1 /*
2 * pnm2png.c --- conversion from PBM/PGM/PPM-file to PNG-file
3 * copyright (C) 1999-2019 by Willem van Schaik <willem at schaik dot com>
4 *
5 * This software is released under the MIT license. For conditions of
6 * distribution and use, see the LICENSE file part of this package.
7 */
8
9 #include <stdio.h>
10 #include <stdlib.h>
11 #include <fcntl.h>
12
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
23 /* make pnm2png verbose so we can find problems (needs to be before png.h) */
24 #ifndef PNG_DEBUG
25 #define PNG_DEBUG 0
26 #endif
27
28 #include "png.h"
29
30 /* function prototypes */
31
32 int main (int argc, char *argv[]);
33 void usage ();
34 BOOL pnm2png (FILE *pnm_file, FILE *png_file, FILE *alpha_file,
35 BOOL interlace, BOOL alpha);
36 void get_token (FILE *pnm_file, char *token_buf, size_t token_buf_size);
37 png_uint_32 get_data (FILE *pnm_file, int depth);
38 png_uint_32 get_value (FILE *pnm_file, int depth);
39
40 /*
41 * main
42 */
43
main(int argc,char * argv[])44 int main (int argc, char *argv[])
45 {
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",
69 argv[argi]);
70 exit (1);
71 }
72 break;
73 case 'h':
74 case '?':
75 usage ();
76 exit (0);
77 break;
78 default:
79 fprintf (stderr, "PNM2PNG\n");
80 fprintf (stderr, "Error: unknown option %s\n", argv[argi]);
81 usage ();
82 exit (1);
83 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");
100 fprintf (stderr, "Error: cannot create PNG-file %s\n", argv[argi]);
101 exit (1);
102 }
103 }
104 else
105 {
106 fprintf (stderr, "PNM2PNG\n");
107 fprintf (stderr, "Error: too many parameters\n");
108 usage ();
109 exit (1);
110 }
111 } /* end for */
112
113 #if defined(O_BINARY) && (O_BINARY != 0)
114 /* set stdin/stdout to binary,
115 * we're reading the PNM always! in binary format
116 */
117 if (fp_rd == stdin)
118 setmode (fileno (stdin), O_BINARY);
119 if (fp_wr == stdout)
120 setmode (fileno (stdout), O_BINARY);
121 #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
usage()146 void usage ()
147 {
148 fprintf (stderr, "PNM2PNG\n");
149 fprintf (stderr, " by Willem van Schaik, 1999\n");
150 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");
154 fprintf (stderr,
155 " -a[lpha] <file>.pgm read PNG alpha channel as pgm-file\n");
156 fprintf (stderr, " -h | -? print this help-information\n");
157 }
158
159 /*
160 * pnm2png
161 */
162
pnm2png(FILE * pnm_file,FILE * png_file,FILE * alpha_file,BOOL interlace,BOOL alpha)163 BOOL pnm2png (FILE *pnm_file, FILE *png_file, FILE *alpha_file,
164 BOOL interlace, BOOL alpha)
165 {
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;
171 volatile png_uint_32 row_bytes;
172
173 char type_token[16];
174 char width_token[16];
175 char height_token[16];
176 char maxval_token[16];
177 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;
183 png_uint_32 maxval;
184 volatile int bit_depth = 0;
185 int channels = 0;
186 int alpha_depth = 0;
187 int alpha_present = 0;
188 int row, col;
189 BOOL raw, alpha_raw = FALSE;
190 #if defined(PNG_WRITE_INVERT_SUPPORTED) || defined(PNG_WRITE_PACK_SUPPORTED)
191 BOOL packed_bitmap = FALSE;
192 #endif
193 png_uint_32 tmp16;
194 int i;
195
196 /* read header of PNM file */
197
198 get_token (pnm_file, type_token, sizeof (type_token));
199 if (type_token[0] != 'P')
200 {
201 return FALSE;
202 }
203 else if ((type_token[1] == '1') || (type_token[1] == '4'))
204 {
205 #if defined(PNG_WRITE_INVERT_SUPPORTED) || defined(PNG_WRITE_PACK_SUPPORTED)
206 raw = (type_token[1] == '4');
207 color_type = PNG_COLOR_TYPE_GRAY;
208 get_token (pnm_file, width_token, sizeof (width_token));
209 sscanf (width_token, "%lu", &ul_width);
210 width = (png_uint_32) ul_width;
211 get_token (pnm_file, height_token, sizeof (height_token));
212 sscanf (height_token, "%lu", &ul_height);
213 height = (png_uint_32) ul_height;
214 bit_depth = 1;
215 packed_bitmap = TRUE;
216 #else
217 fprintf (stderr, "PNM2PNG built without PNG_WRITE_INVERT_SUPPORTED and\n");
218 fprintf (stderr, "PNG_WRITE_PACK_SUPPORTED can't read PBM (P1,P4) files\n");
219 return FALSE;
220 #endif
221 }
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;
226 get_token (pnm_file, width_token, sizeof (width_token));
227 sscanf (width_token, "%lu", &ul_width);
228 width = (png_uint_32) ul_width;
229 get_token (pnm_file, height_token, sizeof (height_token));
230 sscanf (height_token, "%lu", &ul_height);
231 height = (png_uint_32) ul_height;
232 get_token (pnm_file, maxval_token, sizeof (maxval_token));
233 sscanf (maxval_token, "%lu", &ul_maxval);
234 maxval = (png_uint_32) ul_maxval;
235
236 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;
244 else if (maxval <= 65535U)
245 bit_depth = 16;
246 else /* maxval > 65535U */
247 return FALSE;
248 }
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;
253 get_token (pnm_file, width_token, sizeof (width_token));
254 sscanf (width_token, "%lu", &ul_width);
255 width = (png_uint_32) ul_width;
256 get_token (pnm_file, height_token, sizeof (height_token));
257 sscanf (height_token, "%lu", &ul_height);
258 height = (png_uint_32) ul_height;
259 get_token (pnm_file, maxval_token, sizeof (maxval_token));
260 sscanf (maxval_token, "%lu", &ul_maxval);
261 maxval = (png_uint_32) ul_maxval;
262 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;
270 else if (maxval <= 65535U)
271 bit_depth = 16;
272 else /* maxval > 65535U */
273 return FALSE;
274 }
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
289 get_token (alpha_file, type_token, sizeof (type_token));
290 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');
297 get_token (alpha_file, width_token, sizeof (width_token));
298 sscanf (width_token, "%lu", &ul_alpha_width);
299 alpha_width = (png_uint_32) ul_alpha_width;
300 if (alpha_width != width)
301 return FALSE;
302 get_token (alpha_file, height_token, sizeof (height_token));
303 sscanf (height_token, "%lu", &ul_alpha_height);
304 alpha_height = (png_uint_32) ul_alpha_height;
305 if (alpha_height != height)
306 return FALSE;
307 get_token (alpha_file, maxval_token, sizeof (maxval_token));
308 sscanf (maxval_token, "%lu", &ul_maxval);
309 maxval = (png_uint_32) ul_maxval;
310 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;
318 else if (maxval <= 65535U)
319 alpha_depth = 16;
320 else /* maxval > 65535U */
321 return FALSE;
322 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;
340 #if 0
341 else
342 channels = 0; /* cannot happen */
343 #endif
344
345 alpha_present = (channels - 1) % 2;
346
347 #if defined(PNG_WRITE_INVERT_SUPPORTED) || defined(PNG_WRITE_PACK_SUPPORTED)
348 if (packed_bitmap)
349 {
350 /* row data is as many bytes as can fit width x channels x bit_depth */
351 row_bytes = (width * channels * bit_depth + 7) / 8;
352 }
353 else
354 #endif
355 {
356 /* 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 */
364 return FALSE;
365 }
366 if ((png_pixels = (png_byte *)
367 malloc ((size_t) row_bytes * (size_t) height)) == NULL)
368 {
369 /* out of memory */
370 return FALSE;
371 }
372
373 /* read data from PNM file */
374 pix_ptr = png_pixels;
375
376 for (row = 0; row < (int) height; row++)
377 {
378 #if defined(PNG_WRITE_INVERT_SUPPORTED) || defined(PNG_WRITE_PACK_SUPPORTED)
379 if (packed_bitmap)
380 {
381 for (i = 0; i < (int) row_bytes; i++)
382 {
383 /* png supports this format natively so no conversion is needed */
384 *pix_ptr++ = get_data (pnm_file, 8);
385 }
386 }
387 else
388 #endif
389 {
390 for (col = 0; col < (int) width; col++)
391 {
392 for (i = 0; i < (channels - alpha_present); i++)
393 {
394 if (raw)
395 {
396 *pix_ptr++ = get_data (pnm_file, bit_depth);
397 }
398 else
399 {
400 if (bit_depth <= 8)
401 {
402 *pix_ptr++ = get_value (pnm_file, bit_depth);
403 }
404 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 }
412 }
413 }
414
415 if (alpha) /* read alpha-channel from pgm file */
416 {
417 if (alpha_raw)
418 {
419 *pix_ptr++ = get_data (alpha_file, alpha_depth);
420 }
421 else
422 {
423 if (alpha_depth <= 8)
424 {
425 *pix_ptr++ = get_value (alpha_file, bit_depth);
426 }
427 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 }
433 }
434 } /* end if alpha */
435 } /* end if packed_bitmap */
436 } /* end for col */
437 } /* end for row */
438
439 /* prepare the standard PNG structures */
440 png_ptr = png_create_write_struct (png_get_libpng_ver(NULL),
441 NULL, NULL, NULL);
442 if (!png_ptr)
443 {
444 free (png_pixels);
445 return FALSE;
446 }
447 info_ptr = png_create_info_struct (png_ptr);
448 if (!info_ptr)
449 {
450 png_destroy_write_struct (&png_ptr, NULL);
451 free (png_pixels);
452 return FALSE;
453 }
454
455 #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
463 if (setjmp (png_jmpbuf (png_ptr)))
464 {
465 png_destroy_write_struct (&png_ptr, &info_ptr);
466 free (png_pixels);
467 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,
475 (!interlace) ? PNG_INTERLACE_NONE : PNG_INTERLACE_ADAM7,
476 PNG_COMPRESSION_TYPE_BASE, PNG_FILTER_TYPE_BASE);
477
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 */
482 if (row_pointers == NULL)
483 {
484 if ((row_pointers = (png_byte **)
485 malloc (height * sizeof (png_byte *))) == NULL)
486 {
487 png_destroy_write_struct (&png_ptr, &info_ptr);
488 free (png_pixels);
489 return FALSE;
490 }
491 }
492
493 /* set the individual row_pointers to point at the correct offsets */
494 for (i = 0; i < (int) height; i++)
495 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
500 /* write the additional chunks to the PNG file (not really needed) */
501 png_write_end (png_ptr, info_ptr);
502
503 /* clean up after the write, and free any memory allocated */
504 png_destroy_write_struct (&png_ptr, &info_ptr);
505
506 if (row_pointers != NULL)
507 free (row_pointers);
508 if (png_pixels != NULL)
509 free (png_pixels);
510
511 return TRUE;
512 } /* end of pnm2png */
513
514 /*
515 * get_token - gets the first string after whitespace
516 */
517
get_token(FILE * pnm_file,char * token_buf,size_t token_buf_size)518 void get_token (FILE *pnm_file, char *token_buf, size_t token_buf_size)
519 {
520 size_t i = 0;
521 int ret;
522
523 /* remove white-space and comment lines */
524 do
525 {
526 ret = fgetc (pnm_file);
527 if (ret == '#')
528 {
529 /* the rest of this line is a comment */
530 do
531 {
532 ret = fgetc (pnm_file);
533 }
534 while ((ret != '\n') && (ret != '\r') && (ret != EOF));
535 }
536 if (ret == EOF) break;
537 token_buf[i] = (char) ret;
538 }
539 while ((ret == '\n') || (ret == '\r') || (ret == ' '));
540
541 /* read string */
542 do
543 {
544 ret = fgetc (pnm_file);
545 if (ret == EOF) break;
546 if (++i == token_buf_size - 1) break;
547 token_buf[i] = (char) ret;
548 }
549 while ((ret != '\n') && (ret != '\r') && (ret != ' '));
550
551 token_buf[i] = '\0';
552
553 return;
554 }
555
556 /*
557 * 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)
560 */
561
get_data(FILE * pnm_file,int depth)562 png_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 /*
591 * get_value - takes first (numeric) string and converts into number,
592 * using the bit-depth to fill up a byte (0Ah -> AAh)
593 */
594
get_value(FILE * pnm_file,int depth)595 png_uint_32 get_value (FILE *pnm_file, int depth)
596 {
597 static png_uint_32 mask = 0;
598 char token[16];
599 unsigned long ul_ret_value;
600 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
607 get_token (pnm_file, token, sizeof (token));
608 sscanf (token, "%lu", &ul_ret_value);
609 ret_value = (png_uint_32) ul_ret_value;
610
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 */
621