2016-04-07 19:13:43 +03:00
/* hanxin.c - Han Xin Code
libzint - the open source barcode library
Copyright ( C ) 2009 - 2016 Robin Stuart < rstuart114 @ gmail . com >
Redistribution and use in source and binary forms , with or without
modification , are permitted provided that the following conditions
are met :
1. Redistributions of source code must retain the above copyright
notice , this list of conditions and the following disclaimer .
2. Redistributions in binary form must reproduce the above copyright
notice , this list of conditions and the following disclaimer in the
documentation and / or other materials provided with the distribution .
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
without specific prior written permission .
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
OUT OF THE USE OF THIS SOFTWARE , EVEN IF ADVISED OF THE POSSIBILITY OF
SUCH DAMAGE .
*/
/* This code attempts to implement Han Xin Code according to AIMD-015:2010 (Rev 0.8) */
# include <stdio.h>
# include <string.h>
# include <stdlib.h>
# ifdef _MSC_VER
# include <malloc.h>
# endif
# include "common.h"
# include "reedsol.h"
# include "hanxin.h"
2016-05-02 00:10:50 +03:00
# include "gb18030.h"
2016-04-07 19:13:43 +03:00
2016-04-09 19:01:21 +03:00
/* Find which submode to use for a text character */
int getsubmode ( char input ) {
int submode = 2 ;
if ( ( input > = ' 0 ' ) & & ( input < = ' 9 ' ) ) {
submode = 1 ;
}
if ( ( input > = ' A ' ) & & ( input < = ' Z ' ) ) {
submode = 1 ;
}
if ( ( input > = ' a ' ) & & ( input < = ' z ' ) ) {
submode = 1 ;
}
return submode ;
}
2016-04-08 00:30:37 +03:00
/* Calculate the approximate length of the binary string */
2016-05-02 00:10:50 +03:00
int calculate_binlength ( char mode [ ] , int source [ ] , int length ) {
2016-04-08 00:30:37 +03:00
int i ;
2016-04-10 22:36:57 +03:00
char lastmode = ' t ' ;
2016-04-08 00:30:37 +03:00
int est_binlen = 0 ;
2016-04-09 19:01:21 +03:00
int submode = 1 ;
2016-04-08 00:30:37 +03:00
2016-05-02 00:10:50 +03:00
i = 0 ;
do {
2016-04-08 00:30:37 +03:00
switch ( mode [ i ] ) {
case ' n ' :
if ( lastmode ! = ' n ' ) {
est_binlen + = 14 ;
lastmode = ' n ' ;
}
est_binlen + = 4 ;
break ;
case ' t ' :
if ( lastmode ! = ' t ' ) {
est_binlen + = 10 ;
lastmode = ' t ' ;
2016-04-09 19:01:21 +03:00
submode = 1 ;
}
if ( getsubmode ( ( char ) source [ i ] ) ! = submode ) {
est_binlen + = 6 ;
submode = getsubmode ( ( char ) source [ i ] ) ;
2016-04-08 00:30:37 +03:00
}
est_binlen + = 6 ;
2016-04-10 22:36:57 +03:00
break ;
2016-04-08 00:30:37 +03:00
case ' b ' :
if ( lastmode ! = ' b ' ) {
est_binlen + = 17 ;
lastmode = ' b ' ;
}
2016-04-10 22:36:57 +03:00
est_binlen + = 8 ;
break ;
2016-05-02 00:10:50 +03:00
case ' 1 ' :
if ( lastmode ! = ' 1 ' ) {
est_binlen + = 16 ;
lastmode = ' 1 ' ;
}
est_binlen + = 12 ;
break ;
case ' 2 ' :
if ( lastmode ! = ' 2 ' ) {
est_binlen + = 16 ;
lastmode = ' 2 ' ;
}
est_binlen + = 12 ;
break ;
case ' d ' :
if ( lastmode ! = ' d ' ) {
est_binlen + = 16 ;
lastmode = ' d ' ;
}
est_binlen + = 15 ;
break ;
case ' f ' :
if ( lastmode ! = ' f ' ) {
est_binlen + = 4 ;
lastmode = ' f ' ;
}
est_binlen + = 21 ;
i + + ;
break ;
2016-04-08 00:30:37 +03:00
}
2016-05-02 00:10:50 +03:00
i + + ;
} while ( i < length ) ;
2016-04-08 00:30:37 +03:00
return est_binlen ;
}
2016-05-02 00:10:50 +03:00
int isRegion1 ( int glyph ) {
int first_byte , second_byte ;
int valid = 0 ;
first_byte = ( glyph & 0xff00 ) > > 8 ;
second_byte = glyph & 0xff ;
if ( ( first_byte > = 0xb0 ) & & ( first_byte < = 0xd7 ) ) {
if ( ( second_byte > = 0xa1 ) & & ( second_byte < = 0xfe ) ) {
valid = 1 ;
}
}
if ( ( first_byte > = 0xa1 ) & & ( first_byte < = 0xa3 ) ) {
if ( ( second_byte > = 0xa1 ) & & ( second_byte < = 0xfe ) ) {
valid = 1 ;
}
}
if ( ( glyph > = 0xa8a1 ) & & ( glyph < = 0xa8c0 ) ) {
valid = 1 ;
}
return valid ;
}
int isRegion2 ( int glyph ) {
int first_byte , second_byte ;
int valid = 0 ;
first_byte = ( glyph & 0xff00 ) > > 8 ;
second_byte = glyph & 0xff ;
if ( ( first_byte > = 0xd8 ) & & ( first_byte < = 0xf7 ) ) {
if ( ( second_byte > = 0xa1 ) & & ( second_byte < = 0xfe ) ) {
valid = 1 ;
}
}
return valid ;
}
int isDoubleByte ( int glyph ) {
int first_byte , second_byte ;
int valid = 0 ;
first_byte = ( glyph & 0xff00 ) > > 8 ;
second_byte = glyph & 0xff ;
if ( ( first_byte > = 0x81 ) & & ( first_byte < = 0xfe ) ) {
if ( ( second_byte > = 0x40 ) & & ( second_byte < = 0x7e ) ) {
valid = 1 ;
}
if ( ( second_byte > = 0x80 ) & & ( second_byte < = 0xfe ) ) {
valid = 1 ;
}
}
return valid ;
}
int isFourByte ( int glyph , int glyph2 ) {
int first_byte , second_byte ;
int third_byte , fourth_byte ;
int valid = 0 ;
first_byte = ( glyph & 0xff00 ) > > 8 ;
second_byte = glyph & 0xff ;
third_byte = ( glyph2 & 0xff00 ) > > 8 ;
fourth_byte = glyph2 & 0xff ;
if ( ( first_byte > = 0x81 ) & & ( first_byte < = 0xfe ) ) {
if ( ( second_byte > = 0x30 ) & & ( second_byte < = 0x39 ) ) {
if ( ( third_byte > = 0x81 ) & & ( third_byte < = 0xfe ) ) {
if ( ( fourth_byte > = 0x30 ) & & ( fourth_byte < = 0x39 ) ) {
valid = 1 ;
}
}
}
}
return valid ;
}
2016-04-08 00:30:37 +03:00
/* Calculate mode switching */
2016-05-02 00:10:50 +03:00
void hx_define_mode ( char mode [ ] , int source [ ] , int length ) {
2016-04-08 00:30:37 +03:00
int i ;
char lastmode = ' t ' ;
2016-05-02 00:10:50 +03:00
int done ;
2016-04-08 00:30:37 +03:00
2016-05-02 00:10:50 +03:00
i = 0 ;
do {
done = 0 ;
if ( isRegion1 ( source [ i ] ) ) {
mode [ i ] = ' 1 ' ;
done = 1 ;
i + + ;
}
if ( ( done = = 0 ) & & ( isRegion2 ( source [ i ] ) ) ) {
mode [ i ] = ' 2 ' ;
done = 1 ;
i + + ;
}
if ( ( done = = 0 ) & & ( isDoubleByte ( source [ i ] ) ) ) {
mode [ i ] = ' d ' ;
done = 1 ;
i + + ;
}
if ( ( done = = 0 ) & & ( i < length - 1 ) ) {
if ( isFourByte ( source [ i ] , source [ i + 1 ] ) ) {
mode [ i ] = ' f ' ;
mode [ i + 1 ] = ' f ' ;
done = 1 ;
i + = 2 ;
2016-04-08 00:30:37 +03:00
}
2016-05-02 00:10:50 +03:00
}
if ( done = = 0 ) {
if ( ( source [ i ] > = ' 0 ' ) & & ( source [ i ] < = ' 9 ' ) ) {
mode [ i ] = ' n ' ;
if ( lastmode ! = ' n ' ) {
lastmode = ' n ' ;
2016-04-08 00:30:37 +03:00
}
} else {
2016-05-02 00:10:50 +03:00
if ( ( source [ i ] < = 127 ) & & ( ( source [ i ] < = 27 ) | | ( source [ i ] > = 32 ) ) ) {
mode [ i ] = ' t ' ;
if ( lastmode ! = ' t ' ) {
lastmode = ' t ' ;
}
} else {
mode [ i ] = ' b ' ;
if ( lastmode ! = ' b ' ) {
lastmode = ' b ' ;
}
2016-04-08 00:30:37 +03:00
}
}
2016-05-02 00:10:50 +03:00
i + + ;
2016-04-08 00:30:37 +03:00
}
2016-05-02 00:10:50 +03:00
} while ( i < length ) ;
2016-04-08 00:30:37 +03:00
mode [ length ] = ' \0 ' ;
}
/* Convert Text 1 sub-mode character to encoding value, as given in table 3 */
int lookup_text1 ( char input ) {
int encoding_value = 0 ;
if ( ( input > = ' 0 ' ) & & ( input < = ' 9 ' ) ) {
encoding_value = input - ' 0 ' ;
}
if ( ( input > = ' A ' ) & & ( input < = ' Z ' ) ) {
encoding_value = input - ' A ' + 10 ;
}
if ( ( input > = ' a ' ) & & ( input < = ' z ' ) ) {
encoding_value = input - ' a ' + 36 ;
}
return encoding_value ;
}
/* Convert Text 2 sub-mode character to encoding value, as given in table 4 */
int lookup_text2 ( char input ) {
int encoding_value = 0 ;
if ( ( input > = 0 ) & & ( input < = 27 ) ) {
encoding_value = input ;
}
if ( ( input > = ' ' ) & & ( input < = ' / ' ) ) {
encoding_value = input - ' ' + 28 ;
}
if ( ( input > = ' [ ' ) & & ( input < = 96 ) ) {
encoding_value = input - ' [ ' + 51 ;
}
if ( ( input > = ' { ' ) & & ( input < = 127 ) ) {
encoding_value = input - ' { ' + 57 ;
}
return encoding_value ;
}
/* Convert input data to binary stream */
2016-05-02 00:10:50 +03:00
void calculate_binary ( char binary [ ] , char mode [ ] , int source [ ] , int length ) {
2016-04-08 00:30:37 +03:00
int block_length ;
int position = 0 ;
int i , p , count , encoding_value ;
2016-04-23 17:26:51 +03:00
int debug = 0 ;
2016-05-02 00:10:50 +03:00
int first_byte , second_byte ;
int third_byte , fourth_byte ;
int glyph ;
2016-04-08 00:30:37 +03:00
do {
block_length = 0 ;
do {
block_length + + ;
} while ( mode [ position + block_length ] = = mode [ position ] ) ;
switch ( mode [ position ] ) {
case ' n ' :
/* Numeric mode */
/* Mode indicator */
strcat ( binary , " 0001 " ) ;
if ( debug ) {
printf ( " Numeric \n " ) ;
}
i = 0 ;
while ( i < block_length ) {
int first = 0 , second = 0 , third = 0 ;
first = posn ( NEON , ( char ) source [ position + i ] ) ;
count = 1 ;
encoding_value = first ;
if ( i + 1 < block_length & & mode [ position + i + 1 ] = = ' n ' ) {
second = posn ( NEON , ( char ) source [ position + i + 1 ] ) ;
count = 2 ;
encoding_value = ( encoding_value * 10 ) + second ;
if ( i + 2 < block_length & & mode [ position + i + 2 ] = = ' n ' ) {
third = posn ( NEON , ( char ) source [ position + i + 2 ] ) ;
count = 3 ;
encoding_value = ( encoding_value * 10 ) + third ;
}
}
for ( p = 0 ; p < 10 ; p + + ) {
if ( encoding_value & ( 0x200 > > p ) ) {
strcat ( binary , " 1 " ) ;
} else {
strcat ( binary , " 0 " ) ;
}
}
if ( debug ) {
2016-06-19 11:07:50 +03:00
printf ( " 0x%4x (%d) " , encoding_value , encoding_value ) ;
2016-04-08 00:30:37 +03:00
}
i + = count ;
}
/* Mode terminator depends on number of characters in last group (Table 2) */
switch ( count ) {
case 1 :
strcat ( binary , " 1111111101 " ) ;
break ;
case 2 :
strcat ( binary , " 1111111110 " ) ;
break ;
case 3 :
strcat ( binary , " 1111111111 " ) ;
break ;
}
if ( debug ) {
printf ( " (TERM %d) \n " , count ) ;
}
break ;
case ' t ' :
/* Text mode */
if ( position ! = 0 ) {
/* Mode indicator */
strcat ( binary , " 0010 " ) ;
if ( debug ) {
printf ( " Text \n " ) ;
}
}
int submode = 1 ;
i = 0 ;
while ( i < block_length ) {
2016-05-02 00:10:50 +03:00
if ( getsubmode ( ( char ) source [ i + position ] ) ! = submode ) {
2016-04-08 00:30:37 +03:00
/* Change submode */
strcat ( binary , " 111110 " ) ;
2016-05-02 00:10:50 +03:00
submode = getsubmode ( ( char ) source [ i + position ] ) ;
2016-04-08 00:30:37 +03:00
if ( debug ) {
printf ( " SWITCH " ) ;
}
}
if ( submode = = 1 ) {
encoding_value = lookup_text1 ( ( char ) source [ i + position ] ) ;
} else {
encoding_value = lookup_text2 ( ( char ) source [ i + position ] ) ;
}
for ( p = 0 ; p < 6 ; p + + ) {
if ( encoding_value & ( 0x20 > > p ) ) {
strcat ( binary , " 1 " ) ;
} else {
strcat ( binary , " 0 " ) ;
}
}
if ( debug ) {
printf ( " %c (%d) " , ( char ) source [ i ] , encoding_value ) ;
}
i + + ;
}
/* Terminator */
strcat ( binary , " 111111 " ) ;
if ( debug ) {
printf ( " \n " ) ;
}
break ;
case ' b ' :
/* Binary Mode */
/* Mode indicator */
strcat ( binary , " 0011 " ) ;
/* Count indicator */
for ( p = 0 ; p < 13 ; p + + ) {
if ( block_length & ( 0x1000 > > p ) ) {
strcat ( binary , " 1 " ) ;
} else {
strcat ( binary , " 0 " ) ;
}
}
if ( debug ) {
printf ( " Binary (length %d) \n " , block_length ) ;
}
i = 0 ;
while ( i < block_length ) {
/* 8-bit bytes with no conversion */
for ( p = 0 ; p < 8 ; p + + ) {
if ( source [ i + position ] & ( 0x80 > > p ) ) {
strcat ( binary , " 1 " ) ;
} else {
strcat ( binary , " 0 " ) ;
}
}
if ( debug ) {
printf ( " %d " , source [ i + position ] ) ;
}
i + + ;
}
if ( debug ) {
printf ( " \n " ) ;
}
break ;
2016-05-02 00:10:50 +03:00
case ' 1 ' :
/* Region 1 encoding */
/* Mode indicator */
strcat ( binary , " 0100 " ) ;
if ( debug ) {
printf ( " Region 1 \n " ) ;
}
i = 0 ;
while ( i < block_length ) {
first_byte = ( source [ i + position ] & 0xff00 ) > > 8 ;
second_byte = source [ i + position ] & 0xff ;
/* Subset 1 */
glyph = ( 0x5e * ( first_byte - 0xb0 ) ) + ( second_byte - 0xa1 ) ;
/* Subset 2 */
if ( ( first_byte > = 0xa1 ) & & ( first_byte < = 0xa3 ) ) {
if ( ( second_byte > = 0xa1 ) & & ( second_byte < = 0xfe ) ) {
glyph = ( 0x5e * first_byte - 0xa1 ) + ( second_byte - 0xa1 ) + 0xeb0 ;
}
}
/* Subset 3 */
if ( ( source [ i + position ] > = 0xa8a1 ) & & ( source [ i + position ] < = 0xa8c0 ) ) {
glyph = ( second_byte - 0xa1 ) + 0xfca ;
}
if ( debug ) {
printf ( " %d " , glyph ) ;
}
for ( p = 0 ; p < 12 ; p + + ) {
if ( glyph & ( 0x800 > > p ) ) {
strcat ( binary , " 1 " ) ;
} else {
strcat ( binary , " 0 " ) ;
}
}
i + + ;
}
/* Terminator */
strcat ( binary , " 111111111111 " ) ;
if ( debug ) {
printf ( " \n " ) ;
}
break ;
case ' 2 ' :
/* Region 2 encoding */
/* Mode indicator */
strcat ( binary , " 0101 " ) ;
if ( debug ) {
printf ( " Region 2 \n " ) ;
}
i = 0 ;
while ( i < block_length ) {
first_byte = ( source [ i + position ] & 0xff00 ) > > 8 ;
second_byte = source [ i + position ] & 0xff ;
glyph = ( 0x5e * ( first_byte - 0xd8 ) ) + ( second_byte - 0xa1 ) ;
if ( debug ) {
printf ( " %d " , glyph ) ;
}
for ( p = 0 ; p < 12 ; p + + ) {
if ( glyph & ( 0x800 > > p ) ) {
strcat ( binary , " 1 " ) ;
} else {
strcat ( binary , " 0 " ) ;
}
}
i + + ;
}
/* Terminator */
strcat ( binary , " 111111111111 " ) ;
if ( debug ) {
printf ( " \n " ) ;
}
break ;
case ' d ' :
/* Double byte encoding */
/* Mode indicator */
strcat ( binary , " 0110 " ) ;
if ( debug ) {
printf ( " Double byte \n " ) ;
}
i = 0 ;
while ( i < block_length ) {
first_byte = ( source [ i + position ] & 0xff00 ) > > 8 ;
second_byte = source [ i + position ] & 0xff ;
if ( second_byte < = 0x7e ) {
glyph = ( 0xbe * ( first_byte - 0x81 ) ) + ( second_byte - 0x40 ) ;
} else {
glyph = ( 0xbe * ( first_byte - 0x81 ) ) + ( second_byte - 0x41 ) ;
}
if ( debug ) {
printf ( " %d " , glyph ) ;
}
for ( p = 0 ; p < 15 ; p + + ) {
if ( glyph & ( 0x4000 > > p ) ) {
strcat ( binary , " 1 " ) ;
} else {
strcat ( binary , " 0 " ) ;
}
}
i + + ;
}
/* Terminator */
2016-06-19 11:07:50 +03:00
strcat ( binary , " 111111111111111 " ) ;
/* Terminator sequence of length 12 is assumed to be a mistake */
2016-05-02 00:10:50 +03:00
if ( debug ) {
printf ( " \n " ) ;
}
break ;
case ' f ' :
/* Four-byte encoding */
if ( debug ) {
printf ( " Four byte \n " ) ;
}
i = 0 ;
while ( i < block_length ) {
/* Mode indicator */
strcat ( binary , " 0111 " ) ;
first_byte = ( source [ i + position ] & 0xff00 ) > > 8 ;
second_byte = source [ i + position ] & 0xff ;
third_byte = ( source [ i + position + 1 ] & 0xff00 ) > > 8 ;
fourth_byte = source [ i + position + 1 ] & 0xff ;
glyph = ( 0x3138 * ( first_byte - 0x81 ) ) + ( 0x04ec * ( second_byte - 0x30 ) ) +
( 0x0a * ( third_byte - 0x81 ) ) + ( fourth_byte - 0x30 ) ;
if ( debug ) {
printf ( " %d " , glyph ) ;
}
for ( p = 0 ; p < 15 ; p + + ) {
if ( glyph & ( 0x4000 > > p ) ) {
strcat ( binary , " 1 " ) ;
} else {
strcat ( binary , " 0 " ) ;
}
}
i + = 2 ;
}
/* No terminator */
if ( debug ) {
printf ( " \n " ) ;
}
break ;
2016-04-08 00:30:37 +03:00
}
position + = block_length ;
} while ( position < length ) ;
}
2016-04-09 19:01:21 +03:00
/* Finder pattern for top left of symbol */
void hx_place_finder_top_left ( unsigned char * grid , int size ) {
int xp , yp ;
int x = 0 , y = 0 ;
int finder [ ] = {
1 , 1 , 1 , 1 , 1 , 1 , 1 ,
1 , 0 , 0 , 0 , 0 , 0 , 0 ,
1 , 0 , 1 , 1 , 1 , 1 , 1 ,
1 , 0 , 1 , 0 , 0 , 0 , 0 ,
1 , 0 , 1 , 0 , 1 , 1 , 1 ,
1 , 0 , 1 , 0 , 1 , 1 , 1 ,
1 , 0 , 1 , 0 , 1 , 1 , 1
} ;
for ( xp = 0 ; xp < 7 ; xp + + ) {
for ( yp = 0 ; yp < 7 ; yp + + ) {
if ( finder [ xp + ( 7 * yp ) ] = = 1 ) {
grid [ ( ( yp + y ) * size ) + ( xp + x ) ] = 0x11 ;
} else {
grid [ ( ( yp + y ) * size ) + ( xp + x ) ] = 0x10 ;
}
}
}
}
/* Finder pattern for top right and bottom left of symbol */
void hx_place_finder ( unsigned char * grid , int size , int x , int y ) {
int xp , yp ;
int finder [ ] = {
1 , 1 , 1 , 1 , 1 , 1 , 1 ,
0 , 0 , 0 , 0 , 0 , 0 , 1 ,
1 , 1 , 1 , 1 , 1 , 0 , 1 ,
0 , 0 , 0 , 0 , 1 , 0 , 1 ,
1 , 1 , 1 , 0 , 1 , 0 , 1 ,
1 , 1 , 1 , 0 , 1 , 0 , 1 ,
1 , 1 , 1 , 0 , 1 , 0 , 1
} ;
for ( xp = 0 ; xp < 7 ; xp + + ) {
for ( yp = 0 ; yp < 7 ; yp + + ) {
if ( finder [ xp + ( 7 * yp ) ] = = 1 ) {
grid [ ( ( yp + y ) * size ) + ( xp + x ) ] = 0x11 ;
} else {
grid [ ( ( yp + y ) * size ) + ( xp + x ) ] = 0x10 ;
}
}
}
}
/* Finder pattern for bottom right of symbol */
void hx_place_finder_bottom_right ( unsigned char * grid , int size ) {
int xp , yp ;
int x = size - 7 , y = size - 7 ;
int finder [ ] = {
1 , 1 , 1 , 0 , 1 , 0 , 1 ,
1 , 1 , 1 , 0 , 1 , 0 , 1 ,
1 , 1 , 1 , 0 , 1 , 0 , 1 ,
0 , 0 , 0 , 0 , 1 , 0 , 1 ,
1 , 1 , 1 , 1 , 1 , 0 , 1 ,
0 , 0 , 0 , 0 , 0 , 0 , 1 ,
1 , 1 , 1 , 1 , 1 , 1 , 1
} ;
for ( xp = 0 ; xp < 7 ; xp + + ) {
for ( yp = 0 ; yp < 7 ; yp + + ) {
if ( finder [ xp + ( 7 * yp ) ] = = 1 ) {
grid [ ( ( yp + y ) * size ) + ( xp + x ) ] = 0x11 ;
} else {
grid [ ( ( yp + y ) * size ) + ( xp + x ) ] = 0x10 ;
}
}
}
}
/* Avoid plotting outside symbol or over finder patterns */
void hx_safe_plot ( unsigned char * grid , int size , int x , int y , int value ) {
if ( ( x > = 0 ) & & ( x < size ) ) {
if ( ( y > = 0 ) & & ( y < size ) ) {
if ( grid [ ( y * size ) + x ] = = 0 ) {
grid [ ( y * size ) + x ] = value ;
}
}
}
}
/* Plot an alignment pattern around top and right of a module */
void hx_plot_alignment ( unsigned char * grid , int size , int x , int y , int w , int h ) {
int i ;
hx_safe_plot ( grid , size , x , y , 0x11 ) ;
hx_safe_plot ( grid , size , x - 1 , y + 1 , 0x10 ) ;
for ( i = 1 ; i < = w ; i + + ) {
/* Top */
hx_safe_plot ( grid , size , x - i , y , 0x11 ) ;
hx_safe_plot ( grid , size , x - i - 1 , y + 1 , 0x10 ) ;
}
for ( i = 1 ; i < h ; i + + ) {
/* Right */
hx_safe_plot ( grid , size , x , y + i , 0x11 ) ;
hx_safe_plot ( grid , size , x - 1 , y + i + 1 , 0x10 ) ;
}
}
2016-04-10 22:36:57 +03:00
/* Plot assistant alignment patterns */
2016-04-09 19:01:21 +03:00
void hx_plot_assistant ( unsigned char * grid , int size , int x , int y ) {
hx_safe_plot ( grid , size , x - 1 , y - 1 , 0x10 ) ;
hx_safe_plot ( grid , size , x , y - 1 , 0x10 ) ;
hx_safe_plot ( grid , size , x + 1 , y - 1 , 0x10 ) ;
hx_safe_plot ( grid , size , x - 1 , y , 0x10 ) ;
hx_safe_plot ( grid , size , x , y , 0x11 ) ;
hx_safe_plot ( grid , size , x + 1 , y , 0x10 ) ;
hx_safe_plot ( grid , size , x - 1 , y + 1 , 0x10 ) ;
hx_safe_plot ( grid , size , x , y + 1 , 0x10 ) ;
hx_safe_plot ( grid , size , x + 1 , y + 1 , 0x10 ) ;
}
/* Put static elements in the grid */
void hx_setup_grid ( unsigned char * grid , int size , int version ) {
int i , j ;
for ( i = 0 ; i < size ; i + + ) {
for ( j = 0 ; j < size ; j + + ) {
grid [ ( i * size ) + j ] = 0 ;
}
}
/* Add finder patterns */
hx_place_finder_top_left ( grid , size ) ;
hx_place_finder ( grid , size , 0 , size - 7 ) ;
hx_place_finder ( grid , size , size - 7 , 0 ) ;
hx_place_finder_bottom_right ( grid , size ) ;
/* Add finder pattern separator region */
for ( i = 0 ; i < 8 ; i + + ) {
/* Top left */
grid [ ( 7 * size ) + i ] = 0x10 ;
grid [ ( i * size ) + 7 ] = 0x10 ;
/* Top right */
grid [ ( 7 * size ) + ( size - i - 1 ) ] = 0x10 ;
grid [ ( ( size - i - 1 ) * size ) + 7 ] = 0x10 ;
/* Bottom left */
grid [ ( i * size ) + ( size - 8 ) ] = 0x10 ;
grid [ ( ( size - 8 ) * size ) + i ] = 0x10 ;
/* Bottom right */
grid [ ( ( size - 8 ) * size ) + ( size - i - 1 ) ] = 0x10 ;
grid [ ( ( size - i - 1 ) * size ) + ( size - 8 ) ] = 0x10 ;
}
/* Reserve function information region */
for ( i = 0 ; i < 9 ; i + + ) {
/* Top left */
grid [ ( 8 * size ) + i ] = 0x10 ;
grid [ ( i * size ) + 8 ] = 0x10 ;
/* Top right */
grid [ ( 8 * size ) + ( size - i - 1 ) ] = 0x10 ;
grid [ ( ( size - i - 1 ) * size ) + 8 ] = 0x10 ;
/* Bottom left */
grid [ ( i * size ) + ( size - 9 ) ] = 0x10 ;
grid [ ( ( size - 9 ) * size ) + i ] = 0x10 ;
/* Bottom right */
grid [ ( ( size - 9 ) * size ) + ( size - i - 1 ) ] = 0x10 ;
grid [ ( ( size - i - 1 ) * size ) + ( size - 9 ) ] = 0x10 ;
}
if ( version > 3 ) {
int k = hx_module_k [ version - 1 ] ;
int r = hx_module_r [ version - 1 ] ;
int m = hx_module_m [ version - 1 ] ;
int x , y , row_switch , column_switch ;
int module_height , module_width ;
int mod_x , mod_y ;
/* Add assistant alignment patterns to left and right */
y = 0 ;
mod_y = 0 ;
do {
if ( mod_y < m ) {
module_height = k ;
} else {
module_height = r - 1 ;
}
if ( ( mod_y % 2 ) = = 0 ) {
if ( ( m % 2 ) = = 1 ) {
hx_plot_assistant ( grid , size , 0 , y ) ;
}
} else {
if ( ( m % 2 ) = = 0 ) {
hx_plot_assistant ( grid , size , 0 , y ) ;
}
hx_plot_assistant ( grid , size , size - 1 , y ) ;
}
mod_y + + ;
y + = module_height ;
} while ( y < size ) ;
/* Add assistant alignment patterns to top and bottom */
x = ( size - 1 ) ;
mod_x = 0 ;
do {
if ( mod_x < m ) {
module_width = k ;
} else {
module_width = r - 1 ;
}
if ( ( mod_x % 2 ) = = 0 ) {
if ( ( m % 2 ) = = 1 ) {
hx_plot_assistant ( grid , size , x , ( size - 1 ) ) ;
}
} else {
if ( ( m % 2 ) = = 0 ) {
hx_plot_assistant ( grid , size , x , ( size - 1 ) ) ;
}
hx_plot_assistant ( grid , size , x , 0 ) ;
}
mod_x + + ;
x - = module_width ;
} while ( x > = 0 ) ;
/* Add alignment pattern */
column_switch = 1 ;
y = 0 ;
mod_y = 0 ;
do {
if ( mod_y < m ) {
module_height = k ;
} else {
module_height = r - 1 ;
}
if ( column_switch = = 1 ) {
row_switch = 1 ;
column_switch = 0 ;
} else {
row_switch = 0 ;
column_switch = 1 ;
}
x = ( size - 1 ) ;
mod_x = 0 ;
do {
if ( mod_x < m ) {
module_width = k ;
} else {
module_width = r - 1 ;
}
if ( row_switch = = 1 ) {
if ( ! ( y = = 0 & & x = = ( size - 1 ) ) ) {
hx_plot_alignment ( grid , size , x , y , module_width , module_height ) ;
}
row_switch = 0 ;
} else {
row_switch = 1 ;
}
mod_x + + ;
x - = module_width ;
} while ( x > = 0 ) ;
mod_y + + ;
y + = module_height ;
} while ( y < size ) ;
}
}
2016-04-16 14:57:45 +03:00
/* Calculate error correction codes */
void hx_add_ecc ( unsigned char fullstream [ ] , unsigned char datastream [ ] , int version , int ecc_level ) {
unsigned char data_block [ 180 ] ;
unsigned char ecc_block [ 36 ] ;
int i , j , block ;
int batch_size , data_length , ecc_length ;
int input_position = - 1 ;
int output_position = - 1 ;
for ( i = 0 ; i < 3 ; i + + ) {
batch_size = hx_table_d1 [ ( ( ( version - 1 ) + ( ecc_level - 1 ) ) * 9 ) + ( 3 * i ) ] ;
data_length = hx_table_d1 [ ( ( ( version - 1 ) + ( ecc_level - 1 ) ) * 9 ) + ( 3 * i ) + 1 ] ;
ecc_length = hx_table_d1 [ ( ( ( version - 1 ) + ( ecc_level - 1 ) ) * 9 ) + ( 3 * i ) + 2 ] ;
for ( block = 0 ; block < batch_size ; block + + ) {
for ( j = 0 ; j < data_length ; j + + ) {
input_position + + ;
output_position + + ;
data_block [ j ] = datastream [ input_position ] ;
fullstream [ output_position ] = datastream [ input_position ] ;
}
rs_init_gf ( 0x163 ) ; // x^8 + x^6 + x^5 + x + 1 = 0
rs_init_code ( ecc_length , 1 ) ;
rs_encode ( data_length , data_block , ecc_block ) ;
rs_free ( ) ;
for ( j = 0 ; j < ecc_length ; j + + ) {
output_position + + ;
fullstream [ output_position ] = ecc_block [ ecc_length - j - 1 ] ;
}
}
}
}
/* Rearrange data in batches of 13 codewords (section 5.8.2) */
void make_picket_fence ( unsigned char fullstream [ ] , unsigned char picket_fence [ ] , int streamsize ) {
int i , start ;
int output_position = 0 ;
for ( start = 0 ; start < 13 ; start + + ) {
for ( i = start ; i < streamsize ; i + = 13 ) {
if ( i < streamsize ) {
picket_fence [ output_position ] = fullstream [ i ] ;
output_position + + ;
}
}
}
}
2016-04-20 22:44:59 +03:00
/* Evaluate a bitmask according to table 9 */
int hx_evaluate ( unsigned char * eval , int size , int pattern ) {
int x , y , i , block , weight ;
int result = 0 ;
char state ;
int p ;
int a , b , afterCount , beforeCount ;
# ifndef _MSC_VER
char local [ size * size ] ;
# else
char * local = ( char * ) _alloca ( ( size * size ) * sizeof ( char ) ) ;
# endif
2016-04-23 17:26:51 +03:00
/* all four bitmask variants have been encoded in the 4 bits of the bytes
2016-04-20 22:44:59 +03:00
* that make up the grid array . select them for evaluation according to the
* desired pattern . */
for ( x = 0 ; x < size ; x + + ) {
for ( y = 0 ; y < size ; y + + ) {
if ( ( eval [ ( y * size ) + x ] & ( 0x01 < < pattern ) ) ! = 0 ) {
local [ ( y * size ) + x ] = ' 1 ' ;
} else {
local [ ( y * size ) + x ] = ' 0 ' ;
}
}
}
2016-04-23 17:26:51 +03:00
/* Test 1: 1:1:1:1:3 or 3:1:1:1:1 ratio pattern in row/column */
2016-04-20 22:44:59 +03:00
/* Vertical */
for ( x = 0 ; x < size ; x + + ) {
for ( y = 0 ; y < ( size - 7 ) ; y + + ) {
p = 0 ;
for ( weight = 0 ; weight < 7 ; weight + + ) {
if ( local [ ( ( y + weight ) * size ) + x ] = = ' 1 ' ) {
p + = ( 0x40 > > weight ) ;
}
}
if ( ( p = = 0x57 ) | | ( p = 0x75 ) ) {
/* Pattern found, check before and after */
beforeCount = 0 ;
for ( b = ( y - 3 ) ; b < y ; b + + ) {
if ( b < 0 ) {
beforeCount + + ;
} else {
if ( local [ ( b * size ) + x ] = = ' 0 ' ) {
beforeCount + + ;
} else {
beforeCount = 0 ;
}
}
}
afterCount = 0 ;
for ( a = ( y + 7 ) ; a < = ( y + 9 ) ; a + + ) {
if ( a > = size ) {
afterCount + + ;
} else {
if ( local [ ( a * size ) + x ] = = ' 0 ' ) {
afterCount + + ;
} else {
afterCount = 0 ;
}
}
}
if ( ( beforeCount = = 3 ) | | ( afterCount = = 3 ) ) {
/* Pattern is preceeded or followed by light area
3 modules wide */
result + = 50 ;
}
}
}
}
/* Horizontal */
for ( y = 0 ; y < size ; y + + ) {
for ( x = 0 ; x < ( size - 7 ) ; x + + ) {
p = 0 ;
for ( weight = 0 ; weight < 7 ; weight + + ) {
if ( local [ ( y * size ) + x + weight ] = = ' 1 ' ) {
p + = ( 0x40 > > weight ) ;
}
}
if ( ( p = = 0x57 ) | | ( p = 0x75 ) ) {
/* Pattern found, check before and after */
beforeCount = 0 ;
for ( b = ( x - 3 ) ; b < x ; b + + ) {
if ( b < 0 ) {
beforeCount + + ;
} else {
if ( local [ ( y * size ) + b ] = = ' 0 ' ) {
beforeCount + + ;
} else {
beforeCount = 0 ;
}
}
}
afterCount = 0 ;
for ( a = ( x + 7 ) ; a < = ( x + 9 ) ; a + + ) {
if ( a > = size ) {
afterCount + + ;
} else {
if ( local [ ( y * size ) + a ] = = ' 0 ' ) {
afterCount + + ;
} else {
afterCount = 0 ;
}
}
}
if ( ( beforeCount = = 3 ) | | ( afterCount = = 3 ) ) {
/* Pattern is preceeded or followed by light area
3 modules wide */
result + = 50 ;
}
}
}
}
/* Test 2: Adjacent modules in row/column in same colour */
2016-06-19 11:07:50 +03:00
/* In AIMD-15 section 5.8.3.2 it is stated... “In Table 9 below, i refers to the row
* position of the module . ” - however i being the length of the run of the
* same colour ( i . e . " block " below ) in the same fashion as ISO / IEC 18004
* makes more sense . The implementation below matches AIMD - 015. */
2016-04-20 22:44:59 +03:00
/* Vertical */
for ( x = 0 ; x < size ; x + + ) {
state = local [ x ] ;
block = 0 ;
for ( y = 0 ; y < size ; y + + ) {
i = y + 1 ;
if ( local [ ( y * size ) + x ] = = state ) {
block + + ;
} else {
if ( block > ( 3 + i ) ) {
result + = ( 3 + i ) * 4 ;
}
block = 0 ;
state = local [ ( y * size ) + x ] ;
}
}
if ( block > ( 3 + i ) ) {
result + = ( 3 + i ) * 4 ;
}
}
/* Horizontal */
for ( y = 0 ; y < size ; y + + ) {
i = y + 1 ;
state = local [ y * size ] ;
block = 0 ;
for ( x = 0 ; x < size ; x + + ) {
if ( local [ ( y * size ) + x ] = = state ) {
block + + ;
} else {
if ( block > ( 3 + i ) ) {
result + = ( 3 + i ) * 4 ;
}
block = 0 ;
state = local [ ( y * size ) + x ] ;
}
}
if ( block > ( 3 + i ) ) {
result + = ( 3 + i ) * 4 ;
}
}
return result ;
}
/* Apply the four possible bitmasks for evaluation */
int hx_apply_bitmask ( unsigned char * grid , int size ) {
int x , y ;
int i , j ;
int pattern , penalty [ 4 ] ;
int best_pattern , best_val ;
int bit ;
unsigned char p ;
# ifndef _MSC_VER
unsigned char mask [ size * size ] ;
unsigned char eval [ size * size ] ;
# else
unsigned char * mask = ( unsigned char * ) _alloca ( ( size * size ) * sizeof ( unsigned char ) ) ;
unsigned char * eval = ( unsigned char * ) _alloca ( ( size * size ) * sizeof ( unsigned char ) ) ;
# endif
/* Perform data masking */
for ( x = 0 ; x < size ; x + + ) {
for ( y = 0 ; y < size ; y + + ) {
mask [ ( y * size ) + x ] = 0x00 ;
j = x + 1 ;
i = y + 1 ;
if ( ! ( grid [ ( y * size ) + x ] & 0xf0 ) ) {
if ( ( i + j ) % 2 = = 0 ) {
mask [ ( y * size ) + x ] + = 0x02 ;
}
if ( ( ( ( i + j ) % 3 ) + ( j % 3 ) ) % 2 = = 0 ) {
mask [ ( y * size ) + x ] + = 0x04 ;
}
if ( ( ( i % j ) + ( j % i ) + ( i % 3 ) + ( j % 3 ) ) % 2 = = 0 ) {
mask [ ( y * size ) + x ] + = 0x08 ;
}
}
}
}
// apply data masks to grid, result in eval
for ( x = 0 ; x < size ; x + + ) {
for ( y = 0 ; y < size ; y + + ) {
if ( grid [ ( y * size ) + x ] & 0x01 ) {
p = 0xff ;
} else {
p = 0x00 ;
}
eval [ ( y * size ) + x ] = mask [ ( y * size ) + x ] ^ p ;
}
}
/* Evaluate result */
for ( pattern = 0 ; pattern < 4 ; pattern + + ) {
penalty [ pattern ] = hx_evaluate ( eval , size , pattern ) ;
}
best_pattern = 0 ;
best_val = penalty [ 0 ] ;
for ( pattern = 1 ; pattern < 4 ; pattern + + ) {
if ( penalty [ pattern ] < best_val ) {
best_pattern = pattern ;
best_val = penalty [ pattern ] ;
}
}
/* Apply mask */
for ( x = 0 ; x < size ; x + + ) {
for ( y = 0 ; y < size ; y + + ) {
bit = 0 ;
switch ( best_pattern ) {
case 0 : if ( mask [ ( y * size ) + x ] & 0x01 ) {
bit = 1 ;
}
break ;
case 1 : if ( mask [ ( y * size ) + x ] & 0x02 ) {
bit = 1 ;
}
break ;
case 2 : if ( mask [ ( y * size ) + x ] & 0x04 ) {
bit = 1 ;
}
break ;
case 3 : if ( mask [ ( y * size ) + x ] & 0x08 ) {
bit = 1 ;
}
break ;
}
if ( bit = = 1 ) {
if ( grid [ ( y * size ) + x ] & 0x01 ) {
grid [ ( y * size ) + x ] = 0x00 ;
} else {
grid [ ( y * size ) + x ] = 0x01 ;
}
}
}
}
return best_pattern ;
}
2016-04-08 00:30:37 +03:00
/* Han Xin Code - main */
2016-04-07 19:13:43 +03:00
int han_xin ( struct zint_symbol * symbol , const unsigned char source [ ] , int length ) {
2016-04-08 00:30:37 +03:00
char mode [ length + 1 ] ;
int est_binlen ;
2016-04-30 12:25:16 +03:00
int ecc_level = symbol - > option_1 ;
2016-05-02 00:10:50 +03:00
int i , j , version , posn = 0 , glyph , glyph2 ;
2016-04-10 22:36:57 +03:00
int data_codewords , size ;
int est_codewords ;
2016-04-20 22:44:59 +03:00
int bitmask ;
2016-05-02 00:10:50 +03:00
int error_number ;
2016-04-23 17:26:51 +03:00
char function_information [ 36 ] ;
unsigned char fi_cw [ 3 ] = { 0 , 0 , 0 } ;
unsigned char fi_ecc [ 4 ] ;
2016-05-02 00:10:50 +03:00
# ifndef _MSC_VER
int utfdata [ length + 1 ] ;
int gbdata [ ( length + 1 ) * 2 ] ;
# else
int * utfdata = ( int * ) _alloca ( ( length + 1 ) * sizeof ( int ) ) ;
int * gbdata = ( int * ) _alloca ( ( ( length + 1 ) * 2 ) * sizeof ( int ) ) ;
# endif
switch ( symbol - > input_mode ) {
case DATA_MODE :
for ( i = 0 ; i < length ; i + + ) {
gbdata [ i ] = ( int ) source [ i ] ;
}
break ;
default :
/* Convert Unicode input to GB-18030 */
error_number = utf8toutf16 ( symbol , source , utfdata , & length ) ;
if ( error_number ! = 0 ) {
return error_number ;
}
posn = 0 ;
for ( i = 0 ; i < length ; i + + ) {
if ( utfdata [ i ] < = 0x7f ) {
gbdata [ posn ] = utfdata [ i ] ;
posn + + ;
} else {
j = 0 ;
glyph = 0 ;
do {
if ( gb18030_twobyte_lookup [ j * 2 ] = = utfdata [ i ] ) {
glyph = gb18030_twobyte_lookup [ ( j * 2 ) + 1 ] ;
}
j + + ;
} while ( ( j < 23940 ) & & ( glyph = = 0 ) ) ;
if ( glyph = = 0 ) {
j = 0 ;
glyph = 0 ;
glyph2 = 0 ;
do {
if ( gb18030_fourbyte_lookup [ j * 3 ] = = utfdata [ i ] ) {
glyph = gb18030_fourbyte_lookup [ ( j * 3 ) + 1 ] ;
glyph2 = gb18030_fourbyte_lookup [ ( j * 3 ) + 2 ] ;
}
j + + ;
} while ( ( j < 6793 ) & & ( glyph = = 0 ) ) ;
if ( glyph = = 0 ) {
strcpy ( symbol - > errtxt , " Unknown character in input data " ) ;
return ZINT_ERROR_INVALID_DATA ;
} else {
gbdata [ posn ] = glyph ;
gbdata [ posn + 1 ] = glyph2 ;
posn + = 2 ;
}
} else {
gbdata [ posn ] = glyph ;
posn + + ;
}
}
}
break ;
}
2016-04-08 00:30:37 +03:00
2016-05-02 00:10:50 +03:00
length = posn ;
hx_define_mode ( mode , gbdata , length ) ;
2016-04-08 00:30:37 +03:00
2016-05-02 00:10:50 +03:00
est_binlen = calculate_binlength ( mode , gbdata , length ) ;
2016-04-10 22:36:57 +03:00
est_codewords = est_binlen / 8 ;
if ( est_binlen % 8 ! = 0 ) {
est_codewords + + ;
}
char binary [ est_binlen + 1 ] ;
for ( i = 0 ; i < est_binlen + 1 ; i + + ) {
binary [ i ] = ' \0 ' ;
}
2016-04-08 00:30:37 +03:00
binary [ 0 ] = ' \0 ' ;
2016-04-30 12:25:16 +03:00
if ( ( ecc_level < = 0 ) | | ( ecc_level > = 5 ) ) {
ecc_level = 1 ;
}
2016-05-02 00:10:50 +03:00
calculate_binary ( binary , mode , gbdata , length ) ;
2016-04-08 00:30:37 +03:00
2016-04-09 19:01:21 +03:00
version = 85 ;
for ( i = 84 ; i > 0 ; i - - ) {
switch ( ecc_level ) {
case 1 :
2016-04-10 22:36:57 +03:00
if ( hx_data_codewords_L1 [ i - 1 ] > est_codewords ) {
2016-04-09 19:01:21 +03:00
version = i ;
2016-04-10 22:36:57 +03:00
data_codewords = hx_data_codewords_L1 [ i - 1 ] ;
2016-04-09 19:01:21 +03:00
}
break ;
case 2 :
2016-04-10 22:36:57 +03:00
if ( hx_data_codewords_L2 [ i - 1 ] > est_codewords ) {
2016-04-09 19:01:21 +03:00
version = i ;
2016-04-10 22:36:57 +03:00
data_codewords = hx_data_codewords_L2 [ i - 1 ] ;
2016-04-09 19:01:21 +03:00
}
break ;
case 3 :
2016-04-10 22:36:57 +03:00
if ( hx_data_codewords_L3 [ i - 1 ] > est_codewords ) {
2016-04-09 19:01:21 +03:00
version = i ;
2016-04-10 22:36:57 +03:00
data_codewords = hx_data_codewords_L3 [ i - 1 ] ;
2016-04-09 19:01:21 +03:00
}
break ;
case 4 :
2016-04-10 22:36:57 +03:00
if ( hx_data_codewords_L4 [ i - 1 ] > est_codewords ) {
2016-04-09 19:01:21 +03:00
version = i ;
2016-04-10 22:36:57 +03:00
data_codewords = hx_data_codewords_L4 [ i - 1 ] ;
2016-04-09 19:01:21 +03:00
}
break ;
}
}
if ( version = = 85 ) {
strcpy ( symbol - > errtxt , " Input too long for selected error correction level " ) ;
return ZINT_ERROR_TOO_LONG ;
}
2016-04-30 12:25:16 +03:00
if ( ( symbol - > option_2 < 0 ) | | ( symbol - > option_2 > 84 ) ) {
symbol - > option_2 = 0 ;
}
if ( symbol - > option_2 > version ) {
version = symbol - > option_2 ;
}
/* If there is spare capacity, increase the level of ECC */
switch ( ecc_level ) {
case 1 :
if ( est_codewords < hx_data_codewords_L2 [ version - 1 ] ) {
ecc_level = 2 ;
data_codewords = hx_data_codewords_L2 [ version - 1 ] ;
}
case 2 :
if ( est_codewords < hx_data_codewords_L3 [ version - 1 ] ) {
ecc_level = 3 ;
data_codewords = hx_data_codewords_L3 [ version - 1 ] ;
}
case 3 :
if ( est_codewords < hx_data_codewords_L4 [ version - 1 ] ) {
ecc_level = 4 ;
data_codewords = hx_data_codewords_L4 [ version - 1 ] ;
}
}
2016-05-02 00:10:50 +03:00
//printf("Version %d, ECC %d\n", version, ecc_level);
2016-04-30 12:25:16 +03:00
2016-04-09 19:01:21 +03:00
size = ( version * 2 ) + 21 ;
2016-04-16 14:57:45 +03:00
2016-04-09 19:01:21 +03:00
# ifndef _MSC_VER
2016-04-16 14:57:45 +03:00
unsigned char datastream [ data_codewords ] ;
unsigned char fullstream [ hx_total_codewords [ version - 1 ] ] ;
unsigned char picket_fence [ hx_total_codewords [ version - 1 ] ] ;
2016-04-09 19:01:21 +03:00
unsigned char grid [ size * size ] ;
# else
2016-04-16 14:57:45 +03:00
datastream = ( unsigned char * ) _alloca ( ( data_codewords ) * sizeof ( unsigned char ) ) ;
fullstream = ( unsigned char * ) _alloca ( ( hx_total_codewords [ version - 1 ] ) * sizeof ( unsigned char ) ) ;
picket_fence = ( unsigned char * ) _alloca ( ( hx_total_codewords [ version - 1 ] ) * sizeof ( unsigned char ) ) ;
2016-04-09 19:01:21 +03:00
grid = ( unsigned char * ) _alloca ( ( size * size ) * sizeof ( unsigned char ) ) ;
# endif
2016-04-16 14:57:45 +03:00
2016-04-10 22:36:57 +03:00
for ( i = 0 ; i < data_codewords ; i + + ) {
datastream [ i ] = 0 ;
}
for ( i = 0 ; i < est_binlen ; i + + ) {
if ( binary [ i ] = = ' 1 ' ) {
datastream [ i / 8 ] + = 0x80 > > ( i % 8 ) ;
}
}
2016-04-16 14:57:45 +03:00
2016-04-09 19:01:21 +03:00
hx_setup_grid ( grid , size , version ) ;
2016-04-10 22:36:57 +03:00
2016-04-16 14:57:45 +03:00
hx_add_ecc ( fullstream , datastream , version , ecc_level ) ;
make_picket_fence ( fullstream , picket_fence , hx_total_codewords [ version - 1 ] ) ;
2016-04-20 22:44:59 +03:00
/* Populate grid */
j = 0 ;
for ( i = 0 ; i < ( size * size ) ; i + + ) {
if ( grid [ i ] = = 0x00 ) {
if ( j < ( hx_total_codewords [ version - 1 ] * 8 ) ) {
if ( picket_fence [ ( j / 8 ) ] & ( 0x80 > > ( j % 8 ) ) ) {
grid [ i ] = 0x01 ;
}
j + + ;
}
}
}
bitmask = hx_apply_bitmask ( grid , size ) ;
2016-04-23 17:26:51 +03:00
/* Form function information string */
for ( i = 0 ; i < 34 ; i + + ) {
if ( i % 2 ) {
function_information [ i ] = ' 1 ' ;
} else {
function_information [ i ] = ' 0 ' ;
}
}
function_information [ 34 ] = ' \0 ' ;
for ( i = 0 ; i < 8 ; i + + ) {
if ( ( version + 20 ) & ( 0x80 > > i ) ) {
function_information [ i ] = ' 1 ' ;
} else {
function_information [ i ] = ' 0 ' ;
}
}
for ( i = 0 ; i < 2 ; i + + ) {
if ( ecc_level & ( 0x02 > > i ) ) {
function_information [ i + 8 ] = ' 1 ' ;
} else {
function_information [ i + 8 ] = ' 0 ' ;
}
}
for ( i = 0 ; i < 2 ; i + + ) {
if ( bitmask & ( 0x02 > > i ) ) {
function_information [ i + 10 ] = ' 1 ' ;
} else {
function_information [ i + 10 ] = ' 0 ' ;
}
}
for ( i = 0 ; i < 3 ; i + + ) {
for ( j = 0 ; j < 4 ; j + + ) {
if ( function_information [ ( i * 4 ) + j ] = = ' 1 ' ) {
fi_cw [ i ] + = ( 0x08 > > j ) ;
}
}
}
rs_init_gf ( 0x13 ) ;
rs_init_code ( 4 , 1 ) ;
rs_encode ( 3 , fi_cw , fi_ecc ) ;
rs_free ( ) ;
for ( i = 0 ; i < 4 ; i + + ) {
for ( j = 0 ; j < 4 ; j + + ) {
if ( fi_ecc [ 3 - i ] & ( 0x08 > > j ) ) {
function_information [ ( i * 4 ) + j + 12 ] = ' 1 ' ;
} else {
function_information [ ( i * 4 ) + j + 12 ] = ' 0 ' ;
}
}
}
/* Add function information to symbol */
for ( i = 0 ; i < 9 ; i + + ) {
if ( function_information [ i ] = = ' 1 ' ) {
grid [ ( 8 * size ) + i ] = 0x01 ;
grid [ ( ( size - 8 - 1 ) * size ) + ( size - i - 1 ) ] = 0x01 ;
}
if ( function_information [ i + 8 ] = = ' 1 ' ) {
grid [ ( ( 8 - i ) * size ) + 8 ] = 0x01 ;
grid [ ( ( size - 8 - 1 + i ) * size ) + ( size - 8 - 1 ) ] = 0x01 ;
}
if ( function_information [ i + 17 ] = = ' 1 ' ) {
grid [ ( i * size ) + ( size - 1 - 8 ) ] = 0x01 ;
grid [ ( ( size - 1 - i ) * size ) + 8 ] = 0x01 ;
}
if ( function_information [ i + 25 ] = = ' 1 ' ) {
grid [ ( 8 * size ) + ( size - 1 - 8 + i ) ] = 0x01 ;
grid [ ( ( size - 1 - 8 ) * size ) + ( 8 - i ) ] = 0x01 ;
}
}
2016-04-16 14:57:45 +03:00
2016-04-09 19:01:21 +03:00
symbol - > width = size ;
symbol - > rows = size ;
for ( i = 0 ; i < size ; i + + ) {
for ( j = 0 ; j < size ; j + + ) {
if ( grid [ ( i * size ) + j ] & 0x01 ) {
set_module ( symbol , i , j ) ;
}
}
symbol - > row_height [ i ] = 1 ;
}
2016-04-16 14:57:45 +03:00
2016-04-30 12:25:16 +03:00
return 0 ;
2016-04-07 19:13:43 +03:00
}