#include "RooFit.h"
#include "TMath.h"
#include "TMath.h"
#include "TCollection.h"
#include "RooHashTable.h"
#include "RooLinkedList.h"
#include "RooAbsArg.h"
#include "RooSetPair.h"
using namespace std;
ClassImp(RooHashTable)
;
// RooHashTable implements a hash table for TObjects. The hashing can be
// done on the object addresses, object names, or using the objects
// internal hash method. This is a utility class for RooLinkedList
// that uses RooHashTable to speed up direct access to large collections.
// END_HTML
RooHashTable::RooHashTable(Int_t capacity, HashMethod hashMethod) :
_hashMethod(hashMethod)
{
if (capacity <= 0) {
capacity = TCollection::kInitHashTableCapacity;
}
_size = (Int_t)TMath::NextPrime(TMath::Max(capacity,(int)TCollection::kInitHashTableCapacity));
_arr = new RooLinkedList* [_size] ;
memset(_arr, 0, _size*sizeof(RooLinkedList*));
_usedSlots = 0 ;
_entries = 0 ;
}
RooHashTable::RooHashTable(const RooHashTable& other) :
TObject(other),
_hashMethod(other._hashMethod),
_usedSlots(other._usedSlots),
_entries(other._entries),
_size(other._size)
{
_arr = new RooLinkedList* [_size] ;
memset(_arr, 0, _size*sizeof(RooLinkedList*));
Int_t i ;
for (i=0 ; i<_size ; i++) {
if (other._arr[i]) {
_arr[i] = new RooLinkedList(*other._arr[i]) ;
}
}
}
void RooHashTable::add(TObject* arg, TObject* hashArg)
{
Int_t slot = hash(hashArg?hashArg:arg) % _size ;
if (!_arr[slot]) {
_arr[slot] = new RooLinkedList(0) ;
_arr[slot]->useNptr(kFALSE) ;
_usedSlots++ ;
}
_arr[slot]->Add(arg);
_entries++;
}
Bool_t RooHashTable::remove(TObject* arg, TObject* hashArg)
{
Int_t slot = hash(hashArg?hashArg:arg) % _size ;
if (_arr[slot]) {
if (_arr[slot]->Remove(arg)) {
_entries-- ;
if (_arr[slot]->GetSize()==0) {
delete _arr[slot] ;
_arr[slot] = 0 ;
_usedSlots-- ;
}
return kTRUE ;
}
}
if (_hashMethod != Name) return kFALSE;
RooAbsArg* p = dynamic_cast<RooAbsArg*>(arg);
if (p && !p->namePtr()->TestBit(RooNameReg::kRenamedArg)) return kFALSE;
Int_t i;
for (i=0 ; i<_size ; i++) {
if (i != slot && _arr[i] && _arr[i]->Remove(arg)) {
_entries-- ;
if (_arr[i]->GetSize()==0) {
delete _arr[i] ;
_arr[i] = 0 ;
_usedSlots-- ;
}
return kTRUE ;
}
}
return kFALSE ;
}
Double_t RooHashTable::avgCollisions() const
{
Int_t i,h[20] ;
for (i=0 ; i<20 ; i++) h[i]=0 ;
for (i=0 ; i<_size ; i++) {
if (_arr[i]) {
Int_t count = _arr[i]->GetSize() ;
if (count<20) {
h[count]++ ;
} else {
h[19]++ ;
}
} else {
h[0]++ ;
}
}
return 0 ;
}
Bool_t RooHashTable::replace(const TObject* oldArg, const TObject* newArg, const TObject* oldHashArg)
{
Int_t slot = hash(oldHashArg?oldHashArg:oldArg) % _size ;
if (_arr[slot]) {
Int_t newSlot = hash(newArg) % _size ;
if (newSlot == slot) {
return _arr[slot]->Replace(oldArg,newArg) ;
}
}
if (remove((TObject*)oldArg,(TObject*)oldHashArg)) {
add((TObject*)newArg);
return kTRUE;
}
return kFALSE ;
}
TObject* RooHashTable::find(const char* name) const
{
if (_hashMethod != Name) assert(0) ;
Int_t slot = TMath::Hash(name) % _size ;
if (_arr[slot]) return _arr[slot]->find(name) ;
return 0;
}
RooAbsArg* RooHashTable::findArg(const RooAbsArg* arg) const
{
if (_hashMethod != Name) assert(0) ;
Int_t slot = TMath::Hash(arg->GetName()) % _size ;
if (_arr[slot]) return _arr[slot]->findArg(arg) ;
return 0;
}
TObject* RooHashTable::find(const TObject* hashArg) const
{
RooLinkedListElem* elem = findLinkTo(hashArg) ;
return elem ? elem->_arg : 0 ;
}
RooLinkedListElem* RooHashTable::findLinkTo(const TObject* hashArg) const
{
if (_hashMethod != Pointer) assert(0) ;
Int_t slot = hash(hashArg) % _size ;
RooLinkedList* lst = _arr[slot];
if (lst) {
RooFIter it = lst->fwdIterator() ;
TObject* obj;
while ((obj=it.next())) {
RooLinkedListElem* elem = (RooLinkedListElem*)obj ;
if (elem->_arg == hashArg) return elem ;
}
}
return 0;
}
RooSetPair* RooHashTable::findSetPair(const RooArgSet* set1, const RooArgSet* set2) const
{
if (_hashMethod != Intrinsic) assert(0) ;
Int_t slot = RooSetPair(set1,set2).Hash() % _size ;
if (_arr[slot]) {
Int_t i ;
for (i=0 ; i<_arr[slot]->GetSize() ; i++) {
RooSetPair* pair = (RooSetPair*)_arr[slot]->At(i) ;
if (pair->_set1==set1 && pair->_set2==set2) {
return pair ;
}
}
}
return 0 ;
}
RooHashTable::~RooHashTable()
{
Int_t i ;
for (i=0 ; i<_size ; i++) {
if (_arr[i]) delete _arr[i] ;
}
delete[] _arr ;
}