43 typedef RooProduct::ProdMap::iterator RPPMIter ;
44 std::pair<RPPMIter,RPPMIter> findOverlap2nd(RPPMIter i, RPPMIter end) ;
45 void dump_map(std::ostream& os, RPPMIter i, RPPMIter end) ;
73 _compRSet(
"!compRSet",
"Set of real product components",this),
74 _compCSet(
"!compCSet",
"Set of category product components",this),
77 for (
auto comp : prodSet) {
93 _compRSet(
"!compRSet",this,other._compRSet),
94 _compCSet(
"!compCSet",this,other._compCSet),
95 _cacheMgr(other._cacheMgr,this)
109 coutE(InputArguments) <<
"RooProduct::addTerm(" <<
GetName() <<
") ERROR: component " << term->
GetName()
110 <<
" is not of type RooAbsReal or RooAbsCategory" << std::endl ;
111 throw std::invalid_argument(
"RooProduct can only handle terms deriving from RooAbsReal or RooAbsCategory.");
125 for (
auto const* rcomp : static_range_cast<RooAbsReal*>(
_compRSet)) {
127 depends = rcomp->dependsOn(dep);
145 for (
auto const* rcomp : static_range_cast<RooAbsReal*>(
_compRSet)) {
146 if( !rcomp->dependsOn(allVars) ) indep->
add(*rcomp);
148 if (!indep->
empty()) {
149 map->push_back( std::make_pair(
new RooArgSet(),indep) );
155 for (
auto const* var : static_range_cast<RooAbsReal*>(allVars)) {
159 for (
auto const* rcomp2 : static_range_cast<RooAbsReal*>(
_compRSet)) {
160 if( rcomp2->dependsOn(*var) ) comps->
add(*rcomp2);
162 map->push_back( std::make_pair(vars,comps) );
168 std::pair<ProdMap::iterator,ProdMap::iterator> i = findOverlap2nd(map->begin(),map->end());
169 overlap = (i.first!=i.second);
171 i.first->first->add(*i.second->first);
174 for (
auto const* targ : *(i.second->second)) {
175 if (!i.first->second->find(*targ)) {
176 i.first->second->add(*targ) ;
181 delete i.second->first;
182 delete i.second->second;
183 map->erase(i.second);
192 for (ProdMap::iterator i = map->begin();i!=map->end();++i) {
193 nVar+=i->first->size();
194 nFunc+=i->second->size();
196 assert(nVar==allVars.
size());
213 Int_t sterileIndex(-1);
215 if (cache!=
nullptr) {
222 cxcoutD(Integration) <<
"RooProduct::getPartIntList(" <<
GetName() <<
") groupProductTerms returned map" ;
223 if (
dologD(Integration)) {
224 dump_map(
ccoutD(Integration),map->begin(),map->end());
225 ccoutD(Integration) << std::endl;
231 for (ProdMap::iterator iter = map->begin() ; iter != map->end() ; ++iter) {
233 delete iter->second ;
240 for (ProdMap::const_iterator i = map->begin();i!=map->end();++i) {
242 if (i->second->size()>1) {
244 auto ownedTerm = std::make_unique<RooProduct>(
name,
name,*i->second);
245 term = ownedTerm.get();
247 cxcoutD(Integration) <<
"RooProduct::getPartIntList(" <<
GetName() <<
") created subexpression " << term->
GetName() << std::endl;
249 assert(i->second->size()==1);
250 term =
static_cast<RooAbsReal*
>(i->second->at(0));
252 assert(term!=
nullptr);
253 if (i->first->empty()) {
255 cxcoutD(Integration) <<
"RooProduct::getPartIntList(" <<
GetName() <<
") adding simple factor " << term->
GetName() << std::endl;
257 std::unique_ptr<RooAbsReal> integral{term->
createIntegral(*i->first,isetRange)};
259 cxcoutD(Integration) <<
"RooProduct::getPartIntList(" <<
GetName() <<
") adding integral for " << term->
GetName() <<
" : " << integral->GetName() << std::endl;
266 cxcoutD(Integration) <<
"RooProduct::getPartIntList(" <<
GetName() <<
") created list " << cache->
_prodList <<
" with code " << code+1 << std::endl
267 <<
" for iset=" << *iset <<
" @" << iset <<
" range: " << (isetRange?isetRange:
"<none>") << std::endl ;
269 for (ProdMap::iterator iter = map->begin() ; iter != map->end() ; ++iter) {
271 delete iter->second ;
282 const char* rangeName)
const
289 assert(analVars.
empty());
290 analVars.
add(allVars) ;
303 if (cache==
nullptr) {
311 assert(cache!=
nullptr);
323 for (
const auto arg : partIntList) {
324 const auto term =
static_cast<const RooAbsReal*
>(arg);
325 double x = term->getVal();
340 for (
auto const* arg : terms) {
341 if (first) { first=
false;}
343 pname.
Append(arg->GetName());
359 auto rcomp =
static_cast<const RooAbsReal*
>(item);
361 prod *= rcomp->getVal(nset) ;
367 prod *= ccomp->getCurrentIndex() ;
377 std::size_t nEvents =
output.size();
379 for (
unsigned int i = 0; i < nEvents; ++i) {
384 auto rcomp =
static_cast<const RooAbsReal*
>(item);
385 auto componentValues = ctx.
at(rcomp);
387 for (
unsigned int i = 0; i < nEvents; ++i) {
388 output[i] *= componentValues.size() == 1 ? componentValues[0] : componentValues[i];
396 for (
unsigned int i = 0; i < nEvents; ++i) {
409 auto func =
static_cast<const RooAbsReal*
>(item);
411 if (std::list<double>* binb = func->binBoundaries(obs,xlo,xhi)) {
426 auto func =
static_cast<const RooAbsReal*
>(item);
428 if (func->dependsOn(obs) && !func->isBinnedDistribution(obs)) {
444 auto func =
static_cast<const RooAbsReal*
>(item);
446 if (std::list<double>* hint = func->plotSamplingHint(obs,xlo,xhi)) {
482 for (
const auto parg : comp) {
483 if (parg->isDerived()) {
484 if (parg->canNodeBeCached()==
Always) {
485 trackNodes.
add(*parg) ;
500 if (!first) { os <<
" * " ; }
else { first = false ; }
501 os << rcomp->GetName() ;
507 if (!first) { os <<
" * " ; }
else { first = false ; }
508 os << ccomp->GetName() ;
519 throw std::runtime_error(
"RooProduct::ioStreamerPass2(): the number of proxies in the proxy list should be at least 2!");
539 auto expectProxyIs = [
this](std::size_t idx,
RooAbsProxy * proxyInArg,
RooListProxy * ourProxy,
const char* memberName) {
540 if(proxyInArg != ourProxy) {
548 std::stringstream ss;
549 ss <<
"Problem when reading RooProduct instance \"" <<
GetName() <<
"\"!\n"
550 <<
" _proxyList[" << idx <<
"] was expected to be equal to " << memberName <<
", but it's not.\n"
551 <<
" - proxyList[" << idx <<
"] : ";
552 proxyInArg->print(ss,
true);
553 ss <<
"\n - " << memberName <<
" : " ;
554 ourProxy->print(ss,
true);
555 ss <<
"\n RooFit will resolve this inconsistency by making _proxyList[" << idx <<
"] point to " << memberName
557 coutW(LinkStateMgmt) << ss.str() << std::endl;
561 expectProxyIs(0, p0, &
_compRSet,
"_compRSet");
562 expectProxyIs(1, p1, &
_compCSet,
"_compCSet");
568std::pair<RPPMIter,RPPMIter> findOverlap2nd(RPPMIter i, RPPMIter end)
571 for (; i != end; ++i) {
572 for (RPPMIter j(i + 1); j != end; ++j) {
573 if (i->second->overlaps(*j->second)) {
574 return std::make_pair(i, j);
578 return std::make_pair(end,end);
582void dump_map(std::ostream& os, RPPMIter i, RPPMIter end)
588 if (first) { first=
false; }
589 else { os <<
" , " ; }
590 os << *(i->first) <<
" -> " << *(i->second) ;
Common abstract base class for objects that represent a value and a "shape" in RooFit.
RooFit::OwningPtr< RooArgSet > getParameters(const RooAbsData *data, bool stripDisconnected=true) const
Create a list of leaf nodes in the arg tree starting with ourself as top node that don't match any of...
virtual void ioStreamerPass2()
Method called by workspace container to finalize schema evolution issues that cannot be handled in a ...
Int_t numProxies() const
Return the number of registered proxies.
RooAbsProxy * getProxy(Int_t index) const
Return the nth proxy from the proxy list.
Abstract base class for objects to be stored in RooAbsCache cache manager objects.
A space to attach TBranches.
virtual value_type getCurrentIndex() const
Return index number of current state.
virtual bool add(const RooAbsArg &var, bool silent=false)
Add the specified argument to list.
Storage_t::size_type size() const
RooAbsArg * first() const
virtual bool addOwned(RooAbsArg &var, bool silent=false)
Add an argument and transfer the ownership to the collection.
Abstract interface for proxy classes.
const RooArgSet * nset() const
Abstract base class for objects that represent a real value that may appear on the left hand side of ...
Abstract base class for objects that represent a real value and implements functionality common to al...
bool _forceNumInt
Force numerical integration if flag set.
RooFit::OwningPtr< RooAbsReal > createIntegral(const RooArgSet &iset, const RooCmdArg &arg1, const RooCmdArg &arg2={}, const RooCmdArg &arg3={}, const RooCmdArg &arg4={}, const RooCmdArg &arg5={}, const RooCmdArg &arg6={}, const RooCmdArg &arg7={}, const RooCmdArg &arg8={}) const
Create an object that represents the integral of the function over one or more observables listed in ...
RooArgList is a container object that can hold multiple RooAbsArg objects.
RooArgSet is a container object that can hold multiple RooAbsArg objects.
Int_t setObj(const RooArgSet *nset, T *obj, const TNamed *isetRangeName=nullptr)
Setter function without integration set.
T * getObjByIndex(Int_t index) const
Retrieve payload object by slot index.
RooArgSet selectFromSet2(RooArgSet const &argSet, int index) const
Create RooArgSet containing the objects that are both in the cached set 2 with a given index and an i...
Int_t lastIndex() const
Return index of slot used in last get or set operation.
T * getObj(const RooArgSet *nset, Int_t *sterileIndex=nullptr, const TNamed *isetRangeName=nullptr)
Getter function without integration set.
bool add(const RooAbsArg &var, bool valueServer, bool shapeServer, bool silent)
Overloaded RooCollection_t::add() method insert object into set and registers object as server to own...
std::span< const double > at(RooAbsArg const *arg, RooAbsArg const *caller=nullptr)
std::span< double > output()
static const TNamed * ptr(const char *stringPtr)
Return a unique TNamed pointer for given C++ string.
RooArgList containedArgs(Action) override
Return list of all RooAbsArgs in cache element.
~CacheElem() override
Destructor.
Represents the product of a given set of RooAbsReal objects.
std::list< double > * binBoundaries(RooAbsRealLValue &, double, double) const override
Forward the plot sampling hint from the p.d.f. that defines the observable obs.
void doEval(RooFit::EvalContext &) const override
Base function for computing multiple values of a RooAbsReal.
double evaluate() const override
Evaluate product of input functions.
double analyticalIntegral(Int_t code, const char *rangeName=nullptr) const override
Calculate integral internally from appropriate partial integral cache.
void addTerm(RooAbsArg *term)
Add a term to this product.
std::list< double > * plotSamplingHint(RooAbsRealLValue &, double, double) const override
Forward the plot sampling hint from the p.d.f. that defines the observable obs.
ProdMap * groupProductTerms(const RooArgSet &) const
Group observables into subsets in which the product factorizes and that can thus be integrated separa...
double calculate(const RooArgList &partIntList) const
The cache manager.
bool forceAnalyticalInt(const RooAbsArg &dep) const override
Force internal handling of integration of given observable if any of the product terms depend on it.
void setCacheAndTrackHints(RooArgSet &) override
Label OK'ed components of a RooProduct with cache-and-track.
~RooProduct() override
Destructor.
bool isBinnedDistribution(const RooArgSet &obs) const override
Tests if the distribution is binned. Unless overridden by derived classes, this always returns false.
void ioStreamerPass2() override
Method called by workspace container to finalize schema evolution issues that cannot be handled in a ...
Int_t getPartIntList(const RooArgSet *iset, const char *rangeName=nullptr) const
Return list of (partial) integrals whose product defines the integral of this RooProduct over the obs...
Int_t getAnalyticalIntegralWN(RooArgSet &allVars, RooArgSet &analVars, const RooArgSet *normSet, const char *rangeName=nullptr) const override
Declare that we handle all integrations internally.
void printMetaArgs(std::ostream &os) const override
Customized printing of arguments of a RooProduct to more intuitively reflect the contents of the prod...
const char * makeFPName(const char *pfx, const RooArgSet &terms) const
Construct automatic name for internal product terms.
RooProduct()
Default constructor.
RooObjCacheManager _cacheMgr
const char * GetName() const override
Returns name of object.
void AddAt(TObject *obj, Int_t idx) override
Add object at position ids.
TObject * RemoveAt(Int_t idx) override
Remove object at index idx.
const char * Data() const
TString & Append(const char *cs)