ATLAS Offline Software
Loading...
Searching...
No Matches
TreeBranchHelpers.cxx
Go to the documentation of this file.
1/*
2 Copyright (C) 2002-2026 CERN for the benefit of the ATLAS collaboration
3*/
4
5//
6// includes
7//
8
10
11// EDM include(s):
19
20// ROOT include(s):
21#include <TClass.h>
22#include <TTree.h>
23#include <TBranch.h>
24#include <TVirtualCollectionProxy.h>
25#include "Math/Vector4D.h"
26
27using ROOT::Math::PtEtaPhiEVector;
28using ROOT::Math::PtEtaPhiMVector;
29using ROOT::Math::PxPyPzEVector;
30using ROOT::Math::PxPyPzMVector;
31
32
33// System include(s):
34#include <regex>
35#include <algorithm>
36#include <functional>
37#include <sstream>
38
39//
40// method implementations
41//
42
43namespace {
44
45
46class TempInterface
47 : public SG::AuxVectorData
48{
49public:
50 TempInterface (size_t size) : m_size (size) {}
51 TempInterface (size_t size, SG::auxid_t auxid, void* ptr) :
52 m_size (size)
53 {
54 setCache (auxid, ptr);
55 }
56
57 using AuxVectorData::setStore;
58
59 virtual size_t size_v() const { return m_size; }
60 virtual size_t capacity_v() const { return m_size; }
61
62private:
63 size_t m_size;
64};
65
66
67} // anonymous namespace
68
69
70namespace {
71
72#ifdef XAOD_STANDALONE
73
86 const SG::AuxVectorBase* getVector( const std::string& key,
87 asg::SgEvent& evtStore,
88 bool allowMissing,
89 const TClass*& cl,
90 MsgStream& msg ) {
91 if( allowMissing &&
92 ( ! evtStore.contains< const SG::AuxVectorBase >( key ) ) ) {
93 return nullptr;
94 }
95 const SG::AuxVectorBase* c = nullptr;
96 if( ! evtStore.retrieve( c, key ).isSuccess() ) {
97 msg << MSG::ERROR << "Couldn't retrieve container with key \"" << key
98 << "\"" << endmsg;
99 return nullptr;
100 }
101 const xAOD::THolder* holder = evtStore.tds()->holder( key );
102 if( holder != nullptr ) {
103 // If the object is in the transient store, get the type of it from
104 // the transient store itself. So that ConstDataVector types would be
105 // handled correctly.
106 const std::type_info* ti = holder->getTypeInfo();
107 cl = TClass::GetClass( *ti );
108 } else {
109 // If the object is not in the transient store, let's just use its
110 // "actual type".
111 cl = TClass::GetClass( typeid( *c ) );
112 }
113 if( ( allowMissing == false ) && ( cl == nullptr ) ) {
114 msg << MSG::ERROR
115 << "Couldn't find TClass dictionary for container \"" << key
116 << "\"" << endmsg;
117 return nullptr;
118 }
119
120 // Return the vector object.
121 return c;
122 }
123
135 const SG::AuxElement* getElement( const std::string& key,
136 asg::SgEvent& evtStore,
137 bool allowMissing,
138 MsgStream& msg ) {
139 if( allowMissing &&
140 ( ! evtStore.contains< const SG::AuxElement >( key ) ) ) {
141 return nullptr;
142 }
143 const SG::AuxElement* e = nullptr;
144 if( ! evtStore.retrieve( e, key ).isSuccess() ) {
145 msg << MSG::ERROR << "Couldn't retrieve object with key \"" << key
146 << "\"" << endmsg;
147 return nullptr;
148 }
149 return e;
150 }
151
152#else
153
155 class ProxyWithName {
156 public:
158 typedef const SG::DataProxy* argument_type;
160 ProxyWithName( const std::string& name ) : m_name( name ) {}
162 bool operator()( argument_type proxy ) const {
163 return ( proxy->name() == m_name );
164 }
165 private:
166 std::string m_name;
167 }; // class ProxyWithName
168
181 const SG::AuxVectorBase* getVector ATLAS_NOT_CONST_THREAD_SAFE ( const std::string& key,
182 IProxyDict& evtStore,
183 bool allowMissing,
184 const TClass*& cl,
185 MsgStream& msg ) {
186
187 // Find all proxies with this key:
188 auto proxies = evtStore.proxies();
189 proxies.erase( std::remove_if( proxies.begin(), proxies.end(),
190 std::not_fn( ProxyWithName( key ) ) ),
191 proxies.end() );
192 // Now iterate over them:
193 for( const SG::DataProxy* proxy : proxies ) {
194 // We need a non-const version of it... :-(
195 SG::DataProxy* proxy_nc = const_cast< SG::DataProxy* >( proxy );
196 // Try to get the right object out of it.
197 DataBucketBase* bucket =
198 dynamic_cast< DataBucketBase* >( proxy_nc->accessData() );
199 if( ! bucket ) {
200 // This is a big problem in the job. Return right away.
201 msg << MSG::ERROR
202 << "Couldn't access data object as a data bucket?!?" << endmsg;
203 return nullptr;
204 }
205 // Get the dictionary for the type:
206 cl = TClass::GetClass( bucket->tinfo() );
207 if( ! cl ) {
208 if( msg.level() <= MSG::VERBOSE ) {
209 msg << MSG::VERBOSE << "No dictionary found for: "
210 << bucket->tinfo().name() << endmsg;
211 }
212 continue;
213 }
214 // Check whether the object inherits from AuxVectorBase:
215 if( ! cl->InheritsFrom( "SG::AuxVectorBase" ) ) {
216 if( msg.level() <= MSG::VERBOSE ) {
217 msg << MSG::VERBOSE << "Object \"" << key << "/" << cl->GetName()
218 << "\" does not inherit from SG::AuxVectorBase" << endmsg;
219 }
220 continue;
221 }
222 // If all is well, just assume that the inheritance is direct/simple:
224 reinterpret_cast< const SG::AuxVectorBase* >( bucket->object() );
225 return result;
226 }
227
228 // Apparently we failed...
229 if( ! allowMissing ) {
230 msg << MSG::ERROR << "Couldn't retrieve object \"" << key
231 << "\" as SG::AuxVectorBase" << endmsg;
232 }
233 return nullptr;
234 }
235
247 const SG::AuxElement* getElement ATLAS_NOT_CONST_THREAD_SAFE ( const std::string& key,
248 StoreGateSvc& evtStore,
249 bool allowMissing,
250 MsgStream& msg ) {
251
252
253 const SG::AuxElement* e = nullptr;
254 if( !evtStore.retrieve( e, key ).isSuccess() ) {
255 if(!allowMissing) {
256 msg << MSG::ERROR << "Couldn't retrieve object with key \"" << key
257 << "\"" << endmsg;
258 }
259 return nullptr;
260 }
261 return e;
262
263 }
264#endif // XAOD_STANDALONE
265
274 char rootType( char typeidType, MsgStream& msg ) {
275
276 // Do the hard-coded translation:
277 switch( typeidType ) {
278
279 case 'c':
280 return 'B';
281 break;
282 case 'h':
283 return 'b';
284 break;
285 case 's':
286 return 'S';
287 break;
288 case 't':
289 return 's';
290 break;
291 case 'i':
292 return 'I';
293 break;
294 case 'j':
295 return 'i';
296 break;
297 case 'f':
298 return 'F';
299 break;
300 case 'd':
301 return 'D';
302 break;
303 case 'x':
304 return 'L';
305 break;
306 case 'y':
307 case 'm': // Not sure how platform-independent this one is...
308 return 'l';
309 break;
310 case 'b':
311 return 'O';
312 break;
313 default:
314 // If we didn't find this type:
315 msg << MSG::ERROR << "Received an unknown type: " << typeidType
316 << endmsg;
317 return '\0';
318 break;
319 }
320 }
321} // private namespace
322
323namespace CP
324{
325 namespace TreeBranchHelpers
326 {
327 StatusCode BranchConfig ::
328 parse (const std::string& branchDecl, MsgStream& msg)
329 {
330 // The regular expression used to extract the needed info. The logic
331 // is supposed to be:
332 //
333 // (match[1]).(match[2])<any whitespace>-><any whitespace>(match[3])[<any whitespace>type=(match[5])][<any whitespace>metTerm=(match[7])][<any whitespace>basketSize=(match[9])]
334 //
335 // Like:
336 // "Electrons.eta -> el_eta"
337 // "Electrons.eta -> el_eta type=float"
338 // "MissingET.px -> met_px metTerm=Final"
339 static const std::regex
340 re( "\\s*([\\w%]+)\\.([\\w%]+)\\s*->\\s*([\\w%]+)(\\s+type=([\\w%]+))?(\\s+metTerm=([\\w%]+))?(\\s+basketSize=([\\w%]+))?" );
341
342 // Interpret this branch declaration.
343 std::smatch match;
344 if( ! std::regex_match( branchDecl, match, re ) ) {
345 msg << MSG::ERROR << "Expression \"" << branchDecl << "\" doesn't match \"<object>.<variable> -> <branch>\"" << endmsg;
346 return StatusCode::FAILURE;
347 }
348 this->branchDecl = branchDecl;
349 sgName = match[ 1 ];
350 auxName = match[ 2 ];
351 branchName = match[ 3 ];
352 typeName = match[ 5 ];
353 metTermName = match[ 7 ];
354 if (match[9].matched) {
355 try {
356 basketSize = std::stoi(match[9]);
357 } catch (const std::exception& ) {
358 msg << MSG::ERROR << "Could not parse basket size value: " << match[9] << endmsg;
359 return StatusCode::FAILURE;
360 }
361 }
362 return StatusCode::SUCCESS;
363 }
364
365
366
367 StatusCode BranchConfig ::
368 configureTypes (std::set<std::string>& decosWithoutType, MsgStream& msg)
369 {
370 std::string nominalAuxName = auxName;
371 if (auto pos = nominalAuxName.find ("%SYS%"); pos != std::string::npos)
372 nominalAuxName.replace (pos, 5, "NOSYS");
373 if (!typeName.empty())
374 {
375 if (typeName == "char")
376 SG::ConstAccessor<char> {nominalAuxName};
377 else if (typeName == "float")
378 SG::ConstAccessor<float> {nominalAuxName};
379 else if (typeName == "double")
380 SG::ConstAccessor<double> {nominalAuxName};
381 else if (typeName == "int")
382 SG::ConstAccessor<int> {nominalAuxName};
383 else if (typeName == "unsigned")
384 SG::ConstAccessor<unsigned> {nominalAuxName};
385 else if (typeName == "unsigned_char")
386 SG::ConstAccessor<unsigned char> {nominalAuxName};
387 else if (typeName == "unsigned_long")
388 SG::ConstAccessor<unsigned long> {nominalAuxName};
389 else if (typeName == "unsigned_long_long")
391 else if (typeName == "int8")
392 SG::ConstAccessor<std::int8_t> {nominalAuxName};
393 else if (typeName == "int16")
394 SG::ConstAccessor<std::int16_t> {nominalAuxName};
395 else if (typeName == "int32")
396 SG::ConstAccessor<std::int32_t> {nominalAuxName};
397 else if (typeName == "int64")
398 SG::ConstAccessor<std::int64_t> {nominalAuxName};
399 else if (typeName == "uint8")
400 SG::ConstAccessor<std::uint8_t> {nominalAuxName};
401 else if (typeName == "uint16")
402 SG::ConstAccessor<std::uint16_t> {nominalAuxName};
403 else if (typeName == "uint32")
404 SG::ConstAccessor<std::uint32_t> {nominalAuxName};
405 else if (typeName == "uint64")
406 SG::ConstAccessor<std::uint64_t> {nominalAuxName};
407 else if (typeName == "vector_float")
408 SG::ConstAccessor<std::vector<float>> {nominalAuxName};
409 else if (typeName == "vector_int")
410 SG::ConstAccessor<std::vector<int>> {nominalAuxName};
411 else if (typeName == "vector_vector_float")
413 else if (typeName == "vector_vector_int")
415 else if (typeName == "PtEtaPhiEVector")
416 SG::ConstAccessor<PtEtaPhiEVector> {nominalAuxName};
417 else if (typeName == "PtEtaPhiMVector")
418 SG::ConstAccessor<PtEtaPhiMVector> {nominalAuxName};
419 else if (typeName == "PxPyPzEVector")
420 SG::ConstAccessor<PxPyPzEVector> {nominalAuxName};
421 else if (typeName == "PxPyPzMVector")
422 SG::ConstAccessor<PxPyPzMVector> {nominalAuxName};
423 else if (typeName == "vector_PtEtaPhiEVector")
425 else if (typeName == "vector_PtEtaPhiMVector")
427 else if (typeName == "vector_PxPyPzEVector")
429 else if (typeName == "vector_PxPyPzMVector")
431 else if (typeName == "vector_vector_PtEtaPhiEVector")
433 else if (typeName == "vector_vector_PtEtaPhiMVector")
435 else if (typeName == "vector_vector_PxPyPzEVector")
437 else if (typeName == "vector_vector_PxPyPzMVector")
439 else
440 {
441 unsigned line = __LINE__ - 2;
442 std::string file = __FILE__;
443 file = file.substr (file.find_last_of("/\\") + 1);
444 msg << MSG::ERROR << "Unknown type requested, please extend " << file << " near line " << line << " for type " << typeName << endmsg;
445 return StatusCode::FAILURE;
446 }
447 }
450 {
451 nominalAuxId = reg.findAuxID (nominalAuxName);
453 {
454 decosWithoutType.insert (nominalAuxName);
455 msg << MSG::DEBUG << "No aux ID found for auxiliary variable: " << nominalAuxName << endmsg;
456 // just returning SUCCESS here, our caller will report failure
457 return StatusCode::SUCCESS;
458 }
459 }
460 if (auxType == nullptr)
461 {
462 auxType = reg.getType (nominalAuxId);
463 if (auxType == nullptr)
464 {
465 msg << MSG::ERROR
466 << "No std::type_info available for aux-store variable: "
467 << nominalAuxName << endmsg;
468 return StatusCode::FAILURE;
469 }
470 }
471 if (auxVecType == nullptr)
472 {
473 auxVecType = reg.getVecType (nominalAuxId);
474 if (auxVecType == nullptr)
475 {
476 msg << MSG::ERROR
477 << "No std::type_info available for aux-store variable: "
478 << nominalAuxName << endmsg;
479 return StatusCode::FAILURE;
480 }
481 }
482 if (auxFactory == nullptr)
483 {
484 auxFactory = reg.getFactory (nominalAuxId);
485 if (auxFactory == nullptr)
486 {
487 msg << MSG::ERROR
488 << "No factory found for auxiliary variable: "
489 << nominalAuxName << endmsg;
490 return StatusCode::FAILURE;
491 }
492 }
493 return StatusCode::SUCCESS;
494 }
495
496
497
498 StatusCode BranchConfig ::
499 configureSystematics (ISystematicsSvc& sysSvc, MsgStream& msg)
500 {
501 if (sgName.find ("%SYS%") == std::string::npos &&
502 auxName.find ("%SYS%") == std::string::npos &&
503 branchName.find ("%SYS%") == std::string::npos)
504 {
505 nominalOnly = true;
506 }
507 if (!nominalOnly)
508 {
509 if (branchName.find ("%SYS%") == std::string::npos)
510 {
511 msg << MSG::ERROR << "Branch with systematics without %SYS% in branch name: "
512 << branchName << endmsg;
513 return StatusCode::FAILURE;
514 }
515 if (sgName.find ("%SYS%") == std::string::npos &&
516 auxName.find ("%SYS%") == std::string::npos)
517 {
518 msg << MSG::ERROR << "Branch with systematics without %SYS% in SG or aux name: "
519 << sgName << "." << auxName << endmsg;
520 return StatusCode::FAILURE;
521 }
522 if (auxName.find ("NOSYS") != std::string::npos)
523 {
524 msg << MSG::ERROR << "Branch with systematics with NOSYS in aux name: "
525 << sgName << "." << auxName << endmsg;
526 return StatusCode::FAILURE;
527 }
528 if (sgName.find ("NOSYS") != std::string::npos && auxName.find ("%SYS%") == std::string::npos)
529 {
530 msg << MSG::ERROR << "Branch with NOSYS in SG name but without %SYS% in aux name: "
531 << sgName << "." << auxName << endmsg;
532 return StatusCode::FAILURE;
533 }
534
535 if (sgName.find ("%SYS%") != std::string::npos)
537
538 if (auxName.find ("%SYS%") != std::string::npos)
539 {
540 if (auto pos = sgName.find ("NOSYS"); pos == std::string::npos)
542 else
543 {
544 // Sometimes while object systematics were applied we are not interested in them,
545 // NOSYS will then be used on the container name.
546 // Decoration systematics however will only be aware of containers with %SYS% included.
547 // Some special handling is needed to translate from NOSYS back to %SYS%.
548 std::string sgNameSys = sgName;
549 sgNameSys.replace (pos, 5, "%SYS%");
550
551 // these will be the object systematics
552 auto objectSys = sysSvc.getObjectSystematics (sgNameSys);
553
554 // these will be all systematics (object+decor)
555 auto allSys = sysSvc.getDecorSystematics (sgNameSys, auxName);
556
557 // we now need to filter-out object systematics
558 for (auto& variation : allSys)
559 {
560 if (objectSys.find (variation) == objectSys.end())
561 auxNameFilterSys.insert (variation);
562 }
563 }
564 }
565
568 if (branchNameFilterSys.empty())
569 nominalOnly = true;
570 }
571
572 return StatusCode::SUCCESS;
573 }
574
575
576
577 StatusCode OutputBranchData ::
578 configureNames (const BranchConfig& branchConfig, const CP::SystematicSet& sys, ISystematicsSvc& sysSvc, MsgStream& msg)
579 {
580 isNominal = true;
581
582 if (branchConfig.sgName.find ("%SYS%") != std::string::npos)
583 {
584 CP::SystematicSet matching;
585 if (SystematicSet::filterForAffectingSystematics (sys, branchConfig.sgNameFilterSys, matching).isFailure())
586 return StatusCode::FAILURE;
587 if (sysSvc.makeSystematicsName (sgName, branchConfig.sgName, matching).isFailure())
588 return StatusCode::FAILURE;
589 if (!matching.empty())
590 isNominal = false;
591 } else
592 sgName = branchConfig.sgName;
593
594 if (branchConfig.auxName.find ("%SYS%") != std::string::npos)
595 {
596 CP::SystematicSet matching;
597 if (SystematicSet::filterForAffectingSystematics (sys, branchConfig.auxNameFilterSys, matching).isFailure())
598 return StatusCode::FAILURE;
599 if (sysSvc.makeSystematicsName (auxName, branchConfig.auxName, matching).isFailure())
600 return StatusCode::FAILURE;
601 if (!matching.empty())
602 isNominal = false;
603 } else
604 auxName = branchConfig.auxName;
605
606 if (branchConfig.branchName.find ("%SYS%") != std::string::npos)
607 {
608 CP::SystematicSet matching;
609 if (SystematicSet::filterForAffectingSystematics (sys, branchConfig.branchNameFilterSys, matching).isFailure())
610 return StatusCode::FAILURE;
611 if (sysSvc.makeSystematicsName (branchName, branchConfig.branchName, matching).isFailure())
612 return StatusCode::FAILURE;
613 if (matching.empty() && !isNominal)
614 {
615 msg << MSG::FATAL << "Branch \"" << branchName << "\" is not affected by any of the requested systematics but is not nominal." << endmsg;
616 return StatusCode::FAILURE;
617 }
618 } else
619 {
620 branchName = branchConfig.branchName;
621 if (!sys.empty())
622 {
623 msg << MSG::FATAL << "Branch \"" << branchName << "\" without systematics is evaluated in a non-nominal context." << endmsg;
624 return StatusCode::FAILURE;
625 }
626 }
627
628 return StatusCode::SUCCESS;
629 }
630
631
632
633 StatusCode
635 setup( TTree& tree, const BranchConfig& branchConfig, OutputBranchData& outputData, MsgStream& msg ) {
636
637 // Remember the branch name.
638 m_branchName = outputData.branchName;
639
640 // Create the accessor.
641 m_acc.reset( new SG::TypelessConstAccessor( *branchConfig.auxType, outputData.auxName ) );
642
643 // Get a pointer to the vector factory.
644 m_factory = branchConfig.auxFactory;
645
646 // Create the data object.
647 m_data = m_factory->create( m_acc->auxid(), 1, 1, false );
648
649 // Pointer to the branch, to be created.
650 TBranch* br = nullptr;
651
652 // Decide whether we're dealing with a "primitive" or an "object" branch.
653 if( strlen( branchConfig.auxType->name() ) == 1 ) {
654
655 // This is a "primitive" variable...
656
657 // Get the type identifier for it that ROOT will understand.
658 const char rType = rootType( branchConfig.auxType->name()[ 0 ], msg );
659 if( rType == '\0' ) {
660 msg << MSG::ERROR << "Type not recognised for variable: "
661 << outputData.branchName << endmsg;
662 return StatusCode::FAILURE;
663 }
664
665 // Construct the type description.
666 std::ostringstream typeDesc;
667 typeDesc << outputData.branchName << "/" << rType;
668
669 // Create the primitive branch.
670 br = tree.Branch( outputData.branchName.c_str(), m_data->toPtr(),
671 typeDesc.str().c_str() );
672 if (branchConfig.basketSize.has_value())
673 br->SetBasketSize(branchConfig.basketSize.value());
674
675 } else {
676
677 // This is an "object" variable...
678
679 // Get a proper type name for the variable.
680 const std::string typeName = SG::normalizedTypeinfoName( *branchConfig.auxType );
681
682 // Access the dictionary for the type.
683 TClass* cl = TClass::GetClass( *branchConfig.auxType );
684 if( ! cl ) {
685 cl = TClass::GetClass( typeName.c_str() );
686 }
687 if( ! cl ) {
688 msg << MSG::ERROR << "Couldn't find dictionary for type: "
689 << typeName << endmsg;
690 return StatusCode::FAILURE;
691 }
692 if( ! cl->GetStreamerInfo() ) {
693 msg << MSG::ERROR << "No streamer info available for type: "
694 << cl->GetName() << endmsg;
695 return StatusCode::FAILURE;
696 }
697
698 // Create the object branch.
699 m_dataPtr = m_data->toPtr();
700 br = tree.Branch( outputData.branchName.c_str(), cl->GetName(), &m_dataPtr );
701 if (branchConfig.basketSize.has_value())
702 br->SetBasketSize(branchConfig.basketSize.value());
703
704 }
705
706 // Check that the branch creation succeeded.
707 if( ! br ) {
708 msg << MSG::ERROR << "Failed to create branch: " << outputData.branchName
709 << endmsg;
710 return StatusCode::FAILURE;
711 }
712
713 // Return gracefully.
714 return StatusCode::SUCCESS;
715 }
716
717 StatusCode
719 process( const SG::AuxElement& element, MsgStream& msg ) {
720
721 // A security check.
722 if( ( ! m_acc ) || ( ! m_factory ) || ( ! m_data ) ) {
723 msg << MSG::FATAL << "Internal logic error detected" << endmsg;
724 return StatusCode::FAILURE;
725 }
726
727 // Get the data out of the xAOD object.
728 //const void* auxData = ( *m_acc )( element );
729
730 // Copy it into the output variable.
731 TempInterface dstiface (m_data->size(), m_acc->auxid(), m_data->toPtr());
732 m_factory->copy( m_acc->auxid(), dstiface, 0,
733 *element.container(), element.index(), 1 );
734
735 // Return gracefully.
736 return StatusCode::SUCCESS;
737 }
738
740 setup ( ROOT::RNTupleModel& /* model */, const BranchConfig& /* branchConfig */, OutputBranchData& /* outputData */, MsgStream& msg ) {
741 msg << MSG::ERROR << "ElementBranchProcessor::setup for RNTuple should not be called" << endmsg;
742 return StatusCode::FAILURE;
743 }
744
746 setup( TTree& tree, const BranchConfig& branchConfig, OutputBranchData& outputData, MsgStream& msg ) {
747
748 // Remember the branch name.
749 m_branchName = outputData.branchName;
750
751 // Create the accessor.
752 m_acc.reset( new SG::TypelessConstAccessor( *branchConfig.auxType, outputData.auxName ) );
753
754 // Get a pointer to the vector factory.
755 m_factory = branchConfig.auxFactory;
756
757 // Create the data object.
758 m_data = m_factory->create( m_acc->auxid(), 0, 0, false );
759
760 // Get a proper type name for the variable.
761 const std::string typeName = SG::normalizedTypeinfoName( *branchConfig.auxVecType );
762
763 // Access the dictionary for the type.
764 TClass* cl = TClass::GetClass( *branchConfig.auxVecType );
765 if( ! cl ) {
766 cl = TClass::GetClass( typeName.c_str() );
767 }
768 if( ! cl ) {
769 msg << MSG::ERROR << "Couldn't find dictionary for type: "
770 << typeName << endmsg;
771 return StatusCode::FAILURE;
772 }
773 if( ! cl->GetStreamerInfo() ) {
774 msg << MSG::ERROR << "No streamer info available for type: "
775 << cl->GetName() << endmsg;
776 return StatusCode::FAILURE;
777 }
778
779 // Create the branch.
780 m_dataPtr = m_data->toVector();
781 TBranch* br = tree.Branch( outputData.branchName.c_str(), cl->GetName(),
782 &m_dataPtr );
783 if( ! br ) {
784 msg << MSG::ERROR << "Failed to create branch: " << outputData.branchName
785 << endmsg;
786 return StatusCode::FAILURE;
787 }
788 if (branchConfig.basketSize.has_value())
789 br->SetBasketSize(branchConfig.basketSize.value());
790
791 // Return gracefully.
792 return StatusCode::SUCCESS;
793 }
794
796 resize( size_t size, MsgStream& msg ) {
797
798 // A security check.
799 if( ! m_data ) {
800 msg << MSG::FATAL << "Internal logic error detected" << endmsg;
801 return StatusCode::FAILURE;
802 }
803
804 // Do the deed.
805 m_data->resize( 0 );
806 m_data->resize( size );
807
808 // Return gracefully.
809 return StatusCode::SUCCESS;
810 }
811
813 process( const SG::AuxElement& element, size_t index, MsgStream& msg ) {
814
815 // A security check.
816 if( ( ! m_acc ) || ( ! m_factory ) || ( ! m_data ) ) {
817 msg << MSG::FATAL << "Internal logic error detected" << endmsg;
818 return StatusCode::FAILURE;
819 }
820
821 // Get the data out of the xAOD object.
822 //const void* auxData = ( *m_acc )( element );
823
824 // Copy it into the output variable.
825 TempInterface dstiface (m_data->size(), m_acc->auxid(), m_data->toPtr());
826 m_factory->copy( m_acc->auxid(), dstiface, index,
827 *element.container(), element.index(), 1 );
828
829 // Return gracefully.
830 return StatusCode::SUCCESS;
831 }
832
834 setup ( ROOT::RNTupleModel& /* model */, const BranchConfig& /* branchConfig */, OutputBranchData& /* outputData */, MsgStream& msg ) {
835 msg << MSG::ERROR << "ContainerBranchProcessor::setup for RNTuple should not be called" << endmsg;
836 return StatusCode::FAILURE;
837 }
838
839
840
842 : asg::AsgMessaging( ("CP::TreeBranchHelpers::ElementProcessorRegular/" + sgName).c_str() ),
843 m_sgName(sgName) {
844
845 }
846
848 retrieveProcess( StoreType& evtStore ) {
849
850 // Retrieve the object:
851 static const bool ALLOW_MISSING = false;
852 const SG::AuxElement* el = getElement( m_sgName,
853 evtStore,
854 ALLOW_MISSING, msg() );
855 if( ! el ) {
856 ATH_MSG_ERROR( "Failed to retrieve object \"" << m_sgName
857 << "\"" );
858 return StatusCode::FAILURE;
859 }
860 const SG::AuxElement& element = *el;
861
862 // Process all branches.
863 for( auto& p : m_branches ) {
864 ATH_CHECK( p->process( element, msg() ) );
865 }
866
867 // Return gracefully.
868 return StatusCode::SUCCESS;
869 }
870
872 addBranch( TTree& tree, const BranchConfig& branchConfig, OutputBranchData& outputData ) {
873
874 // Set up the new branch.
875 m_branches.emplace_back(std::make_unique<ElementBranchProcessor>());
876 ATH_CHECK( m_branches.back()->setup( tree, branchConfig, outputData, msg() ) );
877
878 // Return gracefully.
879 return StatusCode::SUCCESS;
880 }
881
883 addBranch( ROOT::RNTupleModel& /* model */, const BranchConfig& /* branchConfig */, OutputBranchData& /* outputData */ ) {
884 ATH_MSG_ERROR("ElementProcessorRegular::addBranch for RNTuple should not be called");
885 return StatusCode::FAILURE;
886 }
887
889 : asg::AsgMessaging( ("CP::TreeBranchHelpers::ContainerProcessorRegular/" + sgName).c_str() ),
890 m_sgName(sgName) {
891
892 }
893
895 retrieveProcess( StoreType& evtStore ) {
896
897 // Retrieve the container:
898 static const bool ALLOW_MISSING = false;
899 const TClass* cl = nullptr;
900 const SG::AuxVectorBase* vec = getVector( m_sgName,
901 evtStore,
902 ALLOW_MISSING, cl, msg() );
903 if( ! vec ) {
904 ATH_MSG_ERROR( "Failed to retrieve container \""
905 << m_sgName << "\"" );
906 return StatusCode::FAILURE;
907 }
908 const SG::AuxVectorBase& container = *vec;
909
910 // Get the collection proxy for the type if it's not available yet.
911 if( ! m_collProxy ) {
912
913 // Get the collection proxy from the dictionary.
914 m_collProxy = cl->GetCollectionProxy();
915 if( ! m_collProxy ) {
916 ATH_MSG_ERROR( "No collection proxy provided by type: "
917 << cl->GetName() );
918 return StatusCode::FAILURE;
919 }
920
921 // Get the offset that one needs to use to get from the element
922 // pointers to SG::AuxElement pointers.
923 static const TClass* const auxElementClass =
924 TClass::GetClass( typeid( SG::AuxElement ) );
926 m_collProxy->GetValueClass()->GetBaseClassOffset( auxElementClass );
927 if( m_auxElementOffset < 0 ) {
928 ATH_MSG_ERROR( "Vector element type \""
929 << m_collProxy->GetValueClass()->GetName()
930 << "\" doesn't seem to inherit from \""
931 << auxElementClass->GetName() << "\"" );
932 return StatusCode::FAILURE;
933 }
934 }
935
936 // Set up the iteration over the elements of the container. In a really
937 // low level / ugly way...
938 void* cPtr =
939 const_cast< void* >( static_cast< const void* >( &container ) );
940 TVirtualCollectionProxy::TPushPop helper( m_collProxy, cPtr );
941 const UInt_t cSize = m_collProxy->Size();
942
943 // Tell all branch processors to resize their variables.
944 for( auto& p : m_branches ) {
945 ATH_CHECK( p->resize( cSize, msg() ) );
946 }
947
948 // Now iterate over the container.
949 for( UInt_t i = 0; i < cSize; ++i ) {
950
951 // Get the element.
952 char* elPtr = static_cast< char* >( m_collProxy->At( i ) );
953 if( ! elPtr ) {
954 ATH_MSG_ERROR( "Failed to get element " << i << " from container" );
955 return StatusCode::FAILURE;
956 }
957 const SG::AuxElement* element =
958 reinterpret_cast< const SG::AuxElement* >( elPtr +
960
961 // Execute all branch processors on this element.
962 for( auto& p : m_branches ) {
963 ATH_CHECK( p->process( *element, i, msg() ) );
964 }
965 }
966
967 // Return gracefully.
968 return StatusCode::SUCCESS;
969 }
970
972 addBranch( TTree& tree, const BranchConfig& branchConfig, OutputBranchData& outputData ) {
973
974 // Set up the new branch.
975 m_branches.emplace_back(std::make_unique<ContainerBranchProcessor>());
976 ATH_CHECK( m_branches.back()->setup( tree, branchConfig, outputData, msg() ) );
977
978 // Return gracefully.
979 return StatusCode::SUCCESS;
980 }
981
983 addBranch( ROOT::RNTupleModel& /* model */, const BranchConfig& /* branchConfig */, OutputBranchData& /* outputData */ ) {
984 ATH_MSG_ERROR("ContainerProcessorRegular::addBranch for RNTuple should not be called");
985 return StatusCode::FAILURE;
986 }
987
988 ElementProcessorMet::ElementProcessorMet (const std::string& sgName, const std::string& termName)
989 : asg::AsgMessaging( ("CP::TreeBranchHelpers::ElementProcessorMet/" + sgName).c_str() ),
990 m_sgName(sgName),
991 m_termName(termName) {
992
993 }
994
996 retrieveProcess( StoreType& evtStore ) {
997
998 const xAOD::MissingETContainer *met = nullptr;
999 ANA_CHECK (evtStore.retrieve (met, m_sgName));
1000 const SG::AuxElement& element = *(*met)[m_termName];
1001 // Process all branches.
1002 for( auto& p : m_branches ) {
1003 ATH_CHECK( p->process( element, msg() ) );
1004 }
1005
1006 // Return gracefully.
1007 return StatusCode::SUCCESS;
1008 }
1009
1011 addBranch( TTree& tree, const BranchConfig& branchConfig, OutputBranchData& outputData ) {
1012
1013 // Set up the new branch.
1014 m_branches.emplace_back(std::make_unique<ElementBranchProcessor>());
1015 ATH_CHECK( m_branches.back()->setup( tree, branchConfig, outputData, msg() ) );
1016
1017 // Return gracefully.
1018 return StatusCode::SUCCESS;
1019 }
1020
1022 addBranch( ROOT::RNTupleModel& /* model */, const BranchConfig& /* branchConfig */, OutputBranchData& /* outputData */ ) {
1023 ATH_MSG_ERROR("ElementProcessorMet::addBranch for RNTuple should not be called");
1024 return StatusCode::FAILURE;
1025 }
1026
1027
1028
1029 StatusCode ProcessorList ::
1030 setupTree(const std::vector<std::string>& branches, std::unordered_set<std::string> nonContainers, ISystematicsSvc& sysSvc, TTree& tree) {
1031
1032 m_nonContainers = std::move (nonContainers);
1033
1034 std::vector<BranchConfig> branchConfigs;
1035 branchConfigs.reserve( branches.size() );
1036 for ( const std::string& branchDecl : branches ) {
1037 branchConfigs.emplace_back();
1038 ATH_CHECK( branchConfigs.back().parse( branchDecl, msg() ) );
1039 if (!branchConfigs.back().basketSize.has_value())
1040 branchConfigs.back().basketSize = defaultBasketSize;
1041 }
1042
1043 // This will loop over all branches, collect the name of any
1044 // aux-store decorations that have no type and report them at the
1045 // end. This allows to get the full list of missing decorations in
1046 // a single run, as opposed to having to re-run the job once per
1047 // missing decoration.
1048 std::set<std::string> decosWithoutType;
1049 for (auto& branchConfig : branchConfigs) {
1050 ATH_CHECK ( branchConfig.configureTypes (decosWithoutType, msg()) );
1051 }
1052 if (!decosWithoutType.empty()) {
1053 msg() << MSG::ERROR << "The following decorations have no type information:";
1054 for (const auto& deco : decosWithoutType) {
1055 msg() << " " << deco;
1056 }
1057 msg() << endmsg;
1058 return StatusCode::FAILURE;
1059 }
1060
1061
1062 for (auto& branchConfig : branchConfigs) {
1063 ATH_CHECK ( branchConfig.configureSystematics (sysSvc, msg()) );
1064 }
1065
1066 auto sysVector = sysSvc.makeSystematicsVector();
1067 // Ensure that the nominal systematic is first
1068 if (!sysVector.at(0).empty()) {
1069 ATH_MSG_ERROR ("The first systematic in the list is not nominal!");
1070 return StatusCode::FAILURE;
1071 }
1072
1073 // The branches we intend to write out
1074 std::vector<OutputBranchData> outputBranches;
1075
1076 // All the branches that will be created
1077 std::unordered_set<std::string> allBranches;
1078
1079 // Iterate over the branch specifications.
1080 for( const auto& branchConfig : branchConfigs ) {
1081
1082 // All the branches that will be created for this rule
1083 std::unordered_set<std::string> branchesForRule;
1084
1085 // Consider all systematics but skip the nominal one
1086 for( const auto& sys : sysVector ) {
1087
1088 if (branchConfig.nominalOnly && !sys.empty()) continue;
1089 OutputBranchData outputData;
1090 outputData.branchConfig = &branchConfig;
1091 outputData.sysIndex = &sys - &sysVector.front();
1092 ATH_CHECK( outputData.configureNames (branchConfig, sys, sysSvc, msg()) );
1093
1094 // Skip branches that have already been created for other
1095 // systematics for this rule. That's mostly nominal, but for
1096 // systematics correlation studies it can also do other things.
1097 if (branchesForRule.contains(outputData.branchName))
1098 {
1099 ANA_MSG_VERBOSE ("Branch \"" << outputData.branchName << "\" for rule \"" << branchConfig.branchDecl << "\" and systematic \"" << sys.name() << "\" already exists, skipping." );
1100 continue;
1101 }
1102 branchesForRule.insert(outputData.branchName);
1103
1104 // If this branch already exists from another rule, report
1105 // it as an error.
1106 if (allBranches.contains(outputData.branchName))
1107 {
1108 ANA_MSG_ERROR ("Branch \"" << outputData.branchName << "\" would be created twice!" );
1109 return StatusCode::FAILURE;
1110 }
1111 allBranches.insert(outputData.branchName);
1112 outputBranches.push_back(std::move(outputData));
1113 }
1114 }
1115
1116 // Group all branches by systematic index to ensure that when
1117 // reading a single systematic the branches are contiguous on
1118 // disk.
1119 std::stable_sort (outputBranches.begin(), outputBranches.end(),
1120 [](const OutputBranchData& a, const OutputBranchData& b) {
1121 return a.sysIndex < b.sysIndex; });
1122
1123 for (auto &outputData : outputBranches)
1124 ATH_CHECK( setupBranch( *outputData.branchConfig, outputData, tree ) );
1125
1126 // Return gracefully.
1127 return StatusCode::SUCCESS;
1128 }
1129
1130 StatusCode ProcessorList::setupBranch( const BranchConfig& branchConfig, OutputBranchData& outputData, TTree& tree ) {
1131
1132 ATH_CHECK( getObjectProcessor( branchConfig, outputData.sgName ).addBranch( tree,
1133 branchConfig, outputData ) );
1134 ATH_MSG_DEBUG( "Writing branch \"" << outputData.branchName
1135 << "\" from container/variable \"" << outputData.sgName
1136 << "." << outputData.auxName << "\"" );
1137
1138 // Return gracefully.
1139 return StatusCode::SUCCESS;
1140 }
1141
1142 StatusCode ProcessorList ::
1143 process (StoreType& evtStore)
1144 {
1145 // Process the standalone objects:
1146 for( auto& [name, processor] : m_processors )
1147 {
1148 // Process it:
1149 ATH_CHECK (processor->retrieveProcess (evtStore));
1150 }
1151 return StatusCode::SUCCESS;
1152 }
1153
1154
1155
1156 IObjectProcessor& ProcessorList ::
1157 getObjectProcessor( const BranchConfig& branchConfig, const std::string& sgName )
1158 {
1159 std::string processorName = sgName;
1160 if (!branchConfig.metTermName.empty())
1161 processorName += ":metTerm=" + branchConfig.metTermName;
1162
1163 if (auto iter = m_processors.find(processorName); iter != m_processors.end())
1164 return *iter->second;
1165
1166 if (!branchConfig.metTermName.empty())
1167 return *m_processors.emplace (processorName, std::make_unique<ElementProcessorMet>(sgName, branchConfig.metTermName)).first->second;
1168
1169 if (m_nonContainers.contains(sgName))
1170 return *m_processors.emplace (processorName, std::make_unique<ElementProcessorRegular>(sgName)).first->second;
1171
1172 return *m_processors.emplace (processorName, std::make_unique<ContainerProcessorRegular>(sgName)).first->second;
1173 }
1174 }
1175}
const std::regex re(r_e)
#define endmsg
#define ATH_CHECK
Evaluate an expression and check for errors.
#define ATH_MSG_ERROR(x)
#define ATH_MSG_DEBUG(x)
Base class for elements of a container that can have aux data.
Manage index tracking and synchronization of auxiliary data.
std::vector< size_t > vec
#define ANA_MSG_ERROR(xmsg)
Macro printing error messages.
#define ANA_MSG_VERBOSE(xmsg)
Macro printing verbose messages.
#define ANA_CHECK(EXP)
check whether the given expression was successful
Interface providing I/O for a generic auxiliary store.
Interface for factory objects that create vectors.
static Double_t a
static HEPVis_BooleanProcessor processor
void operator()(T1)
size_t size() const
Number of registered mappings.
#define ATLAS_NOT_CONST_THREAD_SAFE
the interface for the central systematics service
virtual CP::SystematicSet getObjectSystematics(const std::string &name) const =0
get the systematics for the given object in the event store
virtual std::vector< CP::SystematicSet > makeSystematicsVector() const =0
get the list of systematics
virtual CP::SystematicSet getDecorSystematics(const std::string &objectName, const std::string &decorName) const =0
get the systematics for the given object in the event store
virtual StatusCode makeSystematicsName(std::string &result, const std::string &name, const CP::SystematicSet &sys) const =0
make the name for the given systematics
Class to wrap a set of SystematicVariations.
bool empty() const
returns: whether the set is empty
static StatusCode filterForAffectingSystematics(const SystematicSet &systConfig, const SystematicSet &affectingSystematics, SystematicSet &filteredSystematics)
description: filter the systematics for the affected systematics returns: success guarantee: strong f...
std::string m_branchName
Name of the branch being written.
StatusCode resize(size_t size, MsgStream &msg)
Function (re)sizing the variable for a new event.
std::unique_ptr< SG::IAuxTypeVector > m_data
The object managing the memory of the written variable.
StatusCode process(const SG::AuxElement &element, size_t index, MsgStream &msg)
Function processing the object, filling the variable.
std::unique_ptr< SG::TypelessConstAccessor > m_acc
Object accessing the variable in question.
virtual StatusCode setup(TTree &tree, const BranchConfig &branchConfig, OutputBranchData &outputData, MsgStream &msg) override
Function setting up the object, and the branch.
const SG::IAuxTypeVectorFactory * m_factory
Pointer to the helper object that handles this variable.
void * m_dataPtr
Helper variable, pointing at the object to be written.
TVirtualCollectionProxy * m_collProxy
Collection proxy used for iterating over the container.
std::string m_sgName
Name of the object in the event store.
int m_auxElementOffset
Offset of the element type to SG::AuxElement.
virtual StatusCode retrieveProcess(StoreType &evtStore) override
retrieve and process the object
virtual StatusCode addBranch(TTree &tree, const BranchConfig &branchConfig, OutputBranchData &outputData) override
Add one branch to the output tree.
std::vector< std::unique_ptr< ContainerBranchProcessor > > m_branches
List of branch processors set up for this xAOD object.
ContainerProcessorRegular(const std::string &sgName)
Default constructor.
std::unique_ptr< SG::IAuxTypeVector > m_data
The object managing the memory of the written variable.
void * m_dataPtr
Helper variable, pointing at the object to be written.
const SG::IAuxTypeVectorFactory * m_factory
Pointer to the helper object that handles this variable.
std::string m_branchName
Name of the branch being written.
virtual StatusCode setup(TTree &tree, const BranchConfig &branchConfig, OutputBranchData &outputData, MsgStream &msg) override
Function setting up the object, and the branch.
std::unique_ptr< SG::TypelessConstAccessor > m_acc
Object accessing the variable in question.
StatusCode process(const SG::AuxElement &element, MsgStream &msg)
Function processing the object, filling the variable.
std::vector< std::unique_ptr< ElementBranchProcessor > > m_branches
List of branch processors set up for this xAOD object.
std::string m_termName
Name of the MET term to retrieve.
virtual StatusCode addBranch(TTree &tree, const BranchConfig &branchConfig, OutputBranchData &outputData) override
Add one branch to the output tree.
ElementProcessorMet(const std::string &sgName, const std::string &termName)
Default constructor.
virtual StatusCode retrieveProcess(StoreType &evtStore) override
retrieve and process the object
std::string m_sgName
Name of the object in the event store.
virtual StatusCode retrieveProcess(StoreType &evtStore) override
retrieve and process the object
virtual StatusCode addBranch(TTree &tree, const BranchConfig &branchConfig, OutputBranchData &outputData) override
Add one branch to the output tree.
std::string m_sgName
Name of the object in the event store.
ElementProcessorRegular(const std::string &sgName)
Default constructor.
std::vector< std::unique_ptr< ElementBranchProcessor > > m_branches
List of branch processors set up for this xAOD object.
the interface class for classes reading an object from the event store and processing it
virtual StatusCode addBranch(TTree &, const BranchConfig &, OutputBranchData &)=0
Add one branch to the output tree.
StatusCode setupBranch(const BranchConfig &branchConfig, OutputBranchData &outputData, TTree &tree)
Function setting up an individual branch on the first event.
std::unordered_set< std::string > m_nonContainers
the non-containers
IObjectProcessor & getObjectProcessor(const BranchConfig &branchConfig, const std::string &sgName)
std::optional< int > defaultBasketSize
the default basket size
std::unordered_map< std::string, std::unique_ptr< IObjectProcessor > > m_processors
object processors
A non-templated base class for DataBucket, allows to access the transient object address as a void*.
virtual void * object()=0
virtual const std::type_info & tinfo() const =0
Return the type_info for the stored object.
virtual std::vector< const SG::DataProxy * > proxies() const =0
Return the list of all current proxies in store.
Handle mappings between names and auxid_t.
static AuxTypeRegistry & instance()
Return the singleton registry instance.
Manage index tracking and synchronization of auxiliary data.
Manage lookup of vectors of auxiliary data.
Helper class to provide constant type-safe access to aux data.
DataObject * accessData()
Access DataObject on-demand using conversion service.
Helper class to provide const generic access to aux data.
The Athena Transient Store API.
StatusCode retrieve(const T *&ptr) const
Retrieve the default object into a const T*.
MsgStream & msg() const
The standard message stream.
MsgStream & msg() const
The standard message stream.
AsgMessaging(const std::string &name)
Constructor with a name.
Wrapper for Event to make it look like StoreGate.
Definition SgEvent.h:44
bool contains(const std::string &name) const
Check if an object is available for constant access.
xAOD::TStore * tds() const
Return the underlying transient data store.
Definition SgEvent.cxx:33
T * retrieve(const std::string &name) const
Function retrieving a constant or non-constant object.
This class takes care of holding EDM objects in memory.
Definition THolder.h:35
const std::type_info * getTypeInfo() const
Definition THolder.cxx:412
const THolder * holder(const std::string &key) const
return holder for key
Definition TStore.cxx:50
bool match(std::string s1, std::string s2)
match the individual directories of two strings
Definition hcg.cxx:359
a namespace for helper functions and objects for filling tree branches
StoreGateSvc StoreType
the type of the event store in the current environment
Select isolated Photons, Electrons and Muons.
std::string normalizedTypeinfoName(const std::type_info &info)
Convert a type_info to a normalized string representation (matching the names used in the root dictio...
static const auxid_t null_auxid
To signal no aux data item.
Definition AuxTypes.h:30
AuxElement(SG::AuxVectorData *container, size_t index)
Base class for elements of a container that can have aux data.
virtual const IAuxTypeVector * getVector(SG::auxid_t auxid) const override
Return vector interface for one aux data item.
size_t auxid_t
Identifier for a particular aux data item.
Definition AuxTypes.h:27
cl
print [x.__class__ for x in toList(dqregion.getSubRegions()) ]
Definition index.py:1
void stable_sort(DataModel_detail::iterator< DVL > beg, DataModel_detail::iterator< DVL > end)
Specialization of stable_sort for DataVector/List.
DataModel_detail::iterator< DVL > remove_if(typename DataModel_detail::iterator< DVL > beg, typename DataModel_detail::iterator< DVL > end, Predicate pred)
Specialization of remove_if for DataVector/List.
char rootType(char typeidType)
This function is used internally in the code when creating primitive dynamic auxiliary branches.
Convert a type_info to a normalized string representation (matching the names used in the root dictio...
the user configuration of an output branch
std::string branchDecl
the original user configuration string
std::string sgName
the SG name of the object to read from
const SG::IAuxTypeVectorFactory * auxFactory
pointer to the aux vector factory
std::string auxName
the aux data variable name to read from
const std::type_info * auxVecType
the vector type of the decoration we read
std::optional< int > basketSize
the basket size for this branch
CP::SystematicSet auxNameFilterSys
the affecting systematics for the auxName
CP::SystematicSet sgNameFilterSys
the affecting systematics for the sgName
CP::SystematicSet branchNameFilterSys
the affecting systematics for the branchName
const std::type_info * auxType
the type of the decoration we read
SG::auxid_t nominalAuxId
the aux-id for the nominal decoration
std::string branchName
the name of the output branch
std::string metTermName
MET ONLY: the name of the MET term to write out.
bool nominalOnly
whether we only want to write out the nominal
std::string typeName
the name of the type (or empty to read from aux-registry)
the data for a single output branch
bool isNominal
whether this is unaffected by systematics (i.e. nominal)
const BranchConfig * branchConfig
the BranchConfig we are based on
std::size_t sysIndex
the index in the systematics list
StatusCode configureNames(const BranchConfig &branchConfig, const CP::SystematicSet &sys, ISystematicsSvc &sysSvc, MsgStream &msg)
configure names for systematics
std::string auxName
the name of the decoration in the aux-store
std::string sgName
the SG name of the object to read from
std::string branchName
the name of the output branch
MsgStream & msg
Definition testRead.cxx:32
TChain * tree
TFile * file