Project
Loading...
Searching...
No Matches
RootTreeReader.h
Go to the documentation of this file.
1// Copyright 2019-2020 CERN and copyright holders of ALICE O2.
2// See https://alice-o2.web.cern.ch/copyright for details of the copyright holders.
3// All rights not expressly granted are reserved.
4//
5// This software is distributed under the terms of the GNU General Public
6// License v3 (GPL Version 3), copied verbatim in the file "COPYING".
7//
8// In applying this license CERN does not waive the privileges and immunities
9// granted to it by virtue of its status as an Intergovernmental Organization
10// or submit itself to any jurisdiction.
11#ifndef FRAMEWORK_ROOTTREEREADER_H
12#define FRAMEWORK_ROOTTREEREADER_H
13
18
20#include "Framework/Output.h"
24#include "Framework/Logger.h"
25#include "Headers/DataHeader.h"
26#include <TChain.h>
27#include <TTree.h>
28#include <TBranch.h>
29#include <TClass.h>
30#include <vector>
31#include <string>
32#include <stdexcept> // std::runtime_error
33#include <type_traits>
34#include <memory> // std::make_unique
35#include <functional> // std::function
36#include <utility> // std::forward
37
38namespace o2::framework
39{
40
41namespace rtr
42{
43// The class 'Output' has a unique header stack member which makes the class not
44// usable as key, in particular there is no copy constructor. Thats why a dedicated
45// structure is used which can be transformed into Output. Moreover it also handles the
46// conversion from OutputStack to Output
60} // namespace rtr
61
140template <typename KeyType>
142{
143 public:
145 using key_type = KeyType;
146 // The RootTreeWriter/Reader support storage of binary data as a vector of char.
147 // This allows to store e.g. raw data or some intermediate binary data alongside
148 // with ROOT objects like the MC labels.
149 using BinaryDataStoreType = std::vector<char>;
150
152 enum struct PublishingMode {
154 Single,
156 Loop,
157 };
158
169 std::function<bool(std::string_view name, ProcessingContext& context, Output const&, char* data)> hook;
170 };
171
172 // the key must not be of type const char* to make sure that the variable argument
173 // list of the constructor can be parsed
174 static_assert(std::is_same<KeyType, const char*>::value == false, "the key type must not be const char*");
175
178 ConstructorArg(key_type _key, const char* _name)
179 : key(_key), name(_name) {}
180 ConstructorArg(key_type _key, std::string const& _name)
181 : key(_key), name(_name) {}
183 std::string name;
184 };
185
186 // the vector of arguments is filled during the build up of the branch configuration and
187 // passed to the construction of the constructed mixin class
188 using ConstructorArgs = std::vector<ConstructorArg>;
189
199 {
200 public:
201 static const size_t STAGE = 0;
204 virtual ~BranchConfigurationInterface() = default;
205
207 virtual void setup(TTree&, SpecialPublishHook* h = nullptr) {}
213 virtual void exec(ProcessingContext& ctx, int entry, std::function<o2::header::Stack()> stackcreator) {}
214
215 private:
216 };
217
220 template <typename DataT, typename BASE>
222 {
223 public:
224 using PrevT = BASE;
225 using value_type = DataT;
227 static const size_t STAGE = BASE::STAGE + 1;
230 : PrevT(args), mKey(args[STAGE - 1].key), mName(args[STAGE - 1].name)
231 {
232 }
233 ~BranchConfigurationElement() override = default;
234
237 void exec(ProcessingContext& ctx, int entry, std::function<o2::header::Stack()> stackcreator) override
238 {
239 process(ctx, entry, stackcreator);
240 }
241
244 void setup(TTree& tree, SpecialPublishHook* publishhook = nullptr) override
245 {
246 setupInstance(tree, publishhook);
247 }
248
251 void setupInstance(TTree& tree, SpecialPublishHook* publishhook = nullptr)
252 {
253 // recursing through the tree structure by simply using method of the previous type,
254 // i.e. the base class method.
255 if constexpr (STAGE > 1) {
256 PrevT::setupInstance(tree, publishhook);
257 }
258 mPublishHook = publishhook;
259
260 // right now we allow the same key to appear for multiple branches
261 mBranch = tree.GetBranch(mName.c_str());
262 if (mBranch) {
263 std::string sizebranchName = std::string(mName) + "Size";
264 auto sizebranch = tree.GetBranch(sizebranchName.c_str());
265 auto* classinfo = TClass::GetClass(mBranch->GetClassName());
266 if constexpr (not std::is_void<value_type>::value) {
267 // check if the configured type matches the stored type
268 auto* storedclass = TClass::GetClass(typeid(value_type));
269 if (classinfo != storedclass) {
270 throw std::runtime_error(std::string("Configured type ") +
271 (storedclass != nullptr ? storedclass->GetName() : typeid(value_type).name()) +
272 " does not match the stored data type " +
273 (classinfo != nullptr ? classinfo->GetName() : "") +
274 " in branch " + mName);
275 }
276 }
277 if (!sizebranch) {
278 if (classinfo == nullptr) {
279 throw std::runtime_error(std::string("can not find class description for branch ") + mName);
280 }
281 LOG(info) << "branch set up: " << mName;
282 } else {
283 if (classinfo == nullptr || classinfo != TClass::GetClass(typeid(BinaryDataStoreType))) {
284 throw std::runtime_error("mismatching class type, expecting std::vector<char> for binary branch");
285 }
286 mSizeBranch = sizebranch;
287 LOG(info) << "binary branch set up: " << mName;
288 }
289 mClassInfo = classinfo;
290 } else {
291 throw std::runtime_error(std::string("can not find branch ") + mName);
292 }
293 }
294
296 void process(ProcessingContext& context, int entry, std::function<o2::header::Stack()>& stackcreator)
297 {
298 // recursing through the tree structure by simply using method of the previous type,
299 // i.e. the base class method.
300 if constexpr (STAGE > 1) {
301 PrevT::process(context, entry, stackcreator);
302 }
303
304 auto snapshot = [&context, &stackcreator](const KeyType& key, const auto& object) {
305 context.outputs().snapshot(Output{key.origin, key.description, key.subSpec, std::move(stackcreator())}, object);
306 };
307
308 char* data = nullptr;
309 mBranch->SetAddress(&data);
310 mBranch->GetEntry(entry);
311
312 // execute hook if it was registered; if this return true do not proceed further
313 if (mPublishHook != nullptr && (*mPublishHook).hook(mName, context, Output{mKey.origin, mKey.description, mKey.subSpec, std::move(stackcreator())}, data)) {
314
315 }
316 // try to figureout when we need to do something special
317 else {
318 if (mSizeBranch != nullptr) {
319 size_t datasize = 0;
320 mSizeBranch->SetAddress(&datasize);
321 mSizeBranch->GetEntry(entry);
322 auto* buffer = reinterpret_cast<BinaryDataStoreType*>(data);
323 if (buffer->size() == datasize) {
324 LOG(debug) << "branch " << mName << ": publishing binary chunk of " << datasize << " bytes(s)";
325 snapshot(mKey, std::move(*buffer));
326 } else {
327 LOG(error) << "branch " << mName << ": inconsitent size of binary chunk "
328 << buffer->size() << " vs " << datasize;
330 snapshot(mKey, empty);
331 }
332 } else {
333 if constexpr (std::is_void<value_type>::value == true) {
334 // the default branch configuration publishes the object ROOT serialized
335 snapshot(mKey, std::move(ROOTSerializedByClass(*data, mClassInfo)));
336 } else {
337 // if type is specified in the branch configuration, the allocator API decides
338 // upon serialization
339 snapshot(mKey, *reinterpret_cast<value_type*>(data));
340 }
341 }
342 }
343 // cleanup the memory
344 auto* delfunc = mClassInfo->GetDelete();
345 if (delfunc) {
346 (*delfunc)(data);
347 }
348 mBranch->DropBaskets("all");
349 }
350
351 private:
352 key_type mKey;
353 std::string mName;
354 TBranch* mBranch = nullptr;
355 TBranch* mSizeBranch = nullptr;
356 TClass* mClassInfo = nullptr;
357 SpecialPublishHook* mPublishHook = nullptr;
358 };
359
364 template <typename T>
366 using type = T;
367 template <typename U>
368 BranchDefinition(U _key, const char* _name)
369 : key(_key), name(_name)
370 {
371 }
372 template <typename U>
373 BranchDefinition(U _key, std::string const& _name)
374 : key(_key), name(_name)
375 {
376 }
377 template <typename U>
378 BranchDefinition(U _key, std::string&& _name)
379 : key(_key), name(std::move(_name))
380 {
381 }
382
384 std::string name;
385 };
386
389
394 template <typename... Args>
395 GenericRootTreeReader(const char* treename, // name of the tree to read from
396 Args&&... args) // file names, followed by branch info
397 : mInput(treename)
398 {
399 mInput.SetCacheSize(0);
400 parseConstructorArgs<0>(std::forward<Args>(args)...);
401 mBranchConfiguration->setup(mInput, mPublishHook);
402 }
403
405 void addFile(const char* fileName)
406 {
407 mInput.AddFile(fileName);
408 mNEntries = mInput.GetEntries();
409 }
410
413 bool next()
414 {
415 if ((mReadEntry + 1) >= mNEntries || mNEntries == 0) {
416 if (mPublishingMode == PublishingMode::Single) {
417 // stop here
418 if (mReadEntry < mNEntries) {
419 mReadEntry = mNEntries;
420 }
421 return false;
422 }
423 // start over in loop mode
424 mReadEntry = -1;
425 }
426 if (mMaxEntries > 0 && (mNofPublished + 1) >= mMaxEntries) {
427 if (mReadEntry < mNEntries) {
428 mReadEntry = mNEntries;
429 }
430 return false;
431 }
432 ++mReadEntry;
433 ++mNofPublished;
434 return true;
435 }
436
439 {
440 next();
441 return *this;
442 }
444 self_type& operator++(int) = delete;
445
449
457 template <typename ContextType, typename... HeaderTypes>
458 bool operator()(ContextType& context,
459 HeaderTypes&&... headers) const
460 {
461 if (mReadEntry >= mNEntries || mNEntries == 0 || (mMaxEntries > 0 && mNofPublished >= mMaxEntries)) {
462 return false;
463 }
464
465 auto stackcreator = [&headers...]() {
466 return o2::header::Stack{std::forward<HeaderTypes>(headers)...};
467 };
468
469 mBranchConfiguration->exec(context, mReadEntry, stackcreator);
470 return true;
471 }
472
474 int getCount() const
475 {
476 return mNofPublished + 1;
477 }
478
479 private:
480 // special helper to get the char argument from the argument pack
481 template <typename T, typename... Args>
482 const char* getCharArg(T arg, Args&&...)
483 {
484 static_assert(std::is_same<T, const char*>::value, "missing branch name after publishing key, use const char* argument right after the key");
485 return arg;
486 }
487
489 template <size_t skip, typename U, typename... Args>
490 void parseConstructorArgs(U key, Args&&... args)
491 {
492 // all argument parsing is done in the creation of the branch configuration
493 mBranchConfiguration = createBranchConfiguration<0, BranchConfigurationInterface>({}, std::forward<U>(key), std::forward<Args>(args)...);
494 }
495
499 template <size_t skip, typename BASE, typename U, typename... Args>
500 std::unique_ptr<BranchConfigurationInterface> createBranchConfiguration(ConstructorArgs&& cargs, U def, Args&&... args)
501 {
502 if constexpr (skip > 0) {
503 return createBranchConfiguration<skip - 1, BASE>(std::move(cargs), std::forward<Args>(args)...);
504 } else if constexpr (std::is_same<U, const char*>::value) {
505 addFile(def);
506 } else if constexpr (std::is_same<U, int>::value) {
507 mMaxEntries = def;
508 } else if constexpr (std::is_same<U, PublishingMode>::value) {
509 mPublishingMode = def;
510 } else if constexpr (std::is_same<U, SpecialPublishHook*>::value) {
511 mPublishHook = def;
512 } else if constexpr (is_specialization_v<U, BranchDefinition>) {
513 cargs.emplace_back(key_type(def.key), def.name);
514 using type = BranchConfigurationElement<typename U::type, BASE>;
515 return std::move(createBranchConfiguration<0, type>(std::move(cargs), std::forward<Args>(args)...));
516 } else if constexpr (sizeof...(Args) > 0) {
517 const char* arg = getCharArg(std::forward<Args>(args)...);
518 if (arg != nullptr && *arg != 0) {
519 cargs.emplace_back(key_type(def), arg);
520 using type = BranchConfigurationElement<void, BASE>;
521 return std::move(createBranchConfiguration<1, type>(std::move(cargs), std::forward<Args>(args)...));
522 }
523 throw std::runtime_error("expecting valid branch name string after key");
524 } else {
525 static_assert(always_static_assert<U>::value, "argument mismatch, define branches either as argument pairs of key and branchname or using the BranchDefinition helper struct");
526 }
527 return createBranchConfiguration<0, BASE>(std::move(cargs), std::forward<Args>(args)...);
528 }
529
532 template <size_t skip, typename T>
533 std::unique_ptr<BranchConfigurationInterface> createBranchConfiguration(ConstructorArgs&& cargs)
534 {
535 static_assert(skip == 0);
536 return std::move(std::make_unique<T>(cargs));
537 }
538
540 TChain mInput;
542 std::unique_ptr<BranchConfigurationInterface> mBranchConfiguration;
544 int mNEntries = 0;
546 int mReadEntry = -1;
548 int mNofPublished = -1;
550 int mMaxEntries = -1;
552 PublishingMode mPublishingMode = PublishingMode::Single;
554 SpecialPublishHook* mPublishHook = nullptr;
555};
556
558
559} // namespace o2::framework
560#endif
std::ostringstream debug
StringRef key
Class for time synchronization of RawReader instances.
void snapshot(const Output &spec, T const &object)
void process(ProcessingContext &context, int entry, std::function< o2::header::Stack()> &stackcreator)
Run the reader, first recursively for all lower stages, and then the current stage.
void exec(ProcessingContext &ctx, int entry, std::function< o2::header::Stack()> stackcreator) override
void setupInstance(TTree &tree, SpecialPublishHook *publishhook=nullptr)
void setup(TTree &tree, SpecialPublishHook *publishhook=nullptr) override
virtual void setup(TTree &, SpecialPublishHook *h=nullptr)
Setup the branch configuration, namely get the branch and class information from the tree.
virtual void exec(ProcessingContext &ctx, int entry, std::function< o2::header::Stack()> stackcreator)
bool operator()(ContextType &context, HeaderTypes &&... headers) const
self_type & operator++(int)=delete
postfix increment forbidden
void addFile(const char *fileName)
add a file as source for the tree
self_type & operator++()
prefix increment, move to the next entry
o2::framework::ROOTSerialized< char, TClass > ROOTSerializedByClass
PublishingMode
Publishing mode determines what to do when the number of entries in the tree is reached.
GenericRootTreeReader(const char *treename, Args &&... args)
GenericRootTreeReader()
default constructor
int getCount() const
return the number of published entries
std::vector< ConstructorArg > ConstructorArgs
DataAllocator & outputs()
The data allocator is used to allocate memory for the output data.
GLuint buffer
Definition glcorearb.h:655
GLuint entry
Definition glcorearb.h:5735
GLuint const GLchar * name
Definition glcorearb.h:781
GLint GLint GLsizei GLint GLenum GLenum type
Definition glcorearb.h:275
GLboolean * data
Definition glcorearb.h:298
typedef void(APIENTRYP PFNGLCULLFACEPROC)(GLenum mode)
GLuint object
Definition glcorearb.h:4041
Defining PrimaryVertex explicitly as messageable.
Definition TFIDInfo.h:20
Lifetime
Possible Lifetime of objects being exchanged by the DPL.
Definition Lifetime.h:18
Defining DataPointCompositeObject explicitly as copiable.
void empty(int)
helper structure to hold the constructor arguments for BranchConfigurationElement stages
ConstructorArg(key_type _key, const char *_name)
ConstructorArg(key_type _key, std::string const &_name)
std::function< bool(std::string_view name, ProcessingContext &context, Output const &, char *data)> hook
header::DataOrigin origin
Definition Output.h:28
header::DataHeader::SubSpecificationType subSpec
DefaultKey(const Output &desc)
header::DataDescription description
uint32_t SubSpecificationType
Definition DataHeader.h:620
a move-only header stack with serialized headers This is the flat buffer where all the headers in a m...
Definition Stack.h:36
LOG(info)<< "Compressed in "<< sw.CpuTime()<< " s"
std::unique_ptr< TTree > tree((TTree *) flIn.Get(std::string(o2::base::NameConf::CTFTREENAME).c_str()))