aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorDavid Robillard <d@drobilla.net>2018-08-07 17:29:35 +0200
committerDavid Robillard <d@drobilla.net>2018-09-29 14:46:14 +0200
commit4984100559a00e055a502fc8fe85d7ca66589b36 (patch)
treecaf0a0a9e3e3b4e4bee1d1a43b9a9beb30d7610e
parent0bcaa46d0fb0af324fc2991ddcdf3d34c594f741 (diff)
downloadchilbert-4984100559a00e055a502fc8fe85d7ca66589b36.tar.gz
chilbert-4984100559a00e055a502fc8fe85d7ca66589b36.tar.bz2
chilbert-4984100559a00e055a502fc8fe85d7ca66589b36.zip
Clean up bit vector code
-rw-r--r--chilbert/BigBitVec.hpp608
-rw-r--r--chilbert/FixBitVec.hpp331
-rw-r--r--chilbert/Operations.hpp18
3 files changed, 367 insertions, 590 deletions
diff --git a/chilbert/BigBitVec.hpp b/chilbert/BigBitVec.hpp
index 129f5d8..c5ade5d 100644
--- a/chilbert/BigBitVec.hpp
+++ b/chilbert/BigBitVec.hpp
@@ -19,574 +19,450 @@
#ifndef CHILBERT_BIGBITVEC_HPP
#define CHILBERT_BIGBITVEC_HPP
-
#include "chilbert/FixBitVec.hpp"
+#include <algorithm>
#include <cassert>
#include <cstring>
#include <memory>
-#define BBV_MIN(a,b) ((a)<(b)?(a):(b))
-#define BBV_MAX(a,b) ((a)>(b)?(a):(b))
-#define FBVS_NEEDED(b) ((BBV_MAX(b,1)+FBV_BITS-1)/FBV_BITS)
-#define BBV_MODSPLIT(r,b,k) { b=(k); r=b/FBV_BITS; b-=r*FBV_BITS; }
+#define FBVS_NEEDED(b) ((std::max(b, 1) + FBV_BITS - 1) / FBV_BITS)
+#define BBV_MODSPLIT(r, b, k) \
+ { \
+ b = (k); \
+ r = b / FBV_BITS; \
+ b -= r * FBV_BITS; \
+ }
namespace chilbert {
class CBigBitVec
{
public:
-
- // Constructor, with optional number of bits.
- CBigBitVec(
- int iBits = FBV_BITS
- )
- : m_pcRacks{new FBV_UINT[iBits == 0 ? 0 : FBVS_NEEDED(iBits)]}
- , m_iRacks{iBits == 0 ? 0 : FBVS_NEEDED(iBits)}
+ CBigBitVec(const int bits = 0)
+ : m_pcRacks{new FBV_UINT[bits == 0 ? 0 : FBVS_NEEDED(bits)]}
+ , m_iRacks{bits == 0 ? 0 : FBVS_NEEDED(bits)}
{
}
-
-
- // Copy construct. Creates duplicate.
- CBigBitVec(
- const CBigBitVec &cBBV
- )
- : m_pcRacks{new FBV_UINT[cBBV.m_iRacks]}
- , m_iRacks{cBBV.m_iRacks}
+
+ CBigBitVec(const CBigBitVec& vec)
+ : m_pcRacks{new FBV_UINT[vec.m_iRacks]}
+ , m_iRacks{vec.m_iRacks}
{
- if (cBBV.m_pcRacks) {
+ if (vec.m_pcRacks) {
memcpy(m_pcRacks.get(),
- cBBV.m_pcRacks.get(),
+ vec.m_pcRacks.get(),
sizeof(FBV_UINT) * m_iRacks);
}
}
- // Move construct.
- CBigBitVec(
- CBigBitVec &&cBBV
- ) = default;
+ CBigBitVec(CBigBitVec&& vec) = default;
- // Copy constructor.
- CBigBitVec(
- const CFixBitVec &cFBV
- )
- : m_pcRacks{new FBV_UINT[1]}
- , m_iRacks{1}
+ CBigBitVec(const CFixBitVec& vec)
+ : m_pcRacks{new FBV_UINT[1]}
+ , m_iRacks{1}
{
- m_pcRacks[0] = cFBV.rack();
+ m_pcRacks[0] = vec.rack();
}
+ /// Return the size in bits
+ int size() const { return m_iRacks * FBV_BITS; }
- // Returns the current size in bits.
- int
- size() const
+ /// Set all bits to zero
+ CBigBitVec& reset()
{
- return m_iRacks*FBV_BITS;
- }
-
-
- // zeros the bit-vector.
- CBigBitVec &
- reset()
- {
-
memset(m_pcRacks.get(), 0, sizeof(CFixBitVec) * m_iRacks);
-
- return (*this);
+ return *this;
}
-
- // truncates the bit-vector to a given precision in
- // bits (zeroes MSBs without shrinking the vector)
- CBigBitVec &
- truncate(
- int iBits
- )
+ /// Truncate to a given precision in bits (zero MSBs)
+ CBigBitVec& truncate(const int bits)
{
- assert( iBits >= 0 && iBits <= size() );
- int r, b, i;
+ assert(bits >= 0 && bits <= size());
+ int r, b;
+
+ BBV_MODSPLIT(r, b, bits);
- BBV_MODSPLIT(r,b,iBits);
-
- if ( r >= m_iRacks )
- return (*this);
+ if (r >= m_iRacks) {
+ return *this;
+ }
// Truncate rack that contains the split point
- m_pcRacks[r] &= FBVN1S(iBits);
-
- for ( i = r+1; i < m_iRacks; i++ )
+ m_pcRacks[r] &= FBVN1S(bits);
+
+ for (int i = r + 1; i < m_iRacks; ++i) {
m_pcRacks[i] = 0;
+ }
- return (*this);
+ return *this;
}
-
- // Assignment operator. No resizing.
- CBigBitVec &
- operator=(
- const CBigBitVec &cBBV
- )
+ /// Non-resizing assignment
+ CBigBitVec& operator=(const CBigBitVec& vec)
{
- if ( m_iRacks < cBBV.m_iRacks )
- {
- memcpy(m_pcRacks.get(), cBBV.m_pcRacks.get(), sizeof(CFixBitVec) * m_iRacks);
+ if (m_iRacks < vec.m_iRacks) {
+ memcpy(m_pcRacks.get(),
+ vec.m_pcRacks.get(),
+ sizeof(CFixBitVec) * m_iRacks);
} else {
if (m_pcRacks) {
- if (cBBV.m_pcRacks) {
+ if (vec.m_pcRacks) {
memcpy(m_pcRacks.get(),
- cBBV.m_pcRacks.get(),
- sizeof(CFixBitVec) * cBBV.m_iRacks);
+ vec.m_pcRacks.get(),
+ sizeof(CFixBitVec) * vec.m_iRacks);
}
- memset(m_pcRacks.get() + cBBV.m_iRacks,
+ memset(m_pcRacks.get() + vec.m_iRacks,
0,
- sizeof(CFixBitVec) * (m_iRacks - cBBV.m_iRacks));
+ sizeof(CFixBitVec) * (m_iRacks - vec.m_iRacks));
}
}
- return (*this);
+ return *this;
}
- CBigBitVec &
- operator=(
- CBigBitVec &&cBBV
- ) = default;
- CBigBitVec &
- operator=(
- const CFixBitVec &cFBV
- )
+ CBigBitVec& operator=(CBigBitVec&& vec) = default;
+
+ CBigBitVec& operator=(const CFixBitVec& vec)
{
- m_pcRacks[0] = cFBV.rack();
+ m_pcRacks[0] = vec.rack();
memset(m_pcRacks.get() + 1, 0, sizeof(FBV_UINT) * (m_iRacks - 1));
- return (*this);
+ return *this;
}
- CBigBitVec &
- operator=(
- FBV_UINT j
- )
+
+ CBigBitVec& operator=(const FBV_UINT j)
{
m_pcRacks[0] = j;
memset(m_pcRacks.get() + 1, 0, sizeof(CFixBitVec) * (m_iRacks - 1));
- return (*this);
+ return *this;
}
- // Returns the value of the nth bit.
- bool
- test(
- int iIndex
- ) const
+ /// Return the value of the `index`th bit
+ bool test(const int index) const
{
- assert( iIndex >= 0 && iIndex < size() );
+ assert(index >= 0 && index < size());
int r, b;
- BBV_MODSPLIT(r,b,iIndex);
+ BBV_MODSPLIT(r, b, index);
return testBit(m_pcRacks[r], b);
}
-
- // Sets the value of the nth bit.
- CBigBitVec &
- set(
- int iIndex,
- bool bBit = true
- )
+ /// Set the `index`th bit to 1
+ CBigBitVec& set(const int index)
{
- assert( iIndex >= 0 && iIndex < size() );
+ assert(index >= 0 && index < size());
int r, b;
- BBV_MODSPLIT(r,b,iIndex);
- setBit(m_pcRacks[r], b, bBit);
- return (*this);
+ BBV_MODSPLIT(r, b, index);
+ setBit(m_pcRacks[r], b);
+ return *this;
}
-
- // Toggles the value of the nth bit.
- CBigBitVec &
- toggle(
- int iIndex
- )
+ /// Reset the `index`th bit to 0
+ CBigBitVec& reset(const int index)
{
- assert( iIndex >= 0 && iIndex < size() );
+ assert(index >= 0 && index < size());
int r, b;
- BBV_MODSPLIT(r,b,iIndex);
- m_pcRacks[r] ^= (FBV1<<iIndex);
- return (*this);
+ BBV_MODSPLIT(r, b, index);
+ m_pcRacks[r] &= ~((FBV_UINT)1 << b);
+ return *this;
}
+ /// Set the `index`th bit to `value`
+ CBigBitVec& set(const int index, const bool value)
+ {
+ assert(index >= 0 && index < size());
+ int r, b;
+ BBV_MODSPLIT(r, b, index);
+ setBit(m_pcRacks[r], b, value);
+ return *this;
+ }
- // In place AND.
- CBigBitVec &
- operator&=(
- const CBigBitVec &cBBV
- )
+ /// Toggle the value of the `index`th bit
+ CBigBitVec& toggle(const int index)
{
- int i;
-
- for ( i = 0; i < BBV_MIN(m_iRacks,cBBV.m_iRacks); i++ )
- m_pcRacks[i] &= cBBV.m_pcRacks[i];
-
- return (*this);
+ assert(index >= 0 && index < size());
+ int r, b;
+ BBV_MODSPLIT(r, b, index);
+ m_pcRacks[r] ^= (FBV1 << index);
+ return *this;
}
+ CBigBitVec& operator&=(const CBigBitVec& vec)
+ {
+ for (int i = 0; i < std::min(m_iRacks, vec.m_iRacks); ++i) {
+ m_pcRacks[i] &= vec.m_pcRacks[i];
+ }
- // AND operator.
- CBigBitVec
- operator&(
- const CBigBitVec &cBBV
- ) const
+ return *this;
+ }
+
+ CBigBitVec operator&(const CBigBitVec& vec) const
{
- CBigBitVec t( *this );
- t &= cBBV;
+ CBigBitVec t(*this);
+ t &= vec;
return t;
}
-
- // In place OR.
- CBigBitVec &
- operator|=(
- const CBigBitVec &cBBV
- )
+ CBigBitVec& operator|=(const CBigBitVec& vec)
{
- int i;
-
- for ( i = 0; i < BBV_MIN(m_iRacks,cBBV.m_iRacks); i++ )
- m_pcRacks[i] |= cBBV.m_pcRacks[i];
-
- return (*this);
- }
+ for (int i = 0; i < std::min(m_iRacks, vec.m_iRacks); ++i) {
+ m_pcRacks[i] |= vec.m_pcRacks[i];
+ }
+ return *this;
+ }
- // OR operator.
- CBigBitVec
- operator|(
- const CBigBitVec &cBBV
- ) const
+ CBigBitVec operator|(const CBigBitVec& vec) const
{
- CBigBitVec t( *this );
- t |= cBBV;
+ CBigBitVec t(*this);
+ t |= vec;
return t;
}
-
- // In place XOR.
- CBigBitVec &
- operator^=(
- const CBigBitVec &cBBV
- )
+ CBigBitVec& operator^=(const CBigBitVec& vec)
{
- int i;
-
- for ( i = 0; i < BBV_MIN(m_iRacks,cBBV.m_iRacks); i++ )
- m_pcRacks[i] ^= cBBV.m_pcRacks[i];
-
- return (*this);
- }
+ for (int i = 0; i < std::min(m_iRacks, vec.m_iRacks); ++i) {
+ m_pcRacks[i] ^= vec.m_pcRacks[i];
+ }
+ return *this;
+ }
- // XOR operator.
- CBigBitVec
- operator^(
- const CBigBitVec &cBBV
- ) const
+ CBigBitVec operator^(const CBigBitVec& vec) const
{
- CBigBitVec t( *this );
- t ^= cBBV;
+ CBigBitVec t(*this);
+ t ^= vec;
return t;
}
-
- // Shift left operation, in place.
- CBigBitVec &
- operator<<=(
- int iBits
- )
+ CBigBitVec& operator<<=(const int bits)
{
- assert( iBits >= 0 );
+ assert(bits >= 0);
int r, b, i;
// No shift?
- if ( iBits == 0 )
- return (*this);
+ if (bits == 0) {
+ return *this;
+ }
- BBV_MODSPLIT(r,b,iBits);
+ BBV_MODSPLIT(r, b, bits);
// All racks?
- if ( r >= m_iRacks )
- {
+ if (r >= m_iRacks) {
reset();
- return (*this);
+ return *this;
}
// Do rack shifts.
- if ( r > 0 )
- {
- for ( i = m_iRacks-1; i >= r; i-- )
- m_pcRacks[i] = m_pcRacks[i-r];
- for ( ; i >= 0; i-- )
+ if (r > 0) {
+ for (i = m_iRacks - 1; i >= r; --i) {
+ m_pcRacks[i] = m_pcRacks[i - r];
+ }
+ for (; i >= 0; --i) {
m_pcRacks[i] = 0;
+ }
}
// Do bit shifts.
- if ( b > 0 )
- {
+ if (b > 0) {
int bi = FBV_BITS - b;
- for ( i = m_iRacks-1; i >= r+1; i-- )
- {
+ for (i = m_iRacks - 1; i >= r + 1; --i) {
m_pcRacks[i] <<= b;
- m_pcRacks[i] |= m_pcRacks[i-1] >> bi;
+ m_pcRacks[i] |= m_pcRacks[i - 1] >> bi;
}
m_pcRacks[i] <<= b;
}
- return (*this);
+ return *this;
}
-
- // Shift left operation.
- CBigBitVec
- operator<<(
- int iBits
- ) const
+ CBigBitVec operator<<(const int bits) const
{
- CBigBitVec t( *this );
- t <<= iBits;
+ CBigBitVec t(*this);
+ t <<= bits;
return t;
}
-
- // Shift right operation, in place.
- CBigBitVec &
- operator>>=(
- int iBits
- )
+ CBigBitVec& operator>>=(const int bits)
{
- assert( iBits >= 0 );
+ assert(bits >= 0);
int r, b, i;
// No shift?
- if ( iBits == 0 )
- return (*this);
+ if (bits == 0) {
+ return *this;
+ }
- BBV_MODSPLIT(r,b,iBits);
+ BBV_MODSPLIT(r, b, bits);
// All racks?
- if ( r >= m_iRacks )
- {
+ if (r >= m_iRacks) {
reset();
- return (*this);
+ return *this;
}
// Do rack shifts.
- if ( r > 0 )
- {
- for ( i = 0; i < m_iRacks-r; i++ )
- m_pcRacks[i] = m_pcRacks[i+r];
- for ( ; i < m_iRacks; i++ )
+ if (r > 0) {
+ for (i = 0; i < m_iRacks - r; ++i) {
+ m_pcRacks[i] = m_pcRacks[i + r];
+ }
+ for (; i < m_iRacks; ++i) {
m_pcRacks[i] = 0;
+ }
}
// Do bit shifts.
- if ( b > 0 )
- {
+ if (b > 0) {
int bi = FBV_BITS - b;
- for ( i = 0; i < m_iRacks-r-1; i++ )
- {
+ for (i = 0; i < m_iRacks - r - 1; ++i) {
m_pcRacks[i] >>= b;
- m_pcRacks[i] |= m_pcRacks[i+1] << bi;
+ m_pcRacks[i] |= m_pcRacks[i + 1] << bi;
}
m_pcRacks[i] >>= b;
}
- return (*this);
+ return *this;
}
-
- // Shift right operation.
- CBigBitVec
- operator>>(
- int iBits
- ) const
+ CBigBitVec operator>>(const int bits) const
{
- CBigBitVec t( *this );
- t >>= iBits;
+ CBigBitVec t(*this);
+ t >>= bits;
return t;
}
-
- // Right rotation, in place.
- CBigBitVec &
- rotr(
- int iBits,
- int iWidth
- )
+ /// Right-rotate the least significant `width` bits by `bits` positions
+ CBigBitVec& rotr(const int bits, int width)
{
- assert( iBits >= 0 );
+ assert(bits >= 0);
// Fill in the width, if necessary.
- if ( iWidth <= 0 )
- iWidth = size();
+ if (width <= 0) {
+ width = size();
+ }
// Modulo the number of bits.
- assert( iBits < iWidth );
- if ( iBits == 0 ) return (*this);
+ assert(bits < width);
+ if (bits == 0) {
+ return *this;
+ }
// Ensure we are truncated appropriately.
- truncate(iWidth);
+ truncate(width);
- CBigBitVec t1( *this );
- (*this) >>= iBits;
- t1 <<= (iWidth-iBits);
+ CBigBitVec t1(*this);
+ (*this) >>= bits;
+ t1 <<= (width - bits);
(*this) |= t1;
-
- truncate(iWidth);
- return (*this);
- }
+ truncate(width);
+ return *this;
+ }
- // Left rotation, in place.
- CBigBitVec &
- rotl(
- int iBits,
- int iWidth
- )
+ /// Left-rotate the least significant `width` bits by `bits` positions
+ CBigBitVec& rotl(const int bits, int width)
{
- assert( iBits >= 0 );
+ assert(bits >= 0);
// Fill in the width, if necessary.
- if ( iWidth <= 0 )
- iWidth = size();
+ if (width <= 0) {
+ width = size();
+ }
// Modulo the number of bits.
- assert( iBits < iWidth );
- if ( iBits == 0 ) return (*this);
+ assert(bits < width);
+ if (bits == 0) {
+ return *this;
+ }
// Ensure we are truncated appropriately.
- truncate(iWidth);
+ truncate(width);
- CBigBitVec t1( *this );
- (*this) <<= iBits;
- t1 >>= (iWidth-iBits);
+ CBigBitVec t1(*this);
+ (*this) <<= bits;
+ t1 >>= (width - bits);
(*this) |= t1;
- truncate(iWidth);
+ truncate(width);
- return (*this);
+ return *this;
}
-
- // Returns true if the rack is zero valued.
- bool
- none() const
+ /// Return true iff all bits are zero
+ bool none() const
{
- int i;
- for ( i = 0; i < m_iRacks; i++ )
- if ( m_pcRacks[i] ) return false;
+ for (int i = 0; i < m_iRacks; ++i) {
+ if (m_pcRacks[i]) {
+ return false;
+ }
+ }
return true;
}
-
- // Returns the index of the first set bit.
- // (numbered 1 to n, with 0 meaning no bits were set)
- int
- fsb() const
+ /// Return 1 + the index of the first set bit, or 0 if there are none
+ int fsb() const
{
- for (int i = 0; i < m_iRacks; ++i )
- {
+ for (int i = 0; i < m_iRacks; ++i) {
const int j = ffs(m_pcRacks[i]);
- if ( j ) {
+ if (j) {
return (i * FBV_BITS) + j;
}
}
return 0;
}
-
- // Complement
- CBigBitVec &
- flip()
+ /// Flip all bits (one's complement)
+ CBigBitVec& flip()
{
- int i;
- for ( i = 0; i < m_iRacks; i++ )
+ for (int i = 0; i < m_iRacks; ++i) {
m_pcRacks[i] = ~m_pcRacks[i];
- return (*this);
- }
-
-
- // Returns the first rack.
- FBV_UINT &
- rack()
- {
- return m_pcRacks[0];
- }
- FBV_UINT
- rack() const
- {
- return m_pcRacks[0];
+ }
+ return *this;
}
+ /// Return the first rack
+ FBV_UINT& rack() { return m_pcRacks[0]; }
+ FBV_UINT rack() const { return m_pcRacks[0]; }
- // Returns the racks.
- FBV_UINT *
- racks()
- {
- return m_pcRacks.get();
- }
- const FBV_UINT *
- racks() const
- {
- return m_pcRacks.get();
- }
-
+ /// Return a pointer to the racks
+ FBV_UINT* racks() { return m_pcRacks.get(); }
+ const FBV_UINT* racks() const { return m_pcRacks.get(); }
- // Returns the number of racks.
- int
- rackCount() const
- {
- return m_iRacks;
- }
+ /// Return the number of racks
+ int rackCount() const { return m_iRacks; }
-
private:
- friend void grayCode<CBigBitVec>(CBigBitVec&);
- friend void grayCodeInv<CBigBitVec>(CBigBitVec&);
-
// Right rotates entire racks (in place).
- void
- rackRotr(
- int k
- )
+ void rackRotr(int k)
{
- assert( 0 <= k && k < m_iRacks );
-
- int c, v;
- FBV_UINT tmp;
-
- if (k == 0) return;
-
- c = 0;
- for (v = 0; c < m_iRacks; v++)
- {
- int t = v, tp = v + k;
- tmp = m_pcRacks[v];
+ assert(0 <= k && k < m_iRacks);
+
+ if (k == 0) {
+ return;
+ }
+
+ int c = 0;
+ for (int v = 0; c < m_iRacks; ++v) {
+ int t = v, tp = v + k;
+ const int tmp = m_pcRacks[v];
c++;
- while (tp != v)
- {
+ while (tp != v) {
m_pcRacks[t] = m_pcRacks[tp];
- t = tp;
+ t = tp;
tp += k;
- if (tp >= m_iRacks) tp -= m_iRacks;
+ if (tp >= m_iRacks) {
+ tp -= m_iRacks;
+ }
c++;
}
m_pcRacks[t] = tmp;
}
-
- return;
}
-
std::unique_ptr<FBV_UINT[]> m_pcRacks;
- int m_iRacks;
+ int m_iRacks;
};
template <>
@@ -595,10 +471,10 @@ grayCode(CBigBitVec& value)
{
FBV_UINT s = 0;
- for (int i = value.rackCount() - 1; i >= 0; i--) {
+ for (int i = value.rackCount() - 1; i >= 0; --i) {
const FBV_UINT t = value.racks()[i] & 1;
- grayCode(value.m_pcRacks[i]);
- value.m_pcRacks[i] ^= (s << (FBV_BITS - 1));
+ grayCode(value.racks()[i]);
+ value.racks()[i] ^= (s << (FBV_BITS - 1));
s = t;
}
}
@@ -610,7 +486,7 @@ grayCodeInv(CBigBitVec& value)
FBV_UINT s = 0;
for (int i = value.rackCount() - 1; i >= 0; --i) {
- FBV_UINT& rack = value.m_pcRacks[i];
+ FBV_UINT& rack = value.racks()[i];
grayCodeInv(rack);
if (s) {
rack = ~rack;
diff --git a/chilbert/FixBitVec.hpp b/chilbert/FixBitVec.hpp
index d8fee9a..bec5ba1 100644
--- a/chilbert/FixBitVec.hpp
+++ b/chilbert/FixBitVec.hpp
@@ -26,316 +26,199 @@
namespace chilbert {
-// This must be an unsigned integer that is either
-// 32 or 64 bits. Otherwise, there are places in the
-// code that simply will not work.
-// For speed, this should be the native word size.
+/* This must be an unsigned integer that is either 32 or 64 bits. Otherwise,
+ there are places in the code that simply will not work. For speed, this
+ should be the native word size. */
typedef uint64_t FBV_UINT;
-#define FBV_BITS 64
-#define FBV0 ((FBV_UINT)0)
-#define FBV1 ((FBV_UINT)1)
-#define FBV1S (~FBV0)
-#define FBVN1S(n) (n==FBV_BITS?FBV1S:(FBV1<<n)-1)
+#define FBV_BITS 64
+
+#define FBV1 ((FBV_UINT)1)
+#define FBV1S (~(FBV_UINT)0)
+#define FBVN1S(n) (n == FBV_BITS ? FBV1S : (FBV1 << n) - 1)
class CFixBitVec
{
public:
+ /// Return the size in bits
+ int size() const { return FBV_BITS; }
- // Default constructor. The bits parameter
- // is completely ignored, but accepted in order
- // to look and feel the same as a BigBitVec.
- CFixBitVec(
- int iBits = FBV_BITS
- )
- : m_uiRack{}
- {
- }
-
- // Copy constructor.
- CFixBitVec(
- const CFixBitVec &cFBV
- )
- : m_uiRack{cFBV.m_uiRack}
- {
- }
-
- // Returns the current size in bits.
- int
- size() const
+ /// Set all bits to zero
+ CFixBitVec& reset()
{
- return FBV_BITS;
+ m_rack = 0;
+ return *this;
}
- // Zeros the bit-vector.
- CFixBitVec &
- reset()
+ /// Return the value of the `index`th bit
+ bool test(const int index) const
{
- m_uiRack = 0;
- return (*this);
+ assert(0 <= index && index < FBV_BITS);
+ return ((m_rack & (FBV1 << index)) > 0);
}
- // Assignment operator.
- CFixBitVec &
- operator=(
- const CFixBitVec &cFBV
- )
+ /// Set the `index`th bit to 1
+ CFixBitVec& set(const int index)
{
- m_uiRack = cFBV.m_uiRack;
- return (*this);
+ assert(0 <= index && index < FBV_BITS);
+ m_rack |= ((FBV_UINT)1 << index);
+ return *this;
}
- // Assignment operator.
- CFixBitVec &
- operator=(
- FBV_UINT i
- )
+ /// Reset the `index`th bit to 0
+ CFixBitVec& reset(const int index)
{
- m_uiRack = i;
- return (*this);
+ assert(0 <= index && index < FBV_BITS);
+ m_rack &= ~((FBV_UINT)1 << index);
+ return *this;
}
- // Returns the value of the nth bit.
- bool
- test(
- int iIndex
- ) const
+ /// Set the `index`th bit to `value`
+ CFixBitVec& set(const int index, const bool value)
{
- assert( 0 <= iIndex && iIndex < FBV_BITS );
- return ( (m_uiRack & (FBV1<<iIndex)) > 0 );
+ assert(0 <= index && index < FBV_BITS);
+ m_rack ^= (-value ^ m_rack) & ((FBV_UINT)1 << index);
+ return *this;
}
- // Sets the value of the nth bit.
- CFixBitVec &
- set(
- int iIndex,
- bool bBit = true
- )
+ /// Toggle the value of the `index`th bit
+ CFixBitVec& toggle(const int index)
{
- assert( 0 <= iIndex && iIndex < FBV_BITS );
- FBV_UINT m = (FBV1<<iIndex);
- m_uiRack |= m;
- if ( !bBit ) m_uiRack ^= m;
- return (*this);
+ assert(0 <= index && index < FBV_BITS);
+ m_rack ^= (FBV1 << index);
+ return *this;
}
- // Toggles the value of the nth bit.
- CFixBitVec &
- toggle(
- int iIndex
- )
+ CFixBitVec& operator=(const FBV_UINT i)
{
- assert( 0 <= iIndex && iIndex < FBV_BITS );
- m_uiRack ^= (FBV1<<iIndex);
- return (*this);
+ m_rack = i;
+ return *this;
}
- // AND operation in place.
- CFixBitVec &
- operator&=(
- const CFixBitVec &cFBV
- )
+ CFixBitVec& operator&=(const CFixBitVec& vec)
{
- m_uiRack &= cFBV.m_uiRack;
- return (*this);
+ m_rack &= vec.m_rack;
+ return *this;
}
- // AND operation.
- CFixBitVec
- operator&(
- const CFixBitVec &cFBV
- ) const
+ CFixBitVec operator&(const CFixBitVec& vec) const
{
CFixBitVec t(*this);
- t &= cFBV;
+ t &= vec;
return t;
}
- // OR operation in place.
- CFixBitVec &
- operator|=(
- const CFixBitVec &cFBV
- )
+ CFixBitVec& operator|=(const CFixBitVec& vec)
{
- m_uiRack |= cFBV.m_uiRack;
- return (*this);
+ m_rack |= vec.m_rack;
+ return *this;
}
- // OR operation.
- CFixBitVec
- operator|(
- const CFixBitVec &cFBV
- ) const
+ CFixBitVec operator|(const CFixBitVec& vec) const
{
CFixBitVec t(*this);
- t |= cFBV;
+ t |= vec;
return t;
}
-
- // XOR operation in place.
- CFixBitVec &
- operator^=(
- const CFixBitVec &cFBV
- )
+ CFixBitVec& operator^=(const CFixBitVec& vec)
{
- m_uiRack ^= cFBV.m_uiRack;
- return (*this);
+ m_rack ^= vec.m_rack;
+ return *this;
}
- CFixBitVec &
- operator^=(
- FBV_UINT i
- )
+
+ CFixBitVec& operator^=(const FBV_UINT i)
{
- m_uiRack ^= i;
- return (*this);
+ m_rack ^= i;
+ return *this;
}
- // XOR operation.
- CFixBitVec
- operator^(
- const CFixBitVec &cFBV
- ) const
+ CFixBitVec operator^(const CFixBitVec& vec) const
{
CFixBitVec t(*this);
- t ^= cFBV;
+ t ^= vec;
return t;
}
- // Shift left operation, in place.
- CFixBitVec &
- operator<<=(
- int iBits
- )
+ CFixBitVec& operator<<=(const int bits)
{
- m_uiRack <<= iBits;
- return (*this);
+ m_rack <<= bits;
+ return *this;
}
- // Shift left operation.
- CFixBitVec
- operator<<(
- int iBits
- ) const
+ CFixBitVec operator<<(const int bits) const
{
CFixBitVec t(*this);
- t <<= iBits;
+ t <<= bits;
return t;
}
- // Shift right operation, in place.
- CFixBitVec &
- operator>>=(
- int iBits
- )
+ CFixBitVec& operator>>=(const int bits)
{
- m_uiRack >>= iBits;
- return (*this);
+ m_rack >>= bits;
+ return *this;
}
- // Shift right operation.
- CFixBitVec
- operator>>(
- int iBits
- ) const
+ CFixBitVec operator>>(const int bits) const
{
CFixBitVec t(*this);
- t >>= iBits;
+ t >>= bits;
return t;
}
- // Right rotation, in place.
- CFixBitVec &
- rotr(
- int iBits,
- int iWidth
- )
- {
- assert( iBits >= 0 );
- assert( iWidth > 0 );
- assert( iBits < iWidth );
- m_uiRack &= FBVN1S(iWidth);
- m_uiRack = (m_uiRack>>iBits) | (m_uiRack<<(iWidth-iBits));
- m_uiRack &= FBVN1S(iWidth);
- return (*this);
- }
-
- // Left rotation, in place.
- CFixBitVec &
- rotl(
- int iBits,
- int iWidth
- )
+ /// Right-rotate the least significant `width` bits by `bits` positions
+ CFixBitVec& rotr(const int bits, const int width)
{
- assert( iBits >= 0 );
- assert( iWidth > 0 );
- assert( iBits < iWidth );
- m_uiRack &= FBVN1S(iWidth);
- m_uiRack = (m_uiRack<<iBits) | (m_uiRack>>(iWidth-iBits));
- m_uiRack &= FBVN1S(iWidth);
- return (*this);
+ assert(bits >= 0);
+ assert(width > 0);
+ assert(bits < width);
+ m_rack &= FBVN1S(width);
+ m_rack = (m_rack >> bits) | (m_rack << (width - bits));
+ m_rack &= FBVN1S(width);
+ return *this;
}
- // Is the bit rack zero valued?
- bool
- none() const
+ /// Left-rotate the least significant `width` bits by `bits` positions
+ CFixBitVec& rotl(int bits, int width)
{
- return m_uiRack == 0;
+ assert(bits >= 0);
+ assert(width > 0);
+ assert(bits < width);
+ m_rack &= FBVN1S(width);
+ m_rack = (m_rack << bits) | (m_rack >> (width - bits));
+ m_rack &= FBVN1S(width);
+ return *this;
}
- // Returns the index of the first set bit, numbered from
- // 1 to n. 0 means there were no set bits.
- int
- fsb() const
- {
- return chilbert::ffs(m_uiRack);
- }
+ /// Return true iff all bits are zero
+ bool none() const { return m_rack == 0; }
+ /// Return 1 + the index of the first set bit, or 0 if there are none
+ int fsb() const { return chilbert::ffs(m_rack); }
- // Ones-complements the rack
- CFixBitVec &
- flip()
+ /// Flip all bits (one's complement)
+ CFixBitVec& flip()
{
- m_uiRack = ~m_uiRack;
- return (*this);
+ m_rack = ~m_rack;
+ return *this;
}
- // Returns the first rack.
- FBV_UINT &
- rack()
- {
- return m_uiRack;
- }
- FBV_UINT
- rack() const
- {
- return m_uiRack;
- }
+ /// Return the first rack
+ FBV_UINT& rack() { return m_rack; }
+ FBV_UINT rack() const { return m_rack; }
- // Return a pointer to the racks
- FBV_UINT *
- racks()
- {
- return &m_uiRack;
- }
- const FBV_UINT *
- racks() const
- {
- return &m_uiRack;
- }
+ /// Return a pointer to the racks
+ FBV_UINT* racks() { return &m_rack; }
+ const FBV_UINT* racks() const { return &m_rack; }
- // Returns the number of racks.
- int
- rackCount()
- {
- return 1;
- }
+ /// Return the number of racks
+ int rackCount() { return 1; }
private:
- static_assert( 8*sizeof(FBV_UINT) == FBV_BITS, "" );
- static_assert( (sizeof(FBV_UINT) == 4) || (sizeof(FBV_UINT) == 8), "" );
+ static_assert(8 * sizeof(FBV_UINT) == FBV_BITS, "");
+ static_assert((sizeof(FBV_UINT) == 4) || (sizeof(FBV_UINT) == 8), "");
- FBV_UINT m_uiRack;
+ FBV_UINT m_rack{};
};
template <>
diff --git a/chilbert/Operations.hpp b/chilbert/Operations.hpp
index 578daaf..06a06fa 100644
--- a/chilbert/Operations.hpp
+++ b/chilbert/Operations.hpp
@@ -52,6 +52,16 @@ testBit(const T& field, const BitsetIndex<T> index)
return field.test(index);
}
+/// Set the `index`th bit in `field`
+template <typename T>
+void
+setBit(T& field, const IntegralIndex<T> index)
+{
+ assert(size_t(index) < sizeof(field) * CHAR_BIT);
+ field |= ((T)1 << index);
+ assert(testBit(field, index));
+}
+
/// Set the `index`th bit in `field` to `value`
template <typename T>
void
@@ -62,6 +72,14 @@ setBit(T& field, const IntegralIndex<T> index, const bool value)
assert(testBit(field, index) == value);
}
+/// Set the `index`th bit in `field`
+template <typename T>
+void
+setBit(T& field, const BitsetIndex<T> index)
+{
+ field.set(index);
+}
+
/// Set the `index`th bit in `field` to `value`
template <typename T>
void