blob: a2c2551da92ab3fe86e170f6426d812f58480350 [file] [log] [blame]
The Android Open Source Project893912b2009-03-03 19:30:05 -08001/*
2 * pnm2png.c --- conversion from PBM/PGM/PPM-file to PNG-file
xNombred07bb0d2020-03-10 20:17:12 +01003 * copyright (C) 1999-2019 by Willem van Schaik <willem at schaik dot com>
The Android Open Source Project893912b2009-03-03 19:30:05 -08004 *
xNombred07bb0d2020-03-10 20:17:12 +01005 * This software is released under the MIT license. For conditions of
6 * distribution and use, see the LICENSE file part of this package.
The Android Open Source Project893912b2009-03-03 19:30:05 -08007 */
8
9#include <stdio.h>
10#include <stdlib.h>
The Android Open Source Project893912b2009-03-03 19:30:05 -080011#include <fcntl.h>
The Android Open Source Project893912b2009-03-03 19:30:05 -080012
13#ifndef BOOL
14#define BOOL unsigned char
15#endif
16#ifndef TRUE
17#define TRUE (BOOL) 1
18#endif
19#ifndef FALSE
20#define FALSE (BOOL) 0
21#endif
22
xNombred07bb0d2020-03-10 20:17:12 +010023/* make pnm2png verbose so we can find problems (needs to be before png.h) */
The Android Open Source Project893912b2009-03-03 19:30:05 -080024#ifndef PNG_DEBUG
25#define PNG_DEBUG 0
26#endif
27
28#include "png.h"
29
The Android Open Source Project893912b2009-03-03 19:30:05 -080030/* function prototypes */
31
xNombred07bb0d2020-03-10 20:17:12 +010032int main (int argc, char *argv[]);
The Android Open Source Project893912b2009-03-03 19:30:05 -080033void usage ();
xNombred07bb0d2020-03-10 20:17:12 +010034BOOL pnm2png (FILE *pnm_file, FILE *png_file, FILE *alpha_file,
35 BOOL interlace, BOOL alpha);
36void get_token (FILE *pnm_file, char *token_buf, size_t token_buf_size);
The Android Open Source Project893912b2009-03-03 19:30:05 -080037png_uint_32 get_data (FILE *pnm_file, int depth);
38png_uint_32 get_value (FILE *pnm_file, int depth);
39
40/*
41 * main
42 */
43
xNombred07bb0d2020-03-10 20:17:12 +010044int main (int argc, char *argv[])
The Android Open Source Project893912b2009-03-03 19:30:05 -080045{
46 FILE *fp_rd = stdin;
47 FILE *fp_al = NULL;
48 FILE *fp_wr = stdout;
49 BOOL interlace = FALSE;
50 BOOL alpha = FALSE;
51 int argi;
52
53 for (argi = 1; argi < argc; argi++)
54 {
55 if (argv[argi][0] == '-')
56 {
57 switch (argv[argi][1])
58 {
59 case 'i':
60 interlace = TRUE;
61 break;
62 case 'a':
63 alpha = TRUE;
64 argi++;
65 if ((fp_al = fopen (argv[argi], "rb")) == NULL)
66 {
67 fprintf (stderr, "PNM2PNG\n");
68 fprintf (stderr, "Error: alpha-channel file %s does not exist\n",
xNombred07bb0d2020-03-10 20:17:12 +010069 argv[argi]);
The Android Open Source Project893912b2009-03-03 19:30:05 -080070 exit (1);
71 }
72 break;
73 case 'h':
74 case '?':
xNombred07bb0d2020-03-10 20:17:12 +010075 usage ();
76 exit (0);
The Android Open Source Project893912b2009-03-03 19:30:05 -080077 break;
78 default:
79 fprintf (stderr, "PNM2PNG\n");
80 fprintf (stderr, "Error: unknown option %s\n", argv[argi]);
xNombred07bb0d2020-03-10 20:17:12 +010081 usage ();
82 exit (1);
The Android Open Source Project893912b2009-03-03 19:30:05 -080083 break;
84 } /* end switch */
85 }
86 else if (fp_rd == stdin)
87 {
88 if ((fp_rd = fopen (argv[argi], "rb")) == NULL)
89 {
90 fprintf (stderr, "PNM2PNG\n");
91 fprintf (stderr, "Error: file %s does not exist\n", argv[argi]);
92 exit (1);
93 }
94 }
95 else if (fp_wr == stdout)
96 {
97 if ((fp_wr = fopen (argv[argi], "wb")) == NULL)
98 {
99 fprintf (stderr, "PNM2PNG\n");
xNombred07bb0d2020-03-10 20:17:12 +0100100 fprintf (stderr, "Error: cannot create PNG-file %s\n", argv[argi]);
The Android Open Source Project893912b2009-03-03 19:30:05 -0800101 exit (1);
102 }
103 }
104 else
105 {
106 fprintf (stderr, "PNM2PNG\n");
107 fprintf (stderr, "Error: too many parameters\n");
xNombred07bb0d2020-03-10 20:17:12 +0100108 usage ();
The Android Open Source Project893912b2009-03-03 19:30:05 -0800109 exit (1);
110 }
111 } /* end for */
112
xNombred07bb0d2020-03-10 20:17:12 +0100113#if defined(O_BINARY) && (O_BINARY != 0)
114 /* set stdin/stdout to binary,
115 * we're reading the PNM always! in binary format
116 */
The Android Open Source Project893912b2009-03-03 19:30:05 -0800117 if (fp_rd == stdin)
xNombred07bb0d2020-03-10 20:17:12 +0100118 setmode (fileno (stdin), O_BINARY);
The Android Open Source Project893912b2009-03-03 19:30:05 -0800119 if (fp_wr == stdout)
xNombred07bb0d2020-03-10 20:17:12 +0100120 setmode (fileno (stdout), O_BINARY);
The Android Open Source Project893912b2009-03-03 19:30:05 -0800121#endif
122
123 /* call the conversion program itself */
124 if (pnm2png (fp_rd, fp_wr, fp_al, interlace, alpha) == FALSE)
125 {
126 fprintf (stderr, "PNM2PNG\n");
127 fprintf (stderr, "Error: unsuccessful converting to PNG-image\n");
128 exit (1);
129 }
130
131 /* close input file */
132 fclose (fp_rd);
133 /* close output file */
134 fclose (fp_wr);
135 /* close alpha file */
136 if (alpha)
137 fclose (fp_al);
138
139 return 0;
140}
141
142/*
143 * usage
144 */
145
xNombred07bb0d2020-03-10 20:17:12 +0100146void usage ()
The Android Open Source Project893912b2009-03-03 19:30:05 -0800147{
148 fprintf (stderr, "PNM2PNG\n");
149 fprintf (stderr, " by Willem van Schaik, 1999\n");
The Android Open Source Project893912b2009-03-03 19:30:05 -0800150 fprintf (stderr, "Usage: pnm2png [options] <file>.<pnm> [<file>.png]\n");
151 fprintf (stderr, " or: ... | pnm2png [options]\n");
152 fprintf (stderr, "Options:\n");
153 fprintf (stderr, " -i[nterlace] write png-file with interlacing on\n");
Matt Sarett9b1fe632015-11-25 10:21:17 -0500154 fprintf (stderr,
155 " -a[lpha] <file>.pgm read PNG alpha channel as pgm-file\n");
The Android Open Source Project893912b2009-03-03 19:30:05 -0800156 fprintf (stderr, " -h | -? print this help-information\n");
157}
158
159/*
160 * pnm2png
161 */
162
xNombred07bb0d2020-03-10 20:17:12 +0100163BOOL pnm2png (FILE *pnm_file, FILE *png_file, FILE *alpha_file,
164 BOOL interlace, BOOL alpha)
The Android Open Source Project893912b2009-03-03 19:30:05 -0800165{
166 png_struct *png_ptr = NULL;
167 png_info *info_ptr = NULL;
168 png_byte *png_pixels = NULL;
169 png_byte **row_pointers = NULL;
170 png_byte *pix_ptr = NULL;
xNombred07bb0d2020-03-10 20:17:12 +0100171 volatile png_uint_32 row_bytes;
The Android Open Source Project893912b2009-03-03 19:30:05 -0800172
173 char type_token[16];
174 char width_token[16];
175 char height_token[16];
176 char maxval_token[16];
xNombred07bb0d2020-03-10 20:17:12 +0100177 volatile int color_type = 1;
178 unsigned long ul_width = 0, ul_alpha_width = 0;
179 unsigned long ul_height = 0, ul_alpha_height = 0;
180 unsigned long ul_maxval = 0;
181 volatile png_uint_32 width = 0, height = 0;
182 volatile png_uint_32 alpha_width = 0, alpha_height = 0;
The Android Open Source Project893912b2009-03-03 19:30:05 -0800183 png_uint_32 maxval;
xNombred07bb0d2020-03-10 20:17:12 +0100184 volatile int bit_depth = 0;
185 int channels = 0;
The Android Open Source Project893912b2009-03-03 19:30:05 -0800186 int alpha_depth = 0;
xNombred07bb0d2020-03-10 20:17:12 +0100187 int alpha_present = 0;
The Android Open Source Project893912b2009-03-03 19:30:05 -0800188 int row, col;
189 BOOL raw, alpha_raw = FALSE;
Chris Craikca2bf812013-07-29 15:28:30 -0700190#if defined(PNG_WRITE_INVERT_SUPPORTED) || defined(PNG_WRITE_PACK_SUPPORTED)
191 BOOL packed_bitmap = FALSE;
192#endif
The Android Open Source Project893912b2009-03-03 19:30:05 -0800193 png_uint_32 tmp16;
194 int i;
195
196 /* read header of PNM file */
197
xNombred07bb0d2020-03-10 20:17:12 +0100198 get_token (pnm_file, type_token, sizeof (type_token));
The Android Open Source Project893912b2009-03-03 19:30:05 -0800199 if (type_token[0] != 'P')
200 {
201 return FALSE;
202 }
203 else if ((type_token[1] == '1') || (type_token[1] == '4'))
204 {
Chris Craikca2bf812013-07-29 15:28:30 -0700205#if defined(PNG_WRITE_INVERT_SUPPORTED) || defined(PNG_WRITE_PACK_SUPPORTED)
The Android Open Source Project893912b2009-03-03 19:30:05 -0800206 raw = (type_token[1] == '4');
207 color_type = PNG_COLOR_TYPE_GRAY;
xNombred07bb0d2020-03-10 20:17:12 +0100208 get_token (pnm_file, width_token, sizeof (width_token));
Chris Craikca2bf812013-07-29 15:28:30 -0700209 sscanf (width_token, "%lu", &ul_width);
210 width = (png_uint_32) ul_width;
xNombred07bb0d2020-03-10 20:17:12 +0100211 get_token (pnm_file, height_token, sizeof (height_token));
Chris Craikca2bf812013-07-29 15:28:30 -0700212 sscanf (height_token, "%lu", &ul_height);
213 height = (png_uint_32) ul_height;
The Android Open Source Project893912b2009-03-03 19:30:05 -0800214 bit_depth = 1;
Chris Craikca2bf812013-07-29 15:28:30 -0700215 packed_bitmap = TRUE;
216#else
xNombred07bb0d2020-03-10 20:17:12 +0100217 fprintf (stderr, "PNM2PNG built without PNG_WRITE_INVERT_SUPPORTED and\n");
Chris Craikca2bf812013-07-29 15:28:30 -0700218 fprintf (stderr, "PNG_WRITE_PACK_SUPPORTED can't read PBM (P1,P4) files\n");
xNombred07bb0d2020-03-10 20:17:12 +0100219 return FALSE;
Chris Craikca2bf812013-07-29 15:28:30 -0700220#endif
The Android Open Source Project893912b2009-03-03 19:30:05 -0800221 }
222 else if ((type_token[1] == '2') || (type_token[1] == '5'))
223 {
224 raw = (type_token[1] == '5');
225 color_type = PNG_COLOR_TYPE_GRAY;
xNombred07bb0d2020-03-10 20:17:12 +0100226 get_token (pnm_file, width_token, sizeof (width_token));
Chris Craikca2bf812013-07-29 15:28:30 -0700227 sscanf (width_token, "%lu", &ul_width);
228 width = (png_uint_32) ul_width;
xNombred07bb0d2020-03-10 20:17:12 +0100229 get_token (pnm_file, height_token, sizeof (height_token));
Chris Craikca2bf812013-07-29 15:28:30 -0700230 sscanf (height_token, "%lu", &ul_height);
231 height = (png_uint_32) ul_height;
xNombred07bb0d2020-03-10 20:17:12 +0100232 get_token (pnm_file, maxval_token, sizeof (maxval_token));
Chris Craikca2bf812013-07-29 15:28:30 -0700233 sscanf (maxval_token, "%lu", &ul_maxval);
234 maxval = (png_uint_32) ul_maxval;
235
The Android Open Source Project893912b2009-03-03 19:30:05 -0800236 if (maxval <= 1)
237 bit_depth = 1;
238 else if (maxval <= 3)
239 bit_depth = 2;
240 else if (maxval <= 15)
241 bit_depth = 4;
242 else if (maxval <= 255)
243 bit_depth = 8;
xNombred07bb0d2020-03-10 20:17:12 +0100244 else if (maxval <= 65535U)
The Android Open Source Project893912b2009-03-03 19:30:05 -0800245 bit_depth = 16;
xNombred07bb0d2020-03-10 20:17:12 +0100246 else /* maxval > 65535U */
247 return FALSE;
The Android Open Source Project893912b2009-03-03 19:30:05 -0800248 }
249 else if ((type_token[1] == '3') || (type_token[1] == '6'))
250 {
251 raw = (type_token[1] == '6');
252 color_type = PNG_COLOR_TYPE_RGB;
xNombred07bb0d2020-03-10 20:17:12 +0100253 get_token (pnm_file, width_token, sizeof (width_token));
Chris Craikca2bf812013-07-29 15:28:30 -0700254 sscanf (width_token, "%lu", &ul_width);
255 width = (png_uint_32) ul_width;
xNombred07bb0d2020-03-10 20:17:12 +0100256 get_token (pnm_file, height_token, sizeof (height_token));
Chris Craikca2bf812013-07-29 15:28:30 -0700257 sscanf (height_token, "%lu", &ul_height);
258 height = (png_uint_32) ul_height;
xNombred07bb0d2020-03-10 20:17:12 +0100259 get_token (pnm_file, maxval_token, sizeof (maxval_token));
Chris Craikca2bf812013-07-29 15:28:30 -0700260 sscanf (maxval_token, "%lu", &ul_maxval);
261 maxval = (png_uint_32) ul_maxval;
The Android Open Source Project893912b2009-03-03 19:30:05 -0800262 if (maxval <= 1)
263 bit_depth = 1;
264 else if (maxval <= 3)
265 bit_depth = 2;
266 else if (maxval <= 15)
267 bit_depth = 4;
268 else if (maxval <= 255)
269 bit_depth = 8;
xNombred07bb0d2020-03-10 20:17:12 +0100270 else if (maxval <= 65535U)
The Android Open Source Project893912b2009-03-03 19:30:05 -0800271 bit_depth = 16;
xNombred07bb0d2020-03-10 20:17:12 +0100272 else /* maxval > 65535U */
273 return FALSE;
The Android Open Source Project893912b2009-03-03 19:30:05 -0800274 }
275 else
276 {
277 return FALSE;
278 }
279
280 /* read header of PGM file with alpha channel */
281
282 if (alpha)
283 {
284 if (color_type == PNG_COLOR_TYPE_GRAY)
285 color_type = PNG_COLOR_TYPE_GRAY_ALPHA;
286 if (color_type == PNG_COLOR_TYPE_RGB)
287 color_type = PNG_COLOR_TYPE_RGB_ALPHA;
288
xNombred07bb0d2020-03-10 20:17:12 +0100289 get_token (alpha_file, type_token, sizeof (type_token));
The Android Open Source Project893912b2009-03-03 19:30:05 -0800290 if (type_token[0] != 'P')
291 {
292 return FALSE;
293 }
294 else if ((type_token[1] == '2') || (type_token[1] == '5'))
295 {
296 alpha_raw = (type_token[1] == '5');
xNombred07bb0d2020-03-10 20:17:12 +0100297 get_token (alpha_file, width_token, sizeof (width_token));
Chris Craikca2bf812013-07-29 15:28:30 -0700298 sscanf (width_token, "%lu", &ul_alpha_width);
xNombred07bb0d2020-03-10 20:17:12 +0100299 alpha_width = (png_uint_32) ul_alpha_width;
The Android Open Source Project893912b2009-03-03 19:30:05 -0800300 if (alpha_width != width)
301 return FALSE;
xNombred07bb0d2020-03-10 20:17:12 +0100302 get_token (alpha_file, height_token, sizeof (height_token));
Chris Craikca2bf812013-07-29 15:28:30 -0700303 sscanf (height_token, "%lu", &ul_alpha_height);
304 alpha_height = (png_uint_32) ul_alpha_height;
The Android Open Source Project893912b2009-03-03 19:30:05 -0800305 if (alpha_height != height)
306 return FALSE;
xNombred07bb0d2020-03-10 20:17:12 +0100307 get_token (alpha_file, maxval_token, sizeof (maxval_token));
Chris Craikca2bf812013-07-29 15:28:30 -0700308 sscanf (maxval_token, "%lu", &ul_maxval);
309 maxval = (png_uint_32) ul_maxval;
The Android Open Source Project893912b2009-03-03 19:30:05 -0800310 if (maxval <= 1)
311 alpha_depth = 1;
312 else if (maxval <= 3)
313 alpha_depth = 2;
314 else if (maxval <= 15)
315 alpha_depth = 4;
316 else if (maxval <= 255)
317 alpha_depth = 8;
xNombred07bb0d2020-03-10 20:17:12 +0100318 else if (maxval <= 65535U)
The Android Open Source Project893912b2009-03-03 19:30:05 -0800319 alpha_depth = 16;
xNombred07bb0d2020-03-10 20:17:12 +0100320 else /* maxval > 65535U */
321 return FALSE;
The Android Open Source Project893912b2009-03-03 19:30:05 -0800322 if (alpha_depth != bit_depth)
323 return FALSE;
324 }
325 else
326 {
327 return FALSE;
328 }
329 } /* end if alpha */
330
331 /* calculate the number of channels and store alpha-presence */
332 if (color_type == PNG_COLOR_TYPE_GRAY)
333 channels = 1;
334 else if (color_type == PNG_COLOR_TYPE_GRAY_ALPHA)
335 channels = 2;
336 else if (color_type == PNG_COLOR_TYPE_RGB)
337 channels = 3;
338 else if (color_type == PNG_COLOR_TYPE_RGB_ALPHA)
339 channels = 4;
Matt Sarett9b1fe632015-11-25 10:21:17 -0500340#if 0
The Android Open Source Project893912b2009-03-03 19:30:05 -0800341 else
Matt Sarett9b1fe632015-11-25 10:21:17 -0500342 channels = 0; /* cannot happen */
343#endif
The Android Open Source Project893912b2009-03-03 19:30:05 -0800344
345 alpha_present = (channels - 1) % 2;
346
Chris Craikca2bf812013-07-29 15:28:30 -0700347#if defined(PNG_WRITE_INVERT_SUPPORTED) || defined(PNG_WRITE_PACK_SUPPORTED)
348 if (packed_bitmap)
xNombred07bb0d2020-03-10 20:17:12 +0100349 {
Chris Craikca2bf812013-07-29 15:28:30 -0700350 /* row data is as many bytes as can fit width x channels x bit_depth */
351 row_bytes = (width * channels * bit_depth + 7) / 8;
xNombred07bb0d2020-03-10 20:17:12 +0100352 }
Chris Craikca2bf812013-07-29 15:28:30 -0700353 else
354#endif
Leon Scroggins III3cc83ac2017-10-06 11:02:56 -0400355 {
xNombred07bb0d2020-03-10 20:17:12 +0100356 /* row_bytes is the width x number of channels x (bit-depth / 8) */
357 row_bytes = width * channels * ((bit_depth <= 8) ? 1 : 2);
358 }
359
360 if ((row_bytes == 0) ||
361 ((size_t) height > (size_t) (-1) / (size_t) row_bytes))
362 {
363 /* too big */
Leon Scroggins III3cc83ac2017-10-06 11:02:56 -0400364 return FALSE;
365 }
Matt Sarett9b1fe632015-11-25 10:21:17 -0500366 if ((png_pixels = (png_byte *)
xNombred07bb0d2020-03-10 20:17:12 +0100367 malloc ((size_t) row_bytes * (size_t) height)) == NULL)
368 {
369 /* out of memory */
The Android Open Source Project893912b2009-03-03 19:30:05 -0800370 return FALSE;
xNombred07bb0d2020-03-10 20:17:12 +0100371 }
The Android Open Source Project893912b2009-03-03 19:30:05 -0800372
373 /* read data from PNM file */
374 pix_ptr = png_pixels;
375
Matt Sarett9b1fe632015-11-25 10:21:17 -0500376 for (row = 0; row < (int) height; row++)
The Android Open Source Project893912b2009-03-03 19:30:05 -0800377 {
Chris Craikca2bf812013-07-29 15:28:30 -0700378#if defined(PNG_WRITE_INVERT_SUPPORTED) || defined(PNG_WRITE_PACK_SUPPORTED)
Leon Scroggins III3cc83ac2017-10-06 11:02:56 -0400379 if (packed_bitmap)
380 {
Matt Sarett9b1fe632015-11-25 10:21:17 -0500381 for (i = 0; i < (int) row_bytes; i++)
xNombred07bb0d2020-03-10 20:17:12 +0100382 {
Chris Craikca2bf812013-07-29 15:28:30 -0700383 /* png supports this format natively so no conversion is needed */
384 *pix_ptr++ = get_data (pnm_file, 8);
xNombred07bb0d2020-03-10 20:17:12 +0100385 }
386 }
387 else
Chris Craikca2bf812013-07-29 15:28:30 -0700388#endif
The Android Open Source Project893912b2009-03-03 19:30:05 -0800389 {
Matt Sarett9b1fe632015-11-25 10:21:17 -0500390 for (col = 0; col < (int) width; col++)
The Android Open Source Project893912b2009-03-03 19:30:05 -0800391 {
Chris Craikca2bf812013-07-29 15:28:30 -0700392 for (i = 0; i < (channels - alpha_present); i++)
393 {
394 if (raw)
xNombred07bb0d2020-03-10 20:17:12 +0100395 {
Chris Craikca2bf812013-07-29 15:28:30 -0700396 *pix_ptr++ = get_data (pnm_file, bit_depth);
xNombred07bb0d2020-03-10 20:17:12 +0100397 }
The Android Open Source Project893912b2009-03-03 19:30:05 -0800398 else
xNombred07bb0d2020-03-10 20:17:12 +0100399 {
Chris Craikca2bf812013-07-29 15:28:30 -0700400 if (bit_depth <= 8)
xNombred07bb0d2020-03-10 20:17:12 +0100401 {
Chris Craikca2bf812013-07-29 15:28:30 -0700402 *pix_ptr++ = get_value (pnm_file, bit_depth);
xNombred07bb0d2020-03-10 20:17:12 +0100403 }
Chris Craikca2bf812013-07-29 15:28:30 -0700404 else
405 {
406 tmp16 = get_value (pnm_file, bit_depth);
407 *pix_ptr = (png_byte) ((tmp16 >> 8) & 0xFF);
408 pix_ptr++;
409 *pix_ptr = (png_byte) (tmp16 & 0xFF);
410 pix_ptr++;
411 }
xNombred07bb0d2020-03-10 20:17:12 +0100412 }
Chris Craikca2bf812013-07-29 15:28:30 -0700413 }
The Android Open Source Project893912b2009-03-03 19:30:05 -0800414
Chris Craikca2bf812013-07-29 15:28:30 -0700415 if (alpha) /* read alpha-channel from pgm file */
416 {
417 if (alpha_raw)
xNombred07bb0d2020-03-10 20:17:12 +0100418 {
Chris Craikca2bf812013-07-29 15:28:30 -0700419 *pix_ptr++ = get_data (alpha_file, alpha_depth);
xNombred07bb0d2020-03-10 20:17:12 +0100420 }
The Android Open Source Project893912b2009-03-03 19:30:05 -0800421 else
xNombred07bb0d2020-03-10 20:17:12 +0100422 {
Chris Craikca2bf812013-07-29 15:28:30 -0700423 if (alpha_depth <= 8)
xNombred07bb0d2020-03-10 20:17:12 +0100424 {
Chris Craikca2bf812013-07-29 15:28:30 -0700425 *pix_ptr++ = get_value (alpha_file, bit_depth);
xNombred07bb0d2020-03-10 20:17:12 +0100426 }
Chris Craikca2bf812013-07-29 15:28:30 -0700427 else
428 {
429 tmp16 = get_value (alpha_file, bit_depth);
430 *pix_ptr++ = (png_byte) ((tmp16 >> 8) & 0xFF);
431 *pix_ptr++ = (png_byte) (tmp16 & 0xFF);
432 }
xNombred07bb0d2020-03-10 20:17:12 +0100433 }
434 } /* end if alpha */
435 } /* end if packed_bitmap */
The Android Open Source Project893912b2009-03-03 19:30:05 -0800436 } /* end for col */
437 } /* end for row */
438
439 /* prepare the standard PNG structures */
xNombred07bb0d2020-03-10 20:17:12 +0100440 png_ptr = png_create_write_struct (png_get_libpng_ver(NULL),
441 NULL, NULL, NULL);
The Android Open Source Project893912b2009-03-03 19:30:05 -0800442 if (!png_ptr)
443 {
Matt Sarett9b1fe632015-11-25 10:21:17 -0500444 free (png_pixels);
The Android Open Source Project893912b2009-03-03 19:30:05 -0800445 return FALSE;
446 }
447 info_ptr = png_create_info_struct (png_ptr);
448 if (!info_ptr)
449 {
xNombred07bb0d2020-03-10 20:17:12 +0100450 png_destroy_write_struct (&png_ptr, NULL);
Matt Sarett9b1fe632015-11-25 10:21:17 -0500451 free (png_pixels);
The Android Open Source Project893912b2009-03-03 19:30:05 -0800452 return FALSE;
453 }
454
Chris Craikca2bf812013-07-29 15:28:30 -0700455#if defined(PNG_WRITE_INVERT_SUPPORTED) || defined(PNG_WRITE_PACK_SUPPORTED)
456 if (packed_bitmap == TRUE)
457 {
458 png_set_packing (png_ptr);
459 png_set_invert_mono (png_ptr);
460 }
461#endif
462
xNombred07bb0d2020-03-10 20:17:12 +0100463 if (setjmp (png_jmpbuf (png_ptr)))
The Android Open Source Project893912b2009-03-03 19:30:05 -0800464 {
Matt Sarett9b1fe632015-11-25 10:21:17 -0500465 png_destroy_write_struct (&png_ptr, &info_ptr);
466 free (png_pixels);
The Android Open Source Project893912b2009-03-03 19:30:05 -0800467 return FALSE;
468 }
469
470 /* initialize the png structure */
471 png_init_io (png_ptr, png_file);
472
473 /* we're going to write more or less the same PNG as the input file */
474 png_set_IHDR (png_ptr, info_ptr, width, height, bit_depth, color_type,
xNombred07bb0d2020-03-10 20:17:12 +0100475 (!interlace) ? PNG_INTERLACE_NONE : PNG_INTERLACE_ADAM7,
476 PNG_COMPRESSION_TYPE_BASE, PNG_FILTER_TYPE_BASE);
The Android Open Source Project893912b2009-03-03 19:30:05 -0800477
478 /* write the file header information */
479 png_write_info (png_ptr, info_ptr);
480
481 /* if needed we will allocate memory for an new array of row-pointers */
xNombred07bb0d2020-03-10 20:17:12 +0100482 if (row_pointers == NULL)
The Android Open Source Project893912b2009-03-03 19:30:05 -0800483 {
Matt Sarett9b1fe632015-11-25 10:21:17 -0500484 if ((row_pointers = (png_byte **)
xNombred07bb0d2020-03-10 20:17:12 +0100485 malloc (height * sizeof (png_byte *))) == NULL)
The Android Open Source Project893912b2009-03-03 19:30:05 -0800486 {
Matt Sarett9b1fe632015-11-25 10:21:17 -0500487 png_destroy_write_struct (&png_ptr, &info_ptr);
488 free (png_pixels);
The Android Open Source Project893912b2009-03-03 19:30:05 -0800489 return FALSE;
490 }
491 }
492
493 /* set the individual row_pointers to point at the correct offsets */
Matt Sarett9b1fe632015-11-25 10:21:17 -0500494 for (i = 0; i < (int) height; i++)
The Android Open Source Project893912b2009-03-03 19:30:05 -0800495 row_pointers[i] = png_pixels + i * row_bytes;
496
497 /* write out the entire image data in one call */
498 png_write_image (png_ptr, row_pointers);
499
Matt Sarett9b1fe632015-11-25 10:21:17 -0500500 /* write the additional chunks to the PNG file (not really needed) */
The Android Open Source Project893912b2009-03-03 19:30:05 -0800501 png_write_end (png_ptr, info_ptr);
502
503 /* clean up after the write, and free any memory allocated */
Matt Sarett9b1fe632015-11-25 10:21:17 -0500504 png_destroy_write_struct (&png_ptr, &info_ptr);
The Android Open Source Project893912b2009-03-03 19:30:05 -0800505
xNombred07bb0d2020-03-10 20:17:12 +0100506 if (row_pointers != NULL)
The Android Open Source Project893912b2009-03-03 19:30:05 -0800507 free (row_pointers);
xNombred07bb0d2020-03-10 20:17:12 +0100508 if (png_pixels != NULL)
The Android Open Source Project893912b2009-03-03 19:30:05 -0800509 free (png_pixels);
510
511 return TRUE;
512} /* end of pnm2png */
513
514/*
xNombred07bb0d2020-03-10 20:17:12 +0100515 * get_token - gets the first string after whitespace
The Android Open Source Project893912b2009-03-03 19:30:05 -0800516 */
517
xNombred07bb0d2020-03-10 20:17:12 +0100518void get_token (FILE *pnm_file, char *token_buf, size_t token_buf_size)
The Android Open Source Project893912b2009-03-03 19:30:05 -0800519{
xNombred07bb0d2020-03-10 20:17:12 +0100520 size_t i = 0;
Chris Craikca2bf812013-07-29 15:28:30 -0700521 int ret;
The Android Open Source Project893912b2009-03-03 19:30:05 -0800522
Chris Craikca2bf812013-07-29 15:28:30 -0700523 /* remove white-space and comment lines */
The Android Open Source Project893912b2009-03-03 19:30:05 -0800524 do
525 {
xNombred07bb0d2020-03-10 20:17:12 +0100526 ret = fgetc (pnm_file);
Leon Scroggins III3cc83ac2017-10-06 11:02:56 -0400527 if (ret == '#')
528 {
Chris Craikca2bf812013-07-29 15:28:30 -0700529 /* the rest of this line is a comment */
530 do
531 {
xNombred07bb0d2020-03-10 20:17:12 +0100532 ret = fgetc (pnm_file);
Chris Craikca2bf812013-07-29 15:28:30 -0700533 }
534 while ((ret != '\n') && (ret != '\r') && (ret != EOF));
535 }
536 if (ret == EOF) break;
xNombred07bb0d2020-03-10 20:17:12 +0100537 token_buf[i] = (char) ret;
The Android Open Source Project893912b2009-03-03 19:30:05 -0800538 }
xNombred07bb0d2020-03-10 20:17:12 +0100539 while ((ret == '\n') || (ret == '\r') || (ret == ' '));
The Android Open Source Project893912b2009-03-03 19:30:05 -0800540
541 /* read string */
542 do
543 {
xNombred07bb0d2020-03-10 20:17:12 +0100544 ret = fgetc (pnm_file);
Chris Craikca2bf812013-07-29 15:28:30 -0700545 if (ret == EOF) break;
xNombred07bb0d2020-03-10 20:17:12 +0100546 if (++i == token_buf_size - 1) break;
547 token_buf[i] = (char) ret;
The Android Open Source Project893912b2009-03-03 19:30:05 -0800548 }
xNombred07bb0d2020-03-10 20:17:12 +0100549 while ((ret != '\n') && (ret != '\r') && (ret != ' '));
The Android Open Source Project893912b2009-03-03 19:30:05 -0800550
xNombred07bb0d2020-03-10 20:17:12 +0100551 token_buf[i] = '\0';
The Android Open Source Project893912b2009-03-03 19:30:05 -0800552
553 return;
554}
555
556/*
xNombred07bb0d2020-03-10 20:17:12 +0100557 * get_data - takes first byte and converts into next pixel value,
558 * taking as much bits as defined by bit-depth and
559 * using the bit-depth to fill up a byte (0Ah -> AAh)
The Android Open Source Project893912b2009-03-03 19:30:05 -0800560 */
561
562png_uint_32 get_data (FILE *pnm_file, int depth)
563{
564 static int bits_left = 0;
565 static int old_value = 0;
566 static int mask = 0;
567 int i;
568 png_uint_32 ret_value;
569
570 if (mask == 0)
571 for (i = 0; i < depth; i++)
572 mask = (mask >> 1) | 0x80;
573
574 if (bits_left <= 0)
575 {
576 old_value = fgetc (pnm_file);
577 bits_left = 8;
578 }
579
580 ret_value = old_value & mask;
581 for (i = 1; i < (8 / depth); i++)
582 ret_value = ret_value || (ret_value >> depth);
583
584 old_value = (old_value << depth) & 0xFF;
585 bits_left -= depth;
586
587 return ret_value;
588}
589
590/*
xNombred07bb0d2020-03-10 20:17:12 +0100591 * get_value - takes first (numeric) string and converts into number,
592 * using the bit-depth to fill up a byte (0Ah -> AAh)
The Android Open Source Project893912b2009-03-03 19:30:05 -0800593 */
594
595png_uint_32 get_value (FILE *pnm_file, int depth)
596{
597 static png_uint_32 mask = 0;
xNombred07bb0d2020-03-10 20:17:12 +0100598 char token[16];
Chris Craikca2bf812013-07-29 15:28:30 -0700599 unsigned long ul_ret_value;
The Android Open Source Project893912b2009-03-03 19:30:05 -0800600 png_uint_32 ret_value;
601 int i = 0;
602
603 if (mask == 0)
604 for (i = 0; i < depth; i++)
605 mask = (mask << 1) | 0x01;
606
xNombred07bb0d2020-03-10 20:17:12 +0100607 get_token (pnm_file, token, sizeof (token));
608 sscanf (token, "%lu", &ul_ret_value);
Chris Craikca2bf812013-07-29 15:28:30 -0700609 ret_value = (png_uint_32) ul_ret_value;
The Android Open Source Project893912b2009-03-03 19:30:05 -0800610
611 ret_value &= mask;
612
613 if (depth < 8)
614 for (i = 0; i < (8 / depth); i++)
615 ret_value = (ret_value << depth) || ret_value;
616
617 return ret_value;
618}
619
620/* end of source */