All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.itextpdf.barcodes.qrcode.BitVector Maven / Gradle / Ivy

/*
    This file is part of the iText (R) project.
    Copyright (c) 1998-2023 Apryse Group NV
    Authors: Apryse Software.

    This program is offered under a commercial and under the AGPL license.
    For commercial licensing, contact us at https://itextpdf.com/sales.  For AGPL licensing, see below.

    AGPL licensing:
    This program is free software: you can redistribute it and/or modify
    it under the terms of the GNU Affero 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 Affero General Public License for more details.

    You should have received a copy of the GNU Affero General Public License
    along with this program.  If not, see .
 */
package com.itextpdf.barcodes.qrcode;

/**
 * JAVAPORT: This should be combined with BitArray in the future, although that class is not yet
 * dynamically resizeable. This implementation is reasonable but there is a lot of function calling
 * in loops I'd like to get rid of.
 *
 * @author [email protected] (Satoru Takabayashi) - creator
 * @author [email protected] (Daniel Switkin) - ported from C++
 */
final class BitVector {

    private int sizeInBits;
    private byte[] array;

    // For efficiency, start out with some room to work.
    private static final int DEFAULT_SIZE_IN_BYTES = 32;

    /**
     * Create a bitvector usng the default size
     */
    public BitVector() {
        sizeInBits = 0;
        array = new byte[DEFAULT_SIZE_IN_BYTES];
    }

    // Return the bit value at "index".

    /**
     * Return the bit value at "index".
     * @param index index in the vector
     * @return bit value at "index"
     */
    public int at(int index) {
        if (index < 0 || index >= sizeInBits) {
            throw new IllegalArgumentException("Bad index: " + index);
        }
        int value = array[index >> 3] & 0xff;
        return (value >> (7 - (index & 0x7))) & 1;
    }

    /**
     * @return the number of bits in the bit vector.
     */
    public int size() {
        return sizeInBits;
    }


    /**
     * @return the number of bytes in the bit vector.
     */
    public int sizeInBytes() {
        return (sizeInBits + 7) >> 3;
    }

    // Append one bit to the bit vector.

    /**
     * Append the a bit to the bit vector
     * @param bit 0 or 1
     */
    public void appendBit(int bit) {
        if (!(bit == 0 || bit == 1)) {
            throw new IllegalArgumentException("Bad bit");
        }
        int numBitsInLastByte = sizeInBits & 0x7;
        // We'll expand array if we don't have bits in the last byte.
        if (numBitsInLastByte == 0) {
            appendByte(0);
            sizeInBits -= 8;
        }
        // Modify the last byte.
        array[sizeInBits >> 3] |= (byte) (bit << (7 - numBitsInLastByte));
        ++sizeInBits;
    }

    //
    // REQUIRES:
    //
    //
    //
    //
    //

    /**
     * Append "numBits" bits in "value" to the bit vector.
     *
     * Examples:
     * - appendBits(0x00, 1) adds 0.
     * - appendBits(0x00, 4) adds 0000.
     * - appendBits(0xff, 8) adds 11111111.
     * @param value int interpreted as bitvector
     * @param numBits 0 <= numBits <= 32.
     */
    public void appendBits(int value, int numBits) {
        if (numBits < 0 || numBits > 32) {
            throw new IllegalArgumentException("Num bits must be between 0 and 32");
        }
        int numBitsLeft = numBits;
        while (numBitsLeft > 0) {
            // Optimization for byte-oriented appending.
            if ((sizeInBits & 0x7) == 0 && numBitsLeft >= 8) {
                int newByte = (value >> (numBitsLeft - 8)) & 0xff;
                appendByte(newByte);
                numBitsLeft -= 8;
            } else {
                int bit = (value >> (numBitsLeft - 1)) & 1;
                appendBit(bit);
                --numBitsLeft;
            }
        }
    }

    /**
     * Append a different BitVector to this BitVector
     * @param bits BitVector to append
     */
    public void appendBitVector(BitVector bits) {
        int size = bits.size();
        for (int i = 0; i < size; ++i) {
            appendBit(bits.at(i));
        }
    }


    /**
     * XOR the contents of this bitvector with the contetns of "other"
     * @param other Bitvector of equal length
     */
    public void xor(BitVector other) {
        if (sizeInBits != other.size()) {
            throw new IllegalArgumentException("BitVector sizes don't match");
        }
        int sizeInBytes = (sizeInBits + 7) >> 3;
        for (int i = 0; i < sizeInBytes; ++i) {
            // The last byte could be incomplete (i.e. not have 8 bits in
            // it) but there is no problem since 0 XOR 0 == 0.
            array[i] ^= other.array[i];
        }
    }

    // Return String like "01110111" for debugging.

    /**
     * @return String representation of the bitvector
     */
    public String toString() {
        StringBuffer result = new StringBuffer(sizeInBits);
        for (int i = 0; i < sizeInBits; ++i) {
            if (at(i) == 0) {
                result.append('0');
            } else if (at(i) == 1) {
                result.append('1');
            } else {
                throw new IllegalArgumentException("Byte isn't 0 or 1");
            }
        }
        return result.toString();
    }

    //
    //

    /**
     * Callers should not assume that array.length is the exact number of bytes needed to hold
     * sizeInBits - it will typically be larger for efficiency.
     * @return size of the array containing the bitvector
     */
    public byte[] getArray() {
        return array;
    }

    //
    //

    /**
     * Add a new byte to the end, possibly reallocating and doubling the size of the array if we've
     * run out of room.
     * @param value byte to add.
     */
    private void appendByte(int value) {
        if ((sizeInBits >> 3) == array.length) {
            byte[] newArray = new byte[(array.length << 1)];
            System.arraycopy(array, 0, newArray, 0, array.length);
            array = newArray;
        }
        array[sizeInBits >> 3] = (byte) value;
        sizeInBits += 8;
    }

}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy