用glReadPixels读取纹理字节?

时间:2012-08-08 11:20:35

标签: ios opengl-es textures glreadpixels

我想将原始纹理数据转储到磁盘上(稍后再回读),我不确定glReadPixel会从当前绑定的纹理中读取。

如何从纹理中读取缓冲区?

2 个答案:

答案 0 :(得分:34)

glReadPixels 函数从帧缓冲区读取,而不是纹理。要读取纹理对象,必须使用 glGetTexImage ,但在OpenGL ES中不可用 :(

如果你想从纹理中读取缓冲区,那么你可以将它绑定到FBO(FrameBuffer对象)并使用glReadPixels

//Generate a new FBO. It will contain your texture.
glGenFramebuffersOES(1, &offscreen_framebuffer);
glBindFramebufferOES(GL_FRAMEBUFFER_OES, offscreen_framebuffer);

//Create the texture 
glGenTextures(1, &my_texture);
glBindTexture(GL_TEXTURE_2D, my_texture);
glTexImage2D(GL_TEXTURE_2D, 0, GL_RGBA,  width, height, 0, GL_RGBA, GL_UNSIGNED_BYTE, NULL);
glTexParameteri(GL_TEXTURE_2D,GL_TEXTURE_MIN_FILTER,GL_LINEAR);
glTexParameteri(GL_TEXTURE_2D,GL_TEXTURE_MAG_FILTER,GL_LINEAR);

//Bind the texture to your FBO
glFramebufferTexture2DOES(GL_FRAMEBUFFER_OES, GL_COLOR_ATTACHMENT0_OES, GL_TEXTURE_2D, my_texture, 0);

//Test if everything failed    
GLenum status = glCheckFramebufferStatusOES(GL_FRAMEBUFFER_OES);
if(status != GL_FRAMEBUFFER_COMPLETE_OES) {
    printf("failed to make complete framebuffer object %x", status);
}

然后,当你想要从纹理中读取时,你只需要调用glReadPixels:

//Bind the FBO
glBindFramebufferOES(GL_FRAMEBUFFER_OES, offscreen_framebuffer);
// set the viewport as the FBO won't be the same dimension as the screen
glViewport(0, 0, width, height);

GLubyte* pixels = (GLubyte*) malloc(width * height * sizeof(GLubyte) * 4);
glReadPixels(0, 0, width, height, GL_RGBA, GL_UNSIGNED_BYTE, pixels);

//Bind your main FBO again
glBindFramebufferOES(GL_FRAMEBUFFER_OES, screen_framebuffer);
// set the viewport as the FBO won't be the same dimension as the screen
glViewport(0, 0, screen_width, screen_height);

答案 1 :(得分:2)

感谢Gergonzale的回答。我今天早上花了一些时间试图弄清楚如何使用16位纹理,这个代码片段可能对其他任何将GL_UNSIGNED_SHORT_5_6_5转换为GL_UNSIGNED_BYTE的人都有用

glTexImage2D(GL_TEXTURE_2D, 0, GL_RGB,  tSizeW, tSizeH, 0, GL_RGB, GL_UNSIGNED_SHORT_5_6_5, NULL);
GLubyte* pixels = (GLubyte*) malloc(tSizeW * tSizeH * sizeof(GLubyte) * 2);
glReadPixels(0, 0, tSizeW, tSizeH, GL_RGB, GL_UNSIGNED_SHORT_5_6_5, pixels);

int index = (x*tSizeH + y)*2;
unsigned int rgb = pixels[index + 1]*256 + pixels[index + 0];
unsigned int r = rgb;
r &= 0xF800;    // 1111 1000 0000 0000
r >>= 11;       // 0001 1111
r *= (255/31.); // Convert from 31 max to 255 max

unsigned int g = rgb;
g &= 0x7E0;     // 0000 0111 1110 0000
g >>= 5;        // 0011 1111
g *= (255/63.); // Convert from 63 max to 255 max

unsigned int b = rgb;
b &= 0x1F;      // 0000 0000 0001 1111
//g >>= 0;      // 0001 1111
b *= (255/31.); // Convert from 31 max to 255 max