// RooNumIntFactory is a factory to instantiate numeric integrators
// from a given function binding and a given configuration. The factory
// searches for a numeric integrator registered with the factory that
// has the ability to perform the numeric integration. The choice of
// method may depend on the number of dimensions integrated,
// the nature of the integration limits (closed or open ended) and
// the preference of the caller as encoded in the configuration object.
// END_HTML
#include "TClass.h"
#include "Riostream.h"
#include "RooFit.h"
#include "RooNumIntFactory.h"
#include "RooArgSet.h"
#include "RooAbsFunc.h"
#include "RooNumIntConfig.h"
#include "RooNumber.h"
#include "RooIntegrator1D.h"
#include "RooIntegrator2D.h"
#include "RooSegmentedIntegrator1D.h"
#include "RooSegmentedIntegrator2D.h"
#include "RooImproperIntegrator1D.h"
#include "RooMCIntegrator.h"
#include "RooGaussKronrodIntegrator1D.h"
#include "RooAdaptiveGaussKronrodIntegrator1D.h"
#include "RooAdaptiveIntegratorND.h"
#include "RooSentinel.h"
#include "RooMsgService.h"
using namespace std ;
ClassImp(RooNumIntFactory)
;
RooNumIntFactory* RooNumIntFactory::_instance = 0 ;
RooNumIntFactory::RooNumIntFactory()
{
_instance = this ;
RooIntegrator1D::registerIntegrator(*this) ;
RooIntegrator2D::registerIntegrator(*this) ;
RooSegmentedIntegrator1D::registerIntegrator(*this) ;
RooSegmentedIntegrator2D::registerIntegrator(*this) ;
RooImproperIntegrator1D::registerIntegrator(*this) ;
RooMCIntegrator::registerIntegrator(*this) ;
RooAdaptiveGaussKronrodIntegrator1D::registerIntegrator(*this) ;
RooGaussKronrodIntegrator1D::registerIntegrator(*this) ;
RooAdaptiveIntegratorND::registerIntegrator(*this) ;
RooNumIntConfig::defaultConfig().method1D().setLabel("RooIntegrator1D") ;
RooNumIntConfig::defaultConfig().method1DOpen().setLabel("RooImproperIntegrator1D") ;
RooNumIntConfig::defaultConfig().method2D().setLabel("RooAdaptiveIntegratorND") ;
RooNumIntConfig::defaultConfig().methodND().setLabel("RooAdaptiveIntegratorND") ;
}
RooNumIntFactory::~RooNumIntFactory()
{
std::map<std::string,pair<RooAbsIntegrator*,std::string> >::iterator iter = _map.begin() ;
while (iter != _map.end()) {
delete iter->second.first ;
++iter ;
}
}
RooNumIntFactory::RooNumIntFactory(const RooNumIntFactory& other) : TObject(other)
{
}
RooNumIntFactory& RooNumIntFactory::instance()
{
if (_instance==0) {
new RooNumIntFactory ;
RooSentinel::activate() ;
}
return *_instance ;
}
void RooNumIntFactory::cleanup()
{
if (_instance) {
delete _instance ;
_instance = 0 ;
}
}
Bool_t RooNumIntFactory::storeProtoIntegrator(RooAbsIntegrator* proto, const RooArgSet& defConfig, const char* depName)
{
TString name = proto->IsA()->GetName() ;
if (getProtoIntegrator(name)) {
return kTRUE ;
}
_map[name.Data()] = make_pair<RooAbsIntegrator*,std::string>(proto,depName) ;
RooNumIntConfig::defaultConfig().addConfigSection(proto,defConfig) ;
return kFALSE ;
}
const RooAbsIntegrator* RooNumIntFactory::getProtoIntegrator(const char* name)
{
if (_map.count(name)==0) {
return 0 ;
}
return _map[name].first ;
}
const char* RooNumIntFactory::getDepIntegratorName(const char* name)
{
if (_map.count(name)==0) {
return 0 ;
}
return _map[name].second.c_str() ;
}
RooAbsIntegrator* RooNumIntFactory::createIntegrator(RooAbsFunc& func, const RooNumIntConfig& config, Int_t ndimPreset)
{
Int_t ndim = ndimPreset>0 ? ndimPreset : ((Int_t)func.getDimension()) ;
Bool_t openEnded = kFALSE ;
Int_t i ;
for (i=0 ; i<ndim ; i++) {
if(RooNumber::isInfinite(func.getMinLimit(i)) ||
RooNumber::isInfinite(func.getMaxLimit(i))) {
openEnded = kTRUE ;
}
}
TString method ;
switch(ndim) {
case 1:
method = openEnded ? config.method1DOpen().getLabel() : config.method1D().getLabel() ;
break ;
case 2:
method = openEnded ? config.method2DOpen().getLabel() : config.method2D().getLabel() ;
break ;
default:
method = openEnded ? config.methodNDOpen().getLabel() : config.methodND().getLabel() ;
break ;
}
if (!method.CompareTo("N/A")) {
oocoutE((TObject*)0,Integration) << "RooNumIntFactory::createIntegrator: No integration method has been defined for "
<< (openEnded?"an open ended ":"a ") << ndim << "-dimensional integral" << endl ;
return 0 ;
}
const RooAbsIntegrator* proto = getProtoIntegrator(method) ;
RooAbsIntegrator* engine = proto->clone(func,config) ;
if (config.printEvalCounter()) {
engine->setPrintEvalCounter(kTRUE) ;
}
return engine ;
}