ATLAS Offline Software
Loading...
Searching...
No Matches
eFexTowerBuilder.cxx
Go to the documentation of this file.
1/*
2 Copyright (C) 2002-2024 CERN for the benefit of the ATLAS collaboration
3*/
4
5//***************************************************************************
6// eFexTowerBuilder - description:
7// Builds an eFexTowerContainer from a CaloCellContainer (for supercells) and TriggerTowerContainer (for ppm tile towers)
8// -------------------
9// begin : 06 12 2022
10// email : will@cern.ch
11//***************************************************************************/
12
13
14// MyPackage includes
15#include "eFexTowerBuilder.h"
16
18
20
21
22#include "TFile.h"
23#include "TTree.h"
25
26namespace LVL1 {
27
28eFexTowerBuilder::eFexTowerBuilder( const std::string& name, ISvcLocator* pSvcLocator ) : AthReentrantAlgorithm( name, pSvcLocator ){
29
30
31}
32
34 ATH_MSG_INFO ("Initializing " << name() << "...");
35
36 CHECK( m_ddmKey.initialize(true) );
37 CHECK( m_ttKey.initialize(true) );
38 CHECK( m_scellKey.initialize(true) );
39 CHECK( m_outKey.initialize(true) );
40 CHECK( m_eiKey.initialize(true) );
42
43 if(!m_mappingFile.empty()) {
44 if (auto fileName = PathResolverFindCalibFile(m_mappingFile); !fileName.empty()) {
45 std::unique_ptr <TFile> f(TFile::Open(fileName.c_str()));
46 if (f) {
47 TTree *t = f->Get<TTree>("mapping");
48 if (t) {
49 unsigned long long scid = 0;
50 std::pair<int, int> coord = {0, 0};
51 std::pair<int, int> slot;
52 t->SetBranchAddress("scid", &scid);
53 t->SetBranchAddress("etaIndex", &coord.first);
54 t->SetBranchAddress("phiIndex", &coord.second);
55 t->SetBranchAddress("slot1", &slot.first);
56 t->SetBranchAddress("slot2", &slot.second);
57 for (Long64_t i = 0; i < t->GetEntries(); i++) {
58 t->GetEntry(i);
59 m_scMap[scid] = std::make_pair(coord, slot);
60 }
61 }
62 }
63 if (m_scMap.empty()) {
64 ATH_MSG_WARNING("Failed to load sc -> eFexTower map from " << fileName);
65 } else {
66 ATH_MSG_INFO("Loaded sc -> eFexTower map from " << fileName);
67 }
68 }
69 }
70
71 return StatusCode::SUCCESS;
72}
73
74StatusCode eFexTowerBuilder::fillTowers(const EventContext& ctx) const {
75
76
79 if(!tTowers.isValid()){
80 ATH_MSG_FATAL("Could not retrieve collection " << m_ttKey.key() );
81 return StatusCode::FAILURE;
82 }
83 SG::ReadHandle<CaloCellContainer> scells(m_scellKey,ctx); // n.b. 34048 is a full complement of scells
84 if(!scells.isValid()){
85 ATH_MSG_FATAL("Could not retrieve collection " << m_scellKey.key() );
86 return StatusCode::FAILURE;
87 }
88
90 if(!ei.isValid()) {
91 ATH_MSG_FATAL("Cannot retrieve eventinfo");
92 return StatusCode::FAILURE;
93 }
94 bool isMC = ei->eventType(xAOD::EventInfo::IS_SIMULATION); // currently only used to decide if should set a saturation code or not
95
96
97 std::map<std::pair<int,int>,std::array<int,11>> towers;
98
99 constexpr int INVALID_VALUE = -99999; // use this value to indicate invalid
100 constexpr int MASKED_VALUE = std::numeric_limits<int>::max(); // use this value to indicate masked
101 constexpr int SATURATED_VALUE = std::numeric_limits<int>::max()-1; // use this value to indicate saturation
102 constexpr int MISSING_VALUE = -99998; // use this value to indicate missing supercell
103
104 for (auto digi: *scells) {
105 const auto itr = m_scMap.find(digi->ID().get_compact());
106 if (itr == m_scMap.end()) { continue; } // not in map so not mapping to a tower
107 int val = std::round(digi->energy()/(12.5*std::cosh(digi->eta()))); // 12.5 is b.c. energy is in units of 12.5MeV per count
108 // note: a val of < -99998 is what is produced if efex was sent an invalid code of 1022 (see LArRawtoSuperCell)
109 if (isMC && m_applyTimingCut && !((digi)->provenance()&0x200)) val = 0; // apply timing cut to MC (already present in Data)
110 bool isSaturated = (!isMC) ? (digi->quality()) : false; // not applying saturation codes in MC until the changes to trigger counts has been investigated
111 bool isMasked = ((digi)->provenance()&0x80);
112 bool isInvalid = m_applyMasking ? ((digi)->provenance()&0x40) : false;
113 // note: if debugging, the SCIDs have value: digi->ID().get_compact()>>32
114 if(isInvalid) {
115 val = INVALID_VALUE;
116 }
117 if(isSaturated) {
118 val = SATURATED_VALUE;
119 }
120
121 auto towerItr = towers.emplace(itr->second.first,std::array<int,11>{}); // returns pair<itr,bool> with bool indicating if emplaced
122 if(towerItr.second) { // did an emplace
123 towerItr.first->second.fill(MISSING_VALUE); // ensure all slots initialize with missing value
124 }
125 auto& tower = (towerItr.first->second);
126 if (itr->second.second.second<11) {
127 // doing an energy split between slots ... don't include a masked channel (or invalid channel)
128 if (!isMasked && val!=INVALID_VALUE) {
129 if(isSaturated) {
130 // mark both as saturated
131 tower.at(itr->second.second.first) = SATURATED_VALUE;
132 tower.at(itr->second.second.second) = SATURATED_VALUE;
133 }
134 if(tower.at(itr->second.second.first)!=(SATURATED_VALUE)) { // don't override saturation
135 // if the other contribution was masked or invalid or missing, revert to 0 before adding this contribution
136 if (tower.at(itr->second.second.first)==MASKED_VALUE || tower.at(itr->second.second.first)==INVALID_VALUE || tower.at(itr->second.second.first)==MISSING_VALUE) {
137 tower.at(itr->second.second.first)=0;
138 }
139 tower.at(itr->second.second.first) += val >> 1;
140 }
141 if(tower.at(itr->second.second.second)!=(SATURATED_VALUE)) { // don't override saturation
142 // if the other contribution was masked or invalid or missing, revert to 0 before adding this contribution
143 if (tower.at(itr->second.second.second)==MASKED_VALUE || tower.at(itr->second.second.second)==INVALID_VALUE || tower.at(itr->second.second.second)==MISSING_VALUE) {
144 tower.at(itr->second.second.second)=0;
145 }
146 tower.at(itr->second.second.second) += (val - (val >> 1)); // HW seems fixed now!
147 }
148 }
149 // hw is incorrectly ignoring masking on the second part
150 // so always add the 2nd bit
151 //tower.at(itr->second.second.second) += (val - (val >> 1)); // Removed b.c. of fix above - leaving this comment here until resolved!
152 } else {
153 auto& v = tower.at(itr->second.second.first);
154 if (isMasked) {
155 // dont mark it masked if it already has a contribution
156 if(v==MISSING_VALUE) v = MASKED_VALUE;
157 } else if(isSaturated) {
158 v = val;
159 } else {
160 if(v==INVALID_VALUE || v==MISSING_VALUE) v = 0;
161 v += val;
162 }
163 }
164
165 }
166
167 // add tile energies from TriggerTowers
168 static const auto etaIndex = [](float eta) { return int( eta*10 ) + ((eta<0) ? -1 : 1); };
169 static const auto phiIndex = [](float phi) { return int( phi*32./M_PI ) + (phi<0 ? -1 : 1); };
170 for(const xAOD::TriggerTower_v2* tTower : *tTowers) {
171 if (std::abs(tTower->eta()) > 1.5) continue;
172 if (tTower->sampling() != 1) continue;
173 double phi = tTower->phi(); if(phi > M_PI) phi -= 2.*M_PI;
174 auto towerItr = towers.emplace(std::pair(etaIndex(tTower->eta()),phiIndex(phi)),std::array<int,11>{}); // returns pair<itr,bool> with bool indicating if emplaced
175 if(towerItr.second) { // did an emplace
176 towerItr.first->second.fill(MISSING_VALUE); // ensure all slots initialize with missing value
177 }
178 (towerItr.first->second).at(10) = tTower->cpET();
179 }
180
181
183 ATH_CHECK( eTowers.record(std::make_unique<xAOD::eFexTowerContainer>(),std::make_unique<xAOD::eFexTowerAuxContainer>()) );
184
185 static const auto calToFex = [](int calEt) {
186 if(calEt == MASKED_VALUE) return 0; // indicates masked channel
187 if(calEt == SATURATED_VALUE) return 1023; // saturated channel
188 if( calEt == INVALID_VALUE ) return 1022; // invalid channel value
189 if( calEt == MISSING_VALUE ) return 1025; // missing channel value
190 if(calEt<448) return std::max((calEt&~1)/2+32,1); // 25 MeV per eFexTower count
191 if(calEt<1472) return (calEt-448)/4+256; // 50 MeV per eFexTower count
192 if(calEt<3520) return (calEt-1472)/8+512; // 100 MeV ...
193 if(calEt<11584) return (calEt-3520)/32+768; // 400 MeV ...
194 return 1020;
195 };
196
197 // now create the towers. Note that we need a code for "missing" input (1025), because a tower can contain some but not all
198 // inputs, depending on which sources are present in the run (e.g. if tile is present but not LAr).
199 // This is different to e.g. jFex, which creates a separate tower for each source at each location
200 // so jFex doesn't need a "missing" input code.
201 for(auto& [coord,counts] : towers) {
202 size_t ni = (std::abs(coord.first)<=15) ? 10 : 11; // ensures we skip the tile towers for next line
203 for(size_t i=0;i<ni;++i) counts[i] = (scells->empty() ? 1025 : calToFex(counts[i])); // do latome energy scaling to non-tile towers - if had no cells will use code "1025" to indicate
204 eTowers->push_back( std::make_unique<xAOD::eFexTower>() );
205 eTowers->back()->initialize( ( (coord.first<0 ? 0.5:-0.5) + coord.first)*0.1 ,
206 ( (coord.second<0 ? 0.5:-0.5) + coord.second)*M_PI/32,
207 std::vector<uint16_t>(counts.begin(), counts.end()),
208 -1, /* module number */
209 -1, /* fpga number */
210 0,0 /* status flags ... could use to indicate which cells were actually present?? */);
211 }
212
213 return StatusCode::SUCCESS;
214
215}
216
217StatusCode eFexTowerBuilder::fillMap(const EventContext& ctx) const {
218
219 ATH_MSG_INFO("Filling sc -> eFexTower map");
220
222 SG::ReadHandle<CaloCellContainer> scells(m_scellKey,ctx); // 34048 is a full complement of scells
223 if(!scells.isValid()){
224 ATH_MSG_FATAL("Could not retrieve collection " << m_scellKey.key() );
225 return StatusCode::FAILURE;
226 }
227 if (scells->size() != 34048 && !m_mappingFile.empty()) {
228 ATH_MSG_FATAL("Cannot fill sc -> eFexTower mapping with an incomplete sc collection");
229 return StatusCode::FAILURE;
230 }
231
232 // read the LATOME header if a key is given, so that we can determine LATOME version and get mapping right
233 bool doV6Mapping = m_v6Mapping;
234
235 if(!m_LArLatomeHeaderContainerKey.empty()) {
237 if(hdrCont.isValid()) {
238 for (const LArLATOMEHeader* hit : *hdrCont) {
239 doV6Mapping = (hit->FWversion()>1600);
240 }
241 if (doV6Mapping != m_v6Mapping) {
242 ATH_MSG_WARNING("Used LATOME Hardware to determine mapping different to python configuration (use V6 Mapping = " << doV6Mapping << " )");
243 }
244 }
245
246 }
247 struct TowerSCells {
248 std::vector<unsigned long long> ps;
249 std::vector<std::pair<float,unsigned long long>> l1;
250 std::vector<std::pair<float,unsigned long long>> l2;
251 std::vector<unsigned long long> l3;
252 std::vector<unsigned long long> had;
253 std::vector<unsigned long long> other;
254 };
255 static const auto etaIndex = [](float eta) { return int( eta*10 ) + ((eta<0) ? -1 : 1); }; // runs from -25 to 25, skipping over 0 (so gives outer edge eta)
256 static const auto phiIndex = [](float phi) { return int( phi*32./ROOT::Math::Pi() ) + (phi<0 ? -1 : 1); }; // runs from -pi to pi, skipping over 0 (gives out edge phi)
257 std::map<std::pair<int,int>,TowerSCells> towers;
258 std::map<unsigned long long,int> eTowerSlots; // not used by this alg, but we produce the map for benefit of eFexTower->eTower alg
259
260 for (auto digi: *scells) {
261 Identifier id = digi->ID(); // this is if using supercells
262
263 if (auto elem = ddm->get_element(id); elem && std::abs(elem->eta_raw())<2.5) {
264 float eta = elem->eta_raw(); // this seems more symmetric
265 int sampling = elem->getSampling();
266 if(sampling==6 && ddm->getCaloCell_ID()->region(id)==0 && eta<0) eta-=0.01; // nudge this L2 endcap supercell into correct tower (right on boundary)
267
268 unsigned long long val = id.get_compact();
269
270 int towerid = -1;int slot = -1;bool issplit = false;
271 CHECK(m_eFEXSuperCellTowerIdProviderTool->geteTowerIDandslot(id.get_compact(), towerid, slot, issplit));
272 eTowerSlots[id.get_compact()] = slot;
273
274 auto& sc = towers[std::pair(etaIndex(eta),phiIndex(elem->phi_raw()))];
275 switch(sampling) {
276 case 0: case 4: //lar barrel/endcap presampler
277 sc.ps.push_back(val);
278 break;
279 case 1: case 5: //lar barrel/endcap l1
280 sc.l1.push_back({elem->eta(),val}); break;
281 case 2: case 6: //lar barrel/endcap l2
282 sc.l2.push_back({elem->eta(),val}); break;
283 case 3: case 7: //lar barrel/endcap l3
284 sc.l3.push_back(val); break;
285 case 8: case 9: case 10: case 11: //lar hec
286 sc.had.push_back(val); break;
287 default:
288 sc.other.push_back(val); break;
289 }
290 }
291 }
292
293
294 // sort (by increasing eta) l1/l2 sc and handle special cases
295 // finally also output the eTower slot vector
296 std::vector<size_t> slotVector(11);
297 for(auto& [coord,sc] : towers) {
298 std::sort(sc.l1.begin(),sc.l1.end());
299 std::sort(sc.l2.begin(),sc.l2.end());
300 // we have 5 l2 cells @ |eta|=1.45 ... put lowest |eta| one in l3 slot
301 if (sc.l2.size()==5) {
302 if (coord.first >= 0) {
303 sc.l3.push_back(sc.l2.front().second);
304 sc.l2.erase(sc.l2.begin()); // remove first
305 } else {
306 sc.l3.push_back(sc.l2.back().second);
307 sc.l2.resize(sc.l2.size()-1); // remove last
308 }
309 }
310 if (std::abs(coord.first)==15) { //|eta| = 1.45
311 // in the overlap region it seems like the latome id with highest |eta| is swapped with next highest
312 // so to compare we swap the first and second (3rd and 4th are fine) if eta < 0, or 3rd and 4th if eta > 0
313 if (coord.first<0) {std::swap(sc.l1.at(0),sc.l1.at(1)); }
314 else {std::swap(sc.l1.at(2),sc.l1.at(3));}
315 }
316 // handle case @ |eta|~1.8-2 with 6 L1 cells
317 if (sc.l1.size()==6) {
318 m_scMap[sc.l1.at(0).second] = std::pair(coord,std::pair(1,11));
319 m_scMap[sc.l1.at(1).second] = std::pair(coord,(doV6Mapping && coord.first < 0) ? std::pair(2,1) : std::pair(1,2)); // in LATOME v5 FW this was (1,2) for both sides
320 m_scMap[sc.l1.at(2).second] = std::pair(coord,std::pair(2,11));
321 m_scMap[sc.l1.at(3).second] = std::pair(coord,std::pair(3,11));
322 m_scMap[sc.l1.at(4).second] = std::pair(coord,(doV6Mapping && coord.first < 0) ? std::pair(4,3) : std::pair(3,4)); // in LATOME v5 FW this was (3,4) for both sides
323 m_scMap[sc.l1.at(5).second] = std::pair(coord,std::pair(4,11));
324 slotVector[1] = eTowerSlots[sc.l1.at(0).second];
325 slotVector[2] = eTowerSlots[sc.l1.at(2).second];
326 slotVector[3] = eTowerSlots[sc.l1.at(3).second];
327 slotVector[4] = eTowerSlots[sc.l1.at(5).second];
328 }
329
330 // for |eta|>2.4 there's only 1 l1 sc, to match hardware this should be compared placed in the 'last' l1 input
331 if (sc.l1.size()==1) {
332 m_scMap[sc.l1.at(0).second] = std::pair(coord,std::pair(4,11));
333 slotVector[1] = 1; slotVector[2] = 2; slotVector[3] = 3; slotVector[4] = eTowerSlots[sc.l1.at(0).second];
334 }
335
336 // fill the map with sc ids -> tower coord + slot
337 if (!sc.ps.empty()) {m_scMap[sc.ps.at(0)] = std::pair(coord,std::pair(0,11)); slotVector[0] = eTowerSlots[sc.ps.at(0)]; }
338 if(sc.l1.size()==4) for(size_t i=0;i<4;i++) if(sc.l1.size() > i) {m_scMap[sc.l1.at(i).second] = std::pair(coord,std::pair(i+1,11)); slotVector[i+1] = eTowerSlots[sc.l1.at(i).second]; }
339 for(size_t i=0;i<4;i++) if(sc.l2.size() > i) { m_scMap[sc.l2.at(i).second] = std::pair(coord,std::pair(i+5,11)); slotVector[i+5] = eTowerSlots[sc.l2.at(i).second]; }
340 if (!sc.l3.empty()) {m_scMap[sc.l3.at(0)] = std::pair(coord,std::pair(9,11)); slotVector[9] = eTowerSlots[sc.l3.at(0)]; }
341 if (!sc.had.empty()) {m_scMap[sc.had.at(0)] = std::pair(coord,std::pair(10,11));slotVector[10] = eTowerSlots[sc.had.at(0)]; }
342
343 // finally output the slotVector for this tower
344 // do only for the slots that don't match
345 // note to self: seems like everything is fine apart from the l1->ps remap for |eta|>2.4
346 // so leaving this bit commented out for now ... useful to leave it here in case need to recheck in future
347// for(size_t i=0;i<slotVector.size();i++) {
348// if(slotVector[i] != i) {
349// std::cout << coord.first << "," << coord.second << "," << i << "," << slotVector[i] << std::endl;
350// }
351// }
352 }
353
354 // save the map to disk, if required
355 if(!m_mappingFile.empty()) {
356 TFile f(m_mappingFile.value().c_str(), "RECREATE");
357 TTree *t = new TTree("mapping", "mapping");
358 unsigned long long scid = 0;
359 std::pair<int, int> coord = {0, 0};
360 std::pair<int, int> slot = {-1, -1};
361 t->Branch("scid", &scid);
362 t->Branch("etaIndex", &coord.first);
363 t->Branch("phiIndex", &coord.second);
364 t->Branch("slot1", &slot.first);
365 t->Branch("slot2", &slot.second);
366 for (auto &[id, val]: m_scMap) {
367 scid = id;
368 coord = val.first;
369 slot = val.second;
370 t->Fill();
371 }
372 t->Write();
373 f.Close();
374 }
375 return StatusCode::SUCCESS;
376
377}
378
379
380StatusCode eFexTowerBuilder::execute(const EventContext& ctx) const {
381 ATH_MSG_DEBUG("Executing " << name() << "...");
382 setFilterPassed(true, ctx);
383
384
385 {
386 std::lock_guard lock(m_fillMapMutex);
387 if (m_scMap.empty()) CHECK( fillMap(ctx) );
388 }
389
390 return fillTowers(ctx);
391
392}
393
394} // LVL1 Namespace
#define M_PI
Scalar eta() const
pseudorapidity method
Scalar phi() const
phi method
#define ATH_CHECK
Evaluate an expression and check for errors.
#define ATH_MSG_FATAL(x)
#define ATH_MSG_INFO(x)
#define ATH_MSG_WARNING(x)
#define ATH_MSG_DEBUG(x)
Helper class for offline supercell identifiers.
#define CHECK(...)
Evaluate an expression and check for errors.
double coord
Type of coordination system.
static Double_t sc
std::string PathResolverFindCalibFile(const std::string &logical_file_name)
virtual void setFilterPassed(bool state, const EventContext &ctx) const
An algorithm that can be simultaneously executed in multiple threads.
Holds information from the LATOME Header.
SG::ReadHandleKey< xAOD::TriggerTowerContainer > m_ttKey
SG::ReadHandleKey< xAOD::EventInfo > m_eiKey
virtual StatusCode execute(const EventContext &ctx) const
StatusCode fillMap(const EventContext &ctx) const
SG::ReadCondHandleKey< CaloSuperCellDetDescrManager > m_ddmKey
Gaudi::Property< bool > m_applyTimingCut
eFexTowerBuilder(const std::string &name, ISvcLocator *pSvcLocator)
SG::ReadHandleKey< LArLATOMEHeaderContainer > m_LArLatomeHeaderContainerKey
Gaudi::Property< bool > m_v6Mapping
ToolHandle< eFEXSuperCellTowerIdProvider > m_eFEXSuperCellTowerIdProviderTool
virtual StatusCode initialize()
Gaudi::Property< std::string > m_mappingFile
SG::ReadHandleKey< CaloCellContainer > m_scellKey
StatusCode fillTowers(const EventContext &ctx) const
SG::WriteHandleKey< xAOD::eFexTowerContainer > m_outKey
Gaudi::Property< bool > m_applyMasking
virtual bool isValid() override final
Can the handle be successfully dereferenced?
StatusCode record(std::unique_ptr< T > data)
Record a const object to the store.
@ IS_SIMULATION
true: simulation, false: data
Description of TriggerTower_v2.
eFexTowerBuilder creates xAOD::eFexTowerContainer from supercells (LATOME) and triggerTowers (TREX) i...
void sort(typename DataModel_detail::iterator< DVL > beg, typename DataModel_detail::iterator< DVL > end)
Specialization of sort for DataVector/List.
void swap(ElementLinkVector< DOBJ > &lhs, ElementLinkVector< DOBJ > &rhs)