2008-07-14 01:15:55 +04:00
/* 2of5.c - Handles Code 2 of 5 barcodes */
/*
libzint - the open source barcode library
2008-11-17 11:47:42 +03:00
Copyright ( C ) 2008 Robin Stuart < robin @ zint . org . uk >
2008-07-14 01:15:55 +04:00
This program is free software ; you can redistribute it and / or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation ; either version 3 of the License , or
( at your option ) any later version .
This program is distributed in the hope that it will be useful ,
but WITHOUT ANY WARRANTY ; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
GNU General Public License for more details .
You should have received a copy of the GNU General Public License along
with this program ; if not , write to the Free Software Foundation , Inc . ,
51 Franklin Street , Fifth Floor , Boston , MA 02110 - 1301 USA .
*/
# include <string.h>
# include <stdio.h>
# include <stdlib.h>
# include "common.h"
static char * C25MatrixTable [ 10 ] = { " 113311 " , " 311131 " , " 131131 " , " 331111 " , " 113131 " , " 313111 " ,
" 133111 " , " 111331 " , " 311311 " , " 131311 " } ;
static char * C25IndustTable [ 10 ] = { " 1111313111 " , " 3111111131 " , " 1131111131 " , " 3131111111 " , " 1111311131 " ,
" 3111311111 " , " 1131311111 " , " 1111113131 " , " 3111113111 " , " 1131113111 " } ;
static char * C25InterTable [ 10 ] = { " 11331 " , " 31113 " , " 13113 " , " 33111 " , " 11313 " , " 31311 " , " 13311 " , " 11133 " ,
" 31131 " , " 13131 " } ;
2009-09-29 13:45:46 +04:00
int matrix_two_of_five ( struct zint_symbol * symbol , unsigned char source [ ] , int length )
2008-07-14 01:15:55 +04:00
{ /* Code 2 of 5 Standard (Code 2 of 5 Matrix) */
2008-09-19 12:47:28 +04:00
int i , error_number ;
2009-10-06 23:03:00 +04:00
char dest [ 512 ] ; /* 6 + 80 * 6 + 6 + 1 ~ 512*/
2008-07-14 01:15:55 +04:00
2008-09-19 12:47:28 +04:00
error_number = 0 ;
2008-07-14 01:15:55 +04:00
2009-09-29 13:45:46 +04:00
if ( length > 80 ) {
2009-06-01 00:33:54 +04:00
strcpy ( symbol - > errtxt , " Input too long " ) ;
2008-07-14 01:15:55 +04:00
return ERROR_TOO_LONG ;
}
2009-10-06 23:03:00 +04:00
error_number = is_sane ( NEON , source , length ) ;
2008-09-19 12:47:28 +04:00
if ( error_number = = ERROR_INVALID_DATA ) {
2009-06-01 00:33:54 +04:00
strcpy ( symbol - > errtxt , " Invalid characters in data " ) ;
2008-09-19 12:47:28 +04:00
return error_number ;
2008-07-14 01:15:55 +04:00
}
/* start character */
2009-10-06 23:03:00 +04:00
strcpy ( dest , " 411111 " ) ;
2008-07-14 01:15:55 +04:00
2009-10-06 23:03:00 +04:00
for ( i = 0 ; i < length ; i + + ) {
lookup ( NEON , C25MatrixTable , source [ i ] , dest ) ;
2008-07-14 01:15:55 +04:00
}
/* Stop character */
concat ( dest , " 41111 " ) ;
expand ( symbol , dest ) ;
2009-02-19 22:09:57 +03:00
ustrcpy ( symbol - > text , source ) ;
2008-09-19 12:47:28 +04:00
return error_number ;
2008-07-14 01:15:55 +04:00
}
2009-09-29 13:45:46 +04:00
int industrial_two_of_five ( struct zint_symbol * symbol , unsigned char source [ ] , int length )
2008-07-14 01:15:55 +04:00
{ /* Code 2 of 5 Industrial */
2008-09-19 12:47:28 +04:00
int i , error_number ;
2009-10-06 23:03:00 +04:00
char dest [ 512 ] ; /* 6 + 40 * 10 + 6 + 1 */
2008-07-14 01:15:55 +04:00
2008-09-19 12:47:28 +04:00
error_number = 0 ;
2008-07-14 01:15:55 +04:00
2009-09-29 13:45:46 +04:00
if ( length > 45 ) {
2009-06-01 00:33:54 +04:00
strcpy ( symbol - > errtxt , " Input too long " ) ;
2008-07-14 01:15:55 +04:00
return ERROR_TOO_LONG ;
}
2009-10-06 23:03:00 +04:00
error_number = is_sane ( NEON , source , length ) ;
2008-09-19 12:47:28 +04:00
if ( error_number = = ERROR_INVALID_DATA ) {
2009-06-01 00:33:54 +04:00
strcpy ( symbol - > errtxt , " Invalid character in data " ) ;
2008-09-19 12:47:28 +04:00
return error_number ;
2008-07-14 01:15:55 +04:00
}
/* start character */
2009-10-06 23:03:00 +04:00
strcpy ( dest , " 313111 " ) ;
2008-07-14 01:15:55 +04:00
2009-10-06 23:03:00 +04:00
for ( i = 0 ; i < length ; i + + ) {
lookup ( NEON , C25IndustTable , source [ i ] , dest ) ;
2008-07-14 01:15:55 +04:00
}
/* Stop character */
concat ( dest , " 31113 " ) ;
expand ( symbol , dest ) ;
2009-02-19 22:09:57 +03:00
ustrcpy ( symbol - > text , source ) ;
2008-09-19 12:47:28 +04:00
return error_number ;
2008-07-14 01:15:55 +04:00
}
2009-09-29 13:45:46 +04:00
int iata_two_of_five ( struct zint_symbol * symbol , unsigned char source [ ] , int length )
2008-07-14 01:15:55 +04:00
{ /* Code 2 of 5 IATA */
2008-09-19 12:47:28 +04:00
int i , error_number ;
2009-10-06 23:03:00 +04:00
char dest [ 512 ] ; /* 4 + 45 * 10 + 3 + 1 */
2008-07-14 01:15:55 +04:00
2008-09-19 12:47:28 +04:00
error_number = 0 ;
2008-07-14 01:15:55 +04:00
2009-09-29 13:45:46 +04:00
if ( length > 45 ) {
2009-06-01 00:33:54 +04:00
strcpy ( symbol - > errtxt , " Input too long " ) ;
2008-07-14 01:15:55 +04:00
return ERROR_TOO_LONG ;
}
2009-10-06 23:03:00 +04:00
error_number = is_sane ( NEON , source , length ) ;
2008-09-19 12:47:28 +04:00
if ( error_number = = ERROR_INVALID_DATA ) {
2009-06-01 00:33:54 +04:00
strcpy ( symbol - > errtxt , " Invalid characters in data " ) ;
2008-09-19 12:47:28 +04:00
return error_number ;
2008-07-14 01:15:55 +04:00
}
/* start */
2009-10-06 23:03:00 +04:00
strcpy ( dest , " 1111 " ) ;
2008-07-14 01:15:55 +04:00
2009-09-29 13:45:46 +04:00
for ( i = 0 ; i < length ; i + + ) {
2009-10-06 23:03:00 +04:00
lookup ( NEON , C25IndustTable , source [ i ] , dest ) ;
2008-07-14 01:15:55 +04:00
}
/* stop */
concat ( dest , " 311 " ) ;
expand ( symbol , dest ) ;
2009-02-19 22:09:57 +03:00
ustrcpy ( symbol - > text , source ) ;
2008-09-19 12:47:28 +04:00
return error_number ;
2008-07-14 01:15:55 +04:00
}
2009-09-29 13:45:46 +04:00
int logic_two_of_five ( struct zint_symbol * symbol , unsigned char source [ ] , int length )
2008-07-14 01:15:55 +04:00
{ /* Code 2 of 5 Data Logic */
2008-09-19 12:47:28 +04:00
int i , error_number ;
2009-10-06 23:03:00 +04:00
char dest [ 512 ] ; /* 4 + 80 * 6 + 3 + 1 */
2008-07-14 01:15:55 +04:00
2008-09-19 12:47:28 +04:00
error_number = 0 ;
2008-07-14 01:15:55 +04:00
2009-09-29 13:45:46 +04:00
if ( length > 80 ) {
2009-06-01 00:33:54 +04:00
strcpy ( symbol - > errtxt , " Input too long " ) ;
2008-07-14 01:15:55 +04:00
return ERROR_TOO_LONG ;
}
2009-10-06 23:03:00 +04:00
error_number = is_sane ( NEON , source , length ) ;
2008-09-19 12:47:28 +04:00
if ( error_number = = ERROR_INVALID_DATA ) {
2009-06-01 00:33:54 +04:00
strcpy ( symbol - > errtxt , " Invalid characters in data " ) ;
2008-09-19 12:47:28 +04:00
return error_number ;
2008-07-14 01:15:55 +04:00
}
/* start character */
2009-10-06 23:03:00 +04:00
strcpy ( dest , " 1111 " ) ;
2008-07-14 01:15:55 +04:00
2009-10-06 23:03:00 +04:00
for ( i = 0 ; i < length ; i + + ) {
lookup ( NEON , C25MatrixTable , source [ i ] , dest ) ;
2008-07-14 01:15:55 +04:00
}
/* Stop character */
concat ( dest , " 311 " ) ;
expand ( symbol , dest ) ;
2009-02-19 22:09:57 +03:00
ustrcpy ( symbol - > text , source ) ;
2008-09-19 12:47:28 +04:00
return error_number ;
2008-07-14 01:15:55 +04:00
}
2009-09-29 13:45:46 +04:00
int interleaved_two_of_five ( struct zint_symbol * symbol , unsigned char source [ ] , int length )
2008-07-14 01:15:55 +04:00
{ /* Code 2 of 5 Interleaved */
2008-09-19 12:47:28 +04:00
int i , j , k , error_number ;
2008-07-14 01:15:55 +04:00
char bars [ 7 ] , spaces [ 7 ] , mixed [ 14 ] , dest [ 1000 ] ;
2009-11-02 18:47:08 +03:00
# ifndef _MSC_VER
unsigned char temp [ length + 2 ] ;
# else
unsigned char * temp = ( unsigned char * ) _alloca ( ( length + 2 ) * sizeof ( unsigned char ) ) ;
# endif
2008-07-14 01:15:55 +04:00
2008-09-19 12:47:28 +04:00
error_number = 0 ;
2008-07-14 01:15:55 +04:00
2009-11-02 18:47:08 +03:00
if ( length > 89 ) {
2009-06-01 00:33:54 +04:00
strcpy ( symbol - > errtxt , " Input too long " ) ;
2008-07-14 01:15:55 +04:00
return ERROR_TOO_LONG ;
}
2009-10-06 23:03:00 +04:00
error_number = is_sane ( NEON , source , length ) ;
2008-09-19 12:47:28 +04:00
if ( error_number = = ERROR_INVALID_DATA ) {
2009-06-01 00:33:54 +04:00
strcpy ( symbol - > errtxt , " Invalid characters in data " ) ;
2008-09-19 12:47:28 +04:00
return error_number ;
2008-07-14 01:15:55 +04:00
}
2009-11-02 18:47:08 +03:00
ustrcpy ( temp , ( unsigned char * ) " " ) ;
2008-07-14 01:15:55 +04:00
/* Input must be an even number of characters for Interlaced 2 of 5 to work:
if an odd number of characters has been entered then add a leading zero */
2009-10-06 23:03:00 +04:00
if ( ( length % 2 ) ! = 0 )
2008-07-14 01:15:55 +04:00
{
2009-11-02 18:47:08 +03:00
ustrcpy ( temp , ( unsigned char * ) " 0 " ) ;
length + + ;
2008-07-14 01:15:55 +04:00
}
2009-11-02 18:47:08 +03:00
uconcat ( temp , source ) ;
2008-07-14 01:15:55 +04:00
/* start character */
2009-10-06 23:03:00 +04:00
strcpy ( dest , " 1111 " ) ;
2008-07-14 01:15:55 +04:00
2009-11-02 18:47:08 +03:00
for ( i = 0 ; i < length ; i + = 2 )
2008-07-14 01:15:55 +04:00
{
/* look up the bars and the spaces and put them in two strings */
strcpy ( bars , " " ) ;
2009-10-06 23:03:00 +04:00
lookup ( NEON , C25InterTable , temp [ i ] , bars ) ;
2008-07-14 01:15:55 +04:00
strcpy ( spaces , " " ) ;
2009-10-06 23:03:00 +04:00
lookup ( NEON , C25InterTable , temp [ i + 1 ] , spaces ) ;
2008-07-14 01:15:55 +04:00
/* then merge (interlace) the strings together */
k = 0 ;
for ( j = 0 ; j < = 4 ; j + + )
{
mixed [ k ] = bars [ j ] ; k + + ;
mixed [ k ] = spaces [ j ] ; k + + ;
}
mixed [ k ] = ' \0 ' ;
concat ( dest , mixed ) ;
}
/* Stop character */
concat ( dest , " 311 " ) ;
expand ( symbol , dest ) ;
2009-10-06 23:03:00 +04:00
ustrcpy ( symbol - > text , temp ) ;
2008-09-19 12:47:28 +04:00
return error_number ;
2008-07-14 01:15:55 +04:00
}
2009-09-29 13:45:46 +04:00
int itf14 ( struct zint_symbol * symbol , unsigned char source [ ] , int length )
2008-07-14 01:15:55 +04:00
{
2009-09-29 13:45:46 +04:00
int i , error_number , zeroes ;
2008-07-14 01:15:55 +04:00
unsigned int count , check_digit ;
2009-10-06 23:03:00 +04:00
char localstr [ 16 ] ;
2008-07-14 01:15:55 +04:00
2008-09-19 12:47:28 +04:00
error_number = 0 ;
2008-07-14 01:15:55 +04:00
count = 0 ;
2009-09-29 13:45:46 +04:00
if ( length > 13 ) {
2009-06-01 00:33:54 +04:00
strcpy ( symbol - > errtxt , " Input too long " ) ;
2008-07-14 01:15:55 +04:00
return ERROR_TOO_LONG ;
}
2008-09-19 12:47:28 +04:00
2009-10-06 23:03:00 +04:00
error_number = is_sane ( NEON , source , length ) ;
2008-09-19 12:47:28 +04:00
if ( error_number = = ERROR_INVALID_DATA ) {
2009-06-01 00:33:54 +04:00
strcpy ( symbol - > errtxt , " Invalid character in data " ) ;
2008-09-19 12:47:28 +04:00
return error_number ;
2008-07-14 01:15:55 +04:00
}
2008-10-13 01:05:53 +04:00
/* Add leading zeros as required */
2009-09-29 13:45:46 +04:00
zeroes = 13 - length ;
2008-10-13 01:05:53 +04:00
for ( i = 0 ; i < zeroes ; i + + ) {
2009-10-06 23:03:00 +04:00
localstr [ i ] = ' 0 ' ;
2008-10-13 01:05:53 +04:00
}
2009-10-06 23:03:00 +04:00
strcpy ( localstr + zeroes , ( char * ) source ) ;
2008-10-13 01:05:53 +04:00
2008-07-14 01:15:55 +04:00
/* Calculate the check digit - the same method used for EAN-13 */
2009-07-05 00:49:17 +04:00
for ( i = 12 ; i > = 0 ; i - - )
2008-07-14 01:15:55 +04:00
{
2008-10-13 01:05:53 +04:00
count + = ctoi ( localstr [ i ] ) ;
2008-07-14 01:15:55 +04:00
2009-07-05 00:49:17 +04:00
if ( ( i % 2 ) = = 0 )
2008-07-14 01:15:55 +04:00
{
2008-10-13 01:05:53 +04:00
count + = 2 * ctoi ( localstr [ i ] ) ;
2008-07-14 01:15:55 +04:00
}
}
check_digit = 10 - ( count % 10 ) ;
if ( check_digit = = 10 ) { check_digit = 0 ; }
2009-10-06 23:03:00 +04:00
localstr [ 13 ] = itoc ( check_digit ) ;
localstr [ 14 ] = ' \0 ' ;
2009-09-29 13:45:46 +04:00
error_number = interleaved_two_of_five ( symbol , ( unsigned char * ) localstr , strlen ( localstr ) ) ;
2009-02-19 22:09:57 +03:00
ustrcpy ( symbol - > text , ( unsigned char * ) localstr ) ;
2008-09-19 12:47:28 +04:00
return error_number ;
2008-07-14 01:15:55 +04:00
}
2009-09-29 13:45:46 +04:00
int dpleit ( struct zint_symbol * symbol , unsigned char source [ ] , int length )
2008-07-14 01:15:55 +04:00
{ /* Deutshe Post Leitcode */
2008-09-19 12:47:28 +04:00
int i , error_number ;
2009-09-29 13:45:46 +04:00
unsigned int count , check_digit ;
2009-10-06 23:03:00 +04:00
char localstr [ 16 ] ;
2008-10-13 01:05:53 +04:00
int zeroes ;
2008-07-14 01:15:55 +04:00
2008-09-19 12:47:28 +04:00
error_number = 0 ;
2008-07-14 01:15:55 +04:00
count = 0 ;
2009-09-29 13:45:46 +04:00
if ( length > 13 ) {
2009-06-01 00:33:54 +04:00
strcpy ( symbol - > errtxt , " Input wrong length " ) ;
2008-07-14 01:15:55 +04:00
return ERROR_TOO_LONG ;
}
2009-10-06 23:03:00 +04:00
error_number = is_sane ( NEON , source , length ) ;
2008-09-19 12:47:28 +04:00
if ( error_number = = ERROR_INVALID_DATA ) {
2009-06-01 00:33:54 +04:00
strcpy ( symbol - > errtxt , " Invalid characters in data " ) ;
2008-09-19 12:47:28 +04:00
return error_number ;
2008-07-14 01:15:55 +04:00
}
2009-09-29 13:45:46 +04:00
zeroes = 13 - length ;
2008-10-13 01:05:53 +04:00
for ( i = 0 ; i < zeroes ; i + + )
2009-10-06 23:03:00 +04:00
localstr [ i ] = ' 0 ' ;
strcpy ( localstr + zeroes , ( char * ) source ) ;
2008-10-13 01:05:53 +04:00
for ( i = 12 ; i > = 0 ; i - - )
2008-07-14 01:15:55 +04:00
{
2008-10-13 01:05:53 +04:00
count + = 4 * ctoi ( localstr [ i ] ) ;
2008-07-14 01:15:55 +04:00
if ( ! ( ( i % 2 ) = = 0 ) )
{
2008-10-13 01:05:53 +04:00
count + = 5 * ctoi ( localstr [ i ] ) ;
2008-07-14 01:15:55 +04:00
}
}
check_digit = 10 - ( count % 10 ) ;
if ( check_digit = = 10 ) { check_digit = 0 ; }
2009-10-06 23:03:00 +04:00
localstr [ 13 ] = itoc ( check_digit ) ;
localstr [ 14 ] = ' \0 ' ;
error_number = interleaved_two_of_five ( symbol , ( unsigned char * ) localstr , strlen ( localstr ) ) ;
2009-02-19 22:09:57 +03:00
ustrcpy ( symbol - > text , ( unsigned char * ) localstr ) ;
2008-09-19 12:47:28 +04:00
return error_number ;
2008-07-14 01:15:55 +04:00
}
2009-09-29 13:45:46 +04:00
int dpident ( struct zint_symbol * symbol , unsigned char source [ ] , int length )
2008-07-14 01:15:55 +04:00
{ /* Deutsche Post Identcode */
2008-10-13 01:05:53 +04:00
int i , error_number , zeroes ;
2009-09-29 13:45:46 +04:00
unsigned int count , check_digit ;
2009-10-06 23:03:00 +04:00
char localstr [ 16 ] ;
2008-07-14 01:15:55 +04:00
count = 0 ;
2009-09-29 13:45:46 +04:00
if ( length > 11 ) {
2009-06-01 00:33:54 +04:00
strcpy ( symbol - > errtxt , " Input wrong length " ) ;
2008-07-14 01:15:55 +04:00
return ERROR_TOO_LONG ;
}
2009-10-06 23:03:00 +04:00
error_number = is_sane ( NEON , source , length ) ;
2008-09-19 12:47:28 +04:00
if ( error_number = = ERROR_INVALID_DATA ) {
2009-06-01 00:33:54 +04:00
strcpy ( symbol - > errtxt , " Invalid characters in data " ) ;
2008-09-19 12:47:28 +04:00
return error_number ;
2008-07-14 01:15:55 +04:00
}
2009-09-29 13:45:46 +04:00
zeroes = 11 - length ;
2008-10-13 01:05:53 +04:00
for ( i = 0 ; i < zeroes ; i + + )
2009-10-06 23:03:00 +04:00
localstr [ i ] = ' 0 ' ;
strcpy ( localstr + zeroes , ( char * ) source ) ;
2008-10-13 01:05:53 +04:00
for ( i = 10 ; i > = 0 ; i - - )
2008-07-14 01:15:55 +04:00
{
2008-10-13 01:05:53 +04:00
count + = 4 * ctoi ( localstr [ i ] ) ;
2008-07-14 01:15:55 +04:00
if ( ! ( ( i % 2 ) = = 0 ) )
{
2008-10-13 01:05:53 +04:00
count + = 5 * ctoi ( localstr [ i ] ) ;
2008-07-14 01:15:55 +04:00
}
}
check_digit = 10 - ( count % 10 ) ;
if ( check_digit = = 10 ) { check_digit = 0 ; }
2009-10-06 23:03:00 +04:00
localstr [ 11 ] = itoc ( check_digit ) ;
localstr [ 12 ] = ' \0 ' ;
error_number = interleaved_two_of_five ( symbol , ( unsigned char * ) localstr , strlen ( localstr ) ) ;
2009-02-19 22:09:57 +03:00
ustrcpy ( symbol - > text , ( unsigned char * ) localstr ) ;
2008-09-19 12:47:28 +04:00
return error_number ;
2008-07-14 01:15:55 +04:00
}