2016-02-17 13:52:48 +03:00
/* png.c - Handles output to PNG file */
/*
libzint - the open source barcode library
2017-05-11 23:14:38 +03:00
Copyright ( C ) 2009 - 2017 Robin Stuart < rstuart114 @ gmail . com >
2016-02-17 13:52:48 +03:00
Redistribution and use in source and binary forms , with or without
modification , are permitted provided that the following conditions
are met :
2017-10-23 22:37:52 +03:00
1. Redistributions of source code must retain the above copyright
notice , this list of conditions and the following disclaimer .
2016-02-17 13:52:48 +03:00
2. Redistributions in binary form must reproduce the above copyright
notice , this list of conditions and the following disclaimer in the
2017-10-23 22:37:52 +03:00
documentation and / or other materials provided with the distribution .
2016-02-17 13:52:48 +03:00
3. Neither the name of the project nor the names of its contributors
may be used to endorse or promote products derived from this software
2017-10-23 22:37:52 +03:00
without specific prior written permission .
2016-02-17 13:52:48 +03:00
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS " AS IS " AND
ANY EXPRESS OR IMPLIED WARRANTIES , INCLUDING , BUT NOT LIMITED TO , THE
IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
ARE DISCLAIMED . IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE
FOR ANY DIRECT , INDIRECT , INCIDENTAL , SPECIAL , EXEMPLARY , OR CONSEQUENTIAL
DAMAGES ( INCLUDING , BUT NOT LIMITED TO , PROCUREMENT OF SUBSTITUTE GOODS
OR SERVICES ; LOSS OF USE , DATA , OR PROFITS ; OR BUSINESS INTERRUPTION )
HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY , WHETHER IN CONTRACT , STRICT
LIABILITY , OR TORT ( INCLUDING NEGLIGENCE OR OTHERWISE ) ARISING IN ANY WAY
2017-10-23 22:37:52 +03:00
OUT OF THE USE OF THIS SOFTWARE , EVEN IF ADVISED OF THE POSSIBILITY OF
2016-02-17 13:52:48 +03:00
SUCH DAMAGE .
2016-02-20 13:50:15 +03:00
*/
2019-12-19 03:37:55 +03:00
/* vim: set ts=4 sw=4 et : */
2016-02-17 13:52:48 +03:00
# include <stdio.h>
# ifdef _MSC_VER
# include <fcntl.h>
# include <io.h>
2016-11-06 00:06:58 +03:00
# include <malloc.h>
2016-02-17 13:52:48 +03:00
# endif
# include <stdlib.h>
# include <string.h>
# include "common.h"
# ifndef NO_PNG
# include <png.h>
# include <zlib.h>
# include <setjmp.h>
2016-07-18 22:14:56 +03:00
2016-02-17 13:52:48 +03:00
# define SSET "0123456789ABCDEF"
struct mainprog_info_type {
long width ;
long height ;
FILE * outfile ;
jmp_buf jmpbuf ;
} ;
2016-02-20 13:50:15 +03:00
static void writepng_error_handler ( png_structp png_ptr , png_const_charp msg ) {
struct mainprog_info_type * graphic ;
2016-02-17 13:52:48 +03:00
2016-10-29 00:40:40 +03:00
fprintf ( stderr , " writepng libpng error: %s (F30) \n " , msg ) ;
2016-02-17 13:52:48 +03:00
fflush ( stderr ) ;
2016-02-20 13:50:15 +03:00
graphic = ( struct mainprog_info_type * ) png_get_error_ptr ( png_ptr ) ;
if ( graphic = = NULL ) {
/* we are completely hosed now */
2016-02-17 13:52:48 +03:00
fprintf ( stderr ,
2016-10-29 00:40:40 +03:00
" writepng severe error: jmpbuf not recoverable; terminating. (F31) \n " ) ;
2016-02-17 13:52:48 +03:00
fflush ( stderr ) ;
return ;
}
longjmp ( graphic - > jmpbuf , 1 ) ;
}
2019-12-19 03:37:55 +03:00
INTERNAL int png_pixel_plot ( struct zint_symbol * symbol , char * pixelbuf ) {
2016-02-20 13:50:15 +03:00
struct mainprog_info_type wpng_info ;
struct mainprog_info_type * graphic ;
png_structp png_ptr ;
png_infop info_ptr ;
2016-09-11 10:42:31 +03:00
int i , row , column ;
2016-02-20 13:50:15 +03:00
int fgred , fggrn , fgblu , bgred , bggrn , bgblu ;
2020-08-03 00:26:39 +03:00
int fgalpha , bgalpha , use_alpha ;
2016-02-20 13:50:15 +03:00
2016-02-17 13:52:48 +03:00
# ifndef _MSC_VER
2020-08-03 00:26:39 +03:00
unsigned char outdata [ symbol - > bitmap_width * 4 ] ;
2016-02-17 13:52:48 +03:00
# else
2020-08-03 00:26:39 +03:00
unsigned char * outdata = ( unsigned char * ) _alloca ( symbol - > bitmap_width * 4 ) ;
2016-02-17 13:52:48 +03:00
# endif
2016-02-20 13:50:15 +03:00
graphic = & wpng_info ;
2017-10-23 22:37:52 +03:00
2016-09-11 10:42:31 +03:00
graphic - > width = symbol - > bitmap_width ;
graphic - > height = symbol - > bitmap_height ;
2016-02-20 13:50:15 +03:00
fgred = ( 16 * ctoi ( symbol - > fgcolour [ 0 ] ) ) + ctoi ( symbol - > fgcolour [ 1 ] ) ;
fggrn = ( 16 * ctoi ( symbol - > fgcolour [ 2 ] ) ) + ctoi ( symbol - > fgcolour [ 3 ] ) ;
fgblu = ( 16 * ctoi ( symbol - > fgcolour [ 4 ] ) ) + ctoi ( symbol - > fgcolour [ 5 ] ) ;
bgred = ( 16 * ctoi ( symbol - > bgcolour [ 0 ] ) ) + ctoi ( symbol - > bgcolour [ 1 ] ) ;
bggrn = ( 16 * ctoi ( symbol - > bgcolour [ 2 ] ) ) + ctoi ( symbol - > bgcolour [ 3 ] ) ;
bgblu = ( 16 * ctoi ( symbol - > bgcolour [ 4 ] ) ) + ctoi ( symbol - > bgcolour [ 5 ] ) ;
2020-08-03 00:26:39 +03:00
use_alpha = 0 ;
if ( strlen ( symbol - > fgcolour ) > 6 ) {
fgalpha = ( 16 * ctoi ( symbol - > fgcolour [ 6 ] ) ) + ctoi ( symbol - > fgcolour [ 7 ] ) ;
if ( fgalpha ! = 0xff ) use_alpha = 1 ;
} else {
fgalpha = 0xff ;
}
if ( strlen ( symbol - > bgcolour ) > 6 ) {
bgalpha = ( 16 * ctoi ( symbol - > bgcolour [ 6 ] ) ) + ctoi ( symbol - > bgcolour [ 7 ] ) ;
if ( bgalpha ! = 0xff ) use_alpha = 1 ;
} else {
bgalpha = 0xff ;
}
2016-02-20 13:50:15 +03:00
/* Open output file in binary mode */
2016-08-26 17:13:40 +03:00
if ( symbol - > output_options & BARCODE_STDOUT ) {
2016-02-17 13:52:48 +03:00
# ifdef _MSC_VER
2016-02-20 13:50:15 +03:00
if ( - 1 = = _setmode ( _fileno ( stdout ) , _O_BINARY ) ) {
2017-07-27 18:01:53 +03:00
strcpy ( symbol - > errtxt , " 631: Can't open output file " ) ;
2016-02-20 13:50:15 +03:00
return ZINT_ERROR_FILE_ACCESS ;
}
2016-02-17 13:52:48 +03:00
# endif
2016-02-20 13:50:15 +03:00
graphic - > outfile = stdout ;
} else {
if ( ! ( graphic - > outfile = fopen ( symbol - > outfile , " wb " ) ) ) {
2017-07-27 18:01:53 +03:00
strcpy ( symbol - > errtxt , " 632: Can't open output file " ) ;
2016-02-20 13:50:15 +03:00
return ZINT_ERROR_FILE_ACCESS ;
}
}
/* Set up error handling routine as proc() above */
png_ptr = png_create_write_struct ( PNG_LIBPNG_VER_STRING , graphic , writepng_error_handler , NULL ) ;
if ( ! png_ptr ) {
2017-07-27 18:01:53 +03:00
strcpy ( symbol - > errtxt , " 633: Out of memory " ) ;
2016-02-20 13:50:15 +03:00
return ZINT_ERROR_MEMORY ;
}
info_ptr = png_create_info_struct ( png_ptr ) ;
if ( ! info_ptr ) {
png_destroy_write_struct ( & png_ptr , NULL ) ;
2017-07-27 18:01:53 +03:00
strcpy ( symbol - > errtxt , " 634: Out of memory " ) ;
2016-02-20 13:50:15 +03:00
return ZINT_ERROR_MEMORY ;
}
/* catch jumping here */
if ( setjmp ( graphic - > jmpbuf ) ) {
png_destroy_write_struct ( & png_ptr , & info_ptr ) ;
2017-07-27 18:01:53 +03:00
strcpy ( symbol - > errtxt , " 635: libpng error occurred " ) ;
2016-02-20 13:50:15 +03:00
return ZINT_ERROR_MEMORY ;
}
/* open output file with libpng */
png_init_io ( png_ptr , graphic - > outfile ) ;
/* set compression */
png_set_compression_level ( png_ptr , 9 ) ;
/* set Header block */
2020-08-03 00:26:39 +03:00
if ( use_alpha )
png_set_IHDR ( png_ptr , info_ptr , graphic - > width , graphic - > height ,
8 , PNG_COLOR_TYPE_RGBA , PNG_INTERLACE_NONE ,
PNG_COMPRESSION_TYPE_DEFAULT , PNG_FILTER_TYPE_DEFAULT ) ;
else
png_set_IHDR ( png_ptr , info_ptr , graphic - > width , graphic - > height ,
8 , PNG_COLOR_TYPE_RGB , PNG_INTERLACE_NONE ,
PNG_COMPRESSION_TYPE_DEFAULT , PNG_FILTER_TYPE_DEFAULT ) ;
2016-02-20 13:50:15 +03:00
/* write all chunks up to (but not including) first IDAT */
png_write_info ( png_ptr , info_ptr ) ;
/* set up the transformations: for now, just pack low-bit-depth pixels
into bytes ( one , two or four pixels per byte ) */
png_set_packing ( png_ptr ) ;
/* Pixel Plotting */
2016-09-11 10:42:31 +03:00
for ( row = 0 ; row < symbol - > bitmap_height ; row + + ) {
2017-10-23 22:34:31 +03:00
unsigned char * image_data ;
2016-09-11 10:42:31 +03:00
for ( column = 0 ; column < symbol - > bitmap_width ; column + + ) {
i = column * 3 ;
2020-08-03 00:26:39 +03:00
if ( use_alpha ) i + = column ;
2016-09-11 10:42:31 +03:00
switch ( * ( pixelbuf + ( symbol - > bitmap_width * row ) + column ) ) {
2020-01-06 23:01:48 +03:00
case ' W ' : // White
outdata [ i ] = 255 ;
outdata [ i + 1 ] = 255 ;
outdata [ i + 2 ] = 255 ;
2020-08-03 00:26:39 +03:00
if ( use_alpha ) outdata [ i + 3 ] = fgalpha ;
2020-01-06 23:01:48 +03:00
break ;
case ' C ' : // Cyan
outdata [ i ] = 0 ;
outdata [ i + 1 ] = 255 ;
outdata [ i + 2 ] = 255 ;
2020-08-03 00:26:39 +03:00
if ( use_alpha ) outdata [ i + 3 ] = fgalpha ;
2020-01-06 23:01:48 +03:00
break ;
case ' B ' : // Blue
outdata [ i ] = 0 ;
outdata [ i + 1 ] = 0 ;
outdata [ i + 2 ] = 255 ;
2020-08-03 00:26:39 +03:00
if ( use_alpha ) outdata [ i + 3 ] = fgalpha ;
2020-01-06 23:01:48 +03:00
break ;
case ' M ' : // Magenta
outdata [ i ] = 255 ;
outdata [ i + 1 ] = 0 ;
outdata [ i + 2 ] = 255 ;
2020-08-03 00:26:39 +03:00
if ( use_alpha ) outdata [ i + 3 ] = fgalpha ;
2020-01-06 23:01:48 +03:00
break ;
case ' R ' : // Red
outdata [ i ] = 255 ;
outdata [ i + 1 ] = 0 ;
outdata [ i + 2 ] = 0 ;
2020-08-03 00:26:39 +03:00
if ( use_alpha ) outdata [ i + 3 ] = fgalpha ;
2020-01-06 23:01:48 +03:00
break ;
case ' Y ' : // Yellow
outdata [ i ] = 255 ;
outdata [ i + 1 ] = 255 ;
outdata [ i + 2 ] = 0 ;
2020-08-03 00:26:39 +03:00
if ( use_alpha ) outdata [ i + 3 ] = fgalpha ;
2020-01-06 23:01:48 +03:00
break ;
case ' G ' : // Green
outdata [ i ] = 0 ;
outdata [ i + 1 ] = 255 ;
outdata [ i + 2 ] = 0 ;
2020-08-03 00:26:39 +03:00
if ( use_alpha ) outdata [ i + 3 ] = fgalpha ;
2020-01-06 23:01:48 +03:00
break ;
case ' K ' : // Black
outdata [ i ] = 0 ;
outdata [ i + 1 ] = 0 ;
outdata [ i + 2 ] = 0 ;
2020-08-03 00:26:39 +03:00
if ( use_alpha ) outdata [ i + 3 ] = fgalpha ;
2020-01-06 23:01:48 +03:00
break ;
2016-09-11 10:42:31 +03:00
case ' 1 ' :
outdata [ i ] = fgred ;
outdata [ i + 1 ] = fggrn ;
outdata [ i + 2 ] = fgblu ;
2020-08-03 00:26:39 +03:00
if ( use_alpha ) outdata [ i + 3 ] = fgalpha ;
2016-09-11 10:42:31 +03:00
break ;
default :
outdata [ i ] = bgred ;
outdata [ i + 1 ] = bggrn ;
outdata [ i + 2 ] = bgblu ;
2020-08-03 00:26:39 +03:00
if ( use_alpha ) outdata [ i + 3 ] = bgalpha ;
2016-09-11 10:42:31 +03:00
break ;
2016-02-20 13:50:15 +03:00
}
2016-09-11 10:42:31 +03:00
}
/* write row contents to file */
2017-10-16 20:26:54 +03:00
image_data = outdata ;
2016-09-11 10:42:31 +03:00
png_write_row ( png_ptr , image_data ) ;
2016-02-20 13:50:15 +03:00
}
/* End the file */
png_write_end ( png_ptr , NULL ) ;
/* make sure we have disengaged */
if ( png_ptr & & info_ptr ) png_destroy_write_struct ( & png_ptr , & info_ptr ) ;
if ( symbol - > output_options & BARCODE_STDOUT ) {
fflush ( wpng_info . outfile ) ;
} else {
fclose ( wpng_info . outfile ) ;
}
return 0 ;
2016-02-17 13:52:48 +03:00
}
# endif /* NO_PNG */