Coverage Report

Created: 2026-04-12 06:05

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/libjpeg-turbo.main/src/rdpng.c
Line
Count
Source
1
/*
2
 * rdpng.c
3
 *
4
 * This file was part of the Independent JPEG Group's software:
5
 * Copyright (C) 1991-1997, Thomas G. Lane.
6
 * libjpeg-turbo Modifications:
7
 * Copyright (C) 2015-2017, 2020-2026, D. R. Commander.
8
 * For conditions of distribution and use, see the accompanying README.ijg
9
 * file.
10
 *
11
 * This file contains routines to read input images in 8-bit-per-channel or
12
 * 16-bit-per-channel PNG format.  libspng is required to compile this
13
 * software.
14
 */
15
16
#ifdef _MSC_VER
17
#define _CRT_SECURE_NO_DEPRECATE
18
#endif
19
20
#include "cmyk.h"
21
#include "cdjpeg.h"             /* Common decls for cjpeg/djpeg applications */
22
#include "spng/spng.h"
23
24
#if defined(PNG_SUPPORTED) && \
25
    (BITS_IN_JSAMPLE != 16 || defined(C_LOSSLESS_SUPPORTED))
26
27
28
static int alpha_index[JPEG_NUMCS] = {
29
  -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, 3, 3, 0, 0, -1
30
};
31
32
33
0
#define TRY_SPNG(f) { \
34
0
  int __spng_error = (f); \
35
0
  if (__spng_error) \
36
0
    ERREXITS(cinfo, JERR_PNG_LIBSPNG, spng_strerror(__spng_error)); \
37
0
}
38
39
40
/* Private version of data source object */
41
42
typedef struct {
43
  struct cjpeg_source_struct pub; /* public fields */
44
45
  spng_ctx *ctx;
46
  uint8_t png_bit_depth, png_color_type;
47
  int png_alpha;
48
  struct spng_plte colormap;    /* PNG colormap */
49
50
  /* Usually these two pointers point to the same place: */
51
  unsigned char *iobuffer;      /* libspng's I/O buffer */
52
  _JSAMPROW pixrow;             /* compressor input buffer */
53
  size_t buffer_width;          /* width of I/O buffer */
54
  _JSAMPLE *rescale;            /* PNG bit depth => data precision remapping
55
                                   array, or NULL */
56
} png_source_struct;
57
58
typedef png_source_struct *png_source_ptr;
59
60
61
/*
62
 * Read one row of pixels.
63
 *
64
 * We provide several different versions depending on input file format.
65
 * In all cases, input is scaled to cinfo->data_precision.
66
 *
67
 * get_raw_row() is used when the pixel format is JCS_EXT_RGB/JCS_RGB or
68
 * JCS_GRAYSCALE, cinfo->data_precision is 8 or 16, and the PNG file doesn't
69
 * have an alpha channel.  Thus, we can read the pixels directly from the PNG
70
 * file.
71
 */
72
73
METHODDEF(JDIMENSION)
74
get_raw_row(j_compress_ptr cinfo, cjpeg_source_ptr sinfo)
75
0
{
76
0
  png_source_ptr source = (png_source_ptr)sinfo;
77
0
  int spng_error;
78
79
0
  spng_error = spng_decode_row(source->ctx, source->iobuffer,
80
0
                               source->buffer_width);
81
0
  if (spng_error && spng_error != SPNG_EOI)
82
0
    ERREXITS(cinfo, JERR_PNG_LIBSPNG, spng_strerror(spng_error)); \
83
0
  return 1;
84
0
}
Unexecuted instantiation: rdpng-8.c:get_raw_row
Unexecuted instantiation: rdpng-12.c:get_raw_row
Unexecuted instantiation: rdpng-16.c:get_raw_row
85
86
87
0
#define GRAY_RGB_READ_LOOP(read_op, alpha_set_op, pointer_op) { \
88
0
  for (col = cinfo->image_width; col > 0; col--) { \
89
0
    ptr[rindex] = ptr[gindex] = ptr[bindex] = read_op; \
90
0
    alpha_set_op \
91
0
    ptr += ps; \
92
0
    pointer_op \
93
0
  } \
94
0
}
95
96
97
METHODDEF(JDIMENSION)
98
get_gray_row(j_compress_ptr cinfo, cjpeg_source_ptr sinfo)
99
/* This version is for reading 8-bit-per-channel or 16-bit-per-channel
100
 * grayscale or grayscale + alpha PNG files.
101
 */
102
0
{
103
0
  png_source_ptr source = (png_source_ptr)sinfo;
104
0
  register _JSAMPROW ptr;
105
0
  register _JSAMPLE *rescale = source->rescale;
106
0
  JDIMENSION col;
107
108
0
  get_raw_row(cinfo, sinfo);
109
0
  ptr = source->pub._buffer[0];
110
#if BITS_IN_JSAMPLE != 12
111
0
  if (source->png_bit_depth == cinfo->data_precision) {
112
0
    _JSAMPLE *bufferptr = (_JSAMPLE *)source->iobuffer;
113
0
    for (col = cinfo->image_width; col > 0; col--) {
114
0
      *ptr++ = *bufferptr++;
115
0
      bufferptr += source->png_alpha;
116
0
    }
117
0
  } else
118
0
#endif
119
0
  if (source->png_bit_depth == 16) {
120
0
    register unsigned short *bufferptr = (unsigned short *)source->iobuffer;
121
0
    for (col = cinfo->image_width; col > 0; col--) {
122
0
      *ptr++ = rescale[*bufferptr++];
123
0
      bufferptr += source->png_alpha;
124
0
    }
125
0
  } else {
126
0
    register unsigned char *bufferptr = source->iobuffer;
127
0
    for (col = cinfo->image_width; col > 0; col--) {
128
0
      *ptr++ = rescale[*bufferptr++];
129
0
      bufferptr += source->png_alpha;
130
0
    }
131
0
  }
132
0
  return 1;
133
0
}
Unexecuted instantiation: rdpng-8.c:get_gray_row
Unexecuted instantiation: rdpng-12.c:get_gray_row
Unexecuted instantiation: rdpng-16.c:get_gray_row
134
135
136
METHODDEF(JDIMENSION)
137
get_gray_rgb_row(j_compress_ptr cinfo, cjpeg_source_ptr sinfo)
138
/* This version is for reading 8-bit-per-channel or 16-bit-per-channel
139
 * grayscale or grayscale + alpha PNG files and converting to extended RGB.
140
 */
141
0
{
142
0
  png_source_ptr source = (png_source_ptr)sinfo;
143
0
  register _JSAMPROW ptr;
144
0
  register _JSAMPLE *rescale = source->rescale;
145
0
  JDIMENSION col;
146
0
  register int rindex = rgb_red[cinfo->in_color_space];
147
0
  register int gindex = rgb_green[cinfo->in_color_space];
148
0
  register int bindex = rgb_blue[cinfo->in_color_space];
149
0
  register int aindex = alpha_index[cinfo->in_color_space];
150
0
  register int ps = rgb_pixelsize[cinfo->in_color_space];
151
152
0
  get_raw_row(cinfo, sinfo);
153
0
  ptr = source->pub._buffer[0];
154
#if BITS_IN_JSAMPLE != 12
155
0
  if (source->png_bit_depth == cinfo->data_precision) {
156
0
    _JSAMPLE *bufferptr = (_JSAMPLE *)source->iobuffer;
157
0
    if (aindex >= 0)
158
0
      GRAY_RGB_READ_LOOP(*bufferptr++, ptr[aindex] = _MAXJSAMPLE;,
159
0
                         bufferptr += source->png_alpha;)
160
0
    else
161
0
      GRAY_RGB_READ_LOOP(*bufferptr++, {}, bufferptr += source->png_alpha;)
162
0
  } else
163
0
#endif
164
0
  if (source->png_bit_depth == 16) {
165
0
    register unsigned short *bufferptr = (unsigned short *)source->iobuffer;
166
0
    if (aindex >= 0)
167
0
      GRAY_RGB_READ_LOOP(rescale[*bufferptr++],
168
0
                         ptr[aindex] = (1 << cinfo->data_precision) - 1;,
169
0
                         bufferptr += source->png_alpha;)
170
0
    else
171
0
      GRAY_RGB_READ_LOOP(rescale[*bufferptr++], {},
172
0
                         bufferptr += source->png_alpha;)
173
0
  } else {
174
0
    register unsigned char *bufferptr = source->iobuffer;
175
0
    if (aindex >= 0)
176
0
      GRAY_RGB_READ_LOOP(rescale[*bufferptr++],
177
0
                         ptr[aindex] = (1 << cinfo->data_precision) - 1;,
178
0
                         bufferptr += source->png_alpha;)
179
0
    else
180
0
      GRAY_RGB_READ_LOOP(rescale[*bufferptr++], {},
181
0
                         bufferptr += source->png_alpha;)
182
0
  }
183
0
  return 1;
184
0
}
Unexecuted instantiation: rdpng-8.c:get_gray_rgb_row
Unexecuted instantiation: rdpng-12.c:get_gray_rgb_row
Unexecuted instantiation: rdpng-16.c:get_gray_rgb_row
185
186
187
METHODDEF(JDIMENSION)
188
get_gray_cmyk_row(j_compress_ptr cinfo, cjpeg_source_ptr sinfo)
189
/* This version is for reading 8-bit-per-channel or 16-bit-per-channel
190
 * grayscale or grayscale + alpha PNG files and converting to CMYK.
191
 */
192
0
{
193
0
  png_source_ptr source = (png_source_ptr)sinfo;
194
0
  register _JSAMPROW ptr;
195
0
  register _JSAMPLE *rescale = source->rescale;
196
0
  JDIMENSION col;
197
198
0
  get_raw_row(cinfo, sinfo);
199
0
  ptr = source->pub._buffer[0];
200
#if BITS_IN_JSAMPLE != 12
201
0
  if (source->png_bit_depth == cinfo->data_precision) {
202
0
    register _JSAMPLE *bufferptr = (_JSAMPLE *)source->iobuffer;
203
0
    for (col = cinfo->image_width; col > 0; col--) {
204
0
      _JSAMPLE gray = *bufferptr++;
205
0
      bufferptr += source->png_alpha;
206
0
      rgb_to_cmyk(_MAXJSAMPLE, gray, gray, gray, ptr, ptr + 1, ptr + 2,
207
0
                  ptr + 3);
208
0
      ptr += 4;
209
0
    }
210
0
  } else
211
0
#endif
212
0
  if (source->png_bit_depth == 16) {
213
0
    unsigned short *bufferptr = (unsigned short *)source->iobuffer;
214
0
    for (col = cinfo->image_width; col > 0; col--) {
215
0
      _JSAMPLE gray = rescale[*bufferptr++];
216
0
      bufferptr += source->png_alpha;
217
0
      rgb_to_cmyk((1 << cinfo->data_precision) - 1, gray, gray, gray, ptr,
218
0
                  ptr + 1, ptr + 2, ptr + 3);
219
0
      ptr += 4;
220
0
    }
221
0
  } else {
222
0
    register unsigned char *bufferptr = source->iobuffer;
223
0
    for (col = cinfo->image_width; col > 0; col--) {
224
0
      _JSAMPLE gray = rescale[*bufferptr++];
225
0
      bufferptr += source->png_alpha;
226
0
      rgb_to_cmyk((1 << cinfo->data_precision) - 1, gray, gray, gray, ptr,
227
0
                  ptr + 1, ptr + 2, ptr + 3);
228
0
      ptr += 4;
229
0
    }
230
0
  }
231
0
  return 1;
232
0
}
Unexecuted instantiation: rdpng-8.c:get_gray_cmyk_row
Unexecuted instantiation: rdpng-12.c:get_gray_cmyk_row
Unexecuted instantiation: rdpng-16.c:get_gray_cmyk_row
233
234
235
0
#define RGB_READ_LOOP(read_op, alpha_set_op, pointer_op) { \
236
0
  for (col = cinfo->image_width; col > 0; col--) { \
237
0
    ptr[rindex] = read_op; \
238
0
    ptr[gindex] = read_op; \
239
0
    ptr[bindex] = read_op; \
240
0
    alpha_set_op \
241
0
    pointer_op \
242
0
    ptr += ps; \
243
0
  } \
244
0
}
245
246
247
METHODDEF(JDIMENSION)
248
get_rgb_row(j_compress_ptr cinfo, cjpeg_source_ptr sinfo)
249
/* This version is for reading 8-bit-per-channel or 16-bit-per-channel
250
 * truecolor or truecolor + alpha PNG files and converting to extended RGB.
251
 */
252
0
{
253
0
  png_source_ptr source = (png_source_ptr)sinfo;
254
0
  register _JSAMPROW ptr;
255
0
  register _JSAMPLE *rescale = source->rescale;
256
0
  JDIMENSION col;
257
0
  register int rindex = rgb_red[cinfo->in_color_space];
258
0
  register int gindex = rgb_green[cinfo->in_color_space];
259
0
  register int bindex = rgb_blue[cinfo->in_color_space];
260
0
  register int aindex = alpha_index[cinfo->in_color_space];
261
0
  register int ps = rgb_pixelsize[cinfo->in_color_space];
262
263
0
  get_raw_row(cinfo, sinfo);
264
0
  ptr = source->pub._buffer[0];
265
#if BITS_IN_JSAMPLE != 12
266
0
  if (source->png_bit_depth == cinfo->data_precision) {
267
0
    register _JSAMPLE *bufferptr = (_JSAMPLE *)source->iobuffer;
268
0
    if (aindex >= 0)
269
0
      RGB_READ_LOOP(*bufferptr++, ptr[aindex] = _MAXJSAMPLE;,
270
0
                    bufferptr += source->png_alpha;)
271
0
    else
272
0
      RGB_READ_LOOP(*bufferptr++, {}, bufferptr += source->png_alpha;)
273
0
  } else
274
0
#endif
275
0
  if (source->png_bit_depth == 16) {
276
0
    unsigned short *bufferptr = (unsigned short *)source->iobuffer;
277
0
    if (aindex >= 0)
278
0
      RGB_READ_LOOP(rescale[*bufferptr++],
279
0
                    ptr[aindex] = (1 << cinfo->data_precision) - 1;,
280
0
                    bufferptr += source->png_alpha;)
281
0
    else
282
0
      RGB_READ_LOOP(rescale[*bufferptr++], {},
283
0
                    bufferptr += source->png_alpha;)
284
0
  } else {
285
0
    register unsigned char *bufferptr = source->iobuffer;
286
0
    if (aindex >= 0)
287
0
      RGB_READ_LOOP(rescale[*bufferptr++],
288
0
                    ptr[aindex] = (1 << cinfo->data_precision) - 1;,
289
0
                    bufferptr += source->png_alpha;)
290
0
    else
291
0
      RGB_READ_LOOP(rescale[*bufferptr++], {},
292
0
                    bufferptr += source->png_alpha;)
293
0
  }
294
0
  return 1;
295
0
}
Unexecuted instantiation: rdpng-8.c:get_rgb_row
Unexecuted instantiation: rdpng-12.c:get_rgb_row
Unexecuted instantiation: rdpng-16.c:get_rgb_row
296
297
298
METHODDEF(JDIMENSION)
299
get_rgb_cmyk_row(j_compress_ptr cinfo, cjpeg_source_ptr sinfo)
300
/* This version is for reading 8-bit-per-channel or 16-bit-per-channel
301
 * truecolor or truecolor + alpha PNG files and converting to CMYK.
302
 */
303
0
{
304
0
  png_source_ptr source = (png_source_ptr)sinfo;
305
0
  register _JSAMPROW ptr;
306
0
  register _JSAMPLE *rescale = source->rescale;
307
0
  JDIMENSION col;
308
309
0
  get_raw_row(cinfo, sinfo);
310
0
  ptr = source->pub._buffer[0];
311
#if BITS_IN_JSAMPLE != 12
312
0
  if (source->png_bit_depth == cinfo->data_precision) {
313
0
    register _JSAMPLE *bufferptr = (_JSAMPLE *)source->iobuffer;
314
0
    for (col = cinfo->image_width; col > 0; col--) {
315
0
      _JSAMPLE r = *bufferptr++;
316
0
      _JSAMPLE g = *bufferptr++;
317
0
      _JSAMPLE b = *bufferptr++;
318
0
      bufferptr += source->png_alpha;
319
0
      rgb_to_cmyk(_MAXJSAMPLE, r, g, b, ptr, ptr + 1, ptr + 2, ptr + 3);
320
0
      ptr += 4;
321
0
    }
322
0
  } else
323
0
#endif
324
0
  if (source->png_bit_depth == 16) {
325
0
    unsigned short *bufferptr = (unsigned short *)source->iobuffer;
326
0
    for (col = cinfo->image_width; col > 0; col--) {
327
0
      _JSAMPLE r = rescale[*bufferptr++];
328
0
      _JSAMPLE g = rescale[*bufferptr++];
329
0
      _JSAMPLE b = rescale[*bufferptr++];
330
0
      bufferptr += source->png_alpha;
331
0
      rgb_to_cmyk((1 << cinfo->data_precision) - 1, r, g, b, ptr, ptr + 1,
332
0
                  ptr + 2, ptr + 3);
333
0
      ptr += 4;
334
0
    }
335
0
  } else {
336
0
    register unsigned char *bufferptr = source->iobuffer;
337
0
    for (col = cinfo->image_width; col > 0; col--) {
338
0
      _JSAMPLE r = rescale[*bufferptr++];
339
0
      _JSAMPLE g = rescale[*bufferptr++];
340
0
      _JSAMPLE b = rescale[*bufferptr++];
341
0
      bufferptr += source->png_alpha;
342
0
      rgb_to_cmyk((1 << cinfo->data_precision) - 1, r, g, b, ptr, ptr + 1,
343
0
                  ptr + 2, ptr + 3);
344
0
      ptr += 4;
345
0
    }
346
0
  }
347
0
  return 1;
348
0
}
Unexecuted instantiation: rdpng-8.c:get_rgb_cmyk_row
Unexecuted instantiation: rdpng-12.c:get_rgb_cmyk_row
Unexecuted instantiation: rdpng-16.c:get_rgb_cmyk_row
349
350
351
METHODDEF(JDIMENSION)
352
get_indexed_row(j_compress_ptr cinfo, cjpeg_source_ptr sinfo)
353
0
{
354
/* This version is for reading 8-bit-per-channel indexed-color PNG files and
355
 * converting to extended RGB or CMYK.
356
 */
357
0
  png_source_ptr source = (png_source_ptr)sinfo;
358
0
  register _JSAMPROW ptr;
359
0
  register JSAMPLE *bufferptr = (JSAMPLE *)source->iobuffer;
360
0
  register _JSAMPLE *rescale = source->rescale;
361
0
  JDIMENSION col;
362
363
0
  get_raw_row(cinfo, sinfo);
364
0
  ptr = source->pub._buffer[0];
365
#if BITS_IN_JSAMPLE == 8
366
0
  if (source->png_bit_depth == cinfo->data_precision) {
367
0
    if (cinfo->in_color_space == JCS_GRAYSCALE) {
368
0
      for (col = cinfo->image_width; col > 0; col--) {
369
0
        JSAMPLE index = *bufferptr++;
370
371
0
        if (index >= source->colormap.n_entries)
372
0
          ERREXIT(cinfo, JERR_PNG_OUTOFRANGE);
373
0
        *ptr++ = source->colormap.entries[index].red;
374
0
      }
375
0
    } else if (cinfo->in_color_space == JCS_CMYK) {
376
0
      for (col = cinfo->image_width; col > 0; col--) {
377
0
        JSAMPLE index = *bufferptr++;
378
379
0
        if (index >= source->colormap.n_entries)
380
0
          ERREXIT(cinfo, JERR_PNG_OUTOFRANGE);
381
0
        rgb_to_cmyk(_MAXJSAMPLE, source->colormap.entries[index].red,
382
0
                    source->colormap.entries[index].green,
383
0
                    source->colormap.entries[index].blue, ptr, ptr + 1,
384
0
                    ptr + 2, ptr + 3);
385
0
        ptr += 4;
386
0
      }
387
0
    } else {
388
0
      register int rindex = rgb_red[cinfo->in_color_space];
389
0
      register int gindex = rgb_green[cinfo->in_color_space];
390
0
      register int bindex = rgb_blue[cinfo->in_color_space];
391
0
      register int aindex = alpha_index[cinfo->in_color_space];
392
0
      register int ps = rgb_pixelsize[cinfo->in_color_space];
393
394
0
      for (col = cinfo->image_width; col > 0; col--) {
395
0
        JSAMPLE index = *bufferptr++;
396
397
0
        if (index >= source->colormap.n_entries)
398
0
          ERREXIT(cinfo, JERR_PNG_OUTOFRANGE);
399
0
        ptr[rindex] = source->colormap.entries[index].red;
400
0
        ptr[gindex] = source->colormap.entries[index].green;
401
0
        ptr[bindex] = source->colormap.entries[index].blue;
402
0
        if (aindex >= 0)
403
0
          ptr[aindex] = _MAXJSAMPLE;
404
0
        ptr += ps;
405
0
      }
406
0
    }
407
0
  } else
408
0
#endif
409
0
  {
410
0
    if (cinfo->in_color_space == JCS_GRAYSCALE) {
411
0
      for (col = cinfo->image_width; col > 0; col--) {
412
0
        JSAMPLE index = *bufferptr++;
413
414
0
        if (index >= source->colormap.n_entries)
415
0
          ERREXIT(cinfo, JERR_PNG_OUTOFRANGE);
416
0
        *ptr++ = rescale[source->colormap.entries[index].red];
417
0
      }
418
0
    } else if (cinfo->in_color_space == JCS_CMYK) {
419
0
      for (col = cinfo->image_width; col > 0; col--) {
420
0
        JSAMPLE index = *bufferptr++;
421
422
0
        if (index >= source->colormap.n_entries)
423
0
          ERREXIT(cinfo, JERR_PNG_OUTOFRANGE);
424
0
        rgb_to_cmyk((1 << cinfo->data_precision) - 1,
425
0
                    rescale[source->colormap.entries[index].red],
426
0
                    rescale[source->colormap.entries[index].green],
427
0
                    rescale[source->colormap.entries[index].blue], ptr,
428
0
                    ptr + 1, ptr + 2, ptr + 3);
429
0
        ptr += 4;
430
0
      }
431
0
    } else {
432
0
      register int rindex = rgb_red[cinfo->in_color_space];
433
0
      register int gindex = rgb_green[cinfo->in_color_space];
434
0
      register int bindex = rgb_blue[cinfo->in_color_space];
435
0
      register int aindex = alpha_index[cinfo->in_color_space];
436
0
      register int ps = rgb_pixelsize[cinfo->in_color_space];
437
438
0
      for (col = cinfo->image_width; col > 0; col--) {
439
0
        JSAMPLE index = *bufferptr++;
440
441
0
        if (index >= source->colormap.n_entries)
442
0
          ERREXIT(cinfo, JERR_PNG_OUTOFRANGE);
443
0
        ptr[rindex] = rescale[source->colormap.entries[index].red];
444
0
        ptr[gindex] = rescale[source->colormap.entries[index].green];
445
0
        ptr[bindex] = rescale[source->colormap.entries[index].blue];
446
0
        if (aindex >= 0)
447
0
          ptr[aindex] = (1 << cinfo->data_precision) - 1;
448
0
        ptr += ps;
449
0
      }
450
0
    }
451
0
  }
452
0
  return 1;
453
0
}
Unexecuted instantiation: rdpng-8.c:get_indexed_row
Unexecuted instantiation: rdpng-12.c:get_indexed_row
Unexecuted instantiation: rdpng-16.c:get_indexed_row
454
455
456
#ifdef ZERO_BUFFERS
457
458
static void *spng_malloc(size_t size)
459
{
460
  return calloc(1, size);
461
}
462
463
#endif
464
465
466
/*
467
 * Read the file header; return image size and component count.
468
 */
469
470
METHODDEF(void)
471
start_input_png(j_compress_ptr cinfo, cjpeg_source_ptr sinfo)
472
0
{
473
0
  png_source_ptr source = (png_source_ptr)sinfo;
474
0
  struct spng_ihdr ihdr;
475
0
  int png_components = 3;
476
0
  boolean use_raw_buffer;
477
#ifdef ZERO_BUFFERS
478
  struct spng_alloc alloc;
479
480
  alloc.malloc_fn = spng_malloc;
481
  alloc.realloc_fn = realloc;
482
  alloc.calloc_fn = calloc;
483
  alloc.free_fn = free;
484
  source->ctx = spng_ctx_new2(&alloc, 0);
485
#else
486
0
  source->ctx = spng_ctx_new(0);
487
0
#endif
488
0
  if (!source->ctx)
489
0
    ERREXITS(cinfo, JERR_PNG_LIBSPNG, "Could not create context");
490
491
0
  TRY_SPNG(spng_set_png_file(source->ctx, sinfo->input_file));
492
493
0
  TRY_SPNG(spng_get_ihdr(source->ctx, &ihdr));
494
495
0
  if (ihdr.width > JPEG_MAX_DIMENSION || ihdr.height > JPEG_MAX_DIMENSION)
496
0
    ERREXIT1(cinfo, JERR_IMAGE_TOO_BIG, JPEG_MAX_DIMENSION);
497
0
  if (ihdr.bit_depth != 8 && ihdr.bit_depth != 16)
498
0
    ERREXIT(cinfo, JERR_PNG_BADDEPTH);
499
0
  if (sinfo->max_pixels &&
500
0
      (unsigned long long)ihdr.width * ihdr.height > sinfo->max_pixels)
501
0
    ERREXIT1(cinfo, JERR_IMAGE_TOO_BIG, sinfo->max_pixels);
502
503
0
  cinfo->image_width = (JDIMENSION)ihdr.width;
504
0
  cinfo->image_height = (JDIMENSION)ihdr.height;
505
0
  source->png_bit_depth = ihdr.bit_depth;
506
0
  source->png_color_type = ihdr.color_type;
507
508
  /* initialize flags to most common settings */
509
0
  use_raw_buffer = FALSE;       /* do we map input buffer onto I/O buffer? */
510
511
0
  switch (ihdr.color_type) {
512
0
  case SPNG_COLOR_TYPE_GRAYSCALE_ALPHA:
513
0
  case SPNG_COLOR_TYPE_GRAYSCALE:
514
0
    if (cinfo->in_color_space == JCS_UNKNOWN ||
515
0
        cinfo->in_color_space == JCS_RGB)
516
0
      cinfo->in_color_space = JCS_GRAYSCALE;
517
0
    TRACEMS3(cinfo, 1, JTRC_PNG_GRAYSCALE, ihdr.width, ihdr.height,
518
0
             ihdr.bit_depth);
519
0
    if (cinfo->in_color_space == JCS_GRAYSCALE) {
520
0
      if (ihdr.color_type == SPNG_COLOR_TYPE_GRAYSCALE &&
521
0
          cinfo->data_precision == ihdr.bit_depth) {
522
0
        source->pub.get_pixel_rows = get_raw_row;
523
0
        use_raw_buffer = TRUE;
524
0
      } else
525
0
        source->pub.get_pixel_rows = get_gray_row;
526
0
    } else if (IsExtRGB(cinfo->in_color_space))
527
0
      source->pub.get_pixel_rows = get_gray_rgb_row;
528
0
    else if (cinfo->in_color_space == JCS_CMYK)
529
0
        source->pub.get_pixel_rows = get_gray_cmyk_row;
530
0
    else
531
0
      ERREXIT(cinfo, JERR_BAD_IN_COLORSPACE);
532
0
    if (ihdr.color_type == SPNG_COLOR_TYPE_GRAYSCALE_ALPHA) {
533
0
      png_components = 2;
534
0
      source->png_alpha = 1;
535
0
    } else {
536
0
      png_components = 1;
537
0
      source->png_alpha = 0;
538
0
    }
539
0
    break;
540
541
0
  case SPNG_COLOR_TYPE_TRUECOLOR:
542
0
  case SPNG_COLOR_TYPE_TRUECOLOR_ALPHA:
543
0
    if (cinfo->in_color_space == JCS_UNKNOWN)
544
0
      cinfo->in_color_space = JCS_EXT_RGB;
545
0
    TRACEMS3(cinfo, 1, JTRC_PNG_TRUECOLOR, ihdr.width, ihdr.height,
546
0
             ihdr.bit_depth);
547
0
    if (ihdr.color_type == SPNG_COLOR_TYPE_TRUECOLOR &&
548
0
        cinfo->data_precision == ihdr.bit_depth &&
549
0
#if RGB_RED == 0 && RGB_GREEN == 1 && RGB_BLUE == 2 && RGB_PIXELSIZE == 3
550
0
        (cinfo->in_color_space == JCS_EXT_RGB ||
551
0
         cinfo->in_color_space == JCS_RGB)) {
552
#else
553
        cinfo->in_color_space == JCS_EXT_RGB) {
554
#endif
555
0
      source->pub.get_pixel_rows = get_raw_row;
556
0
      use_raw_buffer = TRUE;
557
0
    } else if (IsExtRGB(cinfo->in_color_space))
558
0
      source->pub.get_pixel_rows = get_rgb_row;
559
0
    else if (cinfo->in_color_space == JCS_CMYK)
560
0
      source->pub.get_pixel_rows = get_rgb_cmyk_row;
561
0
    else
562
0
      ERREXIT(cinfo, JERR_BAD_IN_COLORSPACE);
563
0
    if (ihdr.color_type == SPNG_COLOR_TYPE_TRUECOLOR_ALPHA) {
564
0
      png_components = 4;
565
0
      source->png_alpha = 1;
566
0
    } else {
567
0
      png_components = 3;
568
0
      source->png_alpha = 0;
569
0
    }
570
0
    break;
571
572
0
  case SPNG_COLOR_TYPE_INDEXED:
573
0
  {
574
0
    int i, gray = 1;
575
576
0
    TRACEMS3(cinfo, 1, JTRC_PNG_INDEXED, ihdr.width, ihdr.height,
577
0
             ihdr.bit_depth);
578
0
    TRY_SPNG(spng_get_plte(source->ctx, &source->colormap));
579
0
    if (source->png_bit_depth != 8 || source->colormap.n_entries > 256)
580
0
      ERREXIT(cinfo, JERR_PNG_OUTOFRANGE);
581
582
0
    for (i = 0; i < (int)source->colormap.n_entries; i++) {
583
0
      if (source->colormap.entries[i].red !=
584
0
          source->colormap.entries[i].green ||
585
0
          source->colormap.entries[i].green !=
586
0
          source->colormap.entries[i].blue)
587
0
        gray = 0;
588
0
    }
589
590
0
    if ((cinfo->in_color_space == JCS_UNKNOWN ||
591
0
         cinfo->in_color_space == JCS_RGB) && gray)
592
0
      cinfo->in_color_space = JCS_GRAYSCALE;
593
0
    if (cinfo->in_color_space == JCS_GRAYSCALE && !gray)
594
0
      ERREXIT(cinfo, JERR_BAD_IN_COLORSPACE);
595
596
0
    source->pub.get_pixel_rows = get_indexed_row;
597
0
    png_components = 1;
598
0
    source->png_alpha = 0;
599
0
    break;
600
0
  }
601
602
0
  default:
603
0
    ERREXIT(cinfo, JERR_PNG_OUTOFRANGE);
604
0
  }
605
606
0
  if (IsExtRGB(cinfo->in_color_space))
607
0
    cinfo->input_components = rgb_pixelsize[cinfo->in_color_space];
608
0
  else if (cinfo->in_color_space == JCS_GRAYSCALE)
609
0
    cinfo->input_components = 1;
610
0
  else if (cinfo->in_color_space == JCS_CMYK)
611
0
    cinfo->input_components = 4;
612
613
  /* Allocate space for I/O buffer: 1 or 3 bytes or words/pixel. */
614
0
  source->buffer_width =
615
0
    (size_t)ihdr.width * png_components * source->png_bit_depth / 8;
616
0
  source->iobuffer = (unsigned char *)
617
0
    (*cinfo->mem->alloc_small) ((j_common_ptr)cinfo, JPOOL_IMAGE,
618
0
                                source->buffer_width);
619
620
  /* Create compressor input buffer. */
621
0
  if (use_raw_buffer) {
622
    /* For unscaled raw-input case, we can just map it onto the I/O buffer. */
623
    /* Synthesize a _JSAMPARRAY pointer structure */
624
0
    source->pixrow = (_JSAMPROW)source->iobuffer;
625
0
    source->pub._buffer = &source->pixrow;
626
0
    source->pub.buffer_height = 1;
627
0
  } else {
628
0
    unsigned int maxval = source->png_bit_depth == 16 ? 65535 : 255;
629
0
    size_t val, half_maxval;
630
631
    /* Need to translate anyway, so make a separate sample buffer. */
632
0
    source->pub._buffer = (_JSAMPARRAY)(*cinfo->mem->alloc_sarray)
633
0
      ((j_common_ptr)cinfo, JPOOL_IMAGE,
634
0
       (JDIMENSION)ihdr.width * cinfo->input_components, (JDIMENSION)1);
635
0
    source->pub.buffer_height = 1;
636
637
    /* Compute the rescaling array. */
638
0
    source->rescale = (_JSAMPLE *)
639
0
      (*cinfo->mem->alloc_small) ((j_common_ptr)cinfo, JPOOL_IMAGE,
640
0
                                  (maxval + 1L) * sizeof(_JSAMPLE));
641
0
    memset(source->rescale, 0, (maxval + 1L) * sizeof(_JSAMPLE));
642
0
    half_maxval = maxval / 2;
643
0
    for (val = 0; val <= maxval; val++) {
644
      /* The multiplication here must be done in 32 bits to avoid overflow */
645
0
      source->rescale[val] =
646
0
        (_JSAMPLE)((val * ((1 << cinfo->data_precision) - 1) + half_maxval) /
647
0
                   maxval);
648
0
    }
649
0
  }
650
651
0
  TRY_SPNG(spng_decode_image(source->ctx, NULL, 0, SPNG_FMT_PNG,
652
0
                             SPNG_DECODE_PROGRESSIVE));
653
0
}
Unexecuted instantiation: rdpng-8.c:start_input_png
Unexecuted instantiation: rdpng-12.c:start_input_png
Unexecuted instantiation: rdpng-16.c:start_input_png
654
655
656
/*
657
 * Return the ICC profile (if any) embedded in the PNG image.
658
 */
659
660
METHODDEF(boolean)
661
read_icc_profile_png(j_compress_ptr cinfo, cjpeg_source_ptr sinfo,
662
                     JOCTET **icc_data_ptr, unsigned int *icc_data_len)
663
0
{
664
0
  png_source_ptr source = (png_source_ptr)sinfo;
665
0
  struct spng_iccp iccp;
666
667
0
  if (!icc_data_ptr || !icc_data_len)
668
0
    ERREXIT(cinfo, JERR_BUFFER_SIZE);
669
670
0
  if (source->ctx && spng_get_iccp(source->ctx, &iccp) == 0) {
671
0
    *icc_data_ptr = (JOCTET *)iccp.profile;
672
0
    *icc_data_len = (unsigned int)iccp.profile_len;
673
0
    return TRUE;
674
0
  }
675
676
0
  return FALSE;
677
0
}
Unexecuted instantiation: rdpng-8.c:read_icc_profile_png
Unexecuted instantiation: rdpng-12.c:read_icc_profile_png
Unexecuted instantiation: rdpng-16.c:read_icc_profile_png
678
679
680
/*
681
 * Finish up at the end of the file.
682
 */
683
684
METHODDEF(void)
685
finish_input_png(j_compress_ptr cinfo, cjpeg_source_ptr sinfo)
686
0
{
687
0
  png_source_ptr source = (png_source_ptr)sinfo;
688
689
0
  if (source->ctx) {
690
0
    spng_decode_chunks(source->ctx);
691
0
    spng_ctx_free(source->ctx);
692
0
    source->ctx = NULL;
693
0
  }
694
0
}
Unexecuted instantiation: rdpng-8.c:finish_input_png
Unexecuted instantiation: rdpng-12.c:finish_input_png
Unexecuted instantiation: rdpng-16.c:finish_input_png
695
696
697
/*
698
 * The module selection routine for PNG format input.
699
 */
700
701
GLOBAL(cjpeg_source_ptr)
702
_jinit_read_png(j_compress_ptr cinfo)
703
0
{
704
0
  png_source_ptr source;
705
706
#if BITS_IN_JSAMPLE == 8
707
0
  if (cinfo->data_precision > BITS_IN_JSAMPLE || cinfo->data_precision < 2)
708
#else
709
0
  if (cinfo->data_precision > BITS_IN_JSAMPLE ||
710
0
      cinfo->data_precision < BITS_IN_JSAMPLE - 3)
711
0
#endif
712
0
    ERREXIT1(cinfo, JERR_BAD_PRECISION, cinfo->data_precision);
713
714
  /* Create module interface object */
715
0
  source = (png_source_ptr)
716
0
    (*cinfo->mem->alloc_small) ((j_common_ptr)cinfo, JPOOL_IMAGE,
717
0
                                sizeof(png_source_struct));
718
  /* Fill in method ptrs, except get_pixel_rows which start_input sets */
719
0
  source->pub.start_input = start_input_png;
720
0
  source->pub.read_icc_profile = read_icc_profile_png;
721
0
  source->pub.finish_input = finish_input_png;
722
0
  source->pub.max_pixels = 0;
723
724
0
  return (cjpeg_source_ptr)source;
725
0
}
Unexecuted instantiation: jinit_read_png
Unexecuted instantiation: j12init_read_png
Unexecuted instantiation: j16init_read_png
726
727
#endif /* defined(PNG_SUPPORTED) &&
728
          (BITS_IN_JSAMPLE != 16 || defined(C_LOSSLESS_SUPPORTED)) */