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
46.1k
#define TRY_SPNG(f) { \
34
46.1k
  int __spng_error = (f); \
35
46.1k
  if (__spng_error) \
36
46.1k
    ERREXITS(cinfo, JERR_PNG_LIBSPNG, spng_strerror(__spng_error)); \
37
46.1k
}
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
44.7k
{
76
44.7k
  png_source_ptr source = (png_source_ptr)sinfo;
77
44.7k
  int spng_error;
78
79
44.7k
  spng_error = spng_decode_row(source->ctx, source->iobuffer,
80
44.7k
                               source->buffer_width);
81
44.7k
  if (spng_error && spng_error != SPNG_EOI)
82
1.35k
    ERREXITS(cinfo, JERR_PNG_LIBSPNG, spng_strerror(spng_error)); \
83
44.7k
  return 1;
84
44.7k
}
Unexecuted instantiation: rdpng-8.c:get_raw_row
rdpng-12.c:get_raw_row
Line
Count
Source
75
44.7k
{
76
44.7k
  png_source_ptr source = (png_source_ptr)sinfo;
77
44.7k
  int spng_error;
78
79
44.7k
  spng_error = spng_decode_row(source->ctx, source->iobuffer,
80
44.7k
                               source->buffer_width);
81
44.7k
  if (spng_error && spng_error != SPNG_EOI)
82
1.35k
    ERREXITS(cinfo, JERR_PNG_LIBSPNG, spng_strerror(spng_error)); \
83
44.7k
  return 1;
84
44.7k
}
Unexecuted instantiation: rdpng-16.c:get_raw_row
85
86
87
16.1k
#define GRAY_RGB_READ_LOOP(read_op, alpha_set_op, pointer_op) { \
88
1.50M
  for (col = cinfo->image_width; col > 0; col--) { \
89
1.48M
    ptr[rindex] = ptr[gindex] = ptr[bindex] = read_op; \
90
1.48M
    alpha_set_op \
91
1.48M
    ptr += ps; \
92
1.48M
    pointer_op \
93
1.48M
  } \
94
16.1k
}
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
3.22k
{
103
3.22k
  png_source_ptr source = (png_source_ptr)sinfo;
104
3.22k
  register _JSAMPROW ptr;
105
3.22k
  register _JSAMPLE *rescale = source->rescale;
106
3.22k
  JDIMENSION col;
107
108
3.22k
  get_raw_row(cinfo, sinfo);
109
3.22k
  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
3.22k
  if (source->png_bit_depth == 16) {
120
1.10k
    register unsigned short *bufferptr = (unsigned short *)source->iobuffer;
121
109k
    for (col = cinfo->image_width; col > 0; col--) {
122
108k
      *ptr++ = rescale[*bufferptr++];
123
108k
      bufferptr += source->png_alpha;
124
108k
    }
125
2.11k
  } else {
126
2.11k
    register unsigned char *bufferptr = source->iobuffer;
127
191k
    for (col = cinfo->image_width; col > 0; col--) {
128
189k
      *ptr++ = rescale[*bufferptr++];
129
189k
      bufferptr += source->png_alpha;
130
189k
    }
131
2.11k
  }
132
3.22k
  return 1;
133
3.22k
}
Unexecuted instantiation: rdpng-8.c:get_gray_row
rdpng-12.c:get_gray_row
Line
Count
Source
102
3.22k
{
103
3.22k
  png_source_ptr source = (png_source_ptr)sinfo;
104
3.22k
  register _JSAMPROW ptr;
105
3.22k
  register _JSAMPLE *rescale = source->rescale;
106
3.22k
  JDIMENSION col;
107
108
3.22k
  get_raw_row(cinfo, sinfo);
109
3.22k
  ptr = source->pub._buffer[0];
110
#if BITS_IN_JSAMPLE != 12
111
  if (source->png_bit_depth == cinfo->data_precision) {
112
    _JSAMPLE *bufferptr = (_JSAMPLE *)source->iobuffer;
113
    for (col = cinfo->image_width; col > 0; col--) {
114
      *ptr++ = *bufferptr++;
115
      bufferptr += source->png_alpha;
116
    }
117
  } else
118
#endif
119
3.22k
  if (source->png_bit_depth == 16) {
120
1.10k
    register unsigned short *bufferptr = (unsigned short *)source->iobuffer;
121
109k
    for (col = cinfo->image_width; col > 0; col--) {
122
108k
      *ptr++ = rescale[*bufferptr++];
123
108k
      bufferptr += source->png_alpha;
124
108k
    }
125
2.11k
  } else {
126
2.11k
    register unsigned char *bufferptr = source->iobuffer;
127
191k
    for (col = cinfo->image_width; col > 0; col--) {
128
189k
      *ptr++ = rescale[*bufferptr++];
129
189k
      bufferptr += source->png_alpha;
130
189k
    }
131
2.11k
  }
132
3.22k
  return 1;
133
3.22k
}
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
16.1k
{
142
16.1k
  png_source_ptr source = (png_source_ptr)sinfo;
143
16.1k
  register _JSAMPROW ptr;
144
16.1k
  register _JSAMPLE *rescale = source->rescale;
145
16.1k
  JDIMENSION col;
146
16.1k
  register int rindex = rgb_red[cinfo->in_color_space];
147
16.1k
  register int gindex = rgb_green[cinfo->in_color_space];
148
16.1k
  register int bindex = rgb_blue[cinfo->in_color_space];
149
16.1k
  register int aindex = alpha_index[cinfo->in_color_space];
150
16.1k
  register int ps = rgb_pixelsize[cinfo->in_color_space];
151
152
16.1k
  get_raw_row(cinfo, sinfo);
153
16.1k
  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
16.1k
  if (source->png_bit_depth == 16) {
165
5.54k
    register unsigned short *bufferptr = (unsigned short *)source->iobuffer;
166
5.54k
    if (aindex >= 0)
167
1.10k
      GRAY_RGB_READ_LOOP(rescale[*bufferptr++],
168
5.54k
                         ptr[aindex] = (1 << cinfo->data_precision) - 1;,
169
5.54k
                         bufferptr += source->png_alpha;)
170
4.43k
    else
171
4.43k
      GRAY_RGB_READ_LOOP(rescale[*bufferptr++], {},
172
5.54k
                         bufferptr += source->png_alpha;)
173
10.5k
  } else {
174
10.5k
    register unsigned char *bufferptr = source->iobuffer;
175
10.5k
    if (aindex >= 0)
176
2.03k
      GRAY_RGB_READ_LOOP(rescale[*bufferptr++],
177
10.5k
                         ptr[aindex] = (1 << cinfo->data_precision) - 1;,
178
10.5k
                         bufferptr += source->png_alpha;)
179
8.55k
    else
180
8.55k
      GRAY_RGB_READ_LOOP(rescale[*bufferptr++], {},
181
10.5k
                         bufferptr += source->png_alpha;)
182
10.5k
  }
183
16.1k
  return 1;
184
16.1k
}
Unexecuted instantiation: rdpng-8.c:get_gray_rgb_row
rdpng-12.c:get_gray_rgb_row
Line
Count
Source
141
16.1k
{
142
16.1k
  png_source_ptr source = (png_source_ptr)sinfo;
143
16.1k
  register _JSAMPROW ptr;
144
16.1k
  register _JSAMPLE *rescale = source->rescale;
145
16.1k
  JDIMENSION col;
146
16.1k
  register int rindex = rgb_red[cinfo->in_color_space];
147
16.1k
  register int gindex = rgb_green[cinfo->in_color_space];
148
16.1k
  register int bindex = rgb_blue[cinfo->in_color_space];
149
16.1k
  register int aindex = alpha_index[cinfo->in_color_space];
150
16.1k
  register int ps = rgb_pixelsize[cinfo->in_color_space];
151
152
16.1k
  get_raw_row(cinfo, sinfo);
153
16.1k
  ptr = source->pub._buffer[0];
154
#if BITS_IN_JSAMPLE != 12
155
  if (source->png_bit_depth == cinfo->data_precision) {
156
    _JSAMPLE *bufferptr = (_JSAMPLE *)source->iobuffer;
157
    if (aindex >= 0)
158
      GRAY_RGB_READ_LOOP(*bufferptr++, ptr[aindex] = _MAXJSAMPLE;,
159
                         bufferptr += source->png_alpha;)
160
    else
161
      GRAY_RGB_READ_LOOP(*bufferptr++, {}, bufferptr += source->png_alpha;)
162
  } else
163
#endif
164
16.1k
  if (source->png_bit_depth == 16) {
165
5.54k
    register unsigned short *bufferptr = (unsigned short *)source->iobuffer;
166
5.54k
    if (aindex >= 0)
167
1.10k
      GRAY_RGB_READ_LOOP(rescale[*bufferptr++],
168
5.54k
                         ptr[aindex] = (1 << cinfo->data_precision) - 1;,
169
5.54k
                         bufferptr += source->png_alpha;)
170
4.43k
    else
171
4.43k
      GRAY_RGB_READ_LOOP(rescale[*bufferptr++], {},
172
5.54k
                         bufferptr += source->png_alpha;)
173
10.5k
  } else {
174
10.5k
    register unsigned char *bufferptr = source->iobuffer;
175
10.5k
    if (aindex >= 0)
176
2.03k
      GRAY_RGB_READ_LOOP(rescale[*bufferptr++],
177
10.5k
                         ptr[aindex] = (1 << cinfo->data_precision) - 1;,
178
10.5k
                         bufferptr += source->png_alpha;)
179
8.55k
    else
180
8.55k
      GRAY_RGB_READ_LOOP(rescale[*bufferptr++], {},
181
10.5k
                         bufferptr += source->png_alpha;)
182
10.5k
  }
183
16.1k
  return 1;
184
16.1k
}
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
3.22k
{
193
3.22k
  png_source_ptr source = (png_source_ptr)sinfo;
194
3.22k
  register _JSAMPROW ptr;
195
3.22k
  register _JSAMPLE *rescale = source->rescale;
196
3.22k
  JDIMENSION col;
197
198
3.22k
  get_raw_row(cinfo, sinfo);
199
3.22k
  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
3.22k
  if (source->png_bit_depth == 16) {
213
1.10k
    unsigned short *bufferptr = (unsigned short *)source->iobuffer;
214
109k
    for (col = cinfo->image_width; col > 0; col--) {
215
108k
      _JSAMPLE gray = rescale[*bufferptr++];
216
108k
      bufferptr += source->png_alpha;
217
108k
      rgb_to_cmyk((1 << cinfo->data_precision) - 1, gray, gray, gray, ptr,
218
108k
                  ptr + 1, ptr + 2, ptr + 3);
219
108k
      ptr += 4;
220
108k
    }
221
2.11k
  } else {
222
2.11k
    register unsigned char *bufferptr = source->iobuffer;
223
191k
    for (col = cinfo->image_width; col > 0; col--) {
224
189k
      _JSAMPLE gray = rescale[*bufferptr++];
225
189k
      bufferptr += source->png_alpha;
226
189k
      rgb_to_cmyk((1 << cinfo->data_precision) - 1, gray, gray, gray, ptr,
227
189k
                  ptr + 1, ptr + 2, ptr + 3);
228
189k
      ptr += 4;
229
189k
    }
230
2.11k
  }
231
3.22k
  return 1;
232
3.22k
}
Unexecuted instantiation: rdpng-8.c:get_gray_cmyk_row
rdpng-12.c:get_gray_cmyk_row
Line
Count
Source
192
3.22k
{
193
3.22k
  png_source_ptr source = (png_source_ptr)sinfo;
194
3.22k
  register _JSAMPROW ptr;
195
3.22k
  register _JSAMPLE *rescale = source->rescale;
196
3.22k
  JDIMENSION col;
197
198
3.22k
  get_raw_row(cinfo, sinfo);
199
3.22k
  ptr = source->pub._buffer[0];
200
#if BITS_IN_JSAMPLE != 12
201
  if (source->png_bit_depth == cinfo->data_precision) {
202
    register _JSAMPLE *bufferptr = (_JSAMPLE *)source->iobuffer;
203
    for (col = cinfo->image_width; col > 0; col--) {
204
      _JSAMPLE gray = *bufferptr++;
205
      bufferptr += source->png_alpha;
206
      rgb_to_cmyk(_MAXJSAMPLE, gray, gray, gray, ptr, ptr + 1, ptr + 2,
207
                  ptr + 3);
208
      ptr += 4;
209
    }
210
  } else
211
#endif
212
3.22k
  if (source->png_bit_depth == 16) {
213
1.10k
    unsigned short *bufferptr = (unsigned short *)source->iobuffer;
214
109k
    for (col = cinfo->image_width; col > 0; col--) {
215
108k
      _JSAMPLE gray = rescale[*bufferptr++];
216
108k
      bufferptr += source->png_alpha;
217
108k
      rgb_to_cmyk((1 << cinfo->data_precision) - 1, gray, gray, gray, ptr,
218
108k
                  ptr + 1, ptr + 2, ptr + 3);
219
108k
      ptr += 4;
220
108k
    }
221
2.11k
  } else {
222
2.11k
    register unsigned char *bufferptr = source->iobuffer;
223
191k
    for (col = cinfo->image_width; col > 0; col--) {
224
189k
      _JSAMPLE gray = rescale[*bufferptr++];
225
189k
      bufferptr += source->png_alpha;
226
189k
      rgb_to_cmyk((1 << cinfo->data_precision) - 1, gray, gray, gray, ptr,
227
189k
                  ptr + 1, ptr + 2, ptr + 3);
228
189k
      ptr += 4;
229
189k
    }
230
2.11k
  }
231
3.22k
  return 1;
232
3.22k
}
Unexecuted instantiation: rdpng-16.c:get_gray_cmyk_row
233
234
235
17.6k
#define RGB_READ_LOOP(read_op, alpha_set_op, pointer_op) { \
236
1.63M
  for (col = cinfo->image_width; col > 0; col--) { \
237
1.61M
    ptr[rindex] = read_op; \
238
1.61M
    ptr[gindex] = read_op; \
239
1.61M
    ptr[bindex] = read_op; \
240
1.61M
    alpha_set_op \
241
1.61M
    pointer_op \
242
1.61M
    ptr += ps; \
243
1.61M
  } \
244
17.6k
}
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
17.6k
{
253
17.6k
  png_source_ptr source = (png_source_ptr)sinfo;
254
17.6k
  register _JSAMPROW ptr;
255
17.6k
  register _JSAMPLE *rescale = source->rescale;
256
17.6k
  JDIMENSION col;
257
17.6k
  register int rindex = rgb_red[cinfo->in_color_space];
258
17.6k
  register int gindex = rgb_green[cinfo->in_color_space];
259
17.6k
  register int bindex = rgb_blue[cinfo->in_color_space];
260
17.6k
  register int aindex = alpha_index[cinfo->in_color_space];
261
17.6k
  register int ps = rgb_pixelsize[cinfo->in_color_space];
262
263
17.6k
  get_raw_row(cinfo, sinfo);
264
17.6k
  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
17.6k
  if (source->png_bit_depth == 16) {
276
8.66k
    unsigned short *bufferptr = (unsigned short *)source->iobuffer;
277
8.66k
    if (aindex >= 0)
278
1.73k
      RGB_READ_LOOP(rescale[*bufferptr++],
279
8.66k
                    ptr[aindex] = (1 << cinfo->data_precision) - 1;,
280
8.66k
                    bufferptr += source->png_alpha;)
281
6.92k
    else
282
6.92k
      RGB_READ_LOOP(rescale[*bufferptr++], {},
283
8.66k
                    bufferptr += source->png_alpha;)
284
8.96k
  } else {
285
8.96k
    register unsigned char *bufferptr = source->iobuffer;
286
8.96k
    if (aindex >= 0)
287
1.68k
      RGB_READ_LOOP(rescale[*bufferptr++],
288
8.96k
                    ptr[aindex] = (1 << cinfo->data_precision) - 1;,
289
8.96k
                    bufferptr += source->png_alpha;)
290
7.28k
    else
291
7.28k
      RGB_READ_LOOP(rescale[*bufferptr++], {},
292
8.96k
                    bufferptr += source->png_alpha;)
293
8.96k
  }
294
17.6k
  return 1;
295
17.6k
}
Unexecuted instantiation: rdpng-8.c:get_rgb_row
rdpng-12.c:get_rgb_row
Line
Count
Source
252
17.6k
{
253
17.6k
  png_source_ptr source = (png_source_ptr)sinfo;
254
17.6k
  register _JSAMPROW ptr;
255
17.6k
  register _JSAMPLE *rescale = source->rescale;
256
17.6k
  JDIMENSION col;
257
17.6k
  register int rindex = rgb_red[cinfo->in_color_space];
258
17.6k
  register int gindex = rgb_green[cinfo->in_color_space];
259
17.6k
  register int bindex = rgb_blue[cinfo->in_color_space];
260
17.6k
  register int aindex = alpha_index[cinfo->in_color_space];
261
17.6k
  register int ps = rgb_pixelsize[cinfo->in_color_space];
262
263
17.6k
  get_raw_row(cinfo, sinfo);
264
17.6k
  ptr = source->pub._buffer[0];
265
#if BITS_IN_JSAMPLE != 12
266
  if (source->png_bit_depth == cinfo->data_precision) {
267
    register _JSAMPLE *bufferptr = (_JSAMPLE *)source->iobuffer;
268
    if (aindex >= 0)
269
      RGB_READ_LOOP(*bufferptr++, ptr[aindex] = _MAXJSAMPLE;,
270
                    bufferptr += source->png_alpha;)
271
    else
272
      RGB_READ_LOOP(*bufferptr++, {}, bufferptr += source->png_alpha;)
273
  } else
274
#endif
275
17.6k
  if (source->png_bit_depth == 16) {
276
8.66k
    unsigned short *bufferptr = (unsigned short *)source->iobuffer;
277
8.66k
    if (aindex >= 0)
278
1.73k
      RGB_READ_LOOP(rescale[*bufferptr++],
279
8.66k
                    ptr[aindex] = (1 << cinfo->data_precision) - 1;,
280
8.66k
                    bufferptr += source->png_alpha;)
281
6.92k
    else
282
6.92k
      RGB_READ_LOOP(rescale[*bufferptr++], {},
283
8.66k
                    bufferptr += source->png_alpha;)
284
8.96k
  } else {
285
8.96k
    register unsigned char *bufferptr = source->iobuffer;
286
8.96k
    if (aindex >= 0)
287
1.68k
      RGB_READ_LOOP(rescale[*bufferptr++],
288
8.96k
                    ptr[aindex] = (1 << cinfo->data_precision) - 1;,
289
8.96k
                    bufferptr += source->png_alpha;)
290
7.28k
    else
291
7.28k
      RGB_READ_LOOP(rescale[*bufferptr++], {},
292
8.96k
                    bufferptr += source->png_alpha;)
293
8.96k
  }
294
17.6k
  return 1;
295
17.6k
}
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
3.52k
{
304
3.52k
  png_source_ptr source = (png_source_ptr)sinfo;
305
3.52k
  register _JSAMPROW ptr;
306
3.52k
  register _JSAMPLE *rescale = source->rescale;
307
3.52k
  JDIMENSION col;
308
309
3.52k
  get_raw_row(cinfo, sinfo);
310
3.52k
  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
3.52k
  if (source->png_bit_depth == 16) {
325
1.73k
    unsigned short *bufferptr = (unsigned short *)source->iobuffer;
326
168k
    for (col = cinfo->image_width; col > 0; col--) {
327
166k
      _JSAMPLE r = rescale[*bufferptr++];
328
166k
      _JSAMPLE g = rescale[*bufferptr++];
329
166k
      _JSAMPLE b = rescale[*bufferptr++];
330
166k
      bufferptr += source->png_alpha;
331
166k
      rgb_to_cmyk((1 << cinfo->data_precision) - 1, r, g, b, ptr, ptr + 1,
332
166k
                  ptr + 2, ptr + 3);
333
166k
      ptr += 4;
334
166k
    }
335
1.79k
  } else {
336
1.79k
    register unsigned char *bufferptr = source->iobuffer;
337
159k
    for (col = cinfo->image_width; col > 0; col--) {
338
157k
      _JSAMPLE r = rescale[*bufferptr++];
339
157k
      _JSAMPLE g = rescale[*bufferptr++];
340
157k
      _JSAMPLE b = rescale[*bufferptr++];
341
157k
      bufferptr += source->png_alpha;
342
157k
      rgb_to_cmyk((1 << cinfo->data_precision) - 1, r, g, b, ptr, ptr + 1,
343
157k
                  ptr + 2, ptr + 3);
344
157k
      ptr += 4;
345
157k
    }
346
1.79k
  }
347
3.52k
  return 1;
348
3.52k
}
Unexecuted instantiation: rdpng-8.c:get_rgb_cmyk_row
rdpng-12.c:get_rgb_cmyk_row
Line
Count
Source
303
3.52k
{
304
3.52k
  png_source_ptr source = (png_source_ptr)sinfo;
305
3.52k
  register _JSAMPROW ptr;
306
3.52k
  register _JSAMPLE *rescale = source->rescale;
307
3.52k
  JDIMENSION col;
308
309
3.52k
  get_raw_row(cinfo, sinfo);
310
3.52k
  ptr = source->pub._buffer[0];
311
#if BITS_IN_JSAMPLE != 12
312
  if (source->png_bit_depth == cinfo->data_precision) {
313
    register _JSAMPLE *bufferptr = (_JSAMPLE *)source->iobuffer;
314
    for (col = cinfo->image_width; col > 0; col--) {
315
      _JSAMPLE r = *bufferptr++;
316
      _JSAMPLE g = *bufferptr++;
317
      _JSAMPLE b = *bufferptr++;
318
      bufferptr += source->png_alpha;
319
      rgb_to_cmyk(_MAXJSAMPLE, r, g, b, ptr, ptr + 1, ptr + 2, ptr + 3);
320
      ptr += 4;
321
    }
322
  } else
323
#endif
324
3.52k
  if (source->png_bit_depth == 16) {
325
1.73k
    unsigned short *bufferptr = (unsigned short *)source->iobuffer;
326
168k
    for (col = cinfo->image_width; col > 0; col--) {
327
166k
      _JSAMPLE r = rescale[*bufferptr++];
328
166k
      _JSAMPLE g = rescale[*bufferptr++];
329
166k
      _JSAMPLE b = rescale[*bufferptr++];
330
166k
      bufferptr += source->png_alpha;
331
166k
      rgb_to_cmyk((1 << cinfo->data_precision) - 1, r, g, b, ptr, ptr + 1,
332
166k
                  ptr + 2, ptr + 3);
333
166k
      ptr += 4;
334
166k
    }
335
1.79k
  } else {
336
1.79k
    register unsigned char *bufferptr = source->iobuffer;
337
159k
    for (col = cinfo->image_width; col > 0; col--) {
338
157k
      _JSAMPLE r = rescale[*bufferptr++];
339
157k
      _JSAMPLE g = rescale[*bufferptr++];
340
157k
      _JSAMPLE b = rescale[*bufferptr++];
341
157k
      bufferptr += source->png_alpha;
342
157k
      rgb_to_cmyk((1 << cinfo->data_precision) - 1, r, g, b, ptr, ptr + 1,
343
157k
                  ptr + 2, ptr + 3);
344
157k
      ptr += 4;
345
157k
    }
346
1.79k
  }
347
3.52k
  return 1;
348
3.52k
}
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
1.05k
{
354
/* This version is for reading 8-bit-per-channel indexed-color PNG files and
355
 * converting to extended RGB or CMYK.
356
 */
357
1.05k
  png_source_ptr source = (png_source_ptr)sinfo;
358
1.05k
  register _JSAMPROW ptr;
359
1.05k
  register JSAMPLE *bufferptr = (JSAMPLE *)source->iobuffer;
360
1.05k
  register _JSAMPLE *rescale = source->rescale;
361
1.05k
  JDIMENSION col;
362
363
1.05k
  get_raw_row(cinfo, sinfo);
364
1.05k
  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
1.05k
    if (cinfo->in_color_space == JCS_GRAYSCALE) {
411
841
      for (col = cinfo->image_width; col > 0; col--) {
412
745
        JSAMPLE index = *bufferptr++;
413
414
745
        if (index >= source->colormap.n_entries)
415
9
          ERREXIT(cinfo, JERR_PNG_OUTOFRANGE);
416
745
        *ptr++ = rescale[source->colormap.entries[index].red];
417
745
      }
418
962
    } else if (cinfo->in_color_space == JCS_CMYK) {
419
1.22k
      for (col = cinfo->image_width; col > 0; col--) {
420
1.08k
        JSAMPLE index = *bufferptr++;
421
422
1.08k
        if (index >= source->colormap.n_entries)
423
14
          ERREXIT(cinfo, JERR_PNG_OUTOFRANGE);
424
1.08k
        rgb_to_cmyk((1 << cinfo->data_precision) - 1,
425
1.08k
                    rescale[source->colormap.entries[index].red],
426
1.08k
                    rescale[source->colormap.entries[index].green],
427
1.08k
                    rescale[source->colormap.entries[index].blue], ptr,
428
1.08k
                    ptr + 1, ptr + 2, ptr + 3);
429
1.08k
        ptr += 4;
430
1.08k
      }
431
821
    } else {
432
821
      register int rindex = rgb_red[cinfo->in_color_space];
433
821
      register int gindex = rgb_green[cinfo->in_color_space];
434
821
      register int bindex = rgb_blue[cinfo->in_color_space];
435
821
      register int aindex = alpha_index[cinfo->in_color_space];
436
821
      register int ps = rgb_pixelsize[cinfo->in_color_space];
437
438
6.25k
      for (col = cinfo->image_width; col > 0; col--) {
439
5.43k
        JSAMPLE index = *bufferptr++;
440
441
5.43k
        if (index >= source->colormap.n_entries)
442
70
          ERREXIT(cinfo, JERR_PNG_OUTOFRANGE);
443
5.43k
        ptr[rindex] = rescale[source->colormap.entries[index].red];
444
5.43k
        ptr[gindex] = rescale[source->colormap.entries[index].green];
445
5.43k
        ptr[bindex] = rescale[source->colormap.entries[index].blue];
446
5.43k
        if (aindex >= 0)
447
1.07k
          ptr[aindex] = (1 << cinfo->data_precision) - 1;
448
5.43k
        ptr += ps;
449
5.43k
      }
450
821
    }
451
0
  }
452
1.05k
  return 1;
453
1.05k
}
Unexecuted instantiation: rdpng-8.c:get_indexed_row
rdpng-12.c:get_indexed_row
Line
Count
Source
353
1.05k
{
354
/* This version is for reading 8-bit-per-channel indexed-color PNG files and
355
 * converting to extended RGB or CMYK.
356
 */
357
1.05k
  png_source_ptr source = (png_source_ptr)sinfo;
358
1.05k
  register _JSAMPROW ptr;
359
1.05k
  register JSAMPLE *bufferptr = (JSAMPLE *)source->iobuffer;
360
1.05k
  register _JSAMPLE *rescale = source->rescale;
361
1.05k
  JDIMENSION col;
362
363
1.05k
  get_raw_row(cinfo, sinfo);
364
1.05k
  ptr = source->pub._buffer[0];
365
#if BITS_IN_JSAMPLE == 8
366
  if (source->png_bit_depth == cinfo->data_precision) {
367
    if (cinfo->in_color_space == JCS_GRAYSCALE) {
368
      for (col = cinfo->image_width; col > 0; col--) {
369
        JSAMPLE index = *bufferptr++;
370
371
        if (index >= source->colormap.n_entries)
372
          ERREXIT(cinfo, JERR_PNG_OUTOFRANGE);
373
        *ptr++ = source->colormap.entries[index].red;
374
      }
375
    } else if (cinfo->in_color_space == JCS_CMYK) {
376
      for (col = cinfo->image_width; col > 0; col--) {
377
        JSAMPLE index = *bufferptr++;
378
379
        if (index >= source->colormap.n_entries)
380
          ERREXIT(cinfo, JERR_PNG_OUTOFRANGE);
381
        rgb_to_cmyk(_MAXJSAMPLE, source->colormap.entries[index].red,
382
                    source->colormap.entries[index].green,
383
                    source->colormap.entries[index].blue, ptr, ptr + 1,
384
                    ptr + 2, ptr + 3);
385
        ptr += 4;
386
      }
387
    } else {
388
      register int rindex = rgb_red[cinfo->in_color_space];
389
      register int gindex = rgb_green[cinfo->in_color_space];
390
      register int bindex = rgb_blue[cinfo->in_color_space];
391
      register int aindex = alpha_index[cinfo->in_color_space];
392
      register int ps = rgb_pixelsize[cinfo->in_color_space];
393
394
      for (col = cinfo->image_width; col > 0; col--) {
395
        JSAMPLE index = *bufferptr++;
396
397
        if (index >= source->colormap.n_entries)
398
          ERREXIT(cinfo, JERR_PNG_OUTOFRANGE);
399
        ptr[rindex] = source->colormap.entries[index].red;
400
        ptr[gindex] = source->colormap.entries[index].green;
401
        ptr[bindex] = source->colormap.entries[index].blue;
402
        if (aindex >= 0)
403
          ptr[aindex] = _MAXJSAMPLE;
404
        ptr += ps;
405
      }
406
    }
407
  } else
408
#endif
409
1.05k
  {
410
1.05k
    if (cinfo->in_color_space == JCS_GRAYSCALE) {
411
841
      for (col = cinfo->image_width; col > 0; col--) {
412
745
        JSAMPLE index = *bufferptr++;
413
414
745
        if (index >= source->colormap.n_entries)
415
9
          ERREXIT(cinfo, JERR_PNG_OUTOFRANGE);
416
745
        *ptr++ = rescale[source->colormap.entries[index].red];
417
745
      }
418
962
    } else if (cinfo->in_color_space == JCS_CMYK) {
419
1.22k
      for (col = cinfo->image_width; col > 0; col--) {
420
1.08k
        JSAMPLE index = *bufferptr++;
421
422
1.08k
        if (index >= source->colormap.n_entries)
423
14
          ERREXIT(cinfo, JERR_PNG_OUTOFRANGE);
424
1.08k
        rgb_to_cmyk((1 << cinfo->data_precision) - 1,
425
1.08k
                    rescale[source->colormap.entries[index].red],
426
1.08k
                    rescale[source->colormap.entries[index].green],
427
1.08k
                    rescale[source->colormap.entries[index].blue], ptr,
428
1.08k
                    ptr + 1, ptr + 2, ptr + 3);
429
1.08k
        ptr += 4;
430
1.08k
      }
431
821
    } else {
432
821
      register int rindex = rgb_red[cinfo->in_color_space];
433
821
      register int gindex = rgb_green[cinfo->in_color_space];
434
821
      register int bindex = rgb_blue[cinfo->in_color_space];
435
821
      register int aindex = alpha_index[cinfo->in_color_space];
436
821
      register int ps = rgb_pixelsize[cinfo->in_color_space];
437
438
6.25k
      for (col = cinfo->image_width; col > 0; col--) {
439
5.43k
        JSAMPLE index = *bufferptr++;
440
441
5.43k
        if (index >= source->colormap.n_entries)
442
70
          ERREXIT(cinfo, JERR_PNG_OUTOFRANGE);
443
5.43k
        ptr[rindex] = rescale[source->colormap.entries[index].red];
444
5.43k
        ptr[gindex] = rescale[source->colormap.entries[index].green];
445
5.43k
        ptr[bindex] = rescale[source->colormap.entries[index].blue];
446
5.43k
        if (aindex >= 0)
447
1.07k
          ptr[aindex] = (1 << cinfo->data_precision) - 1;
448
5.43k
        ptr += ps;
449
5.43k
      }
450
821
    }
451
1.05k
  }
452
1.05k
  return 1;
453
1.05k
}
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
18.5k
{
473
18.5k
  png_source_ptr source = (png_source_ptr)sinfo;
474
18.5k
  struct spng_ihdr ihdr;
475
18.5k
  int png_components = 3;
476
18.5k
  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
18.5k
  source->ctx = spng_ctx_new(0);
487
18.5k
#endif
488
18.5k
  if (!source->ctx)
489
0
    ERREXITS(cinfo, JERR_PNG_LIBSPNG, "Could not create context");
490
491
18.5k
  TRY_SPNG(spng_set_png_file(source->ctx, sinfo->input_file));
492
493
18.5k
  TRY_SPNG(spng_get_ihdr(source->ctx, &ihdr));
494
495
18.5k
  if (ihdr.width > JPEG_MAX_DIMENSION || ihdr.height > JPEG_MAX_DIMENSION)
496
3.56k
    ERREXIT1(cinfo, JERR_IMAGE_TOO_BIG, JPEG_MAX_DIMENSION);
497
18.5k
  if (ihdr.bit_depth != 8 && ihdr.bit_depth != 16)
498
3.88k
    ERREXIT(cinfo, JERR_PNG_BADDEPTH);
499
18.5k
  if (sinfo->max_pixels &&
500
9.53k
      (unsigned long long)ihdr.width * ihdr.height > sinfo->max_pixels)
501
161
    ERREXIT1(cinfo, JERR_IMAGE_TOO_BIG, sinfo->max_pixels);
502
503
18.5k
  cinfo->image_width = (JDIMENSION)ihdr.width;
504
18.5k
  cinfo->image_height = (JDIMENSION)ihdr.height;
505
18.5k
  source->png_bit_depth = ihdr.bit_depth;
506
18.5k
  source->png_color_type = ihdr.color_type;
507
508
  /* initialize flags to most common settings */
509
18.5k
  use_raw_buffer = FALSE;       /* do we map input buffer onto I/O buffer? */
510
511
18.5k
  switch (ihdr.color_type) {
512
7
  case SPNG_COLOR_TYPE_GRAYSCALE_ALPHA:
513
4.07k
  case SPNG_COLOR_TYPE_GRAYSCALE:
514
4.07k
    if (cinfo->in_color_space == JCS_UNKNOWN ||
515
4.07k
        cinfo->in_color_space == JCS_RGB)
516
0
      cinfo->in_color_space = JCS_GRAYSCALE;
517
4.07k
    TRACEMS3(cinfo, 1, JTRC_PNG_GRAYSCALE, ihdr.width, ihdr.height,
518
4.07k
             ihdr.bit_depth);
519
4.07k
    if (cinfo->in_color_space == JCS_GRAYSCALE) {
520
582
      if (ihdr.color_type == SPNG_COLOR_TYPE_GRAYSCALE &&
521
581
          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
582
        source->pub.get_pixel_rows = get_gray_row;
526
3.49k
    } else if (IsExtRGB(cinfo->in_color_space))
527
2.91k
      source->pub.get_pixel_rows = get_gray_rgb_row;
528
582
    else if (cinfo->in_color_space == JCS_CMYK)
529
582
        source->pub.get_pixel_rows = get_gray_cmyk_row;
530
0
    else
531
0
      ERREXIT(cinfo, JERR_BAD_IN_COLORSPACE);
532
4.07k
    if (ihdr.color_type == SPNG_COLOR_TYPE_GRAYSCALE_ALPHA) {
533
7
      png_components = 2;
534
7
      source->png_alpha = 1;
535
4.06k
    } else {
536
4.06k
      png_components = 1;
537
4.06k
      source->png_alpha = 0;
538
4.06k
    }
539
4.07k
    break;
540
541
4.70k
  case SPNG_COLOR_TYPE_TRUECOLOR:
542
4.71k
  case SPNG_COLOR_TYPE_TRUECOLOR_ALPHA:
543
4.71k
    if (cinfo->in_color_space == JCS_UNKNOWN)
544
0
      cinfo->in_color_space = JCS_EXT_RGB;
545
4.71k
    TRACEMS3(cinfo, 1, JTRC_PNG_TRUECOLOR, ihdr.width, ihdr.height,
546
4.71k
             ihdr.bit_depth);
547
4.71k
    if (ihdr.color_type == SPNG_COLOR_TYPE_TRUECOLOR &&
548
4.70k
        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
4.71k
    } else if (IsExtRGB(cinfo->in_color_space))
558
3.36k
      source->pub.get_pixel_rows = get_rgb_row;
559
1.34k
    else if (cinfo->in_color_space == JCS_CMYK)
560
673
      source->pub.get_pixel_rows = get_rgb_cmyk_row;
561
673
    else
562
673
      ERREXIT(cinfo, JERR_BAD_IN_COLORSPACE);
563
4.71k
    if (ihdr.color_type == SPNG_COLOR_TYPE_TRUECOLOR_ALPHA) {
564
6
      png_components = 4;
565
6
      source->png_alpha = 1;
566
4.70k
    } else {
567
4.70k
      png_components = 3;
568
4.70k
      source->png_alpha = 0;
569
4.70k
    }
570
4.71k
    break;
571
572
588
  case SPNG_COLOR_TYPE_INDEXED:
573
588
  {
574
588
    int i, gray = 1;
575
576
588
    TRACEMS3(cinfo, 1, JTRC_PNG_INDEXED, ihdr.width, ihdr.height,
577
588
             ihdr.bit_depth);
578
588
    TRY_SPNG(spng_get_plte(source->ctx, &source->colormap));
579
588
    if (source->png_bit_depth != 8 || source->colormap.n_entries > 256)
580
0
      ERREXIT(cinfo, JERR_PNG_OUTOFRANGE);
581
582
16.3k
    for (i = 0; i < (int)source->colormap.n_entries; i++) {
583
15.8k
      if (source->colormap.entries[i].red !=
584
15.8k
          source->colormap.entries[i].green ||
585
8.63k
          source->colormap.entries[i].green !=
586
8.63k
          source->colormap.entries[i].blue)
587
7.16k
        gray = 0;
588
15.8k
    }
589
590
588
    if ((cinfo->in_color_space == JCS_UNKNOWN ||
591
259
         cinfo->in_color_space == JCS_RGB) && gray)
592
0
      cinfo->in_color_space = JCS_GRAYSCALE;
593
588
    if (cinfo->in_color_space == JCS_GRAYSCALE && !gray)
594
16
      ERREXIT(cinfo, JERR_BAD_IN_COLORSPACE);
595
596
588
    source->pub.get_pixel_rows = get_indexed_row;
597
588
    png_components = 1;
598
588
    source->png_alpha = 0;
599
588
    break;
600
4.70k
  }
601
602
0
  default:
603
0
    ERREXIT(cinfo, JERR_PNG_OUTOFRANGE);
604
18.5k
  }
605
606
8.35k
  if (IsExtRGB(cinfo->in_color_space))
607
6.46k
    cinfo->input_components = rgb_pixelsize[cinfo->in_color_space];
608
1.89k
  else if (cinfo->in_color_space == JCS_GRAYSCALE)
609
603
    cinfo->input_components = 1;
610
1.29k
  else if (cinfo->in_color_space == JCS_CMYK)
611
1.29k
    cinfo->input_components = 4;
612
613
  /* Allocate space for I/O buffer: 1 or 3 bytes or words/pixel. */
614
8.35k
  source->buffer_width =
615
8.35k
    (size_t)ihdr.width * png_components * source->png_bit_depth / 8;
616
8.35k
  source->iobuffer = (unsigned char *)
617
8.35k
    (*cinfo->mem->alloc_small) ((j_common_ptr)cinfo, JPOOL_IMAGE,
618
8.35k
                                source->buffer_width);
619
620
  /* Create compressor input buffer. */
621
8.35k
  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
8.35k
  } else {
628
8.35k
    unsigned int maxval = source->png_bit_depth == 16 ? 65535 : 255;
629
8.35k
    size_t val, half_maxval;
630
631
    /* Need to translate anyway, so make a separate sample buffer. */
632
8.35k
    source->pub._buffer = (_JSAMPARRAY)(*cinfo->mem->alloc_sarray)
633
8.35k
      ((j_common_ptr)cinfo, JPOOL_IMAGE,
634
8.35k
       (JDIMENSION)ihdr.width * cinfo->input_components, (JDIMENSION)1);
635
8.35k
    source->pub.buffer_height = 1;
636
637
    /* Compute the rescaling array. */
638
8.35k
    source->rescale = (_JSAMPLE *)
639
8.35k
      (*cinfo->mem->alloc_small) ((j_common_ptr)cinfo, JPOOL_IMAGE,
640
8.35k
                                  (maxval + 1L) * sizeof(_JSAMPLE));
641
8.35k
    memset(source->rescale, 0, (maxval + 1L) * sizeof(_JSAMPLE));
642
8.35k
    half_maxval = maxval / 2;
643
58.9M
    for (val = 0; val <= maxval; val++) {
644
      /* The multiplication here must be done in 32 bits to avoid overflow */
645
58.9M
      source->rescale[val] =
646
58.9M
        (_JSAMPLE)((val * ((1 << cinfo->data_precision) - 1) + half_maxval) /
647
58.9M
                   maxval);
648
58.9M
    }
649
8.35k
  }
650
651
8.35k
  TRY_SPNG(spng_decode_image(source->ctx, NULL, 0, SPNG_FMT_PNG,
652
8.35k
                             SPNG_DECODE_PROGRESSIVE));
653
8.35k
}
Unexecuted instantiation: rdpng-8.c:start_input_png
rdpng-12.c:start_input_png
Line
Count
Source
472
18.5k
{
473
18.5k
  png_source_ptr source = (png_source_ptr)sinfo;
474
18.5k
  struct spng_ihdr ihdr;
475
18.5k
  int png_components = 3;
476
18.5k
  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
18.5k
  source->ctx = spng_ctx_new(0);
487
18.5k
#endif
488
18.5k
  if (!source->ctx)
489
0
    ERREXITS(cinfo, JERR_PNG_LIBSPNG, "Could not create context");
490
491
18.5k
  TRY_SPNG(spng_set_png_file(source->ctx, sinfo->input_file));
492
493
18.5k
  TRY_SPNG(spng_get_ihdr(source->ctx, &ihdr));
494
495
18.5k
  if (ihdr.width > JPEG_MAX_DIMENSION || ihdr.height > JPEG_MAX_DIMENSION)
496
3.56k
    ERREXIT1(cinfo, JERR_IMAGE_TOO_BIG, JPEG_MAX_DIMENSION);
497
18.5k
  if (ihdr.bit_depth != 8 && ihdr.bit_depth != 16)
498
3.88k
    ERREXIT(cinfo, JERR_PNG_BADDEPTH);
499
18.5k
  if (sinfo->max_pixels &&
500
9.53k
      (unsigned long long)ihdr.width * ihdr.height > sinfo->max_pixels)
501
161
    ERREXIT1(cinfo, JERR_IMAGE_TOO_BIG, sinfo->max_pixels);
502
503
18.5k
  cinfo->image_width = (JDIMENSION)ihdr.width;
504
18.5k
  cinfo->image_height = (JDIMENSION)ihdr.height;
505
18.5k
  source->png_bit_depth = ihdr.bit_depth;
506
18.5k
  source->png_color_type = ihdr.color_type;
507
508
  /* initialize flags to most common settings */
509
18.5k
  use_raw_buffer = FALSE;       /* do we map input buffer onto I/O buffer? */
510
511
18.5k
  switch (ihdr.color_type) {
512
7
  case SPNG_COLOR_TYPE_GRAYSCALE_ALPHA:
513
4.07k
  case SPNG_COLOR_TYPE_GRAYSCALE:
514
4.07k
    if (cinfo->in_color_space == JCS_UNKNOWN ||
515
4.07k
        cinfo->in_color_space == JCS_RGB)
516
0
      cinfo->in_color_space = JCS_GRAYSCALE;
517
4.07k
    TRACEMS3(cinfo, 1, JTRC_PNG_GRAYSCALE, ihdr.width, ihdr.height,
518
4.07k
             ihdr.bit_depth);
519
4.07k
    if (cinfo->in_color_space == JCS_GRAYSCALE) {
520
582
      if (ihdr.color_type == SPNG_COLOR_TYPE_GRAYSCALE &&
521
581
          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
582
        source->pub.get_pixel_rows = get_gray_row;
526
3.49k
    } else if (IsExtRGB(cinfo->in_color_space))
527
2.91k
      source->pub.get_pixel_rows = get_gray_rgb_row;
528
582
    else if (cinfo->in_color_space == JCS_CMYK)
529
582
        source->pub.get_pixel_rows = get_gray_cmyk_row;
530
0
    else
531
0
      ERREXIT(cinfo, JERR_BAD_IN_COLORSPACE);
532
4.07k
    if (ihdr.color_type == SPNG_COLOR_TYPE_GRAYSCALE_ALPHA) {
533
7
      png_components = 2;
534
7
      source->png_alpha = 1;
535
4.06k
    } else {
536
4.06k
      png_components = 1;
537
4.06k
      source->png_alpha = 0;
538
4.06k
    }
539
4.07k
    break;
540
541
4.70k
  case SPNG_COLOR_TYPE_TRUECOLOR:
542
4.71k
  case SPNG_COLOR_TYPE_TRUECOLOR_ALPHA:
543
4.71k
    if (cinfo->in_color_space == JCS_UNKNOWN)
544
0
      cinfo->in_color_space = JCS_EXT_RGB;
545
4.71k
    TRACEMS3(cinfo, 1, JTRC_PNG_TRUECOLOR, ihdr.width, ihdr.height,
546
4.71k
             ihdr.bit_depth);
547
4.71k
    if (ihdr.color_type == SPNG_COLOR_TYPE_TRUECOLOR &&
548
4.70k
        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
4.71k
    } else if (IsExtRGB(cinfo->in_color_space))
558
3.36k
      source->pub.get_pixel_rows = get_rgb_row;
559
1.34k
    else if (cinfo->in_color_space == JCS_CMYK)
560
673
      source->pub.get_pixel_rows = get_rgb_cmyk_row;
561
673
    else
562
673
      ERREXIT(cinfo, JERR_BAD_IN_COLORSPACE);
563
4.71k
    if (ihdr.color_type == SPNG_COLOR_TYPE_TRUECOLOR_ALPHA) {
564
6
      png_components = 4;
565
6
      source->png_alpha = 1;
566
4.70k
    } else {
567
4.70k
      png_components = 3;
568
4.70k
      source->png_alpha = 0;
569
4.70k
    }
570
4.71k
    break;
571
572
588
  case SPNG_COLOR_TYPE_INDEXED:
573
588
  {
574
588
    int i, gray = 1;
575
576
588
    TRACEMS3(cinfo, 1, JTRC_PNG_INDEXED, ihdr.width, ihdr.height,
577
588
             ihdr.bit_depth);
578
588
    TRY_SPNG(spng_get_plte(source->ctx, &source->colormap));
579
588
    if (source->png_bit_depth != 8 || source->colormap.n_entries > 256)
580
0
      ERREXIT(cinfo, JERR_PNG_OUTOFRANGE);
581
582
16.3k
    for (i = 0; i < (int)source->colormap.n_entries; i++) {
583
15.8k
      if (source->colormap.entries[i].red !=
584
15.8k
          source->colormap.entries[i].green ||
585
8.63k
          source->colormap.entries[i].green !=
586
8.63k
          source->colormap.entries[i].blue)
587
7.16k
        gray = 0;
588
15.8k
    }
589
590
588
    if ((cinfo->in_color_space == JCS_UNKNOWN ||
591
259
         cinfo->in_color_space == JCS_RGB) && gray)
592
0
      cinfo->in_color_space = JCS_GRAYSCALE;
593
588
    if (cinfo->in_color_space == JCS_GRAYSCALE && !gray)
594
16
      ERREXIT(cinfo, JERR_BAD_IN_COLORSPACE);
595
596
588
    source->pub.get_pixel_rows = get_indexed_row;
597
588
    png_components = 1;
598
588
    source->png_alpha = 0;
599
588
    break;
600
4.70k
  }
601
602
0
  default:
603
0
    ERREXIT(cinfo, JERR_PNG_OUTOFRANGE);
604
18.5k
  }
605
606
8.35k
  if (IsExtRGB(cinfo->in_color_space))
607
6.46k
    cinfo->input_components = rgb_pixelsize[cinfo->in_color_space];
608
1.89k
  else if (cinfo->in_color_space == JCS_GRAYSCALE)
609
603
    cinfo->input_components = 1;
610
1.29k
  else if (cinfo->in_color_space == JCS_CMYK)
611
1.29k
    cinfo->input_components = 4;
612
613
  /* Allocate space for I/O buffer: 1 or 3 bytes or words/pixel. */
614
8.35k
  source->buffer_width =
615
8.35k
    (size_t)ihdr.width * png_components * source->png_bit_depth / 8;
616
8.35k
  source->iobuffer = (unsigned char *)
617
8.35k
    (*cinfo->mem->alloc_small) ((j_common_ptr)cinfo, JPOOL_IMAGE,
618
8.35k
                                source->buffer_width);
619
620
  /* Create compressor input buffer. */
621
8.35k
  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
8.35k
  } else {
628
8.35k
    unsigned int maxval = source->png_bit_depth == 16 ? 65535 : 255;
629
8.35k
    size_t val, half_maxval;
630
631
    /* Need to translate anyway, so make a separate sample buffer. */
632
8.35k
    source->pub._buffer = (_JSAMPARRAY)(*cinfo->mem->alloc_sarray)
633
8.35k
      ((j_common_ptr)cinfo, JPOOL_IMAGE,
634
8.35k
       (JDIMENSION)ihdr.width * cinfo->input_components, (JDIMENSION)1);
635
8.35k
    source->pub.buffer_height = 1;
636
637
    /* Compute the rescaling array. */
638
8.35k
    source->rescale = (_JSAMPLE *)
639
8.35k
      (*cinfo->mem->alloc_small) ((j_common_ptr)cinfo, JPOOL_IMAGE,
640
8.35k
                                  (maxval + 1L) * sizeof(_JSAMPLE));
641
8.35k
    memset(source->rescale, 0, (maxval + 1L) * sizeof(_JSAMPLE));
642
8.35k
    half_maxval = maxval / 2;
643
58.9M
    for (val = 0; val <= maxval; val++) {
644
      /* The multiplication here must be done in 32 bits to avoid overflow */
645
58.9M
      source->rescale[val] =
646
58.9M
        (_JSAMPLE)((val * ((1 << cinfo->data_precision) - 1) + half_maxval) /
647
58.9M
                   maxval);
648
58.9M
    }
649
8.35k
  }
650
651
8.35k
  TRY_SPNG(spng_decode_image(source->ctx, NULL, 0, SPNG_FMT_PNG,
652
8.35k
                             SPNG_DECODE_PROGRESSIVE));
653
8.35k
}
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
1.95k
{
664
1.95k
  png_source_ptr source = (png_source_ptr)sinfo;
665
1.95k
  struct spng_iccp iccp;
666
667
1.95k
  if (!icc_data_ptr || !icc_data_len)
668
0
    ERREXIT(cinfo, JERR_BUFFER_SIZE);
669
670
1.95k
  if (source->ctx && spng_get_iccp(source->ctx, &iccp) == 0) {
671
12
    *icc_data_ptr = (JOCTET *)iccp.profile;
672
12
    *icc_data_len = (unsigned int)iccp.profile_len;
673
12
    return TRUE;
674
12
  }
675
676
1.94k
  return FALSE;
677
1.95k
}
Unexecuted instantiation: rdpng-8.c:read_icc_profile_png
rdpng-12.c:read_icc_profile_png
Line
Count
Source
663
1.95k
{
664
1.95k
  png_source_ptr source = (png_source_ptr)sinfo;
665
1.95k
  struct spng_iccp iccp;
666
667
1.95k
  if (!icc_data_ptr || !icc_data_len)
668
0
    ERREXIT(cinfo, JERR_BUFFER_SIZE);
669
670
1.95k
  if (source->ctx && spng_get_iccp(source->ctx, &iccp) == 0) {
671
12
    *icc_data_ptr = (JOCTET *)iccp.profile;
672
12
    *icc_data_len = (unsigned int)iccp.profile_len;
673
12
    return TRUE;
674
12
  }
675
676
1.94k
  return FALSE;
677
1.95k
}
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
18.5k
{
687
18.5k
  png_source_ptr source = (png_source_ptr)sinfo;
688
689
18.5k
  if (source->ctx) {
690
18.5k
    spng_decode_chunks(source->ctx);
691
18.5k
    spng_ctx_free(source->ctx);
692
18.5k
    source->ctx = NULL;
693
18.5k
  }
694
18.5k
}
Unexecuted instantiation: rdpng-8.c:finish_input_png
rdpng-12.c:finish_input_png
Line
Count
Source
686
18.5k
{
687
18.5k
  png_source_ptr source = (png_source_ptr)sinfo;
688
689
18.5k
  if (source->ctx) {
690
18.5k
    spng_decode_chunks(source->ctx);
691
18.5k
    spng_ctx_free(source->ctx);
692
    source->ctx = NULL;
693
18.5k
  }
694
18.5k
}
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
18.5k
{
704
18.5k
  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
18.5k
  if (cinfo->data_precision > BITS_IN_JSAMPLE ||
710
18.5k
      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
18.5k
  source = (png_source_ptr)
716
18.5k
    (*cinfo->mem->alloc_small) ((j_common_ptr)cinfo, JPOOL_IMAGE,
717
18.5k
                                sizeof(png_source_struct));
718
  /* Fill in method ptrs, except get_pixel_rows which start_input sets */
719
18.5k
  source->pub.start_input = start_input_png;
720
18.5k
  source->pub.read_icc_profile = read_icc_profile_png;
721
18.5k
  source->pub.finish_input = finish_input_png;
722
18.5k
  source->pub.max_pixels = 0;
723
724
18.5k
  return (cjpeg_source_ptr)source;
725
18.5k
}
Unexecuted instantiation: jinit_read_png
j12init_read_png
Line
Count
Source
703
18.5k
{
704
18.5k
  png_source_ptr source;
705
706
#if BITS_IN_JSAMPLE == 8
707
  if (cinfo->data_precision > BITS_IN_JSAMPLE || cinfo->data_precision < 2)
708
#else
709
18.5k
  if (cinfo->data_precision > BITS_IN_JSAMPLE ||
710
18.5k
      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
18.5k
  source = (png_source_ptr)
716
18.5k
    (*cinfo->mem->alloc_small) ((j_common_ptr)cinfo, JPOOL_IMAGE,
717
18.5k
                                sizeof(png_source_struct));
718
  /* Fill in method ptrs, except get_pixel_rows which start_input sets */
719
18.5k
  source->pub.start_input = start_input_png;
720
18.5k
  source->pub.read_icc_profile = read_icc_profile_png;
721
18.5k
  source->pub.finish_input = finish_input_png;
722
18.5k
  source->pub.max_pixels = 0;
723
724
18.5k
  return (cjpeg_source_ptr)source;
725
18.5k
}
Unexecuted instantiation: j16init_read_png
726
727
#endif /* defined(PNG_SUPPORTED) &&
728
          (BITS_IN_JSAMPLE != 16 || defined(C_LOSSLESS_SUPPORTED)) */