• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright (C)2009-2019 D. R. Commander.  All Rights Reserved.
3  *
4  * Redistribution and use in source and binary forms, with or without
5  * modification, are permitted provided that the following conditions are met:
6  *
7  * - Redistributions of source code must retain the above copyright notice,
8  *   this list of conditions and the following disclaimer.
9  * - Redistributions in binary form must reproduce the above copyright notice,
10  *   this list of conditions and the following disclaimer in the documentation
11  *   and/or other materials provided with the distribution.
12  * - Neither the name of the libjpeg-turbo Project nor the names of its
13  *   contributors may be used to endorse or promote products derived from this
14  *   software without specific prior written permission.
15  *
16  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS",
17  * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
18  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
19  * ARE DISCLAIMED.  IN NO EVENT SHALL THE COPYRIGHT HOLDERS OR CONTRIBUTORS BE
20  * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
21  * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
22  * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
23  * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
24  * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
25  * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
26  * POSSIBILITY OF SUCH DAMAGE.
27  */
28 
29 /* TurboJPEG/LJT:  this implements the TurboJPEG API using libjpeg or
30    libjpeg-turbo */
31 
32 #include <stdio.h>
33 #include <stdlib.h>
34 #include <ctype.h>
35 #include <jinclude.h>
36 #define JPEG_INTERNALS
37 #include <jpeglib.h>
38 #include <jerror.h>
39 #include <setjmp.h>
40 #include <errno.h>
41 #include "./turbojpeg.h"
42 #include "./tjutil.h"
43 #include "transupp.h"
44 #include "./jpegcomp.h"
45 #include "./cdjpeg.h"
46 
47 extern void jpeg_mem_dest_tj(j_compress_ptr, unsigned char **, unsigned long *,
48                              boolean);
49 extern void jpeg_mem_src_tj(j_decompress_ptr, const unsigned char *,
50                             unsigned long);
51 
52 #define PAD(v, p)  ((v + (p) - 1) & (~((p) - 1)))
53 #define IS_POW2(x)  (((x) & (x - 1)) == 0)
54 
55 
56 /* Error handling (based on example in example.txt) */
57 
58 static char errStr[JMSG_LENGTH_MAX] = "No error";
59 
60 struct my_error_mgr {
61   struct jpeg_error_mgr pub;
62   jmp_buf setjmp_buffer;
63   void (*emit_message) (j_common_ptr, int);
64   boolean warning, stopOnWarning;
65 };
66 typedef struct my_error_mgr *my_error_ptr;
67 
68 #define JMESSAGE(code, string)  string,
69 static const char *turbojpeg_message_table[] = {
70 #include "cderror.h"
71   NULL
72 };
73 
my_error_exit(j_common_ptr cinfo)74 static void my_error_exit(j_common_ptr cinfo)
75 {
76   my_error_ptr myerr = (my_error_ptr)cinfo->err;
77 
78   (*cinfo->err->output_message) (cinfo);
79   longjmp(myerr->setjmp_buffer, 1);
80 }
81 
82 /* Based on output_message() in jerror.c */
83 
my_output_message(j_common_ptr cinfo)84 static void my_output_message(j_common_ptr cinfo)
85 {
86   (*cinfo->err->format_message) (cinfo, errStr);
87 }
88 
my_emit_message(j_common_ptr cinfo,int msg_level)89 static void my_emit_message(j_common_ptr cinfo, int msg_level)
90 {
91   my_error_ptr myerr = (my_error_ptr)cinfo->err;
92 
93   myerr->emit_message(cinfo, msg_level);
94   if (msg_level < 0) {
95     myerr->warning = TRUE;
96     if (myerr->stopOnWarning) longjmp(myerr->setjmp_buffer, 1);
97   }
98 }
99 
100 
101 /* Global structures, macros, etc. */
102 
103 enum { COMPRESS = 1, DECOMPRESS = 2 };
104 
105 typedef struct _tjinstance {
106   struct jpeg_compress_struct cinfo;
107   struct jpeg_decompress_struct dinfo;
108   struct my_error_mgr jerr;
109   int init, headerRead;
110   char errStr[JMSG_LENGTH_MAX];
111   boolean isInstanceError;
112 } tjinstance;
113 
114 static const int pixelsize[TJ_NUMSAMP] = { 3, 3, 3, 1, 3, 3 };
115 
116 static const JXFORM_CODE xformtypes[TJ_NUMXOP] = {
117   JXFORM_NONE, JXFORM_FLIP_H, JXFORM_FLIP_V, JXFORM_TRANSPOSE,
118   JXFORM_TRANSVERSE, JXFORM_ROT_90, JXFORM_ROT_180, JXFORM_ROT_270
119 };
120 
121 #define NUMSF  16
122 static const tjscalingfactor sf[NUMSF] = {
123   { 2, 1 },
124   { 15, 8 },
125   { 7, 4 },
126   { 13, 8 },
127   { 3, 2 },
128   { 11, 8 },
129   { 5, 4 },
130   { 9, 8 },
131   { 1, 1 },
132   { 7, 8 },
133   { 3, 4 },
134   { 5, 8 },
135   { 1, 2 },
136   { 3, 8 },
137   { 1, 4 },
138   { 1, 8 }
139 };
140 
141 static J_COLOR_SPACE pf2cs[TJ_NUMPF] = {
142   JCS_EXT_RGB, JCS_EXT_BGR, JCS_EXT_RGBX, JCS_EXT_BGRX, JCS_EXT_XBGR,
143   JCS_EXT_XRGB, JCS_GRAYSCALE, JCS_EXT_RGBA, JCS_EXT_BGRA, JCS_EXT_ABGR,
144   JCS_EXT_ARGB, JCS_CMYK
145 };
146 
147 static int cs2pf[JPEG_NUMCS] = {
148   TJPF_UNKNOWN, TJPF_GRAY,
149 #if RGB_RED == 0 && RGB_GREEN == 1 && RGB_BLUE == 2 && RGB_PIXELSIZE == 3
150   TJPF_RGB,
151 #elif RGB_RED == 2 && RGB_GREEN == 1 && RGB_BLUE == 0 && RGB_PIXELSIZE == 3
152   TJPF_BGR,
153 #elif RGB_RED == 0 && RGB_GREEN == 1 && RGB_BLUE == 2 && RGB_PIXELSIZE == 4
154   TJPF_RGBX,
155 #elif RGB_RED == 2 && RGB_GREEN == 1 && RGB_BLUE == 0 && RGB_PIXELSIZE == 4
156   TJPF_BGRX,
157 #elif RGB_RED == 3 && RGB_GREEN == 2 && RGB_BLUE == 1 && RGB_PIXELSIZE == 4
158   TJPF_XBGR,
159 #elif RGB_RED == 1 && RGB_GREEN == 2 && RGB_BLUE == 3 && RGB_PIXELSIZE == 4
160   TJPF_XRGB,
161 #endif
162   TJPF_UNKNOWN, TJPF_CMYK, TJPF_UNKNOWN, TJPF_RGB, TJPF_RGBX, TJPF_BGR,
163   TJPF_BGRX, TJPF_XBGR, TJPF_XRGB, TJPF_RGBA, TJPF_BGRA, TJPF_ABGR, TJPF_ARGB,
164   TJPF_UNKNOWN
165 };
166 
167 #define THROWG(m) { \
168   snprintf(errStr, JMSG_LENGTH_MAX, "%s", m); \
169   retval = -1;  goto bailout; \
170 }
171 #define THROW_UNIX(m) { \
172   snprintf(errStr, JMSG_LENGTH_MAX, "%s\n%s", m, strerror(errno)); \
173   retval = -1;  goto bailout; \
174 }
175 #define THROW(m) { \
176   snprintf(this->errStr, JMSG_LENGTH_MAX, "%s", m); \
177   this->isInstanceError = TRUE;  THROWG(m) \
178 }
179 
180 #define GET_INSTANCE(handle) \
181   tjinstance *this = (tjinstance *)handle; \
182   j_compress_ptr cinfo = NULL; \
183   j_decompress_ptr dinfo = NULL; \
184   \
185   if (!this) { \
186     snprintf(errStr, JMSG_LENGTH_MAX, "Invalid handle"); \
187     return -1; \
188   } \
189   cinfo = &this->cinfo;  dinfo = &this->dinfo; \
190   this->jerr.warning = FALSE; \
191   this->isInstanceError = FALSE;
192 
193 #define GET_CINSTANCE(handle) \
194   tjinstance *this = (tjinstance *)handle; \
195   j_compress_ptr cinfo = NULL; \
196   \
197   if (!this) { \
198     snprintf(errStr, JMSG_LENGTH_MAX, "Invalid handle"); \
199     return -1; \
200   } \
201   cinfo = &this->cinfo; \
202   this->jerr.warning = FALSE; \
203   this->isInstanceError = FALSE;
204 
205 #define GET_DINSTANCE(handle) \
206   tjinstance *this = (tjinstance *)handle; \
207   j_decompress_ptr dinfo = NULL; \
208   \
209   if (!this) { \
210     snprintf(errStr, JMSG_LENGTH_MAX, "Invalid handle"); \
211     return -1; \
212   } \
213   dinfo = &this->dinfo; \
214   this->jerr.warning = FALSE; \
215   this->isInstanceError = FALSE;
216 
getPixelFormat(int pixelSize,int flags)217 static int getPixelFormat(int pixelSize, int flags)
218 {
219   if (pixelSize == 1) return TJPF_GRAY;
220   if (pixelSize == 3) {
221     if (flags & TJ_BGR) return TJPF_BGR;
222     else return TJPF_RGB;
223   }
224   if (pixelSize == 4) {
225     if (flags & TJ_ALPHAFIRST) {
226       if (flags & TJ_BGR) return TJPF_XBGR;
227       else return TJPF_XRGB;
228     } else {
229       if (flags & TJ_BGR) return TJPF_BGRX;
230       else return TJPF_RGBX;
231     }
232   }
233   return -1;
234 }
235 
setCompDefaults(struct jpeg_compress_struct * cinfo,int pixelFormat,int subsamp,int jpegQual,int flags)236 static int setCompDefaults(struct jpeg_compress_struct *cinfo, int pixelFormat,
237                            int subsamp, int jpegQual, int flags)
238 {
239   int retval = 0;
240 #ifndef NO_GETENV
241   char *env = NULL;
242 #endif
243 
244   cinfo->in_color_space = pf2cs[pixelFormat];
245   cinfo->input_components = tjPixelSize[pixelFormat];
246   jpeg_set_defaults(cinfo);
247 
248 #ifndef NO_GETENV
249   if ((env = getenv("TJ_OPTIMIZE")) != NULL && strlen(env) > 0 &&
250       !strcmp(env, "1"))
251     cinfo->optimize_coding = TRUE;
252   if ((env = getenv("TJ_ARITHMETIC")) != NULL && strlen(env) > 0 &&
253       !strcmp(env, "1"))
254     cinfo->arith_code = TRUE;
255   if ((env = getenv("TJ_RESTART")) != NULL && strlen(env) > 0) {
256     int temp = -1;
257     char tempc = 0;
258 
259     if (sscanf(env, "%d%c", &temp, &tempc) >= 1 && temp >= 0 &&
260         temp <= 65535) {
261       if (toupper(tempc) == 'B') {
262         cinfo->restart_interval = temp;
263         cinfo->restart_in_rows = 0;
264       } else
265         cinfo->restart_in_rows = temp;
266     }
267   }
268 #endif
269 
270   if (jpegQual >= 0) {
271     jpeg_set_quality(cinfo, jpegQual, TRUE);
272     if (jpegQual >= 96 || flags & TJFLAG_ACCURATEDCT)
273       cinfo->dct_method = JDCT_ISLOW;
274     else
275       cinfo->dct_method = JDCT_FASTEST;
276   }
277   if (subsamp == TJSAMP_GRAY)
278     jpeg_set_colorspace(cinfo, JCS_GRAYSCALE);
279   else if (pixelFormat == TJPF_CMYK)
280     jpeg_set_colorspace(cinfo, JCS_YCCK);
281   else
282     jpeg_set_colorspace(cinfo, JCS_YCbCr);
283 
284   if (flags & TJFLAG_PROGRESSIVE)
285     jpeg_simple_progression(cinfo);
286 #ifndef NO_GETENV
287   else if ((env = getenv("TJ_PROGRESSIVE")) != NULL && strlen(env) > 0 &&
288            !strcmp(env, "1"))
289     jpeg_simple_progression(cinfo);
290 #endif
291 
292   cinfo->comp_info[0].h_samp_factor = tjMCUWidth[subsamp] / 8;
293   cinfo->comp_info[1].h_samp_factor = 1;
294   cinfo->comp_info[2].h_samp_factor = 1;
295   if (cinfo->num_components > 3)
296     cinfo->comp_info[3].h_samp_factor = tjMCUWidth[subsamp] / 8;
297   cinfo->comp_info[0].v_samp_factor = tjMCUHeight[subsamp] / 8;
298   cinfo->comp_info[1].v_samp_factor = 1;
299   cinfo->comp_info[2].v_samp_factor = 1;
300   if (cinfo->num_components > 3)
301     cinfo->comp_info[3].v_samp_factor = tjMCUHeight[subsamp] / 8;
302 
303   return retval;
304 }
305 
306 
getSubsamp(j_decompress_ptr dinfo)307 static int getSubsamp(j_decompress_ptr dinfo)
308 {
309   int retval = -1, i, k;
310 
311   /* The sampling factors actually have no meaning with grayscale JPEG files,
312      and in fact it's possible to generate grayscale JPEGs with sampling
313      factors > 1 (even though those sampling factors are ignored by the
314      decompressor.)  Thus, we need to treat grayscale as a special case. */
315   if (dinfo->num_components == 1 && dinfo->jpeg_color_space == JCS_GRAYSCALE)
316     return TJSAMP_GRAY;
317 
318   for (i = 0; i < NUMSUBOPT; i++) {
319     if (dinfo->num_components == pixelsize[i] ||
320         ((dinfo->jpeg_color_space == JCS_YCCK ||
321           dinfo->jpeg_color_space == JCS_CMYK) &&
322          pixelsize[i] == 3 && dinfo->num_components == 4)) {
323       if (dinfo->comp_info[0].h_samp_factor == tjMCUWidth[i] / 8 &&
324           dinfo->comp_info[0].v_samp_factor == tjMCUHeight[i] / 8) {
325         int match = 0;
326 
327         for (k = 1; k < dinfo->num_components; k++) {
328           int href = 1, vref = 1;
329 
330           if ((dinfo->jpeg_color_space == JCS_YCCK ||
331                dinfo->jpeg_color_space == JCS_CMYK) && k == 3) {
332             href = tjMCUWidth[i] / 8;  vref = tjMCUHeight[i] / 8;
333           }
334           if (dinfo->comp_info[k].h_samp_factor == href &&
335               dinfo->comp_info[k].v_samp_factor == vref)
336             match++;
337         }
338         if (match == dinfo->num_components - 1) {
339           retval = i;  break;
340         }
341       }
342       /* Handle 4:2:2 and 4:4:0 images whose sampling factors are specified
343          in non-standard ways. */
344       if (dinfo->comp_info[0].h_samp_factor == 2 &&
345           dinfo->comp_info[0].v_samp_factor == 2 &&
346           (i == TJSAMP_422 || i == TJSAMP_440)) {
347         int match = 0;
348 
349         for (k = 1; k < dinfo->num_components; k++) {
350           int href = tjMCUHeight[i] / 8, vref = tjMCUWidth[i] / 8;
351 
352           if ((dinfo->jpeg_color_space == JCS_YCCK ||
353                dinfo->jpeg_color_space == JCS_CMYK) && k == 3) {
354             href = vref = 2;
355           }
356           if (dinfo->comp_info[k].h_samp_factor == href &&
357               dinfo->comp_info[k].v_samp_factor == vref)
358             match++;
359         }
360         if (match == dinfo->num_components - 1) {
361           retval = i;  break;
362         }
363       }
364       /* Handle 4:4:4 images whose sampling factors are specified in
365          non-standard ways. */
366       if (dinfo->comp_info[0].h_samp_factor *
367           dinfo->comp_info[0].v_samp_factor <=
368           D_MAX_BLOCKS_IN_MCU / pixelsize[i] && i == TJSAMP_444) {
369         int match = 0;
370         for (k = 1; k < dinfo->num_components; k++) {
371           if (dinfo->comp_info[i].h_samp_factor ==
372               dinfo->comp_info[0].h_samp_factor &&
373               dinfo->comp_info[i].v_samp_factor ==
374               dinfo->comp_info[0].v_samp_factor)
375             match++;
376           if (match == dinfo->num_components - 1) {
377             retval = i;  break;
378           }
379         }
380       }
381     }
382   }
383   return retval;
384 }
385 
386 
387 /* General API functions */
388 
tjGetErrorStr2(tjhandle handle)389 DLLEXPORT char *tjGetErrorStr2(tjhandle handle)
390 {
391   tjinstance *this = (tjinstance *)handle;
392 
393   if (this && this->isInstanceError) {
394     this->isInstanceError = FALSE;
395     return this->errStr;
396   } else
397     return errStr;
398 }
399 
400 
tjGetErrorStr(void)401 DLLEXPORT char *tjGetErrorStr(void)
402 {
403   return errStr;
404 }
405 
406 
tjGetErrorCode(tjhandle handle)407 DLLEXPORT int tjGetErrorCode(tjhandle handle)
408 {
409   tjinstance *this = (tjinstance *)handle;
410 
411   if (this && this->jerr.warning) return TJERR_WARNING;
412   else return TJERR_FATAL;
413 }
414 
415 
tjDestroy(tjhandle handle)416 DLLEXPORT int tjDestroy(tjhandle handle)
417 {
418   GET_INSTANCE(handle);
419 
420   if (setjmp(this->jerr.setjmp_buffer)) return -1;
421   if (this->init & COMPRESS) jpeg_destroy_compress(cinfo);
422   if (this->init & DECOMPRESS) jpeg_destroy_decompress(dinfo);
423   free(this);
424   return 0;
425 }
426 
427 
428 /* These are exposed mainly because Windows can't malloc() and free() across
429    DLL boundaries except when the CRT DLL is used, and we don't use the CRT DLL
430    with turbojpeg.dll for compatibility reasons.  However, these functions
431    can potentially be used for other purposes by different implementations. */
432 
tjFree(unsigned char * buf)433 DLLEXPORT void tjFree(unsigned char *buf)
434 {
435   if (buf) free(buf);
436 }
437 
438 
tjAlloc(int bytes)439 DLLEXPORT unsigned char *tjAlloc(int bytes)
440 {
441   return (unsigned char *)malloc(bytes);
442 }
443 
444 
445 /* Compressor  */
446 
_tjInitCompress(tjinstance * this)447 static tjhandle _tjInitCompress(tjinstance *this)
448 {
449   static unsigned char buffer[1];
450   unsigned char *buf = buffer;
451   unsigned long size = 1;
452 
453   /* This is also straight out of example.txt */
454   this->cinfo.err = jpeg_std_error(&this->jerr.pub);
455   this->jerr.pub.error_exit = my_error_exit;
456   this->jerr.pub.output_message = my_output_message;
457   this->jerr.emit_message = this->jerr.pub.emit_message;
458   this->jerr.pub.emit_message = my_emit_message;
459   this->jerr.pub.addon_message_table = turbojpeg_message_table;
460   this->jerr.pub.first_addon_message = JMSG_FIRSTADDONCODE;
461   this->jerr.pub.last_addon_message = JMSG_LASTADDONCODE;
462 
463   if (setjmp(this->jerr.setjmp_buffer)) {
464     /* If we get here, the JPEG code has signaled an error. */
465     if (this) free(this);
466     return NULL;
467   }
468 
469   jpeg_create_compress(&this->cinfo);
470   /* Make an initial call so it will create the destination manager */
471   jpeg_mem_dest_tj(&this->cinfo, &buf, &size, 0);
472 
473   this->init |= COMPRESS;
474   return (tjhandle)this;
475 }
476 
tjInitCompress(void)477 DLLEXPORT tjhandle tjInitCompress(void)
478 {
479   tjinstance *this = NULL;
480 
481   if ((this = (tjinstance *)malloc(sizeof(tjinstance))) == NULL) {
482     snprintf(errStr, JMSG_LENGTH_MAX,
483              "tjInitCompress(): Memory allocation failure");
484     return NULL;
485   }
486   MEMZERO(this, sizeof(tjinstance));
487   snprintf(this->errStr, JMSG_LENGTH_MAX, "No error");
488   return _tjInitCompress(this);
489 }
490 
491 
tjBufSize(int width,int height,int jpegSubsamp)492 DLLEXPORT unsigned long tjBufSize(int width, int height, int jpegSubsamp)
493 {
494   unsigned long long retval = 0;
495   int mcuw, mcuh, chromasf;
496 
497   if (width < 1 || height < 1 || jpegSubsamp < 0 || jpegSubsamp >= NUMSUBOPT)
498     THROWG("tjBufSize(): Invalid argument");
499 
500   /* This allows for rare corner cases in which a JPEG image can actually be
501      larger than the uncompressed input (we wouldn't mention it if it hadn't
502      happened before.) */
503   mcuw = tjMCUWidth[jpegSubsamp];
504   mcuh = tjMCUHeight[jpegSubsamp];
505   chromasf = jpegSubsamp == TJSAMP_GRAY ? 0 : 4 * 64 / (mcuw * mcuh);
506   retval = PAD(width, mcuw) * PAD(height, mcuh) * (2ULL + chromasf) + 2048ULL;
507   if (retval > (unsigned long long)((unsigned long)-1))
508     THROWG("tjBufSize(): Image is too large");
509 
510 bailout:
511   return (unsigned long)retval;
512 }
513 
TJBUFSIZE(int width,int height)514 DLLEXPORT unsigned long TJBUFSIZE(int width, int height)
515 {
516   unsigned long long retval = 0;
517 
518   if (width < 1 || height < 1)
519     THROWG("TJBUFSIZE(): Invalid argument");
520 
521   /* This allows for rare corner cases in which a JPEG image can actually be
522      larger than the uncompressed input (we wouldn't mention it if it hadn't
523      happened before.) */
524   retval = PAD(width, 16) * PAD(height, 16) * 6ULL + 2048ULL;
525   if (retval > (unsigned long long)((unsigned long)-1))
526     THROWG("TJBUFSIZE(): Image is too large");
527 
528 bailout:
529   return (unsigned long)retval;
530 }
531 
532 
tjBufSizeYUV2(int width,int pad,int height,int subsamp)533 DLLEXPORT unsigned long tjBufSizeYUV2(int width, int pad, int height,
534                                       int subsamp)
535 {
536   unsigned long long retval = 0;
537   int nc, i;
538 
539   if (subsamp < 0 || subsamp >= NUMSUBOPT)
540     THROWG("tjBufSizeYUV2(): Invalid argument");
541 
542   nc = (subsamp == TJSAMP_GRAY ? 1 : 3);
543   for (i = 0; i < nc; i++) {
544     int pw = tjPlaneWidth(i, width, subsamp);
545     int stride = PAD(pw, pad);
546     int ph = tjPlaneHeight(i, height, subsamp);
547 
548     if (pw < 0 || ph < 0) return -1;
549     else retval += (unsigned long long)stride * ph;
550   }
551   if (retval > (unsigned long long)((unsigned long)-1))
552     THROWG("tjBufSizeYUV2(): Image is too large");
553 
554 bailout:
555   return (unsigned long)retval;
556 }
557 
tjBufSizeYUV(int width,int height,int subsamp)558 DLLEXPORT unsigned long tjBufSizeYUV(int width, int height, int subsamp)
559 {
560   return tjBufSizeYUV2(width, 4, height, subsamp);
561 }
562 
TJBUFSIZEYUV(int width,int height,int subsamp)563 DLLEXPORT unsigned long TJBUFSIZEYUV(int width, int height, int subsamp)
564 {
565   return tjBufSizeYUV(width, height, subsamp);
566 }
567 
568 
tjPlaneWidth(int componentID,int width,int subsamp)569 DLLEXPORT int tjPlaneWidth(int componentID, int width, int subsamp)
570 {
571   int pw, nc, retval = 0;
572 
573   if (width < 1 || subsamp < 0 || subsamp >= TJ_NUMSAMP)
574     THROWG("tjPlaneWidth(): Invalid argument");
575   nc = (subsamp == TJSAMP_GRAY ? 1 : 3);
576   if (componentID < 0 || componentID >= nc)
577     THROWG("tjPlaneWidth(): Invalid argument");
578 
579   pw = PAD(width, tjMCUWidth[subsamp] / 8);
580   if (componentID == 0)
581     retval = pw;
582   else
583     retval = pw * 8 / tjMCUWidth[subsamp];
584 
585 bailout:
586   return retval;
587 }
588 
589 
tjPlaneHeight(int componentID,int height,int subsamp)590 DLLEXPORT int tjPlaneHeight(int componentID, int height, int subsamp)
591 {
592   int ph, nc, retval = 0;
593 
594   if (height < 1 || subsamp < 0 || subsamp >= TJ_NUMSAMP)
595     THROWG("tjPlaneHeight(): Invalid argument");
596   nc = (subsamp == TJSAMP_GRAY ? 1 : 3);
597   if (componentID < 0 || componentID >= nc)
598     THROWG("tjPlaneHeight(): Invalid argument");
599 
600   ph = PAD(height, tjMCUHeight[subsamp] / 8);
601   if (componentID == 0)
602     retval = ph;
603   else
604     retval = ph * 8 / tjMCUHeight[subsamp];
605 
606 bailout:
607   return retval;
608 }
609 
610 
tjPlaneSizeYUV(int componentID,int width,int stride,int height,int subsamp)611 DLLEXPORT unsigned long tjPlaneSizeYUV(int componentID, int width, int stride,
612                                        int height, int subsamp)
613 {
614   unsigned long long retval = 0;
615   int pw, ph;
616 
617   if (width < 1 || height < 1 || subsamp < 0 || subsamp >= NUMSUBOPT)
618     THROWG("tjPlaneSizeYUV(): Invalid argument");
619 
620   pw = tjPlaneWidth(componentID, width, subsamp);
621   ph = tjPlaneHeight(componentID, height, subsamp);
622   if (pw < 0 || ph < 0) return -1;
623 
624   if (stride == 0) stride = pw;
625   else stride = abs(stride);
626 
627   retval = (unsigned long long)stride * (ph - 1) + pw;
628   if (retval > (unsigned long long)((unsigned long)-1))
629     THROWG("tjPlaneSizeYUV(): Image is too large");
630 
631 bailout:
632   return (unsigned long)retval;
633 }
634 
635 
tjCompress2(tjhandle handle,const unsigned char * srcBuf,int width,int pitch,int height,int pixelFormat,unsigned char ** jpegBuf,unsigned long * jpegSize,int jpegSubsamp,int jpegQual,int flags)636 DLLEXPORT int tjCompress2(tjhandle handle, const unsigned char *srcBuf,
637                           int width, int pitch, int height, int pixelFormat,
638                           unsigned char **jpegBuf, unsigned long *jpegSize,
639                           int jpegSubsamp, int jpegQual, int flags)
640 {
641   int i, retval = 0, alloc = 1;
642   JSAMPROW *row_pointer = NULL;
643 
644   GET_CINSTANCE(handle)
645   this->jerr.stopOnWarning = (flags & TJFLAG_STOPONWARNING) ? TRUE : FALSE;
646   if ((this->init & COMPRESS) == 0)
647     THROW("tjCompress2(): Instance has not been initialized for compression");
648 
649   if (srcBuf == NULL || width <= 0 || pitch < 0 || height <= 0 ||
650       pixelFormat < 0 || pixelFormat >= TJ_NUMPF || jpegBuf == NULL ||
651       jpegSize == NULL || jpegSubsamp < 0 || jpegSubsamp >= NUMSUBOPT ||
652       jpegQual < 0 || jpegQual > 100)
653     THROW("tjCompress2(): Invalid argument");
654 
655   if (pitch == 0) pitch = width * tjPixelSize[pixelFormat];
656 
657   if ((row_pointer = (JSAMPROW *)malloc(sizeof(JSAMPROW) * height)) == NULL)
658     THROW("tjCompress2(): Memory allocation failure");
659 
660   if (setjmp(this->jerr.setjmp_buffer)) {
661     /* If we get here, the JPEG code has signaled an error. */
662     retval = -1;  goto bailout;
663   }
664 
665   cinfo->image_width = width;
666   cinfo->image_height = height;
667 
668 #ifndef NO_PUTENV
669   if (flags & TJFLAG_FORCEMMX) putenv("JSIMD_FORCEMMX=1");
670   else if (flags & TJFLAG_FORCESSE) putenv("JSIMD_FORCESSE=1");
671   else if (flags & TJFLAG_FORCESSE2) putenv("JSIMD_FORCESSE2=1");
672 #endif
673 
674   if (flags & TJFLAG_NOREALLOC) {
675     alloc = 0;  *jpegSize = tjBufSize(width, height, jpegSubsamp);
676   }
677   jpeg_mem_dest_tj(cinfo, jpegBuf, jpegSize, alloc);
678   if (setCompDefaults(cinfo, pixelFormat, jpegSubsamp, jpegQual, flags) == -1)
679     return -1;
680 
681   jpeg_start_compress(cinfo, TRUE);
682   for (i = 0; i < height; i++) {
683     if (flags & TJFLAG_BOTTOMUP)
684       row_pointer[i] = (JSAMPROW)&srcBuf[(height - i - 1) * (size_t)pitch];
685     else
686       row_pointer[i] = (JSAMPROW)&srcBuf[i * (size_t)pitch];
687   }
688   while (cinfo->next_scanline < cinfo->image_height)
689     jpeg_write_scanlines(cinfo, &row_pointer[cinfo->next_scanline],
690                          cinfo->image_height - cinfo->next_scanline);
691   jpeg_finish_compress(cinfo);
692 
693 bailout:
694   if (cinfo->global_state > CSTATE_START) jpeg_abort_compress(cinfo);
695   if (row_pointer) free(row_pointer);
696   if (this->jerr.warning) retval = -1;
697   this->jerr.stopOnWarning = FALSE;
698   return retval;
699 }
700 
tjCompress(tjhandle handle,unsigned char * srcBuf,int width,int pitch,int height,int pixelSize,unsigned char * jpegBuf,unsigned long * jpegSize,int jpegSubsamp,int jpegQual,int flags)701 DLLEXPORT int tjCompress(tjhandle handle, unsigned char *srcBuf, int width,
702                          int pitch, int height, int pixelSize,
703                          unsigned char *jpegBuf, unsigned long *jpegSize,
704                          int jpegSubsamp, int jpegQual, int flags)
705 {
706   int retval = 0;
707   unsigned long size;
708 
709   if (flags & TJ_YUV) {
710     size = tjBufSizeYUV(width, height, jpegSubsamp);
711     retval = tjEncodeYUV2(handle, srcBuf, width, pitch, height,
712                           getPixelFormat(pixelSize, flags), jpegBuf,
713                           jpegSubsamp, flags);
714   } else {
715     retval = tjCompress2(handle, srcBuf, width, pitch, height,
716                          getPixelFormat(pixelSize, flags), &jpegBuf, &size,
717                          jpegSubsamp, jpegQual, flags | TJFLAG_NOREALLOC);
718   }
719   *jpegSize = size;
720   return retval;
721 }
722 
723 
tjEncodeYUVPlanes(tjhandle handle,const unsigned char * srcBuf,int width,int pitch,int height,int pixelFormat,unsigned char ** dstPlanes,int * strides,int subsamp,int flags)724 DLLEXPORT int tjEncodeYUVPlanes(tjhandle handle, const unsigned char *srcBuf,
725                                 int width, int pitch, int height,
726                                 int pixelFormat, unsigned char **dstPlanes,
727                                 int *strides, int subsamp, int flags)
728 {
729   JSAMPROW *row_pointer = NULL;
730   JSAMPLE *_tmpbuf[MAX_COMPONENTS], *_tmpbuf2[MAX_COMPONENTS];
731   JSAMPROW *tmpbuf[MAX_COMPONENTS], *tmpbuf2[MAX_COMPONENTS];
732   JSAMPROW *outbuf[MAX_COMPONENTS];
733   int i, retval = 0, row, pw0, ph0, pw[MAX_COMPONENTS], ph[MAX_COMPONENTS];
734   JSAMPLE *ptr;
735   jpeg_component_info *compptr;
736 
737   GET_CINSTANCE(handle);
738   this->jerr.stopOnWarning = (flags & TJFLAG_STOPONWARNING) ? TRUE : FALSE;
739 
740   for (i = 0; i < MAX_COMPONENTS; i++) {
741     tmpbuf[i] = NULL;  _tmpbuf[i] = NULL;
742     tmpbuf2[i] = NULL;  _tmpbuf2[i] = NULL;  outbuf[i] = NULL;
743   }
744 
745   if ((this->init & COMPRESS) == 0)
746     THROW("tjEncodeYUVPlanes(): Instance has not been initialized for compression");
747 
748   if (srcBuf == NULL || width <= 0 || pitch < 0 || height <= 0 ||
749       pixelFormat < 0 || pixelFormat >= TJ_NUMPF || !dstPlanes ||
750       !dstPlanes[0] || subsamp < 0 || subsamp >= NUMSUBOPT)
751     THROW("tjEncodeYUVPlanes(): Invalid argument");
752   if (subsamp != TJSAMP_GRAY && (!dstPlanes[1] || !dstPlanes[2]))
753     THROW("tjEncodeYUVPlanes(): Invalid argument");
754 
755   if (pixelFormat == TJPF_CMYK)
756     THROW("tjEncodeYUVPlanes(): Cannot generate YUV images from CMYK pixels");
757 
758   if (pitch == 0) pitch = width * tjPixelSize[pixelFormat];
759 
760   if (setjmp(this->jerr.setjmp_buffer)) {
761     /* If we get here, the JPEG code has signaled an error. */
762     retval = -1;  goto bailout;
763   }
764 
765   cinfo->image_width = width;
766   cinfo->image_height = height;
767 
768 #ifndef NO_PUTENV
769   if (flags & TJFLAG_FORCEMMX) putenv("JSIMD_FORCEMMX=1");
770   else if (flags & TJFLAG_FORCESSE) putenv("JSIMD_FORCESSE=1");
771   else if (flags & TJFLAG_FORCESSE2) putenv("JSIMD_FORCESSE2=1");
772 #endif
773 
774   if (setCompDefaults(cinfo, pixelFormat, subsamp, -1, flags) == -1) return -1;
775 
776   /* Execute only the parts of jpeg_start_compress() that we need.  If we
777      were to call the whole jpeg_start_compress() function, then it would try
778      to write the file headers, which could overflow the output buffer if the
779      YUV image were very small. */
780   if (cinfo->global_state != CSTATE_START)
781     THROW("tjEncodeYUVPlanes(): libjpeg API is in the wrong state");
782   (*cinfo->err->reset_error_mgr) ((j_common_ptr)cinfo);
783   jinit_c_master_control(cinfo, FALSE);
784   jinit_color_converter(cinfo);
785   jinit_downsampler(cinfo);
786   (*cinfo->cconvert->start_pass) (cinfo);
787 
788   pw0 = PAD(width, cinfo->max_h_samp_factor);
789   ph0 = PAD(height, cinfo->max_v_samp_factor);
790 
791   if ((row_pointer = (JSAMPROW *)malloc(sizeof(JSAMPROW) * ph0)) == NULL)
792     THROW("tjEncodeYUVPlanes(): Memory allocation failure");
793   for (i = 0; i < height; i++) {
794     if (flags & TJFLAG_BOTTOMUP)
795       row_pointer[i] = (JSAMPROW)&srcBuf[(height - i - 1) * (size_t)pitch];
796     else
797       row_pointer[i] = (JSAMPROW)&srcBuf[i * (size_t)pitch];
798   }
799   if (height < ph0)
800     for (i = height; i < ph0; i++) row_pointer[i] = row_pointer[height - 1];
801 
802   for (i = 0; i < cinfo->num_components; i++) {
803     compptr = &cinfo->comp_info[i];
804     _tmpbuf[i] = (JSAMPLE *)malloc(
805       PAD((compptr->width_in_blocks * cinfo->max_h_samp_factor * DCTSIZE) /
806           compptr->h_samp_factor, 32) *
807       cinfo->max_v_samp_factor + 32);
808     if (!_tmpbuf[i])
809       THROW("tjEncodeYUVPlanes(): Memory allocation failure");
810     tmpbuf[i] =
811       (JSAMPROW *)malloc(sizeof(JSAMPROW) * cinfo->max_v_samp_factor);
812     if (!tmpbuf[i])
813       THROW("tjEncodeYUVPlanes(): Memory allocation failure");
814     for (row = 0; row < cinfo->max_v_samp_factor; row++) {
815       unsigned char *_tmpbuf_aligned =
816         (unsigned char *)PAD((size_t)_tmpbuf[i], 32);
817 
818       tmpbuf[i][row] = &_tmpbuf_aligned[
819         PAD((compptr->width_in_blocks * cinfo->max_h_samp_factor * DCTSIZE) /
820             compptr->h_samp_factor, 32) * row];
821     }
822     _tmpbuf2[i] =
823       (JSAMPLE *)malloc(PAD(compptr->width_in_blocks * DCTSIZE, 32) *
824                         compptr->v_samp_factor + 32);
825     if (!_tmpbuf2[i])
826       THROW("tjEncodeYUVPlanes(): Memory allocation failure");
827     tmpbuf2[i] = (JSAMPROW *)malloc(sizeof(JSAMPROW) * compptr->v_samp_factor);
828     if (!tmpbuf2[i])
829       THROW("tjEncodeYUVPlanes(): Memory allocation failure");
830     for (row = 0; row < compptr->v_samp_factor; row++) {
831       unsigned char *_tmpbuf2_aligned =
832         (unsigned char *)PAD((size_t)_tmpbuf2[i], 32);
833 
834       tmpbuf2[i][row] =
835         &_tmpbuf2_aligned[PAD(compptr->width_in_blocks * DCTSIZE, 32) * row];
836     }
837     pw[i] = pw0 * compptr->h_samp_factor / cinfo->max_h_samp_factor;
838     ph[i] = ph0 * compptr->v_samp_factor / cinfo->max_v_samp_factor;
839     outbuf[i] = (JSAMPROW *)malloc(sizeof(JSAMPROW) * ph[i]);
840     if (!outbuf[i])
841       THROW("tjEncodeYUVPlanes(): Memory allocation failure");
842     ptr = dstPlanes[i];
843     for (row = 0; row < ph[i]; row++) {
844       outbuf[i][row] = ptr;
845       ptr += (strides && strides[i] != 0) ? strides[i] : pw[i];
846     }
847   }
848 
849   if (setjmp(this->jerr.setjmp_buffer)) {
850     /* If we get here, the JPEG code has signaled an error. */
851     retval = -1;  goto bailout;
852   }
853 
854   for (row = 0; row < ph0; row += cinfo->max_v_samp_factor) {
855     (*cinfo->cconvert->color_convert) (cinfo, &row_pointer[row], tmpbuf, 0,
856                                        cinfo->max_v_samp_factor);
857     (cinfo->downsample->downsample) (cinfo, tmpbuf, 0, tmpbuf2, 0);
858     for (i = 0, compptr = cinfo->comp_info; i < cinfo->num_components;
859          i++, compptr++)
860       jcopy_sample_rows(tmpbuf2[i], 0, outbuf[i],
861         row * compptr->v_samp_factor / cinfo->max_v_samp_factor,
862         compptr->v_samp_factor, pw[i]);
863   }
864   cinfo->next_scanline += height;
865   jpeg_abort_compress(cinfo);
866 
867 bailout:
868   if (cinfo->global_state > CSTATE_START) jpeg_abort_compress(cinfo);
869   if (row_pointer) free(row_pointer);
870   for (i = 0; i < MAX_COMPONENTS; i++) {
871     if (tmpbuf[i] != NULL) free(tmpbuf[i]);
872     if (_tmpbuf[i] != NULL) free(_tmpbuf[i]);
873     if (tmpbuf2[i] != NULL) free(tmpbuf2[i]);
874     if (_tmpbuf2[i] != NULL) free(_tmpbuf2[i]);
875     if (outbuf[i] != NULL) free(outbuf[i]);
876   }
877   if (this->jerr.warning) retval = -1;
878   this->jerr.stopOnWarning = FALSE;
879   return retval;
880 }
881 
tjEncodeYUV3(tjhandle handle,const unsigned char * srcBuf,int width,int pitch,int height,int pixelFormat,unsigned char * dstBuf,int pad,int subsamp,int flags)882 DLLEXPORT int tjEncodeYUV3(tjhandle handle, const unsigned char *srcBuf,
883                            int width, int pitch, int height, int pixelFormat,
884                            unsigned char *dstBuf, int pad, int subsamp,
885                            int flags)
886 {
887   unsigned char *dstPlanes[3];
888   int pw0, ph0, strides[3], retval = -1;
889   tjinstance *this = (tjinstance *)handle;
890 
891   if (!this) THROWG("tjEncodeYUV3(): Invalid handle");
892   this->isInstanceError = FALSE;
893 
894   if (width <= 0 || height <= 0 || dstBuf == NULL || pad < 0 ||
895       !IS_POW2(pad) || subsamp < 0 || subsamp >= NUMSUBOPT)
896     THROW("tjEncodeYUV3(): Invalid argument");
897 
898   pw0 = tjPlaneWidth(0, width, subsamp);
899   ph0 = tjPlaneHeight(0, height, subsamp);
900   dstPlanes[0] = dstBuf;
901   strides[0] = PAD(pw0, pad);
902   if (subsamp == TJSAMP_GRAY) {
903     strides[1] = strides[2] = 0;
904     dstPlanes[1] = dstPlanes[2] = NULL;
905   } else {
906     int pw1 = tjPlaneWidth(1, width, subsamp);
907     int ph1 = tjPlaneHeight(1, height, subsamp);
908 
909     strides[1] = strides[2] = PAD(pw1, pad);
910     dstPlanes[1] = dstPlanes[0] + strides[0] * ph0;
911     dstPlanes[2] = dstPlanes[1] + strides[1] * ph1;
912   }
913 
914   return tjEncodeYUVPlanes(handle, srcBuf, width, pitch, height, pixelFormat,
915                            dstPlanes, strides, subsamp, flags);
916 
917 bailout:
918   return retval;
919 }
920 
tjEncodeYUV2(tjhandle handle,unsigned char * srcBuf,int width,int pitch,int height,int pixelFormat,unsigned char * dstBuf,int subsamp,int flags)921 DLLEXPORT int tjEncodeYUV2(tjhandle handle, unsigned char *srcBuf, int width,
922                            int pitch, int height, int pixelFormat,
923                            unsigned char *dstBuf, int subsamp, int flags)
924 {
925   return tjEncodeYUV3(handle, srcBuf, width, pitch, height, pixelFormat,
926                       dstBuf, 4, subsamp, flags);
927 }
928 
tjEncodeYUV(tjhandle handle,unsigned char * srcBuf,int width,int pitch,int height,int pixelSize,unsigned char * dstBuf,int subsamp,int flags)929 DLLEXPORT int tjEncodeYUV(tjhandle handle, unsigned char *srcBuf, int width,
930                           int pitch, int height, int pixelSize,
931                           unsigned char *dstBuf, int subsamp, int flags)
932 {
933   return tjEncodeYUV2(handle, srcBuf, width, pitch, height,
934                       getPixelFormat(pixelSize, flags), dstBuf, subsamp,
935                       flags);
936 }
937 
938 
tjCompressFromYUVPlanes(tjhandle handle,const unsigned char ** srcPlanes,int width,const int * strides,int height,int subsamp,unsigned char ** jpegBuf,unsigned long * jpegSize,int jpegQual,int flags)939 DLLEXPORT int tjCompressFromYUVPlanes(tjhandle handle,
940                                       const unsigned char **srcPlanes,
941                                       int width, const int *strides,
942                                       int height, int subsamp,
943                                       unsigned char **jpegBuf,
944                                       unsigned long *jpegSize, int jpegQual,
945                                       int flags)
946 {
947   int i, row, retval = 0, alloc = 1;
948   int pw[MAX_COMPONENTS], ph[MAX_COMPONENTS], iw[MAX_COMPONENTS],
949     tmpbufsize = 0, usetmpbuf = 0, th[MAX_COMPONENTS];
950   JSAMPLE *_tmpbuf = NULL, *ptr;
951   JSAMPROW *inbuf[MAX_COMPONENTS], *tmpbuf[MAX_COMPONENTS];
952 
953   GET_CINSTANCE(handle)
954   this->jerr.stopOnWarning = (flags & TJFLAG_STOPONWARNING) ? TRUE : FALSE;
955 
956   for (i = 0; i < MAX_COMPONENTS; i++) {
957     tmpbuf[i] = NULL;  inbuf[i] = NULL;
958   }
959 
960   if ((this->init & COMPRESS) == 0)
961     THROW("tjCompressFromYUVPlanes(): Instance has not been initialized for compression");
962 
963   if (!srcPlanes || !srcPlanes[0] || width <= 0 || height <= 0 ||
964       subsamp < 0 || subsamp >= NUMSUBOPT || jpegBuf == NULL ||
965       jpegSize == NULL || jpegQual < 0 || jpegQual > 100)
966     THROW("tjCompressFromYUVPlanes(): Invalid argument");
967   if (subsamp != TJSAMP_GRAY && (!srcPlanes[1] || !srcPlanes[2]))
968     THROW("tjCompressFromYUVPlanes(): Invalid argument");
969 
970   if (setjmp(this->jerr.setjmp_buffer)) {
971     /* If we get here, the JPEG code has signaled an error. */
972     retval = -1;  goto bailout;
973   }
974 
975   cinfo->image_width = width;
976   cinfo->image_height = height;
977 
978 #ifndef NO_PUTENV
979   if (flags & TJFLAG_FORCEMMX) putenv("JSIMD_FORCEMMX=1");
980   else if (flags & TJFLAG_FORCESSE) putenv("JSIMD_FORCESSE=1");
981   else if (flags & TJFLAG_FORCESSE2) putenv("JSIMD_FORCESSE2=1");
982 #endif
983 
984   if (flags & TJFLAG_NOREALLOC) {
985     alloc = 0;  *jpegSize = tjBufSize(width, height, subsamp);
986   }
987   jpeg_mem_dest_tj(cinfo, jpegBuf, jpegSize, alloc);
988   if (setCompDefaults(cinfo, TJPF_RGB, subsamp, jpegQual, flags) == -1)
989     return -1;
990   cinfo->raw_data_in = TRUE;
991 
992   jpeg_start_compress(cinfo, TRUE);
993   for (i = 0; i < cinfo->num_components; i++) {
994     jpeg_component_info *compptr = &cinfo->comp_info[i];
995     int ih;
996 
997     iw[i] = compptr->width_in_blocks * DCTSIZE;
998     ih = compptr->height_in_blocks * DCTSIZE;
999     pw[i] = PAD(cinfo->image_width, cinfo->max_h_samp_factor) *
1000             compptr->h_samp_factor / cinfo->max_h_samp_factor;
1001     ph[i] = PAD(cinfo->image_height, cinfo->max_v_samp_factor) *
1002             compptr->v_samp_factor / cinfo->max_v_samp_factor;
1003     if (iw[i] != pw[i] || ih != ph[i]) usetmpbuf = 1;
1004     th[i] = compptr->v_samp_factor * DCTSIZE;
1005     tmpbufsize += iw[i] * th[i];
1006     if ((inbuf[i] = (JSAMPROW *)malloc(sizeof(JSAMPROW) * ph[i])) == NULL)
1007       THROW("tjCompressFromYUVPlanes(): Memory allocation failure");
1008     ptr = (JSAMPLE *)srcPlanes[i];
1009     for (row = 0; row < ph[i]; row++) {
1010       inbuf[i][row] = ptr;
1011       ptr += (strides && strides[i] != 0) ? strides[i] : pw[i];
1012     }
1013   }
1014   if (usetmpbuf) {
1015     if ((_tmpbuf = (JSAMPLE *)malloc(sizeof(JSAMPLE) * tmpbufsize)) == NULL)
1016       THROW("tjCompressFromYUVPlanes(): Memory allocation failure");
1017     ptr = _tmpbuf;
1018     for (i = 0; i < cinfo->num_components; i++) {
1019       if ((tmpbuf[i] = (JSAMPROW *)malloc(sizeof(JSAMPROW) * th[i])) == NULL)
1020         THROW("tjCompressFromYUVPlanes(): Memory allocation failure");
1021       for (row = 0; row < th[i]; row++) {
1022         tmpbuf[i][row] = ptr;
1023         ptr += iw[i];
1024       }
1025     }
1026   }
1027 
1028   if (setjmp(this->jerr.setjmp_buffer)) {
1029     /* If we get here, the JPEG code has signaled an error. */
1030     retval = -1;  goto bailout;
1031   }
1032 
1033   for (row = 0; row < (int)cinfo->image_height;
1034        row += cinfo->max_v_samp_factor * DCTSIZE) {
1035     JSAMPARRAY yuvptr[MAX_COMPONENTS];
1036     int crow[MAX_COMPONENTS];
1037 
1038     for (i = 0; i < cinfo->num_components; i++) {
1039       jpeg_component_info *compptr = &cinfo->comp_info[i];
1040 
1041       crow[i] = row * compptr->v_samp_factor / cinfo->max_v_samp_factor;
1042       if (usetmpbuf) {
1043         int j, k;
1044 
1045         for (j = 0; j < MIN(th[i], ph[i] - crow[i]); j++) {
1046           memcpy(tmpbuf[i][j], inbuf[i][crow[i] + j], pw[i]);
1047           /* Duplicate last sample in row to fill out MCU */
1048           for (k = pw[i]; k < iw[i]; k++)
1049             tmpbuf[i][j][k] = tmpbuf[i][j][pw[i] - 1];
1050         }
1051         /* Duplicate last row to fill out MCU */
1052         for (j = ph[i] - crow[i]; j < th[i]; j++)
1053           memcpy(tmpbuf[i][j], tmpbuf[i][ph[i] - crow[i] - 1], iw[i]);
1054         yuvptr[i] = tmpbuf[i];
1055       } else
1056         yuvptr[i] = &inbuf[i][crow[i]];
1057     }
1058     jpeg_write_raw_data(cinfo, yuvptr, cinfo->max_v_samp_factor * DCTSIZE);
1059   }
1060   jpeg_finish_compress(cinfo);
1061 
1062 bailout:
1063   if (cinfo->global_state > CSTATE_START) jpeg_abort_compress(cinfo);
1064   for (i = 0; i < MAX_COMPONENTS; i++) {
1065     if (tmpbuf[i]) free(tmpbuf[i]);
1066     if (inbuf[i]) free(inbuf[i]);
1067   }
1068   if (_tmpbuf) free(_tmpbuf);
1069   if (this->jerr.warning) retval = -1;
1070   this->jerr.stopOnWarning = FALSE;
1071   return retval;
1072 }
1073 
tjCompressFromYUV(tjhandle handle,const unsigned char * srcBuf,int width,int pad,int height,int subsamp,unsigned char ** jpegBuf,unsigned long * jpegSize,int jpegQual,int flags)1074 DLLEXPORT int tjCompressFromYUV(tjhandle handle, const unsigned char *srcBuf,
1075                                 int width, int pad, int height, int subsamp,
1076                                 unsigned char **jpegBuf,
1077                                 unsigned long *jpegSize, int jpegQual,
1078                                 int flags)
1079 {
1080   const unsigned char *srcPlanes[3];
1081   int pw0, ph0, strides[3], retval = -1;
1082   tjinstance *this = (tjinstance *)handle;
1083 
1084   if (!this) THROWG("tjCompressFromYUV(): Invalid handle");
1085   this->isInstanceError = FALSE;
1086 
1087   if (srcBuf == NULL || width <= 0 || pad < 1 || height <= 0 || subsamp < 0 ||
1088       subsamp >= NUMSUBOPT)
1089     THROW("tjCompressFromYUV(): Invalid argument");
1090 
1091   pw0 = tjPlaneWidth(0, width, subsamp);
1092   ph0 = tjPlaneHeight(0, height, subsamp);
1093   srcPlanes[0] = srcBuf;
1094   strides[0] = PAD(pw0, pad);
1095   if (subsamp == TJSAMP_GRAY) {
1096     strides[1] = strides[2] = 0;
1097     srcPlanes[1] = srcPlanes[2] = NULL;
1098   } else {
1099     int pw1 = tjPlaneWidth(1, width, subsamp);
1100     int ph1 = tjPlaneHeight(1, height, subsamp);
1101 
1102     strides[1] = strides[2] = PAD(pw1, pad);
1103     srcPlanes[1] = srcPlanes[0] + strides[0] * ph0;
1104     srcPlanes[2] = srcPlanes[1] + strides[1] * ph1;
1105   }
1106 
1107   return tjCompressFromYUVPlanes(handle, srcPlanes, width, strides, height,
1108                                  subsamp, jpegBuf, jpegSize, jpegQual, flags);
1109 
1110 bailout:
1111   return retval;
1112 }
1113 
1114 
1115 /* Decompressor */
1116 
_tjInitDecompress(tjinstance * this)1117 static tjhandle _tjInitDecompress(tjinstance *this)
1118 {
1119   static unsigned char buffer[1];
1120 
1121   /* This is also straight out of example.txt */
1122   this->dinfo.err = jpeg_std_error(&this->jerr.pub);
1123   this->jerr.pub.error_exit = my_error_exit;
1124   this->jerr.pub.output_message = my_output_message;
1125   this->jerr.emit_message = this->jerr.pub.emit_message;
1126   this->jerr.pub.emit_message = my_emit_message;
1127   this->jerr.pub.addon_message_table = turbojpeg_message_table;
1128   this->jerr.pub.first_addon_message = JMSG_FIRSTADDONCODE;
1129   this->jerr.pub.last_addon_message = JMSG_LASTADDONCODE;
1130 
1131   if (setjmp(this->jerr.setjmp_buffer)) {
1132     /* If we get here, the JPEG code has signaled an error. */
1133     if (this) free(this);
1134     return NULL;
1135   }
1136 
1137   jpeg_create_decompress(&this->dinfo);
1138   /* Make an initial call so it will create the source manager */
1139   jpeg_mem_src_tj(&this->dinfo, buffer, 1);
1140 
1141   this->init |= DECOMPRESS;
1142   return (tjhandle)this;
1143 }
1144 
tjInitDecompress(void)1145 DLLEXPORT tjhandle tjInitDecompress(void)
1146 {
1147   tjinstance *this;
1148 
1149   if ((this = (tjinstance *)malloc(sizeof(tjinstance))) == NULL) {
1150     snprintf(errStr, JMSG_LENGTH_MAX,
1151              "tjInitDecompress(): Memory allocation failure");
1152     return NULL;
1153   }
1154   MEMZERO(this, sizeof(tjinstance));
1155   snprintf(this->errStr, JMSG_LENGTH_MAX, "No error");
1156   return _tjInitDecompress(this);
1157 }
1158 
1159 
tjDecompressHeader3(tjhandle handle,const unsigned char * jpegBuf,unsigned long jpegSize,int * width,int * height,int * jpegSubsamp,int * jpegColorspace)1160 DLLEXPORT int tjDecompressHeader3(tjhandle handle,
1161                                   const unsigned char *jpegBuf,
1162                                   unsigned long jpegSize, int *width,
1163                                   int *height, int *jpegSubsamp,
1164                                   int *jpegColorspace)
1165 {
1166   int retval = 0;
1167 
1168   GET_DINSTANCE(handle);
1169   if ((this->init & DECOMPRESS) == 0)
1170     THROW("tjDecompressHeader3(): Instance has not been initialized for decompression");
1171 
1172   if (jpegBuf == NULL || jpegSize <= 0 || width == NULL || height == NULL ||
1173       jpegSubsamp == NULL || jpegColorspace == NULL)
1174     THROW("tjDecompressHeader3(): Invalid argument");
1175 
1176   if (setjmp(this->jerr.setjmp_buffer)) {
1177     /* If we get here, the JPEG code has signaled an error. */
1178     return -1;
1179   }
1180 
1181   jpeg_mem_src_tj(dinfo, jpegBuf, jpegSize);
1182   jpeg_read_header(dinfo, TRUE);
1183 
1184   *width = dinfo->image_width;
1185   *height = dinfo->image_height;
1186   *jpegSubsamp = getSubsamp(dinfo);
1187   switch (dinfo->jpeg_color_space) {
1188   case JCS_GRAYSCALE:  *jpegColorspace = TJCS_GRAY;  break;
1189   case JCS_RGB:        *jpegColorspace = TJCS_RGB;  break;
1190   case JCS_YCbCr:      *jpegColorspace = TJCS_YCbCr;  break;
1191   case JCS_CMYK:       *jpegColorspace = TJCS_CMYK;  break;
1192   case JCS_YCCK:       *jpegColorspace = TJCS_YCCK;  break;
1193   default:             *jpegColorspace = -1;  break;
1194   }
1195 
1196   jpeg_abort_decompress(dinfo);
1197 
1198   if (*jpegSubsamp < 0)
1199     THROW("tjDecompressHeader3(): Could not determine subsampling type for JPEG image");
1200   if (*jpegColorspace < 0)
1201     THROW("tjDecompressHeader3(): Could not determine colorspace of JPEG image");
1202   if (*width < 1 || *height < 1)
1203     THROW("tjDecompressHeader3(): Invalid data returned in header");
1204 
1205 bailout:
1206   if (this->jerr.warning) retval = -1;
1207   return retval;
1208 }
1209 
tjDecompressHeader2(tjhandle handle,unsigned char * jpegBuf,unsigned long jpegSize,int * width,int * height,int * jpegSubsamp)1210 DLLEXPORT int tjDecompressHeader2(tjhandle handle, unsigned char *jpegBuf,
1211                                   unsigned long jpegSize, int *width,
1212                                   int *height, int *jpegSubsamp)
1213 {
1214   int jpegColorspace;
1215 
1216   return tjDecompressHeader3(handle, jpegBuf, jpegSize, width, height,
1217                              jpegSubsamp, &jpegColorspace);
1218 }
1219 
tjDecompressHeader(tjhandle handle,unsigned char * jpegBuf,unsigned long jpegSize,int * width,int * height)1220 DLLEXPORT int tjDecompressHeader(tjhandle handle, unsigned char *jpegBuf,
1221                                  unsigned long jpegSize, int *width,
1222                                  int *height)
1223 {
1224   int jpegSubsamp;
1225 
1226   return tjDecompressHeader2(handle, jpegBuf, jpegSize, width, height,
1227                              &jpegSubsamp);
1228 }
1229 
1230 
tjGetScalingFactors(int * numscalingfactors)1231 DLLEXPORT tjscalingfactor *tjGetScalingFactors(int *numscalingfactors)
1232 {
1233   if (numscalingfactors == NULL) {
1234     snprintf(errStr, JMSG_LENGTH_MAX,
1235              "tjGetScalingFactors(): Invalid argument");
1236     return NULL;
1237   }
1238 
1239   *numscalingfactors = NUMSF;
1240   return (tjscalingfactor *)sf;
1241 }
1242 
1243 
tjDecompress2(tjhandle handle,const unsigned char * jpegBuf,unsigned long jpegSize,unsigned char * dstBuf,int width,int pitch,int height,int pixelFormat,int flags)1244 DLLEXPORT int tjDecompress2(tjhandle handle, const unsigned char *jpegBuf,
1245                             unsigned long jpegSize, unsigned char *dstBuf,
1246                             int width, int pitch, int height, int pixelFormat,
1247                             int flags)
1248 {
1249   JSAMPROW *row_pointer = NULL;
1250   int i, retval = 0, jpegwidth, jpegheight, scaledw, scaledh;
1251 
1252   GET_DINSTANCE(handle);
1253   this->jerr.stopOnWarning = (flags & TJFLAG_STOPONWARNING) ? TRUE : FALSE;
1254   if ((this->init & DECOMPRESS) == 0)
1255     THROW("tjDecompress2(): Instance has not been initialized for decompression");
1256 
1257   if (jpegBuf == NULL || jpegSize <= 0 || dstBuf == NULL || width < 0 ||
1258       pitch < 0 || height < 0 || pixelFormat < 0 || pixelFormat >= TJ_NUMPF)
1259     THROW("tjDecompress2(): Invalid argument");
1260 
1261 #ifndef NO_PUTENV
1262   if (flags & TJFLAG_FORCEMMX) putenv("JSIMD_FORCEMMX=1");
1263   else if (flags & TJFLAG_FORCESSE) putenv("JSIMD_FORCESSE=1");
1264   else if (flags & TJFLAG_FORCESSE2) putenv("JSIMD_FORCESSE2=1");
1265 #endif
1266 
1267   if (setjmp(this->jerr.setjmp_buffer)) {
1268     /* If we get here, the JPEG code has signaled an error. */
1269     retval = -1;  goto bailout;
1270   }
1271 
1272   jpeg_mem_src_tj(dinfo, jpegBuf, jpegSize);
1273   jpeg_read_header(dinfo, TRUE);
1274   this->dinfo.out_color_space = pf2cs[pixelFormat];
1275   if (flags & TJFLAG_FASTDCT) this->dinfo.dct_method = JDCT_FASTEST;
1276   if (flags & TJFLAG_FASTUPSAMPLE) dinfo->do_fancy_upsampling = FALSE;
1277 
1278   jpegwidth = dinfo->image_width;  jpegheight = dinfo->image_height;
1279   if (width == 0) width = jpegwidth;
1280   if (height == 0) height = jpegheight;
1281   for (i = 0; i < NUMSF; i++) {
1282     scaledw = TJSCALED(jpegwidth, sf[i]);
1283     scaledh = TJSCALED(jpegheight, sf[i]);
1284     if (scaledw <= width && scaledh <= height)
1285       break;
1286   }
1287   if (i >= NUMSF)
1288     THROW("tjDecompress2(): Could not scale down to desired image dimensions");
1289   width = scaledw;  height = scaledh;
1290   dinfo->scale_num = sf[i].num;
1291   dinfo->scale_denom = sf[i].denom;
1292 
1293   jpeg_start_decompress(dinfo);
1294   if (pitch == 0) pitch = dinfo->output_width * tjPixelSize[pixelFormat];
1295 
1296   if ((row_pointer =
1297        (JSAMPROW *)malloc(sizeof(JSAMPROW) * dinfo->output_height)) == NULL)
1298     THROW("tjDecompress2(): Memory allocation failure");
1299   if (setjmp(this->jerr.setjmp_buffer)) {
1300     /* If we get here, the JPEG code has signaled an error. */
1301     retval = -1;  goto bailout;
1302   }
1303   for (i = 0; i < (int)dinfo->output_height; i++) {
1304     if (flags & TJFLAG_BOTTOMUP)
1305       row_pointer[i] = &dstBuf[(dinfo->output_height - i - 1) * (size_t)pitch];
1306     else
1307       row_pointer[i] = &dstBuf[i * (size_t)pitch];
1308   }
1309   while (dinfo->output_scanline < dinfo->output_height)
1310     jpeg_read_scanlines(dinfo, &row_pointer[dinfo->output_scanline],
1311                         dinfo->output_height - dinfo->output_scanline);
1312   jpeg_finish_decompress(dinfo);
1313 
1314 bailout:
1315   if (dinfo->global_state > DSTATE_START) jpeg_abort_decompress(dinfo);
1316   if (row_pointer) free(row_pointer);
1317   if (this->jerr.warning) retval = -1;
1318   this->jerr.stopOnWarning = FALSE;
1319   return retval;
1320 }
1321 
tjDecompress(tjhandle handle,unsigned char * jpegBuf,unsigned long jpegSize,unsigned char * dstBuf,int width,int pitch,int height,int pixelSize,int flags)1322 DLLEXPORT int tjDecompress(tjhandle handle, unsigned char *jpegBuf,
1323                            unsigned long jpegSize, unsigned char *dstBuf,
1324                            int width, int pitch, int height, int pixelSize,
1325                            int flags)
1326 {
1327   if (flags & TJ_YUV)
1328     return tjDecompressToYUV(handle, jpegBuf, jpegSize, dstBuf, flags);
1329   else
1330     return tjDecompress2(handle, jpegBuf, jpegSize, dstBuf, width, pitch,
1331                          height, getPixelFormat(pixelSize, flags), flags);
1332 }
1333 
1334 
setDecodeDefaults(struct jpeg_decompress_struct * dinfo,int pixelFormat,int subsamp,int flags)1335 static int setDecodeDefaults(struct jpeg_decompress_struct *dinfo,
1336                              int pixelFormat, int subsamp, int flags)
1337 {
1338   int i;
1339 
1340   dinfo->scale_num = dinfo->scale_denom = 1;
1341 
1342   if (subsamp == TJSAMP_GRAY) {
1343     dinfo->num_components = dinfo->comps_in_scan = 1;
1344     dinfo->jpeg_color_space = JCS_GRAYSCALE;
1345   } else {
1346     dinfo->num_components = dinfo->comps_in_scan = 3;
1347     dinfo->jpeg_color_space = JCS_YCbCr;
1348   }
1349 
1350   dinfo->comp_info = (jpeg_component_info *)
1351     (*dinfo->mem->alloc_small) ((j_common_ptr)dinfo, JPOOL_IMAGE,
1352                                 dinfo->num_components *
1353                                 sizeof(jpeg_component_info));
1354 
1355   for (i = 0; i < dinfo->num_components; i++) {
1356     jpeg_component_info *compptr = &dinfo->comp_info[i];
1357 
1358     compptr->h_samp_factor = (i == 0) ? tjMCUWidth[subsamp] / 8 : 1;
1359     compptr->v_samp_factor = (i == 0) ? tjMCUHeight[subsamp] / 8 : 1;
1360     compptr->component_index = i;
1361     compptr->component_id = i + 1;
1362     compptr->quant_tbl_no = compptr->dc_tbl_no =
1363       compptr->ac_tbl_no = (i == 0) ? 0 : 1;
1364     dinfo->cur_comp_info[i] = compptr;
1365   }
1366   dinfo->data_precision = 8;
1367   for (i = 0; i < 2; i++) {
1368     if (dinfo->quant_tbl_ptrs[i] == NULL)
1369       dinfo->quant_tbl_ptrs[i] = jpeg_alloc_quant_table((j_common_ptr)dinfo);
1370   }
1371 
1372   return 0;
1373 }
1374 
1375 
my_read_markers(j_decompress_ptr dinfo)1376 static int my_read_markers(j_decompress_ptr dinfo)
1377 {
1378   return JPEG_REACHED_SOS;
1379 }
1380 
my_reset_marker_reader(j_decompress_ptr dinfo)1381 static void my_reset_marker_reader(j_decompress_ptr dinfo)
1382 {
1383 }
1384 
tjDecodeYUVPlanes(tjhandle handle,const unsigned char ** srcPlanes,const int * strides,int subsamp,unsigned char * dstBuf,int width,int pitch,int height,int pixelFormat,int flags)1385 DLLEXPORT int tjDecodeYUVPlanes(tjhandle handle,
1386                                 const unsigned char **srcPlanes,
1387                                 const int *strides, int subsamp,
1388                                 unsigned char *dstBuf, int width, int pitch,
1389                                 int height, int pixelFormat, int flags)
1390 {
1391   JSAMPROW *row_pointer = NULL;
1392   JSAMPLE *_tmpbuf[MAX_COMPONENTS];
1393   JSAMPROW *tmpbuf[MAX_COMPONENTS], *inbuf[MAX_COMPONENTS];
1394   int i, retval = 0, row, pw0, ph0, pw[MAX_COMPONENTS], ph[MAX_COMPONENTS];
1395   JSAMPLE *ptr;
1396   jpeg_component_info *compptr;
1397   int (*old_read_markers) (j_decompress_ptr);
1398   void (*old_reset_marker_reader) (j_decompress_ptr);
1399 
1400   GET_DINSTANCE(handle);
1401   this->jerr.stopOnWarning = (flags & TJFLAG_STOPONWARNING) ? TRUE : FALSE;
1402 
1403   for (i = 0; i < MAX_COMPONENTS; i++) {
1404     tmpbuf[i] = NULL;  _tmpbuf[i] = NULL;  inbuf[i] = NULL;
1405   }
1406 
1407   if ((this->init & DECOMPRESS) == 0)
1408     THROW("tjDecodeYUVPlanes(): Instance has not been initialized for decompression");
1409 
1410   if (!srcPlanes || !srcPlanes[0] || subsamp < 0 || subsamp >= NUMSUBOPT ||
1411       dstBuf == NULL || width <= 0 || pitch < 0 || height <= 0 ||
1412       pixelFormat < 0 || pixelFormat >= TJ_NUMPF)
1413     THROW("tjDecodeYUVPlanes(): Invalid argument");
1414   if (subsamp != TJSAMP_GRAY && (!srcPlanes[1] || !srcPlanes[2]))
1415     THROW("tjDecodeYUVPlanes(): Invalid argument");
1416 
1417   if (setjmp(this->jerr.setjmp_buffer)) {
1418     /* If we get here, the JPEG code has signaled an error. */
1419     retval = -1;  goto bailout;
1420   }
1421 
1422   if (pixelFormat == TJPF_CMYK)
1423     THROW("tjDecodeYUVPlanes(): Cannot decode YUV images into CMYK pixels.");
1424 
1425   if (pitch == 0) pitch = width * tjPixelSize[pixelFormat];
1426   dinfo->image_width = width;
1427   dinfo->image_height = height;
1428 
1429 #ifndef NO_PUTENV
1430   if (flags & TJFLAG_FORCEMMX) putenv("JSIMD_FORCEMMX=1");
1431   else if (flags & TJFLAG_FORCESSE) putenv("JSIMD_FORCESSE=1");
1432   else if (flags & TJFLAG_FORCESSE2) putenv("JSIMD_FORCESSE2=1");
1433 #endif
1434 
1435   dinfo->progressive_mode = dinfo->inputctl->has_multiple_scans = FALSE;
1436   dinfo->Ss = dinfo->Ah = dinfo->Al = 0;
1437   dinfo->Se = DCTSIZE2 - 1;
1438   if (setDecodeDefaults(dinfo, pixelFormat, subsamp, flags) == -1) {
1439     retval = -1;  goto bailout;
1440   }
1441   old_read_markers = dinfo->marker->read_markers;
1442   dinfo->marker->read_markers = my_read_markers;
1443   old_reset_marker_reader = dinfo->marker->reset_marker_reader;
1444   dinfo->marker->reset_marker_reader = my_reset_marker_reader;
1445   jpeg_read_header(dinfo, TRUE);
1446   dinfo->marker->read_markers = old_read_markers;
1447   dinfo->marker->reset_marker_reader = old_reset_marker_reader;
1448 
1449   this->dinfo.out_color_space = pf2cs[pixelFormat];
1450   if (flags & TJFLAG_FASTDCT) this->dinfo.dct_method = JDCT_FASTEST;
1451   dinfo->do_fancy_upsampling = FALSE;
1452   dinfo->Se = DCTSIZE2 - 1;
1453   jinit_master_decompress(dinfo);
1454   (*dinfo->upsample->start_pass) (dinfo);
1455 
1456   pw0 = PAD(width, dinfo->max_h_samp_factor);
1457   ph0 = PAD(height, dinfo->max_v_samp_factor);
1458 
1459   if (pitch == 0) pitch = dinfo->output_width * tjPixelSize[pixelFormat];
1460 
1461   if ((row_pointer = (JSAMPROW *)malloc(sizeof(JSAMPROW) * ph0)) == NULL)
1462     THROW("tjDecodeYUVPlanes(): Memory allocation failure");
1463   for (i = 0; i < height; i++) {
1464     if (flags & TJFLAG_BOTTOMUP)
1465       row_pointer[i] = &dstBuf[(height - i - 1) * (size_t)pitch];
1466     else
1467       row_pointer[i] = &dstBuf[i * (size_t)pitch];
1468   }
1469   if (height < ph0)
1470     for (i = height; i < ph0; i++) row_pointer[i] = row_pointer[height - 1];
1471 
1472   for (i = 0; i < dinfo->num_components; i++) {
1473     compptr = &dinfo->comp_info[i];
1474     _tmpbuf[i] =
1475       (JSAMPLE *)malloc(PAD(compptr->width_in_blocks * DCTSIZE, 32) *
1476                         compptr->v_samp_factor + 32);
1477     if (!_tmpbuf[i])
1478       THROW("tjDecodeYUVPlanes(): Memory allocation failure");
1479     tmpbuf[i] = (JSAMPROW *)malloc(sizeof(JSAMPROW) * compptr->v_samp_factor);
1480     if (!tmpbuf[i])
1481       THROW("tjDecodeYUVPlanes(): Memory allocation failure");
1482     for (row = 0; row < compptr->v_samp_factor; row++) {
1483       unsigned char *_tmpbuf_aligned =
1484         (unsigned char *)PAD((size_t)_tmpbuf[i], 32);
1485 
1486       tmpbuf[i][row] =
1487         &_tmpbuf_aligned[PAD(compptr->width_in_blocks * DCTSIZE, 32) * row];
1488     }
1489     pw[i] = pw0 * compptr->h_samp_factor / dinfo->max_h_samp_factor;
1490     ph[i] = ph0 * compptr->v_samp_factor / dinfo->max_v_samp_factor;
1491     inbuf[i] = (JSAMPROW *)malloc(sizeof(JSAMPROW) * ph[i]);
1492     if (!inbuf[i])
1493       THROW("tjDecodeYUVPlanes(): Memory allocation failure");
1494     ptr = (JSAMPLE *)srcPlanes[i];
1495     for (row = 0; row < ph[i]; row++) {
1496       inbuf[i][row] = ptr;
1497       ptr += (strides && strides[i] != 0) ? strides[i] : pw[i];
1498     }
1499   }
1500 
1501   if (setjmp(this->jerr.setjmp_buffer)) {
1502     /* If we get here, the JPEG code has signaled an error. */
1503     retval = -1;  goto bailout;
1504   }
1505 
1506   for (row = 0; row < ph0; row += dinfo->max_v_samp_factor) {
1507     JDIMENSION inrow = 0, outrow = 0;
1508 
1509     for (i = 0, compptr = dinfo->comp_info; i < dinfo->num_components;
1510          i++, compptr++)
1511       jcopy_sample_rows(inbuf[i],
1512         row * compptr->v_samp_factor / dinfo->max_v_samp_factor, tmpbuf[i], 0,
1513         compptr->v_samp_factor, pw[i]);
1514     (dinfo->upsample->upsample) (dinfo, tmpbuf, &inrow,
1515                                  dinfo->max_v_samp_factor, &row_pointer[row],
1516                                  &outrow, dinfo->max_v_samp_factor);
1517   }
1518   jpeg_abort_decompress(dinfo);
1519 
1520 bailout:
1521   if (dinfo->global_state > DSTATE_START) jpeg_abort_decompress(dinfo);
1522   if (row_pointer) free(row_pointer);
1523   for (i = 0; i < MAX_COMPONENTS; i++) {
1524     if (tmpbuf[i] != NULL) free(tmpbuf[i]);
1525     if (_tmpbuf[i] != NULL) free(_tmpbuf[i]);
1526     if (inbuf[i] != NULL) free(inbuf[i]);
1527   }
1528   if (this->jerr.warning) retval = -1;
1529   this->jerr.stopOnWarning = FALSE;
1530   return retval;
1531 }
1532 
tjDecodeYUV(tjhandle handle,const unsigned char * srcBuf,int pad,int subsamp,unsigned char * dstBuf,int width,int pitch,int height,int pixelFormat,int flags)1533 DLLEXPORT int tjDecodeYUV(tjhandle handle, const unsigned char *srcBuf,
1534                           int pad, int subsamp, unsigned char *dstBuf,
1535                           int width, int pitch, int height, int pixelFormat,
1536                           int flags)
1537 {
1538   const unsigned char *srcPlanes[3];
1539   int pw0, ph0, strides[3], retval = -1;
1540   tjinstance *this = (tjinstance *)handle;
1541 
1542   if (!this) THROWG("tjDecodeYUV(): Invalid handle");
1543   this->isInstanceError = FALSE;
1544 
1545   if (srcBuf == NULL || pad < 0 || !IS_POW2(pad) || subsamp < 0 ||
1546       subsamp >= NUMSUBOPT || width <= 0 || height <= 0)
1547     THROW("tjDecodeYUV(): Invalid argument");
1548 
1549   pw0 = tjPlaneWidth(0, width, subsamp);
1550   ph0 = tjPlaneHeight(0, height, subsamp);
1551   srcPlanes[0] = srcBuf;
1552   strides[0] = PAD(pw0, pad);
1553   if (subsamp == TJSAMP_GRAY) {
1554     strides[1] = strides[2] = 0;
1555     srcPlanes[1] = srcPlanes[2] = NULL;
1556   } else {
1557     int pw1 = tjPlaneWidth(1, width, subsamp);
1558     int ph1 = tjPlaneHeight(1, height, subsamp);
1559 
1560     strides[1] = strides[2] = PAD(pw1, pad);
1561     srcPlanes[1] = srcPlanes[0] + strides[0] * ph0;
1562     srcPlanes[2] = srcPlanes[1] + strides[1] * ph1;
1563   }
1564 
1565   return tjDecodeYUVPlanes(handle, srcPlanes, strides, subsamp, dstBuf, width,
1566                            pitch, height, pixelFormat, flags);
1567 
1568 bailout:
1569   return retval;
1570 }
1571 
tjDecompressToYUVPlanes(tjhandle handle,const unsigned char * jpegBuf,unsigned long jpegSize,unsigned char ** dstPlanes,int width,int * strides,int height,int flags)1572 DLLEXPORT int tjDecompressToYUVPlanes(tjhandle handle,
1573                                       const unsigned char *jpegBuf,
1574                                       unsigned long jpegSize,
1575                                       unsigned char **dstPlanes, int width,
1576                                       int *strides, int height, int flags)
1577 {
1578   int i, sfi, row, retval = 0;
1579   int jpegwidth, jpegheight, jpegSubsamp, scaledw, scaledh;
1580   int pw[MAX_COMPONENTS], ph[MAX_COMPONENTS], iw[MAX_COMPONENTS],
1581     tmpbufsize = 0, usetmpbuf = 0, th[MAX_COMPONENTS];
1582   JSAMPLE *_tmpbuf = NULL, *ptr;
1583   JSAMPROW *outbuf[MAX_COMPONENTS], *tmpbuf[MAX_COMPONENTS];
1584   int dctsize;
1585 
1586   GET_DINSTANCE(handle);
1587   this->jerr.stopOnWarning = (flags & TJFLAG_STOPONWARNING) ? TRUE : FALSE;
1588 
1589   for (i = 0; i < MAX_COMPONENTS; i++) {
1590     tmpbuf[i] = NULL;  outbuf[i] = NULL;
1591   }
1592 
1593   if ((this->init & DECOMPRESS) == 0)
1594     THROW("tjDecompressToYUVPlanes(): Instance has not been initialized for decompression");
1595 
1596   if (jpegBuf == NULL || jpegSize <= 0 || !dstPlanes || !dstPlanes[0] ||
1597       width < 0 || height < 0)
1598     THROW("tjDecompressToYUVPlanes(): Invalid argument");
1599 
1600 #ifndef NO_PUTENV
1601   if (flags & TJFLAG_FORCEMMX) putenv("JSIMD_FORCEMMX=1");
1602   else if (flags & TJFLAG_FORCESSE) putenv("JSIMD_FORCESSE=1");
1603   else if (flags & TJFLAG_FORCESSE2) putenv("JSIMD_FORCESSE2=1");
1604 #endif
1605 
1606   if (setjmp(this->jerr.setjmp_buffer)) {
1607     /* If we get here, the JPEG code has signaled an error. */
1608     retval = -1;  goto bailout;
1609   }
1610 
1611   if (!this->headerRead) {
1612     jpeg_mem_src_tj(dinfo, jpegBuf, jpegSize);
1613     jpeg_read_header(dinfo, TRUE);
1614   }
1615   this->headerRead = 0;
1616   jpegSubsamp = getSubsamp(dinfo);
1617   if (jpegSubsamp < 0)
1618     THROW("tjDecompressToYUVPlanes(): Could not determine subsampling type for JPEG image");
1619 
1620   if (jpegSubsamp != TJSAMP_GRAY && (!dstPlanes[1] || !dstPlanes[2]))
1621     THROW("tjDecompressToYUVPlanes(): Invalid argument");
1622 
1623   jpegwidth = dinfo->image_width;  jpegheight = dinfo->image_height;
1624   if (width == 0) width = jpegwidth;
1625   if (height == 0) height = jpegheight;
1626   for (i = 0; i < NUMSF; i++) {
1627     scaledw = TJSCALED(jpegwidth, sf[i]);
1628     scaledh = TJSCALED(jpegheight, sf[i]);
1629     if (scaledw <= width && scaledh <= height)
1630       break;
1631   }
1632   if (i >= NUMSF)
1633     THROW("tjDecompressToYUVPlanes(): Could not scale down to desired image dimensions");
1634   if (dinfo->num_components > 3)
1635     THROW("tjDecompressToYUVPlanes(): JPEG image must have 3 or fewer components");
1636 
1637   width = scaledw;  height = scaledh;
1638   dinfo->scale_num = sf[i].num;
1639   dinfo->scale_denom = sf[i].denom;
1640   sfi = i;
1641   jpeg_calc_output_dimensions(dinfo);
1642 
1643   dctsize = DCTSIZE * sf[sfi].num / sf[sfi].denom;
1644 
1645   for (i = 0; i < dinfo->num_components; i++) {
1646     jpeg_component_info *compptr = &dinfo->comp_info[i];
1647     int ih;
1648 
1649     iw[i] = compptr->width_in_blocks * dctsize;
1650     ih = compptr->height_in_blocks * dctsize;
1651     pw[i] = PAD(dinfo->output_width, dinfo->max_h_samp_factor) *
1652             compptr->h_samp_factor / dinfo->max_h_samp_factor;
1653     ph[i] = PAD(dinfo->output_height, dinfo->max_v_samp_factor) *
1654             compptr->v_samp_factor / dinfo->max_v_samp_factor;
1655     if (iw[i] != pw[i] || ih != ph[i]) usetmpbuf = 1;
1656     th[i] = compptr->v_samp_factor * dctsize;
1657     tmpbufsize += iw[i] * th[i];
1658     if ((outbuf[i] = (JSAMPROW *)malloc(sizeof(JSAMPROW) * ph[i])) == NULL)
1659       THROW("tjDecompressToYUVPlanes(): Memory allocation failure");
1660     ptr = dstPlanes[i];
1661     for (row = 0; row < ph[i]; row++) {
1662       outbuf[i][row] = ptr;
1663       ptr += (strides && strides[i] != 0) ? strides[i] : pw[i];
1664     }
1665   }
1666   if (usetmpbuf) {
1667     if ((_tmpbuf = (JSAMPLE *)malloc(sizeof(JSAMPLE) * tmpbufsize)) == NULL)
1668       THROW("tjDecompressToYUVPlanes(): Memory allocation failure");
1669     ptr = _tmpbuf;
1670     for (i = 0; i < dinfo->num_components; i++) {
1671       if ((tmpbuf[i] = (JSAMPROW *)malloc(sizeof(JSAMPROW) * th[i])) == NULL)
1672         THROW("tjDecompressToYUVPlanes(): Memory allocation failure");
1673       for (row = 0; row < th[i]; row++) {
1674         tmpbuf[i][row] = ptr;
1675         ptr += iw[i];
1676       }
1677     }
1678   }
1679 
1680   if (setjmp(this->jerr.setjmp_buffer)) {
1681     /* If we get here, the JPEG code has signaled an error. */
1682     retval = -1;  goto bailout;
1683   }
1684 
1685   if (flags & TJFLAG_FASTUPSAMPLE) dinfo->do_fancy_upsampling = FALSE;
1686   if (flags & TJFLAG_FASTDCT) dinfo->dct_method = JDCT_FASTEST;
1687   dinfo->raw_data_out = TRUE;
1688 
1689   jpeg_start_decompress(dinfo);
1690   for (row = 0; row < (int)dinfo->output_height;
1691        row += dinfo->max_v_samp_factor * dinfo->_min_DCT_scaled_size) {
1692     JSAMPARRAY yuvptr[MAX_COMPONENTS];
1693     int crow[MAX_COMPONENTS];
1694 
1695     for (i = 0; i < dinfo->num_components; i++) {
1696       jpeg_component_info *compptr = &dinfo->comp_info[i];
1697 
1698       if (jpegSubsamp == TJ_420) {
1699         /* When 4:2:0 subsampling is used with IDCT scaling, libjpeg will try
1700            to be clever and use the IDCT to perform upsampling on the U and V
1701            planes.  For instance, if the output image is to be scaled by 1/2
1702            relative to the JPEG image, then the scaling factor and upsampling
1703            effectively cancel each other, so a normal 8x8 IDCT can be used.
1704            However, this is not desirable when using the decompress-to-YUV
1705            functionality in TurboJPEG, since we want to output the U and V
1706            planes in their subsampled form.  Thus, we have to override some
1707            internal libjpeg parameters to force it to use the "scaled" IDCT
1708            functions on the U and V planes. */
1709         compptr->_DCT_scaled_size = dctsize;
1710         compptr->MCU_sample_width = tjMCUWidth[jpegSubsamp] *
1711           sf[sfi].num / sf[sfi].denom *
1712           compptr->v_samp_factor / dinfo->max_v_samp_factor;
1713         dinfo->idct->inverse_DCT[i] = dinfo->idct->inverse_DCT[0];
1714       }
1715       crow[i] = row * compptr->v_samp_factor / dinfo->max_v_samp_factor;
1716       if (usetmpbuf) yuvptr[i] = tmpbuf[i];
1717       else yuvptr[i] = &outbuf[i][crow[i]];
1718     }
1719     jpeg_read_raw_data(dinfo, yuvptr,
1720                        dinfo->max_v_samp_factor * dinfo->_min_DCT_scaled_size);
1721     if (usetmpbuf) {
1722       int j;
1723 
1724       for (i = 0; i < dinfo->num_components; i++) {
1725         for (j = 0; j < MIN(th[i], ph[i] - crow[i]); j++) {
1726           memcpy(outbuf[i][crow[i] + j], tmpbuf[i][j], pw[i]);
1727         }
1728       }
1729     }
1730   }
1731   jpeg_finish_decompress(dinfo);
1732 
1733 bailout:
1734   if (dinfo->global_state > DSTATE_START) jpeg_abort_decompress(dinfo);
1735   for (i = 0; i < MAX_COMPONENTS; i++) {
1736     if (tmpbuf[i]) free(tmpbuf[i]);
1737     if (outbuf[i]) free(outbuf[i]);
1738   }
1739   if (_tmpbuf) free(_tmpbuf);
1740   if (this->jerr.warning) retval = -1;
1741   this->jerr.stopOnWarning = FALSE;
1742   return retval;
1743 }
1744 
tjDecompressToYUV2(tjhandle handle,const unsigned char * jpegBuf,unsigned long jpegSize,unsigned char * dstBuf,int width,int pad,int height,int flags)1745 DLLEXPORT int tjDecompressToYUV2(tjhandle handle, const unsigned char *jpegBuf,
1746                                  unsigned long jpegSize, unsigned char *dstBuf,
1747                                  int width, int pad, int height, int flags)
1748 {
1749   unsigned char *dstPlanes[3];
1750   int pw0, ph0, strides[3], retval = -1, jpegSubsamp = -1;
1751   int i, jpegwidth, jpegheight, scaledw, scaledh;
1752 
1753   GET_DINSTANCE(handle);
1754   this->jerr.stopOnWarning = (flags & TJFLAG_STOPONWARNING) ? TRUE : FALSE;
1755 
1756   if (jpegBuf == NULL || jpegSize <= 0 || dstBuf == NULL || width < 0 ||
1757       pad < 1 || !IS_POW2(pad) || height < 0)
1758     THROW("tjDecompressToYUV2(): Invalid argument");
1759 
1760   if (setjmp(this->jerr.setjmp_buffer)) {
1761     /* If we get here, the JPEG code has signaled an error. */
1762     return -1;
1763   }
1764 
1765   jpeg_mem_src_tj(dinfo, jpegBuf, jpegSize);
1766   jpeg_read_header(dinfo, TRUE);
1767   jpegSubsamp = getSubsamp(dinfo);
1768   if (jpegSubsamp < 0)
1769     THROW("tjDecompressToYUV2(): Could not determine subsampling type for JPEG image");
1770 
1771   jpegwidth = dinfo->image_width;  jpegheight = dinfo->image_height;
1772   if (width == 0) width = jpegwidth;
1773   if (height == 0) height = jpegheight;
1774 
1775   for (i = 0; i < NUMSF; i++) {
1776     scaledw = TJSCALED(jpegwidth, sf[i]);
1777     scaledh = TJSCALED(jpegheight, sf[i]);
1778     if (scaledw <= width && scaledh <= height)
1779       break;
1780   }
1781   if (i >= NUMSF)
1782     THROW("tjDecompressToYUV2(): Could not scale down to desired image dimensions");
1783 
1784   pw0 = tjPlaneWidth(0, width, jpegSubsamp);
1785   ph0 = tjPlaneHeight(0, height, jpegSubsamp);
1786   dstPlanes[0] = dstBuf;
1787   strides[0] = PAD(pw0, pad);
1788   if (jpegSubsamp == TJSAMP_GRAY) {
1789     strides[1] = strides[2] = 0;
1790     dstPlanes[1] = dstPlanes[2] = NULL;
1791   } else {
1792     int pw1 = tjPlaneWidth(1, width, jpegSubsamp);
1793     int ph1 = tjPlaneHeight(1, height, jpegSubsamp);
1794 
1795     strides[1] = strides[2] = PAD(pw1, pad);
1796     dstPlanes[1] = dstPlanes[0] + strides[0] * ph0;
1797     dstPlanes[2] = dstPlanes[1] + strides[1] * ph1;
1798   }
1799 
1800   this->headerRead = 1;
1801   return tjDecompressToYUVPlanes(handle, jpegBuf, jpegSize, dstPlanes, width,
1802                                  strides, height, flags);
1803 
1804 bailout:
1805   this->jerr.stopOnWarning = FALSE;
1806   return retval;
1807 }
1808 
tjDecompressToYUV(tjhandle handle,unsigned char * jpegBuf,unsigned long jpegSize,unsigned char * dstBuf,int flags)1809 DLLEXPORT int tjDecompressToYUV(tjhandle handle, unsigned char *jpegBuf,
1810                                 unsigned long jpegSize, unsigned char *dstBuf,
1811                                 int flags)
1812 {
1813   return tjDecompressToYUV2(handle, jpegBuf, jpegSize, dstBuf, 0, 4, 0, flags);
1814 }
1815 
1816 
1817 /* Transformer */
1818 
tjInitTransform(void)1819 DLLEXPORT tjhandle tjInitTransform(void)
1820 {
1821   tjinstance *this = NULL;
1822   tjhandle handle = NULL;
1823 
1824   if ((this = (tjinstance *)malloc(sizeof(tjinstance))) == NULL) {
1825     snprintf(errStr, JMSG_LENGTH_MAX,
1826              "tjInitTransform(): Memory allocation failure");
1827     return NULL;
1828   }
1829   MEMZERO(this, sizeof(tjinstance));
1830   snprintf(this->errStr, JMSG_LENGTH_MAX, "No error");
1831   handle = _tjInitCompress(this);
1832   if (!handle) return NULL;
1833   handle = _tjInitDecompress(this);
1834   return handle;
1835 }
1836 
1837 
tjTransform(tjhandle handle,const unsigned char * jpegBuf,unsigned long jpegSize,int n,unsigned char ** dstBufs,unsigned long * dstSizes,tjtransform * t,int flags)1838 DLLEXPORT int tjTransform(tjhandle handle, const unsigned char *jpegBuf,
1839                           unsigned long jpegSize, int n,
1840                           unsigned char **dstBufs, unsigned long *dstSizes,
1841                           tjtransform *t, int flags)
1842 {
1843   jpeg_transform_info *xinfo = NULL;
1844   jvirt_barray_ptr *srccoefs, *dstcoefs;
1845   int retval = 0, i, jpegSubsamp, saveMarkers = 0;
1846 
1847   GET_INSTANCE(handle);
1848   this->jerr.stopOnWarning = (flags & TJFLAG_STOPONWARNING) ? TRUE : FALSE;
1849   if ((this->init & COMPRESS) == 0 || (this->init & DECOMPRESS) == 0)
1850     THROW("tjTransform(): Instance has not been initialized for transformation");
1851 
1852   if (jpegBuf == NULL || jpegSize <= 0 || n < 1 || dstBufs == NULL ||
1853       dstSizes == NULL || t == NULL || flags < 0)
1854     THROW("tjTransform(): Invalid argument");
1855 
1856 #ifndef NO_PUTENV
1857   if (flags & TJFLAG_FORCEMMX) putenv("JSIMD_FORCEMMX=1");
1858   else if (flags & TJFLAG_FORCESSE) putenv("JSIMD_FORCESSE=1");
1859   else if (flags & TJFLAG_FORCESSE2) putenv("JSIMD_FORCESSE2=1");
1860 #endif
1861 
1862   if ((xinfo =
1863        (jpeg_transform_info *)malloc(sizeof(jpeg_transform_info) * n)) == NULL)
1864     THROW("tjTransform(): Memory allocation failure");
1865   MEMZERO(xinfo, sizeof(jpeg_transform_info) * n);
1866 
1867   if (setjmp(this->jerr.setjmp_buffer)) {
1868     /* If we get here, the JPEG code has signaled an error. */
1869     retval = -1;  goto bailout;
1870   }
1871 
1872   jpeg_mem_src_tj(dinfo, jpegBuf, jpegSize);
1873 
1874   for (i = 0; i < n; i++) {
1875     xinfo[i].transform = xformtypes[t[i].op];
1876     xinfo[i].perfect = (t[i].options & TJXOPT_PERFECT) ? 1 : 0;
1877     xinfo[i].trim = (t[i].options & TJXOPT_TRIM) ? 1 : 0;
1878     xinfo[i].force_grayscale = (t[i].options & TJXOPT_GRAY) ? 1 : 0;
1879     xinfo[i].crop = (t[i].options & TJXOPT_CROP) ? 1 : 0;
1880     if (n != 1 && t[i].op == TJXOP_HFLIP) xinfo[i].slow_hflip = 1;
1881     else xinfo[i].slow_hflip = 0;
1882 
1883     if (xinfo[i].crop) {
1884       xinfo[i].crop_xoffset = t[i].r.x;  xinfo[i].crop_xoffset_set = JCROP_POS;
1885       xinfo[i].crop_yoffset = t[i].r.y;  xinfo[i].crop_yoffset_set = JCROP_POS;
1886       if (t[i].r.w != 0) {
1887         xinfo[i].crop_width = t[i].r.w;  xinfo[i].crop_width_set = JCROP_POS;
1888       } else
1889         xinfo[i].crop_width = JCROP_UNSET;
1890       if (t[i].r.h != 0) {
1891         xinfo[i].crop_height = t[i].r.h;  xinfo[i].crop_height_set = JCROP_POS;
1892       } else
1893         xinfo[i].crop_height = JCROP_UNSET;
1894     }
1895     if (!(t[i].options & TJXOPT_COPYNONE)) saveMarkers = 1;
1896   }
1897 
1898   jcopy_markers_setup(dinfo, saveMarkers ? JCOPYOPT_ALL : JCOPYOPT_NONE);
1899   jpeg_read_header(dinfo, TRUE);
1900   jpegSubsamp = getSubsamp(dinfo);
1901   if (jpegSubsamp < 0)
1902     THROW("tjTransform(): Could not determine subsampling type for JPEG image");
1903 
1904   for (i = 0; i < n; i++) {
1905     if (!jtransform_request_workspace(dinfo, &xinfo[i]))
1906       THROW("tjTransform(): Transform is not perfect");
1907 
1908     if (xinfo[i].crop) {
1909       if ((t[i].r.x % xinfo[i].iMCU_sample_width) != 0 ||
1910           (t[i].r.y % xinfo[i].iMCU_sample_height) != 0) {
1911         snprintf(errStr, JMSG_LENGTH_MAX,
1912                  "To crop this JPEG image, x must be a multiple of %d\n"
1913                  "and y must be a multiple of %d.\n",
1914                  xinfo[i].iMCU_sample_width, xinfo[i].iMCU_sample_height);
1915         retval = -1;  goto bailout;
1916       }
1917     }
1918   }
1919 
1920   srccoefs = jpeg_read_coefficients(dinfo);
1921 
1922   for (i = 0; i < n; i++) {
1923     int w, h, alloc = 1;
1924 
1925     if (!xinfo[i].crop) {
1926       w = dinfo->image_width;  h = dinfo->image_height;
1927     } else {
1928       w = xinfo[i].crop_width;  h = xinfo[i].crop_height;
1929     }
1930     if (flags & TJFLAG_NOREALLOC) {
1931       alloc = 0;  dstSizes[i] = tjBufSize(w, h, jpegSubsamp);
1932     }
1933     if (!(t[i].options & TJXOPT_NOOUTPUT))
1934       jpeg_mem_dest_tj(cinfo, &dstBufs[i], &dstSizes[i], alloc);
1935     jpeg_copy_critical_parameters(dinfo, cinfo);
1936     dstcoefs = jtransform_adjust_parameters(dinfo, cinfo, srccoefs, &xinfo[i]);
1937     if (flags & TJFLAG_PROGRESSIVE || t[i].options & TJXOPT_PROGRESSIVE)
1938       jpeg_simple_progression(cinfo);
1939     if (!(t[i].options & TJXOPT_NOOUTPUT)) {
1940       jpeg_write_coefficients(cinfo, dstcoefs);
1941       jcopy_markers_execute(dinfo, cinfo, t[i].options & TJXOPT_COPYNONE ?
1942                                           JCOPYOPT_NONE : JCOPYOPT_ALL);
1943     } else
1944       jinit_c_master_control(cinfo, TRUE);
1945     jtransform_execute_transformation(dinfo, cinfo, srccoefs, &xinfo[i]);
1946     if (t[i].customFilter) {
1947       int ci, y;
1948       JDIMENSION by;
1949 
1950       for (ci = 0; ci < cinfo->num_components; ci++) {
1951         jpeg_component_info *compptr = &cinfo->comp_info[ci];
1952         tjregion arrayRegion = {
1953           0, 0, compptr->width_in_blocks * DCTSIZE, DCTSIZE
1954         };
1955         tjregion planeRegion = {
1956           0, 0, compptr->width_in_blocks * DCTSIZE,
1957           compptr->height_in_blocks * DCTSIZE
1958         };
1959 
1960         for (by = 0; by < compptr->height_in_blocks;
1961              by += compptr->v_samp_factor) {
1962           JBLOCKARRAY barray = (dinfo->mem->access_virt_barray)
1963             ((j_common_ptr)dinfo, dstcoefs[ci], by, compptr->v_samp_factor,
1964              TRUE);
1965 
1966           for (y = 0; y < compptr->v_samp_factor; y++) {
1967             if (t[i].customFilter(barray[y][0], arrayRegion, planeRegion, ci,
1968                                   i, &t[i]) == -1)
1969               THROW("tjTransform(): Error in custom filter");
1970             arrayRegion.y += DCTSIZE;
1971           }
1972         }
1973       }
1974     }
1975     if (!(t[i].options & TJXOPT_NOOUTPUT)) jpeg_finish_compress(cinfo);
1976   }
1977 
1978   jpeg_finish_decompress(dinfo);
1979 
1980 bailout:
1981   if (cinfo->global_state > CSTATE_START) jpeg_abort_compress(cinfo);
1982   if (dinfo->global_state > DSTATE_START) jpeg_abort_decompress(dinfo);
1983   if (xinfo) free(xinfo);
1984   if (this->jerr.warning) retval = -1;
1985   this->jerr.stopOnWarning = FALSE;
1986   return retval;
1987 }
1988 
1989 
tjLoadImage(const char * filename,int * width,int align,int * height,int * pixelFormat,int flags)1990 DLLEXPORT unsigned char *tjLoadImage(const char *filename, int *width,
1991                                      int align, int *height, int *pixelFormat,
1992                                      int flags)
1993 {
1994   int retval = 0, tempc;
1995   size_t pitch;
1996   tjhandle handle = NULL;
1997   tjinstance *this;
1998   j_compress_ptr cinfo = NULL;
1999   cjpeg_source_ptr src;
2000   unsigned char *dstBuf = NULL;
2001   FILE *file = NULL;
2002   boolean invert;
2003 
2004   if (!filename || !width || align < 1 || !height || !pixelFormat ||
2005       *pixelFormat < TJPF_UNKNOWN || *pixelFormat >= TJ_NUMPF)
2006     THROWG("tjLoadImage(): Invalid argument");
2007   if ((align & (align - 1)) != 0)
2008     THROWG("tjLoadImage(): Alignment must be a power of 2");
2009 
2010   if ((handle = tjInitCompress()) == NULL) return NULL;
2011   this = (tjinstance *)handle;
2012   cinfo = &this->cinfo;
2013 
2014   if ((file = fopen(filename, "rb")) == NULL)
2015     THROW_UNIX("tjLoadImage(): Cannot open input file");
2016 
2017   if ((tempc = getc(file)) < 0 || ungetc(tempc, file) == EOF)
2018     THROW_UNIX("tjLoadImage(): Could not read input file")
2019   else if (tempc == EOF)
2020     THROWG("tjLoadImage(): Input file contains no data");
2021 
2022   if (setjmp(this->jerr.setjmp_buffer)) {
2023     /* If we get here, the JPEG code has signaled an error. */
2024     retval = -1;  goto bailout;
2025   }
2026 
2027   if (*pixelFormat == TJPF_UNKNOWN) cinfo->in_color_space = JCS_UNKNOWN;
2028   else cinfo->in_color_space = pf2cs[*pixelFormat];
2029   if (tempc == 'B') {
2030     if ((src = jinit_read_bmp(cinfo, FALSE)) == NULL)
2031       THROWG("tjLoadImage(): Could not initialize bitmap loader");
2032     invert = (flags & TJFLAG_BOTTOMUP) == 0;
2033   } else if (tempc == 'P') {
2034     if ((src = jinit_read_ppm(cinfo)) == NULL)
2035       THROWG("tjLoadImage(): Could not initialize bitmap loader");
2036     invert = (flags & TJFLAG_BOTTOMUP) != 0;
2037   } else
2038     THROWG("tjLoadImage(): Unsupported file type");
2039 
2040   src->input_file = file;
2041   (*src->start_input) (cinfo, src);
2042   (*cinfo->mem->realize_virt_arrays) ((j_common_ptr)cinfo);
2043 
2044   *width = cinfo->image_width;  *height = cinfo->image_height;
2045   *pixelFormat = cs2pf[cinfo->in_color_space];
2046 
2047   pitch = PAD((*width) * tjPixelSize[*pixelFormat], align);
2048   if ((unsigned long long)pitch * (unsigned long long)(*height) >
2049       (unsigned long long)((size_t)-1) ||
2050       (dstBuf = (unsigned char *)malloc(pitch * (*height))) == NULL)
2051     THROWG("tjLoadImage(): Memory allocation failure");
2052 
2053   if (setjmp(this->jerr.setjmp_buffer)) {
2054     /* If we get here, the JPEG code has signaled an error. */
2055     retval = -1;  goto bailout;
2056   }
2057 
2058   while (cinfo->next_scanline < cinfo->image_height) {
2059     int i, nlines = (*src->get_pixel_rows) (cinfo, src);
2060 
2061     for (i = 0; i < nlines; i++) {
2062       unsigned char *dstptr;
2063       int row;
2064 
2065       row = cinfo->next_scanline + i;
2066       if (invert) dstptr = &dstBuf[((*height) - row - 1) * pitch];
2067       else dstptr = &dstBuf[row * pitch];
2068       memcpy(dstptr, src->buffer[i], (*width) * tjPixelSize[*pixelFormat]);
2069     }
2070     cinfo->next_scanline += nlines;
2071   }
2072 
2073   (*src->finish_input) (cinfo, src);
2074 
2075 bailout:
2076   if (handle) tjDestroy(handle);
2077   if (file) fclose(file);
2078   if (retval < 0 && dstBuf) { free(dstBuf);  dstBuf = NULL; }
2079   return dstBuf;
2080 }
2081 
2082 
tjSaveImage(const char * filename,unsigned char * buffer,int width,int pitch,int height,int pixelFormat,int flags)2083 DLLEXPORT int tjSaveImage(const char *filename, unsigned char *buffer,
2084                           int width, int pitch, int height, int pixelFormat,
2085                           int flags)
2086 {
2087   int retval = 0;
2088   tjhandle handle = NULL;
2089   tjinstance *this;
2090   j_decompress_ptr dinfo = NULL;
2091   djpeg_dest_ptr dst;
2092   FILE *file = NULL;
2093   char *ptr = NULL;
2094   boolean invert;
2095 
2096   if (!filename || !buffer || width < 1 || pitch < 0 || height < 1 ||
2097       pixelFormat < 0 || pixelFormat >= TJ_NUMPF)
2098     THROWG("tjSaveImage(): Invalid argument");
2099 
2100   if ((handle = tjInitDecompress()) == NULL)
2101     return -1;
2102   this = (tjinstance *)handle;
2103   dinfo = &this->dinfo;
2104 
2105   if ((file = fopen(filename, "wb")) == NULL)
2106     THROW_UNIX("tjSaveImage(): Cannot open output file");
2107 
2108   if (setjmp(this->jerr.setjmp_buffer)) {
2109     /* If we get here, the JPEG code has signaled an error. */
2110     retval = -1;  goto bailout;
2111   }
2112 
2113   this->dinfo.out_color_space = pf2cs[pixelFormat];
2114   dinfo->image_width = width;  dinfo->image_height = height;
2115   dinfo->global_state = DSTATE_READY;
2116   dinfo->scale_num = dinfo->scale_denom = 1;
2117 
2118   ptr = strrchr(filename, '.');
2119   if (ptr && !strcasecmp(ptr, ".bmp")) {
2120     if ((dst = jinit_write_bmp(dinfo, FALSE, FALSE)) == NULL)
2121       THROWG("tjSaveImage(): Could not initialize bitmap writer");
2122     invert = (flags & TJFLAG_BOTTOMUP) == 0;
2123   } else {
2124     if ((dst = jinit_write_ppm(dinfo)) == NULL)
2125       THROWG("tjSaveImage(): Could not initialize PPM writer");
2126     invert = (flags & TJFLAG_BOTTOMUP) != 0;
2127   }
2128 
2129   dst->output_file = file;
2130   (*dst->start_output) (dinfo, dst);
2131   (*dinfo->mem->realize_virt_arrays) ((j_common_ptr)dinfo);
2132 
2133   if (pitch == 0) pitch = width * tjPixelSize[pixelFormat];
2134 
2135   while (dinfo->output_scanline < dinfo->output_height) {
2136     unsigned char *rowptr;
2137 
2138     if (invert)
2139       rowptr = &buffer[(height - dinfo->output_scanline - 1) * pitch];
2140     else
2141       rowptr = &buffer[dinfo->output_scanline * pitch];
2142     memcpy(dst->buffer[0], rowptr, width * tjPixelSize[pixelFormat]);
2143     (*dst->put_pixel_rows) (dinfo, dst, 1);
2144     dinfo->output_scanline++;
2145   }
2146 
2147   (*dst->finish_output) (dinfo, dst);
2148 
2149 bailout:
2150   if (handle) tjDestroy(handle);
2151   if (file) fclose(file);
2152   return retval;
2153 }
2154