/* rss.c - Handles Reduced Space Symbology (GS1 DataBar) */ /* libzint - the open source barcode library Copyright (C) 2008-2020 Robin Stuart 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. */ /* vim: set ts=4 sw=4 et : */ /* The functions "combins" and "getRSSwidths" are copyright BSI and are released with permission under the following terms: "Copyright subsists in all BSI publications. BSI also holds the copyright, in the UK, of the international standardisation bodies. Except as permitted under the Copyright, Designs and Patents Act 1988 no extract may be reproduced, stored in a retrieval system or transmitted in any form or by any means - electronic, photocopying, recording or otherwise - without prior written permission from BSI. "This does not preclude the free use, in the course of implementing the standard, of necessary details such as symbols, and size, type or grade designations. If these details are to be used for any other purpose than implementation then the prior written permission of BSI must be obtained." The date of publication for these functions is 30 November 2006 */ /* Includes numerous bugfixes thanks to Pablo Orduña @ the PIRAmIDE project */ /* Note: This code reflects the symbol names as used in ISO/IEC 24724:2006. These names * were updated in ISO/IEC 24724:2011 as follows: * * RSS-14 > GS1 DataBar Omnidirectional * RSS-14 Truncated > GS1 DataBar Truncated * RSS-14 Stacked > GS1 DataBar Stacked * RSS-14 Stacked Omnidirectional > GS1 DataBar Stacked Omnidirectional * RSS Limited > GS1 DataBar Limited * RSS Expanded > GS1 DataBar Expanded Omnidirectional * RSS Expanded Stacked > GS1 DataBar Expanded Stacked Omnidirectional */ #include #ifdef _MSC_VER #include #endif #include "common.h" #include "large.h" #include "rss.h" #include "gs1.h" #include "general_field.h" /********************************************************************** * combins(n,r): returns the number of Combinations of r selected from n: * Combinations = n! / ((n - r)! * r!) **********************************************************************/ static int combins(int n, int r) { int i, j; int maxDenom, minDenom; int val; if (n - r > r) { minDenom = r; maxDenom = n - r; } else { minDenom = n - r; maxDenom = r; } val = 1; j = 1; for (i = n; i > maxDenom; i--) { val *= i; if (j <= minDenom) { val /= j; j++; } } for (; j <= minDenom; j++) { val /= j; } return (val); } /********************************************************************** * getRSSwidths * routine to generate widths for RSS elements for a given value.# * * Calling arguments: * int widths[] = element widths * val = required value * n = number of modules * elements = elements in a set (RSS-14 & Expanded = 4; RSS Limited = 7) * maxWidth = maximum module width of an element * noNarrow = 0 will skip patterns without a one module wide element * **********************************************************************/ static void getRSSwidths(int widths[], int val, int n, int elements, int maxWidth, int noNarrow) { int bar; int elmWidth; int mxwElement; int subVal, lessVal; int narrowMask = 0; for (bar = 0; bar < elements - 1; bar++) { for (elmWidth = 1, narrowMask |= (1 << bar); ; elmWidth++, narrowMask &= ~(1 << bar)) { /* get all combinations */ subVal = combins(n - elmWidth - 1, elements - bar - 2); /* less combinations with no single-module element */ if ((!noNarrow) && (!narrowMask) && (n - elmWidth - (elements - bar - 1) >= elements - bar - 1)) { subVal -= combins(n - elmWidth - (elements - bar), elements - bar - 2); } /* less combinations with elements > maxVal */ if (elements - bar - 1 > 1) { lessVal = 0; for (mxwElement = n - elmWidth - (elements - bar - 2); mxwElement > maxWidth; mxwElement--) { lessVal += combins(n - elmWidth - mxwElement - 1, elements - bar - 3); } subVal -= lessVal * (elements - 1 - bar); } else if (n - elmWidth > maxWidth) { subVal--; } val -= subVal; if (val < 0) break; } val += subVal; n -= elmWidth; widths[bar] = elmWidth; } widths[bar] = n; return; } /* Calculate check digit from Annex A */ static int calc_check_digit(unsigned char *src) { int i, check_digit; int count = 0; for (i = 0; i < 13; i++) { count += (i & 1) ? ctoi(src[i]) : 3 * ctoi(src[i]); } check_digit = 10 - (count % 10); if (check_digit == 10) { check_digit = 0; } return check_digit; } /* Set GTIN-14 human readable text */ static void set_gtin14_hrt(struct zint_symbol *symbol, unsigned char *source, int src_len) { int i; unsigned char hrt[15]; ustrcpy(symbol->text, "(01)"); for (i = 0; i < 12; i++) { hrt[i] = '0'; } for (i = 0; i < src_len; i++) { hrt[12 - i] = source[src_len - i - 1]; } hrt[13] = itoc(calc_check_digit(hrt)); hrt[14] = '\0'; ustrcat(symbol->text, hrt); } /* Expand from a width pattern to a bit pattern */ static int rss_expand(struct zint_symbol *symbol, int writer, char *p_latch, int width) { int j; int latch = *p_latch; for (j = 0; j < width; j++) { if (latch == '1') { set_module(symbol, symbol->rows, writer); } else { unset_module(symbol, symbol->rows, writer); } writer++; } if (latch == '1') { *p_latch = '0'; } else { *p_latch = '1'; } return writer; } /* Adjust top/bottom separator for finder patterns */ static void rss14_finder_adjust(struct zint_symbol *symbol, int separator_row, int above_below, int finder_start) { int i, finder_end; int module_row = separator_row + above_below; int latch; /* Alternation is always left-to-right for Omnidirectional separators (unlike for Expanded) */ latch = '1'; for (i = finder_start, finder_end = finder_start + 13; i < finder_end; i++) { if (!module_is_set(symbol, module_row, i)) { if (latch == '1') { set_module(symbol, separator_row, i); latch = '0'; } else { unset_module(symbol, separator_row, i); latch = '1'; } } else { unset_module(symbol, separator_row, i); latch = '1'; } } } /* Top/bottom separator for DataBar */ static void rss14_separator(struct zint_symbol *symbol, int width, int separator_row, int above_below, int finder_start, int finder2_start, int bottom_finder_value_3) { int i, finder_end, finder_value_3_set; int module_row = separator_row + above_below; for (i = 4, width -= 4; i < width; i++) { if (!module_is_set(symbol, module_row, i)) { set_module(symbol, separator_row, i); } } if (bottom_finder_value_3) { /* ISO/IEC 24724:2011 5.3.2.2 "The single dark module that occurs in the 13 modules over finder value 3 is * shifted one module to the right so that it is over the start of the three module-wide finder bar." */ finder_value_3_set = finder_start + 10; for (i = finder_start, finder_end = finder_start + 13; i < finder_end; i++) { if (i == finder_value_3_set) { set_module(symbol, separator_row, i); } else { unset_module(symbol, separator_row, i); } } } else { if (finder_start) { rss14_finder_adjust(symbol, separator_row, above_below, finder_start); } if (finder2_start) { rss14_finder_adjust(symbol, separator_row, above_below, finder2_start); } } } /* GS1 DataBar Omnidirectional/Truncated/Stacked */ INTERNAL int rss14(struct zint_symbol *symbol, unsigned char source[], int src_len) { int error_number = 0, i; large_int accum; uint64_t left_pair, right_pair; int data_character[4] = {0}, data_group[4] = {0}, v_odd[4], v_even[4]; int data_widths[8][4], checksum, c_left, c_right, total_widths[46], writer; char latch; int separator_row; int widths[4]; separator_row = 0; if (src_len > 14) { /* Allow check digit to be specified (will be verified and ignored) */ strcpy(symbol->errtxt, "380: Input too long"); return ZINT_ERROR_TOO_LONG; } error_number = is_sane(NEON, source, src_len); if (error_number == ZINT_ERROR_INVALID_DATA) { strcpy(symbol->errtxt, "381: Invalid characters in data"); return error_number; } if (src_len == 14) { /* Verify check digit */ if (calc_check_digit(source) != ctoi(source[13])) { strcpy(symbol->errtxt, "388: Invalid check digit"); return ZINT_ERROR_INVALID_CHECK; } src_len--; /* Ignore */ } /* make some room for a separator row for composite symbols */ switch (symbol->symbology) { case BARCODE_RSS14_CC: case BARCODE_RSS14STACK_CC: case BARCODE_RSS14_OMNI_CC: separator_row = symbol->rows; symbol->row_height[separator_row] = 1; symbol->rows += 1; break; } large_load_str_u64(&accum, source, src_len); if (symbol->option_1 == 2) { /* Add symbol linkage flag */ large_add_u64(&accum, 10000000000000); } /* Calculate left and right pair values */ right_pair = large_div_u64(&accum, 4537077); left_pair = large_lo(&accum); /* Calculate four data characters */ data_character[0] = left_pair / 1597; data_character[1] = left_pair % 1597; data_character[2] = right_pair / 1597; data_character[3] = right_pair % 1597; /* Calculate odd and even subset values */ if ((data_character[0] >= 0) && (data_character[0] <= 160)) { data_group[0] = 0; } if ((data_character[0] >= 161) && (data_character[0] <= 960)) { data_group[0] = 1; } if ((data_character[0] >= 961) && (data_character[0] <= 2014)) { data_group[0] = 2; } if ((data_character[0] >= 2015) && (data_character[0] <= 2714)) { data_group[0] = 3; } if ((data_character[0] >= 2715) && (data_character[0] <= 2840)) { data_group[0] = 4; } if ((data_character[1] >= 0) && (data_character[1] <= 335)) { data_group[1] = 5; } if ((data_character[1] >= 336) && (data_character[1] <= 1035)) { data_group[1] = 6; } if ((data_character[1] >= 1036) && (data_character[1] <= 1515)) { data_group[1] = 7; } if ((data_character[1] >= 1516) && (data_character[1] <= 1596)) { data_group[1] = 8; } if ((data_character[3] >= 0) && (data_character[3] <= 335)) { data_group[3] = 5; } if ((data_character[3] >= 336) && (data_character[3] <= 1035)) { data_group[3] = 6; } if ((data_character[3] >= 1036) && (data_character[3] <= 1515)) { data_group[3] = 7; } if ((data_character[3] >= 1516) && (data_character[3] <= 1596)) { data_group[3] = 8; } if ((data_character[2] >= 0) && (data_character[2] <= 160)) { data_group[2] = 0; } if ((data_character[2] >= 161) && (data_character[2] <= 960)) { data_group[2] = 1; } if ((data_character[2] >= 961) && (data_character[2] <= 2014)) { data_group[2] = 2; } if ((data_character[2] >= 2015) && (data_character[2] <= 2714)) { data_group[2] = 3; } if ((data_character[2] >= 2715) && (data_character[2] <= 2840)) { data_group[2] = 4; } v_odd[0] = (data_character[0] - g_sum_table[data_group[0]]) / t_table[data_group[0]]; v_even[0] = (data_character[0] - g_sum_table[data_group[0]]) % t_table[data_group[0]]; v_odd[1] = (data_character[1] - g_sum_table[data_group[1]]) % t_table[data_group[1]]; v_even[1] = (data_character[1] - g_sum_table[data_group[1]]) / t_table[data_group[1]]; v_odd[3] = (data_character[3] - g_sum_table[data_group[3]]) % t_table[data_group[3]]; v_even[3] = (data_character[3] - g_sum_table[data_group[3]]) / t_table[data_group[3]]; v_odd[2] = (data_character[2] - g_sum_table[data_group[2]]) / t_table[data_group[2]]; v_even[2] = (data_character[2] - g_sum_table[data_group[2]]) % t_table[data_group[2]]; /* Use RSS subset width algorithm */ for (i = 0; i < 4; i++) { if ((i == 0) || (i == 2)) { getRSSwidths(widths, v_odd[i], modules_odd[data_group[i]], 4, widest_odd[data_group[i]], 1); data_widths[0][i] = widths[0]; data_widths[2][i] = widths[1]; data_widths[4][i] = widths[2]; data_widths[6][i] = widths[3]; getRSSwidths(widths, v_even[i], modules_even[data_group[i]], 4, widest_even[data_group[i]], 0); data_widths[1][i] = widths[0]; data_widths[3][i] = widths[1]; data_widths[5][i] = widths[2]; data_widths[7][i] = widths[3]; } else { getRSSwidths(widths, v_odd[i], modules_odd[data_group[i]], 4, widest_odd[data_group[i]], 0); data_widths[0][i] = widths[0]; data_widths[2][i] = widths[1]; data_widths[4][i] = widths[2]; data_widths[6][i] = widths[3]; getRSSwidths(widths, v_even[i], modules_even[data_group[i]], 4, widest_even[data_group[i]], 1); data_widths[1][i] = widths[0]; data_widths[3][i] = widths[1]; data_widths[5][i] = widths[2]; data_widths[7][i] = widths[3]; } } checksum = 0; /* Calculate the checksum */ for (i = 0; i < 8; i++) { checksum += checksum_weight[i] * data_widths[i][0]; checksum += checksum_weight[i + 8] * data_widths[i][1]; checksum += checksum_weight[i + 16] * data_widths[i][2]; checksum += checksum_weight[i + 24] * data_widths[i][3]; } checksum %= 79; /* Calculate the two check characters */ if (checksum >= 8) { checksum++; } if (checksum >= 72) { checksum++; } c_left = checksum / 9; c_right = checksum % 9; if (symbol->debug & ZINT_DEBUG_PRINT) { printf("c_left: %d, c_right: %d\n", c_left, c_right); } /* Put element widths together */ total_widths[0] = 1; total_widths[1] = 1; total_widths[44] = 1; total_widths[45] = 1; for (i = 0; i < 8; i++) { total_widths[i + 2] = data_widths[i][0]; total_widths[i + 15] = data_widths[7 - i][1]; total_widths[i + 23] = data_widths[i][3]; total_widths[i + 36] = data_widths[7 - i][2]; } for (i = 0; i < 5; i++) { total_widths[i + 10] = finder_pattern[i + (5 * c_left)]; total_widths[i + 31] = finder_pattern[(4 - i) + (5 * c_right)]; } /* Put this data into the symbol */ if ((symbol->symbology == BARCODE_RSS14) || (symbol->symbology == BARCODE_RSS14_CC)) { writer = 0; latch = '0'; for (i = 0; i < 46; i++) { writer = rss_expand(symbol, writer, &latch, total_widths[i]); } if (symbol->width < writer) { symbol->width = writer; } if (symbol->symbology == BARCODE_RSS14_CC) { /* separator pattern for composite symbol */ rss14_separator(symbol, 96, separator_row, 1 /*above*/, 18, 63, 0 /*bottom_finder_value_3*/); } symbol->rows = symbol->rows + 1; /* Set human readable text */ set_gtin14_hrt(symbol, source, src_len); set_minimum_height(symbol, 14); // Minimum height is 14X for truncated symbol } if ((symbol->symbology == BARCODE_RSS14STACK) || (symbol->symbology == BARCODE_RSS14STACK_CC)) { /* top row */ writer = 0; latch = '0'; for (i = 0; i < 23; i++) { writer = rss_expand(symbol, writer, &latch, total_widths[i]); } set_module(symbol, symbol->rows, writer); unset_module(symbol, symbol->rows, writer + 1); symbol->row_height[symbol->rows] = 5; /* bottom row */ symbol->rows = symbol->rows + 2; set_module(symbol, symbol->rows, 0); unset_module(symbol, symbol->rows, 1); writer = 2; latch = '1'; for (i = 23; i < 46; i++) { writer = rss_expand(symbol, writer, &latch, total_widths[i]); } symbol->row_height[symbol->rows] = 7; /* separator pattern */ /* See #183 for this interpretation of ISO/IEC 24724:2011 5.3.2.1 */ for (i = 1; i < 46; i++) { if (module_is_set(symbol, symbol->rows - 2, i) == module_is_set(symbol, symbol->rows, i)) { if (!(module_is_set(symbol, symbol->rows - 2, i))) { set_module(symbol, symbol->rows - 1, i); } } else { if (!(module_is_set(symbol, symbol->rows - 1, i - 1))) { set_module(symbol, symbol->rows - 1, i); } } } unset_module(symbol, symbol->rows - 1, 1); unset_module(symbol, symbol->rows - 1, 2); unset_module(symbol, symbol->rows - 1, 3); symbol->row_height[symbol->rows - 1] = 1; if (symbol->symbology == BARCODE_RSS14STACK_CC) { /* separator pattern for composite symbol */ rss14_separator(symbol, 50, separator_row, 1 /*above*/, 18, 0, 0 /*bottom_finder_value_3*/); } symbol->rows = symbol->rows + 1; if (symbol->width < 50) { symbol->width = 50; } } if ((symbol->symbology == BARCODE_RSS14STACK_OMNI) || (symbol->symbology == BARCODE_RSS14_OMNI_CC)) { /* top row */ writer = 0; latch = '0'; for (i = 0; i < 23; i++) { writer = rss_expand(symbol, writer, &latch, total_widths[i]); } set_module(symbol, symbol->rows, writer); unset_module(symbol, symbol->rows, writer + 1); /* bottom row */ symbol->rows = symbol->rows + 4; set_module(symbol, symbol->rows, 0); unset_module(symbol, symbol->rows, 1); writer = 2; latch = '1'; for (i = 23; i < 46; i++) { writer = rss_expand(symbol, writer, &latch, total_widths[i]); } /* middle separator */ for (i = 5; i < 46; i += 2) { set_module(symbol, symbol->rows - 2, i); } symbol->row_height[symbol->rows - 2] = 1; /* top separator */ rss14_separator(symbol, 50, symbol->rows - 3, -1 /*below*/, 18, 0, 0 /*bottom_finder_value_3*/); symbol->row_height[symbol->rows - 3] = 1; /* bottom separator */ /* 17 == 2 (guard) + 15 (inner char); +2 to skip over finder elements 4 & 5 (right to left) */ rss14_separator(symbol, 50, symbol->rows - 1, 1 /*above*/, 17 + 2, 0, c_right == 3); symbol->row_height[symbol->rows - 1] = 1; if (symbol->width < 50) { symbol->width = 50; } if (symbol->symbology == BARCODE_RSS14_OMNI_CC) { /* separator pattern for composite symbol */ rss14_separator(symbol, 50, separator_row, 1 /*above*/, 18, 0, 0 /*bottom_finder_value_3*/); } symbol->rows = symbol->rows + 1; set_minimum_height(symbol, 33); } return error_number; } /* GS1 DataBar Limited */ INTERNAL int rsslimited(struct zint_symbol *symbol, unsigned char source[], int src_len) { int error_number = 0, i; large_int accum; uint64_t left_character, right_character; int left_group, right_group, left_odd, left_even, right_odd, right_even; int left_widths[14], right_widths[14]; int checksum, check_elements[14], total_widths[46], writer; char latch; int separator_row; int widths[7]; separator_row = 0; if (src_len > 14) { /* Allow check digit to be specified (will be verified and ignored) */ strcpy(symbol->errtxt, "382: Input too long"); return ZINT_ERROR_TOO_LONG; } error_number = is_sane(NEON, source, src_len); if (error_number == ZINT_ERROR_INVALID_DATA) { strcpy(symbol->errtxt, "383: Invalid characters in data"); return error_number; } if (src_len == 14) { /* Verify check digit */ if (calc_check_digit(source) != ctoi(source[13])) { strcpy(symbol->errtxt, "389: Invalid check digit"); return ZINT_ERROR_INVALID_CHECK; } src_len--; /* Ignore */ } if (src_len == 13) { if ((source[0] != '0') && (source[0] != '1')) { strcpy(symbol->errtxt, "384: Input out of range"); return ZINT_ERROR_INVALID_DATA; } } /* make some room for a separator row for composite symbols */ if (symbol->symbology == BARCODE_RSS_LTD_CC) { separator_row = symbol->rows; symbol->row_height[separator_row] = 1; symbol->rows += 1; } large_load_str_u64(&accum, source, src_len); if (symbol->option_1 == 2) { /* Add symbol linkage flag */ large_add_u64(&accum, 2015133531096); } /* Calculate left and right pair values */ right_character = large_div_u64(&accum, 2013571); left_character = large_lo(&accum); if (left_character >= 1996939) { left_group = 6; left_character -= 1996939; } else if (left_character >= 1979845) { left_group = 5; left_character -= 1979845; } else if (left_character >= 1491021) { left_group = 4; left_character -= 1491021; } else if (left_character >= 1000776) { left_group = 3; left_character -= 1000776; } else if (left_character >= 820064) { left_group = 2; left_character -= 820064; } else if (left_character >= 183064) { left_group = 1; left_character -= 183064; } else { left_group = 0; } if (right_character >= 1996939) { right_group = 6; right_character -= 1996939; } else if (right_character >= 1979845) { right_group = 5; right_character -= 1979845; } else if (right_character >= 1491021) { right_group = 4; right_character -= 1491021; } else if (right_character >= 1000776) { right_group = 3; right_character -= 1000776; } else if (right_character >= 820064) { right_group = 2; right_character -= 820064; } else if (right_character >= 183064) { right_group = 1; right_character -= 183064; } else { right_group = 0; } left_odd = left_character / t_even_ltd[left_group]; left_even = left_character % t_even_ltd[left_group]; right_odd = right_character / t_even_ltd[right_group]; right_even = right_character % t_even_ltd[right_group]; getRSSwidths(widths, left_odd, modules_odd_ltd[left_group], 7, widest_odd_ltd[left_group], 1); for (i = 0; i <= 6; i++) { left_widths[i * 2] = widths[i]; } getRSSwidths(widths, left_even, modules_even_ltd[left_group], 7, widest_even_ltd[left_group], 0); for (i = 0; i <= 6; i++) { left_widths[i * 2 + 1] = widths[i]; } getRSSwidths(widths, right_odd, modules_odd_ltd[right_group], 7, widest_odd_ltd[right_group], 1); for (i = 0; i <= 6; i++) { right_widths[i * 2] = widths[i]; } getRSSwidths(widths, right_even, modules_even_ltd[right_group], 7, widest_even_ltd[right_group], 0); for (i = 0; i <= 6; i++) { right_widths[i * 2 + 1] = widths[i]; } checksum = 0; /* Calculate the checksum */ for (i = 0; i < 14; i++) { checksum += checksum_weight_ltd[i] * left_widths[i]; checksum += checksum_weight_ltd[i + 14] * right_widths[i]; } checksum %= 89; for (i = 0; i < 14; i++) { check_elements[i] = finder_pattern_ltd[i + (checksum * 14)]; } total_widths[0] = 1; total_widths[1] = 1; total_widths[44] = 1; total_widths[45] = 1; for (i = 0; i < 14; i++) { total_widths[i + 2] = left_widths[i]; total_widths[i + 16] = check_elements[i]; total_widths[i + 30] = right_widths[i]; } writer = 0; latch = '0'; for (i = 0; i < 46; i++) { writer = rss_expand(symbol, writer, &latch, total_widths[i]); } if (symbol->width < writer) { symbol->width = writer; } symbol->rows = symbol->rows + 1; /* add separator pattern if composite symbol */ if (symbol->symbology == BARCODE_RSS_LTD_CC) { for (i = 4; i < 70; i++) { if (!(module_is_set(symbol, separator_row + 1, i))) { set_module(symbol, separator_row, i); } } } /* Set human readable text */ set_gtin14_hrt(symbol, source, src_len); set_minimum_height(symbol, 10); return error_number; } /* Handles all data encodation from section 7.2.5 of ISO/IEC 24724 */ static int rss_binary_string(struct zint_symbol *symbol, char source[], char binary_string[]) { int encoding_method, i, j, read_posn, last_digit, debug = (symbol->debug & ZINT_DEBUG_PRINT), mode = NUMERIC; int symbol_characters, characters_per_row; #ifndef _MSC_VER char general_field[strlen(source) + 1]; #else char* general_field = (char*) _alloca(strlen(source) + 1); #endif int remainder, d1, d2; char padstring[40]; /* Decide whether a compressed data field is required and if so what method to use - method 2 = no compressed data field */ if ((strlen(source) >= 16) && ((source[0] == '0') && (source[1] == '1'))) { /* (01) and other AIs */ encoding_method = 1; if (debug) printf("Choosing Method 1\n"); } else { /* any AIs */ encoding_method = 2; if (debug) printf("Choosing Method 2\n"); } if (((strlen(source) >= 20) && (encoding_method == 1)) && ((source[2] == '9') && (source[16] == '3'))) { /* Possibly encoding method > 2 */ if (debug) printf("Checking for other methods\n"); if ((strlen(source) >= 26) && (source[17] == '1')) { /* Methods 3, 7, 9, 11 and 13 */ if (source[18] == '0') { /* (01) and (310x) */ char weight_str[7]; for (i = 0; i < 6; i++) { weight_str[i] = source[20 + i]; } weight_str[6] = '\0'; if (weight_str[0] == '0') { /* Maximum weight = 99999 */ if ((source[19] == '3') && (strlen(source) == 26)) { /* (01) and (3103) */ float weight; /* In kilos */ weight = atof(weight_str) / 1000.0; if (weight <= 32.767) { encoding_method = 3; } } if (strlen(source) == 34) { if ((source[26] == '1') && (source[27] == '1')) { /* (01), (310x) and (11) - metric weight and production date */ encoding_method = 7; } if ((source[26] == '1') && (source[27] == '3')) { /* (01), (310x) and (13) - metric weight and packaging date */ encoding_method = 9; } if ((source[26] == '1') && (source[27] == '5')) { /* (01), (310x) and (15) - metric weight and "best before" date */ encoding_method = 11; } if ((source[26] == '1') && (source[27] == '7')) { /* (01), (310x) and (17) - metric weight and expiration date */ encoding_method = 13; } } } } if (debug) printf("Now using method %d\n", encoding_method); } if ((strlen(source) >= 26) && (source[17] == '2')) { /* Methods 4, 8, 10, 12 and 14 */ if (source[18] == '0') { /* (01) and (320x) */ char weight_str[7]; for (i = 0; i < 6; i++) { weight_str[i] = source[20 + i]; } weight_str[6] = '\0'; if (weight_str[0] == '0') { /* Maximum weight = 99999 */ if (((source[19] == '2') || (source[19] == '3')) && (strlen(source) == 26)) { /* (01) and (3202)/(3203) */ float weight; /* In pounds */ if (source[19] == '3') { weight = (float) (atof(weight_str) / 1000.0F); if (weight <= 22.767) { encoding_method = 4; } } else { weight = (float) (atof(weight_str) / 100.0F); if (weight <= 99.99) { encoding_method = 4; } } } if (strlen(source) == 34) { if ((source[26] == '1') && (source[27] == '1')) { /* (01), (320x) and (11) - English weight and production date */ encoding_method = 8; } if ((source[26] == '1') && (source[27] == '3')) { /* (01), (320x) and (13) - English weight and packaging date */ encoding_method = 10; } if ((source[26] == '1') && (source[27] == '5')) { /* (01), (320x) and (15) - English weight and "best before" date */ encoding_method = 12; } if ((source[26] == '1') && (source[27] == '7')) { /* (01), (320x) and (17) - English weight and expiration date */ encoding_method = 14; } } } } if (debug) printf("Now using method %d\n", encoding_method); } if (source[17] == '9') { /* Methods 5 and 6 */ if ((source[18] == '2') && ((source[19] >= '0') && (source[19] <= '3'))) { /* (01) and (392x) */ encoding_method = 5; } if ((source[18] == '3') && ((source[19] >= '0') && (source[19] <= '3'))) { /* (01) and (393x) */ encoding_method = 6; } if (debug) printf("Now using method %d\n", encoding_method); } } switch (encoding_method) { /* Encoding method - Table 10 */ case 1: strcat(binary_string, "1XX"); read_posn = 16; break; case 2: strcat(binary_string, "00XX"); read_posn = 0; break; case 3: // 0100 case 4: // 0101 bin_append(4 + (encoding_method - 3), 4, binary_string); read_posn = strlen(source); break; case 5: strcat(binary_string, "01100XX"); read_posn = 20; break; case 6: strcat(binary_string, "01101XX"); read_posn = 23; break; default: /* modes 7 to 14 */ bin_append(56 + (encoding_method - 7), 7, binary_string); read_posn = strlen(source); break; } if (debug) printf("Setting binary = %s\n", binary_string); /* Variable length symbol bit field is just given a place holder (XX) for the time being */ /* Verify that the data to be placed in the compressed data field is all numeric data before carrying out compression */ for (i = 0; i < read_posn; i++) { if ((source[i] < '0') || (source[i] > '9')) { if ((source[i] != '[') && (source[i] != ']')) { /* Something is wrong */ strcpy(symbol->errtxt, "385: Invalid characters in input data"); return ZINT_ERROR_INVALID_DATA; } } } /* Now encode the compressed data field */ if (debug) printf("Proceeding to encode data\n"); if (encoding_method == 1) { /* Encoding method field "1" - general item identification data */ char group[4]; group[0] = source[2]; group[1] = '\0'; bin_append(atoi(group), 4, binary_string); for (i = 1; i < 5; i++) { group[0] = source[(i * 3)]; group[1] = source[(i * 3) + 1]; group[2] = source[(i * 3) + 2]; group[3] = '\0'; bin_append(atoi(group), 10, binary_string); } } if ((encoding_method == 3) || (encoding_method == 4)) { /* Encoding method field "0100" - variable weight item (0,001 kilogram icrements) */ /* Encoding method field "0101" - variable weight item (0,01 or 0,001 pound increment) */ char group[4]; char weight_str[7]; for (i = 1; i < 5; i++) { group[0] = source[(i * 3)]; group[1] = source[(i * 3) + 1]; group[2] = source[(i * 3) + 2]; group[3] = '\0'; bin_append(atoi(group), 10, binary_string); } for (i = 0; i < 6; i++) { weight_str[i] = source[20 + i]; } weight_str[6] = '\0'; if ((encoding_method == 4) && (source[19] == '3')) { bin_append(atoi(weight_str) + 10000, 15, binary_string); } else { bin_append(atoi(weight_str), 15, binary_string); } } if ((encoding_method == 5) || (encoding_method == 6)) { /* Encoding method "01100" - variable measure item and price */ /* Encoding method "01101" - variable measure item and price with ISO 4217 Currency Code */ char group[4]; for (i = 1; i < 5; i++) { group[0] = source[(i * 3)]; group[1] = source[(i * 3) + 1]; group[2] = source[(i * 3) + 2]; group[3] = '\0'; bin_append(atoi(group), 10, binary_string); } bin_append(source[19] - '0', 2, binary_string); if (encoding_method == 6) { char currency_str[5]; for (i = 0; i < 3; i++) { currency_str[i] = source[20 + i]; } currency_str[3] = '\0'; bin_append(atoi(currency_str), 10, binary_string); } } if ((encoding_method >= 7) && (encoding_method <= 14)) { /* Encoding method fields "0111000" through "0111111" - variable weight item plus date */ char group[4]; int group_val; char weight_str[8]; for (i = 1; i < 5; i++) { group[0] = source[(i * 3)]; group[1] = source[(i * 3) + 1]; group[2] = source[(i * 3) + 2]; group[3] = '\0'; bin_append(atoi(group), 10, binary_string); } weight_str[0] = source[19]; for (i = 0; i < 5; i++) { weight_str[i + 1] = source[21 + i]; } weight_str[6] = '\0'; bin_append(atoi(weight_str), 20, binary_string); if (strlen(source) == 34) { /* Date information is included */ char date_str[4]; date_str[0] = source[28]; date_str[1] = source[29]; date_str[2] = '\0'; group_val = atoi(date_str) * 384; date_str[0] = source[30]; date_str[1] = source[31]; group_val += (atoi(date_str) - 1) * 32; date_str[0] = source[32]; date_str[1] = source[33]; group_val += atoi(date_str); } else { group_val = 38400; } bin_append(group_val, 16, binary_string); } /* The compressed data field has been processed if appropriate - the rest of the data (if any) goes into a general-purpose data compaction field */ j = 0; for (i = read_posn; i < (int) strlen(source); i++) { general_field[j] = source[i]; j++; } general_field[j] = '\0'; if (debug) printf("General field data = %s\n", general_field); if (!general_field_encode(general_field, &mode, &last_digit, binary_string)) { /* Invalid characters in input data */ strcpy(symbol->errtxt, "386: Invalid characters in input data"); return ZINT_ERROR_INVALID_DATA; } if (debug) printf("Resultant binary = %s\n", binary_string); if (debug) printf("\tLength: %d\n", (int) strlen(binary_string)); remainder = 12 - (strlen(binary_string) % 12); if (remainder == 12) { remainder = 0; } symbol_characters = ((strlen(binary_string) + remainder) / 12) + 1; if ((symbol->symbology == BARCODE_RSS_EXPSTACK) || (symbol->symbology == BARCODE_RSS_EXPSTACK_CC)) { characters_per_row = symbol->option_2 * 2; if ((characters_per_row < 2) || (characters_per_row > 20)) { characters_per_row = 4; } if ((symbol_characters % characters_per_row) == 1) { symbol_characters++; } } if (symbol_characters < 4) { symbol_characters = 4; } remainder = (12 * (symbol_characters - 1)) - strlen(binary_string); if (last_digit) { /* There is still one more numeric digit to encode */ if (debug) printf("Adding extra (odd) numeric digit\n"); if ((remainder >= 4) && (remainder <= 6)) { bin_append(ctoi(last_digit) + 1, 4, binary_string); } else { d1 = ctoi(last_digit); d2 = 10; bin_append((11 * d1) + d2 + 8, 7, binary_string); } remainder = 12 - (strlen(binary_string) % 12); if (remainder == 12) { remainder = 0; } symbol_characters = ((strlen(binary_string) + remainder) / 12) + 1; if ((symbol->symbology == BARCODE_RSS_EXPSTACK) || (symbol->symbology == BARCODE_RSS_EXPSTACK_CC)) { characters_per_row = symbol->option_2 * 2; if ((characters_per_row < 2) || (characters_per_row > 20)) { characters_per_row = 4; } if ((symbol_characters % characters_per_row) == 1) { symbol_characters++; } } if (symbol_characters < 4) { symbol_characters = 4; } remainder = (12 * (symbol_characters - 1)) - strlen(binary_string); if (debug) printf("Resultant binary = %s\n", binary_string); if (debug) printf("\tLength: %d\n", (int) strlen(binary_string)); } if (strlen(binary_string) > 252) { /* 252 = (21 * 12) */ strcpy(symbol->errtxt, "387: Input too long"); return ZINT_ERROR_TOO_LONG; } /* Now add padding to binary string (7.2.5.5.4) */ i = remainder; if (mode == NUMERIC) { strcpy(padstring, "0000"); i -= 4; } else { strcpy(padstring, ""); } for (; i > 0; i -= 5) { strcat(padstring, "00100"); } padstring[remainder] = '\0'; strcat(binary_string, padstring); /* Patch variable length symbol bit field */ d1 = symbol_characters & 1; if (symbol_characters <= 14) { d2 = 0; } else { d2 = 1; } if (encoding_method == 1) { binary_string[2] = d1 ? '1' : '0'; binary_string[3] = d2 ? '1' : '0'; } if (encoding_method == 2) { binary_string[3] = d1 ? '1' : '0'; binary_string[4] = d2 ? '1' : '0'; } if ((encoding_method == 5) || (encoding_method == 6)) { binary_string[6] = d1 ? '1' : '0'; binary_string[7] = d2 ? '1' : '0'; } if (debug) printf("Resultant binary = %s\n", binary_string); if (debug) printf("\tLength: %d\n", (int) strlen(binary_string)); return 0; } static void rssexp_separator(struct zint_symbol *symbol, int width, int cols, int separator_row, int above_below, int special_case_row, int left_to_right, int odd_last_row, int *p_v2_latch) { int i, i_start, i_end, j, k; int module_row = separator_row + above_below; int v2_latch = p_v2_latch ? *p_v2_latch : 0; int space_latch = 0; for (j = 4 + special_case_row, width -= 4; j < width; j++) { if (module_is_set(symbol, module_row, j)) { unset_module(symbol, separator_row, j); } else { set_module(symbol, separator_row, j); } } /* finder adjustment */ for (j = 0; j < cols; j++) { k = (49 * j) + 19 + special_case_row; /* 49 == data (17) + finder (15) + data(17) triplet, 19 == 2 (guard) + 17 (initial check/data character) */ if (left_to_right) { i_start = v2_latch ? 2 : 0; /* Last 13 modules of version 2 finder and first 13 modules of version 1 finder */ i_end = v2_latch ? 15 : 13; for (i = i_start; i < i_end; i++) { if (module_is_set(symbol, module_row, i + k)) { unset_module(symbol, separator_row, i + k); space_latch = 0; } else { if (space_latch) { unset_module(symbol, separator_row, i + k); } else { set_module(symbol, separator_row, i + k); } space_latch = !space_latch; } } } else { if (odd_last_row) { k -= 17; /* No data char at beginning of row, i.e. ends with finder */ } i_start = v2_latch ? 14 : 12; /* First 13 modules of version 1 finder and last 13 modules of version 2 finder */ i_end = v2_latch ? 2 : 0; for (i = i_start; i >= i_end; i--) { if (module_is_set(symbol, module_row, i + k)) { unset_module(symbol, separator_row, i + k); space_latch = 0; } else { if (space_latch) { unset_module(symbol, separator_row, i + k); } else { set_module(symbol, separator_row, i + k); } space_latch = !space_latch; } } } v2_latch = !v2_latch; } if (p_v2_latch && above_below == -1) { /* Only set if below */ *p_v2_latch = v2_latch; } } /* GS1 DataBar Expanded */ INTERNAL int rssexpanded(struct zint_symbol *symbol, unsigned char source[], int src_len) { int i, j, k, p, data_chars, vs[21], group[21], v_odd[21], v_even[21]; char substring[21][14], latch; int char_widths[21][8], checksum, check_widths[8], c_group; int check_char, c_odd, c_even, elements[235], pattern_width, reader, writer; int separator_row; unsigned int bin_len = 13 * src_len + 200 + 1; /* Allow for 8 bits + 5-bit latch per char + 200 bits overhead/padding */ int widths[4]; #ifndef _MSC_VER char reduced[src_len + 1], binary_string[bin_len]; #else char* reduced = (char*) _alloca(src_len + 1); char* binary_string = (char*) _alloca(bin_len); #endif i = gs1_verify(symbol, source, src_len, reduced); if (i != 0) { return i; } if ((symbol->symbology == BARCODE_RSS_EXP_CC) || (symbol->symbology == BARCODE_RSS_EXPSTACK_CC)) { /* make space for a composite separator pattern */ separator_row = symbol->rows; symbol->row_height[separator_row] = 1; symbol->rows += 1; } strcpy(binary_string, ""); if (symbol->option_1 == 2) { /* The "component linkage" flag */ strcat(binary_string, "1"); } else { strcat(binary_string, "0"); } i = rss_binary_string(symbol, reduced, binary_string); if (i != 0) { return i; } data_chars = strlen(binary_string) / 12; for (i = 0; i < data_chars; i++) { for (j = 0; j < 12; j++) { substring[i][j] = binary_string[(i * 12) + j]; } substring[i][12] = '\0'; } for (i = 0; i < data_chars; i++) { vs[i] = 0; for (p = 0; p < 12; p++) { if (substring[i][p] == '1') { vs[i] += (0x800 >> p); } } } for (i = 0; i < data_chars; i++) { if (vs[i] <= 347) { group[i] = 1; } if ((vs[i] >= 348) && (vs[i] <= 1387)) { group[i] = 2; } if ((vs[i] >= 1388) && (vs[i] <= 2947)) { group[i] = 3; } if ((vs[i] >= 2948) && (vs[i] <= 3987)) { group[i] = 4; } if (vs[i] >= 3988) { group[i] = 5; } v_odd[i] = (vs[i] - g_sum_exp[group[i] - 1]) / t_even_exp[group[i] - 1]; v_even[i] = (vs[i] - g_sum_exp[group[i] - 1]) % t_even_exp[group[i] - 1]; getRSSwidths(widths, v_odd[i], modules_odd_exp[group[i] - 1], 4, widest_odd_exp[group[i] - 1], 0); char_widths[i][0] = widths[0]; char_widths[i][2] = widths[1]; char_widths[i][4] = widths[2]; char_widths[i][6] = widths[3]; getRSSwidths(widths, v_even[i], modules_even_exp[group[i] - 1], 4, widest_even_exp[group[i] - 1], 1); char_widths[i][1] = widths[0]; char_widths[i][3] = widths[1]; char_widths[i][5] = widths[2]; char_widths[i][7] = widths[3]; } /* 7.2.6 Check character */ /* The checksum value is equal to the mod 211 residue of the weighted sum of the widths of the elements in the data characters. */ checksum = 0; for (i = 0; i < data_chars; i++) { int row = weight_rows[(((data_chars - 2) / 2) * 21) + i]; for (j = 0; j < 8; j++) { checksum += (char_widths[i][j] * checksum_weight_exp[(row * 8) + j]); } } check_char = (211 * ((data_chars + 1) - 4)) + (checksum % 211); if (symbol->debug & ZINT_DEBUG_PRINT) { printf("Data chars: %d, Check char: %d\n", data_chars, check_char); } if (check_char <= 347) { c_group = 1; } if ((check_char >= 348) && (check_char <= 1387)) { c_group = 2; } if ((check_char >= 1388) && (check_char <= 2947)) { c_group = 3; } if ((check_char >= 2948) && (check_char <= 3987)) { c_group = 4; } if (check_char >= 3988) { c_group = 5; } c_odd = (check_char - g_sum_exp[c_group - 1]) / t_even_exp[c_group - 1]; c_even = (check_char - g_sum_exp[c_group - 1]) % t_even_exp[c_group - 1]; getRSSwidths(widths, c_odd, modules_odd_exp[c_group - 1], 4, widest_odd_exp[c_group - 1], 0); check_widths[0] = widths[0]; check_widths[2] = widths[1]; check_widths[4] = widths[2]; check_widths[6] = widths[3]; getRSSwidths(widths, c_even, modules_even_exp[c_group - 1], 4, widest_even_exp[c_group - 1], 1); check_widths[1] = widths[0]; check_widths[3] = widths[1]; check_widths[5] = widths[2]; check_widths[7] = widths[3]; /* Initialise element array */ pattern_width = ((((data_chars + 1) / 2) + ((data_chars + 1) & 1)) * 5) + ((data_chars + 1) * 8) + 4; for (i = 0; i < pattern_width; i++) { elements[i] = 0; } /* Put finder patterns in element array */ for (i = 0; i < (((data_chars + 1) / 2) + ((data_chars + 1) & 1)); i++) { k = ((((((data_chars + 1) - 2) / 2) + ((data_chars + 1) & 1)) - 1) * 11) + i; for (j = 0; j < 5; j++) { elements[(21 * i) + j + 10] = finder_pattern_exp[((finder_sequence[k] - 1) * 5) + j]; } } /* Put check character in element array */ for (i = 0; i < 8; i++) { elements[i + 2] = check_widths[i]; } /* Put forward reading data characters in element array */ for (i = 1; i < data_chars; i += 2) { for (j = 0; j < 8; j++) { elements[(((i - 1) / 2) * 21) + 23 + j] = char_widths[i][j]; } } /* Put reversed data characters in element array */ for (i = 0; i < data_chars; i += 2) { for (j = 0; j < 8; j++) { elements[((i / 2) * 21) + 15 + j] = char_widths[i][7 - j]; } } if ((symbol->symbology == BARCODE_RSS_EXP) || (symbol->symbology == BARCODE_RSS_EXP_CC)) { /* Copy elements into symbol */ elements[0] = 1; // left guard elements[1] = 1; elements[pattern_width - 2] = 1; // right guard elements[pattern_width - 1] = 1; writer = 0; latch = '0'; for (i = 0; i < pattern_width; i++) { writer = rss_expand(symbol, writer, &latch, elements[i]); } if (symbol->width < writer) { symbol->width = writer; } symbol->rows = symbol->rows + 1; /* Add human readable text */ for (i = 0; i <= src_len; i++) { if ((source[i] != '[') && (source[i] != ']')) { symbol->text[i] = source[i]; } else { if (source[i] == '[') { symbol->text[i] = '('; } if (source[i] == ']') { symbol->text[i] = ')'; } } } } else { int stack_rows; int current_row, current_block, left_to_right; int v2_latch = 0; /* RSS Expanded Stacked */ /* Bug corrected: Character missing for message * [01]90614141999996[10]1234222222222221 * Patch by Daniel Frede */ int codeblocks = (data_chars + 1) / 2 + ((data_chars + 1) % 2); if ((symbol->option_2 < 1) || (symbol->option_2 > 11)) { symbol->option_2 = 2; } if ((symbol->option_1 == 2) && (symbol->option_2 == 1)) { /* "There shall be a minimum of four symbol characters in the first row of an RSS Expanded Stacked symbol when it is the linear component of an EAN.UCC Composite symbol." */ symbol->option_2 = 2; } stack_rows = codeblocks / symbol->option_2; if (codeblocks % symbol->option_2 > 0) { stack_rows++; } current_block = 0; for (current_row = 1; current_row <= stack_rows; current_row++) { int special_case_row = 0; int elements_in_sub; int sub_elements[235] = {0}; int num_columns; /* Number of columns in current row */ if (current_row * symbol->option_2 > codeblocks) { num_columns = codeblocks - current_block; } else { num_columns = symbol->option_2; } /* Row Start */ sub_elements[0] = 1; // left guard sub_elements[1] = 1; elements_in_sub = 2; /* If last row and is partial and even-numbered, and have even columns (segment pairs), and odd number of finders */ if ((current_row == stack_rows) && (num_columns != symbol->option_2) && !(current_row & 1) && !(symbol->option_2 & 1) && (num_columns & 1)) { /* Odd number of finders == odd number of columns */ /* Special case bottom row */ special_case_row = 1; sub_elements[0] = 2; /* Extra space (latch set below) */ } /* If odd number of columns or current row odd-numbered or special case last row then left-to-right, else right-to-left */ if ((symbol->option_2 & 1) || (current_row & 1) || special_case_row) { left_to_right = 1; } else { left_to_right = 0; } if (symbol->debug & ZINT_DEBUG_PRINT) { if (current_row == stack_rows) { printf("Last row: number of columns: %d / %d, left to right: %d, special case: %d\n", num_columns, symbol->option_2, left_to_right, special_case_row); } } /* Row Data */ reader = 0; do { i = 2 + (current_block * 21); for (j = 0; j < 21; j++) { if ((i + j) < pattern_width) { if (left_to_right) { sub_elements[j + (reader * 21) + 2] = elements[i + j]; } else { sub_elements[(20 - j) + (num_columns - 1 - reader) * 21 + 2] = elements[i + j]; } } elements_in_sub++; } reader++; current_block++; } while ((reader < symbol->option_2) && (current_block < codeblocks)); /* Row Stop */ sub_elements[elements_in_sub] = 1; // right guard sub_elements[elements_in_sub + 1] = 1; elements_in_sub += 2; latch = (current_row & 1) || special_case_row ? '0' : '1'; writer = 0; for (i = 0; i < elements_in_sub; i++) { writer = rss_expand(symbol, writer, &latch, sub_elements[i]); } if (symbol->width < writer) { symbol->width = writer; } if (current_row != 1) { int odd_last_row = (current_row == stack_rows) && (data_chars % 2 == 0); /* middle separator pattern (above current row) */ for (j = 5; j < (49 * symbol->option_2); j += 2) { set_module(symbol, symbol->rows - 2, j); } symbol->row_height[symbol->rows - 2] = 1; /* bottom separator pattern (above current row) */ rssexp_separator(symbol, writer, reader, symbol->rows - 1, 1 /*above*/, special_case_row, left_to_right, odd_last_row, &v2_latch); symbol->row_height[symbol->rows - 1] = 1; } if (current_row != stack_rows) { /* top separator pattern (below current row) */ rssexp_separator(symbol, writer, reader, symbol->rows + 1, -1 /*below*/, 0 /*special_case_row*/, left_to_right, 0 /*odd_last_row*/, &v2_latch); symbol->row_height[symbol->rows + 1] = 1; } symbol->rows = symbol->rows + 4; } symbol->rows = symbol->rows - 3; } if (symbol->symbology == BARCODE_RSS_EXP_CC || symbol->symbology == BARCODE_RSS_EXPSTACK_CC) { /* Composite separator */ rssexp_separator(symbol, symbol->width, 4, separator_row, 1 /*above*/, 0 /*special_case_row*/, 1 /*left_to_right*/, 0 /*odd_last_row*/, NULL); } for (i = 0; i < symbol->rows; i++) { if (symbol->row_height[i] == 0) { symbol->row_height[i] = 34; } } return 0; }