81 return StatusCode::FAILURE;
86 return StatusCode::FAILURE;
92 return StatusCode::FAILURE;
97 std::map<std::pair<int,int>,std::array<int,11>> towers;
99 constexpr int INVALID_VALUE = -99999;
100 constexpr int MASKED_VALUE = std::numeric_limits<int>::max();
101 constexpr int SATURATED_VALUE = std::numeric_limits<int>::max()-1;
102 constexpr int MISSING_VALUE = -99998;
104 for (
auto digi: *scells) {
105 const auto itr = m_scMap.find(digi->ID().get_compact());
106 if (itr == m_scMap.end()) {
continue; }
107 int val = std::round(digi->energy()/(12.5*std::cosh(digi->eta())));
110 bool isSaturated = (!isMC) ? (digi->quality()) :
false;
111 bool isMasked = ((digi)->provenance()&0x80);
112 bool isInvalid =
m_applyMasking ? ((digi)->provenance()&0x40) :
false;
118 val = SATURATED_VALUE;
121 auto towerItr = towers.emplace(itr->second.first,std::array<int,11>{});
122 if(towerItr.second) {
123 towerItr.first->second.fill(MISSING_VALUE);
125 auto& tower = (towerItr.first->second);
126 if (itr->second.second.second<11) {
128 if (!isMasked && val!=INVALID_VALUE) {
131 tower.at(itr->second.second.first) = SATURATED_VALUE;
132 tower.at(itr->second.second.second) = SATURATED_VALUE;
134 if(tower.at(itr->second.second.first)!=(SATURATED_VALUE)) {
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;
139 tower.at(itr->second.second.first) += val >> 1;
141 if(tower.at(itr->second.second.second)!=(SATURATED_VALUE)) {
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;
146 tower.at(itr->second.second.second) += (val - (val >> 1));
153 auto& v = tower.at(itr->second.second.first);
156 if(v==MISSING_VALUE) v = MASKED_VALUE;
157 }
else if(isSaturated) {
160 if(v==INVALID_VALUE || v==MISSING_VALUE) v = 0;
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); };
171 if (std::abs(tTower->eta()) > 1.5)
continue;
172 if (tTower->sampling() != 1)
continue;
174 auto towerItr = towers.emplace(std::pair(etaIndex(tTower->eta()),phiIndex(
phi)),std::array<int,11>{});
175 if(towerItr.second) {
176 towerItr.first->second.fill(MISSING_VALUE);
178 (towerItr.first->second).at(10) = tTower->cpET();
183 ATH_CHECK( eTowers.
record(std::make_unique<xAOD::eFexTowerContainer>(),std::make_unique<xAOD::eFexTowerAuxContainer>()) );
185 static const auto calToFex = [](
int calEt) {
186 if(calEt == MASKED_VALUE)
return 0;
187 if(calEt == SATURATED_VALUE)
return 1023;
188 if( calEt == INVALID_VALUE )
return 1022;
189 if( calEt == MISSING_VALUE )
return 1025;
190 if(calEt<448)
return std::max((calEt&~1)/2+32,1);
191 if(calEt<1472)
return (calEt-448)/4+256;
192 if(calEt<3520)
return (calEt-1472)/8+512;
193 if(calEt<11584)
return (calEt-3520)/32+768;
201 for(
auto& [
coord,counts] : towers) {
202 size_t ni = (std::abs(
coord.first)<=15) ? 10 : 11;
203 for(
size_t i=0;i<ni;++i) counts[i] = (scells->empty() ? 1025 : calToFex(counts[i]));
204 eTowers->push_back( std::make_unique<xAOD::eFexTower>() );
205 eTowers->back()->initialize( ( (
coord.first<0 ? 0.5:-0.5) +
coord.first)*0.1 ,
207 std::vector<uint16_t>(counts.begin(), counts.end()),
213 return StatusCode::SUCCESS;
225 return StatusCode::FAILURE;
228 ATH_MSG_FATAL(
"Cannot fill sc -> eFexTower mapping with an incomplete sc collection");
229 return StatusCode::FAILURE;
239 doV6Mapping = (hit->FWversion()>1600);
242 ATH_MSG_WARNING(
"Used LATOME Hardware to determine mapping different to python configuration (use V6 Mapping = " << doV6Mapping <<
" )");
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;
255 static const auto etaIndex = [](
float eta) {
return int(
eta*10 ) + ((
eta<0) ? -1 : 1); };
256 static const auto phiIndex = [](
float phi) {
return int(
phi*32./ROOT::Math::Pi() ) + (
phi<0 ? -1 : 1); };
257 std::map<std::pair<int,int>,TowerSCells> towers;
258 std::map<unsigned long long,int> eTowerSlots;
260 for (
auto digi: *scells) {
263 if (
auto elem = ddm->get_element(
id); elem && std::abs(elem->eta_raw())<2.5) {
264 float eta = elem->eta_raw();
265 int sampling = elem->getSampling();
266 if(sampling==6 && ddm->getCaloCell_ID()->region(
id)==0 &&
eta<0)
eta-=0.01;
268 unsigned long long val =
id.get_compact();
270 int towerid = -1;
int slot = -1;
bool issplit =
false;
272 eTowerSlots[
id.get_compact()] = slot;
274 auto&
sc = towers[std::pair(etaIndex(
eta),phiIndex(elem->phi_raw()))];
277 sc.ps.push_back(val);
280 sc.l1.push_back({elem->eta(),val});
break;
282 sc.l2.push_back({elem->eta(),val});
break;
284 sc.l3.push_back(val);
break;
285 case 8:
case 9:
case 10:
case 11:
286 sc.had.push_back(val);
break;
288 sc.other.push_back(val);
break;
296 std::vector<size_t> slotVector(11);
297 for(
auto& [
coord,
sc] : towers) {
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());
306 sc.l3.push_back(
sc.l2.back().second);
307 sc.l2.resize(
sc.l2.size()-1);
310 if (std::abs(
coord.first)==15) {
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));
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));
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];
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];
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)]; }
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) {
375 return StatusCode::SUCCESS;