#ifndef ROOT_TBits
#define ROOT_TBits
#ifndef ROOT_TObject
#include "TObject.h"
#endif
#ifndef ROOT_Riosfwd
#include "Riosfwd.h"
#endif
#ifndef __CINT__
#include <string.h>
#endif
class TBits : public TObject {
protected:
UInt_t fNbits;
UInt_t fNbytes;
UChar_t *fAllBits;
void ReserveBytes(UInt_t nbytes);
void DoAndEqual(const TBits& rhs);
void DoOrEqual (const TBits& rhs);
void DoXorEqual(const TBits& rhs);
void DoLeftShift(UInt_t shift);
void DoRightShift(UInt_t shift);
void DoFlip();
public:
TBits(UInt_t nbits = 8);
TBits(const TBits&);
TBits& operator=(const TBits&);
virtual ~TBits();
class TReference {
friend class TBits;
TBits &fBits;
UInt_t fPos;
TReference();
public:
TReference(TBits& bit, UInt_t pos) : fBits(bit),fPos(pos) { }
~TReference() { }
TReference& operator=(Bool_t val);
TReference& operator=(const TReference& rhs);
#ifndef __CINT__
Bool_t operator~() const;
#endif
operator Bool_t() const;
};
void ResetAllBits(Bool_t value=kFALSE);
void ResetBitNumber(UInt_t bitnumber);
void SetBitNumber(UInt_t bitnumber, Bool_t value = kTRUE);
Bool_t TestBitNumber(UInt_t bitnumber) const;
TBits::TReference operator[](UInt_t bitnumber) { return TReference(*this,bitnumber); }
Bool_t operator[](UInt_t bitnumber) const;
TBits& operator&=(const TBits& rhs) { DoAndEqual(rhs); return *this; }
TBits& operator|=(const TBits& rhs) { DoOrEqual(rhs); return *this; }
TBits& operator^=(const TBits& rhs) { DoXorEqual(rhs); return *this; }
TBits& operator<<=(UInt_t rhs) { DoLeftShift(rhs); return *this; }
TBits& operator>>=(UInt_t rhs) { DoRightShift(rhs); return *this; }
TBits operator<<(UInt_t rhs) { return TBits(*this)<<= rhs; }
TBits operator>>(UInt_t rhs) { return TBits(*this)>>= rhs; }
TBits operator~() { TBits res(*this); res.DoFlip(); return res; }
void Set(UInt_t nbits, const Char_t *array);
void Set(UInt_t nbits, const UChar_t *array) { Set(nbits, (const Char_t*)array); }
void Set(UInt_t nbits, const Short_t *array);
void Set(UInt_t nbits, const UShort_t *array) { Set(nbits, (const Short_t*)array); }
void Set(UInt_t nbits, const Int_t *array);
void Set(UInt_t nbits, const UInt_t *array) { Set(nbits, (const Int_t*)array); }
void Set(UInt_t nbits, const Long64_t *array);
void Set(UInt_t nbits, const ULong64_t *array) { Set(nbits, (const Long64_t*)array); }
void Get(Char_t *array) const;
void Get(UChar_t *array) const { Get((Char_t*)array); }
void Get(Short_t *array) const;
void Get(UShort_t *array) const { Get((Short_t*)array); }
void Get(Int_t *array) const;
void Get(UInt_t *array) const { Get((Int_t*)array); }
void Get(Long64_t *array) const;
void Get(ULong64_t *array) const { Get((Long64_t*)array); }
void Clear(Option_t *option="");
void Compact();
UInt_t CountBits(UInt_t startBit=0) const ;
UInt_t FirstNullBit(UInt_t startBit=0) const;
UInt_t FirstSetBit(UInt_t startBit=0) const;
UInt_t GetNbits() const { return fNbits; }
UInt_t GetNbytes() const { return fNbytes; }
Bool_t operator==(const TBits &other) const;
Bool_t operator!=(const TBits &other) const { return !(*this==other); }
void Paint(Option_t *option="");
void Print(Option_t *option="") const;
void Output(ostream &) const;
ClassDef(TBits,1)
};
inline Bool_t operator&(const TBits::TReference& lhs, const TBits::TReference& rhs)
{
return (bool)lhs & rhs;
}
inline Bool_t operator|(const TBits::TReference& lhs, const TBits::TReference& rhs)
{
return (bool)lhs | rhs;
}
inline Bool_t operator^(const TBits::TReference& lhs, const TBits::TReference& rhs)
{
return (bool)lhs ^ rhs;
}
inline TBits operator&(const TBits& lhs, const TBits& rhs)
{
TBits result(lhs);
result &= rhs;
return result;
}
inline TBits operator|(const TBits& lhs, const TBits& rhs)
{
TBits result(lhs);
result |= rhs;
return result;
}
inline TBits operator^(const TBits& lhs, const TBits& rhs)
{
TBits result(lhs);
result ^= rhs;
return result;
}
inline ostream &operator<<(ostream& os, const TBits& rhs)
{
rhs.Output(os); return os;
}
inline void TBits::SetBitNumber(UInt_t bitnumber, Bool_t value)
{
if (bitnumber >= fNbits) {
UInt_t new_size = (bitnumber/8) + 1;
if (new_size > fNbytes) {
new_size *= 2;
UChar_t *old_location = fAllBits;
fAllBits = new UChar_t[new_size];
memcpy(fAllBits,old_location,fNbytes);
memset(fAllBits+fNbytes ,0, new_size-fNbytes);
fNbytes = new_size;
delete [] old_location;
}
fNbits = bitnumber+1;
}
UInt_t loc = bitnumber/8;
UChar_t bit = bitnumber%8;
if (value)
fAllBits[loc] |= (1<<bit);
else
fAllBits[loc] &= (0xFF ^ (1<<bit));
}
inline Bool_t TBits::TestBitNumber(UInt_t bitnumber) const
{
if (bitnumber >= fNbits) return kFALSE;
UInt_t loc = bitnumber/8;
UChar_t value = fAllBits[loc];
UChar_t bit = bitnumber%8;
Bool_t result = (value & (1<<bit)) != 0;
return result;
}
inline void TBits::ResetBitNumber(UInt_t bitnumber)
{
SetBitNumber(bitnumber,kFALSE);
}
inline Bool_t TBits::operator[](UInt_t bitnumber) const
{
return TestBitNumber(bitnumber);
}
inline TBits::TReference& TBits::TReference::operator=(Bool_t val)
{
fBits.SetBitNumber(fPos,val); return *this;
}
inline TBits::TReference& TBits::TReference::operator=(const TReference& rhs)
{
fBits.SetBitNumber(fPos,rhs.fBits.TestBitNumber(rhs.fPos)); return *this;
}
#ifndef __CINT__
inline Bool_t TBits::TReference::operator~() const
{
return !fBits.TestBitNumber(fPos);
}
#endif
inline TBits::TReference::operator Bool_t() const
{
return fBits.TestBitNumber(fPos);
}
#endif