2009-11-22 18:38:15 +03:00
/* library.c - external functions of libzint
libzint - the open source barcode library
2021-01-11 21:11:41 +03:00
Copyright ( C ) 2009 - 2021 Robin Stuart < rstuart114 @ gmail . com >
2009-11-22 18:38:15 +03:00
2013-05-16 21:26:38 +04: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 .
2013-05-16 21:26:38 +04: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 .
2013-05-16 21:26:38 +04: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 .
2013-05-16 21:26:38 +04: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
2013-05-16 21:26:38 +04:00
SUCH DAMAGE .
2016-02-20 13:50:15 +03:00
*/
2019-10-17 12:06:21 +03:00
/* vim: set ts=4 sw=4 et : */
2009-11-22 18:38:15 +03:00
# include <stdio.h>
# include <errno.h>
2020-11-01 21:32:55 +03:00
# include <limits.h>
2009-11-22 18:38:15 +03:00
# ifdef _MSC_VER
2017-10-23 22:37:52 +03:00
# include <malloc.h>
2009-11-22 18:38:15 +03:00
# endif
# include "common.h"
2021-01-11 21:11:41 +03:00
# include "eci.h"
2009-11-22 18:38:15 +03:00
# include "gs1.h"
2021-05-15 14:23:46 +03:00
# include "zfiletypes.h"
2009-11-22 18:38:15 +03:00
2020-06-04 20:45:25 +03:00
# define TECHNETIUM "0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZ-. $ / +%"
2009-11-22 18:38:15 +03:00
2020-12-21 22:30:07 +03:00
/* It's assumed that int is at least 32 bits, the following will compile-time fail if not
Add Structured Append support for AZTEC, CODEONE, DATAMATRIX, DOTCODE,
GRIDMATRIX, MAXICODE, MICROPDF417, PDF417, QRCODE, ULTRA
DOTCODE: use pre-calculated generator poly coeffs in Reed-Solomon for
performance improvement
PDF417/MICROPDF417: use common routine pdf417_initial()
GUI: code lines <= 118, shorthand widget_obj(),
shorten calling upcean_addon_gap(), upcean_guard_descent()
various backend: var name debug -> debug_print
2021-09-28 23:42:44 +03:00
* https : //stackoverflow.com/a/1980056 */
2020-12-21 22:30:07 +03:00
typedef int static_assert_int_at_least_32bits [ CHAR_BIT ! = 8 | | sizeof ( int ) < 4 ? - 1 : 1 ] ;
2021-07-06 21:53:31 +03:00
/* Create and initialize a symbol structure */
2016-02-20 13:50:15 +03:00
struct zint_symbol * ZBarcode_Create ( ) {
struct zint_symbol * symbol ;
2021-05-15 14:23:46 +03:00
symbol = ( struct zint_symbol * ) malloc ( sizeof ( * symbol ) ) ;
2016-02-20 13:50:15 +03:00
if ( ! symbol ) return NULL ;
2021-05-15 14:23:46 +03:00
memset ( symbol , 0 , sizeof ( * symbol ) ) ;
2020-11-01 21:32:55 +03:00
2016-02-20 13:50:15 +03:00
symbol - > symbology = BARCODE_CODE128 ;
2021-09-22 02:04:15 +03:00
symbol - > scale = 1.0f ;
2016-02-20 13:50:15 +03:00
strcpy ( symbol - > fgcolour , " 000000 " ) ;
2020-08-03 00:26:39 +03:00
symbol - > fgcolor = & symbol - > fgcolour [ 0 ] ;
2016-02-20 13:50:15 +03:00
strcpy ( symbol - > bgcolour , " ffffff " ) ;
2020-08-03 00:26:39 +03:00
symbol - > bgcolor = & symbol - > bgcolour [ 0 ] ;
2020-12-21 22:30:07 +03:00
# ifdef NO_PNG
strcpy ( symbol - > outfile , " out.gif " ) ;
# else
2017-10-21 14:45:50 +03:00
strcpy ( symbol - > outfile , " out.png " ) ;
2020-12-21 22:30:07 +03:00
# endif
2016-02-20 13:50:15 +03:00
symbol - > option_1 = - 1 ;
symbol - > show_hrt = 1 ; // Show human readable text
2018-02-11 12:55:28 +03:00
symbol - > fontsize = 8 ;
2016-02-20 13:50:15 +03:00
symbol - > input_mode = DATA_MODE ;
2019-10-06 19:39:54 +03:00
symbol - > eci = 0 ; // Default 0 uses ECI 3
2020-09-30 14:19:12 +03:00
symbol - > dot_size = 4.0f / 5.0f ;
2021-09-22 02:04:15 +03:00
symbol - > guard_descent = 5.0f ;
2020-08-22 13:09:57 +03:00
symbol - > warn_level = WARN_DEFAULT ;
2021-09-22 02:04:15 +03:00
symbol - > bitmap = NULL ;
symbol - > alphamap = NULL ;
symbol - > vector = NULL ;
2020-11-01 21:32:55 +03:00
2016-02-20 13:50:15 +03:00
return symbol ;
2009-11-22 18:38:15 +03:00
}
2019-12-19 03:37:55 +03:00
INTERNAL void vector_free ( struct zint_symbol * symbol ) ; /* Free vector structures */
2017-10-16 10:49:44 +03:00
2021-07-06 21:53:31 +03:00
/* Free any output buffers that may have been created and initialize output fields */
2016-02-20 13:50:15 +03:00
void ZBarcode_Clear ( struct zint_symbol * symbol ) {
int i , j ;
2020-09-30 14:19:12 +03:00
if ( ! symbol ) return ;
2016-02-20 13:50:15 +03:00
for ( i = 0 ; i < symbol - > rows ; i + + ) {
for ( j = 0 ; j < symbol - > width ; j + + ) {
unset_module ( symbol , i , j ) ;
}
}
symbol - > rows = 0 ;
symbol - > width = 0 ;
2021-07-06 21:53:31 +03:00
memset ( symbol - > row_height , 0 , sizeof ( symbol - > row_height ) ) ;
2016-10-27 22:55:26 +03:00
memset ( symbol - > text , 0 , sizeof ( symbol - > text ) ) ;
2016-02-20 13:50:15 +03:00
symbol - > errtxt [ 0 ] = ' \0 ' ;
if ( symbol - > bitmap ! = NULL ) {
free ( symbol - > bitmap ) ;
symbol - > bitmap = NULL ;
}
2020-08-03 14:13:05 +03:00
if ( symbol - > alphamap ! = NULL ) {
free ( symbol - > alphamap ) ;
symbol - > alphamap = NULL ;
}
2016-02-20 13:50:15 +03:00
symbol - > bitmap_width = 0 ;
symbol - > bitmap_height = 0 ;
2021-07-06 21:53:31 +03:00
symbol - > bitmap_byte_length = 0 ;
2017-10-16 10:49:44 +03:00
// If there is a rendered version, ensure its memory is released
2018-06-10 11:16:18 +03:00
vector_free ( symbol ) ;
2009-12-02 12:09:45 +03:00
}
2009-11-22 18:38:15 +03:00
2021-07-06 21:53:31 +03:00
/* Free a symbol structure, including any output buffers */
2016-02-20 13:50:15 +03:00
void ZBarcode_Delete ( struct zint_symbol * symbol ) {
2020-09-30 14:19:12 +03:00
if ( ! symbol ) return ;
2016-02-20 13:50:15 +03:00
if ( symbol - > bitmap ! = NULL )
free ( symbol - > bitmap ) ;
2020-08-03 14:13:05 +03:00
if ( symbol - > alphamap ! = NULL )
free ( symbol - > alphamap ) ;
2016-02-20 13:50:15 +03:00
// If there is a rendered version, ensure its memory is released
2018-06-10 11:16:18 +03:00
vector_free ( symbol ) ;
2016-09-06 00:06:50 +03:00
2016-02-20 13:50:15 +03:00
free ( symbol ) ;
2009-11-22 18:38:15 +03:00
}
2019-12-19 03:37:55 +03:00
INTERNAL int eanx ( struct zint_symbol * symbol , unsigned char source [ ] , int length ) ; /* EAN system barcodes */
2020-12-23 13:57:24 +03:00
INTERNAL int c39 ( struct zint_symbol * symbol , unsigned char source [ ] , int length ) ; /* Code 3 from 9 (or Code 39) */
2021-06-19 15:11:23 +03:00
/* Pharmazentral Nummer (PZN) */
INTERNAL int pharmazentral ( struct zint_symbol * symbol , unsigned char source [ ] , int length ) ;
/* Extended Code 3 from 9 (or Code 39+) */
INTERNAL int ec39 ( struct zint_symbol * symbol , unsigned char source [ ] , int length ) ;
/* Codabar - a simple substitution cipher */
INTERNAL int codabar ( struct zint_symbol * symbol , unsigned char source [ ] , int length ) ;
/* Code 2 of 5 Standard (& Matrix) */
INTERNAL int matrix_two_of_five ( struct zint_symbol * symbol , unsigned char source [ ] , int length ) ;
/* Code 2 of 5 Industrial */
INTERNAL int industrial_two_of_five ( struct zint_symbol * symbol , unsigned char source [ ] , int length ) ;
2019-12-19 03:37:55 +03:00
INTERNAL int iata_two_of_five ( struct zint_symbol * symbol , unsigned char source [ ] , int length ) ; /* Code 2 of 5 IATA */
2021-06-19 15:11:23 +03:00
/* Code 2 of 5 Interleaved */
INTERNAL int interleaved_two_of_five ( struct zint_symbol * symbol , unsigned char source [ ] , int length ) ;
/* Code 2 of 5 Data Logic */
INTERNAL int logic_two_of_five ( struct zint_symbol * symbol , unsigned char source [ ] , int length ) ;
2019-12-19 03:37:55 +03:00
INTERNAL int itf14 ( struct zint_symbol * symbol , unsigned char source [ ] , int length ) ; /* ITF-14 */
INTERNAL int dpleit ( struct zint_symbol * symbol , unsigned char source [ ] , int length ) ; /* Deutsche Post Leitcode */
INTERNAL int dpident ( struct zint_symbol * symbol , unsigned char source [ ] , int length ) ; /* Deutsche Post Identcode */
2021-06-19 15:11:23 +03:00
/* Code 93 - a re-working of Code 39+, generates 2 check digits */
INTERNAL int c93 ( struct zint_symbol * symbol , unsigned char source [ ] , int length ) ;
2020-12-21 22:30:07 +03:00
INTERNAL int code_128 ( struct zint_symbol * symbol , unsigned char source [ ] , int length ) ; /* Code 128 and NVE-18 */
INTERNAL int ean_128 ( struct zint_symbol * symbol , unsigned char source [ ] , int length ) ; /* EAN-128 (GS1-128) */
2019-12-19 03:37:55 +03:00
INTERNAL int code_11 ( struct zint_symbol * symbol , unsigned char source [ ] , int length ) ; /* Code 11 */
INTERNAL int msi_handle ( struct zint_symbol * symbol , unsigned char source [ ] , int length ) ; /* MSI Plessey */
2020-12-23 13:57:24 +03:00
INTERNAL int telepen ( struct zint_symbol * symbol , unsigned char source [ ] , int length ) ; /* Telepen ASCII */
INTERNAL int telepen_num ( struct zint_symbol * symbol , unsigned char source [ ] , int length ) ; /* Telepen Numeric */
INTERNAL int plessey ( struct zint_symbol * symbol , unsigned char source [ ] , int length ) ; /* Plessey Code */
2019-12-19 03:37:55 +03:00
INTERNAL int pharma_one ( struct zint_symbol * symbol , unsigned char source [ ] , int length ) ; /* Pharmacode One Track */
INTERNAL int flattermarken ( struct zint_symbol * symbol , unsigned char source [ ] , int length ) ; /* Flattermarken */
INTERNAL int fim ( struct zint_symbol * symbol , unsigned char source [ ] , int length ) ; /* Facing Identification Mark */
INTERNAL int pharma_two ( struct zint_symbol * symbol , unsigned char source [ ] , int length ) ; /* Pharmacode Two Track */
INTERNAL int post_plot ( struct zint_symbol * symbol , unsigned char source [ ] , int length ) ; /* Postnet */
INTERNAL int planet_plot ( struct zint_symbol * symbol , unsigned char source [ ] , int length ) ; /* PLANET */
2021-06-19 15:11:23 +03:00
/* Intelligent Mail (aka USPS OneCode) */
INTERNAL int imail ( struct zint_symbol * symbol , unsigned char source [ ] , int length ) ;
2019-12-19 03:37:55 +03:00
INTERNAL int royal_plot ( struct zint_symbol * symbol , unsigned char source [ ] , int length ) ; /* RM4SCC */
2021-06-19 15:11:23 +03:00
/* Australia Post 4-state */
INTERNAL int australia_post ( struct zint_symbol * symbol , unsigned char source [ ] , int length ) ;
2020-12-23 13:57:24 +03:00
INTERNAL int code16k ( struct zint_symbol * symbol , unsigned char source [ ] , int length ) ; /* Code 16k */
2020-11-01 21:32:55 +03:00
INTERNAL int pdf417enc ( struct zint_symbol * symbol , unsigned char source [ ] , int length ) ; /* PDF417 */
INTERNAL int micro_pdf417 ( struct zint_symbol * symbol , unsigned char chaine [ ] , int length ) ; /* Micro PDF417 */
2019-12-19 03:37:55 +03:00
INTERNAL int maxicode ( struct zint_symbol * symbol , unsigned char source [ ] , int length ) ; /* Maxicode */
INTERNAL int rss14 ( struct zint_symbol * symbol , unsigned char source [ ] , int length ) ; /* RSS-14 */
INTERNAL int rsslimited ( struct zint_symbol * symbol , unsigned char source [ ] , int length ) ; /* RSS Limited */
INTERNAL int rssexpanded ( struct zint_symbol * symbol , unsigned char source [ ] , int length ) ; /* RSS Expanded */
INTERNAL int composite ( struct zint_symbol * symbol , unsigned char source [ ] , int length ) ; /* Composite Symbology */
INTERNAL int kix_code ( struct zint_symbol * symbol , unsigned char source [ ] , int length ) ; /* TNT KIX Code */
2020-11-27 15:54:44 +03:00
INTERNAL int aztec ( struct zint_symbol * symbol , unsigned char source [ ] , int length ) ; /* Aztec Code */
2019-12-19 03:37:55 +03:00
INTERNAL int code32 ( struct zint_symbol * symbol , unsigned char source [ ] , int length ) ; /* Italian Pharmacode */
INTERNAL int daft_code ( struct zint_symbol * symbol , unsigned char source [ ] , int length ) ; /* DAFT Code */
INTERNAL int ean_14 ( struct zint_symbol * symbol , unsigned char source [ ] , int length ) ; /* EAN-14 */
INTERNAL int nve_18 ( struct zint_symbol * symbol , unsigned char source [ ] , int length ) ; /* NVE-18 */
2020-11-27 15:54:44 +03:00
INTERNAL int microqr ( struct zint_symbol * symbol , unsigned char source [ ] , int length ) ; /* Micro QR Code */
2019-12-19 03:37:55 +03:00
INTERNAL int aztec_runes ( struct zint_symbol * symbol , unsigned char source [ ] , int length ) ; /* Aztec Runes */
INTERNAL int korea_post ( struct zint_symbol * symbol , unsigned char source [ ] , int length ) ; /* Korea Post */
INTERNAL int japan_post ( struct zint_symbol * symbol , unsigned char source [ ] , int length ) ; /* Japanese Post */
2020-12-23 13:57:24 +03:00
INTERNAL int code_49 ( struct zint_symbol * symbol , unsigned char source [ ] , int length ) ; /* Code 49 */
2019-12-19 03:37:55 +03:00
INTERNAL int channel_code ( struct zint_symbol * symbol , unsigned char source [ ] , int length ) ; /* Channel Code */
INTERNAL int code_one ( struct zint_symbol * symbol , unsigned char source [ ] , int length ) ; /* Code One */
2020-11-27 15:54:44 +03:00
INTERNAL int grid_matrix ( struct zint_symbol * symbol , unsigned char source [ ] , int length ) ; /* Grid Matrix */
2020-12-23 13:57:24 +03:00
INTERNAL int han_xin ( struct zint_symbol * symbol , unsigned char source [ ] , int length ) ; /* Han Xin */
INTERNAL int dotcode ( struct zint_symbol * symbol , unsigned char source [ ] , int length ) ; /* DotCode */
2020-12-21 22:30:07 +03:00
INTERNAL int codablock ( struct zint_symbol * symbol , unsigned char source [ ] , int length ) ; /* Codablock */
2020-11-27 15:54:44 +03:00
INTERNAL int upnqr ( struct zint_symbol * symbol , unsigned char source [ ] , int length ) ; /* UPNQR */
INTERNAL int qr_code ( struct zint_symbol * symbol , unsigned char source [ ] , int length ) ; /* QR Code */
2020-12-23 13:57:24 +03:00
INTERNAL int dmatrix ( struct zint_symbol * symbol , unsigned char source [ ] , int length ) ; /* Data Matrix (IEC16022) */
2021-06-19 15:11:23 +03:00
/* VIN Code (Vehicle Identification Number) */
INTERNAL int vin ( struct zint_symbol * symbol , unsigned char source [ ] , int length ) ;
/* Royal Mail 4-state Mailmark */
INTERNAL int mailmark ( struct zint_symbol * symbol , unsigned char source [ ] , int length ) ;
2020-12-23 13:57:24 +03:00
INTERNAL int ultracode ( struct zint_symbol * symbol , unsigned char source [ ] , int length ) ; /* Ultracode */
INTERNAL int rmqr ( struct zint_symbol * symbol , unsigned char source [ ] , int length ) ; /* rMQR */
2020-09-03 18:36:57 +03:00
INTERNAL int dpd_parcel ( struct zint_symbol * symbol , unsigned char source [ ] , int length ) ; /* DPD Code */
2019-12-19 03:37:55 +03:00
INTERNAL int plot_raster ( struct zint_symbol * symbol , int rotate_angle , int file_type ) ; /* Plot to PNG/BMP/PCX */
INTERNAL int plot_vector ( struct zint_symbol * symbol , int rotate_angle , int file_type ) ; /* Plot to EPS/EMF/SVG */
2021-06-19 15:11:23 +03:00
/* Prefix error message with Error/Warning */
2021-07-06 21:53:31 +03:00
STATIC_UNLESS_ZINT_TEST int error_tag ( struct zint_symbol * symbol , int error_number , const char * error_string ) {
2012-12-31 17:41:59 +04:00
2016-02-20 13:50:15 +03:00
if ( error_number ! = 0 ) {
2021-07-06 21:53:31 +03:00
static const char * error_fmt = " Error %.93s " ; /* Truncate if too long */
static const char * warn_fmt = " Warning %.91s " ; /* Truncate if too long */
const char * fmt = error_number > = ZINT_ERROR ? error_fmt : warn_fmt ;
2017-09-10 18:03:09 +03:00
char error_buffer [ 100 ] ;
2021-07-06 21:53:31 +03:00
2021-08-31 20:34:29 +03:00
if ( error_number < ZINT_ERROR & & symbol - > warn_level = = WARN_FAIL_ALL ) {
2021-07-06 21:53:31 +03:00
/* Convert to error equivalent */
if ( error_number = = ZINT_WARN_NONCOMPLIANT ) {
error_number = ZINT_ERROR_NONCOMPLIANT ;
} else if ( error_number = = ZINT_WARN_USES_ECI ) {
error_number = ZINT_ERROR_USES_ECI ;
} else { /* ZINT_WARN_INVALID_OPTION */
error_number = ZINT_ERROR_INVALID_OPTION ;
}
fmt = error_fmt ;
}
sprintf ( error_buffer , fmt , error_string ? error_string : symbol - > errtxt ) ;
strcpy ( symbol - > errtxt , error_buffer ) ;
2016-02-20 13:50:15 +03:00
}
2021-01-12 22:51:54 +03:00
return error_number ;
2009-11-22 18:38:15 +03:00
}
2016-02-20 13:50:15 +03:00
/* Output a hexadecimal representation of the rendered symbol */
2019-12-19 03:37:55 +03:00
static int dump_plot ( struct zint_symbol * symbol ) {
2016-02-20 13:50:15 +03:00
FILE * f ;
int i , r ;
char hex [ ] = { ' 0 ' , ' 1 ' , ' 2 ' , ' 3 ' , ' 4 ' , ' 5 ' , ' 6 ' , ' 7 ' , ' 8 ' ,
' 9 ' , ' A ' , ' B ' , ' C ' , ' D ' , ' E ' , ' F ' } ;
int space = 0 ;
2016-09-06 00:06:50 +03:00
2016-02-20 13:50:15 +03:00
if ( symbol - > output_options & BARCODE_STDOUT ) {
f = stdout ;
} else {
f = fopen ( symbol - > outfile , " w " ) ;
if ( ! f ) {
2017-07-27 18:01:53 +03:00
strcpy ( symbol - > errtxt , " 201: Could not open output file " ) ;
2016-02-20 13:50:15 +03:00
return ZINT_ERROR_FILE_ACCESS ;
}
}
for ( r = 0 ; r < symbol - > rows ; r + + ) {
2017-09-10 18:03:09 +03:00
int byt = 0 ;
2016-02-20 13:50:15 +03:00
for ( i = 0 ; i < symbol - > width ; i + + ) {
byt = byt < < 1 ;
2020-11-01 21:32:55 +03:00
if ( symbol - > symbology = = BARCODE_ULTRA ) {
if ( module_colour_is_set ( symbol , r , i ) ) {
byt + = 1 ;
}
} else {
if ( module_is_set ( symbol , r , i ) ) {
byt + = 1 ;
}
2016-02-20 13:50:15 +03:00
}
if ( ( ( i + 1 ) % 4 ) = = 0 ) {
fputc ( hex [ byt ] , f ) ;
space + + ;
2016-02-19 02:23:31 +03:00
byt = 0 ;
2016-02-20 13:50:15 +03:00
}
2020-08-05 00:22:26 +03:00
if ( space = = 2 & & i + 1 < symbol - > width ) {
2016-02-20 13:50:15 +03:00
fputc ( ' ' , f ) ;
2016-02-19 02:23:31 +03:00
space = 0 ;
2016-02-20 13:50:15 +03:00
}
}
2010-09-12 17:52:44 +04:00
2016-02-20 13:50:15 +03:00
if ( ( symbol - > width % 4 ) ! = 0 ) {
byt = byt < < ( 4 - ( symbol - > width % 4 ) ) ;
fputc ( hex [ byt ] , f ) ;
}
fputs ( " \n " , f ) ;
space = 0 ;
}
2010-09-12 17:52:44 +04:00
2016-02-20 13:50:15 +03:00
if ( symbol - > output_options & BARCODE_STDOUT ) {
fflush ( f ) ;
} else {
fclose ( f ) ;
}
return 0 ;
2010-09-12 17:52:44 +04:00
}
2016-02-20 13:50:15 +03:00
/* Process health industry bar code data */
2020-11-27 15:54:44 +03:00
static int hibc ( struct zint_symbol * symbol , unsigned char source [ ] , int length ) {
int i ;
2017-05-29 12:43:47 +03:00
int counter , error_number ;
2020-11-01 21:32:55 +03:00
char to_process [ 113 ] , check_digit ;
2016-02-20 13:50:15 +03:00
2016-10-26 20:34:21 +03:00
/* without "+" and check: max 110 characters in HIBC 2.6 */
if ( length > 110 ) {
2021-07-06 21:53:31 +03:00
strcpy ( symbol - > errtxt , " 202: Data too long for HIBC LIC (110 character maximum) " ) ;
2016-02-20 13:50:15 +03:00
return ZINT_ERROR_TOO_LONG ;
}
to_upper ( source ) ;
error_number = is_sane ( TECHNETIUM , source , length ) ;
if ( error_number = = ZINT_ERROR_INVALID_DATA ) {
2021-07-06 21:53:31 +03:00
strcpy ( symbol - > errtxt , " 203: Invalid character in data (alphanumerics, space and \" -.$/+% \" only) " ) ;
2016-02-20 13:50:15 +03:00
return error_number ;
}
counter = 41 ;
for ( i = 0 ; i < length ; i + + ) {
counter + = posn ( TECHNETIUM , source [ i ] ) ;
}
counter = counter % 43 ;
if ( counter < 10 ) {
check_digit = itoc ( counter ) ;
} else {
if ( counter < 36 ) {
check_digit = ( counter - 10 ) + ' A ' ;
} else {
switch ( counter ) {
case 36 : check_digit = ' - ' ;
break ;
case 37 : check_digit = ' . ' ;
break ;
case 38 : check_digit = ' ' ;
break ;
case 39 : check_digit = ' $ ' ;
break ;
case 40 : check_digit = ' / ' ;
break ;
case 41 : check_digit = ' + ' ;
break ;
case 42 : check_digit = ' % ' ;
break ;
default : check_digit = ' ' ;
break ; /* Keep compiler happy */
}
}
}
2020-11-01 21:32:55 +03:00
to_process [ 0 ] = ' + ' ;
memcpy ( to_process + 1 , source , length ) ;
to_process [ length + 1 ] = check_digit ;
length + = 2 ;
to_process [ length ] = ' \0 ' ;
2016-02-20 13:50:15 +03:00
switch ( symbol - > symbology ) {
case BARCODE_HIBC_128 :
error_number = code_128 ( symbol , ( unsigned char * ) to_process , length ) ;
2020-11-01 21:32:55 +03:00
ustrcpy ( symbol - > text , " * " ) ;
ustrcat ( symbol - > text , to_process ) ;
ustrcat ( symbol - > text , " * " ) ;
2016-02-20 13:50:15 +03:00
break ;
case BARCODE_HIBC_39 :
symbol - > option_2 = 0 ;
error_number = c39 ( symbol , ( unsigned char * ) to_process , length ) ;
2020-11-01 21:32:55 +03:00
ustrcpy ( symbol - > text , " * " ) ;
ustrcat ( symbol - > text , to_process ) ;
ustrcat ( symbol - > text , " * " ) ;
2016-02-20 13:50:15 +03:00
break ;
case BARCODE_HIBC_DM :
error_number = dmatrix ( symbol , ( unsigned char * ) to_process , length ) ;
break ;
case BARCODE_HIBC_QR :
error_number = qr_code ( symbol , ( unsigned char * ) to_process , length ) ;
break ;
case BARCODE_HIBC_PDF :
error_number = pdf417enc ( symbol , ( unsigned char * ) to_process , length ) ;
break ;
case BARCODE_HIBC_MICPDF :
error_number = micro_pdf417 ( symbol , ( unsigned char * ) to_process , length ) ;
break ;
case BARCODE_HIBC_AZTEC :
error_number = aztec ( symbol , ( unsigned char * ) to_process , length ) ;
break ;
2016-08-24 21:37:49 +03:00
case BARCODE_HIBC_BLOCKF :
error_number = codablock ( symbol , ( unsigned char * ) to_process , length ) ;
break ;
2016-02-20 13:50:15 +03:00
}
return error_number ;
2009-11-22 18:38:15 +03:00
}
2019-10-06 13:30:21 +03:00
static int check_force_gs1 ( const int symbology ) {
/* Returns 1 if symbology MUST have GS1 data */
2019-12-01 17:09:46 +03:00
2016-02-20 13:50:15 +03:00
switch ( symbology ) {
2020-07-29 22:43:08 +03:00
case BARCODE_GS1_128 :
2019-10-17 12:06:21 +03:00
case BARCODE_EAN14 :
case BARCODE_NVE18 :
2020-07-29 22:43:08 +03:00
case BARCODE_DBAR_EXP :
case BARCODE_DBAR_EXPSTK :
2021-07-06 21:53:31 +03:00
return 1 ;
2019-10-06 13:30:21 +03:00
break ;
}
2019-12-01 17:09:46 +03:00
2021-07-06 21:53:31 +03:00
return is_composite ( symbology ) ;
2019-10-06 13:30:21 +03:00
}
static int gs1_compliant ( const int symbology ) {
/* Returns 1 if symbology supports GS1 data */
switch ( symbology ) {
2016-02-20 13:50:15 +03:00
case BARCODE_CODE16K :
case BARCODE_AZTEC :
case BARCODE_DATAMATRIX :
case BARCODE_CODE49 :
case BARCODE_QRCODE :
2016-07-26 00:52:29 +03:00
case BARCODE_DOTCODE :
Add Structured Append support for AZTEC, CODEONE, DATAMATRIX, DOTCODE,
GRIDMATRIX, MAXICODE, MICROPDF417, PDF417, QRCODE, ULTRA
DOTCODE: use pre-calculated generator poly coeffs in Reed-Solomon for
performance improvement
PDF417/MICROPDF417: use common routine pdf417_initial()
GUI: code lines <= 118, shorthand widget_obj(),
shorten calling upcean_addon_gap(), upcean_guard_descent()
various backend: var name debug -> debug_print
2021-09-28 23:42:44 +03:00
case BARCODE_CODEONE :
2020-04-07 19:41:21 +03:00
case BARCODE_ULTRA :
Add Structured Append support for AZTEC, CODEONE, DATAMATRIX, DOTCODE,
GRIDMATRIX, MAXICODE, MICROPDF417, PDF417, QRCODE, ULTRA
DOTCODE: use pre-calculated generator poly coeffs in Reed-Solomon for
performance improvement
PDF417/MICROPDF417: use common routine pdf417_initial()
GUI: code lines <= 118, shorthand widget_obj(),
shorten calling upcean_addon_gap(), upcean_guard_descent()
various backend: var name debug -> debug_print
2021-09-28 23:42:44 +03:00
case BARCODE_RMQR :
// TODO: case BARCODE_CODABLOCKF:
// TODO: case BARCODE_HANXIN:
// TODO: case BARCODE_GRIDMATRIX:
2021-07-06 21:53:31 +03:00
return 1 ;
2016-02-20 13:50:15 +03:00
break ;
}
2021-07-06 21:53:31 +03:00
return check_force_gs1 ( symbology ) ;
2009-11-22 18:38:15 +03:00
}
2020-09-30 14:19:12 +03:00
static int is_dotty ( const int symbology ) {
/* Returns 1 if symbology is a matrix design renderable as dots */
2016-09-06 00:06:50 +03:00
2016-08-12 21:30:46 +03:00
switch ( symbology ) {
2020-09-30 14:19:12 +03:00
/* Note MAXICODE and ULTRA absent */
2016-08-12 21:30:46 +03:00
case BARCODE_QRCODE :
case BARCODE_DATAMATRIX :
case BARCODE_MICROQR :
case BARCODE_HIBC_DM :
case BARCODE_AZTEC :
case BARCODE_HIBC_QR :
case BARCODE_HIBC_AZTEC :
case BARCODE_AZRUNE :
case BARCODE_CODEONE :
case BARCODE_GRIDMATRIX :
case BARCODE_HANXIN :
case BARCODE_DOTCODE :
2017-05-21 01:37:50 +03:00
case BARCODE_UPNQR :
2019-11-25 22:08:25 +03:00
case BARCODE_RMQR :
2021-07-06 21:53:31 +03:00
return 1 ;
2016-08-12 21:30:46 +03:00
break ;
}
2016-09-06 00:06:50 +03:00
2021-07-06 21:53:31 +03:00
return 0 ;
2016-08-12 21:30:46 +03:00
}
2020-09-30 14:19:12 +03:00
static int is_fixed_ratio ( const int symbology ) {
/* Returns 1 if symbology has fixed aspect ratio (matrix design) */
if ( is_dotty ( symbology ) ) {
return 1 ;
}
switch ( symbology ) {
case BARCODE_MAXICODE :
case BARCODE_ULTRA :
2021-07-06 21:53:31 +03:00
return 1 ;
2020-09-30 14:19:12 +03:00
break ;
}
2021-07-06 21:53:31 +03:00
return 0 ;
2020-09-30 14:19:12 +03:00
}
2016-08-16 14:43:41 +03:00
static int supports_eci ( const int symbology ) {
/* Returns 1 if symbology can encode the ECI character */
2016-09-06 00:06:50 +03:00
2016-08-16 14:43:41 +03:00
switch ( symbology ) {
case BARCODE_AZTEC :
case BARCODE_DATAMATRIX :
case BARCODE_MAXICODE :
case BARCODE_MICROPDF417 :
case BARCODE_PDF417 :
2020-07-30 00:35:31 +03:00
case BARCODE_PDF417COMP :
2016-08-16 14:43:41 +03:00
case BARCODE_QRCODE :
case BARCODE_DOTCODE :
2021-02-06 02:55:24 +03:00
case BARCODE_CODEONE :
2016-08-16 14:43:41 +03:00
case BARCODE_GRIDMATRIX :
case BARCODE_HANXIN :
2019-03-21 12:14:24 +03:00
case BARCODE_ULTRA :
2021-08-20 18:50:39 +03:00
case BARCODE_RMQR :
2021-07-06 21:53:31 +03:00
return 1 ;
2016-08-16 14:43:41 +03:00
break ;
}
2016-09-06 00:06:50 +03:00
2021-07-06 21:53:31 +03:00
return 0 ;
2016-08-16 14:43:41 +03:00
}
2020-09-30 14:19:12 +03:00
static int has_hrt ( const int symbology ) {
/* Returns 1 if symbology supports HRT */
if ( is_fixed_ratio ( symbology ) ) {
return 0 ;
}
switch ( symbology ) { /* These don't support HRT */
case BARCODE_CODE16K :
case BARCODE_CODE49 :
case BARCODE_FLAT :
case BARCODE_POSTNET :
case BARCODE_FIM :
case BARCODE_PHARMA :
case BARCODE_PHARMA_TWO :
case BARCODE_PDF417 :
case BARCODE_PDF417COMP :
case BARCODE_AUSPOST :
case BARCODE_AUSREPLY :
case BARCODE_AUSROUTE :
case BARCODE_AUSREDIRECT :
case BARCODE_RM4SCC :
case BARCODE_CODABLOCKF :
case BARCODE_JAPANPOST :
case BARCODE_DBAR_STK :
case BARCODE_DBAR_OMNSTK :
case BARCODE_DBAR_EXPSTK :
case BARCODE_PLANET :
case BARCODE_MICROPDF417 :
case BARCODE_USPS_IMAIL :
case BARCODE_KIX :
case BARCODE_DAFT :
case BARCODE_HIBC_PDF :
case BARCODE_HIBC_MICPDF :
case BARCODE_HIBC_BLOCKF :
case BARCODE_MAILMARK :
case BARCODE_DBAR_STK_CC :
case BARCODE_DBAR_OMNSTK_CC :
case BARCODE_DBAR_EXPSTK_CC :
return 0 ;
break ;
}
return 1 ;
}
2021-07-06 21:53:31 +03:00
static int reduced_charset ( struct zint_symbol * symbol , unsigned char * source , int length ) ;
2019-11-27 19:16:14 +03:00
2020-10-26 15:21:43 +03:00
static int extended_or_reduced_charset ( struct zint_symbol * symbol , unsigned char * source , const int length ) {
2016-02-20 13:50:15 +03:00
int error_number = 0 ;
2019-11-27 19:16:14 +03:00
2016-02-20 13:50:15 +03:00
switch ( symbol - > symbology ) {
2019-11-27 19:16:14 +03:00
/* These are the "elite" standards which have support for specific character sets */
2016-02-20 13:50:15 +03:00
case BARCODE_QRCODE : error_number = qr_code ( symbol , source , length ) ;
break ;
case BARCODE_MICROQR : error_number = microqr ( symbol , source , length ) ;
break ;
case BARCODE_GRIDMATRIX : error_number = grid_matrix ( symbol , source , length ) ;
break ;
2016-04-07 19:13:43 +03:00
case BARCODE_HANXIN : error_number = han_xin ( symbol , source , length ) ;
break ;
2017-05-21 01:37:50 +03:00
case BARCODE_UPNQR : error_number = upnqr ( symbol , source , length ) ;
break ;
2019-11-25 22:08:25 +03:00
case BARCODE_RMQR : error_number = rmqr ( symbol , source , length ) ;
break ;
2019-11-27 19:16:14 +03:00
default : error_number = reduced_charset ( symbol , source , length ) ;
break ;
2016-02-20 13:50:15 +03:00
}
2009-11-22 18:38:15 +03:00
2016-02-20 13:50:15 +03:00
return error_number ;
2009-11-22 18:38:15 +03:00
}
2021-07-06 21:53:31 +03:00
static int reduced_charset ( struct zint_symbol * symbol , unsigned char * source , int length ) {
2019-11-27 19:16:14 +03:00
/* These are the "norm" standards which only support Latin-1 at most, though a few support ECI */
2016-02-20 13:50:15 +03:00
int error_number = 0 ;
2020-10-26 15:21:43 +03:00
unsigned char * preprocessed = source ;
2016-02-20 13:50:15 +03:00
2021-07-06 21:53:31 +03:00
int eci_length = get_eci_length ( symbol - > eci , source , length ) ;
2009-11-22 18:38:15 +03:00
# ifndef _MSC_VER
2021-01-11 21:11:41 +03:00
unsigned char preprocessed_buf [ eci_length + 1 ] ;
2009-11-22 18:38:15 +03:00
# else
2021-01-11 21:11:41 +03:00
unsigned char * preprocessed_buf = ( unsigned char * ) _alloca ( eci_length + 1 ) ;
2009-11-22 18:38:15 +03:00
# endif
2019-12-01 17:09:46 +03:00
2021-01-11 21:11:41 +03:00
if ( ( symbol - > input_mode & 0x07 ) = = UNICODE_MODE & & is_eci_convertible ( symbol - > eci ) ) {
2020-10-26 15:21:43 +03:00
/* Prior check ensures ECI only set for those that support it */
preprocessed = preprocessed_buf ;
2021-07-06 21:53:31 +03:00
error_number = utf8_to_eci ( symbol - > eci , source , preprocessed , & length ) ;
2020-10-26 15:21:43 +03:00
if ( error_number ! = 0 ) {
2021-07-06 21:53:31 +03:00
if ( symbol - > eci ) {
sprintf ( symbol - > errtxt , " 244: Invalid character in input data for ECI %d " , symbol - > eci ) ;
} else {
strcpy ( symbol - > errtxt , " 204: Invalid character in input data (ISO/IEC 8859-1 only) " ) ;
}
2020-10-26 15:21:43 +03:00
return error_number ;
}
2016-02-20 13:50:15 +03:00
}
2019-12-01 17:09:46 +03:00
2016-02-20 13:50:15 +03:00
switch ( symbol - > symbology ) {
2021-07-06 21:53:31 +03:00
case BARCODE_C25STANDARD : error_number = matrix_two_of_five ( symbol , preprocessed , length ) ;
2016-02-20 13:50:15 +03:00
break ;
2021-07-06 21:53:31 +03:00
case BARCODE_C25IND : error_number = industrial_two_of_five ( symbol , preprocessed , length ) ;
2016-02-20 13:50:15 +03:00
break ;
2021-07-06 21:53:31 +03:00
case BARCODE_C25INTER : error_number = interleaved_two_of_five ( symbol , preprocessed , length ) ;
2016-02-20 13:50:15 +03:00
break ;
2021-07-06 21:53:31 +03:00
case BARCODE_C25IATA : error_number = iata_two_of_five ( symbol , preprocessed , length ) ;
2016-02-20 13:50:15 +03:00
break ;
2021-07-06 21:53:31 +03:00
case BARCODE_C25LOGIC : error_number = logic_two_of_five ( symbol , preprocessed , length ) ;
2016-02-20 13:50:15 +03:00
break ;
2021-07-06 21:53:31 +03:00
case BARCODE_DPLEIT : error_number = dpleit ( symbol , preprocessed , length ) ;
2016-02-20 13:50:15 +03:00
break ;
2021-07-06 21:53:31 +03:00
case BARCODE_DPIDENT : error_number = dpident ( symbol , preprocessed , length ) ;
2016-02-20 13:50:15 +03:00
break ;
2017-10-23 22:37:52 +03:00
case BARCODE_UPCA :
2016-10-28 21:43:08 +03:00
case BARCODE_UPCA_CHK :
2017-10-23 22:37:52 +03:00
case BARCODE_UPCE :
2016-10-28 21:43:08 +03:00
case BARCODE_UPCE_CHK :
2017-10-23 22:37:52 +03:00
case BARCODE_EANX :
2016-10-28 21:43:08 +03:00
case BARCODE_EANX_CHK :
2021-07-06 21:53:31 +03:00
case BARCODE_ISBNX :
error_number = eanx ( symbol , preprocessed , length ) ;
2016-02-20 13:50:15 +03:00
break ;
2021-07-06 21:53:31 +03:00
case BARCODE_GS1_128 : error_number = ean_128 ( symbol , preprocessed , length ) ;
2016-02-20 13:50:15 +03:00
break ;
2021-07-06 21:53:31 +03:00
case BARCODE_CODE39 : error_number = c39 ( symbol , preprocessed , length ) ;
2016-02-20 13:50:15 +03:00
break ;
2021-07-06 21:53:31 +03:00
case BARCODE_PZN : error_number = pharmazentral ( symbol , preprocessed , length ) ;
2016-02-20 13:50:15 +03:00
break ;
2021-07-06 21:53:31 +03:00
case BARCODE_EXCODE39 : error_number = ec39 ( symbol , preprocessed , length ) ;
2016-02-20 13:50:15 +03:00
break ;
2021-07-06 21:53:31 +03:00
case BARCODE_CODABAR : error_number = codabar ( symbol , preprocessed , length ) ;
2016-02-20 13:50:15 +03:00
break ;
2021-07-06 21:53:31 +03:00
case BARCODE_CODE93 : error_number = c93 ( symbol , preprocessed , length ) ;
2016-02-20 13:50:15 +03:00
break ;
2021-07-06 21:53:31 +03:00
case BARCODE_LOGMARS : error_number = c39 ( symbol , preprocessed , length ) ;
2016-02-20 13:50:15 +03:00
break ;
2016-10-28 21:43:08 +03:00
case BARCODE_CODE128 :
2017-10-23 22:37:52 +03:00
case BARCODE_CODE128B :
2021-07-06 21:53:31 +03:00
error_number = code_128 ( symbol , preprocessed , length ) ;
2016-02-20 13:50:15 +03:00
break ;
2021-07-06 21:53:31 +03:00
case BARCODE_NVE18 : error_number = nve_18 ( symbol , preprocessed , length ) ;
2016-02-20 13:50:15 +03:00
break ;
2021-07-06 21:53:31 +03:00
case BARCODE_CODE11 : error_number = code_11 ( symbol , preprocessed , length ) ;
2016-02-20 13:50:15 +03:00
break ;
2021-07-06 21:53:31 +03:00
case BARCODE_MSI_PLESSEY : error_number = msi_handle ( symbol , preprocessed , length ) ;
2016-02-20 13:50:15 +03:00
break ;
2021-07-06 21:53:31 +03:00
case BARCODE_TELEPEN : error_number = telepen ( symbol , preprocessed , length ) ;
2016-02-20 13:50:15 +03:00
break ;
2021-07-06 21:53:31 +03:00
case BARCODE_TELEPEN_NUM : error_number = telepen_num ( symbol , preprocessed , length ) ;
2016-02-20 13:50:15 +03:00
break ;
2021-07-06 21:53:31 +03:00
case BARCODE_PHARMA : error_number = pharma_one ( symbol , preprocessed , length ) ;
2016-02-20 13:50:15 +03:00
break ;
2021-07-06 21:53:31 +03:00
case BARCODE_PLESSEY : error_number = plessey ( symbol , preprocessed , length ) ;
2016-02-20 13:50:15 +03:00
break ;
2021-07-06 21:53:31 +03:00
case BARCODE_ITF14 : error_number = itf14 ( symbol , preprocessed , length ) ;
2016-02-20 13:50:15 +03:00
break ;
2021-07-06 21:53:31 +03:00
case BARCODE_FLAT : error_number = flattermarken ( symbol , preprocessed , length ) ;
2016-02-20 13:50:15 +03:00
break ;
2021-07-06 21:53:31 +03:00
case BARCODE_FIM : error_number = fim ( symbol , preprocessed , length ) ;
2016-02-20 13:50:15 +03:00
break ;
2021-07-06 21:53:31 +03:00
case BARCODE_POSTNET : error_number = post_plot ( symbol , preprocessed , length ) ;
2016-02-20 13:50:15 +03:00
break ;
2021-07-06 21:53:31 +03:00
case BARCODE_PLANET : error_number = planet_plot ( symbol , preprocessed , length ) ;
2016-02-20 13:50:15 +03:00
break ;
2021-07-06 21:53:31 +03:00
case BARCODE_RM4SCC : error_number = royal_plot ( symbol , preprocessed , length ) ;
2016-02-20 13:50:15 +03:00
break ;
2016-10-28 21:43:08 +03:00
case BARCODE_AUSPOST :
2017-10-23 22:37:52 +03:00
case BARCODE_AUSREPLY :
2016-10-28 21:43:08 +03:00
case BARCODE_AUSROUTE :
case BARCODE_AUSREDIRECT :
2021-07-06 21:53:31 +03:00
error_number = australia_post ( symbol , preprocessed , length ) ;
2016-02-20 13:50:15 +03:00
break ;
2021-07-06 21:53:31 +03:00
case BARCODE_CODE16K : error_number = code16k ( symbol , preprocessed , length ) ;
2016-02-20 13:50:15 +03:00
break ;
2021-07-06 21:53:31 +03:00
case BARCODE_PHARMA_TWO : error_number = pharma_two ( symbol , preprocessed , length ) ;
2016-02-20 13:50:15 +03:00
break ;
2021-07-06 21:53:31 +03:00
case BARCODE_USPS_IMAIL : error_number = imail ( symbol , preprocessed , length ) ;
2016-02-20 13:50:15 +03:00
break ;
2020-07-29 22:43:08 +03:00
case BARCODE_DBAR_OMN :
case BARCODE_DBAR_STK :
case BARCODE_DBAR_OMNSTK :
2021-07-06 21:53:31 +03:00
error_number = rss14 ( symbol , preprocessed , length ) ;
2016-02-20 13:50:15 +03:00
break ;
2021-07-06 21:53:31 +03:00
case BARCODE_DBAR_LTD : error_number = rsslimited ( symbol , preprocessed , length ) ;
2016-02-20 13:50:15 +03:00
break ;
2020-07-29 22:43:08 +03:00
case BARCODE_DBAR_EXP :
case BARCODE_DBAR_EXPSTK :
2021-07-06 21:53:31 +03:00
error_number = rssexpanded ( symbol , preprocessed , length ) ;
2016-10-28 21:43:08 +03:00
break ;
2017-10-23 22:37:52 +03:00
case BARCODE_EANX_CC :
2020-07-29 22:43:08 +03:00
case BARCODE_GS1_128_CC :
case BARCODE_DBAR_OMN_CC :
case BARCODE_DBAR_LTD_CC :
case BARCODE_DBAR_EXP_CC :
2017-10-23 22:37:52 +03:00
case BARCODE_UPCA_CC :
case BARCODE_UPCE_CC :
2020-07-29 22:43:08 +03:00
case BARCODE_DBAR_STK_CC :
case BARCODE_DBAR_OMNSTK_CC :
case BARCODE_DBAR_EXPSTK_CC :
2021-07-06 21:53:31 +03:00
error_number = composite ( symbol , preprocessed , length ) ;
2016-02-20 13:50:15 +03:00
break ;
2021-07-06 21:53:31 +03:00
case BARCODE_KIX : error_number = kix_code ( symbol , preprocessed , length ) ;
2016-02-20 13:50:15 +03:00
break ;
2021-07-06 21:53:31 +03:00
case BARCODE_CODE32 : error_number = code32 ( symbol , preprocessed , length ) ;
2016-02-20 13:50:15 +03:00
break ;
2021-07-06 21:53:31 +03:00
case BARCODE_DAFT : error_number = daft_code ( symbol , preprocessed , length ) ;
2016-02-20 13:50:15 +03:00
break ;
2016-10-28 21:43:08 +03:00
case BARCODE_EAN14 :
2021-07-06 21:53:31 +03:00
error_number = ean_14 ( symbol , preprocessed , length ) ;
2016-02-20 13:50:15 +03:00
break ;
2021-07-06 21:53:31 +03:00
case BARCODE_AZRUNE : error_number = aztec_runes ( symbol , preprocessed , length ) ;
2016-02-20 13:50:15 +03:00
break ;
2021-07-06 21:53:31 +03:00
case BARCODE_KOREAPOST : error_number = korea_post ( symbol , preprocessed , length ) ;
2016-02-20 13:50:15 +03:00
break ;
2017-10-23 22:37:52 +03:00
case BARCODE_HIBC_128 :
case BARCODE_HIBC_39 :
case BARCODE_HIBC_DM :
case BARCODE_HIBC_QR :
case BARCODE_HIBC_PDF :
case BARCODE_HIBC_MICPDF :
case BARCODE_HIBC_AZTEC :
case BARCODE_HIBC_BLOCKF :
2021-07-06 21:53:31 +03:00
error_number = hibc ( symbol , preprocessed , length ) ;
2016-08-24 21:37:49 +03:00
break ;
2021-07-06 21:53:31 +03:00
case BARCODE_JAPANPOST : error_number = japan_post ( symbol , preprocessed , length ) ;
2016-02-20 13:50:15 +03:00
break ;
2021-07-06 21:53:31 +03:00
case BARCODE_CODE49 : error_number = code_49 ( symbol , preprocessed , length ) ;
2016-02-20 13:50:15 +03:00
break ;
2021-07-06 21:53:31 +03:00
case BARCODE_CHANNEL : error_number = channel_code ( symbol , preprocessed , length ) ;
2016-02-20 13:50:15 +03:00
break ;
2021-07-06 21:53:31 +03:00
case BARCODE_CODEONE : error_number = code_one ( symbol , preprocessed , length ) ;
2016-02-20 13:50:15 +03:00
break ;
2021-07-06 21:53:31 +03:00
case BARCODE_DATAMATRIX : error_number = dmatrix ( symbol , preprocessed , length ) ;
2016-02-20 13:50:15 +03:00
break ;
2017-10-23 22:37:52 +03:00
case BARCODE_PDF417 :
2020-07-30 00:35:31 +03:00
case BARCODE_PDF417COMP :
2021-07-06 21:53:31 +03:00
error_number = pdf417enc ( symbol , preprocessed , length ) ;
2016-02-20 13:50:15 +03:00
break ;
2021-07-06 21:53:31 +03:00
case BARCODE_MICROPDF417 : error_number = micro_pdf417 ( symbol , preprocessed , length ) ;
2016-02-20 13:50:15 +03:00
break ;
2021-07-06 21:53:31 +03:00
case BARCODE_MAXICODE : error_number = maxicode ( symbol , preprocessed , length ) ;
2016-02-20 13:50:15 +03:00
break ;
2021-07-06 21:53:31 +03:00
case BARCODE_AZTEC : error_number = aztec ( symbol , preprocessed , length ) ;
2016-02-20 13:50:15 +03:00
break ;
2021-07-06 21:53:31 +03:00
case BARCODE_DOTCODE : error_number = dotcode ( symbol , preprocessed , length ) ;
2016-07-26 00:52:29 +03:00
break ;
2021-07-06 21:53:31 +03:00
case BARCODE_CODABLOCKF : error_number = codablock ( symbol , preprocessed , length ) ;
2016-08-24 21:37:49 +03:00
break ;
2021-07-06 21:53:31 +03:00
case BARCODE_VIN : error_number = vin ( symbol , preprocessed , length ) ;
2018-02-03 21:44:01 +03:00
break ;
2021-07-06 21:53:31 +03:00
case BARCODE_MAILMARK : error_number = mailmark ( symbol , preprocessed , length ) ;
2018-02-06 23:57:01 +03:00
break ;
2021-07-06 21:53:31 +03:00
case BARCODE_ULTRA : error_number = ultracode ( symbol , preprocessed , length ) ;
2019-03-21 12:14:24 +03:00
break ;
2021-07-06 21:53:31 +03:00
case BARCODE_DPD : error_number = dpd_parcel ( symbol , preprocessed , length ) ;
break ;
default : /* Should never happen */
strcpy ( symbol - > errtxt , " 001: Internal error " ) ; /* Not reached */
error_number = ZINT_ERROR_ENCODING_PROBLEM ;
2020-08-03 14:13:05 +03:00
break ;
2016-02-20 13:50:15 +03:00
}
2017-10-23 22:37:52 +03:00
2016-02-20 13:50:15 +03:00
return error_number ;
2009-11-22 18:38:15 +03:00
}
2021-06-10 18:20:14 +03:00
STATIC_UNLESS_ZINT_TEST void strip_bom ( unsigned char * source , int * input_length ) {
2017-10-09 21:59:02 +03:00
int i ;
2017-10-23 22:37:52 +03:00
2021-01-12 22:51:54 +03:00
/* Note if BOM is only data then not stripped */
if ( * input_length > 3 & & ( source [ 0 ] = = 0xef ) & & ( source [ 1 ] = = 0xbb ) & & ( source [ 2 ] = = 0xbf ) ) {
/* BOM at start of input data, strip in accordance with RFC 3629 */
for ( i = 3 ; i < = * input_length ; i + + ) { /* Include terminating NUL */
source [ i - 3 ] = source [ i ] ;
2017-10-09 21:59:02 +03:00
}
2021-01-12 22:51:54 +03:00
* input_length - = 3 ;
2017-10-09 21:59:02 +03:00
}
}
2019-12-19 03:37:55 +03:00
static int escape_char_process ( struct zint_symbol * symbol , unsigned char * input_string , int * length ) {
2017-10-21 14:45:50 +03:00
int in_posn , out_posn ;
int hex1 , hex2 ;
2020-08-04 13:48:43 +03:00
int i , unicode ;
2017-10-21 14:45:50 +03:00
# ifndef _MSC_VER
unsigned char escaped_string [ * length + 1 ] ;
# else
2021-05-15 14:23:46 +03:00
unsigned char * escaped_string = ( unsigned char * ) _alloca ( * length + 1 ) ;
2017-10-21 14:45:50 +03:00
# endif
in_posn = 0 ;
out_posn = 0 ;
2019-12-01 17:09:46 +03:00
2017-10-21 14:45:50 +03:00
do {
if ( input_string [ in_posn ] = = ' \\ ' ) {
2020-05-16 12:22:33 +03:00
if ( in_posn + 1 > = * length ) {
strcpy ( symbol - > errtxt , " 236: Incomplete escape character in input data " ) ;
return ZINT_ERROR_INVALID_DATA ;
}
2017-10-21 14:45:50 +03:00
switch ( input_string [ in_posn + 1 ] ) {
case ' 0 ' : escaped_string [ out_posn ] = 0x00 ; /* Null */
in_posn + = 2 ;
break ;
case ' E ' : escaped_string [ out_posn ] = 0x04 ; /* End of Transmission */
in_posn + = 2 ;
break ;
case ' a ' : escaped_string [ out_posn ] = 0x07 ; /* Bell */
in_posn + = 2 ;
break ;
case ' b ' : escaped_string [ out_posn ] = 0x08 ; /* Backspace */
in_posn + = 2 ;
break ;
case ' t ' : escaped_string [ out_posn ] = 0x09 ; /* Horizontal tab */
in_posn + = 2 ;
break ;
case ' n ' : escaped_string [ out_posn ] = 0x0a ; /* Line feed */
in_posn + = 2 ;
break ;
case ' v ' : escaped_string [ out_posn ] = 0x0b ; /* Vertical tab */
in_posn + = 2 ;
break ;
case ' f ' : escaped_string [ out_posn ] = 0x0c ; /* Form feed */
in_posn + = 2 ;
break ;
case ' r ' : escaped_string [ out_posn ] = 0x0d ; /* Carriage return */
in_posn + = 2 ;
break ;
case ' e ' : escaped_string [ out_posn ] = 0x1b ; /* Escape */
in_posn + = 2 ;
break ;
case ' G ' : escaped_string [ out_posn ] = 0x1d ; /* Group Separator */
in_posn + = 2 ;
break ;
case ' R ' : escaped_string [ out_posn ] = 0x1e ; /* Record Separator */
in_posn + = 2 ;
break ;
case ' x ' : if ( in_posn + 4 > * length ) {
strcpy ( symbol - > errtxt , " 232: Incomplete escape character in input data " ) ;
return ZINT_ERROR_INVALID_DATA ;
}
hex1 = ctoi ( input_string [ in_posn + 2 ] ) ;
hex2 = ctoi ( input_string [ in_posn + 3 ] ) ;
if ( ( hex1 > = 0 ) & & ( hex2 > = 0 ) ) {
2021-01-15 17:22:32 +03:00
escaped_string [ out_posn ] = ( hex1 < < 4 ) + hex2 ;
2017-10-21 14:45:50 +03:00
in_posn + = 4 ;
} else {
strcpy ( symbol - > errtxt , " 233: Corrupt escape character in input data " ) ;
return ZINT_ERROR_INVALID_DATA ;
}
break ;
case ' \\ ' : escaped_string [ out_posn ] = ' \\ ' ;
in_posn + = 2 ;
break ;
2020-08-04 13:48:43 +03:00
case ' u ' :
if ( in_posn + 6 > * length ) {
2021-01-15 17:22:32 +03:00
strcpy ( symbol - > errtxt , " 209: Incomplete Unicode escape character in input data " ) ;
2020-08-04 13:48:43 +03:00
return ZINT_ERROR_INVALID_DATA ;
}
unicode = 0 ;
for ( i = 0 ; i < 4 ; i + + ) {
if ( ctoi ( input_string [ in_posn + i + 2 ] ) = = - 1 ) {
2021-01-15 17:22:32 +03:00
strcpy ( symbol - > errtxt , " 211: Corrupt Unicode escape character in input data " ) ;
2020-08-04 13:48:43 +03:00
return ZINT_ERROR_INVALID_DATA ;
}
unicode = unicode < < 4 ;
unicode + = ctoi ( input_string [ in_posn + i + 2 ] ) ;
}
2021-01-15 17:22:32 +03:00
/* Exclude reversed BOM and surrogates */
if ( unicode = = 0xfffe | | ( unicode > = 0xd800 & & unicode < 0xe000 ) ) {
strcpy ( symbol - > errtxt , " 246: Invalid Unicode BMP escape character in input data " ) ;
return ZINT_ERROR_INVALID_DATA ;
}
2020-08-04 13:48:43 +03:00
if ( unicode > = 0x800 ) {
escaped_string [ out_posn ] = 0xe0 + ( ( unicode & 0xf000 ) > > 12 ) ;
out_posn + + ;
escaped_string [ out_posn ] = 0x80 + ( ( unicode & 0x0fc0 ) > > 6 ) ;
out_posn + + ;
escaped_string [ out_posn ] = 0x80 + ( unicode & 0x003f ) ;
} else if ( unicode > = 0x80 ) {
escaped_string [ out_posn ] = 0xc0 + ( ( unicode & 0x07c0 ) > > 6 ) ;
out_posn + + ;
escaped_string [ out_posn ] = 0x80 + ( unicode & 0x003f ) ;
} else {
escaped_string [ out_posn ] = unicode & 0x7f ;
}
in_posn + = 6 ;
break ;
2017-10-21 14:45:50 +03:00
default : strcpy ( symbol - > errtxt , " 234: Unrecognised escape character in input data " ) ;
return ZINT_ERROR_INVALID_DATA ;
break ;
}
} else {
escaped_string [ out_posn ] = input_string [ in_posn ] ;
in_posn + + ;
}
out_posn + + ;
} while ( in_posn < * length ) ;
2019-12-01 17:09:46 +03:00
2017-10-21 14:45:50 +03:00
memcpy ( input_string , escaped_string , out_posn ) ;
2018-01-13 18:45:26 +03:00
input_string [ out_posn ] = ' \0 ' ;
2017-10-21 14:45:50 +03:00
* length = out_posn ;
2019-12-01 17:09:46 +03:00
2021-07-06 21:53:31 +03:00
return 0 ;
2017-10-21 14:45:50 +03:00
}
2021-07-06 21:53:31 +03:00
/* Encode a barcode. If `length` is 0, `source` must be NUL-terminated. */
int ZBarcode_Encode ( struct zint_symbol * symbol , const unsigned char * source , int length ) {
2021-01-12 22:51:54 +03:00
int error_number , warn_number ;
2015-08-18 14:50:42 +03:00
# ifdef _MSC_VER
2021-01-11 21:11:41 +03:00
unsigned char * local_source ;
2015-08-18 14:50:42 +03:00
# endif
2020-09-30 14:19:12 +03:00
if ( ! symbol ) return ZINT_ERROR_INVALID_DATA ;
2020-12-21 22:30:07 +03:00
if ( symbol - > debug & ZINT_DEBUG_PRINT ) {
printf ( " ZBarcode_Encode: symbology: %d, input_mode: 0x%X, ECI: %d, option_1: %d, option_2: %d, "
2021-05-28 16:37:57 +03:00
" option_3: %d, scale: %g \n output_options: 0x%X, fg: %s, bg: %s, "
2021-07-26 17:29:05 +03:00
" length: %d, First 10 source: \" %.*s \" , First 10 primary: \" %.10s \" \n " ,
2020-12-21 22:30:07 +03:00
symbol - > symbology , symbol - > input_mode , symbol - > eci , symbol - > option_1 , symbol - > option_2 ,
2021-05-28 16:37:57 +03:00
symbol - > option_3 , symbol - > scale , symbol - > output_options , symbol - > fgcolour , symbol - > bgcolour ,
2021-07-26 17:29:05 +03:00
length , length < 10 ? length : 10 , source ? ( const char * ) source : " <NULL> " , symbol - > primary ) ;
2020-12-21 22:30:07 +03:00
}
2021-01-12 22:51:54 +03:00
warn_number = 0 ;
2017-10-23 22:37:52 +03:00
2020-09-30 14:19:12 +03:00
if ( source = = NULL ) {
2021-07-06 21:53:31 +03:00
return error_tag ( symbol , ZINT_ERROR_INVALID_DATA , " 200: Input data NULL " ) ;
2020-09-30 14:19:12 +03:00
}
2021-07-06 21:53:31 +03:00
if ( length < = 0 ) {
length = ( int ) ustrlen ( source ) ;
2016-02-20 13:50:15 +03:00
}
2021-07-06 21:53:31 +03:00
if ( length < = 0 ) {
return error_tag ( symbol , ZINT_ERROR_INVALID_DATA , " 205: No input data " ) ;
2016-02-20 13:50:15 +03:00
}
2021-07-06 21:53:31 +03:00
if ( length > ZINT_MAX_DATA_LEN ) {
return error_tag ( symbol , ZINT_ERROR_TOO_LONG , " 243: Input data too long " ) ;
2020-12-21 22:30:07 +03:00
}
2016-02-20 13:50:15 +03:00
/* First check the symbology field */
2020-11-01 21:32:55 +03:00
if ( ! ZBarcode_ValidID ( symbol - > symbology ) ) {
2021-07-06 21:53:31 +03:00
int orig_symbology = symbol - > symbology ; /* For self-check */
2020-11-01 21:32:55 +03:00
if ( symbol - > symbology < 1 ) {
2021-07-06 21:53:31 +03:00
warn_number = error_tag ( symbol , ZINT_WARN_INVALID_OPTION , " 206: Symbology out of range " ) ;
if ( warn_number > = ZINT_ERROR ) {
return warn_number ;
2020-11-01 21:32:55 +03:00
}
2021-01-12 22:51:54 +03:00
symbol - > symbology = BARCODE_CODE128 ;
2021-07-06 21:53:31 +03:00
/* symbol->symbologys 1 to 126 are defined by tbarcode */
2020-11-01 21:32:55 +03:00
} else if ( symbol - > symbology = = 5 ) {
symbol - > symbology = BARCODE_C25STANDARD ;
} else if ( ( symbol - > symbology > = 10 ) & & ( symbol - > symbology < = 12 ) ) {
symbol - > symbology = BARCODE_EANX ;
} else if ( symbol - > symbology = = 15 ) {
symbol - > symbology = BARCODE_EANX ;
} else if ( symbol - > symbology = = 17 ) {
symbol - > symbology = BARCODE_UPCA ;
} else if ( symbol - > symbology = = 19 ) {
2021-07-06 21:53:31 +03:00
warn_number = error_tag ( symbol , ZINT_WARN_INVALID_OPTION , " 207: Codabar 18 not supported " ) ;
if ( warn_number > = ZINT_ERROR ) {
return warn_number ;
2020-11-01 21:32:55 +03:00
}
2021-01-12 22:51:54 +03:00
symbol - > symbology = BARCODE_CODABAR ;
2021-07-06 21:53:31 +03:00
} else if ( symbol - > symbology = = 26 ) { /* UPC-A up to tbarcode 9, ISSN for tbarcode 10+ */
2020-11-01 21:32:55 +03:00
symbol - > symbology = BARCODE_UPCA ;
2021-07-06 21:53:31 +03:00
} else if ( symbol - > symbology = = 27 ) { /* UPCD1 up to tbarcode 9, ISSN + 2 digit add-on for tbarcode 10+ */
return error_tag ( symbol , ZINT_ERROR_INVALID_OPTION , " 208: UPCD1 not supported " ) ;
2020-11-01 21:32:55 +03:00
} else if ( symbol - > symbology = = 33 ) {
symbol - > symbology = BARCODE_GS1_128 ;
} else if ( symbol - > symbology = = 36 ) {
symbol - > symbology = BARCODE_UPCA ;
2021-07-06 21:53:31 +03:00
} else if ( symbol - > symbology = = 39 ) {
symbol - > symbology = BARCODE_UPCE ;
2020-11-01 21:32:55 +03:00
} else if ( ( symbol - > symbology > = 41 ) & & ( symbol - > symbology < = 45 ) ) {
symbol - > symbology = BARCODE_POSTNET ;
} else if ( symbol - > symbology = = 46 ) {
symbol - > symbology = BARCODE_PLESSEY ;
} else if ( symbol - > symbology = = 48 ) {
symbol - > symbology = BARCODE_NVE18 ;
Add Structured Append support for AZTEC, CODEONE, DATAMATRIX, DOTCODE,
GRIDMATRIX, MAXICODE, MICROPDF417, PDF417, QRCODE, ULTRA
DOTCODE: use pre-calculated generator poly coeffs in Reed-Solomon for
performance improvement
PDF417/MICROPDF417: use common routine pdf417_initial()
GUI: code lines <= 118, shorthand widget_obj(),
shorten calling upcean_addon_gap(), upcean_guard_descent()
various backend: var name debug -> debug_print
2021-09-28 23:42:44 +03:00
} else if ( symbol - > symbology = = 54 ) { /* General Parcel up to tbarcode 9, Brazilian CEPNet for tbarcode 10+ */
2021-07-06 21:53:31 +03:00
warn_number = error_tag ( symbol , ZINT_WARN_INVALID_OPTION , " 210: General Parcel Code not supported " ) ;
if ( warn_number > = ZINT_ERROR ) {
return warn_number ;
2020-11-01 21:32:55 +03:00
}
2021-01-12 22:51:54 +03:00
symbol - > symbology = BARCODE_CODE128 ;
2020-11-01 21:32:55 +03:00
} else if ( ( symbol - > symbology = = 59 ) | | ( symbol - > symbology = = 61 ) ) {
2020-08-22 13:09:57 +03:00
symbol - > symbology = BARCODE_CODE128 ;
2020-11-01 21:32:55 +03:00
} else if ( symbol - > symbology = = 62 ) {
symbol - > symbology = BARCODE_CODE93 ;
} else if ( ( symbol - > symbology = = 64 ) | | ( symbol - > symbology = = 65 ) ) {
symbol - > symbology = BARCODE_AUSPOST ;
} else if ( symbol - > symbology = = 78 ) {
symbol - > symbology = BARCODE_DBAR_OMN ;
} else if ( symbol - > symbology = = 83 ) {
symbol - > symbology = BARCODE_PLANET ;
} else if ( symbol - > symbology = = 88 ) {
symbol - > symbology = BARCODE_GS1_128 ;
2021-07-06 21:53:31 +03:00
} else if ( symbol - > symbology = = 91 ) { /* BC412 up to tbarcode 9, Code 32 for tbarcode 10+ */
warn_number = error_tag ( symbol , ZINT_WARN_INVALID_OPTION , " 212: Symbology out of range " ) ;
if ( warn_number > = ZINT_ERROR ) {
return warn_number ;
2020-11-01 21:32:55 +03:00
}
2021-01-12 22:51:54 +03:00
symbol - > symbology = BARCODE_CODE128 ;
2020-11-01 21:32:55 +03:00
} else if ( ( symbol - > symbology > = 94 ) & & ( symbol - > symbology < = 95 ) ) {
2021-07-06 21:53:31 +03:00
warn_number = error_tag ( symbol , ZINT_WARN_INVALID_OPTION , " 213: Symbology out of range " ) ;
if ( warn_number > = ZINT_ERROR ) {
return warn_number ;
2020-11-01 21:32:55 +03:00
}
2021-01-12 22:51:54 +03:00
symbol - > symbology = BARCODE_CODE128 ;
2020-11-01 21:32:55 +03:00
} else if ( symbol - > symbology = = 100 ) {
symbol - > symbology = BARCODE_HIBC_128 ;
} else if ( symbol - > symbology = = 101 ) {
symbol - > symbology = BARCODE_HIBC_39 ;
} else if ( symbol - > symbology = = 103 ) {
symbol - > symbology = BARCODE_HIBC_DM ;
} else if ( symbol - > symbology = = 105 ) {
symbol - > symbology = BARCODE_HIBC_QR ;
} else if ( symbol - > symbology = = 107 ) {
symbol - > symbology = BARCODE_HIBC_PDF ;
} else if ( symbol - > symbology = = 109 ) {
symbol - > symbology = BARCODE_HIBC_MICPDF ;
} else if ( symbol - > symbology = = 111 ) {
symbol - > symbology = BARCODE_HIBC_BLOCKF ;
} else if ( ( symbol - > symbology = = 113 ) | | ( symbol - > symbology = = 114 ) ) {
2021-07-06 21:53:31 +03:00
warn_number = error_tag ( symbol , ZINT_WARN_INVALID_OPTION , " 214: Symbology out of range " ) ;
if ( warn_number > = ZINT_ERROR ) {
return warn_number ;
2020-11-01 21:32:55 +03:00
}
2021-01-12 22:51:54 +03:00
symbol - > symbology = BARCODE_CODE128 ;
2020-11-01 21:32:55 +03:00
} else if ( ( symbol - > symbology > = 117 ) & & ( symbol - > symbology < = 127 ) ) {
2021-07-06 21:53:31 +03:00
if ( symbol - > symbology ! = 121 ) { /* BARCODE_MAILMARK */
warn_number = error_tag ( symbol , ZINT_WARN_INVALID_OPTION , " 215: Symbology out of range " ) ;
if ( warn_number > = ZINT_ERROR ) {
return warn_number ;
2020-11-01 21:32:55 +03:00
}
2021-01-12 22:51:54 +03:00
symbol - > symbology = BARCODE_CODE128 ;
2020-11-01 21:32:55 +03:00
}
/* Everything from 128 up is Zint-specific */
} else if ( symbol - > symbology > 145 ) {
2021-07-06 21:53:31 +03:00
warn_number = error_tag ( symbol , ZINT_WARN_INVALID_OPTION , " 216: Symbology out of range " ) ;
if ( warn_number > = ZINT_ERROR ) {
return warn_number ;
2020-08-22 13:09:57 +03:00
}
2021-01-12 22:51:54 +03:00
symbol - > symbology = BARCODE_CODE128 ;
2021-07-06 21:53:31 +03:00
}
if ( symbol - > symbology = = orig_symbology ) { /* Should never happen */
return error_tag ( symbol , ZINT_ERROR_ENCODING_PROBLEM , " 000: Internal error " ) ; /* Not reached */
2020-08-22 13:09:57 +03:00
}
2016-02-20 13:50:15 +03:00
}
2020-11-01 21:32:55 +03:00
if ( symbol - > eci ! = 0 ) {
if ( ! ( supports_eci ( symbol - > symbology ) ) ) {
2021-07-06 21:53:31 +03:00
return error_tag ( symbol , ZINT_ERROR_INVALID_OPTION , " 217: Symbology does not support ECI switching " ) ;
2021-01-12 22:51:54 +03:00
}
if ( ( symbol - > eci < 0 ) | | ( symbol - > eci = = 1 ) | | ( symbol - > eci = = 2 ) | | ( symbol - > eci > 999999 ) ) {
2021-07-06 21:53:31 +03:00
return error_tag ( symbol , ZINT_ERROR_INVALID_OPTION , " 218: Invalid ECI mode " ) ;
2020-11-01 21:32:55 +03:00
}
2016-08-16 14:43:41 +03:00
}
2016-02-20 13:50:15 +03:00
2021-09-20 16:56:27 +03:00
if ( ( symbol - > scale < 0.01f ) | | ( symbol - > scale > 100.0f ) ) {
2021-09-22 02:04:15 +03:00
return error_tag ( symbol , ZINT_ERROR_INVALID_OPTION , " 227: Scale out of range (0.01 to 100) " ) ;
2021-09-20 16:56:27 +03:00
}
2020-09-30 14:19:12 +03:00
if ( ( symbol - > dot_size < 0.01f ) | | ( symbol - > dot_size > 20.0f ) ) {
2021-09-22 02:04:15 +03:00
return error_tag ( symbol , ZINT_ERROR_INVALID_OPTION , " 221: Dot size out of range (0.01 to 20) " ) ;
2021-09-20 16:56:27 +03:00
}
if ( ( symbol - > height < 0.0f ) | | ( symbol - > height > 500.0f ) ) {
2021-09-22 02:04:15 +03:00
return error_tag ( symbol , ZINT_ERROR_INVALID_OPTION , " 765: Height out of range (0 to 500) " ) ;
}
if ( ( symbol - > guard_descent < 0.0f ) | | ( symbol - > guard_descent > 50.0f ) ) {
return error_tag ( symbol , ZINT_ERROR_INVALID_OPTION , " 769: Guard bar descent out of range (0 to 50) " ) ;
2021-09-20 16:56:27 +03:00
}
if ( ( symbol - > whitespace_width < 0 ) | | ( symbol - > whitespace_width > 100 ) ) {
2021-09-22 02:04:15 +03:00
return error_tag ( symbol , ZINT_ERROR_INVALID_OPTION , " 766: Whitespace width out of range (0 to 100) " ) ;
2021-09-20 16:56:27 +03:00
}
if ( ( symbol - > whitespace_height < 0 ) | | ( symbol - > whitespace_height > 100 ) ) {
2021-09-22 02:04:15 +03:00
return error_tag ( symbol , ZINT_ERROR_INVALID_OPTION , " 767: Whitespace height out of range (0 to 100) " ) ;
2021-09-20 16:56:27 +03:00
}
if ( ( symbol - > border_width < 0 ) | | ( symbol - > border_width > 100 ) ) {
2021-09-22 02:04:15 +03:00
return error_tag ( symbol , ZINT_ERROR_INVALID_OPTION , " 768: Border width out of range (0 to 100) " ) ;
2019-10-17 12:06:21 +03:00
}
2021-07-06 21:53:31 +03:00
if ( ( symbol - > input_mode & 0x07 ) > 2 ) {
symbol - > input_mode = DATA_MODE ; /* Reset completely TODO: in future, warn/error */
2021-01-11 21:11:41 +03:00
}
2021-07-06 21:53:31 +03:00
if ( ( symbol - > input_mode & 0x07 ) = = UNICODE_MODE & & ! is_valid_utf8 ( source , length ) ) {
return error_tag ( symbol , ZINT_ERROR_INVALID_DATA , " 245: Invalid UTF-8 in input data " ) ;
2019-11-27 19:16:14 +03:00
}
2021-01-11 21:11:41 +03:00
# ifndef _MSC_VER
2021-07-06 21:53:31 +03:00
unsigned char local_source [ length + 1 ] ;
2021-01-11 21:11:41 +03:00
# else
2021-07-06 21:53:31 +03:00
local_source = ( unsigned char * ) _alloca ( length + 1 ) ;
2021-01-11 21:11:41 +03:00
# endif
2021-07-06 21:53:31 +03:00
memcpy ( local_source , source , length ) ;
local_source [ length ] = ' \0 ' ;
2019-10-17 12:06:21 +03:00
/* Start acting on input mode */
2019-11-27 19:16:14 +03:00
if ( symbol - > input_mode & ESCAPE_MODE ) {
2021-07-06 21:53:31 +03:00
error_number = escape_char_process ( symbol , local_source , & length ) ; /* Only returns errors, not warnings */
2017-10-21 14:45:50 +03:00
if ( error_number ! = 0 ) {
2021-07-06 21:53:31 +03:00
return error_tag ( symbol , error_number , NULL ) ;
2017-10-21 14:45:50 +03:00
}
}
2017-10-23 22:37:52 +03:00
2019-11-27 19:16:14 +03:00
if ( ( symbol - > input_mode & 0x07 ) = = UNICODE_MODE ) {
2021-07-06 21:53:31 +03:00
strip_bom ( local_source , & length ) ;
2019-11-27 19:16:14 +03:00
}
if ( ( ( symbol - > input_mode & 0x07 ) = = GS1_MODE ) | | ( check_force_gs1 ( symbol - > symbology ) ) ) {
2019-10-17 12:06:21 +03:00
if ( gs1_compliant ( symbol - > symbology ) = = 1 ) {
2020-12-23 13:57:24 +03:00
// Reduce input for composite and non-forced symbologies, others (EAN128 and RSS_EXP based) will
// handle it themselves
2019-10-17 12:06:21 +03:00
if ( is_composite ( symbol - > symbology ) | | ! check_force_gs1 ( symbol - > symbology ) ) {
# ifndef _MSC_VER
2021-07-06 21:53:31 +03:00
unsigned char reduced [ length + 1 ] ;
2019-10-17 12:06:21 +03:00
# else
2021-07-06 21:53:31 +03:00
unsigned char * reduced = ( unsigned char * ) _alloca ( length + 1 ) ;
2019-10-17 12:06:21 +03:00
# endif
2021-07-06 21:53:31 +03:00
error_number = gs1_verify ( symbol , local_source , length , reduced ) ;
if ( error_number ) {
2021-07-13 00:27:16 +03:00
static const char in_2d_comp [ ] = " in 2D component " ;
if ( is_composite ( symbol - > symbology )
& & strlen ( symbol - > errtxt ) + strlen ( in_2d_comp ) < sizeof ( symbol - > errtxt ) ) {
2021-01-12 22:51:54 +03:00
strcat ( symbol - > errtxt , in_2d_comp ) ;
2020-09-30 14:19:12 +03:00
}
2021-07-06 21:53:31 +03:00
error_number = error_tag ( symbol , error_number , NULL ) ;
if ( error_number > = ZINT_ERROR ) {
return error_number ;
}
warn_number = error_number ; /* Override any previous warning (errtxt has been overwritten) */
2021-01-21 00:15:03 +03:00
}
2021-07-06 21:53:31 +03:00
ustrcpy ( local_source , reduced ) ; // Cannot contain NUL char
length = ( int ) ustrlen ( local_source ) ;
2019-10-17 12:06:21 +03:00
}
} else {
2021-07-06 21:53:31 +03:00
return error_tag ( symbol , ZINT_ERROR_INVALID_OPTION , " 220: Selected symbology does not support GS1 mode " ) ;
2019-10-17 12:06:21 +03:00
}
}
2021-07-06 21:53:31 +03:00
error_number = extended_or_reduced_charset ( symbol , local_source , length ) ;
2017-10-23 22:37:52 +03:00
2019-11-27 19:16:14 +03:00
if ( ( error_number = = ZINT_ERROR_INVALID_DATA ) & & symbol - > eci = = 0 & & supports_eci ( symbol - > symbology )
& & ( symbol - > input_mode & 0x07 ) = = UNICODE_MODE ) {
2016-08-16 14:43:41 +03:00
/* Try another ECI mode */
2021-07-06 21:53:31 +03:00
symbol - > eci = get_best_eci ( local_source , length ) ;
2021-03-21 20:35:52 +03:00
if ( symbol - > eci ! = 0 ) {
2021-07-06 21:53:31 +03:00
error_number = extended_or_reduced_charset ( symbol , local_source , length ) ;
/* Inclusion of ECI more noteworthy than other warnings, so overwrite (if any) */
if ( error_number < ZINT_ERROR ) {
2021-03-21 20:35:52 +03:00
error_number = ZINT_WARN_USES_ECI ;
if ( ! ( symbol - > debug & ZINT_DEBUG_TEST ) ) {
2021-07-06 21:53:31 +03:00
sprintf ( symbol - > errtxt , " 222: Encoded data includes ECI %d " , symbol - > eci ) ;
2021-03-21 20:35:52 +03:00
}
if ( symbol - > debug & ZINT_DEBUG_PRINT ) printf ( " Added ECI %d \n " , symbol - > eci ) ;
2019-11-27 19:16:14 +03:00
}
2018-01-21 17:33:54 +03:00
}
2016-08-16 14:43:41 +03:00
}
2016-02-20 13:50:15 +03:00
2017-07-26 13:44:47 +03:00
if ( error_number = = 0 ) {
2021-07-06 21:53:31 +03:00
error_number = warn_number ; /* Already tagged */
} else {
error_number = error_tag ( symbol , error_number , NULL ) ;
2016-02-20 13:50:15 +03:00
}
2017-10-23 22:37:52 +03:00
2020-09-30 14:19:12 +03:00
if ( error_number < ZINT_ERROR ) {
2021-06-19 15:11:23 +03:00
if ( symbol - > height < 0.5f ) { /* Absolute minimum */
( void ) set_height ( symbol , 0.0f , 50.0f , 0.0f , 1 /*no_errtxt*/ ) ;
}
2016-09-15 00:34:59 +03:00
}
2017-10-23 22:37:52 +03:00
2016-02-20 13:50:15 +03:00
return error_number ;
2009-11-22 18:38:15 +03:00
}
2021-07-06 21:53:31 +03:00
/* Output a previously encoded symbol to file `symbol->outfile` */
2016-02-20 13:50:15 +03:00
int ZBarcode_Print ( struct zint_symbol * symbol , int rotate_angle ) {
int error_number ;
2021-07-06 21:53:31 +03:00
int len ;
2016-02-20 13:50:15 +03:00
2020-09-30 14:19:12 +03:00
if ( ! symbol ) return ZINT_ERROR_INVALID_DATA ;
2016-02-20 13:50:15 +03:00
switch ( rotate_angle ) {
case 0 :
case 90 :
case 180 :
case 270 :
break ;
default :
2021-07-06 21:53:31 +03:00
return error_tag ( symbol , ZINT_ERROR_INVALID_OPTION , " 223: Invalid rotation angle " ) ;
2021-01-12 22:51:54 +03:00
break ;
2016-02-20 13:50:15 +03:00
}
2016-08-26 14:15:54 +03:00
if ( symbol - > output_options & BARCODE_DOTTY_MODE ) {
2020-09-30 14:19:12 +03:00
if ( ! ( is_dotty ( symbol - > symbology ) ) ) {
2021-07-06 21:53:31 +03:00
return error_tag ( symbol , ZINT_ERROR_INVALID_OPTION , " 224: Selected symbology cannot be rendered as dots " ) ;
2016-08-12 21:30:46 +03:00
}
}
2016-09-06 00:06:50 +03:00
2021-07-06 21:53:31 +03:00
len = ( int ) strlen ( symbol - > outfile ) ;
if ( len > 3 ) {
2017-09-10 18:03:09 +03:00
char output [ 4 ] ;
2021-07-06 21:53:31 +03:00
output [ 0 ] = symbol - > outfile [ len - 3 ] ;
output [ 1 ] = symbol - > outfile [ len - 2 ] ;
output [ 2 ] = symbol - > outfile [ len - 1 ] ;
2016-02-20 13:50:15 +03:00
output [ 3 ] = ' \0 ' ;
2021-01-12 22:51:54 +03:00
to_upper ( ( unsigned char * ) output ) ;
2009-11-22 18:38:15 +03:00
2016-02-20 13:50:15 +03:00
if ( ! ( strcmp ( output , " PNG " ) ) ) {
2016-07-20 01:02:39 +03:00
error_number = plot_raster ( symbol , rotate_angle , OUT_PNG_FILE ) ;
2021-01-12 22:51:54 +03:00
} else if ( ! ( strcmp ( output , " BMP " ) ) ) {
2016-07-20 01:02:39 +03:00
error_number = plot_raster ( symbol , rotate_angle , OUT_BMP_FILE ) ;
2021-01-12 22:51:54 +03:00
} else if ( ! ( strcmp ( output , " PCX " ) ) ) {
2016-09-06 00:06:50 +03:00
error_number = plot_raster ( symbol , rotate_angle , OUT_PCX_FILE ) ;
2021-01-12 22:51:54 +03:00
} else if ( ! ( strcmp ( output , " GIF " ) ) ) {
2016-07-28 21:58:33 +03:00
error_number = plot_raster ( symbol , rotate_angle , OUT_GIF_FILE ) ;
2021-01-12 22:51:54 +03:00
} else if ( ! ( strcmp ( output , " TIF " ) ) ) {
2016-12-30 23:25:58 +03:00
error_number = plot_raster ( symbol , rotate_angle , OUT_TIF_FILE ) ;
2021-01-12 22:51:54 +03:00
} else if ( ! ( strcmp ( output , " TXT " ) ) ) {
2016-02-20 13:50:15 +03:00
error_number = dump_plot ( symbol ) ;
2021-01-12 22:51:54 +03:00
} else if ( ! ( strcmp ( output , " EPS " ) ) ) {
2019-12-01 17:09:46 +03:00
error_number = plot_vector ( symbol , rotate_angle , OUT_EPS_FILE ) ;
2021-01-12 22:51:54 +03:00
} else if ( ! ( strcmp ( output , " SVG " ) ) ) {
2018-06-18 04:36:40 +03:00
error_number = plot_vector ( symbol , rotate_angle , OUT_SVG_FILE ) ;
2021-01-12 22:51:54 +03:00
} else if ( ! ( strcmp ( output , " EMF " ) ) ) {
2018-06-18 04:36:40 +03:00
error_number = plot_vector ( symbol , rotate_angle , OUT_EMF_FILE ) ;
2021-01-12 22:51:54 +03:00
2016-02-20 13:50:15 +03:00
} else {
2021-07-06 21:53:31 +03:00
return error_tag ( symbol , ZINT_ERROR_INVALID_OPTION , " 225: Unknown output format " ) ;
2016-02-20 13:50:15 +03:00
}
} else {
2021-07-06 21:53:31 +03:00
return error_tag ( symbol , ZINT_ERROR_INVALID_OPTION , " 226: Unknown output format " ) ;
2016-02-20 13:50:15 +03:00
}
2021-07-06 21:53:31 +03:00
return error_tag ( symbol , error_number , NULL ) ;
2009-11-22 18:38:15 +03:00
}
2021-07-13 00:27:16 +03:00
/* Output a previously encoded symbol to memory as raster (`symbol->bitmap`) */
2016-02-20 13:50:15 +03:00
int ZBarcode_Buffer ( struct zint_symbol * symbol , int rotate_angle ) {
int error_number ;
2020-09-30 14:19:12 +03:00
if ( ! symbol ) return ZINT_ERROR_INVALID_DATA ;
2016-02-20 13:50:15 +03:00
switch ( rotate_angle ) {
case 0 :
case 90 :
case 180 :
case 270 :
break ;
default :
2021-07-06 21:53:31 +03:00
return error_tag ( symbol , ZINT_ERROR_INVALID_OPTION , " 228: Invalid rotation angle " ) ;
2021-01-12 22:51:54 +03:00
break ;
2016-02-20 13:50:15 +03:00
}
2020-06-04 20:45:25 +03:00
if ( symbol - > output_options & BARCODE_DOTTY_MODE ) {
2020-09-30 14:19:12 +03:00
if ( ! ( is_dotty ( symbol - > symbology ) ) ) {
2021-07-06 21:53:31 +03:00
return error_tag ( symbol , ZINT_ERROR_INVALID_OPTION , " 237: Selected symbology cannot be rendered as dots " ) ;
2020-06-04 20:45:25 +03:00
}
}
2016-10-02 12:45:47 +03:00
error_number = plot_raster ( symbol , rotate_angle , OUT_BUFFER ) ;
2021-07-06 21:53:31 +03:00
return error_tag ( symbol , error_number , NULL ) ;
2009-11-22 18:38:15 +03:00
}
2021-07-13 00:27:16 +03:00
/* Output a previously encoded symbol to memory as vector (`symbol->vector`) */
2018-06-18 04:36:40 +03:00
int ZBarcode_Buffer_Vector ( struct zint_symbol * symbol , int rotate_angle ) {
int error_number ;
2020-09-30 14:19:12 +03:00
if ( ! symbol ) return ZINT_ERROR_INVALID_DATA ;
2018-06-18 04:36:40 +03:00
switch ( rotate_angle ) {
case 0 :
case 90 :
case 180 :
case 270 :
break ;
default :
2021-07-06 21:53:31 +03:00
return error_tag ( symbol , ZINT_ERROR_INVALID_OPTION , " 219: Invalid rotation angle " ) ;
2021-01-12 22:51:54 +03:00
break ;
2018-06-18 04:36:40 +03:00
}
2020-06-04 20:45:25 +03:00
if ( symbol - > output_options & BARCODE_DOTTY_MODE ) {
2020-09-30 14:19:12 +03:00
if ( ! ( is_dotty ( symbol - > symbology ) ) ) {
2021-07-06 21:53:31 +03:00
return error_tag ( symbol , ZINT_ERROR_INVALID_OPTION , " 238: Selected symbology cannot be rendered as dots " ) ;
2020-06-04 20:45:25 +03:00
}
}
2018-06-18 04:36:40 +03:00
error_number = plot_vector ( symbol , rotate_angle , OUT_BUFFER ) ;
2021-07-06 21:53:31 +03:00
return error_tag ( symbol , error_number , NULL ) ;
2018-06-18 04:36:40 +03:00
}
2021-07-06 21:53:31 +03:00
/* Encode and output a symbol to file `symbol->outfile` */
2021-07-13 00:27:16 +03:00
int ZBarcode_Encode_and_Print ( struct zint_symbol * symbol , const unsigned char * source , int length , int rotate_angle ) {
2016-02-20 13:50:15 +03:00
int error_number ;
2018-01-21 21:10:42 +03:00
int first_err ;
2012-12-31 17:41:59 +04:00
2021-07-06 21:53:31 +03:00
error_number = ZBarcode_Encode ( symbol , source , length ) ;
2020-09-30 14:19:12 +03:00
if ( error_number > = ZINT_ERROR ) {
2016-02-20 13:50:15 +03:00
return error_number ;
}
2009-11-22 18:38:15 +03:00
2018-01-21 21:10:42 +03:00
first_err = error_number ;
2016-02-20 13:50:15 +03:00
error_number = ZBarcode_Print ( symbol , rotate_angle ) ;
2018-01-21 21:10:42 +03:00
if ( error_number = = 0 ) {
error_number = first_err ;
}
2016-02-20 13:50:15 +03:00
return error_number ;
2009-11-22 18:38:15 +03:00
}
2021-07-13 00:27:16 +03:00
/* Encode and output a symbol to memory as raster (`symbol->bitmap`) */
int ZBarcode_Encode_and_Buffer ( struct zint_symbol * symbol , const unsigned char * source , int length ,
int rotate_angle ) {
2016-02-20 13:50:15 +03:00
int error_number ;
2018-01-21 21:10:42 +03:00
int first_err ;
2012-12-31 17:41:59 +04:00
2021-07-06 21:53:31 +03:00
error_number = ZBarcode_Encode ( symbol , source , length ) ;
2020-09-30 14:19:12 +03:00
if ( error_number > = ZINT_ERROR ) {
2016-02-20 13:50:15 +03:00
return error_number ;
}
2009-11-22 18:38:15 +03:00
2018-01-21 21:10:42 +03:00
first_err = error_number ;
2016-02-20 13:50:15 +03:00
error_number = ZBarcode_Buffer ( symbol , rotate_angle ) ;
2018-01-21 21:10:42 +03:00
if ( error_number = = 0 ) {
error_number = first_err ;
}
2019-12-01 17:09:46 +03:00
2016-02-20 13:50:15 +03:00
return error_number ;
2009-11-22 18:38:15 +03:00
}
2021-07-13 00:27:16 +03:00
/* Encode and output a symbol to memory as vector (`symbol->vector`) */
int ZBarcode_Encode_and_Buffer_Vector ( struct zint_symbol * symbol , const unsigned char * source , int length ,
2021-06-19 15:11:23 +03:00
int rotate_angle ) {
2018-06-18 04:36:40 +03:00
int error_number ;
int first_err ;
2021-07-06 21:53:31 +03:00
error_number = ZBarcode_Encode ( symbol , source , length ) ;
2020-09-30 14:19:12 +03:00
if ( error_number > = ZINT_ERROR ) {
2018-06-18 04:36:40 +03:00
return error_number ;
}
first_err = error_number ;
error_number = ZBarcode_Buffer_Vector ( symbol , rotate_angle ) ;
if ( error_number = = 0 ) {
error_number = first_err ;
}
2019-12-01 17:09:46 +03:00
2018-06-18 04:36:40 +03:00
return error_number ;
}
2021-07-06 21:53:31 +03:00
/* Encode a barcode using input data from file `filename` */
2021-07-13 00:27:16 +03:00
int ZBarcode_Encode_File ( struct zint_symbol * symbol , const char * filename ) {
2016-02-20 13:50:15 +03:00
FILE * file ;
2020-12-18 05:36:48 +03:00
int file_opened = 0 ;
2016-02-20 13:50:15 +03:00
unsigned char * buffer ;
2020-03-27 01:17:37 +03:00
long fileLen ;
size_t n ;
2021-06-10 13:15:39 +03:00
size_t nRead = 0 ;
2016-02-20 13:50:15 +03:00
int ret ;
2020-09-30 14:19:12 +03:00
if ( ! symbol ) return ZINT_ERROR_INVALID_DATA ;
if ( ! filename ) {
2021-07-06 21:53:31 +03:00
return error_tag ( symbol , ZINT_ERROR_INVALID_DATA , " 239: Filename NULL " ) ;
2020-09-30 14:19:12 +03:00
}
2016-02-20 13:50:15 +03:00
if ( ! strcmp ( filename , " - " ) ) {
file = stdin ;
2020-12-21 22:30:07 +03:00
fileLen = ZINT_MAX_DATA_LEN ;
2016-02-20 13:50:15 +03:00
} else {
file = fopen ( filename , " rb " ) ;
if ( ! file ) {
2021-06-10 13:15:39 +03:00
sprintf ( symbol - > errtxt , " 229: Unable to read input file (%d: %.30s) " , errno , strerror ( errno ) ) ;
2021-07-06 21:53:31 +03:00
return error_tag ( symbol , ZINT_ERROR_INVALID_DATA , NULL ) ;
2016-02-20 13:50:15 +03:00
}
2020-12-18 05:36:48 +03:00
file_opened = 1 ;
2016-02-20 13:50:15 +03:00
/* Get file length */
fseek ( file , 0 , SEEK_END ) ;
fileLen = ftell ( file ) ;
fseek ( file , 0 , SEEK_SET ) ;
2020-12-23 13:57:24 +03:00
/* On many Linux distros ftell() returns LONG_MAX not -1 on error */
if ( fileLen < = 0 | | fileLen = = LONG_MAX ) {
2016-02-20 13:50:15 +03:00
fclose ( file ) ;
2021-07-06 21:53:31 +03:00
return error_tag ( symbol , ZINT_ERROR_INVALID_DATA , " 235: Input file empty or unseekable " ) ;
2016-02-20 13:50:15 +03:00
}
2020-12-21 22:30:07 +03:00
if ( fileLen > ZINT_MAX_DATA_LEN ) {
2020-03-27 01:17:37 +03:00
fclose ( file ) ;
2021-07-06 21:53:31 +03:00
return error_tag ( symbol , ZINT_ERROR_TOO_LONG , " 230: Input file too long " ) ;
2020-03-27 01:17:37 +03:00
}
2016-02-20 13:50:15 +03:00
}
/* Allocate memory */
2021-01-11 21:11:41 +03:00
buffer = ( unsigned char * ) malloc ( fileLen ) ;
2016-02-20 13:50:15 +03:00
if ( ! buffer ) {
2020-12-18 05:36:48 +03:00
if ( file_opened ) {
2016-02-20 13:50:15 +03:00
fclose ( file ) ;
2020-03-27 01:17:37 +03:00
}
2021-07-06 21:53:31 +03:00
return error_tag ( symbol , ZINT_ERROR_MEMORY , " 231: Insufficient memory for file read buffer " ) ;
2016-02-20 13:50:15 +03:00
}
/* Read file contents into buffer */
do {
n = fread ( buffer + nRead , 1 , fileLen - nRead , file ) ;
if ( ferror ( file ) ) {
2021-06-10 13:15:39 +03:00
sprintf ( symbol - > errtxt , " 241: Input file read error (%d: %.30s) " , errno , strerror ( errno ) ) ;
2020-12-18 05:36:48 +03:00
if ( file_opened ) {
2020-03-27 01:17:37 +03:00
fclose ( file ) ;
}
free ( buffer ) ;
2021-07-06 21:53:31 +03:00
return error_tag ( symbol , ZINT_ERROR_INVALID_DATA , NULL ) ;
2016-02-20 13:50:15 +03:00
}
nRead + = n ;
2021-06-10 13:15:39 +03:00
} while ( ! feof ( file ) & & ( 0 < n ) & & ( ( long ) nRead < fileLen ) ) ;
2016-02-20 13:50:15 +03:00
2020-12-18 05:36:48 +03:00
if ( file_opened ) {
2020-03-27 01:17:37 +03:00
fclose ( file ) ;
}
2021-06-27 13:47:55 +03:00
ret = ZBarcode_Encode ( symbol , buffer , ( int ) nRead ) ;
2016-02-20 13:50:15 +03:00
free ( buffer ) ;
return ret ;
2009-11-22 18:38:15 +03:00
}
2021-07-06 21:53:31 +03:00
/* Encode a symbol using input data from file `filename` and output to file `symbol->outfile` */
2021-07-13 00:27:16 +03:00
int ZBarcode_Encode_File_and_Print ( struct zint_symbol * symbol , const char * filename , int rotate_angle ) {
2016-02-20 13:50:15 +03:00
int error_number ;
2018-01-21 21:10:42 +03:00
int first_err ;
2012-12-31 17:41:59 +04:00
2016-02-20 13:50:15 +03:00
error_number = ZBarcode_Encode_File ( symbol , filename ) ;
2020-09-30 14:19:12 +03:00
if ( error_number > = ZINT_ERROR ) {
2016-02-20 13:50:15 +03:00
return error_number ;
}
2019-12-01 17:09:46 +03:00
2018-01-21 21:10:42 +03:00
first_err = error_number ;
error_number = ZBarcode_Print ( symbol , rotate_angle ) ;
if ( error_number = = 0 ) {
error_number = first_err ;
}
2019-12-01 17:09:46 +03:00
2018-01-21 21:10:42 +03:00
return error_number ;
2009-11-22 18:38:15 +03:00
}
2021-07-06 21:53:31 +03:00
/* Encode a symbol using input data from file `filename` and output to memory as raster (`symbol->bitmap`) */
2021-07-13 00:27:16 +03:00
int ZBarcode_Encode_File_and_Buffer ( struct zint_symbol * symbol , char const * filename , int rotate_angle ) {
2016-02-20 13:50:15 +03:00
int error_number ;
2018-01-21 21:10:42 +03:00
int first_err ;
2012-12-31 17:41:59 +04:00
2016-02-20 13:50:15 +03:00
error_number = ZBarcode_Encode_File ( symbol , filename ) ;
2020-09-30 14:19:12 +03:00
if ( error_number > = ZINT_ERROR ) {
2016-02-20 13:50:15 +03:00
return error_number ;
}
2019-12-01 17:09:46 +03:00
2018-01-21 21:10:42 +03:00
first_err = error_number ;
error_number = ZBarcode_Buffer ( symbol , rotate_angle ) ;
if ( error_number = = 0 ) {
error_number = first_err ;
}
2012-12-31 17:41:59 +04:00
2018-01-21 21:10:42 +03:00
return error_number ;
2009-11-22 18:38:15 +03:00
}
2010-05-30 21:25:24 +04:00
2021-07-06 21:53:31 +03:00
/* Encode a symbol using input data from file `filename` and output to memory as vector (`symbol->vector`) */
2021-07-13 00:27:16 +03:00
int ZBarcode_Encode_File_and_Buffer_Vector ( struct zint_symbol * symbol , const char * filename , int rotate_angle ) {
2018-06-18 04:36:40 +03:00
int error_number ;
int first_err ;
error_number = ZBarcode_Encode_File ( symbol , filename ) ;
2020-09-30 14:19:12 +03:00
if ( error_number > = ZINT_ERROR ) {
2018-06-18 04:36:40 +03:00
return error_number ;
}
2019-12-01 17:09:46 +03:00
2018-06-18 04:36:40 +03:00
first_err = error_number ;
error_number = ZBarcode_Buffer_Vector ( symbol , rotate_angle ) ;
if ( error_number = = 0 ) {
error_number = first_err ;
}
return error_number ;
}
2021-07-13 00:27:16 +03:00
int ZBarcode_ValidID ( int symbol_id ) {
/* Checks whether a symbology is supported */
static const unsigned char ids [ 146 ] = {
0 , 1 , 2 , 3 , 4 , 0 , 6 , 7 , 8 , 9 ,
0 , 0 , 0 , 13 , 14 , 0 , 16 , 0 , 18 , 0 ,
20 , 21 , 22 , 23 , 24 , 25 , 0 , 0 , 28 , 29 ,
30 , 31 , 32 , 0 , 34 , 35 , 0 , 37 , 38 , 0 ,
40 , 0 , 0 , 0 , 0 , 0 , 0 , 47 , 0 , 49 ,
50 , 51 , 52 , 53 , 0 , 55 , 56 , 57 , 58 , 0 ,
60 , 0 , 0 , 63 , 0 , 0 , 66 , 67 , 68 , 69 ,
70 , 71 , 72 , 73 , 74 , 75 , 76 , 77 , 0 , 79 ,
80 , 81 , 82 , 0 , 84 , 85 , 86 , 87 , 0 , 89 ,
90 , 0 , 92 , 93 , 0 , 0 , 96 , 97 , 98 , 99 ,
0 , 0 , 102 , 0 , 104 , 0 , 106 , 0 , 108 , 0 ,
110 , 0 , 112 , 0 , 0 , 115 , 116 , 0 , 0 , 0 ,
0 , 121 , 0 , 0 , 0 , 0 , 0 , 0 , 128 , 129 ,
130 , 131 , 132 , 133 , 134 , 135 , 136 , 137 , 138 , 139 ,
140 , 141 , 142 , 143 , 144 , 145 ,
} ;
if ( symbol_id < = 0 | | symbol_id > 145 ) {
return 0 ;
}
return ids [ symbol_id ] ! = 0 ;
}
/* Return the capability flags for symbology `symbol_id` that match `cap_flag` */
unsigned int ZBarcode_Cap ( int symbol_id , unsigned int cap_flag ) {
unsigned int result = 0 ;
if ( ! ZBarcode_ValidID ( symbol_id ) ) {
return 0 ;
}
if ( ( cap_flag & ZINT_CAP_HRT ) & & has_hrt ( symbol_id ) ) {
result | = ZINT_CAP_HRT ;
}
if ( ( cap_flag & ZINT_CAP_STACKABLE ) & & is_stackable ( symbol_id ) ) {
result | = ZINT_CAP_STACKABLE ;
}
if ( ( cap_flag & ZINT_CAP_EXTENDABLE ) & & is_extendable ( symbol_id ) ) {
result | = ZINT_CAP_EXTENDABLE ;
}
if ( ( cap_flag & ZINT_CAP_COMPOSITE ) & & is_composite ( symbol_id ) ) {
result | = ZINT_CAP_COMPOSITE ;
}
if ( ( cap_flag & ZINT_CAP_ECI ) & & supports_eci ( symbol_id ) ) {
result | = ZINT_CAP_ECI ;
}
if ( ( cap_flag & ZINT_CAP_GS1 ) & & gs1_compliant ( symbol_id ) ) {
result | = ZINT_CAP_GS1 ;
}
if ( ( cap_flag & ZINT_CAP_DOTTY ) & & is_dotty ( symbol_id ) ) {
result | = ZINT_CAP_DOTTY ;
}
2021-09-24 15:21:24 +03:00
if ( cap_flag & ZINT_CAP_QUIET_ZONES ) {
switch ( symbol_id ) { /* See `quiet_zones()` in "output.c" */
case BARCODE_CODE16K :
case BARCODE_CODE49 :
case BARCODE_CODABLOCKF :
case BARCODE_HIBC_BLOCKF :
case BARCODE_ITF14 :
case BARCODE_EANX :
case BARCODE_EANX_CHK :
case BARCODE_EANX_CC :
case BARCODE_ISBNX :
case BARCODE_UPCA :
case BARCODE_UPCA_CHK :
case BARCODE_UPCA_CC :
case BARCODE_UPCE :
case BARCODE_UPCE_CHK :
case BARCODE_UPCE_CC :
result | = ZINT_CAP_QUIET_ZONES ;
break ;
}
}
2021-07-13 00:27:16 +03:00
if ( ( cap_flag & ZINT_CAP_FIXED_RATIO ) & & is_fixed_ratio ( symbol_id ) ) {
result | = ZINT_CAP_FIXED_RATIO ;
}
if ( cap_flag & ZINT_CAP_READER_INIT ) {
/* Note does not include HIBC versions */
switch ( symbol_id ) {
case BARCODE_CODE128 : /* Note does not include GS1_128 or NVE18 */
case BARCODE_CODE128B :
case BARCODE_CODE16K :
case BARCODE_CODABLOCKF :
case BARCODE_PDF417 :
case BARCODE_PDF417COMP :
case BARCODE_DATAMATRIX :
case BARCODE_MICROPDF417 :
case BARCODE_AZTEC :
case BARCODE_DOTCODE :
case BARCODE_GRIDMATRIX :
case BARCODE_ULTRA :
result | = ZINT_CAP_READER_INIT ;
break ;
}
}
if ( cap_flag & ZINT_CAP_FULL_MULTIBYTE ) {
switch ( symbol_id ) {
case BARCODE_QRCODE :
case BARCODE_MICROQR :
//case BARCODE_HIBC_QR: Note character set restricted to ASCII subset
//case BARCODE_UPNQR: Note does not use Kanji mode
case BARCODE_RMQR :
case BARCODE_HANXIN :
case BARCODE_GRIDMATRIX :
result | = ZINT_CAP_FULL_MULTIBYTE ;
break ;
}
}
if ( cap_flag & ZINT_CAP_MASK ) {
switch ( symbol_id ) {
case BARCODE_QRCODE :
case BARCODE_MICROQR :
case BARCODE_HANXIN :
case BARCODE_DOTCODE :
result | = ZINT_CAP_MASK ;
break ;
}
}
Add Structured Append support for AZTEC, CODEONE, DATAMATRIX, DOTCODE,
GRIDMATRIX, MAXICODE, MICROPDF417, PDF417, QRCODE, ULTRA
DOTCODE: use pre-calculated generator poly coeffs in Reed-Solomon for
performance improvement
PDF417/MICROPDF417: use common routine pdf417_initial()
GUI: code lines <= 118, shorthand widget_obj(),
shorten calling upcean_addon_gap(), upcean_guard_descent()
various backend: var name debug -> debug_print
2021-09-28 23:42:44 +03:00
if ( cap_flag & ZINT_CAP_STRUCTAPP ) {
switch ( symbol_id ) {
case BARCODE_PDF417 :
case BARCODE_PDF417COMP :
case BARCODE_MAXICODE :
case BARCODE_QRCODE : /* Note does not include MICROQR, UPNQR or rMQR */
case BARCODE_DATAMATRIX :
case BARCODE_MICROPDF417 :
case BARCODE_AZTEC :
case BARCODE_HIBC_DM :
case BARCODE_HIBC_QR :
case BARCODE_HIBC_PDF :
case BARCODE_HIBC_MICPDF :
case BARCODE_HIBC_AZTEC :
case BARCODE_DOTCODE :
case BARCODE_CODEONE :
case BARCODE_GRIDMATRIX :
case BARCODE_ULTRA :
result | = ZINT_CAP_STRUCTAPP ;
break ;
}
}
2021-07-13 00:27:16 +03:00
return result ;
}
2021-07-06 21:53:31 +03:00
/* Return the version of Zint linked to */
2017-07-01 11:06:47 +03:00
int ZBarcode_Version ( ) {
2020-09-04 18:38:11 +03:00
if ( ZINT_VERSION_BUILD ) {
2020-12-21 22:30:07 +03:00
return ( ZINT_VERSION_MAJOR * 10000 ) + ( ZINT_VERSION_MINOR * 100 ) + ZINT_VERSION_RELEASE * 10
+ ZINT_VERSION_BUILD ;
2020-09-04 18:38:11 +03:00
}
2017-07-01 11:06:47 +03:00
return ( ZINT_VERSION_MAJOR * 10000 ) + ( ZINT_VERSION_MINOR * 100 ) + ZINT_VERSION_RELEASE ;
2017-06-13 22:05:35 +03:00
}