14 #include <TBranchObject.h>
16 #include <TLeafObject.h>
22 #if ROOT_VERSION_CODE >= ROOT_VERSION(3,01,5)
23 #include <TBranchElement.h>
26 #include <boost/algorithm/string.hpp>
27 #include <boost/foreach.hpp>
56 isFunctionalFlag =
setFile(f,
"titled by PHOOL", a) ? 1 : 0;
67 isFunctionalFlag =
setFile(f, title , a) ? 1 : 0;
81 temp << TreeName << treeindex;
82 TreeName = temp.str();
84 isFunctionalFlag =
setFile(f,
"titled by PHOOL" , a) ? 1 : 0;
127 string currdir = gDirectory->GetPath();
132 file = TFile::Open(
filename.c_str(),
"RECREATE", title.c_str());
137 file ->SetCompressionLevel(CompressionLevel);
138 tree =
new TTree(TreeName.c_str(), title.c_str());
139 tree->SetMaxTreeSize(900000000000LL);
140 gROOT->cd(currdir.c_str());
144 file = TFile::Open(
filename.c_str());
151 gROOT->cd(currdir.c_str());
155 file = TFile::Open(
filename.c_str(),
"UPDATE", title.c_str());
160 file ->SetCompressionLevel(CompressionLevel);
161 tree =
new TTree(TreeName.c_str(), title.c_str());
162 gROOT->cd(currdir.c_str());
177 topNode->
write(
this);
186 if (realTimeSave) tree->AutoSave(
"SaveSelf");
199 TBranch *thisBranch = tree->GetBranch(path.c_str());
211 if ((*data)->InheritsFrom(
"PHObject"))
217 tree->Branch(path.c_str(), (*data)->ClassName(),
218 data, bufSize, split);
222 thisBranch->SetAddress(data);
234 if (readEventFromFile(requestedEvent))
252 topNode = reconstructNodeTree(topNode);
256 if (tree && readEventFromFile(requestedEvent))
274 cout <<
"PHNodeIOManager reading " <<
filename << endl;
278 cout <<
"PHNodeIOManager writing " <<
filename << endl;
285 cout <<
"\n\nList of selected objects to read:" << endl;
286 map<string, PHBoolean>::const_iterator classiter;
287 for (classiter = objectToRead.begin(); classiter != objectToRead.end(); ++classiter)
289 cout << classiter->first <<
" is set to " << classiter->second << endl;
294 PHNodeIOManager::getBranchClassName(TBranch* branch)
305 #if ROOT_VERSION_CODE >= ROOT_VERSION(3,01,5)
306 TBranchElement* be =
dynamic_cast<TBranchElement*
>(branch);
311 return be->GetClassName();
315 TBranchObject* bo =
dynamic_cast<TBranchObject*
>(branch);
320 TLeafObject* leaf =
static_cast<TLeafObject*
>
321 (branch->GetLeaf(branch->GetName()));
323 return leaf->GetTypeName();
325 cout <<
PHWHERE <<
"Fatal error, dynamic cast of TBranchObject failed" << endl;
330 PHNodeIOManager::readEventFromFile(
size_t requestedEvent)
337 "Tree not initialized.");
347 string currdir = gDirectory->GetPath();
348 TFile* file_ptr = gFile;
353 if ((bytesRead = tree->GetEvent(requestedEvent)))
364 gROOT->cd(currdir.c_str());
372 cout <<
PHWHERE <<
"Error: Input TTree corrupt, exiting now" << endl;
384 string name = objectName;
385 map<string, TBranch*>::const_iterator p = fBranches.find(name);
387 if (p != fBranches.end())
389 TBranch* branch = p->second;
392 return branch->GetEvent(requestedEvent);
398 "Unknown object name");
410 cout <<
PHWHERE <<
"filename was never set" << endl;
419 tree =
static_cast<TTree*
>(file->Get(TreeName.c_str()));
423 cout <<
PHWHERE <<
"PHNodeIOManager::reconstructNodeTree : Root Tree "
424 << TreeName <<
" not found in file " << file->GetName() << endl;
432 nname << TreeName << file;
435 tree->SetName(nname.str().c_str());
438 map<string, PHBoolean>::const_iterator it;
440 if (tree->GetNbranches() > 0)
442 for (it = objectToRead.begin(); it != objectToRead.end(); ++it)
444 tree->SetBranchStatus((it->first).c_str(),
445 static_cast<bool>(it->second));
450 TObjArray *branchArray = tree->GetListOfBranches();
466 for (i = 0; i < (size_t)(branchArray->GetEntriesFast()); i++)
468 string branchname = (*branchArray)[i]->GetName();
469 vector<string> splitvec;
470 boost::split(splitvec, branchname, boost::is_any_of(delimeters));
471 for (
size_t ia = 1; ia< splitvec.size()-1; ia++)
473 if (!nodeIter.cd(splitvec[ia]))
476 nodeIter.cd(splitvec[ia]);
479 TBranch *thisBranch = (TBranch*)((*branchArray)[i]);
482 if (thisBranch->TestBit(kDoNotProcess))
487 string branchClassName = getBranchClassName(thisBranch);
488 string branchName = thisBranch->GetName();
489 fBranches[branchName] = thisBranch;
492 TClass* thisClass = gROOT->GetClass(branchClassName.c_str());
497 cout <<
"Missing Class: " << branchClassName.c_str() << endl;
498 cout <<
"Did you forget to load the shared library which contains "
499 << branchClassName.c_str() <<
"?" << endl;
503 assert(thisClass != 0);
506 static_cast<PHIODataNode<TObject> *
> (nodeIter.findFirst(
"PHIODataNode", (*splitvec.rbegin()).c_str()));
509 TObject *newTObject =
static_cast<TObject*
>(thisClass->New());
511 nodeIter.addNode(newIODataNode);
515 TObject *oldobject = newIODataNode->
getData();
516 string oldclass = oldobject->ClassName();
517 if (oldclass != branchClassName)
519 cout <<
"You only have to worry if you get this message when reading parallel files"
521 <<
"if you get this when opening the 2nd, 3rd,... file" << endl
522 <<
"It looks like your objects are not of the same version in these files" << endl;
523 cout <<
PHWHERE <<
"Found object " << oldobject->ClassName()
524 <<
" in node tree but the file "
525 <<
filename <<
" contains a " << branchClassName
526 <<
" object. The object will be replaced without harming you" << endl;
527 cout <<
"CAVEAT: If you use local copies of pointers to data nodes" << endl
528 <<
"instead of searching the node tree you are in trouble now" << endl;
529 delete newIODataNode;
530 TObject *newTObject =
static_cast<TObject*
>(thisClass->New());
532 nodeIter.addNode(newIODataNode);
537 if (thisClass->InheritsFrom(
"PHObject"))
543 cout <<
PHWHERE << branchClassName.c_str()
544 <<
" inherits neither from PHTable nor from PHObject"
545 <<
" setting type to PHObject" << endl;
548 thisBranch->SetAddress(&(newIODataNode->
data));
549 for (j = 1; j < splitvec.size() - 1; j++)
561 objectToRead[objectName] = readit;
566 map<string, PHBoolean>::const_iterator it;
568 for (it = objectToRead.begin(); it != objectToRead.end(); ++it)
570 tree->SetBranchStatus((it->first).c_str(),
571 static_cast<bool>(it->second));
581 string name = objectName;
582 map<string, TBranch*>::const_iterator p = fBranches.find(name);
584 if (p != fBranches.end())
599 CompressionLevel = level;
602 file ->SetCompressionLevel(CompressionLevel);
611 if (file)
return file->GetBytesWritten();
615 map<string, TBranch*> *
623 PHNodeIOManager::FillBranchMap()
625 if (fBranches.empty())
627 TTree *treetmp =
static_cast<TTree *
>(file->Get(TreeName.c_str()));
630 TObjArray *branchArray = treetmp->GetListOfBranches();
631 for (
size_t i = 0; i < (size_t)(branchArray->GetEntriesFast()); i++)
633 TBranch *thisBranch = (TBranch*)((*branchArray)[i]);
634 string branchName = (*branchArray)[i]->GetName();
635 fBranches[branchName] = thisBranch;
640 cout <<
PHWHERE <<
" No Root Tree " << TreeName
void PHMessage(const std::string &functionName, int messageType, const std::string &message)
virtual bool write(PHIOManager *, const std::string &="")
virtual void print() const
PHBoolean SetCompressionLevel(const int level)
virtual PHBoolean write(PHCompositeNode *)
int readSpecific(size_t requestedEvent, const char *objectName)
PHBoolean setFile(const std::string &, const std::string &, const PHAccessType=PHReadOnly)
virtual ~PHNodeIOManager()
void selectObjectToRead(const char *objectName, PHBoolean readit)
PHCompositeNode * read(PHCompositeNode *=0, size_t=0)
PHBoolean isSelected(const char *objectName)
std::map< std::string, TBranch * > * GetBranchMap()
void setObjectType(const std::string &type)
void SplitLevel(const int i)
void BufferSize(const int i)
static const std::string branchpathdelim