2009-10-07 08:03:00 +13:00
/* pdf417.c - Handles PDF417 stacked symbology */
/* Zint - A barcode generating program using libpng
2020-03-29 04:10:53 +13:00
Copyright ( C ) 2008 - 2020 Robin Stuart < rstuart114 @ gmail . com >
2009-10-07 08:03:00 +13:00
Portions Copyright ( C ) 2004 Grandzebu
Bug Fixes thanks to KL Chin < klchin @ users . sourceforge . net >
2013-05-17 05:26:38 +12:00
Redistribution and use in source and binary forms , with or without
modification , are permitted provided that the following conditions
are met :
2016-02-20 23:50:15 +13:00
1. Redistributions of source code must retain the above copyright
notice , this list of conditions and the following disclaimer .
2013-05-17 05:26:38 +12:00
2. Redistributions in binary form must reproduce the above copyright
notice , this list of conditions and the following disclaimer in the
2016-02-20 23:50:15 +13:00
documentation and / or other materials provided with the distribution .
2013-05-17 05:26:38 +12: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
2016-02-20 23:50:15 +13:00
without specific prior written permission .
2013-05-17 05:26:38 +12: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
2016-02-20 23:50:15 +13:00
OUT OF THE USE OF THIS SOFTWARE , EVEN IF ADVISED OF THE POSSIBILITY OF
2013-05-17 05:26:38 +12:00
SUCH DAMAGE .
2016-02-20 23:50:15 +13:00
*/
2019-12-19 13:37:55 +13:00
/* vim: set ts=4 sw=4 et : */
2009-10-07 08:03:00 +13:00
/* This code is adapted from "Code barre PDF 417 / PDF 417 barcode" v2.5.0
which is Copyright ( C ) 2004 ( Grandzebu ) .
The original code can be downloaded from http : //grandzebu.net/index.php */
/* NOTE: symbol->option_1 is used to specify the security level (i.e. control the
number of check codewords )
symbol - > option_2 is used to adjust the width of the resulting symbol ( i . e . the
number of codeword columns not including row start and end data ) */
# include <stdio.h>
# include <math.h>
# ifndef _MSC_VER
# include <stdint.h>
# else
2016-02-20 23:50:15 +13:00
# include <malloc.h>
2009-10-07 08:03:00 +13:00
# include "ms_stdint.h"
2016-02-20 23:50:15 +13:00
# endif
2020-11-02 07:32:55 +13:00
# include <assert.h>
2009-10-07 08:03:00 +13:00
# include "common.h"
2020-11-02 07:32:55 +13:00
# include "pdf417.h"
# define TEX 900
# define BYT 901
# define NUM 902
2016-02-20 23:50:15 +13:00
/*
2009-10-07 08:03:00 +13:00
Three figure numbers in comments give the location of command equivalents in the
original Visual Basic source code file pdf417 . frm
this code retains some original ( French ) procedure and variable names to ease conversion */
/* text mode processing tables */
2020-11-02 07:32:55 +13:00
static const char asciix [ 127 ] = {
0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 12 , 8 , 0 , 0 , 12 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 ,
2016-02-20 23:50:15 +13:00
7 , 8 , 8 , 4 , 12 , 4 , 4 , 8 , 8 , 8 , 12 , 4 , 12 , 12 , 12 , 12 , 4 , 4 , 4 , 4 , 4 , 4 , 4 , 4 ,
4 , 4 , 12 , 8 , 8 , 4 , 8 , 8 , 8 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 ,
1 , 1 , 1 , 1 , 8 , 8 , 8 , 4 , 8 , 8 , 2 , 2 , 2 , 2 , 2 , 2 , 2 , 2 , 2 , 2 , 2 , 2 , 2 , 2 , 2 , 2 , 2 , 2 , 2 , 2 , 2 , 2 ,
2 , 2 , 2 , 2 , 8 , 8 , 8 , 8
} ;
2009-10-07 08:03:00 +13:00
2020-11-02 07:32:55 +13:00
static const char asciiy [ 127 ] = {
0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 12 , 15 , 0 , 0 , 11 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 ,
2016-02-20 23:50:15 +13:00
26 , 10 , 20 , 15 , 18 , 21 , 10 , 28 , 23 , 24 , 22 , 20 , 13 , 16 , 17 , 19 , 0 , 1 , 2 , 3 ,
4 , 5 , 6 , 7 , 8 , 9 , 14 , 0 , 1 , 23 , 2 , 25 , 3 , 0 , 1 , 2 , 3 , 4 , 5 , 6 , 7 , 8 , 9 , 10 , 11 , 12 , 13 , 14 , 15 ,
16 , 17 , 18 , 19 , 20 , 21 , 22 , 23 , 24 , 25 , 4 , 5 , 6 , 24 , 7 , 8 , 0 , 1 , 2 , 3 , 4 , 5 , 6 , 7 , 8 , 9 , 10 ,
11 , 12 , 13 , 14 , 15 , 16 , 17 , 18 , 19 , 20 , 21 , 22 , 23 , 24 , 25 , 26 , 21 , 27 , 9
} ;
2009-10-07 08:03:00 +13:00
/* Automatic sizing table */
2017-08-09 20:13:37 +12:00
static const char MicroAutosize [ 56 ] = {
2016-02-20 23:50:15 +13:00
4 , 6 , 7 , 8 , 10 , 12 , 13 , 14 , 16 , 18 , 19 , 20 , 24 , 29 , 30 , 33 , 34 , 37 , 39 , 46 , 54 , 58 , 70 , 72 , 82 , 90 , 108 , 126 ,
1 , 14 , 2 , 7 , 3 , 25 , 8 , 16 , 5 , 17 , 9 , 6 , 10 , 11 , 28 , 12 , 19 , 13 , 29 , 20 , 30 , 21 , 22 , 31 , 23 , 32 , 33 , 34
2009-10-07 08:03:00 +13:00
} ;
2020-03-29 04:10:53 +13:00
# define PDF417_MAX_LEN 2710 /* ISO/IEC 15438:2015 5.1.1 c) 3) Max possible number of characters at error correction level 0 (Numeric Compaction mode) */
# define MICRO_PDF417_MAX_LEN 366 /* ISO/IEC 24728:2006 5.1.1 c) 3) Max possible number of characters (Numeric Compaction mode) */
2009-10-07 08:03:00 +13:00
/* 866 */
2020-11-02 07:32:55 +13:00
static int quelmode ( unsigned char codeascii ) {
if ( ( codeascii < = ' 9 ' ) & & ( codeascii > = ' 0 ' ) ) {
return NUM ;
2019-09-05 08:34:20 +12:00
}
2020-11-02 07:32:55 +13:00
if ( codeascii < 127 & & asciix [ codeascii ] ) {
return TEX ;
2016-02-20 23:50:15 +13:00
}
/* 876 */
2009-10-07 08:03:00 +13:00
2020-11-02 07:32:55 +13:00
return BYT ;
2009-10-07 08:03:00 +13:00
}
/* 844 */
2020-07-18 03:39:01 +12:00
static void regroupe ( int liste [ 2 ] [ PDF417_MAX_LEN ] , int * indexliste ) {
2016-02-20 23:50:15 +13:00
/* bring together same type blocks */
if ( * ( indexliste ) > 1 ) {
2017-09-11 03:03:09 +12:00
int i = 1 ;
2016-02-20 23:50:15 +13:00
while ( i < * ( indexliste ) ) {
if ( liste [ 1 ] [ i - 1 ] = = liste [ 1 ] [ i ] ) {
2017-10-24 08:34:31 +13:00
int j ;
2016-02-20 23:50:15 +13:00
/* bring together */
liste [ 0 ] [ i - 1 ] = liste [ 0 ] [ i - 1 ] + liste [ 0 ] [ i ] ;
2017-10-17 06:26:54 +13:00
j = i + 1 ;
2016-02-20 23:50:15 +13:00
/* decreace the list */
while ( j < * ( indexliste ) ) {
liste [ 0 ] [ j - 1 ] = liste [ 0 ] [ j ] ;
liste [ 1 ] [ j - 1 ] = liste [ 1 ] [ j ] ;
j + + ;
}
* ( indexliste ) = * ( indexliste ) - 1 ;
i - - ;
}
i + + ;
}
}
/* 865 */
2009-10-07 08:03:00 +13:00
}
/* 478 */
2020-07-18 03:39:01 +12:00
static void pdfsmooth ( int liste [ 2 ] [ PDF417_MAX_LEN ] , int * indexliste ) {
2016-02-20 23:50:15 +13:00
int i , crnt , last , next , length ;
for ( i = 0 ; i < * ( indexliste ) ; i + + ) {
crnt = liste [ 1 ] [ i ] ;
length = liste [ 0 ] [ i ] ;
if ( i ! = 0 ) {
last = liste [ 1 ] [ i - 1 ] ;
} else {
last = FALSE ;
}
if ( i ! = * ( indexliste ) - 1 ) {
next = liste [ 1 ] [ i + 1 ] ;
} else {
next = FALSE ;
}
if ( crnt = = NUM ) {
if ( i = = 0 ) {
/* first block */
if ( * ( indexliste ) > 1 ) {
/* and there are others */
if ( ( next = = TEX ) & & ( length < 8 ) ) {
liste [ 1 ] [ i ] = TEX ;
}
if ( ( next = = BYT ) & & ( length = = 1 ) ) {
liste [ 1 ] [ i ] = BYT ;
}
}
} else {
if ( i = = * ( indexliste ) - 1 ) {
/* last block */
if ( ( last = = TEX ) & & ( length < 7 ) ) {
liste [ 1 ] [ i ] = TEX ;
}
if ( ( last = = BYT ) & & ( length = = 1 ) ) {
liste [ 1 ] [ i ] = BYT ;
}
} else {
/* not first or last block */
if ( ( ( last = = BYT ) & & ( next = = BYT ) ) & & ( length < 4 ) ) {
liste [ 1 ] [ i ] = BYT ;
}
if ( ( ( last = = BYT ) & & ( next = = TEX ) ) & & ( length < 4 ) ) {
liste [ 1 ] [ i ] = TEX ;
}
if ( ( ( last = = TEX ) & & ( next = = BYT ) ) & & ( length < 5 ) ) {
liste [ 1 ] [ i ] = TEX ;
}
if ( ( ( last = = TEX ) & & ( next = = TEX ) ) & & ( length < 8 ) ) {
liste [ 1 ] [ i ] = TEX ;
}
}
}
}
}
2020-07-18 03:39:01 +12:00
regroupe ( liste , indexliste ) ;
2016-02-20 23:50:15 +13:00
/* 520 */
for ( i = 0 ; i < * ( indexliste ) ; i + + ) {
crnt = liste [ 1 ] [ i ] ;
length = liste [ 0 ] [ i ] ;
if ( i ! = 0 ) {
last = liste [ 1 ] [ i - 1 ] ;
} else {
last = FALSE ;
}
if ( i ! = * ( indexliste ) - 1 ) {
next = liste [ 1 ] [ i + 1 ] ;
} else {
next = FALSE ;
}
if ( ( crnt = = TEX ) & & ( i > 0 ) ) {
/* not the first */
if ( i = = * ( indexliste ) - 1 ) {
/* the last one */
if ( ( last = = BYT ) & & ( length = = 1 ) ) {
liste [ 1 ] [ i ] = BYT ;
}
} else {
/* not the last one */
if ( ( ( last = = BYT ) & & ( next = = BYT ) ) & & ( length < 5 ) ) {
liste [ 1 ] [ i ] = BYT ;
}
if ( ( ( ( last = = BYT ) & & ( next ! = BYT ) ) | | ( ( last ! = BYT )
& & ( next = = BYT ) ) ) & & ( length < 3 ) ) {
liste [ 1 ] [ i ] = BYT ;
}
}
}
}
/* 540 */
2020-07-18 03:39:01 +12:00
regroupe ( liste , indexliste ) ;
2009-10-07 08:03:00 +13:00
}
/* 547 */
2020-07-18 03:39:01 +12:00
static void textprocess ( int * chainemc , int * mclength , char chaine [ ] , int start , int length , int is_micro ) {
int j , indexlistet , curtable , listet [ 2 ] [ PDF417_MAX_LEN ] = { { 0 } } , chainet [ PDF417_MAX_LEN ] , wnet ;
2016-02-20 23:50:15 +13:00
wnet = 0 ;
/* listet will contain the table numbers and the value of each characters */
for ( indexlistet = 0 ; indexlistet < length ; indexlistet + + ) {
2020-11-02 07:32:55 +13:00
int codeascii = chaine [ start + indexlistet ] ;
listet [ 0 ] [ indexlistet ] = asciix [ codeascii ] ;
listet [ 1 ] [ indexlistet ] = asciiy [ codeascii ] ;
2016-02-20 23:50:15 +13:00
}
/* 570 */
curtable = 1 ; /* default table */
for ( j = 0 ; j < length ; j + + ) {
if ( listet [ 0 ] [ j ] & curtable ) {
/* The character is in the current table */
chainet [ wnet ] = listet [ 1 ] [ j ] ;
wnet + + ;
} else {
/* Obliged to change table */
int flag = FALSE ; /* True if we change table for only one character */
if ( j = = ( length - 1 ) ) {
flag = TRUE ;
} else {
if ( ! ( listet [ 0 ] [ j ] & listet [ 0 ] [ j + 1 ] ) ) {
flag = TRUE ;
}
}
if ( flag ) {
/* we change only one character - look for temporary switch */
if ( ( listet [ 0 ] [ j ] & 1 ) & & ( curtable = = 2 ) ) { /* T_UPP */
chainet [ wnet ] = 27 ;
chainet [ wnet + 1 ] = listet [ 1 ] [ j ] ;
wnet + = 2 ;
2020-11-02 07:32:55 +13:00
} else if ( listet [ 0 ] [ j ] & 8 ) { /* T_PUN (T_PUN and T_UPP not both possible) */
2016-02-20 23:50:15 +13:00
chainet [ wnet ] = 29 ;
chainet [ wnet + 1 ] = listet [ 1 ] [ j ] ;
wnet + = 2 ;
2020-11-02 07:32:55 +13:00
} else {
2016-02-20 23:50:15 +13:00
/* No temporary switch available */
flag = FALSE ;
}
}
/* 599 */
if ( ! ( flag ) ) {
int newtable ;
if ( j = = ( length - 1 ) ) {
newtable = listet [ 0 ] [ j ] ;
} else {
if ( ! ( listet [ 0 ] [ j ] & listet [ 0 ] [ j + 1 ] ) ) {
newtable = listet [ 0 ] [ j ] ;
} else {
newtable = listet [ 0 ] [ j ] & listet [ 0 ] [ j + 1 ] ;
}
}
/* Maintain the first if several tables are possible */
2020-11-02 07:32:55 +13:00
if ( newtable = = 7 ) {
newtable = 1 ;
} else if ( newtable = = 12 ) {
newtable = 4 ;
2016-02-20 23:50:15 +13:00
}
/* 619 - select the switch */
switch ( curtable ) {
case 1 :
switch ( newtable ) {
case 2 : chainet [ wnet ] = 27 ;
wnet + + ;
break ;
case 4 : chainet [ wnet ] = 28 ;
wnet + + ;
break ;
case 8 : chainet [ wnet ] = 28 ;
wnet + + ;
chainet [ wnet ] = 25 ;
wnet + + ;
break ;
}
break ;
case 2 :
switch ( newtable ) {
case 1 : chainet [ wnet ] = 28 ;
wnet + + ;
chainet [ wnet ] = 28 ;
wnet + + ;
break ;
case 4 : chainet [ wnet ] = 28 ;
wnet + + ;
break ;
case 8 : chainet [ wnet ] = 28 ;
wnet + + ;
chainet [ wnet ] = 25 ;
wnet + + ;
break ;
}
break ;
case 4 :
switch ( newtable ) {
case 1 : chainet [ wnet ] = 28 ;
wnet + + ;
break ;
case 2 : chainet [ wnet ] = 27 ;
wnet + + ;
break ;
case 8 : chainet [ wnet ] = 25 ;
wnet + + ;
break ;
}
break ;
case 8 :
switch ( newtable ) {
case 1 : chainet [ wnet ] = 29 ;
wnet + + ;
break ;
case 2 : chainet [ wnet ] = 29 ;
wnet + + ;
chainet [ wnet ] = 27 ;
wnet + + ;
break ;
case 4 : chainet [ wnet ] = 29 ;
wnet + + ;
chainet [ wnet ] = 28 ;
wnet + + ;
break ;
}
break ;
}
curtable = newtable ;
/* 659 - at last we add the character */
chainet [ wnet ] = listet [ 1 ] [ j ] ;
wnet + + ;
}
}
}
/* 663 */
if ( wnet & 1 ) {
chainet [ wnet ] = 29 ;
wnet + + ;
}
/* Now translate the string chainet into codewords */
2020-07-18 03:39:01 +12:00
/* Default mode for PDF417 is Text Compaction Alpha (ISO/IEC 1543:2015 5.4.2.1), and for MICROPDF417 is Byte Compaction
* ( ISO / IEC 24728 : 2006 5.4 .3 ) , so only add flag if not first codeword or is MICROPDF417 */
if ( * mclength | | is_micro ) {
chainemc [ ( * mclength ) + + ] = 900 ;
}
2016-02-20 23:50:15 +13:00
for ( j = 0 ; j < wnet ; j + = 2 ) {
int cw_number ;
cw_number = ( 30 * chainet [ j ] ) + chainet [ j + 1 ] ;
2020-07-18 03:39:01 +12:00
chainemc [ ( * mclength ) + + ] = cw_number ;
2016-02-20 23:50:15 +13:00
}
2009-10-07 08:03:00 +13:00
}
/* 671 */
2020-07-18 03:39:01 +12:00
INTERNAL void byteprocess ( int * chainemc , int * mclength , unsigned char chaine [ ] , int start , int length , int debug ) {
2013-01-01 02:41:59 +13:00
2016-02-20 23:50:15 +13:00
if ( debug ) printf ( " \n Entering byte mode at position %d \n " , start ) ;
if ( length = = 1 ) {
chainemc [ ( * mclength ) + + ] = 913 ;
chainemc [ ( * mclength ) + + ] = chaine [ start ] ;
if ( debug ) {
printf ( " 913 %d \n " , chainemc [ * mclength - 1 ] ) ;
}
} else {
2017-10-24 08:34:31 +13:00
int len ;
2016-02-20 23:50:15 +13:00
/* select the switch for multiple of 6 bytes */
if ( length % 6 = = 0 ) {
chainemc [ ( * mclength ) + + ] = 924 ;
if ( debug ) printf ( " 924 " ) ;
} else {
2020-07-18 03:39:01 +12:00
/* Default mode for MICROPDF417 is Byte Compaction (ISO/IEC 24728:2006 5.4.3), but not emitting it depends on whether
* an ECI has been emitted previously ( or not ) it appears , so simpler and safer to always emit it . */
2016-02-20 23:50:15 +13:00
chainemc [ ( * mclength ) + + ] = 901 ;
if ( debug ) printf ( " 901 " ) ;
}
2017-10-17 06:26:54 +13:00
len = 0 ;
2017-09-11 03:03:09 +12:00
2017-10-24 08:34:31 +13:00
while ( len < length ) {
2020-03-29 04:10:53 +13:00
uint64_t total ;
2017-09-11 03:03:09 +12:00
unsigned int chunkLen = length - len ;
2020-07-18 03:39:01 +12:00
if ( 6 < = chunkLen ) { /* Take groups of 6 */
2016-02-20 23:50:15 +13:00
chunkLen = 6 ;
len + = chunkLen ;
2017-10-17 06:26:54 +13:00
total = 0 ;
2009-10-07 08:03:00 +13:00
2016-02-20 23:50:15 +13:00
while ( chunkLen - - ) {
2017-09-11 03:03:09 +12:00
uint64_t mantisa = chaine [ start + + ] ;
2020-07-18 03:39:01 +12:00
total | = mantisa < < ( chunkLen * 8 ) ;
2016-02-20 23:50:15 +13:00
}
2009-10-07 08:03:00 +13:00
2016-02-20 23:50:15 +13:00
chunkLen = 5 ;
2009-10-07 08:03:00 +13:00
2016-02-20 23:50:15 +13:00
while ( chunkLen - - ) {
chainemc [ * mclength + chunkLen ] = ( int ) ( total % 900 ) ;
total / = 900 ;
}
* mclength + = 5 ;
2020-07-18 03:39:01 +12:00
} else { /* If there remains a group of less than 6 bytes */
2016-02-20 23:50:15 +13:00
len + = chunkLen ;
while ( chunkLen - - ) {
chainemc [ ( * mclength ) + + ] = chaine [ start + + ] ;
}
}
}
}
2009-10-07 08:03:00 +13:00
}
/* 712 */
2019-12-19 13:37:55 +13:00
static void numbprocess ( int * chainemc , int * mclength , char chaine [ ] , int start , int length ) {
2020-07-18 03:39:01 +12:00
int j , loop , dummy [ 50 ] = { 0 } , diviseur , nombre ;
2020-11-02 07:32:55 +13:00
char chainemod [ 46 ] , chainemult [ 46 ] ;
2016-02-20 23:50:15 +13:00
2020-07-18 03:39:01 +12:00
chainemc [ ( * mclength ) + + ] = 902 ;
2016-02-20 23:50:15 +13:00
j = 0 ;
while ( j < length ) {
2017-10-24 08:34:31 +13:00
int longueur ;
2017-09-11 03:03:09 +12:00
int dumlength = 0 ;
2020-11-02 07:32:55 +13:00
int p ;
2017-10-17 06:26:54 +13:00
longueur = length - j ;
2016-02-20 23:50:15 +13:00
if ( longueur > 44 ) {
longueur = 44 ;
}
2020-11-02 07:32:55 +13:00
chainemod [ 0 ] = ' 1 ' ;
2016-02-20 23:50:15 +13:00
for ( loop = 1 ; loop < = longueur ; loop + + ) {
chainemod [ loop ] = chaine [ start + loop + j - 1 ] ;
}
chainemod [ longueur + 1 ] = ' \0 ' ;
do {
diviseur = 900 ;
/* 877 - gosub Modulo */
2020-11-02 07:32:55 +13:00
p = 0 ;
chainemult [ p ] = ' \0 ' ;
2016-02-20 23:50:15 +13:00
nombre = 0 ;
2020-11-02 07:32:55 +13:00
for ( loop = 0 ; chainemod [ loop ] ; loop + + ) {
2016-02-20 23:50:15 +13:00
nombre * = 10 ;
2020-11-02 07:32:55 +13:00
nombre + = ctoi ( chainemod [ loop ] ) ;
2016-02-20 23:50:15 +13:00
if ( nombre < diviseur ) {
2020-11-02 07:32:55 +13:00
if ( p ) {
chainemult [ p + + ] = ' 0 ' ;
2016-02-20 23:50:15 +13:00
}
} else {
2020-11-02 07:32:55 +13:00
chainemult [ p + + ] = ( nombre / diviseur ) + ' 0 ' ;
nombre = nombre % diviseur ;
2016-02-20 23:50:15 +13:00
}
}
2020-11-02 07:32:55 +13:00
chainemult [ p ] = ' \0 ' ;
2016-02-20 23:50:15 +13:00
diviseur = nombre ;
/* return to 723 */
2020-11-02 07:32:55 +13:00
dummy [ dumlength + + ] = diviseur ;
2016-02-20 23:50:15 +13:00
strcpy ( chainemod , chainemult ) ;
2020-11-02 07:32:55 +13:00
} while ( p ) ;
for ( loop = dumlength - 1 ; loop > = 0 ; loop - - ) {
2020-07-18 03:39:01 +12:00
chainemc [ ( * mclength ) + + ] = dummy [ loop ] ;
2016-02-20 23:50:15 +13:00
}
j + = longueur ;
}
2009-10-07 08:03:00 +13:00
}
/* 366 */
2020-11-02 07:32:55 +13:00
static int pdf417 ( struct zint_symbol * symbol , unsigned char chaine [ ] , const int length ) {
int i , k , j , indexchaine , indexliste , mode , longueur , loop , mccorrection [ 520 ] = { 0 } , offset ;
2020-03-29 04:10:53 +13:00
int total , chainemc [ PDF417_MAX_LEN ] , mclength , c1 , c2 , c3 , dummy [ 35 ] , calcheight ;
2020-07-18 03:39:01 +12:00
int liste [ 2 ] [ PDF417_MAX_LEN ] = { { 0 } } ;
2017-06-10 09:00:22 +12:00
char pattern [ 580 ] ;
2020-10-01 00:19:12 +13:00
int error_number = 0 ;
2020-07-18 03:39:01 +12:00
int debug = symbol - > debug & ZINT_DEBUG_PRINT ;
2016-02-20 23:50:15 +13:00
2020-03-29 04:10:53 +13:00
if ( length > PDF417_MAX_LEN ) {
2020-11-08 08:26:10 +13:00
symbol - > err_origin = 463 ;
strcpy ( symbol - > errtxt , _ ( " Input too long " ) ) ;
2020-07-18 03:39:01 +12:00
return ZINT_ERROR_TOO_LONG ;
2020-03-29 04:10:53 +13:00
}
2016-02-20 23:50:15 +13:00
/* 456 */
indexliste = 0 ;
indexchaine = 0 ;
mode = quelmode ( chaine [ indexchaine ] ) ;
/* 463 */
do {
liste [ 1 ] [ indexliste ] = mode ;
2020-11-02 07:32:55 +13:00
while ( ( liste [ 1 ] [ indexliste ] = = mode ) & & ( indexchaine < length ) ) {
2016-02-20 23:50:15 +13:00
liste [ 0 ] [ indexliste ] + + ;
indexchaine + + ;
mode = quelmode ( chaine [ indexchaine ] ) ;
}
indexliste + + ;
2020-11-02 07:32:55 +13:00
} while ( indexchaine < length ) ;
2016-02-20 23:50:15 +13:00
/* 474 */
2020-07-18 03:39:01 +12:00
pdfsmooth ( liste , & indexliste ) ;
2016-02-20 23:50:15 +13:00
if ( debug ) {
printf ( " Initial block pattern: \n " ) ;
for ( i = 0 ; i < indexliste ; i + + ) {
printf ( " Len: %d Type: " , liste [ 0 ] [ i ] ) ;
switch ( liste [ 1 ] [ i ] ) {
case TEX : printf ( " Text \n " ) ;
break ;
case BYT : printf ( " Byte \n " ) ;
break ;
case NUM : printf ( " Number \n " ) ;
break ;
default : printf ( " ERROR \n " ) ;
break ;
}
}
}
/* 541 - now compress the data */
indexchaine = 0 ;
mclength = 0 ;
2017-10-24 08:37:52 +13:00
2016-02-20 23:50:15 +13:00
if ( symbol - > output_options & READER_INIT ) {
chainemc [ mclength ] = 921 ; /* Reader Initialisation */
mclength + + ;
}
2017-10-24 08:37:52 +13:00
2019-10-07 05:39:54 +13:00
if ( symbol - > eci ! = 0 ) {
2020-11-02 07:32:55 +13:00
if ( symbol - > eci > 811799 ) {
2020-11-08 08:26:10 +13:00
symbol - > err_origin = 472 ;
strcpy ( symbol - > errtxt , _ ( " Invalid ECI " ) ) ;
2020-11-02 07:32:55 +13:00
return ZINT_ERROR_INVALID_OPTION ;
}
2017-06-19 00:00:22 +12:00
/* Encoding ECI assignment number, according to Table 8 */
if ( symbol - > eci < = 899 ) {
chainemc [ mclength ] = 927 ; /* ECI */
mclength + + ;
chainemc [ mclength ] = symbol - > eci ;
mclength + + ;
2020-11-02 07:32:55 +13:00
} else if ( symbol - > eci < = 810899 ) {
2017-06-19 00:00:22 +12:00
chainemc [ mclength ] = 926 ; /* ECI */
mclength + + ;
chainemc [ mclength ] = ( symbol - > eci / 900 ) - 1 ;
mclength + + ;
chainemc [ mclength ] = symbol - > eci % 900 ;
mclength + + ;
2020-11-02 07:32:55 +13:00
} else {
2017-06-19 00:00:22 +12:00
chainemc [ mclength ] = 925 ; /* ECI */
mclength + + ;
chainemc [ mclength ] = symbol - > eci - 810900 ;
mclength + + ;
}
2016-08-16 23:43:41 +12:00
}
2017-10-24 08:37:52 +13:00
2016-02-20 23:50:15 +13:00
for ( i = 0 ; i < indexliste ; i + + ) {
switch ( liste [ 1 ] [ i ] ) {
case TEX : /* 547 - text mode */
2020-07-18 03:39:01 +12:00
textprocess ( chainemc , & mclength , ( char * ) chaine , indexchaine , liste [ 0 ] [ i ] , 0 /*is_micro*/ ) ;
2016-02-20 23:50:15 +13:00
break ;
case BYT : /* 670 - octet stream mode */
2020-07-18 03:39:01 +12:00
byteprocess ( chainemc , & mclength , chaine , indexchaine , liste [ 0 ] [ i ] , debug ) ;
2016-02-20 23:50:15 +13:00
break ;
case NUM : /* 712 - numeric mode */
2017-09-11 03:03:09 +12:00
numbprocess ( chainemc , & mclength , ( char * ) chaine , indexchaine , liste [ 0 ] [ i ] ) ;
2016-02-20 23:50:15 +13:00
break ;
}
indexchaine = indexchaine + liste [ 0 ] [ i ] ;
}
2020-11-02 07:32:55 +13:00
assert ( mclength > 0 ) ; /* Suppress clang-analyzer-core.uninitialized.Assign warning */
2016-02-20 23:50:15 +13:00
if ( debug ) {
printf ( " \n Compressed data stream: \n " ) ;
for ( i = 0 ; i < mclength ; i + + ) {
printf ( " %d " , chainemc [ i ] ) ;
}
printf ( " \n \n " ) ;
}
/* 752 - Now take care of the number of CWs per row */
if ( symbol - > option_1 < 0 ) {
if ( mclength < = 40 ) {
symbol - > option_1 = 2 ;
2020-11-02 07:32:55 +13:00
} else if ( mclength < = 160 ) {
symbol - > option_1 = 3 ;
} else if ( mclength < = 320 ) {
symbol - > option_1 = 4 ;
} else if ( mclength < = 863 ) {
symbol - > option_1 = 5 ;
} else {
symbol - > option_1 = 6 ;
2016-02-20 23:50:15 +13:00
}
}
k = 1 ;
for ( loop = 1 ; loop < = ( symbol - > option_1 + 1 ) ; loop + + ) {
k * = 2 ;
}
longueur = mclength ;
if ( longueur + k > 928 ) {
/* Enforce maximum codeword limit */
2020-11-08 08:26:10 +13:00
symbol - > err_origin = 464 ;
strcpy ( symbol - > errtxt , _ ( " Input too long " ) ) ;
2020-07-18 03:39:01 +12:00
return ZINT_ERROR_TOO_LONG ;
2016-02-20 23:50:15 +13:00
}
2020-11-02 07:32:55 +13:00
if ( symbol - > option_2 > 30 ) {
symbol - > option_2 = 30 ;
} else if ( symbol - > option_2 < 1 ) {
symbol - > option_2 = ( int ) ( 0.5 + sqrt ( ( longueur + k ) / 3.0 ) ) ;
}
2016-02-20 23:50:15 +13:00
if ( ( ( longueur + k ) / symbol - > option_2 ) > 90 ) {
2020-11-02 07:32:55 +13:00
/* stop the symbol from becoming too high */
symbol - > option_2 = symbol - > option_2 + 1 ;
if ( ( ( longueur + k ) / symbol - > option_2 ) > 90 ) {
2020-11-08 08:26:10 +13:00
symbol - > err_origin = 465 ;
strcpy ( symbol - > errtxt , _ ( " Data too long for specified number of columns " ) ) ;
2020-11-02 07:32:55 +13:00
return ZINT_ERROR_TOO_LONG ;
}
2016-02-20 23:50:15 +13:00
}
/* 781 - Padding calculation */
longueur = mclength + 1 + k ;
i = 0 ;
if ( ( longueur / symbol - > option_2 ) < 3 ) {
i = ( symbol - > option_2 * 3 ) - longueur ; /* A bar code must have at least three rows */
} else {
if ( ( longueur % symbol - > option_2 ) > 0 ) {
i = symbol - > option_2 - ( longueur % symbol - > option_2 ) ;
}
}
/* We add the padding */
while ( i > 0 ) {
chainemc [ mclength ] = 900 ;
mclength + + ;
i - - ;
}
/* we add the length descriptor */
for ( i = mclength ; i > 0 ; i - - ) {
chainemc [ i ] = chainemc [ i - 1 ] ;
}
chainemc [ 0 ] = mclength + 1 ;
mclength + + ;
/* 796 - we now take care of the Reed Solomon codes */
switch ( symbol - > option_1 ) {
case 1 : offset = 2 ;
break ;
case 2 : offset = 6 ;
break ;
case 3 : offset = 14 ;
break ;
case 4 : offset = 30 ;
break ;
case 5 : offset = 62 ;
break ;
case 6 : offset = 126 ;
break ;
case 7 : offset = 254 ;
break ;
case 8 : offset = 510 ;
break ;
default : offset = 0 ;
break ;
}
longueur = mclength ;
for ( i = 0 ; i < longueur ; i + + ) {
total = ( chainemc [ i ] + mccorrection [ k - 1 ] ) % 929 ;
for ( j = k - 1 ; j > 0 ; j - - ) {
mccorrection [ j ] = ( mccorrection [ j - 1 ] + 929 - ( total * coefrs [ offset + j ] ) % 929 ) % 929 ;
}
2020-11-02 07:32:55 +13:00
mccorrection [ 0 ] = ( 929 - ( total * coefrs [ offset ] ) % 929 ) % 929 ;
2016-02-20 23:50:15 +13:00
}
/* we add these codes to the string */
for ( i = k - 1 ; i > = 0 ; i - - ) {
chainemc [ mclength + + ] = mccorrection [ i ] ? 929 - mccorrection [ i ] : 0 ;
}
2019-09-05 08:34:20 +12:00
if ( debug ) {
printf ( " Complete CW string: \n " ) ;
for ( i = 0 ; i < mclength ; i + + ) {
printf ( " %d " , chainemc [ i ] ) ;
}
printf ( " \n " ) ;
}
2020-07-18 03:39:01 +12:00
# ifdef ZINT_TEST
if ( symbol - > debug & ZINT_DEBUG_TEST ) {
debug_test_codeword_dump_int ( symbol , chainemc , mclength ) ;
}
# endif
2016-02-20 23:50:15 +13:00
2020-10-01 00:19:12 +13:00
symbol - > rows = mclength / symbol - > option_2 ;
2020-11-02 07:32:55 +13:00
assert ( symbol - > rows > 0 ) ; /* Suppress clang-analyzer-core.DivideZero warning */
2020-10-01 00:19:12 +13:00
2016-02-20 23:50:15 +13:00
/* 818 - The CW string is finished */
2020-10-01 00:19:12 +13:00
c1 = ( symbol - > rows - 1 ) / 3 ;
c2 = symbol - > option_1 * 3 + ( symbol - > rows - 1 ) % 3 ;
2016-02-20 23:50:15 +13:00
c3 = symbol - > option_2 - 1 ;
/* we now encode each row */
2020-10-01 00:19:12 +13:00
for ( i = 0 ; i < symbol - > rows ; i + + ) {
2020-11-02 07:32:55 +13:00
int p ;
2016-02-20 23:50:15 +13:00
for ( j = 0 ; j < symbol - > option_2 ; j + + ) {
dummy [ j + 1 ] = chainemc [ i * symbol - > option_2 + j ] ;
}
k = ( i / 3 ) * 30 ;
switch ( i % 3 ) {
case 0 :
dummy [ 0 ] = k + c1 ;
dummy [ symbol - > option_2 + 1 ] = k + c3 ;
2017-06-10 09:00:22 +12:00
offset = 0 ; /* cluster(0) */
2016-02-20 23:50:15 +13:00
break ;
case 1 :
dummy [ 0 ] = k + c2 ;
dummy [ symbol - > option_2 + 1 ] = k + c1 ;
2017-06-10 09:00:22 +12:00
offset = 929 ; /* cluster(3) */
2016-02-20 23:50:15 +13:00
break ;
case 2 :
dummy [ 0 ] = k + c3 ;
dummy [ symbol - > option_2 + 1 ] = k + c2 ;
2017-06-10 09:00:22 +12:00
offset = 1858 ; /* cluster(6) */
2016-02-20 23:50:15 +13:00
break ;
}
2020-11-02 07:32:55 +13:00
p = bin_append_posn ( 0x1FEA8 , 17 , pattern , 0 ) ; /* Row start */
2017-06-10 09:00:22 +12:00
for ( j = 0 ; j < = symbol - > option_2 ; j + + ) {
2020-11-02 07:32:55 +13:00
p = bin_append_posn ( pdf_bitpattern [ offset + dummy [ j ] ] , 16 , pattern , p ) ;
pattern [ p + + ] = ' 0 ' ;
2017-06-10 09:00:22 +12:00
}
2017-10-24 08:37:52 +13:00
2020-07-30 09:35:31 +12:00
if ( symbol - > symbology ! = BARCODE_PDF417COMP ) {
2020-11-02 07:32:55 +13:00
p = bin_append_posn ( pdf_bitpattern [ offset + dummy [ j ] ] , 16 , pattern , p ) ;
pattern [ p + + ] = ' 0 ' ;
p = bin_append_posn ( 0x3FA29 , 18 , pattern , p ) ; /* Row Stop */
2020-07-18 03:39:01 +12:00
} else {
2020-11-02 07:32:55 +13:00
pattern [ p + + ] = ' 1 ' ; /* Compact PDF417 Stop pattern */
2016-02-20 23:50:15 +13:00
}
2020-11-02 07:32:55 +13:00
pattern [ p ] = ' \0 ' ;
2017-10-24 08:37:52 +13:00
2020-11-02 07:32:55 +13:00
for ( loop = 0 ; loop < p ; loop + + ) {
2016-02-20 23:50:15 +13:00
if ( pattern [ loop ] = = ' 1 ' ) {
set_module ( symbol , i , loop ) ;
}
}
}
2019-09-04 06:53:01 +12:00
/* Allow user to adjust height of symbol, but enforce minimum row height of 3X */
calcheight = ( int ) ( symbol - > height / i ) ;
if ( calcheight < 3 ) {
calcheight = 3 ;
}
for ( j = 0 ; j < i ; j + + ) {
symbol - > row_height [ j ] = calcheight ;
}
2020-10-01 00:19:12 +13:00
symbol - > width = ( int ) strlen ( pattern ) ;
2016-02-20 23:50:15 +13:00
/* 843 */
2020-10-01 00:19:12 +13:00
return error_number ;
2009-10-07 08:03:00 +13:00
}
/* 345 */
2020-11-02 07:32:55 +13:00
INTERNAL int pdf417enc ( struct zint_symbol * symbol , unsigned char source [ ] , int length ) {
2016-02-20 23:50:15 +13:00
int codeerr , error_number ;
error_number = 0 ;
if ( ( symbol - > option_1 < - 1 ) | | ( symbol - > option_1 > 8 ) ) {
2020-11-08 08:26:10 +13:00
symbol - > err_origin = 460 ;
strcpy ( symbol - > errtxt , _ ( " Security value out of range " ) ) ;
2020-08-22 22:09:57 +12:00
if ( symbol - > warn_level = = WARN_FAIL_ALL ) {
return ZINT_ERROR_INVALID_OPTION ;
} else {
symbol - > option_1 = - 1 ;
error_number = ZINT_WARN_INVALID_OPTION ;
}
2016-02-20 23:50:15 +13:00
}
if ( ( symbol - > option_2 < 0 ) | | ( symbol - > option_2 > 30 ) ) {
2020-11-08 08:26:10 +13:00
symbol - > err_origin = 461 ;
strcpy ( symbol - > errtxt , _ ( " Number of columns out of range " ) ) ;
2020-08-22 22:09:57 +12:00
if ( symbol - > warn_level = = WARN_FAIL_ALL ) {
return ZINT_ERROR_INVALID_OPTION ;
} else {
symbol - > option_2 = 0 ;
error_number = ZINT_WARN_INVALID_OPTION ;
}
2016-02-20 23:50:15 +13:00
}
/* 349 */
codeerr = pdf417 ( symbol , source , length ) ;
/* 352 */
if ( codeerr ! = 0 ) {
2020-07-18 03:39:01 +12:00
error_number = codeerr ;
2016-02-20 23:50:15 +13:00
}
/* 364 */
return error_number ;
2009-10-07 08:03:00 +13:00
}
2016-02-20 23:50:15 +13:00
/* like PDF417 only much smaller! */
2020-11-02 07:32:55 +13:00
INTERNAL int micro_pdf417 ( struct zint_symbol * symbol , unsigned char chaine [ ] , int length ) {
int i , k , j , indexchaine , indexliste , mode , longueur , mccorrection [ 50 ] = { 0 } , offset ;
2020-03-29 04:10:53 +13:00
int total , chainemc [ PDF417_MAX_LEN ] , mclength , dummy [ 5 ] , codeerr ;
2020-07-18 03:39:01 +12:00
int liste [ 2 ] [ PDF417_MAX_LEN ] = { { 0 } } ;
2017-06-10 09:00:22 +12:00
char pattern [ 580 ] ;
2016-02-20 23:50:15 +13:00
int variant , LeftRAPStart , CentreRAPStart , RightRAPStart , StartCluster ;
2019-09-04 06:53:01 +12:00
int LeftRAP , CentreRAP , RightRAP , Cluster , loop , calcheight ;
2020-07-18 03:39:01 +12:00
int debug = symbol - > debug & ZINT_DEBUG_PRINT ;
2016-02-20 23:50:15 +13:00
2020-03-29 04:10:53 +13:00
if ( length > MICRO_PDF417_MAX_LEN ) {
2020-11-08 08:26:10 +13:00
symbol - > err_origin = 474 ;
strcpy ( symbol - > errtxt , _ ( " Input too long " ) ) ;
2020-03-29 04:10:53 +13:00
return ZINT_ERROR_TOO_LONG ;
}
2016-02-20 23:50:15 +13:00
/* Encoding starts out the same as PDF417, so use the same code */
codeerr = 0 ;
/* 456 */
indexliste = 0 ;
indexchaine = 0 ;
mode = quelmode ( chaine [ indexchaine ] ) ;
/* 463 */
do {
liste [ 1 ] [ indexliste ] = mode ;
2020-11-02 07:32:55 +13:00
while ( ( liste [ 1 ] [ indexliste ] = = mode ) & & ( indexchaine < length ) ) {
2016-02-20 23:50:15 +13:00
liste [ 0 ] [ indexliste ] + + ;
indexchaine + + ;
mode = quelmode ( chaine [ indexchaine ] ) ;
}
indexliste + + ;
2020-11-02 07:32:55 +13:00
} while ( indexchaine < length ) ;
2016-02-20 23:50:15 +13:00
/* 474 */
2020-07-18 03:39:01 +12:00
pdfsmooth ( liste , & indexliste ) ;
2016-02-20 23:50:15 +13:00
if ( debug ) {
printf ( " Initial mapping: \n " ) ;
for ( i = 0 ; i < indexliste ; i + + ) {
printf ( " len: %d type: " , liste [ 0 ] [ i ] ) ;
switch ( liste [ 1 ] [ i ] ) {
case TEX : printf ( " TEXT \n " ) ;
break ;
case BYT : printf ( " BYTE \n " ) ;
break ;
case NUM : printf ( " NUMBER \n " ) ;
break ;
default : printf ( " *ERROR* \n " ) ;
break ;
}
}
}
/* 541 - now compress the data */
indexchaine = 0 ;
mclength = 0 ;
2017-10-24 08:37:52 +13:00
2016-02-20 23:50:15 +13:00
if ( symbol - > output_options & READER_INIT ) {
chainemc [ mclength ] = 921 ; /* Reader Initialisation */
mclength + + ;
}
2017-10-24 08:37:52 +13:00
2017-07-24 04:59:51 +12:00
if ( symbol - > eci > 811799 ) {
2020-11-08 08:26:10 +13:00
symbol - > err_origin = 473 ;
strcpy ( symbol - > errtxt , _ ( " Invalid ECI " ) ) ;
2017-07-24 04:59:51 +12:00
return ZINT_ERROR_INVALID_OPTION ;
}
2017-10-24 08:37:52 +13:00
2019-10-07 05:39:54 +13:00
if ( symbol - > eci ! = 0 ) {
2017-06-19 00:00:22 +12:00
/* Encoding ECI assignment number, according to Table 8 */
if ( symbol - > eci < = 899 ) {
chainemc [ mclength ] = 927 ; /* ECI */
mclength + + ;
chainemc [ mclength ] = symbol - > eci ;
mclength + + ;
}
if ( ( symbol - > eci > = 900 ) & & ( symbol - > eci < = 810899 ) ) {
chainemc [ mclength ] = 926 ; /* ECI */
mclength + + ;
chainemc [ mclength ] = ( symbol - > eci / 900 ) - 1 ;
mclength + + ;
chainemc [ mclength ] = symbol - > eci % 900 ;
mclength + + ;
}
if ( symbol - > eci > = 810900 ) {
chainemc [ mclength ] = 925 ; /* ECI */
mclength + + ;
chainemc [ mclength ] = symbol - > eci - 810900 ;
mclength + + ;
}
2016-08-16 23:43:41 +12:00
}
2017-10-24 08:37:52 +13:00
2016-02-20 23:50:15 +13:00
for ( i = 0 ; i < indexliste ; i + + ) {
switch ( liste [ 1 ] [ i ] ) {
case TEX : /* 547 - text mode */
2020-07-18 03:39:01 +12:00
textprocess ( chainemc , & mclength , ( char * ) chaine , indexchaine , liste [ 0 ] [ i ] , 1 /*is_micro*/ ) ;
2016-02-20 23:50:15 +13:00
break ;
case BYT : /* 670 - octet stream mode */
2020-07-18 03:39:01 +12:00
byteprocess ( chainemc , & mclength , chaine , indexchaine , liste [ 0 ] [ i ] , debug ) ;
2016-02-20 23:50:15 +13:00
break ;
case NUM : /* 712 - numeric mode */
2017-09-11 03:03:09 +12:00
numbprocess ( chainemc , & mclength , ( char * ) chaine , indexchaine , liste [ 0 ] [ i ] ) ;
2016-02-20 23:50:15 +13:00
break ;
}
indexchaine = indexchaine + liste [ 0 ] [ i ] ;
}
/* This is where it all changes! */
if ( mclength > 126 ) {
2020-11-08 08:26:10 +13:00
symbol - > err_origin = 467 ;
strcpy ( symbol - > errtxt , _ ( " Input too long " ) ) ;
2016-02-20 23:50:15 +13:00
return ZINT_ERROR_TOO_LONG ;
}
if ( symbol - > option_2 > 4 ) {
2020-11-08 08:26:10 +13:00
symbol - > err_origin = 468 ;
strcpy ( symbol - > errtxt , _ ( " Specified width out of range " ) ) ;
2020-08-22 22:09:57 +12:00
if ( symbol - > warn_level = = WARN_FAIL_ALL ) {
return ZINT_ERROR_INVALID_OPTION ;
} else {
symbol - > option_2 = 0 ;
codeerr = ZINT_WARN_INVALID_OPTION ;
}
2016-02-20 23:50:15 +13:00
}
if ( debug ) {
printf ( " \n Encoded Data Stream: \n " ) ;
for ( i = 0 ; i < mclength ; i + + ) {
2020-07-18 03:39:01 +12:00
printf ( " %3d " , chainemc [ i ] ) ;
2016-02-20 23:50:15 +13:00
}
printf ( " \n " ) ;
}
/* Now figure out which variant of the symbol to use and load values accordingly */
variant = 0 ;
if ( ( symbol - > option_2 = = 1 ) & & ( mclength > 20 ) ) {
/* the user specified 1 column but the data doesn't fit - go to automatic */
2020-11-08 08:26:10 +13:00
symbol - > err_origin = 469 ;
strcpy ( symbol - > errtxt , _ ( " Specified symbol size too small for data " ) ) ;
2020-08-22 22:09:57 +12:00
if ( symbol - > warn_level = = WARN_FAIL_ALL ) {
return ZINT_ERROR_INVALID_OPTION ;
} else {
symbol - > option_2 = 0 ;
codeerr = ZINT_WARN_INVALID_OPTION ;
}
2016-02-20 23:50:15 +13:00
}
if ( ( symbol - > option_2 = = 2 ) & & ( mclength > 37 ) ) {
/* the user specified 2 columns but the data doesn't fit - go to automatic */
2020-11-08 08:26:10 +13:00
symbol - > err_origin = 470 ;
strcpy ( symbol - > errtxt , _ ( " Specified symbol size too small for data " ) ) ;
2020-08-22 22:09:57 +12:00
if ( symbol - > warn_level = = WARN_FAIL_ALL ) {
return ZINT_ERROR_INVALID_OPTION ;
} else {
symbol - > option_2 = 0 ;
codeerr = ZINT_WARN_INVALID_OPTION ;
}
2016-02-20 23:50:15 +13:00
}
if ( ( symbol - > option_2 = = 3 ) & & ( mclength > 82 ) ) {
/* the user specified 3 columns but the data doesn't fit - go to automatic */
2020-11-08 08:26:10 +13:00
symbol - > err_origin = 471 ;
strcpy ( symbol - > errtxt , _ ( " Specified symbol size too small for data " ) ) ;
2020-08-22 22:09:57 +12:00
if ( symbol - > warn_level = = WARN_FAIL_ALL ) {
return ZINT_ERROR_INVALID_OPTION ;
} else {
symbol - > option_2 = 0 ;
codeerr = ZINT_WARN_INVALID_OPTION ;
}
2016-02-20 23:50:15 +13:00
}
if ( symbol - > option_2 = = 1 ) {
/* the user specified 1 column and the data does fit */
variant = 6 ;
if ( mclength < = 16 ) {
variant = 5 ;
}
if ( mclength < = 12 ) {
variant = 4 ;
}
if ( mclength < = 10 ) {
variant = 3 ;
}
if ( mclength < = 7 ) {
variant = 2 ;
}
if ( mclength < = 4 ) {
variant = 1 ;
}
}
if ( symbol - > option_2 = = 2 ) {
/* the user specified 2 columns and the data does fit */
variant = 13 ;
if ( mclength < = 33 ) {
variant = 12 ;
}
if ( mclength < = 29 ) {
variant = 11 ;
}
if ( mclength < = 24 ) {
variant = 10 ;
}
if ( mclength < = 19 ) {
variant = 9 ;
}
if ( mclength < = 13 ) {
variant = 8 ;
}
if ( mclength < = 8 ) {
variant = 7 ;
}
}
if ( symbol - > option_2 = = 3 ) {
/* the user specified 3 columns and the data does fit */
variant = 23 ;
if ( mclength < = 70 ) {
variant = 22 ;
}
if ( mclength < = 58 ) {
variant = 21 ;
}
if ( mclength < = 46 ) {
variant = 20 ;
}
if ( mclength < = 34 ) {
variant = 19 ;
}
if ( mclength < = 24 ) {
variant = 18 ;
}
if ( mclength < = 18 ) {
variant = 17 ;
}
if ( mclength < = 14 ) {
variant = 16 ;
}
if ( mclength < = 10 ) {
variant = 15 ;
}
if ( mclength < = 6 ) {
variant = 14 ;
}
}
if ( symbol - > option_2 = = 4 ) {
/* the user specified 4 columns and the data does fit */
variant = 34 ;
if ( mclength < = 108 ) {
variant = 33 ;
}
if ( mclength < = 90 ) {
variant = 32 ;
}
if ( mclength < = 72 ) {
variant = 31 ;
}
if ( mclength < = 54 ) {
variant = 30 ;
}
if ( mclength < = 39 ) {
variant = 29 ;
}
if ( mclength < = 30 ) {
variant = 28 ;
}
if ( mclength < = 24 ) {
variant = 27 ;
}
if ( mclength < = 18 ) {
variant = 26 ;
}
if ( mclength < = 12 ) {
variant = 25 ;
}
if ( mclength < = 8 ) {
variant = 24 ;
}
}
if ( variant = = 0 ) {
/* Zint can choose automatically from all available variations */
for ( i = 27 ; i > = 0 ; i - - ) {
if ( MicroAutosize [ i ] > = mclength ) {
variant = MicroAutosize [ i + 28 ] ;
}
}
}
/* Now we have the variant we can load the data */
variant - - ;
symbol - > option_2 = MicroVariants [ variant ] ; /* columns */
symbol - > rows = MicroVariants [ variant + 34 ] ; /* rows */
k = MicroVariants [ variant + 68 ] ; /* number of EC CWs */
longueur = ( symbol - > option_2 * symbol - > rows ) - k ; /* number of non-EC CWs */
i = longueur - mclength ; /* amount of padding required */
offset = MicroVariants [ variant + 102 ] ; /* coefficient offset */
if ( debug ) {
printf ( " \n Choose symbol size: \n " ) ;
printf ( " %d columns x %d rows \n " , symbol - > option_2 , symbol - > rows ) ;
printf ( " %d data codewords (including %d pads), %d ecc codewords \n " , longueur , i , k ) ;
printf ( " \n " ) ;
}
/* We add the padding */
while ( i > 0 ) {
chainemc [ mclength ] = 900 ;
mclength + + ;
i - - ;
}
/* Reed-Solomon error correction */
longueur = mclength ;
for ( i = 0 ; i < longueur ; i + + ) {
total = ( chainemc [ i ] + mccorrection [ k - 1 ] ) % 929 ;
for ( j = k - 1 ; j > = 0 ; j - - ) {
if ( j = = 0 ) {
mccorrection [ j ] = ( 929 - ( total * Microcoeffs [ offset + j ] ) % 929 ) % 929 ;
} else {
mccorrection [ j ] = ( mccorrection [ j - 1 ] + 929 - ( total * Microcoeffs [ offset + j ] ) % 929 ) % 929 ;
}
}
}
for ( j = 0 ; j < k ; j + + ) {
if ( mccorrection [ j ] ! = 0 ) {
mccorrection [ j ] = 929 - mccorrection [ j ] ;
}
}
/* we add these codes to the string */
for ( i = k - 1 ; i > = 0 ; i - - ) {
chainemc [ mclength ] = mccorrection [ i ] ;
mclength + + ;
}
if ( debug ) {
printf ( " Encoded Data Stream with ECC: \n " ) ;
for ( i = 0 ; i < mclength ; i + + ) {
2020-07-18 03:39:01 +12:00
printf ( " %3d " , chainemc [ i ] ) ;
2016-02-20 23:50:15 +13:00
}
printf ( " \n " ) ;
}
2020-07-18 03:39:01 +12:00
# ifdef ZINT_TEST
if ( symbol - > debug & ZINT_DEBUG_TEST ) {
debug_test_codeword_dump_int ( symbol , chainemc , mclength ) ;
}
# endif
2016-02-20 23:50:15 +13:00
/* Now get the RAP (Row Address Pattern) start values */
LeftRAPStart = RAPTable [ variant ] ;
CentreRAPStart = RAPTable [ variant + 34 ] ;
RightRAPStart = RAPTable [ variant + 68 ] ;
StartCluster = RAPTable [ variant + 102 ] / 3 ;
/* That's all values loaded, get on with the encoding */
LeftRAP = LeftRAPStart ;
CentreRAP = CentreRAPStart ;
RightRAP = RightRAPStart ;
Cluster = StartCluster ;
/* Cluster can be 0, 1 or 2 for Cluster(0), Cluster(3) and Cluster(6) */
if ( debug ) printf ( " \n Internal row representation: \n " ) ;
for ( i = 0 ; i < symbol - > rows ; i + + ) {
if ( debug ) printf ( " row %d: " , i ) ;
2017-06-10 09:00:22 +12:00
strcpy ( pattern , " " ) ;
2016-02-20 23:50:15 +13:00
offset = 929 * Cluster ;
for ( j = 0 ; j < 5 ; j + + ) {
dummy [ j ] = 0 ;
}
for ( j = 0 ; j < symbol - > option_2 ; j + + ) {
dummy [ j + 1 ] = chainemc [ i * symbol - > option_2 + j ] ;
if ( debug ) printf ( " [%d] " , dummy [ j + 1 ] ) ;
}
/* Copy the data into codebarre */
2017-06-10 09:00:22 +12:00
bin_append ( rap_side [ LeftRAP - 1 ] , 10 , pattern ) ;
bin_append ( pdf_bitpattern [ offset + dummy [ 1 ] ] , 16 , pattern ) ;
strcat ( pattern , " 0 " ) ;
2016-02-20 23:50:15 +13:00
if ( symbol - > option_2 = = 3 ) {
2017-06-10 09:00:22 +12:00
bin_append ( rap_centre [ CentreRAP - 1 ] , 10 , pattern ) ;
2016-02-20 23:50:15 +13:00
}
if ( symbol - > option_2 > = 2 ) {
2017-06-10 09:00:22 +12:00
bin_append ( pdf_bitpattern [ offset + dummy [ 2 ] ] , 16 , pattern ) ;
strcat ( pattern , " 0 " ) ;
2016-02-20 23:50:15 +13:00
}
if ( symbol - > option_2 = = 4 ) {
2017-06-10 09:00:22 +12:00
bin_append ( rap_centre [ CentreRAP - 1 ] , 10 , pattern ) ;
2016-02-20 23:50:15 +13:00
}
if ( symbol - > option_2 > = 3 ) {
2017-06-10 09:00:22 +12:00
bin_append ( pdf_bitpattern [ offset + dummy [ 3 ] ] , 16 , pattern ) ;
strcat ( pattern , " 0 " ) ;
2016-02-20 23:50:15 +13:00
}
if ( symbol - > option_2 = = 4 ) {
2017-06-10 09:00:22 +12:00
bin_append ( pdf_bitpattern [ offset + dummy [ 4 ] ] , 16 , pattern ) ;
strcat ( pattern , " 0 " ) ;
2016-02-20 23:50:15 +13:00
}
2017-06-10 09:00:22 +12:00
bin_append ( rap_side [ RightRAP - 1 ] , 10 , pattern ) ;
strcat ( pattern , " 1 " ) ; /* stop */
if ( debug ) printf ( " %s \n " , pattern ) ;
2016-02-20 23:50:15 +13:00
/* so now pattern[] holds the string of '1's and '0's. - copy this to the symbol */
2020-03-29 04:10:53 +13:00
for ( loop = 0 ; loop < ( int ) strlen ( pattern ) ; loop + + ) {
2016-02-20 23:50:15 +13:00
if ( pattern [ loop ] = = ' 1 ' ) {
set_module ( symbol , i , loop ) ;
}
}
symbol - > row_height [ i ] = 2 ;
2017-06-10 09:00:22 +12:00
symbol - > width = strlen ( pattern ) ;
2016-02-20 23:50:15 +13:00
/* Set up RAPs and Cluster for next row */
LeftRAP + + ;
CentreRAP + + ;
RightRAP + + ;
Cluster + + ;
if ( LeftRAP = = 53 ) {
LeftRAP = 1 ;
}
if ( CentreRAP = = 53 ) {
CentreRAP = 1 ;
}
if ( RightRAP = = 53 ) {
RightRAP = 1 ;
}
if ( Cluster = = 3 ) {
Cluster = 0 ;
}
}
2019-09-04 06:53:01 +12:00
/* Allow user to adjust height of symbol, but enforce minimum row height of 2X */
calcheight = ( int ) ( symbol - > height / i ) ;
if ( calcheight < 2 ) {
calcheight = 2 ;
}
for ( j = 0 ; j < i ; j + + ) {
symbol - > row_height [ j ] = calcheight ;
}
2016-02-20 23:50:15 +13:00
return codeerr ;
2009-10-07 08:03:00 +13:00
}