2008-07-14 01:15:55 +04:00
/* medical.c - Handles 1 track and 2 track pharmacode and Codabar */
/*
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
*/
# include <stdio.h>
# include <string.h>
# include <stdlib.h>
# include "common.h"
2009-10-06 23:03:00 +04:00
extern int c39 ( struct zint_symbol * symbol , unsigned char source [ ] , int length ) ;
2008-07-14 01:15:55 +04:00
/* Codabar table checked against EN 798:1995 */
2009-10-06 23:03:00 +04:00
# define CALCIUM "0123456789-$: / .+ABCD"
2008-09-02 23:44:41 +04:00
2012-12-29 22:37:03 +04:00
static const char * CodaTable [ 20 ] = { " 11111221 " , " 11112211 " , " 11121121 " , " 22111111 " , " 11211211 " , " 21111211 " ,
2008-07-14 01:15:55 +04:00
" 12111121 " , " 12112111 " , " 12211111 " , " 21121111 " , " 11122111 " , " 11221111 " , " 21112121 " , " 21211121 " ,
" 21212111 " , " 11212121 " , " 11221211 " , " 12121121 " , " 11121221 " , " 11122211 " } ;
2008-10-03 12:22:08 +04:00
2009-09-29 13:45:46 +04:00
int pharma_one ( struct zint_symbol * symbol , unsigned char source [ ] , int length )
2008-07-14 01:15:55 +04:00
{
/* "Pharmacode can represent only a single integer from 3 to 131070. Unlike other
commonly used one - dimensional barcode schemes , pharmacode does not store the data in a
form corresponding to the human - readable digits ; the number is encoded in binary , rather
than decimal . Pharmacode is read from right to left : with n as the bar position starting
at 0 on the right , each narrow bar adds 2 n to the value and each wide bar adds 2 ( 2 ^ n ) .
The minimum barcode is 2 bars and the maximum 16 , so the smallest number that could
be encoded is 3 ( 2 narrow bars ) and the biggest is 131070 ( 16 wide bars ) . "
- http : //en.wikipedia.org/wiki/Pharmacode */
/* This code uses the One Track Pharamacode calculating algorithm as recommended by
the specification at http : //www.laetus.com/laetus.php?request=file&id=69 */
unsigned long int tester ;
2009-10-06 23:03:00 +04:00
int counter , error_number , h ;
char inter [ 18 ] = { 0 } ; /* 131070 -> 17 bits */
char dest [ 64 ] ; /* 17 * 2 + 1 */
2012-12-31 17:41:59 +04:00
2008-09-02 23:44:41 +04:00
error_number = 0 ;
2008-07-14 01:15:55 +04:00
2009-09-29 13:45:46 +04:00
if ( length > 6 ) {
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-02 23:44:41 +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-02 23:44:41 +04:00
return error_number ;
2008-07-14 01:15:55 +04:00
}
2008-10-03 12:22:08 +04:00
tester = atoi ( ( char * ) source ) ;
2008-07-14 01:15:55 +04:00
if ( ( tester < 3 ) | | ( tester > 131070 ) ) {
2009-06-01 00:33:54 +04:00
strcpy ( symbol - > errtxt , " Data out of range " ) ;
2008-07-14 01:15:55 +04:00
return ERROR_INVALID_DATA ;
}
do
{
2010-09-14 14:36:00 +04:00
if ( ! ( tester & 1 ) ) {
2008-07-14 01:15:55 +04:00
concat ( inter , " W " ) ;
tester = ( tester - 2 ) / 2 ;
} else {
concat ( inter , " N " ) ;
tester = ( tester - 1 ) / 2 ;
}
}
while ( tester ! = 0 ) ;
2009-10-06 23:03:00 +04:00
h = strlen ( inter ) - 1 ;
* dest = ' \0 ' ;
for ( counter = h ; counter > = 0 ; counter - - ) {
2008-07-14 01:15:55 +04:00
if ( inter [ counter ] = = ' W ' ) {
concat ( dest , " 32 " ) ;
} else {
concat ( dest , " 12 " ) ;
}
}
2012-12-31 17:41:59 +04:00
2008-07-14 01:15:55 +04:00
expand ( symbol , dest ) ;
2009-02-19 22:09:57 +03:00
2008-09-02 23:44:41 +04:00
return error_number ;
2008-07-14 01:15:55 +04:00
}
int pharma_two_calc ( struct zint_symbol * symbol , unsigned char source [ ] , char dest [ ] )
{
/* This code uses the Two Track Pharamacode defined in the document at
http : //www.laetus.com/laetus.php?request=file&id=69 and using a modified
algorithm from the One Track system . This standard accepts integet values
from 4 to 64570080. */
unsigned long int tester ;
2009-10-06 23:03:00 +04:00
int counter , h ;
2008-07-14 01:15:55 +04:00
char inter [ 17 ] ;
2008-09-02 23:44:41 +04:00
int error_number ;
2012-12-31 17:41:59 +04:00
2008-10-03 12:22:08 +04:00
tester = atoi ( ( char * ) source ) ;
2008-07-14 01:15:55 +04:00
if ( ( tester < 4 ) | | ( tester > 64570080 ) )
{
2009-06-01 00:33:54 +04:00
strcpy ( symbol - > errtxt , " Data out of range " ) ;
2008-07-14 01:15:55 +04:00
return ERROR_INVALID_DATA ;
}
2009-10-06 23:03:00 +04:00
error_number = 0 ;
strcpy ( inter , " " ) ;
2008-07-14 01:15:55 +04:00
do
{
switch ( tester % 3 ) {
case 0 :
concat ( inter , " 3 " ) ;
tester = ( tester - 3 ) / 3 ;
break ;
case 1 :
concat ( inter , " 1 " ) ;
tester = ( tester - 1 ) / 3 ;
break ;
case 2 :
concat ( inter , " 2 " ) ;
tester = ( tester - 2 ) / 3 ;
break ;
}
}
while ( tester ! = 0 ) ;
2009-10-06 23:03:00 +04:00
h = strlen ( inter ) - 1 ;
for ( counter = h ; counter > = 0 ; counter - - )
2008-07-14 01:15:55 +04:00
{
2009-10-06 23:03:00 +04:00
dest [ h - counter ] = inter [ counter ] ;
2008-07-14 01:15:55 +04:00
}
2009-10-06 23:03:00 +04:00
dest [ h + 1 ] = ' \0 ' ;
2012-12-31 17:41:59 +04:00
2008-09-02 23:44:41 +04:00
return error_number ;
2008-07-14 01:15:55 +04:00
}
2009-09-29 13:45:46 +04:00
int pharma_two ( struct zint_symbol * symbol , unsigned char source [ ] , int length )
2008-07-14 01:15:55 +04:00
{
/* Draws the patterns for two track pharmacode */
char height_pattern [ 200 ] ;
2009-10-06 23:03:00 +04:00
unsigned int loopey , h ;
2008-07-14 01:15:55 +04:00
int writer ;
2008-09-02 23:44:41 +04:00
int error_number = 0 ;
2008-07-14 01:15:55 +04:00
strcpy ( height_pattern , " " ) ;
2009-09-29 13:45:46 +04:00
if ( length > 8 ) {
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-02 23:44:41 +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-02 23:44:41 +04:00
return error_number ;
2008-07-14 01:15:55 +04:00
}
2008-09-02 23:44:41 +04:00
error_number = pharma_two_calc ( symbol , source , height_pattern ) ;
if ( error_number ! = 0 ) {
return error_number ;
2008-07-14 01:15:55 +04:00
}
writer = 0 ;
2009-10-06 23:03:00 +04:00
h = strlen ( height_pattern ) ;
for ( loopey = 0 ; loopey < h ; loopey + + )
2008-07-14 01:15:55 +04:00
{
if ( ( height_pattern [ loopey ] = = ' 2 ' ) | | ( height_pattern [ loopey ] = = ' 3 ' ) )
{
2009-06-01 00:33:54 +04:00
set_module ( symbol , 0 , writer ) ;
2008-07-14 01:15:55 +04:00
}
if ( ( height_pattern [ loopey ] = = ' 1 ' ) | | ( height_pattern [ loopey ] = = ' 3 ' ) )
{
2009-06-01 00:33:54 +04:00
set_module ( symbol , 1 , writer ) ;
2008-07-14 01:15:55 +04:00
}
writer + = 2 ;
}
symbol - > rows = 2 ;
symbol - > width = writer - 1 ;
2012-12-31 17:41:59 +04:00
2009-02-19 22:09:57 +03:00
2008-09-02 23:44:41 +04:00
return error_number ;
2008-07-14 01:15:55 +04:00
}
2009-09-29 13:45:46 +04:00
int codabar ( struct zint_symbol * symbol , unsigned char source [ ] , int length )
2008-07-14 01:15:55 +04:00
{ /* The Codabar system consisting of simple substitution */
2008-09-02 23:44:41 +04:00
int i , error_number ;
2009-10-06 23:03:00 +04:00
char dest [ 512 ] ;
2012-12-31 17:41:59 +04:00
2008-09-02 23:44:41 +04:00
error_number = 0 ;
2008-07-14 01:15:55 +04:00
strcpy ( dest , " " ) ;
2009-09-29 13:45:46 +04:00
if ( length > 60 ) { /* No stack smashing please */
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
to_upper ( source ) ;
error_number = is_sane ( CALCIUM , source , length ) ;
2008-09-02 23:44:41 +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-02 23:44:41 +04:00
return error_number ;
2008-07-14 01:15:55 +04:00
}
/* Codabar must begin and end with the characters A, B, C or D */
2009-10-06 23:03:00 +04:00
if ( ( source [ 0 ] ! = ' A ' ) & & ( source [ 0 ] ! = ' B ' ) & & ( source [ 0 ] ! = ' C ' ) & & ( source [ 0 ] ! = ' D ' ) )
2008-07-14 01:15:55 +04:00
{
2009-06-01 00:33:54 +04:00
strcpy ( symbol - > errtxt , " Invalid characters in data " ) ;
2009-09-29 13:45:46 +04:00
return ERROR_INVALID_DATA ;
2008-07-14 01:15:55 +04:00
}
2009-10-06 23:03:00 +04:00
if ( ( source [ length - 1 ] ! = ' A ' ) & & ( source [ length - 1 ] ! = ' B ' ) & &
( source [ length - 1 ] ! = ' C ' ) & & ( source [ length - 1 ] ! = ' D ' ) )
2008-07-14 01:15:55 +04:00
{
2009-06-01 00:33:54 +04:00
strcpy ( symbol - > errtxt , " Invalid characters in data " ) ;
2009-09-29 13:45:46 +04:00
return ERROR_INVALID_DATA ;
2008-07-14 01:15:55 +04:00
}
2009-09-29 13:45:46 +04:00
for ( i = 0 ; i < length ; i + + )
2008-07-14 01:15:55 +04:00
{
2009-10-06 23:03:00 +04:00
lookup ( CALCIUM , CodaTable , source [ i ] , dest ) ;
2008-07-14 01:15:55 +04:00
}
2012-12-31 17:41:59 +04:00
2008-07-14 01:15:55 +04:00
expand ( symbol , dest ) ;
2009-10-06 23:03:00 +04:00
ustrcpy ( symbol - > text , source ) ;
2008-09-02 23:44:41 +04:00
return error_number ;
}
2009-09-29 13:45:46 +04:00
int code32 ( struct zint_symbol * symbol , unsigned char source [ ] , int length )
2008-09-02 23:44:41 +04:00
{ /* Italian Pharmacode */
int i , zeroes , error_number , checksum , checkpart , checkdigit ;
2009-10-06 23:03:00 +04:00
char localstr [ 10 ] , risultante [ 7 ] ;
2008-09-02 23:44:41 +04:00
long int pharmacode , remainder , devisor ;
int codeword [ 6 ] ;
char tabella [ 34 ] ;
2012-12-31 17:41:59 +04:00
2008-09-02 23:44:41 +04:00
/* Validate the input */
2009-09-29 13:45:46 +04:00
if ( length > 8 ) {
2009-06-01 00:33:54 +04:00
strcpy ( symbol - > errtxt , " Input too long " ) ;
2008-09-02 23:44:41 +04:00
return ERROR_TOO_LONG ;
}
2009-10-06 23:03:00 +04:00
error_number = is_sane ( NEON , source , length ) ;
2008-09-02 23:44:41 +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-02 23:44:41 +04:00
return error_number ;
}
2012-12-31 17:41:59 +04:00
2008-09-02 23:44:41 +04:00
/* Add leading zeros as required */
2009-09-29 13:45:46 +04:00
zeroes = 8 - length ;
2009-10-06 23:03:00 +04:00
memset ( localstr , ' 0 ' , zeroes ) ;
strcpy ( localstr + zeroes , ( char * ) source ) ;
2012-12-31 17:41:59 +04:00
2008-09-02 23:44:41 +04:00
/* Calculate the check digit */
checksum = 0 ;
checkpart = 0 ;
for ( i = 0 ; i < 4 ; i + + ) {
checkpart = ctoi ( localstr [ i * 2 ] ) ;
checksum + = checkpart ;
checkpart = 2 * ( ctoi ( localstr [ ( i * 2 ) + 1 ] ) ) ;
if ( checkpart > = 10 ) {
checksum + = ( checkpart - 10 ) + 1 ;
} else {
checksum + = checkpart ;
}
}
2012-12-31 17:41:59 +04:00
2008-09-02 23:44:41 +04:00
/* Add check digit to data string */
checkdigit = checksum % 10 ;
2009-10-06 23:03:00 +04:00
localstr [ 8 ] = itoc ( checkdigit ) ;
localstr [ 9 ] = ' \0 ' ;
2012-12-31 17:41:59 +04:00
2008-09-02 23:44:41 +04:00
/* Convert string into an integer value */
2009-10-06 23:03:00 +04:00
pharmacode = atoi ( localstr ) ;
2008-09-02 23:44:41 +04:00
/* Convert from decimal to base-32 */
devisor = 33554432 ;
for ( i = 5 ; i > = 0 ; i - - ) {
codeword [ i ] = pharmacode / devisor ;
remainder = pharmacode % devisor ;
pharmacode = remainder ;
devisor / = 32 ;
}
2012-12-31 17:41:59 +04:00
2008-09-02 23:44:41 +04:00
/* Look up values in 'Tabella di conversione' */
strcpy ( tabella , " 0123456789BCDFGHJKLMNPQRSTUVWXYZ " ) ;
for ( i = 5 ; i > = 0 ; i - - ) {
2009-10-06 23:03:00 +04:00
risultante [ 5 - i ] = tabella [ codeword [ i ] ] ;
2008-09-02 23:44:41 +04:00
}
2009-10-06 23:03:00 +04:00
risultante [ 6 ] = ' \0 ' ;
2008-09-02 23:44:41 +04:00
/* Plot the barcode using Code 39 */
2009-09-29 13:45:46 +04:00
error_number = c39 ( symbol , ( unsigned char * ) risultante , strlen ( risultante ) ) ;
2008-09-02 23:44:41 +04:00
if ( error_number ! = 0 ) { return error_number ; }
2012-12-31 17:41:59 +04:00
2008-09-02 23:44:41 +04:00
/* Override the normal text output with the Pharmacode number */
2009-02-19 22:09:57 +03:00
ustrcpy ( symbol - > text , ( unsigned char * ) " A " ) ;
uconcat ( symbol - > text , ( unsigned char * ) localstr ) ;
2012-12-31 17:41:59 +04:00
2008-09-02 23:44:41 +04:00
return error_number ;
2008-07-14 01:15:55 +04:00
}