// RooSimultaneous facilitates simultaneous fitting of multiple PDFs
// to subsets of a given dataset.
// <p>
// The class takes an index category, which is interpreted as
// the data subset indicator, and a list of PDFs, each associated
// with a state of the index category. RooSimultaneous always returns
// the value of the PDF that is associated with the current value
// of the index category
// <p>
// Extended likelihood fitting is supported if all components support
// extended likelihood mode. The expected number of events by a RooSimultaneous
// is that of the component p.d.f. selected by the index category
// END_HTML
#include "RooFit.h"
#include "Riostream.h"
#include "TObjString.h"
#include "RooSimultaneous.h"
#include "RooAbsCategoryLValue.h"
#include "RooPlot.h"
#include "RooCurve.h"
#include "RooRealVar.h"
#include "RooAddPdf.h"
#include "RooAbsData.h"
#include "Roo1DTable.h"
#include "RooSimGenContext.h"
#include "RooSimSplitGenContext.h"
#include "RooDataSet.h"
#include "RooCmdConfig.h"
#include "RooNameReg.h"
#include "RooGlobalFunc.h"
#include "RooNameReg.h"
#include "RooMsgService.h"
#include "RooCategory.h"
#include "RooSuperCategory.h"
#include "RooDataHist.h"
#include "RooRandom.h"
#include "RooArgSet.h"
using namespace std ;
ClassImp(RooSimultaneous)
;
RooSimultaneous::RooSimultaneous(const char *name, const char *title,
RooAbsCategoryLValue& inIndexCat) :
RooAbsPdf(name,title),
_plotCoefNormSet("!plotCoefNormSet","plotCoefNormSet",this,kFALSE,kFALSE),
_plotCoefNormRange(0),
_partIntMgr(this,10),
_indexCat("indexCat","Index category",this,inIndexCat),
_numPdf(0)
{
}
RooSimultaneous::RooSimultaneous(const char *name, const char *title,
const RooArgList& inPdfList, RooAbsCategoryLValue& inIndexCat) :
RooAbsPdf(name,title),
_plotCoefNormSet("!plotCoefNormSet","plotCoefNormSet",this,kFALSE,kFALSE),
_plotCoefNormRange(0),
_partIntMgr(this,10),
_indexCat("indexCat","Index category",this,inIndexCat),
_numPdf(0)
{
if (inPdfList.getSize() != inIndexCat.numTypes()) {
coutE(InputArguments) << "RooSimultaneous::ctor(" << GetName()
<< " ERROR: Number PDF list entries must match number of index category states, no PDFs added" << endl ;
return ;
}
map<string,RooAbsPdf*> pdfMap ;
TIterator* pIter = inPdfList.createIterator() ;
TIterator* cIter = inIndexCat.typeIterator() ;
RooAbsPdf* pdf ;
RooCatType* type(0) ;
while ((pdf=(RooAbsPdf*)pIter->Next())) {
type = (RooCatType*) cIter->Next() ;
pdfMap[string(type->GetName())] = pdf ;
}
delete pIter ;
delete cIter ;
initialize(inIndexCat,pdfMap) ;
}
RooSimultaneous::RooSimultaneous(const char *name, const char *title,
map<string,RooAbsPdf*> pdfMap, RooAbsCategoryLValue& inIndexCat) :
RooAbsPdf(name,title),
_plotCoefNormSet("!plotCoefNormSet","plotCoefNormSet",this,kFALSE,kFALSE),
_plotCoefNormRange(0),
_partIntMgr(this,10),
_indexCat("indexCat","Index category",this,inIndexCat),
_numPdf(0)
{
initialize(inIndexCat,pdfMap) ;
}
namespace RooSimultaneousAux {
struct CompInfo {
RooAbsPdf* pdf ;
RooSimultaneous* simPdf ;
const RooAbsCategoryLValue* subIndex ;
RooArgSet* subIndexComps ;
} ;
}
void RooSimultaneous::initialize(RooAbsCategoryLValue& inIndexCat, std::map<std::string,RooAbsPdf*> pdfMap)
{
Bool_t simComps(kFALSE) ;
for (map<string,RooAbsPdf*>::iterator iter=pdfMap.begin() ; iter!=pdfMap.end() ; iter++) {
if (dynamic_cast<RooSimultaneous*>(iter->second)) {
simComps = kTRUE ;
break ;
}
}
if (!simComps) {
for (map<string,RooAbsPdf*>::iterator iter=pdfMap.begin() ; iter!=pdfMap.end() ; iter++) {
addPdf(*iter->second,iter->first.c_str()) ;
}
return ;
}
coutI(InputArguments) << "RooSimultaneous::initialize(" << GetName() << ") INFO: one or more input component of simultaneous p.d.f.s are"
<< " simultaneous p.d.f.s themselves, rewriting composite expressions as one-level simultaneous p.d.f. in terms of"
<< " final constituents and extended index category" << endl ;
RooArgSet allAuxCats ;
map<string,RooSimultaneousAux::CompInfo> compMap ;
for (map<string,RooAbsPdf*>::iterator iter=pdfMap.begin() ; iter!=pdfMap.end() ; iter++) {
RooSimultaneousAux::CompInfo ci ;
ci.pdf = iter->second ;
RooSimultaneous* simComp = dynamic_cast<RooSimultaneous*>(iter->second) ;
if (simComp) {
ci.simPdf = simComp ;
ci.subIndex = &simComp->indexCat() ;
ci.subIndexComps = simComp->indexCat().isFundamental() ? new RooArgSet(simComp->indexCat()) : simComp->indexCat().getVariables() ;
allAuxCats.add(*(ci.subIndexComps),kTRUE) ;
} else {
ci.simPdf = 0 ;
ci.subIndex = 0 ;
ci.subIndexComps = 0 ;
}
compMap[iter->first] = ci ;
}
RooArgSet allCats(inIndexCat) ;
allCats.add(allAuxCats) ;
string siname = Form("%s_index",GetName()) ;
RooSuperCategory* superIndex = new RooSuperCategory(siname.c_str(),siname.c_str(),allCats) ;
for (map<string,RooSimultaneousAux::CompInfo>::iterator citer = compMap.begin() ; citer != compMap.end() ; citer++) {
RooArgSet repliCats(allAuxCats) ;
if (citer->second.subIndexComps) {
repliCats.remove(*citer->second.subIndexComps) ;
delete citer->second.subIndexComps ;
}
inIndexCat.setLabel(citer->first.c_str()) ;
if (!citer->second.simPdf) {
RooSuperCategory repliSuperCat("tmp","tmp",repliCats) ;
TIterator* titer = repliSuperCat.typeIterator() ;
RooCatType* type ;
while ((type=(RooCatType*)titer->Next())) {
repliSuperCat.setLabel(type->GetName()) ;
string superLabel = superIndex->getLabel() ;
addPdf(*citer->second.pdf,superLabel.c_str()) ;
cxcoutD(InputArguments) << "RooSimultaneous::initialize(" << GetName()
<< ") assigning pdf " << citer->second.pdf->GetName() << " to super label " << superLabel << endl ;
}
} else {
if (repliCats.getSize()==0) {
TIterator* titer = citer->second.subIndex->typeIterator() ;
RooCatType* type ;
while ((type=(RooCatType*)titer->Next())) {
const_cast<RooAbsCategoryLValue*>(citer->second.subIndex)->setLabel(type->GetName()) ;
string superLabel = superIndex->getLabel() ;
RooAbsPdf* compPdf = citer->second.simPdf->getPdf(type->GetName()) ;
if (compPdf) {
addPdf(*compPdf,superLabel.c_str()) ;
cxcoutD(InputArguments) << "RooSimultaneous::initialize(" << GetName()
<< ") assigning pdf " << compPdf->GetName() << "(member of " << citer->second.pdf->GetName()
<< ") to super label " << superLabel << endl ;
} else {
coutW(InputArguments) << "RooSimultaneous::initialize(" << GetName() << ") WARNING: No p.d.f. associated with label "
<< type->GetName() << " for component RooSimultaneous p.d.f " << citer->second.pdf->GetName()
<< "which is associated with master index label " << citer->first << endl ;
}
}
delete titer ;
} else {
RooSuperCategory repliSuperCat("tmp","tmp",repliCats) ;
TIterator* triter = repliSuperCat.typeIterator() ;
TIterator* tsiter = citer->second.subIndex->typeIterator() ;
RooCatType* stype, *rtype ;
while ((stype=(RooCatType*)tsiter->Next())) {
const_cast<RooAbsCategoryLValue*>(citer->second.subIndex)->setLabel(stype->GetName()) ;
triter->Reset() ;
while ((rtype=(RooCatType*)triter->Next())) {
repliSuperCat.setLabel(rtype->GetName()) ;
string superLabel = superIndex->getLabel() ;
RooAbsPdf* compPdf = citer->second.simPdf->getPdf(stype->GetName()) ;
if (compPdf) {
addPdf(*compPdf,superLabel.c_str()) ;
cxcoutD(InputArguments) << "RooSimultaneous::initialize(" << GetName()
<< ") assigning pdf " << compPdf->GetName() << "(member of " << citer->second.pdf->GetName()
<< ") to super label " << superLabel << endl ;
} else {
coutW(InputArguments) << "RooSimultaneous::initialize(" << GetName() << ") WARNING: No p.d.f. associated with label "
<< stype->GetName() << " for component RooSimultaneous p.d.f " << citer->second.pdf->GetName()
<< "which is associated with master index label " << citer->first << endl ;
}
}
}
delete tsiter ;
delete triter ;
}
}
}
_indexCat.setArg(*superIndex) ;
addOwnedComponents(*superIndex) ;
}
RooSimultaneous::RooSimultaneous(const RooSimultaneous& other, const char* name) :
RooAbsPdf(other,name),
_plotCoefNormSet("!plotCoefNormSet",this,other._plotCoefNormSet),
_plotCoefNormRange(other._plotCoefNormRange),
_partIntMgr(other._partIntMgr,this),
_indexCat("indexCat",this,other._indexCat),
_numPdf(other._numPdf)
{
TIterator* pIter = other._pdfProxyList.MakeIterator() ;
RooRealProxy* proxy ;
while ((proxy=(RooRealProxy*)pIter->Next())) {
_pdfProxyList.Add(new RooRealProxy(proxy->GetName(),this,*proxy)) ;
}
delete pIter ;
}
RooSimultaneous::~RooSimultaneous()
{
_pdfProxyList.Delete() ;
}
RooAbsPdf* RooSimultaneous::getPdf(const char* catName) const
{
RooRealProxy* proxy = (RooRealProxy*) _pdfProxyList.FindObject(catName) ;
return proxy ? ((RooAbsPdf*)proxy->absArg()) : 0 ;
}
Bool_t RooSimultaneous::addPdf(const RooAbsPdf& pdf, const char* catLabel)
{
if (pdf.dependsOn(_indexCat.arg())) {
coutE(InputArguments) << "RooSimultaneous::addPdf(" << GetName() << "): ERROR, PDF " << pdf.GetName()
<< " overlaps with index category " << _indexCat.arg().GetName() << endl ;
return kTRUE ;
}
if (_pdfProxyList.FindObject(catLabel)) {
coutE(InputArguments) << "RooSimultaneous::addPdf(" << GetName() << "): ERROR, index state "
<< catLabel << " has already an associated PDF" << endl ;
return kTRUE ;
}
const RooSimultaneous* simPdf = dynamic_cast<const RooSimultaneous*>(&pdf) ;
if (simPdf) {
coutE(InputArguments) << "RooSimultaneous::addPdf(" << GetName()
<< ") ERROR: you cannot add a RooSimultaneous component to a RooSimultaneous using addPdf()."
<< " Use the constructor with RooArgList if input p.d.f.s or the map<string,RooAbsPdf&> instead." << endl ;
return kTRUE ;
} else {
TObject* proxy = new RooRealProxy(catLabel,catLabel,this,(RooAbsPdf&)pdf) ;
_pdfProxyList.Add(proxy) ;
_numPdf += 1 ;
}
return kFALSE ;
}
RooAbsPdf::ExtendMode RooSimultaneous::extendMode() const
{
Bool_t allCanExtend(kTRUE) ;
Bool_t anyMustExtend(kFALSE) ;
for (Int_t i=0 ; i<_numPdf ; i++) {
RooRealProxy* proxy = (RooRealProxy*) _pdfProxyList.FindObject(_indexCat.label()) ;
if (proxy) {
RooAbsPdf* pdf = (RooAbsPdf*) proxy->absArg() ;
if (!pdf->canBeExtended()) {
allCanExtend=kFALSE ;
}
if (pdf->mustBeExtended()) {
anyMustExtend=kTRUE;
}
}
}
if (anyMustExtend) {
return MustBeExtended ;
}
if (allCanExtend) {
return CanBeExtended ;
}
return CanNotBeExtended ;
}
Double_t RooSimultaneous::evaluate() const
{
RooRealProxy* proxy = (RooRealProxy*) _pdfProxyList.FindObject(_indexCat.label()) ;
if (proxy==0) return 0 ;
Double_t catFrac(1) ;
if (canBeExtended()) {
Double_t nEvtCat = ((RooAbsPdf*)(proxy->absArg()))->expectedEvents(_normSet) ;
Double_t nEvtTot(0) ;
TIterator* iter = _pdfProxyList.MakeIterator() ;
RooRealProxy* proxy2 ;
while((proxy2=(RooRealProxy*)iter->Next())) {
nEvtTot += ((RooAbsPdf*)(proxy2->absArg()))->expectedEvents(_normSet) ;
}
delete iter ;
catFrac=nEvtCat/nEvtTot ;
}
return ((RooAbsPdf*)(proxy->absArg()))->getVal(_normSet)*catFrac ;
}
Double_t RooSimultaneous::expectedEvents(const RooArgSet* nset) const
{
if (nset->contains(_indexCat.arg())) {
Double_t sum(0) ;
TIterator* iter = _pdfProxyList.MakeIterator() ;
RooRealProxy* proxy ;
while((proxy=(RooRealProxy*)iter->Next())) {
sum += ((RooAbsPdf*)(proxy->absArg()))->expectedEvents(nset) ;
}
delete iter ;
return sum ;
} else {
RooRealProxy* proxy = (RooRealProxy*) _pdfProxyList.FindObject(_indexCat.label()) ;
if (proxy==0) return 0 ;
return ((RooAbsPdf*)(proxy->absArg()))->expectedEvents(nset);
}
}
Int_t RooSimultaneous::getAnalyticalIntegralWN(RooArgSet& allVars, RooArgSet& analVars,
const RooArgSet* normSet, const char* rangeName) const
{
analVars.add(allVars) ;
Int_t code ;
CacheElem* cache = (CacheElem*) _partIntMgr.getObj(normSet,&analVars,0,RooNameReg::ptr(rangeName)) ;
if (cache) {
code = _partIntMgr.lastIndex() ;
return code+1 ;
}
cache = new CacheElem ;
TIterator* iter = _pdfProxyList.MakeIterator() ;
RooRealProxy* proxy ;
while((proxy=(RooRealProxy*)iter->Next())) {
RooAbsReal* pdfInt = proxy->arg().createIntegral(analVars,normSet,0,rangeName) ;
cache->_partIntList.addOwned(*pdfInt) ;
}
delete iter ;
code = _partIntMgr.setObj(normSet,&analVars,cache,RooNameReg::ptr(rangeName)) ;
return code+1 ;
}
Double_t RooSimultaneous::analyticalIntegralWN(Int_t code, const RooArgSet* normSet, const char* ) const
{
if (code==0) {
return getVal(normSet) ;
}
CacheElem* cache = (CacheElem*) _partIntMgr.getObjByIndex(code-1) ;
RooRealProxy* proxy = (RooRealProxy*) _pdfProxyList.FindObject(_indexCat.label()) ;
Int_t idx = _pdfProxyList.IndexOf(proxy) ;
return ((RooAbsReal*)cache->_partIntList.at(idx))->getVal(normSet) ;
}
RooPlot* RooSimultaneous::plotOn(RooPlot *frame, RooLinkedList& cmdList) const
{
if (plotSanityChecks(frame)) return frame ;
RooCmdConfig pc(Form("RooSimultaneous::plotOn(%s)",GetName())) ;
pc.defineString("sliceCatState","SliceCat",0,"",kTRUE) ;
pc.defineDouble("scaleFactor","Normalization",0,1.0) ;
pc.defineInt("scaleType","Normalization",0,RooAbsPdf::Relative) ;
pc.defineObject("sliceCatList","SliceCat",0,0,kTRUE) ;
pc.defineObject("projSet","Project",0) ;
pc.defineObject("sliceSet","SliceVars",0) ;
pc.defineObject("projDataSet","ProjData",0) ;
pc.defineObject("projData","ProjData",1) ;
pc.defineMutex("Project","SliceVars") ;
pc.allowUndefined() ;
pc.process(cmdList) ;
if (!pc.ok(kTRUE)) {
return frame ;
}
const RooAbsData* projData = (const RooAbsData*) pc.getObject("projData") ;
const RooArgSet* projDataSet = (const RooArgSet*) pc.getObject("projDataSet") ;
const RooArgSet* sliceSetTmp = (const RooArgSet*) pc.getObject("sliceSet") ;
RooArgSet* sliceSet = sliceSetTmp ? ((RooArgSet*) sliceSetTmp->Clone()) : 0 ;
const RooArgSet* projSet = (const RooArgSet*) pc.getObject("projSet") ;
Double_t scaleFactor = pc.getDouble("scaleFactor") ;
ScaleType stype = (ScaleType) pc.getInt("scaleType") ;
const char* sliceCatState = pc.getString("sliceCatState",0,kTRUE) ;
const RooLinkedList& sliceCatList = pc.getObjectList("sliceCatList") ;
if (sliceCatState) {
if (!sliceSet) {
sliceSet = new RooArgSet ;
}
char buf[1024] ;
strlcpy(buf,sliceCatState,1024) ;
const char* slabel = strtok(buf,",") ;
TIterator* iter = sliceCatList.MakeIterator() ;
RooCategory* scat ;
while((scat=(RooCategory*)iter->Next())) {
if (slabel) {
scat->setLabel(slabel) ;
sliceSet->add(*scat,kFALSE) ;
}
slabel = strtok(0,",") ;
}
delete iter ;
}
if (!projData) {
coutE(InputArguments) << "RooSimultaneous::plotOn(" << GetName() << ") ERROR: must have a projection dataset for index category" << endl ;
return frame ;
}
RooArgSet projectedVars ;
if (sliceSet) {
makeProjectionSet(frame->getPlotVar(),frame->getNormVars(),projectedVars,kTRUE) ;
TIterator* iter = sliceSet->createIterator() ;
RooAbsArg* sliceArg ;
while((sliceArg=(RooAbsArg*)iter->Next())) {
RooAbsArg* arg = projectedVars.find(sliceArg->GetName()) ;
if (arg) {
projectedVars.remove(*arg) ;
} else {
coutI(Plotting) << "RooAbsReal::plotOn(" << GetName() << ") slice variable "
<< sliceArg->GetName() << " was not projected anyway" << endl ;
}
}
delete iter ;
} else if (projSet) {
makeProjectionSet(frame->getPlotVar(),projSet,projectedVars,kFALSE) ;
} else {
makeProjectionSet(frame->getPlotVar(),frame->getNormVars(),projectedVars,kTRUE) ;
}
Bool_t projIndex(kFALSE) ;
if (!_indexCat.arg().isDerived()) {
if (!projData->get()->find(_indexCat.arg().GetName())) {
coutE(Plotting) << "RooSimultaneous::plotOn(" << GetName() << ") ERROR: Projection over index category "
<< "requested, but projection data set doesn't contain index category" << endl ;
return frame ;
}
if (projectedVars.find(_indexCat.arg().GetName())) {
projIndex=kTRUE ;
}
} else {
TIterator* sIter = _indexCat.arg().serverIterator() ;
RooAbsArg* server ;
RooArgSet projIdxServers ;
Bool_t anyServers(kFALSE) ;
while((server=(RooAbsArg*)sIter->Next())) {
if (projectedVars.find(server->GetName())) {
anyServers=kTRUE ;
projIdxServers.add(*server) ;
}
}
delete sIter ;
sIter = projIdxServers.createIterator() ;
Bool_t allServers(kTRUE) ;
while((server=(RooAbsArg*)sIter->Next())) {
if (!projData->get()->find(server->GetName())) {
allServers=kFALSE ;
}
}
delete sIter ;
if (!allServers) {
coutE(Plotting) << "RooSimultaneous::plotOn(" << GetName()
<< ") ERROR: Projection dataset doesn't contain complete set of index category dependents" << endl ;
return frame ;
}
if (anyServers) {
projIndex = kTRUE ;
}
}
Roo1DTable* wTable = projData->table(_indexCat.arg()) ;
if (!projIndex) {
coutI(Plotting) << "RooSimultaneous::plotOn(" << GetName() << ") plot on " << frame->getPlotVar()->GetName()
<< " represents a slice in the index category (" << _indexCat.arg().GetName() << ")" << endl ;
const RooAbsData* projDataTmp(projData) ;
if (projData) {
RooArgSet* indexCatComps = _indexCat.arg().getObservables(frame->getNormVars());
TString cutString ;
TIterator* compIter = indexCatComps->createIterator() ;
RooAbsCategory* idxComp ;
Bool_t first(kTRUE) ;
while((idxComp=(RooAbsCategory*)compIter->Next())) {
if (!first) {
cutString.Append("&&") ;
} else {
first=kFALSE ;
}
cutString.Append(Form("%s==%d",idxComp->GetName(),idxComp->getIndex())) ;
}
delete compIter ;
RooArgSet projDataVars(*projData->get()) ;
projDataVars.remove(*indexCatComps,kTRUE,kTRUE) ;
projDataTmp = ((RooAbsData*)projData)->reduce(projDataVars,cutString) ;
delete indexCatComps ;
}
RooCmdArg tmp1 = RooFit::Normalization(scaleFactor*wTable->getFrac(_indexCat.arg().getLabel()),stype) ;
RooCmdArg tmp2 = RooFit::ProjWData(*projDataSet,*projDataTmp) ;
RooLinkedList cmdList2(cmdList) ;
if (!cmdList.find("Asymmetry")) {
cmdList2.Add(&tmp1) ;
}
cmdList2.Add(&tmp2) ;
RooPlot* retFrame = getPdf(_indexCat.arg().getLabel())->plotOn(frame,cmdList2) ;
if (projDataTmp) {
delete projDataTmp ;
}
delete wTable ;
delete sliceSet ;
return retFrame ;
}
RooArgSet* idxCloneSet = (RooArgSet*) RooArgSet(_indexCat.arg()).snapshot(kTRUE) ;
RooAbsCategoryLValue* idxCatClone = (RooAbsCategoryLValue*) idxCloneSet->find(_indexCat.arg().GetName()) ;
RooArgSet* idxCompSliceSet = _indexCat.arg().getObservables(frame->getNormVars()) ;
idxCompSliceSet->remove(projectedVars,kTRUE,kTRUE) ;
TIterator* idxCompSliceIter = idxCompSliceSet->createIterator() ;
RooArgList pdfCompList ;
RooArgList wgtCompList ;
RooRealProxy* proxy ;
TIterator* pIter = _pdfProxyList.MakeIterator() ;
Double_t sumWeight(0) ;
while((proxy=(RooRealProxy*)pIter->Next())) {
idxCatClone->setLabel(proxy->name()) ;
Bool_t skip(kFALSE) ;
idxCompSliceIter->Reset() ;
RooAbsCategory* idxSliceComp ;
while((idxSliceComp=(RooAbsCategory*)idxCompSliceIter->Next())) {
RooAbsCategory* idxComp = (RooAbsCategory*) idxCloneSet->find(idxSliceComp->GetName()) ;
if (idxComp->getIndex()!=idxSliceComp->getIndex()) {
skip=kTRUE ;
break ;
}
}
if (skip) continue ;
RooRealVar *wgtVar = new RooRealVar(proxy->name(),"coef",wTable->getFrac(proxy->name())) ;
wgtCompList.addOwned(*wgtVar) ;
sumWeight += wTable->getFrac(proxy->name()) ;
pdfCompList.add(proxy->arg()) ;
}
TString plotVarName(GetName()) ;
RooAddPdf *plotVar = new RooAddPdf(plotVarName,"weighted sum of RS components",pdfCompList,wgtCompList) ;
if (_plotCoefNormSet.getSize()>0) {
plotVar->fixAddCoefNormalization(_plotCoefNormSet) ;
}
RooAbsData* projDataTmp(0) ;
RooArgSet projSetTmp ;
if (projData) {
TString cutString ;
if (idxCompSliceSet->getSize()>0) {
idxCompSliceIter->Reset() ;
RooAbsCategory* idxSliceComp ;
Bool_t first(kTRUE) ;
while((idxSliceComp=(RooAbsCategory*)idxCompSliceIter->Next())) {
if (!first) {
cutString.Append("&&") ;
} else {
first=kFALSE ;
}
cutString.Append(Form("%s==%d",idxSliceComp->GetName(),idxSliceComp->getIndex())) ;
}
}
RooArgSet projDataVars(*projData->get()) ;
RooArgSet* idxCatServers = _indexCat.arg().getObservables(frame->getNormVars()) ;
projDataVars.remove(*idxCatServers,kTRUE,kTRUE) ;
if (idxCompSliceSet->getSize()>0) {
projDataTmp = ((RooAbsData*)projData)->reduce(projDataVars,cutString) ;
} else {
projDataTmp = ((RooAbsData*)projData)->reduce(projDataVars) ;
}
if (projSet) {
projSetTmp.add(*projSet) ;
projSetTmp.remove(*idxCatServers,kTRUE,kTRUE);
}
delete idxCatServers ;
}
if (_indexCat.arg().isDerived() && idxCompSliceSet->getSize()>0) {
coutI(Plotting) << "RooSimultaneous::plotOn(" << GetName() << ") plot on " << frame->getPlotVar()->GetName()
<< " represents a slice in index category components " << *idxCompSliceSet << endl ;
RooArgSet* idxCompProjSet = _indexCat.arg().getObservables(frame->getNormVars()) ;
idxCompProjSet->remove(*idxCompSliceSet,kTRUE,kTRUE) ;
if (idxCompProjSet->getSize()>0) {
coutI(Plotting) << "RooSimultaneous::plotOn(" << GetName() << ") plot on " << frame->getPlotVar()->GetName()
<< " averages with data index category components " << *idxCompProjSet << endl ;
}
delete idxCompProjSet ;
} else {
coutI(Plotting) << "RooSimultaneous::plotOn(" << GetName() << ") plot on " << frame->getPlotVar()->GetName()
<< " averages with data index category (" << _indexCat.arg().GetName() << ")" << endl ;
}
RooLinkedList cmdList2(cmdList) ;
RooCmdArg tmp1 = RooFit::Normalization(scaleFactor*sumWeight,stype) ;
RooCmdArg tmp2 = RooFit::ProjWData(*projDataSet,*projDataTmp) ;
if (!cmdList.find("Asymmetry")) {
cmdList2.Add(&tmp1) ;
}
cmdList2.Add(&tmp2) ;
RooPlot* frame2 ;
if (projSetTmp.getSize()>0) {
RooCmdArg tmp3 = RooFit::Project(projSetTmp) ;
cmdList2.Add(&tmp3) ;
frame2 = plotVar->plotOn(frame,cmdList2) ;
} else {
frame2 = plotVar->plotOn(frame,cmdList2) ;
}
delete sliceSet ;
delete pIter ;
delete wTable ;
delete idxCloneSet ;
delete idxCompSliceIter ;
delete idxCompSliceSet ;
delete plotVar ;
if (projDataTmp) delete projDataTmp ;
return frame2 ;
}
RooPlot* RooSimultaneous::plotOn(RooPlot *frame, Option_t* drawOptions, Double_t scaleFactor,
ScaleType stype, const RooAbsData* projData, const RooArgSet* projSet,
Double_t , Bool_t , const RooArgSet* ,
Double_t , Double_t , RooCurve::WingMode ) const
{
RooLinkedList cmdList ;
cmdList.Add(new RooCmdArg(RooFit::DrawOption(drawOptions))) ;
cmdList.Add(new RooCmdArg(RooFit::Normalization(scaleFactor,stype))) ;
if (projData) cmdList.Add(new RooCmdArg(RooFit::ProjWData(*projData))) ;
if (projSet) cmdList.Add(new RooCmdArg(RooFit::Project(*projSet))) ;
RooPlot* ret = plotOn(frame,cmdList) ;
cmdList.Delete() ;
return ret ;
}
void RooSimultaneous::selectNormalization(const RooArgSet* normSet, Bool_t )
{
_plotCoefNormSet.removeAll() ;
if (normSet) _plotCoefNormSet.add(*normSet) ;
}
void RooSimultaneous::selectNormalizationRange(const char* normRange2, Bool_t )
{
_plotCoefNormRange = RooNameReg::ptr(normRange2) ;
}
RooAbsGenContext* RooSimultaneous::autoGenContext(const RooArgSet &vars, const RooDataSet* prototype,
const RooArgSet* auxProto, Bool_t verbose, Bool_t autoBinned, const char* binnedTag) const
{
const char* idxCatName = _indexCat.arg().GetName() ;
if (vars.find(idxCatName) && prototype==0 && (auxProto==0 || auxProto->getSize()==0) && (autoBinned || (binnedTag && strlen(binnedTag)))) {
return new RooSimSplitGenContext(*this,vars,verbose,autoBinned,binnedTag) ;
} else {
return genContext(vars,prototype,auxProto,verbose) ;
}
}
RooAbsGenContext* RooSimultaneous::genContext(const RooArgSet &vars, const RooDataSet *prototype,
const RooArgSet* auxProto, Bool_t verbose) const
{
const char* idxCatName = _indexCat.arg().GetName() ;
const RooArgSet* protoVars = prototype ? prototype->get() : 0 ;
if (vars.find(idxCatName) || (protoVars && protoVars->find(idxCatName))) {
return new RooSimGenContext(*this,vars,prototype,auxProto,verbose) ;
} else if (_indexCat.arg().isDerived()) {
Bool_t anyServer(kFALSE), allServers(kTRUE) ;
if (prototype) {
TIterator* sIter = _indexCat.arg().serverIterator() ;
RooAbsArg* server ;
while((server=(RooAbsArg*)sIter->Next())) {
if (prototype->get()->find(server->GetName())) {
anyServer=kTRUE ;
} else {
allServers=kFALSE ;
}
}
delete sIter ;
} else {
allServers=kTRUE ;
}
if (allServers) {
return new RooSimGenContext(*this,vars,prototype,auxProto,verbose) ;
} else if (!allServers && anyServer) {
coutE(Plotting) << "RooSimultaneous::genContext: ERROR: prototype must include either all "
<< " components of the RooSimultaneous index category or none " << endl ;
return 0 ;
}
}
RooRealProxy* proxy = (RooRealProxy*) _pdfProxyList.FindObject(_indexCat.arg().getLabel()) ;
if (!proxy) {
coutE(InputArguments) << "RooSimultaneous::genContext(" << GetName()
<< ") ERROR: no PDF associated with current state ("
<< _indexCat.arg().GetName() << "=" << _indexCat.arg().getLabel() << ")" << endl ;
return 0 ;
}
return ((RooAbsPdf*)proxy->absArg())->genContext(vars,prototype,auxProto,verbose) ;
}
RooDataHist* RooSimultaneous::fillDataHist(RooDataHist *hist,
const RooArgSet* nset,
Double_t scaleFactor,
Bool_t correctForBinVolume,
Bool_t showProgress) const
{
if (RooAbsReal::fillDataHist (hist, nset, scaleFactor,
correctForBinVolume, showProgress) == 0)
return 0;
Double_t sum = 0;
for (int i=0 ; i<hist->numEntries() ; i++) {
hist->get(i) ;
sum += hist->weight();
}
if (sum != 0) {
for (int i=0 ; i<hist->numEntries() ; i++) {
hist->get(i) ;
hist->set (hist->weight() / sum);
}
}
return hist;
}
RooDataSet* RooSimultaneous::generateSimGlobal(const RooArgSet& whatVars, Int_t nEvents)
{
RooArgSet* globClone = (RooArgSet*) whatVars.snapshot() ;
RooDataSet* data = new RooDataSet("gensimglobal","gensimglobal",whatVars) ;
TIterator* iter = indexCat().typeIterator() ;
for (Int_t i=0 ; i<nEvents ; i++) {
iter->Reset() ;
RooCatType* tt ;
while((tt=(RooCatType*) iter->Next())) {
RooAbsPdf* pdftmp = getPdf(tt->GetName()) ;
RooArgSet* globtmp = pdftmp->getObservables(whatVars) ;
RooDataSet* tmp = pdftmp->generate(*globtmp,1) ;
*globClone = *tmp->get(0) ;
delete globtmp ;
delete tmp ;
}
data->add(*globClone) ;
}
delete iter ;
delete globClone ;
return data ;
}