Coverage Report

Created: 2025-06-13 06:48

/src/libtiff/libtiff/tif_getimage.c
Line
Count
Source (jump to first uncovered line)
1
/*
2
 * Copyright (c) 1991-1997 Sam Leffler
3
 * Copyright (c) 1991-1997 Silicon Graphics, Inc.
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
 * 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 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 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
/*
26
 * TIFF Library
27
 *
28
 * Read and return a packed RGBA image.
29
 */
30
#include "tiffiop.h"
31
#include <limits.h>
32
#include <stdio.h>
33
34
static int gtTileContig(TIFFRGBAImage *, uint32_t *, uint32_t, uint32_t);
35
static int gtTileSeparate(TIFFRGBAImage *, uint32_t *, uint32_t, uint32_t);
36
static int gtStripContig(TIFFRGBAImage *, uint32_t *, uint32_t, uint32_t);
37
static int gtStripSeparate(TIFFRGBAImage *, uint32_t *, uint32_t, uint32_t);
38
static int PickContigCase(TIFFRGBAImage *);
39
static int PickSeparateCase(TIFFRGBAImage *);
40
41
static int BuildMapUaToAa(TIFFRGBAImage *img);
42
static int BuildMapBitdepth16To8(TIFFRGBAImage *img);
43
44
static const char photoTag[] = "PhotometricInterpretation";
45
46
/*
47
 * Helper constants used in Orientation tag handling
48
 */
49
0
#define FLIP_VERTICALLY 0x01
50
0
#define FLIP_HORIZONTALLY 0x02
51
52
0
#define EMSG_BUF_SIZE 1024
53
54
/*
55
 * Color conversion constants. We will define display types here.
56
 */
57
58
static const TIFFDisplay display_sRGB = {
59
    {/* XYZ -> luminance matrix */
60
     {3.2410F, -1.5374F, -0.4986F},
61
     {-0.9692F, 1.8760F, 0.0416F},
62
     {0.0556F, -0.2040F, 1.0570F}},
63
    100.0F,
64
    100.0F,
65
    100.0F, /* Light o/p for reference white */
66
    255,
67
    255,
68
    255, /* Pixel values for ref. white */
69
    1.0F,
70
    1.0F,
71
    1.0F, /* Residual light o/p for black pixel */
72
    2.4F,
73
    2.4F,
74
    2.4F, /* Gamma values for the three guns */
75
};
76
77
/*
78
 * Check the image to see if TIFFReadRGBAImage can deal with it.
79
 * 1/0 is returned according to whether or not the image can
80
 * be handled.  If 0 is returned, emsg contains the reason
81
 * why it is being rejected.
82
 */
83
int TIFFRGBAImageOK(TIFF *tif, char emsg[EMSG_BUF_SIZE])
84
0
{
85
0
    TIFFDirectory *td = &tif->tif_dir;
86
0
    uint16_t photometric;
87
0
    int colorchannels;
88
89
0
    if (!tif->tif_decodestatus)
90
0
    {
91
0
        snprintf(emsg, EMSG_BUF_SIZE,
92
0
                 "Sorry, requested compression method is not configured");
93
0
        return (0);
94
0
    }
95
0
    switch (td->td_bitspersample)
96
0
    {
97
0
        case 1:
98
0
        case 2:
99
0
        case 4:
100
0
        case 8:
101
0
        case 16:
102
0
            break;
103
0
        default:
104
0
            snprintf(emsg, EMSG_BUF_SIZE,
105
0
                     "Sorry, can not handle images with %" PRIu16
106
0
                     "-bit samples",
107
0
                     td->td_bitspersample);
108
0
            return (0);
109
0
    }
110
0
    if (td->td_sampleformat == SAMPLEFORMAT_IEEEFP)
111
0
    {
112
0
        snprintf(
113
0
            emsg, EMSG_BUF_SIZE,
114
0
            "Sorry, can not handle images with IEEE floating-point samples");
115
0
        return (0);
116
0
    }
117
0
    colorchannels = td->td_samplesperpixel - td->td_extrasamples;
118
0
    if (!TIFFGetField(tif, TIFFTAG_PHOTOMETRIC, &photometric))
119
0
    {
120
0
        switch (colorchannels)
121
0
        {
122
0
            case 1:
123
0
                photometric = PHOTOMETRIC_MINISBLACK;
124
0
                break;
125
0
            case 3:
126
0
                photometric = PHOTOMETRIC_RGB;
127
0
                break;
128
0
            default:
129
0
                snprintf(emsg, EMSG_BUF_SIZE, "Missing needed %s tag",
130
0
                         photoTag);
131
0
                return (0);
132
0
        }
133
0
    }
134
0
    switch (photometric)
135
0
    {
136
0
        case PHOTOMETRIC_MINISWHITE:
137
0
        case PHOTOMETRIC_MINISBLACK:
138
0
        case PHOTOMETRIC_PALETTE:
139
0
            if (td->td_planarconfig == PLANARCONFIG_CONTIG &&
140
0
                td->td_samplesperpixel != 1 && td->td_bitspersample < 8)
141
0
            {
142
0
                snprintf(
143
0
                    emsg, EMSG_BUF_SIZE,
144
0
                    "Sorry, can not handle contiguous data with %s=%" PRIu16
145
0
                    ", "
146
0
                    "and %s=%" PRIu16 " and Bits/Sample=%" PRIu16 "",
147
0
                    photoTag, photometric, "Samples/pixel",
148
0
                    td->td_samplesperpixel, td->td_bitspersample);
149
0
                return (0);
150
0
            }
151
            /*
152
             * We should likely validate that any extra samples are either
153
             * to be ignored, or are alpha, and if alpha we should try to use
154
             * them.  But for now we won't bother with this.
155
             */
156
0
            break;
157
0
        case PHOTOMETRIC_YCBCR:
158
            /*
159
             * TODO: if at all meaningful and useful, make more complete
160
             * support check here, or better still, refactor to let supporting
161
             * code decide whether there is support and what meaningful
162
             * error to return
163
             */
164
0
            break;
165
0
        case PHOTOMETRIC_RGB:
166
0
            if (colorchannels < 3)
167
0
            {
168
0
                snprintf(emsg, EMSG_BUF_SIZE,
169
0
                         "Sorry, can not handle RGB image with %s=%d",
170
0
                         "Color channels", colorchannels);
171
0
                return (0);
172
0
            }
173
0
            break;
174
0
        case PHOTOMETRIC_SEPARATED:
175
0
        {
176
0
            uint16_t inkset;
177
0
            TIFFGetFieldDefaulted(tif, TIFFTAG_INKSET, &inkset);
178
0
            if (inkset != INKSET_CMYK)
179
0
            {
180
0
                snprintf(emsg, EMSG_BUF_SIZE,
181
0
                         "Sorry, can not handle separated image with %s=%d",
182
0
                         "InkSet", inkset);
183
0
                return 0;
184
0
            }
185
0
            if (td->td_samplesperpixel < 4)
186
0
            {
187
0
                snprintf(
188
0
                    emsg, EMSG_BUF_SIZE,
189
0
                    "Sorry, can not handle separated image with %s=%" PRIu16,
190
0
                    "Samples/pixel", td->td_samplesperpixel);
191
0
                return 0;
192
0
            }
193
0
            break;
194
0
        }
195
0
        case PHOTOMETRIC_LOGL:
196
0
            if (td->td_compression != COMPRESSION_SGILOG)
197
0
            {
198
0
                snprintf(emsg, EMSG_BUF_SIZE,
199
0
                         "Sorry, LogL data must have %s=%d", "Compression",
200
0
                         COMPRESSION_SGILOG);
201
0
                return (0);
202
0
            }
203
0
            break;
204
0
        case PHOTOMETRIC_LOGLUV:
205
0
            if (td->td_compression != COMPRESSION_SGILOG &&
206
0
                td->td_compression != COMPRESSION_SGILOG24)
207
0
            {
208
0
                snprintf(emsg, EMSG_BUF_SIZE,
209
0
                         "Sorry, LogLuv data must have %s=%d or %d",
210
0
                         "Compression", COMPRESSION_SGILOG,
211
0
                         COMPRESSION_SGILOG24);
212
0
                return (0);
213
0
            }
214
0
            if (td->td_planarconfig != PLANARCONFIG_CONTIG)
215
0
            {
216
0
                snprintf(emsg, EMSG_BUF_SIZE,
217
0
                         "Sorry, can not handle LogLuv images with %s=%" PRIu16,
218
0
                         "Planarconfiguration", td->td_planarconfig);
219
0
                return (0);
220
0
            }
221
0
            if (td->td_samplesperpixel != 3 || colorchannels != 3)
222
0
            {
223
0
                snprintf(emsg, EMSG_BUF_SIZE,
224
0
                         "Sorry, can not handle image with %s=%" PRIu16
225
0
                         ", %s=%d",
226
0
                         "Samples/pixel", td->td_samplesperpixel,
227
0
                         "colorchannels", colorchannels);
228
0
                return 0;
229
0
            }
230
0
            break;
231
0
        case PHOTOMETRIC_CIELAB:
232
0
            if (td->td_samplesperpixel != 3 || colorchannels != 3 ||
233
0
                (td->td_bitspersample != 8 && td->td_bitspersample != 16))
234
0
            {
235
0
                snprintf(emsg, EMSG_BUF_SIZE,
236
0
                         "Sorry, can not handle image with %s=%" PRIu16
237
0
                         ", %s=%d and %s=%" PRIu16,
238
0
                         "Samples/pixel", td->td_samplesperpixel,
239
0
                         "colorchannels", colorchannels, "Bits/sample",
240
0
                         td->td_bitspersample);
241
0
                return 0;
242
0
            }
243
0
            break;
244
0
        default:
245
0
            snprintf(emsg, EMSG_BUF_SIZE,
246
0
                     "Sorry, can not handle image with %s=%" PRIu16, photoTag,
247
0
                     photometric);
248
0
            return (0);
249
0
    }
250
0
    return (1);
251
0
}
252
253
void TIFFRGBAImageEnd(TIFFRGBAImage *img)
254
0
{
255
0
    if (img->Map)
256
0
    {
257
0
        _TIFFfreeExt(img->tif, img->Map);
258
0
        img->Map = NULL;
259
0
    }
260
0
    if (img->BWmap)
261
0
    {
262
0
        _TIFFfreeExt(img->tif, img->BWmap);
263
0
        img->BWmap = NULL;
264
0
    }
265
0
    if (img->PALmap)
266
0
    {
267
0
        _TIFFfreeExt(img->tif, img->PALmap);
268
0
        img->PALmap = NULL;
269
0
    }
270
0
    if (img->ycbcr)
271
0
    {
272
0
        _TIFFfreeExt(img->tif, img->ycbcr);
273
0
        img->ycbcr = NULL;
274
0
    }
275
0
    if (img->cielab)
276
0
    {
277
0
        _TIFFfreeExt(img->tif, img->cielab);
278
0
        img->cielab = NULL;
279
0
    }
280
0
    if (img->UaToAa)
281
0
    {
282
0
        _TIFFfreeExt(img->tif, img->UaToAa);
283
0
        img->UaToAa = NULL;
284
0
    }
285
0
    if (img->Bitdepth16To8)
286
0
    {
287
0
        _TIFFfreeExt(img->tif, img->Bitdepth16To8);
288
0
        img->Bitdepth16To8 = NULL;
289
0
    }
290
291
0
    if (img->redcmap)
292
0
    {
293
0
        _TIFFfreeExt(img->tif, img->redcmap);
294
0
        _TIFFfreeExt(img->tif, img->greencmap);
295
0
        _TIFFfreeExt(img->tif, img->bluecmap);
296
0
        img->redcmap = img->greencmap = img->bluecmap = NULL;
297
0
    }
298
0
}
299
300
static int isCCITTCompression(TIFF *tif)
301
0
{
302
0
    uint16_t compress;
303
0
    TIFFGetField(tif, TIFFTAG_COMPRESSION, &compress);
304
0
    return (compress == COMPRESSION_CCITTFAX3 ||
305
0
            compress == COMPRESSION_CCITTFAX4 ||
306
0
            compress == COMPRESSION_CCITTRLE ||
307
0
            compress == COMPRESSION_CCITTRLEW);
308
0
}
309
310
int TIFFRGBAImageBegin(TIFFRGBAImage *img, TIFF *tif, int stop,
311
                       char emsg[EMSG_BUF_SIZE])
312
0
{
313
0
    uint16_t *sampleinfo;
314
0
    uint16_t extrasamples;
315
0
    uint16_t planarconfig;
316
0
    uint16_t compress;
317
0
    int colorchannels;
318
0
    uint16_t *red_orig, *green_orig, *blue_orig;
319
0
    int n_color;
320
321
0
    if (!TIFFRGBAImageOK(tif, emsg))
322
0
        return 0;
323
324
    /* Initialize to normal values */
325
0
    img->row_offset = 0;
326
0
    img->col_offset = 0;
327
0
    img->redcmap = NULL;
328
0
    img->greencmap = NULL;
329
0
    img->bluecmap = NULL;
330
0
    img->Map = NULL;
331
0
    img->BWmap = NULL;
332
0
    img->PALmap = NULL;
333
0
    img->ycbcr = NULL;
334
0
    img->cielab = NULL;
335
0
    img->UaToAa = NULL;
336
0
    img->Bitdepth16To8 = NULL;
337
0
    img->req_orientation = ORIENTATION_BOTLEFT; /* It is the default */
338
339
0
    img->tif = tif;
340
0
    img->stoponerr = stop;
341
0
    TIFFGetFieldDefaulted(tif, TIFFTAG_BITSPERSAMPLE, &img->bitspersample);
342
0
    switch (img->bitspersample)
343
0
    {
344
0
        case 1:
345
0
        case 2:
346
0
        case 4:
347
0
        case 8:
348
0
        case 16:
349
0
            break;
350
0
        default:
351
0
            snprintf(emsg, EMSG_BUF_SIZE,
352
0
                     "Sorry, can not handle images with %" PRIu16
353
0
                     "-bit samples",
354
0
                     img->bitspersample);
355
0
            goto fail_return;
356
0
    }
357
0
    img->alpha = 0;
358
0
    TIFFGetFieldDefaulted(tif, TIFFTAG_SAMPLESPERPIXEL, &img->samplesperpixel);
359
0
    TIFFGetFieldDefaulted(tif, TIFFTAG_EXTRASAMPLES, &extrasamples,
360
0
                          &sampleinfo);
361
0
    if (extrasamples >= 1)
362
0
    {
363
0
        switch (sampleinfo[0])
364
0
        {
365
0
            case EXTRASAMPLE_UNSPECIFIED: /* Workaround for some images without
366
                                           */
367
0
                if (img->samplesperpixel >
368
0
                    3) /* correct info about alpha channel */
369
0
                    img->alpha = EXTRASAMPLE_ASSOCALPHA;
370
0
                break;
371
0
            case EXTRASAMPLE_ASSOCALPHA: /* data is pre-multiplied */
372
0
            case EXTRASAMPLE_UNASSALPHA: /* data is not pre-multiplied */
373
0
                img->alpha = sampleinfo[0];
374
0
                break;
375
0
        }
376
0
    }
377
378
0
#ifdef DEFAULT_EXTRASAMPLE_AS_ALPHA
379
0
    if (!TIFFGetField(tif, TIFFTAG_PHOTOMETRIC, &img->photometric))
380
0
        img->photometric = PHOTOMETRIC_MINISWHITE;
381
382
0
    if (extrasamples == 0 && img->samplesperpixel == 4 &&
383
0
        img->photometric == PHOTOMETRIC_RGB)
384
0
    {
385
0
        img->alpha = EXTRASAMPLE_ASSOCALPHA;
386
0
        extrasamples = 1;
387
0
    }
388
0
#endif
389
390
0
    colorchannels = img->samplesperpixel - extrasamples;
391
0
    TIFFGetFieldDefaulted(tif, TIFFTAG_COMPRESSION, &compress);
392
0
    TIFFGetFieldDefaulted(tif, TIFFTAG_PLANARCONFIG, &planarconfig);
393
0
    if (!TIFFGetField(tif, TIFFTAG_PHOTOMETRIC, &img->photometric))
394
0
    {
395
0
        switch (colorchannels)
396
0
        {
397
0
            case 1:
398
0
                if (isCCITTCompression(tif))
399
0
                    img->photometric = PHOTOMETRIC_MINISWHITE;
400
0
                else
401
0
                    img->photometric = PHOTOMETRIC_MINISBLACK;
402
0
                break;
403
0
            case 3:
404
0
                img->photometric = PHOTOMETRIC_RGB;
405
0
                break;
406
0
            default:
407
0
                snprintf(emsg, EMSG_BUF_SIZE, "Missing needed %s tag",
408
0
                         photoTag);
409
0
                goto fail_return;
410
0
        }
411
0
    }
412
0
    switch (img->photometric)
413
0
    {
414
0
        case PHOTOMETRIC_PALETTE:
415
0
            if (!TIFFGetField(tif, TIFFTAG_COLORMAP, &red_orig, &green_orig,
416
0
                              &blue_orig))
417
0
            {
418
0
                snprintf(emsg, EMSG_BUF_SIZE,
419
0
                         "Missing required \"Colormap\" tag");
420
0
                goto fail_return;
421
0
            }
422
423
            /* copy the colormaps so we can modify them */
424
0
            n_color = (1U << img->bitspersample);
425
0
            img->redcmap =
426
0
                (uint16_t *)_TIFFmallocExt(tif, sizeof(uint16_t) * n_color);
427
0
            img->greencmap =
428
0
                (uint16_t *)_TIFFmallocExt(tif, sizeof(uint16_t) * n_color);
429
0
            img->bluecmap =
430
0
                (uint16_t *)_TIFFmallocExt(tif, sizeof(uint16_t) * n_color);
431
0
            if (!img->redcmap || !img->greencmap || !img->bluecmap)
432
0
            {
433
0
                snprintf(emsg, EMSG_BUF_SIZE,
434
0
                         "Out of memory for colormap copy");
435
0
                goto fail_return;
436
0
            }
437
438
0
            _TIFFmemcpy(img->redcmap, red_orig, n_color * 2);
439
0
            _TIFFmemcpy(img->greencmap, green_orig, n_color * 2);
440
0
            _TIFFmemcpy(img->bluecmap, blue_orig, n_color * 2);
441
442
            /* fall through... */
443
0
        case PHOTOMETRIC_MINISWHITE:
444
0
        case PHOTOMETRIC_MINISBLACK:
445
0
            if (planarconfig == PLANARCONFIG_CONTIG &&
446
0
                img->samplesperpixel != 1 && img->bitspersample < 8)
447
0
            {
448
0
                snprintf(
449
0
                    emsg, EMSG_BUF_SIZE,
450
0
                    "Sorry, can not handle contiguous data with %s=%" PRIu16
451
0
                    ", "
452
0
                    "and %s=%" PRIu16 " and Bits/Sample=%" PRIu16,
453
0
                    photoTag, img->photometric, "Samples/pixel",
454
0
                    img->samplesperpixel, img->bitspersample);
455
0
                goto fail_return;
456
0
            }
457
0
            break;
458
0
        case PHOTOMETRIC_YCBCR:
459
            /* It would probably be nice to have a reality check here. */
460
0
            if (planarconfig == PLANARCONFIG_CONTIG)
461
                /* can rely on libjpeg to convert to RGB */
462
                /* XXX should restore current state on exit */
463
0
                switch (compress)
464
0
                {
465
0
                    case COMPRESSION_JPEG:
466
                        /*
467
                         * TODO: when complete tests verify complete
468
                         * desubsampling and YCbCr handling, remove use of
469
                         * TIFFTAG_JPEGCOLORMODE in favor of tif_getimage.c
470
                         * native handling
471
                         */
472
0
                        TIFFSetField(tif, TIFFTAG_JPEGCOLORMODE,
473
0
                                     JPEGCOLORMODE_RGB);
474
0
                        img->photometric = PHOTOMETRIC_RGB;
475
0
                        break;
476
0
                    default:
477
0
                        /* do nothing */;
478
0
                        break;
479
0
                }
480
            /*
481
             * TODO: if at all meaningful and useful, make more complete
482
             * support check here, or better still, refactor to let supporting
483
             * code decide whether there is support and what meaningful
484
             * error to return
485
             */
486
0
            break;
487
0
        case PHOTOMETRIC_RGB:
488
0
            if (colorchannels < 3)
489
0
            {
490
0
                snprintf(emsg, EMSG_BUF_SIZE,
491
0
                         "Sorry, can not handle RGB image with %s=%d",
492
0
                         "Color channels", colorchannels);
493
0
                goto fail_return;
494
0
            }
495
0
            break;
496
0
        case PHOTOMETRIC_SEPARATED:
497
0
        {
498
0
            uint16_t inkset;
499
0
            TIFFGetFieldDefaulted(tif, TIFFTAG_INKSET, &inkset);
500
0
            if (inkset != INKSET_CMYK)
501
0
            {
502
0
                snprintf(
503
0
                    emsg, EMSG_BUF_SIZE,
504
0
                    "Sorry, can not handle separated image with %s=%" PRIu16,
505
0
                    "InkSet", inkset);
506
0
                goto fail_return;
507
0
            }
508
0
            if (img->samplesperpixel < 4)
509
0
            {
510
0
                snprintf(
511
0
                    emsg, EMSG_BUF_SIZE,
512
0
                    "Sorry, can not handle separated image with %s=%" PRIu16,
513
0
                    "Samples/pixel", img->samplesperpixel);
514
0
                goto fail_return;
515
0
            }
516
0
        }
517
0
        break;
518
0
        case PHOTOMETRIC_LOGL:
519
0
            if (compress != COMPRESSION_SGILOG)
520
0
            {
521
0
                snprintf(emsg, EMSG_BUF_SIZE,
522
0
                         "Sorry, LogL data must have %s=%d", "Compression",
523
0
                         COMPRESSION_SGILOG);
524
0
                goto fail_return;
525
0
            }
526
0
            TIFFSetField(tif, TIFFTAG_SGILOGDATAFMT, SGILOGDATAFMT_8BIT);
527
0
            img->photometric = PHOTOMETRIC_MINISBLACK; /* little white lie */
528
0
            img->bitspersample = 8;
529
0
            break;
530
0
        case PHOTOMETRIC_LOGLUV:
531
0
            if (compress != COMPRESSION_SGILOG &&
532
0
                compress != COMPRESSION_SGILOG24)
533
0
            {
534
0
                snprintf(emsg, EMSG_BUF_SIZE,
535
0
                         "Sorry, LogLuv data must have %s=%d or %d",
536
0
                         "Compression", COMPRESSION_SGILOG,
537
0
                         COMPRESSION_SGILOG24);
538
0
                goto fail_return;
539
0
            }
540
0
            if (planarconfig != PLANARCONFIG_CONTIG)
541
0
            {
542
0
                snprintf(emsg, EMSG_BUF_SIZE,
543
0
                         "Sorry, can not handle LogLuv images with %s=%" PRIu16,
544
0
                         "Planarconfiguration", planarconfig);
545
0
                return (0);
546
0
            }
547
0
            TIFFSetField(tif, TIFFTAG_SGILOGDATAFMT, SGILOGDATAFMT_8BIT);
548
0
            img->photometric = PHOTOMETRIC_RGB; /* little white lie */
549
0
            img->bitspersample = 8;
550
0
            break;
551
0
        case PHOTOMETRIC_CIELAB:
552
0
            break;
553
0
        default:
554
0
            snprintf(emsg, EMSG_BUF_SIZE,
555
0
                     "Sorry, can not handle image with %s=%" PRIu16, photoTag,
556
0
                     img->photometric);
557
0
            goto fail_return;
558
0
    }
559
0
    TIFFGetField(tif, TIFFTAG_IMAGEWIDTH, &img->width);
560
0
    TIFFGetField(tif, TIFFTAG_IMAGELENGTH, &img->height);
561
0
    TIFFGetFieldDefaulted(tif, TIFFTAG_ORIENTATION, &img->orientation);
562
0
    img->isContig =
563
0
        !(planarconfig == PLANARCONFIG_SEPARATE && img->samplesperpixel > 1);
564
0
    if (img->isContig)
565
0
    {
566
0
        if (!PickContigCase(img))
567
0
        {
568
0
            snprintf(emsg, EMSG_BUF_SIZE, "Sorry, can not handle image");
569
0
            goto fail_return;
570
0
        }
571
0
    }
572
0
    else
573
0
    {
574
0
        if (!PickSeparateCase(img))
575
0
        {
576
0
            snprintf(emsg, EMSG_BUF_SIZE, "Sorry, can not handle image");
577
0
            goto fail_return;
578
0
        }
579
0
    }
580
0
    return 1;
581
582
0
fail_return:
583
0
    TIFFRGBAImageEnd(img);
584
0
    return 0;
585
0
}
586
587
int TIFFRGBAImageGet(TIFFRGBAImage *img, uint32_t *raster, uint32_t w,
588
                     uint32_t h)
589
0
{
590
0
    if (img->get == NULL)
591
0
    {
592
0
        TIFFErrorExtR(img->tif, TIFFFileName(img->tif),
593
0
                      "No \"get\" routine setup");
594
0
        return (0);
595
0
    }
596
0
    if (img->put.any == NULL)
597
0
    {
598
0
        TIFFErrorExtR(
599
0
            img->tif, TIFFFileName(img->tif),
600
0
            "No \"put\" routine setupl; probably can not handle image format");
601
0
        return (0);
602
0
    }
603
    /* Verify raster width and height against image width and height. */
604
0
    if (h > img->height)
605
0
    {
606
        /* Adapt parameters to read only available lines and put image at
607
         * the bottom of the raster. */
608
0
        raster += (size_t)(h - img->height) * w;
609
0
        h = img->height;
610
0
    }
611
0
    if (w > img->width)
612
0
    {
613
0
        TIFFWarningExtR(img->tif, TIFFFileName(img->tif),
614
0
                        "Raster width of %d shall not be larger than image "
615
0
                        "width of %d -> raster width adapted for reading",
616
0
                        w, img->width);
617
0
        w = img->width;
618
0
    }
619
0
    return (*img->get)(img, raster, w, h);
620
0
}
621
622
/*
623
 * Read the specified image into an ABGR-format rastertaking in account
624
 * specified orientation.
625
 */
626
int TIFFReadRGBAImageOriented(TIFF *tif, uint32_t rwidth, uint32_t rheight,
627
                              uint32_t *raster, int orientation, int stop)
628
0
{
629
0
    char emsg[EMSG_BUF_SIZE] = "";
630
0
    TIFFRGBAImage img;
631
0
    int ok;
632
633
0
    if (TIFFRGBAImageBegin(&img, tif, stop, emsg))
634
0
    {
635
0
        img.req_orientation = (uint16_t)orientation;
636
0
        ok = TIFFRGBAImageGet(&img, raster, rwidth, rheight);
637
0
        TIFFRGBAImageEnd(&img);
638
0
    }
639
0
    else
640
0
    {
641
0
        TIFFErrorExtR(tif, TIFFFileName(tif), "%s", emsg);
642
0
        ok = 0;
643
0
    }
644
0
    return (ok);
645
0
}
646
647
/*
648
 * Read the specified image into an ABGR-format raster. Use bottom left
649
 * origin for raster by default.
650
 */
651
int TIFFReadRGBAImage(TIFF *tif, uint32_t rwidth, uint32_t rheight,
652
                      uint32_t *raster, int stop)
653
0
{
654
0
    return TIFFReadRGBAImageOriented(tif, rwidth, rheight, raster,
655
0
                                     ORIENTATION_BOTLEFT, stop);
656
0
}
657
658
static int setorientation(TIFFRGBAImage *img)
659
0
{
660
0
    switch (img->orientation)
661
0
    {
662
0
        case ORIENTATION_TOPLEFT:
663
0
        case ORIENTATION_LEFTTOP:
664
0
            if (img->req_orientation == ORIENTATION_TOPRIGHT ||
665
0
                img->req_orientation == ORIENTATION_RIGHTTOP)
666
0
                return FLIP_HORIZONTALLY;
667
0
            else if (img->req_orientation == ORIENTATION_BOTRIGHT ||
668
0
                     img->req_orientation == ORIENTATION_RIGHTBOT)
669
0
                return FLIP_HORIZONTALLY | FLIP_VERTICALLY;
670
0
            else if (img->req_orientation == ORIENTATION_BOTLEFT ||
671
0
                     img->req_orientation == ORIENTATION_LEFTBOT)
672
0
                return FLIP_VERTICALLY;
673
0
            else
674
0
                return 0;
675
0
        case ORIENTATION_TOPRIGHT:
676
0
        case ORIENTATION_RIGHTTOP:
677
0
            if (img->req_orientation == ORIENTATION_TOPLEFT ||
678
0
                img->req_orientation == ORIENTATION_LEFTTOP)
679
0
                return FLIP_HORIZONTALLY;
680
0
            else if (img->req_orientation == ORIENTATION_BOTRIGHT ||
681
0
                     img->req_orientation == ORIENTATION_RIGHTBOT)
682
0
                return FLIP_VERTICALLY;
683
0
            else if (img->req_orientation == ORIENTATION_BOTLEFT ||
684
0
                     img->req_orientation == ORIENTATION_LEFTBOT)
685
0
                return FLIP_HORIZONTALLY | FLIP_VERTICALLY;
686
0
            else
687
0
                return 0;
688
0
        case ORIENTATION_BOTRIGHT:
689
0
        case ORIENTATION_RIGHTBOT:
690
0
            if (img->req_orientation == ORIENTATION_TOPLEFT ||
691
0
                img->req_orientation == ORIENTATION_LEFTTOP)
692
0
                return FLIP_HORIZONTALLY | FLIP_VERTICALLY;
693
0
            else if (img->req_orientation == ORIENTATION_TOPRIGHT ||
694
0
                     img->req_orientation == ORIENTATION_RIGHTTOP)
695
0
                return FLIP_VERTICALLY;
696
0
            else if (img->req_orientation == ORIENTATION_BOTLEFT ||
697
0
                     img->req_orientation == ORIENTATION_LEFTBOT)
698
0
                return FLIP_HORIZONTALLY;
699
0
            else
700
0
                return 0;
701
0
        case ORIENTATION_BOTLEFT:
702
0
        case ORIENTATION_LEFTBOT:
703
0
            if (img->req_orientation == ORIENTATION_TOPLEFT ||
704
0
                img->req_orientation == ORIENTATION_LEFTTOP)
705
0
                return FLIP_VERTICALLY;
706
0
            else if (img->req_orientation == ORIENTATION_TOPRIGHT ||
707
0
                     img->req_orientation == ORIENTATION_RIGHTTOP)
708
0
                return FLIP_HORIZONTALLY | FLIP_VERTICALLY;
709
0
            else if (img->req_orientation == ORIENTATION_BOTRIGHT ||
710
0
                     img->req_orientation == ORIENTATION_RIGHTBOT)
711
0
                return FLIP_HORIZONTALLY;
712
0
            else
713
0
                return 0;
714
0
        default: /* NOTREACHED */
715
0
            return 0;
716
0
    }
717
0
}
718
719
/*
720
 * Get an tile-organized image that has
721
 *  PlanarConfiguration contiguous if SamplesPerPixel > 1
722
 * or
723
 *  SamplesPerPixel == 1
724
 */
725
static int gtTileContig(TIFFRGBAImage *img, uint32_t *raster, uint32_t w,
726
                        uint32_t h)
727
0
{
728
0
    TIFF *tif = img->tif;
729
0
    tileContigRoutine put = img->put.contig;
730
0
    uint32_t col, row, y, rowstoread;
731
0
    tmsize_t pos;
732
0
    uint32_t tw, th;
733
0
    unsigned char *buf = NULL;
734
0
    int32_t fromskew, toskew;
735
0
    uint32_t nrow;
736
0
    int ret = 1, flip;
737
0
    uint32_t this_tw, tocol;
738
0
    int32_t this_toskew, leftmost_toskew;
739
0
    int32_t leftmost_fromskew;
740
0
    uint32_t leftmost_tw;
741
0
    tmsize_t bufsize;
742
743
0
    bufsize = TIFFTileSize(tif);
744
0
    if (bufsize == 0)
745
0
    {
746
0
        TIFFErrorExtR(tif, TIFFFileName(tif), "%s", "No space for tile buffer");
747
0
        return (0);
748
0
    }
749
750
0
    TIFFGetField(tif, TIFFTAG_TILEWIDTH, &tw);
751
0
    TIFFGetField(tif, TIFFTAG_TILELENGTH, &th);
752
753
0
    flip = setorientation(img);
754
0
    if (flip & FLIP_VERTICALLY)
755
0
    {
756
0
        if (((int64_t)tw + w) > INT_MAX)
757
0
        {
758
0
            TIFFErrorExtR(tif, TIFFFileName(tif), "%s",
759
0
                          "unsupported tile size (too wide)");
760
0
            return (0);
761
0
        }
762
0
        y = h - 1;
763
0
        toskew = -(int32_t)(tw + w);
764
0
    }
765
0
    else
766
0
    {
767
0
        if (tw > ((int64_t)INT_MAX + w))
768
0
        {
769
0
            TIFFErrorExtR(tif, TIFFFileName(tif), "%s",
770
0
                          "unsupported tile size (too wide)");
771
0
            return (0);
772
0
        }
773
0
        y = 0;
774
0
        toskew = -(int32_t)(tw - w);
775
0
    }
776
777
0
    if (tw == 0 || th == 0)
778
0
    {
779
0
        TIFFErrorExtR(tif, TIFFFileName(tif), "tile width or height is zero");
780
0
        return (0);
781
0
    }
782
783
    /*
784
     *  Leftmost tile is clipped on left side if col_offset > 0.
785
     */
786
0
    leftmost_fromskew = img->col_offset % tw;
787
0
    leftmost_tw = tw - leftmost_fromskew;
788
0
    int64_t skew_i64 = (int64_t)toskew + leftmost_fromskew;
789
0
    if (skew_i64 > INT_MAX || skew_i64 < INT_MIN)
790
0
    {
791
0
        TIFFErrorExtR(tif, TIFFFileName(tif), "%s %" PRId64, "Invalid skew",
792
0
                      skew_i64);
793
0
        return (0);
794
0
    }
795
0
    leftmost_toskew = (int32_t)skew_i64;
796
0
    for (row = 0; ret != 0 && row < h; row += nrow)
797
0
    {
798
0
        rowstoread = th - (row + img->row_offset) % th;
799
0
        nrow = (row + rowstoread > h ? h - row : rowstoread);
800
0
        fromskew = leftmost_fromskew;
801
0
        this_tw = leftmost_tw;
802
0
        this_toskew = leftmost_toskew;
803
0
        tocol = 0;
804
0
        col = img->col_offset;
805
0
        while (tocol < w)
806
0
        {
807
0
            if (_TIFFReadTileAndAllocBuffer(tif, (void **)&buf, bufsize, col,
808
0
                                            row + img->row_offset, 0,
809
0
                                            0) == (tmsize_t)(-1) &&
810
0
                (buf == NULL || img->stoponerr))
811
0
            {
812
0
                ret = 0;
813
0
                break;
814
0
            }
815
0
            pos = ((row + img->row_offset) % th) * TIFFTileRowSize(tif) +
816
0
                  ((tmsize_t)fromskew * img->samplesperpixel);
817
0
            if (tocol + this_tw > w)
818
0
            {
819
                /*
820
                 * Rightmost tile is clipped on right side.
821
                 */
822
0
                fromskew = tw - (w - tocol);
823
0
                this_tw = tw - fromskew;
824
0
                this_toskew = toskew + fromskew;
825
0
            }
826
0
            tmsize_t roffset = (tmsize_t)y * w + tocol;
827
0
            (*put)(img, raster + roffset, tocol, y, this_tw, nrow, fromskew,
828
0
                   this_toskew, buf + pos);
829
0
            tocol += this_tw;
830
0
            col += this_tw;
831
            /*
832
             * After the leftmost tile, tiles are no longer clipped on left
833
             * side.
834
             */
835
0
            fromskew = 0;
836
0
            this_tw = tw;
837
0
            this_toskew = toskew;
838
0
        }
839
840
0
        y += ((flip & FLIP_VERTICALLY) ? -(int32_t)nrow : (int32_t)nrow);
841
0
    }
842
0
    _TIFFfreeExt(img->tif, buf);
843
844
0
    if (flip & FLIP_HORIZONTALLY)
845
0
    {
846
0
        uint32_t line;
847
848
0
        for (line = 0; line < h; line++)
849
0
        {
850
0
            uint32_t *left = raster + (line * w);
851
0
            uint32_t *right = left + w - 1;
852
853
0
            while (left < right)
854
0
            {
855
0
                uint32_t temp = *left;
856
0
                *left = *right;
857
0
                *right = temp;
858
0
                left++;
859
0
                right--;
860
0
            }
861
0
        }
862
0
    }
863
864
0
    return (ret);
865
0
}
866
867
/*
868
 * Get an tile-organized image that has
869
 *   SamplesPerPixel > 1
870
 *   PlanarConfiguration separated
871
 * We assume that all such images are RGB.
872
 */
873
static int gtTileSeparate(TIFFRGBAImage *img, uint32_t *raster, uint32_t w,
874
                          uint32_t h)
875
0
{
876
0
    TIFF *tif = img->tif;
877
0
    tileSeparateRoutine put = img->put.separate;
878
0
    uint32_t col, row, y, rowstoread;
879
0
    tmsize_t pos;
880
0
    uint32_t tw, th;
881
0
    unsigned char *buf = NULL;
882
0
    unsigned char *p0 = NULL;
883
0
    unsigned char *p1 = NULL;
884
0
    unsigned char *p2 = NULL;
885
0
    unsigned char *pa = NULL;
886
0
    tmsize_t tilesize;
887
0
    tmsize_t bufsize;
888
0
    int32_t fromskew, toskew;
889
0
    int alpha = img->alpha;
890
0
    uint32_t nrow;
891
0
    int ret = 1, flip;
892
0
    uint16_t colorchannels;
893
0
    uint32_t this_tw, tocol;
894
0
    int32_t this_toskew, leftmost_toskew;
895
0
    int32_t leftmost_fromskew;
896
0
    uint32_t leftmost_tw;
897
898
0
    tilesize = TIFFTileSize(tif);
899
0
    bufsize =
900
0
        _TIFFMultiplySSize(tif, alpha ? 4 : 3, tilesize, "gtTileSeparate");
901
0
    if (bufsize == 0)
902
0
    {
903
0
        return (0);
904
0
    }
905
906
0
    TIFFGetField(tif, TIFFTAG_TILEWIDTH, &tw);
907
0
    TIFFGetField(tif, TIFFTAG_TILELENGTH, &th);
908
909
0
    flip = setorientation(img);
910
0
    if (flip & FLIP_VERTICALLY)
911
0
    {
912
0
        if (((int64_t)tw + w) > INT_MAX)
913
0
        {
914
0
            TIFFErrorExtR(tif, TIFFFileName(tif), "%s",
915
0
                          "unsupported tile size (too wide)");
916
0
            return (0);
917
0
        }
918
0
        y = h - 1;
919
0
        toskew = -(int32_t)(tw + w);
920
0
    }
921
0
    else
922
0
    {
923
0
        if (tw > ((int64_t)INT_MAX + w))
924
0
        {
925
0
            TIFFErrorExtR(tif, TIFFFileName(tif), "%s",
926
0
                          "unsupported tile size (too wide)");
927
0
            return (0);
928
0
        }
929
0
        y = 0;
930
0
        toskew = -(int32_t)(tw - w);
931
0
    }
932
933
0
    switch (img->photometric)
934
0
    {
935
0
        case PHOTOMETRIC_MINISWHITE:
936
0
        case PHOTOMETRIC_MINISBLACK:
937
0
        case PHOTOMETRIC_PALETTE:
938
0
            colorchannels = 1;
939
0
            break;
940
941
0
        default:
942
0
            colorchannels = 3;
943
0
            break;
944
0
    }
945
946
0
    if (tw == 0 || th == 0)
947
0
    {
948
0
        TIFFErrorExtR(tif, TIFFFileName(tif), "tile width or height is zero");
949
0
        return (0);
950
0
    }
951
952
    /*
953
     *  Leftmost tile is clipped on left side if col_offset > 0.
954
     */
955
0
    leftmost_fromskew = img->col_offset % tw;
956
0
    leftmost_tw = tw - leftmost_fromskew;
957
0
    int64_t skew_i64 = (int64_t)toskew + leftmost_fromskew;
958
0
    if (skew_i64 > INT_MAX || skew_i64 < INT_MIN)
959
0
    {
960
0
        TIFFErrorExtR(tif, TIFFFileName(tif), "%s %" PRId64, "Invalid skew",
961
0
                      skew_i64);
962
0
        return (0);
963
0
    }
964
0
    leftmost_toskew = (int32_t)skew_i64;
965
0
    for (row = 0; ret != 0 && row < h; row += nrow)
966
0
    {
967
0
        rowstoread = th - (row + img->row_offset) % th;
968
0
        nrow = (row + rowstoread > h ? h - row : rowstoread);
969
0
        fromskew = leftmost_fromskew;
970
0
        this_tw = leftmost_tw;
971
0
        this_toskew = leftmost_toskew;
972
0
        tocol = 0;
973
0
        col = img->col_offset;
974
0
        while (tocol < w)
975
0
        {
976
0
            if (buf == NULL)
977
0
            {
978
0
                if (_TIFFReadTileAndAllocBuffer(tif, (void **)&buf, bufsize,
979
0
                                                col, row + img->row_offset, 0,
980
0
                                                0) == (tmsize_t)(-1) &&
981
0
                    (buf == NULL || img->stoponerr))
982
0
                {
983
0
                    ret = 0;
984
0
                    break;
985
0
                }
986
0
                p0 = buf;
987
0
                if (colorchannels == 1)
988
0
                {
989
0
                    p2 = p1 = p0;
990
0
                    pa = (alpha ? (p0 + 3 * tilesize) : NULL);
991
0
                }
992
0
                else
993
0
                {
994
0
                    p1 = p0 + tilesize;
995
0
                    p2 = p1 + tilesize;
996
0
                    pa = (alpha ? (p2 + tilesize) : NULL);
997
0
                }
998
0
            }
999
0
            else if (TIFFReadTile(tif, p0, col, row + img->row_offset, 0, 0) ==
1000
0
                         (tmsize_t)(-1) &&
1001
0
                     img->stoponerr)
1002
0
            {
1003
0
                ret = 0;
1004
0
                break;
1005
0
            }
1006
0
            if (colorchannels > 1 &&
1007
0
                TIFFReadTile(tif, p1, col, row + img->row_offset, 0, 1) ==
1008
0
                    (tmsize_t)(-1) &&
1009
0
                img->stoponerr)
1010
0
            {
1011
0
                ret = 0;
1012
0
                break;
1013
0
            }
1014
0
            if (colorchannels > 1 &&
1015
0
                TIFFReadTile(tif, p2, col, row + img->row_offset, 0, 2) ==
1016
0
                    (tmsize_t)(-1) &&
1017
0
                img->stoponerr)
1018
0
            {
1019
0
                ret = 0;
1020
0
                break;
1021
0
            }
1022
0
            if (alpha &&
1023
0
                TIFFReadTile(tif, pa, col, row + img->row_offset, 0,
1024
0
                             colorchannels) == (tmsize_t)(-1) &&
1025
0
                img->stoponerr)
1026
0
            {
1027
0
                ret = 0;
1028
0
                break;
1029
0
            }
1030
1031
0
            pos = ((row + img->row_offset) % th) * TIFFTileRowSize(tif) +
1032
0
                  ((tmsize_t)fromskew * img->samplesperpixel);
1033
0
            if (tocol + this_tw > w)
1034
0
            {
1035
                /*
1036
                 * Rightmost tile is clipped on right side.
1037
                 */
1038
0
                fromskew = tw - (w - tocol);
1039
0
                this_tw = tw - fromskew;
1040
0
                this_toskew = toskew + fromskew;
1041
0
            }
1042
0
            tmsize_t roffset = (tmsize_t)y * w + tocol;
1043
0
            (*put)(img, raster + roffset, tocol, y, this_tw, nrow, fromskew,
1044
0
                   this_toskew, p0 + pos, p1 + pos, p2 + pos,
1045
0
                   (alpha ? (pa + pos) : NULL));
1046
0
            tocol += this_tw;
1047
0
            col += this_tw;
1048
            /*
1049
             * After the leftmost tile, tiles are no longer clipped on left
1050
             * side.
1051
             */
1052
0
            fromskew = 0;
1053
0
            this_tw = tw;
1054
0
            this_toskew = toskew;
1055
0
        }
1056
1057
0
        y += ((flip & FLIP_VERTICALLY) ? -(int32_t)nrow : (int32_t)nrow);
1058
0
    }
1059
1060
0
    if (flip & FLIP_HORIZONTALLY)
1061
0
    {
1062
0
        uint32_t line;
1063
1064
0
        for (line = 0; line < h; line++)
1065
0
        {
1066
0
            uint32_t *left = raster + (line * w);
1067
0
            uint32_t *right = left + w - 1;
1068
1069
0
            while (left < right)
1070
0
            {
1071
0
                uint32_t temp = *left;
1072
0
                *left = *right;
1073
0
                *right = temp;
1074
0
                left++;
1075
0
                right--;
1076
0
            }
1077
0
        }
1078
0
    }
1079
1080
0
    _TIFFfreeExt(img->tif, buf);
1081
0
    return (ret);
1082
0
}
1083
1084
/*
1085
 * Get a strip-organized image that has
1086
 *  PlanarConfiguration contiguous if SamplesPerPixel > 1
1087
 * or
1088
 *  SamplesPerPixel == 1
1089
 */
1090
static int gtStripContig(TIFFRGBAImage *img, uint32_t *raster, uint32_t w,
1091
                         uint32_t h)
1092
0
{
1093
0
    TIFF *tif = img->tif;
1094
0
    tileContigRoutine put = img->put.contig;
1095
0
    uint32_t row, y, nrow, nrowsub, rowstoread;
1096
0
    tmsize_t pos;
1097
0
    unsigned char *buf = NULL;
1098
0
    uint32_t rowsperstrip;
1099
0
    uint16_t subsamplinghor, subsamplingver;
1100
0
    uint32_t imagewidth = img->width;
1101
0
    tmsize_t scanline;
1102
0
    int32_t fromskew, toskew;
1103
0
    int ret = 1, flip;
1104
0
    tmsize_t maxstripsize;
1105
1106
0
    TIFFGetFieldDefaulted(tif, TIFFTAG_YCBCRSUBSAMPLING, &subsamplinghor,
1107
0
                          &subsamplingver);
1108
0
    if (subsamplingver == 0)
1109
0
    {
1110
0
        TIFFErrorExtR(tif, TIFFFileName(tif),
1111
0
                      "Invalid vertical YCbCr subsampling");
1112
0
        return (0);
1113
0
    }
1114
1115
0
    maxstripsize = TIFFStripSize(tif);
1116
1117
0
    flip = setorientation(img);
1118
0
    if (flip & FLIP_VERTICALLY)
1119
0
    {
1120
0
        if (w > INT_MAX / 2)
1121
0
        {
1122
0
            TIFFErrorExtR(tif, TIFFFileName(tif), "Width overflow");
1123
0
            return (0);
1124
0
        }
1125
0
        y = h - 1;
1126
0
        toskew = -(int32_t)(w + w);
1127
0
    }
1128
0
    else
1129
0
    {
1130
0
        y = 0;
1131
0
        toskew = 0;
1132
0
    }
1133
1134
0
    TIFFGetFieldDefaulted(tif, TIFFTAG_ROWSPERSTRIP, &rowsperstrip);
1135
0
    if (rowsperstrip == 0)
1136
0
    {
1137
0
        TIFFErrorExtR(tif, TIFFFileName(tif), "rowsperstrip is zero");
1138
0
        return (0);
1139
0
    }
1140
1141
0
    scanline = TIFFScanlineSize(tif);
1142
0
    fromskew = (w < imagewidth ? imagewidth - w : 0);
1143
0
    for (row = 0; row < h; row += nrow)
1144
0
    {
1145
0
        uint32_t temp;
1146
0
        rowstoread = rowsperstrip - (row + img->row_offset) % rowsperstrip;
1147
0
        nrow = (row + rowstoread > h ? h - row : rowstoread);
1148
0
        nrowsub = nrow;
1149
0
        if ((nrowsub % subsamplingver) != 0)
1150
0
            nrowsub += subsamplingver - nrowsub % subsamplingver;
1151
0
        temp = (row + img->row_offset) % rowsperstrip + nrowsub;
1152
0
        if (scanline > 0 && temp > (size_t)(TIFF_TMSIZE_T_MAX / scanline))
1153
0
        {
1154
0
            TIFFErrorExtR(tif, TIFFFileName(tif),
1155
0
                          "Integer overflow in gtStripContig");
1156
0
            return 0;
1157
0
        }
1158
0
        if (_TIFFReadEncodedStripAndAllocBuffer(
1159
0
                tif, TIFFComputeStrip(tif, row + img->row_offset, 0),
1160
0
                (void **)(&buf), maxstripsize,
1161
0
                temp * scanline) == (tmsize_t)(-1) &&
1162
0
            (buf == NULL || img->stoponerr))
1163
0
        {
1164
0
            ret = 0;
1165
0
            break;
1166
0
        }
1167
1168
0
        pos = ((row + img->row_offset) % rowsperstrip) * scanline +
1169
0
              ((tmsize_t)img->col_offset * img->samplesperpixel);
1170
0
        tmsize_t roffset = (tmsize_t)y * w;
1171
0
        (*put)(img, raster + roffset, 0, y, w, nrow, fromskew, toskew,
1172
0
               buf + pos);
1173
0
        y += ((flip & FLIP_VERTICALLY) ? -(int32_t)nrow : (int32_t)nrow);
1174
0
    }
1175
1176
0
    if (flip & FLIP_HORIZONTALLY)
1177
0
    {
1178
0
        uint32_t line;
1179
1180
0
        for (line = 0; line < h; line++)
1181
0
        {
1182
0
            uint32_t *left = raster + (line * w);
1183
0
            uint32_t *right = left + w - 1;
1184
1185
0
            while (left < right)
1186
0
            {
1187
0
                uint32_t temp = *left;
1188
0
                *left = *right;
1189
0
                *right = temp;
1190
0
                left++;
1191
0
                right--;
1192
0
            }
1193
0
        }
1194
0
    }
1195
1196
0
    _TIFFfreeExt(img->tif, buf);
1197
0
    return (ret);
1198
0
}
1199
1200
/*
1201
 * Get a strip-organized image with
1202
 *   SamplesPerPixel > 1
1203
 *   PlanarConfiguration separated
1204
 * We assume that all such images are RGB.
1205
 */
1206
static int gtStripSeparate(TIFFRGBAImage *img, uint32_t *raster, uint32_t w,
1207
                           uint32_t h)
1208
0
{
1209
0
    TIFF *tif = img->tif;
1210
0
    tileSeparateRoutine put = img->put.separate;
1211
0
    unsigned char *buf = NULL;
1212
0
    unsigned char *p0 = NULL, *p1 = NULL, *p2 = NULL, *pa = NULL;
1213
0
    uint32_t row, y, nrow, rowstoread;
1214
0
    tmsize_t pos;
1215
0
    tmsize_t scanline;
1216
0
    uint32_t rowsperstrip, offset_row;
1217
0
    uint32_t imagewidth = img->width;
1218
0
    tmsize_t stripsize;
1219
0
    tmsize_t bufsize;
1220
0
    int32_t fromskew, toskew;
1221
0
    int alpha = img->alpha;
1222
0
    int ret = 1, flip;
1223
0
    uint16_t colorchannels;
1224
1225
0
    stripsize = TIFFStripSize(tif);
1226
0
    bufsize =
1227
0
        _TIFFMultiplySSize(tif, alpha ? 4 : 3, stripsize, "gtStripSeparate");
1228
0
    if (bufsize == 0)
1229
0
    {
1230
0
        return (0);
1231
0
    }
1232
1233
0
    flip = setorientation(img);
1234
0
    if (flip & FLIP_VERTICALLY)
1235
0
    {
1236
0
        if (w > INT_MAX / 2)
1237
0
        {
1238
0
            TIFFErrorExtR(tif, TIFFFileName(tif), "Width overflow");
1239
0
            return (0);
1240
0
        }
1241
0
        y = h - 1;
1242
0
        toskew = -(int32_t)(w + w);
1243
0
    }
1244
0
    else
1245
0
    {
1246
0
        y = 0;
1247
0
        toskew = 0;
1248
0
    }
1249
1250
0
    switch (img->photometric)
1251
0
    {
1252
0
        case PHOTOMETRIC_MINISWHITE:
1253
0
        case PHOTOMETRIC_MINISBLACK:
1254
0
        case PHOTOMETRIC_PALETTE:
1255
0
            colorchannels = 1;
1256
0
            break;
1257
1258
0
        default:
1259
0
            colorchannels = 3;
1260
0
            break;
1261
0
    }
1262
1263
0
    TIFFGetFieldDefaulted(tif, TIFFTAG_ROWSPERSTRIP, &rowsperstrip);
1264
0
    if (rowsperstrip == 0)
1265
0
    {
1266
0
        TIFFErrorExtR(tif, TIFFFileName(tif), "rowsperstrip is zero");
1267
0
        return (0);
1268
0
    }
1269
1270
0
    scanline = TIFFScanlineSize(tif);
1271
0
    fromskew = (w < imagewidth ? imagewidth - w : 0);
1272
0
    for (row = 0; row < h; row += nrow)
1273
0
    {
1274
0
        uint32_t temp;
1275
0
        rowstoread = rowsperstrip - (row + img->row_offset) % rowsperstrip;
1276
0
        nrow = (row + rowstoread > h ? h - row : rowstoread);
1277
0
        offset_row = row + img->row_offset;
1278
0
        temp = (row + img->row_offset) % rowsperstrip + nrow;
1279
0
        if (scanline > 0 && temp > (size_t)(TIFF_TMSIZE_T_MAX / scanline))
1280
0
        {
1281
0
            TIFFErrorExtR(tif, TIFFFileName(tif),
1282
0
                          "Integer overflow in gtStripSeparate");
1283
0
            return 0;
1284
0
        }
1285
0
        if (buf == NULL)
1286
0
        {
1287
0
            if (_TIFFReadEncodedStripAndAllocBuffer(
1288
0
                    tif, TIFFComputeStrip(tif, offset_row, 0), (void **)&buf,
1289
0
                    bufsize, temp * scanline) == (tmsize_t)(-1) &&
1290
0
                (buf == NULL || img->stoponerr))
1291
0
            {
1292
0
                ret = 0;
1293
0
                break;
1294
0
            }
1295
0
            p0 = buf;
1296
0
            if (colorchannels == 1)
1297
0
            {
1298
0
                p2 = p1 = p0;
1299
0
                pa = (alpha ? (p0 + 3 * stripsize) : NULL);
1300
0
            }
1301
0
            else
1302
0
            {
1303
0
                p1 = p0 + stripsize;
1304
0
                p2 = p1 + stripsize;
1305
0
                pa = (alpha ? (p2 + stripsize) : NULL);
1306
0
            }
1307
0
        }
1308
0
        else if (TIFFReadEncodedStrip(tif, TIFFComputeStrip(tif, offset_row, 0),
1309
0
                                      p0, temp * scanline) == (tmsize_t)(-1) &&
1310
0
                 img->stoponerr)
1311
0
        {
1312
0
            ret = 0;
1313
0
            break;
1314
0
        }
1315
0
        if (colorchannels > 1 &&
1316
0
            TIFFReadEncodedStrip(tif, TIFFComputeStrip(tif, offset_row, 1), p1,
1317
0
                                 temp * scanline) == (tmsize_t)(-1) &&
1318
0
            img->stoponerr)
1319
0
        {
1320
0
            ret = 0;
1321
0
            break;
1322
0
        }
1323
0
        if (colorchannels > 1 &&
1324
0
            TIFFReadEncodedStrip(tif, TIFFComputeStrip(tif, offset_row, 2), p2,
1325
0
                                 temp * scanline) == (tmsize_t)(-1) &&
1326
0
            img->stoponerr)
1327
0
        {
1328
0
            ret = 0;
1329
0
            break;
1330
0
        }
1331
0
        if (alpha)
1332
0
        {
1333
0
            if (TIFFReadEncodedStrip(
1334
0
                    tif, TIFFComputeStrip(tif, offset_row, colorchannels), pa,
1335
0
                    temp * scanline) == (tmsize_t)(-1) &&
1336
0
                img->stoponerr)
1337
0
            {
1338
0
                ret = 0;
1339
0
                break;
1340
0
            }
1341
0
        }
1342
1343
0
        pos = ((row + img->row_offset) % rowsperstrip) * scanline +
1344
0
              ((tmsize_t)img->col_offset * img->samplesperpixel);
1345
0
        tmsize_t roffset = (tmsize_t)y * w;
1346
0
        (*put)(img, raster + roffset, 0, y, w, nrow, fromskew, toskew, p0 + pos,
1347
0
               p1 + pos, p2 + pos, (alpha ? (pa + pos) : NULL));
1348
0
        y += ((flip & FLIP_VERTICALLY) ? -(int32_t)nrow : (int32_t)nrow);
1349
0
    }
1350
1351
0
    if (flip & FLIP_HORIZONTALLY)
1352
0
    {
1353
0
        uint32_t line;
1354
1355
0
        for (line = 0; line < h; line++)
1356
0
        {
1357
0
            uint32_t *left = raster + (line * w);
1358
0
            uint32_t *right = left + w - 1;
1359
1360
0
            while (left < right)
1361
0
            {
1362
0
                uint32_t temp = *left;
1363
0
                *left = *right;
1364
0
                *right = temp;
1365
0
                left++;
1366
0
                right--;
1367
0
            }
1368
0
        }
1369
0
    }
1370
1371
0
    _TIFFfreeExt(img->tif, buf);
1372
0
    return (ret);
1373
0
}
1374
1375
/*
1376
 * The following routines move decoded data returned
1377
 * from the TIFF library into rasters filled with packed
1378
 * ABGR pixels (i.e. suitable for passing to lrecwrite.)
1379
 *
1380
 * The routines have been created according to the most
1381
 * important cases and optimized.  PickContigCase and
1382
 * PickSeparateCase analyze the parameters and select
1383
 * the appropriate "get" and "put" routine to use.
1384
 */
1385
#define REPEAT8(op)                                                            \
1386
0
    REPEAT4(op);                                                               \
1387
0
    REPEAT4(op)
1388
#define REPEAT4(op)                                                            \
1389
0
    REPEAT2(op);                                                               \
1390
0
    REPEAT2(op)
1391
#define REPEAT2(op)                                                            \
1392
0
    op;                                                                        \
1393
0
    op
1394
#define CASE8(x, op)                                                           \
1395
0
    switch (x)                                                                 \
1396
0
    {                                                                          \
1397
0
        case 7:                                                                \
1398
0
            op; /*-fallthrough*/                                               \
1399
0
        case 6:                                                                \
1400
0
            op; /*-fallthrough*/                                               \
1401
0
        case 5:                                                                \
1402
0
            op; /*-fallthrough*/                                               \
1403
0
        case 4:                                                                \
1404
0
            op; /*-fallthrough*/                                               \
1405
0
        case 3:                                                                \
1406
0
            op; /*-fallthrough*/                                               \
1407
0
        case 2:                                                                \
1408
0
            op; /*-fallthrough*/                                               \
1409
0
        case 1:                                                                \
1410
0
            op;                                                                \
1411
0
    }
1412
#define CASE4(x, op)                                                           \
1413
0
    switch (x)                                                                 \
1414
0
    {                                                                          \
1415
0
        case 3:                                                                \
1416
0
            op; /*-fallthrough*/                                               \
1417
0
        case 2:                                                                \
1418
0
            op; /*-fallthrough*/                                               \
1419
0
        case 1:                                                                \
1420
0
            op;                                                                \
1421
0
    }
1422
#define NOP
1423
1424
#define UNROLL8(w, op1, op2)                                                   \
1425
0
    {                                                                          \
1426
0
        uint32_t _x;                                                           \
1427
0
        for (_x = w; _x >= 8; _x -= 8)                                         \
1428
0
        {                                                                      \
1429
0
            op1;                                                               \
1430
0
            REPEAT8(op2);                                                      \
1431
0
        }                                                                      \
1432
0
        if (_x > 0)                                                            \
1433
0
        {                                                                      \
1434
0
            op1;                                                               \
1435
0
            CASE8(_x, op2);                                                    \
1436
0
        }                                                                      \
1437
0
    }
1438
#define UNROLL4(w, op1, op2)                                                   \
1439
0
    {                                                                          \
1440
0
        uint32_t _x;                                                           \
1441
0
        for (_x = w; _x >= 4; _x -= 4)                                         \
1442
0
        {                                                                      \
1443
0
            op1;                                                               \
1444
0
            REPEAT4(op2);                                                      \
1445
0
        }                                                                      \
1446
0
        if (_x > 0)                                                            \
1447
0
        {                                                                      \
1448
0
            op1;                                                               \
1449
0
            CASE4(_x, op2);                                                    \
1450
0
        }                                                                      \
1451
0
    }
1452
#define UNROLL2(w, op1, op2)                                                   \
1453
0
    {                                                                          \
1454
0
        uint32_t _x;                                                           \
1455
0
        for (_x = w; _x >= 2; _x -= 2)                                         \
1456
0
        {                                                                      \
1457
0
            op1;                                                               \
1458
0
            REPEAT2(op2);                                                      \
1459
0
        }                                                                      \
1460
0
        if (_x)                                                                \
1461
0
        {                                                                      \
1462
0
            op1;                                                               \
1463
0
            op2;                                                               \
1464
0
        }                                                                      \
1465
0
    }
1466
1467
#define SKEW(r, g, b, skew)                                                    \
1468
0
    {                                                                          \
1469
0
        r += skew;                                                             \
1470
0
        g += skew;                                                             \
1471
0
        b += skew;                                                             \
1472
0
    }
1473
#define SKEW4(r, g, b, a, skew)                                                \
1474
0
    {                                                                          \
1475
0
        r += skew;                                                             \
1476
0
        g += skew;                                                             \
1477
0
        b += skew;                                                             \
1478
0
        a += skew;                                                             \
1479
0
    }
1480
1481
0
#define A1 (((uint32_t)0xffL) << 24)
1482
#define PACK(r, g, b)                                                          \
1483
0
    ((uint32_t)(r) | ((uint32_t)(g) << 8) | ((uint32_t)(b) << 16) | A1)
1484
#define PACK4(r, g, b, a)                                                      \
1485
0
    ((uint32_t)(r) | ((uint32_t)(g) << 8) | ((uint32_t)(b) << 16) |            \
1486
0
     ((uint32_t)(a) << 24))
1487
#define W2B(v) (((v) >> 8) & 0xff)
1488
/* TODO: PACKW should have be made redundant in favor of Bitdepth16To8 LUT */
1489
#define PACKW(r, g, b)                                                         \
1490
    ((uint32_t)W2B(r) | ((uint32_t)W2B(g) << 8) | ((uint32_t)W2B(b) << 16) | A1)
1491
#define PACKW4(r, g, b, a)                                                     \
1492
    ((uint32_t)W2B(r) | ((uint32_t)W2B(g) << 8) | ((uint32_t)W2B(b) << 16) |   \
1493
     ((uint32_t)W2B(a) << 24))
1494
1495
#define DECLAREContigPutFunc(name)                                             \
1496
    static void name(TIFFRGBAImage *img, uint32_t *cp, uint32_t x, uint32_t y, \
1497
                     uint32_t w, uint32_t h, int32_t fromskew, int32_t toskew, \
1498
                     unsigned char *pp)
1499
1500
/*
1501
 * 8-bit palette => colormap/RGB
1502
 */
1503
DECLAREContigPutFunc(put8bitcmaptile)
1504
0
{
1505
0
    uint32_t **PALmap = img->PALmap;
1506
0
    int samplesperpixel = img->samplesperpixel;
1507
1508
0
    (void)y;
1509
0
    for (; h > 0; --h)
1510
0
    {
1511
0
        for (x = w; x > 0; --x)
1512
0
        {
1513
0
            *cp++ = PALmap[*pp][0];
1514
0
            pp += samplesperpixel;
1515
0
        }
1516
0
        cp += toskew;
1517
0
        pp += fromskew;
1518
0
    }
1519
0
}
1520
1521
/*
1522
 * 4-bit palette => colormap/RGB
1523
 */
1524
DECLAREContigPutFunc(put4bitcmaptile)
1525
0
{
1526
0
    uint32_t **PALmap = img->PALmap;
1527
1528
0
    (void)x;
1529
0
    (void)y;
1530
0
    fromskew /= 2;
1531
0
    for (; h > 0; --h)
1532
0
    {
1533
0
        uint32_t *bw;
1534
0
        UNROLL2(w, bw = PALmap[*pp++], *cp++ = *bw++);
1535
0
        cp += toskew;
1536
0
        pp += fromskew;
1537
0
    }
1538
0
}
1539
1540
/*
1541
 * 2-bit palette => colormap/RGB
1542
 */
1543
DECLAREContigPutFunc(put2bitcmaptile)
1544
0
{
1545
0
    uint32_t **PALmap = img->PALmap;
1546
1547
0
    (void)x;
1548
0
    (void)y;
1549
0
    fromskew /= 4;
1550
0
    for (; h > 0; --h)
1551
0
    {
1552
0
        uint32_t *bw;
1553
0
        UNROLL4(w, bw = PALmap[*pp++], *cp++ = *bw++);
1554
0
        cp += toskew;
1555
0
        pp += fromskew;
1556
0
    }
1557
0
}
1558
1559
/*
1560
 * 1-bit palette => colormap/RGB
1561
 */
1562
DECLAREContigPutFunc(put1bitcmaptile)
1563
0
{
1564
0
    uint32_t **PALmap = img->PALmap;
1565
1566
0
    (void)x;
1567
0
    (void)y;
1568
0
    fromskew /= 8;
1569
0
    for (; h > 0; --h)
1570
0
    {
1571
0
        uint32_t *bw;
1572
0
        UNROLL8(w, bw = PALmap[*pp++], *cp++ = *bw++);
1573
0
        cp += toskew;
1574
0
        pp += fromskew;
1575
0
    }
1576
0
}
1577
1578
/*
1579
 * 8-bit greyscale => colormap/RGB
1580
 */
1581
DECLAREContigPutFunc(putgreytile)
1582
0
{
1583
0
    int samplesperpixel = img->samplesperpixel;
1584
0
    uint32_t **BWmap = img->BWmap;
1585
1586
0
    (void)y;
1587
0
    for (; h > 0; --h)
1588
0
    {
1589
0
        for (x = w; x > 0; --x)
1590
0
        {
1591
0
            *cp++ = BWmap[*pp][0];
1592
0
            pp += samplesperpixel;
1593
0
        }
1594
0
        cp += toskew;
1595
0
        pp += fromskew;
1596
0
    }
1597
0
}
1598
1599
/*
1600
 * 8-bit greyscale with associated alpha => colormap/RGBA
1601
 */
1602
DECLAREContigPutFunc(putagreytile)
1603
0
{
1604
0
    int samplesperpixel = img->samplesperpixel;
1605
0
    uint32_t **BWmap = img->BWmap;
1606
1607
0
    (void)y;
1608
0
    for (; h > 0; --h)
1609
0
    {
1610
0
        for (x = w; x > 0; --x)
1611
0
        {
1612
0
            *cp++ = BWmap[*pp][0] & ((uint32_t) * (pp + 1) << 24 | ~A1);
1613
0
            pp += samplesperpixel;
1614
0
        }
1615
0
        cp += toskew;
1616
0
        pp += fromskew;
1617
0
    }
1618
0
}
1619
1620
/*
1621
 * 16-bit greyscale => colormap/RGB
1622
 */
1623
DECLAREContigPutFunc(put16bitbwtile)
1624
0
{
1625
0
    int samplesperpixel = img->samplesperpixel;
1626
0
    uint32_t **BWmap = img->BWmap;
1627
1628
0
    (void)y;
1629
0
    for (; h > 0; --h)
1630
0
    {
1631
0
        uint16_t *wp = (uint16_t *)pp;
1632
1633
0
        for (x = w; x > 0; --x)
1634
0
        {
1635
            /* use high order byte of 16bit value */
1636
1637
0
            *cp++ = BWmap[*wp >> 8][0];
1638
0
            pp += 2 * samplesperpixel;
1639
0
            wp += samplesperpixel;
1640
0
        }
1641
0
        cp += toskew;
1642
0
        pp += fromskew;
1643
0
    }
1644
0
}
1645
1646
/*
1647
 * 1-bit bilevel => colormap/RGB
1648
 */
1649
DECLAREContigPutFunc(put1bitbwtile)
1650
0
{
1651
0
    uint32_t **BWmap = img->BWmap;
1652
1653
0
    (void)x;
1654
0
    (void)y;
1655
0
    fromskew /= 8;
1656
0
    for (; h > 0; --h)
1657
0
    {
1658
0
        uint32_t *bw;
1659
0
        UNROLL8(w, bw = BWmap[*pp++], *cp++ = *bw++);
1660
0
        cp += toskew;
1661
0
        pp += fromskew;
1662
0
    }
1663
0
}
1664
1665
/*
1666
 * 2-bit greyscale => colormap/RGB
1667
 */
1668
DECLAREContigPutFunc(put2bitbwtile)
1669
0
{
1670
0
    uint32_t **BWmap = img->BWmap;
1671
1672
0
    (void)x;
1673
0
    (void)y;
1674
0
    fromskew /= 4;
1675
0
    for (; h > 0; --h)
1676
0
    {
1677
0
        uint32_t *bw;
1678
0
        UNROLL4(w, bw = BWmap[*pp++], *cp++ = *bw++);
1679
0
        cp += toskew;
1680
0
        pp += fromskew;
1681
0
    }
1682
0
}
1683
1684
/*
1685
 * 4-bit greyscale => colormap/RGB
1686
 */
1687
DECLAREContigPutFunc(put4bitbwtile)
1688
0
{
1689
0
    uint32_t **BWmap = img->BWmap;
1690
1691
0
    (void)x;
1692
0
    (void)y;
1693
0
    fromskew /= 2;
1694
0
    for (; h > 0; --h)
1695
0
    {
1696
0
        uint32_t *bw;
1697
0
        UNROLL2(w, bw = BWmap[*pp++], *cp++ = *bw++);
1698
0
        cp += toskew;
1699
0
        pp += fromskew;
1700
0
    }
1701
0
}
1702
1703
/*
1704
 * 8-bit packed samples, no Map => RGB
1705
 */
1706
DECLAREContigPutFunc(putRGBcontig8bittile)
1707
0
{
1708
0
    int samplesperpixel = img->samplesperpixel;
1709
1710
0
    (void)x;
1711
0
    (void)y;
1712
0
    fromskew *= samplesperpixel;
1713
0
    for (; h > 0; --h)
1714
0
    {
1715
0
        UNROLL8(w, NOP, *cp++ = PACK(pp[0], pp[1], pp[2]);
1716
0
                pp += samplesperpixel);
1717
0
        cp += toskew;
1718
0
        pp += fromskew;
1719
0
    }
1720
0
}
1721
1722
/*
1723
 * 8-bit packed samples => RGBA w/ associated alpha
1724
 * (known to have Map == NULL)
1725
 */
1726
DECLAREContigPutFunc(putRGBAAcontig8bittile)
1727
0
{
1728
0
    int samplesperpixel = img->samplesperpixel;
1729
1730
0
    (void)x;
1731
0
    (void)y;
1732
0
    fromskew *= samplesperpixel;
1733
0
    for (; h > 0; --h)
1734
0
    {
1735
0
        UNROLL8(w, NOP, *cp++ = PACK4(pp[0], pp[1], pp[2], pp[3]);
1736
0
                pp += samplesperpixel);
1737
0
        cp += toskew;
1738
0
        pp += fromskew;
1739
0
    }
1740
0
}
1741
1742
/*
1743
 * 8-bit packed samples => RGBA w/ unassociated alpha
1744
 * (known to have Map == NULL)
1745
 */
1746
DECLAREContigPutFunc(putRGBUAcontig8bittile)
1747
0
{
1748
0
    int samplesperpixel = img->samplesperpixel;
1749
0
    (void)y;
1750
0
    fromskew *= samplesperpixel;
1751
0
    for (; h > 0; --h)
1752
0
    {
1753
0
        uint32_t r, g, b, a;
1754
0
        uint8_t *m;
1755
0
        for (x = w; x > 0; --x)
1756
0
        {
1757
0
            a = pp[3];
1758
0
            m = img->UaToAa + ((size_t)a << 8);
1759
0
            r = m[pp[0]];
1760
0
            g = m[pp[1]];
1761
0
            b = m[pp[2]];
1762
0
            *cp++ = PACK4(r, g, b, a);
1763
0
            pp += samplesperpixel;
1764
0
        }
1765
0
        cp += toskew;
1766
0
        pp += fromskew;
1767
0
    }
1768
0
}
1769
1770
/*
1771
 * 16-bit packed samples => RGB
1772
 */
1773
DECLAREContigPutFunc(putRGBcontig16bittile)
1774
0
{
1775
0
    int samplesperpixel = img->samplesperpixel;
1776
0
    uint16_t *wp = (uint16_t *)pp;
1777
0
    (void)y;
1778
0
    fromskew *= samplesperpixel;
1779
0
    for (; h > 0; --h)
1780
0
    {
1781
0
        for (x = w; x > 0; --x)
1782
0
        {
1783
0
            *cp++ = PACK(img->Bitdepth16To8[wp[0]], img->Bitdepth16To8[wp[1]],
1784
0
                         img->Bitdepth16To8[wp[2]]);
1785
0
            wp += samplesperpixel;
1786
0
        }
1787
0
        cp += toskew;
1788
0
        wp += fromskew;
1789
0
    }
1790
0
}
1791
1792
/*
1793
 * 16-bit packed samples => RGBA w/ associated alpha
1794
 * (known to have Map == NULL)
1795
 */
1796
DECLAREContigPutFunc(putRGBAAcontig16bittile)
1797
0
{
1798
0
    int samplesperpixel = img->samplesperpixel;
1799
0
    uint16_t *wp = (uint16_t *)pp;
1800
0
    (void)y;
1801
0
    fromskew *= samplesperpixel;
1802
0
    for (; h > 0; --h)
1803
0
    {
1804
0
        for (x = w; x > 0; --x)
1805
0
        {
1806
0
            *cp++ = PACK4(img->Bitdepth16To8[wp[0]], img->Bitdepth16To8[wp[1]],
1807
0
                          img->Bitdepth16To8[wp[2]], img->Bitdepth16To8[wp[3]]);
1808
0
            wp += samplesperpixel;
1809
0
        }
1810
0
        cp += toskew;
1811
0
        wp += fromskew;
1812
0
    }
1813
0
}
1814
1815
/*
1816
 * 16-bit packed samples => RGBA w/ unassociated alpha
1817
 * (known to have Map == NULL)
1818
 */
1819
DECLAREContigPutFunc(putRGBUAcontig16bittile)
1820
0
{
1821
0
    int samplesperpixel = img->samplesperpixel;
1822
0
    uint16_t *wp = (uint16_t *)pp;
1823
0
    (void)y;
1824
0
    fromskew *= samplesperpixel;
1825
0
    for (; h > 0; --h)
1826
0
    {
1827
0
        uint32_t r, g, b, a;
1828
0
        uint8_t *m;
1829
0
        for (x = w; x > 0; --x)
1830
0
        {
1831
0
            a = img->Bitdepth16To8[wp[3]];
1832
0
            m = img->UaToAa + ((size_t)a << 8);
1833
0
            r = m[img->Bitdepth16To8[wp[0]]];
1834
0
            g = m[img->Bitdepth16To8[wp[1]]];
1835
0
            b = m[img->Bitdepth16To8[wp[2]]];
1836
0
            *cp++ = PACK4(r, g, b, a);
1837
0
            wp += samplesperpixel;
1838
0
        }
1839
0
        cp += toskew;
1840
0
        wp += fromskew;
1841
0
    }
1842
0
}
1843
1844
/*
1845
 * 8-bit packed CMYK samples w/o Map => RGB
1846
 *
1847
 * NB: The conversion of CMYK->RGB is *very* crude.
1848
 */
1849
DECLAREContigPutFunc(putRGBcontig8bitCMYKtile)
1850
0
{
1851
0
    int samplesperpixel = img->samplesperpixel;
1852
0
    uint16_t r, g, b, k;
1853
1854
0
    (void)x;
1855
0
    (void)y;
1856
0
    fromskew *= samplesperpixel;
1857
0
    for (; h > 0; --h)
1858
0
    {
1859
0
        UNROLL8(w, NOP, k = 255 - pp[3]; r = (k * (255 - pp[0])) / 255;
1860
0
                g = (k * (255 - pp[1])) / 255; b = (k * (255 - pp[2])) / 255;
1861
0
                *cp++ = PACK(r, g, b); pp += samplesperpixel);
1862
0
        cp += toskew;
1863
0
        pp += fromskew;
1864
0
    }
1865
0
}
1866
1867
/*
1868
 * 8-bit packed CMYK samples w/Map => RGB
1869
 *
1870
 * NB: The conversion of CMYK->RGB is *very* crude.
1871
 */
1872
DECLAREContigPutFunc(putRGBcontig8bitCMYKMaptile)
1873
0
{
1874
0
    int samplesperpixel = img->samplesperpixel;
1875
0
    TIFFRGBValue *Map = img->Map;
1876
0
    uint16_t r, g, b, k;
1877
1878
0
    (void)y;
1879
0
    fromskew *= samplesperpixel;
1880
0
    for (; h > 0; --h)
1881
0
    {
1882
0
        for (x = w; x > 0; --x)
1883
0
        {
1884
0
            k = 255 - pp[3];
1885
0
            r = (k * (255 - pp[0])) / 255;
1886
0
            g = (k * (255 - pp[1])) / 255;
1887
0
            b = (k * (255 - pp[2])) / 255;
1888
0
            *cp++ = PACK(Map[r], Map[g], Map[b]);
1889
0
            pp += samplesperpixel;
1890
0
        }
1891
0
        pp += fromskew;
1892
0
        cp += toskew;
1893
0
    }
1894
0
}
1895
1896
#define DECLARESepPutFunc(name)                                                \
1897
    static void name(TIFFRGBAImage *img, uint32_t *cp, uint32_t x, uint32_t y, \
1898
                     uint32_t w, uint32_t h, int32_t fromskew, int32_t toskew, \
1899
                     unsigned char *r, unsigned char *g, unsigned char *b,     \
1900
                     unsigned char *a)
1901
1902
/*
1903
 * 8-bit unpacked samples => RGB
1904
 */
1905
DECLARESepPutFunc(putRGBseparate8bittile)
1906
0
{
1907
0
    (void)img;
1908
0
    (void)x;
1909
0
    (void)y;
1910
0
    (void)a;
1911
0
    for (; h > 0; --h)
1912
0
    {
1913
0
        UNROLL8(w, NOP, *cp++ = PACK(*r++, *g++, *b++));
1914
0
        SKEW(r, g, b, fromskew);
1915
0
        cp += toskew;
1916
0
    }
1917
0
}
1918
1919
/*
1920
 * 8-bit unpacked samples => RGBA w/ associated alpha
1921
 */
1922
DECLARESepPutFunc(putRGBAAseparate8bittile)
1923
0
{
1924
0
    (void)img;
1925
0
    (void)x;
1926
0
    (void)y;
1927
0
    for (; h > 0; --h)
1928
0
    {
1929
0
        UNROLL8(w, NOP, *cp++ = PACK4(*r++, *g++, *b++, *a++));
1930
0
        SKEW4(r, g, b, a, fromskew);
1931
0
        cp += toskew;
1932
0
    }
1933
0
}
1934
1935
/*
1936
 * 8-bit unpacked CMYK samples => RGBA
1937
 */
1938
DECLARESepPutFunc(putCMYKseparate8bittile)
1939
0
{
1940
0
    (void)img;
1941
0
    (void)y;
1942
0
    for (; h > 0; --h)
1943
0
    {
1944
0
        uint32_t rv, gv, bv, kv;
1945
0
        for (x = w; x > 0; --x)
1946
0
        {
1947
0
            kv = 255 - *a++;
1948
0
            rv = (kv * (255 - *r++)) / 255;
1949
0
            gv = (kv * (255 - *g++)) / 255;
1950
0
            bv = (kv * (255 - *b++)) / 255;
1951
0
            *cp++ = PACK4(rv, gv, bv, 255);
1952
0
        }
1953
0
        SKEW4(r, g, b, a, fromskew);
1954
0
        cp += toskew;
1955
0
    }
1956
0
}
1957
1958
/*
1959
 * 8-bit unpacked samples => RGBA w/ unassociated alpha
1960
 */
1961
DECLARESepPutFunc(putRGBUAseparate8bittile)
1962
0
{
1963
0
    (void)img;
1964
0
    (void)y;
1965
0
    for (; h > 0; --h)
1966
0
    {
1967
0
        uint32_t rv, gv, bv, av;
1968
0
        uint8_t *m;
1969
0
        for (x = w; x > 0; --x)
1970
0
        {
1971
0
            av = *a++;
1972
0
            m = img->UaToAa + ((size_t)av << 8);
1973
0
            rv = m[*r++];
1974
0
            gv = m[*g++];
1975
0
            bv = m[*b++];
1976
0
            *cp++ = PACK4(rv, gv, bv, av);
1977
0
        }
1978
0
        SKEW4(r, g, b, a, fromskew);
1979
0
        cp += toskew;
1980
0
    }
1981
0
}
1982
1983
/*
1984
 * 16-bit unpacked samples => RGB
1985
 */
1986
DECLARESepPutFunc(putRGBseparate16bittile)
1987
0
{
1988
0
    uint16_t *wr = (uint16_t *)r;
1989
0
    uint16_t *wg = (uint16_t *)g;
1990
0
    uint16_t *wb = (uint16_t *)b;
1991
0
    (void)img;
1992
0
    (void)y;
1993
0
    (void)a;
1994
0
    for (; h > 0; --h)
1995
0
    {
1996
0
        for (x = 0; x < w; x++)
1997
0
            *cp++ = PACK(img->Bitdepth16To8[*wr++], img->Bitdepth16To8[*wg++],
1998
0
                         img->Bitdepth16To8[*wb++]);
1999
0
        SKEW(wr, wg, wb, fromskew);
2000
0
        cp += toskew;
2001
0
    }
2002
0
}
2003
2004
/*
2005
 * 16-bit unpacked samples => RGBA w/ associated alpha
2006
 */
2007
DECLARESepPutFunc(putRGBAAseparate16bittile)
2008
0
{
2009
0
    uint16_t *wr = (uint16_t *)r;
2010
0
    uint16_t *wg = (uint16_t *)g;
2011
0
    uint16_t *wb = (uint16_t *)b;
2012
0
    uint16_t *wa = (uint16_t *)a;
2013
0
    (void)img;
2014
0
    (void)y;
2015
0
    for (; h > 0; --h)
2016
0
    {
2017
0
        for (x = 0; x < w; x++)
2018
0
            *cp++ = PACK4(img->Bitdepth16To8[*wr++], img->Bitdepth16To8[*wg++],
2019
0
                          img->Bitdepth16To8[*wb++], img->Bitdepth16To8[*wa++]);
2020
0
        SKEW4(wr, wg, wb, wa, fromskew);
2021
0
        cp += toskew;
2022
0
    }
2023
0
}
2024
2025
/*
2026
 * 16-bit unpacked samples => RGBA w/ unassociated alpha
2027
 */
2028
DECLARESepPutFunc(putRGBUAseparate16bittile)
2029
0
{
2030
0
    uint16_t *wr = (uint16_t *)r;
2031
0
    uint16_t *wg = (uint16_t *)g;
2032
0
    uint16_t *wb = (uint16_t *)b;
2033
0
    uint16_t *wa = (uint16_t *)a;
2034
0
    (void)img;
2035
0
    (void)y;
2036
0
    for (; h > 0; --h)
2037
0
    {
2038
0
        uint32_t r2, g2, b2, a2;
2039
0
        uint8_t *m;
2040
0
        for (x = w; x > 0; --x)
2041
0
        {
2042
0
            a2 = img->Bitdepth16To8[*wa++];
2043
0
            m = img->UaToAa + ((size_t)a2 << 8);
2044
0
            r2 = m[img->Bitdepth16To8[*wr++]];
2045
0
            g2 = m[img->Bitdepth16To8[*wg++]];
2046
0
            b2 = m[img->Bitdepth16To8[*wb++]];
2047
0
            *cp++ = PACK4(r2, g2, b2, a2);
2048
0
        }
2049
0
        SKEW4(wr, wg, wb, wa, fromskew);
2050
0
        cp += toskew;
2051
0
    }
2052
0
}
2053
2054
/*
2055
 * 8-bit packed CIE L*a*b 1976 samples => RGB
2056
 */
2057
DECLAREContigPutFunc(putcontig8bitCIELab8)
2058
0
{
2059
0
    float X, Y, Z;
2060
0
    uint32_t r, g, b;
2061
0
    (void)y;
2062
0
    fromskew *= 3;
2063
0
    for (; h > 0; --h)
2064
0
    {
2065
0
        for (x = w; x > 0; --x)
2066
0
        {
2067
0
            TIFFCIELabToXYZ(img->cielab, (unsigned char)pp[0],
2068
0
                            (signed char)pp[1], (signed char)pp[2], &X, &Y, &Z);
2069
0
            TIFFXYZToRGB(img->cielab, X, Y, Z, &r, &g, &b);
2070
0
            *cp++ = PACK(r, g, b);
2071
0
            pp += 3;
2072
0
        }
2073
0
        cp += toskew;
2074
0
        pp += fromskew;
2075
0
    }
2076
0
}
2077
2078
/*
2079
 * 16-bit packed CIE L*a*b 1976 samples => RGB
2080
 */
2081
DECLAREContigPutFunc(putcontig8bitCIELab16)
2082
0
{
2083
0
    float X, Y, Z;
2084
0
    uint32_t r, g, b;
2085
0
    uint16_t *wp = (uint16_t *)pp;
2086
0
    (void)y;
2087
0
    fromskew *= 3;
2088
0
    for (; h > 0; --h)
2089
0
    {
2090
0
        for (x = w; x > 0; --x)
2091
0
        {
2092
0
            TIFFCIELab16ToXYZ(img->cielab, (uint16_t)wp[0], (int16_t)wp[1],
2093
0
                              (int16_t)wp[2], &X, &Y, &Z);
2094
0
            TIFFXYZToRGB(img->cielab, X, Y, Z, &r, &g, &b);
2095
0
            *cp++ = PACK(r, g, b);
2096
0
            wp += 3;
2097
0
        }
2098
0
        cp += toskew;
2099
0
        wp += fromskew;
2100
0
    }
2101
0
}
2102
2103
/*
2104
 * YCbCr -> RGB conversion and packing routines.
2105
 */
2106
2107
#define YCbCrtoRGB(dst, Y)                                                     \
2108
0
    {                                                                          \
2109
0
        uint32_t r, g, b;                                                      \
2110
0
        TIFFYCbCrtoRGB(img->ycbcr, (Y), Cb, Cr, &r, &g, &b);                   \
2111
0
        dst = PACK(r, g, b);                                                   \
2112
0
    }
2113
2114
/*
2115
 * 8-bit packed YCbCr samples w/ 4,4 subsampling => RGB
2116
 */
2117
DECLAREContigPutFunc(putcontig8bitYCbCr44tile)
2118
0
{
2119
0
    uint32_t *cp1 = cp + w + toskew;
2120
0
    uint32_t *cp2 = cp1 + w + toskew;
2121
0
    uint32_t *cp3 = cp2 + w + toskew;
2122
0
    int32_t incr = 3 * w + 4 * toskew;
2123
2124
0
    (void)y;
2125
    /* adjust fromskew */
2126
0
    fromskew = (fromskew / 4) * (4 * 2 + 2);
2127
0
    if ((h & 3) == 0 && (w & 3) == 0)
2128
0
    {
2129
0
        for (; h >= 4; h -= 4)
2130
0
        {
2131
0
            x = w >> 2;
2132
0
            do
2133
0
            {
2134
0
                int32_t Cb = pp[16];
2135
0
                int32_t Cr = pp[17];
2136
2137
0
                YCbCrtoRGB(cp[0], pp[0]);
2138
0
                YCbCrtoRGB(cp[1], pp[1]);
2139
0
                YCbCrtoRGB(cp[2], pp[2]);
2140
0
                YCbCrtoRGB(cp[3], pp[3]);
2141
0
                YCbCrtoRGB(cp1[0], pp[4]);
2142
0
                YCbCrtoRGB(cp1[1], pp[5]);
2143
0
                YCbCrtoRGB(cp1[2], pp[6]);
2144
0
                YCbCrtoRGB(cp1[3], pp[7]);
2145
0
                YCbCrtoRGB(cp2[0], pp[8]);
2146
0
                YCbCrtoRGB(cp2[1], pp[9]);
2147
0
                YCbCrtoRGB(cp2[2], pp[10]);
2148
0
                YCbCrtoRGB(cp2[3], pp[11]);
2149
0
                YCbCrtoRGB(cp3[0], pp[12]);
2150
0
                YCbCrtoRGB(cp3[1], pp[13]);
2151
0
                YCbCrtoRGB(cp3[2], pp[14]);
2152
0
                YCbCrtoRGB(cp3[3], pp[15]);
2153
2154
0
                cp += 4;
2155
0
                cp1 += 4;
2156
0
                cp2 += 4;
2157
0
                cp3 += 4;
2158
0
                pp += 18;
2159
0
            } while (--x);
2160
0
            cp += incr;
2161
0
            cp1 += incr;
2162
0
            cp2 += incr;
2163
0
            cp3 += incr;
2164
0
            pp += fromskew;
2165
0
        }
2166
0
    }
2167
0
    else
2168
0
    {
2169
0
        while (h > 0)
2170
0
        {
2171
0
            for (x = w; x > 0;)
2172
0
            {
2173
0
                int32_t Cb = pp[16];
2174
0
                int32_t Cr = pp[17];
2175
0
                switch (x)
2176
0
                {
2177
0
                    default:
2178
0
                        switch (h)
2179
0
                        {
2180
0
                            default:
2181
0
                                YCbCrtoRGB(cp3[3], pp[15]); /* FALLTHROUGH */
2182
0
                            case 3:
2183
0
                                YCbCrtoRGB(cp2[3], pp[11]); /* FALLTHROUGH */
2184
0
                            case 2:
2185
0
                                YCbCrtoRGB(cp1[3], pp[7]); /* FALLTHROUGH */
2186
0
                            case 1:
2187
0
                                YCbCrtoRGB(cp[3], pp[3]); /* FALLTHROUGH */
2188
0
                        }                                 /* FALLTHROUGH */
2189
0
                    case 3:
2190
0
                        switch (h)
2191
0
                        {
2192
0
                            default:
2193
0
                                YCbCrtoRGB(cp3[2], pp[14]); /* FALLTHROUGH */
2194
0
                            case 3:
2195
0
                                YCbCrtoRGB(cp2[2], pp[10]); /* FALLTHROUGH */
2196
0
                            case 2:
2197
0
                                YCbCrtoRGB(cp1[2], pp[6]); /* FALLTHROUGH */
2198
0
                            case 1:
2199
0
                                YCbCrtoRGB(cp[2], pp[2]); /* FALLTHROUGH */
2200
0
                        }                                 /* FALLTHROUGH */
2201
0
                    case 2:
2202
0
                        switch (h)
2203
0
                        {
2204
0
                            default:
2205
0
                                YCbCrtoRGB(cp3[1], pp[13]); /* FALLTHROUGH */
2206
0
                            case 3:
2207
0
                                YCbCrtoRGB(cp2[1], pp[9]); /* FALLTHROUGH */
2208
0
                            case 2:
2209
0
                                YCbCrtoRGB(cp1[1], pp[5]); /* FALLTHROUGH */
2210
0
                            case 1:
2211
0
                                YCbCrtoRGB(cp[1], pp[1]); /* FALLTHROUGH */
2212
0
                        }                                 /* FALLTHROUGH */
2213
0
                    case 1:
2214
0
                        switch (h)
2215
0
                        {
2216
0
                            default:
2217
0
                                YCbCrtoRGB(cp3[0], pp[12]); /* FALLTHROUGH */
2218
0
                            case 3:
2219
0
                                YCbCrtoRGB(cp2[0], pp[8]); /* FALLTHROUGH */
2220
0
                            case 2:
2221
0
                                YCbCrtoRGB(cp1[0], pp[4]); /* FALLTHROUGH */
2222
0
                            case 1:
2223
0
                                YCbCrtoRGB(cp[0], pp[0]); /* FALLTHROUGH */
2224
0
                        }                                 /* FALLTHROUGH */
2225
0
                }
2226
0
                if (x < 4)
2227
0
                {
2228
0
                    cp += x;
2229
0
                    cp1 += x;
2230
0
                    cp2 += x;
2231
0
                    cp3 += x;
2232
0
                    x = 0;
2233
0
                }
2234
0
                else
2235
0
                {
2236
0
                    cp += 4;
2237
0
                    cp1 += 4;
2238
0
                    cp2 += 4;
2239
0
                    cp3 += 4;
2240
0
                    x -= 4;
2241
0
                }
2242
0
                pp += 18;
2243
0
            }
2244
0
            if (h <= 4)
2245
0
                break;
2246
0
            h -= 4;
2247
0
            cp += incr;
2248
0
            cp1 += incr;
2249
0
            cp2 += incr;
2250
0
            cp3 += incr;
2251
0
            pp += fromskew;
2252
0
        }
2253
0
    }
2254
0
}
2255
2256
/*
2257
 * 8-bit packed YCbCr samples w/ 4,2 subsampling => RGB
2258
 */
2259
DECLAREContigPutFunc(putcontig8bitYCbCr42tile)
2260
0
{
2261
0
    uint32_t *cp1 = cp + w + toskew;
2262
0
    int32_t incr = 2 * toskew + w;
2263
2264
0
    (void)y;
2265
0
    fromskew = (fromskew / 4) * (4 * 2 + 2);
2266
0
    if ((w & 3) == 0 && (h & 1) == 0)
2267
0
    {
2268
0
        for (; h >= 2; h -= 2)
2269
0
        {
2270
0
            x = w >> 2;
2271
0
            do
2272
0
            {
2273
0
                int32_t Cb = pp[8];
2274
0
                int32_t Cr = pp[9];
2275
2276
0
                YCbCrtoRGB(cp[0], pp[0]);
2277
0
                YCbCrtoRGB(cp[1], pp[1]);
2278
0
                YCbCrtoRGB(cp[2], pp[2]);
2279
0
                YCbCrtoRGB(cp[3], pp[3]);
2280
0
                YCbCrtoRGB(cp1[0], pp[4]);
2281
0
                YCbCrtoRGB(cp1[1], pp[5]);
2282
0
                YCbCrtoRGB(cp1[2], pp[6]);
2283
0
                YCbCrtoRGB(cp1[3], pp[7]);
2284
2285
0
                cp += 4;
2286
0
                cp1 += 4;
2287
0
                pp += 10;
2288
0
            } while (--x);
2289
0
            cp += incr;
2290
0
            cp1 += incr;
2291
0
            pp += fromskew;
2292
0
        }
2293
0
    }
2294
0
    else
2295
0
    {
2296
0
        while (h > 0)
2297
0
        {
2298
0
            for (x = w; x > 0;)
2299
0
            {
2300
0
                int32_t Cb = pp[8];
2301
0
                int32_t Cr = pp[9];
2302
0
                switch (x)
2303
0
                {
2304
0
                    default:
2305
0
                        switch (h)
2306
0
                        {
2307
0
                            default:
2308
0
                                YCbCrtoRGB(cp1[3], pp[7]); /* FALLTHROUGH */
2309
0
                            case 1:
2310
0
                                YCbCrtoRGB(cp[3], pp[3]); /* FALLTHROUGH */
2311
0
                        }                                 /* FALLTHROUGH */
2312
0
                    case 3:
2313
0
                        switch (h)
2314
0
                        {
2315
0
                            default:
2316
0
                                YCbCrtoRGB(cp1[2], pp[6]); /* FALLTHROUGH */
2317
0
                            case 1:
2318
0
                                YCbCrtoRGB(cp[2], pp[2]); /* FALLTHROUGH */
2319
0
                        }                                 /* FALLTHROUGH */
2320
0
                    case 2:
2321
0
                        switch (h)
2322
0
                        {
2323
0
                            default:
2324
0
                                YCbCrtoRGB(cp1[1], pp[5]); /* FALLTHROUGH */
2325
0
                            case 1:
2326
0
                                YCbCrtoRGB(cp[1], pp[1]); /* FALLTHROUGH */
2327
0
                        }                                 /* FALLTHROUGH */
2328
0
                    case 1:
2329
0
                        switch (h)
2330
0
                        {
2331
0
                            default:
2332
0
                                YCbCrtoRGB(cp1[0], pp[4]); /* FALLTHROUGH */
2333
0
                            case 1:
2334
0
                                YCbCrtoRGB(cp[0], pp[0]); /* FALLTHROUGH */
2335
0
                        }                                 /* FALLTHROUGH */
2336
0
                }
2337
0
                if (x < 4)
2338
0
                {
2339
0
                    cp += x;
2340
0
                    cp1 += x;
2341
0
                    x = 0;
2342
0
                }
2343
0
                else
2344
0
                {
2345
0
                    cp += 4;
2346
0
                    cp1 += 4;
2347
0
                    x -= 4;
2348
0
                }
2349
0
                pp += 10;
2350
0
            }
2351
0
            if (h <= 2)
2352
0
                break;
2353
0
            h -= 2;
2354
0
            cp += incr;
2355
0
            cp1 += incr;
2356
0
            pp += fromskew;
2357
0
        }
2358
0
    }
2359
0
}
2360
2361
/*
2362
 * 8-bit packed YCbCr samples w/ 4,1 subsampling => RGB
2363
 */
2364
DECLAREContigPutFunc(putcontig8bitYCbCr41tile)
2365
0
{
2366
0
    (void)y;
2367
0
    fromskew = (fromskew / 4) * (4 * 1 + 2);
2368
0
    do
2369
0
    {
2370
0
        x = w >> 2;
2371
0
        while (x > 0)
2372
0
        {
2373
0
            int32_t Cb = pp[4];
2374
0
            int32_t Cr = pp[5];
2375
2376
0
            YCbCrtoRGB(cp[0], pp[0]);
2377
0
            YCbCrtoRGB(cp[1], pp[1]);
2378
0
            YCbCrtoRGB(cp[2], pp[2]);
2379
0
            YCbCrtoRGB(cp[3], pp[3]);
2380
2381
0
            cp += 4;
2382
0
            pp += 6;
2383
0
            x--;
2384
0
        }
2385
2386
0
        if ((w & 3) != 0)
2387
0
        {
2388
0
            int32_t Cb = pp[4];
2389
0
            int32_t Cr = pp[5];
2390
2391
0
            switch ((w & 3))
2392
0
            {
2393
0
                case 3:
2394
0
                    YCbCrtoRGB(cp[2], pp[2]); /*-fallthrough*/
2395
0
                case 2:
2396
0
                    YCbCrtoRGB(cp[1], pp[1]); /*-fallthrough*/
2397
0
                case 1:
2398
0
                    YCbCrtoRGB(cp[0], pp[0]); /*-fallthrough*/
2399
0
                case 0:
2400
0
                    break;
2401
0
            }
2402
2403
0
            cp += (w & 3);
2404
0
            pp += 6;
2405
0
        }
2406
2407
0
        cp += toskew;
2408
0
        pp += fromskew;
2409
0
    } while (--h);
2410
0
}
2411
2412
/*
2413
 * 8-bit packed YCbCr samples w/ 2,2 subsampling => RGB
2414
 */
2415
DECLAREContigPutFunc(putcontig8bitYCbCr22tile)
2416
0
{
2417
0
    uint32_t *cp2;
2418
0
    int32_t incr = 2 * toskew + w;
2419
0
    (void)y;
2420
0
    fromskew = (fromskew / 2) * (2 * 2 + 2);
2421
0
    cp2 = cp + w + toskew;
2422
0
    while (h >= 2)
2423
0
    {
2424
0
        x = w;
2425
0
        while (x >= 2)
2426
0
        {
2427
0
            uint32_t Cb = pp[4];
2428
0
            uint32_t Cr = pp[5];
2429
0
            YCbCrtoRGB(cp[0], pp[0]);
2430
0
            YCbCrtoRGB(cp[1], pp[1]);
2431
0
            YCbCrtoRGB(cp2[0], pp[2]);
2432
0
            YCbCrtoRGB(cp2[1], pp[3]);
2433
0
            cp += 2;
2434
0
            cp2 += 2;
2435
0
            pp += 6;
2436
0
            x -= 2;
2437
0
        }
2438
0
        if (x == 1)
2439
0
        {
2440
0
            uint32_t Cb = pp[4];
2441
0
            uint32_t Cr = pp[5];
2442
0
            YCbCrtoRGB(cp[0], pp[0]);
2443
0
            YCbCrtoRGB(cp2[0], pp[2]);
2444
0
            cp++;
2445
0
            cp2++;
2446
0
            pp += 6;
2447
0
        }
2448
0
        cp += incr;
2449
0
        cp2 += incr;
2450
0
        pp += fromskew;
2451
0
        h -= 2;
2452
0
    }
2453
0
    if (h == 1)
2454
0
    {
2455
0
        x = w;
2456
0
        while (x >= 2)
2457
0
        {
2458
0
            uint32_t Cb = pp[4];
2459
0
            uint32_t Cr = pp[5];
2460
0
            YCbCrtoRGB(cp[0], pp[0]);
2461
0
            YCbCrtoRGB(cp[1], pp[1]);
2462
0
            cp += 2;
2463
0
            cp2 += 2;
2464
0
            pp += 6;
2465
0
            x -= 2;
2466
0
        }
2467
0
        if (x == 1)
2468
0
        {
2469
0
            uint32_t Cb = pp[4];
2470
0
            uint32_t Cr = pp[5];
2471
0
            YCbCrtoRGB(cp[0], pp[0]);
2472
0
        }
2473
0
    }
2474
0
}
2475
2476
/*
2477
 * 8-bit packed YCbCr samples w/ 2,1 subsampling => RGB
2478
 */
2479
DECLAREContigPutFunc(putcontig8bitYCbCr21tile)
2480
0
{
2481
0
    (void)y;
2482
0
    fromskew = (fromskew / 2) * (2 * 1 + 2);
2483
0
    do
2484
0
    {
2485
0
        x = w >> 1;
2486
0
        while (x > 0)
2487
0
        {
2488
0
            int32_t Cb = pp[2];
2489
0
            int32_t Cr = pp[3];
2490
2491
0
            YCbCrtoRGB(cp[0], pp[0]);
2492
0
            YCbCrtoRGB(cp[1], pp[1]);
2493
2494
0
            cp += 2;
2495
0
            pp += 4;
2496
0
            x--;
2497
0
        }
2498
2499
0
        if ((w & 1) != 0)
2500
0
        {
2501
0
            int32_t Cb = pp[2];
2502
0
            int32_t Cr = pp[3];
2503
2504
0
            YCbCrtoRGB(cp[0], pp[0]);
2505
2506
0
            cp += 1;
2507
0
            pp += 4;
2508
0
        }
2509
2510
0
        cp += toskew;
2511
0
        pp += fromskew;
2512
0
    } while (--h);
2513
0
}
2514
2515
/*
2516
 * 8-bit packed YCbCr samples w/ 1,2 subsampling => RGB
2517
 */
2518
DECLAREContigPutFunc(putcontig8bitYCbCr12tile)
2519
0
{
2520
0
    uint32_t *cp2;
2521
0
    int32_t incr = 2 * toskew + w;
2522
0
    (void)y;
2523
0
    fromskew = (fromskew / 1) * (1 * 2 + 2);
2524
0
    cp2 = cp + w + toskew;
2525
0
    while (h >= 2)
2526
0
    {
2527
0
        x = w;
2528
0
        do
2529
0
        {
2530
0
            uint32_t Cb = pp[2];
2531
0
            uint32_t Cr = pp[3];
2532
0
            YCbCrtoRGB(cp[0], pp[0]);
2533
0
            YCbCrtoRGB(cp2[0], pp[1]);
2534
0
            cp++;
2535
0
            cp2++;
2536
0
            pp += 4;
2537
0
        } while (--x);
2538
0
        cp += incr;
2539
0
        cp2 += incr;
2540
0
        pp += fromskew;
2541
0
        h -= 2;
2542
0
    }
2543
0
    if (h == 1)
2544
0
    {
2545
0
        x = w;
2546
0
        do
2547
0
        {
2548
0
            uint32_t Cb = pp[2];
2549
0
            uint32_t Cr = pp[3];
2550
0
            YCbCrtoRGB(cp[0], pp[0]);
2551
0
            cp++;
2552
0
            pp += 4;
2553
0
        } while (--x);
2554
0
    }
2555
0
}
2556
2557
/*
2558
 * 8-bit packed YCbCr samples w/ no subsampling => RGB
2559
 */
2560
DECLAREContigPutFunc(putcontig8bitYCbCr11tile)
2561
0
{
2562
0
    (void)y;
2563
0
    fromskew = (fromskew / 1) * (1 * 1 + 2);
2564
0
    do
2565
0
    {
2566
0
        x = w; /* was x = w>>1; patched 2000/09/25 warmerda@home.com */
2567
0
        do
2568
0
        {
2569
0
            int32_t Cb = pp[1];
2570
0
            int32_t Cr = pp[2];
2571
2572
0
            YCbCrtoRGB(*cp++, pp[0]);
2573
2574
0
            pp += 3;
2575
0
        } while (--x);
2576
0
        cp += toskew;
2577
0
        pp += fromskew;
2578
0
    } while (--h);
2579
0
}
2580
2581
/*
2582
 * 8-bit packed YCbCr samples w/ no subsampling => RGB
2583
 */
2584
DECLARESepPutFunc(putseparate8bitYCbCr11tile)
2585
0
{
2586
0
    (void)y;
2587
0
    (void)a;
2588
    /* TODO: naming of input vars is still off, change obfuscating declaration
2589
     * inside define, or resolve obfuscation */
2590
0
    for (; h > 0; --h)
2591
0
    {
2592
0
        x = w;
2593
0
        do
2594
0
        {
2595
0
            uint32_t dr, dg, db;
2596
0
            TIFFYCbCrtoRGB(img->ycbcr, *r++, *g++, *b++, &dr, &dg, &db);
2597
0
            *cp++ = PACK(dr, dg, db);
2598
0
        } while (--x);
2599
0
        SKEW(r, g, b, fromskew);
2600
0
        cp += toskew;
2601
0
    }
2602
0
}
2603
#undef YCbCrtoRGB
2604
2605
static int isInRefBlackWhiteRange(float f)
2606
0
{
2607
0
    return f > (float)(-0x7FFFFFFF + 128) && f < (float)0x7FFFFFFF;
2608
0
}
2609
2610
static int initYCbCrConversion(TIFFRGBAImage *img)
2611
0
{
2612
0
    static const char module[] = "initYCbCrConversion";
2613
2614
0
    float *luma, *refBlackWhite;
2615
2616
0
    if (img->ycbcr == NULL)
2617
0
    {
2618
0
        img->ycbcr = (TIFFYCbCrToRGB *)_TIFFmallocExt(
2619
0
            img->tif, TIFFroundup_32(sizeof(TIFFYCbCrToRGB), sizeof(long)) +
2620
0
                          4 * 256 * sizeof(TIFFRGBValue) +
2621
0
                          2 * 256 * sizeof(int) + 3 * 256 * sizeof(int32_t));
2622
0
        if (img->ycbcr == NULL)
2623
0
        {
2624
0
            TIFFErrorExtR(img->tif, module,
2625
0
                          "No space for YCbCr->RGB conversion state");
2626
0
            return (0);
2627
0
        }
2628
0
    }
2629
2630
0
    TIFFGetFieldDefaulted(img->tif, TIFFTAG_YCBCRCOEFFICIENTS, &luma);
2631
0
    TIFFGetFieldDefaulted(img->tif, TIFFTAG_REFERENCEBLACKWHITE,
2632
0
                          &refBlackWhite);
2633
2634
    /* Do some validation to avoid later issues. Detect NaN for now */
2635
    /* and also if lumaGreen is zero since we divide by it later */
2636
0
    if (luma[0] != luma[0] || luma[1] != luma[1] || luma[1] == 0.0 ||
2637
0
        luma[2] != luma[2])
2638
0
    {
2639
0
        TIFFErrorExtR(img->tif, module,
2640
0
                      "Invalid values for YCbCrCoefficients tag");
2641
0
        return (0);
2642
0
    }
2643
2644
0
    if (!isInRefBlackWhiteRange(refBlackWhite[0]) ||
2645
0
        !isInRefBlackWhiteRange(refBlackWhite[1]) ||
2646
0
        !isInRefBlackWhiteRange(refBlackWhite[2]) ||
2647
0
        !isInRefBlackWhiteRange(refBlackWhite[3]) ||
2648
0
        !isInRefBlackWhiteRange(refBlackWhite[4]) ||
2649
0
        !isInRefBlackWhiteRange(refBlackWhite[5]))
2650
0
    {
2651
0
        TIFFErrorExtR(img->tif, module,
2652
0
                      "Invalid values for ReferenceBlackWhite tag");
2653
0
        return (0);
2654
0
    }
2655
2656
0
    if (TIFFYCbCrToRGBInit(img->ycbcr, luma, refBlackWhite) < 0)
2657
0
        return (0);
2658
0
    return (1);
2659
0
}
2660
2661
static tileContigRoutine initCIELabConversion(TIFFRGBAImage *img)
2662
0
{
2663
0
    static const char module[] = "initCIELabConversion";
2664
2665
0
    float *whitePoint;
2666
0
    float refWhite[3];
2667
2668
0
    TIFFGetFieldDefaulted(img->tif, TIFFTAG_WHITEPOINT, &whitePoint);
2669
0
    if (whitePoint[1] == 0.0f)
2670
0
    {
2671
0
        TIFFErrorExtR(img->tif, module, "Invalid value for WhitePoint tag.");
2672
0
        return NULL;
2673
0
    }
2674
2675
0
    if (!img->cielab)
2676
0
    {
2677
0
        img->cielab = (TIFFCIELabToRGB *)_TIFFmallocExt(
2678
0
            img->tif, sizeof(TIFFCIELabToRGB));
2679
0
        if (!img->cielab)
2680
0
        {
2681
0
            TIFFErrorExtR(img->tif, module,
2682
0
                          "No space for CIE L*a*b*->RGB conversion state.");
2683
0
            return NULL;
2684
0
        }
2685
0
    }
2686
2687
0
    refWhite[1] = 100.0F;
2688
0
    refWhite[0] = whitePoint[0] / whitePoint[1] * refWhite[1];
2689
0
    refWhite[2] =
2690
0
        (1.0F - whitePoint[0] - whitePoint[1]) / whitePoint[1] * refWhite[1];
2691
0
    if (TIFFCIELabToRGBInit(img->cielab, &display_sRGB, refWhite) < 0)
2692
0
    {
2693
0
        TIFFErrorExtR(img->tif, module,
2694
0
                      "Failed to initialize CIE L*a*b*->RGB conversion state.");
2695
0
        _TIFFfreeExt(img->tif, img->cielab);
2696
0
        return NULL;
2697
0
    }
2698
2699
0
    if (img->bitspersample == 8)
2700
0
        return putcontig8bitCIELab8;
2701
0
    else if (img->bitspersample == 16)
2702
0
        return putcontig8bitCIELab16;
2703
0
    return NULL;
2704
0
}
2705
2706
/*
2707
 * Greyscale images with less than 8 bits/sample are handled
2708
 * with a table to avoid lots of shifts and masks.  The table
2709
 * is setup so that put*bwtile (below) can retrieve 8/bitspersample
2710
 * pixel values simply by indexing into the table with one
2711
 * number.
2712
 */
2713
static int makebwmap(TIFFRGBAImage *img)
2714
0
{
2715
0
    TIFFRGBValue *Map = img->Map;
2716
0
    int bitspersample = img->bitspersample;
2717
0
    int nsamples = 8 / bitspersample;
2718
0
    int i;
2719
0
    uint32_t *p;
2720
2721
0
    if (nsamples == 0)
2722
0
        nsamples = 1;
2723
2724
0
    img->BWmap = (uint32_t **)_TIFFmallocExt(
2725
0
        img->tif,
2726
0
        256 * sizeof(uint32_t *) + (256 * nsamples * sizeof(uint32_t)));
2727
0
    if (img->BWmap == NULL)
2728
0
    {
2729
0
        TIFFErrorExtR(img->tif, TIFFFileName(img->tif),
2730
0
                      "No space for B&W mapping table");
2731
0
        return (0);
2732
0
    }
2733
0
    p = (uint32_t *)(img->BWmap + 256);
2734
0
    for (i = 0; i < 256; i++)
2735
0
    {
2736
0
        TIFFRGBValue c;
2737
0
        img->BWmap[i] = p;
2738
0
        switch (bitspersample)
2739
0
        {
2740
0
#define GREY(x)                                                                \
2741
0
    c = Map[x];                                                                \
2742
0
    *p++ = PACK(c, c, c);
2743
0
            case 1:
2744
0
                GREY(i >> 7);
2745
0
                GREY((i >> 6) & 1);
2746
0
                GREY((i >> 5) & 1);
2747
0
                GREY((i >> 4) & 1);
2748
0
                GREY((i >> 3) & 1);
2749
0
                GREY((i >> 2) & 1);
2750
0
                GREY((i >> 1) & 1);
2751
0
                GREY(i & 1);
2752
0
                break;
2753
0
            case 2:
2754
0
                GREY(i >> 6);
2755
0
                GREY((i >> 4) & 3);
2756
0
                GREY((i >> 2) & 3);
2757
0
                GREY(i & 3);
2758
0
                break;
2759
0
            case 4:
2760
0
                GREY(i >> 4);
2761
0
                GREY(i & 0xf);
2762
0
                break;
2763
0
            case 8:
2764
0
            case 16:
2765
0
                GREY(i);
2766
0
                break;
2767
0
        }
2768
0
#undef GREY
2769
0
    }
2770
0
    return (1);
2771
0
}
2772
2773
/*
2774
 * Construct a mapping table to convert from the range
2775
 * of the data samples to [0,255] --for display.  This
2776
 * process also handles inverting B&W images when needed.
2777
 */
2778
static int setupMap(TIFFRGBAImage *img)
2779
0
{
2780
0
    int32_t x, range;
2781
2782
0
    range = (int32_t)((1L << img->bitspersample) - 1);
2783
2784
    /* treat 16 bit the same as eight bit */
2785
0
    if (img->bitspersample == 16)
2786
0
        range = (int32_t)255;
2787
2788
0
    img->Map = (TIFFRGBValue *)_TIFFmallocExt(
2789
0
        img->tif, (range + 1) * sizeof(TIFFRGBValue));
2790
0
    if (img->Map == NULL)
2791
0
    {
2792
0
        TIFFErrorExtR(img->tif, TIFFFileName(img->tif),
2793
0
                      "No space for photometric conversion table");
2794
0
        return (0);
2795
0
    }
2796
0
    if (img->photometric == PHOTOMETRIC_MINISWHITE)
2797
0
    {
2798
0
        for (x = 0; x <= range; x++)
2799
0
            img->Map[x] = (TIFFRGBValue)(((range - x) * 255) / range);
2800
0
    }
2801
0
    else
2802
0
    {
2803
0
        for (x = 0; x <= range; x++)
2804
0
            img->Map[x] = (TIFFRGBValue)((x * 255) / range);
2805
0
    }
2806
0
    if (img->bitspersample <= 16 &&
2807
0
        (img->photometric == PHOTOMETRIC_MINISBLACK ||
2808
0
         img->photometric == PHOTOMETRIC_MINISWHITE))
2809
0
    {
2810
        /*
2811
         * Use photometric mapping table to construct
2812
         * unpacking tables for samples <= 8 bits.
2813
         */
2814
0
        if (!makebwmap(img))
2815
0
            return (0);
2816
        /* no longer need Map, free it */
2817
0
        _TIFFfreeExt(img->tif, img->Map);
2818
0
        img->Map = NULL;
2819
0
    }
2820
0
    return (1);
2821
0
}
2822
2823
static int checkcmap(TIFFRGBAImage *img)
2824
0
{
2825
0
    uint16_t *r = img->redcmap;
2826
0
    uint16_t *g = img->greencmap;
2827
0
    uint16_t *b = img->bluecmap;
2828
0
    long n = 1L << img->bitspersample;
2829
2830
0
    while (n-- > 0)
2831
0
        if (*r++ >= 256 || *g++ >= 256 || *b++ >= 256)
2832
0
            return (16);
2833
0
    return (8);
2834
0
}
2835
2836
static void cvtcmap(TIFFRGBAImage *img)
2837
0
{
2838
0
    uint16_t *r = img->redcmap;
2839
0
    uint16_t *g = img->greencmap;
2840
0
    uint16_t *b = img->bluecmap;
2841
0
    long i;
2842
2843
0
    for (i = (1L << img->bitspersample) - 1; i >= 0; i--)
2844
0
    {
2845
0
#define CVT(x) ((uint16_t)((x) >> 8))
2846
0
        r[i] = CVT(r[i]);
2847
0
        g[i] = CVT(g[i]);
2848
0
        b[i] = CVT(b[i]);
2849
0
#undef CVT
2850
0
    }
2851
0
}
2852
2853
/*
2854
 * Palette images with <= 8 bits/sample are handled
2855
 * with a table to avoid lots of shifts and masks.  The table
2856
 * is setup so that put*cmaptile (below) can retrieve 8/bitspersample
2857
 * pixel values simply by indexing into the table with one
2858
 * number.
2859
 */
2860
static int makecmap(TIFFRGBAImage *img)
2861
0
{
2862
0
    int bitspersample = img->bitspersample;
2863
0
    int nsamples = 8 / bitspersample;
2864
0
    uint16_t *r = img->redcmap;
2865
0
    uint16_t *g = img->greencmap;
2866
0
    uint16_t *b = img->bluecmap;
2867
0
    uint32_t *p;
2868
0
    int i;
2869
2870
0
    img->PALmap = (uint32_t **)_TIFFmallocExt(
2871
0
        img->tif,
2872
0
        256 * sizeof(uint32_t *) + (256 * nsamples * sizeof(uint32_t)));
2873
0
    if (img->PALmap == NULL)
2874
0
    {
2875
0
        TIFFErrorExtR(img->tif, TIFFFileName(img->tif),
2876
0
                      "No space for Palette mapping table");
2877
0
        return (0);
2878
0
    }
2879
0
    p = (uint32_t *)(img->PALmap + 256);
2880
0
    for (i = 0; i < 256; i++)
2881
0
    {
2882
0
        TIFFRGBValue c;
2883
0
        img->PALmap[i] = p;
2884
0
#define CMAP(x)                                                                \
2885
0
    c = (TIFFRGBValue)x;                                                       \
2886
0
    *p++ = PACK(r[c] & 0xff, g[c] & 0xff, b[c] & 0xff);
2887
0
        switch (bitspersample)
2888
0
        {
2889
0
            case 1:
2890
0
                CMAP(i >> 7);
2891
0
                CMAP((i >> 6) & 1);
2892
0
                CMAP((i >> 5) & 1);
2893
0
                CMAP((i >> 4) & 1);
2894
0
                CMAP((i >> 3) & 1);
2895
0
                CMAP((i >> 2) & 1);
2896
0
                CMAP((i >> 1) & 1);
2897
0
                CMAP(i & 1);
2898
0
                break;
2899
0
            case 2:
2900
0
                CMAP(i >> 6);
2901
0
                CMAP((i >> 4) & 3);
2902
0
                CMAP((i >> 2) & 3);
2903
0
                CMAP(i & 3);
2904
0
                break;
2905
0
            case 4:
2906
0
                CMAP(i >> 4);
2907
0
                CMAP(i & 0xf);
2908
0
                break;
2909
0
            case 8:
2910
0
                CMAP(i);
2911
0
                break;
2912
0
        }
2913
0
#undef CMAP
2914
0
    }
2915
0
    return (1);
2916
0
}
2917
2918
/*
2919
 * Construct any mapping table used
2920
 * by the associated put routine.
2921
 */
2922
static int buildMap(TIFFRGBAImage *img)
2923
0
{
2924
0
    switch (img->photometric)
2925
0
    {
2926
0
        case PHOTOMETRIC_RGB:
2927
0
        case PHOTOMETRIC_YCBCR:
2928
0
        case PHOTOMETRIC_SEPARATED:
2929
0
            if (img->bitspersample == 8)
2930
0
                break;
2931
            /* fall through... */
2932
0
        case PHOTOMETRIC_MINISBLACK:
2933
0
        case PHOTOMETRIC_MINISWHITE:
2934
0
            if (!setupMap(img))
2935
0
                return (0);
2936
0
            break;
2937
0
        case PHOTOMETRIC_PALETTE:
2938
            /*
2939
             * Convert 16-bit colormap to 8-bit (unless it looks
2940
             * like an old-style 8-bit colormap).
2941
             */
2942
0
            if (checkcmap(img) == 16)
2943
0
                cvtcmap(img);
2944
0
            else
2945
0
                TIFFWarningExtR(img->tif, TIFFFileName(img->tif),
2946
0
                                "Assuming 8-bit colormap");
2947
            /*
2948
             * Use mapping table and colormap to construct
2949
             * unpacking tables for samples < 8 bits.
2950
             */
2951
0
            if (img->bitspersample <= 8 && !makecmap(img))
2952
0
                return (0);
2953
0
            break;
2954
0
    }
2955
0
    return (1);
2956
0
}
2957
2958
/*
2959
 * Select the appropriate conversion routine for packed data.
2960
 */
2961
static int PickContigCase(TIFFRGBAImage *img)
2962
0
{
2963
0
    img->get = TIFFIsTiled(img->tif) ? gtTileContig : gtStripContig;
2964
0
    img->put.contig = NULL;
2965
0
    switch (img->photometric)
2966
0
    {
2967
0
        case PHOTOMETRIC_RGB:
2968
0
            switch (img->bitspersample)
2969
0
            {
2970
0
                case 8:
2971
0
                    if (img->alpha == EXTRASAMPLE_ASSOCALPHA &&
2972
0
                        img->samplesperpixel >= 4)
2973
0
                        img->put.contig = putRGBAAcontig8bittile;
2974
0
                    else if (img->alpha == EXTRASAMPLE_UNASSALPHA &&
2975
0
                             img->samplesperpixel >= 4)
2976
0
                    {
2977
0
                        if (BuildMapUaToAa(img))
2978
0
                            img->put.contig = putRGBUAcontig8bittile;
2979
0
                    }
2980
0
                    else if (img->samplesperpixel >= 3)
2981
0
                        img->put.contig = putRGBcontig8bittile;
2982
0
                    break;
2983
0
                case 16:
2984
0
                    if (img->alpha == EXTRASAMPLE_ASSOCALPHA &&
2985
0
                        img->samplesperpixel >= 4)
2986
0
                    {
2987
0
                        if (BuildMapBitdepth16To8(img))
2988
0
                            img->put.contig = putRGBAAcontig16bittile;
2989
0
                    }
2990
0
                    else if (img->alpha == EXTRASAMPLE_UNASSALPHA &&
2991
0
                             img->samplesperpixel >= 4)
2992
0
                    {
2993
0
                        if (BuildMapBitdepth16To8(img) && BuildMapUaToAa(img))
2994
0
                            img->put.contig = putRGBUAcontig16bittile;
2995
0
                    }
2996
0
                    else if (img->samplesperpixel >= 3)
2997
0
                    {
2998
0
                        if (BuildMapBitdepth16To8(img))
2999
0
                            img->put.contig = putRGBcontig16bittile;
3000
0
                    }
3001
0
                    break;
3002
0
            }
3003
0
            break;
3004
0
        case PHOTOMETRIC_SEPARATED:
3005
0
            if (img->samplesperpixel >= 4 && buildMap(img))
3006
0
            {
3007
0
                if (img->bitspersample == 8)
3008
0
                {
3009
0
                    if (!img->Map)
3010
0
                        img->put.contig = putRGBcontig8bitCMYKtile;
3011
0
                    else
3012
0
                        img->put.contig = putRGBcontig8bitCMYKMaptile;
3013
0
                }
3014
0
            }
3015
0
            break;
3016
0
        case PHOTOMETRIC_PALETTE:
3017
0
            if (buildMap(img))
3018
0
            {
3019
0
                switch (img->bitspersample)
3020
0
                {
3021
0
                    case 8:
3022
0
                        img->put.contig = put8bitcmaptile;
3023
0
                        break;
3024
0
                    case 4:
3025
0
                        img->put.contig = put4bitcmaptile;
3026
0
                        break;
3027
0
                    case 2:
3028
0
                        img->put.contig = put2bitcmaptile;
3029
0
                        break;
3030
0
                    case 1:
3031
0
                        img->put.contig = put1bitcmaptile;
3032
0
                        break;
3033
0
                }
3034
0
            }
3035
0
            break;
3036
0
        case PHOTOMETRIC_MINISWHITE:
3037
0
        case PHOTOMETRIC_MINISBLACK:
3038
0
            if (buildMap(img))
3039
0
            {
3040
0
                switch (img->bitspersample)
3041
0
                {
3042
0
                    case 16:
3043
0
                        img->put.contig = put16bitbwtile;
3044
0
                        break;
3045
0
                    case 8:
3046
0
                        if (img->alpha && img->samplesperpixel == 2)
3047
0
                            img->put.contig = putagreytile;
3048
0
                        else
3049
0
                            img->put.contig = putgreytile;
3050
0
                        break;
3051
0
                    case 4:
3052
0
                        img->put.contig = put4bitbwtile;
3053
0
                        break;
3054
0
                    case 2:
3055
0
                        img->put.contig = put2bitbwtile;
3056
0
                        break;
3057
0
                    case 1:
3058
0
                        img->put.contig = put1bitbwtile;
3059
0
                        break;
3060
0
                }
3061
0
            }
3062
0
            break;
3063
0
        case PHOTOMETRIC_YCBCR:
3064
0
            if ((img->bitspersample == 8) && (img->samplesperpixel == 3))
3065
0
            {
3066
0
                if (initYCbCrConversion(img) != 0)
3067
0
                {
3068
                    /*
3069
                     * The 6.0 spec says that subsampling must be
3070
                     * one of 1, 2, or 4, and that vertical subsampling
3071
                     * must always be <= horizontal subsampling; so
3072
                     * there are only a few possibilities and we just
3073
                     * enumerate the cases.
3074
                     * Joris: added support for the [1,2] case, nonetheless, to
3075
                     * accommodate some OJPEG files
3076
                     */
3077
0
                    uint16_t SubsamplingHor;
3078
0
                    uint16_t SubsamplingVer;
3079
0
                    TIFFGetFieldDefaulted(img->tif, TIFFTAG_YCBCRSUBSAMPLING,
3080
0
                                          &SubsamplingHor, &SubsamplingVer);
3081
0
                    switch ((SubsamplingHor << 4) | SubsamplingVer)
3082
0
                    {
3083
0
                        case 0x44:
3084
0
                            img->put.contig = putcontig8bitYCbCr44tile;
3085
0
                            break;
3086
0
                        case 0x42:
3087
0
                            img->put.contig = putcontig8bitYCbCr42tile;
3088
0
                            break;
3089
0
                        case 0x41:
3090
0
                            img->put.contig = putcontig8bitYCbCr41tile;
3091
0
                            break;
3092
0
                        case 0x22:
3093
0
                            img->put.contig = putcontig8bitYCbCr22tile;
3094
0
                            break;
3095
0
                        case 0x21:
3096
0
                            img->put.contig = putcontig8bitYCbCr21tile;
3097
0
                            break;
3098
0
                        case 0x12:
3099
0
                            img->put.contig = putcontig8bitYCbCr12tile;
3100
0
                            break;
3101
0
                        case 0x11:
3102
0
                            img->put.contig = putcontig8bitYCbCr11tile;
3103
0
                            break;
3104
0
                    }
3105
0
                }
3106
0
            }
3107
0
            break;
3108
0
        case PHOTOMETRIC_CIELAB:
3109
0
            if (img->samplesperpixel == 3 && buildMap(img))
3110
0
            {
3111
0
                if (img->bitspersample == 8 || img->bitspersample == 16)
3112
0
                    img->put.contig = initCIELabConversion(img);
3113
0
                break;
3114
0
            }
3115
0
    }
3116
0
    return ((img->get != NULL) && (img->put.contig != NULL));
3117
0
}
3118
3119
/*
3120
 * Select the appropriate conversion routine for unpacked data.
3121
 *
3122
 * NB: we assume that unpacked single channel data is directed
3123
 *   to the "packed routines.
3124
 */
3125
static int PickSeparateCase(TIFFRGBAImage *img)
3126
0
{
3127
0
    img->get = TIFFIsTiled(img->tif) ? gtTileSeparate : gtStripSeparate;
3128
0
    img->put.separate = NULL;
3129
0
    switch (img->photometric)
3130
0
    {
3131
0
        case PHOTOMETRIC_MINISWHITE:
3132
0
        case PHOTOMETRIC_MINISBLACK:
3133
            /* greyscale images processed pretty much as RGB by gtTileSeparate
3134
             */
3135
0
        case PHOTOMETRIC_RGB:
3136
0
            switch (img->bitspersample)
3137
0
            {
3138
0
                case 8:
3139
0
                    if (img->alpha == EXTRASAMPLE_ASSOCALPHA)
3140
0
                        img->put.separate = putRGBAAseparate8bittile;
3141
0
                    else if (img->alpha == EXTRASAMPLE_UNASSALPHA)
3142
0
                    {
3143
0
                        if (BuildMapUaToAa(img))
3144
0
                            img->put.separate = putRGBUAseparate8bittile;
3145
0
                    }
3146
0
                    else
3147
0
                        img->put.separate = putRGBseparate8bittile;
3148
0
                    break;
3149
0
                case 16:
3150
0
                    if (img->alpha == EXTRASAMPLE_ASSOCALPHA)
3151
0
                    {
3152
0
                        if (BuildMapBitdepth16To8(img))
3153
0
                            img->put.separate = putRGBAAseparate16bittile;
3154
0
                    }
3155
0
                    else if (img->alpha == EXTRASAMPLE_UNASSALPHA)
3156
0
                    {
3157
0
                        if (BuildMapBitdepth16To8(img) && BuildMapUaToAa(img))
3158
0
                            img->put.separate = putRGBUAseparate16bittile;
3159
0
                    }
3160
0
                    else
3161
0
                    {
3162
0
                        if (BuildMapBitdepth16To8(img))
3163
0
                            img->put.separate = putRGBseparate16bittile;
3164
0
                    }
3165
0
                    break;
3166
0
            }
3167
0
            break;
3168
0
        case PHOTOMETRIC_SEPARATED:
3169
0
            if (img->bitspersample == 8 && img->samplesperpixel == 4)
3170
0
            {
3171
0
                img->alpha =
3172
0
                    1; // Not alpha, but seems like the only way to get 4th band
3173
0
                img->put.separate = putCMYKseparate8bittile;
3174
0
            }
3175
0
            break;
3176
0
        case PHOTOMETRIC_YCBCR:
3177
0
            if ((img->bitspersample == 8) && (img->samplesperpixel == 3))
3178
0
            {
3179
0
                if (initYCbCrConversion(img) != 0)
3180
0
                {
3181
0
                    uint16_t hs, vs;
3182
0
                    TIFFGetFieldDefaulted(img->tif, TIFFTAG_YCBCRSUBSAMPLING,
3183
0
                                          &hs, &vs);
3184
0
                    switch ((hs << 4) | vs)
3185
0
                    {
3186
0
                        case 0x11:
3187
0
                            img->put.separate = putseparate8bitYCbCr11tile;
3188
0
                            break;
3189
                            /* TODO: add other cases here */
3190
0
                    }
3191
0
                }
3192
0
            }
3193
0
            break;
3194
0
    }
3195
0
    return ((img->get != NULL) && (img->put.separate != NULL));
3196
0
}
3197
3198
static int BuildMapUaToAa(TIFFRGBAImage *img)
3199
0
{
3200
0
    static const char module[] = "BuildMapUaToAa";
3201
0
    uint8_t *m;
3202
0
    uint16_t na, nv;
3203
0
    assert(img->UaToAa == NULL);
3204
0
    img->UaToAa = _TIFFmallocExt(img->tif, 65536);
3205
0
    if (img->UaToAa == NULL)
3206
0
    {
3207
0
        TIFFErrorExtR(img->tif, module, "Out of memory");
3208
0
        return (0);
3209
0
    }
3210
0
    m = img->UaToAa;
3211
0
    for (na = 0; na < 256; na++)
3212
0
    {
3213
0
        for (nv = 0; nv < 256; nv++)
3214
0
            *m++ = (uint8_t)((nv * na + 127) / 255);
3215
0
    }
3216
0
    return (1);
3217
0
}
3218
3219
static int BuildMapBitdepth16To8(TIFFRGBAImage *img)
3220
0
{
3221
0
    static const char module[] = "BuildMapBitdepth16To8";
3222
0
    uint8_t *m;
3223
0
    uint32_t n;
3224
0
    assert(img->Bitdepth16To8 == NULL);
3225
0
    img->Bitdepth16To8 = _TIFFmallocExt(img->tif, 65536);
3226
0
    if (img->Bitdepth16To8 == NULL)
3227
0
    {
3228
0
        TIFFErrorExtR(img->tif, module, "Out of memory");
3229
0
        return (0);
3230
0
    }
3231
0
    m = img->Bitdepth16To8;
3232
0
    for (n = 0; n < 65536; n++)
3233
0
        *m++ = (uint8_t)((n + 128) / 257);
3234
0
    return (1);
3235
0
}
3236
3237
/*
3238
 * Read a whole strip off data from the file, and convert to RGBA form.
3239
 * If this is the last strip, then it will only contain the portion of
3240
 * the strip that is actually within the image space.  The result is
3241
 * organized in bottom to top form.
3242
 */
3243
3244
int TIFFReadRGBAStrip(TIFF *tif, uint32_t row, uint32_t *raster)
3245
3246
0
{
3247
0
    return TIFFReadRGBAStripExt(tif, row, raster, 0);
3248
0
}
3249
3250
int TIFFReadRGBAStripExt(TIFF *tif, uint32_t row, uint32_t *raster,
3251
                         int stop_on_error)
3252
3253
0
{
3254
0
    char emsg[EMSG_BUF_SIZE] = "";
3255
0
    TIFFRGBAImage img;
3256
0
    int ok;
3257
0
    uint32_t rowsperstrip, rows_to_read;
3258
3259
0
    if (TIFFIsTiled(tif))
3260
0
    {
3261
0
        TIFFErrorExtR(tif, TIFFFileName(tif),
3262
0
                      "Can't use TIFFReadRGBAStrip() with tiled file.");
3263
0
        return (0);
3264
0
    }
3265
3266
0
    TIFFGetFieldDefaulted(tif, TIFFTAG_ROWSPERSTRIP, &rowsperstrip);
3267
3268
0
    if (rowsperstrip == 0)
3269
0
    {
3270
0
        TIFFErrorExtR(tif, TIFFFileName(tif), "rowsperstrip is zero");
3271
0
        return (0);
3272
0
    }
3273
3274
0
    if ((row % rowsperstrip) != 0)
3275
0
    {
3276
0
        TIFFErrorExtR(
3277
0
            tif, TIFFFileName(tif),
3278
0
            "Row passed to TIFFReadRGBAStrip() must be first in a strip.");
3279
0
        return (0);
3280
0
    }
3281
3282
0
    if (TIFFRGBAImageBegin(&img, tif, stop_on_error, emsg))
3283
0
    {
3284
0
        if (row >= img.height)
3285
0
        {
3286
0
            TIFFErrorExtR(tif, TIFFFileName(tif),
3287
0
                          "Invalid row passed to TIFFReadRGBAStrip().");
3288
0
            TIFFRGBAImageEnd(&img);
3289
0
            return (0);
3290
0
        }
3291
3292
0
        img.row_offset = row;
3293
0
        img.col_offset = 0;
3294
3295
0
        if (row + rowsperstrip > img.height)
3296
0
            rows_to_read = img.height - row;
3297
0
        else
3298
0
            rows_to_read = rowsperstrip;
3299
3300
0
        ok = TIFFRGBAImageGet(&img, raster, img.width, rows_to_read);
3301
3302
0
        TIFFRGBAImageEnd(&img);
3303
0
    }
3304
0
    else
3305
0
    {
3306
0
        TIFFErrorExtR(tif, TIFFFileName(tif), "%s", emsg);
3307
0
        ok = 0;
3308
0
    }
3309
3310
0
    return (ok);
3311
0
}
3312
3313
/*
3314
 * Read a whole tile off data from the file, and convert to RGBA form.
3315
 * The returned RGBA data is organized from bottom to top of tile,
3316
 * and may include zeroed areas if the tile extends off the image.
3317
 */
3318
3319
int TIFFReadRGBATile(TIFF *tif, uint32_t col, uint32_t row, uint32_t *raster)
3320
3321
0
{
3322
0
    return TIFFReadRGBATileExt(tif, col, row, raster, 0);
3323
0
}
3324
3325
int TIFFReadRGBATileExt(TIFF *tif, uint32_t col, uint32_t row, uint32_t *raster,
3326
                        int stop_on_error)
3327
0
{
3328
0
    char emsg[EMSG_BUF_SIZE] = "";
3329
0
    TIFFRGBAImage img;
3330
0
    int ok;
3331
0
    uint32_t tile_xsize, tile_ysize;
3332
0
    uint32_t read_xsize, read_ysize;
3333
0
    uint32_t i_row;
3334
3335
    /*
3336
     * Verify that our request is legal - on a tile file, and on a
3337
     * tile boundary.
3338
     */
3339
3340
0
    if (!TIFFIsTiled(tif))
3341
0
    {
3342
0
        TIFFErrorExtR(tif, TIFFFileName(tif),
3343
0
                      "Can't use TIFFReadRGBATile() with striped file.");
3344
0
        return (0);
3345
0
    }
3346
3347
0
    TIFFGetFieldDefaulted(tif, TIFFTAG_TILEWIDTH, &tile_xsize);
3348
0
    TIFFGetFieldDefaulted(tif, TIFFTAG_TILELENGTH, &tile_ysize);
3349
0
    if (tile_xsize == 0 || tile_ysize == 0)
3350
0
    {
3351
0
        TIFFErrorExtR(tif, TIFFFileName(tif),
3352
0
                      "tile_xsize or tile_ysize is zero");
3353
0
        return (0);
3354
0
    }
3355
3356
0
    if ((col % tile_xsize) != 0 || (row % tile_ysize) != 0)
3357
0
    {
3358
0
        TIFFErrorExtR(tif, TIFFFileName(tif),
3359
0
                      "Row/col passed to TIFFReadRGBATile() must be top"
3360
0
                      "left corner of a tile.");
3361
0
        return (0);
3362
0
    }
3363
3364
    /*
3365
     * Setup the RGBA reader.
3366
     */
3367
3368
0
    if (!TIFFRGBAImageBegin(&img, tif, stop_on_error, emsg))
3369
0
    {
3370
0
        TIFFErrorExtR(tif, TIFFFileName(tif), "%s", emsg);
3371
0
        return (0);
3372
0
    }
3373
3374
0
    if (col >= img.width || row >= img.height)
3375
0
    {
3376
0
        TIFFErrorExtR(tif, TIFFFileName(tif),
3377
0
                      "Invalid row/col passed to TIFFReadRGBATile().");
3378
0
        TIFFRGBAImageEnd(&img);
3379
0
        return (0);
3380
0
    }
3381
3382
    /*
3383
     * The TIFFRGBAImageGet() function doesn't allow us to get off the
3384
     * edge of the image, even to fill an otherwise valid tile.  So we
3385
     * figure out how much we can read, and fix up the tile buffer to
3386
     * a full tile configuration afterwards.
3387
     */
3388
3389
0
    if (row + tile_ysize > img.height)
3390
0
        read_ysize = img.height - row;
3391
0
    else
3392
0
        read_ysize = tile_ysize;
3393
3394
0
    if (col + tile_xsize > img.width)
3395
0
        read_xsize = img.width - col;
3396
0
    else
3397
0
        read_xsize = tile_xsize;
3398
3399
    /*
3400
     * Read the chunk of imagery.
3401
     */
3402
3403
0
    img.row_offset = row;
3404
0
    img.col_offset = col;
3405
3406
0
    ok = TIFFRGBAImageGet(&img, raster, read_xsize, read_ysize);
3407
3408
0
    TIFFRGBAImageEnd(&img);
3409
3410
    /*
3411
     * If our read was incomplete we will need to fix up the tile by
3412
     * shifting the data around as if a full tile of data is being returned.
3413
     *
3414
     * This is all the more complicated because the image is organized in
3415
     * bottom to top format.
3416
     */
3417
3418
0
    if (read_xsize == tile_xsize && read_ysize == tile_ysize)
3419
0
        return (ok);
3420
3421
0
    for (i_row = 0; i_row < read_ysize; i_row++)
3422
0
    {
3423
0
        memmove(raster + (size_t)(tile_ysize - i_row - 1) * tile_xsize,
3424
0
                raster + (size_t)(read_ysize - i_row - 1) * read_xsize,
3425
0
                read_xsize * sizeof(uint32_t));
3426
0
        _TIFFmemset(raster + (size_t)(tile_ysize - i_row - 1) * tile_xsize +
3427
0
                        read_xsize,
3428
0
                    0, sizeof(uint32_t) * (tile_xsize - read_xsize));
3429
0
    }
3430
3431
0
    for (i_row = read_ysize; i_row < tile_ysize; i_row++)
3432
0
    {
3433
0
        _TIFFmemset(raster + (size_t)(tile_ysize - i_row - 1) * tile_xsize, 0,
3434
0
                    sizeof(uint32_t) * tile_xsize);
3435
0
    }
3436
3437
0
    return (ok);
3438
0
}