86 return StatusCode::FAILURE;
94 m_autoRetrieveTools =
false;
95 m_checkToolDeps =
false;
98 << Acts::VersionMajor <<
"." << Acts::VersionMinor <<
"."
99 << Acts::VersionPatch <<
" [" << Acts::CommitHash.value_or(
"unknown hash") <<
"]");
104 ATH_MSG_INFO(
"Configured to build " << buildSubdet.size()
105 <<
" subdetectors:");
106 for (
const auto &s : buildSubdet) {
111 if (buildSubdet.find(
"Pixel") != buildSubdet.end()) {
114 if (buildSubdet.find(
"SCT") != buildSubdet.end()) {
117 if (buildSubdet.find(
"TRT") != buildSubdet.end()) {
121 if (buildSubdet.find(
"ITkPixel") != buildSubdet.end()) {
124 if (buildSubdet.find(
"ITkStrip") != buildSubdet.end()) {
127 if (buildSubdet.find(
"HGTD") != buildSubdet.end()) {
139 ATH_MSG_FATAL(
"Consistency check for ITk inner pixel barrel passive layer construction failed. Please check your inputs! ");
140 return StatusCode::FAILURE;
145 ATH_MSG_FATAL(
"Consistency check for ITk outer pixel barrel passive layer construction failed. Please check your inputs! ");
146 return StatusCode::FAILURE;
151 ATH_MSG_FATAL(
"Consistency check for ITk strip barrel passive layer construction failed. Please check your inputs! ");
152 return StatusCode::FAILURE;
158 ATH_MSG_INFO(
"Using Blueprint API for geometry construction");
165 using enum Acts::AxisDirection;
167 std::vector<ActsTrk::IBlueprintNodeBuilder*> ptrBuilders;
169 std::back_inserter(ptrBuilders),
170 [](ToolHandle<ActsTrk::IBlueprintNodeBuilder>& b) { return b.get(); });
174 Acts::Experimental::Blueprint::Config cfg;
175 cfg.envelope[AxisZ] = {20_mm, 20_mm};
176 cfg.envelope[AxisR] = {0_mm, 20_mm};
178 auto blueprint = std::make_unique<Acts::Experimental::Blueprint>(cfg);
180 auto& root = blueprint->addCylinderContainer(
"Detector", AxisZ);
182 std::shared_ptr<Acts::Experimental::BlueprintNode> currentTop{
nullptr};
184 for (
auto& builder : ptrBuilders) {
185 currentTop = builder->buildBlueprintNode(
getNominalContext().context(), std::move(currentTop));
189 root.addChild(std::move(currentTop));
191 std::unique_ptr<Acts::TrackingGeometry>
trackingGeometry = blueprint->construct(
203 Acts::ObjVisualization3D vis;
205 {.visible =
false}, {.visible =
true});
206 vis.write(
"blueprint_sensitive.obj");
210 {.visible =
false}, {.visible =
false});
211 vis.write(
"blueprint_volume.obj");
215 {.visible =
true}, {.visible =
false});
216 vis.write(
"blueprint_portals.obj");
219 Acts::detail::TrackingGeometryPrintVisitor printer{
m_nominalContext.context()};
221 ATH_MSG_INFO(
"Built tracking geometry \n"<<printer.stream().str());
224 return StatusCode::SUCCESS;
229 Acts::LayerArrayCreator::Config lacCfg;
230 auto layerArrayCreator = std::make_shared<const Acts::LayerArrayCreator>(
233 Acts::TrackingVolumeArrayCreator::Config tvcCfg;
234 auto trackingVolumeArrayCreator =
235 std::make_shared<const Acts::TrackingVolumeArrayCreator>(
238 Acts::CylinderVolumeHelper::Config cvhConfig;
239 cvhConfig.layerArrayCreator = layerArrayCreator;
240 cvhConfig.trackingVolumeArrayCreator = trackingVolumeArrayCreator;
242 auto cylinderVolumeHelper =
243 std::make_shared<const Acts::CylinderVolumeHelper>(
246 Acts::TrackingGeometryBuilder::Config tgbConfig;
247 tgbConfig.trackingVolumeHelper = cylinderVolumeHelper;
250 std::shared_ptr<const Acts::IMaterialDecorator> matDeco =
nullptr;
253 if (matFileFullPath.empty()) {
255 return StatusCode::FAILURE;
257 ATH_MSG_INFO(
"Configured to use material input: " << matFileFullPath);
259 if (matFileFullPath.find(
".json") != std::string::npos) {
261 Acts::MaterialMapJsonConverter::Config jsonGeoConvConfig;
263 matDeco = std::make_shared<const Acts::JsonMaterialDecorator>(
266 tgbConfig.materialDecorator = matDeco;
269 std::array<double, 2> sctECEnvelopeZ{20_mm, 20_mm};
274 tgbConfig.trackingVolumeBuilders.push_back([&](
const auto &gctx,
278 Acts::CylinderVolumeBuilder::Config bpvConfig =
281 Acts::CylinderVolumeBuilder beamPipeVolumeBuilder {
284 return beamPipeVolumeBuilder.trackingVolume(gctx, inner);
291 if (buildSubdet.count(
"Pixel") > 0) {
292 tgbConfig.trackingVolumeBuilders.push_back([&](
const auto &gctx,
297 auto lb = std::make_shared<ActsLayerBuilder>(
299 Acts::CylinderVolumeBuilder::Config cvbConfig;
300 cvbConfig.layerEnvelopeR = {3_mm, 3_mm};
301 cvbConfig.layerEnvelopeZ = 1_mm;
302 cvbConfig.trackingVolumeHelper = cylinderVolumeHelper;
303 cvbConfig.volumeName =
"Pixel";
304 cvbConfig.layerBuilder =
lb;
307 Acts::CylinderVolumeBuilder cvb(
310 return cvb.trackingVolume(gctx, inner);
315 if (buildSubdet.count(
"ITkPixel") > 0) {
316 tgbConfig.trackingVolumeBuilders.push_back(
317 [&](
const auto &gctx,
const auto &inner,
const auto &) {
321 cfg.doEndcapLayerMerging =
true;
325 auto lb = std::make_shared<ActsLayerBuilder>(
328 Acts::CylinderVolumeBuilder::Config cvbConfig;
329 cvbConfig.layerEnvelopeR = {5_mm, 5_mm};
330 cvbConfig.layerEnvelopeZ = 1_mm;
331 cvbConfig.trackingVolumeHelper = cylinderVolumeHelper;
332 cvbConfig.volumeName =
"ITkPixelInner";
333 cvbConfig.layerBuilder =
lb;
336 Acts::CylinderVolumeBuilder cvb(
340 return cvb.trackingVolume(gctx, inner);
343 tgbConfig.trackingVolumeBuilders.push_back(
344 [&](
const auto &gctx,
const auto &inner,
const auto &) {
348 cfg.doEndcapLayerMerging =
false;
352 auto lb = std::make_shared<ActsLayerBuilder>(
355 Acts::CylinderVolumeBuilder::Config cvbConfig;
356 cvbConfig.layerEnvelopeR = {5_mm, 5_mm};
357 cvbConfig.layerEnvelopeZ = 1_mm;
358 cvbConfig.trackingVolumeHelper = cylinderVolumeHelper;
359 cvbConfig.volumeName =
"ITkPixelOuter";
360 cvbConfig.layerBuilder =
lb;
361 cvbConfig.buildToRadiusZero =
false;
362 cvbConfig.checkRingLayout =
true;
363 cvbConfig.ringTolerance = 10_mm;
365 Acts::CylinderVolumeBuilder cvb(
369 return cvb.trackingVolume(gctx, inner);
374 if (buildSubdet.count(
"ITkStrip") > 0) {
375 tgbConfig.trackingVolumeBuilders.push_back(
376 [&](
const auto &gctx,
const auto &inner,
const auto &) {
383 auto lb = std::make_shared<ActsLayerBuilder>(
386 Acts::CylinderVolumeBuilder::Config cvbConfig;
387 cvbConfig.layerEnvelopeR = {5_mm, 5_mm};
388 cvbConfig.layerEnvelopeZ = 1_mm;
389 cvbConfig.trackingVolumeHelper = cylinderVolumeHelper;
390 cvbConfig.volumeName =
"ITkStrip";
391 cvbConfig.layerBuilder =
lb;
392 cvbConfig.buildToRadiusZero =
395 Acts::CylinderVolumeBuilder cvb(
399 return cvb.trackingVolume(gctx, inner);
403 bool buildSCT = buildSubdet.count(
"SCT") > 0;
404 bool buildTRT = buildSubdet.count(
"TRT") > 0;
406 if (buildSCT && buildTRT) {
408 tgbConfig.trackingVolumeBuilders.push_back(
409 [&](
const auto &gctx,
const auto &inner,
const auto &) {
412 cfg.endcapEnvelopeZ = sctECEnvelopeZ;
413 auto sct_lb = std::make_shared<ActsLayerBuilder>(
419 *cylinderVolumeHelper, inner);
422 }
else if (buildSCT) {
423 tgbConfig.trackingVolumeBuilders.push_back(
424 [&](
const auto &gctx,
const auto &inner,
const auto &) {
427 lbCfg.endcapEnvelopeZ = sctECEnvelopeZ;
428 auto lb = std::make_shared<ActsLayerBuilder>(
432 Acts::CylinderVolumeBuilder::Config cvbConfig;
433 cvbConfig.layerEnvelopeR = {5_mm, 5_mm};
434 cvbConfig.layerEnvelopeZ = 2_mm;
435 cvbConfig.trackingVolumeHelper = cylinderVolumeHelper;
436 cvbConfig.volumeName =
"SCT";
437 cvbConfig.layerBuilder =
lb;
438 cvbConfig.buildToRadiusZero =
false;
440 Acts::CylinderVolumeBuilder cvb(
444 return cvb.trackingVolume(gctx, inner);
446 }
else if (buildTRT) {
447 tgbConfig.trackingVolumeBuilders.push_back(
448 [&](
const auto &gctx,
const auto &inner,
const auto &) {
450 Acts::CylinderVolumeBuilder::Config cvbConfig;
451 cvbConfig.layerEnvelopeR = {5_mm, 5_mm};
452 cvbConfig.layerEnvelopeZ = 2_mm;
453 cvbConfig.trackingVolumeHelper = cylinderVolumeHelper;
454 cvbConfig.volumeName =
"TRT";
455 cvbConfig.layerBuilder =
lb;
456 cvbConfig.buildToRadiusZero =
false;
458 Acts::CylinderVolumeBuilder cvb(
462 return cvb.trackingVolume(gctx, inner);
467 if(buildSubdet.count(
"HGTD") > 0) {
468 tgbConfig.trackingVolumeBuilders.push_back(
469 [&](
const auto &gctx,
const auto &inner,
const auto &) {
471 Acts::CylinderVolumeBuilder::Config cvbConfig;
472 cvbConfig.layerEnvelopeR = {5_mm, 5_mm};
473 cvbConfig.layerEnvelopeZ = 1_mm;
474 cvbConfig.trackingVolumeHelper = cylinderVolumeHelper;
475 cvbConfig.volumeName =
"HGTD";
476 cvbConfig.layerBuilder =
lb;
477 cvbConfig.buildToRadiusZero =
false;
479 Acts::CylinderVolumeBuilder cvb(
483 return cvb.trackingVolume(gctx, inner);
489 tgbConfig.trackingVolumeBuilders.push_back(
490 [&](
const auto &gctx,
const auto &inner,
const auto &) {
495 }
catch (
const std::exception &e) {
496 ATH_MSG_ERROR(
"Encountered error when building Acts tracking geometry");
498 return StatusCode::FAILURE;
501 auto trackingGeometryBuilder =
502 std::make_shared<const Acts::TrackingGeometryBuilder>(
511 ATH_MSG_ERROR(
"No ACTS tracking geometry was built. Cannot proceeed");
512 return StatusCode::FAILURE;
517 ATH_MSG_INFO(
"Running extra consistency check! (this is SLOW)");
519 ATH_MSG_ERROR(
"Consistency check has failed! Geometry is not consistent");
520 return StatusCode::FAILURE;
524 ATH_MSG_INFO(
"Acts TrackingGeometry construction completed");
526 return StatusCode::SUCCESS;
532 std::vector<Acts::Vector2> localPoints;
535 std::uniform_real_distribution<> dist(0.0, 1.0);
537 std::optional<std::ofstream> os;
541 throw std::runtime_error{
"Failed to open consistency check output file"};
546 (*os) <<
"geo_id,vol_id,lay_id,sen_id,type,acts_loc0,acts_loc1,acts_inside,trk_loc0,trk_loc1,trk_inside,x,y,z,g2l_loc0,g2l_loc1,trk_x,trk_y,trk_z" << std::endl;
549 localPoints.emplace_back(dist(gen), dist(gen));
554 size_t nTotalSensors = 0;
555 std::array<size_t,3> nInconsistent{0,0,0};
556 size_t nMismatchedCenters = 0;
557 size_t nMismatchedNormals = 0;
561 auto isApprox = [](
auto&
a,
auto& b) ->
bool {
562 return ((
a - b).
array().abs() < 1e-5).all();
568 const auto* actsDetElem =
dynamic_cast<const ActsDetectorElement*
>(surface->surfacePlacement());
569 if(actsDetElem ==
nullptr) {
575 if(siDetElem ==
nullptr) {
579 const auto* regSurface =
dynamic_cast<const Acts::RegularSurface*
>(surface);
580 const auto& trkSurface = siDetElem->
surface();
581 if(regSurface ==
nullptr) {
587 Acts::Vector3 center{regSurface->center(gctx)};
589 if (
dynamic_cast<const Acts::AnnulusBounds *
>(&surface->bounds()))
594 center.head<2>() = trkCenter.head<2>();
597 if(!isApprox(trkCenter, center)) {
599 if (
auto idHelper = siDetElem->getIdHelper())
601 trkName = idHelper->show_to_string(siDetElem->identify());
604 << surface->geometryId()
605 <<
" center (" << center[0] <<
',' << center[1] <<
',' << center[2]
606 <<
") does not match Trk surface " << trkName
607 <<
" center (" << trkCenter[0] <<
',' << trkCenter[1] <<
',' << trkCenter[2] <<
')');
608 nMismatchedCenters++;
612 const auto* lineSurface =
dynamic_cast<const Acts::LineSurface*
>(surface);
613 if(lineSurface ==
nullptr) {
614 Acts::Vector3 norm{regSurface->normal(gctx, regSurface->center(gctx))};
616 if(!isApprox(trkNorm, norm)) {
618 if (
auto idHelper = siDetElem->getIdHelper())
620 trkName = idHelper->show_to_string(siDetElem->identify());
623 << surface->geometryId()
624 <<
" normal (" << norm[0] <<
',' << norm[1] <<
',' << norm[2]
625 <<
") does not match Trk surface " << trkName
626 <<
" normal (" << trkNorm[0] <<
',' << trkNorm[1] <<
',' << trkNorm[2] <<
')');
627 nMismatchedNormals++;
632 auto doPoints = [&](
unsigned int type,
const Acts::Vector2& loc) -> std::array<bool,3> {
633 Acts::Vector3 glb = surface->localToGlobal(gctx, loc, Acts::Vector3::Zero());
637 Acts::Vector2 locg2l = Acts::Vector2::Zero();
638 bool locg2lOk =
false;
639 auto locTrkRes = trkSurface.globalToLocal(glb);
641 locTrk = locTrkRes.value();
642 glbTrk = trkSurface.localToGlobal(locTrk);
644 auto locg2lRes = surface->globalToLocal(gctx, glbTrk, Acts::Vector3::Zero());
645 if (locg2lRes.ok()) {
647 locg2l = locg2lRes.value();
651 auto gId = surface->geometryId();
654 <<
"," << gId.volume()
655 <<
"," << gId.layer()
656 <<
"," << gId.sensitive()
660 <<
"," << surface->insideBounds(loc)
663 <<
"," << trkSurface.insideBounds(locTrk)
675 return {surface->insideBounds(loc) == trkSurface.insideBounds(locTrk),
676 locg2lOk ? isApprox(loc, locg2l) :
true,
677 locTrkRes ? isApprox(glb, glbTrk) :
true};
681 constexpr double envelope = 10.0 * Acts::UnitConstants::mm;
683 std::array<bool,3> allOk{
true,
true,
true};
684 if(
const auto* bounds =
dynamic_cast<const Acts::PlanarBounds*
>(&surface->bounds()); bounds) {
687 const Acts::RectangleBounds& boundingBox = bounds->boundingBox();
688 Acts::Vector2
min = boundingBox.min().array() - envelope;
689 Acts::Vector2
max = boundingBox.max().array() + envelope;
690 Acts::Vector2 diag =
max -
min;
692 for(
const auto& testPoint : localPoints) {
693 Acts::Vector2 loc =
min.array() + (testPoint.array() * diag.array());
694 auto pointOk = doPoints(0, loc);
695 for (
size_t i=0; i<pointOk.size(); ++i) {
704 else if(
const auto* bounds =
dynamic_cast<const Acts::AnnulusBounds*
>(&surface->bounds()); bounds) {
708 std::vector<Acts::Vector2> vertices = bounds->vertices(5);
709 Acts::Vector2
min{std::numeric_limits<double>::max(), std::numeric_limits<double>::max()};
710 Acts::Vector2
max{std::numeric_limits<double>::lowest(), std::numeric_limits<double>::lowest()};
711 for (
const auto& vtx : vertices) {
712 min =
min.array().min(vtx.array());
713 max =
max.array().max(vtx.array());
715 min.array() -= envelope;
716 max.array() += envelope;
717 Acts::Vector2 diag =
max -
min;
719 for(
const auto& testPoint : localPoints) {
720 Acts::Vector2 locXY =
min.array() + (testPoint.array() * diag.array());
721 Acts::Vector2 locPC =
dynamic_cast<const Acts::DiscSurface&
>(*surface).localCartesianToPolar(locXY);
723 auto pointOk = doPoints(1, locPC);
724 for (
size_t i=0; i<pointOk.size(); ++i) {
737 for (
size_t i=0; i<allOk.size(); ++i) {
745 ATH_MSG_INFO(
"Total number of sensors : " << nTotalSensors);
746 ATH_MSG_INFO(
"Number of sensors with mismatched centers : " << nMismatchedCenters);
747 ATH_MSG_INFO(
"Number of sensors with mismatched normals : " << nMismatchedNormals);
748 ATH_MSG_INFO(
"Number of sensors with inconsistent inside: " << nInconsistent[0]);
749 ATH_MSG_INFO(
"Number of sensors with inconsistent g2l : " << nInconsistent[1]);
750 ATH_MSG_INFO(
"Number of sensors with inconsistent l2g : " << nInconsistent[2]);
917 const Acts::GeometryContext &gctx,
const Acts::ILayerBuilder &sct_lb,
918 const Acts::ILayerBuilder &trt_lb,
const Acts::CylinderVolumeHelper &cvh,
919 const std::shared_ptr<const Acts::TrackingVolume> &
pixel) {
922 Acts::CylinderVolumeBuilder::Config cvbCfg;
923 Acts::CylinderVolumeBuilder cvb(
927 Acts::VolumeConfig sctNegEC =
928 cvb.analyzeContent(gctx, sct_lb.negativeLayers(gctx), {});
930 Acts::VolumeConfig sctPosEC =
931 cvb.analyzeContent(gctx, sct_lb.positiveLayers(gctx), {});
933 Acts::VolumeConfig sctBrl =
934 cvb.analyzeContent(gctx, sct_lb.centralLayers(gctx), {});
937 Acts::VolumeConfig trtNegEC =
938 cvb.analyzeContent(gctx, trt_lb.negativeLayers(gctx), {});
940 Acts::VolumeConfig trtPosEC =
941 cvb.analyzeContent(gctx, trt_lb.positiveLayers(gctx), {});
943 Acts::VolumeConfig trtBrl =
944 cvb.analyzeContent(gctx, trt_lb.centralLayers(gctx), {});
948 double absZMinEC = std::min(std::abs(trtNegEC.zMax), std::abs(trtPosEC.zMin));
949 double absZMaxEC = std::max(std::abs(trtNegEC.zMin), std::abs(trtPosEC.zMax));
951 trtNegEC.zMin = -absZMaxEC;
952 trtNegEC.zMax = -absZMinEC;
953 trtPosEC.zMin = absZMinEC;
954 trtPosEC.zMax = absZMaxEC;
956 using CVBBV = Acts::CylinderVolumeBounds::BoundValues;
959 bool isSCTSmallerInZ =
false;
961 ATH_MSG_VERBOSE(
"Shrinking SCT in R (and maybe in increase size in Z) to fit around Pixel");
962 auto pixelBounds =
dynamic_cast<const Acts::CylinderVolumeBounds *
>(
963 &
pixel->volumeBounds());
964 double sctNegECzMin = std::min(sctNegEC.zMin, -pixelBounds->get(CVBBV::eHalfLengthZ));
965 double sctPosECzMax = std::max(sctPosEC.zMax, pixelBounds->get(CVBBV::eHalfLengthZ));
967 ATH_MSG_VERBOSE(
"- SCT +-EC.rMin: " << sctNegEC.rMin <<
" -> " << pixelBounds->get(CVBBV::eMaxR));
968 ATH_MSG_VERBOSE(
"- SCT BRL.rMin: " << sctBrl.rMin <<
" -> " << pixelBounds->get(CVBBV::eMaxR));
969 ATH_MSG_VERBOSE(
"- SCT EC.zMin: " << sctNegEC.zMin <<
" -> " << sctNegECzMin);
970 ATH_MSG_VERBOSE(
"- SCT EC.zMax: " << sctPosEC.zMax <<
" -> " << sctPosECzMax);
972 sctNegEC.rMin = pixelBounds->get(CVBBV::eMaxR);
973 sctPosEC.rMin = pixelBounds->get(CVBBV::eMaxR);
974 sctBrl.rMin = pixelBounds->get(CVBBV::eMaxR);
976 isSCTSmallerInZ = sctPosEC.zMax < pixelBounds->get(CVBBV::eHalfLengthZ);
978 sctNegEC.zMin = sctNegECzMin;
979 sctPosEC.zMax = sctPosECzMax;
989 << sctNegEC.toString());
990 ATH_MSG_VERBOSE(
"- SCT::Barrel: " << sctBrl.layers.size() <<
" layers, "
991 << sctBrl.toString());
994 << sctPosEC.toString());
999 << trtNegEC.toString());
1000 ATH_MSG_VERBOSE(
"- TRT::Barrel: " << trtBrl.layers.size() <<
" layers, "
1001 << trtBrl.toString());
1004 << trtPosEC.toString());
1008 sctBrl.zMax = (sctBrl.zMax + sctPosEC.zMin) / 2.;
1009 sctBrl.zMin = -sctBrl.zMax;
1013 trtBrl.zMin = sctBrl.zMin;
1014 trtBrl.zMax = sctBrl.zMax;
1017 trtNegEC.zMin = sctNegEC.zMin;
1018 trtPosEC.zMax = sctPosEC.zMax;
1021 trtNegEC.zMax = trtBrl.zMin;
1022 sctNegEC.zMax = trtBrl.zMin;
1023 trtPosEC.zMin = trtBrl.zMax;
1024 sctPosEC.zMin = trtBrl.zMax;
1027 sctBrl.rMax = trtBrl.rMin;
1028 sctNegEC.rMax = trtNegEC.rMin;
1029 sctPosEC.rMax = trtPosEC.rMin;
1032 trtNegEC.rMax = trtBrl.rMax;
1033 trtPosEC.rMax = trtBrl.rMax;
1035 ATH_MSG_VERBOSE(
"Dimensions after synchronization between SCT and TRT");
1039 << sctNegEC.toString());
1040 ATH_MSG_VERBOSE(
"- SCT::Barrel: " << sctBrl.layers.size() <<
" layers, "
1041 << sctBrl.toString());
1044 << sctPosEC.toString());
1049 << trtNegEC.toString());
1050 ATH_MSG_VERBOSE(
"- TRT::Barrel: " << trtBrl.layers.size() <<
" layers, "
1051 << trtBrl.toString());
1054 << trtPosEC.toString());
1056 auto makeTVol = [&](
const auto &vConf,
const auto &name) {
1057 return cvh.createTrackingVolume(gctx, vConf.layers, {},
1059 vConf.rMin, vConf.rMax, vConf.zMin,
1064 auto tvSctNegEC = makeTVol(sctNegEC,
"SCT::NegativeEndcap");
1065 auto tvSctBrl = makeTVol(sctBrl,
"SCT::Barrel");
1066 auto tvSctPosEC = makeTVol(sctPosEC,
"SCT::PositiveEndcap");
1068 auto tvTrtNegEC = makeTVol(trtNegEC,
"TRT::NegativeEndcap");
1069 auto tvTrtBrl = makeTVol(trtBrl,
"TRT::Barrel");
1070 auto tvTrtPosEC = makeTVol(trtPosEC,
"TRT::PositiveEndcap");
1074 cvh.createContainerTrackingVolume(gctx, {tvSctNegEC, tvTrtNegEC});
1076 cvh.createContainerTrackingVolume(gctx, {tvSctPosEC, tvTrtPosEC});
1077 auto barrel = cvh.createContainerTrackingVolume(gctx, {tvSctBrl, tvTrtBrl});
1082 cvh.createContainerTrackingVolume(gctx, {negEC,
barrel, posEC});
1087 auto containerBounds =
dynamic_cast<const Acts::CylinderVolumeBounds *
>(
1088 &container->volumeBounds());
1089 auto pixelBounds =
dynamic_cast<const Acts::CylinderVolumeBounds *
>(
1090 &
pixel->volumeBounds());
1091 std::vector<std::shared_ptr<Acts::TrackingVolume>> noVolumes;
1093 if(!isSCTSmallerInZ) {
1095 auto posGap = cvh.createGapTrackingVolume(
1098 pixelBounds->get(CVBBV::eMinR), pixelBounds->get(CVBBV::eMaxR),
1099 pixelBounds->get(CVBBV::eHalfLengthZ),
1100 containerBounds->get(CVBBV::eHalfLengthZ),
1103 "Pixel::PositiveGap");
1104 auto negGap = cvh.createGapTrackingVolume(
1107 pixelBounds->get(CVBBV::eMinR), pixelBounds->get(CVBBV::eMaxR),
1108 -containerBounds->get(CVBBV::eHalfLengthZ),
1109 -pixelBounds->get(CVBBV::eHalfLengthZ),
1112 "Pixel::NegativeGap");
1114 auto pixelContainer =
1115 cvh.createContainerTrackingVolume(gctx, {negGap,
pixel, posGap});
1118 cvh.createContainerTrackingVolume(gctx, {pixelContainer, container});
1123 cvh.createContainerTrackingVolume(gctx, {
pixel, container});
1142 std::shared_ptr<const Acts::CylinderVolumeHelper> cvh)
const {
1146 PVConstLink beamPipeTopVolume =
p_beamPipeMgr->getTreeTop(0);
1149 beamPipeTopVolume =
p_beamPipeMgr->getTreeTop(0)->getChildVol(0)->getChildVol(0);
1152 Acts::Transform3 beamPipeTransform;
1153 beamPipeTransform.setIdentity();
1155 beamPipeTransform = Acts::Translation3(beamPipeTopVolume->getX().translation());
1157 double beamPipeRadius = 20;
1159 const GeoLogVol* beamPipeLogVolume = beamPipeTopVolume->getLogVol();
1160 const GeoTube* beamPipeTube =
nullptr;
1163 if (beamPipeLogVolume ==
nullptr) {
1165 throw std::runtime_error(
"Beam pip volume has no log volume");
1168 beamPipeTube =
dynamic_cast<const GeoTube*
>(beamPipeLogVolume->getShape());
1169 if (beamPipeTube ==
nullptr){
1171 throw std::runtime_error{
"BeamPipeLogVolume was not of type GeoTube"};
1174 for(
unsigned int i=0;i<beamPipeTopVolume->getNChildVols();i++) {
1176 if(beamPipeTopVolume->getNameOfChildVol(i) ==
"SectionC03"){
1178 PVConstLink childTopVolume = beamPipeTopVolume->getChildVol(i);
1179 const GeoLogVol* childLogVolume = childTopVolume->getLogVol();
1180 const GeoTube* childTube =
nullptr;
1182 if (childLogVolume){
1183 childTube =
dynamic_cast<const GeoTube*
>(childLogVolume->getShape());
1185 beamPipeRadius = 0.5 * (childTube->getRMax()+childTube->getRMin());
1194 ATH_MSG_VERBOSE(
"BeamPipe constructed from Database: translation (yes) - radius "
1195 << ( beamPipeTube ?
"(yes)" :
"(no)") <<
" - r = " << beamPipeRadius );
1199 Acts::CylinderVolumeBuilder::Config cfg;
1201 Acts::PassiveLayerBuilder::Config bplConfig;
1202 bplConfig.layerIdentification =
"BeamPipe";
1203 bplConfig.centralLayerRadii = {beamPipeRadius * 1_mm};
1204 bplConfig.centralLayerHalflengthZ = {3000_mm};
1205 bplConfig.centralLayerThickness = {1_mm};
1206 auto beamPipeBuilder = std::make_shared<const Acts::PassiveLayerBuilder>(
1210 cfg.trackingVolumeHelper = cvh;
1211 cfg.volumeName =
"BeamPipe";
1212 cfg.layerBuilder = beamPipeBuilder;
1213 cfg.layerEnvelopeR = {1_mm, 1_mm};
1214 cfg.buildToRadiusZero =
true;