109 Acts::LayerVector &layersOutput)
const {
110 using enum Acts::AxisDirection;
112 ACTS_VERBOSE(
"Build layers: BARREL");
113 std::vector<std::shared_ptr<const ActsDetectorElement>> elements =
116 std::map<int, std::vector<std::shared_ptr<const Acts::Surface>>> layers{};
118 for (
const auto &element : elements) {
125 if(
id.layer_disk() >= 2)
continue;
128 if(
id.layer_disk() <= 1)
continue;
131 m_cfg.elementStore->push_back(element);
134 elementLayer =
id.layer_disk();
136 layers[elementLayer].push_back(element->surface().getSharedPtr());
140 if (
m_cfg.objDebugOutput) {
142 for (
auto &[key, layerSurfaces] : layers) {
144 std::stringstream name;
148 name <<
"obj/" <<
m_cfg.mode <<
"_brl_" << std::setfill(
'0') << std::setw(2)
149 << std::to_string(key) +
"_bec"
152 std::ofstream ofs{name.str()};
153 Acts::ObjVisualization3D vis{};
154 Acts::ViewConfig vc = Acts::s_viewSensitive;
155 vc.quarterSegments = 200;
156 for (
const auto &surface : layerSurfaces) {
157 Acts::GeometryView3D::drawSurface(vis, *surface, gctx,
158 Acts::Transform3::Identity(), vc);
165 ACTS_VERBOSE(
"Found " << layers.size() <<
" barrel layers");
167 if (
logger().doPrint(Acts::Logging::VERBOSE)) {
169 for (
const auto &[key, surfaces] : layers) {
170 Acts::ProtoLayer pl(gctx, surfaces);
171 ACTS_VERBOSE(
"Layer #" << n <<
" with layerKey: (" << key <<
")");
172 ACTS_VERBOSE(
" -> at rMin / rMax: " << pl.min(AxisR) <<
" / "
174 ACTS_VERBOSE(
" -> at zMin / zMax: " << pl.min(AxisZ) <<
" / "
181 for (
const auto &[key, surfaces] : layers) {
183 std::unique_ptr<Acts::ApproachDescriptor> approachDescriptor =
nullptr;
184 std::shared_ptr<const Acts::ProtoSurfaceMaterial> materialProxy =
nullptr;
187 Acts::ProtoLayer pl(gctx, surfaces);
188 pl.envelope[AxisR] =
m_cfg.barrelEnvelopeR;
189 pl.envelope[AxisZ] =
m_cfg.barrelEnvelopeZ;
191 double layerZ = pl.medium(AxisZ,
true);
192 double layerHalfZ = 0.5 * pl.range(AxisZ);
194 Acts::Transform3 transform(Translation3(0., 0., -layerZ));
197 std::shared_ptr<Acts::CylinderSurface> innerBoundary =
198 Acts::Surface::makeShared<Acts::CylinderSurface>(
199 transform, pl.min(AxisR), layerHalfZ);
201 std::shared_ptr<Acts::CylinderSurface> outerBoundary =
202 Acts::Surface::makeShared<Acts::CylinderSurface>(
203 transform, pl.max(AxisR), layerHalfZ);
205 std::shared_ptr<Acts::CylinderSurface> centralSurface =
206 Acts::Surface::makeShared<Acts::CylinderSurface>(
207 transform, (pl.min(AxisR) + pl.max(AxisR)) / 2.,
210 size_t binsPhi =
m_cfg.barrelMaterialBins.first;
211 size_t binsZ =
m_cfg.barrelMaterialBins.second;
213 Acts::BinUtility materialBinUtil(binsPhi, -
M_PI,
M_PI, Acts::closed,
215 materialBinUtil += Acts::BinUtility(binsZ, -layerHalfZ, layerHalfZ,
216 Acts::open, AxisZ, transform);
219 std::make_shared<const Acts::ProtoSurfaceMaterial>(materialBinUtil);
221 ACTS_VERBOSE(
"[L] Layer is marked to carry support material on Surface ( "
222 "inner=0 / center=1 / outer=2 ) : "
224 ACTS_VERBOSE(
"with binning: [" << binsPhi <<
", " << binsZ <<
"]");
226 ACTS_VERBOSE(
"Created ApproachSurfaces for cylinder layer at:");
227 ACTS_VERBOSE(
" - inner: R=" << pl.min(AxisR));
228 ACTS_VERBOSE(
" - central: R=" << (pl.min(AxisR) + pl.max(AxisR)) /
230 ACTS_VERBOSE(
" - outer: R=" << pl.max(AxisR));
233 innerBoundary->assignSurfaceMaterial(materialProxy);
235 std::vector<std::shared_ptr<const Acts::Surface>> aSurfaces;
236 aSurfaces.push_back(std::move(innerBoundary));
237 aSurfaces.push_back(std::move(centralSurface));
238 aSurfaces.push_back(std::move(outerBoundary));
241 std::make_unique<Acts::GenericApproachDescriptor>(std::move(aSurfaces));
244 auto phiEqual = [
this](
const Acts::Surface &
a,
245 const Acts::Surface &b) {
246 Acts::GeometryContext gctx;
247 return m_cfg.surfaceMatcher(gctx, AxisPhi, &
a, &b);
250 auto zEqual = [
this](
const Acts::Surface &
a,
251 const Acts::Surface &b) {
252 Acts::GeometryContext gctx;
253 return m_cfg.surfaceMatcher(gctx, AxisZ, &
a, &b);
258 auto& xsurfaces = surfaces;
259 auto countKey = [&xsurfaces](
auto equal) ->
size_t {
260 std::vector<const Acts::Surface *> keySurfaces;
261 for (
const auto &surface : xsurfaces) {
263 for (
const auto* existing : keySurfaces) {
264 if (equal(*surface, *existing)) {
270 keySurfaces.push_back(surface.get());
273 return keySurfaces.size();
276 size_t nModPhi = countKey(phiEqual);
277 size_t nModZ = countKey(zEqual);
279 ACTS_VERBOSE(
"Found " << nModPhi <<
" modules in phi " << nModZ
282 std::shared_ptr<Acts::Layer> layer;
284 size_t nBinsPhi = nModPhi *
m_cfg.numberOfBinsFactor;
285 size_t nBinsZ = nModZ *
m_cfg.numberOfBinsFactor;
286 layer =
m_cfg.layerCreator->cylinderLayer(gctx, surfaces, nBinsPhi,
287 nBinsZ, pl, transform,
288 std::move(approachDescriptor));
291 size_t nBinsPhi = nModPhi *
m_cfg.numberOfBinsFactor;
292 size_t nBinsZ = nModZ *
m_cfg.numberOfBinsFactor;
298 nBinsZ = nModZ *
m_cfg.numberOfInnermostLayerBinsFactor;
301 layer =
m_cfg.layerCreator->cylinderLayer(gctx, surfaces, nBinsPhi,
302 nBinsZ, pl, transform,
303 std::move(approachDescriptor));
305 layer =
m_cfg.layerCreator->cylinderLayer(
306 gctx, surfaces, Acts::equidistant, Acts::equidistant, pl, transform,
307 std::move(approachDescriptor));
310 layersOutput.push_back(layer);
314 std::size_t numPassiveLayers =
m_cfg.passiveBarrelLayerRadii.size();
315 ACTS_DEBUG(
"Configured to build " << numPassiveLayers <<
" passive central layers.");
316 if (numPassiveLayers != 0u) {
317 for (std::size_t icl = 0; icl < numPassiveLayers; ++icl) {
318 ACTS_VERBOSE(
"- build layer " << icl
319 <<
" with radius = " <<
m_cfg.passiveBarrelLayerRadii.at(icl)
320 <<
" and halfZ = " <<
m_cfg.passiveBarrelLayerHalflengthZ.at(icl)
321 <<
" and thickness = " <<
m_cfg.passiveBarrelLayerThickness.at(icl));
324 auto cBounds = std::make_shared<const Acts::CylinderBounds>(
325 m_cfg.passiveBarrelLayerRadii.at(icl),
m_cfg.passiveBarrelLayerHalflengthZ.at(icl));
328 std::shared_ptr<Acts::Layer> layer =
329 Acts::CylinderLayer::create(Transform3::Identity(), cBounds,
nullptr,
m_cfg.passiveBarrelLayerThickness.at(icl));
331 layersOutput.push_back(layer);
337 Acts::LayerVector &layersOutput,
int type)
const
339 using enum Acts::AxisDirection;
341 ACTS_VERBOSE(
"Build layers: " << (
type < 0 ?
"NEGATIVE" :
"POSITIVE")
343 std::vector<std::shared_ptr<const ActsDetectorElement>> elements =
345 std::map<std::tuple<int, int, int>, std::vector<const Acts::Surface *>>
348 for (
const auto &element : elements) {
351 if (
id.bec() *
type <= 0) {
356 if(
id.layer_disk() >= 3)
continue;
359 if(
id.layer_disk() <= 2)
continue;
362 m_cfg.elementStore->push_back(element);
364 std::tuple<int, int, int> key{
id.bec(),
id.layer_disk(),
id.eta_module()};
366 initialLayers[key].push_back(&element->surface());
369 ACTS_VERBOSE(
"Found " << initialLayers.size() <<
" "
370 << (
type < 0 ?
"NEGATIVE" :
"POSITIVE")
371 <<
" ENDCAP inital layers");
373 std::vector<Acts::ProtoLayer> protoLayers;
374 protoLayers.reserve(initialLayers.size());
376 for (
const auto &[key, surfaces] : initialLayers) {
377 auto &pl = protoLayers.emplace_back(gctx, surfaces);
378 pl.envelope[AxisR] =
m_cfg.endcapEnvelopeR;
379 pl.envelope[AxisZ] =
m_cfg.endcapEnvelopeZ;
383 std::sort(protoLayers.begin(), protoLayers.end(),
384 [
type](
const Acts::ProtoLayer &
a,
const Acts::ProtoLayer &b) {
385 double midA = (a.min(AxisZ) + a.max(AxisZ)) / 2.0;
386 double midB = (b.min(AxisZ) + b.max(AxisZ)) / 2.0;
394 auto plPrintZ = [](
const auto &pl) -> std::string {
395 std::stringstream
ss;
396 double zMid = (pl.min(AxisZ) + pl.max(AxisZ)) / 2.0;
397 ss <<
" < " << pl.min(AxisZ) <<
" | " << zMid <<
" | "
398 << pl.max(AxisZ) <<
" > ";
401 if (
logger().doPrint(Acts::Logging::VERBOSE)) {
402 for (
const auto &pl : protoLayers) {
404 ACTS_VERBOSE(
" -> at < zMin | zMid | zMax >: " << plPrintZ(pl));
406 ACTS_VERBOSE(
" -> at rMin / rMax: " << pl.min(AxisR) <<
" / "
411 std::vector<Acts::ProtoLayer> mergedProtoLayers;
413 if (m_cfg.doEndcapLayerMerging) {
414 mergedProtoLayers.push_back(protoLayers.front());
415 std::vector<const Acts::Surface *> surfaces;
416 for (
size_t i = 1;
i < protoLayers.size();
i++) {
417 auto &pl = protoLayers[
i];
418 auto &pl_prev = mergedProtoLayers.back();
420 ACTS_VERBOSE(
"Compare: " << plPrintZ(pl_prev) <<
" and " << plPrintZ(pl));
421 bool overlap = (pl.min(AxisZ) <= pl_prev.max(AxisZ) &&
422 pl.max(AxisZ) >= pl_prev.min(AxisZ));
423 ACTS_VERBOSE(
" -> overlap? " << (overlap ?
"yes" :
"no"));
425 ACTS_VERBOSE(
" ===> merging");
427 surfaces.reserve(pl.surfaces().size() + pl_prev.surfaces().size());
428 surfaces.insert(surfaces.end(), pl.surfaces().begin(),
429 pl.surfaces().end());
430 surfaces.insert(surfaces.end(), pl_prev.surfaces().begin(),
431 pl_prev.surfaces().end());
432 mergedProtoLayers.pop_back();
434 mergedProtoLayers.emplace_back(gctx, std::move(surfaces));
435 new_pl.envelope[AxisR] = pl.envelope[AxisR];
436 new_pl.envelope[AxisZ] = pl.envelope[AxisZ];
438 mergedProtoLayers.push_back(std::move(pl));
442 ACTS_VERBOSE(
"" << mergedProtoLayers.size() <<
" "
443 << (
type < 0 ?
"NEGATIVE" :
"POSITIVE")
444 <<
" ENDCAP layers remain after merging");
446 mergedProtoLayers = protoLayers;
449 if (m_cfg.objDebugOutput) {
450 for (
size_t i = 0;
i < mergedProtoLayers.size();
i++) {
452 std::stringstream
ss;
453 ss <<
"obj/" << m_cfg.mode <<
"_" << (
type < 0 ?
"neg" :
"pos")
454 <<
"_disk_" << std::setfill(
'0') << std::setw(2) <<
i <<
".obj";
456 std::ofstream ofs{
ss.str()};
457 Acts::ObjVisualization3D vis{};
458 Acts::ViewConfig vc = Acts::s_viewSensitive;
459 vc.quarterSegments = 200;
460 for (
const auto &surface : mergedProtoLayers[i].surfaces()) {
461 Acts::GeometryView3D::drawSurface(vis, *surface, gctx,
462 Acts::Transform3::Identity(), vc);
469 std::vector<std::shared_ptr<const Surface>> ownedSurfaces;
470 for (
const auto &pl : mergedProtoLayers) {
472 std::unique_ptr<Acts::ApproachDescriptor> approachDescriptor =
nullptr;
473 std::shared_ptr<const Acts::ProtoSurfaceMaterial> materialProxy =
nullptr;
475 double layerZ = pl.medium(AxisZ);
476 double layerHalfZ = 0.5 * pl.range(AxisZ);
477 double layerThickness = pl.range(AxisZ);
479 double layerZInner = layerZ - layerHalfZ;
480 double layerZOuter = layerZ + layerHalfZ;
482 if (std::abs(layerZInner) > std::abs(layerZOuter))
485 std::vector<std::shared_ptr<const Acts::Surface>> aSurfaces;
487 Acts::Transform3 transformNominal(Translation3(0., 0., layerZ));
488 Acts::Transform3 transformInner(Translation3(0., 0., layerZInner));
489 Acts::Transform3 transformOuter(Translation3(0., 0., layerZOuter));
491 std::shared_ptr<Acts::DiscSurface> innerBoundary =
492 Acts::Surface::makeShared<Acts::DiscSurface>(
493 transformInner, pl.min(AxisR), pl.max(AxisR));
494 aSurfaces.push_back(innerBoundary);
496 std::shared_ptr<Acts::DiscSurface> nominalSurface =
497 Acts::Surface::makeShared<Acts::DiscSurface>(
498 transformNominal, pl.min(AxisR), pl.max(AxisR));
499 aSurfaces.push_back(nominalSurface);
501 std::shared_ptr<Acts::DiscSurface> outerBoundary =
502 Acts::Surface::makeShared<Acts::DiscSurface>(
503 transformOuter, pl.min(AxisR), pl.max(AxisR));
504 aSurfaces.push_back(outerBoundary);
506 if(layerThickness > 2_mm) {
507 ACTS_VERBOSE(
"Wide disc layer ("<< layerThickness <<
") => adding cylinder like approach surfaces");
508 Acts::Transform3 trf{Translation3{0, 0, layerZ}};
510 Acts::Surface::makeShared<Acts::CylinderSurface>(
511 trf, pl.min(AxisR), layerHalfZ);
512 aSurfaces.push_back(cylinderInner);
515 Acts::Surface::makeShared<Acts::CylinderSurface>(
516 trf, pl.max(AxisR), layerHalfZ);
517 aSurfaces.push_back(cylinderOuter);
521 size_t matBinsPhi = m_cfg.endcapMaterialBins.first;
522 size_t matBinsR = m_cfg.endcapMaterialBins.second;
524 Acts::BinUtility materialBinUtil(matBinsPhi, -
M_PI,
M_PI, Acts::closed,
527 Acts::BinUtility(matBinsR, pl.min(AxisR), pl.max(AxisR),
528 Acts::open, AxisR, transformNominal);
531 std::make_shared<const Acts::ProtoSurfaceMaterial>(materialBinUtil);
533 ACTS_VERBOSE(
"[L] Layer is marked to carry support material on Surface ( "
534 "inner=0 / center=1 / outer=2 ) : "
536 ACTS_VERBOSE(
"with binning: [" << matBinsPhi <<
", " << matBinsR <<
"]");
538 ACTS_VERBOSE(
"Created ApproachSurfaces for disc layer at:");
539 ACTS_VERBOSE(
" - inner: Z=" << layerZInner);
540 ACTS_VERBOSE(
" - central: Z=" << layerZ);
541 ACTS_VERBOSE(
" - outer: Z=" << layerZOuter);
544 innerBoundary->assignSurfaceMaterial(materialProxy);
548 bool isITk = m_cfg.mode == Mode::ITkPixelInner ||
549 m_cfg.mode == Mode::ITkPixelOuter ||
550 m_cfg.mode == Mode::ITkStrip;
552 std::map<int, std::set<int>> phiModuleByRing;
555 for (
const auto &srf : pl.surfaces()) {
557 srf->associatedDetectorElement());
559 auto id = elm->identityHelper();
561 if(m_cfg.mode == Mode::ITkStrip || !isITk) {
562 ring_number =
id.eta_module();
565 ring_number =
id.layer_disk();
567 phiModuleByRing[ring_number].insert(
id.phi_module());
570 size_t nModPhi = std::numeric_limits<size_t>::max();
571 for(
const auto& [ring, phiModules] : phiModuleByRing) {
572 nModPhi = std::min(nModPhi, phiModules.size());
575 size_t nModR = phiModuleByRing.size();
577 ACTS_VERBOSE(
"Identifier reports: " << nModPhi <<
" is lowest for " << nModR
580 size_t nBinsPhi = nModPhi * m_cfg.numberOfBinsFactor;
581 size_t nBinsR = nModR * m_cfg.numberOfBinsFactor;
583 ACTS_VERBOSE(
"Creating r x phi binned layer with " << nBinsR <<
" x "
584 << nBinsPhi <<
" bins");
588 std::make_unique<Acts::GenericApproachDescriptor>(aSurfaces);
591 ownedSurfaces.clear();
592 ownedSurfaces.reserve(pl.surfaces().size());
593 std::transform(pl.surfaces().begin(), pl.surfaces().end(),
594 std::back_inserter(ownedSurfaces),
595 [](
const auto &s) { return s->getSharedPtr(); });
597 auto layer = m_cfg.layerCreator->discLayer(gctx, ownedSurfaces, nBinsR,
598 nBinsPhi, pl, Transform3::Identity(),
599 std::move(approachDescriptor));
601 layersOutput.push_back(layer);