/* ** ** $Id: jpegxr.c,v 1.4 2008/05/25 09:40:46 thor Exp $ ** ** This is the main program ** */ /************************************************************************* * * This software module was originally contributed by Microsoft * Corporation in the course of development of the * ITU-T T.832 | ISO/IEC 29199-2 ("JPEG XR") format standard for * reference purposes and its performance may not have been optimized. * * This software module is an implementation of one or more * tools as specified by the JPEG XR standard. * * ITU/ISO/IEC give You a royalty-free, worldwide, non-exclusive * copyright license to copy, distribute, and make derivative works * of this software module or modifications thereof for use in * products claiming conformance to the JPEG XR standard as * specified by ITU-T T.832 | ISO/IEC 29199-2. * * ITU/ISO/IEC give users the same free license to this software * module or modifications thereof for research purposes and further * ITU/ISO/IEC standardization. * * Those intending to use this software module in products are advised * that its use may infringe existing patents. ITU/ISO/IEC have no * liability for use of this software module or modifications thereof. * * Copyright is not released for products that do not conform to * to the JPEG XR standard as specified by ITU-T T.832 | * ISO/IEC 29199-2. * * Microsoft Corporation retains full right to modify and use the code * for its own purpose, to assign or donate the code to a third party, * and to inhibit third parties from using the code for products that * do not conform to the JPEG XR standard as specified by ITU-T T.832 | * ISO/IEC 29199-2. * * This copyright notice must be included in all copies or derivative * works. * * Copyright (c) ITU-T/ISO/IEC 2008, 2009. ***********************************************************************/ #ifdef _MSC_VER #pragma comment (user,"$Id: jpegxr.c,v 1.4 2008/05/25 09:40:46 thor Exp $") #else #ident "$Id: jpegxr.c,v 1.4 2008/05/25 09:40:46 thor Exp $" #endif # include # include # include # include # include "jpegxr.h" # include "file.h" # include /* MSC doesn't include getopt, so if compiling with MSC then get it from our local copy. Otherwise use the C library version. */ #ifdef _MSC_VER # include "getopt.h" /* MSVC doesn't have strcasecmp. Use _stricmp instead */ # define strcasecmp _stricmp #else # include #endif char*path_out = 0; char*path_debug = 0; char*quant_uniform = 0; char*flags[128]; int nflags = 0; static jxr_bands_present_t bands_present = JXR_BP_ALL; static int trim_flexbits = 0; static int overlap_filter = 1; static jxr_color_fmt_t internal_color_fmt = JXR_YUV444; static int disable_tile_overlap = 0; static int frequency_mode_flag = 0; static int profile_idc = 111; static int level_idc = 255; static int long_word_flag_setting = 1; static int alpha_mode = 2; static int padded_format = 0; static unsigned char window_params[5] = {0,0,0,0,0}; static jxrc_t_pixelFormat pxl_fmt; /* Set this flag if the encoder should try to use the USE_DC_QP and USE_LP_QP flags when it can. */ static int use_dc_qp_flag = 0; # define MAX_CHANNELS 16 static unsigned char quant_per_channel[MAX_CHANNELS]; static unsigned quant_per_channel_count = 0; # define MAX_TILES 4096 static unsigned tile_rows = 1; static unsigned tile_columns = 1; static unsigned tile_width_in_MB[MAX_TILES * 2] = {0}; static unsigned tile_height_in_MB[MAX_TILES * 2] = {0}; static unsigned char shift_bits = 0; static unsigned char len_mantissa = 10; static char exp_bias = 4; static raw_info raw_info_t; static int main_compress(const char*path); static int main_decompress(const char*path); /* This function parses a qp input file and returns an array of jxr_tile_qp with the results. The parser needs to know the tile dimensions of the image in order to get the right number of tiles. */ extern int qp_parse_file(FILE*fd, jxr_image_t image); /* Path to the input quant file. */ static const char*quant_path = 0; int main(int argc, char*argv[]) { int compress_flag = 0; int flag; int derive_flag = 0; char*cp_temp; int idx; /* initialize raw input */ raw_info_t.is_raw = 0; raw_info_t.raw_width = 0; raw_info_t.raw_height = 0; raw_info_t.raw_format = 3; raw_info_t.raw_bpc = 8; while ((flag = getopt(argc, argv, "b:cphmwdra:D:f:F:l:o:U:C:R:P:L:q:Q:X:W:H:M:B:s:")) != -1) { switch (flag) { case 'b': /* Bands present */ if (strcasecmp(optarg,"ALL") == 0) bands_present = JXR_BP_ALL; else if (strcasecmp(optarg,"NOFLEXBITS") == 0) bands_present = JXR_BP_NOFLEXBITS; else if (strcasecmp(optarg,"NOHIGHPASS") == 0) bands_present = JXR_BP_NOHIGHPASS; else if (strcasecmp(optarg,"DCONLY") == 0) bands_present = JXR_BP_DCONLY; else { fprintf(stderr, "Invalid bands flag: %s\n", optarg); fprintf(stderr, "Should be one of: ALL, NOFLEXBITS, NOHIGHPASS, DCONLY\n"); return 1; } break; case 'c': compress_flag = 1; break; case 'd': derive_flag = 1; break; case 'D': path_debug = optarg; break; case 'p': padded_format = 1; break; case 'f': if (strcmp(optarg, "USE_DC_QP") == 0) { use_dc_qp_flag = 1; } else if (strcmp(optarg,"YUV444") == 0) { internal_color_fmt = JXR_YUV444; } else if (strcmp(optarg,"YUV422") == 0) { internal_color_fmt = JXR_YUV422; } else if (strcmp(optarg,"YUV420") == 0) { internal_color_fmt = JXR_YUV420; } else { } break; case 'F': trim_flexbits = strtoul(optarg, 0, 10); if (trim_flexbits > 15) { trim_flexbits = 15; } break; case 'h': disable_tile_overlap = 1; break; case 'm': frequency_mode_flag = 1; break; case 'w': long_word_flag_setting = 0; break; case 'l': overlap_filter = strtoul(optarg, 0, 10); if (overlap_filter >= 3) { fprintf(stderr, "Invalid overlap filter flag. Got %d\n", overlap_filter); return 1; } break; case 'X': flags[nflags] = optarg; nflags += 1; break; case 'o': path_out = optarg; break; case 's': cp_temp = optarg; window_params[0] = 1; unsigned read_value; for (idx = 0 ; idx < 4 ; idx += 1) { read_value = (unsigned) strtoul(cp_temp, &cp_temp, 10); assert(read_value >= 0 && read_value < 64); window_params[idx + 1] = (unsigned char) read_value; if (*cp_temp == ':') cp_temp += 1; } break; case 'U': cp_temp = optarg; tile_rows = (unsigned) strtoul(cp_temp, &cp_temp, 10); if (tile_rows > MAX_TILES) tile_rows = MAX_TILES; if (*cp_temp == ':') { cp_temp += 1; tile_columns = (unsigned) strtoul(cp_temp, &cp_temp, 10); if (tile_columns > MAX_TILES) tile_columns = MAX_TILES; } else { tile_columns = tile_rows; } break; case 'a': alpha_mode = strtoul(optarg, 0, 10); if(alpha_mode <0 || alpha_mode > 2) assert(0); break; case 'C': cp_temp = optarg; for (tile_columns = 0 ; tile_columns < MAX_TILES ; tile_columns++) { tile_width_in_MB[tile_columns] = (unsigned) strtoul(cp_temp, &cp_temp, 10); if (*cp_temp == ':') cp_temp += 1; else break; } tile_columns++; break; case 'R': cp_temp = optarg; for (tile_rows = 0 ; tile_rows < MAX_TILES ; tile_rows++) { tile_height_in_MB[tile_rows] = (unsigned) strtoul(cp_temp, &cp_temp, 10); if (*cp_temp == ':') cp_temp += 1; else break; } tile_rows++; break; case 'q': { /* Uniform quantization */ quant_uniform = optarg; quant_per_channel_count = 0; int last_q = 0; cp_temp = optarg; for (idx = 0 ; idx < MAX_CHANNELS ; idx += 1) { if (*cp_temp == 0) { quant_per_channel[idx] = quant_per_channel[last_q]; continue; } last_q = idx; quant_per_channel_count += 1; quant_per_channel[idx] = (unsigned char) strtoul(cp_temp, &cp_temp, 10); if (*cp_temp == ':') cp_temp += 1; } break; } case 'Q': quant_path = optarg; break; case 'P': profile_idc = strtoul(optarg, 0, 10); if (profile_idc > 255) { profile_idc = 255; } if (profile_idc < 0) { profile_idc = 0; } break; case 'L': level_idc = strtoul(optarg, 0, 10); if (level_idc > 255) { level_idc = 255; } if (level_idc < 0) { level_idc = 0; } break; case 'r': raw_info_t.is_raw = 1; break; case 'W': raw_info_t.raw_width = (unsigned int)atoi(optarg); break; case 'H': raw_info_t.raw_height = (unsigned int)atoi(optarg); break; case 'M': raw_info_t.raw_format = (unsigned char)atoi(optarg); if ((raw_info_t.raw_format < 3) || (raw_info_t.raw_format > 34)) { fprintf(stderr, "Invalid raw format.\n"); return -1; } break; case 'B': raw_info_t.raw_bpc = (unsigned char)atoi(optarg); if ((8 != raw_info_t.raw_bpc) && (10 != raw_info_t.raw_bpc) && (16 != raw_info_t.raw_bpc)) { fprintf(stderr, "Invalid bit/channel.\n"); return -1; } break; case '?': fprintf(stderr, "Abort from bad flag.\n"); return -1; } } if (optind >= argc) { fprintf(stderr,"Usage: %s \n" " DECODER FLAGS:\n" " [-o ] [-w] [-P 44|55|66|111] [-L 4|8|16|32|64|128|255]\n" "\n" "\t-o: selects output file name (.raw/.tif/.pnm)\n" "\t (PNM output can be used only for 24bpp RGB and 8bpp gray Output)\n" "\t (TIF output can be used for all formats except the following:\n" "\t N channels, BGR, RGBE, YCC, CMYKDirect & Premultiplied RGB)\n" "\t (RAW output can be used for all formats)\n" "\t-w: tests whether LONG_WORD_FLAG needs to be equal to TRUE\n" "\t (will still decode the image)\n" "\t-P: selects the maximum accepted profile value\n" "\t (44:Sub-Baseline|55:Baseline|66:Main|111:Advanced)\n" "\t-L: selects the maximum accepted level value\n" "\t (4|8|16|32|64|128)\n" "\n" " ENCODER FLAGS: (Temporary (.tmp) files may be used in encoding)\n" " -c [-o ] [-b ALL|NOFLEXBITS|NOHIGHPASS|DCONLY] [-a 0|1|2] [-p]\n" " [-f YUV444|YUV422|YUV420] [-F bits] [-h] [-m] [-l 0|1|2]\n" " [-q q1[:q2[:q3]]] [-Q ] [-d] [-w] [-U rows:columns]\n" " [-C width1[:width2>[:width3...]]] [-R height1[:height2[:height3...]]]\n" " [-P 44|55|66|111] [-L 4|8|16|32|64|128|255] [-s top|left|bottom|right]\n" " [-r -W width -H height -M 3|4|...|18 [-B 8|16]]\n" "\n" "\t-c: selects encoding instead of decoding\n" "\t this flag is necessary for encoding\n" "\t-o: selects the output file name (.jxr)\n" "\t-b: selects the bands to encode\n" "\t (ALL|NOFLEXBITS|NOHIGHPASS|DCONLY)\n" "\t-a: selects encoder alpha mode\n" "\t (0: no alpha|1:interleaved alpha|2:separate alpha) \n" "\t Default: For tif input files, based on the information in the\n" "\t PhotometricInterpretation and SamplesPerPixel tags in the container,\n" "\t the encoder chooses an input pixel format. If the number\n" "\t of components is 4 and photometric is 2, RGBA input is inferred and \n" "\t the encoder assumes the presence of an alpha channel while encoding.\n" "\t If the number of components is 5 and photometric is 5, CMYKA input is\n" "\t inferred. In both these cases, the encoder infers a pixel format with\n" "\t an alpha channel. In such cases, the default alpha encoder mode is 2.\n" "\t For raw input files, when the -M parameter specified by the user is\n" "\t 9, 10, 11, 12 13, 14, 23, 24, 25, 26 or 28,\n" "\t the default alpha encoder mode is 2\n" "\t In all other cases, the default alpha encoder mode is 0.\n" "\t-p: selects an input pixel format with a padding channel\n" "\t With tif input, when the encoder infers that the input file has an\n" "\t alpha channel (see explanation for -a), this flag causes the encoder\n" "\t to treat the alpha channel as a padding channel instead\n" "\t-f: selects the internal color format\n" "\t (YUV444|YUV422|YUV420)\n" "\t-F: selects the number of flexbits to trim\n" "\t (0 - 15)\n" "\t-h: selects hard tile boundaries\n" "\t (soft tile boundaries by default)\n" "\t-m: encode in frequency order codestream\n" "\t (spatial order by default)\n" "\t-l: selects the overlapped block filtering\n" "\t (0:off|1:HP only|2:all)\n" "\n" "\t-q: sets the quantization values separately, or one per band\n" "\t (0 - 255)\n" "\t-Q: specifies a file containing detailed quantization information\n" "\t See sample.qp\n" "\t-d: selects quantization for U/V channels derived from Y channel\n" "\n" "\t-U: selects uniform tile sizes\n" "\t-C: selects the number of tile columns and the width of each\n" "\t-R: selects the number of tile rows and the height of each\n" "\n" "\t-w: sets LONG_WORD_FLAG equal to FALSE\n" "\t-P: selects the profile value\n" "\t (44:Sub-Baseline|55:Baseline|66:Main|111:Advanced)\n" "\t-L: selects the level value\n" "\t (4|8|16|32|64|128)\n" "\t-s: sets the top, left, bottom, and right margins\n" "\n" "\t-r: selects encoding with RAW images\n" "\t must also specify -W, -H and -M, optional -B\n" "\t-W: RAW image width when encoding with RAW images\n" "\t-H: RAW image height when encoding with RAW images\n" "\t-M: RAW image format when encoding with RAW images\n" "\t 3: 3-channel\n" "\t 4: 4-channel\n" "\t 5: 5-channel\n" "\t 6: 6-channel\n" "\t 7: 7-channel\n" "\t 8: 8-channel\n" "\t 9: 3-channel Alpha\n" "\t 10: 4-channel Alpha\n" "\t 11: 5-channel Alpha\n" "\t 12: 6-channel Alpha\n" "\t 13: 7-channel Alpha\n" "\t 14: 8-channel Alpha\n" "\t 15: 32bppRGBE\n" "\t 16: 16bppBGR555\n" "\t 17: 16bppBGR565\n" "\t 18: 32bppBGR101010\n" "\t 19: YCC420\n" "\t 20: YCC422\n" "\t 21: YCC444\n" "\t 22: YCC444 Fixed Point\n" "\t 23: YCC420 Alpha\n" "\t 24: YCC422 Alpha\n" "\t 25: YCC444 Alpha\n" "\t 26: YCC444 Fixed Point Alpha\n" "\t 27: CMYKDIRECT\n" "\t 28: CMYKDIRECT Alpha\n" "\t 29: 24bppBGR\n" "\t 30: 32bppBGR\n" "\t 31: 32bppBGRA\n" "\t 32: 32bppPBGRA\n" "\t 33: 64bppPRGBA\n" "\t 34: 128bppPRGBAFloat\n" "\t-B: RAW image bit/channel when encoding with RAW images\n" "\t 8: 8-bit/channel (default)\n" "\t 10: 10-bit/channel\n" "\t 16: 16-bit/channel\n" "\n", argv[0]); return 1; } /* ** start mod thor: Derived quantization as in the DPK */ if (derive_flag) { int idx = quant_per_channel[0]; int idxuv = idx; switch (internal_color_fmt) { case JXR_YUV444: if (idx < 16) { idxuv = idx * 2; } else if (idx <= 48) { idxuv = idx + 18; } else { idxuv = idx + 18 + 2; } quant_per_channel_count = 3; break; case JXR_YUV422: if (idx < 16) { idxuv = idx + ((idx + 1) >> 1); } else if (idx <= 48) { idxuv = idx + 8; } else { idxuv = idx + 8 + 2; } quant_per_channel_count = 3; break; case JXR_YUV420: if (idx < 16) { idxuv = idx + ((idx + 2) >> 2); } else if (idx <= 48) { idxuv = idx + 4; } else { idxuv = idx + 4 + 2; } quant_per_channel_count = 3; break; default: idxuv = idx; break; } for(idx = 1;idx < MAX_CHANNELS;idx++) { quant_per_channel[idx] = idxuv; } } /* ** End mod thor */ if(disable_tile_overlap) assert(compress_flag); if (frequency_mode_flag) assert(compress_flag); if (compress_flag) return main_compress(argv[optind]); else return main_decompress(argv[optind]); } int setup_image_params(jxr_image_t *ptr_image, void *input_handle, int alpha_mode, int alpha_plane) { int rc = 0; int wid, hei, ncomp, bpi; short sf, photometric; int padBytes; get_file_parameters(input_handle, &wid, &hei, &ncomp, &bpi, &sf, &photometric, &padBytes); wid -= (window_params[2] + ((window_params[4] >> 4) << 4)); assert(wid > 0); hei -= (window_params[1] + ((window_params[3] >> 4) << 4)); assert(hei > 0); /* Create a stub image. */ jxr_image_t image = jxr_create_image(wid, hei, window_params); *ptr_image = image; if (!raw_info_t.is_raw) { /* Guess a color format from the number of components. */ if(!alpha_plane) { switch (ncomp + padBytes) { case 1: jxr_set_INTERNAL_CLR_FMT(image, JXR_YONLY, 1); jxr_set_OUTPUT_CLR_FMT(image, JXR_OCF_YONLY); break; case 3: jxr_set_INTERNAL_CLR_FMT(image, internal_color_fmt, 3); jxr_set_OUTPUT_CLR_FMT(image, JXR_OCF_RGB); break; case 4: if (photometric == 5){ jxr_set_INTERNAL_CLR_FMT(image, JXR_YUVK, 4); jxr_set_OUTPUT_CLR_FMT(image, JXR_OCF_CMYK); } else if (photometric == 2){ /* RGB_NULL */ jxr_set_INTERNAL_CLR_FMT(image, internal_color_fmt, 3); jxr_set_OUTPUT_CLR_FMT(image, JXR_OCF_RGB); } break; case 5: if (photometric == 5){ /* CMYKA */ jxr_set_INTERNAL_CLR_FMT(image, JXR_YUVK, 4); jxr_set_OUTPUT_CLR_FMT(image, JXR_OCF_CMYK); } break; default: jxr_set_INTERNAL_CLR_FMT(image, JXR_NCOMPONENT, ncomp); jxr_set_OUTPUT_CLR_FMT(image, JXR_OCF_NCOMPONENT); break; } } else { jxr_set_INTERNAL_CLR_FMT(image, JXR_YONLY, 1); jxr_set_OUTPUT_CLR_FMT(image, JXR_OCF_YONLY); } switch (bpi) { case 1: if (photometric) /*white is 1 */ jxr_set_OUTPUT_BITDEPTH(image, JXR_BD1WHITE1); else jxr_set_OUTPUT_BITDEPTH(image, JXR_BD1BLACK1); break; case 8: jxr_set_OUTPUT_BITDEPTH(image, JXR_BD8); break; case 16: if (sf == 1) { jxr_set_OUTPUT_BITDEPTH(image, JXR_BD16); jxr_set_SHIFT_BITS(image, shift_bits); } else if (sf == 2) { jxr_set_OUTPUT_BITDEPTH(image, JXR_BD16S); jxr_set_SHIFT_BITS(image, shift_bits); } else if (sf == 3) jxr_set_OUTPUT_BITDEPTH(image, JXR_BD16F); break; case 32: if (sf == 2) { jxr_set_OUTPUT_BITDEPTH(image, JXR_BD32S); jxr_set_SHIFT_BITS(image, shift_bits); } else if (sf == 3) { jxr_set_OUTPUT_BITDEPTH(image, JXR_BD32F); jxr_set_FLOAT(image, len_mantissa, exp_bias); } break; default: assert(0); break; } } else { /* raw */ if ((raw_info_t.raw_format >= 3) && (raw_info_t.raw_format <= 14)) { /* N-channel and N-channel Alpha*/ if(!alpha_plane) jxr_set_OUTPUT_CLR_FMT(image, JXR_OCF_NCOMPONENT); else jxr_set_OUTPUT_CLR_FMT(image, JXR_OCF_YONLY); if (raw_info_t.raw_format <= 8) jxr_set_INTERNAL_CLR_FMT(image, JXR_NCOMPONENT, ncomp); /* N-channel */ else { if(!alpha_plane) jxr_set_INTERNAL_CLR_FMT(image, JXR_NCOMPONENT, ncomp - 1); /* N-channel Alpha */ else jxr_set_INTERNAL_CLR_FMT(image, JXR_YONLY, 1); /* N-channel Alpha */ } if (8 == raw_info_t.raw_bpc) jxr_set_OUTPUT_BITDEPTH(image, JXR_BD8); else jxr_set_OUTPUT_BITDEPTH(image, JXR_BD16); } else if (raw_info_t.raw_format == 15) { jxr_set_INTERNAL_CLR_FMT(image, JXR_YUV444, 1); jxr_set_OUTPUT_CLR_FMT(image, JXR_OCF_RGBE); jxr_set_OUTPUT_BITDEPTH(image, JXR_BD8); } else if (raw_info_t.raw_format == 16) { jxr_set_INTERNAL_CLR_FMT(image, internal_color_fmt, 1); jxr_set_OUTPUT_CLR_FMT(image, JXR_OCF_RGB); jxr_set_OUTPUT_BITDEPTH(image, JXR_BD5); } else if (raw_info_t.raw_format == 17) { jxr_set_INTERNAL_CLR_FMT(image, internal_color_fmt, 1); jxr_set_OUTPUT_CLR_FMT(image, JXR_OCF_RGB); jxr_set_OUTPUT_BITDEPTH(image, JXR_BD565); } else if (raw_info_t.raw_format == 18) { jxr_set_INTERNAL_CLR_FMT(image, internal_color_fmt, 1); jxr_set_OUTPUT_CLR_FMT(image, JXR_OCF_RGB); jxr_set_OUTPUT_BITDEPTH(image, JXR_BD10); } else if (raw_info_t.raw_format == 19 || raw_info_t.raw_format == 23) { if(!alpha_plane) { jxr_set_INTERNAL_CLR_FMT(image, JXR_YUV420, 3); jxr_set_OUTPUT_CLR_FMT(image, JXR_OCF_YUV420); } else { jxr_set_INTERNAL_CLR_FMT(image, JXR_YONLY, 1); jxr_set_OUTPUT_CLR_FMT(image, JXR_OCF_YONLY); } jxr_set_OUTPUT_BITDEPTH(image, JXR_BD8); } else if (raw_info_t.raw_format == 20 || raw_info_t.raw_format == 24) { if(!alpha_plane) { jxr_set_INTERNAL_CLR_FMT(image, JXR_YUV422, 3); jxr_set_OUTPUT_CLR_FMT(image, JXR_OCF_YUV422); } else { jxr_set_INTERNAL_CLR_FMT(image, JXR_YONLY, 1); jxr_set_OUTPUT_CLR_FMT(image, JXR_OCF_YONLY); } if (8 == raw_info_t.raw_bpc) jxr_set_OUTPUT_BITDEPTH(image, JXR_BD8); else if (10 == raw_info_t.raw_bpc) jxr_set_OUTPUT_BITDEPTH(image, JXR_BD10); else jxr_set_OUTPUT_BITDEPTH(image, JXR_BD16); } else if (raw_info_t.raw_format == 21 || raw_info_t.raw_format == 25) { if(!alpha_plane) { jxr_set_INTERNAL_CLR_FMT(image, internal_color_fmt, 3); jxr_set_OUTPUT_CLR_FMT(image, JXR_OCF_YUV444); } else { jxr_set_INTERNAL_CLR_FMT(image, JXR_YONLY, 1); jxr_set_OUTPUT_CLR_FMT(image, JXR_OCF_YONLY); } if (8 == raw_info_t.raw_bpc) jxr_set_OUTPUT_BITDEPTH(image, JXR_BD8); else if (10 == raw_info_t.raw_bpc) jxr_set_OUTPUT_BITDEPTH(image, JXR_BD10); else jxr_set_OUTPUT_BITDEPTH(image, JXR_BD16); } else if (raw_info_t.raw_format == 22 || raw_info_t.raw_format == 26) { if(!alpha_plane) { jxr_set_INTERNAL_CLR_FMT(image, JXR_YUV444, 3); jxr_set_OUTPUT_CLR_FMT(image, JXR_OCF_YUV444); } else { jxr_set_INTERNAL_CLR_FMT(image, JXR_YONLY, 1); jxr_set_OUTPUT_CLR_FMT(image, JXR_OCF_YONLY); } jxr_set_OUTPUT_BITDEPTH(image, JXR_BD16S); } else if (raw_info_t.raw_format == 27 || raw_info_t.raw_format == 28) { if(!alpha_plane) { jxr_set_INTERNAL_CLR_FMT(image, JXR_YUVK, 4); jxr_set_OUTPUT_CLR_FMT(image, JXR_OCF_CMYKDIRECT); } else { jxr_set_INTERNAL_CLR_FMT(image, JXR_YONLY, 1); jxr_set_OUTPUT_CLR_FMT(image, JXR_OCF_YONLY); } if (8 == raw_info_t.raw_bpc) jxr_set_OUTPUT_BITDEPTH(image, JXR_BD8); else jxr_set_OUTPUT_BITDEPTH(image, JXR_BD16); } else if (raw_info_t.raw_format == 29) { jxr_set_INTERNAL_CLR_FMT(image, internal_color_fmt, 3); jxr_set_OUTPUT_CLR_FMT(image, JXR_OCF_RGB); jxr_set_OUTPUT_BITDEPTH(image, JXR_BD8); } else if (raw_info_t.raw_format == 30) { jxr_set_INTERNAL_CLR_FMT(image, internal_color_fmt, 3); jxr_set_OUTPUT_CLR_FMT(image, JXR_OCF_RGB); jxr_set_OUTPUT_BITDEPTH(image, JXR_BD8); } else if (raw_info_t.raw_format == 31) { if(!alpha_plane) { jxr_set_INTERNAL_CLR_FMT(image, internal_color_fmt, 3); jxr_set_OUTPUT_CLR_FMT(image, JXR_OCF_RGB); jxr_set_OUTPUT_BITDEPTH(image, JXR_BD8); } else { jxr_set_INTERNAL_CLR_FMT(image, JXR_YONLY, 1); jxr_set_OUTPUT_CLR_FMT(image, JXR_OCF_YONLY); jxr_set_OUTPUT_BITDEPTH(image, JXR_BD8); } } else if (raw_info_t.raw_format == 32) { if(!alpha_plane) { jxr_set_INTERNAL_CLR_FMT(image, internal_color_fmt, 3); jxr_set_OUTPUT_CLR_FMT(image, JXR_OCF_RGB); jxr_set_OUTPUT_BITDEPTH(image, JXR_BD8); } else { jxr_set_INTERNAL_CLR_FMT(image, JXR_YONLY, 1); jxr_set_OUTPUT_CLR_FMT(image, JXR_OCF_YONLY); jxr_set_OUTPUT_BITDEPTH(image, JXR_BD8); } } else if (raw_info_t.raw_format == 33) { if(!alpha_plane) { jxr_set_INTERNAL_CLR_FMT(image, internal_color_fmt, 3); jxr_set_OUTPUT_CLR_FMT(image, JXR_OCF_RGB); jxr_set_OUTPUT_BITDEPTH(image, JXR_BD16); } else { jxr_set_INTERNAL_CLR_FMT(image, JXR_YONLY, 1); jxr_set_OUTPUT_CLR_FMT(image, JXR_OCF_YONLY); jxr_set_OUTPUT_BITDEPTH(image, JXR_BD16); } } else if (raw_info_t.raw_format == 34) { if(!alpha_plane) { jxr_set_INTERNAL_CLR_FMT(image, internal_color_fmt, 3); jxr_set_OUTPUT_CLR_FMT(image, JXR_OCF_RGB); jxr_set_OUTPUT_BITDEPTH(image, JXR_BD32F); jxr_set_FLOAT(image, len_mantissa, exp_bias); } else { jxr_set_INTERNAL_CLR_FMT(image, JXR_YONLY, 1); jxr_set_OUTPUT_CLR_FMT(image, JXR_OCF_YONLY); jxr_set_OUTPUT_BITDEPTH(image, JXR_BD32F); jxr_set_FLOAT(image, len_mantissa, exp_bias); } } if(alpha_mode == 2 && !alpha_plane) { set_ncomp(input_handle, ncomp - 1); } else if(alpha_mode == 2 && alpha_plane) { set_ncomp(input_handle, 1); } } jxr_set_BANDS_PRESENT(image, bands_present); jxr_set_TRIM_FLEXBITS(image, trim_flexbits); jxr_set_OVERLAP_FILTER(image, overlap_filter); jxr_set_DISABLE_TILE_OVERLAP(image, disable_tile_overlap); jxr_set_FREQUENCY_MODE_CODESTREAM_FLAG(image, frequency_mode_flag); jxr_set_PROFILE_IDC(image, profile_idc); jxr_set_LEVEL_IDC(image, level_idc); jxr_set_LONG_WORD_FLAG(image, long_word_flag_setting); jxr_set_ALPHA_IMAGE_PLANE_FLAG(image, alpha_mode == 1 ? 1 : 0); jxr_set_NUM_VER_TILES_MINUS1(image, tile_columns); jxr_set_TILE_WIDTH_IN_MB(image, tile_width_in_MB); jxr_set_NUM_HOR_TILES_MINUS1(image, tile_rows); jxr_set_TILE_HEIGHT_IN_MB(image, tile_height_in_MB); jxr_set_pixel_format(image, pxl_fmt); if (quant_uniform) { if (quant_per_channel_count == 1) { jxr_set_QP_UNIFORM(image, quant_per_channel[0]); } else if (ncomp >= 3 && quant_per_channel_count == 2) { jxr_set_QP_SEPARATE(image, quant_per_channel); } else { int idx; for (idx = ncomp ; idx < (int) quant_per_channel_count ; idx += 1) quant_per_channel[idx] = quant_per_channel[ncomp-1]; jxr_set_QP_INDEPENDENT(image, quant_per_channel); } } else if (quant_path) { FILE*quant_fd = fopen(quant_path, "r"); if (quant_fd == 0) { perror(quant_path); return 1; } int rc = qp_parse_file(quant_fd, image); if (rc < 0) return 1; } else { jxr_set_QP_LOSSLESS(image); } rc = jxr_test_PROFILE_IDC(image, 0); if (rc < 0) { return 1; } rc = jxr_test_LEVEL_IDC(image, 0); if (rc < 0) { return 1; } return 0; } int setup_container_params(jxr_container_t container, void *input_handle) { int wid, hei, ncomp, bpi; short sf, photometric; int padBytes; get_file_parameters(input_handle, &wid, &hei, &ncomp, &bpi, &sf, &photometric, &padBytes); int alpha_present = 0; if (!raw_info_t.is_raw) { switch (ncomp + padBytes) { case 1: if (bpi == 1) { pxl_fmt = JXRC_FMT_BlackWhite; } else if (bpi == 8) { pxl_fmt = JXRC_FMT_8bppGray; } else if (bpi == 16) { if(sf == 1) pxl_fmt = JXRC_FMT_16bppGray; else if(sf == 2) pxl_fmt = JXRC_FMT_16bppGrayFixedPoint; else if(sf == 3) pxl_fmt = JXRC_FMT_16bppGrayHalf; } else if (bpi == 32) { if(sf == 2) pxl_fmt = JXRC_FMT_32bppGrayFixedPoint; else if(sf == 3) pxl_fmt = JXRC_FMT_32bppGrayFloat; } break; case 3: /* Assume RGB */ if (bpi == 8) pxl_fmt = (JXRC_FMT_24bppRGB); else if (bpi == 16) { if(sf == 1) pxl_fmt = JXRC_FMT_48bppRGB; else if(sf == 2) pxl_fmt = JXRC_FMT_48bppRGBFixedPoint; else if(sf == 3) pxl_fmt = JXRC_FMT_48bppRGBHalf; } else if (bpi == 32) { if(sf == 2) pxl_fmt = JXRC_FMT_96bppRGBFixedPoint; /* no 96bppRGBFloat */ } break; case 4: /* CMYK or RGBA or RGB_Null*/ if (bpi == 8) { if (photometric == 5) pxl_fmt = JXRC_FMT_32bppCMYK; } else if (bpi == 16) { if (photometric == 5) pxl_fmt = JXRC_FMT_64bppCMYK; else if (photometric == 2) { if (sf == 1) { alpha_present = 1; pxl_fmt = JXRC_FMT_64bppRGBA; /* no 64bppRGB */ } else if (sf == 2) { if (ncomp == 3) pxl_fmt = JXRC_FMT_64bppRGBFixedPoint; else if (ncomp == 4) { alpha_present = 1; pxl_fmt = JXRC_FMT_64bppRGBAFixedPoint; } } else if (sf == 3) { if (ncomp == 3) pxl_fmt = JXRC_FMT_64bppRGBHalf; else if (ncomp == 4) { alpha_present = 1; pxl_fmt = JXRC_FMT_64bppRGBAHalf; } } } } else if (bpi == 32) { if (photometric == 2) { if (sf == 2) { if (ncomp == 3) pxl_fmt = JXRC_FMT_128bppRGBFixedPoint; if (ncomp == 4) { alpha_present = 1; pxl_fmt = JXRC_FMT_128bppRGBAFixedPoint; } } else if(sf == 3) { if (ncomp == 3) pxl_fmt = JXRC_FMT_128bppRGBFloat; else if (ncomp == 4) { alpha_present = 1; pxl_fmt = JXRC_FMT_128bppRGBAFloat; } } } /* add 128bppRGBAFloat here */ } break; case 5: /* CMYKA */ if (bpi == 8) { if (photometric == 5) { alpha_present = 1; pxl_fmt = JXRC_FMT_40bppCMYKAlpha; } } else if (bpi == 16) { if (photometric == 5) { alpha_present = 1; pxl_fmt = JXRC_FMT_80bppCMYKAlpha; } } break; default: assert(0); break; } } else { /* raw */ if ((raw_info_t.raw_format >= 3) && (raw_info_t.raw_format <= 14)) { /* N-channel */ if (8 == raw_info_t.raw_bpc) { switch (raw_info_t.raw_format) { case 3: pxl_fmt = JXRC_FMT_24bpp3Channels; break; case 4: pxl_fmt = JXRC_FMT_32bpp4Channels; break; case 5: pxl_fmt = JXRC_FMT_40bpp5Channels; break; case 6: pxl_fmt = JXRC_FMT_48bpp6Channels; break; case 7: pxl_fmt = JXRC_FMT_56bpp7Channels; break; case 8: pxl_fmt = JXRC_FMT_64bpp8Channels; break; case 9: alpha_present = 1; pxl_fmt = JXRC_FMT_32bpp3ChannelsAlpha; break; case 10: alpha_present = 1; pxl_fmt = JXRC_FMT_40bpp4ChannelsAlpha; break; case 11: alpha_present = 1; pxl_fmt = JXRC_FMT_48bpp5ChannelsAlpha; break; case 12: alpha_present = 1; pxl_fmt = JXRC_FMT_56bpp6ChannelsAlpha; break; case 13: alpha_present = 1; pxl_fmt = JXRC_FMT_64bpp7ChannelsAlpha; break; case 14: alpha_present = 1; pxl_fmt = JXRC_FMT_72bpp8ChannelsAlpha; break; default: assert(0); break; } } else { /* 16bpc */ switch (raw_info_t.raw_format) { case 3: pxl_fmt = JXRC_FMT_48bpp3Channels; break; case 4: pxl_fmt = JXRC_FMT_64bpp4Channels; break; case 5: pxl_fmt = JXRC_FMT_80bpp5Channels; break; case 6: pxl_fmt = JXRC_FMT_96bpp6Channels; break; case 7: pxl_fmt = JXRC_FMT_112bpp7Channels; break; case 8: pxl_fmt = JXRC_FMT_128bpp8Channels; break; case 9: alpha_present = 1; pxl_fmt = JXRC_FMT_64bpp3ChannelsAlpha; break; case 10: alpha_present = 1; pxl_fmt = JXRC_FMT_80bpp4ChannelsAlpha; break; case 11: alpha_present = 1; pxl_fmt = JXRC_FMT_96bpp5ChannelsAlpha; break; case 12: alpha_present = 1; pxl_fmt = JXRC_FMT_112bpp6ChannelsAlpha; break; case 13: alpha_present = 1; pxl_fmt = JXRC_FMT_128bpp7ChannelsAlpha; break; case 14: alpha_present = 1; pxl_fmt = JXRC_FMT_144bpp8ChannelsAlpha; break; default: assert(0); break; } } } else if (raw_info_t.raw_format == 15) { /* RGBE */ pxl_fmt = JXRC_FMT_32bppRGBE; } else if (raw_info_t.raw_format == 16) { pxl_fmt = JXRC_FMT_16bppBGR555; } else if (raw_info_t.raw_format == 17) { pxl_fmt = JXRC_FMT_16bppBGR565; } else if (raw_info_t.raw_format == 18) { pxl_fmt = JXRC_FMT_32bppBGR101010; } else if ((raw_info_t.raw_format >= 19) && (raw_info_t.raw_format <= 26)) { /* YCC */ if (8 == raw_info_t.raw_bpc) { switch (raw_info_t.raw_format) { case 19: pxl_fmt = JXRC_FMT_12bppYCC420; break; case 20: pxl_fmt = JXRC_FMT_16bppYCC422; break; case 21: pxl_fmt = JXRC_FMT_24bppYCC444; break; case 23: alpha_present = 1; pxl_fmt = JXRC_FMT_20bppYCC420Alpha; break; case 24: alpha_present = 1; pxl_fmt = JXRC_FMT_24bppYCC422Alpha; break; case 25: alpha_present = 1; pxl_fmt = JXRC_FMT_32bppYCC444Alpha; break; default: assert(0); break; } } else if (10 == raw_info_t.raw_bpc) { switch (raw_info_t.raw_format) { case 20: pxl_fmt = JXRC_FMT_20bppYCC422; break; case 21: pxl_fmt = JXRC_FMT_30bppYCC444; break; case 24: alpha_present = 1; pxl_fmt = JXRC_FMT_30bppYCC422Alpha; break; case 25: alpha_present = 1; pxl_fmt = JXRC_FMT_40bppYCC444Alpha; break; default: assert(0); break; } } else if (16 == raw_info_t.raw_bpc) { /* 16bpc */ switch (raw_info_t.raw_format) { case 20: pxl_fmt = JXRC_FMT_32bppYCC422; break; case 21: pxl_fmt = JXRC_FMT_48bppYCC444; break; case 22: pxl_fmt = JXRC_FMT_48bppYCC444FixedPoint; break; case 24: alpha_present = 1; pxl_fmt = JXRC_FMT_48bppYCC422Alpha; break; case 25: alpha_present = 1; pxl_fmt = JXRC_FMT_64bppYCC444Alpha; break; case 26: alpha_present = 1; pxl_fmt = JXRC_FMT_64bppYCC444AlphaFixedPoint; break; default: assert(0); break; } } else assert(0); } else if ((raw_info_t.raw_format >= 27) && (raw_info_t.raw_format <= 28)) { /* CMYKDIRECT */ if (8 == raw_info_t.raw_bpc) { switch (raw_info_t.raw_format) { case 27: pxl_fmt = JXRC_FMT_32bppCMYKDIRECT; break; case 28: alpha_present = 1; pxl_fmt = JXRC_FMT_40bppCMYKDIRECTAlpha; break; default: assert(0); break; } } else if (16 == raw_info_t.raw_bpc) { /* 16bpc */ switch (raw_info_t.raw_format) { case 27: pxl_fmt = JXRC_FMT_64bppCMYKDIRECT; break; case 28: alpha_present = 1; pxl_fmt = JXRC_FMT_80bppCMYKDIRECTAlpha; break; default: assert(0); break; } } else assert(0); } else if (raw_info_t.raw_format == 29) { pxl_fmt = JXRC_FMT_24bppBGR; } else if (raw_info_t.raw_format == 30) { pxl_fmt = JXRC_FMT_32bppBGR; } else if (raw_info_t.raw_format == 31) { alpha_present = 1; pxl_fmt = JXRC_FMT_32bppBGRA; } else if (raw_info_t.raw_format == 32) { alpha_present = 1; pxl_fmt = JXRC_FMT_32bppPBGRA; } else if (raw_info_t.raw_format == 33) { alpha_present = 1; pxl_fmt = JXRC_FMT_64bppPRGBA; } else if (raw_info_t.raw_format == 34) { alpha_present = 1; pxl_fmt = JXRC_FMT_128bppPRGBAFloat; } else assert(0); } jxrc_set_pixel_format(container, pxl_fmt); if(alpha_present) { if(alpha_mode == 0) /* No -a option was used at command line, so by default, turn on Separate alpha */ { alpha_mode = 2; /* Separate alpha */ fprintf(stderr, " Setting alpha_mode to 2\n"); } else { fprintf(stderr, " Using alpha_mode = %d\n ", alpha_mode); } } else /* Current output format has no alpha information, so turn off alpha_mode */ { if(alpha_mode != 0) { alpha_mode = 0; fprintf(stderr, " Setting alpha_mode to 0\n"); } } jxrc_set_image_shape(container, wid, hei); jxrc_set_image_band_presence(container, (unsigned) bands_present); /* another call will need to be added for separate alpha */ return 0; } static int main_compress(const char*path) { int rc = 0; jxr_image_t image = NULL; void *input_handle = NULL; jxr_container_t container = NULL; if (path_out == 0) path_out = "out.jxr"; FILE*fd = fopen(path_out, "wb"); if (fd == 0) { perror(path_out); rc = -1; goto exit; } input_handle = open_input_file(path, &raw_info_t, &alpha_mode, &padded_format); char path_primary[2048]; char path_alpha[2048]; int wid, hei, ncomp, bpi; short sf, photometric; int padBytes; get_file_parameters(input_handle, &wid, &hei, &ncomp, &bpi, &sf, &photometric, &padBytes); /* Create the file container and bind the FD to it. */ container = jxr_create_container(); jxrc_start_file(container, fd); /* Start the next ifd entry and load it with information about the image to be processed. */ rc = jxrc_begin_ifd_entry(container); if(rc != 0) goto exit; rc = setup_container_params(container, input_handle); if(rc != 0) goto exit; /* Create a stub image. */ if(alpha_mode == 2) rc = setup_image_params(&image, input_handle, 0, 0); /* Set up parameters for a regular image, fix later after separating primary and alpha */ else rc = setup_image_params(&image, input_handle, alpha_mode, 0); if(rc != 0) goto exit; if(alpha_mode == 2) { /* Open handle to dump primary in tif format */ separate_primary_alpha(image, input_handle, path_out, path_primary, path_alpha, container);//jxrc_image_pixelformat(container, 0)); close_file(input_handle); input_handle = NULL; input_handle = open_input_file(path_primary, &raw_info_t, &alpha_mode, &padded_format); get_file_parameters(input_handle, &wid, &hei, &ncomp, &bpi, &sf, &photometric, &padBytes); rc = setup_image_params(&image, input_handle, alpha_mode, 0); if(rc != 0) goto exit; jxrc_set_separate_alpha_image_plane(container, 1); } else { jxrc_set_separate_alpha_image_plane(container, 0); } jxr_set_pixel_format(image, jxrc_get_pixel_format(container)); /* Close out the current IFD and start with the actual image data to be written. */ jxrc_begin_image_data(container); jxr_set_block_input(image, read_file); jxr_set_user_data(image, input_handle); /* Write image to the destination file. */ rc = jxr_write_image_bitstream(image, fd); if (rc != 0) { goto exit; } /* Finalize IFDs for image. */ jxrc_write_container_post(container); close_file(input_handle); input_handle = NULL; jxr_destroy(image); image = NULL; if(alpha_mode == 2) { input_handle = open_input_file(path_alpha, &raw_info_t, &alpha_mode, &padded_format); get_file_parameters(input_handle, &wid, &hei, &ncomp, &bpi, &sf, &photometric, &padBytes); rc = setup_image_params(&image, input_handle, alpha_mode, 1); jxr_set_pixel_format(image, jxrc_get_pixel_format(container)); if(rc != 0) { goto exit; } jxr_set_block_input(image, read_file); jxr_set_user_data(image, input_handle); /* Write image to the destination file. */ rc = jxr_write_image_bitstream(image, fd); if (rc != 0) { goto exit; } /* Finalize IFDs for alpha plane. */ jxrc_write_container_post_alpha(container); remove(path_alpha); remove(path_primary); } exit: close_file(input_handle); jxr_destroy(image); if(fd) fclose(fd); jxr_destroy_container(container); return rc; } static int decompress_image(FILE *fd, jxr_container_t container, void *output_handle, jxr_image_t *pImage, unsigned char alpha) { int rc, idx; *pImage = jxr_create_input(); jxr_set_block_output(*pImage, write_file); jxr_set_pixel_format(*pImage, jxrc_image_pixelformat(container, 0)); jxr_set_user_data(*pImage, output_handle); jxr_set_PROFILE_IDC(*pImage, profile_idc); jxr_set_LEVEL_IDC(*pImage, level_idc); jxr_set_container_parameters(*pImage, jxrc_image_pixelformat(container, 0), jxrc_image_width(container, 0), jxrc_image_height(container, 0), jxrc_alpha_offset(container, 0), jxrc_image_band_presence(container,0), jxrc_alpha_band_presence(container,0), alpha); for (idx = 0 ; idx < nflags ; idx += 1) { if (strcmp(flags[idx],"SKIP_HP_DATA") == 0) { jxr_flag_SKIP_HP_DATA(*pImage, 1); continue; } if (strcmp(flags[idx],"SKIP_FLEX_DATA") == 0) { jxr_flag_SKIP_FLEX_DATA(*pImage, 1); continue; } } /* Process as an image bitstream. */ rc = jxr_read_image_bitstream(*pImage, fd); if (rc < 0) { switch (rc) { case JXR_EC_BADMAGIC: fprintf(stderr, "No valid magic number. Not an JPEG XR container or bitstream.\n"); break; default: fprintf(stderr, " Error %d reading image bitstream\n", rc); break; } } else rc = jxr_test_LONG_WORD_FLAG(*pImage, long_word_flag_setting); return rc; } #define SAFE_FREE(h) {if(h)free(h); h = NULL;} #define SAFE_CLOSE(h) {if(h)close_file(h); h = NULL;} #define SAFE_JXR_DESTROY(h) {if(h) jxr_destroy(h); h = NULL;} static int main_decompress(const char*path_in) { int rc; int codedImages = 1; unsigned int alphaCodedImagePresent = 0; void *output_handle_primary = NULL; void *output_handle_alpha = NULL; char path_out_primary[2048]; char path_out_alpha[2048]; unsigned long off; jxr_image_t imageAlpha=NULL, image=NULL; if (path_out == 0) path_out = "out.raw"; FILE*fd = fopen(path_in, "rb"); if (fd == 0) { perror(path_in); return -1; } void *output_handle = open_output_file(path_out); jxr_container_t ifile = jxr_create_container(); rc = jxr_read_image_container(ifile, fd); if (rc >= 0) { assert(rc >= 0); # if defined(DETAILED_DEBUG) printf("Detected jxr image container\n"); printf("XXXX Bytes of contained image: %ld\n", jxrc_image_bytecount(ifile, 0)); #endif off = jxrc_image_offset(ifile, 0); #if defined(DETAILED_DEBUG) printf("XXXX Offset of contained image: %ld\n", off); #endif rc = fseek(fd, off, SEEK_SET); assert(rc >= 0); if(jxrc_alpha_offset(ifile, 0)) { alphaCodedImagePresent = 1; # if defined(DETAILED_DEBUG) printf("XXXX Bytes of alpha image: %ld\n", jxrc_alpha_bytecount(ifile, 0)); printf("XXXX Offset of contained image: %ld\n", jxrc_alpha_offset(ifile, 0)); #endif } } else { #if defined(DETAILED_DEBUG) printf("No container found, assuming unwrapped bistream with no alpha coded image\n"); #endif rc = fseek(fd, 0, SEEK_SET); assert(rc >= 0); } /* read optional IFD tags to make certain of conformance */ char * document_name = 0, * image_description = 0, * equipment_make = 0, * equipment_model = 0, * page_name = 0; char * software_name_version = 0, * date_time = 0, * artist_name = 0, * host_computer = 0, * copyright_notice = 0; unsigned char profile, level; unsigned short page_number[2] = {0, 0}, color_space; unsigned long spatial_xfrm, image_type; float width_res, height_res; unsigned char image_band_present, alpha_band_present, buf[4]; rc = jxrc_document_name(ifile, 0, &document_name); rc = jxrc_image_description(ifile, 0, &image_description); rc = jxrc_equipment_make(ifile, 0, &equipment_make); rc = jxrc_equipment_model(ifile, 0, &equipment_model); rc = jxrc_page_name(ifile, 0, &page_name); rc = jxrc_page_number(ifile, 0, page_number); rc = jxrc_software_name_version(ifile, 0, &software_name_version); rc = jxrc_date_time(ifile, 0, &date_time); rc = jxrc_artist_name(ifile, 0, &artist_name); rc = jxrc_host_computer(ifile, 0, &host_computer); rc = jxrc_copyright_notice(ifile, 0, ©right_notice); color_space = jxrc_color_space(ifile, 0); spatial_xfrm = jxrc_spatial_xfrm_primary(ifile, 0); image_type = jxrc_image_type(ifile, 0); rc = jxrc_ptm_color_info(ifile, 0, buf); rc = jxrc_profile_level_container(ifile, 0, &profile, &level); if (rc < 0) { profile = 111; level = 255; } width_res = jxrc_width_resolution(ifile, 0); height_res = jxrc_height_resolution(ifile, 0); image_band_present = jxrc_image_band_presence(ifile, 0); alpha_band_present = jxrc_alpha_band_presence(ifile, 0); rc = jxrc_padding_data(ifile, 0); if(alphaCodedImagePresent) { /* Open handle to dump decoded primary in raw format */ strcpy(path_out_primary, path_out); strcat(path_out_primary, "_primary.raw"); output_handle_primary = open_output_file(path_out_primary); } else { output_handle_primary = open_output_file(path_out); } /*Decode image */ rc = decompress_image(fd, ifile, output_handle_primary, &image, 0); SAFE_CLOSE(output_handle_primary); if(rc < 0) goto exit; if(!alphaCodedImagePresent) goto exit; /* Open handle to dump decoded alpha in raw format*/ strcpy(path_out_alpha, path_out); strcat(path_out_alpha, "_alpha.raw"); output_handle_alpha = open_output_file(path_out_alpha); /*Seek to alpha offset */ off = jxrc_alpha_offset(ifile, 0); rc = fseek(fd, off, SEEK_SET); assert(rc >= 0); /* Decode alpha */ rc = decompress_image(fd, ifile, output_handle_alpha, &imageAlpha, 1); SAFE_CLOSE(output_handle_alpha); if(rc < 0) goto exit; /* For YCC and CMYKDirect formats, concatenate alpha and primary images */ /* For other output pixel formats, interleave alphad and primary images */ { output_handle = open_output_file(path_out); FILE *fpPrimary = fopen(path_out_primary,"rb"); assert(fpPrimary); FILE *fpAlpha = fopen(path_out_alpha,"rb"); assert(fpAlpha); jxr_set_user_data(image, output_handle); write_file_combine_primary_alpha(image, fpPrimary, fpAlpha); fclose(fpPrimary); fclose(fpAlpha); remove(path_out_primary); remove(path_out_alpha); SAFE_CLOSE(output_handle); } exit: SAFE_FREE(document_name); SAFE_FREE(image_description); SAFE_FREE(equipment_make); SAFE_FREE(equipment_model); SAFE_FREE(page_name); SAFE_FREE(software_name_version); SAFE_FREE(date_time); SAFE_FREE(artist_name); SAFE_FREE(host_computer); SAFE_FREE(copyright_notice); SAFE_JXR_DESTROY(image); SAFE_JXR_DESTROY(imageAlpha); fclose(fd); return 0; } /* * $Log: jpegxr.c,v $ * Revision 1.39 2009/05/29 12:00:00 microsoft * Reference Software v1.6 updates. * * Revision 1.38 2009/04/13 12:00:00 microsoft * Reference Software v1.5 updates. * * Revision 1.37 2008/05/25 09:40:46 thor * Added a -d command line option to enable derived quantization. * * Revision 1.36 2008/05/13 13:47:11 thor * Some experiments with a smarter selection for the quantization size, * does not yet compile. * * Revision 1.35 2008-05-09 19:57:48 thor * Reformatted for unix LF. * * Revision 1.34 2008-04-15 14:28:12 thor * Start of the repository for the jpegxr reference software. * * Revision 1.33 2008/03/21 18:05:53 steve * Proper CMYK formatting on input. * * Revision 1.32 2008/03/18 18:36:56 steve * Support compress of CMYK images. * * Revision 1.31 2008/03/11 22:12:49 steve * Encode YUV422 through DC. * * Revision 1.30 2008/03/06 02:05:48 steve * Distributed quantization * * Revision 1.29 2008/03/05 06:58:10 gus * *** empty log message *** * * Revision 1.28 2008/03/05 04:04:30 steve * Clarify constraints on USE_DC_QP in image plane header. * * Revision 1.27 2008/03/05 01:27:15 steve * QP_UNIFORM may use USE_DC_LP optionally. * * Revision 1.26 2008/03/04 23:01:28 steve * Cleanup QP API in preparation for distributed QP * * Revision 1.25 2008/03/02 19:56:27 steve * Infrastructure to read write BD16 files. * * Revision 1.24 2008/03/01 02:46:09 steve * Add support for JXR container. * * Revision 1.23 2008/02/29 01:03:31 steve * MSC doesnt have strcasecmp. Use stricmp instead. * * Revision 1.22 2008/02/29 00:57:59 steve * VisualStudio files * * Revision 1.21 2008/02/26 23:52:44 steve * Remove ident for MS compilers. * * Revision 1.20 2008/02/01 22:49:53 steve * Handle compress of YUV444 color DCONLY * * Revision 1.19 2008/01/19 02:30:46 rick * Re-implement and extend file interface. * * Revision 1.18 2008/01/08 01:06:20 steve * Add first pass overlap filtering. * * Revision 1.17 2008/01/06 01:29:28 steve * Add support for TRIM_FLEXBITS in compression. * * Revision 1.16 2008/01/04 17:07:35 steve * API interface for setting QP values. * * Revision 1.15 2007/12/06 00:24:25 steve * Zero fill strip buffer, so that right pad is zeros. * * Revision 1.14 2007/12/04 22:06:10 steve * Infrastructure for encoding LP. * * Revision 1.13 2007/11/30 01:57:00 steve * Handle comments in PNM files. * * Revision 1.12 2007/11/30 01:50:58 steve * Compression of DCONLY GRAY. * * Revision 1.11 2007/11/26 01:47:15 steve * Add copyright notices per MS request. * * Revision 1.10 2007/11/08 02:52:32 steve * Some progress in some encoding infrastructure. * * Revision 1.9 2007/11/07 18:11:02 steve * Useful error message for bad magic numbers. * * Revision 1.8 2007/10/22 23:08:55 steve * Protect prints with DETAILED_DEBUG. * * Revision 1.7 2007/09/18 17:02:49 steve * Slight PNM header format change. * * Revision 1.6 2007/09/08 01:01:43 steve * YUV444 color parses properly. * * Revision 1.5 2007/08/03 22:49:10 steve * Do not write out the line padding. * * Revision 1.4 2007/07/30 23:09:57 steve * Interleave FLEXBITS within HP block. * * Revision 1.3 2007/07/21 00:25:48 steve * snapshot 2007 07 20 * * Revision 1.2 2007/06/07 18:53:06 steve * Parse HP coeffs that are all 0. * * Revision 1.1 2007/06/06 17:19:12 steve * Introduce to CVS. * */