// RooSimGenContext is an efficient implementation of the generator context
// specific for RooSimultaneous PDFs when generating more than one of the
// component pdfs.
// END_HTML
#include "RooFit.h"
#include "Riostream.h"
#include "RooSimGenContext.h"
#include "RooSimultaneous.h"
#include "RooRealProxy.h"
#include "RooDataSet.h"
#include "Roo1DTable.h"
#include "RooCategory.h"
#include "RooMsgService.h"
#include "RooRandom.h"
#include "RooGlobalFunc.h"
using namespace RooFit ;
#include <string>
ClassImp(RooSimGenContext)
;
RooSimGenContext::RooSimGenContext(const RooSimultaneous &model, const RooArgSet &vars,
const RooDataSet *prototype, const RooArgSet* auxProto, Bool_t verbose) :
RooAbsGenContext(model,vars,prototype,auxProto,verbose), _pdf(&model), _protoData(0)
{
RooAbsCategory *idxCat = (RooAbsCategory*) model._indexCat.absArg() ;
RooArgSet pdfVars(vars) ;
RooArgSet allPdfVars(pdfVars) ;
if (prototype) allPdfVars.add(*prototype->get(),kTRUE) ;
if (!idxCat->isDerived()) {
pdfVars.remove(*idxCat,kTRUE,kTRUE) ;
Bool_t doGenIdx = allPdfVars.find(idxCat->GetName())?kTRUE:kFALSE ;
if (!doGenIdx) {
oocoutE(_pdf,Generation) << "RooSimGenContext::ctor(" << GetName() << ") ERROR: This context must"
<< " generate the index category" << endl ;
_isValid = kFALSE ;
_numPdf = 0 ;
_haveIdxProto = kFALSE ;
return ;
}
} else {
TIterator* sIter = idxCat->serverIterator() ;
RooAbsArg* server ;
Bool_t anyServer(kFALSE), allServers(kTRUE) ;
while((server=(RooAbsArg*)sIter->Next())) {
if (vars.find(server->GetName())) {
anyServer=kTRUE ;
pdfVars.remove(*server,kTRUE,kTRUE) ;
} else {
allServers=kFALSE ;
}
}
delete sIter ;
if (anyServer && !allServers) {
oocoutE(_pdf,Generation) << "RooSimGenContext::ctor(" << GetName() << ") ERROR: This context must"
<< " generate all components of a derived index category" << endl ;
_isValid = kFALSE ;
_numPdf = 0 ;
_haveIdxProto = kFALSE ;
return ;
}
}
_haveIdxProto = prototype ? kTRUE : kFALSE ;
_idxCatName = idxCat->GetName() ;
if (!_haveIdxProto && !model.canBeExtended()) {
oocoutE(_pdf,Generation) << "RooSimGenContext::ctor(" << GetName() << ") ERROR: Need either extended mode"
<< " or prototype data to calculate number of events per category" << endl ;
_isValid = kFALSE ;
_numPdf = 0 ;
return ;
}
_numPdf = model._pdfProxyList.GetSize() ;
_fracThresh = new Double_t[_numPdf+1] ;
_fracThresh[0] = 0 ;
_proxyIter = model._pdfProxyList.MakeIterator() ;
_allVarsPdf.add(allPdfVars) ;
RooRealProxy* proxy ;
RooAbsPdf* pdf ;
Int_t i(1) ;
while((proxy=(RooRealProxy*)_proxyIter->Next())) {
pdf=(RooAbsPdf*)proxy->absArg() ;
RooAbsGenContext* cx = pdf->genContext(pdfVars,prototype,auxProto,verbose) ;
cx->SetName(proxy->name()) ;
_gcList.push_back(cx) ;
_gcIndex.push_back(idxCat->lookupType(proxy->name())->getVal()) ;
_fracThresh[i] = _fracThresh[i-1] + (_haveIdxProto?0:pdf->expectedEvents(&allPdfVars)) ;
i++ ;
}
if (!_haveIdxProto) {
for(i=0 ; i<_numPdf ; i++)
_fracThresh[i] /= _fracThresh[_numPdf] ;
}
_idxCatSet = (RooArgSet*) RooArgSet(model._indexCat.arg()).snapshot(kTRUE) ;
if (!_idxCatSet) {
oocoutE(_pdf,Generation) << "RooSimGenContext::RooSimGenContext(" << GetName() << ") Couldn't deep-clone index category, abort," << endl ;
throw std::string("RooSimGenContext::RooSimGenContext() Couldn't deep-clone index category, abort") ;
}
_idxCat = (RooAbsCategoryLValue*) _idxCatSet->find(model._indexCat.arg().GetName()) ;
}
RooSimGenContext::~RooSimGenContext()
{
delete[] _fracThresh ;
delete _idxCatSet ;
for (vector<RooAbsGenContext*>::iterator iter = _gcList.begin() ; iter!=_gcList.end() ; ++iter) {
delete (*iter) ;
}
delete _proxyIter ;
if (_protoData) delete _protoData ;
}
void RooSimGenContext::attach(const RooArgSet& args)
{
if (_idxCat->isDerived()) {
_idxCat->recursiveRedirectServers(args,kTRUE) ;
}
for (vector<RooAbsGenContext*>::iterator iter = _gcList.begin() ; iter!=_gcList.end() ; ++iter) {
(*iter)->attach(args) ;
}
}
void RooSimGenContext::initGenerator(const RooArgSet &theEvent)
{
if (_idxCat->isDerived()) {
_idxCat->recursiveRedirectServers(theEvent,kTRUE) ;
} else {
_idxCat = (RooAbsCategoryLValue*) theEvent.find(_idxCat->GetName()) ;
}
updateFractions() ;
for (vector<RooAbsGenContext*>::iterator iter = _gcList.begin() ; iter!=_gcList.end() ; ++iter) {
(*iter)->initGenerator(theEvent) ;
}
}
RooDataSet* RooSimGenContext::createDataSet(const char* name, const char* title, const RooArgSet& obs)
{
if (!obs.contains(*_idxCat)) {
return new RooDataSet(name,title,obs) ;
}
if (!_protoData) {
map<string,RooAbsData*> dmap ;
RooCatType* state ;
TIterator* iter = _idxCat->typeIterator() ;
while((state=(RooCatType*)iter->Next())) {
RooAbsPdf* slicePdf = _pdf->getPdf(state->GetName()) ;
RooArgSet* sliceObs = slicePdf->getObservables(obs) ;
std::string sliceName = Form("%s_slice_%s",name,state->GetName()) ;
std::string sliceTitle = Form("%s (index slice %s)",title,state->GetName()) ;
RooDataSet* dset = new RooDataSet(sliceName.c_str(),sliceTitle.c_str(),*sliceObs) ;
dmap[state->GetName()] = dset ;
delete sliceObs ;
}
delete iter ;
_protoData = new RooDataSet(name, title, obs, Index((RooCategory&)*_idxCat), Link(dmap), OwnLinked()) ;
}
RooDataSet* emptyClone = new RooDataSet(*_protoData,name) ;
return emptyClone ;
}
void RooSimGenContext::generateEvent(RooArgSet &theEvent, Int_t remaining)
{
if (_haveIdxProto) {
Int_t gidx(0), cidx =_idxCat->getIndex() ;
for (Int_t i=0 ; i<(Int_t)_gcIndex.size() ; i++) {
if (_gcIndex[i]==cidx) { gidx = i ; break ; }
}
RooAbsGenContext* cx = _gcList[gidx] ;
if (cx) {
cx->generateEvent(theEvent,remaining) ;
} else {
oocoutW(_pdf,Generation) << "RooSimGenContext::generateEvent: WARNING, no PDF to generate event of type " << cidx << endl ;
}
} else {
Double_t rand = RooRandom::uniform() ;
Int_t i=0 ;
for (i=0 ; i<_numPdf ; i++) {
if (rand>_fracThresh[i] && rand<_fracThresh[i+1]) {
RooAbsGenContext* gen=_gcList[i] ;
gen->generateEvent(theEvent,remaining) ;
_idxCat->setIndexFast(_gcIndex[i]) ;
return ;
}
}
}
}
void RooSimGenContext::updateFractions()
{
if (_haveIdxProto) return ;
RooRealProxy* proxy ;
RooAbsPdf* pdf ;
Int_t i(1) ;
_proxyIter->Reset() ;
while((proxy=(RooRealProxy*)_proxyIter->Next())) {
pdf=(RooAbsPdf*)proxy->absArg() ;
_fracThresh[i] = _fracThresh[i-1] + (_haveIdxProto?0:pdf->expectedEvents(&_allVarsPdf)) ;
i++ ;
}
if (!_haveIdxProto) {
for(i=0 ; i<_numPdf ; i++)
_fracThresh[i] /= _fracThresh[_numPdf] ;
}
}
void RooSimGenContext::setProtoDataOrder(Int_t* lut)
{
RooAbsGenContext::setProtoDataOrder(lut) ;
for (vector<RooAbsGenContext*>::iterator iter = _gcList.begin() ; iter!=_gcList.end() ; ++iter) {
(*iter)->setProtoDataOrder(lut) ;
}
}
void RooSimGenContext::printMultiline(ostream &os, Int_t content, Bool_t verbose, TString indent) const
{
RooAbsGenContext::printMultiline(os,content,verbose,indent) ;
os << indent << "--- RooSimGenContext ---" << endl ;
os << indent << "Using PDF ";
_pdf->printStream(os,kName|kArgs|kClassName,kSingleLine,indent);
os << indent << "List of component generators" << endl ;
TString indent2(indent) ;
indent2.Append(" ") ;
for (vector<RooAbsGenContext*>::const_iterator iter = _gcList.begin() ; iter!=_gcList.end() ; ++iter) {
(*iter)->printMultiline(os,content,verbose,indent2);
}
}