将SurfaceTexture保存为JPEG

时间:2012-11-14 18:33:14

标签: android opengl-es libjpeg

我尝试通过JNI保存我平板电脑上OpenGL ES 2修改过的摄像头输出。

为实现这一目标,我使用了NDK-r8b编译的libjpeg库。

我使用以下代码:

在渲染功能中:

renderImage();
if (iIsPictureRequired)
{
  savePicture();
  iIsPictureRequired=false;
}

保存程序:

bool Image::savePicture()
{
 bool l_res =false;
char p_filename[]={"/sdcard/Pictures/testPic.jpg"};
// Allocates the image buffer (RGBA)
int l_size = iWidth*iHeight*4*sizeof(GLubyte);
GLubyte *l_image = (GLubyte*)malloc(l_size);
if (l_image==NULL)
{
  LOGE("Image::savePicture:could not allocate %d bytes",l_size);
  return l_res;
}
// Reads pixels from the color buffer (byte-aligned)
glPixelStorei(GL_PACK_ALIGNMENT, 1);
checkGlError("glPixelStorei");
// Saves the pixel buffer
glReadPixels(0,0,iWidth,iHeight,GL_RGBA,GL_UNSIGNED_BYTE,l_image);
checkGlError("glReadPixels");
// Stores the file
FILE* l_file  = fopen(p_filename, "wb");
if (l_file==NULL)
 {
   LOGE("Image::savePicture:could not create %s:errno=%d",p_filename,errno);
   free(l_image);
   return l_res;
 }
 // JPEG structures
 struct jpeg_compress_struct cinfo;
 struct jpeg_error_mgr       jerr;

 cinfo.err = jpeg_std_error(&jerr);
 jerr.trace_level = 10;

 jpeg_create_compress(&cinfo);
 jpeg_stdio_dest(&cinfo, l_file);
 cinfo.image_width      = iWidth;
 cinfo.image_height     = iHeight;
 cinfo.input_components = 3;
 cinfo.in_color_space   = JCS_RGB;
 jpeg_set_defaults(&cinfo);

 // Image quality [0..100]
 jpeg_set_quality (&cinfo, 70, true);
 jpeg_start_compress(&cinfo, true);

 // Saves the buffer
 JSAMPROW row_pointer[1];          // pointer to a single row

 // JPEG stores the image from top to bottom (OpenGL does the opposite)
 while (cinfo.next_scanline < cinfo.image_height)
{
  row_pointer[0] = (JSAMPROW)&l_image[(cinfo.image_height-1-cinfo.next_scanline)* (cinfo.input_components)*iWidth];
  jpeg_write_scanlines(&cinfo, row_pointer, 1);
}
// End of the process
jpeg_finish_compress(&cinfo);
fclose(l_file);
free(l_image);
l_res =true;
return l_res;

}

显示正确但生成的JPEG似乎是三倍并且从左到右重叠。

Scaled image (original size is 1210x648)

我做错了什么?

1 个答案:

答案 0 :(得分:0)

似乎jpeg lib和画布的内部格式不匹配。其他似乎用RGBRGBRGB读取/编码,其他用RGBARGBARGBA读取。

如果其他一切都失败,您可以重新排列图像数据......

 char *dst_ptr = l_image; char *src_ptr = l_image;
 for (i=0;i<width*height;i++) { *dst_ptr++=*src_ptr++;
  *dst_ptr++=*src_ptr++; *dst_ptr++=*src_ptr++; src_ptr++; }
编辑:既然原因得到了验证,可能会有更简单的修改。 您可能能够以正确的格式从 gl像素缓冲区获取数据:

 int l_size = iWidth*iHeight*3*sizeof(GLubyte);
 ...
 glReadPixels(0,0,iWidth,iHeight,GL_RGB,GL_UNSIGNED_BYTE,l_image);

还有一条警告:如果这个编译,但输出是倾斜,那么这意味着你的屏幕宽度不是4的倍数,但是opengl想要开始每一个新行在双边界。但在这种情况下,崩溃是一个很好的机会,因为在这种情况下l_size应该比预期的大1,2或3个字节。