ROOT  6.07/01
Reference Guide
 All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Properties Friends Macros Groups Pages
ProcFileElements.C
Go to the documentation of this file.
1 /// \file
2 /// \ingroup tutorial_ProcFileElements
3 ///
4 /// Class to hold information about the processed elements of a file
5 ///
6 /// \macro_code
7 ///
8 /// \author Gerardo Ganis (gerardo.ganis@cern.ch)
9 
10 #include "ProcFileElements.h"
11 #include "TCollection.h"
12 
13 //_______________________________________________________________________
15 {
16  // Compare this element with 'e'.
17  // Return
18  // -1 this should come first
19  // 0 same
20  // 1 e should come first
21 
22  const ProcFileElements::ProcFileElement *e =
23  dynamic_cast<const ProcFileElements::ProcFileElement *>(o);
24  if (!e) return -1;
25 
26  if (fFirst == e->fFirst) {
27  // They start at the same point
28  return 0;
29  } else if (fFirst < e->fFirst) {
30  // This starts first
31  return -1;
32  } else {
33  // e starts first
34  return 1;
35  }
36 }
37 
38 //_______________________________________________________________________
39 Int_t ProcFileElements::ProcFileElement::Overlapping(ProcFileElement *e)
40 {
41  // Check overlapping status of this element with 'e'.
42  // Return
43  // -1 not overlapping
44  // 0 adjacent
45  // 1 overlapping
46 
47  if (!e) return -1;
48 
49  if (fFirst == 0 && fLast == -1) {
50  // We cover the full range, so we overlap
51  return 1;
52  }
53 
54  if (fFirst < e->fFirst) {
55  if (fLast >= 0) {
56  if (fLast < e->fFirst - 1) {
57  // Disjoint
58  return -1;
59  } else {
60  // We somehow overlap
61  if (fLast == e->fFirst - 1) {
62  // Just adjacent
63  return 0;
64  } else {
65  // Real overlap
66  return 1;
67  }
68  }
69  } else {
70  // Always overlapping
71  return 1;
72  }
73  } else if (fFirst == e->fFirst) {
74  // Overlapping or adjacent
75  if (fFirst == fLast || e->fFirst == e->fLast) return 0;
76  return 1;
77  } else {
78  // The other way around
79  if (e->fLast >= 0) {
80  if (e->fLast < fFirst - 1) {
81  // Disjoint
82  return -1;
83  } else {
84  // We somehow overlap
85  if (e->fLast == fFirst - 1) {
86  // Just adjacent
87  return 0;
88  } else {
89  // Real overlap
90  return 1;
91  }
92  }
93  } else {
94  // Always overlapping
95  return 1;
96  }
97  }
98 
99  // Should never be here
100  Warning("Overlapping", "should never be here!");
101  return -1;
102 }
103 
104 //_______________________________________________________________________
105 Int_t ProcFileElements::ProcFileElement::MergeElement(ProcFileElement *e)
106 {
107  // Merge this element with element 'e'.
108  // Return -1 if it cannot be done, i.e. thei are disjoint; 0 otherwise
109 
110  // Check if it can be done
111  if (Overlapping(e) < 0) return -1;
112 
113  // Ok, we can merge: set the lower bound
114  if (e->fFirst < fFirst) fFirst = e->fFirst;
115 
116  // Set the upper bound
117  if (fLast == -1 || e->fLast == -1) {
118  fLast = -1;
119  } else {
120  if (fLast < e->fLast) fLast = e->fLast;
121  }
122  // Done
123  return 0;
124 }
125 
126 //_______________________________________________________________________
128 {
129  // Print range of this element
130 
131  Printf("\tfirst: %lld\t last: %lld", fFirst, fLast);
132 }
133 
134 //_______________________________________________________________________
136 {
137  // Add a new element to the list
138  // Return 1 if a new element has been added, 0 if it has been merged
139  // with an existing one, -1 in case of error
140 
141  if (!fElements) fElements = new TSortedList;
142  if (!fElements) {
143  Error("Add", "could not create internal list!");
144  return -1;
145  }
146 
147  // Create (temporary element)
148  ProcFileElements::ProcFileElement *ne =
149  new ProcFileElements::ProcFileElement(fst, lst);
150 
151  // Check if if it is adjacent or overlapping with an existing one
152  TIter nxe(fElements);
153  ProcFileElements::ProcFileElement *e = 0;
154  while ((e = (ProcFileElements::ProcFileElement *)nxe())) {
155  if (e->MergeElement(ne) == 0) break;
156  }
157 
158  Int_t rc = 0;
159  // Remove and re-add the merged element to sort correctly its possibly new position
160  if (e) {
161  fElements->Remove(e);
162  fElements->Add(e);
163  SafeDelete(ne);
164  } else {
165  // Add the new element
166  fElements->Add(ne);
167  rc = 1;
168  }
169 
170  // Make sure that all what can be merged is merged (because of the order, some elements
171  // which could be merged are not merged, making the determination of fFirst and fLast below
172  // to give incorrect values)
173 
174  ProcFileElements::ProcFileElement *ep = 0, *en = 0;
175  TObjLink *olp = fElements->FirstLink(), *oln = 0;
176  while (olp && (ep = (ProcFileElements::ProcFileElement *) olp->GetObject())) {
177  oln = olp->Next();
178  while (oln) {
179  if ((en = (ProcFileElements::ProcFileElement *) oln->GetObject())) {
180  if (ep->MergeElement(en) == 0) {
181  fElements->Remove(en);
182  delete en;
183  }
184  }
185  oln = oln->Next();
186  }
187  olp = olp->Next();
188  }
189 
190  // New overall ranges
191  if ((e = (ProcFileElements::ProcFileElement *) fElements->First())) fFirst = e->fFirst;
192  if ((e = (ProcFileElements::ProcFileElement *) fElements->Last())) fLast = e->fLast;
193 
194  // Done
195  return rc;
196 }
197 
198 //_______________________________________________________________________
199 void ProcFileElements::Print(Option_t *) const
200 {
201  // Print info about this processed file
202 
203  Printf("--- ProcFileElements ----------------------------------------");
204  Printf(" File: %s", fName.Data());
205  Printf(" # proc elements: %d", fElements ? fElements->GetSize() : 0);
206  TIter nxe(fElements);
207  ProcFileElements::ProcFileElement *e = 0;
208  while ((e = (ProcFileElements::ProcFileElement *)nxe())) { e->Print(); }
209  Printf(" Raw overall range: [%lld, %lld]", fFirst, fLast);
210  Printf("-------------------------------------------------------------");
211 }
212 
213 //_______________________________________________________________________
214 Int_t ProcFileElements::Merge(TCollection *li)
215 {
216  // Merge this object with those in the list
217  // Return number of elements added
218 
219  if (!li) return -1;
220 
221  if (li->GetSize() <= 0) return 0;
222 
223  Int_t nadd = 0;
224  TIter nxo(li);
225  ProcFileElements *pfe = 0;
226  while ((pfe = (ProcFileElements *) nxo())) {
227  if (strcmp(GetName(), pfe->GetName()))
228  Warning("Merge", "merging objects of different name! ('%s' != '%s')",
229  GetName(), pfe->GetName());
230  TIter nxe(pfe->GetListOfElements());
231  ProcFileElements::ProcFileElement *e = 0;
232  while ((e = (ProcFileElements::ProcFileElement *)nxe())) {
233  Int_t rc = Add(e->fFirst, e->fLast);
234  if (rc == 1) nadd++;
235  }
236  }
237  // Done
238  return nadd;
239 }
long long Long64_t
Definition: RtypesCore.h:69
TMatrixT< Element > & Add(TMatrixT< Element > &target, Element scalar, const TMatrixT< Element > &source)
Modify addition: target += scalar * source.
Definition: TMatrixT.cxx:2925
const char Option_t
Definition: RtypesCore.h:62
int Int_t
Definition: RtypesCore.h:41
Class to hold information about the processed elements of a file.
#define SafeDelete(p)
Definition: RConfig.h:436
A sorted doubly linked list.
Definition: TSortedList.h:30
void Error(const char *location, const char *msgfmt,...)
Collection abstract base class.
Definition: TCollection.h:48
void Warning(const char *location, const char *msgfmt,...)
#define Printf
Definition: TGeoToOCC.h:18
virtual Int_t GetSize() const
Definition: TCollection.h:95
void Print(std::ostream &os, const OptionType &opt)
Mother of all ROOT objects.
Definition: TObject.h:58
Int_t Compare(const void *item1, const void *item2)