diff mupdf-source/thirdparty/zint/backend/2of5.c @ 2:b50eed0cc0ef upstream

ADD: MuPDF v1.26.7: the MuPDF source as downloaded by a default build of PyMuPDF 1.26.4. The directory name has changed: no version number in the expanded directory now.
author Franz Glasner <fzglas.hg@dom66.de>
date Mon, 15 Sep 2025 11:43:07 +0200
parents
children
line wrap: on
line diff
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/mupdf-source/thirdparty/zint/backend/2of5.c	Mon Sep 15 11:43:07 2025 +0200
@@ -0,0 +1,385 @@
+/* 2of5.c - Handles Code 2 of 5 barcodes */
+/*
+    libzint - the open source barcode library
+    Copyright (C) 2008-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 <stdio.h>
+#include "common.h"
+#include "gs1.h"
+
+/* First 5 of each entry Interleaved also */
+static const char C25MatrixTable[10][6] = {
+    {'1','1','3','3','1','1'}, {'3','1','1','1','3','1'}, {'1','3','1','1','3','1'}, {'3','3','1','1','1','1'},
+    {'1','1','3','1','3','1'}, {'3','1','3','1','1','1'}, {'1','3','3','1','1','1'}, {'1','1','1','3','3','1'},
+    {'3','1','1','3','1','1'}, {'1','3','1','3','1','1'}
+};
+
+static const char C25IndustTable[10][10] = {
+    {'1','1','1','1','3','1','3','1','1','1'}, {'3','1','1','1','1','1','1','1','3','1'},
+    {'1','1','3','1','1','1','1','1','3','1'}, {'3','1','3','1','1','1','1','1','1','1'},
+    {'1','1','1','1','3','1','1','1','3','1'}, {'3','1','1','1','3','1','1','1','1','1'},
+    {'1','1','3','1','3','1','1','1','1','1'}, {'1','1','1','1','1','1','3','1','3','1'},
+    {'3','1','1','1','1','1','3','1','1','1'}, {'1','1','3','1','1','1','3','1','1','1'}
+};
+
+/* Note `c25_common()` assumes Stop string length one less than Start */
+static const char C25MatrixStartStop[2][6] =    { {'4', '1', '1', '1', '1', '1'}, {'4', '1', '1', '1', '1'} };
+static const char C25IndustStartStop[2][6] =    { {'3', '1', '3', '1', '1', '1'}, {'3', '1', '1', '1', '3'} };
+static const char C25IataLogicStartStop[2][6] = { {'1', '1', '1', '1'},           {'3', '1', '1'} };
+
+/* Common to Standard (Matrix), Industrial, IATA, and Data Logic */
+static int c25_common(struct zint_symbol *symbol, const unsigned char source[], int length, const int max,
+            const int is_matrix, const char start_stop[2][6], const int start_length, const int error_base) {
+
+    int i;
+    char dest[818]; /* Largest destination 4 + (80 + 1) * 10 + 3 + 1 = 818 */
+    char *d = dest;
+    unsigned char temp[113 + 1 + 1]; /* Largest maximum 113 + optional check digit */
+    const int have_checkdigit = symbol->option_2 == 1 || symbol->option_2 == 2;
+
+    if (length > max) {
+        /* errtxt 301: 303: 305: 307: */
+        return errtxtf(ZINT_ERROR_TOO_LONG, symbol, error_base, "Input length %1$d too long (maximum %2$d)", length,
+                        max);
+    }
+    if ((i = not_sane(NEON_F, source, length))) {
+        /* Note: for all "at position" error messages, escape sequences not accounted for */
+        /* errtxt 302: 304: 306: 308: */
+        return errtxtf(ZINT_ERROR_INVALID_DATA, symbol, error_base + 1,
+                        "Invalid character at position %d in input (digits only)", i);
+    }
+
+    ustrcpy(temp, source);
+
+    if (have_checkdigit) {
+        /* Add standard GS1 check digit */
+        temp[length] = gs1_check_digit(source, length);
+        temp[++length] = '\0';
+        if (symbol->debug & ZINT_DEBUG_PRINT) printf("Check digit: %c\n", temp[length - 1]);
+    }
+
+    /* Start character */
+    memcpy(d, start_stop[0], start_length);
+    d += start_length;
+
+    if (is_matrix) {
+        for (i = 0; i < length; i++, d += 6) {
+            memcpy(d, C25MatrixTable[temp[i] - '0'], 6);
+        }
+    } else {
+        for (i = 0; i < length; i++, d += 10) {
+            memcpy(d, C25IndustTable[temp[i] - '0'], 10);
+        }
+    }
+
+    /* Stop character */
+    memcpy(d, start_stop[1], start_length - 1);
+    d += start_length - 1;
+
+    expand(symbol, dest, d - dest);
+
+    ustrcpy(symbol->text, temp);
+    if (symbol->option_2 == 2) {
+        /* Remove check digit from HRT */
+        symbol->text[length - 1] = '\0';
+    }
+
+    return 0;
+}
+
+/* Code 2 of 5 Standard (Code 2 of 5 Matrix) */
+INTERNAL int c25standard(struct zint_symbol *symbol, unsigned char source[], int length) {
+    /* 9 + (112 + 1) * 10 + 8 = 1147 */
+    return c25_common(symbol, source, length, 112, 1 /*is_matrix*/, C25MatrixStartStop, 6, 301);
+}
+
+/* Code 2 of 5 Industrial */
+INTERNAL int c25ind(struct zint_symbol *symbol, unsigned char source[], int length) {
+    /* 10 + (79 + 1) * 14 + 9 = 1139 */
+    return c25_common(symbol, source, length, 79, 0 /*is_matrix*/, C25IndustStartStop, 6, 303);
+}
+
+/* Code 2 of 5 IATA */
+INTERNAL int c25iata(struct zint_symbol *symbol, unsigned char source[], int length) {
+    /* 4 + (80 + 1) * 14 + 5 = 1143 */
+    return c25_common(symbol, source, length, 80, 0 /*is_matrix*/, C25IataLogicStartStop, 4, 305);
+}
+
+/* Code 2 of 5 Data Logic */
+INTERNAL int c25logic(struct zint_symbol *symbol, unsigned char source[], int length) {
+    /* 4 + (113 + 1) * 10 + 5 = 1149 */
+    return c25_common(symbol, source, length, 113, 1 /*is_matrix*/, C25IataLogicStartStop, 4, 307);
+}
+
+/* Common to Interleaved, ITF-14, DP Leitcode, DP Identcode */
+static int c25_inter_common(struct zint_symbol *symbol, unsigned char source[], int length,
+            const int checkdigit_option, const int dont_set_height) {
+    int i, j, error_number = 0;
+    char dest[638]; /* 4 + (125 + 1) * 5 + 3 + 1 = 638 */
+    char *d = dest;
+    unsigned char temp[125 + 1 + 1];
+    const int have_checkdigit = checkdigit_option == 1 || checkdigit_option == 2;
+
+    if (length > 125) { /* 4 + (125 + 1) * 9 + 5 = 1143 */
+        return errtxtf(ZINT_ERROR_TOO_LONG, symbol, 309, "Input length %d too long (maximum 125)", length);
+    }
+    if ((i = not_sane(NEON_F, source, length))) {
+        return errtxtf(ZINT_ERROR_INVALID_DATA, symbol, 310,
+                        "Invalid character at position %d in input (digits only)", i);
+    }
+
+    /* Input must be an even number of characters for Interlaced 2 of 5 to work:
+       if an odd number of characters has been entered and no check digit or an even number and have check digit
+       then add a leading zero */
+    if (have_checkdigit == !(length & 1)) {
+        temp[0] = '0';
+        memcpy(temp + 1, source, length++);
+    } else {
+        memcpy(temp, source, length);
+    }
+    temp[length] = '\0';
+
+    if (have_checkdigit) {
+        /* Add standard GS1 check digit */
+        temp[length] = gs1_check_digit(temp, length);
+        temp[++length] = '\0';
+    }
+
+    /* Start character */
+    memcpy(d, "1111", 4);
+    d += 4;
+
+    for (i = 0; i < length; i += 2) {
+        /* Look up the bars and the spaces */
+        const char *const bars = C25MatrixTable[temp[i] - '0'];
+        const char *const spaces = C25MatrixTable[temp[i + 1] - '0'];
+
+        /* Then merge (interlace) the strings together */
+        for (j = 0; j < 5; j++) {
+            *d++ = bars[j];
+            *d++ = spaces[j];
+        }
+    }
+
+    /* Stop character */
+    memcpy(d, "311", 3);
+    d += 3;
+
+    expand(symbol, dest, d - dest);
+
+    ustrcpy(symbol->text, temp);
+    if (checkdigit_option == 2) {
+        /* Remove check digit from HRT */
+        symbol->text[length - 1] = '\0';
+    }
+
+    if (!dont_set_height) {
+        if (symbol->output_options & COMPLIANT_HEIGHT) {
+            /* ISO/IEC 16390:2007 Section 4.4 min height 5mm or 15% of symbol width whichever greater where
+               (P = character pairs, N = wide/narrow ratio = 3)
+               width = (P(4N + 6) + N + 6)X = (length / 2) * 18 + 9 */
+            /* Taking min X = 0.330mm from Annex D.3.1 (application specification) */
+            const float min_height_min = 15.151515f; /* 5.0 / 0.33 */
+            float min_height = stripf((18.0f * (length / 2) + 9.0f) * 0.15f);
+            if (min_height < min_height_min) {
+                min_height = min_height_min;
+            }
+            /* Using 50 as default as none recommended */
+            error_number = set_height(symbol, min_height, min_height > 50.0f ? min_height : 50.0f, 0.0f,
+                                        0 /*no_errtxt*/);
+        } else {
+            (void) set_height(symbol, 0.0f, 50.0f, 0.0f, 1 /*no_errtxt*/);
+        }
+    }
+
+    return error_number;
+}
+
+/* Code 2 of 5 Interleaved ISO/IEC 16390:2007 */
+INTERNAL int c25inter(struct zint_symbol *symbol, unsigned char source[], int length) {
+    return c25_inter_common(symbol, source, length, symbol->option_2 /*checkdigit_option*/, 0 /*dont_set_height*/);
+}
+
+/* Interleaved 2-of-5 (ITF-14) */
+INTERNAL int itf14(struct zint_symbol *symbol, unsigned char source[], int length) {
+    int i, error_number, zeroes;
+    unsigned char localstr[16] = {0};
+
+    if (length > 13) {
+        return errtxtf(ZINT_ERROR_TOO_LONG, symbol, 311, "Input length %d too long (maximum 13)", length);
+    }
+
+    if ((i = not_sane(NEON_F, source, length))) {
+        return errtxtf(ZINT_ERROR_INVALID_DATA, symbol, 312,
+                        "Invalid character at position %d in input (digits only)", i);
+    }
+
+    /* Add leading zeros as required */
+    zeroes = 13 - length;
+    for (i = 0; i < zeroes; i++) {
+        localstr[i] = '0';
+    }
+    ustrcpy(localstr + zeroes, source);
+
+    /* Calculate the check digit - the same method used for EAN-13 */
+    localstr[13] = gs1_check_digit(localstr, 13);
+    localstr[14] = '\0';
+    error_number = c25_inter_common(symbol, localstr, 14, 0 /*checkdigit_option*/, 1 /*dont_set_height*/);
+    ustrcpy(symbol->text, localstr);
+
+    if (error_number < ZINT_ERROR) {
+        if (!(symbol->output_options & (BARCODE_BOX | BARCODE_BIND | BARCODE_BIND_TOP))) {
+            /* If no option has been selected then uses default box option */
+            symbol->output_options |= BARCODE_BOX;
+            if (symbol->border_width == 0) { /* Allow override if non-zero */
+                /* GS1 General Specifications 21.0.1 Sections 5.3.2.4 & 5.3.6 (4.83 / 1.016 ~ 4.75) */
+                symbol->border_width = 5; /* Note change from previous value 8 */
+            }
+        }
+
+        if (symbol->output_options & COMPLIANT_HEIGHT) {
+            /* GS1 General Specifications 21.0.1 5.12.3.2 table 2, including footnote (**): (note bind/box additional
+               to symbol->height), same as GS1-128: "in case of further space constraints"
+               height 5.8mm / 1.016mm (X max) ~ 5.7; default 31.75mm / 0.495mm ~ 64.14 */
+            const float min_height = 5.70866156f; /* 5.8 / 1.016 */
+            const float default_height = 64.1414108f; /* 31.75 / 0.495 */
+            error_number = set_height(symbol, min_height, default_height, 0.0f, 0 /*no_errtxt*/);
+        } else {
+            (void) set_height(symbol, 0.0f, 50.0f, 0.0f, 1 /*no_errtxt*/);
+        }
+    }
+
+    return error_number;
+}
+
+/* Deutsche Post check digit */
+static char c25_dp_check_digit(const unsigned int count) {
+    return itoc((10 - (count % 10)) % 10);
+}
+
+/* Deutsche Post Leitcode */
+/* Documentation (of a very incomplete and non-technical type):
+https://www.deutschepost.de/content/dam/dpag/images/D_d/dialogpost-schwer/dp-dialogpost-schwer-broschuere-072021.pdf
+*/
+INTERNAL int dpleit(struct zint_symbol *symbol, unsigned char source[], int length) {
+    int i, j, error_number;
+    unsigned int count;
+    int factor;
+    unsigned char localstr[16] = {0};
+    int zeroes;
+
+    count = 0;
+    if (length > 13) {
+        return errtxtf(ZINT_ERROR_TOO_LONG, symbol, 313, "Input length %d too long (maximum 13)", length);
+    }
+    if ((i = not_sane(NEON_F, source, length))) {
+        return errtxtf(ZINT_ERROR_INVALID_DATA, symbol, 314,
+                        "Invalid character at position %d in input (digits only)", i);
+    }
+
+    zeroes = 13 - length;
+    for (i = 0; i < zeroes; i++)
+        localstr[i] = '0';
+    ustrcpy(localstr + zeroes, source);
+
+    factor = 4;
+    for (i = 12; i >= 0; i--) {
+        count += factor * ctoi(localstr[i]);
+        factor ^= 0x0D; /* Toggles 4 and 9 */
+    }
+    localstr[13] = c25_dp_check_digit(count);
+    localstr[14] = '\0';
+    error_number = c25_inter_common(symbol, localstr, 14, 0 /*checkdigit_option*/, 1 /*dont_set_height*/);
+
+    /* HRT formatting as per DIALOGPOST SCHWER brochure but TEC-IT differs as do examples at
+       https://www.philaseiten.de/cgi-bin/index.pl?ST=8615&CP=0&F=1#M147 */
+    for (i = 0, j = 0; i <= 14; i++) {
+        symbol->text[j++] = localstr[i];
+        if (i == 4 || i == 7 || i == 10) {
+            symbol->text[j++] = '.';
+        }
+    }
+
+    /* TODO: Find documentation on BARCODE_DPLEIT dimensions/height */
+    /* Based on eyeballing DIALOGPOST SCHWER, using 72X as default */
+    (void) set_height(symbol, 0.0f, 72.0f, 0.0f, 1 /*no_errtxt*/);
+
+    return error_number;
+}
+
+/* Deutsche Post Identcode */
+/* See dpleit() for (sort of) documentation reference */
+INTERNAL int dpident(struct zint_symbol *symbol, unsigned char source[], int length) {
+    int i, j, error_number, zeroes;
+    unsigned int count;
+    int factor;
+    unsigned char localstr[16] = {0};
+
+    count = 0;
+    if (length > 11) {
+        return errtxtf(ZINT_ERROR_TOO_LONG, symbol, 315, "Input length %d too long (maximum 11)", length);
+    }
+    if ((i = not_sane(NEON_F, source, length))) {
+        return errtxtf(ZINT_ERROR_INVALID_DATA, symbol, 316,
+                        "Invalid character at position %d in input (digits only)", i);
+    }
+
+    zeroes = 11 - length;
+    for (i = 0; i < zeroes; i++)
+        localstr[i] = '0';
+    ustrcpy(localstr + zeroes, source);
+
+    factor = 4;
+    for (i = 10; i >= 0; i--) {
+        count += factor * ctoi(localstr[i]);
+        factor ^= 0x0D; /* Toggles 4 and 9 */
+    }
+    localstr[11] = c25_dp_check_digit(count);
+    localstr[12] = '\0';
+    error_number = c25_inter_common(symbol, localstr, 12, 0 /*checkdigit_option*/, 1 /*dont_set_height*/);
+
+    /* HRT formatting as per DIALOGPOST SCHWER brochure but TEC-IT differs as do other examples (see above) */
+    for (i = 0, j = 0; i <= 12; i++) {
+        symbol->text[j++] = localstr[i];
+        if (i == 1 || i == 4 || i == 7) {
+            symbol->text[j++] = '.';
+        } else if (i == 3 || i == 10) {
+            symbol->text[j++] = ' ';
+        }
+    }
+
+    /* TODO: Find documentation on BARCODE_DPIDENT dimensions/height */
+    /* Based on eyeballing DIALOGPOST SCHWER, using 72X as default */
+    (void) set_height(symbol, 0.0f, 72.0f, 0.0f, 1 /*no_errtxt*/);
+
+    return error_number;
+}
+
+/* vim: set ts=4 sw=4 et : */