mirror of
				https://git.code.sf.net/p/libpng/code.git
				synced 2025-07-10 18:04:09 +02:00 
			
		
		
		
	
		
			
				
	
	
		
			866 lines
		
	
	
		
			31 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			866 lines
		
	
	
		
			31 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*---------------------------------------------------------------------------
 | |
| 
 | |
|    wpng - simple PNG-writing program                                 wpng.c
 | |
| 
 | |
|    This program converts certain NetPBM binary files (grayscale and RGB,
 | |
|    maxval = 255) to PNG.  Non-interlaced PNGs are written progressively;
 | |
|    interlaced PNGs are read and written in one memory-intensive blast.
 | |
| 
 | |
|    Thanks to Jean-loup Gailly for providing the necessary trick to read
 | |
|    interactive text from the keyboard while stdin is redirected.  Thanks
 | |
|    to Cosmin Truta for Cygwin fixes.
 | |
| 
 | |
|    NOTE:  includes provisional support for PNM type "8" (portable alphamap)
 | |
|           images, presumed to be a 32-bit interleaved RGBA format; no pro-
 | |
|           vision for possible interleaved grayscale+alpha (16-bit) format.
 | |
|           THIS IS UNLIKELY TO BECOME AN OFFICIAL NETPBM ALPHA FORMAT!
 | |
| 
 | |
|    to do:
 | |
|     - delete output file if quit before calling any writepng routines
 | |
|     - process backspace with -text option under DOS/Win? (currently get ^H)
 | |
| 
 | |
|   ---------------------------------------------------------------------------
 | |
| 
 | |
|    Changelog:
 | |
|     - 1.01:  initial public release
 | |
|     - 1.02:  modified to allow abbreviated options
 | |
|     - 1.03:  removed extraneous character from usage screen; fixed bug in
 | |
|               command-line parsing
 | |
|     - 1.04:  fixed DOS/OS2/Win32 detection, including partial Cygwin fix
 | |
|               (see http://home.att.net/~perlspinr/diffs/GregBook_cygwin.diff)
 | |
|     - 2.00:  dual-licensed (added GNU GPL)
 | |
|     - 2.01:  check for integer overflow (Glenn R-P)
 | |
| 
 | |
|         [REPORTED BUG (win32 only):  "contrib/gregbook/wpng.c - cmd line
 | |
|          dose not work!  In order to do something useful I needed to redirect
 | |
|          both input and output, with cygwin and with bcc32 as well.  Under
 | |
|          Linux, the same wpng appears to work fine.  I don't know what is
 | |
|          the problem."]
 | |
| 
 | |
|   ---------------------------------------------------------------------------
 | |
| 
 | |
|       Copyright (c) 1998-2007, 2017 Greg Roelofs.  All rights reserved.
 | |
| 
 | |
|       This software is provided "as is," without warranty of any kind,
 | |
|       express or implied.  In no event shall the author or contributors
 | |
|       be held liable for any damages arising in any way from the use of
 | |
|       this software.
 | |
| 
 | |
|       The contents of this file are DUAL-LICENSED.  You may modify and/or
 | |
|       redistribute this software according to the terms of one of the
 | |
|       following two licenses (at your option):
 | |
| 
 | |
| 
 | |
|       LICENSE 1 ("BSD-like with advertising clause"):
 | |
| 
 | |
|       Permission is granted to anyone to use this software for any purpose,
 | |
|       including commercial applications, and to alter it and redistribute
 | |
|       it freely, subject to the following restrictions:
 | |
| 
 | |
|       1. Redistributions of source code must retain the above copyright
 | |
|          notice, disclaimer, and this list of conditions.
 | |
|       2. Redistributions in binary form must reproduce the above copyright
 | |
|          notice, disclaimer, and this list of conditions in the documenta-
 | |
|          tion and/or other materials provided with the distribution.
 | |
|       3. All advertising materials mentioning features or use of this
 | |
|          software must display the following acknowledgment:
 | |
| 
 | |
|             This product includes software developed by Greg Roelofs
 | |
|             and contributors for the book, "PNG: The Definitive Guide,"
 | |
|             published by O'Reilly and Associates.
 | |
| 
 | |
| 
 | |
|       LICENSE 2 (GNU GPL v2 or later):
 | |
| 
 | |
|       This program is free software; you can redistribute it and/or modify
 | |
|       it under the terms of the GNU General Public License as published by
 | |
|       the Free Software Foundation; either version 2 of the License, or
 | |
|       (at your option) any later version.
 | |
| 
 | |
|       This program is distributed in the hope that it will be useful,
 | |
|       but WITHOUT ANY WARRANTY; without even the implied warranty of
 | |
|       MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 | |
|       GNU General Public License for more details.
 | |
| 
 | |
|       You should have received a copy of the GNU General Public License
 | |
|       along with this program; if not, write to the Free Software Foundation,
 | |
|       Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
 | |
| 
 | |
|   ---------------------------------------------------------------------------*/
 | |
| 
 | |
| #define PROGNAME  "wpng"
 | |
| #define VERSION   "2.00 of 2 June 2007"
 | |
| #define APPNAME   "Simple PGM/PPM/PAM to PNG Converter"
 | |
| 
 | |
| #if defined(__MSDOS__) || defined(__OS2__)
 | |
| #  define DOS_OS2_W32
 | |
| #elif defined(WIN32) || defined(_WIN32) || defined(__WIN32__)
 | |
| #  ifndef __GNUC__   /* treat Win32 native ports of gcc as Unix environments */
 | |
| #    define DOS_OS2_W32
 | |
| #  endif
 | |
| #endif
 | |
| 
 | |
| #include <stdio.h>
 | |
| #include <stdlib.h>
 | |
| #include <string.h>
 | |
| #include <setjmp.h>     /* for jmpbuf declaration in writepng.h */
 | |
| #include <time.h>
 | |
| 
 | |
| #ifdef DOS_OS2_W32
 | |
| #  include <io.h>       /* for isatty(), setmode() prototypes */
 | |
| #  include <fcntl.h>    /* O_BINARY for fdopen() without text translation */
 | |
| #  ifdef __EMX__
 | |
| #    ifndef getch
 | |
| #      define getch() _read_kbd(0, 1, 0)    /* need getche() */
 | |
| #    endif
 | |
| #  else /* !__EMX__ */
 | |
| #    ifdef __GO32__
 | |
| #      include <pc.h>
 | |
| #      define getch() getkey()  /* GRR:  need getche() */
 | |
| #    else
 | |
| #      include <conio.h>        /* for getche() console input */
 | |
| #    endif
 | |
| #  endif /* ?__EMX__ */
 | |
| #  define FGETS(buf,len,stream)  dos_kbd_gets(buf,len)
 | |
| #else
 | |
| #  include <unistd.h>           /* for isatty() prototype */
 | |
| #  define FGETS fgets
 | |
| #endif
 | |
| 
 | |
| /* #define DEBUG  :  this enables the Trace() macros */
 | |
| 
 | |
| /* #define FORBID_LATIN1_CTRL  :  this requires the user to re-enter any
 | |
|    text that includes control characters discouraged by the PNG spec; text
 | |
|    that includes an escape character (27) must be re-entered regardless */
 | |
| 
 | |
| #include "writepng.h"   /* typedefs, common macros, writepng prototypes */
 | |
| 
 | |
| 
 | |
| 
 | |
| /* local prototypes */
 | |
| 
 | |
| static int  wpng_isvalid_latin1(uch *p, int len);
 | |
| static void wpng_cleanup(void);
 | |
| 
 | |
| #ifdef DOS_OS2_W32
 | |
|    static char *dos_kbd_gets(char *buf, int len);
 | |
| #endif
 | |
| 
 | |
| 
 | |
| 
 | |
| static mainprog_info wpng_info;   /* lone global */
 | |
| 
 | |
| 
 | |
| 
 | |
| int main(int argc, char **argv)
 | |
| {
 | |
| #ifndef DOS_OS2_W32
 | |
|     FILE *keybd;
 | |
| #endif
 | |
| #ifdef sgi
 | |
|     FILE *tmpfile;      /* or we could just use keybd, since no overlap */
 | |
|     char tmpline[80];
 | |
| #endif
 | |
|     char *inname = NULL, outname[256];
 | |
|     char *p, pnmchar, pnmline[256];
 | |
|     char *bgstr, *textbuf = NULL;
 | |
|     ulg rowbytes;
 | |
|     int rc, len = 0;
 | |
|     int error = 0;
 | |
|     int text = FALSE;
 | |
|     int maxval;
 | |
|     double LUT_exponent;                /* just the lookup table */
 | |
|     double CRT_exponent = 2.2;          /* just the monitor */
 | |
|     double default_display_exponent;    /* whole display system */
 | |
|     double default_gamma = 0.0;
 | |
| 
 | |
| 
 | |
|     wpng_info.infile = NULL;
 | |
|     wpng_info.outfile = NULL;
 | |
|     wpng_info.image_data = NULL;
 | |
|     wpng_info.row_pointers = NULL;
 | |
|     wpng_info.filter = FALSE;
 | |
|     wpng_info.interlaced = FALSE;
 | |
|     wpng_info.have_bg = FALSE;
 | |
|     wpng_info.have_time = FALSE;
 | |
|     wpng_info.have_text = 0;
 | |
|     wpng_info.gamma = 0.0;
 | |
| 
 | |
| 
 | |
|     /* First get the default value for our display-system exponent, i.e.,
 | |
|      * the product of the CRT exponent and the exponent corresponding to
 | |
|      * the frame-buffer's lookup table (LUT), if any.  If the PNM image
 | |
|      * looks correct on the user's display system, its file gamma is the
 | |
|      * inverse of this value.  (Note that this is not an exhaustive list
 | |
|      * of LUT values--e.g., OpenStep has a lot of weird ones--but it should
 | |
|      * cover 99% of the current possibilities.  This section must ensure
 | |
|      * that default_display_exponent is positive.) */
 | |
| 
 | |
| #if defined(NeXT)
 | |
|     /* third-party utilities can modify the default LUT exponent */
 | |
|     LUT_exponent = 1.0 / 2.2;
 | |
|     /*
 | |
|     if (some_next_function_that_returns_gamma(&next_gamma))
 | |
|         LUT_exponent = 1.0 / next_gamma;
 | |
|      */
 | |
| #elif defined(sgi)
 | |
|     LUT_exponent = 1.0 / 1.7;
 | |
|     /* there doesn't seem to be any documented function to
 | |
|      * get the "gamma" value, so we do it the hard way */
 | |
|     tmpfile = fopen("/etc/config/system.glGammaVal", "r");
 | |
|     if (tmpfile) {
 | |
|         double sgi_gamma;
 | |
| 
 | |
|         fgets(tmpline, 80, tmpfile);
 | |
|         fclose(tmpfile);
 | |
|         sgi_gamma = atof(tmpline);
 | |
|         if (sgi_gamma > 0.0)
 | |
|             LUT_exponent = 1.0 / sgi_gamma;
 | |
|     }
 | |
| #elif defined(Macintosh)
 | |
|     LUT_exponent = 1.8 / 2.61;
 | |
|     /*
 | |
|     if (some_mac_function_that_returns_gamma(&mac_gamma))
 | |
|         LUT_exponent = mac_gamma / 2.61;
 | |
|      */
 | |
| #else
 | |
|     LUT_exponent = 1.0;   /* assume no LUT:  most PCs */
 | |
| #endif
 | |
| 
 | |
|     /* the defaults above give 1.0, 1.3, 1.5 and 2.2, respectively: */
 | |
|     default_display_exponent = LUT_exponent * CRT_exponent;
 | |
| 
 | |
| 
 | |
|     /* If the user has set the SCREEN_GAMMA environment variable as suggested
 | |
|      * (somewhat imprecisely) in the libpng documentation, use that; otherwise
 | |
|      * use the default value we just calculated.  Either way, the user may
 | |
|      * override this via a command-line option. */
 | |
| 
 | |
|     if ((p = getenv("SCREEN_GAMMA")) != NULL) {
 | |
|         double exponent = atof(p);
 | |
| 
 | |
|         if (exponent > 0.0)
 | |
|             default_gamma = 1.0 / exponent;
 | |
|     }
 | |
| 
 | |
|     if (default_gamma == 0.0)
 | |
|         default_gamma = 1.0 / default_display_exponent;
 | |
| 
 | |
| 
 | |
|     /* Now parse the command line for options and the PNM filename. */
 | |
| 
 | |
|     while (*++argv && !error) {
 | |
|         if (!strncmp(*argv, "-i", 2)) {
 | |
|             wpng_info.interlaced = TRUE;
 | |
|         } else if (!strncmp(*argv, "-time", 3)) {
 | |
|             wpng_info.modtime = time(NULL);
 | |
|             wpng_info.have_time = TRUE;
 | |
|         } else if (!strncmp(*argv, "-text", 3)) {
 | |
|             text = TRUE;
 | |
|         } else if (!strncmp(*argv, "-gamma", 2)) {
 | |
|             if (!*++argv)
 | |
|                 ++error;
 | |
|             else {
 | |
|                 wpng_info.gamma = atof(*argv);
 | |
|                 if (wpng_info.gamma <= 0.0)
 | |
|                     ++error;
 | |
|                 else if (wpng_info.gamma > 1.01)
 | |
|                     fprintf(stderr, PROGNAME
 | |
|                       " warning:  file gammas are usually less than 1.0\n");
 | |
|             }
 | |
|         } else if (!strncmp(*argv, "-bgcolor", 4)) {
 | |
|             if (!*++argv)
 | |
|                 ++error;
 | |
|             else {
 | |
|                 bgstr = *argv;
 | |
|                 if (strlen(bgstr) != 7 || bgstr[0] != '#')
 | |
|                     ++error;
 | |
|                 else {
 | |
|                     unsigned r, g, b;  /* this way quiets compiler warnings */
 | |
| 
 | |
|                     sscanf(bgstr+1, "%2x%2x%2x", &r, &g, &b);
 | |
|                     wpng_info.bg_red   = (uch)r;
 | |
|                     wpng_info.bg_green = (uch)g;
 | |
|                     wpng_info.bg_blue  = (uch)b;
 | |
|                     wpng_info.have_bg = TRUE;
 | |
|                 }
 | |
|             }
 | |
|         } else {
 | |
|             if (**argv != '-') {
 | |
|                 inname = *argv;
 | |
|                 if (argv[1])   /* shouldn't be any more args after filename */
 | |
|                     ++error;
 | |
|             } else
 | |
|                 ++error;   /* not expecting any other options */
 | |
|         }
 | |
|     }
 | |
| 
 | |
| 
 | |
|     /* open the input and output files, or register an error and abort */
 | |
| 
 | |
|     if (!inname) {
 | |
|         if (isatty(0)) {
 | |
|             fprintf(stderr, PROGNAME
 | |
|               ":  must give input filename or provide image data via stdin\n");
 | |
|             ++error;
 | |
|         } else {
 | |
| #ifdef DOS_OS2_W32
 | |
|             /* some buggy C libraries require BOTH setmode() and fdopen(bin) */
 | |
|             setmode(fileno(stdin), O_BINARY);
 | |
|             setmode(fileno(stdout), O_BINARY);
 | |
| #endif
 | |
|             if ((wpng_info.infile = fdopen(fileno(stdin), "rb")) == NULL) {
 | |
|                 fprintf(stderr, PROGNAME
 | |
|                   ":  unable to reopen stdin in binary mode\n");
 | |
|                 ++error;
 | |
|             } else
 | |
|             if ((wpng_info.outfile = fdopen(fileno(stdout), "wb")) == NULL) {
 | |
|                 fprintf(stderr, PROGNAME
 | |
|                   ":  unable to reopen stdout in binary mode\n");
 | |
|                 fclose(wpng_info.infile);
 | |
|                 ++error;
 | |
|             } else
 | |
|                 wpng_info.filter = TRUE;
 | |
|         }
 | |
|     } else if ((len = strlen(inname)) > 250) {
 | |
|         fprintf(stderr, PROGNAME ":  input filename is too long [%d chars]\n",
 | |
|           len);
 | |
|         ++error;
 | |
|     } else if (!(wpng_info.infile = fopen(inname, "rb"))) {
 | |
|         fprintf(stderr, PROGNAME ":  can't open input file [%s]\n", inname);
 | |
|         ++error;
 | |
|     }
 | |
| 
 | |
|     if (!error) {
 | |
|         fgets(pnmline, 256, wpng_info.infile);
 | |
|         if (pnmline[0] != 'P' || ((pnmchar = pnmline[1]) != '5' &&
 | |
|             pnmchar != '6' && pnmchar != '8'))
 | |
|         {
 | |
|             fprintf(stderr, PROGNAME
 | |
|               ":  input file [%s] is not a binary PGM, PPM or PAM file\n",
 | |
|               inname);
 | |
|             ++error;
 | |
|         } else {
 | |
|             wpng_info.pnmtype = (int)(pnmchar - '0');
 | |
|             if (wpng_info.pnmtype != 8)
 | |
|                 wpng_info.have_bg = FALSE;  /* no need for bg if opaque */
 | |
|             do {
 | |
|                 fgets(pnmline, 256, wpng_info.infile);  /* lose any comments */
 | |
|             } while (pnmline[0] == '#');
 | |
|             sscanf(pnmline, "%ld %ld", &wpng_info.width, &wpng_info.height);
 | |
|             do {
 | |
|                 fgets(pnmline, 256, wpng_info.infile);  /* more comment lines */
 | |
|             } while (pnmline[0] == '#');
 | |
|             sscanf(pnmline, "%d", &maxval);
 | |
|             if (wpng_info.width <= 0L || wpng_info.height <= 0L ||
 | |
|                 maxval != 255)
 | |
|             {
 | |
|                 fprintf(stderr, PROGNAME
 | |
|                   ":  only positive width/height, maxval == 255 allowed \n");
 | |
|                 ++error;
 | |
|             }
 | |
|             wpng_info.sample_depth = 8;  /* <==> maxval 255 */
 | |
| 
 | |
|             if (!wpng_info.filter) {
 | |
|                 /* make outname from inname */
 | |
|                 if ((p = strrchr(inname, '.')) == NULL ||
 | |
|                     (p - inname) != (len - 4))
 | |
|                 {
 | |
|                     strcpy(outname, inname);
 | |
|                     strcpy(outname+len, ".png");
 | |
|                 } else {
 | |
|                     len -= 4;
 | |
|                     strncpy(outname, inname, len);
 | |
|                     strcpy(outname+len, ".png");
 | |
|                 }
 | |
|                 /* check if outname already exists; if not, open */
 | |
|                 if ((wpng_info.outfile = fopen(outname, "rb")) != NULL) {
 | |
|                     fprintf(stderr, PROGNAME ":  output file exists [%s]\n",
 | |
|                       outname);
 | |
|                     fclose(wpng_info.outfile);
 | |
|                     ++error;
 | |
|                 } else if (!(wpng_info.outfile = fopen(outname, "wb"))) {
 | |
|                     fprintf(stderr, PROGNAME ":  can't open output file [%s]\n",
 | |
|                       outname);
 | |
|                     ++error;
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
|         if (error) {
 | |
|             fclose(wpng_info.infile);
 | |
|             wpng_info.infile = NULL;
 | |
|             if (wpng_info.filter) {
 | |
|                 fclose(wpng_info.outfile);
 | |
|                 wpng_info.outfile = NULL;
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| 
 | |
| 
 | |
|     /* if we had any errors, print usage and die horrible death...arrr! */
 | |
| 
 | |
|     if (error) {
 | |
|         fprintf(stderr, "\n%s %s:  %s\n", PROGNAME, VERSION, APPNAME);
 | |
|         writepng_version_info();
 | |
|         fprintf(stderr, "\n"
 | |
| "Usage:  %s [-gamma exp] [-bgcolor bg] [-text] [-time] [-interlace] pnmfile\n"
 | |
| "or: ... | %s [-gamma exp] [-bgcolor bg] [-text] [-time] [-interlace] | ...\n"
 | |
|          "    exp \ttransfer-function exponent (``gamma'') of the image in\n"
 | |
|          "\t\t  floating-point format (e.g., ``%.5f''); if image looks\n"
 | |
|          "\t\t  correct on given display system, image gamma is equal to\n"
 | |
|          "\t\t  inverse of display-system exponent, i.e., 1 / (LUT * CRT)\n"
 | |
|          "\t\t  (where LUT = lookup-table exponent and CRT = CRT exponent;\n"
 | |
|          "\t\t  first varies, second is usually 2.2, all are positive)\n"
 | |
|          "    bg  \tdesired background color for alpha-channel images, in\n"
 | |
|          "\t\t  7-character hex RGB format (e.g., ``#ff7700'' for orange:\n"
 | |
|          "\t\t  same as HTML colors)\n"
 | |
|          "    -text\tprompt interactively for text info (tEXt chunks)\n"
 | |
|          "    -time\tinclude a tIME chunk (last modification time)\n"
 | |
|          "    -interlace\twrite interlaced PNG image\n"
 | |
|          "\n"
 | |
| "pnmfile or stdin must be a binary PGM (`P5'), PPM (`P6') or (extremely\n"
 | |
| "unofficial and unsupported!) PAM (`P8') file.  Currently it is required\n"
 | |
| "to have maxval == 255 (i.e., no scaling).  If pnmfile is specified, it\n"
 | |
| "is converted to the corresponding PNG file with the same base name but a\n"
 | |
| "``.png'' extension; files read from stdin are converted and sent to stdout.\n"
 | |
| "The conversion is progressive (low memory usage) unless interlacing is\n"
 | |
| "requested; in that case the whole image will be buffered in memory and\n"
 | |
| "written in one call.\n"
 | |
|          "\n", PROGNAME, PROGNAME, default_gamma);
 | |
|         exit(1);
 | |
|     }
 | |
| 
 | |
| 
 | |
|     /* prepare the text buffers for libpng's use; note that even though
 | |
|      * PNG's png_text struct includes a length field, we don't have to fill
 | |
|      * it out */
 | |
| 
 | |
|     if (text &&
 | |
| #ifndef DOS_OS2_W32
 | |
|         (keybd = fdopen(fileno(stderr), "r")) != NULL &&
 | |
| #endif
 | |
|         (textbuf = (char *)malloc((5 + 9)*75)) != NULL)
 | |
|     {
 | |
|         int i, valid, result;
 | |
| 
 | |
|         fprintf(stderr,
 | |
|           "Enter text info (no more than 72 characters per line);\n");
 | |
|         fprintf(stderr, "to skip a field, hit the <Enter> key.\n");
 | |
|         /* note:  just <Enter> leaves len == 1 */
 | |
| 
 | |
|         do {
 | |
|             valid = TRUE;
 | |
|             p = textbuf + TEXT_TITLE_OFFSET;
 | |
|             fprintf(stderr, "  Title: ");
 | |
|             fflush(stderr);
 | |
|             if (FGETS(p, 74, keybd) && (len = strlen(p)) > 1) {
 | |
|                 if (p[len-1] == '\n')
 | |
|                     p[--len] = '\0';
 | |
|                 wpng_info.title = p;
 | |
|                 wpng_info.have_text |= TEXT_TITLE;
 | |
|                 if ((result = wpng_isvalid_latin1((uch *)p, len)) >= 0) {
 | |
|                     fprintf(stderr, "    " PROGNAME " warning:  character code"
 | |
|                       " %u is %sdiscouraged by the PNG\n    specification "
 | |
|                       "[first occurrence was at character position #%d]\n",
 | |
|                       (unsigned)p[result], (p[result] == 27)? "strongly " : "",
 | |
|                       result+1);
 | |
|                     fflush(stderr);
 | |
| #ifdef FORBID_LATIN1_CTRL
 | |
|                     wpng_info.have_text &= ~TEXT_TITLE;
 | |
|                     valid = FALSE;
 | |
| #else
 | |
|                     if (p[result] == 27) {    /* escape character */
 | |
|                         wpng_info.have_text &= ~TEXT_TITLE;
 | |
|                         valid = FALSE;
 | |
|                     }
 | |
| #endif
 | |
|                 }
 | |
|             }
 | |
|         } while (!valid);
 | |
| 
 | |
|         do {
 | |
|             valid = TRUE;
 | |
|             p = textbuf + TEXT_AUTHOR_OFFSET;
 | |
|             fprintf(stderr, "  Author: ");
 | |
|             fflush(stderr);
 | |
|             if (FGETS(p, 74, keybd) && (len = strlen(p)) > 1) {
 | |
|                 if (p[len-1] == '\n')
 | |
|                     p[--len] = '\0';
 | |
|                 wpng_info.author = p;
 | |
|                 wpng_info.have_text |= TEXT_AUTHOR;
 | |
|                 if ((result = wpng_isvalid_latin1((uch *)p, len)) >= 0) {
 | |
|                     fprintf(stderr, "    " PROGNAME " warning:  character code"
 | |
|                       " %u is %sdiscouraged by the PNG\n    specification "
 | |
|                       "[first occurrence was at character position #%d]\n",
 | |
|                       (unsigned)p[result], (p[result] == 27)? "strongly " : "",
 | |
|                       result+1);
 | |
|                     fflush(stderr);
 | |
| #ifdef FORBID_LATIN1_CTRL
 | |
|                     wpng_info.have_text &= ~TEXT_AUTHOR;
 | |
|                     valid = FALSE;
 | |
| #else
 | |
|                     if (p[result] == 27) {    /* escape character */
 | |
|                         wpng_info.have_text &= ~TEXT_AUTHOR;
 | |
|                         valid = FALSE;
 | |
|                     }
 | |
| #endif
 | |
|                 }
 | |
|             }
 | |
|         } while (!valid);
 | |
| 
 | |
|         do {
 | |
|             valid = TRUE;
 | |
|             p = textbuf + TEXT_DESC_OFFSET;
 | |
|             fprintf(stderr, "  Description (up to 9 lines):\n");
 | |
|             for (i = 1;  i < 10;  ++i) {
 | |
|                 fprintf(stderr, "    [%d] ", i);
 | |
|                 fflush(stderr);
 | |
|                 if (FGETS(p, 74, keybd) && (len = strlen(p)) > 1)
 | |
|                     p += len;   /* now points at NULL; char before is newline */
 | |
|                 else
 | |
|                     break;
 | |
|             }
 | |
|             if ((len = p - (textbuf + TEXT_DESC_OFFSET)) > 1) {
 | |
|                 if (p[-1] == '\n') {
 | |
|                     p[-1] = '\0';
 | |
|                     --len;
 | |
|                 }
 | |
|                 wpng_info.desc = textbuf + TEXT_DESC_OFFSET;
 | |
|                 wpng_info.have_text |= TEXT_DESC;
 | |
|                 p = textbuf + TEXT_DESC_OFFSET;
 | |
|                 if ((result = wpng_isvalid_latin1((uch *)p, len)) >= 0) {
 | |
|                     fprintf(stderr, "    " PROGNAME " warning:  character code"
 | |
|                       " %u is %sdiscouraged by the PNG\n    specification "
 | |
|                       "[first occurrence was at character position #%d]\n",
 | |
|                       (unsigned)p[result], (p[result] == 27)? "strongly " : "",
 | |
|                       result+1);
 | |
|                     fflush(stderr);
 | |
| #ifdef FORBID_LATIN1_CTRL
 | |
|                     wpng_info.have_text &= ~TEXT_DESC;
 | |
|                     valid = FALSE;
 | |
| #else
 | |
|                     if (p[result] == 27) {    /* escape character */
 | |
|                         wpng_info.have_text &= ~TEXT_DESC;
 | |
|                         valid = FALSE;
 | |
|                     }
 | |
| #endif
 | |
|                 }
 | |
|             }
 | |
|         } while (!valid);
 | |
| 
 | |
|         do {
 | |
|             valid = TRUE;
 | |
|             p = textbuf + TEXT_COPY_OFFSET;
 | |
|             fprintf(stderr, "  Copyright: ");
 | |
|             fflush(stderr);
 | |
|             if (FGETS(p, 74, keybd) && (len = strlen(p)) > 1) {
 | |
|                 if (p[len-1] == '\n')
 | |
|                     p[--len] = '\0';
 | |
|                 wpng_info.copyright = p;
 | |
|                 wpng_info.have_text |= TEXT_COPY;
 | |
|                 if ((result = wpng_isvalid_latin1((uch *)p, len)) >= 0) {
 | |
|                     fprintf(stderr, "    " PROGNAME " warning:  character code"
 | |
|                       " %u is %sdiscouraged by the PNG\n    specification "
 | |
|                       "[first occurrence was at character position #%d]\n",
 | |
|                       (unsigned)p[result], (p[result] == 27)? "strongly " : "",
 | |
|                       result+1);
 | |
|                     fflush(stderr);
 | |
| #ifdef FORBID_LATIN1_CTRL
 | |
|                     wpng_info.have_text &= ~TEXT_COPY;
 | |
|                     valid = FALSE;
 | |
| #else
 | |
|                     if (p[result] == 27) {    /* escape character */
 | |
|                         wpng_info.have_text &= ~TEXT_COPY;
 | |
|                         valid = FALSE;
 | |
|                     }
 | |
| #endif
 | |
|                 }
 | |
|             }
 | |
|         } while (!valid);
 | |
| 
 | |
|         do {
 | |
|             valid = TRUE;
 | |
|             p = textbuf + TEXT_EMAIL_OFFSET;
 | |
|             fprintf(stderr, "  E-mail: ");
 | |
|             fflush(stderr);
 | |
|             if (FGETS(p, 74, keybd) && (len = strlen(p)) > 1) {
 | |
|                 if (p[len-1] == '\n')
 | |
|                     p[--len] = '\0';
 | |
|                 wpng_info.email = p;
 | |
|                 wpng_info.have_text |= TEXT_EMAIL;
 | |
|                 if ((result = wpng_isvalid_latin1((uch *)p, len)) >= 0) {
 | |
|                     fprintf(stderr, "    " PROGNAME " warning:  character code"
 | |
|                       " %u is %sdiscouraged by the PNG\n    specification "
 | |
|                       "[first occurrence was at character position #%d]\n",
 | |
|                       (unsigned)p[result], (p[result] == 27)? "strongly " : "",
 | |
|                       result+1);
 | |
|                     fflush(stderr);
 | |
| #ifdef FORBID_LATIN1_CTRL
 | |
|                     wpng_info.have_text &= ~TEXT_EMAIL;
 | |
|                     valid = FALSE;
 | |
| #else
 | |
|                     if (p[result] == 27) {    /* escape character */
 | |
|                         wpng_info.have_text &= ~TEXT_EMAIL;
 | |
|                         valid = FALSE;
 | |
|                     }
 | |
| #endif
 | |
|                 }
 | |
|             }
 | |
|         } while (!valid);
 | |
| 
 | |
|         do {
 | |
|             valid = TRUE;
 | |
|             p = textbuf + TEXT_URL_OFFSET;
 | |
|             fprintf(stderr, "  URL: ");
 | |
|             fflush(stderr);
 | |
|             if (FGETS(p, 74, keybd) && (len = strlen(p)) > 1) {
 | |
|                 if (p[len-1] == '\n')
 | |
|                     p[--len] = '\0';
 | |
|                 wpng_info.url = p;
 | |
|                 wpng_info.have_text |= TEXT_URL;
 | |
|                 if ((result = wpng_isvalid_latin1((uch *)p, len)) >= 0) {
 | |
|                     fprintf(stderr, "    " PROGNAME " warning:  character code"
 | |
|                       " %u is %sdiscouraged by the PNG\n    specification "
 | |
|                       "[first occurrence was at character position #%d]\n",
 | |
|                       (unsigned)p[result], (p[result] == 27)? "strongly " : "",
 | |
|                       result+1);
 | |
|                     fflush(stderr);
 | |
| #ifdef FORBID_LATIN1_CTRL
 | |
|                     wpng_info.have_text &= ~TEXT_URL;
 | |
|                     valid = FALSE;
 | |
| #else
 | |
|                     if (p[result] == 27) {    /* escape character */
 | |
|                         wpng_info.have_text &= ~TEXT_URL;
 | |
|                         valid = FALSE;
 | |
|                     }
 | |
| #endif
 | |
|                 }
 | |
|             }
 | |
|         } while (!valid);
 | |
| 
 | |
| #ifndef DOS_OS2_W32
 | |
|         fclose(keybd);
 | |
| #endif
 | |
| 
 | |
|     } else if (text) {
 | |
|         fprintf(stderr, PROGNAME ":  unable to allocate memory for text\n");
 | |
|         text = FALSE;
 | |
|         wpng_info.have_text = 0;
 | |
|     }
 | |
| 
 | |
| 
 | |
|     /* allocate libpng stuff, initialize transformations, write pre-IDAT data */
 | |
| 
 | |
|     if ((rc = writepng_init(&wpng_info)) != 0) {
 | |
|         switch (rc) {
 | |
|             case 2:
 | |
|                 fprintf(stderr, PROGNAME
 | |
|                   ":  libpng initialization problem (longjmp)\n");
 | |
|                 break;
 | |
|             case 4:
 | |
|                 fprintf(stderr, PROGNAME ":  insufficient memory\n");
 | |
|                 break;
 | |
|             case 11:
 | |
|                 fprintf(stderr, PROGNAME
 | |
|                   ":  internal logic error (unexpected PNM type)\n");
 | |
|                 break;
 | |
|             default:
 | |
|                 fprintf(stderr, PROGNAME
 | |
|                   ":  unknown writepng_init() error\n");
 | |
|                 break;
 | |
|         }
 | |
|         exit(rc);
 | |
|     }
 | |
| 
 | |
| 
 | |
|     /* free textbuf, since it's a completely local variable and all text info
 | |
|      * has just been written to the PNG file */
 | |
| 
 | |
|     if (text && textbuf) {
 | |
|         free(textbuf);
 | |
|         textbuf = NULL;
 | |
|     }
 | |
| 
 | |
| 
 | |
|     /* calculate rowbytes on basis of image type; note that this becomes much
 | |
|      * more complicated if we choose to support PBM type, ASCII PNM types, or
 | |
|      * 16-bit-per-sample binary data [currently not an official NetPBM type] */
 | |
| 
 | |
|     if (wpng_info.pnmtype == 5)
 | |
|         rowbytes = wpng_info.width;
 | |
|     else if (wpng_info.pnmtype == 6)
 | |
|         rowbytes = wpng_info.width * 3;
 | |
|     else /* if (wpng_info.pnmtype == 8) */
 | |
|         rowbytes = wpng_info.width * 4;
 | |
| 
 | |
| 
 | |
|     /* read and write the image, either in its entirety (if writing interlaced
 | |
|      * PNG) or row by row (if non-interlaced) */
 | |
| 
 | |
|     fprintf(stderr, "Encoding image data...\n");
 | |
|     fflush(stderr);
 | |
| 
 | |
|     if (wpng_info.interlaced) {
 | |
|         long i;
 | |
|         ulg bytes;
 | |
|         ulg image_bytes;
 | |
| 
 | |
|         /* Guard against integer overflow */
 | |
|         if (wpng_info_height > ((size_t)(-1)/rowbytes ||
 | |
|             wpng_info_height > ((ulg)(-1)/rowbytes) {
 | |
|             fprintf(stderr, PROGNAME ":  image_data buffer too large\n");
 | |
|             writepng_cleanup(&wpng_info);
 | |
|             wpng_cleanup();
 | |
|             exit(5);
 | |
|         }
 | |
| 
 | |
|         image_bytes = rowbytes * wpng_info.height;
 | |
| 
 | |
|         wpng_info.image_data = (uch *)malloc(image_bytes);
 | |
|         wpng_info.row_pointers = (uch **)malloc(wpng_info.height*sizeof(uch *));
 | |
|         if (wpng_info.image_data == NULL || wpng_info.row_pointers == NULL) {
 | |
|             fprintf(stderr, PROGNAME ":  insufficient memory for image data\n");
 | |
|             writepng_cleanup(&wpng_info);
 | |
|             wpng_cleanup();
 | |
|             exit(5);
 | |
|         }
 | |
|         for (i = 0;  i < wpng_info.height;  ++i)
 | |
|             wpng_info.row_pointers[i] = wpng_info.image_data + i*rowbytes;
 | |
|         bytes = fread(wpng_info.image_data, 1, image_bytes, wpng_info.infile);
 | |
|         if (bytes != image_bytes) {
 | |
|             fprintf(stderr, PROGNAME ":  expected %lu bytes, got %lu bytes\n",
 | |
|               image_bytes, bytes);
 | |
|             fprintf(stderr, "  (continuing anyway)\n");
 | |
|         }
 | |
|         if (writepng_encode_image(&wpng_info) != 0) {
 | |
|             fprintf(stderr, PROGNAME
 | |
|               ":  libpng problem (longjmp) while writing image data\n");
 | |
|             writepng_cleanup(&wpng_info);
 | |
|             wpng_cleanup();
 | |
|             exit(2);
 | |
|         }
 | |
| 
 | |
|     } else /* not interlaced:  write progressively (row by row) */ {
 | |
|         long j;
 | |
|         ulg bytes;
 | |
| 
 | |
|         wpng_info.image_data = (uch *)malloc(rowbytes);
 | |
|         if (wpng_info.image_data == NULL) {
 | |
|             fprintf(stderr, PROGNAME ":  insufficient memory for row data\n");
 | |
|             writepng_cleanup(&wpng_info);
 | |
|             wpng_cleanup();
 | |
|             exit(5);
 | |
|         }
 | |
|         error = 0;
 | |
|         for (j = wpng_info.height;  j > 0L;  --j) {
 | |
|             bytes = fread(wpng_info.image_data, 1, rowbytes, wpng_info.infile);
 | |
|             if (bytes != rowbytes) {
 | |
|                 fprintf(stderr, PROGNAME
 | |
|                   ":  expected %lu bytes, got %lu bytes (row %ld)\n", rowbytes,
 | |
|                   bytes, wpng_info.height-j);
 | |
|                 ++error;
 | |
|                 break;
 | |
|             }
 | |
|             if (writepng_encode_row(&wpng_info) != 0) {
 | |
|                 fprintf(stderr, PROGNAME
 | |
|                   ":  libpng problem (longjmp) while writing row %ld\n",
 | |
|                   wpng_info.height-j);
 | |
|                 ++error;
 | |
|                 break;
 | |
|             }
 | |
|         }
 | |
|         if (error) {
 | |
|             writepng_cleanup(&wpng_info);
 | |
|             wpng_cleanup();
 | |
|             exit(2);
 | |
|         }
 | |
|         if (writepng_encode_finish(&wpng_info) != 0) {
 | |
|             fprintf(stderr, PROGNAME ":  error on final libpng call\n");
 | |
|             writepng_cleanup(&wpng_info);
 | |
|             wpng_cleanup();
 | |
|             exit(2);
 | |
|         }
 | |
|     }
 | |
| 
 | |
| 
 | |
|     /* OK, we're done (successfully):  clean up all resources and quit */
 | |
| 
 | |
|     fprintf(stderr, "Done.\n");
 | |
|     fflush(stderr);
 | |
| 
 | |
|     writepng_cleanup(&wpng_info);
 | |
|     wpng_cleanup();
 | |
| 
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
| static int wpng_isvalid_latin1(uch *p, int len)
 | |
| {
 | |
|     int i, result = -1;
 | |
| 
 | |
|     for (i = 0;  i < len;  ++i) {
 | |
|         if (p[i] == 10 || (p[i] > 31 && p[i] < 127) || p[i] > 160)
 | |
|             continue;           /* character is completely OK */
 | |
|         if (result < 0 || (p[result] != 27 && p[i] == 27))
 | |
|             result = i;         /* mark location of first questionable one */
 | |
|     }                           /*  or of first escape character (bad) */
 | |
| 
 | |
|     return result;
 | |
| }
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
| static void wpng_cleanup(void)
 | |
| {
 | |
|     if (wpng_info.outfile) {
 | |
|         fclose(wpng_info.outfile);
 | |
|         wpng_info.outfile = NULL;
 | |
|     }
 | |
| 
 | |
|     if (wpng_info.infile) {
 | |
|         fclose(wpng_info.infile);
 | |
|         wpng_info.infile = NULL;
 | |
|     }
 | |
| 
 | |
|     if (wpng_info.image_data) {
 | |
|         free(wpng_info.image_data);
 | |
|         wpng_info.image_data = NULL;
 | |
|     }
 | |
| 
 | |
|     if (wpng_info.row_pointers) {
 | |
|         free(wpng_info.row_pointers);
 | |
|         wpng_info.row_pointers = NULL;
 | |
|     }
 | |
| }
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
| #ifdef DOS_OS2_W32
 | |
| 
 | |
| static char *dos_kbd_gets(char *buf, int len)
 | |
| {
 | |
|     int ch, count=0;
 | |
| 
 | |
|     do {
 | |
|         buf[count++] = ch = getche();
 | |
|     } while (ch != '\r' && count < len-1);
 | |
| 
 | |
|     buf[count--] = '\0';        /* terminate string */
 | |
|     if (buf[count] == '\r')     /* Enter key makes CR, so change to newline */
 | |
|         buf[count] = '\n';
 | |
| 
 | |
|     fprintf(stderr, "\n");      /* Enter key does *not* cause a newline */
 | |
|     fflush(stderr);
 | |
| 
 | |
|     return buf;
 | |
| }
 | |
| 
 | |
| #endif /* DOS_OS2_W32 */
 | 
