ATLAS Offline Software
Loading...
Searching...
No Matches
TrigTauMonitorTandPAlgorithm.cxx
Go to the documentation of this file.
1/*
2 Copyright (C) 2002-2025 CERN for the benefit of the ATLAS collaboration
3*/
4
7
8
9TrigTauMonitorTandPAlgorithm::TrigTauMonitorTandPAlgorithm(const std::string& name, ISvcLocator* pSvcLocator)
10 : TrigTauMonitorBaseAlgorithm(name, pSvcLocator)
11{}
12
13
16
17 ATH_CHECK( m_hltElectronKey.initialize() );
18 ATH_CHECK( m_hltMuonKey.initialize() );
19
20 ATH_CHECK( m_offlineElectronKey.initialize() );
21 ATH_CHECK( m_offlineMuonKey.initialize() );
22
23 return StatusCode::SUCCESS;
24}
25
26
27std::vector<const xAOD::Electron*> TrigTauMonitorTandPAlgorithm::getOfflineElectrons(const EventContext& ctx, const float threshold) const
28{
29 std::vector<const xAOD::Electron*> el_vec;
30
32 if(!electrons.isValid()) {
33 ATH_MSG_WARNING("Failed to retrieve offline Electrons");
34 return el_vec;
35 }
36
37 for(const xAOD::Electron* el : *electrons) {
38 // Threshold selection
39 if(el->p4().Pt()/Gaudi::Units::GeV < threshold) continue;
40
41 // Select offline electrons passing good quality cuts
42 if(!(el->passSelection("LHMedium") && el->isGoodOQ(xAOD::EgammaParameters::BADCLUSELECTRON))) continue;
43
44 el_vec.push_back(el);
45 }
46
47 return el_vec;
48}
49
50
51std::vector<const xAOD::Muon*> TrigTauMonitorTandPAlgorithm::getOfflineMuons(const EventContext& ctx, const float threshold) const
52{
53 std::vector<const xAOD::Muon*> mu_vec;
54
56 if(!muons.isValid()) {
57 ATH_MSG_WARNING("Failed to retrieve offline Muons");
58 return mu_vec;
59 }
60
61 for(const xAOD::Muon* mu : *muons) {
62 // Threshold selection
63 if(mu->p4().Pt()/Gaudi::Units::GeV < threshold) continue;
64
65 // Select offline muons passing good quality cuts (quality >= Medium, but the Enum indexes are reversed...)
66 if(!(mu->quality() <= xAOD::Muon::Medium && mu->passesIDCuts())) continue;
67
68 mu_vec.push_back(mu);
69 }
70
71 return mu_vec;
72}
73
74
75std::vector<const xAOD::Electron*> TrigTauMonitorTandPAlgorithm::getOnlineElectrons(const std::string& trigger) const
76{
77 std::vector<const xAOD::Electron*> el_vec;
78
80 for(auto &featLinkInfo : vec) {
81 const auto *feat = *(featLinkInfo.link);
82 if(!feat) continue;
83 el_vec.push_back(feat);
84 }
85
86 return el_vec;
87}
88
89
90std::vector<const xAOD::Muon*> TrigTauMonitorTandPAlgorithm::getOnlineMuons(const std::string& trigger) const
91{
92 std::vector<const xAOD::Muon*> mu_vec;
93
94 auto vec = m_trigDecTool->features<xAOD::MuonContainer>(trigger, TrigDefs::Physics, m_hltMuonKey.key());
95 for(auto &featLinkInfo : vec) {
96 const auto *feat = *(featLinkInfo.link);
97 if(!feat) continue;
98 mu_vec.push_back(feat);
99 }
100
101 return mu_vec;
102}
103
104
105StatusCode TrigTauMonitorTandPAlgorithm::processEvent(const EventContext& ctx) const
106{
107 constexpr float threshold_offset = 10.0;
108
109 // Offline taus
110 auto offline_taus_all = getOfflineTausAll(ctx, 0.0);
111 if(m_requireOfflineTaus && offline_taus_all.empty()) return StatusCode::SUCCESS;
112
113 for(const std::string& trigger : m_triggers) {
114 const TrigTauInfo& info = getTrigInfo(trigger);
115
116 if(!info.isHLTTandP()) {
117 ATH_MSG_WARNING("Chain \"" << trigger << "\" is not a Tag and Probe trigger. Skipping...");
118 continue;
119 }
120
121 const auto passBits = m_trigDecTool->isPassedBits(trigger);
122 const bool l1_accept_flag = passBits & TrigDefs::L1_isPassedAfterVeto;
123 const bool hlt_not_prescaled_flag = (passBits & TrigDefs::EF_prescaled) == 0;
124
125 // Offline tau requirement check
126 std::vector<const xAOD::TauJet*> offline_taus_with_id = classifyTausAll(offline_taus_all, 0, static_cast<TauID>(m_offline_tau_id.value()));
127 if(m_requireOfflineTaus && offline_taus_with_id.empty()) continue;
128
129 // Filter offline taus
130 std::vector<const xAOD::TauJet*> offline_taus = classifyTausAll(offline_taus_with_id, info.getHLTTauThreshold() - threshold_offset);
131
132 // Online taus
133 std::vector<const xAOD::TauJet*> hlt_taus = getOnlineTausAll(trigger, true);
134
135 if(info.hasHLTElectronLeg()) { // Electron channel
136 // Offline Electrons
137 std::vector<const xAOD::IParticle*> offline_electrons;
138 for(const xAOD::Electron* p : getOfflineElectrons(ctx, info.getHLTElecThreshold()+1)) offline_electrons.push_back(dynamic_cast<const xAOD::IParticle*>(p));
139
140 // Overlap removal: dR(Offline Tau, Offline Electron) > 0.2
141 for(int i = offline_taus.size()-1; i >= 0; i--) {
142 bool is_match = matchObjects(offline_taus.at(i), offline_electrons, 0.2);
143 if(is_match) offline_taus.erase(offline_taus.begin() + i);
144 }
145
146 // Online Electrons
147 std::vector<const xAOD::IParticle*> hlt_electrons;
148 for(const xAOD::Electron* p : getOnlineElectrons(trigger)) hlt_electrons.push_back(dynamic_cast<const xAOD::IParticle*>(p));
149
150 if(m_do_variable_plots) fillTagAndProbeVars(trigger, hlt_taus, hlt_electrons);
151 if(m_do_efficiency_plots && l1_accept_flag && hlt_not_prescaled_flag) fillTAndPHLTEfficiencies(ctx, trigger, offline_electrons, hlt_electrons, offline_taus, hlt_taus);
152
153 } else if(info.hasHLTMuonLeg()) { // Muon channel
154 // Offline Muons
155 std::vector<const xAOD::IParticle*> offline_muons;
156 for(const xAOD::Muon* p : getOfflineMuons(ctx, info.getHLTMuonThreshold()+1)) offline_muons.push_back(dynamic_cast<const xAOD::IParticle*>(p));
157
158 // Overlap removal: dR(Offline Tau, Offline Tau) > 0.2
159 for(int i = offline_taus.size()-1; i >= 0; i--) {
160 bool is_match = matchObjects(offline_taus.at(i), offline_muons, 0.2);
161 if(is_match) offline_taus.erase(offline_taus.begin() + i);
162 }
163
164 // Online Muons
165 std::vector<const xAOD::IParticle*> hlt_muons;
166 for(const xAOD::Muon* p : getOnlineMuons(trigger)) hlt_muons.push_back(dynamic_cast<const xAOD::IParticle*>(p));
167
168 if(m_do_variable_plots) fillTagAndProbeVars(trigger, hlt_taus, hlt_muons);
169 if(m_do_efficiency_plots && l1_accept_flag && hlt_not_prescaled_flag) fillTAndPHLTEfficiencies(ctx, trigger, offline_muons, hlt_muons, offline_taus, hlt_taus);
170 }
171 }
172
173 return StatusCode::SUCCESS;
174}
175
176
177void TrigTauMonitorTandPAlgorithm::fillTAndPHLTEfficiencies(const EventContext& ctx, const std::string& trigger, const std::vector<const xAOD::IParticle*>& offline_lep_vec, const std::vector<const xAOD::IParticle*>& online_lep_vec, const std::vector<const xAOD::TauJet*>& offline_tau_vec, const std::vector<const xAOD::TauJet*>& online_tau_vec) const
178{
179 ATH_MSG_DEBUG("Fill Tag and Probe HLT efficiencies: " << trigger);
180
181 // Require 1 offline taus and 1 online taus
182 if(online_tau_vec.size() != 1 || offline_tau_vec.size() != 1) return;
183 // ...and require 1 offline lepton and 1 online lepton
184 if(online_lep_vec.size() != 1 || offline_lep_vec.size() != 1) return;
185
186 auto monGroup = getGroup(trigger+"_TAndPHLT_Efficiency");
187
188 auto tauPt = Monitored::Scalar<float>("tauPt", 0.0);
189 auto tauEta = Monitored::Scalar<float>("tauEta", 0.0);
190 auto tauPhi = Monitored::Scalar<float>("tauPhi", 0.0);
191 auto dR = Monitored::Scalar<float>("dR", 0.0);
192 auto dEta = Monitored::Scalar<float>("dEta", 0.0);
193 auto dPhi = Monitored::Scalar<float>("dPhi", 0.0);
194 auto averageMu = Monitored::Scalar<float>("averageMu", 0.0);
195 auto HLT_match = Monitored::Scalar<bool>("HLT_pass", false);
196 auto HLT_match_highPt = Monitored::Scalar<bool>("HLT_pass_highPt", false);
197
198 // efficiency denominator : 1 offline tau, 1 online tau, 1 offline lepton, 1 online lepton
199 // efficiency numerator : hlt fires + offline and online tau matched + offline and online lepton matched
200 bool hlt_fires = m_trigDecTool->isPassed(trigger, TrigDefs::Physics);
201 bool tau1_match = matchObjects(offline_tau_vec[0], online_tau_vec, 0.2);
202 bool lep1_match = matchObjects(offline_lep_vec[0], online_lep_vec, 0.2);
203
204 tauPt = offline_tau_vec[0]->pt()/Gaudi::Units::GeV;
205 tauEta = offline_tau_vec[0]->eta();
206 tauPhi = offline_tau_vec[0]->phi();
207 dR = offline_tau_vec[0]->p4().DeltaR(offline_lep_vec[0]->p4());
208 dEta = std::abs(offline_tau_vec[0]->eta() - offline_lep_vec[0]->eta());
209 dPhi = offline_tau_vec[0]->p4().DeltaPhi(offline_lep_vec[0]->p4());
210 averageMu = lbAverageInteractionsPerCrossing(ctx);
211 HLT_match = hlt_fires && tau1_match && lep1_match;
212
213 fill(monGroup, tauPt, tauEta, tauPhi, dR, dEta, dPhi, averageMu, HLT_match);
214
215 bool is_highPt = tauPt > getTrigInfo(trigger).getHLTTauThreshold() + 20.0;
216 if(is_highPt) {
217 HLT_match_highPt = static_cast<bool>(HLT_match);
218 fill(monGroup, tauEta, tauPhi, HLT_match_highPt);
219 }
220
221
222 ATH_MSG_DEBUG("After fill Tag and Probe HLT efficiencies: " << trigger);
223}
224
225
226void TrigTauMonitorTandPAlgorithm::fillTagAndProbeVars(const std::string& trigger, const std::vector<const xAOD::TauJet*>& tau_vec, const std::vector<const xAOD::IParticle*>& lep_vec) const
227{
228 ATH_MSG_DEBUG("Fill Tag & Probe Variables: " << trigger);
229
230 auto monGroup = getGroup(trigger+"_TAndPVars");
231
232 // Require 1 tau and 1 lepton
233 if(tau_vec.empty() || lep_vec.empty()) return;
234
235 auto dR = Monitored::Scalar<float>("dR", 0.0);
236 auto dEta = Monitored::Scalar<float>("dEta", 0.0);
237 auto dPhi = Monitored::Scalar<float>("dPhi", 0.0);
238 auto dPt = Monitored::Scalar<float>("dPt", 0.0);
239
240 auto Pt = Monitored::Scalar<float>("Pt", 0.0);
241 auto Eta = Monitored::Scalar<float>("Eta", 0.0);
242 auto Phi = Monitored::Scalar<float>("Phi", 0.0);
243 auto M = Monitored::Scalar<float>("M", 0.0);
244
245 // Choose a pair with dR > 0.3 to fill the plot (there must be always at least one pair with dR > 0.3 if the trigger fires)
246 uint index_tau = 0;
247 uint index_lep = 0;
248
249 for(uint i=0; i < tau_vec.size(); i++) {
250 for(uint j=0; j< lep_vec.size(); j++) {
251 if(tau_vec[i]->p4().DeltaR(lep_vec[j]->p4()) >= 0.3) {
252 index_tau = i;
253 index_lep = j;
254 }
255 }
256 }
257
258 dR = tau_vec[index_tau]->p4().DeltaR(lep_vec[index_lep]->p4());
259 dEta = std::abs(tau_vec[index_tau]->eta() - lep_vec[index_lep]->eta());
260 dPhi = tau_vec[index_tau]->p4().DeltaPhi(lep_vec[index_lep]->p4());
261 dPt = std::abs((tau_vec[index_tau]->pt() - lep_vec[index_lep]->pt())/Gaudi::Units::GeV);
262
263 TLorentzVector diTau4V = tau_vec[index_tau]->p4() + lep_vec[index_lep]->p4();
264
265 Pt = diTau4V.Pt()/Gaudi::Units::GeV;
266 Eta = diTau4V.Eta();
267 Phi = diTau4V.Phi();
268 M = diTau4V.M()/Gaudi::Units::GeV;
269
270 fill(monGroup, dR, dEta, dPhi, dPt, Pt, Eta, Phi, M);
271
272 ATH_MSG_DEBUG("After fill Tag & Probe variables: " << trigger);
273}
Scalar eta() const
pseudorapidity method
#define ATH_CHECK
Evaluate an expression and check for errors.
#define ATH_MSG_WARNING(x)
#define ATH_MSG_DEBUG(x)
std::vector< size_t > vec
unsigned int uint
const ToolHandle< GenericMonitoringTool > & getGroup(const std::string &name) const
Get a specific monitoring tool from the tool handle array.
PublicToolHandle< Trig::TrigDecisionTool > m_trigDecTool
Tool to tell whether a specific trigger is passed.
Declare a monitored scalar variable.
virtual bool isValid() override final
Can the handle be successfully dereferenced?
float getHLTTauThreshold() const
Definition TrigTauInfo.h:25
Gaudi::Property< bool > m_do_efficiency_plots
virtual StatusCode initialize() override
initialize
double dR(const double eta1, const double phi1, const double eta2, const double phi2) const
TrigTauMonitorBaseAlgorithm(const std::string &name, ISvcLocator *pSvcLocator)
std::vector< const xAOD::TauJet * > getOfflineTausAll(const EventContext &ctx, const float threshold=20.0) const
bool matchObjects(const T1 *tau, const std::vector< const T2 * > &tau_vec, float threshold) const
Gaudi::Property< std::vector< std::string > > m_triggers
const TrigTauInfo & getTrigInfo(const std::string &trigger) const
std::vector< const xAOD::TauJet * > classifyTausAll(const std::vector< const xAOD::TauJet * > &taus, const float threshold=0.0, const TauID tau_id=TauID::None) const
Gaudi::Property< bool > m_do_variable_plots
std::vector< const xAOD::TauJet * > getOnlineTausAll(const std::string &trigger, bool include_0P=true) const
SG::ReadHandleKey< xAOD::MuonContainer > m_hltMuonKey
TrigTauMonitorTandPAlgorithm(const std::string &name, ISvcLocator *pSvcLocator)
virtual StatusCode processEvent(const EventContext &ctx) const override
SG::ReadHandleKey< xAOD::ElectronContainer > m_offlineElectronKey
void fillTAndPHLTEfficiencies(const EventContext &ctx, const std::string &trigger, const std::vector< const xAOD::IParticle * > &offline_lep_vec, const std::vector< const xAOD::IParticle * > &online_lep_vec, const std::vector< const xAOD::TauJet * > &offline_tau_vec, const std::vector< const xAOD::TauJet * > &online_tau_vec) const
Gaudi::Property< unsigned int > m_offline_tau_id
std::vector< const xAOD::Electron * > getOfflineElectrons(const EventContext &ctx, const float threshold=0.0) const
std::vector< const xAOD::Muon * > getOfflineMuons(const EventContext &ctx, const float threshold=0.0) const
SG::ReadHandleKey< xAOD::ElectronContainer > m_hltElectronKey
void fillTagAndProbeVars(const std::string &trigger, const std::vector< const xAOD::TauJet * > &tau_vec, const std::vector< const xAOD::IParticle * > &lep_vec) const
std::vector< const xAOD::Muon * > getOnlineMuons(const std::string &trigger) const
std::vector< const xAOD::Electron * > getOnlineElectrons(const std::string &trigger) const
SG::ReadHandleKey< xAOD::MuonContainer > m_offlineMuonKey
virtual StatusCode initialize() override
initialize
Class providing the definition of the 4-vector interface.
void fill(const ToolHandle< GenericMonitoringTool > &groupHandle, std::vector< std::reference_wrapper< Monitored::IMonitoredVariable > > &&variables) const
Fills a vector of variables to a group by reference.
virtual float lbAverageInteractionsPerCrossing(const EventContext &ctx=Gaudi::Hive::currentContext()) const
Calculate the average mu, i.e.
const uint32_t BADCLUSELECTRON
Definition EgammaDefs.h:116
ElectronContainer_v1 ElectronContainer
Definition of the current "electron container version".
Muon_v1 Muon
Reference the current persistent version:
MuonContainer_v1 MuonContainer
Definition of the current "Muon container version".
Electron_v1 Electron
Definition of the current "egamma version".