Class Reference for E1039 Core & Analysis Software
PHNodeIOManager.cc
Go to the documentation of this file.
1 // Implementation of class PHNodeIOManager
2 // Author: Matthias Messer
3 
4 #include "PHNodeIOManager.h"
5 #include "PHCompositeNode.h"
6 #include "PHNodeIterator.h"
7 #include "PHIODataNode.h"
8 #include "PHObject.h"
9 #include "phool.h"
10 #include "phooldefs.h"
11 
12 #include <TFile.h>
13 #include <TTree.h>
14 #include <TBranchObject.h>
15 #include <TObject.h>
16 #include <TLeafObject.h>
17 #include <TClass.h>
18 #include <TROOT.h>
19 #include <RVersion.h>
20 
21 // ROOT version taken from RVersion.h
22 #if ROOT_VERSION_CODE >= ROOT_VERSION(3,01,5)
23 #include <TBranchElement.h>
24 #endif
25 
26 #include <boost/algorithm/string.hpp>
27 #include <boost/foreach.hpp>
28 
29 #include <cassert>
30 #include <cstdlib>
31 #include <sstream>
32 #include <string>
33 
34 using namespace std;
35 
37  file(NULL),
38  tree(NULL),
39  TreeName("T"),
40  bufSize(0),
41  split(0),
42  accessMode(PHReadOnly),
43  CompressionLevel(3),
44  realTimeSave(false),
45  isFunctionalFlag(0)
46 {}
47 
49  const PHAccessType a):
50  file(NULL),
51  tree(NULL),
52  TreeName("T"),
53  CompressionLevel(3),
54  realTimeSave(false)
55 {
56  isFunctionalFlag = setFile(f, "titled by PHOOL", a) ? 1 : 0;
57 }
58 
59 PHNodeIOManager::PHNodeIOManager (const string& f, const string& title,
60  const PHAccessType a):
61  file(NULL),
62  tree(NULL),
63  TreeName("T"),
64  CompressionLevel(3),
65  realTimeSave(false)
66 {
67  isFunctionalFlag = setFile(f, title , a) ? 1 : 0;
68 }
69 
71  const PHTreeType treeindex):
72  file(NULL),
73  tree(NULL),
74  TreeName("T"),
75  CompressionLevel(3),
76  realTimeSave(false)
77 {
78  if (treeindex != PHEventTree)
79  {
80  ostringstream temp;
81  temp << TreeName << treeindex; // create e.g. T1
82  TreeName = temp.str();
83  }
84  isFunctionalFlag = setFile(f, "titled by PHOOL" , a) ? 1 : 0;
85 }
86 
88 {
89  closeFile ();
90  // if (tree)
91  // {
92  // tree->Delete();
93  // }
94  delete file;
95 }
96 
97 void
99 {
100  if (file)
101  {
102  if (accessMode == PHWrite || accessMode == PHUpdate)
103  {
104  file->Write();
105  }
106  file->Close();
107  }
108 }
109 
110 PHBoolean
111 PHNodeIOManager::setFile (const string& f, const string& title,
112  const PHAccessType a)
113 {
114  filename = f;
115  bufSize = 32000;
116  split = 0;
117  accessMode = a;
118  if (file)
119  {
120  if (file->IsOpen())
121  {
122  closeFile();
123  }
124  delete file;
125  file = 0;
126  }
127  string currdir = gDirectory->GetPath();
128  gROOT->cd();
129  switch (accessMode)
130  {
131  case PHWrite:
132  file = TFile::Open(filename.c_str(), "RECREATE", title.c_str());
133  if (!file)
134  {
135  return False;
136  }
137  file ->SetCompressionLevel(CompressionLevel);
138  tree = new TTree(TreeName.c_str(), title.c_str());
139  tree->SetMaxTreeSize(900000000000LL); // set max size to ~900 GB
140  gROOT->cd(currdir.c_str());
141  return True;
142  break;
143  case PHReadOnly:
144  file = TFile::Open(filename.c_str());
145  tree = 0;
146  if (!file)
147  {
148  return False;
149  }
150  selectObjectToRead("*", true);
151  gROOT->cd(currdir.c_str());
152  return True;
153  break;
154  case PHUpdate:
155  file = TFile::Open(filename.c_str(), "UPDATE", title.c_str());
156  if (!file)
157  {
158  return False;
159  }
160  file ->SetCompressionLevel(CompressionLevel);
161  tree = new TTree(TreeName.c_str(), title.c_str());
162  gROOT->cd(currdir.c_str());
163  return True;
164  break;
165  }
166 
167  return False;
168 }
169 
170 PHBoolean
172 {
173  // The write function of the PHCompositeNode topNode will
174  // recursively call the write functions of its subnodes, thus
175  // constructing the path-string which is then stored as name of the
176  // Root-branch corresponding to the data of each PHRootIODataNode.
177  topNode->write(this);
178 
179 
180  // Now all PHRootIODataNodes should have called the write function
181  // of this I/O-manager and thus created their branch. The tree can
182  // be filled.
183  if (file && tree)
184  {
185  tree->Fill();
186  if (realTimeSave) tree->AutoSave("SaveSelf");
187  eventNumber++;
188  return True;
189  }
190 
191  return False;
192 }
193 
194 PHBoolean
195 PHNodeIOManager::write(TObject** data, const string& path)
196 {
197  if (file && tree)
198  {
199  TBranch *thisBranch = tree->GetBranch(path.c_str());
200  if (!thisBranch)
201  {
202  // Here is were we decide how to save the data in the root
203  // tree. split=0(prior to Root3.01/05; needs -1 afterwards
204  // for classes with custom streamers, as our old PHTable(s))
205  // means data are hidden, interactive T->draw() will not
206  // work. The old wrapped tables seem to need that, with
207  // split = 1 they cannot be read back. The new PHObjects
208  // can be saved either way, but split = 1 makes interactive
209  // display possible.
210  split = 99;
211  if ((*data)->InheritsFrom("PHObject"))
212  {
213  PHObject *phob = dynamic_cast<PHObject *> (*data);
214  split = phob->SplitLevel();
215  bufSize = phob->BufferSize();
216  }
217  tree->Branch(path.c_str(), (*data)->ClassName(),
218  data, bufSize, split);
219  }
220  else
221  {
222  thisBranch->SetAddress(data);
223  }
224  return True;
225  }
226 
227  return False;
228 }
229 
230 
231 PHBoolean
232 PHNodeIOManager::read(size_t requestedEvent)
233 {
234  if (readEventFromFile(requestedEvent))
235  {
236  return True;
237  }
238  else
239  {
240  return False;
241  }
242 }
243 
245 PHNodeIOManager::read(PHCompositeNode* topNode, size_t requestedEvent)
246 {
247 
248  // No tree means we have not yet looked at the file,
249  // so we'll reconstruct the node tree now.
250  if (!tree)
251  {
252  topNode = reconstructNodeTree(topNode);
253  }
254 
255  // If everything worked, there should be a tree now.
256  if (tree && readEventFromFile(requestedEvent))
257  {
258  return topNode;
259  }
260  else
261  {
262  return 0;
263  }
264 }
265 
266 
267 void
269 {
270  if (file)
271  {
272  if (accessMode == PHReadOnly)
273  {
274  cout << "PHNodeIOManager reading " << filename << endl;
275  }
276  else
277  {
278  cout << "PHNodeIOManager writing " << filename << endl;
279  }
280  }
281  if (file && tree)
282  {
283  tree->Print();
284  }
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)
288  {
289  cout << classiter->first << " is set to " << classiter->second << endl;
290  }
291 }
292 
293 string
294 PHNodeIOManager::getBranchClassName(TBranch* branch)
295 {
296  // OK. Here all the game is to find out the name of the type
297  // contained in this branch. In ROOT pre-3.01/05 versions, all
298  // branches we used were of the same type = TBranchObject, so that
299  // was easy. Since version 3.01/05 ROOT introduced new branch style
300  // with some TBranchElement objects. So far so good. The problem is
301  // that I did not find a common way to grab the typename of the
302  // object contained in those branches, so I hereby use some durty if
303  // { } else if { } ...
304 
305 #if ROOT_VERSION_CODE >= ROOT_VERSION(3,01,5)
306  TBranchElement* be = dynamic_cast<TBranchElement*>(branch);
307 
308  if (be)
309  {
310  // TBranchElement has a nice GetClassName() method for us :
311  return be->GetClassName();
312  }
313 #endif
314 
315  TBranchObject* bo = dynamic_cast<TBranchObject*>(branch);
316  if (bo)
317  {
318  // For this one we need to go down a little before getting the
319  // name...
320  TLeafObject* leaf = static_cast<TLeafObject*>
321  (branch->GetLeaf(branch->GetName()));
322  assert(leaf != 0);
323  return leaf->GetTypeName();
324  }
325  cout << PHWHERE << "Fatal error, dynamic cast of TBranchObject failed" << endl;
326  exit(1);
327 }
328 
329 PHBoolean
330 PHNodeIOManager::readEventFromFile(size_t requestedEvent)
331 {
332  // Se non c'e niente, non possiamo fare niente. Logisch, n'est ce
333  // pas?
334  if (!tree)
335  {
336  PHMessage("PHNodeIOManager::readEventFromFile", PHError,
337  "Tree not initialized.");
338  return False;
339  }
340 
341  int bytesRead;
342 
343  // Due to the current implementation of TBuffer>>(Long_t) we need
344  // to cd() in the current file before trying to fetch any event,
345  // otherwise mixing of reading 2.25/03 DST with writing some
346  // 3.01/05 trees will fail.
347  string currdir = gDirectory->GetPath();
348  TFile* file_ptr = gFile; // save current gFile
349  file->cd();
350 
351  if (requestedEvent)
352  {
353  if ((bytesRead = tree->GetEvent(requestedEvent)))
354  {
355  eventNumber = requestedEvent + 1;
356  }
357  }
358  else
359  {
360  bytesRead = tree->GetEvent(eventNumber++);
361  }
362 
363  gFile = file_ptr; // recover gFile
364  gROOT->cd(currdir.c_str());
365 
366  if (!bytesRead)
367  {
368  return False;
369  }
370  if (bytesRead == -1)
371  {
372  cout << PHWHERE << "Error: Input TTree corrupt, exiting now" << endl;
373  exit(1);
374  }
375  return True;
376 }
377 
378 int
379 PHNodeIOManager::readSpecific(size_t requestedEvent, const char* objectName)
380 {
381  // objectName should be one of the valid branch name of the "T" TTree, and
382  // should be one of the branches selected by selectObjectToRead() method.
383  // No wildcard allowed for the moment.
384  string name = objectName;
385  map<string, TBranch*>::const_iterator p = fBranches.find(name);
386 
387  if (p != fBranches.end())
388  {
389  TBranch* branch = p->second;
390  if (branch)
391  {
392  return branch->GetEvent(requestedEvent);
393  }
394  }
395  else
396  {
397  PHMessage("PHNodeIOManager::readSpecific", PHError,
398  "Unknown object name");
399  }
400  return 0;
401 }
402 
404 PHNodeIOManager::reconstructNodeTree(PHCompositeNode* topNode)
405 {
406  if (! file)
407  {
408  if (filename.empty())
409  {
410  cout << PHWHERE << "filename was never set" << endl;
411  }
412  else
413  {
414  cout << PHWHERE << "TFile " << filename << " NULL pointer" << endl;
415  }
416  return NULL;
417  }
418 
419  tree = static_cast<TTree*>(file->Get(TreeName.c_str()));
420 
421  if (!tree)
422  {
423  cout << PHWHERE << "PHNodeIOManager::reconstructNodeTree : Root Tree "
424  << TreeName << " not found in file " << file->GetName() << endl;
425  return NULL;
426  }
427 
428  // ROOT sucks, we need a unique name for the tree so we can open multiple
429  // files. So we take the memory location of the file pointer which
430  // should be unique within this process to create it
431  ostringstream nname;
432  nname << TreeName << file;
433 
434 
435  tree->SetName(nname.str().c_str());
436 
437  // Select the branches according to objectToRead
438  map<string, PHBoolean>::const_iterator it;
439 
440  if (tree->GetNbranches() > 0)
441  {
442  for (it = objectToRead.begin(); it != objectToRead.end(); ++it)
443  {
444  tree->SetBranchStatus((it->first).c_str(),
445  static_cast<bool>(it->second));
446  }
447  }
448  // The file contains a TTree with a list of the TBranchObjects
449  // attached to it.
450  TObjArray *branchArray = tree->GetListOfBranches();
451 
452  // We need these in the loops down below...
453  size_t i, j;
454 
455  // If a topNode was provided, we can feed the iterator with it.
456  if (!topNode)
457  {
458  topNode = new PHCompositeNode("TOP"); // create topNode if we got a null pointer
459  }
460  PHNodeIterator nodeIter(topNode);
461 
462  // Loop over all branches in the tree. Each branch-name contains the
463  // full 'path' of composite-nodes in the original node tree. We
464  // split the name and reconstruct the tree.
465  string delimeters = phooldefs::branchpathdelim + "/"; // add old backslash for backward compat
466  for (i = 0; i < (size_t)(branchArray->GetEntriesFast()); i++)
467  {
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++) // -1 so we skip the node name
472  {
473  if (!nodeIter.cd(splitvec[ia]))
474  {
475  nodeIter.addNode(new PHCompositeNode(splitvec[ia]));
476  nodeIter.cd(splitvec[ia]);
477  }
478  }
479  TBranch *thisBranch = (TBranch*)((*branchArray)[i]);
480 
481  // Skip non-selected branches
482  if (thisBranch->TestBit(kDoNotProcess))
483  {
484  continue;
485  }
486 
487  string branchClassName = getBranchClassName(thisBranch);
488  string branchName = thisBranch->GetName();
489  fBranches[branchName] = thisBranch;
490 
491  assert(gROOT != 0);
492  TClass* thisClass = gROOT->GetClass(branchClassName.c_str());
493 
494  if (!thisClass)
495  {
496  cout << PHWHERE << endl;
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;
500  }
501  // it does not make sense to continue - the code coredumps
502  // later if a class is not loaded
503  assert(thisClass != 0);
504 
505  PHIODataNode<TObject> *newIODataNode =
506  static_cast<PHIODataNode<TObject> *> (nodeIter.findFirst("PHIODataNode", (*splitvec.rbegin()).c_str()));
507  if (! newIODataNode)
508  {
509  TObject *newTObject = static_cast<TObject*>(thisClass->New());
510  newIODataNode = new PHIODataNode<TObject>(newTObject, (*splitvec.rbegin()).c_str());
511  nodeIter.addNode(newIODataNode);
512  }
513  else
514  {
515  TObject *oldobject = newIODataNode->getData();
516  string oldclass = oldobject->ClassName();
517  if (oldclass != branchClassName)
518  {
519  cout << "You only have to worry if you get this message when reading parallel files"
520  << endl
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());
531  newIODataNode = new PHIODataNode<TObject>(newTObject, (*splitvec.rbegin()).c_str());
532  nodeIter.addNode(newIODataNode);
533  }
534 
535  }
536 
537  if (thisClass->InheritsFrom("PHObject"))
538  {
539  newIODataNode->setObjectType("PHObject");
540  }
541  else
542  {
543  cout << PHWHERE << branchClassName.c_str()
544  << " inherits neither from PHTable nor from PHObject"
545  << " setting type to PHObject" << endl;
546  newIODataNode->setObjectType("PHObject");
547  }
548  thisBranch->SetAddress(&(newIODataNode->data));
549  for (j = 1; j < splitvec.size() - 1; j++)
550  {
551  nodeIter.cd("..");
552  }
553 
554  }
555  return topNode;
556 }
557 
558 void
559 PHNodeIOManager::selectObjectToRead(const char* objectName, PHBoolean readit)
560 {
561  objectToRead[objectName] = readit;
562 
563  // If tree is already open, loop over map and set branch status
564  if (tree)
565  {
566  map<string, PHBoolean>::const_iterator it;
567 
568  for (it = objectToRead.begin(); it != objectToRead.end(); ++it)
569  {
570  tree->SetBranchStatus((it->first).c_str(),
571  static_cast<bool>(it->second));
572  }
573 
574  }
575  return;
576 }
577 
578 PHBoolean
579 PHNodeIOManager::isSelected(const char* objectName)
580 {
581  string name = objectName;
582  map<string, TBranch*>::const_iterator p = fBranches.find(name);
583 
584  if (p != fBranches.end())
585  {
586  return True;
587  }
588 
589  return False;
590 }
591 
592 PHBoolean
594 {
595  if (level < 0)
596  {
597  return False;
598  }
599  CompressionLevel = level;
600  if (file)
601  {
602  file ->SetCompressionLevel(CompressionLevel);
603  }
604 
605  return True;
606 }
607 
608 double
610 {
611  if (file) return file->GetBytesWritten();
612  return 0.;
613 }
614 
615 map<string, TBranch*> *
617 {
618  FillBranchMap();
619  return &fBranches;
620 }
621 
622 int
623 PHNodeIOManager::FillBranchMap()
624 {
625  if (fBranches.empty())
626  {
627  TTree *treetmp = static_cast<TTree *>(file->Get(TreeName.c_str()));
628  if (treetmp)
629  {
630  TObjArray *branchArray = treetmp->GetListOfBranches();
631  for (size_t i = 0; i < (size_t)(branchArray->GetEntriesFast()); i++)
632  {
633  TBranch *thisBranch = (TBranch*)((*branchArray)[i]);
634  string branchName = (*branchArray)[i]->GetName();
635  fBranches[branchName] = thisBranch;
636  }
637  }
638  else
639  {
640  cout << PHWHERE << " No Root Tree " << TreeName
641  << " on file " << filename << endl;
642  return -1;
643  }
644  }
645  return 0;
646 }
void PHMessage(const std::string &functionName, int messageType, const std::string &message)
Definition: PHMessage.cc:8
#define NULL
Definition: Pdb.h:9
virtual bool write(PHIOManager *, const std::string &="")
T * getData()
Definition: PHDataNode.h:21
tobjcast data
Definition: PHDataNode.h:37
std::string filename
Definition: PHIOManager.h:28
size_t eventNumber
Definition: PHIOManager.h:29
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 void closeFile()
virtual ~PHNodeIOManager()
void selectObjectToRead(const char *objectName, PHBoolean readit)
PHCompositeNode * read(PHCompositeNode *=0, size_t=0)
PHBoolean isSelected(const char *objectName)
double GetBytesWritten()
std::map< std::string, TBranch * > * GetBranchMap()
void setObjectType(const std::string &type)
Definition: PHNode.h:37
void SplitLevel(const int i)
Definition: PHObject.h:42
void BufferSize(const int i)
Definition: PHObject.h:44
static const std::string branchpathdelim
Definition: phooldefs.h:8
PHTreeType
Definition: phool.h:16
@ PHEventTree
Definition: phool.h:16
@ PHError
Definition: phool.h:14
int PHBoolean
Definition: phool.h:13
PHAccessType
Definition: phool.h:15
@ PHReadOnly
Definition: phool.h:15
@ PHWrite
Definition: phool.h:15
@ PHUpdate
Definition: phool.h:15
static const int True
Definition: phool.h:10
#define PHWHERE
Definition: phool.h:23
static const int False
Definition: phool.h:9