all repos — mgba @ 9de8f084ba55460b02d300c1dd8b8e6c56f691d5

mGBA Game Boy Advance Emulator

src/third-party/libpng/contrib/pngminus/png2pnm.c (view raw)

  1/*
  2 *  png2pnm.c --- conversion from PNG-file to PGM/PPM-file
  3 *  copyright (C) 1999,2017 by Willem van Schaik <willem at schaik.com>
  4 *
  5 *  version 1.0 - 1999.10.15 - First version.
  6 *          1.1 - 2017.04.22 - Add buffer-size check (Glenn Randers-Pehrson)
  7 *          1.2 - 2017.08.24 - Fix potential overflow in buffer-size check
  8 *                             (Glenn Randers-Pehrson)
  9 *          1.3 - 2017.08.28 - Add PNGMINUS_UNUSED (Christian Hesse)
 10 *
 11 *  Permission to use, copy, modify, and distribute this software and
 12 *  its documentation for any purpose and without fee is hereby granted,
 13 *  provided that the above copyright notice appear in all copies and
 14 *  that both that copyright notice and this permission notice appear in
 15 *  supporting documentation. This software is provided "as is" without
 16 *  express or implied warranty.
 17 */
 18
 19#include <stdio.h>
 20#include <stdlib.h>
 21#ifdef __TURBOC__
 22#include <mem.h>
 23#include <fcntl.h>
 24#endif
 25#include <zlib.h>
 26
 27#ifndef BOOL
 28#define BOOL unsigned char
 29#endif
 30#ifndef TRUE
 31#define TRUE (BOOL) 1
 32#endif
 33#ifndef FALSE
 34#define FALSE (BOOL) 0
 35#endif
 36
 37#ifdef __TURBOC__
 38#define STDIN  0
 39#define STDOUT 1
 40#define STDERR 2
 41#endif
 42
 43/* to make png2pnm verbose so we can find problems (needs to be before png.h) */
 44#ifndef PNG_DEBUG
 45#define PNG_DEBUG 0
 46#endif
 47
 48
 49#include "png.h"
 50
 51/* Define png_jmpbuf() in case we are using a pre-1.0.6 version of libpng */
 52#ifndef png_jmpbuf
 53#  define png_jmpbuf(png_ptr) ((png_ptr)->jmpbuf)
 54#endif
 55
 56#ifndef PNGMINUS_UNUSED
 57/* Unused formal parameter warnings are silenced using the following macro
 58 * which is expected to have no bad effects on performance (optimizing
 59 * compilers will probably remove it entirely).
 60 */
 61#  define PNGMINUS_UNUSED(param) (void)param
 62#endif
 63
 64/* function prototypes */
 65
 66int  main (int argc, char *argv[]);
 67void usage ();
 68BOOL png2pnm (FILE *png_file, FILE *pnm_file, FILE *alpha_file, BOOL raw,
 69   BOOL alpha);
 70
 71/*
 72 *  main
 73 */
 74
 75int main(int argc, char *argv[])
 76{
 77  FILE *fp_rd = stdin;
 78  FILE *fp_wr = stdout;
 79  FILE *fp_al = NULL;
 80  BOOL raw = TRUE;
 81  BOOL alpha = FALSE;
 82  int argi;
 83
 84  for (argi = 1; argi < argc; argi++)
 85  {
 86    if (argv[argi][0] == '-')
 87    {
 88      switch (argv[argi][1])
 89      {
 90        case 'n':
 91          raw = FALSE;
 92          break;
 93        case 'r':
 94          raw = TRUE;
 95          break;
 96        case 'a':
 97          alpha = TRUE;
 98          argi++;
 99          if ((fp_al = fopen (argv[argi], "wb")) == NULL)
100          {
101            fprintf (stderr, "PNM2PNG\n");
102            fprintf (stderr, "Error:  can not create alpha-channel file %s\n",
103               argv[argi]);
104            exit (1);
105          }
106          break;
107        case 'h':
108        case '?':
109          usage();
110          exit(0);
111          break;
112        default:
113          fprintf (stderr, "PNG2PNM\n");
114          fprintf (stderr, "Error:  unknown option %s\n", argv[argi]);
115          usage();
116          exit(1);
117          break;
118      } /* end switch */
119    }
120    else if (fp_rd == stdin)
121    {
122      if ((fp_rd = fopen (argv[argi], "rb")) == NULL)
123      {
124             fprintf (stderr, "PNG2PNM\n");
125            fprintf (stderr, "Error:  file %s does not exist\n", argv[argi]);
126            exit (1);
127      }
128    }
129    else if (fp_wr == stdout)
130    {
131      if ((fp_wr = fopen (argv[argi], "wb")) == NULL)
132      {
133        fprintf (stderr, "PNG2PNM\n");
134        fprintf (stderr, "Error:  can not create file %s\n", argv[argi]);
135        exit (1);
136      }
137    }
138    else
139    {
140      fprintf (stderr, "PNG2PNM\n");
141      fprintf (stderr, "Error:  too many parameters\n");
142      usage();
143      exit(1);
144    }
145  } /* end for */
146
147#ifdef __TURBOC__
148  /* set stdin/stdout if required to binary */
149  if (fp_rd == stdin)
150  {
151    setmode (STDIN, O_BINARY);
152  }
153  if ((raw) && (fp_wr == stdout))
154  {
155    setmode (STDOUT, O_BINARY);
156  }
157#endif
158
159  /* call the conversion program itself */
160  if (png2pnm (fp_rd, fp_wr, fp_al, raw, alpha) == FALSE)
161  {
162    fprintf (stderr, "PNG2PNM\n");
163    fprintf (stderr, "Error:  unsuccessful conversion of PNG-image\n");
164    exit(1);
165  }
166
167  /* close input file */
168  fclose (fp_rd);
169  /* close output file */
170  fclose (fp_wr);
171  /* close alpha file */
172  if (alpha)
173    fclose (fp_al);
174
175  return 0;
176}
177
178/*
179 *  usage
180 */
181
182void usage()
183{
184  fprintf (stderr, "PNG2PNM\n");
185  fprintf (stderr, "   by Willem van Schaik, 1999\n");
186#ifdef __TURBOC__
187  fprintf (stderr, "   for Turbo-C and Borland-C compilers\n");
188#else
189  fprintf (stderr, "   for Linux (and Unix) compilers\n");
190#endif
191  fprintf (stderr, "Usage:  png2pnm [options] <file>.png [<file>.pnm]\n");
192  fprintf (stderr, "   or:  ... | png2pnm [options]\n");
193  fprintf (stderr, "Options:\n");
194  fprintf (stderr,
195     "   -r[aw]   write pnm-file in binary format (P4/P5/P6) (default)\n");
196  fprintf (stderr, "   -n[oraw] write pnm-file in ascii format (P1/P2/P3)\n");
197  fprintf (stderr,
198     "   -a[lpha] <file>.pgm write PNG alpha channel as pgm-file\n");
199  fprintf (stderr, "   -h | -?  print this help-information\n");
200}
201
202/*
203 *  png2pnm
204 */
205
206BOOL png2pnm (FILE *png_file, FILE *pnm_file, FILE *alpha_file,
207    volatile BOOL raw, BOOL alpha)
208{
209  png_struct    *png_ptr = NULL;
210  png_info        *info_ptr = NULL;
211  png_byte      buf[8];
212  png_byte      *png_pixels = NULL;
213  png_byte      **row_pointers = NULL;
214  png_byte      *pix_ptr = NULL;
215  png_uint_32   row_bytes;
216
217  png_uint_32   width;
218  png_uint_32   height;
219  int           bit_depth;
220  int           channels;
221  int           color_type;
222  int           alpha_present;
223  int           row, col;
224  int           ret;
225  int           i;
226  long          dep_16;
227
228  /* read and check signature in PNG file */
229  ret = fread (buf, 1, 8, png_file);
230  if (ret != 8)
231    return FALSE;
232
233  ret = png_sig_cmp (buf, 0, 8);
234  if (ret)
235    return FALSE;
236
237  /* create png and info structures */
238
239  png_ptr = png_create_read_struct (png_get_libpng_ver(NULL),
240    NULL, NULL, NULL);
241  if (!png_ptr)
242    return FALSE;   /* out of memory */
243
244  info_ptr = png_create_info_struct (png_ptr);
245  if (!info_ptr)
246  {
247    png_destroy_read_struct (&png_ptr, NULL, NULL);
248    return FALSE;   /* out of memory */
249  }
250
251  if (setjmp (png_jmpbuf(png_ptr)))
252  {
253    png_destroy_read_struct (&png_ptr, &info_ptr, NULL);
254    return FALSE;
255  }
256
257  /* set up the input control for C streams */
258  png_init_io (png_ptr, png_file);
259  png_set_sig_bytes (png_ptr, 8);  /* we already read the 8 signature bytes */
260
261  /* read the file information */
262  png_read_info (png_ptr, info_ptr);
263
264  /* get size and bit-depth of the PNG-image */
265  png_get_IHDR (png_ptr, info_ptr,
266    &width, &height, &bit_depth, &color_type,
267    NULL, NULL, NULL);
268
269  /* set-up the transformations */
270
271  /* transform paletted images into full-color rgb */
272  if (color_type == PNG_COLOR_TYPE_PALETTE)
273    png_set_expand (png_ptr);
274  /* expand images to bit-depth 8 (only applicable for grayscale images) */
275  if (color_type == PNG_COLOR_TYPE_GRAY && bit_depth < 8)
276    png_set_expand (png_ptr);
277  /* transform transparency maps into full alpha-channel */
278  if (png_get_valid (png_ptr, info_ptr, PNG_INFO_tRNS))
279    png_set_expand (png_ptr);
280
281#ifdef NJET
282  /* downgrade 16-bit images to 8-bit */
283  if (bit_depth == 16)
284    png_set_strip_16 (png_ptr);
285  /* transform grayscale images into full-color */
286  if (color_type == PNG_COLOR_TYPE_GRAY ||
287    color_type == PNG_COLOR_TYPE_GRAY_ALPHA)
288    png_set_gray_to_rgb (png_ptr);
289  /* only if file has a file gamma, we do a correction */
290  if (png_get_gAMA (png_ptr, info_ptr, &file_gamma))
291    png_set_gamma (png_ptr, (double) 2.2, file_gamma);
292#endif
293
294  /* all transformations have been registered; now update info_ptr data,
295   * get rowbytes and channels, and allocate image memory */
296
297  png_read_update_info (png_ptr, info_ptr);
298
299  /* get the new color-type and bit-depth (after expansion/stripping) */
300  png_get_IHDR (png_ptr, info_ptr, &width, &height, &bit_depth, &color_type,
301    NULL, NULL, NULL);
302
303  /* check for 16-bit files */
304  if (bit_depth == 16)
305  {
306    raw = FALSE;
307#ifdef __TURBOC__
308    pnm_file->flags &= ~((unsigned) _F_BIN);
309#endif
310  }
311
312  /* calculate new number of channels and store alpha-presence */
313  if (color_type == PNG_COLOR_TYPE_GRAY)
314    channels = 1;
315  else if (color_type == PNG_COLOR_TYPE_GRAY_ALPHA)
316    channels = 2;
317  else if (color_type == PNG_COLOR_TYPE_RGB)
318    channels = 3;
319  else if (color_type == PNG_COLOR_TYPE_RGB_ALPHA)
320    channels = 4;
321  else
322    channels = 0; /* should never happen */
323  alpha_present = (channels - 1) % 2;
324
325  /* check if alpha is expected to be present in file */
326  if (alpha && !alpha_present)
327  {
328    fprintf (stderr, "PNG2PNM\n");
329    fprintf (stderr, "Error:  PNG-file doesn't contain alpha channel\n");
330    exit (1);
331  }
332
333  /* row_bytes is the width x number of channels x (bit-depth / 8) */
334  row_bytes = png_get_rowbytes (png_ptr, info_ptr);
335
336  if ((row_bytes == 0 || (size_t)height > ((size_t)(-1))/(size_t)row_bytes))
337  {
338    /* too big */ 
339    png_destroy_read_struct (&png_ptr, &info_ptr, NULL);
340    return FALSE;
341  }
342  if ((png_pixels = (png_byte *)
343     malloc ((size_t)row_bytes * (size_t)height * sizeof (png_byte))) == NULL)
344  {
345    png_destroy_read_struct (&png_ptr, &info_ptr, NULL);
346    return FALSE;
347  }
348
349  if ((row_pointers = (png_byte **)
350     malloc ((size_t)height * sizeof (png_bytep))) == NULL)
351  {
352    png_destroy_read_struct (&png_ptr, &info_ptr, NULL);
353    free (png_pixels);
354    png_pixels = NULL;
355    return FALSE;
356  }
357
358  /* set the individual row_pointers to point at the correct offsets */
359  for (i = 0; i < ((int) height); i++)
360    row_pointers[i] = png_pixels + i * row_bytes;
361
362  /* now we can go ahead and just read the whole image */
363  png_read_image (png_ptr, row_pointers);
364
365  /* read rest of file, and get additional chunks in info_ptr - REQUIRED */
366  png_read_end (png_ptr, info_ptr);
367
368  /* clean up after the read, and free any memory allocated - REQUIRED */
369  png_destroy_read_struct (&png_ptr, &info_ptr, (png_infopp) NULL);
370
371  /* write header of PNM file */
372
373  if ((color_type == PNG_COLOR_TYPE_GRAY) ||
374      (color_type == PNG_COLOR_TYPE_GRAY_ALPHA))
375  {
376    fprintf (pnm_file, "%s\n", (raw) ? "P5" : "P2");
377    fprintf (pnm_file, "%d %d\n", (int) width, (int) height);
378    fprintf (pnm_file, "%ld\n", ((1L << (int) bit_depth) - 1L));
379  }
380  else if ((color_type == PNG_COLOR_TYPE_RGB) ||
381           (color_type == PNG_COLOR_TYPE_RGB_ALPHA))
382  {
383    fprintf (pnm_file, "%s\n", (raw) ? "P6" : "P3");
384    fprintf (pnm_file, "%d %d\n", (int) width, (int) height);
385    fprintf (pnm_file, "%ld\n", ((1L << (int) bit_depth) - 1L));
386  }
387
388  /* write header of PGM file with alpha channel */
389
390  if ((alpha) &&
391      ((color_type == PNG_COLOR_TYPE_GRAY_ALPHA) ||
392       (color_type == PNG_COLOR_TYPE_RGB_ALPHA)))
393  {
394    fprintf (alpha_file, "%s\n", (raw) ? "P5" : "P2");
395    fprintf (alpha_file, "%d %d\n", (int) width, (int) height);
396    fprintf (alpha_file, "%ld\n", ((1L << (int) bit_depth) - 1L));
397  }
398
399  /* write data to PNM file */
400  pix_ptr = png_pixels;
401
402  for (row = 0; row < (int) height; row++)
403  {
404    for (col = 0; col < (int) width; col++)
405    {
406      for (i = 0; i < (channels - alpha_present); i++)
407      {
408        if (raw)
409          fputc ((int) *pix_ptr++ , pnm_file);
410        else
411          if (bit_depth == 16){
412            dep_16 = (long) *pix_ptr++;
413            fprintf (pnm_file, "%ld ", (dep_16 << 8) + ((long) *pix_ptr++));
414          }
415          else
416            fprintf (pnm_file, "%ld ", (long) *pix_ptr++);
417      }
418      if (alpha_present)
419      {
420        if (!alpha)
421        {
422          pix_ptr++; /* alpha */
423          if (bit_depth == 16)
424            pix_ptr++;
425        }
426        else /* output alpha-channel as pgm file */
427        {
428          if (raw)
429            fputc ((int) *pix_ptr++ , alpha_file);
430          else
431            if (bit_depth == 16)
432            {
433              dep_16 = (long) *pix_ptr++;
434              fprintf (alpha_file, "%ld ", (dep_16 << 8) + (long) *pix_ptr++);
435            }
436            else
437              fprintf (alpha_file, "%ld ", (long) *pix_ptr++);
438        }
439      } /* if alpha_present */
440
441      if (!raw)
442        if (col % 4 == 3)
443          fprintf (pnm_file, "\n");
444    } /* end for col */
445
446    if (!raw)
447      if (col % 4 != 0)
448        fprintf (pnm_file, "\n");
449  } /* end for row */
450
451  if (row_pointers != (unsigned char**) NULL)
452    free (row_pointers);
453  if (png_pixels != (unsigned char*) NULL)
454    free (png_pixels);
455
456  PNGMINUS_UNUSED(raw); /* to quiet a Coverity defect */
457  return TRUE;
458
459} /* end of source */
460