36. OpenGL Framebuffers
Introduction
Frame buffers are one of those mythical things that we have all heard of, but many beginner OpenGL developers avoid because there is not much information about them, and they can be confusing at first. A frame buffer can be thought of as another window that we render to, but we don’t end up seeing.
Now you might be thinking, if we draw something, but we don’t end up seeing it, then what is the point? Have you ever heard of off-screen rendering? I might ask you this instead, have you ever seen a reflection in a game of a live scene? Well, to get this reflection, we typically have to render the scene from the angle of the reflection first, and then we turn that render into a texture and bind it to whatever shape has the reflection attached to it. This initial rendering of the reflection is typically done in a frame buffer for speed, as we can bind a frame buffer straight to a texture.
If you have ever heard of pixel buffers, don’t get them confused with frame buffers, they are different, but can be used together. I won’t go into pixel buffers here, I might write a tutorial on them later on, but they are highly useful for asynchronous texture reads and writes. I once had a project, and part of it required I write to two frame buffers and read back the data, and then perform operations on this data and push it back to the GPU as a texture as quickly as possible. Using a combination of frame buffers and pixel buffers, I went from approximately 5 frames per second doing this, to 30 frames per second, and if the demand arises, I will post a tutorial on how to do this. It was a case where offloading the information to the GPU was not possible.
You should now have an idea of what frame buffers are, in short, they are a rendering context that can be bound to a texture, so let’s start coding them and see what we come up with. I am going to aim for a quad, with a texture of a rotating cube on it. It is a simple example, but highly useful, as you can imagine, the entire scene can be rendered into a frame buffer, and then placed onto a quad for some post processing.
Code
Frame buffer objects are stored like textures. OpenGL will store the information on the graphics card, and it will return an ID associated with that texture for us to place inside an unsigned integer. Also, as seen with textures, the frame buffer ID of 0 (zero) is reserved by OpenGL and is used to unbind the current frame buffer. Also, frame buffers can have several buffers attached. The default buffer is the colour buffer, but you can also bind depth buffers, stencil buffers, etc. So let’s create some variables to hold our frame buffer, frame buffers depth buffer and the texture we are going to store our frame buffer in.
unsigned int fbo; // The frame buffer object unsigned int fbo_depth; // The depth buffer for the frame buffer object unsigned int fbo_texture; // The texture object to write our frame buffer object to
After we have all the variables required to create and use our frame buffers (yes, we only need three variables), we are going to add some extra variables. First, we need two integer values for the width and height of our GLUT window, this is because when you create a frame buffer, you need to specify the width and the height of the texture, and when you use the frame buffer, you need to change the viewport size to match the texture size. Because frame buffers are independent of the size of the GLUT window, you can use them to create higher or lower resolution textures than your regular scene. Typically you would use a lower resolution texture for effects such as reflections so that it renders quicker. The last extra variable we are going to use will just hold how much our teapot in our frame buffer scene will rotate.
int window_width = 500; // The width of our window int window_height = 500; // The height of our window float rotation_degree = 0.0f; // The angle of rotation in degrees for our teapot
The next step we have to do is to create/initialize our frame buffer, the associated depth buffer, and the texture we are going to render to. Typically it is done in the following order:
- Create the Depth Buffer that we are going to use with our frame buffer.
- Create the Texture that we are going to bind our frame buffer to.
- Create the actual Frame Buffer.
- Bind the texture to the frame buffer.
- Bind the depth buffer to the frame buffer.
- Check for errors.
You can check for errors along the way if you wish, but I will just be using one check to make sure our final frame buffer is complete.
Frame Buffer – Depth Buffer
Let’s create the depth buffer we are going to use. To begin with, create a method call initFrameBufferDepthBuffer. This method is going to take no parameters and is not going to return anything. This method is going to contain all code related to the depth buffer we are going to use.
void initFrameBufferDepthBuffer(void) { }
Inside of this method to create the depth buffer, the first call we need to make will create a render buffer for OpenGL to use, and is very similar to the code used to create textures.
glGenRenderbuffersEXT(1, &fbo_depth); // Generate one render buffer and store the ID in fbo_depth
This will create a render buffer, and store the ID in our fbo_depth variable for us to access. Once the render buffer is created for the depth buffer, we then need to bind the buffer so that we can play with it and once we are finished with it, we need to bind the null render buffer, which takes the value of 0. When we are binding a frame buffer, it is once again very similar to texturing, so let’s place our binding code inside our method to fill between.
void initFrameBufferDepthBuffer(void) { glGenRenderbuffersEXT(1, &fbo_depth); // Generate one render buffer and store the ID in fbo_depth glBindRenderbufferEXT(GL_RENDERBUFFER_EXT, fbo_depth); // Bind the fbo_depth render buffer ... glBindRenderbufferEXT(GL_RENDERBUFFER_EXT, 0); // Unbind the render buffer }
Next, we need to fill in the … between the binding of the render buffers. These next two lines are the lines that tell OpenGL that this render buffer will be used for depth. The first line tells OpenGL that we are going to be storing the depth component, and we are going to monitor the entire size of the window. The next line then says that we are going to use fbo_depth to render the depth buffer/attachment.
glRenderbufferStorageEXT(GL_RENDERBUFFER_EXT, GL_DEPTH_COMPONENT, window_width, window_height); // Set the render buffer storage to be a depth component, with a width and height of the window glFramebufferRenderbufferEXT(GL_FRAMEBUFFER_EXT, GL_DEPTH_ATTACHMENT_EXT, GL_RENDERBUFFER_EXT, fbo_depth); // Set the render buffer of this buffer to the depth buffer
Frame Buffer – Texture
Recapping on what we have done, we have created a render buffer that will store the depth component of the buffer that it is attached to. Now we need to create the texture that we want to store our frame buffer in. To do this, let’s create another method similar to the one we just created, but I am going to call this one initFrameBufferTexture. Because we are only creating a texture here, I am not really going to explain this code, the code comments should outline what is going on if you don’t know about texturing in OpenGL (in which case, you are diving in fairly deep straight out).
void initFrameBufferTexture(void) { glGenTextures(1, &fbo_texture); // Generate one texture glBindTexture(GL_TEXTURE_2D, fbo_texture); // Bind the texture fbo_texture glTexImage2D(GL_TEXTURE_2D, 0, GL_RGBA, window_width, window_height, 0, GL_RGBA, GL_UNSIGNED_BYTE, NULL); // Create a standard texture with the width and height of our window // Setup the basic texture parameters glTexParameterf(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_CLAMP_TO_EDGE); glTexParameterf(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_CLAMP_TO_EDGE); glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR); glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR); // Unbind the texture glBindTexture(GL_TEXTURE_2D, 0); }
Frame Buffer – Initialization
Finally, we need one more method; this one will be called initFrameBuffer and will make calls to the above methods we have created, as well as creating the frame buffer and attaching the texture and the depth buffer.
void initFrameBuffer(void) { initFrameBufferDepthBuffer(); // Initialize our frame buffer depth buffer initFrameBufferTexture(); // Initialize our frame buffer texture … }
After making the calls to our above methods, we need to create the frame buffer, and bind it. This is extremely similar to how we created the render buffer for the depth component, and how we create textures, so the following should look vaguely familiar.
glGenFramebuffersEXT(1, &fbo); // Generate one frame buffer and store the ID in fbo glBindFramebufferEXT(GL_FRAMEBUFFER_EXT, fbo); // Bind our frame buffer … glBindFramebufferEXT(GL_FRAMEBUFFER_EXT, 0); // Unbind our frame buffer
This code will generate one frame buffer, and then bind it so we can modify it, and then finally it will unbind it. This is fairly straight forward, and the next two calls will attach the texture and render buffer to our frame buffer. First off, let’s attach the texture to the frame buffer.
glFramebufferTexture2DEXT(GL_FRAMEBUFFER_EXT, GL_COLOR_ATTACHMENT0_EXT, GL_TEXTURE_2D, fbo_texture, 0); // Attach the texture fbo_texture to the color buffer in our frame buffer
Now that we have a texture, theoretically we can use this straight out, but you won’t get any depth information. So we need to go ahead and attach the depth render buffer we created called fbo_depth, to our frame buffer.
glFramebufferRenderbufferEXT(GL_FRAMEBUFFER_EXT, GL_DEPTH_ATTACHMENT_EXT, GL_RENDERBUFFER_EXT, fbo_depth); // Attach the depth buffer fbo_depth to our frame buffer
And now we have a frame buffer, which outputs to a texture and also has a depth buffer, which we can use. Well, we think we do, so first let’s check if our frame buffer is complete and there were no problems. To do this, we can make a call to glCheckFramebufferStatusEXT while our frame buffer is bound, and this will give us back a GLenum with the status of our frame buffer. For our frame buffer to be complete, we need to check that the status is equal to GL_FRAMEBUFFER_COMPLETE_EXT. We can do this with a simple if statement. Note that if you want the output to the console, you need to include <iostream> into your file.
GLenum status = glCheckFramebufferStatusEXT(GL_FRAMEBUFFER_EXT); // Check that status of our generated frame buffer if (status != GL_FRAMEBUFFER_COMPLETE_EXT) // If the frame buffer does not report back as complete { std::cout << "Couldn't create frame buffer" << std::endl; // Make sure you include <iostream> exit(0); // Exit the application }
General Initialization
Because in this example, we want to use the frame buffer constantly, I am going to create an init method which will call the initFrameBuffer method, and init will be called straight after we initialize GLEW, which is straight after we call glutCreateWindow. So here is our init method, which in this example, enables texturing, enables depth testing, and then initializes our frame buffer.
void init(void) { glEnable(GL_TEXTURE_2D); // Enable texturing so we can bind our frame buffer texture glEnable(GL_DEPTH_TEST); // Enable depth testing initFrameBuffer(); // Create our frame buffer object }
Next up, here is an example of my main method, which calls glewInit and calls our own init method.
int main (int argc, char **argv) { … glutCreateWindow ("Your first OpenGL Window"); // Set the title for the window if (GLEW_OK != glewInit()) { std::cout << "Couldn't initialize GLEW" << std::endl; exit(0); } init(); … }
Frame Buffer – Usage
To use our frame buffer, I am first going to give you the display method, and whilst you could put all your code in here, I am going to make a call to a method to render our teapot scene. This display method builds upon the tutorial for creating a square.
So it looks something like the following code, which will create a square with texture coordinates, and I have placed it back 2 units so that it fills up most of the screen. Keep in mind, that I am not doing anything with the frame buffer at the moment; this is just a standard display method.
void display (void) { keyOperations(); glClearColor(1.0f, 0.0f, 0.0f, 1.0f); // Clear the background of our window to red glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT); //Clear the colour buffer (more buffers later on) glLoadIdentity(); // Load the Identity Matrix to reset our drawing locations glTranslatef(0.0f, 0.0f, -2.0f); glBegin(GL_QUADS); glTexCoord2f(0.0f, 0.0f); glVertex3f(-1.0f, -1.0f, 0.0f); // The bottom left corner glTexCoord2f(0.0f, 1.0f); glVertex3f(-1.0f, 1.0f, 0.0f); // The top left corner glTexCoord2f(1.0f, 1.0f); glVertex3f(1.0f, 1.0f, 0.0f); // The top right corner glTexCoord2f(1.0f, 0.0f); glVertex3f(1.0f, -1.0f, 0.0f); // The bottom right corner glEnd(); glutSwapBuffers(); }
Now we will go ahead and bind our frame buffer texture to the quad. This is done just like binding a regular texture, because it is just a regular texture. The only difference between the texture we use for our frame buffer, and a standard texture we load in from a file, is that this texture is filled in on the GPU by OpenGL.
So right before your quad begins, bind your frame buffer texture, and then after you finish drawing your quad, unbind it.
glBindTexture(GL_TEXTURE_2D, fbo_texture); glBegin(GL_QUADS); … glEnd(); glBindTexture(GL_TEXTURE_2D, 0);
You should be able to run this now, but unfortunately because we haven’t rendered anything into the frame buffer, nothing will appear on our quad, it will be as if we never bound a texture. So let’s create a method where we are going to render our frame buffer. I am going to call this renderTeapotScene, because I am using it to render a teapot.
void renderTeapotScene(void) { }
renderTeapotScene is going to be called from inside our display method, right after our keyOperations and before we do any actual rendering.
void display(void) { keyOperations(); renderTeapotScene(); ... }
Inside of our renderTeapotScene method, we need to first bind our frame buffer so that we can render to it, and then unbind it when we are finished rendering. I am also going to add in some extra code at the very end of this method, just to update the amount in which we are going to rotate our teapot.
void renderTeapotScene(void) { glBindFramebufferEXT(GL_FRAMEBUFFER_EXT, fbo); // Bind our frame buffer for rendering glBindFramebufferEXT(GL_FRAMEBUFFER_EXT, 0); // Unbind our texture rotation_degree += 0.5f; if (rotation_degree > 360.0f) rotation_degree = 0.0f; }
So far, everything should be fine, except when you bind a frame buffer, you need to set the size of the viewport that the frame buffer will render to. This is done with a call to glViewport, but first I am going to do some more management orientated activities, and store the current state of our GL_ENABLE_BIT and our GL_VIEWPORT_BIT. This is so that when we finish, we can put these back to how they were and not having any glEnable or glViewport calls modify our regular rendering. It is also wise to store information such as GL_LIGHTING_BIT if you are using lighting, and any other states you might need.
void renderTeapotScene(void) { glBindFramebufferEXT(GL_FRAMEBUFFER_EXT, fbo); // Bind our frame buffer for rendering glPushAttrib(GL_VIEWPORT_BIT | GL_ENABLE_BIT); // Push our glEnable and glViewport states glPopAttrib(); // Restore our glEnable and glViewport states glBindFramebufferEXT(GL_FRAMEBUFFER_EXT, 0); // Unbind our texture rotation_degree += 0.5f; if (rotation_degree > 360.0f) rotation_degree = 0.0f; }
It is our calls to glPushAttrib and glPopAttrib which will save and restore our states. The next step as I said is to set the viewport size for rendering, and then I am going to clear the colour buffer, depth buffer and reset the model view matrix, just like you would in your regular display method.
void renderTeapotScene(void) { glBindFramebufferEXT(GL_FRAMEBUFFER_EXT, fbo); // Bind our frame buffer for rendering glPushAttrib(GL_VIEWPORT_BIT | GL_ENABLE_BIT); // Push our glEnable and glViewport states glViewport(0, 0, window_width, window_height); // Set the size of the frame buffer view port glClearColor (0.0f, 0.0f, 1.0f, 1.0f); // Set the clear colour glClear (GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT); // Clear the depth and colour buffers glLoadIdentity(); // Reset the modelview matrix glPopAttrib(); // Restore our glEnable and glViewport states glBindFramebufferEXT(GL_FRAMEBUFFER_EXT, 0); // Unbind our texture rotation_degree += 0.5f; if (rotation_degree > 360.0f) rotation_degree = 0.0f; }
This should be starting to look a little familiar inside of our frame buffer and push attributes calls, and you are probably right, everything you do from now on is just like rendering a normal scene, only you don’t need to call glFlush or glutSwapBuffers like you would with a regular render as we are not pushing the output to the screen. So now all we need to do is add the code in for our teapot rendering. I am going to translate it back five units, and then rotate it along the x and the y axis.
void renderTeapotScene(void) { … glClear (GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT); // Clear the depth and colour buffers glLoadIdentity(); // Reset the modelview matrix glTranslatef(0.0f, 0.0f, -5.0f); // Translate back 5 units glRotatef(rotation_degree, 1.0f, 1.0f, 0.0f); // Rotate according to our rotation_degree value glutSolidTeapot(1.0f); // Render a teapot glPopAttrib(); // Restore our glEnable and glViewport states glBindFramebufferEXT(GL_FRAMEBUFFER_EXT, 0); // Unbind our texture … }
If you followed everything as I have said correctly, then you should be able to compile this and run it. You should then see a quad with a red background, with a texture of a spinning teapot with a blue background attached to the quad.
Well done if you managed that, and now have your head around frame buffers!
If you have any questions, you can always contact me at swiftless@gmail.com
Tutorial Code
#include <GL/glew.h> // Include the GLEW header file #include <GL/glut.h> // Include the GLUT header file #include <iostream> // Allow us to print to the console bool* keyStates = new bool[256]; // Create an array of boolean values of length 256 (0-255) unsigned int fbo; // The frame buffer object unsigned int fbo_depth; // The depth buffer for the frame buffer object unsigned int fbo_texture; // The texture object to write our frame buffer object to int window_width = 500; // The width of our window int window_height = 500; // The height of our window float rotation_degree = 0.0f; // The angle of rotation in degrees for our teapot void initFrameBufferDepthBuffer(void) { glGenRenderbuffersEXT(1, &fbo_depth); // Generate one render buffer and store the ID in fbo_depth glBindRenderbufferEXT(GL_RENDERBUFFER_EXT, fbo_depth); // Bind the fbo_depth render buffer glRenderbufferStorageEXT(GL_RENDERBUFFER_EXT, GL_DEPTH_COMPONENT, window_width, window_height); // Set the render buffer storage to be a depth component, with a width and height of the window glFramebufferRenderbufferEXT(GL_FRAMEBUFFER_EXT, GL_DEPTH_ATTACHMENT_EXT, GL_RENDERBUFFER_EXT, fbo_depth); // Set the render buffer of this buffer to the depth buffer glBindRenderbufferEXT(GL_RENDERBUFFER_EXT, 0); // Unbind the render buffer } void initFrameBufferTexture(void) { glGenTextures(1, &fbo_texture); // Generate one texture glBindTexture(GL_TEXTURE_2D, fbo_texture); // Bind the texture fbo_texture glTexImage2D(GL_TEXTURE_2D, 0, GL_RGBA, window_width, window_height, 0, GL_RGBA, GL_UNSIGNED_BYTE, NULL); // Create a standard texture with the width and height of our window // Setup the basic texture parameters glTexParameterf(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_CLAMP_TO_EDGE); glTexParameterf(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_CLAMP_TO_EDGE); glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR); glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR); // Unbind the texture glBindTexture(GL_TEXTURE_2D, 0); } void initFrameBuffer(void) { initFrameBufferDepthBuffer(); // Initialize our frame buffer depth buffer initFrameBufferTexture(); // Initialize our frame buffer texture glGenFramebuffersEXT(1, &fbo); // Generate one frame buffer and store the ID in fbo glBindFramebufferEXT(GL_FRAMEBUFFER_EXT, fbo); // Bind our frame buffer glFramebufferTexture2DEXT(GL_FRAMEBUFFER_EXT, GL_COLOR_ATTACHMENT0_EXT, GL_TEXTURE_2D, fbo_texture, 0); // Attach the texture fbo_texture to the color buffer in our frame buffer glFramebufferRenderbufferEXT(GL_FRAMEBUFFER_EXT, GL_DEPTH_ATTACHMENT_EXT, GL_RENDERBUFFER_EXT, fbo_depth); // Attach the depth buffer fbo_depth to our frame buffer GLenum status = glCheckFramebufferStatusEXT(GL_FRAMEBUFFER_EXT); // Check that status of our generated frame buffer if (status != GL_FRAMEBUFFER_COMPLETE_EXT) // If the frame buffer does not report back as complete { std::cout << "Couldn't create frame buffer" << std::endl; // Output an error to the console exit(0); // Exit the application } glBindFramebufferEXT(GL_FRAMEBUFFER_EXT, 0); // Unbind our frame buffer } void init(void) { glEnable(GL_TEXTURE_2D); // Enable texturing so we can bind our frame buffer texture glEnable(GL_DEPTH_TEST); // Enable depth testing initFrameBuffer(); // Create our frame buffer object } void keyOperations (void) { if (keyStates['a']) { // If the a key has been pressed // Perform 'a' key operations } } void renderTeapotScene(void) { glBindFramebufferEXT(GL_FRAMEBUFFER_EXT, fbo); // Bind our frame buffer for rendering glPushAttrib(GL_VIEWPORT_BIT | GL_ENABLE_BIT); // Push our glEnable and glViewport states glViewport(0, 0, window_width, window_height); // Set the size of the frame buffer view port glClearColor (0.0f, 0.0f, 1.0f, 1.0f); // Set the clear colour glClear (GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT); // Clear the depth and colour buffers glLoadIdentity(); // Reset the modelview matrix glTranslatef(0.0f, 0.0f, -5.0f); // Translate back 5 units glRotatef(rotation_degree, 1.0f, 1.0f, 0.0f); // Rotate according to our rotation_degree value glutSolidTeapot(1.0f); // Render a teapot glPopAttrib(); // Restore our glEnable and glViewport states glBindFramebufferEXT(GL_FRAMEBUFFER_EXT, 0); // Unbind our texture rotation_degree += 0.5f; if (rotation_degree > 360.0f) rotation_degree = 0.0f; } void display (void) { keyOperations(); // Perform any key presses renderTeapotScene(); // Render our teapot scene into our frame buffer glClearColor(1.0f, 0.0f, 0.0f, 1.0f); // Clear the background of our window to red glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT); //Clear the colour buffer (more buffers later on) glLoadIdentity(); // Load the Identity Matrix to reset our drawing locations glTranslatef(0.0f, 0.0f, -2.0f); glBindTexture(GL_TEXTURE_2D, fbo_texture); // Bind our frame buffer texture glBegin(GL_QUADS); glTexCoord2f(0.0f, 0.0f); glVertex3f(-1.0f, -1.0f, 0.0f); // The bottom left corner glTexCoord2f(0.0f, 1.0f); glVertex3f(-1.0f, 1.0f, 0.0f); // The top left corner glTexCoord2f(1.0f, 1.0f); glVertex3f(1.0f, 1.0f, 0.0f); // The top right corner glTexCoord2f(1.0f, 0.0f); glVertex3f(1.0f, -1.0f, 0.0f); // The bottom right corner glEnd(); glBindTexture(GL_TEXTURE_2D, 0); // Unbind any textures glutSwapBuffers(); } void reshape (int width, int height) { glViewport(0, 0, (GLsizei)width, (GLsizei)height); // Set our viewport to the size of our window glMatrixMode(GL_PROJECTION); // Switch to the projection matrix so that we can manipulate how our scene is viewed glLoadIdentity(); // Reset the projection matrix to the identity matrix so that we don't get any artifacts (cleaning up) gluPerspective(60, (GLfloat)width / (GLfloat)height, 1.0, 100.0); // Set the Field of view angle (in degrees), the aspect ratio of our window, and the new and far planes glMatrixMode(GL_MODELVIEW); // Switch back to the model view matrix, so that we can start drawing shapes correctly } void keyPressed (unsigned char key, int x, int y) { keyStates[key] = true; // Set the state of the current key to pressed } void keyUp (unsigned char key, int x, int y) { keyStates[key] = false; // Set the state of the current key to not pressed } int main (int argc, char **argv) { glutInit(&argc, argv); // Initialize GLUT glutInitDisplayMode (GLUT_DOUBLE | GLUT_DEPTH | GLUT_RGBA); // Set up a basic display buffer (only single buffered for now) glutInitWindowSize (500, 500); // Set the width and height of the window glutInitWindowPosition (100, 100); // Set the position of the window glutCreateWindow ("Your first OpenGL Window"); // Set the title for the window if (GLEW_OK != glewInit()) { std::cout << "Couldn't initialize GLEW" << std::endl; exit(0); } init(); glutDisplayFunc(display); // Tell GLUT to use the method "display" for rendering glutIdleFunc(display); // Tell GLUT to use the method "display" for rendering glutReshapeFunc(reshape); // Tell GLUT to use the method "reshape" for reshaping glutKeyboardFunc(keyPressed); // Tell GLUT to use the method "keyPressed" for key presses glutKeyboardUpFunc(keyUp); // Tell GLUT to use the method "keyUp" for key up events glutMainLoop(); // Enter GLUT's main loop }
sir its not wrking on my studio..it underline “glClearColor()” and “glLoadIdentity()”
plzz tell me its solution….
Hi Donald, what I observed here is even if i remove the depthbuffer initialisation code and do not do the binding with it I am still able to get the correct output, so what’s the point of using depth buffer here.
Hi!
I tried to integrate your code in a project of mine, but the rendered texture don’t contains the teapot, wich shows up without problems if I render it direct to screen.
Do you know any reasons to why this might happen?
I’d like to alter your code to do something similar. Instead of displaying the rendered texture to the quad, I’d like to display the depth mask stored in the renderbuffer we attached. But when I replace fbo_texture with fbo_depth in the glBindTexture call, I get the same texture out. I’ve tried changing the call to read:
glBindRenderbuffer(GL_RENDERBUFFER_EXT, fbo_depth);
and all sorts of other variants, but I either get the textured image, or a white square. How would I display the depth map on the quad?
Thank you so much for this tutorial. By the way, when I set a color for the teapot, the blue background disappears or turns black, any idea why this happens?
Nice article, but the formatting on this site crops the right side of it!
This is known. I’m hoping to get time to fix it soon!
Thanks,
Swiftless
Nice intro!!
Btw, i have a question, where does the fbo_depth variable obtain the depth buffer data?
or does this line
glFramebufferRenderbufferEXT(GL_FRAMEBUFFER_EXT, GL_DEPTH_ATTACHMENT_EXT, GL_RENDERBUFFER_EXT, fbo_depth);
actually do the magic for assigning the depth buffer to fbo_depth?
Hi Fath,
Yes, that is exactly right, that one line is what specifically attaches a depth buffer to a framebuffer object. Framebuffer objects are fully customisable, you don’t need to have depth, colour or stencil buffers, you can do a combination of buffers to get exactly what you want. There are times when a colour buffer isn’t required, but a depth buffer is, and it saves a lot of processing.
Cheers,
Swiftless
Nice tutorial, I’d like to share an experience I had tonight. So i’ve got a basic deferred renderer and its time to add downsampling (HDR). From what I’ve understood a single FBO is prefered but the textures are too plenty to fit in a row (more than 8). I wrote some book keeping to simplify attachment and got a weird defect in the display mode where all texture images are shown side by side. The buffers had shrank 1/4 by 1/4. Aha that caused by the new smaller downsampling texture. After some shotgun debugging i found that several diversly sized images where attached in the same FBO whereby (for unexamined reasons) the smaller one got used.
So here I wish to share my contract snippet to guard against this error in the future.
#define haltitude(expr) breakitude(expr)
//#define haltitude(expr) assert(expr)
#define breakitude(expr) {if(expr){}else{_asm int 3}}
void assertSameSize(unsigned int t[8])
{
int firstWidth = 0;
int firstHeight = 0;
int width;
int height;
for(int i=0; i<8; ++i)
{
if(t[i] && glIsTexture(t[i]))
{
glBindTexture(GL_TEXTURE_2D, t[i]);
glGetTexLevelParameteriv(GL_TEXTURE_2D,0,GL_TEXTURE_WIDTH,&width);
glGetTexLevelParameteriv(GL_TEXTURE_2D,0,GL_TEXTURE_HEIGHT,&height);
glBindTexture(GL_TEXTURE_2D,0);
if(0==i)
{
firstWidth = width;
firstHeight = height;
continue;
}
haltitude(width==firstWidth);
haltitude(height==firstHeight);
}
}
}
thank you for your perseverance Swiftfless!
When I’m using your code I keep getting a problem with the fbo not being complete (it closes saying there’s a problem from the if statement in the fbo init function). Any reasons as to why this might happen?
I forgot to include that the status variable returns 30655 as the variable.
Ok so it turns out you can’t write a function that does this for any unsigned int. it didn’t work, but now it does once I changed that.
I am trying to do a project which will simulate navigation only with out the user interface(the routes will be predefined and based upon user selection the routes will be mapped). i have mapped the texture(a map) on the screen and i am trying to move the car over it. the problem i am facing is that i am not able to see the actual movement of the car and also,it is leaving a trail on th texture. please suggest a solution for this problem. thank u.