1
0
mirror of https://github.com/alliedmodders/hl2sdk.git synced 2025-01-05 17:13:36 +08:00
hl2sdk/public/bitvec.h

317 lines
6.9 KiB
C++

//===== Copyright © 1996-2005, Valve Corporation, All rights reserved. ======//
//
// Purpose:
//
//===========================================================================//
#ifndef BITVEC_H
#define BITVEC_H
#ifdef _WIN32
#pragma once
#endif
#include "tier0/dbg.h"
#include "tier0/basetypes.h"
class CBitVecAccessor
{
public:
CBitVecAccessor(unsigned long *pDWords, int iBit);
void operator=(int val);
operator unsigned long();
private:
unsigned long *m_pDWords;
int m_iBit;
};
// CBitVec allows you to store a list of bits and do operations on them like they were an atomic type.
template<int NUM_BITS>
class CBitVec
{
public:
CBitVec();
// Get a pointer to the raw bits.
unsigned long* Base();
// Set all values to the specified value (0 or 1..)
void Init(int val = 0);
// Access the bits like an array.
CBitVecAccessor operator[](int i);
// Clear or set all bits.
void ClearAll( int nBits=NUM_BITS );
void SetAll( int nBits=NUM_BITS );
// Get the state of a bit.
unsigned long Get( unsigned long iBit ) const;
bool IsBitSet( unsigned long iBit ) const;
// Set or clear a bit.
void Set( unsigned long iBit );
void Clear( unsigned long iBit );
// Set a bit either on or off.
void Set( unsigned long iBit, int bOn );
// Copy the state of another CBitVec.
void Copy( CBitVec<NUM_BITS> const &other, int nBits=NUM_BITS );
// Returns true if the bit vectors have the same values.
bool Compare( CBitVec<NUM_BITS> const &other, int nBits=NUM_BITS );
// Operations on other bit vectors.
CBitVec& operator=(CBitVec<NUM_BITS> const &other);
bool operator==(CBitVec<NUM_BITS> const &other);
bool operator!=(CBitVec<NUM_BITS> const &other);
// Get underlying dword representations of the bits.
int GetNumDWords() const;
unsigned long GetDWord(int i) const;
void SetDWord(int i, unsigned long val);
int GetNumBits() const;
int FindNextSetBit(int iStartBit) const; // returns -1 if no set bit was found
private:
enum {NUM_DWORDS = NUM_BITS/32 + !!(NUM_BITS & 31)};
unsigned long m_DWords[NUM_DWORDS];
};
// ------------------------------------------------------------------------ //
// CBitVecAccessor inlines.
// ------------------------------------------------------------------------ //
inline CBitVecAccessor::CBitVecAccessor(unsigned long *pDWords, int iBit)
{
m_pDWords = pDWords;
m_iBit = iBit;
}
inline void CBitVecAccessor::operator=(int val)
{
if(val)
m_pDWords[m_iBit >> 5] |= (1 << (m_iBit & 31));
else
m_pDWords[m_iBit >> 5] &= ~(unsigned long)(1 << (m_iBit & 31));
}
inline CBitVecAccessor::operator unsigned long()
{
return m_pDWords[m_iBit >> 5] & (1 << (m_iBit & 31));
}
// ------------------------------------------------------------------------ //
// CBitVec inlines.
// ------------------------------------------------------------------------ //
template<int NUM_BITS>
inline int CBitVec<NUM_BITS>::GetNumBits() const
{
return NUM_BITS;
}
template<int NUM_BITS>
inline CBitVec<NUM_BITS>::CBitVec()
{
for(int i=0; i < NUM_DWORDS; i++)
m_DWords[i] = 0;
}
template<int NUM_BITS>
inline void CBitVec<NUM_BITS>::Init(int val)
{
unsigned long dwVal = val ? 0xFFFFFFFF : 0;
for( int i=0; i < NUM_DWORDS; i++ )
m_DWords[i] = dwVal;
}
template<int NUM_BITS>
inline unsigned long* CBitVec<NUM_BITS>::Base()
{
return m_DWords;
}
template<int NUM_BITS>
inline CBitVec<NUM_BITS>& CBitVec<NUM_BITS>::operator=(CBitVec<NUM_BITS> const &other)
{
memcpy(m_DWords, other.m_DWords, sizeof(m_DWords));
return *this;
}
template<int NUM_BITS>
inline CBitVecAccessor CBitVec<NUM_BITS>::operator[](int i)
{
Assert(i >= 0 && i < GetNumBits());
return CBitVecAccessor(m_DWords, i);
}
template< int SIZE >
inline void CBitVec<SIZE>::ClearAll( int nBits )
{
int nDWords = PAD_NUMBER( nBits, 32 ) >> 5;
Assert( nDWords <= NUM_DWORDS );
for( int i=0; i < nDWords; i++ )
m_DWords[i] = 0;
}
template< int SIZE >
inline void CBitVec<SIZE>::SetAll( int nBits )
{
int nDWords = PAD_NUMBER( nBits, 32 ) >> 5;
Assert( nDWords <= NUM_DWORDS );
for( int i=0; i < nDWords; i++ )
m_DWords[i] = 0xFFFFFFFF;
}
template< int SIZE >
inline unsigned long CBitVec<SIZE>::Get( unsigned long iBit ) const
{
Assert( iBit < SIZE );
return m_DWords[iBit >> 5] & (1 << (iBit & 31));
}
template< int SIZE >
bool CBitVec<SIZE>::IsBitSet( unsigned long iBit ) const
{
return ( m_DWords[iBit >> 5] & (1 << (iBit & 31)) ) != 0;
}
template< int SIZE >
inline void CBitVec<SIZE>::Set( unsigned long iBit )
{
Assert( iBit < SIZE );
m_DWords[iBit >> 5] |= (1 << (iBit & 31));
}
template< int SIZE >
inline void CBitVec<SIZE>::Clear( unsigned long iBit )
{
Assert( iBit < SIZE );
m_DWords[iBit >> 5] &= ~(1 << (iBit & 31));
}
template< int SIZE >
inline void CBitVec<SIZE>::Set( unsigned long iBit, int bOn )
{
if( bOn )
Set( iBit );
else
Clear( iBit );
}
template< int SIZE >
inline void CBitVec<SIZE>::Copy( CBitVec<SIZE> const &other, int nBits )
{
int nBytes = PAD_NUMBER( nBits, 8 ) >> 3;
Assert( nBytes <= NUM_DWORDS*4 );
memcpy( m_DWords, other.m_DWords, nBytes );
}
template< int SIZE >
inline bool CBitVec<SIZE>::Compare( CBitVec<SIZE> const &other, int nBits )
{
int nBytes = PAD_NUMBER( nBits, 8 ) >> 3;
Assert( nBytes <= NUM_DWORDS*4 );
return memcmp( m_DWords, other.m_DWords, nBytes ) == 0;
}
template<int NUM_BITS>
inline bool CBitVec<NUM_BITS>::operator==(CBitVec<NUM_BITS> const &other)
{
for(int i=0; i < NUM_DWORDS; i++)
if(m_DWords[i] != other.m_DWords[i])
return false;
return true;
}
template<int NUM_BITS>
inline bool CBitVec<NUM_BITS>::operator!=(CBitVec<NUM_BITS> const &other)
{
return !(*this == other);
}
template<int NUM_BITS>
inline int CBitVec<NUM_BITS>::GetNumDWords() const
{
return NUM_DWORDS;
}
template<int NUM_BITS>
inline unsigned long CBitVec<NUM_BITS>::GetDWord(int i) const
{
Assert(i >= 0 && i < NUM_DWORDS);
return m_DWords[i];
}
template<int NUM_BITS>
inline void CBitVec<NUM_BITS>::SetDWord(int i, unsigned long val)
{
Assert(i >= 0 && i < NUM_DWORDS);
m_DWords[i] = val;
}
template<int NUM_BITS>
inline int CBitVec<NUM_BITS>::FindNextSetBit(int iStartBit) const
{
int currentBit = iStartBit;
while (currentBit < NUM_BITS)
{
// Early out for lots of zeros in a row
// check for 32 bit boundary...
if ( (currentBit & 31) == 0)
{
if (m_DWords[currentBit>>5] == 0)
{
currentBit += 32;
continue;
}
}
// One bit at a time...until we reach next DWord border
do
{
if( m_DWords[currentBit >> 5] & (1 << (currentBit & 31)) )
return currentBit; // found a set bit
++currentBit; // next please
} while ( (currentBit & 31) != 0 );
}
return -1; // no bit set in rest of bit array
}
#endif // BITVEC_H