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
2021-03-19 16:09:21 +03:00
Copyright ( C ) 2016 - 2021 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
2021-06-10 13:15:39 +03:00
# include <errno.h>
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>
2021-03-19 16:09:21 +03:00
# include <limits.h>
2016-12-30 23:25:58 +03:00
# include "common.h"
# include "tif.h"
2021-03-19 16:09:21 +03:00
# include "tif_lzw.h"
2016-12-30 23:25:58 +03:00
# ifdef _MSC_VER
# include <io.h>
# include <fcntl.h>
# include <malloc.h>
# endif
2021-03-19 16:09:21 +03:00
/* PhotometricInterpretation */
# define TIF_PMI_WHITEISZERO 0
# define TIF_PMI_BLACKISZERO 1
# define TIF_PMI_RGB 2
# define TIF_PMI_PALETTE_COLOR 3
# define TIF_PMI_SEPARATED 5 /* CMYK */
/* Compression */
# define TIF_NO_COMPRESSION 1
# define TIF_LZW 5
2021-10-31 00:00:31 +03:00
static void to_color_map ( const unsigned char rgb [ 4 ] , tiff_color_t * color_map_entry ) {
2021-03-19 16:09:21 +03:00
color_map_entry - > red = ( rgb [ 0 ] < < 8 ) | rgb [ 0 ] ;
color_map_entry - > green = ( rgb [ 1 ] < < 8 ) | rgb [ 1 ] ;
color_map_entry - > blue = ( rgb [ 2 ] < < 8 ) | rgb [ 2 ] ;
}
2021-10-31 00:00:31 +03:00
static void to_cmyk ( const unsigned char rgb [ 3 ] , const unsigned char alpha , unsigned char * cmyk ) {
2021-03-19 16:09:21 +03:00
unsigned char max = rgb [ 0 ] ;
if ( rgb [ 1 ] > max ) {
max = rgb [ 1 ] ;
}
if ( rgb [ 2 ] > max ) {
max = rgb [ 2 ] ;
}
cmyk [ 0 ] = max - rgb [ 0 ] ;
cmyk [ 1 ] = max - rgb [ 1 ] ;
cmyk [ 2 ] = max - rgb [ 2 ] ;
cmyk [ 3 ] = 0xff - max ;
cmyk [ 4 ] = alpha ;
}
2021-10-31 00:00:31 +03:00
static int is_big_endian ( void ) {
2021-03-23 18:37:18 +03:00
return ( * ( ( const uint16_t * ) " \x11 \x22 " ) = = 0x1122 ) ;
}
2020-11-01 21:32:55 +03:00
INTERNAL int tif_pixel_plot ( struct zint_symbol * symbol , unsigned char * pixelbuf ) {
2021-03-19 16:09:21 +03:00
unsigned char fg [ 4 ] , bg [ 4 ] ;
2016-12-30 23:25:58 +03:00
int i ;
2021-03-19 16:09:21 +03:00
int pmi ; /* PhotometricInterpretation */
2016-12-30 23:25:58 +03:00
int rows_per_strip , strip_count ;
2021-03-19 16:09:21 +03:00
int rows_last_strip ;
int bytes_per_strip ;
uint16_t bits_per_sample ;
int samples_per_pixel ;
int pixels_per_sample ;
unsigned char map [ 128 ] ;
2021-09-13 01:37:44 +03:00
tiff_color_t color_map [ 256 ] = { { 0 } } ;
2021-03-19 16:09:21 +03:00
unsigned char palette [ 32 ] [ 5 ] ;
int color_map_size = 0 ;
int extra_samples = 0 ;
uint32_t free_memory ;
2020-04-04 18:53:29 +03:00
int row , column , strip ;
2021-03-19 16:09:21 +03:00
int strip_row ;
2020-04-04 18:53:29 +03:00
unsigned int bytes_put ;
2021-03-19 16:09:21 +03:00
long total_bytes_put ;
2016-12-30 23:25:58 +03:00
FILE * tif_file ;
2021-03-19 16:09:21 +03:00
unsigned char * pb ;
int compression = TIF_NO_COMPRESSION ;
tif_lzw_state lzw_state ;
long file_pos ;
2021-07-26 17:29:05 +03:00
const int output_to_stdout = symbol - > output_options & BARCODE_STDOUT ;
2017-07-27 18:01:53 +03:00
# ifdef _MSC_VER
uint32_t * strip_offset ;
uint32_t * strip_bytes ;
2021-03-19 16:09:21 +03:00
unsigned char * strip_buf ;
2017-07-27 18:01:53 +03:00
# endif
2017-10-23 22:37:52 +03:00
2016-12-30 23:25:58 +03:00
tiff_header_t header ;
2021-03-19 16:09:21 +03:00
uint16_t entries = 0 ;
tiff_tag_t tags [ 20 ] ;
uint32_t offset = 0 ;
int update_offsets [ 20 ] ;
int offsets = 0 ;
int ifd_size ;
2016-12-30 23:25:58 +03:00
uint32_t temp32 ;
2017-10-23 22:37:52 +03:00
2021-03-19 16:09:21 +03:00
fg [ 0 ] = ( 16 * ctoi ( symbol - > fgcolour [ 0 ] ) ) + ctoi ( symbol - > fgcolour [ 1 ] ) ;
fg [ 1 ] = ( 16 * ctoi ( symbol - > fgcolour [ 2 ] ) ) + ctoi ( symbol - > fgcolour [ 3 ] ) ;
fg [ 2 ] = ( 16 * ctoi ( symbol - > fgcolour [ 4 ] ) ) + ctoi ( symbol - > fgcolour [ 5 ] ) ;
bg [ 0 ] = ( 16 * ctoi ( symbol - > bgcolour [ 0 ] ) ) + ctoi ( symbol - > bgcolour [ 1 ] ) ;
bg [ 1 ] = ( 16 * ctoi ( symbol - > bgcolour [ 2 ] ) ) + ctoi ( symbol - > bgcolour [ 3 ] ) ;
bg [ 2 ] = ( 16 * ctoi ( symbol - > bgcolour [ 4 ] ) ) + ctoi ( symbol - > bgcolour [ 5 ] ) ;
if ( strlen ( symbol - > fgcolour ) > 6 ) {
fg [ 3 ] = ( 16 * ctoi ( symbol - > fgcolour [ 6 ] ) ) + ctoi ( symbol - > fgcolour [ 7 ] ) ;
} else {
fg [ 3 ] = 0xff ;
}
if ( strlen ( symbol - > bgcolour ) > 6 ) {
bg [ 3 ] = ( 16 * ctoi ( symbol - > bgcolour [ 6 ] ) ) + ctoi ( symbol - > bgcolour [ 7 ] ) ;
} else {
bg [ 3 ] = 0xff ;
}
if ( symbol - > symbology = = BARCODE_ULTRA ) {
static const int ultra_chars [ 8 ] = { ' W ' , ' C ' , ' B ' , ' M ' , ' R ' , ' Y ' , ' G ' , ' K ' } ;
2021-10-31 00:00:31 +03:00
static const unsigned char ultra_rgbs [ 8 ] [ 3 ] = {
2021-03-19 16:09:21 +03:00
{ 0xff , 0xff , 0xff , } , /* White */
{ 0 , 0xff , 0xff , } , /* Cyan */
{ 0 , 0 , 0xff , } , /* Blue */
{ 0xff , 0 , 0xff , } , /* Magenta */
{ 0xff , 0 , 0 , } , /* Red */
{ 0xff , 0xff , 0 , } , /* Yellow */
{ 0 , 0xff , 0 , } , /* Green */
{ 0 , 0 , 0 , } , /* Black */
} ;
if ( symbol - > output_options & CMYK_COLOUR ) {
for ( i = 0 ; i < 8 ; i + + ) {
map [ ultra_chars [ i ] ] = i ;
to_cmyk ( ultra_rgbs [ i ] , fg [ 3 ] , palette [ i ] ) ;
}
map [ ' 0 ' ] = 8 ;
to_cmyk ( bg , bg [ 3 ] , palette [ 8 ] ) ;
map [ ' 1 ' ] = 9 ;
to_cmyk ( fg , fg [ 3 ] , palette [ 9 ] ) ;
pmi = TIF_PMI_SEPARATED ;
bits_per_sample = 8 ;
if ( fg [ 3 ] = = 0xff & & bg [ 3 ] = = 0xff ) { /* If no alpha */
samples_per_pixel = 4 ;
} else {
samples_per_pixel = 5 ;
extra_samples = 1 ; /* Associated alpha */
}
pixels_per_sample = 1 ;
} else {
for ( i = 0 ; i < 8 ; i + + ) {
map [ ultra_chars [ i ] ] = i ;
memcpy ( palette [ i ] , ultra_rgbs [ i ] , 3 ) ;
palette [ i ] [ 3 ] = fg [ 3 ] ;
}
map [ ' 0 ' ] = 8 ;
memcpy ( palette [ 8 ] , bg , 4 ) ;
map [ ' 1 ' ] = 9 ;
memcpy ( palette [ 9 ] , fg , 4 ) ;
if ( fg [ 3 ] = = 0xff & & bg [ 3 ] = = 0xff ) { /* If no alpha */
pmi = TIF_PMI_PALETTE_COLOR ;
for ( i = 0 ; i < 10 ; i + + ) {
to_color_map ( palette [ i ] , & color_map [ i ] ) ;
}
bits_per_sample = 4 ;
samples_per_pixel = 1 ;
pixels_per_sample = 2 ;
color_map_size = 16 ; /* 2**BitsPerSample */
} else {
pmi = TIF_PMI_RGB ;
bits_per_sample = 8 ;
samples_per_pixel = 4 ;
pixels_per_sample = 1 ;
extra_samples = 1 ; /* Associated alpha */
}
}
} else { /* fg/bg only */
if ( symbol - > output_options & CMYK_COLOUR ) {
map [ ' 0 ' ] = 0 ;
to_cmyk ( bg , bg [ 3 ] , palette [ 0 ] ) ;
map [ ' 1 ' ] = 1 ;
to_cmyk ( fg , fg [ 3 ] , palette [ 1 ] ) ;
pmi = TIF_PMI_SEPARATED ;
bits_per_sample = 8 ;
if ( fg [ 3 ] = = 0xff & & bg [ 3 ] = = 0xff ) { /* If no alpha */
samples_per_pixel = 4 ;
} else {
samples_per_pixel = 5 ;
extra_samples = 1 ; /* Associated alpha */
}
pixels_per_sample = 1 ;
} else if ( bg [ 0 ] = = 0xff & & bg [ 1 ] = = 0xff & & bg [ 2 ] = = 0xff & & bg [ 3 ] = = 0xff
& & fg [ 0 ] = = 0 & & fg [ 1 ] = = 0 & & fg [ 2 ] = = 0 & & fg [ 3 ] = = 0xff ) {
map [ ' 0 ' ] = 0 ;
map [ ' 1 ' ] = 1 ;
pmi = TIF_PMI_WHITEISZERO ;
bits_per_sample = 1 ;
samples_per_pixel = 1 ;
pixels_per_sample = 8 ;
} else if ( bg [ 0 ] = = 0 & & bg [ 1 ] = = 0 & & bg [ 2 ] = = 0 & & bg [ 3 ] = = 0xff
& & fg [ 0 ] = = 0xff & & fg [ 1 ] = = 0xff & & fg [ 2 ] = = 0xff & & fg [ 3 ] = = 0xff ) {
2021-07-26 17:29:05 +03:00
map [ ' 0 ' ] = 0 ;
map [ ' 1 ' ] = 1 ;
2021-03-19 16:09:21 +03:00
pmi = TIF_PMI_BLACKISZERO ;
bits_per_sample = 1 ;
samples_per_pixel = 1 ;
pixels_per_sample = 8 ;
} else {
map [ ' 0 ' ] = 0 ;
memcpy ( palette [ 0 ] , bg , 4 ) ;
map [ ' 1 ' ] = 1 ;
memcpy ( palette [ 1 ] , fg , 4 ) ;
pmi = TIF_PMI_PALETTE_COLOR ;
for ( i = 0 ; i < 2 ; i + + ) {
to_color_map ( palette [ i ] , & color_map [ i ] ) ;
}
if ( fg [ 3 ] = = 0xff & & bg [ 3 ] = = 0xff ) { /* If no alpha */
bits_per_sample = 4 ;
samples_per_pixel = 1 ;
pixels_per_sample = 2 ;
color_map_size = 16 ; /* 2**BitsPerSample */
} else {
bits_per_sample = 8 ;
samples_per_pixel = 2 ;
pixels_per_sample = 1 ;
color_map_size = 256 ; /* 2**BitsPerSample */
extra_samples = 1 ; /* Associated alpha */
}
}
}
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 . " */
2021-03-19 16:09:21 +03:00
rows_per_strip = ( 8192 * pixels_per_sample ) / ( symbol - > bitmap_width * samples_per_pixel ) ;
2016-12-30 23:25:58 +03:00
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 ) ;
2021-03-19 16:09:21 +03:00
if ( rows_per_strip > = symbol - > bitmap_height ) {
strip_count = 1 ;
rows_per_strip = rows_last_strip = symbol - > bitmap_height ;
} else {
strip_count = symbol - > bitmap_height / rows_per_strip ;
rows_last_strip = symbol - > bitmap_height % rows_per_strip ;
if ( rows_last_strip ! = 0 ) {
strip_count + + ;
}
if ( rows_per_strip > symbol - > bitmap_height ) {
rows_per_strip = rows_last_strip = symbol - > bitmap_height ;
}
2018-02-11 16:01:43 +03:00
}
2021-03-19 16:09:21 +03:00
assert ( strip_count > 0 ) ; /* Suppress clang-analyzer-core.UndefinedBinaryOperatorResult */
2020-01-06 23:01:48 +03:00
2020-05-06 21:57:27 +03:00
if ( symbol - > debug & ZINT_DEBUG_PRINT ) {
2021-03-22 00:20:16 +03:00
printf ( " TIFF (%dx%d) Strip Count %d, Rows Per Strip %d, Pixels Per Sample %d, Samples Per Pixel %d, PMI %d \n " ,
symbol - > bitmap_width , symbol - > bitmap_height , strip_count , rows_per_strip , pixels_per_sample ,
samples_per_pixel , pmi ) ;
2018-02-11 16:01:43 +03:00
}
2020-01-06 23:01:48 +03:00
2021-03-19 16:09:21 +03:00
bytes_per_strip = rows_per_strip * ( ( symbol - > bitmap_width + pixels_per_sample - 1 ) / pixels_per_sample )
* samples_per_pixel ;
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 ] ;
2021-03-19 16:09:21 +03:00
unsigned char strip_buf [ bytes_per_strip + 1 ] ;
2016-12-30 23:25:58 +03:00
# else
2021-03-19 16:09:21 +03:00
strip_offset = ( uint32_t * ) _alloca ( strip_count * sizeof ( uint32_t ) ) ;
strip_bytes = ( uint32_t * ) _alloca ( strip_count * sizeof ( uint32_t ) ) ;
strip_buf = ( unsigned char * ) _alloca ( bytes_per_strip + 1 ) ;
2016-12-30 23:25:58 +03:00
# endif
2021-03-19 16:09:21 +03:00
free_memory = sizeof ( tiff_header_t ) ;
2017-10-23 22:37:52 +03:00
2021-03-19 16:09:21 +03:00
for ( i = 0 ; i < strip_count ; i + + ) {
2016-12-30 23:25:58 +03:00
strip_offset [ i ] = free_memory ;
if ( i ! = ( strip_count - 1 ) ) {
2021-03-19 16:09:21 +03:00
strip_bytes [ i ] = bytes_per_strip ;
2016-12-30 23:25:58 +03:00
} else {
2021-03-19 16:09:21 +03:00
if ( rows_last_strip ) {
2021-07-26 17:29:05 +03:00
strip_bytes [ i ] = rows_last_strip
* ( ( symbol - > bitmap_width + pixels_per_sample - 1 ) / pixels_per_sample )
2021-03-19 16:09:21 +03:00
* samples_per_pixel ;
2016-12-31 13:44:09 +03:00
} else {
2021-03-19 16:09:21 +03:00
strip_bytes [ i ] = bytes_per_strip ;
2016-12-31 13:44:09 +03:00
}
2016-12-30 23:25:58 +03:00
}
free_memory + = strip_bytes [ i ] ;
2021-03-19 16:09:21 +03:00
}
if ( free_memory & 1 ) {
free_memory + + ; // IFD must be on word boundary
2016-12-30 23:25:58 +03:00
}
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 */
2021-07-26 17:29:05 +03:00
if ( output_to_stdout ) {
2016-12-30 23:25:58 +03:00
# ifdef _MSC_VER
if ( - 1 = = _setmode ( _fileno ( stdout ) , _O_BINARY ) ) {
2021-07-26 17:29:05 +03:00
sprintf ( symbol - > errtxt , " 671: Could not set stdout to binary (%d: %.30s) " , errno , strerror ( errno ) ) ;
2016-12-30 23:25:58 +03:00
return ZINT_ERROR_FILE_ACCESS ;
}
# endif
tif_file = stdout ;
} else {
2021-07-26 17:29:05 +03:00
if ( ! ( tif_file = fopen ( symbol - > outfile , " wb+ " ) ) ) { /* '+' as use fseek/ftell() */
sprintf ( symbol - > errtxt , " 672: Could not open output file (%d: %.30s) " , errno , strerror ( errno ) ) ;
2016-12-30 23:25:58 +03:00
return ZINT_ERROR_FILE_ACCESS ;
}
2021-03-19 16:09:21 +03:00
compression = TIF_LZW ;
tif_lzw_init ( & lzw_state ) ;
2016-12-30 23:25:58 +03:00
}
2017-10-23 22:37:52 +03:00
2016-12-30 23:25:58 +03:00
/* Header */
2021-03-23 18:37:18 +03:00
if ( is_big_endian ( ) ) {
header . byte_order = 0x4D4D ; // "MM" big-endian
} else {
header . byte_order = 0x4949 ; // "II" little-endian
}
2016-12-30 23:25:58 +03:00
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 ) ;
2021-03-19 16:09:21 +03:00
total_bytes_put = sizeof ( tiff_header_t ) ;
2020-01-06 23:01:48 +03:00
2016-12-30 23:25:58 +03:00
/* Pixel data */
2021-03-19 16:09:21 +03:00
pb = pixelbuf ;
2018-02-11 16:01:43 +03:00
strip = 0 ;
2021-03-19 16:09:21 +03:00
strip_row = 0 ;
2018-02-11 16:01:43 +03:00
bytes_put = 0 ;
2016-12-30 23:25:58 +03:00
for ( row = 0 ; row < symbol - > bitmap_height ; row + + ) {
2021-03-19 16:09:21 +03:00
if ( samples_per_pixel = = 1 ) {
if ( bits_per_sample = = 1 ) { /* WHITEISZERO or BLACKISZERO */
for ( column = 0 ; column < symbol - > bitmap_width ; column + = 8 ) {
unsigned char byte = 0 ;
for ( i = 0 ; i < 8 & & column + i < symbol - > bitmap_width ; i + + , pb + + ) {
byte | = map [ * pb ] < < ( 7 - i ) ;
}
strip_buf [ bytes_put + + ] = byte ;
}
} else { /* bits_per_sample == 4, PALETTE_COLOR with no alpha */
for ( column = 0 ; column < symbol - > bitmap_width ; column + = 2 ) {
unsigned char byte = map [ * pb + + ] < < 4 ;
if ( column + 1 < symbol - > bitmap_width ) {
byte | = map [ * pb + + ] ;
}
strip_buf [ bytes_put + + ] = byte ;
}
}
} else if ( samples_per_pixel = = 2 ) { /* PALETTE_COLOR with alpha */
for ( column = 0 ; column < symbol - > bitmap_width ; column + + ) {
2021-10-31 00:00:31 +03:00
const int idx = map [ * pb + + ] ;
2021-03-19 16:09:21 +03:00
strip_buf [ bytes_put + + ] = idx ;
strip_buf [ bytes_put + + ] = palette [ idx ] [ 3 ] ;
}
} else { /* samples_per_pixel >= 4, RGB with alpha (4) or CMYK with (5) or without (4) alpha */
for ( column = 0 ; column < symbol - > bitmap_width ; column + + ) {
2021-10-31 00:00:31 +03:00
const int idx = map [ * pb + + ] ;
2021-03-19 16:09:21 +03:00
memcpy ( & strip_buf [ bytes_put ] , & palette [ idx ] , samples_per_pixel ) ;
bytes_put + = samples_per_pixel ;
2016-12-30 23:25:58 +03:00
}
}
2020-01-06 23:01:48 +03:00
2021-03-19 16:09:21 +03:00
strip_row + + ;
if ( strip_row = = rows_per_strip | | ( strip = = strip_count - 1 & & strip_row = = rows_last_strip ) ) {
// End of strip
if ( compression = = TIF_LZW ) {
file_pos = ftell ( tif_file ) ;
if ( ! tif_lzw_encode ( & lzw_state , tif_file , strip_buf , bytes_put ) ) { /* Only fails if can't malloc */
tif_lzw_cleanup ( & lzw_state ) ;
fclose ( tif_file ) ; /* Only use LZW if not STDOUT, so ok to close */
strcpy ( symbol - > errtxt , " 673: Failed to malloc LZW hash table " ) ;
return ZINT_ERROR_MEMORY ;
}
bytes_put = ftell ( tif_file ) - file_pos ;
if ( bytes_put ! = strip_bytes [ strip ] ) {
2021-10-31 00:00:31 +03:00
const int diff = bytes_put - strip_bytes [ strip ] ;
2021-03-19 16:09:21 +03:00
strip_bytes [ strip ] = bytes_put ;
for ( i = strip + 1 ; i < strip_count ; i + + ) {
strip_offset [ i ] + = diff ;
}
}
} else {
fwrite ( strip_buf , 1 , bytes_put , tif_file ) ;
2016-12-30 23:25:58 +03:00
}
2018-02-11 16:01:43 +03:00
strip + + ;
2021-03-19 16:09:21 +03:00
total_bytes_put + = bytes_put ;
2018-02-11 16:01:43 +03:00
bytes_put = 0 ;
2021-03-19 16:09:21 +03:00
strip_row = 0 ;
/* Suppress clang-analyzer-core.UndefinedBinaryOperatorResult */
assert ( strip < strip_count | | row + 1 = = symbol - > bitmap_height ) ;
2016-12-30 23:25:58 +03:00
}
}
2017-10-23 22:37:52 +03:00
2021-03-19 16:09:21 +03:00
if ( total_bytes_put & 1 ) {
putc ( 0 , tif_file ) ; // IFD must be on word boundary
total_bytes_put + + ;
}
if ( compression = = TIF_LZW ) {
tif_lzw_cleanup ( & lzw_state ) ;
file_pos = ftell ( tif_file ) ;
fseek ( tif_file , 4 , SEEK_SET ) ;
free_memory = file_pos ;
fwrite ( & free_memory , 4 , 1 , tif_file ) ;
fseek ( tif_file , file_pos , SEEK_SET ) ;
}
2017-10-23 22:37:52 +03:00
/* Image File Directory */
2021-03-19 16:09:21 +03:00
tags [ entries ] . tag = 0x0100 ; // ImageWidth
tags [ entries ] . type = 3 ; // SHORT
tags [ entries ] . count = 1 ;
tags [ entries + + ] . offset = symbol - > bitmap_width ;
tags [ entries ] . tag = 0x0101 ; // ImageLength - number of rows
tags [ entries ] . type = 3 ; // SHORT
tags [ entries ] . count = 1 ;
tags [ entries + + ] . offset = symbol - > bitmap_height ;
if ( samples_per_pixel ! = 1 | | bits_per_sample ! = 1 ) {
tags [ entries ] . tag = 0x0102 ; // BitsPerSample
tags [ entries ] . type = 3 ; // SHORT
tags [ entries ] . count = samples_per_pixel ;
if ( samples_per_pixel = = 1 ) {
tags [ entries + + ] . offset = bits_per_sample ;
} else if ( samples_per_pixel = = 2 ) { /* 2 SHORTS fit into LONG offset so packed into offset */
tags [ entries + + ] . offset = ( bits_per_sample < < 16 ) | bits_per_sample ;
} else {
update_offsets [ offsets + + ] = entries ;
tags [ entries + + ] . offset = free_memory ;
free_memory + = samples_per_pixel * 2 ;
}
}
tags [ entries ] . tag = 0x0103 ; // Compression
tags [ entries ] . type = 3 ; // SHORT
tags [ entries ] . count = 1 ;
tags [ entries + + ] . offset = compression ;
tags [ entries ] . tag = 0x0106 ; // PhotometricInterpretation
tags [ entries ] . type = 3 ; // SHORT
tags [ entries ] . count = 1 ;
tags [ entries + + ] . offset = pmi ;
tags [ entries ] . tag = 0x0111 ; // StripOffsets
tags [ entries ] . type = 4 ; // LONG
tags [ entries ] . count = strip_count ;
2020-05-06 21:57:27 +03:00
if ( strip_count = = 1 ) {
2021-03-19 16:09:21 +03:00
tags [ entries + + ] . offset = strip_offset [ 0 ] ;
2020-05-06 21:57:27 +03:00
} else {
2021-03-19 16:09:21 +03:00
update_offsets [ offsets + + ] = entries ;
tags [ entries + + ] . offset = free_memory ;
2020-05-06 21:57:27 +03:00
free_memory + = strip_count * 4 ;
}
2017-10-23 22:37:52 +03:00
2021-03-19 16:09:21 +03:00
if ( samples_per_pixel > 1 ) {
tags [ entries ] . tag = 0x0115 ; // SamplesPerPixel
tags [ entries ] . type = 3 ; // SHORT
tags [ entries ] . count = 1 ;
tags [ entries + + ] . offset = samples_per_pixel ;
}
2017-10-23 22:37:52 +03:00
2021-03-19 16:09:21 +03:00
tags [ entries ] . tag = 0x0116 ; // RowsPerStrip
tags [ entries ] . type = 4 ; // LONG
tags [ entries ] . count = 1 ;
tags [ entries + + ] . offset = rows_per_strip ;
2017-10-23 22:37:52 +03:00
2021-03-19 16:09:21 +03:00
tags [ entries ] . tag = 0x0117 ; // StripByteCounts
tags [ entries ] . type = 4 ; // LONG
tags [ entries ] . count = strip_count ;
2020-05-06 21:57:27 +03:00
if ( strip_count = = 1 ) {
2021-03-19 16:09:21 +03:00
tags [ entries + + ] . offset = strip_bytes [ 0 ] ;
2020-05-06 21:57:27 +03:00
} else {
2021-03-19 16:09:21 +03:00
update_offsets [ offsets + + ] = entries ;
tags [ entries + + ] . offset = free_memory ;
2020-05-06 21:57:27 +03:00
free_memory + = strip_count * 4 ;
}
2017-10-23 22:37:52 +03:00
2021-03-19 16:09:21 +03:00
tags [ entries ] . tag = 0x011a ; // XResolution
tags [ entries ] . type = 5 ; // RATIONAL
tags [ entries ] . count = 1 ;
update_offsets [ offsets + + ] = entries ;
tags [ entries + + ] . offset = free_memory ;
2016-12-30 23:25:58 +03:00
free_memory + = 8 ;
2017-10-23 22:37:52 +03:00
2021-03-19 16:09:21 +03:00
tags [ entries ] . tag = 0x011b ; // YResolution
tags [ entries ] . type = 5 ; // RATIONAL
tags [ entries ] . count = 1 ;
update_offsets [ offsets + + ] = entries ;
tags [ entries + + ] . offset = free_memory ;
free_memory + = 8 ;
2017-10-23 22:37:52 +03:00
2021-03-19 16:09:21 +03:00
tags [ entries ] . tag = 0x0128 ; // ResolutionUnit
tags [ entries ] . type = 3 ; // SHORT
tags [ entries ] . count = 1 ;
tags [ entries + + ] . offset = 2 ; // Inches
if ( color_map_size ) {
tags [ entries ] . tag = 0x0140 ; // ColorMap
tags [ entries ] . type = 3 ; // SHORT
tags [ entries ] . count = color_map_size * 3 ;
update_offsets [ offsets + + ] = entries ;
tags [ entries + + ] . offset = free_memory ;
//free_memory += color_map_size * 3 * 2; /* Unnecessary as long as last use */
}
2017-10-23 22:37:52 +03:00
2021-03-19 16:09:21 +03:00
if ( extra_samples ) {
tags [ entries ] . tag = 0x0152 ; // ExtraSamples
tags [ entries ] . type = 3 ; // SHORT
tags [ entries ] . count = 1 ;
tags [ entries + + ] . offset = extra_samples ;
}
2017-10-23 22:37:52 +03:00
2021-03-19 16:09:21 +03:00
ifd_size = sizeof ( entries ) + sizeof ( tiff_tag_t ) * entries + sizeof ( offset ) ;
for ( i = 0 ; i < offsets ; i + + ) {
tags [ update_offsets [ i ] ] . offset + = ifd_size ;
}
2017-10-23 22:37:52 +03:00
2021-03-19 16:09:21 +03:00
fwrite ( & entries , sizeof ( entries ) , 1 , tif_file ) ;
fwrite ( & tags , sizeof ( tiff_tag_t ) , entries , tif_file ) ;
fwrite ( & offset , sizeof ( offset ) , 1 , tif_file ) ;
total_bytes_put + = ifd_size ;
if ( samples_per_pixel > 2 ) {
for ( i = 0 ; i < samples_per_pixel ; i + + ) {
fwrite ( & bits_per_sample , sizeof ( bits_per_sample ) , 1 , tif_file ) ;
}
total_bytes_put + = sizeof ( bits_per_sample ) * samples_per_pixel ;
}
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 ) ;
}
2021-03-19 16:09:21 +03:00
total_bytes_put + = strip_count * 8 ;
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 ) ;
2021-03-19 16:09:21 +03:00
total_bytes_put + = 8 ;
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 ) ;
2021-03-19 16:09:21 +03:00
total_bytes_put + = 8 ;
if ( color_map_size ) {
for ( i = 0 ; i < color_map_size ; i + + ) {
fwrite ( & color_map [ i ] . red , 2 , 1 , tif_file ) ;
}
for ( i = 0 ; i < color_map_size ; i + + ) {
fwrite ( & color_map [ i ] . green , 2 , 1 , tif_file ) ;
}
for ( i = 0 ; i < color_map_size ; i + + ) {
fwrite ( & color_map [ i ] . blue , 2 , 1 , tif_file ) ;
}
total_bytes_put + = 6 * color_map_size ;
}
2017-10-23 22:37:52 +03:00
2021-07-26 17:29:05 +03:00
if ( output_to_stdout ) {
2016-12-30 23:25:58 +03:00
fflush ( tif_file ) ;
} else {
2021-03-19 16:09:21 +03:00
if ( ftell ( tif_file ) ! = total_bytes_put ) {
fclose ( tif_file ) ;
strcpy ( symbol - > errtxt , " 674: Failed to write all output " ) ;
return ZINT_ERROR_FILE_WRITE ;
}
2016-12-30 23:25:58 +03:00
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
}