2008-07-14 01:15:55 +04:00
/* plessey.c - Handles Plessey and MSI Plessey */
/*
libzint - the open source barcode library
2017-08-27 11:31:02 +03:00
Copyright ( C ) 2008 - 2017 Robin Stuart < rstuart114 @ gmail . com >
2008-07-14 01:15:55 +04:00
2013-05-16 21:26:38 +04: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 22:37:52 +03:00
1. Redistributions of source code must retain the above copyright
notice , this list of conditions and the following disclaimer .
2013-05-16 21:26:38 +04: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 22:37:52 +03:00
documentation and / or other materials provided with the distribution .
2013-05-16 21:26:38 +04: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 22:37:52 +03:00
without specific prior written permission .
2013-05-16 21:26:38 +04: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 22:37:52 +03:00
OUT OF THE USE OF THIS SOFTWARE , EVEN IF ADVISED OF THE POSSIBILITY OF
2013-05-16 21:26:38 +04:00
SUCH DAMAGE .
2016-02-20 13:50:15 +03:00
*/
2019-12-19 03:37:55 +03:00
/* vim: set ts=4 sw=4 et : */
2008-07-14 01:15:55 +04:00
# include <stdio.h>
# include <stdlib.h>
# include <string.h>
# include "common.h"
# define SSET "0123456789ABCDEF"
2016-02-20 13:50:15 +03:00
static const char * PlessTable [ 16 ] = {
" 13131313 " , " 31131313 " , " 13311313 " , " 31311313 " , " 13133113 " , " 31133113 " ,
" 13313113 " , " 31313113 " , " 13131331 " , " 31131331 " , " 13311331 " , " 31311331 " , " 13133131 " ,
" 31133131 " , " 13313131 " , " 31313131 "
} ;
static const char * MSITable [ 10 ] = {
" 12121212 " , " 12121221 " , " 12122112 " , " 12122121 " , " 12211212 " , " 12211221 " ,
" 12212112 " , " 12212121 " , " 21121212 " , " 21121221 "
} ;
/* Not MSI/Plessey but the older Plessey standard */
2019-12-19 03:37:55 +03:00
INTERNAL int plessey ( struct zint_symbol * symbol , unsigned char source [ ] , const size_t length ) {
2016-02-20 13:50:15 +03:00
2017-09-10 18:03:09 +03:00
unsigned int i ;
2016-02-20 13:50:15 +03:00
unsigned char * checkptr ;
static const char grid [ 9 ] = { 1 , 1 , 1 , 1 , 0 , 1 , 0 , 0 , 1 } ;
char dest [ 1024 ] ; /* 8 + 65 * 8 + 8 * 2 + 9 + 1 ~ 1024 */
int error_number ;
if ( length > 65 ) {
2017-07-27 18:01:53 +03:00
strcpy ( symbol - > errtxt , " 370: Input too long " ) ;
2016-02-20 13:50:15 +03:00
return ZINT_ERROR_TOO_LONG ;
}
error_number = is_sane ( SSET , source , length ) ;
if ( error_number = = ZINT_ERROR_INVALID_DATA ) {
2017-07-27 18:01:53 +03:00
strcpy ( symbol - > errtxt , " 371: Invalid characters in data " ) ;
2016-02-20 13:50:15 +03:00
return error_number ;
}
checkptr = ( unsigned char * ) calloc ( 1 , length * 4 + 8 ) ;
/* Start character */
strcpy ( dest , " 31311331 " ) ;
/* Data area */
for ( i = 0 ; i < length ; i + + ) {
2017-09-10 18:03:09 +03:00
unsigned int check = posn ( SSET , source [ i ] ) ;
2016-02-20 13:50:15 +03:00
lookup ( SSET , PlessTable , source [ i ] , dest ) ;
checkptr [ 4 * i ] = check & 1 ;
checkptr [ 4 * i + 1 ] = ( check > > 1 ) & 1 ;
checkptr [ 4 * i + 2 ] = ( check > > 2 ) & 1 ;
checkptr [ 4 * i + 3 ] = ( check > > 3 ) & 1 ;
}
/* CRC check digit code adapted from code by Leonid A. Broukhis
used in GNU Barcode */
for ( i = 0 ; i < ( 4 * length ) ; i + + ) {
2017-09-10 18:03:09 +03:00
if ( checkptr [ i ] ) {
int j ;
2016-02-20 13:50:15 +03:00
for ( j = 0 ; j < 9 ; j + + )
checkptr [ i + j ] ^ = grid [ j ] ;
2017-09-10 18:03:09 +03:00
}
2016-02-20 13:50:15 +03:00
}
for ( i = 0 ; i < 8 ; i + + ) {
switch ( checkptr [ length * 4 + i ] ) {
2016-03-03 00:12:38 +03:00
case 0 : strcat ( dest , " 13 " ) ;
2016-02-20 13:50:15 +03:00
break ;
2016-03-03 00:12:38 +03:00
case 1 : strcat ( dest , " 31 " ) ;
2016-02-20 13:50:15 +03:00
break ;
}
}
/* Stop character */
2016-03-03 00:12:38 +03:00
strcat ( dest , " 331311313 " ) ;
2016-02-20 13:50:15 +03:00
expand ( symbol , dest ) ;
ustrcpy ( symbol - > text , source ) ;
free ( checkptr ) ;
return error_number ;
2008-07-14 01:15:55 +04:00
}
2016-02-20 13:50:15 +03:00
/* Plain MSI Plessey - does not calculate any check character */
2019-12-19 03:37:55 +03:00
static int msi_plessey ( struct zint_symbol * symbol , unsigned char source [ ] , const size_t length ) {
2008-07-14 01:15:55 +04:00
2017-05-29 12:43:47 +03:00
size_t i ;
2016-02-20 13:50:15 +03:00
char dest [ 512 ] ; /* 2 + 55 * 8 + 3 + 1 ~ 512 */
2012-12-31 17:41:59 +04:00
2016-02-20 13:50:15 +03:00
if ( length > 55 ) {
2017-07-27 18:01:53 +03:00
strcpy ( symbol - > errtxt , " 372: Input too long " ) ;
2016-02-20 13:50:15 +03:00
return ZINT_ERROR_TOO_LONG ;
}
2008-07-14 01:15:55 +04:00
2016-02-20 13:50:15 +03:00
/* start character */
strcpy ( dest , " 21 " ) ;
2008-07-14 01:15:55 +04:00
2016-02-20 13:50:15 +03:00
for ( i = 0 ; i < length ; i + + ) {
lookup ( NEON , MSITable , source [ i ] , dest ) ;
}
2008-07-14 01:15:55 +04:00
2016-02-20 13:50:15 +03:00
/* Stop character */
2016-03-03 00:12:38 +03:00
strcat ( dest , " 121 " ) ;
2012-12-31 17:41:59 +04:00
2016-02-20 13:50:15 +03:00
expand ( symbol , dest ) ;
ustrcpy ( symbol - > text , source ) ;
return 0 ;
2008-07-14 01:15:55 +04:00
}
2016-02-20 13:50:15 +03:00
/* MSI Plessey with Modulo 10 check digit - algorithm from Barcode Island
* http : //www.barcodeisland.com/ */
2019-12-19 03:37:55 +03:00
static int msi_plessey_mod10 ( struct zint_symbol * symbol , unsigned char source [ ] , int length ) {
2016-03-03 00:12:38 +03:00
2016-02-20 13:50:15 +03:00
unsigned long i , wright , dau , pedwar , pump , n ;
char un [ 200 ] , tri [ 32 ] ;
int error_number , h ;
char dest [ 1000 ] ;
error_number = 0 ;
if ( length > 18 ) {
2017-07-27 18:01:53 +03:00
strcpy ( symbol - > errtxt , " 373: Input too long " ) ;
2016-02-20 13:50:15 +03:00
return ZINT_ERROR_TOO_LONG ;
}
/* start character */
strcpy ( dest , " 21 " ) ;
/* draw data section */
for ( i = 0 ; i < length ; i + + ) {
lookup ( NEON , MSITable , source [ i ] , dest ) ;
}
/* calculate check digit */
wright = 0 ;
n = ! ( length & 1 ) ;
for ( i = n ; i < length ; i + = 2 ) {
un [ wright + + ] = source [ i ] ;
}
un [ wright ] = ' \0 ' ;
dau = strtoul ( un , NULL , 10 ) ;
dau * = 2 ;
2017-09-10 18:03:09 +03:00
sprintf ( tri , " %lu " , dau ) ;
2016-02-20 13:50:15 +03:00
pedwar = 0 ;
h = strlen ( tri ) ;
for ( i = 0 ; i < h ; i + + ) {
pedwar + = ctoi ( tri [ i ] ) ;
}
n = length & 1 ;
for ( i = n ; i < length ; i + = 2 ) {
pedwar + = ctoi ( source [ i ] ) ;
}
pump = ( 10 - pedwar % 10 ) ;
if ( pump = = 10 ) {
pump = 0 ;
}
/* draw check digit */
lookup ( NEON , MSITable , itoc ( pump ) , dest ) ;
/* Stop character */
2016-03-03 00:12:38 +03:00
strcat ( dest , " 121 " ) ;
2016-02-20 13:50:15 +03:00
expand ( symbol , dest ) ;
ustrcpy ( symbol - > text , source ) ;
symbol - > text [ length ] = itoc ( pump ) ;
symbol - > text [ length + 1 ] = ' \0 ' ;
return error_number ;
2008-07-14 01:15:55 +04:00
}
2016-02-20 13:50:15 +03:00
/* MSI Plessey with two Modulo 10 check digits - algorithm from
* Barcode Island http : //www.barcodeisland.com/ */
2019-12-19 03:37:55 +03:00
static int msi_plessey_mod1010 ( struct zint_symbol * symbol , unsigned char source [ ] , const unsigned int src_len ) {
2016-03-03 00:12:38 +03:00
2008-07-14 01:15:55 +04:00
2016-02-20 13:50:15 +03:00
unsigned long i , n , wright , dau , pedwar , pump , chwech ;
char un [ 16 ] , tri [ 32 ] ;
int error_number , h ;
char dest [ 1000 ] ;
2008-07-14 01:15:55 +04:00
2016-02-20 13:50:15 +03:00
error_number = 0 ;
2008-07-14 01:15:55 +04:00
2016-02-20 13:50:15 +03:00
if ( src_len > 18 ) {
/* No Entry Stack Smashers! limit because of str->number conversion*/
2017-07-27 18:01:53 +03:00
strcpy ( symbol - > errtxt , " 374: Input too long " ) ;
2016-02-20 13:50:15 +03:00
return ZINT_ERROR_TOO_LONG ;
}
2008-09-16 11:44:01 +04:00
2016-02-20 13:50:15 +03:00
/* start character */
strcpy ( dest , " 21 " ) ;
/* draw data section */
for ( i = 0 ; i < src_len ; i + + ) {
lookup ( NEON , MSITable , source [ i ] , dest ) ;
}
/* calculate first check digit */
wright = 0 ;
n = ! ( src_len & 1 ) ;
for ( i = n ; i < src_len ; i + = 2 ) {
un [ wright + + ] = source [ i ] ;
}
un [ wright ] = ' \0 ' ;
dau = strtoul ( un , NULL , 10 ) ;
dau * = 2 ;
2017-09-10 18:03:09 +03:00
sprintf ( tri , " %lu " , dau ) ;
2016-02-20 13:50:15 +03:00
pedwar = 0 ;
h = strlen ( tri ) ;
for ( i = 0 ; i < h ; i + + ) {
pedwar + = ctoi ( tri [ i ] ) ;
}
n = src_len & 1 ;
for ( i = n ; i < src_len ; i + = 2 ) {
pedwar + = ctoi ( source [ i ] ) ;
}
pump = 10 - pedwar % 10 ;
if ( pump = = 10 ) {
pump = 0 ;
}
/* calculate second check digit */
wright = 0 ;
n = src_len & 1 ;
for ( i = n ; i < src_len ; i + = 2 ) {
un [ wright + + ] = source [ i ] ;
}
un [ wright + + ] = itoc ( pump ) ;
un [ wright ] = ' \0 ' ;
2008-10-03 12:22:08 +04:00
2016-02-20 13:50:15 +03:00
dau = strtoul ( un , NULL , 10 ) ;
dau * = 2 ;
2012-12-31 17:41:59 +04:00
2017-09-10 18:03:09 +03:00
sprintf ( tri , " %lu " , dau ) ;
2009-10-06 23:03:00 +04:00
2016-02-20 13:50:15 +03:00
pedwar = 0 ;
h = strlen ( tri ) ;
for ( i = 0 ; i < h ; i + + ) {
pedwar + = ctoi ( tri [ i ] ) ;
}
2008-09-16 11:44:01 +04:00
2012-12-31 17:41:59 +04:00
2016-02-20 13:50:15 +03:00
i = ! ( src_len & 1 ) ;
for ( ; i < src_len ; i + = 2 ) {
pedwar + = ctoi ( source [ i ] ) ;
}
chwech = 10 - pedwar % 10 ;
if ( chwech = = 10 ) {
chwech = 0 ;
}
/* Draw check digits */
lookup ( NEON , MSITable , itoc ( pump ) , dest ) ;
lookup ( NEON , MSITable , itoc ( chwech ) , dest ) ;
/* Stop character */
2016-03-03 00:12:38 +03:00
strcat ( dest , " 121 " ) ;
2016-02-20 13:50:15 +03:00
expand ( symbol , dest ) ;
ustrcpy ( symbol - > text , source ) ;
symbol - > text [ src_len ] = itoc ( pump ) ;
symbol - > text [ src_len + 1 ] = itoc ( chwech ) ;
symbol - > text [ src_len + 2 ] = ' \0 ' ;
return error_number ;
}
2017-10-23 22:37:52 +03:00
/* Calculate a Modulo 11 check digit using the system discussed on Wikipedia -
2016-02-20 13:50:15 +03:00
see http : //en.wikipedia.org/wiki/Talk:MSI_Barcode */
2019-12-19 03:37:55 +03:00
static int msi_plessey_mod11 ( struct zint_symbol * symbol , unsigned char source [ ] , const unsigned int src_len ) {
2016-02-20 13:50:15 +03:00
/* uses the IBM weight system */
int i , weight , x , check ;
int error_number ;
char dest [ 1000 ] ;
error_number = 0 ;
if ( src_len > 55 ) {
2017-07-27 18:01:53 +03:00
strcpy ( symbol - > errtxt , " 375: Input too long " ) ;
2016-02-20 13:50:15 +03:00
return ZINT_ERROR_TOO_LONG ;
}
/* start character */
strcpy ( dest , " 21 " ) ;
/* draw data section */
for ( i = 0 ; i < src_len ; i + + ) {
lookup ( NEON , MSITable , source [ i ] , dest ) ;
}
/* calculate check digit */
x = 0 ;
weight = 2 ;
for ( i = src_len - 1 ; i > = 0 ; i - - ) {
x + = weight * ctoi ( source [ i ] ) ;
weight + + ;
if ( weight > 7 ) {
weight = 2 ;
}
}
check = ( 11 - ( x % 11 ) ) % 11 ;
if ( check = = 10 ) {
lookup ( NEON , MSITable , ' 1 ' , dest ) ;
lookup ( NEON , MSITable , ' 0 ' , dest ) ;
} else {
lookup ( NEON , MSITable , itoc ( check ) , dest ) ;
}
/* stop character */
2016-03-03 00:12:38 +03:00
strcat ( dest , " 121 " ) ;
2016-02-20 13:50:15 +03:00
expand ( symbol , dest ) ;
ustrcpy ( symbol - > text , source ) ;
if ( check = = 10 ) {
2016-03-03 00:12:38 +03:00
strcat ( ( char * ) symbol - > text , " 10 " ) ;
2016-02-20 13:50:15 +03:00
} else {
symbol - > text [ src_len ] = itoc ( check ) ;
symbol - > text [ src_len + 1 ] = ' \0 ' ;
}
return error_number ;
}
/* Combining the Barcode Island and Wikipedia code
* Verified against http : //www.bokai.com/BarcodeJSP/applet/BarcodeSampleApplet.htm */
2019-12-19 03:37:55 +03:00
static int msi_plessey_mod1110 ( struct zint_symbol * symbol , unsigned char source [ ] , const unsigned int src_len ) {
2016-02-20 13:50:15 +03:00
/* Weighted using the IBM system */
2017-05-29 12:43:47 +03:00
unsigned long i , weight , x , check , wright , dau , pedwar , pump ;
size_t h ;
2016-02-20 13:50:15 +03:00
long si ;
char un [ 16 ] , tri [ 16 ] ;
int error_number ;
char dest [ 1000 ] ;
unsigned char temp [ 32 ] ;
unsigned int temp_len ;
error_number = 0 ;
if ( src_len > 18 ) {
2017-07-27 18:01:53 +03:00
strcpy ( symbol - > errtxt , " 376: Input too long " ) ;
2016-02-20 13:50:15 +03:00
return ZINT_ERROR_TOO_LONG ;
}
/* start character */
strcpy ( dest , " 21 " ) ;
/* draw data section */
for ( i = 0 ; i < src_len ; i + + ) {
lookup ( NEON , MSITable , source [ i ] , dest ) ;
}
/* calculate first (mod 11) digit */
x = 0 ;
weight = 2 ;
for ( si = src_len - 1 ; si > = 0 ; si - - ) {
x + = weight * ctoi ( source [ si ] ) ;
weight + + ;
if ( weight > 7 ) {
weight = 2 ;
}
}
check = ( 11 - ( x % 11 ) ) % 11 ;
ustrcpy ( temp , source ) ;
temp_len = src_len ;
if ( check = = 10 ) {
lookup ( NEON , MSITable , ' 1 ' , dest ) ;
lookup ( NEON , MSITable , ' 0 ' , dest ) ;
2016-03-03 00:12:38 +03:00
strcat ( ( char * ) temp , " 10 " ) ;
2016-02-20 13:50:15 +03:00
temp_len + = 2 ;
} else {
lookup ( NEON , MSITable , itoc ( check ) , dest ) ;
temp [ temp_len + + ] = itoc ( check ) ;
temp [ temp_len ] = ' \0 ' ;
}
/* calculate second (mod 10) check digit */
wright = 0 ;
i = ! ( temp_len & 1 ) ;
for ( ; i < temp_len ; i + = 2 ) {
un [ wright + + ] = temp [ i ] ;
}
un [ wright ] = ' \0 ' ;
dau = strtoul ( un , NULL , 10 ) ;
dau * = 2 ;
2017-09-10 18:03:09 +03:00
sprintf ( tri , " %lu " , dau ) ;
2016-02-20 13:50:15 +03:00
pedwar = 0 ;
h = strlen ( tri ) ;
for ( i = 0 ; i < h ; i + + ) {
pedwar + = ctoi ( tri [ i ] ) ;
}
i = temp_len & 1 ;
for ( ; i < temp_len ; i + = 2 ) {
pedwar + = ctoi ( temp [ i ] ) ;
}
pump = 10 - pedwar % 10 ;
if ( pump = = 10 ) {
pump = 0 ;
}
/* draw check digit */
lookup ( NEON , MSITable , itoc ( pump ) , dest ) ;
/* stop character */
2016-03-03 00:12:38 +03:00
strcat ( dest , " 121 " ) ;
2016-02-20 13:50:15 +03:00
expand ( symbol , dest ) ;
temp [ temp_len + + ] = itoc ( pump ) ;
temp [ temp_len ] = ' \0 ' ;
ustrcpy ( symbol - > text , temp ) ;
return error_number ;
}
2019-12-19 03:37:55 +03:00
INTERNAL int msi_handle ( struct zint_symbol * symbol , unsigned char source [ ] , int length ) {
2016-02-20 13:50:15 +03:00
int error_number ;
error_number = is_sane ( NEON , source , length ) ;
if ( error_number ! = 0 ) {
2017-07-27 18:01:53 +03:00
strcpy ( symbol - > errtxt , " 377: Invalid characters in input data " ) ;
2016-02-20 13:50:15 +03:00
return ZINT_ERROR_INVALID_DATA ;
}
if ( ( symbol - > option_2 < 0 ) | | ( symbol - > option_2 > 4 ) ) {
symbol - > option_2 = 0 ;
}
switch ( symbol - > option_2 ) {
case 0 : error_number = msi_plessey ( symbol , source , length ) ;
break ;
case 1 : error_number = msi_plessey_mod10 ( symbol , source , length ) ;
break ;
case 2 : error_number = msi_plessey_mod1010 ( symbol , source , length ) ;
break ;
case 3 : error_number = msi_plessey_mod11 ( symbol , source , length ) ;
break ;
case 4 : error_number = msi_plessey_mod1110 ( symbol , source , length ) ;
break ;
}
return error_number ;
2008-07-14 01:15:55 +04:00
}