/* * Copyright 2009 ZXing authors * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ using System; using ReaderException = com.google.zxing.ReaderException; using BitMatrix = com.google.zxing.common.BitMatrix; using DecoderResult = com.google.zxing.common.DecoderResult; namespace com.google.zxing.pdf417.decoder { //import com.google.zxing.pdf417.reedsolomon.ReedSolomonDecoder; ///

The main class which implements PDF417 Code decoding -- as /// opposed to locating and extracting the PDF417 Code from an image.

/// ///
/// SITA Lab (kevin.osullivan@sita.aero) /// /// www.Redivivus.in (suraj.supekar@redivivus.in) - Ported from ZXING Java Source /// public sealed class Decoder { private const int MAX_ERRORS = 3; private const int MAX_EC_CODEWORDS = 512; //private final ReedSolomonDecoder rsDecoder; public Decoder() { // TODO MGMG //rsDecoder = new ReedSolomonDecoder(); } ///

Convenience method that can decode a PDF417 Code represented as a 2D array of booleans. /// "true" is taken to mean a black module.

/// ///
/// booleans representing white/black PDF417 modules /// /// text and bytes encoded within the PDF417 Code /// /// ReaderException if the PDF417 Code cannot be decoded public DecoderResult decode(bool[][] image) { int dimension = image.Length; BitMatrix bits = new BitMatrix(dimension); for (int i = 0; i < dimension; i++) { for (int j = 0; j < dimension; j++) { if (image[j][i]) { bits.set_Renamed(j, i); } } } return decode(bits); } ///

Decodes a PDF417 Code represented as a {@link BitMatrix}. /// A 1 or "true" is taken to mean a black module.

/// ///
/// booleans representing white/black PDF417 Code modules /// /// text and bytes encoded within the PDF417 Code /// /// ReaderException if the PDF417 Code cannot be decoded public DecoderResult decode(BitMatrix bits) { // Construct a parser to read the data codewords and error-correction level BitMatrixParser parser = new BitMatrixParser(bits); int[] codewords = parser.readCodewords(); if (codewords == null || codewords.Length == 0) { throw ReaderException.Instance; } int ecLevel = parser.ECLevel; int numECCodewords = 1 << (ecLevel + 1); int[] erasures = parser.Erasures; correctErrors(codewords, erasures, numECCodewords); verifyCodewordCount(codewords, numECCodewords); // Decode the codewords return DecodedBitStreamParser.decode(codewords); } /// Verify that all is OK with the codeword array. /// /// /// /// /// an index to the first data codeword. /// /// ReaderException private static void verifyCodewordCount(int[] codewords, int numECCodewords) { if (codewords.Length < 4) { // Codeword array size should be at least 4 allowing for // Count CW, At least one Data CW, Error Correction CW, Error Correction CW throw ReaderException.Instance; } // The first codeword, the Symbol Length Descriptor, shall always encode the total number of data // codewords in the symbol, including the Symbol Length Descriptor itself, data codewords and pad // codewords, but excluding the number of error correction codewords. int numberOfCodewords = codewords[0]; if (numberOfCodewords > codewords.Length) { throw ReaderException.Instance; } if (numberOfCodewords == 0) { // Reset to the length of the array - 8 (Allow for at least level 3 Error Correction (8 Error Codewords) if (numECCodewords < codewords.Length) { codewords[0] = codewords.Length - numECCodewords; } else { throw ReaderException.Instance; } } } ///

Given data and error-correction codewords received, possibly corrupted by errors, attempts to /// correct the errors in-place using Reed-Solomon error correction.

/// ///
/// data and error correction codewords /// /// ReaderException if error correction fails private static int correctErrors(int[] codewords, int[] erasures, int numECCodewords) { if ((erasures != null && erasures.Length > numECCodewords / 2 + MAX_ERRORS) || (numECCodewords < 0 || numECCodewords > MAX_EC_CODEWORDS)) { // Too many errors or EC Codewords is corrupted throw ReaderException.Instance; } // Try to correct the errors int result = 0; // rsDecoder.correctErrors(codewords, numECCodewords); if (erasures != null) { int numErasures = erasures.Length; if (result > 0) { numErasures -= result; } if (numErasures > MAX_ERRORS) { // Still too many errors throw ReaderException.Instance; } } return result; } } }