1*dfc6aa5cSAndroid Build Coastguard Worker /*
2*dfc6aa5cSAndroid Build Coastguard Worker * Copyright (C)2009-2019, 2021-2023 D. R. Commander. All Rights Reserved.
3*dfc6aa5cSAndroid Build Coastguard Worker *
4*dfc6aa5cSAndroid Build Coastguard Worker * Redistribution and use in source and binary forms, with or without
5*dfc6aa5cSAndroid Build Coastguard Worker * modification, are permitted provided that the following conditions are met:
6*dfc6aa5cSAndroid Build Coastguard Worker *
7*dfc6aa5cSAndroid Build Coastguard Worker * - Redistributions of source code must retain the above copyright notice,
8*dfc6aa5cSAndroid Build Coastguard Worker * this list of conditions and the following disclaimer.
9*dfc6aa5cSAndroid Build Coastguard Worker * - Redistributions in binary form must reproduce the above copyright notice,
10*dfc6aa5cSAndroid Build Coastguard Worker * this list of conditions and the following disclaimer in the documentation
11*dfc6aa5cSAndroid Build Coastguard Worker * and/or other materials provided with the distribution.
12*dfc6aa5cSAndroid Build Coastguard Worker * - Neither the name of the libjpeg-turbo Project nor the names of its
13*dfc6aa5cSAndroid Build Coastguard Worker * contributors may be used to endorse or promote products derived from this
14*dfc6aa5cSAndroid Build Coastguard Worker * software without specific prior written permission.
15*dfc6aa5cSAndroid Build Coastguard Worker *
16*dfc6aa5cSAndroid Build Coastguard Worker * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS",
17*dfc6aa5cSAndroid Build Coastguard Worker * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
18*dfc6aa5cSAndroid Build Coastguard Worker * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
19*dfc6aa5cSAndroid Build Coastguard Worker * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDERS OR CONTRIBUTORS BE
20*dfc6aa5cSAndroid Build Coastguard Worker * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
21*dfc6aa5cSAndroid Build Coastguard Worker * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
22*dfc6aa5cSAndroid Build Coastguard Worker * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
23*dfc6aa5cSAndroid Build Coastguard Worker * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
24*dfc6aa5cSAndroid Build Coastguard Worker * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
25*dfc6aa5cSAndroid Build Coastguard Worker * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
26*dfc6aa5cSAndroid Build Coastguard Worker * POSSIBILITY OF SUCH DAMAGE.
27*dfc6aa5cSAndroid Build Coastguard Worker */
28*dfc6aa5cSAndroid Build Coastguard Worker
29*dfc6aa5cSAndroid Build Coastguard Worker #ifdef _MSC_VER
30*dfc6aa5cSAndroid Build Coastguard Worker #define _CRT_SECURE_NO_DEPRECATE
31*dfc6aa5cSAndroid Build Coastguard Worker #endif
32*dfc6aa5cSAndroid Build Coastguard Worker
33*dfc6aa5cSAndroid Build Coastguard Worker #include <stdio.h>
34*dfc6aa5cSAndroid Build Coastguard Worker #include <stdlib.h>
35*dfc6aa5cSAndroid Build Coastguard Worker #include <string.h>
36*dfc6aa5cSAndroid Build Coastguard Worker #include <ctype.h>
37*dfc6aa5cSAndroid Build Coastguard Worker #include <math.h>
38*dfc6aa5cSAndroid Build Coastguard Worker #include <errno.h>
39*dfc6aa5cSAndroid Build Coastguard Worker #include <limits.h>
40*dfc6aa5cSAndroid Build Coastguard Worker #include <cdjpeg.h>
41*dfc6aa5cSAndroid Build Coastguard Worker #include "./tjutil.h"
42*dfc6aa5cSAndroid Build Coastguard Worker #include "./turbojpeg.h"
43*dfc6aa5cSAndroid Build Coastguard Worker
44*dfc6aa5cSAndroid Build Coastguard Worker
45*dfc6aa5cSAndroid Build Coastguard Worker #define THROW(op, err) { \
46*dfc6aa5cSAndroid Build Coastguard Worker fprintf(stderr, "ERROR in line %d while %s:\n%s\n", __LINE__, op, err); \
47*dfc6aa5cSAndroid Build Coastguard Worker retval = -1; goto bailout; \
48*dfc6aa5cSAndroid Build Coastguard Worker }
49*dfc6aa5cSAndroid Build Coastguard Worker #define THROW_UNIX(m) THROW(m, strerror(errno))
50*dfc6aa5cSAndroid Build Coastguard Worker
51*dfc6aa5cSAndroid Build Coastguard Worker static char tjErrorStr[JMSG_LENGTH_MAX] = "\0",
52*dfc6aa5cSAndroid Build Coastguard Worker tjErrorMsg[JMSG_LENGTH_MAX] = "\0";
53*dfc6aa5cSAndroid Build Coastguard Worker static int tjErrorLine = -1, tjErrorCode = -1;
54*dfc6aa5cSAndroid Build Coastguard Worker
55*dfc6aa5cSAndroid Build Coastguard Worker #define THROW_TJG(m) { \
56*dfc6aa5cSAndroid Build Coastguard Worker fprintf(stderr, "ERROR in line %d while %s:\n%s\n", __LINE__, m, \
57*dfc6aa5cSAndroid Build Coastguard Worker tjGetErrorStr2(NULL)); \
58*dfc6aa5cSAndroid Build Coastguard Worker retval = -1; goto bailout; \
59*dfc6aa5cSAndroid Build Coastguard Worker }
60*dfc6aa5cSAndroid Build Coastguard Worker
61*dfc6aa5cSAndroid Build Coastguard Worker #define THROW_TJ(m) { \
62*dfc6aa5cSAndroid Build Coastguard Worker int _tjErrorCode = tjGetErrorCode(handle); \
63*dfc6aa5cSAndroid Build Coastguard Worker char *_tjErrorStr = tjGetErrorStr2(handle); \
64*dfc6aa5cSAndroid Build Coastguard Worker \
65*dfc6aa5cSAndroid Build Coastguard Worker if (!(flags & TJFLAG_STOPONWARNING) && _tjErrorCode == TJERR_WARNING) { \
66*dfc6aa5cSAndroid Build Coastguard Worker if (strncmp(tjErrorStr, _tjErrorStr, JMSG_LENGTH_MAX) || \
67*dfc6aa5cSAndroid Build Coastguard Worker strncmp(tjErrorMsg, m, JMSG_LENGTH_MAX) || \
68*dfc6aa5cSAndroid Build Coastguard Worker tjErrorCode != _tjErrorCode || tjErrorLine != __LINE__) { \
69*dfc6aa5cSAndroid Build Coastguard Worker strncpy(tjErrorStr, _tjErrorStr, JMSG_LENGTH_MAX); \
70*dfc6aa5cSAndroid Build Coastguard Worker tjErrorStr[JMSG_LENGTH_MAX - 1] = '\0'; \
71*dfc6aa5cSAndroid Build Coastguard Worker strncpy(tjErrorMsg, m, JMSG_LENGTH_MAX); \
72*dfc6aa5cSAndroid Build Coastguard Worker tjErrorMsg[JMSG_LENGTH_MAX - 1] = '\0'; \
73*dfc6aa5cSAndroid Build Coastguard Worker tjErrorCode = _tjErrorCode; \
74*dfc6aa5cSAndroid Build Coastguard Worker tjErrorLine = __LINE__; \
75*dfc6aa5cSAndroid Build Coastguard Worker fprintf(stderr, "WARNING in line %d while %s:\n%s\n", __LINE__, m, \
76*dfc6aa5cSAndroid Build Coastguard Worker _tjErrorStr); \
77*dfc6aa5cSAndroid Build Coastguard Worker } \
78*dfc6aa5cSAndroid Build Coastguard Worker } else { \
79*dfc6aa5cSAndroid Build Coastguard Worker fprintf(stderr, "%s in line %d while %s:\n%s\n", \
80*dfc6aa5cSAndroid Build Coastguard Worker _tjErrorCode == TJERR_WARNING ? "WARNING" : "ERROR", __LINE__, m, \
81*dfc6aa5cSAndroid Build Coastguard Worker _tjErrorStr); \
82*dfc6aa5cSAndroid Build Coastguard Worker retval = -1; goto bailout; \
83*dfc6aa5cSAndroid Build Coastguard Worker } \
84*dfc6aa5cSAndroid Build Coastguard Worker }
85*dfc6aa5cSAndroid Build Coastguard Worker
86*dfc6aa5cSAndroid Build Coastguard Worker static int flags = TJFLAG_NOREALLOC, compOnly = 0, decompOnly = 0, doYUV = 0,
87*dfc6aa5cSAndroid Build Coastguard Worker quiet = 0, doTile = 0, pf = TJPF_BGR, yuvAlign = 1, doWrite = 1;
88*dfc6aa5cSAndroid Build Coastguard Worker static char *ext = "ppm";
89*dfc6aa5cSAndroid Build Coastguard Worker static const char *pixFormatStr[TJ_NUMPF] = {
90*dfc6aa5cSAndroid Build Coastguard Worker "RGB", "BGR", "RGBX", "BGRX", "XBGR", "XRGB", "GRAY", "", "", "", "", "CMYK"
91*dfc6aa5cSAndroid Build Coastguard Worker };
92*dfc6aa5cSAndroid Build Coastguard Worker static const char *subNameLong[TJ_NUMSAMP] = {
93*dfc6aa5cSAndroid Build Coastguard Worker "4:4:4", "4:2:2", "4:2:0", "GRAY", "4:4:0", "4:1:1"
94*dfc6aa5cSAndroid Build Coastguard Worker };
95*dfc6aa5cSAndroid Build Coastguard Worker static const char *csName[TJ_NUMCS] = {
96*dfc6aa5cSAndroid Build Coastguard Worker "RGB", "YCbCr", "GRAY", "CMYK", "YCCK"
97*dfc6aa5cSAndroid Build Coastguard Worker };
98*dfc6aa5cSAndroid Build Coastguard Worker static const char *subName[TJ_NUMSAMP] = {
99*dfc6aa5cSAndroid Build Coastguard Worker "444", "422", "420", "GRAY", "440", "411"
100*dfc6aa5cSAndroid Build Coastguard Worker };
101*dfc6aa5cSAndroid Build Coastguard Worker static tjscalingfactor *scalingFactors = NULL, sf = { 1, 1 };
102*dfc6aa5cSAndroid Build Coastguard Worker static int nsf = 0, xformOp = TJXOP_NONE, xformOpt = 0;
103*dfc6aa5cSAndroid Build Coastguard Worker static int (*customFilter) (short *, tjregion, tjregion, int, int,
104*dfc6aa5cSAndroid Build Coastguard Worker tjtransform *);
105*dfc6aa5cSAndroid Build Coastguard Worker static double benchTime = 5.0, warmup = 1.0;
106*dfc6aa5cSAndroid Build Coastguard Worker
107*dfc6aa5cSAndroid Build Coastguard Worker
formatName(int subsamp,int cs,char * buf)108*dfc6aa5cSAndroid Build Coastguard Worker static char *formatName(int subsamp, int cs, char *buf)
109*dfc6aa5cSAndroid Build Coastguard Worker {
110*dfc6aa5cSAndroid Build Coastguard Worker if (cs == TJCS_YCbCr)
111*dfc6aa5cSAndroid Build Coastguard Worker return (char *)subNameLong[subsamp];
112*dfc6aa5cSAndroid Build Coastguard Worker else if (cs == TJCS_YCCK || cs == TJCS_CMYK) {
113*dfc6aa5cSAndroid Build Coastguard Worker SNPRINTF(buf, 80, "%s %s", csName[cs], subNameLong[subsamp]);
114*dfc6aa5cSAndroid Build Coastguard Worker return buf;
115*dfc6aa5cSAndroid Build Coastguard Worker } else
116*dfc6aa5cSAndroid Build Coastguard Worker return (char *)csName[cs];
117*dfc6aa5cSAndroid Build Coastguard Worker }
118*dfc6aa5cSAndroid Build Coastguard Worker
119*dfc6aa5cSAndroid Build Coastguard Worker
sigfig(double val,int figs,char * buf,int len)120*dfc6aa5cSAndroid Build Coastguard Worker static char *sigfig(double val, int figs, char *buf, int len)
121*dfc6aa5cSAndroid Build Coastguard Worker {
122*dfc6aa5cSAndroid Build Coastguard Worker char format[80];
123*dfc6aa5cSAndroid Build Coastguard Worker int digitsAfterDecimal = figs - (int)ceil(log10(fabs(val)));
124*dfc6aa5cSAndroid Build Coastguard Worker
125*dfc6aa5cSAndroid Build Coastguard Worker if (digitsAfterDecimal < 1)
126*dfc6aa5cSAndroid Build Coastguard Worker SNPRINTF(format, 80, "%%.0f");
127*dfc6aa5cSAndroid Build Coastguard Worker else
128*dfc6aa5cSAndroid Build Coastguard Worker SNPRINTF(format, 80, "%%.%df", digitsAfterDecimal);
129*dfc6aa5cSAndroid Build Coastguard Worker SNPRINTF(buf, len, format, val);
130*dfc6aa5cSAndroid Build Coastguard Worker return buf;
131*dfc6aa5cSAndroid Build Coastguard Worker }
132*dfc6aa5cSAndroid Build Coastguard Worker
133*dfc6aa5cSAndroid Build Coastguard Worker
134*dfc6aa5cSAndroid Build Coastguard Worker /* Custom DCT filter which produces a negative of the image */
dummyDCTFilter(short * coeffs,tjregion arrayRegion,tjregion planeRegion,int componentIndex,int transformIndex,tjtransform * transform)135*dfc6aa5cSAndroid Build Coastguard Worker static int dummyDCTFilter(short *coeffs, tjregion arrayRegion,
136*dfc6aa5cSAndroid Build Coastguard Worker tjregion planeRegion, int componentIndex,
137*dfc6aa5cSAndroid Build Coastguard Worker int transformIndex, tjtransform *transform)
138*dfc6aa5cSAndroid Build Coastguard Worker {
139*dfc6aa5cSAndroid Build Coastguard Worker int i;
140*dfc6aa5cSAndroid Build Coastguard Worker
141*dfc6aa5cSAndroid Build Coastguard Worker for (i = 0; i < arrayRegion.w * arrayRegion.h; i++)
142*dfc6aa5cSAndroid Build Coastguard Worker coeffs[i] = -coeffs[i];
143*dfc6aa5cSAndroid Build Coastguard Worker return 0;
144*dfc6aa5cSAndroid Build Coastguard Worker }
145*dfc6aa5cSAndroid Build Coastguard Worker
146*dfc6aa5cSAndroid Build Coastguard Worker
147*dfc6aa5cSAndroid Build Coastguard Worker /* Decompression test */
decomp(unsigned char * srcBuf,unsigned char ** jpegBuf,unsigned long * jpegSize,unsigned char * dstBuf,int w,int h,int subsamp,int jpegQual,char * fileName,int tilew,int tileh)148*dfc6aa5cSAndroid Build Coastguard Worker static int decomp(unsigned char *srcBuf, unsigned char **jpegBuf,
149*dfc6aa5cSAndroid Build Coastguard Worker unsigned long *jpegSize, unsigned char *dstBuf, int w, int h,
150*dfc6aa5cSAndroid Build Coastguard Worker int subsamp, int jpegQual, char *fileName, int tilew,
151*dfc6aa5cSAndroid Build Coastguard Worker int tileh)
152*dfc6aa5cSAndroid Build Coastguard Worker {
153*dfc6aa5cSAndroid Build Coastguard Worker char tempStr[1024], sizeStr[24] = "\0", qualStr[13] = "\0", *ptr;
154*dfc6aa5cSAndroid Build Coastguard Worker FILE *file = NULL;
155*dfc6aa5cSAndroid Build Coastguard Worker tjhandle handle = NULL;
156*dfc6aa5cSAndroid Build Coastguard Worker int row, col, iter = 0, dstBufAlloc = 0, retval = 0;
157*dfc6aa5cSAndroid Build Coastguard Worker double elapsed, elapsedDecode;
158*dfc6aa5cSAndroid Build Coastguard Worker int ps = tjPixelSize[pf];
159*dfc6aa5cSAndroid Build Coastguard Worker int scaledw = TJSCALED(w, sf);
160*dfc6aa5cSAndroid Build Coastguard Worker int scaledh = TJSCALED(h, sf);
161*dfc6aa5cSAndroid Build Coastguard Worker int pitch = scaledw * ps;
162*dfc6aa5cSAndroid Build Coastguard Worker int ntilesw = (w + tilew - 1) / tilew, ntilesh = (h + tileh - 1) / tileh;
163*dfc6aa5cSAndroid Build Coastguard Worker unsigned char *dstPtr, *dstPtr2, *yuvBuf = NULL;
164*dfc6aa5cSAndroid Build Coastguard Worker
165*dfc6aa5cSAndroid Build Coastguard Worker if (jpegQual > 0) {
166*dfc6aa5cSAndroid Build Coastguard Worker SNPRINTF(qualStr, 13, "_Q%d", jpegQual);
167*dfc6aa5cSAndroid Build Coastguard Worker qualStr[12] = 0;
168*dfc6aa5cSAndroid Build Coastguard Worker }
169*dfc6aa5cSAndroid Build Coastguard Worker
170*dfc6aa5cSAndroid Build Coastguard Worker if ((handle = tjInitDecompress()) == NULL)
171*dfc6aa5cSAndroid Build Coastguard Worker THROW_TJ("executing tjInitDecompress()");
172*dfc6aa5cSAndroid Build Coastguard Worker
173*dfc6aa5cSAndroid Build Coastguard Worker if (dstBuf == NULL) {
174*dfc6aa5cSAndroid Build Coastguard Worker if ((unsigned long long)pitch * (unsigned long long)scaledh >
175*dfc6aa5cSAndroid Build Coastguard Worker (unsigned long long)((size_t)-1))
176*dfc6aa5cSAndroid Build Coastguard Worker THROW("allocating destination buffer", "Image is too large");
177*dfc6aa5cSAndroid Build Coastguard Worker if ((dstBuf = (unsigned char *)malloc((size_t)pitch * scaledh)) == NULL)
178*dfc6aa5cSAndroid Build Coastguard Worker THROW_UNIX("allocating destination buffer");
179*dfc6aa5cSAndroid Build Coastguard Worker dstBufAlloc = 1;
180*dfc6aa5cSAndroid Build Coastguard Worker }
181*dfc6aa5cSAndroid Build Coastguard Worker /* Set the destination buffer to gray so we know whether the decompressor
182*dfc6aa5cSAndroid Build Coastguard Worker attempted to write to it */
183*dfc6aa5cSAndroid Build Coastguard Worker memset(dstBuf, 127, (size_t)pitch * scaledh);
184*dfc6aa5cSAndroid Build Coastguard Worker
185*dfc6aa5cSAndroid Build Coastguard Worker if (doYUV) {
186*dfc6aa5cSAndroid Build Coastguard Worker int width = doTile ? tilew : scaledw;
187*dfc6aa5cSAndroid Build Coastguard Worker int height = doTile ? tileh : scaledh;
188*dfc6aa5cSAndroid Build Coastguard Worker unsigned long yuvSize = tjBufSizeYUV2(width, yuvAlign, height, subsamp);
189*dfc6aa5cSAndroid Build Coastguard Worker
190*dfc6aa5cSAndroid Build Coastguard Worker if (yuvSize == (unsigned long)-1)
191*dfc6aa5cSAndroid Build Coastguard Worker THROW_TJ("allocating YUV buffer");
192*dfc6aa5cSAndroid Build Coastguard Worker if ((yuvBuf = (unsigned char *)malloc(yuvSize)) == NULL)
193*dfc6aa5cSAndroid Build Coastguard Worker THROW_UNIX("allocating YUV buffer");
194*dfc6aa5cSAndroid Build Coastguard Worker memset(yuvBuf, 127, yuvSize);
195*dfc6aa5cSAndroid Build Coastguard Worker }
196*dfc6aa5cSAndroid Build Coastguard Worker
197*dfc6aa5cSAndroid Build Coastguard Worker /* Benchmark */
198*dfc6aa5cSAndroid Build Coastguard Worker iter = -1;
199*dfc6aa5cSAndroid Build Coastguard Worker elapsed = elapsedDecode = 0.;
200*dfc6aa5cSAndroid Build Coastguard Worker while (1) {
201*dfc6aa5cSAndroid Build Coastguard Worker int tile = 0;
202*dfc6aa5cSAndroid Build Coastguard Worker double start = getTime();
203*dfc6aa5cSAndroid Build Coastguard Worker
204*dfc6aa5cSAndroid Build Coastguard Worker for (row = 0, dstPtr = dstBuf; row < ntilesh;
205*dfc6aa5cSAndroid Build Coastguard Worker row++, dstPtr += (size_t)pitch * tileh) {
206*dfc6aa5cSAndroid Build Coastguard Worker for (col = 0, dstPtr2 = dstPtr; col < ntilesw;
207*dfc6aa5cSAndroid Build Coastguard Worker col++, tile++, dstPtr2 += ps * tilew) {
208*dfc6aa5cSAndroid Build Coastguard Worker int width = doTile ? min(tilew, w - col * tilew) : scaledw;
209*dfc6aa5cSAndroid Build Coastguard Worker int height = doTile ? min(tileh, h - row * tileh) : scaledh;
210*dfc6aa5cSAndroid Build Coastguard Worker
211*dfc6aa5cSAndroid Build Coastguard Worker if (doYUV) {
212*dfc6aa5cSAndroid Build Coastguard Worker double startDecode;
213*dfc6aa5cSAndroid Build Coastguard Worker
214*dfc6aa5cSAndroid Build Coastguard Worker if (tjDecompressToYUV2(handle, jpegBuf[tile], jpegSize[tile], yuvBuf,
215*dfc6aa5cSAndroid Build Coastguard Worker width, yuvAlign, height, flags) == -1)
216*dfc6aa5cSAndroid Build Coastguard Worker THROW_TJ("executing tjDecompressToYUV2()");
217*dfc6aa5cSAndroid Build Coastguard Worker startDecode = getTime();
218*dfc6aa5cSAndroid Build Coastguard Worker if (tjDecodeYUV(handle, yuvBuf, yuvAlign, subsamp, dstPtr2, width,
219*dfc6aa5cSAndroid Build Coastguard Worker pitch, height, pf, flags) == -1)
220*dfc6aa5cSAndroid Build Coastguard Worker THROW_TJ("executing tjDecodeYUV()");
221*dfc6aa5cSAndroid Build Coastguard Worker if (iter >= 0) elapsedDecode += getTime() - startDecode;
222*dfc6aa5cSAndroid Build Coastguard Worker } else if (tjDecompress2(handle, jpegBuf[tile], jpegSize[tile],
223*dfc6aa5cSAndroid Build Coastguard Worker dstPtr2, width, pitch, height, pf,
224*dfc6aa5cSAndroid Build Coastguard Worker flags) == -1)
225*dfc6aa5cSAndroid Build Coastguard Worker THROW_TJ("executing tjDecompress2()");
226*dfc6aa5cSAndroid Build Coastguard Worker }
227*dfc6aa5cSAndroid Build Coastguard Worker }
228*dfc6aa5cSAndroid Build Coastguard Worker elapsed += getTime() - start;
229*dfc6aa5cSAndroid Build Coastguard Worker if (iter >= 0) {
230*dfc6aa5cSAndroid Build Coastguard Worker iter++;
231*dfc6aa5cSAndroid Build Coastguard Worker if (elapsed >= benchTime) break;
232*dfc6aa5cSAndroid Build Coastguard Worker } else if (elapsed >= warmup) {
233*dfc6aa5cSAndroid Build Coastguard Worker iter = 0;
234*dfc6aa5cSAndroid Build Coastguard Worker elapsed = elapsedDecode = 0.;
235*dfc6aa5cSAndroid Build Coastguard Worker }
236*dfc6aa5cSAndroid Build Coastguard Worker }
237*dfc6aa5cSAndroid Build Coastguard Worker if (doYUV) elapsed -= elapsedDecode;
238*dfc6aa5cSAndroid Build Coastguard Worker
239*dfc6aa5cSAndroid Build Coastguard Worker if (tjDestroy(handle) == -1) THROW_TJ("executing tjDestroy()");
240*dfc6aa5cSAndroid Build Coastguard Worker handle = NULL;
241*dfc6aa5cSAndroid Build Coastguard Worker
242*dfc6aa5cSAndroid Build Coastguard Worker if (quiet) {
243*dfc6aa5cSAndroid Build Coastguard Worker fprintf(stderr, "%-6s%s",
244*dfc6aa5cSAndroid Build Coastguard Worker sigfig((double)(w * h) / 1000000. * (double)iter / elapsed, 4,
245*dfc6aa5cSAndroid Build Coastguard Worker tempStr, 1024),
246*dfc6aa5cSAndroid Build Coastguard Worker quiet == 2 ? "\n" : " ");
247*dfc6aa5cSAndroid Build Coastguard Worker if (doYUV)
248*dfc6aa5cSAndroid Build Coastguard Worker fprintf(stderr, "%s\n",
249*dfc6aa5cSAndroid Build Coastguard Worker sigfig((double)(w * h) / 1000000. * (double)iter / elapsedDecode,
250*dfc6aa5cSAndroid Build Coastguard Worker 4, tempStr, 1024));
251*dfc6aa5cSAndroid Build Coastguard Worker else if (quiet != 2) fprintf(stderr, "\n");
252*dfc6aa5cSAndroid Build Coastguard Worker } else {
253*dfc6aa5cSAndroid Build Coastguard Worker fprintf(stderr, "%s --> Frame rate: %f fps\n",
254*dfc6aa5cSAndroid Build Coastguard Worker doYUV ? "Decomp to YUV" : "Decompress ", (double)iter / elapsed);
255*dfc6aa5cSAndroid Build Coastguard Worker fprintf(stderr,
256*dfc6aa5cSAndroid Build Coastguard Worker " Throughput: %f Megapixels/sec\n",
257*dfc6aa5cSAndroid Build Coastguard Worker (double)(w * h) / 1000000. * (double)iter / elapsed);
258*dfc6aa5cSAndroid Build Coastguard Worker if (doYUV) {
259*dfc6aa5cSAndroid Build Coastguard Worker fprintf(stderr, "YUV Decode --> Frame rate: %f fps\n",
260*dfc6aa5cSAndroid Build Coastguard Worker (double)iter / elapsedDecode);
261*dfc6aa5cSAndroid Build Coastguard Worker fprintf(stderr,
262*dfc6aa5cSAndroid Build Coastguard Worker " Throughput: %f Megapixels/sec\n",
263*dfc6aa5cSAndroid Build Coastguard Worker (double)(w * h) / 1000000. * (double)iter / elapsedDecode);
264*dfc6aa5cSAndroid Build Coastguard Worker }
265*dfc6aa5cSAndroid Build Coastguard Worker }
266*dfc6aa5cSAndroid Build Coastguard Worker
267*dfc6aa5cSAndroid Build Coastguard Worker if (!doWrite) goto bailout;
268*dfc6aa5cSAndroid Build Coastguard Worker
269*dfc6aa5cSAndroid Build Coastguard Worker if (sf.num != 1 || sf.denom != 1)
270*dfc6aa5cSAndroid Build Coastguard Worker SNPRINTF(sizeStr, 24, "%d_%d", sf.num, sf.denom);
271*dfc6aa5cSAndroid Build Coastguard Worker else if (tilew != w || tileh != h)
272*dfc6aa5cSAndroid Build Coastguard Worker SNPRINTF(sizeStr, 24, "%dx%d", tilew, tileh);
273*dfc6aa5cSAndroid Build Coastguard Worker else SNPRINTF(sizeStr, 24, "full");
274*dfc6aa5cSAndroid Build Coastguard Worker if (decompOnly)
275*dfc6aa5cSAndroid Build Coastguard Worker SNPRINTF(tempStr, 1024, "%s_%s.%s", fileName, sizeStr, ext);
276*dfc6aa5cSAndroid Build Coastguard Worker else
277*dfc6aa5cSAndroid Build Coastguard Worker SNPRINTF(tempStr, 1024, "%s_%s%s_%s.%s", fileName, subName[subsamp],
278*dfc6aa5cSAndroid Build Coastguard Worker qualStr, sizeStr, ext);
279*dfc6aa5cSAndroid Build Coastguard Worker
280*dfc6aa5cSAndroid Build Coastguard Worker if (tjSaveImage(tempStr, dstBuf, scaledw, 0, scaledh, pf, flags) == -1)
281*dfc6aa5cSAndroid Build Coastguard Worker THROW_TJG("saving output image");
282*dfc6aa5cSAndroid Build Coastguard Worker ptr = strrchr(tempStr, '.');
283*dfc6aa5cSAndroid Build Coastguard Worker SNPRINTF(ptr, 1024 - (ptr - tempStr), "-err.%s", ext);
284*dfc6aa5cSAndroid Build Coastguard Worker if (srcBuf && sf.num == 1 && sf.denom == 1) {
285*dfc6aa5cSAndroid Build Coastguard Worker if (!quiet) fprintf(stderr, "Compression error written to %s.\n", tempStr);
286*dfc6aa5cSAndroid Build Coastguard Worker if (subsamp == TJSAMP_GRAY) {
287*dfc6aa5cSAndroid Build Coastguard Worker unsigned long index, index2;
288*dfc6aa5cSAndroid Build Coastguard Worker
289*dfc6aa5cSAndroid Build Coastguard Worker for (row = 0, index = 0; row < h; row++, index += pitch) {
290*dfc6aa5cSAndroid Build Coastguard Worker for (col = 0, index2 = index; col < w; col++, index2 += ps) {
291*dfc6aa5cSAndroid Build Coastguard Worker unsigned long rindex = index2 + tjRedOffset[pf];
292*dfc6aa5cSAndroid Build Coastguard Worker unsigned long gindex = index2 + tjGreenOffset[pf];
293*dfc6aa5cSAndroid Build Coastguard Worker unsigned long bindex = index2 + tjBlueOffset[pf];
294*dfc6aa5cSAndroid Build Coastguard Worker int y = (int)((double)srcBuf[rindex] * 0.299 +
295*dfc6aa5cSAndroid Build Coastguard Worker (double)srcBuf[gindex] * 0.587 +
296*dfc6aa5cSAndroid Build Coastguard Worker (double)srcBuf[bindex] * 0.114 + 0.5);
297*dfc6aa5cSAndroid Build Coastguard Worker
298*dfc6aa5cSAndroid Build Coastguard Worker if (y > 255) y = 255;
299*dfc6aa5cSAndroid Build Coastguard Worker if (y < 0) y = 0;
300*dfc6aa5cSAndroid Build Coastguard Worker dstBuf[rindex] = (unsigned char)abs(dstBuf[rindex] - y);
301*dfc6aa5cSAndroid Build Coastguard Worker dstBuf[gindex] = (unsigned char)abs(dstBuf[gindex] - y);
302*dfc6aa5cSAndroid Build Coastguard Worker dstBuf[bindex] = (unsigned char)abs(dstBuf[bindex] - y);
303*dfc6aa5cSAndroid Build Coastguard Worker }
304*dfc6aa5cSAndroid Build Coastguard Worker }
305*dfc6aa5cSAndroid Build Coastguard Worker } else {
306*dfc6aa5cSAndroid Build Coastguard Worker for (row = 0; row < h; row++)
307*dfc6aa5cSAndroid Build Coastguard Worker for (col = 0; col < w * ps; col++)
308*dfc6aa5cSAndroid Build Coastguard Worker dstBuf[pitch * row + col] =
309*dfc6aa5cSAndroid Build Coastguard Worker (unsigned char)abs(dstBuf[pitch * row + col] -
310*dfc6aa5cSAndroid Build Coastguard Worker srcBuf[pitch * row + col]);
311*dfc6aa5cSAndroid Build Coastguard Worker }
312*dfc6aa5cSAndroid Build Coastguard Worker if (tjSaveImage(tempStr, dstBuf, w, 0, h, pf, flags) == -1)
313*dfc6aa5cSAndroid Build Coastguard Worker THROW_TJG("saving output image");
314*dfc6aa5cSAndroid Build Coastguard Worker }
315*dfc6aa5cSAndroid Build Coastguard Worker
316*dfc6aa5cSAndroid Build Coastguard Worker bailout:
317*dfc6aa5cSAndroid Build Coastguard Worker if (file) fclose(file);
318*dfc6aa5cSAndroid Build Coastguard Worker if (handle) tjDestroy(handle);
319*dfc6aa5cSAndroid Build Coastguard Worker if (dstBufAlloc) free(dstBuf);
320*dfc6aa5cSAndroid Build Coastguard Worker free(yuvBuf);
321*dfc6aa5cSAndroid Build Coastguard Worker return retval;
322*dfc6aa5cSAndroid Build Coastguard Worker }
323*dfc6aa5cSAndroid Build Coastguard Worker
324*dfc6aa5cSAndroid Build Coastguard Worker
fullTest(unsigned char * srcBuf,int w,int h,int subsamp,int jpegQual,char * fileName)325*dfc6aa5cSAndroid Build Coastguard Worker static int fullTest(unsigned char *srcBuf, int w, int h, int subsamp,
326*dfc6aa5cSAndroid Build Coastguard Worker int jpegQual, char *fileName)
327*dfc6aa5cSAndroid Build Coastguard Worker {
328*dfc6aa5cSAndroid Build Coastguard Worker char tempStr[1024], tempStr2[80];
329*dfc6aa5cSAndroid Build Coastguard Worker FILE *file = NULL;
330*dfc6aa5cSAndroid Build Coastguard Worker tjhandle handle = NULL;
331*dfc6aa5cSAndroid Build Coastguard Worker unsigned char **jpegBuf = NULL, *yuvBuf = NULL, *tmpBuf = NULL, *srcPtr,
332*dfc6aa5cSAndroid Build Coastguard Worker *srcPtr2;
333*dfc6aa5cSAndroid Build Coastguard Worker double start, elapsed, elapsedEncode;
334*dfc6aa5cSAndroid Build Coastguard Worker int totalJpegSize = 0, row, col, i, tilew = w, tileh = h, retval = 0;
335*dfc6aa5cSAndroid Build Coastguard Worker int iter;
336*dfc6aa5cSAndroid Build Coastguard Worker unsigned long *jpegSize = NULL, yuvSize = 0;
337*dfc6aa5cSAndroid Build Coastguard Worker int ps = tjPixelSize[pf];
338*dfc6aa5cSAndroid Build Coastguard Worker int ntilesw = 1, ntilesh = 1, pitch = w * ps;
339*dfc6aa5cSAndroid Build Coastguard Worker const char *pfStr = pixFormatStr[pf];
340*dfc6aa5cSAndroid Build Coastguard Worker
341*dfc6aa5cSAndroid Build Coastguard Worker if ((unsigned long long)pitch * (unsigned long long)h >
342*dfc6aa5cSAndroid Build Coastguard Worker (unsigned long long)((size_t)-1))
343*dfc6aa5cSAndroid Build Coastguard Worker THROW("allocating temporary image buffer", "Image is too large");
344*dfc6aa5cSAndroid Build Coastguard Worker if ((tmpBuf = (unsigned char *)malloc((size_t)pitch * h)) == NULL)
345*dfc6aa5cSAndroid Build Coastguard Worker THROW_UNIX("allocating temporary image buffer");
346*dfc6aa5cSAndroid Build Coastguard Worker
347*dfc6aa5cSAndroid Build Coastguard Worker if (!quiet)
348*dfc6aa5cSAndroid Build Coastguard Worker fprintf(stderr, ">>>>> %s (%s) <--> JPEG %s Q%d <<<<<\n", pfStr,
349*dfc6aa5cSAndroid Build Coastguard Worker (flags & TJFLAG_BOTTOMUP) ? "Bottom-up" : "Top-down",
350*dfc6aa5cSAndroid Build Coastguard Worker subNameLong[subsamp], jpegQual);
351*dfc6aa5cSAndroid Build Coastguard Worker
352*dfc6aa5cSAndroid Build Coastguard Worker for (tilew = doTile ? 8 : w, tileh = doTile ? 8 : h; ;
353*dfc6aa5cSAndroid Build Coastguard Worker tilew *= 2, tileh *= 2) {
354*dfc6aa5cSAndroid Build Coastguard Worker if (tilew > w) tilew = w;
355*dfc6aa5cSAndroid Build Coastguard Worker if (tileh > h) tileh = h;
356*dfc6aa5cSAndroid Build Coastguard Worker ntilesw = (w + tilew - 1) / tilew;
357*dfc6aa5cSAndroid Build Coastguard Worker ntilesh = (h + tileh - 1) / tileh;
358*dfc6aa5cSAndroid Build Coastguard Worker
359*dfc6aa5cSAndroid Build Coastguard Worker if ((jpegBuf = (unsigned char **)malloc(sizeof(unsigned char *) *
360*dfc6aa5cSAndroid Build Coastguard Worker ntilesw * ntilesh)) == NULL)
361*dfc6aa5cSAndroid Build Coastguard Worker THROW_UNIX("allocating JPEG tile array");
362*dfc6aa5cSAndroid Build Coastguard Worker memset(jpegBuf, 0, sizeof(unsigned char *) * ntilesw * ntilesh);
363*dfc6aa5cSAndroid Build Coastguard Worker if ((jpegSize = (unsigned long *)malloc(sizeof(unsigned long) *
364*dfc6aa5cSAndroid Build Coastguard Worker ntilesw * ntilesh)) == NULL)
365*dfc6aa5cSAndroid Build Coastguard Worker THROW_UNIX("allocating JPEG size array");
366*dfc6aa5cSAndroid Build Coastguard Worker memset(jpegSize, 0, sizeof(unsigned long) * ntilesw * ntilesh);
367*dfc6aa5cSAndroid Build Coastguard Worker
368*dfc6aa5cSAndroid Build Coastguard Worker if ((flags & TJFLAG_NOREALLOC) != 0)
369*dfc6aa5cSAndroid Build Coastguard Worker for (i = 0; i < ntilesw * ntilesh; i++) {
370*dfc6aa5cSAndroid Build Coastguard Worker if (tjBufSize(tilew, tileh, subsamp) > (unsigned long)INT_MAX)
371*dfc6aa5cSAndroid Build Coastguard Worker THROW("getting buffer size", "Image is too large");
372*dfc6aa5cSAndroid Build Coastguard Worker if ((jpegBuf[i] = (unsigned char *)
373*dfc6aa5cSAndroid Build Coastguard Worker tjAlloc(tjBufSize(tilew, tileh, subsamp))) == NULL)
374*dfc6aa5cSAndroid Build Coastguard Worker THROW_UNIX("allocating JPEG tiles");
375*dfc6aa5cSAndroid Build Coastguard Worker }
376*dfc6aa5cSAndroid Build Coastguard Worker
377*dfc6aa5cSAndroid Build Coastguard Worker /* Compression test */
378*dfc6aa5cSAndroid Build Coastguard Worker if (quiet == 1)
379*dfc6aa5cSAndroid Build Coastguard Worker fprintf(stderr, "%-4s (%s) %-5s %-3d ", pfStr,
380*dfc6aa5cSAndroid Build Coastguard Worker (flags & TJFLAG_BOTTOMUP) ? "BU" : "TD", subNameLong[subsamp],
381*dfc6aa5cSAndroid Build Coastguard Worker jpegQual);
382*dfc6aa5cSAndroid Build Coastguard Worker for (i = 0; i < h; i++)
383*dfc6aa5cSAndroid Build Coastguard Worker memcpy(&tmpBuf[pitch * i], &srcBuf[w * ps * i], w * ps);
384*dfc6aa5cSAndroid Build Coastguard Worker if ((handle = tjInitCompress()) == NULL)
385*dfc6aa5cSAndroid Build Coastguard Worker THROW_TJ("executing tjInitCompress()");
386*dfc6aa5cSAndroid Build Coastguard Worker
387*dfc6aa5cSAndroid Build Coastguard Worker if (doYUV) {
388*dfc6aa5cSAndroid Build Coastguard Worker yuvSize = tjBufSizeYUV2(tilew, yuvAlign, tileh, subsamp);
389*dfc6aa5cSAndroid Build Coastguard Worker if (yuvSize == (unsigned long)-1)
390*dfc6aa5cSAndroid Build Coastguard Worker THROW_TJ("allocating YUV buffer");
391*dfc6aa5cSAndroid Build Coastguard Worker if ((yuvBuf = (unsigned char *)malloc(yuvSize)) == NULL)
392*dfc6aa5cSAndroid Build Coastguard Worker THROW_UNIX("allocating YUV buffer");
393*dfc6aa5cSAndroid Build Coastguard Worker memset(yuvBuf, 127, yuvSize);
394*dfc6aa5cSAndroid Build Coastguard Worker }
395*dfc6aa5cSAndroid Build Coastguard Worker
396*dfc6aa5cSAndroid Build Coastguard Worker /* Benchmark */
397*dfc6aa5cSAndroid Build Coastguard Worker iter = -1;
398*dfc6aa5cSAndroid Build Coastguard Worker elapsed = elapsedEncode = 0.;
399*dfc6aa5cSAndroid Build Coastguard Worker while (1) {
400*dfc6aa5cSAndroid Build Coastguard Worker int tile = 0;
401*dfc6aa5cSAndroid Build Coastguard Worker
402*dfc6aa5cSAndroid Build Coastguard Worker totalJpegSize = 0;
403*dfc6aa5cSAndroid Build Coastguard Worker start = getTime();
404*dfc6aa5cSAndroid Build Coastguard Worker for (row = 0, srcPtr = srcBuf; row < ntilesh;
405*dfc6aa5cSAndroid Build Coastguard Worker row++, srcPtr += pitch * tileh) {
406*dfc6aa5cSAndroid Build Coastguard Worker for (col = 0, srcPtr2 = srcPtr; col < ntilesw;
407*dfc6aa5cSAndroid Build Coastguard Worker col++, tile++, srcPtr2 += ps * tilew) {
408*dfc6aa5cSAndroid Build Coastguard Worker int width = min(tilew, w - col * tilew);
409*dfc6aa5cSAndroid Build Coastguard Worker int height = min(tileh, h - row * tileh);
410*dfc6aa5cSAndroid Build Coastguard Worker
411*dfc6aa5cSAndroid Build Coastguard Worker if (doYUV) {
412*dfc6aa5cSAndroid Build Coastguard Worker double startEncode = getTime();
413*dfc6aa5cSAndroid Build Coastguard Worker
414*dfc6aa5cSAndroid Build Coastguard Worker if (tjEncodeYUV3(handle, srcPtr2, width, pitch, height, pf, yuvBuf,
415*dfc6aa5cSAndroid Build Coastguard Worker yuvAlign, subsamp, flags) == -1)
416*dfc6aa5cSAndroid Build Coastguard Worker THROW_TJ("executing tjEncodeYUV3()");
417*dfc6aa5cSAndroid Build Coastguard Worker if (iter >= 0) elapsedEncode += getTime() - startEncode;
418*dfc6aa5cSAndroid Build Coastguard Worker if (tjCompressFromYUV(handle, yuvBuf, width, yuvAlign, height,
419*dfc6aa5cSAndroid Build Coastguard Worker subsamp, &jpegBuf[tile], &jpegSize[tile],
420*dfc6aa5cSAndroid Build Coastguard Worker jpegQual, flags) == -1)
421*dfc6aa5cSAndroid Build Coastguard Worker THROW_TJ("executing tjCompressFromYUV()");
422*dfc6aa5cSAndroid Build Coastguard Worker } else {
423*dfc6aa5cSAndroid Build Coastguard Worker if (tjCompress2(handle, srcPtr2, width, pitch, height, pf,
424*dfc6aa5cSAndroid Build Coastguard Worker &jpegBuf[tile], &jpegSize[tile], subsamp, jpegQual,
425*dfc6aa5cSAndroid Build Coastguard Worker flags) == -1)
426*dfc6aa5cSAndroid Build Coastguard Worker THROW_TJ("executing tjCompress2()");
427*dfc6aa5cSAndroid Build Coastguard Worker }
428*dfc6aa5cSAndroid Build Coastguard Worker totalJpegSize += jpegSize[tile];
429*dfc6aa5cSAndroid Build Coastguard Worker }
430*dfc6aa5cSAndroid Build Coastguard Worker }
431*dfc6aa5cSAndroid Build Coastguard Worker elapsed += getTime() - start;
432*dfc6aa5cSAndroid Build Coastguard Worker if (iter >= 0) {
433*dfc6aa5cSAndroid Build Coastguard Worker iter++;
434*dfc6aa5cSAndroid Build Coastguard Worker if (elapsed >= benchTime) break;
435*dfc6aa5cSAndroid Build Coastguard Worker } else if (elapsed >= warmup) {
436*dfc6aa5cSAndroid Build Coastguard Worker iter = 0;
437*dfc6aa5cSAndroid Build Coastguard Worker elapsed = elapsedEncode = 0.;
438*dfc6aa5cSAndroid Build Coastguard Worker }
439*dfc6aa5cSAndroid Build Coastguard Worker }
440*dfc6aa5cSAndroid Build Coastguard Worker if (doYUV) elapsed -= elapsedEncode;
441*dfc6aa5cSAndroid Build Coastguard Worker
442*dfc6aa5cSAndroid Build Coastguard Worker if (tjDestroy(handle) == -1) THROW_TJ("executing tjDestroy()");
443*dfc6aa5cSAndroid Build Coastguard Worker handle = NULL;
444*dfc6aa5cSAndroid Build Coastguard Worker
445*dfc6aa5cSAndroid Build Coastguard Worker if (quiet == 1) fprintf(stderr, "%-5d %-5d ", tilew, tileh);
446*dfc6aa5cSAndroid Build Coastguard Worker if (quiet) {
447*dfc6aa5cSAndroid Build Coastguard Worker if (doYUV)
448*dfc6aa5cSAndroid Build Coastguard Worker fprintf(stderr, "%-6s%s",
449*dfc6aa5cSAndroid Build Coastguard Worker sigfig((double)(w * h) / 1000000. *
450*dfc6aa5cSAndroid Build Coastguard Worker (double)iter / elapsedEncode, 4, tempStr, 1024),
451*dfc6aa5cSAndroid Build Coastguard Worker quiet == 2 ? "\n" : " ");
452*dfc6aa5cSAndroid Build Coastguard Worker fprintf(stderr, "%-6s%s",
453*dfc6aa5cSAndroid Build Coastguard Worker sigfig((double)(w * h) / 1000000. * (double)iter / elapsed, 4,
454*dfc6aa5cSAndroid Build Coastguard Worker tempStr, 1024),
455*dfc6aa5cSAndroid Build Coastguard Worker quiet == 2 ? "\n" : " ");
456*dfc6aa5cSAndroid Build Coastguard Worker fprintf(stderr, "%-6s%s",
457*dfc6aa5cSAndroid Build Coastguard Worker sigfig((double)(w * h * ps) / (double)totalJpegSize, 4, tempStr2,
458*dfc6aa5cSAndroid Build Coastguard Worker 80),
459*dfc6aa5cSAndroid Build Coastguard Worker quiet == 2 ? "\n" : " ");
460*dfc6aa5cSAndroid Build Coastguard Worker } else {
461*dfc6aa5cSAndroid Build Coastguard Worker fprintf(stderr, "\n%s size: %d x %d\n", doTile ? "Tile" : "Image", tilew,
462*dfc6aa5cSAndroid Build Coastguard Worker tileh);
463*dfc6aa5cSAndroid Build Coastguard Worker if (doYUV) {
464*dfc6aa5cSAndroid Build Coastguard Worker fprintf(stderr, "Encode YUV --> Frame rate: %f fps\n",
465*dfc6aa5cSAndroid Build Coastguard Worker (double)iter / elapsedEncode);
466*dfc6aa5cSAndroid Build Coastguard Worker fprintf(stderr, " Output image size: %lu bytes\n",
467*dfc6aa5cSAndroid Build Coastguard Worker yuvSize);
468*dfc6aa5cSAndroid Build Coastguard Worker fprintf(stderr, " Compression ratio: %f:1\n",
469*dfc6aa5cSAndroid Build Coastguard Worker (double)(w * h * ps) / (double)yuvSize);
470*dfc6aa5cSAndroid Build Coastguard Worker fprintf(stderr,
471*dfc6aa5cSAndroid Build Coastguard Worker " Throughput: %f Megapixels/sec\n",
472*dfc6aa5cSAndroid Build Coastguard Worker (double)(w * h) / 1000000. * (double)iter / elapsedEncode);
473*dfc6aa5cSAndroid Build Coastguard Worker fprintf(stderr,
474*dfc6aa5cSAndroid Build Coastguard Worker " Output bit stream: %f Megabits/sec\n",
475*dfc6aa5cSAndroid Build Coastguard Worker (double)yuvSize * 8. / 1000000. * (double)iter / elapsedEncode);
476*dfc6aa5cSAndroid Build Coastguard Worker }
477*dfc6aa5cSAndroid Build Coastguard Worker fprintf(stderr, "%s --> Frame rate: %f fps\n",
478*dfc6aa5cSAndroid Build Coastguard Worker doYUV ? "Comp from YUV" : "Compress ",
479*dfc6aa5cSAndroid Build Coastguard Worker (double)iter / elapsed);
480*dfc6aa5cSAndroid Build Coastguard Worker fprintf(stderr, " Output image size: %d bytes\n",
481*dfc6aa5cSAndroid Build Coastguard Worker totalJpegSize);
482*dfc6aa5cSAndroid Build Coastguard Worker fprintf(stderr, " Compression ratio: %f:1\n",
483*dfc6aa5cSAndroid Build Coastguard Worker (double)(w * h * ps) / (double)totalJpegSize);
484*dfc6aa5cSAndroid Build Coastguard Worker fprintf(stderr,
485*dfc6aa5cSAndroid Build Coastguard Worker " Throughput: %f Megapixels/sec\n",
486*dfc6aa5cSAndroid Build Coastguard Worker (double)(w * h) / 1000000. * (double)iter / elapsed);
487*dfc6aa5cSAndroid Build Coastguard Worker fprintf(stderr,
488*dfc6aa5cSAndroid Build Coastguard Worker " Output bit stream: %f Megabits/sec\n",
489*dfc6aa5cSAndroid Build Coastguard Worker (double)totalJpegSize * 8. / 1000000. * (double)iter / elapsed);
490*dfc6aa5cSAndroid Build Coastguard Worker }
491*dfc6aa5cSAndroid Build Coastguard Worker if (tilew == w && tileh == h && doWrite) {
492*dfc6aa5cSAndroid Build Coastguard Worker SNPRINTF(tempStr, 1024, "%s_%s_Q%d.jpg", fileName, subName[subsamp],
493*dfc6aa5cSAndroid Build Coastguard Worker jpegQual);
494*dfc6aa5cSAndroid Build Coastguard Worker if ((file = fopen(tempStr, "wb")) == NULL)
495*dfc6aa5cSAndroid Build Coastguard Worker THROW_UNIX("opening reference image");
496*dfc6aa5cSAndroid Build Coastguard Worker if (fwrite(jpegBuf[0], jpegSize[0], 1, file) != 1)
497*dfc6aa5cSAndroid Build Coastguard Worker THROW_UNIX("writing reference image");
498*dfc6aa5cSAndroid Build Coastguard Worker fclose(file); file = NULL;
499*dfc6aa5cSAndroid Build Coastguard Worker if (!quiet) fprintf(stderr, "Reference image written to %s\n", tempStr);
500*dfc6aa5cSAndroid Build Coastguard Worker }
501*dfc6aa5cSAndroid Build Coastguard Worker
502*dfc6aa5cSAndroid Build Coastguard Worker /* Decompression test */
503*dfc6aa5cSAndroid Build Coastguard Worker if (!compOnly) {
504*dfc6aa5cSAndroid Build Coastguard Worker if (decomp(srcBuf, jpegBuf, jpegSize, tmpBuf, w, h, subsamp, jpegQual,
505*dfc6aa5cSAndroid Build Coastguard Worker fileName, tilew, tileh) == -1)
506*dfc6aa5cSAndroid Build Coastguard Worker goto bailout;
507*dfc6aa5cSAndroid Build Coastguard Worker } else if (quiet == 1) fprintf(stderr, "N/A\n");
508*dfc6aa5cSAndroid Build Coastguard Worker
509*dfc6aa5cSAndroid Build Coastguard Worker for (i = 0; i < ntilesw * ntilesh; i++) {
510*dfc6aa5cSAndroid Build Coastguard Worker tjFree(jpegBuf[i]);
511*dfc6aa5cSAndroid Build Coastguard Worker jpegBuf[i] = NULL;
512*dfc6aa5cSAndroid Build Coastguard Worker }
513*dfc6aa5cSAndroid Build Coastguard Worker free(jpegBuf); jpegBuf = NULL;
514*dfc6aa5cSAndroid Build Coastguard Worker free(jpegSize); jpegSize = NULL;
515*dfc6aa5cSAndroid Build Coastguard Worker if (doYUV) {
516*dfc6aa5cSAndroid Build Coastguard Worker free(yuvBuf); yuvBuf = NULL;
517*dfc6aa5cSAndroid Build Coastguard Worker }
518*dfc6aa5cSAndroid Build Coastguard Worker
519*dfc6aa5cSAndroid Build Coastguard Worker if (tilew == w && tileh == h) break;
520*dfc6aa5cSAndroid Build Coastguard Worker }
521*dfc6aa5cSAndroid Build Coastguard Worker
522*dfc6aa5cSAndroid Build Coastguard Worker bailout:
523*dfc6aa5cSAndroid Build Coastguard Worker if (file) fclose(file);
524*dfc6aa5cSAndroid Build Coastguard Worker if (jpegBuf) {
525*dfc6aa5cSAndroid Build Coastguard Worker for (i = 0; i < ntilesw * ntilesh; i++)
526*dfc6aa5cSAndroid Build Coastguard Worker tjFree(jpegBuf[i]);
527*dfc6aa5cSAndroid Build Coastguard Worker }
528*dfc6aa5cSAndroid Build Coastguard Worker free(jpegBuf);
529*dfc6aa5cSAndroid Build Coastguard Worker free(yuvBuf);
530*dfc6aa5cSAndroid Build Coastguard Worker free(jpegSize);
531*dfc6aa5cSAndroid Build Coastguard Worker free(tmpBuf);
532*dfc6aa5cSAndroid Build Coastguard Worker if (handle) tjDestroy(handle);
533*dfc6aa5cSAndroid Build Coastguard Worker return retval;
534*dfc6aa5cSAndroid Build Coastguard Worker }
535*dfc6aa5cSAndroid Build Coastguard Worker
536*dfc6aa5cSAndroid Build Coastguard Worker
decompTest(char * fileName)537*dfc6aa5cSAndroid Build Coastguard Worker static int decompTest(char *fileName)
538*dfc6aa5cSAndroid Build Coastguard Worker {
539*dfc6aa5cSAndroid Build Coastguard Worker FILE *file = NULL;
540*dfc6aa5cSAndroid Build Coastguard Worker tjhandle handle = NULL;
541*dfc6aa5cSAndroid Build Coastguard Worker unsigned char **jpegBuf = NULL, *srcBuf = NULL;
542*dfc6aa5cSAndroid Build Coastguard Worker unsigned long *jpegSize = NULL, srcSize, totalJpegSize;
543*dfc6aa5cSAndroid Build Coastguard Worker tjtransform *t = NULL;
544*dfc6aa5cSAndroid Build Coastguard Worker double start, elapsed;
545*dfc6aa5cSAndroid Build Coastguard Worker int ps = tjPixelSize[pf], tile, row, col, i, iter, retval = 0, decompsrc = 0;
546*dfc6aa5cSAndroid Build Coastguard Worker char *temp = NULL, tempStr[80], tempStr2[80];
547*dfc6aa5cSAndroid Build Coastguard Worker /* Original image */
548*dfc6aa5cSAndroid Build Coastguard Worker int w = 0, h = 0, tilew, tileh, ntilesw = 1, ntilesh = 1, subsamp = -1,
549*dfc6aa5cSAndroid Build Coastguard Worker cs = -1;
550*dfc6aa5cSAndroid Build Coastguard Worker /* Transformed image */
551*dfc6aa5cSAndroid Build Coastguard Worker int tw, th, ttilew, ttileh, tntilesw, tntilesh, tsubsamp;
552*dfc6aa5cSAndroid Build Coastguard Worker
553*dfc6aa5cSAndroid Build Coastguard Worker if ((file = fopen(fileName, "rb")) == NULL)
554*dfc6aa5cSAndroid Build Coastguard Worker THROW_UNIX("opening file");
555*dfc6aa5cSAndroid Build Coastguard Worker if (fseek(file, 0, SEEK_END) < 0 ||
556*dfc6aa5cSAndroid Build Coastguard Worker (srcSize = ftell(file)) == (unsigned long)-1)
557*dfc6aa5cSAndroid Build Coastguard Worker THROW_UNIX("determining file size");
558*dfc6aa5cSAndroid Build Coastguard Worker if ((srcBuf = (unsigned char *)malloc(srcSize)) == NULL)
559*dfc6aa5cSAndroid Build Coastguard Worker THROW_UNIX("allocating memory");
560*dfc6aa5cSAndroid Build Coastguard Worker if (fseek(file, 0, SEEK_SET) < 0)
561*dfc6aa5cSAndroid Build Coastguard Worker THROW_UNIX("setting file position");
562*dfc6aa5cSAndroid Build Coastguard Worker if (fread(srcBuf, srcSize, 1, file) < 1)
563*dfc6aa5cSAndroid Build Coastguard Worker THROW_UNIX("reading JPEG data");
564*dfc6aa5cSAndroid Build Coastguard Worker fclose(file); file = NULL;
565*dfc6aa5cSAndroid Build Coastguard Worker
566*dfc6aa5cSAndroid Build Coastguard Worker temp = strrchr(fileName, '.');
567*dfc6aa5cSAndroid Build Coastguard Worker if (temp != NULL) *temp = '\0';
568*dfc6aa5cSAndroid Build Coastguard Worker
569*dfc6aa5cSAndroid Build Coastguard Worker if ((handle = tjInitTransform()) == NULL)
570*dfc6aa5cSAndroid Build Coastguard Worker THROW_TJ("executing tjInitTransform()");
571*dfc6aa5cSAndroid Build Coastguard Worker if (tjDecompressHeader3(handle, srcBuf, srcSize, &w, &h, &subsamp,
572*dfc6aa5cSAndroid Build Coastguard Worker &cs) == -1)
573*dfc6aa5cSAndroid Build Coastguard Worker THROW_TJ("executing tjDecompressHeader3()");
574*dfc6aa5cSAndroid Build Coastguard Worker if (w < 1 || h < 1)
575*dfc6aa5cSAndroid Build Coastguard Worker THROW("reading JPEG header", "Invalid image dimensions");
576*dfc6aa5cSAndroid Build Coastguard Worker if (cs == TJCS_YCCK || cs == TJCS_CMYK) {
577*dfc6aa5cSAndroid Build Coastguard Worker pf = TJPF_CMYK; ps = tjPixelSize[pf];
578*dfc6aa5cSAndroid Build Coastguard Worker }
579*dfc6aa5cSAndroid Build Coastguard Worker
580*dfc6aa5cSAndroid Build Coastguard Worker if (quiet == 1) {
581*dfc6aa5cSAndroid Build Coastguard Worker fprintf(stderr, "All performance values in Mpixels/sec\n\n");
582*dfc6aa5cSAndroid Build Coastguard Worker fprintf(stderr,
583*dfc6aa5cSAndroid Build Coastguard Worker "Pixel JPEG JPEG %s %s Xform Comp Decomp ",
584*dfc6aa5cSAndroid Build Coastguard Worker doTile ? "Tile " : "Image", doTile ? "Tile " : "Image");
585*dfc6aa5cSAndroid Build Coastguard Worker if (doYUV) fprintf(stderr, "Decode");
586*dfc6aa5cSAndroid Build Coastguard Worker fprintf(stderr, "\n");
587*dfc6aa5cSAndroid Build Coastguard Worker fprintf(stderr,
588*dfc6aa5cSAndroid Build Coastguard Worker "Format CS Subsamp Width Height Perf Ratio Perf ");
589*dfc6aa5cSAndroid Build Coastguard Worker if (doYUV) fprintf(stderr, "Perf");
590*dfc6aa5cSAndroid Build Coastguard Worker fprintf(stderr, "\n\n");
591*dfc6aa5cSAndroid Build Coastguard Worker } else if (!quiet)
592*dfc6aa5cSAndroid Build Coastguard Worker fprintf(stderr, ">>>>> JPEG %s --> %s (%s) <<<<<\n",
593*dfc6aa5cSAndroid Build Coastguard Worker formatName(subsamp, cs, tempStr), pixFormatStr[pf],
594*dfc6aa5cSAndroid Build Coastguard Worker (flags & TJFLAG_BOTTOMUP) ? "Bottom-up" : "Top-down");
595*dfc6aa5cSAndroid Build Coastguard Worker
596*dfc6aa5cSAndroid Build Coastguard Worker for (tilew = doTile ? 16 : w, tileh = doTile ? 16 : h; ;
597*dfc6aa5cSAndroid Build Coastguard Worker tilew *= 2, tileh *= 2) {
598*dfc6aa5cSAndroid Build Coastguard Worker if (tilew > w) tilew = w;
599*dfc6aa5cSAndroid Build Coastguard Worker if (tileh > h) tileh = h;
600*dfc6aa5cSAndroid Build Coastguard Worker ntilesw = (w + tilew - 1) / tilew;
601*dfc6aa5cSAndroid Build Coastguard Worker ntilesh = (h + tileh - 1) / tileh;
602*dfc6aa5cSAndroid Build Coastguard Worker
603*dfc6aa5cSAndroid Build Coastguard Worker if ((jpegBuf = (unsigned char **)malloc(sizeof(unsigned char *) *
604*dfc6aa5cSAndroid Build Coastguard Worker ntilesw * ntilesh)) == NULL)
605*dfc6aa5cSAndroid Build Coastguard Worker THROW_UNIX("allocating JPEG tile array");
606*dfc6aa5cSAndroid Build Coastguard Worker memset(jpegBuf, 0, sizeof(unsigned char *) * ntilesw * ntilesh);
607*dfc6aa5cSAndroid Build Coastguard Worker if ((jpegSize = (unsigned long *)malloc(sizeof(unsigned long) *
608*dfc6aa5cSAndroid Build Coastguard Worker ntilesw * ntilesh)) == NULL)
609*dfc6aa5cSAndroid Build Coastguard Worker THROW_UNIX("allocating JPEG size array");
610*dfc6aa5cSAndroid Build Coastguard Worker memset(jpegSize, 0, sizeof(unsigned long) * ntilesw * ntilesh);
611*dfc6aa5cSAndroid Build Coastguard Worker
612*dfc6aa5cSAndroid Build Coastguard Worker if ((flags & TJFLAG_NOREALLOC) != 0 &&
613*dfc6aa5cSAndroid Build Coastguard Worker (doTile || xformOp != TJXOP_NONE || xformOpt != 0 || customFilter))
614*dfc6aa5cSAndroid Build Coastguard Worker for (i = 0; i < ntilesw * ntilesh; i++) {
615*dfc6aa5cSAndroid Build Coastguard Worker if (tjBufSize(tilew, tileh, subsamp) > (unsigned long)INT_MAX)
616*dfc6aa5cSAndroid Build Coastguard Worker THROW("getting buffer size", "Image is too large");
617*dfc6aa5cSAndroid Build Coastguard Worker if ((jpegBuf[i] = (unsigned char *)
618*dfc6aa5cSAndroid Build Coastguard Worker tjAlloc(tjBufSize(tilew, tileh, subsamp))) == NULL)
619*dfc6aa5cSAndroid Build Coastguard Worker THROW_UNIX("allocating JPEG tiles");
620*dfc6aa5cSAndroid Build Coastguard Worker }
621*dfc6aa5cSAndroid Build Coastguard Worker
622*dfc6aa5cSAndroid Build Coastguard Worker tw = w; th = h; ttilew = tilew; ttileh = tileh;
623*dfc6aa5cSAndroid Build Coastguard Worker if (!quiet) {
624*dfc6aa5cSAndroid Build Coastguard Worker fprintf(stderr, "\n%s size: %d x %d", doTile ? "Tile" : "Image", ttilew,
625*dfc6aa5cSAndroid Build Coastguard Worker ttileh);
626*dfc6aa5cSAndroid Build Coastguard Worker if (sf.num != 1 || sf.denom != 1)
627*dfc6aa5cSAndroid Build Coastguard Worker fprintf(stderr, " --> %d x %d", TJSCALED(tw, sf), TJSCALED(th, sf));
628*dfc6aa5cSAndroid Build Coastguard Worker fprintf(stderr, "\n");
629*dfc6aa5cSAndroid Build Coastguard Worker } else if (quiet == 1) {
630*dfc6aa5cSAndroid Build Coastguard Worker fprintf(stderr, "%-4s (%s) %-5s %-5s ", pixFormatStr[pf],
631*dfc6aa5cSAndroid Build Coastguard Worker (flags & TJFLAG_BOTTOMUP) ? "BU" : "TD", csName[cs],
632*dfc6aa5cSAndroid Build Coastguard Worker subNameLong[subsamp]);
633*dfc6aa5cSAndroid Build Coastguard Worker fprintf(stderr, "%-5d %-5d ", tilew, tileh);
634*dfc6aa5cSAndroid Build Coastguard Worker }
635*dfc6aa5cSAndroid Build Coastguard Worker
636*dfc6aa5cSAndroid Build Coastguard Worker tsubsamp = subsamp;
637*dfc6aa5cSAndroid Build Coastguard Worker if (doTile || xformOp != TJXOP_NONE || xformOpt != 0 || customFilter) {
638*dfc6aa5cSAndroid Build Coastguard Worker if ((t = (tjtransform *)malloc(sizeof(tjtransform) * ntilesw *
639*dfc6aa5cSAndroid Build Coastguard Worker ntilesh)) == NULL)
640*dfc6aa5cSAndroid Build Coastguard Worker THROW_UNIX("allocating image transform array");
641*dfc6aa5cSAndroid Build Coastguard Worker
642*dfc6aa5cSAndroid Build Coastguard Worker if (xformOp == TJXOP_TRANSPOSE || xformOp == TJXOP_TRANSVERSE ||
643*dfc6aa5cSAndroid Build Coastguard Worker xformOp == TJXOP_ROT90 || xformOp == TJXOP_ROT270) {
644*dfc6aa5cSAndroid Build Coastguard Worker tw = h; th = w; ttilew = tileh; ttileh = tilew;
645*dfc6aa5cSAndroid Build Coastguard Worker }
646*dfc6aa5cSAndroid Build Coastguard Worker
647*dfc6aa5cSAndroid Build Coastguard Worker if (xformOpt & TJXOPT_GRAY) tsubsamp = TJSAMP_GRAY;
648*dfc6aa5cSAndroid Build Coastguard Worker if (xformOp == TJXOP_HFLIP || xformOp == TJXOP_ROT180)
649*dfc6aa5cSAndroid Build Coastguard Worker tw = tw - (tw % tjMCUWidth[tsubsamp]);
650*dfc6aa5cSAndroid Build Coastguard Worker if (xformOp == TJXOP_VFLIP || xformOp == TJXOP_ROT180)
651*dfc6aa5cSAndroid Build Coastguard Worker th = th - (th % tjMCUHeight[tsubsamp]);
652*dfc6aa5cSAndroid Build Coastguard Worker if (xformOp == TJXOP_TRANSVERSE || xformOp == TJXOP_ROT90)
653*dfc6aa5cSAndroid Build Coastguard Worker tw = tw - (tw % tjMCUHeight[tsubsamp]);
654*dfc6aa5cSAndroid Build Coastguard Worker if (xformOp == TJXOP_TRANSVERSE || xformOp == TJXOP_ROT270)
655*dfc6aa5cSAndroid Build Coastguard Worker th = th - (th % tjMCUWidth[tsubsamp]);
656*dfc6aa5cSAndroid Build Coastguard Worker tntilesw = (tw + ttilew - 1) / ttilew;
657*dfc6aa5cSAndroid Build Coastguard Worker tntilesh = (th + ttileh - 1) / ttileh;
658*dfc6aa5cSAndroid Build Coastguard Worker
659*dfc6aa5cSAndroid Build Coastguard Worker if (xformOp == TJXOP_TRANSPOSE || xformOp == TJXOP_TRANSVERSE ||
660*dfc6aa5cSAndroid Build Coastguard Worker xformOp == TJXOP_ROT90 || xformOp == TJXOP_ROT270) {
661*dfc6aa5cSAndroid Build Coastguard Worker if (tsubsamp == TJSAMP_422) tsubsamp = TJSAMP_440;
662*dfc6aa5cSAndroid Build Coastguard Worker else if (tsubsamp == TJSAMP_440) tsubsamp = TJSAMP_422;
663*dfc6aa5cSAndroid Build Coastguard Worker }
664*dfc6aa5cSAndroid Build Coastguard Worker
665*dfc6aa5cSAndroid Build Coastguard Worker for (row = 0, tile = 0; row < tntilesh; row++) {
666*dfc6aa5cSAndroid Build Coastguard Worker for (col = 0; col < tntilesw; col++, tile++) {
667*dfc6aa5cSAndroid Build Coastguard Worker t[tile].r.w = min(ttilew, tw - col * ttilew);
668*dfc6aa5cSAndroid Build Coastguard Worker t[tile].r.h = min(ttileh, th - row * ttileh);
669*dfc6aa5cSAndroid Build Coastguard Worker t[tile].r.x = col * ttilew;
670*dfc6aa5cSAndroid Build Coastguard Worker t[tile].r.y = row * ttileh;
671*dfc6aa5cSAndroid Build Coastguard Worker t[tile].op = xformOp;
672*dfc6aa5cSAndroid Build Coastguard Worker t[tile].options = xformOpt | TJXOPT_TRIM;
673*dfc6aa5cSAndroid Build Coastguard Worker t[tile].customFilter = customFilter;
674*dfc6aa5cSAndroid Build Coastguard Worker if (t[tile].options & TJXOPT_NOOUTPUT && jpegBuf[tile]) {
675*dfc6aa5cSAndroid Build Coastguard Worker tjFree(jpegBuf[tile]); jpegBuf[tile] = NULL;
676*dfc6aa5cSAndroid Build Coastguard Worker }
677*dfc6aa5cSAndroid Build Coastguard Worker }
678*dfc6aa5cSAndroid Build Coastguard Worker }
679*dfc6aa5cSAndroid Build Coastguard Worker
680*dfc6aa5cSAndroid Build Coastguard Worker iter = -1;
681*dfc6aa5cSAndroid Build Coastguard Worker elapsed = 0.;
682*dfc6aa5cSAndroid Build Coastguard Worker while (1) {
683*dfc6aa5cSAndroid Build Coastguard Worker start = getTime();
684*dfc6aa5cSAndroid Build Coastguard Worker if (tjTransform(handle, srcBuf, srcSize, tntilesw * tntilesh, jpegBuf,
685*dfc6aa5cSAndroid Build Coastguard Worker jpegSize, t, flags) == -1)
686*dfc6aa5cSAndroid Build Coastguard Worker THROW_TJ("executing tjTransform()");
687*dfc6aa5cSAndroid Build Coastguard Worker elapsed += getTime() - start;
688*dfc6aa5cSAndroid Build Coastguard Worker if (iter >= 0) {
689*dfc6aa5cSAndroid Build Coastguard Worker iter++;
690*dfc6aa5cSAndroid Build Coastguard Worker if (elapsed >= benchTime) break;
691*dfc6aa5cSAndroid Build Coastguard Worker } else if (elapsed >= warmup) {
692*dfc6aa5cSAndroid Build Coastguard Worker iter = 0;
693*dfc6aa5cSAndroid Build Coastguard Worker elapsed = 0.;
694*dfc6aa5cSAndroid Build Coastguard Worker }
695*dfc6aa5cSAndroid Build Coastguard Worker }
696*dfc6aa5cSAndroid Build Coastguard Worker
697*dfc6aa5cSAndroid Build Coastguard Worker free(t); t = NULL;
698*dfc6aa5cSAndroid Build Coastguard Worker
699*dfc6aa5cSAndroid Build Coastguard Worker for (tile = 0, totalJpegSize = 0; tile < tntilesw * tntilesh; tile++)
700*dfc6aa5cSAndroid Build Coastguard Worker totalJpegSize += jpegSize[tile];
701*dfc6aa5cSAndroid Build Coastguard Worker
702*dfc6aa5cSAndroid Build Coastguard Worker if (quiet) {
703*dfc6aa5cSAndroid Build Coastguard Worker fprintf(stderr, "%-6s%s%-6s%s",
704*dfc6aa5cSAndroid Build Coastguard Worker sigfig((double)(w * h) / 1000000. / elapsed, 4, tempStr, 80),
705*dfc6aa5cSAndroid Build Coastguard Worker quiet == 2 ? "\n" : " ",
706*dfc6aa5cSAndroid Build Coastguard Worker sigfig((double)(w * h * ps) / (double)totalJpegSize, 4,
707*dfc6aa5cSAndroid Build Coastguard Worker tempStr2, 80),
708*dfc6aa5cSAndroid Build Coastguard Worker quiet == 2 ? "\n" : " ");
709*dfc6aa5cSAndroid Build Coastguard Worker } else {
710*dfc6aa5cSAndroid Build Coastguard Worker fprintf(stderr, "Transform --> Frame rate: %f fps\n",
711*dfc6aa5cSAndroid Build Coastguard Worker 1.0 / elapsed);
712*dfc6aa5cSAndroid Build Coastguard Worker fprintf(stderr, " Output image size: %lu bytes\n",
713*dfc6aa5cSAndroid Build Coastguard Worker totalJpegSize);
714*dfc6aa5cSAndroid Build Coastguard Worker fprintf(stderr, " Compression ratio: %f:1\n",
715*dfc6aa5cSAndroid Build Coastguard Worker (double)(w * h * ps) / (double)totalJpegSize);
716*dfc6aa5cSAndroid Build Coastguard Worker fprintf(stderr,
717*dfc6aa5cSAndroid Build Coastguard Worker " Throughput: %f Megapixels/sec\n",
718*dfc6aa5cSAndroid Build Coastguard Worker (double)(w * h) / 1000000. / elapsed);
719*dfc6aa5cSAndroid Build Coastguard Worker fprintf(stderr,
720*dfc6aa5cSAndroid Build Coastguard Worker " Output bit stream: %f Megabits/sec\n",
721*dfc6aa5cSAndroid Build Coastguard Worker (double)totalJpegSize * 8. / 1000000. / elapsed);
722*dfc6aa5cSAndroid Build Coastguard Worker }
723*dfc6aa5cSAndroid Build Coastguard Worker } else {
724*dfc6aa5cSAndroid Build Coastguard Worker if (quiet == 1) fprintf(stderr, "N/A N/A ");
725*dfc6aa5cSAndroid Build Coastguard Worker tjFree(jpegBuf[0]);
726*dfc6aa5cSAndroid Build Coastguard Worker jpegBuf[0] = NULL;
727*dfc6aa5cSAndroid Build Coastguard Worker decompsrc = 1;
728*dfc6aa5cSAndroid Build Coastguard Worker }
729*dfc6aa5cSAndroid Build Coastguard Worker
730*dfc6aa5cSAndroid Build Coastguard Worker if (w == tilew) ttilew = tw;
731*dfc6aa5cSAndroid Build Coastguard Worker if (h == tileh) ttileh = th;
732*dfc6aa5cSAndroid Build Coastguard Worker if (!(xformOpt & TJXOPT_NOOUTPUT)) {
733*dfc6aa5cSAndroid Build Coastguard Worker if (decomp(NULL, decompsrc ? &srcBuf : jpegBuf,
734*dfc6aa5cSAndroid Build Coastguard Worker decompsrc ? &srcSize : jpegSize, NULL, tw, th, tsubsamp, 0,
735*dfc6aa5cSAndroid Build Coastguard Worker fileName, ttilew, ttileh) == -1)
736*dfc6aa5cSAndroid Build Coastguard Worker goto bailout;
737*dfc6aa5cSAndroid Build Coastguard Worker } else if (quiet == 1) fprintf(stderr, "N/A\n");
738*dfc6aa5cSAndroid Build Coastguard Worker
739*dfc6aa5cSAndroid Build Coastguard Worker for (i = 0; i < ntilesw * ntilesh; i++) {
740*dfc6aa5cSAndroid Build Coastguard Worker tjFree(jpegBuf[i]);
741*dfc6aa5cSAndroid Build Coastguard Worker jpegBuf[i] = NULL;
742*dfc6aa5cSAndroid Build Coastguard Worker }
743*dfc6aa5cSAndroid Build Coastguard Worker free(jpegBuf); jpegBuf = NULL;
744*dfc6aa5cSAndroid Build Coastguard Worker free(jpegSize); jpegSize = NULL;
745*dfc6aa5cSAndroid Build Coastguard Worker
746*dfc6aa5cSAndroid Build Coastguard Worker if (tilew == w && tileh == h) break;
747*dfc6aa5cSAndroid Build Coastguard Worker }
748*dfc6aa5cSAndroid Build Coastguard Worker
749*dfc6aa5cSAndroid Build Coastguard Worker bailout:
750*dfc6aa5cSAndroid Build Coastguard Worker if (file) fclose(file);
751*dfc6aa5cSAndroid Build Coastguard Worker if (jpegBuf) {
752*dfc6aa5cSAndroid Build Coastguard Worker for (i = 0; i < ntilesw * ntilesh; i++)
753*dfc6aa5cSAndroid Build Coastguard Worker tjFree(jpegBuf[i]);
754*dfc6aa5cSAndroid Build Coastguard Worker }
755*dfc6aa5cSAndroid Build Coastguard Worker free(jpegBuf);
756*dfc6aa5cSAndroid Build Coastguard Worker free(jpegSize);
757*dfc6aa5cSAndroid Build Coastguard Worker free(srcBuf);
758*dfc6aa5cSAndroid Build Coastguard Worker free(t);
759*dfc6aa5cSAndroid Build Coastguard Worker if (handle) { tjDestroy(handle); handle = NULL; }
760*dfc6aa5cSAndroid Build Coastguard Worker return retval;
761*dfc6aa5cSAndroid Build Coastguard Worker }
762*dfc6aa5cSAndroid Build Coastguard Worker
763*dfc6aa5cSAndroid Build Coastguard Worker
usage(char * progName)764*dfc6aa5cSAndroid Build Coastguard Worker static void usage(char *progName)
765*dfc6aa5cSAndroid Build Coastguard Worker {
766*dfc6aa5cSAndroid Build Coastguard Worker int i;
767*dfc6aa5cSAndroid Build Coastguard Worker
768*dfc6aa5cSAndroid Build Coastguard Worker printf("USAGE: %s\n", progName);
769*dfc6aa5cSAndroid Build Coastguard Worker printf(" <Inputimage (BMP|PPM)> <Quality> [options]\n\n");
770*dfc6aa5cSAndroid Build Coastguard Worker printf(" %s\n", progName);
771*dfc6aa5cSAndroid Build Coastguard Worker printf(" <Inputimage (JPG)> [options]\n\n");
772*dfc6aa5cSAndroid Build Coastguard Worker printf("Options:\n\n");
773*dfc6aa5cSAndroid Build Coastguard Worker printf("-alloc = Dynamically allocate JPEG buffers\n");
774*dfc6aa5cSAndroid Build Coastguard Worker printf("-bmp = Use Windows Bitmap format for output images [default = PPM]\n");
775*dfc6aa5cSAndroid Build Coastguard Worker printf("-bottomup = Use bottom-up row order for packed-pixel source/destination buffers\n");
776*dfc6aa5cSAndroid Build Coastguard Worker printf("-tile = Compress/transform the input image into separate JPEG tiles of varying\n");
777*dfc6aa5cSAndroid Build Coastguard Worker printf(" sizes (useful for measuring JPEG overhead)\n");
778*dfc6aa5cSAndroid Build Coastguard Worker printf("-rgb, -bgr, -rgbx, -bgrx, -xbgr, -xrgb =\n");
779*dfc6aa5cSAndroid Build Coastguard Worker printf(" Use the specified pixel format for packed-pixel source/destination buffers\n");
780*dfc6aa5cSAndroid Build Coastguard Worker printf(" [default = BGR]\n");
781*dfc6aa5cSAndroid Build Coastguard Worker printf("-cmyk = Indirectly test YCCK JPEG compression/decompression\n");
782*dfc6aa5cSAndroid Build Coastguard Worker printf(" (use the CMYK pixel format for packed-pixel source/destination buffers)\n");
783*dfc6aa5cSAndroid Build Coastguard Worker printf("-fastupsample = Use the fastest chrominance upsampling algorithm available\n");
784*dfc6aa5cSAndroid Build Coastguard Worker printf("-fastdct = Use the fastest DCT/IDCT algorithm available\n");
785*dfc6aa5cSAndroid Build Coastguard Worker printf("-accuratedct = Use the most accurate DCT/IDCT algorithm available\n");
786*dfc6aa5cSAndroid Build Coastguard Worker printf("-progressive = Use progressive entropy coding in JPEG images generated by\n");
787*dfc6aa5cSAndroid Build Coastguard Worker printf(" compression and transform operations\n");
788*dfc6aa5cSAndroid Build Coastguard Worker printf("-subsamp <s> = When compressing, use the specified level of chrominance\n");
789*dfc6aa5cSAndroid Build Coastguard Worker printf(" subsampling (<s> = 444, 422, 440, 420, 411, or GRAY) [default = test\n");
790*dfc6aa5cSAndroid Build Coastguard Worker printf(" Grayscale, 4:2:0, 4:2:2, and 4:4:4 in sequence]\n");
791*dfc6aa5cSAndroid Build Coastguard Worker printf("-quiet = Output results in tabular rather than verbose format\n");
792*dfc6aa5cSAndroid Build Coastguard Worker printf("-yuv = Compress from/decompress to intermediate planar YUV images\n");
793*dfc6aa5cSAndroid Build Coastguard Worker printf("-yuvpad <p> = The number of bytes by which each row in each plane of an\n");
794*dfc6aa5cSAndroid Build Coastguard Worker printf(" intermediate YUV image is evenly divisible (must be a power of 2)\n");
795*dfc6aa5cSAndroid Build Coastguard Worker printf(" [default = 1]\n");
796*dfc6aa5cSAndroid Build Coastguard Worker printf("-scale M/N = When decompressing, scale the width/height of the JPEG image by a\n");
797*dfc6aa5cSAndroid Build Coastguard Worker printf(" factor of M/N (M/N = ");
798*dfc6aa5cSAndroid Build Coastguard Worker for (i = 0; i < nsf; i++) {
799*dfc6aa5cSAndroid Build Coastguard Worker printf("%d/%d", scalingFactors[i].num, scalingFactors[i].denom);
800*dfc6aa5cSAndroid Build Coastguard Worker if (nsf == 2 && i != nsf - 1) printf(" or ");
801*dfc6aa5cSAndroid Build Coastguard Worker else if (nsf > 2) {
802*dfc6aa5cSAndroid Build Coastguard Worker if (i != nsf - 1) printf(", ");
803*dfc6aa5cSAndroid Build Coastguard Worker if (i == nsf - 2) printf("or ");
804*dfc6aa5cSAndroid Build Coastguard Worker }
805*dfc6aa5cSAndroid Build Coastguard Worker if (i % 8 == 0 && i != 0) printf("\n ");
806*dfc6aa5cSAndroid Build Coastguard Worker }
807*dfc6aa5cSAndroid Build Coastguard Worker printf(")\n");
808*dfc6aa5cSAndroid Build Coastguard Worker printf("-hflip, -vflip, -transpose, -transverse, -rot90, -rot180, -rot270 =\n");
809*dfc6aa5cSAndroid Build Coastguard Worker printf(" Perform the specified lossless transform operation on the input image\n");
810*dfc6aa5cSAndroid Build Coastguard Worker printf(" prior to decompression (these operations are mutually exclusive)\n");
811*dfc6aa5cSAndroid Build Coastguard Worker printf("-grayscale = Transform the input image into a grayscale JPEG image prior to\n");
812*dfc6aa5cSAndroid Build Coastguard Worker printf(" decompression (can be combined with the other transform operations above)\n");
813*dfc6aa5cSAndroid Build Coastguard Worker printf("-copynone = Do not copy any extra markers (including EXIF and ICC profile data)\n");
814*dfc6aa5cSAndroid Build Coastguard Worker printf(" when transforming the input image\n");
815*dfc6aa5cSAndroid Build Coastguard Worker printf("-benchtime <t> = Run each benchmark for at least <t> seconds [default = 5.0]\n");
816*dfc6aa5cSAndroid Build Coastguard Worker printf("-warmup <t> = Run each benchmark for <t> seconds [default = 1.0] prior to\n");
817*dfc6aa5cSAndroid Build Coastguard Worker printf(" starting the timer, in order to prime the caches and thus improve the\n");
818*dfc6aa5cSAndroid Build Coastguard Worker printf(" consistency of the benchmark results\n");
819*dfc6aa5cSAndroid Build Coastguard Worker printf("-componly = Stop after running compression tests. Do not test decompression.\n");
820*dfc6aa5cSAndroid Build Coastguard Worker printf("-nowrite = Do not write reference or output images (improves consistency of\n");
821*dfc6aa5cSAndroid Build Coastguard Worker printf(" benchmark results)\n");
822*dfc6aa5cSAndroid Build Coastguard Worker printf("-limitscans = Refuse to decompress or transform progressive JPEG images that\n");
823*dfc6aa5cSAndroid Build Coastguard Worker printf(" have an unreasonably large number of scans\n");
824*dfc6aa5cSAndroid Build Coastguard Worker printf("-stoponwarning = Immediately discontinue the current\n");
825*dfc6aa5cSAndroid Build Coastguard Worker printf(" compression/decompression/transform operation if a warning (non-fatal\n");
826*dfc6aa5cSAndroid Build Coastguard Worker printf(" error) occurs\n\n");
827*dfc6aa5cSAndroid Build Coastguard Worker printf("NOTE: If the quality is specified as a range (e.g. 90-100), a separate\n");
828*dfc6aa5cSAndroid Build Coastguard Worker printf("test will be performed for all quality values in the range.\n\n");
829*dfc6aa5cSAndroid Build Coastguard Worker exit(1);
830*dfc6aa5cSAndroid Build Coastguard Worker }
831*dfc6aa5cSAndroid Build Coastguard Worker
832*dfc6aa5cSAndroid Build Coastguard Worker #ifndef GTEST
main(int argc,char * argv[])833*dfc6aa5cSAndroid Build Coastguard Worker int main(int argc, char *argv[])
834*dfc6aa5cSAndroid Build Coastguard Worker #else
835*dfc6aa5cSAndroid Build Coastguard Worker int tjbench(int argc, char *argv[])
836*dfc6aa5cSAndroid Build Coastguard Worker #endif
837*dfc6aa5cSAndroid Build Coastguard Worker {
838*dfc6aa5cSAndroid Build Coastguard Worker unsigned char *srcBuf = NULL;
839*dfc6aa5cSAndroid Build Coastguard Worker int w = 0, h = 0, i, j, minQual = -1, maxQual = -1;
840*dfc6aa5cSAndroid Build Coastguard Worker char *temp;
841*dfc6aa5cSAndroid Build Coastguard Worker int minArg = 2, retval = 0, subsamp = -1;
842*dfc6aa5cSAndroid Build Coastguard Worker
843*dfc6aa5cSAndroid Build Coastguard Worker if ((scalingFactors = tjGetScalingFactors(&nsf)) == NULL || nsf == 0)
844*dfc6aa5cSAndroid Build Coastguard Worker THROW("executing tjGetScalingFactors()", tjGetErrorStr());
845*dfc6aa5cSAndroid Build Coastguard Worker
846*dfc6aa5cSAndroid Build Coastguard Worker if (argc < minArg) usage(argv[0]);
847*dfc6aa5cSAndroid Build Coastguard Worker
848*dfc6aa5cSAndroid Build Coastguard Worker temp = strrchr(argv[1], '.');
849*dfc6aa5cSAndroid Build Coastguard Worker if (temp != NULL) {
850*dfc6aa5cSAndroid Build Coastguard Worker if (!strcasecmp(temp, ".bmp")) ext = "bmp";
851*dfc6aa5cSAndroid Build Coastguard Worker if (!strcasecmp(temp, ".jpg") || !strcasecmp(temp, ".jpeg"))
852*dfc6aa5cSAndroid Build Coastguard Worker decompOnly = 1;
853*dfc6aa5cSAndroid Build Coastguard Worker }
854*dfc6aa5cSAndroid Build Coastguard Worker
855*dfc6aa5cSAndroid Build Coastguard Worker fprintf(stderr, "\n");
856*dfc6aa5cSAndroid Build Coastguard Worker
857*dfc6aa5cSAndroid Build Coastguard Worker if (!decompOnly) {
858*dfc6aa5cSAndroid Build Coastguard Worker minArg = 3;
859*dfc6aa5cSAndroid Build Coastguard Worker if (argc < minArg) usage(argv[0]);
860*dfc6aa5cSAndroid Build Coastguard Worker if ((minQual = atoi(argv[2])) < 1 || minQual > 100) {
861*dfc6aa5cSAndroid Build Coastguard Worker puts("ERROR: Quality must be between 1 and 100.");
862*dfc6aa5cSAndroid Build Coastguard Worker exit(1);
863*dfc6aa5cSAndroid Build Coastguard Worker }
864*dfc6aa5cSAndroid Build Coastguard Worker if ((temp = strchr(argv[2], '-')) != NULL && strlen(temp) > 1 &&
865*dfc6aa5cSAndroid Build Coastguard Worker sscanf(&temp[1], "%d", &maxQual) == 1 && maxQual > minQual &&
866*dfc6aa5cSAndroid Build Coastguard Worker maxQual >= 1 && maxQual <= 100) {}
867*dfc6aa5cSAndroid Build Coastguard Worker else maxQual = minQual;
868*dfc6aa5cSAndroid Build Coastguard Worker }
869*dfc6aa5cSAndroid Build Coastguard Worker
870*dfc6aa5cSAndroid Build Coastguard Worker if (argc > minArg) {
871*dfc6aa5cSAndroid Build Coastguard Worker for (i = minArg; i < argc; i++) {
872*dfc6aa5cSAndroid Build Coastguard Worker if (!strcasecmp(argv[i], "-tile")) {
873*dfc6aa5cSAndroid Build Coastguard Worker doTile = 1; xformOpt |= TJXOPT_CROP;
874*dfc6aa5cSAndroid Build Coastguard Worker } else if (!strcasecmp(argv[i], "-fastupsample")) {
875*dfc6aa5cSAndroid Build Coastguard Worker fprintf(stderr, "Using fastest upsampling algorithm\n\n");
876*dfc6aa5cSAndroid Build Coastguard Worker flags |= TJFLAG_FASTUPSAMPLE;
877*dfc6aa5cSAndroid Build Coastguard Worker } else if (!strcasecmp(argv[i], "-fastdct")) {
878*dfc6aa5cSAndroid Build Coastguard Worker fprintf(stderr, "Using fastest DCT/IDCT algorithm\n\n");
879*dfc6aa5cSAndroid Build Coastguard Worker flags |= TJFLAG_FASTDCT;
880*dfc6aa5cSAndroid Build Coastguard Worker } else if (!strcasecmp(argv[i], "-accuratedct")) {
881*dfc6aa5cSAndroid Build Coastguard Worker fprintf(stderr, "Using most accurate DCT/IDCT algorithm\n\n");
882*dfc6aa5cSAndroid Build Coastguard Worker flags |= TJFLAG_ACCURATEDCT;
883*dfc6aa5cSAndroid Build Coastguard Worker } else if (!strcasecmp(argv[i], "-progressive")) {
884*dfc6aa5cSAndroid Build Coastguard Worker fprintf(stderr, "Using progressive entropy coding\n\n");
885*dfc6aa5cSAndroid Build Coastguard Worker flags |= TJFLAG_PROGRESSIVE;
886*dfc6aa5cSAndroid Build Coastguard Worker xformOpt |= TJXOPT_PROGRESSIVE;
887*dfc6aa5cSAndroid Build Coastguard Worker } else if (!strcasecmp(argv[i], "-rgb"))
888*dfc6aa5cSAndroid Build Coastguard Worker pf = TJPF_RGB;
889*dfc6aa5cSAndroid Build Coastguard Worker else if (!strcasecmp(argv[i], "-rgbx"))
890*dfc6aa5cSAndroid Build Coastguard Worker pf = TJPF_RGBX;
891*dfc6aa5cSAndroid Build Coastguard Worker else if (!strcasecmp(argv[i], "-bgr"))
892*dfc6aa5cSAndroid Build Coastguard Worker pf = TJPF_BGR;
893*dfc6aa5cSAndroid Build Coastguard Worker else if (!strcasecmp(argv[i], "-bgrx"))
894*dfc6aa5cSAndroid Build Coastguard Worker pf = TJPF_BGRX;
895*dfc6aa5cSAndroid Build Coastguard Worker else if (!strcasecmp(argv[i], "-xbgr"))
896*dfc6aa5cSAndroid Build Coastguard Worker pf = TJPF_XBGR;
897*dfc6aa5cSAndroid Build Coastguard Worker else if (!strcasecmp(argv[i], "-xrgb"))
898*dfc6aa5cSAndroid Build Coastguard Worker pf = TJPF_XRGB;
899*dfc6aa5cSAndroid Build Coastguard Worker else if (!strcasecmp(argv[i], "-cmyk"))
900*dfc6aa5cSAndroid Build Coastguard Worker pf = TJPF_CMYK;
901*dfc6aa5cSAndroid Build Coastguard Worker else if (!strcasecmp(argv[i], "-bottomup"))
902*dfc6aa5cSAndroid Build Coastguard Worker flags |= TJFLAG_BOTTOMUP;
903*dfc6aa5cSAndroid Build Coastguard Worker else if (!strcasecmp(argv[i], "-quiet"))
904*dfc6aa5cSAndroid Build Coastguard Worker quiet = 1;
905*dfc6aa5cSAndroid Build Coastguard Worker else if (!strcasecmp(argv[i], "-qq"))
906*dfc6aa5cSAndroid Build Coastguard Worker quiet = 2;
907*dfc6aa5cSAndroid Build Coastguard Worker else if (!strcasecmp(argv[i], "-scale") && i < argc - 1) {
908*dfc6aa5cSAndroid Build Coastguard Worker int temp1 = 0, temp2 = 0, match = 0;
909*dfc6aa5cSAndroid Build Coastguard Worker
910*dfc6aa5cSAndroid Build Coastguard Worker if (sscanf(argv[++i], "%d/%d", &temp1, &temp2) == 2) {
911*dfc6aa5cSAndroid Build Coastguard Worker for (j = 0; j < nsf; j++) {
912*dfc6aa5cSAndroid Build Coastguard Worker if ((double)temp1 / (double)temp2 ==
913*dfc6aa5cSAndroid Build Coastguard Worker (double)scalingFactors[j].num /
914*dfc6aa5cSAndroid Build Coastguard Worker (double)scalingFactors[j].denom) {
915*dfc6aa5cSAndroid Build Coastguard Worker sf = scalingFactors[j];
916*dfc6aa5cSAndroid Build Coastguard Worker match = 1; break;
917*dfc6aa5cSAndroid Build Coastguard Worker }
918*dfc6aa5cSAndroid Build Coastguard Worker }
919*dfc6aa5cSAndroid Build Coastguard Worker if (!match) usage(argv[0]);
920*dfc6aa5cSAndroid Build Coastguard Worker } else usage(argv[0]);
921*dfc6aa5cSAndroid Build Coastguard Worker } else if (!strcasecmp(argv[i], "-hflip"))
922*dfc6aa5cSAndroid Build Coastguard Worker xformOp = TJXOP_HFLIP;
923*dfc6aa5cSAndroid Build Coastguard Worker else if (!strcasecmp(argv[i], "-vflip"))
924*dfc6aa5cSAndroid Build Coastguard Worker xformOp = TJXOP_VFLIP;
925*dfc6aa5cSAndroid Build Coastguard Worker else if (!strcasecmp(argv[i], "-transpose"))
926*dfc6aa5cSAndroid Build Coastguard Worker xformOp = TJXOP_TRANSPOSE;
927*dfc6aa5cSAndroid Build Coastguard Worker else if (!strcasecmp(argv[i], "-transverse"))
928*dfc6aa5cSAndroid Build Coastguard Worker xformOp = TJXOP_TRANSVERSE;
929*dfc6aa5cSAndroid Build Coastguard Worker else if (!strcasecmp(argv[i], "-rot90"))
930*dfc6aa5cSAndroid Build Coastguard Worker xformOp = TJXOP_ROT90;
931*dfc6aa5cSAndroid Build Coastguard Worker else if (!strcasecmp(argv[i], "-rot180"))
932*dfc6aa5cSAndroid Build Coastguard Worker xformOp = TJXOP_ROT180;
933*dfc6aa5cSAndroid Build Coastguard Worker else if (!strcasecmp(argv[i], "-rot270"))
934*dfc6aa5cSAndroid Build Coastguard Worker xformOp = TJXOP_ROT270;
935*dfc6aa5cSAndroid Build Coastguard Worker else if (!strcasecmp(argv[i], "-grayscale"))
936*dfc6aa5cSAndroid Build Coastguard Worker xformOpt |= TJXOPT_GRAY;
937*dfc6aa5cSAndroid Build Coastguard Worker else if (!strcasecmp(argv[i], "-custom"))
938*dfc6aa5cSAndroid Build Coastguard Worker customFilter = dummyDCTFilter;
939*dfc6aa5cSAndroid Build Coastguard Worker else if (!strcasecmp(argv[i], "-nooutput"))
940*dfc6aa5cSAndroid Build Coastguard Worker xformOpt |= TJXOPT_NOOUTPUT;
941*dfc6aa5cSAndroid Build Coastguard Worker else if (!strcasecmp(argv[i], "-copynone"))
942*dfc6aa5cSAndroid Build Coastguard Worker xformOpt |= TJXOPT_COPYNONE;
943*dfc6aa5cSAndroid Build Coastguard Worker else if (!strcasecmp(argv[i], "-benchtime") && i < argc - 1) {
944*dfc6aa5cSAndroid Build Coastguard Worker double tempd = atof(argv[++i]);
945*dfc6aa5cSAndroid Build Coastguard Worker
946*dfc6aa5cSAndroid Build Coastguard Worker if (tempd > 0.0) benchTime = tempd;
947*dfc6aa5cSAndroid Build Coastguard Worker else usage(argv[0]);
948*dfc6aa5cSAndroid Build Coastguard Worker } else if (!strcasecmp(argv[i], "-warmup") && i < argc - 1) {
949*dfc6aa5cSAndroid Build Coastguard Worker double tempd = atof(argv[++i]);
950*dfc6aa5cSAndroid Build Coastguard Worker
951*dfc6aa5cSAndroid Build Coastguard Worker if (tempd >= 0.0) warmup = tempd;
952*dfc6aa5cSAndroid Build Coastguard Worker else usage(argv[0]);
953*dfc6aa5cSAndroid Build Coastguard Worker fprintf(stderr, "Warmup time = %.1f seconds\n\n", warmup);
954*dfc6aa5cSAndroid Build Coastguard Worker } else if (!strcasecmp(argv[i], "-alloc"))
955*dfc6aa5cSAndroid Build Coastguard Worker flags &= (~TJFLAG_NOREALLOC);
956*dfc6aa5cSAndroid Build Coastguard Worker else if (!strcasecmp(argv[i], "-bmp"))
957*dfc6aa5cSAndroid Build Coastguard Worker ext = "bmp";
958*dfc6aa5cSAndroid Build Coastguard Worker else if (!strcasecmp(argv[i], "-yuv")) {
959*dfc6aa5cSAndroid Build Coastguard Worker fprintf(stderr, "Testing planar YUV encoding/decoding\n\n");
960*dfc6aa5cSAndroid Build Coastguard Worker doYUV = 1;
961*dfc6aa5cSAndroid Build Coastguard Worker } else if (!strcasecmp(argv[i], "-yuvpad") && i < argc - 1) {
962*dfc6aa5cSAndroid Build Coastguard Worker int tempi = atoi(argv[++i]);
963*dfc6aa5cSAndroid Build Coastguard Worker
964*dfc6aa5cSAndroid Build Coastguard Worker if (tempi >= 1 && (tempi & (tempi - 1)) == 0) yuvAlign = tempi;
965*dfc6aa5cSAndroid Build Coastguard Worker else usage(argv[0]);
966*dfc6aa5cSAndroid Build Coastguard Worker } else if (!strcasecmp(argv[i], "-subsamp") && i < argc - 1) {
967*dfc6aa5cSAndroid Build Coastguard Worker i++;
968*dfc6aa5cSAndroid Build Coastguard Worker if (toupper(argv[i][0]) == 'G') subsamp = TJSAMP_GRAY;
969*dfc6aa5cSAndroid Build Coastguard Worker else {
970*dfc6aa5cSAndroid Build Coastguard Worker int tempi = atoi(argv[i]);
971*dfc6aa5cSAndroid Build Coastguard Worker
972*dfc6aa5cSAndroid Build Coastguard Worker switch (tempi) {
973*dfc6aa5cSAndroid Build Coastguard Worker case 444: subsamp = TJSAMP_444; break;
974*dfc6aa5cSAndroid Build Coastguard Worker case 422: subsamp = TJSAMP_422; break;
975*dfc6aa5cSAndroid Build Coastguard Worker case 440: subsamp = TJSAMP_440; break;
976*dfc6aa5cSAndroid Build Coastguard Worker case 420: subsamp = TJSAMP_420; break;
977*dfc6aa5cSAndroid Build Coastguard Worker case 411: subsamp = TJSAMP_411; break;
978*dfc6aa5cSAndroid Build Coastguard Worker default: usage(argv[0]);
979*dfc6aa5cSAndroid Build Coastguard Worker }
980*dfc6aa5cSAndroid Build Coastguard Worker }
981*dfc6aa5cSAndroid Build Coastguard Worker } else if (!strcasecmp(argv[i], "-componly"))
982*dfc6aa5cSAndroid Build Coastguard Worker compOnly = 1;
983*dfc6aa5cSAndroid Build Coastguard Worker else if (!strcasecmp(argv[i], "-nowrite"))
984*dfc6aa5cSAndroid Build Coastguard Worker doWrite = 0;
985*dfc6aa5cSAndroid Build Coastguard Worker else if (!strcasecmp(argv[i], "-limitscans"))
986*dfc6aa5cSAndroid Build Coastguard Worker flags |= TJFLAG_LIMITSCANS;
987*dfc6aa5cSAndroid Build Coastguard Worker else if (!strcasecmp(argv[i], "-stoponwarning"))
988*dfc6aa5cSAndroid Build Coastguard Worker flags |= TJFLAG_STOPONWARNING;
989*dfc6aa5cSAndroid Build Coastguard Worker else usage(argv[0]);
990*dfc6aa5cSAndroid Build Coastguard Worker }
991*dfc6aa5cSAndroid Build Coastguard Worker }
992*dfc6aa5cSAndroid Build Coastguard Worker
993*dfc6aa5cSAndroid Build Coastguard Worker if ((sf.num != 1 || sf.denom != 1) && doTile) {
994*dfc6aa5cSAndroid Build Coastguard Worker fprintf(stderr, "Disabling tiled compression/decompression tests, because those tests do not\n");
995*dfc6aa5cSAndroid Build Coastguard Worker fprintf(stderr, "work when scaled decompression is enabled.\n\n");
996*dfc6aa5cSAndroid Build Coastguard Worker doTile = 0; xformOpt &= (~TJXOPT_CROP);
997*dfc6aa5cSAndroid Build Coastguard Worker }
998*dfc6aa5cSAndroid Build Coastguard Worker
999*dfc6aa5cSAndroid Build Coastguard Worker if ((flags & TJFLAG_NOREALLOC) == 0 && doTile) {
1000*dfc6aa5cSAndroid Build Coastguard Worker fprintf(stderr, "Disabling tiled compression/decompression tests, because those tests do not\n");
1001*dfc6aa5cSAndroid Build Coastguard Worker fprintf(stderr, "work when dynamic JPEG buffer allocation is enabled.\n\n");
1002*dfc6aa5cSAndroid Build Coastguard Worker doTile = 0; xformOpt &= (~TJXOPT_CROP);
1003*dfc6aa5cSAndroid Build Coastguard Worker }
1004*dfc6aa5cSAndroid Build Coastguard Worker
1005*dfc6aa5cSAndroid Build Coastguard Worker if (!decompOnly) {
1006*dfc6aa5cSAndroid Build Coastguard Worker if ((srcBuf = tjLoadImage(argv[1], &w, 1, &h, &pf, flags)) == NULL)
1007*dfc6aa5cSAndroid Build Coastguard Worker THROW_TJG("loading input image");
1008*dfc6aa5cSAndroid Build Coastguard Worker temp = strrchr(argv[1], '.');
1009*dfc6aa5cSAndroid Build Coastguard Worker if (temp != NULL) *temp = '\0';
1010*dfc6aa5cSAndroid Build Coastguard Worker }
1011*dfc6aa5cSAndroid Build Coastguard Worker
1012*dfc6aa5cSAndroid Build Coastguard Worker if (quiet == 1 && !decompOnly) {
1013*dfc6aa5cSAndroid Build Coastguard Worker fprintf(stderr, "All performance values in Mpixels/sec\n\n");
1014*dfc6aa5cSAndroid Build Coastguard Worker fprintf(stderr, "Pixel JPEG JPEG %s %s ",
1015*dfc6aa5cSAndroid Build Coastguard Worker doTile ? "Tile " : "Image", doTile ? "Tile " : "Image");
1016*dfc6aa5cSAndroid Build Coastguard Worker if (doYUV) fprintf(stderr, "Encode ");
1017*dfc6aa5cSAndroid Build Coastguard Worker fprintf(stderr, "Comp Comp Decomp ");
1018*dfc6aa5cSAndroid Build Coastguard Worker if (doYUV) fprintf(stderr, "Decode");
1019*dfc6aa5cSAndroid Build Coastguard Worker fprintf(stderr, "\n");
1020*dfc6aa5cSAndroid Build Coastguard Worker fprintf(stderr, "Format Subsamp Qual Width Height ");
1021*dfc6aa5cSAndroid Build Coastguard Worker if (doYUV) fprintf(stderr, "Perf ");
1022*dfc6aa5cSAndroid Build Coastguard Worker fprintf(stderr, "Perf Ratio Perf ");
1023*dfc6aa5cSAndroid Build Coastguard Worker if (doYUV) fprintf(stderr, "Perf");
1024*dfc6aa5cSAndroid Build Coastguard Worker fprintf(stderr, "\n\n");
1025*dfc6aa5cSAndroid Build Coastguard Worker }
1026*dfc6aa5cSAndroid Build Coastguard Worker
1027*dfc6aa5cSAndroid Build Coastguard Worker if (decompOnly) {
1028*dfc6aa5cSAndroid Build Coastguard Worker decompTest(argv[1]);
1029*dfc6aa5cSAndroid Build Coastguard Worker fprintf(stderr, "\n");
1030*dfc6aa5cSAndroid Build Coastguard Worker goto bailout;
1031*dfc6aa5cSAndroid Build Coastguard Worker }
1032*dfc6aa5cSAndroid Build Coastguard Worker if (subsamp >= 0 && subsamp < TJ_NUMSAMP) {
1033*dfc6aa5cSAndroid Build Coastguard Worker for (i = maxQual; i >= minQual; i--)
1034*dfc6aa5cSAndroid Build Coastguard Worker fullTest(srcBuf, w, h, subsamp, i, argv[1]);
1035*dfc6aa5cSAndroid Build Coastguard Worker fprintf(stderr, "\n");
1036*dfc6aa5cSAndroid Build Coastguard Worker } else {
1037*dfc6aa5cSAndroid Build Coastguard Worker if (pf != TJPF_CMYK) {
1038*dfc6aa5cSAndroid Build Coastguard Worker for (i = maxQual; i >= minQual; i--)
1039*dfc6aa5cSAndroid Build Coastguard Worker fullTest(srcBuf, w, h, TJSAMP_GRAY, i, argv[1]);
1040*dfc6aa5cSAndroid Build Coastguard Worker fprintf(stderr, "\n");
1041*dfc6aa5cSAndroid Build Coastguard Worker }
1042*dfc6aa5cSAndroid Build Coastguard Worker for (i = maxQual; i >= minQual; i--)
1043*dfc6aa5cSAndroid Build Coastguard Worker fullTest(srcBuf, w, h, TJSAMP_420, i, argv[1]);
1044*dfc6aa5cSAndroid Build Coastguard Worker fprintf(stderr, "\n");
1045*dfc6aa5cSAndroid Build Coastguard Worker for (i = maxQual; i >= minQual; i--)
1046*dfc6aa5cSAndroid Build Coastguard Worker fullTest(srcBuf, w, h, TJSAMP_422, i, argv[1]);
1047*dfc6aa5cSAndroid Build Coastguard Worker fprintf(stderr, "\n");
1048*dfc6aa5cSAndroid Build Coastguard Worker for (i = maxQual; i >= minQual; i--)
1049*dfc6aa5cSAndroid Build Coastguard Worker fullTest(srcBuf, w, h, TJSAMP_444, i, argv[1]);
1050*dfc6aa5cSAndroid Build Coastguard Worker fprintf(stderr, "\n");
1051*dfc6aa5cSAndroid Build Coastguard Worker }
1052*dfc6aa5cSAndroid Build Coastguard Worker
1053*dfc6aa5cSAndroid Build Coastguard Worker bailout:
1054*dfc6aa5cSAndroid Build Coastguard Worker tjFree(srcBuf);
1055*dfc6aa5cSAndroid Build Coastguard Worker return retval;
1056*dfc6aa5cSAndroid Build Coastguard Worker }
1057