从纹理渲染到屏幕外纹理(FBO)

时间:2013-10-21 04:19:03

标签: opengl textures glsl gpgpu fbo

我正在尝试编写一个GPGPU程序,只需将纹理内容复制到附加到FBO的输出纹理即可。但是,我看到输入纹理数据未正确加载/未正确绑定。我无法理解什么是错的。

这是我的代码:

void render()
{

  int i;
  float * result, *in;

  //allocating and filling data in the input texture

  result = (float * ) malloc(4*32*32*sizeof(float));
  in = (float *) malloc(4*32*32*sizeof(float));
  for(i=0;i<32*32*4; i++)
  {
    in[i]=1.0f;
  }


makeBuffers(); //makes the vertex and element buffers
makeTexture(in);  //makes the input textures
makeShaders();    //makes the vertex n pixel shaders
makeProgram();   //makes the program

glClearColor(1.0f, 1.0f, 1.0f, 0.0f);
glClear(GL_COLOR_BUFFER_BIT);

glBindBuffer(GL_ARRAY_BUFFER, res.vertexBuffer);
glVertexAttribPointer(res.position, 2, GL_FLOAT, GL_FALSE, sizeof(GLfloat)*2, (void*)0);
glEnableVertexAttribArray(res.position);
glBindBuffer(GL_ELEMENT_ARRAY_BUFFER, res.elementBuffer);
res.texture= glGetUniformLocation(res.shaderProgram, "intexture");

/ *绑定纹理* /     makefbo(); //制作屏幕外帧缓冲对象

glActiveTexture(GL_TEXTURE0);
glBindTexture(GL_TEXTURE_2D, res.texture);
glUniform1i(res.texture, 0);


glDrawElements(GL_TRIANGLE_STRIP,4, GL_UNSIGNED_SHORT, (void*)0);

//    glFlush();   // i think drawing would be enough now. 

glReadBuffer(GL_COLOR_ATTACHMENT0_EXT);   //reading back data from the framebuffer
glReadPixels(0, 0, 32, 32,GL_RGBA,GL_FLOAT,result);

printf(" float values \n");
for(i=0; i< 32*4 ; i++)
  printf(" %f   ", result[i]);
printf(" \n");

}

这是我的maketexture()函数

 glGenTextures (1, &res.texture);
glBindTexture(GL_TEXTURE_2D,res.texture);
// set texture parameters
glTexParameteri(GL_TEXTURE_2D,
                GL_TEXTURE_MIN_FILTER, GL_NEAREST);
glTexParameteri(GL_TEXTURE_2D,
                GL_TEXTURE_MAG_FILTER, GL_NEAREST);
glTexParameteri(GL_TEXTURE_2D,
                GL_TEXTURE_WRAP_S, GL_CLAMP);
glTexParameteri(GL_TEXTURE_2D,
                GL_TEXTURE_WRAP_T, GL_CLAMP);
// define texture with floating point format
glTexImage2D(GL_TEXTURE_2D,0,GL_RGBA32F,
             texSize,texSize,0,GL_RGBA,GL_FLOAT,0);

// transfer data to texture
glTexSubImage2D(GL_TEXTURE_2D,0,0,0,texSize,texSize,
                GL_RGBA,GL_FLOAT,data);

这是我的framebufferoject准备和输出纹理biding:

glGenFramebuffers(1, &res.fbo);
glBindFramebuffer(GL_FRAMEBUFFER, res.fbo);
glViewport(0, 0, 32, 32);

glGenTextures(1, &res.tex);
glBindTexture(GL_TEXTURE_2D, res.fbo);
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_NEAREST);
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_NEAREST);
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_CLAMP);
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_CLAMP);
glTexImage2D(GL_TEXTURE_2D, 0, GL_RGBA32F, 
         32, 32, 0, GL_RGBA, GL_FLOAT, 0);
 glFramebufferTexture2DEXT(GL_FRAMEBUFFER, 
                      GL_COLOR_ATTACHMENT0, 
                      GL_TEXTURE_2D, res.tex, 0);

数据似乎没有得到正确读取。我试图将输入纹理附加到帧缓冲对象,并尝试立即读回纹理数据,然后数据显示正常。但是当我尝试渲染到屏幕外纹理时,它并没有提供正确的输出。它在像素着色器执行结束时只给出了一堆零。

像素着色器获取输入纹理并将数据复制到输出纹理中:outColor = texture2D(intexture,texcoord);

有人可以看到这个问题吗?

1 个答案:

答案 0 :(得分:1)

你的代码缺少很多上下文,但我的印象是你只需要一次绑定所有内容然后就这样离开。这不是它的意思,它可能不会起作用。

基本要点是,如果纹理用作FBO渲染目标,则它不能用作采样源。所以你必须根据需要解开纹理/ FBO,这样事情就不会踩在彼此的脚趾上,或者你正在加载着色器,这些着色器只是禁止纹理访问。恕我直言解除绑定是更简单的选择(但是使用OpenGL-4.4发布了无绑定纹理扩展,这可能成为4.5中的核心。)