22#include "Acts/Seeding/CombinatorialSeedSolver.hpp"
27using namespace Acts::Experimental::CombinatorialSeedSolver;
32 return prd->readoutElement()->stripLayer(prd->measurementHash()).design();
35 const auto*
re = prd->readoutElement();
36 switch(prd->channelType()) {
38 return re->stripDesign(prd->measurementHash());
40 return re->wireDesign(prd->measurementHash());
42 return re->padDesign(prd->measurementHash());
45 THROW_EXCEPTION(
"Invalid space point for design retrieval "<<
sp.msSector()->idHelperSvc()->toString(
sp.identify()));
48 const auto& design = getDesign(
sp);
51 return 0.5* design.stripLength(prd->channelNumber());
56 auto padCorners = padDesign.
padCorners(prd->channelNumber());
57 return 0.5* std::abs(padCorners[0].
x() - padCorners[1].
x());
59 return 0.5* design.stripLength(prd->channelNumber());
64 inline std::string sTgcChannelType(
const int chType) {
71 double minPull{std::numeric_limits<float>::max()};
72 const MuonR4::SpacePoint* spacePoint{
nullptr};
94 ATH_MSG_ERROR(
"MM or STGC not part of initialized detector layout");
95 return StatusCode::FAILURE;
101 fitCfg.calcAlongStrip =
false;
103 fitCfg.parsToUse = {ParamDefs::x0, ParamDefs::y0, ParamDefs::theta, ParamDefs::phi};
105 m_lineFitter = std::make_unique<SegmentFit::SegmentLineFitter>(name(), std::move(fitCfg));
108 m_seedCounter = std::make_unique<SeedStatistics>();
111 return StatusCode::SUCCESS;
117 for (std::size_t l = 0; l < sortedSp.size(); ++l) {
118 emptyKeeper[l].resize(sortedSp[l].
size(), 0);
127 const auto& design = getDesign(
sp);
128 if (!design.hasStereoAngle()) {
134 if (
sp.dimension() == 2) {
150 const Amg::Vector3D estPlaneArrivalUp = SeedingAux::extrapolateToPlane(beamSpotPos, dirEstUp, testHit);
151 const Amg::Vector3D estPlaneArrivalDn = SeedingAux::extrapolateToPlane(beamSpotPos, dirEstDn, testHit);
153 bool below{
true}, above{
true};
158 const double halfLength = 0.5* stripHalfLength(testHit);
164 below = estPlaneArrivalDn.y() > std::max(leftEdge.y(), rightEdge.y());
166 above = estPlaneArrivalUp.y() < std::min(leftEdge.y(), rightEdge.y());
172 below = estPlaneArrivalDn.y() > hY;
174 above = estPlaneArrivalUp.y() < hY;
186 << (below || above ?
" is outside the window" :
" is inside the window"));
197#define TEST_HIT_CORRIDOR(LAYER, HIT_ITER, START_LAYER) \
199 const SpacePoint* testMe = combinatoricLayers[LAYER].get()[HIT_ITER]; \
200 if (usedHits[LAYER].get()[HIT_ITER] > m_maxUsed) { \
201 ATH_MSG_VERBOSE(__func__<<":"<<__LINE__<<" - " \
202 <<m_idHelperSvc->toString(testMe->identify()) \
203 <<" already used in good seed." ); \
206 const HitWindow inWindow = hitFromIPCorridor(*testMe, beamSpot, dirEstUp, dirEstDn); \
207 if(inWindow == HitWindow::tooHigh) { \
208 ATH_MSG_VERBOSE(__func__<<":"<<__LINE__<<" - Hit " \
209 <<m_idHelperSvc->toString(testMe->identify()) \
210 <<" is beyond the corridor. Break loop"); \
212 } else if (inWindow == HitWindow::tooLow) { \
213 START_LAYER = HIT_ITER + 1; \
214 ATH_MSG_VERBOSE(__func__<<":"<<__LINE__<<" - Hit " \
215 <<m_idHelperSvc->toString(testMe->identify()) \
216 <<" is still below the corridor. Update start to " \
227 seedHitsFromLayers.clear();
228 std::size_t maxSize{1};
229 for (
const HitVec& hitVec : combinatoricLayers) {
230 maxSize = maxSize * hitVec.size();
232 seedHitsFromLayers.reserve(maxSize);
234 unsigned iterLay0{0}, iterLay1{0}, iterLay2{0}, iterLay3{0};
235 unsigned startLay1{0}, startLay2{0}, startLay3{0};
237 for( ; iterLay0 < combinatoricLayers[0].get().
size() ; ++iterLay0){
242 const SpacePoint* hit0 = combinatoricLayers[0].get()[iterLay0];
251 <<
", seed plane: "<<
Amg::toString(SeedingAux::extrapolateToPlane(beamSpot, initSeedDir, *hit0)));
253 for( iterLay1 = startLay1; iterLay1 < combinatoricLayers[1].get().
size() ; ++iterLay1){
255 for( iterLay2 = startLay2; iterLay2 < combinatoricLayers[2].get().
size() ; ++iterLay2){
257 for( iterLay3 = startLay3; iterLay3 < combinatoricLayers[3].get().
size(); ++iterLay3){
259 seedHitsFromLayers.emplace_back(std::array{hit0, combinatoricLayers[1].get()[iterLay1],
260 combinatoricLayers[2].get()[iterLay2],
261 combinatoricLayers[3].get()[iterLay3]});
267#undef TEST_HIT_CORRIDOR
278 for (std::size_t i = 0; i < extensionLayers.size(); ++i) {
279 const HitVec& layer{extensionLayers[i].get()};
280 const Amg::Vector3D extrapPos = SeedingAux::extrapolateToPlane(startPos, direction, *layer.front());
283 unsigned triedHit{0};
284 HitCandidate precisionHit, noPrecisionHit;
287 for (
unsigned j = 0; j < layer.size(); ++j) {
291 auto hit = layer.at(j);
292 const double pull = std::sqrt(SeedingAux::chi2Term(extrapPos, direction, *hit));
295 double minPull = isPrecision ? precisionHit.minPull : noPrecisionHit.minPull;
297 if (pull > minPull) {
307 precisionHit.spacePoint = hit;
308 precisionHit.minPull = pull;
312 noPrecisionHit.spacePoint = hit;
313 noPrecisionHit.minPull = pull;
320 bestCand = precisionHit.spacePoint;
322 bestCand = noPrecisionHit.spacePoint;
330 combinatoricHits.push_back(bestCand);
333 return combinatoricHits;
336std::unique_ptr<SegmentSeed>
342 bool allValid = std::any_of(initialSeed.begin(), initialSeed.end(),
343 [
this](
const auto& hit){
344 if (hit->type() == xAOD::UncalibMeasType::MMClusterType) {
345 const auto* mmClust = static_cast<const xAOD::MMCluster*>(hit->primaryMeasurement());
346 return mmClust->stripNumbers().size() >= m_minClusSize;
352 ATH_MSG_VERBOSE(
"Seed rejection: Not all clusters meet minimum strip size");
357 std::array<double, 4> params = defineParameters(bMatrix, initialSeed);
359 const auto [segPos, direction] = seedSolution(initialSeed, params);
363 for (std::size_t i = 0; i < 4; ++i) {
364 const double halfLength = stripHalfLength(*initialSeed[i]);
366 if (std::abs(params[i]) > halfLength) {
367 ATH_MSG_VERBOSE(
"Seed Rejection: Invalid seed - outside of the strip's length "<< m_idHelperSvc->toString(initialSeed[i]->identify())
368 <<
", param: "<<params[i]<<
", halfLength: "<<halfLength);
375 double interceptX = segPos.x();
376 double interceptY = segPos.y();
379 if(std::abs(tanAlpha) > m_maxTanAlpha){
380 ATH_MSG_VERBOSE(
"Seed Rejection: Invalid seed - tanAlpha "<<tanAlpha<<
" above threshold "<<m_maxTanAlpha);
386 auto extendedHits = extendHits(segPos, direction, extensionLayers, usedHits);
387 HitVec hits{initialSeed.begin(),initialSeed.end()};
388 std::ranges::move(extendedHits, std::back_inserter(hits));
390 return std::make_unique<SegmentSeed>(tanBeta, interceptY, tanAlpha,
391 interceptX,
hits.size(),
392 std::move(hits),
max.parentBucket());
401 ATH_MSG_VERBOSE(
"Not enough hits in the SegmentSeed to fit a segment");
406 if (
msgLvl(MSG::VERBOSE)) {
407 std::stringstream hitStream{};
409 hitStream<<
"**** "<< (*hit)<<std::endl;
411 ATH_MSG_VERBOSE(__func__<<
"() - "<<__LINE__ <<
": Uncalibrated space points for the segment fit: "<<std::endl
423 locToGlob, std::move(calibratedHits));
444 << segment->measurements().size()
445 <<
" hits, chi2/ndof: "
446 << segment->chi2() / std::max(1u,segment->nDoF()));
449 segMeasSP.reserve(segment->measurements().size());
451 std::ranges::transform(
452 segment->measurements(),
453 std::back_inserter(segMeasSP),
454 [](
const auto& m) { return m->spacePoint(); }
460 segments.push_back(std::move(segment));
464std::pair<NswSegmentFinderAlg::SegmentSeedVec_t, NswSegmentFinderAlg::SegmentVec_t>
476 std::size_t layerSize = hitLayers.size();
480 auto isUnusedCombined = [&](std::size_t layIdx, std::size_t hitIdx) ->
bool {
483 THROW_EXCEPTION(
"Space point is not of sTgc type: "<<
sp->msSector()->idHelperSvc()->toString(
sp->identify()));
486 bool isCombined = sTgcChannelType(prd->channelType()) ==
"S" &&
sp->dimension() == 2;
487 bool isUnused = usedHits[layIdx].at(hitIdx) <=
m_maxUsed;
488 return isCombined && isUnused;
493 for(std::size_t layIdx1 = 0; layIdx1 < 2; ++layIdx1){
494 for(std::size_t layIdx2 = layerSize-1; layIdx2 >= layerSize-2; --layIdx2){
498 ATH_MSG_VERBOSE(
"Outermost layers are MM - stop searching for sTgc Measurements");
499 return std::make_pair(std::move(seeds), std::move(segments));
503 for(std::size_t hitIdx1 = 0; hitIdx1 < hitLayers[layIdx1].size(); ++hitIdx1) {
504 const SpacePoint* hit1 = hitLayers[layIdx1][hitIdx1];
505 if(!isUnusedCombined(layIdx1, hitIdx1)){
508 for(std::size_t hitIdx2 = 0; hitIdx2 < hitLayers[layIdx2].size(); ++hitIdx2) {
509 const SpacePoint* hit2 = hitLayers[layIdx2][hitIdx2];
510 if(!isUnusedCombined(layIdx2, hitIdx2)){
516 const double cosAngle = beamSpotHitDir.dot(seedDir);
518 if(std::abs(cosAngle) < thetaWindowCut){
522 HitVec seedHits{hit1, hit2};
530 extensionLayers.reserve(hitLayers.size());
531 usedExtensionHits.reserve(hitLayers.size());
532 for (std::size_t e = 0 ; e < hitLayers.size(); ++e) {
533 if (!(e == layIdx1 || e == layIdx2)){
534 extensionLayers.emplace_back(hitLayers[e]);
535 usedExtensionHits.emplace_back(usedHits[e]);
538 auto extendedHits =
extendHits(seedPosZ0, seedDir, extensionLayers, usedExtensionHits);
539 std::ranges::move(extendedHits, std::back_inserter(seedHits));
542 auto seed = std::make_unique<SegmentSeed>(
houghTanBeta(seedDir), seedPosZ0.y(),
544 seedHits.size(), std::move(seedHits),
549 seeds.push_back(std::move(seed));
553 std::unique_ptr<Segment> segment =
fitSegmentSeed(ctx, gctx, seed.get());
554 processSegment(std::move(segment), seed->getHitsInMax(), hitLayers, usedHits, segments);
555 seeds.push_back(std::move(seed));
561 return std::make_pair(std::move(seeds),std::move(segments));
564std::pair<NswSegmentFinderAlg::SegmentSeedVec_t, NswSegmentFinderAlg::SegmentVec_t>
571 bool useOnlyMM)
const {
576 std::size_t layerSize = hitLayers.size();
580 return {std::move(seeds), std::move(segments)};
584 auto unusedStripHit = [&](
const HitVec& layerHits,
unsigned int layIdx) ->
const SpacePoint* {
587 for (
auto [idx, hit] : Acts::enumerate(layerHits)) {
590 bool isUnused = usedHits[layIdx].at(idx) <=
m_maxUsed;
591 if (isStrip && isUnused && isMM) {
598 std::array<const SpacePoint*, 4> seedHits{};
601 for (std::size_t i = 0; i < layerSize - 3; ++i) {
602 seedHits[0] = unusedStripHit(hitLayers[i], i);
606 for (std::size_t j = i + 1; j < layerSize - 2; ++j) {
607 seedHits[1] = unusedStripHit(hitLayers[j], j);
611 for (std::size_t l = layerSize - 1; l > j+1; --l) {
612 seedHits[3] = unusedStripHit(hitLayers[l], l);
616 for (std::size_t k = l-1; k > j ; --k) {
617 seedHits[2] = unusedStripHit(hitLayers[k], k);
622 const HitLaySpan_t layers{hitLayers[i], hitLayers[j], hitLayers[k], hitLayers[l]};
624 bool tooBusy = std::ranges::any_of(layers,
625 [
this](
const auto& layer) {
634 if (std::abs(bMatrix.determinant()) < 1.e-6) {
638 <<(*seedHits[0]) <<
",\n"
639 <<(*seedHits[1]) <<
",\n"
640 <<(*seedHits[2]) <<
",\n"
644 UsedHitSpan_t usedHitsSpan{usedHits[i], usedHits[j], usedHits[k], usedHits[l]};
651 usedExtensionHits.reserve(hitLayers.size());
652 extensionLayers.reserve(hitLayers.size());
653 for (std::size_t e = 0 ; e < hitLayers.size(); ++e) {
654 if (!(e == i || e == j || e == k || e == l)){
655 extensionLayers.emplace_back(hitLayers[e]);
656 usedExtensionHits.emplace_back(usedHits[e]);
661 for (
auto &combinatoricHits : preLimSeeds) {
667 seeds.push_back(std::move(seed));
670 std::unique_ptr<Segment> segment =
fitSegmentSeed(ctx, gctx, seed.get());
671 processSegment(std::move(segment), seed->getHitsInMax(), hitLayers, usedHits, segments);
672 seeds.push_back(std::move(seed));
679 return std::make_pair(std::move(seeds),std::move(segments));
682std::pair<NswSegmentFinderAlg::SegmentSeedVec_t, NswSegmentFinderAlg::SegmentVec_t>
685 const EventContext& ctx)
const {
690 const std::size_t layerSize = stripHitsLayers.size();
701 return std::make_pair(std::move(seeds),std::move(segments));
706 constexpr double legX{0.2};
720 const double pull = std::sqrt(SeedingAux::chi2Term(hitPos, hitDir, *
sp));
723 const auto* mmClust =
static_cast<const xAOD::MMCluster*
>(
sp->primaryMeasurement());
725 const auto& design = mmEle->
stripLayer(mmClust->measurementHash()).
design();
726 std::string stereoDesign{!design.hasStereoAngle() ?
"X" : design.stereoAngle() >0 ?
"U":
"V"};
727 primitives.push_back(
MuonValR4::drawLabel(std::format(
"ml: {:1d}, gap: {:1d}, {:}, pull: {:.2f}",
729 stereoDesign, pull), legX, legY, 14));
732 std::string channelString =
sp->secondaryMeasurement() ==
nullptr ?
733 sTgcChannelType(sTgcMeas->channelType()) :
734 std::format(
"{:}/{:}", sTgcChannelType(sTgcMeas->channelType()),
736 primitives.push_back(
MuonValR4::drawLabel(std::format(
"ml: {:1d}, gap: {:1d}, type: {:}, pull: {:.2f}",
737 sTgcMeas->readoutElement()->multilayer(), sTgcMeas->gasGap(),
738 channelString, pull), legX, legY, 14));
743 "truth", std::move(primitives));
748 Acts::ObjVisualization3D visualHelper{};
756 visualHelper.write(std::format(
"Event_{:}_{:}_spacepoints_truth.obj", ctx.eventID().event_number(),
max.getHitsInMax().front()->chamber()->identString()));
761 std::size_t nSeeds{0}, nExtSeeds{0}, nSegments{0};
765 auto processSeedsAndSegments = [&](std::pair<SegmentSeedVec_t, SegmentVec_t>&& seedSegmentPairs, std::string_view source) {
766 auto& [returnSeeds, returnSegments] = seedSegmentPairs;
767 ATH_MSG_DEBUG(
"From " << source <<
": built " << returnSeeds.size() <<
" seeds and " << returnSegments.size() <<
" segments.");
768 for(
auto& seed : returnSeeds) {
770 Acts::ObjVisualization3D visualHelper{};
772 ATH_MSG_VERBOSE(
"Seed with "<< seed->getHitsInMax().size() <<
" hits rejected");
773 for(
const auto& hit : seed->getHitsInMax()){
783 visualHelper.write(std::format(
"Event_{:}_{:}_notExtendedSeed.obj", ctx.eventID().event_number(), seed->getHitsInMax().front()->chamber()->identString()));
788 seeds.push_back(std::move(seed));
791 std::ranges::move(returnSegments, std::back_inserter(segments));
792 nSegments += returnSegments.size();
797 processSeedsAndSegments(
buildSegmentsFromSTGC(ctx, gctx, stripHitsLayers,
max, globToLocal.translation(), allUsedHits),
"sTgc segment seeds");
802 ATH_MSG_VERBOSE(
"Start building combinatoric seeds only from Micromegas");
803 processSeedsAndSegments(
buildSegmentsFromMM(ctx, gctx, stripHitsLayers,
max, globToLocal.translation(), allUsedHits,
true),
"MM combinatoric segment seeds");
807 ATH_MSG_VERBOSE(
"Start building combinatoric seeds from Micromegas and sTgc hits");
808 processSeedsAndSegments(
buildSegmentsFromMM(ctx, gctx, stripHitsLayers,
max, globToLocal.translation(), allUsedHits,
true),
"MM combinatoric segment seeds");
809 processSeedsAndSegments(
buildSegmentsFromMM(ctx, gctx, stripHitsLayers,
max, globToLocal.translation(), allUsedHits,
false),
"MM and STGC combinatoric segment seeds");
814 m_seedCounter->addToStat(
max.msSector(), nSeeds, nExtSeeds, nSegments);
817 return std::make_pair(std::move(seeds),std::move(segments));
824 bool markNeighborHits)
const {
828 for(
const auto&
sp : spacePoints){
838 switch (
sp->primaryMeasurement()->numDimensions()) {
840 spPosX[
Amg::x] =
sp->primaryMeasurement()->localPosition<1>().
x();
843 spPosX = xAOD::toEigen(
sp->primaryMeasurement()->localPosition<2>());
849 for (std::size_t lIdx = 0; lIdx < allSortHits.size(); ++lIdx) {
850 const HitVec& hVec{allSortHits[lIdx]};
853 if(hitLayer != measLayer) {
854 ATH_MSG_VERBOSE(
"Not in the same layer since measLayer = "<< measLayer <<
" and "<<hitLayer);
857 for (std::size_t hIdx = 0 ; hIdx < hVec.size(); ++hIdx) {
859 auto testHit = hVec[hIdx];
861 usedHitMarker[lIdx][hIdx] += incr;
862 if(!markNeighborHits){
865 }
else if (markNeighborHits) {
868 switch (testHit->primaryMeasurement()->numDimensions()) {
870 testPosX[
Amg::x] = testHit->primaryMeasurement()->localPosition<1>().
x();
873 testPosX = xAOD::toEigen(testHit->primaryMeasurement()->localPosition<2>());
879 double deltaX = (testPosX - spPosX).
mag();
881 usedHitMarker[lIdx][hIdx] += incr;
902 ATH_CHECK(writeSegmentSeeds.
record(std::make_unique<SegmentSeedContainer>()));
910 if (
msgLvl(MSG::VERBOSE)) {
912 for(
const auto& hitMax :
max->getHitsInMax()){
919 for(
auto& seed: seeds){
921 if (
msgLvl(MSG::VERBOSE)){
922 std::stringstream sstr{};
923 sstr<<
"Seed tanBeta = "<<seed->tanBeta()<<
", y0 = "<<seed->interceptY()
924 <<
", tanAlpha = "<<seed->tanAlpha()<<
", x0 = "<<seed->interceptX()<<
", hits in the seed "
925 <<seed->getHitsInMax().size()<<std::endl;
927 for(
const auto& hit : seed->getHitsInMax()){
928 sstr<<
" *** Hit "<<
m_idHelperSvc->toString(hit->identify())<<
", "
934 m_visionTool->visualizeSeed(ctx, *seed,
"#phi-combinatorialSeed");
937 writeSegmentSeeds->push_back(std::move(seed));
941 for (
auto &seg : segments) {
947 m_visionTool->visualizeSegment(ctx, *seg,
"#phi-segment");
951 Acts::ObjVisualization3D visualHelper{};
954 visualHelper.write(std::format(
"Event_{:}_segment_{:}.obj", ctx.eventID().event_number(), seg->msSector()->identString()));
957 writeSegments->push_back(std::move(seg));
962 return StatusCode::SUCCESS;
967 m_seedCounter->printTableSeedStats(msgStream());
969 return StatusCode::SUCCESS;
973 std::unique_lock guard{
m_mutex};
977 key.eta = msSector->
chambers().front()->stationEta();
980 entry.nSeeds += seeds;
981 entry.nExtSeeds += extSeeds;
982 entry.nSegments += segments;
988 std::stringstream sstr{};
989 sstr<<
"Seed statistics per sector:"<<std::endl;
990 sstr<<
"-----------------------------------------------------"<<std::endl;
991 sstr<<
"| Chamber | Phi | Eta | Seeds | ExtSeeds | Segments |"<<std::endl;
992 sstr<<
"-----------------------------------------------------"<<std::endl;
996 for (
const auto& [sector, stats] :
m_seedStat) {
997 sstr <<
"| " << std::setw(3) <<
chName(sector.chIdx)
998 <<
" | " << std::setw(2) <<
static_cast<unsigned>(sector.phi)
999 <<
" | " << std::setw(3) <<
static_cast<int>(sector.eta)
1000 <<
" | " << std::setw(7) << stats.nSeeds
1001 <<
" | " << std::setw(8) << stats.nExtSeeds
1002 <<
" | " << std::setw(8) << stats.nSegments
1006 sstr<<
"------------------------------------------------------------"<<std::endl;
1007 msg<<MSG::ALWAYS<<
"\n"<<sstr.str()<<
endmsg;
const boost::regex re(r_e)
Scalar mag() const
mag method
#define ATH_CHECK
Evaluate an expression and check for errors.
#define ATH_MSG_VERBOSE(x)
#define ATH_MSG_WARNING(x)
#define AmgSymMatrix(dim)
ATLAS-specific HepMC functions.
sTgcIdHelper::sTgcChannelTypes chType
#define TEST_HIT_CORRIDOR(LAYER, HIT_ITER, START_LAYER)
Macro to check whether a hit is compatible with the hit corridor.
size_t size() const
Number of registered mappings.
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
SeedStatistic_T m_seedStat
void printTableSeedStats(MsgStream &msg) const
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
DoubleProperty m_minPullThreshold
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.
DoubleProperty m_maxdYWindow
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
DoubleProperty m_windowTheta
BooleanProperty m_dumpSeedStatistics
ServiceHandle< Muon::IMuonIdHelperSvc > m_idHelperSvc
SpacePointPerLayerSplitter::HitLayVec HitLayVec
SpacePointPerLayerSplitter::HitVec HitVec
std::vector< InitialSeed_t > InitialSeedVec_t
Vector of initial seeds.
BooleanProperty m_doOnlyMMCombinatorics
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.
@ C
Single phi measurements.
@ 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.
BooleanProperty m_dumpObj
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.
BooleanProperty m_markHitsFromSeed
Representation of a segment seed (a fully processed hough maximum) produced by the hough transform.
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 Amg::Vector3D & sensorDirection() const
const Identifier & identify() const
: Identifier of the primary measurement
const Amg::Vector3D & localPosition() const
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?)
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
double houghTanBeta(const Amg::Vector3D &v)
Returns the hough tanBeta [y] / [z].
DataVector< HoughMaximum > EtaHoughMaxContainer
const xAOD::MuonSimHit * getTruthMatchedHit(const xAOD::MuonMeasurement &prdHit)
Returns the MuonSimHit, if there's any, matched to the uncalibrated muon measurement.
constexpr unsigned minLayers
bool isPrecisionHit(const SpacePoint &hit)
Returns whether the uncalibrated spacepoint is a precision hit (Mdt, micromegas, stgc strips).
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.
sTgcMeasurement_v1 sTgcMeasurement
sector's field to dump the seed statistics
const ISpacePointCalibrator * calibrator
Pointer to the calibrator.
const Muon::IMuonIdHelperSvc * idHelperSvc
Pointer to the idHelperSvc.
const MuonValR4::IPatternVisualizationTool * visionTool
Pointer to the visualization tool.
Full configuration object.
#define THROW_EXCEPTION(MESSAGE)