Fixed _strnicmp bug
[openjpeg.git] / mj2 / mj2_to_frames.c
1 /*
2 * Copyright (c) 2003-2004, Fran�ois-Olivier Devaux
3 * Copyright (c) 2002-2004,  Communications and remote sensing Laboratory, Universite catholique de Louvain, Belgium
4 * All rights reserved.
5 *
6 * Redistribution and use in source and binary forms, with or without
7 * modification, are permitted provided that the following conditions
8 * are met:
9 * 1. Redistributions of source code must retain the above copyright
10 *    notice, this list of conditions and the following disclaimer.
11 * 2. Redistributions in binary form must reproduce the above copyright
12 *    notice, this list of conditions and the following disclaimer in the
13 *    documentation and/or other materials provided with the distribution.
14 *
15 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS `AS IS'
16 * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
17 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
18 * ARE DISCLAIMED.  IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
19 * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
20 * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
21 * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
22 * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
23 * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
24 * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
25 * POSSIBILITY OF SUCH DAMAGE.
26 */
27
28 #include "opj_includes.h"
29 #include "mj2.h"
30 #include "mj2_convert.h"
31
32 /* -------------------------------------------------------------------------- */
33
34 /**
35 sample error callback expecting a FILE* client object
36 */
37 void error_callback(const char *msg, void *client_data) {
38         FILE *stream = (FILE*)client_data;
39         fprintf(stream, "[ERROR] %s", msg);
40 }
41 /**
42 sample warning callback expecting a FILE* client object
43 */
44 void warning_callback(const char *msg, void *client_data) {
45         FILE *stream = (FILE*)client_data;
46         fprintf(stream, "[WARNING] %s", msg);
47 }
48 /**
49 sample debug callback expecting a FILE* client object
50 */
51 void info_callback(const char *msg, void *client_data) {
52         FILE *stream = (FILE*)client_data;
53         fprintf(stream, "[INFO] %s", msg);
54 }
55
56 /* -------------------------------------------------------------------------- */
57
58
59 int main(int argc, char *argv[]) {
60         mj2_dparameters_t mj2_parameters;                       /* decompression parameters */
61         opj_dinfo_t* dinfo; 
62         opj_event_mgr_t event_mgr;              /* event manager */     
63         opj_cio_t *cio = NULL;
64   unsigned int tnum, snum;
65   opj_mj2_t *movie;
66   mj2_tk_t *track;
67   mj2_sample_t *sample;
68   unsigned char* frame_codestream;
69   FILE *file, *outfile;
70   char outfilename[50];
71   opj_image_t *img = NULL;
72         unsigned int max_codstrm_size = 0;
73         double total_time = 0;
74         unsigned int numframes = 0;
75                         
76   if (argc != 3) {
77     printf("Bad syntax: Usage: mj2_to_frames inputfile.mj2 outputfile.yuv\n"); 
78     printf("Example: MJ2_decoder foreman.mj2 foreman.yuv\n");
79     return 1;
80   }
81   
82   file = fopen(argv[1], "rb");
83   
84   if (!file) {
85     fprintf(stderr, "failed to open %s for reading\n", argv[1]);
86     return 1;
87   }
88         
89   // Checking output file
90   outfile = fopen(argv[2], "w");
91   if (!file) {
92     fprintf(stderr, "failed to open %s for writing\n", argv[2]);
93     return 1;
94   }
95   fclose(outfile);
96         
97         /*
98         configure the event callbacks (not required)
99         setting of each callback is optionnal
100         */
101         memset(&event_mgr, 0, sizeof(opj_event_mgr_t));
102         event_mgr.error_handler = error_callback;
103         event_mgr.warning_handler = warning_callback;
104         event_mgr.info_handler = NULL;
105         
106         /* get a MJ2 decompressor handle */
107         dinfo = mj2_create_decompress();
108         movie = dinfo->mj2_handle;
109         
110         /* catch events using our callbacks and give a local context */
111         opj_set_event_mgr((opj_common_ptr)dinfo, &event_mgr, stderr);           
112
113         /* set J2K decoding parameters to default values */
114         opj_set_default_decoder_parameters(&mj2_parameters.j2k_parameters);
115         
116         /* setup the decoder decoding parameters using user parameters */
117         mj2_setup_decoder(dinfo->mj2_handle, &mj2_parameters);
118                         
119   if (mj2_read_struct(file, movie)) // Creating the movie structure
120     return 1;   
121         
122   // Decode first video track 
123         for (tnum=0; tnum < (unsigned int)(movie->num_htk + movie->num_stk + movie->num_vtk); tnum++) {
124                 if (movie->tk[tnum].track_type == 0) 
125                         break;
126         }
127         
128         if (movie->tk[tnum].track_type != 0) {
129                 printf("Error. Movie does not contain any video track\n");
130                 return 1;
131         }
132         
133   track = &movie->tk[tnum];
134         
135   // Output info on first video tracl
136   fprintf(stdout,"The first video track contains %d frames.\nWidth: %d, Height: %d \n\n",
137     track->num_samples, track->w, track->h);
138         
139         max_codstrm_size = track->sample[0].sample_size-8;
140         frame_codestream = (unsigned char*) malloc(max_codstrm_size * sizeof(unsigned char)); 
141
142         numframes = track->num_samples;
143         
144   for (snum=0; snum < numframes; snum++)
145   {
146                 double init_time = opj_clock();
147                 double elapsed_time;
148     sample = &track->sample[snum];
149                 if (sample->sample_size-8 > max_codstrm_size) {
150                         max_codstrm_size =  sample->sample_size-8;
151                         if ((frame_codestream = realloc(frame_codestream, max_codstrm_size)) == NULL) {
152                                 printf("Error reallocation memory\n");
153                                 return 1;
154                         };              
155                 }
156     fseek(file,sample->offset+8,SEEK_SET);
157     fread(frame_codestream, sample->sample_size-8, 1, file);  // Assuming that jp and ftyp markers size do
158                 
159                 /* open a byte stream */
160                 cio = opj_cio_open((opj_common_ptr)dinfo, frame_codestream, sample->sample_size-8);
161                 
162                 img = opj_decode(dinfo, cio); // Decode J2K to image
163                                 
164     if (((img->numcomps == 3) && (img->comps[0].dx == img->comps[1].dx / 2) 
165       && (img->comps[0].dx == img->comps[2].dx / 2 ) && (img->comps[0].dx == 1)) 
166       || (img->numcomps == 1)) {
167       
168       if (!imagetoyuv(img, argv[2]))    // Convert image to YUV
169                                 return 1;
170     }
171     else if ((img->numcomps == 3) && 
172       (img->comps[0].dx == 1) && (img->comps[1].dx == 1)&&
173       (img->comps[2].dx == 1))// If YUV 4:4:4 input --> to bmp
174     {
175       fprintf(stdout,"The frames will be output in a bmp format (output_1.bmp, ...)\n");
176       sprintf(outfilename,"output_%d.bmp",snum);
177       if (imagetobmp(img, outfilename)) // Convert image to YUV
178                                 return 1;
179       
180     }
181     else {
182       fprintf(stdout,"Image component dimensions are unknown. Unable to output image\n");
183       fprintf(stdout,"The frames will be output in a j2k file (output_1.j2k, ...)\n");
184                         
185       sprintf(outfilename,"output_%d.j2k",snum);
186       outfile = fopen(outfilename, "wb");
187       if (!outfile) {
188                                 fprintf(stderr, "failed to open %s for writing\n",outfilename);
189                                 return 1;
190       }
191       fwrite(frame_codestream,sample->sample_size-8,1,outfile);
192       fclose(outfile);
193     }
194                 /* close the byte stream */
195                 opj_cio_close(cio);     
196                 /* free image data structure */
197                 opj_image_destroy(img);
198                 elapsed_time = opj_clock()-init_time;
199                 fprintf(stderr, "Frame number %d/%d decoded in %.2f mseconds\n", snum + 1, numframes, elapsed_time*1000);
200                 total_time += elapsed_time;
201
202   }
203         
204         free(frame_codestream); 
205   fclose(file); 
206         /* free remaining structures */
207         if(dinfo) {
208                 mj2_destroy_decompress((opj_mj2_t*)dinfo->mj2_handle);
209         }
210         free(dinfo);
211         
212         fprintf(stdout, "%d frame(s) correctly decompressed\n", snum);
213         fprintf(stdout,"Total decoding time: %.2f seconds (%.1f fps)\n", total_time, (float)numframes/total_time);
214                 
215   return 0;
216 }