2016-12-30 23:25:58 +03:00
/* tif.c - Aldus Tagged Image File Format support */
2020-05-06 21:57:27 +03:00
/* TIFF Revision 6.0 https://www.adobe.io/content/dam/udp/en/open/standards/tiff/TIFF6.pdf */
2016-12-30 23:25:58 +03:00
/*
libzint - the open source barcode library
2020-04-04 18:53:29 +03:00
Copyright ( C ) 2016 - 2020 Robin Stuart < rstuart114 @ gmail . com >
2016-12-30 23:25:58 +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-12-30 23:25:58 +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-12-30 23:25:58 +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-12-30 23:25:58 +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-12-30 23:25:58 +03:00
SUCH DAMAGE .
*/
2019-12-19 03:37:55 +03:00
/* vim: set ts=4 sw=4 et : */
2016-12-30 23:25:58 +03:00
# include <stdio.h>
# include <math.h>
2020-06-04 20:45:25 +03:00
# include <assert.h>
2016-12-30 23:25:58 +03:00
# include "common.h"
# include "tif.h"
# ifdef _MSC_VER
# include <io.h>
# include <fcntl.h>
# include <malloc.h>
# endif
2020-11-01 21:32:55 +03:00
INTERNAL int tif_pixel_plot ( struct zint_symbol * symbol , unsigned char * pixelbuf ) {
2016-12-30 23:25:58 +03:00
int fgred , fggrn , fgblu , bgred , bggrn , bgblu ;
int i ;
int rows_per_strip , strip_count ;
2020-04-04 18:53:29 +03:00
unsigned int free_memory ;
int row , column , strip ;
unsigned int bytes_put ;
2016-12-30 23:25:58 +03:00
FILE * tif_file ;
2017-07-27 18:01:53 +03:00
# ifdef _MSC_VER
uint32_t * strip_offset ;
uint32_t * strip_bytes ;
# endif
2017-10-23 22:37:52 +03:00
2016-12-30 23:25:58 +03:00
tiff_header_t header ;
tiff_ifd_t ifd ;
uint16_t temp ;
uint32_t temp32 ;
2017-10-23 22:37:52 +03:00
2016-12-30 23:25:58 +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 ] ) ;
2017-10-23 22:37:52 +03:00
2020-05-06 21:57:27 +03:00
/* TIFF Rev 6 Section 7 p.27 "Set RowsPerStrip such that the size of each strip is about 8K bytes...
* Note that extremely wide high resolution images may have rows larger than 8 K bytes ; in this case ,
* RowsPerStrip should be 1 , and the strip will be larger than 8 K . " */
2016-12-30 23:25:58 +03:00
rows_per_strip = 8192 / ( symbol - > bitmap_width * 3 ) ;
if ( rows_per_strip = = 0 ) {
rows_per_strip = 1 ;
}
2020-06-04 20:45:25 +03:00
/* Suppresses clang-tidy clang-analyzer-core.VLASize warning */
assert ( symbol - > bitmap_height > 0 ) ;
2016-12-30 23:25:58 +03:00
strip_count = symbol - > bitmap_height / rows_per_strip ;
if ( ( symbol - > bitmap_height % rows_per_strip ) ! = 0 ) {
strip_count + + ;
}
2020-01-06 23:01:48 +03:00
2018-02-11 16:01:43 +03:00
if ( rows_per_strip > symbol - > bitmap_height ) {
rows_per_strip = symbol - > bitmap_height ;
}
2020-01-06 23:01:48 +03:00
2020-05-06 21:57:27 +03:00
if ( symbol - > debug & ZINT_DEBUG_PRINT ) {
printf ( " TIFF (%dx%d) Strip Count %d, Rows Per Strip %d \n " , symbol - > bitmap_width , symbol - > bitmap_height , strip_count , rows_per_strip ) ;
2018-02-11 16:01:43 +03:00
}
2020-01-06 23:01:48 +03:00
2016-12-30 23:25:58 +03:00
# ifndef _MSC_VER
2020-06-04 20:45:25 +03:00
uint32_t strip_offset [ strip_count ] ;
2016-12-30 23:25:58 +03:00
uint32_t strip_bytes [ strip_count ] ;
# else
2017-03-28 19:06:08 +03:00
strip_offset = ( uint32_t * ) _alloca ( strip_count * sizeof ( uint32_t ) ) ;
strip_bytes = ( uint32_t * ) _alloca ( strip_count * sizeof ( uint32_t ) ) ;
2016-12-30 23:25:58 +03:00
# endif
free_memory = 8 ;
2017-10-23 22:37:52 +03:00
2016-12-30 23:25:58 +03:00
for ( i = 0 ; i < strip_count ; i + + ) {
strip_offset [ i ] = free_memory ;
if ( i ! = ( strip_count - 1 ) ) {
strip_bytes [ i ] = rows_per_strip * symbol - > bitmap_width * 3 ;
} else {
2016-12-31 13:44:09 +03:00
if ( ( symbol - > bitmap_height % rows_per_strip ) ! = 0 ) {
strip_bytes [ i ] = ( symbol - > bitmap_height % rows_per_strip ) * symbol - > bitmap_width * 3 ;
} else {
strip_bytes [ i ] = rows_per_strip * symbol - > bitmap_width * 3 ;
}
2016-12-30 23:25:58 +03:00
}
free_memory + = strip_bytes [ i ] ;
if ( ( free_memory % 2 ) = = 1 ) {
free_memory + + ;
}
}
2017-10-23 22:37:52 +03:00
2016-12-30 23:25:58 +03:00
if ( free_memory > 0xffff0000 ) {
2017-07-27 18:01:53 +03:00
strcpy ( symbol - > errtxt , " 670: Output file size too big " ) ;
2016-12-30 23:25:58 +03:00
return ZINT_ERROR_MEMORY ;
}
/* Open output file in binary mode */
if ( symbol - > output_options & BARCODE_STDOUT ) {
# ifdef _MSC_VER
if ( - 1 = = _setmode ( _fileno ( stdout ) , _O_BINARY ) ) {
2017-07-27 18:01:53 +03:00
strcpy ( symbol - > errtxt , " 671: Can't open output file " ) ;
2016-12-30 23:25:58 +03:00
return ZINT_ERROR_FILE_ACCESS ;
}
# endif
tif_file = stdout ;
} else {
if ( ! ( tif_file = fopen ( symbol - > outfile , " wb " ) ) ) {
2017-07-27 18:01:53 +03:00
strcpy ( symbol - > errtxt , " 672: Can't open output file " ) ;
2016-12-30 23:25:58 +03:00
return ZINT_ERROR_FILE_ACCESS ;
}
}
2017-10-23 22:37:52 +03:00
2016-12-30 23:25:58 +03:00
/* Header */
header . byte_order = 0x4949 ;
header . identity = 42 ;
header . offset = free_memory ;
2017-10-23 22:37:52 +03:00
2016-12-30 23:25:58 +03:00
fwrite ( & header , sizeof ( tiff_header_t ) , 1 , tif_file ) ;
free_memory + = sizeof ( tiff_ifd_t ) ;
2020-01-06 23:01:48 +03:00
2016-12-30 23:25:58 +03:00
/* Pixel data */
2018-02-11 16:01:43 +03:00
strip = 0 ;
bytes_put = 0 ;
2016-12-30 23:25:58 +03:00
for ( row = 0 ; row < symbol - > bitmap_height ; row + + ) {
for ( column = 0 ; column < symbol - > bitmap_width ; column + + ) {
2020-01-06 23:01:48 +03:00
switch ( pixelbuf [ ( row * symbol - > bitmap_width ) + column ] ) {
case ' W ' : // White
putc ( 255 , tif_file ) ;
putc ( 255 , tif_file ) ;
putc ( 255 , tif_file ) ;
break ;
case ' C ' : // Cyan
putc ( 0 , tif_file ) ;
putc ( 255 , tif_file ) ;
putc ( 255 , tif_file ) ;
break ;
case ' B ' : // Blue
putc ( 0 , tif_file ) ;
putc ( 0 , tif_file ) ;
putc ( 255 , tif_file ) ;
break ;
case ' M ' : // Magenta
putc ( 255 , tif_file ) ;
putc ( 0 , tif_file ) ;
putc ( 255 , tif_file ) ;
break ;
case ' R ' : // Red
putc ( 255 , tif_file ) ;
putc ( 0 , tif_file ) ;
putc ( 0 , tif_file ) ;
break ;
case ' Y ' : // Yellow
putc ( 255 , tif_file ) ;
putc ( 255 , tif_file ) ;
putc ( 0 , tif_file ) ;
break ;
case ' G ' : // Green
putc ( 0 , tif_file ) ;
putc ( 255 , tif_file ) ;
putc ( 0 , tif_file ) ;
break ;
case ' K ' : // Black
putc ( 0 , tif_file ) ;
putc ( 0 , tif_file ) ;
putc ( 0 , tif_file ) ;
break ;
case ' 1 ' :
putc ( fgred , tif_file ) ;
putc ( fggrn , tif_file ) ;
putc ( fgblu , tif_file ) ;
break ;
default :
putc ( bgred , tif_file ) ;
putc ( bggrn , tif_file ) ;
putc ( bgblu , tif_file ) ;
break ;
2016-12-30 23:25:58 +03:00
}
2018-02-11 16:01:43 +03:00
bytes_put + = 3 ;
2016-12-30 23:25:58 +03:00
}
2020-01-06 23:01:48 +03:00
2020-05-06 21:57:27 +03:00
if ( strip < strip_count & & ( bytes_put + 3 ) > = strip_bytes [ strip ] ) {
2018-02-11 16:01:43 +03:00
// End of strip, pad if strip length is odd
if ( strip_bytes [ strip ] % 2 = = 1 ) {
2016-12-30 23:25:58 +03:00
putc ( 0 , tif_file ) ;
}
2018-02-11 16:01:43 +03:00
strip + + ;
bytes_put = 0 ;
2016-12-30 23:25:58 +03:00
}
}
2017-10-23 22:37:52 +03:00
/* Image File Directory */
2016-12-30 23:25:58 +03:00
ifd . entries = 14 ;
ifd . offset = 0 ;
2017-10-23 22:37:52 +03:00
2016-12-30 23:25:58 +03:00
ifd . new_subset . tag = 0xfe ;
ifd . new_subset . type = 4 ;
ifd . new_subset . count = 1 ;
ifd . new_subset . offset = 0 ;
2017-10-23 22:37:52 +03:00
2016-12-30 23:25:58 +03:00
ifd . image_width . tag = 0x0100 ;
ifd . image_width . type = 3 ; // SHORT
ifd . image_width . count = 1 ;
ifd . image_width . offset = symbol - > bitmap_width ;
2017-10-23 22:37:52 +03:00
2016-12-30 23:25:58 +03:00
ifd . image_length . tag = 0x0101 ;
ifd . image_length . type = 3 ; // SHORT
ifd . image_length . count = 1 ;
ifd . image_length . offset = symbol - > bitmap_height ;
2017-10-23 22:37:52 +03:00
2016-12-30 23:25:58 +03:00
ifd . bits_per_sample . tag = 0x0102 ;
ifd . bits_per_sample . type = 3 ; // SHORT
ifd . bits_per_sample . count = 3 ;
ifd . bits_per_sample . offset = free_memory ;
free_memory + = 6 ;
2017-10-23 22:37:52 +03:00
2016-12-30 23:25:58 +03:00
ifd . compression . tag = 0x0103 ;
ifd . compression . type = 3 ;
ifd . compression . count = 1 ;
ifd . compression . offset = 1 ; // Uncompressed
2017-10-23 22:37:52 +03:00
2016-12-30 23:25:58 +03:00
ifd . photometric . tag = 0x0106 ;
ifd . photometric . type = 3 ; // SHORT
ifd . photometric . count = 1 ;
ifd . photometric . offset = 2 ; // RGB Model
2017-10-23 22:37:52 +03:00
2016-12-30 23:25:58 +03:00
ifd . strip_offsets . tag = 0x0111 ;
ifd . strip_offsets . type = 4 ; // LONG
ifd . strip_offsets . count = strip_count ;
2020-05-06 21:57:27 +03:00
if ( strip_count = = 1 ) {
ifd . strip_offsets . offset = strip_offset [ 0 ] ;
} else {
ifd . strip_offsets . offset = free_memory ;
free_memory + = strip_count * 4 ;
}
2017-10-23 22:37:52 +03:00
2016-12-30 23:25:58 +03:00
ifd . samples_per_pixel . tag = 0x0115 ;
ifd . samples_per_pixel . type = 3 ;
ifd . samples_per_pixel . count = 1 ;
ifd . samples_per_pixel . offset = 3 ;
2017-10-23 22:37:52 +03:00
2016-12-30 23:25:58 +03:00
ifd . rows_per_strip . tag = 0x0116 ;
ifd . rows_per_strip . type = 4 ;
ifd . rows_per_strip . count = 1 ;
ifd . rows_per_strip . offset = rows_per_strip ;
2017-10-23 22:37:52 +03:00
2016-12-30 23:25:58 +03:00
ifd . strip_byte_counts . tag = 0x0117 ;
ifd . strip_byte_counts . type = 4 ;
ifd . strip_byte_counts . count = strip_count ;
2020-05-06 21:57:27 +03:00
if ( strip_count = = 1 ) {
ifd . strip_byte_counts . offset = strip_bytes [ 0 ] ;
} else {
ifd . strip_byte_counts . offset = free_memory ;
free_memory + = strip_count * 4 ;
}
2017-10-23 22:37:52 +03:00
2016-12-30 23:25:58 +03:00
ifd . x_resolution . tag = 0x011a ;
ifd . x_resolution . type = 5 ;
ifd . x_resolution . count = 1 ;
ifd . x_resolution . offset = free_memory ;
free_memory + = 8 ;
2017-10-23 22:37:52 +03:00
2016-12-30 23:25:58 +03:00
ifd . y_resolution . tag = 0x011b ;
ifd . y_resolution . type = 5 ;
ifd . y_resolution . count = 1 ;
ifd . y_resolution . offset = free_memory ;
2017-09-10 18:03:09 +03:00
// free_memory += 8;
2017-10-23 22:37:52 +03:00
2016-12-30 23:25:58 +03:00
ifd . planar_config . tag = 0x11c ;
ifd . planar_config . type = 3 ;
ifd . planar_config . count = 1 ;
ifd . planar_config . offset = 1 ;
2017-10-23 22:37:52 +03:00
2016-12-30 23:25:58 +03:00
ifd . resolution_unit . tag = 0x0128 ;
ifd . resolution_unit . type = 3 ;
ifd . resolution_unit . count = 1 ;
ifd . resolution_unit . offset = 2 ; // Inches
2017-10-23 22:37:52 +03:00
2016-12-30 23:25:58 +03:00
fwrite ( & ifd , sizeof ( tiff_ifd_t ) , 1 , tif_file ) ;
2017-10-23 22:37:52 +03:00
2016-12-30 23:25:58 +03:00
/* Bits per sample */
temp = 8 ;
fwrite ( & temp , 2 , 1 , tif_file ) ; // Red Bytes
fwrite ( & temp , 2 , 1 , tif_file ) ; // Green Bytes
fwrite ( & temp , 2 , 1 , tif_file ) ; // Blue Bytes
2017-10-23 22:37:52 +03:00
2020-05-06 21:57:27 +03:00
if ( strip_count ! = 1 ) {
/* Strip offsets */
for ( i = 0 ; i < strip_count ; i + + ) {
fwrite ( & strip_offset [ i ] , 4 , 1 , tif_file ) ;
}
2017-10-23 22:37:52 +03:00
2020-05-06 21:57:27 +03:00
/* Strip byte lengths */
for ( i = 0 ; i < strip_count ; i + + ) {
fwrite ( & strip_bytes [ i ] , 4 , 1 , tif_file ) ;
}
2016-12-30 23:25:58 +03:00
}
2017-10-23 22:37:52 +03:00
2016-12-30 23:25:58 +03:00
/* X Resolution */
temp32 = 72 ;
fwrite ( & temp32 , 4 , 1 , tif_file ) ;
temp32 = 1 ;
fwrite ( & temp32 , 4 , 1 , tif_file ) ;
2017-10-23 22:37:52 +03:00
2016-12-30 23:25:58 +03:00
/* Y Resolution */
temp32 = 72 ;
fwrite ( & temp32 , 4 , 1 , tif_file ) ;
temp32 = 1 ;
fwrite ( & temp32 , 4 , 1 , tif_file ) ;
2017-10-23 22:37:52 +03:00
2016-12-30 23:25:58 +03:00
if ( symbol - > output_options & BARCODE_STDOUT ) {
fflush ( tif_file ) ;
} else {
fclose ( tif_file ) ;
}
2017-10-23 22:37:52 +03:00
2016-12-30 23:25:58 +03:00
return 0 ;
2017-10-23 22:37:52 +03:00
}