|
@ -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 */ |
|
|