/* dmatrix.c Handles Data Matrix ECC 200 symbols */ /* libzint - the open source barcode library Copyright (C) 2009 - 2021 Robin Stuart developed from and including some functions from: IEC16022 bar code generation Adrian Kennard, Andrews & Arnold Ltd with help from Cliff Hones on the RS coding (c) 2004 Adrian Kennard, Andrews & Arnold Ltd (c) 2006 Stefan Schmidt 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 : */ #include #include #include #ifdef _MSC_VER #include #endif #include "common.h" #include "reedsol.h" #include "dmatrix.h" /* Annex M placement algorithm low level */ static void ecc200placementbit(int *array, const int NR, const int NC, int r, int c, const int p, const char b) { if (r < 0) { r += NR; c += 4 - ((NR + 4) % 8); } if (c < 0) { c += NC; r += 4 - ((NC + 4) % 8); } // Necessary for 26x32,26x40,26x48,36x120,36x144,72x120,72x144 if (r >= NR) { #ifdef DEBUG fprintf(stderr, "r >= NR:%i,%i at r=%i->", p, b, r); #endif r -= NR; #ifdef DEBUG fprintf(stderr, "%i,c=%i\n", r, c); #endif } #ifdef DEBUG if (0 != array[r * NC + c]) { int a = array[r * NC + c]; fprintf(stderr, "Double:%i,%i->%i,%i at r=%i,c=%i\n", a >> 3, a & 7, p, b, r, c); return; } #endif // Check index limits assert(r < NR); assert(c < NC); // Check double-assignment assert(0 == array[r * NC + c]); array[r * NC + c] = (p << 3) + b; } static void ecc200placementblock(int *array, const int NR, const int NC, const int r, const int c, const int p) { ecc200placementbit(array, NR, NC, r - 2, c - 2, p, 7); ecc200placementbit(array, NR, NC, r - 2, c - 1, p, 6); ecc200placementbit(array, NR, NC, r - 1, c - 2, p, 5); ecc200placementbit(array, NR, NC, r - 1, c - 1, p, 4); ecc200placementbit(array, NR, NC, r - 1, c - 0, p, 3); ecc200placementbit(array, NR, NC, r - 0, c - 2, p, 2); ecc200placementbit(array, NR, NC, r - 0, c - 1, p, 1); ecc200placementbit(array, NR, NC, r - 0, c - 0, p, 0); } static void ecc200placementcornerA(int *array, const int NR, const int NC, const int p) { ecc200placementbit(array, NR, NC, NR - 1, 0, p, 7); ecc200placementbit(array, NR, NC, NR - 1, 1, p, 6); ecc200placementbit(array, NR, NC, NR - 1, 2, p, 5); ecc200placementbit(array, NR, NC, 0, NC - 2, p, 4); ecc200placementbit(array, NR, NC, 0, NC - 1, p, 3); ecc200placementbit(array, NR, NC, 1, NC - 1, p, 2); ecc200placementbit(array, NR, NC, 2, NC - 1, p, 1); ecc200placementbit(array, NR, NC, 3, NC - 1, p, 0); } static void ecc200placementcornerB(int *array, const int NR, const int NC, const int p) { ecc200placementbit(array, NR, NC, NR - 3, 0, p, 7); ecc200placementbit(array, NR, NC, NR - 2, 0, p, 6); ecc200placementbit(array, NR, NC, NR - 1, 0, p, 5); ecc200placementbit(array, NR, NC, 0, NC - 4, p, 4); ecc200placementbit(array, NR, NC, 0, NC - 3, p, 3); ecc200placementbit(array, NR, NC, 0, NC - 2, p, 2); ecc200placementbit(array, NR, NC, 0, NC - 1, p, 1); ecc200placementbit(array, NR, NC, 1, NC - 1, p, 0); } static void ecc200placementcornerC(int *array, const int NR, const int NC, const int p) { ecc200placementbit(array, NR, NC, NR - 3, 0, p, 7); ecc200placementbit(array, NR, NC, NR - 2, 0, p, 6); ecc200placementbit(array, NR, NC, NR - 1, 0, p, 5); ecc200placementbit(array, NR, NC, 0, NC - 2, p, 4); ecc200placementbit(array, NR, NC, 0, NC - 1, p, 3); ecc200placementbit(array, NR, NC, 1, NC - 1, p, 2); ecc200placementbit(array, NR, NC, 2, NC - 1, p, 1); ecc200placementbit(array, NR, NC, 3, NC - 1, p, 0); } static void ecc200placementcornerD(int *array, const int NR, const int NC, const int p) { ecc200placementbit(array, NR, NC, NR - 1, 0, p, 7); ecc200placementbit(array, NR, NC, NR - 1, NC - 1, p, 6); ecc200placementbit(array, NR, NC, 0, NC - 3, p, 5); ecc200placementbit(array, NR, NC, 0, NC - 2, p, 4); ecc200placementbit(array, NR, NC, 0, NC - 1, p, 3); ecc200placementbit(array, NR, NC, 1, NC - 3, p, 2); ecc200placementbit(array, NR, NC, 1, NC - 2, p, 1); ecc200placementbit(array, NR, NC, 1, NC - 1, p, 0); } /* Annex M placement alorithm main function */ static void ecc200placement(int *array, const int NR, const int NC) { int r, c, p; // invalidate for (r = 0; r < NR; r++) for (c = 0; c < NC; c++) array[r * NC + c] = 0; // start p = 1; r = 4; c = 0; do { // check corner if (r == NR && !c) ecc200placementcornerA(array, NR, NC, p++); if (r == NR - 2 && !c && NC % 4) ecc200placementcornerB(array, NR, NC, p++); if (r == NR - 2 && !c && (NC % 8) == 4) ecc200placementcornerC(array, NR, NC, p++); if (r == NR + 4 && c == 2 && !(NC % 8)) ecc200placementcornerD(array, NR, NC, p++); // up/right do { if (r < NR && c >= 0 && !array[r * NC + c]) ecc200placementblock(array, NR, NC, r, c, p++); r -= 2; c += 2; } while (r >= 0 && c < NC); r++; c += 3; // down/left do { if (r >= 0 && c < NC && !array[r * NC + c]) ecc200placementblock(array, NR, NC, r, c, p++); r += 2; c -= 2; } while (r < NR && c >= 0); r += 3; c++; } while (r < NR || c < NC); // unfilled corner if (!array[NR * NC - 1]) array[NR * NC - 1] = array[NR * NC - NC - 2] = 1; } /* calculate and append ecc code, and if necessary interleave */ static void ecc200(unsigned char *binary, const int bytes, const int datablock, const int rsblock, const int skew) { int blocks = (bytes + 2) / datablock, b; int rsblocks = rsblock * blocks; int n; rs_t rs; rs_init_gf(&rs, 0x12d); rs_init_code(&rs, rsblock, 1); for (b = 0; b < blocks; b++) { unsigned char buf[256], ecc[256]; int p = 0; for (n = b; n < bytes; n += blocks) buf[p++] = binary[n]; rs_encode(&rs, p, buf, ecc); p = rsblock - 1; // comes back reversed for (n = b; n < rsblocks; n += blocks) { if (skew) { /* Rotate ecc data to make 144x144 size symbols acceptable */ /* See http://groups.google.com/group/postscriptbarcode/msg/5ae8fda7757477da */ if (b < 8) { binary[bytes + n + 2] = ecc[p--]; } else { binary[bytes + n - 8] = ecc[p--]; } } else { binary[bytes + n] = ecc[p--]; } } } } /* Is basic (non-shifted) C40? */ static int isc40(const unsigned char input) { if ((input >= '0' && input <= '9') || (input >= 'A' && input <= 'Z') || input == ' ') { return 1; } return 0; } /* Is basic (non-shifted) TEXT? */ static int istext(const unsigned char input) { if ((input >= '0' && input <= '9') || (input >= 'a' && input <= 'z') || input == ' ') { return 1; } return 0; } /* Is basic (non-shifted) C40/TEXT? */ static int isc40text(const int current_mode, const unsigned char input) { return current_mode == DM_C40 ? isc40(input) : istext(input); } /* Return true (1) if a character is valid in X12 set */ static int isX12(const unsigned char input) { if (isc40(input)) { return 1; } if (input == 13 || input == '*' || input == '>') { return 1; } return 0; } static int p_r_6_2_1(const unsigned char inputData[], const int position, const int sourcelen) { /* Annex P section (r)(6)(ii)(I) "If one of the three X12 terminator/separator characters first occurs in the yet to be processed data before a non-X12 character..." */ int i; for (i = position; i < sourcelen && isX12(inputData[i]); i++) { if (inputData[i] == 13 || inputData[i] == '*' || inputData[i] == '>') { return 1; } } return 0; } /* 'look ahead test' from Annex P */ static int look_ahead_test(const unsigned char inputData[], const int sourcelen, const int position, const int current_mode, const int gs1) { float ascii_count, c40_count, text_count, x12_count, edf_count, b256_count; int ascii_rnded, c40_rnded, text_rnded, x12_rnded, edf_rnded, b256_rnded; float cnt_1; int sp; /* step (j) */ if (current_mode == DM_ASCII) { ascii_count = 0.0F; c40_count = 1.0F; text_count = 1.0F; x12_count = 1.0F; edf_count = 1.0F; b256_count = 1.25F; } else { ascii_count = 1.0F; c40_count = 2.0F; text_count = 2.0F; x12_count = 2.0F; edf_count = 2.0F; b256_count = 2.25F; } switch (current_mode) { case DM_C40: c40_count = 0.0F; break; case DM_TEXT: text_count = 0.0F; break; case DM_X12: x12_count = 0.0F; break; case DM_EDIFACT: edf_count = 0.0F; break; case DM_BASE256: b256_count = 0.0F; break; } for (sp = position; sp < sourcelen; sp++) { /* ascii ... step (l) */ if ((inputData[sp] >= '0') && (inputData[sp] <= '9')) { ascii_count += 0.5F; // (l)(1) } else { if (inputData[sp] > 127) { ascii_count = ceilf(ascii_count) + 2.0F; // (l)(2) } else { ascii_count = ceilf(ascii_count) + 1.0F; // (l)(3) } } /* c40 ... step (m) */ if (isc40(inputData[sp])) { c40_count += (2.0F / 3.0F); // (m)(1) } else { if (inputData[sp] > 127) { c40_count += (8.0F / 3.0F); // (m)(2) } else { c40_count += (4.0F / 3.0F); // (m)(3) } } /* text ... step (n) */ if (istext(inputData[sp])) { text_count += (2.0F / 3.0F); // (n)(1) } else { if (inputData[sp] > 127) { text_count += (8.0F / 3.0F); // (n)(2) } else { text_count += (4.0F / 3.0F); // (n)(3) } } /* x12 ... step (o) */ if (isX12(inputData[sp])) { x12_count += (2.0F / 3.0F); // (o)(1) } else { if (inputData[sp] > 127) { x12_count += (13.0F / 3.0F); // (o)(2) } else { x12_count += (10.0F / 3.0F); // (o)(3) } } /* edifact ... step (p) */ if ((inputData[sp] >= ' ') && (inputData[sp] <= '^')) { edf_count += (3.0F / 4.0F); // (p)(1) } else { if (inputData[sp] > 127) { edf_count += 17.0F / 4.0f; // (p)(2) } else { edf_count += 13.0F / 4.0f; // (p)(3) } } /* base 256 ... step (q) */ if ((gs1 == 1) && (inputData[sp] == '[')) { /* FNC1 separator */ b256_count += 4.0F; // (q)(1) } else { b256_count += 1.0F; // (q)(2) } if (sp >= position + 4) { /* At least 5 data characters processed ... step (r) */ /* NOTE: different than spec, where it's at least 4. Following previous behaviour here (and BWIPP) */ cnt_1 = ascii_count + 1.0f; if (cnt_1 <= b256_count && cnt_1 <= edf_count && cnt_1 <= text_count && cnt_1 <= x12_count && cnt_1 <= c40_count) { return DM_ASCII; /* step (r)(1) */ } cnt_1 = b256_count + 1.0f; if (cnt_1 <= ascii_count || (cnt_1 < edf_count && cnt_1 < text_count && cnt_1 < x12_count && cnt_1 < c40_count)) { return DM_BASE256; /* step (r)(2) */ } cnt_1 = edf_count + 1.0f; if (cnt_1 < ascii_count && cnt_1 < b256_count && cnt_1 < text_count && cnt_1 < x12_count && cnt_1 < c40_count) { return DM_EDIFACT; /* step (r)(3) */ } cnt_1 = text_count + 1.0f; if (cnt_1 < ascii_count && cnt_1 < b256_count && cnt_1 < edf_count && cnt_1 < x12_count && cnt_1 < c40_count) { return DM_TEXT; /* step (r)(4) */ } cnt_1 = x12_count + 1.0f; if (cnt_1 < ascii_count && cnt_1 < b256_count && cnt_1 < edf_count && cnt_1 < text_count && cnt_1 < c40_count) { return DM_X12; /* step (r)(5) */ } cnt_1 = c40_count + 1.0f; if (cnt_1 < ascii_count && cnt_1 < b256_count && cnt_1 < edf_count && cnt_1 < text_count) { if (c40_count < x12_count) { return DM_C40; /* step (r)(6)(i) */ } if (c40_count == x12_count) { if (p_r_6_2_1(inputData, sp, sourcelen) == 1) { return DM_X12; /* step (r)(6)(ii)(I) */ } return DM_C40; /* step (r)(6)(ii)(II) */ } } } } /* At the end of data ... step (k) */ /* step (k)(1) */ ascii_rnded = (int) ceilf(stripf(ascii_count)); b256_rnded = (int) ceilf(stripf(b256_count)); edf_rnded = (int) ceilf(stripf(edf_count)); text_rnded = (int) ceilf(stripf(text_count)); x12_rnded = (int) ceilf(stripf(x12_count)); c40_rnded = (int) ceilf(stripf(c40_count)); if (ascii_rnded <= b256_rnded && ascii_rnded <= edf_rnded && ascii_rnded <= text_rnded && ascii_rnded <= x12_rnded && ascii_rnded <= c40_rnded) { return DM_ASCII; /* step (k)(2) */ } if (b256_rnded < ascii_rnded && b256_rnded < edf_rnded && b256_rnded < text_rnded && b256_rnded < x12_rnded && b256_rnded < c40_rnded) { return DM_BASE256; /* step (k)(3) */ } if (edf_rnded < ascii_rnded && edf_rnded < b256_rnded && edf_rnded < text_rnded && edf_rnded < x12_rnded && edf_rnded < c40_rnded) { return DM_EDIFACT; /* step (k)(4) */ } if (text_rnded < ascii_rnded && text_rnded < b256_rnded && text_rnded < edf_rnded && text_rnded < x12_rnded && text_rnded < c40_rnded) { return DM_TEXT; /* step (k)(5) */ } if (x12_rnded < ascii_rnded && x12_rnded < b256_rnded && x12_rnded < edf_rnded && x12_rnded < text_rnded && x12_rnded < c40_rnded) { return DM_X12; /* step (k)(6) */ } /* Note the algorithm is particularly sub-optimal here, returning C40 even if X12/EDIFACT (much) better, due to the < comparisons of rounded X12/EDIFACT values to each other above - comparisons would need to be <= or unrounded (cf. very similar Code One algorithm). Not changed to maintain compatibility with spec and BWIPP */ return DM_C40; /* step (k)(7) */ } /* Copy C40/TEXT/X12 triplets from buffer to target. Returns elements left in buffer (< 3) */ static int ctx_process_buffer_transfer(int process_buffer[8], int process_p, unsigned char target[], int *p_tp, int debug) { int i, process_e; int tp = *p_tp; process_e = (process_p / 3) * 3; for (i = 0; i < process_e; i += 3) { int iv = (1600 * process_buffer[i]) + (40 * process_buffer[i + 1]) + (process_buffer[i + 2]) + 1; target[tp++] = (unsigned char) (iv >> 8); target[tp++] = (unsigned char) (iv & 0xFF); if (debug) { printf("[%d %d %d (%d %d)] ", process_buffer[i], process_buffer[i + 1], process_buffer[i + 2], target[tp - 2], target[tp - 1]); } } process_p -= process_e; if (process_p) { memmove(process_buffer, process_buffer + process_e, sizeof(int) * process_p); } *p_tp = tp; return process_p; } /* Copy EDIFACT quadruplets from buffer to target. Returns elements left in buffer (< 4) */ static int edi_process_buffer_transfer(int process_buffer[8], int process_p, unsigned char target[], int *p_tp, int debug) { int i, process_e; int tp = *p_tp; process_e = (process_p / 4) * 4; for (i = 0; i < process_e; i += 4) { target[tp++] = (unsigned char) (process_buffer[i] << 2 | (process_buffer[i + 1] & 0x30) >> 4); target[tp++] = (unsigned char) ((process_buffer[i + 1] & 0x0f) << 4 | (process_buffer[i + 2] & 0x3c) >> 2); target[tp++] = (unsigned char) ((process_buffer[i + 2] & 0x03) << 6 | process_buffer[i + 3]); if (debug) { printf("[%d %d %d %d (%d %d %d)] ", process_buffer[i], process_buffer[i + 1], process_buffer[i + 2], process_buffer[i + 3], target[tp - 3], target[tp - 2], target[tp - 1]); } } process_p -= process_e; if (process_p) { memmove(process_buffer, process_buffer + process_e, sizeof(int) * process_p); } *p_tp = tp; return process_p; } /* Get symbol size, as specified or else smallest containing `minimum` codewords */ static int get_symbolsize(struct zint_symbol *symbol, const int minimum) { int i; if ((symbol->option_2 >= 1) && (symbol->option_2 <= DMSIZESCOUNT)) { return intsymbol[symbol->option_2 - 1]; } for (i = DMSIZESCOUNT - 2; i >= 0; i--) { if (minimum > matrixbytes[i]) { if (symbol->option_3 == DM_DMRE) { return i + 1; } if (symbol->option_3 == DM_SQUARE) { /* Skip rectangular symbols in square only mode */ while (i + 1 < DMSIZESCOUNT && matrixH[i + 1] != matrixW[i + 1]) { i++; } return i + 1 < DMSIZESCOUNT ? i + 1 : 0; } /* Skip DMRE symbols in no dmre mode */ while (i + 1 < DMSIZESCOUNT && isDMRE[i + 1]) { i++; } return i + 1 < DMSIZESCOUNT ? i + 1 : 0; } } return 0; } /* Number of codewords remaining in a particular version (may be negative) */ static int codewords_remaining(struct zint_symbol *symbol, const int tp, const int process_p) { int symbolsize = get_symbolsize(symbol, tp + process_p); /* Allow for the remaining data characters */ return matrixbytes[symbolsize] - tp; } /* Number of C40/TEXT elements needed to encode `input` */ static int c40text_cnt(const int current_mode, const int gs1, unsigned char input) { int cnt; if (gs1 && input == '[') { return 2; } cnt = 1; if (input > 127) { cnt += 2; input = input - 128; } if ((current_mode == DM_C40 && c40_shift[input]) || (current_mode == DM_TEXT && text_shift[input])) { cnt += 1; } return cnt; } /* Update Base 256 field length */ static int update_b256_field_length(unsigned char target[], int tp, int b256_start) { int b256_count = tp - (b256_start + 1); if (b256_count <= 249) { target[b256_start] = b256_count; } else { /* Insert extra codeword */ memmove(target + b256_start + 2, target + b256_start + 1, b256_count); target[b256_start] = (unsigned char) (249 + (b256_count / 250)); target[b256_start + 1] = (unsigned char) (b256_count % 250); tp++; } return tp; } /* Encodes data using ASCII, C40, Text, X12, EDIFACT or Base 256 modes as appropriate Supports encoding FNC1 in supporting systems */ static int dm200encode(struct zint_symbol *symbol, const unsigned char source[], unsigned char target[], int *p_length, int *p_binlen) { int sp; int tp, i, gs1; int current_mode, next_mode; int inputlen = *p_length; int process_buffer[8]; /* holds remaining data to finalised */ int process_p = 0; /* number of characters left to finalise */ int b256_start = 0; int symbols_left; int debug = symbol->debug & ZINT_DEBUG_PRINT; sp = 0; tp = 0; /* step (a) */ current_mode = DM_ASCII; next_mode = DM_ASCII; /* gs1 flag values: 0: no gs1, 1: gs1 with FNC1 serparator, 2: GS separator */ if ((symbol->input_mode & 0x07) == GS1_MODE) { if (symbol->output_options & GS1_GS_SEPARATOR) { gs1 = 2; } else { gs1 = 1; } } else { gs1 = 0; } if (gs1) { target[tp] = 232; tp++; if (debug) printf("FN1 "); } /* FNC1 */ if (symbol->output_options & READER_INIT) { if (gs1) { strcpy(symbol->errtxt, "521: Cannot encode in GS1 mode and Reader Initialisation at the same time"); return ZINT_ERROR_INVALID_OPTION; } else { target[tp] = 234; tp++; /* Reader Programming */ if (debug) printf("RP "); } } if (symbol->eci > 0) { /* Encode ECI numbers according to Table 6 */ target[tp] = 241; /* ECI Character */ tp++; if (symbol->eci <= 126) { target[tp] = (unsigned char) (symbol->eci + 1); tp++; } else if (symbol->eci <= 16382) { target[tp] = (unsigned char) ((symbol->eci - 127) / 254 + 128); tp++; target[tp] = (unsigned char) ((symbol->eci - 127) % 254 + 1); tp++; } else { target[tp] = (unsigned char) ((symbol->eci - 16383) / 64516 + 192); tp++; target[tp] = (unsigned char) (((symbol->eci - 16383) / 254) % 254 + 1); tp++; target[tp] = (unsigned char) ((symbol->eci - 16383) % 254 + 1); tp++; } if (debug) printf("ECI %d ", symbol->eci + 1); } /* Check for Macro05/Macro06 */ /* "[)>[RS]05[GS]...[RS][EOT]" -> CW 236 */ /* "[)>[RS]06[GS]...[RS][EOT]" -> CW 237 */ if (tp == 0 && sp == 0 && inputlen >= 9 && source[0] == '[' && source[1] == ')' && source[2] == '>' && source[3] == '\x1e' && source[4] == '0' && (source[5] == '5' || source[5] == '6') && source[6] == '\x1d' && source[inputlen - 2] == '\x1e' && source[inputlen - 1] == '\x04') { /* Output macro Codeword */ if (source[5] == '5') { target[tp] = 236; if (debug) printf("Macro05 "); } else { target[tp] = 237; if (debug) printf("Macro06 "); } tp++; /* Remove macro characters from input string */ sp = 7; inputlen -= 2; *p_length -= 2; } while (sp < inputlen) { current_mode = next_mode; /* step (b) - ASCII encodation */ if (current_mode == DM_ASCII) { next_mode = DM_ASCII; if (istwodigits(source, inputlen, sp)) { target[tp] = (unsigned char) ((10 * ctoi(source[sp])) + ctoi(source[sp + 1]) + 130); if (debug) printf("N%02d ", target[tp] - 130); tp++; sp += 2; } else { next_mode = look_ahead_test(source, inputlen, sp, current_mode, gs1); if (next_mode != DM_ASCII) { switch (next_mode) { case DM_C40: target[tp] = 230; tp++; if (debug) printf("C40 "); break; case DM_TEXT: target[tp] = 239; tp++; if (debug) printf("TEX "); break; case DM_X12: target[tp] = 238; tp++; if (debug) printf("X12 "); break; case DM_EDIFACT: target[tp] = 240; tp++; if (debug) printf("EDI "); break; case DM_BASE256: target[tp] = 231; tp++; b256_start = tp; target[tp++] = 0; /* Byte count holder (may be expanded to 2 codewords) */ if (debug) printf("BAS "); break; } } else { if (source[sp] > 127) { target[tp] = 235; /* FNC4 */ tp++; target[tp] = (source[sp] - 128) + 1; tp++; if (debug) printf("FN4 A%02X ", target[tp - 1] - 1); } else { if (gs1 && (source[sp] == '[')) { if (gs1 == 2) { target[tp] = 29 + 1; /* GS */ if (debug) printf("GS "); } else { target[tp] = 232; /* FNC1 */ if (debug) printf("FN1 "); } } else { target[tp] = source[sp] + 1; if (debug) printf("A%02X ", target[tp] - 1); } tp++; } sp++; } } /* step (c)/(d) C40/TEXT encodation */ } else if (current_mode == DM_C40 || current_mode == DM_TEXT) { next_mode = current_mode; if (process_p == 0) { next_mode = look_ahead_test(source, inputlen, sp, current_mode, gs1); } if (next_mode != current_mode) { target[tp] = 254; /* Unlatch */ tp++; next_mode = DM_ASCII; if (debug) printf("ASC "); } else { int shift_set, value; const char *ct_shift, *ct_value; if (current_mode == DM_C40) { ct_shift = c40_shift; ct_value = c40_value; } else { ct_shift = text_shift; ct_value = text_value; } if (source[sp] > 127) { process_buffer[process_p++] = 1; process_buffer[process_p++] = 30; /* Upper Shift */ shift_set = ct_shift[source[sp] - 128]; value = ct_value[source[sp] - 128]; } else { if (gs1 && (source[sp] == '[')) { if (gs1 == 2) { shift_set = ct_shift[29]; value = ct_value[29]; /* GS */ } else { shift_set = 2; value = 27; /* FNC1 */ } } else { shift_set = ct_shift[source[sp]]; value = ct_value[source[sp]]; } } if (shift_set != 0) { process_buffer[process_p++] = shift_set - 1; } process_buffer[process_p++] = value; if (process_p >= 3) { process_p = ctx_process_buffer_transfer(process_buffer, process_p, target, &tp, debug); } sp++; } /* step (e) X12 encodation */ } else if (current_mode == DM_X12) { next_mode = DM_X12; if (process_p == 0) { next_mode = look_ahead_test(source, inputlen, sp, current_mode, gs1); } if (next_mode != DM_X12) { target[tp] = 254; /* Unlatch */ tp++; next_mode = DM_ASCII; if (debug) printf("ASC "); } else { static const char x12_nonalphanum_chars[] = "\015*> "; int value = 0; if ((source[sp] >= '0') && (source[sp] <= '9')) { value = (source[sp] - '0') + 4; } else if ((source[sp] >= 'A') && (source[sp] <= 'Z')) { value = (source[sp] - 'A') + 14; } else { value = posn(x12_nonalphanum_chars, source[sp]); } process_buffer[process_p++] = value; if (process_p >= 3) { process_p = ctx_process_buffer_transfer(process_buffer, process_p, target, &tp, debug); } sp++; } /* step (f) EDIFACT encodation */ } else if (current_mode == DM_EDIFACT) { next_mode = DM_EDIFACT; if (process_p == 3) { /* Note different then spec Step (f)(1), which suggests checking when 0, but this seems to work better in many cases. */ next_mode = look_ahead_test(source, inputlen, sp, current_mode, gs1); } if (next_mode != DM_EDIFACT) { process_buffer[process_p++] = 31; next_mode = DM_ASCII; } else { int value = source[sp]; if (value >= 64) { // '@' value -= 64; } process_buffer[process_p++] = value; sp++; } if (process_p >= 4) { process_p = edi_process_buffer_transfer(process_buffer, process_p, target, &tp, debug); } if (debug && next_mode == DM_ASCII) printf("ASC "); /* step (g) Base 256 encodation */ } else if (current_mode == DM_BASE256) { next_mode = look_ahead_test(source, inputlen, sp, current_mode, gs1); if (next_mode == DM_BASE256) { target[tp] = source[sp]; tp++; sp++; if (debug) printf("B%02X ", target[tp - 1]); } else { tp = update_b256_field_length(target, tp, b256_start); /* B.2.1 255-state randomising algorithm */ for (i = b256_start; i < tp; i++) { int prn = ((149 * (i + 1)) % 255) + 1; target[i] = (unsigned char) ((target[i] + prn) & 0xFF); } next_mode = DM_ASCII; if (debug) printf("ASC "); } } if (tp > 1558) { strcpy(symbol->errtxt, "520: Data too long to fit in symbol"); return ZINT_ERROR_TOO_LONG; } } /* while */ symbols_left = codewords_remaining(symbol, tp, process_p); if (debug) printf("\nsymbols_left %d, process_p %d ", symbols_left, process_p); if (current_mode == DM_C40 || current_mode == DM_TEXT) { /* NOTE: changed to follow spec exactly here, only using Shift 1 padded triplets when 2 symbol chars remain. This matches the behaviour of BWIPP but not tec-it, nor figures 4.15.1-1 and 4.15-1-2 in GS1 General Specifications 21.0.1. */ if (debug) printf("%s ", current_mode == DM_C40 ? "C40" : "TEX"); if (process_p == 0) { if (symbols_left > 0) { target[tp++] = 254; // Unlatch if (debug) printf("ASC "); } } else { if (process_p == 2 && symbols_left == 2) { /* 5.2.5.2 (b) */ process_buffer[process_p++] = 0; // Shift 1 (void) ctx_process_buffer_transfer(process_buffer, process_p, target, &tp, debug); } else if (process_p == 1 && symbols_left <= 2 && isc40text(current_mode, source[inputlen - 1])) { /* 5.2.5.2 (c)/(d) */ if (symbols_left > 1) { /* 5.2.5.2 (c) */ target[tp++] = 254; // Unlatch and encode remaining data in ascii. if (debug) printf("ASC "); } target[tp++] = source[inputlen - 1] + 1; if (debug) printf("A%02X ", target[tp - 1] - 1); } else { int cnt, total_cnt = 0; /* Backtrack to last complete triplet (same technique as BWIPP) */ while (sp > 0 && process_p % 3) { sp--; cnt = c40text_cnt(current_mode, gs1, source[sp]); total_cnt += cnt; process_p -= cnt; } tp -= (total_cnt / 3) * 2; target[tp++] = 254; // Unlatch if (debug) printf("ASC "); for (; sp < inputlen; sp++) { if (istwodigits(source, inputlen, sp)) { target[tp++] = (unsigned char) ((10 * ctoi(source[sp])) + ctoi(source[sp + 1]) + 130); if (debug) printf("N%02d ", target[tp - 1] - 130); sp++; } else if (source[sp] > 127) { target[tp++] = 235; /* FNC4 */ target[tp++] = (source[sp] - 128) + 1; if (debug) printf("FN4 A%02X ", target[tp - 1] - 1); } else if (gs1 && source[sp] == '[') { if (gs1 == 2) { target[tp] = 29 + 1; /* GS */ if (debug) printf("GS "); } else { target[tp] = 232; /* FNC1 */ if (debug) printf("FN1 "); } } else { target[tp++] = source[sp] + 1; if (debug) printf("A%02X ", target[tp - 1] - 1); } } } } } else if (current_mode == DM_X12) { if (debug) printf("X12 "); if ((symbols_left == 1) && (process_p == 1)) { // Unlatch not required! target[tp++] = source[inputlen - 1] + 1; if (debug) printf("A%02X ", target[tp - 1] - 1); } else { if (symbols_left > 0) { target[tp++] = (254); // Unlatch. if (debug) printf("ASC "); } if (process_p == 1) { target[tp++] = source[inputlen - 1] + 1; if (debug) printf("A%02X ", target[tp - 1] - 1); } else if (process_p == 2) { target[tp++] = source[inputlen - 2] + 1; target[tp++] = source[inputlen - 1] + 1; if (debug) printf("A%02X A%02X ", target[tp - 2] - 1, target[tp - 1] - 1); } } } else if (current_mode == DM_EDIFACT) { if (debug) printf("EDI "); if (symbols_left <= 2 && process_p <= symbols_left) { // Unlatch not required! if (process_p == 1) { target[tp++] = source[inputlen - 1] + 1; if (debug) printf("A%02X ", target[tp - 1] - 1); } else if (process_p == 2) { target[tp++] = source[inputlen - 2] + 1; target[tp++] = source[inputlen - 1] + 1; if (debug) printf("A%02X A%02X ", target[tp - 2] - 1, target[tp - 1] - 1); } } else { // Append edifact unlatch value (31) and empty buffer if (process_p <= 3) { process_buffer[process_p++] = 31; if (process_p < 4) { memset(process_buffer + process_p, 0, sizeof(int) * (4 - process_p)); } } (void) edi_process_buffer_transfer(process_buffer, 4, target, &tp, debug); } } else if (current_mode == DM_BASE256) { if (symbols_left > 0) { tp = update_b256_field_length(target, tp, b256_start); } /* B.2.1 255-state randomising algorithm */ for (i = b256_start; i < tp; i++) { int prn = ((149 * (i + 1)) % 255) + 1; target[i] = (unsigned char) ((target[i] + prn) & 0xFF); } } if (debug) { printf("\nData (%d): ", tp); for (i = 0; i < tp; i++) printf("%d ", target[i]); printf("\n"); } *p_binlen = tp; return 0; } /* add pad bits */ static void add_tail(unsigned char target[], int tp, const int tail_length) { int i, prn, temp; for (i = tail_length; i > 0; i--) { if (i == tail_length) { target[tp] = 129; tp++; /* Pad */ } else { /* B.1.1 253-state randomising algorithm */ prn = ((149 * (tp + 1)) % 253) + 1; temp = 129 + prn; if (temp <= 254) { target[tp] = (unsigned char) (temp); tp++; } else { target[tp] = (unsigned char) (temp - 254); tp++; } } } } static int data_matrix_200(struct zint_symbol *symbol, const unsigned char source[], int inputlen) { int i, skew = 0; unsigned char binary[2200]; int binlen; int symbolsize; int taillength, error_number = 0; int H, W, FH, FW, datablock, bytes, rsblock; int debug = symbol->debug & ZINT_DEBUG_PRINT; /* inputlen may be decremented by 2 if macro character is used */ error_number = dm200encode(symbol, source, binary, &inputlen, &binlen); if (error_number != 0) { return error_number; } symbolsize = get_symbolsize(symbol, binlen); if (binlen > matrixbytes[symbolsize]) { if ((symbol->option_2 >= 1) && (symbol->option_2 <= DMSIZESCOUNT)) { // The symbol size was given by --ver (option_2) strcpy(symbol->errtxt, "522: Input too long for selected symbol size"); } else { strcpy(symbol->errtxt, "523: Data too long to fit in symbol"); } return ZINT_ERROR_TOO_LONG; } H = matrixH[symbolsize]; W = matrixW[symbolsize]; FH = matrixFH[symbolsize]; FW = matrixFW[symbolsize]; bytes = matrixbytes[symbolsize]; datablock = matrixdatablock[symbolsize]; rsblock = matrixrsblock[symbolsize]; taillength = bytes - binlen; if (taillength != 0) { add_tail(binary, binlen, taillength); } if (debug) { printf("Pads (%d): ", taillength); for (i = binlen; i < binlen + taillength; i++) printf("%d ", binary[i]); printf("\n"); } // ecc code if (symbolsize == INTSYMBOL144) { skew = 1; } ecc200(binary, bytes, datablock, rsblock, skew); if (debug) { printf("ECC (%d): ", rsblock * (bytes / datablock)); for (i = bytes; i < bytes + rsblock * (bytes / datablock); i++) printf("%d ", binary[i]); printf("\n"); } #ifdef ZINT_TEST if (symbol->debug & ZINT_DEBUG_TEST) { debug_test_codeword_dump(symbol, binary, skew ? 1558 + 620 : bytes + rsblock * (bytes / datablock)); } #endif { // placement int x, y, NC, NR, *places; unsigned char *grid; NC = W - 2 * (W / FW); NR = H - 2 * (H / FH); places = (int *) malloc(sizeof(int) * NC * NR); ecc200placement(places, NR, NC); grid = (unsigned char *) malloc((size_t) W * H); memset(grid, 0, W * H); for (y = 0; y < H; y += FH) { for (x = 0; x < W; x++) grid[y * W + x] = 1; for (x = 0; x < W; x += 2) grid[(y + FH - 1) * W + x] = 1; } for (x = 0; x < W; x += FW) { for (y = 0; y < H; y++) grid[y * W + x] = 1; for (y = 0; y < H; y += 2) grid[y * W + x + FW - 1] = 1; } #ifdef DEBUG // Print position matrix as in standard for (y = NR - 1; y >= 0; y--) { for (x = 0; x < NC; x++) { int v; if (x != 0) fprintf(stderr, "|"); v = places[(NR - y - 1) * NC + x]; fprintf(stderr, "%3d.%2d", (v >> 3), 8 - (v & 7)); } fprintf(stderr, "\n"); } #endif for (y = 0; y < NR; y++) { for (x = 0; x < NC; x++) { int v = places[(NR - y - 1) * NC + x]; if (v == 1 || (v > 7 && (binary[(v >> 3) - 1] & (1 << (v & 7))))) grid[(1 + y + 2 * (y / (FH - 2))) * W + 1 + x + 2 * (x / (FW - 2))] = 1; } } for (y = H - 1; y >= 0; y--) { for (x = 0; x < W; x++) { if (grid[W * y + x]) { set_module(symbol, (H - y) - 1, x); } } symbol->row_height[(H - y) - 1] = 1; } free(grid); free(places); } symbol->height = H; symbol->rows = H; symbol->width = W; return error_number; } INTERNAL int dmatrix(struct zint_symbol *symbol, unsigned char source[], int length) { int error_number; if (symbol->option_1 <= 1) { /* ECC 200 */ error_number = data_matrix_200(symbol, source, length); } else { /* ECC 000 - 140 */ strcpy(symbol->errtxt, "524: Older Data Matrix standards are no longer supported"); error_number = ZINT_ERROR_INVALID_OPTION; } return error_number; }