int LoadJPGBuff( void *src_buffer, int src_size, unsigned char **pic, int *width, int *height ) { struct jpeg_decompress_struct cinfo; struct my_jpeg_error_mgr jerr; JSAMPARRAY buffer; int row_stride, size; cinfo.err = jpeg_std_error( &jerr.pub ); jerr.pub.error_exit = my_jpeg_error_exit; if ( setjmp( jerr.setjmp_buffer ) ) { *pic = (unsigned char*)errormsg; jpeg_destroy_decompress( &cinfo ); return -1; } jpeg_create_decompress( &cinfo ); jpeg_buffer_src( &cinfo, src_buffer, src_size ); jpeg_read_header( &cinfo, TRUE ); jpeg_start_decompress( &cinfo ); row_stride = cinfo.output_width * cinfo.output_components; size = cinfo.output_width * cinfo.output_height * 4; *width = cinfo.output_width; *height = cinfo.output_height; *pic = (unsigned char*)( malloc( size + 1 ) ); memset( *pic, 0, size + 1 ); buffer = ( *cinfo.mem->alloc_sarray )( ( j_common_ptr ) & cinfo, JPOOL_IMAGE, row_stride, 1 ); while ( cinfo.output_scanline < cinfo.output_height ) { jpeg_read_scanlines( &cinfo, buffer, 1 ); if ( cinfo.out_color_components == 4 ) { j_putRGBAScanline( buffer[0], cinfo.output_width, *pic, cinfo.output_scanline - 1 ); } else if ( cinfo.out_color_components == 3 ) { j_putRGBScanline( buffer[0], cinfo.output_width, *pic, cinfo.output_scanline - 1 ); } else if ( cinfo.out_color_components == 1 ) { j_putGrayScanlineToRGB( buffer[0], cinfo.output_width, *pic, cinfo.output_scanline - 1 ); } } jpeg_finish_decompress( &cinfo ); jpeg_destroy_decompress( &cinfo ); return 0; }
static Image* LoadJPGBuff_( const void *src_buffer, int src_size ){ struct jpeg_decompress_struct cinfo; struct my_jpeg_error_mgr jerr; cinfo.err = jpeg_std_error( &jerr.pub ); jerr.pub.error_exit = my_jpeg_error_exit; if ( setjmp( jerr.setjmp_buffer ) ) { //< TODO: use c++ exceptions instead of setjmp/longjmp to handle errors globalErrorStream() << "WARNING: JPEG library error: " << errormsg << "\n"; jpeg_destroy_decompress( &cinfo ); return 0; } jpeg_create_decompress( &cinfo ); jpeg_buffer_src( &cinfo, const_cast<void*>( src_buffer ), src_size ); jpeg_read_header( &cinfo, TRUE ); jpeg_start_decompress( &cinfo ); int row_stride = cinfo.output_width * cinfo.output_components; RGBAImage* image = new RGBAImage( cinfo.output_width, cinfo.output_height ); JSAMPARRAY buffer = ( *cinfo.mem->alloc_sarray )( ( j_common_ptr ) & cinfo, JPOOL_IMAGE, row_stride, 1 ); while ( cinfo.output_scanline < cinfo.output_height ) { jpeg_read_scanlines( &cinfo, buffer, 1 ); if ( cinfo.out_color_components == 4 ) { j_putRGBAScanline( buffer[0], cinfo.output_width, image->getRGBAPixels(), cinfo.output_scanline - 1 ); } else if ( cinfo.out_color_components == 3 ) { j_putRGBScanline( buffer[0], cinfo.output_width, image->getRGBAPixels(), cinfo.output_scanline - 1 ); } else if ( cinfo.out_color_components == 1 ) { j_putGrayScanlineToRGB( buffer[0], cinfo.output_width, image->getRGBAPixels(), cinfo.output_scanline - 1 ); } } jpeg_finish_decompress( &cinfo ); jpeg_destroy_decompress( &cinfo ); return image; }
// // read a JPEG file // BYTE* CJpeg::ReadJPEGFile(LPCSTR lpstrFileName, UINT *uWidth, UINT *uHeight) { // basic code from IJG Jpeg Code v6 example.c *uWidth=0; *uHeight=0; /* This struct contains the JPEG decompression parameters and pointers to * working space (which is allocated as needed by the JPEG library). */ struct jpeg_decompress_struct cinfo; /* We use our private extension JPEG error handler. * Note that this struct must live as long as the main JPEG parameter * struct, to avoid dangling-pointer problems. */ struct my_error_mgr jerr; /* More stuff */ FILE * infile; /* source file */ JSAMPARRAY buffer; /* Output row buffer */ int row_stride; /* physical row width in output buffer */ char buf[250]; /* In this example we want to open the input file before doing anything else, * so that the setjmp() error recovery below can assume the file is open. * VERY IMPORTANT: use "b" option to fopen() if you are on a machine that * requires it in order to read binary files. */ if ((infile = fopen(lpstrFileName, "rb")) == NULL) { sprintf(buf, "JPEG :\nCan't open %s\n", lpstrFileName); m_strJPEGError = buf; return NULL; } /* Step 1: allocate and initialize JPEG decompression object */ /* We set up the normal JPEG error routines, then override error_exit. */ cinfo.err = jpeg_std_error(&jerr.pub); jerr.pub.error_exit = my_error_exit; /* Establish the setjmp return context for my_error_exit to use. */ if (setjmp(jerr.setjmp_buffer)) { /* If we get here, the JPEG code has signaled an error. * We need to clean up the JPEG object, close the input file, and return. */ jpeg_destroy_decompress(&cinfo); fclose(infile); return NULL; } /* Now we can initialize the JPEG decompression object. */ jpeg_create_decompress(&cinfo); /* Step 2: specify data source (eg, a file) */ jpeg_stdio_src(&cinfo, infile); /* Step 3: read file parameters with jpeg_read_header() */ (void) jpeg_read_header(&cinfo, TRUE); /* We can ignore the return value from jpeg_read_header since * (a) suspension is not possible with the stdio data source, and * (b) we passed TRUE to reject a tables-only JPEG file as an error. * See libjpeg.doc for more info. */ /* Step 4: set parameters for decompression */ /* In this example, we don't need to change any of the defaults set by * jpeg_read_header(), so we do nothing here. */ /* Step 5: Start decompressor */ (void) jpeg_start_decompress(&cinfo); /* We can ignore the return value since suspension is not possible * with the stdio data source. */ /* We may need to do some setup of our own at this point before reading * the data. After jpeg_start_decompress() we have the correct scaled * output image dimensions available, as well as the output colormap * if we asked for color quantization. * In this example, we need to make an output work buffer of the right size. */ // get our buffer set to hold data BYTE *dataBuf; //////////////////////////////////////////////////////////// // alloc and open our new buffer dataBuf=(BYTE *)new BYTE[cinfo.output_width * 3 * cinfo.output_height]; if (dataBuf==NULL) { m_strJPEGError = "JpegFile :\nOut of memory"; jpeg_destroy_decompress(&cinfo); fclose(infile); return NULL; } // how big is this thing gonna be? *uWidth = cinfo.output_width; *uHeight = cinfo.output_height; /* JSAMPLEs per row in output buffer */ row_stride = cinfo.output_width * cinfo.output_components; /* Make a one-row-high sample array that will go away when done with image */ buffer = (*cinfo.mem->alloc_sarray) ((j_common_ptr) &cinfo, JPOOL_IMAGE, row_stride, 1); /* Step 6: while (scan lines remain to be read) */ /* jpeg_read_scanlines(...); */ /* Here we use the library's state variable cinfo.output_scanline as the * loop counter, so that we don't have to keep track ourselves. */ while (cinfo.output_scanline < cinfo.output_height) { /* jpeg_read_scanlines expects an array of pointers to scanlines. * Here the array is only one element long, but you could ask for * more than one scanline at a time if that's more convenient. */ (void) jpeg_read_scanlines(&cinfo, buffer, 1); /* Assume put_scanline_someplace wants a pointer and sample count. */ // asuumer all 3-components are RGBs if (cinfo.out_color_components==3) { j_putRGBScanline(buffer[0], *uWidth, dataBuf, cinfo.output_scanline-1); } else if (cinfo.out_color_components==1) { // assume all single component images are grayscale j_putGrayScanlineToRGB(buffer[0], *uWidth, dataBuf, cinfo.output_scanline-1); } } /* Step 7: Finish decompression */ (void) jpeg_finish_decompress(&cinfo); /* We can ignore the return value since suspension is not possible * with the stdio data source. */ /* Step 8: Release JPEG decompression object */ /* This is an important step since it will release a good deal of memory. */ jpeg_destroy_decompress(&cinfo); /* After finish_decompress, we can close the input file. * Here we postpone it until after no more JPEG errors are possible, * so as to simplify the setjmp error logic above. (Actually, I don't * think that jpeg_destroy can do an error exit, but why assume anything...) */ fclose(infile); /* At this point you may want to check to see whether any corrupt-data * warnings occurred (test whether jerr.pub.num_warnings is nonzero). */ return dataBuf; }
BYTE* CJpeg::ReadJPEGFile(LPCSTR lpstrFileName, UINT *uWidth, UINT *uHeight) { *uWidth=0; *uHeight=0; //定义JPEG文件的解压信息 struct jpeg_decompress_struct cinfo; //定义JPEG文件的错误信息 struct my_error_mgr jerr; //定义缓冲区 FILE * infile; JSAMPARRAY buffer; int row_stride; char buf[250]; //打开JPEG文件 if ((infile = fopen(lpstrFileName, "rb")) == NULL) { sprintf(buf, "JPEG :\nCan't open %s\n", lpstrFileName); m_strJPEGError = buf; return NULL; } //为JPEG文件解压对象分配内存并对其初始化 cinfo.err = jpeg_std_error(&jerr.pub); jerr.pub.error_exit = my_error_exit; if (setjmp(jerr.setjmp_buffer)) { jpeg_destroy_decompress(&cinfo); fclose(infile); return NULL; } jpeg_create_decompress(&cinfo); //设定数据源 jpeg_stdio_src(&cinfo, infile); //读取JPEG文件参数 (void) jpeg_read_header(&cinfo, TRUE); //开始解压 (void) jpeg_start_decompress(&cinfo); BYTE *dataBuf; dataBuf=(BYTE *)new BYTE[cinfo.output_width * 3 * cinfo.output_height]; if (dataBuf==NULL) { m_strJPEGError = "JpegFile :\nOut of memory"; jpeg_destroy_decompress(&cinfo); fclose(infile); return NULL; } *uWidth = cinfo.output_width; *uHeight = cinfo.output_height; row_stride = cinfo.output_width * cinfo.output_components; buffer = (*cinfo.mem->alloc_sarray) ((j_common_ptr) &cinfo, JPOOL_IMAGE, row_stride, 1); //读取扫描线 while (cinfo.output_scanline < cinfo.output_height) { (void) jpeg_read_scanlines(&cinfo, buffer, 1); if (cinfo.out_color_components==3) { j_putRGBScanline(buffer[0], *uWidth, dataBuf, cinfo.output_scanline-1); } else if (cinfo.out_color_components==1) { j_putGrayScanlineToRGB(buffer[0], *uWidth, dataBuf, cinfo.output_scanline-1); } } //完成解压 (void) jpeg_finish_decompress(&cinfo); //释放JPEG解压对象 jpeg_destroy_decompress(&cinfo); fclose(infile); return dataBuf; }