ECC-303: Cannot build with OpenJPEG version 2

This commit is contained in:
Shahram Najm 2016-07-25 15:57:48 +01:00
parent 7dccccb936
commit fd1b31edec
1 changed files with 330 additions and 12 deletions

View File

@ -11,17 +11,17 @@
#include "grib_api_internal.h"
#if HAVE_LIBOPENJPEG
#include "openjpeg.h"
/* TODO: Strip the \n from msg */
static void openjpeg_warning(const char *msg, void *client_data)
{
grib_context_log((grib_context*)client_data,GRIB_LOG_WARNING,"OPENJPEG: %s",msg);
grib_context_log((grib_context*)client_data,GRIB_LOG_WARNING,"openjpeg: %s",msg);
}
static void openjpeg_error(const char *msg, void *client_data)
{
grib_context_log((grib_context*)client_data,GRIB_LOG_ERROR,"OPENJPEG: %s",msg);
grib_context_log((grib_context*)client_data,GRIB_LOG_ERROR,"openjpeg: %s",msg);
}
static void openjpeg_info(const char *msg, void *client_data)
@ -29,8 +29,12 @@ static void openjpeg_info(const char *msg, void *client_data)
/* grib_context_log((grib_context*)client_data,GRIB_LOG_INFO,"openjpeg: %s",msg); */
}
int grib_openjpeg_encode(grib_context *c, j2k_encode_helper *helper) {
/* Note: The old OpenJPEG versions (e.g. v1.5.2) did not have this macro.
* From OpenJPEG v2.1.0 onwards there is the macro OPJ_VERSION_MAJOR */
#if !defined(OPJ_VERSION_MAJOR) /* The old interface */
int grib_openjpeg_encode(grib_context *c, j2k_encode_helper *helper)
{
int err = GRIB_SUCCESS;
const int numcomps = 1;
@ -54,6 +58,8 @@ int grib_openjpeg_encode(grib_context *c, j2k_encode_helper *helper) {
/* set encoding parameters to default values */
opj_set_default_encoder_parameters(&parameters);
grib_context_log(c, GRIB_LOG_DEBUG, "grib_openjpeg_encode: OpenJPEG version %s", opj_version());
parameters.tcp_numlayers = 1;
parameters.cp_disto_alloc = 1;
parameters.tcp_rates[0] = helper->compression;
@ -139,11 +145,11 @@ int grib_openjpeg_decode(grib_context *c, unsigned char *buf, size_t *buflen, do
opj_image_t *image = NULL;
opj_image_comp_t comp = {0,};
/* set decoding parameters to default values */
opj_set_default_decoder_parameters(&parameters);
/* JPEG-2000 codestream */
grib_context_log(c, GRIB_LOG_DEBUG, "grib_openjpeg_decode: OpenJPEG version %s", opj_version());
/* get a decoder handle */
dinfo = opj_create_decompress(CODEC_J2K);
@ -164,7 +170,7 @@ int grib_openjpeg_decode(grib_context *c, unsigned char *buf, size_t *buflen, do
image = opj_decode(dinfo, cio);
if(!image) {
grib_context_log(c, GRIB_LOG_ERROR, "OPENJPEG: failed to decode image");
grib_context_log(c, GRIB_LOG_ERROR, "openjpeg: failed to decode image");
err = GRIB_DECODING_ERROR;
goto cleanup;
}
@ -201,19 +207,331 @@ cleanup:
return err;
}
#else
#else /* OPENJPEG VERSION 2 */
/* OpenJPEG 2.1 version of grib_openjpeg_encoding.c */
/* opj_* Helper code from https://groups.google.com/forum/#!topic/openjpeg/8cebr0u7JgY */
/* These routines are added to use memory instead of a file for input and output */
/* struct need to treat memory as a stream */
typedef struct
{
OPJ_UINT8* pData; /* our data */
OPJ_SIZE_T dataSize; /* how big is our data */
OPJ_SIZE_T offset; /* where we are currently in our data */
j2k_encode_helper *helper;
} opj_memory_stream;
/* This will read from our memory to the buffer */
static OPJ_SIZE_T opj_memory_stream_read(void *buffer, OPJ_SIZE_T nb_bytes, void * p_user_data)
{
opj_memory_stream* mstream = (opj_memory_stream*) p_user_data; /* Our data */
OPJ_SIZE_T nb_bytes_read = nb_bytes; /* Amount to move to buffer */
/* Check if the current offset is outside our data buffer */
if (mstream->offset >= mstream->dataSize)
return (OPJ_SIZE_T) -1;
/* Check if we are reading more than we have */
if (nb_bytes > (mstream->dataSize - mstream->offset))
nb_bytes_read = mstream->dataSize - mstream->offset;
memcpy(buffer, &(mstream->pData[mstream->offset]), nb_bytes_read);
mstream->offset += nb_bytes_read; /* Update the pointer to the new location */
return nb_bytes_read;
}
/* Write from the buffer to our memory */
static OPJ_SIZE_T opj_memory_stream_write(void *buffer, OPJ_SIZE_T nb_bytes, void *user_data)
{
opj_memory_stream* mstream = (opj_memory_stream*) user_data; /* our data */
OPJ_SIZE_T nb_bytes_write = nb_bytes; /* Amount to move to buffer */
/* Check if the current offset is outside our data buffer */
if (mstream->offset >= mstream->dataSize)
return (OPJ_SIZE_T)-1;
/* Check if we are writing more than we have space for */
if (nb_bytes > (mstream->dataSize - mstream->offset))
nb_bytes_write = mstream->dataSize - mstream->offset;
/* Copy the data from the internal buffer */
memcpy(&(mstream->pData[mstream->offset]), buffer, nb_bytes_write);
mstream->offset += nb_bytes_write; /* Update the pointer to the new location */
return nb_bytes_write;
}
/* Moves the pointer forward, but never more than we have */
static OPJ_OFF_T opj_memory_stream_skip(OPJ_OFF_T nb_bytes, void *user_data)
{
opj_memory_stream* mstream = (opj_memory_stream*) user_data;
OPJ_SIZE_T l_nb_bytes;
if (nb_bytes < 0)
return -1; /* No skipping backwards */
l_nb_bytes = (OPJ_SIZE_T) nb_bytes; /* Allowed because it is positive */
/* Do not allow jumping past the end */
if (l_nb_bytes > mstream->dataSize - mstream->offset)
nb_bytes = mstream->dataSize - mstream->offset;
mstream->offset += l_nb_bytes;
return l_nb_bytes; /* Returm how far we jumped */
}
/* Sets the pointer to anywhere in the memory */
static OPJ_BOOL opj_memory_stream_seek(OPJ_OFF_T nb_bytes, void * user_data)
{
opj_memory_stream* mstream = (opj_memory_stream*) user_data;
if (nb_bytes < 0)
return OPJ_FALSE; /* Not before the buffer */
if (nb_bytes >(OPJ_OFF_T) mstream->dataSize)
return OPJ_FALSE; /* Not after the buffer */
mstream->offset = (OPJ_SIZE_T) nb_bytes; /* Move to new position */
return OPJ_TRUE;
}
static void opj_memory_stream_do_nothing(void * p_user_data)
{
OPJ_ARG_NOT_USED(p_user_data);
}
/* Create a stream to use memory as the input or output */
opj_stream_t* opj_stream_create_default_memory_stream(opj_memory_stream* memoryStream, OPJ_BOOL is_read_stream)
{
opj_stream_t* stream;
if (!(stream = opj_stream_default_create(is_read_stream)))
return (NULL);
/* Set how to work with the frame buffer */
if (is_read_stream)
opj_stream_set_read_function(stream, opj_memory_stream_read);
else
opj_stream_set_write_function(stream, opj_memory_stream_write);
opj_stream_set_seek_function(stream, opj_memory_stream_seek);
opj_stream_set_skip_function(stream, opj_memory_stream_skip);
opj_stream_set_user_data(stream, memoryStream, opj_memory_stream_do_nothing);
opj_stream_set_user_data_length(stream, memoryStream->dataSize);
return stream;
}
int grib_openjpeg_encode(grib_context *c, j2k_encode_helper *helper) {
int err = GRIB_SUCCESS;
const int numcomps = 1;
int i;
const double * values = helper->values;
long no_values = helper->no_values;
double reference_value = helper->reference_value;
double divisor = helper->divisor;
double decimal = helper->decimal;
int* data;
opj_cparameters_t parameters = {0,}; /* compression parameters */
opj_codec_t *codec = NULL;
opj_image_t *image = NULL;
opj_image_cmptparm_t cmptparm = {0,};
opj_stream_t *stream = NULL;
opj_memory_stream mstream;
/* set encoding parameters to default values */
opj_set_default_encoder_parameters(&parameters);
grib_context_log(c, GRIB_LOG_DEBUG, "grib_openjpeg_encode: OpenJPEG version %s", opj_version());
parameters.tcp_numlayers = 1;
parameters.cp_disto_alloc = 1;
parameters.numresolution = 1;
parameters.tcp_rates[0] = helper->compression;
/* initialize image component */
cmptparm.prec = helper->bits_per_value;
cmptparm.bpp = helper->bits_per_value; /* Not sure about this one and the previous. What is the difference? */
cmptparm.sgnd = 0;
cmptparm.dx = 1;
cmptparm.dy = 1;
cmptparm.w = helper->width;
cmptparm.h = helper->height;
/* create the image */
image = opj_image_create(numcomps, &cmptparm, OPJ_CLRSPC_GRAY);
if(!image) {
err = GRIB_ENCODING_ERROR;
goto cleanup;
}
image->x0 = 0;
image->y0 = 0;
image->x1 = helper->width;
image->y1 = helper->height;
Assert(cmptparm.prec <= sizeof(image->comps[0].data[0])*8 - 1); /* BR: -1 because I don't know what happens if the sign bit is set */
Assert(helper->no_values == image->comps[0].h * image->comps[0].w);
/* Simple packing */
data = image->comps[0].data;
for(i=0;i< no_values;i++){
unsigned long unsigned_val = (unsigned long)((((values[i]*decimal)-(reference_value))*divisor)+0.5);
data[i] = unsigned_val;
}
/* get a J2K compressor handle */
codec = opj_create_compress(OPJ_CODEC_J2K);
opj_set_info_handler(codec, openjpeg_info, c);
opj_set_warning_handler(codec, openjpeg_warning, c);
opj_set_error_handler(codec, openjpeg_error,c);
/* setup the encoder parameters using the current image and user parameters */
if (!opj_setup_encoder(codec, &parameters, image)) {
grib_context_log(c, GRIB_LOG_ERROR, "openjpeg: failed to setup encoder");
err = GRIB_ENCODING_ERROR;
goto cleanup;
}
/* open a byte stream for writing */
mstream.helper = helper;
mstream.pData = (OPJ_UINT8*) helper->jpeg_buffer;
mstream.offset = 0;
mstream.dataSize = helper->buffer_size;
stream = opj_stream_create_default_memory_stream( &mstream, OPJ_STREAM_WRITE);
if (stream == NULL) {
grib_context_log(c, GRIB_LOG_ERROR, "openjpeg: failed create default memory stream");
err = GRIB_ENCODING_ERROR;
goto cleanup;
}
if (!opj_start_compress(codec, image, stream)) {
grib_context_log(c, GRIB_LOG_ERROR, "openjpeg: failed to setup encoder");
err = GRIB_ENCODING_ERROR;
goto cleanup;
}
/* encode image */
if (!opj_encode(codec, stream)) {
grib_context_log(c, GRIB_LOG_ERROR, "openjpeg: opj_encode failed");
err = GRIB_ENCODING_ERROR;
goto cleanup;
}
if (!opj_end_compress(codec, stream)) {
grib_context_log(c, GRIB_LOG_ERROR, "openjpeg: opj_end_compress failed");
err = GRIB_ENCODING_ERROR;
goto cleanup;
}
helper->jpeg_length = mstream.offset;
cleanup:
if (codec) opj_destroy_codec(codec);
if (stream) opj_stream_destroy(stream);
if (image) opj_image_destroy(image);
return err;
}
int grib_openjpeg_decode(grib_context *c, unsigned char *buf, size_t *buflen, double *val, size_t *n_vals) {
grib_context_log(c, GRIB_LOG_ERROR,
"grib_accessor_data_jpeg2000_packing: OpenJPEG JPEG support not enabled.");
int err = GRIB_SUCCESS;
int i;
unsigned long mask;
int *data;
size_t count;
opj_dparameters_t parameters = {0,}; /* decompression parameters */
opj_stream_t *stream = NULL;
opj_memory_stream mstream;
opj_image_t *image = NULL;
opj_codec_t *codec = NULL;
opj_image_comp_t comp = {0,};
/* set decoding parameters to default values */
opj_set_default_decoder_parameters(&parameters);
parameters.decod_format = 1; /* JP2_FMT */
/* JPEG-2000 codestream */
grib_context_log(c, GRIB_LOG_DEBUG, "grib_openjpeg_decode: OpenJPEG version %s", opj_version());
/* get a decoder handle */
codec = opj_create_decompress(OPJ_CODEC_J2K);
/* catch events using our callbacks and give a local context */
opj_set_info_handler(codec, openjpeg_info, c);
opj_set_warning_handler(codec, openjpeg_warning, c);
opj_set_error_handler(codec, openjpeg_error,c);
/* initialize our memory stream */
mstream.pData = buf;
mstream.dataSize = *buflen;
mstream.offset = 0;
/* open a byte stream from memory stream */
stream = opj_stream_create_default_memory_stream( &mstream, OPJ_STREAM_READ);
/* setup the decoder decoding parameters using user parameters */
if (!opj_setup_decoder(codec, &parameters)) {
grib_context_log(c, GRIB_LOG_ERROR, "openjpeg: failed to setup decoder");
err = GRIB_DECODING_ERROR;
goto cleanup;
}
if (!opj_read_header(stream, codec, &image)) {
grib_context_log(c, GRIB_LOG_ERROR, "openjpeg: failed to read the header");
err = GRIB_DECODING_ERROR;
goto cleanup;
}
if (!opj_decode(codec, stream, image)) {
grib_context_log(c, GRIB_LOG_ERROR, "openjpeg: failed to decode");
err = GRIB_DECODING_ERROR;
goto cleanup;
}
if ( !(*n_vals <= image->comps[0].w * image->comps[0].h) ) {
err = GRIB_DECODING_ERROR;
goto cleanup;
}
if ( (image->numcomps != 1) || !(image->x1 * image->y1) ) {
err = GRIB_DECODING_ERROR;
goto cleanup;
}
Assert(image->comps[0].sgnd == 0);
Assert(comp.prec <= sizeof(image->comps[0].data[0])*8 - 1); /* BR: -1 because I don't know what happens if the sign bit is set */
Assert(image->comps[0].prec < sizeof(mask)*8-1);
data = image->comps[0].data;
mask = (1 << image->comps[0].prec) - 1;
count = image->comps[0].w * image->comps[0].h;
for(i = 0; i <count ; i++)
val[i] = data[i] & mask;
if (!opj_end_decompress(codec, stream)) {
grib_context_log(c, GRIB_LOG_ERROR, "openjpeg: failed in opj_end_decompress");
err = GRIB_DECODING_ERROR;
}
cleanup:
/* close the byte stream */
if (codec) opj_destroy_codec(codec);
if (stream) opj_stream_destroy(stream);
if (image) opj_image_destroy(image);
return err;
}
#endif /* OPENJPEG_VERSION */
#else /* No OpenJPEG */
int grib_openjpeg_decode(grib_context *c, unsigned char *buf, size_t *buflen, double *val, size_t *n_vals) {
grib_context_log(c, GRIB_LOG_ERROR, "grib_openjpeg_encoding.c: OpenJPEG JPEG support not enabled.");
return GRIB_NOT_IMPLEMENTED;
}
int grib_openjpeg_encode(grib_context *c, j2k_encode_helper *helper) {
grib_context_log(c, GRIB_LOG_ERROR,
"grib_accessor_data_jpeg2000_packing: OpenJPEG JPEG support not enabled.");
grib_context_log(c, GRIB_LOG_ERROR, "grib_openjpeg_encoding.c: OpenJPEG JPEG support not enabled.");
return GRIB_NOT_IMPLEMENTED;
}
#endif