313 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			313 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /* mj2_to_metadata.c */
 | |
| /* Dump MJ2, JP2 metadata (partial so far) to xml file */
 | |
| /* Contributed to Open JPEG by Glenn Pearson, contract software developer, U.S. National Library of Medicine.
 | |
| 
 | |
| The base code in this file was developed by the author as part of a video archiving
 | |
| project for the U.S. National Library of Medicine, Bethesda, MD. 
 | |
| It is the policy of NLM (and U.S. government) to not assert copyright.
 | |
| 
 | |
| A non-exclusive copy of this code has been contributed to the Open JPEG project.
 | |
| Except for copyright, inclusion of the code within Open JPEG for distribution and use
 | |
| can be bound by the Open JPEG open-source license and disclaimer, expressed elsewhere.
 | |
| */
 | |
| 
 | |
| #include "opj_includes.h"
 | |
| #include "mj2.h"
 | |
| 
 | |
| #include "mj2_to_metadata.h"
 | |
| #include <string.h>
 | |
| #include "opj_getopt.h"
 | |
| 
 | |
| /* -------------------------------------------------------------------------- */
 | |
| 
 | |
| /**
 | |
| sample error callback expecting a FILE* client object
 | |
| */
 | |
| void error_callback(const char *msg, void *client_data) {
 | |
| 	FILE *stream = (FILE*)client_data;
 | |
| 	fprintf(stream, "[ERROR] %s", msg);
 | |
| }
 | |
| /**
 | |
| sample warning callback expecting a FILE* client object
 | |
| */
 | |
| void warning_callback(const char *msg, void *client_data) {
 | |
| 	FILE *stream = (FILE*)client_data;
 | |
| 	fprintf(stream, "[WARNING] %s", msg);
 | |
| }
 | |
| /**
 | |
| sample debug callback expecting a FILE* client object
 | |
| */
 | |
| void info_callback(const char *msg, void *client_data) {
 | |
| 	FILE *stream = (FILE*)client_data;
 | |
| 	fprintf(stream, "[INFO] %s", msg);
 | |
| }
 | |
| 
 | |
| /* -------------------------------------------------------------------------- */
 | |
| 
 | |
| 
 | |
| 
 | |
| /* ------------- */
 | |
| 
 | |
| void help_display()
 | |
| {
 | |
|   /*             "1234567890123456789012345678901234567890123456789012345678901234567890123456789" */
 | |
|   fprintf(stdout,"                Help for the 'mj2_to_metadata' Program\n");
 | |
|   fprintf(stdout,"                ======================================\n");
 | |
|   fprintf(stdout,"The -h option displays this information on screen.\n\n");
 | |
|   
 | |
|   fprintf(stdout,"mj2_to_metadata generates an XML file from a Motion JPEG 2000 file.\n");
 | |
|   fprintf(stdout,"The generated XML shows the structural, but not (yet) curatorial,\n");
 | |
|   fprintf(stdout,"metadata from the movie header and from the JPEG 2000 image and tile\n");
 | |
|   fprintf(stdout,"headers of a sample frame.  Excluded: low-level packed-bits image data.\n\n");
 | |
| 
 | |
|   fprintf(stdout,"By Default\n");
 | |
|   fprintf(stdout,"----------\n");
 | |
|   fprintf(stdout,"The metadata includes the jp2 image and tile headers of the first frame.\n");
 | |
|   fprintf(stdout,"\n");
 | |
|   fprintf(stdout,"Metadata values are shown in 'raw' form (e.g., hexidecimal) as stored in the\n");
 | |
|   fprintf(stdout,"file, and, if apt, in a 'derived' form that is more quickly grasped.\n");
 | |
|   fprintf(stdout,"\n");
 | |
|   fprintf(stdout,"Notes explaining the XML are embedded as terse comments.  These include\n");
 | |
|   fprintf(stdout,"   meaning of non-obvious tag abbreviations;\n");
 | |
|   fprintf(stdout,"   range and precision of valid values;\n");
 | |
|   fprintf(stdout,"   interpretations of values, such as enumerations; and\n");
 | |
|   fprintf(stdout,"   current implementation limitations.\n");
 | |
|   fprintf(stdout,"\n");
 | |
|   fprintf(stdout,"The sample-size and chunk-offset tables, each with 1 row per frame, are not reported.\n");
 | |
|   fprintf(stdout,"\n");
 | |
|   fprintf(stdout,"The file is self-contained and no verification (e.g., against a DTD) is requested.\n");
 | |
|   fprintf(stdout,"\n");
 | |
|   fprintf(stdout,"Required Parameters (except with -h)\n");
 | |
|   fprintf(stdout,"------------------------------------\n");
 | |
|   fprintf(stdout,"[Caution: file strings that contain spaces should be wrapped with quotes.]\n");
 | |
|   fprintf(stdout,"-i input.mj2  : where 'input' is any source file name or path.\n");
 | |
|   fprintf(stdout,"                MJ2 files created with 'frames_to_mj2' are supported so far.\n");
 | |
|   fprintf(stdout,"                These are silent, single-track, 'MJ2 Simple Profile' videos.\n");
 | |
|   fprintf(stdout,"-o output.xml : where 'output' is any destination file name or path.\n");
 | |
|   fprintf(stdout,"\n");
 | |
|   fprintf(stdout,"Optional Parameters\n");
 | |
|   fprintf(stdout,"-------------------\n");
 | |
|   fprintf(stdout,"-h            : Display this help information.\n");
 | |
|   fprintf(stdout,"-n            : Suppress all mj2_to_metadata notes.\n");
 | |
|   fprintf(stdout,"-t            : Include sample-size and chunk-offset tables.\n");
 | |
|   fprintf(stdout,"-f n          : where n > 0.  Include jp2 header info for frame n [default=1].\n");
 | |
|   fprintf(stdout,"-f 0          : No jp2 header info.\n");
 | |
|   fprintf(stdout,"-r            : Suppress all 'raw' data for which a 'derived' form exists.\n");
 | |
|   fprintf(stdout,"-d            : Suppress all 'derived' data.\n");
 | |
|   fprintf(stdout,"                (If both -r and -d given, -r will be ignored.)\n");
 | |
|   fprintf(stdout,"-v string     : Verify against the DTD file located by the string.\n");
 | |
|   fprintf(stdout,"                Prepend quoted 'string' with either SYSTEM or PUBLIC keyword.\n");
 | |
|   fprintf(stdout,"                Thus, for the distributed DTD placed in the same directory as\n");
 | |
|   fprintf(stdout,"                the output file: -v \"SYSTEM mj2_to_metadata.dtd\"\n");
 | |
|   fprintf(stdout,"                \"PUBLIC\" is used with an access protocol (e.g., http:) + URL.\n");
 | |
|   /* More to come */
 | |
|   fprintf(stdout,"\n");
 | |
|   /*             "1234567890123456789012345678901234567890123456789012345678901234567890123456789" */
 | |
| }
 | |
| 
 | |
| /* ------------- */
 | |
| 
 | |
| int main(int argc, char *argv[]) {
 | |
| 
 | |
| 	opj_dinfo_t* dinfo; 
 | |
| 	opj_event_mgr_t event_mgr;		/* event manager */
 | |
| 
 | |
|   FILE *file, *xmlout;
 | |
| /*  char xmloutname[50]; */
 | |
|   opj_mj2_t *movie;
 | |
| 
 | |
|   char* infile = 0;
 | |
|   char* outfile = 0;
 | |
|   char* s, S1, S2, S3;
 | |
|   int len;
 | |
|   unsigned int sampleframe = 1; /* First frame */
 | |
|   char* stringDTD = NULL;
 | |
|   BOOL notes = TRUE;
 | |
|   BOOL sampletables = FALSE;
 | |
|   BOOL raw = TRUE;
 | |
|   BOOL derived = TRUE;
 | |
| 	mj2_dparameters_t parameters;
 | |
| 
 | |
|   while (TRUE) {
 | |
| 	/* ':' after letter means it takes an argument */
 | |
|     int c = getopt(argc, argv, "i:o:f:v:hntrd");
 | |
| 	/* FUTURE:  Reserve 'p' for pruning file (which will probably make -t redundant) */
 | |
|     if (c == -1)
 | |
|       break;
 | |
|     switch (c) {
 | |
|     case 'i':			/* IN file */
 | |
|       infile = optarg;
 | |
|       s = optarg;
 | |
|       while (*s) { s++; } /* Run to filename end */
 | |
|       s--;
 | |
|       S3 = *s;
 | |
|       s--;
 | |
|       S2 = *s;
 | |
|       s--;
 | |
|       S1 = *s;
 | |
|       
 | |
|       if ((S1 == 'm' && S2 == 'j' && S3 == '2')
 | |
|       || (S1 == 'M' && S2 == 'J' && S3 == '2')) {
 | |
|        break;
 | |
|       }
 | |
|       fprintf(stderr, "Input file name must have .mj2 extension, not .%c%c%c.\n", S1, S2, S3);
 | |
|       return 1;
 | |
| 
 | |
|       /* ----------------------------------------------------- */
 | |
|     case 'o':			/* OUT file */
 | |
|       outfile = optarg;
 | |
|       while (*outfile) { outfile++; } /* Run to filename end */
 | |
|       outfile--;
 | |
|       S3 = *outfile;
 | |
|       outfile--;
 | |
|       S2 = *outfile;
 | |
|       outfile--;
 | |
|       S1 = *outfile;
 | |
|       
 | |
|       outfile = optarg;
 | |
|       
 | |
|       if ((S1 == 'x' && S2 == 'm' && S3 == 'l')
 | |
| 	  || (S1 == 'X' && S2 == 'M' && S3 == 'L'))
 | |
|         break;
 | |
|     
 | |
|       fprintf(stderr,
 | |
| 	  "Output file name must have .xml extension, not .%c%c%c\n", S1, S2, S3);
 | |
| 	  return 1;
 | |
| 
 | |
|       /* ----------------------------------------------------- */
 | |
|     case 'f':			/* Choose sample frame.  0 = none */
 | |
|       sscanf(optarg, "%u", &sampleframe);
 | |
|       break;
 | |
| 
 | |
|       /* ----------------------------------------------------- */
 | |
|     case 'v':			/* Verification by DTD. */
 | |
|       stringDTD = optarg;
 | |
| 	  /* We will not insist upon last 3 chars being "dtd", since non-file
 | |
| 	  access protocol may be used. */
 | |
| 	  if(strchr(stringDTD,'"') != NULL) {
 | |
|         fprintf(stderr, "-D's string must not contain any embedded double-quote characters.\n");
 | |
| 	    return 1;
 | |
| 	  }
 | |
| 
 | |
|       if (strncmp(stringDTD,"PUBLIC ",7) == 0 || strncmp(stringDTD,"SYSTEM ",7) == 0)
 | |
|         break;
 | |
|     
 | |
|       fprintf(stderr, "-D's string must start with \"PUBLIC \" or \"SYSTEM \"\n");
 | |
| 	  return 1;
 | |
| 
 | |
|     /* ----------------------------------------------------- */
 | |
|     case 'n':			/* Suppress comments */
 | |
|       notes = FALSE;
 | |
|       break;
 | |
| 
 | |
|     /* ----------------------------------------------------- */
 | |
|     case 't':			/* Show sample size and chunk offset tables */
 | |
|       sampletables = TRUE;
 | |
|       break;
 | |
| 
 | |
|     /* ----------------------------------------------------- */
 | |
|     case 'h':			/* Display an help description */
 | |
|       help_display();
 | |
|       return 0;
 | |
| 
 | |
|     /* ----------------------------------------------------- */
 | |
|     case 'r':			/* Suppress raw data */
 | |
|       raw = FALSE;
 | |
|       break;
 | |
| 
 | |
|     /* ----------------------------------------------------- */
 | |
|     case 'd':			/* Suppress derived data */
 | |
|       derived = FALSE;
 | |
|       break;
 | |
| 
 | |
|    /* ----------------------------------------------------- */
 | |
|     default:
 | |
|       return 1;
 | |
|     } /* switch */
 | |
|   } /* while */
 | |
| 
 | |
|   if(!raw && !derived)
 | |
| 	  raw = TRUE; /* At least one of 'raw' and 'derived' must be true */
 | |
| 
 | |
|     /* Error messages */
 | |
|   /* -------------- */
 | |
|   if (!infile || !outfile) {
 | |
|     fprintf(stderr,"Correct usage: mj2_to_metadata -i mj2-file -o xml-file (plus options)\n");
 | |
|     return 1;
 | |
|   }
 | |
| 
 | |
| /* was:
 | |
|   if (argc != 3) {
 | |
|     printf("Bad syntax: Usage: MJ2_to_metadata inputfile.mj2 outputfile.xml\n"); 
 | |
|     printf("Example: MJ2_to_metadata foreman.mj2 foreman.xml\n");
 | |
|     return 1;
 | |
|   }
 | |
| */
 | |
|   len = strlen(infile);
 | |
|   if(infile[0] == ' ')
 | |
|   {
 | |
|     infile++; /* There may be a leading blank if user put space after -i */
 | |
|   }
 | |
|   
 | |
|   file = fopen(infile, "rb"); /* was: argv[1] */
 | |
|   
 | |
|   if (!file) {
 | |
|     fprintf(stderr, "Failed to open %s for reading.\n", infile); /* was: argv[1] */
 | |
|     return 1;
 | |
|   }
 | |
| 
 | |
|   len = strlen(outfile);
 | |
|   if(outfile[0] == ' ')
 | |
|   {
 | |
|     outfile++; /* There may be a leading blank if user put space after -o */
 | |
|   }
 | |
| 
 | |
|   // Checking output file
 | |
|   xmlout = fopen(outfile, "w"); /* was: argv[2] */
 | |
|   if (!xmlout) {
 | |
|     fprintf(stderr, "Failed to open %s for writing.\n", outfile); /* was: argv[2] */
 | |
|     return 1;
 | |
|   }
 | |
|   // Leave it open
 | |
| 
 | |
| 	/*
 | |
| 	configure the event callbacks (not required)
 | |
| 	setting of each callback is optionnal
 | |
| 	*/
 | |
| 	memset(&event_mgr, 0, sizeof(opj_event_mgr_t));
 | |
| 	event_mgr.error_handler = error_callback;
 | |
| 	event_mgr.warning_handler = warning_callback;
 | |
| 	event_mgr.info_handler = info_callback;
 | |
| 
 | |
| 	/* get a MJ2 decompressor handle */
 | |
| 	dinfo = mj2_create_decompress();
 | |
| 
 | |
| 	/* catch events using our callbacks and give a local context */
 | |
| 	opj_set_event_mgr((opj_common_ptr)dinfo, &event_mgr, stderr);		
 | |
| 
 | |
| 	/* setup the decoder decoding parameters using user parameters */
 | |
| 	movie = (opj_mj2_t*) dinfo->mj2_handle;
 | |
| 	mj2_setup_decoder(dinfo->mj2_handle, ¶meters);
 | |
| 
 | |
|   if (mj2_read_struct(file, movie)) // Creating the movie structure
 | |
|   {
 | |
|     fclose(xmlout);
 | |
|     return 1;
 | |
|   }
 | |
| 
 | |
|   xml_write_init(notes, sampletables, raw, derived);
 | |
|   xml_write_struct(file, xmlout, movie, sampleframe, stringDTD, &event_mgr);
 | |
|   fclose(xmlout);
 | |
| 
 | |
| 	fprintf(stderr,"Metadata correctly extracted to XML file \n");;	
 | |
| 
 | |
| 	/* free remaining structures */
 | |
| 	if(dinfo) {
 | |
| 		mj2_destroy_decompress((opj_mj2_t*)dinfo->mj2_handle);
 | |
| 	}
 | |
| 
 | |
|   return 0;
 | |
| }
 | |
| 
 | |
| 
 | 
