Browse Source

Refactor JPEG decoding into decodeJPEGIntoSurface

v1.x
c-spencer 13 years ago
parent
commit
d5838bc4aa
  1. 128
      src/Image.cc
  2. 12
      src/Image.h

128
src/Image.cc

@ -12,11 +12,6 @@
#include <errno.h> #include <errno.h>
#include <node_buffer.h> #include <node_buffer.h>
#ifdef HAVE_JPEG
#include <jpeglib.h>
#include <jerror.h>
#endif
#ifdef HAVE_GIF #ifdef HAVE_GIF
#include <gif_lib.h> #include <gif_lib.h>
typedef struct { typedef struct {
@ -601,45 +596,28 @@ static void jpeg_mem_src (j_decompress_ptr cinfo, void* buffer, long nbytes) {
#endif #endif
/*
* Load jpeg from buffer.
*/
cairo_status_t cairo_status_t
Image::loadJPEGFromBuffer(uint8_t *buf, unsigned len) { Image::decodeJPEGIntoSurface(jpeg_decompress_struct *info) {
// TODO: remove this duplicate logic
// JPEG setup
struct jpeg_decompress_struct info;
struct jpeg_error_mgr err;
info.err = jpeg_std_error(&err);
jpeg_create_decompress(&info);
jpeg_mem_src(&info, buf, len);
jpeg_read_header(&info, 1);
jpeg_start_decompress(&info);
width = info.output_width;
height = info.output_height;
// Data alloc
int stride = width * 4; int stride = width * 4;
uint8_t *data = (uint8_t *) malloc(width * height * 4); cairo_status_t status;
uint8_t *data = (uint8_t *) malloc(width * height * 4);
if (!data) { if (!data) {
jpeg_finish_decompress(&info); jpeg_finish_decompress(info);
jpeg_destroy_decompress(&info); jpeg_destroy_decompress(info);
return CAIRO_STATUS_NO_MEMORY; return CAIRO_STATUS_NO_MEMORY;
} }
uint8_t *src = (uint8_t *) malloc(width * 3); uint8_t *src = (uint8_t *) malloc(width * 3);
if (!src) { if (!src) {
free(data); free(data);
jpeg_finish_decompress(&info); jpeg_finish_decompress(info);
jpeg_destroy_decompress(&info); jpeg_destroy_decompress(info);
return CAIRO_STATUS_NO_MEMORY; return CAIRO_STATUS_NO_MEMORY;
} }
// Copy RGB -> ARGB
for (int y = 0; y < height; ++y) { for (int y = 0; y < height; ++y) {
jpeg_read_scanlines(&info, &src, 1); jpeg_read_scanlines(info, &src, 1);
uint32_t *row = (uint32_t *)(data + stride * y); uint32_t *row = (uint32_t *)(data + stride * y);
for (int x = 0; x < width; ++x) { for (int x = 0; x < width; ++x) {
int bx = 3 * x; int bx = 3 * x;
@ -651,7 +629,6 @@ Image::loadJPEGFromBuffer(uint8_t *buf, unsigned len) {
} }
} }
// New image surface
_surface = cairo_image_surface_create_for_data( _surface = cairo_image_surface_create_for_data(
data data
, CAIRO_FORMAT_ARGB32 , CAIRO_FORMAT_ARGB32
@ -659,94 +636,69 @@ Image::loadJPEGFromBuffer(uint8_t *buf, unsigned len) {
, height , height
, cairo_format_stride_for_width(CAIRO_FORMAT_ARGB32, width)); , cairo_format_stride_for_width(CAIRO_FORMAT_ARGB32, width));
// Cleanup jpeg_finish_decompress(info);
free(src); jpeg_destroy_decompress(info);
jpeg_finish_decompress(&info); status = cairo_surface_status(_surface);
jpeg_destroy_decompress(&info);
cairo_status_t status = cairo_surface_status(_surface);
if (status) { if (status) {
free(data); free(data);
free(src);
return status; return status;
} }
free(src);
_data = data; _data = data;
return CAIRO_STATUS_SUCCESS; return CAIRO_STATUS_SUCCESS;
} }
/* /*
* Load JPEG, convert RGB to ARGB. * Load jpeg from buffer.
*/ */
cairo_status_t cairo_status_t
Image::loadJPEG(FILE *stream) { Image::loadJPEGFromBuffer(uint8_t *buf, unsigned len) {
// TODO: remove this duplicate logic
// JPEG setup // JPEG setup
struct jpeg_decompress_struct info; struct jpeg_decompress_struct info;
struct jpeg_error_mgr err; struct jpeg_error_mgr err;
info.err = jpeg_std_error(&err); info.err = jpeg_std_error(&err);
jpeg_create_decompress(&info); jpeg_create_decompress(&info);
jpeg_stdio_src(&info, stream);
jpeg_mem_src(&info, buf, len);
jpeg_read_header(&info, 1); jpeg_read_header(&info, 1);
jpeg_start_decompress(&info); jpeg_start_decompress(&info);
width = info.output_width; width = info.output_width;
height = info.output_height; height = info.output_height;
// Data alloc return decodeJPEGIntoSurface(&info);
int stride = width * 4; }
uint8_t *data = (uint8_t *) malloc(width * height * 4);
if (!data) { /*
fclose(stream); * Load JPEG, convert RGB to ARGB.
jpeg_finish_decompress(&info); */
jpeg_destroy_decompress(&info);
return CAIRO_STATUS_NO_MEMORY;
}
uint8_t *src = (uint8_t *) malloc(width * 3); cairo_status_t
Image::loadJPEG(FILE *stream) {
// JPEG setup
struct jpeg_decompress_struct info;
struct jpeg_error_mgr err;
info.err = jpeg_std_error(&err);
jpeg_create_decompress(&info);
if (!src) { jpeg_stdio_src(&info, stream);
free(data);
fclose(stream);
jpeg_finish_decompress(&info);
jpeg_destroy_decompress(&info);
return CAIRO_STATUS_NO_MEMORY;
}
// Copy RGB -> ARGB jpeg_read_header(&info, 1);
for (int y = 0; y < height; ++y) { jpeg_start_decompress(&info);
jpeg_read_scanlines(&info, &src, 1); width = info.output_width;
uint32_t *row = (uint32_t *)(data + stride * y); height = info.output_height;
for (int x = 0; x < width; ++x) {
int bx = 3 * x;
uint32_t *pixel = row + x;
*pixel = 255 << 24
| src[bx + 0] << 16
| src[bx + 1] << 8
| src[bx + 2];
}
}
// New image surface cairo_status_t status;
_surface = cairo_image_surface_create_for_data(
data
, CAIRO_FORMAT_ARGB32
, width
, height
, cairo_format_stride_for_width(CAIRO_FORMAT_ARGB32, width));
// Cleanup status = decodeJPEGIntoSurface(&info);
free(src);
fclose(stream); fclose(stream);
jpeg_finish_decompress(&info);
jpeg_destroy_decompress(&info);
cairo_status_t status = cairo_surface_status(_surface);
if (status) { return status;
free(data);
return status;
}
return CAIRO_STATUS_SUCCESS;
} }
#endif /* HAVE_JPEG */ #endif /* HAVE_JPEG */

12
src/Image.h

@ -10,6 +10,11 @@
#include "Canvas.h" #include "Canvas.h"
#ifdef HAVE_JPEG
#include <jpeglib.h>
#include <jerror.h>
#endif
class Image: public node::ObjectWrap { class Image: public node::ObjectWrap {
public: public:
char *filename; char *filename;
@ -47,6 +52,7 @@ class Image: public node::ObjectWrap {
#ifdef HAVE_JPEG #ifdef HAVE_JPEG
cairo_status_t loadJPEGFromBuffer(uint8_t *buf, unsigned len); cairo_status_t loadJPEGFromBuffer(uint8_t *buf, unsigned len);
cairo_status_t loadJPEG(FILE *stream); cairo_status_t loadJPEG(FILE *stream);
cairo_status_t decodeJPEGIntoSurface(jpeg_decompress_struct *info);
#endif #endif
void error(Local<Value> error); void error(Local<Value> error);
void loaded(); void loaded();
@ -59,6 +65,12 @@ class Image: public node::ObjectWrap {
, COMPLETE , COMPLETE
} state; } state;
enum {
DATA_IMAGE,
DATA_MIME,
DATA_IMAGE_AND_MIME
} data_mode;
typedef enum { typedef enum {
UNKNOWN UNKNOWN
, GIF , GIF

Loading…
Cancel
Save