/* pdf417.c - Handles PDF417 stacked symbology */ /* Zint - A barcode generating program using libpng Copyright (C) 2008-2020 Robin Stuart Portions Copyright (C) 2004 Grandzebu Bug Fixes thanks to KL Chin Redistribution and use in source and binary forms, with or without modification, are permitted provided that the following conditions are met: 1. Redistributions of source code must retain the above copyright notice, this list of conditions and the following disclaimer. 2. Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the following disclaimer in the documentation and/or other materials provided with the distribution. 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 without specific prior written permission. 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 OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ /* vim: set ts=4 sw=4 et : */ /* 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 #include #ifndef _MSC_VER #include #else #include #include "ms_stdint.h" #endif #include "pdf417.h" #include "common.h" /* 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 */ static const char asciix[95] = { 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 }; static const char asciiy[95] = { 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 }; /* Automatic sizing table */ static const char MicroAutosize[56] = { 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 }; #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) */ /* 866 */ static int quelmode(char codeascii) { int mode = BYT; if ((codeascii == '\t') || (codeascii == '\n') || (codeascii == '\r') || ((codeascii >= ' ') && (codeascii <= '~'))) { mode = TEX; } if ((codeascii >= '0') && (codeascii <= '9')) { mode = NUM; } /* 876 */ return mode; } /* 844 */ static void regroupe(int liste[2][PDF417_MAX_LEN], int *indexliste) { /* bring together same type blocks */ if (*(indexliste) > 1) { int i = 1; while (i < *(indexliste)) { if (liste[1][i - 1] == liste[1][i]) { int j; /* bring together */ liste[0][i - 1] = liste[0][i - 1] + liste[0][i]; j = i + 1; /* 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 */ } /* 478 */ static void pdfsmooth(int liste[2][PDF417_MAX_LEN], int *indexliste) { 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; } } } } } regroupe(liste, indexliste); /* 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 */ regroupe(liste, indexliste); } /* 547 */ 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; wnet = 0; /* listet will contain the table numbers and the value of each characters */ for (indexlistet = 0; indexlistet < length; indexlistet++) { char codeascii = chaine[start + indexlistet]; switch (codeascii) { case '\t': listet[0][indexlistet] = 12; listet[1][indexlistet] = 12; break; case '\n': listet[0][indexlistet] = 8; listet[1][indexlistet] = 15; break; case 13: listet[0][indexlistet] = 12; listet[1][indexlistet] = 11; break; default: listet[0][indexlistet] = asciix[codeascii - 32]; listet[1][indexlistet] = asciiy[codeascii - 32]; break; } } /* 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; } if (listet[0][j] & 8) { /* T_PUN */ chainet[wnet] = 29; chainet[wnet + 1] = listet[1][j]; wnet += 2; } if (!(((listet[0][j] & 1) && (curtable == 2)) || (listet[0][j] & 8))) { /* 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 */ switch (newtable) { case 3: case 5: case 7: case 9: case 11: case 13: case 15: newtable = 1; break; case 6: case 10: case 14: newtable = 2; break; case 12: newtable = 4; break; } /* 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 */ /* 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; } for (j = 0; j < wnet; j += 2) { int cw_number; cw_number = (30 * chainet[j]) + chainet[j + 1]; chainemc[(*mclength)++] = cw_number; } } /* 671 */ INTERNAL void byteprocess(int *chainemc, int *mclength, unsigned char chaine[], int start, int length, int debug) { if (debug) printf("\nEntering 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 { int len; /* select the switch for multiple of 6 bytes */ if (length % 6 == 0) { chainemc[(*mclength)++] = 924; if (debug) printf("924 "); } else { /* 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. */ chainemc[(*mclength)++] = 901; if (debug) printf("901 "); } len = 0; while (len < length) { uint64_t total; unsigned int chunkLen = length - len; if (6 <= chunkLen) { /* Take groups of 6 */ chunkLen = 6; len += chunkLen; total = 0; while (chunkLen--) { uint64_t mantisa = chaine[start++]; total |= mantisa << (chunkLen * 8); } chunkLen = 5; while (chunkLen--) { chainemc[*mclength + chunkLen] = (int) (total % 900); total /= 900; } *mclength += 5; } else { /* If there remains a group of less than 6 bytes */ len += chunkLen; while (chunkLen--) { chainemc[(*mclength)++] = chaine[start++]; } } } } } /* 712 */ static void numbprocess(int *chainemc, int *mclength, char chaine[], int start, int length) { int j, loop, dummy[50] = {0}, diviseur, nombre; char chainemod[50], chainemult[100], temp; strcpy(chainemod, ""); chainemc[(*mclength)++] = 902; j = 0; while (j < length) { int longueur; int dumlength = 0; strcpy(chainemod, ""); longueur = length - j; if (longueur > 44) { longueur = 44; } strcat(chainemod, "1"); for (loop = 1; loop <= longueur; loop++) { chainemod[loop] = chaine[start + loop + j - 1]; } chainemod[longueur + 1] = '\0'; do { diviseur = 900; /* 877 - gosub Modulo */ strcpy(chainemult, ""); nombre = 0; while (strlen(chainemod) != 0) { nombre *= 10; nombre += ctoi(chainemod[0]); for (loop = 0; loop < (int)strlen(chainemod); loop++) { chainemod[loop] = chainemod[loop + 1]; } if (nombre < diviseur) { if (strlen(chainemult) != 0) { strcat(chainemult, "0"); } } else { temp = (nombre / diviseur) + '0'; chainemult[strlen(chainemult) + 1] = '\0'; chainemult[strlen(chainemult)] = temp; } nombre = nombre % diviseur; } diviseur = nombre; /* return to 723 */ for (loop = dumlength; loop > 0; loop--) { dummy[loop] = dummy[loop - 1]; } dummy[0] = diviseur; dumlength++; strcpy(chainemod, chainemult); } while (strlen(chainemult) != 0); for (loop = 0; loop < dumlength; loop++) { chainemc[(*mclength)++] = dummy[loop]; } j += longueur; } } /* 366 */ static int pdf417(struct zint_symbol *symbol, unsigned char chaine[], const size_t length) { int i, k, j, indexchaine, indexliste, mode, longueur, loop, mccorrection[520], offset; int total, chainemc[PDF417_MAX_LEN], mclength, c1, c2, c3, dummy[35], calcheight; int liste[2][PDF417_MAX_LEN] = {{0}}; char pattern[580]; int error_number = 0; int debug = symbol->debug & ZINT_DEBUG_PRINT; if (length > PDF417_MAX_LEN) { strcpy(symbol->errtxt, "463: Input string too long"); return ZINT_ERROR_TOO_LONG; } /* 456 */ indexliste = 0; indexchaine = 0; mode = quelmode(chaine[indexchaine]); /* 463 */ do { liste[1][indexliste] = mode; while ((liste[1][indexliste] == mode) && (indexchaine < (int)length)) { liste[0][indexliste]++; indexchaine++; mode = quelmode(chaine[indexchaine]); } indexliste++; } while (indexchaine < (int)length); /* 474 */ pdfsmooth(liste, &indexliste); 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; if (symbol->output_options & READER_INIT) { chainemc[mclength] = 921; /* Reader Initialisation */ mclength++; } if (symbol->eci != 0) { /* 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++; } } if (symbol->eci > 811799) { strcpy(symbol->errtxt, "472: Invalid ECI"); return ZINT_ERROR_INVALID_OPTION; } for (i = 0; i < indexliste; i++) { switch (liste[1][i]) { case TEX: /* 547 - text mode */ textprocess(chainemc, &mclength, (char*) chaine, indexchaine, liste[0][i], 0 /*is_micro*/); break; case BYT: /* 670 - octet stream mode */ byteprocess(chainemc, &mclength, chaine, indexchaine, liste[0][i], debug); break; case NUM: /* 712 - numeric mode */ numbprocess(chainemc, &mclength, (char*) chaine, indexchaine, liste[0][i]); break; } indexchaine = indexchaine + liste[0][i]; } if (debug) { printf("\nCompressed 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) { symbol->option_1 = 6; if (mclength <= 863) { symbol->option_1 = 5; } if (mclength <= 320) { symbol->option_1 = 4; } if (mclength <= 160) { symbol->option_1 = 3; } if (mclength <= 40) { symbol->option_1 = 2; } } k = 1; for (loop = 1; loop <= (symbol->option_1 + 1); loop++) { k *= 2; } longueur = mclength; if (symbol->option_2 > 30) { symbol->option_2 = 30; } if (symbol->option_2 < 1) { symbol->option_2 =(int)(0.5 + sqrt((longueur + k) / 3.0)); } if (((longueur + k) / symbol->option_2) > 90) { /* stop the symbol from becoming too high */ symbol->option_2 = symbol->option_2 + 1; } if (longueur + k > 928) { /* Enforce maximum codeword limit */ strcpy(symbol->errtxt, "464: Input string too long"); return ZINT_ERROR_TOO_LONG; } if (((longueur + k) / symbol->option_2) > 90) { strcpy(symbol->errtxt, "465: Data too long for specified number of columns"); return ZINT_ERROR_TOO_LONG; } /* 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 (loop = 0; loop < 520; loop++) { mccorrection[loop] = 0; } 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; } mccorrection[0] = (929 - (total * coefrs[offset + j]) % 929) % 929; } /* we add these codes to the string */ for (i = k - 1; i >= 0; i--) { chainemc[mclength++] = mccorrection[i] ? 929 - mccorrection[i] : 0; } if (debug) { printf("Complete CW string:\n"); for (i = 0; i < mclength; i++) { printf("%d ", chainemc[i]); } printf("\n"); } #ifdef ZINT_TEST if (symbol->debug & ZINT_DEBUG_TEST) { debug_test_codeword_dump_int(symbol, chainemc, mclength); } #endif symbol->rows = mclength / symbol->option_2; /* 818 - The CW string is finished */ c1 = (symbol->rows - 1) / 3; c2 = symbol->option_1 * 3 + (symbol->rows - 1) % 3; c3 = symbol->option_2 - 1; /* we now encode each row */ for (i = 0; i < symbol->rows; i++) { 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; offset = 0; /* cluster(0) */ break; case 1: dummy[0] = k + c2; dummy[symbol->option_2 + 1] = k + c1; offset = 929; /* cluster(3) */ break; case 2: dummy[0] = k + c3; dummy[symbol->option_2 + 1] = k + c2; offset = 1858; /* cluster(6) */ break; } strcpy(pattern, ""); bin_append(0x1FEA8, 17, pattern); /* Row start */ for (j = 0; j <= symbol->option_2; j++) { bin_append(pdf_bitpattern[offset + dummy[j]], 16, pattern); strcat(pattern, "0"); } if (symbol->symbology != BARCODE_PDF417COMP) { bin_append(pdf_bitpattern[offset + dummy[j]], 16, pattern); strcat(pattern, "0"); bin_append(0x3FA29, 18, pattern); /* Row Stop */ } else { strcat(pattern, "1"); /* Compact PDF417 Stop pattern */ } for (loop = 0; loop < (int)strlen(pattern); loop++) { if (pattern[loop] == '1') { set_module(symbol, i, loop); } } } /* 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; } symbol->width = (int) strlen(pattern); /* 843 */ return error_number; } /* 345 */ INTERNAL int pdf417enc(struct zint_symbol *symbol, unsigned char source[], const size_t length) { int codeerr, error_number; error_number = 0; if ((symbol->option_1 < -1) || (symbol->option_1 > 8)) { strcpy(symbol->errtxt, "460: Security value out of range"); if (symbol->warn_level == WARN_FAIL_ALL) { return ZINT_ERROR_INVALID_OPTION; } else { symbol->option_1 = -1; error_number = ZINT_WARN_INVALID_OPTION; } } if ((symbol->option_2 < 0) || (symbol->option_2 > 30)) { strcpy(symbol->errtxt, "461: Number of columns out of range"); if (symbol->warn_level == WARN_FAIL_ALL) { return ZINT_ERROR_INVALID_OPTION; } else { symbol->option_2 = 0; error_number = ZINT_WARN_INVALID_OPTION; } } /* 349 */ codeerr = pdf417(symbol, source, length); /* 352 */ if (codeerr != 0) { error_number = codeerr; } /* 364 */ return error_number; } /* like PDF417 only much smaller! */ INTERNAL int micro_pdf417(struct zint_symbol *symbol, unsigned char chaine[], const size_t length) { int i, k, j, indexchaine, indexliste, mode, longueur, mccorrection[50], offset; int total, chainemc[PDF417_MAX_LEN], mclength, dummy[5], codeerr; int liste[2][PDF417_MAX_LEN] = {{0}}; char pattern[580]; int variant, LeftRAPStart, CentreRAPStart, RightRAPStart, StartCluster; int LeftRAP, CentreRAP, RightRAP, Cluster, loop, calcheight; int debug = symbol->debug & ZINT_DEBUG_PRINT; if (length > MICRO_PDF417_MAX_LEN) { strcpy(symbol->errtxt, "474: Input data too long"); return ZINT_ERROR_TOO_LONG; } /* 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; while ((liste[1][indexliste] == mode) && (indexchaine < (int)length)) { liste[0][indexliste]++; indexchaine++; mode = quelmode(chaine[indexchaine]); } indexliste++; } while (indexchaine < (int)length); /* 474 */ pdfsmooth(liste, &indexliste); 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; if (symbol->output_options & READER_INIT) { chainemc[mclength] = 921; /* Reader Initialisation */ mclength++; } if (symbol->eci > 811799) { strcpy(symbol->errtxt, "473: Invalid ECI"); return ZINT_ERROR_INVALID_OPTION; } if (symbol->eci != 0) { /* 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++; } } for (i = 0; i < indexliste; i++) { switch (liste[1][i]) { case TEX: /* 547 - text mode */ textprocess(chainemc, &mclength, (char*) chaine, indexchaine, liste[0][i], 1 /*is_micro*/); break; case BYT: /* 670 - octet stream mode */ byteprocess(chainemc, &mclength, chaine, indexchaine, liste[0][i], debug); break; case NUM: /* 712 - numeric mode */ numbprocess(chainemc, &mclength, (char*) chaine, indexchaine, liste[0][i]); break; } indexchaine = indexchaine + liste[0][i]; } /* This is where it all changes! */ if (mclength > 126) { strcpy(symbol->errtxt, "467: Input data too long"); return ZINT_ERROR_TOO_LONG; } if (symbol->option_2 > 4) { strcpy(symbol->errtxt, "468: Specified width out of range"); if (symbol->warn_level == WARN_FAIL_ALL) { return ZINT_ERROR_INVALID_OPTION; } else { symbol->option_2 = 0; codeerr = ZINT_WARN_INVALID_OPTION; } } if (debug) { printf("\nEncoded Data Stream:\n"); for (i = 0; i < mclength; i++) { printf("%3d ", chainemc[i]); } 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 */ strcpy(symbol->errtxt, "469: Specified symbol size too small for data"); if (symbol->warn_level == WARN_FAIL_ALL) { return ZINT_ERROR_INVALID_OPTION; } else { symbol->option_2 = 0; codeerr = ZINT_WARN_INVALID_OPTION; } } if ((symbol->option_2 == 2) && (mclength > 37)) { /* the user specified 2 columns but the data doesn't fit - go to automatic */ strcpy(symbol->errtxt, "470: Specified symbol size too small for data"); if (symbol->warn_level == WARN_FAIL_ALL) { return ZINT_ERROR_INVALID_OPTION; } else { symbol->option_2 = 0; codeerr = ZINT_WARN_INVALID_OPTION; } } if ((symbol->option_2 == 3) && (mclength > 82)) { /* the user specified 3 columns but the data doesn't fit - go to automatic */ strcpy(symbol->errtxt, "471: Specified symbol size too small for data"); if (symbol->warn_level == WARN_FAIL_ALL) { return ZINT_ERROR_INVALID_OPTION; } else { symbol->option_2 = 0; codeerr = ZINT_WARN_INVALID_OPTION; } } 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("\nChoose 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 (loop = 0; loop < 50; loop++) { mccorrection[loop] = 0; } 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++) { printf("%3d ", chainemc[i]); } printf("\n"); } #ifdef ZINT_TEST if (symbol->debug & ZINT_DEBUG_TEST) { debug_test_codeword_dump_int(symbol, chainemc, mclength); } #endif /* 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("\nInternal row representation:\n"); for (i = 0; i < symbol->rows; i++) { if (debug) printf("row %d: ", i); strcpy(pattern, ""); 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 */ bin_append(rap_side[LeftRAP - 1], 10, pattern); bin_append(pdf_bitpattern[offset + dummy[1]], 16, pattern); strcat(pattern, "0"); if (symbol->option_2 == 3) { bin_append(rap_centre[CentreRAP - 1], 10, pattern); } if (symbol->option_2 >= 2) { bin_append(pdf_bitpattern[offset + dummy[2]], 16, pattern); strcat(pattern, "0"); } if (symbol->option_2 == 4) { bin_append(rap_centre[CentreRAP - 1], 10, pattern); } if (symbol->option_2 >= 3) { bin_append(pdf_bitpattern[offset + dummy[3]], 16, pattern); strcat(pattern, "0"); } if (symbol->option_2 == 4) { bin_append(pdf_bitpattern[offset + dummy[4]], 16, pattern); strcat(pattern, "0"); } bin_append(rap_side[RightRAP - 1], 10, pattern); strcat(pattern, "1"); /* stop */ if (debug) printf("%s\n", pattern); /* so now pattern[] holds the string of '1's and '0's. - copy this to the symbol */ for (loop = 0; loop < (int)strlen(pattern); loop++) { if (pattern[loop] == '1') { set_module(symbol, i, loop); } } symbol->row_height[i] = 2; symbol->width = strlen(pattern); /* 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; } } /* 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; } return codeerr; }