Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

Render multiple objects with OpenGL ES 2.0

I am trying to learn OpenGL ES 2.0 to do some iPhone game development. I have read through multiple tutorials and some of the OpenGL ES 2.0 spec. All of the examples I have seen have created a single mesh, loaded it into a vertex buffer and then rendered it (with the expected translation, rotation, gradient, etc.)

My question is this: how do you render multiple objects in your scene that have different meshes and are moving independently? If I have a car and a motorcycle for example, can I create 2 vertex buffers and keep the mesh data for both around for each render call, and then just send in different matrices for the shader for each object? Or do I need to somehow translate the meshes and then combine them into a single mesh so that they can be rendered in one pass? I'm looking for more of the high-level strategy / program structure rather than code examples. I think I just have the wrong mental modal of how this works.

Thanks!

like image 916
Matthew Daugherty Avatar asked Aug 27 '11 23:08

Matthew Daugherty


1 Answers

The best way I found to do this is using VAOs in addition to VBOs.

I'll first answer you question using VBOs only.

First of all, assume you have the two meshes of your two objects stored in the following arrays:

GLuint _vertexBufferCube1; GLuint _vertexBufferCube2; 

where:

GLfloat gCubeVertexData1[36] = {...}; GLfloat gCubeVertexData2[36] = {...}; 

And you also have to vertix buffers:

GLuint _vertexBufferCube1; GLuint _vertexBufferCube2; 

Now, to draw those two cubes (without VAOs), you have to do something like that: in draw function (from OpenGLES template):

//Draw first object, bind VBO, adjust your attributes then call DrawArrays glGenBuffers(1, &_vertexBufferCube1); glBindBuffer(GL_ARRAY_BUFFER, _vertexBufferCube1); glBufferData(GL_ARRAY_BUFFER, sizeof(gCubeVertexData1), gCubeVertexData1, GL_STATIC_DRAW);  glEnableVertexAttribArray(GLKVertexAttribPosition); glVertexAttribPointer(GLKVertexAttribPosition, 3, GL_FLOAT, GL_FALSE, 24, BUFFER_OFFSET(0)); glEnableVertexAttribArray(GLKVertexAttribNormal); glVertexAttribPointer(GLKVertexAttribNormal, 3, GL_FLOAT, GL_FALSE, 24, BUFFER_OFFSET(12));   glDrawArrays(GL_TRIANGLES, 0, 36);    //Repeat for second object: glGenBuffers(1, &_vertexBufferCube2); glBindBuffer(GL_ARRAY_BUFFER, _vertexBufferCube2); glBufferData(GL_ARRAY_BUFFER, sizeof(gCubeVertexData2), gCubeVertexData2, GL_STATIC_DRAW);  glEnableVertexAttribArray(GLKVertexAttribPosition); glVertexAttribPointer(GLKVertexAttribPosition, 3, GL_FLOAT, GL_FALSE, 24, BUFFER_OFFSET(0)); glEnableVertexAttribArray(GLKVertexAttribNormal); glVertexAttribPointer(GLKVertexAttribNormal, 3, GL_FLOAT, GL_FALSE, 24, BUFFER_OFFSET(12)); glUseProgram(_program);  glDrawArrays(GL_TRIANGLES, 0, 36); 

This will answer you question. But now to use VAOs, your draw function code is much simpler (which is good because it is the repeated function):

First you will define to VAOs:

GLuint _vertexArray1; GLuint _vertexArray2; 

and then you will do all the steps previously done in draw method, you will do it in setupGL function but after binding to the VAO. Then in your draw function you just bind to the VAO you want.

VAO here is like a profile that contains a lot of properties (imagine a smart device profile). Instead of changing color, desktop, fonts.. etc every time you wish to change them, you do that once and save it under a profile name. Then you just switch the profile.

So you do that once, inside setupGL, then you switch between them in draw.

Of course you may say that you could have put the code (without VAO) in a function and call it. That's true, but VAOs are more efficient according to Apple:

http://developer.apple.com/library/ios/#documentation/3DDrawing/Conceptual/OpenGLES_ProgrammingGuide/TechniquesforWorkingwithVertexData/TechniquesforWorkingwithVertexData.html#//apple_ref/doc/uid/TP40008793-CH107-SW1

Now to the code:

In setupGL:

glGenVertexArraysOES(1, &_vertexArray1); //Bind to first VAO glBindVertexArrayOES(_vertexArray1);  glGenBuffers(1, &_vertexBufferCube1); //All steps from this one are done to first VAO only glBindBuffer(GL_ARRAY_BUFFER, _vertexBufferCube1); glBufferData(GL_ARRAY_BUFFER, sizeof(gCubeVertexData1), gCubeVertexData1, GL_STATIC_DRAW);  glEnableVertexAttribArray(GLKVertexAttribPosition); glVertexAttribPointer(GLKVertexAttribPosition, 3, GL_FLOAT, GL_FALSE, 24, BUFFER_OFFSET(0)); glEnableVertexAttribArray(GLKVertexAttribNormal); glVertexAttribPointer(GLKVertexAttribNormal, 3, GL_FLOAT, GL_FALSE, 24, BUFFER_OFFSET(12));  glGenVertexArraysOES(1, &_vertexArray2); // now bind to the second glBindVertexArrayOES(_vertexArray2);  glGenBuffers(1, &_vertexBufferCube2); //repeat with the second mesh glBindBuffer(GL_ARRAY_BUFFER, _vertexBufferCube2); glBufferData(GL_ARRAY_BUFFER, sizeof(gCubeVertexData2), gCubeVertexData2, GL_STATIC_DRAW);  glEnableVertexAttribArray(GLKVertexAttribPosition); glVertexAttribPointer(GLKVertexAttribPosition, 3, GL_FLOAT, GL_FALSE, 24, BUFFER_OFFSET(0)); glEnableVertexAttribArray(GLKVertexAttribNormal); glVertexAttribPointer(GLKVertexAttribNormal, 3, GL_FLOAT, GL_FALSE, 24, BUFFER_OFFSET(12));   glBindVertexArrayOES(0); 

Then finally in your draw method:

glBindVertexArrayOES(_vertexArray1); glDrawArrays(GL_TRIANGLES, 0, 36);   glBindVertexArrayOES(_vertexArray2);     glDrawArrays(GL_TRIANGLES, 0, 36); 
like image 117
Tony Avatar answered Oct 07 '22 10:10

Tony