Project
Loading...
Searching...
No Matches
Detector.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
14
20
21#include "DetectorsBase/Stack.h"
23
24// FairRoot includes
25#include "FairDetector.h" // for FairDetector
26#include <fairlogger/Logger.h> // for LOG, LOG_IF
27#include "FairRootManager.h" // for FairRootManager
28#include "FairRun.h" // for FairRun
29#include "FairRuntimeDb.h" // for FairRuntimeDb
30#include "FairVolume.h" // for FairVolume
31#include "FairRootManager.h"
32
33#include "TGeoManager.h" // for TGeoManager, gGeoManager
34#include "TGeoTube.h" // for TGeoTube
35#include "TGeoPcon.h" // for TGeoPcon
36#include "TGeoVolume.h" // for TGeoVolume, TGeoVolumeAssembly
37#include "TString.h" // for TString, operator+
38#include "TVirtualMC.h" // for gMC, TVirtualMC
39#include "TVirtualMCStack.h" // for TVirtualMCStack
40
41#include <cstdio> // for NULL, snprintf
42
43class FairModule;
44
45class TGeoMedium;
46
47class TParticle;
48
49using namespace o2::ft3;
50using o2::itsmft::Hit;
51
52//_________________________________________________________________________________________________
54 : o2::base::DetImpl<Detector>("FT3", kTRUE),
55 mTrackData(),
56 mHits(o2::utils::createSimVector<o2::itsmft::Hit>())
57{
58}
59
60//_________________________________________________________________________________________________
61void Detector::buildFT3FromFile(std::string configFileName)
62{
63 // Geometry description from file. One line per disk
64 // z_layer r_in r_out Layerx2X0
65 // This simple file reader is not failproof. Do not add empty lines!
66
67 /*
68 # Sample FT3 configuration
69 # z_layer r_in r_out Layerx2X0
70 -45.3 2.5 9.26 0.0042
71 -46.7 2.5 9.26 0.0042
72 -48.6 2.5 9.8 0.0042
73 -50.0 2.5 9.8 0.0042
74 -52.4 2.5 10.43 0.0042
75 -53.8 2.5 10.43 0.0042
76 -67.7 3.82 13.01 0.0042
77 -69.1 3.82 13.01 0.0042
78 -76.1 3.92 14.35 0.0042
79 -77.5 3.92 14.35 0.0042
80 */
81
82 mLayerName.clear();
83 mLayers.clear();
84 mLayerID.clear();
85 mLayerName.resize(1);
86 mLayers.resize(1);
87
88 LOG(info) << "Building FT3 Detector: From file";
89 LOG(info) << " FT3 detector configuration: " << configFileName;
90 std::ifstream ifs(configFileName.c_str());
91 if (!ifs.good()) {
92 LOG(fatal) << " Invalid FT3Base.configFile!";
93 }
94 std::string tempstr;
95 float z_layer, r_in, r_out, Layerx2X0;
96 char delimiter;
97 int layerNumber = 0;
98 while (std::getline(ifs, tempstr)) {
99 if (tempstr[0] == '#') {
100 LOG(info) << " Comment: " << tempstr;
101 continue;
102 }
103 std::istringstream iss(tempstr);
104 iss >> z_layer;
105 iss >> r_in;
106 iss >> r_out;
107 iss >> Layerx2X0;
108
109 int direction = 1; // Forwards
110 if (z_layer < 0) {
111 // Backwards
112 direction = 0;
113 }
114
115 std::string directionName = std::to_string(direction);
116 std::string layerName = GeometryTGeo::getFT3LayerPattern() + directionName + std::string("_") + std::to_string(layerNumber);
117 mLayerName[0].push_back(layerName);
118 LOG(info) << "Adding Layer " << layerName << " at z = " << z_layer << " ; direction = " << direction << " ; r_in = " << r_in << " ; r_out = " << r_out << " x/X0 = " << Layerx2X0;
119 auto& thisLayer = mLayers[0].emplace_back(direction, layerNumber, layerName, z_layer, r_in, r_out, Layerx2X0);
120 layerNumber++;
121 }
122
123 mNumberOfLayers = layerNumber;
124 LOG(info) << " Loaded FT3 Detector with " << mNumberOfLayers << " layers";
125}
126
127//_________________________________________________________________________________________________
129{
130 // Export FT3 Layout description to file. One line per disk
131 // z_layer r_in r_out Layerx2X0
132
133 std::string configFileName = "FT3_layout.cfg";
134
135 LOG(info) << "Exporting FT3 Detector layout to " << configFileName;
136
137 std::ofstream fOut(configFileName.c_str(), std::ios::out);
138 if (!fOut) {
139 printf("Cannot open file\n");
140 return;
141 }
142 fOut << "# z_layer r_in r_out Layerx2X0" << std::endl;
143 for (auto layers_dir : mLayers) {
144 for (auto layer : layers_dir) {
145 fOut << layer.getZ() << " " << layer.getInnerRadius() << " " << layer.getOuterRadius() << " " << layer.getx2X0() << std::endl;
146 }
147 }
148}
149
150//_________________________________________________________________________________________________
152{
153 // Build a basic parametrized FT3 detector with nLayers equally spaced between z_first and z_first+z_length
154 // Covering pseudo rapidity [etaIn,etaOut]. Silicon thinkness computed to match layer x/X0
155
156 LOG(info) << "Building FT3 Detector: Conical Telescope";
157
158 auto z_first = param.z0;
159 auto z_length = param.zLength;
160 auto etaIn = param.etaIn;
161 auto etaOut = param.etaOut;
162 auto Layerx2X0 = param.Layerx2X0;
163 mNumberOfLayers = param.nLayers;
164 mLayerName.resize(2);
165 mLayerName[0].resize(mNumberOfLayers);
166 mLayerName[1].resize(mNumberOfLayers);
167 mLayerID.clear();
168 mLayers.resize(2);
169
170 for (int direction : {0, 1}) {
171 for (int layerNumber = 0; layerNumber < mNumberOfLayers; layerNumber++) {
172 std::string layerName = GeometryTGeo::getFT3LayerPattern() + std::to_string(layerNumber + mNumberOfLayers * direction);
173 mLayerName[direction][layerNumber] = layerName;
174
175 // Adds evenly spaced layers
176 float layerZ = z_first + (layerNumber * z_length / (mNumberOfLayers - 1)) * std::copysign(1, z_first);
177 float rIn = std::abs(layerZ * std::tan(2.f * std::atan(std::exp(-etaIn))));
178 float rOut = std::abs(layerZ * std::tan(2.f * std::atan(std::exp(-etaOut))));
179 auto& thisLayer = mLayers[direction].emplace_back(direction, layerNumber, layerName, layerZ, rIn, rOut, Layerx2X0);
180 }
181 }
182}
183
184//_________________________________________________________________________________________________
186{
187 // Build FT3 detector according to
188 // https://indico.cern.ch/event/992488/contributions/4174473/attachments/2168881/3661331/tracker_parameters_werner_jan_11_2021.pdf
189
190 LOG(info) << "Building FT3 Detector: V1";
191
192 mNumberOfLayers = 10;
193 float sensorThickness = 30.e-4;
194 float layersx2X0 = 1.e-2;
195 std::vector<std::array<float, 5>> layersConfig{
196 {26., .5, 3., 0.1f * layersx2X0}, // {z_layer, r_in, r_out, Layerx2X0}
197 {30., .5, 3., 0.1f * layersx2X0},
198 {34., .5, 3., 0.1f * layersx2X0},
199 {77., 3.5, 35., layersx2X0},
200 {100., 3.5, 35., layersx2X0},
201 {122., 3.5, 35., layersx2X0},
202 {150., 3.5, 80.f, layersx2X0},
203 {180., 3.5, 80.f, layersx2X0},
204 {220., 3.5, 80.f, layersx2X0},
205 {279., 3.5, 80.f, layersx2X0}};
206
207 mLayerName.resize(2);
208 mLayerName[0].resize(mNumberOfLayers);
209 mLayerName[1].resize(mNumberOfLayers);
210 mLayerID.clear();
211 mLayers.resize(2);
212
213 for (auto direction : {0, 1}) {
214 for (int layerNumber = 0; layerNumber < mNumberOfLayers; layerNumber++) {
215 std::string directionName = std::to_string(direction);
216 std::string layerName = GeometryTGeo::getFT3LayerPattern() + directionName + std::string("_") + std::to_string(layerNumber);
217 mLayerName[direction][layerNumber] = layerName;
218 auto& z = layersConfig[layerNumber][0];
219
220 auto& rIn = layersConfig[layerNumber][1];
221 auto& rOut = layersConfig[layerNumber][2];
222 auto& x0 = layersConfig[layerNumber][3];
223
224 LOG(info) << "Adding Layer " << layerName << " at z = " << z;
225 // Add layers
226 auto& thisLayer = mLayers[direction].emplace_back(direction, layerNumber, layerName, z, rIn, rOut, x0);
227 }
228 }
229}
230
231//_________________________________________________________________________________________________
233{
234 // Build FT3 detector according to
235 // https://www.overleaf.com/project/6051acc870e39aaeb4653621
236
237 LOG(info) << "Building FT3 Detector: V3b";
238
239 mNumberOfLayers = 12;
240 float sensorThickness = 30.e-4;
241 float layersx2X0 = 1.e-2;
242 std::vector<std::array<float, 5>> layersConfig{
243 {26., .5, 3., 0.1f * layersx2X0}, // {z_layer, r_in, r_out, Layerx2X0}
244 {30., .5, 3., 0.1f * layersx2X0},
245 {34., .5, 3., 0.1f * layersx2X0},
246 {77., 5.0, 35., layersx2X0},
247 {100., 5.0, 35., layersx2X0},
248 {122., 5.0, 35., layersx2X0},
249 {150., 5.5, 80.f, layersx2X0},
250 {180., 6.6, 80.f, layersx2X0},
251 {220., 8.1, 80.f, layersx2X0},
252 {279., 10.2, 80.f, layersx2X0},
253 {340., 12.5, 80.f, layersx2X0},
254 {400., 14.7, 80.f, layersx2X0}};
255
256 mLayerName.resize(2);
257 mLayerName[0].resize(mNumberOfLayers);
258 mLayerName[1].resize(mNumberOfLayers);
259 mLayerID.clear();
260 mLayers.resize(2);
261
262 for (auto direction : {0, 1}) {
263 for (int layerNumber = 0; layerNumber < mNumberOfLayers; layerNumber++) {
264 std::string directionName = std::to_string(direction);
265 std::string layerName = GeometryTGeo::getFT3LayerPattern() + directionName + std::string("_") + std::to_string(layerNumber);
266 mLayerName[direction][layerNumber] = layerName;
267 auto& z = layersConfig[layerNumber][0];
268
269 auto& rIn = layersConfig[layerNumber][1];
270 auto& rOut = layersConfig[layerNumber][2];
271 auto& x0 = layersConfig[layerNumber][3];
272
273 LOG(info) << "Adding Layer " << layerName << " at z = " << z;
274 // Add layers
275 auto& thisLayer = mLayers[direction].emplace_back(direction, layerNumber, layerName, z, rIn, rOut, x0);
276 }
277 }
278}
279
281{
282 // Build the FT3 detector according to changes proposed during
283 // https://indico.cern.ch/event/1407704/
284 // to adhere to the changes that were presented at the ALICE 3 Upgrade days in March 2024
285 // Inner radius at C-side to 7 cm
286 // Inner radius at A-side stays at 5 cm
287 // 06.02.2025 update: IRIS layers are now in TRK
288
289 LOG(info) << "Building FT3 Detector: After Upgrade Days March 2024 version";
290
291 mNumberOfLayers = 9;
292 float sensorThickness = 30.e-4;
293 float layersx2X0 = 1.e-2;
294 std::vector<std::array<float, 5>> layersConfigCSide{
295 {77., 7.0, 35., layersx2X0}, // {z_layer, r_in, r_out, Layerx2X0}
296 {100., 7.0, 35., layersx2X0},
297 {122., 7.0, 35., layersx2X0},
298 {150., 7.0, 68.f, layersx2X0},
299 {180., 7.0, 68.f, layersx2X0},
300 {220., 7.0, 68.f, layersx2X0},
301 {260., 7.0, 68.f, layersx2X0},
302 {300., 7.0, 68.f, layersx2X0},
303 {350., 7.0, 68.f, layersx2X0}};
304
305 std::vector<std::array<float, 5>> layersConfigASide{
306 {77., 5.0, 35., layersx2X0}, // {z_layer, r_in, r_out, Layerx2X0}
307 {100., 5.0, 35., layersx2X0},
308 {122., 5.0, 35., layersx2X0},
309 {150., 5.0, 68.f, layersx2X0},
310 {180., 5.0, 68.f, layersx2X0},
311 {220., 5.0, 68.f, layersx2X0},
312 {260., 5.0, 68.f, layersx2X0},
313 {300., 5.0, 68.f, layersx2X0},
314 {350., 5.0, 68.f, layersx2X0}};
315
316 mLayerName.resize(2);
317 mLayerName[0].resize(mNumberOfLayers);
318 mLayerName[1].resize(mNumberOfLayers);
319 mLayerID.clear();
320 mLayers.resize(2);
321
322 for (auto direction : {0, 1}) {
323 for (int layerNumber = 0; layerNumber < mNumberOfLayers; layerNumber++) {
324 std::string directionName = std::to_string(direction);
325 std::string layerName = GeometryTGeo::getFT3LayerPattern() + directionName + std::string("_") + std::to_string(layerNumber);
326 mLayerName[direction][layerNumber] = layerName;
327 float z, rIn, rOut, x0;
328 if (direction == 0) { // C-Side
329 z = layersConfigCSide[layerNumber][0];
330 rIn = layersConfigCSide[layerNumber][1];
331 rOut = layersConfigCSide[layerNumber][2];
332 x0 = layersConfigCSide[layerNumber][3];
333 } else if (direction == 1) { // A-Side
334 z = layersConfigASide[layerNumber][0];
335 rIn = layersConfigASide[layerNumber][1];
336 rOut = layersConfigASide[layerNumber][2];
337 x0 = layersConfigASide[layerNumber][3];
338 }
339
340 LOG(info) << "Adding Layer " << layerName << " at z = " << z;
341 // Add layers
342 auto& thisLayer = mLayers[direction].emplace_back(direction, layerNumber, layerName, z, rIn, rOut, x0);
343 }
344 }
345}
346
347//_________________________________________________________________________________________________
349{
350 // Build FT3 detector according to the scoping document
351
352 LOG(info) << "Building FT3 Detector: Scoping document version";
353
354 mNumberOfLayers = 12;
355 float sensorThickness = 30.e-4;
356 float layersx2X0 = 1.e-2;
357 std::vector<std::array<float, 5>> layersConfig{
358 {26., .5, 2.5, 0.1f * layersx2X0}, // {z_layer, r_in, r_out, Layerx2X0}
359 {30., .5, 2.5, 0.1f * layersx2X0},
360 {34., .5, 2.5, 0.1f * layersx2X0},
361 {77., 5.0, 35., layersx2X0},
362 {100., 5.0, 35., layersx2X0},
363 {122., 5.0, 35., layersx2X0},
364 {150., 5.0, 68.f, layersx2X0},
365 {180., 5.0, 68.f, layersx2X0},
366 {220., 5.0, 68.f, layersx2X0},
367 {260., 5.0, 68.f, layersx2X0},
368 {300., 5.0, 68.f, layersx2X0},
369 {350., 5.0, 68.f, layersx2X0}};
370
371 mLayerName.resize(2);
372 mLayerName[0].resize(mNumberOfLayers);
373 mLayerName[1].resize(mNumberOfLayers);
374 mLayerID.clear();
375 mLayers.resize(2);
376
377 for (auto direction : {0, 1}) {
378 for (int layerNumber = 0; layerNumber < mNumberOfLayers; layerNumber++) {
379 std::string directionName = std::to_string(direction);
380 std::string layerName = GeometryTGeo::getFT3LayerPattern() + directionName + std::string("_") + std::to_string(layerNumber);
381 mLayerName[direction][layerNumber] = layerName;
382 auto& z = layersConfig[layerNumber][0];
383
384 auto& rIn = layersConfig[layerNumber][1];
385 auto& rOut = layersConfig[layerNumber][2];
386 auto& x0 = layersConfig[layerNumber][3];
387
388 LOG(info) << "Adding Layer " << layerName << " at z = " << z;
389 // Add layers
390 auto& thisLayer = mLayers[direction].emplace_back(direction, layerNumber, layerName, z, rIn, rOut, x0);
391 }
392 }
393}
394
395//_________________________________________________________________________________________________
397 : o2::base::DetImpl<Detector>("FT3", active),
398 mTrackData(),
399 mHits(o2::utils::createSimVector<o2::itsmft::Hit>())
400{
401
402 // FT3 Base configuration parameters
403 auto& ft3BaseParam = FT3BaseParam::Instance();
404
405 if (ft3BaseParam.configFile != "") {
406 LOG(info) << "FT3 Geometry configuration file provided. Overriding FT3Base.geoModel configuration.";
407 buildFT3FromFile(ft3BaseParam.configFile);
408
409 } else {
410 switch (ft3BaseParam.geoModel) {
411 case Default:
412 buildFT3NewVacuumVessel(); // FT3 after Upgrade days March 2024
413 break;
414 case Telescope:
415 buildBasicFT3(ft3BaseParam); // BasicFT3 = Parametrized telescopic detector (equidistant layers)
416 break;
417 default:
418 LOG(fatal) << "Invalid Geometry.\n";
419 break;
420 }
421 }
422 exportLayout();
423}
424
425//_________________________________________________________________________________________________
427 : o2::base::DetImpl<Detector>(rhs),
428 mTrackData(),
429
431 mHits(o2::utils::createSimVector<o2::itsmft::Hit>())
432{
433 mLayerID = rhs.mLayerID;
434 mLayerName = rhs.mLayerName;
435 mNumberOfLayers = rhs.mNumberOfLayers;
436}
437
438//_________________________________________________________________________________________________
440{
441
442 if (mHits) {
443 // delete mHits;
445 }
446}
447
448//_________________________________________________________________________________________________
450{
451 // The standard = operator
452 // Inputs:
453 // Detector &h the sourse of this copy
454 // Outputs:
455 // none.
456 // Return:
457 // A copy of the sourse hit h
458
459 if (this == &rhs) {
460 return *this;
461 }
462
463 // base class assignment
465
466 mLayerID = rhs.mLayerID;
467 mLayerName = rhs.mLayerName;
468 mNumberOfLayers = rhs.mNumberOfLayers;
469 mLayers = rhs.mLayers;
470 mTrackData = rhs.mTrackData;
471
473 mHits = nullptr;
474
475 return *this;
476}
477
478//_________________________________________________________________________________________________
480{
481 // Define the list of sensitive volumes
482 LOG(info) << "Initialize FT3 O2Detector";
483
484 mGeometryTGeo = GeometryTGeo::Instance();
485
486 defineSensitiveVolumes();
487}
488
489//_________________________________________________________________________________________________
490bool Detector::ProcessHits(FairVolume* vol)
491{
492 // This method is called from the MC stepping
493 if (!(fMC->TrackCharge())) {
494 return kFALSE;
495 }
496
497 int lay = 0, volID = vol->getMCid();
498 while ((lay <= mLayerID.size()) && (volID != mLayerID[lay])) {
499 ++lay;
500 }
501
502 auto stack = (o2::data::Stack*)fMC->GetStack();
503
504 bool startHit = false, stopHit = false;
505 unsigned char status = 0;
506 if (fMC->IsTrackEntering()) {
507 status |= Hit::kTrackEntering;
508 }
509 if (fMC->IsTrackInside()) {
510 status |= Hit::kTrackInside;
511 }
512 if (fMC->IsTrackExiting()) {
513 status |= Hit::kTrackExiting;
514 }
515 if (fMC->IsTrackOut()) {
516 status |= Hit::kTrackOut;
517 }
518 if (fMC->IsTrackStop()) {
519 status |= Hit::kTrackStopped;
520 }
521 if (fMC->IsTrackAlive()) {
522 status |= Hit::kTrackAlive;
523 }
524
525 // track is entering or created in the volume
526 if ((status & Hit::kTrackEntering) || (status & Hit::kTrackInside && !mTrackData.mHitStarted)) {
527 startHit = true;
528 } else if ((status & (Hit::kTrackExiting | Hit::kTrackOut | Hit::kTrackStopped))) {
529 stopHit = true;
530 }
531
532 // increment energy loss at all steps except entrance
533 if (!startHit) {
534 mTrackData.mEnergyLoss += fMC->Edep();
535 }
536 if (!(startHit | stopHit)) {
537 return kFALSE; // do noting
538 }
539 if (startHit) {
540 mTrackData.mEnergyLoss = 0.;
541 fMC->TrackMomentum(mTrackData.mMomentumStart);
542 fMC->TrackPosition(mTrackData.mPositionStart);
543 mTrackData.mTrkStatusStart = status;
544 mTrackData.mHitStarted = true;
545 }
546 if (stopHit) {
547 TLorentzVector positionStop;
548 fMC->TrackPosition(positionStop);
549 // Retrieve the indices with the volume path
550 int chipindex = lay;
551
552 Hit* p = addHit(stack->GetCurrentTrackNumber(), chipindex, mTrackData.mPositionStart.Vect(), positionStop.Vect(),
553 mTrackData.mMomentumStart.Vect(), mTrackData.mMomentumStart.E(), positionStop.T(),
554 mTrackData.mEnergyLoss, mTrackData.mTrkStatusStart, status);
555 // p->SetTotalEnergy(vmc->Etot());
556
557 // RS: not sure this is needed
558 // Increment number of Detector det points in TParticle
559 stack->addHit(GetDetId());
560 }
561
562 return kTRUE;
563}
564
565//_________________________________________________________________________________________________
566void Detector::createMaterials()
567{
568 int ifield = 2;
569 float fieldm = 10.0;
571
572 float tmaxfdSi = 0.1; // .10000E+01; // Degree
573 float stemaxSi = 0.0075; // .10000E+01; // cm
574 float deemaxSi = 0.1; // 0.30000E-02; // Fraction of particle's energy 0<deemax<=1
575 float epsilSi = 1.0E-4; // .10000E+01;
576 float stminSi = 0.0; // cm "Default value used"
577
578 float tmaxfdAir = 0.1; // .10000E+01; // Degree
579 float stemaxAir = .10000E+01; // cm
580 float deemaxAir = 0.1; // 0.30000E-02; // Fraction of particle's energy 0<deemax<=1
581 float epsilAir = 1.0E-4; // .10000E+01;
582 float stminAir = 0.0; // cm "Default value used"
583
584 // AIR
585 float aAir[4] = {12.0107, 14.0067, 15.9994, 39.948};
586 float zAir[4] = {6., 7., 8., 18.};
587 float wAir[4] = {0.000124, 0.755267, 0.231781, 0.012827};
588 float dAir = 1.20479E-3;
589
590 o2::base::Detector::Mixture(1, "AIR$", aAir, zAir, dAir, 4, wAir);
591 o2::base::Detector::Medium(1, "AIR$", 1, 0, ifield, fieldm, tmaxfdAir, stemaxAir, deemaxAir, epsilAir, stminAir);
592
593 o2::base::Detector::Material(3, "SILICON$", 0.28086E+02, 0.14000E+02, 0.23300E+01, 0.93600E+01, 0.99900E+03);
594 o2::base::Detector::Medium(3, "SILICON$", 3, 0, ifield, fieldm, tmaxfdSi, stemaxSi, deemaxSi, epsilSi, stminSi);
595}
596
597//_________________________________________________________________________________________________
598void Detector::EndOfEvent() { Reset(); }
599
600//_________________________________________________________________________________________________
602{
603 // This will create a branch in the output tree called Hit, setting the last
604 // parameter to kFALSE means that this collection will not be written to the file,
605 // it will exist only during the simulation
606
607 if (FairRootManager::Instance()) {
608 FairRootManager::Instance()->RegisterAny(addNameTo("Hit").data(), mHits, kTRUE);
609 }
610}
611
612//_________________________________________________________________________________________________
614{
615 if (!o2::utils::ShmManager::Instance().isOperational()) {
616 mHits->clear();
617 }
618}
619
620//_________________________________________________________________________________________________
622{
623 // Create detector materials
624 createMaterials();
625
626 // Construct the detector geometry
627 createGeometry();
628}
629
630//_________________________________________________________________________________________________
631void Detector::createGeometry()
632{
633
634 mGeometryTGeo = GeometryTGeo::Instance();
635
636 TGeoVolume* volFT3 = new TGeoVolumeAssembly(GeometryTGeo::getFT3VolPattern());
637 TGeoVolume* volIFT3 = new TGeoVolumeAssembly(GeometryTGeo::getFT3InnerVolPattern());
638
639 LOG(info) << "GeometryBuilder::buildGeometry volume name = " << GeometryTGeo::getFT3VolPattern();
640
641 TGeoVolume* vALIC = gGeoManager->GetVolume("barrel");
642 if (!vALIC) {
643 LOG(fatal) << "Could not find the top volume";
644 }
645
646 TGeoVolume* A3IPvac = gGeoManager->GetVolume("OUT_PIPEVACUUM");
647 if (!A3IPvac) {
648 LOG(info) << "Running simulation with no beam pipe.";
649 }
650
651 LOG(debug) << "FT3 createGeometry: "
652 << Form("gGeoManager name is %s title is %s", gGeoManager->GetName(), gGeoManager->GetTitle());
653
654 if (mLayers.size() == 2) { // V1 and telescope
655 if (!A3IPvac) {
656 for (int direction : {0, 1}) { // Backward layers at mLayers[0]; Forward layers at mLayers[1]
657 std::string directionString = direction ? "Forward" : "Backward";
658 LOG(info) << "Creating FT3 " << directionString << " layers:";
659 for (int iLayer = 0; iLayer < mLayers[direction].size(); iLayer++) {
660 mLayers[direction][iLayer].createLayer(volFT3);
661 }
662 }
663 vALIC->AddNode(volFT3, 2, new TGeoTranslation(0., 30., 0.));
664 } else { // If beampipe is enabled append inner disks to beampipe filling volume, this should be temporary.
665 for (int direction : {0, 1}) {
666 std::string directionString = direction ? "Forward" : "Backward";
667 LOG(info) << "Creating FT3 " << directionString << " layers:";
668 for (int iLayer = 0; iLayer < mLayers[direction].size(); iLayer++) {
669 if (iLayer < 3) {
670 mLayers[direction][iLayer].createLayer(volIFT3);
671 } else {
672 mLayers[direction][iLayer].createLayer(volFT3);
673 }
674 }
675 }
676 A3IPvac->AddNode(volIFT3, 2, new TGeoTranslation(0., 0., 0.));
677 vALIC->AddNode(volFT3, 2, new TGeoTranslation(0., 30., 0.));
678 }
679
680 for (auto direction : {0, 1}) {
681 std::string directionString = direction ? "Forward" : "Backward";
682 LOG(info) << "Registering FT3 " << directionString << " LayerIDs:";
683 for (int iLayer = 0; iLayer < mLayers[direction].size(); iLayer++) {
684 auto layerID = gMC ? TVirtualMC::GetMC()->VolId(Form("%s_%d_%d", GeometryTGeo::getFT3SensorPattern(), direction, iLayer)) : 0;
685 mLayerID.push_back(layerID);
686 LOG(info) << " " << directionString << " layer " << iLayer << " LayerID " << layerID;
687 }
688 }
689 }
690
691 if (mLayers.size() == 1) { // All layers registered at mLayers[0], used when building from file
692 LOG(info) << "Creating FT3 layers:";
693 if (A3IPvac) {
694 for (int iLayer = 0; iLayer < mLayers[0].size(); iLayer++) {
695 if (std::abs(mLayers[0][iLayer].getZ()) < 25) {
696 mLayers[0][iLayer].createLayer(volIFT3);
697 } else {
698 mLayers[0][iLayer].createLayer(volFT3);
699 }
700 }
701 A3IPvac->AddNode(volIFT3, 2, new TGeoTranslation(0., 0., 0.));
702 vALIC->AddNode(volFT3, 2, new TGeoTranslation(0., 30., 0.));
703 } else {
704 for (int iLayer = 0; iLayer < mLayers[0].size(); iLayer++) {
705 mLayers[0][iLayer].createLayer(volFT3);
706 }
707 vALIC->AddNode(volFT3, 2, new TGeoTranslation(0., 30., 0.));
708 }
709 LOG(info) << "Registering FT3 LayerIDs:";
710 for (int iLayer = 0; iLayer < mLayers[0].size(); iLayer++) {
711 auto layerID = gMC ? TVirtualMC::GetMC()->VolId(Form("%s_%d_%d", GeometryTGeo::getFT3SensorPattern(), 0, iLayer)) : 0;
712 mLayerID.push_back(layerID);
713 LOG(info) << " mLayerID[" << iLayer << "] = " << layerID;
714 }
715 }
716}
717
718//_________________________________________________________________________________________________
719void Detector::defineSensitiveVolumes()
720{
721 TGeoManager* geoManager = gGeoManager;
722 TGeoVolume* v;
723
724 TString volumeName;
725 LOG(info) << "Adding FT3 Sensitive Volumes";
726
727 // The names of the FT3 sensitive volumes have the format: FT3Sensor_(0,1)_(0...sNumberLayers-1)
728 if (mLayers.size() == 2) {
729 for (int direction : {0, 1}) {
730 for (int iLayer = 0; iLayer < mNumberOfLayers; iLayer++) {
732 v = geoManager->GetVolume(Form("%s_%d_%d", GeometryTGeo::getFT3SensorPattern(), direction, iLayer));
733 LOG(info) << "Adding FT3 Sensitive Volume => " << v->GetName();
734 AddSensitiveVolume(v);
735 }
736 }
737 }
738
739 if (mLayers.size() == 1) {
740 for (int iLayer = 0; iLayer < mLayers[0].size(); iLayer++) {
742 v = geoManager->GetVolume(Form("%s_%d_%d", GeometryTGeo::getFT3SensorPattern(), mLayers[0][iLayer].getDirection(), iLayer));
743 LOG(info) << "Adding FT3 Sensitive Volume => " << v->GetName();
744 AddSensitiveVolume(v);
745 }
746 }
747}
748
749//_________________________________________________________________________________________________
750Hit* Detector::addHit(int trackID, int detID, const TVector3& startPos, const TVector3& endPos,
751 const TVector3& startMom, double startE, double endTime, double eLoss, unsigned char startStatus,
752 unsigned char endStatus)
753{
754 mHits->emplace_back(trackID, detID, startPos, endPos, startMom, startE, endTime, eLoss, startStatus, endStatus);
755 return &(mHits->back());
756}
757
Definition of the Stack class.
Definition of the ITSMFT Hit class.
Definition of the FT3Layer class.
ClassImp(IdPath)
uint32_t stack
Definition RawData.h:1
Definition of the GeometryTGeo class.
Definition of the Detector class.
std::ostringstream debug
Detector & operator=(const Detector &)
Definition Detector.cxx:46
void Mixture(Int_t imat, const char *name, Float_t *a, Float_t *z, Float_t dens, Int_t nlmat, Float_t *wmat)
Definition Detector.cxx:66
void Medium(Int_t numed, const char *name, Int_t nmat, Int_t isvol, Int_t ifield, Float_t fieldm, Float_t tmaxfd, Float_t stemax, Float_t deemax, Float_t epsil, Float_t stmin, Float_t *ubuf=nullptr, Int_t nbuf=0)
Definition Detector.cxx:72
static void initFieldTrackingParams(int &mode, float &maxfield)
Definition Detector.cxx:143
virtual void InitializeO2Detector()=0
Definition Detector.cxx:98
void Material(Int_t imat, const char *name, Float_t a, Float_t z, Float_t dens, Float_t radl, Float_t absl, Float_t *buf=nullptr, Int_t nwbuf=0)
Definition Detector.cxx:59
std::string addNameTo(const char *ext) const
Definition Detector.h:150
std::vector< std::vector< TString > > mLayerName
Definition Detector.h:127
void EndOfEvent() override
Definition Detector.cxx:598
o2::itsmft::Hit * addHit(int trackID, int detID, const TVector3 &startPos, const TVector3 &endPos, const TVector3 &startMom, double startE, double endTime, double eLoss, unsigned char startStatus, unsigned char endStatus)
This method is an example of how to add your own point of type Hit to the clones array.
Definition Detector.cxx:750
void Register() override
Registers the produced collections in FAIRRootManager.
Definition Detector.cxx:601
void buildFT3FromFile(std::string)
Definition Detector.cxx:61
void ConstructGeometry() override
Base class to create the detector geometry.
Definition Detector.cxx:621
Detector()
Default constructor.
Definition Detector.cxx:53
Bool_t ProcessHits(FairVolume *v=nullptr) override
This method is called for each step during simulation (see FairMCApplication::Stepping())
Definition Detector.cxx:490
std::vector< Int_t > mLayerID
Definition Detector.h:126
void buildBasicFT3(const FT3BaseParam &param)
Definition Detector.cxx:151
void buildFT3NewVacuumVessel()
Definition Detector.cxx:280
void Reset() override
Has to be called after each event to reset the containers.
Definition Detector.cxx:613
void exportLayout()
access to geometry details
Definition Detector.cxx:128
Int_t mNumberOfLayers
Definition Detector.h:128
static const char * getFT3SensorPattern()
static const char * getFT3LayerPattern()
static GeometryTGeo * Instance()
static const char * getFT3VolPattern()
static const char * getFT3InnerVolPattern()
static ShmManager & Instance()
Definition ShmManager.h:61
const GLdouble * v
Definition glcorearb.h:832
GLboolean * data
Definition glcorearb.h:298
GLuint GLfloat x0
Definition glcorearb.h:5034
GLenum GLuint GLint GLint layer
Definition glcorearb.h:1310
GLenum GLfloat param
Definition glcorearb.h:271
GLdouble GLdouble GLdouble z
Definition glcorearb.h:843
void freeSimVector(std::vector< T > *ptr)
a couple of static helper functions to create timestamp values for CCDB queries or override obsolete ...
std::string to_string(gsl::span< T, Size > span)
Definition common.h:52
Common utility functions.
LOG(info)<< "Compressed in "<< sw.CpuTime()<< " s"