NCEPLIBS-g2c 2.0.0
Loading...
Searching...
No Matches
decenc_openjpeg.c
Go to the documentation of this file.
1
6/* Copyright 2005-2019 ECMWF.
7 *
8 * This software is licensed under the terms of the Apache Licence
9 * Version 2.0 which can be obtained at
10 * http://www.apache.org/licenses/LICENSE-2.0.
11 *
12 * In applying this licence, ECMWF does not waive the privileges and
13 * immunities granted to it by virtue of its status as an
14 * intergovernmental organisation nor does it submit to any
15 * jurisdiction.
16 */
17
18#if !defined USE_JPEG2000 && defined USE_OPENJPEG
19
20#include "grib2_int.h"
21#include "openjpeg.h"
22
23#include <assert.h>
24#include <stdio.h>
25#include <stdlib.h>
26#include <string.h>
27
36static void
37openjpeg_warning(const char *msg, void *client_data)
38{
39 (void)client_data;
40 fprintf(stderr, "openjpeg: %s", msg);
41}
42
51static void
52openjpeg_error(const char *msg, void *client_data)
53{
54 (void)client_data;
55 fprintf(stderr, "openjpeg: %s", msg);
56}
57
66static void
67openjpeg_info(const char *msg, void *client_data)
68{
69 (void)msg;
70 (void)client_data;
71 /* fprintf(stderr,"openjpeg: %s",msg); */
72}
73
74/* opj_* Helper code from
75 * https://groups.google.com/forum/#!topic/openjpeg/8cebr0u7JgY
76 */
77/* These routines are added to use memory instead of a file for input and output */
78/* struct need to treat memory as a stream */
79typedef struct
80{
81 OPJ_UINT8 *pData; /* our data */
82 OPJ_SIZE_T dataSize; /* how big is our data */
83 OPJ_SIZE_T offset; /* where we are currently in our data */
84} opj_memory_stream;
85
97static OPJ_SIZE_T
98opj_memory_stream_read(void *buffer, OPJ_SIZE_T nb_bytes, void *p_user_data)
99{
100 opj_memory_stream *mstream = (opj_memory_stream *)p_user_data; /* Our data */
101 OPJ_SIZE_T nb_bytes_read = nb_bytes; /* Amount to move to buffer */
102
103 /* Check if the current offset is outside our data buffer */
104 if (mstream->offset >= mstream->dataSize)
105 return (OPJ_SIZE_T)-1;
106
107 /* Check if we are reading more than we have */
108 if (nb_bytes > (mstream->dataSize - mstream->offset))
109 nb_bytes_read = mstream->dataSize - mstream->offset;
110
111 memcpy(buffer, &(mstream->pData[mstream->offset]), nb_bytes_read);
112 mstream->offset += nb_bytes_read; /* Update the pointer to the new location */
113 return nb_bytes_read;
114}
115
127static OPJ_SIZE_T
128opj_memory_stream_write(void *buffer, OPJ_SIZE_T nb_bytes, void *user_data)
129{
130 opj_memory_stream *mstream = (opj_memory_stream *)user_data; /* our data */
131 OPJ_SIZE_T nb_bytes_write = nb_bytes; /* Amount to move to buffer */
132
133 /* Check if the current offset is outside our data buffer */
134 if (mstream->offset >= mstream->dataSize)
135 return (OPJ_SIZE_T)-1;
136
137 /* Check if we are writing more than we have space for */
138 if (nb_bytes > (mstream->dataSize - mstream->offset))
139 nb_bytes_write = mstream->dataSize - mstream->offset;
140
141 /* Copy the data from the internal buffer */
142 memcpy(&(mstream->pData[mstream->offset]), buffer, nb_bytes_write);
143 mstream->offset += nb_bytes_write; /* Update the pointer to the new location */
144 return nb_bytes_write;
145}
146
157static OPJ_OFF_T
158opj_memory_stream_skip(OPJ_OFF_T nb_bytes, void *user_data)
159{
160 opj_memory_stream *mstream = (opj_memory_stream *)user_data;
161 OPJ_SIZE_T l_nb_bytes;
162
163 if (nb_bytes < 0)
164 return -1; /* No skipping backwards */
165 l_nb_bytes = (OPJ_SIZE_T)nb_bytes; /* Allowed because it is positive */
166 /* Do not allow jumping past the end */
167 if (l_nb_bytes > mstream->dataSize - mstream->offset)
168 l_nb_bytes = mstream->dataSize - mstream->offset;
169 mstream->offset += l_nb_bytes;
170 return (OPJ_OFF_T)l_nb_bytes; /* Return how far we jumped */
171}
172
183static OPJ_BOOL
184opj_memory_stream_seek(OPJ_OFF_T nb_bytes, void *user_data)
185{
186 opj_memory_stream *mstream = (opj_memory_stream *)user_data;
187
188 if (nb_bytes < 0)
189 return OPJ_FALSE; /* Not before the buffer */
190 if (nb_bytes > (OPJ_OFF_T)mstream->dataSize)
191 return OPJ_FALSE; /* Not after the buffer */
192 mstream->offset = (OPJ_SIZE_T)nb_bytes; /* Move to new position */
193 return OPJ_TRUE;
194}
195
203static void
204opj_memory_stream_do_nothing(void *p_user_data)
205{
206 OPJ_ARG_NOT_USED(p_user_data);
207}
208
219static opj_stream_t *
220opj_stream_create_default_memory_stream(opj_memory_stream *memoryStream, OPJ_BOOL is_read_stream)
221{
222 opj_stream_t *stream;
223
224 if (!(stream = opj_stream_default_create(is_read_stream)))
225 return (NULL);
226 /* Set how to work with the frame buffer */
227 if (is_read_stream)
228 opj_stream_set_read_function(stream, opj_memory_stream_read);
229 else
230 opj_stream_set_write_function(stream, opj_memory_stream_write);
231
232 opj_stream_set_seek_function(stream, opj_memory_stream_seek);
233 opj_stream_set_skip_function(stream, opj_memory_stream_skip);
234 opj_stream_set_user_data(stream, memoryStream, opj_memory_stream_do_nothing);
235 opj_stream_set_user_data_length(stream, memoryStream->dataSize);
236 return stream;
237}
238
260int
261dec_jpeg2000(char *injpc, g2int bufsize, g2int *outfld)
262{
263 int iret = 0;
264 OPJ_INT32 mask;
265
266 opj_stream_t *stream = NULL;
267 opj_image_t *image = NULL;
268 opj_codec_t *codec = NULL;
269
270 /* set decoding parameters to default values */
271 opj_dparameters_t parameters = {
272 0,
273 }; /* decompression parameters */
274 opj_set_default_decoder_parameters(&parameters);
275 parameters.decod_format = 1; /* JP2_FMT */
276
277 /* get a decoder handle */
278 codec = opj_create_decompress(OPJ_CODEC_J2K);
279
280 /* catch events using our callbacks */
281 opj_set_info_handler(codec, openjpeg_info, NULL);
282 opj_set_warning_handler(codec, openjpeg_warning, NULL);
283 opj_set_error_handler(codec, openjpeg_error, NULL);
284
285 /* initialize our memory stream */
286 opj_memory_stream mstream;
287 mstream.pData = (OPJ_UINT8 *)injpc;
288 mstream.dataSize = (OPJ_SIZE_T)bufsize;
289 mstream.offset = 0;
290 /* open a byte stream from memory stream */
291 stream = opj_stream_create_default_memory_stream(&mstream, OPJ_STREAM_READ);
292
293 /* setup the decoder decoding parameters using user parameters */
294 if (!opj_setup_decoder(codec, &parameters))
295 {
296 fprintf(stderr, "openjpeg: failed to setup decoder");
297 iret = -3;
298 goto cleanup;
299 }
300 if (!opj_read_header(stream, codec, &image))
301 {
302 fprintf(stderr, "openjpeg: failed to read the header");
303 iret = -3;
304 goto cleanup;
305 }
306 if (!opj_decode(codec, stream, image))
307 {
308 fprintf(stderr, "openjpeg: failed to decode");
309 iret = -3;
310 goto cleanup;
311 }
312
313 if ((image->numcomps != 1) || (image->x1 * image->y1) == 0)
314 {
315 iret = -3;
316 goto cleanup;
317 }
318
319 assert(image->comps[0].sgnd == 0);
320 assert(image->comps[0].prec < sizeof(mask) * 8 - 1);
321
322 mask = (1 << image->comps[0].prec) - 1;
323
324 for (unsigned int i = 0; i < image->comps[0].w * image->comps[0].h; i++)
325 outfld[i] = (g2int)(image->comps[0].data[i] & mask);
326
327 if (!opj_end_decompress(codec, stream))
328 {
329 fprintf(stderr, "openjpeg: failed in opj_end_decompress");
330 iret = -3;
331 }
332
333cleanup:
334 /* close the byte stream */
335 if (codec)
336 opj_destroy_codec(codec);
337 if (stream)
338 opj_stream_destroy(stream);
339 if (image)
340 opj_image_destroy(image);
341
342 return iret;
343}
344
378int
379enc_jpeg2000(unsigned char *cin, g2int width, g2int height, g2int nbits,
380 g2int ltype, g2int ratio, g2int retry, char *outjpc,
381 g2int jpclen)
382{
383 (void)retry;
384 int iret = 0;
385 int nbytes = 0;
386 const int numcomps = 1;
387 g2int *ifld = NULL;
388
389 opj_codec_t *codec = NULL;
390 opj_image_t *image = NULL;
391 opj_stream_t *stream = NULL;
392
393 /* set encoding parameters to default values */
394 opj_cparameters_t parameters = {
395 0,
396 }; /* compression parameters */
397 opj_set_default_encoder_parameters(&parameters);
398
399 parameters.tcp_numlayers = 1;
400 parameters.cp_disto_alloc = 1;
401 if (ltype == 1)
402 {
403 assert(ratio != 255);
404 parameters.tcp_rates[0] = 1.0f / (float)ratio;
405 }
406
407 /* By default numresolution = 6 (must be between 1 and 32)
408 * This may be too large for some of our datasets, eg. 1xn, so adjust ...
409 */
410 parameters.numresolution = 6;
411 while ((width < (1 << (parameters.numresolution - 1))) ||
412 (height < (1 << (parameters.numresolution - 1))))
413 {
414 parameters.numresolution--;
415 }
416
417 /* initialize image component */
418 opj_image_cmptparm_t cmptparm = {
419 0,
420 };
421 cmptparm.prec = (OPJ_UINT32)nbits;
422 cmptparm.bpp = (OPJ_UINT32)nbits;
423 cmptparm.sgnd = 0;
424 cmptparm.dx = 1;
425 cmptparm.dy = 1;
426 cmptparm.w = (OPJ_UINT32)width;
427 cmptparm.h = (OPJ_UINT32)height;
428
429 /* create the image */
430 image = opj_image_create(numcomps, &cmptparm, OPJ_CLRSPC_GRAY);
431 if (!image)
432 {
433 iret = -3;
434 goto cleanup;
435 }
436 image->x0 = 0;
437 image->y0 = 0;
438 image->x1 = (OPJ_UINT32)width;
439 image->y1 = (OPJ_UINT32)height;
440
441 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 */
442
443 ifld = malloc(width * height * sizeof(g2int));
444 nbytes = (nbits + 7) / 8;
445 gbits(cin, ifld, 0, nbytes * 8, 0, width * height);
446 /* Simple packing */
447 for (int i = 0; i < width * height; i++)
448 {
449 image->comps[0].data[i] = ifld[i];
450 }
451 free(ifld);
452
453 /* get a J2K compressor handle */
454 codec = opj_create_compress(OPJ_CODEC_J2K);
455
456 opj_set_info_handler(codec, openjpeg_info, NULL);
457 opj_set_warning_handler(codec, openjpeg_warning, NULL);
458 opj_set_error_handler(codec, openjpeg_error, NULL);
459
460 /* setup the encoder parameters using the current image and user parameters */
461 if (!opj_setup_encoder(codec, &parameters, image))
462 {
463 fprintf(stderr, "openjpeg: failed to setup encoder");
464 iret = -3;
465 goto cleanup;
466 }
467
468 /* open a byte stream for writing */
469 opj_memory_stream mstream;
470 mstream.pData = (OPJ_UINT8 *)outjpc;
471 mstream.offset = 0;
472 mstream.dataSize = (OPJ_SIZE_T)jpclen;
473 stream = opj_stream_create_default_memory_stream(&mstream, OPJ_STREAM_WRITE);
474 if (!stream)
475 {
476 fprintf(stderr, "openjpeg: failed create default memory stream");
477 iret = -3;
478 goto cleanup;
479 }
480
481 if (!opj_start_compress(codec, image, stream))
482 {
483 fprintf(stderr, "openjpeg: failed to setup encoder");
484 iret = -3;
485 goto cleanup;
486 }
487
488 /* encode image */
489 if (!opj_encode(codec, stream))
490 {
491 fprintf(stderr, "openjpeg: opj_encode failed");
492 iret = -3;
493 goto cleanup;
494 }
495
496 if (!opj_end_compress(codec, stream))
497 {
498 fprintf(stderr, "openjpeg: opj_end_compress failed");
499 iret = -3;
500 goto cleanup;
501 }
502 iret = (int)mstream.offset;
503
504cleanup:
505 if (codec)
506 opj_destroy_codec(codec);
507 if (stream)
508 opj_stream_destroy(stream);
509 if (image)
510 opj_image_destroy(image);
511
512 return iret;
513}
514
515#endif
int dec_jpeg2000(char *injpc, g2int bufsize, g2int *outfld)
Decode a JPEG2000 code stream specified in the JPEG2000 Part-1 standard (i.e., ISO/IEC 15444-1) using...
int enc_jpeg2000(unsigned char *cin, g2int width, g2int height, g2int nbits, g2int ltype, g2int ratio, g2int retry, char *outjpc, g2int jpclen)
Encode a grayscale image into a JPEG2000 code stream specified in the JPEG2000 Part-1 standard (i....
void gbits(unsigned char *in, g2int *iout, g2int iskip, g2int nbits, g2int nskip, g2int n)
Unpack arbitrary size values from a packed bit string, right justifying each value in the unpacked io...
Definition gbits.c:57
int64_t g2int
Long integer type.
Definition grib2.h:32
Header file with internal function prototypes NCEPLIBS-g2c library.