/*****************************************************************************
* Project: RooFit *
* Package: RooFitCore *
* File: $Id: RooNumConvPdf.cc,v 1.8 2005/06/20 15:44:55 wverkerke Exp $
* 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) *
*****************************************************************************/
// -- CLASS DESCRIPTION [PDF] --
// Numeric 1-dimensional convolution operator PDF. This class can convolve any PDF
// with any other PDF
//
// This class should not be used blindly as numeric convolution is computing
// intensive and prone to stability fitting problems. If an analytic convolution
// can be calculated, you should use that or implement it if not available.
// RooNumConvPdf implements reasonable defaults that should convolve most
// functions reasonably well, but results strongly depend on the shape of your
// input PDFS so always check your result.
//
// The default integration engine for the numeric convolution is the
// adaptive Gauss-Kronrod method, which empirically seems the most robust
// for this task. You can override the convolution integration settings via
// the RooNumIntConfig object reference returned by the convIntConfig() member
// function
//
// By default the numeric convolution is integrated from -infinity to
// +infinity through a x -> 1/x coordinate transformation of the
// tails. For convolution with a very small bandwidth it may be
// advantageous (for both CPU consumption and stability) if the
// integration domain is limited to a finite range. The function
// setConvolutionWindow(mean,width,scale) allows to set a sliding
// window around the x value to be calculated taking a RooAbsReal
// expression for an offset and a width to be taken around the x
// value. These input expression can be RooFormulaVars or other
// function objects although the 3d 'scale' argument 'scale'
// multiplies the width RooAbsReal expression given in the 2nd
// argument, allowing for an appropriate window definition for most
// cases without need for a RooFormulaVar object: e.g. a Gaussian
// resolution PDF do setConvolutionWindow(gaussMean,gaussSigma,5)
// Note that for a 'wide' Gaussian the -inf to +inf integration
// may converge more quickly than that over a finite range!
//
// The default numeric precision is 1e-7, i.e. the global default for
// numeric integration but you should experiment with this value to
// see if it is sufficient for example by studying the number of function
// calls that MINUIT needs to fit your function as function of the
// convolution precision.
#include "RooFit.h"
#include "Riostream.h"
#include "Riostream.h"
#include "TH2F.h"
#include "RooNumConvPdf.h"
#include "RooArgList.h"
#include "RooRealVar.h"
#include "RooFormulaVar.h"
#include "RooCustomizer.h"
#include "RooConvIntegrandBinding.h"
#include "RooNumIntFactory.h"
#include "RooGenContext.h"
#include "RooConvGenContext.h"
ClassImp(RooNumConvPdf)
;
RooNumConvPdf::RooNumConvPdf(const char *name, const char *title, RooRealVar& convVar, RooAbsPdf& pdf, RooAbsPdf& resmodel) :
RooAbsPdf(name,title),
_init(kFALSE),
_conv(0),
_origVar("origVar","Original Convolution variable",this,convVar),
_origPdf("origPdf","Original Input PDF",this,pdf),
_origModel("origModel","Original Resolution model",this,resmodel)
{
// Constructor of convolution operator PDF
//
// convVar : convolution variable (on which both pdf and resmodel should depend)
// pdf : input 'physics' pdf
// resmodel : input 'resultion' pdf
//
// output is pdf(x) (X) resmodel(x) = Int [ pdf(x') resmodel (x-x') ] dx'
//
}
RooNumConvPdf::RooNumConvPdf(const RooNumConvPdf& other, const char* name) :
RooAbsPdf(other,name),
_init(kFALSE),
_origVar("origVar",this,other._origVar),
_origPdf("origPdf",this,other._origPdf),
_origModel("origModel",this,other._origModel)
{
// Copy constructor
// Make temporary clone of original convolution to preserve configuration information
// This information will be propagated to a newly create convolution in a subsequent
// call to initialize()
if (other._conv) {
_conv = new RooNumConvolution(*other._conv,Form("%s_CONV",name?name:GetName())) ;
} else {
_conv = 0 ;
}
}
RooNumConvPdf::~RooNumConvPdf()
{
// Destructor
if (_init) {
delete _conv ;
}
}
Double_t RooNumConvPdf::evaluate() const
{
if (!_init) initialize() ;
return _conv->evaluate() ;
}
void RooNumConvPdf::initialize() const
{
// Save pointer to any prototype convolution object (only present if this object is made through
// a copy constructor)
RooNumConvolution* protoConv = _conv ;
// Optionally pass along configuration data from prototype object
_conv = new RooNumConvolution(Form("%s_CONV",GetName()),GetTitle(),var(),pdf(),model(),protoConv) ;
// Delete prototype object now
if (protoConv) {
delete protoConv ;
}
_init = kTRUE ;
}
RooAbsGenContext* RooNumConvPdf::genContext(const RooArgSet &vars, const RooDataSet *prototype,
const RooArgSet* auxProto, Bool_t verbose) const
{
if (!_init) initialize() ;
// Check if physics PDF and resolution model can both directly generate the convolution variable
RooArgSet* modelDep = _conv->model().getObservables(&vars) ;
modelDep->remove(_conv->var(),kTRUE,kTRUE) ;
Int_t numAddDep = modelDep->getSize() ;
delete modelDep ;
RooArgSet dummy ;
Bool_t pdfCanDir = (((RooAbsPdf&)_conv->pdf()).getGenerator(_conv->var(),dummy) != 0 && \
((RooAbsPdf&)_conv->pdf()).isDirectGenSafe(_conv->var())) ;
Bool_t resCanDir = (((RooAbsPdf&)_conv->model()).getGenerator(_conv->var(),dummy) !=0 &&
((RooAbsPdf&)_conv->model()).isDirectGenSafe(_conv->var())) ;
if (numAddDep>0 || !pdfCanDir || !resCanDir) {
// Any resolution model with more dependents than the convolution variable
// or pdf or resmodel do not support direct generation
return new RooGenContext(*this,vars,prototype,auxProto,verbose) ;
}
// Any other resolution model: use specialized generator context
return new RooConvGenContext(*this,vars,prototype,auxProto,verbose) ;
}
ROOT page - Class index - Class Hierarchy - Top of the page
This page has been automatically generated. If you have any comments or suggestions about the page layout send a mail to ROOT support, or contact the developers with any questions or problems regarding ROOT.