2020-06-04 18:45:25 +01:00
/* code.c - Handles Code 11, 39, 39+, 93, PZN, Channel and VIN */
2008-07-13 21:15:55 +00:00
/*
libzint - the open source barcode library
2022-07-14 16:01:30 +01:00
Copyright ( C ) 2008 - 2022 Robin Stuart < rstuart114 @ gmail . com >
2013-05-16 19:26:38 +02:00
Redistribution and use in source and binary forms , with or without
modification , are permitted provided that the following conditions
are met :
2017-10-23 21:37:52 +02:00
1. Redistributions of source code must retain the above copyright
notice , this list of conditions and the following disclaimer .
2013-05-16 19:26:38 +02:00
2. Redistributions in binary form must reproduce the above copyright
notice , this list of conditions and the following disclaimer in the
2017-10-23 21:37:52 +02:00
documentation and / or other materials provided with the distribution .
2013-05-16 19:26:38 +02:00
3. Neither the name of the project nor the names of its contributors
may be used to endorse or promote products derived from this software
2017-10-23 21:37:52 +02:00
without specific prior written permission .
2013-05-16 19:26:38 +02:00
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS " AS IS " AND
ANY EXPRESS OR IMPLIED WARRANTIES , INCLUDING , BUT NOT LIMITED TO , THE
IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
ARE DISCLAIMED . IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE
FOR ANY DIRECT , INDIRECT , INCIDENTAL , SPECIAL , EXEMPLARY , OR CONSEQUENTIAL
DAMAGES ( INCLUDING , BUT NOT LIMITED TO , PROCUREMENT OF SUBSTITUTE GOODS
OR SERVICES ; LOSS OF USE , DATA , OR PROFITS ; OR BUSINESS INTERRUPTION )
HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY , WHETHER IN CONTRACT , STRICT
LIABILITY , OR TORT ( INCLUDING NEGLIGENCE OR OTHERWISE ) ARISING IN ANY WAY
2017-10-23 21:37:52 +02:00
OUT OF THE USE OF THIS SOFTWARE , EVEN IF ADVISED OF THE POSSIBILITY OF
2013-05-16 19:26:38 +02:00
SUCH DAMAGE .
2016-02-20 09:38:03 +00:00
*/
2022-07-14 16:01:30 +01:00
/* SPDX-License-Identifier: BSD-3-Clause */
2008-07-13 21:15:55 +00:00
/* In version 0.5 this file was 1,553 lines long! */
2020-06-04 18:45:25 +01:00
# include <assert.h>
2022-07-14 16:01:30 +01:00
# include <stdio.h>
2008-07-13 21:15:55 +00:00
# include "common.h"
2021-10-20 23:05:30 +01:00
# define SODIUM_MNS_F (IS_NUM_F | IS_MNS_F) /* SODIUM "0123456789-" */
/* Same as TECHNETIUM (HIBC) with "abcd" added for CODE93 */
static const char SILVER [ ] = " 0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZ-. $/+%abcd " ;
# define ARSENIC_F (IS_NUM_F | IS_ARS_F) /* ARSENIC "0123456789ABCDEFGHJKLMNPRSTUVWXYZ" */
2008-07-13 21:15:55 +00:00
2021-10-20 23:05:30 +01:00
static const char C11Table [ 11 ] [ 6 ] = {
{ ' 1 ' , ' 1 ' , ' 1 ' , ' 1 ' , ' 2 ' , ' 1 ' } , { ' 2 ' , ' 1 ' , ' 1 ' , ' 1 ' , ' 2 ' , ' 1 ' } , { ' 1 ' , ' 2 ' , ' 1 ' , ' 1 ' , ' 2 ' , ' 1 ' } , { ' 2 ' , ' 2 ' , ' 1 ' , ' 1 ' , ' 1 ' , ' 1 ' } ,
{ ' 1 ' , ' 1 ' , ' 2 ' , ' 1 ' , ' 2 ' , ' 1 ' } , { ' 2 ' , ' 1 ' , ' 2 ' , ' 1 ' , ' 1 ' , ' 1 ' } , { ' 1 ' , ' 2 ' , ' 2 ' , ' 1 ' , ' 1 ' , ' 1 ' } , { ' 1 ' , ' 1 ' , ' 1 ' , ' 2 ' , ' 2 ' , ' 1 ' } ,
{ ' 2 ' , ' 1 ' , ' 1 ' , ' 2 ' , ' 1 ' , ' 1 ' } , { ' 2 ' , ' 1 ' , ' 1 ' , ' 1 ' , ' 1 ' , ' 1 ' } , { ' 1 ' , ' 1 ' , ' 2 ' , ' 1 ' , ' 1 ' , ' 1 ' }
2016-02-20 09:38:03 +00:00
} ;
2008-07-13 21:15:55 +00:00
/* Code 39 tables checked against ISO/IEC 16388:2007 */
2012-12-29 19:37:03 +01:00
2008-07-13 21:15:55 +00:00
/* Incorporates Table A1 */
2021-10-20 23:05:30 +01:00
static const char C39Table [ 43 ] [ 10 ] = {
2016-02-20 09:38:03 +00:00
/* Code 39 character assignments (Table 1) */
2021-10-20 23:05:30 +01:00
{ ' 1 ' , ' 1 ' , ' 1 ' , ' 2 ' , ' 2 ' , ' 1 ' , ' 2 ' , ' 1 ' , ' 1 ' , ' 1 ' } , { ' 2 ' , ' 1 ' , ' 1 ' , ' 2 ' , ' 1 ' , ' 1 ' , ' 1 ' , ' 1 ' , ' 2 ' , ' 1 ' } ,
{ ' 1 ' , ' 1 ' , ' 2 ' , ' 2 ' , ' 1 ' , ' 1 ' , ' 1 ' , ' 1 ' , ' 2 ' , ' 1 ' } , { ' 2 ' , ' 1 ' , ' 2 ' , ' 2 ' , ' 1 ' , ' 1 ' , ' 1 ' , ' 1 ' , ' 1 ' , ' 1 ' } ,
{ ' 1 ' , ' 1 ' , ' 1 ' , ' 2 ' , ' 2 ' , ' 1 ' , ' 1 ' , ' 1 ' , ' 2 ' , ' 1 ' } , { ' 2 ' , ' 1 ' , ' 1 ' , ' 2 ' , ' 2 ' , ' 1 ' , ' 1 ' , ' 1 ' , ' 1 ' , ' 1 ' } ,
{ ' 1 ' , ' 1 ' , ' 2 ' , ' 2 ' , ' 2 ' , ' 1 ' , ' 1 ' , ' 1 ' , ' 1 ' , ' 1 ' } , { ' 1 ' , ' 1 ' , ' 1 ' , ' 2 ' , ' 1 ' , ' 1 ' , ' 2 ' , ' 1 ' , ' 2 ' , ' 1 ' } ,
{ ' 2 ' , ' 1 ' , ' 1 ' , ' 2 ' , ' 1 ' , ' 1 ' , ' 2 ' , ' 1 ' , ' 1 ' , ' 1 ' } , { ' 1 ' , ' 1 ' , ' 2 ' , ' 2 ' , ' 1 ' , ' 1 ' , ' 2 ' , ' 1 ' , ' 1 ' , ' 1 ' } ,
{ ' 2 ' , ' 1 ' , ' 1 ' , ' 1 ' , ' 1 ' , ' 2 ' , ' 1 ' , ' 1 ' , ' 2 ' , ' 1 ' } , { ' 1 ' , ' 1 ' , ' 2 ' , ' 1 ' , ' 1 ' , ' 2 ' , ' 1 ' , ' 1 ' , ' 2 ' , ' 1 ' } ,
{ ' 2 ' , ' 1 ' , ' 2 ' , ' 1 ' , ' 1 ' , ' 2 ' , ' 1 ' , ' 1 ' , ' 1 ' , ' 1 ' } , { ' 1 ' , ' 1 ' , ' 1 ' , ' 1 ' , ' 2 ' , ' 2 ' , ' 1 ' , ' 1 ' , ' 2 ' , ' 1 ' } ,
{ ' 2 ' , ' 1 ' , ' 1 ' , ' 1 ' , ' 2 ' , ' 2 ' , ' 1 ' , ' 1 ' , ' 1 ' , ' 1 ' } , { ' 1 ' , ' 1 ' , ' 2 ' , ' 1 ' , ' 2 ' , ' 2 ' , ' 1 ' , ' 1 ' , ' 1 ' , ' 1 ' } ,
{ ' 1 ' , ' 1 ' , ' 1 ' , ' 1 ' , ' 1 ' , ' 2 ' , ' 2 ' , ' 1 ' , ' 2 ' , ' 1 ' } , { ' 2 ' , ' 1 ' , ' 1 ' , ' 1 ' , ' 1 ' , ' 2 ' , ' 2 ' , ' 1 ' , ' 1 ' , ' 1 ' } ,
{ ' 1 ' , ' 1 ' , ' 2 ' , ' 1 ' , ' 1 ' , ' 2 ' , ' 2 ' , ' 1 ' , ' 1 ' , ' 1 ' } , { ' 1 ' , ' 1 ' , ' 1 ' , ' 1 ' , ' 2 ' , ' 2 ' , ' 2 ' , ' 1 ' , ' 1 ' , ' 1 ' } ,
{ ' 2 ' , ' 1 ' , ' 1 ' , ' 1 ' , ' 1 ' , ' 1 ' , ' 1 ' , ' 2 ' , ' 2 ' , ' 1 ' } , { ' 1 ' , ' 1 ' , ' 2 ' , ' 1 ' , ' 1 ' , ' 1 ' , ' 1 ' , ' 2 ' , ' 2 ' , ' 1 ' } ,
{ ' 2 ' , ' 1 ' , ' 2 ' , ' 1 ' , ' 1 ' , ' 1 ' , ' 1 ' , ' 2 ' , ' 1 ' , ' 1 ' } , { ' 1 ' , ' 1 ' , ' 1 ' , ' 1 ' , ' 2 ' , ' 1 ' , ' 1 ' , ' 2 ' , ' 2 ' , ' 1 ' } ,
{ ' 2 ' , ' 1 ' , ' 1 ' , ' 1 ' , ' 2 ' , ' 1 ' , ' 1 ' , ' 2 ' , ' 1 ' , ' 1 ' } , { ' 1 ' , ' 1 ' , ' 2 ' , ' 1 ' , ' 2 ' , ' 1 ' , ' 1 ' , ' 2 ' , ' 1 ' , ' 1 ' } ,
{ ' 1 ' , ' 1 ' , ' 1 ' , ' 1 ' , ' 1 ' , ' 1 ' , ' 2 ' , ' 2 ' , ' 2 ' , ' 1 ' } , { ' 2 ' , ' 1 ' , ' 1 ' , ' 1 ' , ' 1 ' , ' 1 ' , ' 2 ' , ' 2 ' , ' 1 ' , ' 1 ' } ,
{ ' 1 ' , ' 1 ' , ' 2 ' , ' 1 ' , ' 1 ' , ' 1 ' , ' 2 ' , ' 2 ' , ' 1 ' , ' 1 ' } , { ' 1 ' , ' 1 ' , ' 1 ' , ' 1 ' , ' 2 ' , ' 1 ' , ' 2 ' , ' 2 ' , ' 1 ' , ' 1 ' } ,
{ ' 2 ' , ' 2 ' , ' 1 ' , ' 1 ' , ' 1 ' , ' 1 ' , ' 1 ' , ' 1 ' , ' 2 ' , ' 1 ' } , { ' 1 ' , ' 2 ' , ' 2 ' , ' 1 ' , ' 1 ' , ' 1 ' , ' 1 ' , ' 1 ' , ' 2 ' , ' 1 ' } ,
{ ' 2 ' , ' 2 ' , ' 2 ' , ' 1 ' , ' 1 ' , ' 1 ' , ' 1 ' , ' 1 ' , ' 1 ' , ' 1 ' } , { ' 1 ' , ' 2 ' , ' 1 ' , ' 1 ' , ' 2 ' , ' 1 ' , ' 1 ' , ' 1 ' , ' 2 ' , ' 1 ' } ,
{ ' 2 ' , ' 2 ' , ' 1 ' , ' 1 ' , ' 2 ' , ' 1 ' , ' 1 ' , ' 1 ' , ' 1 ' , ' 1 ' } , { ' 1 ' , ' 2 ' , ' 2 ' , ' 1 ' , ' 2 ' , ' 1 ' , ' 1 ' , ' 1 ' , ' 1 ' , ' 1 ' } ,
{ ' 1 ' , ' 2 ' , ' 1 ' , ' 1 ' , ' 1 ' , ' 1 ' , ' 2 ' , ' 1 ' , ' 2 ' , ' 1 ' } , { ' 2 ' , ' 2 ' , ' 1 ' , ' 1 ' , ' 1 ' , ' 1 ' , ' 2 ' , ' 1 ' , ' 1 ' , ' 1 ' } ,
{ ' 1 ' , ' 2 ' , ' 2 ' , ' 1 ' , ' 1 ' , ' 1 ' , ' 2 ' , ' 1 ' , ' 1 ' , ' 1 ' } , { ' 1 ' , ' 2 ' , ' 1 ' , ' 2 ' , ' 1 ' , ' 2 ' , ' 1 ' , ' 1 ' , ' 1 ' , ' 1 ' } ,
{ ' 1 ' , ' 2 ' , ' 1 ' , ' 2 ' , ' 1 ' , ' 1 ' , ' 1 ' , ' 2 ' , ' 1 ' , ' 1 ' } , { ' 1 ' , ' 2 ' , ' 1 ' , ' 1 ' , ' 1 ' , ' 2 ' , ' 1 ' , ' 2 ' , ' 1 ' , ' 1 ' } ,
{ ' 1 ' , ' 1 ' , ' 1 ' , ' 2 ' , ' 1 ' , ' 2 ' , ' 1 ' , ' 2 ' , ' 1 ' , ' 1 ' }
2016-02-20 09:38:03 +00:00
} ;
2021-10-20 23:05:30 +01:00
static const char EC39Ctrl [ 128 ] [ 2 ] = {
2016-02-20 09:38:03 +00:00
/* Encoding the full ASCII character set in Code 39 (Table A2) */
2021-10-20 23:05:30 +01:00
{ ' % ' , ' U ' } , { ' $ ' , ' A ' } , { ' $ ' , ' B ' } , { ' $ ' , ' C ' } , { ' $ ' , ' D ' } , { ' $ ' , ' E ' } , { ' $ ' , ' F ' } , { ' $ ' , ' G ' } , { ' $ ' , ' H ' } , { ' $ ' , ' I ' } ,
{ ' $ ' , ' J ' } , { ' $ ' , ' K ' } , { ' $ ' , ' L ' } , { ' $ ' , ' M ' } , { ' $ ' , ' N ' } , { ' $ ' , ' O ' } , { ' $ ' , ' P ' } , { ' $ ' , ' Q ' } , { ' $ ' , ' R ' } , { ' $ ' , ' S ' } ,
{ ' $ ' , ' T ' } , { ' $ ' , ' U ' } , { ' $ ' , ' V ' } , { ' $ ' , ' W ' } , { ' $ ' , ' X ' } , { ' $ ' , ' Y ' } , { ' $ ' , ' Z ' } , { ' % ' , ' A ' } , { ' % ' , ' B ' } , { ' % ' , ' C ' } ,
{ ' % ' , ' D ' } , { ' % ' , ' E ' } , { " " } , { ' / ' , ' A ' } , { ' / ' , ' B ' } , { ' / ' , ' C ' } , { ' / ' , ' D ' } , { ' / ' , ' E ' } , { ' / ' , ' F ' } , { ' / ' , ' G ' } ,
{ ' / ' , ' H ' } , { ' / ' , ' I ' } , { ' / ' , ' J ' } , { ' / ' , ' K ' } , { ' / ' , ' L ' } , { " - " } , { " . " } , { ' / ' , ' O ' } , { " 0 " } , { " 1 " } ,
{ " 2 " } , { " 3 " } , { " 4 " } , { " 5 " } , { " 6 " } , { " 7 " } , { " 8 " } , { " 9 " } , { ' / ' , ' Z ' } , { ' % ' , ' F ' } ,
{ ' % ' , ' G ' } , { ' % ' , ' H ' } , { ' % ' , ' I ' } , { ' % ' , ' J ' } , { ' % ' , ' V ' } , { " A " } , { " B " } , { " C " } , { " D " } , { " E " } ,
{ " F " } , { " G " } , { " H " } , { " I " } , { " J " } , { " K " } , { " L " } , { " M " } , { " N " } , { " O " } ,
{ " P " } , { " Q " } , { " R " } , { " S " } , { " T " } , { " U " } , { " V " } , { " W " } , { " X " } , { " Y " } ,
{ " Z " } , { ' % ' , ' K ' } , { ' % ' , ' L ' } , { ' % ' , ' M ' } , { ' % ' , ' N ' } , { ' % ' , ' O ' } , { ' % ' , ' W ' } , { ' + ' , ' A ' } , { ' + ' , ' B ' } , { ' + ' , ' C ' } ,
{ ' + ' , ' D ' } , { ' + ' , ' E ' } , { ' + ' , ' F ' } , { ' + ' , ' G ' } , { ' + ' , ' H ' } , { ' + ' , ' I ' } , { ' + ' , ' J ' } , { ' + ' , ' K ' } , { ' + ' , ' L ' } , { ' + ' , ' M ' } ,
{ ' + ' , ' N ' } , { ' + ' , ' O ' } , { ' + ' , ' P ' } , { ' + ' , ' Q ' } , { ' + ' , ' R ' } , { ' + ' , ' S ' } , { ' + ' , ' T ' } , { ' + ' , ' U ' } , { ' + ' , ' V ' } , { ' + ' , ' W ' } ,
{ ' + ' , ' X ' } , { ' + ' , ' Y ' } , { ' + ' , ' Z ' } , { ' % ' , ' P ' } , { ' % ' , ' Q ' } , { ' % ' , ' R ' } , { ' % ' , ' S ' } , { ' % ' , ' T ' }
2016-02-20 09:38:03 +00:00
} ;
2021-10-20 23:05:30 +01:00
static const char C93Ctrl [ 128 ] [ 2 ] = {
{ ' b ' , ' U ' } , { ' a ' , ' A ' } , { ' a ' , ' B ' } , { ' a ' , ' C ' } , { ' a ' , ' D ' } , { ' a ' , ' E ' } , { ' a ' , ' F ' } , { ' a ' , ' G ' } , { ' a ' , ' H ' } , { ' a ' , ' I ' } ,
{ ' a ' , ' J ' } , { ' a ' , ' K ' } , { ' a ' , ' L ' } , { ' a ' , ' M ' } , { ' a ' , ' N ' } , { ' a ' , ' O ' } , { ' a ' , ' P ' } , { ' a ' , ' Q ' } , { ' a ' , ' R ' } , { ' a ' , ' S ' } ,
{ ' a ' , ' T ' } , { ' a ' , ' U ' } , { ' a ' , ' V ' } , { ' a ' , ' W ' } , { ' a ' , ' X ' } , { ' a ' , ' Y ' } , { ' a ' , ' Z ' } , { ' b ' , ' A ' } , { ' b ' , ' B ' } , { ' b ' , ' C ' } ,
{ ' b ' , ' D ' } , { ' b ' , ' E ' } , { " " } , { ' c ' , ' A ' } , { ' c ' , ' B ' } , { ' c ' , ' C ' } , { " $ " } , { " % " } , { ' c ' , ' F ' } , { ' c ' , ' G ' } ,
{ ' c ' , ' H ' } , { ' c ' , ' I ' } , { ' c ' , ' J ' } , { " + " } , { ' c ' , ' L ' } , { " - " } , { " . " } , { " / " } , { " 0 " } , { " 1 " } ,
{ " 2 " } , { " 3 " } , { " 4 " } , { " 5 " } , { " 6 " } , { " 7 " } , { " 8 " } , { " 9 " } , { ' c ' , ' Z ' } , { ' b ' , ' F ' } ,
{ ' b ' , ' G ' } , { ' b ' , ' H ' } , { ' b ' , ' I ' } , { ' b ' , ' J ' } , { ' b ' , ' V ' } , { " A " } , { " B " } , { " C " } , { " D " } , { " E " } ,
{ " F " } , { " G " } , { " H " } , { " I " } , { " J " } , { " K " } , { " L " } , { " M " } , { " N " } , { " O " } ,
{ " P " } , { " Q " } , { " R " } , { " S " } , { " T " } , { " U " } , { " V " } , { " W " } , { " X " } , { " Y " } ,
{ " Z " } , { ' b ' , ' K ' } , { ' b ' , ' L ' } , { ' b ' , ' M ' } , { ' b ' , ' N ' } , { ' b ' , ' O ' } , { ' b ' , ' W ' } , { ' d ' , ' A ' } , { ' d ' , ' B ' } , { ' d ' , ' C ' } ,
{ ' d ' , ' D ' } , { ' d ' , ' E ' } , { ' d ' , ' F ' } , { ' d ' , ' G ' } , { ' d ' , ' H ' } , { ' d ' , ' I ' } , { ' d ' , ' J ' } , { ' d ' , ' K ' } , { ' d ' , ' L ' } , { ' d ' , ' M ' } ,
{ ' d ' , ' N ' } , { ' d ' , ' O ' } , { ' d ' , ' P ' } , { ' d ' , ' Q ' } , { ' d ' , ' R ' } , { ' d ' , ' S ' } , { ' d ' , ' T ' } , { ' d ' , ' U ' } , { ' d ' , ' V ' } , { ' d ' , ' W ' } ,
{ ' d ' , ' X ' } , { ' d ' , ' Y ' } , { ' d ' , ' Z ' } , { ' b ' , ' P ' } , { ' b ' , ' Q ' } , { ' b ' , ' R ' } , { ' b ' , ' S ' } , { ' b ' , ' T ' }
2016-02-20 09:38:03 +00:00
} ;
2021-10-20 23:05:30 +01:00
static const char C93Table [ 47 ] [ 6 ] = {
{ ' 1 ' , ' 3 ' , ' 1 ' , ' 1 ' , ' 1 ' , ' 2 ' } , { ' 1 ' , ' 1 ' , ' 1 ' , ' 2 ' , ' 1 ' , ' 3 ' } , { ' 1 ' , ' 1 ' , ' 1 ' , ' 3 ' , ' 1 ' , ' 2 ' } , { ' 1 ' , ' 1 ' , ' 1 ' , ' 4 ' , ' 1 ' , ' 1 ' } ,
{ ' 1 ' , ' 2 ' , ' 1 ' , ' 1 ' , ' 1 ' , ' 3 ' } , { ' 1 ' , ' 2 ' , ' 1 ' , ' 2 ' , ' 1 ' , ' 2 ' } , { ' 1 ' , ' 2 ' , ' 1 ' , ' 3 ' , ' 1 ' , ' 1 ' } , { ' 1 ' , ' 1 ' , ' 1 ' , ' 1 ' , ' 1 ' , ' 4 ' } ,
{ ' 1 ' , ' 3 ' , ' 1 ' , ' 2 ' , ' 1 ' , ' 1 ' } , { ' 1 ' , ' 4 ' , ' 1 ' , ' 1 ' , ' 1 ' , ' 1 ' } , { ' 2 ' , ' 1 ' , ' 1 ' , ' 1 ' , ' 1 ' , ' 3 ' } , { ' 2 ' , ' 1 ' , ' 1 ' , ' 2 ' , ' 1 ' , ' 2 ' } ,
{ ' 2 ' , ' 1 ' , ' 1 ' , ' 3 ' , ' 1 ' , ' 1 ' } , { ' 2 ' , ' 2 ' , ' 1 ' , ' 1 ' , ' 1 ' , ' 2 ' } , { ' 2 ' , ' 2 ' , ' 1 ' , ' 2 ' , ' 1 ' , ' 1 ' } , { ' 2 ' , ' 3 ' , ' 1 ' , ' 1 ' , ' 1 ' , ' 1 ' } ,
{ ' 1 ' , ' 1 ' , ' 2 ' , ' 1 ' , ' 1 ' , ' 3 ' } , { ' 1 ' , ' 1 ' , ' 2 ' , ' 2 ' , ' 1 ' , ' 2 ' } , { ' 1 ' , ' 1 ' , ' 2 ' , ' 3 ' , ' 1 ' , ' 1 ' } , { ' 1 ' , ' 2 ' , ' 2 ' , ' 1 ' , ' 1 ' , ' 2 ' } ,
{ ' 1 ' , ' 3 ' , ' 2 ' , ' 1 ' , ' 1 ' , ' 1 ' } , { ' 1 ' , ' 1 ' , ' 1 ' , ' 1 ' , ' 2 ' , ' 3 ' } , { ' 1 ' , ' 1 ' , ' 1 ' , ' 2 ' , ' 2 ' , ' 2 ' } , { ' 1 ' , ' 1 ' , ' 1 ' , ' 3 ' , ' 2 ' , ' 1 ' } ,
{ ' 1 ' , ' 2 ' , ' 1 ' , ' 1 ' , ' 2 ' , ' 2 ' } , { ' 1 ' , ' 3 ' , ' 1 ' , ' 1 ' , ' 2 ' , ' 1 ' } , { ' 2 ' , ' 1 ' , ' 2 ' , ' 1 ' , ' 1 ' , ' 2 ' } , { ' 2 ' , ' 1 ' , ' 2 ' , ' 2 ' , ' 1 ' , ' 1 ' } ,
{ ' 2 ' , ' 1 ' , ' 1 ' , ' 1 ' , ' 2 ' , ' 2 ' } , { ' 2 ' , ' 1 ' , ' 1 ' , ' 2 ' , ' 2 ' , ' 1 ' } , { ' 2 ' , ' 2 ' , ' 1 ' , ' 1 ' , ' 2 ' , ' 1 ' } , { ' 2 ' , ' 2 ' , ' 2 ' , ' 1 ' , ' 1 ' , ' 1 ' } ,
{ ' 1 ' , ' 1 ' , ' 2 ' , ' 1 ' , ' 2 ' , ' 2 ' } , { ' 1 ' , ' 1 ' , ' 2 ' , ' 2 ' , ' 2 ' , ' 1 ' } , { ' 1 ' , ' 2 ' , ' 2 ' , ' 1 ' , ' 2 ' , ' 1 ' } , { ' 1 ' , ' 2 ' , ' 3 ' , ' 1 ' , ' 1 ' , ' 1 ' } ,
{ ' 1 ' , ' 2 ' , ' 1 ' , ' 1 ' , ' 3 ' , ' 1 ' } , { ' 3 ' , ' 1 ' , ' 1 ' , ' 1 ' , ' 1 ' , ' 2 ' } , { ' 3 ' , ' 1 ' , ' 1 ' , ' 2 ' , ' 1 ' , ' 1 ' } , { ' 3 ' , ' 2 ' , ' 1 ' , ' 1 ' , ' 1 ' , ' 1 ' } ,
{ ' 1 ' , ' 1 ' , ' 2 ' , ' 1 ' , ' 3 ' , ' 1 ' } , { ' 1 ' , ' 1 ' , ' 3 ' , ' 1 ' , ' 2 ' , ' 1 ' } , { ' 2 ' , ' 1 ' , ' 1 ' , ' 1 ' , ' 3 ' , ' 1 ' } , { ' 1 ' , ' 2 ' , ' 1 ' , ' 2 ' , ' 2 ' , ' 1 ' } ,
{ ' 3 ' , ' 1 ' , ' 2 ' , ' 1 ' , ' 1 ' , ' 1 ' } , { ' 3 ' , ' 1 ' , ' 1 ' , ' 1 ' , ' 2 ' , ' 1 ' } , { ' 1 ' , ' 2 ' , ' 2 ' , ' 2 ' , ' 1 ' , ' 1 ' }
2016-02-20 09:38:03 +00:00
} ;
2008-07-13 21:15:55 +00:00
/* *********************** CODE 11 ******************** */
Add compliant height, using ZINT_COMPLIANT_HEIGHT flag for back-compatibility
Rename barcode funcs to same as BARCODE_XXX name
library: barcode funcs array for dispatch, used for ZBarcode_ValidID() also
general: change is_sane() comparison to nonzero from ZINT_ERROR_INVALID_OPTION
MAILMARK: fuller error messages
CODABAR: add option to show check character in HRT
zint.h: use 0xNNNN for OR-able defines
GUI: add guard descent height reset button, add Zint version to window title,
static get_zint_version() method, use QStringLiteral (QSL shorthand),
use SIGNAL(toggled()), add errtxt "popup" and status bar, add icons,
add saveAs shortcut, add main menu, context menus and actions, add help,
reset_view() -> reset_colours(), add copy to clipboard as EMF/GIF/PNG/TIF,
lessen triggering of update_preview(), shorten names of getters/setters,
simplify/shorten some update_preview() logic in switch,
CODEONE disable structapp for Version S
qzint.cpp: add on_errored signal, add missing getters, add test
2021-10-09 00:13:39 +01:00
INTERNAL int code11 ( struct zint_symbol * symbol , unsigned char source [ ] , int length ) { /* Code 11 */
2016-02-20 09:38:03 +00:00
2020-06-04 18:45:25 +01:00
int i ;
2016-02-20 09:38:03 +00:00
int h , c_digit , c_weight , c_count , k_digit , k_weight , k_count ;
Add compliant height, using ZINT_COMPLIANT_HEIGHT flag for back-compatibility
Rename barcode funcs to same as BARCODE_XXX name
library: barcode funcs array for dispatch, used for ZBarcode_ValidID() also
general: change is_sane() comparison to nonzero from ZINT_ERROR_INVALID_OPTION
MAILMARK: fuller error messages
CODABAR: add option to show check character in HRT
zint.h: use 0xNNNN for OR-able defines
GUI: add guard descent height reset button, add Zint version to window title,
static get_zint_version() method, use QStringLiteral (QSL shorthand),
use SIGNAL(toggled()), add errtxt "popup" and status bar, add icons,
add saveAs shortcut, add main menu, context menus and actions, add help,
reset_view() -> reset_colours(), add copy to clipboard as EMF/GIF/PNG/TIF,
lessen triggering of update_preview(), shorten names of getters/setters,
simplify/shorten some update_preview() logic in switch,
CODEONE disable structapp for Version S
qzint.cpp: add on_errored signal, add missing getters, add test
2021-10-09 00:13:39 +01:00
int weight [ 122 ] , error_number = 0 ;
2020-06-04 18:45:25 +01:00
char dest [ 750 ] ; /* 6 + 121 * 6 + 2 * 6 + 5 + 1 == 750 */
2021-10-20 23:05:30 +01:00
char * d = dest ;
2020-06-04 18:45:25 +01:00
int num_check_digits ;
2021-10-20 23:05:30 +01:00
char checkstr [ 3 ] = { 0 } ;
2021-10-30 22:00:31 +01:00
static const char checkchrs [ 11 ] = { ' 0 ' , ' 1 ' , ' 2 ' , ' 3 ' , ' 4 ' , ' 5 ' , ' 6 ' , ' 7 ' , ' 8 ' , ' 9 ' , ' - ' } ;
2020-06-04 18:45:25 +01:00
/* Suppresses clang-tidy clang-analyzer-core.UndefinedBinaryOperatorResult warning */
assert ( length > 0 ) ;
2016-02-20 09:38:03 +00:00
if ( length > 121 ) {
2021-06-27 11:47:55 +01:00
strcpy ( symbol - > errtxt , " 320: Input too long (121 character maximum) " ) ;
2016-02-20 09:38:03 +00:00
return ZINT_ERROR_TOO_LONG ;
}
2021-10-20 23:05:30 +01:00
if ( ! is_sane ( SODIUM_MNS_F , source , length ) ) {
2021-07-06 19:53:31 +01:00
strcpy ( symbol - > errtxt , " 321: Invalid character in data (digits and \" - \" only) " ) ;
Add compliant height, using ZINT_COMPLIANT_HEIGHT flag for back-compatibility
Rename barcode funcs to same as BARCODE_XXX name
library: barcode funcs array for dispatch, used for ZBarcode_ValidID() also
general: change is_sane() comparison to nonzero from ZINT_ERROR_INVALID_OPTION
MAILMARK: fuller error messages
CODABAR: add option to show check character in HRT
zint.h: use 0xNNNN for OR-able defines
GUI: add guard descent height reset button, add Zint version to window title,
static get_zint_version() method, use QStringLiteral (QSL shorthand),
use SIGNAL(toggled()), add errtxt "popup" and status bar, add icons,
add saveAs shortcut, add main menu, context menus and actions, add help,
reset_view() -> reset_colours(), add copy to clipboard as EMF/GIF/PNG/TIF,
lessen triggering of update_preview(), shorten names of getters/setters,
simplify/shorten some update_preview() logic in switch,
CODEONE disable structapp for Version S
qzint.cpp: add on_errored signal, add missing getters, add test
2021-10-09 00:13:39 +01:00
return ZINT_ERROR_INVALID_DATA ;
2016-02-20 09:38:03 +00:00
}
2020-06-04 18:45:25 +01:00
if ( symbol - > option_2 < 0 | | symbol - > option_2 > 2 ) {
2021-06-27 11:47:55 +01:00
strcpy ( symbol - > errtxt , " 339: Invalid check digit version (1, 2 only) " ) ;
2020-06-04 18:45:25 +01:00
return ZINT_ERROR_INVALID_OPTION ;
}
if ( symbol - > option_2 = = 2 ) {
num_check_digits = 0 ;
} else if ( symbol - > option_2 = = 1 ) {
num_check_digits = 1 ;
} else {
num_check_digits = 2 ;
}
2016-02-20 09:38:03 +00:00
c_weight = 1 ;
c_count = 0 ;
k_weight = 1 ;
k_count = 0 ;
/* start character */
2021-10-20 23:05:30 +01:00
memcpy ( d , " 112211 " , 6 ) ;
d + = 6 ;
2016-02-20 09:38:03 +00:00
/* Draw main body of barcode */
2021-10-20 23:05:30 +01:00
for ( i = 0 ; i < length ; i + + , d + = 6 ) {
2016-02-20 09:38:03 +00:00
if ( source [ i ] = = ' - ' )
weight [ i ] = 10 ;
else
weight [ i ] = ctoi ( source [ i ] ) ;
2021-10-20 23:05:30 +01:00
memcpy ( d , C11Table [ weight [ i ] ] , 6 ) ;
2016-02-20 09:38:03 +00:00
}
2020-06-04 18:45:25 +01:00
if ( num_check_digits ) {
/* Calculate C checksum */
for ( h = length - 1 ; h > = 0 ; h - - ) {
c_count + = ( c_weight * weight [ h ] ) ;
c_weight + + ;
2016-02-20 09:38:03 +00:00
2020-06-04 18:45:25 +01:00
if ( c_weight > 10 ) {
c_weight = 1 ;
}
2016-02-20 09:38:03 +00:00
}
2020-06-04 18:45:25 +01:00
c_digit = c_count % 11 ;
2016-02-20 09:38:03 +00:00
2021-10-20 23:05:30 +01:00
checkstr [ 0 ] = checkchrs [ c_digit ] ;
memcpy ( d , C11Table [ c_digit ] , 6 ) ;
d + = 6 ;
if ( num_check_digits = = 2 ) {
2020-06-04 18:45:25 +01:00
weight [ length ] = c_digit ;
2016-02-20 09:38:03 +00:00
2020-06-04 18:45:25 +01:00
/* Calculate K checksum */
for ( h = length ; h > = 0 ; h - - ) {
k_count + = ( k_weight * weight [ h ] ) ;
k_weight + + ;
2016-02-20 09:38:03 +00:00
2020-06-04 18:45:25 +01:00
if ( k_weight > 9 ) {
k_weight = 1 ;
}
}
k_digit = k_count % 11 ;
2016-02-20 09:38:03 +00:00
2021-10-20 23:05:30 +01:00
checkstr [ 1 ] = checkchrs [ k_digit ] ;
memcpy ( d , C11Table [ k_digit ] , 6 ) ;
d + = 6 ;
2020-06-04 18:45:25 +01:00
}
2016-02-20 09:38:03 +00:00
}
2020-07-10 19:39:32 +01:00
if ( symbol - > debug & ZINT_DEBUG_PRINT ) {
printf ( " Check digit (%d): %s \n " , num_check_digits , num_check_digits ? checkstr : " <none> " ) ;
}
2016-02-20 09:38:03 +00:00
/* Stop character */
2021-10-20 23:05:30 +01:00
memcpy ( d , " 11221 " , 5 ) ;
d + = 5 ;
2016-02-20 09:38:03 +00:00
2021-10-20 23:05:30 +01:00
expand ( symbol , dest , d - dest ) ;
2016-02-20 09:38:03 +00:00
2022-07-14 16:01:30 +01:00
/* TODO: Find documentation on BARCODE_CODE11 dimensions/height */
2021-06-19 13:11:23 +01:00
2016-02-20 09:38:03 +00:00
ustrcpy ( symbol - > text , source ) ;
2020-06-04 18:45:25 +01:00
if ( num_check_digits ) {
ustrcat ( symbol - > text , checkstr ) ;
}
2016-02-20 09:38:03 +00:00
return error_number ;
2008-07-13 21:15:55 +00:00
}
2016-02-20 09:38:03 +00:00
/* Code 39 */
Add compliant height, using ZINT_COMPLIANT_HEIGHT flag for back-compatibility
Rename barcode funcs to same as BARCODE_XXX name
library: barcode funcs array for dispatch, used for ZBarcode_ValidID() also
general: change is_sane() comparison to nonzero from ZINT_ERROR_INVALID_OPTION
MAILMARK: fuller error messages
CODABAR: add option to show check character in HRT
zint.h: use 0xNNNN for OR-able defines
GUI: add guard descent height reset button, add Zint version to window title,
static get_zint_version() method, use QStringLiteral (QSL shorthand),
use SIGNAL(toggled()), add errtxt "popup" and status bar, add icons,
add saveAs shortcut, add main menu, context menus and actions, add help,
reset_view() -> reset_colours(), add copy to clipboard as EMF/GIF/PNG/TIF,
lessen triggering of update_preview(), shorten names of getters/setters,
simplify/shorten some update_preview() logic in switch,
CODEONE disable structapp for Version S
qzint.cpp: add on_errored signal, add missing getters, add test
2021-10-09 00:13:39 +01:00
INTERNAL int code39 ( struct zint_symbol * symbol , unsigned char source [ ] , int length ) {
2020-06-04 18:45:25 +01:00
int i ;
int counter ;
Add compliant height, using ZINT_COMPLIANT_HEIGHT flag for back-compatibility
Rename barcode funcs to same as BARCODE_XXX name
library: barcode funcs array for dispatch, used for ZBarcode_ValidID() also
general: change is_sane() comparison to nonzero from ZINT_ERROR_INVALID_OPTION
MAILMARK: fuller error messages
CODABAR: add option to show check character in HRT
zint.h: use 0xNNNN for OR-able defines
GUI: add guard descent height reset button, add Zint version to window title,
static get_zint_version() method, use QStringLiteral (QSL shorthand),
use SIGNAL(toggled()), add errtxt "popup" and status bar, add icons,
add saveAs shortcut, add main menu, context menus and actions, add help,
reset_view() -> reset_colours(), add copy to clipboard as EMF/GIF/PNG/TIF,
lessen triggering of update_preview(), shorten names of getters/setters,
simplify/shorten some update_preview() logic in switch,
CODEONE disable structapp for Version S
qzint.cpp: add on_errored signal, add missing getters, add test
2021-10-09 00:13:39 +01:00
int error_number = 0 ;
2021-10-20 23:05:30 +01:00
int posns [ 85 ] ;
2020-06-04 18:45:25 +01:00
char dest [ 880 ] ; /* 10 (Start) + 85 * 10 + 10 (Check) + 9 (Stop) + 1 = 880 */
2021-10-20 23:05:30 +01:00
char * d = dest ;
2016-02-20 09:38:03 +00:00
char localstr [ 2 ] = { 0 } ;
counter = 0 ;
if ( ( symbol - > option_2 < 0 ) | | ( symbol - > option_2 > 1 ) ) {
symbol - > option_2 = 0 ;
}
2021-10-20 23:05:30 +01:00
/* LOGMARS MIL-STD-1189 Rev. B https://apps.dtic.mil/dtic/tr/fulltext/u2/a473534.pdf */
2020-06-04 18:45:25 +01:00
if ( ( symbol - > symbology = = BARCODE_LOGMARS ) & & ( length > 30 ) ) { /* MIL-STD-1189 Rev. B Section 5.2.6.2 */
2021-06-27 11:47:55 +01:00
strcpy ( symbol - > errtxt , " 322: Input too long (30 character maximum) " ) ;
2016-02-20 09:38:03 +00:00
return ZINT_ERROR_TOO_LONG ;
2021-06-19 13:11:23 +01:00
/* Prevent encoded_data out-of-bounds >= 143 for BARCODE_HIBC_39 due to wider 'wide' bars */
2021-07-06 19:53:31 +01:00
} else if ( ( symbol - > symbology = = BARCODE_HIBC_39 ) & & ( length > 69 ) ) {
2021-06-27 11:47:55 +01:00
/* Note use 319 (2of5 range) as 340 taken by CODE128 */
2021-07-06 19:53:31 +01:00
strcpy ( symbol - > errtxt , " 319: Input too long (67 character maximum) " ) ; /* 69 less '+' and check */
2020-06-04 18:45:25 +01:00
return ZINT_ERROR_TOO_LONG ;
} else if ( length > 85 ) {
2021-06-27 11:47:55 +01:00
strcpy ( symbol - > errtxt , " 323: Input too long (85 character maximum) " ) ;
2016-02-20 09:38:03 +00:00
return ZINT_ERROR_TOO_LONG ;
}
2021-10-20 23:05:30 +01:00
to_upper ( source , length ) ;
if ( ! is_sane_lookup ( SILVER , 43 , source , length , posns ) ) {
2021-07-06 19:53:31 +01:00
strcpy ( symbol - > errtxt , " 324: Invalid character in data (alphanumerics, space and \" -.$/+% \" only) " ) ;
Add compliant height, using ZINT_COMPLIANT_HEIGHT flag for back-compatibility
Rename barcode funcs to same as BARCODE_XXX name
library: barcode funcs array for dispatch, used for ZBarcode_ValidID() also
general: change is_sane() comparison to nonzero from ZINT_ERROR_INVALID_OPTION
MAILMARK: fuller error messages
CODABAR: add option to show check character in HRT
zint.h: use 0xNNNN for OR-able defines
GUI: add guard descent height reset button, add Zint version to window title,
static get_zint_version() method, use QStringLiteral (QSL shorthand),
use SIGNAL(toggled()), add errtxt "popup" and status bar, add icons,
add saveAs shortcut, add main menu, context menus and actions, add help,
reset_view() -> reset_colours(), add copy to clipboard as EMF/GIF/PNG/TIF,
lessen triggering of update_preview(), shorten names of getters/setters,
simplify/shorten some update_preview() logic in switch,
CODEONE disable structapp for Version S
qzint.cpp: add on_errored signal, add missing getters, add test
2021-10-09 00:13:39 +01:00
return ZINT_ERROR_INVALID_DATA ;
2016-02-20 09:38:03 +00:00
}
/* Start character */
2021-10-20 23:05:30 +01:00
memcpy ( d , " 1211212111 " , 10 ) ;
d + = 10 ;
2016-02-20 09:38:03 +00:00
2021-10-20 23:05:30 +01:00
for ( i = 0 ; i < length ; i + + , d + = 10 ) {
memcpy ( d , C39Table [ posns [ i ] ] , 10 ) ;
counter + = posns [ i ] ;
2016-02-20 09:38:03 +00:00
}
2020-06-04 18:45:25 +01:00
if ( symbol - > option_2 = = 1 ) {
2016-02-20 09:38:03 +00:00
2017-09-10 12:03:09 -03:00
char check_digit ;
2021-07-26 15:29:05 +01:00
counter % = 43 ;
check_digit = SILVER [ counter ] ;
2021-10-20 23:05:30 +01:00
memcpy ( d , C39Table [ counter ] , 10 ) ;
d + = 10 ;
2016-02-20 09:38:03 +00:00
/* Display a space check digit as _, otherwise it looks like an error */
if ( check_digit = = ' ' ) {
check_digit = ' _ ' ;
}
localstr [ 0 ] = check_digit ;
localstr [ 1 ] = ' \0 ' ;
}
/* Stop character */
2021-10-20 23:05:30 +01:00
memcpy ( d , " 121121211 " , 9 ) ;
d + = 9 ;
2016-02-20 09:38:03 +00:00
if ( ( symbol - > symbology = = BARCODE_LOGMARS ) | | ( symbol - > symbology = = BARCODE_HIBC_39 ) ) {
/* LOGMARS uses wider 'wide' bars than normal Code 39 */
2021-10-20 23:05:30 +01:00
counter = d - dest ;
2016-02-20 09:38:03 +00:00
for ( i = 0 ; i < counter ; i + + ) {
if ( dest [ i ] = = ' 2 ' ) {
dest [ i ] = ' 3 ' ;
}
}
}
2021-10-20 23:05:30 +01:00
if ( symbol - > debug & ZINT_DEBUG_PRINT ) {
printf ( " Barspaces: %.*s \n " , ( int ) ( d - dest ) , dest ) ;
}
2021-05-15 12:23:46 +01:00
2021-10-20 23:05:30 +01:00
expand ( symbol , dest , d - dest ) ;
2016-02-20 09:38:03 +00:00
Add compliant height, using ZINT_COMPLIANT_HEIGHT flag for back-compatibility
Rename barcode funcs to same as BARCODE_XXX name
library: barcode funcs array for dispatch, used for ZBarcode_ValidID() also
general: change is_sane() comparison to nonzero from ZINT_ERROR_INVALID_OPTION
MAILMARK: fuller error messages
CODABAR: add option to show check character in HRT
zint.h: use 0xNNNN for OR-able defines
GUI: add guard descent height reset button, add Zint version to window title,
static get_zint_version() method, use QStringLiteral (QSL shorthand),
use SIGNAL(toggled()), add errtxt "popup" and status bar, add icons,
add saveAs shortcut, add main menu, context menus and actions, add help,
reset_view() -> reset_colours(), add copy to clipboard as EMF/GIF/PNG/TIF,
lessen triggering of update_preview(), shorten names of getters/setters,
simplify/shorten some update_preview() logic in switch,
CODEONE disable structapp for Version S
qzint.cpp: add on_errored signal, add missing getters, add test
2021-10-09 00:13:39 +01:00
if ( symbol - > output_options & COMPLIANT_HEIGHT ) {
if ( symbol - > symbology = = BARCODE_LOGMARS ) {
/* MIL-STD-1189 Rev. B Section 5.2
Min height 0.25 " / 0.04 " ( X max ) = 6.25
Default height 0.625 " (average of 0.375 " - 0.875 " ) / 0.01375 " ( average of 0.0075 " - 0.02 " ) ~ 45.45 */
error_number = set_height ( symbol , 6.25f , stripf ( 0.625f / 0.01375f ) , stripf ( 0.875f / 0.0075f ) ,
0 /*no_errtxt*/ ) ;
} else if ( symbol - > symbology = = BARCODE_CODE39 | | symbol - > symbology = = BARCODE_EXCODE39
| | symbol - > symbology = = BARCODE_HIBC_39 ) {
/* ISO/IEC 16388:2007 4.4 (e) recommended min height 5.0mm or 15% of width excluding quiet zones;
as X left to application specification use
width = ( C + 2 ) * ( 3 * N + 6 ) * X + ( C + 1 ) * I = ( C + 2 ) * 9 + C + 1 ) * X = ( 10 * C + 19 ) ;
use 50 as default as none recommended */
const float min_height = stripf ( ( 10.0f * ( symbol - > option_2 = = 1 ? length + 1 : length ) + 19.0f ) * 0.15f ) ;
error_number = set_height ( symbol , min_height , min_height > 50.0f ? min_height : 50.0f , 0.0f ,
0 /*no_errtxt*/ ) ;
}
2022-07-14 16:01:30 +01:00
/* PZN and CODE32 set their own heights */
Add compliant height, using ZINT_COMPLIANT_HEIGHT flag for back-compatibility
Rename barcode funcs to same as BARCODE_XXX name
library: barcode funcs array for dispatch, used for ZBarcode_ValidID() also
general: change is_sane() comparison to nonzero from ZINT_ERROR_INVALID_OPTION
MAILMARK: fuller error messages
CODABAR: add option to show check character in HRT
zint.h: use 0xNNNN for OR-able defines
GUI: add guard descent height reset button, add Zint version to window title,
static get_zint_version() method, use QStringLiteral (QSL shorthand),
use SIGNAL(toggled()), add errtxt "popup" and status bar, add icons,
add saveAs shortcut, add main menu, context menus and actions, add help,
reset_view() -> reset_colours(), add copy to clipboard as EMF/GIF/PNG/TIF,
lessen triggering of update_preview(), shorten names of getters/setters,
simplify/shorten some update_preview() logic in switch,
CODEONE disable structapp for Version S
qzint.cpp: add on_errored signal, add missing getters, add test
2021-10-09 00:13:39 +01:00
} else {
( void ) set_height ( symbol , 0.0f , 50.f , 0.0f , 1 /*no_errtxt*/ ) ;
2021-06-19 13:11:23 +01:00
}
2016-02-20 09:38:03 +00:00
if ( symbol - > symbology = = BARCODE_CODE39 ) {
2020-06-04 18:45:25 +01:00
ustrcpy ( symbol - > text , " * " ) ;
2020-12-23 10:57:24 +00:00
ustrncat ( symbol - > text , source , length ) ;
2020-06-04 18:45:25 +01:00
ustrcat ( symbol - > text , localstr ) ;
ustrcat ( symbol - > text , " * " ) ;
2016-02-20 09:38:03 +00:00
} else {
2020-06-04 18:45:25 +01:00
ustrcpy ( symbol - > text , source ) ;
ustrcat ( symbol - > text , localstr ) ;
2016-02-20 09:38:03 +00:00
}
return error_number ;
2008-07-13 21:15:55 +00:00
}
2022-12-09 14:51:26 +00:00
/* Pharmazentralnummer (PZN) */
2021-10-20 23:05:30 +01:00
/* PZN https://www.ifaffm.de/mandanten/1/documents/04_ifa_coding_system/IFA_Info_Code_39_EN.pdf */
/* PZN https://www.ifaffm.de/mandanten/1/documents/04_ifa_coding_system/
IFA - Info_Check_Digit_Calculations_PZN_PPN_UDI_EN . pdf */
Add compliant height, using ZINT_COMPLIANT_HEIGHT flag for back-compatibility
Rename barcode funcs to same as BARCODE_XXX name
library: barcode funcs array for dispatch, used for ZBarcode_ValidID() also
general: change is_sane() comparison to nonzero from ZINT_ERROR_INVALID_OPTION
MAILMARK: fuller error messages
CODABAR: add option to show check character in HRT
zint.h: use 0xNNNN for OR-able defines
GUI: add guard descent height reset button, add Zint version to window title,
static get_zint_version() method, use QStringLiteral (QSL shorthand),
use SIGNAL(toggled()), add errtxt "popup" and status bar, add icons,
add saveAs shortcut, add main menu, context menus and actions, add help,
reset_view() -> reset_colours(), add copy to clipboard as EMF/GIF/PNG/TIF,
lessen triggering of update_preview(), shorten names of getters/setters,
simplify/shorten some update_preview() logic in switch,
CODEONE disable structapp for Version S
qzint.cpp: add on_errored signal, add missing getters, add test
2021-10-09 00:13:39 +01:00
INTERNAL int pzn ( struct zint_symbol * symbol , unsigned char source [ ] , int length ) {
2016-02-20 09:38:03 +00:00
int i , error_number , zeroes ;
2022-12-09 14:51:26 +00:00
int count , check_digit ;
unsigned char have_check_digit = ' \0 ' ;
char localstr [ 1 + 8 + 1 ] ; /* '-' prefix + 8 digits + NUL */
const int pzn7 = symbol - > option_2 = = 1 ;
2016-02-20 09:38:03 +00:00
2022-12-09 14:51:26 +00:00
if ( length > 8 - pzn7 ) {
sprintf ( symbol - > errtxt , " 325: Input wrong length (%d character maximum) " , 8 - pzn7 ) ;
2016-02-20 09:38:03 +00:00
return ZINT_ERROR_TOO_LONG ;
}
2022-12-09 14:51:26 +00:00
if ( length = = 8 - pzn7 ) {
have_check_digit = source [ 7 - pzn7 ] ;
length - - ;
}
2021-10-20 23:05:30 +01:00
if ( ! is_sane ( NEON_F , source , length ) ) {
2021-07-06 19:53:31 +01:00
strcpy ( symbol - > errtxt , " 326: Invalid character in data (digits only) " ) ;
Add compliant height, using ZINT_COMPLIANT_HEIGHT flag for back-compatibility
Rename barcode funcs to same as BARCODE_XXX name
library: barcode funcs array for dispatch, used for ZBarcode_ValidID() also
general: change is_sane() comparison to nonzero from ZINT_ERROR_INVALID_OPTION
MAILMARK: fuller error messages
CODABAR: add option to show check character in HRT
zint.h: use 0xNNNN for OR-able defines
GUI: add guard descent height reset button, add Zint version to window title,
static get_zint_version() method, use QStringLiteral (QSL shorthand),
use SIGNAL(toggled()), add errtxt "popup" and status bar, add icons,
add saveAs shortcut, add main menu, context menus and actions, add help,
reset_view() -> reset_colours(), add copy to clipboard as EMF/GIF/PNG/TIF,
lessen triggering of update_preview(), shorten names of getters/setters,
simplify/shorten some update_preview() logic in switch,
CODEONE disable structapp for Version S
qzint.cpp: add on_errored signal, add missing getters, add test
2021-10-09 00:13:39 +01:00
return ZINT_ERROR_INVALID_DATA ;
2016-02-20 09:38:03 +00:00
}
localstr [ 0 ] = ' - ' ;
2022-12-09 14:51:26 +00:00
zeroes = 7 - pzn7 - length + 1 ;
2016-02-20 09:38:03 +00:00
for ( i = 1 ; i < zeroes ; i + + )
localstr [ i ] = ' 0 ' ;
2020-06-04 18:45:25 +01:00
ustrcpy ( localstr + zeroes , source ) ;
2016-02-20 09:38:03 +00:00
2020-06-04 18:45:25 +01:00
count = 0 ;
2022-12-09 14:51:26 +00:00
for ( i = 1 ; i < 8 - pzn7 ; i + + ) {
count + = ( i + pzn7 ) * ctoi ( localstr [ i ] ) ;
2016-02-20 09:38:03 +00:00
}
check_digit = count % 11 ;
2020-06-04 18:45:25 +01:00
if ( symbol - > debug & ZINT_DEBUG_PRINT ) {
2021-07-07 13:58:04 +01:00
printf ( " PZN: %s, check digit %d \n " , localstr , ( int ) check_digit ) ;
2016-02-20 09:38:03 +00:00
}
2020-06-04 18:45:25 +01:00
if ( check_digit = = 10 ) {
2021-06-27 11:47:55 +01:00
strcpy ( symbol - > errtxt , " 327: Invalid PZN, check digit is '10' " ) ;
2016-02-20 09:38:03 +00:00
return ZINT_ERROR_INVALID_DATA ;
}
2022-12-09 14:51:26 +00:00
if ( have_check_digit & & ctoi ( have_check_digit ) ! = check_digit ) {
sprintf ( symbol - > errtxt , " 890: Invalid check digit '%c', expecting '%c' " , have_check_digit , itoc ( check_digit ) ) ;
return ZINT_ERROR_INVALID_CHECK ;
}
localstr [ 8 - pzn7 ] = itoc ( check_digit ) ;
localstr [ 9 - pzn7 ] = ' \0 ' ;
if ( pzn7 ) {
symbol - > option_2 = 0 ; /* Need to overwrite this so `code39()` doesn't add a check digit itself */
}
error_number = code39 ( symbol , ( unsigned char * ) localstr , 9 - pzn7 ) ;
if ( pzn7 ) {
symbol - > option_2 = 1 ; /* Restore */
}
ustrcpy ( symbol - > text , " PZN - " ) ; /* Note changed to put space after hyphen */
ustrcat ( symbol - > text , localstr + 1 ) ;
2021-06-19 13:11:23 +01:00
Add compliant height, using ZINT_COMPLIANT_HEIGHT flag for back-compatibility
Rename barcode funcs to same as BARCODE_XXX name
library: barcode funcs array for dispatch, used for ZBarcode_ValidID() also
general: change is_sane() comparison to nonzero from ZINT_ERROR_INVALID_OPTION
MAILMARK: fuller error messages
CODABAR: add option to show check character in HRT
zint.h: use 0xNNNN for OR-able defines
GUI: add guard descent height reset button, add Zint version to window title,
static get_zint_version() method, use QStringLiteral (QSL shorthand),
use SIGNAL(toggled()), add errtxt "popup" and status bar, add icons,
add saveAs shortcut, add main menu, context menus and actions, add help,
reset_view() -> reset_colours(), add copy to clipboard as EMF/GIF/PNG/TIF,
lessen triggering of update_preview(), shorten names of getters/setters,
simplify/shorten some update_preview() logic in switch,
CODEONE disable structapp for Version S
qzint.cpp: add on_errored signal, add missing getters, add test
2021-10-09 00:13:39 +01:00
if ( symbol - > output_options & COMPLIANT_HEIGHT ) {
/* Technical Information regarding PZN Coding V 2.1 (25 Feb 2019) Code size
https : //www.ifaffm.de/mandanten/1/documents/04_ifa_coding_system/IFA_Info_Code_39_EN.pdf
" normal " X 0.25 mm ( 0.187 mm - 0.45 mm ) , height 8 mm - 20 mm for 0.25 mm X , 10 mm mentioned so use that
as default , 10 mm / 0.25 mm = 40 */
if ( error_number < ZINT_ERROR ) {
error_number = set_height ( symbol , stripf ( 8.0f / 0.45f ) , 40.0f , stripf ( 20.0f / 0.187f ) , 0 /*no_errtxt*/ ) ;
}
} else {
if ( error_number < ZINT_ERROR ) {
( void ) set_height ( symbol , 0.0f , 50.0f , 0.0f , 1 /*no_errtxt*/ ) ;
}
2021-06-19 13:11:23 +01:00
}
2016-02-20 09:38:03 +00:00
return error_number ;
2008-07-13 21:15:55 +00:00
}
2016-02-20 09:38:03 +00:00
/* Extended Code 39 - ISO/IEC 16388:2007 Annex A */
Add compliant height, using ZINT_COMPLIANT_HEIGHT flag for back-compatibility
Rename barcode funcs to same as BARCODE_XXX name
library: barcode funcs array for dispatch, used for ZBarcode_ValidID() also
general: change is_sane() comparison to nonzero from ZINT_ERROR_INVALID_OPTION
MAILMARK: fuller error messages
CODABAR: add option to show check character in HRT
zint.h: use 0xNNNN for OR-able defines
GUI: add guard descent height reset button, add Zint version to window title,
static get_zint_version() method, use QStringLiteral (QSL shorthand),
use SIGNAL(toggled()), add errtxt "popup" and status bar, add icons,
add saveAs shortcut, add main menu, context menus and actions, add help,
reset_view() -> reset_colours(), add copy to clipboard as EMF/GIF/PNG/TIF,
lessen triggering of update_preview(), shorten names of getters/setters,
simplify/shorten some update_preview() logic in switch,
CODEONE disable structapp for Version S
qzint.cpp: add on_errored signal, add missing getters, add test
2021-10-09 00:13:39 +01:00
INTERNAL int excode39 ( struct zint_symbol * symbol , unsigned char source [ ] , int length ) {
2008-07-13 21:15:55 +00:00
2020-06-04 18:45:25 +01:00
unsigned char buffer [ 85 * 2 + 1 ] = { 0 } ;
2021-10-20 23:05:30 +01:00
unsigned char * b = buffer ;
2020-06-04 18:45:25 +01:00
int i ;
2016-02-20 09:38:03 +00:00
int error_number ;
2008-07-13 21:15:55 +00:00
2020-06-04 18:45:25 +01:00
if ( length > 85 ) {
2021-06-27 11:47:55 +01:00
strcpy ( symbol - > errtxt , " 328: Input too long (85 character maximum) " ) ;
2016-02-20 09:38:03 +00:00
return ZINT_ERROR_TOO_LONG ;
}
2008-07-13 21:15:55 +00:00
2016-02-20 09:38:03 +00:00
/* Creates a buffer string and places control characters into it */
2020-06-04 18:45:25 +01:00
for ( i = 0 ; i < length ; i + + ) {
2016-02-20 09:38:03 +00:00
if ( source [ i ] > 127 ) {
/* Cannot encode extended ASCII */
2021-07-06 19:53:31 +01:00
strcpy ( symbol - > errtxt , " 329: Invalid character in data, extended ASCII not allowed " ) ;
2016-02-20 09:38:03 +00:00
return ZINT_ERROR_INVALID_DATA ;
}
2021-10-20 23:05:30 +01:00
memcpy ( b , EC39Ctrl [ source [ i ] ] , 2 ) ;
b + = EC39Ctrl [ source [ i ] ] [ 1 ] ? 2 : 1 ;
2016-02-20 09:38:03 +00:00
}
2021-10-20 23:05:30 +01:00
if ( b - buffer > 85 ) {
strcpy ( symbol - > errtxt , " 317: Expanded input too long (85 symbol character maximum) " ) ;
return ZINT_ERROR_TOO_LONG ;
}
* b = ' \0 ' ;
2012-12-29 19:37:03 +01:00
2016-02-20 09:38:03 +00:00
/* Then sends the buffer to the C39 function */
2021-10-20 23:05:30 +01:00
error_number = code39 ( symbol , buffer , b - buffer ) ;
2008-07-13 21:15:55 +00:00
2020-06-04 18:45:25 +01:00
for ( i = 0 ; i < length ; i + + )
2020-07-19 00:13:03 +01:00
symbol - > text [ i ] = source [ i ] > = ' ' & & source [ i ] ! = 0x7F ? source [ i ] : ' ' ;
Add compliant height, using ZINT_COMPLIANT_HEIGHT flag for back-compatibility
Rename barcode funcs to same as BARCODE_XXX name
library: barcode funcs array for dispatch, used for ZBarcode_ValidID() also
general: change is_sane() comparison to nonzero from ZINT_ERROR_INVALID_OPTION
MAILMARK: fuller error messages
CODABAR: add option to show check character in HRT
zint.h: use 0xNNNN for OR-able defines
GUI: add guard descent height reset button, add Zint version to window title,
static get_zint_version() method, use QStringLiteral (QSL shorthand),
use SIGNAL(toggled()), add errtxt "popup" and status bar, add icons,
add saveAs shortcut, add main menu, context menus and actions, add help,
reset_view() -> reset_colours(), add copy to clipboard as EMF/GIF/PNG/TIF,
lessen triggering of update_preview(), shorten names of getters/setters,
simplify/shorten some update_preview() logic in switch,
CODEONE disable structapp for Version S
qzint.cpp: add on_errored signal, add missing getters, add test
2021-10-09 00:13:39 +01:00
symbol - > text [ length ] = ' \0 ' ; /* Chops off check digit */
2012-12-29 19:37:03 +01:00
2016-02-20 09:38:03 +00:00
return error_number ;
2008-07-13 21:15:55 +00:00
}
2016-02-20 09:38:03 +00:00
/* Code 93 is an advancement on Code 39 and the definition is a lot tighter */
Add compliant height, using ZINT_COMPLIANT_HEIGHT flag for back-compatibility
Rename barcode funcs to same as BARCODE_XXX name
library: barcode funcs array for dispatch, used for ZBarcode_ValidID() also
general: change is_sane() comparison to nonzero from ZINT_ERROR_INVALID_OPTION
MAILMARK: fuller error messages
CODABAR: add option to show check character in HRT
zint.h: use 0xNNNN for OR-able defines
GUI: add guard descent height reset button, add Zint version to window title,
static get_zint_version() method, use QStringLiteral (QSL shorthand),
use SIGNAL(toggled()), add errtxt "popup" and status bar, add icons,
add saveAs shortcut, add main menu, context menus and actions, add help,
reset_view() -> reset_colours(), add copy to clipboard as EMF/GIF/PNG/TIF,
lessen triggering of update_preview(), shorten names of getters/setters,
simplify/shorten some update_preview() logic in switch,
CODEONE disable structapp for Version S
qzint.cpp: add on_errored signal, add missing getters, add test
2021-10-09 00:13:39 +01:00
INTERNAL int code93 ( struct zint_symbol * symbol , unsigned char source [ ] , int length ) {
2008-07-13 21:15:55 +00:00
2016-02-20 09:38:03 +00:00
/* SILVER includes the extra characters a, b, c and d to represent Code 93 specific
shift characters 1 , 2 , 3 and 4 respectively . These characters are never used by
Add compliant height, using ZINT_COMPLIANT_HEIGHT flag for back-compatibility
Rename barcode funcs to same as BARCODE_XXX name
library: barcode funcs array for dispatch, used for ZBarcode_ValidID() also
general: change is_sane() comparison to nonzero from ZINT_ERROR_INVALID_OPTION
MAILMARK: fuller error messages
CODABAR: add option to show check character in HRT
zint.h: use 0xNNNN for OR-able defines
GUI: add guard descent height reset button, add Zint version to window title,
static get_zint_version() method, use QStringLiteral (QSL shorthand),
use SIGNAL(toggled()), add errtxt "popup" and status bar, add icons,
add saveAs shortcut, add main menu, context menus and actions, add help,
reset_view() -> reset_colours(), add copy to clipboard as EMF/GIF/PNG/TIF,
lessen triggering of update_preview(), shorten names of getters/setters,
simplify/shorten some update_preview() logic in switch,
CODEONE disable structapp for Version S
qzint.cpp: add on_errored signal, add missing getters, add test
2021-10-09 00:13:39 +01:00
` code39 ( ) ` and ` excode39 ( ) ` */
2008-07-13 21:15:55 +00:00
2016-02-20 09:38:03 +00:00
int i ;
2021-08-28 20:17:26 +01:00
int h , weight , c , k , error_number = 0 ;
2021-10-20 23:05:30 +01:00
int values [ 110 ] ; /* 107 + 2 (Checks) */
2021-08-28 20:17:26 +01:00
char buffer [ 216 ] ; /* 107*2 (107 full ASCII) + 1 = 215 */
2021-10-20 23:05:30 +01:00
char * b = buffer ;
2021-08-28 20:17:26 +01:00
char dest [ 668 ] ; /* 6 (Start) + 107*6 + 2*6 (Checks) + 7 (Stop) + 1 (NUL) = 668 */
2021-10-20 23:05:30 +01:00
char * d = dest ;
2012-12-29 19:37:03 +01:00
2021-08-28 20:17:26 +01:00
/* Suppresses clang-tidy clang-analyzer-core.CallAndMessage warning */
assert ( length > 0 ) ;
2012-12-29 19:37:03 +01:00
2021-08-28 20:17:26 +01:00
if ( length > 107 ) { /* 9 (Start) + 107*9 + 2*9 (Checks) + 10 (Stop) == 1000 */
2021-06-27 11:47:55 +01:00
strcpy ( symbol - > errtxt , " 330: Input too long (107 character maximum) " ) ;
2016-02-20 09:38:03 +00:00
return ZINT_ERROR_TOO_LONG ;
}
/* Message Content */
for ( i = 0 ; i < length ; i + + ) {
if ( source [ i ] > 127 ) {
/* Cannot encode extended ASCII */
2021-07-06 19:53:31 +01:00
strcpy ( symbol - > errtxt , " 331: Invalid character in data, extended ASCII not allowed " ) ;
2016-02-20 09:38:03 +00:00
return ZINT_ERROR_INVALID_DATA ;
}
2021-10-20 23:05:30 +01:00
memcpy ( b , C93Ctrl [ source [ i ] ] , 2 ) ;
b + = C93Ctrl [ source [ i ] ] [ 1 ] ? 2 : 1 ;
2020-07-19 00:13:03 +01:00
symbol - > text [ i ] = source [ i ] > = ' ' & & source [ i ] ! = 0x7F ? source [ i ] : ' ' ;
2016-02-20 09:38:03 +00:00
}
/* Now we can check the true length of the barcode */
2021-10-20 23:05:30 +01:00
h = b - buffer ;
2016-02-20 09:38:03 +00:00
if ( h > 107 ) {
2021-10-20 23:05:30 +01:00
strcpy ( symbol - > errtxt , " 332: Expanded input too long (107 symbol character maximum) " ) ;
2016-02-20 09:38:03 +00:00
return ZINT_ERROR_TOO_LONG ;
}
for ( i = 0 ; i < h ; i + + ) {
values [ i ] = posn ( SILVER , buffer [ i ] ) ;
}
/* Putting the data into dest[] is not done until after check digits are calculated */
/* Check digit C */
c = 0 ;
weight = 1 ;
for ( i = h - 1 ; i > = 0 ; i - - ) {
c + = values [ i ] * weight ;
weight + + ;
if ( weight = = 21 )
weight = 1 ;
}
c = c % 47 ;
values [ h ] = c ;
/* Check digit K */
k = 0 ;
weight = 1 ;
for ( i = h ; i > = 0 ; i - - ) {
k + = values [ i ] * weight ;
weight + + ;
if ( weight = = 16 )
weight = 1 ;
}
k = k % 47 ;
2021-10-20 23:05:30 +01:00
values [ h + 1 ] = k ;
h + = 2 ;
if ( symbol - > debug & ZINT_DEBUG_PRINT ) {
printf ( " Check digit c: %d, k: %d \n " , c , k ) ;
}
2016-02-20 09:38:03 +00:00
/* Start character */
2021-10-20 23:05:30 +01:00
memcpy ( d , " 111141 " , 6 ) ;
d + = 6 ;
2016-02-20 09:38:03 +00:00
2021-10-20 23:05:30 +01:00
for ( i = 0 ; i < h ; i + + , d + = 6 ) {
memcpy ( d , C93Table [ values [ i ] ] , 6 ) ;
2016-02-20 09:38:03 +00:00
}
/* Stop character */
2021-10-20 23:05:30 +01:00
memcpy ( d , " 1111411 " , 7 ) ;
d + = 7 ;
expand ( symbol , dest , d - dest ) ;
2016-02-20 09:38:03 +00:00
Add compliant height, using ZINT_COMPLIANT_HEIGHT flag for back-compatibility
Rename barcode funcs to same as BARCODE_XXX name
library: barcode funcs array for dispatch, used for ZBarcode_ValidID() also
general: change is_sane() comparison to nonzero from ZINT_ERROR_INVALID_OPTION
MAILMARK: fuller error messages
CODABAR: add option to show check character in HRT
zint.h: use 0xNNNN for OR-able defines
GUI: add guard descent height reset button, add Zint version to window title,
static get_zint_version() method, use QStringLiteral (QSL shorthand),
use SIGNAL(toggled()), add errtxt "popup" and status bar, add icons,
add saveAs shortcut, add main menu, context menus and actions, add help,
reset_view() -> reset_colours(), add copy to clipboard as EMF/GIF/PNG/TIF,
lessen triggering of update_preview(), shorten names of getters/setters,
simplify/shorten some update_preview() logic in switch,
CODEONE disable structapp for Version S
qzint.cpp: add on_errored signal, add missing getters, add test
2021-10-09 00:13:39 +01:00
if ( symbol - > output_options & COMPLIANT_HEIGHT ) {
/* ANSI/AIM BC5-1995 Section 2.6 minimum height 0.2" or 15% of symbol length, whichever is greater
no max X given so for min height use symbol length = ( 9 * ( C + 4 ) + 1 ) * X + 2 * Q = symbol - > width + 20 ;
use 40 as default height based on figures in spec */
float min_height = stripf ( ( symbol - > width + 20 ) * 0.15f ) ;
error_number = set_height ( symbol , min_height , min_height > 40.0f ? min_height : 40.0f , 0.0f , 0 /*no_errtxt*/ ) ;
} else {
( void ) set_height ( symbol , 0.0f , 50.0f , 0.0f , 1 /*no_errtxt*/ ) ;
2021-06-19 13:11:23 +01:00
}
2021-09-13 18:49:25 +01:00
if ( symbol - > option_2 = = 1 ) {
2021-10-20 23:05:30 +01:00
symbol - > text [ length ] = SILVER [ c ] ;
symbol - > text [ length + 1 ] = SILVER [ k ] ;
2021-09-13 18:49:25 +01:00
symbol - > text [ length + 2 ] = ' \0 ' ;
}
2016-02-20 09:38:03 +00:00
return error_number ;
2008-07-13 21:15:55 +00:00
}
2009-05-16 14:19:43 +00:00
2020-06-10 18:41:36 +01:00
typedef const struct s_channel_precalc {
long value ; unsigned char B [ 8 ] ; unsigned char S [ 8 ] ; unsigned char bmax [ 7 ] ; unsigned char smax [ 7 ] ;
} channel_precalc ;
2016-02-20 09:38:03 +00:00
2022-11-06 22:25:19 +00:00
#if 0
# define CHANNEL_GENERATE_PRECALCS
# endif
2009-05-16 14:19:43 +00:00
2020-06-10 18:41:36 +01:00
# ifdef CHANNEL_GENERATE_PRECALCS
Add Structured Append support for AZTEC, CODEONE, DATAMATRIX, DOTCODE,
GRIDMATRIX, MAXICODE, MICROPDF417, PDF417, QRCODE, ULTRA
DOTCODE: use pre-calculated generator poly coeffs in Reed-Solomon for
performance improvement
PDF417/MICROPDF417: use common routine pdf417_initial()
GUI: code lines <= 118, shorthand widget_obj(),
shorten calling upcean_addon_gap(), upcean_guard_descent()
various backend: var name debug -> debug_print
2021-09-28 21:42:44 +01:00
/* To generate precalc tables uncomment CHANNEL_GENERATE_PRECALCS define and run
" backend/tests/test_channel -f generate -g " and place result in " channel_precalcs.h " */
2021-06-19 13:11:23 +01:00
static void channel_generate_precalc ( int channels , long value , int mod , int last , int B [ 8 ] , int S [ 8 ] , int bmax [ 7 ] ,
int smax [ 7 ] ) {
2020-06-10 18:41:36 +01:00
int i ;
if ( value = = mod ) printf ( " static channel_precalc channel_precalcs%d[] = { \n " , channels ) ;
printf ( " { %7ld, { " , value ) ; for ( i = 0 ; i < 8 ; i + + ) printf ( " %d, " , B [ i ] ) ; printf ( " }, " ) ;
printf ( " { " ) ; for ( i = 0 ; i < 8 ; i + + ) printf ( " %d, " , S [ i ] ) ; printf ( " }, " ) ;
printf ( " { " ) ; for ( i = 0 ; i < 7 ; i + + ) printf ( " %d, " , bmax [ i ] ) ; printf ( " }, " ) ;
printf ( " { " ) ; for ( i = 0 ; i < 7 ; i + + ) printf ( " %d, " , smax [ i ] ) ; printf ( " }, }, \n " ) ;
if ( value = = last ) printf ( " }; \n " ) ;
2009-05-16 14:19:43 +00:00
}
2020-06-10 18:41:36 +01:00
# else
# include "channel_precalcs.h"
# endif
2009-05-16 14:19:43 +00:00
2020-06-10 18:41:36 +01:00
static long channel_copy_precalc ( channel_precalc precalc , int B [ 8 ] , int S [ 8 ] , int bmax [ 7 ] , int smax [ 7 ] ) {
int i ;
2016-02-20 09:38:03 +00:00
2020-06-10 18:41:36 +01:00
for ( i = 0 ; i < 7 ; i + + ) {
B [ i ] = precalc . B [ i ] ;
S [ i ] = precalc . S [ i ] ;
bmax [ i ] = precalc . bmax [ i ] ;
smax [ i ] = precalc . smax [ i ] ;
2016-02-20 09:38:03 +00:00
}
2020-06-10 18:41:36 +01:00
B [ 7 ] = precalc . B [ 7 ] ;
S [ 7 ] = precalc . S [ 7 ] ;
return precalc . value ;
2009-05-16 14:19:43 +00:00
}
2020-06-10 18:41:36 +01:00
/* CHNCHR is adapted from ANSI/AIM BC12-1998 Annex D Figure D5 and is Copyright (c) AIM 1997 */
2012-12-29 19:37:03 +01:00
2020-06-10 18:41:36 +01:00
/* It is used here on the understanding that it forms part of the specification
for Channel Code and therefore its use is permitted under the following terms
set out in that document :
" It is the intent and understanding of AIM [t]hat the symbology presented in this
specification is entirely in the public domain and free of all use restrictions ,
licenses and fees . AIM USA , its member companies , or individual officers
assume no liability for the use of this document . " */
static void CHNCHR ( int channels , long target_value , int B [ 8 ] , int S [ 8 ] ) {
2021-06-19 13:11:23 +01:00
/* Use of initial pre-calculations taken from Barcode Writer in Pure PostScript (BWIPP)
2020-06-10 18:41:36 +01:00
* Copyright ( c ) 2004 - 2020 Terry Burton ( MIT / X - Consortium license ) */
static channel_precalc initial_precalcs [ 6 ] = {
Add compliant height, using ZINT_COMPLIANT_HEIGHT flag for back-compatibility
Rename barcode funcs to same as BARCODE_XXX name
library: barcode funcs array for dispatch, used for ZBarcode_ValidID() also
general: change is_sane() comparison to nonzero from ZINT_ERROR_INVALID_OPTION
MAILMARK: fuller error messages
CODABAR: add option to show check character in HRT
zint.h: use 0xNNNN for OR-able defines
GUI: add guard descent height reset button, add Zint version to window title,
static get_zint_version() method, use QStringLiteral (QSL shorthand),
use SIGNAL(toggled()), add errtxt "popup" and status bar, add icons,
add saveAs shortcut, add main menu, context menus and actions, add help,
reset_view() -> reset_colours(), add copy to clipboard as EMF/GIF/PNG/TIF,
lessen triggering of update_preview(), shorten names of getters/setters,
simplify/shorten some update_preview() logic in switch,
CODEONE disable structapp for Version S
qzint.cpp: add on_errored signal, add missing getters, add test
2021-10-09 00:13:39 +01:00
{ 0 , { 1 , 1 , 1 , 1 , 1 , 2 , 1 , 2 , } , { 1 , 1 , 1 , 1 , 1 , 1 , 1 , 3 , } , { 1 , 1 , 1 , 1 , 1 , 3 , 2 , } ,
{ 1 , 1 , 1 , 1 , 1 , 3 , 3 , } , } ,
{ 0 , { 1 , 1 , 1 , 1 , 2 , 1 , 1 , 3 , } , { 1 , 1 , 1 , 1 , 1 , 1 , 1 , 4 , } , { 1 , 1 , 1 , 1 , 4 , 3 , 3 , } ,
{ 1 , 1 , 1 , 1 , 4 , 4 , 4 , } , } ,
{ 0 , { 1 , 1 , 1 , 2 , 1 , 1 , 2 , 3 , } , { 1 , 1 , 1 , 1 , 1 , 1 , 1 , 5 , } , { 1 , 1 , 1 , 5 , 4 , 4 , 4 , } ,
{ 1 , 1 , 1 , 5 , 5 , 5 , 5 , } , } ,
{ 0 , { 1 , 1 , 2 , 1 , 1 , 2 , 1 , 4 , } , { 1 , 1 , 1 , 1 , 1 , 1 , 1 , 6 , } , { 1 , 1 , 6 , 5 , 5 , 5 , 4 , } ,
{ 1 , 1 , 6 , 6 , 6 , 6 , 6 , } , } ,
{ 0 , { 1 , 2 , 1 , 1 , 2 , 1 , 1 , 5 , } , { 1 , 1 , 1 , 1 , 1 , 1 , 1 , 7 , } , { 1 , 7 , 6 , 6 , 6 , 5 , 5 , } ,
{ 1 , 7 , 7 , 7 , 7 , 7 , 7 , } , } ,
{ 0 , { 2 , 1 , 1 , 2 , 1 , 1 , 2 , 5 , } , { 1 , 1 , 1 , 1 , 1 , 1 , 1 , 8 , } , { 8 , 7 , 7 , 7 , 6 , 6 , 6 , } ,
{ 8 , 8 , 8 , 8 , 8 , 8 , 8 , } , } ,
2020-06-10 18:41:36 +01:00
} ;
int bmax [ 7 ] , smax [ 7 ] ;
long value = 0 ;
channel_copy_precalc ( initial_precalcs [ channels - 3 ] , B , S , bmax , smax ) ;
# ifndef CHANNEL_GENERATE_PRECALCS
if ( channels = = 7 & & target_value > = channel_precalcs7 [ 0 ] . value ) {
2021-06-19 13:11:23 +01:00
value = channel_copy_precalc ( channel_precalcs7 [ ( target_value / channel_precalcs7 [ 0 ] . value ) - 1 ] , B , S , bmax ,
smax ) ;
2020-06-10 18:41:36 +01:00
} else if ( channels = = 8 & & target_value > = channel_precalcs8 [ 0 ] . value ) {
2021-06-19 13:11:23 +01:00
value = channel_copy_precalc ( channel_precalcs8 [ ( target_value / channel_precalcs8 [ 0 ] . value ) - 1 ] , B , S , bmax ,
smax ) ;
2020-06-10 18:41:36 +01:00
}
# endif
goto chkchr ;
ls0 : smax [ 1 ] = smax [ 0 ] + 1 - S [ 0 ] ; B [ 0 ] = 1 ;
if ( S [ 0 ] = = 1 ) goto nb0 ;
lb0 : bmax [ 1 ] = bmax [ 0 ] + 1 - B [ 0 ] ; S [ 1 ] = 1 ;
ls1 : smax [ 2 ] = smax [ 1 ] + 1 - S [ 1 ] ; B [ 1 ] = 1 ;
if ( S [ 0 ] + B [ 0 ] + S [ 1 ] = = 3 ) goto nb1 ;
lb1 : bmax [ 2 ] = bmax [ 1 ] + 1 - B [ 1 ] ; S [ 2 ] = 1 ;
ls2 : smax [ 3 ] = smax [ 2 ] + 1 - S [ 2 ] ; B [ 2 ] = 1 ;
if ( B [ 0 ] + S [ 1 ] + B [ 1 ] + S [ 2 ] = = 4 ) goto nb2 ;
lb2 : bmax [ 3 ] = bmax [ 2 ] + 1 - B [ 2 ] ; S [ 3 ] = 1 ;
ls3 : smax [ 4 ] = smax [ 3 ] + 1 - S [ 3 ] ; B [ 3 ] = 1 ;
if ( B [ 1 ] + S [ 2 ] + B [ 2 ] + S [ 3 ] = = 4 ) goto nb3 ;
lb3 : bmax [ 4 ] = bmax [ 3 ] + 1 - B [ 3 ] ; S [ 4 ] = 1 ;
ls4 : smax [ 5 ] = smax [ 4 ] + 1 - S [ 4 ] ; B [ 4 ] = 1 ;
if ( B [ 2 ] + S [ 3 ] + B [ 3 ] + S [ 4 ] = = 4 ) goto nb4 ;
lb4 : bmax [ 5 ] = bmax [ 4 ] + 1 - B [ 4 ] ; S [ 5 ] = 1 ;
ls5 : smax [ 6 ] = smax [ 5 ] + 1 - S [ 5 ] ; B [ 5 ] = 1 ;
if ( B [ 3 ] + S [ 4 ] + B [ 4 ] + S [ 5 ] = = 4 ) goto nb5 ;
lb5 : bmax [ 6 ] = bmax [ 5 ] + 1 - B [ 5 ] ; S [ 6 ] = 1 ;
ls6 : S [ 7 ] = smax [ 6 ] + 1 - S [ 6 ] ; B [ 6 ] = 1 ;
if ( B [ 4 ] + S [ 5 ] + B [ 5 ] + S [ 6 ] = = 4 ) goto nb6 ;
lb6 : B [ 7 ] = bmax [ 6 ] + 1 - B [ 6 ] ;
if ( B [ 5 ] + S [ 6 ] + B [ 6 ] + S [ 7 ] + B [ 7 ] = = 5 ) goto nb6 ;
chkchr :
# ifdef CHANNEL_GENERATE_PRECALCS
2021-06-19 13:11:23 +01:00
/* 115338 == (576688 + 2) / 5 */
if ( channels = = 7 & & value & & value % 115338 = = 0 ) {
channel_generate_precalc ( channels , value , 115338 ,
115338 * ( 5 - 1 ) , B , S , bmax , smax ) ;
/* 119121 == (7742862 + 3) / 65 */
} else if ( channels = = 8 & & value & & value % 119121 = = 0 ) {
channel_generate_precalc ( channels , value , 119121 ,
119121 * ( 65 - 1 ) , B , S , bmax , smax ) ;
2020-06-10 18:41:36 +01:00
}
# endif
if ( value = = target_value ) return ;
value + + ;
nb6 : if ( + + B [ 6 ] < = bmax [ 6 ] ) goto lb6 ;
if ( + + S [ 6 ] < = smax [ 6 ] ) goto ls6 ;
nb5 : if ( + + B [ 5 ] < = bmax [ 5 ] ) goto lb5 ;
if ( + + S [ 5 ] < = smax [ 5 ] ) goto ls5 ;
nb4 : if ( + + B [ 4 ] < = bmax [ 4 ] ) goto lb4 ;
if ( + + S [ 4 ] < = smax [ 4 ] ) goto ls4 ;
nb3 : if ( + + B [ 3 ] < = bmax [ 3 ] ) goto lb3 ;
if ( + + S [ 3 ] < = smax [ 3 ] ) goto ls3 ;
nb2 : if ( + + B [ 2 ] < = bmax [ 2 ] ) goto lb2 ;
if ( + + S [ 2 ] < = smax [ 2 ] ) goto ls2 ;
nb1 : if ( + + B [ 1 ] < = bmax [ 1 ] ) goto lb1 ;
if ( + + S [ 1 ] < = smax [ 1 ] ) goto ls1 ;
nb0 : if ( + + B [ 0 ] < = bmax [ 0 ] ) goto lb0 ;
if ( + + S [ 0 ] < = smax [ 0 ] ) goto ls0 ;
2009-05-16 14:19:43 +00:00
}
2016-02-20 09:38:03 +00:00
/* Channel Code - According to ANSI/AIM BC12-1998 */
Add compliant height, using ZINT_COMPLIANT_HEIGHT flag for back-compatibility
Rename barcode funcs to same as BARCODE_XXX name
library: barcode funcs array for dispatch, used for ZBarcode_ValidID() also
general: change is_sane() comparison to nonzero from ZINT_ERROR_INVALID_OPTION
MAILMARK: fuller error messages
CODABAR: add option to show check character in HRT
zint.h: use 0xNNNN for OR-able defines
GUI: add guard descent height reset button, add Zint version to window title,
static get_zint_version() method, use QStringLiteral (QSL shorthand),
use SIGNAL(toggled()), add errtxt "popup" and status bar, add icons,
add saveAs shortcut, add main menu, context menus and actions, add help,
reset_view() -> reset_colours(), add copy to clipboard as EMF/GIF/PNG/TIF,
lessen triggering of update_preview(), shorten names of getters/setters,
simplify/shorten some update_preview() logic in switch,
CODEONE disable structapp for Version S
qzint.cpp: add on_errored signal, add missing getters, add test
2021-10-09 00:13:39 +01:00
INTERNAL int channel ( struct zint_symbol * symbol , unsigned char source [ ] , int length ) {
2021-10-30 22:00:31 +01:00
static const int max_ranges [ ] = { - 1 , - 1 , - 1 , 26 , 292 , 3493 , 44072 , 576688 , 7742862 } ;
2020-06-10 18:41:36 +01:00
int S [ 8 ] = { 0 } , B [ 8 ] = { 0 } ;
long target_value = 0 ;
2021-10-20 23:05:30 +01:00
char dest [ 30 ] ;
char * d = dest ;
2016-02-20 09:38:03 +00:00
int channels , i ;
Add compliant height, using ZINT_COMPLIANT_HEIGHT flag for back-compatibility
Rename barcode funcs to same as BARCODE_XXX name
library: barcode funcs array for dispatch, used for ZBarcode_ValidID() also
general: change is_sane() comparison to nonzero from ZINT_ERROR_INVALID_OPTION
MAILMARK: fuller error messages
CODABAR: add option to show check character in HRT
zint.h: use 0xNNNN for OR-able defines
GUI: add guard descent height reset button, add Zint version to window title,
static get_zint_version() method, use QStringLiteral (QSL shorthand),
use SIGNAL(toggled()), add errtxt "popup" and status bar, add icons,
add saveAs shortcut, add main menu, context menus and actions, add help,
reset_view() -> reset_colours(), add copy to clipboard as EMF/GIF/PNG/TIF,
lessen triggering of update_preview(), shorten names of getters/setters,
simplify/shorten some update_preview() logic in switch,
CODEONE disable structapp for Version S
qzint.cpp: add on_errored signal, add missing getters, add test
2021-10-09 00:13:39 +01:00
int error_number = 0 , zeroes ;
2016-02-20 09:38:03 +00:00
if ( length > 7 ) {
2021-06-27 11:47:55 +01:00
strcpy ( symbol - > errtxt , " 333: Input too long (7 character maximum) " ) ;
2016-02-20 09:38:03 +00:00
return ZINT_ERROR_TOO_LONG ;
}
2021-10-20 23:05:30 +01:00
if ( ! is_sane ( NEON_F , source , length ) ) {
2021-07-06 19:53:31 +01:00
strcpy ( symbol - > errtxt , " 334: Invalid character in data (digits only) " ) ;
Add compliant height, using ZINT_COMPLIANT_HEIGHT flag for back-compatibility
Rename barcode funcs to same as BARCODE_XXX name
library: barcode funcs array for dispatch, used for ZBarcode_ValidID() also
general: change is_sane() comparison to nonzero from ZINT_ERROR_INVALID_OPTION
MAILMARK: fuller error messages
CODABAR: add option to show check character in HRT
zint.h: use 0xNNNN for OR-able defines
GUI: add guard descent height reset button, add Zint version to window title,
static get_zint_version() method, use QStringLiteral (QSL shorthand),
use SIGNAL(toggled()), add errtxt "popup" and status bar, add icons,
add saveAs shortcut, add main menu, context menus and actions, add help,
reset_view() -> reset_colours(), add copy to clipboard as EMF/GIF/PNG/TIF,
lessen triggering of update_preview(), shorten names of getters/setters,
simplify/shorten some update_preview() logic in switch,
CODEONE disable structapp for Version S
qzint.cpp: add on_errored signal, add missing getters, add test
2021-10-09 00:13:39 +01:00
return ZINT_ERROR_INVALID_DATA ;
2016-02-20 09:38:03 +00:00
}
if ( ( symbol - > option_2 < 3 ) | | ( symbol - > option_2 > 8 ) ) {
channels = 0 ;
} else {
channels = symbol - > option_2 ;
}
2020-06-10 18:41:36 +01:00
for ( i = 0 ; i < length ; i + + ) {
target_value * = 10 ;
target_value + = ctoi ( ( char ) source [ i ] ) ;
}
2016-02-20 09:38:03 +00:00
if ( channels = = 0 ) {
channels = length + 1 ;
2020-06-10 18:41:36 +01:00
if ( target_value > 576688 & & channels < 8 ) {
channels = 8 ;
} else if ( target_value > 44072 & & channels < 7 ) {
channels = 7 ;
} else if ( target_value > 3493 & & channels < 6 ) {
channels = 6 ;
} else if ( target_value > 292 & & channels < 5 ) {
channels = 5 ;
} else if ( target_value > 26 & & channels < 4 ) {
channels = 4 ;
}
2016-02-20 09:38:03 +00:00
}
if ( channels = = 2 ) {
channels = 3 ;
}
2021-06-27 11:47:55 +01:00
if ( target_value > max_ranges [ channels ] ) {
2021-07-06 19:53:31 +01:00
if ( channels = = 8 ) {
Add Structured Append support for AZTEC, CODEONE, DATAMATRIX, DOTCODE,
GRIDMATRIX, MAXICODE, MICROPDF417, PDF417, QRCODE, ULTRA
DOTCODE: use pre-calculated generator poly coeffs in Reed-Solomon for
performance improvement
PDF417/MICROPDF417: use common routine pdf417_initial()
GUI: code lines <= 118, shorthand widget_obj(),
shorten calling upcean_addon_gap(), upcean_guard_descent()
various backend: var name debug -> debug_print
2021-09-28 21:42:44 +01:00
sprintf ( symbol - > errtxt , " 318: Value out of range (0 to %d) " , max_ranges [ channels ] ) ;
2021-07-06 19:53:31 +01:00
} else {
2021-07-26 15:29:05 +01:00
sprintf ( symbol - > errtxt , " 335: Value out of range (0 to %d) for %d channels " ,
max_ranges [ channels ] , channels ) ;
2021-07-06 19:53:31 +01:00
}
2016-02-20 09:38:03 +00:00
return ZINT_ERROR_INVALID_DATA ;
}
2020-06-10 18:41:36 +01:00
CHNCHR ( channels , target_value , B , S ) ;
2016-02-20 09:38:03 +00:00
2021-10-20 23:05:30 +01:00
memcpy ( d , " 111111111 " , 9 ) ; /* Finder pattern */
d + = 9 ;
2020-06-10 18:41:36 +01:00
for ( i = 8 - channels ; i < 8 ; i + + ) {
2021-10-20 23:05:30 +01:00
* d + + = itoc ( S [ i ] ) ;
* d + + = itoc ( B [ i ] ) ;
2020-06-10 18:41:36 +01:00
}
2016-02-20 09:38:03 +00:00
zeroes = channels - 1 - length ;
2019-09-01 20:23:15 +01:00
if ( zeroes < 0 ) {
zeroes = 0 ;
}
2021-10-20 23:05:30 +01:00
if ( zeroes ) {
memset ( symbol - > text , ' 0 ' , zeroes ) ;
}
ustrcpy ( symbol - > text + zeroes , source ) ;
2016-02-20 09:38:03 +00:00
2021-10-20 23:05:30 +01:00
expand ( symbol , dest , d - dest ) ;
2016-02-20 09:38:03 +00:00
Add compliant height, using ZINT_COMPLIANT_HEIGHT flag for back-compatibility
Rename barcode funcs to same as BARCODE_XXX name
library: barcode funcs array for dispatch, used for ZBarcode_ValidID() also
general: change is_sane() comparison to nonzero from ZINT_ERROR_INVALID_OPTION
MAILMARK: fuller error messages
CODABAR: add option to show check character in HRT
zint.h: use 0xNNNN for OR-able defines
GUI: add guard descent height reset button, add Zint version to window title,
static get_zint_version() method, use QStringLiteral (QSL shorthand),
use SIGNAL(toggled()), add errtxt "popup" and status bar, add icons,
add saveAs shortcut, add main menu, context menus and actions, add help,
reset_view() -> reset_colours(), add copy to clipboard as EMF/GIF/PNG/TIF,
lessen triggering of update_preview(), shorten names of getters/setters,
simplify/shorten some update_preview() logic in switch,
CODEONE disable structapp for Version S
qzint.cpp: add on_errored signal, add missing getters, add test
2021-10-09 00:13:39 +01:00
if ( symbol - > output_options & COMPLIANT_HEIGHT ) {
/* ANSI/AIM BC12-1998 gives min height as 5mm or 15% of length; X left as application specification so use
length = 1 X ( left qz ) + ( 9 ( finder ) + 4 * 8 - 2 ) * X + 2 X ( right qz ) ;
use 20 as default based on figures in spec */
const float min_height = stripf ( ( 1 + 9 + 4 * channels - 2 + 2 ) * 0.15f ) ;
error_number = set_height ( symbol , min_height , 20.0f , 0.0f , 0 /*no_errtxt*/ ) ;
} else {
( void ) set_height ( symbol , 0.0f , 50.0f , 0.0f , 1 /*no_errtxt*/ ) ;
}
2021-06-19 13:11:23 +01:00
2016-02-20 09:38:03 +00:00
return error_number ;
2009-08-06 18:48:42 +00:00
}
2017-10-23 21:34:31 +02:00
2018-02-03 18:44:01 +00:00
/* Vehicle Identification Number (VIN) */
2020-12-23 10:57:24 +00:00
INTERNAL int vin ( struct zint_symbol * symbol , unsigned char source [ ] , int length ) {
2020-06-04 18:45:25 +01:00
2018-02-03 18:44:01 +00:00
/* This code verifies the check digit present in North American VIN codes */
2020-06-04 18:45:25 +01:00
2020-12-23 10:57:24 +00:00
char dest [ 200 ] ; /* 10 + 10 + 17 * 10 + 9 + 1 = 200 */
2021-10-20 23:05:30 +01:00
char * d = dest ;
2018-02-03 18:44:01 +00:00
char input_check ;
char output_check ;
int sum ;
int i ;
2021-10-30 22:00:31 +01:00
static const int weight [ 17 ] = { 8 , 7 , 6 , 5 , 4 , 3 , 2 , 10 , 0 , 9 , 8 , 7 , 6 , 5 , 4 , 3 , 2 } ;
2020-06-04 18:45:25 +01:00
2022-07-14 16:01:30 +01:00
/* Check length */
2020-06-04 18:45:25 +01:00
if ( length ! = 17 ) {
2021-06-27 11:47:55 +01:00
strcpy ( symbol - > errtxt , " 336: Input wrong length (17 characters required) " ) ;
2018-02-03 18:44:01 +00:00
return ZINT_ERROR_TOO_LONG ;
}
2020-06-04 18:45:25 +01:00
2022-07-14 16:01:30 +01:00
/* Check input characters, I, O and Q are not allowed */
2021-10-20 23:05:30 +01:00
if ( ! is_sane ( ARSENIC_F , source , length ) ) {
2021-10-30 22:00:31 +01:00
strcpy ( symbol - > errtxt ,
" 337: Invalid character in data (alphanumerics only, excluding \" I \" , \" O \" and \" Q \" ) " ) ;
2018-02-04 08:42:01 +00:00
return ZINT_ERROR_INVALID_DATA ;
}
2020-06-04 18:45:25 +01:00
2021-10-20 23:05:30 +01:00
to_upper ( source , length ) ;
2020-06-04 18:45:25 +01:00
2022-07-14 16:01:30 +01:00
/* Check digit only valid for North America */
2021-10-20 23:05:30 +01:00
if ( source [ 0 ] > = ' 1 ' & & source [ 0 ] < = ' 5 ' ) {
input_check = source [ 8 ] ;
2020-06-04 18:45:25 +01:00
sum = 0 ;
for ( i = 0 ; i < 17 ; i + + ) {
2021-10-20 23:05:30 +01:00
int value ;
if ( source [ i ] < = ' 9 ' ) {
value = source [ i ] - ' 0 ' ;
} else if ( source [ i ] < = ' H ' ) {
value = ( source [ i ] - ' A ' ) + 1 ;
} else if ( source [ i ] < = ' R ' ) {
value = ( source [ i ] - ' J ' ) + 1 ;
} else { /* (source[i] >= 'S') && (source[i] <= 'Z') */
value = ( source [ i ] - ' S ' ) + 2 ;
}
sum + = value * weight [ i ] ;
2018-02-03 18:44:01 +00:00
}
2020-06-04 18:45:25 +01:00
output_check = ' 0 ' + ( sum % 11 ) ;
if ( output_check = = ' : ' ) {
2022-07-14 16:01:30 +01:00
/* Check digit was 10 */
2020-06-04 18:45:25 +01:00
output_check = ' X ' ;
2018-02-03 18:44:01 +00:00
}
2020-06-04 18:45:25 +01:00
if ( symbol - > debug & ZINT_DEBUG_PRINT ) {
2021-10-20 23:05:30 +01:00
printf ( " Producing VIN code: %s \n " , source ) ;
2020-06-04 18:45:25 +01:00
printf ( " Input check was %c, calculated check is %c \n " , input_check , output_check ) ;
2018-02-03 18:44:01 +00:00
}
2020-06-04 18:45:25 +01:00
if ( input_check ! = output_check ) {
2021-10-20 23:05:30 +01:00
sprintf ( symbol - > errtxt , " 338: Invalid check digit '%c' (position 9), expecting '%c' " ,
input_check , output_check ) ;
2021-06-27 11:47:55 +01:00
return ZINT_ERROR_INVALID_CHECK ;
2018-02-03 18:44:01 +00:00
}
}
2020-06-04 18:45:25 +01:00
2018-02-03 18:44:01 +00:00
/* Start character */
2021-10-20 23:05:30 +01:00
memcpy ( d , " 1211212111 " , 10 ) ;
d + = 10 ;
2020-06-04 18:45:25 +01:00
/* Import character 'I' prefix? */
if ( symbol - > option_2 & 1 ) {
2021-10-20 23:05:30 +01:00
memcpy ( d , C39Table [ 18 ] , 10 ) ;
d + = 10 ;
2020-06-04 18:45:25 +01:00
}
2022-07-14 16:01:30 +01:00
/* Copy glyphs to symbol */
2021-10-20 23:05:30 +01:00
for ( i = 0 ; i < 17 ; i + + , d + = 10 ) {
memcpy ( d , C39Table [ posn ( SILVER , source [ i ] ) ] , 10 ) ;
2018-02-03 18:44:01 +00:00
}
2020-06-04 18:45:25 +01:00
2021-10-20 23:05:30 +01:00
/* Stop character */
memcpy ( d , " 121121211 " , 9 ) ;
d + = 9 ;
expand ( symbol , dest , d - dest ) ;
2020-06-04 18:45:25 +01:00
2021-10-20 23:05:30 +01:00
ustrcpy ( symbol - > text , source ) ;
2020-06-04 18:45:25 +01:00
2021-06-19 13:11:23 +01:00
/* Specification of dimensions/height for BARCODE_VIN unlikely */
2018-02-03 18:44:01 +00:00
return 0 ;
}
2022-07-14 16:01:30 +01:00
/* vim: set ts=4 sw=4 et : */