2008-07-14 09:15:55 +12:00
|
|
|
/* auspost.c - Handles Australia Post 4-State Barcode */
|
|
|
|
|
|
|
|
/*
|
|
|
|
libzint - the open source barcode library
|
2008-11-17 21:47:42 +13:00
|
|
|
Copyright (C) 2008 Robin Stuart <robin@zint.org.uk>
|
2008-07-14 09:15:55 +12: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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#define GDSET "0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz #"
|
|
|
|
|
|
|
|
static char *AusNTable[10] = {"00", "01", "02", "10", "11", "12", "20", "21", "22", "30"};
|
|
|
|
|
|
|
|
static char *AusCTable[64] = {"222", "300", "301", "302", "310", "311", "312", "320", "321", "322",
|
|
|
|
"000", "001", "002", "010", "011", "012", "020", "021", "022", "100", "101", "102", "110",
|
|
|
|
"111", "112", "120", "121", "122", "200", "201", "202", "210", "211", "212", "220", "221",
|
|
|
|
"023", "030", "031", "032", "033", "103", "113", "123", "130", "131", "132", "133", "203",
|
|
|
|
"213", "223", "230", "231", "232", "233", "303", "313", "323", "330", "331", "332", "333",
|
|
|
|
"003", "013"};
|
|
|
|
|
|
|
|
static char *AusBarTable[64] = {"000", "001", "002", "003", "010", "011", "012", "013", "020", "021",
|
|
|
|
"022", "023", "030", "031", "032", "033", "100", "101", "102", "103", "110", "111", "112",
|
|
|
|
"113", "120", "121", "122", "123", "130", "131", "132", "133", "200", "201", "202", "203",
|
|
|
|
"210", "211", "212", "213", "220", "221", "222", "223", "230", "231", "232", "233", "300",
|
|
|
|
"301", "302", "303", "310", "311", "312", "313", "320", "321", "322", "323", "330", "331",
|
|
|
|
"332", "333"};
|
|
|
|
|
|
|
|
#include <string.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include "common.h"
|
|
|
|
#include "reedsol.h"
|
|
|
|
|
|
|
|
void rs_error(char data_pattern[])
|
|
|
|
{
|
|
|
|
/* Adds Reed-Solomon error correction to auspost */
|
|
|
|
|
|
|
|
int reader, triple_writer;
|
|
|
|
char triple[31], inv_triple[31];
|
2008-10-03 22:26:27 +13:00
|
|
|
unsigned char result[5];
|
2008-07-14 09:15:55 +12:00
|
|
|
|
|
|
|
triple_writer = 0;
|
|
|
|
|
|
|
|
for(reader = 2; reader < strlen(data_pattern); reader+= 3)
|
|
|
|
{
|
|
|
|
triple[triple_writer] = 0;
|
|
|
|
switch(data_pattern[reader])
|
|
|
|
{
|
|
|
|
case '1': triple[triple_writer] += 16; break;
|
|
|
|
case '2': triple[triple_writer] += 32; break;
|
|
|
|
case '3': triple[triple_writer] += 48; break;
|
|
|
|
}
|
|
|
|
switch(data_pattern[reader + 1])
|
|
|
|
{
|
|
|
|
case '1': triple[triple_writer] += 4; break;
|
|
|
|
case '2': triple[triple_writer] += 8; break;
|
|
|
|
case '3': triple[triple_writer] += 12; break;
|
|
|
|
}
|
|
|
|
switch(data_pattern[reader + 2])
|
|
|
|
{
|
|
|
|
case '1': triple[triple_writer] += 1; break;
|
|
|
|
case '2': triple[triple_writer] += 2; break;
|
|
|
|
case '3': triple[triple_writer] += 3; break;
|
|
|
|
}
|
|
|
|
triple_writer++;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
for(reader = 0; reader < triple_writer; reader++)
|
|
|
|
{
|
|
|
|
inv_triple[reader] = triple[(triple_writer - 1) - reader];
|
|
|
|
}
|
|
|
|
|
|
|
|
rs_init_gf(0x43);
|
|
|
|
rs_init_code(4, 1);
|
2008-10-03 22:26:27 +13:00
|
|
|
rs_encode(triple_writer, (unsigned char*) inv_triple, result);
|
2008-07-14 09:15:55 +12:00
|
|
|
|
|
|
|
for(reader = 4; reader > 0; reader--)
|
|
|
|
{
|
2008-10-05 18:51:58 +13:00
|
|
|
concat(data_pattern, AusBarTable[(int)result[reader - 1]]);
|
2008-07-14 09:15:55 +12:00
|
|
|
}
|
2008-10-13 07:52:54 +13:00
|
|
|
rs_free();
|
2008-07-14 09:15:55 +12:00
|
|
|
}
|
|
|
|
|
2009-09-29 22:45:46 +13:00
|
|
|
int australia_post(struct zint_symbol *symbol, unsigned char source[], int length)
|
2008-07-14 09:15:55 +12:00
|
|
|
{
|
|
|
|
/* Handles Australia Posts's 4 State Codes */
|
|
|
|
/* Customer Standard Barcode, Barcode 2 or Barcode 3 system determined automatically
|
|
|
|
(i.e. the FCC doesn't need to be specified by the user) dependent
|
|
|
|
on the length of the input string */
|
|
|
|
|
|
|
|
/* The contents of data_pattern conform to the following standard:
|
|
|
|
0 = Tracker, Ascender and Descender
|
|
|
|
1 = Tracker and Ascender
|
|
|
|
2 = Tracker and Descender
|
|
|
|
3 = Tracker only */
|
2009-08-07 06:48:42 +12:00
|
|
|
int error_number, zeroes;
|
|
|
|
int writer, i;
|
|
|
|
unsigned int loopey, reader;
|
2008-07-14 09:15:55 +12:00
|
|
|
|
|
|
|
char data_pattern[200];
|
|
|
|
char fcc[3], dpid[10];
|
2008-10-13 10:05:53 +13:00
|
|
|
char localstr[30];
|
2008-07-14 09:15:55 +12:00
|
|
|
|
2008-12-25 10:26:38 +13:00
|
|
|
error_number = 0;
|
2009-08-07 06:48:42 +12:00
|
|
|
strcpy (data_pattern, "");
|
|
|
|
strcpy(localstr, "");
|
2008-07-14 09:15:55 +12:00
|
|
|
|
|
|
|
/* Do all of the length checking first to avoid stack smashing */
|
|
|
|
if(symbol->symbology == BARCODE_AUSPOST) {
|
|
|
|
/* Format control code (FCC) */
|
2009-09-29 22:45:46 +13:00
|
|
|
switch(length)
|
2008-07-14 09:15:55 +12:00
|
|
|
{
|
|
|
|
case 8: strcpy(fcc, "11"); break;
|
|
|
|
case 13: strcpy(fcc, "59"); break;
|
2009-09-29 22:45:46 +13:00
|
|
|
case 16: strcpy(fcc, "59"); error_number = is_sane(NESET, source, length); break;
|
2008-07-14 09:15:55 +12:00
|
|
|
case 18: strcpy(fcc, "62"); break;
|
2009-09-29 22:45:46 +13:00
|
|
|
case 23: strcpy(fcc, "62"); error_number = is_sane(NESET, source, length); break;
|
2009-06-01 08:33:54 +12:00
|
|
|
default: strcpy(symbol->errtxt, "Auspost input is wrong length");
|
2008-07-14 09:15:55 +12:00
|
|
|
return ERROR_TOO_LONG;
|
|
|
|
break;
|
|
|
|
}
|
2008-12-25 10:26:38 +13:00
|
|
|
if(error_number == ERROR_INVALID_DATA) {
|
2009-06-01 08:33:54 +12:00
|
|
|
strcpy(symbol->errtxt, "Invalid characters in data");
|
2008-12-25 10:26:38 +13:00
|
|
|
return error_number;
|
2008-07-14 09:15:55 +12:00
|
|
|
}
|
|
|
|
} else {
|
2008-10-13 10:05:53 +13:00
|
|
|
if(ustrlen(source) > 8) {
|
2009-06-01 08:33:54 +12:00
|
|
|
strcpy(symbol->errtxt, "Auspost input is too long");
|
2008-07-14 09:15:55 +12:00
|
|
|
return ERROR_TOO_LONG;
|
|
|
|
}
|
|
|
|
switch(symbol->symbology) {
|
|
|
|
case BARCODE_AUSREPLY: strcpy(fcc, "45"); break;
|
|
|
|
case BARCODE_AUSROUTE: strcpy(fcc, "87"); break;
|
|
|
|
case BARCODE_AUSREDIRECT: strcpy(fcc, "92"); break;
|
|
|
|
}
|
2008-10-13 10:05:53 +13:00
|
|
|
|
|
|
|
/* Add leading zeros as required */
|
2009-09-29 22:45:46 +13:00
|
|
|
zeroes = 8 - length;
|
2008-10-13 10:05:53 +13:00
|
|
|
for(i = 0; i < zeroes; i++) {
|
|
|
|
concat(localstr, "0");
|
|
|
|
}
|
2008-07-14 09:15:55 +12:00
|
|
|
}
|
|
|
|
|
2008-10-13 10:05:53 +13:00
|
|
|
concat(localstr, (char*)source);
|
2009-09-29 22:45:46 +13:00
|
|
|
error_number = is_sane(GDSET, (unsigned char *)localstr, length);
|
2008-12-25 10:26:38 +13:00
|
|
|
if(error_number == ERROR_INVALID_DATA) {
|
2009-06-01 08:33:54 +12:00
|
|
|
strcpy(symbol->errtxt, "Invalid characters in data");
|
2008-12-25 10:26:38 +13:00
|
|
|
return error_number;
|
2008-07-14 09:15:55 +12:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Verifiy that the first 8 characters are numbers */
|
|
|
|
for(loopey = 0; loopey < 8; loopey++) {
|
2008-10-13 10:05:53 +13:00
|
|
|
dpid[loopey] = localstr[loopey];
|
2008-07-14 09:15:55 +12:00
|
|
|
}
|
|
|
|
dpid[8] = '\0';
|
2009-09-29 22:45:46 +13:00
|
|
|
error_number = is_sane(NESET, (unsigned char *)dpid, 8);
|
2008-12-25 10:26:38 +13:00
|
|
|
if(error_number == ERROR_INVALID_DATA) {
|
2009-06-01 08:33:54 +12:00
|
|
|
strcpy(symbol->errtxt, "Invalid characters in DPID");
|
2008-12-25 10:26:38 +13:00
|
|
|
return error_number;
|
2008-07-14 09:15:55 +12:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Start character */
|
|
|
|
concat(data_pattern, "13");
|
|
|
|
|
|
|
|
/* Encode the FCC */
|
|
|
|
for(reader = 0; reader < 2; reader++)
|
|
|
|
{
|
|
|
|
lookup(NESET, AusNTable, fcc[reader], data_pattern);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* printf("AUSPOST FCC: %s ", fcc); */
|
|
|
|
|
|
|
|
/* Delivery Point Identifier (DPID) */
|
|
|
|
for(reader = 0; reader < 8; reader++)
|
|
|
|
{
|
|
|
|
lookup(NESET, AusNTable, dpid[reader], data_pattern);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Customer Information */
|
2008-10-13 10:05:53 +13:00
|
|
|
if(strlen(localstr) > 8)
|
2008-07-14 09:15:55 +12:00
|
|
|
{
|
2008-10-13 10:05:53 +13:00
|
|
|
if((strlen(localstr) == 13) || (strlen(localstr) == 18)) {
|
|
|
|
for(reader = 8; reader < strlen(localstr); reader++) {
|
|
|
|
lookup(GDSET, AusCTable, localstr[reader], data_pattern);
|
2008-07-14 09:15:55 +12:00
|
|
|
}
|
|
|
|
}
|
2008-10-13 10:05:53 +13:00
|
|
|
if((strlen(localstr) == 16) || (strlen(localstr) == 23)) {
|
|
|
|
for(reader = 8; reader < strlen(localstr); reader++) {
|
|
|
|
lookup(NESET, AusNTable, localstr[reader], data_pattern);
|
2008-07-14 09:15:55 +12:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Filler bar */
|
|
|
|
if(strlen(data_pattern) == 22) {
|
|
|
|
concat(data_pattern, "3");
|
|
|
|
}
|
|
|
|
if(strlen(data_pattern) == 37) {
|
|
|
|
concat(data_pattern, "3");
|
|
|
|
}
|
|
|
|
if(strlen(data_pattern) == 52) {
|
|
|
|
concat(data_pattern, "3");
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Reed Solomon error correction */
|
|
|
|
rs_error(data_pattern);
|
|
|
|
|
|
|
|
/* Stop character */
|
|
|
|
concat(data_pattern, "13");
|
|
|
|
|
|
|
|
/* Turn the symbol into a bar pattern ready for plotting */
|
|
|
|
writer = 0;
|
|
|
|
for(loopey = 0; loopey < strlen(data_pattern); loopey++)
|
|
|
|
{
|
|
|
|
if((data_pattern[loopey] == '1') || (data_pattern[loopey] == '0'))
|
|
|
|
{
|
2009-06-01 08:33:54 +12:00
|
|
|
set_module(symbol, 0, writer);
|
2008-07-14 09:15:55 +12:00
|
|
|
}
|
2009-06-01 08:33:54 +12:00
|
|
|
set_module(symbol, 1, writer);
|
2008-07-14 09:15:55 +12:00
|
|
|
if((data_pattern[loopey] == '2') || (data_pattern[loopey] == '0'))
|
|
|
|
{
|
2009-06-01 08:33:54 +12:00
|
|
|
set_module(symbol, 2, writer);
|
2008-07-14 09:15:55 +12:00
|
|
|
}
|
|
|
|
writer += 2;
|
|
|
|
}
|
|
|
|
|
|
|
|
symbol->row_height[0] = 4;
|
|
|
|
symbol->row_height[1] = 2;
|
|
|
|
symbol->row_height[2] = 4;
|
|
|
|
|
|
|
|
symbol->rows = 3;
|
|
|
|
symbol->width = writer - 1;
|
|
|
|
|
2008-12-25 10:26:38 +13:00
|
|
|
return error_number;
|
2008-07-14 09:15:55 +12:00
|
|
|
}
|
|
|
|
|