Logo ROOT   6.30.04
Reference Guide
 All Namespaces Files Pages
TLeafObject.cxx
Go to the documentation of this file.
1 // @(#)root/tree:$Id$
2 // Author: Rene Brun 27/01/96
3 
4 /*************************************************************************
5  * Copyright (C) 1995-2000, Rene Brun and Fons Rademakers. *
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 TLeafObject
13 \ingroup tree
14 
15 A TLeaf for a general object derived from TObject.
16 */
17 
18 #include "TLeafObject.h"
19 #include "TBranch.h"
20 #include "TBuffer.h"
21 #include "TClass.h"
22 #include "TMethodCall.h"
23 #include "TDataType.h"
24 
25 ClassImp(TLeafObject);
26 
27 ////////////////////////////////////////////////////////////////////////////////
28 /// Default constructor for LeafObject.
29 
30 TLeafObject::TLeafObject(): TLeaf()
31 {
32  fClass = 0;
33  fObjAddress = 0;
34  fVirtual = kTRUE;
35 }
36 
37 ////////////////////////////////////////////////////////////////////////////////
38 /// Create a LeafObject.
39 
40 TLeafObject::TLeafObject(TBranch *parent, const char *name, const char *type)
41  :TLeaf(parent, name,type)
42 {
43  SetTitle(type);
44  fClass = TClass::GetClass(type);
45  fObjAddress = 0;
46  fVirtual = kTRUE;
47 }
48 
49 ////////////////////////////////////////////////////////////////////////////////
50 /// Default destructor for a LeafObject.
51 
52 TLeafObject::~TLeafObject()
53 {
54 }
55 
56 ////////////////////////////////////////////////////////////////////////////////
57 /// Pack leaf elements in Basket output buffer.
58 
59 void TLeafObject::FillBasket(TBuffer &b)
60 {
61  if (!fObjAddress) return;
62  TObject *object = GetObject();
63  if (object) {
64  if (fVirtual) {
65  UChar_t n = (UChar_t) strlen(object->ClassName());
66  b << n;
67  b.WriteFastArray(object->ClassName(),n+1);
68  }
69  object->Streamer(b);
70  } else {
71  if (fClass) {
72  if (fClass->Property() & kIsAbstract) object = new TObject;
73  else object = (TObject *)fClass->New();
74  object->SetBit(kInvalidObject);
75  object->SetUniqueID(123456789);
76  object->Streamer(b);
77  if (fClass->Property() & kIsAbstract) delete object;
78  else fClass->Destructor(object);
79  } else {
80  Error("FillBasket","Attempt to write a NULL object in leaf:%s",GetName());
81  }
82  }
83 }
84 
85 ////////////////////////////////////////////////////////////////////////////////
86 /// Returns pointer to method corresponding to name.
87 ///
88 /// name is a string with the general form "method(list of params)"
89 /// If list of params is omitted, () is assumed;
90 
91 TMethodCall *TLeafObject::GetMethodCall(const char *name)
92 {
93  char *namecpy = new char[strlen(name)+1];
94  strcpy(namecpy,name);
95  char *params = strchr(namecpy,'(');
96  if (params) { *params = 0; params++; }
97  else params = (char *) ")";
98 
99  if (!fClass) fClass = TClass::GetClass(GetTitle());
100  TMethodCall *m = new TMethodCall(fClass, namecpy, params);
101  delete [] namecpy;
102  if (m->GetMethod()) return m;
103  Error("GetMethodCall","Unknown method:%s",name);
104  delete m;
105  return 0;
106 }
107 
108 ////////////////////////////////////////////////////////////////////////////////
109 /// Returns name of leaf type.
110 
111 const char *TLeafObject::GetTypeName() const
112 {
113  return fTitle.Data();
114 }
115 
116 ////////////////////////////////////////////////////////////////////////////////
117 /// This method must be overridden to handle object notifcation.
118 
119 Bool_t TLeafObject::Notify()
120 {
121  fClass = TClass::GetClass(GetTitle());
122  return kFALSE;
123 }
124 
125 ////////////////////////////////////////////////////////////////////////////////
126 /// Prints leaf value.
127 
128 void TLeafObject::PrintValue(Int_t) const
129 {
130  printf("%lx\n",(Long_t)GetValuePointer());
131 }
132 
133 ////////////////////////////////////////////////////////////////////////////////
134 /// Read leaf elements from Basket input buffer.
135 
136 void TLeafObject::ReadBasket(TBuffer &b)
137 {
138  char classname[128];
139  UChar_t n;
140  if (fVirtual) {
141  b >> n;
142  b.ReadFastArray(classname,n+1);
143  fClass = TClass::GetClass(classname);
144  }
145  if (fClass) {
146  TObject *object;
147  if (!fObjAddress) {
148  Long_t *voidobj = new Long_t[1];
149  fObjAddress = (void **)voidobj;
150  *fObjAddress = (TObject *)fClass->New();
151  }
152  object = (TObject*)(*fObjAddress);
153  if (fBranch->IsAutoDelete()) {
154  fClass->Destructor(object);
155  object = (TObject *)fClass->New();
156  }
157  if (!object) return;
158 
159  if (fClass->GetState() > TClass::kEmulated) {
160  object->Streamer(b);
161  } else {
162  //emulated class has no Streamer
163  if (!TestBit(kWarn)) {
164  Warning("ReadBasket","%s::Streamer not available, using TClass::ReadBuffer instead",fClass->GetName());
165  SetBit(kWarn);
166  }
167  fClass->ReadBuffer(b,object);
168  }
169  // in case we had written a null pointer a Zombie object was created
170  // we must delete it
171  if (object->TestBit(kInvalidObject)) {
172  if (object->GetUniqueID() == 123456789) {
173  fClass->Destructor(object);
174  object = 0;
175  }
176  }
177  *fObjAddress = object;
178  } else GetBranch()->SetAddress(0);
179 }
180 
181 ////////////////////////////////////////////////////////////////////////////////
182 /// Set leaf buffer data address.
183 
184 void TLeafObject::SetAddress(void *add)
185 {
186  fObjAddress = (void **)add;
187 }
188 
189 ////////////////////////////////////////////////////////////////////////////////
190 /// Stream an object of class TLeafObject.
191 
192 void TLeafObject::Streamer(TBuffer &b)
193 {
194  if (b.IsReading()) {
195  UInt_t R__s, R__c;
196  Version_t R__v = b.ReadVersion(&R__s, &R__c);
197  if (R__v > 3 || R__v == 2) {
198  b.ReadClassBuffer(TLeafObject::Class(), this, R__v, R__s, R__c);
199  if (R__v == 2) fVirtual = kTRUE;
200  fObjAddress = 0;
201  fClass = TClass::GetClass(fTitle.Data());
202  if (!fClass) Warning("Streamer","Cannot find class:%s",fTitle.Data());
203 
204  // We should rewarn in this process.
205  ResetBit(kWarn);
206  ResetBit(kOldWarn);
207 
208  return;
209  }
210  //====process old versions before automatic schema evolution
211  TLeaf::Streamer(b);
212  fObjAddress = 0;
213  fClass = TClass::GetClass(fTitle.Data());
214  if (!fClass) Warning("Streamer","Cannot find class:%s",fTitle.Data());
215  if (R__v < 1) fVirtual = kFALSE;
216  if (R__v == 1) fVirtual = kTRUE;
217  if (R__v == 3) b >> fVirtual;
218  // We should rewarn in this process.
219  ResetBit(kOldWarn);
220  //====end of old versions
221 
222  } else {
223  b.WriteClassBuffer(TLeafObject::Class(),this);
224  }
225 }
226 
227 ////////////////////////////////////////////////////////////////////////////////
228 /// Return true if this leaf is does not have any sub-branch/leaf.
229 
230 Bool_t TLeafObject::IsOnTerminalBranch() const
231 {
232  if (fBranch->GetListOfBranches()->GetEntriesFast()) return kFALSE;
233  return kTRUE;
234 }