sl@4: /*************************************************************************** sl@4: * Arrays of Arbitrary Bit Length sl@4: * sl@4: * File : bitarray.cpp sl@4: * Purpose : Provides object with methods for creation and manipulation of sl@4: * arbitrary length arrays of bits. sl@4: * sl@4: * Bit arrays are implemented as vectors of unsigned chars. Bit sl@4: * 0 is the MSB of char 0, and the last bit is the least sl@4: * significant (non-spare) bit of the last unsigned char. sl@4: * sl@4: * Example: array of 20 bits (0 through 19) with 8 bit unsigned sl@4: * chars requires 3 unsigned chars (0 through 2) to sl@4: * store all the bits. sl@4: * sl@4: * char 0 1 2 sl@4: * +--------+--------+--------+ sl@4: * | | | | sl@4: * +--------+--------+--------+ sl@4: * bit 01234567 8911111111111XXXX sl@4: * 012345 6789 sl@4: * sl@4: * Author : Michael Dipperstein sl@4: * Date : July 23, 2004 sl@4: * sl@4: **************************************************************************** sl@4: * HISTORY sl@4: * sl@4: * $Id: bitarray.cpp,v 1.7 2010/02/04 03:31:43 michael Exp $ sl@4: * $Log: bitarray.cpp,v $ sl@4: * Revision 1.7 2010/02/04 03:31:43 michael sl@4: * Replaced vector with an array of unsigned char. sl@4: * sl@4: * Made updates for GCC 4.4. sl@4: * sl@4: * Revision 1.5 2007/08/06 05:23:29 michael sl@4: * Updated for LGPL Version 3. sl@4: * sl@4: * All methods that don't modify object have been made sl@4: * const to increase functionality of const bit_array_c. sl@4: * sl@4: * All assignment operators return a reference to the object being assigned a value so that operator chaining will work. sl@4: * sl@4: * Added >> and << operators. sl@4: * sl@4: * Revision 1.3 2006/04/30 23:34:07 michael sl@4: * Improved performance by incorporating Benjamin Schindler's sl@4: * changes to pass arguments as a reference. sl@4: * sl@4: * Revision 1.2 2004/08/05 22:16:49 michael sl@4: * Add overloads for bitwise operators returning values. sl@4: * Add a more natural looking way to set bit values. sl@4: * sl@4: * Revision 1.1.1.1 2004/08/04 13:28:20 michael sl@4: * bit_array_c sl@4: * sl@4: **************************************************************************** sl@4: * sl@4: * Bitarray: An ANSI C++ class for manipulating arbitrary length bit arrays sl@4: * Copyright (C) 2004, 2006-2007, 2010 by sl@4: * Michael Dipperstein (mdipper@alumni.engr.ucsb.edu) sl@4: * sl@4: * This file is part of the bit array library. sl@4: * sl@4: * The bit array library is free software; you can redistribute it and/or sl@4: * modify it under the terms of the GNU Lesser General Public License as sl@4: * published by the Free Software Foundation; either version 3 of the sl@4: * License, or (at your option) any later version. sl@4: * sl@4: * The bit array library is distributed in the hope that it will be useful, sl@4: * but WITHOUT ANY WARRANTY; without even the implied warranty of sl@4: * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser sl@4: * General Public License for more details. sl@4: * sl@4: * You should have received a copy of the GNU Lesser General Public License sl@4: * along with this program. If not, see . sl@4: * sl@4: ***************************************************************************/ sl@4: sl@4: /*************************************************************************** sl@4: * INCLUDED FILES sl@4: ***************************************************************************/ sl@4: #include sl@4: #include sl@4: #include "bitarray.h" sl@4: sl@4: using namespace std; sl@4: sl@4: /*************************************************************************** sl@4: * MACROS sl@4: ***************************************************************************/ sl@4: sl@4: /* make CHAR_BIT 8 if it's not defined in limits.h */ sl@4: #ifndef CHAR_BIT sl@4: #warning CHAR_BIT not defined. Assuming 8 bits. sl@4: #define CHAR_BIT 8 sl@4: #endif sl@4: sl@4: /* position of bit within character */ sl@4: #define BIT_CHAR(bit) ((bit) / CHAR_BIT) sl@4: sl@4: /* array index for character containing bit */ sl@4: //SL: We had to change that macro since bits in our frame buffer are the other way around sl@4: //TODO: Find a solution to tackle that problem sl@4: //#define BIT_IN_CHAR(bit) (1 << (CHAR_BIT - 1 - ((bit) % CHAR_BIT))) sl@4: #define BIT_IN_CHAR(bit) (1 << (((bit) % CHAR_BIT))) sl@4: sl@4: /* number of characters required to contain number of bits */ sl@4: #define BITS_TO_CHARS(bits) ((((bits) - 1) / CHAR_BIT) + 1) sl@4: sl@4: /* most significant bit in a character */ sl@4: #define MS_BIT (1 << (CHAR_BIT - 1)) sl@4: sl@4: /*************************************************************************** sl@4: * METHODS sl@4: ***************************************************************************/ sl@4: sl@4: /*************************************************************************** sl@4: * Method : bit_array_c - constructor sl@4: * Description: This is the bit_array_c constructor. It reserves memory sl@4: * for the vector storing the array. sl@4: * Parameters : numBits - number of bits in the array sl@4: * Effects : Allocates vectory for array bits sl@4: * Returned : None sl@4: ***************************************************************************/ sl@4: BitArray::BitArray(const int numBits): sl@6: m_NumBits(numBits), sl@6: m_Array(NULL), sl@6: m_OwnsBuffer(true) sl@4: { sl@4: m_SizeInBytes = BITS_TO_CHARS(numBits); sl@4: sl@4: sl@4: /* allocate space for bit array */ sl@4: m_Array = new unsigned char[m_SizeInBytes]; sl@4: sl@4: /* set all bits to 0 */ sl@4: fill_n(m_Array, m_SizeInBytes, 0); sl@4: } sl@4: sl@4: /*************************************************************************** sl@4: * Method : bit_array_c - constructor sl@4: * Description: This is the bit_array_c constructor. It copies the sl@4: * for contents of a vector of unsigned char into the sl@4: * bitarray. sl@4: * Parameters : vect - vector to be copied sl@4: * numBits - number of bits in the array sl@4: * Effects : Allocates vectory for array bits sl@4: * Returned : None sl@4: ***************************************************************************/ sl@6: BitArray::BitArray(unsigned char *array, const int numBits,bool aOwnsBuffer): sl@4: m_NumBits(numBits), sl@6: m_Array(array), sl@6: m_OwnsBuffer(aOwnsBuffer) sl@4: { sl@6: sl@4: } sl@4: sl@4: /*************************************************************************** sl@4: * Method : ~bit_array_c - destructor sl@4: * Description: This is the bit_array_c destructor. At this point it's sl@4: * just a place holder. sl@4: * Parameters : None sl@4: * Effects : None sl@4: * Returned : None sl@4: ***************************************************************************/ sl@4: BitArray::~BitArray(void) sl@4: { sl@6: if (m_OwnsBuffer) sl@6: { sl@6: delete[] m_Array; sl@6: m_Array = NULL; sl@6: } sl@4: } sl@4: sl@4: /*************************************************************************** sl@4: * Method : Dump sl@4: * Description: This method dumps the conents of a bit array to stdout. sl@4: * The format of the dump is a series of bytes represented in sl@4: * hexadecimal. sl@4: * Parameters : outStream - stream to write to sl@4: * Effects : Array contents are dumped to stdout sl@4: * Returned : None sl@4: ***************************************************************************/ sl@4: void BitArray::Dump(std::ostream &outStream) sl@4: { sl@4: int size; sl@4: sl@4: size = BITS_TO_CHARS(m_NumBits); sl@4: sl@4: outStream.width(2); sl@4: outStream.fill('0'); sl@4: outStream << uppercase << hex << (int)(m_Array[0]); /* first byte */ sl@4: sl@4: for (int i = 1; i < size; i++) sl@4: { sl@4: /* remaining bytes with a leading space */ sl@4: outStream << " "; sl@4: outStream.width(2); sl@4: outStream.fill('0'); sl@4: outStream << (int)(m_Array[i]); sl@4: } sl@4: sl@4: outStream << dec; sl@4: } sl@4: sl@4: /*************************************************************************** sl@4: * Method : SetAll sl@4: * Description: This method sets every bit in the bit array to 1. This sl@4: * method uses UCHAR_MAX to determine what it means to set sl@4: * all bits in an unsigned char, so it is crucial that the sl@4: * machine implementation of unsigned char utilizes all of sl@4: * the bits in the memory allocated for an unsigned char. sl@4: * Parameters : None sl@4: * Effects : Each of the bits used in the bit array are set to 1. sl@4: * Unused (spare) bits are set to 0. sl@4: * Returned : None sl@4: ***************************************************************************/ sl@4: void BitArray::SetAll(void) sl@4: { sl@4: int bits, size; sl@4: unsigned char mask; sl@4: sl@4: size = BITS_TO_CHARS(m_NumBits); sl@4: sl@4: /* set bits in all bytes to 1 */ sl@4: fill_n(m_Array, size, UCHAR_MAX); sl@4: sl@4: /* zero any spare bits so increment and decrement are consistent */ sl@4: bits = m_NumBits % CHAR_BIT; sl@4: if (bits != 0) sl@4: { sl@4: mask = UCHAR_MAX << (CHAR_BIT - bits); sl@4: m_Array[BIT_CHAR(m_NumBits - 1)] = mask; sl@4: } sl@4: } sl@4: sl@4: /*************************************************************************** sl@4: * Method : ClearAll sl@4: * Description: This method sets every bit in the bit array to 0. sl@4: * Parameters : None sl@4: * Effects : Each of the bits in the bit array are set to 0. sl@4: * Returned : None sl@4: ***************************************************************************/ sl@4: void BitArray::ClearAll(void) sl@4: { sl@4: int size; sl@4: sl@4: size = BITS_TO_CHARS(m_NumBits); sl@4: sl@4: /* set bits in all bytes to 0 */ sl@4: fill_n(m_Array, size, 0); sl@4: } sl@4: sl@4: /*************************************************************************** sl@4: * Method : SetBit sl@4: * Description: This method sets a bit in the bit array to 1. sl@4: * Parameters : bit - the number of the bit to set sl@4: * Effects : The specified bit will be set to 1. sl@4: * Returned : None sl@4: ***************************************************************************/ sl@4: void BitArray::SetBit(const unsigned int bit) sl@4: { sl@4: if (m_NumBits <= bit) sl@4: { sl@4: return; /* bit out of range */ sl@4: } sl@4: sl@4: m_Array[BIT_CHAR(bit)] |= BIT_IN_CHAR(bit); sl@4: } sl@4: sl@4: /** sl@4: */ sl@4: void BitArray::SetBitValue(const unsigned int bit, bool aValue) sl@4: { sl@4: if (aValue) sl@4: { sl@4: SetBit(bit); sl@4: } sl@4: else sl@4: { sl@4: ClearBit(bit); sl@4: } sl@4: } sl@4: sl@4: /*************************************************************************** sl@4: * Method : ClearBit sl@4: * Description: This method sets a bit in the bit array to 0. sl@4: * Parameters : bit - the number of the bit to clear sl@4: * Effects : The specified bit will be set to 0. sl@4: * Returned : None sl@4: ***************************************************************************/ sl@4: void BitArray::ClearBit(const unsigned int bit) sl@4: { sl@4: unsigned char mask; sl@4: sl@4: if (m_NumBits <= bit) sl@4: { sl@4: return; /* bit out of range */ sl@4: } sl@4: sl@4: /* create a mask to zero out desired bit */ sl@4: mask = BIT_IN_CHAR(bit); sl@4: mask = ~mask; sl@4: sl@4: m_Array[BIT_CHAR(bit)] &= mask; sl@4: } sl@4: sl@4: /*************************************************************************** sl@4: * Method : operator() sl@4: * Description: Overload of the () operator. This method approximates sl@4: * array indices used for assignment. It returns a sl@4: * bit_array_index_c which includes an = method used to sl@4: * set bit values. sl@4: * Parameters : bit - index of array bit sl@4: * Effects : None sl@4: * Returned : bit_array_index_c (pointer to bit) sl@4: ***************************************************************************/ sl@4: BitArrayIndex BitArray::operator()(const unsigned int bit) sl@4: { sl@4: BitArrayIndex result(this, bit); sl@4: sl@4: return result; sl@4: } sl@4: sl@4: /*************************************************************************** sl@4: * Method : operator[] sl@4: * Description: Overload of the [] operator. This method returns the sl@4: * value of a bit in the bit array. sl@4: * Parameters : bit - index of array bit sl@4: * Effects : None sl@4: * Returned : The value of the specified bit. sl@4: ***************************************************************************/ sl@4: bool BitArray::operator[](const unsigned int bit) const sl@4: { sl@4: return((m_Array[BIT_CHAR(bit)] & BIT_IN_CHAR(bit)) != 0); sl@4: } sl@4: sl@4: /*************************************************************************** sl@4: * Method : operator== sl@4: * Description: overload of the == operator sl@4: * Parameters : other - bit array to compare sl@4: * Effects : None sl@4: * Returned : True if this == other. Otherwise false. sl@4: ***************************************************************************/ sl@4: bool BitArray::operator==(const BitArray &other) const sl@4: { sl@4: if (m_NumBits != other.m_NumBits) sl@4: { sl@4: /* unequal sizes */ sl@4: return false; sl@4: } sl@4: sl@4: return (this->m_Array == other.m_Array); sl@4: } sl@4: sl@4: /*************************************************************************** sl@4: * Method : operator!= sl@4: * Description: overload of the != operator sl@4: * Parameters : other - bit array to compare sl@4: * Effects : None sl@4: * Returned : True if this != other. Otherwise false. sl@4: ***************************************************************************/ sl@4: bool BitArray::operator!=(const BitArray &other) const sl@4: { sl@4: if (m_NumBits != other.m_NumBits) sl@4: { sl@4: /* unequal sizes */ sl@4: return true; sl@4: } sl@4: sl@4: return (this->m_Array != other.m_Array); sl@4: } sl@4: sl@4: /*************************************************************************** sl@4: * Method : operator< sl@4: * Description: overload of the < operator sl@4: * Parameters : other - bit array to compare sl@4: * Effects : None sl@4: * Returned : True if this < other. Otherwise false. sl@4: ***************************************************************************/ sl@4: bool BitArray::operator<(const BitArray &other) const sl@4: { sl@4: if (m_NumBits != other.m_NumBits) sl@4: { sl@4: /* unequal sizes */ sl@4: return false; sl@4: } sl@4: sl@4: return (this->m_Array < other.m_Array); sl@4: } sl@4: sl@4: /*************************************************************************** sl@4: * Method : operator<= sl@4: * Description: overload of the <= operator sl@4: * Parameters : other - bit array to compare sl@4: * Effects : None sl@4: * Returned : True if this <= other. Otherwise false. sl@4: ***************************************************************************/ sl@4: bool BitArray::operator<=(const BitArray &other) const sl@4: { sl@4: if (m_NumBits != other.m_NumBits) sl@4: { sl@4: /* unequal sizes */ sl@4: return false; sl@4: } sl@4: sl@4: return (this->m_Array <= other.m_Array); sl@4: } sl@4: sl@4: /*************************************************************************** sl@4: * Method : operator> sl@4: * Description: overload of the > operator sl@4: * Parameters : other - bit array to compare sl@4: * Effects : None sl@4: * Returned : True if this > other. Otherwise false. sl@4: ***************************************************************************/ sl@4: bool BitArray::operator>(const BitArray &other) const sl@4: { sl@4: if (m_NumBits != other.m_NumBits) sl@4: { sl@4: /* unequal sizes */ sl@4: return false; sl@4: } sl@4: sl@4: return (this->m_Array > other.m_Array); sl@4: } sl@4: sl@4: /*************************************************************************** sl@4: * Method : operator>= sl@4: * Description: overload of the >= operator sl@4: * Parameters : other - bit array to compare sl@4: * Effects : None sl@4: * Returned : True if this >= other. Otherwise false. sl@4: ***************************************************************************/ sl@4: bool BitArray::operator>=(const BitArray &other) const sl@4: { sl@4: if (m_NumBits != other.m_NumBits) sl@4: { sl@4: /* unequal sizes */ sl@4: return false; sl@4: } sl@4: sl@4: return (this->m_Array >= other.m_Array); sl@4: } sl@4: sl@4: /*************************************************************************** sl@4: * Method : operator~ sl@4: * Description: overload of the ~ operator. Negates all non-spare bits in sl@4: * bit array sl@4: * Parameters : None sl@4: * Effects : None sl@4: * Returned : value of this after bitwise not sl@4: ***************************************************************************/ sl@4: BitArray BitArray::operator~(void) const sl@4: { sl@4: BitArray result(this->m_NumBits); sl@4: result = *this; sl@4: result.Not(); sl@4: sl@4: return result; sl@4: } sl@4: sl@4: /*************************************************************************** sl@4: * Method : operator& sl@4: * Description: overload of the & operator. Performs a bitwise and sl@4: * between the source array and this bit array. sl@4: * Parameters : other - bit array on righthand side of & sl@4: * Effects : None sl@4: * Returned : value of bitwise and of this and other. sl@4: ***************************************************************************/ sl@4: BitArray BitArray::operator&(const BitArray &other) const sl@4: { sl@4: BitArray result(this->m_NumBits); sl@4: result = *this; sl@4: result &= other; sl@4: sl@4: return result; sl@4: } sl@4: sl@4: sl@4: /*************************************************************************** sl@4: * Method : operator^ sl@4: * Description: overload of the ^ operator. Performs a bitwise xor sl@4: * between the source array and this bit array. sl@4: * Parameters : other - bit array on righthand side of ^ sl@4: * Effects : None sl@4: * Returned : value of bitwise xor of this and other. sl@4: ***************************************************************************/ sl@4: BitArray BitArray::operator^(const BitArray &other) const sl@4: { sl@4: BitArray result(this->m_NumBits); sl@4: result = *this; sl@4: result ^= other; sl@4: sl@4: return result; sl@4: } sl@4: sl@4: /*************************************************************************** sl@4: * Method : operator| sl@4: * Description: overload of the | operator. Performs a bitwise or sl@4: * between the source array and this bit array. sl@4: * Parameters : other - bit array on righthand side of | sl@4: * Effects : None sl@4: * Returned : value of bitwise or of this and other. sl@4: ***************************************************************************/ sl@4: BitArray BitArray::operator|(const BitArray &other) const sl@4: { sl@4: BitArray result(this->m_NumBits); sl@4: result = *this; sl@4: result |= other; sl@4: sl@4: return result; sl@4: } sl@4: sl@4: /*************************************************************************** sl@4: * Method : operator<< sl@4: * Description: overload of the << operator. Performs a bitwise left sl@4: * shift of this bit array. sl@4: * Parameters : count - the number of bits to shift left sl@4: * Effects : None sl@4: * Returned : result of bitwise left shift sl@4: ***************************************************************************/ sl@4: BitArray BitArray::operator<<(const unsigned int count) const sl@4: { sl@4: BitArray result(this->m_NumBits); sl@4: result = *this; sl@4: result <<= count; sl@4: sl@4: return result; sl@4: } sl@4: sl@4: /*************************************************************************** sl@4: * Method : operator>> sl@4: * Description: overload of the >> operator. Performs a bitwise right sl@4: * shift of this bit array. sl@4: * Parameters : count - the number of bits to shift right sl@4: * Effects : None sl@4: * Returned : result of bitwise right shift sl@4: ***************************************************************************/ sl@4: BitArray BitArray::operator>>(const unsigned int count) const sl@4: { sl@4: BitArray result(this->m_NumBits); sl@4: result = *this; sl@4: result >>= count; sl@4: sl@4: return result; sl@4: } sl@4: sl@4: /*************************************************************************** sl@4: * Method : operator++ (prefix) sl@4: * Description: overload of the ++ operator. Increments the contents of sl@4: * a bit array. Overflows cause rollover. sl@4: * Parameters : None sl@4: * Effects : Bit array contents are incremented sl@4: * Returned : Reference to this array after increment sl@4: ***************************************************************************/ sl@4: BitArray& BitArray::operator++(void) sl@4: { sl@4: int i; sl@4: unsigned char maxValue; /* maximum value for current char */ sl@4: unsigned char one; /* least significant bit in current char */ sl@4: sl@4: if (m_NumBits == 0) sl@4: { sl@4: return *this; /* nothing to increment */ sl@4: } sl@4: sl@4: /* handle arrays that don't use every bit in the last character */ sl@4: i = (m_NumBits % CHAR_BIT); sl@4: if (i != 0) sl@4: { sl@4: maxValue = UCHAR_MAX << (CHAR_BIT - i); sl@4: one = 1 << (CHAR_BIT - i); sl@4: } sl@4: else sl@4: { sl@4: maxValue = UCHAR_MAX; sl@4: one = 1; sl@4: } sl@4: sl@4: for (i = BIT_CHAR(m_NumBits - 1); i >= 0; i--) sl@4: { sl@4: if (m_Array[i] != maxValue) sl@4: { sl@4: m_Array[i] = m_Array[i] + one; sl@4: return *this; sl@4: } sl@4: else sl@4: { sl@4: /* need to carry to next byte */ sl@4: m_Array[i] = 0; sl@4: sl@4: /* remaining characters must use all bits */ sl@4: maxValue = UCHAR_MAX; sl@4: one = 1; sl@4: } sl@4: } sl@4: sl@4: return *this; sl@4: } sl@4: sl@4: /*************************************************************************** sl@4: * Method : operator++ (postfix) sl@4: * Description: overload of the ++ operator. Increments the contents of sl@4: * a bit array. Overflows cause rollover. sl@4: * Parameters : dumy - needed for postfix increment sl@4: * Effects : Bit array contents are incremented sl@4: * Returned : Reference to this array after increment sl@4: ***************************************************************************/ sl@4: BitArray& BitArray::operator++(int dummy) sl@4: { sl@4: ++(*this); sl@4: return *this; sl@4: } sl@4: sl@4: /*************************************************************************** sl@4: * Method : operator-- (prefix) sl@4: * Description: overload of the -- operator. Decrements the contents of sl@4: * a bit array. Underflows cause rollover. sl@4: * Parameters : None sl@4: * Effects : Bit array contents are decremented sl@4: * Returned : None sl@4: ***************************************************************************/ sl@4: BitArray& BitArray::operator--(void) sl@4: { sl@4: int i; sl@4: unsigned char maxValue; /* maximum value for current char */ sl@4: unsigned char one; /* least significant bit in current char */ sl@4: sl@4: if (m_NumBits == 0) sl@4: { sl@4: return *this; /* nothing to decrement */ sl@4: } sl@4: sl@4: /* handle arrays that don't use every bit in the last character */ sl@4: i = (m_NumBits % CHAR_BIT); sl@4: if (i != 0) sl@4: { sl@4: maxValue = UCHAR_MAX << (CHAR_BIT - i); sl@4: one = 1 << (CHAR_BIT - i); sl@4: } sl@4: else sl@4: { sl@4: maxValue = UCHAR_MAX; sl@4: one = 1; sl@4: } sl@4: sl@4: for (i = BIT_CHAR(m_NumBits - 1); i >= 0; i--) sl@4: { sl@4: if (m_Array[i] >= one) sl@4: { sl@4: m_Array[i] = m_Array[i] - one; sl@4: return *this; sl@4: } sl@4: else sl@4: { sl@4: /* need to borrow from the next byte */ sl@4: m_Array[i] = maxValue; sl@4: sl@4: /* remaining characters must use all bits */ sl@4: maxValue = UCHAR_MAX; sl@4: one = 1; sl@4: } sl@4: } sl@4: sl@4: return *this; sl@4: } sl@4: sl@4: /*************************************************************************** sl@4: * Method : operator-- (postfix) sl@4: * Description: overload of the -- operator. Decrements the contents of sl@4: * a bit array. Underflows cause rollover. sl@4: * Parameters : dumy - needed for postfix decrement sl@4: * Effects : Bit array contents are decremented sl@4: * Returned : None sl@4: ***************************************************************************/ sl@4: BitArray& BitArray::operator--(int dummy) sl@4: { sl@4: --(*this); sl@4: return *this; sl@4: } sl@4: sl@4: /*************************************************************************** sl@4: * Method : operator= sl@4: * Description: overload of the = operator. Copies source contents into sl@4: * this bit array. sl@4: * Parameters : src - Source bit array sl@4: * Effects : Source bit array contents are copied into this array sl@4: * Returned : Reference to this array after copy sl@4: ***************************************************************************/ sl@4: BitArray& BitArray::operator=(const BitArray &src) sl@4: { sl@4: if (*this == src) sl@4: { sl@4: /* don't do anything for a self assignment */ sl@4: return *this; sl@4: } sl@4: sl@4: if (m_NumBits != src.m_NumBits) sl@4: { sl@4: /* don't do assignment with different array sizes */ sl@4: return *this; sl@4: } sl@4: sl@4: if ((m_NumBits == 0) || (src.m_NumBits == 0)) sl@4: { sl@4: /* don't do assignment with unallocated array */ sl@4: return *this; sl@4: } sl@4: sl@4: /* copy bits from source */ sl@4: int size; sl@4: size = BITS_TO_CHARS(m_NumBits); sl@4: sl@4: copy(src.m_Array, &src.m_Array[size], this->m_Array); sl@4: return *this; sl@4: } sl@4: sl@4: /*************************************************************************** sl@4: * Method : operator&= sl@4: * Description: overload of the &= operator. Performs a bitwise and sl@4: * between the source array and this bit array. This bit sl@4: * array will contain the result. sl@4: * Parameters : src - Source bit array sl@4: * Effects : Results of bitwise and are stored in this array sl@4: * Returned : Reference to this array after and sl@4: ***************************************************************************/ sl@4: BitArray& BitArray::operator&=(const BitArray &src) sl@4: { sl@4: int size; sl@4: sl@4: size = BITS_TO_CHARS(m_NumBits); sl@4: sl@4: if (m_NumBits != src.m_NumBits) sl@4: { sl@4: /* don't do assignment with different array sizes */ sl@4: return *this; sl@4: } sl@4: sl@4: /* AND array one unsigned char at a time */ sl@4: for(int i = 0; i < size; i++) sl@4: { sl@4: m_Array[i] = m_Array[i] & src.m_Array[i]; sl@4: } sl@4: sl@4: return *this; sl@4: } sl@4: sl@4: /*************************************************************************** sl@4: * Method : operator^= sl@4: * Description: overload of the ^= operator. Performs a bitwise xor sl@4: * between the source array and this bit array. This bit sl@4: * array will contain the result. sl@4: * Parameters : src - Source bit array sl@4: * Effects : Results of bitwise xor are stored in this array sl@4: * Returned : Reference to this array after xor sl@4: ***************************************************************************/ sl@4: BitArray& BitArray::operator^=(const BitArray &src) sl@4: { sl@4: int size; sl@4: sl@4: size = BITS_TO_CHARS(m_NumBits); sl@4: sl@4: if (m_NumBits != src.m_NumBits) sl@4: { sl@4: /* don't do assignment with different array sizes */ sl@4: return *this; sl@4: } sl@4: sl@4: /* XOR array one unsigned char at a time */ sl@4: for(int i = 0; i < size; i++) sl@4: { sl@4: m_Array[i] = m_Array[i] ^ src.m_Array[i]; sl@4: } sl@4: sl@4: return *this; sl@4: } sl@4: sl@4: /*************************************************************************** sl@4: * Method : operator|= sl@4: * Description: overload of the |= operator. Performs a bitwise or sl@4: * between the source array and this bit array. This bit sl@4: * array will contain the result. sl@4: * Parameters : src - Source bit array sl@4: * Effects : Results of bitwise or are stored in this array sl@4: * Returned : Reference to this array after or sl@4: ***************************************************************************/ sl@4: BitArray& BitArray::operator|=(const BitArray &src) sl@4: { sl@4: int size; sl@4: sl@4: size = BITS_TO_CHARS(m_NumBits); sl@4: sl@4: if (m_NumBits != src.m_NumBits) sl@4: { sl@4: /* don't do assignment with different array sizes */ sl@4: return *this; sl@4: } sl@4: sl@4: /* OR array one unsigned char at a time */ sl@4: for(int i = 0; i < size; i++) sl@4: { sl@4: m_Array[i] = m_Array[i] | src.m_Array[i]; sl@4: } sl@4: sl@4: return *this; sl@4: } sl@4: sl@4: /*************************************************************************** sl@4: * Method : Not sl@4: * Description: Negates all non-spare bits in bit array. sl@4: * Parameters : None sl@4: * Effects : Contents of bit array are negated. Any spare bits are sl@4: * left at 0. sl@4: * Returned : Reference to this array after not sl@4: ***************************************************************************/ sl@4: BitArray& BitArray::Not(void) sl@4: { sl@4: int bits; sl@4: unsigned char mask; sl@4: int size; sl@4: sl@4: size = BITS_TO_CHARS(m_NumBits); sl@4: sl@4: if (m_NumBits == 0) sl@4: { sl@4: /* don't do not with unallocated array */ sl@4: return *this; sl@4: } sl@4: sl@4: /* NOT array one unsigned char at a time */ sl@4: for(int i = 0; i < size; i++) sl@4: { sl@4: m_Array[i] = ~m_Array[i]; sl@4: } sl@4: sl@4: /* zero any spare bits so increment and decrement are consistent */ sl@4: bits = m_NumBits % CHAR_BIT; sl@4: if (bits != 0) sl@4: { sl@4: mask = UCHAR_MAX << (CHAR_BIT - bits); sl@4: m_Array[BIT_CHAR(m_NumBits - 1)] &= mask; sl@4: } sl@4: sl@4: return *this; sl@4: } sl@4: sl@4: /*************************************************************************** sl@4: * Method : operator<<= sl@4: * Description: overload of the <<= operator. Performs a left shift on sl@4: * this bit array. This bit array will contain the result. sl@4: * Parameters : shifts - number of bit positions to shift sl@4: * Effects : Results of the shifts are stored in this array sl@4: * Returned : Reference to this array after shift sl@4: ***************************************************************************/ sl@4: BitArray& BitArray::operator<<=(const unsigned int shifts) sl@4: { sl@4: int i; sl@4: int chars = shifts / CHAR_BIT; /* number of whole byte shifts */ sl@4: sl@4: if (shifts >= m_NumBits) sl@4: { sl@4: /* all bits have been shifted off */ sl@4: this->ClearAll(); sl@4: return *this; sl@4: } sl@4: sl@4: /* first handle big jumps of bytes */ sl@4: if (chars > 0) sl@4: { sl@4: int size; sl@4: sl@4: size = BITS_TO_CHARS(m_NumBits); sl@4: sl@4: for (i = 0; (i + chars) < size; i++) sl@4: { sl@4: m_Array[i] = m_Array[i + chars]; sl@4: } sl@4: sl@4: /* now zero out new bytes on the right */ sl@4: for (i = size; chars > 0; chars--) sl@4: { sl@4: m_Array[i - chars] = 0; sl@4: } sl@4: } sl@4: sl@4: /* now we have at most CHAR_BIT - 1 bit shifts across the whole array */ sl@4: for (i = 0; i < (int)(shifts % CHAR_BIT); i++) sl@4: { sl@4: for (unsigned int j = 0; j < BIT_CHAR(m_NumBits - 1); j++) sl@4: { sl@4: m_Array[j] <<= 1; sl@4: sl@4: /* handle shifts across byte bounds */ sl@4: if (m_Array[j + 1] & MS_BIT) sl@4: { sl@4: m_Array[j] |= 0x01; sl@4: } sl@4: } sl@4: sl@4: m_Array[BIT_CHAR(m_NumBits - 1)] <<= 1; sl@4: } sl@4: sl@4: return *this; sl@4: } sl@4: sl@4: /*************************************************************************** sl@4: * Method : operator>>= sl@4: * Description: overload of the >>= operator. Performs a right shift on sl@4: * this bit array. This bit array will contain the result. sl@4: * Parameters : shifts - number of bit positions to shift sl@4: * Effects : Results of the shifts are stored in this array sl@4: * Returned : Reference to this array after shift sl@4: ***************************************************************************/ sl@4: BitArray& BitArray::operator>>=(const unsigned int shifts) sl@4: { sl@4: int i; sl@4: char mask; sl@4: int chars = shifts / CHAR_BIT; /* number of whole byte shifts */ sl@4: sl@4: if (shifts >= m_NumBits) sl@4: { sl@4: /* all bits have been shifted off */ sl@4: this->ClearAll(); sl@4: return *this; sl@4: } sl@4: sl@4: /* first handle big jumps of bytes */ sl@4: if (chars > 0) sl@4: { sl@4: for (i = BIT_CHAR(m_NumBits - 1); (i - chars) >= 0; i--) sl@4: { sl@4: m_Array[i] = m_Array[i - chars]; sl@4: } sl@4: sl@4: /* now zero out new bytes on the right */ sl@4: for (; chars > 0; chars--) sl@4: { sl@4: m_Array[chars - 1] = 0; sl@4: } sl@4: } sl@4: sl@4: /* now we have at most CHAR_BIT - 1 bit shifts across the whole array */ sl@4: for (i = 0; i < (int)(shifts % CHAR_BIT); i++) sl@4: { sl@4: for (unsigned int j = BIT_CHAR(m_NumBits - 1); j > 0; j--) sl@4: { sl@4: m_Array[j] >>= 1; sl@4: sl@4: /* handle shifts across byte bounds */ sl@4: if (m_Array[j - 1] & 0x01) sl@4: { sl@4: m_Array[j] |= MS_BIT; sl@4: } sl@4: } sl@4: sl@4: m_Array[0] >>= 1; sl@4: } sl@4: sl@4: /*********************************************************************** sl@4: * zero any spare bits that are shifted beyond the end of the bit array sl@4: * so that increment and decrement are consistent. sl@4: ***********************************************************************/ sl@4: i = m_NumBits % CHAR_BIT; sl@4: if (i != 0) sl@4: { sl@4: mask = UCHAR_MAX << (CHAR_BIT - i); sl@4: m_Array[BIT_CHAR(m_NumBits - 1)] &= mask; sl@4: } sl@4: sl@4: return *this; sl@4: } sl@4: sl@4: /*************************************************************************** sl@4: * Method : bit_array_index_c - constructor sl@4: * Description: This is the bit_array_index_c constructor. It stores a sl@4: * pointer to the bit array and the bit index. sl@4: * Parameters : array - pointer to bit array sl@4: * index - index of bit in array sl@4: * Effects : Pointer to bit array and bit index are stored. sl@4: * Returned : None sl@4: ***************************************************************************/ sl@4: BitArrayIndex::BitArrayIndex(BitArray *array, sl@4: const unsigned int index) sl@4: { sl@4: m_BitArray = array; sl@4: m_Index = index; sl@4: } sl@4: sl@4: /*************************************************************************** sl@4: * Method : operator= sl@4: * Description: overload of the = operator. Sets the bit array bit to sl@4: * the value of src. sl@4: * Parameters : src - bit value sl@4: * Effects : Bit pointed to by this object is set to the value of sl@4: * source. sl@4: * Returned : None sl@4: ***************************************************************************/ sl@4: void BitArrayIndex::operator=(const bool src) sl@4: { sl@4: if (m_BitArray == NULL) sl@4: { sl@4: return; /* no array */ sl@4: } sl@4: sl@4: if (m_BitArray->SizeInBits() <= m_Index) sl@4: { sl@4: return; /* index is out of bounds */ sl@4: } sl@4: sl@4: if (src) sl@4: { sl@4: m_BitArray->SetBit(m_Index); sl@4: } sl@4: else sl@4: { sl@4: m_BitArray->ClearBit(m_Index); sl@4: } sl@4: }