ATLAS Offline Software
Loading...
Searching...
No Matches
CopyTruthJetParticles.cxx
Go to the documentation of this file.
1/*
2 Copyright (C) 2002-2025 CERN for the benefit of the ATLAS collaboration
3*/
4
10
15#include "AsgMessaging/Check.h"
18
19#include <mutex> // std::call_once, std::once_flag
20
21#ifndef XAOD_STANDALONE
22// Usage of metadata is for now only possible in Athena...
23//#include "CoralBase/AttributeListException.h"
25#endif
26
27// For std::find in comesFrom()
28#include <algorithm>
29
30using namespace std;
31using namespace MCTruthPartClassifier;
32
34 : AsgTool(name) {}
35
37
38 ATH_CHECK(m_truthParticleKey.initialize());
39 ATH_CHECK(m_outTruthPartKey.initialize());
40 ATH_CHECK(m_dressingNames.initialize());
41
42 // ATLASRECTS-8290: this is for backward compatability, remove eventually
44
45 return StatusCode::SUCCESS;
46}
47
48
49
51 std::vector<const xAOD::TruthParticle*>& promptLeptons,
52 std::map<const xAOD::TruthParticle*,unsigned int>& tc_results) const {
53
54 // Needed for the dressed photon decorations
55 const EventContext& ctx = Gaudi::Hive::currentContext();
56
57 // Check if this thing is a candidate to be in a truth jet
58 // First block is largely copied from isGenStable, which works on HepMC only
59 if (HepMC::is_simulation_particle(tp)) return false; // Particle is from G4
60 int pdgid = tp->pdgId();
61 if (MC::isZeroEnergyPhoton(tp)) return false; // Work around for an old generator bug
62
63 // -- changed for dark jet clustering -- //
64 if ( !MC::isStable(tp) && !m_includeDark ) return false; // dark hadrons will not be status 1
65 // ----------------------------------- //
66
67 // Easy classifiers by PDG ID
68 if(MC::isNeutrino(pdgid)) {
69 if (!m_includeNu) return false;
70 } else {
71 if (!m_includeBSMNonInt && !MC::isInteracting(pdgid)) return false;
72 }
73 if (!m_includeMu && MC::isMuon(pdgid)) return false;
74
75 // Already built a list of prompt leptons, just use it here
76 if (!m_includePromptLeptons && std::find(promptLeptons.begin(),promptLeptons.end(),tp)!=promptLeptons.end()){
77 ATH_MSG_VERBOSE("Veto prompt lepton (" << pdgid << ") with pt " << tp->pt());
78 return false;
79 }
80
81 // Extra catch. If we aren't supposed to include prompt leptons, we aren't supposed to include prompt neutrinos
82 unsigned int tc_res = getTCresult(tp, tc_results);
84 return false;
85 }
86
87 // -- added for dark jet clustering -- //
88 // new classifiers to account for dark particles
89 // for dark jets: ignore SM particles; include only "stable" dark hadrons
90 if (!m_includeSM && !MC::isHiddenValley(tp)) return false;
91 if (m_includeDark) {
92 if (abs(tp->pdgId()) <= 4900101) return false; // ignore Xd, qd, gd
93 if (tp->hasDecayVtx()) {
94 size_t good_hadrons = 0;
95 for (size_t p = 0; p < tp->end_vertex()->nOutgoingParticles(); ++p) {
96 if (!MC::isHiddenValley(tp->child(p))) {
97 good_hadrons++;
98 }
99 }
100 if (good_hadrons == 0) return false; // ignore "non-stable" dark hadrons (decaying to dark sector) -- "stable" if decaying to SM
101 }
102 }
103 // for SM jets: ignore dark particles - probably unnecessary bc of status requirement above
104 if (!m_includeDark && MC::isHiddenValley(tp)) return false;
105 // ----------------------------------- //
106
107 if (!m_includePromptPhotons && MC::isPhoton(pdgid) && tp->hasProdVtx()){
108 //ParticleOrigin orig = getPartOrigin(tp, originMap);
109 //if (orig==Higgs || orig==HiggsMSSM) return false;
110 if (MCTruthPartClassifier::isPrompt(tc_res)) return false;
111 }
112
113 // If we want to remove photons via the dressing decoration
114 if (!m_dressingNames.empty()){
115 // Accessor for the dressing decoration above
116 bool foundDressDec{false};
117 for(const auto &decName : m_dressingNames){
119 if (MC::isPhoton(pdgid) && dressAcc(*tp)) foundDressDec = true;
120 }
121 if (foundDressDec) return false;
122 } // End of removal via dressing decoration
123
124 // Pseudo-rapidity cut
125 if(std::abs(tp->eta())>m_maxAbsEta) return false;
126
127 // Vetoes of specific origins. Not fast, but if no particles are specified should not execute
128 if (m_vetoPDG_IDs.size()>0){
129 std::vector<int> used_vertices;
130 for (int anID : m_vetoPDG_IDs){
131 used_vertices.clear();
132 if (comesFrom(tp,anID,used_vertices)) return false;
133 }
134 }
135
136 // Made it!
137 return true;
138}
139
140
142 std::map<const xAOD::TruthParticle*,unsigned int>& tc_results) const
143{
144 if(tc_results.find(tp) == tc_results.end()) {
145 const unsigned int result = tp ? std::get<0>(MCTruthPartClassifier::defOrigOfParticle(tp)) : 0;
146 tc_results[tp] = result;
147 }
148 return tc_results[tp];
149}
150
152
153 // retrieve barcode Offset for this event from metadata.
154 // We'd need a cleaner solution where this offset is set only at
155 // each new file, but this requires some tool interface which does
156 // not exist in RootCore yet.
157 // So we use the less disruptive solution in Athena for now...
158
159 // the function used below is
160 // std::call_once(metaDataFlag,basicMetaDataCheck(), this);
161 // std::call_once(metaDataFlag,this->basicMetaDataCheck());
162 // the syntax is explained in http://stackoverflow.com/questions/23197333/why-is-this-pointer-needed-when-calling-stdcall-once
163 // this syntax requires the call_once function to receive the object the function is called upon
164 //": these are all non-static member functions , they act on objects
165 // they need the "this" pointer which always point to the object the function is working on
166 //http://www.learncpp.com/cpp-tutorial/812-static-member-functions/
167
168 std::vector<const xAOD::TruthParticle*> promptLeptons;
169 promptLeptons.reserve(10);
170
171 // Retrieve the xAOD truth objects
172 auto truthParticles = SG::makeHandle(m_truthParticleKey);
173 if ( !truthParticles.isValid() ) {
174 ATH_MSG_ERROR("Failed to retrieve truth particle container " << m_truthParticleKey.key());
175 return 1;
176 }
177
178 // Classify particles for tagging and add to the TruthParticleContainer
179 std::unique_ptr<ConstDataVector<xAOD::TruthParticleContainer> > ptruth(new ConstDataVector<xAOD::TruthParticleContainer>(SG::VIEW_ELEMENTS));
180 std::map<const xAOD::TruthParticle*,unsigned int> tc_results;
181 tc_results.clear();
182 size_t numCopied = 0;
183
184 for (const xAOD::TruthParticle* tp : *truthParticles) {
185 if(!tp) continue;
186 if (tp->pt() < m_ptmin)
187 continue;
188 // Cannot use the truth helper functions; they're written for HepMC
189 // Last two switches only apply if the thing is a lepton and not a tau
190 if ((MC::isElectron(tp) || MC::isMuon(tp)) && tp->hasProdVtx()) {
191 // If this is a prompt, generator stable lepton, then we can use it
193 promptLeptons.push_back(tp);
194 }
195 }
196 }
197
198 for (const xAOD::TruthParticle* tp : *truthParticles) {
199 if(!tp) continue;
200 if (tp->pt() < m_ptmin)
201 continue;
202
203 if (classifyJetInput(tp, promptLeptons, tc_results)) {
204 ptruth->push_back(tp);
205 numCopied += 1;
206 }
207 }
208
209 ATH_MSG_DEBUG("Copied " << numCopied << " truth particles into " << m_outTruthPartKey.key() << " TruthParticle container");
210
211 // record
212 auto truthParticles_out = SG::makeHandle(m_outTruthPartKey);
213 ATH_MSG_DEBUG("Recorded truth particle collection " << m_outTruthPartKey.key());
214 // notify
215 if (!truthParticles_out.put(std::move(ptruth))) {
216 ATH_MSG_ERROR("Unable to write new TruthParticleContainer to event store: "
217 << m_outTruthPartKey.key());
218 } else {
219 ATH_MSG_DEBUG("Created new TruthParticleContainer in event store: "
220 << m_outTruthPartKey.key());
221 }
222
223 return 0;
224}
225
226
227bool CopyTruthJetParticles::comesFrom( const xAOD::TruthParticle* tp, const int pdgID, std::vector<int>& used_vertices ) const {
228 // If it's not a particle, then it doesn't come from something...
229 if (!tp) return false;
230 // If it doesn't have a production vertex or has no parents, it doesn't come from much of anything
231 if (!tp->prodVtx() || tp->nParents()==0) return false;
232 // If we have seen it before, then skip this production vertex
233 // ATLASRECTS-8290: this should be replaced with ->uid()
234 if (std::find(used_vertices.begin(),used_vertices.end(), m_uid(*tp->prodVtx()))!=used_vertices.end()) return false;
235 // Add the production vertex to our used list
236 // ATLASRECTS-8290: this should be replaced with ->uid()
237 used_vertices.push_back( m_uid(*tp->prodVtx()) );
238 // Loop over the parents
239 for (size_t par=0;par<tp->nParents();++par){
240 // Check for null pointers in case of skimming
241 if (!tp->parent(par)) continue;
242 // Check for a match
243 if (tp->parent(par)->absPdgId()==pdgID) return true;
244 // Recurse on this parent
245 if (comesFrom(tp->parent(par), pdgID, used_vertices)) return true;
246 }
247 // No hits -- all done with the checks!
248 return false;
249}
250
#define ATH_CHECK
Evaluate an expression and check for errors.
#define ATH_MSG_ERROR(x)
#define ATH_MSG_VERBOSE(x)
#define ATH_MSG_DEBUG(x)
Handle class for reading from StoreGate.
Handle class for recording to StoreGate.
DataVector adapter that acts like it holds const pointers.
ATLAS-specific HepMC functions.
Define macros for attributes used to control the static checker.
DataVector adapter that acts like it holds const pointers.
SG::ConstAccessor< int > m_uid
Gaudi::Property< bool > m_includeDark
bool comesFrom(const xAOD::TruthParticle *tp, const int pdgID, std::vector< int > &used_vertices) const
virtual StatusCode initialize() override final
Function initialising the tool.
Gaudi::Property< bool > m_includeSM
Gaudi::Property< std::vector< int > > m_vetoPDG_IDs
Gaudi::Property< bool > m_includeBSMNonInt
bool classifyJetInput(const xAOD::TruthParticle *tp, std::vector< const xAOD::TruthParticle * > &promptLeptons, std::map< const xAOD::TruthParticle *, unsigned int > &tc_results) const
Redefine our own Classifier function(s)
Gaudi::Property< bool > m_includeNu
SG::WriteHandleKey< ConstDataVector< xAOD::TruthParticleContainer > > m_outTruthPartKey
Key for output truth particles.
unsigned int getTCresult(const xAOD::TruthParticle *tp, std::map< const xAOD::TruthParticle *, unsigned int > &tc_results) const
Gaudi::Property< bool > m_includePromptLeptons
Gaudi::Property< float > m_maxAbsEta
Maximum allowed eta for particles in jets.
Gaudi::Property< bool > m_includePromptPhotons
Gaudi::Property< bool > m_use_barcode
SG::ReadHandleKey< xAOD::TruthParticleContainer > m_truthParticleKey
Key for input truth event.
Gaudi::Property< float > m_ptmin
Minimum pT for particle selection (in MeV)
CopyTruthJetParticles(const std::string &name)
Constructor.
virtual int execute() const override final
redefine execute so we can call our own classify()
Gaudi::Property< bool > m_includeMu
SG::ReadDecorHandleKeyArray< xAOD::TruthParticleContainer > m_dressingNames
Name of the decoration to be used for identifying FSR (dressing) photons.
Helper class to provide constant type-safe access to aux data.
Handle class for reading a decoration on an object.
AsgTool(const std::string &name)
Constructor specifying the tool instance's name.
Definition AsgTool.cxx:58
bool is_simulation_particle(const T &p)
Method to establish if a particle (or barcode) was created during the simulation (TODO update to be s...
int isPrompt(const unsigned int classify, bool allow_prompt_tau_decays=true)
std::tuple< unsigned int, T > defOrigOfParticle(T thePart)
bool isZeroEnergyPhoton(const T &p)
Identify a photon with zero energy. Probably a workaround for a generator bug.
bool isPhoton(const T &p)
bool isNeutrino(const T &p)
APID: the fourth generation neutrinos are neutrinos.
bool isElectron(const T &p)
bool isStable(const T &p)
Identify if the particle is stable, i.e. has not decayed.
bool isInteracting(const T &p)
Identify if the particle with given PDG ID would not interact with the detector, i....
bool isMuon(const T &p)
bool isHiddenValley(const T &p)
PDG rule 11k Hidden Valley particles have n = 4 and n_r = 9, and trailing numbers in agreement with t...
@ VIEW_ELEMENTS
this data object is a view, it does not own its elmts
SG::ReadCondHandle< T > makeHandle(const SG::ReadCondHandleKey< T > &key, const EventContext &ctx=Gaudi::Hive::currentContext())
STL namespace.
TruthParticle_v1 TruthParticle
Typedef to implementation.