我想将场景渲染为最初为空的纹理。为此,我使用了一个帧缓冲区对象,我将一个空的 2d 纹理和一个深度缓冲区附加到该对象上。设置完成后,至于测试,我在场景中绘制了一个简单的四边形。每个顶点都有不同的颜色,所以我最终希望纹理中有一个颜色插值的四边形。然后我使用包含四边形的纹理并将其映射到另一个四边形上。所以最终,我的默认帧缓冲区中有一个四边形,它的纹理包含一个彩色四边形。我希望这不会太混乱......
无论如何,我必须在这里遗漏一些东西,因为我得到的只是灰色纹理。我基本上遵循了这个非常简单的说明。但是,我无法完全弄清楚我在这里缺少什么。如果有人能给我一个线索,我将不胜感激。
谢谢沃尔特
这是我到目前为止的代码: // 创建帧缓冲区对象 glGenFramebuffers(1, &frameBufferObject);
// create depth buffer
glGenRenderbuffers(1, &depthAttachment);
// create empty texture
int width = 512;
int height = 512;
int numberOfChannels = 3;
GLuint internalFormat = GL_RGB;
GLuint format = GL_RGB;
unsigned char* texels = new unsigned char[width * height * numberOfChannels];
glGenTextures(1, &texture);
glBindTexture(GL_TEXTURE_2D, texture);
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_REPEAT);
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_REPEAT);
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR);
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR);
glTexImage2D(GL_TEXTURE_2D, 0, internalFormat, width, height, 0, format, GL_UNSIGNED_BYTE, texels);
glGenerateMipmap(GL_TEXTURE_2D);
delete[] texels;
texels = NULL;
// activate & bind empty texture
glActiveTexture(GL_TEXTURE0);
glBindTexture(GL_TEXTURE_2D, texture);
// attach empty texture to framebuffer object
glBindFramebuffer(GL_FRAMEBUFFER, frameBufferObject);
glFramebufferTexture2D(GL_FRAMEBUFFER, GL_COLOR_ATTACHMENT0, GL_TEXTURE_2D, texture, 0);
// define render targets (empty texture is at GL_COLOR_ATTACHMENT0)
glDrawBuffers(1, GL_COLOR_ATTACHMENT0);
// attach depth buffer
glBindRenderbuffer(GL_RENDERBUFFER, depthAttachment);
glRenderbufferStorage(GL_RENDERBUFFER, GL_DEPTH_COMPONENT, width, height);
glFramebufferRenderbuffer(GL_FRAMEBUFFER, GL_DEPTH_ATTACHMENT, GL_RENDERBUFFER, depthAttachment);
// use framebuffer object
glBindFramebuffer(GL_FRAMEBUFFER, frameBufferObject);
// draw the colored quad into the initially empty texture
glDisable(GL_CULL_FACE);
glDisable(GL_DEPTH_TEST);
// store attibutes
glPushAttrib(GL_VIEWPORT_BIT | GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT);
// reset viewport
glViewport(0, 0, width, height);
// make background yellow
glClearColor(1.0f, 1.0f, 0.0f, 1.0f);
// draw quad into texture attached to frame buffer object
glColor4f(1.0f, 1.0f, 1.0f, 1.0f);
glBegin(GL_QUADS);
glColor4f(1.0f, 1.0f, 1.0f, 1.0f); glVertex2f(0.0f, 100.0f); // top left
glColor4f(1.0f, 0.0f, 0.0f, 1.0f); glVertex2f(0.0f, 0.0f); // bottom left
glColor4f(0.0f, 1.0f, 0.0f, 1.0f); glVertex2f(100.0f, 0.0f); // bottom right
glColor4f(0.0f, 0.0f, 1.0f, 1.0f); glVertex2f(100.0f, 100.0f); // top right
glEnd();
// restore attributes
glPopAttrib();
glEnable(GL_DEPTH_TEST);
glEnable(GL_CULL_FACE);
// use default framebuffer
glBindFramebuffer(GL_FRAMEBUFFER, 0);
// clear default framebuffer
glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT);
// draw the scene
glMatrixMode(GL_MODELVIEW);
glPushMatrix();
glLoadIdentity();
glColor4f(1.0, 1.0, 1.0, 1.0);
// begin texture mapping
glEnable(GL_TEXTURE_2D);
glTexEnvf(GL_TEXTURE_ENV, GL_TEXTURE_ENV_MODE, GL_MODULATE);
glBegin(GL_QUADS);
glNormal3d(0.0f, 0.0f, 1.0f);
glTexCoord2f(0.0f, 0.0f); glVertex3f(0.0f, 50.0f, -100.0f); // top left
glTexCoord2f(0.0f, 1.0f); glVertex3f(0.0f, 0.0f, -100.0f); // bottom left
glTexCoord2f(1.0f, 1.0f); glVertex3f(50.0f, 0.0f, -100.0f); // bottom right
glTexCoord2f(1.0f, 0.0f); glVertex3f(50.0f, 50.0f, -100.0f); // top right
glEnd();
glDisable(GL_TEXTURE_2D);
glPopMatrix();
// swap buffers (I forgot to mention that I use SDL)
SDL_GL_SwapBuffers();
预期结果:
- 黄色背景上带有彩色四边形的纹理
- 该纹理映射到另一个四边形
实际结果:
- 灰色的纹理
- 纹理可以成功映射到另一个四边形
编辑:我错过了提到错误处理。这就是我检查错误的方式。它包括 Paul S 建议的其他案例。感谢那。
GLuint status = glCheckFramebufferStatus(GL_FRAMEBUFFER);
switch(status) {
case GL_FRAMEBUFFER_COMPLETE:
return;
break;
case GL_FRAMEBUFFER_INCOMPLETE_ATTACHMENT:
throw FramebufferIncompleteException("An attachment could not be bound to frame buffer object!");
break;
case GL_FRAMEBUFFER_INCOMPLETE_MISSING_ATTACHMENT:
throw FramebufferIncompleteException("Attachments are missing! At least one image (texture) must be bound to the frame buffer object!");
break;
case GL_FRAMEBUFFER_INCOMPLETE_DIMENSIONS_EXT:
throw FramebufferIncompleteException("The dimensions of the buffers attached to the currently used frame buffer object do not match!");
break;
case GL_FRAMEBUFFER_INCOMPLETE_FORMATS_EXT:
throw FramebufferIncompleteException("The formats of the currently used frame buffer object are not supported or do not fit together!");
break;
case GL_FRAMEBUFFER_INCOMPLETE_DRAW_BUFFER:
throw FramebufferIncompleteException("A Draw buffer is incomplete or undefinied. All draw buffers must specify attachment points that have images attached.");
break;
case GL_FRAMEBUFFER_INCOMPLETE_READ_BUFFER:
throw FramebufferIncompleteException("A Read buffer is incomplete or undefinied. All read buffers must specify attachment points that have images attached.");
break;
case GL_FRAMEBUFFER_INCOMPLETE_MULTISAMPLE:
throw FramebufferIncompleteException("All images must have the same number of multisample samples.");
break;
case GL_FRAMEBUFFER_INCOMPLETE_LAYER_TARGETS :
throw FramebufferIncompleteException("If a layered image is attached to one attachment, then all attachments must be layered attachments. The attached layers do not have to have the same number of layers, nor do the layers have to come from the same kind of texture.");
break;
case GL_FRAMEBUFFER_UNSUPPORTED:
throw FramebufferIncompleteException("Attempt to use an unsupported format combinaton!");
break;
default:
throw FramebufferIncompleteException("Unknown error while attempting to create frame buffer object!");
break;
}
编辑 2:这是我用来检查 GL 错误的方法
checkForGLErrors(string sourceFile, int line)
GLenum error = glGetError();
ostringstream o;
switch(error) {
case GL_NO_ERROR:
return;
break;
case GL_INVALID_ENUM:
o<<"OpenGL Error in "<<sourceFile<<" at line "<<line<<": Invalid enum!"<<endl;
throw GLErrorException(o.str());
break;
case GL_INVALID_VALUE:
o<<"OpenGL Error in "<<sourceFile<<" at line "<<line<<": Invalid value!"<<endl;
throw GLErrorException(o.str());
break;
case GL_INVALID_OPERATION:
o<<"OpenGL Error in "<<sourceFile<<" at line "<<line<<": Invalid operation!"<<endl;
throw GLErrorException(o.str());
break;
case GL_STACK_OVERFLOW:
o<<"OpenGL Error in "<<sourceFile<<" at line "<<line<<": Stack overflow!"<<endl;
throw GLErrorException(o.str());
break;
case GL_STACK_UNDERFLOW:
o<<"OpenGL Error in "<<sourceFile<<" at line "<<line<<": Stack underflow!"<<endl;
throw GLErrorException(o.str());
break;
case GL_OUT_OF_MEMORY:
o<<"OpenGL Error in "<<sourceFile<<" at line "<<line<<": Out Of memory!"<<endl;
throw GLErrorException(o.str());
break;
case GL_TABLE_TOO_LARGE:
o<<"OpenGL Error in "<<sourceFile<<" at line "<<line<<": Table too large!"<<endl;
throw GLErrorException(o.str());
break;
default:
o<<"OpenGL Error in "<<sourceFile<<" at line "<<line<<": Unknown error!"<<endl;
throw GLErrorException(o.str());
break;
}
}