Logo ROOT  
Reference Guide
 
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Properties Friends Macros Modules Pages
Loading...
Searching...
No Matches
TBranchProxy.cxx
Go to the documentation of this file.
1// @(#)root/base:$Id$
2// Author: Philippe Canal 13/05/2003
3
4/*************************************************************************
5 * Copyright (C) 1995-2000, Rene Brun, Fons Rademakers and al. *
6 * All rights reserved. *
7 * *
8 * For the licensing terms see $ROOTSYS/LICENSE. *
9 * For the list of contributors see $ROOTSYS/README/CREDITS. *
10 *************************************************************************/
11
12/** \class ROOT::Detail::TBranchProxy
13Base class for all the proxy object. It includes the implementation
14of the autoloading of branches as well as all the generic setup routine.
15*/
16
17#include "TBranchProxy.h"
18#include "TLeaf.h"
19#include "TBranchElement.h"
20#include "TBranchObject.h"
21#include "TCollection.h" // TRangeStaticCast
22#include "TStreamerElement.h"
23#include "TStreamerInfo.h"
24#include "ROOT/InternalTreeUtils.hxx" // GetFileNamesFromTree, GetTreeFullPaths
25
26#include <string>
27#include <string_view>
28
30
31using namespace ROOT::Internal;
32
33namespace {
34/**
35 * \brief Find if the input branch name is the prefix for other sub-branches in the same tree.
36 */
37bool AreThereSubBranches(std::string_view parentBrName, TTree &tree)
38{
39 const std::string prefix = [&parentBrName]() {
40 if (parentBrName.back() == '.')
41 return std::string(parentBrName);
42 return std::string(parentBrName) + '.';
43 }();
44
45 for (auto *tree.GetListOfLeaves()))
46 // Compares the prefix string with the leaf name, checking if the first
47 // `prefix.size()` characters match those of the prefix, i.e. if the leaf
48 // name starts with the prefix
49 if (prefix.compare(0, prefix.size(), leaf->GetName(), prefix.size()) == 0)
51 return false;
52}
53} // namespace
54
55////////////////////////////////////////////////////////////////////////////////
56/// Constructor.
57
59 fDirector(nullptr), fInitialized(false), fIsMember(false), fIsClone(false), fIsaPointer(false),
60 fHasLeafCount(false), fBranchName(""), fParent(nullptr), fDataMember(""),
61 fClassName(""), fClass(nullptr), fElement(nullptr), fMemberOffset(0), fOffset(0), fArrayLength(1),
62 fBranch(nullptr), fBranchCount(nullptr),
63 fNotify(this),
64 fRead(-1), fWhere(nullptr),fCollection(nullptr)
65{
66};
67
68////////////////////////////////////////////////////////////////////////////////
69/// Constructor.
70
72 const char* name) :
73 fDirector(boss), fInitialized(false), fIsMember(false), fIsClone(false), fIsaPointer(false),
74 fHasLeafCount(false), fBranchName(top), fParent(nullptr), fDataMember(""),
75 fClassName(""), fClass(nullptr), fElement(nullptr), fMemberOffset(0), fOffset(0), fArrayLength(1),
76 fBranch(nullptr), fBranchCount(nullptr),
77 fNotify(this),
78 fRead(-1), fWhere(nullptr),fCollection(nullptr)
79{
80 if (fBranchName.Length() && fBranchName[fBranchName.Length()-1]!='.' && name) {
81 ((TString&)fBranchName).Append(".");
82 }
83 if (name) ((TString&)fBranchName).Append(name);
84 boss->Attach(this);
85}
86
87////////////////////////////////////////////////////////////////////////////////
88/// Constructor.
89
91 fDirector(boss), fInitialized(false), fIsMember(true), fIsClone(false), fIsaPointer(false),
92 fHasLeafCount(false), fBranchName(top), fParent(nullptr), fDataMember(membername),
93 fClassName(""), fClass(nullptr), fElement(nullptr), fMemberOffset(0), fOffset(0), fArrayLength(1),
94 fBranch(nullptr), fBranchCount(nullptr),
95 fNotify(this),
96 fRead(-1), fWhere(nullptr),fCollection(nullptr)
97{
98 if (name && strlen(name)) {
99 if (fBranchName.Length() && fBranchName[fBranchName.Length()-1]!='.') {
100 ((TString&)fBranchName).Append(".");
101 }
102 ((TString&)fBranchName).Append(name);
103 }
104 boss->Attach(this);
105}
106
107////////////////////////////////////////////////////////////////////////////////
108/// Constructor.
109
111 const char* name) :
112 fDirector(boss), fInitialized(false), fIsMember(true), fIsClone(false), fIsaPointer(false),
113 fHasLeafCount(false), fBranchName(top), fParent(parent), fDataMember(membername),
114 fClassName(""), fClass(nullptr), fElement(nullptr), fMemberOffset(0), fOffset(0), fArrayLength(1),
115 fBranch(nullptr), fBranchCount(nullptr),
116 fNotify(this),
117 fRead(-1), fWhere(nullptr),fCollection(nullptr)
119 if (name && strlen(name)) {
120 if (fBranchName.Length() && fBranchName[fBranchName.Length()-1]!='.') {
121 ((TString&)fBranchName).Append(".");
122 }
123 ((TString&)fBranchName).Append(name);
124 }
125 boss->Attach(this);
126}
127
128////////////////////////////////////////////////////////////////////////////////
129/// Constructor.
130
132 fDirector(boss), fInitialized(false), fIsMember(membername != nullptr && membername[0]), fIsClone(false), fIsaPointer(false),
133 fHasLeafCount(false), fBranchName(branch->GetName()), fParent(nullptr), fDataMember(membername),
134 fClassName(""), fClass(nullptr), fElement(nullptr), fMemberOffset(0), fOffset(0), fArrayLength(1),
135 fBranch(nullptr), fBranchCount(nullptr),
136 fNotify(this),
137 fRead(-1), fWhere(nullptr),fCollection(nullptr)
138{
139 boss->Attach(this);
140}
141
142////////////////////////////////////////////////////////////////////////////////
143/// For a fullBranchName that might contain a leading friend tree path (but
144/// access elements designating a leaf), but the leaf name such that it matches
145/// the "path" to branch.
146
148{
149 // ROOT-10046: Here we need to ask for the tree with GetTree otherwise, if directorTree
150 // is a chain, this check is bogus and a bug can occur (ROOT-10046)
151 if (directorTree->GetTree() == branch->GetTree())
152 return branch->GetFullName().Data();
153
154 // Friend case:
155 std::string sFullBranchName = fullBranchName;
156 std::string::size_type pos = sFullBranchName.rfind(branch->GetFullName());
157 if (pos != std::string::npos) {
158 sFullBranchName.erase(pos);
159 sFullBranchName += branch->GetFullName();
160 }
161 return sFullBranchName;
162}
163
164////////////////////////////////////////////////////////////////////////////////
165/// Constructor taking the branch name, possibly of a friended tree.
166/// Used by TTreeReaderValue in place of TFriendProxy.
167
169 const char *membername, bool suppressMissingBranchError)
170 : fDirector(boss),
171 fInitialized(false),
172 fIsMember(membername != nullptr && membername[0]),
173 fIsClone(false),
174 fIsaPointer(false),
175 fHasLeafCount(false),
176 fSuppressMissingBranchError(suppressMissingBranchError),
177 fBranchName(GetFriendBranchName(boss->GetTree(), branch, branchname)),
178 fParent(nullptr),
179 fDataMember(membername),
180 fClassName(""),
181 fClass(nullptr),
182 fElement(nullptr),
183 fMemberOffset(0),
184 fOffset(0),
185 fArrayLength(1),
186 fBranch(nullptr),
187 fBranchCount(nullptr),
188 fNotify(this),
189 fRead(-1),
190 fWhere(nullptr),
191 fCollection(nullptr)
192{
193 // Constructor.
194
195 boss->Attach(this);
196}
197
198////////////////////////////////////////////////////////////////////////////////
199/// Typical Destructor
200
202{
203 if (fNotify.IsLinked() && fDirector && fDirector->GetTree())
204 fNotify.RemoveLink(*(fDirector->GetTree()));
205}
206
207////////////////////////////////////////////////////////////////////////////////
208/// Completely reset the object.
209
211{
212 fWhere = nullptr;
213 fBranch = nullptr;
214 fBranchCount = nullptr;
215 fRead = -1;
216 fClass = nullptr;
217 fElement = nullptr;
218 fMemberOffset = 0;
219 fOffset = 0;
220 fArrayLength = 1;
221 fIsClone = false;
222 fInitialized = false;
223 fHasLeafCount = false;
224 delete fCollection;
225 fCollection = nullptr;
226}
227
228////////////////////////////////////////////////////////////////////////////////
229/// Display the content of the object
230
232{
233 std::cout << "fBranchName " << fBranchName << std::endl;
234 //std::cout << "fTree " << fDirector->fTree << std::endl;
235 std::cout << "fBranch " << fBranch << std::endl;
236 if (fHasLeafCount)
237 std::cout << "fLeafCount " << fLeafCount << std::endl;
238 else if (fBranchCount)
239 std::cout << "fBranchCount " << fBranchCount << std::endl;
240}
241
242
243////////////////////////////////////////////////////////////////////////////////
244/// Initialize/cache the necessary information.
245
247{
248 // Should we check the type?
249
250 if (!fDirector->GetTree()) {
251 return false;
252 }
253 if (!fNotify.IsLinked()) {
254 fNotify.PrependLink(*fDirector->GetTree());
255 }
256 if (fParent) {
257
258 if (!fParent->Setup()) {
259 return false;
260 }
261
262 TClass *pcl = fParent->GetClass();
263 R__ASSERT(pcl);
264
265 if (pcl==TClonesArray::Class()) {
266 // We always skip the clones array
267
268 Int_t i = fDirector->GetReadEntry();
269 if (i<0) fDirector->SetReadEntry(0);
270 if (fParent->Read()) {
271 if (i<0) fDirector->SetReadEntry(i);
272
274 clones = (TClonesArray*)fParent->GetStart();
275
276 if (clones) pcl = clones->GetClass();
277 }
278 } else if (pcl->GetCollectionProxy()) {
279 // We always skip the collections.
280
281 if (fCollection) delete fCollection;
282 fCollection = pcl->GetCollectionProxy()->Generate();
283 pcl = fCollection->GetValueClass();
284 if (pcl == nullptr) {
285 // coverity[dereference] fparent is checked jus a bit earlier and can not be null here
286 Error("Setup","Not finding TClass for collection for the data member %s seems no longer be in class %s",fDataMember.Data(),fParent->GetClass()->GetName());
287 return false;
288 }
289 }
290
291 fElement = (TStreamerElement*)pcl->GetStreamerInfo()->GetStreamerElement(fDataMember, fOffset);
292 if (fElement) {
293 fIsaPointer = fElement->IsaPointer();
294 fClass = fElement->GetClassPointer();
295 fMemberOffset = fElement->GetOffset();
296 fArrayLength = fElement->GetArrayLength();
297 fValueSize = fElement->GetSize();
298 } else {
299 Error("Setup","Data member %s seems no longer be in class %s",fDataMember.Data(),pcl->GetName());
300 return false;
301 }
302
303 fIsClone = (fClass==TClonesArray::Class());
304
305 fWhere = fParent->fWhere; // not really used ... it is reset by GetStart and GetClStart
306
307 if (fParent->IsaPointer()) {
308 // fprintf(stderr,"non-split pointer de-referencing non implemented yet \n");
309 // nothing to do!
310 } else {
311 // Accumulate offsets.
312 // or not!? fOffset = fMemberOffset = fMemberOffset + fParent->fOffset;
313 }
314
315 // This is not sufficient for following pointers
316
317 } else {
318
319 // This does not allow (yet) to precede the branch name with
320 // its mother's name
321 fBranch = fDirector->GetTree()->GetBranch(fBranchName.Data());
322 if (!fBranch) {
323 if (!fSuppressMissingBranchError && !AreThereSubBranches(fBranchName.View(), *fDirector->GetTree())) {
324 // The next error refers specifically to the situation where the branch identified by fBranchName
325 // is not present and that is not expected. An example is when traversing a chain of files, the branch
326 // is missing from the file that we are switching into.
327 // Conversely, there are situations where the missing branch is indeed expected. A notable example is when
328 // the TTree contains a split object, the branch referring to the whole object type will actually be elided
329 // and will not be found by `TTree::GetBranch`, only the data members will be present as sub branches.
330 auto *tree = fDirector->GetTree()->GetTree(); // Double GetTree to extract the current TTree being processed
331 Error("TBranchProxy::Setup()", "%s",
332 Form("Branch '%s' is not available from tree '%s' in file '%s'.", fBranchName.Data(),
335 }
336 return false;
337 }
338
339 fWhere = (double*)fBranch->GetAddress();
340
341 if (!fWhere && fBranch->IsA()==TBranchElement::Class()
342 && ((TBranchElement*)fBranch)->GetMother()) {
343
344 TBranchElement* be = ((TBranchElement*)fBranch);
345
346 be->GetMother()->SetAddress(nullptr);
347 fWhere = (double*)fBranch->GetAddress();
348
349 }
350 if (fBranch->IsA()==TBranch::Class()) {
351 TLeaf *leaf2 = nullptr;
352 if (fDataMember.Length()) {
353 leaf2 = fBranch->GetLeaf(fDataMember);
354 } else if (!fWhere) {
355 leaf2 = (TLeaf*)fBranch->GetListOfLeaves()->At(0); // fBranch->GetLeaf(fLeafname);
356 fWhere = leaf2->GetValuePointer();
357 }
358 if (leaf2) {
359 fWhere = leaf2->GetValuePointer();
360 fArrayLength = leaf2->GetLen();
361 fValueSize = leaf2->GetLenType();
362 if (leaf2->GetLeafCount()) {
363 fLeafCount = leaf2->GetLeafCount();
364 fHasLeafCount = true;
365 }
366 }
367 } else if (fBranch->IsA() == TBranchElement::Class()) {
368 // Calculate fBranchCount for a leaf.
369 TLeaf *leaf = (TLeaf*) fBranch->GetListOfLeaves()->At(0); // fBranch->GetLeaf(fLeafname);
370 if (leaf)
371 leaf = leaf->GetLeafCount();
372 if (leaf) {
373 fBranchCount = dynamic_cast<TBranchElement*>(leaf->GetBranch());
374 }
375 }
376
377 if (!fWhere) {
378 fBranch->SetupAddresses();
379 fWhere = (double*)fBranch->GetAddress();
380 }
381
382
383 if (fWhere && fBranch->IsA()==TBranchElement::Class()) {
384
385 TBranchElement* be = ((TBranchElement*)fBranch);
386
387 TStreamerInfo * info = be->GetInfo();
388 Int_t id = be->GetID();
389 if (be->GetType() == 3) {
390 fClassName = "TClonesArray";
392 } else if (id>=0) {
393 fOffset = info->GetElementOffset(id);
394 fElement = (TStreamerElement*)info->GetElements()->At(id);
395 fIsaPointer = fElement->IsaPointer();
396 fClass = fElement->GetClassPointer();
397 fValueSize = fElement->GetSize();
398
399 if ((fIsMember || (be->GetType()!=3 && be->GetType() !=4))
400 && (be->GetType()!=31 && be->GetType()!=41)) {
401
403 Int_t i = be->GetTree()->GetReadEntry();
404 if (i<0) i = 0;
405 be->GetEntry(i);
406
408 if ( fIsMember && be->GetType()==3 ) {
409 clones = (TClonesArray*)be->GetObject();
410 } else if (fIsaPointer) {
411 clones = (TClonesArray*)*(void**)((char*)fWhere+fOffset);
412 } else {
413 clones = (TClonesArray*)((char*)fWhere+fOffset);
414 }
415 if (!fIsMember) fIsClone = true;
416 fClass = clones->GetClass();
417 } else if (fClass && fClass->GetCollectionProxy()) {
418 delete fCollection;
419 fCollection = fClass->GetCollectionProxy()->Generate();
420 fClass = fCollection->GetValueClass();
421 }
422
423 }
424 if (fClass) fClassName = fClass->GetName();
425 } else {
426 fClassName = be->GetClassName();
427 fClass = TClass::GetClass(fClassName);
428 }
429
430 if (be->GetType()==3) {
431 // top level TClonesArray
432
433 if (!fIsMember) fIsClone = true;
434 fIsaPointer = false;
435 fWhere = be->GetObject();
436
437 } else if (be->GetType()==4) {
438 // top level TClonesArray
439
440 fCollection = be->GetCollectionProxy()->Generate();
441 fIsaPointer = false;
442 fWhere = be->GetObject();
443
444 } else if (id<0) {
445 // top level object
446
447 fIsaPointer = false;
448 fWhere = be->GetObject();
449
450 } else if (be->GetType()==41) {
451
452 fCollection = be->GetCollectionProxy()->Generate();
453 fWhere = be->GetObject();
454 fOffset += be->GetOffset();
455
456 } else if (be->GetType()==31) {
457
458 fWhere = be->GetObject();
459 fOffset += be->GetOffset();
460
461 } else if (be->GetType()==2) {
462 // this might also be the right path for GetType()==1
463
464 fWhere = be->GetObject();
465
466 } else {
467
468 // fWhere = ((unsigned char*)fWhere) + fOffset;
469 fWhere = ((unsigned char*)be->GetObject()) + fOffset;
470
471 }
472 } else if (fBranch->IsA() == TBranchObject::Class()) {
473 fIsaPointer = true; // this holds for all cases we test
474 fClassName = fBranch->GetClassName();
475 fClass = TClass::GetClass(fClassName);
476 } else {
477 fClassName = fBranch->GetClassName();
478 fClass = TClass::GetClass(fClassName);
479 }
480
481 /*
482 fClassName = fBranch->GetClassName(); // What about TClonesArray?
483 if ( fBranch->IsA()==TBranchElement::Class() &&
484 ((TBranchElement*)fBranch)->GetType()==31 ||((TBranchElement*)fBranch)->GetType()==3 ) {
485
486 Int_t id = ((TBranchElement*)fBranch)->GetID();
487 if (id>=0) {
488
489 fElement = ((TStreamerElement*)(((TBranchElement*)fBranch)->GetInfo())->GetElements()->At(id));
490 fClass = fElement->GetClassPointer();
491 if (fClass) fClassName = fClass->GetName();
492
493 }
494 }
495 if (fClass==0 && fClassName.Length()) fClass = TClass::GetClass(fClassName);
496 */
497 //fprintf(stderr,"For %s fClass is %p which is %s\n",
498 // fBranchName.Data(),fClass,fClass==0?"not set":fClass->GetName());
499
500 if ( fBranch->IsA()==TBranchElement::Class() &&
501 (((TBranchElement*)fBranch)->GetType()==3 || fClass==TClonesArray::Class()) &&
502 !fIsMember ) {
503 fIsClone = true;
504 }
505
506 if (fIsMember) {
507 if ( fBranch->IsA()==TBranchElement::Class() &&
509 (((TBranchElement*)fBranch)->GetType()==31 || ((TBranchElement*)fBranch)->GetType()==3) ) {
510
511 TBranchElement *bcount = ((TBranchElement*)fBranch)->GetBranchCount();
513 if (bcount) {
514 TString bname = fBranch->GetName();
515 TString bcname = bcount->GetName();
516 member = bname.Remove(0,bcname.Length()+1);
517 } else {
518 member = fDataMember;
519 }
520
521 fMemberOffset = fClass->GetDataMemberOffset(member);
522
523 if (fMemberOffset < 0) {
524 Error("Setup","%s",Form("Negative offset %d for %s in %s",
525 fMemberOffset,fBranch->GetName(),
526 bcount?bcount->GetName():"unknown"));
527 fMemberOffset = 0;
528 } else if (fMemberOffset == TVirtualStreamerInfo::kMissing) {
529 Error("Setup", "%s",
530 Form("Missing data member in a TClonesArray, %s in %s and %s", fDataMember.Data(),
531 fBranch->GetName(), bcount ? bcount->GetName() : "unknown"));
532 fMemberOffset = 0;
533 }
534
535 } else if (fParent && fClass) {
536
537 fElement = (TStreamerElement*)
538 fClass->GetStreamerInfo()->GetElements()->FindObject(fDataMember);
539 if (fElement) {
540 fMemberOffset = fElement->GetOffset();
541 fValueSize = fElement->GetSize();
542 } else {
543 // Need to compose the proper sub name
544
546
547 member += fDataMember;
548 fMemberOffset = fClass->GetDataMemberOffset(member);
549 }
550 if (fMemberOffset < 0) {
551 Error("Setup", "%s",
552 Form("Negative offset %d for %s in %s, class: %s", fMemberOffset, fDataMember.Data(),
553 fBranch->GetName(), fClass->GetName()));
554 fMemberOffset = 0;
555 } else if (fMemberOffset == TVirtualStreamerInfo::kMissing) {
556 Error("Setup", "%s",
557 Form("Missing data member %s in %s, class: %s", fDataMember.Data(), fBranch->GetName(),
558 fClass->GetName()));
559 fMemberOffset = 0;
560 }
561
562 // The extra condition (fElement is not a TStreamerSTL) is to handle the case where fBranch is a
563 // TBranchElement and fElement is a TStreamerSTL. Without the extra condition we get an error
564 // message, although the vector (i.e. the TBranchElement) is accessible.
565 } else if (fParent && fBranch->IsA() != TBranch::Class() && fElement->IsA() != TStreamerBasicType::Class() &&
566 fElement->IsA() != TStreamerSTL::Class()) {
567 Error("Setup", "%s", Form("Missing TClass object for %s", fClassName.Data()));
568 }
569
570 if ( fBranch->IsA()==TBranchElement::Class()
571 && (((TBranchElement*)fBranch)->GetType()==31 || ((TBranchElement*)fBranch)->GetType()==3) ) {
572
573 fOffset = fMemberOffset;
574
575 } else {
576
577 fWhere = ((unsigned char*)fWhere) + fMemberOffset;
578 }
579 }
580 }
581 if (fClass==TClonesArray::Class()) fIsClone = true;
582 if (fWhere!=nullptr) {
583 fInitialized = true;
584 return true;
585 } else {
586 return false;
587 }
588}
size_t fValueSize
Cppyy::TCppType_t fClass
size_t size(const MatrixT &matrix)
retrieve the size of a square matrix
#define ClassImp(name)
Definition Rtypes.h:374
static std::string GetFriendBranchName(TTree *directorTree, TBranch *branch, const char *fullBranchName)
For a fullBranchName that might contain a leading friend tree path (but access elements designating a...
ROOT::Detail::TRangeCast< T, true > TRangeDynCast
TRangeDynCast is an adapter class that allows the typed iteration through a TCollection.
#define R__ASSERT(e)
Checks condition e and reports a fatal error if it's false.
Definition TError.h:125
void Error(const char *location, const char *msgfmt,...)
Use this function in case an error occurred.
Definition TError.cxx:185
char name[80]
Definition TGX11.cxx:110
char * Form(const char *fmt,...)
Formats a string in a circular formatting buffer.
Definition TString.cxx:2489
Base class for all the proxy object.
bool Setup()
Initialize/cache the necessary information.
void Reset()
Completely reset the object.
virtual void Print()
Display the content of the object.
virtual ~TBranchProxy()
Typical Destructor.
A Branch for the case of an object.
static TClass * Class()
void SetupAddresses() override
If the branch address is not set, we set all addresses starting with the top level parent branch.
TClass * IsA() const override
static TClass * Class()
A TTree is a list of TBranches.
Definition TBranch.h:93
static TClass * Class()
TClass instances represent classes, structs and namespaces in the ROOT type system.
Definition TClass.h:84
static TClass * GetClass(const char *name, Bool_t load=kTRUE, Bool_t silent=kFALSE)
Static method returning pointer to TClass of the specified class name.
Definition TClass.cxx:3069
An array of clone (identical) objects.
static TClass * Class()
A TLeaf describes individual elements of a TBranch See TBranch structure in TTree.
Definition TLeaf.h:57
const char * GetName() const override
Returns name of object.
Definition TNamed.h:49
static TClass * Class()
Describes a persistent version of a class.
static TClass * Class()
Basic string class.
Definition TString.h:139
TString & Remove(Ssiz_t pos)
Definition TString.h:685
A TTree represents a columnar dataset.
Definition TTree.h:79
std::vector< std::string > GetTreeFullPaths(const TTree &tree)
std::vector< std::string > GetFileNamesFromTree(const TTree &tree)
tbb::task_arena is an alias of tbb::interface7::task_arena, which doesn't allow to forward declare tb...