zint/backend/code1.c
gitlost 5e2044ff2e CODE128: reduce extended latch cut-off from 5 to 4 for better
encodation in certain cases (and no pessimizations found so far),
  props lyngklip (BWIPP);
  fix extended char latching when exactly 3 extended chars at end;
  count code set C (not digits) in loop deciding when to
  shift/latch to extended for better estimate
AZTEC: return warning if ECC < 5% (due to bit-stuffing when version
  given); return error if > 22 layers (Zint 26) for Reader
  Initialisation symbol requested for better error message
AZTEC/HANXIN/QRCODE: consolidate different ECC data size tables
  into one indexed by ECC
DBAR_EXP: check for reduced length <= 77 up front for better error
  message
HANXIN: use `malloc()` rather than `z_alloca()` for large binary
  array
QRCODE: `ecc_level` now 0-based (not 1-based)
MICROQR: consolidate different version end routines into one
  `microqr_end()` and use new `microqr_data` table to simplify code
MICROPDF417: use table for max codewords per column
library: centralize all error messages using new `errtxt()`,
  `errtxtf()`, `errtxt_adj()` funcs that protect `symbol->errtxt`
  from overflow, & try to make error messages more consistent
  thru-out, adding more feedback info to many, & use positional
  args "%n$" in prep for l10n (maybe);
  `is_sane/is_sane_lookup()` -> `not_sane/not_sane_lookup()`,
  returning 1-based position (zero on failure) instead of bool;
  `long` ints -> plain `int` (except those dealing with `ftell()`,
  `fread()` etc) as depend on int being 32-bits already
GUI: in "grpDATF.ui" use "PlainText" rather than "RichText" for
  tracker ratio examples as height of text messing up sometimes
manual: clarify Codablock-F length maximum & add examples
docs: README: pandoc 3.5, Ubuntu 24.04
CMake: use "-Wpedantic" for Clang only as GNU complains about
  `errtxtf()` positional args "%n$"
2024-10-27 21:33:33 +00:00

1592 lines
61 KiB
C

/* code1.c - USS Code One */
/*
libzint - the open source barcode library
Copyright (C) 2009-2024 Robin Stuart <rstuart114@gmail.com>
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.
*/
/* SPDX-License-Identifier: BSD-3-Clause */
#include <assert.h>
#include <math.h>
#include <stdio.h>
#include "common.h"
#include "code1.h"
#include "reedsol.h"
#include "large.h"
#define C1_MAX_CWS 1480 /* Max data codewords for Version H */
#define C1_MAX_CWS_S "1480" /* String version of above */
#define C1_MAX_ECCS 560 /* Max ECC codewords for Version H */
#define C1_ASCII 1
#define C1_C40 2
#define C1_DECIMAL 3
#define C1_TEXT 4
#define C1_EDI 5
#define C1_BYTE 6
/* Add solid bar */
static void c1_horiz(struct zint_symbol *symbol, const int row_no, const int full) {
int i;
if (full) {
for (i = 0; i < symbol->width; i++) {
set_module(symbol, row_no, i);
}
} else {
for (i = 1; i < symbol->width - 1; i++) {
set_module(symbol, row_no, i);
}
}
}
/* Central recognition pattern for Versions A-H */
static void c1_central_finder(struct zint_symbol *symbol, const int start_row, const int row_count,
const int full_rows) {
int i;
for (i = 0; i < row_count; i++) {
if (i < full_rows) {
c1_horiz(symbol, start_row + (i * 2), 1);
} else {
c1_horiz(symbol, start_row + (i * 2), 0);
if (i != row_count - 1) {
set_module(symbol, start_row + (i * 2) + 1, 1);
set_module(symbol, start_row + (i * 2) + 1, symbol->width - 2);
}
}
}
}
/* Add solid column */
static void c1_vert(struct zint_symbol *symbol, const int column, const int height, const int top) {
int i;
if (top) {
for (i = 0; i < height; i++) {
set_module(symbol, i, column);
}
} else {
for (i = 0; i < height; i++) {
set_module(symbol, symbol->rows - i - 1, column);
}
}
}
/* Add bump to the right of the vertical recognition pattern column (Versions A-H) */
static void c1_spigot(struct zint_symbol *symbol, const int row_no) {
int i;
for (i = symbol->width - 1; i > 0; i--) {
if (module_is_set(symbol, row_no, i - 1)) {
set_module(symbol, row_no, i);
}
}
}
/* Is basic (non-shifted) C40? */
static int c1_isc40(const unsigned char input) {
if (z_isdigit(input) || z_isupper(input) || input == ' ') {
return 1;
}
return 0;
}
/* Is basic (non-shifted) TEXT? */
static int c1_istext(const unsigned char input) {
if (z_isdigit(input) || z_islower(input) || input == ' ') {
return 1;
}
return 0;
}
/* Is basic (non-shifted) C40/TEXT? */
static int c1_isc40text(const int current_mode, const unsigned char input) {
return current_mode == C1_C40 ? c1_isc40(input) : c1_istext(input);
}
/* EDI characters are uppercase alphanumerics plus space plus EDI terminator (CR) plus 2 EDI separator chars */
static int c1_isedi(const unsigned char input) {
if (c1_isc40(input)) {
return 1;
}
if (input == 13 || input == '*' || input == '>') {
return 1;
}
return 0;
}
/* Whether Step Q4bi applies, i.e. if one of the 3 EDI terminator/separator chars appears before a non-EDI char */
static int c1_is_step_Q4bi_applicable(const unsigned char source[], const int length, const int position) {
int i;
for (i = position; i < length && c1_isedi(source[i]); i++) {
if (source[i] == 13 || source[i] == '*' || source[i] == '>') {
return 1;
}
}
return 0;
}
/* Character counts are multiplied by this, so as to be whole integer divisible by 2 and 3 */
#define C1_MULT 6
#define C1_MULT_1_DIV_2 3
#define C1_MULT_2_DIV_3 4
#define C1_MULT_1 6
#define C1_MULT_4_DIV_3 8
#define C1_MULT_2 12
#define C1_MULT_8_DIV_3 16
#define C1_MULT_3 18
#define C1_MULT_10_DIV_3 20
#define C1_MULT_13_DIV_3 26
#define C1_MULT_MINUS_1 5
#define C1_MULT_CEIL(n) ((((n) + C1_MULT_MINUS_1) / C1_MULT) * C1_MULT)
/* AIM USS Code One Annex D Steps J-R */
static int c1_look_ahead_test(const unsigned char source[], const int length, const int position,
const int current_mode, const int gs1) {
int ascii_count, c40_count, text_count, edi_count, byte_count;
int ascii_rnded, c40_rnded, text_rnded, edi_rnded, byte_rnded;
int cnt_1;
int sp;
/* Step J1 */
if (current_mode == C1_ASCII) {
ascii_count = 0;
c40_count = C1_MULT_1;
text_count = C1_MULT_1;
edi_count = C1_MULT_1;
byte_count = C1_MULT_2;
} else {
ascii_count = C1_MULT_1;
c40_count = C1_MULT_2;
text_count = C1_MULT_2;
edi_count = C1_MULT_2;
byte_count = C1_MULT_3;
}
switch (current_mode) {
case C1_C40: c40_count = 0; /* Step J2 */
break;
case C1_TEXT: text_count = 0; /* Step J3 */
break;
case C1_EDI: edi_count = 0; /* Missing in spec */
break;
case C1_BYTE: byte_count = 0; /* Step J4 */
break;
}
for (sp = position; sp < length; sp++) {
const unsigned char c = source[sp];
const int is_extended = c & 0x80;
/* Step L */
if (z_isdigit(c)) {
ascii_count += C1_MULT_1_DIV_2; /* Step L1 */
} else {
if (is_extended) {
ascii_count = ceilf(ascii_count) + C1_MULT_2; /* Step L2 */
} else {
ascii_count = ceilf(ascii_count) + C1_MULT_1; /* Step L3 */
}
}
/* Step M */
if (c1_isc40(c)) {
c40_count += C1_MULT_2_DIV_3; /* Step M1 */
} else if (is_extended) {
c40_count += C1_MULT_8_DIV_3; /* Step M2 */
} else {
c40_count += C1_MULT_4_DIV_3; /* Step M3 */
}
/* Step N */
if (c1_istext(c)) {
text_count += C1_MULT_2_DIV_3; /* Step N1 */
} else if (is_extended) {
text_count += C1_MULT_8_DIV_3; /* Step N2 */
} else {
text_count += C1_MULT_4_DIV_3; /* Step N3 */
}
/* Step O */
if (c1_isedi(c)) {
edi_count += C1_MULT_2_DIV_3; /* Step O1 */
} else if (is_extended) {
edi_count += C1_MULT_13_DIV_3; /* Step O2 */
} else {
edi_count += C1_MULT_10_DIV_3; /* Step O3 */
}
/* Step P */
if (gs1 && c == '\x1D') {
byte_count += C1_MULT_3; /* Step P1 */
} else {
byte_count += C1_MULT_1; /* Step P2 */
}
/* If at least 4 characters processed */
/* NOTE: different than spec, where it's at least 3, but that ends up suppressing C40/TEXT/EDI.
BWIPP also uses 4 (cf very similar Data Matrix ISO/IEC 16022:2006 Annex P algorithm) */
if (sp >= position + 3) {
/* Step Q */
ascii_rnded = C1_MULT_CEIL(ascii_count);
c40_rnded = C1_MULT_CEIL(c40_count);
text_rnded = C1_MULT_CEIL(text_count);
edi_rnded = C1_MULT_CEIL(edi_count);
byte_rnded = C1_MULT_CEIL(byte_count);
cnt_1 = byte_count + C1_MULT_1;
if (cnt_1 <= ascii_rnded && cnt_1 <= c40_rnded && cnt_1 <= text_rnded && cnt_1 <= edi_rnded) {
return C1_BYTE; /* Step Q1 */
}
cnt_1 = ascii_count + C1_MULT_1;
if (cnt_1 <= c40_rnded && cnt_1 <= text_rnded && cnt_1 <= edi_rnded && cnt_1 <= byte_rnded) {
return C1_ASCII; /* Step Q2 */
}
cnt_1 = text_rnded + C1_MULT_1;
if (cnt_1 <= ascii_rnded && cnt_1 <= c40_rnded && cnt_1 <= edi_rnded && cnt_1 <= byte_rnded) {
return C1_TEXT; /* Step Q3 */
}
cnt_1 = c40_rnded + C1_MULT_1;
if (cnt_1 <= ascii_rnded && cnt_1 <= text_rnded) {
/* Step Q4 */
if (c40_rnded < edi_rnded) {
return C1_C40; /* Step Q4a */
}
if (c40_rnded == edi_rnded) {
/* Step Q4b */
if (c1_is_step_Q4bi_applicable(source, length, sp + 1)) {
return C1_EDI; /* Step Q4bi */
}
return C1_C40; /* Step Q4bii */
}
}
cnt_1 = edi_rnded + C1_MULT_1;
if (cnt_1 <= ascii_rnded && cnt_1 <= c40_rnded && cnt_1 <= text_rnded && cnt_1 <= byte_rnded) {
return C1_EDI; /* Step Q5 */
}
}
}
/* Step K */
ascii_rnded = C1_MULT_CEIL(ascii_count);
c40_rnded = C1_MULT_CEIL(c40_count);
text_rnded = C1_MULT_CEIL(text_count);
edi_rnded = C1_MULT_CEIL(edi_count);
byte_rnded = C1_MULT_CEIL(byte_count);
if (byte_count <= ascii_rnded && byte_count <= c40_rnded && byte_count <= text_rnded && byte_count <= edi_rnded) {
return C1_BYTE; /* Step K1 */
}
if (ascii_count <= c40_rnded && ascii_count <= text_rnded && ascii_count <= edi_rnded
&& ascii_count <= byte_rnded) {
return C1_ASCII; /* Step K2 */
}
if (c40_rnded <= text_rnded && c40_rnded <= edi_rnded) {
return C1_C40; /* Step K3 */
}
if (text_rnded <= edi_rnded) {
return C1_TEXT; /* Step K4 */
}
return C1_EDI; /* Step K5 */
}
/* Whether can fit last character or characters in a single ASCII codeword */
static int c1_is_last_single_ascii(const unsigned char source[], const int length, const int sp) {
if (length - sp == 1 && source[sp] <= 127) {
return 1;
}
if (length - sp == 2 && is_twodigits(source, length, sp)) {
return 1;
}
return 0;
}
/* Initialize number of digits array (taken from BWIPP) */
static void c1_set_num_digits(const unsigned char source[], const int length, int num_digits[]) {
int i;
for (i = length - 1; i >= 0; i--) {
if (z_isdigit(source[i])) {
num_digits[i] = num_digits[i + 1] + 1;
}
}
}
/* Copy C40/TEXT/EDI triplets from buffer to `target`. Returns elements left in buffer (< 3) */
static int c1_cte_buffer_transfer(int cte_buffer[6], int cte_p, unsigned target[], int *p_tp) {
int cte_i, cte_e;
int tp = *p_tp;
cte_e = (cte_p / 3) * 3;
for (cte_i = 0; cte_i < cte_e; cte_i += 3) {
int iv = (1600 * cte_buffer[cte_i]) + (40 * cte_buffer[cte_i + 1]) + (cte_buffer[cte_i + 2]) + 1;
target[tp++] = iv >> 8;
target[tp++] = iv & 0xFF;
}
cte_p -= cte_e;
if (cte_p) {
memmove(cte_buffer, cte_buffer + cte_e, sizeof(int) * cte_p);
}
*p_tp = tp;
return cte_p;
}
/* Copy DECIMAL bytes to `target`. Returns bits left in buffer (< 8) */
static int c1_decimal_binary_transfer(char decimal_binary[24], int db_p, unsigned int target[], int *p_tp) {
int b_i, b_e, p;
int tp = *p_tp;
/* Transfer full bytes to target */
b_e = db_p & 0xF8;
for (b_i = 0; b_i < b_e; b_i += 8) {
int value = 0;
for (p = 0; p < 8; p++) {
value <<= 1;
value += decimal_binary[b_i + p] == '1';
}
target[tp++] = value;
}
db_p &= 0x07; /* Bits remaining */
if (db_p) {
memmove(decimal_binary, decimal_binary + b_e, db_p);
}
*p_tp = tp;
return db_p;
}
/* Unlatch to ASCII from DECIMAL mode using 6 ones flag. DECIMAL binary buffer will be empty */
static int c1_decimal_unlatch(char decimal_binary[24], int db_p, unsigned int target[], int *p_tp,
const int decimal_count, const unsigned char source[], int *p_sp) {
int sp = *p_sp;
int bits_left;
db_p = bin_append_posn(63, 6, decimal_binary, db_p); /* Unlatch */
if (db_p >= 8) {
db_p = c1_decimal_binary_transfer(decimal_binary, db_p, target, p_tp);
}
bits_left = (8 - db_p) & 0x07;
if (decimal_count >= 1 && bits_left >= 4) {
db_p = bin_append_posn(ctoi(source[sp]) + 1, 4, decimal_binary, db_p);
sp++;
if (bits_left == 6) {
db_p = bin_append_posn(1, 2, decimal_binary, db_p);
}
(void) c1_decimal_binary_transfer(decimal_binary, db_p, target, p_tp);
} else if (bits_left) {
if (bits_left >= 4) {
db_p = bin_append_posn(15, 4, decimal_binary, db_p);
}
if (bits_left == 2 || bits_left == 6) {
db_p = bin_append_posn(1, 2, decimal_binary, db_p);
}
(void) c1_decimal_binary_transfer(decimal_binary, db_p, target, p_tp);
}
*p_sp = sp;
return 0;
}
/* Number of codewords remaining in a particular version (may be negative) */
static int c1_codewords_remaining(struct zint_symbol *symbol, const int tp) {
int i;
if (symbol->option_2 == 10) { /* Version T */
if (tp > 24) {
return 38 - tp;
}
if (tp > 10) {
return 24 - tp;
}
return 10 - tp;
}
/* Versions A to H */
for (i = 6; i >= 0; i--) {
if (tp > c1_data_length[i]) {
return c1_data_length[i + 1] - tp;
}
}
return c1_data_length[0] - tp;
}
/* Number of C40/TEXT elements needed to encode `input` */
static int c1_c40text_cnt(const int current_mode, const int gs1, unsigned char input) {
int cnt;
if (gs1 && input == '\x1D') {
return 2;
}
cnt = 1;
if (input & 0x80) {
cnt += 2;
input -= 128;
}
if ((current_mode == C1_C40 && c1_c40_shift[input]) || (current_mode == C1_TEXT && c1_text_shift[input])) {
cnt += 1;
}
return cnt;
}
/* Copy `source` to `eci_buf` with "\NNNNNN" ECI indicator at start and backslashes escaped */
static void c1_eci_escape(const int eci, unsigned char source[], const int length, unsigned char eci_buf[],
const int eci_length) {
int i, j;
j = sprintf((char *) eci_buf, "\\%06d", eci);
for (i = 0; i < length && j < eci_length; i++) {
if (source[i] == '\\') {
eci_buf[j++] = '\\';
}
eci_buf[j++] = source[i];
}
eci_buf[j] = '\0';
}
/* Convert to codewords */
static int c1_encode(struct zint_symbol *symbol, unsigned char source[], int length, const int eci,
const int seg_count, const int gs1, unsigned int target[], int *p_tp, int *p_last_mode) {
int current_mode, next_mode, last_mode;
int sp = 0;
int tp = *p_tp;
int i;
int cte_buffer[6], cte_p = 0; /* C1_C40/TEXT/EDI buffer and index */
char decimal_binary[24]; /* C1_DECIMAL buffer */
int db_p = 0;
int byte_start = 0;
const int debug_print = symbol->debug & ZINT_DEBUG_PRINT;
const int eci_length = length + 7 + chr_cnt(source, length, '\\');
unsigned char *eci_buf = (unsigned char *) z_alloca(eci_length + 1);
int *num_digits = (int *) z_alloca(sizeof(int) * (eci_length + 1));
memset(num_digits, 0, sizeof(int) * (eci_length + 1));
/* Step A */
current_mode = C1_ASCII;
next_mode = C1_ASCII;
if (gs1 && tp == 0) {
c1_set_num_digits(source, length, num_digits);
if (length >= 15 && num_digits[0] >= 15) {
target[tp++] = 236; /* FNC1 and change to Decimal */
next_mode = C1_DECIMAL;
if (debug_print) fputs("FNC1Dec ", stdout);
} else if (length >= 7 && num_digits[0] == length) {
target[tp++] = 236; /* FNC1 and change to Decimal */
next_mode = C1_DECIMAL;
if (debug_print) fputs("FNC1Dec ", stdout);
} else {
target[tp++] = 232; /* FNC1 */
if (debug_print) fputs("FNC1 ", stdout);
}
/* Note ignoring Structured Append and ECI if GS1 mode (up to caller to warn/error) */
} else {
if (symbol->structapp.count && tp == 0) {
if (symbol->structapp.count < 16) { /* Group mode */
if ((eci || seg_count > 1) && symbol->structapp.index == 1) {
/* Initial pad indicator for 1st symbol only */
target[tp++] = 129; /* Pad */
target[tp++] = 233; /* FNC2 */
target[tp++] = (symbol->structapp.index - 1) * 15 + (symbol->structapp.count - 1);
target[tp++] = '\\' + 1; /* Escape char */
if (debug_print) fputs("SAGrp ", stdout);
} else {
target[tp++] = (symbol->structapp.index - 1) * 15 + (symbol->structapp.count - 1);
target[tp++] = 233; /* FNC2 */
if (debug_print) fputs("FNC2 ", stdout);
}
} else { /* Extended Group mode */
if ((eci || seg_count > 1) && symbol->structapp.index == 1) {
/* Initial pad indicator for 1st symbol only */
target[tp++] = 129; /* Pad */
target[tp++] = '\\' + 1; /* Escape char */
target[tp++] = 233; /* FNC2 */
target[tp++] = symbol->structapp.index;
target[tp++] = symbol->structapp.count;
if (debug_print) fputs("SAExGrp ", stdout);
} else {
target[tp++] = symbol->structapp.index;
target[tp++] = symbol->structapp.count;
target[tp++] = 233; /* FNC2 */
if (debug_print) fputs("FNC2 ", stdout);
}
}
if (eci) {
c1_eci_escape(eci, source, length, eci_buf, eci_length);
source = eci_buf;
length = eci_length;
}
} else if (eci || seg_count > 1) {
if (tp == 0) {
target[tp++] = 129; /* Pad */
target[tp++] = '\\' + 1; /* Escape char */
if (debug_print) fputs("PADEsc ", stdout);
}
if (eci) {
c1_eci_escape(eci, source, length, eci_buf, eci_length);
source = eci_buf;
length = eci_length;
}
}
c1_set_num_digits(source, length, num_digits);
}
do {
last_mode = current_mode;
if (current_mode != next_mode) {
/* Change mode */
switch (next_mode) {
case C1_C40: target[tp++] = 230;
if (debug_print) fputs("->C40 ", stdout);
break;
case C1_TEXT: target[tp++] = 239;
if (debug_print) fputs("->Text ", stdout);
break;
case C1_EDI: target[tp++] = 238;
if (debug_print) fputs("->EDI ", stdout);
break;
case C1_BYTE: target[tp++] = 231;
if (debug_print) fputs("->Byte ", stdout);
byte_start = tp;
target[tp++] = 0; /* Byte count holder (may be expanded to 2 codewords) */
break;
}
current_mode = next_mode;
}
if (current_mode == C1_ASCII) {
/* Step B - ASCII encodation */
next_mode = C1_ASCII;
if ((length - sp) >= 21 && num_digits[sp] >= 21) {
/* Step B1 */
next_mode = C1_DECIMAL;
db_p = bin_append_posn(15, 4, decimal_binary, db_p);
} else if ((length - sp) >= 13 && num_digits[sp] == (length - sp)) {
/* Step B2 */
next_mode = C1_DECIMAL;
db_p = bin_append_posn(15, 4, decimal_binary, db_p);
}
if (next_mode == C1_ASCII) {
if (is_twodigits(source, length, sp)) {
/* Step B3 */
target[tp++] = (10 * ctoi(source[sp])) + ctoi(source[sp + 1]) + 130;
if (debug_print) printf("ASCDD(%.2s) ", source + sp);
sp += 2;
} else {
if (gs1 && source[sp] == '\x1D') {
if (length - (sp + 1) >= 15 && num_digits[sp + 1] >= 15) {
/* Step B4 */
target[tp++] = 236; /* FNC1 and change to Decimal */
if (debug_print) fputs("FNC1 ", stdout);
sp++;
next_mode = C1_DECIMAL;
} else if (length - (sp + 1) >= 7 && num_digits[sp + 1] == length - (sp + 1)) {
/* Step B5 */
target[tp++] = 236; /* FNC1 and change to Decimal */
if (debug_print) fputs("FNC1 ", stdout);
sp++;
next_mode = C1_DECIMAL;
}
}
if (next_mode == C1_ASCII) {
/* Step B6 */
next_mode = c1_look_ahead_test(source, length, sp, current_mode, gs1);
if (next_mode == last_mode) { /* Avoid looping on latch (ticket #300 (#8) Andre Maute) */
next_mode = C1_ASCII;
}
if (next_mode == C1_ASCII) {
if (debug_print) printf("ASC(%d) ", source[sp]);
if (source[sp] & 0x80) {
/* Step B7 */
target[tp++] = 235; /* FNC4 (Upper Shift) */
target[tp++] = (source[sp] - 128) + 1;
if (debug_print) printf("UpSh(%d) ", source[sp]);
} else if (gs1 && source[sp] == '\x1D') {
/* Step B8 */
target[tp++] = 232; /* FNC1 */
if (debug_print) fputs("FNC1 ", stdout);
} else {
/* Step B8 */
target[tp++] = source[sp] + 1;
if (debug_print) printf("ASC(%d) ", source[sp]);
}
sp++;
}
}
}
}
} else if (current_mode == C1_C40 || current_mode == C1_TEXT) {
/* Step C/D - C40/TEXT encodation */
next_mode = current_mode;
if (cte_p == 0) {
/* Step C/D1 */
if ((length - sp) >= 12 && num_digits[sp] >= 12) {
/* Step C/D1a */
next_mode = C1_ASCII;
} else if ((length - sp) >= 8 && num_digits[sp] == (length - sp)) {
/* Step C/D1b */
next_mode = C1_ASCII;
} else {
next_mode = c1_look_ahead_test(source, length, sp, current_mode, gs1);
}
}
if (next_mode != current_mode) {
/* Step C/D1c */
target[tp++] = 255; /* Unlatch */
if (debug_print) fputs("Unlatch ", stdout);
} else {
/* Step C/D2 */
const char *ct_shift, *ct_value;
if (current_mode == C1_C40) {
ct_shift = c1_c40_shift;
ct_value = c1_c40_value;
} else {
ct_shift = c1_text_shift;
ct_value = c1_text_value;
}
if (debug_print) fputs(current_mode == C1_C40 ? "C40 " : "TEXT ", stdout);
if (source[sp] & 0x80) {
cte_buffer[cte_p++] = 1; /* Shift 2 */
cte_buffer[cte_p++] = 30; /* FNC4 (Upper Shift) */
if (ct_shift[source[sp] - 128]) {
cte_buffer[cte_p++] = ct_shift[source[sp] - 128] - 1;
}
cte_buffer[cte_p++] = ct_value[source[sp] - 128];
} else if (gs1 && source[sp] == '\x1D') {
cte_buffer[cte_p++] = 1; /* Shift 2 */
cte_buffer[cte_p++] = 27; /* FNC1 */
} else {
if (ct_shift[source[sp]]) {
cte_buffer[cte_p++] = ct_shift[source[sp]] - 1;
}
cte_buffer[cte_p++] = ct_value[source[sp]];
}
if (cte_p >= 3) {
cte_p = c1_cte_buffer_transfer(cte_buffer, cte_p, target, &tp);
}
sp++;
}
} else if (current_mode == C1_EDI) {
/* Step E - EDI Encodation */
next_mode = C1_EDI;
if (cte_p == 0) {
/* Step E1 */
if ((length - sp) >= 12 && num_digits[sp] >= 12) {
/* Step E1a */
next_mode = C1_ASCII;
} else if ((length - sp) >= 8 && num_digits[sp] == (length - sp)) {
/* Step E1b */
next_mode = C1_ASCII;
} else if ((length - sp) < 3 || !c1_isedi(source[sp]) || !c1_isedi(source[sp + 1])
|| !c1_isedi(source[sp + 2])) {
/* Step E1c */
/* This ensures ASCII switch if don't have EDI triplet, so cte_p will be zero on loop exit */
next_mode = C1_ASCII;
}
}
if (next_mode != C1_EDI) {
if (c1_is_last_single_ascii(source, length, sp) && c1_codewords_remaining(symbol, tp) == 1) {
/* No unlatch needed if data fits as ASCII in last data codeword */
} else {
target[tp++] = 255; /* Unlatch */
if (debug_print) fputs("Unlatch ", stdout);
}
} else {
/* Step E2 */
static const char edi_nonalphanum_chars[] = "\015*> ";
if (debug_print) fputs("EDI ", stdout);
if (z_isdigit(source[sp])) {
cte_buffer[cte_p++] = source[sp] - '0' + 4;
} else if (z_isupper(source[sp])) {
cte_buffer[cte_p++] = source[sp] - 'A' + 14;
} else {
cte_buffer[cte_p++] = posn(edi_nonalphanum_chars, source[sp]);
}
if (cte_p >= 3) {
cte_p = c1_cte_buffer_transfer(cte_buffer, cte_p, target, &tp);
}
sp++;
}
} else if (current_mode == C1_DECIMAL) {
/* Step F - Decimal encodation */
if (debug_print) fputs("DEC ", stdout);
next_mode = C1_DECIMAL;
if (length - sp < 3) {
/* Step F1 */
const int bits_left = 8 - db_p;
const int can_ascii = bits_left == 8 && c1_is_last_single_ascii(source, length, sp);
if (c1_codewords_remaining(symbol, tp) == 1
&& (can_ascii || (num_digits[sp] == 1 && bits_left >= 4))) {
if (can_ascii) {
/* Encode last character or last 2 digits as ASCII */
if (is_twodigits(source, length, sp)) {
target[tp++] = (10 * ctoi(source[sp])) + ctoi(source[sp + 1]) + 130;
if (debug_print) printf("ASCDD(%.2s) ", source + sp);
sp += 2;
} else {
target[tp++] = source[sp] + 1;
if (debug_print) printf("ASC(%d) ", source[sp]);
sp++;
}
} else {
/* Encode last digit in 4 bits */
db_p = bin_append_posn(ctoi(source[sp]) + 1, 4, decimal_binary, db_p);
sp++;
if (bits_left == 6) {
db_p = bin_append_posn(1, 2, decimal_binary, db_p);
}
db_p = c1_decimal_binary_transfer(decimal_binary, db_p, target, &tp);
}
} else {
db_p = c1_decimal_unlatch(decimal_binary, db_p, target, &tp, num_digits[sp], source, &sp);
current_mode = C1_ASCII; /* Note need to set current_mode also in case exit loop */
}
next_mode = C1_ASCII;
} else {
if (num_digits[sp] < 3) {
/* Step F2 */
db_p = c1_decimal_unlatch(decimal_binary, db_p, target, &tp, num_digits[sp], source, &sp);
current_mode = next_mode = C1_ASCII; /* Note need to set current_mode also in case exit loop */
} else {
/* Step F3 */
/* There are three digits - convert the value to binary */
int value = (100 * ctoi(source[sp])) + (10 * ctoi(source[sp + 1])) + ctoi(source[sp + 2]) + 1;
db_p = bin_append_posn(value, 10, decimal_binary, db_p);
if (db_p >= 8) {
db_p = c1_decimal_binary_transfer(decimal_binary, db_p, target, &tp);
}
sp += 3;
}
}
} else if (current_mode == C1_BYTE) {
next_mode = C1_BYTE;
if (gs1 && source[sp] == '\x1D') {
next_mode = C1_ASCII;
} else {
if (source[sp] <= 127) {
next_mode = c1_look_ahead_test(source, length, sp, current_mode, gs1);
}
}
if (next_mode != C1_BYTE) {
/* Update byte field length */
int byte_count = tp - (byte_start + 1);
if (byte_count <= 249) {
target[byte_start] = byte_count;
} else {
/* Insert extra codeword */
memmove(target + byte_start + 2, target + byte_start + 1, sizeof(unsigned int) * byte_count);
target[byte_start] = 249 + (byte_count / 250);
target[byte_start + 1] = (byte_count % 250);
tp++;
}
} else {
if (debug_print) printf("BYTE(%d) ", source[sp]);
target[tp++] = source[sp];
sp++;
}
}
if (tp > C1_MAX_CWS) {
if (debug_print) fputc('\n', stdout);
/* Data is too large for symbol */
return 0;
}
} while (sp < length);
if (debug_print) {
printf("\nEnd Current Mode: %d, tp %d, cte_p %d, db_p %d\n", current_mode, tp, cte_p, db_p);
}
/* Empty buffers (note cte_buffer will be empty if current_mode C1_EDI) */
if (current_mode == C1_C40 || current_mode == C1_TEXT) {
if (cte_p >= 1) {
const int cws_remaining = c1_codewords_remaining(symbol, tp);
/* Note doing strict interpretation of spec here (same as BWIPP), as now also done in Data Matrix case */
if (cws_remaining == 1 && cte_p == 1 && c1_isc40text(current_mode, source[sp - 1])) {
/* 2.2.2.2 "...except when a single symbol character is left at the end before the first
error correction character. This single character is encoded in the ASCII code set." */
target[tp++] = source[sp - 1] + 1; /* As ASCII */
cte_p = 0;
} else if (cws_remaining == 2 && cte_p == 2) {
/* 2.2.2.2 "Two characters may be encoded in C40 mode in the last two data symbol characters of the
symbol as two C40 values followed by one of the C40 shift characters." */
cte_buffer[cte_p++] = 0; /* Shift 0 */
cte_p = c1_cte_buffer_transfer(cte_buffer, cte_p, target, &tp);
}
if (cte_p >= 1) {
int cnt, total_cnt = 0;
/* Backtrack to last complete triplet (same technique as BWIPP) */
while (sp > 0 && cte_p % 3) {
sp--;
cnt = c1_c40text_cnt(current_mode, gs1, source[sp]);
total_cnt += cnt;
cte_p -= cnt;
}
tp -= (total_cnt / 3) * 2;
target[tp++] = 255; /* Unlatch */
for (; sp < length; sp++) {
if (is_twodigits(source, length, sp)) {
target[tp++] = (10 * ctoi(source[sp])) + ctoi(source[sp + 1]) + 130;
sp++;
} else if (source[sp] & 0x80) {
target[tp++] = 235; /* FNC4 (Upper Shift) */
target[tp++] = (source[sp] - 128) + 1;
} else if (gs1 && source[sp] == '\x1D') {
target[tp++] = 232; /* FNC1 */
} else {
target[tp++] = source[sp] + 1;
}
}
current_mode = C1_ASCII;
}
}
} else if (current_mode == C1_DECIMAL) {
int bits_left;
/* Finish Decimal mode and go back to ASCII unless only one codeword remaining */
if (c1_codewords_remaining(symbol, tp) > 1) {
db_p = bin_append_posn(63, 6, decimal_binary, db_p); /* Unlatch */
}
if (db_p >= 8) {
db_p = c1_decimal_binary_transfer(decimal_binary, db_p, target, &tp);
}
bits_left = (8 - db_p) & 0x07;
if (bits_left) {
if ((bits_left == 4) || (bits_left == 6)) {
db_p = bin_append_posn(15, 4, decimal_binary, db_p);
}
if (bits_left == 2 || bits_left == 6) {
db_p = bin_append_posn(1, 2, decimal_binary, db_p);
}
(void) c1_decimal_binary_transfer(decimal_binary, db_p, target, &tp);
}
current_mode = C1_ASCII;
} else if (current_mode == C1_BYTE) {
/* Update byte field length unless no codewords remaining */
if (c1_codewords_remaining(symbol, tp) > 0) {
int byte_count = tp - (byte_start + 1);
if (byte_count <= 249) {
target[byte_start] = byte_count;
} else {
/* Insert extra byte field byte */
memmove(target + byte_start + 2, target + byte_start + 1, sizeof(unsigned int) * byte_count);
target[byte_start] = 249 + (byte_count / 250);
target[byte_start + 1] = (byte_count % 250);
tp++;
}
}
}
/* Re-check length of data */
if (tp > C1_MAX_CWS) {
/* Data is too large for symbol */
return 0;
}
*p_last_mode = current_mode;
if (debug_print) {
printf("Target (%d):", tp);
for (i = 0; i < tp; i++) {
printf(" [%d]", (int) target[i]);
}
printf("\nLast Mode: %d\n", *p_last_mode);
}
return tp;
}
/* Call `c1_encode()` for each segment */
static int c1_encode_segs(struct zint_symbol *symbol, struct zint_seg segs[], const int seg_count, const int gs1,
unsigned int target[], int *p_last_mode) {
int i;
int tp = 0;
for (i = 0; i < seg_count; i++) {
tp = c1_encode(symbol, segs[i].source, segs[i].length, segs[i].eci, seg_count, gs1, target, &tp, p_last_mode);
}
return tp;
}
/* Set symbol from datagrid */
static void c1_block_copy(struct zint_symbol *symbol, char datagrid[136][120], const int start_row,
const int start_col, const int height, const int width, const int row_offset, const int col_offset) {
int i, j;
for (i = start_row; i < (start_row + height); i++) {
for (j = start_col; j < (start_col + width); j++) {
if (datagrid[i][j]) {
set_module(symbol, i + row_offset, j + col_offset);
}
}
}
}
/* Get total length allowing for ECIs and escaping backslashes */
static int c1_total_length_segs(struct zint_seg segs[], const int seg_count) {
int total_len = 0;
int i;
if (segs[0].eci || seg_count > 1) {
for (i = 0; i < seg_count; i++) {
total_len += segs[i].length + 7 + chr_cnt(segs[i].source, segs[i].length, '\\');
}
} else {
total_len = segs[0].length;
}
return total_len;
}
INTERNAL int codeone(struct zint_symbol *symbol, struct zint_seg segs[], const int seg_count) {
int size = 1, i, j;
char datagrid[136][120];
int row, col;
int sub_version = 0;
rs_t rs;
const int gs1 = (symbol->input_mode & 0x07) == GS1_MODE;
const int debug_print = symbol->debug & ZINT_DEBUG_PRINT;
if ((symbol->option_2 < 0) || (symbol->option_2 > 10)) {
return errtxtf(ZINT_ERROR_INVALID_OPTION, symbol, 513, "Version '%d' out of range (1 to 10)",
symbol->option_2);
}
if (symbol->structapp.count) {
if (symbol->option_2 == 9) { /* Version S */
return errtxt(ZINT_ERROR_INVALID_OPTION, symbol, 714, "Structured Append not available for Version S");
}
if (gs1) {
return errtxt(ZINT_ERROR_INVALID_OPTION, symbol, 710,
"Cannot have Structured Append and GS1 mode at the same time");
}
if (symbol->structapp.count < 2 || symbol->structapp.count > 128) {
return errtxtf(ZINT_ERROR_INVALID_OPTION, symbol, 711,
"Structured Append count '%d' out of range (2 to 128)", symbol->structapp.count);
}
if (symbol->structapp.index < 1 || symbol->structapp.index > symbol->structapp.count) {
return errtxtf(ZINT_ERROR_INVALID_OPTION, symbol, 712,
"Structured Append index '%1$d' out of range (1 to count %2$d)",
symbol->structapp.index, symbol->structapp.count);
}
if (symbol->structapp.id[0]) {
return errtxt(ZINT_ERROR_INVALID_OPTION, symbol, 713, "Structured Append ID not available for Code One");
}
}
if (symbol->option_2 == 9) {
/* Version S */
int codewords;
large_uint elreg;
unsigned int target[30], ecc[15];
int block_width;
if (seg_count > 1) {
return errtxt(ZINT_ERROR_INVALID_OPTION, symbol, 715, "Multiple segments not supported for Version S");
}
if (segs[0].length > 18) {
return errtxtf(ZINT_ERROR_TOO_LONG, symbol, 514, "Input length %d too long for Version S (maximum 18)",
segs[0].length);
}
if ((i = not_sane(NEON_F, segs[0].source, segs[0].length))) {
return errtxtf(ZINT_ERROR_INVALID_DATA, symbol, 515,
"Invalid character at position %d in input (Version S encodes digits only)", i);
}
size = 9;
if (segs[0].length <= 6) {
/* Version S-10 */
sub_version = 1;
codewords = 4;
block_width = 2;
} else if (segs[0].length <= 12) {
/* Version S-20 */
sub_version = 2;
codewords = 8;
block_width = 4;
} else {
/* Version S-30 */
sub_version = 3;
codewords = 12;
block_width = 6;
}
if (debug_print) {
printf("Subversion: %d\n", sub_version);
}
/* Convert value plus one to binary */
large_load_str_u64(&elreg, segs[0].source, segs[0].length);
large_add_u64(&elreg, 1);
large_uint_array(&elreg, target, codewords, 5 /*bits*/);
rs_init_gf(&rs, 0x25);
rs_init_code(&rs, codewords, 0);
rs_encode_uint(&rs, codewords, target, ecc);
for (i = 0; i < codewords; i++) {
target[i + codewords] = ecc[codewords - i - 1];
}
if (debug_print) {
printf("Codewords (%d): ", codewords);
for (i = 0; i < codewords * 2; i++) printf(" %d", (int) target[i]);
fputc('\n', stdout);
}
i = 0;
for (row = 0; row < 2; row++) {
for (col = 0; col < block_width; col++) {
datagrid[row * 2][col * 5] = target[i] & 0x10;
datagrid[row * 2][(col * 5) + 1] = target[i] & 0x08;
datagrid[row * 2][(col * 5) + 2] = target[i] & 0x04;
datagrid[(row * 2) + 1][col * 5] = target[i] & 0x02;
datagrid[(row * 2) + 1][(col * 5) + 1] = target[i] & 0x01;
datagrid[row * 2][(col * 5) + 3] = target[i + 1] & 0x10;
datagrid[row * 2][(col * 5) + 4] = target[i + 1] & 0x08;
datagrid[(row * 2) + 1][(col * 5) + 2] = target[i + 1] & 0x04;
datagrid[(row * 2) + 1][(col * 5) + 3] = target[i + 1] & 0x02;
datagrid[(row * 2) + 1][(col * 5) + 4] = target[i + 1] & 0x01;
i += 2;
}
}
symbol->rows = 8;
symbol->width = 10 * sub_version + 1;
} else if (symbol->option_2 == 10) {
/* Version T */
unsigned int target[C1_MAX_CWS + C1_MAX_ECCS]; /* Use same buffer size as A to H to avail of loop checks */
unsigned int ecc[22];
int data_length;
int data_cw, ecc_cw, block_width;
int last_mode = 0; /* Suppress gcc 14 "-Wmaybe-uninitialized" false positive */
if ((i = c1_total_length_segs(segs, seg_count)) > 90) {
return errtxtf(ZINT_ERROR_TOO_LONG, symbol, 519, "Input length %d too long for Version T (maximum 90)",
i);
}
data_length = c1_encode_segs(symbol, segs, seg_count, gs1, target, &last_mode);
assert(data_length); /* Can't exceed C1_MAX_CWS as input <= 90 */
if (data_length > 38) {
return errtxtf(ZINT_ERROR_TOO_LONG, symbol, 516,
"Input too long for Version T, requires %d codewords (maximum 38)", data_length);
}
size = 10;
if (data_length <= 10) {
sub_version = 1;
data_cw = 10;
ecc_cw = 10;
block_width = 4;
} else if (data_length <= 24) {
sub_version = 2;
data_cw = 24;
ecc_cw = 16;
block_width = 8;
} else {
sub_version = 3;
data_cw = 38;
ecc_cw = 22;
block_width = 12;
}
if (debug_print) {
printf("Padding: %d, Subversion: %d\n", data_cw - data_length, sub_version);
}
/* If require padding */
if (data_cw > data_length) {
/* If did not finish in ASCII or BYTE mode, switch to ASCII */
if (last_mode != C1_ASCII && last_mode != C1_BYTE) {
target[data_length++] = 255; /* Unlatch */
}
for (i = data_length; i < data_cw; i++) {
target[i] = 129; /* Pad */
}
}
/* Calculate error correction data */
rs_init_gf(&rs, 0x12d);
rs_init_code(&rs, ecc_cw, 0);
rs_encode_uint(&rs, data_cw, target, ecc);
for (i = 0; i < ecc_cw; i++) {
target[data_cw + i] = ecc[ecc_cw - i - 1];
}
if (debug_print) {
printf("Codewords (%d):", data_cw + ecc_cw);
for (i = 0; i < data_cw + ecc_cw; i++) printf(" %d", (int) target[i]);
fputc('\n', stdout);
}
i = 0;
for (row = 0; row < 5; row++) {
for (col = 0; col < block_width; col++) {
datagrid[row * 2][col * 4] = target[i] & 0x80;
datagrid[row * 2][(col * 4) + 1] = target[i] & 0x40;
datagrid[row * 2][(col * 4) + 2] = target[i] & 0x20;
datagrid[row * 2][(col * 4) + 3] = target[i] & 0x10;
datagrid[(row * 2) + 1][col * 4] = target[i] & 0x08;
datagrid[(row * 2) + 1][(col * 4) + 1] = target[i] & 0x04;
datagrid[(row * 2) + 1][(col * 4) + 2] = target[i] & 0x02;
datagrid[(row * 2) + 1][(col * 4) + 3] = target[i] & 0x01;
i++;
}
}
symbol->rows = 16;
symbol->width = (sub_version * 16) + 1;
} else {
/* Versions A to H */
unsigned int target[C1_MAX_CWS + C1_MAX_ECCS];
unsigned int sub_data[185], sub_ecc[70];
int data_length;
int data_cw;
int blocks, data_blocks, ecc_blocks, ecc_length;
int last_mode;
data_length = c1_encode_segs(symbol, segs, seg_count, gs1, target, &last_mode);
if (data_length == 0) {
return errtxt(ZINT_ERROR_TOO_LONG, symbol, 517,
"Input too long, requires too many codewords (maximum " C1_MAX_CWS_S ")");
}
for (i = 7; i >= 0; i--) {
if (c1_data_length[i] >= data_length) {
size = i + 1;
}
}
if (symbol->option_2 > size) {
size = symbol->option_2;
}
if ((symbol->option_2 != 0) && (symbol->option_2 < size)) {
return errtxtf(ZINT_ERROR_TOO_LONG, symbol, 518,
"Input too long for Version %1$c, requires %2$d codewords (maximum %3$d)",
'A' + symbol->option_2 - 1, data_length, c1_data_length[symbol->option_2 - 1]);
}
data_cw = c1_data_length[size - 1];
/* If require padding */
if (data_cw > data_length) {
/* If did not finish in ASCII or BYTE mode, switch to ASCII */
if (last_mode != C1_ASCII && last_mode != C1_BYTE) {
target[data_length++] = 255; /* Unlatch */
}
if (debug_print) {
printf("Padding: %d\n", data_cw - data_length);
}
for (i = data_length; i < data_cw; i++) {
target[i] = 129; /* Pad */
}
} else if (debug_print) {
fputs("No padding\n", stdout);
}
/* Calculate error correction data */
blocks = c1_blocks[size - 1];
data_blocks = c1_data_blocks[size - 1];
ecc_blocks = c1_ecc_blocks[size - 1];
ecc_length = c1_ecc_length[size - 1];
rs_init_gf(&rs, 0x12d);
rs_init_code(&rs, ecc_blocks, 0);
for (i = 0; i < blocks; i++) {
for (j = 0; j < data_blocks; j++) {
sub_data[j] = target[j * blocks + i];
}
rs_encode_uint(&rs, data_blocks, sub_data, sub_ecc);
for (j = 0; j < ecc_blocks; j++) {
target[data_cw + j * blocks + i] = sub_ecc[ecc_blocks - 1 - j];
}
}
if (debug_print) {
printf("Codewords (%d):", data_cw + ecc_length);
for (i = 0; i < data_cw + ecc_length; i++) printf(" %d", (int) target[i]);
fputc('\n', stdout);
}
i = 0;
for (row = 0; row < c1_grid_height[size - 1]; row++) {
for (col = 0; col < c1_grid_width[size - 1]; col++) {
datagrid[row * 2][col * 4] = target[i] & 0x80;
datagrid[row * 2][(col * 4) + 1] = target[i] & 0x40;
datagrid[row * 2][(col * 4) + 2] = target[i] & 0x20;
datagrid[row * 2][(col * 4) + 3] = target[i] & 0x10;
datagrid[(row * 2) + 1][col * 4] = target[i] & 0x08;
datagrid[(row * 2) + 1][(col * 4) + 1] = target[i] & 0x04;
datagrid[(row * 2) + 1][(col * 4) + 2] = target[i] & 0x02;
datagrid[(row * 2) + 1][(col * 4) + 3] = target[i] & 0x01;
i++;
}
}
symbol->rows = c1_height[size - 1];
symbol->width = c1_width[size - 1];
}
if (debug_print) {
printf("Version: %d\n", size);
}
switch (size) {
case 1: /* Version A */
c1_central_finder(symbol, 6, 3, 1);
c1_vert(symbol, 4, 6, 1);
c1_vert(symbol, 12, 5, 0);
set_module(symbol, 5, 12);
c1_spigot(symbol, 0);
c1_spigot(symbol, 15);
c1_block_copy(symbol, datagrid, 0, 0, 5, 4, 0, 0);
c1_block_copy(symbol, datagrid, 0, 4, 5, 12, 0, 2);
c1_block_copy(symbol, datagrid, 5, 0, 5, 12, 6, 0);
c1_block_copy(symbol, datagrid, 5, 12, 5, 4, 6, 2);
break;
case 2: /* Version B */
c1_central_finder(symbol, 8, 4, 1);
c1_vert(symbol, 4, 8, 1);
c1_vert(symbol, 16, 7, 0);
set_module(symbol, 7, 16);
c1_spigot(symbol, 0);
c1_spigot(symbol, 21);
c1_block_copy(symbol, datagrid, 0, 0, 7, 4, 0, 0);
c1_block_copy(symbol, datagrid, 0, 4, 7, 16, 0, 2);
c1_block_copy(symbol, datagrid, 7, 0, 7, 16, 8, 0);
c1_block_copy(symbol, datagrid, 7, 16, 7, 4, 8, 2);
break;
case 3: /* Version C */
c1_central_finder(symbol, 11, 4, 2);
c1_vert(symbol, 4, 11, 1);
c1_vert(symbol, 26, 13, 1);
c1_vert(symbol, 4, 10, 0);
c1_vert(symbol, 26, 10, 0);
c1_spigot(symbol, 0);
c1_spigot(symbol, 27);
c1_block_copy(symbol, datagrid, 0, 0, 10, 4, 0, 0);
c1_block_copy(symbol, datagrid, 0, 4, 10, 20, 0, 2);
c1_block_copy(symbol, datagrid, 0, 24, 10, 4, 0, 4);
c1_block_copy(symbol, datagrid, 10, 0, 10, 4, 8, 0);
c1_block_copy(symbol, datagrid, 10, 4, 10, 20, 8, 2);
c1_block_copy(symbol, datagrid, 10, 24, 10, 4, 8, 4);
break;
case 4: /* Version D */
c1_central_finder(symbol, 16, 5, 1);
c1_vert(symbol, 4, 16, 1);
c1_vert(symbol, 20, 16, 1);
c1_vert(symbol, 36, 16, 1);
c1_vert(symbol, 4, 15, 0);
c1_vert(symbol, 20, 15, 0);
c1_vert(symbol, 36, 15, 0);
c1_spigot(symbol, 0);
c1_spigot(symbol, 12);
c1_spigot(symbol, 27);
c1_spigot(symbol, 39);
c1_block_copy(symbol, datagrid, 0, 0, 15, 4, 0, 0);
c1_block_copy(symbol, datagrid, 0, 4, 15, 14, 0, 2);
c1_block_copy(symbol, datagrid, 0, 18, 15, 14, 0, 4);
c1_block_copy(symbol, datagrid, 0, 32, 15, 4, 0, 6);
c1_block_copy(symbol, datagrid, 15, 0, 15, 4, 10, 0);
c1_block_copy(symbol, datagrid, 15, 4, 15, 14, 10, 2);
c1_block_copy(symbol, datagrid, 15, 18, 15, 14, 10, 4);
c1_block_copy(symbol, datagrid, 15, 32, 15, 4, 10, 6);
break;
case 5: /* Version E */
c1_central_finder(symbol, 22, 5, 2);
c1_vert(symbol, 4, 22, 1);
c1_vert(symbol, 26, 24, 1);
c1_vert(symbol, 48, 22, 1);
c1_vert(symbol, 4, 21, 0);
c1_vert(symbol, 26, 21, 0);
c1_vert(symbol, 48, 21, 0);
c1_spigot(symbol, 0);
c1_spigot(symbol, 12);
c1_spigot(symbol, 39);
c1_spigot(symbol, 51);
c1_block_copy(symbol, datagrid, 0, 0, 21, 4, 0, 0);
c1_block_copy(symbol, datagrid, 0, 4, 21, 20, 0, 2);
c1_block_copy(symbol, datagrid, 0, 24, 21, 20, 0, 4);
c1_block_copy(symbol, datagrid, 0, 44, 21, 4, 0, 6);
c1_block_copy(symbol, datagrid, 21, 0, 21, 4, 10, 0);
c1_block_copy(symbol, datagrid, 21, 4, 21, 20, 10, 2);
c1_block_copy(symbol, datagrid, 21, 24, 21, 20, 10, 4);
c1_block_copy(symbol, datagrid, 21, 44, 21, 4, 10, 6);
break;
case 6: /* Version F */
c1_central_finder(symbol, 31, 5, 3);
c1_vert(symbol, 4, 31, 1);
c1_vert(symbol, 26, 35, 1);
c1_vert(symbol, 48, 31, 1);
c1_vert(symbol, 70, 35, 1);
c1_vert(symbol, 4, 30, 0);
c1_vert(symbol, 26, 30, 0);
c1_vert(symbol, 48, 30, 0);
c1_vert(symbol, 70, 30, 0);
c1_spigot(symbol, 0);
c1_spigot(symbol, 12);
c1_spigot(symbol, 24);
c1_spigot(symbol, 45);
c1_spigot(symbol, 57);
c1_spigot(symbol, 69);
c1_block_copy(symbol, datagrid, 0, 0, 30, 4, 0, 0);
c1_block_copy(symbol, datagrid, 0, 4, 30, 20, 0, 2);
c1_block_copy(symbol, datagrid, 0, 24, 30, 20, 0, 4);
c1_block_copy(symbol, datagrid, 0, 44, 30, 20, 0, 6);
c1_block_copy(symbol, datagrid, 0, 64, 30, 4, 0, 8);
c1_block_copy(symbol, datagrid, 30, 0, 30, 4, 10, 0);
c1_block_copy(symbol, datagrid, 30, 4, 30, 20, 10, 2);
c1_block_copy(symbol, datagrid, 30, 24, 30, 20, 10, 4);
c1_block_copy(symbol, datagrid, 30, 44, 30, 20, 10, 6);
c1_block_copy(symbol, datagrid, 30, 64, 30, 4, 10, 8);
break;
case 7: /* Version G */
c1_central_finder(symbol, 47, 6, 2);
c1_vert(symbol, 6, 47, 1);
c1_vert(symbol, 27, 49, 1);
c1_vert(symbol, 48, 47, 1);
c1_vert(symbol, 69, 49, 1);
c1_vert(symbol, 90, 47, 1);
c1_vert(symbol, 6, 46, 0);
c1_vert(symbol, 27, 46, 0);
c1_vert(symbol, 48, 46, 0);
c1_vert(symbol, 69, 46, 0);
c1_vert(symbol, 90, 46, 0);
c1_spigot(symbol, 0);
c1_spigot(symbol, 12);
c1_spigot(symbol, 24);
c1_spigot(symbol, 36);
c1_spigot(symbol, 67);
c1_spigot(symbol, 79);
c1_spigot(symbol, 91);
c1_spigot(symbol, 103);
c1_block_copy(symbol, datagrid, 0, 0, 46, 6, 0, 0);
c1_block_copy(symbol, datagrid, 0, 6, 46, 19, 0, 2);
c1_block_copy(symbol, datagrid, 0, 25, 46, 19, 0, 4);
c1_block_copy(symbol, datagrid, 0, 44, 46, 19, 0, 6);
c1_block_copy(symbol, datagrid, 0, 63, 46, 19, 0, 8);
c1_block_copy(symbol, datagrid, 0, 82, 46, 6, 0, 10);
c1_block_copy(symbol, datagrid, 46, 0, 46, 6, 12, 0);
c1_block_copy(symbol, datagrid, 46, 6, 46, 19, 12, 2);
c1_block_copy(symbol, datagrid, 46, 25, 46, 19, 12, 4);
c1_block_copy(symbol, datagrid, 46, 44, 46, 19, 12, 6);
c1_block_copy(symbol, datagrid, 46, 63, 46, 19, 12, 8);
c1_block_copy(symbol, datagrid, 46, 82, 46, 6, 12, 10);
break;
case 8: /* Version H */
c1_central_finder(symbol, 69, 6, 3);
c1_vert(symbol, 6, 69, 1);
c1_vert(symbol, 26, 73, 1);
c1_vert(symbol, 46, 69, 1);
c1_vert(symbol, 66, 73, 1);
c1_vert(symbol, 86, 69, 1);
c1_vert(symbol, 106, 73, 1);
c1_vert(symbol, 126, 69, 1);
c1_vert(symbol, 6, 68, 0);
c1_vert(symbol, 26, 68, 0);
c1_vert(symbol, 46, 68, 0);
c1_vert(symbol, 66, 68, 0);
c1_vert(symbol, 86, 68, 0);
c1_vert(symbol, 106, 68, 0);
c1_vert(symbol, 126, 68, 0);
c1_spigot(symbol, 0);
c1_spigot(symbol, 12);
c1_spigot(symbol, 24);
c1_spigot(symbol, 36);
c1_spigot(symbol, 48);
c1_spigot(symbol, 60);
c1_spigot(symbol, 87);
c1_spigot(symbol, 99);
c1_spigot(symbol, 111);
c1_spigot(symbol, 123);
c1_spigot(symbol, 135);
c1_spigot(symbol, 147);
c1_block_copy(symbol, datagrid, 0, 0, 68, 6, 0, 0);
c1_block_copy(symbol, datagrid, 0, 6, 68, 18, 0, 2);
c1_block_copy(symbol, datagrid, 0, 24, 68, 18, 0, 4);
c1_block_copy(symbol, datagrid, 0, 42, 68, 18, 0, 6);
c1_block_copy(symbol, datagrid, 0, 60, 68, 18, 0, 8);
c1_block_copy(symbol, datagrid, 0, 78, 68, 18, 0, 10);
c1_block_copy(symbol, datagrid, 0, 96, 68, 18, 0, 12);
c1_block_copy(symbol, datagrid, 0, 114, 68, 6, 0, 14);
c1_block_copy(symbol, datagrid, 68, 0, 68, 6, 12, 0);
c1_block_copy(symbol, datagrid, 68, 6, 68, 18, 12, 2);
c1_block_copy(symbol, datagrid, 68, 24, 68, 18, 12, 4);
c1_block_copy(symbol, datagrid, 68, 42, 68, 18, 12, 6);
c1_block_copy(symbol, datagrid, 68, 60, 68, 18, 12, 8);
c1_block_copy(symbol, datagrid, 68, 78, 68, 18, 12, 10);
c1_block_copy(symbol, datagrid, 68, 96, 68, 18, 12, 12);
c1_block_copy(symbol, datagrid, 68, 114, 68, 6, 12, 14);
break;
case 9: /* Version S */
c1_horiz(symbol, 5, 1);
c1_horiz(symbol, 7, 1);
set_module(symbol, 6, 0);
set_module(symbol, 6, symbol->width - 1);
unset_module(symbol, 7, 1);
unset_module(symbol, 7, symbol->width - 2);
switch (sub_version) {
case 1: /* Version S-10 */
set_module(symbol, 0, 5);
c1_block_copy(symbol, datagrid, 0, 0, 4, 5, 0, 0);
c1_block_copy(symbol, datagrid, 0, 5, 4, 5, 0, 1);
break;
case 2: /* Version S-20 */
set_module(symbol, 0, 10);
set_module(symbol, 4, 10);
c1_block_copy(symbol, datagrid, 0, 0, 4, 10, 0, 0);
c1_block_copy(symbol, datagrid, 0, 10, 4, 10, 0, 1);
break;
case 3: /* Version S-30 */
set_module(symbol, 0, 15);
set_module(symbol, 4, 15);
set_module(symbol, 6, 15);
c1_block_copy(symbol, datagrid, 0, 0, 4, 15, 0, 0);
c1_block_copy(symbol, datagrid, 0, 15, 4, 15, 0, 1);
break;
}
break;
case 10: /* Version T */
c1_horiz(symbol, 11, 1);
c1_horiz(symbol, 13, 1);
c1_horiz(symbol, 15, 1);
set_module(symbol, 12, 0);
set_module(symbol, 12, symbol->width - 1);
set_module(symbol, 14, 0);
set_module(symbol, 14, symbol->width - 1);
unset_module(symbol, 13, 1);
unset_module(symbol, 13, symbol->width - 2);
unset_module(symbol, 15, 1);
unset_module(symbol, 15, symbol->width - 2);
switch (sub_version) {
case 1: /* Version T-16 */
set_module(symbol, 0, 8);
set_module(symbol, 10, 8);
c1_block_copy(symbol, datagrid, 0, 0, 10, 8, 0, 0);
c1_block_copy(symbol, datagrid, 0, 8, 10, 8, 0, 1);
break;
case 2: /* Version T-32 */
set_module(symbol, 0, 16);
set_module(symbol, 10, 16);
set_module(symbol, 12, 16);
c1_block_copy(symbol, datagrid, 0, 0, 10, 16, 0, 0);
c1_block_copy(symbol, datagrid, 0, 16, 10, 16, 0, 1);
break;
case 3: /* Verion T-48 */
set_module(symbol, 0, 24);
set_module(symbol, 10, 24);
set_module(symbol, 12, 24);
set_module(symbol, 14, 24);
c1_block_copy(symbol, datagrid, 0, 0, 10, 24, 0, 0);
c1_block_copy(symbol, datagrid, 0, 24, 10, 24, 0, 1);
break;
}
break;
}
for (i = 0; i < symbol->rows; i++) {
symbol->row_height[i] = 1;
}
symbol->height = symbol->rows;
if (symbol->option_2 == 9) { /* Version S */
if (symbol->eci || gs1) {
return errtxtf(ZINT_WARN_INVALID_OPTION, symbol, 511, "%s ignored for Version S",
symbol->eci && gs1 ? "ECI and GS1 mode" : symbol->eci ? "ECI" : "GS1 mode");
}
} else if (symbol->eci && gs1) {
return errtxt(ZINT_WARN_INVALID_OPTION, symbol, 512, "ECI ignored for GS1 mode");
}
return 0;
}
/* vim: set ts=4 sw=4 et : */