File indexing completed on 2025-08-05 08:09:35
0001
0002
0003
0004
0005
0006
0007
0008
0009 #include "Acts/Geometry/CuboidVolumeBuilder.hpp"
0010
0011 #include "Acts/Definitions/Algebra.hpp"
0012 #include "Acts/Definitions/Units.hpp"
0013 #include "Acts/Geometry/BoundarySurfaceFace.hpp"
0014 #include "Acts/Geometry/CuboidVolumeBounds.hpp"
0015 #include "Acts/Geometry/Extent.hpp"
0016 #include "Acts/Geometry/LayerArrayCreator.hpp"
0017 #include "Acts/Geometry/LayerCreator.hpp"
0018 #include "Acts/Geometry/ProtoLayer.hpp"
0019 #include "Acts/Geometry/SurfaceArrayCreator.hpp"
0020 #include "Acts/Geometry/TrackingGeometry.hpp"
0021 #include "Acts/Geometry/TrackingVolume.hpp"
0022 #include "Acts/Geometry/Volume.hpp"
0023 #include "Acts/Surfaces/PlaneSurface.hpp"
0024 #include "Acts/Surfaces/RectangleBounds.hpp"
0025 #include "Acts/Surfaces/Surface.hpp"
0026 #include "Acts/Utilities/BinUtility.hpp"
0027 #include "Acts/Utilities/BinnedArrayXD.hpp"
0028 #include "Acts/Utilities/BinningData.hpp"
0029 #include "Acts/Utilities/Logger.hpp"
0030
0031 #include <algorithm>
0032 #include <limits>
0033 #include <stdexcept>
0034 #include <type_traits>
0035
0036 std::shared_ptr<const Acts::Surface> Acts::CuboidVolumeBuilder::buildSurface(
0037 const GeometryContext& ,
0038 const CuboidVolumeBuilder::SurfaceConfig& cfg) const {
0039 std::shared_ptr<PlaneSurface> surface;
0040
0041
0042 Transform3 trafo(Transform3::Identity() * cfg.rotation);
0043 trafo.translation() = cfg.position;
0044
0045
0046 if (cfg.detElementConstructor) {
0047 surface = Surface::makeShared<PlaneSurface>(
0048 cfg.rBounds,
0049 *(cfg.detElementConstructor(trafo, cfg.rBounds, cfg.thickness)));
0050 } else {
0051 surface = Surface::makeShared<PlaneSurface>(trafo, cfg.rBounds);
0052 }
0053 surface->assignSurfaceMaterial(cfg.surMat);
0054 return surface;
0055 }
0056
0057 std::shared_ptr<const Acts::Layer> Acts::CuboidVolumeBuilder::buildLayer(
0058 const GeometryContext& gctx,
0059 Acts::CuboidVolumeBuilder::LayerConfig& cfg) const {
0060 if (cfg.surfaces.empty() && cfg.surfaceCfg.empty()) {
0061 throw std::runtime_error{
0062 "Neither surfaces nor config to build surfaces was provided. Cannot "
0063 "proceed"};
0064 }
0065
0066
0067 if (cfg.surfaces.empty()) {
0068 for (const auto& sCfg : cfg.surfaceCfg) {
0069 cfg.surfaces.push_back(buildSurface(gctx, sCfg));
0070 }
0071 }
0072
0073 Vector3 centroid{0., 0., 0.};
0074
0075 for (const auto& surface : cfg.surfaces) {
0076 centroid += surface->transform(gctx).translation();
0077 }
0078
0079 centroid /= cfg.surfaces.size();
0080
0081
0082
0083
0084 Transform3 trafo = Transform3::Identity();
0085 trafo.translation() = centroid;
0086 if (cfg.rotation) {
0087 trafo.linear() = *cfg.rotation;
0088 } else {
0089 trafo.linear() = cfg.surfaces.front()->transform(gctx).rotation();
0090 }
0091
0092 LayerCreator::Config lCfg;
0093 lCfg.surfaceArrayCreator = std::make_shared<const SurfaceArrayCreator>();
0094 LayerCreator layerCreator(lCfg);
0095 ProtoLayer pl{gctx, cfg.surfaces};
0096 pl.envelope[binX] = cfg.envelopeX;
0097 pl.envelope[binY] = cfg.envelopeY;
0098 pl.envelope[binZ] = cfg.envelopeZ;
0099 return layerCreator.planeLayer(gctx, cfg.surfaces, cfg.binsY, cfg.binsZ,
0100 BinningValue::binX, pl, trafo);
0101 }
0102
0103 std::pair<double, double> Acts::CuboidVolumeBuilder::binningRange(
0104 const GeometryContext& gctx,
0105 const Acts::CuboidVolumeBuilder::VolumeConfig& cfg) const {
0106 using namespace UnitLiterals;
0107
0108 std::pair<double, double> minMax = std::make_pair(
0109 std::numeric_limits<double>::max(), -std::numeric_limits<double>::max());
0110
0111
0112
0113
0114
0115
0116 Vector3 minVolumeBoundaries = cfg.position - 0.5 * cfg.length;
0117 Vector3 maxVolumeBoundaries = cfg.position + 0.5 * cfg.length;
0118
0119
0120
0121 for (const auto& layercfg : cfg.layerCfg) {
0122
0123 ProtoLayer pl{gctx, layercfg.surfaces};
0124 pl.envelope[binX] = layercfg.envelopeX;
0125
0126 double surfacePosMin = pl.min(binX);
0127 double surfacePosMax = pl.max(binX);
0128
0129
0130 if (surfacePosMin < minMax.first) {
0131 minMax.first = surfacePosMin;
0132 }
0133 if (surfacePosMax > minMax.second) {
0134 minMax.second = surfacePosMax;
0135 }
0136 }
0137
0138
0139 minMax.first = std::min(minMax.first, minVolumeBoundaries(binX));
0140 minMax.second = std::max(minMax.second, maxVolumeBoundaries(binX));
0141
0142 return minMax;
0143 }
0144
0145 std::shared_ptr<Acts::TrackingVolume> Acts::CuboidVolumeBuilder::buildVolume(
0146 const GeometryContext& gctx,
0147 Acts::CuboidVolumeBuilder::VolumeConfig& cfg) const {
0148
0149 Transform3 trafo(Transform3::Identity());
0150 trafo.translation() = cfg.position;
0151
0152 auto bounds = std::make_shared<CuboidVolumeBounds>(
0153 cfg.length.x() * 0.5, cfg.length.y() * 0.5, cfg.length.z() * 0.5);
0154
0155
0156 LayerVector layVec;
0157 if (cfg.layers.empty()) {
0158 cfg.layers.reserve(cfg.layerCfg.size());
0159
0160 for (auto& layerCfg : cfg.layerCfg) {
0161 cfg.layers.push_back(buildLayer(gctx, layerCfg));
0162 layVec.push_back(cfg.layers.back());
0163 }
0164 } else {
0165 for (auto& lay : cfg.layers) {
0166 layVec.push_back(lay);
0167 }
0168 }
0169
0170
0171 std::pair<double, double> minMax = binningRange(gctx, cfg);
0172 LayerArrayCreator::Config lacCnf;
0173 LayerArrayCreator layArrCreator(
0174 lacCnf, getDefaultLogger("LayerArrayCreator", Logging::INFO));
0175 std::unique_ptr<const LayerArray> layArr(
0176 layArrCreator.layerArray(gctx, layVec, minMax.first, minMax.second,
0177 BinningType::arbitrary, BinningValue::binX));
0178
0179
0180 if (cfg.trackingVolumes.empty()) {
0181 for (VolumeConfig vc : cfg.volumeCfg) {
0182 cfg.trackingVolumes.push_back(buildVolume(gctx, vc));
0183 }
0184 }
0185
0186 std::shared_ptr<TrackingVolume> trackVolume;
0187 if (layVec.empty()) {
0188
0189 trackVolume = std::make_shared<TrackingVolume>(
0190 trafo, bounds, cfg.volumeMaterial, nullptr, nullptr,
0191 cfg.trackingVolumes, cfg.name);
0192 } else {
0193
0194 trackVolume = std::make_shared<TrackingVolume>(
0195 trafo, bounds, cfg.volumeMaterial, std::move(layArr), nullptr,
0196 cfg.trackingVolumes, cfg.name);
0197 }
0198 return trackVolume;
0199 }
0200
0201 Acts::MutableTrackingVolumePtr Acts::CuboidVolumeBuilder::trackingVolume(
0202 const GeometryContext& gctx, Acts::TrackingVolumePtr ,
0203 std::shared_ptr<const VolumeBounds> ) const {
0204
0205 std::vector<std::shared_ptr<TrackingVolume>> volumes;
0206 volumes.reserve(m_cfg.volumeCfg.size());
0207 for (VolumeConfig volCfg : m_cfg.volumeCfg) {
0208 volumes.push_back(buildVolume(gctx, volCfg));
0209 }
0210
0211
0212
0213 std::sort(volumes.begin(), volumes.end(),
0214 [](const TrackingVolumePtr& lhs, const TrackingVolumePtr& rhs) {
0215 return lhs->center().x() < rhs->center().x();
0216 });
0217
0218
0219 for (unsigned int i = 0; i < volumes.size() - 1; i++) {
0220 volumes[i + 1]->glueTrackingVolume(
0221 gctx, BoundarySurfaceFace::negativeFaceYZ, volumes[i].get(),
0222 BoundarySurfaceFace::positiveFaceYZ);
0223 volumes[i]->glueTrackingVolume(gctx, BoundarySurfaceFace::positiveFaceYZ,
0224 volumes[i + 1].get(),
0225 BoundarySurfaceFace::negativeFaceYZ);
0226 }
0227
0228
0229 Transform3 trafo(Transform3::Identity());
0230 trafo.translation() = m_cfg.position;
0231
0232
0233 auto volumeBounds = std::make_shared<CuboidVolumeBounds>(
0234 m_cfg.length.x() * 0.5, m_cfg.length.y() * 0.5, m_cfg.length.z() * 0.5);
0235
0236
0237 std::vector<std::pair<TrackingVolumePtr, Vector3>> tapVec;
0238 tapVec.reserve(m_cfg.volumeCfg.size());
0239 for (auto& tVol : volumes) {
0240 tapVec.push_back(std::make_pair(tVol, tVol->center()));
0241 }
0242
0243
0244 std::vector<float> binBoundaries;
0245 binBoundaries.push_back(volumes[0]->center().x() -
0246 m_cfg.volumeCfg[0].length.x() * 0.5);
0247 for (std::size_t i = 0; i < volumes.size(); i++) {
0248 binBoundaries.push_back(volumes[i]->center().x() +
0249 m_cfg.volumeCfg[i].length.x() * 0.5);
0250 }
0251
0252
0253 BinningData binData(BinningOption::open, BinningValue::binX, binBoundaries);
0254 auto bu = std::make_unique<const BinUtility>(binData);
0255
0256
0257 std::shared_ptr<const TrackingVolumeArray> trVolArr(
0258 new BinnedArrayXD<TrackingVolumePtr>(tapVec, std::move(bu)));
0259
0260
0261 MutableTrackingVolumePtr mtvp(std::make_shared<TrackingVolume>(
0262 trafo, volumeBounds, nullptr, nullptr, trVolArr,
0263 MutableTrackingVolumeVector{}, "World"));
0264
0265 return mtvp;
0266 }