Coverage Report

Created: 2023-12-08 06:53

/src/freeimage-svn/FreeImage/trunk/Source/LibTIFF4/tif_pixarlog.c
Line
Count
Source (jump to first uncovered line)
1
/*
2
 * Copyright (c) 1996-1997 Sam Leffler
3
 * Copyright (c) 1996 Pixar
4
 *
5
 * Permission to use, copy, modify, distribute, and sell this software and
6
 * its documentation for any purpose is hereby granted without fee, provided
7
 * that (i) the above copyright notices and this permission notice appear in
8
 * all copies of the software and related documentation, and (ii) the names of
9
 * Pixar, Sam Leffler and Silicon Graphics may not be used in any advertising or
10
 * publicity relating to the software without the specific, prior written
11
 * permission of Pixar, Sam Leffler and Silicon Graphics.
12
 *
13
 * THE SOFTWARE IS PROVIDED "AS-IS" AND WITHOUT WARRANTY OF ANY KIND,
14
 * EXPRESS, IMPLIED OR OTHERWISE, INCLUDING WITHOUT LIMITATION, ANY
15
 * WARRANTY OF MERCHANTABILITY OR FITNESS FOR A PARTICULAR PURPOSE.
16
 *
17
 * IN NO EVENT SHALL PIXAR, SAM LEFFLER OR SILICON GRAPHICS BE LIABLE FOR
18
 * ANY SPECIAL, INCIDENTAL, INDIRECT OR CONSEQUENTIAL DAMAGES OF ANY KIND,
19
 * OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS,
20
 * WHETHER OR NOT ADVISED OF THE POSSIBILITY OF DAMAGE, AND ON ANY THEORY OF
21
 * LIABILITY, ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE
22
 * OF THIS SOFTWARE.
23
 */
24
25
#include "tiffiop.h"
26
#ifdef PIXARLOG_SUPPORT
27
28
/*
29
 * TIFF Library.
30
 * PixarLog Compression Support
31
 *
32
 * Contributed by Dan McCoy.
33
 *
34
 * PixarLog film support uses the TIFF library to store companded
35
 * 11 bit values into a tiff file, which are compressed using the
36
 * zip compressor.
37
 *
38
 * The codec can take as input and produce as output 32-bit IEEE float values
39
 * as well as 16-bit or 8-bit unsigned integer values.
40
 *
41
 * On writing any of the above are converted into the internal
42
 * 11-bit log format.   In the case of  8 and 16 bit values, the
43
 * input is assumed to be unsigned linear color values that represent
44
 * the range 0-1.  In the case of IEEE values, the 0-1 range is assumed to
45
 * be the normal linear color range, in addition over 1 values are
46
 * accepted up to a value of about 25.0 to encode "hot" highlights and such.
47
 * The encoding is lossless for 8-bit values, slightly lossy for the
48
 * other bit depths.  The actual color precision should be better
49
 * than the human eye can perceive with extra room to allow for
50
 * error introduced by further image computation.  As with any quantized
51
 * color format, it is possible to perform image calculations which
52
 * expose the quantization error. This format should certainly be less
53
 * susceptible to such errors than standard 8-bit encodings, but more
54
 * susceptible than straight 16-bit or 32-bit encodings.
55
 *
56
 * On reading the internal format is converted to the desired output format.
57
 * The program can request which format it desires by setting the internal
58
 * pseudo tag TIFFTAG_PIXARLOGDATAFMT to one of these possible values:
59
 *  PIXARLOGDATAFMT_FLOAT     = provide IEEE float values.
60
 *  PIXARLOGDATAFMT_16BIT     = provide unsigned 16-bit integer values
61
 *  PIXARLOGDATAFMT_8BIT      = provide unsigned 8-bit integer values
62
 *
63
 * alternately PIXARLOGDATAFMT_8BITABGR provides unsigned 8-bit integer
64
 * values with the difference that if there are exactly three or four channels
65
 * (rgb or rgba) it swaps the channel order (bgr or abgr).
66
 *
67
 * PIXARLOGDATAFMT_11BITLOG provides the internal encoding directly
68
 * packed in 16-bit values.   However no tools are supplied for interpreting
69
 * these values.
70
 *
71
 * "hot" (over 1.0) areas written in floating point get clamped to
72
 * 1.0 in the integer data types.
73
 *
74
 * When the file is closed after writing, the bit depth and sample format
75
 * are set always to appear as if 8-bit data has been written into it.
76
 * That way a naive program unaware of the particulars of the encoding
77
 * gets the format it is most likely able to handle.
78
 *
79
 * The codec does it's own horizontal differencing step on the coded
80
 * values so the libraries predictor stuff should be turned off.
81
 * The codec also handle byte swapping the encoded values as necessary
82
 * since the library does not have the information necessary
83
 * to know the bit depth of the raw unencoded buffer.
84
 *
85
 * NOTE: This decoder does not appear to update tif_rawcp, and tif_rawcc.
86
 * This can cause problems with the implementation of CHUNKY_STRIP_READ_SUPPORT
87
 * as noted in http://trac.osgeo.org/gdal/ticket/3894.   FrankW - Jan'11
88
 */
89
90
#include "tif_predict.h"
91
#include "zlib.h"
92
93
#include <math.h>
94
#include <stdio.h>
95
#include <stdlib.h>
96
97
/* Tables for converting to/from 11 bit coded values */
98
99
0
#define TSIZE 2048   /* decode table size (11-bit tokens) */
100
0
#define TSIZEP1 2049 /* Plus one for slop */
101
0
#define ONE 1250     /* token value of 1.0 exactly */
102
0
#define RATIO 1.004  /* nominal ratio for log part */
103
104
0
#define CODE_MASK 0x7ff /* 11 bits. */
105
106
static float Fltsize;
107
static float LogK1, LogK2;
108
109
#define REPEAT(n, op)                                                          \
110
0
    {                                                                          \
111
0
        int i;                                                                 \
112
0
        i = n;                                                                 \
113
0
        do                                                                     \
114
0
        {                                                                      \
115
0
            i--;                                                               \
116
0
            op;                                                                \
117
0
        } while (i > 0);                                                       \
118
0
    }
119
120
static void horizontalAccumulateF(uint16_t *wp, int n, int stride, float *op,
121
                                  float *ToLinearF)
122
0
{
123
0
    register unsigned int cr, cg, cb, ca, mask;
124
0
    register float t0, t1, t2, t3;
125
126
0
    if (n >= stride)
127
0
    {
128
0
        mask = CODE_MASK;
129
0
        if (stride == 3)
130
0
        {
131
0
            t0 = ToLinearF[cr = (wp[0] & mask)];
132
0
            t1 = ToLinearF[cg = (wp[1] & mask)];
133
0
            t2 = ToLinearF[cb = (wp[2] & mask)];
134
0
            op[0] = t0;
135
0
            op[1] = t1;
136
0
            op[2] = t2;
137
0
            n -= 3;
138
0
            while (n > 0)
139
0
            {
140
0
                wp += 3;
141
0
                op += 3;
142
0
                n -= 3;
143
0
                t0 = ToLinearF[(cr += wp[0]) & mask];
144
0
                t1 = ToLinearF[(cg += wp[1]) & mask];
145
0
                t2 = ToLinearF[(cb += wp[2]) & mask];
146
0
                op[0] = t0;
147
0
                op[1] = t1;
148
0
                op[2] = t2;
149
0
            }
150
0
        }
151
0
        else if (stride == 4)
152
0
        {
153
0
            t0 = ToLinearF[cr = (wp[0] & mask)];
154
0
            t1 = ToLinearF[cg = (wp[1] & mask)];
155
0
            t2 = ToLinearF[cb = (wp[2] & mask)];
156
0
            t3 = ToLinearF[ca = (wp[3] & mask)];
157
0
            op[0] = t0;
158
0
            op[1] = t1;
159
0
            op[2] = t2;
160
0
            op[3] = t3;
161
0
            n -= 4;
162
0
            while (n > 0)
163
0
            {
164
0
                wp += 4;
165
0
                op += 4;
166
0
                n -= 4;
167
0
                t0 = ToLinearF[(cr += wp[0]) & mask];
168
0
                t1 = ToLinearF[(cg += wp[1]) & mask];
169
0
                t2 = ToLinearF[(cb += wp[2]) & mask];
170
0
                t3 = ToLinearF[(ca += wp[3]) & mask];
171
0
                op[0] = t0;
172
0
                op[1] = t1;
173
0
                op[2] = t2;
174
0
                op[3] = t3;
175
0
            }
176
0
        }
177
0
        else
178
0
        {
179
0
            REPEAT(stride, *op = ToLinearF[*wp & mask]; wp++; op++)
180
0
            n -= stride;
181
0
            while (n > 0)
182
0
            {
183
0
                REPEAT(stride, wp[stride] += *wp; *op = ToLinearF[*wp & mask];
184
0
                       wp++; op++)
185
0
                n -= stride;
186
0
            }
187
0
        }
188
0
    }
189
0
}
190
191
static void horizontalAccumulate12(uint16_t *wp, int n, int stride, int16_t *op,
192
                                   float *ToLinearF)
193
0
{
194
0
    register unsigned int cr, cg, cb, ca, mask;
195
0
    register float t0, t1, t2, t3;
196
197
0
#define SCALE12 2048.0F
198
0
#define CLAMP12(t) (((t) < 3071) ? (uint16_t)(t) : 3071)
199
200
0
    if (n >= stride)
201
0
    {
202
0
        mask = CODE_MASK;
203
0
        if (stride == 3)
204
0
        {
205
0
            t0 = ToLinearF[cr = (wp[0] & mask)] * SCALE12;
206
0
            t1 = ToLinearF[cg = (wp[1] & mask)] * SCALE12;
207
0
            t2 = ToLinearF[cb = (wp[2] & mask)] * SCALE12;
208
0
            op[0] = CLAMP12(t0);
209
0
            op[1] = CLAMP12(t1);
210
0
            op[2] = CLAMP12(t2);
211
0
            n -= 3;
212
0
            while (n > 0)
213
0
            {
214
0
                wp += 3;
215
0
                op += 3;
216
0
                n -= 3;
217
0
                t0 = ToLinearF[(cr += wp[0]) & mask] * SCALE12;
218
0
                t1 = ToLinearF[(cg += wp[1]) & mask] * SCALE12;
219
0
                t2 = ToLinearF[(cb += wp[2]) & mask] * SCALE12;
220
0
                op[0] = CLAMP12(t0);
221
0
                op[1] = CLAMP12(t1);
222
0
                op[2] = CLAMP12(t2);
223
0
            }
224
0
        }
225
0
        else if (stride == 4)
226
0
        {
227
0
            t0 = ToLinearF[cr = (wp[0] & mask)] * SCALE12;
228
0
            t1 = ToLinearF[cg = (wp[1] & mask)] * SCALE12;
229
0
            t2 = ToLinearF[cb = (wp[2] & mask)] * SCALE12;
230
0
            t3 = ToLinearF[ca = (wp[3] & mask)] * SCALE12;
231
0
            op[0] = CLAMP12(t0);
232
0
            op[1] = CLAMP12(t1);
233
0
            op[2] = CLAMP12(t2);
234
0
            op[3] = CLAMP12(t3);
235
0
            n -= 4;
236
0
            while (n > 0)
237
0
            {
238
0
                wp += 4;
239
0
                op += 4;
240
0
                n -= 4;
241
0
                t0 = ToLinearF[(cr += wp[0]) & mask] * SCALE12;
242
0
                t1 = ToLinearF[(cg += wp[1]) & mask] * SCALE12;
243
0
                t2 = ToLinearF[(cb += wp[2]) & mask] * SCALE12;
244
0
                t3 = ToLinearF[(ca += wp[3]) & mask] * SCALE12;
245
0
                op[0] = CLAMP12(t0);
246
0
                op[1] = CLAMP12(t1);
247
0
                op[2] = CLAMP12(t2);
248
0
                op[3] = CLAMP12(t3);
249
0
            }
250
0
        }
251
0
        else
252
0
        {
253
0
            REPEAT(stride, t0 = ToLinearF[*wp & mask] * SCALE12;
254
0
                   *op = CLAMP12(t0); wp++; op++)
255
0
            n -= stride;
256
0
            while (n > 0)
257
0
            {
258
0
                REPEAT(stride, wp[stride] += *wp;
259
0
                       t0 = ToLinearF[wp[stride] & mask] * SCALE12;
260
0
                       *op = CLAMP12(t0); wp++; op++)
261
0
                n -= stride;
262
0
            }
263
0
        }
264
0
    }
265
0
}
266
267
static void horizontalAccumulate16(uint16_t *wp, int n, int stride,
268
                                   uint16_t *op, uint16_t *ToLinear16)
269
0
{
270
0
    register unsigned int cr, cg, cb, ca, mask;
271
272
0
    if (n >= stride)
273
0
    {
274
0
        mask = CODE_MASK;
275
0
        if (stride == 3)
276
0
        {
277
0
            op[0] = ToLinear16[cr = (wp[0] & mask)];
278
0
            op[1] = ToLinear16[cg = (wp[1] & mask)];
279
0
            op[2] = ToLinear16[cb = (wp[2] & mask)];
280
0
            n -= 3;
281
0
            while (n > 0)
282
0
            {
283
0
                wp += 3;
284
0
                op += 3;
285
0
                n -= 3;
286
0
                op[0] = ToLinear16[(cr += wp[0]) & mask];
287
0
                op[1] = ToLinear16[(cg += wp[1]) & mask];
288
0
                op[2] = ToLinear16[(cb += wp[2]) & mask];
289
0
            }
290
0
        }
291
0
        else if (stride == 4)
292
0
        {
293
0
            op[0] = ToLinear16[cr = (wp[0] & mask)];
294
0
            op[1] = ToLinear16[cg = (wp[1] & mask)];
295
0
            op[2] = ToLinear16[cb = (wp[2] & mask)];
296
0
            op[3] = ToLinear16[ca = (wp[3] & mask)];
297
0
            n -= 4;
298
0
            while (n > 0)
299
0
            {
300
0
                wp += 4;
301
0
                op += 4;
302
0
                n -= 4;
303
0
                op[0] = ToLinear16[(cr += wp[0]) & mask];
304
0
                op[1] = ToLinear16[(cg += wp[1]) & mask];
305
0
                op[2] = ToLinear16[(cb += wp[2]) & mask];
306
0
                op[3] = ToLinear16[(ca += wp[3]) & mask];
307
0
            }
308
0
        }
309
0
        else
310
0
        {
311
0
            REPEAT(stride, *op = ToLinear16[*wp & mask]; wp++; op++)
312
0
            n -= stride;
313
0
            while (n > 0)
314
0
            {
315
0
                REPEAT(stride, wp[stride] += *wp; *op = ToLinear16[*wp & mask];
316
0
                       wp++; op++)
317
0
                n -= stride;
318
0
            }
319
0
        }
320
0
    }
321
0
}
322
323
/*
324
 * Returns the log encoded 11-bit values with the horizontal
325
 * differencing undone.
326
 */
327
static void horizontalAccumulate11(uint16_t *wp, int n, int stride,
328
                                   uint16_t *op)
329
0
{
330
0
    register unsigned int cr, cg, cb, ca, mask;
331
332
0
    if (n >= stride)
333
0
    {
334
0
        mask = CODE_MASK;
335
0
        if (stride == 3)
336
0
        {
337
0
            op[0] = wp[0];
338
0
            op[1] = wp[1];
339
0
            op[2] = wp[2];
340
0
            cr = wp[0];
341
0
            cg = wp[1];
342
0
            cb = wp[2];
343
0
            n -= 3;
344
0
            while (n > 0)
345
0
            {
346
0
                wp += 3;
347
0
                op += 3;
348
0
                n -= 3;
349
0
                op[0] = (uint16_t)((cr += wp[0]) & mask);
350
0
                op[1] = (uint16_t)((cg += wp[1]) & mask);
351
0
                op[2] = (uint16_t)((cb += wp[2]) & mask);
352
0
            }
353
0
        }
354
0
        else if (stride == 4)
355
0
        {
356
0
            op[0] = wp[0];
357
0
            op[1] = wp[1];
358
0
            op[2] = wp[2];
359
0
            op[3] = wp[3];
360
0
            cr = wp[0];
361
0
            cg = wp[1];
362
0
            cb = wp[2];
363
0
            ca = wp[3];
364
0
            n -= 4;
365
0
            while (n > 0)
366
0
            {
367
0
                wp += 4;
368
0
                op += 4;
369
0
                n -= 4;
370
0
                op[0] = (uint16_t)((cr += wp[0]) & mask);
371
0
                op[1] = (uint16_t)((cg += wp[1]) & mask);
372
0
                op[2] = (uint16_t)((cb += wp[2]) & mask);
373
0
                op[3] = (uint16_t)((ca += wp[3]) & mask);
374
0
            }
375
0
        }
376
0
        else
377
0
        {
378
0
            REPEAT(stride, *op = *wp & mask; wp++; op++)
379
0
            n -= stride;
380
0
            while (n > 0)
381
0
            {
382
0
                REPEAT(stride, wp[stride] += *wp; *op = *wp & mask; wp++; op++)
383
0
                n -= stride;
384
0
            }
385
0
        }
386
0
    }
387
0
}
388
389
static void horizontalAccumulate8(uint16_t *wp, int n, int stride,
390
                                  unsigned char *op, unsigned char *ToLinear8)
391
0
{
392
0
    register unsigned int cr, cg, cb, ca, mask;
393
394
0
    if (n >= stride)
395
0
    {
396
0
        mask = CODE_MASK;
397
0
        if (stride == 3)
398
0
        {
399
0
            op[0] = ToLinear8[cr = (wp[0] & mask)];
400
0
            op[1] = ToLinear8[cg = (wp[1] & mask)];
401
0
            op[2] = ToLinear8[cb = (wp[2] & mask)];
402
0
            n -= 3;
403
0
            while (n > 0)
404
0
            {
405
0
                n -= 3;
406
0
                wp += 3;
407
0
                op += 3;
408
0
                op[0] = ToLinear8[(cr += wp[0]) & mask];
409
0
                op[1] = ToLinear8[(cg += wp[1]) & mask];
410
0
                op[2] = ToLinear8[(cb += wp[2]) & mask];
411
0
            }
412
0
        }
413
0
        else if (stride == 4)
414
0
        {
415
0
            op[0] = ToLinear8[cr = (wp[0] & mask)];
416
0
            op[1] = ToLinear8[cg = (wp[1] & mask)];
417
0
            op[2] = ToLinear8[cb = (wp[2] & mask)];
418
0
            op[3] = ToLinear8[ca = (wp[3] & mask)];
419
0
            n -= 4;
420
0
            while (n > 0)
421
0
            {
422
0
                n -= 4;
423
0
                wp += 4;
424
0
                op += 4;
425
0
                op[0] = ToLinear8[(cr += wp[0]) & mask];
426
0
                op[1] = ToLinear8[(cg += wp[1]) & mask];
427
0
                op[2] = ToLinear8[(cb += wp[2]) & mask];
428
0
                op[3] = ToLinear8[(ca += wp[3]) & mask];
429
0
            }
430
0
        }
431
0
        else
432
0
        {
433
0
            REPEAT(stride, *op = ToLinear8[*wp & mask]; wp++; op++)
434
0
            n -= stride;
435
0
            while (n > 0)
436
0
            {
437
0
                REPEAT(stride, wp[stride] += *wp; *op = ToLinear8[*wp & mask];
438
0
                       wp++; op++)
439
0
                n -= stride;
440
0
            }
441
0
        }
442
0
    }
443
0
}
444
445
static void horizontalAccumulate8abgr(uint16_t *wp, int n, int stride,
446
                                      unsigned char *op,
447
                                      unsigned char *ToLinear8)
448
0
{
449
0
    register unsigned int cr, cg, cb, ca, mask;
450
0
    register unsigned char t0, t1, t2, t3;
451
452
0
    if (n >= stride)
453
0
    {
454
0
        mask = CODE_MASK;
455
0
        if (stride == 3)
456
0
        {
457
0
            op[0] = 0;
458
0
            t1 = ToLinear8[cb = (wp[2] & mask)];
459
0
            t2 = ToLinear8[cg = (wp[1] & mask)];
460
0
            t3 = ToLinear8[cr = (wp[0] & mask)];
461
0
            op[1] = t1;
462
0
            op[2] = t2;
463
0
            op[3] = t3;
464
0
            n -= 3;
465
0
            while (n > 0)
466
0
            {
467
0
                n -= 3;
468
0
                wp += 3;
469
0
                op += 4;
470
0
                op[0] = 0;
471
0
                t1 = ToLinear8[(cb += wp[2]) & mask];
472
0
                t2 = ToLinear8[(cg += wp[1]) & mask];
473
0
                t3 = ToLinear8[(cr += wp[0]) & mask];
474
0
                op[1] = t1;
475
0
                op[2] = t2;
476
0
                op[3] = t3;
477
0
            }
478
0
        }
479
0
        else if (stride == 4)
480
0
        {
481
0
            t0 = ToLinear8[ca = (wp[3] & mask)];
482
0
            t1 = ToLinear8[cb = (wp[2] & mask)];
483
0
            t2 = ToLinear8[cg = (wp[1] & mask)];
484
0
            t3 = ToLinear8[cr = (wp[0] & mask)];
485
0
            op[0] = t0;
486
0
            op[1] = t1;
487
0
            op[2] = t2;
488
0
            op[3] = t3;
489
0
            n -= 4;
490
0
            while (n > 0)
491
0
            {
492
0
                n -= 4;
493
0
                wp += 4;
494
0
                op += 4;
495
0
                t0 = ToLinear8[(ca += wp[3]) & mask];
496
0
                t1 = ToLinear8[(cb += wp[2]) & mask];
497
0
                t2 = ToLinear8[(cg += wp[1]) & mask];
498
0
                t3 = ToLinear8[(cr += wp[0]) & mask];
499
0
                op[0] = t0;
500
0
                op[1] = t1;
501
0
                op[2] = t2;
502
0
                op[3] = t3;
503
0
            }
504
0
        }
505
0
        else
506
0
        {
507
0
            REPEAT(stride, *op = ToLinear8[*wp & mask]; wp++; op++)
508
0
            n -= stride;
509
0
            while (n > 0)
510
0
            {
511
0
                REPEAT(stride, wp[stride] += *wp; *op = ToLinear8[*wp & mask];
512
0
                       wp++; op++)
513
0
                n -= stride;
514
0
            }
515
0
        }
516
0
    }
517
0
}
518
519
/*
520
 * State block for each open TIFF
521
 * file using PixarLog compression/decompression.
522
 */
523
typedef struct
524
{
525
    TIFFPredictorState predict;
526
    z_stream stream;
527
    tmsize_t tbuf_size; /* only set/used on reading for now */
528
    uint16_t *tbuf;
529
    uint16_t stride;
530
    int state;
531
    int user_datafmt;
532
    int quality;
533
0
#define PLSTATE_INIT 1
534
535
    TIFFVSetMethod vgetparent; /* super-class method */
536
    TIFFVSetMethod vsetparent; /* super-class method */
537
538
    float *ToLinearF;
539
    uint16_t *ToLinear16;
540
    unsigned char *ToLinear8;
541
    uint16_t *FromLT2;
542
    uint16_t *From14; /* Really for 16-bit data, but we shift down 2 */
543
    uint16_t *From8;
544
545
} PixarLogState;
546
547
static int PixarLogMakeTables(TIFF *tif, PixarLogState *sp)
548
0
{
549
550
    /*
551
     *    We make several tables here to convert between various external
552
     *    representations (float, 16-bit, and 8-bit) and the internal
553
     *    11-bit companded representation.  The 11-bit representation has two
554
     *    distinct regions.  A linear bottom end up through .018316 in steps
555
     *    of about .000073, and a region of constant ratio up to about 25.
556
     *    These floating point numbers are stored in the main table ToLinearF.
557
     *    All other tables are derived from this one.  The tables (and the
558
     *    ratios) are continuous at the internal seam.
559
     */
560
561
0
    int nlin, lt2size;
562
0
    int i, j;
563
0
    double b, c, linstep, v;
564
0
    float *ToLinearF;
565
0
    uint16_t *ToLinear16;
566
0
    unsigned char *ToLinear8;
567
0
    uint16_t *FromLT2;
568
0
    uint16_t *From14; /* Really for 16-bit data, but we shift down 2 */
569
0
    uint16_t *From8;
570
571
0
    c = log(RATIO);
572
0
    nlin = (int)(1. / c); /* nlin must be an integer */
573
0
    c = 1. / nlin;
574
0
    b = exp(-c * ONE); /* multiplicative scale factor [b*exp(c*ONE) = 1] */
575
0
    linstep = b * c * exp(1.);
576
577
0
    LogK1 = (float)(1. / c); /* if (v >= 2)  token = k1*log(v*k2) */
578
0
    LogK2 = (float)(1. / b);
579
0
    lt2size = (int)(2. / linstep) + 1;
580
0
    FromLT2 = (uint16_t *)_TIFFmallocExt(tif, lt2size * sizeof(uint16_t));
581
0
    From14 = (uint16_t *)_TIFFmallocExt(tif, 16384 * sizeof(uint16_t));
582
0
    From8 = (uint16_t *)_TIFFmallocExt(tif, 256 * sizeof(uint16_t));
583
0
    ToLinearF = (float *)_TIFFmallocExt(tif, TSIZEP1 * sizeof(float));
584
0
    ToLinear16 = (uint16_t *)_TIFFmallocExt(tif, TSIZEP1 * sizeof(uint16_t));
585
0
    ToLinear8 =
586
0
        (unsigned char *)_TIFFmallocExt(tif, TSIZEP1 * sizeof(unsigned char));
587
0
    if (FromLT2 == NULL || From14 == NULL || From8 == NULL ||
588
0
        ToLinearF == NULL || ToLinear16 == NULL || ToLinear8 == NULL)
589
0
    {
590
0
        if (FromLT2)
591
0
            _TIFFfreeExt(tif, FromLT2);
592
0
        if (From14)
593
0
            _TIFFfreeExt(tif, From14);
594
0
        if (From8)
595
0
            _TIFFfreeExt(tif, From8);
596
0
        if (ToLinearF)
597
0
            _TIFFfreeExt(tif, ToLinearF);
598
0
        if (ToLinear16)
599
0
            _TIFFfreeExt(tif, ToLinear16);
600
0
        if (ToLinear8)
601
0
            _TIFFfreeExt(tif, ToLinear8);
602
0
        sp->FromLT2 = NULL;
603
0
        sp->From14 = NULL;
604
0
        sp->From8 = NULL;
605
0
        sp->ToLinearF = NULL;
606
0
        sp->ToLinear16 = NULL;
607
0
        sp->ToLinear8 = NULL;
608
0
        return 0;
609
0
    }
610
611
0
    j = 0;
612
613
0
    for (i = 0; i < nlin; i++)
614
0
    {
615
0
        v = i * linstep;
616
0
        ToLinearF[j++] = (float)v;
617
0
    }
618
619
0
    for (i = nlin; i < TSIZE; i++)
620
0
        ToLinearF[j++] = (float)(b * exp(c * i));
621
622
0
    ToLinearF[2048] = ToLinearF[2047];
623
624
0
    for (i = 0; i < TSIZEP1; i++)
625
0
    {
626
0
        v = ToLinearF[i] * 65535.0 + 0.5;
627
0
        ToLinear16[i] = (v > 65535.0) ? 65535 : (uint16_t)v;
628
0
        v = ToLinearF[i] * 255.0 + 0.5;
629
0
        ToLinear8[i] = (v > 255.0) ? 255 : (unsigned char)v;
630
0
    }
631
632
0
    j = 0;
633
0
    for (i = 0; i < lt2size; i++)
634
0
    {
635
0
        if ((i * linstep) * (i * linstep) > ToLinearF[j] * ToLinearF[j + 1])
636
0
            j++;
637
0
        FromLT2[i] = (uint16_t)j;
638
0
    }
639
640
    /*
641
     * Since we lose info anyway on 16-bit data, we set up a 14-bit
642
     * table and shift 16-bit values down two bits on input.
643
     * saves a little table space.
644
     */
645
0
    j = 0;
646
0
    for (i = 0; i < 16384; i++)
647
0
    {
648
0
        while ((i / 16383.) * (i / 16383.) > ToLinearF[j] * ToLinearF[j + 1])
649
0
            j++;
650
0
        From14[i] = (uint16_t)j;
651
0
    }
652
653
0
    j = 0;
654
0
    for (i = 0; i < 256; i++)
655
0
    {
656
0
        while ((i / 255.) * (i / 255.) > ToLinearF[j] * ToLinearF[j + 1])
657
0
            j++;
658
0
        From8[i] = (uint16_t)j;
659
0
    }
660
661
0
    Fltsize = (float)(lt2size / 2);
662
663
0
    sp->ToLinearF = ToLinearF;
664
0
    sp->ToLinear16 = ToLinear16;
665
0
    sp->ToLinear8 = ToLinear8;
666
0
    sp->FromLT2 = FromLT2;
667
0
    sp->From14 = From14;
668
0
    sp->From8 = From8;
669
670
0
    return 1;
671
0
}
672
673
0
#define DecoderState(tif) ((PixarLogState *)(tif)->tif_data)
674
0
#define EncoderState(tif) ((PixarLogState *)(tif)->tif_data)
675
676
static int PixarLogEncode(TIFF *tif, uint8_t *bp, tmsize_t cc, uint16_t s);
677
static int PixarLogDecode(TIFF *tif, uint8_t *op, tmsize_t occ, uint16_t s);
678
679
0
#define PIXARLOGDATAFMT_UNKNOWN -1
680
681
static int PixarLogGuessDataFmt(TIFFDirectory *td)
682
0
{
683
0
    int guess = PIXARLOGDATAFMT_UNKNOWN;
684
0
    int format = td->td_sampleformat;
685
686
    /* If the user didn't tell us his datafmt,
687
     * take our best guess from the bitspersample.
688
     */
689
0
    switch (td->td_bitspersample)
690
0
    {
691
0
        case 32:
692
0
            if (format == SAMPLEFORMAT_IEEEFP)
693
0
                guess = PIXARLOGDATAFMT_FLOAT;
694
0
            break;
695
0
        case 16:
696
0
            if (format == SAMPLEFORMAT_VOID || format == SAMPLEFORMAT_UINT)
697
0
                guess = PIXARLOGDATAFMT_16BIT;
698
0
            break;
699
0
        case 12:
700
0
            if (format == SAMPLEFORMAT_VOID || format == SAMPLEFORMAT_INT)
701
0
                guess = PIXARLOGDATAFMT_12BITPICIO;
702
0
            break;
703
0
        case 11:
704
0
            if (format == SAMPLEFORMAT_VOID || format == SAMPLEFORMAT_UINT)
705
0
                guess = PIXARLOGDATAFMT_11BITLOG;
706
0
            break;
707
0
        case 8:
708
0
            if (format == SAMPLEFORMAT_VOID || format == SAMPLEFORMAT_UINT)
709
0
                guess = PIXARLOGDATAFMT_8BIT;
710
0
            break;
711
0
    }
712
713
0
    return guess;
714
0
}
715
716
static tmsize_t multiply_ms(tmsize_t m1, tmsize_t m2)
717
0
{
718
0
    return _TIFFMultiplySSize(NULL, m1, m2, NULL);
719
0
}
720
721
static tmsize_t add_ms(tmsize_t m1, tmsize_t m2)
722
0
{
723
0
    assert(m1 >= 0 && m2 >= 0);
724
    /* if either input is zero, assume overflow already occurred */
725
0
    if (m1 == 0 || m2 == 0)
726
0
        return 0;
727
0
    else if (m1 > TIFF_TMSIZE_T_MAX - m2)
728
0
        return 0;
729
730
0
    return m1 + m2;
731
0
}
732
733
static int PixarLogFixupTags(TIFF *tif)
734
0
{
735
0
    (void)tif;
736
0
    return (1);
737
0
}
738
739
static int PixarLogSetupDecode(TIFF *tif)
740
0
{
741
0
    static const char module[] = "PixarLogSetupDecode";
742
0
    TIFFDirectory *td = &tif->tif_dir;
743
0
    PixarLogState *sp = DecoderState(tif);
744
0
    tmsize_t tbuf_size;
745
0
    uint32_t strip_height;
746
747
0
    assert(sp != NULL);
748
749
    /* This function can possibly be called several times by */
750
    /* PredictorSetupDecode() if this function succeeds but */
751
    /* PredictorSetup() fails */
752
0
    if ((sp->state & PLSTATE_INIT) != 0)
753
0
        return 1;
754
755
0
    strip_height = td->td_rowsperstrip;
756
0
    if (strip_height > td->td_imagelength)
757
0
        strip_height = td->td_imagelength;
758
759
    /* Make sure no byte swapping happens on the data
760
     * after decompression. */
761
0
    tif->tif_postdecode = _TIFFNoPostDecode;
762
763
    /* for some reason, we can't do this in TIFFInitPixarLog */
764
765
0
    sp->stride =
766
0
        (td->td_planarconfig == PLANARCONFIG_CONTIG ? td->td_samplesperpixel
767
0
                                                    : 1);
768
0
    tbuf_size = multiply_ms(
769
0
        multiply_ms(multiply_ms(sp->stride, td->td_imagewidth), strip_height),
770
0
        sizeof(uint16_t));
771
    /* add one more stride in case input ends mid-stride */
772
0
    tbuf_size = add_ms(tbuf_size, sizeof(uint16_t) * sp->stride);
773
0
    if (tbuf_size == 0)
774
0
        return (0); /* TODO: this is an error return without error report
775
                       through TIFFErrorExt */
776
0
    sp->tbuf = (uint16_t *)_TIFFmallocExt(tif, tbuf_size);
777
0
    if (sp->tbuf == NULL)
778
0
        return (0);
779
0
    sp->tbuf_size = tbuf_size;
780
0
    if (sp->user_datafmt == PIXARLOGDATAFMT_UNKNOWN)
781
0
        sp->user_datafmt = PixarLogGuessDataFmt(td);
782
0
    if (sp->user_datafmt == PIXARLOGDATAFMT_UNKNOWN)
783
0
    {
784
0
        _TIFFfreeExt(tif, sp->tbuf);
785
0
        sp->tbuf = NULL;
786
0
        sp->tbuf_size = 0;
787
0
        TIFFErrorExtR(tif, module,
788
0
                      "PixarLog compression can't handle bits depth/data "
789
0
                      "format combination (depth: %" PRIu16 ")",
790
0
                      td->td_bitspersample);
791
0
        return (0);
792
0
    }
793
794
0
    if (inflateInit(&sp->stream) != Z_OK)
795
0
    {
796
0
        _TIFFfreeExt(tif, sp->tbuf);
797
0
        sp->tbuf = NULL;
798
0
        sp->tbuf_size = 0;
799
0
        TIFFErrorExtR(tif, module, "%s",
800
0
                      sp->stream.msg ? sp->stream.msg : "(null)");
801
0
        return (0);
802
0
    }
803
0
    else
804
0
    {
805
0
        sp->state |= PLSTATE_INIT;
806
0
        return (1);
807
0
    }
808
0
}
809
810
/*
811
 * Setup state for decoding a strip.
812
 */
813
static int PixarLogPreDecode(TIFF *tif, uint16_t s)
814
0
{
815
0
    static const char module[] = "PixarLogPreDecode";
816
0
    PixarLogState *sp = DecoderState(tif);
817
818
0
    (void)s;
819
0
    assert(sp != NULL);
820
0
    sp->stream.next_in = tif->tif_rawdata;
821
0
    assert(sizeof(sp->stream.avail_in) == 4); /* if this assert gets raised,
822
         we need to simplify this code to reflect a ZLib that is likely updated
823
         to deal with 8byte memory sizes, though this code will respond
824
         appropriately even before we simplify it */
825
0
    sp->stream.avail_in = (uInt)tif->tif_rawcc;
826
0
    if ((tmsize_t)sp->stream.avail_in != tif->tif_rawcc)
827
0
    {
828
0
        TIFFErrorExtR(tif, module, "ZLib cannot deal with buffers this size");
829
0
        return (0);
830
0
    }
831
0
    return (inflateReset(&sp->stream) == Z_OK);
832
0
}
833
834
static int PixarLogDecode(TIFF *tif, uint8_t *op, tmsize_t occ, uint16_t s)
835
0
{
836
0
    static const char module[] = "PixarLogDecode";
837
0
    TIFFDirectory *td = &tif->tif_dir;
838
0
    PixarLogState *sp = DecoderState(tif);
839
0
    tmsize_t i;
840
0
    tmsize_t nsamples;
841
0
    int llen;
842
0
    uint16_t *up;
843
844
0
    switch (sp->user_datafmt)
845
0
    {
846
0
        case PIXARLOGDATAFMT_FLOAT:
847
0
            nsamples = occ / sizeof(float); /* XXX float == 32 bits */
848
0
            break;
849
0
        case PIXARLOGDATAFMT_16BIT:
850
0
        case PIXARLOGDATAFMT_12BITPICIO:
851
0
        case PIXARLOGDATAFMT_11BITLOG:
852
0
            nsamples = occ / sizeof(uint16_t); /* XXX uint16_t == 16 bits */
853
0
            break;
854
0
        case PIXARLOGDATAFMT_8BIT:
855
0
        case PIXARLOGDATAFMT_8BITABGR:
856
0
            nsamples = occ;
857
0
            break;
858
0
        default:
859
0
            TIFFErrorExtR(tif, module,
860
0
                          "%" PRIu16 " bit input not supported in PixarLog",
861
0
                          td->td_bitspersample);
862
0
            return 0;
863
0
    }
864
865
0
    llen = sp->stride * td->td_imagewidth;
866
867
0
    (void)s;
868
0
    assert(sp != NULL);
869
870
0
    sp->stream.next_in = tif->tif_rawcp;
871
0
    sp->stream.avail_in = (uInt)tif->tif_rawcc;
872
873
0
    sp->stream.next_out = (unsigned char *)sp->tbuf;
874
0
    assert(sizeof(sp->stream.avail_out) == 4); /* if this assert gets raised,
875
         we need to simplify this code to reflect a ZLib that is likely updated
876
         to deal with 8byte memory sizes, though this code will respond
877
         appropriately even before we simplify it */
878
0
    sp->stream.avail_out = (uInt)(nsamples * sizeof(uint16_t));
879
0
    if (sp->stream.avail_out != nsamples * sizeof(uint16_t))
880
0
    {
881
0
        TIFFErrorExtR(tif, module, "ZLib cannot deal with buffers this size");
882
0
        return (0);
883
0
    }
884
    /* Check that we will not fill more than what was allocated */
885
0
    if ((tmsize_t)sp->stream.avail_out > sp->tbuf_size)
886
0
    {
887
0
        TIFFErrorExtR(tif, module, "sp->stream.avail_out > sp->tbuf_size");
888
0
        return (0);
889
0
    }
890
0
    do
891
0
    {
892
0
        int state = inflate(&sp->stream, Z_PARTIAL_FLUSH);
893
0
        if (state == Z_STREAM_END)
894
0
        {
895
0
            break; /* XXX */
896
0
        }
897
0
        if (state == Z_DATA_ERROR)
898
0
        {
899
0
            TIFFErrorExtR(
900
0
                tif, module, "Decoding error at scanline %" PRIu32 ", %s",
901
0
                tif->tif_row, sp->stream.msg ? sp->stream.msg : "(null)");
902
0
            return (0);
903
0
        }
904
0
        if (state != Z_OK)
905
0
        {
906
0
            TIFFErrorExtR(tif, module, "ZLib error: %s",
907
0
                          sp->stream.msg ? sp->stream.msg : "(null)");
908
0
            return (0);
909
0
        }
910
0
    } while (sp->stream.avail_out > 0);
911
912
    /* hopefully, we got all the bytes we needed */
913
0
    if (sp->stream.avail_out != 0)
914
0
    {
915
0
        TIFFErrorExtR(tif, module,
916
0
                      "Not enough data at scanline %" PRIu32
917
0
                      " (short %u bytes)",
918
0
                      tif->tif_row, sp->stream.avail_out);
919
0
        return (0);
920
0
    }
921
922
0
    tif->tif_rawcp = sp->stream.next_in;
923
0
    tif->tif_rawcc = sp->stream.avail_in;
924
925
0
    up = sp->tbuf;
926
    /* Swap bytes in the data if from a different endian machine. */
927
0
    if (tif->tif_flags & TIFF_SWAB)
928
0
        TIFFSwabArrayOfShort(up, nsamples);
929
930
    /*
931
     * if llen is not an exact multiple of nsamples, the decode operation
932
     * may overflow the output buffer, so truncate it enough to prevent
933
     * that but still salvage as much data as possible.
934
     */
935
0
    if (nsamples % llen)
936
0
    {
937
0
        TIFFWarningExtR(tif, module,
938
0
                        "stride %d is not a multiple of sample count, "
939
0
                        "%" TIFF_SSIZE_FORMAT ", data truncated.",
940
0
                        llen, nsamples);
941
0
        nsamples -= nsamples % llen;
942
0
    }
943
944
0
    for (i = 0; i < nsamples; i += llen, up += llen)
945
0
    {
946
0
        switch (sp->user_datafmt)
947
0
        {
948
0
            case PIXARLOGDATAFMT_FLOAT:
949
0
                horizontalAccumulateF(up, llen, sp->stride, (float *)op,
950
0
                                      sp->ToLinearF);
951
0
                op += llen * sizeof(float);
952
0
                break;
953
0
            case PIXARLOGDATAFMT_16BIT:
954
0
                horizontalAccumulate16(up, llen, sp->stride, (uint16_t *)op,
955
0
                                       sp->ToLinear16);
956
0
                op += llen * sizeof(uint16_t);
957
0
                break;
958
0
            case PIXARLOGDATAFMT_12BITPICIO:
959
0
                horizontalAccumulate12(up, llen, sp->stride, (int16_t *)op,
960
0
                                       sp->ToLinearF);
961
0
                op += llen * sizeof(int16_t);
962
0
                break;
963
0
            case PIXARLOGDATAFMT_11BITLOG:
964
0
                horizontalAccumulate11(up, llen, sp->stride, (uint16_t *)op);
965
0
                op += llen * sizeof(uint16_t);
966
0
                break;
967
0
            case PIXARLOGDATAFMT_8BIT:
968
0
                horizontalAccumulate8(up, llen, sp->stride, (unsigned char *)op,
969
0
                                      sp->ToLinear8);
970
0
                op += llen * sizeof(unsigned char);
971
0
                break;
972
0
            case PIXARLOGDATAFMT_8BITABGR:
973
0
                horizontalAccumulate8abgr(up, llen, sp->stride,
974
0
                                          (unsigned char *)op, sp->ToLinear8);
975
0
                op += llen * sizeof(unsigned char);
976
0
                break;
977
0
            default:
978
0
                TIFFErrorExtR(tif, module, "Unsupported bits/sample: %" PRIu16,
979
0
                              td->td_bitspersample);
980
0
                return (0);
981
0
        }
982
0
    }
983
984
0
    return (1);
985
0
}
986
987
static int PixarLogSetupEncode(TIFF *tif)
988
0
{
989
0
    static const char module[] = "PixarLogSetupEncode";
990
0
    TIFFDirectory *td = &tif->tif_dir;
991
0
    PixarLogState *sp = EncoderState(tif);
992
0
    tmsize_t tbuf_size;
993
994
0
    assert(sp != NULL);
995
996
    /* for some reason, we can't do this in TIFFInitPixarLog */
997
998
0
    sp->stride =
999
0
        (td->td_planarconfig == PLANARCONFIG_CONTIG ? td->td_samplesperpixel
1000
0
                                                    : 1);
1001
0
    tbuf_size =
1002
0
        multiply_ms(multiply_ms(multiply_ms(sp->stride, td->td_imagewidth),
1003
0
                                td->td_rowsperstrip),
1004
0
                    sizeof(uint16_t));
1005
0
    if (tbuf_size == 0)
1006
0
        return (0); /* TODO: this is an error return without error report
1007
                       through TIFFErrorExt */
1008
0
    sp->tbuf = (uint16_t *)_TIFFmallocExt(tif, tbuf_size);
1009
0
    if (sp->tbuf == NULL)
1010
0
        return (0);
1011
0
    if (sp->user_datafmt == PIXARLOGDATAFMT_UNKNOWN)
1012
0
        sp->user_datafmt = PixarLogGuessDataFmt(td);
1013
0
    if (sp->user_datafmt == PIXARLOGDATAFMT_UNKNOWN)
1014
0
    {
1015
0
        TIFFErrorExtR(tif, module,
1016
0
                      "PixarLog compression can't handle %" PRIu16
1017
0
                      " bit linear encodings",
1018
0
                      td->td_bitspersample);
1019
0
        return (0);
1020
0
    }
1021
1022
0
    if (deflateInit(&sp->stream, sp->quality) != Z_OK)
1023
0
    {
1024
0
        TIFFErrorExtR(tif, module, "%s",
1025
0
                      sp->stream.msg ? sp->stream.msg : "(null)");
1026
0
        return (0);
1027
0
    }
1028
0
    else
1029
0
    {
1030
0
        sp->state |= PLSTATE_INIT;
1031
0
        return (1);
1032
0
    }
1033
0
}
1034
1035
/*
1036
 * Reset encoding state at the start of a strip.
1037
 */
1038
static int PixarLogPreEncode(TIFF *tif, uint16_t s)
1039
0
{
1040
0
    static const char module[] = "PixarLogPreEncode";
1041
0
    PixarLogState *sp = EncoderState(tif);
1042
1043
0
    (void)s;
1044
0
    assert(sp != NULL);
1045
0
    sp->stream.next_out = tif->tif_rawdata;
1046
0
    assert(sizeof(sp->stream.avail_out) == 4); /* if this assert gets raised,
1047
         we need to simplify this code to reflect a ZLib that is likely updated
1048
         to deal with 8byte memory sizes, though this code will respond
1049
         appropriately even before we simplify it */
1050
0
    sp->stream.avail_out = (uInt)tif->tif_rawdatasize;
1051
0
    if ((tmsize_t)sp->stream.avail_out != tif->tif_rawdatasize)
1052
0
    {
1053
0
        TIFFErrorExtR(tif, module, "ZLib cannot deal with buffers this size");
1054
0
        return (0);
1055
0
    }
1056
0
    return (deflateReset(&sp->stream) == Z_OK);
1057
0
}
1058
1059
static void horizontalDifferenceF(float *ip, int n, int stride, uint16_t *wp,
1060
                                  uint16_t *FromLT2)
1061
0
{
1062
0
    int32_t r1, g1, b1, a1, r2, g2, b2, a2, mask;
1063
0
    float fltsize = Fltsize;
1064
1065
0
#define CLAMP(v)                                                               \
1066
0
    ((v < (float)0.)     ? 0                                                   \
1067
0
     : (v < (float)2.)   ? FromLT2[(int)(v * fltsize)]                         \
1068
0
     : (v > (float)24.2) ? 2047                                                \
1069
0
                         : LogK1 * log(v * LogK2) + 0.5)
1070
1071
0
    mask = CODE_MASK;
1072
0
    if (n >= stride)
1073
0
    {
1074
0
        if (stride == 3)
1075
0
        {
1076
0
            r2 = wp[0] = (uint16_t)CLAMP(ip[0]);
1077
0
            g2 = wp[1] = (uint16_t)CLAMP(ip[1]);
1078
0
            b2 = wp[2] = (uint16_t)CLAMP(ip[2]);
1079
0
            n -= 3;
1080
0
            while (n > 0)
1081
0
            {
1082
0
                n -= 3;
1083
0
                wp += 3;
1084
0
                ip += 3;
1085
0
                r1 = (int32_t)CLAMP(ip[0]);
1086
0
                wp[0] = (uint16_t)((r1 - r2) & mask);
1087
0
                r2 = r1;
1088
0
                g1 = (int32_t)CLAMP(ip[1]);
1089
0
                wp[1] = (uint16_t)((g1 - g2) & mask);
1090
0
                g2 = g1;
1091
0
                b1 = (int32_t)CLAMP(ip[2]);
1092
0
                wp[2] = (uint16_t)((b1 - b2) & mask);
1093
0
                b2 = b1;
1094
0
            }
1095
0
        }
1096
0
        else if (stride == 4)
1097
0
        {
1098
0
            r2 = wp[0] = (uint16_t)CLAMP(ip[0]);
1099
0
            g2 = wp[1] = (uint16_t)CLAMP(ip[1]);
1100
0
            b2 = wp[2] = (uint16_t)CLAMP(ip[2]);
1101
0
            a2 = wp[3] = (uint16_t)CLAMP(ip[3]);
1102
0
            n -= 4;
1103
0
            while (n > 0)
1104
0
            {
1105
0
                n -= 4;
1106
0
                wp += 4;
1107
0
                ip += 4;
1108
0
                r1 = (int32_t)CLAMP(ip[0]);
1109
0
                wp[0] = (uint16_t)((r1 - r2) & mask);
1110
0
                r2 = r1;
1111
0
                g1 = (int32_t)CLAMP(ip[1]);
1112
0
                wp[1] = (uint16_t)((g1 - g2) & mask);
1113
0
                g2 = g1;
1114
0
                b1 = (int32_t)CLAMP(ip[2]);
1115
0
                wp[2] = (uint16_t)((b1 - b2) & mask);
1116
0
                b2 = b1;
1117
0
                a1 = (int32_t)CLAMP(ip[3]);
1118
0
                wp[3] = (uint16_t)((a1 - a2) & mask);
1119
0
                a2 = a1;
1120
0
            }
1121
0
        }
1122
0
        else
1123
0
        {
1124
0
            REPEAT(stride, wp[0] = (uint16_t)CLAMP(ip[0]); wp++; ip++)
1125
0
            n -= stride;
1126
0
            while (n > 0)
1127
0
            {
1128
0
                REPEAT(stride,
1129
0
                       wp[0] = (uint16_t)(((int32_t)CLAMP(ip[0]) -
1130
0
                                           (int32_t)CLAMP(ip[-stride])) &
1131
0
                                          mask);
1132
0
                       wp++; ip++)
1133
0
                n -= stride;
1134
0
            }
1135
0
        }
1136
0
    }
1137
0
}
1138
1139
static void horizontalDifference16(unsigned short *ip, int n, int stride,
1140
                                   unsigned short *wp, uint16_t *From14)
1141
0
{
1142
0
    register int r1, g1, b1, a1, r2, g2, b2, a2, mask;
1143
1144
/* assumption is unsigned pixel values */
1145
0
#undef CLAMP
1146
0
#define CLAMP(v) From14[(v) >> 2]
1147
1148
0
    mask = CODE_MASK;
1149
0
    if (n >= stride)
1150
0
    {
1151
0
        if (stride == 3)
1152
0
        {
1153
0
            r2 = wp[0] = CLAMP(ip[0]);
1154
0
            g2 = wp[1] = CLAMP(ip[1]);
1155
0
            b2 = wp[2] = CLAMP(ip[2]);
1156
0
            n -= 3;
1157
0
            while (n > 0)
1158
0
            {
1159
0
                n -= 3;
1160
0
                wp += 3;
1161
0
                ip += 3;
1162
0
                r1 = CLAMP(ip[0]);
1163
0
                wp[0] = (uint16_t)((r1 - r2) & mask);
1164
0
                r2 = r1;
1165
0
                g1 = CLAMP(ip[1]);
1166
0
                wp[1] = (uint16_t)((g1 - g2) & mask);
1167
0
                g2 = g1;
1168
0
                b1 = CLAMP(ip[2]);
1169
0
                wp[2] = (uint16_t)((b1 - b2) & mask);
1170
0
                b2 = b1;
1171
0
            }
1172
0
        }
1173
0
        else if (stride == 4)
1174
0
        {
1175
0
            r2 = wp[0] = CLAMP(ip[0]);
1176
0
            g2 = wp[1] = CLAMP(ip[1]);
1177
0
            b2 = wp[2] = CLAMP(ip[2]);
1178
0
            a2 = wp[3] = CLAMP(ip[3]);
1179
0
            n -= 4;
1180
0
            while (n > 0)
1181
0
            {
1182
0
                n -= 4;
1183
0
                wp += 4;
1184
0
                ip += 4;
1185
0
                r1 = CLAMP(ip[0]);
1186
0
                wp[0] = (uint16_t)((r1 - r2) & mask);
1187
0
                r2 = r1;
1188
0
                g1 = CLAMP(ip[1]);
1189
0
                wp[1] = (uint16_t)((g1 - g2) & mask);
1190
0
                g2 = g1;
1191
0
                b1 = CLAMP(ip[2]);
1192
0
                wp[2] = (uint16_t)((b1 - b2) & mask);
1193
0
                b2 = b1;
1194
0
                a1 = CLAMP(ip[3]);
1195
0
                wp[3] = (uint16_t)((a1 - a2) & mask);
1196
0
                a2 = a1;
1197
0
            }
1198
0
        }
1199
0
        else
1200
0
        {
1201
0
            REPEAT(stride, wp[0] = CLAMP(ip[0]); wp++; ip++)
1202
0
            n -= stride;
1203
0
            while (n > 0)
1204
0
            {
1205
0
                REPEAT(stride,
1206
0
                       wp[0] = (uint16_t)((CLAMP(ip[0]) - CLAMP(ip[-stride])) &
1207
0
                                          mask);
1208
0
                       wp++; ip++)
1209
0
                n -= stride;
1210
0
            }
1211
0
        }
1212
0
    }
1213
0
}
1214
1215
static void horizontalDifference8(unsigned char *ip, int n, int stride,
1216
                                  unsigned short *wp, uint16_t *From8)
1217
0
{
1218
0
    register int r1, g1, b1, a1, r2, g2, b2, a2, mask;
1219
1220
0
#undef CLAMP
1221
0
#define CLAMP(v) (From8[(v)])
1222
1223
0
    mask = CODE_MASK;
1224
0
    if (n >= stride)
1225
0
    {
1226
0
        if (stride == 3)
1227
0
        {
1228
0
            r2 = wp[0] = CLAMP(ip[0]);
1229
0
            g2 = wp[1] = CLAMP(ip[1]);
1230
0
            b2 = wp[2] = CLAMP(ip[2]);
1231
0
            n -= 3;
1232
0
            while (n > 0)
1233
0
            {
1234
0
                n -= 3;
1235
0
                r1 = CLAMP(ip[3]);
1236
0
                wp[3] = (uint16_t)((r1 - r2) & mask);
1237
0
                r2 = r1;
1238
0
                g1 = CLAMP(ip[4]);
1239
0
                wp[4] = (uint16_t)((g1 - g2) & mask);
1240
0
                g2 = g1;
1241
0
                b1 = CLAMP(ip[5]);
1242
0
                wp[5] = (uint16_t)((b1 - b2) & mask);
1243
0
                b2 = b1;
1244
0
                wp += 3;
1245
0
                ip += 3;
1246
0
            }
1247
0
        }
1248
0
        else if (stride == 4)
1249
0
        {
1250
0
            r2 = wp[0] = CLAMP(ip[0]);
1251
0
            g2 = wp[1] = CLAMP(ip[1]);
1252
0
            b2 = wp[2] = CLAMP(ip[2]);
1253
0
            a2 = wp[3] = CLAMP(ip[3]);
1254
0
            n -= 4;
1255
0
            while (n > 0)
1256
0
            {
1257
0
                n -= 4;
1258
0
                r1 = CLAMP(ip[4]);
1259
0
                wp[4] = (uint16_t)((r1 - r2) & mask);
1260
0
                r2 = r1;
1261
0
                g1 = CLAMP(ip[5]);
1262
0
                wp[5] = (uint16_t)((g1 - g2) & mask);
1263
0
                g2 = g1;
1264
0
                b1 = CLAMP(ip[6]);
1265
0
                wp[6] = (uint16_t)((b1 - b2) & mask);
1266
0
                b2 = b1;
1267
0
                a1 = CLAMP(ip[7]);
1268
0
                wp[7] = (uint16_t)((a1 - a2) & mask);
1269
0
                a2 = a1;
1270
0
                wp += 4;
1271
0
                ip += 4;
1272
0
            }
1273
0
        }
1274
0
        else
1275
0
        {
1276
0
            REPEAT(stride, wp[0] = CLAMP(ip[0]); wp++; ip++)
1277
0
            n -= stride;
1278
0
            while (n > 0)
1279
0
            {
1280
0
                REPEAT(stride,
1281
0
                       wp[0] = (uint16_t)((CLAMP(ip[0]) - CLAMP(ip[-stride])) &
1282
0
                                          mask);
1283
0
                       wp++; ip++)
1284
0
                n -= stride;
1285
0
            }
1286
0
        }
1287
0
    }
1288
0
}
1289
1290
/*
1291
 * Encode a chunk of pixels.
1292
 */
1293
static int PixarLogEncode(TIFF *tif, uint8_t *bp, tmsize_t cc, uint16_t s)
1294
0
{
1295
0
    static const char module[] = "PixarLogEncode";
1296
0
    TIFFDirectory *td = &tif->tif_dir;
1297
0
    PixarLogState *sp = EncoderState(tif);
1298
0
    tmsize_t i;
1299
0
    tmsize_t n;
1300
0
    int llen;
1301
0
    unsigned short *up;
1302
1303
0
    (void)s;
1304
1305
0
    switch (sp->user_datafmt)
1306
0
    {
1307
0
        case PIXARLOGDATAFMT_FLOAT:
1308
0
            n = cc / sizeof(float); /* XXX float == 32 bits */
1309
0
            break;
1310
0
        case PIXARLOGDATAFMT_16BIT:
1311
0
        case PIXARLOGDATAFMT_12BITPICIO:
1312
0
        case PIXARLOGDATAFMT_11BITLOG:
1313
0
            n = cc / sizeof(uint16_t); /* XXX uint16_t == 16 bits */
1314
0
            break;
1315
0
        case PIXARLOGDATAFMT_8BIT:
1316
0
        case PIXARLOGDATAFMT_8BITABGR:
1317
0
            n = cc;
1318
0
            break;
1319
0
        default:
1320
0
            TIFFErrorExtR(tif, module,
1321
0
                          "%" PRIu16 " bit input not supported in PixarLog",
1322
0
                          td->td_bitspersample);
1323
0
            return 0;
1324
0
    }
1325
1326
0
    llen = sp->stride * td->td_imagewidth;
1327
    /* Check against the number of elements (of size uint16_t) of sp->tbuf */
1328
0
    if (n > ((tmsize_t)td->td_rowsperstrip * llen))
1329
0
    {
1330
0
        TIFFErrorExtR(tif, module, "Too many input bytes provided");
1331
0
        return 0;
1332
0
    }
1333
1334
0
    for (i = 0, up = sp->tbuf; i < n; i += llen, up += llen)
1335
0
    {
1336
0
        switch (sp->user_datafmt)
1337
0
        {
1338
0
            case PIXARLOGDATAFMT_FLOAT:
1339
0
                horizontalDifferenceF((float *)bp, llen, sp->stride, up,
1340
0
                                      sp->FromLT2);
1341
0
                bp += llen * sizeof(float);
1342
0
                break;
1343
0
            case PIXARLOGDATAFMT_16BIT:
1344
0
                horizontalDifference16((uint16_t *)bp, llen, sp->stride, up,
1345
0
                                       sp->From14);
1346
0
                bp += llen * sizeof(uint16_t);
1347
0
                break;
1348
0
            case PIXARLOGDATAFMT_8BIT:
1349
0
                horizontalDifference8((unsigned char *)bp, llen, sp->stride, up,
1350
0
                                      sp->From8);
1351
0
                bp += llen * sizeof(unsigned char);
1352
0
                break;
1353
0
            default:
1354
0
                TIFFErrorExtR(tif, module,
1355
0
                              "%" PRIu16 " bit input not supported in PixarLog",
1356
0
                              td->td_bitspersample);
1357
0
                return 0;
1358
0
        }
1359
0
    }
1360
1361
0
    sp->stream.next_in = (unsigned char *)sp->tbuf;
1362
0
    assert(sizeof(sp->stream.avail_in) == 4); /* if this assert gets raised,
1363
         we need to simplify this code to reflect a ZLib that is likely updated
1364
         to deal with 8byte memory sizes, though this code will respond
1365
         appropriately even before we simplify it */
1366
0
    sp->stream.avail_in = (uInt)(n * sizeof(uint16_t));
1367
0
    if ((sp->stream.avail_in / sizeof(uint16_t)) != (uInt)n)
1368
0
    {
1369
0
        TIFFErrorExtR(tif, module, "ZLib cannot deal with buffers this size");
1370
0
        return (0);
1371
0
    }
1372
1373
0
    do
1374
0
    {
1375
0
        if (deflate(&sp->stream, Z_NO_FLUSH) != Z_OK)
1376
0
        {
1377
0
            TIFFErrorExtR(tif, module, "Encoder error: %s",
1378
0
                          sp->stream.msg ? sp->stream.msg : "(null)");
1379
0
            return (0);
1380
0
        }
1381
0
        if (sp->stream.avail_out == 0)
1382
0
        {
1383
0
            tif->tif_rawcc = tif->tif_rawdatasize;
1384
0
            if (!TIFFFlushData1(tif))
1385
0
                return 0;
1386
0
            sp->stream.next_out = tif->tif_rawdata;
1387
0
            sp->stream.avail_out =
1388
0
                (uInt)tif
1389
0
                    ->tif_rawdatasize; /* this is a safe typecast, as check is
1390
                                          made already in PixarLogPreEncode */
1391
0
        }
1392
0
    } while (sp->stream.avail_in > 0);
1393
0
    return (1);
1394
0
}
1395
1396
/*
1397
 * Finish off an encoded strip by flushing the last
1398
 * string and tacking on an End Of Information code.
1399
 */
1400
1401
static int PixarLogPostEncode(TIFF *tif)
1402
0
{
1403
0
    static const char module[] = "PixarLogPostEncode";
1404
0
    PixarLogState *sp = EncoderState(tif);
1405
0
    int state;
1406
1407
0
    sp->stream.avail_in = 0;
1408
1409
0
    do
1410
0
    {
1411
0
        state = deflate(&sp->stream, Z_FINISH);
1412
0
        switch (state)
1413
0
        {
1414
0
            case Z_STREAM_END:
1415
0
            case Z_OK:
1416
0
                if ((tmsize_t)sp->stream.avail_out != tif->tif_rawdatasize)
1417
0
                {
1418
0
                    tif->tif_rawcc =
1419
0
                        tif->tif_rawdatasize - sp->stream.avail_out;
1420
0
                    if (!TIFFFlushData1(tif))
1421
0
                        return 0;
1422
0
                    sp->stream.next_out = tif->tif_rawdata;
1423
0
                    sp->stream.avail_out =
1424
0
                        (uInt)tif->tif_rawdatasize; /* this is a safe typecast,
1425
                                                       as check is made already
1426
                                                       in PixarLogPreEncode */
1427
0
                }
1428
0
                break;
1429
0
            default:
1430
0
                TIFFErrorExtR(tif, module, "ZLib error: %s",
1431
0
                              sp->stream.msg ? sp->stream.msg : "(null)");
1432
0
                return (0);
1433
0
        }
1434
0
    } while (state != Z_STREAM_END);
1435
0
    return (1);
1436
0
}
1437
1438
static void PixarLogClose(TIFF *tif)
1439
0
{
1440
0
    PixarLogState *sp = (PixarLogState *)tif->tif_data;
1441
0
    TIFFDirectory *td = &tif->tif_dir;
1442
1443
0
    assert(sp != 0);
1444
    /* In a really sneaky (and really incorrect, and untruthful, and
1445
     * troublesome, and error-prone) maneuver that completely goes against
1446
     * the spirit of TIFF, and breaks TIFF, on close, we covertly
1447
     * modify both bitspersample and sampleformat in the directory to
1448
     * indicate 8-bit linear.  This way, the decode "just works" even for
1449
     * readers that don't know about PixarLog, or how to set
1450
     * the PIXARLOGDATFMT pseudo-tag.
1451
     */
1452
1453
0
    if (sp->state & PLSTATE_INIT)
1454
0
    {
1455
        /* We test the state to avoid an issue such as in
1456
         * http://bugzilla.maptools.org/show_bug.cgi?id=2604
1457
         * What appends in that case is that the bitspersample is 1 and
1458
         * a TransferFunction is set. The size of the TransferFunction
1459
         * depends on 1<<bitspersample. So if we increase it, an access
1460
         * out of the buffer will happen at directory flushing.
1461
         * Another option would be to clear those targs.
1462
         */
1463
0
        td->td_bitspersample = 8;
1464
0
        td->td_sampleformat = SAMPLEFORMAT_UINT;
1465
0
    }
1466
0
}
1467
1468
static void PixarLogCleanup(TIFF *tif)
1469
0
{
1470
0
    PixarLogState *sp = (PixarLogState *)tif->tif_data;
1471
1472
0
    assert(sp != 0);
1473
1474
0
    (void)TIFFPredictorCleanup(tif);
1475
1476
0
    tif->tif_tagmethods.vgetfield = sp->vgetparent;
1477
0
    tif->tif_tagmethods.vsetfield = sp->vsetparent;
1478
1479
0
    if (sp->FromLT2)
1480
0
        _TIFFfreeExt(tif, sp->FromLT2);
1481
0
    if (sp->From14)
1482
0
        _TIFFfreeExt(tif, sp->From14);
1483
0
    if (sp->From8)
1484
0
        _TIFFfreeExt(tif, sp->From8);
1485
0
    if (sp->ToLinearF)
1486
0
        _TIFFfreeExt(tif, sp->ToLinearF);
1487
0
    if (sp->ToLinear16)
1488
0
        _TIFFfreeExt(tif, sp->ToLinear16);
1489
0
    if (sp->ToLinear8)
1490
0
        _TIFFfreeExt(tif, sp->ToLinear8);
1491
0
    if (sp->state & PLSTATE_INIT)
1492
0
    {
1493
0
        if (tif->tif_mode == O_RDONLY)
1494
0
            inflateEnd(&sp->stream);
1495
0
        else
1496
0
            deflateEnd(&sp->stream);
1497
0
    }
1498
0
    if (sp->tbuf)
1499
0
        _TIFFfreeExt(tif, sp->tbuf);
1500
0
    _TIFFfreeExt(tif, sp);
1501
0
    tif->tif_data = NULL;
1502
1503
0
    _TIFFSetDefaultCompressionState(tif);
1504
0
}
1505
1506
static int PixarLogVSetField(TIFF *tif, uint32_t tag, va_list ap)
1507
0
{
1508
0
    static const char module[] = "PixarLogVSetField";
1509
0
    PixarLogState *sp = (PixarLogState *)tif->tif_data;
1510
0
    int result;
1511
1512
0
    switch (tag)
1513
0
    {
1514
0
        case TIFFTAG_PIXARLOGQUALITY:
1515
0
            sp->quality = (int)va_arg(ap, int);
1516
0
            if (tif->tif_mode != O_RDONLY && (sp->state & PLSTATE_INIT))
1517
0
            {
1518
0
                if (deflateParams(&sp->stream, sp->quality,
1519
0
                                  Z_DEFAULT_STRATEGY) != Z_OK)
1520
0
                {
1521
0
                    TIFFErrorExtR(tif, module, "ZLib error: %s",
1522
0
                                  sp->stream.msg ? sp->stream.msg : "(null)");
1523
0
                    return (0);
1524
0
                }
1525
0
            }
1526
0
            return (1);
1527
0
        case TIFFTAG_PIXARLOGDATAFMT:
1528
0
            sp->user_datafmt = (int)va_arg(ap, int);
1529
            /* Tweak the TIFF header so that the rest of libtiff knows what
1530
             * size of data will be passed between app and library, and
1531
             * assume that the app knows what it is doing and is not
1532
             * confused by these header manipulations...
1533
             */
1534
0
            switch (sp->user_datafmt)
1535
0
            {
1536
0
                case PIXARLOGDATAFMT_8BIT:
1537
0
                case PIXARLOGDATAFMT_8BITABGR:
1538
0
                    TIFFSetField(tif, TIFFTAG_BITSPERSAMPLE, 8);
1539
0
                    TIFFSetField(tif, TIFFTAG_SAMPLEFORMAT, SAMPLEFORMAT_UINT);
1540
0
                    break;
1541
0
                case PIXARLOGDATAFMT_11BITLOG:
1542
0
                    TIFFSetField(tif, TIFFTAG_BITSPERSAMPLE, 16);
1543
0
                    TIFFSetField(tif, TIFFTAG_SAMPLEFORMAT, SAMPLEFORMAT_UINT);
1544
0
                    break;
1545
0
                case PIXARLOGDATAFMT_12BITPICIO:
1546
0
                    TIFFSetField(tif, TIFFTAG_BITSPERSAMPLE, 16);
1547
0
                    TIFFSetField(tif, TIFFTAG_SAMPLEFORMAT, SAMPLEFORMAT_INT);
1548
0
                    break;
1549
0
                case PIXARLOGDATAFMT_16BIT:
1550
0
                    TIFFSetField(tif, TIFFTAG_BITSPERSAMPLE, 16);
1551
0
                    TIFFSetField(tif, TIFFTAG_SAMPLEFORMAT, SAMPLEFORMAT_UINT);
1552
0
                    break;
1553
0
                case PIXARLOGDATAFMT_FLOAT:
1554
0
                    TIFFSetField(tif, TIFFTAG_BITSPERSAMPLE, 32);
1555
0
                    TIFFSetField(tif, TIFFTAG_SAMPLEFORMAT,
1556
0
                                 SAMPLEFORMAT_IEEEFP);
1557
0
                    break;
1558
0
            }
1559
            /*
1560
             * Must recalculate sizes should bits/sample change.
1561
             */
1562
0
            tif->tif_tilesize =
1563
0
                isTiled(tif) ? TIFFTileSize(tif) : (tmsize_t)(-1);
1564
0
            tif->tif_scanlinesize = TIFFScanlineSize(tif);
1565
0
            result = 1; /* NB: pseudo tag */
1566
0
            break;
1567
0
        default:
1568
0
            result = (*sp->vsetparent)(tif, tag, ap);
1569
0
    }
1570
0
    return (result);
1571
0
}
1572
1573
static int PixarLogVGetField(TIFF *tif, uint32_t tag, va_list ap)
1574
0
{
1575
0
    PixarLogState *sp = (PixarLogState *)tif->tif_data;
1576
1577
0
    switch (tag)
1578
0
    {
1579
0
        case TIFFTAG_PIXARLOGQUALITY:
1580
0
            *va_arg(ap, int *) = sp->quality;
1581
0
            break;
1582
0
        case TIFFTAG_PIXARLOGDATAFMT:
1583
0
            *va_arg(ap, int *) = sp->user_datafmt;
1584
0
            break;
1585
0
        default:
1586
0
            return (*sp->vgetparent)(tif, tag, ap);
1587
0
    }
1588
0
    return (1);
1589
0
}
1590
1591
static const TIFFField pixarlogFields[] = {
1592
    {TIFFTAG_PIXARLOGDATAFMT, 0, 0, TIFF_ANY, 0, TIFF_SETGET_INT,
1593
     TIFF_SETGET_UNDEFINED, FIELD_PSEUDO, FALSE, FALSE, "", NULL},
1594
    {TIFFTAG_PIXARLOGQUALITY, 0, 0, TIFF_ANY, 0, TIFF_SETGET_INT,
1595
     TIFF_SETGET_UNDEFINED, FIELD_PSEUDO, FALSE, FALSE, "", NULL}};
1596
1597
int TIFFInitPixarLog(TIFF *tif, int scheme)
1598
0
{
1599
0
    static const char module[] = "TIFFInitPixarLog";
1600
1601
0
    PixarLogState *sp;
1602
1603
0
    (void)scheme;
1604
0
    assert(scheme == COMPRESSION_PIXARLOG);
1605
1606
    /*
1607
     * Merge codec-specific tag information.
1608
     */
1609
0
    if (!_TIFFMergeFields(tif, pixarlogFields, TIFFArrayCount(pixarlogFields)))
1610
0
    {
1611
0
        TIFFErrorExtR(tif, module,
1612
0
                      "Merging PixarLog codec-specific tags failed");
1613
0
        return 0;
1614
0
    }
1615
1616
    /*
1617
     * Allocate state block so tag methods have storage to record values.
1618
     */
1619
0
    tif->tif_data = (uint8_t *)_TIFFmallocExt(tif, sizeof(PixarLogState));
1620
0
    if (tif->tif_data == NULL)
1621
0
        goto bad;
1622
0
    sp = (PixarLogState *)tif->tif_data;
1623
0
    _TIFFmemset(sp, 0, sizeof(*sp));
1624
0
    sp->stream.data_type = Z_BINARY;
1625
0
    sp->user_datafmt = PIXARLOGDATAFMT_UNKNOWN;
1626
1627
    /*
1628
     * Install codec methods.
1629
     */
1630
0
    tif->tif_fixuptags = PixarLogFixupTags;
1631
0
    tif->tif_setupdecode = PixarLogSetupDecode;
1632
0
    tif->tif_predecode = PixarLogPreDecode;
1633
0
    tif->tif_decoderow = PixarLogDecode;
1634
0
    tif->tif_decodestrip = PixarLogDecode;
1635
0
    tif->tif_decodetile = PixarLogDecode;
1636
0
    tif->tif_setupencode = PixarLogSetupEncode;
1637
0
    tif->tif_preencode = PixarLogPreEncode;
1638
0
    tif->tif_postencode = PixarLogPostEncode;
1639
0
    tif->tif_encoderow = PixarLogEncode;
1640
0
    tif->tif_encodestrip = PixarLogEncode;
1641
0
    tif->tif_encodetile = PixarLogEncode;
1642
0
    tif->tif_close = PixarLogClose;
1643
0
    tif->tif_cleanup = PixarLogCleanup;
1644
1645
    /* Override SetField so we can handle our private pseudo-tag */
1646
0
    sp->vgetparent = tif->tif_tagmethods.vgetfield;
1647
0
    tif->tif_tagmethods.vgetfield = PixarLogVGetField; /* hook for codec tags */
1648
0
    sp->vsetparent = tif->tif_tagmethods.vsetfield;
1649
0
    tif->tif_tagmethods.vsetfield = PixarLogVSetField; /* hook for codec tags */
1650
1651
    /* Default values for codec-specific fields */
1652
0
    sp->quality = Z_DEFAULT_COMPRESSION; /* default comp. level */
1653
0
    sp->state = 0;
1654
1655
    /* we don't wish to use the predictor,
1656
     * the default is none, which predictor value 1
1657
     */
1658
0
    (void)TIFFPredictorInit(tif);
1659
1660
    /*
1661
     * build the companding tables
1662
     */
1663
0
    PixarLogMakeTables(tif, sp);
1664
1665
0
    return (1);
1666
0
bad:
1667
0
    TIFFErrorExtR(tif, module, "No space for PixarLog state block");
1668
0
    return (0);
1669
0
}
1670
#endif /* PIXARLOG_SUPPORT */