ATLAS Offline Software
Loading...
Searching...
No Matches
NswSegmentFinderAlg.cxx
Go to the documentation of this file.
1/*
2 Copyright (C) 2002-2026 CERN for the benefit of the ATLAS collaboration
3*/
4
6
12
13
15
18
20
21#include "Acts/Seeding/CombinatorialSeedSolver.hpp"
22
23#include <ranges>
24#include <format>
25
26using namespace Acts::Experimental::CombinatorialSeedSolver;
27namespace {
28 inline const MuonGMR4::StripDesign& getDesign(const MuonR4::SpacePoint& sp) {
30 const auto* prd = static_cast<const xAOD::MMCluster*>(sp.primaryMeasurement());
31 return prd->readoutElement()->stripLayer(prd->measurementHash()).design();
32 } else if (sp.type() == xAOD::UncalibMeasType::sTgcStripType) {
33 const auto* prd = static_cast<const xAOD::sTgcMeasurement*>(sp.primaryMeasurement());
34 const auto* re = prd->readoutElement();
35 switch(prd->channelType()) {
37 return re->stripDesign(prd->measurementHash());
39 return re->wireDesign(prd->measurementHash());
41 return re->padDesign(prd->measurementHash());
42 }
43 }
44 THROW_EXCEPTION("Invalid space point for design retrieval "<<sp.msSector()->idHelperSvc()->toString(sp.identify()));
45 }
46 inline double stripHalfLength(const MuonR4::SpacePoint& sp) {
47 const auto& design = getDesign(sp);
49 const auto* prd = static_cast<const xAOD::MMCluster*>(sp.primaryMeasurement());
50 return 0.5* design.stripLength(prd->channelNumber());
51 } else{
52 const auto* prd = static_cast<const xAOD::sTgcMeasurement*>(sp.primaryMeasurement());
53 if(prd->channelType() == sTgcIdHelper::Pad){
54 const auto& padDesign = static_cast<const MuonGMR4::PadDesign&>(design);
55 auto padCorners = padDesign.padCorners(prd->channelNumber());
56 return 0.5* std::abs(padCorners[0].x() - padCorners[1].x());
57 }
58 return 0.5* design.stripLength(prd->channelNumber());
59 }
60
61 return 0.;
62 }
63 inline std::string sTgcChannelType(const int chType) {
64 return chType == sTgcIdHelper::Strip ? "S" :
65 chType == sTgcIdHelper::Wire ? "W" : "P";
66 }
67}
68
69namespace MuonR4 {
70
71using namespace SegmentFit;
72constexpr unsigned minLayers{4};
73
75
77 ATH_CHECK(m_geoCtxKey.initialize());
78 ATH_CHECK(m_etaKey.initialize());
79 ATH_CHECK(m_writeSegmentKey.initialize());
80 ATH_CHECK(m_writeSegmentSeedKey.initialize());
81 ATH_CHECK(m_idHelperSvc.retrieve());
82 ATH_CHECK(m_calibTool.retrieve());
83 ATH_CHECK(m_visionTool.retrieve(DisableTool{m_visionTool.empty()}));
84 ATH_CHECK(detStore()->retrieve(m_detMgr));
85
86 if (!(m_idHelperSvc->hasMM() || m_idHelperSvc->hasSTGC())) {
87 ATH_MSG_ERROR("MM or STGC not part of initialized detector layout");
88 return StatusCode::FAILURE;
89 }
90
92 fitCfg.calibrator = m_calibTool.get();
93 fitCfg.visionTool = m_visionTool.get();
94 fitCfg.calcAlongStrip = false;
95 fitCfg.idHelperSvc = m_idHelperSvc.get();
96 fitCfg.parsToUse = {ParamDefs::x0, ParamDefs::y0, ParamDefs::theta, ParamDefs::phi};
97
98 m_lineFitter = std::make_unique<SegmentFit::SegmentLineFitter>(name(), std::move(fitCfg));
99
101 m_seedCounter = std::make_unique<SeedStatistics>();
102 }
103
104 return StatusCode::SUCCESS;
105}
106
109 UsedHitMarker_t emptyKeeper(sortedSp.size());
110 for (std::size_t l = 0; l < sortedSp.size(); ++l) {
111 emptyKeeper[l].resize(sortedSp[l].size(), 0);
112 }
113 return emptyKeeper;
114}
115
118
120 const auto& design = getDesign(sp);
121 if (!design.hasStereoAngle()) {
122 return StripOrient::X;
123 }
124 return design.stereoAngle() > 0. ? StripOrient::U : StripOrient::V;
125 } else if (sp.type() == xAOD::UncalibMeasType::sTgcStripType) {
126 const auto* prd = static_cast<const xAOD::sTgcMeasurement*>(sp.primaryMeasurement());
127 if (sp.dimension() == 2) {
128 return StripOrient::C;
129 }
130 //check if we have strip only or wire only measurements
131 return prd->channelType() == sTgcIdHelper::Strip ? StripOrient::X : StripOrient::P;
132
133 }
134 ATH_MSG_WARNING("Cannot classify orientation of "<<m_idHelperSvc->toString(sp.identify()));
136}
139 const Amg::Vector3D& beamSpotPos,
140 const Amg::Vector3D& dirEstUp,
141 const Amg::Vector3D& dirEstDn) const{
142
143 const Amg::Vector3D estPlaneArrivalUp = SeedingAux::extrapolateToPlane(beamSpotPos, dirEstUp, testHit);
144 const Amg::Vector3D estPlaneArrivalDn = SeedingAux::extrapolateToPlane(beamSpotPos, dirEstDn, testHit);
145
146 bool below{true}, above{true};
147 switch (classifyStrip(testHit)) {
148 using enum StripOrient;
149 case U:
150 case V:{
151 const double halfLength = 0.5* stripHalfLength(testHit);
153 const Amg::Vector3D leftEdge = testHit.localPosition() - halfLength * testHit.sensorDirection();
154 const Amg::Vector3D rightEdge = testHit.localPosition() + halfLength * testHit.sensorDirection();
155
157 below = estPlaneArrivalDn.y() > std::max(leftEdge.y(), rightEdge.y());
159 above = estPlaneArrivalUp.y() < std::min(leftEdge.y(), rightEdge.y());
160 break;
161 } case X:
162 case C: {
164 const double hY = testHit.localPosition().y();
165 below = estPlaneArrivalDn.y() > hY;
167 above = estPlaneArrivalUp.y() < hY;
168 break;
169 }
170 case P:{
171 break;
172 }
173 case Unknown:{
174 break;
175 }
176
177 }
178 ATH_MSG_VERBOSE("Hit " << m_idHelperSvc->toString(testHit.identify())
179 << (below || above ? " is outside the window" : " is inside the window"));
180 if(below) {
181 return HitWindow::tooLow;
182 }
183 if(above) {
184 return HitWindow::tooHigh;
185 }
186 return HitWindow::inside;
187};
188
190#define TEST_HIT_CORRIDOR(LAYER, HIT_ITER, START_LAYER) \
191{ \
192 const SpacePoint* testMe = combinatoricLayers[LAYER].get()[HIT_ITER]; \
193 if (usedHits[LAYER].get()[HIT_ITER] > m_maxUsed) { \
194 ATH_MSG_VERBOSE(__func__<<":"<<__LINE__<<" - " \
195 <<m_idHelperSvc->toString(testMe->identify()) \
196 <<" already used in good seed." ); \
197 continue; \
198 } \
199 const HitWindow inWindow = hitFromIPCorridor(*testMe, beamSpot, dirEstUp, dirEstDn); \
200 if(inWindow == HitWindow::tooHigh) { \
201 ATH_MSG_VERBOSE(__func__<<":"<<__LINE__<<" - Hit " \
202 <<m_idHelperSvc->toString(testMe->identify()) \
203 <<" is beyond the corridor. Break loop"); \
204 break; \
205 } else if (inWindow == HitWindow::tooLow) { \
206 START_LAYER = HIT_ITER + 1; \
207 ATH_MSG_VERBOSE(__func__<<":"<<__LINE__<<" - Hit " \
208 <<m_idHelperSvc->toString(testMe->identify()) \
209 <<" is still below the corridor. Update start to " \
210 <<START_LAYER); \
211 continue; \
212 } \
213}
214
216 const HitLaySpan_t& combinatoricLayers,
217 const UsedHitSpan_t& usedHits,
218 InitialSeedVec_t& seedHitsFromLayers) const {
220 seedHitsFromLayers.clear();
221 std::size_t maxSize{1};
222 for (const HitVec& hitVec : combinatoricLayers) {
223 maxSize = maxSize * hitVec.size();
224 }
225 seedHitsFromLayers.reserve(maxSize);
226
227 unsigned iterLay0{0}, iterLay1{0}, iterLay2{0}, iterLay3{0};
228 unsigned startLay1{0}, startLay2{0}, startLay3{0};
229
230 for( ; iterLay0 < combinatoricLayers[0].get().size() ; ++iterLay0){
232 if (usedHits[0].get()[iterLay0] > m_maxUsed) {
233 continue;
234 }
235 const SpacePoint* hit0 = combinatoricLayers[0].get()[iterLay0];
237 const Amg::Vector3D initSeedDir{(beamSpot - hit0->localPosition()).unit()};
238 const Amg::Vector3D dirEstUp = Amg::dirFromAngles(initSeedDir.phi(), initSeedDir.theta() - m_windowTheta);
239 const Amg::Vector3D dirEstDn = Amg::dirFromAngles(initSeedDir.phi(), initSeedDir.theta() + m_windowTheta);
240
241 ATH_MSG_VERBOSE("Reference hit: "<<m_idHelperSvc->toString(hit0->identify())
242 <<", position: "<<Amg::toString(hit0->localPosition())
243 <<", seed dir: "<<Amg::toString(initSeedDir)
244 <<", seed plane: "<<Amg::toString(SeedingAux::extrapolateToPlane(beamSpot, initSeedDir, *hit0)));
246 for( iterLay1 = startLay1; iterLay1 < combinatoricLayers[1].get().size() ; ++iterLay1){
247 TEST_HIT_CORRIDOR(1, iterLay1, startLay1);
248 for( iterLay2 = startLay2; iterLay2 < combinatoricLayers[2].get().size() ; ++iterLay2){
249 TEST_HIT_CORRIDOR(2, iterLay2, startLay2);
250 for( iterLay3 = startLay3; iterLay3 < combinatoricLayers[3].get().size(); ++iterLay3){
251 TEST_HIT_CORRIDOR(3, iterLay3, startLay3);
252 seedHitsFromLayers.emplace_back(std::array{hit0, combinatoricLayers[1].get()[iterLay1],
253 combinatoricLayers[2].get()[iterLay2],
254 combinatoricLayers[3].get()[iterLay3]});
255 }
256 }
257 }
258 }
259}
260#undef TEST_HIT_CORRIDOR
261
264 const Amg::Vector3D& direction,
265 const HitLaySpan_t& extensionLayers,
266 const UsedHitSpan_t& usedHits) const {
267
268 //the hits we need to return to extend the segment seed
269 HitVec combinatoricHits;
270
271 for (std::size_t i = 0; i < extensionLayers.size(); ++i) {
272 const HitVec& layer{extensionLayers[i].get()};
273 const Amg::Vector3D extrapPos = SeedingAux::extrapolateToPlane(startPos, direction, *layer.front());
274
275 unsigned indexOfHit = layer.size() + 1;
276 unsigned triedHit{0};
277 double minPull{std::numeric_limits<double>::max()};
278
279 // loop over the hits on the same layer
280 for (unsigned j = 0; j < layer.size(); ++j) {
281 if (usedHits[i].get().at(j) > m_maxUsed) {
282 continue;
283 }
284 auto hit = layer.at(j);
285 const double pull = std::sqrt(SeedingAux::chi2Term(extrapPos, direction, *hit));
286 ATH_MSG_VERBOSE("Trying extension with hit " << m_idHelperSvc->toString(hit->identify())<<" and pull "<<pull);
287
288 //find the hit with the minimum pull (check at least one hit after we have increasing pulls)
289 if (pull > minPull) {
290 triedHit+=1;
291 continue;
292 }
293
294 if(triedHit>1){
295 break;
296 }
297
298 indexOfHit = j;
299 minPull = pull;
300 }
301
302 // complete the seed with the extended hits
303 if (minPull < m_minPullThreshold) {
304 const auto* bestCand = layer.at(indexOfHit);
305 ATH_MSG_VERBOSE("Extension successfull - hit" << m_idHelperSvc->toString(bestCand->identify())
306 <<", pos: "<<Amg::toString(bestCand->localPosition())
307 <<", dir: "<<Amg::toString(bestCand->sensorDirection())<<" found with pull "<<minPull);
308 combinatoricHits.push_back(bestCand);
309 }
310 }
311 return combinatoricHits;
312}
313
314std::unique_ptr<SegmentSeed>
316 const AmgSymMatrix(2)& bMatrix,
317 const HoughMaximum& max,
318 const HitLaySpan_t& extensionLayers,
319 const UsedHitSpan_t& usedHits) const {
320 bool allValid = std::any_of(initialSeed.begin(), initialSeed.end(),
321 [this](const auto& hit){
322 if (hit->type() == xAOD::UncalibMeasType::MMClusterType) {
323 const auto* mmClust = static_cast<const xAOD::MMCluster*>(hit->primaryMeasurement());
324 return mmClust->stripNumbers().size() >= m_minClusSize;
325 }
326 return true;
327 });
328
329 if (!allValid) {
330 ATH_MSG_VERBOSE("Seed rejection: Not all clusters meet minimum strip size");
331 return nullptr;
332 }
333
334
335 std::array<double, 4> params = defineParameters(bMatrix, initialSeed);
336
337 const auto [segPos, direction] = seedSolution(initialSeed, params);
338
339 // check the consistency of the parameters - expected to lay in the strip's
340 // length
341 for (std::size_t i = 0; i < 4; ++i) {
342 const double halfLength = stripHalfLength(*initialSeed[i]);
343
344 if (std::abs(params[i]) > halfLength) {
345 ATH_MSG_VERBOSE("Seed Rejection: Invalid seed - outside of the strip's length "<< m_idHelperSvc->toString(initialSeed[i]->identify())
346 <<", param: "<<params[i]<<", halfLength: "<<halfLength);
347 return nullptr;
348 }
349 }
350 double tanAlpha = houghTanAlpha(direction);
351 double tanBeta = houghTanBeta(direction);
352
353 double interceptX = segPos.x();
354 double interceptY = segPos.y();
355
356 //seed quality check - we expect the tanAlpha not to be too big which would mean big deflection along the strip layers
357 if(std::abs(tanAlpha) > m_maxTanAlpha){
358 ATH_MSG_VERBOSE("Seed Rejection: Invalid seed - tanAlpha "<<tanAlpha<<" above threshold "<<m_maxTanAlpha);
359 return nullptr;
360 }
361
362
363 // extend the seed to the segment -- include hits from the other layers too
364 auto extendedHits = extendHits(segPos, direction, extensionLayers, usedHits);
365 HitVec hits{initialSeed.begin(),initialSeed.end()};
366 std::ranges::move(extendedHits, std::back_inserter(hits));
367
368 return std::make_unique<SegmentSeed>(tanBeta, interceptY, tanAlpha,
369 interceptX, hits.size(),
370 std::move(hits), max.parentBucket());
371}
372
373
374std::unique_ptr<Segment> NswSegmentFinderAlg::fitSegmentSeed(const EventContext& ctx,
375 const ActsTrk::GeometryContext& gctx,
376 const SegmentSeed* patternSeed) const{
377
378 if(patternSeed->getHitsInMax().size() < m_minSeedHits){
379 ATH_MSG_VERBOSE("Not enough hits in the SegmentSeed to fit a segment");
380 return nullptr;
381 }
382
383 ATH_MSG_VERBOSE("Fit the SegmentSeed");
384 if (msgLvl(MSG::VERBOSE)) {
385 std::stringstream hitStream{};
386 for (const auto& hit : patternSeed->getHitsInMax()) {
387 hitStream<<"**** "<< (*hit)<<std::endl;
388 }
389 ATH_MSG_VERBOSE(__func__<<"() - "<<__LINE__ <<": Uncalibrated space points for the segment fit: "<<std::endl
390 <<hitStream.str());
391 }
392
393 //Calibration of the seed spacepoints
394 CalibSpacePointVec calibratedHits = m_calibTool->calibrate(ctx, patternSeed->getHitsInMax(),
395 patternSeed->localPosition(),
396 patternSeed->localDirection(), 0.);
397
398 const Amg::Transform3D& locToGlob{patternSeed->msSector()->localToGlobalTransform(gctx)};
399
400 return m_lineFitter->fitSegment(ctx, patternSeed, patternSeed->parameters(),
401 locToGlob, std::move(calibratedHits));
402}
403
404void NswSegmentFinderAlg::processSegment(std::unique_ptr<Segment> segment,
405 const HitVec& seedHits,
406 const HitLayVec& hitLayers,
407 UsedHitMarker_t& usedHits,
408 SegmentVec_t& segments) const {
409
410 if (!segment) {
411 ATH_MSG_VERBOSE("Seed Rejection: Segment fit failed");
412
413 if (m_markHitsFromSeed && seedHits.size() > m_minSeedHits) {
414 // Mark hits from extended seed (used increment by 1)
415 markHitsAsUsed(seedHits, hitLayers, usedHits, 1, false);
416 }
417 return;
418 }
419
420 // -------- success path --------
421 ATH_MSG_DEBUG("Segment built with "
422 << segment->measurements().size()
423 << " hits, chi2/ndof: "
424 << segment->chi2() / std::max(1u,segment->nDoF()));
425
426 HitVec segMeasSP;
427 segMeasSP.reserve(segment->measurements().size());
428
429 std::ranges::transform(
430 segment->measurements(),
431 std::back_inserter(segMeasSP),
432 [](const auto& m) { return m->spacePoint(); }
433 );
434
435 // Mark segment hits as fully used (used increment by 10,
436 // hits are effectively removed)
437 markHitsAsUsed(segMeasSP, hitLayers, usedHits, 10, true);
438 segments.push_back(std::move(segment));
439
440}
441
442std::pair<NswSegmentFinderAlg::SegmentSeedVec_t, NswSegmentFinderAlg::SegmentVec_t>
444 const ActsTrk::GeometryContext &gctx,
445 const HitLayVec& hitLayers,
446 const HoughMaximum& max,
447 const Amg::Vector3D& beamSpotPos,
448 UsedHitMarker_t& usedHits) const {
449
450 //go through the layers and build seeds from the combinations of hits
451 //starting from the outermost layers with 2D measurements (excluding pads)
452 SegmentSeedVec_t seeds{};
453 SegmentVec_t segments{};
454 std::size_t layerSize = hitLayers.size();
455 double thetaWindowCut{std::cos(2*m_windowTheta)};
456
457 // lamda helper to check if the spacepoint is combined (but not pad) and unused in an already constructed seed
458 auto isUnusedCombined = [&](std::size_t layIdx, std::size_t hitIdx) -> bool {
459 const SpacePoint* sp = hitLayers[layIdx][hitIdx];
461 THROW_EXCEPTION("Space point is not of sTgc type: "<<sp->msSector()->idHelperSvc()->toString(sp->identify()));
462 }
463 const auto* prd = static_cast<const xAOD::sTgcMeasurement*>(sp->primaryMeasurement());
464 bool isCombined = sTgcChannelType(prd->channelType()) == "S" && sp->dimension() == 2;
465 bool isUnused = usedHits[layIdx].at(hitIdx) <= m_maxUsed;
466 return isCombined && isUnused;
467
468 };
469
470 // find the 2D measurements from the outermost layers - even move one layer inside
471 for(std::size_t layIdx1 = 0; layIdx1 < 2; ++layIdx1){
472 for(std::size_t layIdx2 = layerSize-1; layIdx2 >= layerSize-2; --layIdx2){
473 //in case of MM layers we stop - the layers are sorted in Z
474 if(hitLayers[layIdx1].front()->type() == xAOD::UncalibMeasType::MMClusterType ||
475 hitLayers[layIdx2].front()->type() == xAOD::UncalibMeasType::MMClusterType){
476 ATH_MSG_VERBOSE("Outermost layers are MM - stop searching for sTgc Measurements");
477 return std::make_pair(std::move(seeds), std::move(segments));
478 }
479
480 //check if we have 2D measurements on these layers that are unused (excluding the pads)
481 for(std::size_t hitIdx1 = 0; hitIdx1 < hitLayers[layIdx1].size(); ++hitIdx1) {
482 const SpacePoint* hit1 = hitLayers[layIdx1][hitIdx1];
483 if(!isUnusedCombined(layIdx1, hitIdx1)){
484 continue;
485 }
486 for(std::size_t hitIdx2 = 0; hitIdx2 < hitLayers[layIdx2].size(); ++hitIdx2) {
487 const SpacePoint* hit2 = hitLayers[layIdx2][hitIdx2];
488 if(!isUnusedCombined(layIdx2, hitIdx2)){
489 continue;
490 }
491 //test if this selection of the hits from the two layers is aligned with the beam spot
492 const Amg::Vector3D beamSpotHitDir{(beamSpotPos - hit1->localPosition()).unit()};
493 const Amg::Vector3D seedDir{(hit2->localPosition() - hit1->localPosition()).unit()};
494 const double cosAngle = beamSpotHitDir.dot(seedDir);
495 //accept the deflection of direction with a tolerance of 1 deg
496 if(std::abs(cosAngle) < thetaWindowCut){
497 continue;
498 }
499 //found 2D hits on the outermost layers - build a seed
500 HitVec seedHits{hit1, hit2};
501 //get the seed direction and position from the two 2D hits
502 const Amg::Vector3D seedPos = hit1->localPosition();
503 //express position in z=0
504 const Amg::Vector3D seedPosZ0 = seedPos + Amg::intersect<3>(seedPos, seedDir, Amg::Vector3D::UnitZ(), 0.).value_or(0.)*seedDir;
505 // extend the seed to the other layers
506 HitLaySpan_t extensionLayers{};
507 UsedHitSpan_t usedExtensionHits{};
508 extensionLayers.reserve(hitLayers.size());
509 usedExtensionHits.reserve(hitLayers.size());
510 for (std::size_t e = 0 ; e < hitLayers.size(); ++e) {
511 if (!(e == layIdx1 || e == layIdx2)){
512 extensionLayers.emplace_back(hitLayers[e]);
513 usedExtensionHits.emplace_back(usedHits[e]);
514 }
515 }
516 auto extendedHits = extendHits(seedPosZ0, seedDir, extensionLayers, usedExtensionHits);
517 std::ranges::move(extendedHits, std::back_inserter(seedHits));
518
519 //make seed
520 auto seed = std::make_unique<SegmentSeed>(houghTanBeta(seedDir), seedPosZ0.y(),
521 houghTanAlpha(seedDir), seedPosZ0.x(),
522 seedHits.size(), std::move(seedHits),
523 max.parentBucket());
524
525 //skip segment fit with less than 5 hits
526 if(seed->getHitsInMax().size() < m_minSeedHits){
527 seeds.push_back(std::move(seed));
528 continue;
529 }
530 //fit the segment seed
531 std::unique_ptr<Segment> segment = fitSegmentSeed(ctx, gctx, seed.get());
532 processSegment(std::move(segment), seed->getHitsInMax(), hitLayers, usedHits, segments);
533 seeds.push_back(std::move(seed));
534
535 }
536 }
537 }
538 }
539 return std::make_pair(std::move(seeds),std::move(segments));
540}
541
542std::pair<NswSegmentFinderAlg::SegmentSeedVec_t, NswSegmentFinderAlg::SegmentVec_t>
544 const ActsTrk::GeometryContext &gctx,
545 const HitLayVec& hitLayers,
546 const HoughMaximum& max,
547 const Amg::Vector3D& beamSpotPos,
548 UsedHitMarker_t& usedHits,
549 bool useOnlyMM) const {
550
551 //go through the layers and build seeds from the combinations of hits
552 SegmentSeedVec_t seeds{};
553 SegmentVec_t segments{};
554 std::size_t layerSize = hitLayers.size();
555
556 if(layerSize < minLayers){
557 ATH_MSG_VERBOSE("Not enough layers to build a seed");
558 return {std::move(seeds), std::move(segments)};
559 }
560
561 //lamda helper to find the first unused strip hit on the layer
562 auto unusedStripHit = [&](const HitVec& layerHits, unsigned int layIdx) -> const SpacePoint* {
563 //in case of MM only combinatorial seeding - we consider only MM strip hits for seeding
564 bool isMM = useOnlyMM ? layerHits.front()->type() == xAOD::UncalibMeasType::MMClusterType : true;
565 for (auto [idx, hit] : Acts::enumerate(layerHits)) {
566 auto spOrient = classifyStrip(*hit);
567 bool isStrip = spOrient == StripOrient::X || spOrient == StripOrient::U || spOrient == StripOrient::V;
568 bool isUnused = usedHits[layIdx].at(idx) <= m_maxUsed;
569 if (isStrip && isUnused && isMM) {
570 return hit;
571 }
572 }
573 return nullptr;
574 };
575
576 std::array<const SpacePoint*, 4> seedHits{};
577 InitialSeedVec_t preLimSeeds{};
578
579 for (std::size_t i = 0; i < layerSize - 3; ++i) {
580 seedHits[0] = unusedStripHit(hitLayers[i], i);
581 if(!seedHits[0]) {
582 continue;
583 }
584 for (std::size_t j = i + 1; j < layerSize - 2; ++j) {
585 seedHits[1] = unusedStripHit(hitLayers[j], j);
586 if(!seedHits[1]){
587 continue;
588 }
589 for (std::size_t l = layerSize - 1; l > j+1; --l) {
590 seedHits[3] = unusedStripHit(hitLayers[l], l);
591 if(!seedHits[3]){
592 continue;
593 }
594 for (std::size_t k = l-1; k > j ; --k) {
595 seedHits[2] = unusedStripHit(hitLayers[k], k);
596 if(!seedHits[2]){
597 continue;
598 }
599
600 const HitLaySpan_t layers{hitLayers[i], hitLayers[j], hitLayers[k], hitLayers[l]};
601 //skip combination with at least one too busy layer
602 bool tooBusy = std::ranges::any_of(layers,
603 [this](const auto& layer) {
604 return layer.get().size() > m_maxClustersInLayer;
605 });
606 if (tooBusy) {
607 continue; // skip this combination
608 }
609
610 AmgSymMatrix(2) bMatrix = betaMatrix(seedHits);
611
612 if (std::abs(bMatrix.determinant()) < 1.e-6) {
613 continue;
614 }
615 ATH_MSG_DEBUG("Space point positions for seed layers: \n"
616 <<(*seedHits[0]) << ",\n"
617 <<(*seedHits[1]) << ",\n"
618 <<(*seedHits[2]) << ",\n"
619 <<(*seedHits[3]));
620
621
622 UsedHitSpan_t usedHitsSpan{usedHits[i], usedHits[j], usedHits[k], usedHits[l]};
623 // each layer may have more than one hit - take the hit combinations
624 constructPreliminarySeeds(beamSpotPos, layers, usedHitsSpan, preLimSeeds);
625
626 //the layers not participated in the seed build - gonna be used for the extension
627 HitLaySpan_t extensionLayers{};
628 UsedHitSpan_t usedExtensionHits{};
629 usedExtensionHits.reserve(hitLayers.size());
630 extensionLayers.reserve(hitLayers.size());
631 for (std::size_t e = 0 ; e < hitLayers.size(); ++e) {
632 if (!(e == i || e == j || e == k || e == l)){
633 extensionLayers.emplace_back(hitLayers[e]);
634 usedExtensionHits.emplace_back(usedHits[e]);
635 }
636 }
637 // we have made sure to have hits from all the four layers -
638 // start by 4 hits for the seed and try to build the extended seed for the combinatorics found
639 for (auto &combinatoricHits : preLimSeeds) {
640 auto seed = constructCombinatorialSeed(combinatoricHits, bMatrix, max, extensionLayers, usedExtensionHits);
641 if(!seed){
642 continue;
643 }
644 if (seed->getHitsInMax().size() < m_minSeedHits) {
645 seeds.push_back(std::move(seed));
646 continue;
647 }
648 std::unique_ptr<Segment> segment = fitSegmentSeed(ctx, gctx, seed.get());
649 processSegment(std::move(segment), seed->getHitsInMax(), hitLayers, usedHits, segments);
650 seeds.push_back(std::move(seed));
651
652 }
653 }
654 }
655 }
656 }
657 return std::make_pair(std::move(seeds),std::move(segments));
658}
659
660std::pair<NswSegmentFinderAlg::SegmentSeedVec_t, NswSegmentFinderAlg::SegmentVec_t>
662 const ActsTrk::GeometryContext &gctx,
663 const EventContext& ctx) const {
664 // first sort the hits per layer from the maximum
665 SpacePointPerLayerSplitter hitLayers{max.getHitsInMax()};
666
667 const HitLayVec& stripHitsLayers{hitLayers.stripHits()};
668 const std::size_t layerSize = stripHitsLayers.size();
669
670 //seeds and segments containers
671 SegmentSeedVec_t seeds{};
672 SegmentVec_t segments{};
673
674 const Amg::Transform3D globToLocal = max.msSector()->globalToLocalTransform(gctx);
675 //counters for the number of seeds, extented seeds and segments
676
677 if (layerSize < minLayers) {
678 ATH_MSG_VERBOSE("Not enough layers to build a seed");
679 return std::make_pair(std::move(seeds),std::move(segments));
680 }
681
682 if (m_visionTool.isEnabled()) {
684 constexpr double legX{0.2};
685 double legY{0.8};
686 for (const SpacePoint* sp : max.getHitsInMax()) {
687 const xAOD::MuonSimHit* simHit = getTruthMatchedHit(*sp->primaryMeasurement());
688 if (!simHit) {
689 continue;
690 }
691
692 const MuonGMR4::MuonReadoutElement* reEle = m_detMgr->getReadoutElement(simHit->identify());
693 const Amg::Transform3D toChamb = reEle->msSector()->globalToLocalTransform(gctx) *
694 reEle->localToGlobalTransform(gctx, sp->identify());
695
696 const Amg::Vector3D hitPos = toChamb * xAOD::toEigen(simHit->localPosition());
697 const Amg::Vector3D hitDir = toChamb.linear() * xAOD::toEigen(simHit->localDirection());
698 const double pull = std::sqrt(SeedingAux::chi2Term(hitPos, hitDir, *sp));
699
701 const auto* mmClust = static_cast<const xAOD::MMCluster*>(sp->primaryMeasurement());
702 const MuonGMR4::MmReadoutElement* mmEle = mmClust->readoutElement();
703 const auto& design = mmEle->stripLayer(mmClust->measurementHash()).design();
704 std::string stereoDesign{!design.hasStereoAngle() ? "X" : design.stereoAngle() >0 ? "U": "V"};
705 primitives.push_back(MuonValR4::drawLabel(std::format("ml: {:1d}, gap: {:1d}, {:}, pull: {:.2f}",
706 mmEle->multilayer(), mmClust->gasGap(),
707 stereoDesign, pull), legX, legY, 14));
708 } else if(sp->type() == xAOD::UncalibMeasType::sTgcStripType) {
709 const auto* sTgcMeas = static_cast<const xAOD::sTgcMeasurement*>(sp->primaryMeasurement());
710 std::string channelString = sp->secondaryMeasurement() == nullptr ?
711 sTgcChannelType(sTgcMeas->channelType()) :
712 std::format("{:}/{:}", sTgcChannelType(sTgcMeas->channelType()),
713 sTgcChannelType(static_cast<const xAOD::sTgcMeasurement*>(sp->secondaryMeasurement())->channelType()));
714 primitives.push_back(MuonValR4::drawLabel(std::format("ml: {:1d}, gap: {:1d}, type: {:}, pull: {:.2f}",
715 sTgcMeas->readoutElement()->multilayer(), sTgcMeas->gasGap(),
716 channelString, pull), legX, legY, 14));
717 }
718 legY-=0.05;
719 }
720 m_visionTool->visualizeBucket(ctx, *max.parentBucket(),
721 "truth", std::move(primitives));
722 }
723
724 //dump spacepoints associated with truth sim hits to an obj file
725 if(m_dumpObj){
726 Acts::ObjVisualization3D visualHelper{};
727 for (const SpacePoint* sp : max.getHitsInMax()) {
728 const xAOD::MuonSimHit* simHit = getTruthMatchedHit(*sp->primaryMeasurement());
729 if (!simHit) {
730 continue;
731 }
732 MuonValR4::drawSpacePoint(gctx, *sp, visualHelper);
733 }
734 visualHelper.write(std::format("Event_{:}_{:}_spacepoints_truth.obj", ctx.eventID().event_number(), max.getHitsInMax().front()->chamber()->identString()));
735 }
736
737
738 UsedHitMarker_t allUsedHits = emptyBookKeeper(stripHitsLayers);
739 std::size_t nSeeds{0}, nExtSeeds{0}, nSegments{0}; //for the seed statistics
740
741 // helper lamda function to increase counters and fill the seeds and segments we want to return after we construct them
742 // the extended seeds are returned even if they did not make it to a segment and the segments only if successfully fitted
743 auto processSeedsAndSegments = [&](std::pair<SegmentSeedVec_t, SegmentVec_t>&& seedSegmentPairs, std::string_view source) {
744 auto& [returnSeeds, returnSegments] = seedSegmentPairs;
745 ATH_MSG_DEBUG("From " << source << ": built " << returnSeeds.size() << " seeds and " << returnSegments.size() << " segments.");
746 for(auto& seed : returnSeeds) {
747 ++nSeeds;
748 Acts::ObjVisualization3D visualHelper{};
749 if(seed->getHitsInMax().size() < m_minSeedHits){
750 ATH_MSG_VERBOSE("Seed with "<< seed->getHitsInMax().size() <<" hits rejected");
751 for(const auto& hit : seed->getHitsInMax()){
752 ATH_MSG_VERBOSE("Hit "<<m_idHelperSvc->toString(hit->identify())<<", "
753 <<Amg::toString(hit->localPosition())<<", dir: "
754 <<Amg::toString(hit->sensorDirection()));
755 if(m_dumpObj){
756 MuonValR4::drawSpacePoint(gctx, *hit, visualHelper);
757 }
758
759 }
760 if(m_dumpObj){
761 visualHelper.write(std::format("Event_{:}_{:}_notExtendedSeed.obj", ctx.eventID().event_number(), seed->getHitsInMax().front()->chamber()->identString()));
762 }
763 continue;
764 }
765 ++nExtSeeds;
766 seeds.push_back(std::move(seed));
767 }
768 //move all the segments to the output container
769 std::ranges::move(returnSegments, std::back_inserter(segments));
770 nSegments += returnSegments.size();
771 };
772
773 //Start from outermost sTgc layers with combined 2D measurements
774 ATH_MSG_VERBOSE("Start building seed from sTgc outermost layers");
775 processSeedsAndSegments(buildSegmentsFromSTGC(ctx, gctx, stripHitsLayers, max, globToLocal.translation(), allUsedHits), "sTgc segment seeds");
776
777 //continue with the combinatorial seeding for the strip measurements
779
780 ATH_MSG_VERBOSE("Start building combinatoric seeds only from Micromegas");
781 processSeedsAndSegments(buildSegmentsFromMM(ctx, gctx, stripHitsLayers, max, globToLocal.translation(), allUsedHits, true), "MM combinatoric segment seeds");
782
783 }else{
784
785 ATH_MSG_VERBOSE("Start building combinatoric seeds from Micromegas and sTgc hits");
786 processSeedsAndSegments(buildSegmentsFromMM(ctx, gctx, stripHitsLayers, max, globToLocal.translation(), allUsedHits, true), "MM combinatoric segment seeds");
787 processSeedsAndSegments(buildSegmentsFromMM(ctx, gctx, stripHitsLayers, max, globToLocal.translation(), allUsedHits, false), "MM and STGC combinatoric segment seeds");
788
789 }
790
791 if(m_seedCounter) {
792 m_seedCounter->addToStat(max.msSector(), nSeeds, nExtSeeds, nSegments);
793 }
794
795 return std::make_pair(std::move(seeds),std::move(segments));
796}
797
799 const HitLayVec& allSortHits,
800 UsedHitMarker_t& usedHitMarker,
801 unsigned incr,
802 bool markNeighborHits) const {
803
804 SpacePointPerLayerSorter layerSorter{};
805
806 for(const auto& sp : spacePoints){
807 // Proection against the auxiliary measurement
808 if(!sp){
809 continue;
810 }
811
812 unsigned measLayer = layerSorter.sectorLayerNum(*sp);
813
814 Amg::Vector2D spPosX{Amg::Vector2D::Zero()};
816 switch (sp->primaryMeasurement()->numDimensions()) {
817 case 1:
818 spPosX[Amg::x] = sp->primaryMeasurement()->localPosition<1>().x();
819 break;
820 case 2:
821 spPosX = xAOD::toEigen(sp->primaryMeasurement()->localPosition<2>());
822 break;
823 default:
824 THROW_EXCEPTION("Unsupported dimension");
825 }
826
827 for (std::size_t lIdx = 0; lIdx < allSortHits.size(); ++lIdx) {
828 const HitVec& hVec{allSortHits[lIdx]};
829 //check if they are not in the same layer
830 unsigned hitLayer = layerSorter.sectorLayerNum(*hVec.front());
831 if(hitLayer != measLayer) {
832 ATH_MSG_VERBOSE("Not in the same layer since measLayer = "<< measLayer << " and "<<hitLayer);
833 continue;
834 }
835 for (std::size_t hIdx = 0 ; hIdx < hVec.size(); ++hIdx) {
836 //check the dY between the measurement and the hits
837 auto testHit = hVec[hIdx];
838 if (testHit == sp) {
839 usedHitMarker[lIdx][hIdx] += incr;
840 if(!markNeighborHits){
841 break;
842 }
843 } else if (markNeighborHits) {
844 Amg::Vector2D testPosX{Amg::Vector2D::Zero()};
846 switch (testHit->primaryMeasurement()->numDimensions()) {
847 case 1:
848 testPosX[Amg::x] = testHit->primaryMeasurement()->localPosition<1>().x();
849 break;
850 case 2:
851 testPosX = xAOD::toEigen(testHit->primaryMeasurement()->localPosition<2>());
852 break;
853 default:
854 THROW_EXCEPTION("Unsupported dimension");
855 }
856 //if the hit not found let's see if it is too close to the segment's measurement
857 double deltaX = (testPosX - spPosX).mag();
858 if(deltaX < m_maxdYWindow){
859 usedHitMarker[lIdx][hIdx] += incr;
860 }
861 }
862 }
863 }
864 }
865}
866
867StatusCode NswSegmentFinderAlg::execute(const EventContext &ctx) const {
868 // read the inputs
869 const EtaHoughMaxContainer *maxima{nullptr};
870 ATH_CHECK(SG::get( maxima, m_etaKey, ctx));
871
872 const ActsTrk::GeometryContext *gctx{nullptr};
873 ATH_CHECK(SG::get(gctx, m_geoCtxKey, ctx));
874
875 // prepare our output collection
876 SG::WriteHandle writeSegments{m_writeSegmentKey, ctx};
877 ATH_CHECK(writeSegments.record(std::make_unique<SegmentContainer>()));
878
879 SG::WriteHandle writeSegmentSeeds{m_writeSegmentSeedKey, ctx};
880 ATH_CHECK(writeSegmentSeeds.record(std::make_unique<SegmentSeedContainer>()));
881
882 // we use the information from the previous eta-hough transform
883 // to get the combined hits that belong in the same maxima
884 for (const HoughMaximum *max : *maxima) {
885
886 auto [seeds, segments] = findSegmentsFromMaximum(*max, *gctx, ctx);
887
888 if (msgLvl(MSG::VERBOSE)) {
889 ATH_MSG_VERBOSE("Hits from Hough maximum");
890 for(const auto& hitMax : max->getHitsInMax()){
891 ATH_MSG_VERBOSE("Hit "<<m_idHelperSvc->toString(hitMax->identify())<<", "
892 <<Amg::toString(hitMax->localPosition())<<", dir: "
893 <<Amg::toString(hitMax->sensorDirection()));
894 }
895 }
896
897 for(auto& seed: seeds){
898
899 if (msgLvl(MSG::VERBOSE)){
900 std::stringstream sstr{};
901 sstr<<"Seed tanBeta = "<<seed->tanBeta()<<", y0 = "<<seed->interceptY()
902 <<", tanAlpha = "<<seed->tanAlpha()<<", x0 = "<<seed->interceptX()<<", hits in the seed "
903 <<seed->getHitsInMax().size()<<std::endl;
904
905 for(const auto& hit : seed->getHitsInMax()){
906 sstr<<" *** Hit "<<m_idHelperSvc->toString(hit->identify())<<", "
907 << Amg::toString(hit->localPosition())<<", dir: "<<Amg::toString(hit->sensorDirection())<<std::endl;
908 }
909 ATH_MSG_VERBOSE(sstr.str());
910 }
911 if (m_visionTool.isEnabled()) {
912 m_visionTool->visualizeSeed(ctx, *seed, "#phi-combinatorialSeed");
913 }
914
915 writeSegmentSeeds->push_back(std::move(seed));
916
917 }
918
919 for (auto &seg : segments) {
920 const Parameters pars = localSegmentPars(*gctx, *seg);
921
922 ATH_MSG_VERBOSE("Segment parameters : "<<toString(pars));
923
924 if (m_visionTool.isEnabled()) {
925 m_visionTool->visualizeSegment(ctx, *seg, "#phi-segment");
926 }
927
928 if(m_dumpObj){
929 Acts::ObjVisualization3D visualHelper{};
930 MuonValR4::drawSegmentMeasurements(*gctx, *seg, visualHelper);
931 MuonValR4::drawSegmentLine(*gctx, *seg, visualHelper);
932 visualHelper.write(std::format("Event_{:}_segment_{:}.obj", ctx.eventID().event_number(), seg->msSector()->identString()));
933 }
934
935 writeSegments->push_back(std::move(seg));
936
937 }
938 }
939
940 return StatusCode::SUCCESS;
941}
942
944 if(m_seedCounter) {
945 m_seedCounter->printTableSeedStats(msgStream());
946 }
947 return StatusCode::SUCCESS;
948}
949
950void NswSegmentFinderAlg::SeedStatistics::addToStat(const MuonGMR4::SpectrometerSector* msSector, unsigned seeds, unsigned extSeeds, unsigned segments){
951 std::unique_lock guard{m_mutex};
952 SectorField key{};
953 key.chIdx = msSector->chamberIndex();
954 key.phi = msSector->stationPhi();
955 key.eta = msSector->chambers().front()->stationEta();
956
957 auto &entry = m_seedStat[key];
958 entry.nSeeds += seeds;
959 entry.nExtSeeds += extSeeds;
960 entry.nSegments += segments;
961}
962
964
965
966 std::stringstream sstr{};
967 sstr<<"Seed statistics per sector:"<<std::endl;
968 sstr<<"-----------------------------------------------------"<<std::endl;
969 sstr<<"| Chamber | Phi | Eta | Seeds | ExtSeeds | Segments |"<<std::endl;
970 sstr<<"-----------------------------------------------------"<<std::endl;
971
972 using namespace Muon::MuonStationIndex;
973
974 for (const auto& [sector, stats] : m_seedStat) {
975 sstr << "| " << std::setw(3) << chName(sector.chIdx)
976 << " | " << std::setw(2) << static_cast<unsigned>(sector.phi)
977 << " | " << std::setw(3) << static_cast<int>(sector.eta)
978 << " | " << std::setw(7) << stats.nSeeds
979 << " | " << std::setw(8) << stats.nExtSeeds
980 << " | " << std::setw(8) << stats.nSegments
981 << " |"<<std::endl;
982 }
983
984 sstr<<"------------------------------------------------------------"<<std::endl;
985 msg<<MSG::ALWAYS<<"\n"<<sstr.str()<<endmsg;
986 }
987
988
989} // namespace MuonR4
const boost::regex re(r_e)
Scalar mag() const
mag method
#define endmsg
#define ATH_CHECK
Evaluate an expression and check for errors.
#define ATH_MSG_ERROR(x)
#define ATH_MSG_VERBOSE(x)
#define ATH_MSG_WARNING(x)
#define ATH_MSG_DEBUG(x)
#define AmgSymMatrix(dim)
ATLAS-specific HepMC functions.
static Double_t sp
#define TEST_HIT_CORRIDOR(LAYER, HIT_ITER, START_LAYER)
Macro to check whether a hit is compatible with the hit corridor.
#define x
#define max(a, b)
Definition cfImp.cxx:41
const ServiceHandle< StoreGateSvc > & detStore() const
bool msgLvl(const MSG::Level lvl) const
const StripLayer & stripLayer(const Identifier &measId) const
int multilayer() const
Returns the multi layer of the element [1-2].
MuonReadoutElement is an abstract class representing the geometry of a muon detector.
const Amg::Transform3D & localToGlobalTransform(const ActsTrk::GeometryContext &ctx) const
Returns the transformation from the local coordinate system of the readout element into the global AT...
const SpectrometerSector * msSector() const
Returns the pointer to the envelope volume enclosing all chambers in the sector.
A spectrometer sector forms the envelope of all chambers that are placed in the same MS sector & laye...
const Amg::Transform3D & localToGlobalTransform(const ActsTrk::GeometryContext &gctx) const
Returns the local -> global tarnsformation from the sector.
Amg::Transform3D globalToLocalTransform(const ActsTrk::GeometryContext &gctx) const
Returns the global -> local transformation from the ATLAS global.
const ChamberSet & chambers() const
Returns the associated chambers with this sector.
int stationPhi() const
: Returns the station phi of the sector
Muon::MuonStationIndex::ChIndex chamberIndex() const
Returns the chamber index scheme.
const StripDesign & design(bool phiView=false) const
Returns the underlying strip design.
Data class to represent an eta maximum in hough space.
std::vector< CalibSpacePointPtr > CalibSpacePointVec
void addToStat(const MuonGMR4::SpectrometerSector *msSector, unsigned int nSeeds, unsigned int nExtSeeds, unsigned int nSegments)
const MuonGMR4::MuonDetectorManager * m_detMgr
UnsignedIntegerProperty m_maxUsed
std::pair< SegmentSeedVec_t, SegmentVec_t > findSegmentsFromMaximum(const HoughMaximum &max, const ActsTrk::GeometryContext &gctx, const EventContext &ctx) const
Find seed and segment from an eta hough maximum.
std::unique_ptr< SegmentSeed > constructCombinatorialSeed(const InitialSeed_t &initialSeed, const AmgSymMatrix(2)&bMatrix, const HoughMaximum &max, const HitLaySpan_t &extensionLayers, const UsedHitSpan_t &usedHits) const
Construct a combinatorial seed from the initial 4-layer seed hits.
std::unique_ptr< SegmentFit::SegmentLineFitter > m_lineFitter
std::pair< SegmentSeedVec_t, SegmentVec_t > buildSegmentsFromMM(const EventContext &ctx, const ActsTrk::GeometryContext &gctx, const HitLayVec &hitLayers, const HoughMaximum &max, const Amg::Vector3D &beamSpotPos, UsedHitMarker_t &usedHits, bool useOnlyMM) const
Build the final segment seed from strip like measurements using the combinatorial seeding for MicroMe...
UnsignedIntegerProperty m_maxClustersInLayer
virtual StatusCode initialize() override
virtual StatusCode execute(const EventContext &ctx) const override
std::pair< SegmentSeedVec_t, SegmentVec_t > buildSegmentsFromSTGC(const EventContext &ctx, const ActsTrk::GeometryContext &gctx, const HitLayVec &hitLayers, const HoughMaximum &max, const Amg::Vector3D &beamSpotPos, UsedHitMarker_t &usedHits) const
Build the segment for a seed from STGC 2D measurement layers directly and then attempt to append hits...
HitWindow
To fastly check whether a hit is roughly compatible with a muon trajectory a narrow corridor is opene...
@ inside
The hit is below the predefined corridor.
@ tooHigh
The hit is inside the defined window and hence an initial candidate.
std::unique_ptr< Segment > fitSegmentSeed(const EventContext &ctx, const ActsTrk::GeometryContext &gctx, const SegmentSeed *patternSeed) const
Fit the segment seeds.
ToolHandle< MuonValR4::IPatternVisualizationTool > m_visionTool
Pattern visualization tool.
std::vector< std::reference_wrapper< const HitVec > > HitLaySpan_t
Abbrivation of the space comprising multiple hit vectors without copy.
std::array< const SpacePoint *, 4 > InitialSeed_t
Abbrivation of the initial seed.
SG::WriteHandleKey< SegmentSeedContainer > m_writeSegmentSeedKey
HitWindow hitFromIPCorridor(const SpacePoint &testHit, const Amg::Vector3D &beamSpotPos, const Amg::Vector3D &dirEstUp, const Amg::Vector3D &dirEstDn) const
The hit is above the predefined corridor.
std::vector< std::unique_ptr< SegmentSeed > > SegmentSeedVec_t
Abbrivation of the seed vector.
void constructPreliminarySeeds(const Amg::Vector3D &beamSpot, const HitLaySpan_t &combinatoricLayers, const UsedHitSpan_t &usedHits, InitialSeedVec_t &outVec) const
Construct a set of prelimnary seeds from the selected combinatoric layers.
void processSegment(std::unique_ptr< Segment > segment, const HitVec &seedHits, const HitLayVec &hitLayers, UsedHitMarker_t &usedHits, SegmentVec_t &segments) const
Process the segment and mark the hits if it is successfully built or not by differently mark the hits...
void markHitsAsUsed(const HitVec &spacePoints, const HitLayVec &allSortHits, UsedHitMarker_t &usedHitMarker, unsigned int increase, bool markNeighborHits) const
Hits that are used in a good seed/segment built should be flagged as used and not contribute to other...
UsedHitMarker_t emptyBookKeeper(const HitLayVec &sortedSp) const
Constructs an empty HitMarker from the split space points.
virtual StatusCode finalize() override
ServiceHandle< Muon::IMuonIdHelperSvc > m_idHelperSvc
SpacePointPerLayerSplitter::HitLayVec HitLayVec
SpacePointPerLayerSplitter::HitVec HitVec
std::vector< InitialSeed_t > InitialSeedVec_t
Vector of initial seeds.
StripOrient classifyStrip(const SpacePoint &spacePoint) const
Determines the orientation of the strip space point.
StripOrient
Enumeration to classify the orientation of a NSW strip.
@ X
Stereo strips with negative angle.
@ V
Stereo strips with positive angle.
@ Unknown
Combined 2D space point (sTGC wire + strip / sTgc pad)
std::vector< std::unique_ptr< Segment > > SegmentVec_t
Abbrivation of the final segment vector.
SG::WriteHandleKey< SegmentContainer > m_writeSegmentKey
SG::ReadHandleKey< ActsTrk::GeometryContext > m_geoCtxKey
std::vector< std::vector< unsigned int > > UsedHitMarker_t
Abbrivation of the container book keeping whether a hit is used or not.
ToolHandle< ISpacePointCalibrator > m_calibTool
HitVec extendHits(const Amg::Vector3D &startPos, const Amg::Vector3D &direction, const HitLaySpan_t &extensionLayers, const UsedHitSpan_t &usedHits) const
Extend the seed with the hits from the other layers.
UnsignedIntegerProperty m_minSeedHits
SG::ReadHandleKey< EtaHoughMaxContainer > m_etaKey
std::vector< std::reference_wrapper< std::vector< unsigned int > > > UsedHitSpan_t
Abbrivation of the container to pass a subset of markers wtihout copy.
Representation of a segment seed (a fully processed hough maximum) produced by the hough transform.
Definition SegmentSeed.h:14
const std::vector< HitType > & getHitsInMax() const
Returns the list of assigned hits.
const Parameters & parameters() const
Returns the parameter array.
Amg::Vector3D localDirection() const
Returns the direction of the seed in the sector frame.
const MuonGMR4::SpectrometerSector * msSector() const
Returns the associated chamber.
Amg::Vector3D localPosition() const
Returns the position of the seed in the sector frame.
The SpacePointPerLayerSorter sort two given space points by their layer Identifier.
unsigned int sectorLayerNum(const SpacePoint &sp) const
method returning the logic layer number
The SpacePointPerLayerSplitter takes a set of spacepoints already sorted by layer Identifier (see Muo...
const HitLayVec & stripHits() const
Returns the sorted strip hits.
The muon space point is the combination of two uncalibrated measurements one of them measures the eta...
const Identifier & identify() const
: Identifier of the primary measurement
StatusCode record(std::unique_ptr< T > data)
Record a const object to the store.
ConstVectorMap< 3 > localDirection() const
Returns the local direction of the traversing particle.
Identifier identify() const
Returns the global ATLAS identifier of the SimHit.
ConstVectorMap< 3 > localPosition() const
Returns the local postion of the traversing particle.
T * get(TKey *tobj)
get a TObject* from a TKey* (why can't a TObject be a TKey?)
Definition hcg.cxx:130
std::optional< double > intersect(const AmgVector(N)&posA, const AmgVector(N)&dirA, const AmgVector(N)&posB, const AmgVector(N)&dirB)
Calculates the point B' along the line B that's closest to a second line A.
std::string toString(const Translation3D &translation, int precision=4)
GeoPrimitvesToStringConverter.
Amg::Vector3D dirFromAngles(const double phi, const double theta)
Constructs a direction vector from the azimuthal & polar angles.
Eigen::Affine3d Transform3D
Eigen::Matrix< double, 2, 1 > Vector2D
Eigen::Matrix< double, 3, 1 > Vector3D
Parameters localSegmentPars(const xAOD::MuonSegment &seg)
Returns the localSegPars decoration from a xAODMuon::Segment.
Acts::Experimental::CompositeSpacePointLineFitter::ParamVec_t Parameters
std::string toString(const Parameters &pars)
Dumps the parameters into a string with labels in front of each number.
This header ties the generic definitions in this package.
ISpacePointCalibrator::CalibSpacePointVec CalibSpacePointVec
const xAOD::MuonSimHit * getTruthMatchedHit(const xAOD::UncalibratedMeasurement &prdHit)
Returns the MuonSimHit, if there's any, matched to the uncalibrated muon measurement.
double houghTanBeta(const Amg::Vector3D &v)
Returns the hough tanBeta [y] / [z].
DataVector< HoughMaximum > EtaHoughMaxContainer
constexpr unsigned minLayers
SpacePointPerLayerSplitter::HitVec HitVec
double houghTanAlpha(const Amg::Vector3D &v)
: Returns the hough tanAlpha [x] / [z]
std::unique_ptr< TLatex > drawLabel(const std::string &text, const double xPos, const double yPos, const double textSize=18, const bool useNDC=true, const int color=kBlack)
Create a TLatex label,.
void drawSpacePoint(const ActsTrk::GeometryContext &gctx, const MuonR4::SpacePoint &spacePoint, Acts::ObjVisualization3D &visualHelper, const Acts::ViewConfig &viewConfig=Acts::s_viewSensitive)
Draw an uncalibrated space point inside the obj file.
void drawSegmentMeasurements(const ActsTrk::GeometryContext &gctx, const xAOD::MuonSegment &segment, Acts::ObjVisualization3D &visualHelper, const Acts::ViewConfig &viewConfig=Acts::s_viewSensitive)
Draw all uncalibrated measurements associated to the segment.
void drawSegmentLine(const ActsTrk::GeometryContext &gctx, const xAOD::MuonSegment &segment, Acts::ObjVisualization3D &visualHelper, const Acts::ViewConfig &viewConfig=Acts::s_viewLine, const double standardLength=1.*Gaudi::Units::m)
Draw a segment line inside the obj file.
const std::string & chName(ChIndex index)
convert ChIndex into a string
const T * get(const ReadCondHandleKey< T > &key, const EventContext &ctx)
Convenience function to retrieve an object given a ReadCondHandleKey.
MuonSimHit_v1 MuonSimHit
Defined the version of the MuonSimHit.
Definition MuonSimHit.h:12
MMCluster_v1 MMCluster
sTgcMeasurement_v1 sTgcMeasurement
const ISpacePointCalibrator * calibrator
Pointer to the calibrator.
const Muon::IMuonIdHelperSvc * idHelperSvc
Pointer to the idHelperSvc.
const MuonValR4::IPatternVisualizationTool * visionTool
Pointer to the visualization tool.
#define THROW_EXCEPTION(MESSAGE)
Definition throwExcept.h:10