mirror of https://github.com/ecmwf/eccodes.git
ECC-303: Cannot build with OpenJPEG version 2
This commit is contained in:
parent
7dccccb936
commit
fd1b31edec
|
@ -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(¶meters);
|
||||
|
||||
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(¶meters);
|
||||
|
||||
/* 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(¶meters);
|
||||
|
||||
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, ¶meters, 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(¶meters);
|
||||
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, ¶meters)) {
|
||||
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
|
||||
|
||||
|
|
Loading…
Reference in New Issue