// The RooWorkspace is a persistable container for RooFit projects. A workspace
// can contain and own variables, p.d.f.s, functions and datasets. All objects
// that live in the workspace are owned by the workspace. The import() method
// enforces consistency of objects upon insertion into the workspace (e.g. no
// duplicate object with the same name are allowed) and makes sure all objects
// in the workspace are connected to each other. Easy accessor methods like
// pdf(), var() and data() allow to refer to the contents of the workspace by
// object name. The entire RooWorkspace can be saved into a ROOT TFile and organises
// the consistent streaming of its contents without duplication.
// <p>
// If a RooWorkspace contains custom classes, i.e. classes not in the
// ROOT distribution, portability of workspaces can be enhanced by
// storing the source code of those classes in the workspace as well.
// This process is also organized by the workspace through the
// importClassCode() method.
// END_HTML
#include "RooFit.h"
#include "RooWorkspace.h"
#include "RooAbsPdf.h"
#include "RooRealVar.h"
#include "RooCategory.h"
#include "RooAbsData.h"
#include "RooCmdConfig.h"
#include "RooMsgService.h"
#include "RooConstVar.h"
#include "RooResolutionModel.h"
#include "TInterpreter.h"
#include "TClassTable.h"
#include "TBaseClass.h"
#include "TSystem.h"
#include "TRegexp.h"
#include "RooFactoryWSTool.h"
#include "RooAbsStudy.h"
#include "RooTObjWrap.h"
#include "RooAbsOptTestStatistic.h"
#include "TROOT.h"
#include "TFile.h"
#include "TH1.h"
#include "Api.h"
#include <map>
#include <string>
#include <list>
#include <set>
using namespace std ;
#if ROOT_VERSION_CODE <= ROOT_VERSION(5,19,02)
#include "Api.h"
#endif
#include "TClass.h"
#include "Riostream.h"
#include <string.h>
#include <assert.h>
ClassImp(RooWorkspace)
;
ClassImp(RooWorkspace::CodeRepo)
;
ClassImp(RooWorkspace::WSDir)
;
list<string> RooWorkspace::_classDeclDirList ;
list<string> RooWorkspace::_classImplDirList ;
string RooWorkspace::_classFileExportDir = ".wscode.%s.%s" ;
Bool_t RooWorkspace::_autoClass = kFALSE ;
void RooWorkspace::addClassDeclImportDir(const char* dir)
{
_classDeclDirList.push_back(dir) ;
}
void RooWorkspace::addClassImplImportDir(const char* dir)
{
_classImplDirList.push_back(dir) ;
}
void RooWorkspace::setClassFileExportDir(const char* dir)
{
if (dir) {
_classFileExportDir = dir ;
} else {
_classFileExportDir = ".wscode.%s.%s" ;
}
}
void RooWorkspace::autoImportClassCode(Bool_t flag)
{
_autoClass = flag ;
}
RooWorkspace::RooWorkspace() : _classes(this), _dir(0), _factory(0), _doExport(kFALSE), _openTrans(kFALSE)
{
}
RooWorkspace::RooWorkspace(const char* name, const char* title) :
TNamed(name,title?title:name), _classes(this), _dir(0), _factory(0), _doExport(kFALSE), _openTrans(kFALSE)
{
}
RooWorkspace::RooWorkspace(const char* name, Bool_t doCINTExport) :
TNamed(name,name), _classes(this), _dir(0), _factory(0), _doExport(kFALSE), _openTrans(kFALSE)
{
if (doCINTExport) {
exportToCint(name) ;
}
}
RooWorkspace::RooWorkspace(const RooWorkspace& other) :
TNamed(other), _uuid(other._uuid), _classes(this), _dir(0), _factory(0), _doExport(kFALSE), _openTrans(kFALSE)
{
other._allOwnedNodes.snapshot(_allOwnedNodes,kTRUE) ;
TIterator* iter = other._dataList.MakeIterator() ;
TObject* data2 ;
while((data2=iter->Next())) {
_dataList.Add(data2->Clone()) ;
}
delete iter ;
TIterator* iter2 = other._snapshots.MakeIterator() ;
RooArgSet* snap ;
while((snap=(RooArgSet*)iter2->Next())) {
RooArgSet* snapClone = (RooArgSet*) snap->snapshot() ;
snapClone->setName(snap->GetName()) ;
_snapshots.Add(snapClone) ;
}
delete iter2 ;
for (map<string,RooArgSet>::const_iterator iter3 = other._namedSets.begin() ; iter3 != other._namedSets.end() ; ++iter3) {
RooArgSet* tmp = (RooArgSet*) _allOwnedNodes.selectCommon(iter3->second) ;
_namedSets[iter3->first].add(*tmp) ;
delete tmp ;
}
TIterator* iter4 = other._genObjects.MakeIterator() ;
TObject* gobj ;
while((gobj=iter4->Next())) {
_genObjects.Add(gobj->Clone()) ;
}
delete iter4 ;
}
RooWorkspace::~RooWorkspace()
{
if (_doExport) {
unExport() ;
}
_dataList.Delete() ;
if (_dir) {
delete _dir ;
}
_snapshots.Delete() ;
_genObjects.Delete() ;
}
Bool_t RooWorkspace::import(const char* fileSpec, const RooCmdArg& arg1, const RooCmdArg& arg2, const RooCmdArg& arg3)
{
char buf[1024] ;
strlcpy(buf,fileSpec,1024) ;
char* filename = strtok(buf,":") ;
char* wsname = strtok(0,":") ;
char* objname = strtok(0,":") ;
if (!filename||!wsname||!objname) {
coutE(InputArguments) << "RooWorkspace(" << GetName() << ") ERROR in file specification, expecting for 'filename:wsname:objname'" << endl ;
return kTRUE ;
}
TFile* f = TFile::Open(filename) ;
if (f==0) {
coutE(InputArguments) << "RooWorkspace(" << GetName() << ") ERROR opening file " << filename << endl ;
return 0 ;
}
RooWorkspace* w = dynamic_cast<RooWorkspace*>(f->Get(wsname)) ;
if (w==0) {
coutE(InputArguments) << "RooWorkspace(" << GetName() << ") ERROR: No object named " << wsname << " in file " << filename
<< " or object is not a RooWorkspace" << endl ;
return 0 ;
}
RooAbsArg* warg = w->arg(objname) ;
if (warg) {
Bool_t ret = import(*warg,arg1,arg2,arg3) ;
delete f ;
return ret ;
}
RooAbsData* wdata = w->data(objname) ;
if (wdata) {
Bool_t ret = import(*wdata,arg1,arg2,arg3) ;
delete f ;
return ret ;
}
coutE(InputArguments) << "RooWorkspace(" << GetName() << ") ERROR: No RooAbsArg or RooAbsData object named " << objname
<< " in workspace " << wsname << " in file " << filename << endl ;
return kTRUE ;
}
Bool_t RooWorkspace::import(const RooArgSet& args, const RooCmdArg& arg1, const RooCmdArg& arg2, const RooCmdArg& arg3)
{
TIterator* iter = args.createIterator() ;
RooAbsArg* oneArg ;
Bool_t ret(kFALSE) ;
while((oneArg=(RooAbsArg*)iter->Next())) {
ret |= import(*oneArg,arg1,arg2,arg3) ;
}
return ret ;
}
Bool_t RooWorkspace::import(const RooAbsArg& inArg, const RooCmdArg& arg1, const RooCmdArg& arg2, const RooCmdArg& arg3)
{
RooLinkedList args ;
args.Add((TObject*)&arg1) ;
args.Add((TObject*)&arg2) ;
args.Add((TObject*)&arg3) ;
RooCmdConfig pc(Form("RooWorkspace::import(%s)",GetName())) ;
pc.defineString("conflictSuffix","RenameConflictNodes",0) ;
pc.defineString("allSuffix","RenameAllNodes",0) ;
pc.defineString("allVarsSuffix","RenameAllVariables",0) ;
pc.defineString("allVarsExcept","RenameAllVariables",1) ;
pc.defineString("varChangeIn","RenameVar",0,"",kTRUE) ;
pc.defineString("varChangeOut","RenameVar",1,"",kTRUE) ;
pc.defineString("factoryTag","FactoryTag",0) ;
pc.defineInt("useExistingNodes","RecycleConflictNodes",0,0) ;
pc.defineInt("silence","Silence",0,0) ;
pc.defineMutex("RenameConflictNodes","RenameAllNodes") ;
pc.defineMutex("RenameConflictNodes","RecycleConflictNodes") ;
pc.defineMutex("RenameAllNodes","RecycleConflictNodes") ;
pc.defineMutex("RenameVariable","RenameAllVariables") ;
pc.process(args) ;
if (!pc.ok(kTRUE)) {
return kTRUE ;
}
const char* suffixC = pc.getString("conflictSuffix") ;
const char* suffixA = pc.getString("allSuffix") ;
const char* suffixV = pc.getString("allVarsSuffix") ;
const char* exceptVars = pc.getString("allVarsExcept") ;
const char* varChangeIn = pc.getString("varChangeIn") ;
const char* varChangeOut = pc.getString("varChangeOut") ;
Int_t useExistingNodes = pc.getInt("useExistingNodes") ;
Int_t silence = pc.getInt("silence") ;
if (suffixC && strlen(suffixC)==0) suffixC = 0 ;
if (suffixA && strlen(suffixA)==0) suffixA = 0 ;
Bool_t conflictOnly = suffixA ? kFALSE : kTRUE ;
const char* suffix = suffixA ? suffixA : suffixC ;
map<string,string> varMap ;
if (strlen(varChangeIn)>0) {
char tmp[1024] ;
strlcpy(tmp,varChangeIn,1024) ;
list<string> tmpIn,tmpOut ;
char* ptr = strtok(tmp,",") ;
while (ptr) {
tmpIn.push_back(ptr) ;
ptr = strtok(0,",") ;
}
strlcpy(tmp,varChangeOut,1024) ;
ptr = strtok(tmp,",") ;
while (ptr) {
tmpOut.push_back(ptr) ;
ptr = strtok(0,",") ;
}
list<string>::iterator iin = tmpIn.begin() ;
list<string>::iterator iout = tmpOut.begin() ;
for (;iin!=tmpIn.end() ; ++iin,++iout) {
varMap[*iin]=*iout ;
}
}
std::set<string> exceptVarNames ;
char tmp[1024] ;
if (exceptVars && strlen(exceptVars)) {
strlcpy(tmp,exceptVars,1024) ;
char* ptr = strtok(tmp,",") ;
while(ptr) {
exceptVarNames.insert(ptr) ;
ptr = strtok(0,",") ;
}
}
if (suffixV != 0 && strlen(suffixV)>0) {
RooArgSet* vars = inArg.getVariables() ;
TIterator* iter = vars->createIterator() ;
RooAbsArg* v ;
while((v=(RooAbsArg*)iter->Next())) {
if (exceptVarNames.find(v->GetName())==exceptVarNames.end()) {
varMap[v->GetName()] = Form("%s_%s",v->GetName(),suffixV) ;
}
}
delete iter ;
delete vars ;
}
RooAbsArg* wsarg = _allOwnedNodes.find(inArg.GetName()) ;
const char* tagIn = inArg.getStringAttribute("factory_tag") ;
const char* tagWs = wsarg ? wsarg->getStringAttribute("factory_tag") : 0 ;
Bool_t factoryMatch = (tagIn && tagWs && !strcmp(tagIn,tagWs)) ;
if (factoryMatch) {
((RooAbsArg&)inArg).setAttribute("RooWorkspace::Recycle") ;
}
if (!suffix && wsarg && !useExistingNodes && !(inArg.isFundamental() && varMap[inArg.GetName()]!="")) {
if (!factoryMatch) {
if (wsarg!=&inArg) {
coutE(ObjectHandling) << "RooWorkSpace::import(" << GetName() << ") ERROR importing object named " << inArg.GetName()
<< ": another instance with same name already in the workspace and no conflict resolution protocol specified" << endl ;
return kTRUE ;
} else {
if (!silence) {
coutI(ObjectHandling) << "RooWorkSpace::import(" << GetName() << ") Object " << inArg.GetName() << " is already in workspace!" << endl ;
}
return kTRUE ;
}
} else {
coutI(ObjectHandling) << "RooWorkSpace::import(" << GetName() << ") Recycling existing object " << inArg.GetName() << " created with identical factory specification" << endl ;
}
}
RooArgSet conflictNodes ;
RooArgSet branchSet ;
inArg.branchNodeServerList(&branchSet) ;
TIterator* iter = branchSet.createIterator() ;
RooAbsArg* branch ;
while ((branch=(RooAbsArg*)iter->Next())) {
RooAbsArg* wsbranch = _allOwnedNodes.find(branch->GetName()) ;
if (wsbranch && wsbranch!=branch && !branch->getAttribute("RooWorkspace::Recycle") && !useExistingNodes) {
conflictNodes.add(*branch) ;
}
}
delete iter ;
if (conflictNodes.getSize()>0 && !suffix && !useExistingNodes) {
coutE(ObjectHandling) << "RooWorkSpace::import(" << GetName() << ") ERROR object named " << inArg.GetName() << ": component(s) "
<< conflictNodes << " already in the workspace and no conflict resolution protocol specified" << endl ;
return kTRUE ;
}
RooArgSet* cloneSet = (RooArgSet*) RooArgSet(inArg).snapshot(kTRUE) ;
RooAbsArg* cloneTop = cloneSet->find(inArg.GetName()) ;
if (!conflictOnly) {
conflictNodes.removeAll() ;
conflictNodes.add(branchSet) ;
}
TIterator* citer = conflictNodes.createIterator() ;
string topName2 = cloneTop->GetName() ;
RooAbsArg* cnode ;
while ((cnode=(RooAbsArg*)citer->Next())) {
RooAbsArg* cnode2 = cloneSet->find(cnode->GetName()) ;
string origName = cnode2->GetName() ;
cnode2->SetName(Form("%s_%s",cnode2->GetName(),suffix)) ;
cnode2->SetTitle(Form("%s (%s)",cnode2->GetTitle(),suffix)) ;
string tag = Form("ORIGNAME:%s",origName.c_str()) ;
cnode2->setAttribute(tag.c_str()) ;
if (cnode2==cloneTop) {
topName2 = cnode2->GetName() ;
}
if (!silence) {
coutI(ObjectHandling) << "RooWorkspace::import(" << GetName()
<< ") Resolving name conflict in workspace by changing name of imported node "
<< origName << " to " << cnode2->GetName() << endl ;
}
}
delete citer ;
if (strlen(varChangeIn)>0 || (suffixV && strlen(suffixV)>0)) {
TIterator* cliter = cloneSet->createIterator() ;
while ((cnode=(RooAbsArg*)cliter->Next())) {
if (varMap.find(cnode->GetName())!=varMap.end()) {
string origName = cnode->GetName() ;
cnode->SetName(varMap[cnode->GetName()].c_str()) ;
string tag = Form("ORIGNAME:%s",origName.c_str()) ;
cnode->setAttribute(tag.c_str()) ;
if (!silence) {
coutI(ObjectHandling) << "RooWorkspace::import(" << GetName() << ") Changing name of variable "
<< origName << " to " << cnode->GetName() << " on request" << endl ;
}
if (cnode==cloneTop) {
topName2 = cnode->GetName() ;
}
}
}
delete cliter ;
}
RooArgSet* cloneSet2 = (RooArgSet*) RooArgSet(*cloneTop).snapshot(kTRUE) ;
RooAbsArg* cloneTop2 = cloneSet2->find(topName2.c_str()) ;
RooArgSet conflictNodes2 ;
RooArgSet branchSet2 ;
inArg.branchNodeServerList(&branchSet) ;
TIterator* iter2 = branchSet2.createIterator() ;
RooAbsArg* branch2 ;
while ((branch2=(RooAbsArg*)iter2->Next())) {
if (_allOwnedNodes.find(branch2->GetName())) {
conflictNodes2.add(*branch2) ;
}
}
delete iter2 ;
if (conflictNodes2.getSize()) {
coutE(ObjectHandling) << "RooWorkSpace::import(" << GetName() << ") ERROR object named " << inArg.GetName() << ": component(s) "
<< conflictNodes2 << " cause naming conflict after conflict resolution protocol was executed" << endl ;
return kTRUE ;
}
iter = cloneSet2->createIterator() ;
RooAbsArg* node ;
while((node=(RooAbsArg*)iter->Next())) {
if (node->importWorkspaceHook(*this)) {
coutE(ObjectHandling) << "RooWorkSpace::import(" << GetName() << ") ERROR object named " << node->GetName()
<< " has an error in importing in one or more of its auxiliary objects, aborting" << endl ;
return kTRUE ;
}
}
iter->Reset() ;
RooArgSet recycledNodes ;
RooArgSet nodesToBeDeleted ;
while((node=(RooAbsArg*)iter->Next())) {
if (_autoClass) {
if (!_classes.autoImportClass(node->IsA())) {
coutW(ObjectHandling) << "RooWorkspace::import(" << GetName() << ") WARNING: problems import class code of object "
<< node->IsA()->GetName() << "::" << node->GetName() << ", reading of workspace will require external definition of class" << endl ;
}
}
RooExpensiveObjectCache& oldCache = node->expensiveObjectCache() ;
node->setExpensiveObjectCache(_eocache) ;
_eocache.importCacheObjects(oldCache,node->GetName(),kTRUE) ;
RooAbsArg* wsnode = _allOwnedNodes.find(node->GetName()) ;
if (wsnode) {
if (!silence && useExistingNodes) {
coutI(ObjectHandling) << "RooWorkspace::import(" << GetName() << ") using existing copy of " << node->IsA()->GetName()
<< "::" << node->GetName() << " for import of " << cloneTop2->IsA()->GetName() << "::"
<< cloneTop2->GetName() << endl ;
}
recycledNodes.add(*_allOwnedNodes.find(node->GetName())) ;
nodesToBeDeleted.addOwned(*node) ;
} else {
if (!silence) {
coutI(ObjectHandling) << "RooWorkspace::import(" << GetName() << ") importing " << node->IsA()->GetName() << "::"
<< node->GetName() << endl ;
}
_allOwnedNodes.addOwned(*node) ;
if (_openTrans) {
_sandboxNodes.add(*node) ;
} else {
if (_dir && node->IsA() != RooConstVar::Class()) {
_dir->InternalAppend(node) ;
}
if (_doExport && node->IsA() != RooConstVar::Class()) {
exportObj(node) ;
}
}
}
}
delete cloneSet ;
if (recycledNodes.getSize()>0) {
iter->Reset() ;
while((node=(RooAbsArg*)iter->Next())) {
node->redirectServers(recycledNodes) ;
}
}
delete iter ;
cloneSet2->releaseOwnership() ;
delete cloneSet2 ;
return kFALSE ;
}
Bool_t RooWorkspace::import(RooAbsData& inData, const RooCmdArg& arg1, const RooCmdArg& arg2, const RooCmdArg& arg3)
{
coutI(ObjectHandling) << "RooWorkspace::import(" << GetName() << ") importing dataset " << inData.GetName() << endl ;
RooLinkedList args ;
args.Add((TObject*)&arg1) ;
args.Add((TObject*)&arg2) ;
args.Add((TObject*)&arg3) ;
RooCmdConfig pc(Form("RooWorkspace::import(%s)",GetName())) ;
pc.defineString("dsetName","Rename",0,"") ;
pc.defineString("varChangeIn","RenameVar",0,"",kTRUE) ;
pc.defineString("varChangeOut","RenameVar",1,"",kTRUE) ;
pc.process(args) ;
if (!pc.ok(kTRUE)) {
return kTRUE ;
}
const char* dsetName = pc.getString("dsetName") ;
const char* varChangeIn = pc.getString("varChangeIn") ;
const char* varChangeOut = pc.getString("varChangeOut") ;
if (dsetName && strlen(dsetName)==0) {
dsetName=0 ;
}
if (dsetName && _dataList.FindObject(dsetName)) {
coutE(ObjectHandling) << "RooWorkspace::import(" << GetName() << ") ERROR dataset with name " << dsetName << " already exists in workspace, import aborted" << endl ;
return kTRUE ;
}
if (!dsetName && _dataList.FindObject(inData.GetName())) {
coutE(ObjectHandling) << "RooWorkspace::import(" << GetName() << ") ERROR dataset with name " << inData.GetName() << " already exists in workspace, import aborted" << endl ;
return kTRUE ;
}
RooAbsData* clone ;
if (dsetName) {
coutI(ObjectHandling) << "RooWorkSpace::import(" << GetName() << ") changing name of dataset from " << inData.GetName() << " to " << dsetName << endl ;
clone = (RooAbsData*) inData.Clone(dsetName) ;
} else {
clone = (RooAbsData*) inData.Clone(inData.GetName()) ;
}
if (strlen(varChangeIn)>0) {
char tmp[1024] ;
strlcpy(tmp,varChangeIn,1024) ;
list<string> tmpIn,tmpOut ;
char* ptr = strtok(tmp,",") ;
while (ptr) {
tmpIn.push_back(ptr) ;
ptr = strtok(0,",") ;
}
strlcpy(tmp,varChangeOut,1024) ;
ptr = strtok(tmp,",") ;
while (ptr) {
tmpOut.push_back(ptr) ;
ptr = strtok(0,",") ;
}
list<string>::iterator iin = tmpIn.begin() ;
list<string>::iterator iout = tmpOut.begin() ;
for (; iin!=tmpIn.end() ; ++iin,++iout) {
coutI(ObjectHandling) << "RooWorkSpace::import(" << GetName() << ") changing name of dataset observable " << *iin << " to " << *iout << endl ;
clone->changeObservableName(iin->c_str(),iout->c_str()) ;
}
}
TIterator* iter = clone->get()->createIterator() ;
RooAbsArg* carg ;
while((carg=(RooAbsArg*)iter->Next())) {
if (!arg(carg->GetName())) {
import(*carg) ;
}
}
delete iter ;
_dataList.Add(clone) ;
if (_dir) {
_dir->InternalAppend(clone) ;
}
if (_doExport) {
exportObj(clone) ;
}
return kFALSE ;
}
Bool_t RooWorkspace::defineSet(const char* name, const RooArgSet& aset, Bool_t importMissing)
{
map<string,RooArgSet>::iterator i = _namedSets.find(name) ;
if (i!=_namedSets.end()) {
coutW(InputArguments) << "RooWorkspace::defineSet(" << GetName() << ") WARNING redefining previously defined named set " << name << endl ;
}
RooArgSet wsargs ;
TIterator* iter = aset.createIterator() ;
RooAbsArg* sarg ;
while((sarg=(RooAbsArg*)iter->Next())) {
if (!arg(sarg->GetName())) {
if (importMissing) {
import(*sarg) ;
} else {
coutE(InputArguments) << "RooWorkspace::defineSet(" << GetName() << ") ERROR set constituent " << sarg->GetName()
<< " is not in workspace and importMissing option is disabled" << endl ;
return kTRUE ;
}
}
wsargs.add(*arg(sarg->GetName())) ;
}
delete iter ;
_namedSets[name].removeAll() ;
_namedSets[name].add(wsargs) ;
return kFALSE ;
}
Bool_t RooWorkspace::defineSet(const char* name, const char* contentList)
{
map<string,RooArgSet>::iterator i = _namedSets.find(name) ;
if (i!=_namedSets.end()) {
coutW(InputArguments) << "RooWorkspace::defineSet(" << GetName() << ") WARNING redefining previously defined named set " << name << endl ;
}
RooArgSet wsargs ;
char buf[1024] ;
strlcpy(buf,contentList,1024) ;
char* token = strtok(buf,",") ;
while(token) {
if (!arg(token)) {
coutE(InputArguments) << "RooWorkspace::defineSet(" << GetName() << ") ERROR proposed set constituent " << token
<< " is not in workspace" << endl ;
return kTRUE ;
}
wsargs.add(*arg(token)) ;
token = strtok(0,",") ;
}
_namedSets[name].removeAll() ;
_namedSets[name].add(wsargs) ;
return kFALSE ;
}
Bool_t RooWorkspace::extendSet(const char* name, const char* newContents)
{
RooArgSet wsargs ;
char buf[1024] ;
strlcpy(buf,newContents,1024) ;
char* token = strtok(buf,",") ;
while(token) {
if (!arg(token)) {
coutE(InputArguments) << "RooWorkspace::defineSet(" << GetName() << ") ERROR proposed set constituent " << token
<< " is not in workspace" << endl ;
return kTRUE ;
}
wsargs.add(*arg(token)) ;
token = strtok(0,",") ;
}
_namedSets[name].add(wsargs,kTRUE) ;
return kFALSE ;
}
const RooArgSet* RooWorkspace::set(const char* name)
{
map<string,RooArgSet>::iterator i = _namedSets.find(name) ;
return (i!=_namedSets.end()) ? &(i->second) : 0 ;
}
Bool_t RooWorkspace::renameSet(const char* name, const char* newName)
{
if (!set(name)) {
coutE(InputArguments) << "RooWorkspace::renameSet(" << GetName() << ") ERROR a set with name " << name
<< " does not exist" << endl ;
return kTRUE ;
}
if (set(newName)) {
coutE(InputArguments) << "RooWorkspace::renameSet(" << GetName() << ") ERROR a set with name " << newName
<< " already exists" << endl ;
return kTRUE ;
}
_namedSets[newName].add(_namedSets[name]) ;
_namedSets.erase(name) ;
return kFALSE ;
}
Bool_t RooWorkspace::removeSet(const char* name)
{
if (!set(name)) {
coutE(InputArguments) << "RooWorkspace::removeSet(" << GetName() << ") ERROR a set with name " << name
<< " does not exist" << endl ;
return kTRUE ;
}
_namedSets.erase(name) ;
return kFALSE ;
}
Bool_t RooWorkspace::startTransaction()
{
if (_openTrans) {
return kFALSE ;
}
_openTrans = kTRUE ;
return kTRUE ;
}
Bool_t RooWorkspace::cancelTransaction()
{
if (!_openTrans) {
return kFALSE ;
}
TIterator* iter = _sandboxNodes.createIterator() ;
RooAbsArg* tmpArg ;
while((tmpArg=(RooAbsArg*)iter->Next())) {
_allOwnedNodes.remove(*tmpArg) ;
}
delete iter ;
_sandboxNodes.removeAll() ;
_openTrans = kFALSE ;
return kTRUE ;
}
Bool_t RooWorkspace::commitTransaction()
{
if (!_openTrans) {
return kFALSE ;
}
TIterator* iter = _sandboxNodes.createIterator() ;
RooAbsArg* sarg ;
while((sarg=(RooAbsArg*)iter->Next())) {
if (_dir && sarg->IsA() != RooConstVar::Class()) {
_dir->InternalAppend(sarg) ;
}
if (_doExport && sarg->IsA() != RooConstVar::Class()) {
exportObj(sarg) ;
}
}
delete iter ;
_sandboxNodes.removeAll() ;
_openTrans = kFALSE ;
return kTRUE ;
}
Bool_t RooWorkspace::importClassCode(TClass* theClass, Bool_t doReplace)
{
return _classes.autoImportClass(theClass,doReplace) ;
}
Bool_t RooWorkspace::importClassCode(const char* pat, Bool_t doReplace)
{
Bool_t ret(kTRUE) ;
TRegexp re(pat,kTRUE) ;
TIterator* iter = componentIterator() ;
RooAbsArg* carg ;
while((carg=(RooAbsArg*)iter->Next())) {
TString className = carg->IsA()->GetName() ;
if (className.Index(re)>=0 && !_classes.autoImportClass(carg->IsA(),doReplace)) {
coutW(ObjectHandling) << "RooWorkspace::import(" << GetName() << ") WARNING: problems import class code of object "
<< carg->IsA()->GetName() << "::" << carg->GetName() << ", reading of workspace will require external definition of class" << endl ;
ret = kFALSE ;
}
}
delete iter ;
return ret ;
}
Bool_t RooWorkspace::saveSnapshot(const char* name, const char* paramNames)
{
return saveSnapshot(name,argSet(paramNames),kFALSE) ;
}
Bool_t RooWorkspace::saveSnapshot(const char* name, const RooArgSet& params, Bool_t importValues)
{
RooArgSet* actualParams = (RooArgSet*) _allOwnedNodes.selectCommon(params) ;
RooArgSet* snapshot = (RooArgSet*) actualParams->snapshot() ;
delete actualParams ;
snapshot->setName(name) ;
if (importValues) {
*snapshot = params ;
}
RooArgSet* oldSnap = (RooArgSet*) _snapshots.FindObject(name) ;
if (oldSnap) {
coutI(ObjectHandling) << "RooWorkspace::saveSnaphot(" << GetName() << ") replacing previous snapshot with name " << name << endl ;
_snapshots.Remove(oldSnap) ;
delete oldSnap ;
}
_snapshots.Add(snapshot) ;
return kTRUE ;
}
Bool_t RooWorkspace::loadSnapshot(const char* name)
{
RooArgSet* snap = (RooArgSet*) _snapshots.find(name) ;
if (!snap) {
coutE(ObjectHandling) << "RooWorkspace::loadSnapshot(" << GetName() << ") no snapshot with name " << name << " is available" << endl ;
return kFALSE ;
}
RooArgSet* actualParams = (RooArgSet*) _allOwnedNodes.selectCommon(*snap) ;
*actualParams = *snap ;
delete actualParams ;
return kTRUE ;
}
RooAbsPdf* RooWorkspace::pdf(const char* name) const
{
return dynamic_cast<RooAbsPdf*>(_allOwnedNodes.find(name)) ;
}
RooAbsReal* RooWorkspace::function(const char* name) const
{
return dynamic_cast<RooAbsReal*>(_allOwnedNodes.find(name)) ;
}
RooRealVar* RooWorkspace::var(const char* name) const
{
return dynamic_cast<RooRealVar*>(_allOwnedNodes.find(name)) ;
}
RooCategory* RooWorkspace::cat(const char* name) const
{
return dynamic_cast<RooCategory*>(_allOwnedNodes.find(name)) ;
}
RooAbsCategory* RooWorkspace::catfunc(const char* name) const
{
return dynamic_cast<RooAbsCategory*>(_allOwnedNodes.find(name)) ;
}
RooAbsArg* RooWorkspace::arg(const char* name) const
{
return _allOwnedNodes.find(name) ;
}
RooArgSet RooWorkspace::argSet(const char* nameList) const
{
RooArgSet ret ;
char tmp[1024] ;
strlcpy(tmp,nameList,1024) ;
char* token = strtok(tmp,",") ;
while(token) {
RooAbsArg* oneArg = arg(token) ;
if (oneArg) {
ret.add(*oneArg) ;
} else {
coutE(InputArguments) << " RooWorkspace::argSet(" << GetName() << ") no RooAbsArg named " << token << " in workspace" << endl ;
}
token = strtok(0,",") ;
}
return ret ;
}
RooAbsArg* RooWorkspace::fundArg(const char* name) const
{
RooAbsArg* tmp = arg(name) ;
if (!tmp) {
return 0 ;
}
return tmp->isFundamental() ? tmp : 0 ;
}
RooAbsData* RooWorkspace::data(const char* name) const
{
return (RooAbsData*)_dataList.FindObject(name) ;
}
RooArgSet RooWorkspace::allVars() const
{
RooArgSet ret ;
TIterator* iter = _allOwnedNodes.createIterator() ;
RooAbsArg* parg ;
while((parg=(RooAbsArg*)iter->Next())) {
if (parg->IsA()->InheritsFrom(RooRealVar::Class())) {
ret.add(*parg) ;
}
}
delete iter ;
return ret ;
}
RooArgSet RooWorkspace::allCats() const
{
RooArgSet ret ;
TIterator* iter = _allOwnedNodes.createIterator() ;
RooAbsArg* parg ;
while((parg=(RooAbsArg*)iter->Next())) {
if (parg->IsA()->InheritsFrom(RooCategory::Class())) {
ret.add(*parg) ;
}
}
delete iter ;
return ret ;
}
RooArgSet RooWorkspace::allFunctions() const
{
RooArgSet ret ;
TIterator* iter = _allOwnedNodes.createIterator() ;
RooAbsArg* parg ;
while((parg=(RooAbsArg*)iter->Next())) {
if (parg->IsA()->InheritsFrom(RooAbsReal::Class()) &&
!parg->IsA()->InheritsFrom(RooAbsPdf::Class()) &&
!parg->IsA()->InheritsFrom(RooConstVar::Class()) &&
!parg->IsA()->InheritsFrom(RooRealVar::Class())) {
ret.add(*parg) ;
}
}
return ret ;
}
RooArgSet RooWorkspace::allCatFunctions() const
{
RooArgSet ret ;
TIterator* iter = _allOwnedNodes.createIterator() ;
RooAbsArg* parg ;
while((parg=(RooAbsArg*)iter->Next())) {
if (parg->IsA()->InheritsFrom(RooAbsCategory::Class()) &&
!parg->IsA()->InheritsFrom(RooCategory::Class())) {
ret.add(*parg) ;
}
}
return ret ;
}
RooArgSet RooWorkspace::allResolutionModels() const
{
RooArgSet ret ;
TIterator* iter = _allOwnedNodes.createIterator() ;
RooAbsArg* parg ;
while((parg=(RooAbsArg*)iter->Next())) {
if (parg->IsA()->InheritsFrom(RooResolutionModel::Class())) {
if (!((RooResolutionModel*)parg)->isConvolved()) {
ret.add(*parg) ;
}
}
}
return ret ;
}
RooArgSet RooWorkspace::allPdfs() const
{
RooArgSet ret ;
TIterator* iter = _allOwnedNodes.createIterator() ;
RooAbsArg* parg ;
while((parg=(RooAbsArg*)iter->Next())) {
if (parg->IsA()->InheritsFrom(RooAbsPdf::Class()) &&
!parg->IsA()->InheritsFrom(RooResolutionModel::Class())) {
ret.add(*parg) ;
}
}
return ret ;
}
list<RooAbsData*> RooWorkspace::allData() const
{
list<RooAbsData*> ret ;
TIterator* iter = _dataList.MakeIterator() ;
RooAbsData* dat ;
while((dat=(RooAbsData*)iter->Next())) {
ret.push_back(dat) ;
}
delete iter ;
return ret ;
}
list<TObject*> RooWorkspace::allGenericObjects() const
{
list<TObject*> ret ;
TIterator* iter = _genObjects.MakeIterator() ;
TObject* gobj ;
while((gobj=(RooAbsData*)iter->Next())) {
if (gobj->IsA()==RooTObjWrap::Class()) {
ret.push_back(((RooTObjWrap*)gobj)->obj()) ;
} else {
ret.push_back(gobj) ;
}
}
delete iter ;
return ret ;
}
Bool_t RooWorkspace::CodeRepo::autoImportClass(TClass* tc, Bool_t doReplace)
{
oocxcoutD(_wspace,ObjectHandling) << "RooWorkspace::CodeRepo(" << _wspace->GetName() << ") request to import code of class " << tc->GetName() << endl ;
if (!doReplace && _c2fmap.find(tc->GetName())!=_c2fmap.end()) {
oocxcoutD(_wspace,ObjectHandling) << "RooWorkspace::CodeRepo(" << _wspace->GetName() << ") code of class " << tc->GetName() << " already imported, skipping" << endl ;
return kTRUE ;
}
string implfile = tc->GetImplFileName() ;
string declfile = tc->GetDeclFileName() ;
if (implfile.empty() || declfile.empty()) {
oocoutE(_wspace,ObjectHandling) << "RooWorkspace::CodeRepo(" << _wspace->GetName() << ") ERROR: cannot retrieve code file names for class "
<< tc->GetName() << " through ROOT TClass interface, unable to import code" << endl ;
return kFALSE ;
}
TString rootsys = gSystem->Getenv("ROOTSYS") ;
if (TString(implfile.c_str()).Index(rootsys)>=0) {
oocxcoutD(_wspace,ObjectHandling) << "RooWorkspace::CodeRepo(" << _wspace->GetName() << ") code of class " << tc->GetName() << " is in ROOT distribution, skipping " << endl ;
return kTRUE ;
}
const char* implpath=0 ;
#if ROOT_VERSION_CODE <= ROOT_VERSION(5,19,02)
if (!(tc->GetClassInfo()->Property()&G__BIT_ISABSTRACT) && !tc->HasDefaultConstructor()) {
#else
if (!(gCint->ClassInfo_Property(tc->GetClassInfo())&G__BIT_ISABSTRACT) && !tc->HasDefaultConstructor()) {
#endif
oocoutW(_wspace,ObjectHandling) << "RooWorkspace::autoImportClass(" << _wspace->GetName() << ") WARNING cannot import class "
<< tc->GetName() << " : it cannot be persisted because it doesn't have a default constructor. Please fix " << endl ;
return kFALSE ;
}
char* declpath = 0 ;
if (gSystem->AccessPathName(declfile.c_str())) {
list<string>::iterator diter = RooWorkspace::_classDeclDirList.begin() ;
while(diter!= RooWorkspace::_classDeclDirList.end()) {
declpath = gSystem->ConcatFileName(diter->c_str(),declfile.c_str()) ;
if (!gSystem->AccessPathName(declpath)) {
break ;
}
delete[] declpath ;
declpath=0 ;
++diter ;
}
if (!declpath) {
oocoutW(_wspace,ObjectHandling) << "RooWorkspace::autoImportClass(" << _wspace->GetName() << ") WARNING Cannot access code of class "
<< tc->GetName() << " because header file " << declfile << " is not found in current directory nor in $ROOTSYS" ;
if (_classDeclDirList.size()>0) {
ooccoutW(_wspace,ObjectHandling) << ", nor in the search path " ;
diter = RooWorkspace::_classDeclDirList.begin() ;
while(diter!= RooWorkspace::_classDeclDirList.end()) {
if (diter!=RooWorkspace::_classDeclDirList.begin()) {
ooccoutW(_wspace,ObjectHandling) << "," ;
}
ooccoutW(_wspace,ObjectHandling) << diter->c_str() ;
++diter ;
}
}
ooccoutW(_wspace,ObjectHandling) << ". To fix this problem add the required directory to the search "
<< "path using RooWorkspace::addClassDeclDir(const char* dir)" << endl ;
return kFALSE ;
}
}
if (gSystem->AccessPathName(implfile.c_str())) {
list<string>::iterator iiter = RooWorkspace::_classImplDirList.begin() ;
while(iiter!= RooWorkspace::_classImplDirList.end()) {
implpath = gSystem->ConcatFileName(iiter->c_str(),implfile.c_str()) ;
if (!gSystem->AccessPathName(implpath)) {
break ;
}
delete[] implpath ;
implpath=0 ;
++iiter ;
}
if (!implpath) {
oocoutW(_wspace,ObjectHandling) << "RooWorkspace::autoImportClass(" << _wspace->GetName() << ") WARNING Cannot access code of class "
<< tc->GetName() << " because implementation file " << implfile << " is not found in current directory nor in $ROOTSYS" ;
if (_classDeclDirList.size()>0) {
ooccoutW(_wspace,ObjectHandling) << ", nor in the search path " ;
iiter = RooWorkspace::_classImplDirList.begin() ;
while(iiter!= RooWorkspace::_classImplDirList.end()) {
if (iiter!=RooWorkspace::_classImplDirList.begin()) {
ooccoutW(_wspace,ObjectHandling) << "," ;
}
ooccoutW(_wspace,ObjectHandling) << iiter->c_str() ;
++iiter ;
}
}
ooccoutW(_wspace,ObjectHandling) << ". To fix this problem add the required directory to the search "
<< "path using RooWorkspace::addClassImplDir(const char* dir)" << endl ;
return kFALSE ;
}
}
char buf[1024] ;
string declfilename = declpath?gSystem->BaseName(declpath):gSystem->BaseName(declfile.c_str()) ;
int dotpos2 = strrchr(declfilename.c_str(),'.') - declfilename.c_str() ;
string declfilebase = declfilename.substr(0,dotpos2) ;
string declfileext = declfilename.substr(dotpos2+1) ;
if (_fmap.find(declfilebase) == _fmap.end()) {
fstream fdecl(declpath?declpath:declfile.c_str()) ;
if (!fdecl) {
oocoutE(_wspace,ObjectHandling) << "RooWorkspace::autoImportClass(" << _wspace->GetName()
<< ") ERROR opening declaration file " << declfile << endl ;
return kFALSE ;
}
oocoutI(_wspace,ObjectHandling) << "RooWorkspace::autoImportClass(" << _wspace->GetName()
<< ") importing code of class " << tc->GetName()
<< " from " << (implpath?implpath:implfile.c_str())
<< " and " << (declpath?declpath:declfile.c_str()) << endl ;
string decl ;
while(fdecl.getline(buf,1023)) {
decl += buf ;
decl += '\n' ;
}
fstream fimpl(implpath?implpath:implfile.c_str()) ;
if (!fimpl) {
oocoutE(_wspace,ObjectHandling) << "RooWorkspace::autoImportClass(" << _wspace->GetName()
<< ") ERROR opening implementation file " << implfile << endl ;
return kFALSE ;
}
string impl ;
while(fimpl.getline(buf,1023)) {
Bool_t foundSelfInclude=kFALSE ;
if (strstr(buf,"#include")) {
char tmp[1024] ;
strlcpy(tmp,buf,1024) ;
strtok(tmp," <\"") ;
char* incfile = strtok(0," <\"") ;
if (strstr(incfile,declfilename.c_str())) {
foundSelfInclude=kTRUE ;
}
}
if (foundSelfInclude) {
impl += Form("#include \"%s.%s\"\n",declfilebase.c_str(),declfileext.c_str()) ;
} else {
impl += buf ;
impl += '\n' ;
}
}
_fmap[declfilebase]._hfile = decl ;
_fmap[declfilebase]._cxxfile = impl ;
_fmap[declfilebase]._hext = declfileext ;
} else {
oocoutI(_wspace,ObjectHandling) << "RooWorkspace::autoImportClass(" << _wspace->GetName()
<< ") code of class " << tc->GetName()
<< " was already imported from " << (implpath?implpath:implfile.c_str())
<< " and " << (declpath?declpath:declfile.c_str()) << endl ;
}
TString baseNameList ;
TList* bl = tc->GetListOfBases() ;
TIterator* iter = bl->MakeIterator() ;
TBaseClass* base ;
list<TClass*> bases ;
while((base=(TBaseClass*)iter->Next())) {
if (baseNameList.Length()>0) {
baseNameList += "," ;
}
baseNameList += base->GetClassPointer()->GetName() ;
bases.push_back(base->GetClassPointer()) ;
}
_c2fmap[tc->GetName()]._baseName = baseNameList ;
_c2fmap[tc->GetName()]._fileBase = declfilebase ;
list<TClass*>::iterator biter = bases.begin() ;
while(biter!=bases.end()) {
autoImportClass(*biter,doReplace) ;
++biter ;
}
if (implpath) {
delete[] implpath ;
}
if (declpath) {
delete[] declpath ;
}
return kTRUE ;
}
Bool_t RooWorkspace::makeDir()
{
if (_dir) return kTRUE ;
TString title= Form("TDirectory representation of RooWorkspace %s",GetName()) ;
_dir = new WSDir(GetName(),title.Data(),this) ;
TIterator* iter = componentIterator() ;
RooAbsArg* darg ;
while((darg=(RooAbsArg*)iter->Next())) {
if (darg->IsA() != RooConstVar::Class()) {
_dir->InternalAppend(darg) ;
}
}
return kTRUE ;
}
Bool_t RooWorkspace::import(TObject& object, Bool_t replaceExisting)
{
TObject* oldObj = _genObjects.FindObject(object.GetName()) ;
if (oldObj && !replaceExisting) {
coutE(InputArguments) << "RooWorkspace::import(" << GetName() << ") generic object with name "
<< object.GetName() << " is already in workspace and replaceExisting flag is set to false" << endl ;
return kTRUE ;
}
TH1::AddDirectory(kFALSE) ;
if (oldObj) {
_genObjects.Replace(oldObj,object.Clone()) ;
delete oldObj ;
} else {
_genObjects.Add(object.Clone()) ;
}
TH1::AddDirectory(kTRUE) ;
return kFALSE ;
}
Bool_t RooWorkspace::import(TObject& object, const char* aliasName, Bool_t replaceExisting)
{
TObject* oldObj = _genObjects.FindObject(object.GetName()) ;
if (oldObj && !replaceExisting) {
coutE(InputArguments) << "RooWorkspace::import(" << GetName() << ") generic object with name "
<< object.GetName() << " is already in workspace and replaceExisting flag is set to false" << endl ;
return kTRUE ;
}
TH1::AddDirectory(kFALSE) ;
RooTObjWrap* wrapper = new RooTObjWrap(object.Clone()) ;
TH1::AddDirectory(kTRUE) ;
wrapper->setOwning(kTRUE) ;
wrapper->SetName(aliasName) ;
wrapper->SetTitle(aliasName) ;
if (oldObj) {
_genObjects.Replace(oldObj,wrapper) ;
delete oldObj ;
} else {
_genObjects.Add(wrapper) ;
}
return kFALSE ;
}
Bool_t RooWorkspace::addStudy(RooAbsStudy& study)
{
RooAbsStudy* clone = (RooAbsStudy*) study.Clone() ;
_studyMods.Add(clone) ;
return kFALSE ;
}
void RooWorkspace::clearStudies()
{
_studyMods.Delete() ;
}
TObject* RooWorkspace::obj(const char* name) const
{
TObject* ret = arg(name) ;
if (ret) return ret ;
ret = data(name) ;
if (ret) return ret ;
return genobj(name) ;
}
TObject* RooWorkspace::genobj(const char* name) const
{
TObject* gobj = _genObjects.FindObject(name) ;
if (!gobj) return 0 ;
if (gobj->IsA()==RooTObjWrap::Class()) return ((RooTObjWrap*)gobj)->obj() ;
return gobj ;
}
Bool_t RooWorkspace::cd(const char* path)
{
makeDir() ;
return _dir->cd(path) ;
}
Bool_t RooWorkspace::writeToFile(const char* fileName, Bool_t recreate)
{
TFile f(fileName,recreate?"RECREATE":"UPDATE") ;
Write() ;
return kFALSE ;
}
RooFactoryWSTool& RooWorkspace::factory()
{
if (_factory) {
return *_factory ;
}
cxcoutD(ObjectHandling) << "INFO: Creating RooFactoryWSTool associated with this workspace" << endl ;
_factory = new RooFactoryWSTool(*this) ;
return *_factory ;
}
RooAbsArg* RooWorkspace::factory(const char* expr)
{
return factory().process(expr) ;
}
void RooWorkspace::Print(Option_t* opts) const
{
Bool_t treeMode(kFALSE) ;
if (TString(opts).Contains("t")) {
treeMode=kTRUE ;
}
cout << endl << "RooWorkspace(" << GetName() << ") " << GetTitle() << " contents" << endl << endl ;
RooAbsArg* parg ;
RooArgSet pdfSet ;
RooArgSet funcSet ;
RooArgSet varSet ;
RooArgSet catfuncSet ;
RooArgSet convResoSet ;
RooArgSet resoSet ;
TIterator* iter = _allOwnedNodes.createIterator() ;
while((parg=(RooAbsArg*)iter->Next())) {
if (treeMode) {
if (parg->IsA()->InheritsFrom(RooAbsPdf::Class())) {
if (!parg->hasClients()) {
pdfSet.add(*parg) ;
}
}
if (parg->IsA()->InheritsFrom(RooAbsReal::Class()) &&
!parg->IsA()->InheritsFrom(RooAbsPdf::Class()) &&
!parg->IsA()->InheritsFrom(RooConstVar::Class()) &&
!parg->IsA()->InheritsFrom(RooRealVar::Class())) {
if (!parg->hasClients()) {
funcSet.add(*parg) ;
}
}
if (parg->IsA()->InheritsFrom(RooAbsCategory::Class()) &&
!parg->IsA()->InheritsFrom(RooCategory::Class())) {
if (!parg->hasClients()) {
catfuncSet.add(*parg) ;
}
}
} else {
if (parg->IsA()->InheritsFrom(RooResolutionModel::Class())) {
if (((RooResolutionModel*)parg)->isConvolved()) {
convResoSet.add(*parg) ;
} else {
resoSet.add(*parg) ;
}
}
if (parg->IsA()->InheritsFrom(RooAbsPdf::Class()) &&
!parg->IsA()->InheritsFrom(RooResolutionModel::Class())) {
pdfSet.add(*parg) ;
}
if (parg->IsA()->InheritsFrom(RooAbsReal::Class()) &&
!parg->IsA()->InheritsFrom(RooAbsPdf::Class()) &&
!parg->IsA()->InheritsFrom(RooConstVar::Class()) &&
!parg->IsA()->InheritsFrom(RooRealVar::Class())) {
funcSet.add(*parg) ;
}
if (parg->IsA()->InheritsFrom(RooAbsCategory::Class()) &&
!parg->IsA()->InheritsFrom(RooCategory::Class())) {
catfuncSet.add(*parg) ;
}
}
if (parg->IsA()->InheritsFrom(RooRealVar::Class())) {
varSet.add(*parg) ;
}
if (parg->IsA()->InheritsFrom(RooCategory::Class())) {
varSet.add(*parg) ;
}
}
delete iter ;
RooFit::MsgLevel oldLevel = RooMsgService::instance().globalKillBelow() ;
RooMsgService::instance().setGlobalKillBelow(RooFit::WARNING) ;
if (varSet.getSize()>0) {
varSet.sort() ;
cout << "variables" << endl ;
cout << "---------" << endl ;
cout << varSet << endl ;
cout << endl ;
}
if (pdfSet.getSize()>0) {
cout << "p.d.f.s" << endl ;
cout << "-------" << endl ;
pdfSet.sort() ;
iter = pdfSet.createIterator() ;
while((parg=(RooAbsArg*)iter->Next())) {
if (treeMode) {
parg->printComponentTree() ;
} else {
parg->Print() ;
}
}
delete iter ;
cout << endl ;
}
if (!treeMode) {
if (resoSet.getSize()>0) {
cout << "analytical resolution models" << endl ;
cout << "----------------------------" << endl ;
resoSet.sort() ;
iter = resoSet.createIterator() ;
while((parg=(RooAbsArg*)iter->Next())) {
parg->Print() ;
}
delete iter ;
cout << endl ;
}
}
if (funcSet.getSize()>0) {
cout << "functions" << endl ;
cout << "--------" << endl ;
funcSet.sort() ;
iter = funcSet.createIterator() ;
while((parg=(RooAbsArg*)iter->Next())) {
if (treeMode) {
parg->printComponentTree() ;
} else {
parg->Print() ;
}
}
delete iter ;
cout << endl ;
}
if (catfuncSet.getSize()>0) {
cout << "category functions" << endl ;
cout << "------------------" << endl ;
catfuncSet.sort() ;
iter = catfuncSet.createIterator() ;
while((parg=(RooAbsArg*)iter->Next())) {
if (treeMode) {
parg->printComponentTree() ;
} else {
parg->Print() ;
}
}
delete iter ;
cout << endl ;
}
if (_dataList.GetSize()>0) {
cout << "datasets" << endl ;
cout << "--------" << endl ;
iter = _dataList.MakeIterator() ;
RooAbsData* data2 ;
while((data2=(RooAbsData*)iter->Next())) {
cout << data2->IsA()->GetName() << "::" << data2->GetName() << *data2->get() << endl ;
}
delete iter ;
cout << endl ;
}
if (_snapshots.GetSize()>0) {
cout << "parameter snapshots" << endl ;
cout << "-------------------" << endl ;
iter = _snapshots.MakeIterator() ;
RooArgSet* snap ;
while((snap=(RooArgSet*)iter->Next())) {
cout << snap->GetName() << " = (" ;
TIterator* aiter = snap->createIterator() ;
RooAbsArg* a ;
Bool_t first(kTRUE) ;
while((a=(RooAbsArg*)aiter->Next())) {
if (first) { first=kFALSE ; } else { cout << "," ; }
cout << a->GetName() << "=" ;
a->printValue(cout) ;
if (a->isConstant()) {
cout << "[C]" ;
}
}
cout << ")" << endl ;
delete aiter ;
}
delete iter ;
cout << endl ;
}
if (_namedSets.size()>0) {
cout << "named sets" << endl ;
cout << "----------" << endl ;
for (map<string,RooArgSet>::const_iterator it = _namedSets.begin() ; it != _namedSets.end() ; it++) {
cout << it->first << ":" << it->second << endl ;
}
cout << endl ;
}
if (_genObjects.GetSize()>0) {
cout << "generic objects" << endl ;
cout << "---------------" << endl ;
iter = _genObjects.MakeIterator() ;
TObject* gobj ;
while((gobj=(TObject*)iter->Next())) {
if (gobj->IsA()==RooTObjWrap::Class()) {
cout << ((RooTObjWrap*)gobj)->obj()->IsA()->GetName() << "::" << gobj->GetName() << endl ;
} else {
cout << gobj->IsA()->GetName() << "::" << gobj->GetName() << endl ;
}
}
delete iter ;
cout << endl ;
}
if (_studyMods.GetSize()>0) {
cout << "study modules" << endl ;
cout << "-------------" << endl ;
iter = _studyMods.MakeIterator() ;
TObject* smobj ;
while((smobj=(TObject*)iter->Next())) {
cout << smobj->IsA()->GetName() << "::" << smobj->GetName() << endl ;
}
delete iter ;
cout << endl ;
}
if (_classes.listOfClassNames().size()>0) {
cout << "embedded class code" << endl ;
cout << "-------------------" << endl ;
cout << _classes.listOfClassNames() << endl ;
cout << endl ;
}
if (_eocache.size()>0) {
cout << "embedded precalculated expensive components" << endl ;
cout << "-------------------------------------------" << endl ;
_eocache.print() ;
}
RooMsgService::instance().setGlobalKillBelow(oldLevel) ;
return ;
}
void RooWorkspace::CodeRepo::Streamer(TBuffer &R__b)
{
typedef ::RooWorkspace::CodeRepo thisClass;
if (R__b.IsReading()) {
UInt_t R__s, R__c;
R__b.ReadVersion(&R__s, &R__c);
Int_t count(0) ;
R__b >> count ;
while(count--) {
TString name ;
name.Streamer(R__b) ;
_fmap[name]._hext.Streamer(R__b) ;
_fmap[name]._hfile.Streamer(R__b) ;
_fmap[name]._cxxfile.Streamer(R__b) ;
}
count=0 ;
R__b >> count ;
while(count--) {
TString name,bname,fbase ;
name.Streamer(R__b) ;
_c2fmap[name]._baseName.Streamer(R__b) ;
_c2fmap[name]._fileBase.Streamer(R__b) ;
}
R__b.CheckByteCount(R__s, R__c, thisClass::IsA());
_compiledOK = !compileClasses() ;
} else {
UInt_t R__c;
R__c = R__b.WriteVersion(thisClass::IsA(), kTRUE);
UInt_t count = _fmap.size() ;
R__b << count ;
map<TString,ClassFiles>::iterator iter = _fmap.begin() ;
while(iter!=_fmap.end()) {
TString key_copy(iter->first) ;
key_copy.Streamer(R__b) ;
iter->second._hext.Streamer(R__b) ;
iter->second._hfile.Streamer(R__b);
iter->second._cxxfile.Streamer(R__b);
++iter ;
}
count = _c2fmap.size() ;
R__b << count ;
map<TString,ClassRelInfo>::iterator iter2 = _c2fmap.begin() ;
while(iter2!=_c2fmap.end()) {
TString key_copy(iter2->first) ;
key_copy.Streamer(R__b) ;
iter2->second._baseName.Streamer(R__b) ;
iter2->second._fileBase.Streamer(R__b);
++iter2 ;
}
R__b.SetByteCount(R__c, kTRUE);
}
}
void RooWorkspace::Streamer(TBuffer &R__b)
{
if (R__b.IsReading()) {
R__b.ReadClassBuffer(RooWorkspace::Class(),this);
TIterator* iter = _allOwnedNodes.createIterator() ;
RooAbsArg* node ;
while((node=(RooAbsArg*)iter->Next())) {
node->setExpensiveObjectCache(_eocache) ;
if (node->IsA()->InheritsFrom(RooAbsOptTestStatistic::Class())) {
RooAbsOptTestStatistic* tmp = (RooAbsOptTestStatistic*) node ;
if (tmp->isSealed() && tmp->sealNotice() && strlen(tmp->sealNotice())>0) {
cout << "RooWorkspace::Streamer(" << GetName() << ") " << node->IsA()->GetName() << "::" << node->GetName() << " : " << tmp->sealNotice() << endl ;
}
}
}
delete iter ;
} else {
map<RooAbsArg*,list<RooAbsArg*> > extClients, extValueClients, extShapeClients ;
TIterator* iter = _allOwnedNodes.createIterator() ;
RooAbsArg* tmparg ;
while((tmparg=(RooAbsArg*)iter->Next())) {
TIterator* clientIter = tmparg->_clientList.MakeIterator() ;
RooAbsArg* client ;
while((client=(RooAbsArg*)clientIter->Next())) {
if (!_allOwnedNodes.containsInstance(*client)) {
while(tmparg->_clientList.refCount(client)>0) {
tmparg->_clientList.Remove(client) ;
extClients[tmparg].push_back(client) ;
}
}
}
delete clientIter ;
TIterator* vclientIter = tmparg->_clientListValue.MakeIterator() ;
RooAbsArg* vclient ;
while((vclient=(RooAbsArg*)vclientIter->Next())) {
if (!_allOwnedNodes.containsInstance(*vclient)) {
cxcoutD(ObjectHandling) << "RooWorkspace::Streamer(" << GetName() << ") element " << tmparg->GetName()
<< " has external value client link to " << vclient << " (" << vclient->GetName() << ") with ref count " << tmparg->_clientListValue.refCount(vclient) << endl ;
while(tmparg->_clientListValue.refCount(vclient)>0) {
tmparg->_clientListValue.Remove(vclient) ;
extValueClients[tmparg].push_back(vclient) ;
}
}
}
delete vclientIter ;
TIterator* sclientIter = tmparg->_clientListShape.MakeIterator() ;
RooAbsArg* sclient ;
while((sclient=(RooAbsArg*)sclientIter->Next())) {
if (!_allOwnedNodes.containsInstance(*sclient)) {
cxcoutD(ObjectHandling) << "RooWorkspace::Streamer(" << GetName() << ") element " << tmparg->GetName()
<< " has external shape client link to " << sclient << " (" << sclient->GetName() << ") with ref count " << tmparg->_clientListShape.refCount(sclient) << endl ;
while(tmparg->_clientListShape.refCount(sclient)>0) {
tmparg->_clientListShape.Remove(sclient) ;
extShapeClients[tmparg].push_back(sclient) ;
}
}
}
delete sclientIter ;
}
delete iter ;
R__b.WriteClassBuffer(RooWorkspace::Class(),this);
for (map<RooAbsArg*,list<RooAbsArg*> >::iterator iterx = extClients.begin() ; iterx!=extClients.end() ; iterx++) {
for (list<RooAbsArg*>::iterator citer = iterx->second.begin() ; citer!=iterx->second.end() ; citer++) {
iterx->first->_clientList.Add(*citer) ;
}
}
for (map<RooAbsArg*,list<RooAbsArg*> >::iterator iterx = extValueClients.begin() ; iterx!=extValueClients.end() ; iterx++) {
for (list<RooAbsArg*>::iterator citer = iterx->second.begin() ; citer!=iterx->second.end() ; citer++) {
iterx->first->_clientListValue.Add(*citer) ;
}
}
for (map<RooAbsArg*,list<RooAbsArg*> >::iterator iterx = extShapeClients.begin() ; iterx!=extShapeClients.end() ; iterx++) {
for (list<RooAbsArg*>::iterator citer = iterx->second.begin() ; citer!=iterx->second.end() ; citer++) {
iterx->first->_clientListShape.Add(*citer) ;
}
}
}
}
std::string RooWorkspace::CodeRepo::listOfClassNames() const
{
string ret ;
map<TString,ClassRelInfo>::const_iterator iter = _c2fmap.begin() ;
while(iter!=_c2fmap.end()) {
if (ret.size()>0) {
ret += ", " ;
}
ret += iter->first ;
++iter ;
}
return ret ;
}
Bool_t RooWorkspace::CodeRepo::compileClasses()
{
Bool_t haveDir=kFALSE ;
string dirName = Form(_classFileExportDir.c_str(),_wspace->uuid().AsString(),_wspace->GetName()) ;
map<TString,ClassRelInfo>::iterator iter = _c2fmap.begin() ;
while(iter!=_c2fmap.end()) {
oocxcoutD(_wspace,ObjectHandling) << "RooWorkspace::CodeRepo::compileClasses() now processing class " << iter->first.Data() << endl ;
if (gClassTable->GetDict(iter->first.Data())) {
oocoutI(_wspace,ObjectHandling) << "RooWorkspace::CodeRepo::compileClasses() Embedded class "
<< iter->first << " already in ROOT class table, skipping" << endl ;
++iter ;
continue ;
}
if (!haveDir) {
if (!gSystem->AccessPathName(dirName.c_str())) {
oocoutI(_wspace,ObjectHandling) << "RooWorkspace::CodeRepo::compileClasses() reusing code export directory " << dirName.c_str()
<< " to extract coded embedded in workspace" << endl ;
} else {
if (gSystem->MakeDirectory(dirName.c_str())==0) {
oocoutI(_wspace,ObjectHandling) << "RooWorkspace::CodeRepo::compileClasses() creating code export directory " << dirName.c_str()
<< " to extract coded embedded in workspace" << endl ;
} else {
oocoutE(_wspace,ObjectHandling) << "RooWorkspace::CodeRepo::compileClasses() ERROR creating code export directory " << dirName.c_str()
<< " to extract coded embedded in workspace" << endl ;
return kFALSE ;
}
}
haveDir=kTRUE ;
}
ClassFiles& cfinfo = _fmap[iter->second._fileBase] ;
oocxcoutD(_wspace,ObjectHandling) << "RooWorkspace::CodeRepo::compileClasses() now processing file with base " << iter->second._fileBase << endl ;
if (cfinfo._extracted) {
oocxcoutD(_wspace,ObjectHandling) << "RooWorkspace::CodeRepo::compileClasses() file with base name " << iter->second._fileBase
<< " has already been extracted, skipping to next class" << endl ;
continue ;
}
Bool_t needDeclWrite=kTRUE ;
string fdname = Form("%s/%s.%s",dirName.c_str(),iter->second._fileBase.Data(),cfinfo._hext.Data()) ;
ifstream ifdecl(fdname.c_str()) ;
if (ifdecl) {
TString contents ;
char buf[1024] ;
while(ifdecl.getline(buf,1024)) {
contents += buf ;
contents += "\n" ;
}
UInt_t crcFile = RooAbsArg::crc32(contents.Data()) ;
UInt_t crcWS = RooAbsArg::crc32(cfinfo._hfile.Data()) ;
needDeclWrite = (crcFile!=crcWS) ;
}
if (needDeclWrite) {
oocoutI(_wspace,ObjectHandling) << "RooWorkspace::CodeRepo::compileClasses() Extracting declaration code of class " << iter->first << ", file " << fdname << endl ;
ofstream fdecl(fdname.c_str()) ;
if (!fdecl) {
oocoutE(_wspace,ObjectHandling) << "RooWorkspace::CodeRepo::compileClasses() ERROR opening file"
<< fdname << " for writing" << endl ;
return kFALSE ;
}
fdecl << cfinfo._hfile ;
fdecl.close() ;
}
Bool_t needImplWrite=kTRUE ;
string finame = Form("%s/%s.cxx",dirName.c_str(),iter->second._fileBase.Data()) ;
ifstream ifimpl(finame.c_str()) ;
if (ifimpl) {
TString contents ;
char buf[1024] ;
while(ifimpl.getline(buf,1024)) {
contents += buf ;
contents += "\n" ;
}
UInt_t crcFile = RooAbsArg::crc32(contents.Data()) ;
UInt_t crcWS = RooAbsArg::crc32(cfinfo._cxxfile.Data()) ;
needImplWrite = (crcFile!=crcWS) ;
}
if (needImplWrite) {
oocoutI(_wspace,ObjectHandling) << "RooWorkspace::CodeRepo::compileClasses() Extracting implementation code of class " << iter->first << ", file " << finame << endl ;
ofstream fimpl(finame.c_str()) ;
if (!fimpl) {
oocoutE(_wspace,ObjectHandling) << "RooWorkspace::CodeRepo::compileClasses() ERROR opening file"
<< finame << " for writing" << endl ;
return kFALSE ;
}
fimpl << cfinfo._cxxfile ;
fimpl.close() ;
}
cfinfo._extracted = kTRUE ;
oocxcoutD(_wspace,ObjectHandling) << "RooWorkspace::CodeRepo::compileClasses() marking code unit " << iter->second._fileBase << " as extracted" << endl ;
oocoutI(_wspace,ObjectHandling) << "RooWorkspace::CodeRepo::compileClasses() Compiling code unit " << iter->second._fileBase.Data() << " to define class " << iter->first << endl ;
Bool_t ok = gSystem->CompileMacro(finame.c_str(),"k") ;
if (!ok) {
oocoutE(_wspace,ObjectHandling) << "RooWorkspace::CodeRepo::compileClasses() ERROR compiling class " << iter->first.Data() << ", to fix this you can do the following: " << endl
<< " 1) Fix extracted source code files in directory " << dirName.c_str() << "/" << endl
<< " 2) In clean ROOT session compiled fixed classes by hand using '.x " << dirName.c_str() << "/ClassName.cxx+'" << endl
<< " 3) Reopen file with RooWorkspace with broken source code in UPDATE mode. Access RooWorkspace to force loading of class" << endl
<< " Broken instances in workspace will _not_ be compiled, instead precompiled fixed instances will be used." << endl
<< " 4) Reimport fixed code in workspace using 'RooWorkspace::importClassCode(\"*\",kTRUE)' method, Write() updated workspace to file and close file" << endl
<< " 5) Reopen file in clean ROOT session to confirm that problems are fixed" << endl ;
return kFALSE ;
}
++iter ;
}
return kTRUE ;
}
void RooWorkspace::WSDir::InternalAppend(TObject* obj)
{
#if ROOT_VERSION_CODE <= ROOT_VERSION(5,19,02)
TDirectory::Append(obj) ;
#else
TDirectory::Append(obj,kFALSE) ;
#endif
}
#if ROOT_VERSION_CODE <= ROOT_VERSION(5,19,02)
void RooWorkspace::WSDir::Add(TObject* obj)
#else
void RooWorkspace::WSDir::Add(TObject* obj,Bool_t)
#endif
{
if (dynamic_cast<RooAbsArg*>(obj) || dynamic_cast<RooAbsData*>(obj)) {
coutE(ObjectHandling) << "RooWorkspace::WSDir::Add(" << GetName() << ") ERROR: Directory is read-only representation of a RooWorkspace, use RooWorkspace::import() to add objects" << endl ;
} else {
InternalAppend(obj) ;
}
}
#if ROOT_VERSION_CODE <= ROOT_VERSION(5,19,02)
void RooWorkspace::WSDir::Append(TObject* obj)
#else
void RooWorkspace::WSDir::Append(TObject* obj,Bool_t)
#endif
{
if (dynamic_cast<RooAbsArg*>(obj) || dynamic_cast<RooAbsData*>(obj)) {
coutE(ObjectHandling) << "RooWorkspace::WSDir::Add(" << GetName() << ") ERROR: Directory is read-only representation of a RooWorkspace, use RooWorkspace::import() to add objects" << endl ;
} else {
InternalAppend(obj) ;
}
}
void RooWorkspace::exportToCint(const char* nsname)
{
if (_doExport) {
coutE(ObjectHandling) << "RooWorkspace::exportToCint(" << GetName() << ") WARNING: repeated calls to exportToCint() have no effect" << endl ;
return ;
}
_doExport = kTRUE ;
if (!nsname) nsname = GetName() ;
_exportNSName = nsname ;
coutI(ObjectHandling) << "RooWorkspace::exportToCint(" << GetName()
<< ") INFO: references to all objects in this workspace will be created in CINT in 'namespace " << _exportNSName << "'" << endl ;
TIterator* iter = _allOwnedNodes.createIterator() ;
TObject* wobj ;
while((wobj=iter->Next())) {
exportObj(wobj) ;
}
delete iter ;
iter = _dataList.MakeIterator() ;
while((wobj=iter->Next())) {
exportObj(wobj) ;
}
delete iter ;
}
void RooWorkspace::exportObj(TObject* wobj)
{
if (!_doExport) return ;
if (wobj->IsA() == RooConstVar::Class()) {
return ;
}
if (!isValidCPPID(wobj->GetName())) {
cxcoutD(ObjectHandling) << "RooWorkspace::exportObj(" << GetName() << ") INFO: Workspace object name " << wobj->GetName() << " is not a valid C++ identifier and is not exported to CINT" << endl ;
return ;
}
string cintExpr = Form("namespace %s { %s& %s = *(%s *)%p ; }",_exportNSName.c_str(),wobj->IsA()->GetName(),wobj->GetName(),wobj->IsA()->GetName(),wobj) ;
gROOT->ProcessLine(cintExpr.c_str()) ;
}
Bool_t RooWorkspace::isValidCPPID(const char* name)
{
string oname(name) ;
if (isdigit(oname[0])) {
return kFALSE ;
} else {
for (UInt_t i=0 ; i<oname.size() ; i++) {
char c = oname[i] ;
if (!isalnum(c) && (c!='_')) {
return kFALSE ;
}
}
}
return kTRUE ;
}
void RooWorkspace::unExport()
{
char buf[1024] ;
TIterator* iter = _allOwnedNodes.createIterator() ;
TObject* wobj ;
while((wobj=iter->Next())) {
if (isValidCPPID(wobj->GetName())) {
strlcpy(buf,Form("%s::%s",_exportNSName.c_str(),wobj->GetName()),1024) ;
G__deletevariable(buf) ;
}
}
delete iter ;
}