/***************************************************************************** * Project: RooFit * * Package: RooFitCore * * @(#)root/roofitcore:$Id$ * Authors: * * WV, Wouter Verkerke, UC Santa Barbara, verkerke@slac.stanford.edu * * DK, David Kirkby, UC Irvine, dkirkby@uci.edu * * * * Copyright (c) 2000-2005, Regents of the University of California * * and Stanford University. All rights reserved. * * * * Redistribution and use in source and binary forms, * * with or without modification, are permitted according to the terms * * listed in LICENSE (http://roofit.sourceforge.net/license.txt) * *****************************************************************************/ ////////////////////////////////////////////////////////////////////////////// // // BEGIN_HTML // Class RooNormSet cache manage the bookkeeping of multiple instances // of sets of integration and normalization observables that effectively // have the same definition. In complex function expression many // RooArgSets with the same contents may be passed to an object that // caches intermediate results dependent on the normalization/integration set // To avoid unnecessary cache faulting, This class tracks all instances // with the same contents and reports to the owner if the present nset/iset // is truely different from the current reference. Class RooNormSet only // evaluates each RooArgSet pointer once, it therefore assumes that // RooArgSets with normalization and/or integration sets are not changes // during their lifetime. // END_HTML // #include "RooFit.h" #include "RooNormSetCache.h" #include "RooArgSet.h" ClassImp(RooNormSetCache) ; #include <iostream> using namespace std ; //_____________________________________________________________________________ RooNormSetCache::RooNormSetCache(ULong_t max) : _max(max), _next(0), _set2RangeName(0) { } //_____________________________________________________________________________ RooNormSetCache::~RooNormSetCache() { // Destructor } //_____________________________________________________________________________ void RooNormSetCache::clear() { // Clear contents { PairIdxMapType tmpmap; tmpmap.swap(_pairToIdx); } { PairVectType tmppairvect; tmppairvect.swap(_pairs); } _next = 0; } //_____________________________________________________________________________ void RooNormSetCache::add(const RooArgSet* set1, const RooArgSet* set2) { // Add given pair of RooArgSet pointers to our store const Pair pair(set1, set2); PairIdxMapType::iterator it = _pairToIdx.lower_bound(pair); if (_pairToIdx.end() != it && !PairCmp()(it->first, pair) && !PairCmp()(pair, it->first)) { // not empty, and keys match - nothing to do return; } // register pair -> index mapping _pairToIdx.insert(it, std::make_pair(pair, ULong_t(_pairs.size()))); // save pair at that index _pairs.push_back(pair); // if the cache grew too large, start replacing in a round-robin fashion while (_pairs.size() > _max) { // new index of the pair: replace slot _next it->second = _next; // find and erase mapping of old pair in that slot _pairToIdx.erase(_pairs[_next]); // put new pair into new slot _pairs[_next] = _pairs.back(); // and erase the copy we no longer need _pairs.erase(_pairs.end() - 1); ++_next; _next %= _max; } } //_____________________________________________________________________________ Bool_t RooNormSetCache::autoCache(const RooAbsArg* self, const RooArgSet* set1, const RooArgSet* set2, const TNamed* set2RangeName, Bool_t doRefill) { // If RooArgSets set1 and set2 or sets with similar contents have // been seen by this cache manager before return kFALSE If not, // return kTRUE. If sets have not been seen and doRefill is true, // update cache reference to current input sets. // Automated cache management function - Returns kTRUE if cache is invalidated // A - Check if set1/2 are in cache and range name is identical if (set2RangeName == _set2RangeName && contains(set1,set2)) { return kFALSE ; } // B - Check if dependents(set1/set2) are compatible with current cache RooNameSet nset1d, nset2d; // cout << "RooNormSetCache::autoCache set1 = " << (set1?*set1:RooArgSet()) << " set2 = " << (set2?*set2:RooArgSet()) << endl; // if (set1) set1->Print("v"); // if (set2) set2->Print("v"); //if (self) self->Print("v"); RooArgSet *set1d, *set2d ; if (self) { set1d = set1 ? self->getObservables(*set1,kFALSE) : new RooArgSet; set2d = set2 ? self->getObservables(*set2,kFALSE) : new RooArgSet; } else { set1d = set1 ? (RooArgSet*)set1->snapshot() : new RooArgSet; set2d = set2 ? (RooArgSet*)set2->snapshot() : new RooArgSet; } // cout << "RooNormSetCache::autoCache set1d = " << *set1d << " set2 = " << *set2d << endl; nset1d.refill(*set1d); nset2d.refill(*set2d); if (nset1d == _name1 && nset2d == _name2 && _set2RangeName == set2RangeName) { // Compatible - Add current set1/2 to cache add(set1,set2); delete set1d; delete set2d; return kFALSE; } // C - Reset cache and refill with current state if (doRefill) { clear(); add(set1,set2); _name1.refill(*set1d); _name2.refill(*set2d); // cout << "RooNormSetCache::autoCache() _name1 refilled from " << *set1d << " to " ; _name1.printValue(cout) ; cout << endl; // cout << "RooNormSetCache::autoCache() _name2 refilled from " << *set2d << " to " ; _name2.printValue(cout) ; cout << endl; _set2RangeName = (TNamed*) set2RangeName; } delete set1d; delete set2d; return kTRUE; }