Patch to read GML geo metadata and other associated data for inclusion in openjpeg

This commit is contained in:
hw 2018-03-22 13:33:24 +01:00
parent 2c7eb4fed9
commit 53df2bf860
5 changed files with 265 additions and 16 deletions

View File

@ -10443,6 +10443,9 @@ opj_codestream_info_v2_t* j2k_get_cstr_info(opj_j2k_t* p_j2k)
return NULL; return NULL;
} }
cstr_info->nbasoc = 0;
cstr_info->asoc_info = 00;
cstr_info->nbcomps = p_j2k->m_private_image->numcomps; cstr_info->nbcomps = p_j2k->m_private_image->numcomps;
cstr_info->tx0 = p_j2k->m_cp.tx0; cstr_info->tx0 = p_j2k->m_cp.tx0;

View File

@ -111,6 +111,11 @@ static OPJ_BOOL opj_jp2_read_cdef(opj_jp2_t * jp2,
static void opj_jp2_apply_cdef(opj_image_t *image, opj_jp2_color_t *color, static void opj_jp2_apply_cdef(opj_image_t *image, opj_jp2_color_t *color,
opj_event_mgr_t *); opj_event_mgr_t *);
/**
* Destroy list of ASOC entities
*/
static void opj_jp2_asoc_destroy( opj_jp2_asoc_t *p_asoc, OPJ_UINT32 num );
/** /**
* Writes the Channel Definition box. * Writes the Channel Definition box.
* *
@ -161,6 +166,22 @@ static OPJ_BOOL opj_jp2_read_ftyp(opj_jp2_t *jp2,
OPJ_UINT32 p_header_size, OPJ_UINT32 p_header_size,
opj_event_mgr_t * p_manager); opj_event_mgr_t * p_manager);
/**
* Reads a ASOC box - Associated data box.
* Also reads contained label (LBL) and XML boxes
*
* @param p_header_data the data contained in the ASOC box.
* @param jp2 the jpeg2000 file codec.
* @param p_header_size the size of the data contained in the ASOC box.
* @param p_manager the user event manager.
*
* @return true if the ASOC box is valid.
*/
static OPJ_BOOL opj_jp2_read_asoc( opj_jp2_t *jp2,
OPJ_BYTE * p_header_data,
OPJ_UINT32 p_header_size,
opj_event_mgr_t * p_manager);
static OPJ_BOOL opj_jp2_skip_jp2c(opj_jp2_t *jp2, static OPJ_BOOL opj_jp2_skip_jp2c(opj_jp2_t *jp2,
opj_stream_private_t *stream, opj_stream_private_t *stream,
opj_event_mgr_t * p_manager); opj_event_mgr_t * p_manager);
@ -425,7 +446,8 @@ static const opj_jp2_header_handler_t * opj_jp2_find_handler(OPJ_UINT32 p_id);
static const opj_jp2_header_handler_t jp2_header [] = { static const opj_jp2_header_handler_t jp2_header [] = {
{JP2_JP, opj_jp2_read_jp}, {JP2_JP, opj_jp2_read_jp},
{JP2_FTYP, opj_jp2_read_ftyp}, {JP2_FTYP, opj_jp2_read_ftyp},
{JP2_JP2H, opj_jp2_read_jp2h} {JP2_JP2H, opj_jp2_read_jp2h},
{JP2_ASOC, opj_jp2_read_asoc}
}; };
static const opj_jp2_header_handler_t jp2_img_header [] = { static const opj_jp2_header_handler_t jp2_img_header [] = {
@ -2638,6 +2660,104 @@ static OPJ_BOOL opj_jp2_read_ftyp(opj_jp2_t *jp2,
return OPJ_TRUE; return OPJ_TRUE;
} }
static OPJ_BOOL opj_jp2_read_asoc( opj_jp2_t *jp2,
OPJ_BYTE * p_header_data,
OPJ_UINT32 p_header_size,
opj_event_mgr_t * p_manager)
{
OPJ_UINT32 label_tag;
OPJ_UINT32 asoc_tag;
OPJ_UINT32 asoc_size;
opj_jp2_asoc_t *asoc;
/* preconditions */
assert(jp2 != 00);
assert(p_header_data != 00);
assert(p_manager != 00);
opj_read_bytes(p_header_data,&asoc_size,4);
p_header_data += 4;
p_header_size -= 4;
opj_read_bytes(p_header_data,&label_tag,4);
p_header_data += 4;
p_header_size -= 4;
asoc_size -= 4;
if (label_tag != JP2_LBL) {
/* TODO: Verify that ASOC must have a following label ? */
opj_event_msg(p_manager, EVT_WARNING, "ASOC data does not have a label (LBL)\n");
return OPJ_TRUE; // No error if we could not parse
}
if ( jp2->numasoc == 0 ) {
jp2->numasoc = 1;
jp2->asoc = opj_malloc(sizeof(opj_jp2_asoc_t));
}
else {
(jp2->numasoc)++;
jp2->asoc = opj_realloc(jp2->asoc, jp2->numasoc * sizeof(opj_jp2_asoc_t));
}
asoc = &(jp2->asoc[jp2->numasoc-1]);
asoc->level = jp2->numasoc-1; /* TODO: This is not correct if a parent asoc contains multiple child asocs! */
asoc->label_length = asoc_size;
asoc->label = opj_malloc(asoc_size);
memcpy(asoc->label, p_header_data, asoc_size);
asoc->xml_buf = 00;
asoc->xml_len = 0;
p_header_data += asoc_size;
p_header_size -= asoc_size;
opj_read_bytes(p_header_data,&asoc_tag,4);
p_header_data += 4;
p_header_size -= 4;
switch (asoc_tag) {
case JP2_ASOC: {
/* Start of nested ASOC tags. Parse this level. */
if (!opj_jp2_read_asoc( jp2, p_header_data, p_header_size, p_manager )) {
return OPJ_FALSE;
}
}
break;
case JP2_XML: {
asoc->xml_len = p_header_size;
asoc->xml_buf = opj_malloc(p_header_size);
memcpy( asoc->xml_buf, p_header_data, p_header_size );
}
break;
default: {
/* Copy the unknown data for external handling.
NOTE: This is not tested, but does the same as if an XML tag was found.*/
asoc->xml_len = p_header_size;
asoc->xml_buf = opj_malloc(p_header_size);
memcpy( asoc->xml_buf, p_header_data, p_header_size );
}
}
return OPJ_TRUE;
}
static void opj_jp2_asoc_destroy( opj_jp2_asoc_t *p_asoc, OPJ_UINT32 num )
{
OPJ_UINT32 i;
opj_jp2_asoc_t *asoc;
for (i=0; i<num; i++)
{
asoc = &(p_asoc[i]);
opj_free( asoc->label );
asoc->label = 00;
asoc->label_length = 0;
opj_free( asoc->xml_buf );
asoc->xml_buf = 00;
asoc->xml_len = 0;
}
}
static OPJ_BOOL opj_jp2_skip_jp2c(opj_jp2_t *jp2, static OPJ_BOOL opj_jp2_skip_jp2c(opj_jp2_t *jp2,
opj_stream_private_t *stream, opj_stream_private_t *stream,
opj_event_mgr_t * p_manager) opj_event_mgr_t * p_manager)
@ -3070,6 +3190,12 @@ void opj_jp2_destroy(opj_jp2_t *jp2)
jp2->m_procedure_list = 00; jp2->m_procedure_list = 00;
} }
if ( jp2->numasoc ) {
opj_jp2_asoc_destroy( jp2->asoc, jp2->numasoc );
jp2->asoc = 00;
jp2->numasoc = 0;
}
opj_free(jp2); opj_free(jp2);
} }
} }
@ -3205,6 +3331,11 @@ opj_jp2_t* opj_jp2_create(OPJ_BOOL p_is_decoder)
opj_jp2_destroy(jp2); opj_jp2_destroy(jp2);
return 00; return 00;
} }
/* Association data */
jp2->asoc = 00;
jp2->numasoc = 0;
} }
return jp2; return jp2;
@ -3227,7 +3358,30 @@ opj_codestream_index_t* jp2_get_cstr_index(opj_jp2_t* p_jp2)
opj_codestream_info_v2_t* jp2_get_cstr_info(opj_jp2_t* p_jp2) opj_codestream_info_v2_t* jp2_get_cstr_info(opj_jp2_t* p_jp2)
{ {
return j2k_get_cstr_info(p_jp2->j2k); opj_codestream_info_v2_t* p_info = j2k_get_cstr_info(p_jp2->j2k);
jp2_copy_asoc_data( p_jp2, p_info );
return p_info;
}
OPJ_BOOL jp2_copy_asoc_data( opj_jp2_t* p_jp2, opj_codestream_info_v2_t* p_info )
{
OPJ_UINT32 i;
opj_jp2_asoc_t *asoc, *to_asoc;
p_info->nbasoc = p_jp2->numasoc;
p_info->asoc_info = opj_malloc(p_info->nbasoc * sizeof(opj_jp2_asoc_t));
for (i=0; i<p_info->nbasoc; i++) {
asoc = &(p_jp2->asoc[i]);
to_asoc = &(p_info->asoc_info[i]);
to_asoc->level = asoc->level;
to_asoc->label_length = asoc->label_length;
to_asoc->xml_len = asoc->xml_len;
to_asoc->label = opj_malloc( to_asoc->label_length );
memcpy(to_asoc->label, asoc->label, to_asoc->label_length);
to_asoc->xml_buf = opj_malloc( to_asoc->xml_len);
memcpy(to_asoc->xml_buf, asoc->xml_buf, to_asoc->xml_len);
}
return OPJ_TRUE;
} }
OPJ_BOOL opj_jp2_set_decoded_resolution_factor(opj_jp2_t *p_jp2, OPJ_BOOL opj_jp2_set_decoded_resolution_factor(opj_jp2_t *p_jp2,

View File

@ -59,6 +59,9 @@
#define JP2_DTBL 0x6474626c /**< Data Reference box */ #define JP2_DTBL 0x6474626c /**< Data Reference box */
#define JP2_BPCC 0x62706363 /**< Bits per component box */ #define JP2_BPCC 0x62706363 /**< Bits per component box */
#define JP2_JP2 0x6a703220 /**< File type fields */ #define JP2_JP2 0x6a703220 /**< File type fields */
#define JP2_ASOC 0x61736f63 /**< Associated data */
#define JP2_LBL 0x6c626c20 /**< Association label */
#define JP2_XML 0x786d6c20 /**< XML data */
/* For the future */ /* For the future */
/* #define JP2_RES 0x72657320 */ /**< Resolution box (super-box) */ /* #define JP2_RES 0x72657320 */ /**< Resolution box (super-box) */
@ -186,6 +189,9 @@ typedef struct opj_jp2 {
opj_jp2_color_t color; opj_jp2_color_t color;
opj_jp2_asoc_t *asoc;
OPJ_UINT32 numasoc;
OPJ_BOOL ignore_pclr_cmap_cdef; OPJ_BOOL ignore_pclr_cmap_cdef;
OPJ_BYTE has_jp2h; OPJ_BYTE has_jp2h;
OPJ_BYTE has_ihdr; OPJ_BYTE has_ihdr;
@ -480,6 +486,11 @@ void jp2_dump(opj_jp2_t* p_jp2, OPJ_INT32 flag, FILE* out_stream);
*/ */
opj_codestream_info_v2_t* jp2_get_cstr_info(opj_jp2_t* p_jp2); opj_codestream_info_v2_t* jp2_get_cstr_info(opj_jp2_t* p_jp2);
/**
* Copy associated data
*/
OPJ_BOOL jp2_copy_asoc_data( opj_jp2_t* p_jp2, opj_codestream_info_v2_t* p_info );
/** /**
* Get the codestream index from a JPEG2000 codec. * Get the codestream index from a JPEG2000 codec.
* *

View File

@ -37,6 +37,7 @@
#include "opj_includes.h" #include "opj_includes.h"
static void opj_asoc_destroy( opj_jp2_asoc_t *p_asoc, OPJ_UINT32 num );
/* ---------------------------------------------------------------------- */ /* ---------------------------------------------------------------------- */
/* Functions to set the message handlers */ /* Functions to set the message handlers */
@ -952,6 +953,26 @@ void OPJ_CALLCONV opj_dump_codec(opj_codec_t *p_codec,
return; return;
} }
void OPJ_CALLCONV opj_dump_associated_data(
opj_codestream_info_v2_t* cstr_info,
FILE* output_stream )
{
OPJ_UINT32 i;
if ( cstr_info->asoc_info ) {
fprintf(output_stream, "\n\nAssociated data: {\n");
for (i=0; i<cstr_info->nbasoc; i++) {
fprintf(output_stream, "\tlabel=%s, xml/data=", (char*) cstr_info->asoc_info[i].label);
if (cstr_info->asoc_info[i].xml_buf) {
fprintf(output_stream, "%s\n", (char*) cstr_info->asoc_info[i].xml_buf);
}
else {
fprintf(output_stream, "NULL\n");
}
}
fprintf(output_stream, "}\n");
}
}
opj_codestream_info_v2_t* OPJ_CALLCONV opj_get_cstr_info(opj_codec_t *p_codec) opj_codestream_info_v2_t* OPJ_CALLCONV opj_get_cstr_info(opj_codec_t *p_codec)
{ {
if (p_codec) { if (p_codec) {
@ -963,6 +984,23 @@ opj_codestream_info_v2_t* OPJ_CALLCONV opj_get_cstr_info(opj_codec_t *p_codec)
return NULL; return NULL;
} }
void opj_asoc_destroy( opj_jp2_asoc_t *p_asoc, OPJ_UINT32 num )
{
OPJ_UINT32 i;
opj_jp2_asoc_t *asoc;
for (i=0; i<num; i++)
{
asoc = &(p_asoc[i]);
opj_free( asoc->label );
asoc->label = 00;
asoc->label_length = 0;
opj_free( asoc->xml_buf );
asoc->xml_buf = 00;
asoc->xml_len = 0;
}
}
void OPJ_CALLCONV opj_destroy_cstr_info(opj_codestream_info_v2_t **cstr_info) void OPJ_CALLCONV opj_destroy_cstr_info(opj_codestream_info_v2_t **cstr_info)
{ {
if (cstr_info) { if (cstr_info) {
@ -975,6 +1013,12 @@ void OPJ_CALLCONV opj_destroy_cstr_info(opj_codestream_info_v2_t **cstr_info)
/* FIXME not used for the moment*/ /* FIXME not used for the moment*/
} }
if ((*cstr_info)->nbasoc) {
opj_asoc_destroy((*cstr_info)->asoc_info, (*cstr_info)->nbasoc);
(*cstr_info)->asoc_info = 00;
(*cstr_info)->nbasoc = 0;
}
opj_free((*cstr_info)); opj_free((*cstr_info));
(*cstr_info) = NULL; (*cstr_info) = NULL;
} }
@ -1050,6 +1094,11 @@ opj_stream_t* OPJ_CALLCONV opj_stream_create_file_stream(
return l_stream; return l_stream;
} }
OPJ_OFF_T opj_stream_skip_api(opj_stream_t * p_stream, OPJ_OFF_T p_size)
{
opj_stream_private_t* l_stream = (opj_stream_private_t*) p_stream;
return l_stream->m_opj_skip(l_stream, p_size, NULL);
}
void* OPJ_CALLCONV opj_image_data_alloc(OPJ_SIZE_T size) void* OPJ_CALLCONV opj_image_data_alloc(OPJ_SIZE_T size)
{ {

View File

@ -916,6 +916,22 @@ typedef struct opj_tile_v2_info {
} opj_tile_info_v2_t; } opj_tile_info_v2_t;
/**
Collector for association box (ASOC data), defined by a level, label and optionally XML data.
E.g. georeferencing with GML uses this (http://docs.opengeospatial.org/is/08-085r4/08-085r4.html).
In this case the first asoc is labelled 'gml.data' and has no XML data. The second asoc is named
'gml.root-instance' and contains XML formatted geo-information.
*/
typedef struct opj_jp2_asoc
{
OPJ_UINT32 level;
OPJ_BYTE *label;
OPJ_UINT32 label_length;
OPJ_BYTE *xml_buf;
OPJ_UINT32 xml_len;
} opj_jp2_asoc_t;
/** /**
* Information structure about the codestream (FIXME should be expand and enhance) * Information structure about the codestream (FIXME should be expand and enhance)
*/ */
@ -943,6 +959,12 @@ typedef struct opj_codestream_info_v2 {
/** information regarding tiles inside image */ /** information regarding tiles inside image */
opj_tile_info_v2_t *tile_info; /* FIXME not used for the moment */ opj_tile_info_v2_t *tile_info; /* FIXME not used for the moment */
/** Number of associated data boxes*/
OPJ_UINT32 nbasoc;
/** Associated data, e.g. GML geoinformation */
opj_jp2_asoc_t *asoc_info;
} opj_codestream_info_v2_t; } opj_codestream_info_v2_t;
@ -1178,14 +1200,23 @@ OPJ_API void OPJ_CALLCONV opj_stream_set_skip_function(opj_stream_t* p_stream,
opj_stream_skip_fn p_function); opj_stream_skip_fn p_function);
/** /**
* Sets the given function to be used as a seek function, the stream is then seekable, * Sets the given function to be used as a seek function, the stream is then seekable.
* using SEEK_SET behavior.
* @param p_stream the stream to modify * @param p_stream the stream to modify
* @param p_function the function to use a skip function. * @param p_function the function to use a skip function.
*/ */
OPJ_API void OPJ_CALLCONV opj_stream_set_seek_function(opj_stream_t* p_stream, OPJ_API void OPJ_CALLCONV opj_stream_set_seek_function(opj_stream_t* p_stream,
opj_stream_seek_fn p_function); opj_stream_seek_fn p_function);
/**
* Skips a number of bytes from the stream.
* @param p_stream the stream to skip data from.
* @param p_size the number of bytes to skip.
* @param p_event_mgr the user event manager to be notified of special events.
* @return the number of bytes skipped, or -1 if an error occurred.
*/
OPJ_API OPJ_OFF_T OPJ_CALLCONV opj_stream_skip_api(opj_stream_t * p_stream, OPJ_OFF_T p_size);
/** /**
* Sets the given data to be used as a user data for the stream. * Sets the given data to be used as a user data for the stream.
* @param p_stream the stream to modify * @param p_stream the stream to modify
@ -1314,9 +1345,6 @@ OPJ_API OPJ_BOOL OPJ_CALLCONV opj_setup_decoder(opj_codec_t *p_codec,
* number, or "ALL_CPUS". If OPJ_NUM_THREADS is set and this function is called, * number, or "ALL_CPUS". If OPJ_NUM_THREADS is set and this function is called,
* this function will override the behaviour of the environment variable. * this function will override the behaviour of the environment variable.
* *
* Currently this function must be called after opj_setup_decoder() and
* before opj_read_header().
*
* Note: currently only has effect on the decompressor. * Note: currently only has effect on the decompressor.
* *
* @param p_codec decompressor handler * @param p_codec decompressor handler
@ -1611,6 +1639,10 @@ OPJ_API void OPJ_CALLCONV opj_dump_codec(opj_codec_t *p_codec,
OPJ_API opj_codestream_info_v2_t* OPJ_CALLCONV opj_get_cstr_info( OPJ_API opj_codestream_info_v2_t* OPJ_CALLCONV opj_get_cstr_info(
opj_codec_t *p_codec); opj_codec_t *p_codec);
OPJ_API void OPJ_CALLCONV opj_dump_associated_data(
opj_codestream_info_v2_t* cstr_info,
FILE* output_stream );
/** /**
* Get the codestream index from the codec * Get the codestream index from the codec
* *