mirror of
https://github.com/python-pillow/Pillow.git
synced 2025-01-19 13:54:18 +03:00
637 lines
18 KiB
C
637 lines
18 KiB
C
/*
|
|
* The Python Imaging Library.
|
|
* $Id$
|
|
*
|
|
* decoder for JPEG2000 image data.
|
|
*
|
|
* history:
|
|
* 2014-03-12 ajh Created
|
|
*
|
|
* Copyright (c) 2014 Coriolis Systems Limited
|
|
* Copyright (c) 2014 Alastair Houghton
|
|
*
|
|
* See the README file for details on usage and redistribution.
|
|
*/
|
|
|
|
#include "Imaging.h"
|
|
|
|
#ifdef HAVE_OPENJPEG
|
|
|
|
#include "Jpeg2K.h"
|
|
|
|
#define CINEMA_24_CS_LENGTH 1302083
|
|
#define CINEMA_48_CS_LENGTH 651041
|
|
#define COMP_24_CS_MAX_LENGTH 1041666
|
|
#define COMP_48_CS_MAX_LENGTH 520833
|
|
|
|
/* -------------------------------------------------------------------- */
|
|
/* Error handler */
|
|
/* -------------------------------------------------------------------- */
|
|
|
|
static void
|
|
j2k_error(const char *msg, void *client_data) {
|
|
JPEG2KENCODESTATE *state = (JPEG2KENCODESTATE *)client_data;
|
|
free((void *)state->error_msg);
|
|
state->error_msg = strdup(msg);
|
|
}
|
|
|
|
static void
|
|
j2k_warn(const char *msg, void *client_data) {
|
|
// Null handler
|
|
}
|
|
|
|
/* -------------------------------------------------------------------- */
|
|
/* Buffer output stream */
|
|
/* -------------------------------------------------------------------- */
|
|
|
|
static OPJ_SIZE_T
|
|
j2k_write(void *p_buffer, OPJ_SIZE_T p_nb_bytes, void *p_user_data) {
|
|
ImagingCodecState state = (ImagingCodecState)p_user_data;
|
|
unsigned int result;
|
|
|
|
result = _imaging_write_pyFd(state->fd, p_buffer, p_nb_bytes);
|
|
|
|
return result ? result : (OPJ_SIZE_T)-1;
|
|
}
|
|
|
|
static OPJ_OFF_T
|
|
j2k_skip(OPJ_OFF_T p_nb_bytes, void *p_user_data) {
|
|
ImagingCodecState state = (ImagingCodecState)p_user_data;
|
|
char *buffer;
|
|
int result;
|
|
|
|
/* Explicitly write zeros */
|
|
buffer = calloc(p_nb_bytes, 1);
|
|
if (!buffer) {
|
|
return (OPJ_OFF_T)-1;
|
|
}
|
|
|
|
result = _imaging_write_pyFd(state->fd, buffer, p_nb_bytes);
|
|
|
|
free(buffer);
|
|
|
|
return result ? result : p_nb_bytes;
|
|
}
|
|
|
|
static OPJ_BOOL
|
|
j2k_seek(OPJ_OFF_T p_nb_bytes, void *p_user_data) {
|
|
ImagingCodecState state = (ImagingCodecState)p_user_data;
|
|
off_t pos = 0;
|
|
|
|
_imaging_seek_pyFd(state->fd, p_nb_bytes, SEEK_SET);
|
|
pos = _imaging_tell_pyFd(state->fd);
|
|
|
|
return pos == p_nb_bytes;
|
|
}
|
|
|
|
/* -------------------------------------------------------------------- */
|
|
/* Encoder */
|
|
/* -------------------------------------------------------------------- */
|
|
|
|
typedef void (*j2k_pack_tile_t)(
|
|
Imaging im, UINT8 *buf, unsigned x0, unsigned y0, unsigned w, unsigned h);
|
|
|
|
static void
|
|
j2k_pack_l(Imaging im, UINT8 *buf, unsigned x0, unsigned y0, unsigned w, unsigned h) {
|
|
UINT8 *ptr = buf;
|
|
unsigned x, y;
|
|
for (y = 0; y < h; ++y) {
|
|
UINT8 *data = (UINT8 *)(im->image[y + y0] + x0);
|
|
for (x = 0; x < w; ++x) {
|
|
*ptr++ = *data++;
|
|
}
|
|
}
|
|
}
|
|
|
|
static void
|
|
j2k_pack_i16(Imaging im, UINT8 *buf, unsigned x0, unsigned y0, unsigned w, unsigned h) {
|
|
UINT8 *ptr = buf;
|
|
unsigned x, y;
|
|
for (y = 0; y < h; ++y) {
|
|
UINT8 *data = (UINT8 *)(im->image[y + y0] + x0);
|
|
for (x = 0; x < w; ++x) {
|
|
#ifdef WORDS_BIGENDIAN
|
|
ptr[0] = data[1];
|
|
ptr[1] = data[0];
|
|
#else
|
|
ptr[0] = data[0];
|
|
ptr[1] = data[1];
|
|
#endif
|
|
ptr += 2;
|
|
data += 2;
|
|
}
|
|
}
|
|
}
|
|
|
|
static void
|
|
j2k_pack_la(Imaging im, UINT8 *buf, unsigned x0, unsigned y0, unsigned w, unsigned h) {
|
|
UINT8 *ptr = buf;
|
|
UINT8 *ptra = buf + w * h;
|
|
unsigned x, y;
|
|
for (y = 0; y < h; ++y) {
|
|
UINT8 *data = (UINT8 *)(im->image[y + y0] + 4 * x0);
|
|
for (x = 0; x < w; ++x) {
|
|
*ptr++ = data[0];
|
|
*ptra++ = data[3];
|
|
data += 4;
|
|
}
|
|
}
|
|
}
|
|
|
|
static void
|
|
j2k_pack_rgb(Imaging im, UINT8 *buf, unsigned x0, unsigned y0, unsigned w, unsigned h) {
|
|
UINT8 *pr = buf;
|
|
UINT8 *pg = pr + w * h;
|
|
UINT8 *pb = pg + w * h;
|
|
unsigned x, y;
|
|
for (y = 0; y < h; ++y) {
|
|
UINT8 *data = (UINT8 *)(im->image[y + y0] + 4 * x0);
|
|
for (x = 0; x < w; ++x) {
|
|
*pr++ = data[0];
|
|
*pg++ = data[1];
|
|
*pb++ = data[2];
|
|
data += 4;
|
|
}
|
|
}
|
|
}
|
|
|
|
static void
|
|
j2k_pack_rgba(
|
|
Imaging im, UINT8 *buf, unsigned x0, unsigned y0, unsigned w, unsigned h) {
|
|
UINT8 *pr = buf;
|
|
UINT8 *pg = pr + w * h;
|
|
UINT8 *pb = pg + w * h;
|
|
UINT8 *pa = pb + w * h;
|
|
unsigned x, y;
|
|
for (y = 0; y < h; ++y) {
|
|
UINT8 *data = (UINT8 *)(im->image[y + y0] + 4 * x0);
|
|
for (x = 0; x < w; ++x) {
|
|
*pr++ = *data++;
|
|
*pg++ = *data++;
|
|
*pb++ = *data++;
|
|
*pa++ = *data++;
|
|
}
|
|
}
|
|
}
|
|
|
|
enum {
|
|
J2K_STATE_START = 0,
|
|
J2K_STATE_ENCODING = 1,
|
|
J2K_STATE_DONE = 2,
|
|
J2K_STATE_FAILED = 3,
|
|
};
|
|
|
|
static void
|
|
j2k_set_cinema_params(Imaging im, int components, opj_cparameters_t *params) {
|
|
float rate;
|
|
int n;
|
|
|
|
/* These settings have been copied from opj_compress in the OpenJPEG
|
|
sources. */
|
|
|
|
params->tile_size_on = OPJ_FALSE;
|
|
params->cp_tdx = params->cp_tdy = 1;
|
|
params->tp_flag = 'C';
|
|
params->tp_on = 1;
|
|
params->cp_tx0 = params->cp_ty0 = 0;
|
|
params->image_offset_x0 = params->image_offset_y0 = 0;
|
|
params->cblockw_init = 32;
|
|
params->cblockh_init = 32;
|
|
params->csty |= 0x01;
|
|
params->prog_order = OPJ_CPRL;
|
|
params->roi_compno = -1;
|
|
params->subsampling_dx = params->subsampling_dy = 1;
|
|
params->irreversible = 1;
|
|
|
|
if (params->cp_cinema == OPJ_CINEMA4K_24) {
|
|
float max_rate =
|
|
((float)(components * im->xsize * im->ysize * 8) /
|
|
(CINEMA_24_CS_LENGTH * 8));
|
|
|
|
params->POC[0].tile = 1;
|
|
params->POC[0].resno0 = 0;
|
|
params->POC[0].compno0 = 0;
|
|
params->POC[0].layno1 = 1;
|
|
params->POC[0].resno1 = params->numresolution - 1;
|
|
params->POC[0].compno1 = 3;
|
|
params->POC[0].prg1 = OPJ_CPRL;
|
|
params->POC[1].tile = 1;
|
|
params->POC[1].resno0 = 0;
|
|
params->POC[1].compno0 = 0;
|
|
params->POC[1].layno1 = 1;
|
|
params->POC[1].resno1 = params->numresolution - 1;
|
|
params->POC[1].compno1 = 3;
|
|
params->POC[1].prg1 = OPJ_CPRL;
|
|
params->numpocs = 2;
|
|
|
|
for (n = 0; n < params->tcp_numlayers; ++n) {
|
|
rate = 0;
|
|
if (params->tcp_rates[0] == 0) {
|
|
params->tcp_rates[n] = max_rate;
|
|
} else {
|
|
rate =
|
|
((float)(components * im->xsize * im->ysize * 8) /
|
|
(params->tcp_rates[n] * 8));
|
|
if (rate > CINEMA_24_CS_LENGTH) {
|
|
params->tcp_rates[n] = max_rate;
|
|
}
|
|
}
|
|
}
|
|
|
|
params->max_comp_size = COMP_24_CS_MAX_LENGTH;
|
|
} else {
|
|
float max_rate =
|
|
((float)(components * im->xsize * im->ysize * 8) /
|
|
(CINEMA_48_CS_LENGTH * 8));
|
|
|
|
for (n = 0; n < params->tcp_numlayers; ++n) {
|
|
rate = 0;
|
|
if (params->tcp_rates[0] == 0) {
|
|
params->tcp_rates[n] = max_rate;
|
|
} else {
|
|
rate =
|
|
((float)(components * im->xsize * im->ysize * 8) /
|
|
(params->tcp_rates[n] * 8));
|
|
if (rate > CINEMA_48_CS_LENGTH) {
|
|
params->tcp_rates[n] = max_rate;
|
|
}
|
|
}
|
|
}
|
|
|
|
params->max_comp_size = COMP_48_CS_MAX_LENGTH;
|
|
}
|
|
}
|
|
|
|
static int
|
|
j2k_encode_entry(Imaging im, ImagingCodecState state) {
|
|
JPEG2KENCODESTATE *context = (JPEG2KENCODESTATE *)state->context;
|
|
opj_stream_t *stream = NULL;
|
|
opj_image_t *image = NULL;
|
|
opj_codec_t *codec = NULL;
|
|
opj_cparameters_t params;
|
|
unsigned components;
|
|
OPJ_COLOR_SPACE color_space;
|
|
opj_image_cmptparm_t image_params[4];
|
|
unsigned xsiz, ysiz;
|
|
unsigned tile_width, tile_height;
|
|
unsigned tiles_x, tiles_y;
|
|
unsigned x, y, tile_ndx;
|
|
unsigned n;
|
|
j2k_pack_tile_t pack;
|
|
int ret = -1;
|
|
|
|
unsigned prec = 8;
|
|
unsigned bpp = 8;
|
|
unsigned _overflow_scale_factor;
|
|
|
|
stream = opj_stream_create(BUFFER_SIZE, OPJ_FALSE);
|
|
|
|
if (!stream) {
|
|
state->errcode = IMAGING_CODEC_BROKEN;
|
|
state->state = J2K_STATE_FAILED;
|
|
goto quick_exit;
|
|
}
|
|
|
|
opj_stream_set_write_function(stream, j2k_write);
|
|
opj_stream_set_skip_function(stream, j2k_skip);
|
|
opj_stream_set_seek_function(stream, j2k_seek);
|
|
|
|
/* OpenJPEG 2.0 doesn't have OPJ_VERSION_MAJOR */
|
|
#ifndef OPJ_VERSION_MAJOR
|
|
opj_stream_set_user_data(stream, state);
|
|
#else
|
|
opj_stream_set_user_data(stream, state, NULL);
|
|
#endif
|
|
|
|
/* Setup an opj_image */
|
|
if (strcmp(im->mode, "L") == 0) {
|
|
components = 1;
|
|
color_space = OPJ_CLRSPC_GRAY;
|
|
pack = j2k_pack_l;
|
|
} else if (strcmp(im->mode, "I;16") == 0 || strcmp(im->mode, "I;16B") == 0) {
|
|
components = 1;
|
|
color_space = OPJ_CLRSPC_GRAY;
|
|
pack = j2k_pack_i16;
|
|
prec = 16;
|
|
bpp = 12;
|
|
} else if (strcmp(im->mode, "LA") == 0) {
|
|
components = 2;
|
|
color_space = OPJ_CLRSPC_GRAY;
|
|
pack = j2k_pack_la;
|
|
} else if (strcmp(im->mode, "RGB") == 0) {
|
|
components = 3;
|
|
color_space = OPJ_CLRSPC_SRGB;
|
|
pack = j2k_pack_rgb;
|
|
} else if (strcmp(im->mode, "YCbCr") == 0) {
|
|
components = 3;
|
|
color_space = OPJ_CLRSPC_SYCC;
|
|
pack = j2k_pack_rgb;
|
|
} else if (strcmp(im->mode, "RGBA") == 0) {
|
|
components = 4;
|
|
color_space = OPJ_CLRSPC_SRGB;
|
|
pack = j2k_pack_rgba;
|
|
} else {
|
|
state->errcode = IMAGING_CODEC_BROKEN;
|
|
state->state = J2K_STATE_FAILED;
|
|
goto quick_exit;
|
|
}
|
|
|
|
for (n = 0; n < components; ++n) {
|
|
image_params[n].dx = image_params[n].dy = 1;
|
|
image_params[n].w = im->xsize;
|
|
image_params[n].h = im->ysize;
|
|
image_params[n].x0 = image_params[n].y0 = 0;
|
|
image_params[n].prec = prec;
|
|
image_params[n].bpp = bpp;
|
|
image_params[n].sgnd = 0;
|
|
}
|
|
|
|
image = opj_image_create(components, image_params, color_space);
|
|
if (!image) {
|
|
state->errcode = IMAGING_CODEC_BROKEN;
|
|
state->state = J2K_STATE_FAILED;
|
|
goto quick_exit;
|
|
}
|
|
|
|
/* Setup compression context */
|
|
context->error_msg = NULL;
|
|
|
|
opj_set_default_encoder_parameters(¶ms);
|
|
|
|
params.image_offset_x0 = context->offset_x;
|
|
params.image_offset_y0 = context->offset_y;
|
|
|
|
if (context->tile_size_x && context->tile_size_y) {
|
|
params.tile_size_on = OPJ_TRUE;
|
|
params.cp_tx0 = context->tile_offset_x;
|
|
params.cp_ty0 = context->tile_offset_y;
|
|
params.cp_tdx = context->tile_size_x;
|
|
params.cp_tdy = context->tile_size_y;
|
|
|
|
tile_width = params.cp_tdx;
|
|
tile_height = params.cp_tdy;
|
|
} else {
|
|
params.cp_tx0 = 0;
|
|
params.cp_ty0 = 0;
|
|
params.cp_tdx = 1;
|
|
params.cp_tdy = 1;
|
|
|
|
tile_width = im->xsize;
|
|
tile_height = im->ysize;
|
|
}
|
|
|
|
if (context->quality_layers && PySequence_Check(context->quality_layers)) {
|
|
Py_ssize_t len = PySequence_Length(context->quality_layers);
|
|
Py_ssize_t n;
|
|
float *pq;
|
|
|
|
if (len > 0) {
|
|
if ((size_t)len >
|
|
sizeof(params.tcp_rates) / sizeof(params.tcp_rates[0])) {
|
|
len = sizeof(params.tcp_rates) / sizeof(params.tcp_rates[0]);
|
|
}
|
|
|
|
params.tcp_numlayers = (int)len;
|
|
|
|
if (context->quality_is_in_db) {
|
|
params.cp_disto_alloc = params.cp_fixed_alloc = 0;
|
|
params.cp_fixed_quality = 1;
|
|
pq = params.tcp_distoratio;
|
|
} else {
|
|
params.cp_disto_alloc = 1;
|
|
params.cp_fixed_alloc = params.cp_fixed_quality = 0;
|
|
pq = params.tcp_rates;
|
|
}
|
|
|
|
for (n = 0; n < len; ++n) {
|
|
PyObject *obj = PySequence_ITEM(context->quality_layers, n);
|
|
pq[n] = PyFloat_AsDouble(obj);
|
|
}
|
|
}
|
|
} else {
|
|
params.tcp_numlayers = 1;
|
|
params.tcp_rates[0] = 0;
|
|
params.cp_disto_alloc = 1;
|
|
}
|
|
|
|
if (context->num_resolutions) {
|
|
params.numresolution = context->num_resolutions;
|
|
}
|
|
|
|
if (context->cblk_width >= 4 && context->cblk_width <= 1024 &&
|
|
context->cblk_height >= 4 && context->cblk_height <= 1024 &&
|
|
context->cblk_width * context->cblk_height <= 4096) {
|
|
params.cblockw_init = context->cblk_width;
|
|
params.cblockh_init = context->cblk_height;
|
|
}
|
|
|
|
if (context->precinct_width >= 4 && context->precinct_height >= 4 &&
|
|
context->precinct_width >= context->cblk_width &&
|
|
context->precinct_height > context->cblk_height) {
|
|
params.prcw_init[0] = context->precinct_width;
|
|
params.prch_init[0] = context->precinct_height;
|
|
params.res_spec = 1;
|
|
params.csty |= 0x01;
|
|
}
|
|
|
|
params.irreversible = context->irreversible;
|
|
|
|
params.prog_order = context->progression;
|
|
|
|
params.cp_cinema = context->cinema_mode;
|
|
|
|
switch (params.cp_cinema) {
|
|
case OPJ_OFF:
|
|
params.cp_rsiz = OPJ_STD_RSIZ;
|
|
break;
|
|
case OPJ_CINEMA2K_24:
|
|
case OPJ_CINEMA2K_48:
|
|
params.cp_rsiz = OPJ_CINEMA2K;
|
|
if (params.numresolution > 6) {
|
|
params.numresolution = 6;
|
|
}
|
|
break;
|
|
case OPJ_CINEMA4K_24:
|
|
params.cp_rsiz = OPJ_CINEMA4K;
|
|
if (params.numresolution > 7) {
|
|
params.numresolution = 7;
|
|
}
|
|
break;
|
|
}
|
|
|
|
if (!context->num_resolutions) {
|
|
while (tile_width < (1 << (params.numresolution - 1U)) || tile_height < (1 << (params.numresolution - 1U))) {
|
|
params.numresolution -= 1;
|
|
}
|
|
}
|
|
|
|
if (context->cinema_mode != OPJ_OFF) {
|
|
j2k_set_cinema_params(im, components, ¶ms);
|
|
}
|
|
|
|
/* Set up the reference grid in the image */
|
|
image->x0 = params.image_offset_x0;
|
|
image->y0 = params.image_offset_y0;
|
|
image->x1 = xsiz = im->xsize + params.image_offset_x0;
|
|
image->y1 = ysiz = im->ysize + params.image_offset_y0;
|
|
|
|
/* Create the compressor */
|
|
codec = opj_create_compress(context->format);
|
|
|
|
if (!codec) {
|
|
state->errcode = IMAGING_CODEC_BROKEN;
|
|
state->state = J2K_STATE_FAILED;
|
|
goto quick_exit;
|
|
}
|
|
|
|
opj_set_error_handler(codec, j2k_error, context);
|
|
opj_set_info_handler(codec, j2k_warn, context);
|
|
opj_set_warning_handler(codec, j2k_warn, context);
|
|
opj_setup_encoder(codec, ¶ms, image);
|
|
|
|
/* Start encoding */
|
|
if (!opj_start_compress(codec, image, stream)) {
|
|
state->errcode = IMAGING_CODEC_BROKEN;
|
|
state->state = J2K_STATE_FAILED;
|
|
goto quick_exit;
|
|
}
|
|
|
|
/* Write each tile */
|
|
tiles_x = (im->xsize + (params.image_offset_x0 - params.cp_tx0) + tile_width - 1) /
|
|
tile_width;
|
|
tiles_y = (im->ysize + (params.image_offset_y0 - params.cp_ty0) + tile_height - 1) /
|
|
tile_height;
|
|
|
|
/* check for integer overflow for the malloc line, checking any expression
|
|
that may multiply either tile_width or tile_height */
|
|
_overflow_scale_factor = components * prec;
|
|
if ((tile_width > UINT_MAX / _overflow_scale_factor) ||
|
|
(tile_height > UINT_MAX / _overflow_scale_factor) ||
|
|
(tile_width > UINT_MAX / (tile_height * _overflow_scale_factor)) ||
|
|
(tile_height > UINT_MAX / (tile_width * _overflow_scale_factor))) {
|
|
state->errcode = IMAGING_CODEC_BROKEN;
|
|
state->state = J2K_STATE_FAILED;
|
|
goto quick_exit;
|
|
}
|
|
/* malloc check ok, checked for overflow above */
|
|
state->buffer = malloc(tile_width * tile_height * components * prec / 8);
|
|
if (!state->buffer) {
|
|
state->errcode = IMAGING_CODEC_BROKEN;
|
|
state->state = J2K_STATE_FAILED;
|
|
goto quick_exit;
|
|
}
|
|
|
|
tile_ndx = 0;
|
|
for (y = 0; y < tiles_y; ++y) {
|
|
int ty0 = params.cp_ty0 + y * tile_height;
|
|
unsigned ty1 = ty0 + tile_height;
|
|
unsigned pixy, pixh;
|
|
|
|
if (ty0 < params.image_offset_y0) {
|
|
ty0 = params.image_offset_y0;
|
|
}
|
|
if (ty1 > ysiz) {
|
|
ty1 = ysiz;
|
|
}
|
|
|
|
pixy = ty0 - params.image_offset_y0;
|
|
pixh = ty1 - ty0;
|
|
|
|
for (x = 0; x < tiles_x; ++x) {
|
|
int tx0 = params.cp_tx0 + x * tile_width;
|
|
unsigned tx1 = tx0 + tile_width;
|
|
unsigned pixx, pixw;
|
|
unsigned data_size;
|
|
|
|
if (tx0 < params.image_offset_x0) {
|
|
tx0 = params.image_offset_x0;
|
|
}
|
|
if (tx1 > xsiz) {
|
|
tx1 = xsiz;
|
|
}
|
|
|
|
pixx = tx0 - params.image_offset_x0;
|
|
pixw = tx1 - tx0;
|
|
|
|
pack(im, state->buffer, pixx, pixy, pixw, pixh);
|
|
|
|
data_size = pixw * pixh * components * prec / 8;
|
|
|
|
if (!opj_write_tile(codec, tile_ndx++, state->buffer, data_size, stream)) {
|
|
state->errcode = IMAGING_CODEC_BROKEN;
|
|
state->state = J2K_STATE_FAILED;
|
|
goto quick_exit;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (!opj_end_compress(codec, stream)) {
|
|
state->errcode = IMAGING_CODEC_BROKEN;
|
|
state->state = J2K_STATE_FAILED;
|
|
goto quick_exit;
|
|
}
|
|
|
|
state->errcode = IMAGING_CODEC_END;
|
|
state->state = J2K_STATE_DONE;
|
|
ret = -1;
|
|
|
|
quick_exit:
|
|
if (codec) {
|
|
opj_destroy_codec(codec);
|
|
}
|
|
if (image) {
|
|
opj_image_destroy(image);
|
|
}
|
|
if (stream) {
|
|
opj_stream_destroy(stream);
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
int
|
|
ImagingJpeg2KEncode(Imaging im, ImagingCodecState state, UINT8 *buf, int bytes) {
|
|
if (state->state == J2K_STATE_FAILED) {
|
|
return -1;
|
|
}
|
|
|
|
if (state->state == J2K_STATE_START) {
|
|
state->state = J2K_STATE_ENCODING;
|
|
|
|
return j2k_encode_entry(im, state);
|
|
}
|
|
|
|
return -1;
|
|
}
|
|
|
|
/* -------------------------------------------------------------------- */
|
|
/* Cleanup */
|
|
/* -------------------------------------------------------------------- */
|
|
|
|
int
|
|
ImagingJpeg2KEncodeCleanup(ImagingCodecState state) {
|
|
JPEG2KENCODESTATE *context = (JPEG2KENCODESTATE *)state->context;
|
|
|
|
if (context->quality_layers) {
|
|
Py_XDECREF(context->quality_layers);
|
|
context->quality_layers = NULL;
|
|
}
|
|
|
|
if (context->error_msg) {
|
|
free((void *)context->error_msg);
|
|
}
|
|
|
|
context->error_msg = NULL;
|
|
|
|
return -1;
|
|
}
|
|
|
|
#endif /* HAVE_OPENJPEG */
|
|
|
|
/*
|
|
* Local Variables:
|
|
* c-basic-offset: 4
|
|
* End:
|
|
*
|
|
*/
|