blob: ecef521f37b450574bf2b30cc3b7e44a1c2d753c [file] [log] [blame]
Andreas Dilger47a0c421997-05-16 02:46:07 -05001
2/* pngset.c - storage of image information into info struct
Glenn Randers-Pehrsonb6ce43d1998-01-01 07:13:13 -06003 *
Glenn Randers-Pehrsond60b8fa2006-04-20 21:31:14 -05004 * Last changed in libpng 1.4.0 April 20, 2006
Glenn Randers-Pehrsonb6ce43d1998-01-01 07:13:13 -06005 * For conditions of distribution and use, see copyright notice in png.h
Glenn Randers-Pehrson9c3ab682006-02-20 22:09:05 -06006 * Copyright (c) 1998-2006 Glenn Randers-Pehrson
Glenn Randers-Pehrsond4366722000-06-04 14:29:29 -05007 * (Version 0.96 Copyright (c) 1996, 1997 Andreas Dilger)
8 * (Version 0.88 Copyright (c) 1995, 1996 Guy Eric Schalnat, Group 42, Inc.)
Glenn Randers-Pehrsonb6ce43d1998-01-01 07:13:13 -06009 *
10 * The functions here are used during reads to store data from the file
11 * into the info struct, and during writes to store application data
12 * into the info struct for writing into the file. This abstracts the
13 * info struct and allows us to change the structure in the future.
14 */
Andreas Dilger47a0c421997-05-16 02:46:07 -050015
Andreas Dilger47a0c421997-05-16 02:46:07 -050016#include "png.h"
Glenn Randers-Pehrsonbeb572e2006-08-19 13:59:24 -050017#include "pngpriv.h"
Andreas Dilger47a0c421997-05-16 02:46:07 -050018
Glenn Randers-Pehrson9c3ab682006-02-20 22:09:05 -060019#if defined(PNG_READ_SUPPORTED) || defined(PNG_WRITE_SUPPORTED)
20
Glenn Randers-Pehrson166c5a31999-12-10 09:43:02 -060021#if defined(PNG_bKGD_SUPPORTED)
Glenn Randers-Pehrson75294572000-05-06 14:09:57 -050022void PNGAPI
Andreas Dilger47a0c421997-05-16 02:46:07 -050023png_set_bKGD(png_structp png_ptr, png_infop info_ptr, png_color_16p background)
24{
25 png_debug1(1, "in %s storage function\n", "bKGD");
Glenn Randers-Pehrsona357b991998-02-08 20:56:40 -060026 if (png_ptr == NULL || info_ptr == NULL)
Andreas Dilger47a0c421997-05-16 02:46:07 -050027 return;
28
Glenn Randers-Pehrsonbeb572e2006-08-19 13:59:24 -050029 png_memcpy(&(info_ptr->background), background, sizeof(png_color_16));
Andreas Dilger47a0c421997-05-16 02:46:07 -050030 info_ptr->valid |= PNG_INFO_bKGD;
31}
32#endif
33
Glenn Randers-Pehrson166c5a31999-12-10 09:43:02 -060034#if defined(PNG_cHRM_SUPPORTED)
35#ifdef PNG_FLOATING_POINT_SUPPORTED
Glenn Randers-Pehrson75294572000-05-06 14:09:57 -050036void PNGAPI
Andreas Dilger47a0c421997-05-16 02:46:07 -050037png_set_cHRM(png_structp png_ptr, png_infop info_ptr,
38 double white_x, double white_y, double red_x, double red_y,
39 double green_x, double green_y, double blue_x, double blue_y)
40{
41 png_debug1(1, "in %s storage function\n", "cHRM");
Glenn Randers-Pehrsona357b991998-02-08 20:56:40 -060042 if (png_ptr == NULL || info_ptr == NULL)
Andreas Dilger47a0c421997-05-16 02:46:07 -050043 return;
44
Glenn Randers-Pehrson9c0f0942002-02-21 23:14:23 -060045 if (white_x < 0.0 || white_y < 0.0 ||
46 red_x < 0.0 || red_y < 0.0 ||
47 green_x < 0.0 || green_y < 0.0 ||
48 blue_x < 0.0 || blue_y < 0.0)
49 {
50 png_warning(png_ptr,
51 "Ignoring attempt to set negative chromaticity value");
52 return;
53 }
54 if (white_x > 21474.83 || white_y > 21474.83 ||
55 red_x > 21474.83 || red_y > 21474.83 ||
56 green_x > 21474.83 || green_y > 21474.83 ||
57 blue_x > 21474.83 || blue_y > 21474.83)
58 {
59 png_warning(png_ptr,
60 "Ignoring attempt to set chromaticity value exceeding 21474.83");
61 return;
62 }
63
Andreas Dilger47a0c421997-05-16 02:46:07 -050064 info_ptr->x_white = (float)white_x;
65 info_ptr->y_white = (float)white_y;
66 info_ptr->x_red = (float)red_x;
67 info_ptr->y_red = (float)red_y;
68 info_ptr->x_green = (float)green_x;
69 info_ptr->y_green = (float)green_y;
70 info_ptr->x_blue = (float)blue_x;
71 info_ptr->y_blue = (float)blue_y;
Glenn Randers-Pehrson61c32d92000-02-04 23:40:16 -060072#ifdef PNG_FIXED_POINT_SUPPORTED
73 info_ptr->int_x_white = (png_fixed_point)(white_x*100000.+0.5);
74 info_ptr->int_y_white = (png_fixed_point)(white_y*100000.+0.5);
Glenn Randers-Pehrson9c0f0942002-02-21 23:14:23 -060075 info_ptr->int_x_red = (png_fixed_point)( red_x*100000.+0.5);
76 info_ptr->int_y_red = (png_fixed_point)( red_y*100000.+0.5);
Glenn Randers-Pehrson61c32d92000-02-04 23:40:16 -060077 info_ptr->int_x_green = (png_fixed_point)(green_x*100000.+0.5);
78 info_ptr->int_y_green = (png_fixed_point)(green_y*100000.+0.5);
Glenn Randers-Pehrson9c0f0942002-02-21 23:14:23 -060079 info_ptr->int_x_blue = (png_fixed_point)( blue_x*100000.+0.5);
80 info_ptr->int_y_blue = (png_fixed_point)( blue_y*100000.+0.5);
Glenn Randers-Pehrson61c32d92000-02-04 23:40:16 -060081#endif
Andreas Dilger47a0c421997-05-16 02:46:07 -050082 info_ptr->valid |= PNG_INFO_cHRM;
83}
84#endif
Glenn Randers-Pehrson166c5a31999-12-10 09:43:02 -060085#ifdef PNG_FIXED_POINT_SUPPORTED
Glenn Randers-Pehrson75294572000-05-06 14:09:57 -050086void PNGAPI
Glenn Randers-Pehrson166c5a31999-12-10 09:43:02 -060087png_set_cHRM_fixed(png_structp png_ptr, png_infop info_ptr,
Glenn Randers-Pehrson61c32d92000-02-04 23:40:16 -060088 png_fixed_point white_x, png_fixed_point white_y, png_fixed_point red_x,
89 png_fixed_point red_y, png_fixed_point green_x, png_fixed_point green_y,
90 png_fixed_point blue_x, png_fixed_point blue_y)
Glenn Randers-Pehrson166c5a31999-12-10 09:43:02 -060091{
92 png_debug1(1, "in %s storage function\n", "cHRM");
93 if (png_ptr == NULL || info_ptr == NULL)
94 return;
Andreas Dilger47a0c421997-05-16 02:46:07 -050095
Glenn Randers-Pehrson9c0f0942002-02-21 23:14:23 -060096 if (white_x < 0 || white_y < 0 ||
97 red_x < 0 || red_y < 0 ||
98 green_x < 0 || green_y < 0 ||
99 blue_x < 0 || blue_y < 0)
100 {
101 png_warning(png_ptr,
102 "Ignoring attempt to set negative chromaticity value");
103 return;
104 }
Glenn Randers-Pehrson9c3ab682006-02-20 22:09:05 -0600105#ifdef PNG_FLOATING_POINT_SUPPORTED
Glenn Randers-Pehrson5fea36f2004-07-28 08:20:44 -0500106 if (white_x > (double) PNG_UINT_31_MAX ||
107 white_y > (double) PNG_UINT_31_MAX ||
108 red_x > (double) PNG_UINT_31_MAX ||
109 red_y > (double) PNG_UINT_31_MAX ||
110 green_x > (double) PNG_UINT_31_MAX ||
111 green_y > (double) PNG_UINT_31_MAX ||
112 blue_x > (double) PNG_UINT_31_MAX ||
113 blue_y > (double) PNG_UINT_31_MAX)
Glenn Randers-Pehrson9c3ab682006-02-20 22:09:05 -0600114#else
115 if (white_x > (png_fixed_point) PNG_UINT_31_MAX/100000L ||
116 white_y > (png_fixed_point) PNG_UINT_31_MAX/100000L ||
117 red_x > (png_fixed_point) PNG_UINT_31_MAX/100000L ||
118 red_y > (png_fixed_point) PNG_UINT_31_MAX/100000L ||
119 green_x > (png_fixed_point) PNG_UINT_31_MAX/100000L ||
120 green_y > (png_fixed_point) PNG_UINT_31_MAX/100000L ||
121 blue_x > (png_fixed_point) PNG_UINT_31_MAX/100000L ||
122 blue_y > (png_fixed_point) PNG_UINT_31_MAX/100000L)
123#endif
Glenn Randers-Pehrson9c0f0942002-02-21 23:14:23 -0600124 {
125 png_warning(png_ptr,
126 "Ignoring attempt to set chromaticity value exceeding 21474.83");
127 return;
128 }
Glenn Randers-Pehrson166c5a31999-12-10 09:43:02 -0600129 info_ptr->int_x_white = white_x;
130 info_ptr->int_y_white = white_y;
131 info_ptr->int_x_red = red_x;
132 info_ptr->int_y_red = red_y;
133 info_ptr->int_x_green = green_x;
134 info_ptr->int_y_green = green_y;
135 info_ptr->int_x_blue = blue_x;
136 info_ptr->int_y_blue = blue_y;
Glenn Randers-Pehrson61c32d92000-02-04 23:40:16 -0600137#ifdef PNG_FLOATING_POINT_SUPPORTED
138 info_ptr->x_white = (float)(white_x/100000.);
139 info_ptr->y_white = (float)(white_y/100000.);
Glenn Randers-Pehrson9c0f0942002-02-21 23:14:23 -0600140 info_ptr->x_red = (float)( red_x/100000.);
141 info_ptr->y_red = (float)( red_y/100000.);
Glenn Randers-Pehrson61c32d92000-02-04 23:40:16 -0600142 info_ptr->x_green = (float)(green_x/100000.);
143 info_ptr->y_green = (float)(green_y/100000.);
Glenn Randers-Pehrson9c0f0942002-02-21 23:14:23 -0600144 info_ptr->x_blue = (float)( blue_x/100000.);
145 info_ptr->y_blue = (float)( blue_y/100000.);
Glenn Randers-Pehrson61c32d92000-02-04 23:40:16 -0600146#endif
Glenn Randers-Pehrson166c5a31999-12-10 09:43:02 -0600147 info_ptr->valid |= PNG_INFO_cHRM;
148}
149#endif
150#endif
151
152#if defined(PNG_gAMA_SUPPORTED)
153#ifdef PNG_FLOATING_POINT_SUPPORTED
Glenn Randers-Pehrson75294572000-05-06 14:09:57 -0500154void PNGAPI
Andreas Dilger47a0c421997-05-16 02:46:07 -0500155png_set_gAMA(png_structp png_ptr, png_infop info_ptr, double file_gamma)
156{
Glenn Randers-Pehrson9c0f0942002-02-21 23:14:23 -0600157 double gamma;
Andreas Dilger47a0c421997-05-16 02:46:07 -0500158 png_debug1(1, "in %s storage function\n", "gAMA");
Glenn Randers-Pehrsona357b991998-02-08 20:56:40 -0600159 if (png_ptr == NULL || info_ptr == NULL)
Andreas Dilger47a0c421997-05-16 02:46:07 -0500160 return;
161
Glenn Randers-Pehrson9c0f0942002-02-21 23:14:23 -0600162 /* Check for overflow */
163 if (file_gamma > 21474.83)
164 {
165 png_warning(png_ptr, "Limiting gamma to 21474.83");
166 gamma=21474.83;
167 }
168 else
169 gamma=file_gamma;
170 info_ptr->gamma = (float)gamma;
Glenn Randers-Pehrson166c5a31999-12-10 09:43:02 -0600171#ifdef PNG_FIXED_POINT_SUPPORTED
Glenn Randers-Pehrson9c0f0942002-02-21 23:14:23 -0600172 info_ptr->int_gamma = (int)(gamma*100000.+.5);
Glenn Randers-Pehrson166c5a31999-12-10 09:43:02 -0600173#endif
174 info_ptr->valid |= PNG_INFO_gAMA;
Glenn Randers-Pehrson9c0f0942002-02-21 23:14:23 -0600175 if(gamma == 0.0)
Glenn Randers-Pehrson3097f612001-05-07 14:52:45 -0500176 png_warning(png_ptr, "Setting gamma=0");
Glenn Randers-Pehrson166c5a31999-12-10 09:43:02 -0600177}
178#endif
Glenn Randers-Pehrson75294572000-05-06 14:09:57 -0500179void PNGAPI
Glenn Randers-Pehrson61c32d92000-02-04 23:40:16 -0600180png_set_gAMA_fixed(png_structp png_ptr, png_infop info_ptr, png_fixed_point
181 int_gamma)
Glenn Randers-Pehrson166c5a31999-12-10 09:43:02 -0600182{
Glenn Randers-Pehrson9c0f0942002-02-21 23:14:23 -0600183 png_fixed_point gamma;
184
Glenn Randers-Pehrson166c5a31999-12-10 09:43:02 -0600185 png_debug1(1, "in %s storage function\n", "gAMA");
186 if (png_ptr == NULL || info_ptr == NULL)
187 return;
188
Glenn Randers-Pehrson5fea36f2004-07-28 08:20:44 -0500189 if (int_gamma > (png_fixed_point) PNG_UINT_31_MAX)
Glenn Randers-Pehrson9c0f0942002-02-21 23:14:23 -0600190 {
191 png_warning(png_ptr, "Limiting gamma to 21474.83");
Glenn Randers-Pehrson5fea36f2004-07-28 08:20:44 -0500192 gamma=PNG_UINT_31_MAX;
Glenn Randers-Pehrson9c0f0942002-02-21 23:14:23 -0600193 }
194 else
195 {
196 if (int_gamma < 0)
197 {
198 png_warning(png_ptr, "Setting negative gamma to zero");
199 gamma=0;
200 }
201 else
202 gamma=int_gamma;
203 }
Glenn Randers-Pehrson166c5a31999-12-10 09:43:02 -0600204#ifdef PNG_FLOATING_POINT_SUPPORTED
Glenn Randers-Pehrson9c0f0942002-02-21 23:14:23 -0600205 info_ptr->gamma = (float)(gamma/100000.);
Glenn Randers-Pehrson166c5a31999-12-10 09:43:02 -0600206#endif
Glenn Randers-Pehrson6942d532000-05-01 09:31:54 -0500207#ifdef PNG_FIXED_POINT_SUPPORTED
Glenn Randers-Pehrson9c0f0942002-02-21 23:14:23 -0600208 info_ptr->int_gamma = gamma;
Glenn Randers-Pehrson6942d532000-05-01 09:31:54 -0500209#endif
Andreas Dilger47a0c421997-05-16 02:46:07 -0500210 info_ptr->valid |= PNG_INFO_gAMA;
Glenn Randers-Pehrson9c0f0942002-02-21 23:14:23 -0600211 if(gamma == 0)
Glenn Randers-Pehrson3097f612001-05-07 14:52:45 -0500212 png_warning(png_ptr, "Setting gamma=0");
Andreas Dilger47a0c421997-05-16 02:46:07 -0500213}
Glenn Randers-Pehrson3097f612001-05-07 14:52:45 -0500214#endif
Andreas Dilger47a0c421997-05-16 02:46:07 -0500215
Glenn Randers-Pehrson166c5a31999-12-10 09:43:02 -0600216#if defined(PNG_hIST_SUPPORTED)
Glenn Randers-Pehrson75294572000-05-06 14:09:57 -0500217void PNGAPI
Andreas Dilger47a0c421997-05-16 02:46:07 -0500218png_set_hIST(png_structp png_ptr, png_infop info_ptr, png_uint_16p hist)
219{
Glenn Randers-Pehrsond1e8c862002-06-20 06:54:34 -0500220 int i;
Glenn Randers-Pehrson76e5fd62000-12-28 07:50:05 -0600221
Andreas Dilger47a0c421997-05-16 02:46:07 -0500222 png_debug1(1, "in %s storage function\n", "hIST");
Glenn Randers-Pehrsona357b991998-02-08 20:56:40 -0600223 if (png_ptr == NULL || info_ptr == NULL)
Andreas Dilger47a0c421997-05-16 02:46:07 -0500224 return;
Glenn Randers-Pehrson9c3ab682006-02-20 22:09:05 -0600225 if (info_ptr->num_palette <= 0 || info_ptr->num_palette
226 > PNG_MAX_PALETTE_LENGTH)
Glenn Randers-Pehrson3097f612001-05-07 14:52:45 -0500227 {
Glenn Randers-Pehrson76e5fd62000-12-28 07:50:05 -0600228 png_warning(png_ptr,
Glenn Randers-Pehrsonbeb572e2006-08-19 13:59:24 -0500229 "Invalid palette size, hIST allocation skipped");
Glenn Randers-Pehrson3097f612001-05-07 14:52:45 -0500230 return;
231 }
Andreas Dilger47a0c421997-05-16 02:46:07 -0500232
Glenn Randers-Pehrson76e5fd62000-12-28 07:50:05 -0600233#ifdef PNG_FREE_ME_SUPPORTED
234 png_free_data(png_ptr, info_ptr, PNG_FREE_HIST, 0);
235#endif
Glenn Randers-Pehrson9c3ab682006-02-20 22:09:05 -0600236 /* Changed from info->num_palette to PNG_MAX_PALETTE_LENGTH in version
237 1.2.1 */
Glenn Randers-Pehrson07748d12002-05-25 11:12:10 -0500238 png_ptr->hist = (png_uint_16p)png_malloc_warn(png_ptr,
Glenn Randers-Pehrsonbeb572e2006-08-19 13:59:24 -0500239 PNG_MAX_PALETTE_LENGTH * sizeof(png_uint_16));
Glenn Randers-Pehrson07748d12002-05-25 11:12:10 -0500240 if (png_ptr->hist == NULL)
241 {
Glenn Randers-Pehrsonbeb572e2006-08-19 13:59:24 -0500242 png_warning(png_ptr, "Insufficient memory for hIST chunk data");
Glenn Randers-Pehrson07748d12002-05-25 11:12:10 -0500243 return;
244 }
Glenn Randers-Pehrson76e5fd62000-12-28 07:50:05 -0600245
246 for (i = 0; i < info_ptr->num_palette; i++)
247 png_ptr->hist[i] = hist[i];
248 info_ptr->hist = png_ptr->hist;
Andreas Dilger47a0c421997-05-16 02:46:07 -0500249 info_ptr->valid |= PNG_INFO_hIST;
Glenn Randers-Pehrson76e5fd62000-12-28 07:50:05 -0600250
251#ifdef PNG_FREE_ME_SUPPORTED
252 info_ptr->free_me |= PNG_FREE_HIST;
253#else
254 png_ptr->flags |= PNG_FLAG_FREE_HIST;
255#endif
Andreas Dilger47a0c421997-05-16 02:46:07 -0500256}
257#endif
258
Glenn Randers-Pehrson75294572000-05-06 14:09:57 -0500259void PNGAPI
Andreas Dilger47a0c421997-05-16 02:46:07 -0500260png_set_IHDR(png_structp png_ptr, png_infop info_ptr,
261 png_uint_32 width, png_uint_32 height, int bit_depth,
262 int color_type, int interlace_type, int compression_type,
263 int filter_type)
264{
265 png_debug1(1, "in %s storage function\n", "IHDR");
Glenn Randers-Pehrsona357b991998-02-08 20:56:40 -0600266 if (png_ptr == NULL || info_ptr == NULL)
Andreas Dilger47a0c421997-05-16 02:46:07 -0500267 return;
268
Glenn Randers-Pehrson3097f612001-05-07 14:52:45 -0500269 /* check for width and height valid values */
270 if (width == 0 || height == 0)
271 png_error(png_ptr, "Image width or height is zero in IHDR");
Glenn Randers-Pehrson272489d2004-08-04 06:34:52 -0500272#ifdef PNG_SET_USER_LIMITS_SUPPORTED
273 if (width > png_ptr->user_width_max || height > png_ptr->user_height_max)
274 png_error(png_ptr, "image size exceeds user limits in IHDR");
275#else
276 if (width > PNG_USER_WIDTH_MAX || height > PNG_USER_HEIGHT_MAX)
277 png_error(png_ptr, "image size exceeds user limits in IHDR");
278#endif
Glenn Randers-Pehrson5fea36f2004-07-28 08:20:44 -0500279 if (width > PNG_UINT_31_MAX || height > PNG_UINT_31_MAX)
Glenn Randers-Pehrson3097f612001-05-07 14:52:45 -0500280 png_error(png_ptr, "Invalid image size in IHDR");
Glenn Randers-Pehrson272489d2004-08-04 06:34:52 -0500281 if ( width > (PNG_UINT_32_MAX
282 >> 3) /* 8-byte RGBA pixels */
283 - 64 /* bigrowbuf hack */
284 - 1 /* filter byte */
285 - 7*8 /* rounding of width to multiple of 8 pixels */
286 - 8) /* extra max_pixel_depth pad */
287 png_warning(png_ptr, "Width is too large for libpng to process pixels");
Glenn Randers-Pehrson3097f612001-05-07 14:52:45 -0500288
289 /* check other values */
290 if (bit_depth != 1 && bit_depth != 2 && bit_depth != 4 &&
291 bit_depth != 8 && bit_depth != 16)
292 png_error(png_ptr, "Invalid bit depth in IHDR");
293
294 if (color_type < 0 || color_type == 1 ||
295 color_type == 5 || color_type > 6)
296 png_error(png_ptr, "Invalid color type in IHDR");
297
298 if (((color_type == PNG_COLOR_TYPE_PALETTE) && bit_depth > 8) ||
299 ((color_type == PNG_COLOR_TYPE_RGB ||
300 color_type == PNG_COLOR_TYPE_GRAY_ALPHA ||
301 color_type == PNG_COLOR_TYPE_RGB_ALPHA) && bit_depth < 8))
302 png_error(png_ptr, "Invalid color type/bit depth combination in IHDR");
303
304 if (interlace_type >= PNG_INTERLACE_LAST)
305 png_error(png_ptr, "Unknown interlace method in IHDR");
306
307 if (compression_type != PNG_COMPRESSION_TYPE_BASE)
308 png_error(png_ptr, "Unknown compression method in IHDR");
309
310#if defined(PNG_MNG_FEATURES_SUPPORTED)
311 /* Accept filter_method 64 (intrapixel differencing) only if
312 * 1. Libpng was compiled with PNG_MNG_FEATURES_SUPPORTED and
313 * 2. Libpng did not read a PNG signature (this filter_method is only
314 * used in PNG datastreams that are embedded in MNG datastreams) and
315 * 3. The application called png_permit_mng_features with a mask that
316 * included PNG_FLAG_MNG_FILTER_64 and
317 * 4. The filter_method is 64 and
318 * 5. The color_type is RGB or RGBA
319 */
320 if((png_ptr->mode&PNG_HAVE_PNG_SIGNATURE)&&png_ptr->mng_features_permitted)
Glenn Randers-Pehrsonc3d51c12006-03-02 07:23:18 -0600321 png_warning(png_ptr,"MNG features are not allowed in a PNG datastream");
Glenn Randers-Pehrson3097f612001-05-07 14:52:45 -0500322 if(filter_type != PNG_FILTER_TYPE_BASE)
323 {
324 if(!((png_ptr->mng_features_permitted & PNG_FLAG_MNG_FILTER_64) &&
325 (filter_type == PNG_INTRAPIXEL_DIFFERENCING) &&
326 ((png_ptr->mode&PNG_HAVE_PNG_SIGNATURE) == 0) &&
Glenn Randers-Pehrsone6474622006-03-04 16:50:47 -0600327 (color_type == PNG_COLOR_TYPE_RGB ||
Glenn Randers-Pehrson3097f612001-05-07 14:52:45 -0500328 color_type == PNG_COLOR_TYPE_RGB_ALPHA)))
329 png_error(png_ptr, "Unknown filter method in IHDR");
330 if(png_ptr->mode&PNG_HAVE_PNG_SIGNATURE)
331 png_warning(png_ptr, "Invalid filter method in IHDR");
332 }
333#else
334 if(filter_type != PNG_FILTER_TYPE_BASE)
335 png_error(png_ptr, "Unknown filter method in IHDR");
336#endif
337
Andreas Dilger47a0c421997-05-16 02:46:07 -0500338 info_ptr->width = width;
339 info_ptr->height = height;
340 info_ptr->bit_depth = (png_byte)bit_depth;
341 info_ptr->color_type =(png_byte) color_type;
342 info_ptr->compression_type = (png_byte)compression_type;
343 info_ptr->filter_type = (png_byte)filter_type;
344 info_ptr->interlace_type = (png_byte)interlace_type;
Glenn Randers-Pehrson25d82242002-05-01 11:51:26 -0500345 if (info_ptr->color_type == PNG_COLOR_TYPE_PALETTE)
346 info_ptr->channels = 1;
347 else if (info_ptr->color_type & PNG_COLOR_MASK_COLOR)
Andreas Dilger47a0c421997-05-16 02:46:07 -0500348 info_ptr->channels = 3;
349 else
350 info_ptr->channels = 1;
351 if (info_ptr->color_type & PNG_COLOR_MASK_ALPHA)
352 info_ptr->channels++;
353 info_ptr->pixel_depth = (png_byte)(info_ptr->channels * info_ptr->bit_depth);
Glenn Randers-Pehrson0f881d61998-02-07 10:20:57 -0600354
Glenn Randers-Pehrson272489d2004-08-04 06:34:52 -0500355 /* check for potential overflow */
Glenn Randers-Pehrsonbeb572e2006-08-19 13:59:24 -0500356 if (width > (PNG_UINT_32_MAX
Glenn Randers-Pehrson272489d2004-08-04 06:34:52 -0500357 >> 3) /* 8-byte RGBA pixels */
358 - 64 /* bigrowbuf hack */
359 - 1 /* filter byte */
360 - 7*8 /* rounding of width to multiple of 8 pixels */
361 - 8) /* extra max_pixel_depth pad */
Glenn Randers-Pehrsonbeb572e2006-08-19 13:59:24 -0500362 info_ptr->rowbytes = 0;
Glenn Randers-Pehrson896239b1998-04-21 15:03:57 -0500363 else
Glenn Randers-Pehrson272489d2004-08-04 06:34:52 -0500364 info_ptr->rowbytes = PNG_ROWBYTES(info_ptr->pixel_depth,width);
Andreas Dilger47a0c421997-05-16 02:46:07 -0500365}
366
Glenn Randers-Pehrson166c5a31999-12-10 09:43:02 -0600367#if defined(PNG_oFFs_SUPPORTED)
Glenn Randers-Pehrson75294572000-05-06 14:09:57 -0500368void PNGAPI
Andreas Dilger47a0c421997-05-16 02:46:07 -0500369png_set_oFFs(png_structp png_ptr, png_infop info_ptr,
Glenn Randers-Pehrson166c5a31999-12-10 09:43:02 -0600370 png_int_32 offset_x, png_int_32 offset_y, int unit_type)
Andreas Dilger47a0c421997-05-16 02:46:07 -0500371{
372 png_debug1(1, "in %s storage function\n", "oFFs");
Glenn Randers-Pehrsona357b991998-02-08 20:56:40 -0600373 if (png_ptr == NULL || info_ptr == NULL)
Andreas Dilger47a0c421997-05-16 02:46:07 -0500374 return;
375
376 info_ptr->x_offset = offset_x;
377 info_ptr->y_offset = offset_y;
378 info_ptr->offset_unit_type = (png_byte)unit_type;
379 info_ptr->valid |= PNG_INFO_oFFs;
380}
381#endif
382
Glenn Randers-Pehrson166c5a31999-12-10 09:43:02 -0600383#if defined(PNG_pCAL_SUPPORTED)
Glenn Randers-Pehrson75294572000-05-06 14:09:57 -0500384void PNGAPI
Andreas Dilger47a0c421997-05-16 02:46:07 -0500385png_set_pCAL(png_structp png_ptr, png_infop info_ptr,
386 png_charp purpose, png_int_32 X0, png_int_32 X1, int type, int nparams,
387 png_charp units, png_charpp params)
388{
Glenn Randers-Pehrsonbeb572e2006-08-19 13:59:24 -0500389 png_size_t length;
Glenn Randers-Pehrsona357b991998-02-08 20:56:40 -0600390 int i;
Andreas Dilger47a0c421997-05-16 02:46:07 -0500391
392 png_debug1(1, "in %s storage function\n", "pCAL");
Glenn Randers-Pehrsona357b991998-02-08 20:56:40 -0600393 if (png_ptr == NULL || info_ptr == NULL)
Andreas Dilger47a0c421997-05-16 02:46:07 -0500394 return;
395
396 length = png_strlen(purpose) + 1;
Glenn Randers-Pehrson6bc53be2006-06-16 07:52:03 -0500397 png_debug1(3, "allocating purpose for info (%lu bytes)\n",
398 (unsigned long) length);
Glenn Randers-Pehrson07748d12002-05-25 11:12:10 -0500399 info_ptr->pcal_purpose = (png_charp)png_malloc_warn(png_ptr, length);
400 if (info_ptr->pcal_purpose == NULL)
Glenn Randers-Pehrsonbeb572e2006-08-19 13:59:24 -0500401 {
402 png_warning(png_ptr, "Insufficient memory for pCAL purpose");
403 return;
404 }
405 png_memcpy(info_ptr->pcal_purpose, purpose, length);
Andreas Dilger47a0c421997-05-16 02:46:07 -0500406
407 png_debug(3, "storing X0, X1, type, and nparams in info\n");
408 info_ptr->pcal_X0 = X0;
409 info_ptr->pcal_X1 = X1;
410 info_ptr->pcal_type = (png_byte)type;
411 info_ptr->pcal_nparams = (png_byte)nparams;
412
413 length = png_strlen(units) + 1;
Glenn Randers-Pehrson6bc53be2006-06-16 07:52:03 -0500414 png_debug1(3, "allocating units for info (%lu bytes)\n",
415 (unsigned long)length);
Glenn Randers-Pehrson07748d12002-05-25 11:12:10 -0500416 info_ptr->pcal_units = (png_charp)png_malloc_warn(png_ptr, length);
417 if (info_ptr->pcal_units == NULL)
Glenn Randers-Pehrsonbeb572e2006-08-19 13:59:24 -0500418 {
419 png_warning(png_ptr, "Insufficient memory for pCAL units");
420 return;
421 }
422 png_memcpy(info_ptr->pcal_units, units, length);
Andreas Dilger47a0c421997-05-16 02:46:07 -0500423
Glenn Randers-Pehrson07748d12002-05-25 11:12:10 -0500424 info_ptr->pcal_params = (png_charpp)png_malloc_warn(png_ptr,
Glenn Randers-Pehrsonbeb572e2006-08-19 13:59:24 -0500425 (png_size_t)((nparams + 1) * sizeof(png_charp)));
Glenn Randers-Pehrson07748d12002-05-25 11:12:10 -0500426 if (info_ptr->pcal_params == NULL)
Glenn Randers-Pehrsonbeb572e2006-08-19 13:59:24 -0500427 {
428 png_warning(png_ptr, "Insufficient memory for pCAL params");
429 return;
430 }
Glenn Randers-Pehrson3097f612001-05-07 14:52:45 -0500431
Andreas Dilger47a0c421997-05-16 02:46:07 -0500432 info_ptr->pcal_params[nparams] = NULL;
433
434 for (i = 0; i < nparams; i++)
435 {
436 length = png_strlen(params[i]) + 1;
Glenn Randers-Pehrson6bc53be2006-06-16 07:52:03 -0500437 png_debug2(3, "allocating parameter %d for info (%lu bytes)\n", i,
438 (unsigned long) length);
Glenn Randers-Pehrson07748d12002-05-25 11:12:10 -0500439 info_ptr->pcal_params[i] = (png_charp)png_malloc_warn(png_ptr, length);
440 if (info_ptr->pcal_params[i] == NULL)
Glenn Randers-Pehrsonbeb572e2006-08-19 13:59:24 -0500441 {
442 png_warning(png_ptr, "Insufficient memory for pCAL parameter");
443 return;
444 }
445 png_memcpy(info_ptr->pcal_params[i], params[i], length);
Andreas Dilger47a0c421997-05-16 02:46:07 -0500446 }
447
448 info_ptr->valid |= PNG_INFO_pCAL;
Glenn Randers-Pehrson1ef65b62000-05-12 06:19:53 -0500449#ifdef PNG_FREE_ME_SUPPORTED
450 info_ptr->free_me |= PNG_FREE_PCAL;
451#endif
Andreas Dilger47a0c421997-05-16 02:46:07 -0500452}
453#endif
454
Glenn Randers-Pehrson166c5a31999-12-10 09:43:02 -0600455#if defined(PNG_READ_sCAL_SUPPORTED) || defined(PNG_WRITE_sCAL_SUPPORTED)
456#ifdef PNG_FLOATING_POINT_SUPPORTED
Glenn Randers-Pehrson75294572000-05-06 14:09:57 -0500457void PNGAPI
Glenn Randers-Pehrson166c5a31999-12-10 09:43:02 -0600458png_set_sCAL(png_structp png_ptr, png_infop info_ptr,
Glenn Randers-Pehrson61c32d92000-02-04 23:40:16 -0600459 int unit, double width, double height)
Glenn Randers-Pehrson166c5a31999-12-10 09:43:02 -0600460{
Glenn Randers-Pehrson166c5a31999-12-10 09:43:02 -0600461 png_debug1(1, "in %s storage function\n", "sCAL");
462 if (png_ptr == NULL || info_ptr == NULL)
463 return;
464
Glenn Randers-Pehrson61c32d92000-02-04 23:40:16 -0600465 info_ptr->scal_unit = (png_byte)unit;
Glenn Randers-Pehrson166c5a31999-12-10 09:43:02 -0600466 info_ptr->scal_pixel_width = width;
467 info_ptr->scal_pixel_height = height;
468
469 info_ptr->valid |= PNG_INFO_sCAL;
470}
Glenn Randers-Pehrson61c32d92000-02-04 23:40:16 -0600471#else
472#ifdef PNG_FIXED_POINT_SUPPORTED
Glenn Randers-Pehrson75294572000-05-06 14:09:57 -0500473void PNGAPI
Glenn Randers-Pehrson166c5a31999-12-10 09:43:02 -0600474png_set_sCAL_s(png_structp png_ptr, png_infop info_ptr,
Glenn Randers-Pehrson61c32d92000-02-04 23:40:16 -0600475 int unit, png_charp swidth, png_charp sheight)
Glenn Randers-Pehrson166c5a31999-12-10 09:43:02 -0600476{
Glenn Randers-Pehrsonbeb572e2006-08-19 13:59:24 -0500477 png_size_t length;
Glenn Randers-Pehrson166c5a31999-12-10 09:43:02 -0600478
479 png_debug1(1, "in %s storage function\n", "sCAL");
480 if (png_ptr == NULL || info_ptr == NULL)
481 return;
482
Glenn Randers-Pehrson61c32d92000-02-04 23:40:16 -0600483 info_ptr->scal_unit = (png_byte)unit;
Glenn Randers-Pehrson166c5a31999-12-10 09:43:02 -0600484
485 length = png_strlen(swidth) + 1;
Glenn Randers-Pehrsonbeb572e2006-08-19 13:59:24 -0500486 png_debug1(3, "allocating unit for info (%u bytes)\n",
487 (unsigned int)length);
Glenn Randers-Pehrson5fea36f2004-07-28 08:20:44 -0500488 info_ptr->scal_s_width = (png_charp)png_malloc_warn(png_ptr, length);
489 if (info_ptr->scal_s_width == NULL)
490 {
Glenn Randers-Pehrsonbeb572e2006-08-19 13:59:24 -0500491 png_warning(png_ptr, "Memory allocation failed while processing sCAL");
Glenn Randers-Pehrson5fea36f2004-07-28 08:20:44 -0500492 }
Glenn Randers-Pehrsonbeb572e2006-08-19 13:59:24 -0500493 png_memcpy(info_ptr->scal_s_width, swidth, length);
Glenn Randers-Pehrson166c5a31999-12-10 09:43:02 -0600494
495 length = png_strlen(sheight) + 1;
Glenn Randers-Pehrsonbeb572e2006-08-19 13:59:24 -0500496 png_debug1(3, "allocating unit for info (%u bytes)\n",
497 (unsigned int)length);
Glenn Randers-Pehrson5fea36f2004-07-28 08:20:44 -0500498 info_ptr->scal_s_height = (png_charp)png_malloc_warn(png_ptr, length);
499 if (info_ptr->scal_s_height == NULL)
500 {
501 png_free (png_ptr, info_ptr->scal_s_width);
Glenn Randers-Pehrsonbeb572e2006-08-19 13:59:24 -0500502 png_warning(png_ptr, "Memory allocation failed while processing sCAL");
Glenn Randers-Pehrson5fea36f2004-07-28 08:20:44 -0500503 }
Glenn Randers-Pehrsonbeb572e2006-08-19 13:59:24 -0500504 png_memcpy(info_ptr->scal_s_height, sheight, length);
Glenn Randers-Pehrson166c5a31999-12-10 09:43:02 -0600505
506 info_ptr->valid |= PNG_INFO_sCAL;
Glenn Randers-Pehrson1ef65b62000-05-12 06:19:53 -0500507#ifdef PNG_FREE_ME_SUPPORTED
508 info_ptr->free_me |= PNG_FREE_SCAL;
509#endif
Glenn Randers-Pehrson166c5a31999-12-10 09:43:02 -0600510}
511#endif
Glenn Randers-Pehrson61c32d92000-02-04 23:40:16 -0600512#endif
513#endif
Glenn Randers-Pehrson166c5a31999-12-10 09:43:02 -0600514
515#if defined(PNG_pHYs_SUPPORTED)
Glenn Randers-Pehrson75294572000-05-06 14:09:57 -0500516void PNGAPI
Andreas Dilger47a0c421997-05-16 02:46:07 -0500517png_set_pHYs(png_structp png_ptr, png_infop info_ptr,
518 png_uint_32 res_x, png_uint_32 res_y, int unit_type)
519{
520 png_debug1(1, "in %s storage function\n", "pHYs");
Glenn Randers-Pehrsona357b991998-02-08 20:56:40 -0600521 if (png_ptr == NULL || info_ptr == NULL)
Andreas Dilger47a0c421997-05-16 02:46:07 -0500522 return;
523
524 info_ptr->x_pixels_per_unit = res_x;
525 info_ptr->y_pixels_per_unit = res_y;
526 info_ptr->phys_unit_type = (png_byte)unit_type;
527 info_ptr->valid |= PNG_INFO_pHYs;
528}
529#endif
530
Glenn Randers-Pehrson75294572000-05-06 14:09:57 -0500531void PNGAPI
Andreas Dilger47a0c421997-05-16 02:46:07 -0500532png_set_PLTE(png_structp png_ptr, png_infop info_ptr,
533 png_colorp palette, int num_palette)
534{
Glenn Randers-Pehrson61c32d92000-02-04 23:40:16 -0600535
Andreas Dilger47a0c421997-05-16 02:46:07 -0500536 png_debug1(1, "in %s storage function\n", "PLTE");
Glenn Randers-Pehrsona357b991998-02-08 20:56:40 -0600537 if (png_ptr == NULL || info_ptr == NULL)
Andreas Dilger47a0c421997-05-16 02:46:07 -0500538 return;
539
Glenn Randers-Pehrson9c3ab682006-02-20 22:09:05 -0600540 if (num_palette < 0 || num_palette > PNG_MAX_PALETTE_LENGTH)
541 {
542 if (info_ptr->color_type == PNG_COLOR_TYPE_PALETTE)
543 png_error(png_ptr, "Invalid palette length");
544 else
545 {
546 png_warning(png_ptr, "Invalid palette length");
547 return;
548 }
549 }
550
Glenn Randers-Pehrson76e5fd62000-12-28 07:50:05 -0600551 /*
552 * It may not actually be necessary to set png_ptr->palette here;
553 * we do it for backward compatibility with the way the png_handle_tRNS
554 * function used to do the allocation.
555 */
556#ifdef PNG_FREE_ME_SUPPORTED
557 png_free_data(png_ptr, info_ptr, PNG_FREE_PLTE, 0);
558#endif
Glenn Randers-Pehrson5fea36f2004-07-28 08:20:44 -0500559
Glenn Randers-Pehrson9c3ab682006-02-20 22:09:05 -0600560 /* Changed in libpng-1.2.1 to allocate PNG_MAX_PALETTE_LENGTH instead
561 of num_palette entries,
Glenn Randers-Pehrsondb3b88d2001-12-04 06:30:43 -0600562 in case of an invalid PNG file that has too-large sample values. */
Glenn Randers-Pehrson5fea36f2004-07-28 08:20:44 -0500563 png_ptr->palette = (png_colorp)png_malloc(png_ptr,
Glenn Randers-Pehrsonbeb572e2006-08-19 13:59:24 -0500564 PNG_MAX_PALETTE_LENGTH * sizeof(png_color));
Glenn Randers-Pehrson9c3ab682006-02-20 22:09:05 -0600565 png_memset(png_ptr->palette, 0, PNG_MAX_PALETTE_LENGTH *
Glenn Randers-Pehrsonbeb572e2006-08-19 13:59:24 -0500566 sizeof(png_color));
567 png_memcpy(png_ptr->palette, palette, num_palette * sizeof(png_color));
Glenn Randers-Pehrson76e5fd62000-12-28 07:50:05 -0600568 info_ptr->palette = png_ptr->palette;
569 info_ptr->num_palette = png_ptr->num_palette = (png_uint_16)num_palette;
Glenn Randers-Pehrson61c32d92000-02-04 23:40:16 -0600570
Glenn Randers-Pehrson76e5fd62000-12-28 07:50:05 -0600571#ifdef PNG_FREE_ME_SUPPORTED
572 info_ptr->free_me |= PNG_FREE_PLTE;
573#else
574 png_ptr->flags |= PNG_FLAG_FREE_PLTE;
575#endif
576
Glenn Randers-Pehrsona77ef622000-02-18 13:48:52 -0600577 info_ptr->valid |= PNG_INFO_PLTE;
Andreas Dilger47a0c421997-05-16 02:46:07 -0500578}
579
Glenn Randers-Pehrson166c5a31999-12-10 09:43:02 -0600580#if defined(PNG_sBIT_SUPPORTED)
Glenn Randers-Pehrson75294572000-05-06 14:09:57 -0500581void PNGAPI
Andreas Dilger47a0c421997-05-16 02:46:07 -0500582png_set_sBIT(png_structp png_ptr, png_infop info_ptr,
583 png_color_8p sig_bit)
584{
585 png_debug1(1, "in %s storage function\n", "sBIT");
Glenn Randers-Pehrsona357b991998-02-08 20:56:40 -0600586 if (png_ptr == NULL || info_ptr == NULL)
Andreas Dilger47a0c421997-05-16 02:46:07 -0500587 return;
588
Glenn Randers-Pehrsonbeb572e2006-08-19 13:59:24 -0500589 png_memcpy(&(info_ptr->sig_bit), sig_bit, sizeof(png_color_8));
Andreas Dilger47a0c421997-05-16 02:46:07 -0500590 info_ptr->valid |= PNG_INFO_sBIT;
591}
592#endif
593
Glenn Randers-Pehrson166c5a31999-12-10 09:43:02 -0600594#if defined(PNG_sRGB_SUPPORTED)
Glenn Randers-Pehrson75294572000-05-06 14:09:57 -0500595void PNGAPI
Glenn Randers-Pehrsonc4a2ae61998-01-16 22:06:18 -0600596png_set_sRGB(png_structp png_ptr, png_infop info_ptr, int intent)
Glenn Randers-Pehrsonb6ce43d1998-01-01 07:13:13 -0600597{
598 png_debug1(1, "in %s storage function\n", "sRGB");
Glenn Randers-Pehrsona357b991998-02-08 20:56:40 -0600599 if (png_ptr == NULL || info_ptr == NULL)
Glenn Randers-Pehrsonb6ce43d1998-01-01 07:13:13 -0600600 return;
601
Glenn Randers-Pehrsonc4a2ae61998-01-16 22:06:18 -0600602 info_ptr->srgb_intent = (png_byte)intent;
Glenn Randers-Pehrsonb6ce43d1998-01-01 07:13:13 -0600603 info_ptr->valid |= PNG_INFO_sRGB;
604}
Glenn Randers-Pehrson166c5a31999-12-10 09:43:02 -0600605
Glenn Randers-Pehrson75294572000-05-06 14:09:57 -0500606void PNGAPI
Glenn Randers-Pehrsonb6ce43d1998-01-01 07:13:13 -0600607png_set_sRGB_gAMA_and_cHRM(png_structp png_ptr, png_infop info_ptr,
Glenn Randers-Pehrsonc4a2ae61998-01-16 22:06:18 -0600608 int intent)
Glenn Randers-Pehrsonb6ce43d1998-01-01 07:13:13 -0600609{
Glenn Randers-Pehrson166c5a31999-12-10 09:43:02 -0600610#if defined(PNG_gAMA_SUPPORTED)
611#ifdef PNG_FLOATING_POINT_SUPPORTED
Glenn Randers-Pehrsonb6ce43d1998-01-01 07:13:13 -0600612 float file_gamma;
613#endif
Glenn Randers-Pehrson166c5a31999-12-10 09:43:02 -0600614#ifdef PNG_FIXED_POINT_SUPPORTED
Glenn Randers-Pehrson61c32d92000-02-04 23:40:16 -0600615 png_fixed_point int_file_gamma;
Glenn Randers-Pehrson166c5a31999-12-10 09:43:02 -0600616#endif
617#endif
618#if defined(PNG_cHRM_SUPPORTED)
619#ifdef PNG_FLOATING_POINT_SUPPORTED
Glenn Randers-Pehrsonb6ce43d1998-01-01 07:13:13 -0600620 float white_x, white_y, red_x, red_y, green_x, green_y, blue_x, blue_y;
621#endif
Glenn Randers-Pehrson166c5a31999-12-10 09:43:02 -0600622#ifdef PNG_FIXED_POINT_SUPPORTED
Glenn Randers-Pehrson61c32d92000-02-04 23:40:16 -0600623 png_fixed_point int_white_x, int_white_y, int_red_x, int_red_y, int_green_x,
Glenn Randers-Pehrson166c5a31999-12-10 09:43:02 -0600624 int_green_y, int_blue_x, int_blue_y;
625#endif
626#endif
Glenn Randers-Pehrsonb6ce43d1998-01-01 07:13:13 -0600627 png_debug1(1, "in %s storage function\n", "sRGB_gAMA_and_cHRM");
Glenn Randers-Pehrsona357b991998-02-08 20:56:40 -0600628 if (png_ptr == NULL || info_ptr == NULL)
Glenn Randers-Pehrsonb6ce43d1998-01-01 07:13:13 -0600629 return;
630
631 png_set_sRGB(png_ptr, info_ptr, intent);
632
Glenn Randers-Pehrson166c5a31999-12-10 09:43:02 -0600633#if defined(PNG_gAMA_SUPPORTED)
634#ifdef PNG_FLOATING_POINT_SUPPORTED
Glenn Randers-Pehrsonab1e5831999-10-06 04:57:42 -0500635 file_gamma = (float).45455;
Glenn Randers-Pehrsonb6ce43d1998-01-01 07:13:13 -0600636 png_set_gAMA(png_ptr, info_ptr, file_gamma);
637#endif
Glenn Randers-Pehrson166c5a31999-12-10 09:43:02 -0600638#ifdef PNG_FIXED_POINT_SUPPORTED
639 int_file_gamma = 45455L;
640 png_set_gAMA_fixed(png_ptr, info_ptr, int_file_gamma);
641#endif
642#endif
Glenn Randers-Pehrsonb6ce43d1998-01-01 07:13:13 -0600643
Glenn Randers-Pehrson166c5a31999-12-10 09:43:02 -0600644#if defined(PNG_cHRM_SUPPORTED)
645#ifdef PNG_FIXED_POINT_SUPPORTED
646 int_white_x = 31270L;
647 int_white_y = 32900L;
648 int_red_x = 64000L;
649 int_red_y = 33000L;
650 int_green_x = 30000L;
651 int_green_y = 60000L;
652 int_blue_x = 15000L;
653 int_blue_y = 6000L;
654
655 png_set_cHRM_fixed(png_ptr, info_ptr,
656 int_white_x, int_white_y, int_red_x, int_red_y, int_green_x, int_green_y,
657 int_blue_x, int_blue_y);
658#endif
659#ifdef PNG_FLOATING_POINT_SUPPORTED
Glenn Randers-Pehrson70e3f541998-01-03 22:40:55 -0600660 white_x = (float).3127;
661 white_y = (float).3290;
662 red_x = (float).64;
663 red_y = (float).33;
664 green_x = (float).30;
665 green_y = (float).60;
666 blue_x = (float).15;
667 blue_y = (float).06;
Glenn Randers-Pehrsonb6ce43d1998-01-01 07:13:13 -0600668
669 png_set_cHRM(png_ptr, info_ptr,
670 white_x, white_y, red_x, red_y, green_x, green_y, blue_x, blue_y);
Glenn Randers-Pehrson166c5a31999-12-10 09:43:02 -0600671#endif
Glenn Randers-Pehrsonb6ce43d1998-01-01 07:13:13 -0600672#endif
673}
674#endif
675
Glenn Randers-Pehrson166c5a31999-12-10 09:43:02 -0600676
677#if defined(PNG_iCCP_SUPPORTED)
Glenn Randers-Pehrson75294572000-05-06 14:09:57 -0500678void PNGAPI
Glenn Randers-Pehrson166c5a31999-12-10 09:43:02 -0600679png_set_iCCP(png_structp png_ptr, png_infop info_ptr,
680 png_charp name, int compression_type,
Glenn Randers-Pehrson61c32d92000-02-04 23:40:16 -0600681 png_charp profile, png_uint_32 proflen)
Glenn Randers-Pehrson166c5a31999-12-10 09:43:02 -0600682{
Glenn Randers-Pehrson4accabb2000-04-14 14:20:47 -0500683 png_charp new_iccp_name;
684 png_charp new_iccp_profile;
685
Glenn Randers-Pehrson166c5a31999-12-10 09:43:02 -0600686 png_debug1(1, "in %s storage function\n", "iCCP");
687 if (png_ptr == NULL || info_ptr == NULL || name == NULL || profile == NULL)
688 return;
689
Glenn Randers-Pehrson5fea36f2004-07-28 08:20:44 -0500690 new_iccp_name = (png_charp)png_malloc_warn(png_ptr, png_strlen(name)+1);
691 if (new_iccp_name == NULL)
692 {
Glenn Randers-Pehrsonbeb572e2006-08-19 13:59:24 -0500693 png_warning(png_ptr, "Insufficient memory to process iCCP chunk");
Glenn Randers-Pehrson5fea36f2004-07-28 08:20:44 -0500694 return;
695 }
Glenn Randers-Pehrson3097f612001-05-07 14:52:45 -0500696 png_strcpy(new_iccp_name, name);
Glenn Randers-Pehrson5fea36f2004-07-28 08:20:44 -0500697 new_iccp_profile = (png_charp)png_malloc_warn(png_ptr, proflen);
698 if (new_iccp_profile == NULL)
699 {
700 png_free (png_ptr, new_iccp_name);
Glenn Randers-Pehrsonbeb572e2006-08-19 13:59:24 -0500701 png_warning(png_ptr, "Insufficient memory to process iCCP profile");
Glenn Randers-Pehrson5fea36f2004-07-28 08:20:44 -0500702 return;
703 }
Glenn Randers-Pehrson4accabb2000-04-14 14:20:47 -0500704 png_memcpy(new_iccp_profile, profile, (png_size_t)proflen);
705
Glenn Randers-Pehrson38e6e772000-04-09 19:06:13 -0500706 png_free_data(png_ptr, info_ptr, PNG_FREE_ICCP, 0);
Glenn Randers-Pehrson4accabb2000-04-14 14:20:47 -0500707
Glenn Randers-Pehrson61c32d92000-02-04 23:40:16 -0600708 info_ptr->iccp_proflen = proflen;
Glenn Randers-Pehrson4accabb2000-04-14 14:20:47 -0500709 info_ptr->iccp_name = new_iccp_name;
710 info_ptr->iccp_profile = new_iccp_profile;
Glenn Randers-Pehrson166c5a31999-12-10 09:43:02 -0600711 /* Compression is always zero but is here so the API and info structure
Glenn Randers-Pehrson4accabb2000-04-14 14:20:47 -0500712 * does not have to change if we introduce multiple compression types */
Glenn Randers-Pehrson166c5a31999-12-10 09:43:02 -0600713 info_ptr->iccp_compression = (png_byte)compression_type;
Glenn Randers-Pehrson6942d532000-05-01 09:31:54 -0500714#ifdef PNG_FREE_ME_SUPPORTED
Glenn Randers-Pehrsona77ef622000-02-18 13:48:52 -0600715 info_ptr->free_me |= PNG_FREE_ICCP;
Glenn Randers-Pehrson6942d532000-05-01 09:31:54 -0500716#endif
Glenn Randers-Pehrson166c5a31999-12-10 09:43:02 -0600717 info_ptr->valid |= PNG_INFO_iCCP;
718}
719#endif
720
721#if defined(PNG_TEXT_SUPPORTED)
Glenn Randers-Pehrson75294572000-05-06 14:09:57 -0500722void PNGAPI
Andreas Dilger47a0c421997-05-16 02:46:07 -0500723png_set_text(png_structp png_ptr, png_infop info_ptr, png_textp text_ptr,
724 int num_text)
725{
Glenn Randers-Pehrson07748d12002-05-25 11:12:10 -0500726 int ret;
727 ret=png_set_text_2(png_ptr, info_ptr, text_ptr, num_text);
728 if (ret)
729 png_error(png_ptr, "Insufficient memory to store text");
730}
731
732int /* PRIVATE */
733png_set_text_2(png_structp png_ptr, png_infop info_ptr, png_textp text_ptr,
734 int num_text)
735{
Andreas Dilger47a0c421997-05-16 02:46:07 -0500736 int i;
737
738 png_debug1(1, "in %s storage function\n", (png_ptr->chunk_name[0] == '\0' ?
Glenn Randers-Pehrsonc4a2ae61998-01-16 22:06:18 -0600739 "text" : (png_const_charp)png_ptr->chunk_name));
Andreas Dilger47a0c421997-05-16 02:46:07 -0500740
Glenn Randers-Pehrsona357b991998-02-08 20:56:40 -0600741 if (png_ptr == NULL || info_ptr == NULL || num_text == 0)
Glenn Randers-Pehrson07748d12002-05-25 11:12:10 -0500742 return(0);
Andreas Dilger47a0c421997-05-16 02:46:07 -0500743
744 /* Make sure we have enough space in the "text" array in info_struct
745 * to hold all of the incoming text_ptr objects.
746 */
747 if (info_ptr->num_text + num_text > info_ptr->max_text)
748 {
749 if (info_ptr->text != NULL)
750 {
751 png_textp old_text;
752 int old_max;
753
754 old_max = info_ptr->max_text;
755 info_ptr->max_text = info_ptr->num_text + num_text + 8;
756 old_text = info_ptr->text;
Glenn Randers-Pehrson07748d12002-05-25 11:12:10 -0500757 info_ptr->text = (png_textp)png_malloc_warn(png_ptr,
Glenn Randers-Pehrsonbeb572e2006-08-19 13:59:24 -0500758 (png_size_t)(info_ptr->max_text * sizeof(png_text)));
Glenn Randers-Pehrson07748d12002-05-25 11:12:10 -0500759 if (info_ptr->text == NULL)
760 {
761 png_free(png_ptr, old_text);
762 return(1);
763 }
Glenn Randers-Pehrson0f881d61998-02-07 10:20:57 -0600764 png_memcpy(info_ptr->text, old_text, (png_size_t)(old_max *
Glenn Randers-Pehrsonbeb572e2006-08-19 13:59:24 -0500765 sizeof(png_text)));
Andreas Dilger47a0c421997-05-16 02:46:07 -0500766 png_free(png_ptr, old_text);
767 }
768 else
769 {
770 info_ptr->max_text = num_text + 8;
771 info_ptr->num_text = 0;
Glenn Randers-Pehrson07748d12002-05-25 11:12:10 -0500772 info_ptr->text = (png_textp)png_malloc_warn(png_ptr,
Glenn Randers-Pehrsonbeb572e2006-08-19 13:59:24 -0500773 (png_uint_32)(info_ptr->max_text * sizeof(png_text)));
Glenn Randers-Pehrson07748d12002-05-25 11:12:10 -0500774 if (info_ptr->text == NULL)
775 return(1);
Glenn Randers-Pehrson1ef65b62000-05-12 06:19:53 -0500776#ifdef PNG_FREE_ME_SUPPORTED
777 info_ptr->free_me |= PNG_FREE_TEXT;
778#endif
Andreas Dilger47a0c421997-05-16 02:46:07 -0500779 }
780 png_debug1(3, "allocated %d entries for info_ptr->text\n",
781 info_ptr->max_text);
782 }
Andreas Dilger47a0c421997-05-16 02:46:07 -0500783 for (i = 0; i < num_text; i++)
784 {
Glenn Randers-Pehrson75294572000-05-06 14:09:57 -0500785 png_size_t text_length,key_len;
Glenn Randers-Pehrson75294572000-05-06 14:09:57 -0500786 png_size_t lang_len,lang_key_len;
Andreas Dilger47a0c421997-05-16 02:46:07 -0500787 png_textp textp = &(info_ptr->text[info_ptr->num_text]);
788
Glenn Randers-Pehrson3097f612001-05-07 14:52:45 -0500789 if (text_ptr[i].key == NULL)
Glenn Randers-Pehrson166c5a31999-12-10 09:43:02 -0600790 continue;
791
Glenn Randers-Pehrson61c32d92000-02-04 23:40:16 -0600792 key_len = png_strlen(text_ptr[i].key);
793
Glenn Randers-Pehrson1ef65b62000-05-12 06:19:53 -0500794 if(text_ptr[i].compression <= 0)
795 {
796 lang_len = 0;
797 lang_key_len = 0;
798 }
799 else
Glenn Randers-Pehrson75294572000-05-06 14:09:57 -0500800#ifdef PNG_iTXt_SUPPORTED
Glenn Randers-Pehrson61c32d92000-02-04 23:40:16 -0600801 {
802 /* set iTXt data */
Glenn Randers-Pehrson73d57cb2002-03-25 18:49:08 -0600803 if (text_ptr[i].lang != NULL)
Glenn Randers-Pehrson4accabb2000-04-14 14:20:47 -0500804 lang_len = png_strlen(text_ptr[i].lang);
805 else
806 lang_len = 0;
Glenn Randers-Pehrson3097f612001-05-07 14:52:45 -0500807 if (text_ptr[i].lang_key != NULL)
Glenn Randers-Pehrson4accabb2000-04-14 14:20:47 -0500808 lang_key_len = png_strlen(text_ptr[i].lang_key);
809 else
810 lang_key_len = 0;
Glenn Randers-Pehrson61c32d92000-02-04 23:40:16 -0600811 }
Glenn Randers-Pehrson75294572000-05-06 14:09:57 -0500812#else
813 {
Glenn Randers-Pehrsonbeb572e2006-08-19 13:59:24 -0500814 png_warning(png_ptr, "iTXt chunk not supported");
Glenn Randers-Pehrson75294572000-05-06 14:09:57 -0500815 continue;
816 }
817#endif
Andreas Dilger47a0c421997-05-16 02:46:07 -0500818
Glenn Randers-Pehrson3097f612001-05-07 14:52:45 -0500819 if (text_ptr[i].text == NULL || text_ptr[i].text[0] == '\0')
Andreas Dilger47a0c421997-05-16 02:46:07 -0500820 {
Glenn Randers-Pehrson61c32d92000-02-04 23:40:16 -0600821 text_length = 0;
Glenn Randers-Pehrson75294572000-05-06 14:09:57 -0500822#ifdef PNG_iTXt_SUPPORTED
Glenn Randers-Pehrson61c32d92000-02-04 23:40:16 -0600823 if(text_ptr[i].compression > 0)
824 textp->compression = PNG_ITXT_COMPRESSION_NONE;
825 else
Glenn Randers-Pehrson75294572000-05-06 14:09:57 -0500826#endif
Glenn Randers-Pehrson61c32d92000-02-04 23:40:16 -0600827 textp->compression = PNG_TEXT_COMPRESSION_NONE;
Andreas Dilger47a0c421997-05-16 02:46:07 -0500828 }
829 else
830 {
Glenn Randers-Pehrson61c32d92000-02-04 23:40:16 -0600831 text_length = png_strlen(text_ptr[i].text);
Andreas Dilger47a0c421997-05-16 02:46:07 -0500832 textp->compression = text_ptr[i].compression;
833 }
Glenn Randers-Pehrson6d8f3b01999-10-23 08:39:18 -0500834
Glenn Randers-Pehrson07748d12002-05-25 11:12:10 -0500835 textp->key = (png_charp)png_malloc_warn(png_ptr,
Glenn Randers-Pehrsonbeb572e2006-08-19 13:59:24 -0500836 (png_size_t)(key_len + text_length + lang_len + lang_key_len + 4));
Glenn Randers-Pehrson07748d12002-05-25 11:12:10 -0500837 if (textp->key == NULL)
838 return(1);
Glenn Randers-Pehrson8b6a8892001-05-18 04:54:50 -0500839 png_debug2(2, "Allocated %lu bytes at %x in png_set_text\n",
Glenn Randers-Pehrson6bc53be2006-06-16 07:52:03 -0500840 (unsigned long)(png_uint_32)(key_len + lang_len + lang_key_len
841 + text_length + 4),
Glenn Randers-Pehrson8b6a8892001-05-18 04:54:50 -0500842 (int)textp->key);
Glenn Randers-Pehrson6d8f3b01999-10-23 08:39:18 -0500843
844 png_memcpy(textp->key, text_ptr[i].key,
Glenn Randers-Pehrson61c32d92000-02-04 23:40:16 -0600845 (png_size_t)(key_len));
846 *(textp->key+key_len) = '\0';
Glenn Randers-Pehrson75294572000-05-06 14:09:57 -0500847#ifdef PNG_iTXt_SUPPORTED
Glenn Randers-Pehrson61c32d92000-02-04 23:40:16 -0600848 if (text_ptr[i].compression > 0)
849 {
850 textp->lang=textp->key + key_len + 1;
851 png_memcpy(textp->lang, text_ptr[i].lang, lang_len);
852 *(textp->lang+lang_len) = '\0';
853 textp->lang_key=textp->lang + lang_len + 1;
854 png_memcpy(textp->lang_key, text_ptr[i].lang_key, lang_key_len);
855 *(textp->lang_key+lang_key_len) = '\0';
856 textp->text=textp->lang_key + lang_key_len + 1;
857 }
858 else
Glenn Randers-Pehrson75294572000-05-06 14:09:57 -0500859#endif
Glenn Randers-Pehrson61c32d92000-02-04 23:40:16 -0600860 {
Glenn Randers-Pehrson75294572000-05-06 14:09:57 -0500861#ifdef PNG_iTXt_SUPPORTED
Glenn Randers-Pehrson3f549252001-10-27 07:35:13 -0500862 textp->lang=NULL;
863 textp->lang_key=NULL;
Glenn Randers-Pehrson75294572000-05-06 14:09:57 -0500864#endif
Glenn Randers-Pehrson61c32d92000-02-04 23:40:16 -0600865 textp->text=textp->key + key_len + 1;
866 }
Glenn Randers-Pehrson61c32d92000-02-04 23:40:16 -0600867 if(text_length)
Glenn Randers-Pehrson6d8f3b01999-10-23 08:39:18 -0500868 png_memcpy(textp->text, text_ptr[i].text,
Glenn Randers-Pehrson61c32d92000-02-04 23:40:16 -0600869 (png_size_t)(text_length));
Glenn Randers-Pehrson4accabb2000-04-14 14:20:47 -0500870 *(textp->text+text_length) = '\0';
Glenn Randers-Pehrson61c32d92000-02-04 23:40:16 -0600871
Glenn Randers-Pehrson75294572000-05-06 14:09:57 -0500872#ifdef PNG_iTXt_SUPPORTED
Glenn Randers-Pehrson61c32d92000-02-04 23:40:16 -0600873 if(textp->compression > 0)
874 {
875 textp->text_length = 0;
876 textp->itxt_length = text_length;
877 }
878 else
Glenn Randers-Pehrson75294572000-05-06 14:09:57 -0500879#endif
Glenn Randers-Pehrson61c32d92000-02-04 23:40:16 -0600880 {
881 textp->text_length = text_length;
Glenn Randers-Pehrson75294572000-05-06 14:09:57 -0500882#ifdef PNG_iTXt_SUPPORTED
Glenn Randers-Pehrson61c32d92000-02-04 23:40:16 -0600883 textp->itxt_length = 0;
Glenn Randers-Pehrson75294572000-05-06 14:09:57 -0500884#endif
Glenn Randers-Pehrson61c32d92000-02-04 23:40:16 -0600885 }
Glenn Randers-Pehrson6d8f3b01999-10-23 08:39:18 -0500886 info_ptr->text[info_ptr->num_text]= *textp;
Andreas Dilger47a0c421997-05-16 02:46:07 -0500887 info_ptr->num_text++;
888 png_debug1(3, "transferred text chunk %d\n", info_ptr->num_text);
889 }
Glenn Randers-Pehrson07748d12002-05-25 11:12:10 -0500890 return(0);
Andreas Dilger47a0c421997-05-16 02:46:07 -0500891}
892#endif
893
Glenn Randers-Pehrson166c5a31999-12-10 09:43:02 -0600894#if defined(PNG_tIME_SUPPORTED)
Glenn Randers-Pehrson75294572000-05-06 14:09:57 -0500895void PNGAPI
Andreas Dilger47a0c421997-05-16 02:46:07 -0500896png_set_tIME(png_structp png_ptr, png_infop info_ptr, png_timep mod_time)
897{
898 png_debug1(1, "in %s storage function\n", "tIME");
Glenn Randers-Pehrson4393a9a1999-09-17 12:27:26 -0500899 if (png_ptr == NULL || info_ptr == NULL ||
Glenn Randers-Pehrson5379b241999-11-27 10:22:33 -0600900 (png_ptr->mode & PNG_WROTE_tIME))
Andreas Dilger47a0c421997-05-16 02:46:07 -0500901 return;
902
Glenn Randers-Pehrsonbeb572e2006-08-19 13:59:24 -0500903 png_memcpy(&(info_ptr->mod_time), mod_time, sizeof(png_time));
Andreas Dilger47a0c421997-05-16 02:46:07 -0500904 info_ptr->valid |= PNG_INFO_tIME;
905}
906#endif
907
Glenn Randers-Pehrson166c5a31999-12-10 09:43:02 -0600908#if defined(PNG_tRNS_SUPPORTED)
Glenn Randers-Pehrson75294572000-05-06 14:09:57 -0500909void PNGAPI
Andreas Dilger47a0c421997-05-16 02:46:07 -0500910png_set_tRNS(png_structp png_ptr, png_infop info_ptr,
911 png_bytep trans, int num_trans, png_color_16p trans_values)
912{
913 png_debug1(1, "in %s storage function\n", "tRNS");
Glenn Randers-Pehrsona357b991998-02-08 20:56:40 -0600914 if (png_ptr == NULL || info_ptr == NULL)
Andreas Dilger47a0c421997-05-16 02:46:07 -0500915 return;
916
917 if (trans != NULL)
Glenn Randers-Pehrson76e5fd62000-12-28 07:50:05 -0600918 {
919 /*
Glenn Randers-Pehrsond1e8c862002-06-20 06:54:34 -0500920 * It may not actually be necessary to set png_ptr->trans here;
921 * we do it for backward compatibility with the way the png_handle_tRNS
922 * function used to do the allocation.
923 */
Glenn Randers-Pehrson76e5fd62000-12-28 07:50:05 -0600924#ifdef PNG_FREE_ME_SUPPORTED
925 png_free_data(png_ptr, info_ptr, PNG_FREE_TRNS, 0);
926#endif
Glenn Randers-Pehrson9c3ab682006-02-20 22:09:05 -0600927 /* Changed from num_trans to PNG_MAX_PALETTE_LENGTH in version 1.2.1 */
Glenn Randers-Pehrson3097f612001-05-07 14:52:45 -0500928 png_ptr->trans = info_ptr->trans = (png_bytep)png_malloc(png_ptr,
Glenn Randers-Pehrsonbeb572e2006-08-19 13:59:24 -0500929 PNG_MAX_PALETTE_LENGTH);
Glenn Randers-Pehrson9c3ab682006-02-20 22:09:05 -0600930 if (num_trans <= PNG_MAX_PALETTE_LENGTH)
931 png_memcpy(info_ptr->trans, trans, (png_size_t)num_trans);
Glenn Randers-Pehrson76e5fd62000-12-28 07:50:05 -0600932#ifdef PNG_FREE_ME_SUPPORTED
933 info_ptr->free_me |= PNG_FREE_TRNS;
934#else
935 png_ptr->flags |= PNG_FLAG_FREE_TRNS;
936#endif
937 }
Andreas Dilger47a0c421997-05-16 02:46:07 -0500938
939 if (trans_values != NULL)
940 {
941 png_memcpy(&(info_ptr->trans_values), trans_values,
Glenn Randers-Pehrsonbeb572e2006-08-19 13:59:24 -0500942 sizeof(png_color_16));
Glenn Randers-Pehrsonb6ce43d1998-01-01 07:13:13 -0600943 if (num_trans == 0)
944 num_trans = 1;
Andreas Dilger47a0c421997-05-16 02:46:07 -0500945 }
946 info_ptr->num_trans = (png_uint_16)num_trans;
947 info_ptr->valid |= PNG_INFO_tRNS;
948}
949#endif
950
Glenn Randers-Pehrson166c5a31999-12-10 09:43:02 -0600951#if defined(PNG_sPLT_SUPPORTED)
Glenn Randers-Pehrson75294572000-05-06 14:09:57 -0500952void PNGAPI
Glenn Randers-Pehrsona77ef622000-02-18 13:48:52 -0600953png_set_sPLT(png_structp png_ptr,
Glenn Randers-Pehrson520a7642000-03-21 05:13:06 -0600954 png_infop info_ptr, png_sPLT_tp entries, int nentries)
Glenn Randers-Pehrson166c5a31999-12-10 09:43:02 -0600955{
Glenn Randers-Pehrson520a7642000-03-21 05:13:06 -0600956 png_sPLT_tp np;
Glenn Randers-Pehrson166c5a31999-12-10 09:43:02 -0600957 int i;
958
Glenn Randers-Pehrson170b70c2006-03-10 10:19:04 -0600959 if (png_ptr == NULL || info_ptr == NULL)
960 return;
961
Glenn Randers-Pehrson07748d12002-05-25 11:12:10 -0500962 np = (png_sPLT_tp)png_malloc_warn(png_ptr,
Glenn Randers-Pehrsonbeb572e2006-08-19 13:59:24 -0500963 (info_ptr->splt_palettes_num + nentries) * sizeof(png_sPLT_t));
Glenn Randers-Pehrson07748d12002-05-25 11:12:10 -0500964 if (np == NULL)
965 {
Glenn Randers-Pehrsonbeb572e2006-08-19 13:59:24 -0500966 png_warning(png_ptr, "No memory for sPLT palettes");
Glenn Randers-Pehrson07748d12002-05-25 11:12:10 -0500967 return;
968 }
Glenn Randers-Pehrson166c5a31999-12-10 09:43:02 -0600969
Glenn Randers-Pehrson61c32d92000-02-04 23:40:16 -0600970 png_memcpy(np, info_ptr->splt_palettes,
Glenn Randers-Pehrsonbeb572e2006-08-19 13:59:24 -0500971 info_ptr->splt_palettes_num * sizeof(png_sPLT_t));
Glenn Randers-Pehrson166c5a31999-12-10 09:43:02 -0600972 png_free(png_ptr, info_ptr->splt_palettes);
Glenn Randers-Pehrson4766a242000-07-17 06:17:09 -0500973 info_ptr->splt_palettes=NULL;
Glenn Randers-Pehrson166c5a31999-12-10 09:43:02 -0600974
975 for (i = 0; i < nentries; i++)
976 {
Glenn Randers-Pehrson520a7642000-03-21 05:13:06 -0600977 png_sPLT_tp to = np + info_ptr->splt_palettes_num + i;
978 png_sPLT_tp from = entries + i;
Glenn Randers-Pehrson166c5a31999-12-10 09:43:02 -0600979
Glenn Randers-Pehrsonbeb572e2006-08-19 13:59:24 -0500980 to->name = (png_charp)png_malloc(png_ptr, png_strlen(from->name) + 1);
Glenn Randers-Pehrson5fea36f2004-07-28 08:20:44 -0500981 /* TODO: use png_malloc_warn */
Glenn Randers-Pehrson166c5a31999-12-10 09:43:02 -0600982 png_strcpy(to->name, from->name);
Glenn Randers-Pehrson520a7642000-03-21 05:13:06 -0600983 to->entries = (png_sPLT_entryp)png_malloc(png_ptr,
Glenn Randers-Pehrsonbeb572e2006-08-19 13:59:24 -0500984 from->nentries * sizeof(png_sPLT_t));
Glenn Randers-Pehrson5fea36f2004-07-28 08:20:44 -0500985 /* TODO: use png_malloc_warn */
Glenn Randers-Pehrson82ae3832001-04-20 10:32:10 -0500986 png_memcpy(to->entries, from->entries,
Glenn Randers-Pehrsonbeb572e2006-08-19 13:59:24 -0500987 from->nentries * sizeof(png_sPLT_t));
Glenn Randers-Pehrson82ae3832001-04-20 10:32:10 -0500988 to->nentries = from->nentries;
989 to->depth = from->depth;
Glenn Randers-Pehrson166c5a31999-12-10 09:43:02 -0600990 }
991
992 info_ptr->splt_palettes = np;
993 info_ptr->splt_palettes_num += nentries;
Glenn Randers-Pehrson61c32d92000-02-04 23:40:16 -0600994 info_ptr->valid |= PNG_INFO_sPLT;
Glenn Randers-Pehrson6942d532000-05-01 09:31:54 -0500995#ifdef PNG_FREE_ME_SUPPORTED
Glenn Randers-Pehrsona77ef622000-02-18 13:48:52 -0600996 info_ptr->free_me |= PNG_FREE_SPLT;
Glenn Randers-Pehrson6942d532000-05-01 09:31:54 -0500997#endif
Glenn Randers-Pehrson166c5a31999-12-10 09:43:02 -0600998}
999#endif /* PNG_sPLT_SUPPORTED */
1000
Glenn Randers-Pehrson3097f612001-05-07 14:52:45 -05001001#if defined(PNG_UNKNOWN_CHUNKS_SUPPORTED)
Glenn Randers-Pehrson75294572000-05-06 14:09:57 -05001002void PNGAPI
Glenn Randers-Pehrson166c5a31999-12-10 09:43:02 -06001003png_set_unknown_chunks(png_structp png_ptr,
Glenn Randers-Pehrson61c32d92000-02-04 23:40:16 -06001004 png_infop info_ptr, png_unknown_chunkp unknowns, int num_unknowns)
Glenn Randers-Pehrson166c5a31999-12-10 09:43:02 -06001005{
1006 png_unknown_chunkp np;
1007 int i;
1008
Glenn Randers-Pehrson61c32d92000-02-04 23:40:16 -06001009 if (png_ptr == NULL || info_ptr == NULL || num_unknowns == 0)
1010 return;
Glenn Randers-Pehrson166c5a31999-12-10 09:43:02 -06001011
Glenn Randers-Pehrson07748d12002-05-25 11:12:10 -05001012 np = (png_unknown_chunkp)png_malloc_warn(png_ptr,
Glenn Randers-Pehrson61c32d92000-02-04 23:40:16 -06001013 (info_ptr->unknown_chunks_num + num_unknowns) *
Glenn Randers-Pehrsonbeb572e2006-08-19 13:59:24 -05001014 sizeof(png_unknown_chunk));
Glenn Randers-Pehrson07748d12002-05-25 11:12:10 -05001015 if (np == NULL)
1016 {
Glenn Randers-Pehrsonbeb572e2006-08-19 13:59:24 -05001017 png_warning(png_ptr, "Out of memory while processing unknown chunk");
Glenn Randers-Pehrson07748d12002-05-25 11:12:10 -05001018 return;
1019 }
Glenn Randers-Pehrson61c32d92000-02-04 23:40:16 -06001020
1021 png_memcpy(np, info_ptr->unknown_chunks,
Glenn Randers-Pehrsonbeb572e2006-08-19 13:59:24 -05001022 info_ptr->unknown_chunks_num * sizeof(png_unknown_chunk));
Glenn Randers-Pehrson166c5a31999-12-10 09:43:02 -06001023 png_free(png_ptr, info_ptr->unknown_chunks);
Glenn Randers-Pehrson4766a242000-07-17 06:17:09 -05001024 info_ptr->unknown_chunks=NULL;
Glenn Randers-Pehrson166c5a31999-12-10 09:43:02 -06001025
Glenn Randers-Pehrson61c32d92000-02-04 23:40:16 -06001026 for (i = 0; i < num_unknowns; i++)
Glenn Randers-Pehrson166c5a31999-12-10 09:43:02 -06001027 {
1028 png_unknown_chunkp to = np + info_ptr->unknown_chunks_num + i;
1029 png_unknown_chunkp from = unknowns + i;
1030
Glenn Randers-Pehrson5fea36f2004-07-28 08:20:44 -05001031 png_strncpy((png_charp)to->name, (png_charp)from->name, 5);
1032 to->data = (png_bytep)png_malloc_warn(png_ptr, from->size);
Glenn Randers-Pehrson07748d12002-05-25 11:12:10 -05001033 if (to->data == NULL)
Glenn Randers-Pehrson5fea36f2004-07-28 08:20:44 -05001034 {
Glenn Randers-Pehrsonbeb572e2006-08-19 13:59:24 -05001035 png_warning(png_ptr, "Out of memory processing unknown chunk");
Glenn Randers-Pehrson5fea36f2004-07-28 08:20:44 -05001036 }
Glenn Randers-Pehrson07748d12002-05-25 11:12:10 -05001037 else
1038 {
Glenn Randers-Pehrson5fea36f2004-07-28 08:20:44 -05001039 png_memcpy(to->data, from->data, from->size);
1040 to->size = from->size;
Glenn Randers-Pehrson166c5a31999-12-10 09:43:02 -06001041
Glenn Randers-Pehrson5fea36f2004-07-28 08:20:44 -05001042 /* note our location in the read or write sequence */
1043 to->location = (png_byte)(png_ptr->mode & 0xff);
Glenn Randers-Pehrson07748d12002-05-25 11:12:10 -05001044 }
Glenn Randers-Pehrson166c5a31999-12-10 09:43:02 -06001045 }
1046
1047 info_ptr->unknown_chunks = np;
Glenn Randers-Pehrson61c32d92000-02-04 23:40:16 -06001048 info_ptr->unknown_chunks_num += num_unknowns;
Glenn Randers-Pehrson6942d532000-05-01 09:31:54 -05001049#ifdef PNG_FREE_ME_SUPPORTED
Glenn Randers-Pehrsona77ef622000-02-18 13:48:52 -06001050 info_ptr->free_me |= PNG_FREE_UNKN;
Glenn Randers-Pehrson6942d532000-05-01 09:31:54 -05001051#endif
Glenn Randers-Pehrson166c5a31999-12-10 09:43:02 -06001052}
Glenn Randers-Pehrson75294572000-05-06 14:09:57 -05001053void PNGAPI
Glenn Randers-Pehrson228bd392000-04-23 23:14:02 -05001054png_set_unknown_chunk_location(png_structp png_ptr, png_infop info_ptr,
1055 int chunk, int location)
1056{
Glenn Randers-Pehrsond56aca72000-11-23 11:51:42 -06001057 if(png_ptr != NULL && info_ptr != NULL && chunk >= 0 && chunk <
Glenn Randers-Pehrson228bd392000-04-23 23:14:02 -05001058 (int)info_ptr->unknown_chunks_num)
1059 info_ptr->unknown_chunks[chunk].location = (png_byte)location;
1060}
Glenn Randers-Pehrson166c5a31999-12-10 09:43:02 -06001061#endif
1062
Glenn Randers-Pehrson5e5c1e12000-11-10 12:26:19 -06001063#if defined(PNG_MNG_FEATURES_SUPPORTED)
1064png_uint_32 PNGAPI
1065png_permit_mng_features (png_structp png_ptr, png_uint_32 mng_features)
1066{
1067 png_debug(1, "in png_permit_mng_features\n");
1068 if (png_ptr == NULL)
1069 return (png_uint_32)0;
1070 png_ptr->mng_features_permitted =
1071 (png_byte)(mng_features & PNG_ALL_MNG_FEATURES);
1072 return (png_uint_32)png_ptr->mng_features_permitted;
Glenn Randers-Pehrson4393a9a1999-09-17 12:27:26 -05001073}
1074#endif
Glenn Randers-Pehrson166c5a31999-12-10 09:43:02 -06001075
Glenn Randers-Pehrson61c32d92000-02-04 23:40:16 -06001076#if defined(PNG_UNKNOWN_CHUNKS_SUPPORTED)
Glenn Randers-Pehrson75294572000-05-06 14:09:57 -05001077void PNGAPI
Glenn Randers-Pehrson61c32d92000-02-04 23:40:16 -06001078png_set_keep_unknown_chunks(png_structp png_ptr, int keep, png_bytep
1079 chunk_list, int num_chunks)
1080{
1081 png_bytep new_list, p;
1082 int i, old_num_chunks;
Glenn Randers-Pehrson170b70c2006-03-10 10:19:04 -06001083 if (png_ptr == NULL)
1084 return;
Glenn Randers-Pehrson61c32d92000-02-04 23:40:16 -06001085 if (num_chunks == 0)
1086 {
Glenn Randers-Pehrsondff799e2004-08-07 21:42:49 -05001087 if(keep == PNG_HANDLE_CHUNK_ALWAYS || keep == PNG_HANDLE_CHUNK_IF_SAFE)
Glenn Randers-Pehrson61c32d92000-02-04 23:40:16 -06001088 png_ptr->flags |= PNG_FLAG_KEEP_UNKNOWN_CHUNKS;
1089 else
1090 png_ptr->flags &= ~PNG_FLAG_KEEP_UNKNOWN_CHUNKS;
Glenn Randers-Pehrson166c5a31999-12-10 09:43:02 -06001091
Glenn Randers-Pehrsondff799e2004-08-07 21:42:49 -05001092 if(keep == PNG_HANDLE_CHUNK_ALWAYS)
Glenn Randers-Pehrson61c32d92000-02-04 23:40:16 -06001093 png_ptr->flags |= PNG_FLAG_KEEP_UNSAFE_CHUNKS;
1094 else
1095 png_ptr->flags &= ~PNG_FLAG_KEEP_UNSAFE_CHUNKS;
1096 return;
1097 }
1098 if (chunk_list == NULL)
1099 return;
1100 old_num_chunks=png_ptr->num_chunk_list;
Glenn Randers-Pehrsonb1828932001-06-23 08:03:17 -05001101 new_list=(png_bytep)png_malloc(png_ptr,
Glenn Randers-Pehrsonbeb572e2006-08-19 13:59:24 -05001102 (png_size_t)(5*(num_chunks+old_num_chunks)));
Glenn Randers-Pehrson3097f612001-05-07 14:52:45 -05001103 if(png_ptr->chunk_list != NULL)
Glenn Randers-Pehrson61c32d92000-02-04 23:40:16 -06001104 {
Glenn Randers-Pehrsonb1828932001-06-23 08:03:17 -05001105 png_memcpy(new_list, png_ptr->chunk_list,
1106 (png_size_t)(5*old_num_chunks));
Glenn Randers-Pehrsona77ef622000-02-18 13:48:52 -06001107 png_free(png_ptr, png_ptr->chunk_list);
Glenn Randers-Pehrson4766a242000-07-17 06:17:09 -05001108 png_ptr->chunk_list=NULL;
Glenn Randers-Pehrson61c32d92000-02-04 23:40:16 -06001109 }
Glenn Randers-Pehrsonb1828932001-06-23 08:03:17 -05001110 png_memcpy(new_list+5*old_num_chunks, chunk_list,
1111 (png_size_t)(5*num_chunks));
Glenn Randers-Pehrson61c32d92000-02-04 23:40:16 -06001112 for (p=new_list+5*old_num_chunks+4, i=0; i<num_chunks; i++, p+=5)
1113 *p=(png_byte)keep;
1114 png_ptr->num_chunk_list=old_num_chunks+num_chunks;
1115 png_ptr->chunk_list=new_list;
Glenn Randers-Pehrson6942d532000-05-01 09:31:54 -05001116#ifdef PNG_FREE_ME_SUPPORTED
Glenn Randers-Pehrsona77ef622000-02-18 13:48:52 -06001117 png_ptr->free_me |= PNG_FREE_LIST;
Glenn Randers-Pehrson6942d532000-05-01 09:31:54 -05001118#endif
Glenn Randers-Pehrson61c32d92000-02-04 23:40:16 -06001119}
1120#endif
Glenn Randers-Pehrson166c5a31999-12-10 09:43:02 -06001121
Glenn Randers-Pehrson61c32d92000-02-04 23:40:16 -06001122#if defined(PNG_READ_USER_CHUNKS_SUPPORTED)
Glenn Randers-Pehrson75294572000-05-06 14:09:57 -05001123void PNGAPI
Glenn Randers-Pehrson61c32d92000-02-04 23:40:16 -06001124png_set_read_user_chunk_fn(png_structp png_ptr, png_voidp user_chunk_ptr,
1125 png_user_chunk_ptr read_user_chunk_fn)
1126{
1127 png_debug(1, "in png_set_read_user_chunk_fn\n");
Glenn Randers-Pehrson170b70c2006-03-10 10:19:04 -06001128 if (png_ptr == NULL)
1129 return;
Glenn Randers-Pehrson61c32d92000-02-04 23:40:16 -06001130 png_ptr->read_user_chunk_fn = read_user_chunk_fn;
1131 png_ptr->user_chunk_ptr = user_chunk_ptr;
1132}
1133#endif
Glenn Randers-Pehrson166c5a31999-12-10 09:43:02 -06001134
Glenn Randers-Pehrsona77ef622000-02-18 13:48:52 -06001135#if defined(PNG_INFO_IMAGE_SUPPORTED)
Glenn Randers-Pehrson75294572000-05-06 14:09:57 -05001136void PNGAPI
Glenn Randers-Pehrsona77ef622000-02-18 13:48:52 -06001137png_set_rows(png_structp png_ptr, png_infop info_ptr, png_bytepp row_pointers)
1138{
1139 png_debug1(1, "in %s storage function\n", "rows");
Glenn Randers-Pehrsonfc4a1432000-05-17 17:39:34 -05001140
Glenn Randers-Pehrsona77ef622000-02-18 13:48:52 -06001141 if (png_ptr == NULL || info_ptr == NULL)
1142 return;
1143
Glenn Randers-Pehrsonec61c232000-05-16 06:17:36 -05001144 if(info_ptr->row_pointers && (info_ptr->row_pointers != row_pointers))
Glenn Randers-Pehrson4accabb2000-04-14 14:20:47 -05001145 png_free_data(png_ptr, info_ptr, PNG_FREE_ROWS, 0);
Glenn Randers-Pehrsonfc4a1432000-05-17 17:39:34 -05001146 info_ptr->row_pointers = row_pointers;
1147 if(row_pointers)
1148 info_ptr->valid |= PNG_INFO_IDAT;
Glenn Randers-Pehrsona77ef622000-02-18 13:48:52 -06001149}
1150#endif
1151
Glenn Randers-Pehrson5fea36f2004-07-28 08:20:44 -05001152#ifdef PNG_WRITE_SUPPORTED
Glenn Randers-Pehrson75294572000-05-06 14:09:57 -05001153void PNGAPI
Glenn Randers-Pehrsonbeb572e2006-08-19 13:59:24 -05001154png_set_compression_buffer_size(png_structp png_ptr, png_size_t size)
Glenn Randers-Pehrson228bd392000-04-23 23:14:02 -05001155{
Glenn Randers-Pehrson170b70c2006-03-10 10:19:04 -06001156 if (png_ptr == NULL)
1157 return;
Glenn Randers-Pehrson228bd392000-04-23 23:14:02 -05001158 if(png_ptr->zbuf)
1159 png_free(png_ptr, png_ptr->zbuf);
Glenn Randers-Pehrsonbeb572e2006-08-19 13:59:24 -05001160 png_ptr->zbuf_size = size;
Glenn Randers-Pehrson228bd392000-04-23 23:14:02 -05001161 png_ptr->zbuf = (png_bytep)png_malloc(png_ptr, size);
1162 png_ptr->zstream.next_out = png_ptr->zbuf;
1163 png_ptr->zstream.avail_out = (uInt)png_ptr->zbuf_size;
1164}
Glenn Randers-Pehrson5fea36f2004-07-28 08:20:44 -05001165#endif
Glenn Randers-Pehrsonfc4a1432000-05-17 17:39:34 -05001166
1167void PNGAPI
1168png_set_invalid(png_structp png_ptr, png_infop info_ptr, int mask)
1169{
1170 if (png_ptr && info_ptr)
1171 info_ptr->valid &= ~(mask);
1172}
Glenn Randers-Pehrson231e6872001-01-12 15:13:06 -06001173
Glenn Randers-Pehrsone68f5a32001-05-14 09:20:53 -05001174
1175#ifdef PNG_ASSEMBLER_CODE_SUPPORTED
1176/* this function was added to libpng 1.2.0 and should always exist by default */
1177void PNGAPI
1178png_set_asm_flags (png_structp png_ptr, png_uint_32 asm_flags)
1179{
1180 png_uint_32 settable_asm_flags;
1181 png_uint_32 settable_mmx_flags;
1182
Glenn Randers-Pehrson170b70c2006-03-10 10:19:04 -06001183 if (png_ptr == NULL)
1184 return;
1185
Glenn Randers-Pehrsone68f5a32001-05-14 09:20:53 -05001186 settable_mmx_flags =
Glenn Randers-Pehrsond60b8fa2006-04-20 21:31:14 -05001187#ifdef PNG_MMX_CODE_SUPPORTED
Glenn Randers-Pehrson17218292006-04-20 07:20:46 -05001188#ifdef PNG_HAVE_MMX_COMBINE_ROW
Glenn Randers-Pehrsone68f5a32001-05-14 09:20:53 -05001189 PNG_ASM_FLAG_MMX_READ_COMBINE_ROW |
1190#endif
Glenn Randers-Pehrson17218292006-04-20 07:20:46 -05001191#ifdef PNG_HAVE_MMX_READ_INTERLACE
Glenn Randers-Pehrsone68f5a32001-05-14 09:20:53 -05001192 PNG_ASM_FLAG_MMX_READ_INTERLACE |
1193#endif
Glenn Randers-Pehrson17218292006-04-20 07:20:46 -05001194#ifdef PNG_HAVE_MMX_READ_FILTER_ROW
Glenn Randers-Pehrsone68f5a32001-05-14 09:20:53 -05001195 PNG_ASM_FLAG_MMX_READ_FILTER_SUB |
1196 PNG_ASM_FLAG_MMX_READ_FILTER_UP |
1197 PNG_ASM_FLAG_MMX_READ_FILTER_AVG |
1198 PNG_ASM_FLAG_MMX_READ_FILTER_PAETH |
1199#endif
Glenn Randers-Pehrsond60b8fa2006-04-20 21:31:14 -05001200#endif
Glenn Randers-Pehrsone68f5a32001-05-14 09:20:53 -05001201 0;
1202
1203 /* could be some non-MMX ones in the future, but not currently: */
1204 settable_asm_flags = settable_mmx_flags;
1205
Glenn Randers-Pehrson17218292006-04-20 07:20:46 -05001206#ifdef PNG_MMX_CODE_SUPPORTED
Glenn Randers-Pehrsone68f5a32001-05-14 09:20:53 -05001207 if (!(png_ptr->asm_flags & PNG_ASM_FLAG_MMX_SUPPORT_COMPILED) ||
1208 !(png_ptr->asm_flags & PNG_ASM_FLAG_MMX_SUPPORT_IN_CPU))
1209 {
1210 /* clear all MMX flags if MMX isn't supported */
1211 settable_asm_flags &= ~settable_mmx_flags;
1212 png_ptr->asm_flags &= ~settable_mmx_flags;
1213 }
Glenn Randers-Pehrson17218292006-04-20 07:20:46 -05001214#endif
Glenn Randers-Pehrsone68f5a32001-05-14 09:20:53 -05001215
1216 /* we're replacing the settable bits with those passed in by the user,
1217 * so first zero them out of the master copy, then logical-OR in the
1218 * allowed subset that was requested */
1219
Glenn Randers-Pehrsond1e8c862002-06-20 06:54:34 -05001220 png_ptr->asm_flags &= ~settable_asm_flags; /* zero them */
1221 png_ptr->asm_flags |= (asm_flags & settable_asm_flags); /* set them */
Glenn Randers-Pehrsone68f5a32001-05-14 09:20:53 -05001222}
1223#endif /* ?PNG_ASSEMBLER_CODE_SUPPORTED */
1224
Glenn Randers-Pehrson17218292006-04-20 07:20:46 -05001225#ifdef PNG_MMX_CODE_SUPPORTED
Glenn Randers-Pehrsone68f5a32001-05-14 09:20:53 -05001226/* this function was added to libpng 1.2.0 */
1227void PNGAPI
1228png_set_mmx_thresholds (png_structp png_ptr,
1229 png_byte mmx_bitdepth_threshold,
1230 png_uint_32 mmx_rowbytes_threshold)
1231{
Glenn Randers-Pehrson170b70c2006-03-10 10:19:04 -06001232 if (png_ptr == NULL)
1233 return;
Glenn Randers-Pehrsone68f5a32001-05-14 09:20:53 -05001234 png_ptr->mmx_bitdepth_threshold = mmx_bitdepth_threshold;
1235 png_ptr->mmx_rowbytes_threshold = mmx_rowbytes_threshold;
1236}
Glenn Randers-Pehrson17218292006-04-20 07:20:46 -05001237#endif /* ?PNG_MMX_CODE_SUPPORTED */
Glenn Randers-Pehrson272489d2004-08-04 06:34:52 -05001238
1239#ifdef PNG_SET_USER_LIMITS_SUPPORTED
1240/* this function was added to libpng 1.2.6 */
1241void PNGAPI
1242png_set_user_limits (png_structp png_ptr, png_uint_32 user_width_max,
1243 png_uint_32 user_height_max)
1244{
1245 /* Images with dimensions larger than these limits will be
1246 * rejected by png_set_IHDR(). To accept any PNG datastream
1247 * regardless of dimensions, set both limits to 0x7ffffffL.
1248 */
1249 png_ptr->user_width_max = user_width_max;
1250 png_ptr->user_height_max = user_height_max;
1251}
1252#endif /* ?PNG_SET_USER_LIMITS_SUPPORTED */
1253
Glenn Randers-Pehrson6bc53be2006-06-16 07:52:03 -05001254
1255#if defined(PNG_BENIGN_ERRORS_SUPPORTED)
1256void PNGAPI
1257png_set_benign_errors(png_structp png_ptr, int allowed)
1258{
1259 png_debug(1, "in png_set_benign_errors\n");
1260 if (allowed)
1261 png_ptr->flags |= PNG_FLAG_BENIGN_ERRORS_WARN;
1262 else
1263 png_ptr->flags &= ~PNG_FLAG_BENIGN_ERRORS_WARN;
1264}
1265#endif /* PNG_BENIGN_ERRORS_SUPPORTED */
1266
Glenn Randers-Pehrson9c3ab682006-02-20 22:09:05 -06001267#endif /* PNG_READ_SUPPORTED || PNG_WRITE_SUPPORTED */