Project
Loading...
Searching...
No Matches
Geometry.cxx
Go to the documentation of this file.
1// Copyright 2019-2020 CERN and copyright holders of ALICE O2.
2// See https://alice-o2.web.cern.ch/copyright for details of the copyright holders.
3// All rights not expressly granted are reserved.
4//
5// This software is distributed under the terms of the GNU General Public
6// License v3 (GPL Version 3), copied verbatim in the file "COPYING".
7//
8// In applying this license CERN does not waive the privileges and immunities
9// granted to it by virtue of its status as an Intergovernmental Organization
10// or submit itself to any jurisdiction.
11#include <iomanip>
12
13#include <TGeoBBox.h>
14#include <TGeoManager.h>
15#include <TGeoMatrix.h>
16#include <TList.h>
17
18#include <fairlogger/Logger.h>
19
20#include "EMCALBase/Geometry.h"
22
23#include <boost/algorithm/string/predicate.hpp>
24
25using namespace o2::emcal;
26
27// these initialisations are needed for a singleton
28Geometry* Geometry::sGeom = nullptr;
29
31 : mGeoName(geo.mGeoName),
32 mKey110DEG(geo.mKey110DEG),
33 mnSupModInDCAL(geo.mnSupModInDCAL),
34 mNCellsInSupMod(geo.mNCellsInSupMod),
35 mNETAdiv(geo.mNETAdiv),
36 mNPHIdiv(geo.mNPHIdiv),
37 mNCellsInModule(geo.mNCellsInModule),
38 mPhiBoundariesOfSM(geo.mPhiBoundariesOfSM),
39 mPhiCentersOfSM(geo.mPhiCentersOfSM),
40 mPhiCentersOfSMSec(geo.mPhiCentersOfSMSec),
41 mPhiCentersOfCells(geo.mPhiCentersOfCells),
42 mCentersOfCellsEtaDir(geo.mCentersOfCellsEtaDir),
43 mCentersOfCellsPhiDir(geo.mCentersOfCellsPhiDir),
44 mEtaCentersOfCells(geo.mEtaCentersOfCells),
45 mNCells(geo.mNCells),
46 mNPhi(geo.mNPhi),
47 mCentersOfCellsXDir(geo.mCentersOfCellsXDir),
48 mArm1EtaMin(geo.mArm1EtaMin),
49 mArm1EtaMax(geo.mArm1EtaMax),
50 mArm1PhiMin(geo.mArm1PhiMin),
51 mArm1PhiMax(geo.mArm1PhiMax),
52 mEtaMaxOfTRD1(geo.mEtaMaxOfTRD1),
53 mDCALPhiMin(geo.mDCALPhiMin),
54 mDCALPhiMax(geo.mDCALPhiMax),
55 mEMCALPhiMax(geo.mEMCALPhiMax),
56 mDCALStandardPhiMax(geo.mDCALStandardPhiMax),
57 mDCALInnerExtandedEta(geo.mDCALInnerExtandedEta),
58 mDCALInnerEdge(geo.mDCALInnerEdge),
59 mShishKebabTrd1Modules(geo.mShishKebabTrd1Modules),
60 mPhiModuleSize(geo.mPhiModuleSize),
61 mEtaModuleSize(geo.mEtaModuleSize),
62 mPhiTileSize(geo.mPhiTileSize),
63 mEtaTileSize(geo.mEtaTileSize),
64 mNZ(geo.mNZ),
65 mIPDistance(geo.mIPDistance),
66 mLongModuleSize(geo.mLongModuleSize),
67 mShellThickness(geo.mShellThickness),
68 mZLength(geo.mZLength),
69 mSampling(geo.mSampling),
70 mECPbRadThickness(geo.mECPbRadThickness),
71 mECScintThick(geo.mECScintThick),
72 mNECLayers(geo.mNECLayers),
73 mNumberOfSuperModules(geo.mNumberOfSuperModules),
74 mEMCSMSystem(geo.mEMCSMSystem),
75 mFrontSteelStrip(geo.mFrontSteelStrip),
76 mLateralSteelStrip(geo.mLateralSteelStrip),
77 mPassiveScintThick(geo.mPassiveScintThick),
78 mPhiSuperModule(geo.mPhiSuperModule),
79 mNPhiSuperModule(geo.mNPhiSuperModule),
80 mTrd1Angle(geo.mTrd1Angle),
81 m2Trd1Dx2(geo.m2Trd1Dx2),
82 mPhiGapForSM(geo.mPhiGapForSM),
83 mTrd1AlFrontThick(geo.mTrd1AlFrontThick),
84 mTrd1BondPaperThick(geo.mTrd1BondPaperThick),
85 mILOSS(geo.mILOSS),
86 mIHADR(geo.mIHADR),
87 mSteelFrontThick(geo.mSteelFrontThick), // obsolete data member?
88 mCellIndexLookup(geo.mCellIndexLookup)
89{
90 memcpy(mEnvelop, geo.mEnvelop, sizeof(Float_t) * 3);
91 memcpy(mParSM, geo.mParSM, sizeof(Float_t) * 3);
92
93 memset(SMODULEMATRIX, 0, sizeof(TGeoHMatrix*) * EMCAL_MODULES);
94}
95
96Geometry::Geometry(const std::string_view name, const std::string_view mcname, const std::string_view mctitle)
97 : mGeoName(name),
98 mKey110DEG(0),
99 mnSupModInDCAL(0),
100 mNCellsInSupMod(0),
101 mNETAdiv(0),
102 mNPHIdiv(0),
103 mNCellsInModule(0),
104 mPhiBoundariesOfSM(),
105 mPhiCentersOfSM(),
106 mPhiCentersOfSMSec(),
107 mPhiCentersOfCells(),
108 mCentersOfCellsEtaDir(),
109 mCentersOfCellsPhiDir(),
110 mEtaCentersOfCells(),
111 mNCells(0),
112 mNPhi(0),
113 mCentersOfCellsXDir(),
114 mArm1EtaMin(0),
115 mArm1EtaMax(0),
116 mArm1PhiMin(0),
117 mArm1PhiMax(0),
118 mEtaMaxOfTRD1(0),
119 mDCALPhiMin(0),
120 mDCALPhiMax(0),
121 mEMCALPhiMax(0),
122 mDCALStandardPhiMax(0),
123 mDCALInnerExtandedEta(0),
124 mDCALInnerEdge(0.),
125 mShishKebabTrd1Modules(),
126 mPhiModuleSize(0.),
127 mEtaModuleSize(0.),
128 mPhiTileSize(0.),
129 mEtaTileSize(0.),
130 mNZ(0),
131 mIPDistance(0.),
132 mLongModuleSize(0.),
133 mShellThickness(0.),
134 mZLength(0.),
135 mSampling(0.),
136 mECPbRadThickness(0.),
137 mECScintThick(0.),
138 mNECLayers(0),
139 mNumberOfSuperModules(0),
140 mEMCSMSystem(),
141 mFrontSteelStrip(0.),
142 mLateralSteelStrip(0.),
143 mPassiveScintThick(0.),
144 mPhiSuperModule(0),
145 mNPhiSuperModule(0),
146 mTrd1Angle(0.),
147 m2Trd1Dx2(0.),
148 mPhiGapForSM(0.),
149 mTrd1AlFrontThick(0.0),
150 mTrd1BondPaperThick(0.),
151 mILOSS(-1),
152 mIHADR(-1),
153 mSteelFrontThick(0.) // obsolete data member?
154{
155 DefineEMC(mcname, mctitle);
157
159
161 for (auto icell = 0; icell < mNCells; icell++) {
162 mCellIndexLookup[icell] = CalculateCellIndex(icell);
163 }
164
165 memset(SMODULEMATRIX, 0, sizeof(TGeoHMatrix*) * EMCAL_MODULES);
166
167 LOG(debug) << "Name <<" << name << ">>";
168}
169
171{
172 LOG(fatal) << "assignment operator, not implemented";
173 return *this;
174}
175
177{
178 if (this == sGeom) {
179 LOG(error) << "Do not call delete on me";
180 return;
181 }
182
183 for (Int_t smod = 0; smod < mNumberOfSuperModules; smod++) {
184 if (SMODULEMATRIX[smod]) {
185 delete SMODULEMATRIX[smod];
186 }
187 }
188}
189
191{
192 Geometry* rv = static_cast<Geometry*>(sGeom);
193 if (!rv) {
195 }
196 return rv;
197}
198
199Geometry* Geometry::GetInstance(const std::string_view name, const std::string_view mcname,
200 const std::string_view mctitle)
201{
202 if (!sGeom) {
203 if (!name.length()) { // get default geometry
204 sGeom = new Geometry(DEFAULT_GEOMETRY, mcname, mctitle);
205 } else {
206 sGeom = new Geometry(name, mcname, mctitle);
207 } // end if strcmp(name,"")
208 return sGeom;
209 } else {
210 if (sGeom->GetName() != name) {
211 LOG(info) << "\n current geometry is " << sGeom->GetName() << " : you should not call " << name;
212 } // end
213 return sGeom;
214 } // end if sGeom
215
216 return nullptr;
217}
218
219Geometry* Geometry::GetInstanceFromRunNumber(Int_t runNumber, const std::string_view geoName,
220 const std::string_view mcname, const std::string_view mctitle)
221{
222 using boost::algorithm::contains;
223
224 // printf("AliEMCALGeometry::GetInstanceFromRunNumber() - run %d, geoName <<%s>> \n",runNumber,geoName.Data());
225
226 if (runNumber >= 104064 && runNumber < 140000) {
227 // 2009-2010 runs
228 // First year geometry, 4 SM.
229
230 if (contains(geoName, "FIRSTYEARV1") && geoName != std::string("")) {
231 LOG(info) << "o2::emcal::Geometry::GetInstanceFromRunNumber() *** ATTENTION *** \n"
232 << "\t Specified geometry name <<" << geoName << ">> for run " << runNumber
233 << " is not considered! \n"
234 << "\t In use <<EMCAL_FIRSTYEARV1>>, check run number and year";
235 } else {
236 LOG(info)
237 << "o2::emcal::Geometry::GetInstanceFromRunNumber() - Initialized geometry with name <<EMCAL_FIRSTYEARV1>>";
238 }
239
240 return Geometry::GetInstance("EMCAL_FIRSTYEARV1", mcname, mctitle);
241 } else if (runNumber >= 140000 && runNumber <= 170593) {
242 // Almost complete EMCAL geometry, 10 SM. Year 2011 configuration
243
244 if (contains(geoName, "COMPLETEV1") && geoName != std::string("")) {
245 LOG(info) << "o2::emcal::Geometry::GetInstanceFromRunNumber() *** ATTENTION *** \n"
246 << "\t Specified geometry name <<" << geoName << ">> for run " << runNumber
247 << " is not considered! \n"
248 << "\t In use <<EMCAL_COMPLETEV1>>, check run number and year";
249 } else {
250 LOG(info)
251 << "o2::emcal::Geometry::GetInstanceFromRunNumber() - Initialized geometry with name <<EMCAL_COMPLETEV1>>";
252 }
253 return Geometry::GetInstance("EMCAL_COMPLETEV1", mcname, mctitle);
254 } else if (runNumber > 176000 && runNumber <= 197692) {
255 // Complete EMCAL geometry, 12 SM. Year 2012 and on
256 // The last 2 SM were not active, anyway they were there.
257
258 if (contains(geoName, "COMPLETE12SMV1") && geoName != std::string("")) {
259 LOG(info) << "o2::emcal::Geometry::GetInstanceFromRunNumber() *** ATTENTION *** \n"
260 << "\t Specified geometry name <<" << geoName << " >> for run " << runNumber
261 << " is not considered! \n"
262 << "\t In use <<EMCAL_COMPLETE12SMV1>>, check run number and year";
263 } else {
264 LOG(info) << "o2::emcal::Geometry::GetInstanceFromRunNumber() - Initialized geometry with name "
265 "<<EMCAL_COMPLETE12SMV1>>";
266 }
267 return Geometry::GetInstance("EMCAL_COMPLETE12SMV1", mcname, mctitle);
268 } else // Run 2
269 {
270 // EMCAL + DCAL geometry, 20 SM. Year 2015 and on
271
272 if (contains(geoName, "DCAL_8SM") && geoName != std::string("")) {
273 LOG(info) << "o2::emcal::Geometry::GetInstanceFromRunNumber() *** ATTENTION *** \n"
274 << "\t Specified geometry name <<" << geoName << ">> for run " << runNumber
275 << " is not considered! \n"
276 << "\t In use <<EMCAL_COMPLETE12SMV1_DCAL_8SM>>, check run number and year";
277 } else {
278 LOG(info) << "o2::emcal::Geometry::GetInstanceFromRunNumber() - Initialized geometry with name "
279 "<<EMCAL_COMPLETE12SMV1_DCAL_8SM>>";
280 }
281 return Geometry::GetInstance("EMCAL_COMPLETE12SMV1_DCAL_8SM", mcname, mctitle);
282 }
283}
284
285void Geometry::DefineSamplingFraction(const std::string_view mcname, const std::string_view mctitle)
286{
287 // Jun 05,2006
288 // Look http://rhic.physics.wayne.edu/~pavlinov/ALICE/SHISHKEBAB/RES/linearityAndResolutionForTRD1.html
289 // Keep for compatibility
290 //
291 using boost::algorithm::contains;
292
293 // Sampling factor for G3
294 mSampling = 10.87; // Default value - Nov 25,2010
295 if (mNECLayers == 69) { // 10% layer reduction
296 mSampling = 12.55;
297 } else if (mNECLayers == 61) { // 20% layer reduction
298 mSampling = 12.80;
299 } else if (mNECLayers == 77) {
300 if (contains(mGeoName, "V1")) {
301 mSampling = 10.87; // Adding paper sheets and cover plate; Nov 25,2010
302 } else if (mECScintThick > 0.159 && mECScintThick < 0.161) { // original sampling fraction, equal layers
303 mSampling = 12.327; // fECScintThick = fECPbRadThickness = 0.160;
304 } else if (mECScintThick > 0.175 && mECScintThick < 0.177) { // 10% Pb thicknes reduction
305 mSampling = 10.5; // fECScintThick = 0.176, fECPbRadThickness=0.144;
306 } else if (mECScintThick > 0.191 && mECScintThick < 0.193) { // 20% Pb thicknes reduction
307 mSampling = 8.93; // fECScintThick = 0.192, fECPbRadThickness=0.128;
308 }
309 }
310
311 Float_t samplingFactorTranportModel = 1.;
312 // Note: The sampling factors are chosen so that results from the simulation
313 // engines correspond well with testbeam data
314
315 if (contains(mcname, "Geant3")) {
316 samplingFactorTranportModel = 1.; // 0.988 // Do nothing
317 } else if (contains(mcname, "Fluka")) {
318 samplingFactorTranportModel = 1.; // To be set
319 } else if (contains(mcname, "Geant4")) {
320 std::string physicslist = mctitle.substr(mctitle.find(":") + 2).data();
321 LOG(info) << "Selected physics list: " << physicslist;
322 // sampling factors for different Geant4 physics list
323 // GEANT4 10.7 -> EMCAL-784
324
325 // set a default (there may be many physics list strings)
326 samplingFactorTranportModel = 0.81;
327 if (physicslist == "FTFP_BERT_EMV+optical") {
328 samplingFactorTranportModel = 0.821;
329 } else if (physicslist == "FTFP_BERT_EMV+optical+biasing") {
330 samplingFactorTranportModel = 0.81;
331 } else if (physicslist == "FTFP_INCLXX_EMV+optical") {
332 samplingFactorTranportModel = 0.81;
333 }
334 }
335
336 LOG(info) << "MC modeler <" << mcname << ">, Title <" << mctitle << ">: Sampling " << std::setw(2)
337 << std::setprecision(3) << mSampling << ", model fraction with respect to G3 "
338 << samplingFactorTranportModel << ", final sampling " << mSampling * samplingFactorTranportModel;
339
340 mSampling *= samplingFactorTranportModel;
341}
342
343void Geometry::DefineEMC(std::string_view mcname, std::string_view mctitle)
344{
345 using boost::algorithm::contains;
346
347 // geometry
348 std::transform(mGeoName.begin(), mGeoName.end(), mGeoName.begin(), ::toupper);
349
350 // Convert old geometry names to new ones
351 if (contains(mGeoName, "SHISH_77_TRD1_2X2_FINAL_110DEG")) {
352 if (contains(mGeoName, "PBTH=0.144") && contains(mGeoName, "SCTH=0.176")) {
353 mGeoName = "EMCAL_COMPLETE";
354 } else {
355 mGeoName = "EMCAL_PDC06";
356 }
357 }
358
359 if (contains(mGeoName, "WSUC")) {
360 mGeoName = "EMCAL_WSUC";
361 }
362
363 // check that we have a valid geometry name
364 if (!(contains(mGeoName, "EMCAL_PDC06") || contains(mGeoName, "EMCAL_WSUC") || contains(mGeoName, "EMCAL_COMPLETE") ||
365 contains(mGeoName, "EMCAL_COMPLETEV1") || contains(mGeoName, "EMCAL_COMPLETE12SMV1") ||
366 contains(mGeoName, "EMCAL_FIRSTYEAR") || contains(mGeoName, "EMCAL_FIRSTYEARV1"))) {
367 LOG(fatal) << "Init, " << mGeoName << " is an undefined geometry!\n";
368 }
369
370 // Option to know whether we have the "half" supermodule(s) or not
371 mKey110DEG = 0;
372 if (contains(mGeoName, "COMPLETE") || contains(mGeoName, "PDC06") || contains(mGeoName, "12SM")) {
373 mKey110DEG = 1; // for GetAbsCellId
374 }
375 if (contains(mGeoName, "COMPLETEV1")) {
376 mKey110DEG = 0;
377 }
378
379 mnSupModInDCAL = 0;
380 if (contains(mGeoName, "DCAL_DEV")) {
381 mnSupModInDCAL = 10;
382 } else if (contains(mGeoName, "DCAL_8SM")) {
383 mnSupModInDCAL = 8;
384 } else if (contains(mGeoName, "DCAL")) {
385 mnSupModInDCAL = 6;
386 }
387
388 // JLK 13-Apr-2008
389 // default parameters are those of EMCAL_COMPLETE geometry
390 // all others render variations from these at the end of
391 // geometry-name specific options
392
393 mNumberOfSuperModules = 12; // 12 = 6 * 2 (6 in phi, 2 in Z)
394 mNPhi = 12; // module granularity in phi within smod (azimuth)
395 mNZ = 24; // module granularity along Z within smod (eta)
396 mNPHIdiv = mNETAdiv = 2; // tower granularity within module
397 mArm1PhiMin = 80.0; // degrees, Starting EMCAL Phi position
398 mArm1PhiMax = 200.0; // degrees, Ending EMCAL Phi position
399 mArm1EtaMin = -0.7; // pseudorapidity, Starting EMCAL Eta position
400 mArm1EtaMax = +0.7; // pseudorapidity, Ending EMCAL Eta position
401 mIPDistance = 428.0; // cm, radial distance to front face from nominal vertex point
402 mPhiGapForSM = 2.; // cm, only for final TRD1 geometry
403 mFrontSteelStrip = 0.025; // 0.025cm = 0.25mm (13-may-05 from V.Petrov)
404 mPassiveScintThick = 0.8; // 0.8cm = 8mm (13-may-05 from V.Petrov)
405 mLateralSteelStrip = 0.01; // 0.01cm = 0.1mm (13-may-05 from V.Petrov) - was 0.025
406 mTrd1Angle = 1.5; // in degrees
407
408 mSampling = 1.; // should be calculated with call to DefineSamplingFraction()
409 mNECLayers = 77; // (13-may-05 from V.Petrov) - can be changed with additional options
410 mECScintThick = 0.176; // scintillator layer thickness
411 mECPbRadThickness = 0.144; // lead layer thickness
412
413 mPhiModuleSize = 12.26 - mPhiGapForSM / Float_t(mNPhi); // first assumption
415
416 mZLength = 700.; // Z coverage (cm)
417 mPhiSuperModule = 20.; // phi in degree
418 mDCALInnerEdge = mIPDistance * TMath::Tan(mTrd1Angle * 8. * TMath::DegToRad());
419
420 // modifications to the above for PDC06 geometry
421 if (contains(mGeoName, "PDC06")) { // 18-may-05 - about common structure
422 mECScintThick = mECPbRadThickness = 0.16; // (13-may-05 from V.Petrov)
423 }
424
425 // modifications to the above for WSUC geometry
426 if (contains(mGeoName, "WSUC")) { // 18-may-05 - about common structure
427 mNumberOfSuperModules = 2; // 27-may-05; Nov 24,2010 for TB
428 mNPhi = mNZ = 4;
429 mTrd1AlFrontThick = 1.0; // one cm
430 // Bond paper - two sheets around Sc tile
431 mTrd1BondPaperThick = 0.01; // 0.01cm = 0.1 mm
432
433 mPhiModuleSize = 12.0;
435 mLateralSteelStrip = 0.015; // 0.015cm = 0.15mm
436 }
437
438 // In 2009-2010 data taking runs only 4 SM, in the upper position.
439 if (contains(mGeoName, "FIRSTYEAR")) {
441 mArm1PhiMax = 120.0;
442 }
443
444 if (contains(mGeoName, "FIRSTYEARV1") || contains(mGeoName, "COMPLETEV1") || contains(mGeoName, "COMPLETE12SMV1")) {
445 // Oct 26,2010 : First module has tilt = 0.75 degree :
446 // look to AliEMCALShishKebabTrd1Module::DefineFirstModule(key)
447 // New sizes from production drawing, added Al front plate.
448 // The thickness of sampling is change due to existing two sheets of paper.
449
450 // Will replace fFrontSteelStrip
451 mTrd1AlFrontThick = 1.0; // one cm
452 // Bond paper - two sheets around Sc tile
453 mTrd1BondPaperThick = 0.01; // 0.01cm = 0.1 mm
454
455 mPhiModuleSize = 12.0;
457 mLateralSteelStrip = 0.015; // 0.015cm = 0.15mm
458
459 if (contains(mGeoName, "COMPLETEV1")) {
461 mArm1PhiMax = 180.0;
462 } else if (contains(mGeoName, "COMPLETE12SMV1")) {
464 mArm1PhiMax = 200.0;
465 }
466 if (contains(mGeoName, "DCAL")) {
468 mArm1PhiMax = 320.0;
469 if (contains(mGeoName, "DCAL_8SM")) {
470 mArm1PhiMax = 340.0; // degrees, End of DCAL Phi position
471 } else if (contains(mGeoName, "DCAL_DEV")) {
472 mArm1PhiMin = 40.0; // degrees, Starting EMCAL(shifted) Phi position
473 }
475 }
476 }
477
478 //
479 // Init EMCal/DCal SMs type array
480 mEMCSMSystem.clear();
482
483 for (Int_t i = 0; i < mNumberOfSuperModules; i++) {
485 }
486
487 Int_t iSM = 0;
488
489 //
490 // BASIC EMCAL SM
491 if (contains(mGeoName, "WSUC")) {
492 for (int i = 0; i < 2; i++) {
494 iSM++;
495 }
496 } else if (contains(mGeoName, "FIRSTYEAR")) {
497 for (int i = 0; i < 4; i++) {
499 iSM++;
500 }
501 } else if (contains(mGeoName, "PDC06") || contains(mGeoName, "COMPLETE")) {
502 for (int i = 0; i < 10; i++) {
504 iSM++;
505 }
506 }
507
508 //
509 // EMCAL 110SM
510 if (mKey110DEG && contains(mGeoName, "12SM")) {
511 for (int i = 0; i < 2; i++) {
513 if (contains(mGeoName, "12SMV1")) {
515 }
516 iSM++;
517 }
518 }
519
520 //
521 // DCAL SM
522 if (mnSupModInDCAL && contains(mGeoName, "DCAL")) {
523 if (contains(mGeoName, "8SM")) {
524 for (int i = 0; i < mnSupModInDCAL - 2; i++) {
526 iSM++;
527 }
528 for (int i = 0; i < 2; i++) {
529 mEMCSMSystem[iSM] = DCAL_EXT;
530 iSM++;
531 }
532 } else {
533 for (int i = 0; i < mnSupModInDCAL; i++) {
535 iSM++;
536 }
537 }
538 }
539
540 // constant for transition absid <--> indexes
543 mNCells = 0;
544 for (int i = 0; i < mNumberOfSuperModules; i++) {
545 if (GetSMType(i) == EMCAL_STANDARD) {
547 } else if (GetSMType(i) == EMCAL_HALF) {
549 } else if (GetSMType(i) == EMCAL_THIRD) {
551 } else if (GetSMType(i) == DCAL_STANDARD) {
552 mNCells += 2 * mNCellsInSupMod / 3;
553 } else if (GetSMType(i) == DCAL_EXT) {
555 } else {
556 LOG(error) << "Uknown SuperModule Type !!\n";
557 }
558 }
559
561 if (mNPhiSuperModule < 1) {
563 }
564
565 mPhiTileSize = mPhiModuleSize / double(mNPHIdiv) - mLateralSteelStrip; // 13-may-05
566 mEtaTileSize = mEtaModuleSize / double(mNETAdiv) - mLateralSteelStrip; // 13-may-05
567
569 if (contains(mGeoName, "V1")) {
570 Double_t ws = mECScintThick + mECPbRadThickness + 2. * mTrd1BondPaperThick; // sampling width
571 // Number of Pb tiles = Number of Sc tiles - 1
573 }
574 m2Trd1Dx2 = mEtaModuleSize + 2. * mLongModuleSize * TMath::Tan(mTrd1Angle * TMath::DegToRad() / 2.);
575
576 if (!contains(mGeoName, "WSUC")) {
578 }
579
580 // These parameters are used to create the mother volume to hold the supermodules
581 // 2cm padding added to allow for misalignments - JLK 30-May-2008
582 mEnvelop[0] = mIPDistance - 1.; // mother volume inner radius
583 mEnvelop[1] = mIPDistance + mShellThickness + 1.; // mother volume outer r.
584 mEnvelop[2] = mZLength + 2.; // mother volume length
585
586 // Local coordinates
587 mParSM[0] = GetShellThickness() / 2.;
588 mParSM[1] = GetPhiModuleSize() * GetNPhi() / 2.;
589 mParSM[2] = mZLength / 4.; // divide by 4 to get half-length of SM
590
591 // SM phi boundaries - (0,1),(2,3) ... - has the same boundaries;
595 Double_t kfSupermodulePhiWidth = mPhiSuperModule * TMath::DegToRad();
596 mPhiCentersOfSM[0] = (mArm1PhiMin + mPhiSuperModule / 2.) * TMath::DegToRad(); // Define from First SM
597 mPhiCentersOfSMSec[0] = mPhiCentersOfSM[0]; // the same in the First SM
598 mPhiBoundariesOfSM[0] = mPhiCentersOfSM[0] - TMath::ATan2(mParSM[1], mIPDistance); // 1th and 2th modules)
599 mPhiBoundariesOfSM[1] = mPhiCentersOfSM[0] + TMath::ATan2(mParSM[1], mIPDistance);
600
601 if (mNumberOfSuperModules > 2) { // 2 to Max
602 Int_t tmpSMType = GetSMType(2);
603 for (int i = 1; i < mNPhiSuperModule; i++) {
604 mPhiBoundariesOfSM[2 * i] += mPhiBoundariesOfSM[2 * i - 2] + kfSupermodulePhiWidth;
605 if (tmpSMType == GetSMType(2 * i)) {
606 mPhiBoundariesOfSM[2 * i + 1] += mPhiBoundariesOfSM[2 * i - 1] + kfSupermodulePhiWidth;
607 } else {
608 // changed SM Type, redefine the [2*i+1] Boundaries
609 tmpSMType = GetSMType(2 * i);
610 switch (GetSMType(2 * i)) {
611 case EMCAL_STANDARD:
612 mPhiBoundariesOfSM[2 * i + 1] = mPhiBoundariesOfSM[2 * i] + kfSupermodulePhiWidth;
613 break;
614 case EMCAL_HALF:
615 mPhiBoundariesOfSM[2 * i + 1] = mPhiBoundariesOfSM[2 * i] + 2. * TMath::ATan2((mParSM[1]) / 2, mIPDistance);
616 break;
617 case EMCAL_THIRD:
618 mPhiBoundariesOfSM[2 * i + 1] = mPhiBoundariesOfSM[2 * i] + 2. * TMath::ATan2((mParSM[1]) / 3, mIPDistance);
619 break;
620 case DCAL_STANDARD:
621 mPhiBoundariesOfSM[2 * i] = (mDCALPhiMin - mArm1PhiMin) * TMath::DegToRad() + mPhiBoundariesOfSM[0];
622 mPhiBoundariesOfSM[2 * i + 1] = (mDCALPhiMin - mArm1PhiMin) * TMath::DegToRad() + mPhiBoundariesOfSM[1];
623 break;
624 case DCAL_EXT:
625 mPhiBoundariesOfSM[2 * i + 1] = mPhiBoundariesOfSM[2 * i] + 2. * TMath::ATan2((mParSM[1]) / 3, mIPDistance);
626 break;
627 default:
628 break;
629 };
630 }
631 mPhiCentersOfSM[i] = (mPhiBoundariesOfSM[2 * i] + mPhiBoundariesOfSM[2 * i + 1]) / 2.;
632 mPhiCentersOfSMSec[i] = mPhiBoundariesOfSM[2 * i] + TMath::ATan2(mParSM[1], mIPDistance);
633 }
634 }
635
636 // inner extend in eta (same as outer part) for DCal (0.189917), //calculated from the smallest gap (1# cell to the
637 // 80-degree-edge),
638 const double INNNER_EXTENDED_PHI =
639 1.102840997; // calculated from the smallest gap (1# cell to the 80-degree-edge), too complicatd to explain...
640 mDCALInnerExtandedEta = -TMath::Log(
641 TMath::Tan((TMath::Pi() / 2. - 8 * mTrd1Angle * TMath::DegToRad() +
642 (TMath::Pi() / 2 - mNZ * mTrd1Angle * TMath::DegToRad() - TMath::ATan(TMath::Exp(mArm1EtaMin)) * 2)) /
643 2.));
644
646 mDCALPhiMax = mDCALPhiMin; // DCAl extention will not be included
647 for (Int_t i = 0; i < mNumberOfSuperModules; i += 2) {
648 switch (GetSMType(i)) {
649 case EMCAL_STANDARD:
650 mEMCALPhiMax += 20.;
651 break;
652 case EMCAL_HALF:
653 mEMCALPhiMax += mPhiSuperModule / 2. + INNNER_EXTENDED_PHI;
654 break;
655 case EMCAL_THIRD:
656 mEMCALPhiMax += mPhiSuperModule / 3. + 4.0 * INNNER_EXTENDED_PHI / 3.0;
657 break;
658 case DCAL_STANDARD:
659 mDCALPhiMax += 20.;
661 break;
662 case DCAL_EXT:
663 mDCALPhiMax += mPhiSuperModule / 3. + 4.0 * INNNER_EXTENDED_PHI / 3.0;
664 break;
665 default:
666 LOG(error) << "Unkown SM Type!!\n";
667 break;
668 };
669 }
670 // for compatible reason
671 // if(fNumberOfSuperModules == 4) {fEMCALPhiMax = fArm1PhiMax ;}
672 if (mNumberOfSuperModules == 12) {
674 }
675
676 // called after setting of scintillator and lead layer parameters
677 // called now in AliEMCALv0::CreateGeometry() - 15/03/16
678 // DefineSamplingFraction(mcname,mctitle);
679}
680
681void Geometry::GetGlobal(const Double_t* loc, Double_t* glob, int iSM) const
682{
683 const TGeoHMatrix* m = GetMatrixForSuperModule(iSM);
684 if (m) {
685 m->LocalToMaster(loc, glob);
686 } else {
687 LOG(fatal) << "Geo matrixes are not loaded \n";
688 }
689}
690
691void Geometry::GetGlobal(const TVector3& vloc, TVector3& vglob, int iSM) const
692{
693 Double_t tglob[3], tloc[3];
694 vloc.GetXYZ(tloc);
695 GetGlobal(tloc, tglob, iSM);
696 vglob.SetXYZ(tglob[0], tglob[1], tglob[2]);
697}
698
699void Geometry::GetGlobal(Int_t absId, Double_t glob[3]) const
700{
701 double loc[3];
702
703 memset(glob, 0, sizeof(Double_t) * 3);
704 try {
705 auto cellpos = RelPosCellInSModule(absId);
706 loc[0] = cellpos.X();
707 loc[1] = cellpos.Y();
708 loc[2] = cellpos.Z();
709 } catch (InvalidCellIDException& e) {
710 LOG(error) << e.what();
711 return;
712 }
713
714 Int_t nSupMod = std::get<0>(GetCellIndex(absId));
715 const TGeoHMatrix* m = GetMatrixForSuperModule(nSupMod);
716 if (m) {
717 m->LocalToMaster(loc, glob);
718 } else {
719 LOG(fatal) << "Geo matrixes are not loaded \n";
720 }
721}
722
723void Geometry::GetGlobal(Int_t absId, TVector3& vglob) const
724{
725 Double_t glob[3];
726
727 GetGlobal(absId, glob);
728 vglob.SetXYZ(glob[0], glob[1], glob[2]);
729}
730
731std::tuple<double, double> Geometry::EtaPhiFromIndex(Int_t absId) const
732{
733 TVector3 vglob;
734 GetGlobal(absId, vglob);
735 return std::make_tuple(vglob.Eta(), vglob.Phi());
736}
737
738int Geometry::GetAbsCellId(int supermoduleID, int moduleID, int phiInModule, int etaInModule) const
739{
740 // 0 <= nSupMod < fNumberOfSuperModules
741 // 0 <= nModule < fNPHI * fNZ ( fNPHI * fNZ/2 for fKey110DEG=1)
742 // 0 <= nIphi < fNPHIdiv
743 // 0 <= nIeta < fNETAdiv
744 // 0 <= absid < fNCells
745 int cellid = 0; // have to change from 0 to fNCells-1
746 for (int i = 0; i < supermoduleID; i++) {
747 switch (GetSMType(i)) {
748 case EMCAL_STANDARD:
749 cellid += mNCellsInSupMod;
750 break;
751 case EMCAL_HALF:
752 cellid += mNCellsInSupMod / 2;
753 break;
754 case EMCAL_THIRD:
755 cellid += mNCellsInSupMod / 3;
756 break;
757 case DCAL_STANDARD:
758 cellid += 2 * mNCellsInSupMod / 3;
759 break;
760 case DCAL_EXT:
761 cellid += mNCellsInSupMod / 3;
762 break;
763 default:
765 };
766 }
767
768 cellid += mNCellsInModule * moduleID;
769 cellid += mNPHIdiv * phiInModule;
770 cellid += etaInModule;
771 if (!CheckAbsCellId(cellid)) {
772 throw InvalidCellIDException(cellid);
773 }
774
775 return cellid;
776}
777
778std::tuple<int, int, int> Geometry::GetModuleIndexesFromCellIndexesInSModule(int supermoduleID, int phiInSupermodule, int etaInSupermodule) const
779{
780 int nModulesInSMPhi = GetNumberOfModuleInPhiDirection(supermoduleID);
781
782 int moduleEta = etaInSupermodule / mNETAdiv,
783 modulePhi = phiInSupermodule / mNPHIdiv,
784 moduleID = moduleEta * nModulesInSMPhi + modulePhi;
785 int etaInModule = etaInSupermodule % mNETAdiv,
786 phiInModule = phiInSupermodule % mNPHIdiv;
787 // return std::make_tuple(modulePhi, moduleEta, moduleID);
788 return std::make_tuple(phiInModule, etaInModule, moduleID);
789}
790
791Int_t Geometry::GetAbsCellIdFromCellIndexes(Int_t nSupMod, Int_t iphi, Int_t ieta) const
792{
793 // Check if the indeces correspond to existing SM or tower indeces
794 if (iphi < 0 || iphi >= EMCAL_ROWS || ieta < 0 || ieta >= EMCAL_COLS || nSupMod < 0 ||
795 nSupMod >= GetNumberOfSuperModules()) {
796 LOG(debug) << "Wrong cell indexes : SM " << nSupMod << ", column (eta) " << ieta << ", row (phi) " << iphi;
797 return -1;
798 }
799
800 auto indexmod = GetModuleIndexesFromCellIndexesInSModule(nSupMod, iphi, ieta);
801
802 Int_t nIeta = ieta % mNETAdiv,
803 nIphi = iphi % mNPHIdiv;
804 nIeta = mNETAdiv - 1 - nIeta;
805 return GetAbsCellId(nSupMod, std::get<2>(indexmod), nIphi, nIeta);
806}
807
808std::tuple<int, int> Geometry::GlobalRowColFromIndex(int cellID) const
809{
810 if (!CheckAbsCellId(cellID)) {
811 throw InvalidCellIDException(cellID);
812 }
813 auto [supermodule, module, phiInModule, etaInModule] = GetCellIndex(cellID);
814 auto [row, col] = GetCellPhiEtaIndexInSModule(supermodule, module, phiInModule, etaInModule);
815 // add offsets (row / col per supermodule)
816 if (supermodule == 13 || supermodule == 15 || supermodule == 17) {
817 // DCal odd SMs need shift of the col. index in oder to get the global col. index
818 col += 16;
819 }
820 if (supermodule % 2) {
821 col += mNZ * 2;
822 }
823 int sector = supermodule / 2;
824 if (sector > 0) {
825 for (int isec = 0; isec < sector; isec++) {
826 auto smtype = GetSMType(isec * 2);
827 auto nphism = (smtype == EMCAL_THIRD || smtype == DCAL_EXT) ? GetNPhi() / 3 : GetNPhi();
828 row += 2 * nphism;
829 }
830 }
831 return std::make_tuple(row, col);
832}
833
834std::tuple<int, int, int> Geometry::GetPositionInSupermoduleFromGlobalRowCol(int row, int col) const
835{
836 if (col < 0 || col >= 4 * GetNEta()) {
837 throw RowColException(row, col);
838 }
839 int side = col < GetNEta() * 2 ? 0 : 1,
840 colSM = col % (GetNEta() * 2);
841 int sector = -1,
842 rowSM = row;
843 for (int isec = 0; isec < GetNPhiSuperModule(); isec++) {
844 auto smtype = GetSMType(isec * 2);
845 auto nphism = GetNPhi() * 2;
846 if (smtype == EMCAL_THIRD || smtype == DCAL_EXT) {
847 nphism /= 3;
848 }
849 if (rowSM < nphism) {
850 sector = isec;
851 break;
852 }
853 rowSM -= nphism;
854 }
855 if (sector < 0) {
856 throw RowColException(row, col);
857 }
858 int supermodule = sector * 2 + side;
859 if (supermodule == 13 || supermodule == 15 || supermodule == 17) {
860 // DCal odd SMs need shift of the col. index as global col index includes PHOS hole
861 colSM -= 16;
862 if (colSM < 0) {
863 throw RowColException(row, col); // Position inside PHOS hole specified
864 }
865 }
866 if (supermodule == 12 || supermodule == 14 || supermodule == 16) {
867 if (colSM > 32) {
868 throw RowColException(row, col); // Position inside PHOS hole specified
869 }
870 }
871 return std::make_tuple(supermodule, rowSM, colSM);
872}
873
875{
877 return GetAbsCellIdFromCellIndexes(supermodule, rowSM, colSM);
878}
879
880std::tuple<int, int, int, int> Geometry::GetCellIndexFromGlobalRowCol(int row, int col) const
881{
883 auto indexmod = GetModuleIndexesFromCellIndexesInSModule(supermodule, rowSM, colSM);
884
885 Int_t colInModule = colSM % mNETAdiv,
886 rowInMOdule = rowSM % mNPHIdiv;
887 colInModule = mNETAdiv - 1 - colInModule;
888 return std::make_tuple(supermodule, std::get<2>(indexmod), rowInMOdule, colInModule);
889}
890
891int Geometry::GlobalCol(int cellID) const
892{
893 return std::get<1>(GlobalRowColFromIndex(cellID));
894}
895
896int Geometry::GlobalRow(int cellID) const
897{
898 return std::get<0>(GlobalRowColFromIndex(cellID));
899}
900
901Int_t Geometry::SuperModuleNumberFromEtaPhi(Double_t eta, Double_t phi) const
902{
903 if (TMath::Abs(eta) > mEtaMaxOfTRD1) {
904 throw InvalidPositionException(eta, phi);
905 }
906
907 phi = TVector2::Phi_0_2pi(phi); // move phi to (0,2pi) boundaries
908 Int_t nphism = mNumberOfSuperModules / 2;
909 Int_t nSupMod = 0;
910 for (Int_t i = 0; i < nphism; i++) {
911 LOG(debug) << "Sec " << i << ": Min " << mPhiBoundariesOfSM[2 * i] << ", Max " << mPhiBoundariesOfSM[2 * i + 1];
912 if (phi >= mPhiBoundariesOfSM[2 * i] && phi <= mPhiBoundariesOfSM[2 * i + 1]) {
913 nSupMod = 2 * i;
914 if (eta < 0.0) {
915 nSupMod++;
916 }
917
918 if (GetSMType(nSupMod) == DCAL_STANDARD) { // Gap between DCAL
919 if (TMath::Abs(eta) < GetNEta() / 3 * mTrd1Angle * TMath::DegToRad()) {
920 throw InvalidPositionException(eta, phi);
921 }
922 }
923
924 LOG(debug) << "eta " << eta << " phi " << phi << " (" << std::setw(5) << std::setprecision(2)
925 << phi * TMath::RadToDeg() << ") : nSupMod " << nSupMod << ": #bound " << i;
926 return nSupMod;
927 }
928 }
929 throw InvalidPositionException(eta, phi);
930}
931
932Int_t Geometry::GetAbsCellIdFromEtaPhi(Double_t eta, Double_t phi) const
933{
934 Int_t nSupMod = SuperModuleNumberFromEtaPhi(eta, phi);
935
936 // phi index first
937 phi = TVector2::Phi_0_2pi(phi);
938 Double_t phiLoc = phi - mPhiCentersOfSMSec[nSupMod / 2];
939 Int_t nphi = mPhiCentersOfCells.size();
940 switch (GetSMType(nSupMod)) {
941 case EMCAL_HALF:
942 nphi /= 2;
943 case EMCAL_THIRD:
944 case DCAL_EXT:
945 nphi /= 3;
946 break;
947 default:
948 // All other supermodules have full number of cells in phi
949 break;
950 };
951
952 Double_t dmin = TMath::Abs(mPhiCentersOfCells[0] - phiLoc),
953 d = 0.;
954 Int_t iphi = 0;
955 for (Int_t i = 1; i < nphi; i++) {
956 d = TMath::Abs(mPhiCentersOfCells[i] - phiLoc);
957 if (d < dmin) {
958 dmin = d;
959 iphi = i;
960 }
961 // printf(" i %i : d %f : dmin %f : fPhiCentersOfCells[i] %f \n", i, d, dmin, fPhiCentersOfCells[i]);
962 }
963 // odd SM are turned with respect of even SM - reverse indexes
964 LOG(debug2) << " iphi " << iphi << " : dmin " << dmin << " (phi " << phi << ", phiLoc " << phiLoc << ")\n";
965
966 // eta index
967 Double_t absEta = TMath::Abs(eta);
968 Int_t neta = mCentersOfCellsEtaDir.size(),
969 etaShift = iphi * neta,
970 ieta = 0;
971 if (GetSMType(nSupMod) == DCAL_STANDARD) {
972 ieta += 16; // jump 16 cells for DCSM
973 }
974 dmin = TMath::Abs(mEtaCentersOfCells[etaShift + ieta] - absEta);
975 for (Int_t i = ieta + 1; i < neta; i++) {
976 d = TMath::Abs(mEtaCentersOfCells[i + etaShift] - absEta);
977 if (d < dmin) {
978 dmin = d;
979 ieta = i;
980 }
981 }
982
983 if (GetSMType(nSupMod) == DCAL_STANDARD) {
984 ieta -= 16; // jump 16 cells for DCSM
985 }
986
987 LOG(debug2) << " ieta " << ieta << " : dmin " << dmin << " (eta=" << eta << ") : nSupMod " << nSupMod;
988
989 // patch for mapping following alice convention
990 if (nSupMod % 2 ==
991 0) { // 47 + 16 -ieta for DCSM, 47 - ieta for others, revert the ordering on A side in order to keep convention.
992 ieta = (neta - 1) - ieta;
993 if (GetSMType(nSupMod) == DCAL_STANDARD) {
994 ieta -= 16; // recover cells for DCSM
995 }
996 }
997
998 return GetAbsCellIdFromCellIndexes(nSupMod, iphi, ieta);
999}
1000
1001std::tuple<int, int, int, int> Geometry::CalculateCellIndex(Int_t absId) const
1002{
1003 if (!CheckAbsCellId(absId)) {
1004 throw InvalidCellIDException(absId);
1005 }
1006
1007 Int_t tmp = absId;
1008 Int_t test = absId;
1009
1010 Int_t nSupMod;
1011 for (nSupMod = -1; test >= 0;) {
1012 nSupMod++;
1013 tmp = test;
1014 switch (GetSMType(nSupMod)) {
1015 case EMCAL_STANDARD:
1017 break;
1018 case EMCAL_HALF:
1019 test -= mNCellsInSupMod / 2;
1020 break;
1021 case DCAL_STANDARD:
1022 test -= 2 * mNCellsInSupMod / 3;
1023 break;
1024 case EMCAL_THIRD:
1025 case DCAL_EXT:
1026 test -= mNCellsInSupMod / 3;
1027 break;
1028 default:
1030 };
1031 }
1032
1033 Int_t nModule = tmp / mNCellsInModule;
1034 tmp = tmp % mNCellsInModule;
1035 Int_t nIphi = tmp / mNPHIdiv, nIeta = tmp % mNETAdiv;
1036 return std::make_tuple(nSupMod, nModule, nIphi, nIeta);
1037}
1038
1039std::tuple<int, int, int, int> Geometry::GetCellIndex(Int_t absId) const
1040{
1041 if (!CheckAbsCellId(absId)) {
1042 throw InvalidCellIDException(absId);
1043 }
1044 return mCellIndexLookup[absId];
1045}
1046
1047Int_t Geometry::GetSuperModuleNumber(Int_t absId) const { return std::get<0>(GetCellIndex(absId)); }
1048
1049std::tuple<int, int> Geometry::GetModulePhiEtaIndexInSModule(int supermoduleID, int moduleID) const
1050{
1051 int nModulesInPhi = -1;
1052 switch (GetSMType(supermoduleID)) {
1053 case EMCAL_HALF:
1054 nModulesInPhi = mNPhi / 2; // halfSM
1055 break;
1056 case EMCAL_THIRD:
1057 case DCAL_EXT:
1058 nModulesInPhi = mNPhi / 3; // 1/3 SM
1059 break;
1060 default:
1061 nModulesInPhi = mNPhi; // full SM
1062 break;
1063 };
1064 return std::make_tuple(int(moduleID % nModulesInPhi), int(moduleID / nModulesInPhi));
1065}
1066
1067std::tuple<int, int> Geometry::GetCellPhiEtaIndexInSModule(int supermoduleID, int moduleID, int phiInModule,
1068 int etaInModule) const
1069{
1070 auto [phiOfModule, etaOfModule] = GetModulePhiEtaIndexInSModule(supermoduleID, moduleID);
1071
1072 // ieta = etaOfModule*fNETAdiv + (1-etaInModule); // x(module) = -z(SM)
1073 int etaInSupermodule = etaOfModule * mNETAdiv + (mNETAdiv - 1 - etaInModule); // x(module) = -z(SM)
1074 int phiInSupermodule = phiOfModule * mNPHIdiv + phiInModule; // y(module) = y(SM)
1075
1076 if (phiInSupermodule < 0 || etaInSupermodule < 0) {
1077 LOG(debug) << " Supermodule " << supermoduleID << ", Module " << moduleID << " (phi " << phiInModule << ", eta " << etaInModule << ")"
1078 << " => in Supermodule: eta " << etaInSupermodule << ", phi " << phiInSupermodule;
1079 }
1080 return std::make_tuple(phiInSupermodule, etaInSupermodule);
1081}
1082
1083std::tuple<int, int> Geometry::ShiftOnlineToOfflineCellIndexes(Int_t supermoduleID, Int_t iphi, Int_t ieta) const
1084{
1085 if (supermoduleID == 13 || supermoduleID == 15 || supermoduleID == 17) {
1086 // DCal odd SMs
1087 ieta -= 16; // Same cabling mapping as for EMCal, not considered offline.
1088 } else if (supermoduleID == 18 || supermoduleID == 19) {
1089 // DCal 1/3 SMs
1090 iphi -= 16; // Needed due to cabling mistake.
1091 }
1092 return std::tuple<int, int>(iphi, ieta);
1093}
1094
1095std::tuple<int, int> Geometry::ShiftOfflineToOnlineCellIndexes(Int_t supermoduleID, Int_t iphi, Int_t ieta) const
1096{
1097 if (supermoduleID == 13 || supermoduleID == 15 || supermoduleID == 17) {
1098 // DCal odd SMs
1099 ieta += 16; // Same cabling mapping as for EMCal, not considered offline.
1100 } else if (supermoduleID == 18 || supermoduleID == 19) {
1101 // DCal 1/3 SMs
1102 iphi += 16; // Needed due to cabling mistake.
1103 }
1104 return std::tuple<int, int>(iphi, ieta);
1105}
1106
1108{
1109 // Shift index taking into account the difference between standard SM
1110 // and SM of half (or one third) size in phi direction
1111
1112 Int_t phiindex = mCentersOfCellsPhiDir.size();
1113 Double_t zshift = 0.5 * GetDCALInnerEdge();
1114 Double_t xr, yr, zr;
1115
1116 if (!CheckAbsCellId(absId)) {
1117 throw InvalidCellIDException(absId);
1118 }
1119
1120 auto cellindex = GetCellIndex(absId);
1121 Int_t nSupMod = std::get<0>(cellindex), nModule = std::get<1>(cellindex), nIphi = std::get<2>(cellindex),
1122 nIeta = std::get<3>(cellindex);
1123 auto indexinsm = GetCellPhiEtaIndexInSModule(nSupMod, nModule, nIphi, nIeta);
1124 Int_t iphi = std::get<0>(indexinsm), ieta = std::get<1>(indexinsm);
1125
1126 // Get eta position. Careful with ALICE conventions (increase index decrease eta)
1127 Int_t ieta2 = ieta;
1128 if (nSupMod % 2 == 0) {
1129 ieta2 = (mCentersOfCellsEtaDir.size() - 1) -
1130 ieta; // 47-ieta, revert the ordering on A side in order to keep convention.
1131 }
1132
1133 if (GetSMType(nSupMod) == DCAL_STANDARD && nSupMod % 2) {
1134 ieta2 += 16; // DCAL revert the ordering on C side ...
1135 }
1136 zr = mCentersOfCellsEtaDir[ieta2];
1137 if (GetSMType(nSupMod) == DCAL_STANDARD) {
1138 zr -= zshift; // DCAL shift (SMALLER SM)
1139 }
1140 xr = mCentersOfCellsXDir[ieta2];
1141
1142 // Get phi position. Careful with ALICE conventions (increase index increase phi)
1143 Int_t iphi2 = iphi;
1144 if (GetSMType(nSupMod) == DCAL_EXT) {
1145 if (nSupMod % 2 != 0) {
1146 iphi2 = (phiindex / 3 - 1) - iphi; // 7-iphi [1/3SM], revert the ordering on C side in order to keep convention.
1147 }
1148 yr = mCentersOfCellsPhiDir[iphi2 + phiindex / 3];
1149 } else if (GetSMType(nSupMod) == EMCAL_HALF) {
1150 if (nSupMod % 2 != 0) {
1151 iphi2 = (phiindex / 2 - 1) - iphi; // 11-iphi [1/2SM], revert the ordering on C side in order to keep
1152 }
1153 // convention.
1154 yr = mCentersOfCellsPhiDir[iphi2 + phiindex / 4];
1155 } else if (GetSMType(nSupMod) == EMCAL_THIRD) {
1156 if (nSupMod % 2 != 0) {
1157 iphi2 = (phiindex / 3 - 1) - iphi; // 7-iphi [1/3SM], revert the ordering on C side in order to keep convention.
1158 }
1159 yr = mCentersOfCellsPhiDir[iphi2 + phiindex / 3];
1160 } else {
1161 if (nSupMod % 2 != 0) {
1162 iphi2 = (phiindex - 1) - iphi; // 23-iphi, revert the ordering on C side in order to keep conventi
1163 }
1164 yr = mCentersOfCellsPhiDir[iphi2];
1165 }
1166
1167 LOG(debug) << "absId " << absId << " nSupMod " << nSupMod << " iphi " << iphi << " ieta " << ieta << " xr " << xr
1168 << " yr " << yr << " zr " << zr;
1169 return o2::math_utils::Point3D<double>(xr, yr, zr);
1170}
1171
1173{
1174 // Shift index taking into account the difference between standard SM
1175 // and SM of half (or one third) size in phi direction
1176 Double_t xr, yr, zr;
1177 Int_t nphiIndex = mCentersOfCellsPhiDir.size();
1178 Double_t zshift = 0.5 * GetDCALInnerEdge();
1179 Int_t kDCalshift = 8; // wangml DCal cut first 8 modules(16 cells)
1180
1181 Int_t iphim = -1, ietam = -1;
1182 TVector2 v;
1183 if (!CheckAbsCellId(absId)) {
1184 throw InvalidCellIDException(absId);
1185 }
1186
1187 auto cellindex = GetCellIndex(absId);
1188 Int_t nSupMod = std::get<0>(cellindex), nModule = std::get<1>(cellindex), nIphi = std::get<2>(cellindex),
1189 nIeta = std::get<3>(cellindex);
1190 auto indmodep = GetModulePhiEtaIndexInSModule(nSupMod, nModule);
1191 iphim = std::get<0>(indmodep);
1192 ietam = std::get<1>(indmodep);
1193 auto indexinsm = GetCellPhiEtaIndexInSModule(nSupMod, nModule, nIphi, nIeta);
1194 Int_t iphi = std::get<0>(indexinsm), ieta = std::get<1>(indexinsm);
1195
1196 // Get eta position. Careful with ALICE conventions (increase index decrease eta)
1197 if (nSupMod % 2 == 0) {
1198 ietam = (mCentersOfCellsEtaDir.size() / 2 - 1) -
1199 ietam; // 24-ietam, revert the ordering on A side in order to keep convention.
1200 if (nIeta == 0) {
1201 nIeta = 1;
1202 } else {
1203 nIeta = 0;
1204 }
1205 }
1206
1207 if (GetSMType(nSupMod) == DCAL_STANDARD && nSupMod % 2) {
1208 ietam += kDCalshift; // DCAL revert the ordering on C side ....
1209 }
1210 const ShishKebabTrd1Module& mod = GetShishKebabModule(ietam);
1211 mod.GetPositionAtCenterCellLine(nIeta, distEff, v);
1212 xr = v.Y() - mParSM[0];
1213 zr = v.X() - mParSM[2];
1214 if (GetSMType(nSupMod) == DCAL_STANDARD) {
1215 zr -= zshift; // DCAL shift (SMALLER SM)
1216 }
1217
1218 // Get phi position. Careful with ALICE conventions (increase index increase phi)
1219 Int_t iphi2 = iphi;
1220 if (GetSMType(nSupMod) == DCAL_EXT) {
1221 if (nSupMod % 2 != 0) {
1222 iphi2 = (nphiIndex / 3 - 1) - iphi; // 7-iphi [1/3SM], revert the ordering on C side in order to keep convention.
1223 }
1224 yr = mCentersOfCellsPhiDir[iphi2 + nphiIndex / 3];
1225 } else if (GetSMType(nSupMod) == EMCAL_HALF) {
1226 if (nSupMod % 2 != 0) {
1227 iphi2 = (nphiIndex / 2 - 1) - iphi; // 11-iphi [1/2SM], revert the ordering on C side in order to keep
1228 }
1229 // convention.
1230 yr = mCentersOfCellsPhiDir[iphi2 + nphiIndex / 2];
1231 } else if (GetSMType(nSupMod) == EMCAL_THIRD) {
1232 if (nSupMod % 2 != 0) {
1233 iphi2 = (nphiIndex / 3 - 1) - iphi; // 7-iphi [1/3SM], revert the ordering on C side in order to keep convention.
1234 }
1235 yr = mCentersOfCellsPhiDir[iphi2 + nphiIndex / 3];
1236 } else {
1237 if (nSupMod % 2 != 0) {
1238 iphi2 = (nphiIndex - 1) - iphi; // 23-iphi, revert the ordering on C side in order to keep convention.
1239 }
1240 yr = mCentersOfCellsPhiDir[iphi2];
1241 }
1242
1243 LOG(debug) << "absId " << absId << " nSupMod " << nSupMod << " iphi " << iphi << " ieta " << ieta << " xr " << xr
1244 << " yr " << yr << " zr " << zr;
1245 return math_utils::Point3D<double>(xr, yr, zr);
1246}
1247
1249{
1250 LOG(debug2) << " o2::emcal::Geometry::CreateListOfTrd1Modules() started\n";
1251
1252 if (!mShishKebabTrd1Modules.size()) {
1253 for (int iz = 0; iz < mNZ; iz++) {
1254 if (iz == 0) {
1255 // mod = new AliEMCALShishKebabTrd1Module(TMath::Pi()/2.,this);
1256 mShishKebabTrd1Modules.emplace_back(ShishKebabTrd1Module(TMath::Pi() / 2., this));
1257 } else {
1259 }
1260 }
1261 } else {
1262 LOG(debug2) << " Already exits :\n";
1263 }
1264
1267 LOG(debug2) << " mShishKebabTrd1Modules has " << mShishKebabTrd1Modules.size() << " modules : max eta "
1268 << std::setw(5) << std::setprecision(4) << mEtaMaxOfTRD1;
1269
1270 // define grid for cells in eta(z) and x directions in local coordinates system of SM
1271 // Works just for 2x2 case only -- ?? start here
1272 //
1273 //
1274 // Define grid for cells in phi(y) direction in local coordinates system of SM
1275 // as for 2X2 as for 3X3 - Nov 8,2006
1276 //
1277 LOG(debug2) << " Cells grid in phi directions : size " << mCentersOfCellsPhiDir.size();
1278
1279 Int_t ind = 0; // this is phi index
1280 Int_t ieta = 0, nModule = 0;
1281 Double_t xr = 0., zr = 0., theta = 0., phi = 0., eta = 0., r = 0., x = 0., y = 0.;
1282 TVector3 vglob;
1283 Double_t ytCenterModule = 0.0, ytCenterCell = 0.0;
1284
1287
1288 Double_t r0 = mIPDistance + mLongModuleSize / 2.;
1289 for (Int_t it = 0; it < mNPhi; it++) { // cycle on modules
1290 ytCenterModule = -mParSM[1] + mPhiModuleSize * (2 * it + 1) / 2; // center of module
1291 for (Int_t ic = 0; ic < mNPHIdiv; ic++) { // cycle on cells in module
1292 if (mNPHIdiv == 2) {
1293 ytCenterCell = ytCenterModule + mPhiTileSize * (2 * ic - 1) / 2.;
1294 } else if (mNPHIdiv == 3) {
1295 ytCenterCell = ytCenterModule + mPhiTileSize * (ic - 1);
1296 } else if (mNPHIdiv == 1) {
1297 ytCenterCell = ytCenterModule;
1298 }
1299 mCentersOfCellsPhiDir[ind] = ytCenterCell;
1300 // Define grid on phi direction
1301 // Grid is not the same for different eta bin;
1302 // Effect is small but is still here
1303 phi = TMath::ATan2(ytCenterCell, r0);
1304 mPhiCentersOfCells[ind] = phi;
1305
1306 LOG(debug2) << " ind " << std::setw(2) << std::setprecision(2) << ind << " : y " << std::setw(8)
1307 << std::setprecision(3) << mCentersOfCellsPhiDir[ind];
1308 ind++;
1309 }
1310 }
1311
1315
1316 LOG(debug2) << " Cells grid in eta directions : size " << mCentersOfCellsEtaDir.size();
1317
1318 for (Int_t it = 0; it < mNZ; it++) {
1319 const ShishKebabTrd1Module& trd1 = GetShishKebabModule(it);
1320 nModule = mNPhi * it;
1321 for (Int_t ic = 0; ic < mNETAdiv; ic++) {
1322 if (mNPHIdiv == 2) {
1323 trd1.GetCenterOfCellInLocalCoordinateofSM(ic, xr, zr); // case of 2X2
1324 auto indexinsm = GetCellPhiEtaIndexInSModule(0, nModule, 0, ic);
1325 ieta = std::get<1>(indexinsm);
1326 }
1327 if (mNPHIdiv == 3) {
1328 trd1.GetCenterOfCellInLocalCoordinateofSM3X3(ic, xr, zr); // case of 3X3
1329 auto indexinsm = GetCellPhiEtaIndexInSModule(0, nModule, 0, ic);
1330 ieta = std::get<1>(indexinsm);
1331 }
1332 if (mNPHIdiv == 1) {
1333 trd1.GetCenterOfCellInLocalCoordinateofSM1X1(xr, zr); // case of 1X1
1334 auto indexinsm = GetCellPhiEtaIndexInSModule(0, nModule, 0, ic);
1335 ieta = std::get<1>(indexinsm);
1336 }
1337 mCentersOfCellsXDir[ieta] = float(xr) - mParSM[0];
1338 mCentersOfCellsEtaDir[ieta] = float(zr) - mParSM[2];
1339 // Define grid on eta direction for each bin in phi
1340 for (int iphi = 0; iphi < mCentersOfCellsPhiDir.size(); iphi++) {
1341 x = xr + trd1.GetRadius();
1342 y = mCentersOfCellsPhiDir[iphi];
1343 r = TMath::Sqrt(x * x + y * y + zr * zr);
1344 theta = TMath::ACos(zr / r);
1346 // ind = ieta*fCentersOfCellsPhiDir.GetSize() + iphi;
1347 ind = iphi * mCentersOfCellsEtaDir.size() + ieta;
1348 mEtaCentersOfCells[ind] = eta;
1349 }
1350 // printf(" ieta %i : xr + trd1->GetRadius() %f : zr %f : eta %f \n", ieta, xr + trd1->GetRadius(), zr, eta);
1351 }
1352 }
1353
1354 for (Int_t i = 0; i < mCentersOfCellsEtaDir.size(); i++) {
1355 LOG(debug2) << " ind " << std::setw(2) << std::setprecision(2) << i + 1 << " : z " << std::setw(8)
1356 << std::setprecision(3) << mCentersOfCellsEtaDir[i] << " : x " << std::setw(8)
1357 << std::setprecision(3) << mCentersOfCellsXDir[i];
1358 }
1359}
1360
1362{
1363 if (mShishKebabTrd1Modules.size() && neta >= 0 && neta < mShishKebabTrd1Modules.size()) {
1364 return mShishKebabTrd1Modules.at(neta);
1365 }
1367}
1368
1369Bool_t Geometry::Impact(const TParticle* particle) const
1370{
1371 Bool_t in = kFALSE;
1372 Int_t absID = 0;
1373 math_utils::Point3D<double> vimpact = {0, 0, 0};
1374
1375 ImpactOnEmcal({particle->Vx(), particle->Vy(), particle->Vz()}, particle->Theta(), particle->Phi(), absID, vimpact);
1376
1377 if (absID >= 0) {
1378 in = kTRUE;
1379 }
1380
1381 return in;
1382}
1383
1384void Geometry::ImpactOnEmcal(const math_utils::Point3D<double>& vtx, Double_t theta, Double_t phi, Int_t& absId, math_utils::Point3D<double>& vimpact) const
1385{
1386 math_utils::Vector3D<double> p(TMath::Sin(theta) * TMath::Cos(phi), TMath::Sin(theta) * TMath::Sin(phi), TMath::Cos(theta));
1387
1388 vimpact.SetXYZ(0, 0, 0);
1389 absId = -1;
1390 if (phi == 0 || theta == 0) {
1391 return;
1392 }
1393
1395 Double_t factor = (mIPDistance - vtx.Y()) / p.Y();
1396 direction = vtx + factor * p;
1397
1398 // from particle direction -> tower hitted
1399 absId = GetAbsCellIdFromEtaPhi(direction.Eta(), direction.Phi());
1400
1401 // tower absID hitted -> tower/module plane (evaluated at the center of the tower)
1402
1403 Double_t loc[3], loc2[3], loc3[3];
1404 Double_t glob[3] = {}, glob2[3] = {}, glob3[3] = {};
1405
1406 try {
1407 RelPosCellInSModule(absId).GetCoordinates(loc[0], loc[1], loc[2]);
1408 } catch (InvalidCellIDException& e) {
1409 LOG(error) << e.what();
1410 return;
1411 }
1412
1413 // loc is cell center of tower
1414 auto cellindex = GetCellIndex(absId);
1415 Int_t nSupMod = std::get<0>(cellindex), nModule = std::get<1>(cellindex), nIphi = std::get<2>(cellindex),
1416 nIeta = std::get<3>(cellindex);
1417 // look at 2 neighbours-s cell using nIphi={0,1} and nIeta={0,1}
1418 Int_t nIphi2 = -1, nIeta2 = -1, absId2 = -1, absId3 = -1;
1419 if (nIeta == 0) {
1420 nIeta2 = 1;
1421 } else {
1422 nIeta2 = 0;
1423 }
1424 absId2 = GetAbsCellId(nSupMod, nModule, nIphi, nIeta2);
1425 if (nIphi == 0) {
1426 nIphi2 = 1;
1427 } else {
1428 nIphi2 = 0;
1429 }
1430 absId3 = GetAbsCellId(nSupMod, nModule, nIphi2, nIeta);
1431
1432 // 2nd point on emcal cell plane
1433 try {
1434 RelPosCellInSModule(absId2).GetCoordinates(loc2[0], loc2[1], loc2[2]);
1435 } catch (InvalidCellIDException& e) {
1436 LOG(error) << e.what();
1437 return;
1438 }
1439
1440 // 3rd point on emcal cell plane
1441 try {
1442 RelPosCellInSModule(absId3).GetCoordinates(loc3[0], loc3[1], loc3[2]);
1443 } catch (InvalidCellIDException& e) {
1444 LOG(error) << e.what();
1445 return;
1446 }
1447
1448 // Get Matrix
1449 const TGeoHMatrix* m = GetMatrixForSuperModule(nSupMod);
1450 if (m) {
1451 m->LocalToMaster(loc, glob);
1452 m->LocalToMaster(loc2, glob2);
1453 m->LocalToMaster(loc3, glob3);
1454 } else {
1455 LOG(fatal) << "Geo matrixes are not loaded \n";
1456 }
1457
1458 // Equation of Plane from glob,glob2,glob3 (Ax+By+Cz+D=0)
1459 Double_t a = glob[1] * (glob2[2] - glob3[2]) + glob2[1] * (glob3[2] - glob[2]) + glob3[1] * (glob[2] - glob2[2]);
1460 Double_t b = glob[2] * (glob2[0] - glob3[0]) + glob2[2] * (glob3[0] - glob[0]) + glob3[2] * (glob[0] - glob2[0]);
1461 Double_t c = glob[0] * (glob2[1] - glob3[1]) + glob2[0] * (glob3[1] - glob[1]) + glob3[0] * (glob[1] - glob2[1]);
1462 Double_t d = glob[0] * (glob2[1] * glob3[2] - glob3[1] * glob2[2]) +
1463 glob2[0] * (glob3[1] * glob[2] - glob[1] * glob3[2]) +
1464 glob3[0] * (glob[1] * glob2[2] - glob2[1] * glob[2]);
1465 d = -d;
1466
1467 // shift equation of plane from tower/module center to surface along vector (A,B,C) normal to tower/module plane
1468 Double_t dist = mLongModuleSize / 2.;
1469 Double_t norm = TMath::Sqrt(a * a + b * b + c * c);
1470 Double_t glob4[3] = {};
1472 math_utils::Point3D<double> point = {glob[0], glob[1], glob[2]};
1473 if (point.Dot(dir) < 0) {
1474 dist *= -1;
1475 }
1476 glob4[0] = glob[0] - dist * a / norm;
1477 glob4[1] = glob[1] - dist * b / norm;
1478 glob4[2] = glob[2] - dist * c / norm;
1479 d = glob4[0] * a + glob4[1] * b + glob4[2] * c;
1480 d = -d;
1481
1482 // Line determination (2 points for equation of line : vtx and direction)
1483 // impact between line (particle) and plane (module/tower plane)
1484 Double_t den = a * (vtx.X() - direction.X()) + b * (vtx.Y() - direction.Y()) + c * (vtx.Z() - direction.Z());
1485 if (den == 0) {
1486 LOG(error) << "ImpactOnEmcal() No solution :\n";
1487 return;
1488 }
1489
1490 Double_t length = a * vtx.X() + b * vtx.Y() + c * vtx.Z() + d;
1491 length /= den;
1492
1493 vimpact.SetXYZ(vtx.X() + length * (direction.X() - vtx.X()), vtx.Y() + length * (direction.Y() - vtx.Y()),
1494 vtx.Z() + length * (direction.Z() - vtx.Z()));
1495
1496 // shift vimpact from tower/module surface to center along vector (A,B,C) normal to tower/module plane
1497 vimpact.SetXYZ(vimpact.Z() + dist * a / norm, vimpact.Y() + dist * b / norm, vimpact.Z() + dist * c / norm);
1498}
1499
1501{
1502 if (IsInEMCALOrDCAL(pnt) == EMCAL_ACCEPTANCE) {
1503 return kTRUE;
1504 } else {
1505 return kFALSE;
1506 }
1507}
1508
1510{
1511 if (IsInEMCALOrDCAL(pnt) == DCAL_ACCEPTANCE) {
1512 return kTRUE;
1513 } else {
1514 return kFALSE;
1515 }
1516}
1517
1519{
1520 Double_t r = sqrt(pnt.X() * pnt.X() + pnt.Y() * pnt.Y());
1521
1522 if (r <= mEnvelop[0]) {
1523 return NON_ACCEPTANCE;
1524 } else {
1525 Double_t theta = TMath::ATan2(r, pnt.Z());
1526 Double_t eta;
1527 if (theta == 0) {
1528 eta = 9999;
1529 } else {
1530 eta = -TMath::Log(TMath::Tan(theta / 2.));
1531 }
1532 if (eta < mArm1EtaMin || eta > mArm1EtaMax) {
1533 return NON_ACCEPTANCE;
1534 }
1535
1536 Double_t phi = TMath::ATan2(pnt.Y(), pnt.X()) * 180. / TMath::Pi();
1537 if (phi < 0) {
1538 phi += 360; // phi should go from 0 to 360 in this case
1539 }
1540
1541 if (phi >= mArm1PhiMin && phi <= mEMCALPhiMax) {
1542 return EMCAL_ACCEPTANCE;
1543 } else if (phi >= mDCALPhiMin && phi <= mDCALStandardPhiMax && TMath::Abs(eta) > mDCALInnerExtandedEta) {
1544 return DCAL_ACCEPTANCE;
1545 } else if (phi > mDCALStandardPhiMax && phi <= mDCALPhiMax) {
1546 return DCAL_ACCEPTANCE;
1547 }
1548 return NON_ACCEPTANCE;
1549 }
1550}
1551
1552const TGeoHMatrix* Geometry::GetMatrixForSuperModule(Int_t smod) const
1553{
1554 if (smod < 0 || smod > mNumberOfSuperModules) {
1555 LOG(fatal) << "Wrong supermodule index -> " << smod;
1556 }
1557
1558 if (!SMODULEMATRIX[smod]) {
1559 if (gGeoManager) {
1561 } else {
1562 LOG(fatal) << "Cannot find EMCAL misalignment matrices! Recover them either: \n"
1563 << "\t - importing TGeoManager from file geometry.root or \n"
1564 << "\t - from OADB in file OADB/EMCAL/EMCALlocal2master.root or \n"
1565 << "\t - from OCDB in directory OCDB/EMCAL/Align/Data/ or \n"
1566 << "\t - from AliESDs (not in AliAOD) via AliESDRun::GetEMCALMatrix(Int_t superModIndex). \n"
1567 << "Store them via AliEMCALGeometry::SetMisalMatrix(Int_t superModIndex)";
1568 }
1569 }
1570
1571 return SMODULEMATRIX[smod];
1572}
1573
1574const TGeoHMatrix* Geometry::GetMatrixForSuperModuleFromArray(Int_t smod) const
1575{
1576 if (smod < 0 || smod > mNumberOfSuperModules) {
1577 LOG(fatal) << "Wrong supermodule index -> " << smod;
1578 }
1579
1580 return SMODULEMATRIX[smod];
1581}
1582
1583const TGeoHMatrix* Geometry::GetMatrixForSuperModuleFromGeoManager(Int_t smod) const
1584{
1585 const Int_t buffersize = 255;
1586 char path[buffersize];
1587 Int_t tmpType = -1;
1588 Int_t smOrder = 0;
1589
1590 // Get the order for SM
1591 for (Int_t i = 0; i < smod + 1; i++) {
1592 if (GetSMType(i) == tmpType) {
1593 smOrder++;
1594 } else {
1595 tmpType = GetSMType(i);
1596 smOrder = 1;
1597 }
1598 }
1599
1600 Int_t smType = GetSMType(smod);
1601 TString smName = "";
1602
1603 if (smType == EMCAL_STANDARD) {
1604 smName = "SMOD";
1605 } else if (smType == EMCAL_HALF) {
1606 smName = "SM10";
1607 } else if (smType == EMCAL_THIRD) {
1608 smName = "SM3rd";
1609 } else if (smType == DCAL_STANDARD) {
1610 smName = "DCSM";
1611 } else if (smType == DCAL_EXT) {
1612 smName = "DCEXT";
1613 } else {
1614 LOG(error) << "Unkown SM Type!!\n";
1615 }
1616
1617 snprintf(path, buffersize, "/cave/barrel_1/%s_%d", smName.Data(), smOrder);
1618
1619 if (!gGeoManager->cd(path)) {
1620 LOG(fatal) << "Geo manager can not find path " << path << "!\n";
1621 }
1622
1623 return gGeoManager->GetCurrentMatrix();
1624}
1625
1626void Geometry::RecalculateTowerPosition(Float_t drow, Float_t dcol, const Int_t sm, const Float_t depth,
1627 const Float_t misaligTransShifts[15], const Float_t misaligRotShifts[15],
1628 Float_t global[3]) const
1629{
1630 // To use in a print later
1631 Float_t droworg = drow;
1632 Float_t dcolorg = dcol;
1633
1634 if (gGeoManager) {
1635 // Recover some stuff
1636
1637 const Int_t nSMod = mNumberOfSuperModules;
1638
1639 gGeoManager->cd("/cave/barrel_1/");
1640 TGeoNode* geoXEn1 = gGeoManager->GetCurrentNode();
1641 TGeoNodeMatrix* geoSM[nSMod];
1642 TGeoVolume* geoSMVol[nSMod];
1643 TGeoShape* geoSMShape[nSMod];
1644 TGeoBBox* geoBox[nSMod];
1645 TGeoMatrix* geoSMMatrix[nSMod];
1646
1647 for (int iSM = 0; iSM < nSMod; iSM++) {
1648 geoSM[iSM] = dynamic_cast<TGeoNodeMatrix*>(geoXEn1->GetDaughter(iSM));
1649 geoSMVol[iSM] = geoSM[iSM]->GetVolume();
1650 geoSMShape[iSM] = geoSMVol[iSM]->GetShape();
1651 geoBox[iSM] = dynamic_cast<TGeoBBox*>(geoSMShape[iSM]);
1652 geoSMMatrix[iSM] = geoSM[iSM]->GetMatrix();
1653 }
1654
1655 if (sm % 2 == 0) {
1656 dcol = 47. - dcol;
1657 drow = 23. - drow;
1658 }
1659
1660 Int_t istrip = 0;
1661 Float_t z0 = 0;
1662 Float_t zb = 0;
1663 Float_t zIs = 0;
1664
1665 Float_t x, y, z; // return variables in terry's RF
1666
1667 //***********************************************************
1668 // Do not like this: too many hardcoded values, is it not already stored somewhere else?
1669 // : need more comments in the code
1670 //***********************************************************
1671
1672 Float_t dz = 6.0; // base cell width in eta
1673 Float_t dx = 6.004; // base cell width in phi
1674
1675 // Float_t L = 26.04; // active tower length for hadron (lead+scint+paper)
1676 // we use the geant numbers 13.87*2=27.74
1677 Float_t teta1 = 0.;
1678
1679 // Do some basic checks
1680 if (dcol >= 47.5 || dcol < -0.5) {
1681 LOG(error) << "Bad tower coordinate dcol=" << dcol << ", where dcol >= 47.5 || dcol<-0.5; org: " << dcolorg;
1682 return;
1683 }
1684 if (drow >= 23.5 || drow < -0.5) {
1685 LOG(error) << "Bad tower coordinate drow=" << drow << ", where drow >= 23.5 || drow<-0.5; org: " << droworg;
1686 return;
1687 }
1688 if (sm >= nSMod || sm < 0) {
1689 LOG(error) << "Bad SM number sm=" << nSMod << ", where sm >= " << sm << " || sm < 0\n";
1690 return;
1691 }
1692
1693 istrip = int((dcol + 0.5) / 2);
1694
1695 // tapering angle
1696 teta1 = TMath::DegToRad() * istrip * 1.5;
1697
1698 // calculation of module corner along z
1699 // as a function of strip
1700
1701 for (int is = 0; is <= istrip; is++) {
1702 teta1 = TMath::DegToRad() * (is * 1.5 + 0.75);
1703 if (is == 0) {
1704 zIs = zIs + 2 * dz * TMath::Cos(teta1);
1705 } else {
1706 zIs =
1707 zIs + 2 * dz * TMath::Cos(teta1) + 2 * dz * TMath::Sin(teta1) * TMath::Tan(teta1 - 0.75 * TMath::DegToRad());
1708 }
1709 }
1710
1711 z0 = dz * (dcol - 2 * istrip + 0.5);
1712 zb = (2 * dz - z0 - depth * TMath::Tan(teta1));
1713
1714 z = zIs - zb * TMath::Cos(teta1);
1715 y = depth / TMath::Cos(teta1) + zb * TMath::Sin(teta1);
1716
1717 x = (drow + 0.5) * dx;
1718
1719 // moving the origin from terry's RF
1720 // to the GEANT one
1721
1722 double xx = y - geoBox[sm]->GetDX();
1723 double yy = -x + geoBox[sm]->GetDY();
1724 double zz = z - geoBox[sm]->GetDZ();
1725 const double localIn[3] = {xx, yy, zz};
1726 double dglobal[3];
1727 // geoSMMatrix[sm]->Print();
1728 // printf("TFF Local (row = %d, col = %d, x = %3.2f, y = %3.2f, z = %3.2f)\n", iroworg, icolorg, localIn[0],
1729 // localIn[1], localIn[2]);
1730 geoSMMatrix[sm]->LocalToMaster(localIn, dglobal);
1731 // printf("TFF Global (row = %2.0f, col = %2.0f, x = %3.2f, y = %3.2f, z = %3.2f)\n", drow, dcol, dglobal[0],
1732 // dglobal[1], dglobal[2]);
1733
1734 // apply global shifts
1735 if (sm == 2 || sm == 3) { // sector 1
1736 global[0] = dglobal[0] + misaligTransShifts[3] + misaligRotShifts[3] * TMath::Sin(TMath::DegToRad() * 20);
1737 global[1] = dglobal[1] + misaligTransShifts[4] + misaligRotShifts[4] * TMath::Cos(TMath::DegToRad() * 20);
1738 global[2] = dglobal[2] + misaligTransShifts[5];
1739 } else if (sm == 0 || sm == 1) { // sector 0
1740 global[0] = dglobal[0] + misaligTransShifts[0];
1741 global[1] = dglobal[1] + misaligTransShifts[1];
1742 global[2] = dglobal[2] + misaligTransShifts[2];
1743 } else {
1744 LOG(info) << "Careful, correction not implemented yet!\n";
1745 global[0] = dglobal[0];
1746 global[1] = dglobal[1];
1747 global[2] = dglobal[2];
1748 }
1749 } else {
1750 LOG(fatal) << "Geometry boxes information, check that geometry.root is loaded\n";
1751 }
1752}
1753
1754void Geometry::SetMisalMatrix(const TGeoHMatrix* m, Int_t smod) const
1755{
1756 if (smod >= 0 && smod < mNumberOfSuperModules) {
1757 if (!SMODULEMATRIX[smod]) {
1758 SMODULEMATRIX[smod] = new TGeoHMatrix(*m); // Set only if not set yet
1759 }
1760 } else {
1761 LOG(fatal) << "Wrong supermodule index -> " << smod << std::endl;
1762 }
1763}
1764
1765Bool_t Geometry::IsDCALSM(Int_t iSupMod) const
1766{
1767 if (mEMCSMSystem[iSupMod] == DCAL_STANDARD || mEMCSMSystem[iSupMod] == DCAL_EXT) {
1768 return kTRUE;
1769 }
1770
1771 return kFALSE;
1772}
1773
1774Bool_t Geometry::IsDCALExtSM(Int_t iSupMod) const
1775{
1776 if (mEMCSMSystem[iSupMod] == DCAL_EXT) {
1777 return kTRUE;
1778 }
1779
1780 return kFALSE;
1781}
1782
1783Double_t Geometry::GetPhiCenterOfSMSec(Int_t nsupmod) const
1784{
1785 int i = nsupmod / 2;
1786 return mPhiCentersOfSMSec[i];
1787}
1788
1789Double_t Geometry::GetPhiCenterOfSM(Int_t nsupmod) const
1790{
1791 int i = nsupmod / 2;
1792 return mPhiCentersOfSM[i];
1793}
1794
1795std::tuple<double, double> Geometry::GetPhiBoundariesOfSM(Int_t nSupMod) const
1796{
1797 int i;
1798 if (nSupMod < 0 || nSupMod > 12 + mnSupModInDCAL - 1) {
1799 throw InvalidModuleException(nSupMod, 12 + mnSupModInDCAL);
1800 }
1801 i = nSupMod / 2;
1802 return std::make_tuple((Double_t)mPhiBoundariesOfSM[2 * i], (Double_t)mPhiBoundariesOfSM[2 * i + 1]);
1803}
1804
1805std::tuple<double, double> Geometry::GetPhiBoundariesOfSMGap(Int_t nPhiSec) const
1806{
1807 if (nPhiSec < 0 || nPhiSec > 5 + mnSupModInDCAL / 2 - 1) {
1808 throw InvalidModuleException(nPhiSec, 5 + mnSupModInDCAL / 2);
1809 }
1810 return std::make_tuple(mPhiBoundariesOfSM[2 * nPhiSec + 1], mPhiBoundariesOfSM[2 * nPhiSec + 2]);
1811}
1812
1813std::tuple<int, int, int> Geometry::getOnlineID(int towerID)
1814{
1815 auto cellindex = GetCellIndex(towerID);
1816 auto supermoduleID = std::get<0>(cellindex);
1817 auto etaphi = GetCellPhiEtaIndexInSModule(supermoduleID, std::get<1>(cellindex), std::get<2>(cellindex), std::get<3>(cellindex));
1818 auto etaphishift = ShiftOfflineToOnlineCellIndexes(supermoduleID, std::get<0>(etaphi), std::get<1>(etaphi));
1819 int row = std::get<0>(etaphishift), col = std::get<1>(etaphishift);
1820
1821 int ddlInSupermoudel = -1;
1822 if (0 <= row && row < 8) {
1823 ddlInSupermoudel = 0; // first cable row
1824 } else if (8 <= row && row < 16 && 0 <= col && col < 24) {
1825 ddlInSupermoudel = 0; // first half;
1826 } else if (8 <= row && row < 16 && 24 <= col && col < 48) {
1827 ddlInSupermoudel = 1; // second half;
1828 } else if (16 <= row && row < 24) {
1829 ddlInSupermoudel = 1; // third cable row
1830 }
1831 if (supermoduleID % 2 == 1) {
1832 ddlInSupermoudel = 1 - ddlInSupermoudel; // swap for odd=C side, to allow us to cable both sides the same
1833 }
1834
1835 return std::make_tuple(supermoduleID * 2 + ddlInSupermoudel, row, col);
1836}
int32_t i
uint32_t supermodule
Definition RawData.h:3
uint32_t side
Definition RawData.h:0
uint32_t col
Definition RawData.h:4
uint32_t c
Definition RawData.h:2
std::ostringstream debug
Error handling access to non-initialized geometry.
EMCAL geometry definition.
Definition Geometry.h:40
std::tuple< int, int, int > GetModuleIndexesFromCellIndexesInSModule(int supermoduleID, int phiInSupermodule, int etaInSupermodule) const
Transition from cell indexes (iphi, ieta) to module indexes (iphim, ietam, nModule)
Definition Geometry.cxx:778
Float_t mFrontSteelStrip
13-may-05
Definition Geometry.h:672
static Geometry * GetInstanceFromRunNumber(Int_t runNumber, const std::string_view="", const std::string_view mcname="TGeant3", const std::string_view mctitle="")
Instanciate geometry depending on the run number. Mostly used in analysis and MC anchors.
Definition Geometry.cxx:219
Float_t mEtaMaxOfTRD1
Max eta in case of TRD1 geometry (see AliEMCALShishKebabTrd1Module)
Definition Geometry.h:639
Int_t GetSuperModuleNumber(Int_t absId) const
Get cell SM, from absolute ID number.
void RecalculateTowerPosition(Float_t drow, Float_t dcol, const Int_t sm, const Float_t depth, const Float_t misaligTransShifts[15], const Float_t misaligRotShifts[15], Float_t global[3]) const
std::tuple< int, int, int, int > CalculateCellIndex(Int_t absId) const
Calculate cell SM, module numbers from absolute ID number.
Float_t mIPDistance
Radial Distance of the inner surface of the EMCAL.
Definition Geometry.h:653
Float_t mDCALPhiMin
Minimum angular position of DCAL in Phi (degrees)
Definition Geometry.h:640
Float_t mPassiveScintThick
13-may-05
Definition Geometry.h:674
Float_t mPhiTileSize
Size of phi tile.
Definition Geometry.h:650
int GlobalRow(int cellID) const
Get row number of cell in global numbering scheme.
Definition Geometry.cxx:896
Int_t mNZ
Number of Towers in the Z direction.
Definition Geometry.h:652
Float_t mECScintThick
cm, Thickness of the scintillators
Definition Geometry.h:663
const TGeoHMatrix * GetMatrixForSuperModuleFromGeoManager(Int_t smod) const
Provides shift-rotation matrix for EMCAL from the TGeoManager.
std::tuple< int, int > GetCellPhiEtaIndexInSModule(int supermoduleID, int moduleID, int phiInModule, int etaInModule) const
Get eta-phi indexes of cell in SM.
std::tuple< int, int, int, int > GetCellIndex(Int_t absId) const
Get cell SM, module numbers from absolute ID number.
std::tuple< double, double > GetPhiBoundariesOfSM(Int_t nSupMod) const
Int_t mNECLayers
number of scintillator layers
Definition Geometry.h:664
std::vector< Double_t > mPhiCentersOfCells
[fNPhi*fNPHIdiv] from center of SM (-10. < phi < +10.)
Definition Geometry.h:626
const TGeoHMatrix * GetMatrixForSuperModule(Int_t smod) const
Provides shift-rotation matrix for EMCAL from externally set matrix or from TGeoManager.
Geometry & operator=(const Geometry &rvalue)
Assignment operator.
Definition Geometry.cxx:170
Float_t GetDCALInnerEdge() const
Definition Geometry.h:184
std::tuple< int, int, int > getOnlineID(int towerID)
Get link ID, row and column from cell ID, have a look here: https://alice.its.cern....
Int_t GetNEta() const
Get the number of modules in supermodule in #eta direction.
Definition Geometry.h:198
Float_t mEtaModuleSize
Eta -> Y.
Definition Geometry.h:649
Double_t GetPhiCenterOfSMSec(Int_t nsupmod) const
Float_t GetPhiModuleSize() const
Definition Geometry.h:209
Bool_t CheckAbsCellId(Int_t absId) const
Check whether a cell number is valid.
Definition Geometry.h:703
Float_t mZLength
Total length in z direction.
Definition Geometry.h:658
Float_t mEnvelop[3]
The GEANT TUB for the detector.
Definition Geometry.h:634
Float_t mDCALInnerEdge
Inner edge for DCAL.
Definition Geometry.h:645
void CreateListOfTrd1Modules()
Float_t mParSM[3]
SM sizes as in GEANT (TRD1)
Definition Geometry.h:647
Float_t GetShellThickness() const
Definition Geometry.h:182
Int_t mNCellsInModule
Number cell in module.
Definition Geometry.h:620
std::vector< EMCALSMType > mEMCSMSystem
geometry structure
Definition Geometry.h:670
Bool_t IsDCALExtSM(Int_t nSupMod) const
Check if iSupMod is a valid DCal 1/3rd SM.
std::tuple< int, int, int, int > GetCellIndexFromGlobalRowCol(int row, int col) const
Get the cell indices from global position in the EMCAL.
Definition Geometry.cxx:880
Double_t GetPhiCenterOfSM(Int_t nsupmod) const
const TGeoHMatrix * SMODULEMATRIX[EMCAL_MODULES]
Orientations of EMCAL super modules.
Definition Geometry.h:696
std::tuple< int, int > GetModulePhiEtaIndexInSModule(int supermoduleID, int moduleID) const
Get eta-phi indexes of module in SM.
std::tuple< int, int, int > GetPositionInSupermoduleFromGlobalRowCol(int row, int col) const
Get the posision (row, col) of a global row-col position.
Definition Geometry.cxx:834
const TGeoHMatrix * GetMatrixForSuperModuleFromArray(Int_t smod) const
Provides shift-rotation matrix for EMCAL from fkSModuleMatrix[smod].
Float_t mTrd1AlFrontThick
Thickness of the Al front plate.
Definition Geometry.h:685
Float_t mSampling
Sampling factor.
Definition Geometry.h:659
Float_t mDCALPhiMax
Maximum angular position of DCAL in Phi (degrees)
Definition Geometry.h:641
Float_t mTrd1BondPaperThick
Thickness of the Bond Paper sheet.
Definition Geometry.h:686
Int_t mNCells
Number of cells in calo.
Definition Geometry.h:631
Float_t mArm1PhiMin
Minimum angular position of EMCAL in Phi (degrees)
Definition Geometry.h:637
void SetMisalMatrix(const TGeoHMatrix *m, Int_t smod) const
Int_t mNETAdiv
Number eta division of module.
Definition Geometry.h:618
Float_t mLateralSteelStrip
13-may-05
Definition Geometry.h:673
std::tuple< int, int > GlobalRowColFromIndex(int cellID) const
get (Column,Row) pair of cell in global numbering scheme
Definition Geometry.cxx:808
Float_t mTrd1Angle
angle in x-z plane (in degree)
Definition Geometry.h:680
Float_t mECPbRadThickness
cm, Thickness of the Pb radiators
Definition Geometry.h:662
void GetGlobal(const Double_t *loc, Double_t *glob, int ind) const
Figure out the global coordinates from local coordinates on a supermodule.
Definition Geometry.cxx:681
std::vector< Double_t > mEtaCentersOfCells
[fNEta*fNETAdiv*fNPhi*fNPHIdiv], positive direction (eta>0); eta depend from phi position;
Definition Geometry.h:630
int GetAbsCellId(int supermoduleID, int moduleID, int phiInModule, int etaInModule) const
Get cell absolute ID number from location module (2 times 2 cells) of a super module.
Definition Geometry.cxx:738
Float_t mArm1PhiMax
Maximum angular position of EMCAL in Phi (degrees)
Definition Geometry.h:638
static Geometry * GetInstance()
Get geometry instance. It should have been set before.
Definition Geometry.cxx:190
std::tuple< double, double > EtaPhiFromIndex(Int_t absId) const
Figure out the eta/phi coordinates of a cell.
Definition Geometry.cxx:731
Int_t mKey110DEG
For calculation abs cell id; 19-oct-05.
Definition Geometry.h:615
std::vector< Double_t > mPhiCentersOfSMSec
Phi of centers of section where SM lies; size is fNumberOfSuperModules/2.
Definition Geometry.h:623
std::vector< Double_t > mCentersOfCellsEtaDir
Size fNEta*fNETAdiv (for TRD1 only) (eta or z in SM, in cm)
Definition Geometry.h:627
std::string mGeoName
Geometry name string.
Definition Geometry.h:614
Int_t GetNPhiSuperModule() const
Definition Geometry.h:217
int GlobalCol(int cellID) const
Get column number of cell in global numbering scheme.
Definition Geometry.cxx:891
Float_t mShellThickness
Total thickness in (x,y) direction.
Definition Geometry.h:657
EMCALSMType GetSMType(Int_t nSupMod) const
Definition Geometry.h:242
std::vector< Double_t > mCentersOfCellsPhiDir
Size fNPhi*fNPHIdiv (for TRD1 only) (phi or y in SM, in cm)
Definition Geometry.h:628
Int_t mNPHIdiv
Number phi division of module.
Definition Geometry.h:619
Int_t mnSupModInDCAL
For calculation abs cell id; 06-nov-12.
Definition Geometry.h:616
Float_t mEtaTileSize
Size of eta tile.
Definition Geometry.h:651
int SuperModuleNumberFromEtaPhi(Double_t eta, Double_t phi) const
Given a global eta/phi point check if it belongs to a supermodule covered region.
Definition Geometry.cxx:901
void DefineEMC(std::string_view mcname, std::string_view mctitle)
Init function of previous class EMCGeometry.
Definition Geometry.cxx:343
Bool_t IsDCALSM(Int_t nSupMod) const
Check if iSupMod is a valid DCal standard SM.
Bool_t Impact(const TParticle *particle) const
Check if particle falls in the EMCal/DCal geometry.
std::vector< ShishKebabTrd1Module > mShishKebabTrd1Modules
List of modules.
Definition Geometry.h:646
Bool_t IsInEMCAL(const math_utils::Point3D< double > &pnt) const
Checks whether point is inside the EMCal volume.
math_utils::Point3D< double > RelPosCellInSModule(Int_t absId, Double_t distEf) const
Look to see what the relative position inside a given cell is for a recpoint.
const std::string & GetName() const
Definition Geometry.h:109
Float_t mEMCALPhiMax
Maximum angular position of EMCAL in Phi (degrees)
Definition Geometry.h:642
Bool_t IsInDCAL(const math_utils::Point3D< double > &pnt) const
Checks whether point is inside the DCal volume.
Int_t mNPhiSuperModule
9 - number supermodule in phi direction
Definition Geometry.h:677
Float_t mPhiModuleSize
Phi -> X.
Definition Geometry.h:648
Float_t mPhiGapForSM
Gap betweeen supermodules in phi direction.
Definition Geometry.h:682
Geometry()=default
Default constructor. It must be kept public for root persistency purposes, but should never be called...
Float_t m2Trd1Dx2
2*dx2 for TRD1
Definition Geometry.h:681
AcceptanceType_t IsInEMCALOrDCAL(const math_utils::Point3D< double > &pnt) const
Checks whether point is inside the EMCal volume (included DCal)
Int_t GetNumberOfSuperModules() const
Definition Geometry.h:207
~Geometry()
Destructor.
Definition Geometry.cxx:176
Int_t GetNumberOfModuleInPhiDirection(Int_t nSupMod) const
Definition Geometry.h:462
std::tuple< int, int > ShiftOnlineToOfflineCellIndexes(Int_t supermoduleID, Int_t iphi, Int_t ieta) const
Adapt cell indices in supermodule to online indexing.
Float_t mPhiSuperModule
Phi of normal supermodule (20, in degree)
Definition Geometry.h:676
Float_t mArm1EtaMin
Minimum pseudorapidity position of EMCAL in Eta.
Definition Geometry.h:635
void ImpactOnEmcal(const math_utils::Point3D< double > &vtx, Double_t theta, Double_t phi, Int_t &absId, math_utils::Point3D< double > &vimpact) const
Get the impact coordinates on EMCAL.
Int_t GetNPhi() const
Get the number of modules in supermodule in #phi direction.
Definition Geometry.h:202
Float_t mArm1EtaMax
Maximum pseudorapidity position of EMCAL in Eta.
Definition Geometry.h:636
Int_t GetAbsCellIdFromCellIndexes(Int_t nSupMod, Int_t iphi, Int_t ieta) const
Transition from super module number (nSupMod) and cell indexes (ieta,iphi) to cell absolute ID number...
Definition Geometry.cxx:791
std::vector< Double_t > mCentersOfCellsXDir
Size fNEta*fNETAdiv (for TRD1 only) ( x in SM, in cm)
Definition Geometry.h:633
Int_t mNPhi
Number of Towers in the PHI direction.
Definition Geometry.h:632
const ShishKebabTrd1Module & GetShishKebabModule(Int_t neta) const
Get the Module parameters for a eta.
Int_t mNumberOfSuperModules
default is 12 = 6 * 2
Definition Geometry.h:667
std::tuple< double, double > GetPhiBoundariesOfSMGap(Int_t nPhiSec) const
std::vector< Double_t > mPhiBoundariesOfSM
Phi boundaries of SM in rad; size is fNumberOfSuperModules;.
Definition Geometry.h:621
std::vector< std::tuple< int, int, int, int > > mCellIndexLookup
Lookup table for cell indices.
Definition Geometry.h:697
void DefineSamplingFraction(const std::string_view mcname="", const std::string_view mctitle="")
Set the value of the Sampling used to calibrate the MC hits energy (check)
Definition Geometry.cxx:285
Float_t mDCALStandardPhiMax
Special edge for the case that DCAL contian extension.
Definition Geometry.h:643
Float_t mLongModuleSize
Size of long module.
Definition Geometry.h:654
int GetAbsCellIdFromEtaPhi(Double_t eta, Double_t phi) const
Get cell absolute ID number from eta and phi location.
Definition Geometry.cxx:932
Int_t mNCellsInSupMod
Number cell in super module.
Definition Geometry.h:617
int GetCellAbsIDFromGlobalRowCol(int row, int col) const
Get the absolute cell ID from global position in the EMCAL.
Definition Geometry.cxx:874
std::tuple< int, int > ShiftOfflineToOnlineCellIndexes(Int_t supermoduleID, Int_t iphi, Int_t ieta) const
Adapt cell indices in supermodule to offline indexing.
std::vector< Double_t > mPhiCentersOfSM
Phi of centers of SM; size is fNumberOfSuperModules/2.
Definition Geometry.h:622
Float_t mDCALInnerExtandedEta
DCAL inner edge in Eta (with some extension)
Definition Geometry.h:644
Exception handling non-existing cell IDs.
const char * what() const noexcept final
Access to error message of the exception.
Error Handling when an invalid module ID (outside the limits) is called.
Exception handling errors due to positions not in the EMCAL area.
Exception handling improper or uninitialized supermodule types.
Handling error for invalid positions in row-column space.
Main class for TRD1 geometry of Shish-Kebab case.
void GetCenterOfCellInLocalCoordinateofSM3X3(Int_t ieta, Double_t &xr, Double_t &zr) const
void GetCenterOfCellInLocalCoordinateofSM1X1(Double_t &xr, Double_t &zr) const
static Double_t ThetaToEta(Double_t theta)
void GetPositionAtCenterCellLine(Int_t ieta, Double_t dist, TVector2 &v) const
const TVector2 & GetCenterOfCellInLocalCoordinateofSM(Int_t ieta) const
GLint GLenum GLint x
Definition glcorearb.h:403
const GLfloat * m
Definition glcorearb.h:4066
const GLdouble * v
Definition glcorearb.h:832
GLuint const GLchar * name
Definition glcorearb.h:781
GLboolean GLboolean GLboolean b
Definition glcorearb.h:1233
GLint y
Definition glcorearb.h:270
GLuint GLsizei GLsizei * length
Definition glcorearb.h:790
GLint GLint GLsizei GLsizei GLsizei depth
Definition glcorearb.h:470
GLsizei const GLchar *const * path
Definition glcorearb.h:3591
GLboolean r
Definition glcorearb.h:1233
GLboolean GLboolean GLboolean GLboolean a
Definition glcorearb.h:1233
GLdouble GLdouble GLdouble z
Definition glcorearb.h:843
const std::string DEFAULT_GEOMETRY
@ EMCAL_MODULES
Number of modules, 12 for EMCal + 8 for DCAL.
Definition Constants.h:24
@ EMCAL_ROWS
Number of rows per module for EMCAL.
Definition Constants.h:25
@ EMCAL_COLS
Number of columns per module for EMCAL.
Definition Constants.h:26
FIXME: do not use data model tables.
LOG(info)<< "Compressed in "<< sw.CpuTime()<< " s"
std::vector< int > row