Unbind Your Vertex Array Objects.

When I first started learning about OpenGL, I always made sure to unbind each resource as soon as I was done using it. Over time I became more lenient with unbinding, as in many cases leaving resources bound is okay and leads to fewer API calls.

In addition, when I’m developing on my PC I have access to the newer Direct State Access (DSA) functions, so binding resources isn’t something I have to worry about as much. However, I do a fair amount of programming on my macbook which means I don’t always have access to the DSA functions.

I was working on a project recently that involved the vertex and index data for a mesh frequently changing. The actual mesh struct has a bit more going on, but we can think of it like this for now:

1typedef struct {
2    Vec3 *vertices;
3    GLuint *indices;
4    GLuint vao;
5    GLuint vbo;
6    GLuint ebo;
7} Mesh;

So when it comes time to update the vertex or index data, we map the vertex buffer object to the vertices pointer and the element buffer object to the indices pointer. Then we can write all of our vertex and index data directly to the buffers, and then unmap them when we’re all done and need to draw.

 1...
 2
 3glBindBuffer(GL_ARRAY_BUFFER, mesh->vbo);
 4mesh->vertices = glMapBuffer(GL_ARRAY_BUFFER, GL_WRITE_ONLY);
 5
 6glBindBuffer(GL_ELEMENT_ARRAY_BUFFER, mesh->ebo);
 7mesh->indices = glMapBuffer(GL_ELEMENT_ARRAY_BUFFER, GL_WRITE_ONLY);
 8
 9// While we have the buffers mapped, we can write data directly to
10// mesh->vertices and mesh->indices.
11
12glBindBuffer(GL_ARRAY_BUFFER, mesh->vbo);
13glUnmapBuffer(GL_ARRAY_BUFFER);
14mesh->vertices = NULL
15
16glBindBuffer(GL_ELEMENT_ARRAY_BUFFER, mesh->ebo);
17glUnmapBuffer(GL_ELEMENT_ARRAY_BUFFER);
18mesh->indices = NULL
19
20// Now that the buffers are unmapped, we can draw the mesh again.
21
22...

Leaving the buffers bound in the above snippet didn’t cause problems and seemed a reasonable thing to do to reduce unnecessary OpenGL calls. The issues I was running into were actually caused by how I was drawing each mesh:

1...
2
3glBindVertexArray(mesh->vao);
4glDrawElements(...);
5
6...

Since I wasn’t unbinding the mesh’s vertex array object when I finished drawing it, when I updated another mesh’s vertex or index data and bound that mesh’s buffer objects, the vertex array object I forgot to unbind would now be using the incorrect buffers to draw with.

Luckily, there was a simple fix, and now I know to be more careful with unbinding vertex array objects:

1...
2
3glBindVertexArray(mesh->vao);
4glDrawElements(...);
5glBindVertexArray(0);
6
7...

#opengl #c

Reply to this post by email ↪