From 5865281cdd91b962085f63f75650908fc1d70553 Mon Sep 17 00:00:00 2001 From: Ian Tomalin Date: Fri, 29 Mar 2024 14:50:50 +0000 Subject: [PATCH 01/14] Run combined modules by default (#265) * Make combined modules default * tweak * Improve USEHYBRID ifdef range * Fix compiler error for pure Tracklet algo * Move fitpattern.txt refs, so only used for pure Tracklet algo * code format --- L1Trigger/TrackFindingTracklet/plugins/L1FPGATrackProducer.cc | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/L1Trigger/TrackFindingTracklet/plugins/L1FPGATrackProducer.cc b/L1Trigger/TrackFindingTracklet/plugins/L1FPGATrackProducer.cc index 6bc9104b41764..8a3055c3173ee 100644 --- a/L1Trigger/TrackFindingTracklet/plugins/L1FPGATrackProducer.cc +++ b/L1Trigger/TrackFindingTracklet/plugins/L1FPGATrackProducer.cc @@ -282,6 +282,10 @@ L1FPGATrackProducer::L1FPGATrackProducer(edm::ParameterSet const& iConfig) settings_.setExtended(extended_); settings_.setReduced(reduced_); settings_.setNHelixPar(nHelixPar_); + // combined_ must be false for extended tracking, regardless of whether + // combined modules are used or not. + if (extended_) + settings_.setCombined(false); #ifndef USEHYBRID fitPatternFile = iConfig.getParameter("fitPatternFile"); From a274918defc6b32494e03c307841f2108a9cc3ce Mon Sep 17 00:00:00 2001 From: cgsavard Date: Fri, 17 May 2024 12:47:12 -0600 Subject: [PATCH 02/14] Bin TQ MVA output (#272) * add TQ MVA output binning scheme * remove code no longer used * move logistic sigmoid into TTTrack class * fix for proper precision after MVA conversion in track word * label all MVA variables before logistic sigmoid with 'Pre' * formatting * fix filling of track word before running TQ MVA * switch to post-sigmoid mva bins * remove any onnx instances * fix MVA1 initialization * bin TQ MVA in pre-sigmoided bins in KFout * move pre-sigmoid bins to track quality class * change minimum bin --- L1Trigger/TrackFindingTracklet/plugins/L1FPGATrackProducer.cc | 3 +++ 1 file changed, 3 insertions(+) diff --git a/L1Trigger/TrackFindingTracklet/plugins/L1FPGATrackProducer.cc b/L1Trigger/TrackFindingTracklet/plugins/L1FPGATrackProducer.cc index 8a3055c3173ee..76790d82e49e3 100644 --- a/L1Trigger/TrackFindingTracklet/plugins/L1FPGATrackProducer.cc +++ b/L1Trigger/TrackFindingTracklet/plugins/L1FPGATrackProducer.cc @@ -765,6 +765,9 @@ void L1FPGATrackProducer::produce(edm::Event& iEvent, const edm::EventSetup& iSe // test track word //aTrack.testTrackWordBits(); + // set track word again to set MVA variable from TTTrack into track word + aTrack.setTrackWordBits(); + L1TkTracksForOutput->push_back(aTrack); } From 944c6ad26377eb24e3536becf4ca87147fd2d003 Mon Sep 17 00:00:00 2001 From: Thomas Schuh Date: Fri, 22 Nov 2024 15:32:00 +0000 Subject: [PATCH 03/14] squash --- .../python/L1TrackTrigger_cff.py | 4 +- DataFormats/L1TrackTrigger/interface/TTBV.h | 129 ++- .../L1TrackTrigger/src/classes_def.xml | 2 + .../test/L1TrackObjectNtupleMaker_cfg.py | 4 +- .../make_l1ctLayer1_dumpFiles_fromRAW_cfg.py | 5 +- .../make_l1ct_patternFiles_fromRAW_cfg.py | 5 +- .../TrackFindingTMTT/interface/L1track3D.h | 29 + L1Trigger/TrackFindingTMTT/interface/Stub.h | 13 + .../python/TMTrackProducer_Defaults_cfi.py | 2 +- L1Trigger/TrackFindingTMTT/src/Settings.cc | 4 +- L1Trigger/TrackFindingTMTT/src/Stub.cc | 47 + .../interface/ChannelAssignment.h | 44 +- .../interface/DataFormats.h | 492 +++++++++ .../interface/DataFormatsRcd.h | 17 + .../interface/{DR.h => DuplicateRemoval.h} | 48 +- .../interface/HitPatternHelper.h | 10 +- .../TrackFindingTracklet/interface/KFin.h | 95 -- .../interface/KalmanFilter.h | 164 +++ .../interface/KalmanFilterFormats.h | 243 +++++ .../TrackFindingTracklet/interface/State.h | 143 +++ .../interface/{DRin.h => TrackMultiplexer.h} | 92 +- .../plugins/L1FPGATrackProducer.cc | 146 +-- .../plugins/ProducerDR.cc | 86 +- .../plugins/ProducerDRin.cc | 152 --- .../plugins/ProducerDataFormats.cc | 46 + .../plugins/ProducerKF.cc | 178 ++++ .../plugins/ProducerKFin.cc | 147 --- .../plugins/ProducerKFout.cc | 381 ------- .../plugins/ProducerTBout.cc | 192 ---- .../plugins/ProducerTM.cc | 127 +++ .../python/Analyzer_cff.py | 14 +- .../python/Analyzer_cfi.py | 9 +- .../python/ChannelAssignment_cff.py | 2 + .../python/ChannelAssignment_cfi.py | 18 +- .../python/Customize_cff.py | 48 +- .../python/DataFormats_cff.py | 11 + .../python/Demonstrator_cff.py | 3 + .../python/Demonstrator_cfi.py | 15 +- .../python/KalmanFilterFormats_cfi.py | 66 ++ .../python/ProducerHPH_cff.py | 6 +- .../python/Producer_cff.py | 25 +- .../python/Producer_cfi.py | 38 +- .../l1tTTTracksFromTrackletEmulation_cfi.py | 14 +- .../src/ChannelAssignment.cc | 28 +- L1Trigger/TrackFindingTracklet/src/DR.cc | 160 --- L1Trigger/TrackFindingTracklet/src/DRin.cc | 459 --------- .../TrackFindingTracklet/src/DataFormats.cc | 227 ++++ .../src/DuplicateRemoval.cc | 183 ++++ .../src/ES_DataFormats.cc | 6 + .../TrackFindingTracklet/src/FitTrack.cc | 4 +- .../src/HitPatternHelper.cc | 17 +- L1Trigger/TrackFindingTracklet/src/KFin.cc | 270 ----- .../TrackFindingTracklet/src/KalmanFilter.cc | 721 +++++++++++++ .../src/KalmanFilterFormats.cc | 588 +++++++++++ L1Trigger/TrackFindingTracklet/src/State.cc | 170 +++ .../src/TrackMultiplexer.cc | 426 ++++++++ .../TrackFindingTracklet/test/AnalyzerDR.cc | 117 +-- .../test/AnalyzerDemonstrator.cc | 62 +- .../TrackFindingTracklet/test/AnalyzerKF.cc | 413 ++++++++ .../test/AnalyzerKFout.cc | 6 +- .../test/AnalyzerTBout.cc | 429 -------- .../test/AnalyzerTFP.cc} | 219 ++-- .../test/{AnalyzerDRin.cc => AnalyzerTM.cc} | 163 ++- .../test/AnalyzerTracklet.cc | 22 +- .../test/HybridTracksNewKF_cfg.py | 47 +- .../test/HybridTracks_cfg.py | 21 +- .../test/L1TrackNtupleMaker.cc | 92 +- .../test/L1TrackNtupleMaker_cfg.py | 24 +- .../TrackFindingTracklet/test/ProducerIRin.cc | 23 +- .../test/demonstrator_cfg.py | 33 +- .../TrackTrigger/interface/L1TrackQuality.h | 73 -- .../TrackTrigger/interface/SensorModule.h | 16 + L1Trigger/TrackTrigger/interface/Setup.h | 564 +++++----- L1Trigger/TrackTrigger/interface/SetupRcd.h | 18 +- .../TrackTrigger/plugins/ProducerSetup.cc | 50 +- .../TrackTrigger/python/ProducerSetup_cff.py | 14 - .../TrackTrigger/python/ProducerSetup_cfi.py | 230 ----- L1Trigger/TrackTrigger/python/Setup_cff.py | 6 + L1Trigger/TrackTrigger/python/Setup_cfi.py | 204 ++++ .../python/TTStubAlgorithmRegister_cfi.py | 3 +- .../python/TrackQualityParams_cfi.py | 11 - L1Trigger/TrackTrigger/src/L1TrackQuality.cc | 115 --- L1Trigger/TrackTrigger/src/SensorModule.cc | 7 + L1Trigger/TrackTrigger/src/Setup.cc | 459 +++------ .../TrackTrigger/test/CleanRelVal_cfg.py | 78 ++ L1Trigger/TrackerDTC/BuildFile.xml | 3 +- L1Trigger/TrackerDTC/interface/DTC.h | 4 + .../TrackerDTC/interface/LayerEncoding.h | 5 +- L1Trigger/TrackerDTC/interface/Stub.h | 37 +- .../plugins/{ProducerED.cc => ProducerDTC.cc} | 77 +- .../plugins/ProducerLayerEncoding.cc | 5 +- .../TrackerDTC/python/AnalyzerDAQ_cff.py | 4 +- .../TrackerDTC/python/AnalyzerDAQ_cfi.py | 2 + L1Trigger/TrackerDTC/python/Analyzer_cff.py | 8 +- L1Trigger/TrackerDTC/python/Analyzer_cfi.py | 13 +- L1Trigger/TrackerDTC/python/Customize_cff.py | 20 +- .../python/{ProducerED_cff.py => DTC_cff.py} | 9 +- .../python/{ProducerED_cfi.py => DTC_cfi.py} | 5 +- .../TrackerDTC/python/LayerEncoding_cff.py | 3 + .../python/ProducerLayerEncoding_cff.py | 5 - .../python/ProducerLayerEncoding_cfi.py | 7 - L1Trigger/TrackerDTC/src/DTC.cc | 17 +- L1Trigger/TrackerDTC/src/LayerEncoding.cc | 5 +- L1Trigger/TrackerDTC/src/Stub.cc | 285 +++-- L1Trigger/TrackerDTC/test/Analyzer.cc | 429 +++----- L1Trigger/TrackerDTC/test/testDAQ_cfg.py | 4 +- L1Trigger/TrackerDTC/test/test_cfg.py | 6 +- L1Trigger/TrackerTFP/BuildFile.xml | 1 + L1Trigger/TrackerTFP/README.md | 28 +- .../TrackerTFP/interface/CleanTrackBuilder.h | 138 +++ L1Trigger/TrackerTFP/interface/DataFormats.h | 972 ++++++------------ L1Trigger/TrackerTFP/interface/Demonstrator.h | 16 +- .../TrackerTFP/interface/DuplicateRemoval.h | 58 ++ .../TrackerTFP/interface/GeometricProcessor.h | 26 +- .../TrackerTFP/interface/HoughTransform.h | 52 +- L1Trigger/TrackerTFP/interface/KalmanFilter.h | 156 +-- .../interface/KalmanFilterFormats.h | 120 ++- .../interface/KalmanFilterFormatsRcd.h | 17 - .../TrackerTFP/interface/LayerEncoding.h | 38 +- .../TrackerTFP/interface/MiniHoughTransform.h | 69 -- L1Trigger/TrackerTFP/interface/State.h | 125 ++- .../interface/TrackFindingProcessor.h | 89 ++ L1Trigger/TrackerTFP/interface/TrackQuality.h | 156 +++ .../TrackerTFP/interface/TrackQualityRcd.h | 17 + .../TrackerTFP/interface/ZHoughTransform.h | 56 - L1Trigger/TrackerTFP/plugins/ProducerCTB.cc | 246 +++++ L1Trigger/TrackerTFP/plugins/ProducerDR.cc | 188 ++++ .../{ProducerES.cc => ProducerDataFormats.cc} | 16 +- .../plugins/ProducerDemonstrator.cc | 3 +- .../TrackerTFP/plugins/ProducerFormatsKF.cc | 45 - L1Trigger/TrackerTFP/plugins/ProducerGP.cc | 118 ++- L1Trigger/TrackerTFP/plugins/ProducerHT.cc | 101 +- L1Trigger/TrackerTFP/plugins/ProducerKF.cc | 195 ++-- L1Trigger/TrackerTFP/plugins/ProducerKFin.cc | 225 ---- L1Trigger/TrackerTFP/plugins/ProducerMHT.cc | 110 -- L1Trigger/TrackerTFP/plugins/ProducerPP.cc | 81 ++ L1Trigger/TrackerTFP/plugins/ProducerTFP.cc | 120 +++ L1Trigger/TrackerTFP/plugins/ProducerTQ.cc | 140 +++ L1Trigger/TrackerTFP/plugins/ProducerTT.cc | 124 --- .../plugins/ProducerTrackQuality.cc | 39 + L1Trigger/TrackerTFP/plugins/ProducerZHT.cc | 110 -- .../TrackerTFP/plugins/ProducerZHTout.cc | 135 --- L1Trigger/TrackerTFP/python/Analyzer_cff.py | 14 +- L1Trigger/TrackerTFP/python/Analyzer_cfi.py | 8 +- L1Trigger/TrackerTFP/python/Customize_cff.py | 28 +- .../TrackerTFP/python/DataFormats_cff.py | 5 + .../TrackerTFP/python/Demonstrator_cff.py | 3 + .../TrackerTFP/python/Demonstrator_cfi.py | 16 +- .../python/KalmanFilterFormats_cff.py | 5 - .../python/KalmanFilterFormats_cfi.py | 121 +-- .../TrackerTFP/python/LayerEncoding_cff.py | 5 + L1Trigger/TrackerTFP/python/ProducerES_cff.py | 5 - L1Trigger/TrackerTFP/python/ProducerES_cfi.py | 28 - .../python/ProducerLayerEncoding_cff.py | 5 - .../python/ProducerLayerEncoding_cfi.py | 7 - L1Trigger/TrackerTFP/python/Producer_cff.py | 26 +- L1Trigger/TrackerTFP/python/Producer_cfi.py | 34 +- .../TrackerTFP/python/TrackQuality_cff.py | 12 + .../TrackerTFP/python/TrackQuality_cfi.py | 35 + L1Trigger/TrackerTFP/src/CleanTrackBuilder.cc | 489 +++++++++ L1Trigger/TrackerTFP/src/DataFormats.cc | 889 +++------------- L1Trigger/TrackerTFP/src/Demonstrator.cc | 63 +- L1Trigger/TrackerTFP/src/DuplicateRemoval.cc | 142 +++ .../TrackerTFP/src/ES_KalmanFilterFormats.cc | 4 - L1Trigger/TrackerTFP/src/ES_TrackQuality.cc | 5 + .../TrackerTFP/src/GeometricProcessor.cc | 150 +-- L1Trigger/TrackerTFP/src/HoughTransform.cc | 296 +++--- L1Trigger/TrackerTFP/src/KalmanFilter.cc | 787 ++++++++------ .../TrackerTFP/src/KalmanFilterFormats.cc | 410 ++++++-- .../TrackerTFP/src/KalmanFilterFormatsRcd.cc | 4 - L1Trigger/TrackerTFP/src/LayerEncoding.cc | 203 ++-- .../TrackerTFP/src/MiniHoughTransform.cc | 317 ------ L1Trigger/TrackerTFP/src/State.cc | 246 +++-- .../TrackerTFP/src/TrackFindingProcessor.cc | 278 +++++ L1Trigger/TrackerTFP/src/TrackQuality.cc | 284 +++++ L1Trigger/TrackerTFP/src/ZHoughTransform.cc | 322 ------ .../test/{AnalyzerKFin.cc => AnalyzerCTB.cc} | 209 ++-- .../test/AnalyzerDR.cc} | 178 ++-- .../TrackerTFP/test/AnalyzerDemonstrator.cc | 17 +- L1Trigger/TrackerTFP/test/AnalyzerGP.cc | 105 +- L1Trigger/TrackerTFP/test/AnalyzerHT.cc | 141 ++- L1Trigger/TrackerTFP/test/AnalyzerKF.cc | 349 ++++--- L1Trigger/TrackerTFP/test/AnalyzerTT.cc | 131 --- L1Trigger/TrackerTFP/test/AnalyzerZHT.cc | 294 ------ L1Trigger/TrackerTFP/test/ProducerAS.cc | 97 -- L1Trigger/TrackerTFP/test/demonstrator_cfg.py | 44 +- L1Trigger/TrackerTFP/test/test_cfg.py | 45 +- .../interface/StubAssociation.h | 13 +- .../plugins/StubAssociator.cc | 102 +- .../python/StubAssociator_cff.py | 2 +- .../python/StubAssociator_cfi.py | 24 +- .../src/StubAssociation.cc | 28 +- 192 files changed, 12058 insertions(+), 10144 deletions(-) create mode 100644 L1Trigger/TrackFindingTracklet/interface/DataFormats.h create mode 100644 L1Trigger/TrackFindingTracklet/interface/DataFormatsRcd.h rename L1Trigger/TrackFindingTracklet/interface/{DR.h => DuplicateRemoval.h} (56%) delete mode 100644 L1Trigger/TrackFindingTracklet/interface/KFin.h create mode 100644 L1Trigger/TrackFindingTracklet/interface/KalmanFilter.h create mode 100644 L1Trigger/TrackFindingTracklet/interface/KalmanFilterFormats.h create mode 100644 L1Trigger/TrackFindingTracklet/interface/State.h rename L1Trigger/TrackFindingTracklet/interface/{DRin.h => TrackMultiplexer.h} (64%) delete mode 100644 L1Trigger/TrackFindingTracklet/plugins/ProducerDRin.cc create mode 100644 L1Trigger/TrackFindingTracklet/plugins/ProducerDataFormats.cc create mode 100644 L1Trigger/TrackFindingTracklet/plugins/ProducerKF.cc delete mode 100644 L1Trigger/TrackFindingTracklet/plugins/ProducerKFin.cc delete mode 100644 L1Trigger/TrackFindingTracklet/plugins/ProducerKFout.cc delete mode 100644 L1Trigger/TrackFindingTracklet/plugins/ProducerTBout.cc create mode 100644 L1Trigger/TrackFindingTracklet/plugins/ProducerTM.cc create mode 100644 L1Trigger/TrackFindingTracklet/python/DataFormats_cff.py create mode 100644 L1Trigger/TrackFindingTracklet/python/KalmanFilterFormats_cfi.py delete mode 100644 L1Trigger/TrackFindingTracklet/src/DR.cc delete mode 100644 L1Trigger/TrackFindingTracklet/src/DRin.cc create mode 100644 L1Trigger/TrackFindingTracklet/src/DataFormats.cc create mode 100644 L1Trigger/TrackFindingTracklet/src/DuplicateRemoval.cc create mode 100644 L1Trigger/TrackFindingTracklet/src/ES_DataFormats.cc delete mode 100644 L1Trigger/TrackFindingTracklet/src/KFin.cc create mode 100644 L1Trigger/TrackFindingTracklet/src/KalmanFilter.cc create mode 100644 L1Trigger/TrackFindingTracklet/src/KalmanFilterFormats.cc create mode 100644 L1Trigger/TrackFindingTracklet/src/State.cc create mode 100644 L1Trigger/TrackFindingTracklet/src/TrackMultiplexer.cc create mode 100644 L1Trigger/TrackFindingTracklet/test/AnalyzerKF.cc delete mode 100644 L1Trigger/TrackFindingTracklet/test/AnalyzerTBout.cc rename L1Trigger/{TrackerTFP/test/AnalyzerMHT.cc => TrackFindingTracklet/test/AnalyzerTFP.cc} (51%) rename L1Trigger/TrackFindingTracklet/test/{AnalyzerDRin.cc => AnalyzerTM.cc} (62%) delete mode 100644 L1Trigger/TrackTrigger/interface/L1TrackQuality.h delete mode 100644 L1Trigger/TrackTrigger/python/ProducerSetup_cff.py delete mode 100644 L1Trigger/TrackTrigger/python/ProducerSetup_cfi.py create mode 100644 L1Trigger/TrackTrigger/python/Setup_cff.py create mode 100644 L1Trigger/TrackTrigger/python/Setup_cfi.py delete mode 100644 L1Trigger/TrackTrigger/python/TrackQualityParams_cfi.py delete mode 100644 L1Trigger/TrackTrigger/src/L1TrackQuality.cc create mode 100644 L1Trigger/TrackTrigger/test/CleanRelVal_cfg.py rename L1Trigger/TrackerDTC/plugins/{ProducerED.cc => ProducerDTC.cc} (59%) rename L1Trigger/TrackerDTC/python/{ProducerED_cff.py => DTC_cff.py} (51%) rename L1Trigger/TrackerDTC/python/{ProducerED_cfi.py => DTC_cfi.py} (77%) create mode 100644 L1Trigger/TrackerDTC/python/LayerEncoding_cff.py delete mode 100644 L1Trigger/TrackerDTC/python/ProducerLayerEncoding_cff.py delete mode 100644 L1Trigger/TrackerDTC/python/ProducerLayerEncoding_cfi.py create mode 100644 L1Trigger/TrackerTFP/interface/CleanTrackBuilder.h create mode 100644 L1Trigger/TrackerTFP/interface/DuplicateRemoval.h delete mode 100644 L1Trigger/TrackerTFP/interface/KalmanFilterFormatsRcd.h delete mode 100644 L1Trigger/TrackerTFP/interface/MiniHoughTransform.h create mode 100644 L1Trigger/TrackerTFP/interface/TrackFindingProcessor.h create mode 100644 L1Trigger/TrackerTFP/interface/TrackQuality.h create mode 100644 L1Trigger/TrackerTFP/interface/TrackQualityRcd.h delete mode 100644 L1Trigger/TrackerTFP/interface/ZHoughTransform.h create mode 100644 L1Trigger/TrackerTFP/plugins/ProducerCTB.cc create mode 100644 L1Trigger/TrackerTFP/plugins/ProducerDR.cc rename L1Trigger/TrackerTFP/plugins/{ProducerES.cc => ProducerDataFormats.cc} (67%) delete mode 100644 L1Trigger/TrackerTFP/plugins/ProducerFormatsKF.cc delete mode 100644 L1Trigger/TrackerTFP/plugins/ProducerKFin.cc delete mode 100644 L1Trigger/TrackerTFP/plugins/ProducerMHT.cc create mode 100644 L1Trigger/TrackerTFP/plugins/ProducerPP.cc create mode 100644 L1Trigger/TrackerTFP/plugins/ProducerTFP.cc create mode 100644 L1Trigger/TrackerTFP/plugins/ProducerTQ.cc delete mode 100644 L1Trigger/TrackerTFP/plugins/ProducerTT.cc create mode 100644 L1Trigger/TrackerTFP/plugins/ProducerTrackQuality.cc delete mode 100644 L1Trigger/TrackerTFP/plugins/ProducerZHT.cc delete mode 100644 L1Trigger/TrackerTFP/plugins/ProducerZHTout.cc create mode 100644 L1Trigger/TrackerTFP/python/DataFormats_cff.py delete mode 100644 L1Trigger/TrackerTFP/python/KalmanFilterFormats_cff.py create mode 100644 L1Trigger/TrackerTFP/python/LayerEncoding_cff.py delete mode 100644 L1Trigger/TrackerTFP/python/ProducerES_cff.py delete mode 100644 L1Trigger/TrackerTFP/python/ProducerES_cfi.py delete mode 100644 L1Trigger/TrackerTFP/python/ProducerLayerEncoding_cff.py delete mode 100644 L1Trigger/TrackerTFP/python/ProducerLayerEncoding_cfi.py create mode 100644 L1Trigger/TrackerTFP/python/TrackQuality_cff.py create mode 100644 L1Trigger/TrackerTFP/python/TrackQuality_cfi.py create mode 100644 L1Trigger/TrackerTFP/src/CleanTrackBuilder.cc create mode 100644 L1Trigger/TrackerTFP/src/DuplicateRemoval.cc delete mode 100644 L1Trigger/TrackerTFP/src/ES_KalmanFilterFormats.cc create mode 100644 L1Trigger/TrackerTFP/src/ES_TrackQuality.cc delete mode 100644 L1Trigger/TrackerTFP/src/KalmanFilterFormatsRcd.cc delete mode 100644 L1Trigger/TrackerTFP/src/MiniHoughTransform.cc create mode 100644 L1Trigger/TrackerTFP/src/TrackFindingProcessor.cc create mode 100644 L1Trigger/TrackerTFP/src/TrackQuality.cc delete mode 100644 L1Trigger/TrackerTFP/src/ZHoughTransform.cc rename L1Trigger/TrackerTFP/test/{AnalyzerKFin.cc => AnalyzerCTB.cc} (58%) rename L1Trigger/{TrackFindingTracklet/test/AnalyzerKFin.cc => TrackerTFP/test/AnalyzerDR.cc} (57%) delete mode 100644 L1Trigger/TrackerTFP/test/AnalyzerTT.cc delete mode 100644 L1Trigger/TrackerTFP/test/AnalyzerZHT.cc delete mode 100644 L1Trigger/TrackerTFP/test/ProducerAS.cc diff --git a/Configuration/StandardSequences/python/L1TrackTrigger_cff.py b/Configuration/StandardSequences/python/L1TrackTrigger_cff.py index 213f0c82c7c5e..197bac97529e7 100644 --- a/Configuration/StandardSequences/python/L1TrackTrigger_cff.py +++ b/Configuration/StandardSequences/python/L1TrackTrigger_cff.py @@ -2,10 +2,10 @@ from L1Trigger.TrackTrigger.TrackTrigger_cff import * from SimTracker.TrackTriggerAssociation.TrackTriggerAssociator_cff import * -from L1Trigger.TrackerDTC.ProducerED_cff import * +from L1Trigger.TrackerDTC.DTC_cff import * from L1Trigger.TrackFindingTracklet.L1HybridEmulationTracks_cff import * -L1TrackTrigger=cms.Sequence(TrackTriggerClustersStubs*TrackTriggerAssociatorClustersStubs*TrackerDTCProducer) +L1TrackTrigger=cms.Sequence(TrackTriggerClustersStubs*TrackTriggerAssociatorClustersStubs*ProducerDTC) # Customisation to enable TTTracks in geometry D41 and later (corresponding to phase2_trackerV14 or later). Includes the HGCAL L1 trigger _tttracks_l1tracktrigger = L1TrackTrigger.copy() diff --git a/DataFormats/L1TrackTrigger/interface/TTBV.h b/DataFormats/L1TrackTrigger/interface/TTBV.h index f18f96cacd900..c7ceb3cda0b08 100644 --- a/DataFormats/L1TrackTrigger/interface/TTBV.h +++ b/DataFormats/L1TrackTrigger/interface/TTBV.h @@ -20,16 +20,13 @@ class TTBV { public: static constexpr int S_ = 64; // Frame width of emp infrastructure f/w, max number of bits a TTBV can handle - private: bool twos_; // Two's complement (true) or binary (false) int size_; // number or bits std::bitset bs_; // underlying storage - public: // constructor: default TTBV() : twos_(false), size_(0), bs_() {} - // constructor: double precision (IEEE 754); from most to least significant bit: 1 bit sign + 11 bit binary exponent + 52 bit binary mantisse TTBV(const double d) : twos_(false), size_(S_) { int index(0); @@ -42,14 +39,17 @@ class TTBV { } // constructor: unsigned int value - TTBV(unsigned long long int value, int size) : twos_(false), size_(size), bs_(value) {} + TTBV(unsigned long long int value, int size) : twos_(false), size_(size), bs_(value) { checkU(value); } // constructor: int value TTBV(int value, int size, bool twos = false) - : twos_(twos), size_(size), bs_((!twos || value >= 0) ? value : value + iMax()) {} + : twos_(twos), size_(size), bs_((!twos || value >= 0) ? value : value + iMax()) { + checkI(value); + } // constructor: double value + precision, biased (floor) representation - TTBV(double value, double base, int size, bool twos = false) : TTBV((int)std::floor(value / base), size, twos) {} + TTBV(double value, double base, int size, bool twos = false) + : TTBV((int)std::floor(value / base + 1.e-12), size, twos) {} // constructor: string TTBV(const std::string& str, bool twos = false) : twos_(twos), size_(str.size()), bs_(str) {} @@ -70,10 +70,15 @@ class TTBV { // underlying storage const std::bitset& bs() const { return bs_; } - // access: single bit + // access: single bit value bool operator[](int pos) const { return bs_[pos]; } + + // access: single bit reference std::bitset::reference operator[](int pos) { return bs_[pos]; } + // access: single bit value with bounds check + bool test(int pos) const { return bs_.test(pos); } + // access: most significant bit copy bool msb() const { return bs_[size_ - 1]; } @@ -95,31 +100,31 @@ class TTBV { // operator: boolean and TTBV& operator&=(const TTBV& rhs) { - const int m(std::max(size_, rhs.size())); - this->resize(m); - TTBV bv(rhs); - bv.resize(m); - bs_ &= bv.bs_; + bs_ &= rhs.bs_; return *this; } + // operator: boolean and + TTBV operator&&(const TTBV& rhs) { + TTBV copy(*this); + return copy &= rhs; + } + // operator: boolean or TTBV& operator|=(const TTBV& rhs) { - const int m(std::max(size_, rhs.size())); - this->resize(m); - TTBV bv(rhs); - bv.resize(m); - bs_ |= bv.bs_; + bs_ |= rhs.bs_; return *this; } + // operator: boolean or + TTBV operator||(const TTBV& rhs) { + TTBV copy(*this); + return copy |= rhs; + } + // operator: boolean xor TTBV& operator^=(const TTBV& rhs) { - const int m(std::max(size_, rhs.size())); - this->resize(m); - TTBV bv(rhs); - bv.resize(m); - bs_ ^= bv.bs_; + bs_ ^= rhs.bs_; return *this; } @@ -242,7 +247,7 @@ class TTBV { bs_.set(n, msb); size_ = size; } else if (size < size_ && size > 0) { - this->operator<<=(size - size_); + this->operator<<=(size_ - size); if (twos_) this->msb() = msb; } @@ -281,11 +286,18 @@ class TTBV { // maniplulation and conversion: extracts range based to int reinterpret sign and removes these bits int extract(int size, bool twos = false) { - double val = this->val(size, 0, twos); + int val = this->val(size, 0, twos); this->operator>>=(size); return val; } + // maniplulation and conversion: extracts bool and removes this bit + bool extract() { + bool val = bs_[0]; + this->operator>>=(1); + return val; + } + // manipulation: extracts slice and removes these bits TTBV slice(int size, bool twos = false) { TTBV ttBV(*this, size, 0, twos); @@ -310,6 +322,14 @@ class TTBV { return size_; } + // position of least significant '1' or '0' in range [begin, end) + int plEncode(int begin, int end, bool b = true) const { + for (int e = begin; e < end; e++) + if (bs_.test(e) == b) + return e; + return size_; + } + // position of most significant '1' or '0' int pmEncode(bool b = true) const { for (int e = size_ - 1; e > -1; e--) @@ -318,6 +338,14 @@ class TTBV { return size_; } + // position of most significant '1' or '0' in range [begin, end) + int pmEncode(int begin, int end, bool b = true) const { + for (int e = end - 1; e >= begin; e--) + if (bs_.test(e) == b) + return e; + return end; + } + // position for n'th '1' or '0' counted from least to most significant bit int encode(int n, bool b = true) const { int sum(0); @@ -344,17 +372,58 @@ class TTBV { private: // look up table initializer for powers of 2 - constexpr std::array powersOfTwo() const { - std::array lut = {}; - for (int i = 0; i < S_; i++) + constexpr std::array powersOfTwo() const { + std::array lut = {}; + for (int i = 0; i <= S_; i++) lut[i] = std::pow(2, i); return lut; } // returns 2 ** size_ - unsigned long long int iMax() const { - static const std::array lut = powersOfTwo(); - return lut[size_]; + double iMax() const { + static const std::array lut = powersOfTwo(); + return std::round(lut[size_]); + } + + // check if value fits into binary BV + void checkU(unsigned long long int value) { + if (size_ == 0) + return; + if (value < iMax()) + return; + cms::Exception exception("RunTimeError."); + exception << "Value " << value << " does not fit into a " << size_ << "b binary."; + exception.addContext("TTBV::checkU"); + throw exception; + } + + // check if value fits into twos's complement BV + void checkT(int value) { + if (size_ == 0) + return; + static const std::array lut = powersOfTwo(); + auto abs = [](int val) { return val < 0 ? std::abs(val) - 1 : val; }; + if (abs(value) < std::round(lut[size_ - 1])) + return; + cms::Exception exception("RunTimeError."); + exception << "Value " << value << " does not fit into a " << size_ << "b two's complement."; + exception.addContext("TTBV::checkT"); + throw exception; + } + + // check if value fits into twos complement / binary BV + void checkI(int value) { + if (size_ == 0) + return; + if (twos_) + checkT(value); + else if (value < 0) { + cms::Exception exception("RunTimeError."); + exception << size_ << "b Binary TTBV constructor called with negative value (" << value << ")."; + exception.addContext("TTBV::checkI"); + throw exception; + } else + checkU(value); } }; diff --git a/DataFormats/L1TrackTrigger/src/classes_def.xml b/DataFormats/L1TrackTrigger/src/classes_def.xml index 800cf2528af7f..5f23a6744b6a4 100644 --- a/DataFormats/L1TrackTrigger/src/classes_def.xml +++ b/DataFormats/L1TrackTrigger/src/classes_def.xml @@ -51,5 +51,7 @@ + + diff --git a/L1Trigger/L1TTrackMatch/test/L1TrackObjectNtupleMaker_cfg.py b/L1Trigger/L1TTrackMatch/test/L1TrackObjectNtupleMaker_cfg.py index 0ad414a072c7b..564c5b4caf3d7 100644 --- a/L1Trigger/L1TTrackMatch/test/L1TrackObjectNtupleMaker_cfg.py +++ b/L1Trigger/L1TTrackMatch/test/L1TrackObjectNtupleMaker_cfg.py @@ -80,8 +80,8 @@ # DTC emulation -process.load('L1Trigger.TrackerDTC.ProducerED_cff') -process.dtc = cms.Path(process.TrackerDTCProducer) +process.load('L1Trigger.TrackerDTC.DTC_cff') +process.dtc = cms.Path(process.ProducerDTC) process.load("L1Trigger.TrackFindingTracklet.L1HybridEmulationTracks_cff") process.load("L1Trigger.L1TTrackMatch.l1tTrackSelectionProducer_cfi") diff --git a/L1Trigger/Phase2L1ParticleFlow/test/make_l1ctLayer1_dumpFiles_fromRAW_cfg.py b/L1Trigger/Phase2L1ParticleFlow/test/make_l1ctLayer1_dumpFiles_fromRAW_cfg.py index c0a15ebcdff85..5b9f89f76d3a2 100644 --- a/L1Trigger/Phase2L1ParticleFlow/test/make_l1ctLayer1_dumpFiles_fromRAW_cfg.py +++ b/L1Trigger/Phase2L1ParticleFlow/test/make_l1ctLayer1_dumpFiles_fromRAW_cfg.py @@ -33,8 +33,7 @@ process.load('Configuration.StandardSequences.SimL1Emulator_cff') process.load('L1Trigger.TrackTrigger.TrackTrigger_cff') process.load("L1Trigger.TrackFindingTracklet.L1HybridEmulationTracks_cff") -process.load("L1Trigger.TrackerDTC.ProducerES_cff") -process.load("L1Trigger.TrackerDTC.ProducerED_cff") +process.load("L1Trigger.TrackerDTC.DTC_cff") process.load("RecoVertex.BeamSpotProducer.BeamSpot_cfi") process.l1tLayer1Barrel9 = process.l1tLayer1Barrel.clone() @@ -52,7 +51,7 @@ process.PFInputsTask = cms.Task( process.TTClustersFromPhase2TrackerDigis, process.TTStubsFromPhase2TrackerDigis, - process.TrackerDTCProducer, + process.ProducerDTC, process.offlineBeamSpot, process.l1tTTTracksFromTrackletEmulation, process.SimL1EmulatorTask diff --git a/L1Trigger/Phase2L1ParticleFlow/test/make_l1ct_patternFiles_fromRAW_cfg.py b/L1Trigger/Phase2L1ParticleFlow/test/make_l1ct_patternFiles_fromRAW_cfg.py index c519341d5963e..7ddaf341bbce9 100644 --- a/L1Trigger/Phase2L1ParticleFlow/test/make_l1ct_patternFiles_fromRAW_cfg.py +++ b/L1Trigger/Phase2L1ParticleFlow/test/make_l1ct_patternFiles_fromRAW_cfg.py @@ -33,8 +33,7 @@ process.load('Configuration.StandardSequences.SimL1Emulator_cff') process.load('L1Trigger.TrackTrigger.TrackTrigger_cff') process.load("L1Trigger.TrackFindingTracklet.L1HybridEmulationTracks_cff") -process.load("L1Trigger.TrackerDTC.ProducerES_cff") -process.load("L1Trigger.TrackerDTC.ProducerED_cff") +process.load("L1Trigger.TrackerDTC.DTC_cff") process.load("RecoVertex.BeamSpotProducer.BeamSpot_cfi") from L1Trigger.Phase2L1ParticleFlow.l1tSeedConePFJetProducer_cfi import l1tSeedConePFJetEmulatorProducer @@ -69,7 +68,7 @@ process.PFInputsTask = cms.Task( process.TTClustersFromPhase2TrackerDigis, process.TTStubsFromPhase2TrackerDigis, - process.TrackerDTCProducer, + process.ProducerDTC, process.offlineBeamSpot, process.l1tTTTracksFromTrackletEmulation, process.SimL1EmulatorTask diff --git a/L1Trigger/TrackFindingTMTT/interface/L1track3D.h b/L1Trigger/TrackFindingTMTT/interface/L1track3D.h index 3b3b7dc77e3a2..530d67aa6eed5 100644 --- a/L1Trigger/TrackFindingTMTT/interface/L1track3D.h +++ b/L1Trigger/TrackFindingTMTT/interface/L1track3D.h @@ -72,6 +72,35 @@ namespace tmtt { : L1track3D( settings, stubs, cellLocationHT, helixRphi, helixRz, 0.0, iPhiSec, iEtaReg, optoLinkID, mergedHTcell) {} + // KF emulator: constructor + L1track3D(Settings* settings, + std::vector stubs, + double qOverPt, + double phi0, + double z0, + double tanLambda, + double helixD0, + int iPhiSec, + int iEtaReg) + : settings_(settings), + stubs_(stubs), + stubsConst_(std::vector()), + bestStubs_(std::unordered_set()), + nLayers_(0), + cellLocationHT_(0, 0), + helixRphi_(qOverPt, phi0), + helixRz_(z0, tanLambda), + helixD0_(helixD0), + iPhiSec_(iPhiSec), + iEtaReg_(iEtaReg), + optoLinkID_(0), + mergedHTcell_(false), + seedLayerType_(TrackletSeedType()), + seedPS_(0), + matchedTP_(nullptr), + matchedStubs_(std::vector()), + nMatchedLayers_(0) {} + ~L1track3D() override = default; //--- Set/get optional info for tracklet tracks. diff --git a/L1Trigger/TrackFindingTMTT/interface/Stub.h b/L1Trigger/TrackFindingTMTT/interface/Stub.h index 526e8944d63b7..bbbcc6980f678 100644 --- a/L1Trigger/TrackFindingTMTT/interface/Stub.h +++ b/L1Trigger/TrackFindingTMTT/interface/Stub.h @@ -69,6 +69,19 @@ namespace tmtt { const DegradeBend* degradeBend, const StubKiller* stubKiller); + // KF emualtor: stub constructor + Stub(const TTStubRef& ttStubRef, + double r, + double phi, + double z, + int layerId, + int layerIdReduced, + double stripPitch, + double stripLength, + bool psModule, + bool barrel, + bool tiltedBarrel); + bool operator==(const Stub& stubOther) { return (this->index() == stubOther.index()); } // Return reference to original TTStub. diff --git a/L1Trigger/TrackFindingTMTT/python/TMTrackProducer_Defaults_cfi.py b/L1Trigger/TrackFindingTMTT/python/TMTrackProducer_Defaults_cfi.py index 83d6d3763bdda..fc9ed066f1287 100644 --- a/L1Trigger/TrackFindingTMTT/python/TMTrackProducer_Defaults_cfi.py +++ b/L1Trigger/TrackFindingTMTT/python/TMTrackProducer_Defaults_cfi.py @@ -358,7 +358,7 @@ #--- Options for Kalman filter track fitters --- # # Larger number has more debug printout. "1" is useful for understanding why tracks are lost, best combined with TrackFitCheat=True. - KalmanDebugLevel = cms.uint32(0), + KalmanDebugLevel = cms.uint32(2), # Fit will reject fitted tracks unless it can assign at least this number of stubs to them. KalmanMinNumStubs = cms.uint32(4), # Fit will attempt to add up to this nummber of stubs to each fitted tracks, but won't bother adding more. diff --git a/L1Trigger/TrackFindingTMTT/src/Settings.cc b/L1Trigger/TrackFindingTMTT/src/Settings.cc index 8f54fec900240..d1a246cb51e74 100644 --- a/L1Trigger/TrackFindingTMTT/src/Settings.cc +++ b/L1Trigger/TrackFindingTMTT/src/Settings.cc @@ -38,8 +38,8 @@ namespace tmtt { stubMatchStrict_(false), // Kalman filter track fit cfg - kalmanDebugLevel_(0), - //kalmanDebugLevel_(2), // Good for debugging + //kalmanDebugLevel_(0), + kalmanDebugLevel_(2), // Good for debugging kalmanMinNumStubs_(4), kalmanMaxNumStubs_(6), kalmanAddBeamConstr_(false), // Apply post-fit beam-spot constraint to 5-param fit diff --git a/L1Trigger/TrackFindingTMTT/src/Stub.cc b/L1Trigger/TrackFindingTMTT/src/Stub.cc index 094a36d9bbc92..3a61edd624563 100644 --- a/L1Trigger/TrackFindingTMTT/src/Stub.cc +++ b/L1Trigger/TrackFindingTMTT/src/Stub.cc @@ -158,6 +158,53 @@ namespace tmtt { } } + // KF emualtor: stub constructor + Stub::Stub(const TTStubRef& ttStubRef, + double r, + double phi, + double z, + int layerId, + int layerIdReduced, + double stripPitch, + double stripLength, + bool psModule, + bool barrel, + bool tiltedBarrel) + : ttStubRef_(ttStubRef), + settings_(nullptr), + index_in_vStubs_(0), + phi_(phi), + r_(r), + z_(z), + bend_(0.), + dphiOverBend_(0.), + min_qOverPt_bin_(0), + max_qOverPt_bin_(0), + localU_cluster_({{0., 0.}}), + localV_cluster_({{0., 0.}}), + iphi_(0), + alpha_(0.), + frontendPass_(false), + stubFailedDegradeWindow_(false), + bendInFrontend_(0), + numMergedBend_(0), + assocTP_(nullptr), + assocTPs_(set()), + assocTPofCluster_({{nullptr, nullptr}}), + digitalStub_(nullptr), + lastDigiStep_(DigiStage()), + digitizeWarningsOn_(false), + trackerModule_(nullptr), + degradeBend_(nullptr), + layerId_(layerId), + layerIdReduced_(layerIdReduced), + stripPitch_(stripPitch), + stripLength_(stripLength), + nStrips_(0), + psModule_(psModule), + barrel_(barrel), + tiltedBarrel_(tiltedBarrel) {} + //=== Calculate bin range along q/Pt axis of r-phi Hough transform array consistent with bend of this stub. void Stub::calcQoverPtrange() { diff --git a/L1Trigger/TrackFindingTracklet/interface/ChannelAssignment.h b/L1Trigger/TrackFindingTracklet/interface/ChannelAssignment.h index c4d644d41d56b..d3cab07b1e3a1 100644 --- a/L1Trigger/TrackFindingTracklet/interface/ChannelAssignment.h +++ b/L1Trigger/TrackFindingTracklet/interface/ChannelAssignment.h @@ -22,28 +22,24 @@ namespace trklet { ChannelAssignment() {} ChannelAssignment(const edm::ParameterSet& iConfig, const tt::Setup* setup); ~ChannelAssignment() {} + // helper class to store configurations + const tt::Setup* setup() const { return setup_; } // returns channelId of given TTTrackRef from TrackBuilder int channelId(const TTTrackRef& ttTrackRef) const; // number of used TB channels for tracks int numChannelsTrack() const { return numChannelsTrack_; } // number of used TB channels for stubs int numChannelsStub() const { return numChannelsStub_; } - // number of bits used to represent layer id [barrel: 0-5, discs: 6-10] - int widthLayerId() const { return widthLayerId_; } + // number of layers per rtack + int tmNumLayers() const { return tmNumLayers_; } // number of bits used to represent stub id for projected stubs - int widthStubId() const { return widthStubId_; } - // number of bits used to represent stub id for seed stubs - int widthSeedStubId() const { return widthSeedStubId_; } - // number of bits used to distinguish between tilted and untilded barrel modules or 2S and PS endcap modules - int widthPSTilt() const { return widthPSTilt_; } - // depth of fifos within systolic array - int depthMemory() const { return depthMemory_; } + int tmWidthStubId() const { return tmWidthStubId_; } + // + int tmWidthCot() const { return tmWidthCot_; } // number of comparison modules used in each DR node int numComparisonModules() const { return numComparisonModules_; } // min number of shared stubs to identify duplicates int minIdenticalStubs() const { return minIdenticalStubs_; } - // number of DR nodes - int numNodesDR() const { return numNodesDR_; } // number of used seed types in tracklet algorithm int numSeedTypes() const { return numSeedTypes_; } // sets layerId (0-7 in sequence the seed type projects to) of given TTStubRef and seedType, returns false if seeed stub @@ -66,34 +62,24 @@ namespace trklet { int channelId(int seedType, int layerId) const; // max number of seeding layers int numSeedingLayers() const { return numSeedingLayers_; } - // return DR node for given ttTrackRef - int nodeDR(const TTTrackRef& ttTrackRef) const; private: // helper class to store configurations const tt::Setup* setup_; - // DRin parameter - edm::ParameterSet pSetDRin_; - // number of bits used to represent layer id [barrel: 0-5, discs: 6-10] - int widthLayerId_; + // TM parameter + edm::ParameterSet pSetTM_; + // number of layers per rtack + int tmNumLayers_; // number of bits used to represent stub id for projected stubs - int widthStubId_; - // number of bits used to represent stub id for seed stubs - int widthSeedStubId_; - // number of bits used to distinguish between tilted and untilded barrel modules or 2S and PS endcap modules - int widthPSTilt_; - // depth of fifos within systolic array - int depthMemory_; - // positive pt Boundaries in GeV (symmetric negatives are assumed), first boundary is pt cut, last boundary is infinity, defining ot bins used by DR - std::vector ptBoundaries_; - // DRin parameter + int tmWidthStubId_; + // + int tmWidthCot_; + // DR parameter edm::ParameterSet pSetDR_; // number of comparison modules used in each DR node int numComparisonModules_; // min number of shared stubs to identify duplicates [default: 3] int minIdenticalStubs_; - // number of DR nodes - int numNodesDR_; // seed type names std::vector seedTypeNames_; // number of used seed types in tracklet algorithm diff --git a/L1Trigger/TrackFindingTracklet/interface/DataFormats.h b/L1Trigger/TrackFindingTracklet/interface/DataFormats.h new file mode 100644 index 0000000000000..24a6e7ca2d1b4 --- /dev/null +++ b/L1Trigger/TrackFindingTracklet/interface/DataFormats.h @@ -0,0 +1,492 @@ +#ifndef L1Trigger_TrackFindingTracklet_DataFormats_h +#define L1Trigger_TrackFindingTracklet_DataFormats_h + +/*---------------------------------------------------------------------- +Classes to calculate and provide dataformats used by Hybrid emulator +enabling automated conversions from frames to stubs/tracks and vice versa +In data members of classes Stub* & Track* below, the variables describing +stubs/tracks are stored both in digitial format as a 64b word in frame_, +and in undigitized format in an std::tuple. (This saves CPU) +----------------------------------------------------------------------*/ + +#include "FWCore/Framework/interface/data_default_record_trait.h" +#include "L1Trigger/TrackFindingTracklet/interface/DataFormatsRcd.h" +#include "L1Trigger/TrackFindingTracklet/interface/ChannelAssignment.h" +#include "L1Trigger/TrackTrigger/interface/Setup.h" +#include "DataFormats/L1TrackTrigger/interface/TTBV.h" + +#include +#include +#include +#include +#include +#include + +namespace trklet { + + // hybrid processes + enum class Process { begin, tm = begin, dr, kf, tfp, end, x }; + // hybrid variables + enum class Variable { begin, stubId = begin, r, phi, z, dPhi, dZ, inv2R, phiT, cot, zT, end, x }; + // hybrid process order + constexpr std::initializer_list Processes = {Process::tm, Process::dr, Process::kf, Process::tfp}; + // conversion: Process to int + inline constexpr int operator+(Process p) { return static_cast(p); } + // conversion: Variable to int + inline constexpr int operator+(Variable v) { return static_cast(v); } + // increment of Process + inline constexpr Process operator++(Process p) { return Process(+p + 1); } + // increment of Variable + inline constexpr Variable operator++(Variable v) { return Variable(+v + 1); } + + //Base class representing format of a variable + class DataFormat { + public: + DataFormat(bool twos, bool biased = true) : twos_(twos), width_(0), base_(1.), range_(0.) {} + DataFormat(bool twos, int width, double base, double range) + : twos_(twos), width_(width), base_(base), range_(range) {} + DataFormat() {} + ~DataFormat() {} + // converts int to bitvector + TTBV ttBV(int i) const { return TTBV(i, width_, twos_); } + // converts double to bitvector + TTBV ttBV(double d) const { return TTBV(d, base_, width_, twos_); } + // extracts int from bitvector, removing these bits from bitvector + void extract(TTBV& in, int& out) const { out = in.extract(width_, twos_); } + // extracts double from bitvector, removing these bits from bitvector + void extract(TTBV& in, double& out) const { out = in.extract(base_, width_, twos_); } + // extracts double from bitvector, removing these bits from bitvector + void extract(TTBV& in, TTBV& out) const { out = in.slice(width_, twos_); } + // extracts bool from bitvector, removing these bits from bitvector + void extract(TTBV& in, bool& out) const { out = in.extract(); } + // attaches integer to bitvector + void attach(const int i, TTBV& ttBV) const { ttBV += TTBV(i, width_, twos_); } + // attaches double to bitvector + void attach(const double d, TTBV& ttBV) const { ttBV += TTBV(d, base_, width_, twos_); } + // attaches bitvector to bitvector + void attach(const TTBV& bv, TTBV& ttBV) const { ttBV += bv; } + // converts int to double + double floating(int i) const { return (i + .5) * base_; } + // converts double to int + int integer(double d) const { return std::floor(d / base_ + 1.e-12); } + // converts double to int and back to double + double digi(double d) const { return floating(integer(d)); } + // converts binary integer value to twos complement integer value + int toSigned(int i) const { return i - std::pow(2, width_) / 2; } + // converts twos complement integer value to binary integer value + int toUnsigned(int i) const { return i + std::pow(2, width_) / 2; } + // converts floating point value to binary integer value + int toUnsigned(double d) const { return this->integer(d) + std::pow(2, width_) / 2; } + // biggest representable floating point value + //double limit() const { return (range_ - base_) / (twos_ ? 2. : 1.); } + // returns false if data format would oferflow for this double value + bool inRange(double d, bool digi = true) const { + const double range = digi ? base_ * pow(2, width_) : range_; + return d >= -range / 2. && d < range / 2.; + } + // returns false if data format would oferflow for this int value + bool inRange(int i) const { return inRange(floating(i)); } + // true if twos'complement or false if binary representation is chosen + bool twos() const { return twos_; } + // number of used bits + int width() const { return width_; } + // precision + double base() const { return base_; } + // covered range + double range() const { return range_; } + + protected: + // true if twos'complement or false if binary representation is chosen + bool twos_; + // number of used bits + int width_; + // precision + double base_; + // covered range + double range_; + }; + + // class representing format of a specific variable + template + class Format : public DataFormat { + public: + Format(const ChannelAssignment* ca); + ~Format() {} + }; + + template <> + Format::Format(const ChannelAssignment* ca); + template <> + Format::Format(const ChannelAssignment* ca); + template <> + Format::Format(const ChannelAssignment* ca); + template <> + Format::Format(const ChannelAssignment* ca); + + template <> + Format::Format(const ChannelAssignment* ca); + template <> + Format::Format(const ChannelAssignment* ca); + template <> + Format::Format(const ChannelAssignment* ca); + template <> + Format::Format(const ChannelAssignment* ca); + + template <> + Format::Format(const ChannelAssignment* ca); + template <> + Format::Format(const ChannelAssignment* ca); + template <> + Format::Format(const ChannelAssignment* ca); + template <> + Format::Format(const ChannelAssignment* ca); + template <> + Format::Format(const ChannelAssignment* ca); + template <> + Format::Format(const ChannelAssignment* ca); + + template <> + Format::Format(const ChannelAssignment* ca); + template <> + Format::Format(const ChannelAssignment* ca); + template <> + Format::Format(const ChannelAssignment* ca); + template <> + Format::Format(const ChannelAssignment* ca); + + /*! \class trklet::DataFormats + * \brief Class to calculate and provide dataformats used by Hybrid emulator + * \author Thomas Schuh + * \date 2024, Sep + */ + class DataFormats { + private: + // variable flavour mapping, Each row below declares which processing steps use the variable named in the comment at the end of the row + static constexpr std::array, +Variable::end> config_ = {{ + // Process::tm Process::dr Process::kf Process::tfp + {{Process::tm, Process::x, Process::x, Process::x}}, // Variable::stubId + {{Process::tm, Process::tm, Process::tm, Process::x}}, // Variable::r + {{Process::tm, Process::tm, Process::tm, Process::x}}, // Variable::phi + {{Process::tm, Process::tm, Process::tm, Process::x}}, // Variable::z + {{Process::tm, Process::tm, Process::tm, Process::x}}, // Variable::dPhi + {{Process::tm, Process::tm, Process::tm, Process::x}}, // Variable::dZ + {{Process::tm, Process::tm, Process::kf, Process::tfp}}, // Variable::inv2R + {{Process::tm, Process::tm, Process::kf, Process::tfp}}, // Variable::phiT + {{Process::tm, Process::tm, Process::kf, Process::tfp}}, // Variable::cot + {{Process::tm, Process::tm, Process::kf, Process::tfp}} // Variable::zT + }}; + // stub word assembly, shows which stub variables are used by each process + static constexpr std::array, +Process::end> stubs_ = {{ + {Variable::stubId, Variable::r, Variable::phi, Variable::z}, // Process::tm + {Variable::r, Variable::phi, Variable::z, Variable::dPhi, Variable::dZ}, // Process::dr + {Variable::r, Variable::phi, Variable::z, Variable::dPhi, Variable::dZ}, // Process::kf + {} // Process::tfp + }}; + // track word assembly, shows which track variables are used by each process + static constexpr std::array, +Process::end> tracks_ = {{ + {Variable::inv2R, Variable::phiT, Variable::zT}, // Process::tm + {Variable::inv2R, Variable::phiT, Variable::zT}, // Process::dr + {Variable::inv2R, Variable::phiT, Variable::cot, Variable::zT}, // Process::kf + {} // Process::tfp + }}; + + public: + DataFormats(); + DataFormats(const ChannelAssignment* ca); + ~DataFormats() {} + // converts bits to ntuple of variables + template + void convertStub(Process p, const tt::Frame& bv, std::tuple& data) const { + TTBV ttBV(bv); + extractStub(p, ttBV, data); + } + // converts ntuple of variables to bits + template + void convertStub(Process p, const std::tuple& data, tt::Frame& bv) const { + TTBV ttBV(1, numUnusedBitsStubs_[+p]); + attachStub(p, data, ttBV); + bv = ttBV.bs(); + } + // converts bits to ntuple of variables + template + void convertTrack(Process p, const tt::Frame& bv, std::tuple& data) const { + TTBV ttBV(bv); + extractTrack(p, ttBV, data); + } + // converts ntuple of variables to bits + template + void convertTrack(Process p, const std::tuple& data, tt::Frame& bv) const { + TTBV ttBV(1, numUnusedBitsTracks_[+p]); + attachTrack(p, data, ttBV); + bv = ttBV.bs(); + } + // access to run-time constants + const tt::Setup* setup() const { return channelAssignment_->setup(); } + // number of bits being used for specific variable flavour + int width(Variable v, Process p) const { return formats_[+v][+p]->width(); } + // precision being used for specific variable flavour + double base(Variable v, Process p) const { return formats_[+v][+p]->base(); } + // covered range for specific variable flavour + double range(Variable v, Process p) const { return formats_[+v][+p]->range(); } + // access to spedific format + const DataFormat& format(Variable v, Process p) const { return *formats_[+v][+p]; } + + private: + // number of unique data formats + int numDataFormats_; + // method to count number of unique data formats + template + void countFormats(); + // constructs data formats of all unique used variables and flavours + template + void fillDataFormats(); + // helper (loop) data formats of all unique used variables and flavours + template + void fillFormats(); + // helper (loop) to convert bits to ntuple of variables + template + void extractStub(Process p, TTBV& ttBV, std::tuple& data) const { + Variable v = *std::next(stubs_[+p].begin(), sizeof...(Ts) - 1 - it); + formats_[+v][+p]->extract(ttBV, std::get(data)); + if constexpr (it + 1 != sizeof...(Ts)) + extractStub(p, ttBV, data); + } + // helper (loop) to convert bits to ntuple of variables + template + void extractTrack(Process p, TTBV& ttBV, std::tuple& data) const { + Variable v = *std::next(tracks_[+p].begin(), sizeof...(Ts) - 1 - it); + formats_[+v][+p]->extract(ttBV, std::get(data)); + if constexpr (it + 1 != sizeof...(Ts)) + extractTrack(p, ttBV, data); + } + // helper (loop) to convert ntuple of variables to bits + template + void attachStub(Process p, const std::tuple& data, TTBV& ttBV) const { + Variable v = *std::next(stubs_[+p].begin(), it); + formats_[+v][+p]->attach(std::get(data), ttBV); + if constexpr (it + 1 != sizeof...(Ts)) + attachStub(p, data, ttBV); + } + // helper (loop) to convert ntuple of variables to bits + template + void attachTrack(Process p, const std::tuple& data, TTBV& ttBV) const { + Variable v = *std::next(tracks_[+p].begin(), it); + formats_[+v][+p]->attach(std::get(data), ttBV); + if constexpr (it + 1 != sizeof...(Ts)) + attachTrack(p, data, ttBV); + } + // configuration during construction + edm::ParameterSet iConfig_; + // stored run-time constants + const ChannelAssignment* channelAssignment_; + // collection of unique formats + std::vector dataFormats_; + // variable flavour mapping + std::vector> formats_; + // number of unused frame bits for a all Stub flavours + std::vector numUnusedBitsStubs_; + // number of unused frame bits for a all Track flavours + std::vector numUnusedBitsTracks_; + }; + + // base class to represent stubs + template + class Stub { + public: + // construct Stub from Frame + Stub(const tt::FrameStub& fs, const DataFormats* df, Process p) : dataFormats_(df), p_(p), frame_(fs) { + dataFormats_->convertStub(p_, frame_.second, data_); + } + template + // construct Stub from other Stub + Stub(const Stub& stub, Ts... data) + : dataFormats_(stub.dataFormats()), p_(++stub.p()), frame_(stub.frame()), data_(data...) { + dataFormats_->convertStub(p_, data_, frame_.second); + } + // construct Stub from TTStubRef + Stub(const TTStubRef& ttStubRef, const DataFormats* df, Process p, Ts... data) + : dataFormats_(df), p_(p), frame_(ttStubRef, tt::Frame()), data_(data...) { + dataFormats_->convertStub(p_, data_, frame_.second); + } + Stub() {} + ~Stub() {} + // true if frame valid, false if gap in data stream + explicit operator bool() const { return frame_.first.isNonnull(); } + // access to DataFormats + const DataFormats* dataFormats() const { return dataFormats_; } + // stub flavour + Process p() const { return p_; } + // acess to frame + const tt::FrameStub& frame() const { return frame_; } + + protected: + // all dataformats + const DataFormats* dataFormats_; + // stub flavour + Process p_; + // underlying TTStubRef and bitvector + tt::FrameStub frame_; + // ntuple of variables this stub is assemled of + std::tuple data_; + }; + + // class to represent stubs generated by process TrackMulitplexer + class StubTM : public Stub { + public: + // construct StubTM from Frame + StubTM(const tt::FrameStub& fs, const DataFormats* df) : Stub(fs, df, Process::tm) {} + // construct StubTM from TTStubRef + StubTM(const TTStubRef& ttStubRef, const DataFormats* df, int stubId, double r, double phi, double z) + : Stub(ttStubRef, df, Process::tm, stubId, r, phi, z) {} + ~StubTM() {} + // stub Id + int stubId() const { return std::get<0>(data_); } + // stub radius wrt chosenRofPhi + double r() const { return std::get<1>(data_); } + // stub phi wrt processing nonant centre + double phi() const { return std::get<2>(data_); } + // stub z + double z() const { return std::get<3>(data_); } + }; + + // class to represent stubs generated by process DuplicateRemoval + class StubDR : public Stub { + public: + // construct StubDR from Frame + StubDR(const tt::FrameStub& fs, const DataFormats* df) : Stub(fs, df, Process::dr) {} + // construct StubDR from StubTM + StubDR(const StubTM& stub, double r, double phi, double z, double dPhi, double dZ) + : Stub(stub, r, phi, z, dPhi, dZ) {} + ~StubDR() {} + // stub radius wrt chosenRofPhi + double r() const { return std::get<0>(data_); } + // stub phi wrt phi sector centre + double phi() const { return std::get<1>(data_); } + // stub z residual wrt eta sector + double z() const { return std::get<2>(data_); } + // stub phi uncertainty + double dPhi() const { return std::get<3>(data_); } + // stub z uncertainty + double dZ() const { return std::get<4>(data_); } + }; + + // class to represent stubs generated by process KalmanFilter + class StubKF : public Stub { + public: + // construct StubKF from Frame + StubKF(const tt::FrameStub& fs, const DataFormats* df) : Stub(fs, df, Process::kf) {} + // construct StubKF from StubDR + StubKF(const StubDR& stub, double r, double phi, double z, double dPhi, double dZ) + : Stub(stub, r, phi, z, dPhi, dZ) {} + ~StubKF() {} + // stub radius wrt chosenRofPhi + double r() const { return std::get<0>(data_); }; + // stub phi residual wrt track parameter + double phi() const { return std::get<1>(data_); }; + // stub z residual wrt eta sector + double z() const { return std::get<2>(data_); }; + // stub phi uncertainty + double dPhi() const { return std::get<3>(data_); } + // stub z uncertainty + double dZ() const { return std::get<4>(data_); } + }; + + // base class to represent tracks + template + class Track { + public: + // construct Track from Frame + Track(const tt::FrameTrack& ft, const DataFormats* df, Process p) : dataFormats_(df), p_(p), frame_(ft) { + dataFormats_->convertTrack(p_, frame_.second, data_); + } + // construct Track from TTTrackRef + Track(const TTTrackRef& ttTrackRef, const DataFormats* df, Process p, Ts... data) + : dataFormats_(df), p_(p), frame_(ttTrackRef, tt::Frame()), data_(data...) { + dataFormats_->convertTrack(p_, data_, frame_.second); + } + // construct Track from other Track + template + Track(const Track& track, Ts... data) + : dataFormats_(track.dataFormats()), p_(++track.p()), frame_(track.frame()), data_(data...) { + dataFormats_->convertTrack(p_, data_, frame_.second); + } + Track() {} + ~Track() {} + // true if frame valid, false if gap in data stream + explicit operator bool() const { return frame_.first.isNonnull(); } + // access to DataFormats + const DataFormats* dataFormats() const { return dataFormats_; } + // track flavour + Process p() const { return p_; } + // acces to frame + const tt::FrameTrack& frame() const { return frame_; } + + protected: + // all data formats + const DataFormats* dataFormats_; + // track flavour + Process p_; + // underlying TTTrackRef and bitvector + tt::FrameTrack frame_; + // ntuple of variables this track is assemled of + std::tuple data_; + }; + + // class to represent tracks generated by process TrackMultiplexer + class TrackTM : public Track { + public: + // construct TrackTM from Frame + TrackTM(const tt::FrameTrack& ft, const DataFormats* df) : Track(ft, df, Process::tm) {} + // construct TrackTM from TTTrack + TrackTM(const TTTrackRef& tTTrackRef, const DataFormats* df, double inv2R, double phiT, double zT) + : Track(tTTrackRef, df, Process::tm, inv2R, phiT, zT) {} + ~TrackTM() {} + // track inv2R + double inv2R() const { return std::get<0>(data_); } + // track phi at radius chosenRofPhi wrt pprocessing centre + double phiT() const { return std::get<1>(data_); } + // track z at radius chosenRofZ + double zT() const { return std::get<2>(data_); } + }; + + // class to represent tracks generated by process DuplicateRemoval + class TrackDR : public Track { + public: + // construct TrackDR from Frame + TrackDR(const tt::FrameTrack& ft, const DataFormats* df) : Track(ft, df, Process::dr) {} + // construct TrackDR from TrackTM + TrackDR(const TrackTM& track) : Track(track, track.inv2R(), track.phiT(), track.zT()) {} + ~TrackDR() {} + // track qOver pt + double inv2R() const { return std::get<0>(data_); } + // track phi at radius chosenRofPhi wrt processing nonant centre + double phiT() const { return std::get<1>(data_); } + // track z at radius chosenRofZ + double zT() const { return std::get<2>(data_); } + }; + + // class to represent tracks generated by process KalmanFilter + class TrackKF : public Track { + public: + // construct TrackKF from Frame + TrackKF(const tt::FrameTrack& ft, const DataFormats* df) : Track(ft, df, Process::kf) {} + // construct TrackKF from TrackDR + TrackKF(const TrackDR& track, double inv2R, double phiT, double cot, double zT) + : Track(track, inv2R, phiT, cot, zT) {} + TrackKF() {} + ~TrackKF() {} + // track inv2R + double inv2R() const { return std::get<0>(data_); } + // track phi at radius 0 wrt processing nonant centre + double phiT() const { return std::get<1>(data_); } + // track cotThea + double cot() const { return std::get<2>(data_); } + // track z at radius 0 + double zT() const { return std::get<3>(data_); } + }; + +} // namespace trklet + +EVENTSETUP_DATA_DEFAULT_RECORD(trklet::DataFormats, trklet::DataFormatsRcd); + +#endif diff --git a/L1Trigger/TrackFindingTracklet/interface/DataFormatsRcd.h b/L1Trigger/TrackFindingTracklet/interface/DataFormatsRcd.h new file mode 100644 index 0000000000000..25f45d0cdb4ec --- /dev/null +++ b/L1Trigger/TrackFindingTracklet/interface/DataFormatsRcd.h @@ -0,0 +1,17 @@ +#ifndef L1Trigger_TrackFindingTracklet_DataFormatsRcd_h +#define L1Trigger_TrackFindingTracklet_DataFormatsRcd_h + +#include "FWCore/Framework/interface/DependentRecordImplementation.h" +#include "L1Trigger/TrackFindingTracklet/interface/ChannelAssignmentRcd.h" +#include "FWCore/Utilities/interface/mplVector.h" + +namespace trklet { + + typedef edm::mpl::Vector RcdsDataFormats; + + // record of trklet::DataFormats + class DataFormatsRcd : public edm::eventsetup::DependentRecordImplementation {}; + +} // namespace trklet + +#endif \ No newline at end of file diff --git a/L1Trigger/TrackFindingTracklet/interface/DR.h b/L1Trigger/TrackFindingTracklet/interface/DuplicateRemoval.h similarity index 56% rename from L1Trigger/TrackFindingTracklet/interface/DR.h rename to L1Trigger/TrackFindingTracklet/interface/DuplicateRemoval.h index b3956fb0744ec..8fbb22a993603 100644 --- a/L1Trigger/TrackFindingTracklet/interface/DR.h +++ b/L1Trigger/TrackFindingTracklet/interface/DuplicateRemoval.h @@ -1,50 +1,50 @@ -#ifndef L1Trigger_TrackFindingTracklet_DR_h -#define L1Trigger_TrackFindingTracklet_DR_h +#ifndef L1Trigger_TrackFindingTracklet_DuplicateRemoval_h +#define L1Trigger_TrackFindingTracklet_DuplicateRemoval_h #include "L1Trigger/TrackTrigger/interface/Setup.h" -#include "L1Trigger/TrackerTFP/interface/DataFormats.h" +#include "L1Trigger/TrackerTFP/interface/LayerEncoding.h" +#include "L1Trigger/TrackFindingTracklet/interface/DataFormats.h" #include "L1Trigger/TrackFindingTracklet/interface/ChannelAssignment.h" #include namespace trklet { - /*! \class trklet::DR + /*! \class trklet::DuplicateRemoval * \brief Class to bit- and clock-accurate emulate duplicate removal * DR identifies duplicates based on pairs of tracks that share stubs in at least 3 layers. - * It keeps the first such track in each pair. + * It keeps the first such track in each pair. The Track order is determined by TrackMultiplexer, + * provided by ProducerTM. * \author Thomas Schuh * \date 2023, Feb */ - class DR { + class DuplicateRemoval { public: - DR(const edm::ParameterSet& iConfig, - const tt::Setup* setup_, - const trackerTFP::DataFormats* dataFormats, - const ChannelAssignment* channelAssignment, - int region); - ~DR() {} + DuplicateRemoval(const edm::ParameterSet& iConfig, + const tt::Setup* setup_, + const trackerTFP::LayerEncoding* layerEncoding, + const DataFormats* dataFormats, + const ChannelAssignment* channelAssignment, + int region); + ~DuplicateRemoval() {} // read in and organize input tracks and stubs void consume(const tt::StreamsTrack& streamsTrack, const tt::StreamsStub& streamsStub); // fill output products - void produce(tt::StreamsStub& accpetedStubs, - tt::StreamsTrack& acceptedTracks, - tt::StreamsStub& lostStubs, - tt::StreamsTrack& lostTracks); + void produce(tt::StreamsTrack& acceptedTracks, tt::StreamsStub& accpetedStubs); private: struct Stub { - Stub(const tt::FrameStub& frame, int stubId, int channel) : frame_(frame), stubId_(stubId), channel_(channel) {} - bool operator==(const Stub& s) const { return s.stubId_ == stubId_; } + Stub(const tt::FrameStub& frame, int stubId, int layer) : frame_(frame), stubId_(stubId), layer_(layer) {} + // output frame tt::FrameStub frame_; // all stubs id int stubId_; // kf layer id - int channel_; + int layer_; }; struct Track { // max number of stubs a track may formed of (we allow only one stub per layer) - static constexpr int max_ = 7; + static constexpr int max_ = 11; Track() { stubs_.reserve(max_); } Track(const tt::FrameTrack& frame, const std::vector& stubs) : frame_(frame), stubs_(stubs) {} tt::FrameTrack frame_; @@ -56,8 +56,10 @@ namespace trklet { bool enableTruncation_; // provides run-time constants const tt::Setup* setup_; + // helper class to encode layer + const trackerTFP::LayerEncoding* layerEncoding_; // provides dataformats - const trackerTFP::DataFormats* dataFormats_; + const DataFormats* dataFormats_; // helper class to assign tracks to channel const ChannelAssignment* channelAssignment_; // processing region (0 - 8) aka processing phi nonant @@ -67,7 +69,9 @@ namespace trklet { // storage of input stubs std::vector stubs_; // h/w liked organized pointer to input tracks - std::vector> input_; + std::vector input_; + // dataformat used to calculate pitch over stubs radius + DataFormat r_; }; } // namespace trklet diff --git a/L1Trigger/TrackFindingTracklet/interface/HitPatternHelper.h b/L1Trigger/TrackFindingTracklet/interface/HitPatternHelper.h index 6b1c81db4f342..20b4b7160c02f 100644 --- a/L1Trigger/TrackFindingTracklet/interface/HitPatternHelper.h +++ b/L1Trigger/TrackFindingTracklet/interface/HitPatternHelper.h @@ -59,11 +59,13 @@ namespace hph { int etaRegion(double z0, double cot, bool useNewKF) const; int digiCot(double cot, int binEta) const; int digiZT(double z0, double cot, int binEta) const; - const std::vector& layerEncoding(int binEta, int binZT, int binCot) const { - return layerEncoding_.layerEncoding(binEta, binZT, binCot); + const std::vector layerEncoding(int binEta, int binZT, int binCot) const { + //return layerEncoding_.layerEncoding(binEta, binZT, binCot); + return std::vector(); } - const std::map& layerEncodingMap(int binEta, int binZT, int binCot) const { - return layerEncoding_.layerEncodingMap(binEta, binZT, binCot); + const std::map layerEncodingMap(int binEta, int binZT, int binCot) const { + //return layerEncoding_.layerEncodingMap(binEta, binZT, binCot); + return std::map(); } private: diff --git a/L1Trigger/TrackFindingTracklet/interface/KFin.h b/L1Trigger/TrackFindingTracklet/interface/KFin.h deleted file mode 100644 index 9408c83e23a38..0000000000000 --- a/L1Trigger/TrackFindingTracklet/interface/KFin.h +++ /dev/null @@ -1,95 +0,0 @@ -#ifndef L1Trigger_TrackFindingTracklet_KFin_h -#define L1Trigger_TrackFindingTracklet_KFin_h - -#include "L1Trigger/TrackTrigger/interface/Setup.h" -#include "L1Trigger/TrackerTFP/interface/DataFormats.h" -#include "L1Trigger/TrackerTFP/interface/LayerEncoding.h" -#include "L1Trigger/TrackFindingTracklet/interface/ChannelAssignment.h" - -#include - -namespace trklet { - - /*! \class trklet::KFin - * \brief Class to emulate the data transformation happening betwwen DR and KF - * \author Thomas Schuh - * \date 2023, Feb - */ - class KFin { - public: - KFin(const edm::ParameterSet& iConfig, - const tt::Setup* setup_, - const trackerTFP::DataFormats* dataFormats, - const trackerTFP::LayerEncoding* layerEncoding, - const ChannelAssignment* channelAssignment, - int region); - ~KFin() {} - // read in and organize input tracks and stubs - void consume(const tt::StreamsTrack& streamsTrack, const tt::StreamsStub& streamsStub); - // fill output products - void produce(tt::StreamsStub& accpetedStubs, - tt::StreamsTrack& acceptedTracks, - tt::StreamsStub& lostStubs, - tt::StreamsTrack& lostTracks); - - private: - // truncates double precision of val into base precision, +1.e-12 restores robustness of addition of 2 digitised values - double digi(double val, double base) const { return (floor(val / base + 1.e-12) + .5) * base; } - struct Stub { - Stub(const TTStubRef& ttStubRef, double r, double phi, double z, int layerId, bool psTilt, int channel) - : ttStubRef_(ttStubRef), r_(r), phi_(phi), z_(z), layerId_(layerId), psTilt_(psTilt), channel_(channel) {} - TTStubRef ttStubRef_; - double r_; - double phi_; - double z_; - int layerId_; - bool psTilt_; - int channel_; - // phi uncertainty * sqrt(12) + additional terms in rad - double dPhi_; - // z uncertainty * sqrt(12) + additional terms in cm - double dZ_; - }; - struct Track { - static constexpr int max_ = 7; - Track() { stubs_.reserve(max_); } - Track(const tt::FrameTrack& frame, - const std::vector& stubs, - double cot, - double zT, - double inv2R, - int sectorEta) - : frame_(frame), stubs_(stubs), cot_(cot), zT_(zT), inv2R_(inv2R), sectorEta_(sectorEta) {} - tt::FrameTrack frame_; - std::vector stubs_; - double cot_; - double zT_; - double inv2R_; - int sectorEta_; - }; - // remove and return first element of deque, returns nullptr if empty - template - T* pop_front(std::deque& ts) const; - // true if truncation is enbaled - bool enableTruncation_; - // provides run-time constants - const tt::Setup* setup_; - // provides dataformats - const trackerTFP::DataFormats* dataFormats_; - // helper class to encode layer - const trackerTFP::LayerEncoding* layerEncoding_; - // helper class to assign tracks to channel - const ChannelAssignment* channelAssignment_; - // processing region (0 - 8) aka processing phi nonant - const int region_; - // storage of input tracks - std::vector tracks_; - // storage of input stubs - std::vector stubs_; - // h/w liked organized pointer to input tracks - std::vector> input_; - }; - -} // namespace trklet - -#endif diff --git a/L1Trigger/TrackFindingTracklet/interface/KalmanFilter.h b/L1Trigger/TrackFindingTracklet/interface/KalmanFilter.h new file mode 100644 index 0000000000000..51eaff05a5585 --- /dev/null +++ b/L1Trigger/TrackFindingTracklet/interface/KalmanFilter.h @@ -0,0 +1,164 @@ +#ifndef L1Trigger_TrackFindingTracklet_KalmanFilter_h +#define L1Trigger_TrackFindingTracklet_KalmanFilter_h + +#include "L1Trigger/TrackTrigger/interface/Setup.h" +#include "L1Trigger/TrackFindingTracklet/interface/DataFormats.h" +#include "L1Trigger/TrackFindingTracklet/interface/KalmanFilterFormats.h" +#include "L1Trigger/TrackFindingTracklet/interface/State.h" +#include "DataFormats/L1TrackTrigger/interface/TTTypes.h" +#include "L1Trigger/TrackFindingTMTT/interface/Settings.h" +#include "L1Trigger/TrackFindingTMTT/interface/KFParamsComb.h" + +#include +#include + +namespace trklet { + + /*! \class trklet::KalmanFilter + * \brief Class to do helix fit to all tracks in a region. + * All variable names & equations come from Fruhwirth KF paper + * http://dx.doi.org/10.1016/0168-9002%2887%2990887-4 + * Summary of variables: + * m = hit position (phi,z) + * V = hit position 2x2 covariance matrix in (phi,z). + * x = helix params + * C = helix params 4x4 covariance matrix + * r = residuals + * H = 2x4 derivative matrix (expected stub position w.r.t. helix params) + * K = KF gain 2x2 matrix + * x' & C': Updated values of x & C after KF iteration + * Boring: F = unit matrix; pxcov = C + * Summary of equations: + * S = H*C (2x4 matrix); St = Transpose S + * R = V + H*C*Ht (KF paper) = V + H*St (used here at simpler): 2x2 matrix + * Rinv = Inverse R + * K = St * Rinv : 2x2 Kalman gain matrix * det(R) + * r = m - H*x + * x' = x + K*r + * C' = C - K*H*C (KF paper) = C - K*S (used here as simpler) + * \author Thomas Schuh + * \date 2024, Sep + */ + class KalmanFilter { + public: + typedef State::Stub Stub; + KalmanFilter(const edm::ParameterSet& iConfig, + const tt::Setup* setup, + const DataFormats* dataFormats, + KalmanFilterFormats* kalmanFilterFormats, + tmtt::Settings* settings, + tmtt::KFParamsComb* tmtt, + int region, + tt::TTTracks& ttTracks); + ~KalmanFilter() {} + // read in and organize input tracks and stubs + void consume(const tt::StreamsTrack& streamsTrack, const tt::StreamsStub& streamsStub); + // fill output products + void produce(tt::StreamsStub& streamsStub, + tt::StreamsTrack& streamsTrack, + int& numAcceptedStates, + int& numLostStates); + + private: + // + struct Track { + Track() {} + Track(int trackId, + int numConsistent, + int numConsistentPS, + double d0, + const TTBV& hitPattern, + const TrackKF& trackKF, + const std::vector& stubsKF) + : trackId_(trackId), + numConsistent_(numConsistent), + numConsistentPS_(numConsistentPS), + d0_(d0), + hitPattern_(hitPattern), + trackKF_(trackKF), + stubsKF_(stubsKF) {} + int trackId_; + int numConsistent_; + int numConsistentPS_; + double d0_; + TTBV hitPattern_; + TrackKF trackKF_; + std::vector stubsKF_; + }; + // call old KF + void simulate(tt::StreamsStub& streamsStub, tt::StreamsTrack& streamsTrack); + // constraints double precision + double digi(VariableKF var, double val) { return kalmanFilterFormats_->format(var).digi(val); } + // + int integer(VariableKF var, double val) { return kalmanFilterFormats_->format(var).integer(val); } + // + void updateRangeActual(VariableKF var, double val) { + return kalmanFilterFormats_->format(var).updateRangeActual(val); + } + // + double base(VariableKF var) { return kalmanFilterFormats_->format(var).base(); } + // + int width(VariableKF var) { return kalmanFilterFormats_->format(var).width(); } + // remove and return first element of deque, returns nullptr if empty + template + T* pop_front(std::deque& ts) const; + // calculates the helix params & their cov. matrix from a pair of stubs + void calcSeeds(); + // Transform States into output products + void conv(tt::StreamsStub& streamsStub, tt::StreamsTrack& streamsTrack); + // adds a layer to states + void addLayer(); + // adds a layer to states to build seeds + void addSeedLayer(); + // Assign next combinatoric (i.e. not first in layer) stub to state + void comb(State*& state); + // apply final cuts + void finalize(); + // best state selection + void accumulator(); + // updates state + void update(State*& state) { use5ParameterFit_ ? update5(state) : update4(state); } + // updates state using 4 paramter fit + void update4(State*& state); + // updates state using 5 parameter fit + void update5(State*& state); + + // true if truncation is enbaled + bool enableTruncation_; + // + bool use5ParameterFit_; + // + bool useSimmulation_; + // + bool useTTStubResiduals_; + // provides run-time constants + const tt::Setup* setup_; + // provides dataformats + const DataFormats* dataFormats_; + // provides dataformats of Kalman filter internals + KalmanFilterFormats* kalmanFilterFormats_; + // + tmtt::Settings* settings_; + // + tmtt::KFParamsComb* tmtt_; + // processing region + int region_; + // + tt::TTTracks& ttTracks_; + // container of tracks + std::vector tracks_; + // container of stubs + std::vector stubs_; + // container of all Kalman Filter states + std::deque states_; + // processing stream + std::deque stream_; + // + std::vector finals_; + // current layer used during state propagation + int layer_; + }; + +} // namespace trklet + +#endif \ No newline at end of file diff --git a/L1Trigger/TrackFindingTracklet/interface/KalmanFilterFormats.h b/L1Trigger/TrackFindingTracklet/interface/KalmanFilterFormats.h new file mode 100644 index 0000000000000..1fc7c9775da8c --- /dev/null +++ b/L1Trigger/TrackFindingTracklet/interface/KalmanFilterFormats.h @@ -0,0 +1,243 @@ +#ifndef L1Trigger_TrackFindingTracklet_KalmanFilterFormats_h +#define L1Trigger_TrackFindingTracklet_KalmanFilterFormats_h + +/*---------------------------------------------------------------------- +Classes to calculate and provide dataformats used by Kalman Filter emulator +enabling tuning of bit widths +----------------------------------------------------------------------*/ + +#include "FWCore/ParameterSet/interface/ParameterSet.h" +#include "L1Trigger/TrackFindingTracklet/interface/DataFormats.h" + +#include +#include +#include +#include +#include +#include +#include + +namespace trklet { + + enum class VariableKF { + begin, + x0 = begin, + x1, + x2, + x3, + H00, + H12, + m0, + m1, + v0, + v1, + r0, + r1, + S00, + S01, + S12, + S13, + S00Shifted, + S01Shifted, + S12Shifted, + S13Shifted, + K00, + K10, + K21, + K31, + R00, + R11, + R00Rough, + R11Rough, + invR00Approx, + invR11Approx, + invR00Cor, + invR11Cor, + invR00, + invR11, + C00, + C01, + C11, + C22, + C23, + C33, + dH, + invdH, + invdH2, + H2, + Hm0, + Hm1, + Hv0, + Hv1, + H2v0, + H2v1, + end + }; + inline constexpr int operator+(VariableKF v) { return static_cast(v); } + inline constexpr VariableKF operator++(VariableKF v) { return VariableKF(+v + 1); } + + class DataFormatKF { + public: + DataFormatKF(const VariableKF& v, bool twos, const edm::ParameterSet& iConfig); + virtual ~DataFormatKF() {} + double digi(double val) const { + return enableIntegerEmulation_ ? (std::floor(val / base_ + 1.e-11) + .5) * base_ : val; + } + bool twos() const { return twos_; } + int width() const { return width_; } + double base() const { return base_; } + double range() const { return range_; } + double min() const { return min_; } + double abs() const { return abs_; } + double max() const { return max_; } + // returns false if data format would oferflow for this double value + bool inRange(double d) const; + void updateRangeActual(double d); + int integer(double d) const { return floor(d / base_ + 1.e-11); } + + protected: + VariableKF v_; + bool twos_; + bool enableIntegerEmulation_; + int width_; + double base_; + double range_; + double min_; + double abs_; + double max_; + }; + + class KalmanFilterFormats { + public: + KalmanFilterFormats(const edm::ParameterSet& iConfig); + ~KalmanFilterFormats() {} + DataFormatKF& format(VariableKF v) { return formats_[+v]; } + const tt::Setup* setup() const { return dataFormats_->setup(); } + const DataFormats* dataFormats() const { return dataFormats_; } + void consume(const DataFormats* dataFormats); + void endJob(); + + private: + template + void fillFormats(); + const edm::ParameterSet iConfig_; + const DataFormats* dataFormats_; + std::vector formats_; + }; + + template + class FormatKF : public DataFormatKF { + public: + FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig); + ~FormatKF() override {} + + private: + void calcRange() { range_ = base_ * pow(2, width_); } + void calcWidth() { width_ = ceil(log2(range_ / base_) - 1.e-11); } + }; + + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig); + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig); + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig); + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig); + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig); + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig); + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig); + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig); + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig); + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig); + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig); + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig); + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig); + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig); + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig); + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig); + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig); + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig); + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig); + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig); + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig); + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig); + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig); + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig); + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig); + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig); + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig); + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig); + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig); + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig); + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig); + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig); + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig); + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig); + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig); + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig); + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig); + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig); + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig); + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig); + + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig); + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig); + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig); + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig); + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig); + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig); + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig); + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig); + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig); + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig); + +} // namespace trklet + +#endif diff --git a/L1Trigger/TrackFindingTracklet/interface/State.h b/L1Trigger/TrackFindingTracklet/interface/State.h new file mode 100644 index 0000000000000..029106d7a3bcc --- /dev/null +++ b/L1Trigger/TrackFindingTracklet/interface/State.h @@ -0,0 +1,143 @@ +#ifndef L1Trigger_TrackFindingTracklet_State_h +#define L1Trigger_TrackFindingTracklet_State_h + +#include "L1Trigger/TrackTrigger/interface/Setup.h" +#include "L1Trigger/TrackFindingTracklet/interface/DataFormats.h" +#include "L1Trigger/TrackFindingTracklet/interface/KalmanFilterFormats.h" + +#include +#include + +namespace trklet { + + // Class to represent a Kalman Filter helix State + class State { + public: + // + struct Stub { + Stub(KalmanFilterFormats* kff, const tt::FrameStub& frame); + StubDR stubDR_; + double H12_; + double H04_; + double v0_; + double v1_; + }; + // copy constructor + State(State* state); + // proto state constructor + State(KalmanFilterFormats* kff, TrackDR* track, const std::vector& stubs, int trackId); + // updated state constructor + State(State* state, const std::vector& doubles); + // combinatoric and seed building state constructor + State(State* state, State* parent, int layer); + ~State() {} + // + State* comb(std::deque& states, int layer); + // + State* combSeed(std::deque& states, int layer); + // + State* update(std::deque& states, int layer); + // input track + TrackDR* track() const { return track_; } + // parent state (nullpointer if no parent available) + State* parent() const { return parent_; } + // stub to add to state + Stub* stub() const { return stub_; } + // hitPattern of so far added stubs + const TTBV& hitPattern() const { return hitPattern_; } + // shows which layer the found track has stubs on + const TTBV& trackPattern() const { return trackPattern_; } + // track id of input track + int trackId() const { return trackId_; } + // helix inv2R wrt input helix + double x0() const { return x0_; } + // helix phi at radius ChosenRofPhi wrt input helix + double x1() const { return x1_; } + // helix cot(Theta) wrt input helix + double x2() const { return x2_; } + // helix z at radius chosenRofZ wrt input helix + double x3() const { return x3_; } + // + double x4() const { return x4_; } + // cov. matrix element + double C00() const { return C00_; } + // cov. matrix element + double C01() const { return C01_; } + // cov. matrix element + double C11() const { return C11_; } + // cov. matrix element + double C22() const { return C22_; } + // cov. matrix element + double C23() const { return C23_; } + // cov. matrix element + double C33() const { return C33_; } + double C44() const { return C44_; } + double C40() const { return C40_; } + double C41() const { return C41_; } + // Derivative of predicted stub coords wrt helix params: stub radius minus chosenRofPhi + double H00() const { return stub_->stubDR_.r(); } + // Derivative of predicted stub coords wrt helix params: stub radius minus chosenRofZ + double H12() const { return stub_->H12_; } + // + double H04() const { return stub_->H04_; } + // stub phi residual wrt input helix + double m0() const { return stub_->stubDR_.phi(); } + // stub z residual wrt input helix + double m1() const { return stub_->stubDR_.z(); } + // stub projected phi uncertainty + double d0() const { return stub_->stubDR_.dPhi(); } + // stub projected z uncertainty + double d1() const { return stub_->stubDR_.dZ(); } + // squared stub projected phi uncertainty instead of wheight (wrong but simpler) + double v0() const { return stub_->v0_; } + // squared stub projected z uncertainty instead of wheight (wrong but simpler) + double v1() const { return stub_->v1_; } + // layer of current to add stub + int layer() const { return std::distance(stubs_.begin(), std::find(stubs_.begin(), stubs_.end(), stub_)); } + // + std::vector stubs() const { return stubs_; } + + private: + // provides data fomats + KalmanFilterFormats* kff_; + // provides run-time constants + const tt::Setup* setup_; + // input track + TrackDR* track_; + // input track stubs + std::vector stubs_; + // track id + int trackId_; + // previous state, nullptr for first states + State* parent_; + // stub to add + Stub* stub_; + // shows which layer has been added so far + TTBV hitPattern_; + // shows which layer the found track has stubs on + TTBV trackPattern_; + // helix inv2R wrt input helix + double x0_; + // helix phi at radius ChosenRofPhi wrt input helix + double x1_; + // helix cot(Theta) wrt input helix + double x2_; + // helix z at radius chosenRofZ wrt input helix + double x3_; + // impact parameter in 1/cm + double x4_; + // cov. matrix + double C00_; + double C01_; + double C11_; + double C22_; + double C23_; + double C33_; + double C44_; + double C40_; + double C41_; + }; + +} // namespace trklet + +#endif \ No newline at end of file diff --git a/L1Trigger/TrackFindingTracklet/interface/DRin.h b/L1Trigger/TrackFindingTracklet/interface/TrackMultiplexer.h similarity index 64% rename from L1Trigger/TrackFindingTracklet/interface/DRin.h rename to L1Trigger/TrackFindingTracklet/interface/TrackMultiplexer.h index f18d985405823..e0641691f6503 100644 --- a/L1Trigger/TrackFindingTracklet/interface/DRin.h +++ b/L1Trigger/TrackFindingTracklet/interface/TrackMultiplexer.h @@ -1,10 +1,9 @@ -#ifndef L1Trigger_TrackFindingTracklet_DRin_h -#define L1Trigger_TrackFindingTracklet_DRin_h +#ifndef L1Trigger_TrackFindingTracklet_TrackMultiplexer_h +#define L1Trigger_TrackFindingTracklet_TrackMultiplexer_h #include "L1Trigger/TrackTrigger/interface/Setup.h" -#include "L1Trigger/TrackerTFP/interface/DataFormats.h" -#include "L1Trigger/TrackerTFP/interface/LayerEncoding.h" #include "L1Trigger/TrackFindingTracklet/interface/ChannelAssignment.h" +#include "L1Trigger/TrackFindingTracklet/interface/DataFormats.h" #include "L1Trigger/TrackFindingTracklet/interface/Settings.h" #include @@ -12,29 +11,25 @@ namespace trklet { - /*! \class trklet::DRin + /*! \class trklet::TrackMultiplexer * \brief Class to emulate transformation of tracklet tracks and stubs into TMTT format - * and routing of seed type streams into inv2R streams + * and routing of seed type streams into single stream * \author Thomas Schuh * \date 2023, Jan */ - class DRin { + class TrackMultiplexer { public: - DRin(const edm::ParameterSet& iConfig, - const tt::Setup* setup_, - const trackerTFP::DataFormats* dataFormats, - const trackerTFP::LayerEncoding* layerEncoding, - const ChannelAssignment* channelAssignment, - const Settings* settings, - int region); - ~DRin() {} + TrackMultiplexer(const edm::ParameterSet& iConfig, + const tt::Setup* setup_, + const DataFormats* dataFormats, + const ChannelAssignment* channelAssignment, + const Settings* settings, + int region); + ~TrackMultiplexer() {} // read in and organize input tracks and stubs void consume(const tt::StreamsTrack& streamsTrack, const tt::StreamsStub& streamsStub); // fill output products - void produce(tt::StreamsStub& accpetedStubs, - tt::StreamsTrack& acceptedTracks, - tt::StreamsStub& lostStubs, - tt::StreamsTrack& lostTracks); + void produce(tt::StreamsTrack& streamsTrack, tt::StreamsStub& streamsStub); private: // truncates double precision of val into base precision, +1.e-12 restores robustness of addition of 2 digitised values @@ -42,37 +37,16 @@ namespace trklet { // basetransformation of val from baseLow into baseHigh using widthMultiplier bit multiplication double redigi(double val, double baseLow, double baseHigh, int widthMultiplier) const; struct Stub { - Stub(const TTStubRef& ttStubRef, - int layer, - int layerDet, - bool seed, - int stubId, - double r, - double phi, - double z, - bool psTilt) - : valid_(true), - ttStubRef_(ttStubRef), - layer_(layer), - layerDet_(layerDet), - layerKF_(-1), - seed_(seed), - stubId_(stubId), - r_(r), - phi_(phi), - z_(z), - psTilt_(psTilt) {} + Stub(const TTStubRef& ttStubRef, int layer, int stubId, double r, double phi, double z, bool psTilt) + : valid_(true), ttStubRef_(ttStubRef), layer_(layer), stubId_(stubId), r_(r), phi_(phi), z_(z) { + stubId_ = 2 * stubId_ + (psTilt ? 1 : 0); + } + tt::FrameStub frame(const DataFormats* df) const { return StubTM(ttStubRef_, df, stubId_, r_, phi_, z_).frame(); } bool valid_; TTStubRef ttStubRef_; - // layers a seed types can project to using default layer id [barrel: 1-6, discs: 11-15] + // kf layer id int layer_; - // layer id [0-5] barrel [6-10] end cap discs - int layerDet_; - // layer id [0-6] counted from inside-out along track - int layerKF_; - // true if stub was part of the seed - bool seed_; - // traclet stub id + // tracklet stub id, used to identify duplicates int stubId_; // radius w.r.t. chosenRofPhi in cm double r_; @@ -80,14 +54,13 @@ namespace trklet { double phi_; // z residual in cm double z_; - // true if barrel tilted module or encap PS module - bool psTilt_; }; struct Track { - static constexpr int max_ = 8; + static constexpr int max_ = 11; Track() { stubs_.reserve(max_); } Track(const TTTrackRef& ttTrackRef, bool valid, + int seedType, double inv2R, double phiT, double cot, @@ -95,16 +68,16 @@ namespace trklet { const std::vector& stubs) : ttTrackRef_(ttTrackRef), valid_(valid), - sector_(-1), + seedType_(seedType), inv2R_(inv2R), phiT_(phiT), cot_(cot), zT_(zT), stubs_(stubs) {} + tt::FrameTrack frame(const DataFormats* df) const { return TrackTM(ttTrackRef_, df, inv2R_, phiT_, zT_).frame(); } TTTrackRef ttTrackRef_; bool valid_; - TTBV maybe_; - int sector_; + int seedType_; double inv2R_; double phiT_; double cot_; @@ -118,12 +91,14 @@ namespace trklet { bool enableTruncation_; // stub residuals are recalculated from seed parameter and TTStub position bool useTTStubResiduals_; + // track parameter are recalculated from seed TTStub positions + bool useTTStubParameters_; + // + bool applyNonLinearCorrection_; // provides run-time constants const tt::Setup* setup_; // provides dataformats - const trackerTFP::DataFormats* dataFormats_; - // helper class to encode layer - const trackerTFP::LayerEncoding* layerEncoding_; + const DataFormats* dataFormats_; // helper class to assign tracks to channel const ChannelAssignment* channelAssignment_; // provides tracklet constants @@ -147,21 +122,22 @@ namespace trklet { // KF input format digitisation granularity (identical to TMTT) double baseLinv2R_; double baseLphiT_; - double baseLcot_; double baseLzT_; double baseLr_; double baseLphi_; double baseLz_; + double baseLcot_; // Finer granularity (by powers of 2) than the TMTT one. Used to transform from Tracklet to TMTT base. double baseHinv2R_; double baseHphiT_; - double baseHcot_; double baseHzT_; double baseHr_; double baseHphi_; double baseHz_; + double baseHcot_; // digitisation granularity used for inverted cot(theta) double baseInvCot_; + double baseScot_; }; } // namespace trklet diff --git a/L1Trigger/TrackFindingTracklet/plugins/L1FPGATrackProducer.cc b/L1Trigger/TrackFindingTracklet/plugins/L1FPGATrackProducer.cc index 76790d82e49e3..88b79adc0b268 100644 --- a/L1Trigger/TrackFindingTracklet/plugins/L1FPGATrackProducer.cc +++ b/L1Trigger/TrackFindingTracklet/plugins/L1FPGATrackProducer.cc @@ -24,6 +24,7 @@ // DATA FORMATS HEADERS #include "DataFormats/Common/interface/Handle.h" #include "DataFormats/Common/interface/Ref.h" +#include "DataFormats/Common/interface/OrphanHandle.h" // #include "DataFormats/DetId/interface/DetId.h" #include "DataFormats/SiPixelDetId/interface/PXBDetId.h" @@ -53,6 +54,7 @@ #include "DataFormats/L1TrackTrigger/interface/TTTypes.h" #include "DataFormats/L1TrackTrigger/interface/TTDTC.h" #include "L1Trigger/TrackTrigger/interface/Setup.h" +#include "L1Trigger/TrackerTFP/interface/TrackQuality.h" // #include "DataFormats/HepMCCandidate/interface/GenParticle.h" #include "DataFormats/Candidate/interface/Candidate.h" @@ -99,7 +101,6 @@ #include "DataFormats/GeometrySurface/interface/BoundPlane.h" #include "L1Trigger/TrackTrigger/interface/StubPtConsistency.h" -#include "L1Trigger/TrackTrigger/interface/L1TrackQuality.h" ////////////// // STD HEADERS @@ -114,6 +115,7 @@ using namespace edm; using namespace std; using namespace tt; using namespace trklet; +using namespace trackerTFP; ////////////////////////////// // // @@ -180,9 +182,6 @@ class L1FPGATrackProducer : public edm::one::EDProducer { bool extended_; bool reduced_; - bool trackQuality_; - std::unique_ptr trackQualityModel_; - std::map> dtclayerdisk; edm::InputTag MCTruthClusterInputTag; @@ -194,26 +193,30 @@ class L1FPGATrackProducer : public edm::one::EDProducer { edm::EDGetTokenT> getTokenTTClusterMCTruth_; edm::EDGetTokenT> getTokenTrackingParticle_; + // ED output token for TTTracks + const EDPutTokenT putTokenTTTracks_; // ED output token for clock and bit accurate tracks - const edm::EDPutTokenT putTokenTracks_; + EDPutTokenT putTokenTracks_; // ED output token for clock and bit accurate stubs - const edm::EDPutTokenT putTokenStubs_; - // ChannelAssignment token - const ESGetToken esGetTokenChannelAssignment_; - // helper class to assign tracks to channel - const ChannelAssignment* channelAssignment_; - - // helper class to store DTC configuration - const Setup* setup_; + EDPutTokenT putTokenStubs_; + + // helper class to store Track Trigger configuration + const Setup* setup_ = nullptr; + // helper class to store Tracklet specific configuration + const ChannelAssignment* channelAssignment_ = nullptr; + // helper class to determine track quality + const TrackQuality* trackQuality_ = nullptr; // helper class to store configuration needed by HitPatternHelper - const hph::Setup* setupHPH_; + const hph::Setup* setupHPH_ = nullptr; // Setup token - const edm::ESGetToken esGetTokenBfield_; - const edm::ESGetToken esGetTokenTGeom_; - const edm::ESGetToken esGetTokenTTopo_; - const edm::ESGetToken esGetToken_; - const edm::ESGetToken esGetTokenHPH_; + const ESGetToken esGetTokenBfield_; + const ESGetToken esGetTokenTGeom_; + const ESGetToken esGetTokenTTopo_; + const ESGetToken esGetTokenSetup_; + const ESGetToken esGetTokenChannelAssignment_; + const ESGetToken esGetTokenTrackQuality_; + const ESGetToken esGetTokenHPH_; /// ///////////////// /// /// MANDATORY METHODS /// @@ -236,23 +239,26 @@ L1FPGATrackProducer::L1FPGATrackProducer(edm::ParameterSet const& iConfig) // book ED products getTokenBS_(consumes(config.getParameter("BeamSpotSource"))), getTokenDTC_(consumes(edm::InputTag(iConfig.getParameter("InputTagTTDTC")))), - // book ED output token for clock and bit accurate tracks - putTokenTracks_(produces("Level1TTTracks")), - // book ED output token for clock and bit accurate stubs - putTokenStubs_(produces("Level1TTTracks")), + // book ED output token for TTTracks + putTokenTTTracks_(produces("Level1TTTracks")), // book ES products - esGetTokenChannelAssignment_(esConsumes()), esGetTokenBfield_(esConsumes()), esGetTokenTGeom_(esConsumes()), esGetTokenTTopo_(esConsumes()), - esGetToken_(esConsumes()), + esGetTokenSetup_(esConsumes()), + esGetTokenChannelAssignment_(esConsumes()), + esGetTokenTrackQuality_(esConsumes()), esGetTokenHPH_(esConsumes()) { if (readMoreMcTruth_) { getTokenTTClusterMCTruth_ = consumes>(MCTruthClusterInputTag); getTokenTrackingParticle_ = consumes>(TrackingParticleInputTag); } - - produces>>("Level1TTTracks").setBranchAlias("Level1TTTracks"); + const string& branchStubs = iConfig.getParameter("BranchStubs"); + const string& branchTracks = iConfig.getParameter("BranchTracks"); + // book ED output token for clock and bit accurate tracks + putTokenTracks_ = produces(branchTracks); + // book ED output token for clock and bit accurate stubs + putTokenStubs_ = produces(branchStubs); asciiEventOutName_ = iConfig.getUntrackedParameter("asciiFileName", ""); @@ -271,10 +277,6 @@ L1FPGATrackProducer::L1FPGATrackProducer(edm::ParameterSet const& iConfig) tableTREFile = iConfig.getParameter("tableTREFile"); } - // initial ES products - channelAssignment_ = nullptr; - setup_ = nullptr; - // -------------------------------------------------------------------------------- // set options in Settings based on inputs from configuration files // -------------------------------------------------------------------------------- @@ -327,11 +329,6 @@ L1FPGATrackProducer::L1FPGATrackProducer(edm::ParameterSet const& iConfig) << "\n table_TRE : " << tableTREFile.fullPath(); } } - - trackQuality_ = iConfig.getParameter("TrackQuality"); - if (trackQuality_) { - trackQualityModel_ = std::make_unique(iConfig.getParameter("TrackQualityPSet")); - } if (settings_.storeTrackBuilderOutput() && (settings_.doMultipleMatches() || !settings_.removalType().empty())) { cms::Exception exception("ConfigurationNotSupported."); exception.addContext("L1FPGATrackProducer::produce"); @@ -364,13 +361,16 @@ void L1FPGATrackProducer::beginRun(const edm::Run& run, const edm::EventSetup& i double mMagneticFieldStrength = theMagneticField->inTesla(GlobalPoint(0, 0, 0)).z(); settings_.setBfield(mMagneticFieldStrength); - setup_ = &iSetup.getData(esGetToken_); + // helper class to store Track Trigger configuration + setup_ = &iSetup.getData(esGetTokenSetup_); + // helper class to store Tracklet spezific configuration + channelAssignment_ = &iSetup.getData(esGetTokenChannelAssignment_); + // helper class to determine track quality + trackQuality_ = &iSetup.getData(esGetTokenTrackQuality_); settings_.passSetup(setup_); setupHPH_ = &iSetup.getData(esGetTokenHPH_); - // Tracklet pattern reco output channel info. - channelAssignment_ = &iSetup.getData(esGetTokenChannelAssignment_); // initialize the tracklet event processing (this sets all the processing & memory modules, wiring, etc) eventProcessor.init(settings_, setup_); } @@ -752,7 +752,7 @@ void L1FPGATrackProducer::produce(edm::Event& iEvent, const edm::EventSetup& iSe aTrack.setTrackWordBits(); if (trackQuality_) { - trackQualityModel_->setL1TrackQuality(aTrack); + trackQuality_->setL1TrackQuality(aTrack); } // hph::HitPatternHelper hph(setupHPH_, tmp_hit, tmp_tanL, tmp_z0); @@ -771,47 +771,51 @@ void L1FPGATrackProducer::produce(edm::Event& iEvent, const edm::EventSetup& iSe L1TkTracksForOutput->push_back(aTrack); } - iEvent.put(std::move(L1TkTracksForOutput), "Level1TTTracks"); + const OrphanHandle oh = iEvent.emplace(putTokenTTTracks_, move(*L1TkTracksForOutput)); // produce clock and bit accurate stream output tracks and stubs. // from end of tracklet pattern recognition. // Convertion here is from stream format that allows this code to run // outside CMSSW to the EDProduct one. - Streams streamsTrack(numStreamsTrack); + int iTrk(0); + StreamsTrack streamsTrack(numStreamsTrack); StreamsStub streamsStub(numStreamsStub); - - for (unsigned int chanTrk = 0; chanTrk < numStreamsTrack; chanTrk++) { - for (unsigned int itk = 0; itk < streamsTrackRaw[chanTrk].size(); itk++) { - std::string bitsTrk = streamsTrackRaw[chanTrk][itk]; - int iSeed = chanTrk % channelAssignment_->numChannelsTrack(); // seed type - streamsTrack[chanTrk].emplace_back(bitsTrk); - - const unsigned int chanStubOffsetIn = chanTrk * numStubChannel; - const unsigned int chanStubOffsetOut = channelAssignment_->offsetStub(chanTrk); - const unsigned int numProjLayers = channelAssignment_->numProjectionLayers(iSeed); - TTBV hitMap(0, numProjLayers + numSeedingLayers); - // remove padding from stub stream - for (unsigned int iproj = 0; iproj < numStubChannel; iproj++) { - // FW current has one (perhaps invalid) stub per layer per track. - const StubStreamData& stubdata = streamsStubRaw[chanStubOffsetIn + iproj][itk]; - const L1TStub& stub = stubdata.stub(); - if (!stubdata.valid()) - continue; - const TTStubRef& ttStubRef = stubMap[stub]; - const int seedType = stubdata.iSeed(); - const int layerId = setup_->layerId(ttStubRef); - const int channelId = channelAssignment_->channelId(seedType, layerId); - hitMap.set(channelId); - streamsStub[chanStubOffsetOut + channelId].emplace_back(ttStubRef, stubdata.dataBits()); + for (int channel = 0; channel < (int)numStreamsTrack; channel++) { + const int seedType = channel % channelAssignment_->numChannelsTrack(); + const int numLayers = channelAssignment_->numProjectionLayers(seedType) + channelAssignment_->numSeedingLayers(); + const int offsetIn = channel * numStubChannel; + const int offsetOut = channelAssignment_->offsetStub(channel); + const vector& tracks = streamsTrackRaw[channel]; + StreamTrack& streamTrack = streamsTrack[channel]; + streamTrack.reserve(tracks.size()); + for (int layer = 0; layer < numLayers; layer++) + streamsStub[offsetOut + layer].reserve(tracks.size()); + for (int frame = 0; frame < (int)tracks.size(); frame++) { + const Frame bitsTrk(tracks[frame]); + if (bitsTrk.none()) { + streamTrack.emplace_back(FrameTrack()); + for (int layer = 0; layer < numLayers; layer++) + streamsStub[offsetOut + layer].emplace_back(FrameStub()); + continue; } - for (int layerId : hitMap.ids(false)) { // invalid stubs - streamsStub[chanStubOffsetOut + layerId].emplace_back(tt::FrameStub()); + const TTTrackRef ttTrackRef(oh, iTrk++); + streamTrack.emplace_back(ttTrackRef, bitsTrk); + StreamStub stubs(numLayers, FrameStub()); + for (int layer = 0; layer < numLayers; layer++) { + const StubStreamData& stub = streamsStubRaw[offsetIn + layer][frame]; + if (!stub.valid()) + continue; + const TTStubRef& ttStubRef = stubMap[stub.stub()]; + const int index = channelAssignment_->channelId(seedType, setup_->layerId(ttStubRef)); + stubs[index] = FrameStub(ttStubRef, stub.dataBits()); } + int layer(0); + for (const FrameStub& fs : stubs) + streamsStub[offsetOut + layer++].push_back(fs); } } - - iEvent.emplace(putTokenTracks_, std::move(streamsTrack)); - iEvent.emplace(putTokenStubs_, std::move(streamsStub)); + iEvent.emplace(putTokenTracks_, move(streamsTrack)); + iEvent.emplace(putTokenStubs_, move(streamsStub)); } /// End of produce() diff --git a/L1Trigger/TrackFindingTracklet/plugins/ProducerDR.cc b/L1Trigger/TrackFindingTracklet/plugins/ProducerDR.cc index 1c8055c022282..225acc4c1aeee 100644 --- a/L1Trigger/TrackFindingTracklet/plugins/ProducerDR.cc +++ b/L1Trigger/TrackFindingTracklet/plugins/ProducerDR.cc @@ -11,7 +11,8 @@ #include "DataFormats/Common/interface/Handle.h" #include "L1Trigger/TrackTrigger/interface/Setup.h" -#include "L1Trigger/TrackerTFP/interface/DataFormats.h" +#include "L1Trigger/TrackerTFP/interface/LayerEncoding.h" +#include "L1Trigger/TrackFindingTracklet/interface/DataFormats.h" #include "L1Trigger/TrackFindingTracklet/interface/ChannelAssignment.h" #include "L1Trigger/TrackFindingTracklet/interface/DR.h" #include "SimDataFormats/Associations/interface/TTTypes.h" @@ -25,13 +26,14 @@ using namespace std; using namespace edm; -using namespace trackerTFP; using namespace tt; +using namespace trackerTFP; namespace trklet { /*! \class trklet::ProducerDR - * \brief Emulates removal of duplicated TTTracks f/w + * \brief Emulates removal of duplicated TTTracks f/w. + * Track order determined by TrackMultiplexer affects performance * \author Thomas Schuh * \date 2023, Feb */ @@ -49,13 +51,13 @@ namespace trklet { // ED input token of Stubs EDGetTokenT edGetTokenStubs_; // ED output token for stubs - EDPutTokenT edPutTokenAcceptedStubs_; - EDPutTokenT edPutTokenLostStubs_; + EDPutTokenT edPutTokenStubs_; // ED output token for tracks - EDPutTokenT edPutTokenAcceptedTracks_; - EDPutTokenT edPutTokenLostTracks_; + EDPutTokenT edPutTokenTracks_; // Setup token ESGetToken esGetTokenSetup_; + // LayerEncoding token + ESGetToken esGetTokenLayerEncoding_; // DataFormats token ESGetToken esGetTokenDataFormats_; // ChannelAssignment token @@ -64,6 +66,8 @@ namespace trklet { ParameterSet iConfig_; // helper class to store configurations const Setup* setup_ = nullptr; + // helper class to encode layer + const LayerEncoding* layerEncoding_ = nullptr; // helper class to extract structured data from tt::Frames const DataFormats* dataFormats_ = nullptr; // helper class to assign tracks to channel @@ -71,20 +75,17 @@ namespace trklet { }; ProducerDR::ProducerDR(const ParameterSet& iConfig) : iConfig_(iConfig) { - const string& label = iConfig.getParameter("LabelDRin"); - const string& branchAcceptedStubs = iConfig.getParameter("BranchAcceptedStubs"); - const string& branchAcceptedTracks = iConfig.getParameter("BranchAcceptedTracks"); - const string& branchLostStubs = iConfig.getParameter("BranchLostStubs"); - const string& branchLostTracks = iConfig.getParameter("BranchLostTracks"); + const string& label = iConfig.getParameter("InputLabelDR"); + const string& branchStubs = iConfig.getParameter("BranchStubs"); + const string& branchTracks = iConfig.getParameter("BranchTracks"); // book in- and output ED products - edGetTokenTracks_ = consumes(InputTag(label, branchAcceptedTracks)); - edGetTokenStubs_ = consumes(InputTag(label, branchAcceptedStubs)); - edPutTokenAcceptedStubs_ = produces(branchAcceptedStubs); - edPutTokenAcceptedTracks_ = produces(branchAcceptedTracks); - edPutTokenLostStubs_ = produces(branchLostStubs); - edPutTokenLostTracks_ = produces(branchLostTracks); + edGetTokenTracks_ = consumes(InputTag(label, branchTracks)); + edGetTokenStubs_ = consumes(InputTag(label, branchStubs)); + edPutTokenTracks_ = produces(branchTracks); + edPutTokenStubs_ = produces(branchStubs); // book ES products esGetTokenSetup_ = esConsumes(); + esGetTokenLayerEncoding_ = esConsumes(); esGetTokenDataFormats_ = esConsumes(); esGetTokenChannelAssignment_ = esConsumes(); } @@ -92,11 +93,8 @@ namespace trklet { void ProducerDR::beginRun(const Run& iRun, const EventSetup& iSetup) { // helper class to store configurations setup_ = &iSetup.getData(esGetTokenSetup_); - if (!setup_->configurationSupported()) - return; - // check process history if desired - if (iConfig_.getParameter("CheckHistory")) - setup_->checkHistory(iRun.processHistory()); + // helper class to encode layer + layerEncoding_ = &iSetup.getData(esGetTokenLayerEncoding_); // helper class to extract structured data from tt::Frames dataFormats_ = &iSetup.getData(esGetTokenDataFormats_); // helper class to assign tracks to channel @@ -105,34 +103,28 @@ namespace trklet { void ProducerDR::produce(Event& iEvent, const EventSetup& iSetup) { // empty DR products - const int numStreamsTracks = channelAssignment_->numNodesDR() * setup_->numRegions(); + const int numStreamsTracks = setup_->numRegions(); const int numStreamsStubs = numStreamsTracks * setup_->numLayers(); - StreamsStub acceptedStubs(numStreamsStubs); - StreamsTrack acceptedTracks(numStreamsTracks); - StreamsStub lostStubs(numStreamsStubs); - StreamsTrack lostTracks(numStreamsTracks); + StreamsStub streamsStub(numStreamsStubs); + StreamsTrack streamsTrack(numStreamsTracks); // read in TBout Product and produce KFin product - if (setup_->configurationSupported()) { - Handle handleStubs; - iEvent.getByToken(edGetTokenStubs_, handleStubs); - const StreamsStub& stubs = *handleStubs; - Handle handleTracks; - iEvent.getByToken(edGetTokenTracks_, handleTracks); - const StreamsTrack& tracks = *handleTracks; - for (int region = 0; region < setup_->numRegions(); region++) { - // object to remove duplicated tracks in a processing region - DR dr(iConfig_, setup_, dataFormats_, channelAssignment_, region); - // read in and organize input tracks and stubs - dr.consume(tracks, stubs); - // fill output products - dr.produce(acceptedStubs, acceptedTracks, lostStubs, lostTracks); - } + Handle handleStubs; + iEvent.getByToken(edGetTokenStubs_, handleStubs); + const StreamsStub& stubs = *handleStubs; + Handle handleTracks; + iEvent.getByToken(edGetTokenTracks_, handleTracks); + const StreamsTrack& tracks = *handleTracks; + for (int region = 0; region < setup_->numRegions(); region++) { + // object to remove duplicated tracks in a processing region + DuplicateRemoval dr(iConfig_, setup_, layerEncoding_, dataFormats_, channelAssignment_, region); + // read in and organize input tracks and stubs + dr.consume(tracks, stubs); + // fill output products + dr.produce(streamsTrack, streamsStub); } // store products - iEvent.emplace(edPutTokenAcceptedStubs_, std::move(acceptedStubs)); - iEvent.emplace(edPutTokenAcceptedTracks_, std::move(acceptedTracks)); - iEvent.emplace(edPutTokenLostStubs_, std::move(lostStubs)); - iEvent.emplace(edPutTokenLostTracks_, std::move(lostTracks)); + iEvent.emplace(edPutTokenStubs_, move(streamsStub)); + iEvent.emplace(edPutTokenTracks_, move(streamsTrack)); } } // namespace trklet diff --git a/L1Trigger/TrackFindingTracklet/plugins/ProducerDRin.cc b/L1Trigger/TrackFindingTracklet/plugins/ProducerDRin.cc deleted file mode 100644 index 86b0dd8f3e48b..0000000000000 --- a/L1Trigger/TrackFindingTracklet/plugins/ProducerDRin.cc +++ /dev/null @@ -1,152 +0,0 @@ -#include "FWCore/Framework/interface/stream/EDProducer.h" -#include "FWCore/Framework/interface/Run.h" -#include "FWCore/Framework/interface/EventSetup.h" -#include "FWCore/Framework/interface/Event.h" -#include "FWCore/Framework/interface/MakerMacros.h" -#include "FWCore/Utilities/interface/EDGetToken.h" -#include "FWCore/Utilities/interface/EDPutToken.h" -#include "FWCore/Utilities/interface/ESGetToken.h" -#include "FWCore/Utilities/interface/InputTag.h" -#include "FWCore/ParameterSet/interface/ParameterSet.h" -#include "DataFormats/Common/interface/Handle.h" - -#include "L1Trigger/TrackTrigger/interface/Setup.h" -#include "L1Trigger/TrackerTFP/interface/DataFormats.h" -#include "L1Trigger/TrackerTFP/interface/LayerEncoding.h" -#include "L1Trigger/TrackFindingTracklet/interface/ChannelAssignment.h" -#include "L1Trigger/TrackFindingTracklet/interface/Settings.h" -#include "L1Trigger/TrackFindingTracklet/interface/DRin.h" -#include "SimDataFormats/Associations/interface/TTTypes.h" - -#include -#include -#include -#include -#include -#include - -using namespace std; -using namespace edm; -using namespace trackerTFP; -using namespace tt; - -namespace trklet { - - /*! \class trklet::ProducerDRin - * \brief Transforms format of TBout into that expected by DR input. - * \author Thomas Schuh - * \date 2023, Jan - */ - class ProducerDRin : public stream::EDProducer<> { - public: - explicit ProducerDRin(const ParameterSet&); - ~ProducerDRin() override {} - - private: - void beginRun(const Run&, const EventSetup&) override; - void produce(Event&, const EventSetup&) override; - virtual void endJob() {} - - // ED input token of Tracks - EDGetTokenT edGetTokenTracks_; - // ED input token of Stubs - EDGetTokenT edGetTokenStubs_; - // ED output token for stubs - EDPutTokenT edPutTokenAcceptedStubs_; - EDPutTokenT edPutTokenLostStubs_; - // ED output token for tracks - EDPutTokenT edPutTokenAcceptedTracks_; - EDPutTokenT edPutTokenLostTracks_; - // Setup token - ESGetToken esGetTokenSetup_; - // DataFormats token - ESGetToken esGetTokenDataFormats_; - // LayerEncoding token - ESGetToken esGetTokenLayerEncoding_; - // ChannelAssignment token - ESGetToken esGetTokenChannelAssignment_; - // configuration - ParameterSet iConfig_; - // helper class to store configurations - const Setup* setup_ = nullptr; - // helper class to extract structured data from tt::Frames - const DataFormats* dataFormats_ = nullptr; - // helper class to encode layer - const LayerEncoding* layerEncoding_ = nullptr; - // helper class to assign tracks to channel - const ChannelAssignment* channelAssignment_ = nullptr; - // helper class to store tracklet configurations - Settings settings_; - }; - - ProducerDRin::ProducerDRin(const ParameterSet& iConfig) : iConfig_(iConfig) { - const string& label = iConfig.getParameter("LabelTBout"); - const string& branchAcceptedStubs = iConfig.getParameter("BranchAcceptedStubs"); - const string& branchAcceptedTracks = iConfig.getParameter("BranchAcceptedTracks"); - const string& branchLostStubs = iConfig.getParameter("BranchLostStubs"); - const string& branchLostTracks = iConfig.getParameter("BranchLostTracks"); - // book in- and output ED products - edGetTokenTracks_ = consumes(InputTag(label, branchAcceptedTracks)); - edGetTokenStubs_ = consumes(InputTag(label, branchAcceptedStubs)); - edPutTokenAcceptedStubs_ = produces(branchAcceptedStubs); - edPutTokenAcceptedTracks_ = produces(branchAcceptedTracks); - edPutTokenLostStubs_ = produces(branchLostStubs); - edPutTokenLostTracks_ = produces(branchLostTracks); - // book ES products - esGetTokenSetup_ = esConsumes(); - esGetTokenDataFormats_ = esConsumes(); - esGetTokenLayerEncoding_ = esConsumes(); - esGetTokenChannelAssignment_ = esConsumes(); - } - - void ProducerDRin::beginRun(const Run& iRun, const EventSetup& iSetup) { - // helper class to store configurations - setup_ = &iSetup.getData(esGetTokenSetup_); - if (!setup_->configurationSupported()) - return; - // check process history if desired - if (iConfig_.getParameter("CheckHistory")) - setup_->checkHistory(iRun.processHistory()); - // helper class to extract structured data from tt::Frames - dataFormats_ = &iSetup.getData(esGetTokenDataFormats_); - // helper class to encode layer - layerEncoding_ = &iSetup.getData(esGetTokenLayerEncoding_); - // helper class to assign tracks to channel - channelAssignment_ = &iSetup.getData(esGetTokenChannelAssignment_); - } - - void ProducerDRin::produce(Event& iEvent, const EventSetup& iSetup) { - // empty KFin products - const int numStreamsTracks = channelAssignment_->numNodesDR() * setup_->numRegions(); - const int numStreamsStubs = numStreamsTracks * setup_->numLayers(); - StreamsStub acceptedStubs(numStreamsStubs); - StreamsTrack acceptedTracks(numStreamsTracks); - StreamsStub lostStubs(numStreamsStubs); - StreamsTrack lostTracks(numStreamsTracks); - // read in TBout Product and produce KFin product - if (setup_->configurationSupported()) { - Handle handleStubs; - iEvent.getByToken(edGetTokenStubs_, handleStubs); - const StreamsStub& stubs = *handleStubs; - Handle handleTracks; - iEvent.getByToken(edGetTokenTracks_, handleTracks); - const StreamsTrack& tracks = *handleTracks; - for (int region = 0; region < setup_->numRegions(); region++) { - // object to reformat tracks from tracklet fromat to TMTT format in a processing region - DRin drin(iConfig_, setup_, dataFormats_, layerEncoding_, channelAssignment_, &settings_, region); - // read in and organize input tracks and stubs - drin.consume(tracks, stubs); - // fill output products - drin.produce(acceptedStubs, acceptedTracks, lostStubs, lostTracks); - } - } - // store products - iEvent.emplace(edPutTokenAcceptedStubs_, std::move(acceptedStubs)); - iEvent.emplace(edPutTokenAcceptedTracks_, std::move(acceptedTracks)); - iEvent.emplace(edPutTokenLostStubs_, std::move(lostStubs)); - iEvent.emplace(edPutTokenLostTracks_, std::move(lostTracks)); - } - -} // namespace trklet - -DEFINE_FWK_MODULE(trklet::ProducerDRin); diff --git a/L1Trigger/TrackFindingTracklet/plugins/ProducerDataFormats.cc b/L1Trigger/TrackFindingTracklet/plugins/ProducerDataFormats.cc new file mode 100644 index 0000000000000..dbf9d104e4b46 --- /dev/null +++ b/L1Trigger/TrackFindingTracklet/plugins/ProducerDataFormats.cc @@ -0,0 +1,46 @@ +#include "FWCore/Framework/interface/ESProducer.h" +#include "FWCore/Framework/interface/ESHandle.h" +#include "FWCore/ParameterSet/interface/ParameterSet.h" +#include "FWCore/Utilities/interface/ESGetToken.h" +#include "FWCore/Utilities/interface/ESInputTag.h" +#include "DataFormats/Provenance/interface/ParameterSetID.h" +#include "L1Trigger/TrackFindingTracklet/interface/ChannelAssignment.h" +#include "L1Trigger/TrackFindingTracklet/interface/DataFormats.h" + +#include + +using namespace std; +using namespace edm; +using namespace tt; + +namespace trklet { + + /*! \class trklet::ProducerDataFormats + * \brief Class to produce setup of Hybrid emulator data formats + * \author Thomas Schuh + * \date 2024, Sep + */ + class ProducerDataFormats : public ESProducer { + public: + ProducerDataFormats(const ParameterSet& iConfig); + ~ProducerDataFormats() override {} + unique_ptr produce(const DataFormatsRcd& rcd); + + private: + const ParameterSet iConfig_; + ESGetToken esGetToken_; + }; + + ProducerDataFormats::ProducerDataFormats(const ParameterSet& iConfig) { + auto cc = setWhatProduced(this); + esGetToken_ = cc.consumes(); + } + + unique_ptr ProducerDataFormats::produce(const DataFormatsRcd& rcd) { + const ChannelAssignment* ca = &rcd.get(esGetToken_); + return make_unique(ca); + } + +} // namespace trklet + +DEFINE_FWK_EVENTSETUP_MODULE(trklet::ProducerDataFormats); \ No newline at end of file diff --git a/L1Trigger/TrackFindingTracklet/plugins/ProducerKF.cc b/L1Trigger/TrackFindingTracklet/plugins/ProducerKF.cc new file mode 100644 index 0000000000000..68b7ad13ca971 --- /dev/null +++ b/L1Trigger/TrackFindingTracklet/plugins/ProducerKF.cc @@ -0,0 +1,178 @@ +#include "FWCore/Framework/interface/stream/EDProducer.h" +#include "FWCore/Framework/interface/Run.h" +#include "FWCore/Framework/interface/EventSetup.h" +#include "FWCore/Framework/interface/Event.h" +#include "FWCore/Framework/interface/MakerMacros.h" +#include "FWCore/Utilities/interface/EDGetToken.h" +#include "FWCore/Utilities/interface/EDPutToken.h" +#include "FWCore/Utilities/interface/ESGetToken.h" +#include "FWCore/Utilities/interface/InputTag.h" +#include "FWCore/ParameterSet/interface/ParameterSet.h" +#include "DataFormats/Common/interface/Handle.h" + +#include "DataFormats/L1TrackTrigger/interface/TTTypes.h" +#include "L1Trigger/TrackTrigger/interface/Setup.h" +#include "L1Trigger/TrackFindingTracklet/interface/DataFormats.h" +#include "L1Trigger/TrackFindingTracklet/interface/KalmanFilter.h" +#include "L1Trigger/TrackFindingTMTT/interface/Settings.h" +#include "L1Trigger/TrackFindingTMTT/interface/KFParamsComb.h" + +#include +#include + +using namespace std; +using namespace edm; +using namespace tt; +using namespace tmtt; + +namespace trklet { + + /*! \class trklet::ProducerKF + * \brief L1TrackTrigger Kamlan Filter emulator + * \author Thomas Schuh + * \date 2020, July + */ + class ProducerKF : public stream::EDProducer<> { + public: + explicit ProducerKF(const ParameterSet&); + ~ProducerKF() override {} + + private: + void beginRun(const Run&, const EventSetup&) override; + void produce(Event&, const EventSetup&) override; + void endStream() override { + if (printDebug_) + kalmanFilterFormats_.endJob(); + } + // ED input token of sf stubs and tracks + EDGetTokenT edGetTokenStubs_; + EDGetTokenT edGetTokenTracks_; + // ED output token for accepted stubs and tracks + EDPutTokenT edPutTokenTTTracks_; + EDPutTokenT edPutTokenStubs_; + EDPutTokenT edPutTokenTracks_; + // ED output token for number of accepted and lost States + EDPutTokenT edPutTokenNumStatesAccepted_; + EDPutTokenT edPutTokenNumStatesTruncated_; + // Setup token + ESGetToken esGetTokenSetup_; + // DataFormats token + ESGetToken esGetTokenDataFormats_; + // configuration + ParameterSet iConfig_; + // helper class to store configurations + const Setup* setup_ = nullptr; + // helper class to extract structured data from tt::Frames + const DataFormats* dataFormats_ = nullptr; + // provides dataformats of Kalman filter internals + KalmanFilterFormats kalmanFilterFormats_; + // + Settings settings_; + // + KFParamsComb tmtt4_; + // + KFParamsComb tmtt5_; + // + KFParamsComb* tmtt_; + // print end job internal unused MSB + bool printDebug_; + // + bool use5ParameterFit_; + }; + + ProducerKF::ProducerKF(const ParameterSet& iConfig) + : iConfig_(iConfig), + kalmanFilterFormats_(iConfig), + settings_(iConfig), + tmtt4_(&settings_, 4, "KF5ParamsComb"), + tmtt5_(&settings_, 5, "KF4ParamsComb"), + tmtt_(&tmtt4_) { + printDebug_ = iConfig.getParameter("PrintKFDebug"); + use5ParameterFit_ = iConfig.getParameter("Use5ParameterFit"); + if (use5ParameterFit_) + tmtt_ = &tmtt5_; + const string& label = iConfig.getParameter("InputLabelKF"); + const string& branchStubs = iConfig.getParameter("BranchStubs"); + const string& branchTracks = iConfig.getParameter("BranchTracks"); + const string& branchTruncated = iConfig.getParameter("BranchTruncated"); + // book in- and output ED products + edGetTokenStubs_ = consumes(InputTag(label, branchStubs)); + edGetTokenTracks_ = consumes(InputTag(label, branchTracks)); + edPutTokenStubs_ = produces(branchStubs); + edPutTokenTracks_ = produces(branchTracks); + edPutTokenTTTracks_ = produces(branchTracks); + edPutTokenNumStatesAccepted_ = produces(branchTracks); + edPutTokenNumStatesTruncated_ = produces(branchTruncated); + // book ES products + esGetTokenSetup_ = esConsumes(); + esGetTokenDataFormats_ = esConsumes(); + } + + void ProducerKF::beginRun(const Run& iRun, const EventSetup& iSetup) { + // helper class to store configurations + setup_ = &iSetup.getData(esGetTokenSetup_); + // helper class to extract structured data from tt::Frames + dataFormats_ = &iSetup.getData(esGetTokenDataFormats_); + // provides dataformats of Kalman filter internals + kalmanFilterFormats_.consume(dataFormats_); + settings_.setMagneticField(setup_->bField()); + } + + void ProducerKF::produce(Event& iEvent, const EventSetup& iSetup) { + auto valid = [](int& sum, const FrameTrack& f) { return sum += f.first.isNull() ? 0 : 1; }; + static const int numRegions = setup_->numRegions(); + static const int numLayers = setup_->numLayers(); + // empty KF products + StreamsStub streamsStub(numRegions * numLayers); + StreamsTrack streamsTrack(numRegions); + int numStatesAccepted(0); + int numStatesTruncated(0); + // read in DR Product and produce KF product + Handle handleStubs; + iEvent.getByToken(edGetTokenStubs_, handleStubs); + const StreamsStub& stubs = *handleStubs; + Handle handleTracks; + iEvent.getByToken(edGetTokenTracks_, handleTracks); + const StreamsTrack& tracks = *handleTracks; + // prep TTTracks + TTTracks ttTracks; + vector ttTrackRefs; + if (use5ParameterFit_) { + int nTracks(0); + for (const StreamTrack& stream : tracks) + nTracks += accumulate(stream.begin(), stream.end(), 0, valid); + ttTracks.reserve(nTracks); + ttTrackRefs.reserve(nTracks); + for (const StreamTrack& stream : tracks) + for (const FrameTrack& frame : stream) + if (frame.first.isNonnull()) + ttTrackRefs.push_back(frame.first); + } + for (int region = 0; region < setup_->numRegions(); region++) { + // object to fit tracks in a processing region + KalmanFilter kf(iConfig_, setup_, dataFormats_, &kalmanFilterFormats_, &settings_, tmtt_, region, ttTracks); + // read in and organize input tracks and stubs + kf.consume(tracks, stubs); + // fill output products + kf.produce(streamsStub, streamsTrack, numStatesAccepted, numStatesTruncated); + } + if (use5ParameterFit_) { + // store ttTracks + const OrphanHandle oh = iEvent.emplace(edPutTokenTTTracks_, move(ttTracks)); + // replace ttTrackRefs in track streams + int iTrk(0); + for (StreamTrack& stream : streamsTrack) + for (FrameTrack& frame : stream) + if (frame.first.isNonnull()) + frame.first = TTTrackRef(oh, iTrk++); + } + // store products + iEvent.emplace(edPutTokenStubs_, move(streamsStub)); + iEvent.emplace(edPutTokenTracks_, move(streamsTrack)); + iEvent.emplace(edPutTokenNumStatesAccepted_, numStatesAccepted); + iEvent.emplace(edPutTokenNumStatesTruncated_, numStatesTruncated); + } + +} // namespace trklet + +DEFINE_FWK_MODULE(trklet::ProducerKF); diff --git a/L1Trigger/TrackFindingTracklet/plugins/ProducerKFin.cc b/L1Trigger/TrackFindingTracklet/plugins/ProducerKFin.cc deleted file mode 100644 index 1a23f97513f01..0000000000000 --- a/L1Trigger/TrackFindingTracklet/plugins/ProducerKFin.cc +++ /dev/null @@ -1,147 +0,0 @@ -#include "FWCore/Framework/interface/stream/EDProducer.h" -#include "FWCore/Framework/interface/Run.h" -#include "FWCore/Framework/interface/EventSetup.h" -#include "FWCore/Framework/interface/Event.h" -#include "FWCore/Framework/interface/MakerMacros.h" -#include "FWCore/Utilities/interface/EDGetToken.h" -#include "FWCore/Utilities/interface/EDPutToken.h" -#include "FWCore/Utilities/interface/ESGetToken.h" -#include "FWCore/Utilities/interface/InputTag.h" -#include "FWCore/ParameterSet/interface/ParameterSet.h" -#include "DataFormats/Common/interface/Handle.h" - -#include "L1Trigger/TrackTrigger/interface/Setup.h" -#include "L1Trigger/TrackerTFP/interface/DataFormats.h" -#include "L1Trigger/TrackFindingTracklet/interface/ChannelAssignment.h" -#include "L1Trigger/TrackFindingTracklet/interface/KFin.h" -#include "SimDataFormats/Associations/interface/TTTypes.h" - -#include -#include -#include -#include -#include -#include - -using namespace std; -using namespace edm; -using namespace trackerTFP; -using namespace tt; - -namespace trklet { - - /*! \class trklet::ProducerKFin - * \brief Transforms format of DR into that expected by KF input. - * \author Thomas Schuh - * \date 2023, Feb - */ - class ProducerKFin : public stream::EDProducer<> { - public: - explicit ProducerKFin(const ParameterSet&); - ~ProducerKFin() override {} - - private: - void beginRun(const Run&, const EventSetup&) override; - void produce(Event&, const EventSetup&) override; - virtual void endJob() {} - // ED input token of Tracks - EDGetTokenT edGetTokenTracks_; - // ED input token of Stubs - EDGetTokenT edGetTokenStubs_; - // ED output token for stubs - EDPutTokenT edPutTokenAcceptedStubs_; - EDPutTokenT edPutTokenLostStubs_; - // ED output token for tracks - EDPutTokenT edPutTokenAcceptedTracks_; - EDPutTokenT edPutTokenLostTracks_; - // Setup token - ESGetToken esGetTokenSetup_; - // DataFormats token - ESGetToken esGetTokenDataFormats_; - // LayerEncoding token - ESGetToken esGetTokenLayerEncoding_; - // ChannelAssignment token - ESGetToken esGetTokenChannelAssignment_; - // configuration - ParameterSet iConfig_; - // helper class to store configurations - const Setup* setup_ = nullptr; - // helper class to extract structured data from tt::Frames - const DataFormats* dataFormats_ = nullptr; - // helper class to encode layer - const LayerEncoding* layerEncoding_ = nullptr; - // helper class to assign tracks to channel - const ChannelAssignment* channelAssignment_ = nullptr; - }; - - ProducerKFin::ProducerKFin(const ParameterSet& iConfig) : iConfig_(iConfig) { - const string& label = iConfig.getParameter("LabelDR"); - const string& branchAcceptedStubs = iConfig.getParameter("BranchAcceptedStubs"); - const string& branchAcceptedTracks = iConfig.getParameter("BranchAcceptedTracks"); - const string& branchLostStubs = iConfig.getParameter("BranchLostStubs"); - const string& branchLostTracks = iConfig.getParameter("BranchLostTracks"); - // book in- and output ED products - edGetTokenTracks_ = consumes(InputTag(label, branchAcceptedTracks)); - edGetTokenStubs_ = consumes(InputTag(label, branchAcceptedStubs)); - edPutTokenAcceptedStubs_ = produces(branchAcceptedStubs); - edPutTokenAcceptedTracks_ = produces(branchAcceptedTracks); - edPutTokenLostStubs_ = produces(branchLostStubs); - edPutTokenLostTracks_ = produces(branchLostTracks); - // book ES products - esGetTokenSetup_ = esConsumes(); - esGetTokenDataFormats_ = esConsumes(); - esGetTokenLayerEncoding_ = esConsumes(); - esGetTokenChannelAssignment_ = esConsumes(); - } - - void ProducerKFin::beginRun(const Run& iRun, const EventSetup& iSetup) { - // helper class to store configurations - setup_ = &iSetup.getData(esGetTokenSetup_); - if (!setup_->configurationSupported()) - return; - // check process history if desired - if (iConfig_.getParameter("CheckHistory")) - setup_->checkHistory(iRun.processHistory()); - // helper class to extract structured data from tt::Frames - dataFormats_ = &iSetup.getData(esGetTokenDataFormats_); - // helper class to encode layer - layerEncoding_ = &iSetup.getData(esGetTokenLayerEncoding_); - // helper class to assign tracks to channel - channelAssignment_ = &iSetup.getData(esGetTokenChannelAssignment_); - } - - void ProducerKFin::produce(Event& iEvent, const EventSetup& iSetup) { - // empty KFin products - const int numStreamsTracks = setup_->kfNumWorker() * setup_->numRegions(); - const int numStreamsStubs = numStreamsTracks * setup_->numLayers(); - StreamsStub acceptedStubs(numStreamsStubs); - StreamsTrack acceptedTracks(numStreamsTracks); - StreamsStub lostStubs(numStreamsStubs); - StreamsTrack lostTracks(numStreamsTracks); - // read in TBout Product and produce KFin product - if (setup_->configurationSupported()) { - Handle handleStubs; - iEvent.getByToken(edGetTokenStubs_, handleStubs); - const StreamsStub& stubs = *handleStubs; - Handle handleTracks; - iEvent.getByToken(edGetTokenTracks_, handleTracks); - const StreamsTrack& tracks = *handleTracks; - for (int region = 0; region < setup_->numRegions(); region++) { - // object to reformat tracks from DR fromat to KF format in a processing region - KFin kfin(iConfig_, setup_, dataFormats_, layerEncoding_, channelAssignment_, region); - // read in and organize input tracks and stubs - kfin.consume(tracks, stubs); - // fill output products - kfin.produce(acceptedStubs, acceptedTracks, lostStubs, lostTracks); - } - } - // store products - iEvent.emplace(edPutTokenAcceptedStubs_, std::move(acceptedStubs)); - iEvent.emplace(edPutTokenAcceptedTracks_, std::move(acceptedTracks)); - iEvent.emplace(edPutTokenLostStubs_, std::move(lostStubs)); - iEvent.emplace(edPutTokenLostTracks_, std::move(lostTracks)); - } - -} // namespace trklet - -DEFINE_FWK_MODULE(trklet::ProducerKFin); diff --git a/L1Trigger/TrackFindingTracklet/plugins/ProducerKFout.cc b/L1Trigger/TrackFindingTracklet/plugins/ProducerKFout.cc deleted file mode 100644 index f7001add46306..0000000000000 --- a/L1Trigger/TrackFindingTracklet/plugins/ProducerKFout.cc +++ /dev/null @@ -1,381 +0,0 @@ -#include "FWCore/Framework/interface/stream/EDProducer.h" -#include "FWCore/Framework/interface/Run.h" -#include "FWCore/Framework/interface/EventSetup.h" -#include "FWCore/Framework/interface/Event.h" -#include "FWCore/Framework/interface/MakerMacros.h" -#include "FWCore/Utilities/interface/EDGetToken.h" -#include "FWCore/Utilities/interface/EDPutToken.h" -#include "FWCore/Utilities/interface/ESGetToken.h" -#include "FWCore/Utilities/interface/InputTag.h" -#include "FWCore/ParameterSet/interface/ParameterSet.h" -#include "DataFormats/Common/interface/Handle.h" - -#include "L1Trigger/TrackTrigger/interface/Setup.h" -#include "L1Trigger/TrackerTFP/interface/DataFormats.h" -#include "L1Trigger/TrackTrigger/interface/L1TrackQuality.h" - -#include -#include - -using namespace std; -using namespace edm; -using namespace trackerTFP; -using namespace tt; - -namespace trklet { - - /*! \class trklet::ProducerKFout - * \brief Converts KF output into tttrack collection and TFP output - * A bit accurate emulation of the track transformation, the - * eta routing and splitting of the 96-bit track words into 64-bit - * packets. Also run is a bit accurate emulation of the track quality - * BDT, whose output is also added to the track word. - * \author Christopher Brown - * \date 2021, Aug - * \update 2024, June by Claire Savard - */ - class ProducerKFout : public stream::EDProducer<> { - public: - explicit ProducerKFout(const ParameterSet&); - ~ProducerKFout() override {} - - private: - void beginRun(const Run&, const EventSetup&) override; - void produce(Event&, const EventSetup&) override; - void endJob() {} - - // ED input token of kf stubs - EDGetTokenT edGetTokenStubs_; - // ED input token of kf tracks - EDGetTokenT edGetTokenTracks_; - // ED output token for accepted kfout tracks - EDPutTokenT edPutTokenAccepted_; - // ED output token for TTTracks - EDPutTokenT edPutTokenTTTracks_; - // ED output token for truncated kfout tracks - EDPutTokenT edPutTokenLost_; - // Setup token - ESGetToken esGetTokenSetup_; - // DataFormats token - ESGetToken esGetTokenDataFormats_; - // configuration - ParameterSet iConfig_; - // helper class to store configurations - const Setup* setup_; - // helper class to extract structured data from tt::Frames - const DataFormats* dataFormats_; - // Bins for dPhi/dZ use to create weight LUT - vector dPhiBins_; - vector dZBins_; - - std::unique_ptr trackQualityModel_; - double tqTanlScale_; - double tqZ0Scale_; - static constexpr double ap_fixed_rescale = 32.0; - - // For convenience and keeping readable code, accessed many times - int numWorkers_; - int partialTrackWordBits_; - - // Helper function to convert floating value to bin - template - unsigned int digitise(const T& bins, double value, double factor) { - unsigned int bin = 0; - for (unsigned int i = 0; i < bins.size() - 1; i++) { - if (value * factor > bins[i] && value * factor <= bins[i + 1]) - break; - bin++; - } - return bin; - } - }; - - ProducerKFout::ProducerKFout(const ParameterSet& iConfig) : iConfig_(iConfig) { - const string& labelKF = iConfig.getParameter("LabelKF"); - const string& branchStubs = iConfig.getParameter("BranchAcceptedStubs"); - const string& branchTracks = iConfig.getParameter("BranchAcceptedTracks"); - const string& branchTTTracks = iConfig.getParameter("BranchAcceptedTTTracks"); - const string& branchLost = iConfig.getParameter("BranchLostTracks"); - // book in- and output ED products - edGetTokenStubs_ = consumes(InputTag(labelKF, branchStubs)); - edGetTokenTracks_ = consumes(InputTag(labelKF, branchTracks)); - edPutTokenAccepted_ = produces(branchTracks); - edPutTokenTTTracks_ = produces(branchTTTracks); - edPutTokenLost_ = produces(branchLost); - // book ES products - esGetTokenSetup_ = esConsumes(); - esGetTokenDataFormats_ = esConsumes(); - // initial ES products - setup_ = nullptr; - dataFormats_ = nullptr; - - trackQualityModel_ = std::make_unique(iConfig.getParameter("TrackQualityPSet")); - edm::ParameterSet trackQualityPSset = iConfig.getParameter("TrackQualityPSet"); - tqTanlScale_ = trackQualityPSset.getParameter("tqemu_TanlScale"); - tqZ0Scale_ = trackQualityPSset.getParameter("tqemu_Z0Scale"); - } - - void ProducerKFout::beginRun(const Run& iRun, const EventSetup& iSetup) { - // helper class to store configurations - setup_ = &iSetup.getData(esGetTokenSetup_); - if (!setup_->configurationSupported()) - return; - // check process history if desired - if (iConfig_.getParameter("CheckHistory")) - setup_->checkHistory(iRun.processHistory()); - // helper class to extract structured data from tt::Frames - dataFormats_ = &iSetup.getData(esGetTokenDataFormats_); - - // Calculate 1/dz**2 and 1/dphi**2 bins for v0 and v1 weightings - float temp_dphi = 0.0; - float temp_dz = 0.0; - for (int i = 0; - i < pow(2, dataFormats_->width(Variable::dPhi, Process::kfin)) / pow(2, setup_->weightBinFraction()); - i++) { - temp_dphi = - pow(dataFormats_->base(Variable::dPhi, Process::kfin) * (i + 1) * pow(2, setup_->weightBinFraction()), -2); - temp_dphi = temp_dphi / setup_->dphiTruncation(); - temp_dphi = std::floor(temp_dphi); - dPhiBins_.push_back(temp_dphi * setup_->dphiTruncation()); - } - for (int i = 0; i < pow(2, dataFormats_->width(Variable::dZ, Process::kfin)) / pow(2, setup_->weightBinFraction()); - i++) { - temp_dz = - pow(dataFormats_->base(Variable::dZ, Process::kfin) * (i + 1) * pow(2, setup_->weightBinFraction()), -2); - temp_dz = temp_dz * setup_->dzTruncation(); - temp_dz = std::ceil(temp_dz); - dZBins_.push_back(temp_dz / setup_->dzTruncation()); - } - numWorkers_ = setup_->kfNumWorker(); - partialTrackWordBits_ = TTBV::S_ / 2; - } - - void ProducerKFout::produce(Event& iEvent, const EventSetup& iSetup) { - // empty KFout product - StreamsTrack accepted(setup_->numRegions() * setup_->tfpNumChannel()); - StreamsTrack lost(setup_->numRegions() * setup_->tfpNumChannel()); - // read in KF Product and produce KFout product - if (setup_->configurationSupported()) { - Handle handleStubs; - iEvent.getByToken(edGetTokenStubs_, handleStubs); - const StreamsStub& streamsStubs = *handleStubs.product(); - Handle handleTracks; - iEvent.getByToken(edGetTokenTracks_, handleTracks); - const StreamsTrack& streamsTracks = *handleTracks.product(); - - // Setup KFout track collection - TrackKFOutSAPtrCollection KFoutTracks; - - // Setup containers for track quality - float tempTQMVAPreSig = 0.0; - // Due to ap_fixed implementation in CMSSW this 10,5 must be specified at compile time, TODO make this a changeable parameter - std::vector> trackQuality_inputs = {0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0}; - - // calculate track quality and fill TTTracks - TTTracks ttTracks; - int nTracks(0); - for (const StreamTrack& stream : streamsTracks) - nTracks += accumulate(stream.begin(), stream.end(), 0, [](int sum, const FrameTrack& frame) { - return sum + (frame.first.isNonnull() ? 1 : 0); - }); - ttTracks.reserve(nTracks); - for (int iLink = 0; iLink < (int)streamsTracks.size(); iLink++) { - for (int iTrack = 0; iTrack < (int)streamsTracks[iLink].size(); iTrack++) { - const auto& track = streamsTracks[iLink].at(iTrack); - TrackKF inTrack(track, dataFormats_); - - double temp_z0 = inTrack.zT() - ((inTrack.cot() * setup_->chosenRofZ())); - // Correction to Phi calcuation depending if +ve/-ve phi sector - const double baseSectorCorr = inTrack.sectorPhi() ? -setup_->baseSector() : setup_->baseSector(); - double temp_phi0 = inTrack.phiT() - ((inTrack.inv2R()) * setup_->hybridChosenRofPhi()) + baseSectorCorr; - double temp_tanL = inTrack.cotGlobal(); - - TTBV hitPattern(0, setup_->numLayers()); - double tempchi2rphi = 0; - double tempchi2rz = 0; - int temp_nstub = 0; - int temp_ninterior = 0; - bool counter = false; - - vector stubs; - stubs.reserve(setup_->numLayers()); - for (int iStub = 0; iStub < setup_->numLayers(); iStub++) { - const auto& stub = streamsStubs[setup_->numLayers() * iLink + iStub].at(iTrack); - StubKF inStub(stub, dataFormats_, iStub); - if (stub.first.isNonnull()) - stubs.emplace_back(stub, dataFormats_, iStub); - - if (!stub.first.isNonnull()) { - if (counter) - temp_ninterior += 1; - continue; - } - - counter = true; - - hitPattern.set(iStub); - temp_nstub += 1; - double phiSquared = pow(inStub.phi(), 2); - double zSquared = pow(inStub.z(), 2); - - double tempv0 = dPhiBins_[(inStub.dPhi() / (dataFormats_->base(Variable::dPhi, Process::kfin) * - pow(2, setup_->weightBinFraction())))]; - double tempv1 = dZBins_[( - inStub.dZ() / (dataFormats_->base(Variable::dZ, Process::kfin) * pow(2, setup_->weightBinFraction())))]; - - double tempRphi = phiSquared * tempv0; - double tempRz = zSquared * tempv1; - - tempchi2rphi += tempRphi; - tempchi2rz += tempRz; - } // Iterate over track stubs - - // Create bit vectors for each output, including digitisation of chi2 - // TODO implement extraMVA, bendChi2, d0 - TTBV trackValid(1, TTTrack_TrackWord::TrackBitWidths::kValidSize, false); - TTBV extraMVA(0, TTTrack_TrackWord::TrackBitWidths::kMVAOtherSize, false); - TTBV bendChi2(0, TTTrack_TrackWord::TrackBitWidths::kBendChi2Size, false); - TTBV chi2rphi(digitise(TTTrack_TrackWord::chi2RPhiBins, tempchi2rphi, (double)setup_->kfoutchi2rphiConv()), - TTTrack_TrackWord::TrackBitWidths::kChi2RPhiSize, - false); - TTBV chi2rz(digitise(TTTrack_TrackWord::chi2RZBins, tempchi2rz, (double)setup_->kfoutchi2rzConv()), - TTTrack_TrackWord::TrackBitWidths::kChi2RZSize, - false); - TTBV d0(0, TTTrack_TrackWord::TrackBitWidths::kD0Size, false); - TTBV z0( - temp_z0, dataFormats_->base(Variable::zT, Process::kf), TTTrack_TrackWord::TrackBitWidths::kZ0Size, true); - TTBV tanL(temp_tanL, - dataFormats_->base(Variable::cot, Process::kf), - TTTrack_TrackWord::TrackBitWidths::kTanlSize, - true); - TTBV phi0(temp_phi0, - dataFormats_->base(Variable::phiT, Process::kf), - TTTrack_TrackWord::TrackBitWidths::kPhiSize, - true); - TTBV invR(-inTrack.inv2R(), - dataFormats_->base(Variable::inv2R, Process::kf), - TTTrack_TrackWord::TrackBitWidths::kRinvSize + 1, - true); - invR.resize(TTTrack_TrackWord::TrackBitWidths::kRinvSize); - - // conversion to tttrack to calculate bendchi2 - // temporary fix for MVA1 while bendchi2 not implemented - TTTrack temp_tttrack = inTrack.ttTrack(stubs); - double tempbendchi2 = temp_tttrack.chi2BendRed(); - - // Create input vector for BDT - trackQuality_inputs = { - (std::trunc(tanL.val() / tqTanlScale_)) / ap_fixed_rescale, - (std::trunc(z0.val() / tqZ0Scale_)) / ap_fixed_rescale, - digitise(TTTrack_TrackWord::bendChi2Bins, tempbendchi2, 1.), - temp_nstub, - temp_ninterior, - digitise(TTTrack_TrackWord::chi2RPhiBins, tempchi2rphi, (double)setup_->kfoutchi2rphiConv()), - digitise(TTTrack_TrackWord::chi2RZBins, tempchi2rz, (double)setup_->kfoutchi2rzConv())}; - - // Run BDT emulation and package output into 3 bits - // output needs sigmoid transformation applied - tempTQMVAPreSig = trackQualityModel_->runEmulatedTQ(trackQuality_inputs); - TTBV tqMVA(digitise(L1TrackQuality::getTqMVAPreSigBins(), tempTQMVAPreSig, 1.0), - TTTrack_TrackWord::TrackBitWidths::kMVAQualitySize, - false); - - // Build 32 bit partial tracks for outputting in 64 bit packets - // 12 + 3 + 7 + 3 + 6 - TTBV partialTrack3((d0 + bendChi2 + hitPattern + tqMVA + extraMVA), partialTrackWordBits_, false); - // 16 + 12 + 4 - TTBV partialTrack2((tanL + z0 + chi2rz), partialTrackWordBits_, false); - // 1 + 15 + 12 + 4 - TTBV partialTrack1((trackValid + invR + phi0 + chi2rphi), partialTrackWordBits_, false); - - int sortKey = (inTrack.sectorEta() < (int)(setup_->numSectorsEta() / 2)) ? 0 : 1; - int nonantId = iLink / setup_->kfNumWorker(); - // Set correct bit to valid for track valid - TrackKFOut temp_track(partialTrack1.set((partialTrackWordBits_ - 1)), - partialTrack2, - partialTrack3, - sortKey, - nonantId, - track, - iTrack, - iLink, - true); - KFoutTracks.push_back(std::make_shared(temp_track)); - - // add MVA to tttrack and add tttrack to collection - temp_tttrack.settrkMVA1(1. / (1. + exp(tempTQMVAPreSig))); - temp_tttrack.setTrackWordBits(); - ttTracks.emplace_back(temp_tttrack); - } // Iterate over Tracks - } // Iterate over Links - const OrphanHandle orphanHandleTTTracks = iEvent.emplace(edPutTokenTTTracks_, std::move(ttTracks)); - - // sort partial KFout tracks into 18 separate links (nonant idx * eta idx) with tttrack ref info - // 0th index order: [nonant 0 + negative eta, nonant 0 + positive eta, nonant 1 + negative eta, ...] - struct kfoTrack_info { - TTBV partialBits; - TTTrackRef trackRef; - }; - vector> sortedPartialTracks(setup_->numRegions() * setup_->tfpNumChannel(), - vector(0)); - for (int i = 0; i < (int)KFoutTracks.size(); i++) { - auto& kfoTrack = KFoutTracks.at(i); - if (kfoTrack->dataValid()) { - sortedPartialTracks[kfoTrack->nonantId() * setup_->tfpNumChannel() + kfoTrack->sortKey()].push_back( - {kfoTrack->PartialTrack1(), TTTrackRef(orphanHandleTTTracks, i)}); - sortedPartialTracks[kfoTrack->nonantId() * setup_->tfpNumChannel() + kfoTrack->sortKey()].push_back( - {kfoTrack->PartialTrack2(), TTTrackRef(orphanHandleTTTracks, i)}); - sortedPartialTracks[kfoTrack->nonantId() * setup_->tfpNumChannel() + kfoTrack->sortKey()].push_back( - {kfoTrack->PartialTrack3(), TTTrackRef(orphanHandleTTTracks, i)}); - } - } - // fill remaining tracks allowed on each link (setup_->numFramesIO()) with null info - kfoTrack_info nullTrack_info; - for (int i = 0; i < (int)sortedPartialTracks.size(); i++) { - // will not fill if any additional tracks if already above limit - while ((int)sortedPartialTracks.at(i).size() < setup_->numFramesIO() * 2) - sortedPartialTracks.at(i).push_back(nullTrack_info); - } - - // combine sorted partial tracks into proper format: - // < TTTrackRef A, first 64 A bits > - // < TTTrackRef B, last 32 A bits + first 32 B bits > - // < TTTrackRef null, last 64 B bits > - // ... repeat for next tracks - const TTBV nullPartialBits(0, partialTrackWordBits_, false); - const TTTrackRef nullTrackRef; - int partialFactor = TTBV::S_ / partialTrackWordBits_; //how many partial track words to combine in an output - for (int iLink = 0; iLink < (int)sortedPartialTracks.size(); iLink++) { - for (int iTrack = 0; iTrack < (int)sortedPartialTracks[iLink].size(); iTrack += partialFactor) { - // if a partial track has no pair, pair it with null partial track - if (iTrack + 1 == (int)sortedPartialTracks[iLink].size()) - sortedPartialTracks[iLink].push_back({nullPartialBits, nullTrackRef}); - // keep TTTrackRef null every third (96 bits / 32 partial bits) output packet - TTTrackRef fillTrackRef; - if ((iTrack / partialFactor + 1) % (TTTrack_TrackWord::kTrackWordSize / partialTrackWordBits_) != 0) - fillTrackRef = sortedPartialTracks[iLink][iTrack + 1].trackRef; - - // if there are too many output packets, truncate and put remaining outputs in lost collection - if (iTrack / partialFactor < setup_->numFramesIO()) - accepted[iLink].emplace_back( - std::make_pair(fillTrackRef, - (sortedPartialTracks[iLink][iTrack].partialBits.slice(partialTrackWordBits_) + - sortedPartialTracks[iLink][iTrack + 1].partialBits.slice(partialTrackWordBits_)) - .bs())); - else - lost[iLink].emplace_back( - std::make_pair(fillTrackRef, - (sortedPartialTracks[iLink][iTrack].partialBits.slice(partialTrackWordBits_) + - sortedPartialTracks[iLink][iTrack + 1].partialBits.slice(partialTrackWordBits_)) - .bs())); - } - } - } // Config Supported - - // store products - iEvent.emplace(edPutTokenAccepted_, std::move(accepted)); - iEvent.emplace(edPutTokenLost_, std::move(lost)); - } -} // namespace trklet - -DEFINE_FWK_MODULE(trklet::ProducerKFout); diff --git a/L1Trigger/TrackFindingTracklet/plugins/ProducerTBout.cc b/L1Trigger/TrackFindingTracklet/plugins/ProducerTBout.cc deleted file mode 100644 index 981ed2ba75c6b..0000000000000 --- a/L1Trigger/TrackFindingTracklet/plugins/ProducerTBout.cc +++ /dev/null @@ -1,192 +0,0 @@ -#include "FWCore/Framework/interface/stream/EDProducer.h" -#include "FWCore/Framework/interface/Run.h" -#include "FWCore/Framework/interface/EventSetup.h" -#include "FWCore/Framework/interface/Event.h" -#include "FWCore/Framework/interface/MakerMacros.h" -#include "FWCore/Utilities/interface/EDGetToken.h" -#include "FWCore/Utilities/interface/EDPutToken.h" -#include "FWCore/Utilities/interface/ESGetToken.h" -#include "FWCore/Utilities/interface/InputTag.h" -#include "FWCore/ParameterSet/interface/ParameterSet.h" -#include "DataFormats/Common/interface/Handle.h" - -#include "L1Trigger/TrackTrigger/interface/Setup.h" -#include "L1Trigger/TrackerTFP/interface/DataFormats.h" -#include "L1Trigger/TrackFindingTracklet/interface/ChannelAssignment.h" - -#include -#include -#include -#include -#include -#include - -using namespace std; -using namespace edm; -using namespace trackerTFP; -using namespace tt; - -namespace trklet { - - /*! \class trklet::ProducerTBout - * \brief Transforms TTTracks and Streams from Tracklet pattern reco. into StreamsTrack - * by adding to the digitised track stream a reference to the corresponding TTTrack. - * (Could not be done in previous L1TrackFPGAProducer, as single EDProducer can't - * produce output containing both an EDProduct and refs to that product). - * Writes Tracks & stubs rejected/kept after truncation to separate StreamsTrack & StreamsStub branches. - * \author Thomas Schuh - * \date 2021, Oct - */ - class ProducerTBout : public stream::EDProducer<> { - public: - explicit ProducerTBout(const ParameterSet&); - ~ProducerTBout() override {} - - private: - void beginRun(const Run&, const EventSetup&) override; - void produce(Event&, const EventSetup&) override; - virtual void endJob() {} - - // ED input token of TTTracks - EDGetTokenT edGetTokenTTTracks_; - // ED input token of Tracklet tracks - EDGetTokenT edGetTokenTracks_; - // ED input token of Tracklet Stubs - EDGetTokenT edGetTokenStubs_; - // ED output token for stubs - EDPutTokenT edPutTokenAcceptedStubs_; - EDPutTokenT edPutTokenLostStubs_; - // ED output token for tracks - EDPutTokenT edPutTokenAcceptedTracks_; - EDPutTokenT edPutTokenLostTracks_; - // Setup token - ESGetToken esGetTokenSetup_; - // DataFormats token - ESGetToken esGetTokenDataFormats_; - // ChannelAssignment token - ESGetToken esGetTokenChannelAssignment_; - // configuration - ParameterSet iConfig_; - // helper class to store configurations - const Setup* setup_ = nullptr; - // helper class to extract structured data from tt::Frames - const DataFormats* dataFormats_ = nullptr; - // helper class to assign tracks to channel - ChannelAssignment* channelAssignment_ = nullptr; - // - bool enableTruncation_; - }; - - ProducerTBout::ProducerTBout(const ParameterSet& iConfig) : iConfig_(iConfig) { - const InputTag& inputTag = iConfig.getParameter("InputTag"); - const string& branchAcceptedStubs = iConfig.getParameter("BranchAcceptedStubs"); - const string& branchAcceptedTracks = iConfig.getParameter("BranchAcceptedTracks"); - const string& branchLostStubs = iConfig.getParameter("BranchLostStubs"); - const string& branchLostTracks = iConfig.getParameter("BranchLostTracks"); - // book in- and output ED products - edGetTokenTTTracks_ = consumes(inputTag); - edGetTokenTracks_ = consumes(inputTag); - edGetTokenStubs_ = consumes(inputTag); - edPutTokenAcceptedStubs_ = produces(branchAcceptedStubs); - edPutTokenAcceptedTracks_ = produces(branchAcceptedTracks); - edPutTokenLostStubs_ = produces(branchLostStubs); - edPutTokenLostTracks_ = produces(branchLostTracks); - // book ES products - esGetTokenSetup_ = esConsumes(); - esGetTokenDataFormats_ = esConsumes(); - esGetTokenChannelAssignment_ = esConsumes(); - // - enableTruncation_ = iConfig.getParameter("EnableTruncation"); - } - - void ProducerTBout::beginRun(const Run& iRun, const EventSetup& iSetup) { - // helper class to store configurations - setup_ = &iSetup.getData(esGetTokenSetup_); - if (!setup_->configurationSupported()) - return; - // check process history if desired - if (iConfig_.getParameter("CheckHistory")) - setup_->checkHistory(iRun.processHistory()); - // helper class to extract structured data from tt::Frames - dataFormats_ = &iSetup.getData(esGetTokenDataFormats_); - // helper class to assign tracks to channel - channelAssignment_ = const_cast(&iSetup.getData(esGetTokenChannelAssignment_)); - } - - void ProducerTBout::produce(Event& iEvent, const EventSetup& iSetup) { - const int numStreamsTracks = setup_->numRegions() * channelAssignment_->numChannelsTrack(); - const int numStreamsStubs = setup_->numRegions() * channelAssignment_->numChannelsStub(); - // empty KFin products - StreamsStub streamAcceptedStubs(numStreamsStubs); - StreamsTrack streamAcceptedTracks(numStreamsTracks); - StreamsStub streamLostStubs(numStreamsStubs); - StreamsTrack streamLostTracks(numStreamsTracks); - // read in hybrid track finding product and produce KFin product - if (setup_->configurationSupported()) { - // create and structure TTrackRefs in h/w channel - vector> ttTrackRefs(numStreamsTracks); - Handle handleTTTracks; - iEvent.getByToken(edGetTokenTTTracks_, handleTTTracks); - int channelId(-1); - for (int i = 0; i < (int)handleTTTracks->size(); i++) { - const TTTrackRef ttTrackRef(handleTTTracks, i); - const int channelId = channelAssignment_->channelId(ttTrackRef); - ttTrackRefs[channelId].push_back(ttTrackRef); - } - // get and trunacte tracks - Handle handleTracks; - iEvent.getByToken(edGetTokenTracks_, handleTracks); - channelId = 0; - for (const Stream& streamTrack : *handleTracks) { - const int nTracks = accumulate( - streamTrack.begin(), streamTrack.end(), 0, [](int sum, const Frame& f) { return sum + (f.any() ? 1 : 0); }); - StreamTrack& accepted = streamAcceptedTracks[channelId]; - StreamTrack& lost = streamLostTracks[channelId]; - auto limit = streamTrack.end(); - if (enableTruncation_ && (int)streamTrack.size() > setup_->numFrames()) - limit = next(streamTrack.begin(), setup_->numFrames()); - accepted.reserve(distance(streamTrack.begin(), limit)); - lost.reserve(distance(limit, streamTrack.end())); - int nFrame(0); - const deque& ttTracks = ttTrackRefs[channelId++]; - if ((int)ttTracks.size() != nTracks) { - cms::Exception exception("LogicError."); - const int region = channelId / channelAssignment_->numChannelsTrack(); - const int channel = channelId % channelAssignment_->numChannelsTrack(); - exception << "Region " << region << " output channel " << channel << " has " << nTracks - << " tracks found but created " << ttTracks.size() << " TTTracks."; - exception.addContext("trklet::ProducerTBout::produce"); - throw exception; - } - auto toFrameTrack = [&nFrame, &ttTracks](const Frame& frame) { - if (frame.any()) - return FrameTrack(ttTracks[nFrame++], frame); - return FrameTrack(); - }; - transform(streamTrack.begin(), limit, back_inserter(accepted), toFrameTrack); - transform(limit, streamTrack.end(), back_inserter(lost), toFrameTrack); - } - // get and trunacte stubs - Handle handleStubs; - iEvent.getByToken(edGetTokenStubs_, handleStubs); - const StreamsStub& streamsStub = *handleStubs; - // reserve output ed products - channelId = 0; - for (const StreamStub& streamStub : streamsStub) { - auto limit = streamStub.end(); - if (enableTruncation_ && (int)streamStub.size() > setup_->numFrames()) - limit = next(streamStub.begin(), setup_->numFrames()); - streamAcceptedStubs[channelId] = StreamStub(streamStub.begin(), limit); - streamLostStubs[channelId++] = StreamStub(limit, streamStub.end()); - } - } - // store products - iEvent.emplace(edPutTokenAcceptedStubs_, std::move(streamAcceptedStubs)); - iEvent.emplace(edPutTokenAcceptedTracks_, std::move(streamAcceptedTracks)); - iEvent.emplace(edPutTokenLostStubs_, std::move(streamLostStubs)); - iEvent.emplace(edPutTokenLostTracks_, std::move(streamLostTracks)); - } - -} // namespace trklet - -DEFINE_FWK_MODULE(trklet::ProducerTBout); diff --git a/L1Trigger/TrackFindingTracklet/plugins/ProducerTM.cc b/L1Trigger/TrackFindingTracklet/plugins/ProducerTM.cc new file mode 100644 index 0000000000000..bc487ae8c766c --- /dev/null +++ b/L1Trigger/TrackFindingTracklet/plugins/ProducerTM.cc @@ -0,0 +1,127 @@ +#include "FWCore/Framework/interface/stream/EDProducer.h" +#include "FWCore/Framework/interface/Run.h" +#include "FWCore/Framework/interface/EventSetup.h" +#include "FWCore/Framework/interface/Event.h" +#include "FWCore/Framework/interface/MakerMacros.h" +#include "FWCore/Utilities/interface/EDGetToken.h" +#include "FWCore/Utilities/interface/EDPutToken.h" +#include "FWCore/Utilities/interface/ESGetToken.h" +#include "FWCore/Utilities/interface/InputTag.h" +#include "FWCore/ParameterSet/interface/ParameterSet.h" +#include "DataFormats/Common/interface/Handle.h" + +#include "L1Trigger/TrackTrigger/interface/Setup.h" +#include "L1Trigger/TrackFindingTracklet/interface/ChannelAssignment.h" +#include "L1Trigger/TrackFindingTracklet/interface/DataFormats.h" +#include "L1Trigger/TrackFindingTracklet/interface/Settings.h" +#include "L1Trigger/TrackFindingTracklet/interface/TrackMultiplexer.h" +#include "SimTracker/TrackTriggerAssociation/interface/TTTypes.h" + +#include +#include +#include +#include +#include +#include + +using namespace std; +using namespace edm; +using namespace tt; + +namespace trklet { + + /*! \class trklet::ProducerTM + * \brief Transforms format of Track Builder into that expected by DR input and muxes all channels to 1. + Since DR keeps first tracks the mux ordering (currently from low seed id to high seed id) is important. + * \author Thomas Schuh + * \date 2023, Jan + */ + class ProducerTM : public stream::EDProducer<> { + public: + explicit ProducerTM(const ParameterSet&); + ~ProducerTM() override {} + + private: + void beginRun(const Run&, const EventSetup&) override; + void produce(Event&, const EventSetup&) override; + virtual void endJob() {} + + // ED input token of Tracks + EDGetTokenT edGetTokenTracks_; + // ED input token of Stubs + EDGetTokenT edGetTokenStubs_; + // ED output token for stubs + EDPutTokenT edPutTokenStubs_; + // ED output token for tracks + EDPutTokenT edPutTokenTracks_; + // Setup token + ESGetToken esGetTokenSetup_; + // DataFormats token + ESGetToken esGetTokenDataFormats_; + // ChannelAssignment token + ESGetToken esGetTokenChannelAssignment_; + // configuration + ParameterSet iConfig_; + // helper class to store configurations + const Setup* setup_ = nullptr; + // helper class to extract structured data from tt::Frames + const DataFormats* dataFormats_ = nullptr; + // helper class to assign tracks to channel + const ChannelAssignment* channelAssignment_ = nullptr; + // helper class to store tracklet configurations + Settings settings_; + }; + + ProducerTM::ProducerTM(const ParameterSet& iConfig) : iConfig_(iConfig) { + const string& label = iConfig.getParameter("InputLabelTM"); + const string& branchStubs = iConfig.getParameter("BranchStubs"); + const string& branchTracks = iConfig.getParameter("BranchTracks"); + // book in- and output ED products + edGetTokenTracks_ = consumes(InputTag(label, branchTracks)); + edGetTokenStubs_ = consumes(InputTag(label, branchStubs)); + edPutTokenStubs_ = produces(branchStubs); + edPutTokenTracks_ = produces(branchTracks); + // book ES products + esGetTokenSetup_ = esConsumes(); + esGetTokenDataFormats_ = esConsumes(); + esGetTokenChannelAssignment_ = esConsumes(); + } + + void ProducerTM::beginRun(const Run& iRun, const EventSetup& iSetup) { + // helper class to store configurations + setup_ = &iSetup.getData(esGetTokenSetup_); + // helper class to extract structured data from tt::Frames + dataFormats_ = &iSetup.getData(esGetTokenDataFormats_); + // helper class to assign tracks to channel + channelAssignment_ = &iSetup.getData(esGetTokenChannelAssignment_); + } + + void ProducerTM::produce(Event& iEvent, const EventSetup& iSetup) { + // empty TM products + const int numStreamsTracks = setup_->numRegions(); + const int numStreamsStubs = numStreamsTracks * channelAssignment_->tmNumLayers(); + StreamsStub streamsStub(numStreamsStubs); + StreamsTrack streamsTrack(numStreamsTracks); + // read in TBout Product and produce TM product + Handle handleStubs; + iEvent.getByToken(edGetTokenStubs_, handleStubs); + const StreamsStub& stubs = *handleStubs; + Handle handleTracks; + iEvent.getByToken(edGetTokenTracks_, handleTracks); + const StreamsTrack& tracks = *handleTracks; + for (int region = 0; region < setup_->numRegions(); region++) { + // object to reformat tracks from tracklet fromat to TMTT format in a processing region + TrackMultiplexer tm(iConfig_, setup_, dataFormats_, channelAssignment_, &settings_, region); + // read in and organize input tracks and stubs + tm.consume(tracks, stubs); + // fill output products + tm.produce(streamsTrack, streamsStub); + } + // store products + iEvent.emplace(edPutTokenTracks_, move(streamsTrack)); + iEvent.emplace(edPutTokenStubs_, move(streamsStub)); + } + +} // namespace trklet + +DEFINE_FWK_MODULE(trklet::ProducerTM); diff --git a/L1Trigger/TrackFindingTracklet/python/Analyzer_cff.py b/L1Trigger/TrackFindingTracklet/python/Analyzer_cff.py index ddc1b5377044a..f38d6311b0a57 100644 --- a/L1Trigger/TrackFindingTracklet/python/Analyzer_cff.py +++ b/L1Trigger/TrackFindingTracklet/python/Analyzer_cff.py @@ -1,12 +1,12 @@ +# EDAnalyzer to analyze hybrid track reconstruction emulation chain + import FWCore.ParameterSet.Config as cms from L1Trigger.TrackFindingTracklet.Analyzer_cfi import TrackFindingTrackletAnalyzer_params from L1Trigger.TrackFindingTracklet.Producer_cfi import TrackFindingTrackletProducer_params -TrackFindingTrackletAnalyzerTBout = cms.EDAnalyzer( 'trklet::AnalyzerTBout', TrackFindingTrackletAnalyzer_params, TrackFindingTrackletProducer_params ) -TrackFindingTrackletAnalyzerTracklet = cms.EDAnalyzer( 'trklet::AnalyzerTracklet', TrackFindingTrackletAnalyzer_params, TrackFindingTrackletProducer_params ) -TrackFindingTrackletAnalyzerDRin = cms.EDAnalyzer( 'trklet::AnalyzerDRin', TrackFindingTrackletAnalyzer_params, TrackFindingTrackletProducer_params ) -TrackFindingTrackletAnalyzerDR = cms.EDAnalyzer( 'trklet::AnalyzerDR', TrackFindingTrackletAnalyzer_params, TrackFindingTrackletProducer_params ) -TrackFindingTrackletAnalyzerKFin = cms.EDAnalyzer( 'trklet::AnalyzerKFin', TrackFindingTrackletAnalyzer_params, TrackFindingTrackletProducer_params ) -TrackFindingTrackletAnalyzerKF = cms.EDAnalyzer( 'trackerTFP::AnalyzerKF', TrackFindingTrackletAnalyzer_params, TrackFindingTrackletProducer_params ) -TrackFindingTrackletAnalyzerKFout = cms.EDAnalyzer( 'trklet::AnalyzerKFout', TrackFindingTrackletAnalyzer_params, TrackFindingTrackletProducer_params ) +AnalyzerTracklet = cms.EDAnalyzer( 'trklet::AnalyzerTracklet', TrackFindingTrackletAnalyzer_params, TrackFindingTrackletProducer_params ) +AnalyzerTM = cms.EDAnalyzer( 'trklet::AnalyzerTM', TrackFindingTrackletAnalyzer_params, TrackFindingTrackletProducer_params ) +AnalyzerDR = cms.EDAnalyzer( 'trklet::AnalyzerDR', TrackFindingTrackletAnalyzer_params, TrackFindingTrackletProducer_params ) +AnalyzerKF = cms.EDAnalyzer( 'trklet::AnalyzerKF', TrackFindingTrackletAnalyzer_params, TrackFindingTrackletProducer_params ) +AnalyzerTFP = cms.EDAnalyzer( 'trklet::AnalyzerTFP', TrackFindingTrackletAnalyzer_params, TrackFindingTrackletProducer_params ) diff --git a/L1Trigger/TrackFindingTracklet/python/Analyzer_cfi.py b/L1Trigger/TrackFindingTracklet/python/Analyzer_cfi.py index af2a269ca3793..8a5ae3a29c8be 100644 --- a/L1Trigger/TrackFindingTracklet/python/Analyzer_cfi.py +++ b/L1Trigger/TrackFindingTracklet/python/Analyzer_cfi.py @@ -1,4 +1,5 @@ -# ParameterSet used by AnalyzerKFin and AnalyzerTracklet +# configuration for hybrid track reconstruction chain analyzer + import FWCore.ParameterSet.Config as cms TrackFindingTrackletAnalyzer_params = cms.PSet ( @@ -6,6 +7,12 @@ UseMCTruth = cms.bool( True ), # enables analyze of TPs InputTagReconstructable = cms.InputTag("StubAssociator", "Reconstructable"), # InputTagSelection = cms.InputTag("StubAssociator", "UseForAlgEff"), # + InputTag = cms.InputTag( "l1tTTTracksFromTrackletEmulation", "Level1TTTracks"), # + OutputLabelTM = cms.string ( "ProducerTM" ), # + OutputLabelDR = cms.string ( "ProducerDR" ), # + OutputLabelKF = cms.string ( "ProducerKF" ), # + OutputLabelTQ = cms.string ( "ProducerTQ" ), # + OutputLabelTFP = cms.string ( "ProducerTFP" ), # ) diff --git a/L1Trigger/TrackFindingTracklet/python/ChannelAssignment_cff.py b/L1Trigger/TrackFindingTracklet/python/ChannelAssignment_cff.py index adb369d78606f..46df2cd3cf444 100644 --- a/L1Trigger/TrackFindingTracklet/python/ChannelAssignment_cff.py +++ b/L1Trigger/TrackFindingTracklet/python/ChannelAssignment_cff.py @@ -1,3 +1,5 @@ +# ESProducer providing the algorithm to assign tracklet tracks and stubs to output channel based on their Pt or seed type as well as DTC stubs to input channel + import FWCore.ParameterSet.Config as cms from L1Trigger.TrackFindingTracklet.ChannelAssignment_cfi import ChannelAssignment_params diff --git a/L1Trigger/TrackFindingTracklet/python/ChannelAssignment_cfi.py b/L1Trigger/TrackFindingTracklet/python/ChannelAssignment_cfi.py index 64bee96e8f6bf..91023650a2e3e 100644 --- a/L1Trigger/TrackFindingTracklet/python/ChannelAssignment_cfi.py +++ b/L1Trigger/TrackFindingTracklet/python/ChannelAssignment_cfi.py @@ -1,21 +1,19 @@ -# defines PSet to assign tracklet tracks and stubs to output channel based on their Pt or seed type as well as DTC stubs to input channel +# configuration for ChannelAssignment + import FWCore.ParameterSet.Config as cms ChannelAssignment_params = cms.PSet ( - # DRin parameter - DRin = cms.PSet ( - WidthLayerId = cms.int32( 4 ), # number of bits used to represent layer id [barrel: 0-5, discs: 6-10] - WidthStubId = cms.int32( 10 ), # number of bits used to represent stub id for projected stubs - WidthSeedStubId = cms.int32( 7 ), # number of bits used to represent stub id for seed stubs - WidthPSTilt = cms.int32( 1 ), # number of bits used to distinguish between tilted and untilded barrel modules or 2S and PS endcap modules - DepthMemory = cms.int32( 32 ), # depth of fifos within systolic array - PtBoundaries = cms.vdouble( 3.0, 5.0, 8.0, 12.0, 24.0 ) # positive pt Boundaries in GeV (symmetric negatives are assumed), first boundary is pt cut, last boundary is infinity, defining pt bins used by DR + # TM parameter + TM = cms.PSet ( + NumLayers = cms.int32( 11 ), # number of layers per track + WidthStubId = cms.int32( 10 ), # number of bits used to represent stub id for projected stubs + WidthCot = cms.int32( 14 ) ), # DR parameter DR = cms.PSet ( - NumComparisonModules = cms.int32( 16 ), # number of comparison modules used in each DR node + NumComparisonModules = cms.int32( 32 ), # number of comparison modules used in each DR node MinIdenticalStubs = cms.int32( 3 ) # min number of shared stubs to identify duplicates ), diff --git a/L1Trigger/TrackFindingTracklet/python/Customize_cff.py b/L1Trigger/TrackFindingTracklet/python/Customize_cff.py index 6124b40f047e8..430e52bea0ba7 100644 --- a/L1Trigger/TrackFindingTracklet/python/Customize_cff.py +++ b/L1Trigger/TrackFindingTracklet/python/Customize_cff.py @@ -1,13 +1,14 @@ +# functions to alter configurations + import FWCore.ParameterSet.Config as cms # configures track finding s/w to use KF emulator instead of KF simulator def newKFConfig(process): - process.l1tTTTracksFromTrackletEmulation.Fakefit = True + process.l1tTTTracksFromTrackletEmulation.Fakefit = True # configures track finding s/w to behave as track finding f/w def fwConfig(process): newKFConfig(process) - process.TrackTriggerSetup.Firmware.FreqBE = 240 # Frequency of DTC & KF (determines truncation) process.l1tTTTracksFromTrackletEmulation.RemovalType = "" process.l1tTTTracksFromTrackletEmulation.DoMultipleMatches = False process.l1tTTTracksFromTrackletEmulation.StoreTrackBuilderOutput = True @@ -15,6 +16,7 @@ def fwConfig(process): # configures track finding s/w to behave as a subchain of processing steps def reducedConfig(process): fwConfig(process) + process.TrackTriggerSetup.Firmware.FreqBEHigh = 240 # Frequency of DTC & KF (determines truncation) process.TrackTriggerSetup.KalmanFilter.NumWorker = 1 process.ChannelAssignment.SeedTypes = cms.vstring( "L1L2" ) process.ChannelAssignment.SeedTypesSeedLayers = cms.PSet( L1L2 = cms.vint32( 1, 2 ) ) @@ -30,4 +32,44 @@ def reducedConfig(process): # configures pure tracklet algorithm (as opposed to Hybrid algorithm) def trackletConfig(process): - process.l1tTTTracksFromTrackletEmulation.fitPatternFile = cms.FileInPath('L1Trigger/TrackFindingTracklet/data/fitpattern.txt') + process.l1tTTTracksFromTrackletEmulation.fitPatternFile = cms.FileInPath('L1Trigger/TrackFindingTracklet/data/fitpattern.txt') + +# configures KF simulation in emulation chain +def oldKFConfig(process): + process.ProducerKF.Hybrid = True + process.ProducerKF.DeadModuleOpts.KillScenario = 0 + process.ProducerKF.DeadModuleOpts.KillRecover = False + process.ProducerKF.HTArraySpecRphi.HoughMinPt = 2. + process.ProducerKF.TrackDigi.KF_skipTrackDigi = True + process.ProducerKF.StubDigitize.EnableDigitize = False + process.ProducerKF.GeometricProc.UseApproxB = True + process.ProducerKF.GeometricProc.BApprox_gradient = 0.886454 + process.ProducerKF.GeometricProc.BApprox_intercept = 0.504148 + process.ProducerKF.PhiSectors.NumPhiSectors = 9 + process.ProducerKF.PhiSectors.NumPhiNonants = 9 + process.ProducerKF.PhiSectors.ChosenRofPhi = 55. + process.ProducerKF.EtaSectors.EtaRegions = [-2.4, -2.08, -1.68, -1.26, -0.90, -0.62, -0.41, -0.20, 0.0, 0.20, 0.41, 0.62, 0.90, 1.26, 1.68, 2.08, 2.4] + process.ProducerKF.EtaSectors.ChosenRofZ = 50.0 + process.ProducerKF.TrackFitSettings.KalmanMinNumStubs = 4 + process.ProducerKF.TrackFitSettings.KalmanMaxNumStubs = 6 + process.ProducerKF.TrackFitSettings.KalmanMaxSkipLayersHard = 1 + process.ProducerKF.TrackFitSettings.KalmanMaxSkipLayersEasy = 2 + process.ProducerKF.TrackFitSettings.KalmanMaxStubsEasy = 10 + process.ProducerKF.TrackFitSettings.KalmanMaxStubsPerLayer = 4 + process.ProducerKF.TrackFitSettings.KalmanMultiScattTerm = 0.00075 + process.ProducerKF.TrackFitSettings.KalmanChi2RphiScale = 8 + process.ProducerKF.TrackFitSettings.KFUseMaybeLayers = True + process.ProducerKF.TrackFitSettings.KalmanRemove2PScut = True + process.ProducerKF.TrackFitSettings.KFLayerVsPtToler = [999., 999., 0.1, 0.1, 0.05, 0.05, 0.05] + process.ProducerKF.TrackFitSettings.KFLayerVsD0Cut5 = [999., 999., 999., 10., 10., 10., 10.] + process.ProducerKF.TrackFitSettings.KFLayerVsZ0Cut5 = [999., 999., 25.5, 25.5, 25.5, 25.5, 25.5] + process.ProducerKF.TrackFitSettings.KFLayerVsZ0Cut4 = [999., 999., 15., 15., 15., 15., 15.] + process.ProducerKF.TrackFitSettings.KFLayerVsChiSq5 = [999., 999., 10., 30., 80., 120., 160.] + process.ProducerKF.TrackFitSettings.KFLayerVsChiSq4 = [999., 999., 10., 30., 80., 120., 160.] + process.ProducerKF.TrackFitSettings.KalmanAddBeamConstr = False + process.ProducerKF.TrackFitSettings.KalmanHOfw = False + process.ProducerKF.TrackFitSettings.KalmanHOtilted = True + process.ProducerKF.TrackFitSettings.KalmanHOprojZcorr = 1 + process.ProducerKF.TrackFitSettings.KalmanHOalpha = 0 + process.ProducerKF.TrackFitSettings.KalmanHOhelixExp = True + process.ProducerKF.TrackFitSettings.KalmanDebugLevel = 0 \ No newline at end of file diff --git a/L1Trigger/TrackFindingTracklet/python/DataFormats_cff.py b/L1Trigger/TrackFindingTracklet/python/DataFormats_cff.py new file mode 100644 index 0000000000000..5e7416487b67c --- /dev/null +++ b/L1Trigger/TrackFindingTracklet/python/DataFormats_cff.py @@ -0,0 +1,11 @@ +# ESProducer to provide and calculate and provide dataformats used by Hybrid emulator + +import FWCore.ParameterSet.Config as cms + +HybridDataFormats = cms.ESProducer("trklet::ProducerDataFormats") + +fakeHybridDataFormatsSource = cms.ESSource("EmptyESSource", + recordName = cms.string('trklet::DataFormatsRcd'), + iovIsRunNotTime = cms.bool(True), + firstValid = cms.vuint32(1) +) \ No newline at end of file diff --git a/L1Trigger/TrackFindingTracklet/python/Demonstrator_cff.py b/L1Trigger/TrackFindingTracklet/python/Demonstrator_cff.py index 4c01ad1e086a2..d14422e2ac499 100644 --- a/L1Trigger/TrackFindingTracklet/python/Demonstrator_cff.py +++ b/L1Trigger/TrackFindingTracklet/python/Demonstrator_cff.py @@ -1,3 +1,6 @@ +# ESProducer providing the algorithm to run input data through modelsim and to compares results with expected output data +# and EDAnalyzer running the ESProduct produced by above ESProducer + import FWCore.ParameterSet.Config as cms from L1Trigger.TrackFindingTracklet.Demonstrator_cfi import TrackTriggerDemonstrator_params diff --git a/L1Trigger/TrackFindingTracklet/python/Demonstrator_cfi.py b/L1Trigger/TrackFindingTracklet/python/Demonstrator_cfi.py index 8331a613b1813..1f02bb9d1e762 100644 --- a/L1Trigger/TrackFindingTracklet/python/Demonstrator_cfi.py +++ b/L1Trigger/TrackFindingTracklet/python/Demonstrator_cfi.py @@ -1,11 +1,18 @@ # configuration of Demonstrator. This is used to compare FW with SW for the subset fo the chain between LabelIn & LabelOut. FW must be wrapped by EMP & compiled with IPBB. import FWCore.ParameterSet.Config as cms +from L1Trigger.TrackFindingTracklet.Producer_cfi import TrackFindingTrackletProducer_params +from L1Trigger.TrackFindingTracklet.Analyzer_cfi import TrackFindingTrackletAnalyzer_params + +# these parameters a for ModelSim runs of FW TrackTriggerDemonstrator_params = cms.PSet ( - LabelIn = cms.string( "TrackFindingTrackletProducerDRin" ), # - LabelOut = cms.string( "TrackFindingTrackletProducerDR" ), # - DirIPBB = cms.string( "/heplnw039/tschuh/work/proj/DRinDR/" ), # path to ipbb proj area - RunTime = cms.double( 4.5 ) # runtime in us + LabelIn = TrackFindingTrackletProducer_params.InputLabelDR, # + LabelOut = TrackFindingTrackletAnalyzer_params.OutputLabelDR, # + DirIPBB = cms.string( "/heplnw039/tschuh/work/proj/tmdr/" ), # path to ipbb proj area + RunTime = cms.double( 7.5 ), # runtime in us + + LinkMappingIn = cms.vint32( ), + LinkMappingOut = cms.vint32( ) ) diff --git a/L1Trigger/TrackFindingTracklet/python/KalmanFilterFormats_cfi.py b/L1Trigger/TrackFindingTracklet/python/KalmanFilterFormats_cfi.py new file mode 100644 index 0000000000000..d22bbe5ffde91 --- /dev/null +++ b/L1Trigger/TrackFindingTracklet/python/KalmanFilterFormats_cfi.py @@ -0,0 +1,66 @@ +# configuration of internal KF variable bases which can be shifted by powers of 2 w.r.t. KF output track parameter +# TrackerTFPProducer_params.PrintKFDebug printouts unused MSB for each variable, so that one could consider decreasing the basseshift by that amount +# numerical instabillity (negative C00, C11, C22, C33) requires smaller baseshifts of related variables (rx, Sxx, Kxx, Rxx, invRxx) +# if a variable overflows an Exception will be thrown and the corresponding baseshift needs to be increased. +import FWCore.ParameterSet.Config as cms + +HybridKalmanFilterFormats_params = cms.PSet ( + + EnableIntegerEmulation = cms.bool( True ), + + WidthR00 = cms.int32( 20 ), + WidthR11 = cms.int32( 20 ), + + WidthC00 = cms.int32( 20 ), + WidthC01 = cms.int32( 20 ), + WidthC11 = cms.int32( 20 ), + WidthC22 = cms.int32( 20 ), + WidthC23 = cms.int32( 20 ), + WidthC33 = cms.int32( 20 ), + + BaseShiftx0 = cms.int32( -1 ), + BaseShiftx1 = cms.int32( -8 ), + BaseShiftx2 = cms.int32( -1 ), + BaseShiftx3 = cms.int32( -1 ), + + BaseShiftr0 = cms.int32( -8 ), + BaseShiftr1 = cms.int32( 0 ), + + BaseShiftS00 = cms.int32( -4 ), + BaseShiftS01 = cms.int32( -12 ), + BaseShiftS12 = cms.int32( 0 ), + BaseShiftS13 = cms.int32( -1 ), + + BaseShiftR00 = cms.int32( -5 ), + BaseShiftR11 = cms.int32( 6 ), + + BaseShiftInvR00Approx = cms.int32( -30 ), + BaseShiftInvR11Approx = cms.int32( -41 ), + BaseShiftInvR00Cor = cms.int32( -24 ), + BaseShiftInvR11Cor = cms.int32( -24 ), + BaseShiftInvR00 = cms.int32( -30 ), + BaseShiftInvR11 = cms.int32( -41 ), + + BaseShiftS00Shifted = cms.int32( -1 ), + BaseShiftS01Shifted = cms.int32( -7 ), + BaseShiftS12Shifted = cms.int32( 4 ), + BaseShiftS13Shifted = cms.int32( 4 ), + + BaseShiftK00 = cms.int32( -7 ), + BaseShiftK10 = cms.int32( -13 ), + BaseShiftK21 = cms.int32( -13 ), + BaseShiftK31 = cms.int32( -13 ), + + BaseShiftC00 = cms.int32( 6 ), + BaseShiftC01 = cms.int32( 1 ), + BaseShiftC11 = cms.int32( -6 ), + BaseShiftC22 = cms.int32( 5 ), + BaseShiftC23 = cms.int32( 6 ), + BaseShiftC33 = cms.int32( 5 ), + + BaseShiftr02 = cms.int32( -2 ), + BaseShiftr12 = cms.int32( 10 ), + BaseShiftchi20 = cms.int32( -10 ), + BaseShiftchi21 = cms.int32( -10 ) + +) \ No newline at end of file diff --git a/L1Trigger/TrackFindingTracklet/python/ProducerHPH_cff.py b/L1Trigger/TrackFindingTracklet/python/ProducerHPH_cff.py index cf287e9b34b4c..947519004024b 100644 --- a/L1Trigger/TrackFindingTracklet/python/ProducerHPH_cff.py +++ b/L1Trigger/TrackFindingTracklet/python/ProducerHPH_cff.py @@ -1,9 +1,9 @@ import FWCore.ParameterSet.Config as cms # Get required input ESProducts -from L1Trigger.TrackerTFP.ProducerES_cff import TrackTriggerDataFormats -from L1Trigger.TrackerTFP.ProducerLayerEncoding_cff import TrackTriggerLayerEncoding -from L1Trigger.TrackTrigger.ProducerSetup_cff import TrackTriggerSetup +from L1Trigger.TrackerTFP.DataFormats_cff import TrackTriggerDataFormats +from L1Trigger.TrackerTFP.LayerEncoding_cff import TrackTriggerLayerEncoding +from L1Trigger.TrackTrigger.Setup_cff import TrackTriggerSetup # HitPatternHelper configuration from L1Trigger.TrackFindingTracklet.ProducerHPH_cfi import HitPatternHelper_params diff --git a/L1Trigger/TrackFindingTracklet/python/Producer_cff.py b/L1Trigger/TrackFindingTracklet/python/Producer_cff.py index c295baa0e2dd9..f51299b301958 100644 --- a/L1Trigger/TrackFindingTracklet/python/Producer_cff.py +++ b/L1Trigger/TrackFindingTracklet/python/Producer_cff.py @@ -1,17 +1,20 @@ +# EDProducer to emulate hybrid track reconstruction chain after tracklet track fingding + import FWCore.ParameterSet.Config as cms -from L1Trigger.TrackTrigger.ProducerSetup_cff import TrackTriggerSetup -from L1Trigger.TrackerTFP.Producer_cfi import TrackerTFPProducer_params -from L1Trigger.TrackerTFP.ProducerES_cff import TrackTriggerDataFormats -from L1Trigger.TrackerTFP.ProducerLayerEncoding_cff import TrackTriggerLayerEncoding +from L1Trigger.TrackTrigger.Setup_cff import TrackTriggerSetup +from L1Trigger.TrackerTFP.LayerEncoding_cff import TrackTriggerLayerEncoding from L1Trigger.TrackerTFP.KalmanFilterFormats_cff import TrackTriggerKalmanFilterFormats +from L1Trigger.TrackerTFP.TrackQuality_cff import * from L1Trigger.TrackFindingTracklet.ChannelAssignment_cff import ChannelAssignment +from L1Trigger.TrackFindingTracklet.DataFormats_cff import * +from L1Trigger.TrackFindingTracklet.KalmanFilterFormats_cfi import HybridKalmanFilterFormats_params from L1Trigger.TrackFindingTracklet.Producer_cfi import TrackFindingTrackletProducer_params +from L1Trigger.TrackFindingTMTT.TMTrackProducer_Defaults_cfi import TMTrackProducer_params -TrackFindingTrackletProducerIRin = cms.EDProducer( 'trklet::ProducerIRin', TrackFindingTrackletProducer_params ) -TrackFindingTrackletProducerTBout = cms.EDProducer( 'trklet::ProducerTBout', TrackFindingTrackletProducer_params ) -TrackFindingTrackletProducerDRin = cms.EDProducer( 'trklet::ProducerDRin', TrackFindingTrackletProducer_params ) -TrackFindingTrackletProducerDR = cms.EDProducer( 'trklet::ProducerDR', TrackFindingTrackletProducer_params ) -TrackFindingTrackletProducerKFin = cms.EDProducer( 'trklet::ProducerKFin', TrackFindingTrackletProducer_params ) -TrackFindingTrackletProducerKF = cms.EDProducer( 'trackerTFP::ProducerKF', TrackFindingTrackletProducer_params ) -TrackFindingTrackletProducerKFout = cms.EDProducer( 'trklet::ProducerKFout', TrackFindingTrackletProducer_params ) +ProducerIRin = cms.EDProducer( 'trklet::ProducerIRin', TrackFindingTrackletProducer_params ) +ProducerTM = cms.EDProducer( 'trklet::ProducerTM', TrackFindingTrackletProducer_params ) +ProducerDR = cms.EDProducer( 'trklet::ProducerDR', TrackFindingTrackletProducer_params ) +ProducerKF = cms.EDProducer( 'trklet::ProducerKF', TrackFindingTrackletProducer_params, HybridKalmanFilterFormats_params, TMTrackProducer_params ) +ProducerTQ = cms.EDProducer( 'trackerTFP::ProducerTQ', TrackFindingTrackletProducer_params ) +ProducerTFP = cms.EDProducer( 'trackerTFP::ProducerTFP', TrackFindingTrackletProducer_params ) diff --git a/L1Trigger/TrackFindingTracklet/python/Producer_cfi.py b/L1Trigger/TrackFindingTracklet/python/Producer_cfi.py index 3ca33e23bd4e0..234b327985d6e 100644 --- a/L1Trigger/TrackFindingTracklet/python/Producer_cfi.py +++ b/L1Trigger/TrackFindingTracklet/python/Producer_cfi.py @@ -1,26 +1,24 @@ +# configuration for hybrid track reconstruction chain emulating EDProducer + import FWCore.ParameterSet.Config as cms -from L1Trigger.TrackTrigger.TrackQualityParams_cfi import * TrackFindingTrackletProducer_params = cms.PSet ( - InputTag = cms.InputTag( "l1tTTTracksFromTrackletEmulation", "Level1TTTracks"), # - InputTagDTC = cms.InputTag( "TrackerDTCProducer", "StubAccepted"), # - LabelTBout = cms.string ( "TrackFindingTrackletProducerTBout" ), # - LabelDRin = cms.string ( "TrackFindingTrackletProducerDRin" ), # - LabelDR = cms.string ( "TrackFindingTrackletProducerDR" ), # - LabelKFin = cms.string ( "TrackFindingTrackletProducerKFin" ), # - LabelKF = cms.string ( "TrackFindingTrackletProducerKF" ), # - LabelKFout = cms.string ( "TrackFindingTrackletProducerKFout" ), # - BranchAcceptedStubs = cms.string ( "StubAccepted" ), # - BranchAcceptedTracks = cms.string ( "TrackAccepted" ), # - BranchAcceptedTTTracks = cms.string ( "TTTrackAccepted" ), # - BranchLostStubs = cms.string ( "StubLost" ), # - BranchLostTracks = cms.string ( "TrackLost" ), # - CheckHistory = cms.bool ( False ), # checks if input sample production is configured as current process - EnableTruncation = cms.bool ( True ), # enable emulation of truncation for TBout, KF, KFin, lost stubs are filled in BranchLost - PrintKFDebug = cms.bool ( False ), # print end job internal unused MSB - UseTTStubResiduals = cms.bool ( False ), # stub residuals are recalculated from seed parameter and TTStub position - TrackQualityPSet = cms.PSet ( TrackQualityParams ), - + InputLabelTFP = cms.string( "ProducerTQ" ), # + InputLabelTQ = cms.string( "ProducerKF" ), # + InputLabelKF = cms.string( "ProducerDR" ), # + InputLabelDR = cms.string( "ProducerTM" ), # + InputLabelTM = cms.string( "l1tTTTracksFromTrackletEmulation" ), # + BranchStubs = cms.string( "StubAccepted" ), # + BranchTracks = cms.string( "TrackAccepted" ), # + BranchTTTracks = cms.string( "TTTrackAccepted" ), # + BranchTruncated = cms.string( "Truncated" ), # + EnableTruncation = cms.bool ( True ), # enable emulation of truncation for TM, DR, KF, TQ and TFP + PrintKFDebug = cms.bool ( False ), # print end job internal unused MSB + UseTTStubResiduals = cms.bool ( True ), # stub residuals and radius are recalculated from seed parameter and TTStub position + UseTTStubParameters = cms.bool ( True ), # track parameter are recalculated from seed TTStub positions + ApplyNonLinearCorrection = cms.bool ( True ), # + Use5ParameterFit = cms.bool ( False ), # double precision simulation of 5 parameter fit instead of bit accurate emulation of 4 parameter fit + UseKFsimmulation = cms.bool ( False ) # simulate KF instead of emulate ) diff --git a/L1Trigger/TrackFindingTracklet/python/l1tTTTracksFromTrackletEmulation_cfi.py b/L1Trigger/TrackFindingTracklet/python/l1tTTTracksFromTrackletEmulation_cfi.py index 7515a7c089c62..4066a576a5250 100644 --- a/L1Trigger/TrackFindingTracklet/python/l1tTTTracksFromTrackletEmulation_cfi.py +++ b/L1Trigger/TrackFindingTracklet/python/l1tTTTracksFromTrackletEmulation_cfi.py @@ -1,11 +1,15 @@ import FWCore.ParameterSet.Config as cms -from L1Trigger.TrackTrigger.TrackQualityParams_cfi import * + +from L1Trigger.TrackFindingTracklet.Producer_cfi import TrackFindingTrackletProducer_params from L1Trigger.TrackFindingTracklet.ChannelAssignment_cff import ChannelAssignment +from L1Trigger.TrackerTFP.TrackQuality_cff import * +from L1Trigger.TrackerTFP.LayerEncoding_cff import TrackTriggerLayerEncoding l1tTTTracksFromTrackletEmulation = cms.EDProducer("L1FPGATrackProducer", + TrackFindingTrackletProducer_params, TTStubSource = cms.InputTag("TTStubsFromPhase2TrackerDigis","StubAccepted"), - InputTagTTDTC = cms.InputTag("TrackerDTCProducer", "StubAccepted"), - readMoreMcTruth = cms.bool(True), + InputTagTTDTC = cms.InputTag("ProducerDTC", "StubAccepted"), + readMoreMcTruth = cms.bool(False), MCTruthClusterInputTag = cms.InputTag("TTClusterAssociatorFromPixelDigis", "ClusterAccepted"), MCTruthStubInputTag = cms.InputTag("TTStubAssociatorFromPixelDigis", "StubAccepted"), TrackingParticleInputTag = cms.InputTag("mix", "MergedTrackTruth"), @@ -21,7 +25,6 @@ wiresFile = cms.FileInPath('L1Trigger/TrackFindingTracklet/data/wires_hourglassExtendedAllCombined.dat'), # Quality Flag and Quality params TrackQuality = cms.bool(True), - TrackQualityPSet = cms.PSet(TrackQualityParams), Fakefit = cms.bool(False), # True causes Tracklet reco to output TTTracks before DR & KF StoreTrackBuilderOutput = cms.bool(False), # if True EDProducts for TrackBuilder tracks and stubs will be filled RemovalType = cms.string("merge"), # Duplicate track removal @@ -36,6 +39,5 @@ tableTEDFile = cms.FileInPath('L1Trigger/TrackFindingTracklet/data/table_TED/table_TED_D1PHIA1_D2PHIA1.txt'), tableTREFile = cms.FileInPath('L1Trigger/TrackFindingTracklet/data/table_TRE/table_TRE_D1AD2A_1.txt'), # Quality Flag and Quality params - TrackQuality = cms.bool(False), - TrackQualityPSet = cms.PSet(TrackQualityParams) + TrackQuality = cms.bool(False) ) diff --git a/L1Trigger/TrackFindingTracklet/src/ChannelAssignment.cc b/L1Trigger/TrackFindingTracklet/src/ChannelAssignment.cc index a040bbc8125c3..0479e3f3702ce 100644 --- a/L1Trigger/TrackFindingTracklet/src/ChannelAssignment.cc +++ b/L1Trigger/TrackFindingTracklet/src/ChannelAssignment.cc @@ -16,17 +16,13 @@ namespace trklet { ChannelAssignment::ChannelAssignment(const edm::ParameterSet& iConfig, const Setup* setup) : setup_(setup), - pSetDRin_(iConfig.getParameter("DRin")), - widthLayerId_(pSetDRin_.getParameter("WidthLayerId")), - widthStubId_(pSetDRin_.getParameter("WidthStubId")), - widthSeedStubId_(pSetDRin_.getParameter("WidthSeedStubId")), - widthPSTilt_(pSetDRin_.getParameter("WidthPSTilt")), - depthMemory_(pSetDRin_.getParameter("DepthMemory")), - ptBoundaries_(pSetDRin_.getParameter>("PtBoundaries")), + pSetTM_(iConfig.getParameter("TM")), + tmNumLayers_((pSetTM_.getParameter("NumLayers"))), + tmWidthStubId_(pSetTM_.getParameter("WidthStubId")), + tmWidthCot_(pSetTM_.getParameter("WidthCot")), pSetDR_(iConfig.getParameter("DR")), numComparisonModules_(pSetDR_.getParameter("NumComparisonModules")), minIdenticalStubs_(pSetDR_.getParameter("MinIdenticalStubs")), - numNodesDR_(2 * (ptBoundaries_.size() + 1)), seedTypeNames_(iConfig.getParameter>("SeedTypes")), numSeedTypes_(seedTypeNames_.size()), numChannelsTrack_(numSeedTypes_), @@ -186,22 +182,6 @@ namespace trklet { return -1; } - // return DR node for given ttTrackRef - int ChannelAssignment::nodeDR(const TTTrackRef& ttTrackRef) const { - const double pt = ttTrackRef->momentum().perp(); - int bin(0); - for (double b : ptBoundaries_) { - if (pt < b) - break; - bin++; - } - if (ttTrackRef->rInv() >= 0.) - bin += numNodesDR_ / 2; - else - bin = numNodesDR_ / 2 - 1 - bin; - return bin; - } - // layers a seed types can project to using default layer id [barrel: 1-6, discs: 11-15] int ChannelAssignment::layerId(int seedType, int channel) const { if (channel < numProjectionLayers(seedType)) diff --git a/L1Trigger/TrackFindingTracklet/src/DR.cc b/L1Trigger/TrackFindingTracklet/src/DR.cc deleted file mode 100644 index 14afa089a247e..0000000000000 --- a/L1Trigger/TrackFindingTracklet/src/DR.cc +++ /dev/null @@ -1,160 +0,0 @@ -#include "L1Trigger/TrackFindingTracklet/interface/DR.h" - -#include -#include -#include - -using namespace std; -using namespace edm; -using namespace tt; -using namespace trackerTFP; - -namespace trklet { - - DR::DR(const ParameterSet& iConfig, - const Setup* setup, - const DataFormats* dataFormats, - const ChannelAssignment* channelAssignment, - int region) - : enableTruncation_(iConfig.getParameter("EnableTruncation")), - setup_(setup), - dataFormats_(dataFormats), - channelAssignment_(channelAssignment), - region_(region), - input_(channelAssignment_->numNodesDR()) {} - - // read in and organize input tracks and stubs - void DR::consume(const StreamsTrack& streamsTrack, const StreamsStub& streamsStub) { - const int offsetTrack = region_ * channelAssignment_->numNodesDR(); - auto nonNullTrack = [](int sum, const FrameTrack& frame) { return sum + (frame.first.isNonnull() ? 1 : 0); }; - auto nonNullStub = [](int sum, const FrameStub& frame) { return sum + (frame.first.isNonnull() ? 1 : 0); }; - // count tracks and stubs and reserve corresponding vectors - int sizeTracks(0); - int sizeStubs(0); - for (int channel = 0; channel < channelAssignment_->numNodesDR(); channel++) { - const int streamTrackId = offsetTrack + channel; - const int offsetStub = streamTrackId * setup_->numLayers(); - const StreamTrack& streamTrack = streamsTrack[streamTrackId]; - input_[channel].reserve(streamTrack.size()); - sizeTracks += accumulate(streamTrack.begin(), streamTrack.end(), 0, nonNullTrack); - for (int layer = 0; layer < setup_->numLayers(); layer++) { - const StreamStub& streamStub = streamsStub[offsetStub + layer]; - sizeStubs += accumulate(streamStub.begin(), streamStub.end(), 0, nonNullStub); - } - } - tracks_.reserve(sizeTracks); - stubs_.reserve(sizeStubs); - // transform input data into handy structs - for (int channel = 0; channel < channelAssignment_->numNodesDR(); channel++) { - vector& input = input_[channel]; - const int streamTrackId = offsetTrack + channel; - const int offsetStub = streamTrackId * setup_->numLayers(); - const StreamTrack& streamTrack = streamsTrack[streamTrackId]; - for (int frame = 0; frame < (int)streamTrack.size(); frame++) { - const FrameTrack& frameTrack = streamTrack[frame]; - if (frameTrack.first.isNull()) { - input.push_back(nullptr); - continue; - } - vector stubs; - stubs.reserve(setup_->numLayers()); - for (int layer = 0; layer < setup_->numLayers(); layer++) { - const FrameStub& frameStub = streamsStub[offsetStub + layer][frame]; - if (frameStub.first.isNull()) - continue; - TTBV ttBV = frameStub.second; - const TTBV z(ttBV, dataFormats_->format(Variable::z, Process::kfin).width(), 0, true); - ttBV >>= dataFormats_->format(Variable::z, Process::kfin).width(); - const TTBV phi(ttBV, dataFormats_->format(Variable::phi, Process::kfin).width(), 0, true); - ttBV >>= dataFormats_->format(Variable::phi, Process::kfin).width(); - const TTBV r(ttBV, dataFormats_->format(Variable::r, Process::kfin).width(), 0, true); - ttBV >>= dataFormats_->format(Variable::r, Process::kfin).width(); - const TTBV stubId(ttBV, channelAssignment_->widthSeedStubId(), 0); - ttBV >>= channelAssignment_->widthSeedStubId(); - const TTBV layerId(ttBV, channelAssignment_->widthLayerId(), 0); - ttBV >>= channelAssignment_->widthLayerId(); - const TTBV tilt(ttBV, channelAssignment_->widthPSTilt(), 0); - const FrameStub frame(frameStub.first, - Frame("1" + tilt.str() + layerId.str() + r.str() + phi.str() + z.str())); - stubs_.emplace_back(frame, stubId.val(), layer); - stubs.push_back(&stubs_.back()); - } - tracks_.emplace_back(frameTrack, stubs); - input.push_back(&tracks_.back()); - } - // remove all gaps between end and last track - for (auto it = input.end(); it != input.begin();) - it = (*--it) ? input.begin() : input.erase(it); - } - } - - // fill output products - void DR::produce(StreamsStub& accpetedStubs, - StreamsTrack& acceptedTracks, - StreamsStub& lostStubs, - StreamsTrack& lostTracks) { - const int offsetTrack = region_ * channelAssignment_->numNodesDR(); - for (int node = 0; node < channelAssignment_->numNodesDR(); node++) { - const int channelTrack = offsetTrack + node; - const int offsetStub = channelTrack * setup_->numLayers(); - // remove duplicated tracks, no merge of stubs, one stub per layer expected - vector cms(channelAssignment_->numComparisonModules(), nullptr); - vector& tracks = input_[node]; - for (Track*& track : tracks) { - if (!track) - // gaps propagate trough chain and appear in output stream - continue; - for (Track*& trackCM : cms) { - if (!trackCM) { - // tracks used in CMs propagate trough chain and appear in output stream unaltered - trackCM = track; - break; - } - if (equalEnough(track, trackCM)) { - // tracks compared in CMs propagate trough chain and appear in output stream as gap if identified as duplicate or unaltered elsewise - track = nullptr; - break; - } - } - } - // remove all gaps between end and last track - for (auto it = tracks.end(); it != tracks.begin();) - it = (*--it) ? tracks.begin() : tracks.erase(it); - // store output - StreamTrack& streamTrack = acceptedTracks[channelTrack]; - streamTrack.reserve(tracks.size()); - for (int layer = 0; layer < setup_->numLayers(); layer++) - accpetedStubs[offsetStub + layer].reserve(tracks.size()); - for (Track* track : tracks) { - if (!track) { - streamTrack.emplace_back(FrameTrack()); - for (int layer = 0; layer < setup_->numLayers(); layer++) - accpetedStubs[offsetStub + layer].emplace_back(FrameStub()); - continue; - } - streamTrack.push_back(track->frame_); - TTBV hitPattern(0, setup_->numLayers()); - for (Stub* stub : track->stubs_) { - hitPattern.set(stub->channel_); - accpetedStubs[offsetStub + stub->channel_].push_back(stub->frame_); - } - for (int layer : hitPattern.ids(false)) - accpetedStubs[offsetStub + layer].emplace_back(FrameStub()); - } - } - } - - // compares two tracks, returns true if those are considered duplicates - bool DR::equalEnough(Track* t0, Track* t1) const { - int same(0); - for (int layer = 0; layer < setup_->numLayers(); layer++) { - auto onLayer = [layer](Stub* stub) { return stub->channel_ == layer; }; - const auto s0 = find_if(t0->stubs_.begin(), t0->stubs_.end(), onLayer); - const auto s1 = find_if(t1->stubs_.begin(), t1->stubs_.end(), onLayer); - if (s0 != t0->stubs_.end() && s1 != t1->stubs_.end() && **s0 == **s1) - same++; - } - return same >= channelAssignment_->minIdenticalStubs(); - } - -} // namespace trklet diff --git a/L1Trigger/TrackFindingTracklet/src/DRin.cc b/L1Trigger/TrackFindingTracklet/src/DRin.cc deleted file mode 100644 index 9196ff1fee994..0000000000000 --- a/L1Trigger/TrackFindingTracklet/src/DRin.cc +++ /dev/null @@ -1,459 +0,0 @@ -#include "L1Trigger/TrackFindingTracklet/interface/DRin.h" - -#include -#include -#include -#include - -using namespace std; -using namespace edm; -using namespace tt; -using namespace trackerTFP; - -namespace trklet { - - DRin::DRin(const ParameterSet& iConfig, - const Setup* setup, - const DataFormats* dataFormats, - const LayerEncoding* layerEncoding, - const ChannelAssignment* channelAssignment, - const Settings* settings, - int region) - : enableTruncation_(iConfig.getParameter("EnableTruncation")), - useTTStubResiduals_(iConfig.getParameter("UseTTStubResiduals")), - setup_(setup), - dataFormats_(dataFormats), - layerEncoding_(layerEncoding), - channelAssignment_(channelAssignment), - settings_(settings), - region_(region), - input_(channelAssignment_->numChannelsTrack()), - // unified tracklet digitisation granularity - baseUinv2R_(.5 * settings_->kphi1() / settings_->kr() * pow(2, settings_->rinv_shift())), - baseUphiT_(settings_->kphi1() * pow(2, settings_->phi0_shift())), - baseUcot_(settings_->kz() / settings_->kr() * pow(2, settings_->t_shift())), - baseUzT_(settings_->kz() * pow(2, settings_->z0_shift())), - baseUr_(settings_->kr()), - baseUphi_(settings_->kphi1()), - baseUz_(settings_->kz()), - // KF input format digitisation granularity (identical to TMTT) - baseLinv2R_(dataFormats->base(Variable::inv2R, Process::kfin)), - baseLphiT_(dataFormats->base(Variable::phiT, Process::kfin)), - baseLcot_(dataFormats->base(Variable::cot, Process::kfin)), - baseLzT_(dataFormats->base(Variable::zT, Process::kfin)), - baseLr_(dataFormats->base(Variable::r, Process::kfin)), - baseLphi_(dataFormats->base(Variable::phi, Process::kfin)), - baseLz_(dataFormats->base(Variable::z, Process::kfin)), - // Finer granularity (by powers of 2) than the TMTT one. Used to transform from Tracklet to TMTT base. - baseHinv2R_(baseLinv2R_ * pow(2, floor(log2(baseUinv2R_ / baseLinv2R_)))), - baseHphiT_(baseLphiT_ * pow(2, floor(log2(baseUphiT_ / baseLphiT_)))), - baseHcot_(baseLcot_ * pow(2, floor(log2(baseUcot_ / baseLcot_)))), - baseHzT_(baseLzT_ * pow(2, floor(log2(baseUzT_ / baseLzT_)))), - baseHr_(baseLr_ * pow(2, floor(log2(baseUr_ / baseLr_)))), - baseHphi_(baseLphi_ * pow(2, floor(log2(baseUphi_ / baseLphi_)))), - baseHz_(baseLz_ * pow(2, floor(log2(baseUz_ / baseLz_)))) { - // calculate digitisation granularity used for inverted cot(theta) - const int baseShiftInvCot = ceil(log2(setup_->outerRadius() / setup_->hybridRangeR())) - setup_->widthDSPbu(); - baseInvCot_ = pow(2, baseShiftInvCot); - } - - // read in and organize input tracks and stubs - void DRin::consume(const StreamsTrack& streamsTrack, const StreamsStub& streamsStub) { - static const double maxCot = sinh(setup_->maxEta()) + setup_->beamWindowZ() / setup_->chosenRofZ(); - static const int unusedMSBcot = floor(log2(baseUcot_ * pow(2, settings_->nbitst()) / (2. * maxCot))); - static const double baseCot = - baseUcot_ * pow(2, settings_->nbitst() - unusedMSBcot - 1 - setup_->widthAddrBRAM18()); - const int offsetTrack = region_ * channelAssignment_->numChannelsTrack(); - // count tracks and stubs to reserve container - int nTracks(0); - int nStubs(0); - for (int channel = 0; channel < channelAssignment_->numChannelsTrack(); channel++) { - const int channelTrack = offsetTrack + channel; - const int offsetStub = channelAssignment_->offsetStub(channelTrack); - const StreamTrack& streamTrack = streamsTrack[channelTrack]; - input_[channel].reserve(streamTrack.size()); - for (int frame = 0; frame < (int)streamTrack.size(); frame++) { - if (streamTrack[frame].first.isNull()) - continue; - nTracks++; - for (int layer = 0; layer < channelAssignment_->numProjectionLayers(channel); layer++) - if (streamsStub[offsetStub + layer][frame].first.isNonnull()) - nStubs++; - } - } - stubs_.reserve(nStubs + nTracks * channelAssignment_->numSeedingLayers()); - tracks_.reserve(nTracks); - // store tracks and stubs - for (int channel = 0; channel < channelAssignment_->numChannelsTrack(); channel++) { - const int channelTrack = offsetTrack + channel; - const int offsetStub = channelAssignment_->offsetStub(channelTrack); - const StreamTrack& streamTrack = streamsTrack[channelTrack]; - vector& input = input_[channel]; - for (int frame = 0; frame < (int)streamTrack.size(); frame++) { - const TTTrackRef& ttTrackRef = streamTrack[frame].first; - if (ttTrackRef.isNull()) { - input.push_back(nullptr); - continue; - } - //convert track parameter - const double r2Inv = digi(-ttTrackRef->rInv() / 2., baseUinv2R_); - const double phi0U = - digi(tt::deltaPhi(ttTrackRef->phi() - region_ * setup_->baseRegion() + setup_->hybridRangePhi() / 2.), - baseUphiT_); - const double phi0S = digi(phi0U - setup_->hybridRangePhi() / 2., baseUphiT_); - const double cot = digi(ttTrackRef->tanL(), baseUcot_); - const double z0 = digi(ttTrackRef->z0(), baseUzT_); - const double phiT = digi(phi0S + r2Inv * digi(dataFormats_->chosenRofPhi(), baseUr_), baseUphiT_); - const double zT = digi(z0 + cot * digi(setup_->chosenRofZ(), baseUr_), baseUzT_); - // kill tracks outside of fiducial range - if (abs(phiT) > setup_->baseRegion() / 2. || abs(zT) > setup_->hybridMaxCot() * setup_->chosenRofZ() || - abs(z0) > setup_->beamWindowZ()) { - input.push_back(nullptr); - continue; - } - // convert stubs - vector stubs; - stubs.reserve(channelAssignment_->numProjectionLayers(channel) + channelAssignment_->numSeedingLayers()); - for (int layer = 0; layer < channelAssignment_->numProjectionLayers(channel); layer++) { - const FrameStub& frameStub = streamsStub[offsetStub + layer][frame]; - const TTStubRef& ttStubRef = frameStub.first; - if (ttStubRef.isNull()) - continue; - const int layerId = channelAssignment_->layerId(channel, layer); - // parse residuals from tt::Frame and take r and layerId from tt::TTStubRef - const bool barrel = setup_->barrel(ttStubRef); - const int layerIdTracklet = setup_->trackletLayerId(ttStubRef); - const double basePhi = barrel ? settings_->kphi1() : settings_->kphi(layerIdTracklet); - const double baseRZ = barrel ? settings_->kz(layerIdTracklet) : settings_->kz(); - const int widthRZ = barrel ? settings_->zresidbits() : settings_->rresidbits(); - TTBV hw(frameStub.second); - const TTBV hwRZ(hw, widthRZ, 0, true); - hw >>= widthRZ; - const TTBV hwPhi(hw, settings_->phiresidbits(), 0, true); - hw >>= settings_->phiresidbits(); - const int indexLayerId = setup_->indexLayerId(ttStubRef); - const SensorModule::Type type = setup_->type(ttStubRef); - const int widthR = setup_->tbWidthR(type); - const double baseR = setup_->hybridBaseR(type); - const TTBV hwR(hw, widthR, 0, barrel); - hw >>= widthR; - double r = hwR.val(baseR) + (barrel ? setup_->hybridLayerR(indexLayerId) : 0.); - if (type == SensorModule::Disk2S) - r = setup_->disk2SR(indexLayerId, r); - r = digi(r - dataFormats_->chosenRofPhi(), baseUr_); - double phi = hwPhi.val(basePhi); - if (basePhi > baseUphi_) - phi += baseUphi_ / 2.; - const double z = digi(hwRZ.val(baseRZ) * (barrel ? 1. : -cot), baseUz_); - const TTBV hwStubId(hw, channelAssignment_->widthSeedStubId(), 0, false); - const int stubId = hwStubId.val(); - // determine module type - bool psTilt; - if (barrel) { - const double posZ = (r + digi(dataFormats_->chosenRofPhi(), baseUr_)) * cot + z0 + z; - const int indexLayerId = setup_->indexLayerId(ttStubRef); - const double limit = setup_->tiltedLayerLimitZ(indexLayerId); - psTilt = abs(posZ) < limit; - } else - psTilt = setup_->psModule(ttStubRef); - if (useTTStubResiduals_) { - const GlobalPoint gp = setup_->stubPos(ttStubRef); - const double ttR = r; - const double ttZ = gp.z() - (z0 + (ttR + dataFormats_->chosenRofPhi()) * cot); - stubs_.emplace_back(ttStubRef, layerId, layerIdTracklet, false, stubId, ttR, phi, ttZ, psTilt); - } else - stubs_.emplace_back(ttStubRef, layerId, layerIdTracklet, false, stubId, r, phi, z, psTilt); - stubs.push_back(&stubs_.back()); - } - // create fake seed stubs, since TrackBuilder doesn't output these stubs, required by the KF. - for (int seedingLayer = 0; seedingLayer < channelAssignment_->numSeedingLayers(); seedingLayer++) { - const int channelStub = channelAssignment_->numProjectionLayers(channel) + seedingLayer; - const FrameStub& frameStub = streamsStub[offsetStub + channelStub][frame]; - const TTStubRef& ttStubRef = frameStub.first; - if (ttStubRef.isNull()) - continue; - const int layerId = channelAssignment_->layerId(channel, channelStub); - const int layerIdTracklet = setup_->trackletLayerId(ttStubRef); - const int stubId = TTBV(frameStub.second).val(channelAssignment_->widthSeedStubId()); - const bool barrel = setup_->barrel(ttStubRef); - double r; - if (barrel) - r = digi(setup_->hybridLayerR(layerId - setup_->offsetLayerId()) - dataFormats_->chosenRofPhi(), baseUr_); - else { - r = (z0 + - digi(setup_->hybridDiskZ(layerId - setup_->offsetLayerId() - setup_->offsetLayerDisks()), baseUzT_)) * - digi(1. / digi(abs(cot), baseCot), baseInvCot_); - r = digi(r - digi(dataFormats_->chosenRofPhi(), baseUr_), baseUr_); - } - static constexpr double phi = 0.; - static constexpr double z = 0.; - // determine module type - bool psTilt; - if (barrel) { - const double posZ = - digi(digi(setup_->hybridLayerR(layerId - setup_->offsetLayerId()), baseUr_) * cot + z0, baseUz_); - const int indexLayerId = setup_->indexLayerId(ttStubRef); - const double limit = digi(setup_->tiltedLayerLimitZ(indexLayerId), baseUz_); - psTilt = abs(posZ) < limit; - } else - psTilt = true; - const GlobalPoint gp = setup_->stubPos(ttStubRef); - const double ttR = gp.perp() - dataFormats_->chosenRofPhi(); - const double ttZ = gp.z() - (z0 + (ttR + dataFormats_->chosenRofPhi()) * cot); - if (useTTStubResiduals_) - stubs_.emplace_back(ttStubRef, layerId, layerIdTracklet, true, stubId, ttR, phi, ttZ, psTilt); - else - stubs_.emplace_back(ttStubRef, layerId, layerIdTracklet, true, stubId, r, phi, z, psTilt); - stubs.push_back(&stubs_.back()); - } - const bool valid = frame < setup_->numFrames() ? true : enableTruncation_; - tracks_.emplace_back(ttTrackRef, valid, r2Inv, phiT, cot, zT, stubs); - input.push_back(&tracks_.back()); - } - } - } - - // fill output products - void DRin::produce(StreamsStub& accpetedStubs, - StreamsTrack& acceptedTracks, - StreamsStub& lostStubs, - StreamsTrack& lostTracks) { - // base transform into high precision TMTT format - for (Track& track : tracks_) { - track.inv2R_ = redigi(track.inv2R_, baseUinv2R_, baseHinv2R_, setup_->widthDSPbu()); - track.phiT_ = redigi(track.phiT_, baseUphiT_, baseHphiT_, setup_->widthDSPbu()); - track.cot_ = redigi(track.cot_, baseUcot_, baseHcot_, setup_->widthDSPbu()); - track.zT_ = redigi(track.zT_, baseUzT_, baseHzT_, setup_->widthDSPbu()); - for (Stub* stub : track.stubs_) { - stub->r_ = redigi(stub->r_, baseUr_, baseHr_, setup_->widthDSPbu()); - stub->phi_ = redigi(stub->phi_, baseUphi_, baseHphi_, setup_->widthDSPbu()); - stub->z_ = redigi(stub->z_, baseUz_, baseHz_, setup_->widthDSPbu()); - } - } - // find sector - for (Track& track : tracks_) { - const int sectorPhi = track.phiT_ < 0. ? 0 : 1; - track.phiT_ -= (sectorPhi - .5) * setup_->baseSector(); - int sectorEta(-1); - for (; sectorEta < setup_->numSectorsEta(); sectorEta++) - if (track.zT_ < digi(setup_->chosenRofZ() * sinh(setup_->boundarieEta(sectorEta + 1)), baseHzT_)) - break; - if (sectorEta >= setup_->numSectorsEta() || sectorEta <= -1) { - track.valid_ = false; - continue; - } - track.cot_ = track.cot_ - digi(setup_->sectorCot(sectorEta), baseHcot_); - track.zT_ = track.zT_ - digi(setup_->chosenRofZ() * setup_->sectorCot(sectorEta), baseHzT_); - track.sector_ = sectorPhi * setup_->numSectorsEta() + sectorEta; - } - // base transform into TMTT format - for (Track& track : tracks_) { - if (!track.valid_) - continue; - // store track parameter shifts - const double dinv2R = digi(track.inv2R_ - digi(track.inv2R_, baseLinv2R_), baseHinv2R_); - const double dphiT = digi(track.phiT_ - digi(track.phiT_, baseLphiT_), baseHphiT_); - const double dcot = digi(track.cot_ - digi(track.cot_, baseLcot_), baseHcot_); - const double dzT = digi(track.zT_ - digi(track.zT_, baseLzT_), baseHzT_); - // shift track parameter; - track.inv2R_ = digi(track.inv2R_, baseLinv2R_); - track.phiT_ = digi(track.phiT_, baseLphiT_); - track.cot_ = digi(track.cot_, baseLcot_); - track.zT_ = digi(track.zT_, baseLzT_); - // range checks - if (!dataFormats_->format(Variable::inv2R, Process::kfin).inRange(track.inv2R_, true)) - track.valid_ = false; - if (!dataFormats_->format(Variable::phiT, Process::kfin).inRange(track.phiT_, true)) - track.valid_ = false; - if (!dataFormats_->format(Variable::cot, Process::kfin).inRange(track.cot_, true)) - track.valid_ = false; - if (!dataFormats_->format(Variable::zT, Process::kfin).inRange(track.zT_, true)) - track.valid_ = false; - if (!track.valid_) - continue; - // adjust stub residuals by track parameter shifts - for (Stub* stub : track.stubs_) { - const double dphi = digi(dphiT + stub->r_ * dinv2R, baseHphi_); - const double r = stub->r_ + digi(dataFormats_->chosenRofPhi() - setup_->chosenRofZ(), baseHr_); - const double dz = digi(dzT + r * dcot, baseHz_); - stub->phi_ = digi(stub->phi_ + dphi, baseLphi_); - stub->z_ = digi(stub->z_ + dz, baseLz_); - // range checks - if (!dataFormats_->format(Variable::phi, Process::kfin).inRange(stub->phi_)) - stub->valid_ = false; - if (!dataFormats_->format(Variable::z, Process::kfin).inRange(stub->z_)) - stub->valid_ = false; - } - } - // encode layer id - for (Track& track : tracks_) { - if (!track.valid_) - continue; - const int sectorEta = track.sector_ % setup_->numSectorsEta(); - const int zT = dataFormats_->format(Variable::zT, Process::kfin).toUnsigned(track.zT_); - const int cot = dataFormats_->format(Variable::cot, Process::kfin).toUnsigned(track.cot_); - for (Stub* stub : track.stubs_) { - if (!stub->valid_) - continue; - // store encoded layerId - stub->layerKF_ = layerEncoding_->layerIdKF(sectorEta, zT, cot, stub->layer_); - // kill stubs from layers which can't be crossed by track - if (stub->layerKF_ == -1) - stub->valid_ = false; - } - TTBV hitPattern(0, setup_->numLayers()); - // kill multiple stubs from same kf layer - for (Stub* stub : track.stubs_) { - if (!stub->valid_) - continue; - if (hitPattern[stub->layerKF_]) - stub->valid_ = false; - else - hitPattern.set(stub->layerKF_); - } - // lookup maybe layers - track.maybe_ = layerEncoding_->maybePattern(sectorEta, zT, cot); - } - // kill tracks with not enough layer - for (Track& track : tracks_) { - if (!track.valid_) - continue; - TTBV hits(0, setup_->numLayers()); - for (const Stub* stub : track.stubs_) - if (stub->valid_) - hits.set(stub->layerKF_); - if (hits.count() < setup_->kfMinLayers()) - track.valid_ = false; - } - // store helper - auto frameTrack = [this](Track* track) { - const TTBV sectorPhi( - dataFormats_->format(Variable::sectorPhi, Process::kfin).ttBV(track->sector_ / setup_->numSectorsEta())); - const TTBV sectorEta( - dataFormats_->format(Variable::sectorEta, Process::kfin).ttBV(track->sector_ % setup_->numSectorsEta())); - const TTBV inv2R(dataFormats_->format(Variable::inv2R, Process::kfin).ttBV(track->inv2R_)); - const TTBV phiT(dataFormats_->format(Variable::phiT, Process::kfin).ttBV(track->phiT_)); - const TTBV cot(dataFormats_->format(Variable::cot, Process::kfin).ttBV(track->cot_)); - const TTBV zT(dataFormats_->format(Variable::zT, Process::kfin).ttBV(track->zT_)); - return FrameTrack( - track->ttTrackRef_, - Frame("1" + sectorPhi.str() + sectorEta.str() + inv2R.str() + phiT.str() + zT.str() + cot.str())); - }; - auto frameStub = [this](Track* track, int layer) { - auto equal = [layer](Stub* stub) { return stub->valid_ && stub->layerKF_ == layer; }; - const auto it = find_if(track->stubs_.begin(), track->stubs_.end(), equal); - if (it == track->stubs_.end() || !(*it)->valid_) - return FrameStub(); - Stub* stub = *it; - const TTBV layerId(stub->layerDet_, channelAssignment_->widthLayerId()); - const TTBV stubId(stub->stubId_, channelAssignment_->widthSeedStubId(), true); - const TTBV r(dataFormats_->format(Variable::r, Process::kfin).ttBV(stub->r_)); - const TTBV phi(dataFormats_->format(Variable::phi, Process::kfin).ttBV(stub->phi_)); - const TTBV z(dataFormats_->format(Variable::z, Process::kfin).ttBV(stub->z_)); - return FrameStub( - stub->ttStubRef_, - Frame("1" + to_string(stub->psTilt_) + layerId.str() + stubId.str() + r.str() + phi.str() + z.str())); - }; - // route tracks into pt bins and store result - const int offsetTrack = region_ * channelAssignment_->numNodesDR(); - for (int nodeDR = 0; nodeDR < channelAssignment_->numNodesDR(); nodeDR++) { - deque accepted; - deque lost; - vector> stacks(channelAssignment_->numChannelsTrack()); - vector> inputs(channelAssignment_->numChannelsTrack()); - for (int channel = 0; channel < channelAssignment_->numChannelsTrack(); channel++) { - for (Track* track : input_[channel]) { - const bool match = track && channelAssignment_->nodeDR(track->ttTrackRef_) == nodeDR; - if (match && !track->valid_) - lost.push_back(track); - inputs[channel].push_back(match && track->valid_ ? track : nullptr); - } - } - // remove all gaps between end and last track - for (deque& input : inputs) - for (auto it = input.end(); it != input.begin();) - it = (*--it) ? input.begin() : input.erase(it); - // clock accurate firmware emulation, each while trip describes one clock tick, one stub in and one stub out per tick - while (!all_of(inputs.begin(), inputs.end(), [](const deque& tracks) { return tracks.empty(); }) or - !all_of(stacks.begin(), stacks.end(), [](const deque& tracks) { return tracks.empty(); })) { - // fill input fifos - for (int channel = 0; channel < channelAssignment_->numChannelsTrack(); channel++) { - deque& stack = stacks[channel]; - Track* track = pop_front(inputs[channel]); - if (track) { - if (enableTruncation_ && (int)stack.size() == channelAssignment_->depthMemory() - 1) - lost.push_back(pop_front(stack)); - stack.push_back(track); - } - } - // merge input fifos to one stream, prioritizing higher input channel over lower channel - bool nothingToRoute(true); - for (int channel = channelAssignment_->numChannelsTrack() - 1; channel >= 0; channel--) { - Track* track = pop_front(stacks[channel]); - if (track) { - nothingToRoute = false; - accepted.push_back(track); - break; - } - } - if (nothingToRoute) - accepted.push_back(nullptr); - } - // truncate if desired - if (enableTruncation_ && (int)accepted.size() > setup_->numFrames()) { - const auto limit = next(accepted.begin(), setup_->numFrames()); - copy_if(limit, accepted.end(), back_inserter(lost), [](const Track* track) { return track; }); - accepted.erase(limit, accepted.end()); - } - // remove all gaps between end and last track - for (auto it = accepted.end(); it != accepted.begin();) - it = (*--it) ? accepted.begin() : accepted.erase(it); - // fill products StreamsStub& accpetedStubs, StreamsTrack& acceptedTracks, StreamsStub& lostStubs, StreamsTrack& lostTracks - const int channelTrack = offsetTrack + nodeDR; - const int offsetStub = channelTrack * setup_->numLayers(); - // fill lost tracks and stubs without gaps - lostTracks[channelTrack].reserve(lost.size()); - for (int layer = 0; layer < setup_->numLayers(); layer++) - lostStubs[offsetStub + layer].reserve(lost.size()); - for (Track* track : lost) { - lostTracks[channelTrack].emplace_back(frameTrack(track)); - for (int layer = 0; layer < setup_->numLayers(); layer++) - lostStubs[offsetStub + layer].emplace_back(frameStub(track, layer)); - } - // fill accepted tracks and stubs with gaps - acceptedTracks[channelTrack].reserve(accepted.size()); - for (int layer = 0; layer < setup_->numLayers(); layer++) - accpetedStubs[offsetStub + layer].reserve(accepted.size()); - for (Track* track : accepted) { - if (!track) { // fill gap - acceptedTracks[channelTrack].emplace_back(FrameTrack()); - for (int layer = 0; layer < setup_->numLayers(); layer++) - accpetedStubs[offsetStub + layer].emplace_back(FrameStub()); - continue; - } - acceptedTracks[channelTrack].emplace_back(frameTrack(track)); - for (int layer = 0; layer < setup_->numLayers(); layer++) - accpetedStubs[offsetStub + layer].emplace_back(frameStub(track, layer)); - } - } - } - - // remove and return first element of deque, returns nullptr if empty - template - T* DRin::pop_front(deque& ts) const { - T* t = nullptr; - if (!ts.empty()) { - t = ts.front(); - ts.pop_front(); - } - return t; - } - - // basetransformation of val from baseLow into baseHigh using widthMultiplier bit multiplication - double DRin::redigi(double val, double baseLow, double baseHigh, int widthMultiplier) const { - const double base = pow(2, 1 - widthMultiplier); - const double transform = digi(baseLow / baseHigh, base); - return (floor(val * transform / baseLow) + .5) * baseHigh; - } - -} // namespace trklet diff --git a/L1Trigger/TrackFindingTracklet/src/DataFormats.cc b/L1Trigger/TrackFindingTracklet/src/DataFormats.cc new file mode 100644 index 0000000000000..4a0b091944474 --- /dev/null +++ b/L1Trigger/TrackFindingTracklet/src/DataFormats.cc @@ -0,0 +1,227 @@ +#include "L1Trigger/TrackFindingTracklet/interface/DataFormats.h" +#include "DataFormats/L1TrackTrigger/interface/TTTrack_TrackWord.h" + +#include +#include +#include +#include +#include +#include +#include +#include +#include + +using namespace std; +using namespace edm; +using namespace tt; + +namespace trklet { + + // default constructor, trying to need space as proper constructed object + DataFormats::DataFormats() + : numDataFormats_(0), + formats_(+Variable::end, std::vector(+Process::end, nullptr)), + numUnusedBitsStubs_(+Process::end, TTBV::S_ - 1), + numUnusedBitsTracks_(+Process::end, TTBV::S_ - 1) { + channelAssignment_ = nullptr; + countFormats(); + dataFormats_.reserve(numDataFormats_); + } + + // method to count number of unique data formats + template + void DataFormats::countFormats() { + if constexpr (config_[+v][+p] == p) + numDataFormats_++; + if constexpr (++p != Process::end) + countFormats(); + else if constexpr (++v != Variable::end) + countFormats<++v>(); + } + + // proper constructor + DataFormats::DataFormats(const ChannelAssignment* channelAssignment) : DataFormats() { + channelAssignment_ = channelAssignment; + fillDataFormats(); + for (const Process p : Processes) + for (const Variable v : stubs_[+p]) + numUnusedBitsStubs_[+p] -= formats_[+v][+p] ? formats_[+v][+p]->width() : 0; + for (const Process p : Processes) + for (const Variable v : tracks_[+p]) + numUnusedBitsTracks_[+p] -= formats_[+v][+p] ? formats_[+v][+p]->width() : 0; + } + + // constructs data formats of all unique used variables and flavours + template + void DataFormats::fillDataFormats() { + if constexpr (config_[+v][+p] == p) { + dataFormats_.emplace_back(Format(channelAssignment_)); + fillFormats(); + } + if constexpr (++p != Process::end) + fillDataFormats(); + else if constexpr (++v != Variable::end) + fillDataFormats<++v>(); + } + + // helper (loop) data formats of all unique used variables and flavours + template + void DataFormats::fillFormats() { + if (config_[+v][+it] == p) { + formats_[+v][+it] = &dataFormats_.back(); + } + if constexpr (++it != Process::end) + fillFormats(); + } + + template <> + Format::Format(const ChannelAssignment* ca) : DataFormat(true) { + width_ = TTTrack_TrackWord::TrackBitWidths::kRinvSize; + range_ = -2. * TTTrack_TrackWord::minRinv; + base_ = range_ * pow(2, -width_); + } + template <> + Format::Format(const ChannelAssignment* ca) : DataFormat(true) { + width_ = TTTrack_TrackWord::TrackBitWidths::kPhiSize; + range_ = -2. * TTTrack_TrackWord::minPhi0; + base_ = range_ * pow(2, -width_); + } + template <> + Format::Format(const ChannelAssignment* ca) : DataFormat(true) { + width_ = TTTrack_TrackWord::TrackBitWidths::kTanlSize; + range_ = -2. * TTTrack_TrackWord::minTanl; + base_ = range_ * pow(2, -width_); + } + template <> + Format::Format(const ChannelAssignment* ca) : DataFormat(true) { + width_ = TTTrack_TrackWord::TrackBitWidths::kZ0Size; + range_ = -2. * TTTrack_TrackWord::minZ0; + base_ = range_ * pow(2, -width_); + } + + template <> + Format::Format(const ChannelAssignment* ca) : DataFormat(true) { + const Setup* s = ca->setup(); + static const double thight = s->htNumBinsInv2R(); + static const double loose = thight + 2; + range_ = 2. * s->invPtToDphi() / s->minPt() * loose / thight; + base_ = range_ / loose; + width_ = ceil(log2(range_ / base_)); + } + template <> + Format::Format(const ChannelAssignment* ca) : DataFormat(true) { + const Setup* s = ca->setup(); + static const double thight = s->gpNumBinsPhiT() * s->htNumBinsPhiT(); + static const double loose = thight + 2 * s->gpNumBinsPhiT(); + range_ = 2. * M_PI / s->numRegions() * loose / thight; + base_ = range_ / loose; + width_ = ceil(log2(range_ / base_)); + } + template <> + Format::Format(const ChannelAssignment* ca) : DataFormat(true) { + const Setup* s = ca->setup(); + static const double thight = s->gpNumBinsZT(); + static const double loose = thight + 2; + range_ = 2. * sinh(s->maxEta()) * s->chosenRofZ() * loose / thight; + base_ = range_ / loose; + width_ = ceil(log2(range_ / base_)); + } + template <> + Format::Format(const ChannelAssignment* ca) : DataFormat(true) { + const Setup* s = ca->setup(); + const Format zT(ca); + range_ = (zT.range() + 2. * s->beamWindowZ()) / s->chosenRofZ(); + base_ = (zT.base() + 2. * s->beamWindowZ()) / s->chosenRofZ(); + } + + template <> + Format::Format(const ChannelAssignment* ca) : DataFormat(false) { + width_ = ca->tmWidthStubId() + 1; + } + template <> + Format::Format(const ChannelAssignment* ca) : DataFormat(true) { + const Setup* s = ca->setup(); + const Format phiT(ca); + const Format inv2R(ca); + range_ = 2. * s->maxRphi(); + base_ = phiT.base() / inv2R.base(); + const int shift = ceil(log2(range_ / base_)) - s->tmttWidthR(); + base_ *= pow(2., shift); + width_ = ceil(log2(range_ / base_)); + } + template <> + Format::Format(const ChannelAssignment* ca) : DataFormat(true) { + const Setup* s = ca->setup(); + const Format phiT(ca); + const Format inv2R(ca); + const double range = s->baseRegion() + s->maxRphi() * inv2R.range(); + range_ = phiT.base() + s->maxRphi() * inv2R.base(); + const int shift = ceil(log2(range / phiT.base())) - s->tmttWidthPhi(); + base_ = phiT.base() * pow(2., shift); + width_ = ceil(log2(range_ / base_)); + } + template <> + Format::Format(const ChannelAssignment* ca) : DataFormat(true) { + const Setup* s = ca->setup(); + const Format zT(ca); + const Format cot(ca); + const double range = 2. * s->halfLength(); + range_ = zT.base() + s->maxRz() * cot.base(); + const int shift = ceil(log2(range / zT.base())) - s->tmttWidthZ(); + base_ = zT.base() * pow(2., shift); + width_ = ceil(log2(range_ / base_)); + } + template <> + Format::Format(const ChannelAssignment* ca) : DataFormat(false) { + const Setup* s = ca->setup(); + const Format phi(ca); + const Format inv2R(ca); + range_ = .5 * s->pitchRowPS() / s->innerRadius() + .25 * (s->pitchCol2S() + s->scattering()) * inv2R.range(); + base_ = phi.base(); + width_ = ceil(log2(range_ / base_)); + } + template <> + Format::Format(const ChannelAssignment* ca) : DataFormat(false) { + const Setup* s = ca->setup(); + const Format z(ca); + range_ = .5 * s->pitchCol2S() * sinh(s->maxEta()); + base_ = z.base(); + width_ = ceil(log2(range_ / base_)); + } + + template <> + Format::Format(const ChannelAssignment* ca) : DataFormat(true) { + const Format tfp(ca); + const Format tm(ca); + range_ = tm.range(); + base_ = tm.base() * pow(2., floor(log2(.5 * tfp.base() / tm.base()))); + width_ = ceil(log2(range_ / base_)); + } + template <> + Format::Format(const ChannelAssignment* ca) : DataFormat(true) { + const Format tfp(ca); + const Format tm(ca); + range_ = tm.range(); + base_ = tm.base() * pow(2., floor(log2(tfp.base() / tm.base()))); + width_ = ceil(log2(range_ / base_)); + } + template <> + Format::Format(const ChannelAssignment* ca) : DataFormat(true) { + const Format tfp(ca); + const Format cot(ca); + const Format z(ca); + const Format r(ca); + range_ = cot.range(); + base_ = z.base() / r.base() * pow(2., floor(log2(tfp.base() / z.base() * r.base()))); + width_ = ceil(log2(range_ / base_)); + } + template <> + Format::Format(const ChannelAssignment* ca) : DataFormat(true) { + const Format tfp(ca); + const Format tm(ca); + range_ = tm.range(); + base_ = tm.base() * pow(2., floor(log2(tfp.base() / tm.base()))); + width_ = ceil(log2(range_ / base_)); + } + +} // namespace trklet diff --git a/L1Trigger/TrackFindingTracklet/src/DuplicateRemoval.cc b/L1Trigger/TrackFindingTracklet/src/DuplicateRemoval.cc new file mode 100644 index 0000000000000..2f4192a0c2ac1 --- /dev/null +++ b/L1Trigger/TrackFindingTracklet/src/DuplicateRemoval.cc @@ -0,0 +1,183 @@ +#include "L1Trigger/TrackFindingTracklet/interface/DuplicateRemoval.h" + +#include +#include +#include + +using namespace std; +using namespace edm; +using namespace tt; +using namespace trackerTFP; + +namespace trklet { + + DuplicateRemoval::DuplicateRemoval(const ParameterSet& iConfig, + const Setup* setup, + const LayerEncoding* layerEncoding, + const DataFormats* dataFormats, + const ChannelAssignment* channelAssignment, + int region) + : enableTruncation_(iConfig.getParameter("EnableTruncation")), + setup_(setup), + layerEncoding_(layerEncoding), + dataFormats_(dataFormats), + channelAssignment_(channelAssignment), + region_(region) { + const DataFormat& r = dataFormats_->format(Variable::r, Process::dr); + const int width = setup_->widthAddrBRAM18() - 1; + const double base = r.base() * pow(2., r.width() - width); + const double range = r.range(); + r_ = DataFormat(true, width, base, range); + } + + // read in and organize input tracks and stubs + void DuplicateRemoval::consume(const StreamsTrack& streamsTrack, const StreamsStub& streamsStub) { + auto nonNullTrack = [](int& sum, const FrameTrack& frame) { return sum += (frame.first.isNonnull() ? 1 : 0); }; + auto nonNullStub = [](int& sum, const FrameStub& frame) { return sum += (frame.first.isNonnull() ? 1 : 0); }; + // count tracks and stubs and reserve corresponding vectors + int sizeStubs(0); + static const int numLayers = channelAssignment_->tmNumLayers(); + const int offset = region_ * numLayers; + const StreamTrack& streamTrack = streamsTrack[region_]; + input_.reserve(streamTrack.size()); + const int sizeTracks = accumulate(streamTrack.begin(), streamTrack.end(), 0, nonNullTrack); + for (int layer = 0; layer < numLayers; layer++) { + const StreamStub& streamStub = streamsStub[offset + layer]; + sizeStubs += accumulate(streamStub.begin(), streamStub.end(), 0, nonNullStub); + } + tracks_.reserve(sizeTracks); + stubs_.reserve(sizeStubs); + // transform input data into handy structs + for (int frame = 0; frame < (int)streamTrack.size(); frame++) { + const FrameTrack& frameTrack = streamTrack[frame]; + if (frameTrack.first.isNull()) { + input_.push_back(nullptr); + continue; + } + // lookup layerEncoding + const TrackTM track(frameTrack, dataFormats_); + const double inv2R = abs(track.inv2R()); + const double zT = abs(track.zT()); + const double cot = zT / setup_->chosenRofZ(); + const vector& layerEncoding = layerEncoding_->layerEncoding(zT); + vector stubs(numLayers, nullptr); + TTBV hitPattern(0, setup_->numLayers()); + for (int layer = 0; layer < numLayers; layer++) { + const FrameStub& frameStub = streamsStub[offset + layer][frame]; + const TTStubRef& ttStubRef = frameStub.first; + if (ttStubRef.isNull()) + continue; + // encode layerId + const int decodedLayerId = + layer + setup_->offsetLayerId() + + (layer < setup_->numBarrelLayer() ? 0 : setup_->offsetLayerDisks() - setup_->numBarrelLayer()); + const auto it = find(layerEncoding.begin(), layerEncoding.end(), decodedLayerId); + const int encodedLayerId = min((int)distance(layerEncoding.begin(), it), setup_->numLayers() - 1); + // kill stub on already occupied layer + if (hitPattern.test(encodedLayerId)) + continue; + hitPattern.set(encodedLayerId); + const StubTM stubTM(frameStub, dataFormats_); + const int stubId = stubTM.stubId() / 2; + const bool psTilt = stubTM.stubId() % 2 == 1; + // calculate stub uncertainties + static const DataFormat& dfPhi = dataFormats_->format(Variable::phi, Process::dr); + static constexpr int numBarrelPSLayer = 3; + const bool barrel = layer < setup_->numBarrelLayer(); + const bool ps = barrel ? layer < numBarrelPSLayer : psTilt; + const bool tilt = barrel && psTilt; + const double length = .5 * (ps ? setup_->pitchColPS() : setup_->pitchCol2S()); + const double pitch = .5 * (ps ? setup_->pitchRowPS() : setup_->pitchRow2S()); + const double pitchOverR = dfPhi.digi(pitch / (r_.digi(stubTM.r()) + setup_->chosenRofPhi())); + double lengthZ = length; + double lengthR = 0.; + if (!barrel) { + lengthZ = length * cot; + lengthR = length; + } else if (tilt) { + lengthZ = length * (setup_->tiltApproxSlope() * cot + setup_->tiltApproxIntercept()); + lengthR = .5 * setup_->tiltUncertaintyR(); + } + const double dR = lengthR + .5 * setup_->scattering(); + const double dZ = lengthZ; + const double dPhi = dfPhi.digi(dR * inv2R) + pitchOverR; + const StubDR stubDR(stubTM, stubTM.r(), stubTM.phi(), stubTM.z(), dPhi, dZ); + stubs_.emplace_back(stubDR.frame(), stubId, encodedLayerId); + stubs[layer] = &stubs_.back(); + } + // kill tracks with not enough layers + if (hitPattern.count() < setup_->kfMinLayers()) { + input_.push_back(nullptr); + continue; + } + tracks_.emplace_back(frameTrack, stubs); + input_.push_back(&tracks_.back()); + } + // remove all gaps between end and last track + for (auto it = input_.end(); it != input_.begin();) + it = (*--it) ? input_.begin() : input_.erase(it); + } + + // fill output products + void DuplicateRemoval::produce(StreamsTrack& streamsTrack, StreamsStub& streamsStub) { + const int offset = region_ * setup_->numLayers(); + // remove duplicated tracks, no merge of stubs, one stub per layer expected + vector cms(channelAssignment_->numComparisonModules(), nullptr); + for (Track*& track : input_) { + if (!track) + // gaps propagate through chain and appear in output stream + continue; + for (Track*& trackCM : cms) { + if (!trackCM) { + // tracks used in CMs propagate through chain and do appear in output stream unaltered + trackCM = track; + break; + } + if (equalEnough(track, trackCM)) { + // tracks compared in CMs propagate through chain and appear in output stream as gap if identified as duplicate or unaltered elsewise + track = nullptr; + break; + } + } + } + // remove all gaps between end and last track + for (auto it = input_.end(); it != input_.begin();) + it = (*--it) ? input_.begin() : input_.erase(it); + // store output + StreamTrack& streamTrack = streamsTrack[region_]; + streamTrack.reserve(input_.size()); + for (int layer = 0; layer < setup_->numLayers(); layer++) + streamsStub[offset + layer].reserve(input_.size()); + for (Track* track : input_) { + if (!track) { + streamTrack.emplace_back(FrameTrack()); + for (int layer = 0; layer < setup_->numLayers(); layer++) + streamsStub[offset + layer].emplace_back(FrameStub()); + continue; + } + streamTrack.push_back(track->frame_); + TTBV hitPattern(0, setup_->numLayers()); + for (Stub* stub : track->stubs_) { + if (!stub) + continue; + hitPattern.set(stub->layer_); + streamsStub[offset + stub->layer_].emplace_back(stub->frame_); + } + for (int layer : hitPattern.ids(false)) + streamsStub[offset + layer].emplace_back(FrameStub()); + } + } + + // compares two tracks, returns true if those are considered duplicates + bool DuplicateRemoval::equalEnough(Track* t0, Track* t1) const { + int same(0); + for (int layer = 0; layer < channelAssignment_->tmNumLayers(); layer++) { + Stub* s0 = t0->stubs_[layer]; + Stub* s1 = t1->stubs_[layer]; + if (s0 && s1 && s0->stubId_ == s1->stubId_) + same++; + } + return same >= channelAssignment_->minIdenticalStubs(); + } + +} // namespace trklet diff --git a/L1Trigger/TrackFindingTracklet/src/ES_DataFormats.cc b/L1Trigger/TrackFindingTracklet/src/ES_DataFormats.cc new file mode 100644 index 0000000000000..411f1e9f3e36f --- /dev/null +++ b/L1Trigger/TrackFindingTracklet/src/ES_DataFormats.cc @@ -0,0 +1,6 @@ +#include "FWCore/Utilities/interface/typelookup.h" +#include "L1Trigger/TrackFindingTracklet/interface/DataFormats.h" +#include "L1Trigger/TrackFindingTracklet/interface/DataFormatsRcd.h" + +TYPELOOKUP_DATA_REG(trklet::DataFormats); +TYPELOOKUP_DATA_REG(trklet::DataFormatsRcd); \ No newline at end of file diff --git a/L1Trigger/TrackFindingTracklet/src/FitTrack.cc b/L1Trigger/TrackFindingTracklet/src/FitTrack.cc index 04c5f565f335f..b0b0b9412da91 100644 --- a/L1Trigger/TrackFindingTracklet/src/FitTrack.cc +++ b/L1Trigger/TrackFindingTracklet/src/FitTrack.cc @@ -1077,10 +1077,10 @@ void FitTrack::execute(deque& streamTrackRaw, } } // convert seed stubs - const string& stubId0 = bestTracklet->innerFPGAStub()->stubindex().str(); + const string& stubId0 = bestTracklet->innerFPGAStub()->phiregionaddressstr(); const L1TStub* stub0 = bestTracklet->innerFPGAStub()->l1tstub(); streamsStubRaw[ihit++].emplace_back(seedType, *stub0, valid + stubId0); - const string& stubId1 = bestTracklet->outerFPGAStub()->stubindex().str(); + const string& stubId1 = bestTracklet->outerFPGAStub()->phiregionaddressstr(); const L1TStub* stub1 = bestTracklet->outerFPGAStub()->l1tstub(); streamsStubRaw[ihit++].emplace_back(seedType, *stub1, valid + stubId1); // fill all layers that have no stubs with gaps diff --git a/L1Trigger/TrackFindingTracklet/src/HitPatternHelper.cc b/L1Trigger/TrackFindingTracklet/src/HitPatternHelper.cc index 9ff54c3d35aeb..3b475819c0e0f 100644 --- a/L1Trigger/TrackFindingTracklet/src/HitPatternHelper.cc +++ b/L1Trigger/TrackFindingTracklet/src/HitPatternHelper.cc @@ -21,13 +21,12 @@ namespace hph { oldKFPSet_(iConfig.getParameter("oldKFPSet")), setupTT_(setupTT), dataFormats_(dataFormats), - dfcot_(dataFormats_.format(trackerTFP::Variable::cot, trackerTFP::Process::kfin)), - dfzT_(dataFormats_.format(trackerTFP::Variable::zT, trackerTFP::Process::kfin)), + dfcot_(dataFormats_.format(trackerTFP::Variable::cot, trackerTFP::Process::gp)), + dfzT_(dataFormats_.format(trackerTFP::Variable::zT, trackerTFP::Process::gp)), layerEncoding_(layerEncoding), hphDebug_(iConfig.getParameter("hphDebug")), useNewKF_(iConfig.getParameter("useNewKF")), chosenRofZNewKF_(setupTT_.chosenRofZ()), - etaRegionsNewKF_(setupTT_.boundarieEta()), layermap_(), nEtaRegions_(tmtt::KFbase::nEta_ / 2), nKalmanLayers_(tmtt::KFbase::nKFlayer_) { @@ -259,14 +258,16 @@ namespace hph { } int Setup::digiCot(double cot, int binEta) const { - double cotLocal = dfcot_.digi(cot - setupTT_.sectorCot(binEta)); - return dfcot_.toUnsigned(dfcot_.integer(cotLocal)); + //double cotLocal = dfcot_.digi(cot - setupTT_.sectorCot(binEta)); + //return dfcot_.toUnsigned(dfcot_.integer(cotLocal)); + return int(); } int Setup::digiZT(double z0, double cot, int binEta) const { - double zT = z0 + setupTT_.chosenRofZ() * cot; - double zTLocal = dfzT_.digi(zT - setupTT_.sectorCot(binEta) * setupTT_.chosenRofZ()); - return dfzT_.toUnsigned(dfzT_.integer(zTLocal)); + //double zT = z0 + setupTT_.chosenRofZ() * cot; + //double zTLocal = dfzT_.digi(zT - setupTT_.sectorCot(binEta) * setupTT_.chosenRofZ()); + //return dfzT_.toUnsigned(dfzT_.integer(zTLocal)); + return int(); } int HitPatternHelper::reducedId(int layerId) { diff --git a/L1Trigger/TrackFindingTracklet/src/KFin.cc b/L1Trigger/TrackFindingTracklet/src/KFin.cc deleted file mode 100644 index c5cbc3d469648..0000000000000 --- a/L1Trigger/TrackFindingTracklet/src/KFin.cc +++ /dev/null @@ -1,270 +0,0 @@ -#include "L1Trigger/TrackFindingTracklet/interface/KFin.h" - -#include -#include -#include - -using namespace std; -using namespace edm; -using namespace tt; -using namespace trackerTFP; - -namespace trklet { - - KFin::KFin(const ParameterSet& iConfig, - const Setup* setup, - const DataFormats* dataFormats, - const LayerEncoding* layerEncoding, - const ChannelAssignment* channelAssignment, - int region) - : enableTruncation_(iConfig.getParameter("EnableTruncation")), - setup_(setup), - dataFormats_(dataFormats), - layerEncoding_(layerEncoding), - channelAssignment_(channelAssignment), - region_(region), - input_(channelAssignment_->numNodesDR()) {} - - // read in and organize input tracks and stubs - void KFin::consume(const StreamsTrack& streamsTrack, const StreamsStub& streamsStub) { - const int offsetTrack = region_ * channelAssignment_->numNodesDR(); - auto nonNullTrack = [](int sum, const FrameTrack& frame) { return sum + (frame.first.isNonnull() ? 1 : 0); }; - auto nonNullStub = [](int sum, const FrameStub& frame) { return sum + (frame.first.isNonnull() ? 1 : 0); }; - // count tracks and stubs and reserve corresponding vectors - int sizeTracks(0); - int sizeStubs(0); - for (int channel = 0; channel < channelAssignment_->numNodesDR(); channel++) { - const int streamTrackId = offsetTrack + channel; - const int offsetStub = streamTrackId * setup_->numLayers(); - const StreamTrack& streamTrack = streamsTrack[streamTrackId]; - input_[channel].reserve(streamTrack.size()); - sizeTracks += accumulate(streamTrack.begin(), streamTrack.end(), 0, nonNullTrack); - for (int layer = 0; layer < setup_->numLayers(); layer++) { - const StreamStub& streamStub = streamsStub[offsetStub + layer]; - sizeStubs += accumulate(streamStub.begin(), streamStub.end(), 0, nonNullStub); - } - } - tracks_.reserve(sizeTracks); - stubs_.reserve(sizeStubs); - // transform input data into handy structs - for (int channel = 0; channel < channelAssignment_->numNodesDR(); channel++) { - vector& input = input_[channel]; - const int streamTrackId = offsetTrack + channel; - const int offsetStub = streamTrackId * setup_->numLayers(); - const StreamTrack& streamTrack = streamsTrack[streamTrackId]; - for (int frame = 0; frame < (int)streamTrack.size(); frame++) { - const FrameTrack& frameTrack = streamTrack[frame]; - if (frameTrack.first.isNull()) { - input.push_back(nullptr); - continue; - } - vector stubs; - stubs.reserve(setup_->numLayers()); - for (int layer = 0; layer < setup_->numLayers(); layer++) { - const FrameStub& frameStub = streamsStub[offsetStub + layer][frame]; - if (frameStub.first.isNull()) - continue; - TTBV ttBV = frameStub.second; - const TTBV zBV(ttBV, dataFormats_->format(Variable::z, Process::kfin).width(), 0, true); - ttBV >>= dataFormats_->format(Variable::z, Process::kfin).width(); - const TTBV phiBV(ttBV, dataFormats_->format(Variable::phi, Process::kfin).width(), 0, true); - ttBV >>= dataFormats_->format(Variable::phi, Process::kfin).width(); - const TTBV rBV(ttBV, dataFormats_->format(Variable::r, Process::kfin).width(), 0, true); - ttBV >>= dataFormats_->format(Variable::r, Process::kfin).width(); - const TTBV layerIdBV(ttBV, channelAssignment_->widthLayerId(), 0); - ttBV >>= channelAssignment_->widthPSTilt(); - const TTBV tiltBV(ttBV, channelAssignment_->widthPSTilt(), 0); - const double r = rBV.val(dataFormats_->base(Variable::r, Process::kfin)); - const double phi = phiBV.val(dataFormats_->base(Variable::phi, Process::kfin)); - const double z = zBV.val(dataFormats_->base(Variable::z, Process::kfin)); - stubs_.emplace_back(frameStub.first, r, phi, z, layerIdBV.val(), tiltBV.val(), layer); - stubs.push_back(&stubs_.back()); - } - TTBV ttBV = frameTrack.second; - const TTBV cotBV(ttBV, dataFormats_->format(Variable::cot, Process::kfin).width(), 0, true); - ttBV >>= dataFormats_->format(Variable::cot, Process::kfin).width(); - const TTBV zTBV(ttBV, dataFormats_->format(Variable::zT, Process::kfin).width(), 0, true); - ttBV >>= dataFormats_->format(Variable::zT, Process::kfin).width(); - const TTBV phiTBV(ttBV, dataFormats_->format(Variable::phiT, Process::kfin).width(), 0, true); - ttBV >>= dataFormats_->format(Variable::phiT, Process::kfin).width(); - const TTBV inv2RBV(ttBV, dataFormats_->format(Variable::inv2R, Process::kfin).width(), 0, true); - ttBV >>= dataFormats_->format(Variable::inv2R, Process::kfin).width(); - const TTBV sectorEtaBV(ttBV, dataFormats_->format(Variable::sectorEta, Process::kfin).width(), 0); - ttBV >>= dataFormats_->format(Variable::sectorEta, Process::kfin).width(); - const TTBV sectorPhiBV(ttBV, dataFormats_->format(Variable::sectorPhi, Process::kfin).width(), 0); - const double cot = cotBV.val(dataFormats_->base(Variable::cot, Process::kfin)); - const double zT = zTBV.val(dataFormats_->base(Variable::zT, Process::kfin)); - const double inv2R = inv2RBV.val(dataFormats_->base(Variable::inv2R, Process::kfin)); - const int sectorEta = sectorEtaBV.val(); - const int zTu = dataFormats_->format(Variable::zT, Process::kfin).toUnsigned(zT); - const int cotu = dataFormats_->format(Variable::cot, Process::kfin).toUnsigned(cot); - const TTBV maybe = layerEncoding_->maybePattern(sectorEta, zTu, cotu); - const FrameTrack frameT(frameTrack.first, - Frame("1" + maybe.str() + sectorPhiBV.str() + sectorEtaBV.str() + phiTBV.str() + - inv2RBV.str() + zTBV.str() + cotBV.str())); - tracks_.emplace_back(frameT, stubs, cot, zT, inv2R, sectorEtaBV.val()); - input.push_back(&tracks_.back()); - } - // remove all gaps between end and last track - for (auto it = input.end(); it != input.begin();) - it = (*--it) ? input.begin() : input.erase(it); - } - } - - // fill output products - void KFin::produce(StreamsStub& accpetedStubs, - StreamsTrack& acceptedTracks, - StreamsStub& lostStubs, - StreamsTrack& lostTracks) { - // calculate stub uncertainties - static constexpr int usedMSBpitchOverRaddr = 1; - static const double baseRlut = - dataFormats_->base(Variable::r, Process::kfin) * - pow(2, dataFormats_->width(Variable::r, Process::zht) - setup_->widthAddrBRAM18() + usedMSBpitchOverRaddr); - static const double baseRinvR = dataFormats_->base(Variable::r, Process::kfin) * - pow(2, dataFormats_->width(Variable::r, Process::zht) - setup_->widthAddrBRAM18()); - static const double basePhi = - dataFormats_->base(Variable::inv2R, Process::kfin) * dataFormats_->base(Variable::r, Process::kfin); - static const double baseInvR = - pow(2., - ceil(log2(dataFormats_->base(Variable::r, Process::kfin) / setup_->tbInnerRadius())) - - setup_->widthDSPbu()) / - dataFormats_->base(Variable::r, Process::kfin); - static const double maxCot = sinh(setup_->maxEta()) + setup_->beamWindowZ() / setup_->chosenRofZ(); - static constexpr int usedMSBCotLutaddr = 3; - static const double baseCotLut = pow(2., ceil(log2(maxCot)) - setup_->widthAddrBRAM18() + usedMSBCotLutaddr); - static const double baseCot = dataFormats_->base(Variable::cot, Process::kfin); - static const double baseZ = dataFormats_->base(Variable::z, Process::kfin); - static const double baseR = dataFormats_->base(Variable::r, Process::kfin); - for (const Track& track : tracks_) { - const int sectorEta = track.sectorEta_; - const double inv2R = abs(track.inv2R_); - for (Stub* stub : track.stubs_) { - const bool barrel = setup_->barrel(stub->ttStubRef_); - const bool ps = barrel ? setup_->psModule(stub->ttStubRef_) : stub->psTilt_; - const bool tilt = barrel ? (ps && !stub->psTilt_) : false; - const double length = ps ? setup_->lengthPS() : setup_->length2S(); - const double pitch = ps ? setup_->pitchPS() : setup_->pitch2S(); - const double pitchOverR = digi(pitch / (digi(stub->r_, baseRlut) + dataFormats_->chosenRofPhi()), basePhi); - const double r = digi(stub->r_, baseRinvR) + dataFormats_->chosenRofPhi(); - const double sumdz = track.zT_ + stub->z_; - const double dZ = digi(sumdz - digi(setup_->chosenRofZ(), baseR) * track.cot_, baseCot * baseR); - const double sumcot = track.cot_ + digi(setup_->sectorCot(sectorEta), baseCot); - const double cot = digi(abs(dZ * digi(1. / r, baseInvR) + sumcot), baseCotLut); - double lengthZ = length; - double lengthR = 0.; - if (!barrel) { - lengthZ = length * cot; - lengthR = length; - } else if (tilt) { - lengthZ = length * abs(setup_->tiltApproxSlope() * cot + setup_->tiltApproxIntercept()); - lengthR = setup_->tiltUncertaintyR(); - } - const double scat = digi(setup_->scattering(), baseR); - stub->dZ_ = lengthZ + baseZ; - stub->dPhi_ = (scat + digi(lengthR, baseR)) * inv2R + pitchOverR; - stub->dPhi_ = digi(stub->dPhi_, basePhi) + basePhi; - } - } - // store helper - auto frameTrack = [](Track* track) { return track->frame_; }; - auto frameStub = [this](Track* track, int layer) { - auto equal = [layer](Stub* stub) { return stub->channel_ == layer; }; - const auto it = find_if(track->stubs_.begin(), track->stubs_.end(), equal); - if (it == track->stubs_.end()) - return FrameStub(); - Stub* stub = *it; - const TTBV r(dataFormats_->format(Variable::r, Process::kfin).ttBV(stub->r_)); - const TTBV phi(dataFormats_->format(Variable::phi, Process::kfin).ttBV(stub->phi_)); - const TTBV z(dataFormats_->format(Variable::z, Process::kfin).ttBV(stub->z_)); - const TTBV dPhi(dataFormats_->format(Variable::dPhi, Process::kfin).ttBV(stub->dPhi_)); - const TTBV dZ(dataFormats_->format(Variable::dZ, Process::kfin).ttBV(stub->dZ_)); - return FrameStub(stub->ttStubRef_, Frame("1" + r.str() + phi.str() + z.str() + dPhi.str() + dZ.str())); - }; - // merge number of nodes DR to number of Nodes KF and store result - static const int nMux = channelAssignment_->numNodesDR() / setup_->kfNumWorker(); - const int offsetTrack = region_ * setup_->kfNumWorker(); - for (int nodeKF = 0; nodeKF < setup_->kfNumWorker(); nodeKF++) { - const int offset = nodeKF * nMux; - deque accepted; - deque lost; - vector> stacks(nMux); - vector> inputs(nMux); - for (int channel = 0; channel < nMux; channel++) { - const vector& input = input_[offset + channel]; - inputs[channel] = deque(input.begin(), input.end()); - } - // clock accurate firmware emulation, each while trip describes one clock tick, one stub in and one stub out per tick - while (!all_of(inputs.begin(), inputs.end(), [](const deque& tracks) { return tracks.empty(); }) or - !all_of(stacks.begin(), stacks.end(), [](const deque& tracks) { return tracks.empty(); })) { - // fill input fifos - for (int channel = 0; channel < nMux; channel++) { - deque& stack = stacks[channel]; - Track* track = pop_front(inputs[channel]); - if (track) - stack.push_back(track); - } - // merge input fifos to one stream, prioritizing higher input channel over lower channel - bool nothingToRoute(true); - for (int channel = nMux - 1; channel >= 0; channel--) { - Track* track = pop_front(stacks[channel]); - if (track) { - nothingToRoute = false; - accepted.push_back(track); - break; - } - } - if (nothingToRoute) - accepted.push_back(nullptr); - } - // truncate if desired - if (enableTruncation_ && (int)accepted.size() > setup_->numFrames()) { - const auto limit = next(accepted.begin(), setup_->numFrames()); - copy_if(limit, accepted.end(), back_inserter(lost), [](const Track* track) { return track; }); - accepted.erase(limit, accepted.end()); - } - // remove all gaps between end and last track - for (auto it = accepted.end(); it != accepted.begin();) - it = (*--it) ? accepted.begin() : accepted.erase(it); - // fill products StreamsStub& accpetedStubs, StreamsTrack& acceptedTracks, StreamsStub& lostStubs, StreamsTrack& lostTracks - const int channelTrack = offsetTrack + nodeKF; - const int offsetStub = channelTrack * setup_->numLayers(); - // fill lost tracks and stubs without gaps - lostTracks[channelTrack].reserve(lost.size()); - for (int layer = 0; layer < setup_->numLayers(); layer++) - lostStubs[offsetStub + layer].reserve(lost.size()); - for (Track* track : lost) { - lostTracks[channelTrack].emplace_back(frameTrack(track)); - for (int layer = 0; layer < setup_->numLayers(); layer++) - lostStubs[offsetStub + layer].emplace_back(frameStub(track, layer)); - } - // fill accepted tracks and stubs with gaps - acceptedTracks[channelTrack].reserve(accepted.size()); - for (int layer = 0; layer < setup_->numLayers(); layer++) - accpetedStubs[offsetStub + layer].reserve(accepted.size()); - for (Track* track : accepted) { - if (!track) { // fill gap - acceptedTracks[channelTrack].emplace_back(FrameTrack()); - for (int layer = 0; layer < setup_->numLayers(); layer++) - accpetedStubs[offsetStub + layer].emplace_back(FrameStub()); - continue; - } - acceptedTracks[channelTrack].emplace_back(frameTrack(track)); - for (int layer = 0; layer < setup_->numLayers(); layer++) - accpetedStubs[offsetStub + layer].emplace_back(frameStub(track, layer)); - } - } - } - - // remove and return first element of deque, returns nullptr if empty - template - T* KFin::pop_front(deque& ts) const { - T* t = nullptr; - if (!ts.empty()) { - t = ts.front(); - ts.pop_front(); - } - return t; - } - -} // namespace trklet diff --git a/L1Trigger/TrackFindingTracklet/src/KalmanFilter.cc b/L1Trigger/TrackFindingTracklet/src/KalmanFilter.cc new file mode 100644 index 0000000000000..27ff45ac6793a --- /dev/null +++ b/L1Trigger/TrackFindingTracklet/src/KalmanFilter.cc @@ -0,0 +1,721 @@ +#include "L1Trigger/TrackFindingTracklet/interface/KalmanFilter.h" +#include "L1Trigger/TrackFindingTMTT/interface/L1track3D.h" +#include "L1Trigger/TrackFindingTMTT/interface/Stub.h" +#include "L1Trigger/TrackFindingTMTT/interface/L1fittedTrack.h" + +#include +#include +#include +#include +#include +#include +#include + +using namespace std; +using namespace edm; +using namespace tt; +using namespace tmtt; + +namespace trklet { + + KalmanFilter::KalmanFilter(const ParameterSet& iConfig, + const Setup* setup, + const DataFormats* dataFormats, + KalmanFilterFormats* kalmanFilterFormats, + Settings* settings, + KFParamsComb* tmtt, + int region, + TTTracks& ttTracks) + : enableTruncation_(iConfig.getParameter("EnableTruncation")), + use5ParameterFit_(iConfig.getParameter("Use5ParameterFit")), + useSimmulation_(iConfig.getParameter("UseKFsimmulation")), + useTTStubResiduals_(iConfig.getParameter("UseTTStubResiduals")), + setup_(setup), + dataFormats_(dataFormats), + kalmanFilterFormats_(kalmanFilterFormats), + settings_(settings), + tmtt_(tmtt), + region_(region), + ttTracks_(ttTracks), + layer_(0) {} + + // read in and organize input tracks and stubs + void KalmanFilter::consume(const StreamsTrack& streamsTrack, const StreamsStub& streamsStub) { + static const int numLayers = setup_->numLayers(); + const int offset = region_ * numLayers; + const StreamTrack& streamTrack = streamsTrack[region_]; + const int numTracks = accumulate(streamTrack.begin(), streamTrack.end(), 0, [](int& sum, const FrameTrack& f) { + return sum += (f.first.isNull() ? 0 : 1); + }); + int numStubs(0); + for (int layer = 0; layer < numLayers; layer++) { + const StreamStub& streamStub = streamsStub[offset + layer]; + numStubs += accumulate(streamStub.begin(), streamStub.end(), 0, [](int& sum, const FrameStub& f) { + return sum += (f.first.isNull() ? 0 : 1); + }); + } + tracks_.reserve(numTracks); + stubs_.reserve(numStubs); + int trackId(0); + for (int frame = 0; frame < (int)streamTrack.size(); frame++) { + const FrameTrack& frameTrack = streamTrack[frame]; + if (frameTrack.first.isNull()) { + stream_.push_back(nullptr); + continue; + } + tracks_.emplace_back(frameTrack, dataFormats_); + TrackDR* track = &tracks_.back(); + vector stubs(numLayers, nullptr); + TTBV hitPattern(0, setup_->numLayers()); + for (int layer = 0; layer < numLayers; layer++) { + const FrameStub& frameStub = streamsStub[offset + layer][frame]; + if (frameStub.first.isNull()) + continue; + stubs_.emplace_back(kalmanFilterFormats_, frameStub); + stubs[layer] = &stubs_.back(); + hitPattern.set(layer); + } + if (hitPattern.count(0, setup_->kfMaxSeedingLayer()) < setup_->kfNumSeedStubs()) { + stream_.push_back(nullptr); + continue; + } + states_.emplace_back(kalmanFilterFormats_, track, stubs, trackId++); + stream_.push_back(&states_.back()); + if (enableTruncation_ && trackId == setup_->kfMaxTracks()) + break; + } + } + + // call old KF + void KalmanFilter::simulate(tt::StreamsStub& streamsStub, tt::StreamsTrack& streamsTrack) { + static vector zTs; + if (zTs.empty()) { + zTs.reserve(settings_->etaRegions().size()); + for (double eta : settings_->etaRegions()) + zTs.emplace_back(sinh(eta) * settings_->chosenRofZ()); + } + finals_.reserve(states_.size()); + for (const State& state : states_) { + TrackDR* trackFound = state.track(); + const TTTrackRef& ttTrackRef = trackFound->frame().first; + const double qOverPt = -trackFound->inv2R() / setup_->invPtToDphi(); + const double phi0 = + deltaPhi(trackFound->phiT() - setup_->chosenRofPhi() * trackFound->inv2R() + region_ * setup_->baseRegion()); + const double tanLambda = trackFound->zT() / setup_->chosenRofZ(); + static constexpr double z0 = 0; + static constexpr double helixD0 = 0.; + vector stubs; + vector stubsFound; + stubs.reserve(state.trackPattern().count()); + stubsFound.reserve(state.trackPattern().count()); + const vector& stubsState = state.stubs(); + for (int layer = 0; layer < setup_->numLayers(); layer++) { + if (!stubsState[layer]) + continue; + const StubDR& stub = stubsState[layer]->stubDR_; + const TTStubRef& ttStubRef = stub.frame().first; + SensorModule* sensorModule = setup_->sensorModule(ttStubRef); + double r, phi, z; + if (useTTStubResiduals_) { + const GlobalPoint gp = setup_->stubPos(ttStubRef); + r = gp.perp(); + phi = gp.phi(); + z = gp.z(); + } else { + r = stub.r() + setup_->chosenRofPhi(); + phi = deltaPhi(stub.phi() + trackFound->phiT() + stub.r() * trackFound->inv2R() + + region_ * setup_->baseRegion()); + z = stub.z() + trackFound->zT() + (r - setup_->chosenRofZ()) * tanLambda; + } + int layerId = setup_->layerId(ttStubRef); + if (layerId > 10 && z < 0.) + layerId += 10; + int layerIdReduced = setup_->layerId(ttStubRef); + if (layerIdReduced == 6) + layerIdReduced = 11; + else if (layerIdReduced == 5) + layerIdReduced = 12; + else if (layerIdReduced == 4) + layerIdReduced = 13; + else if (layerIdReduced == 3) + layerIdReduced = 15; + if (layerIdReduced > 10) + layerIdReduced -= 8; + const double stripPitch = sensorModule->pitchRow(); + const double stripLength = sensorModule->pitchCol(); + const bool psModule = sensorModule->psModule(); + const bool barrel = sensorModule->barrel(); + const bool tiltedBarrel = sensorModule->tilted(); + stubs.emplace_back( + ttStubRef, r, phi, z, layerId, layerIdReduced, stripPitch, stripLength, psModule, barrel, tiltedBarrel); + stubsFound.push_back(&stubs.back()); + } + const int iPhiSec = region_; + const double zTtrack = ttTrackRef->z0() + settings_->chosenRofZ() * ttTrackRef->tanL(); + int iEtaReg = 0; + for (; iEtaReg < 15; iEtaReg++) + if (zTtrack < zTs[iEtaReg + 1]) + break; + const L1track3D l1track3D(settings_, stubsFound, qOverPt, phi0, z0, tanLambda, helixD0, iPhiSec, iEtaReg); + //cout << "In " << qOverPt << " " << phi0 << " " << tanLambda << " " << z0 << endl; + const L1fittedTrack trackFitted(tmtt_->fit(l1track3D)); + //cout << "Out " << trackFitted.qOverPt() << " " << trackFitted.phi0() << " " << trackFitted.tanLambda() << " " << trackFitted.z0() << endl; + if (!trackFitted.accepted()) + //throw cms::Exception("..."); + continue; + static constexpr int trackId = 0; + static constexpr int numConsistent = 0; + static constexpr int numConsistentPS = 0; + const double inv2R = -trackFitted.qOverPt() * setup_->invPtToDphi(); + const double phiT = + deltaPhi(trackFitted.phi0() + inv2R * setup_->chosenRofPhi() - region_ * setup_->baseRegion()); + const double cot = trackFitted.tanLambda(); + const double zT = trackFitted.z0() + cot * setup_->chosenRofZ(); + if (!dataFormats_->format(Variable::inv2R, Process::kf).inRange(inv2R, true)) + continue; + if (!dataFormats_->format(Variable::phiT, Process::kf).inRange(phiT, true)) + continue; + if (!dataFormats_->format(Variable::cot, Process::kf).inRange(cot, true)) + continue; + if (!dataFormats_->format(Variable::zT, Process::kf).inRange(zT, true)) + continue; + const double d0 = trackFitted.d0(); + const double x0 = inv2R - trackFound->inv2R(); + const double x1 = phiT - trackFound->phiT(); + const double x2 = cot - tanLambda; + const double x3 = zT - trackFound->zT(); + const double x4 = d0; + TTBV hitPattern(0, setup_->numLayers()); + vector stubsKF; + stubsKF.reserve(setup_->numLayers()); + for (tmtt::Stub* stub : trackFitted.stubs()) { + if (!stub) + continue; + const auto it = find_if(stubsState.begin(), stubsState.end(), [stub](Stub* state) { + return state && (stub->ttStubRef() == state->stubDR_.frame().first); + }); + const StubDR& s = (*it)->stubDR_; + const double r = s.r(); + const double r0 = r + setup_->chosenRofPhi(); + const double phi = s.phi() - (x1 + r * x0 + x4 / r0); + const double z = s.z() - (x3 + (r0 - setup_->chosenRofZ()) * x2); + const double dPhi = s.dPhi(); + const double dZ = s.dZ(); + const int layer = distance(stubsState.begin(), it); + if (!dataFormats_->format(Variable::phi, Process::kf).inRange(phi, true)) + continue; + if (!dataFormats_->format(Variable::z, Process::kf).inRange(z, true)) + continue; + hitPattern.set(layer); + stubsKF.emplace_back(s, r, phi, z, dPhi, dZ); + } + if (hitPattern.count() < setup_->kfMinLayers()) + continue; + const TrackKF trackKF(*trackFound, inv2R, phiT, cot, zT); + finals_.emplace_back(trackId, numConsistent, numConsistentPS, d0, hitPattern, trackKF, stubsKF); + } + conv(streamsStub, streamsTrack); + } + + // fill output products + void KalmanFilter::produce(StreamsStub& streamsStub, + StreamsTrack& streamsTrack, + int& numAcceptedStates, + int& numLostStates) { + if (useSimmulation_) + return simulate(streamsStub, streamsTrack); + // 5 parameter fit simulation + if (use5ParameterFit_) { + // Propagate state to each layer in turn, updating it with all viable stub combinations there, using KF maths + for (layer_ = 0; layer_ < setup_->numLayers(); layer_++) + addLayer(); + } else { // 4 parameter fit emulation + // seed building + for (layer_ = 0; layer_ < setup_->kfMaxSeedingLayer(); layer_++) + addSeedLayer(); + // calulcate seed parameter + calcSeeds(); + // Propagate state to each layer in turn, updating it with all viable stub combinations there, using KF maths + for (layer_ = setup_->kfNumSeedStubs(); layer_ < setup_->numLayers(); layer_++) + addLayer(); + } + // count total number of final states + const int nStates = + accumulate(stream_.begin(), stream_.end(), 0, [](int& sum, State* state) { return sum += (state ? 1 : 0); }); + // apply truncation + if (enableTruncation_ && (int)stream_.size() > setup_->numFramesHigh()) + stream_.resize(setup_->numFramesHigh()); + // cycle event, remove gaps + stream_.erase(remove(stream_.begin(), stream_.end(), nullptr), stream_.end()); + // store number of states which got taken into account + numAcceptedStates += (int)stream_.size(); + // store number of states which got not taken into account due to truncation + numLostStates += nStates - (int)stream_.size(); + // apply final cuts + finalize(); + // best track per candidate selection + accumulator(); + // Transform States into output products + conv(streamsStub, streamsTrack); + } + + // apply final cuts + void KalmanFilter::finalize() { + finals_.reserve(stream_.size()); + for (State* state : stream_) { + int numConsistent(0); + int numConsistentPS(0); + TTBV hitPattern = state->hitPattern(); + vector stubsKF; + stubsKF.reserve(setup_->numLayers()); + // stub residual cut + State* s = state; + while ((s = s->parent())) { + const double dPhi = state->x1() + s->H00() * state->x0() + state->x4() / s->H04(); + const double dZ = state->x3() + s->H12() * state->x2(); + const double phi = digi(VariableKF::m0, s->m0() - dPhi); + const double z = digi(VariableKF::m1, s->m1() - dZ); + const bool validPhi = dataFormats_->format(Variable::phi, Process::kf).inRange(phi); + const bool validZ = dataFormats_->format(Variable::z, Process::kf).inRange(z); + if (validPhi && validZ) { + const double r = s->H00(); + const double dPhi = s->d0(); + const double dZ = s->d1(); + const StubDR& stubDR = s->stub()->stubDR_; + stubsKF.emplace_back(stubDR, r, phi, z, dPhi, dZ); + if (abs(phi) <= dPhi && abs(z) <= dZ) { + numConsistent++; + if (setup_->psModule(stubDR.frame().first)) + numConsistentPS++; + } + } else + hitPattern.reset(s->layer()); + } + reverse(stubsKF.begin(), stubsKF.end()); + // layer cut + bool validLayers = hitPattern.count() >= setup_->kfMinLayers(); + // track parameter cuts + const double cotTrack = + dataFormats_->format(Variable::cot, Process::kf).digi(state->track()->zT() / setup_->chosenRofZ()); + const double inv2R = state->x0() + state->track()->inv2R(); + const double phiT = state->x1() + state->track()->phiT(); + const double cot = state->x2() + cotTrack; + const double zT = state->x3() + state->track()->zT(); + const double d0 = state->x4(); + // pt cut + const bool validX0 = dataFormats_->format(Variable::inv2R, Process::kf).inRange(inv2R); + // cut on phi sector boundaries + const bool validX1 = abs(phiT) < setup_->baseRegion() / 2.; + // cot cut + const bool validX2 = dataFormats_->format(Variable::cot, Process::kf).inRange(cot); + // zT cut + const bool validX3 = dataFormats_->format(Variable::zT, Process::kf).inRange(zT); + if (!validLayers || !validX0 || !validX1 || !validX2 || !validX3) + continue; + const int trackId = state->trackId(); + const TrackKF trackKF(*state->track(), inv2R, phiT, cot, zT); + finals_.emplace_back(trackId, numConsistent, numConsistentPS, d0, hitPattern, trackKF, stubsKF); + } + } + + // best state selection + void KalmanFilter::accumulator() { + // create container of pointer to make sorts less CPU intense + vector finals; + finals.reserve(finals_.size()); + transform(finals_.begin(), finals_.end(), back_inserter(finals), [](Track& track) { return &track; }); + // prepare arrival order + vector trackIds; + trackIds.reserve(tracks_.size()); + for (Track* track : finals) { + const int trackId = track->trackId_; + if (find_if(trackIds.begin(), trackIds.end(), [trackId](int id) { return id == trackId; }) == trackIds.end()) + trackIds.push_back(trackId); + } + // sort in number of consistent stubs + auto moreConsistentLayers = [](Track* lhs, Track* rhs) { return lhs->numConsistent_ > rhs->numConsistent_; }; + stable_sort(finals.begin(), finals.end(), moreConsistentLayers); + // sort in number of consistent ps stubs + auto moreConsistentLayersPS = [](Track* lhs, Track* rhs) { return lhs->numConsistentPS_ > rhs->numConsistentPS_; }; + stable_sort(finals.begin(), finals.end(), moreConsistentLayersPS); + // sort in track id as arrived + auto order = [&trackIds](auto lhs, auto rhs) { + const auto l = find(trackIds.begin(), trackIds.end(), lhs->trackId_); + const auto r = find(trackIds.begin(), trackIds.end(), rhs->trackId_); + return distance(r, l) < 0; + }; + stable_sort(finals.begin(), finals.end(), order); + // keep first state (best due to previous sorts) per track id + const auto it = + unique(finals.begin(), finals.end(), [](Track* lhs, Track* rhs) { return lhs->trackId_ == rhs->trackId_; }); + finals.erase(it, finals.end()); + // apply to actual track container + int i(0); + for (Track* track : finals) + finals_[i++] = *track; + finals_.resize(i); + } + + // Transform States into output products + void KalmanFilter::conv(StreamsStub& streamsStub, StreamsTrack& streamsTrack) { + const int offset = region_ * setup_->numLayers(); + StreamTrack& streamTrack = streamsTrack[region_]; + streamTrack.reserve(stream_.size()); + for (int layer = 0; layer < setup_->numLayers(); layer++) + streamsStub[offset + layer].reserve(stream_.size()); + for (const Track& track : finals_) { + streamTrack.emplace_back(track.trackKF_.frame()); + const TTBV& hitPattern = track.hitPattern_; + const vector& stubsKF = track.stubsKF_; + int i(0); + for (int layer = 0; layer < setup_->numLayers(); layer++) + streamsStub[offset + layer].emplace_back(hitPattern.test(layer) ? stubsKF[i++].frame() : FrameStub()); + // store d0 in copied TTTracks + if (use5ParameterFit_) { + const TTTrackRef& ttTrackRef = track.trackKF_.frame().first; + ttTracks_.emplace_back(ttTrackRef->rInv(), + ttTrackRef->phi(), + ttTrackRef->tanL(), + ttTrackRef->z0(), + track.d0_, + ttTrackRef->chi2XY(), + ttTrackRef->chi2Z(), + ttTrackRef->trkMVA1(), + ttTrackRef->trkMVA2(), + ttTrackRef->trkMVA3(), + ttTrackRef->hitPattern(), + 5, + setup_->bField()); + ttTracks_.back().setPhiSector(ttTrackRef->phiSector()); + ttTracks_.back().setEtaSector(ttTrackRef->etaSector()); + ttTracks_.back().setTrackSeedType(ttTrackRef->trackSeedType()); + ttTracks_.back().setStubPtConsistency(ttTrackRef->stubPtConsistency()); + ttTracks_.back().setStubRefs(ttTrackRef->getStubRefs()); + } + } + } + + // calculates the helix params & their cov. matrix from a pair of stubs + void KalmanFilter::calcSeeds() { + auto update = [this](State* s) { + updateRangeActual(VariableKF::m0, s->m0()); + updateRangeActual(VariableKF::m1, s->m1()); + updateRangeActual(VariableKF::v0, s->v0()); + updateRangeActual(VariableKF::v1, s->v1()); + updateRangeActual(VariableKF::H00, s->H00()); + updateRangeActual(VariableKF::H12, s->H12()); + }; + for (State*& state : stream_) { + if (!state) + continue; + State* s1 = state->parent(); + State* s0 = s1->parent(); + update(s0); + update(s1); + const double dH = digi(VariableKF::dH, s1->H00() - s0->H00()); + const double invdH = digi(VariableKF::invdH, 1.0 / dH); + const double invdH2 = digi(VariableKF::invdH2, 1.0 / dH / dH); + const double H12 = digi(VariableKF::H2, s1->H00() * s1->H00()); + const double H02 = digi(VariableKF::H2, s0->H00() * s0->H00()); + const double H32 = digi(VariableKF::H2, s1->H12() * s1->H12()); + const double H22 = digi(VariableKF::H2, s0->H12() * s0->H12()); + const double H1m0 = digi(VariableKF::Hm0, s1->H00() * s0->m0()); + const double H0m1 = digi(VariableKF::Hm0, s0->H00() * s1->m0()); + const double H3m2 = digi(VariableKF::Hm1, s1->H12() * s0->m1()); + const double H2m3 = digi(VariableKF::Hm1, s0->H12() * s1->m1()); + const double H1v0 = digi(VariableKF::Hv0, s1->H00() * s0->v0()); + const double H0v1 = digi(VariableKF::Hv0, s0->H00() * s1->v0()); + const double H3v2 = digi(VariableKF::Hv1, s1->H12() * s0->v1()); + const double H2v3 = digi(VariableKF::Hv1, s0->H12() * s1->v1()); + const double H12v0 = digi(VariableKF::H2v0, H12 * s0->v0()); + const double H02v1 = digi(VariableKF::H2v0, H02 * s1->v0()); + const double H32v2 = digi(VariableKF::H2v1, H32 * s0->v1()); + const double H22v3 = digi(VariableKF::H2v1, H22 * s1->v1()); + const double x0 = digi(VariableKF::x0, (s1->m0() - s0->m0()) * invdH); + const double x2 = digi(VariableKF::x2, (s1->m1() - s0->m1()) * invdH); + const double x1 = digi(VariableKF::x1, (H1m0 - H0m1) * invdH); + const double x3 = digi(VariableKF::x3, (H3m2 - H2m3) * invdH); + const double C00 = digi(VariableKF::C00, (s1->v0() + s0->v0()) * invdH2); + const double C22 = digi(VariableKF::C22, (s1->v1() + s0->v1()) * invdH2); + const double C01 = -digi(VariableKF::C01, (H1v0 + H0v1) * invdH2); + const double C23 = -digi(VariableKF::C23, (H3v2 + H2v3) * invdH2); + const double C11 = digi(VariableKF::C11, (H12v0 + H02v1) * invdH2); + const double C33 = digi(VariableKF::C33, (H32v2 + H22v3) * invdH2); + // create updated state + states_.emplace_back(State(s1, {x0, x1, x2, x3, 0., C00, C11, C22, C33, C01, C23, 0., 0., 0.})); + state = &states_.back(); + updateRangeActual(VariableKF::x0, x0); + updateRangeActual(VariableKF::x1, x1); + updateRangeActual(VariableKF::x2, x2); + updateRangeActual(VariableKF::x3, x3); + updateRangeActual(VariableKF::C00, C00); + updateRangeActual(VariableKF::C01, C01); + updateRangeActual(VariableKF::C11, C11); + updateRangeActual(VariableKF::C22, C22); + updateRangeActual(VariableKF::C23, C23); + updateRangeActual(VariableKF::C33, C33); + } + } + + // adds a layer to states to build seeds + void KalmanFilter::addSeedLayer() { + // Latency of KF Associator block firmware + static constexpr int latency = 5; + // dynamic state container for clock accurate emulation + deque streamOutput; + // Memory stack used to handle combinatorics + deque stack; + // static delay container + deque delay(latency, nullptr); + // each trip corresponds to a f/w clock tick + // done if no states to process left, taking as much time as needed + while (!stream_.empty() || !stack.empty() || + !all_of(delay.begin(), delay.end(), [](const State* state) { return state == nullptr; })) { + State* state = pop_front(stream_); + // Process a combinatoric state if no (non-combinatoric?) state available + if (!state) + state = pop_front(stack); + streamOutput.push_back(state); + // The remainder of the code in this loop deals with combinatoric states. + if (state) + state = state->combSeed(states_, layer_); + delay.push_back(state); + state = pop_front(delay); + if (state) + stack.push_back(state); + } + stream_ = streamOutput; + // Update state with next stub using KF maths + for (State*& state : stream_) + if (state) + state = state->update(states_, layer_); + } + + // adds a layer to states + void KalmanFilter::addLayer() { + // Latency of KF Associator block firmware + static constexpr int latency = 5; + // dynamic state container for clock accurate emulation + deque streamOutput; + // Memory stack used to handle combinatorics + deque stack; + // static delay container + deque delay(latency, nullptr); + // each trip corresponds to a f/w clock tick + // done if no states to process left, taking as much time as needed + while (!stream_.empty() || !stack.empty() || + !all_of(delay.begin(), delay.end(), [](const State* state) { return state == nullptr; })) { + State* state = pop_front(stream_); + // Process a combinatoric state if no (non-combinatoric?) state available + if (!state) + state = pop_front(stack); + streamOutput.push_back(state); + // The remainder of the code in this loop deals with combinatoric states. + if (state) + state = state->comb(states_, layer_); + delay.push_back(state); + state = pop_front(delay); + if (state) + stack.push_back(state); + } + stream_ = streamOutput; + // Update state with next stub using KF maths + for (State*& state : stream_) + if (state && state->hitPattern().pmEncode() == layer_) + update(state); + } + + // updates state + void KalmanFilter::update4(State*& state) { + // All variable names & equations come from Fruhwirth KF paper http://dx.doi.org/10.1016/0168-9002%2887%2990887-4", where F taken as unit matrix. Stub uncertainties projected onto (phi,z), assuming no correlations between r-phi & r-z planes. + // stub phi residual wrt input helix + const double m0 = state->m0(); + // stub z residual wrt input helix + const double m1 = state->m1(); + // stub projected phi uncertainty squared); + const double v0 = state->v0(); + // stub projected z uncertainty squared + const double v1 = state->v1(); + // Derivative of predicted stub coords wrt helix params: stub radius minus chosenRofPhi + const double H00 = state->H00(); + // Derivative of predicted stub coords wrt helix params: stub radius minus chosenRofZ + const double H12 = state->H12(); + updateRangeActual(VariableKF::m0, m0); + updateRangeActual(VariableKF::m1, m1); + updateRangeActual(VariableKF::v0, v0); + updateRangeActual(VariableKF::v1, v1); + updateRangeActual(VariableKF::H00, H00); + updateRangeActual(VariableKF::H12, H12); + // helix inv2R wrt input helix + double x0 = state->x0(); + // helix phi at radius ChosenRofPhi wrt input helix + double x1 = state->x1(); + // helix cot(Theta) wrt input helix + double x2 = state->x2(); + // helix z at radius chosenRofZ wrt input helix + double x3 = state->x3(); + // cov. matrix + double C00 = state->C00(); + double C01 = state->C01(); + double C11 = state->C11(); + double C22 = state->C22(); + double C23 = state->C23(); + double C33 = state->C33(); + // stub phi residual wrt current state + const double r0C = digi(VariableKF::x1, m0 - x1); + const double r0 = digi(VariableKF::r0, r0C - x0 * H00); + // stub z residual wrt current state + const double r1C = digi(VariableKF::x3, m1 - x3); + const double r1 = digi(VariableKF::r1, r1C - x2 * H12); + // matrix S = H*C + const double S00 = digi(VariableKF::S00, C01 + H00 * C00); + const double S01 = digi(VariableKF::S01, C11 + H00 * C01); + const double S12 = digi(VariableKF::S12, C23 + H12 * C22); + const double S13 = digi(VariableKF::S13, C33 + H12 * C23); + // Cov. matrix of predicted residuals R = V+HCHt = C+H*St + const double R00 = digi(VariableKF::R00, v0 + S01 + H00 * S00); + const double R11 = digi(VariableKF::R11, v1 + S13 + H12 * S12); + // improved dynamic cancelling + const int msb0 = max(0, (int)ceil(log2(R00 / base(VariableKF::R00)))); + const int msb1 = max(0, (int)ceil(log2(R11 / base(VariableKF::R11)))); + const int shift0 = width(VariableKF::R00) - msb0; + const int shift1 = width(VariableKF::R11) - msb1; + const double R00Shifted = R00 * pow(2., shift0); + const double R11Shifted = R11 * pow(2., shift1); + const double R00Rough = digi(VariableKF::R00Rough, R00Shifted); + const double R11Rough = digi(VariableKF::R11Rough, R11Shifted); + const double invR00Approx = digi(VariableKF::invR00Approx, 1. / R00Rough); + const double invR11Approx = digi(VariableKF::invR11Approx, 1. / R11Rough); + const double invR00Cor = digi(VariableKF::invR00Cor, 2. - invR00Approx * R00Shifted); + const double invR11Cor = digi(VariableKF::invR11Cor, 2. - invR11Approx * R11Shifted); + const double invR00 = digi(VariableKF::invR00, invR00Approx * invR00Cor); + const double invR11 = digi(VariableKF::invR11, invR11Approx * invR11Cor); + // shift S to "undo" shifting of R + auto digiShifted = [](double val, double base) { return floor(val / base * 2. + 1.e-11) * base / 2.; }; + const double S00Shifted = digiShifted(S00 * pow(2., shift0), base(VariableKF::S00Shifted)); + const double S01Shifted = digiShifted(S01 * pow(2., shift0), base(VariableKF::S01Shifted)); + const double S12Shifted = digiShifted(S12 * pow(2., shift1), base(VariableKF::S12Shifted)); + const double S13Shifted = digiShifted(S13 * pow(2., shift1), base(VariableKF::S13Shifted)); + // Kalman gain matrix K = S*R(inv) + const double K00 = digi(VariableKF::K00, S00Shifted * invR00); + const double K10 = digi(VariableKF::K10, S01Shifted * invR00); + const double K21 = digi(VariableKF::K21, S12Shifted * invR11); + const double K31 = digi(VariableKF::K31, S13Shifted * invR11); + // Updated helix params, their cov. matrix + x0 = digi(VariableKF::x0, x0 + r0 * K00); + x1 = digi(VariableKF::x1, x1 + r0 * K10); + x2 = digi(VariableKF::x2, x2 + r1 * K21); + x3 = digi(VariableKF::x3, x3 + r1 * K31); + C00 = digi(VariableKF::C00, C00 - S00 * K00); + C01 = digi(VariableKF::C01, C01 - S01 * K00); + C11 = digi(VariableKF::C11, C11 - S01 * K10); + C22 = digi(VariableKF::C22, C22 - S12 * K21); + C23 = digi(VariableKF::C23, C23 - S13 * K21); + C33 = digi(VariableKF::C33, C33 - S13 * K31); + // update variable ranges to tune variable granularity + updateRangeActual(VariableKF::r0, r0); + updateRangeActual(VariableKF::r1, r1); + updateRangeActual(VariableKF::S00, S00); + updateRangeActual(VariableKF::S01, S01); + updateRangeActual(VariableKF::S12, S12); + updateRangeActual(VariableKF::S13, S13); + updateRangeActual(VariableKF::S00Shifted, S00Shifted); + updateRangeActual(VariableKF::S01Shifted, S01Shifted); + updateRangeActual(VariableKF::S12Shifted, S12Shifted); + updateRangeActual(VariableKF::S13Shifted, S13Shifted); + updateRangeActual(VariableKF::R00, R00); + updateRangeActual(VariableKF::R11, R11); + updateRangeActual(VariableKF::R00Rough, R00Rough); + updateRangeActual(VariableKF::R11Rough, R11Rough); + updateRangeActual(VariableKF::invR00Approx, invR00Approx); + updateRangeActual(VariableKF::invR11Approx, invR11Approx); + updateRangeActual(VariableKF::invR00Cor, invR00Cor); + updateRangeActual(VariableKF::invR11Cor, invR11Cor); + updateRangeActual(VariableKF::invR00, invR00); + updateRangeActual(VariableKF::invR11, invR11); + updateRangeActual(VariableKF::K00, K00); + updateRangeActual(VariableKF::K10, K10); + updateRangeActual(VariableKF::K21, K21); + updateRangeActual(VariableKF::K31, K31); + // create updated state + states_.emplace_back(State(state, {x0, x1, x2, x3, 0., C00, C11, C22, C33, C01, C23, 0., 0., 0.})); + state = &states_.back(); + updateRangeActual(VariableKF::x0, x0); + updateRangeActual(VariableKF::x1, x1); + updateRangeActual(VariableKF::x2, x2); + updateRangeActual(VariableKF::x3, x3); + updateRangeActual(VariableKF::C00, C00); + updateRangeActual(VariableKF::C01, C01); + updateRangeActual(VariableKF::C11, C11); + updateRangeActual(VariableKF::C22, C22); + updateRangeActual(VariableKF::C23, C23); + updateRangeActual(VariableKF::C33, C33); + } + + // updates state + void KalmanFilter::update5(State*& state) { + const double m0 = state->m0(); + const double m1 = state->m1(); + const double v0 = state->v0(); + const double v1 = state->v1(); + const double H00 = state->H00(); + const double H12 = state->H12(); + const double H04 = state->H04(); + double x0 = state->x0(); + double x1 = state->x1(); + double x2 = state->x2(); + double x3 = state->x3(); + double x4 = state->x4(); + double C00 = state->C00(); + double C01 = state->C01(); + double C11 = state->C11(); + double C22 = state->C22(); + double C23 = state->C23(); + double C33 = state->C33(); + double C44 = state->C44(); + double C40 = state->C40(); + double C41 = state->C41(); + const double r0 = m0 - x1 - x0 * H00 - x4 / H04; + const double r1 = m1 - x3 - x2 * H12; + const double S00 = C01 + H00 * C00 + C40 / H04; + const double S01 = C11 + H00 * C01 + C41 / H04; + const double S12 = C23 + H12 * C22; + const double S13 = C33 + H12 * C23; + const double S04 = C41 + H00 * C40 + C44 / H04; + const double R00 = v0 + S01 + H00 * S00 + S04 / H04; + const double R11 = v1 + S13 + H12 * S12; + const double K00 = S00 / R00; + const double K10 = S01 / R00; + const double K21 = S12 / R11; + const double K31 = S13 / R11; + const double K40 = S04 / R00; + x0 += r0 * K00; + x1 += r0 * K10; + x2 += r1 * K21; + x3 += r1 * K31; + x4 += r0 * K40; + C00 -= S00 * K00; + C01 -= S01 * K00; + C11 -= S01 * K10; + C22 -= S12 * K21; + C23 -= S13 * K21; + C33 -= S13 * K31; + C44 -= S04 * K40; + C40 -= S04 * K00; + C41 -= S04 * K10; + states_.emplace_back(State(state, {x0, x1, x2, x3, x4, C00, C11, C22, C33, C01, C23, C44, C40, C41})); + state = &states_.back(); + } + + // remove and return first element of deque, returns nullptr if empty + template + T* KalmanFilter::pop_front(deque& ts) const { + T* t = nullptr; + if (!ts.empty()) { + t = ts.front(); + ts.pop_front(); + } + return t; + } + +} // namespace trklet diff --git a/L1Trigger/TrackFindingTracklet/src/KalmanFilterFormats.cc b/L1Trigger/TrackFindingTracklet/src/KalmanFilterFormats.cc new file mode 100644 index 0000000000000..b105968dbd8eb --- /dev/null +++ b/L1Trigger/TrackFindingTracklet/src/KalmanFilterFormats.cc @@ -0,0 +1,588 @@ +#include "L1Trigger/TrackFindingTracklet/interface/KalmanFilterFormats.h" + +#include +#include +#include +#include +#include +#include +#include +#include + +using namespace std; +using namespace edm; +using namespace tt; + +namespace trklet { + + constexpr auto variableKFstrs_ = { + "x0", "x1", "x2", "x3", "H00", "H12", "m0", "m1", + "v0", "v1", "r0", "r1", "S00", "S01", "S12", "S13", + "S00Shifted", "S01Shifted", "S12Shifted", "S13Shifted", "K00", "K10", "K21", "K31", + "R00", "R11", "R00Rough", "R11Rough", "invR00Approx", "invR11Approx", "invR00Cor", "invR11Cor", + "invR00", "invR11", "C00", "C01", "C11", "C22", "C23", "C33"}; + + void KalmanFilterFormats::endJob() { + const int wName = + strlen(*max_element(variableKFstrs_.begin(), variableKFstrs_.end(), [](const auto& a, const auto& b) { + return strlen(a) < strlen(b); + })); + for (VariableKF v = VariableKF::begin; v != VariableKF::dH; v = VariableKF(+v + 1)) { + const double r = + format(v).twos() ? std::max(std::abs(format(v).min()), std::abs(format(v).max())) * 2. : format(v).max(); + const int delta = format(v).width() - ceil(log2(r / format(v).base())); + cout << setw(wName) << *next(variableKFstrs_.begin(), +v) << ": "; + cout << setw(3) << (delta == -2147483648 ? "-" : to_string(delta)) << endl; + } + } + + KalmanFilterFormats::KalmanFilterFormats(const ParameterSet& iConfig) : iConfig_(iConfig) { + formats_.reserve(+VariableKF::end); + } + + void KalmanFilterFormats::consume(const DataFormats* dataFormats) { + dataFormats_ = dataFormats; + fillFormats(); + } + + template + void KalmanFilterFormats::fillFormats() { + formats_.emplace_back(FormatKF(dataFormats_, iConfig_)); + if constexpr (++it != VariableKF::end) + fillFormats<++it>(); + } + + DataFormatKF::DataFormatKF(const VariableKF& v, bool twos, const ParameterSet& iConfig) + : v_(v), + twos_(twos), + enableIntegerEmulation_(iConfig.getParameter("EnableIntegerEmulation")), + width_(0), + base_(1.), + range_(0.), + min_(numeric_limits::max()), + abs_(numeric_limits::max()), + max_(numeric_limits::lowest()) {} + + // returns false if data format would oferflow for this double value + bool DataFormatKF::inRange(double d) const { + if (twos_) + return d >= -range_ / 2. && d < range_ / 2.; + return d >= 0 && d < range_; + } + + void DataFormatKF::updateRangeActual(double d) { + min_ = std::min(min_, d); + abs_ = std::min(abs_, std::abs(d)); + max_ = std::max(max_, d); + if (enableIntegerEmulation_ && !inRange(d)) { + string v = *next(variableKFstrs_.begin(), +v_); + cms::Exception exception("out_of_range"); + exception.addContext("trackFindingTracklet:DataFormatKF::updateRangeActual"); + exception << "Variable " << v << " = " << d << " is out of range " << (twos_ ? -range_ / 2. : 0) << " to " + << (twos_ ? range_ / 2. : range_) << "." << endl; + if (twos_ || d >= 0.) + exception.addAdditionalInfo("Consider raising BaseShift" + v + " in KalmanFilterFormats_cfi.py."); + exception.addAdditionalInfo("Consider disabling integer emulation in KalmanFilterFormats_cfi.py."); + throw exception; + } + } + + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const ParameterSet& iConfig) + : DataFormatKF(VariableKF::x0, true, iConfig) { + const DataFormat& input = dataFormats->format(Variable::inv2R, Process::kf); + const int baseShift = iConfig.getParameter("BaseShiftx0"); + base_ = pow(2, baseShift) * input.base(); + width_ = dataFormats->setup()->widthDSPbb(); + calcRange(); + } + + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const ParameterSet& iConfig) + : DataFormatKF(VariableKF::x1, true, iConfig) { + const DataFormat& input = dataFormats->format(Variable::phiT, Process::kf); + const int baseShift = iConfig.getParameter("BaseShiftx1"); + base_ = pow(2, baseShift) * input.base(); + width_ = dataFormats->setup()->widthDSPbb(); + calcRange(); + } + + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const ParameterSet& iConfig) + : DataFormatKF(VariableKF::x2, true, iConfig) { + const DataFormat& input = dataFormats->format(Variable::cot, Process::kf); + const int baseShift = iConfig.getParameter("BaseShiftx2"); + base_ = pow(2, baseShift) * input.base(); + width_ = dataFormats->setup()->widthDSPbb(); + calcRange(); + } + + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const ParameterSet& iConfig) + : DataFormatKF(VariableKF::x3, true, iConfig) { + const DataFormat& input = dataFormats->format(Variable::zT, Process::kf); + const int baseShift = iConfig.getParameter("BaseShiftx3"); + base_ = pow(2, baseShift) * input.base(); + width_ = dataFormats->setup()->widthDSPbb(); + calcRange(); + } + + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const ParameterSet& iConfig) + : DataFormatKF(VariableKF::H00, true, iConfig) { + const DataFormat& tm = dataFormats->format(Variable::r, Process::tm); + base_ = tm.base(); + width_ = tm.width(); + calcRange(); + } + + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const ParameterSet& iConfig) + : DataFormatKF(VariableKF::H12, true, iConfig) { + const Setup* setup = dataFormats->setup(); + const DataFormat& tm = dataFormats->format(Variable::r, Process::tm); + base_ = tm.base(); + range_ = 2. * setup->maxRz(); + width_ = ceil(log2(range_ / base_)); + calcRange(); + } + + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const ParameterSet& iConfig) + : DataFormatKF(VariableKF::m0, true, iConfig) { + const DataFormat& tm = dataFormats->format(Variable::phi, Process::tm); + base_ = tm.base(); + width_ = tm.width(); + calcRange(); + } + + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const ParameterSet& iConfig) + : DataFormatKF(VariableKF::m1, true, iConfig) { + const DataFormat& tm = dataFormats->format(Variable::z, Process::tm); + base_ = tm.base(); + width_ = tm.width(); + calcRange(); + } + + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const ParameterSet& iConfig) + : DataFormatKF(VariableKF::v0, false, iConfig) { + const DataFormat& dPhi = dataFormats->format(Variable::dPhi, Process::tm); + const FormatKF S01(dataFormats, iConfig); + range_ = 4. * dPhi.range() * dPhi.range(); + base_ = S01.base(); + calcWidth(); + } + + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const ParameterSet& iConfig) + : DataFormatKF(VariableKF::v1, false, iConfig) { + const DataFormat& dZ = dataFormats->format(Variable::dZ, Process::tm); + const FormatKF S13(dataFormats, iConfig); + range_ = 4. * dZ.range() * dZ.range(); + base_ = S13.base(); + calcWidth(); + } + + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const ParameterSet& iConfig) + : DataFormatKF(VariableKF::r0, true, iConfig) { + const DataFormat& x1 = dataFormats->format(Variable::phiT, Process::kf); + const int baseShift = iConfig.getParameter("BaseShiftr0"); + base_ = pow(2., baseShift) * x1.base(); + width_ = dataFormats->setup()->widthDSPbb(); + calcRange(); + } + + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const ParameterSet& iConfig) + : DataFormatKF(VariableKF::r1, true, iConfig) { + const DataFormat& x3 = dataFormats->format(Variable::zT, Process::kf); + const int baseShift = iConfig.getParameter("BaseShiftr1"); + base_ = pow(2., baseShift) * x3.base(); + width_ = dataFormats->setup()->widthDSPbb(); + calcRange(); + } + + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const ParameterSet& iConfig) + : DataFormatKF(VariableKF::S00, true, iConfig) { + const DataFormat& x0 = dataFormats->format(Variable::inv2R, Process::kf); + const DataFormat& x1 = dataFormats->format(Variable::phiT, Process::kf); + const int baseShift = iConfig.getParameter("BaseShiftS00"); + base_ = pow(2., baseShift) * x0.base() * x1.base(); + width_ = dataFormats->setup()->widthDSPab(); + calcRange(); + } + + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const ParameterSet& iConfig) + : DataFormatKF(VariableKF::S01, true, iConfig) { + const DataFormat& x1 = dataFormats->format(Variable::phiT, Process::kf); + const int baseShift = iConfig.getParameter("BaseShiftS01"); + base_ = pow(2., baseShift) * x1.base() * x1.base(); + width_ = dataFormats->setup()->widthDSPab(); + calcRange(); + } + + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const ParameterSet& iConfig) + : DataFormatKF(VariableKF::S12, true, iConfig) { + const DataFormat& x2 = dataFormats->format(Variable::cot, Process::kf); + const DataFormat& x3 = dataFormats->format(Variable::zT, Process::kf); + const int baseShift = iConfig.getParameter("BaseShiftS12"); + base_ = pow(2., baseShift) * x2.base() * x3.base(); + width_ = dataFormats->setup()->widthDSPab(); + calcRange(); + } + + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const ParameterSet& iConfig) + : DataFormatKF(VariableKF::S13, true, iConfig) { + const DataFormat& x3 = dataFormats->format(Variable::zT, Process::kf); + const int baseShift = iConfig.getParameter("BaseShiftS13"); + base_ = pow(2., baseShift) * x3.base() * x3.base(); + width_ = dataFormats->setup()->widthDSPab(); + calcRange(); + } + + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const ParameterSet& iConfig) + : DataFormatKF(VariableKF::S00Shifted, true, iConfig) { + const DataFormat& x0 = dataFormats->format(Variable::inv2R, Process::kf); + const DataFormat& x1 = dataFormats->format(Variable::phiT, Process::kf); + const int baseShift = iConfig.getParameter("BaseShiftS00Shifted"); + base_ = pow(2., baseShift) * x0.base() * x1.base(); + width_ = dataFormats->setup()->widthDSPab(); + calcRange(); + } + + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const ParameterSet& iConfig) + : DataFormatKF(VariableKF::S01Shifted, true, iConfig) { + const DataFormat& x1 = dataFormats->format(Variable::phiT, Process::kf); + const int baseShift = iConfig.getParameter("BaseShiftS01Shifted"); + base_ = pow(2., baseShift) * x1.base() * x1.base(); + width_ = dataFormats->setup()->widthDSPab(); + calcRange(); + } + + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const ParameterSet& iConfig) + : DataFormatKF(VariableKF::S12Shifted, true, iConfig) { + const DataFormat& x2 = dataFormats->format(Variable::cot, Process::kf); + const DataFormat& x3 = dataFormats->format(Variable::zT, Process::kf); + const int baseShift = iConfig.getParameter("BaseShiftS12Shifted"); + base_ = pow(2., baseShift) * x2.base() * x3.base(); + width_ = dataFormats->setup()->widthDSPab(); + calcRange(); + } + + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const ParameterSet& iConfig) + : DataFormatKF(VariableKF::S13Shifted, true, iConfig) { + const DataFormat& x3 = dataFormats->format(Variable::zT, Process::kf); + const int baseShift = iConfig.getParameter("BaseShiftS13Shifted"); + base_ = pow(2., baseShift) * x3.base() * x3.base(); + width_ = dataFormats->setup()->widthDSPab(); + calcRange(); + } + + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const ParameterSet& iConfig) + : DataFormatKF(VariableKF::K00, true, iConfig) { + const DataFormat& x0 = dataFormats->format(Variable::inv2R, Process::kf); + const DataFormat& x1 = dataFormats->format(Variable::phiT, Process::kf); + const int baseShift = iConfig.getParameter("BaseShiftK00"); + base_ = pow(2., baseShift) * x0.base() / x1.base(); + width_ = dataFormats->setup()->widthDSPbb(); + calcRange(); + } + + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const ParameterSet& iConfig) + : DataFormatKF(VariableKF::K10, true, iConfig) { + const int baseShift = iConfig.getParameter("BaseShiftK10"); + base_ = pow(2., baseShift); + width_ = dataFormats->setup()->widthDSPbb(); + calcRange(); + } + + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const ParameterSet& iConfig) + : DataFormatKF(VariableKF::K21, true, iConfig) { + const DataFormat& x2 = dataFormats->format(Variable::cot, Process::kf); + const DataFormat& x3 = dataFormats->format(Variable::zT, Process::kf); + const int baseShift = iConfig.getParameter("BaseShiftK21"); + base_ = pow(2., baseShift) * x2.base() / x3.base(); + width_ = dataFormats->setup()->widthDSPbb(); + calcRange(); + } + + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const ParameterSet& iConfig) + : DataFormatKF(VariableKF::K31, true, iConfig) { + const int baseShift = iConfig.getParameter("BaseShiftK31"); + base_ = pow(2., baseShift); + width_ = dataFormats->setup()->widthDSPbb(); + calcRange(); + } + + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const ParameterSet& iConfig) + : DataFormatKF(VariableKF::R00, false, iConfig) { + const DataFormat& x1 = dataFormats->format(Variable::phiT, Process::kf); + const int baseShift = iConfig.getParameter("BaseShiftR00"); + width_ = iConfig.getParameter("WidthR00"); + base_ = pow(2., baseShift) * x1.base() * x1.base(); + calcRange(); + } + + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const ParameterSet& iConfig) + : DataFormatKF(VariableKF::R11, false, iConfig) { + const DataFormat& x3 = dataFormats->format(Variable::zT, Process::kf); + const int baseShift = iConfig.getParameter("BaseShiftR11"); + width_ = iConfig.getParameter("WidthR11"); + base_ = pow(2., baseShift) * x3.base() * x3.base(); + calcRange(); + } + + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const ParameterSet& iConfig) + : DataFormatKF(VariableKF::R00Rough, false, iConfig) { + const FormatKF R00(dataFormats, iConfig); + width_ = dataFormats->setup()->widthAddrBRAM18(); + range_ = R00.range(); + const int baseShift = R00.width() - width_ - 1; + base_ = pow(2., baseShift) * R00.base(); + } + + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const ParameterSet& iConfig) + : DataFormatKF(VariableKF::R11Rough, false, iConfig) { + const FormatKF R11(dataFormats, iConfig); + width_ = dataFormats->setup()->widthAddrBRAM18(); + range_ = R11.range(); + const int baseShift = R11.width() - width_ - 1; + base_ = pow(2., baseShift) * R11.base(); + } + + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const ParameterSet& iConfig) + : DataFormatKF(VariableKF::invR00Approx, false, iConfig) { + const DataFormat& x1 = dataFormats->format(Variable::phiT, Process::kf); + const int baseShift = iConfig.getParameter("BaseShiftInvR00Approx"); + base_ = pow(2., baseShift) / x1.base() / x1.base(); + width_ = dataFormats->setup()->widthDSPbu(); + calcRange(); + } + + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const ParameterSet& iConfig) + : DataFormatKF(VariableKF::invR11Approx, false, iConfig) { + const DataFormat& x3 = dataFormats->format(Variable::zT, Process::kf); + const int baseShift = iConfig.getParameter("BaseShiftInvR11Approx"); + base_ = pow(2., baseShift) / x3.base() / x3.base(); + width_ = dataFormats->setup()->widthDSPbu(); + calcRange(); + } + + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const ParameterSet& iConfig) + : DataFormatKF(VariableKF::invR00Cor, false, iConfig) { + const int baseShift = iConfig.getParameter("BaseShiftInvR00Cor"); + base_ = pow(2., baseShift); + width_ = dataFormats->setup()->widthDSPau(); + calcRange(); + } + + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const ParameterSet& iConfig) + : DataFormatKF(VariableKF::invR11Cor, false, iConfig) { + const int baseShift = iConfig.getParameter("BaseShiftInvR11Cor"); + base_ = pow(2., baseShift); + width_ = dataFormats->setup()->widthDSPau(); + calcRange(); + } + + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const ParameterSet& iConfig) + : DataFormatKF(VariableKF::invR00, false, iConfig) { + const DataFormat& x1 = dataFormats->format(Variable::phiT, Process::kf); + const int baseShift = iConfig.getParameter("BaseShiftInvR00"); + base_ = pow(2., baseShift) / x1.base() / x1.base(); + width_ = dataFormats->setup()->widthDSPbu(); + calcRange(); + } + + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const ParameterSet& iConfig) + : DataFormatKF(VariableKF::invR11, false, iConfig) { + const DataFormat& x3 = dataFormats->format(Variable::zT, Process::kf); + const int baseShift = iConfig.getParameter("BaseShiftInvR11"); + base_ = pow(2., baseShift) / x3.base() / x3.base(); + width_ = dataFormats->setup()->widthDSPbu(); + calcRange(); + } + + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const ParameterSet& iConfig) + : DataFormatKF(VariableKF::C00, false, iConfig) { + const DataFormat& x0 = dataFormats->format(Variable::inv2R, Process::kf); + const int baseShift = iConfig.getParameter("BaseShiftC00"); + width_ = iConfig.getParameter("WidthC00"); + base_ = pow(2., baseShift) * x0.base() * x0.base(); + calcRange(); + } + + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const ParameterSet& iConfig) + : DataFormatKF(VariableKF::C01, true, iConfig) { + const DataFormat& x0 = dataFormats->format(Variable::inv2R, Process::kf); + const DataFormat& x1 = dataFormats->format(Variable::phiT, Process::kf); + const int baseShift = iConfig.getParameter("BaseShiftC01"); + width_ = iConfig.getParameter("WidthC01"); + base_ = pow(2., baseShift) * x0.base() * x1.base(); + calcRange(); + } + + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const ParameterSet& iConfig) + : DataFormatKF(VariableKF::C11, false, iConfig) { + const DataFormat& x1 = dataFormats->format(Variable::phiT, Process::kf); + const int baseShift = iConfig.getParameter("BaseShiftC11"); + width_ = iConfig.getParameter("WidthC11"); + base_ = pow(2., baseShift) * x1.base() * x1.base(); + calcRange(); + } + + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const ParameterSet& iConfig) + : DataFormatKF(VariableKF::C22, false, iConfig) { + const DataFormat& x2 = dataFormats->format(Variable::cot, Process::kf); + const int baseShift = iConfig.getParameter("BaseShiftC22"); + width_ = iConfig.getParameter("WidthC22"); + base_ = pow(2., baseShift) * x2.base() * x2.base(); + calcRange(); + } + + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const ParameterSet& iConfig) + : DataFormatKF(VariableKF::C23, true, iConfig) { + const DataFormat& x2 = dataFormats->format(Variable::cot, Process::kf); + const DataFormat& x3 = dataFormats->format(Variable::zT, Process::kf); + const int baseShift = iConfig.getParameter("BaseShiftC23"); + width_ = iConfig.getParameter("WidthC23"); + base_ = pow(2., baseShift) * x2.base() * x3.base(); + calcRange(); + } + + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const ParameterSet& iConfig) + : DataFormatKF(VariableKF::C33, false, iConfig) { + const DataFormat& x3 = dataFormats->format(Variable::zT, Process::kf); + const int baseShift = iConfig.getParameter("BaseShiftC33"); + width_ = iConfig.getParameter("WidthC33"); + base_ = pow(2., baseShift) * x3.base() * x3.base(); + calcRange(); + } + + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const ParameterSet& iConfig) + : DataFormatKF(VariableKF::end, false, iConfig) { + const Setup* setup = dataFormats->setup(); + const DataFormat& tm = dataFormats->format(Variable::r, Process::tm); + width_ = setup->widthAddrBRAM18(); + range_ = setup->outerRadius() - setup->innerRadius(); + base_ = tm.base() * pow(2, ceil(log2(range_ / tm.base())) - width_); + } + + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const ParameterSet& iConfig) + : DataFormatKF(VariableKF::end, false, iConfig) { + const FormatKF H00(dataFormats, iConfig); + const Setup* setup = dataFormats->setup(); + width_ = setup->widthDSPbu(); + range_ = 1. / setup->kfMinSeedDeltaR(); + const int baseShift = ceil(log2(range_ * pow(2., -width_) * H00.base())); + base_ = pow(2., baseShift) / H00.base(); + } + + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const ParameterSet& iConfig) + : DataFormatKF(VariableKF::end, false, iConfig) { + const FormatKF H00(dataFormats, iConfig); + const Setup* setup = dataFormats->setup(); + width_ = setup->widthDSPbu(); + range_ = 1. / pow(setup->kfMinSeedDeltaR(), 2); + const double baseH2 = H00.base() * H00.base(); + const int baseShift = ceil(log2(range_ * pow(2., -width_) * baseH2)); + base_ = pow(2., baseShift) / baseH2; + } + + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const ParameterSet& iConfig) + : DataFormatKF(VariableKF::end, false, iConfig) { + const FormatKF H00(dataFormats, iConfig); + base_ = H00.base() * H00.base(); + } + + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const ParameterSet& iConfig) + : DataFormatKF(VariableKF::end, true, iConfig) { + const FormatKF H00(dataFormats, iConfig); + const FormatKF m0(dataFormats, iConfig); + base_ = H00.base() * m0.base(); + } + + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const ParameterSet& iConfig) + : DataFormatKF(VariableKF::end, true, iConfig) { + const FormatKF H12(dataFormats, iConfig); + const FormatKF m1(dataFormats, iConfig); + base_ = H12.base() * m1.base(); + } + + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const ParameterSet& iConfig) + : DataFormatKF(VariableKF::end, false, iConfig) { + const Setup* setup = dataFormats->setup(); + const FormatKF H00(dataFormats, iConfig); + const FormatKF v0(dataFormats, iConfig); + width_ = setup->widthDSPab(); + base_ = H00.base() * v0.base() * pow(2, H00.width() + v0.width() - width_); + } + + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const ParameterSet& iConfig) + : DataFormatKF(VariableKF::end, false, iConfig) { + const FormatKF H12(dataFormats, iConfig); + const Setup* setup = dataFormats->setup(); + const FormatKF v1(dataFormats, iConfig); + width_ = setup->widthDSPab(); + base_ = H12.base() * v1.base() * pow(2, H12.width() + v1.width() - width_); + } + + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const ParameterSet& iConfig) + : DataFormatKF(VariableKF::end, false, iConfig) { + const Setup* setup = dataFormats->setup(); + const FormatKF H00(dataFormats, iConfig); + const FormatKF v0(dataFormats, iConfig); + width_ = setup->widthDSPau(); + base_ = H00.base() * H00.base() * v0.base() * pow(2, 2 * H00.width() + v0.width() - width_); + } + + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const ParameterSet& iConfig) + : DataFormatKF(VariableKF::end, false, iConfig) { + const Setup* setup = dataFormats->setup(); + const FormatKF H12(dataFormats, iConfig); + const FormatKF v1(dataFormats, iConfig); + width_ = setup->widthDSPau(); + base_ = H12.base() * H12.base() * v1.base() * pow(2, 2 * H12.width() + v1.width() - width_); + } + +} // namespace trklet diff --git a/L1Trigger/TrackFindingTracklet/src/State.cc b/L1Trigger/TrackFindingTracklet/src/State.cc new file mode 100644 index 0000000000000..0bf4a08162479 --- /dev/null +++ b/L1Trigger/TrackFindingTracklet/src/State.cc @@ -0,0 +1,170 @@ +#include "L1Trigger/TrackFindingTracklet/interface/State.h" + +#include +#include +#include +#include +#include + +using namespace std; +using namespace tt; + +namespace trklet { + + // + State::Stub::Stub(KalmanFilterFormats* kff, const FrameStub& frame) : stubDR_(frame, kff->dataFormats()) { + const Setup* setup = kff->setup(); + H12_ = kff->format(VariableKF::H12).digi(stubDR_.r() + setup->chosenRofPhi() - setup->chosenRofZ()); + H04_ = stubDR_.r() + setup->chosenRofPhi(); + v0_ = kff->format(VariableKF::v0).digi(pow(2. * stubDR_.dPhi(), 2)); + v1_ = kff->format(VariableKF::v1).digi(pow(2. * stubDR_.dZ(), 2)); + } + + // proto state constructor + State::State(KalmanFilterFormats* kff, TrackDR* track, const vector& stubs, int trackId) + : kff_(kff), + setup_(kff->setup()), + track_(track), + stubs_(stubs), + trackId_(trackId), + parent_(nullptr), + stub_(nullptr), + hitPattern_(0, setup_->numLayers()), + trackPattern_(0, setup_->numLayers()), + x0_(0.), + x1_(0.), + x2_(0.), + x3_(0.), + x4_(0.), + C00_(9.e9), + C01_(0.), + C11_(9.e9), + C22_(9.e9), + C23_(0.), + C33_(9.e9), + C44_(pow(setup_->maxD0(), 2)), + C40_(0.), + C41_(0.) { + int layer(0); + for (Stub* stub : stubs_) + trackPattern_[layer++] = (bool)stub; + layer = trackPattern_.plEncode(); + stub_ = stubs_[layer]; + hitPattern_.set(layer); + } + + // updated state constructor + State::State(State* state, const vector& doubles) : State(state) { + parent_ = state; + // updated track parameter and uncertainties + x0_ = doubles[0]; + x1_ = doubles[1]; + x2_ = doubles[2]; + x3_ = doubles[3]; + x4_ = doubles[4]; + C00_ = doubles[5]; + C11_ = doubles[6]; + C22_ = doubles[7]; + C33_ = doubles[8]; + C01_ = doubles[9]; + C23_ = doubles[10]; + C44_ = doubles[11]; + C40_ = doubles[12]; + C41_ = doubles[13]; + // pick next stub + const int layer = this->layer(); + stub_ = nullptr; + if (hitPattern_.count() >= setup_->kfMinLayers() || hitPattern_.count() == setup_->kfMaxLayers()) + return; + const int nextLayer = trackPattern_.plEncode(layer + 1, setup_->numLayers()); + if (nextLayer == setup_->numLayers()) + return; + stub_ = stubs_[nextLayer]; + hitPattern_.set(nextLayer); + } + + // combinatoric and seed building state constructor + State::State(State* state, State* parent, int layer) : State(state) { + parent_ = parent; + hitPattern_ = parent ? parent->hitPattern() : TTBV(0, setup_->numLayers()); + stub_ = stubs_[layer]; + hitPattern_.set(layer); + } + + // + State* State::update(deque& states, int layer) { + if (!hitPattern_.test(layer) || hitPattern_.count() > setup_->kfNumSeedStubs()) + return this; + const int nextLayer = trackPattern_.plEncode(layer + 1, setup_->numLayers()); + states.emplace_back(this, this, nextLayer); + return &states.back(); + } + + // + State* State::combSeed(deque& states, int layer) { + // handle trivial state + if (!hitPattern_.test(layer) || hitPattern_.count() > setup_->kfNumSeedStubs()) + return nullptr; + // skip layers + const int nextLayer = trackPattern_.plEncode(layer + 1, setup_->numLayers()); + const int maxSeedStubs = hitPattern_.count(0, layer) + trackPattern_.count(nextLayer, setup_->kfMaxSeedingLayer()); + if (maxSeedStubs < setup_->kfNumSeedStubs()) + return nullptr; + const int maxStubs = maxSeedStubs + trackPattern_.count(setup_->kfMaxSeedingLayer(), setup_->numLayers()); + if (maxStubs < setup_->kfMinLayers()) + return nullptr; + states.emplace_back(this, parent_, nextLayer); + return &states.back(); + } + + // + State* State::comb(deque& states, int layer) { + // handle skipping and min reached + if (!hitPattern_.test(layer)) { + if (!stub_ && trackPattern_[layer] && hitPattern_.count() < setup_->kfMaxLayers()) { + states.emplace_back(this, parent_, layer); + return &states.back(); + } + return nullptr; + } + // handle part of seed + if (hitPattern_.pmEncode() != layer) + return nullptr; + // handle skip + const int nextLayer = trackPattern_.plEncode(layer + 1, setup_->numLayers()); + if (nextLayer == setup_->numLayers()) + return nullptr; + // not enough layer left + if (hitPattern_.count() - 1 + trackPattern_.count(nextLayer, setup_->numLayers()) < setup_->kfMinLayers()) + return nullptr; + states.emplace_back(this, parent_, nextLayer); + return &states.back(); + } + + // copy constructor + State::State(State* state) + : kff_(state->kff_), + setup_(state->setup_), + track_(state->track_), + stubs_(state->stubs_), + trackId_(state->trackId_), + parent_(state->parent_), + stub_(state->stub_), + hitPattern_(state->hitPattern_), + trackPattern_(state->trackPattern_), + x0_(state->x0_), + x1_(state->x1_), + x2_(state->x2_), + x3_(state->x3_), + x4_(state->x4_), + C00_(state->C00_), + C01_(state->C01_), + C11_(state->C11_), + C22_(state->C22_), + C23_(state->C23_), + C33_(state->C33_), + C44_(state->C44_), + C40_(state->C40_), + C41_(state->C41_) {} + +} // namespace trklet diff --git a/L1Trigger/TrackFindingTracklet/src/TrackMultiplexer.cc b/L1Trigger/TrackFindingTracklet/src/TrackMultiplexer.cc new file mode 100644 index 0000000000000..5dd59d8d691f2 --- /dev/null +++ b/L1Trigger/TrackFindingTracklet/src/TrackMultiplexer.cc @@ -0,0 +1,426 @@ +#include "L1Trigger/TrackFindingTracklet/interface/TrackMultiplexer.h" + +#include +#include +#include +#include +#include + +using namespace std; +using namespace edm; +using namespace tt; + +namespace trklet { + + TrackMultiplexer::TrackMultiplexer(const ParameterSet& iConfig, + const Setup* setup, + const DataFormats* dataFormats, + const ChannelAssignment* channelAssignment, + const Settings* settings, + int region) + : enableTruncation_(iConfig.getParameter("EnableTruncation")), + useTTStubResiduals_(iConfig.getParameter("UseTTStubResiduals")), + useTTStubParameters_(iConfig.getParameter("UseTTStubParameters")), + applyNonLinearCorrection_(iConfig.getParameter("ApplyNonLinearCorrection")), + setup_(setup), + dataFormats_(dataFormats), + channelAssignment_(channelAssignment), + settings_(settings), + region_(region), + input_(channelAssignment_->numChannelsTrack()) { + // unified tracklet digitisation granularity + baseUinv2R_ = .5 * settings_->kphi1() / settings_->kr() * pow(2, settings_->rinv_shift()); + baseUphiT_ = settings_->kphi1() * pow(2, settings_->phi0_shift()); + baseUcot_ = settings_->kz() / settings_->kr() * pow(2, settings_->t_shift()); + baseUzT_ = settings_->kz() * pow(2, settings_->z0_shift()); + baseUr_ = settings_->kr(); + baseUphi_ = settings_->kphi1(); + baseUz_ = settings_->kz(); + // DR input format digitisation granularity (identical to TMTT) + baseLinv2R_ = dataFormats->base(Variable::inv2R, Process::tm); + baseLphiT_ = dataFormats->base(Variable::phiT, Process::tm); + baseLzT_ = dataFormats->base(Variable::zT, Process::tm); + baseLr_ = dataFormats->base(Variable::r, Process::tm); + baseLphi_ = dataFormats->base(Variable::phi, Process::tm); + baseLz_ = dataFormats->base(Variable::z, Process::tm); + baseLcot_ = baseLz_ / baseLr_; + // Finer granularity (by powers of 2) than the TMTT one. Used to transform from Tracklet to TMTT base. + baseHinv2R_ = baseLinv2R_ * pow(2, floor(log2(baseUinv2R_ / baseLinv2R_))); + baseHphiT_ = baseLphiT_ * pow(2, floor(log2(baseUphiT_ / baseLphiT_))); + baseHzT_ = baseLzT_ * pow(2, floor(log2(baseUzT_ / baseLzT_))); + baseHr_ = baseLr_ * pow(2, floor(log2(baseUr_ / baseLr_))); + baseHphi_ = baseLphi_ * pow(2, floor(log2(baseUphi_ / baseLphi_))); + baseHz_ = baseLz_ * pow(2, floor(log2(baseUz_ / baseLz_))); + baseHcot_ = baseLcot_ * pow(2, floor(log2(baseUcot_ / baseLcot_))); + // calculate digitisation granularity used for inverted cot(theta) + const int baseShiftInvCot = ceil(log2(setup_->outerRadius() / setup_->hybridRangeR())) - setup_->widthDSPbu(); + baseInvCot_ = pow(2, baseShiftInvCot); + const int unusedMSBScot = + floor(log2(baseUcot_ * pow(2.0, channelAssignment_->tmWidthCot()) / 2. / setup_->maxCot())); + const int baseShiftScot = channelAssignment_->tmWidthCot() - unusedMSBScot - 1 - setup_->widthAddrBRAM18(); + baseScot_ = baseUcot_ * pow(2.0, baseShiftScot); + } + + // read in and organize input tracks and stubs + void TrackMultiplexer::consume(const StreamsTrack& streamsTrack, const StreamsStub& streamsStub) { + static const int numS = channelAssignment_->numSeedingLayers(); + const int offsetTrack = region_ * channelAssignment_->numChannelsTrack(); + // count tracks and stubs to reserve container + int nTracks(0); + int nStubs(0); + for (int channel = 0; channel < channelAssignment_->numChannelsTrack(); channel++) { + const int channelTrack = offsetTrack + channel; + const int offsetStub = channelAssignment_->offsetStub(channelTrack); + const StreamTrack& streamTrack = streamsTrack[channelTrack]; + input_[channel].reserve(streamTrack.size()); + for (int frame = 0; frame < (int)streamTrack.size(); frame++) { + if (streamTrack[frame].first.isNull()) + continue; + nTracks++; + for (int layer = 0; layer < channelAssignment_->numProjectionLayers(channel); layer++) + if (streamsStub[offsetStub + layer][frame].first.isNonnull()) + nStubs++; + } + } + stubs_.reserve(nStubs + nTracks * numS); + tracks_.reserve(nTracks); + // store tracks and stubs + for (int channel = 0; channel < channelAssignment_->numChannelsTrack(); channel++) { + const int numP = channelAssignment_->numProjectionLayers(channel); + const int channelTrack = offsetTrack + channel; + const int offsetStub = channelAssignment_->offsetStub(channelTrack); + const StreamTrack& streamTrack = streamsTrack[channelTrack]; + vector& input = input_[channel]; + for (int frame = 0; frame < (int)streamTrack.size(); frame++) { + const TTTrackRef& ttTrackRef = streamTrack[frame].first; + if (ttTrackRef.isNull()) { + input.push_back(nullptr); + continue; + } + //convert track parameter + const double offset = region_ * setup_->baseRegion(); + double inv2R = digi(-ttTrackRef->rInv() / 2., baseUinv2R_); + const double phi0U = digi(tt::deltaPhi(ttTrackRef->phi() - offset + setup_->hybridRangePhi() / 2.), baseUphiT_); + const double phi0S = digi(phi0U - setup_->hybridRangePhi() / 2., baseUphiT_); + double cot = digi(ttTrackRef->tanL(), baseUcot_); + double z0 = digi(ttTrackRef->z0(), baseUzT_); + double phiT = digi(phi0S + inv2R * digi(setup_->chosenRofPhi(), baseUr_), baseUphiT_); + double zT = digi(z0 + cot * digi(setup_->chosenRofZ(), baseUr_), baseUzT_); + // convert stubs + vector stubs; + stubs.reserve(numS + numP); + for (int layer = 0; layer < numP; layer++) { + const FrameStub& frameStub = streamsStub[offsetStub + layer][frame]; + const TTStubRef& ttStubRef = frameStub.first; + if (ttStubRef.isNull()) + continue; + // parse residuals from tt::Frame and take layerId from tt::TTStubRef + const bool barrel = setup_->barrel(ttStubRef); + const int layerIdTracklet = setup_->trackletLayerId(ttStubRef); + const double basePhi = barrel ? settings_->kphi1() : settings_->kphi(layerIdTracklet); + const double baseRZ = barrel ? settings_->kz(layerIdTracklet) : settings_->kz(); + const int widthRZ = barrel ? settings_->zresidbits() : settings_->rresidbits(); + TTBV hw(frameStub.second); + const TTBV hwRZ(hw, widthRZ, 0, true); + hw >>= widthRZ; + const TTBV hwPhi(hw, settings_->phiresidbits(), 0, true); + hw >>= settings_->phiresidbits(); + const int indexLayerId = setup_->indexLayerId(ttStubRef); + const SensorModule::Type type = setup_->type(ttStubRef); + const int widthR = setup_->tbWidthR(type); + const double baseR = setup_->hybridBaseR(type); + const TTBV hwR(hw, widthR, 0, barrel); + hw >>= widthR; + const TTBV hwStubId(hw, channelAssignment_->tmWidthStubId(), 0, false); + const int stubId = hwStubId.val(); + double r = hwR.val(baseR) + (barrel ? setup_->hybridLayerR(indexLayerId) : 0.); + if (type == SensorModule::Disk2S) + r = setup_->disk2SR(indexLayerId, r); + r = digi(r - setup_->chosenRofPhi(), baseUr_); + double phi = hwPhi.val(basePhi); + if (basePhi > baseUphi_) + phi += baseUphi_ / 2.; + double z = digi(hwRZ.val(baseRZ) * (barrel ? 1. : -cot), baseUz_); + // determine module type + bool psTilt = setup_->psModule(ttStubRef); + if (barrel) { + const double posZ = (r + digi(setup_->chosenRofPhi(), baseUr_)) * cot + z0 + z; + const int indexLayerId = setup_->indexLayerId(ttStubRef); + const double limit = setup_->tiltedLayerLimitZ(indexLayerId); + psTilt = abs(posZ) < limit; + } + stubs_.emplace_back(ttStubRef, layerIdTracklet, stubId, r, phi, z, psTilt); + stubs.push_back(&stubs_.back()); + } + if (useTTStubParameters_) { + vector seedTTStubRefs; + seedTTStubRefs.reserve(numS); + map mapStubs; + for (TTStubRef& ttStubRef : ttTrackRef->getStubRefs()) + mapStubs.emplace(setup_->layerId(ttStubRef), ttStubRef); + for (int layer : channelAssignment_->seedingLayers(ttTrackRef->trackSeedType())) + seedTTStubRefs.push_back(mapStubs[layer]); + const GlobalPoint gp0 = setup_->stubPos(seedTTStubRefs[0]); + const GlobalPoint gp1 = setup_->stubPos(seedTTStubRefs[1]); + const double dH = gp1.perp() - gp0.perp(); + const double H1m0 = (gp1.perp() - setup_->chosenRofPhi()) * deltaPhi(gp0.phi() - offset); + const double H0m1 = (gp0.perp() - setup_->chosenRofPhi()) * deltaPhi(gp1.phi() - offset); + const double H3m2 = (gp1.perp() - setup_->chosenRofZ()) * gp0.z(); + const double H2m3 = (gp0.perp() - setup_->chosenRofZ()) * gp1.z(); + const double dinv2R = inv2R - (gp1.phi() - gp0.phi()) / dH; + const double dcot = cot - (gp1.z() - gp0.z()) / dH; + const double dphiT = phiT - (H1m0 - H0m1) / dH; + const double dzT = zT - (H3m2 - H2m3) / dH; + inv2R -= dinv2R; + cot -= dcot; + phiT -= dphiT; + zT -= dzT; + z0 = zT - cot * setup_->chosenRofZ(); + // adjust stub residuals by track parameter shifts + for (Stub* stub : stubs) { + const double dphi = digi(dphiT + stub->r_ * dinv2R, baseUphi_); + const double r = stub->r_ + digi(setup_->chosenRofPhi() - setup_->chosenRofZ(), baseUr_); + const double dz = digi(dzT + r * dcot, baseUz_); + stub->phi_ = digi(stub->phi_ + dphi, baseUphi_); + stub->z_ = digi(stub->z_ + dz, baseUz_); + } + } + // create fake seed stubs, since TrackBuilder doesn't output these stubs, required by the KF. + for (int seedingLayer = 0; seedingLayer < numS; seedingLayer++) { + const int channelStub = numP + seedingLayer; + const FrameStub& frameStub = streamsStub[offsetStub + channelStub][frame]; + const TTStubRef& ttStubRef = frameStub.first; + const int trackletLayerId = setup_->trackletLayerId(ttStubRef); + const int layerId = channelAssignment_->layerId(channel, channelStub); + const int stubId = TTBV(frameStub.second).val(channelAssignment_->tmWidthStubId()); + const bool barrel = setup_->barrel(ttStubRef); + double r; + if (barrel) { + const int index = layerId - setup_->offsetLayerId(); + const double layer = digi(setup_->hybridLayerR(index), baseUr_); + const double z = digi(z0 + layer * cot, baseUz_); + if (abs(z) < digi(setup_->tbBarrelHalfLength(), baseUz_) || index > 0) + r = digi(setup_->hybridLayerR(index) - setup_->chosenRofPhi(), baseUr_); + else { + r = digi(setup_->innerRadius() - setup_->chosenRofPhi(), baseUr_); + } + } else { + const int index = layerId - setup_->offsetLayerId() - setup_->offsetLayerDisks(); + const double side = cot < 0. ? -1. : 1.; + const double disk = digi(setup_->hybridDiskZ(index), baseUzT_); + const double invCot = digi(1. / digi(abs(cot), baseScot_), baseInvCot_); + const double offset = digi(setup_->chosenRofPhi(), baseUr_); + r = digi((disk - side * z0) * invCot - offset, baseUr_); + } + double phi = 0.; + double z = 0.; + // determine module type + bool psTilt; + if (barrel) { + const int indexLayerId = setup_->indexLayerId(ttStubRef); + const double limit = digi(setup_->tiltedLayerLimitZ(indexLayerId), baseUz_); + const double posR = digi(setup_->hybridLayerR(layerId - setup_->offsetLayerId()), baseUr_); + const double posZ = digi(posR * cot + z0, baseUz_); + psTilt = abs(posZ) < limit; + } else + psTilt = true; + stubs_.emplace_back(ttStubRef, trackletLayerId, stubId, r, phi, z, psTilt); + stubs.push_back(&stubs_.back()); + } + if (useTTStubResiduals_) { + for (Stub* stub : stubs) { + const GlobalPoint gp = setup_->stubPos(stub->ttStubRef_); + stub->r_ = gp.perp() - setup_->chosenRofPhi(); + stub->phi_ = deltaPhi(gp.phi() - region_ * setup_->baseRegion()); + stub->phi_ -= phiT + stub->r_ * inv2R; + stub->z_ = gp.z() - (z0 + gp.perp() * cot); + } + } + // non linear corrections + if (applyNonLinearCorrection_) { + for (Stub* stub : stubs) { + const double d = inv2R * (stub->r_ + setup_->chosenRofPhi()); + const double dPhi = asin(d) - d; + stub->phi_ -= dPhi; + stub->z_ -= dPhi / inv2R * cot; + } + } + // check track validity + bool valid = true; + // kill truncated rtacks + if (enableTruncation_ && frame >= setup_->numFramesHigh()) + valid = false; + // kill tracks outside of fiducial range + if (!dataFormats_->format(Variable::phiT, Process::tm).inRange(phiT, true)) + valid = false; + if (!dataFormats_->format(Variable::zT, Process::tm).inRange(zT, true)) + valid = false; + // stub range checks + for (Stub* stub : stubs) { + if (!dataFormats_->format(Variable::phi, Process::tm).inRange(stub->phi_, true)) + stub->valid_ = false; + if (!dataFormats_->format(Variable::z, Process::tm).inRange(stub->z_, true)) + stub->valid_ = false; + } + // layer check + set layers, layersPS; + for (Stub* stub : stubs) { + if (!stub->valid_) + continue; + const int layerId = setup_->layerId(stub->ttStubRef_); + layers.insert(layerId); + if (setup_->psModule(stub->ttStubRef_)) + layersPS.insert(layerId); + } + if ((int)layers.size() < setup_->kfMinLayers() || (int)(int)layersPS.size() < setup_->kfMinLayersPS()) + valid = false; + // create track + tracks_.emplace_back(ttTrackRef, valid, channel, inv2R, phiT, cot, zT, stubs); + input.push_back(&tracks_.back()); + } + } + } + + // fill output products + void TrackMultiplexer::produce(StreamsTrack& streamsTrack, StreamsStub& streamsStub) { + // base transform into high precision TMTT format + for (Track& track : tracks_) { + track.inv2R_ = redigi(track.inv2R_, baseUinv2R_, baseHinv2R_, setup_->widthDSPbu()); + track.phiT_ = redigi(track.phiT_, baseUphiT_, baseHphiT_, setup_->widthDSPbu()); + track.cot_ = redigi(track.cot_, baseUcot_, baseHcot_, setup_->widthDSPbu()); + track.zT_ = redigi(track.zT_, baseUzT_, baseHzT_, setup_->widthDSPbu()); + for (Stub* stub : track.stubs_) { + stub->r_ = redigi(stub->r_, baseUr_, baseHr_, setup_->widthDSPbu()); + stub->phi_ = redigi(stub->phi_, baseUphi_, baseHphi_, setup_->widthDSPbu()); + stub->z_ = redigi(stub->z_, baseUz_, baseHz_, setup_->widthDSPbu()); + } + } + // base transform into TMTT format + for (Track& track : tracks_) { + // store track parameter shifts + const double dinv2R = digi(track.inv2R_ - digi(track.inv2R_, baseLinv2R_), baseHinv2R_); + const double dphiT = digi(track.phiT_ - digi(track.phiT_, baseLphiT_), baseHphiT_); + const double dcot = track.cot_ - digi(digi(track.zT_, baseLzT_) / setup_->chosenRofZ(), baseHcot_); + const double dzT = digi(track.zT_ - digi(track.zT_, baseLzT_), baseHzT_); + // shift track parameter; + track.inv2R_ -= dinv2R; + track.phiT_ -= dphiT; + track.cot_ -= dcot; + track.zT_ -= dzT; + // range checks + if (!dataFormats_->format(Variable::inv2R, Process::tm).inRange(track.inv2R_, true)) + track.valid_ = false; + if (!dataFormats_->format(Variable::phiT, Process::tm).inRange(track.phiT_, true)) + track.valid_ = false; + if (!dataFormats_->format(Variable::zT, Process::tm).inRange(track.zT_, true)) + track.valid_ = false; + // adjust stub residuals by track parameter shifts + for (Stub* stub : track.stubs_) { + const double dphi = digi(dphiT + stub->r_ * dinv2R, baseHphi_); + const double r = stub->r_ + digi(setup_->chosenRofPhi() - setup_->chosenRofZ(), baseHr_); + const double dz = digi(dzT + r * dcot, baseHz_); + stub->phi_ = digi(stub->phi_ + dphi, baseLphi_); + stub->z_ = digi(stub->z_ + dz, baseLz_); + // range checks + if (!dataFormats_->format(Variable::phi, Process::tm).inRange(stub->phi_, true)) + stub->valid_ = false; + if (!dataFormats_->format(Variable::z, Process::tm).inRange(stub->z_, true)) + stub->valid_ = false; + } + } + // emualte clock domain crossing + static constexpr int ticksPerGap = 3; + static constexpr int gapPos = 1; + vector> streams(channelAssignment_->numChannelsTrack()); + for (int channel = 0; channel < channelAssignment_->numChannelsTrack(); channel++) { + int iTrack(0); + deque& stream = streams[channel]; + const vector& intput = input_[channel]; + for (int tick = 0; iTrack < (int)intput.size(); tick++) { + Track* track = tick % ticksPerGap != gapPos ? intput[iTrack++] : nullptr; + stream.push_back(track && track->valid_ ? track : nullptr); + } + } + // remove all gaps between end and last track + for (deque& stream : streams) + for (auto it = stream.end(); it != stream.begin();) + it = (*--it) ? stream.begin() : stream.erase(it); + // route into single channel + deque accepted; + vector> stacks(channelAssignment_->numChannelsTrack()); + // clock accurate firmware emulation, each while trip describes one clock tick, one stub in and one stub out per tick + while (!all_of(streams.begin(), streams.end(), [](const deque& tracks) { return tracks.empty(); }) or + !all_of(stacks.begin(), stacks.end(), [](const deque& tracks) { return tracks.empty(); })) { + // fill input fifos + for (int channel = 0; channel < channelAssignment_->numChannelsTrack(); channel++) { + Track* track = pop_front(streams[channel]); + if (track) + stacks[channel].push_back(track); + } + // merge input fifos to one stream, prioritizing lower input channel over higher channel, affects DR + bool nothingToRoute(true); + for (int channel = 0; channel < channelAssignment_->numChannelsTrack(); channel++) { + Track* track = pop_front(stacks[channel]); + if (track) { + nothingToRoute = false; + accepted.push_back(track); + break; + } + } + if (nothingToRoute) + accepted.push_back(nullptr); + } + // truncate if desired + if (enableTruncation_ && (int)accepted.size() > setup_->numFramesHigh()) + accepted.resize(setup_->numFramesHigh()); + // remove all gaps between end and last track + for (auto it = accepted.end(); it != accepted.begin();) + it = (*--it) ? accepted.begin() : accepted.erase(it); + // store helper + auto frameTrack = [this](Track* track) { return track->valid_ ? track->frame(dataFormats_) : FrameTrack(); }; + auto frameStub = [this](Track* track, int layer) { + const auto it = + find_if(track->stubs_.begin(), track->stubs_.end(), [layer](Stub* stub) { return stub->layer_ == layer; }); + if (!track->valid_ || it == track->stubs_.end() || !(*it)->valid_) + return FrameStub(); + + return (*it)->frame(dataFormats_); + }; + static const int numLayers = channelAssignment_->tmNumLayers(); + const int offsetStub = region_ * numLayers; + // fill output tracks and stubs + streamsTrack[region_].reserve(accepted.size()); + for (int layer = 0; layer < numLayers; layer++) + streamsStub[offsetStub + layer].reserve(accepted.size()); + for (Track* track : accepted) { + if (!track) { // fill gaps + streamsTrack[region_].emplace_back(FrameTrack()); + for (int layer = 0; layer < numLayers; layer++) + streamsStub[offsetStub + layer].emplace_back(FrameStub()); + continue; + } + streamsTrack[region_].emplace_back(frameTrack(track)); + for (int layer = 0; layer < numLayers; layer++) + streamsStub[offsetStub + layer].emplace_back(frameStub(track, layer)); + } + } + + // remove and return first element of deque, returns nullptr if empty + template + T* TrackMultiplexer::pop_front(deque& ts) const { + T* t = nullptr; + if (!ts.empty()) { + t = ts.front(); + ts.pop_front(); + } + return t; + } + + // basetransformation of val from baseLow into baseHigh using widthMultiplier bit multiplication + double TrackMultiplexer::redigi(double val, double baseLow, double baseHigh, int widthMultiplier) const { + const double base = pow(2, 1 - widthMultiplier); + const double transform = digi(baseLow / baseHigh, base); + return (floor(val * transform / baseLow) + .5) * baseHigh; + } + +} // namespace trklet diff --git a/L1Trigger/TrackFindingTracklet/test/AnalyzerDR.cc b/L1Trigger/TrackFindingTracklet/test/AnalyzerDR.cc index fc5e36f397075..d48e360bb2c7d 100644 --- a/L1Trigger/TrackFindingTracklet/test/AnalyzerDR.cc +++ b/L1Trigger/TrackFindingTracklet/test/AnalyzerDR.cc @@ -62,13 +62,9 @@ namespace trklet { int& sum, bool perfect = false) const; // ED input token of stubs - EDGetTokenT edGetTokenAcceptedStubs_; + EDGetTokenT edGetTokenStubs_; // ED input token of tracks - EDGetTokenT edGetTokenAcceptedTracks_; - // ED input token of lost stubs - EDGetTokenT edGetTokenLostStubs_; - // ED input token of lost tracks - EDGetTokenT edGetTokenLostTracks_; + EDGetTokenT edGetTokenTracks_; // ED input token of TTStubRef to TPPtr association for tracking efficiency EDGetTokenT edGetTokenSelection_; // ED input token of TTStubRef to recontructable TPPtr association @@ -95,6 +91,7 @@ namespace trklet { TProfile* prof_; TProfile* profChannel_; TH1F* hisChannel_; + TH1F* hisTracks_; // printout stringstream log_; @@ -103,15 +100,11 @@ namespace trklet { AnalyzerDR::AnalyzerDR(const ParameterSet& iConfig) : useMCTruth_(iConfig.getParameter("UseMCTruth")) { usesResource("TFileService"); // book in- and output ED products - const string& label = iConfig.getParameter("LabelDR"); - const string& branchAcceptedStubs = iConfig.getParameter("BranchAcceptedStubs"); - const string& branchAcceptedTracks = iConfig.getParameter("BranchAcceptedTracks"); - const string& branchLostStubs = iConfig.getParameter("BranchLostStubs"); - const string& branchLostTracks = iConfig.getParameter("BranchLostTracks"); - edGetTokenAcceptedStubs_ = consumes(InputTag(label, branchAcceptedStubs)); - edGetTokenAcceptedTracks_ = consumes(InputTag(label, branchAcceptedTracks)); - edGetTokenLostStubs_ = consumes(InputTag(label, branchLostStubs)); - edGetTokenLostTracks_ = consumes(InputTag(label, branchLostTracks)); + const string& label = iConfig.getParameter("OutputLabelDR"); + const string& branchStubs = iConfig.getParameter("BranchStubs"); + const string& branchTracks = iConfig.getParameter("BranchTracks"); + edGetTokenStubs_ = consumes(InputTag(label, branchStubs)); + edGetTokenTracks_ = consumes(InputTag(label, branchTracks)); if (useMCTruth_) { const auto& inputTagSelecttion = iConfig.getParameter("InputTagSelection"); const auto& inputTagReconstructable = iConfig.getParameter("InputTagReconstructable"); @@ -141,35 +134,28 @@ namespace trklet { prof_ = dir.make("Counts", ";", 10, 0.5, 10.5); prof_->GetXaxis()->SetBinLabel(1, "Stubs"); prof_->GetXaxis()->SetBinLabel(2, "Tracks"); - prof_->GetXaxis()->SetBinLabel(3, "Lost Tracks"); prof_->GetXaxis()->SetBinLabel(4, "Matched Tracks"); prof_->GetXaxis()->SetBinLabel(5, "All Tracks"); prof_->GetXaxis()->SetBinLabel(6, "Found TPs"); prof_->GetXaxis()->SetBinLabel(7, "Found selected TPs"); - prof_->GetXaxis()->SetBinLabel(8, "Lost TPs"); prof_->GetXaxis()->SetBinLabel(9, "All TPs"); prof_->GetXaxis()->SetBinLabel(10, "Perfect TPs"); // channel occupancy constexpr int maxOcc = 180; - const int numChannels = channelAssignment_->numNodesDR(); + const int numChannels = 1; hisChannel_ = dir.make("His Channel Occupancy", ";", maxOcc, -.5, maxOcc - .5); profChannel_ = dir.make("Prof Channel Occupancy", ";", numChannels, -.5, numChannels - .5); + hisTracks_ = dir.make("His Number of Tracks", ";", maxOcc, -.5, maxOcc - .5); } void AnalyzerDR::analyze(const Event& iEvent, const EventSetup& iSetup) { // read in ht products - Handle handleAcceptedStubs; - iEvent.getByToken(edGetTokenAcceptedStubs_, handleAcceptedStubs); - const StreamsStub& acceptedStubs = *handleAcceptedStubs; - Handle handleAcceptedTracks; - iEvent.getByToken(edGetTokenAcceptedTracks_, handleAcceptedTracks); - const StreamsTrack& acceptedTracks = *handleAcceptedTracks; - Handle handleLostStubs; - iEvent.getByToken(edGetTokenLostStubs_, handleLostStubs); - const StreamsStub& lostStubs = *handleLostStubs; - Handle handleLostTracks; - iEvent.getByToken(edGetTokenLostTracks_, handleLostTracks); - const StreamsTrack& lostTracks = *handleLostTracks; + Handle handleStubs; + iEvent.getByToken(edGetTokenStubs_, handleStubs); + const StreamsStub& streamsStub = *handleStubs; + Handle handleTracks; + iEvent.getByToken(edGetTokenTracks_, handleTracks); + const StreamsTrack& streamsTrack = *handleTracks; // read in MCTruth const StubAssociation* selection = nullptr; const StubAssociation* reconstructable = nullptr; @@ -186,53 +172,38 @@ namespace trklet { set tpPtrs; set tpPtrsSelection; set tpPtrsPerfect; - set tpPtrsLost; int allMatched(0); int allTracks(0); for (int region = 0; region < setup_->numRegions(); region++) { - const int offset = region * channelAssignment_->numNodesDR(); int nStubs(0); int nTracks(0); - int nLost(0); - for (int channel = 0; channel < channelAssignment_->numNodesDR(); channel++) { - vector> tracks; - formTracks(acceptedTracks, acceptedStubs, tracks, offset + channel); - vector> lost; - formTracks(lostTracks, lostStubs, lost, offset + channel); - nTracks += tracks.size(); - nStubs += accumulate(tracks.begin(), tracks.end(), 0, [](int sum, const vector& track) { - return sum + static_cast(track.size()); - }); - nLost += lost.size(); - allTracks += tracks.size(); - if (!useMCTruth_) - continue; - int tmp(0); - associate(tracks, selection, tpPtrsSelection, tmp); - associate(tracks, selection, tpPtrsPerfect, tmp, true); - associate(lost, selection, tpPtrsLost, tmp); - associate(tracks, reconstructable, tpPtrs, allMatched); - const StreamTrack& stream = acceptedTracks[offset + channel]; - const auto end = - find_if(stream.rbegin(), stream.rend(), [](const FrameTrack& frame) { return frame.first.isNonnull(); }); - const int size = distance(stream.begin(), end.base()) - 1; - hisChannel_->Fill(size); - profChannel_->Fill(channel, size); - } + vector> tracks; + formTracks(streamsTrack, streamsStub, tracks, region); + nTracks += tracks.size(); + nStubs += accumulate(tracks.begin(), tracks.end(), 0, [](int& sum, const vector& track) { + return sum += (int)track.size(); + }); + allTracks += tracks.size(); + if (!useMCTruth_) + continue; + int tmp(0); + associate(tracks, selection, tpPtrsSelection, tmp); + associate(tracks, selection, tpPtrsPerfect, tmp, true); + associate(tracks, reconstructable, tpPtrs, allMatched); + const StreamTrack& stream = streamsTrack[region]; + const auto end = + find_if(stream.rbegin(), stream.rend(), [](const FrameTrack& frame) { return frame.first.isNonnull(); }); + const int size = distance(stream.begin(), end.base()) - 1; + hisTracks_->Fill(nTracks); + hisChannel_->Fill(size); + profChannel_->Fill(1, size); prof_->Fill(1, nStubs); prof_->Fill(2, nTracks); - prof_->Fill(3, nLost); } - vector recovered; - recovered.reserve(tpPtrsLost.size()); - set_intersection(tpPtrsLost.begin(), tpPtrsLost.end(), tpPtrs.begin(), tpPtrs.end(), back_inserter(recovered)); - for (const TPPtr& tpPtr : recovered) - tpPtrsLost.erase(tpPtr); prof_->Fill(4, allMatched); prof_->Fill(5, allTracks); prof_->Fill(6, tpPtrs.size()); prof_->Fill(7, tpPtrsSelection.size()); - prof_->Fill(8, tpPtrsLost.size()); prof_->Fill(10, tpPtrsPerfect.size()); nEvents_++; } @@ -240,46 +211,38 @@ namespace trklet { void AnalyzerDR::endJob() { if (nEvents_ == 0) return; - // printout SF summary + // printout summary const double totalTPs = prof_->GetBinContent(9); const double numStubs = prof_->GetBinContent(1); const double numTracks = prof_->GetBinContent(2); - const double numTracksLost = prof_->GetBinContent(3); const double totalTracks = prof_->GetBinContent(5); const double numTracksMatched = prof_->GetBinContent(4); const double numTPsAll = prof_->GetBinContent(6); const double numTPsEff = prof_->GetBinContent(7); - const double numTPsLost = prof_->GetBinContent(8); const double numTPsEffPerfect = prof_->GetBinContent(10); const double errStubs = prof_->GetBinError(1); const double errTracks = prof_->GetBinError(2); - const double errTracksLost = prof_->GetBinError(3); const double fracFake = (totalTracks - numTracksMatched) / totalTracks; const double fracDup = (numTracksMatched - numTPsAll) / totalTracks; const double eff = numTPsEff / totalTPs; const double errEff = sqrt(eff * (1. - eff) / totalTPs / nEvents_); - const double effLoss = numTPsLost / totalTPs; - const double errEffLoss = sqrt(effLoss * (1. - effLoss) / totalTPs / nEvents_); const double effPerfect = numTPsEffPerfect / totalTPs; const double errEffPerfect = sqrt(effPerfect * (1. - effPerfect) / totalTPs / nEvents_); - const vector nums = {numStubs, numTracks, numTracksLost}; - const vector errs = {errStubs, errTracks, errTracksLost}; + const vector nums = {numStubs, numTracks}; + const vector errs = {errStubs, errTracks}; const int wNums = ceil(log10(*max_element(nums.begin(), nums.end()))) + 5; const int wErrs = ceil(log10(*max_element(errs.begin(), errs.end()))) + 5; log_ << " DR SUMMARY " << endl; log_ << "number of stubs per TFP = " << setw(wNums) << numStubs << " +- " << setw(wErrs) << errStubs << endl; log_ << "number of tracks per TFP = " << setw(wNums) << numTracks << " +- " << setw(wErrs) << errTracks << endl; - log_ << "number of lost tracks per TFP = " << setw(wNums) << numTracksLost << " +- " << setw(wErrs) << errTracksLost - << endl; log_ << " current tracking efficiency = " << setw(wNums) << effPerfect << " +- " << setw(wErrs) << errEffPerfect << endl; log_ << " max tracking efficiency = " << setw(wNums) << eff << " +- " << setw(wErrs) << errEff << endl; - log_ << " lost tracking efficiency = " << setw(wNums) << effLoss << " +- " << setw(wErrs) << errEffLoss << endl; log_ << " fake rate = " << setw(wNums) << fracFake << endl; log_ << " duplicate rate = " << setw(wNums) << fracDup << endl; log_ << "============================================================="; - LogPrint("L1Trigger/TrackerTFP") << log_.str(); + LogPrint(moduleDescription().moduleName()) << log_.str(); } // diff --git a/L1Trigger/TrackFindingTracklet/test/AnalyzerDemonstrator.cc b/L1Trigger/TrackFindingTracklet/test/AnalyzerDemonstrator.cc index cd0e596601bfb..98b224b1b0df5 100644 --- a/L1Trigger/TrackFindingTracklet/test/AnalyzerDemonstrator.cc +++ b/L1Trigger/TrackFindingTracklet/test/AnalyzerDemonstrator.cc @@ -25,8 +25,8 @@ using namespace trackerTFP; namespace trklet { /*! \class trklet::AnalyzerDemonstrator - * \brief Class to demontrate correctness of track trigger emulators - * by comparing FW with SW + * \brief calls questasim to simulate the f/w and compares the results with clock-and-bit-accurate emulation. + * At the end the number of passing events (not a single bit error) are reported. * \author Thomas Schuh * \date 2022, March */ @@ -44,7 +44,8 @@ namespace trklet { void convert(const Event& iEvent, const EDGetTokenT& tokenTracks, const EDGetTokenT& tokenStubs, - vector>& bits) const; + vector>& bits, + bool TB = false) const; // template void convert(const T& collection, vector>& bits) const; @@ -70,26 +71,30 @@ namespace trklet { int nEvents_ = 0; // int nEventsSuccessful_ = 0; + // + bool TBin_; + bool TBout_; }; AnalyzerDemonstrator::AnalyzerDemonstrator(const ParameterSet& iConfig) { // book in- and output ED products const string& labelIn = iConfig.getParameter("LabelIn"); const string& labelOut = iConfig.getParameter("LabelOut"); - const string& branchStubs = iConfig.getParameter("BranchAcceptedStubs"); - const string& branchTracks = iConfig.getParameter("BranchAcceptedTracks"); + const string& branchStubs = iConfig.getParameter("BranchStubs"); + const string& branchTracks = iConfig.getParameter("BranchTracks"); edGetTokenStubsIn_ = consumes(InputTag(labelIn, branchStubs)); edGetTokenStubsOut_ = consumes(InputTag(labelOut, branchStubs)); - if (labelOut != "TrackFindingTrackletProducerKFout") - edGetTokenStubsOut_ = consumes(InputTag(labelOut, branchStubs)); - if (labelIn != "TrackFindingTrackletProducerIRin") + if (labelIn != "ProducerIRin") edGetTokenTracksIn_ = consumes(InputTag(labelIn, branchTracks)); - if (labelOut != "TrackFindingTrackletProducerIRin") + if (labelOut != "ProducerIRin") edGetTokenTracksOut_ = consumes(InputTag(labelOut, branchTracks)); // book ES products esGetTokenSetup_ = esConsumes(); esGetTokenChannelAssignment_ = esConsumes(); esGetTokenDemonstrator_ = esConsumes(); + // + TBin_ = labelIn == "l1tTTTracksFromTrackletEmulation"; + TBout_ = labelOut == "l1tTTTracksFromTrackletEmulation"; } void AnalyzerDemonstrator::beginRun(const Run& iEvent, const EventSetup& iSetup) { @@ -105,8 +110,8 @@ namespace trklet { nEvents_++; vector> input; vector> output; - convert(iEvent, edGetTokenTracksIn_, edGetTokenStubsIn_, input); - convert(iEvent, edGetTokenTracksOut_, edGetTokenStubsOut_, output); + convert(iEvent, edGetTokenTracksIn_, edGetTokenStubsIn_, input, TBin_); + convert(iEvent, edGetTokenTracksOut_, edGetTokenStubsOut_, output, TBout_); if (demonstrator_->analyze(input, output)) nEventsSuccessful_++; } @@ -115,36 +120,41 @@ namespace trklet { void AnalyzerDemonstrator::convert(const Event& iEvent, const EDGetTokenT& tokenTracks, const EDGetTokenT& tokenStubs, - vector>& bits) const { + vector>& bits, + bool TB) const { const bool tracks = !tokenTracks.isUninitialized(); const bool stubs = !tokenStubs.isUninitialized(); Handle handleStubs; Handle handleTracks; + int numChannelStubs(0); + if (stubs) { + iEvent.getByToken(tokenStubs, handleStubs); + numChannelStubs = handleStubs->size(); + } int numChannelTracks(0); if (tracks) { iEvent.getByToken(tokenTracks, handleTracks); numChannelTracks = handleTracks->size(); } numChannelTracks /= setup_->numRegions(); - int numChannelStubs(0); - vector numChannelsStubs(numChannelTracks, 0); - if (stubs) { - iEvent.getByToken(tokenStubs, handleStubs); - numChannelStubs = handleStubs->size() / setup_->numRegions(); - const int numChannel = tracks ? numChannelStubs / numChannelTracks : numChannelStubs; - numChannelsStubs = vector(numChannelTracks, numChannel); - } + numChannelStubs /= (setup_->numRegions() * (tracks ? numChannelTracks : 1)); + if (TB) + numChannelStubs = channelAssignment_->numChannelsStub(); bits.reserve(numChannelTracks + numChannelStubs); for (int region = 0; region < setup_->numRegions(); region++) { if (tracks) { const int offsetTracks = region * numChannelTracks; for (int channelTracks = 0; channelTracks < numChannelTracks; channelTracks++) { - const int offsetStubs = - region * numChannelStubs + - accumulate(numChannelsStubs.begin(), next(numChannelsStubs.begin(), channelTracks), 0); - convert(handleTracks->at(offsetTracks + channelTracks), bits); + int offsetStubs = (region * numChannelTracks + channelTracks) * numChannelStubs; + if (TB) { + numChannelStubs = + channelAssignment_->numProjectionLayers(channelTracks) + channelAssignment_->numSeedingLayers(); + offsetStubs = channelAssignment_->offsetStub(offsetTracks + channelTracks); + } + if (tracks) + convert(handleTracks->at(offsetTracks + channelTracks), bits); if (stubs) { - for (int channelStubs = 0; channelStubs < numChannelsStubs[channelTracks]; channelStubs++) + for (int channelStubs = 0; channelStubs < numChannelStubs; channelStubs++) convert(handleStubs->at(offsetStubs + channelStubs), bits); } } @@ -168,7 +178,7 @@ namespace trklet { void AnalyzerDemonstrator::endJob() { stringstream log; log << "Successrate: " << nEventsSuccessful_ << " / " << nEvents_ << " = " << nEventsSuccessful_ / (double)nEvents_; - LogPrint("L1Trigger/TrackerTFP") << log.str(); + LogPrint(moduleDescription().moduleName()) << log.str(); } } // namespace trklet diff --git a/L1Trigger/TrackFindingTracklet/test/AnalyzerKF.cc b/L1Trigger/TrackFindingTracklet/test/AnalyzerKF.cc new file mode 100644 index 0000000000000..6eeb2220a6ede --- /dev/null +++ b/L1Trigger/TrackFindingTracklet/test/AnalyzerKF.cc @@ -0,0 +1,413 @@ +#include "FWCore/Framework/interface/one/EDAnalyzer.h" +#include "FWCore/Framework/interface/Run.h" +#include "FWCore/Framework/interface/Event.h" +#include "FWCore/Framework/interface/EventSetup.h" +#include "FWCore/Framework/interface/MakerMacros.h" +#include "FWCore/ParameterSet/interface/ParameterSet.h" +#include "FWCore/ServiceRegistry/interface/Service.h" +#include "FWCore/MessageLogger/interface/MessageLogger.h" +#include "FWCore/Utilities/interface/EDGetToken.h" +#include "FWCore/Utilities/interface/InputTag.h" +#include "FWCore/Utilities/interface/Exception.h" +#include "CommonTools/UtilAlgos/interface/TFileService.h" +#include "DataFormats/Common/interface/Handle.h" + +#include "SimTracker/TrackTriggerAssociation/interface/StubAssociation.h" +#include "L1Trigger/TrackTrigger/interface/Setup.h" +#include "L1Trigger/TrackFindingTracklet/interface/DataFormats.h" + +#include +#include +#include + +#include +#include +#include +#include +#include +#include +#include + +using namespace std; +using namespace edm; +using namespace tt; + +namespace trklet { + + /*! \class trklet::AnalyzerKF + * \brief Class to analyze hardware like structured TTTrack Collection generated by Kalman Filter + * \author Thomas Schuh + * \date 2020, Sep + */ + class AnalyzerKF : public one::EDAnalyzer { + public: + AnalyzerKF(const ParameterSet& iConfig); + void beginJob() override {} + void beginRun(const Run& iEvent, const EventSetup& iSetup) override; + void analyze(const Event& iEvent, const EventSetup& iSetup) override; + void endRun(const Run& iEvent, const EventSetup& iSetup) override {} + void endJob() override; + + private: + // + void associate(const vector& tracks, + const vector>& stubs, + int region, + const StubAssociation* ass, + set& tps, + int& sum, + const vector& his, + const vector& prof, + bool perfect = true) const; + // + void analyzeTPz0(const StubAssociation* sa); + // ED input token of accepted Tracks + EDGetTokenT edGetTokenStubs_; + // ED input token of accepted Stubs + EDGetTokenT edGetTokenTracks_; + // ED input token for number of accepted States + EDGetTokenT edGetTokenNumStatesAccepted_; + // ED input token for number of lost States + EDGetTokenT edGetTokenNumStatesTruncated_; + // ED input token of TTStubRef to TPPtr association for tracking efficiency + EDGetTokenT edGetTokenSelection_; + // ED input token of TTStubRef to recontructable TPPtr association + EDGetTokenT edGetTokenReconstructable_; + // Setup token + ESGetToken esGetTokenSetup_; + // DataFormats token + ESGetToken esGetTokenDataFormats_; + // stores, calculates and provides run-time constants + const Setup* setup_ = nullptr; + // + const DataFormats* dataFormats_ = nullptr; + // enables analyze of TPs + bool useMCTruth_; + // + int nEvents_ = 0; + // + string label_; + + // Histograms + + TProfile* prof_; + TProfile* profChannel_; + TH1F* hisChannel_; + vector hisRes_; + vector profRes_; + TH1F* hisEffD0_; + TH1F* hisEffD0Total_; + TEfficiency* effD0_; + TH1F* hisEffEta_; + TH1F* hisEffEtaTotal_; + TEfficiency* effEta_; + TH1F* hisEffInv2R_; + TH1F* hisEffInv2RTotal_; + TEfficiency* effInv2R_; + TH1F* hisEffPT_; + TH1F* hisEffPTTotal_; + TEfficiency* effPT_; + TH1F* hisTracks_; + TH1F* hisLayers_; + TH1F* hisNumLayers_; + TProfile* profNumLayers_; + + // printout + stringstream log_; + }; + + AnalyzerKF::AnalyzerKF(const ParameterSet& iConfig) + : useMCTruth_(iConfig.getParameter("UseMCTruth")), nEvents_(0), hisRes_(9), profRes_(5) { + usesResource("TFileService"); + // book in- and output ED products + label_ = iConfig.getParameter("OutputLabelKF"); + const string& branchStubs = iConfig.getParameter("BranchStubs"); + const string& branchTracks = iConfig.getParameter("BranchTracks"); + const string& branchTruncated = iConfig.getParameter("BranchTruncated"); + edGetTokenStubs_ = consumes(InputTag(label_, branchStubs)); + edGetTokenTracks_ = consumes(InputTag(label_, branchTracks)); + edGetTokenNumStatesAccepted_ = consumes(InputTag(label_, branchTracks)); + edGetTokenNumStatesTruncated_ = consumes(InputTag(label_, branchTruncated)); + if (useMCTruth_) { + const auto& inputTagSelecttion = iConfig.getParameter("InputTagSelection"); + const auto& inputTagReconstructable = iConfig.getParameter("InputTagReconstructable"); + edGetTokenSelection_ = consumes(inputTagSelecttion); + edGetTokenReconstructable_ = consumes(inputTagReconstructable); + } + // book ES products + esGetTokenSetup_ = esConsumes(); + esGetTokenDataFormats_ = esConsumes(); + // log config + log_.setf(ios::fixed, ios::floatfield); + log_.precision(4); + } + + void AnalyzerKF::beginRun(const Run& iEvent, const EventSetup& iSetup) { + // helper class to store configurations + setup_ = &iSetup.getData(esGetTokenSetup_); + dataFormats_ = &iSetup.getData(esGetTokenDataFormats_); + // book histograms + Service fs; + TFileDirectory dir; + dir = fs->mkdir("KF"); + prof_ = dir.make("Counts", ";", 14, 0.5, 14.5); + prof_->GetXaxis()->SetBinLabel(1, "Stubs"); + prof_->GetXaxis()->SetBinLabel(2, "Tracks"); + prof_->GetXaxis()->SetBinLabel(4, "Matched Tracks"); + prof_->GetXaxis()->SetBinLabel(5, "All Tracks"); + prof_->GetXaxis()->SetBinLabel(6, "Found TPs"); + prof_->GetXaxis()->SetBinLabel(7, "Found selected TPs"); + prof_->GetXaxis()->SetBinLabel(9, "All TPs"); + prof_->GetXaxis()->SetBinLabel(10, "states"); + prof_->GetXaxis()->SetBinLabel(12, "max tp"); + prof_->GetXaxis()->SetBinLabel(13, "All electron TPs"); + prof_->GetXaxis()->SetBinLabel(14, "max electron tp"); + // channel occupancy + constexpr int maxOcc = 180; + const int numChannels = 1; + hisChannel_ = dir.make("His Channel Occupancy", ";", maxOcc, -.5, maxOcc - .5); + profChannel_ = dir.make("Prof Channel Occupancy", ";", numChannels, -.5, numChannels - .5); + // resoultions + static const vector names = {"phi0", "inv2R", "z0", "cot", "d0"}; + static const vector ranges = {.01, .004, 5., .4, 5.}; + for (int i = 0; i < 5; i++) { + hisRes_[i] = dir.make(("HisRes" + names[i]).c_str(), ";", 100, -ranges[i], ranges[i]); + profRes_[i] = dir.make(("ProfRes" + names[i]).c_str(), ";", 32, 0, 2.4); + } + for (int i = 5; i < 9; i++) { + const string name = (i < 7 ? names[2] : names[3]) + (i % 2 == 1 ? "plus" : "minus"); + const double range = (i < 7 ? ranges[2] : ranges[3]); + hisRes_[i] = dir.make(("HisRes" + name).c_str(), ";", 100, -range, range); + } + // Efficiencies + const double rangeD0 = 10.; + hisEffD0_ = dir.make("HisTPD0", ";", 32, -rangeD0 / 2., rangeD0 / 2.); + hisEffD0Total_ = dir.make("HisTPD0Total", ";", 32, -rangeD0 / 2., rangeD0 / 2.); + effD0_ = dir.make("EffD0", ";", 32, -rangeD0 / 2., rangeD0 / 2.); + hisEffEtaTotal_ = dir.make("HisTPEtaTotal", ";", 48, -2.4, 2.4); + hisEffEta_ = dir.make("HisTPEta", ";", 48, -2.4, 2.4); + effEta_ = dir.make("EffEta", ";", 48, -2.4, 2.4); + const double rangeInv2R = dataFormats_->format(Variable::inv2R, Process::dr).range(); + hisEffInv2R_ = dir.make("HisTPInv2R", ";", 32, -rangeInv2R / 2., rangeInv2R / 2.); + hisEffInv2RTotal_ = dir.make("HisTPInv2RTotal", ";", 32, -rangeInv2R / 2., rangeInv2R / 2.); + effInv2R_ = dir.make("EffInv2R", ";", 32, -rangeInv2R / 2., rangeInv2R / 2.); + hisEffPT_ = dir.make("HisTPPT", ";", 100, 0, 100); + hisEffPTTotal_ = dir.make("HisTPPTTotal", ";", 100, 0, 100); + effPT_ = dir.make("EffPT", ";", 100, 0, 100); + // tracks + hisTracks_ = dir.make("HisTracks", ";", 40, 0., 400); + // layers + hisLayers_ = dir.make("HisLayers", ";", 8, 0, 8); + hisNumLayers_ = dir.make("HisNumLayers", ";", 9, 0, 9); + profNumLayers_ = dir.make("Prof NumLayers", ";", 32, 0, 2.4); + } + + void AnalyzerKF::analyze(const Event& iEvent, const EventSetup& iSetup) { + static const int numLayers = setup_->numLayers(); + auto fill = [this](const TPPtr& tpPtr, TH1F* hisEta, TH1F* hisInv2R, TH1F* hisPT, TH1F* hisD0) { + hisEta->Fill(tpPtr->eta()); + hisInv2R->Fill(tpPtr->charge() / tpPtr->pt() * setup_->invPtToDphi()); + hisPT->Fill(tpPtr->pt()); + hisD0->Fill(tpPtr->d0()); + }; + // read in kf products + Handle handleStubs; + iEvent.getByToken(edGetTokenStubs_, handleStubs); + const StreamsStub& allStubs = *handleStubs; + Handle handleTracks; + iEvent.getByToken(edGetTokenTracks_, handleTracks); + const StreamsTrack& allTracks = *handleTracks; + Handle handleNumStatesAccepted; + iEvent.getByToken(edGetTokenNumStatesAccepted_, handleNumStatesAccepted); + Handle handleNumStatesTruncated; + iEvent.getByToken(edGetTokenNumStatesTruncated_, handleNumStatesTruncated); + // read in MCTruth + const StubAssociation* selection = nullptr; + const StubAssociation* reconstructable = nullptr; + if (useMCTruth_) { + Handle handleSelection; + iEvent.getByToken(edGetTokenSelection_, handleSelection); + selection = handleSelection.product(); + prof_->Fill(9, selection->numTPs()); + Handle handleReconstructable; + iEvent.getByToken(edGetTokenReconstructable_, handleReconstructable); + reconstructable = handleReconstructable.product(); + for (const auto& p : selection->getTrackingParticleToTTStubsMap()) + fill(p.first, hisEffEtaTotal_, hisEffInv2RTotal_, hisEffPTTotal_, hisEffD0Total_); + } + // analyze kf products and associate found tracks with reconstrucable TrackingParticles + set tpPtrs; + set tpPtrsSelection; + set tpPtrsMax; + int numMatched(0); + int numTracks(0); + for (int region = 0; region < setup_->numRegions(); region++) { + int nRegionStubs(0); + int nRegionTracks(0); + const int offset = region * numLayers; + const StreamTrack& channelTracks = allTracks[region]; + hisChannel_->Fill(channelTracks.size()); + profChannel_->Fill(1, channelTracks.size()); + vector tracks; + vector stubs; + vector> tracksStubs(channelTracks.size(), vector(numLayers, nullptr)); + tracks.reserve(channelTracks.size()); + stubs.reserve(channelTracks.size() * numLayers); + for (int frame = 0; frame < (int)channelTracks.size(); frame++) { + tracks.emplace_back(channelTracks[frame], dataFormats_); + const double eta = abs(asinh(tracks.back().cot())); + int nLayers(0); + for (int layer = 0; layer < numLayers; layer++) { + const FrameStub& fs = allStubs[offset + layer][frame]; + if (fs.first.isNull()) + continue; + stubs.emplace_back(fs, dataFormats_); + tracksStubs[frame][layer] = &stubs.back(); + hisLayers_->Fill(layer); + nLayers++; + } + hisNumLayers_->Fill(nLayers); + profNumLayers_->Fill(eta, nLayers); + } + nRegionStubs += stubs.size(); + nRegionTracks += tracks.size(); + if (!useMCTruth_) + continue; + int tmp(0); + associate(tracks, tracksStubs, region, selection, tpPtrsSelection, tmp, hisRes_, profRes_); + associate(tracks, tracksStubs, region, reconstructable, tpPtrs, numMatched, vector(), vector()); + associate(tracks, tracksStubs, region, selection, tpPtrsMax, tmp, vector(), vector(), false); + numTracks += nRegionTracks; + prof_->Fill(1, nRegionStubs); + prof_->Fill(2, nRegionTracks); + } + for (const TPPtr& tpPtr : tpPtrsSelection) + fill(tpPtr, hisEffEta_, hisEffInv2R_, hisEffPT_, hisEffD0_); + prof_->Fill(4, numMatched); + prof_->Fill(5, numTracks); + prof_->Fill(6, tpPtrs.size()); + prof_->Fill(7, tpPtrsSelection.size()); + prof_->Fill(10, *handleNumStatesAccepted); + prof_->Fill(11, *handleNumStatesTruncated); + prof_->Fill(12, tpPtrsMax.size()); + hisTracks_->Fill(numTracks); + nEvents_++; + } + + void AnalyzerKF::endJob() { + if (nEvents_ == 0) + return; + // effi + effD0_->SetPassedHistogram(*hisEffD0_, "f"); + effD0_->SetTotalHistogram(*hisEffD0Total_, "f"); + effEta_->SetPassedHistogram(*hisEffEta_, "f"); + effEta_->SetTotalHistogram(*hisEffEtaTotal_, "f"); + effInv2R_->SetPassedHistogram(*hisEffInv2R_, "f"); + effInv2R_->SetTotalHistogram(*hisEffInv2RTotal_, "f"); + effPT_->SetPassedHistogram(*hisEffPT_, "f"); + effPT_->SetTotalHistogram(*hisEffPTTotal_, "f"); + // printout SF summary + const double totalTPs = prof_->GetBinContent(9); + const double numStubs = prof_->GetBinContent(1); + const double numTracks = prof_->GetBinContent(2); + const double totalTracks = prof_->GetBinContent(5); + const double numTracksMatched = prof_->GetBinContent(4); + const double numTPsAll = prof_->GetBinContent(6); + const double numTPsEff = prof_->GetBinContent(7); + const double numTPsEffMax = prof_->GetBinContent(12); + const double errStubs = prof_->GetBinError(1); + const double errTracks = prof_->GetBinError(2); + const double fracFake = (totalTracks - numTracksMatched) / totalTracks; + const double fracDup = (numTracksMatched - numTPsAll) / totalTracks; + const double eff = numTPsEff / totalTPs; + const double errEff = sqrt(eff * (1. - eff) / totalTPs / nEvents_); + const double effMax = numTPsEffMax / totalTPs; + const double errEffMax = sqrt(effMax * (1. - effMax) / totalTPs / nEvents_); + const int numStates = prof_->GetBinContent(10); + const int numStatesLost = prof_->GetBinContent(11); + const double fracSatest = numStates / (double)(numStates + numStatesLost); + const vector nums = {numStubs, numTracks}; + const vector errs = {errStubs, errTracks}; + const int wNums = ceil(log10(*max_element(nums.begin(), nums.end()))) + 5; + const int wErrs = ceil(log10(*max_element(errs.begin(), errs.end()))) + 5; + log_ << " KF SUMMARY " << endl; + log_ << "number of stubs per TFP = " << setw(wNums) << numStubs << " +- " << setw(wErrs) << errStubs << endl; + log_ << "number of tracks per TFP = " << setw(wNums) << numTracks << " +- " << setw(wErrs) << errTracks + << endl; + log_ << " tracking efficiency = " << setw(wNums) << eff << " +- " << setw(wErrs) << errEff << endl; + log_ << " max tracking efficiency = " << setw(wNums) << effMax << " +- " << setw(wErrs) << errEffMax << endl; + log_ << " fake rate = " << setw(wNums) << fracFake << endl; + log_ << " duplicate rate = " << setw(wNums) << fracDup << endl; + log_ << " state assessment fraction = " << setw(wNums) << fracSatest << endl; + log_ << " number of states per TFP = " << setw(wNums) << (numStates + numStatesLost) / setup_->numRegions() + << endl; + log_ << "============================================================="; + LogPrint(moduleDescription().moduleName()) << log_.str(); + } + + // + void AnalyzerKF::associate(const vector& tracks, + const vector>& tracksStubs, + int region, + const StubAssociation* ass, + set& tps, + int& sum, + const vector& his, + const vector& prof, + bool perfect) const { + for (int frame = 0; frame < (int)tracks.size(); frame++) { + const TrackKF& track = tracks[frame]; + const vector& stubs = tracksStubs[frame]; + vector ttStubRefs; + ttStubRefs.reserve(stubs.size()); + TTBV hitPattern(0, setup_->numLayers()); + int layer(-1); + for (StubKF* stub : stubs) { + layer++; + if (!stub) + continue; + hitPattern.set(layer); + ttStubRefs.push_back(stub->frame().first); + } + const vector& tpPtrs = perfect ? ass->associateFinal(ttStubRefs) : ass->associate(ttStubRefs); + if (tpPtrs.empty()) + continue; + sum++; + copy(tpPtrs.begin(), tpPtrs.end(), inserter(tps, tps.begin())); + if (his.empty()) + continue; + const double cot = track.cot(); + const double z0 = track.zT() - setup_->chosenRofZ() * cot; + const double inv2R = track.inv2R(); + const double phi0 = deltaPhi(track.phiT() - setup_->chosenRofPhi() * inv2R + + region * dataFormats_->format(Variable::phiT, Process::kf).range()); + for (const TPPtr& tpPtr : tpPtrs) { + const double tpPhi0 = tpPtr->phi(); + const double tpCot = sinh(tpPtr->eta()); + const double tpInv2R = -setup_->invPtToDphi() * tpPtr->charge() / tpPtr->pt(); + const math::XYZPointD& v = tpPtr->vertex(); + const double tpZ0 = v.z() - tpCot * (v.x() * cos(tpPhi0) + v.y() * sin(tpPhi0)); + const double dCot = tpCot - cot; + const double dZ0 = tpZ0 - z0; + const double dInv2R = tpInv2R - inv2R; + const double dPhi0 = deltaPhi(tpPhi0 - phi0); + const double dD0 = tpPtr->d0() + track.frame().first->d0(); + const vector ds = {dPhi0, dInv2R / setup_->invPtToDphi(), dZ0, dCot, dD0}; + for (int i = 0; i < (int)ds.size(); i++) { + his[i]->Fill(ds[i]); + prof[i]->Fill(abs(tpPtr->eta()), abs(ds[i])); + } + if (tpCot < 0) { + his[6]->Fill(dZ0); + his[8]->Fill(dCot); + } else { + his[5]->Fill(dZ0); + his[7]->Fill(dCot); + } + } + } + } + +} // namespace trklet + +DEFINE_FWK_MODULE(trklet::AnalyzerKF); diff --git a/L1Trigger/TrackFindingTracklet/test/AnalyzerKFout.cc b/L1Trigger/TrackFindingTracklet/test/AnalyzerKFout.cc index 1418a03e4c409..b4c625c9d8074 100644 --- a/L1Trigger/TrackFindingTracklet/test/AnalyzerKFout.cc +++ b/L1Trigger/TrackFindingTracklet/test/AnalyzerKFout.cc @@ -86,8 +86,8 @@ namespace trklet { usesResource("TFileService"); // book in- and output ED products const string& label = iConfig.getParameter("LabelKFout"); - const string& branchAccepted = iConfig.getParameter("BranchAcceptedTracks"); - const string& branchLost = iConfig.getParameter("BranchLostTracks"); + const string& branchAccepted = iConfig.getParameter("BranchTracksAccepted"); + const string& branchLost = iConfig.getParameter("BranchTracksTruncated"); edGetTokenAccepted_ = consumes(InputTag(label, branchAccepted)); edGetTokenLost_ = consumes(InputTag(label, branchLost)); if (useMCTruth_) { @@ -231,7 +231,7 @@ namespace trklet { log_ << " fake rate = " << setw(wNums) << fracFake << endl; log_ << " duplicate rate = " << setw(wNums) << fracDup << endl; log_ << "============================================================="; - LogPrint("L1Trigger/TrackerTFP") << log_.str(); + LogPrint(moduleDescription().moduleName()) << log_.str(); } // diff --git a/L1Trigger/TrackFindingTracklet/test/AnalyzerTBout.cc b/L1Trigger/TrackFindingTracklet/test/AnalyzerTBout.cc deleted file mode 100644 index d4a89e1a96100..0000000000000 --- a/L1Trigger/TrackFindingTracklet/test/AnalyzerTBout.cc +++ /dev/null @@ -1,429 +0,0 @@ -#include "FWCore/Framework/interface/one/EDAnalyzer.h" -#include "FWCore/Framework/interface/Run.h" -#include "FWCore/Framework/interface/Event.h" -#include "FWCore/Framework/interface/EventSetup.h" -#include "FWCore/Framework/interface/MakerMacros.h" -#include "FWCore/ParameterSet/interface/ParameterSet.h" -#include "FWCore/ServiceRegistry/interface/Service.h" -#include "FWCore/MessageLogger/interface/MessageLogger.h" -#include "FWCore/Utilities/interface/EDGetToken.h" -#include "FWCore/Utilities/interface/InputTag.h" -#include "FWCore/Utilities/interface/Exception.h" -#include "CommonTools/UtilAlgos/interface/TFileService.h" -#include "DataFormats/Common/interface/Handle.h" - -#include "SimTracker/TrackTriggerAssociation/interface/StubAssociation.h" -#include "L1Trigger/TrackTrigger/interface/Setup.h" -#include "L1Trigger/TrackerTFP/interface/DataFormats.h" -#include "L1Trigger/TrackFindingTracklet/interface/ChannelAssignment.h" -#include "L1Trigger/TrackFindingTracklet/interface/Settings.h" - -#include -#include -#include -#include - -#include -#include -#include -#include -#include -#include - -using namespace std; -using namespace edm; -using namespace trackerTFP; -using namespace tt; - -namespace trklet { - - // stub resolution plots helper - enum Resolution { Phi, Z, NumResolution }; - constexpr initializer_list AllResolution = {Phi, Z}; - constexpr auto NameResolution = {"Phi", "Z"}; - inline string name(Resolution r) { return string(*(NameResolution.begin() + r)); } - - /*! \class trklet::AnalyzerTBout - * \brief Class to analyze hardware like structured TTStub Collection generated by Tracklet - * \author Thomas Schuh - * \date 2021, Nov - */ - class AnalyzerTBout : public one::EDAnalyzer { - public: - AnalyzerTBout(const ParameterSet& iConfig); - void beginJob() override {} - void beginRun(const Run& iEvent, const EventSetup& iSetup) override; - void analyze(const Event& iEvent, const EventSetup& iSetup) override; - void endRun(const Run& iEvent, const EventSetup& iSetup) override {} - void endJob() override; - - private: - // - void formTracks(const StreamsTrack& streamsTrack, - const StreamsStub& streamsStubs, - vector>& tracks, - int channel); - // - void associate(const vector>& tracks, const StubAssociation* ass, set& tps, int& sum) const; - // - void fill(const FrameTrack& frameTrack, const FrameStub& frameStub); - - // ED input token of dtc stubs - EDGetTokenT edGetTokenTTDTC_; - // ED input token of stubs - EDGetTokenT edGetTokenAcceptedStubs_; - // ED input token of tracks - EDGetTokenT edGetTokenAcceptedTracks_; - // ED input token of lost stubs - EDGetTokenT edGetTokenLostStubs_; - // ED input token of lost tracks - EDGetTokenT edGetTokenLostTracks_; - // ED input token of TTStubRef to TPPtr association for tracking efficiency - EDGetTokenT edGetTokenSelection_; - // ED input token of TTStubRef to recontructable TPPtr association - EDGetTokenT edGetTokenReconstructable_; - // Setup token - ESGetToken esGetTokenSetup_; - // DataFormats token - ESGetToken esGetTokenDataFormats_; - // ChannelAssignment token - ESGetToken esGetTokenChannelAssignment_; - // stores, calculates and provides run-time constants - const Setup* setup_ = nullptr; - // - const Settings settings_; - // helper class to extract structured data from tt::Frames - const DataFormats* dataFormats_ = nullptr; - // helper class to assign tracklet track to channel - const ChannelAssignment* channelAssignment_ = nullptr; - // enables analyze of TPs - bool useMCTruth_; - // - int nEvents_ = 0; - // - vector> regionStubs_; - // - int region_; - - // Histograms - - TProfile* prof_; - TProfile* profChannel_; - TH1F* hisChannel_; - vector hisResolution_; - vector profResolution_; - vector hisResolutionMe_; - vector hisResolutionThey_; - vector his2Resolution_; - - // printout - stringstream log_; - }; - - AnalyzerTBout::AnalyzerTBout(const ParameterSet& iConfig) : useMCTruth_(iConfig.getParameter("UseMCTruth")) { - usesResource("TFileService"); - // book in- and output ED products - const InputTag& inputTag = iConfig.getParameter("InputTagDTC"); - const string& label = iConfig.getParameter("LabelTBout"); - const string& branchAcceptedStubs = iConfig.getParameter("BranchAcceptedStubs"); - const string& branchAcceptedTracks = iConfig.getParameter("BranchAcceptedTracks"); - const string& branchLostStubs = iConfig.getParameter("BranchLostStubs"); - const string& branchLostTracks = iConfig.getParameter("BranchLostTracks"); - edGetTokenTTDTC_ = consumes(inputTag); - edGetTokenAcceptedStubs_ = consumes(InputTag(label, branchAcceptedStubs)); - edGetTokenAcceptedTracks_ = consumes(InputTag(label, branchAcceptedTracks)); - edGetTokenLostStubs_ = consumes(InputTag(label, branchLostStubs)); - edGetTokenLostTracks_ = consumes(InputTag(label, branchLostTracks)); - if (useMCTruth_) { - const auto& inputTagSelecttion = iConfig.getParameter("InputTagSelection"); - const auto& inputTagReconstructable = iConfig.getParameter("InputTagReconstructable"); - edGetTokenSelection_ = consumes(inputTagSelecttion); - edGetTokenReconstructable_ = consumes(inputTagReconstructable); - } - // book ES products - esGetTokenSetup_ = esConsumes(); - esGetTokenDataFormats_ = esConsumes(); - esGetTokenChannelAssignment_ = esConsumes(); - // log config - log_.setf(ios::fixed, ios::floatfield); - log_.precision(4); - } - - void AnalyzerTBout::beginRun(const Run& iEvent, const EventSetup& iSetup) { - // helper class to store configurations - setup_ = &iSetup.getData(esGetTokenSetup_); - // helper class to extract structured data from tt::Frames - dataFormats_ = &iSetup.getData(esGetTokenDataFormats_); - // helper class to assign tracklet track to channel - channelAssignment_ = &iSetup.getData(esGetTokenChannelAssignment_); - // book histograms - Service fs; - TFileDirectory dir; - dir = fs->mkdir("TBout"); - prof_ = dir.make("Counts", ";", 9, 0.5, 9.5); - prof_->GetXaxis()->SetBinLabel(1, "Stubs"); - prof_->GetXaxis()->SetBinLabel(2, "Tracks"); - prof_->GetXaxis()->SetBinLabel(3, "Lost Tracks"); - prof_->GetXaxis()->SetBinLabel(4, "Matched Tracks"); - prof_->GetXaxis()->SetBinLabel(5, "All Tracks"); - prof_->GetXaxis()->SetBinLabel(6, "Found TPs"); - prof_->GetXaxis()->SetBinLabel(7, "Found selected TPs"); - prof_->GetXaxis()->SetBinLabel(8, "Lost TPs"); - prof_->GetXaxis()->SetBinLabel(9, "All TPs"); - // channel occupancy - constexpr int maxOcc = 180; - const int numChannels = channelAssignment_->numChannelsStub() * setup_->numRegions(); - hisChannel_ = dir.make("His Channel Occupancy", ";", maxOcc, -.5, maxOcc - .5); - profChannel_ = dir.make("Prof Channel Occupancy", ";", numChannels, -.5, numChannels - .5); - // stub parameter resolutions - constexpr int bins = 400; - constexpr int binsHis = 100; - constexpr double maxZ = 300.; - constexpr double maxR = 120.; - constexpr array ranges{{.01, 5.}}; - hisResolution_.reserve(NumResolution); - profResolution_.reserve(NumResolution); - for (Resolution r : AllResolution) { - hisResolution_.emplace_back(dir.make(("HisRes" + name(r)).c_str(), ";", binsHis, -ranges[r], ranges[r])); - profResolution_.emplace_back( - dir.make(("ProfRes" + name(r)).c_str(), ";;", bins, -maxZ, maxZ, bins, 0., maxR)); - hisResolutionMe_.emplace_back( - dir.make(("HisResMe" + name(r)).c_str(), ";", binsHis, -ranges[r], ranges[r])); - hisResolutionThey_.emplace_back( - dir.make(("HisResThey" + name(r)).c_str(), ";", binsHis, -ranges[r], ranges[r])); - his2Resolution_.emplace_back( - dir.make(("His2" + name(r)).c_str(), ";;", bins, -ranges[r], ranges[r], bins, -ranges[r], ranges[r])); - } - regionStubs_ = vector>(setup_->numRegions()); - } - - void AnalyzerTBout::analyze(const Event& iEvent, const EventSetup& iSetup) { - // read in TTDTC - Handle handleTTDTC; - iEvent.getByToken(edGetTokenTTDTC_, handleTTDTC); - const TTDTC& ttDTC = *handleTTDTC; - for (deque& region : regionStubs_) - region.clear(); - for (int r : ttDTC.tfpRegions()) { - for (int c : ttDTC.tfpChannels()) { - const StreamStub& s = ttDTC.stream(r, c); - copy_if( - s.begin(), s.end(), back_inserter(regionStubs_[r]), [](const FrameStub& f) { return f.first.isNonnull(); }); - } - } - // read in TBout products - Handle handleAcceptedStubs; - iEvent.getByToken(edGetTokenAcceptedStubs_, handleAcceptedStubs); - const StreamsStub& acceptedStubs = *handleAcceptedStubs; - Handle handleAcceptedTracks; - iEvent.getByToken(edGetTokenAcceptedTracks_, handleAcceptedTracks); - const StreamsTrack& acceptedTracks = *handleAcceptedTracks; - Handle handleLostStubs; - iEvent.getByToken(edGetTokenLostStubs_, handleLostStubs); - const StreamsStub& lostStubs = *handleLostStubs; - Handle handleLostTracks; - iEvent.getByToken(edGetTokenLostTracks_, handleLostTracks); - const StreamsTrack& lostTracks = *handleLostTracks; - // read in MCTruth - const StubAssociation* selection = nullptr; - const StubAssociation* reconstructable = nullptr; - if (useMCTruth_) { - Handle handleSelection; - iEvent.getByToken(edGetTokenSelection_, handleSelection); - selection = handleSelection.product(); - prof_->Fill(9, selection->numTPs()); - Handle handleReconstructable; - iEvent.getByToken(edGetTokenReconstructable_, handleReconstructable); - reconstructable = handleReconstructable.product(); - } - // analyze ht products and associate found tracks with reconstrucable TrackingParticles - set tpPtrs; - set tpPtrsSelection; - set tpPtrsLost; - int allMatched(0); - int allTracks(0); - for (region_ = 0; region_ < setup_->numRegions(); region_++) { - const int offset = region_ * channelAssignment_->numChannelsTrack(); - int nStubs(0); - int nTracks(0); - int nLost(0); - for (int channel = 0; channel < channelAssignment_->numChannelsTrack(); channel++) { - vector> tracks; - formTracks(acceptedTracks, acceptedStubs, tracks, offset + channel); - vector> lost; - formTracks(lostTracks, lostStubs, lost, offset + channel); - nTracks += tracks.size(); - nStubs += - accumulate(tracks.begin(), tracks.end(), 0, [](int sum, const auto& v) { return sum + (int)v.size(); }); - nLost += lost.size(); - allTracks += tracks.size(); - if (!useMCTruth_) - continue; - int tmp(0); - associate(tracks, selection, tpPtrsSelection, tmp); - associate(lost, selection, tpPtrsLost, tmp); - associate(tracks, reconstructable, tpPtrs, allMatched); - } - prof_->Fill(1, nStubs); - prof_->Fill(2, nTracks); - prof_->Fill(3, nLost); - } - vector recovered; - recovered.reserve(tpPtrsLost.size()); - set_intersection(tpPtrsLost.begin(), tpPtrsLost.end(), tpPtrs.begin(), tpPtrs.end(), back_inserter(recovered)); - for (const TPPtr& tpPtr : recovered) - tpPtrsLost.erase(tpPtr); - prof_->Fill(4, allMatched); - prof_->Fill(5, allTracks); - prof_->Fill(6, tpPtrs.size()); - prof_->Fill(7, tpPtrsSelection.size()); - prof_->Fill(8, tpPtrsLost.size()); - nEvents_++; - } - - void AnalyzerTBout::endJob() { - if (nEvents_ == 0) - return; - // printout TBout summary - const double totalTPs = prof_->GetBinContent(9); - const double numStubs = prof_->GetBinContent(1); - const double numTracks = prof_->GetBinContent(2); - const double numTracksLost = prof_->GetBinContent(3); - const double totalTracks = prof_->GetBinContent(5); - const double numTracksMatched = prof_->GetBinContent(4); - const double numTPsAll = prof_->GetBinContent(6); - const double numTPsEff = prof_->GetBinContent(7); - const double numTPsLost = prof_->GetBinContent(8); - const double errStubs = prof_->GetBinError(1); - const double errTracks = prof_->GetBinError(2); - const double errTracksLost = prof_->GetBinError(3); - const double fracFake = (totalTracks - numTracksMatched) / totalTracks; - const double fracDup = (numTracksMatched - numTPsAll) / totalTracks; - const double eff = numTPsEff / totalTPs; - const double errEff = sqrt(eff * (1. - eff) / totalTPs / nEvents_); - const double effLoss = numTPsLost / totalTPs; - const double errEffLoss = sqrt(effLoss * (1. - effLoss) / totalTPs / nEvents_); - const vector nums = {numStubs, numTracks, numTracksLost}; - const vector errs = {errStubs, errTracks, errTracksLost}; - const int wNums = ceil(log10(*max_element(nums.begin(), nums.end()))) + 5; - const int wErrs = ceil(log10(*max_element(errs.begin(), errs.end()))) + 5; - log_ << " TBout SUMMARY " << endl; - log_ << "number of stubs per TFP = " << setw(wNums) << numStubs << " +- " << setw(wErrs) << errStubs << endl; - log_ << "number of tracks per TFP = " << setw(wNums) << numTracks << " +- " << setw(wErrs) << errTracks - << endl; - log_ << "number of lost tracks per TFP = " << setw(wNums) << numTracksLost << " +- " << setw(wErrs) << errTracksLost - << endl; - log_ << " max tracking efficiency = " << setw(wNums) << eff << " +- " << setw(wErrs) << errEff << endl; - log_ << " lost tracking efficiency = " << setw(wNums) << effLoss << " +- " << setw(wErrs) << errEffLoss << endl; - log_ << " fake rate = " << setw(wNums) << fracFake << endl; - log_ << " duplicate rate = " << setw(wNums) << fracDup << endl; - log_ << "============================================================="; - LogPrint("L1Trigger/TrackerTFP") << log_.str(); - } - - // - void AnalyzerTBout::formTracks(const StreamsTrack& streamsTrack, - const StreamsStub& streamsStubs, - vector>& tracks, - int channel) { - const int seedType = channel % channelAssignment_->numChannelsTrack(); - const int offset = channelAssignment_->offsetStub(channel); - const StreamTrack& streamTrack = streamsTrack[channel]; - const int numTracks = accumulate(streamTrack.begin(), streamTrack.end(), 0, [](int sum, const FrameTrack& frame) { - return sum + (frame.first.isNonnull() ? 1 : 0); - }); - tracks.reserve(numTracks); - for (int frame = 0; frame < (int)streamTrack.size(); frame++) { - const FrameTrack& frameTrack = streamTrack[frame]; - if (frameTrack.first.isNull()) - continue; - vector ttStubRefs; - const int numProjectionLayers = channelAssignment_->numProjectionLayers(seedType); - const int numSeedingLayers = channelAssignment_->seedingLayers(seedType).size(); - ttStubRefs.reserve(numProjectionLayers + numSeedingLayers); - for (int channel = 0; channel < numProjectionLayers + numSeedingLayers; channel++) { - const FrameStub& stub = streamsStubs[offset + channel][frame]; - if (stub.first.isNull()) - continue; - if (channel < numProjectionLayers) - this->fill(frameTrack, stub); - ttStubRefs.push_back(stub.first); - } - tracks.push_back(ttStubRefs); - } - } - - // - void AnalyzerTBout::associate(const vector>& tracks, - const StubAssociation* ass, - set& tps, - int& sum) const { - for (const vector& ttStubRefs : tracks) { - const vector& tpPtrs = ass->associate(ttStubRefs); - if (tpPtrs.empty()) - continue; - sum++; - copy(tpPtrs.begin(), tpPtrs.end(), inserter(tps, tps.begin())); - } - } - - // - void AnalyzerTBout::fill(const FrameTrack& frameTrack, const FrameStub& frameStub) { - // get dtc stub frame - const deque& region = regionStubs_[region_]; - const auto it = - find_if(region.begin(), region.end(), [&frameStub](const FrameStub& f) { return f.first == frameStub.first; }); - if (it == region.end()) - throw cms::Exception("LgociError.") << "Stub on track was not in DTC collection."; - const GlobalPoint ttPos = setup_->stubPos(frameStub.first); - const GlobalPoint pos = setup_->stubPos(true, *it, region_); - static constexpr int widthPhi = 12; - static constexpr int widthZ = 9; - static constexpr int widthR = 7; - const bool barrel = setup_->barrel(frameStub.first); - const int layerIdTracklet = setup_->trackletLayerId(frameStub.first); - static const double baseR = settings_.kz(); - const double basePhi = barrel ? settings_.kphi1() : settings_.kphi(layerIdTracklet); - const double baseZ = settings_.kz(layerIdTracklet); - static const double baseInvR = settings_.kphi1() / settings_.kr() * pow(2, settings_.rinv_shift()); - static const double basePhi0 = settings_.kphi1() * pow(2, settings_.phi0_shift()); - static const double baseZ0 = settings_.kz() * pow(2, settings_.z0_shift()); - static const double baseTanL = settings_.kz() / settings_.kr() * pow(2, settings_.t_shift()); - const int widthRZ = barrel ? widthZ : widthR; - const double baseRZ = barrel ? baseZ : baseR; - // calc residuals - const double rInv = (frameTrack.first->rInv() / baseInvR + .5) * baseInvR; - const double phi0 = (frameTrack.first->phi() / basePhi0 + .5) * basePhi0; - const double z0 = (frameTrack.first->z0() / baseZ0 + .5) * baseZ0; - const double tanL = (frameTrack.first->tanL() / baseTanL + .5) * baseTanL; - const double phi = deltaPhi(pos.phi() - (phi0 - rInv * pos.perp() / 2.)); - const double r = pos.perp() - (pos.z() - z0) / tanL; - const double z = pos.z() - (z0 + tanL * pos.perp()); - const double rz = barrel ? z : r; - const int phii = floor(phi / basePhi); - const int rzi = floor(rz / baseRZ); - const double phid = (phii + .5) * basePhi; - const double rzd = (rzi + .5) * baseRZ; - // parse residuals - TTBV hw(frameStub.second); - const TTBV hwRZ(hw, widthRZ, 0, true); - hw >>= widthRZ; - const TTBV hwPhi(hw, widthPhi, 0, true); - const double hwPhid = hwPhi.val(basePhi); - const double hwRZd = hwRZ.val(baseRZ); - const vector resolutions = {phid - hwPhid, rzd - hwRZd}; - for (Resolution r : AllResolution) { - hisResolution_[r]->Fill(resolutions[r]); - profResolution_[r]->Fill(ttPos.z(), ttPos.perp(), abs(resolutions[r])); - } - hisResolutionMe_[0]->Fill(phid); - hisResolutionMe_[1]->Fill(rzd); - hisResolutionThey_[0]->Fill(hwPhid); - hisResolutionThey_[1]->Fill(hwRZd); - his2Resolution_[0]->Fill(phid, hwPhid); - his2Resolution_[1]->Fill(rzd, hwRZd); - } - -} // namespace trklet - -DEFINE_FWK_MODULE(trklet::AnalyzerTBout); diff --git a/L1Trigger/TrackerTFP/test/AnalyzerMHT.cc b/L1Trigger/TrackFindingTracklet/test/AnalyzerTFP.cc similarity index 51% rename from L1Trigger/TrackerTFP/test/AnalyzerMHT.cc rename to L1Trigger/TrackFindingTracklet/test/AnalyzerTFP.cc index 530e7f59bcc16..2e79e64bb6fae 100644 --- a/L1Trigger/TrackerTFP/test/AnalyzerMHT.cc +++ b/L1Trigger/TrackFindingTracklet/test/AnalyzerTFP.cc @@ -14,7 +14,6 @@ #include "SimTracker/TrackTriggerAssociation/interface/StubAssociation.h" #include "L1Trigger/TrackTrigger/interface/Setup.h" -#include "L1Trigger/TrackerTFP/interface/DataFormats.h" #include #include @@ -31,16 +30,16 @@ using namespace std; using namespace edm; using namespace tt; -namespace trackerTFP { +namespace trklet { - /*! \class trackerTFP::AnalyzerMHT - * \brief Class to analyze hardware like structured TTStub Collection generated by Mini Hough Transform + /*! \class trklet::AnalyzerTFP + * \brief Class to analyze TTTracks found by tfp * \author Thomas Schuh - * \date 2020, Apr + * \date 20204 Aug */ - class AnalyzerMHT : public one::EDAnalyzer { + class AnalyzerTFP : public one::EDAnalyzer { public: - AnalyzerMHT(const ParameterSet& iConfig); + AnalyzerTFP(const ParameterSet& iConfig); void beginJob() override {} void beginRun(const Run& iEvent, const EventSetup& iSetup) override; void analyze(const Event& iEvent, const EventSetup& iSetup) override; @@ -48,27 +47,23 @@ namespace trackerTFP { void endJob() override; private: - // - void formTracks(const StreamStub& stream, vector>& tracks) const; - // - void associate(const vector>& tracks, const StubAssociation* ass, set& tps, int& sum) const; + // gets all TPs associated too any of the tracks & number of tracks matching at least one TP + void associate(const vector>& tracks, + const StubAssociation* ass, + set& tps, + int& nMatchTrk, + bool perfect = false) const; - // ED input token of stubs - EDGetTokenT edGetTokenAccepted_; - // ED input token of lost stubs - EDGetTokenT edGetTokenLost_; + // ED input token of tracks + EDGetTokenT edGetToken_; // ED input token of TTStubRef to TPPtr association for tracking efficiency EDGetTokenT edGetTokenSelection_; // ED input token of TTStubRef to recontructable TPPtr association EDGetTokenT edGetTokenReconstructable_; // Setup token ESGetToken esGetTokenSetup_; - // DataFormats token - ESGetToken esGetTokenDataFormats_; // stores, calculates and provides run-time constants const Setup* setup_ = nullptr; - // helper class to extract structured data from tt::Frames - const DataFormats* dataFormats_ = nullptr; // enables analyze of TPs bool useMCTruth_; // @@ -76,7 +71,9 @@ namespace trackerTFP { // Histograms + // counts per TFP (processing nonant and event) TProfile* prof_; + // no. of tracks per nonant TProfile* profChannel_; TH1F* hisChannel_; TH1F* hisEff_; @@ -87,14 +84,12 @@ namespace trackerTFP { stringstream log_; }; - AnalyzerMHT::AnalyzerMHT(const ParameterSet& iConfig) : useMCTruth_(iConfig.getParameter("UseMCTruth")) { + AnalyzerTFP::AnalyzerTFP(const ParameterSet& iConfig) : useMCTruth_(iConfig.getParameter("UseMCTruth")) { usesResource("TFileService"); // book in- and output ED products - const string& label = iConfig.getParameter("LabelMHT"); - const string& branchAccepted = iConfig.getParameter("BranchAcceptedStubs"); - const string& branchLost = iConfig.getParameter("BranchLostStubs"); - edGetTokenAccepted_ = consumes(InputTag(label, branchAccepted)); - edGetTokenLost_ = consumes(InputTag(label, branchLost)); + const string& label = iConfig.getParameter("OutputLabelTFP"); + const string& branch = iConfig.getParameter("BranchTTTracks"); + edGetToken_ = consumes(InputTag(label, branch)); if (useMCTruth_) { const auto& inputTagSelecttion = iConfig.getParameter("InputTagSelection"); const auto& inputTagReconstructable = iConfig.getParameter("InputTagReconstructable"); @@ -103,22 +98,19 @@ namespace trackerTFP { } // book ES products esGetTokenSetup_ = esConsumes(); - esGetTokenDataFormats_ = esConsumes(); // log config log_.setf(ios::fixed, ios::floatfield); log_.precision(4); } - void AnalyzerMHT::beginRun(const Run& iEvent, const EventSetup& iSetup) { + void AnalyzerTFP::beginRun(const Run& iEvent, const EventSetup& iSetup) { // helper class to store configurations setup_ = &iSetup.getData(esGetTokenSetup_); - // helper class to extract structured data from tt::Frames - dataFormats_ = &iSetup.getData(esGetTokenDataFormats_); // book histograms Service fs; TFileDirectory dir; - dir = fs->mkdir("MHT"); - prof_ = dir.make("Counts", ";", 9, 0.5, 9.5); + dir = fs->mkdir("TFP"); + prof_ = dir.make("Counts", ";", 10, 0.5, 10.5); prof_->GetXaxis()->SetBinLabel(1, "Stubs"); prof_->GetXaxis()->SetBinLabel(2, "Tracks"); prof_->GetXaxis()->SetBinLabel(3, "Lost Tracks"); @@ -128,9 +120,10 @@ namespace trackerTFP { prof_->GetXaxis()->SetBinLabel(7, "Found selected TPs"); prof_->GetXaxis()->SetBinLabel(8, "Lost TPs"); prof_->GetXaxis()->SetBinLabel(9, "All TPs"); + prof_->GetXaxis()->SetBinLabel(10, "Perfectly Found selected TPs"); // channel occupancy constexpr int maxOcc = 180; - const int numChannels = dataFormats_->numChannel(Process::mht); + const int numChannels = setup_->numRegions(); hisChannel_ = dir.make("His Channel Occupancy", ";", maxOcc, -.5, maxOcc - .5); profChannel_ = dir.make("Prof Channel Occupancy", ";", numChannels, -.5, numChannels - .5); // Efficiencies @@ -139,13 +132,11 @@ namespace trackerTFP { eff_ = dir.make("EffEta", ";", 128, -2.5, 2.5); } - void AnalyzerMHT::analyze(const Event& iEvent, const EventSetup& iSetup) { + void AnalyzerTFP::analyze(const Event& iEvent, const EventSetup& iSetup) { auto fill = [](const TPPtr& tpPtr, TH1F* his) { his->Fill(tpPtr->eta()); }; - // read in ht products - Handle handleAccepted; - iEvent.getByToken(edGetTokenAccepted_, handleAccepted); - Handle handleLost; - iEvent.getByToken(edGetTokenLost_, handleLost); + // read in tracklet products + Handle handle; + iEvent.getByToken(edGetToken_, handle); // read in MCTruth const StubAssociation* selection = nullptr; const StubAssociation* reconstructable = nullptr; @@ -160,133 +151,115 @@ namespace trackerTFP { for (const auto& p : selection->getTrackingParticleToTTStubsMap()) fill(p.first, hisEffTotal_); } - // analyze ht products and associate found tracks with reconstrucable TrackingParticles - set tpPtrs; - set tpPtrsSelection; - set tpPtrsLost; - int allMatched(0); - int allTracks(0); + // + const TTTracks& ttTracks = *handle.product(); + vector> ttTrackRefsRegions(setup_->numRegions()); + vector nTTTracksRegions(setup_->numRegions(), 0); + for (const TTTrack& ttTrack : ttTracks) + nTTTracksRegions[ttTrack.phiSector()]++; + for (int region = 0; region < setup_->numRegions(); region++) + ttTrackRefsRegions[region].reserve(nTTTracksRegions[region]); + int i(0); + for (const TTTrack& ttTrack : ttTracks) + ttTrackRefsRegions[ttTrack.phiSector()].emplace_back(TTTrackRef(handle, i++)); for (int region = 0; region < setup_->numRegions(); region++) { - int nStubs(0); - int nTracks(0); - int nLost(0); - for (int channel = 0; channel < dataFormats_->numChannel(Process::mht); channel++) { - const int index = region * dataFormats_->numChannel(Process::mht) + channel; - const StreamStub& accepted = handleAccepted->at(index); - hisChannel_->Fill(accepted.size()); - profChannel_->Fill(channel, accepted.size()); - nStubs += accumulate(accepted.begin(), accepted.end(), 0, [](int sum, const FrameStub& frame) { - return sum + (frame.first.isNonnull() ? 1 : 0); - }); - vector> tracks; - vector> lost; - formTracks(accepted, tracks); - formTracks(handleLost->at(index), lost); - nTracks += tracks.size(); - allTracks += tracks.size(); - nLost += lost.size(); - if (!useMCTruth_) - continue; - int tmp(0); - associate(tracks, selection, tpPtrsSelection, tmp); - associate(lost, selection, tpPtrsLost, tmp); - associate(tracks, reconstructable, tpPtrs, allMatched); - } + const vector& ttTrackRefs = ttTrackRefsRegions[region]; + const int nStubs = + accumulate(ttTrackRefs.begin(), ttTrackRefs.end(), 0, [](int sum, const TTTrackRef& ttTrackRef) { + return sum + ttTrackRef->getStubRefs().size(); + }); + const int nTracks = ttTrackRefs.size(); prof_->Fill(1, nStubs); prof_->Fill(2, nTracks); - prof_->Fill(3, nLost); + // no access to lost tracks + prof_->Fill(3, 0); + hisChannel_->Fill(nTracks); + profChannel_->Fill(region, nTracks); + } + // analyze tracklet products and associate found tracks with reconstrucable TrackingParticles + set tpPtrs; + set tpPtrsSelection; + set tpPtrsPerfect; + int nAllMatched(0); + // convert vector of tracks to vector of vector of associated stubs + vector> tracks; + tracks.reserve(ttTracks.size()); + transform( + ttTracks.begin(), ttTracks.end(), back_inserter(tracks), [](const TTTrack& ttTrack) { + return ttTrack.getStubRefs(); + }); + if (useMCTruth_) { + int tmp(0); + associate(tracks, selection, tpPtrsSelection, tmp); + associate(tracks, selection, tpPtrsPerfect, tmp, true); + associate(tracks, reconstructable, tpPtrs, nAllMatched); } for (const TPPtr& tpPtr : tpPtrsSelection) fill(tpPtr, hisEff_); - vector recovered; - recovered.reserve(tpPtrsLost.size()); - set_intersection(tpPtrsLost.begin(), tpPtrsLost.end(), tpPtrs.begin(), tpPtrs.end(), back_inserter(recovered)); - for (const TPPtr& tpPtr : recovered) - tpPtrsLost.erase(tpPtr); - prof_->Fill(4, allMatched); - prof_->Fill(5, allTracks); + prof_->Fill(4, nAllMatched); + prof_->Fill(5, ttTracks.size()); prof_->Fill(6, tpPtrs.size()); prof_->Fill(7, tpPtrsSelection.size()); - prof_->Fill(8, tpPtrsLost.size()); + // no access to lost tp + prof_->Fill(8, 0); + prof_->Fill(10, tpPtrsPerfect.size()); nEvents_++; } - void AnalyzerMHT::endJob() { + void AnalyzerTFP::endJob() { if (nEvents_ == 0) return; // effi eff_->SetPassedHistogram(*hisEff_, "f"); eff_->SetTotalHistogram(*hisEffTotal_, "f"); - // printout MHT summary + // printout SF summary const double totalTPs = prof_->GetBinContent(9); const double numStubs = prof_->GetBinContent(1); const double numTracks = prof_->GetBinContent(2); - const double numTracksLost = prof_->GetBinContent(3); const double totalTracks = prof_->GetBinContent(5); const double numTracksMatched = prof_->GetBinContent(4); const double numTPsAll = prof_->GetBinContent(6); const double numTPsEff = prof_->GetBinContent(7); - const double numTPsLost = prof_->GetBinContent(8); + const double numTPsEffPerfect = prof_->GetBinContent(10); const double errStubs = prof_->GetBinError(1); const double errTracks = prof_->GetBinError(2); - const double errTracksLost = prof_->GetBinError(3); const double fracFake = (totalTracks - numTracksMatched) / totalTracks; const double fracDup = (numTracksMatched - numTPsAll) / totalTracks; const double eff = numTPsEff / totalTPs; const double errEff = sqrt(eff * (1. - eff) / totalTPs / nEvents_); - const double effLoss = numTPsLost / totalTPs; - const double errEffLoss = sqrt(effLoss * (1. - effLoss) / totalTPs / nEvents_); - const vector nums = {numStubs, numTracks, numTracksLost}; - const vector errs = {errStubs, errTracks, errTracksLost}; + const double effPerfect = numTPsEffPerfect / totalTPs; + const double errEffPerfect = sqrt(effPerfect * (1. - effPerfect) / totalTPs / nEvents_); + const vector nums = {numStubs, numTracks}; + const vector errs = {errStubs, errTracks}; const int wNums = ceil(log10(*max_element(nums.begin(), nums.end()))) + 5; const int wErrs = ceil(log10(*max_element(errs.begin(), errs.end()))) + 5; - log_ << " MHT SUMMARY " << endl; - log_ << "number of stubs per TFP = " << setw(wNums) << numStubs << " +- " << setw(wErrs) << errStubs << endl; - log_ << "number of tracks per TFP = " << setw(wNums) << numTracks << " +- " << setw(wErrs) << errTracks - << endl; - log_ << "number of lost tracks per TFP = " << setw(wNums) << numTracksLost << " +- " << setw(wErrs) << errTracksLost + log_ << " TFP SUMMARY " << endl; + log_ << "number of stubs per TFP = " << setw(wNums) << numStubs << " +- " << setw(wErrs) << errStubs << endl; + log_ << "number of tracks per TFP = " << setw(wNums) << numTracks << " +- " << setw(wErrs) << errTracks << endl; + log_ << "current tracking efficiency = " << setw(wNums) << effPerfect << " +- " << setw(wErrs) << errEffPerfect << endl; - log_ << " max tracking efficiency = " << setw(wNums) << eff << " +- " << setw(wErrs) << errEff << endl; - log_ << " lost tracking efficiency = " << setw(wNums) << effLoss << " +- " << setw(wErrs) << errEffLoss << endl; - log_ << " fake rate = " << setw(wNums) << fracFake << endl; - log_ << " duplicate rate = " << setw(wNums) << fracDup << endl; + log_ << "max tracking efficiency = " << setw(wNums) << eff << " +- " << setw(wErrs) << errEff << endl; + log_ << " fake rate = " << setw(wNums) << fracFake << endl; + log_ << " duplicate rate = " << setw(wNums) << fracDup << endl; log_ << "============================================================="; - LogPrint("L1Trigger/TrackerTFP") << log_.str(); - } - - // - void AnalyzerMHT::formTracks(const StreamStub& stream, vector>& tracks) const { - vector stubs; - stubs.reserve(stream.size()); - for (const FrameStub& frame : stream) - if (frame.first.isNonnull()) - stubs.emplace_back(frame, dataFormats_); - for (auto it = stubs.begin(); it != stubs.end();) { - const auto start = it; - const int id = it->trackId(); - auto different = [id](const StubMHT& stub) { return id != stub.trackId(); }; - it = find_if(it, stubs.end(), different); - vector ttStubRefs; - ttStubRefs.reserve(distance(start, it)); - transform(start, it, back_inserter(ttStubRefs), [](const StubMHT& stub) { return stub.ttStubRef(); }); - tracks.push_back(ttStubRefs); - } + LogPrint(moduleDescription().moduleName()) << log_.str(); } - // - void AnalyzerMHT::associate(const vector>& tracks, + // gets all TPs associated too any of the tracks & number of tracks matching at least one TP + void AnalyzerTFP::associate(const vector>& tracks, const StubAssociation* ass, set& tps, - int& sum) const { + int& nMatchTrk, + bool perfect) const { for (const vector& ttStubRefs : tracks) { - const vector& tpPtrs = ass->associate(ttStubRefs); + const vector& tpPtrs = perfect ? ass->associateFinal(ttStubRefs) : ass->associate(ttStubRefs); if (tpPtrs.empty()) continue; - sum++; + nMatchTrk++; copy(tpPtrs.begin(), tpPtrs.end(), inserter(tps, tps.begin())); } } -} // namespace trackerTFP +} // namespace trklet -DEFINE_FWK_MODULE(trackerTFP::AnalyzerMHT); +DEFINE_FWK_MODULE(trklet::AnalyzerTFP); diff --git a/L1Trigger/TrackFindingTracklet/test/AnalyzerDRin.cc b/L1Trigger/TrackFindingTracklet/test/AnalyzerTM.cc similarity index 62% rename from L1Trigger/TrackFindingTracklet/test/AnalyzerDRin.cc rename to L1Trigger/TrackFindingTracklet/test/AnalyzerTM.cc index e91d45c0b06e2..4ade077b3a398 100644 --- a/L1Trigger/TrackFindingTracklet/test/AnalyzerDRin.cc +++ b/L1Trigger/TrackFindingTracklet/test/AnalyzerTM.cc @@ -35,14 +35,14 @@ using namespace tt; namespace trklet { - /*! \class trklet::AnalyzerDRin - * \brief Class to analyze hardware like structured TTStub Collection generated by DRin module + /*! \class trklet::AnalyzerTM + * \brief Class to analyze hardware like structured TTStub Collection generated by TM module * \author Thomas Schuh * \date 2023, Jan */ - class AnalyzerDRin : public one::EDAnalyzer { + class AnalyzerTM : public one::EDAnalyzer { public: - AnalyzerDRin(const ParameterSet& iConfig); + AnalyzerTM(const ParameterSet& iConfig); void beginJob() override {} void beginRun(const Run& iEvent, const EventSetup& iSetup) override; void analyze(const Event& iEvent, const EventSetup& iSetup) override; @@ -63,13 +63,9 @@ namespace trklet { bool perfect = false) const; // ED input token of stubs - EDGetTokenT edGetTokenAcceptedStubs_; + EDGetTokenT edGetTokenStubs_; // ED input token of tracks - EDGetTokenT edGetTokenAcceptedTracks_; - // ED input token of lost stubs - EDGetTokenT edGetTokenLostStubs_; - // ED input token of lost tracks - EDGetTokenT edGetTokenLostTracks_; + EDGetTokenT edGetTokenTracks_; // ED input token of TTStubRef to TPPtr association for tracking efficiency EDGetTokenT edGetTokenSelection_; // ED input token of TTStubRef to recontructable TPPtr association @@ -101,18 +97,14 @@ namespace trklet { stringstream log_; }; - AnalyzerDRin::AnalyzerDRin(const ParameterSet& iConfig) : useMCTruth_(iConfig.getParameter("UseMCTruth")) { + AnalyzerTM::AnalyzerTM(const ParameterSet& iConfig) : useMCTruth_(iConfig.getParameter("UseMCTruth")) { usesResource("TFileService"); // book in- and output ED products - const string& label = iConfig.getParameter("LabelDRin"); - const string& branchAcceptedStubs = iConfig.getParameter("BranchAcceptedStubs"); - const string& branchAcceptedTracks = iConfig.getParameter("BranchAcceptedTracks"); - const string& branchLostStubs = iConfig.getParameter("BranchLostStubs"); - const string& branchLostTracks = iConfig.getParameter("BranchLostTracks"); - edGetTokenAcceptedStubs_ = consumes(InputTag(label, branchAcceptedStubs)); - edGetTokenAcceptedTracks_ = consumes(InputTag(label, branchAcceptedTracks)); - edGetTokenLostStubs_ = consumes(InputTag(label, branchLostStubs)); - edGetTokenLostTracks_ = consumes(InputTag(label, branchLostTracks)); + const string& label = iConfig.getParameter("OutputLabelTM"); + const string& branchStubs = iConfig.getParameter("BranchStubs"); + const string& branchTracks = iConfig.getParameter("BranchTracks"); + edGetTokenStubs_ = consumes(InputTag(label, branchStubs)); + edGetTokenTracks_ = consumes(InputTag(label, branchTracks)); if (useMCTruth_) { const auto& inputTagSelecttion = iConfig.getParameter("InputTagSelection"); const auto& inputTagReconstructable = iConfig.getParameter("InputTagReconstructable"); @@ -128,7 +120,7 @@ namespace trklet { log_.precision(4); } - void AnalyzerDRin::beginRun(const Run& iEvent, const EventSetup& iSetup) { + void AnalyzerTM::beginRun(const Run& iEvent, const EventSetup& iSetup) { // helper class to store configurations setup_ = &iSetup.getData(esGetTokenSetup_); // helper class to extract structured data from tt::Frames @@ -138,39 +130,31 @@ namespace trklet { // book histograms Service fs; TFileDirectory dir; - dir = fs->mkdir("DRin"); + dir = fs->mkdir("TM"); prof_ = dir.make("Counts", ";", 10, 0.5, 10.5); prof_->GetXaxis()->SetBinLabel(1, "Stubs"); prof_->GetXaxis()->SetBinLabel(2, "Tracks"); - prof_->GetXaxis()->SetBinLabel(3, "Lost Tracks"); prof_->GetXaxis()->SetBinLabel(4, "Matched Tracks"); prof_->GetXaxis()->SetBinLabel(5, "All Tracks"); prof_->GetXaxis()->SetBinLabel(6, "Found TPs"); prof_->GetXaxis()->SetBinLabel(7, "Found selected TPs"); - prof_->GetXaxis()->SetBinLabel(8, "Lost TPs"); prof_->GetXaxis()->SetBinLabel(9, "All TPs"); prof_->GetXaxis()->SetBinLabel(10, "Perfect TPs"); // channel occupancy constexpr int maxOcc = 180; - const int numChannels = channelAssignment_->numNodesDR(); + const int numChannels = 1; hisChannel_ = dir.make("His Channel Occupancy", ";", maxOcc, -.5, maxOcc - .5); profChannel_ = dir.make("Prof Channel Occupancy", ";", numChannels, -.5, numChannels - .5); } - void AnalyzerDRin::analyze(const Event& iEvent, const EventSetup& iSetup) { + void AnalyzerTM::analyze(const Event& iEvent, const EventSetup& iSetup) { // read in ht products - Handle handleAcceptedStubs; - iEvent.getByToken(edGetTokenAcceptedStubs_, handleAcceptedStubs); - const StreamsStub& acceptedStubs = *handleAcceptedStubs; - Handle handleAcceptedTracks; - iEvent.getByToken(edGetTokenAcceptedTracks_, handleAcceptedTracks); - const StreamsTrack& acceptedTracks = *handleAcceptedTracks; - Handle handleLostStubs; - iEvent.getByToken(edGetTokenLostStubs_, handleLostStubs); - const StreamsStub& lostStubs = *handleLostStubs; - Handle handleLostTracks; - iEvent.getByToken(edGetTokenLostTracks_, handleLostTracks); - const StreamsTrack& lostTracks = *handleLostTracks; + Handle handleStubs; + iEvent.getByToken(edGetTokenStubs_, handleStubs); + const StreamsStub& streamsStub = *handleStubs; + Handle handleTracks; + iEvent.getByToken(edGetTokenTracks_, handleTracks); + const StreamsTrack& streamsTrack = *handleTracks; // read in MCTruth const StubAssociation* selection = nullptr; const StubAssociation* reconstructable = nullptr; @@ -187,108 +171,85 @@ namespace trklet { set tpPtrs; set tpPtrsSelection; set tpPtrsPerfect; - set tpPtrsLost; int allMatched(0); int allTracks(0); for (int region = 0; region < setup_->numRegions(); region++) { - const int offset = region * channelAssignment_->numNodesDR(); int nStubs(0); int nTracks(0); - int nLost(0); - for (int channel = 0; channel < channelAssignment_->numNodesDR(); channel++) { - vector> tracks; - formTracks(acceptedTracks, acceptedStubs, tracks, offset + channel); - vector> lost; - formTracks(lostTracks, lostStubs, lost, offset + channel); - nTracks += tracks.size(); - nStubs += accumulate(tracks.begin(), tracks.end(), 0, [](int sum, const vector& track) { - return sum + (int)track.size(); - }); - nLost += lost.size(); - allTracks += tracks.size(); - if (!useMCTruth_) - continue; - int tmp(0); - associate(tracks, selection, tpPtrsSelection, tmp); - associate(tracks, selection, tpPtrsPerfect, tmp, true); - associate(lost, selection, tpPtrsLost, tmp); - associate(tracks, reconstructable, tpPtrs, allMatched); - const StreamTrack& stream = acceptedTracks[offset + channel]; - const auto end = - find_if(stream.rbegin(), stream.rend(), [](const FrameTrack& frame) { return frame.first.isNonnull(); }); - const int size = distance(stream.begin(), end.base()) - 1; - hisChannel_->Fill(size); - profChannel_->Fill(channel, size); - } + vector> tracks; + formTracks(streamsTrack, streamsStub, tracks, region); + nTracks += tracks.size(); + nStubs += accumulate(tracks.begin(), tracks.end(), 0, [](int& sum, const vector& track) { + return sum += (int)track.size(); + }); + allTracks += tracks.size(); + if (!useMCTruth_) + continue; + int tmp(0); + associate(tracks, selection, tpPtrsSelection, tmp); + associate(tracks, selection, tpPtrsPerfect, tmp, true); + associate(tracks, reconstructable, tpPtrs, allMatched); + const StreamTrack& stream = streamsTrack[region]; + const auto end = + find_if(stream.rbegin(), stream.rend(), [](const FrameTrack& frame) { return frame.first.isNonnull(); }); + const int size = distance(stream.begin(), end.base()) - 1; + hisChannel_->Fill(size); + profChannel_->Fill(1, size); prof_->Fill(1, nStubs); prof_->Fill(2, nTracks); - prof_->Fill(3, nLost); } - vector recovered; - recovered.reserve(tpPtrsLost.size()); - set_intersection(tpPtrsLost.begin(), tpPtrsLost.end(), tpPtrs.begin(), tpPtrs.end(), back_inserter(recovered)); - for (const TPPtr& tpPtr : recovered) - tpPtrsLost.erase(tpPtr); prof_->Fill(4, allMatched); prof_->Fill(5, allTracks); prof_->Fill(6, tpPtrs.size()); prof_->Fill(7, tpPtrsSelection.size()); - prof_->Fill(8, tpPtrsLost.size()); prof_->Fill(10, tpPtrsPerfect.size()); nEvents_++; } - void AnalyzerDRin::endJob() { + void AnalyzerTM::endJob() { if (nEvents_ == 0) return; - // printout SF summary + // printout summary const double totalTPs = prof_->GetBinContent(9); const double numStubs = prof_->GetBinContent(1); const double numTracks = prof_->GetBinContent(2); - const double numTracksLost = prof_->GetBinContent(3); const double totalTracks = prof_->GetBinContent(5); const double numTracksMatched = prof_->GetBinContent(4); const double numTPsAll = prof_->GetBinContent(6); const double numTPsEff = prof_->GetBinContent(7); - const double numTPsLost = prof_->GetBinContent(8); const double numTPsEffPerfect = prof_->GetBinContent(10); const double errStubs = prof_->GetBinError(1); const double errTracks = prof_->GetBinError(2); - const double errTracksLost = prof_->GetBinError(3); const double fracFake = (totalTracks - numTracksMatched) / totalTracks; const double fracDup = (numTracksMatched - numTPsAll) / totalTracks; const double eff = numTPsEff / totalTPs; const double errEff = sqrt(eff * (1. - eff) / totalTPs / nEvents_); - const double effLoss = numTPsLost / totalTPs; - const double errEffLoss = sqrt(effLoss * (1. - effLoss) / totalTPs / nEvents_); const double effPerfect = numTPsEffPerfect / totalTPs; const double errEffPerfect = sqrt(effPerfect * (1. - effPerfect) / totalTPs / nEvents_); - const vector nums = {numStubs, numTracks, numTracksLost}; - const vector errs = {errStubs, errTracks, errTracksLost}; + const vector nums = {numStubs, numTracks}; + const vector errs = {errStubs, errTracks}; const int wNums = ceil(log10(*max_element(nums.begin(), nums.end()))) + 5; const int wErrs = ceil(log10(*max_element(errs.begin(), errs.end()))) + 5; - log_ << " DRin SUMMARY " << endl; + log_ << " TM SUMMARY " << endl; log_ << "number of stubs per TFP = " << setw(wNums) << numStubs << " +- " << setw(wErrs) << errStubs << endl; log_ << "number of tracks per TFP = " << setw(wNums) << numTracks << " +- " << setw(wErrs) << errTracks << endl; - log_ << "number of lost tracks per TFP = " << setw(wNums) << numTracksLost << " +- " << setw(wErrs) << errTracksLost - << endl; log_ << " current tracking efficiency = " << setw(wNums) << effPerfect << " +- " << setw(wErrs) << errEffPerfect << endl; log_ << " max tracking efficiency = " << setw(wNums) << eff << " +- " << setw(wErrs) << errEff << endl; - log_ << " lost tracking efficiency = " << setw(wNums) << effLoss << " +- " << setw(wErrs) << errEffLoss << endl; log_ << " fake rate = " << setw(wNums) << fracFake << endl; log_ << " duplicate rate = " << setw(wNums) << fracDup << endl; log_ << "============================================================="; - LogPrint("L1Trigger/TrackerTFP") << log_.str(); + LogPrint(moduleDescription().moduleName()) << log_.str(); } // - void AnalyzerDRin::formTracks(const StreamsTrack& streamsTrack, - const StreamsStub& streamsStubs, - vector>& tracks, - int channel) const { - const int offset = channel * setup_->numLayers(); + void AnalyzerTM::formTracks(const StreamsTrack& streamsTrack, + const StreamsStub& streamsStubs, + vector>& tracks, + int channel) const { + static const int numLayers = channelAssignment_->tmNumLayers(); + const int offset = channel * numLayers; const StreamTrack& streamTrack = streamsTrack[channel]; const int numTracks = accumulate(streamTrack.begin(), streamTrack.end(), 0, [](int sum, const FrameTrack& frame) { return sum + (frame.first.isNonnull() ? 1 : 0); @@ -299,8 +260,8 @@ namespace trklet { if (frameTrack.first.isNull()) continue; vector ttStubRefs; - ttStubRefs.reserve(setup_->numLayers()); - for (int layer = 0; layer < setup_->numLayers(); layer++) { + ttStubRefs.reserve(numLayers); + for (int layer = 0; layer < numLayers; layer++) { const FrameStub& stub = streamsStubs[offset + layer][frame]; if (stub.first.isNonnull()) ttStubRefs.push_back(stub.first); @@ -310,11 +271,11 @@ namespace trklet { } // - void AnalyzerDRin::associate(const vector>& tracks, - const StubAssociation* ass, - set& tps, - int& sum, - bool perfect) const { + void AnalyzerTM::associate(const vector>& tracks, + const StubAssociation* ass, + set& tps, + int& sum, + bool perfect) const { for (const vector& ttStubRefs : tracks) { const vector& tpPtrs = perfect ? ass->associateFinal(ttStubRefs) : ass->associate(ttStubRefs); if (tpPtrs.empty()) @@ -326,4 +287,8 @@ namespace trklet { } // namespace trklet +<<<<<<< HEAD:L1Trigger/TrackFindingTracklet/test/AnalyzerDRin.cc DEFINE_FWK_MODULE(trklet::AnalyzerDRin); +======= +DEFINE_FWK_MODULE(trklet::AnalyzerTM); +>>>>>>> c9a89dd5637 (squash):L1Trigger/TrackFindingTracklet/test/AnalyzerTM.cc diff --git a/L1Trigger/TrackFindingTracklet/test/AnalyzerTracklet.cc b/L1Trigger/TrackFindingTracklet/test/AnalyzerTracklet.cc index 0306313194801..59711d55880c2 100644 --- a/L1Trigger/TrackFindingTracklet/test/AnalyzerTracklet.cc +++ b/L1Trigger/TrackFindingTracklet/test/AnalyzerTracklet.cc @@ -15,6 +15,7 @@ #include "SimTracker/TrackTriggerAssociation/interface/StubAssociation.h" #include "L1Trigger/TrackTrigger/interface/Setup.h" #include "L1Trigger/TrackerTFP/interface/DataFormats.h" +#include "L1Trigger/TrackFindingTracklet/interface/ChannelAssignment.h" #include #include @@ -66,10 +67,14 @@ namespace trklet { ESGetToken esGetTokenSetup_; // DataFormats token ESGetToken esGetTokenDataFormats_; + // ChannelAssignment token + ESGetToken esGetTokenChannelAssignment_; // stores, calculates and provides run-time constants const Setup* setup_ = nullptr; // helper class to extract structured data from tt::Frames const DataFormats* dataFormats_ = nullptr; + // helper class to assign tracks to channel + const ChannelAssignment* channelAssignment_ = nullptr; // enables analyze of TPs bool useMCTruth_; // @@ -105,6 +110,7 @@ namespace trklet { // book ES products esGetTokenSetup_ = esConsumes(); esGetTokenDataFormats_ = esConsumes(); + esGetTokenChannelAssignment_ = esConsumes(); // log config log_.setf(ios::fixed, ios::floatfield); log_.precision(4); @@ -115,6 +121,8 @@ namespace trklet { setup_ = &iSetup.getData(esGetTokenSetup_); // helper class to extract structured data from tt::Frames dataFormats_ = &iSetup.getData(esGetTokenDataFormats_); + // helper class to assign tracks to channel + channelAssignment_ = &iSetup.getData(esGetTokenChannelAssignment_); // book histograms Service fs; TFileDirectory dir; @@ -132,7 +140,7 @@ namespace trklet { prof_->GetXaxis()->SetBinLabel(10, "Perfectly Found selected TPs"); // channel occupancy constexpr int maxOcc = 180; - const int numChannels = setup_->numRegions(); + const int numChannels = channelAssignment_->numSeedTypes(); hisChannel_ = dir.make("His Channel Occupancy", ";", maxOcc, -.5, maxOcc - .5); profChannel_ = dir.make("Prof Channel Occupancy", ";", numChannels, -.5, numChannels - .5); // Efficiencies @@ -178,12 +186,18 @@ namespace trklet { return sum + ttTrackRef->getStubRefs().size(); }); const int nTracks = ttTrackRefs.size(); - hisChannel_->Fill(nTracks); - profChannel_->Fill(region, nTracks); prof_->Fill(1, nStubs); prof_->Fill(2, nTracks); // no access to lost tracks prof_->Fill(3, 0); + for (int seedType = 0; seedType < channelAssignment_->numSeedTypes(); seedType++) { + const int nTracks = + accumulate(ttTrackRefs.begin(), ttTrackRefs.end(), 0, [seedType](int& sum, const TTTrackRef& ttTrackRef) { + return sum += ((int)ttTrackRef->trackSeedType() == seedType ? 1 : 0); + }); + hisChannel_->Fill(nTracks); + profChannel_->Fill(seedType, nTracks); + } } // analyze tracklet products and associate found tracks with reconstrucable TrackingParticles set tpPtrs; @@ -251,7 +265,7 @@ namespace trklet { log_ << " fake rate = " << setw(wNums) << fracFake << endl; log_ << " duplicate rate = " << setw(wNums) << fracDup << endl; log_ << "============================================================="; - LogPrint("L1Trigger/TrackFindingTracklet") << log_.str(); + LogPrint(moduleDescription().moduleName()) << log_.str(); } // gets all TPs associated too any of the tracks & number of tracks matching at least one TP diff --git a/L1Trigger/TrackFindingTracklet/test/HybridTracksNewKF_cfg.py b/L1Trigger/TrackFindingTracklet/test/HybridTracksNewKF_cfg.py index b9e05c5a39e41..f60550c3ada70 100644 --- a/L1Trigger/TrackFindingTracklet/test/HybridTracksNewKF_cfg.py +++ b/L1Trigger/TrackFindingTracklet/test/HybridTracksNewKF_cfg.py @@ -12,8 +12,8 @@ process = cms.Process( "Demo" ) process.load( 'FWCore.MessageService.MessageLogger_cfi' ) process.load( 'Configuration.EventContent.EventContent_cff' ) -process.load( 'Configuration.Geometry.GeometryExtendedRun4D88Reco_cff' ) -process.load( 'Configuration.Geometry.GeometryExtendedRun4D88_cff' ) +process.load( 'Configuration.Geometry.GeometryExtended2026D98Reco_cff' ) +process.load( 'Configuration.Geometry.GeometryExtended2026D98_cff' ) process.load( 'Configuration.StandardSequences.MagneticField_cff' ) process.load( 'Configuration.StandardSequences.FrontierConditions_GlobalTag_cff' ) process.load( 'L1Trigger.TrackTrigger.TrackTrigger_cff' ) @@ -24,36 +24,30 @@ # load code that associates stubs with mctruth process.load( 'SimTracker.TrackTriggerAssociation.StubAssociator_cff' ) # load code that produces DTCStubs -process.load( 'L1Trigger.TrackerDTC.ProducerED_cff' ) +process.load( 'L1Trigger.TrackerDTC.DTC_cff' ) # load code that analyzes DTCStubs process.load( 'L1Trigger.TrackerDTC.Analyzer_cff' ) # L1 tracking => hybrid emulation process.load("L1Trigger.TrackFindingTracklet.L1HybridEmulationTracks_cff") -from L1Trigger.TrackFindingTracklet.Customize_cff import * -fwConfig( process ) #--- Load code that analyzes hybrid emulation process.load( 'L1Trigger.TrackFindingTracklet.Analyzer_cff' ) # load code that fits hybrid tracks process.load( 'L1Trigger.TrackFindingTracklet.Producer_cff' ) - -# load and configure TrackTriggerAssociation -process.load( 'SimTracker.TrackTriggerAssociation.TrackTriggerAssociator_cff' ) -process.TTTrackAssociatorFromPixelDigis.TTTracks = cms.VInputTag( cms.InputTag( - process.TrackFindingTrackletProducer_params.LabelKFout.value(), - process.TrackFindingTrackletProducer_params.BranchAcceptedTTTracks.value() -) ) +from L1Trigger.TrackFindingTracklet.Customize_cff import * +fwConfig( process ) +oldKFConfig( process ) +process.l1tTTTracksFromTrackletEmulation.readMoreMcTruth = False # build schedule -process.mc = cms.Sequence( process.StubAssociator ) -process.dtc = cms.Sequence( process.TrackerDTCProducer + process.TrackerDTCAnalyzer ) -process.tracklet = cms.Sequence( process.L1THybridTracks + process.TrackFindingTrackletAnalyzerTracklet ) -process.TBout = cms.Sequence( process.TrackFindingTrackletProducerTBout + process.TrackFindingTrackletAnalyzerTBout ) -process.drin = cms.Sequence( process.TrackFindingTrackletProducerDRin + process.TrackFindingTrackletAnalyzerDRin ) -process.dr = cms.Sequence( process.TrackFindingTrackletProducerDR + process.TrackFindingTrackletAnalyzerDR ) -process.kfin = cms.Sequence( process.TrackFindingTrackletProducerKFin + process.TrackFindingTrackletAnalyzerKFin ) -process.kf = cms.Sequence( process.TrackFindingTrackletProducerKF + process.TrackFindingTrackletAnalyzerKF ) -process.kfout = cms.Sequence( process.TrackFindingTrackletProducerKFout + process.TrackFindingTrackletAnalyzerKFout ) -process.tt = cms.Path( process.mc + process.dtc + process.tracklet + process.TBout + process.drin + process.dr + process.kfin + process.kf + process.kfout) +process.mc = cms.Sequence( process.StubAssociator ) +process.dtc = cms.Sequence( process.ProducerDTC + process.AnalyzerDTC ) +process.tracklet = cms.Sequence( process.L1THybridTracks + process.AnalyzerTracklet ) +process.tm = cms.Sequence( process.ProducerTM + process.AnalyzerTM ) +process.dr = cms.Sequence( process.ProducerDR + process.AnalyzerDR ) +process.kf = cms.Sequence( process.ProducerKF + process.AnalyzerKF ) +process.tq = cms.Sequence( process.ProducerTQ ) +process.tfp = cms.Sequence( process.ProducerTFP + process.AnalyzerTFP ) +process.tt = cms.Path( process.mc + process.dtc + process.tracklet + process.tm + process.dr + process.kf + process.tq + process.tfp ) process.schedule = cms.Schedule( process.tt ) # create options @@ -64,8 +58,10 @@ #from MCsamples.Scripts.getCMSlocaldata_cfi import * #from MCsamples.RelVal_1260_D88.PU200_TTbar_14TeV_cfi import * #inputMC = getCMSdataFromCards() -inputMC = ["/store/mc/CMSSW_12_6_0/RelValTTbar_14TeV/GEN-SIM-DIGI-RAW/PU_125X_mcRun4_realistic_v5_2026D88PU200RV183v2-v1/30000/0959f326-3f52-48d8-9fcf-65fc41de4e27.root"] -options.register( 'inputMC', inputMC, VarParsing.VarParsing.multiplicity.singleton, VarParsing.VarParsing.varType.string, "Files to be processed" ) +Samples = [ + '/store/relval/CMSSW_14_0_0_pre2/RelValTTbar_14TeV/GEN-SIM-DIGI-RAW/PU_133X_mcRun4_realistic_v1_STD_2026D98_PU200_RV229-v1/2580000/0b2b0b0b-f312-48a8-9d46-ccbadc69bbfd.root' +] +options.register( 'inputMC', Samples, VarParsing.VarParsing.multiplicity.singleton, VarParsing.VarParsing.varType.string, "Files to be processed" ) # specify number of events to process. options.register( 'Events',100,VarParsing.VarParsing.multiplicity.singleton, VarParsing.VarParsing.varType.int, "Number of Events to analyze" ) options.parseArguments() @@ -75,12 +71,13 @@ process.source = cms.Source( "PoolSource", fileNames = cms.untracked.vstring( options.inputMC ), - #skipEvents = cms.untracked.uint32( 250 ), + #skipEvents = cms.untracked.uint32( 3537 ), secondaryFileNames = cms.untracked.vstring(), duplicateCheckMode = cms.untracked.string( 'noDuplicateCheck' ) ) process.Timing = cms.Service( "Timing", summaryOnly = cms.untracked.bool( True ) ) process.MessageLogger.cerr.enableStatistics = False +process.MessageLogger.L1track = dict(limit = -1) process.TFileService = cms.Service( "TFileService", fileName = cms.string( "Hist.root" ) ) if ( False ): diff --git a/L1Trigger/TrackFindingTracklet/test/HybridTracks_cfg.py b/L1Trigger/TrackFindingTracklet/test/HybridTracks_cfg.py index e782d84048448..f794985786064 100644 --- a/L1Trigger/TrackFindingTracklet/test/HybridTracks_cfg.py +++ b/L1Trigger/TrackFindingTracklet/test/HybridTracks_cfg.py @@ -15,8 +15,8 @@ process.load('Configuration.EventContent.EventContent_cff') process.load('Configuration.StandardSequences.MagneticField_cff') -process.load('Configuration.Geometry.GeometryExtendedRun4D88Reco_cff') -process.load('Configuration.Geometry.GeometryExtendedRun4D88_cff') +process.load('Configuration.Geometry.GeometryExtended2026D98Reco_cff') +process.load('Configuration.Geometry.GeometryExtended2026D98_cff') process.load('Configuration.StandardSequences.EndOfProcess_cff') process.load('Configuration.StandardSequences.FrontierConditions_GlobalTag_cff') @@ -30,16 +30,21 @@ # input # ---------------------------------------------------------------------------------- -process.maxEvents = cms.untracked.PSet(input = cms.untracked.int32(10)) -inputMC = ["/store/mc/CMSSW_12_6_0/RelValTTbar_14TeV/GEN-SIM-DIGI-RAW/PU_125X_mcRun4_realistic_v5_2026D88PU200RV183v2-v1/30000/0959f326-3f52-48d8-9fcf-65fc41de4e27.root"] -process.source = cms.Source("PoolSource", fileNames = cms.untracked.vstring(*inputMC)) +# create options +import FWCore.ParameterSet.VarParsing as VarParsing +options = VarParsing.VarParsing( 'analysis' ) +options.register( 'Events',100,VarParsing.VarParsing.multiplicity.singleton, VarParsing.VarParsing.varType.int, "Number of Events to analyze" ) +options.parseArguments() +process.maxEvents = cms.untracked.PSet( input = cms.untracked.int32(options.Events) ) +inputMC = ["/store/relval/CMSSW_14_0_0_pre2/RelValSingleMuFlatPt2To100/GEN-SIM-DIGI-RAW/133X_mcRun4_realistic_v1_STD_2026D98_noPU_RV229-v1/2580000/00b68219-8585-406f-88d0-84da05a13280.root"] +process.source = cms.Source("PoolSource", fileNames = cms.untracked.vstring(*inputMC), skipEvents = cms.untracked.uint32( 17 )) # ---------------------------------------------------------------------------------- # DTC emulation # ---------------------------------------------------------------------------------- -process.load( 'L1Trigger.TrackerDTC.ProducerED_cff' ) -process.dtc = cms.Path( process.TrackerDTCProducer ) +process.load( 'L1Trigger.TrackerDTC.DTC_cff' ) +process.dtc = cms.Path( process.ProducerDTC ) # ---------------------------------------------------------------------------------- # L1 tracking @@ -59,6 +64,8 @@ #process.TTTracksEmulation = cms.Path(process.L1TPromptExtendedHybridTracks) #process.TTTracksEmulationWithTruth = cms.Path(process.L1TPromptExtendedHybridTracksWithAssociators) +process.MessageLogger.L1track = dict(limit = -1) + # ---------------------------------------------------------------------------------- # output module # ---------------------------------------------------------------------------------- diff --git a/L1Trigger/TrackFindingTracklet/test/L1TrackNtupleMaker.cc b/L1Trigger/TrackFindingTracklet/test/L1TrackNtupleMaker.cc index 789a10020b25b..783a19cae3532 100644 --- a/L1Trigger/TrackFindingTracklet/test/L1TrackNtupleMaker.cc +++ b/L1Trigger/TrackFindingTracklet/test/L1TrackNtupleMaker.cc @@ -55,6 +55,8 @@ //////////////// // PHYSICS TOOLS +#include "L1Trigger/TrackTrigger/interface/Setup.h" +#include "L1Trigger/TrackerTFP/interface/LayerEncoding.h" #include "L1Trigger/TrackFindingTracklet/interface/HitPatternHelper.h" #include "CommonTools/UtilAlgos/interface/TFileService.h" #include "CLHEP/Units/PhysicalConstants.h" @@ -128,23 +130,25 @@ class L1TrackNtupleMaker : public one::EDAnalyzer > > ttClusterToken_; - edm::EDGetTokenT > > ttStubToken_; - edm::EDGetTokenT > ttClusterMCTruthToken_; - edm::EDGetTokenT > ttStubMCTruthToken_; + edm::EDGetTokenT>> ttClusterToken_; + edm::EDGetTokenT>> ttStubToken_; + edm::EDGetTokenT> ttClusterMCTruthToken_; + edm::EDGetTokenT> ttStubMCTruthToken_; - edm::EDGetTokenT > > ttTrackToken_; - edm::EDGetTokenT > ttTrackMCTruthToken_; + edm::EDGetTokenT>> ttTrackToken_; + edm::EDGetTokenT> ttTrackMCTruthToken_; - edm::EDGetTokenT > TrackingParticleToken_; - edm::EDGetTokenT > TrackingVertexToken_; + edm::EDGetTokenT> TrackingParticleToken_; + edm::EDGetTokenT> TrackingVertexToken_; - edm::EDGetTokenT > GenJetToken_; + edm::EDGetTokenT> GenJetToken_; edm::ESGetToken getTokenTrackerGeom_; edm::ESGetToken getTokenTrackerTopo_; edm::ESGetToken getTokenBField_; edm::ESGetToken getTokenHPHSetup_; + edm::ESGetToken getTokenSetup_; + edm::ESGetToken getTokenLayerEncoding_; //----------------------------------------------------------------------------------------------- // tree & branches for mini-ntuple @@ -196,6 +200,7 @@ class L1TrackNtupleMaker : public one::EDAnalyzer* m_trk_injet; //is the track within dR<0.4 of a genjet with pt > 30 GeV? std::vector* m_trk_injet_highpt; //is the track within dR<0.4 of a genjet with pt > 100 GeV? std::vector* m_trk_injet_vhighpt; //is the track within dR<0.4 of a genjet with pt > 200 GeV? + std::vector>* m_trk_layers; // all tracking particles std::vector* m_tp_pt; @@ -303,20 +308,22 @@ L1TrackNtupleMaker::L1TrackNtupleMaker(edm::ParameterSet const& iConfig) : confi TrackingVertexInputTag = iConfig.getParameter("TrackingVertexInputTag"); GenJetInputTag = iConfig.getParameter("GenJetInputTag"); - ttTrackToken_ = consumes > >(L1TrackInputTag); - ttTrackMCTruthToken_ = consumes >(MCTruthTrackInputTag); - ttStubToken_ = consumes > >(L1StubInputTag); - ttClusterMCTruthToken_ = consumes >(MCTruthClusterInputTag); - ttStubMCTruthToken_ = consumes >(MCTruthStubInputTag); + ttTrackToken_ = consumes>>(L1TrackInputTag); + ttTrackMCTruthToken_ = consumes>(MCTruthTrackInputTag); + ttStubToken_ = consumes>>(L1StubInputTag); + ttClusterMCTruthToken_ = consumes>(MCTruthClusterInputTag); + ttStubMCTruthToken_ = consumes>(MCTruthStubInputTag); - TrackingParticleToken_ = consumes >(TrackingParticleInputTag); - TrackingVertexToken_ = consumes >(TrackingVertexInputTag); - GenJetToken_ = consumes >(GenJetInputTag); + TrackingParticleToken_ = consumes>(TrackingParticleInputTag); + TrackingVertexToken_ = consumes>(TrackingVertexInputTag); + GenJetToken_ = consumes>(GenJetInputTag); getTokenTrackerGeom_ = esConsumes(); getTokenTrackerTopo_ = esConsumes(); getTokenBField_ = esConsumes(); getTokenHPHSetup_ = esConsumes(); + getTokenSetup_ = esConsumes(); + getTokenLayerEncoding_ = esConsumes(); } ///////////// @@ -494,6 +501,7 @@ void L1TrackNtupleMaker::beginJob() { m_trk_injet = new std::vector; m_trk_injet_highpt = new std::vector; m_trk_injet_vhighpt = new std::vector; + m_trk_layers = new std::vector>; m_tp_pt = new std::vector; m_tp_eta = new std::vector; @@ -610,6 +618,7 @@ void L1TrackNtupleMaker::beginJob() { eventTree->Branch("trk_injet_highpt", &m_trk_injet_highpt); eventTree->Branch("trk_injet_vhighpt", &m_trk_injet_vhighpt); } + eventTree->Branch("m_trk_layers", &m_trk_layers); } eventTree->Branch("tp_pt", &m_tp_pt); @@ -751,6 +760,7 @@ void L1TrackNtupleMaker::analyze(const edm::Event& iEvent, const edm::EventSetup m_trk_injet->clear(); m_trk_injet_highpt->clear(); m_trk_injet_vhighpt->clear(); + m_trk_layers->clear(); } m_tp_pt->clear(); @@ -827,25 +837,25 @@ void L1TrackNtupleMaker::analyze(const edm::Event& iEvent, const edm::EventSetup // ----------------------------------------------------------------------------------------------- // L1 tracks - edm::Handle > > TTTrackHandle; + edm::Handle>> TTTrackHandle; iEvent.getByToken(ttTrackToken_, TTTrackHandle); // L1 stubs - edm::Handle > > TTStubHandle; + edm::Handle>> TTStubHandle; if (SaveStubs) iEvent.getByToken(ttStubToken_, TTStubHandle); // MC truth association maps - edm::Handle > MCTruthTTClusterHandle; + edm::Handle> MCTruthTTClusterHandle; iEvent.getByToken(ttClusterMCTruthToken_, MCTruthTTClusterHandle); - edm::Handle > MCTruthTTStubHandle; + edm::Handle> MCTruthTTStubHandle; iEvent.getByToken(ttStubMCTruthToken_, MCTruthTTStubHandle); - edm::Handle > MCTruthTTTrackHandle; + edm::Handle> MCTruthTTTrackHandle; iEvent.getByToken(ttTrackMCTruthToken_, MCTruthTTTrackHandle); // tracking particles - edm::Handle > TrackingParticleHandle; - edm::Handle > TrackingVertexHandle; + edm::Handle> TrackingParticleHandle; + edm::Handle> TrackingVertexHandle; iEvent.getByToken(TrackingParticleToken_, TrackingParticleHandle); //iEvent.getByToken(TrackingVertexToken_, TrackingVertexHandle); @@ -858,10 +868,14 @@ void L1TrackNtupleMaker::analyze(const edm::Event& iEvent, const edm::EventSetup edm::ESHandle bFieldHandle = iSetup.getHandle(getTokenBField_); edm::ESHandle hphHandle = iSetup.getHandle(getTokenHPHSetup_); + edm::ESHandle handleSetup = iSetup.getHandle(getTokenSetup_); + edm::ESHandle handleLayerEncoding = iSetup.getHandle(getTokenLayerEncoding_); const TrackerTopology* const tTopo = tTopoHandle.product(); const TrackerGeometry* const theTrackerGeom = tGeomHandle.product(); const hph::Setup* hphSetup = hphHandle.product(); + const tt::Setup* setup = handleSetup.product(); + const trackerTFP::LayerEncoding* layerEncoding = handleLayerEncoding.product(); // ---------------------------------------------------------------------------------------------- // loop over L1 stubs @@ -880,14 +894,14 @@ void L1TrackNtupleMaker::analyze(const edm::Event& iEvent, const edm::EventSetup continue; // Get the DetSets of the Clusters - edmNew::DetSet > stubs = (*TTStubHandle)[stackDetid]; + edmNew::DetSet> stubs = (*TTStubHandle)[stackDetid]; const GeomDetUnit* det0 = theTrackerGeom->idToDetUnit(detid); const auto* theGeomDet = dynamic_cast(det0); const PixelTopology* topol = dynamic_cast(&(theGeomDet->specificTopology())); // loop over stubs for (auto stubIter = stubs.begin(); stubIter != stubs.end(); ++stubIter) { - edm::Ref >, TTStub > tempStubPtr = + edm::Ref>, TTStub> tempStubPtr = edmNew::makeRefTo(TTStubHandle, stubIter); int isBarrel = 0; @@ -985,7 +999,7 @@ void L1TrackNtupleMaker::analyze(const edm::Event& iEvent, const edm::EventSetup // gen jets if (DebugMode) edm::LogVerbatim("Tracklet") << "get genjets"; - edm::Handle > GenJetHandle; + edm::Handle> GenJetHandle; iEvent.getByToken(GenJetToken_, GenJetHandle); if (GenJetHandle.isValid()) { @@ -1042,9 +1056,9 @@ void L1TrackNtupleMaker::analyze(const edm::Event& iEvent, const edm::EventSetup } int this_l1track = 0; - std::vector >::const_iterator iterL1Track; + std::vector>::const_iterator iterL1Track; for (iterL1Track = TTTrackHandle->begin(); iterL1Track != TTTrackHandle->end(); iterL1Track++) { - edm::Ptr > l1track_ptr(TTTrackHandle, this_l1track); + edm::Ptr> l1track_ptr(TTTrackHandle, this_l1track); this_l1track++; float tmp_trk_pt = iterL1Track->momentum().perp(); @@ -1095,7 +1109,7 @@ void L1TrackNtupleMaker::analyze(const edm::Event& iEvent, const edm::EventSetup float tmp_trk_bendchi2 = iterL1Track->stubPtConsistency(); float tmp_trk_MVA1 = iterL1Track->trkMVA1(); - std::vector >, TTStub > > + std::vector>, TTStub>> stubRefs = iterL1Track->getStubRefs(); int tmp_trk_nstub = (int)stubRefs.size(); int ndof = 2 * tmp_trk_nstub - L1Tk_nPar; @@ -1330,6 +1344,16 @@ void L1TrackNtupleMaker::analyze(const edm::Event& iEvent, const edm::EventSetup } //end tracking in jets + // layer encoding + const TTBV hitPattern((int)iterL1Track->hitPattern(), setup->numLayers()); + const double zT = iterL1Track->z0() + setup->chosenRofZ() * iterL1Track->tanL(); + const vector& le = layerEncoding->layerEncoding(zT); + vector layers; + layers.reserve(hitPattern.size()); + for (int layer : hitPattern.ids()) + layers.push_back(le[layer]); + m_trk_layers->push_back(layers); + } //end track loop } //end if SaveAllTracks @@ -1430,7 +1454,7 @@ void L1TrackNtupleMaker::analyze(const edm::Event& iEvent, const edm::EventSetup continue; } - std::vector >, TTStub > > + std::vector>, TTStub>> theStubRefs = MCTruthTTStubHandle->findTTStubRefs(tp_ptr); int nStubTP = (int)theStubRefs.size(); @@ -1491,7 +1515,7 @@ void L1TrackNtupleMaker::analyze(const edm::Event& iEvent, const edm::EventSetup // ---------------------------------------------------------------------------------------------- // look for L1 tracks matched to the tracking particle - std::vector > > matchedTracks = + std::vector>> matchedTracks = MCTruthTTTrackHandle->findTTTrackPtrs(tp_ptr); int nMatch = 0; @@ -1544,7 +1568,7 @@ void L1TrackNtupleMaker::analyze(const edm::Event& iEvent, const edm::EventSetup // + have >= L1Tk_minNStub stubs for it to be a valid match (only relevant is your track collection // e.g. stores 3-stub tracks but at plot level you require >= 4 stubs (--> tracklet case) - std::vector >, TTStub > > + std::vector>, TTStub>> stubRefs = matchedTracks.at(it)->getStubRefs(); int tmp_trk_nstub = stubRefs.size(); @@ -1646,7 +1670,7 @@ void L1TrackNtupleMaker::analyze(const edm::Event& iEvent, const edm::EventSetup tmp_matchtrk_dhits = 0; tmp_matchtrk_lhits = 0; - std::vector >, TTStub > > + std::vector>, TTStub>> stubRefs = matchedTracks.at(i_track)->getStubRefs(); int tmp_nstub = stubRefs.size(); diff --git a/L1Trigger/TrackFindingTracklet/test/L1TrackNtupleMaker_cfg.py b/L1Trigger/TrackFindingTracklet/test/L1TrackNtupleMaker_cfg.py index 65a118b51e5a0..dabf98787df05 100644 --- a/L1Trigger/TrackFindingTracklet/test/L1TrackNtupleMaker_cfg.py +++ b/L1Trigger/TrackFindingTracklet/test/L1TrackNtupleMaker_cfg.py @@ -117,8 +117,6 @@ ############################################################ process.load('L1Trigger.TrackTrigger.TrackTrigger_cff') -process.load('L1Trigger.TrackerTFP.ProducerES_cff') -process.load('L1Trigger.TrackerTFP.ProducerLayerEncoding_cff') # remake stubs? #from L1Trigger.TrackTrigger.TTStubAlgorithmRegister_cfi import * @@ -131,19 +129,19 @@ #process.TTClusterStubTruth = cms.Path(process.TrackTriggerAssociatorClustersStubs) +# load code that associates stubs with mctruth +process.load( 'SimTracker.TrackTriggerAssociation.StubAssociator_cff' ) # DTC emulation -process.load('L1Trigger.TrackerDTC.ProducerED_cff') +process.load('L1Trigger.TrackerDTC.DTC_cff') # load code that analyzes DTCStubs -#process.load('L1Trigger.TrackerDTC.Analyzer_cff') +process.load('L1Trigger.TrackerDTC.Analyzer_cff') # modify default cuts #process.TrackTriggerSetup.FrontEnd.BendCut = 5.0 #process.TrackTriggerSetup.Hybrid.MinPt = 1.0 -process.dtc = cms.Path(process.TrackerDTCProducer)#*process.TrackerDTCAnalyzer) -# Throw error if reading MC produced with different stub window sizes. -process.TrackerDTCProducer.CheckHistory = True +process.dtc = cms.Path(process.StubAssociator + process.ProducerDTC + process.AnalyzerDTC) ############################################################ # L1 tracking @@ -153,6 +151,8 @@ # HYBRID: prompt tracking if (L1TRKALGO == 'HYBRID'): + process.TrackTriggerSetup.GeometricProcessor.ChosenRofZ = 50.0 + process.TrackTriggerSetup.TrackFinding.MaxEta = 2.4 process.TTTracksEmulation = cms.Path(process.L1THybridTracks) process.TTTracksEmulationWithTruth = cms.Path(process.L1THybridTracksWithAssociators) NHELIXPAR = 4 @@ -172,18 +172,18 @@ # HYBRID_NEWKF: prompt tracking or reduced elif (L1TRKALGO == 'HYBRID_NEWKF' or L1TRKALGO == 'HYBRID_REDUCED'): process.load( 'L1Trigger.TrackFindingTracklet.Producer_cff' ) + process.load( 'L1Trigger.TrackFindingTracklet.Analyzer_cff' ) NHELIXPAR = 4 - L1TRK_NAME = process.TrackFindingTrackletProducer_params.LabelKFout.value() - L1TRK_LABEL = process.TrackFindingTrackletProducer_params.BranchAcceptedTTTracks.value() + L1TRK_NAME = process.TrackFindingTrackletAnalyzer_params.OutputLabelTFP.value() + L1TRK_LABEL = process.TrackFindingTrackletProducer_params.BranchTTTracks.value() L1TRUTH_NAME = "TTTrackAssociatorFromPixelDigis" process.TTTrackAssociatorFromPixelDigis.TTTracks = cms.VInputTag( cms.InputTag(L1TRK_NAME, L1TRK_LABEL) ) - process.HybridNewKF = cms.Sequence(process.L1THybridTracks + process.TrackFindingTrackletProducerTBout + process.TrackFindingTrackletProducerDRin + process.TrackFindingTrackletProducerDR + process.TrackFindingTrackletProducerKFin + process.TrackFindingTrackletProducerKF + process.TrackFindingTrackletProducerKFout) + process.HybridNewKF = cms.Sequence(process.L1THybridTracks + process.ProducerTM + process.ProducerDR + process.ProducerKF + process.ProducerTQ + process.ProducerTFP) process.TTTracksEmulation = cms.Path(process.HybridNewKF) #process.TTTracksEmulationWithTruth = cms.Path(process.HybridNewKF + process.TrackTriggerAssociatorTracks) # Optionally include code producing performance plots & end-of-job summary. process.load( 'SimTracker.TrackTriggerAssociation.StubAssociator_cff' ) - process.load( 'L1Trigger.TrackFindingTracklet.Analyzer_cff' ) - process.TTTracksEmulationWithTruth = cms.Path(process.HybridNewKF + process.TrackTriggerAssociatorTracks + process.StubAssociator + process.TrackFindingTrackletAnalyzerTracklet + process.TrackFindingTrackletAnalyzerTBout + process.TrackFindingTrackletAnalyzerDRin + process.TrackFindingTrackletAnalyzerDR + process.TrackFindingTrackletAnalyzerKFin + process.TrackFindingTrackletAnalyzerKF + process.TrackFindingTrackletAnalyzerKFout) + process.TTTracksEmulationWithTruth = cms.Path(process.HybridNewKF + process.TrackTriggerAssociatorTracks + process.StubAssociator + process.AnalyzerTracklet + process.AnalyzerTM + process.AnalyzerDR + process.AnalyzerKF ) from L1Trigger.TrackFindingTracklet.Customize_cff import * if (L1TRKALGO == 'HYBRID_NEWKF'): fwConfig( process ) diff --git a/L1Trigger/TrackFindingTracklet/test/ProducerIRin.cc b/L1Trigger/TrackFindingTracklet/test/ProducerIRin.cc index 7ca0960306522..d695c7a28668a 100644 --- a/L1Trigger/TrackFindingTracklet/test/ProducerIRin.cc +++ b/L1Trigger/TrackFindingTracklet/test/ProducerIRin.cc @@ -60,7 +60,7 @@ namespace trklet { ProducerIRin::ProducerIRin(const ParameterSet& iConfig) : iConfig_(iConfig) { const InputTag& inputTag = iConfig.getParameter("InputTagDTC"); - const string& branchStubs = iConfig.getParameter("BranchAcceptedStubs"); + const string& branchStubs = iConfig.getParameter("BranchStubsAccepted"); // book in- and output ED products edGetTokenTTDTC_ = consumes(inputTag); edPutTokenStubs_ = produces(branchStubs); @@ -74,11 +74,6 @@ namespace trklet { void ProducerIRin::beginRun(const Run& iRun, const EventSetup& iSetup) { // helper class to store configurations setup_ = &iSetup.getData(esGetTokenSetup_); - if (!setup_->configurationSupported()) - return; - // check process history if desired - if (iConfig_.getParameter("CheckHistory")) - setup_->checkHistory(iRun.processHistory()); channelAssignment_ = const_cast(&iSetup.getData(esGetTokenChannelAssignment_)); // map of used tfp channels channelEncoding_ = channelAssignment_->channelEncoding(); @@ -88,15 +83,13 @@ namespace trklet { // empty IRin product StreamsStub streamStubs; // read in hybrid track finding product and produce KFin product - if (setup_->configurationSupported()) { - Handle handleTTDTC; - iEvent.getByToken(edGetTokenTTDTC_, handleTTDTC); - const int numChannel = channelEncoding_.size(); - streamStubs.reserve(numChannel); - for (int tfpRegion : handleTTDTC->tfpRegions()) - for (int tfpChannel : channelEncoding_) - streamStubs.emplace_back(handleTTDTC->stream(tfpRegion, tfpChannel)); - } + Handle handleTTDTC; + iEvent.getByToken(edGetTokenTTDTC_, handleTTDTC); + const int numChannel = channelEncoding_.size(); + streamStubs.reserve(numChannel); + for (int tfpRegion : handleTTDTC->tfpRegions()) + for (int tfpChannel : channelEncoding_) + streamStubs.emplace_back(handleTTDTC->stream(tfpRegion, tfpChannel)); // store products iEvent.emplace(edPutTokenStubs_, std::move(streamStubs)); } diff --git a/L1Trigger/TrackFindingTracklet/test/demonstrator_cfg.py b/L1Trigger/TrackFindingTracklet/test/demonstrator_cfg.py index 77406528debd0..86dea01ab5bba 100644 --- a/L1Trigger/TrackFindingTracklet/test/demonstrator_cfg.py +++ b/L1Trigger/TrackFindingTracklet/test/demonstrator_cfg.py @@ -4,8 +4,8 @@ process = cms.Process( "Demo" ) process.load( 'FWCore.MessageService.MessageLogger_cfi' ) process.load( 'Configuration.EventContent.EventContent_cff' ) -process.load( 'Configuration.Geometry.GeometryExtendedRun4D88Reco_cff' ) -process.load( 'Configuration.Geometry.GeometryExtendedRun4D88_cff' ) +process.load( 'Configuration.Geometry.GeometryExtended2026D98Reco_cff' ) +process.load( 'Configuration.Geometry.GeometryExtended2026D98_cff' ) process.load( 'Configuration.StandardSequences.MagneticField_cff' ) process.load( 'Configuration.StandardSequences.FrontierConditions_GlobalTag_cff' ) process.load( 'L1Trigger.TrackTrigger.TrackTrigger_cff' ) @@ -14,7 +14,7 @@ process.GlobalTag = GlobalTag(process.GlobalTag, 'auto:phase2_realistic', '') # load code that produces DTCStubs -process.load( 'L1Trigger.TrackerDTC.ProducerED_cff' ) +process.load( 'L1Trigger.TrackerDTC.DTC_cff' ) # L1 tracking => hybrid emulation process.load("L1Trigger.TrackFindingTracklet.L1HybridEmulationTracks_cff") # load code that fits hybrid tracks @@ -26,12 +26,12 @@ fwConfig( process ) # build schedule -process.tt = cms.Sequence ( process.TrackerDTCProducer +process.tt = cms.Sequence ( process.ProducerDTC + #+ process.ProducerIRin + process.L1THybridTracks - + process.TrackFindingTrackletProducerIRin - + process.TrackFindingTrackletProducerTBout - + process.TrackFindingTrackletProducerDRin - + process.TrackFindingTrackletProducerDR + + process.ProducerTM + + process.ProducerDR + #+ process.ProducerKF ) process.demo = cms.Path( process.tt + process.TrackerTFPDemonstrator ) process.schedule = cms.Schedule( process.demo ) @@ -40,9 +40,20 @@ import FWCore.ParameterSet.VarParsing as VarParsing options = VarParsing.VarParsing( 'analysis' ) # specify input MC -inputMC = ["/store/relval/CMSSW_12_6_0_pre4/RelValTTbar_14TeV/GEN-SIM-DIGI-RAW/PU_125X_mcRun4_realistic_v2_2026D88PU200-v1/2590000/00b3d04b-4c7b-4506-8d82-9538fb21ee19.root"] +Samples = [ +'/store/relval/CMSSW_14_0_0_pre2/RelValTTbar_14TeV/GEN-SIM-DIGI-RAW/PU_133X_mcRun4_realistic_v1_STD_2026D98_PU200_RV229-v1/2580000/0b2b0b0b-f312-48a8-9d46-ccbadc69bbfd.root', +'/store/relval/CMSSW_14_0_0_pre2/RelValTTbar_14TeV/GEN-SIM-DIGI-RAW/PU_133X_mcRun4_realistic_v1_STD_2026D98_PU200_RV229-v1/2580000/0c3cb20d-8556-450d-b4f0-e5c754818f74.root', +'/store/relval/CMSSW_14_0_0_pre2/RelValTTbar_14TeV/GEN-SIM-DIGI-RAW/PU_133X_mcRun4_realistic_v1_STD_2026D98_PU200_RV229-v1/2580000/0eafa2b4-711a-43ec-be1c-7e564c294a9a.root', +'/store/relval/CMSSW_14_0_0_pre2/RelValTTbar_14TeV/GEN-SIM-DIGI-RAW/PU_133X_mcRun4_realistic_v1_STD_2026D98_PU200_RV229-v1/2580000/1450b1bb-171e-495e-a767-68e2796d95c2.root', +'/store/relval/CMSSW_14_0_0_pre2/RelValTTbar_14TeV/GEN-SIM-DIGI-RAW/PU_133X_mcRun4_realistic_v1_STD_2026D98_PU200_RV229-v1/2580000/15498564-9cf0-4219-aab7-f97b3484b122.root', +'/store/relval/CMSSW_14_0_0_pre2/RelValTTbar_14TeV/GEN-SIM-DIGI-RAW/PU_133X_mcRun4_realistic_v1_STD_2026D98_PU200_RV229-v1/2580000/1838a806-316b-4f53-9d22-5b3856019623.root', +'/store/relval/CMSSW_14_0_0_pre2/RelValTTbar_14TeV/GEN-SIM-DIGI-RAW/PU_133X_mcRun4_realistic_v1_STD_2026D98_PU200_RV229-v1/2580000/1a34eb87-b9a3-47fb-b945-57e6f775fcac.root', +'/store/relval/CMSSW_14_0_0_pre2/RelValTTbar_14TeV/GEN-SIM-DIGI-RAW/PU_133X_mcRun4_realistic_v1_STD_2026D98_PU200_RV229-v1/2580000/1add5b2e-19cb-4581-956d-271907d03b72.root', +'/store/relval/CMSSW_14_0_0_pre2/RelValTTbar_14TeV/GEN-SIM-DIGI-RAW/PU_133X_mcRun4_realistic_v1_STD_2026D98_PU200_RV229-v1/2580000/1bed1837-ef65-4e07-a2ac-13c705b20fc1.root', +'/store/relval/CMSSW_14_0_0_pre2/RelValTTbar_14TeV/GEN-SIM-DIGI-RAW/PU_133X_mcRun4_realistic_v1_STD_2026D98_PU200_RV229-v1/2580000/1d057884-72bd-4353-8375-ec4616c00a33.root' +] -options.register( 'inputMC', inputMC, VarParsing.VarParsing.multiplicity.singleton, VarParsing.VarParsing.varType.string, "Files to be processed" ) +options.register( 'inputMC', Samples, VarParsing.VarParsing.multiplicity.singleton, VarParsing.VarParsing.varType.string, "Files to be processed" ) # specify number of events to process. options.register( 'Events',100,VarParsing.VarParsing.multiplicity.singleton, VarParsing.VarParsing.varType.int, "Number of Events to analyze" ) options.parseArguments() @@ -52,7 +63,7 @@ process.source = cms.Source( "PoolSource", fileNames = cms.untracked.vstring( options.inputMC ), - #skipEvents = cms.untracked.uint32( 1 ), + #skipEvents = cms.untracked.uint32( 301 ), secondaryFileNames = cms.untracked.vstring(), duplicateCheckMode = cms.untracked.string( 'noDuplicateCheck' ) ) diff --git a/L1Trigger/TrackTrigger/interface/L1TrackQuality.h b/L1Trigger/TrackTrigger/interface/L1TrackQuality.h deleted file mode 100644 index 1107ed3e74fbb..0000000000000 --- a/L1Trigger/TrackTrigger/interface/L1TrackQuality.h +++ /dev/null @@ -1,73 +0,0 @@ -/* -Track Quality Header file -C.Brown 28/07/20 -*/ - -#ifndef L1Trigger_TrackTrigger_interface_L1TrackQuality_h -#define L1Trigger_TrackTrigger_interface_L1TrackQuality_h - -#include -#include -#include -#include -#include - -#include "FWCore/Framework/interface/Event.h" -#include "FWCore/Framework/interface/EventSetup.h" -#include "FWCore/Framework/interface/Frameworkfwd.h" -#include "FWCore/Framework/interface/MakerMacros.h" -#include "FWCore/ParameterSet/interface/ParameterSet.h" -#include "DataFormats/L1TrackTrigger/interface/TTTrack.h" -#include "DataFormats/L1TrackTrigger/interface/TTTrack_TrackWord.h" -#include "DataFormats/L1TrackTrigger/interface/TTTypes.h" -#include - -#include "conifer.h" -#include "ap_fixed.h" - -class L1TrackQuality { -public: - //Default Constructor - L1TrackQuality(); - - L1TrackQuality(const edm::ParameterSet& qualityParams); - - //Default Destructor - ~L1TrackQuality() = default; - - // Controls the conversion between TTTrack features and ML model training features - std::vector featureTransform(TTTrack& aTrack, - std::vector const& featureNames); - - // Passed by reference a track without MVA filled, method fills the track's MVA field - void setL1TrackQuality(TTTrack& aTrack); - // Function to run the BDT in isolation allowing a feature vector in the ap_fixed datatype to be passed - // and a single output to be returned which is then used to fill the bits in the Track Word for situations - // where a TTTrack datatype is unavailable to be passed to the track quality - float runEmulatedTQ(std::vector> inputFeatures); - - void setModel(edm::FileInPath const& model, std::vector const& featureNames); - - void setBonusFeatures(std::vector bonusFeatures); - - // TQ MVA bin conversions - static constexpr double invSigmoid(double value) { return -log(1. / value - 1.); } - static constexpr std::array getTqMVAPreSigBins() { - return {{-16., - invSigmoid(TTTrack_TrackWord::tqMVABins[1]), - invSigmoid(TTTrack_TrackWord::tqMVABins[2]), - invSigmoid(TTTrack_TrackWord::tqMVABins[3]), - invSigmoid(TTTrack_TrackWord::tqMVABins[4]), - invSigmoid(TTTrack_TrackWord::tqMVABins[5]), - invSigmoid(TTTrack_TrackWord::tqMVABins[6]), - invSigmoid(TTTrack_TrackWord::tqMVABins[7])}}; - } - -private: - // Private Member Data - edm::FileInPath model_; - std::vector featureNames_; - bool useHPH_; - std::vector bonusFeatures_; -}; -#endif diff --git a/L1Trigger/TrackTrigger/interface/SensorModule.h b/L1Trigger/TrackTrigger/interface/SensorModule.h index a10c701f24344..a717ab3b1b26b 100644 --- a/L1Trigger/TrackTrigger/interface/SensorModule.h +++ b/L1Trigger/TrackTrigger/interface/SensorModule.h @@ -25,6 +25,8 @@ namespace tt { bool side() const { return side_; } // barrel or endcap bool barrel() const { return barrel_; } + // tilted barrel or flat barrel + bool tilted() const { return tilted_; } // Pixel-Strip or 2Strip module bool psModule() const { return psModule_; } // main sensor inside or outside @@ -69,6 +71,10 @@ namespace tt { int windowSize() const { return windowSize_; } // double tiltCorrection(double cot) const { return std::abs(tiltCorrectionSlope_ * cot) + tiltCorrectionIntercept_; } + // + double dPhi(double inv2R) const { return dPhi_ + (dR_ + scattering_) * abs(inv2R); } + // + double dZ() const { return dZ_; } unsigned int ringId(const Setup* setup) const; @@ -84,6 +90,8 @@ namespace tt { bool side_; // barrel or endcap bool barrel_; + // tilted barrel or flat barrel + bool tilted_; // Pixel-Strip or 2Strip module bool psModule_; // main sensor inside or outside @@ -132,6 +140,14 @@ namespace tt { double tiltCorrectionSlope_; // tilt correction parameter used to project r to z uncertainty double tiltCorrectionIntercept_; + // + double scattering_; + // + double dR_; + // + double dPhi_; + // + double dZ_; }; } // namespace tt diff --git a/L1Trigger/TrackTrigger/interface/Setup.h b/L1Trigger/TrackTrigger/interface/Setup.h index 07fe6cd88eb5a..2e2724e37c8fe 100644 --- a/L1Trigger/TrackTrigger/interface/Setup.h +++ b/L1Trigger/TrackTrigger/interface/Setup.h @@ -4,8 +4,6 @@ #include "FWCore/Framework/interface/data_default_record_trait.h" #include "FWCore/ParameterSet/interface/ParameterSet.h" #include "FWCore/ParameterSet/interface/Registry.h" -#include "DataFormats/Provenance/interface/ProcessHistory.h" -#include "DataFormats/Provenance/interface/ParameterSetID.h" #include "DataFormats/DetId/interface/DetId.h" #include "DataFormats/GeometryVector/interface/GlobalPoint.h" #include "DataFormats/Math/interface/deltaPhi.h" @@ -13,10 +11,7 @@ #include "DataFormats/SiStripDetId/interface/StripSubdetector.h" #include "Geometry/CommonTopologies/interface/PixelGeomDetUnit.h" #include "Geometry/TrackerGeometryBuilder/interface/TrackerGeometry.h" -#include "DetectorDescription/Core/interface/DDCompactView.h" -#include "DetectorDescription/DDCMS/interface/DDCompactView.h" #include "L1Trigger/TrackTrigger/interface/TTStubAlgorithm_official.h" -#include "MagneticField/Engine/interface/MagneticField.h" #include "CondFormats/SiPhase2TrackerObjects/interface/TrackerDetToDTCELinkCablingMap.h" #include "SimTracker/Common/interface/TrackingParticleSelector.h" @@ -45,21 +40,13 @@ namespace tt { public: Setup() {} Setup(const edm::ParameterSet& iConfig, - const MagneticField& magneticField, const TrackerGeometry& trackerGeometry, const TrackerTopology& trackerTopology, const TrackerDetToDTCELinkCablingMap& cablingMap, const StubAlgorithmOfficial& stubAlgorithm, - const edm::ParameterSet& pSetStubAlgorithm, - const edm::ParameterSet& pSetGeometryConfiguration, - const edm::ParameterSetID& pSetIdTTStubAlgorithm, - const edm::ParameterSetID& pSetIdGeometryConfiguration); + const edm::ParameterSet& pSetStubAlgorithm); ~Setup() {} - // true if tracker geometry and magnetic field supported - bool configurationSupported() const { return configurationSupported_; } - // checks current configuration vs input sample configuration - void checkHistory(const edm::ProcessHistory& processHistory) const; // converts tk layout id into dtc id int dtcId(int tklId) const; // converts dtci id into tk layout id @@ -76,6 +63,8 @@ namespace tt { int slot(int dtcId) const; // sensor module for det id SensorModule* sensorModule(const DetId& detId) const; + // sensor module for ttStubRef + SensorModule* sensorModule(const TTStubRef& ttStubRef) const; // TrackerGeometry const TrackerGeometry* trackerGeometry() const { return trackerGeometry_; } // TrackerTopology @@ -88,12 +77,6 @@ namespace tt { GlobalPoint stubPos(bool hybrid, const tt::FrameStub& frame, int region) const; // empty trackerDTC EDProduct TTDTC ttDTC() const { return TTDTC(numRegions_, numOverlappingRegions_, numDTCsPerRegion_); } - // checks if stub collection is considered forming a reconstructable track - bool reconstructable(const std::vector& ttStubRefs) const; - // checks if tracking particle is selected for efficiency measurements - bool useForAlgEff(const TrackingParticle& tp) const; - // checks if tracking particle is selected for fake and duplicate rate measurements - bool useForReconstructable(const TrackingParticle& tp) const { return tpSelectorLoose_(tp); } // stub layer id (barrel: 1 - 6, endcap: 11 - 15) int layerId(const TTStubRef& ttStubRef) const; // return tracklet layerId (barrel: [0-5], endcap: [6-10]) for given TTStubRef @@ -106,6 +89,8 @@ namespace tt { bool psModule(const TTStubRef& ttStubRef) const; // return sensor moduel type SensorModule::Type type(const TTStubRef& ttStubRef) const; + // checks if stub collection is considered forming a reconstructable track + bool reconstructable(const std::vector& ttStubRefs) const; // TTBV layerMap(const std::vector& ints) const; // @@ -117,13 +102,23 @@ namespace tt { // stub projected phi uncertainty double dPhi(const TTStubRef& ttStubRef, double inv2R) const; // stub projected z uncertainty - double dZ(const TTStubRef& ttStubRef, double cot) const; + double dZ(const TTStubRef& ttStubRef) const; // stub projected chi2phi wheight double v0(const TTStubRef& ttStubRef, double inv2R) const; // stub projected chi2z wheight double v1(const TTStubRef& ttStubRef, double cot) const; // const std::vector& sensorModules() const { return sensorModules_; } + // + TTBV module(double r, double z) const; + // + bool ps(const TTBV& module) const { return module[gpPosPS_]; } + // + bool barrel(const TTBV& module) const { return module[gpPosBarrel_]; } + // + bool tilted(const TTBV& module) const { return module[gpPosTilted_]; } + // stub projected phi uncertainty for given module type, stub radius and track curvature + double dPhi(const TTBV& module, double r, double inv2R) const; // Firmware specific Parameter @@ -150,29 +145,50 @@ namespace tt { // smallest address width of an BRAM18 configured as broadest simple dual port memory int widthAddrBRAM18() const { return widthAddrBRAM18_; } // number of frames betwen 2 resets of 18 BX packets - int numFrames() const { return numFrames_; } + int numFramesHigh() const { return numFramesHigh_; } + // number of frames betwen 2 resets of 18 BX packets + int numFramesLow() const { return numFramesLow_; } // number of frames needed per reset int numFramesInfra() const { return numFramesInfra_; } // number of valid frames per 18 BX packet - int numFramesIO() const { return numFramesIO_; } + int numFramesIOHigh() const { return numFramesIOHigh_; } + // number of valid frames per 18 BX packet + int numFramesIOLow() const { return numFramesIOLow_; } // number of valid frames per 8 BX packet int numFramesFE() const { return numFramesFE_; } - // maximum representable stub phi uncertainty - double maxdPhi() const { return maxdPhi_; } - // maximum representable stub z uncertainty - double maxdZ() const { return maxdZ_; } - // barrel layer limit z value to partition into tilted and untilted region - double tiltedLayerLimitZ(int layer) const { return tiltedLayerLimitsZ_.at(layer); } - // endcap disk limit r value to partition into PS and 2S region - double psDiskLimitR(int layer) const { return psDiskLimitsR_.at(layer); } + + // Tracker specific Parameter + // strip pitch of outer tracker sensors in cm - double pitch2S() const { return pitch2S_; } + double pitchRow2S() const { return pitchRow2S_; } // pixel pitch of outer tracker sensors in cm - double pitchPS() const { return pitchPS_; } + double pitchRowPS() const { return pitchRowPS_; } // strip length of outer tracker sensors in cm - double length2S() const { return length2S_; } + double pitchCol2S() const { return pitchCol2S_; } // pixel length of outer tracker sensors in cm - double lengthPS() const { return lengthPS_; } + double pitchColPS() const { return pitchColPS_; } + // BField used in fw in T + double bField() const { return bField_; } + // outer radius of outer tracker in cm + double outerRadius() const { return outerRadius_; } + // inner radius of outer tracker in cm + double innerRadius() const { return innerRadius_; } + // half length of outer tracker in cm + double halfLength() const { return halfLength_; } + // max strip/pixel length of outer tracker sensors in cm + double maxPitchCol() const { return maxPitchCol_; } + // In tilted barrel, grad*|z|/r + int approximates |cosTilt| + |sinTilt * cotTheta| + double tiltApproxSlope() const { return tiltApproxSlope_; } + // In tilted barrel, grad*|z|/r + int approximates |cosTilt| + |sinTilt * cotTheta| + double tiltApproxIntercept() const { return tiltApproxIntercept_; } + // In tilted barrel, constant assumed stub radial uncertainty * sqrt(12) in cm + double tiltUncertaintyR() const { return tiltUncertaintyR_; } + // scattering term used to add stub phi uncertainty depending on assumed track inv2R + double scattering() const { return scattering_; } + // barrel layer limit z value to partition into tilted and untilted region + double tiltedLayerLimitZ(int layer) const { return tiltedLayerLimitsZ_.at(layer); } + // endcap disk limit r value to partition into PS and 2S region + double psDiskLimitR(int layer) const { return psDiskLimitsR_.at(layer); } // Common track finding parameter @@ -182,37 +198,23 @@ namespace tt { double invPtToDphi() const { return invPtToDphi_; } // region size in rad double baseRegion() const { return baseRegion_; } - // pt cut - double tpMinPt() const { return tpMinPt_; } - // TP eta cut - double tpMaxEta() const { return tpMaxEta_; } - // TP cut on vertex pos r in cm - double tpMaxVertR() const { return tpMaxVertR_; } - // TP cut on vertex pos z in cm - double tpMaxVertZ() const { return tpMaxVertZ_; } - // TP cut on impact parameter in cm - double tpMaxD0() const { return tpMaxD0_; } - // required number of associated layers to a TP to consider it reconstruct-able - int tpMinLayers() const { return tpMinLayers_; } - // required number of associated ps layers to a TP to consider it reconstruct-able - int tpMinLayersPS() const { return tpMinLayersPS_; } - // max number of unassociated 2S stubs allowed to still associate TTTrack with TP - int tpMaxBadStubs2S() const { return tpMaxBadStubs2S_; } - // max number of unassociated PS stubs allowed to still associate TTTrack with TP - int tpMaxBadStubsPS() const { return tpMaxBadStubsPS_; } - // BField used in fw in T - double bField() const { return bField_; } - - // TMTT specific parameter - + // max cot(theta) of found tracks + double maxCot() const { return maxCot_; } // cut on stub and TP pt, also defines region overlap shape in GeV double minPt() const { return minPt_; } + // cut on candidate pt + double minPtCand() const { return minPtCand_; } // cut on stub eta double maxEta() const { return maxEta_; } + // constraints track reconstruction phase space + double maxD0() const { return maxD0_; } // critical radius defining region overlap shape in cm double chosenRofPhi() const { return chosenRofPhi_; } - // number of detector layers a reconstructbale particle may cross + // TMTT: number of detector layers a reconstructbale particle may cross; Hybrid: max number of layers connected to one DTC int numLayers() const { return numLayers_; } + + // TMTT specific parameter + // number of bits used for stub r - ChosenRofPhi int tmttWidthR() const { return tmttWidthR_; } // number of bits used for stub phi w.r.t. phi sector centre @@ -237,35 +239,11 @@ namespace tt { double tmttBasePhiT() const { return tmttBasePhiT_; } // number of padded 0s in output data format int tmttNumUnusedBits() const { return tmttNumUnusedBits_; } - // outer radius of outer tracker in cm - double outerRadius() const { return outerRadius_; } - // inner radius of outer tracker in cm - double innerRadius() const { return innerRadius_; } - // half length of outer tracker in cm - double halfLength() const { return halfLength_; } - // max strip/pixel length of outer tracker sensors in cm - double maxLength() const { return maxLength_; } - // In tilted barrel, grad*|z|/r + int approximates |cosTilt| + |sinTilt * cotTheta| - double tiltApproxSlope() const { return tiltApproxSlope_; } - // In tilted barrel, grad*|z|/r + int approximates |cosTilt| + |sinTilt * cotTheta| - double tiltApproxIntercept() const { return tiltApproxIntercept_; } - // In tilted barrel, constant assumed stub radial uncertainty * sqrt(12) in cm - double tiltUncertaintyR() const { return tiltUncertaintyR_; } - // scattering term used to add stub phi uncertainty depending on assumed track inv2R - double scattering() const { return scattering_; } // Hybrid specific parameter - // cut on stub pt in GeV, also defines region overlap shape - double hybridMinPtStub() const { return hybridMinPtStub_; } - // cut on andidate pt in GeV - double hybridMinPtCand() const { return hybridMinPtCand_; } - // cut on stub eta - double hybridMaxEta() const { return hybridMaxEta_; } - // critical radius defining region overlap shape in cm - double hybridChosenRofPhi() const { return hybridChosenRofPhi_; } - // max number of detector layer connected to one DTC - int hybridNumLayers() const { return hybridNumLayers_; } + // max number of layer connected to one DTC + double hybridNumLayers() const { return hybridNumLayers_; } // number of bits used for stub r w.r.t layer/disk centre for module types (barrelPS, barrel2S, diskPS, disk2S) int hybridWidthR(SensorModule::Type type) const { return hybridWidthsR_.at(type); } // number of bits used for stub z w.r.t layer/disk centre for module types (barrelPS, barrel2S, diskPS, disk2S) @@ -280,10 +258,13 @@ namespace tt { int hybridWidthLayerId() const { return hybridWidthLayerId_; } // precision or r in cm for (barrelPS, barrel2S, diskPS, disk2S) double hybridBaseR(SensorModule::Type type) const { return hybridBasesR_.at(type); } + double hybridBaseR() const { return hybridBaseR_; } // precision or phi in rad for (barrelPS, barrel2S, diskPS, disk2S) double hybridBasePhi(SensorModule::Type type) const { return hybridBasesPhi_.at(type); } + double hybridBasePhi() const { return hybridBasePhi_; } // precision or z in cm for (barrelPS, barrel2S, diskPS, disk2S) double hybridBaseZ(SensorModule::Type type) const { return hybridBasesZ_.at(type); } + double hybridBaseZ() const { return hybridBaseZ_; } // precision or alpha in pitch units for (barrelPS, barrel2S, diskPS, disk2S) double hybridBaseAlpha(SensorModule::Type type) const { return hybridBasesAlpha_.at(type); } // number of padded 0s in output data format for (barrelPS, barrel2S, diskPS, disk2S) @@ -300,6 +281,8 @@ namespace tt { double hybridRangePhi() const { return hybridRangePhi_; } // range of stub r in cm double hybridRangeR() const { return hybridRangesR_[SensorModule::DiskPS]; } + // biggest barrel stub z position after TrackBuilder in cm + double tbBarrelHalfLength() const { return tbBarrelHalfLength_; } // smallest stub radius after TrackBuilder in cm double tbInnerRadius() const { return tbInnerRadius_; } // center radius of outer tracker endcap 2S diks strips @@ -325,6 +308,8 @@ namespace tt { double baseWindowSize() const { return baseWindowSize_; } // index = encoded bend, value = decoded bend for given window size and module type const std::vector& encodingBend(int windowSize, bool psModule) const; + //getBendCut + const StubAlgorithmOfficial* stubAlgorithm() const { return stubAlgorithm_; } // Parameter specifying front-end @@ -398,8 +383,8 @@ namespace tt { // number of bist used for phi0 int tfpWidthPhi0() const { return tfpWidthPhi0_; } - // umber of bist used for inv2R - int tfpWidthInv2R() const { return tfpWidthInv2R_; } + // umber of bist used for invR + int tfpWidthInvR() const { return tfpWidthInvR_; } // number of bist used for cot(theta) int tfpWidthCot() const { return tfpWidthCot_; } // number of bist used for z0 @@ -410,28 +395,23 @@ namespace tt { // Parameter specifying GeometricProcessor // number of phi sectors in a processing nonant used in hough transform - int numSectorsPhi() const { return numSectorsPhi_; } + int gpNumBinsPhiT() const { return gpNumBinsPhiT_; } // number of eta sectors used in hough transform - int numSectorsEta() const { return numSectorsEta_; } + int gpNumBinsZT() const { return gpNumBinsZT_; } // # critical radius defining r-z sector shape in cm double chosenRofZ() const { return chosenRofZ_; } // fifo depth in stub router firmware int gpDepthMemory() const { return gpDepthMemory_; } - // defining r-z sector shape - double boundarieEta(int eta) const { return boundariesEta_.at(eta); } - std::vector boundarieEta() const { return boundariesEta_; } + // + int gpWidthModule() const { return gpWidthModule_; } // phi sector size in rad double baseSector() const { return baseSector_; } - // cut on zT - double maxZT() const { return maxZT_; } - // cut on stub cot theta - double maxCot() const { return maxCot_; } // total number of sectors int numSectors() const { return numSectors_; } - // cot(theta) of given eta sector - double sectorCot(int eta) const { return sectorCots_.at(eta); } // - double neededRangeChiZ() const { return neededRangeChiZ_; } + double maxRphi() const { return maxRphi_; } + // + double maxRz() const { return maxRz_; } // Parameter specifying HoughTransform @@ -444,85 +424,73 @@ namespace tt { // internal fifo depth int htDepthMemory() const { return htDepthMemory_; } - // Parameter specifying MiniHoughTransform + // Parameter specifying Track Builder // number of finer inv2R bins inside HT bin - int mhtNumBinsInv2R() const { return mhtNumBinsInv2R_; } + int ctbNumBinsInv2R() const { return ctbNumBinsInv2R_; } // number of finer phiT bins inside HT bin - int mhtNumBinsPhiT() const { return mhtNumBinsPhiT_; } - // number of dynamic load balancing steps - int mhtNumDLBs() const { return mhtNumDLBs_; } - // number of units per dynamic load balancing step - int mhtNumDLBNodes() const { return mhtNumDLBNodes_; } - // number of inputs per dynamic load balancing unit - int mhtNumDLBChannel() const { return mhtNumDLBChannel_; } - // required number of stub layers to form a candidate - int mhtMinLayers() const { return mhtMinLayers_; } - // number of mht cells - int mhtNumCells() const { return mhtNumCells_; } - - // Parameter specifying ZHoughTransform - - //number of used zT bins - int zhtNumBinsZT() const { return zhtNumBinsZT_; } - // number of used cot bins - int zhtNumBinsCot() const { return zhtNumBinsCot_; } - // number of stages - int zhtNumStages() const { return zhtNumStages_; } + int ctbNumBinsPhiT() const { return ctbNumBinsPhiT_; } + // number of used z0 bins inside GP ZT bin + int ctbNumBinsCot() const { return ctbNumBinsCot_; } + //number of used zT bins inside GP ZT bin + int ctbNumBinsZT() const { return ctbNumBinsZT_; } // required number of stub layers to form a candidate - int zhtMinLayers() const { return zhtMinLayers_; } + int ctbMinLayers() const { return ctbMinLayers_; } // max number of output tracks per node - int zhtMaxTracks() const { return zhtMaxTracks_; } + int ctbMaxTracks() const { return ctbMaxTracks_; } // cut on number of stub per layer for input candidates - int zhtMaxStubsPerLayer() const { return zhtMaxStubsPerLayer_; } - // number of zht cells - int zhtNumCells() const { return zhtNumCells_; } - - // Parameter specifying KalmanFilter Input Formatter - - // power of 2 multiplier of stub phi residual range - int kfinShiftRangePhi() const { return kfinShiftRangePhi_; } - // power of 2 multiplier of stub z residual range - int kfinShiftRangeZ() const { return kfinShiftRangeZ_; } + int ctbMaxStubs() const { return ctbMaxStubs_; } + // internal memory depth + int ctbDepthMemory() const { return ctbDepthMemory_; } // Parameter specifying KalmanFilter // number of kf worker int kfNumWorker() const { return kfNumWorker_; } + // max number of tracks a kf worker can process + int kfMaxTracks() const { return kfMaxTracks_; } // required number of stub layers to form a track int kfMinLayers() const { return kfMinLayers_; } + // required number of ps stub layers to form a track + int kfMinLayersPS() const { return kfMinLayersPS_; } // maximum number of layers added to a track int kfMaxLayers() const { return kfMaxLayers_; } + // + int kfMaxGaps() const { return kfMaxGaps_; } + // + int kfMaxSeedingLayer() const { return kfMaxSeedingLayer_; } + // + int kfNumSeedStubs() const { return kfNumSeedStubs_; } + // + double kfMinSeedDeltaR() const { return kfMinSeedDeltaR_; } // search window of each track parameter in initial uncertainties double kfRangeFactor() const { return kfRangeFactor_; } - // + // initial C00 is given by inv2R uncertainty squared times this power of 2 int kfShiftInitialC00() const { return kfShiftInitialC00_; } - // + // initial C11 is given by phiT uncertainty squared times this power of 2 int kfShiftInitialC11() const { return kfShiftInitialC11_; } - // + // initial C22 is given by cot uncertainty squared times this power of 2 int kfShiftInitialC22() const { return kfShiftInitialC22_; } - // + // initial C33 is given by zT uncertainty squared times this power of 2 int kfShiftInitialC33() const { return kfShiftInitialC33_; } - - // Parameter specifying KalmanFilter Output Formatter - // Conversion factor between dphi^2/weight and chi2rphi - int kfoutchi2rphiConv() const { return kfoutchi2rphiConv_; } - // Conversion factor between dz^2/weight and chi2rz - int kfoutchi2rzConv() const { return kfoutchi2rzConv_; } - // Fraction of total dphi and dz ranges to calculate v0 and v1 LUT for - int weightBinFraction() const { return weightBinFraction_; } - // Constant used in FW to prevent 32-bit int overflow - int dzTruncation() const { return dzTruncation_; } - // Constant used in FW to prevent 32-bit int overflow - int dphiTruncation() const { return dphiTruncation_; } + // + int kfShiftChi20() const { return kfShiftChi20_; } + // + int kfShiftChi21() const { return kfShiftChi21_; } + // + double kfCutChi2() const { return kfCutChi2_; } + // + int kfWidthChi2() const { return kfWidthChi2_; } // Parameter specifying DuplicateRemoval // internal memory depth int drDepthMemory() const { return drDepthMemory_; } - //getBendCut - const StubAlgorithmOfficial* stubAlgorithm() const { return stubAlgorithm_; } + // Parameter specifying TrackQuaility + + // number of output channel + int tqNumChannel() const { return tqNumChannel_; } private: // checks consitency between history and current configuration for a specific module @@ -532,10 +500,6 @@ namespace tt { const edm::ParameterSetID&) const; // dumps pSetHistory where incosistent lines with pSetProcess are highlighted std::string dumpDiff(const edm::ParameterSet& pSetHistory, const edm::ParameterSet& pSetProcess) const; - // check if bField is supported - void checkMagneticField(); - // check if geometry is supported - void checkGeometry(); // derive constants void calculateConstants(); // convert configuration of TTStubAlgorithm @@ -553,8 +517,6 @@ namespace tt { // configure TPSelector void configureTPSelector(); - // MagneticField - const MagneticField* magneticField_; // TrackerGeometry const TrackerGeometry* trackerGeometry_; // TrackerTopology @@ -565,59 +527,28 @@ namespace tt { const StubAlgorithmOfficial* stubAlgorithm_; // pSet of ttStub algorithm, used to identify bend window sizes of sensor modules const edm::ParameterSet* pSetSA_; - // pSet of geometry configuration, used to identify if geometry is supported - const edm::ParameterSet* pSetGC_; - // pset id of current TTStubAlgorithm - edm::ParameterSetID pSetIdTTStubAlgorithm_; - // pset id of current geometry configuration - edm::ParameterSetID pSetIdGeometryConfiguration_; - - // DD4hep - bool fromDD4hep_; - - // Parameter to check if configured Tracker Geometry is supported - edm::ParameterSet pSetSG_; - // label of ESProducer/ESSource - std::string sgXMLLabel_; - // compared path - std::string sgXMLPath_; - // compared filen ame - std::string sgXMLFile_; - // list of supported versions - std::vector sgXMLVersions_; - - // Parameter to check if Process History is consistent with process configuration - edm::ParameterSet pSetPH_; - // label of compared GeometryConfiguration - std::string phGeometryConfiguration_; - // label of compared TTStubAlgorithm - std::string phTTStubAlgorithm_; // Common track finding parameter edm::ParameterSet pSetTF_; // half lumi region size in cm double beamWindowZ_; - // required number of layers a found track has to have in common with a TP to consider it matched to it - int matchedLayers_; - // required number of ps layers a found track has to have in common with a TP to consider it matched to it - int matchedLayersPS_; - // allowed number of stubs a found track may have not in common with its matched TP - int unMatchedStubs_; - // allowed number of PS stubs a found track may have not in common with its matched TP - int unMatchedStubsPS_; - // scattering term used to add stub phi uncertainty depending on assumed track inv2R - double scattering_; - - // TMTT specific parameter - edm::ParameterSet pSetTMTT_; // cut on stub and TP pt, also defines region overlap shape in GeV double minPt_; + // cut on candidate pt + double minPtCand_; // cut on stub eta double maxEta_; + // in cm, constraints track reconstruction phase space + double maxD0_; // critical radius defining region overlap shape in cm double chosenRofPhi_; // number of detector layers a reconstructbale particle may cross int numLayers_; + // required number of stub layers to form a track + int minLayers_; + + // TMTT specific parameter + edm::ParameterSet pSetTMTT_; // number of bits used for stub r - ChosenRofPhi int tmttWidthR_; // number of bits used for stub phi w.r.t. phi sector centre @@ -627,15 +558,7 @@ namespace tt { // Hybrid specific parameter edm::ParameterSet pSetHybrid_; - // cut on stub pt in GeV, also defines region overlap shape - double hybridMinPtStub_; - // cut on andidate pt in GeV - double hybridMinPtCand_; - // cut on stub eta - double hybridMaxEta_; - // critical radius defining region overlap shape in cm - double hybridChosenRofPhi_; - // max number of detector layer connected to one DTC + // max number of layers connected to one DTC int hybridNumLayers_; // number of outer PS rings for disk 1, 2, 3, 4, 5 std::vector hybridNumRingsPS_; @@ -663,32 +586,13 @@ namespace tt { std::vector hybridDisk2SRsSet_; // range of stub phi in rad double hybridRangePhi_; + // biggest barrel stub z position after TrackBuilder in cm + double tbBarrelHalfLength_; // smallest stub radius after TrackBuilder in cm double tbInnerRadius_; // number of bits used for stub r w.r.t layer/disk centre for module types (barrelPS, barrel2S, diskPS, disk2S) after TrackBuilder std::vector tbWidthsR_; - // Parameter specifying TrackingParticle used for Efficiency measurements - edm::ParameterSet pSetTP_; - // pt cut - double tpMinPt_; - // eta cut - double tpMaxEta_; - // cut on vertex pos r in cm - double tpMaxVertR_; - // cut on vertex pos z in cm - double tpMaxVertZ_; - // cut on impact parameter in cm - double tpMaxD0_; - // required number of associated layers to a TP to consider it reconstruct-able - int tpMinLayers_; - // required number of associated ps layers to a TP to consider it reconstruct-able - int tpMinLayersPS_; - // max number of unassociated 2S stubs allowed to still associate TTTrack with TP - int tpMaxBadStubs2S_; - // max number of unassociated PS stubs allowed to still associate TTTrack with TP - int tpMaxBadStubsPS_; - // Firmware specific Parameter edm::ParameterSet pSetFW_; // width of the 'A' port of an DSP slice @@ -718,13 +622,18 @@ namespace tt { // LHC bunch crossing rate in MHz double freqLHC_; // processing Frequency of DTC & TFP in MHz, has to be integer multiple of FreqLHC - double freqBE_; + double freqBEHigh_; + // processing Frequency of DTC & TFP in MHz, has to be integer multiple of FreqLHC + double freqBELow_; // number of events collected in front-end int tmpFE_; // time multiplexed period of track finding processor int tmpTFP_; // speed of light used in FW in e8 m/s double speedOfLight_; + + // Tracker specific Parameter + edm::ParameterSet pSetOT_; // BField used in fw in T double bField_; // accepted BField difference between FW to EventSetup in T @@ -736,31 +645,35 @@ namespace tt { // half length of outer tracker in cm double halfLength_; // max strip/pixel pitch of outer tracker sensors in cm - double maxPitch_; + double maxPitchRow_; // max strip/pixel length of outer tracker sensors in cm - double maxLength_; + double maxPitchCol_; // approximated tilt correction parameter used to project r to z uncertainty double tiltApproxSlope_; // approximated tilt correction parameter used to project r to z uncertainty double tiltApproxIntercept_; // In tilted barrel, constant assumed stub radial uncertainty * sqrt(12) in cm double tiltUncertaintyR_; - // minimum representable stub phi uncertainty - double mindPhi_; - // maximum representable stub phi uncertainty - double maxdPhi_; - // minimum representable stub z uncertainty - double mindZ_; - // maximum representable stub z uncertainty - double maxdZ_; + // scattering term used to add stub phi uncertainty depending on assumed track inv2R + double scattering_; // strip pitch of outer tracker sensors in cm - double pitch2S_; + double pitchRow2S_; // pixel pitch of outer tracker sensors in cm - double pitchPS_; + double pitchRowPS_; // strip length of outer tracker sensors in cm - double length2S_; + double pitchCol2S_; // pixel length of outer tracker sensors in cm - double lengthPS_; + double pitchColPS_; + // barrel layer limit r value to partition into PS and 2S region + double limitPSBarrel_; + // barrel layer limit r value to partition into tilted and untilted region + std::vector limitsTiltedR_; + // barrel layer limit |z| value to partition into tilted and untilted region + std::vector limitsTiltedZ_; + // endcap disk limit |z| value to partition into PS and 2S region + std::vector limitsPSDiksZ_; + // endcap disk limit r value to partition into PS and 2S region + std::vector limitsPSDiksR_; // barrel layer limit |z| value to partition into tilted and untilted region std::vector tiltedLayerLimitsZ_; // endcap disk limit r value to partition into PS and 2S region @@ -824,13 +737,13 @@ namespace tt { // Parameter specifying TFP edm::ParameterSet pSetTFP_; - // number of bist used for phi0 + // number of bits used for phi0 int tfpWidthPhi0_; - // umber of bist used for qOverPt - int tfpWidthInv2R_; - // number of bist used for cot(theta) + // umber of bits used for qOverPt + int tfpWidthInvR_; + // number of bits used for cot(theta) int tfpWidthCot_; - // number of bist used for z0 + // number of bits used for z0 int tfpWidthZ0_; // number of output links int tfpNumChannel_; @@ -838,17 +751,21 @@ namespace tt { // Parameter specifying GeometricProcessor edm::ParameterSet pSetGP_; // number of phi sectors used in hough transform - int numSectorsPhi_; + int gpNumBinsPhiT_; // number of eta sectors used in hough transform - int numSectorsEta_; + int gpNumBinsZT_; // # critical radius defining r-z sector shape in cm double chosenRofZ_; - // range of stub z residual w.r.t. sector center which needs to be covered - double neededRangeChiZ_; // fifo depth in stub router firmware int gpDepthMemory_; - // defining r-z sector shape - std::vector boundariesEta_; + // + int gpWidthModule_; + // + int gpPosPS_; + // + int gpPosBarrel_; + // + int gpPosTilted_; // Parameter specifying HoughTransform edm::ParameterSet pSetHT_; @@ -861,91 +778,78 @@ namespace tt { // internal fifo depth int htDepthMemory_; - // Parameter specifying MiniHoughTransform - edm::ParameterSet pSetMHT_; + // Parameter specifying Clean Track Builder + edm::ParameterSet pSetCTB_; // number of finer inv2R bins inside HT bin - int mhtNumBinsInv2R_; + int ctbNumBinsInv2R_; // number of finer phiT bins inside HT bin - int mhtNumBinsPhiT_; - // number of dynamic load balancing steps - int mhtNumDLBs_; - // number of units per dynamic load balancing step - int mhtNumDLBNodes_; - // number of inputs per dynamic load balancing unit - int mhtNumDLBChannel_; + int ctbNumBinsPhiT_; + // number of used cot bins inside GP ZT bin + int ctbNumBinsCot_; + //number of used zT bins inside GP ZT bin + int ctbNumBinsZT_; // required number of stub layers to form a candidate - int mhtMinLayers_; - - // Parameter specifying ZHoughTransform - edm::ParameterSet pSetZHT_; - //number of used zT bins - int zhtNumBinsZT_; - // number of used cot bins - int zhtNumBinsCot_; - // number of stages - int zhtNumStages_; - // required number of stub layers to form a candidate - int zhtMinLayers_; + int ctbMinLayers_; // max number of output tracks per node - int zhtMaxTracks_; + int ctbMaxTracks_; // cut on number of stub per layer for input candidates - int zhtMaxStubsPerLayer_; - - // Parameter specifying KalmanFilter Input Formatter - edm::ParameterSet pSetKFin_; - // power of 2 multiplier of stub phi residual range - int kfinShiftRangePhi_; - // power of 2 multiplier of stub z residual range - int kfinShiftRangeZ_; + int ctbMaxStubs_; + // internal memory depth + int ctbDepthMemory_; // Parameter specifying KalmanFilter edm::ParameterSet pSetKF_; // number of kf worker int kfNumWorker_; + // max number of tracks a kf worker can process + int kfMaxTracks_; // required number of stub layers to form a track int kfMinLayers_; + // required number of ps stub layers to form a track + int kfMinLayersPS_; // maximum number of layers added to a track int kfMaxLayers_; + // + int kfMaxGaps_; + // + int kfMaxSeedingLayer_; + // + int kfNumSeedStubs_; + // + double kfMinSeedDeltaR_; // search window of each track parameter in initial uncertainties double kfRangeFactor_; - // + // initial C00 is given by inv2R uncertainty squared times this power of 2 int kfShiftInitialC00_; - // + // initial C11 is given by phiT uncertainty squared times this power of 2 int kfShiftInitialC11_; - // + // initial C22 is given by cot uncertainty squared times this power of 2 int kfShiftInitialC22_; - // + // initial C33 is given by zT uncertainty squared times this power of 2 int kfShiftInitialC33_; - - // Parameter specifying KalmanFilter Output Formatter - edm::ParameterSet pSetKFOut_; - // Conversion factor between dphi^2/weight and chi2rphi - int kfoutchi2rphiConv_; - // Conversion factor between dz^2/weight and chi2rz - int kfoutchi2rzConv_; - // Fraction of total dphi and dz ranges to calculate v0 and v1 LUT for - int weightBinFraction_; - // Constant used in FW to prevent 32-bit int overflow - int dzTruncation_; - // Constant used in FW to prevent 32-bit int overflow - int dphiTruncation_; + // + int kfShiftChi20_; + // + int kfShiftChi21_; + // + double kfCutChi2_; + // + int kfWidthChi2_; // Parameter specifying DuplicateRemoval edm::ParameterSet pSetDR_; // internal memory depth int drDepthMemory_; + // Parameter specifying Track Quality + edm::ParameterSet pSetTQ_; + // number of output channel + int tqNumChannel_; + // // Derived constants // - // true if tracker geometry and magnetic field supported - bool configurationSupported_; - // selector to partly select TPs for efficiency measurements - TrackingParticleSelector tpSelector_; - // selector to partly select TPs for fake and duplicate rate measurements - TrackingParticleSelector tpSelectorLoose_; - // TTStubAlgorithm // number of tilted layer rings per barrel layer @@ -962,15 +866,21 @@ namespace tt { // common Track finding // number of frames betwen 2 resets of 18 BX packets - int numFrames_; + int numFramesHigh_; + // number of frames betwen 2 resets of 18 BX packets + int numFramesLow_; + // number of valid frames per 18 BX packet + int numFramesIOHigh_; // number of valid frames per 18 BX packet - int numFramesIO_; + int numFramesIOLow_; // number of valid frames per 8 BX packet int numFramesFE_; // converts GeV in 1/cm double invPtToDphi_; // region size in rad double baseRegion_; + // max cot(theta) of found tracks + double maxCot_; // TMTT @@ -1055,30 +965,28 @@ namespace tt { // phi sector size in rad double baseSector_; - // cut on zT - double maxZT_; - // cut on stub cot theta - double maxCot_; + // + double maxRphi_; + // + double maxRz_; // total number of sectors int numSectors_; - // number of unused bits in GP output format - int gpNumUnusedBits_; - // cot(theta) of eta sectors - std::vector sectorCots_; - - // MHT - - // number of mht cells - int mhtNumCells_; - // ZHT + // CTB - // number of zht cells - int zhtNumCells_; + // number of bits used to count stubs per layer + int ctbWidthLayerCount_; - // KF + // KFout - int kfWidthLayerCount_; + // Bins used to digitize dPhi for chi2 calculation + std::vector kfoutdPhiBins_; + // Bins used to digitize dZ for chi2 calculation + std::vector kfoutdZBins_; + // v0 weight Bins corresponding to dPhi Bins for chi2 calculation + std::vector kfoutv0Bins_; + // v1 weight Bins corresponding to dZ Bins for chi2 calculation + std::vector kfoutv1Bins_; }; } // namespace tt diff --git a/L1Trigger/TrackTrigger/interface/SetupRcd.h b/L1Trigger/TrackTrigger/interface/SetupRcd.h index a354a41e122c2..58ea67f941a38 100644 --- a/L1Trigger/TrackTrigger/interface/SetupRcd.h +++ b/L1Trigger/TrackTrigger/interface/SetupRcd.h @@ -1,11 +1,9 @@ -#ifndef L1Trigger_TrackerDTC_SetupRcd_h -#define L1Trigger_TrackerDTC_SetupRcd_h +#ifndef L1Trigger_TrackTrigger_SetupRcd_h +#define L1Trigger_TrackTrigger_SetupRcd_h #include "FWCore/Framework/interface/DependentRecordImplementation.h" -#include "MagneticField/Records/interface/IdealMagneticFieldRecord.h" #include "Geometry/Records/interface/TrackerDigiGeometryRecord.h" -#include "Geometry/Records/interface/IdealGeometryRecord.h" #include "DataFormats/TrackerCommon/interface/TrackerTopology.h" #include "CondFormats/DataRecord/interface/TrackerDetToDTCELinkCablingMapRcd.h" #include "L1Trigger/TrackTrigger/interface/TTStubAlgorithmRecord.h" @@ -14,16 +12,12 @@ namespace tt { - typedef edm::mpl::Vector - Rcds; + typedef edm::mpl:: + Vector + SetupDepRcds; // record of tt::Setup - class SetupRcd : public edm::eventsetup::DependentRecordImplementation {}; + class SetupRcd : public edm::eventsetup::DependentRecordImplementation {}; } // namespace tt diff --git a/L1Trigger/TrackTrigger/plugins/ProducerSetup.cc b/L1Trigger/TrackTrigger/plugins/ProducerSetup.cc index 51c053df55416..3cc8421c3f125 100644 --- a/L1Trigger/TrackTrigger/plugins/ProducerSetup.cc +++ b/L1Trigger/TrackTrigger/plugins/ProducerSetup.cc @@ -1,9 +1,8 @@ #include "FWCore/Framework/interface/ESProducer.h" +#include "FWCore/Framework/interface/ModuleFactory.h" #include "FWCore/Framework/interface/ESHandle.h" #include "FWCore/ParameterSet/interface/ParameterSet.h" #include "FWCore/Utilities/interface/ESGetToken.h" -#include "FWCore/Utilities/interface/ESInputTag.h" -#include "DataFormats/Provenance/interface/ParameterSetID.h" #include "L1Trigger/TrackTrigger/interface/Setup.h" #include @@ -26,70 +25,29 @@ namespace tt { private: const ParameterSet iConfig_; - ESGetToken getTokenTTStubAlgorithm_; - ESGetToken getTokenMagneticField_; ESGetToken getTokenTrackerGeometry_; ESGetToken getTokenTrackerTopology_; ESGetToken getTokenCablingMap_; - ESGetToken getTokenGeometryConfiguration_; - ESGetToken getTokenGeometryConfigurationDD4hep_; - bool fromDD4hep_; + ESGetToken getTokenTTStubAlgorithm_; }; ProducerSetup::ProducerSetup(const ParameterSet& iConfig) : iConfig_(iConfig) { - fromDD4hep_ = iConfig.getParameter("fromDD4hep"); auto cc = setWhatProduced(this); - getTokenTTStubAlgorithm_ = cc.consumes(); - getTokenMagneticField_ = cc.consumes(); getTokenTrackerGeometry_ = cc.consumes(); getTokenTrackerTopology_ = cc.consumes(); getTokenCablingMap_ = cc.consumes(); - if (fromDD4hep_) - getTokenGeometryConfigurationDD4hep_ = cc.consumes(); - else - getTokenGeometryConfiguration_ = cc.consumes(); + getTokenTTStubAlgorithm_ = cc.consumes(); } unique_ptr ProducerSetup::produce(const SetupRcd& setupRcd) { - const MagneticField& magneticField = setupRcd.get(getTokenMagneticField_); const TrackerGeometry& trackerGeometry = setupRcd.get(getTokenTrackerGeometry_); const TrackerTopology& trackerTopology = setupRcd.get(getTokenTrackerTopology_); const TrackerDetToDTCELinkCablingMap& cablingMap = setupRcd.get(getTokenCablingMap_); const ESHandle handleStubAlgorithm = setupRcd.getHandle(getTokenTTStubAlgorithm_); - const ParameterSetID& pSetIdTTStubAlgorithm = handleStubAlgorithm.description()->pid_; const StubAlgorithmOfficial& stubAlgoritm = *dynamic_cast(&setupRcd.get(getTokenTTStubAlgorithm_)); const ParameterSet& pSetStubAlgorithm = getParameterSet(handleStubAlgorithm.description()->pid_); - if (fromDD4hep_) { - const ESHandle handleGeometryConfiguration = - setupRcd.getHandle(getTokenGeometryConfigurationDD4hep_); - const ParameterSetID& pSetIdGeometryConfiguration = handleGeometryConfiguration.description()->pid_; - const ParameterSet& pSetGeometryConfiguration = getParameterSet(handleGeometryConfiguration.description()->pid_); - return make_unique(iConfig_, - magneticField, - trackerGeometry, - trackerTopology, - cablingMap, - stubAlgoritm, - pSetStubAlgorithm, - pSetGeometryConfiguration, - pSetIdTTStubAlgorithm, - pSetIdGeometryConfiguration); - } else { - const ESHandle handleGeometryConfiguration = setupRcd.getHandle(getTokenGeometryConfiguration_); - const ParameterSetID& pSetIdGeometryConfiguration = handleGeometryConfiguration.description()->pid_; - const ParameterSet& pSetGeometryConfiguration = getParameterSet(handleGeometryConfiguration.description()->pid_); - return make_unique(iConfig_, - magneticField, - trackerGeometry, - trackerTopology, - cablingMap, - stubAlgoritm, - pSetStubAlgorithm, - pSetGeometryConfiguration, - pSetIdTTStubAlgorithm, - pSetIdGeometryConfiguration); - } + return make_unique(iConfig_, trackerGeometry, trackerTopology, cablingMap, stubAlgoritm, pSetStubAlgorithm); } } // namespace tt diff --git a/L1Trigger/TrackTrigger/python/ProducerSetup_cff.py b/L1Trigger/TrackTrigger/python/ProducerSetup_cff.py deleted file mode 100644 index 1bb2fc46f0693..0000000000000 --- a/L1Trigger/TrackTrigger/python/ProducerSetup_cff.py +++ /dev/null @@ -1,14 +0,0 @@ -import FWCore.ParameterSet.Config as cms - -from Configuration.ProcessModifiers.dd4hep_cff import dd4hep -from L1Trigger.TrackTrigger.ProducerSetup_cfi import TrackTrigger_params - -dd4hep.toModify(TrackTrigger_params, - fromDD4hep = cms.bool(True), - ProcessHistory = cms.PSet ( - GeometryConfiguration = cms.string("DDDetectorESProducer@"), - TTStubAlgorithm = cms.string("TTStubAlgorithm_official_Phase2TrackerDigi_@") - ) -) - -TrackTriggerSetup = cms.ESProducer("tt::ProducerSetup", TrackTrigger_params) diff --git a/L1Trigger/TrackTrigger/python/ProducerSetup_cfi.py b/L1Trigger/TrackTrigger/python/ProducerSetup_cfi.py deleted file mode 100644 index 333c47a2770d6..0000000000000 --- a/L1Trigger/TrackTrigger/python/ProducerSetup_cfi.py +++ /dev/null @@ -1,230 +0,0 @@ -import FWCore.ParameterSet.Config as cms - -TrackTrigger_params = cms.PSet ( - - fromDD4hep = cms.bool(False), - - # Parameter to check if configured Tracker Geometry is supported - # this refers to files included by Configuration/Geometry/python/GeometryExtended*_cff.py - UnSupportedGeometry = cms.PSet ( - XMLLabel = cms.string ("geomXMLFiles" ), # label of ESProducer/ESSource - XMLPath = cms.string ("Geometry/TrackerCommonData/data/PhaseII/" ), # compared path - XMLFile = cms.string ("tracker.xml" ), # compared filen ame - XMLVersions = cms.vstring() # list of unsupported versions - ), - - # Parameter to check if Process History is consistent with process configuration - ProcessHistory = cms.PSet ( - GeometryConfiguration = cms.string( "XMLIdealGeometryESSource@" ), # label of compared GeometryConfiguration - TTStubAlgorithm = cms.string( "TTStubAlgorithm_official_Phase2TrackerDigi_@" ) # label of compared TTStubAlgorithm - ), - - # Common track finding parameter - TrackFinding = cms.PSet ( - BeamWindowZ = cms.double( 15. ), # half lumi region size in cm - MatchedLayers = cms.int32 ( 4 ), # required number of layers a found track has to have in common with a TP to consider it matched to it - MatchedLayersPS = cms.int32 ( 0 ), # required number of ps layers a found track has to have in common with a TP to consider it matched to it - UnMatchedStubs = cms.int32 ( 1 ), # allowed number of stubs a found track may have not in common with its matched TP - UnMatchedStubsPS = cms.int32 ( 0 ), # allowed number of PS stubs a found track may have not in common with its matched TP - Scattering = cms.double( 0.131283 ) # additional radial uncertainty in cm used to calculate stub phi residual uncertainty to take multiple scattering into account - ), - - # TMTT specific parameter - TMTT = cms.PSet ( - MinPt = cms.double( 3. ), # cut on stub in GeV, also defines region overlap shape - MaxEta = cms.double( 2.4 ), # cut on stub eta - ChosenRofPhi = cms.double( 67.24 ), # critical radius defining region overlap shape in cm - NumLayers = cms.int32 ( 7 ), # number of detector layers a reconstructbale particle may cross, reduced to 7, 8th layer almost never corssed - WidthR = cms.int32 ( 12 ), # number of bits used for stub r - ChosenRofPhi - WidthPhi = cms.int32 ( 15 ), # number of bits used for stub phi w.r.t. phi region centre - WidthZ = cms.int32 ( 14 ) # number of bits used for stub z - ), - - # Hybrid specific parameter - Hybrid = cms.PSet ( - MinPtStub = cms.double( 2.0 ), # cut on stub pt in GeV, also defines region overlap shape - MinPtCand = cms.double( 1.34 ), # cut on candidate pt in GeV - MaxEta = cms.double( 2.5 ), # cut on stub eta - ChosenRofPhi = cms.double( 55. ), # critical radius defining region overlap shape in cm - NumLayers = cms.int32 ( 4 ), # max number of detector layer connected to one DTC - NumRingsPS = cms.vint32 ( 11, 11, 8, 8, 8 ), # number of outer PS rings for disk 1, 2, 3, 4, 5 - WidthsR = cms.vint32 ( 7, 7, 12, 7 ), # number of bits used for stub r w.r.t layer/disk centre for module types (barrelPS, barrel2S, diskPS, disk2S) - WidthsZ = cms.vint32 ( 12, 8, 7, 7 ), # number of bits used for stub z w.r.t layer/disk centre for module types (barrelPS, barrel2S, diskPS, disk2S) - WidthsPhi = cms.vint32 ( 14, 17, 14, 14 ), # number of bits used for stub phi w.r.t. region centre for module types (barrelPS, barrel2S, diskPS, disk2S) - WidthsAlpha = cms.vint32 ( 0, 0, 0, 4 ), # number of bits used for stub row number for module types (barrelPS, barrel2S, diskPS, disk2S) - WidthsBend = cms.vint32 ( 3, 4, 3, 4 ), # number of bits used for stub bend number for module types (barrelPS, barrel2S, diskPS, disk2S) - RangesR = cms.vdouble( 7.5, 7.5, 120. , 0. ), # range in stub r which needs to be covered for module types (barrelPS, barrel2S, diskPS, disk2S) - RangesZ = cms.vdouble( 240., 240., 7.5, 7.5 ), # range in stub z which needs to be covered for module types (barrelPS, barrel2S, diskPS, disk2S) - RangesAlpha = cms.vdouble( 0., 0., 0., 2048. ), # range in stub row which needs to be covered for module types (barrelPS, barrel2S, diskPS, disk2S) - LayerRs = cms.vdouble( 24.9316, 37.1777, 52.2656, 68.7598, 86.0156, 108.3105 ), # mean radius of outer tracker barrel layer - DiskZs = cms.vdouble( 131.1914, 154.9805, 185.3320, 221.6016, 265.0195 ), # mean z of outer tracker endcap disks - Disk2SRsSet = cms.VPSet( # center radius of outer tracker endcap 2S diks strips - cms.PSet( Disk2SRs = cms.vdouble( 66.4391, 71.4391, 76.2750, 81.2750, 82.9550, 87.9550, 93.8150, 98.8150, 99.8160, 104.8160 ) ), # disk 1 - cms.PSet( Disk2SRs = cms.vdouble( 66.4391, 71.4391, 76.2750, 81.2750, 82.9550, 87.9550, 93.8150, 98.8150, 99.8160, 104.8160 ) ), # disk 2 - cms.PSet( Disk2SRs = cms.vdouble( 63.9903, 68.9903, 74.2750, 79.2750, 81.9562, 86.9562, 92.4920, 97.4920, 99.8160, 104.8160 ) ), # disk 3 - cms.PSet( Disk2SRs = cms.vdouble( 63.9903, 68.9903, 74.2750, 79.2750, 81.9562, 86.9562, 92.4920, 97.4920, 99.8160, 104.8160 ) ), # disk 4 - cms.PSet( Disk2SRs = cms.vdouble( 63.9903, 68.9903, 74.2750, 79.2750, 81.9562, 86.9562, 92.4920, 97.4920, 99.8160, 104.8160 ) ) # disk 5 - ), - InnerRadius = cms.double( 19.6 ), # smallest stub radius after TrackBuilder in cm - WidthsRTB = cms.vint32 ( 7, 7, 12, 12 ), # number of bits used for stub r w.r.t layer/disk centre for module types (barrelPS, barrel2S, diskPS, disk2S) at TB output - ), - - # Parameter specifying TrackingParticle used for Efficiency measurements - TrackingParticle = cms.PSet ( - MinPt = cms.double( 2. ), # pt cut in GeV - MaxEta = cms.double( 2.4 ), # eta cut - MaxVertR = cms.double( 1. ), # cut on vertex pos r in cm - MaxVertZ = cms.double( 30. ), # cut on vertex pos z in cm - MaxD0 = cms.double( 5. ), # cut on impact parameter in cm - MinLayers = cms.int32 ( 4 ), # required number of associated layers to a TP to consider it reconstruct-able and to match it with TTTrack - MinLayersPS = cms.int32 ( 0 ), # required number of associated ps layers to a TP to consider it reconstruct-able - MaxBadStubs2S = cms.int32 ( 1 ), # max number of unassociated 2S stubs allowed to still associate TTTrack with TP - MaxBadStubsPS = cms.int32 ( 0 ) # max number of unassociated PS stubs allowed to still associate TTTrack with TP - ), - - # Fimrware specific Parameter - Firmware = cms.PSet ( - WidthDSPa = cms.int32( 27 ), # width of the 'A' port of an DSP slice - WidthDSPb = cms.int32( 18 ), # width of the 'B' port of an DSP slice - WidthDSPc = cms.int32( 48 ), # width of the 'C' port of an DSP slice - WidthAddrBRAM36 = cms.int32( 9 ), # smallest address width of an BRAM36 configured as broadest simple dual port memory - WidthAddrBRAM18 = cms.int32( 10 ), # smallest address width of an BRAM18 configured as broadest simple dual port memory - NumFramesInfra = cms.int32 ( 6 ), # needed gap between events of emp-infrastructure firmware - FreqLHC = cms.double( 40. ), # LHC bunch crossing rate in MHz - FreqBE = cms.double( 360. ), # processing Frequency of DTC, KF & TFP in MHz, has to be integer multiple of FreqLHC - TMP_FE = cms.int32 ( 8 ), # number of events collected in front-end - TMP_TFP = cms.int32 ( 18 ), # time multiplexed period of track finding processor - SpeedOfLight = cms.double( 2.99792458 ), # in e8 m/s - BField = cms.double( 3.81120228767395 ), # in T - BFieldError = cms.double( 1.e-6 ), # accepted difference to EventSetup in T - OuterRadius = cms.double( 112.7 ), # outer radius of outer tracker in cm - InnerRadius = cms.double( 21.8 ), # inner radius of outer tracker in cm - HalfLength = cms.double( 270. ), # half length of outer tracker in cm - TiltApproxSlope = cms.double( 0.884 ), # In tilted barrel, grad*|z|/r + int approximates |cosTilt| + |sinTilt * cotTheta| - TiltApproxIntercept = cms.double( 0.507 ), # In tilted barrel, grad*|z|/r + int approximates |cosTilt| + |sinTilt * cotTheta| - TiltUncertaintyR = cms.double( 0.12 ), # In tilted barrel, constant assumed stub radial uncertainty * sqrt(12) in cm - MindPhi = cms.double( 0.0001 ), # minimum representable stub phi uncertainty * sqrt(12) + additional terms in rad - MaxdPhi = cms.double( 0.02 ), # maximum representable stub phi uncertainty * sqrt(12) + additional terms in rad - MindZ = cms.double( 0.1 ), # minimum representable stub z uncertainty * sqrt(12) + additional terms in cm - MaxdZ = cms.double( 30. ), # maximum representable stub z uncertainty * sqrt(12) + additional terms in cm - Pitch2S = cms.double( 0.009 ), # strip pitch of outer tracker sensors in cm - PitchPS = cms.double( 0.01 ), # pixel pitch of outer tracker sensors in cm - Length2S = cms.double( 5.025 ), # strip length of outer tracker sensors in cm - LengthPS = cms.double( 0.1467 ), # pixel length of outer tracker sensors in cm - TiltedLayerLimitsZ = cms.vdouble( 15.5, 24.9, 34.3, -1., -1., -1. ), # barrel layer limit |z| value to partition into tilted and untilted region - PSDiskLimitsR = cms.vdouble( 66.4, 66.4, 64.55, 64.55, 64.55 ), # endcap disk limit r value to partition into PS and 2S region - ), - - # Parmeter specifying front-end - FrontEnd = cms.PSet ( - WidthBend = cms.int32 ( 6 ), # number of bits used for internal stub bend - WidthCol = cms.int32 ( 5 ), # number of bits used for internal stub column - WidthRow = cms.int32 ( 11 ), # number of bits used for internal stub row - BaseBend = cms.double( .25 ), # precision of internal stub bend in pitch units - BaseCol = cms.double( 1. ), # precision of internal stub column in pitch units - BaseRow = cms.double( .5 ), # precision of internal stub row in pitch units - BaseWindowSize = cms.double( .5 ), # precision of window sizes in pitch units - BendCut = cms.double( 1.3125 ) # used stub bend uncertainty in pitch units - ), - - # Parmeter specifying DTC - DTC = cms.PSet ( - NumRegions = cms.int32( 9 ), # number of phi slices the outer tracker readout is organized in - NumOverlappingRegions = cms.int32( 2 ), # number of regions a reconstructable particles may cross - NumATCASlots = cms.int32( 12 ), # number of Slots in used ATCA crates - NumDTCsPerRegion = cms.int32( 24 ), # number of DTC boards used to readout a detector region, likely constructed to be an integerer multiple of NumSlots_ - NumModulesPerDTC = cms.int32( 72 ), # max number of sensor modules connected to one DTC board - NumRoutingBlocks = cms.int32( 2 ), # number of systiloic arrays in stub router firmware - DepthMemory = cms.int32( 64 ), # fifo depth in stub router firmware - WidthRowLUT = cms.int32( 4 ), # number of row bits used in look up table - WidthInv2R = cms.int32( 9 ), # number of bits used for stub inv2R. lut addr is col + bend = 11 => 1 BRAM -> 18 bits for min and max val -> 9 - OffsetDetIdDSV = cms.int32( 1 ), # tk layout det id minus DetSetVec->detId - OffsetDetIdTP = cms.int32( -1 ), # tk layout det id minus TrackerTopology lower det id - OffsetLayerDisks = cms.int32( 10 ), # offset in layer ids between barrel layer and endcap disks - OffsetLayerId = cms.int32( 1 ), # offset between 0 and smallest layer id (barrel layer 1) - NumBarrelLayer = cms.int32( 6 ), # - SlotLimitPS = cms.int32( 6 ), # slot number changing from PS to 2S - SlotLimit10gbps = cms.int32( 3 ) # slot number changing from 10 gbps to 5gbps - ), - - # Parmeter specifying TFP - TFP = cms.PSet ( - WidthPhi0 = cms.int32( 12 ), # number of bist used for phi0 - WidthInv2R = cms.int32( 15 ), # number of bist used for inv2R - WidthCot = cms.int32( 16 ), # number of bist used for cot(theta) - WidthZ0 = cms.int32( 12 ), # number of bist used for z0 - NumChannel = cms.int32( 2 ) # number of output links - ), - - # Parmeter specifying GeometricProcessor - GeometricProcessor = cms.PSet ( - NumSectorsPhi = cms.int32 ( 2 ), # number of phi sectors used in hough transform - ChosenRofZ = cms.double( 50. ), # critical radius defining r-z sector shape in cm - RangeChiZ = cms.double( 160. ), # range of stub z residual w.r.t. sector center which needs to be covered - DepthMemory = cms.int32 ( 64 ), # fifo depth in stub router firmware - #BoundariesEta = cms.vdouble( -2.40, -2.08, -1.68, -1.26, -0.90, -0.62, -0.41, -0.20, 0.0, 0.20, 0.41, 0.62, 0.90, 1.26, 1.68, 2.08, 2.40 ) # defining r-z sector shape - BoundariesEta = cms.vdouble( -2.50, -2.23, -1.88, -1.36, -0.90, -0.62, -0.41, -0.20, 0.0, 0.20, 0.41, 0.62, 0.90, 1.36, 1.88, 2.23, 2.50 ) # defining r-z sector shape - ), - - # Parmeter specifying HoughTransform - HoughTransform = cms.PSet ( - NumBinsInv2R = cms.int32( 16 ), # number of used inv2R bins - NumBinsPhiT = cms.int32( 32 ), # number of used phiT bins - MinLayers = cms.int32( 5 ), # required number of stub layers to form a candidate - DepthMemory = cms.int32( 32 ) # internal fifo depth - ), - - # Parmeter specifying MiniHoughTransform - MiniHoughTransform = cms.PSet ( - NumBinsInv2R = cms.int32( 2 ), # number of finer inv2R bins inside HT bin - NumBinsPhiT = cms.int32( 2 ), # number of finer phiT bins inside HT bin - NumDLBs = cms.int32( 2 ), # number of dynamic load balancing steps - NumDLBNodes = cms.int32( 8 ), # number of units per dynamic load balancing step - NumDLBChannel = cms.int32( 2 ), # number of inputs per dynamic load balancing unit - MinLayers = cms.int32( 5 ) # required number of stub layers to form a candidate - ), - - # Parmeter specifying ZHoughTransform - ZHoughTransform = cms.PSet ( - NumBinsZT = cms.int32( 2 ), # - NumBinsCot = cms.int32( 2 ), # - NumStages = cms.int32( 5 ), # - MinLayers = cms.int32( 4 ), # required number of stub layers to form a candidate - MaxTracks = cms.int32( 16 ), # max number of output tracks per node - MaxStubsPerLayer = cms.int32( 4 ) # cut on number of stub per layer for input candidates - ), - - # Parmeter specifying KalmanFilter Input Formatter - - KalmanFilterIn = cms.PSet ( - ShiftRangePhi = cms.int32( 2 ), # power of 2 multiplier of stub phi residual range - ShiftRangeZ = cms.int32( 1 ) # power of 2 multiplier of stub z residual range - ), - - # Parmeter specifying KalmanFilter - KalmanFilter = cms.PSet ( - NumWorker = cms.int32 ( 2 ), # number of kf worker - RangeFactor = cms.double( 2.0 ), # search window of each track parameter in initial uncertainties - MinLayers = cms.int32 ( 4 ), # required number of stub layers to form a track - MaxLayers = cms.int32 ( 7 ), # maximum number of layers added to a track - ShiftInitialC00 = cms.int32 ( 0 ), # - ShiftInitialC11 = cms.int32 ( -2 ), # - ShiftInitialC22 = cms.int32 ( 0 ), # - ShiftInitialC33 = cms.int32 ( 0 ) # - ), - - # Parmeter specifying KalmanFilter Output Formatter - KalmanFilterOut = cms.PSet ( - Chi2rphiConv = cms.int32( 3 ), # Conversion factor between dphi^2/weight and chi2rphi - Chi2rzConv = cms.int32( 13 ), # Conversion factor between dz^2/weight and chi2rz - WeightBinFraction = cms.int32( 0 ), # Number of bits dropped from dphi and dz for v0 and v1 LUTs - DzTruncation = cms.int32( 262144 ), # Constant used in FW to prevent 32-bit int overflow - DphiTruncation = cms.int32( 16 ) # Constant used in FW to prevent 32-bit int overflow - ), - - # Parmeter specifying DuplicateRemoval - DuplicateRemoval = cms.PSet ( - DepthMemory = cms.int32( 16 ) # internal memory depth - ) - -) diff --git a/L1Trigger/TrackTrigger/python/Setup_cff.py b/L1Trigger/TrackTrigger/python/Setup_cff.py new file mode 100644 index 0000000000000..4880aeadc1cb4 --- /dev/null +++ b/L1Trigger/TrackTrigger/python/Setup_cff.py @@ -0,0 +1,6 @@ +# ESProducer processing and providing run-time constants used by Track Trigger emulators + +import FWCore.ParameterSet.Config as cms +from L1Trigger.TrackTrigger.Setup_cfi import TrackTrigger_params + +TrackTriggerSetup = cms.ESProducer("tt::ProducerSetup", TrackTrigger_params) diff --git a/L1Trigger/TrackTrigger/python/Setup_cfi.py b/L1Trigger/TrackTrigger/python/Setup_cfi.py new file mode 100644 index 0000000000000..b070040786aa6 --- /dev/null +++ b/L1Trigger/TrackTrigger/python/Setup_cfi.py @@ -0,0 +1,204 @@ +# configuration for TrackTriggerSetup + +import FWCore.ParameterSet.Config as cms + +TrackTrigger_params = cms.PSet ( + + # Parameter to check if Process History is consistent with process configuration + ProcessHistory = cms.PSet ( + GeometryConfiguration = cms.string( "XMLIdealGeometryESSource@" ), # label of compared GeometryConfiguration + TTStubAlgorithm = cms.string( "TTStubAlgorithm_official_Phase2TrackerDigi_@" ) # label of compared TTStubAlgorithm + ), + + # Common track finding parameter + TrackFinding = cms.PSet ( + BeamWindowZ = cms.double( 15. ), # half lumi region size in cm + NumLayers = cms.int32 ( 8 ), # TMTT: number of detector layers a reconstructbale particle may cross, reduced to 7, 8th layer almost never corssed + MinLayers = cms.int32 ( 4 ), # required number of stub layers to form a track + MinPt = cms.double( 2.0 ), # min track pt in GeV, also defines region overlap shape + MinPtCand = cms.double( 1.34 ), # min candiate pt in GeV + MaxEta = cms.double( 2.5 ), # cut on stub eta + MaxD0 = cms.double( 5.0 ), # in cm, constraints track reconstruction phase space + ChosenRofPhi = cms.double( 55. ), # critical radius defining region overlap shape in cm + ), + + # TMTT specific parameter + TMTT = cms.PSet ( + WidthR = cms.int32 ( 12 ), # number of bits used for stub r - ChosenRofPhi + WidthPhi = cms.int32 ( 15 ), # number of bits used for stub phi w.r.t. phi region centre + WidthZ = cms.int32 ( 14 ) # number of bits used for stub z + ), + + # Hybrid specific parameter + Hybrid = cms.PSet ( + NumLayers = cms.int32 ( 4 ), # max number of layer connected to one DTC + NumRingsPS = cms.vint32 ( 11, 11, 8, 8, 8 ), # number of outer PS rings for disk 1, 2, 3, 4, 5 + WidthsR = cms.vint32 ( 7, 7, 12, 7 ), # number of bits used for stub r w.r.t layer/disk centre for module types (barrelPS, barrel2S, diskPS, disk2S) + WidthsZ = cms.vint32 ( 12, 8, 7, 7 ), # number of bits used for stub z w.r.t layer/disk centre for module types (barrelPS, barrel2S, diskPS, disk2S) + WidthsPhi = cms.vint32 ( 14, 17, 14, 14 ), # number of bits used for stub phi w.r.t. region centre for module types (barrelPS, barrel2S, diskPS, disk2S) + WidthsAlpha = cms.vint32 ( 0, 0, 0, 4 ), # number of bits used for stub row number for module types (barrelPS, barrel2S, diskPS, disk2S) + WidthsBend = cms.vint32 ( 3, 4, 3, 4 ), # number of bits used for stub bend number for module types (barrelPS, barrel2S, diskPS, disk2S) + WidthsRTB = cms.vint32 ( 7, 7, 12, 12 ), # number of bits used for stub r w.r.t layer/disk centre for module types (barrelPS, barrel2S, diskPS, disk2S) at TB output + RangesR = cms.vdouble( 7.5, 7.5, 120. , 0. ), # range in stub r which needs to be covered for module types (barrelPS, barrel2S, diskPS, disk2S) + RangesZ = cms.vdouble( 240., 240., 7.5, 7.5 ), # range in stub z which needs to be covered for module types (barrelPS, barrel2S, diskPS, disk2S) + RangesAlpha = cms.vdouble( 0., 0., 0., 2048. ), # range in stub row which needs to be covered for module types (barrelPS, barrel2S, diskPS, disk2S) + LayerRs = cms.vdouble( 24.9316, 37.1777, 52.2656, 68.7598, 86.0156, 108.3105 ), # mean radius of outer tracker barrel layer + DiskZs = cms.vdouble( 131.1914, 154.9805, 185.3320, 221.6016, 265.0195 ), # mean z of outer tracker endcap disks + Disk2SRsSet = cms.VPSet( # center radius of outer tracker endcap 2S diks strips + cms.PSet( Disk2SRs = cms.vdouble( 66.4391, 71.4391, 76.2750, 81.2750, 82.9550, 87.9550, 93.8150, 98.8150, 99.8160, 104.8160 ) ), # disk 1 + cms.PSet( Disk2SRs = cms.vdouble( 66.4391, 71.4391, 76.2750, 81.2750, 82.9550, 87.9550, 93.8150, 98.8150, 99.8160, 104.8160 ) ), # disk 2 + cms.PSet( Disk2SRs = cms.vdouble( 63.9903, 68.9903, 74.2750, 79.2750, 81.9562, 86.9562, 92.4920, 97.4920, 99.8160, 104.8160 ) ), # disk 3 + cms.PSet( Disk2SRs = cms.vdouble( 63.9903, 68.9903, 74.2750, 79.2750, 81.9562, 86.9562, 92.4920, 97.4920, 99.8160, 104.8160 ) ), # disk 4 + cms.PSet( Disk2SRs = cms.vdouble( 63.9903, 68.9903, 74.2750, 79.2750, 81.9562, 86.9562, 92.4920, 97.4920, 99.8160, 104.8160 ) ) # disk 5 + ), + BarrelHalfLength = cms.double( 120.0 ), # biggest barrel stub z position after TrackBuilder in cm + InnerRadius = cms.double( 19.6 ), # smallest stub radius after TrackBuilder in cm + ), + + # Fimrware specific Parameter + Firmware = cms.PSet ( + WidthDSPa = cms.int32 ( 27 ), # width of the 'A' port of an DSP slice + WidthDSPb = cms.int32 ( 18 ), # width of the 'B' port of an DSP slice + WidthDSPc = cms.int32 ( 48 ), # width of the 'C' port of an DSP slice + WidthAddrBRAM36 = cms.int32 ( 9 ), # smallest address width of an BRAM36 configured as broadest simple dual port memory + WidthAddrBRAM18 = cms.int32 ( 10 ), # smallest address width of an BRAM18 configured as broadest simple dual port memory + NumFramesInfra = cms.int32 ( 6 ), # needed gap between events of emp-infrastructure firmware + FreqLHC = cms.double( 40. ), # LHC bunch crossing rate in MHz + FreqBEHigh = cms.double( 360. ), # processing Frequency of DTC, KF & TFP in MHz, has to be integer multiple of FreqLHC + FreqBELow = cms.double( 240. ), # processing Frequency of DTC, KF & TFP in MHz, has to be integer multiple of FreqLHC + TMP_FE = cms.int32 ( 8 ), # number of events collected in front-end + TMP_TFP = cms.int32 ( 18 ), # time multiplexed period of track finding processor + SpeedOfLight = cms.double( 2.99792458 ), # in e8 m/s + ), + + # Parameter specifying outer tracker + Tracker = cms.PSet ( + BField = cms.double ( 3.81120228767395 ), # in T + BFieldError = cms.double ( 1.e-6 ), # accepted difference to EventSetup in T + OuterRadius = cms.double ( 112.7 ), # outer radius of outer tracker in cm + InnerRadius = cms.double ( 21.8 ), # inner radius of outer tracker in cm + HalfLength = cms.double ( 270. ), # half length of outer tracker in cm + TiltApproxSlope = cms.double ( 0.884 ), # In tilted barrel, grad*|z|/r + int approximates |cosTilt| + |sinTilt * cotTheta| + TiltApproxIntercept = cms.double ( 0.507 ), # In tilted barrel, grad*|z|/r + int approximates |cosTilt| + |sinTilt * cotTheta| + TiltUncertaintyR = cms.double ( 0.12 ), # In tilted barrel, constant assumed stub radial uncertainty * sqrt(12) in cm + Scattering = cms.double ( 0.131283 ), # additional radial uncertainty in cm used to calculate stub phi residual uncertainty to take multiple scattering into account + PitchRow2S = cms.double ( 0.009 ), # strip pitch of outer tracker sensors in cm + PitchRowPS = cms.double ( 0.01 ), # pixel pitch of outer tracker sensors in cm + PitchCol2S = cms.double ( 5.025 ), # strip length of outer tracker sensors in cm + PitchColPS = cms.double ( 0.1467 ), # pixel length of outer tracker sensors in cm + LimitPSBarrel = cms.double ( 125.0 ), # barrel layer limit r value to partition into PS and 2S region + LimitsTiltedR = cms.vdouble( 30.0, 45.0, 60.0 ), # barrel layer limit r value to partition into tilted and untilted region + LimitsTiltedZ = cms.vdouble( 15.5, 25.0, 34.3 ), # barrel layer limit |z| value to partition into tilted and untilted region + LimitsPSDiksZ = cms.vdouble( 125.0, 150.0, 175.0, 200.0, 250.0 ), # endcap disk limit |z| value to partition into PS and 2S region + LimitsPSDiksR = cms.vdouble( 66.0, 66.0, 63.0, 63.0, 63.0 ), # endcap disk limit r value to partition into PS and 2S region + TiltedLayerLimitsZ = cms.vdouble( 15.5, 24.9, 34.3, -1., -1., -1. ), # barrel layer limit |z| value to partition into tilted and untilted region + PSDiskLimitsR = cms.vdouble( 66.4, 66.4, 64.55, 64.55, 64.55 ), # endcap disk limit r value to partition into PS and 2S region + ), + + # Parmeter specifying front-end + FrontEnd = cms.PSet ( + WidthBend = cms.int32 ( 6 ), # number of bits used for internal stub bend + WidthCol = cms.int32 ( 5 ), # number of bits used for internal stub column + WidthRow = cms.int32 ( 11 ), # number of bits used for internal stub row + BaseBend = cms.double( .25 ), # precision of internal stub bend in pitch units + BaseCol = cms.double( 1. ), # precision of internal stub column in pitch units + BaseRow = cms.double( .5 ), # precision of internal stub row in pitch units + BaseWindowSize = cms.double( .5 ), # precision of window sizes in pitch units + BendCut = cms.double( 1.3125 ) # used stub bend uncertainty in pitch units + ), + + # Parmeter specifying DTC + DTC = cms.PSet ( + NumRegions = cms.int32( 9 ), # number of phi slices the outer tracker readout is organized in + NumOverlappingRegions = cms.int32( 2 ), # number of regions a reconstructable particles may cross + NumATCASlots = cms.int32( 12 ), # number of Slots in used ATCA crates + NumDTCsPerRegion = cms.int32( 24 ), # number of DTC boards used to readout a detector region, likely constructed to be an integerer multiple of NumSlots_ + NumModulesPerDTC = cms.int32( 72 ), # max number of sensor modules connected to one DTC board + NumRoutingBlocks = cms.int32( 2 ), # number of systiloic arrays in stub router firmware + DepthMemory = cms.int32( 64 ), # fifo depth in stub router firmware + WidthRowLUT = cms.int32( 4 ), # number of row bits used in look up table + WidthInv2R = cms.int32( 9 ), # number of bits used for stub inv2R. lut addr is col + bend = 11 => 1 BRAM -> 18 bits for min and max val -> 9 + OffsetDetIdDSV = cms.int32( 1 ), # tk layout det id minus DetSetVec->detId + OffsetDetIdTP = cms.int32( -1 ), # tk layout det id minus TrackerTopology lower det id + OffsetLayerDisks = cms.int32( 10 ), # offset in layer ids between barrel layer and endcap disks + OffsetLayerId = cms.int32( 1 ), # offset between 0 and smallest layer id (barrel layer 1) + NumBarrelLayer = cms.int32( 6 ), # + SlotLimitPS = cms.int32( 6 ), # slot number changing from PS to 2S + SlotLimit10gbps = cms.int32( 3 ) # slot number changing from 10 gbps to 5gbps + ), + + # Parmeter specifying TFP + TFP = cms.PSet ( + WidthPhi0 = cms.int32( 12 ), # number of bist used for phi0 + WidthInvR = cms.int32( 15 ), # number of bist used for invR + WidthCot = cms.int32( 16 ), # number of bist used for cot(theta) + WidthZ0 = cms.int32( 12 ), # number of bist used for z0 + NumChannel = cms.int32( 2 ) # number of output links + ), + + # Parmeter specifying GeometricProcessor + GeometricProcessor = cms.PSet ( + NumBinsPhiT = cms.int32 ( 2 ), # number of phi sectors used in hough transform + NumBinsZT = cms.int32 ( 32 ), # number of eta sectors used in hough transform + ChosenRofZ = cms.double( 57.76 ), # critical radius defining r-z sector shape in cm + DepthMemory = cms.int32 ( 32 ), # fifo depth in stub router firmware + WidthModule = cms.int32 ( 3 ), # + PosPS = cms.int32 ( 2 ), # + PosBarrel = cms.int32 ( 1 ), # + PosTilted = cms.int32 ( 0 ) # + ), + + # Parmeter specifying HoughTransform + HoughTransform = cms.PSet ( + NumBinsInv2R = cms.int32( 16 ), # number of used inv2R bins + NumBinsPhiT = cms.int32( 32 ), # number of used phiT bins + MinLayers = cms.int32( 5 ), # required number of stub layers to form a candidate + DepthMemory = cms.int32( 32 ) # internal fifo depth + ), + + # Parmeter specifying Clean Track Builder + + CleanTrackBuilder = cms.PSet ( + NumBinsInv2R = cms.int32( 4 ), # number of inv2R bins + NumBinsPhiT = cms.int32( 4 ), # number of phiT bins + NumBinsCot = cms.int32( 4 ), # number of cot bins + NumBinsZT = cms.int32( 4 ), # number of zT bins + MinLayers = cms.int32( 4 ), # required number of stub layers to form a candidate + MaxTracks = cms.int32( 16 ), # max number of output tracks per node + MaxStubs = cms.int32( 4 ), # cut on number of stub per layer for input candidates + DepthMemory = cms.int32( 16 ) # internal fifo depth + ), + + # Parmeter specifying KalmanFilter + KalmanFilter = cms.PSet ( + NumWorker = cms.int32 ( 1 ), # number of kf worker + MaxTracks = cms.int32 ( 63 ), # max number of tracks a kf worker can process + RangeFactor = cms.double( 3.0 ), # search window of each track parameter in initial uncertainties + MinLayers = cms.int32 ( 4 ), # required number of stub layers to form a track + MinLayersPS = cms.int32 ( 0 ), # required number of ps stub layers to form a track + MaxLayers = cms.int32 ( 8 ), # maximum number of layers added to a track + MaxGaps = cms.int32 ( 4 ), # + MaxSeedingLayer = cms.int32 ( 4 ), # + NumSeedStubs = cms.int32 ( 2 ), # + MinSeedDeltaR = cms.double( 1.6 ), # + ShiftInitialC00 = cms.int32 ( 0 ), # initial C00 is given by inv2R uncertainty squared times this power of 2 + ShiftInitialC11 = cms.int32 ( 0 ), # initial C11 is given by phiT uncertainty squared times this power of 2 + ShiftInitialC22 = cms.int32 ( 0 ), # initial C22 is given by cot uncertainty squared times this power of 2 + ShiftInitialC33 = cms.int32 ( 0 ), # initial C33 is given by zT uncertainty squared times this power of 2 + ShiftChi20 = cms.int32 ( -1 ), # + ShiftChi21 = cms.int32 ( -5 ), # + CutChi2 = cms.double( 2.0 ), # + WidthChi2 = cms.int32 ( 8 ) # + ), + + # Parmeter specifying DuplicateRemoval + DuplicateRemoval = cms.PSet ( + DepthMemory = cms.int32( 16 ) # internal memory depth + ), + + # Parmeter specifying Track Quality + TrackQuality = cms.PSet ( + NumChannel = cms.int32( 2 ) # number of output channel + ) + +) diff --git a/L1Trigger/TrackTrigger/python/TTStubAlgorithmRegister_cfi.py b/L1Trigger/TrackTrigger/python/TTStubAlgorithmRegister_cfi.py index b7dea492ce517..43e8215c4e253 100644 --- a/L1Trigger/TrackTrigger/python/TTStubAlgorithmRegister_cfi.py +++ b/L1Trigger/TrackTrigger/python/TTStubAlgorithmRegister_cfi.py @@ -42,5 +42,4 @@ # We prefer the global geometry algorithm for now in order not to break # anything. Override with process.TTStubAlgorithm_PSimHit_ = ..., # etc. in your configuration. -TTStubAlgorithm_Phase2TrackerDigi_ = cms.ESPrefer("TTStubAlgorithm_official_Phase2TrackerDigi_") - +TTStubAlgorithm_Phase2TrackerDigi_ = cms.ESPrefer("TTStubAlgorithm_official_Phase2TrackerDigi_") \ No newline at end of file diff --git a/L1Trigger/TrackTrigger/python/TrackQualityParams_cfi.py b/L1Trigger/TrackTrigger/python/TrackQualityParams_cfi.py deleted file mode 100644 index e1e8f6cdb5ec2..0000000000000 --- a/L1Trigger/TrackTrigger/python/TrackQualityParams_cfi.py +++ /dev/null @@ -1,11 +0,0 @@ -import FWCore.ParameterSet.Config as cms - -TrackQualityParams = cms.PSet(# This emulation GBDT is optimised for the HYBRID_NEWKF emulation and works with the emulation of the KF out module - # It is compatible with the HYBRID simulation and will give equivilant performance with this workflow - model = cms.FileInPath("L1Trigger/TrackTrigger/data/L1_TrackQuality_GBDT_emulation_digitized.json"), - #Vector of strings of training features, in the order that the model was trained with - featureNames = cms.vstring(["tanl", "z0_scaled", "bendchi2_bin", "nstub", - "nlaymiss_interior", "chi2rphi_bin", "chi2rz_bin"]), - tqemu_TanlScale = cms.double( 128.0), - tqemu_Z0Scale = cms.double( 64.0 ), - ) diff --git a/L1Trigger/TrackTrigger/src/L1TrackQuality.cc b/L1Trigger/TrackTrigger/src/L1TrackQuality.cc deleted file mode 100644 index 7296cb385e665..0000000000000 --- a/L1Trigger/TrackTrigger/src/L1TrackQuality.cc +++ /dev/null @@ -1,115 +0,0 @@ -/* -Track Quality Body file -C.Brown & C.Savard 07/2020 -*/ - -#include "L1Trigger/TrackTrigger/interface/L1TrackQuality.h" - -//Constructors - -L1TrackQuality::L1TrackQuality() {} - -L1TrackQuality::L1TrackQuality(const edm::ParameterSet& qualityParams) : useHPH_(false), bonusFeatures_() { - // Unpacks EDM parameter set itself to save unecessary processing within TrackProducers - setModel(qualityParams.getParameter("model"), - qualityParams.getParameter>("featureNames")); -} - -std::vector L1TrackQuality::featureTransform(TTTrack& aTrack, - std::vector const& featureNames) { - // List input features for MVA in proper order below, the current features options are - // {"phi", "eta", "z0", "bendchi2_bin", "nstub", "nlaymiss_interior", "chi2rphi_bin", - // "chi2rz_bin"} - // - // To use more features, they must be created here and added to feature_map below - - std::vector transformedFeatures; - - // Define feature map, filled as features are generated - std::map feature_map; - - // -------- calculate feature variables -------- - - // calculate number of missed interior layers from hitpattern - int tmp_trk_hitpattern = aTrack.hitPattern(); - int nbits = floor(log2(tmp_trk_hitpattern)) + 1; - int lay_i = 0; - int tmp_trk_nlaymiss_interior = 0; - bool seq = false; - for (int i = 0; i < nbits; i++) { - lay_i = ((1 << i) & tmp_trk_hitpattern) >> i; //0 or 1 in ith bit (right to left) - - if (lay_i && !seq) - seq = true; //sequence starts when first 1 found - if (!lay_i && seq) - tmp_trk_nlaymiss_interior++; - } - - // binned chi2 variables - int tmp_trk_bendchi2_bin = aTrack.getBendChi2Bits(); - int tmp_trk_chi2rphi_bin = aTrack.getChi2RPhiBits(); - int tmp_trk_chi2rz_bin = aTrack.getChi2RZBits(); - - // get the nstub - std::vector>, TTStub>> stubRefs = - aTrack.getStubRefs(); - int tmp_trk_nstub = stubRefs.size(); - - // get other variables directly from TTTrack - float tmp_trk_z0 = aTrack.z0(); - float tmp_trk_z0_scaled = tmp_trk_z0 / abs(aTrack.minZ0); - float tmp_trk_phi = aTrack.phi(); - float tmp_trk_eta = aTrack.eta(); - float tmp_trk_tanl = aTrack.tanL(); - - // -------- fill the feature map --------- - - feature_map["nstub"] = float(tmp_trk_nstub); - feature_map["z0"] = tmp_trk_z0; - feature_map["z0_scaled"] = tmp_trk_z0_scaled; - feature_map["phi"] = tmp_trk_phi; - feature_map["eta"] = tmp_trk_eta; - feature_map["nlaymiss_interior"] = float(tmp_trk_nlaymiss_interior); - feature_map["bendchi2_bin"] = tmp_trk_bendchi2_bin; - feature_map["chi2rphi_bin"] = tmp_trk_chi2rphi_bin; - feature_map["chi2rz_bin"] = tmp_trk_chi2rz_bin; - feature_map["tanl"] = tmp_trk_tanl; - - // fill tensor with track params - transformedFeatures.reserve(featureNames.size()); - for (const std::string& feature : featureNames) - transformedFeatures.push_back(feature_map[feature]); - - return transformedFeatures; -} - -void L1TrackQuality::setL1TrackQuality(TTTrack& aTrack) { - // load in bdt - conifer::BDT bdt(this->model_.fullPath()); - - // collect features and classify using bdt - std::vector inputs = featureTransform(aTrack, this->featureNames_); - std::vector output = bdt.decision_function(inputs); - aTrack.settrkMVA1(1. / (1. + exp(-output.at(0)))); -} - -float L1TrackQuality::runEmulatedTQ(std::vector> inputFeatures) { - // load in bdt - - conifer::BDT, ap_fixed<10, 5>> bdt(this->model_.fullPath()); - - // collect features and classify using bdt - std::vector> output = bdt.decision_function(inputFeatures); - return output.at(0).to_float(); // need logistic sigmoid fcn applied to xgb output -} - -void L1TrackQuality::setModel(edm::FileInPath const& model, std::vector const& featureNames) { - //Convert algorithm string to Enum class for track by track comparison - model_ = model; - featureNames_ = featureNames; -} - -void L1TrackQuality::setBonusFeatures(std::vector bonusFeatures) { - bonusFeatures_ = bonusFeatures; - useHPH_ = true; -} diff --git a/L1Trigger/TrackTrigger/src/SensorModule.cc b/L1Trigger/TrackTrigger/src/SensorModule.cc index e30e75819c744..e71df45626453 100644 --- a/L1Trigger/TrackTrigger/src/SensorModule.cc +++ b/L1Trigger/TrackTrigger/src/SensorModule.cc @@ -113,10 +113,17 @@ namespace tt { // calculate tilt correction parameter used to project r to z uncertainty tiltCorrectionSlope_ = barrel_ ? 0. : 1.; tiltCorrectionIntercept_ = barrel_ ? 1. : 0.; + tilted_ = false; if (typeTilt == tiltedMinus || typeTilt == tiltedPlus) { + tilted_ = true; tiltCorrectionSlope_ = setup->tiltApproxSlope(); tiltCorrectionIntercept_ = setup->tiltApproxIntercept(); } + // stub uncertainty + scattering_ = setup->scattering(); + dR_ = abs(sinTilt_) * pitchCol_; + dPhi_ = pitchRow_ / r_; + dZ_ = abs(cosTilt_) * pitchCol_ + dR_ * abs(z_) / r_; } unsigned int SensorModule::ringId(const Setup* setup) const { diff --git a/L1Trigger/TrackTrigger/src/Setup.cc b/L1Trigger/TrackTrigger/src/Setup.cc index f26ed4ac2c238..e9dab24ac4496 100644 --- a/L1Trigger/TrackTrigger/src/Setup.cc +++ b/L1Trigger/TrackTrigger/src/Setup.cc @@ -1,16 +1,13 @@ #include "L1Trigger/TrackTrigger/interface/Setup.h" #include "FWCore/Utilities/interface/Exception.h" -#include "DataFormats/Provenance/interface/ProcessConfiguration.h" #include "DataFormats/L1TrackTrigger/interface/TTBV.h" #include "DataFormats/L1TrackTrigger/interface/TTTypes.h" #include #include +#include #include #include -#include -#include -#include using namespace std; using namespace edm; @@ -18,59 +15,33 @@ using namespace edm; namespace tt { Setup::Setup(const ParameterSet& iConfig, - const MagneticField& magneticField, const TrackerGeometry& trackerGeometry, const TrackerTopology& trackerTopology, const TrackerDetToDTCELinkCablingMap& cablingMap, const StubAlgorithmOfficial& stubAlgorithm, - const ParameterSet& pSetStubAlgorithm, - const ParameterSet& pSetGeometryConfiguration, - const ParameterSetID& pSetIdTTStubAlgorithm, - const ParameterSetID& pSetIdGeometryConfiguration) - : magneticField_(&magneticField), - trackerGeometry_(&trackerGeometry), + const ParameterSet& pSetStubAlgorithm) + : trackerGeometry_(&trackerGeometry), trackerTopology_(&trackerTopology), cablingMap_(&cablingMap), stubAlgorithm_(&stubAlgorithm), pSetSA_(&pSetStubAlgorithm), - pSetGC_(&pSetGeometryConfiguration), - pSetIdTTStubAlgorithm_(pSetIdTTStubAlgorithm), - pSetIdGeometryConfiguration_(pSetIdGeometryConfiguration), - // DD4hep - fromDD4hep_(iConfig.getParameter("fromDD4hep")), - // Parameter to check if configured Tracker Geometry is supported - pSetSG_(iConfig.getParameter("UnSupportedGeometry")), - sgXMLLabel_(pSetSG_.getParameter("XMLLabel")), - sgXMLPath_(pSetSG_.getParameter("XMLPath")), - sgXMLFile_(pSetSG_.getParameter("XMLFile")), - sgXMLVersions_(pSetSG_.getParameter>("XMLVersions")), - // Parameter to check if Process History is consistent with process configuration - pSetPH_(iConfig.getParameter("ProcessHistory")), - phGeometryConfiguration_(pSetPH_.getParameter("GeometryConfiguration")), - phTTStubAlgorithm_(pSetPH_.getParameter("TTStubAlgorithm")), // Common track finding parameter pSetTF_(iConfig.getParameter("TrackFinding")), beamWindowZ_(pSetTF_.getParameter("BeamWindowZ")), - matchedLayers_(pSetTF_.getParameter("MatchedLayers")), - matchedLayersPS_(pSetTF_.getParameter("MatchedLayersPS")), - unMatchedStubs_(pSetTF_.getParameter("UnMatchedStubs")), - unMatchedStubsPS_(pSetTF_.getParameter("UnMatchedStubsPS")), - scattering_(pSetTF_.getParameter("Scattering")), + minPt_(pSetTF_.getParameter("MinPt")), + minPtCand_(pSetTF_.getParameter("MinPtCand")), + maxEta_(pSetTF_.getParameter("MaxEta")), + maxD0_(pSetTF_.getParameter("MaxD0")), + chosenRofPhi_(pSetTF_.getParameter("ChosenRofPhi")), + numLayers_(pSetTF_.getParameter("NumLayers")), + minLayers_(pSetTF_.getParameter("MinLayers")), // TMTT specific parameter pSetTMTT_(iConfig.getParameter("TMTT")), - minPt_(pSetTMTT_.getParameter("MinPt")), - maxEta_(pSetTMTT_.getParameter("MaxEta")), - chosenRofPhi_(pSetTMTT_.getParameter("ChosenRofPhi")), - numLayers_(pSetTMTT_.getParameter("NumLayers")), tmttWidthR_(pSetTMTT_.getParameter("WidthR")), tmttWidthPhi_(pSetTMTT_.getParameter("WidthPhi")), tmttWidthZ_(pSetTMTT_.getParameter("WidthZ")), // Hybrid specific parameter pSetHybrid_(iConfig.getParameter("Hybrid")), - hybridMinPtStub_(pSetHybrid_.getParameter("MinPtStub")), - hybridMinPtCand_(pSetHybrid_.getParameter("MinPtCand")), - hybridMaxEta_(pSetHybrid_.getParameter("MaxEta")), - hybridChosenRofPhi_(pSetHybrid_.getParameter("ChosenRofPhi")), hybridNumLayers_(pSetHybrid_.getParameter("NumLayers")), hybridNumRingsPS_(pSetHybrid_.getParameter>("NumRingsPS")), hybridWidthsR_(pSetHybrid_.getParameter>("WidthsR")), @@ -84,19 +55,9 @@ namespace tt { hybridLayerRs_(pSetHybrid_.getParameter>("LayerRs")), hybridDiskZs_(pSetHybrid_.getParameter>("DiskZs")), hybridDisk2SRsSet_(pSetHybrid_.getParameter>("Disk2SRsSet")), + tbBarrelHalfLength_(pSetHybrid_.getParameter("BarrelHalfLength")), tbInnerRadius_(pSetHybrid_.getParameter("InnerRadius")), tbWidthsR_(pSetHybrid_.getParameter>("WidthsRTB")), - // Parameter specifying TrackingParticle used for Efficiency measurements - pSetTP_(iConfig.getParameter("TrackingParticle")), - tpMinPt_(pSetTP_.getParameter("MinPt")), - tpMaxEta_(pSetTP_.getParameter("MaxEta")), - tpMaxVertR_(pSetTP_.getParameter("MaxVertR")), - tpMaxVertZ_(pSetTP_.getParameter("MaxVertZ")), - tpMaxD0_(pSetTP_.getParameter("MaxD0")), - tpMinLayers_(pSetTP_.getParameter("MinLayers")), - tpMinLayersPS_(pSetTP_.getParameter("MinLayersPS")), - tpMaxBadStubs2S_(pSetTP_.getParameter("MaxBadStubs2S")), - tpMaxBadStubsPS_(pSetTP_.getParameter("MaxBadStubsPS")), // Fimrware specific Parameter pSetFW_(iConfig.getParameter("Firmware")), widthDSPa_(pSetFW_.getParameter("WidthDSPa")), @@ -106,28 +67,33 @@ namespace tt { widthAddrBRAM18_(pSetFW_.getParameter("WidthAddrBRAM18")), numFramesInfra_(pSetFW_.getParameter("NumFramesInfra")), freqLHC_(pSetFW_.getParameter("FreqLHC")), - freqBE_(pSetFW_.getParameter("FreqBE")), + freqBEHigh_(pSetFW_.getParameter("FreqBEHigh")), + freqBELow_(pSetFW_.getParameter("FreqBELow")), tmpFE_(pSetFW_.getParameter("TMP_FE")), tmpTFP_(pSetFW_.getParameter("TMP_TFP")), speedOfLight_(pSetFW_.getParameter("SpeedOfLight")), - bField_(pSetFW_.getParameter("BField")), - bFieldError_(pSetFW_.getParameter("BFieldError")), - outerRadius_(pSetFW_.getParameter("OuterRadius")), - innerRadius_(pSetFW_.getParameter("InnerRadius")), - halfLength_(pSetFW_.getParameter("HalfLength")), - tiltApproxSlope_(pSetFW_.getParameter("TiltApproxSlope")), - tiltApproxIntercept_(pSetFW_.getParameter("TiltApproxIntercept")), - tiltUncertaintyR_(pSetFW_.getParameter("TiltUncertaintyR")), - mindPhi_(pSetFW_.getParameter("MindPhi")), - maxdPhi_(pSetFW_.getParameter("MaxdPhi")), - mindZ_(pSetFW_.getParameter("MindZ")), - maxdZ_(pSetFW_.getParameter("MaxdZ")), - pitch2S_(pSetFW_.getParameter("Pitch2S")), - pitchPS_(pSetFW_.getParameter("PitchPS")), - length2S_(pSetFW_.getParameter("Length2S")), - lengthPS_(pSetFW_.getParameter("LengthPS")), - tiltedLayerLimitsZ_(pSetFW_.getParameter>("TiltedLayerLimitsZ")), - psDiskLimitsR_(pSetFW_.getParameter>("PSDiskLimitsR")), + // Tracker specific Paramter + pSetOT_(iConfig.getParameter("Tracker")), + bField_(pSetOT_.getParameter("BField")), + bFieldError_(pSetOT_.getParameter("BFieldError")), + outerRadius_(pSetOT_.getParameter("OuterRadius")), + innerRadius_(pSetOT_.getParameter("InnerRadius")), + halfLength_(pSetOT_.getParameter("HalfLength")), + tiltApproxSlope_(pSetOT_.getParameter("TiltApproxSlope")), + tiltApproxIntercept_(pSetOT_.getParameter("TiltApproxIntercept")), + tiltUncertaintyR_(pSetOT_.getParameter("TiltUncertaintyR")), + scattering_(pSetOT_.getParameter("Scattering")), + pitchRow2S_(pSetOT_.getParameter("PitchRow2S")), + pitchRowPS_(pSetOT_.getParameter("PitchRowPS")), + pitchCol2S_(pSetOT_.getParameter("PitchCol2S")), + pitchColPS_(pSetOT_.getParameter("PitchColPS")), + limitPSBarrel_(pSetOT_.getParameter("LimitPSBarrel")), + limitsTiltedR_(pSetOT_.getParameter>("LimitsTiltedR")), + limitsTiltedZ_(pSetOT_.getParameter>("LimitsTiltedZ")), + limitsPSDiksZ_(pSetOT_.getParameter>("LimitsPSDiksZ")), + limitsPSDiksR_(pSetOT_.getParameter>("LimitsPSDiksR")), + tiltedLayerLimitsZ_(pSetOT_.getParameter>("TiltedLayerLimitsZ")), + psDiskLimitsR_(pSetOT_.getParameter>("PSDiskLimitsR")), // Parmeter specifying front-end pSetFE_(iConfig.getParameter("FrontEnd")), widthBend_(pSetFE_.getParameter("WidthBend")), @@ -159,70 +125,62 @@ namespace tt { // Parmeter specifying TFP pSetTFP_(iConfig.getParameter("TFP")), tfpWidthPhi0_(pSetTFP_.getParameter("WidthPhi0")), - tfpWidthInv2R_(pSetTFP_.getParameter("WidthInv2R")), + tfpWidthInvR_(pSetTFP_.getParameter("WidthInvR")), tfpWidthCot_(pSetTFP_.getParameter("WidthCot")), tfpWidthZ0_(pSetTFP_.getParameter("WidthZ0")), tfpNumChannel_(pSetTFP_.getParameter("NumChannel")), // Parmeter specifying GeometricProcessor pSetGP_(iConfig.getParameter("GeometricProcessor")), - numSectorsPhi_(pSetGP_.getParameter("NumSectorsPhi")), + gpNumBinsPhiT_(pSetGP_.getParameter("NumBinsPhiT")), + gpNumBinsZT_(pSetGP_.getParameter("NumBinsZT")), chosenRofZ_(pSetGP_.getParameter("ChosenRofZ")), - neededRangeChiZ_(pSetGP_.getParameter("RangeChiZ")), gpDepthMemory_(pSetGP_.getParameter("DepthMemory")), - boundariesEta_(pSetGP_.getParameter>("BoundariesEta")), + gpWidthModule_(pSetGP_.getParameter("WidthModule")), + gpPosPS_(pSetGP_.getParameter("PosPS")), + gpPosBarrel_(pSetGP_.getParameter("PosBarrel")), + gpPosTilted_(pSetGP_.getParameter("PosTilted")), // Parmeter specifying HoughTransform pSetHT_(iConfig.getParameter("HoughTransform")), htNumBinsInv2R_(pSetHT_.getParameter("NumBinsInv2R")), htNumBinsPhiT_(pSetHT_.getParameter("NumBinsPhiT")), htMinLayers_(pSetHT_.getParameter("MinLayers")), htDepthMemory_(pSetHT_.getParameter("DepthMemory")), - // Parmeter specifying MiniHoughTransform - pSetMHT_(iConfig.getParameter("MiniHoughTransform")), - mhtNumBinsInv2R_(pSetMHT_.getParameter("NumBinsInv2R")), - mhtNumBinsPhiT_(pSetMHT_.getParameter("NumBinsPhiT")), - mhtNumDLBs_(pSetMHT_.getParameter("NumDLBs")), - mhtNumDLBNodes_(pSetMHT_.getParameter("NumDLBNodes")), - mhtNumDLBChannel_(pSetMHT_.getParameter("NumDLBChannel")), - mhtMinLayers_(pSetMHT_.getParameter("MinLayers")), - // Parmeter specifying ZHoughTransform - pSetZHT_(iConfig.getParameter("ZHoughTransform")), - zhtNumBinsZT_(pSetZHT_.getParameter("NumBinsZT")), - zhtNumBinsCot_(pSetZHT_.getParameter("NumBinsCot")), - zhtNumStages_(pSetZHT_.getParameter("NumStages")), - zhtMinLayers_(pSetZHT_.getParameter("MinLayers")), - zhtMaxTracks_(pSetZHT_.getParameter("MaxTracks")), - zhtMaxStubsPerLayer_(pSetZHT_.getParameter("MaxStubsPerLayer")), - // Parameter specifying KalmanFilter Input Formatter - pSetKFin_(iConfig.getParameter("KalmanFilterIn")), - kfinShiftRangePhi_(pSetKFin_.getParameter("ShiftRangePhi")), - kfinShiftRangeZ_(pSetKFin_.getParameter("ShiftRangeZ")), + // Parameter specifying Track Builder + pSetCTB_(iConfig.getParameter("CleanTrackBuilder")), + ctbNumBinsInv2R_(pSetCTB_.getParameter("NumBinsInv2R")), + ctbNumBinsPhiT_(pSetCTB_.getParameter("NumBinsPhiT")), + ctbNumBinsCot_(pSetCTB_.getParameter("NumBinsCot")), + ctbNumBinsZT_(pSetCTB_.getParameter("NumBinsZT")), + ctbMinLayers_(pSetCTB_.getParameter("MinLayers")), + ctbMaxTracks_(pSetCTB_.getParameter("MaxTracks")), + ctbMaxStubs_(pSetCTB_.getParameter("MaxStubs")), + ctbDepthMemory_(pSetCTB_.getParameter("DepthMemory")), // Parmeter specifying KalmanFilter pSetKF_(iConfig.getParameter("KalmanFilter")), kfNumWorker_(pSetKF_.getParameter("NumWorker")), + kfMaxTracks_(pSetKF_.getParameter("MaxTracks")), kfMinLayers_(pSetKF_.getParameter("MinLayers")), + kfMinLayersPS_(pSetKF_.getParameter("MinLayersPS")), kfMaxLayers_(pSetKF_.getParameter("MaxLayers")), + kfMaxGaps_(pSetKF_.getParameter("MaxGaps")), + kfMaxSeedingLayer_(pSetKF_.getParameter("MaxSeedingLayer")), + kfNumSeedStubs_(pSetKF_.getParameter("NumSeedStubs")), + kfMinSeedDeltaR_(pSetKF_.getParameter("MinSeedDeltaR")), kfRangeFactor_(pSetKF_.getParameter("RangeFactor")), kfShiftInitialC00_(pSetKF_.getParameter("ShiftInitialC00")), kfShiftInitialC11_(pSetKF_.getParameter("ShiftInitialC11")), kfShiftInitialC22_(pSetKF_.getParameter("ShiftInitialC22")), kfShiftInitialC33_(pSetKF_.getParameter("ShiftInitialC33")), - // Parmeter specifying KalmanFilter Output Formatter - pSetKFOut_(iConfig.getParameter("KalmanFilterOut")), - kfoutchi2rphiConv_(pSetKFOut_.getParameter("Chi2rphiConv")), - kfoutchi2rzConv_(pSetKFOut_.getParameter("Chi2rzConv")), - weightBinFraction_(pSetKFOut_.getParameter("WeightBinFraction")), - dzTruncation_(pSetKFOut_.getParameter("DzTruncation")), - dphiTruncation_(pSetKFOut_.getParameter("DphiTruncation")), + kfShiftChi20_(pSetKF_.getParameter("ShiftChi20")), + kfShiftChi21_(pSetKF_.getParameter("ShiftChi21")), + kfCutChi2_(pSetKF_.getParameter("CutChi2")), + kfWidthChi2_(pSetKF_.getParameter("WidthChi2")), // Parmeter specifying DuplicateRemoval pSetDR_(iConfig.getParameter("DuplicateRemoval")), - drDepthMemory_(pSetDR_.getParameter("DepthMemory")) { - configurationSupported_ = true; - // check if bField is supported - checkMagneticField(); - // check if geometry is supported - checkGeometry(); - if (!configurationSupported_) - return; + drDepthMemory_(pSetDR_.getParameter("DepthMemory")), + // Parmeter specifying Track Quality + pSetTQ_(iConfig.getParameter("TrackQuality")), + tqNumChannel_(pSetTQ_.getParameter("NumChannel")) { // derive constants calculateConstants(); // convert configuration of TTStubAlgorithm @@ -234,60 +192,6 @@ namespace tt { encodeBend(encodingsBend2S_, false); // create sensor modules produceSensorModules(); - // configure TPSelector - configureTPSelector(); - } - - // checks current configuration vs input sample configuration - void Setup::checkHistory(const ProcessHistory& processHistory) const { - const pset::Registry* psetRegistry = pset::Registry::instance(); - // check used TTStubAlgorithm in input producer - checkHistory(processHistory, psetRegistry, phTTStubAlgorithm_, pSetIdTTStubAlgorithm_); - // check used GeometryConfiguration in input producer - checkHistory(processHistory, psetRegistry, phGeometryConfiguration_, pSetIdGeometryConfiguration_); - } - - // checks consitency between history and current configuration for a specific module - void Setup::checkHistory(const ProcessHistory& ph, - const pset::Registry* pr, - const string& label, - const ParameterSetID& pSetId) const { - vector> pSets; - pSets.reserve(ph.size()); - for (const ProcessConfiguration& pc : ph) { - const ParameterSet* pSet = pr->getMapped(pc.parameterSetID()); - if (pSet && pSet->exists(label)) - pSets.emplace_back(pc.processName(), pSet->getParameterSet(label)); - } - if (pSets.empty()) { - cms::Exception exception("BadConfiguration"); - exception << label << " not found in process history."; - exception.addContext("tt::Setup::checkHistory"); - throw exception; - } - auto consistent = [&pSetId](const pair& p) { return p.second.id() == pSetId; }; - if (!all_of(pSets.begin(), pSets.end(), consistent)) { - const ParameterSet& pSetProcess = getParameterSet(pSetId); - cms::Exception exception("BadConfiguration"); - exception.addContext("tt::Setup::checkHistory"); - exception << label << " inconsistent with History." << endl; - exception << "Current Configuration:" << endl << pSetProcess.dump() << endl; - for (const pair& p : pSets) - if (!consistent(p)) - exception << "Process " << p.first << " Configuration:" << endl << dumpDiff(p.second, pSetProcess) << endl; - throw exception; - } - } - - // dumps pSetHistory where incosistent lines with pSetProcess are highlighted - string Setup::dumpDiff(const ParameterSet& pSetHistory, const ParameterSet& pSetProcess) const { - stringstream ssHistory, ssProcess, ss; - ssHistory << pSetHistory.dump(); - ssProcess << pSetProcess.dump(); - string lineHistory, lineProcess; - for (; getline(ssHistory, lineHistory) && getline(ssProcess, lineProcess);) - ss << (lineHistory != lineProcess ? "\033[1;31m" : "") << lineHistory << "\033[0m" << endl; - return ss.str(); } // converts tk layout id into dtc id @@ -373,49 +277,18 @@ namespace tt { return it->second; } + // sensor module for ttStubRef + SensorModule* Setup::sensorModule(const TTStubRef& ttStubRef) const { + const DetId detId = ttStubRef->getDetId() + offsetDetIdDSV_; + return this->sensorModule(detId); + } + // index = encoded bend, value = decoded bend for given window size and module type const vector& Setup::encodingBend(int windowSize, bool psModule) const { const vector>& encodingsBend = psModule ? encodingsBendPS_ : encodingsBend2S_; return encodingsBend.at(windowSize); } - // check if bField is supported - void Setup::checkMagneticField() { - const double bFieldES = magneticField_->inTesla(GlobalPoint(0., 0., 0.)).z(); - if (abs(bField_ - bFieldES) > bFieldError_) { - configurationSupported_ = false; - LogWarning("ConfigurationNotSupported") - << "Magnetic Field from EventSetup (" << bFieldES << ") differs more then " << bFieldError_ - << " from supported value (" << bField_ << "). "; - } - } - - // check if geometry is supported - void Setup::checkGeometry() { - //FIX ME: Can we assume that geometry used in dd4hep wf supports L1Track? - if (!fromDD4hep_) { - const vector& geomXMLFiles = pSetGC_->getParameter>(sgXMLLabel_); - string version; - for (const string& geomXMLFile : geomXMLFiles) { - const auto begin = geomXMLFile.find(sgXMLPath_) + sgXMLPath_.size(); - const auto end = geomXMLFile.find(sgXMLFile_); - if (begin != string::npos && end != string::npos) - version = geomXMLFile.substr(begin, end - begin - 1); - } - if (version.empty()) { - cms::Exception exception("LogicError"); - exception << "No " << sgXMLPath_ << "*/" << sgXMLFile_ << " found in GeometryConfiguration"; - exception.addContext("tt::Setup::checkGeometry"); - throw exception; - } - if (find(sgXMLVersions_.begin(), sgXMLVersions_.end(), version) != sgXMLVersions_.end()) { - configurationSupported_ = false; - LogWarning("ConfigurationNotSupported") - << "Geometry Configuration " << sgXMLPath_ << version << "/" << sgXMLFile_ << " is not supported. "; - } - } - } - // convert configuration of TTStubAlgorithm void Setup::consumeStubAlgorithm() { numTiltedLayerRings_ = pSetSA_->getParameter>("NTiltedRings"); @@ -488,25 +361,6 @@ namespace tt { } } - // configure TPSelector - void Setup::configureTPSelector() { - // configure TrackingParticleSelector - const double ptMin = tpMinPt_; - constexpr double ptMax = 9.e9; - const double etaMax = tpMaxEta_; - const double tip = tpMaxVertR_; - const double lip = tpMaxVertZ_; - constexpr int minHit = 0; - constexpr bool signalOnly = true; - constexpr bool intimeOnly = true; - constexpr bool chargedOnly = true; - constexpr bool stableOnly = false; - tpSelector_ = TrackingParticleSelector( - ptMin, ptMax, -etaMax, etaMax, tip, lip, minHit, signalOnly, intimeOnly, chargedOnly, stableOnly); - tpSelectorLoose_ = - TrackingParticleSelector(ptMin, ptMax, -etaMax, etaMax, tip, lip, minHit, false, false, false, stableOnly); - } - // stub layer id (barrel: 1 - 6, endcap: 11 - 15) int Setup::layerId(const TTStubRef& ttStubRef) const { const DetId& detId = ttStubRef->getDetId(); @@ -516,12 +370,16 @@ namespace tt { // return tracklet layerId (barrel: [0-5], endcap: [6-10]) for given TTStubRef int Setup::trackletLayerId(const TTStubRef& ttStubRef) const { - return this->layerId(ttStubRef) - (this->barrel(ttStubRef) ? offsetLayerId_ : numBarrelLayer_ - offsetLayerId_); + static constexpr int offsetBarrel = 1; + static constexpr int offsetDisks = 5; + return this->layerId(ttStubRef) - (this->barrel(ttStubRef) ? offsetBarrel : offsetDisks); } // return index layerId (barrel: [0-5], endcap: [0-6]) for given TTStubRef int Setup::indexLayerId(const TTStubRef& ttStubRef) const { - return this->layerId(ttStubRef) - (this->barrel(ttStubRef) ? offsetLayerId_ : offsetLayerId_ + offsetLayerDisks_); + static constexpr int offsetBarrel = 1; + static constexpr int offsetDisks = 11; + return this->layerId(ttStubRef) - (this->barrel(ttStubRef) ? offsetBarrel : offsetDisks); } // true if stub from barrel module @@ -533,7 +391,9 @@ namespace tt { // true if stub from barrel module bool Setup::psModule(const TTStubRef& ttStubRef) const { const DetId& detId = ttStubRef->getDetId(); - return trackerGeometry_->getDetectorType(detId) == TrackerGeometry::ModuleType::Ph2PSP; + SensorModule* sm = sensorModule(detId + 1); + return sm->psModule(); + //return trackerGeometry_->getDetectorType(detId) == TrackerGeometry::ModuleType::Ph2PSP; } // @@ -541,7 +401,7 @@ namespace tt { TTBV ttBV; for (int layer = numLayers_ - 1; layer >= 0; layer--) { const int i = ints[layer]; - ttBV += TTBV(i, kfWidthLayerCount_); + ttBV += TTBV(i, ctbWidthLayerCount_); } return ttBV; } @@ -551,7 +411,7 @@ namespace tt { TTBV ttBV; for (int layer = numLayers_ - 1; layer >= 0; layer--) { const int i = ints[layer]; - ttBV += TTBV((hitPattern[layer] ? i - 1 : 0), kfWidthLayerCount_); + ttBV += TTBV((hitPattern[layer] ? i - 1 : 0), ctbWidthLayerCount_); } return ttBV; } @@ -561,7 +421,7 @@ namespace tt { TTBV bv(ttBV); vector ints(numLayers_, 0); for (int layer = 0; layer < numLayers_; layer++) { - const int i = bv.extract(kfWidthLayerCount_); + const int i = bv.extract(ctbWidthLayerCount_); ints[layer] = i + (hitPattern[layer] ? 1 : 0); } return ints; @@ -572,7 +432,7 @@ namespace tt { TTBV bv(ttBV); vector ints(numLayers_, 0); for (int layer = 0; layer < numLayers_; layer++) - ints[layer] = bv.extract(kfWidthLayerCount_); + ints[layer] = bv.extract(ctbWidthLayerCount_); return ints; } @@ -580,36 +440,14 @@ namespace tt { double Setup::dPhi(const TTStubRef& ttStubRef, double inv2R) const { const DetId& detId = ttStubRef->getDetId(); SensorModule* sm = sensorModule(detId + 1); - const double r = stubPos(ttStubRef).perp(); - const double sigma = sm->pitchRow() / r; - const double scat = scattering_ * abs(inv2R); - const double extra = sm->barrel() ? 0. : sm->pitchCol() * abs(inv2R); - const double digi = tmttBasePhi_; - const double dPhi = sigma + scat + extra + digi; - if (dPhi >= maxdPhi_ || dPhi < mindPhi_) { - cms::Exception exception("out_of_range"); - exception.addContext("tt::Setup::dPhi"); - exception << "Stub phi uncertainty " << dPhi << " " - << "is out of range " << mindPhi_ << " to " << maxdPhi_ << "."; - throw exception; - } - return dPhi; + return sm->dPhi(inv2R); } // stub projected z uncertainty - double Setup::dZ(const TTStubRef& ttStubRef, double cot) const { + double Setup::dZ(const TTStubRef& ttStubRef) const { const DetId& detId = ttStubRef->getDetId(); SensorModule* sm = sensorModule(detId + 1); - const double sigma = sm->pitchCol() * sm->tiltCorrection(cot); - const double digi = tmttBaseZ_; - const double dZ = sigma + digi; - if (dZ >= maxdZ_ || dZ < mindZ_) { - cms::Exception exception("out_of_range"); - exception.addContext("tt::Setup::dZ"); - exception << "Stub z uncertainty " << dZ << " " - << "is out of range " << mindZ_ << " to " << maxdZ_ << "."; - throw exception; - } + const double dZ = sm->dZ(); return dZ; } @@ -639,28 +477,71 @@ namespace tt { set hitPattern; for (const TTStubRef& ttStubRef : ttStubRefs) hitPattern.insert(layerId(ttStubRef)); - return (int)hitPattern.size() >= tpMinLayers_; + return (int)hitPattern.size() >= minLayers_; } - // checks if tracking particle is selected for efficiency measurements - bool Setup::useForAlgEff(const TrackingParticle& tp) const { - const bool selected = tpSelector_(tp); - const double cot = sinh(tp.eta()); - const double s = sin(tp.phi()); - const double c = cos(tp.phi()); - const TrackingParticle::Point& v = tp.vertex(); - const double z0 = v.z() - (v.x() * c + v.y() * s) * cot; - const double d0 = v.x() * s - v.y() * c; - return selected && (abs(d0) < tpMaxD0_) && (abs(z0) < tpMaxVertZ_); + // + TTBV Setup::module(double r, double z) const { + static constexpr int layer1 = 0; + static constexpr int layer2 = 1; + static constexpr int layer3 = 2; + static constexpr int disk1 = 0; + static constexpr int disk2 = 1; + static constexpr int disk3 = 2; + static constexpr int disk4 = 3; + static constexpr int disk5 = 4; + bool ps(false); + bool barrel(false); + bool tilted(false); + if (abs(z) < limitPSBarrel_) { + barrel = true; + if (r < limitsTiltedR_[layer3]) + ps = true; + if (r < limitsTiltedR_[layer1]) + tilted = abs(z) > limitsTiltedZ_[layer1]; + else if (r < limitsTiltedR_[layer2]) + tilted = abs(z) > limitsTiltedZ_[layer2]; + else if (r < limitsTiltedR_[layer3]) + tilted = abs(z) > limitsTiltedZ_[layer3]; + } else if (abs(z) > limitsPSDiksZ_[disk5]) + ps = r < limitsPSDiksR_[disk5]; + else if (abs(z) > limitsPSDiksZ_[disk4]) + ps = r < limitsPSDiksR_[disk4]; + else if (abs(z) > limitsPSDiksZ_[disk3]) + ps = r < limitsPSDiksR_[disk3]; + else if (abs(z) > limitsPSDiksZ_[disk2]) + ps = r < limitsPSDiksR_[disk2]; + else if (abs(z) > limitsPSDiksZ_[disk1]) + ps = r < limitsPSDiksR_[disk1]; + TTBV module(0, gpWidthModule_); + if (ps) + module.set(gpPosPS_); + if (barrel) + module.set(gpPosBarrel_); + if (tilted) + module.set(gpPosTilted_); + return module; + } + + // stub projected phi uncertainty for given module type, stub radius and track curvature + double Setup::dPhi(const TTBV& module, double r, double inv2R) const { + const double sigma = (ps(module) ? pitchRowPS_ : pitchRow2S_) / r; + const double dR = scattering_ + (barrel(module) ? (tilted(module) ? tiltUncertaintyR_ : 0.0) + : (ps(module) ? pitchColPS_ : pitchCol2S_)); + const double dPhi = sigma + dR * abs(inv2R) + tmttBasePhi_; + return dPhi; } // derive constants void Setup::calculateConstants() { // emp - const int numFramesPerBX = freqBE_ / freqLHC_; - numFrames_ = numFramesPerBX * tmpTFP_ - 1; - numFramesIO_ = numFramesPerBX * tmpTFP_ - numFramesInfra_; - numFramesFE_ = numFramesPerBX * tmpFE_ - numFramesInfra_; + const int numFramesPerBXHigh = freqBEHigh_ / freqLHC_; + numFramesHigh_ = numFramesPerBXHigh * tmpTFP_ - 1; + numFramesIOHigh_ = numFramesPerBXHigh * tmpTFP_ - numFramesInfra_; + const int numFramesPerBXLow = freqBELow_ / freqLHC_; + numFramesLow_ = numFramesPerBXLow * tmpTFP_ - 1; + numFramesIOLow_ = numFramesPerBXLow * tmpTFP_ - numFramesInfra_; + numFramesFE_ = numFramesPerBXHigh * tmpFE_ - numFramesInfra_; // dsp widthDSPab_ = widthDSPa_ - 1; widthDSPau_ = widthDSPab_ - 1; @@ -669,26 +550,23 @@ namespace tt { widthDSPcb_ = widthDSPc_ - 1; widthDSPcu_ = widthDSPcb_ - 1; // firmware - maxPitch_ = max(pitchPS_, pitch2S_); - maxLength_ = max(lengthPS_, length2S_); + maxPitchRow_ = max(pitchRowPS_, pitchRow2S_); + maxPitchCol_ = max(pitchColPS_, pitchCol2S_); // common track finding invPtToDphi_ = speedOfLight_ * bField_ / 2000.; baseRegion_ = 2. * M_PI / numRegions_; + maxCot_ = beamWindowZ_ / chosenRofZ_ + sinh(maxEta_); // gp - baseSector_ = baseRegion_ / numSectorsPhi_; - maxCot_ = sinh(maxEta_); - maxZT_ = maxCot_ * chosenRofZ_; - numSectorsEta_ = boundariesEta_.size() - 1; - numSectors_ = numSectorsPhi_ * numSectorsEta_; - sectorCots_.reserve(numSectorsEta_); - for (int eta = 0; eta < numSectorsEta_; eta++) - sectorCots_.emplace_back((sinh(boundariesEta_.at(eta)) + sinh(boundariesEta_.at(eta + 1))) / 2.); + baseSector_ = baseRegion_ / gpNumBinsPhiT_; + maxRphi_ = max(abs(outerRadius_ - chosenRofPhi_), abs(innerRadius_ - chosenRofPhi_)); + maxRz_ = max(abs(outerRadius_ - chosenRofZ_), abs(innerRadius_ - chosenRofZ_)); + numSectors_ = gpNumBinsPhiT_ * gpNumBinsZT_; // tmtt const double rangeInv2R = 2. * invPtToDphi_ / minPt_; tmttBaseInv2R_ = rangeInv2R / htNumBinsInv2R_; tmttBasePhiT_ = baseSector_ / htNumBinsPhiT_; const double baseRgen = tmttBasePhiT_ / tmttBaseInv2R_; - const double rangeR = 2. * max(abs(outerRadius_ - chosenRofPhi_), abs(innerRadius_ - chosenRofPhi_)); + const double rangeR = 2. * maxRphi_; const int baseShiftR = ceil(log2(rangeR / baseRgen / pow(2., tmttWidthR_))); tmttBaseR_ = baseRgen * pow(2., baseShiftR); const double rangeZ = 2. * halfLength_; @@ -698,14 +576,13 @@ namespace tt { const int baseShiftPhi = ceil(log2(rangePhi / tmttBasePhiT_ / pow(2., tmttWidthPhi_))); tmttBasePhi_ = tmttBasePhiT_ * pow(2., baseShiftPhi); tmttWidthLayer_ = ceil(log2(numLayers_)); - tmttWidthSectorEta_ = ceil(log2(numSectorsEta_)); + tmttWidthSectorEta_ = ceil(log2(gpNumBinsZT_)); tmttWidthInv2R_ = ceil(log2(htNumBinsInv2R_)); tmttNumUnusedBits_ = TTBV::S_ - tmttWidthLayer_ - 2 * tmttWidthSectorEta_ - tmttWidthR_ - tmttWidthPhi_ - - tmttWidthZ_ - 2 * tmttWidthInv2R_ - numSectorsPhi_ - 1; + tmttWidthZ_ - 2 * tmttWidthInv2R_ - gpNumBinsPhiT_ - 1; // hybrid - const double hybridRangeInv2R = 2. * invPtToDphi_ / hybridMinPtStub_; - const double hybridRangeR = - 2. * max(abs(outerRadius_ - hybridChosenRofPhi_), abs(innerRadius_ - hybridChosenRofPhi_)); + const double hybridRangeInv2R = 2. * invPtToDphi_ / minPt_; + const double hybridRangeR = 2. * max(abs(outerRadius_ - chosenRofPhi_), abs(innerRadius_ - chosenRofPhi_)); hybridRangePhi_ = baseRegion_ + (hybridRangeR * hybridRangeInv2R) / 2.; hybridWidthLayerId_ = ceil(log2(hybridNumLayers_)); hybridBasesZ_.reserve(SensorModule::NumTypes); @@ -726,7 +603,10 @@ namespace tt { hybridNumsUnusedBits_.emplace_back(TTBV::S_ - hybridWidthsR_.at(type) - hybridWidthsZ_.at(type) - hybridWidthsPhi_.at(type) - hybridWidthsAlpha_.at(type) - hybridWidthsBend_.at(type) - hybridWidthLayerId_ - 1); - hybridMaxCot_ = sinh(hybridMaxEta_); + hybridBaseR_ = *min_element(hybridBasesR_.begin(), hybridBasesR_.end()); + hybridBasePhi_ = *min_element(hybridBasesPhi_.begin(), hybridBasesPhi_.end()); + hybridBaseZ_ = *min_element(hybridBasesZ_.begin(), hybridBasesZ_.end()); + hybridMaxCot_ = sinh(maxEta_); disk2SRs_.reserve(hybridDisk2SRsSet_.size()); for (const auto& pSet : hybridDisk2SRsSet_) disk2SRs_.emplace_back(pSet.getParameter>("Disk2SRs")); @@ -741,17 +621,14 @@ namespace tt { dtcBaseInv2R_ = tmttBaseInv2R_ * pow(2., baseShiftInv2R); const int baseDiffM = dtcWidthRowLUT_ - widthRow_; dtcBaseM_ = tmttBasePhi_ * pow(2., baseDiffM); - const double x1 = pow(2, widthRow_) * baseRow_ * maxPitch_ / 2.; - const double x0 = x1 - pow(2, dtcWidthRowLUT_) * baseRow_ * maxPitch_; + const double x1 = pow(2, widthRow_) * baseRow_ * maxPitchRow_ / 2.; + const double x0 = x1 - pow(2, dtcWidthRowLUT_) * baseRow_ * maxPitchRow_; const double maxM = atan2(x1, innerRadius_) - atan2(x0, innerRadius_); dtcWidthM_ = ceil(log2(maxM / dtcBaseM_)); dtcNumStreams_ = numDTCs_ * numOverlappingRegions_; - // mht - mhtNumCells_ = mhtNumBinsInv2R_ * mhtNumBinsPhiT_; - // zht - zhtNumCells_ = zhtNumBinsCot_ * zhtNumBinsZT_; - // - kfWidthLayerCount_ = ceil(log2(zhtMaxStubsPerLayer_)); + // ctb + ctbWidthLayerCount_ = ceil(log2(ctbMaxStubs_)); + // kf } // returns bit accurate hybrid stub radius for given TTStubRef and h/w bit word @@ -806,7 +683,7 @@ namespace tt { } p = GlobalPoint(GlobalPoint::Cylindrical(r, phi, z)); } else { - bv >>= 2 * tmttWidthInv2R_ + 2 * tmttWidthSectorEta_ + numSectorsPhi_ + tmttWidthLayer_; + bv >>= 2 * tmttWidthInv2R_ + 2 * tmttWidthSectorEta_ + gpNumBinsPhiT_ + tmttWidthLayer_; double z = (bv.val(tmttWidthZ_, 0, true) + .5) * tmttBaseZ_; bv >>= tmttWidthZ_; double phi = (bv.val(tmttWidthPhi_, 0, true) + .5) * tmttBasePhi_; diff --git a/L1Trigger/TrackTrigger/test/CleanRelVal_cfg.py b/L1Trigger/TrackTrigger/test/CleanRelVal_cfg.py new file mode 100644 index 0000000000000..9576afb41be57 --- /dev/null +++ b/L1Trigger/TrackTrigger/test/CleanRelVal_cfg.py @@ -0,0 +1,78 @@ +################################################################################################ +# Run bit-accurate TMTT L1 tracking emulation. +# +# To run execute do +# cmsRun L1Trigger/L1TTrackerTFP/test/test_cfg.py +# where the arguments take default values if you don't specify them. You can change defaults below. +################################################################################################# + +import FWCore.ParameterSet.Config as cms + +process = cms.Process( "CleanUp" ) +process.load( 'FWCore.MessageService.MessageLogger_cfi' ) +process.load('Configuration.Geometry.GeometryExtended2026D98Reco_cff') +process.load('Configuration.Geometry.GeometryExtended2026D98_cff') +process.load( 'Configuration.StandardSequences.MagneticField_cff' ) +process.load( 'Configuration.StandardSequences.FrontierConditions_GlobalTag_cff' ) +process.load( 'Configuration.EventContent.EventContent_cff' ) +process.load( 'L1Trigger.TrackTrigger.TrackTrigger_cff' ) + +from Configuration.AlCa.GlobalTag import GlobalTag +process.GlobalTag = GlobalTag(process.GlobalTag, 'auto:phase2_realistic', '') + +# load code that creates clean TVs +process.load( 'SimTracker.TrackTriggerAssociation.CleanTV_cff' ) +# load code that creates clean TPs +process.load( 'SimTracker.TrackTriggerAssociation.CleanTP_cff' ) +# load code that associates TTStubs with clean TPs +process.load( 'SimTracker.TrackTriggerAssociation.CleanAssoc_cff' ) + +# build schedule +process.clean = cms.Path( process.CleanTV + process.CleanTP + process.CleanAssoc ) +process.schedule = cms.Schedule( process.clean ) + +# create options +import FWCore.ParameterSet.VarParsing as VarParsing +options = VarParsing.VarParsing( 'analysis' ) +# specify input MC +File = '28f2ac9b-b0a6-44a1-b10e-32ea9f59b611.root' +Samples = [ + '/store/relvalOrig/CMSSW_14_0_0_pre2/RelValTTbar_14TeV/GEN-SIM-DIGI-RAW/PU_133X_mcRun4_realistic_v1_STD_2026D98_PU200_RV229-v1/2580000/'+File +] +options.register( 'inputMC', Samples, VarParsing.VarParsing.multiplicity.singleton, VarParsing.VarParsing.varType.string, "Files to be processed" ) +# specify number of events to process. +options.register( 'Events',100,VarParsing.VarParsing.multiplicity.singleton, VarParsing.VarParsing.varType.int, "Number of Events to analyze" ) +options.parseArguments() + +process.options = cms.untracked.PSet( wantSummary = cms.untracked.bool(False) ) +process.maxEvents = cms.untracked.PSet( input = cms.untracked.int32(options.Events) ) +process.source = cms.Source( + "PoolSource", + fileNames = cms.untracked.vstring( options.inputMC ), + noEventSort = cms.untracked.bool( True ), + secondaryFileNames = cms.untracked.vstring(), + duplicateCheckMode = cms.untracked.string( 'noDuplicateCheck' ) +) +#process.Timing = cms.Service( "Timing", summaryOnly = cms.untracked.bool( True ) ) +process.MessageLogger.cerr.enableStatistics = False + +if True : + process.writeDataset = cms.OutputModule("PoolOutputModule", + splitLevel = cms.untracked.int32(0), + eventAutoFlushCompressedSize = cms.untracked.int32(5242880), + outputCommands = process.RAWSIMEventContent.outputCommands, + fileName = cms.untracked.string('/heplnw039/store/relvalTrimmed/CMSSW_14_0_0_pre2/RelValTTbar_14TeV/GEN-SIM-DIGI-RAW/PU_133X_mcRun4_realistic_v1_STD_2026D98_PU200_RV229-v1/2580000/'+File), + dataset = cms.untracked.PSet( + filterName = cms.untracked.string(''), + dataTier = cms.untracked.string('GEN-SIM') + ) + ) + process.writeDataset.outputCommands.append('drop *_*_*_*') + process.writeDataset.outputCommands.append('keep *_CleanTV_AtLeastOneCluster_*') + process.writeDataset.outputCommands.append('keep *_CleanTP_AtLeastOneCluster_*') + process.writeDataset.outputCommands.append('keep *_TTStubsFromPhase2TrackerDigis_ClusterAccepted_*') + process.writeDataset.outputCommands.append('keep *_TTStubsFromPhase2TrackerDigis_StubAccepted_*') + process.writeDataset.outputCommands.append('keep *_CleanAssoc_AtLeastOneCluster_*') + + process.pd = cms.EndPath(process.writeDataset) + process.schedule.append(process.pd) \ No newline at end of file diff --git a/L1Trigger/TrackerDTC/BuildFile.xml b/L1Trigger/TrackerDTC/BuildFile.xml index 5b09818a993ea..d22023419c1d6 100644 --- a/L1Trigger/TrackerDTC/BuildFile.xml +++ b/L1Trigger/TrackerDTC/BuildFile.xml @@ -1,5 +1,6 @@ - + + diff --git a/L1Trigger/TrackerDTC/interface/DTC.h b/L1Trigger/TrackerDTC/interface/DTC.h index 6f5bc72f6f9b7..2716d9d0564fc 100644 --- a/L1Trigger/TrackerDTC/interface/DTC.h +++ b/L1Trigger/TrackerDTC/interface/DTC.h @@ -5,6 +5,7 @@ #include "L1Trigger/TrackTrigger/interface/Setup.h" #include "L1Trigger/TrackerDTC/interface/LayerEncoding.h" #include "L1Trigger/TrackerDTC/interface/Stub.h" +#include "L1Trigger/TrackerTFP/interface/DataFormats.h" #include #include @@ -25,6 +26,7 @@ namespace trackerDTC { public: DTC(const edm::ParameterSet& iConfig, const tt::Setup* setup, + const trackerTFP::DataFormats* dataFormats, const LayerEncoding* layerEncoding, int dtcId, const std::vector>& stubsDTC); @@ -43,6 +45,8 @@ namespace trackerDTC { Stub* pop_front(Stubs& stubs); // helper class to store configurations const tt::Setup* setup_; + // provides dataformats + const trackerTFP::DataFormats* dataFormats_; // enables emulation of truncation bool enableTruncation_; // outer tracker detector region [0-8] diff --git a/L1Trigger/TrackerDTC/interface/LayerEncoding.h b/L1Trigger/TrackerDTC/interface/LayerEncoding.h index 7e18cff679ba8..e1352f27859cc 100644 --- a/L1Trigger/TrackerDTC/interface/LayerEncoding.h +++ b/L1Trigger/TrackerDTC/interface/LayerEncoding.h @@ -5,7 +5,6 @@ #include "L1Trigger/TrackerDTC/interface/LayerEncodingRcd.h" #include "L1Trigger/TrackTrigger/interface/Setup.h" #include "L1Trigger/TrackTrigger/interface/SensorModule.h" -#include "FWCore/ParameterSet/interface/ParameterSet.h" #include @@ -19,10 +18,10 @@ namespace trackerDTC { class LayerEncoding { public: LayerEncoding() {} - LayerEncoding(const edm::ParameterSet& iConfig, const tt::Setup* setup); + LayerEncoding(const tt::Setup* setup); ~LayerEncoding() {} // decode layer id for given sensor module - int decode(tt::SensorModule* sm) const; + int decode(const tt::SensorModule* sm) const; // get encoded layers read by given DTC const std::vector& layers(int dtcId) const { return encodingsLayerId_.at(dtcId % numDTCsPerRegion_); } diff --git a/L1Trigger/TrackerDTC/interface/Stub.h b/L1Trigger/TrackerDTC/interface/Stub.h index 1dde69828ff57..0b64f0c0d645b 100644 --- a/L1Trigger/TrackerDTC/interface/Stub.h +++ b/L1Trigger/TrackerDTC/interface/Stub.h @@ -3,7 +3,8 @@ #include "L1Trigger/TrackTrigger/interface/Setup.h" #include "L1Trigger/TrackerDTC/interface/LayerEncoding.h" -#include "SimDataFormats/Associations/interface/TTTypes.h" +#include "SimTracker/TrackTriggerAssociation/interface/TTTypes.h" +#include "L1Trigger/TrackerTFP/interface/DataFormats.h" #include #include @@ -17,20 +18,30 @@ namespace trackerDTC { */ class Stub { public: - Stub(const edm::ParameterSet&, const tt::Setup*, const LayerEncoding*, tt::SensorModule*, const TTStubRef&); + Stub(const trackerTFP::DataFormats*, const tt::SensorModule*, const TTStubRef&); + Stub(const edm::ParameterSet&, + const tt::Setup*, + const trackerTFP::DataFormats*, + const LayerEncoding*, + const tt::SensorModule*, + const TTStubRef&); ~Stub() {} // underlying TTStubRef - TTStubRef ttStubRef() const { return ttStubRef_; } + const TTStubRef& ttStubRef() const { return ttStubRef_; } // did pass pt and eta cut bool valid() const { return valid_; } // stub bend in quarter pitch units int bend() const { return bend_; } // bit accurate representation of Stub - tt::Frame frame(int region) const; + tt::FrameStub frame(int region) const; // checks stubs region assignment - bool inRegion(int region) const; + bool inRegion(int region) const { return regions_[region]; } + // range of stub extrapolated phi to radius chosenRofPhi in rad + std::pair phiT() const { return phiT_; } + // stub phi w.r.t. detector region centre in rad + double phi() const { return phi_; } - private: + public: // truncates double precision to f/w integer equivalent double digi(double value, double precision) const; // 64 bit stub in hybrid data format @@ -39,12 +50,14 @@ namespace trackerDTC { tt::Frame formatTMTT(int region) const; // stores, calculates and provides run-time constants const tt::Setup* setup_; + // helper class to extract structured data from tt::Frames + const trackerTFP::DataFormats* dataFormats_; // class to encode layer ids used between DTC and TFP in Hybrid const LayerEncoding* layerEncoding_; // representation of an outer tracker sensormodule - tt::SensorModule* sm_; + const tt::SensorModule* sm_; // underlying TTStubRef - TTStubRef ttStubRef_; + const TTStubRef ttStubRef_; // chosen TT algorithm bool hybrid_; // passes pt and eta cut @@ -73,12 +86,12 @@ namespace trackerDTC { double d_; // range of stub inv2R in 1/cm std::pair inv2R_; - // range of stub cot(theta) - std::pair cot_; - // range of stub extrapolated phi to radius chosenRofPhi in rad + // range of stub extrapolated phi to radius chosenRofPhi wrt detector nonant center in rad std::pair phiT_; + // range of stub extrapolated z to radius chosenRofZ in cm + std::pair zT_; // shared regions this stub belongs to [0-1] - std::vector regions_; + TTBV regions_; }; } // namespace trackerDTC diff --git a/L1Trigger/TrackerDTC/plugins/ProducerED.cc b/L1Trigger/TrackerDTC/plugins/ProducerDTC.cc similarity index 59% rename from L1Trigger/TrackerDTC/plugins/ProducerED.cc rename to L1Trigger/TrackerDTC/plugins/ProducerDTC.cc index 8017496fc3358..8f50d1c28604a 100644 --- a/L1Trigger/TrackerDTC/plugins/ProducerED.cc +++ b/L1Trigger/TrackerDTC/plugins/ProducerDTC.cc @@ -16,6 +16,7 @@ #include "L1Trigger/TrackTrigger/interface/SensorModule.h" #include "L1Trigger/TrackerDTC/interface/LayerEncoding.h" #include "L1Trigger/TrackerDTC/interface/DTC.h" +#include "L1Trigger/TrackerTFP/interface/DataFormats.h" #include #include @@ -26,18 +27,19 @@ using namespace std; using namespace edm; using namespace tt; +using namespace trackerTFP; namespace trackerDTC { - /*! \class trackerDTC::ProducerED + /*! \class trackerDTC::ProducerDTC * \brief Class to produce hardware like structured TTStub Collection used by Track Trigger emulators * \author Thomas Schuh * \date 2020, Jan */ - class ProducerED : public stream::EDProducer<> { + class ProducerDTC : public stream::EDProducer<> { public: - explicit ProducerED(const ParameterSet&); - ~ProducerED() override {} + explicit ProducerDTC(const ParameterSet&); + ~ProducerDTC() override {} private: void beginRun(const Run&, const EventSetup&) override; @@ -45,6 +47,8 @@ namespace trackerDTC { void endJob() {} // helper class to store configurations const Setup* setup_ = nullptr; + // helper class to extract structured data from tt::Frames + const DataFormats* dataFormats_ = nullptr; // class to encode layer ids used between DTC and TFP in Hybrid const LayerEncoding* layerEncoding_ = nullptr; // ED input token of TTStubs @@ -55,13 +59,15 @@ namespace trackerDTC { EDPutTokenT edPutTokenLost_; // Setup token ESGetToken esGetTokenSetup_; + // DataFormats token + ESGetToken esGetTokenDataFormats_; // LayerEncoding token ESGetToken esGetTokenLayerEncoding_; // configuration ParameterSet iConfig_; }; - ProducerED::ProducerED(const ParameterSet& iConfig) : iConfig_(iConfig) { + ProducerDTC::ProducerDTC(const ParameterSet& iConfig) : iConfig_(iConfig) { // book in- and output ED products const auto& inputTag = iConfig.getParameter("InputTag"); const auto& branchAccepted = iConfig.getParameter("BranchAccepted"); @@ -71,48 +77,43 @@ namespace trackerDTC { edPutTokenLost_ = produces(branchLost); // book ES products esGetTokenSetup_ = esConsumes(); + esGetTokenDataFormats_ = esConsumes(); esGetTokenLayerEncoding_ = esConsumes(); } - void ProducerED::beginRun(const Run& iRun, const EventSetup& iSetup) { + void ProducerDTC::beginRun(const Run& iRun, const EventSetup& iSetup) { setup_ = &iSetup.getData(esGetTokenSetup_); - if (!setup_->configurationSupported()) - return; - // check process history if desired - if (iConfig_.getParameter("CheckHistory")) - setup_->checkHistory(iRun.processHistory()); + dataFormats_ = &iSetup.getData(esGetTokenDataFormats_); layerEncoding_ = &iSetup.getData(esGetTokenLayerEncoding_); } - void ProducerED::produce(Event& iEvent, const EventSetup& iSetup) { + void ProducerDTC::produce(Event& iEvent, const EventSetup& iSetup) { // empty DTC products TTDTC productAccepted = setup_->ttDTC(); TTDTC productLost = setup_->ttDTC(); - if (setup_->configurationSupported()) { - // read in stub collection - Handle handle; - iEvent.getByToken(edGetToken_, handle); - // apply cabling map, reorganise stub collections - vector>> stubsDTCs(setup_->numDTCs(), - vector>(setup_->numModulesPerDTC())); - for (auto module = handle->begin(); module != handle->end(); module++) { - // DetSetVec->detId + 1 = tk layout det id - const DetId detId = module->detId() + setup_->offsetDetIdDSV(); - // corresponding sensor module - SensorModule* sm = setup_->sensorModule(detId); - // empty stub collection - vector& stubsModule = stubsDTCs[sm->dtcId()][sm->modId()]; - stubsModule.reserve(module->size()); - for (TTStubDetSet::const_iterator ttStub = module->begin(); ttStub != module->end(); ttStub++) - stubsModule.emplace_back(makeRefTo(handle, ttStub)); - } - // board level processing - for (int dtcId = 0; dtcId < setup_->numDTCs(); dtcId++) { - // create single outer tracker DTC board - DTC dtc(iConfig_, setup_, layerEncoding_, dtcId, stubsDTCs.at(dtcId)); - // route stubs and fill products - dtc.produce(productAccepted, productLost); - } + // read in stub collection + Handle handle; + iEvent.getByToken(edGetToken_, handle); + // apply cabling map, reorganise stub collections + vector>> stubsDTCs(setup_->numDTCs(), + vector>(setup_->numModulesPerDTC())); + for (auto module = handle->begin(); module != handle->end(); module++) { + // DetSetVec->detId + 1 = tk layout det id + const DetId detId = module->detId() + setup_->offsetDetIdDSV(); + // corresponding sensor module + SensorModule* sm = setup_->sensorModule(detId); + // empty stub collection + vector& stubsModule = stubsDTCs[sm->dtcId()][sm->modId()]; + stubsModule.reserve(module->size()); + for (TTStubDetSet::const_iterator ttStub = module->begin(); ttStub != module->end(); ttStub++) + stubsModule.emplace_back(makeRefTo(handle, ttStub)); + } + // board level processing + for (int dtcId = 0; dtcId < setup_->numDTCs(); dtcId++) { + // create single outer tracker DTC board + DTC dtc(iConfig_, setup_, dataFormats_, layerEncoding_, dtcId, stubsDTCs.at(dtcId)); + // route stubs and fill products + dtc.produce(productAccepted, productLost); } // store ED products iEvent.emplace(edPutTokenAccepted_, std::move(productAccepted)); @@ -121,4 +122,4 @@ namespace trackerDTC { } // namespace trackerDTC -DEFINE_FWK_MODULE(trackerDTC::ProducerED); +DEFINE_FWK_MODULE(trackerDTC::ProducerDTC); diff --git a/L1Trigger/TrackerDTC/plugins/ProducerLayerEncoding.cc b/L1Trigger/TrackerDTC/plugins/ProducerLayerEncoding.cc index 2adbd96ce57fd..89dedb601c2c0 100644 --- a/L1Trigger/TrackerDTC/plugins/ProducerLayerEncoding.cc +++ b/L1Trigger/TrackerDTC/plugins/ProducerLayerEncoding.cc @@ -26,18 +26,17 @@ namespace trackerDTC { unique_ptr produce(const LayerEncodingRcd& rcd); private: - const ParameterSet iConfig_; ESGetToken esGetToken_; }; - ProducerLayerEncoding::ProducerLayerEncoding(const ParameterSet& iConfig) : iConfig_(iConfig) { + ProducerLayerEncoding::ProducerLayerEncoding(const ParameterSet& iConfig) { auto cc = setWhatProduced(this); esGetToken_ = cc.consumes(); } unique_ptr ProducerLayerEncoding::produce(const LayerEncodingRcd& rcd) { const Setup* setup = &rcd.get(esGetToken_); - return make_unique(iConfig_, setup); + return make_unique(setup); } } // namespace trackerDTC diff --git a/L1Trigger/TrackerDTC/python/AnalyzerDAQ_cff.py b/L1Trigger/TrackerDTC/python/AnalyzerDAQ_cff.py index 7c015b9333806..4b7b561e662ff 100644 --- a/L1Trigger/TrackerDTC/python/AnalyzerDAQ_cff.py +++ b/L1Trigger/TrackerDTC/python/AnalyzerDAQ_cff.py @@ -1,6 +1,8 @@ +# EDAnalyzer to analyze TTCluster Occupancies on DTCs, plots cluster occupancy + import FWCore.ParameterSet.Config as cms from L1Trigger.TrackerDTC.AnalyzerDAQ_cfi import TrackerDTCAnalyzerDAQ_params -from L1Trigger.TrackTrigger.ProducerSetup_cff import TrackTriggerSetup +from L1Trigger.TrackTrigger.Setup_cff import TrackTriggerSetup TrackerDTCAnalyzerDAQ = cms.EDAnalyzer('trackerDTC::AnalyzerDAQ', TrackerDTCAnalyzerDAQ_params) diff --git a/L1Trigger/TrackerDTC/python/AnalyzerDAQ_cfi.py b/L1Trigger/TrackerDTC/python/AnalyzerDAQ_cfi.py index 8c03b9126c3e8..9e8ed768f7ac5 100644 --- a/L1Trigger/TrackerDTC/python/AnalyzerDAQ_cfi.py +++ b/L1Trigger/TrackerDTC/python/AnalyzerDAQ_cfi.py @@ -1,3 +1,5 @@ +# configuration for TrackerDTCAnalyzerDAQ + import FWCore.ParameterSet.Config as cms TrackerDTCAnalyzerDAQ_params = cms.PSet ( diff --git a/L1Trigger/TrackerDTC/python/Analyzer_cff.py b/L1Trigger/TrackerDTC/python/Analyzer_cff.py index aaf563eac18ef..9a673c297a2bd 100644 --- a/L1Trigger/TrackerDTC/python/Analyzer_cff.py +++ b/L1Trigger/TrackerDTC/python/Analyzer_cff.py @@ -1,7 +1,9 @@ +# EDAnalyzer for hardware like structured TTStub Collection used by Track Trigger emulators, runs DTC stub emulation, plots performance & stub occupancy + import FWCore.ParameterSet.Config as cms from L1Trigger.TrackerDTC.Analyzer_cfi import TrackerDTCAnalyzer_params -from L1Trigger.TrackerDTC.ProducerED_cfi import TrackerDTCProducer_params -from L1Trigger.TrackTrigger.ProducerSetup_cff import TrackTriggerSetup +from L1Trigger.TrackerDTC.DTC_cfi import TrackerDTC_params +from L1Trigger.TrackTrigger.Setup_cff import TrackTriggerSetup -TrackerDTCAnalyzer = cms.EDAnalyzer('trackerDTC::Analyzer', TrackerDTCAnalyzer_params, TrackerDTCProducer_params) +AnalyzerDTC = cms.EDAnalyzer('trackerDTC::Analyzer', TrackerDTCAnalyzer_params, TrackerDTC_params) diff --git a/L1Trigger/TrackerDTC/python/Analyzer_cfi.py b/L1Trigger/TrackerDTC/python/Analyzer_cfi.py index 4b6a091d0884f..3f87cc24917d3 100644 --- a/L1Trigger/TrackerDTC/python/Analyzer_cfi.py +++ b/L1Trigger/TrackerDTC/python/Analyzer_cfi.py @@ -1,12 +1,13 @@ +# configuration for AnalyzerDTC + import FWCore.ParameterSet.Config as cms TrackerDTCAnalyzer_params = cms.PSet ( - InputTagAccepted = cms.InputTag( "TrackerDTCProducer", "StubAccepted" ), # dtc passed stubs selection - InputTagLost = cms.InputTag( "TrackerDTCProducer", "StubLost" ), # dtc lost stubs selection - InputTagTTStubDetSetVec = cms.InputTag( "TTStubsFromPhase2TrackerDigis", "StubAccepted" ), # original TTStub selection - InputTagTTClusterDetSetVec = cms.InputTag( "TTClustersFromPhase2TrackerDigis", "ClusterInclusive" ), # original TTCluster selection - InputTagTTClusterAssMap = cms.InputTag( "TTClusterAssociatorFromPixelDigis", "ClusterAccepted" ), # tag of AssociationMap between TTCluster and TrackingParticles - UseMCTruth = cms.bool( True ) # eneables analyze of TPs # eneables analyze of TPs + InputTagAccepted = cms.InputTag( "ProducerDTC", "StubAccepted" ), # dtc passed stubs selection + InputTagLost = cms.InputTag( "ProducerDTC", "StubLost" ), # dtc lost stubs selection + InputTagReconstructable = cms.InputTag( "StubAssociator", "Reconstructable" ), # + InputTagSelection = cms.InputTag( "StubAssociator", "UseForAlgEff" ), # + UseMCTruth = cms.bool( True ) # eneables analyze of TPs ) diff --git a/L1Trigger/TrackerDTC/python/Customize_cff.py b/L1Trigger/TrackerDTC/python/Customize_cff.py index b62f897ab5aac..4576ecac6515a 100644 --- a/L1Trigger/TrackerDTC/python/Customize_cff.py +++ b/L1Trigger/TrackerDTC/python/Customize_cff.py @@ -1,14 +1,22 @@ +# function to manipilate TrackerDTC emulator to match TMTT configuration and support TMTT data formats + import FWCore.ParameterSet.Config as cms def producerUseTMTT(process): - from L1Trigger.TrackerDTC.ProducerED_cfi import TrackerDTCProducer_params - TrackerDTCProducer_params.UseHybrid = cms.bool( False ) - process.TrackerDTCProducer = cms.EDProducer('trackerDTC::ProducerED', TrackerDTCProducer_params) + from L1Trigger.TrackerDTC.DTC_cfi import TrackerDTC_params + TrackerDTC_params.UseHybrid = False + process.TrackTriggerSetup.TrackFinding.MinPt = 3.0 + process.TrackTriggerSetup.TrackFinding.MaxEta = 2.4 + process.TrackTriggerSetup.TrackFinding.ChosenRofPhi = 67.24 + process.ProducerDTC = cms.EDProducer('trackerDTC::ProducerDTC', TrackerDTC_params) return process def analyzerUseTMTT(process): from L1Trigger.TrackerDTC.Analyzer_cfi import TrackerDTCAnalyzer_params - from L1Trigger.TrackerDTC.ProducerED_cfi import TrackerDTCProducer_params - TrackerDTCProducer_params.UseHybrid = cms.bool( False ) - process.TrackerDTCAnalyzer = cms.EDAnalyzer('trackerDTC::Analyzer', TrackerDTCAnalyzer_params, TrackerDTCProducer_params) + from L1Trigger.TrackerDTC.DTC_cfi import TrackerDTC_params + TrackerDTC_params.UseHybrid = False + process.TrackTriggerSetup.TrackFinding.MinPt = 3.0 + process.TrackTriggerSetup.TrackFinding.MaxEta = 2.4 + process.TrackTriggerSetup.TrackFinding.ChosenRofPhi = 67.24 + process.AnalyzerDTC = cms.EDAnalyzer('trackerDTC::Analyzer', TrackerDTCAnalyzer_params, TrackerDTC_params) return process diff --git a/L1Trigger/TrackerDTC/python/ProducerED_cff.py b/L1Trigger/TrackerDTC/python/DTC_cff.py similarity index 51% rename from L1Trigger/TrackerDTC/python/ProducerED_cff.py rename to L1Trigger/TrackerDTC/python/DTC_cff.py index e559be729703f..3fb8662f13b8d 100644 --- a/L1Trigger/TrackerDTC/python/ProducerED_cff.py +++ b/L1Trigger/TrackerDTC/python/DTC_cff.py @@ -6,8 +6,9 @@ #=== Import default values for all parameters & define EDProducer. -from L1Trigger.TrackerDTC.ProducerED_cfi import TrackerDTCProducer_params -from L1Trigger.TrackTrigger.ProducerSetup_cff import TrackTriggerSetup -from L1Trigger.TrackerDTC.ProducerLayerEncoding_cff import TrackerDTCLayerEncoding +from L1Trigger.TrackerDTC.DTC_cfi import TrackerDTC_params +from L1Trigger.TrackTrigger.Setup_cff import TrackTriggerSetup +from L1Trigger.TrackerDTC.LayerEncoding_cff import TrackerDTCLayerEncoding +from L1Trigger.TrackerTFP.DataFormats_cff import TrackTriggerDataFormats -TrackerDTCProducer = cms.EDProducer('trackerDTC::ProducerED', TrackerDTCProducer_params) +ProducerDTC = cms.EDProducer('trackerDTC::ProducerDTC', TrackerDTC_params) diff --git a/L1Trigger/TrackerDTC/python/ProducerED_cfi.py b/L1Trigger/TrackerDTC/python/DTC_cfi.py similarity index 77% rename from L1Trigger/TrackerDTC/python/ProducerED_cfi.py rename to L1Trigger/TrackerDTC/python/DTC_cfi.py index ba6303791d629..cdf8c69bc7075 100644 --- a/L1Trigger/TrackerDTC/python/ProducerED_cfi.py +++ b/L1Trigger/TrackerDTC/python/DTC_cfi.py @@ -1,11 +1,12 @@ +# configuration for ProducerDTC + import FWCore.ParameterSet.Config as cms -TrackerDTCProducer_params = cms.PSet ( +TrackerDTC_params = cms.PSet ( InputTag = cms.InputTag( "TTStubsFromPhase2TrackerDigis", "StubAccepted" ), # original TTStub selection BranchAccepted = cms.string ( "StubAccepted" ), # label for prodcut with passed stubs BranchLost = cms.string ( "StubLost" ), # label for prodcut with lost stubs - CheckHistory = cms.bool ( False ), # checks if input sample production is configured as current process UseHybrid = cms.bool ( True ), # use Hybrid or TMTT as TT algorithm EnableTruncation = cms.bool ( True ) # enable emulation of truncation, lost stubs are filled in BranchLost diff --git a/L1Trigger/TrackerDTC/python/LayerEncoding_cff.py b/L1Trigger/TrackerDTC/python/LayerEncoding_cff.py new file mode 100644 index 0000000000000..5f8ec932fb2d2 --- /dev/null +++ b/L1Trigger/TrackerDTC/python/LayerEncoding_cff.py @@ -0,0 +1,3 @@ +import FWCore.ParameterSet.Config as cms + +TrackerDTCLayerEncoding = cms.ESProducer("trackerDTC::ProducerLayerEncoding") \ No newline at end of file diff --git a/L1Trigger/TrackerDTC/python/ProducerLayerEncoding_cff.py b/L1Trigger/TrackerDTC/python/ProducerLayerEncoding_cff.py deleted file mode 100644 index 49935c483846d..0000000000000 --- a/L1Trigger/TrackerDTC/python/ProducerLayerEncoding_cff.py +++ /dev/null @@ -1,5 +0,0 @@ -import FWCore.ParameterSet.Config as cms - -from L1Trigger.TrackerDTC.ProducerLayerEncoding_cfi import TrackerDTCLayerEncoding_params - -TrackerDTCLayerEncoding = cms.ESProducer("trackerDTC::ProducerLayerEncoding", TrackerDTCLayerEncoding_params) diff --git a/L1Trigger/TrackerDTC/python/ProducerLayerEncoding_cfi.py b/L1Trigger/TrackerDTC/python/ProducerLayerEncoding_cfi.py deleted file mode 100644 index 8f82c976fe812..0000000000000 --- a/L1Trigger/TrackerDTC/python/ProducerLayerEncoding_cfi.py +++ /dev/null @@ -1,7 +0,0 @@ -import FWCore.ParameterSet.Config as cms - -TrackerDTCLayerEncoding_params = cms.PSet ( - - - -) diff --git a/L1Trigger/TrackerDTC/src/DTC.cc b/L1Trigger/TrackerDTC/src/DTC.cc index babcea183e29f..5884cc8efb2e7 100644 --- a/L1Trigger/TrackerDTC/src/DTC.cc +++ b/L1Trigger/TrackerDTC/src/DTC.cc @@ -8,15 +8,18 @@ using namespace std; using namespace edm; using namespace tt; +using namespace trackerTFP; namespace trackerDTC { DTC::DTC(const ParameterSet& iConfig, const Setup* setup, + const DataFormats* dataFormats, const LayerEncoding* layerEncoding, int dtcId, - const std::vector>& stubsDTC) + const vector>& stubsDTC) : setup_(setup), + dataFormats_(dataFormats), enableTruncation_(iConfig.getParameter("EnableTruncation")), region_(dtcId / setup->numDTCsPerRegion()), board_(dtcId % setup->numDTCsPerRegion()), @@ -33,7 +36,7 @@ namespace trackerDTC { if (ttStubRefs.empty()) continue; // Module which produced this ttStubRefs - SensorModule* module = modules_.at(modId); + const SensorModule* module = modules_.at(modId); // DTC routing block id [0-1] const int blockId = modId / setup->dtcNumModulesPerRoutingBlock(); // DTC routing blockc channel id [0-35] @@ -41,7 +44,7 @@ namespace trackerDTC { // convert TTStubs and fill input channel Stubs& stubs = input_[blockId][channelId]; for (const TTStubRef& ttStubRef : ttStubRefs) { - stubs_.emplace_back(iConfig, setup, layerEncoding, module, ttStubRef); + stubs_.emplace_back(iConfig, setup, dataFormats, layerEncoding, module, ttStubRef); Stub& stub = stubs_.back(); if (stub.valid()) // passed pt and eta cut @@ -120,8 +123,8 @@ namespace trackerDTC { output.push_back(nullptr); } // truncate if desired - if (enableTruncation_ && (int)output.size() > setup_->numFramesIO()) { - const auto limit = next(output.begin(), setup_->numFramesIO()); + if (enableTruncation_ && (int)output.size() > setup_->numFramesIOHigh()) { + const auto limit = next(output.begin(), setup_->numFramesIOHigh()); copy_if(limit, output.end(), back_inserter(lost), [](Stub* stub) { return stub; }); output.erase(limit, output.end()); } @@ -151,9 +154,7 @@ namespace trackerDTC { // conversion from Stubss to TTDTC void DTC::produce(const Stubss& stubss, TTDTC& product) { int channel(0); - auto toFrame = [&channel](Stub* stub) { - return stub ? make_pair(stub->ttStubRef(), stub->frame(channel)) : FrameStub(); - }; + auto toFrame = [&channel](Stub* stub) { return stub ? stub->frame(channel) : FrameStub(); }; for (const Stubs& stubs : stubss) { StreamStub stream; stream.reserve(stubs.size()); diff --git a/L1Trigger/TrackerDTC/src/LayerEncoding.cc b/L1Trigger/TrackerDTC/src/LayerEncoding.cc index c72d8968c038e..4802bdca31dca 100644 --- a/L1Trigger/TrackerDTC/src/LayerEncoding.cc +++ b/L1Trigger/TrackerDTC/src/LayerEncoding.cc @@ -13,8 +13,7 @@ using namespace tt; namespace trackerDTC { - LayerEncoding::LayerEncoding(const ParameterSet& iConfig, const Setup* setup) - : setup_(setup), numDTCsPerRegion_(setup->numDTCsPerRegion()) { + LayerEncoding::LayerEncoding(const Setup* setup) : setup_(setup), numDTCsPerRegion_(setup->numDTCsPerRegion()) { encodingsLayerId_.reserve(numDTCsPerRegion_); for (int dtcInRegion = 0; dtcInRegion < setup->numDTCsPerRegion(); dtcInRegion++) { set encodingLayerId; @@ -36,7 +35,7 @@ namespace trackerDTC { } // decode layer id for given sensor module - int LayerEncoding::decode(SensorModule* sm) const { + int LayerEncoding::decode(const SensorModule* sm) const { const vector& encoding = encodingsLayerId_.at(sm->dtcId() % setup_->numDTCsPerRegion()); const auto pos = find(encoding.begin(), encoding.end(), sm->layerId()); return distance(encoding.begin(), pos); diff --git a/L1Trigger/TrackerDTC/src/Stub.cc b/L1Trigger/TrackerDTC/src/Stub.cc index 3d8fda73d2574..7775e53277f74 100644 --- a/L1Trigger/TrackerDTC/src/Stub.cc +++ b/L1Trigger/TrackerDTC/src/Stub.cc @@ -8,129 +8,128 @@ using namespace edm; using namespace std; using namespace tt; +using namespace trackerTFP; namespace trackerDTC { - Stub::Stub(const ParameterSet& iConfig, - const Setup* setup, - const LayerEncoding* layerEncoding, - SensorModule* sm, - const TTStubRef& ttStubRef) - : setup_(setup), - layerEncoding_(layerEncoding), + Stub::Stub(const DataFormats* dataFormats, const SensorModule* sm, const TTStubRef& ttStubRef) + : setup_(dataFormats->setup()), + dataFormats_(dataFormats), + layerEncoding_(nullptr), sm_(sm), ttStubRef_(ttStubRef), - hybrid_(iConfig.getParameter("UseHybrid")), - valid_(true) { - regions_.reserve(setup->numOverlappingRegions()); + hybrid_(false), + valid_(true), + regions_(0, setup_->numOverlappingRegions()) { + const DataFormat& dfR = dataFormats_->format(Variable::r, Process::dtc); + const DataFormat& dfPhi = dataFormats_->format(Variable::phi, Process::dtc); + const DataFormat& dfZ = dataFormats_->format(Variable::z, Process::dtc); + const DataFormat& dfInv2R = dataFormats_->format(Variable::inv2R, Process::ht); // get stub local coordinates const MeasurementPoint& mp = ttStubRef->clusterRef(0)->findAverageLocalCoordinatesCentered(); - // convert to uniformed local coordinates - // column number in pitch units - col_ = (int)floor(pow(-1, sm->signCol()) * (mp.y() - sm->numColumns() / 2) / setup->baseCol()); + col_ = (int)floor(pow(-1, sm_->signCol()) * (mp.y() - sm_->numColumns() / 2) / setup_->baseCol()); // row number in half pitch units - row_ = (int)floor(pow(-1, sm->signRow()) * (mp.x() - sm->numRows() / 2) / setup->baseRow()); + row_ = (int)floor(pow(-1, sm_->signRow()) * (mp.x() - sm_->numRows() / 2) / setup_->baseRow()); // bend number in quarter pitch units - bend_ = (int)floor(pow(-1, sm->signBend()) * (ttStubRef->bendBE()) / setup->baseBend()); + bend_ = (int)floor(pow(-1, sm_->signBend()) * (ttStubRef->bendBE()) / setup_->baseBend()); // reduced row number for look up - rowLUT_ = (int)floor((double)row_ / pow(2., setup->widthRow() - setup->dtcWidthRowLUT())); + rowLUT_ = (int)floor((double)row_ / pow(2., setup_->widthRow() - setup_->dtcWidthRowLUT())); // sub row number inside reduced row number - rowSub_ = row_ - (rowLUT_ + .5) * pow(2, setup->widthRow() - setup->dtcWidthRowLUT()); - + rowSub_ = row_ - (rowLUT_ + .5) * pow(2, setup_->widthRow() - setup_->dtcWidthRowLUT()); // convert local to global coordinates - - const double y = (col_ + .5) * setup->baseCol() * sm->pitchCol(); + const double y = (col_ + .5) * setup_->baseCol() * sm_->pitchCol(); // radius of a column of strips/pixel in cm - d_ = sm->r() + y * sm->sinTilt(); + d_ = sm_->r() + y * sm_->sinTilt(); // stub z in cm - z_ = digi(sm->z() + y * sm->cosTilt(), setup->tmttBaseZ()); - - const double x0 = rowLUT_ * setup->baseRow() * setup->dtcNumMergedRows() * sm->pitchRow(); - const double x1 = (rowLUT_ + 1) * setup->baseRow() * setup->dtcNumMergedRows() * sm->pitchRow(); - const double x = (rowLUT_ + .5) * setup->baseRow() * setup->dtcNumMergedRows() * sm->pitchRow(); - // stub r in cm - r_ = sqrt(d_ * d_ + x * x); - - const double phi0 = sm->phi() + atan2(x0, d_); - const double phi1 = sm->phi() + atan2(x1, d_); + z_ = dfZ.digi(sm_->z() + y * sm_->cosTilt()); + const double x = (rowLUT_ + .5) * setup_->baseRow() * setup_->dtcNumMergedRows() * sm_->pitchRow(); + // stub r wrt chosen RofPhi in cm + r_ = dfR.digi(sqrt(d_ * d_ + x * x) - setup_->chosenRofPhi()); + const double x0 = rowLUT_ * setup_->baseRow() * setup_->dtcNumMergedRows() * sm_->pitchRow(); + const double x1 = (rowLUT_ + 1) * setup_->baseRow() * setup_->dtcNumMergedRows() * sm_->pitchRow(); + const double phi0 = sm_->phi() + atan2(x0, d_); + const double phi1 = sm_->phi() + atan2(x1, d_); const double c = (phi0 + phi1) / 2.; - const double m = (phi1 - phi0) / setup->dtcNumMergedRows(); - + const double m = (phi1 - phi0) / setup_->dtcNumMergedRows(); // intercept of linearized stub phi in rad - c_ = digi(c, setup->tmttBasePhi()); + c_ = digi(c, dfPhi.base() / 2.); // slope of linearized stub phi in rad / strip - m_ = digi(m, setup->dtcBaseM()); - - if (hybrid_) { - if (abs(z_ / r_) > setup->hybridMaxCot()) - // did not pass eta cut - valid_ = false; - } else { - // extrapolated z at radius T assuming z0=0 - const double zT = setup->chosenRofZ() * z_ / r_; - // extrapolated z0 window at radius T - const double dZT = setup->beamWindowZ() * abs(1. - setup->chosenRofZ() / r_); - double zTMin = zT - dZT; - double zTMax = zT + dZT; - if (zTMin >= setup->maxZT() || zTMax < -setup->maxZT()) - // did not pass "eta" cut - valid_ = false; - else { - zTMin = max(zTMin, -setup->maxZT()); - zTMax = min(zTMax, setup->maxZT()); - } - // range of stub cot(theta) - cot_ = {zTMin / setup->chosenRofZ(), zTMax / setup->chosenRofZ()}; - } - - // stub r w.r.t. chosenRofPhi in cm - static const double chosenRofPhi = hybrid_ ? setup->hybridChosenRofPhi() : setup->chosenRofPhi(); - r_ = digi(r_ - chosenRofPhi, setup->tmttBaseR()); - + m_ = digi(m, setup_->dtcBaseM()); + // stub phi w.r.t. detector region centre in rad + phi_ = dfPhi.digi(c_ + rowSub_ * m_); + // assaign stub to processing regions // radial (cylindrical) component of sensor separation - const double dr = sm->sep() / (sm->cosTilt() - sm->sinTilt() * z_ / d_); + const double dr = sm_->sep() / (sm_->cosTilt() - sm_->sinTilt() * z_ / d_); // converts bend into inv2R in 1/cm - const double inv2ROverBend = sm->pitchRow() / dr / d_; + const double inv2ROverBend = sm_->pitchRow() / dr / d_; // inv2R in 1/cm - const double inv2R = -bend_ * setup->baseBend() * inv2ROverBend; + const double inv2R = -bend_ * setup_->baseBend() * inv2ROverBend; // inv2R uncertainty in 1/cm - const double dInv2R = setup->bendCut() * inv2ROverBend; - const double minPt = hybrid_ ? setup->hybridMinPtStub() : setup->minPt(); - const double maxInv2R = setup->invPtToDphi() / minPt - setup->dtcBaseInv2R() / 2.; - double inv2RMin = digi(inv2R - dInv2R, setup->dtcBaseInv2R()); - double inv2RMax = digi(inv2R + dInv2R, setup->dtcBaseInv2R()); - if (inv2RMin > maxInv2R || inv2RMax < -maxInv2R) { - // did not pass pt cut + const double dInv2R = setup_->bendCut() * inv2ROverBend; + inv2R_.first = dfInv2R.digi(inv2R - dInv2R); + inv2R_.second = dfInv2R.digi(inv2R + dInv2R); + //static const double maxInv2R = dfInv2R.limit(); + static const double maxInv2R = dfInv2R.range() / 2.; + // cut on pt + if (inv2R_.first > maxInv2R || inv2R_.second < -maxInv2R) valid_ = false; - } else { - inv2RMin = max(inv2RMin, -maxInv2R); - inv2RMax = min(inv2RMax, maxInv2R); + else { + inv2R_.first = max(inv2R_.first, -maxInv2R); + inv2R_.second = min(inv2R_.second, maxInv2R); } - // range of stub inv2R in 1/cm - inv2R_ = {inv2RMin, inv2RMax}; - - // stub phi w.r.t. detector region centre in rad - phi_ = c_ + rowSub_ * m_; - // range of stub extrapolated phi to radius chosenRofPhi in rad phiT_.first = phi_ - r_ * inv2R_.first; phiT_.second = phi_ - r_ * inv2R_.second; if (phiT_.first > phiT_.second) swap(phiT_.first, phiT_.second); - if (phiT_.first < 0.) - regions_.push_back(0); + regions_.set(0); if (phiT_.second >= 0.) - regions_.push_back(1); + regions_.set(1); + } + Stub::Stub(const ParameterSet& iConfig, + const Setup* setup, + const DataFormats* dataFormats, + const LayerEncoding* layerEncoding, + const SensorModule* sm, + const TTStubRef& ttStubRef) + : setup_(setup), + dataFormats_(dataFormats), + layerEncoding_(layerEncoding), + sm_(sm), + ttStubRef_(ttStubRef), + hybrid_(iConfig.getParameter("UseHybrid")) { + const Stub stub(dataFormats, sm, ttStubRef); + bend_ = stub.bend_; + valid_ = stub.valid_; + row_ = stub.row_; + col_ = stub.col_; + r_ = stub.r_; + phi_ = stub.phi_; + z_ = stub.z_; + phiT_ = stub.phiT_; + inv2R_ = stub.inv2R_; + regions_ = stub.regions_; + // apply "eta" cut + const DataFormat& dfZT = dataFormats->format(Variable::zT, Process::gp); + const double r = r_ + setup->chosenRofPhi(); + const double ratioRZ = setup->chosenRofZ() / r; + // extrapolated z at radius T assuming z0=0 + const double zT = z_ * ratioRZ; + // extrapolated z0 window at radius T + const double dZT = setup->beamWindowZ() * abs(1. - ratioRZ); + zT_ = {zT - dZT, zT + dZT}; + //if (abs(zT) > dfZT.limit() + dZT) + if (abs(zT) > dfZT.range() / 2. + dZT) + valid_ = false; // apply data format specific manipulations if (!hybrid_) return; - // stub r w.r.t. an offset in cm - r_ -= sm->offsetR() - chosenRofPhi; + r_ -= sm->offsetR() - setup->chosenRofPhi(); // stub z w.r.t. an offset in cm z_ -= sm->offsetZ(); if (sm->type() == SensorModule::Disk2S) { @@ -138,7 +137,6 @@ namespace trackerDTC { r_ = sm->encodedR() + (sm->side() ? -col_ : (col_ + sm->numColumns() / 2)); r_ = (r_ + 0.5) * setup->hybridBaseR(sm->type()); } - // encode bend const vector& encodingBend = setup->encodingBend(sm->windowSize(), sm->psModule()); const auto pos = find(encodingBend.begin(), encodingBend.end(), abs(ttStubRef->bendBE())); @@ -147,13 +145,14 @@ namespace trackerDTC { } // returns bit accurate representation of Stub - Frame Stub::frame(int region) const { return hybrid_ ? formatHybrid(region) : formatTMTT(region); } - - // returns true if stub belongs to region - bool Stub::inRegion(int region) const { return find(regions_.begin(), regions_.end(), region) != regions_.end(); } + FrameStub Stub::frame(int region) const { + return make_pair(ttStubRef_, hybrid_ ? formatHybrid(region) : formatTMTT(region)); + } // truncates double precision to f/w integer equivalent - double Stub::digi(double value, double precision) const { return (floor(value / precision) + .5) * precision; } + double Stub::digi(double value, double precision) const { + return (floor(value / precision + 1.e-12) + .5) * precision; + } // returns 64 bit stub in hybrid data format Frame Stub::formatHybrid(int region) const { @@ -162,11 +161,10 @@ namespace trackerDTC { const int decodedLayerId = layerEncoding_->decode(sm_); // stub phi w.r.t. processing region border in rad double phi = phi_ - (region - .5) * setup_->baseRegion() + setup_->hybridRangePhi() / 2.; - if (phi >= setup_->hybridRangePhi()) - phi = setup_->hybridRangePhi() - setup_->hybridBasePhi(type) / 2.; // convert stub variables into bit vectors - const TTBV hwR(r_, setup_->hybridBaseR(type), setup_->hybridWidthR(type), true); - const TTBV hwPhi(phi, setup_->hybridBasePhi(type), setup_->hybridWidthPhi(type), true); + const bool twosR = type == SensorModule::BarrelPS || type == SensorModule::Barrel2S; + const TTBV hwR(r_, setup_->hybridBaseR(type), setup_->hybridWidthR(type), twosR); + const TTBV hwPhi(phi, setup_->hybridBasePhi(type), setup_->hybridWidthPhi(type)); const TTBV hwZ(z_, setup_->hybridBaseZ(type), setup_->hybridWidthZ(type), true); const TTBV hwAlpha(row_, setup_->hybridBaseAlpha(type), setup_->hybridWidthAlpha(type), true); const TTBV hwBend(bend_, setup_->hybridWidthBend(type), true); @@ -179,74 +177,33 @@ namespace trackerDTC { } Frame Stub::formatTMTT(int region) const { - int layerM = sm_->layerId(); - // convert unique layer id [1-6,11-15] into reduced layer id [0-6] - // a fiducial track may not cross more then 7 detector layers, for stubs from a given track the reduced layer id is actually unique - int layer(-1); - if (layerM == 1) - layer = 0; - else if (layerM == 2) - layer = 1; - else if (layerM == 6 || layerM == 11) - layer = 2; - else if (layerM == 5 || layerM == 12) - layer = 3; - else if (layerM == 4 || layerM == 13) - layer = 4; - else if (layerM == 14) - layer = 5; - else if (layerM == 3 || layerM == 15) - layer = 6; - // assign stub to phi sectors within a processing region, to be generalized - TTBV sectorsPhi(0, setup_->numOverlappingRegions() * setup_->numSectorsPhi()); - if (phiT_.first < 0.) { - if (phiT_.first < -setup_->baseSector()) - sectorsPhi.set(0); - else - sectorsPhi.set(1); - if (phiT_.second < 0. && phiT_.second >= -setup_->baseSector()) - sectorsPhi.set(1); - } - if (phiT_.second >= 0.) { - if (phiT_.second < setup_->baseSector()) - sectorsPhi.set(2); - else - sectorsPhi.set(3); - if (phiT_.first >= 0. && phiT_.first < setup_->baseSector()) - sectorsPhi.set(2); - } - // assign stub to eta sectors within a processing region - pair sectorEta({0, setup_->numSectorsEta() - 1}); - for (int bin = 0; bin < setup_->numSectorsEta(); bin++) - if (asinh(cot_.first) < setup_->boundarieEta(bin + 1)) { - sectorEta.first = bin; - break; - } - for (int bin = sectorEta.first; bin < setup_->numSectorsEta(); bin++) - if (asinh(cot_.second) < setup_->boundarieEta(bin + 1)) { - sectorEta.second = bin; - break; - } - // stub phi w.r.t. processing region centre in rad - const double phi = phi_ - (region - .5) * setup_->baseRegion(); - // convert stub variables into bit vectors - const TTBV hwValid(1, 1); - const TTBV hwGap(0, setup_->tmttNumUnusedBits()); - const TTBV hwLayer(layer, setup_->tmttWidthLayer()); - const TTBV hwSectorEtaMin(sectorEta.first, setup_->tmttWidthSectorEta()); - const TTBV hwSectorEtaMax(sectorEta.second, setup_->tmttWidthSectorEta()); - const TTBV hwR(r_, setup_->tmttBaseR(), setup_->tmttWidthR(), true); - const TTBV hwPhi(phi, setup_->tmttBasePhi(), setup_->tmttWidthPhi(), true); - const TTBV hwZ(z_, setup_->tmttBaseZ(), setup_->tmttWidthZ(), true); - const TTBV hwInv2RMin(inv2R_.first, setup_->tmttBaseInv2R(), setup_->tmttWidthInv2R(), true); - const TTBV hwInv2RMax(inv2R_.second, setup_->tmttBaseInv2R(), setup_->tmttWidthInv2R(), true); - TTBV hwSectorPhis(0, setup_->numSectorsPhi()); - for (int sectorPhi = 0; sectorPhi < setup_->numSectorsPhi(); sectorPhi++) - hwSectorPhis[sectorPhi] = sectorsPhi[region * setup_->numSectorsPhi() + sectorPhi]; - // assemble final bitset - return Frame(hwGap.str() + hwValid.str() + hwR.str() + hwPhi.str() + hwZ.str() + hwLayer.str() + - hwSectorPhis.str() + hwSectorEtaMin.str() + hwSectorEtaMax.str() + hwInv2RMin.str() + - hwInv2RMax.str()); + static const DataFormat& dfInv2R = dataFormats_->format(Variable::inv2R, Process::ht); + static const DataFormat& dfPhiT = dataFormats_->format(Variable::phiT, Process::gp); + static const DataFormat& dfZT = dataFormats_->format(Variable::zT, Process::gp); + const double offset = (region - .5) * dfPhiT.range(); + const double r = r_; + const double phi = phi_ - offset; + const double z = z_; + const int indexLayerId = setup_->indexLayerId(ttStubRef_); + TTBV layer(indexLayerId, dataFormats_->width(Variable::layer, Process::dtc)); + if (sm_->barrel()) { + layer.set(4); + if (sm_->tilted()) + layer.set(3); + } else if (sm_->psModule()) + layer.set(3); + int phiTMin = max(dfPhiT.integer(phiT_.first - offset), -setup_->gpNumBinsPhiT() / 2); + int phiTMax = min(dfPhiT.integer(phiT_.second - offset), setup_->gpNumBinsPhiT() / 2 - 1); + if (phiTMin > setup_->gpNumBinsPhiT() / 2 - 1) + phiTMin = setup_->gpNumBinsPhiT() / 2 - 1; + if (phiTMax < -setup_->gpNumBinsPhiT() / 2) + phiTMax = -setup_->gpNumBinsPhiT() / 2; + const int zTMin = max(dfZT.integer(zT_.first), -setup_->gpNumBinsZT() / 2); + const int zTMax = min(dfZT.integer(zT_.second), setup_->gpNumBinsZT() / 2 - 1); + const int inv2RMin = max(dfInv2R.integer(inv2R_.first), -setup_->htNumBinsInv2R() / 2); + const int inv2RMax = min(dfInv2R.integer(inv2R_.second), setup_->htNumBinsInv2R() / 2 - 1); + const StubDTC stub(ttStubRef_, dataFormats_, r, phi, z, layer, phiTMin, phiTMax, zTMin, zTMax, inv2RMin, inv2RMax); + return stub.frame().second; } } // namespace trackerDTC diff --git a/L1Trigger/TrackerDTC/test/Analyzer.cc b/L1Trigger/TrackerDTC/test/Analyzer.cc index 7392f24abb066..4ef94a47cb0f4 100644 --- a/L1Trigger/TrackerDTC/test/Analyzer.cc +++ b/L1Trigger/TrackerDTC/test/Analyzer.cc @@ -10,18 +10,16 @@ #include "FWCore/Utilities/interface/InputTag.h" #include "FWCore/Utilities/interface/Exception.h" #include "CommonTools/UtilAlgos/interface/TFileService.h" -#include "SimDataFormats/TrackingAnalysis/interface/TrackingParticle.h" -#include "SimDataFormats/Associations/interface/TTClusterAssociationMap.h" -#include "SimTracker/Common/interface/TrackingParticleSelector.h" #include "DataFormats/DetId/interface/DetId.h" #include "DataFormats/Common/interface/Ptr.h" #include "DataFormats/Common/interface/Handle.h" -#include "DataFormats/L1TrackTrigger/interface/TTTypes.h" #include "DataFormats/L1TrackTrigger/interface/TTDTC.h" #include "DataFormats/GeometryVector/interface/GlobalPoint.h" #include "DataFormats/GeometrySurface/interface/Plane.h" #include "DataFormats/SiStripDetId/interface/StripSubdetector.h" +#include "SimTracker/TrackTriggerAssociation/interface/StubAssociation.h" +#include "SimTracker/TrackTriggerAssociation/interface/TTTypes.h" #include "L1Trigger/TrackTrigger/interface/Setup.h" #include "L1Trigger/TrackerDTC/interface/LayerEncoding.h" @@ -48,9 +46,6 @@ using namespace tt; namespace trackerDTC { - // mc truth types - typedef TTClusterAssociationMap TTClusterAssMap; - typedef edm::Ptr TPPtr; // stub resolution plots helper enum Resolution { R, Phi, Z, NumResolution }; constexpr initializer_list AllResolution = {R, Phi, Z}; @@ -77,51 +72,27 @@ namespace trackerDTC { void endJob() override; private: - // configuring track particle selector - void configTPSelector(); - // book histograms - void bookHistograms(); - // associate TPPtr with TTStubRef - void assoc(const Handle&, const Handle&, map>&); - // organize reconstrucable TrackingParticles used for efficiency measurements - void convert(const map>&, map>&); - // checks if a stub selection is considered reconstructable - bool reconstructable(const set& ttStubRefs) const; - // checks if TrackingParticle is selected for efficiency measurements - bool select(const TrackingParticle& tp) const; // fills kinematic tp histograms void fill(const TPPtr& tpPtr, const vector th1fs) const; - // analyze DTC products and find still reconstrucable TrackingParticles - void analyzeStubs(const TTDTC*, const TTDTC*, const map>&, map>&); // fill stub related histograms - void analyzeStream(const StreamStub& stream, int region, int channel, int& sum, TH2F* th2f); - // returns layerId [1-6, 11-15] of stub - int layerId(const TTStubRef& ttStubRef) const; - // analyze survived TPs - void analyzeTPs(const map>& mapTPsStubs); + void fill(const StreamStub& stream, int region, int channel, int& sum, TH2F* th2f); // prints out MC summary void endJobMC(); // prints out DTC summary void endJobDTC(); // ED input token of DTC stubs - EDGetTokenT getTokenTTDTCAccepted_; + EDGetTokenT edGetTokenTTDTCAccepted_; // ED input token of lost DTC stubs - EDGetTokenT getTokenTTDTCLost_; - // ED input token of TT stubs - EDGetTokenT getTokenTTStubDetSetVec_; - // ED input token of TTClsuter - EDGetTokenT getTokenTTClusterDetSetVec_; - // ED input token of TTCluster to TPPtr association - EDGetTokenT getTokenTTClusterAssMap_; + EDGetTokenT edGetTokenTTDTCLost_; + // ED input token of TTStubRef to TPPtr association for tracking efficiency + EDGetTokenT edGetTokenSelection_; + // ED input token of TTStubRef to recontructable TPPtr association + EDGetTokenT edGetTokenReconstructable_; // Setup token ESGetToken esGetToken_; // stores, calculates and provides run-time constants - const Setup* setup_ = nullptr; - // selector to partly select TPs for efficiency measurements - TrackingParticleSelector tpSelector_; - // - TrackingParticleSelector tpSelectorLoose_; + const Setup* setup_; // enables analyze of TPs bool useMCTruth_; // specifies used TT algorithm @@ -154,15 +125,13 @@ namespace trackerDTC { // book in- and output ED products const auto& inputTagAccepted = iConfig.getParameter("InputTagAccepted"); const auto& inputTagLost = iConfig.getParameter("InputTagLost"); - getTokenTTDTCAccepted_ = consumes(inputTagAccepted); - getTokenTTDTCLost_ = consumes(inputTagLost); + edGetTokenTTDTCAccepted_ = consumes(inputTagAccepted); + edGetTokenTTDTCLost_ = consumes(inputTagLost); if (useMCTruth_) { - const auto& inputTagTTStubDetSetVec = iConfig.getParameter("InputTagTTStubDetSetVec"); - const auto& inputTagTTClusterDetSetVec = iConfig.getParameter("InputTagTTClusterDetSetVec"); - const auto& inputTagTTClusterAssMap = iConfig.getParameter("InputTagTTClusterAssMap"); - getTokenTTStubDetSetVec_ = consumes(inputTagTTStubDetSetVec); - getTokenTTClusterDetSetVec_ = consumes(inputTagTTClusterDetSetVec); - getTokenTTClusterAssMap_ = consumes(inputTagTTClusterAssMap); + const auto& inputTagSelection = iConfig.getParameter("InputTagSelection"); + const auto& inputTagReconstructable = iConfig.getParameter("InputTagReconstructable"); + edGetTokenSelection_ = consumes(inputTagSelection); + edGetTokenReconstructable_ = consumes(inputTagReconstructable); } // book ES product esGetToken_ = esConsumes(); @@ -174,42 +143,130 @@ namespace trackerDTC { void Analyzer::beginRun(const Run& iEvent, const EventSetup& iSetup) { // helper class to store configurations setup_ = &iSetup.getData(esGetToken_); - // configuring track particle selector - configTPSelector(); // book histograms - bookHistograms(); + Service fs; + TFileDirectory dir; + // mc + dir = fs->mkdir("MC"); + profMC_ = dir.make("Counts", ";", 6, 0.5, 6.5); + profMC_->GetXaxis()->SetBinLabel(1, "Stubs"); + profMC_->GetXaxis()->SetBinLabel(2, "Matched Stubs"); + profMC_->GetXaxis()->SetBinLabel(3, "reco TPs"); + profMC_->GetXaxis()->SetBinLabel(4, "eff TPs"); + profMC_->GetXaxis()->SetBinLabel(5, "total eff TPs"); + profMC_->GetXaxis()->SetBinLabel(6, "Cluster"); + constexpr array binsEff{{9 * 8, 10, 16, 10, 30, 24}}; + constexpr array, NumEfficiency> rangesEff{ + {{-M_PI, M_PI}, {0., 100.}, {-1. / 3., 1. / 3.}, {-5., 5.}, {-15., 15.}, {-2.4, 2.4}}}; + if (useMCTruth_) { + hisEffMC_.reserve(NumEfficiency); + for (Efficiency e : AllEfficiency) + hisEffMC_.emplace_back( + dir.make(("HisTP" + name(e)).c_str(), ";", binsEff[e], rangesEff[e].first, rangesEff[e].second)); + } + // dtc + dir = fs->mkdir("DTC"); + profDTC_ = dir.make("Counts", ";", 3, 0.5, 3.5); + profDTC_->GetXaxis()->SetBinLabel(1, "Stubs"); + profDTC_->GetXaxis()->SetBinLabel(2, "Lost Stubs"); + profDTC_->GetXaxis()->SetBinLabel(3, "TPs"); + // channel occupancy + constexpr int maxOcc = 180; + const int numChannels = setup_->numDTCs() * setup_->numOverlappingRegions(); + hisChannel_ = dir.make("His Channel Occupancy", ";", maxOcc, -.5, maxOcc - .5); + profChannel_ = dir.make("Prof Channel Occupancy", ";", numChannels, -.5, numChannels - .5); + // max tracking efficiencies + if (useMCTruth_) { + dir = fs->mkdir("DTC/Effi"); + hisEff_.reserve(NumEfficiency); + for (Efficiency e : AllEfficiency) + hisEff_.emplace_back( + dir.make(("HisTP" + name(e)).c_str(), ";", binsEff[e], rangesEff[e].first, rangesEff[e].second)); + eff_.reserve(NumEfficiency); + for (Efficiency e : AllEfficiency) + eff_.emplace_back( + dir.make(("Eff" + name(e)).c_str(), ";", binsEff[e], rangesEff[e].first, rangesEff[e].second)); + } + // lost stub fraction in r-z + dir = fs->mkdir("DTC/Loss"); + constexpr int bins = 400; + constexpr double maxZ = 300.; + constexpr double maxR = 120.; + hisRZStubs_ = dir.make("RZ Stubs", ";;", bins, -maxZ, maxZ, bins, 0., maxR); + hisRZStubsLost_ = dir.make("RZ Stubs Lost", ";;", bins, -maxZ, maxZ, bins, 0., maxR); + hisRZStubsEff_ = dir.make("RZ Stubs Eff", ";;", bins, -maxZ, maxZ, bins, 0., maxR); + // stub parameter resolutions + dir = fs->mkdir("DTC/Res"); + constexpr array ranges{{.2, .0001, .5}}; + constexpr int binsHis = 100; + hisResolution_.reserve(NumResolution); + profResolution_.reserve(NumResolution); + for (Resolution r : AllResolution) { + hisResolution_.emplace_back(dir.make(("HisRes" + name(r)).c_str(), ";", binsHis, -ranges[r], ranges[r])); + profResolution_.emplace_back( + dir.make(("ProfRes" + name(r)).c_str(), ";;", bins, -maxZ, maxZ, bins, 0., maxR)); + } } void Analyzer::analyze(const Event& iEvent, const EventSetup& iSetup) { - // read in TrackingParticle - map> mapAllStubsTPs; - if (useMCTruth_) { - Handle handleTTStubDetSetVec; - iEvent.getByToken(getTokenTTStubDetSetVec_, handleTTStubDetSetVec); - Handle handleTTClusterAssMap; - iEvent.getByToken(getTokenTTClusterAssMap_, handleTTClusterAssMap); - // associate TPPtr with TTStubRef - map> mapAllTPsAllStubs; - assoc(handleTTStubDetSetVec, handleTTClusterAssMap, mapAllTPsAllStubs); - // organize reconstrucable TrackingParticles used for efficiency measurements - convert(mapAllTPsAllStubs, mapAllStubsTPs); - Handle handleTTClusterDetSetVec; - iEvent.getByToken(getTokenTTClusterDetSetVec_, handleTTClusterDetSetVec); - int nCluster(0); - for (const auto& detSet : *handleTTClusterDetSetVec) - nCluster += detSet.size(); - profMC_->Fill(6, nCluster / (double)setup_->numRegions()); - } // read in dtc products Handle handleTTDTCAccepted; - iEvent.getByToken(getTokenTTDTCAccepted_, handleTTDTCAccepted); + iEvent.getByToken(edGetTokenTTDTCAccepted_, handleTTDTCAccepted); Handle handleTTDTCLost; - iEvent.getByToken(getTokenTTDTCLost_, handleTTDTCLost); - map> mapTPsTTStubs; - // analyze DTC products and find still reconstrucable TrackingParticles - analyzeStubs(handleTTDTCAccepted.product(), handleTTDTCLost.product(), mapAllStubsTPs, mapTPsTTStubs); - // analyze survived TPs - analyzeTPs(mapTPsTTStubs); + iEvent.getByToken(edGetTokenTTDTCLost_, handleTTDTCLost); + // read in MCTruth + const StubAssociation* selection = nullptr; + const StubAssociation* reconstructable = nullptr; + if (useMCTruth_) { + Handle handleSelection; + iEvent.getByToken(edGetTokenSelection_, handleSelection); + selection = handleSelection.product(); + Handle handleReconstructable; + iEvent.getByToken(edGetTokenReconstructable_, handleReconstructable); + reconstructable = handleReconstructable.product(); + profMC_->Fill(3, reconstructable->numTPs() / (double)setup_->numRegions()); + profMC_->Fill(4, selection->numTPs() / (double)setup_->numRegions()); + profMC_->Fill(5, selection->numTPs()); + for (const auto& p : selection->getTrackingParticleToTTStubsMap()) + fill(p.first, hisEffMC_); + } + // analyze dtc products and find still reconstrucable TrackingParticles + set tpPtrs; + for (int region = 0; region < setup_->numRegions(); region++) { + int nStubs(0); + int nLost(0); + map> mapTPsTTStubs; + for (int channel = 0; channel < setup_->numDTCsPerTFP(); channel++) { + const StreamStub& accepted = handleTTDTCAccepted->stream(region, channel); + const StreamStub& lost = handleTTDTCLost->stream(region, channel); + hisChannel_->Fill(accepted.size()); + profChannel_->Fill(channel, accepted.size()); + fill(accepted, region, channel, nStubs, hisRZStubs_); + fill(lost, region, channel, nLost, hisRZStubsLost_); + if (!useMCTruth_) + continue; + for (const FrameStub& frame : accepted) { + if (frame.first.isNull()) + continue; + for (const TPPtr& tpPtr : selection->findTrackingParticlePtrs(frame.first)) { + auto it = mapTPsTTStubs.find(tpPtr); + if (it == mapTPsTTStubs.end()) { + it = mapTPsTTStubs.emplace(tpPtr, vector()).first; + it->second.reserve(selection->findTTStubRefs(tpPtr).size()); + } + it->second.push_back(frame.first); + } + } + for (const auto& p : mapTPsTTStubs) + if (setup_->reconstructable(p.second)) + tpPtrs.insert(p.first); + } + profDTC_->Fill(1, nStubs); + profDTC_->Fill(2, nLost); + } + for (const TPPtr& tpPtr : tpPtrs) + fill(tpPtr, hisEff_); + profDTC_->Fill(3, tpPtrs.size()); nEvents_++; } @@ -235,86 +292,7 @@ namespace trackerDTC { // printout DTC summary endJobDTC(); log_ << "============================================================="; - LogPrint("L1Trigger/TrackerDTC") << log_.str(); - } - - // associate TPPtr with TTStubRef - void Analyzer::assoc(const Handle& handleTTStubDetSetVec, - const Handle& handleTTClusterAssMap, - map>& mapTPsStubs) { - int nStubs(0); - int nStubsMatched(0); - for (TTStubDetSetVec::const_iterator ttModule = handleTTStubDetSetVec->begin(); - ttModule != handleTTStubDetSetVec->end(); - ttModule++) { - nStubs += ttModule->size(); - for (TTStubDetSet::const_iterator ttStub = ttModule->begin(); ttStub != ttModule->end(); ttStub++) { - set tpPtrs; - for (unsigned int iClus = 0; iClus < 2; iClus++) { - const vector& assocPtrs = handleTTClusterAssMap->findTrackingParticlePtrs(ttStub->clusterRef(iClus)); - copy_if(assocPtrs.begin(), assocPtrs.end(), inserter(tpPtrs, tpPtrs.begin()), [](const TPPtr& tpPtr) { - return tpPtr.isNonnull(); - }); - } - for (const TPPtr& tpPtr : tpPtrs) - mapTPsStubs[tpPtr].emplace(makeRefTo(handleTTStubDetSetVec, ttStub)); - if (!tpPtrs.empty()) - nStubsMatched++; - } - } - profMC_->Fill(1, nStubs / (double)setup_->numRegions()); - profMC_->Fill(2, nStubsMatched / (double)setup_->numRegions()); - } - - // organize reconstrucable TrackingParticles used for efficiency measurements - void Analyzer::convert(const map>& mapTPsStubs, map>& mapStubsTPs) { - int nTPsReco(0); - int nTPsEff(0); - for (const auto& mapTPStubs : mapTPsStubs) { - if (!tpSelectorLoose_(*mapTPStubs.first) || !reconstructable(mapTPStubs.second)) - continue; - nTPsReco++; - const bool useForAlgEff = select(*mapTPStubs.first); - if (useForAlgEff) { - nTPsEff++; - fill(mapTPStubs.first, hisEffMC_); - for (const TTStubRef& ttStubRef : mapTPStubs.second) - mapStubsTPs[ttStubRef].insert(mapTPStubs.first); - } - } - profMC_->Fill(3, nTPsReco / (double)setup_->numRegions()); - profMC_->Fill(4, nTPsEff / (double)setup_->numRegions()); - profMC_->Fill(5, nTPsEff); - } - - // checks if a stub selection is considered reconstructable - bool Analyzer::reconstructable(const set& ttStubRefs) const { - const TrackerGeometry* trackerGeometry = setup_->trackerGeometry(); - const TrackerTopology* trackerTopology = setup_->trackerTopology(); - set hitPattern; - set hitPatternPS; - for (const TTStubRef& ttStubRef : ttStubRefs) { - const DetId detId = ttStubRef->getDetId(); - const bool barrel = detId.subdetId() == StripSubdetector::TOB; - const bool psModule = trackerGeometry->getDetectorType(detId) == TrackerGeometry::ModuleType::Ph2PSP; - const int layerId = barrel ? trackerTopology->layer(detId) : trackerTopology->tidWheel(detId) + 10; - hitPattern.insert(layerId); - if (psModule) - hitPatternPS.insert(layerId); - } - return (int)hitPattern.size() >= setup_->tpMinLayers() && (int)hitPatternPS.size() >= setup_->tpMinLayersPS(); - } - - // checks if TrackingParticle is selected for efficiency measurements - bool Analyzer::select(const TrackingParticle& tp) const { - const bool selected = tpSelector_(tp); - const double cot = sinh(tp.eta()); - const double s = sin(tp.phi()); - const double c = cos(tp.phi()); - const TrackingParticle::Point& v = tp.vertex(); - const double z0 = v.z() - (v.x() * c + v.y() * s) * cot; - const double d0 = v.x() * s - v.y() * c; - return selected && (fabs(d0) < setup_->tpMaxD0()) && (fabs(z0) < setup_->tpMaxVertZ()); + LogPrint(moduleDescription().moduleName()) << log_.str(); } // fills kinematic tp histograms @@ -332,37 +310,8 @@ namespace trackerDTC { th1fs[e]->Fill(x[e]); } - // analyze DTC products and find still reconstrucable TrackingParticles - void Analyzer::analyzeStubs(const TTDTC* accepted, - const TTDTC* lost, - const map>& mapStubsTPs, - map>& mapTPsStubs) { - for (int region = 0; region < setup_->numRegions(); region++) { - int nStubs(0); - int nLost(0); - for (int channel = 0; channel < setup_->numDTCsPerTFP(); channel++) { - const StreamStub& stream = accepted->stream(region, channel); - hisChannel_->Fill(stream.size()); - profChannel_->Fill(region * setup_->numDTCsPerTFP() + channel, stream.size()); - for (const FrameStub& frame : stream) { - if (frame.first.isNull()) - continue; - const auto it = mapStubsTPs.find(frame.first); - if (it == mapStubsTPs.end()) - continue; - for (const TPPtr& tp : it->second) - mapTPsStubs[tp].insert(frame.first); - } - analyzeStream(stream, region, channel, nStubs, hisRZStubs_); - analyzeStream(lost->stream(region, channel), region, channel, nLost, hisRZStubsLost_); - } - profDTC_->Fill(1, nStubs); - profDTC_->Fill(2, nLost); - } - } - // fill stub related histograms - void Analyzer::analyzeStream(const StreamStub& stream, int region, int channel, int& sum, TH2F* th2f) { + void Analyzer::fill(const StreamStub& stream, int region, int channel, int& sum, TH2F* th2f) { for (const FrameStub& frame : stream) { if (frame.first.isNull()) continue; @@ -379,26 +328,6 @@ namespace trackerDTC { } } - // returns layerId [1-6, 11-15] of stub - int Analyzer::layerId(const TTStubRef& ttStubRef) const { - const TrackerTopology* trackerTopology = setup_->trackerTopology(); - const DetId detId = ttStubRef->getDetId() + setup_->offsetDetIdDSV(); - const bool barrel = detId.subdetId() == StripSubdetector::TOB; - return barrel ? trackerTopology->layer(detId) : trackerTopology->tidWheel(detId) + setup_->offsetLayerDisks(); - } - - // analyze survived TPs - void Analyzer::analyzeTPs(const map>& mapTPsStubs) { - int nTPs(0); - for (const auto& mapTPStubs : mapTPsStubs) { - if (!reconstructable(mapTPStubs.second)) - continue; - nTPs++; - fill(mapTPStubs.first, hisEff_); - } - profDTC_->Fill(3, nTPs); - } - // prints out MC summary void Analyzer::endJobMC() { const double numStubs = profMC_->GetBinContent(1); @@ -417,12 +346,12 @@ namespace trackerDTC { const int wErrs = ceil(log10(*max_element(errs.begin(), errs.end()))) + 5; log_ << "=============================================================" << endl; log_ << " MC SUMMARY " << endl; - log_ << "number of cluster per TFP = " << setw(wNums) << numCluster << " +- " << setw(wErrs) << errCluster + /*log_ << "number of cluster per TFP = " << setw(wNums) << numCluster << " +- " << setw(wErrs) << errCluster << endl; log_ << "number of stubs per TFP = " << setw(wNums) << numStubs << " +- " << setw(wErrs) << errStubs << endl; log_ << "number of matched stubs per TFP = " << setw(wNums) << numStubsMatched << " +- " << setw(wErrs) - << errStubsMatched << endl; + << errStubsMatched << endl;*/ log_ << "number of TPs per TFP = " << setw(wNums) << numTPsReco << " +- " << setw(wErrs) << errTPsReco << endl; log_ << "number of TPs for eff per TFP = " << setw(wNums) << numTPsEff << " +- " << setw(wErrs) << errTPsEff @@ -451,90 +380,6 @@ namespace trackerDTC { log_ << " max tracking efficiency = " << setw(wNums) << eff << " +- " << setw(wErrs) << errEff << endl; } - // configuring track particle selector - void Analyzer::configTPSelector() { - const double ptMin = hybrid_ ? setup_->hybridMinPtStub() : setup_->minPt(); - constexpr double ptMax = 9999999999.; - const double etaMax = setup_->tpMaxEta(); - const double tip = setup_->tpMaxVertR(); - const double lip = setup_->tpMaxVertZ(); - constexpr int minHit = 0; - constexpr bool signalOnly = true; - constexpr bool intimeOnly = true; - constexpr bool chargedOnly = true; - constexpr bool stableOnly = false; - tpSelector_ = TrackingParticleSelector( - ptMin, ptMax, -etaMax, etaMax, tip, lip, minHit, signalOnly, intimeOnly, chargedOnly, stableOnly); - tpSelectorLoose_ = - TrackingParticleSelector(ptMin, ptMax, -etaMax, etaMax, tip, lip, minHit, false, false, false, stableOnly); - } - - // book histograms - void Analyzer::bookHistograms() { - Service fs; - TFileDirectory dir; - // mc - dir = fs->mkdir("MC"); - profMC_ = dir.make("Counts", ";", 6, 0.5, 6.5); - profMC_->GetXaxis()->SetBinLabel(1, "Stubs"); - profMC_->GetXaxis()->SetBinLabel(2, "Matched Stubs"); - profMC_->GetXaxis()->SetBinLabel(3, "reco TPs"); - profMC_->GetXaxis()->SetBinLabel(4, "eff TPs"); - profMC_->GetXaxis()->SetBinLabel(5, "total eff TPs"); - profMC_->GetXaxis()->SetBinLabel(6, "Cluster"); - constexpr array binsEff{{9 * 8, 10, 16, 10, 30, 24}}; - constexpr array, NumEfficiency> rangesEff{ - {{-M_PI, M_PI}, {0., 100.}, {-1. / 3., 1. / 3.}, {-5., 5.}, {-15., 15.}, {-2.4, 2.4}}}; - if (useMCTruth_) { - hisEffMC_.reserve(NumEfficiency); - for (Efficiency e : AllEfficiency) - hisEffMC_.emplace_back( - dir.make(("HisTP" + name(e)).c_str(), ";", binsEff[e], rangesEff[e].first, rangesEff[e].second)); - } - // dtc - dir = fs->mkdir("DTC"); - profDTC_ = dir.make("Counts", ";", 3, 0.5, 3.5); - profDTC_->GetXaxis()->SetBinLabel(1, "Stubs"); - profDTC_->GetXaxis()->SetBinLabel(2, "Lost Stubs"); - profDTC_->GetXaxis()->SetBinLabel(3, "TPs"); - // channel occupancy - constexpr int maxOcc = 180; - const int numChannels = setup_->numDTCs() * setup_->numOverlappingRegions(); - hisChannel_ = dir.make("His Channel Occupancy", ";", maxOcc, -.5, maxOcc - .5); - profChannel_ = dir.make("Prof Channel Occupancy", ";", numChannels, -.5, numChannels - .5); - // max tracking efficiencies - if (useMCTruth_) { - dir = fs->mkdir("DTC/Effi"); - hisEff_.reserve(NumEfficiency); - for (Efficiency e : AllEfficiency) - hisEff_.emplace_back( - dir.make(("HisTP" + name(e)).c_str(), ";", binsEff[e], rangesEff[e].first, rangesEff[e].second)); - eff_.reserve(NumEfficiency); - for (Efficiency e : AllEfficiency) - eff_.emplace_back( - dir.make(("Eff" + name(e)).c_str(), ";", binsEff[e], rangesEff[e].first, rangesEff[e].second)); - } - // lost stub fraction in r-z - dir = fs->mkdir("DTC/Loss"); - constexpr int bins = 400; - constexpr double maxZ = 300.; - constexpr double maxR = 120.; - hisRZStubs_ = dir.make("RZ Stubs", ";;", bins, -maxZ, maxZ, bins, 0., maxR); - hisRZStubsLost_ = dir.make("RZ Stubs Lost", ";;", bins, -maxZ, maxZ, bins, 0., maxR); - hisRZStubsEff_ = dir.make("RZ Stubs Eff", ";;", bins, -maxZ, maxZ, bins, 0., maxR); - // stub parameter resolutions - dir = fs->mkdir("DTC/Res"); - constexpr array ranges{{.2, .0001, .5}}; - constexpr int binsHis = 100; - hisResolution_.reserve(NumResolution); - profResolution_.reserve(NumResolution); - for (Resolution r : AllResolution) { - hisResolution_.emplace_back(dir.make(("HisRes" + name(r)).c_str(), ";", binsHis, -ranges[r], ranges[r])); - profResolution_.emplace_back( - dir.make(("ProfRes" + name(r)).c_str(), ";;", bins, -maxZ, maxZ, bins, 0., maxR)); - } - } - } // namespace trackerDTC DEFINE_FWK_MODULE(trackerDTC::Analyzer); diff --git a/L1Trigger/TrackerDTC/test/testDAQ_cfg.py b/L1Trigger/TrackerDTC/test/testDAQ_cfg.py index e3c3e48262f73..6e5fc978d2533 100644 --- a/L1Trigger/TrackerDTC/test/testDAQ_cfg.py +++ b/L1Trigger/TrackerDTC/test/testDAQ_cfg.py @@ -21,12 +21,12 @@ process.GlobalTag = GlobalTag(process.GlobalTag, '140X_mcRun4_realistic_v3', '') # load code that produces DTCStubs -process.load( 'L1Trigger.TrackerDTC.ProducerED_cff' ) +process.load( 'L1Trigger.TrackerDTC.DTC_cff' ) # load code that analyzes TTCluster process.load( 'L1Trigger.TrackerDTC.AnalyzerDAQ_cff' ) # build schedule (not essential to rerun producer). -process.produce = cms.Path( process.TrackerDTCProducer ) +process.produce = cms.Path( process.ProducerDTC ) process.analyzeDAQ = cms.Path( process.TrackerDTCAnalyzerDAQ ) process.schedule = cms.Schedule( process.produce, process.analyzeDAQ ) diff --git a/L1Trigger/TrackerDTC/test/test_cfg.py b/L1Trigger/TrackerDTC/test/test_cfg.py index 8196b0508dc7b..cfc4788d9ad29 100644 --- a/L1Trigger/TrackerDTC/test/test_cfg.py +++ b/L1Trigger/TrackerDTC/test/test_cfg.py @@ -21,7 +21,7 @@ process.GlobalTag = GlobalTag(process.GlobalTag, '140X_mcRun4_realistic_v3', '') # load code that produces DTCStubs -process.load( 'L1Trigger.TrackerDTC.ProducerED_cff' ) +process.load( 'L1Trigger.TrackerDTC.DTC_cff' ) # load code that analyzes DTCStubs process.load( 'L1Trigger.TrackerDTC.Analyzer_cff' ) # cosutmize TT algorithm @@ -30,8 +30,8 @@ #analyzerUseTMTT(process) # build schedule (not essential to rerun producer) -process.produce = cms.Path( process.TrackerDTCProducer ) -process.analyze = cms.Path( process.TrackerDTCAnalyzer ) +process.produce = cms.Path( process.ProducerDTC ) +process.analyze = cms.Path( process.AnalyzerDTC ) process.schedule = cms.Schedule( process.produce, process.analyze ) # create options diff --git a/L1Trigger/TrackerTFP/BuildFile.xml b/L1Trigger/TrackerTFP/BuildFile.xml index 8f490f87eed7e..33536297880e0 100644 --- a/L1Trigger/TrackerTFP/BuildFile.xml +++ b/L1Trigger/TrackerTFP/BuildFile.xml @@ -1,4 +1,5 @@ + diff --git a/L1Trigger/TrackerTFP/README.md b/L1Trigger/TrackerTFP/README.md index 434b589553b04..9dd4b5415f83b 100644 --- a/L1Trigger/TrackerTFP/README.md +++ b/L1Trigger/TrackerTFP/README.md @@ -4,9 +4,9 @@ This directory contains L1 tracking code used by the TMTT & Hybrid algorithms. cmsRun L1Trigger/TrackerTFP/test/test_cfg.py Events= -runs the clock and bit accurate emulation of the TMTT chain. In the run script one may want to change the used event files or tracker geometry. The option CheckHistory in L1Trigger/TrackerTFP/python/Producer_cfi.py is set to false by default but is highly recommended to be set to true if one runs the emulator. This will automatically test if the configuration of the emulation run is consistent with the configuration of the input evevnt production and points out for example if one chooses a different tracker geometry. +runs the clock and bit accurate emulation of the TMTT chain. In the run script one may want to change the used event files or tracker geometry. -Apart from producing TTTrack collection as the f/w will, test_cfg.py analyses the results. It provides a end-of-job summary, which reports data rates and tracking efficiencies at the end of each processing step. The definition of which Tracking Particles are taken into account for this efficiency measurements are described here: L1Trigger/TrackTrigger/python/ProducerSetup_cfi.py in the PSet TrackTrigger_params.TrackingParticle. The "maximal possible tracking efficiency" reported for tracking steps part way through the chain is derived assuming zero efficiency loss in subsequent steps. This method allows to assess which processing steps cause most inefficiency. Additionally a "lost tracking efficiency" is reported, which is the loss due to truncation as a result of bottlenecks in the data throughput of the implemented design. Beside this end job summary test_cfg.py produces Hist.root which contains histograms with more details like efficiencies over certain tracking particle parameter. +Apart from producing TTTrack collection as the f/w will, test_cfg.py analyses the results. It provides a end-of-job summary, which reports data rates and tracking efficiencies at the end of each processing step. The definition of which Tracking Particles are taken into account for this efficiency measurements are described here: L1Trigger/TrackTrigger/python/ProducerSetup_cfi.py in the PSet TrackTrigger_params.TrackingParticle. The "maximal possible tracking efficiency" reported for tracking steps part way through the chain is derived assuming zero efficiency loss in subsequent steps. This method allows to assess which processing steps cause most inefficiency. Beside this end job summary test_cfg.py produces Hist.root which contains histograms with more details like efficiencies over certain tracking particle parameter. cmsRun L1Trigger/TrackerTFP/test/demonstrator_cfg.py Events= @@ -18,11 +18,31 @@ All configuration params to manipulate the algorithms one may want to play with === Code structure === -There are 6 TMTT algorithm steps: GP (Geometric Process), HT (Hough Transform), MHT (Mini Hough Transform), ZHT (r-z Hough Transform), KF (Kalman Filter), DR (Duplicate Removal). Each comes with one EDProducer, one EDAnalyzer and one class which contains the actual emulation of this step for one nonant (1/9 phi slice of outer tracker). Their EDProducts combine the connection to MCTruth (and does not conatain MCTruth) via edm::Refs of either TTStubs or TTTracks with the actual bits used in h/w via std::bitset<64> using a std::pair of those objects. -The track-finding firmware is described in a highly parallel fashion. On the one hand, one has multiple worker nodes for each step and on the other hand is the process pipelined in a way to receive potentially one product per clock tick. This parallelism is reflected by a two dimensional vector of the earlier mentioned pairs. The inner vector describes the output (also called Frame) of one working node per clock tick. If the f/w produces no valid product in a given clock tick, then the bitset will be all '0' and the edm:ref will be null. Valid products do not necessarily form a contiguous block of valid products. The outer vector will have one entry per worker node (also called channel) where all nodes for the whole tracker are counted. Each EDProducer will produce two branches: Accepted Track or Stub and Lost Track or Stub. The Lost branch contains the Tracks or Stubs which are lost since they got not processed in time. Since the KF uses Tracks and Stubs as input the EDProducer ProducerZHTout is used to form TTTracks after the ZHT and ProducerKFin to create the edm::ref to this TTTracks. Finally ProducerTT takes the h/w liked structured output from the KF and produces one collection of TTTracks and ProducerAS creates a map between KF input TTracks to KF output TTTracks. +There are 7 TMTT algorithm steps: GP (Geometric Process), HT (Hough Transform), CTB (Clean Track Builder), KF (Kalman Filter), DR (Duplicate Removal), TQ (Track Quality), TFP (Track Finding Processor). Each comes with one EDProducer, one EDAnalyzer and one class which contains the actual emulation of this step for one nonant (1/9 phi slice of outer tracker). Their EDProducts combine the connection to MCTruth (and does not conatain MCTruth) via edm::Refs of either TTStubs or TTTracks with the actual bits used in h/w via std::bitset<64> using a std::pair of those objects. +The track-finding firmware is described in a highly parallel fashion. On the one hand, one has multiple worker nodes for each step and on the other hand is the process pipelined in a way to receive potentially one product per clock tick. This parallelism is reflected by a two dimensional vector of the earlier mentioned pairs. The inner vector describes the output (also called Frame) of one working node per clock tick. If the f/w produces no valid product in a given clock tick, then the bitset will be all '0' and the edm:ref will be null. Valid products do not necessarily form a contiguous block of valid products. The outer vector will have one entry per worker node (also called channel) where all nodes for the whole tracker are counted. Since the KF uses Tracks and Stubs as input the EDProducer ProducerCTB is used to form TTTracks after the HT. Finally ProducerTFP takes the h/w liked structured output from the TQ and produces one collection of TTTracks. There are 5 additional classes in L1Trigger/TrackerTFP. DataFormats describes Stubs and Tracks for each process step and automates the conversion from floating points to bits as used in h/w and vice versa. Demonstrator allows one to compare s/w with f/w. KalmanFilterFormats describes the used precisions in the Kalman Filter. LayerEncoding allows one to transform the layer encoding used before the KF into the encoding after KF. State is a helper class to simplify the KalmanFilter code. In order to simplify the conversion of floating point values into arbitrary long (within 64 bit) binary or twos-complement number, the class DataFormats/L1TrackTrigger/interface/TTBV.h has been created. In order to simplify the tracking efficiency measurement the class StubAssociator in SimTracker/TrackTriggerAssociation/ has been created. + +=== Details to commonly used Classes === + +Frame is a typedef for std::bitset<64> representing the h/w words which level-1 track finder or level-1 trigger boards will receive and transmit per optical link and internal clock tick. + +TTBV Class representing a BitVector used by TrackTrigger emulators. Based on Frame. The class is mainly used to convert h/w-like structured bits into integers and vice versa. A typical constructors receive an integer values, a bit width (number of bits used to represent this value, an error will be thrown when not enough bits are provided.) and a boolean to distinguish between binary and two's complement representation. Multiple operators are provided, e.g bit wise or, or concatenation with another TTBV. + +FrameStub is a typedef for std::pair. This object is used to represent a Stub in TrackTrigger emulators. On the one hand side it connects to the original TTStub and on the other hand it has the bit string used in h/w to represent this stub. + +FrameTrack same as FrameStub but for Tracks + +StreamStub h/w-like structured collection of stubs. Clock ticks where no Stub can be provided are represented by default constructed FrameStubs (edm:Ref recognising being a null ref and bit set being zero'd). This enables to store stubs bit and clock accurately. + +StreamTrack same as StreamStub but for Tracks + +DataFormat Base class to represent formats of a specific variable at a specific processing step. A format is given by a bit width, an boolean to distinguish between signed and unsigned cover as well as an conversion factor to transform between floating point and biased integer representation. These formats are used to transform h/w words (TTBVs) into variables (supporting conversion to int, double, bool or TTBV). + +DataFormats ESProduct which provides access to all DataFormats used by Track Trigger emulators + +Setup ESProduct providing run time constants configuring Track Trigger emulators \ No newline at end of file diff --git a/L1Trigger/TrackerTFP/interface/CleanTrackBuilder.h b/L1Trigger/TrackerTFP/interface/CleanTrackBuilder.h new file mode 100644 index 0000000000000..46759124c401b --- /dev/null +++ b/L1Trigger/TrackerTFP/interface/CleanTrackBuilder.h @@ -0,0 +1,138 @@ +#ifndef L1Trigger_TrackerTFP_CleanTrackBuilder_h +#define L1Trigger_TrackerTFP_CleanTrackBuilder_h + +#include "DataFormats/L1TrackTrigger/interface/TTTypes.h" +#include "L1Trigger/TrackTrigger/interface/Setup.h" +#include "L1Trigger/TrackerTFP/interface/DataFormats.h" +#include "L1Trigger/TrackerTFP/interface/LayerEncoding.h" +#include "DataFormats/L1TrackTrigger/interface/TTTypes.h" + +#include +#include + +namespace trackerTFP { + + // Class to clean and transform stream of stubs into a stream of tracks with one stub stream per kf layer + class CleanTrackBuilder { + public: + CleanTrackBuilder(const edm::ParameterSet& iConfig, + const tt::Setup* setup, + const DataFormats* dataFormats, + const LayerEncoding* layerEncoding, + const DataFormat& cot, + std::vector& stubs, + std::vector& tracks); + ~CleanTrackBuilder() {} + // fill output products + void produce(const std::vector>& streamsIn, + std::vector>& regionTracks, + std::vector>>& regionStubs); + void put(TrackCTB* track, const std::vector>& stubs, int region, tt::TTTracks& ttTracks) const; + + private: + // struct to represent internal stubs + struct Stub { + // construct Stub from StubHT + Stub(StubHT* stub, int trackId, const TTBV& hitsPhi, const TTBV& hitsZ, int layerId, double dPhi, double dZ) + : stubHT_(stub), + trackId_(trackId), + hitsPhi_(hitsPhi), + hitsZ_(hitsZ), + layerId_(layerId), + dPhi_(dPhi), + dZ_(dZ) {} + // + void update(const TTBV& phi, const TTBV& z, std::vector& ids, int max); + // original ht stub + StubHT* stubHT_; + // + bool valid_ = true; + // + int trackId_; + // + int stubId_ = -1; + // + TTBV hitsPhi_; + // + TTBV hitsZ_; + // + int layerId_; + // + double dPhi_; + // + double dZ_; + }; + + // struct to represent internal tracks + struct Track { + // construct Track from Stubs + Track(const tt::Setup* setup, + int trackId, + const TTBV& hitsPhi, + const TTBV& hitsZ, + const std::vector& stubs, + double inv2R); + // + bool valid_; + // stubs + std::vector stubs_; + // track id + int trackId_; + // + TTBV hitsPhi_; + // + TTBV hitsZ_; + // + double inv2R_; + // size: number of stubs on most occupied layer + int size_; + }; + // + void cleanStream(const std::vector& input, + std::deque& tracks, + std::deque& stubs, + int channelId); + // run single track through r-phi and r-z hough transform + void cleanTrack(const std::vector& track, + std::deque& tracks, + std::deque& stubs, + double inv2R, + int zT, + int trackId); + // + void route(std::vector>& inputs, std::deque& output) const; + // + void route(std::vector>& input, std::vector>& outputs) const; + // + void sort(std::deque& track, std::vector>& stubs) const; + // + void convert(const std::deque& iTracks, + const std::vector>& iStubs, + std::deque& oTracks, + std::vector>& oStubs); + // remove and return first element of deque, returns nullptr if empty + template + T* pop_front(std::deque& ts) const; + // true if truncation is enbaled + bool enableTruncation_; + // provides run-time constants + const tt::Setup* setup_; + // provides dataformats + const DataFormats* dataFormats_; + // + const LayerEncoding* layerEncoding_; + // + DataFormat cot_; + // container of internal stubs + std::vector stubs_; + // container of internal tracks + std::vector tracks_; + // container of output stubs + std::vector& stubsCTB_; + // container of output tracks + std::vector& tracksCTB_; + }; + +} // namespace trackerTFP + +#endif \ No newline at end of file diff --git a/L1Trigger/TrackerTFP/interface/DataFormats.h b/L1Trigger/TrackerTFP/interface/DataFormats.h index 054f97807e20f..86ed628598515 100644 --- a/L1Trigger/TrackerTFP/interface/DataFormats.h +++ b/L1Trigger/TrackerTFP/interface/DataFormats.h @@ -10,7 +10,6 @@ and in undigitized format in an std::tuple. (This saves CPU) ----------------------------------------------------------------------*/ #include "FWCore/Framework/interface/data_default_record_trait.h" -#include "FWCore/ParameterSet/interface/ParameterSet.h" #include "L1Trigger/TrackerTFP/interface/DataFormatsRcd.h" #include "L1Trigger/TrackTrigger/interface/Setup.h" #include "DataFormats/L1TrackTrigger/interface/TTBV.h" @@ -25,41 +24,12 @@ and in undigitized format in an std::tuple. (This saves CPU) namespace trackerTFP { // track trigger processes - enum class Process { begin, fe = begin, dtc, pp, gp, ht, mht, zht, kfin, kf, dr, end, x }; + enum class Process { begin, dtc = begin, pp, gp, ht, ctb, kf, dr, tfp, end, x }; // track trigger variables - enum class Variable { - begin, - r = begin, - phi, - z, - layer, - sectorsPhi, - sectorEta, - sectorPhi, - phiT, - inv2R, - zT, - cot, - dPhi, - dZ, - match, - hitPattern, - phi0, - z0, - end, - x - }; + enum class Variable { begin, r = begin, phi, z, dPhi, dZ, inv2R, phiT, cot, zT, layer, match, end, x }; // track trigger process order - constexpr std::initializer_list Processes = {Process::fe, - Process::dtc, - Process::pp, - Process::gp, - Process::ht, - Process::mht, - Process::zht, - Process::kfin, - Process::kf, - Process::dr}; + constexpr std::initializer_list Processes = { + Process::dtc, Process::pp, Process::gp, Process::ht, Process::ctb, Process::kf, Process::dr, Process::tfp}; // conversion: Process to int inline constexpr int operator+(Process p) { return static_cast(p); } // conversion: Variable to int @@ -72,7 +42,9 @@ namespace trackerTFP { //Base class representing format of a variable class DataFormat { public: - DataFormat(bool twos) : twos_(twos), width_(0), base_(1.), range_(0.) {} + DataFormat() {} + DataFormat(bool twos, bool biased = true) : twos_(twos), width_(0), base_(1.), range_(0.) {} + DataFormat(bool twos, int width, double base, double range) : twos_(twos), width_(width), base_(base), range_(range) {} ~DataFormat() {} // converts int to bitvector TTBV ttBV(int i) const { return TTBV(i, width_, twos_); } @@ -84,12 +56,14 @@ namespace trackerTFP { void extract(TTBV& in, double& out) const { out = in.extract(base_, width_, twos_); } // extracts double from bitvector, removing these bits from bitvector void extract(TTBV& in, TTBV& out) const { out = in.slice(width_, twos_); } + // extracts bool from bitvector, removing these bits from bitvector + void extract(TTBV& in, bool& out) const { out = in.extract(); } // attaches integer to bitvector void attach(const int i, TTBV& ttBV) const { ttBV += TTBV(i, width_, twos_); } // attaches double to bitvector void attach(const double d, TTBV& ttBV) const { ttBV += TTBV(d, base_, width_, twos_); } // attaches bitvector to bitvector - void attach(const TTBV bv, TTBV& ttBV) const { ttBV += bv; } + void attach(const TTBV& bv, TTBV& ttBV) const { ttBV += bv; } // converts int to double double floating(int i) const { return (i + .5) * base_; } // converts double to int @@ -102,8 +76,10 @@ namespace trackerTFP { int toUnsigned(int i) const { return i + std::pow(2, width_) / 2; } // converts floating point value to binary integer value int toUnsigned(double d) const { return this->integer(d) + std::pow(2, width_) / 2; } + // biggest representable floating point value + //double limit() const { return (range_ - base_) / (twos_ ? 2. : 1.); } // returns false if data format would oferflow for this double value - bool inRange(double d, bool digi = false) const { + bool inRange(double d, bool digi = true) const { const double range = digi ? base_ * pow(2, width_) : range_; return d >= -range / 2. && d < range / 2.; } @@ -133,80 +109,70 @@ namespace trackerTFP { template class Format : public DataFormat { public: - Format(const edm::ParameterSet& iConfig, const tt::Setup* setup); + Format(const tt::Setup* setup); ~Format() {} }; template <> - Format::Format(const edm::ParameterSet& iConfig, const tt::Setup* setup); - template <> - Format::Format(const edm::ParameterSet& iConfig, const tt::Setup* setup); - template <> - Format::Format(const edm::ParameterSet& iConfig, const tt::Setup* setup); - template <> - Format::Format(const edm::ParameterSet& iConfig, const tt::Setup* setup); - template <> - Format::Format(const edm::ParameterSet& iConfig, const tt::Setup* setup); - template <> - Format::Format(const edm::ParameterSet& iConfig, const tt::Setup* setup); - template <> - Format::Format(const edm::ParameterSet& iConfig, const tt::Setup* setup); - template <> - Format::Format(const edm::ParameterSet& iConfig, const tt::Setup* setup); - template <> - Format::Format(const edm::ParameterSet& iConfig, const tt::Setup* setup); + Format::Format(const tt::Setup* setup); template <> - Format::Format(const edm::ParameterSet& iConfig, const tt::Setup* setup); + Format::Format(const tt::Setup* setup); template <> - Format::Format(const edm::ParameterSet& iConfig, const tt::Setup* setup); + Format::Format(const tt::Setup* setup); template <> - Format::Format(const edm::ParameterSet& iConfig, const tt::Setup* setup); + Format::Format(const tt::Setup* setup); + template <> - Format::Format(const edm::ParameterSet& iConfig, const tt::Setup* setup); + Format::Format(const tt::Setup* setup); template <> - Format::Format(const edm::ParameterSet& iConfig, const tt::Setup* setup); + Format::Format(const tt::Setup* setup); template <> - Format::Format(const edm::ParameterSet& iConfig, const tt::Setup* setup); + Format::Format(const tt::Setup* setup); template <> - Format::Format(const edm::ParameterSet& iConfig, const tt::Setup* setup); + Format::Format(const tt::Setup* setup); + template <> - Format::Format(const edm::ParameterSet& iConfig, const tt::Setup* setup); + Format::Format(const tt::Setup* setup); template <> - Format::Format(const edm::ParameterSet& iConfig, const tt::Setup* setup); + Format::Format(const tt::Setup* setup); template <> - Format::Format(const edm::ParameterSet& iConfig, const tt::Setup* setup); + Format::Format(const tt::Setup* setup); template <> - Format::Format(const edm::ParameterSet& iConfig, const tt::Setup* setup); + Format::Format(const tt::Setup* setup); template <> - Format::Format(const edm::ParameterSet& iConfig, const tt::Setup* setup); + Format::Format(const tt::Setup* setup); template <> - Format::Format(const edm::ParameterSet& iConfig, const tt::Setup* setup); + Format::Format(const tt::Setup* setup); + template <> - Format::Format(const edm::ParameterSet& iConfig, const tt::Setup* setup); + Format::Format(const tt::Setup* setup); template <> - Format::Format(const edm::ParameterSet& iConfig, const tt::Setup* setup); + Format::Format(const tt::Setup* setup); template <> - Format::Format(const edm::ParameterSet& iConfig, const tt::Setup* setup); + Format::Format(const tt::Setup* setup); + template <> - Format::Format(const edm::ParameterSet& iConfig, const tt::Setup* setup); + Format::Format(const tt::Setup* setup); template <> - Format::Format(const edm::ParameterSet& iConfig, const tt::Setup* setup); + Format::Format(const tt::Setup* setup); template <> - Format::Format(const edm::ParameterSet& iConfig, const tt::Setup* setup); + Format::Format(const tt::Setup* setup); + template <> - Format::Format(const edm::ParameterSet& iConfig, const tt::Setup* setup); + Format::Format(const tt::Setup* setup); template <> - Format::Format(const edm::ParameterSet& iConfig, const tt::Setup* setup); + Format::Format(const tt::Setup* setup); template <> - Format::Format(const edm::ParameterSet& iConfig, const tt::Setup* setup); + Format::Format(const tt::Setup* setup); template <> - Format::Format(const edm::ParameterSet& iConfig, const tt::Setup* setup); + Format::Format(const tt::Setup* setup); template <> - Format::Format(const edm::ParameterSet& iConfig, const tt::Setup* setup); + Format::Format(const tt::Setup* setup); template <> - Format::Format(const edm::ParameterSet& iConfig, const tt::Setup* setup); + Format::Format(const tt::Setup* setup); + template <> - Format::Format(const edm::ParameterSet& iConfig, const tt::Setup* setup); + Format::Format(const tt::Setup* setup); /*! \class trackerTFP::DataFormats * \brief Class to calculate and provide dataformats used by Track Trigger emulator @@ -217,295 +183,180 @@ namespace trackerTFP { private: // variable flavour mapping, Each row below declares which processing steps use the variable named in the comment at the end of the row static constexpr std::array, +Variable::end> config_ = {{ - // Process::fe Process::dtc Process::pp Process::gp Process::ht Process::mht Process::zht Process::kfin Process::kf Process::dr - {{Process::x, - Process::ht, - Process::ht, - Process::ht, - Process::ht, - Process::ht, - Process::ht, - Process::ht, - Process::ht, - Process::x}}, // Variable::r - {{Process::x, + // Process::dtc Process::pp Process::gp Process::ht Process::ctb Process::kf Process::dr, Process::tfp + {{Process::dtc, Process::dtc, Process::dtc, - Process::gp, - Process::ht, - Process::mht, - Process::zht, - Process::kfin, - Process::kfin, - Process::x}}, // Variable::phi - {{Process::x, Process::dtc, + Process::dtc, + Process::dtc, + Process::dtc, + Process::x}}, // Variable::r + {{Process::dtc, Process::dtc, Process::gp, - Process::gp, - Process::gp, - Process::zht, - Process::kfin, - Process::kfin, - Process::x}}, // Variable::z - {{Process::x, - Process::ht, - Process::ht, Process::ht, Process::ht, - Process::ht, - Process::ht, - Process::x, - Process::x, - Process::x}}, // Variable::layer - {{Process::x, - Process::dtc, + Process::kf, + Process::kf, + Process::x}}, // Variable::phi + {{Process::dtc, Process::dtc, - Process::x, - Process::x, - Process::x, - Process::x, - Process::x, - Process::x, - Process::x}}, // Variable::sectorsPhi - {{Process::x, Process::gp, Process::gp, Process::gp, Process::gp, Process::gp, - Process::gp, - Process::gp, - Process::gp, - Process::x}}, // Variable::sectorEta + Process::x}}, // Variable::z {{Process::x, Process::x, Process::x, - Process::gp, - Process::gp, - Process::gp, - Process::gp, - Process::gp, - Process::gp, - Process::x}}, // Variable::sectorPhi + Process::x, + Process::ctb, + Process::ctb, + Process::ctb, + Process::x}}, // Variable::dPhi {{Process::x, + Process::x, + Process::x, + Process::x, + Process::ctb, + Process::ctb, + Process::ctb, + Process::x}}, // Variable::dZ + {{Process::ht, Process::ht, Process::ht, Process::ht, Process::ht, - Process::mht, - Process::mht, - Process::mht, Process::kf, - Process::x}}, // Variable::phiT - {{Process::x, - Process::ht, - Process::ht, + Process::kf, + Process::tfp}}, // Variable::inv2R + {{Process::gp, + Process::gp, + Process::gp, Process::ht, Process::ht, - Process::mht, - Process::mht, - Process::mht, Process::kf, - Process::dr}}, // Variable::inv2R - {{Process::x, - Process::x, - Process::x, - Process::x, - Process::x, - Process::x, - Process::zht, - Process::zht, Process::kf, - Process::x}}, // Variable::zT + Process::tfp}}, // Variable::phiT {{Process::x, Process::x, + Process::gp, Process::x, - Process::x, - Process::x, - Process::x, - Process::zht, - Process::zht, + Process::gp, Process::kf, - Process::dr}}, // Variable::cot - {{Process::x, - Process::x, - Process::x, - Process::x, - Process::x, - Process::x, - Process::x, - Process::kfin, - Process::kfin, - Process::x}}, // Variable::dPhi - {{Process::x, - Process::x, - Process::x, - Process::x, - Process::x, - Process::x, - Process::x, - Process::kfin, - Process::kfin, - Process::x}}, // Variable::dZ - {{Process::x, - Process::x, - Process::x, - Process::x, - Process::x, - Process::x, - Process::x, - Process::x, + Process::dr, + Process::tfp}}, // Variable::cot + {{Process::gp, + Process::gp, + Process::gp, + Process::gp, + Process::gp, Process::kf, - Process::x}}, // Variable::match - {{Process::x, - Process::x, - Process::x, - Process::x, - Process::x, - Process::x, - Process::x, - Process::kfin, - Process::x, - Process::x}}, // Variable::hitPattern - {{Process::x, - Process::x, - Process::x, - Process::x, - Process::x, - Process::x, - Process::x, - Process::x, - Process::x, - Process::dr}}, // Variable::phi0 - {{Process::x, - Process::x, - Process::x, - Process::x, - Process::x, - Process::x, - Process::x, + Process::kf, + Process::tfp}}, // Variable::zT + {{Process::dtc, + Process::dtc, + Process::gp, + Process::gp, + Process::ctb, Process::x, Process::x, - Process::dr}} // Variable::z0 + Process::x}}, // Variable::layer + {{Process::x, Process::x, Process::x, Process::x, Process::x, Process::kf, Process::x, Process::x}} // Variable::match }}; // stub word assembly, shows which stub variables are used by each process static constexpr std::array, +Process::end> stubs_ = {{ - {}, // Process::fe {Variable::r, Variable::phi, Variable::z, Variable::layer, - Variable::sectorsPhi, - Variable::sectorEta, - Variable::sectorEta, + Variable::phiT, + Variable::phiT, + Variable::zT, + Variable::zT, Variable::inv2R, Variable::inv2R}, // Process::dtc {Variable::r, Variable::phi, Variable::z, Variable::layer, - Variable::sectorsPhi, - Variable::sectorEta, - Variable::sectorEta, - Variable::inv2R, - Variable::inv2R}, // Process::pp - {Variable::r, Variable::phi, Variable::z, Variable::layer, Variable::inv2R, Variable::inv2R}, // Process::gp - {Variable::r, - Variable::phi, - Variable::z, - Variable::layer, - Variable::sectorPhi, - Variable::sectorEta, - Variable::phiT}, // Process::ht - {Variable::r, - Variable::phi, - Variable::z, - Variable::layer, - Variable::sectorPhi, - Variable::sectorEta, Variable::phiT, - Variable::inv2R}, // Process::mht - {Variable::r, - Variable::phi, - Variable::z, - Variable::layer, - Variable::sectorPhi, - Variable::sectorEta, Variable::phiT, - Variable::inv2R, Variable::zT, - Variable::cot}, // Process::zht - {Variable::r, Variable::phi, Variable::z, Variable::dPhi, Variable::dZ}, // Process::kfin - {Variable::r, Variable::phi, Variable::z, Variable::dPhi, Variable::dZ}, // Process::kf - {} // Process::dr + Variable::zT, + Variable::inv2R, + Variable::inv2R}, // Process::pp + {Variable::r, Variable::phi, Variable::z, Variable::layer, Variable::inv2R, Variable::inv2R}, // Process::gp + {Variable::r, Variable::phi, Variable::z, Variable::layer, Variable::phiT, Variable::zT}, // Process::ht + {Variable::r, Variable::phi, Variable::z, Variable::dPhi, Variable::dZ}, // Process::ctb + {Variable::r, Variable::phi, Variable::z, Variable::dPhi, Variable::dZ}, // Process::kf + {Variable::r, Variable::phi, Variable::z, Variable::dPhi, Variable::dZ}, // Process::dr + {} // Process::tfp }}; // track word assembly, shows which track variables are used by each process static constexpr std::array, +Process::end> tracks_ = {{ - {}, // Process::fe - {}, // Process::dtc - {}, // Process::pp - {}, // Process::gp - {}, // Process::ht - {}, // Process::mht - {}, // Process::zht - {Variable::hitPattern, - Variable::sectorPhi, - Variable::sectorEta, - Variable::phiT, - Variable::inv2R, - Variable::zT, - Variable::cot}, // Process::kfin - {Variable::match, - Variable::sectorPhi, - Variable::sectorEta, - Variable::phiT, - Variable::inv2R, - Variable::cot, - Variable::zT}, // Process::kf - {Variable::phi0, Variable::inv2R, Variable::z0, Variable::cot} // Process::dr + {}, // Process::dtc + {}, // Process::pp + {}, // Process::gp + {}, // Process::ht + {Variable::inv2R, Variable::phiT, Variable::zT}, // Process::ctb + {Variable::inv2R, Variable::phiT, Variable::cot, Variable::zT, Variable::match}, // Process::kf + {Variable::inv2R, Variable::phiT, Variable::cot, Variable::zT}, // Process::dr + {} // Process::tfp }}; public: DataFormats(); - DataFormats(const edm::ParameterSet& iConfig, const tt::Setup* setup); + DataFormats(const tt::Setup* setup); ~DataFormats() {} - // bool indicating if hybrid or tmtt being used - bool hybrid() const { return iConfig_.getParameter("UseHybrid"); } // converts bits to ntuple of variables template - void convertStub(Process p, const tt::Frame& bv, std::tuple& data) const; + void convertStub(Process p, const tt::Frame& bv, std::tuple& data) const { + TTBV ttBV(bv); + extractStub(p, ttBV, data); + } // converts ntuple of variables to bits template - void convertStub(Process p, const std::tuple& data, tt::Frame& bv) const; + void convertStub(Process p, const std::tuple& data, tt::Frame& bv) const { + TTBV ttBV(1, numUnusedBitsStubs_[+p]); + attachStub(p, data, ttBV); + bv = ttBV.bs(); + } // converts bits to ntuple of variables template - void convertTrack(Process p, const tt::Frame& bv, std::tuple& data) const; + void convertTrack(Process p, const tt::Frame& bv, std::tuple& data) const { + TTBV ttBV(bv); + extractTrack(p, ttBV, data); + } // converts ntuple of variables to bits template - void convertTrack(Process p, const std::tuple& data, tt::Frame& bv) const; + void convertTrack(Process p, const std::tuple& data, tt::Frame& bv) const { + TTBV ttBV(1, numUnusedBitsTracks_[+p]); + attachTrack(p, data, ttBV); + bv = ttBV.bs(); + } // access to run-time constants const tt::Setup* setup() const { return setup_; } // number of bits being used for specific variable flavour int width(Variable v, Process p) const { return formats_[+v][+p]->width(); } // precision being used for specific variable flavour double base(Variable v, Process p) const { return formats_[+v][+p]->base(); } + // covered range for specific variable flavour + double range(Variable v, Process p) const { return formats_[+v][+p]->range(); } // number of unused frame bits for a given Stub flavour int numUnusedBitsStubs(Process p) const { return numUnusedBitsStubs_[+p]; } // number of unused frame bits for a given Track flavour int numUnusedBitsTracks(Process p) const { return numUnusedBitsTracks_[+p]; } // number of channels of a given process on a TFP int numChannel(Process p) const { return numChannel_[+p]; } - // number of channels of a given process for whole system - int numStreams(Process p) const { return numStreams_[+p]; } - // + // number of stub channels of a given process for whole system int numStreamsStubs(Process p) const { return numStreamsStubs_[+p]; } - // + // number of track channels of a given process for whole system int numStreamsTracks(Process p) const { return numStreamsTracks_[+p]; } // access to spedific format const DataFormat& format(Variable v, Process p) const { return *formats_[+v][+p]; } - // critical radius defining region overlap shape in cm - double chosenRofPhi() const { return hybrid() ? setup_->hybridChosenRofPhi() : setup_->chosenRofPhi(); } private: // number of unique data formats @@ -521,16 +372,36 @@ namespace trackerTFP { void fillFormats(); // helper (loop) to convert bits to ntuple of variables template - void extractStub(Process p, TTBV& ttBV, std::tuple& data) const; + void extractStub(Process p, TTBV& ttBV, std::tuple& data) const { + Variable v = *std::next(stubs_[+p].begin(), sizeof...(Ts) - 1 - it); + formats_[+v][+p]->extract(ttBV, std::get(data)); + if constexpr (it + 1 != sizeof...(Ts)) + extractStub(p, ttBV, data); + } // helper (loop) to convert bits to ntuple of variables template - void extractTrack(Process p, TTBV& ttBV, std::tuple& data) const; + void extractTrack(Process p, TTBV& ttBV, std::tuple& data) const { + Variable v = *std::next(tracks_[+p].begin(), sizeof...(Ts) - 1 - it); + formats_[+v][+p]->extract(ttBV, std::get(data)); + if constexpr (it + 1 != sizeof...(Ts)) + extractTrack(p, ttBV, data); + } // helper (loop) to convert ntuple of variables to bits template - void attachStub(Process p, const std::tuple& data, TTBV& ttBV) const; + void attachStub(Process p, const std::tuple& data, TTBV& ttBV) const { + Variable v = *std::next(stubs_[+p].begin(), it); + formats_[+v][+p]->attach(std::get(data), ttBV); + if constexpr (it + 1 != sizeof...(Ts)) + attachStub(p, data, ttBV); + } // helper (loop) to convert ntuple of variables to bits template - void attachTrack(Process p, const std::tuple& data, TTBV& ttBV) const; + void attachTrack(Process p, const std::tuple& data, TTBV& ttBV) const { + Variable v = *std::next(tracks_[+p].begin(), it); + formats_[+v][+p]->attach(std::get(data), ttBV); + if constexpr (it + 1 != sizeof...(Ts)) + attachTrack(p, data, ttBV); + } // configuration during construction edm::ParameterSet iConfig_; // stored run-time constants @@ -545,11 +416,9 @@ namespace trackerTFP { std::vector numUnusedBitsTracks_; // number of channels of all processes on a TFP std::vector numChannel_; - // number of channels of all processes for whole system - std::vector numStreams_; - // + // number of stub channels of all processes for whole system std::vector numStreamsStubs_; - // + // number of track channels of all processes for whole system std::vector numStreamsTracks_; }; @@ -558,12 +427,20 @@ namespace trackerTFP { class Stub { public: // construct Stub from Frame - Stub(const tt::FrameStub& frame, const DataFormats* dataFormats, Process p); + Stub(const tt::FrameStub& fs, const DataFormats* df, Process p) : dataFormats_(df), p_(p), frame_(fs) { + dataFormats_->convertStub(p_, frame_.second, data_); + } template // construct Stub from other Stub - Stub(const Stub& stub, Ts... data); + Stub(const Stub& stub, Ts... data) + : dataFormats_(stub.dataFormats()), p_(++stub.p()), frame_(stub.frame()), data_(data...) { + dataFormats_->convertStub(p_, data_, frame_.second); + } // construct Stub from TTStubRef - Stub(const TTStubRef& ttStubRef, const DataFormats* dataFormats, Process p, Ts... data); + Stub(const TTStubRef& ttStubRef, const DataFormats* df, Process p, Ts... data) + : dataFormats_(df), p_(p), frame_(ttStubRef, tt::Frame()), data_(data...) { + dataFormats_->convertStub(p_, data_, frame_.second); + } Stub() {} ~Stub() {} // true if frame valid, false if gap in data stream @@ -573,21 +450,9 @@ namespace trackerTFP { // stub flavour Process p() const { return p_; } // acess to frame - tt::FrameStub frame() const { return frame_; } - // access to TTStubRef - TTStubRef ttStubRef() const { return frame_.first; } - // access to bitvector - tt::Frame bv() const { return frame_.second; } - // id of collection this stub belongs to - int trackId() const { return trackId_; } + const tt::FrameStub& frame() const { return frame_; } protected: - // number of used bits for given variable - int width(Variable v) const { return dataFormats_->width(v, p_); } - // precision of given variable - double base(Variable v) const { return dataFormats_->base(v, p_); } - // format of given variable - DataFormat format(Variable v) const { return dataFormats_->format(v, p_); } // all dataformats const DataFormats* dataFormats_; // stub flavour @@ -596,208 +461,131 @@ namespace trackerTFP { tt::FrameStub frame_; // ntuple of variables this stub is assemled of std::tuple data_; - // id of collection this stub belongs to - int trackId_; + }; + + // class to represent stubs generated by process DTC + class StubDTC : public Stub { + public: + // construct StubDTC from TTStubRef + StubDTC(const TTStubRef& ttStubRef, + const DataFormats* df, + double r, + double phi, + double z, + const TTBV& layer, + int phiTMin, + int phiTMax, + int zTMin, + int zTMax, + int inv2RMin, + int inv2RMax) + : Stub(ttStubRef, df, Process::dtc, r, phi, z, layer, phiTMin, phiTMax, zTMin, zTMax, inv2RMin, inv2RMax) {} + ~StubDTC() {} + // stub radius wrt chosenRofPhi + double r() const { return std::get<0>(data_); } + // stub phi wrt processing nonant centre + double phi() const { return std::get<1>(data_); } + // stub z + double z() const { return std::get<2>(data_); } + // enhanced layer id + TTBV layer() const { return std::get<3>(data_); } + // first phi sector this stub belongs to + int phiTMin() const { return std::get<4>(data_); } + // last phi sector this stub belongs to + int phiTMax() const { return std::get<5>(data_); } + // first eta sector this stub belongs to + int zTMin() const { return std::get<6>(data_); } + // last eta sector this stub belongs to + int zTMax() const { return std::get<7>(data_); } + // first inv2R bin this stub belongs to + int inv2RMin() const { return std::get<8>(data_); } + // last inv2R bin this stub belongs to + int inv2RMax() const { return std::get<9>(data_); } }; // class to represent stubs generated by process patch pannel - class StubPP : public Stub { + class StubPP : public Stub { public: // construct StubPP from Frame - StubPP(const tt::FrameStub& frame, const DataFormats* dataFormats); + StubPP(const tt::FrameStub& fs, const DataFormats* df) : Stub(fs, df, Process::pp) {} ~StubPP() {} - // true if stub belongs to given sector - bool inSector(int sector) const { return sectors_[sector]; } - // sectors this stub belongs to - std::vector sectors() const { return sectors_.ids(); } // stub radius wrt chosenRofPhi double r() const { return std::get<0>(data_); } // stub phi wrt processing nonant centre double phi() const { return std::get<1>(data_); } // stub z double z() const { return std::get<2>(data_); } - // reduced layer id - int layer() const { return std::get<3>(data_); } - // phi sector map to which this stub belongs to - TTBV sectorsPhi() const { return std::get<4>(data_); } + // enhanced layer id + const TTBV& layer() const { return std::get<3>(data_); } + // first phi sector this stub belongs to + int phiTMin() const { return std::get<4>(data_); } + // last phi sector this stub belongs to + int phiTMax() const { return std::get<5>(data_); } // first eta sector this stub belongs to - int sectorEtaMin() const { return std::get<5>(data_); } + int zTMin() const { return std::get<6>(data_); } // last eta sector this stub belongs to - int sectorEtaMax() const { return std::get<6>(data_); } + int zTMax() const { return std::get<7>(data_); } // first inv2R bin this stub belongs to - int inv2RMin() const { return std::get<7>(data_); } + int inv2RMin() const { return std::get<8>(data_); } // last inv2R bin this stub belongs to - int inv2RMax() const { return std::get<8>(data_); } - - private: - // sectors this stub belongs to - TTBV sectors_; + int inv2RMax() const { return std::get<9>(data_); } }; // class to represent stubs generated by process geometric processor - class StubGP : public Stub { + class StubGP : public Stub { public: // construct StubGP from Frame - StubGP(const tt::FrameStub& frame, const DataFormats* dataFormats, int sectorPhi, int sectorEta); - // construct StubGO from StubPP - StubGP(const StubPP& stub, int sectorPhi, int sectorEta); + StubGP(const tt::FrameStub& fs, const DataFormats* df) : Stub(fs, df, Process::gp) {} + // construct StubGP from StubPP + StubGP(const StubPP& stub, double r, double phi, double z, const TTBV& layer, int inv2RMin, int inv2RMax) + : Stub(stub, r, phi, z, layer, inv2RMin, inv2RMax) {} ~StubGP() {} - // true if stub belongs to given inv2R bin - bool inInv2RBin(int inv2RBin) const { return inv2RBins_[inv2RBin]; } - // inv2R bins this stub belongs to - std::vector inv2RBins() const { return inv2RBins_.ids(); } - // stub phi sector - int sectorPhi() const { return sectorPhi_; } - // stub eta sector - int sectorEta() const { return sectorEta_; } // stub radius wrt chosenRofPhi double r() const { return std::get<0>(data_); } // stub phi wrt phi sector centre double phi() const { return std::get<1>(data_); } // stub z residual wrt eta sector double z() const { return std::get<2>(data_); } - // reduced layer id - int layer() const { return std::get<3>(data_); } + // enhanced layer id + const TTBV& layer() const { return std::get<3>(data_); } // first inv2R bin this stub belongs to int inv2RMin() const { return std::get<4>(data_); } // last inv2R bin this stub belongs to int inv2RMax() const { return std::get<5>(data_); } - - private: - // inv2R bins this stub belongs to - TTBV inv2RBins_; - // stub phi sector - int sectorPhi_; - // stub eta sector - int sectorEta_; }; // class to represent stubs generated by process hough transform - class StubHT : public Stub { + class StubHT : public Stub { public: // construct StubHT from Frame - StubHT(const tt::FrameStub& frame, const DataFormats* dataFormats, int inv2R); - // construct StubHT from StubGP and HT cell assignment - StubHT(const StubGP& stub, int phiT, int inv2R); + StubHT(const tt::FrameStub& fs, const DataFormats* df) : Stub(fs, df, Process::ht) {} + // construct StubHT from StubGP + StubHT(const StubGP& stub, double r, double phi, double z, const TTBV& layer, int phiT, int zT) + : Stub(stub, r, phi, z, layer, phiT, zT) {} ~StubHT() {} - // stub qOver pt - int inv2R() const { return inv2R_; } // stub radius wrt chosenRofPhi double r() const { return std::get<0>(data_); }; // stub phi residual wrt track parameter double phi() const { return std::get<1>(data_); }; // stub z residual wrt eta sector double z() const { return std::get<2>(data_); }; - // reduced layer id - int layer() const { return std::get<3>(data_); }; - // phi sector - int sectorPhi() const { return std::get<4>(data_); }; - // eta sector - int sectorEta() const { return std::get<5>(data_); }; - // stub phi at radius chosenRofPhi wrt phi sector centre - int phiT() const { return std::get<6>(data_); }; - - private: - // fills track id - void fillTrackId(); - // stub qOver pt - int inv2R_; - }; - - // class to represent stubs generated by process mini hough transform - class StubMHT : public Stub { - public: - // construct StubMHT from Frame - StubMHT(const tt::FrameStub& frame, const DataFormats* dataFormats); - // construct StubMHT from StubHT and MHT cell assignment - StubMHT(const StubHT& stub, int phiT, int inv2R); - ~StubMHT() {} - // stub radius wrt choenRofPhi - double r() const { return std::get<0>(data_); } - // stub phi residual wrt finer track parameter - double phi() const { return std::get<1>(data_); } - // stub z rsidual wrt eta sector - double z() const { return std::get<2>(data_); } - // reduced layer id - int layer() const { return std::get<3>(data_); } - // phi sector - int sectorPhi() const { return std::get<4>(data_); } - // eta sector - int sectorEta() const { return std::get<5>(data_); } - // stub phi at radius chosenRofPhi wrt phi sector centre - int phiT() const { return std::get<6>(data_); } - // stub inv2R - int inv2R() const { return std::get<7>(data_); } - - private: - // fills track id - void fillTrackId(); - }; - - // class to represent stubs generated by process z hough transform - class StubZHT : public Stub { - public: - // construct StubZHT from Frame - StubZHT(const tt::FrameStub& frame, const DataFormats* dataFormats); - // construct StubZHT from StubMHT - StubZHT(const StubMHT& stub); - // - StubZHT(const StubZHT& stub, double zT, double cot, int id); - // - StubZHT(const StubZHT& stub, int cot, int zT); - ~StubZHT() {} - // stub radius wrt chonseRofPhi - double r() const { return std::get<0>(data_); } - // stub phiresiudal wrt finer track parameter - double phi() const { return std::get<1>(data_); } - // stub z residual to track parameter - double z() const { return std::get<2>(data_); } - // reduced layer id - int layer() const { return std::get<3>(data_); } - // phi sector - int sectorPhi() const { return std::get<4>(data_); } + // enhanced layer id + const TTBV& layer() const { return std::get<3>(data_); } + // stub phi at radius chosenRofPhi wrt processing nonant centre + int phiT() const { return std::get<4>(data_); }; // eta sector - int sectorEta() const { return std::get<5>(data_); } - // stub phi at radius chosenRofPhi wrt phi sector centre - int phiT() const { return std::get<6>(data_); } - // stub inv2R - int inv2R() const { return std::get<7>(data_); } - // stub z at radius chosenRofZ wrt eta sector centre - int zT() const { return std::get<8>(data_); } - // stub cotTheta wrt eta sector cotTheta - int cot() const { return std::get<9>(data_); } - double cotf() const { return cot_; } - double ztf() const { return zT_; } - double chi() const { return chi_; } - - private: - // fills track id - void fillTrackId(); - double r_; - double chi_; - double cot_; - double zT_; + int zT() const { return std::get<5>(data_); }; }; - // class to represent stubs generated by process kfin - class StubKFin : public Stub { + // class to represent stubs generated by process CTB + class StubCTB : public Stub { public: - // construct StubKFin from Frame - StubKFin(const tt::FrameStub& frame, const DataFormats* dataFormats, int layer); - // construct StubKFin from StubZHT - StubKFin(const StubZHT& stub, double dPhi, double dZ, int layer); - // construct StubKFin from TTStubRef - StubKFin(const TTStubRef& ttStubRef, - const DataFormats* dataFormats, - double r, - double phi, - double z, - double dPhi, - double dZ, - int layer); - ~StubKFin() {} - // kf layer id - int layer() const { return layer_; } + // construct StubTB from Frame + StubCTB(const tt::FrameStub& fs, const DataFormats* df) : Stub(fs, df, Process::ctb) {} + // construct StubTB from StubZHT + StubCTB(const StubHT& stub, double r, double phi, double z, double dPhi, double dZ) + : Stub(stub, r, phi, z, dPhi, dZ) {} + ~StubCTB() {} // stub radius wrt chosenRofPhi double r() const { return std::get<0>(data_); } // stub phi residual wrt finer track parameter @@ -808,22 +596,17 @@ namespace trackerTFP { double dPhi() const { return std::get<3>(data_); } // stub z uncertainty double dZ() const { return std::get<4>(data_); } - - private: - // kf layer id - int layer_; }; // class to represent stubs generated by process kalman filter class StubKF : public Stub { public: // construct StubKF from Frame - StubKF(const tt::FrameStub& frame, const DataFormats* dataFormats, int layer); - // construct StubKF from StubKFin - StubKF(const StubKFin& stub, double inv2R, double phiT, double cot, double zT); + StubKF(const tt::FrameStub& fs, const DataFormats* df) : Stub(fs, df, Process::kf) {} + // construct StubKF from StubCTB + StubKF(const StubCTB& stub, double r, double phi, double z, double dPhi, double dZ) + : Stub(stub, r, phi, z, dPhi, dZ) {} ~StubKF() {} - // kf layer id - int layer() const { return layer_; } // stub radius wrt choenRofPhi double r() const { return std::get<0>(data_); } // stub phi residual wrt fitted parameter @@ -834,10 +617,27 @@ namespace trackerTFP { double dPhi() const { return std::get<3>(data_); } // stub z uncertainty double dZ() const { return std::get<4>(data_); } + }; - private: - // kf layer id - int layer_; + // class to represent stubs generated by process duplicate removal + class StubDR : public Stub { + public: + // construct StubDR from Frame + StubDR(const tt::FrameStub& fs, const DataFormats* df) : Stub(fs, df, Process::dr) {} + // construct StubDR from StubKF + StubDR(const StubKF& stub, double r, double phi, double z, double dPhi, double dZ) + : Stub(stub, r, phi, z, dPhi, dZ) {} + ~StubDR() {} + // stub radius wrt choenRofPhi + double r() const { return std::get<0>(data_); } + // stub phi residual wrt fitted parameter + double phi() const { return std::get<1>(data_); } + // stub z residual wrt fitted parameter + double z() const { return std::get<2>(data_); } + // stub phi uncertainty + double dPhi() const { return std::get<3>(data_); } + // stub z uncertainty + double dZ() const { return std::get<4>(data_); } }; // base class to represent tracks @@ -845,15 +645,20 @@ namespace trackerTFP { class Track { public: // construct Track from Frame - Track(const tt::FrameTrack& frame, const DataFormats* dataFormats, Process p); + Track(const tt::FrameTrack& ft, const DataFormats* df, Process p) : dataFormats_(df), p_(p), frame_(ft) { + dataFormats_->convertTrack(p_, frame_.second, data_); + } + // construct Track from TTTrackRef + Track(const TTTrackRef& ttTrackRef, const DataFormats* df, Process p, Ts... data) + : dataFormats_(df), p_(p), frame_(ttTrackRef, tt::Frame()), data_(data...) { + dataFormats_->convertTrack(p_, data_, frame_.second); + } // construct Track from other Track template - Track(const Track& track, Ts... data); - // construct Track from Stub - template - Track(const Stub& stub, const TTTrackRef& ttTrackRef, Ts... data); - // construct Track from TTTrackRef - Track(const TTTrackRef& ttTrackRef, const DataFormats* dataFormats, Process p, Ts... data); + Track(const Track& track, Ts... data) + : dataFormats_(track.dataFormats()), p_(++track.p()), frame_(track.frame()), data_(data...) { + dataFormats_->convertTrack(p_, data_, frame_.second); + } ~Track() {} // true if frame valid, false if gap in data stream explicit operator bool() const { return frame_.first.isNonnull(); } @@ -862,25 +667,9 @@ namespace trackerTFP { // track flavour Process p() const { return p_; } // acces to frame - tt::FrameTrack frame() const { return frame_; } - // access to TTTrackRef - TTTrackRef ttTrackRef() const { return frame_.first; } - // access to bitvector - tt::Frame bv() const { return frame_.second; } - // access to ntuple of variables this track is assemled of - std::tuple data() const { return data_; } + const tt::FrameTrack& frame() const { return frame_; } protected: - //number of bits uesd of given variable - int width(Variable v) const { return dataFormats_->width(v, p_); } - // precision of given variable - double base(Variable v) const { return dataFormats_->base(v, p_); } - // access to run-time constants - const tt::Setup* setup() const { return dataFormats_->setup(); } - // format of given variable - DataFormat format(Variable v) const { return dataFormats_->format(v, p_); } - // format of given variable and process - DataFormat format(Variable v, Process p) const { return dataFormats_->format(v, p); } // all data formats const DataFormats* dataFormats_; // track flavour @@ -891,168 +680,61 @@ namespace trackerTFP { std::tuple data_; }; - class TrackKFin : public Track { + // class to represent tracks generated by process Clean Track Builder + class TrackCTB : public Track { public: - // construct TrackKFin from Frame - TrackKFin(const tt::FrameTrack& frame, const DataFormats* dataFormats, const std::vector& stubs); - // construct TrackKFin from StubKFin - TrackKFin(const StubZHT& stub, const TTTrackRef& ttTrackRef, const TTBV& maybePattern); - // construct TrackKFin from TTTrackRef - TrackKFin(const TTTrackRef& ttTrackRef, - const DataFormats* dataFormats, - const TTBV& maybePattern, - double phiT, - double qOverPt, - double zT, - double cot, - int sectorPhi, - int sectorEta); - ~TrackKFin() {} - // pattern of layers which are only maybe crossed by found candidate - const TTBV& maybePattern() const { return std::get<0>(data_); } - // phi sector - int sectorPhi() const { return std::get<1>(data_); } - // eta sector - int sectorEta() const { return std::get<2>(data_); } - // track phi at radius chosenRofPhi wrt phi sector centre - double phiT() const { return std::get<3>(data_); } + // construct TrackTB from Frame + TrackCTB(const tt::FrameTrack& ft, const DataFormats* df) : Track(ft, df, Process::ctb) {} + // construct TrackTB from StubsCTB + TrackCTB(const TTTrackRef& tTTrackRef, const DataFormats* df, double inv2R, double phiT, double zT) + : Track(tTTrackRef, df, Process::ctb, inv2R, phiT, zT) {} + ~TrackCTB() {} // track inv2R - double inv2R() const { return std::get<4>(data_); } - // track z at radius chosenRofZ wrt eta sector centre - double zT() const { return std::get<5>(data_); } - // track cotTheta wrt seta sector cotTheta - double cot() const { return std::get<6>(data_); } - // - TTBV hitPattern() const { return hitPattern_; } - // true if given layer has a hit - bool hitPattern(int index) const { return hitPattern_[index]; } - // true if given layer has a hit or is a maybe layer - bool maybePattern(int index) const { return hitPattern_[index] || maybePattern()[index]; } - // stubs on a given layer - std::vector layerStubs(int layer) const { return stubs_[layer]; } - // firts stub on a given layer - StubKFin* layerStub(int layer) const { return stubs_[layer].front(); } - // selection of ttStubRefs for given hit ids on given layers - std::vector ttStubRefs(const TTBV& hitPattern, const std::vector& layerMap) const; - // stubs organized in layer - std::vector> stubs() const { return stubs_; } - // global cotTheta - double cotGlobal() const { return cot() + setup()->sectorCot(sectorEta()); } - - private: - // stubs organized in layer - std::vector> stubs_; - // - TTBV hitPattern_; + double inv2R() const { return std::get<0>(data_); } + // track phi at radius chosenRofPhi wrt pprocessing centre + double phiT() const { return std::get<1>(data_); } + // track z at radius chosenRofZ + double zT() const { return std::get<2>(data_); } }; // class to represent tracks generated by process kalman filter - class TrackKF : public Track { + class TrackKF : public Track { public: // construct TrackKF from Frame - TrackKF(const tt::FrameTrack& frame, const DataFormats* dataFormats); - // construct TrackKF from TrackKFKFin - TrackKF(const TrackKFin& track, double phiT, double inv2R, double zT, double cot); + TrackKF(const tt::FrameTrack& ft, const DataFormats* df) : Track(ft, df, Process::kf) {} + // construct TrackKF from TrackCTB + TrackKF(const TrackCTB& track, double inv2R, double phiT, double cot, double zT, const TTBV& match) + : Track(track, inv2R, phiT, cot, zT, match) {} ~TrackKF() {} - // true if kf prameter consistent with mht parameter - bool match() const { return std::get<0>(data_); } - // phi sector - int sectorPhi() const { return std::get<1>(data_); } - // eta sector - int sectorEta() const { return std::get<2>(data_); } - // track phi at radius chosenRofPhi wrt phi sector centre - double phiT() const { return std::get<3>(data_); } // track qOver pt - double inv2R() const { return std::get<4>(data_); } + double inv2R() const { return std::get<0>(data_); } + // track phi at radius chosenRofPhi wrt processing nonant centre + double phiT() const { return std::get<1>(data_); } // track cotTheta wrt eta sector cotTheta - double cot() const { return std::get<5>(data_); } - // track z at radius chosenRofZ wrt eta sector centre - double zT() const { return std::get<6>(data_); } - // global cotTheta - double cotGlobal() const { return cot() + setup()->sectorCot(sectorEta()); } - // conversion to TTTrack with given stubs - TTTrack ttTrack(const std::vector& stubs) const; - - private: - }; - - //Class to represent KFout 96-bit track for use in distribution server - class TrackKFOut { - public: - TrackKFOut() : TrackKFOut(0, 0, 0, 0, 0, tt::FrameTrack(), 0, 0, false) {} - // construct TrackKF from Partial Tracks - TrackKFOut(TTBV PartialTrack1, - TTBV PartialTrack2, - TTBV PartialTrack3, - int sortKey, - int nonantId, - const tt::FrameTrack& track, - int trackID, - int linkID, - bool valid) - : PartialTrack1_(PartialTrack1), - PartialTrack2_(PartialTrack2), - PartialTrack3_(PartialTrack3), - sortKey_(sortKey), - nonantId_(nonantId), - track_(track), - trackID_(trackID), - linkID_(linkID), - valid_(valid) {} - - ~TrackKFOut() {} - - int sortKey() const { return sortKey_; } - int nonantId() const { return nonantId_; } - - bool dataValid() const { return valid_; } - - int trackID() const { return trackID_; } - int linkID() const { return linkID_; } - - TTBV PartialTrack1() const { return PartialTrack1_; } - TTBV PartialTrack2() const { return PartialTrack2_; } - TTBV PartialTrack3() const { return PartialTrack3_; } - - tt::FrameTrack track() const { return track_; } - - private: - TTBV PartialTrack1_; - TTBV PartialTrack2_; - TTBV PartialTrack3_; - int sortKey_; - int nonantId_; - tt::FrameTrack track_; - int trackID_; - int linkID_; - bool valid_; + double cot() const { return std::get<2>(data_); } + // track z at radius chosenRofZ + double zT() const { return std::get<3>(data_); } + // true if kf prameter consistent with mht parameter + const TTBV& match() const { return std::get<4>(data_); } }; - typedef std::vector TrackKFOutSACollection; - typedef std::shared_ptr TrackKFOutSAPtr; - typedef std::vector TrackKFOutSAPtrCollection; - typedef std::vector>> TrackKFOutSAPtrCollections; - typedef std::vector>>> TrackKFOutSAPtrCollectionss; // class to represent tracks generated by process duplicate removal class TrackDR : public Track { public: // construct TrackDR from Frame - TrackDR(const tt::FrameTrack& frame, const DataFormats* dataFormats); + TrackDR(const tt::FrameTrack& ft, const DataFormats* df) : Track(ft, df, Process::dr) {} // construct TrackDR from TrackKF - TrackDR(const TrackKF& track); + TrackDR(const TrackKF& track, double inv2R, double phiT, double cot, double zT) + : Track(track, inv2R, phiT, cot, zT) {} ~TrackDR() {} - // track phi at radius 0 wrt processing nonant centre - double phi0() const { return std::get<0>(data_); } // track inv2R - double inv2R() const { return std::get<1>(data_); } - // track z at radius 0 - double z0() const { return std::get<2>(data_); } + double inv2R() const { return std::get<0>(data_); } + // track phi at radius 0 wrt processing nonant centre + double phiT() const { return std::get<1>(data_); } // track cotThea - double cot() const { return std::get<3>(data_); } - // conversion to TTTrack - TTTrack ttTrack() const; - - private: + double cot() const { return std::get<2>(data_); } + // track z at radius 0 + double zT() const { return std::get<3>(data_); } }; } // namespace trackerTFP diff --git a/L1Trigger/TrackerTFP/interface/Demonstrator.h b/L1Trigger/TrackerTFP/interface/Demonstrator.h index b76b38f64e76b..7f23fc960ef80 100644 --- a/L1Trigger/TrackerTFP/interface/Demonstrator.h +++ b/L1Trigger/TrackerTFP/interface/Demonstrator.h @@ -12,8 +12,8 @@ namespace trackerTFP { /*! \class trackerTFP::Demonstrator - * \brief Compares emulator with f/w - * \author Thomas Schuh + * \brief ESProduct providing the algorithm to run input data through modelsim + * and to compares results with expected output data * \date 2021, April */ class Demonstrator { @@ -27,15 +27,17 @@ namespace trackerTFP { private: // converts streams of bv into stringstream - void convert(const std::vector>& bits, std::stringstream& ss) const; + void convert(const std::vector>& bits, + std::stringstream& ss, + const std::vector& mapping) const; // plays stringstream through modelsim void sim(const std::stringstream& ss) const; // compares stringstream with modelsim output bool compare(std::stringstream& ss) const; // creates emp file header - std::string header(int numChannel) const; + std::string header(const std::vector& links) const; // creates 6 frame gap between packets - std::string infraGap(int& nFrame, int numChannel) const; + std::string infraGap(int& nFrame, int numLinks) const; // creates frame number std::string frame(int& nFrame) const; // converts bv into hex @@ -45,6 +47,10 @@ namespace trackerTFP { std::string dirIPBB_; // runtime in ms double runTime_; + // + const std::vector linkMappingIn_; + // + const std::vector linkMappingOut_; // path to input text file std::string dirIn_; // path to output text file diff --git a/L1Trigger/TrackerTFP/interface/DuplicateRemoval.h b/L1Trigger/TrackerTFP/interface/DuplicateRemoval.h new file mode 100644 index 0000000000000..4ba4de46f9d51 --- /dev/null +++ b/L1Trigger/TrackerTFP/interface/DuplicateRemoval.h @@ -0,0 +1,58 @@ +#ifndef L1Trigger_TrackerTFP_DuplicateRemoval_h +#define L1Trigger_TrackerTFP_DuplicateRemoval_h + +#include "L1Trigger/TrackTrigger/interface/Setup.h" +#include "L1Trigger/TrackerTFP/interface/DataFormats.h" +#include "DataFormats/L1TrackTrigger/interface/TTTypes.h" + +#include + +namespace trackerTFP { + + // Class to do duplicate removal in a region. + class DuplicateRemoval { + public: + DuplicateRemoval(const edm::ParameterSet& iConfig, + const tt::Setup* setup, + const DataFormats* dataFormats, + std::vector& tracks, + std::vector& stubs); + ~DuplicateRemoval() {} + // fill output products + void produce(const std::vector>& tracksIn, + const std::vector>& stubsIn, + std::vector>& tracksOut, + std::vector>& stubsOut); + + private: + struct Track { + Track(TrackKF* track, const std::vector& stubs, bool match, int inv2R, int phiT, int zT) + : track_(track), stubs_(stubs), match_(match), inv2R_(inv2R), phiT_(phiT), zT_(zT) {} + // + TrackKF* track_; + // + std::vector stubs_; + // + bool match_; + // + int inv2R_; + // + int phiT_; + // + int zT_; + }; + // true if truncation is enbaled + bool enableTruncation_; + // provides run-time constants + const tt::Setup* setup_; + // provides dataformats + const DataFormats* dataFormats_; + // container of output tracks + std::vector& tracks_; + // container of output stubs + std::vector& stubs_; + }; + +} // namespace trackerTFP + +#endif \ No newline at end of file diff --git a/L1Trigger/TrackerTFP/interface/GeometricProcessor.h b/L1Trigger/TrackerTFP/interface/GeometricProcessor.h index 07fc8858c872f..97ef1a2882b35 100644 --- a/L1Trigger/TrackerTFP/interface/GeometricProcessor.h +++ b/L1Trigger/TrackerTFP/interface/GeometricProcessor.h @@ -1,9 +1,11 @@ #ifndef L1Trigger_TrackerTFP_GeometricProcessor_h #define L1Trigger_TrackerTFP_GeometricProcessor_h +#include "FWCore/ParameterSet/interface/ParameterSet.h" #include "L1Trigger/TrackTrigger/interface/Setup.h" #include "L1Trigger/TrackerTFP/interface/DataFormats.h" -#include "DataFormats/L1TrackTrigger/interface/TTDTC.h" +#include "L1Trigger/TrackerTFP/interface/LayerEncoding.h" +#include "DataFormats/L1TrackTrigger/interface/TTTypes.h" #include #include @@ -16,33 +18,29 @@ namespace trackerTFP { GeometricProcessor(const edm::ParameterSet& iConfig, const tt::Setup* setup_, const DataFormats* dataFormats, - int region); + const LayerEncoding* layerEncoding, + std::vector& stubs); ~GeometricProcessor() {} - // read in and organize input product (fill vector input_) - void consume(const TTDTC& ttDTC); - // fill output products - void produce(tt::StreamsStub& accepted, tt::StreamsStub& lost); + // fill output data + void produce(const std::vector>& streamsIn, std::vector>& streamsOut); private: + // convert stub + StubGP* produce(const StubPP& stub, int phiT, int zT); // remove and return first element of deque, returns nullptr if empty template T* pop_front(std::deque& ts) const; - // true if truncation is enbaled bool enableTruncation_; // provides run-time constants const tt::Setup* setup_; // provides dataformats const DataFormats* dataFormats_; - // processing region (0 - 8) - const int region_; - // storage of input stubs - std::vector stubsPP_; + // provides layer encoding + const LayerEncoding* layerEncoding_; // storage of output stubs - std::vector stubsGP_; - // h/w liked organized pointer to input stubs - std::vector>> input_; + std::vector& stubs_; }; } // namespace trackerTFP diff --git a/L1Trigger/TrackerTFP/interface/HoughTransform.h b/L1Trigger/TrackerTFP/interface/HoughTransform.h index 680a76b325aa9..af13c702a6092 100644 --- a/L1Trigger/TrackerTFP/interface/HoughTransform.h +++ b/L1Trigger/TrackerTFP/interface/HoughTransform.h @@ -3,10 +3,10 @@ #include "L1Trigger/TrackTrigger/interface/Setup.h" #include "L1Trigger/TrackerTFP/interface/DataFormats.h" +#include "L1Trigger/TrackerTFP/interface/LayerEncoding.h" #include "DataFormats/L1TrackTrigger/interface/TTTypes.h" #include -#include #include namespace trackerTFP { @@ -14,51 +14,45 @@ namespace trackerTFP { // Class to find initial rough candidates in r-phi in a region class HoughTransform { public: - HoughTransform(const edm::ParameterSet& iConfig, const tt::Setup* setup, const DataFormats* dataFormats, int region); + HoughTransform(const edm::ParameterSet& iConfig, + const tt::Setup* setup, + const DataFormats* dataFormats, + const LayerEncoding* layerEncoding, + std::vector& stubs); ~HoughTransform() {} - - // read in and organize input product - void consume(const tt::StreamsStub& streams); // fill output products - void produce(tt::StreamsStub& accepted, tt::StreamsStub& lost); + void produce(const std::vector>& streamsIn, std::vector>& streamsOut); private: // remove and return first element of deque, returns nullptr if empty template T* pop_front(std::deque& ts) const; // associate stubs with phiT bins in this inv2R column - void fillIn(int inv2R, - std::deque& inputSector, - std::vector& acceptedSector, - std::vector& lostSector); + void fillIn(int inv2R, int sector, const std::vector& input, std::vector& output); // identify tracks - void readOut(const std::vector& acceptedSector, - const std::vector& lostSector, - std::deque& acceptedAll, - std::deque& lostAll) const; - // identify lost tracks - void analyze(); - // store tracks - void put() const; - + void readOut(const std::vector& input, std::deque& output) const; + // + bool noTrack(const TTBV& pattern, int zT) const; // true if truncation is enbaled bool enableTruncation_; // provides run-time constants const tt::Setup* setup_; // provides dataformats const DataFormats* dataFormats_; + // + const LayerEncoding* layerEncoding_; // data format of inv2R - DataFormat inv2R_; + const DataFormat* inv2R_; // data format of phiT - DataFormat phiT_; - // processing region (0 - 8) - int region_; - // container of input stubs - std::vector stubsGP_; - // container of output stubs - std::vector stubsHT_; - // h/w liked organized pointer to input stubs - std::vector>> input_; + const DataFormat* phiT_; + // data format of zT + const DataFormat* zT_; + // data format of phi + const DataFormat* phi_; + // data format of z + const DataFormat* z_; + // container of stubs + std::vector& stubs_; }; } // namespace trackerTFP diff --git a/L1Trigger/TrackerTFP/interface/KalmanFilter.h b/L1Trigger/TrackerTFP/interface/KalmanFilter.h index d875f6ad7756f..445b1342dd6cc 100644 --- a/L1Trigger/TrackerTFP/interface/KalmanFilter.h +++ b/L1Trigger/TrackerTFP/interface/KalmanFilter.h @@ -3,48 +3,125 @@ #include "L1Trigger/TrackTrigger/interface/Setup.h" #include "L1Trigger/TrackerTFP/interface/DataFormats.h" +#include "L1Trigger/TrackerTFP/interface/LayerEncoding.h" #include "L1Trigger/TrackerTFP/interface/KalmanFilterFormats.h" #include "L1Trigger/TrackerTFP/interface/State.h" #include "DataFormats/L1TrackTrigger/interface/TTTypes.h" +#include #include +#include namespace trackerTFP { // Class to do helix fit to all tracks in a region. class KalmanFilter { public: + typedef State::Stub Stub; KalmanFilter(const edm::ParameterSet& iConfig, const tt::Setup* setup, const DataFormats* dataFormats, + const LayerEncoding* layerEncoding, KalmanFilterFormats* kalmanFilterFormats, - int region); + std::vector& tracks, + std::vector& stubs); ~KalmanFilter() {} - // read in and organize input tracks and stubs - void consume(const tt::StreamsTrack& streamsTrack, const tt::StreamsStub& streamsStub); // fill output products - void produce(tt::StreamsStub& accpetedStubs, - tt::StreamsTrack& acceptedTracks, - tt::StreamsStub& lostStubs, - tt::StreamsTrack& lostTracks, + void produce(const std::vector>& tracksIn, + const std::vector>& stubsIn, + std::vector>& tracksOut, + std::vector>>& stubsOut, int& numAcceptedStates, - int& numLostStates); + int& numLostStates, + std::deque>& chi2s); private: + // + struct Track { + Track() {} + Track(int trackId, + int numConsistent, + int numConsistentPS, + double inv2R, + double phiT, + double cot, + double zT, + double chi20, + double chi21, + const TTBV& hitPattern, + TrackCTB* track, + const std::vector& stubs, + const std::vector& phi, + const std::vector& z) + : trackId_(trackId), + numConsistent_(numConsistent), + numConsistentPS_(numConsistentPS), + inv2R_(inv2R), + phiT_(phiT), + cot_(cot), + zT_(zT), + chi20_(chi20), + chi21_(chi21), + hitPattern_(hitPattern), + track_(track), + stubs_(stubs), + phi_(phi), + z_(z) {} + int trackId_; + int numConsistent_; + int numConsistentPS_; + double inv2R_; + double phiT_; + double cot_; + double zT_; + double chi20_; + double chi21_; + TTBV hitPattern_; + TrackCTB* track_; + std::vector stubs_; + std::vector phi_; + std::vector z_; + }; // remove and return first element of deque, returns nullptr if empty template T* pop_front(std::deque& ts) const; - // remove and return first element of vector, returns nullptr if empty - template - T* pop_front(std::vector& ts) const; + // constraints double precision + double digi(VariableKF var, double val) const { return kalmanFilterFormats_->format(var).digi(val); } + // + int integer(VariableKF var, double val) const { return kalmanFilterFormats_->format(var).integer(val); } + // + void updateRangeActual(VariableKF var, double val) { + return kalmanFilterFormats_->format(var).updateRangeActual(val); + } + // + double base(VariableKF var) const { return kalmanFilterFormats_->format(var).base(); } + // + int width(VariableKF var) const { return kalmanFilterFormats_->format(var).width(); } + // + int inRange(VariableKF var, double val) const { return kalmanFilterFormats_->format(var).inRange(val); } + // create Proto States + void createProtoStates(const std::vector>& tracksIn, + const std::vector>& stubsIn, + int channel, + std::deque& stream); + // calulcate seed parameter + void calcSeeds(std::deque& stream); + // apply final cuts + void finalize(const std::deque& stream, std::vector& finals); + // Transform States into Tracks + void conv(const std::vector& best, + std::vector& tracks, + std::vector>& stubs); // adds a layer to states void addLayer(std::deque& stream); + // adds a layer to states to build seeds + void addSeedLayer(std::deque& stream); // Assign next combinatoric (i.e. not first in layer) stub to state void comb(State*& state); // best state selection - void accumulator(std::deque& stream); + void accumulator(std::vector& finals, std::vector& best); // updates state void update(State*& state); @@ -54,59 +131,20 @@ namespace trackerTFP { const tt::Setup* setup_; // provides dataformats const DataFormats* dataFormats_; + // provides layer Encoding + const LayerEncoding* layerEncoding_; // provides dataformats of Kalman filter internals KalmanFilterFormats* kalmanFilterFormats_; - // processing region (0 - 8) - int region_; - // container of input stubs - std::vector stubs_; - // container of input tracks - std::vector tracks_; + // container of output tracks + std::vector& tracks_; + // container of output stubs + std::vector& stubs_; // container of all Kalman Filter states std::deque states_; - // h/w liked organized pointer to input stubs - std::vector> input_; + // + std::vector finals_; // current layer used during state propagation int layer_; - - // dataformats of Kalman filter internals - - DataFormatKF* x0_; - DataFormatKF* x1_; - DataFormatKF* x2_; - DataFormatKF* x3_; - DataFormatKF* H00_; - DataFormatKF* H12_; - DataFormatKF* m0_; - DataFormatKF* m1_; - DataFormatKF* v0_; - DataFormatKF* v1_; - DataFormatKF* r0_; - DataFormatKF* r1_; - DataFormatKF* S00_; - DataFormatKF* S01_; - DataFormatKF* S12_; - DataFormatKF* S13_; - DataFormatKF* K00_; - DataFormatKF* K10_; - DataFormatKF* K21_; - DataFormatKF* K31_; - DataFormatKF* R00_; - DataFormatKF* R11_; - DataFormatKF* R00Rough_; - DataFormatKF* R11Rough_; - DataFormatKF* invR00Approx_; - DataFormatKF* invR11Approx_; - DataFormatKF* invR00Cor_; - DataFormatKF* invR11Cor_; - DataFormatKF* invR00_; - DataFormatKF* invR11_; - DataFormatKF* C00_; - DataFormatKF* C01_; - DataFormatKF* C11_; - DataFormatKF* C22_; - DataFormatKF* C23_; - DataFormatKF* C33_; }; } // namespace trackerTFP diff --git a/L1Trigger/TrackerTFP/interface/KalmanFilterFormats.h b/L1Trigger/TrackerTFP/interface/KalmanFilterFormats.h index 918e61a69fc2c..1e3a35e1dee66 100644 --- a/L1Trigger/TrackerTFP/interface/KalmanFilterFormats.h +++ b/L1Trigger/TrackerTFP/interface/KalmanFilterFormats.h @@ -6,8 +6,6 @@ Classes to calculate and provide dataformats used by Kalman Filter emulator enabling tuning of bit widths ----------------------------------------------------------------------*/ -#include "FWCore/Framework/interface/data_default_record_trait.h" -#include "L1Trigger/TrackerTFP/interface/KalmanFilterFormatsRcd.h" #include "FWCore/ParameterSet/interface/ParameterSet.h" #include "L1Trigger/TrackerTFP/interface/DataFormats.h" @@ -39,6 +37,10 @@ namespace trackerTFP { S01, S12, S13, + S00Shifted, + S01Shifted, + S12Shifted, + S13Shifted, K00, K10, K21, @@ -59,6 +61,22 @@ namespace trackerTFP { C22, C23, C33, + r0Shifted, + r1Shifted, + r02, + r12, + chi20, + chi21, + dH, + invdH, + invdH2, + H2, + Hm0, + Hm1, + Hv0, + Hv1, + H2v0, + H2v1, end, x }; @@ -67,26 +85,51 @@ namespace trackerTFP { class DataFormatKF { public: - DataFormatKF(const VariableKF& v, bool twos); + DataFormatKF(const VariableKF& v, bool twos, const edm::ParameterSet& iConfig); virtual ~DataFormatKF() {} - double digi(double val) const { return (std::floor(val / base_ + 1.e-12) + .5) * base_; } + double digi(double val) const { + return enableIntegerEmulation_ ? (std::floor(val / base_ + 1.e-11) + .5) * base_ : val; + } bool twos() const { return twos_; } int width() const { return width_; } double base() const { return base_; } double range() const { return range_; } - const std::pair& rangeActual() const { return rangeActual_; } + double min() const { return min_; } + double abs() const { return abs_; } + double max() const { return max_; } // returns false if data format would oferflow for this double value bool inRange(double d) const; void updateRangeActual(double d); - int integer(double d) const { return floor(d / base_); } + int integer(double d) const { return floor(d / base_ + 1.e-11); } protected: VariableKF v_; bool twos_; + bool enableIntegerEmulation_; int width_; double base_; double range_; - std::pair rangeActual_; + double min_; + double abs_; + double max_; + }; + + class KalmanFilterFormats { + public: + KalmanFilterFormats(const edm::ParameterSet& iConfig); + ~KalmanFilterFormats() {} + void beginRun(const DataFormats* dataFormats); + const tt::Setup* setup() const { return dataFormats_->setup(); } + const DataFormats* dataFormats() const { return dataFormats_; } + DataFormatKF& format(VariableKF v) { return formats_[+v]; } + void endJob(); + + private: + template + void fillFormats(); + const edm::ParameterSet iConfig_; + const DataFormats* dataFormats_; + std::vector formats_; }; template @@ -97,6 +140,7 @@ namespace trackerTFP { private: void calcRange() { range_ = base_ * pow(2, width_); } + void calcWidth() { width_ = ceil(log2(range_ / base_) - 1.e-11); } }; template <> @@ -132,6 +176,14 @@ namespace trackerTFP { template <> FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig); template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig); + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig); + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig); + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig); + template <> FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig); template <> FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig); @@ -171,30 +223,40 @@ namespace trackerTFP { FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig); template <> FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig); + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig); + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig); + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig); + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig); + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig); + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig); - class KalmanFilterFormats { - public: - KalmanFilterFormats(); - KalmanFilterFormats(const edm::ParameterSet& iConfig, const DataFormats* dataFormats); - ~KalmanFilterFormats() {} - const tt::Setup* setup() const { return setup_; } - const DataFormats* dataFormats() const { return dataFormats_; } - int width(VariableKF v) const { return formats_[+v].width(); } - double base(VariableKF v) const { return formats_[+v].base(); } - DataFormatKF& format(VariableKF v) { return formats_[+v]; } - void endJob(); - - private: - template - void fillFormats(); - const edm::ParameterSet iConfig_; - const DataFormats* dataFormats_; - const tt::Setup* setup_; - std::vector formats_; - }; + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig); + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig); + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig); + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig); + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig); + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig); + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig); + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig); + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig); + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig); } // namespace trackerTFP -EVENTSETUP_DATA_DEFAULT_RECORD(trackerTFP::KalmanFilterFormats, trackerTFP::KalmanFilterFormatsRcd); - #endif diff --git a/L1Trigger/TrackerTFP/interface/KalmanFilterFormatsRcd.h b/L1Trigger/TrackerTFP/interface/KalmanFilterFormatsRcd.h deleted file mode 100644 index 86345133b31a5..0000000000000 --- a/L1Trigger/TrackerTFP/interface/KalmanFilterFormatsRcd.h +++ /dev/null @@ -1,17 +0,0 @@ -#ifndef L1Trigger_TrackerTFP_KalmanFilterFormatsRcd_h -#define L1Trigger_TrackerTFP_KalmanFilterFormatsRcd_h - -#include "FWCore/Framework/interface/DependentRecordImplementation.h" -#include "L1Trigger/TrackerTFP/interface/DataFormatsRcd.h" -#include "FWCore/Utilities/interface/mplVector.h" - -namespace trackerTFP { - - typedef edm::mpl::Vector RcdsKalmanFilterFormats; - - class KalmanFilterFormatsRcd - : public edm::eventsetup::DependentRecordImplementation {}; - -} // namespace trackerTFP - -#endif diff --git a/L1Trigger/TrackerTFP/interface/LayerEncoding.h b/L1Trigger/TrackerTFP/interface/LayerEncoding.h index a5adaf15b8b0b..ef2642d4d2803 100644 --- a/L1Trigger/TrackerTFP/interface/LayerEncoding.h +++ b/L1Trigger/TrackerTFP/interface/LayerEncoding.h @@ -12,7 +12,7 @@ namespace trackerTFP { /*! \class trackerTFP::LayerEncoding * \brief Class to encode layer ids for Kalman Filter - * Layers consitent with rough r-z track parameters are counted from 0 onwards. + * Layers (1 to 6 for barrel, 11 to 15 for end caps) consitent with rough r-z track parameters are counted from 0 onwards (0 to 7). * \author Thomas Schuh * \date 2020, July */ @@ -21,21 +21,14 @@ namespace trackerTFP { LayerEncoding() {} LayerEncoding(const DataFormats* dataFormats); ~LayerEncoding() {} - // Set of layers in each (zT,tanL) digi Bin of each eta sector numbered 0->N - const std::vector& layerEncoding(int binEta, int binZT, int binCot) const { - return layerEncoding_.at(binEta).at(binZT).at(binCot); - } - const std::map& layerEncodingMap(int binEta, int binZT, int binCot) const { - return layerEncodingMap_.at(binEta).at(binZT).at(binCot); - } - // maybe layers for given ets sector, bin in zT and bin in cotThea - const std::vector& maybeLayer(int binEta, int binZT, int binCot) const { - return maybeLayer_.at(binEta).at(binZT).at(binCot); - } - // encoded layer id for given eta sector, bin in zT, bin in cotThea and decoed layer id, returns -1 if layer incositent with track - const int layerIdKF(int binEta, int binZT, int binCot, int layerId) const; - // pattern of maybe layers for given eta sector, bin in zT and bin in cotThea - TTBV maybePattern(int binEta, int binZT, int binCot) const; + // Set of layers for given bin in zT + const std::vector& layerEncoding(int zT) const; + // Set of layers for given zT in cm + const std::vector& layerEncoding(double zT) const; + // pattern of maybe layers for given bin in zT + const TTBV& maybePattern(int zT) const; + // pattern of maybe layers for given zT in cm + const TTBV& maybePattern(double zT) const; private: // helper class providing run-time constants @@ -44,13 +37,12 @@ namespace trackerTFP { const DataFormats* dataFormats_; // data foramt of variable zT const DataFormat* zT_; - // data foramt of variable cotTheta - const DataFormat* cot_; - // outer to inner indices: eta sector, bin in zT, bin in cotTheta, layerId - std::vector>>> layerEncoding_; - std::vector>>> layerEncodingMap_; - // outer to inner indices: eta sector, bin in zT, bin in cotTheta, layerId of maybe layers - std::vector>>> maybeLayer_; + // outer to inner indices: bin in zT, layerId + std::vector> layerEncoding_; + // outer to inner indices: bin in zT, maybe patterns + std::vector maybePattern_; + std::vector nullLE_; + TTBV nullMP_; }; } // namespace trackerTFP diff --git a/L1Trigger/TrackerTFP/interface/MiniHoughTransform.h b/L1Trigger/TrackerTFP/interface/MiniHoughTransform.h deleted file mode 100644 index 2c3a622d1c2be..0000000000000 --- a/L1Trigger/TrackerTFP/interface/MiniHoughTransform.h +++ /dev/null @@ -1,69 +0,0 @@ -#ifndef L1Trigger_TrackerTFP_MiniHoughTransform_h -#define L1Trigger_TrackerTFP_MiniHoughTransform_h - -#include "L1Trigger/TrackTrigger/interface/Setup.h" -#include "L1Trigger/TrackerTFP/interface/DataFormats.h" -#include "DataFormats/L1TrackTrigger/interface/TTTypes.h" - -#include -#include -#include - -namespace trackerTFP { - - // Class to refine HT track candidates in r-phi, by subdividing each HT cell into a finer granularity array - class MiniHoughTransform { - public: - MiniHoughTransform(const edm::ParameterSet& iConfig, - const tt::Setup* setup, - const DataFormats* dataFormats, - int region); - ~MiniHoughTransform() {} - - // read in and organize input product (fill vector input_) - void consume(const tt::StreamsStub& streams); - // fill output products - void produce(tt::StreamsStub& accepted, tt::StreamsStub& lost); - - private: - // remove and return first element of deque, returns nullptr if empty - template - T* pop_front(std::deque& ts) const; - // perform finer pattern recognition per track - void fill(int channel, const std::vector& input, std::vector>& output); - // Static load balancing of inputs: mux 4 streams to 1 stream - void slb(std::vector>& inputs, std::vector& accepted, tt::StreamStub& lost) const; - // Dynamic load balancing of inputs: swapping parts of streams to balance the amount of tracks per stream - void dlb(std::vector>& streams) const; - - // true if truncation is enbaled - bool enableTruncation_; - // provides run-time constants - const tt::Setup* setup_; - // provides dataformats - const DataFormats* dataFormats_; - // dataformat of inv2R - DataFormat inv2R_; - // dataformat of phiT - DataFormat phiT_; - // processing region (0 - 8) - int region_; - // number of inv2R bins used in HT - int numBinsInv2R_; - // number of cells used in MHT - int numCells_; - // number of dynamic load balancing nodes - int numNodes_; - // number of channel per dynamic load balancing node - int numChannel_; - // container of input stubs - std::vector stubsHT_; - // container of output stubs - std::vector stubsMHT_; - // h/w liked organized pointer to input stubs - std::vector> input_; - }; - -} // namespace trackerTFP - -#endif diff --git a/L1Trigger/TrackerTFP/interface/State.h b/L1Trigger/TrackerTFP/interface/State.h index fe6806b1e0ea3..5f49750d2b709 100644 --- a/L1Trigger/TrackerTFP/interface/State.h +++ b/L1Trigger/TrackerTFP/interface/State.h @@ -2,48 +2,59 @@ #define L1Trigger_TrackerTFP_State_h #include "L1Trigger/TrackTrigger/interface/Setup.h" -#include "L1Trigger/TrackerTFP/interface/DataFormats.h" +#include "L1Trigger/TrackerTFP/interface/KalmanFilterFormats.h" #include #include namespace trackerTFP { - // Class to represent a Kalman Filter State + // Class to represent a Kalman Filter helix State class State { public: - // default constructor + // + struct Stub { + Stub(KalmanFilterFormats* formats, const tt::FrameStub& frame); + StubCTB stubCTB_; + double H12_; + double v0_; + double v1_; + }; + // copy constructor State(State* state); // proto state constructor - State(const DataFormats* dataFormats, TrackKFin* track, int trackId); - // combinatoric state constructor - State(State* state, StubKFin* stub); + State(KalmanFilterFormats* formats, + TrackCTB* track, + const std::vector>& stubs, + const TTBV& maybe, + int trackId); // updated state constructor State(State* state, const std::vector& doubles); + // combinatoric and seed building state constructor + State(State* state, State* parent, Stub* stub, int layer); ~State() {} - - // Determine quality of completed state - void finish(); - // number of skipped layers - int numSkippedLayers() const { return numSkippedLayers_; } - // number of consitent layers - int numConsistentLayers() const { return numConsistentLayers_; } + // + State* comb(std::deque& states, int layer); + // + State* combSeed(std::deque& states, int layer); + // + State* update(std::deque& states, int layer); // input track - TrackKFin* track() const { return track_; } + TrackCTB* track() const { return track_; } // parent state (nullpointer if no parent available) State* parent() const { return parent_; } // stub to add to state - StubKFin* stub() const { return stub_; } + Stub* stub() const { return stub_; } // hitPattern of so far added stubs const TTBV& hitPattern() const { return hitPattern_; } + // shows which layer the found track has stubs on + const TTBV& trackPattern() const { return trackPattern_; } // track id of input track int trackId() const { return trackId_; } // pattern of maybe layers for input track - TTBV maybePattern() const { return track_->maybePattern(); } - // stub id per layer of so far added stubs - const std::vector& layerMap() const { return layerMap_; } + const TTBV& maybePattern() const { return maybePattern_; } // layer id of the current stub to add - int layer() const { return stub_->layer(); } + int layer() const { return layer_; } // helix inv2R wrt input helix double x0() const { return x0_; } // helix phi at radius ChosenRofPhi wrt input helix @@ -52,6 +63,10 @@ namespace trackerTFP { double x2() const { return x2_; } // helix z at radius chosenRofZ wrt input helix double x3() const { return x3_; } + // chi2 for the r-phi plane straight line fit + double chi20() const { return chi20_; } + // chi2 for the r-z plane straight line fit + double chi21() const { return chi21_; } // cov. matrix element double C00() const { return C00_; } // cov. matrix element @@ -65,65 +80,67 @@ namespace trackerTFP { // cov. matrix element double C33() const { return C33_; } // Derivative of predicted stub coords wrt helix params: stub radius minus chosenRofPhi - double H00() const { return stub_->r(); } + double H00() const { return stub_->stubCTB_.r(); } // Derivative of predicted stub coords wrt helix params: stub radius minus chosenRofZ - double H12() const { return stub_->r() + dataFormats_->chosenRofPhi() - setup_->chosenRofZ(); } + double H12() const { return stub_->H12_; } // stub phi residual wrt input helix - double m0() const { return stub_->phi(); } + double m0() const { return stub_->stubCTB_.phi(); } // stub z residual wrt input helix - double m1() const { return stub_->z(); } + double m1() const { return stub_->stubCTB_.z(); } // stub projected phi uncertainty - double dPhi() const { return stub_->dPhi(); } + double dPhi() const { return stub_->stubCTB_.dPhi(); } // stub projected z uncertainty - double dZ() const { return stub_->dZ(); } + double dZ() const { return stub_->stubCTB_.dZ(); } // squared stub projected phi uncertainty instead of wheight (wrong but simpler) - double v0() const { return pow(stub_->dPhi(), 2); } + double v0() const { return stub_->v0_; } // squared stub projected z uncertainty instead of wheight (wrong but simpler) - double v1() const { return pow(stub_->dZ(), 2); } - // output frame - tt::FrameTrack frame() const { return TrackKF(*track_, x1_, x0_, x3_, x2_).frame(); } - // fill collection of stubs added so far to state - void fill(std::vector& stubs) const; + double v1() const { return stub_->v1_; } + //const std::vector>& stubs() const { return stubs_; } private: + // + bool gapCheck(int layer) const; // provides data fomats - const DataFormats* dataFormats_; + KalmanFilterFormats* formats_; // provides run-time constants const tt::Setup* setup_; // input track - TrackKFin* track_; + TrackCTB* track_; + // input track stubs + std::vector> stubs_; + // pattern of maybe layers for input track + TTBV maybePattern_; // track id int trackId_; // previous state, nullptr for first states - State* parent_; + State* parent_ = nullptr; // stub to add - StubKFin* stub_; - // shows which stub on each layer has been added so far - std::vector layerMap_; + Stub* stub_ = nullptr; + // layer id of the current stub to add + int layer_ = 0; // shows which layer has been added so far TTBV hitPattern_; + // shows which layer the found track has stubs on + TTBV trackPattern_; // helix inv2R wrt input helix - double x0_; + double x0_ = 0.; // helix phi at radius ChosenRofPhi wrt input helix - double x1_; + double x1_ = 0.; // helix cot(Theta) wrt input helix - double x2_; + double x2_ = 0.; // helix z at radius chosenRofZ wrt input helix - double x3_; - + double x3_ = 0.; + // chi2 for the r-phi plane straight line fit + double chi20_ = 0.; + // chi2 for the r-z plane straight line fit + double chi21_ = 0.; // cov. matrix - - double C00_; - double C01_; - double C11_; - double C22_; - double C23_; - double C33_; - - // number of skipped layers - int numSkippedLayers_; - // number of consistent layers - int numConsistentLayers_; + double C00_ = 9.e9; + double C01_ = 0.; + double C11_ = 9.e9; + double C22_ = 9.e9; + double C23_ = 0.; + double C33_ = 9.e9; }; } // namespace trackerTFP diff --git a/L1Trigger/TrackerTFP/interface/TrackFindingProcessor.h b/L1Trigger/TrackerTFP/interface/TrackFindingProcessor.h new file mode 100644 index 0000000000000..fcc67c4146def --- /dev/null +++ b/L1Trigger/TrackerTFP/interface/TrackFindingProcessor.h @@ -0,0 +1,89 @@ +#ifndef L1Trigger_TrackerTFP_TrackFindingProcessor_h +#define L1Trigger_TrackerTFP_TrackFindingProcessor_h + +#include "FWCore/ParameterSet/interface/ParameterSet.h" +#include "L1Trigger/TrackTrigger/interface/Setup.h" +#include "L1Trigger/TrackerTFP/interface/DataFormats.h" +#include "L1Trigger/TrackerTFP/interface/TrackQuality.h" +#include "DataFormats/L1TrackTrigger/interface/TTTypes.h" + +#include +#include +#include + +namespace trackerTFP { + + // Class to format final tfp output and to prodcue final TTTrackCollection + class TrackFindingProcessor { + public: + TrackFindingProcessor(const edm::ParameterSet& iConfig, + const tt::Setup* setup_, + const DataFormats* dataFormats, + const TrackQuality* trackQuality); + ~TrackFindingProcessor() {} + + // produce TTTracks + void produce(const tt::StreamsTrack& inputs, + const tt::StreamsStub& stubs, + tt::TTTracks& ttTracks, + tt::StreamsTrack& outputs); + // produce StreamsTrack + void produce(const std::vector& inputs, tt::StreamsTrack& outputs) const; + + private: + // + static constexpr int partial_width = 32; + // + static constexpr int partial_in = 3; + // + static constexpr int partial_out = 2; + // + typedef std::bitset PartialFrame; + // + typedef std::pair PartialFrameTrack; + // + struct Track { + Track(const tt::FrameTrack& frameTrack, + const tt::Frame& frameTQ, + const std::vector& ttStubRefs, + const TrackQuality* tq); + const TTTrackRef& ttTrackRef_; + const std::vector ttStubRefs_; + bool valid_; + std::vector partials_; + TTBV hitPattern_; + int channel_; + int mva_; + double inv2R_; + double phiT_; + double cot_; + double zT_; + double chi2rphi_; + double chi2rz_; + }; + // remove and return first element of deque, returns nullptr if empty + template + T* pop_front(std::deque& ts) const; + // + void consume(const tt::StreamsTrack& inputs, + const tt::StreamsStub& stubs, + std::vector>& outputs); + // emualte data format f/w + void produce(std::vector>& inputs, tt::StreamsTrack& outputs) const; + // produce TTTracks + void produce(const tt::StreamsTrack& inputs, tt::TTTracks& ouputs) const; + // true if truncation is enbaled + bool enableTruncation_; + // provides run-time constants + const tt::Setup* setup_; + // provides data formats + const DataFormats* dataFormats_; + // provides Track Quality algo and formats + const TrackQuality* trackQuality_; + // storage of tracks + std::vector tracks_; + }; + +} // namespace trackerTFP + +#endif \ No newline at end of file diff --git a/L1Trigger/TrackerTFP/interface/TrackQuality.h b/L1Trigger/TrackerTFP/interface/TrackQuality.h new file mode 100644 index 0000000000000..cfc32f8bfc8be --- /dev/null +++ b/L1Trigger/TrackerTFP/interface/TrackQuality.h @@ -0,0 +1,156 @@ +/* +Track Quality Header file +C.Brown 28/07/20 +*/ + +#ifndef L1Trigger_TrackerTFP_TrackQuality_h +#define L1Trigger_TrackerTFP_TrackQuality_h + +#include "FWCore/Framework/interface/data_default_record_trait.h" +#include "FWCore/ParameterSet/interface/ParameterSet.h" +#include "DataFormats/L1TrackTrigger/interface/TTTypes.h" +#include "L1Trigger/TrackerTFP/interface/DataFormats.h" +#include "L1Trigger/TrackerTFP/interface/TrackQualityRcd.h" + +#include +#include +#include +#include "ap_fixed.h" + +namespace trackerTFP { + + // number of mva bins + static constexpr int numBinsMVA_ = 1 << TTTrack_TrackWord::TrackBitWidths::kMVAQualitySize; + // number of chi2B bins + static constexpr int numBinsChi2B_ = 1 << TTTrack_TrackWord::TrackBitWidths::kBendChi2Size; + // number of chi2rphi bins + static constexpr int numBinschi2rphi_ = 1 << TTTrack_TrackWord::TrackBitWidths::kChi2RPhiSize; + // number of chi2rz bins + static constexpr int numBinschi2rz_ = 1 << TTTrack_TrackWord::TrackBitWidths::kChi2RZSize; + + // track quality variables + enum class VariableTQ { begin, m20 = begin, m21, invV0, invV1, chi2rphi, chi2rz, end, x }; + // conversion: Variable to int + inline constexpr int operator+(VariableTQ v) { return static_cast(v); } + // increment of Variable + inline constexpr VariableTQ operator++(VariableTQ v) { return VariableTQ(+v + 1); } + + // class representing format of a specific variable + template + class FormatTQ : public DataFormat { + public: + FormatTQ(const DataFormats* dataFormats, const edm::ParameterSet& iConfig); + ~FormatTQ() {} + + private: + void calcRange() { range_ = base_ * pow(2, width_); } + void calcWidth() { width_ = ceil(log2(range_ / base_) - 1.e-11); } + void calcBase() { base_ = range_ * pow(2, -width_); } + }; + template <> + FormatTQ::FormatTQ(const DataFormats* dataFormats, const edm::ParameterSet& iConfig); + template <> + FormatTQ::FormatTQ(const DataFormats* dataFormats, const edm::ParameterSet& iConfig); + template <> + FormatTQ::FormatTQ(const DataFormats* dataFormats, const edm::ParameterSet& iConfig); + template <> + FormatTQ::FormatTQ(const DataFormats* dataFormats, const edm::ParameterSet& iConfig); + template <> + FormatTQ::FormatTQ(const DataFormats* dataFormats, const edm::ParameterSet& iConfig); + template <> + FormatTQ::FormatTQ(const DataFormats* dataFormats, const edm::ParameterSet& iConfig); + + /*! \class trackerTFP::TrackQuality + * \brief Bit accurate emulation of the track quality BDT + * \author C.Brown + * \date 28/07/20 + * \update 2024, June by Claire Savard + * \update 2024, Aug by Thomas Schuh + */ + class TrackQuality { + public: + TrackQuality() {} + TrackQuality(const edm::ParameterSet& iConfig, const DataFormats* dataFormats); + ~TrackQuality() {} + // object to represent tracks + struct Track { + Track(const tt::FrameTrack& frameTrack, const tt::StreamStub& streamStub, const TrackQuality* tq); + // return the iths frame of this track + const tt::FrameTrack& frame(int i) const { return frames_.at(i); } + // collection of frames forming track + tt::StreamTrack frames_; + }; + // provides dataformats + const DataFormats* dataFormats() const { return dataFormats_; } + // Controls the conversion between TTTrack features and ML model training features + std::vector featureTransform(TTTrack& aTrack, + const std::vector& featureNames) const; + // Passed by reference a track without MVA filled, method fills the track's MVA field + void setL1TrackQuality(TTTrack& aTrack) const; + // Helper function to convert mvaPreSig to bin + int toBinMVA(double mva) const; + // Helper function to convert chi2B to bin + int toBinChi2B(double chi2B) const; + // Helper function to convert chi2rphi to bin + int toBinchi2rphi(double chi2rphi) const; + // Helper function to convert chi2rz to bin + int toBinchi2rz(double chi2rz) const; + // + double scaleCot(int cot) const { return scaleAP(scale(cot, baseShiftCot_)); } + // + double scaleZ0(int z0) const { return scaleAP(scale(z0, baseShiftZ0_)); } + // access to spedific format + const DataFormat& format(VariableTQ v) const { return dataFormatsTQ_[+v]; } + // + double base(VariableTQ v) const { return dataFormatsTQ_[+v].base(); } + // + double range(VariableTQ v) const { return dataFormatsTQ_[+v].range(); } + // + const edm::FileInPath& model() const { return model_; } + + private: + // constructs TQ data formats + template + void fillDataFormats(const edm::ParameterSet& iConfig); + // TQ MVA bin conversion LUT + constexpr std::array mvaPreSigBins() const; + // + static constexpr double invSigmoid(double value) { return -log(1. / value - 1.); } + // + template + int toBin(const T& bins, double d) const; + // + int scale(int i, int shift) const { return floor(i * pow(2., shift)); } + // + double scaleAP(int i) const { return i * pow(2., baseShiftAPfixed_); } + // provides dataformats + const DataFormats* dataFormats_; + // + edm::FileInPath model_; + // + std::vector featureNames_; + // + double baseShiftCot_; + // + double baseShiftZ0_; + // + double baseShiftAPfixed_; + // Conversion factor between dphi^2/weight and chi2rphi + int chi2rphiConv_; + // Conversion factor between dz^2/weight and chi2rz + int chi2rzConv_; + // Fraction of total dphi and dz ranges to calculate v0 and v1 LUT for + int weightBinFraction_; + // Constant used in FW to prevent 32-bit int overflow + int dzTruncation_; + // Constant used in FW to prevent 32-bit int overflow + int dphiTruncation_; + // collection of unique formats + std::vector dataFormatsTQ_; + }; + +} // namespace trackerTFP + +EVENTSETUP_DATA_DEFAULT_RECORD(trackerTFP::TrackQuality, trackerTFP::TrackQualityRcd); + +#endif diff --git a/L1Trigger/TrackerTFP/interface/TrackQualityRcd.h b/L1Trigger/TrackerTFP/interface/TrackQualityRcd.h new file mode 100644 index 0000000000000..d65bd922c9393 --- /dev/null +++ b/L1Trigger/TrackerTFP/interface/TrackQualityRcd.h @@ -0,0 +1,17 @@ +#ifndef L1Trigger_TrackerTFP_TrackQualityRcd_h +#define L1Trigger_TrackerTFP_TrackQualityRcd_h + +#include "FWCore/Framework/interface/DependentRecordImplementation.h" +#include "L1Trigger/TrackerTFP/interface/DataFormatsRcd.h" +#include "FWCore/Utilities/interface/mplVector.h" + +namespace trackerTFP { + + typedef edm::mpl::Vector RcdsTrackQuality; + + // record of trackerTFP::TrackQuality + class TrackQualityRcd : public edm::eventsetup::DependentRecordImplementation {}; + +} // namespace trackerTFP + +#endif \ No newline at end of file diff --git a/L1Trigger/TrackerTFP/interface/ZHoughTransform.h b/L1Trigger/TrackerTFP/interface/ZHoughTransform.h deleted file mode 100644 index 049c905f23812..0000000000000 --- a/L1Trigger/TrackerTFP/interface/ZHoughTransform.h +++ /dev/null @@ -1,56 +0,0 @@ -#ifndef L1Trigger_TrackerTFP_ZHoughTransform_h -#define L1Trigger_TrackerTFP_ZHoughTransform_h - -#include "L1Trigger/TrackTrigger/interface/Setup.h" -#include "L1Trigger/TrackerTFP/interface/DataFormats.h" -#include "DataFormats/L1TrackTrigger/interface/TTTypes.h" - -#include -#include - -namespace trackerTFP { - - // Class to refine MHT track candidates in r-z - class ZHoughTransform { - public: - ZHoughTransform(const edm::ParameterSet& iConfig, - const tt::Setup* setup, - const DataFormats* dataFormats, - int region); - ~ZHoughTransform() {} - - // read in and organize input product (fill vector input_) - void consume(const tt::StreamsStub& streams); - // fill output products - void produce(tt::StreamsStub& accepted, tt::StreamsStub& lost); - - private: - // remove and return first element of deque, returns nullptr if empty - template - T* pop_front(std::deque& ts) const; - // perform finer pattern recognition per track - void fill(int channel, const std::deque& input, std::vector>& output); - // Static load balancing of inputs: mux 4 streams to 1 stream - void slb(std::vector>& inputs, std::deque& accepted, tt::StreamStub& lost) const; - // - void merge(std::deque& stubs, tt::StreamStub& stream) const; - - // true if truncation is enbaled - bool enableTruncation_; - // provides run-time constants - const tt::Setup* setup_; - // provides dataformats - const DataFormats* dataFormats_; - // processing region (0 - 8) - int region_; - // container of in- and output stubs - std::vector stubsZHT_; - // h/w liked organized pointer to input stubs - std::vector> input_; - // - int stage_; - }; - -} // namespace trackerTFP - -#endif diff --git a/L1Trigger/TrackerTFP/plugins/ProducerCTB.cc b/L1Trigger/TrackerTFP/plugins/ProducerCTB.cc new file mode 100644 index 0000000000000..517a4dd5525cc --- /dev/null +++ b/L1Trigger/TrackerTFP/plugins/ProducerCTB.cc @@ -0,0 +1,246 @@ +#include "FWCore/Framework/interface/stream/EDProducer.h" +#include "FWCore/Framework/interface/Run.h" +#include "FWCore/Framework/interface/EventSetup.h" +#include "FWCore/Framework/interface/Event.h" +#include "FWCore/Framework/interface/MakerMacros.h" +#include "FWCore/Utilities/interface/EDGetToken.h" +#include "FWCore/Utilities/interface/EDPutToken.h" +#include "FWCore/Utilities/interface/ESGetToken.h" +#include "FWCore/Utilities/interface/InputTag.h" +#include "FWCore/ParameterSet/interface/ParameterSet.h" +#include "DataFormats/Common/interface/Handle.h" +#include "DataFormats/Common/interface/OrphanHandle.h" + +#include "DataFormats/L1TrackTrigger/interface/TTTypes.h" +#include "L1Trigger/TrackTrigger/interface/Setup.h" +#include "L1Trigger/TrackerTFP/interface/DataFormats.h" +#include "L1Trigger/TrackerTFP/interface/CleanTrackBuilder.h" + +#include +#include +#include +#include +#include +#include +#include + +using namespace std; +using namespace edm; +using namespace tt; + +namespace trackerTFP { + + /*! \class trackerTFP::ProducerCTB + * \brief clean HT tracks and rrestructures them + * \author Thomas Schuh + * \date 2020, July + */ + class ProducerCTB : public stream::EDProducer<> { + public: + explicit ProducerCTB(const ParameterSet&); + ~ProducerCTB() override {} + + private: + void beginRun(const Run&, const EventSetup&) override; + void produce(Event&, const EventSetup&) override; + virtual void endJob() {} + // ED input token of Stubs + EDGetTokenT edGetToken_; + // ED output token for TTTracks + EDPutTokenT edPutTokenTTTracks_; + // ED output token for stubs + EDPutTokenT edPutTokenStubs_; + // ED output token for tracks + EDPutTokenT edPutTokenTracks_; + // Setup token + ESGetToken esGetTokenSetup_; + // DataFormats token + ESGetToken esGetTokenDataFormats_; + // LayerEncoding token + ESGetToken esGetTokenLayerEncoding_; + // configuration + ParameterSet iConfig_; + // helper class to store configurations + const Setup* setup_ = nullptr; + // helper class to extract structured data from tt::Frames + const DataFormats* dataFormats_ = nullptr; + // + const LayerEncoding* layerEncoding_ = nullptr; + // + bool enableTruncation_; + // + DataFormat cot_; + }; + + ProducerCTB::ProducerCTB(const ParameterSet& iConfig) : iConfig_(iConfig) { + const string& label = iConfig.getParameter("InputLabelCTB"); + const string& branchStubs = iConfig.getParameter("BranchStubs"); + const string& branchTracks = iConfig.getParameter("BranchTracks"); + // book in- and output ED products + edGetToken_ = consumes(InputTag(label, branchStubs)); + edPutTokenStubs_ = produces(branchStubs); + edPutTokenTTTracks_ = produces(branchTracks); + edPutTokenTracks_ = produces(branchTracks); + // book ES products + esGetTokenSetup_ = esConsumes(); + esGetTokenDataFormats_ = esConsumes(); + esGetTokenLayerEncoding_ = esConsumes(); + // + enableTruncation_ = iConfig.getParameter("EnableTruncation"); + } + + void ProducerCTB::beginRun(const Run& iRun, const EventSetup& iSetup) { + // helper class to store configurations + setup_ = &iSetup.getData(esGetTokenSetup_); + // helper class to extract structured data from tt::Frames + dataFormats_ = &iSetup.getData(esGetTokenDataFormats_); + // + layerEncoding_ = &iSetup.getData(esGetTokenLayerEncoding_); + // + const double baseZ = dataFormats_->base(Variable::z, Process::ctb); + const double baseR = dataFormats_->base(Variable::r, Process::ctb); + const double range = dataFormats_->range(Variable::cot, Process::kf); + const int baseShift = ceil(log2(range / baseZ * baseR / setup_->ctbNumBinsCot())); + const int width = ceil(log2(setup_->ctbNumBinsCot())); + const double base = baseZ / baseR * pow(2, baseShift); + cot_ = DataFormat(true, width, base, range); + } + + void ProducerCTB::produce(Event& iEvent, const EventSetup& iSetup) { + static const int numChannelIn = dataFormats_->numChannel(Process::ht); + static const int numChannelOut = dataFormats_->numChannel(Process::ctb); + static const int numRegions = setup_->numRegions(); + static const int numLayers = setup_->numLayers(); + // empty output products + StreamsTrack acceptedTracks(numRegions * numChannelOut); + StreamsStub acceptedStubs(numRegions * numChannelOut * numLayers); + vector>> streamsTracks(numRegions, vector>(numChannelOut)); + vector>>> streamsStubs( + numRegions, vector>>(numChannelOut, vector>(numLayers))); + // read input Product and produce output product + Handle handleGet; + iEvent.getByToken(edGetToken_, handleGet); + const StreamsStub& streamsStub = *handleGet.product(); + // count stubs + int nStubsHT(0); + auto validFrame = [](int& sum, const FrameStub& frame) { return sum += (frame.first.isNonnull() ? 1 : 0); }; + for (const StreamStub& stream : streamsStub) + nStubsHT += accumulate(stream.begin(), stream.end(), 0, validFrame); + // create input objects and count tracks + vector stubsHT; + stubsHT.reserve(nStubsHT); + // count stubs + int nTracksHT(0); + for (const StreamStub& stream : streamsStub) { + pair trackId({setup_->htNumBinsPhiT(), setup_->gpNumBinsZT()}); + for (const FrameStub& frame : stream) { + if (frame.first.isNull()) + continue; + stubsHT.emplace_back(frame, dataFormats_); + StubHT* stub = &stubsHT.back(); + if (trackId.first != stub->phiT() || trackId.second != stub->zT()) { + nTracksHT++; + trackId = {stub->phiT(), stub->zT()}; + } + } + } + // object to clean and restructure tracks + vector stubsCTB; + vector tracksCTB; + tracksCTB.reserve(nTracksHT); + stubsCTB.reserve(nStubsHT); + CleanTrackBuilder ctb(iConfig_, setup_, dataFormats_, layerEncoding_, cot_, stubsCTB, tracksCTB); + int iStub(0); + for (int region = 0; region < numRegions; region++) { + const int offsetIn = region * numChannelIn; + const int offsetOut = region * numChannelOut; + // read h/w liked organized pointer to input data + vector> streamsIn(numChannelIn); + for (int channelIn = 0; channelIn < numChannelIn; channelIn++) { + const StreamStub& channelStubs = streamsStub[offsetIn + channelIn]; + vector& stream = streamsIn[channelIn]; + stream.reserve(channelStubs.size()); + for (const FrameStub& frame : channelStubs) + stream.push_back(frame.first.isNull() ? nullptr : &stubsHT[iStub++]); + } + // empty h/w liked organized pointer to output data + vector>& regionTracks = streamsTracks[region]; + vector>>& regionStubs = streamsStubs[region]; + // fill output data + ctb.produce(streamsIn, regionTracks, regionStubs); + // fill ed stubs + for (int channelOut = 0; channelOut < numChannelOut; channelOut++) { + const int offset = (offsetOut + channelOut) * numLayers; + const vector>& channelStubs = regionStubs[channelOut]; + for (int layer = 0; layer < numLayers; layer++) { + StreamStub& accepted = acceptedStubs[offset + layer]; + const deque& layerStubs = channelStubs[layer]; + accepted.reserve(layerStubs.size()); + for (StubCTB* stub : layerStubs) + accepted.emplace_back(stub ? stub->frame() : FrameStub()); + } + } + } + // store TTTracks + int nTracks(0); + auto valid = [](int& sum, TrackCTB* track) { return sum += (track ? 1 : 0); }; + for (const vector>& region : streamsTracks) + for (const deque& channel : region) + nTracks += accumulate(channel.begin(), channel.end(), 0, valid); + TTTracks ttTracks; + ttTracks.reserve(nTracks); + for (int region = 0; region < numRegions; region++) { + const vector>& regionTracks = streamsTracks[region]; + const vector>>& regionStubs = streamsStubs[region]; + for (int channelOut = 0; channelOut < numChannelOut; channelOut++) { + const deque& channelTracks = regionTracks[channelOut]; + const vector>& channelStubs = regionStubs[channelOut]; + for (int frame = 0; frame < (int)channelTracks.size(); frame++) { + TrackCTB* track = channelTracks[frame]; + if (!track) + continue; + const auto begin = next(channelTracks.begin(), frame); + const auto end = find_if(begin + 1, channelTracks.end(), [](TrackCTB* track) { return track; }); + const int size = distance(begin, end); + vector> stubs(numLayers); + for (int layer = 0; layer < numLayers; layer++) { + const deque& layerStubs = channelStubs[layer]; + vector& layerTrack = stubs[layer]; + layerTrack.reserve(size); + for (int s = 0; s < size; s++) { + StubCTB* stub = layerStubs[frame + s]; + if (stub) + layerTrack.push_back(stub); + } + } + ctb.put(track, stubs, region, ttTracks); + } + } + } + const OrphanHandle handle = iEvent.emplace(edPutTokenTTTracks_, move(ttTracks)); + // add TTTrackRefs + int iTrk(0); + int iChan(0); + for (const vector>& region : streamsTracks) { + for (const deque& stream : region) { + StreamTrack& streamTrack = acceptedTracks[iChan++]; + for (TrackCTB* track : stream) { + if (!track) { + streamTrack.emplace_back(FrameTrack()); + continue; + } + FrameTrack frame = track->frame(); + frame.first = TTTrackRef(handle, iTrk++); + streamTrack.emplace_back(frame); + } + } + } + // store tracks + iEvent.emplace(edPutTokenTracks_, move(acceptedTracks)); + // store stubs + iEvent.emplace(edPutTokenStubs_, move(acceptedStubs)); + } + +} // namespace trackerTFP + +DEFINE_FWK_MODULE(trackerTFP::ProducerCTB); \ No newline at end of file diff --git a/L1Trigger/TrackerTFP/plugins/ProducerDR.cc b/L1Trigger/TrackerTFP/plugins/ProducerDR.cc new file mode 100644 index 0000000000000..3661adcf49c3c --- /dev/null +++ b/L1Trigger/TrackerTFP/plugins/ProducerDR.cc @@ -0,0 +1,188 @@ +#include "FWCore/Framework/interface/stream/EDProducer.h" +#include "FWCore/Framework/interface/Run.h" +#include "FWCore/Framework/interface/EventSetup.h" +#include "FWCore/Framework/interface/Event.h" +#include "FWCore/Framework/interface/MakerMacros.h" +#include "FWCore/Utilities/interface/EDGetToken.h" +#include "FWCore/Utilities/interface/EDPutToken.h" +#include "FWCore/Utilities/interface/ESGetToken.h" +#include "FWCore/Utilities/interface/InputTag.h" +#include "FWCore/ParameterSet/interface/ParameterSet.h" +#include "DataFormats/Common/interface/Handle.h" + +#include "DataFormats/L1TrackTrigger/interface/TTTypes.h" +#include "L1Trigger/TrackTrigger/interface/Setup.h" +#include "L1Trigger/TrackerTFP/interface/DataFormats.h" +#include "L1Trigger/TrackerTFP/interface/DuplicateRemoval.h" + +#include +#include + +using namespace std; +using namespace edm; +using namespace tt; + +namespace trackerTFP { + + /*! \class trackerTFP::ProducerDR + * \brief L1TrackTrigger duplicate removal emulator + * \author Thomas Schuh + * \date 2023, Feb + */ + class ProducerDR : public stream::EDProducer<> { + public: + explicit ProducerDR(const ParameterSet&); + ~ProducerDR() override {} + + private: + void beginRun(const Run&, const EventSetup&) override; + void produce(Event&, const EventSetup&) override; + void endStream() override {} + // ED input token of sf stubs and tracks + EDGetTokenT edGetTokenStubs_; + EDGetTokenT edGetTokenTracks_; + // ED output token for accepted stubs and tracks + EDPutTokenT edPutTokenStubs_; + EDPutTokenT edPutTokenTracks_; + // Setup token + ESGetToken esGetTokenSetup_; + // DataFormats token + ESGetToken esGetTokenDataFormats_; + // configuration + ParameterSet iConfig_; + // helper class to store configurations + const Setup* setup_ = nullptr; + // helper class to extract structured data from tt::Frames + const DataFormats* dataFormats_ = nullptr; + }; + + ProducerDR::ProducerDR(const ParameterSet& iConfig) : iConfig_(iConfig) { + const string& label = iConfig.getParameter("InputLabelDR"); + const string& branchStubs = iConfig.getParameter("BranchStubs"); + const string& branchTracks = iConfig.getParameter("BranchTracks"); + // book in- and output ED products + edGetTokenStubs_ = consumes(InputTag(label, branchStubs)); + edGetTokenTracks_ = consumes(InputTag(label, branchTracks)); + edPutTokenStubs_ = produces(branchStubs); + edPutTokenTracks_ = produces(branchTracks); + // book ES products + esGetTokenSetup_ = esConsumes(); + esGetTokenDataFormats_ = esConsumes(); + } + + void ProducerDR::beginRun(const Run& iRun, const EventSetup& iSetup) { + // helper class to store configurations + setup_ = &iSetup.getData(esGetTokenSetup_); + // helper class to extract structured data from tt::Frames + dataFormats_ = &iSetup.getData(esGetTokenDataFormats_); + } + + void ProducerDR::produce(Event& iEvent, const EventSetup& iSetup) { + static const int numChannelIn = dataFormats_->numChannel(Process::kf); + static const int numChannelOut = dataFormats_->numChannel(Process::dr); + static const int numRegions = setup_->numRegions(); + static const int numLayers = setup_->numLayers(); + // empty DR products + StreamsStub acceptedStubs(numRegions * numChannelOut * numLayers); + StreamsTrack acceptedTracks(numRegions * numChannelOut); + // read in KF Product and produce DR product + Handle handleStubs; + iEvent.getByToken(edGetTokenStubs_, handleStubs); + const StreamsStub& allStubs = *handleStubs; + Handle handleTracks; + iEvent.getByToken(edGetTokenTracks_, handleTracks); + const StreamsTrack& allTracks = *handleTracks; + // helper + auto validFrameT = [](int& sum, const FrameTrack& frame) { return sum += (frame.first.isNonnull() ? 1 : 0); }; + auto validFrameS = [](int& sum, const FrameStub& frame) { return sum += (frame.first.isNonnull() ? 1 : 0); }; + auto putT = [](const vector& objects, StreamTrack& stream) { + auto toFrame = [](TrackDR* object) { return object ? object->frame() : FrameTrack(); }; + stream.reserve(objects.size()); + transform(objects.begin(), objects.end(), back_inserter(stream), toFrame); + }; + auto putS = [](const vector& objects, StreamStub& stream) { + auto toFrame = [](StubDR* object) { return object ? object->frame() : FrameStub(); }; + stream.reserve(objects.size()); + transform(objects.begin(), objects.end(), back_inserter(stream), toFrame); + }; + for (int region = 0; region < numRegions; region++) { + const int offsetIn = region * numChannelIn; + const int offsetOut = region * numChannelOut; + // count input objects + int nTracks(0); + int nStubs(0); + for (int channelIn = 0; channelIn < numChannelIn; channelIn++) { + const int index = offsetIn + channelIn; + const int offset = index * numLayers; + const StreamTrack& tracks = allTracks[index]; + nTracks += accumulate(tracks.begin(), tracks.end(), 0, validFrameT); + for (int layer = 0; layer < numLayers; layer++) { + const StreamStub& stubs = allStubs[offset + layer]; + nStubs += accumulate(stubs.begin(), stubs.end(), 0, validFrameS); + } + } + // storage of input data + vector tracksKF; + tracksKF.reserve(nTracks); + vector stubsKF; + stubsKF.reserve(nStubs); + // h/w liked organized pointer to input data + vector> regionTracks(numChannelIn); + vector> regionStubs(numChannelIn * numLayers); + // read input data + for (int channelIn = 0; channelIn < numChannelIn; channelIn++) { + const int index = offsetIn + channelIn; + const int offsetAll = index * numLayers; + const int offsetRegion = channelIn * numLayers; + const StreamTrack& streamTrack = allTracks[index]; + vector& tracks = regionTracks[channelIn]; + tracks.reserve(streamTrack.size()); + for (const FrameTrack& frame : streamTrack) { + TrackKF* track = nullptr; + if (frame.first.isNonnull()) { + tracksKF.emplace_back(frame, dataFormats_); + track = &tracksKF.back(); + } + tracks.push_back(track); + } + for (int layer = 0; layer < numLayers; layer++) { + for (const FrameStub& frame : allStubs[offsetAll + layer]) { + StubKF* stub = nullptr; + if (frame.first.isNonnull()) { + stubsKF.emplace_back(frame, dataFormats_); + stub = &stubsKF.back(); + } + regionStubs[offsetRegion + layer].push_back(stub); + } + } + } + // empty storage of output data + vector tracksDR; + tracksDR.reserve(nTracks); + vector stubsDR; + stubsDR.reserve(nStubs); + // object to remove duplicates in a processing region + DuplicateRemoval dr(iConfig_, setup_, dataFormats_, tracksDR, stubsDR); + // empty h/w liked organized pointer to output data + vector> streamsTrack(numChannelOut); + vector> streamsStub(numChannelOut * numLayers); + // fill output data + dr.produce(regionTracks, regionStubs, streamsTrack, streamsStub); + // convert data to ed products + for (int channelOut = 0; channelOut < numChannelOut; channelOut++) { + const int index = offsetOut + channelOut; + const int offsetRegion = channelOut * numLayers; + const int offsetAll = index * numLayers; + putT(streamsTrack[channelOut], acceptedTracks[index]); + for (int layer = 0; layer < numLayers; layer++) + putS(streamsStub[offsetRegion + layer], acceptedStubs[offsetAll + layer]); + } + } + // store products + iEvent.emplace(edPutTokenStubs_, move(acceptedStubs)); + iEvent.emplace(edPutTokenTracks_, move(acceptedTracks)); + } + +} // namespace trackerTFP + +DEFINE_FWK_MODULE(trackerTFP::ProducerDR); \ No newline at end of file diff --git a/L1Trigger/TrackerTFP/plugins/ProducerES.cc b/L1Trigger/TrackerTFP/plugins/ProducerDataFormats.cc similarity index 67% rename from L1Trigger/TrackerTFP/plugins/ProducerES.cc rename to L1Trigger/TrackerTFP/plugins/ProducerDataFormats.cc index 4e308c80851ad..2dbf3922af8f1 100644 --- a/L1Trigger/TrackerTFP/plugins/ProducerES.cc +++ b/L1Trigger/TrackerTFP/plugins/ProducerDataFormats.cc @@ -14,15 +14,15 @@ using namespace tt; namespace trackerTFP { - /*! \class trackerTFP::ProducerES + /*! \class trackerTFP::ProducerDataFormats * \brief Class to produce setup of Track Trigger emulator data formats * \author Thomas Schuh * \date 2020, June */ - class ProducerES : public ESProducer { + class ProducerDataFormats : public ESProducer { public: - ProducerES(const ParameterSet& iConfig); - ~ProducerES() override {} + ProducerDataFormats(const ParameterSet& iConfig); + ~ProducerDataFormats() override {} unique_ptr produce(const DataFormatsRcd& rcd); private: @@ -30,16 +30,16 @@ namespace trackerTFP { ESGetToken esGetToken_; }; - ProducerES::ProducerES(const ParameterSet& iConfig) : iConfig_(iConfig) { + ProducerDataFormats::ProducerDataFormats(const ParameterSet& iConfig) { auto cc = setWhatProduced(this); esGetToken_ = cc.consumes(); } - unique_ptr ProducerES::produce(const DataFormatsRcd& rcd) { + unique_ptr ProducerDataFormats::produce(const DataFormatsRcd& rcd) { const Setup* setup = &rcd.get(esGetToken_); - return make_unique(iConfig_, setup); + return make_unique(setup); } } // namespace trackerTFP -DEFINE_FWK_EVENTSETUP_MODULE(trackerTFP::ProducerES); +DEFINE_FWK_EVENTSETUP_MODULE(trackerTFP::ProducerDataFormats); diff --git a/L1Trigger/TrackerTFP/plugins/ProducerDemonstrator.cc b/L1Trigger/TrackerTFP/plugins/ProducerDemonstrator.cc index a735aa202d281..894cd93eca986 100644 --- a/L1Trigger/TrackerTFP/plugins/ProducerDemonstrator.cc +++ b/L1Trigger/TrackerTFP/plugins/ProducerDemonstrator.cc @@ -13,7 +13,8 @@ using namespace tt; namespace trackerTFP { /*! \class trackerTFP::ProducerDemonstrator - * \brief Class to demontrate correctness of track trigger emulators + * \brief ESProducer providing the algorithm to run input data through modelsim + * and to compares results with expected output data * \author Thomas Schuh * \date 2020, Nov */ diff --git a/L1Trigger/TrackerTFP/plugins/ProducerFormatsKF.cc b/L1Trigger/TrackerTFP/plugins/ProducerFormatsKF.cc deleted file mode 100644 index 569d0516aab4d..0000000000000 --- a/L1Trigger/TrackerTFP/plugins/ProducerFormatsKF.cc +++ /dev/null @@ -1,45 +0,0 @@ -#include "FWCore/Framework/interface/ESProducer.h" -#include "FWCore/Framework/interface/ESHandle.h" -#include "FWCore/ParameterSet/interface/ParameterSet.h" -#include "FWCore/Utilities/interface/ESGetToken.h" -#include "FWCore/Utilities/interface/ESInputTag.h" -#include "DataFormats/Provenance/interface/ParameterSetID.h" -#include "L1Trigger/TrackerTFP/interface/DataFormats.h" -#include "L1Trigger/TrackerTFP/interface/KalmanFilterFormats.h" - -#include - -using namespace std; -using namespace edm; - -namespace trackerTFP { - - /*! \class trackerTFP::ProducerFormatsKF - * \brief Class to produce setup of Kalman Filter emulator data formats - * \author Thomas Schuh - * \date 2020, July - */ - class ProducerFormatsKF : public ESProducer { - public: - ProducerFormatsKF(const ParameterSet& iConfig); - ~ProducerFormatsKF() override {} - unique_ptr produce(const KalmanFilterFormatsRcd& rcd); - - private: - const ParameterSet iConfig_; - ESGetToken esGetToken_; - }; - - ProducerFormatsKF::ProducerFormatsKF(const ParameterSet& iConfig) : iConfig_(iConfig) { - auto cc = setWhatProduced(this); - esGetToken_ = cc.consumes(); - } - - unique_ptr ProducerFormatsKF::produce(const KalmanFilterFormatsRcd& rcd) { - const DataFormats* dataFormats = &rcd.get(esGetToken_); - return make_unique(iConfig_, dataFormats); - } - -} // namespace trackerTFP - -DEFINE_FWK_EVENTSETUP_MODULE(trackerTFP::ProducerFormatsKF); diff --git a/L1Trigger/TrackerTFP/plugins/ProducerGP.cc b/L1Trigger/TrackerTFP/plugins/ProducerGP.cc index b9781b8186411..41c96b5a598c5 100644 --- a/L1Trigger/TrackerTFP/plugins/ProducerGP.cc +++ b/L1Trigger/TrackerTFP/plugins/ProducerGP.cc @@ -10,14 +10,18 @@ #include "FWCore/ParameterSet/interface/ParameterSet.h" #include "DataFormats/Common/interface/Handle.h" -#include "DataFormats/L1TrackTrigger/interface/TTDTC.h" #include "DataFormats/L1TrackTrigger/interface/TTTypes.h" #include "L1Trigger/TrackTrigger/interface/Setup.h" #include "L1Trigger/TrackerTFP/interface/DataFormats.h" +#include "L1Trigger/TrackerTFP/interface/LayerEncoding.h" #include "L1Trigger/TrackerTFP/interface/GeometricProcessor.h" -#include #include +#include +#include +#include +#include +#include using namespace std; using namespace edm; @@ -39,69 +43,113 @@ namespace trackerTFP { void beginRun(const Run&, const EventSetup&) override; void produce(Event&, const EventSetup&) override; virtual void endJob() {} - - // ED input token of DTC stubs - EDGetTokenT edGetToken_; - // ED output token for accepted stubs - EDPutTokenT edPutTokenAccepted_; - // ED output token for lost stubs - EDPutTokenT edPutTokenLost_; + // ED input token of pp objects + EDGetTokenT edGetToken_; + // ED output token for accepted objects + EDPutTokenT edPutToken_; // Setup token ESGetToken esGetTokenSetup_; // DataFormats token ESGetToken esGetTokenDataFormats_; + // LayerEncoding token + ESGetToken esGetTokenLayerEncoding_; // configuration ParameterSet iConfig_; // helper classe to store configurations const Setup* setup_ = nullptr; // helper class to extract structured data from tt::Frames const DataFormats* dataFormats_ = nullptr; + // helper class to encode layer + const LayerEncoding* layerEncoding_ = nullptr; }; ProducerGP::ProducerGP(const ParameterSet& iConfig) : iConfig_(iConfig) { - const string& label = iConfig.getParameter("LabelDTC"); - const string& branchAccepted = iConfig.getParameter("BranchAcceptedStubs"); - const string& branchLost = iConfig.getParameter("BranchLostStubs"); + const string& label = iConfig.getParameter("InputLabelGP"); + const string& branch = iConfig.getParameter("BranchStubs"); // book in- and output ED products - edGetToken_ = consumes(InputTag(label, branchAccepted)); - edPutTokenAccepted_ = produces(branchAccepted); - edPutTokenLost_ = produces(branchLost); + edGetToken_ = consumes(InputTag(label, branch)); + edPutToken_ = produces(branch); // book ES products esGetTokenSetup_ = esConsumes(); esGetTokenDataFormats_ = esConsumes(); + esGetTokenLayerEncoding_ = esConsumes(); } void ProducerGP::beginRun(const Run& iRun, const EventSetup& iSetup) { setup_ = &iSetup.getData(esGetTokenSetup_); - if (!setup_->configurationSupported()) - return; - // check process history if desired - if (iConfig_.getParameter("CheckHistory")) - setup_->checkHistory(iRun.processHistory()); dataFormats_ = &iSetup.getData(esGetTokenDataFormats_); + // helper class to encode layer + layerEncoding_ = &iSetup.getData(esGetTokenLayerEncoding_); } void ProducerGP::produce(Event& iEvent, const EventSetup& iSetup) { + static const int numChannelIn = dataFormats_->numChannel(Process::pp); + static const int numChannelOut = dataFormats_->numChannel(Process::gp); + static const int numRegions = setup_->numRegions(); // empty GP products - StreamsStub accepted(dataFormats_->numStreams(Process::gp)); - StreamsStub lost(dataFormats_->numStreams(Process::gp)); + StreamsStub accepted(numRegions * numChannelOut); // read in DTC Product and produce TFP product - if (setup_->configurationSupported()) { - Handle handle; - iEvent.getByToken(edGetToken_, handle); - const TTDTC& ttDTC = *handle.product(); - for (int region = 0; region < setup_->numRegions(); region++) { - // object to route Stubs of one region to one stream per sector - GeometricProcessor gp(iConfig_, setup_, dataFormats_, region); - // read in and organize input product - gp.consume(ttDTC); - // fill output products - gp.produce(accepted, lost); + Handle handle; + iEvent.getByToken(edGetToken_, handle); + const StreamsStub& streamsStub = *handle.product(); + // helper + auto validFrame = [](int& sum, const FrameStub& frame) { return sum += (frame.first.isNonnull() ? 1 : 0); }; + auto nSectors = [](int& sum, const StubPP& object) { + const int nPhiT = object.phiTMax() - object.phiTMin() + 1; + const int nZT = object.zTMax() - object.zTMin() + 1; + return sum += nPhiT * nZT; + }; + auto toFrame = [](StubGP* object) { return object ? object->frame() : FrameStub(); }; + // produce GP product per region + for (int region = 0; region < numRegions; region++) { + const int offsetIn = region * numChannelIn; + const int offsetOut = region * numChannelOut; + // count input objects + int nStubsPP(0); + for (int channelIn = 0; channelIn < numChannelIn; channelIn++) { + const StreamStub& stream = streamsStub[offsetIn + channelIn]; + nStubsPP += accumulate(stream.begin(), stream.end(), 0, validFrame); + } + // storage of input data + vector stubsPP; + stubsPP.reserve(nStubsPP); + // h/w liked organized pointer to input data + vector> streamsIn(numChannelIn); + // read input data + for (int channelIn = 0; channelIn < numChannelIn; channelIn++) { + const StreamStub& streamStub = streamsStub[offsetIn + channelIn]; + vector& stream = streamsIn[channelIn]; + stream.reserve(streamStub.size()); + for (const FrameStub& frame : streamStub) { + StubPP* stubPP = nullptr; + if (frame.first.isNonnull()) { + stubsPP.emplace_back(frame, dataFormats_); + stubPP = &stubsPP.back(); + } + stream.push_back(stubPP); + } + } + // predict upper limit of GP stubs + const int nStubsGP = accumulate(stubsPP.begin(), stubsPP.end(), 0, nSectors); + // container of GP stubs + vector stubsGP; + stubsGP.reserve(nStubsGP); + // object to route Stubs of one region to one stream per sector + GeometricProcessor gp(iConfig_, setup_, dataFormats_, layerEncoding_, stubsGP); + // empty h/w liked organized pointer to output data + vector> streamsOut(numChannelOut); + // fill output data + gp.produce(streamsIn, streamsOut); + // convert data to ed products + for (int channelOut = 0; channelOut < numChannelOut; channelOut++) { + const deque& objects = streamsOut[channelOut]; + StreamStub& stream = accepted[offsetOut + channelOut]; + stream.reserve(objects.size()); + transform(objects.begin(), objects.end(), back_inserter(stream), toFrame); } } // store products - iEvent.emplace(edPutTokenAccepted_, std::move(accepted)); - iEvent.emplace(edPutTokenLost_, std::move(lost)); + iEvent.emplace(edPutToken_, move(accepted)); } } // namespace trackerTFP diff --git a/L1Trigger/TrackerTFP/plugins/ProducerHT.cc b/L1Trigger/TrackerTFP/plugins/ProducerHT.cc index 15a9701b7ac14..c5089c103df72 100644 --- a/L1Trigger/TrackerTFP/plugins/ProducerHT.cc +++ b/L1Trigger/TrackerTFP/plugins/ProducerHT.cc @@ -13,10 +13,15 @@ #include "DataFormats/L1TrackTrigger/interface/TTTypes.h" #include "L1Trigger/TrackTrigger/interface/Setup.h" #include "L1Trigger/TrackerTFP/interface/DataFormats.h" +#include "L1Trigger/TrackerTFP/interface/LayerEncoding.h" #include "L1Trigger/TrackerTFP/interface/HoughTransform.h" #include +#include +#include #include +#include +#include using namespace std; using namespace edm; @@ -38,71 +43,107 @@ namespace trackerTFP { void beginRun(const Run&, const EventSetup&) override; void produce(Event&, const EventSetup&) override; virtual void endJob() {} - // ED input token of gp stubs EDGetTokenT edGetToken_; // ED output token for accepted stubs - EDPutTokenT edPutTokenAccepted_; - // ED output token for lost stubs - EDPutTokenT edPutTokenLost_; + EDPutTokenT edPutToken_; // Setup token ESGetToken esGetTokenSetup_; // DataFormats token ESGetToken esGetTokenDataFormats_; + // LayerEncoding token + ESGetToken esGetTokenLayerEncoding_; // configuration ParameterSet iConfig_; // helper class to store configurations const Setup* setup_ = nullptr; // helper class to extract structured data from tt::Frames const DataFormats* dataFormats_ = nullptr; + // + const LayerEncoding* layerEncoding_ = nullptr; }; ProducerHT::ProducerHT(const ParameterSet& iConfig) : iConfig_(iConfig) { - const string& label = iConfig.getParameter("LabelGP"); - const string& branchAccepted = iConfig.getParameter("BranchAcceptedStubs"); - const string& branchLost = iConfig.getParameter("BranchLostStubs"); + const string& label = iConfig.getParameter("InputLabelHT"); + const string& branch = iConfig.getParameter("BranchStubs"); // book in- and output ED products - edGetToken_ = consumes(InputTag(label, branchAccepted)); - edPutTokenAccepted_ = produces(branchAccepted); - edPutTokenLost_ = produces(branchLost); + edGetToken_ = consumes(InputTag(label, branch)); + edPutToken_ = produces(branch); // book ES products esGetTokenSetup_ = esConsumes(); esGetTokenDataFormats_ = esConsumes(); + esGetTokenLayerEncoding_ = esConsumes(); } void ProducerHT::beginRun(const Run& iRun, const EventSetup& iSetup) { // helper class to store configurations setup_ = &iSetup.getData(esGetTokenSetup_); - if (!setup_->configurationSupported()) - return; - // check process history if desired - if (iConfig_.getParameter("CheckHistory")) - setup_->checkHistory(iRun.processHistory()); // helper class to extract structured data from tt::Frames dataFormats_ = &iSetup.getData(esGetTokenDataFormats_); + // + layerEncoding_ = &iSetup.getData(esGetTokenLayerEncoding_); } void ProducerHT::produce(Event& iEvent, const EventSetup& iSetup) { + static const int numChannelIn = dataFormats_->numChannel(Process::gp); + static const int numChannelOut = dataFormats_->numChannel(Process::ht); + static const int numRegions = setup_->numRegions(); // empty HT products - StreamsStub accepted(dataFormats_->numStreams(Process::ht)); - StreamsStub lost(dataFormats_->numStreams(Process::ht)); + StreamsStub accepted(numRegions * numChannelOut); // read in DTC Product and produce TFP product - if (setup_->configurationSupported()) { - Handle handle; - iEvent.getByToken(edGetToken_, handle); - const StreamsStub& streams = *handle.product(); - for (int region = 0; region < setup_->numRegions(); region++) { - // object to find initial rough candidates in r-phi in a region - HoughTransform ht(iConfig_, setup_, dataFormats_, region); - // read in and organize input product - ht.consume(streams); - // fill output products - ht.produce(accepted, lost); + Handle handle; + iEvent.getByToken(edGetToken_, handle); + const StreamsStub& streamsStub = *handle.product(); + // helper + auto validFrame = [](int& sum, const FrameStub& frame) { return sum += (frame.first.isNonnull() ? 1 : 0); }; + auto toFrame = [](StubHT* object) { return object ? object->frame() : FrameStub(); }; + // produce HT output per region + for (int region = 0; region < numRegions; region++) { + const int offsetIn = region * numChannelIn; + const int offsetOut = region * numChannelOut; + // count input objects + int nStubsGP(0); + for (int channelIn = 0; channelIn < numChannelIn; channelIn++) { + const StreamStub& stream = streamsStub[offsetIn + channelIn]; + nStubsGP += accumulate(stream.begin(), stream.end(), 0, validFrame); + } + // storage of input data + vector stubsGP; + stubsGP.reserve(nStubsGP); + // h/w liked organized pointer to input data + vector> streamsIn(numChannelIn); + // read input data + for (int channelIn = 0; channelIn < numChannelIn; channelIn++) { + const StreamStub& streamStub = streamsStub[offsetIn + channelIn]; + vector& stream = streamsIn[channelIn]; + stream.reserve(streamStub.size()); + for (const FrameStub& frame : streamStub) { + StubGP* stub = nullptr; + if (frame.first.isNonnull()) { + stubsGP.emplace_back(frame, dataFormats_); + stub = &stubsGP.back(); + } + stream.push_back(stub); + } + } + // container for output stubs + vector stubsHT; + // object to find initial rough candidates in r-phi in a region + HoughTransform ht(iConfig_, setup_, dataFormats_, layerEncoding_, stubsHT); + // empty h/w liked organized pointer to output data + vector> streamsOut(numChannelOut); + // fill output data + ht.produce(streamsIn, streamsOut); + // convert data to ed products + for (int channelOut = 0; channelOut < numChannelOut; channelOut++) { + const deque& objects = streamsOut[channelOut]; + StreamStub& stream = accepted[offsetOut + channelOut]; + stream.reserve(objects.size()); + transform(objects.begin(), objects.end(), back_inserter(stream), toFrame); } } // store products - iEvent.emplace(edPutTokenAccepted_, std::move(accepted)); - iEvent.emplace(edPutTokenLost_, std::move(lost)); + iEvent.emplace(edPutToken_, move(accepted)); } } // namespace trackerTFP diff --git a/L1Trigger/TrackerTFP/plugins/ProducerKF.cc b/L1Trigger/TrackerTFP/plugins/ProducerKF.cc index ce123005215ae..c7d74f0b4ad16 100644 --- a/L1Trigger/TrackerTFP/plugins/ProducerKF.cc +++ b/L1Trigger/TrackerTFP/plugins/ProducerKF.cc @@ -18,6 +18,8 @@ #include "L1Trigger/TrackerTFP/interface/KalmanFilter.h" #include +#include +#include using namespace std; using namespace edm; @@ -36,111 +38,182 @@ namespace trackerTFP { ~ProducerKF() override {} private: + typedef State::Stub Stub; void beginRun(const Run&, const EventSetup&) override; void produce(Event&, const EventSetup&) override; void endStream() override { if (printDebug_) - kalmanFilterFormats_->endJob(); + kalmanFilterFormats_.endJob(); } - // ED input token of sf stubs and tracks EDGetTokenT edGetTokenStubs_; EDGetTokenT edGetTokenTracks_; // ED output token for accepted stubs and tracks - EDPutTokenT edPutTokenAcceptedStubs_; - EDPutTokenT edPutTokenAcceptedTracks_; - // ED output token for lost stubs and tracks - EDPutTokenT edPutTokenLostStubs_; - EDPutTokenT edPutTokenLostTracks_; + EDPutTokenT edPutTokenStubs_; + EDPutTokenT edPutTokenTracks_; // ED output token for number of accepted and lost States - EDPutTokenT edPutTokenNumAcceptedStates_; - EDPutTokenT edPutTokenNumLostStates_; + EDPutTokenT edPutTokenNumStatesAccepted_; + EDPutTokenT edPutTokenNumStatesTruncated_; + // ED output token for chi2s in r-phi and r-z plane + EDPutTokenT>> edPutTokenChi2s_; // Setup token ESGetToken esGetTokenSetup_; // DataFormats token ESGetToken esGetTokenDataFormats_; - // KalmanFilterFormats token - ESGetToken esGetTokenKalmanFilterFormats_; + // LayerEncoding token + ESGetToken esGetTokenLayerEncoding_; // configuration ParameterSet iConfig_; // helper class to store configurations const Setup* setup_ = nullptr; // helper class to extract structured data from tt::Frames const DataFormats* dataFormats_ = nullptr; - // helper class to - KalmanFilterFormats* kalmanFilterFormats_ = nullptr; + // helper class to encode layer + const LayerEncoding* layerEncoding_ = nullptr; + // helper class to tune internal kf variables + KalmanFilterFormats kalmanFilterFormats_; // print end job internal unused MSB bool printDebug_; }; - ProducerKF::ProducerKF(const ParameterSet& iConfig) : iConfig_(iConfig) { + ProducerKF::ProducerKF(const ParameterSet& iConfig) : iConfig_(iConfig), kalmanFilterFormats_(iConfig) { printDebug_ = iConfig.getParameter("PrintKFDebug"); - const string& label = iConfig.getParameter("LabelKFin"); - const string& branchAcceptedStubs = iConfig.getParameter("BranchAcceptedStubs"); - const string& branchAcceptedTracks = iConfig.getParameter("BranchAcceptedTracks"); - const string& branchLostStubs = iConfig.getParameter("BranchLostStubs"); - const string& branchLostTracks = iConfig.getParameter("BranchLostTracks"); + const string& label = iConfig.getParameter("InputLabelKF"); + const string& branchStubs = iConfig.getParameter("BranchStubs"); + const string& branchTracks = iConfig.getParameter("BranchTracks"); + const string& branchTruncated = iConfig.getParameter("BranchTruncated"); // book in- and output ED products - edGetTokenStubs_ = consumes(InputTag(label, branchAcceptedStubs)); - edGetTokenTracks_ = consumes(InputTag(label, branchAcceptedTracks)); - edPutTokenAcceptedStubs_ = produces(branchAcceptedStubs); - edPutTokenAcceptedTracks_ = produces(branchAcceptedTracks); - edPutTokenLostStubs_ = produces(branchLostStubs); - edPutTokenLostTracks_ = produces(branchLostTracks); - edPutTokenNumAcceptedStates_ = produces(branchAcceptedTracks); - edPutTokenNumLostStates_ = produces(branchLostTracks); + edGetTokenStubs_ = consumes(InputTag(label, branchStubs)); + edGetTokenTracks_ = consumes(InputTag(label, branchTracks)); + edPutTokenStubs_ = produces(branchStubs); + edPutTokenTracks_ = produces(branchTracks); + edPutTokenNumStatesAccepted_ = produces(branchTracks); + edPutTokenNumStatesTruncated_ = produces(branchTruncated); + edPutTokenChi2s_ = produces>>(branchTracks); // book ES products esGetTokenSetup_ = esConsumes(); esGetTokenDataFormats_ = esConsumes(); - esGetTokenKalmanFilterFormats_ = esConsumes(); + esGetTokenLayerEncoding_ = esConsumes(); } void ProducerKF::beginRun(const Run& iRun, const EventSetup& iSetup) { // helper class to store configurations setup_ = &iSetup.getData(esGetTokenSetup_); - if (!setup_->configurationSupported()) - return; - // check process history if desired - if (iConfig_.getParameter("CheckHistory")) - setup_->checkHistory(iRun.processHistory()); // helper class to extract structured data from tt::Frames dataFormats_ = &iSetup.getData(esGetTokenDataFormats_); - // helper class to - kalmanFilterFormats_ = const_cast(&iSetup.getData(esGetTokenKalmanFilterFormats_)); + // helper class to encode layer + layerEncoding_ = &iSetup.getData(esGetTokenLayerEncoding_); + // helper class to tune internal kf variables + kalmanFilterFormats_.beginRun(dataFormats_); } void ProducerKF::produce(Event& iEvent, const EventSetup& iSetup) { + static const int numChannel = dataFormats_->numChannel(Process::kf); + static const int numRegions = setup_->numRegions(); + static const int numLayers = setup_->numLayers(); // empty KF products - StreamsStub acceptedStubs(dataFormats_->numStreamsStubs(Process::kf)); - StreamsTrack acceptedTracks(dataFormats_->numStreamsTracks(Process::kf)); - StreamsStub lostStubs(dataFormats_->numStreamsStubs(Process::kf)); - StreamsTrack lostTracks(dataFormats_->numStreamsTracks(Process::kf)); - int numAcceptedStates(0); - int numLostStates(0); + StreamsStub acceptedStubs(numRegions * numChannel * numLayers); + StreamsTrack acceptedTracks(numRegions * numChannel); + int numStatesAccepted(0); + int numStatesTruncated(0); + deque> chi2s; // read in SF Product and produce KF product - if (setup_->configurationSupported()) { - Handle handleStubs; - iEvent.getByToken(edGetTokenStubs_, handleStubs); - const StreamsStub& stubs = *handleStubs; - Handle handleTracks; - iEvent.getByToken(edGetTokenTracks_, handleTracks); - const StreamsTrack& tracks = *handleTracks; - for (int region = 0; region < setup_->numRegions(); region++) { - // object to fit tracks in a processing region - KalmanFilter kf(iConfig_, setup_, dataFormats_, kalmanFilterFormats_, region); - // read in and organize input tracks and stubs - kf.consume(tracks, stubs); - // fill output products - kf.produce(acceptedStubs, acceptedTracks, lostStubs, lostTracks, numAcceptedStates, numLostStates); + Handle handleStubs; + iEvent.getByToken(edGetTokenStubs_, handleStubs); + const StreamsStub& allStubs = *handleStubs; + Handle handleTracks; + iEvent.getByToken(edGetTokenTracks_, handleTracks); + const StreamsTrack& allTracks = *handleTracks; + // helper + auto validFrameT = [](int& sum, const FrameTrack& frame) { return sum += (frame.first.isNonnull() ? 1 : 0); }; + auto validFrameS = [](int& sum, const FrameStub& frame) { return sum += (frame.first.isNonnull() ? 1 : 0); }; + auto putT = [](const vector& objects, StreamTrack& stream) { + auto toFrame = [](TrackKF* object) { return object ? object->frame() : FrameTrack(); }; + stream.reserve(objects.size()); + transform(objects.begin(), objects.end(), back_inserter(stream), toFrame); + }; + auto putS = [](const vector& objects, StreamStub& stream) { + auto toFrame = [](StubKF* object) { return object ? object->frame() : FrameStub(); }; + stream.reserve(objects.size()); + transform(objects.begin(), objects.end(), back_inserter(stream), toFrame); + }; + for (int region = 0; region < numRegions; region++) { + const int offset = region * numChannel; + // count input objects + int nTracks(0); + int nStubs(0); + for (int channel = 0; channel < numChannel; channel++) { + const int index = offset + channel; + const int offsetStubs = index * numLayers; + const StreamTrack& tracks = allTracks[index]; + nTracks += accumulate(tracks.begin(), tracks.end(), 0, validFrameT); + for (int layer = 0; layer < numLayers; layer++) { + const StreamStub& stubs = allStubs[offsetStubs + layer]; + nStubs += accumulate(stubs.begin(), stubs.end(), 0, validFrameS); + } + } + // storage of input data + vector tracksCTB; + tracksCTB.reserve(nTracks); + vector stubs; + stubs.reserve(nStubs); + // h/w liked organized pointer to input data + vector> regionTracks(numChannel); + vector> regionStubs(numChannel * numLayers); + // read input data + for (int channel = 0; channel < numChannel; channel++) { + const int index = offset + channel; + const int offsetAll = index * numLayers; + const int offsetRegion = channel * numLayers; + const StreamTrack& streamTrack = allTracks[index]; + vector& tracks = regionTracks[channel]; + tracks.reserve(streamTrack.size()); + for (const FrameTrack& frame : streamTrack) { + TrackCTB* track = nullptr; + if (frame.first.isNonnull()) { + tracksCTB.emplace_back(frame, dataFormats_); + track = &tracksCTB.back(); + } + tracks.push_back(track); + } + for (int layer = 0; layer < numLayers; layer++) { + for (const FrameStub& frame : allStubs[offsetAll + layer]) { + Stub* stub = nullptr; + if (frame.first.isNonnull()) { + stubs.emplace_back(&kalmanFilterFormats_, frame); + stub = &stubs.back(); + } + regionStubs[offsetRegion + layer].push_back(stub); + } + } + } + // empty storage of output data + vector tracksKF; + tracksKF.reserve(nTracks); + vector stubsKF; + stubsKF.reserve(nStubs); + // object to fit tracks in a processing region + KalmanFilter kf(iConfig_, setup_, dataFormats_, layerEncoding_, &kalmanFilterFormats_, tracksKF, stubsKF); + // empty h/w liked organized pointer to output data + vector> streamsTrack(numChannel); + vector>> streamsStub(numChannel, vector>(numLayers)); + // fill output products + kf.produce(regionTracks, regionStubs, streamsTrack, streamsStub, numStatesAccepted, numStatesTruncated, chi2s); + // convert data to ed products + for (int channel = 0; channel < numChannel; channel++) { + const int index = offset + channel; + const int offsetStubs = index * numLayers; + putT(streamsTrack[channel], acceptedTracks[index]); + for (int layer = 0; layer < numLayers; layer++) + putS(streamsStub[channel][layer], acceptedStubs[offsetStubs + layer]); } } // store products - iEvent.emplace(edPutTokenAcceptedStubs_, std::move(acceptedStubs)); - iEvent.emplace(edPutTokenAcceptedTracks_, std::move(acceptedTracks)); - iEvent.emplace(edPutTokenLostStubs_, std::move(lostStubs)); - iEvent.emplace(edPutTokenLostTracks_, std::move(lostTracks)); - iEvent.emplace(edPutTokenNumAcceptedStates_, numAcceptedStates); - iEvent.emplace(edPutTokenNumLostStates_, numLostStates); + iEvent.emplace(edPutTokenStubs_, move(acceptedStubs)); + iEvent.emplace(edPutTokenTracks_, move(acceptedTracks)); + iEvent.emplace(edPutTokenNumStatesAccepted_, numStatesAccepted); + iEvent.emplace(edPutTokenNumStatesTruncated_, numStatesTruncated); + iEvent.emplace(edPutTokenChi2s_, chi2s.begin(), chi2s.end()); } } // namespace trackerTFP diff --git a/L1Trigger/TrackerTFP/plugins/ProducerKFin.cc b/L1Trigger/TrackerTFP/plugins/ProducerKFin.cc deleted file mode 100644 index a3e539b3270b8..0000000000000 --- a/L1Trigger/TrackerTFP/plugins/ProducerKFin.cc +++ /dev/null @@ -1,225 +0,0 @@ -#include "FWCore/Framework/interface/stream/EDProducer.h" -#include "FWCore/Framework/interface/Run.h" -#include "FWCore/Framework/interface/EventSetup.h" -#include "FWCore/Framework/interface/Event.h" -#include "FWCore/Framework/interface/MakerMacros.h" -#include "FWCore/Utilities/interface/EDGetToken.h" -#include "FWCore/Utilities/interface/EDPutToken.h" -#include "FWCore/Utilities/interface/ESGetToken.h" -#include "FWCore/Utilities/interface/InputTag.h" -#include "FWCore/ParameterSet/interface/ParameterSet.h" -#include "DataFormats/Common/interface/Handle.h" - -#include "DataFormats/L1TrackTrigger/interface/TTTypes.h" -#include "L1Trigger/TrackTrigger/interface/Setup.h" -#include "L1Trigger/TrackerTFP/interface/DataFormats.h" -#include "L1Trigger/TrackerTFP/interface/LayerEncoding.h" - -#include -#include -#include -#include -#include -#include -#include - -using namespace std; -using namespace edm; -using namespace tt; - -namespace trackerTFP { - - /*! \class trackerTFP::ProducerKFin - * \brief transforms TTTracks into KF input - * \author Thomas Schuh - * \date 2020, July - */ - class ProducerKFin : public stream::EDProducer<> { - public: - explicit ProducerKFin(const ParameterSet&); - ~ProducerKFin() override {} - - private: - void beginRun(const Run&, const EventSetup&) override; - void produce(Event&, const EventSetup&) override; - virtual void endJob() {} - - // ED input token of TTTracks - EDGetTokenT>> edGetTokenTTTracks_; - // ED input token of Stubs - EDGetTokenT edGetTokenStubs_; - // ED output token for stubs - EDPutTokenT edPutTokenAcceptedStubs_; - EDPutTokenT edPutTokenLostStubs_; - // ED output token for tracks - EDPutTokenT edPutTokenAcceptedTracks_; - EDPutTokenT edPutTokenLostTracks_; - // Setup token - ESGetToken esGetTokenSetup_; - // DataFormats token - ESGetToken esGetTokenDataFormats_; - // LayerEncoding token - ESGetToken esGetTokenLayerEncoding_; - // configuration - ParameterSet iConfig_; - // helper class to store configurations - const Setup* setup_ = nullptr; - // helper class to extract structured data from tt::Frames - const DataFormats* dataFormats_ = nullptr; - // helper class to encode layer - const LayerEncoding* layerEncoding_ = nullptr; - // - bool enableTruncation_; - }; - - ProducerKFin::ProducerKFin(const ParameterSet& iConfig) : iConfig_(iConfig) { - const string& labelTTTracks = iConfig.getParameter("LabelZHTout"); - const string& labelStubs = iConfig.getParameter("LabelZHT"); - const string& branchAcceptedStubs = iConfig.getParameter("BranchAcceptedStubs"); - const string& branchAcceptedTracks = iConfig.getParameter("BranchAcceptedTracks"); - const string& branchLostStubs = iConfig.getParameter("BranchLostStubs"); - const string& branchLostTracks = iConfig.getParameter("BranchLostTracks"); - // book in- and output ED products - edGetTokenTTTracks_ = - consumes>>(InputTag(labelTTTracks, branchAcceptedTracks)); - edGetTokenStubs_ = consumes(InputTag(labelStubs, branchAcceptedStubs)); - edPutTokenAcceptedStubs_ = produces(branchAcceptedStubs); - edPutTokenAcceptedTracks_ = produces(branchAcceptedTracks); - edPutTokenLostStubs_ = produces(branchLostStubs); - edPutTokenLostTracks_ = produces(branchLostTracks); - // book ES products - esGetTokenSetup_ = esConsumes(); - esGetTokenDataFormats_ = esConsumes(); - esGetTokenLayerEncoding_ = esConsumes(); - // - enableTruncation_ = iConfig.getParameter("EnableTruncation"); - } - - void ProducerKFin::beginRun(const Run& iRun, const EventSetup& iSetup) { - // helper class to store configurations - setup_ = &iSetup.getData(esGetTokenSetup_); - if (!setup_->configurationSupported()) - return; - // check process history if desired - if (iConfig_.getParameter("CheckHistory")) - setup_->checkHistory(iRun.processHistory()); - // helper class to extract structured data from tt::Frames - dataFormats_ = &iSetup.getData(esGetTokenDataFormats_); - // helper class to encode layer - layerEncoding_ = &iSetup.getData(esGetTokenLayerEncoding_); - } - - void ProducerKFin::produce(Event& iEvent, const EventSetup& iSetup) { - const DataFormat& dfcot = dataFormats_->format(Variable::cot, Process::kfin); - const DataFormat& dfzT = dataFormats_->format(Variable::zT, Process::kfin); - const DataFormat& dfinv2R = dataFormats_->format(Variable::inv2R, Process::kfin); - const DataFormat& dfdPhi = dataFormats_->format(Variable::dPhi, Process::kfin); - const DataFormat& dfdZ = dataFormats_->format(Variable::dZ, Process::kfin); - // empty KFin products - StreamsStub streamAcceptedStubs(dataFormats_->numStreamsStubs(Process::kf)); - StreamsTrack streamAcceptedTracks(dataFormats_->numStreamsTracks(Process::kf)); - StreamsStub streamLostStubs(dataFormats_->numStreamsStubs(Process::kf)); - StreamsTrack streamLostTracks(dataFormats_->numStreamsTracks(Process::kf)); - // read in SFout Product and produce KFin product - if (setup_->configurationSupported()) { - Handle handleStubs; - iEvent.getByToken(edGetTokenStubs_, handleStubs); - const StreamsStub& streams = *handleStubs.product(); - Handle>> handleTTTracks; - iEvent.getByToken>>(edGetTokenTTTracks_, handleTTTracks); - const vector>& ttTracks = *handleTTTracks.product(); - for (int region = 0; region < setup_->numRegions(); region++) { - // Unpack input SF data into vector - int nStubsZHR(0); - for (int channel = 0; channel < dataFormats_->numChannel(Process::zht); channel++) { - const int index = region * dataFormats_->numChannel(Process::zht) + channel; - const StreamStub& stream = streams[index]; - nStubsZHR += accumulate(stream.begin(), stream.end(), 0, [](int sum, const FrameStub& frame) { - return sum + (frame.first.isNonnull() ? 1 : 0); - }); - } - vector stubsZHT; - stubsZHT.reserve(nStubsZHR); - for (int channel = 0; channel < dataFormats_->numChannel(Process::zht); channel++) { - const int index = region * dataFormats_->numChannel(Process::zht) + channel; - for (const FrameStub& frame : streams[index]) - if (frame.first.isNonnull()) - stubsZHT.emplace_back(frame, dataFormats_); - } - vector> dequesStubs(dataFormats_->numChannel(Process::kf) * setup_->numLayers()); - vector> dequesTracks(dataFormats_->numChannel(Process::kf)); - int i(0); - for (const TTTrack& ttTrack : ttTracks) { - if ((int)ttTrack.phiSector() / setup_->numSectorsPhi() != region) { - i++; - continue; - } - const int sectorPhi = ttTrack.phiSector() % setup_->numSectorsPhi(); - deque& tracks = dequesTracks[sectorPhi]; - const int binEta = ttTrack.etaSector(); - const int binZT = dfzT.toUnsigned(dfzT.integer(ttTrack.z0())); - const int binCot = dfcot.toUnsigned(dfcot.integer(ttTrack.tanL())); - StubZHT* stubZHT = nullptr; - vector layerCounts(setup_->numLayers(), 0); - for (const TTStubRef& ttStubRef : ttTrack.getStubRefs()) { - const int layerId = setup_->layerId(ttStubRef); - const int layerIdKF = layerEncoding_->layerIdKF(binEta, binZT, binCot, layerId); - if (layerIdKF == -1) - continue; - if (layerCounts[layerIdKF] == setup_->zhtMaxStubsPerLayer()) - continue; - layerCounts[layerIdKF]++; - deque& stubs = dequesStubs[sectorPhi * setup_->numLayers() + layerIdKF]; - auto identical = [ttStubRef, ttTrack](const StubZHT& stub) { - return (int)ttTrack.trackSeedType() == stub.trackId() && ttStubRef == stub.ttStubRef(); - }; - stubZHT = &*find_if(stubsZHT.begin(), stubsZHT.end(), identical); - const double inv2R = dfinv2R.floating(stubZHT->inv2R()); - const double cot = dfcot.floating(stubZHT->cot()) + setup_->sectorCot(binEta); - const double dPhi = dfdPhi.digi(setup_->dPhi(ttStubRef, inv2R)); - const double dZ = dfdZ.digi(setup_->dZ(ttStubRef, cot)); - stubs.emplace_back(StubKFin(*stubZHT, dPhi, dZ, layerIdKF).frame()); - } - const int size = *max_element(layerCounts.begin(), layerCounts.end()); - int layerIdKF(0); - for (int layerCount : layerCounts) { - deque& stubs = dequesStubs[sectorPhi * setup_->numLayers() + layerIdKF++]; - const int nGaps = size - layerCount; - stubs.insert(stubs.end(), nGaps, FrameStub()); - } - const TTBV& maybePattern = layerEncoding_->maybePattern(binEta, binZT, binCot); - const TrackKFin track(*stubZHT, TTTrackRef(handleTTTracks, i++), maybePattern); - tracks.emplace_back(track.frame()); - const int nGaps = size - 1; - tracks.insert(tracks.end(), nGaps, FrameTrack()); - } - // transform deques to vectors & emulate truncation - for (int channel = 0; channel < dataFormats_->numChannel(Process::kf); channel++) { - const int index = region * dataFormats_->numChannel(Process::kf) + channel; - deque& tracks = dequesTracks[channel]; - auto limitTracks = next(tracks.begin(), min(setup_->numFrames(), (int)tracks.size())); - if (!enableTruncation_) - limitTracks = tracks.end(); - streamAcceptedTracks[index] = StreamTrack(tracks.begin(), limitTracks); - streamLostTracks[index] = StreamTrack(limitTracks, tracks.end()); - for (int l = 0; l < setup_->numLayers(); l++) { - deque& stubs = dequesStubs[channel * setup_->numLayers() + l]; - auto limitStubs = next(stubs.begin(), min(setup_->numFrames(), (int)stubs.size())); - if (!enableTruncation_) - limitStubs = stubs.end(); - streamAcceptedStubs[index * setup_->numLayers() + l] = StreamStub(stubs.begin(), limitStubs); - streamLostStubs[index * setup_->numLayers() + l] = StreamStub(limitStubs, stubs.end()); - } - } - } - } - // store products - iEvent.emplace(edPutTokenAcceptedStubs_, std::move(streamAcceptedStubs)); - iEvent.emplace(edPutTokenAcceptedTracks_, std::move(streamAcceptedTracks)); - iEvent.emplace(edPutTokenLostStubs_, std::move(streamLostStubs)); - iEvent.emplace(edPutTokenLostTracks_, std::move(streamLostTracks)); - } - -} // namespace trackerTFP - -DEFINE_FWK_MODULE(trackerTFP::ProducerKFin); diff --git a/L1Trigger/TrackerTFP/plugins/ProducerMHT.cc b/L1Trigger/TrackerTFP/plugins/ProducerMHT.cc deleted file mode 100644 index 7ca6a74f8d108..0000000000000 --- a/L1Trigger/TrackerTFP/plugins/ProducerMHT.cc +++ /dev/null @@ -1,110 +0,0 @@ -#include "FWCore/Framework/interface/stream/EDProducer.h" -#include "FWCore/Framework/interface/Run.h" -#include "FWCore/Framework/interface/EventSetup.h" -#include "FWCore/Framework/interface/Event.h" -#include "FWCore/Framework/interface/MakerMacros.h" -#include "FWCore/Utilities/interface/EDGetToken.h" -#include "FWCore/Utilities/interface/EDPutToken.h" -#include "FWCore/Utilities/interface/ESGetToken.h" -#include "FWCore/Utilities/interface/InputTag.h" -#include "FWCore/ParameterSet/interface/ParameterSet.h" -#include "DataFormats/Common/interface/Handle.h" - -#include "DataFormats/L1TrackTrigger/interface/TTTypes.h" -#include "L1Trigger/TrackTrigger/interface/Setup.h" -#include "L1Trigger/TrackerTFP/interface/DataFormats.h" -#include "L1Trigger/TrackerTFP/interface/MiniHoughTransform.h" - -#include -#include - -using namespace std; -using namespace edm; -using namespace tt; - -namespace trackerTFP { - - /*! \class trackerTFP::ProducerMHT - * \brief L1TrackTrigger Mini Hough Transform emulator - * \author Thomas Schuh - * \date 2020, May - */ - class ProducerMHT : public stream::EDProducer<> { - public: - explicit ProducerMHT(const ParameterSet&); - ~ProducerMHT() override {} - - private: - void beginRun(const Run&, const EventSetup&) override; - void produce(Event&, const EventSetup&) override; - virtual void endJob() {} - - // ED input token of gp stubs - EDGetTokenT edGetToken_; - // ED output token for accepted stubs - EDPutTokenT edPutTokenAccepted_; - // ED output token for lost stubs - EDPutTokenT edPutTokenLost_; - // Setup token - ESGetToken esGetTokenSetup_; - // DataFormats token - ESGetToken esGetTokenDataFormats_; - // configuration - ParameterSet iConfig_; - // helper class to store configurations - const Setup* setup_ = nullptr; - // helper class to extract structured data from tt::Frames - const DataFormats* dataFormats_ = nullptr; - }; - - ProducerMHT::ProducerMHT(const ParameterSet& iConfig) : iConfig_(iConfig) { - const string& label = iConfig.getParameter("LabelHT"); - const string& branchAccepted = iConfig.getParameter("BranchAcceptedStubs"); - const string& branchLost = iConfig.getParameter("BranchLostStubs"); - // book in- and output ED products - edGetToken_ = consumes(InputTag(label, branchAccepted)); - edPutTokenAccepted_ = produces(branchAccepted); - edPutTokenLost_ = produces(branchLost); - // book ES products - esGetTokenSetup_ = esConsumes(); - esGetTokenDataFormats_ = esConsumes(); - } - - void ProducerMHT::beginRun(const Run& iRun, const EventSetup& iSetup) { - // helper class to store configurations - setup_ = &iSetup.getData(esGetTokenSetup_); - if (!setup_->configurationSupported()) - return; - // check process history if desired - if (iConfig_.getParameter("CheckHistory")) - setup_->checkHistory(iRun.processHistory()); - // helper class to extract structured data from tt::Frames - dataFormats_ = &iSetup.getData(esGetTokenDataFormats_); - } - - void ProducerMHT::produce(Event& iEvent, const EventSetup& iSetup) { - // empty MHT products - StreamsStub accepted(dataFormats_->numStreams(Process::mht)); - StreamsStub lost(dataFormats_->numStreams(Process::mht)); - // read in HT Product and produce MHT product - if (setup_->configurationSupported()) { - Handle handle; - iEvent.getByToken(edGetToken_, handle); - const StreamsStub& streams = *handle.product(); - for (int region = 0; region < setup_->numRegions(); region++) { - // object to find in a region finer rough candidates in r-phi - MiniHoughTransform mht(iConfig_, setup_, dataFormats_, region); - // read in and organize input product - mht.consume(streams); - // fill output products - mht.produce(accepted, lost); - } - } - // store products - iEvent.emplace(edPutTokenAccepted_, std::move(accepted)); - iEvent.emplace(edPutTokenLost_, std::move(lost)); - } - -} // namespace trackerTFP - -DEFINE_FWK_MODULE(trackerTFP::ProducerMHT); diff --git a/L1Trigger/TrackerTFP/plugins/ProducerPP.cc b/L1Trigger/TrackerTFP/plugins/ProducerPP.cc new file mode 100644 index 0000000000000..374b6f7430de7 --- /dev/null +++ b/L1Trigger/TrackerTFP/plugins/ProducerPP.cc @@ -0,0 +1,81 @@ +#include "FWCore/Framework/interface/stream/EDProducer.h" +#include "FWCore/Framework/interface/Run.h" +#include "FWCore/Framework/interface/EventSetup.h" +#include "FWCore/Framework/interface/Event.h" +#include "FWCore/Framework/interface/MakerMacros.h" +#include "FWCore/Utilities/interface/EDGetToken.h" +#include "FWCore/Utilities/interface/EDPutToken.h" +#include "FWCore/Utilities/interface/ESGetToken.h" +#include "FWCore/Utilities/interface/InputTag.h" +#include "FWCore/ParameterSet/interface/ParameterSet.h" +#include "DataFormats/Common/interface/Handle.h" + +#include "DataFormats/L1TrackTrigger/interface/TTDTC.h" +#include "DataFormats/L1TrackTrigger/interface/TTTypes.h" +#include "L1Trigger/TrackTrigger/interface/Setup.h" + +#include + +using namespace std; +using namespace edm; +using namespace tt; + +namespace trackerTFP { + + /*! \class trackerTFP::ProducerPP + * \brief L1TrackTrigger PatchPanel between DTC and TFP emulator + * \author Thomas Schuh + * \date 2023, April + */ + class ProducerPP : public stream::EDProducer<> { + public: + explicit ProducerPP(const ParameterSet&); + ~ProducerPP() override {} + + private: + void beginRun(const Run&, const EventSetup&) override; + void produce(Event&, const EventSetup&) override; + virtual void endJob() {} + // ED input token of DTC stubs + EDGetTokenT edGetToken_; + // ED output token for accepted stubs + EDPutTokenT edPutToken_; + // Setup token + ESGetToken esGetTokenSetup_; + // configuration + ParameterSet iConfig_; + // helper classe to store configurations + const Setup* setup_ = nullptr; + }; + + ProducerPP::ProducerPP(const ParameterSet& iConfig) : iConfig_(iConfig) { + const string& label = iConfig.getParameter("InputLabelPP"); + const string& branch = iConfig.getParameter("BranchStubs"); + // book in- and output ED products + edGetToken_ = consumes(InputTag(label, branch)); + edPutToken_ = produces(branch); + // book ES products + esGetTokenSetup_ = esConsumes(); + } + + void ProducerPP::beginRun(const Run& iRun, const EventSetup& iSetup) { setup_ = &iSetup.getData(esGetTokenSetup_); } + + void ProducerPP::produce(Event& iEvent, const EventSetup& iSetup) { + // empty GP products + StreamsStub stubs(setup_->numRegions() * setup_->numDTCsPerTFP()); + // read in DTC Product and produce TFP product + Handle handle; + iEvent.getByToken(edGetToken_, handle); + const TTDTC& ttDTC = *handle.product(); + for (int region = 0; region < setup_->numRegions(); region++) { + const int offset = region * setup_->numDTCsPerTFP(); + for (int channel = 0; channel < setup_->numDTCsPerTFP(); channel++) + stubs[offset + channel] = ttDTC.stream(region, channel); + } + // store products + iEvent.emplace(edPutToken_, move(stubs)); + } + +} // namespace trackerTFP + +DEFINE_FWK_MODULE(trackerTFP::ProducerPP); \ No newline at end of file diff --git a/L1Trigger/TrackerTFP/plugins/ProducerTFP.cc b/L1Trigger/TrackerTFP/plugins/ProducerTFP.cc new file mode 100644 index 0000000000000..7351fa40af5b1 --- /dev/null +++ b/L1Trigger/TrackerTFP/plugins/ProducerTFP.cc @@ -0,0 +1,120 @@ +#include "FWCore/Framework/interface/stream/EDProducer.h" +#include "FWCore/Framework/interface/Run.h" +#include "FWCore/Framework/interface/EventSetup.h" +#include "FWCore/Framework/interface/Event.h" +#include "FWCore/Framework/interface/MakerMacros.h" +#include "FWCore/Utilities/interface/EDGetToken.h" +#include "FWCore/Utilities/interface/EDPutToken.h" +#include "FWCore/Utilities/interface/ESGetToken.h" +#include "FWCore/Utilities/interface/InputTag.h" +#include "FWCore/ParameterSet/interface/ParameterSet.h" +#include "DataFormats/Common/interface/Handle.h" +#include "DataFormats/Common/interface/OrphanHandle.h" + +#include "DataFormats/L1TrackTrigger/interface/TTTypes.h" +#include "L1Trigger/TrackTrigger/interface/Setup.h" +#include "L1Trigger/TrackerTFP/interface/DataFormats.h" +#include "L1Trigger/TrackerTFP/interface/TrackQuality.h" +#include "L1Trigger/TrackerTFP/interface/TrackFindingProcessor.h" + +#include +#include +#include +#include + +using namespace std; +using namespace edm; +using namespace tt; + +namespace trackerTFP { + + /*! \class trackerTFP::ProducerTFP + * \brief L1TrackTrigger final TFP output formatter + * \author Thomas Schuh + * \date 2023, June + */ + class ProducerTFP : public stream::EDProducer<> { + public: + explicit ProducerTFP(const ParameterSet&); + ~ProducerTFP() override {} + + private: + void beginRun(const Run&, const EventSetup&) override; + void produce(Event&, const EventSetup&) override; + void endStream() override {} + // ED input token of stubs and tracks + EDGetTokenT edGetTokenTracks_; + EDGetTokenT edGetTokenStubs_; + // ED output token for accepted stubs and tracks + EDPutTokenT edPutTokenTTTracks_; + EDPutTokenT edPutTokenTracks_; + // Setup token + ESGetToken esGetTokenSetup_; + // DataFormats token + ESGetToken esGetTokenDataFormats_; + // TrackQuality token + ESGetToken esGetTokenTrackQuality_; + // configuration + ParameterSet iConfig_; + // helper class to store configurations + const Setup* setup_ = nullptr; + // helper class to extract structured data from tt::Frames + const DataFormats* dataFormats_ = nullptr; + // helper class to determine track quality + const TrackQuality* trackQuality_ = nullptr; + }; + + ProducerTFP::ProducerTFP(const ParameterSet& iConfig) : iConfig_(iConfig) { + const string& labelTracks = iConfig.getParameter("InputLabelTFP"); + const string& labelStubs = iConfig.getParameter("InputLabelTQ"); + const string& branchTracks = iConfig.getParameter("BranchTracks"); + const string& branchTTTracks = iConfig.getParameter("BranchTTTracks"); + const string& branchStubs = iConfig.getParameter("BranchStubs"); + // book in- and output ED products + edGetTokenTracks_ = consumes(InputTag(labelTracks, branchTracks)); + edGetTokenStubs_ = consumes(InputTag(labelStubs, branchStubs)); + edPutTokenTTTracks_ = produces(branchTTTracks); + edPutTokenTracks_ = produces(branchTracks); + // book ES products + esGetTokenSetup_ = esConsumes(); + esGetTokenDataFormats_ = esConsumes(); + esGetTokenTrackQuality_ = esConsumes(); + } + + void ProducerTFP::beginRun(const Run& iRun, const EventSetup& iSetup) { + // helper class to store configurations + setup_ = &iSetup.getData(esGetTokenSetup_); + // helper class to extract structured data from tt::Frames + dataFormats_ = &iSetup.getData(esGetTokenDataFormats_); + // helper class to determine track quality + trackQuality_ = &iSetup.getData(esGetTokenTrackQuality_); + } + + void ProducerTFP::produce(Event& iEvent, const EventSetup& iSetup) { + // empty TFP products + TTTracks ttTracks; + StreamsTrack streamsTrack(setup_->numRegions() * setup_->tfpNumChannel()); + // read in TQ Products + Handle handleTracks; + iEvent.getByToken(edGetTokenTracks_, handleTracks); + Handle handleStubs; + iEvent.getByToken(edGetTokenStubs_, handleStubs); + // produce TTTracks + TrackFindingProcessor tfp(iConfig_, setup_, dataFormats_, trackQuality_); + tfp.produce(*handleTracks, *handleStubs, ttTracks, streamsTrack); + // put TTTRacks and produce TTTRackRefs + const int nTrks = ttTracks.size(); + const OrphanHandle oh = iEvent.emplace(edPutTokenTTTracks_, move(ttTracks)); + vector ttTrackRefs; + ttTrackRefs.reserve(nTrks); + for (int iTrk = 0; iTrk < nTrks; iTrk++) + ttTrackRefs.emplace_back(oh, iTrk); + // replace old TTTrackRefs in streamsTrack with new TTTrackRefs + tfp.produce(ttTrackRefs, streamsTrack); + // put StreamsTrack + iEvent.emplace(edPutTokenTracks_, move(streamsTrack)); + } + +} // namespace trackerTFP + +DEFINE_FWK_MODULE(trackerTFP::ProducerTFP); \ No newline at end of file diff --git a/L1Trigger/TrackerTFP/plugins/ProducerTQ.cc b/L1Trigger/TrackerTFP/plugins/ProducerTQ.cc new file mode 100644 index 0000000000000..6f6059ca100cd --- /dev/null +++ b/L1Trigger/TrackerTFP/plugins/ProducerTQ.cc @@ -0,0 +1,140 @@ +#include "FWCore/Framework/interface/stream/EDProducer.h" +#include "FWCore/Framework/interface/Run.h" +#include "FWCore/Framework/interface/EventSetup.h" +#include "FWCore/Framework/interface/Event.h" +#include "FWCore/Framework/interface/MakerMacros.h" +#include "FWCore/Utilities/interface/EDGetToken.h" +#include "FWCore/Utilities/interface/EDPutToken.h" +#include "FWCore/Utilities/interface/ESGetToken.h" +#include "FWCore/Utilities/interface/InputTag.h" +#include "FWCore/ParameterSet/interface/ParameterSet.h" +#include "DataFormats/Common/interface/Handle.h" + +#include "L1Trigger/TrackTrigger/interface/Setup.h" +#include "L1Trigger/TrackerTFP/interface/DataFormats.h" +#include "L1Trigger/TrackerTFP/interface/TrackQuality.h" + +#include +#include + +using namespace std; +using namespace edm; +using namespace tt; +using namespace trackerTFP; + +namespace trackerTFP { + + /*! \class trackerTFP::ProducerTQ + * \brief Bit accurate emulation of the track quality BDT + * \author Thomas Schuh + * \date 2024, Aug + */ + class ProducerTQ : public stream::EDProducer<> { + public: + explicit ProducerTQ(const ParameterSet&); + ~ProducerTQ() override {} + void beginRun(const Run&, const EventSetup&) override; + void produce(Event&, const EventSetup&) override; + void endJob() {} + + private: + typedef TrackQuality::Track Track; + // ED input token of kf stubs + EDGetTokenT edGetTokenStubs_; + // ED input token of kf tracks + EDGetTokenT edGetTokenTracks_; + // ED output token for accepted kfout tracks + EDPutTokenT edPutTokenTracks_; + // Setup token + ESGetToken esGetTokenSetup_; + // DataFormats token + ESGetToken esGetTokenDataFormats_; + // TrackQuality token + ESGetToken esGetTokenTrackQuality_; + // helper class to store configurations + const Setup* setup_ = nullptr; + // helper class to extract structured data from tt::Frames + const DataFormats* dataFormats_ = nullptr; + // helper class to determine Track Quality + const TrackQuality* trackQuality_ = nullptr; + }; + + ProducerTQ::ProducerTQ(const ParameterSet& iConfig) { + const string& label = iConfig.getParameter("InputLabelTQ"); + const string& branchStubs = iConfig.getParameter("BranchStubs"); + const string& branchTracks = iConfig.getParameter("BranchTracks"); + // book in- and output ED products + edGetTokenStubs_ = consumes(InputTag(label, branchStubs)); + edGetTokenTracks_ = consumes(InputTag(label, branchTracks)); + edPutTokenTracks_ = produces(branchTracks); + // book ES products + esGetTokenSetup_ = esConsumes(); + esGetTokenDataFormats_ = esConsumes(); + esGetTokenTrackQuality_ = esConsumes(); + } + + void ProducerTQ::beginRun(const Run& iRun, const EventSetup& iSetup) { + // helper class to store configurations + setup_ = &iSetup.getData(esGetTokenSetup_); + // helper class to extract structured data from tt::Frames + dataFormats_ = &iSetup.getData(esGetTokenDataFormats_); + // helper class to determine Track Quality + trackQuality_ = &iSetup.getData(esGetTokenTrackQuality_); + } + + void ProducerTQ::produce(Event& iEvent, const EventSetup& iSetup) { + static const int numRegions = setup_->numRegions(); + static const int numLayers = setup_->numLayers(); + static const int numChannel = setup_->tqNumChannel(); + auto valid = [](int sum, const FrameTrack& frame) { return sum += (frame.first.isNull() ? 0 : 1); }; + // empty TQ product + StreamsTrack output(numRegions * numChannel); + // read in KF Product and produce TQ product + Handle handleStubs; + iEvent.getByToken(edGetTokenStubs_, handleStubs); + const StreamsStub& streamsStubs = *handleStubs.product(); + Handle handleTracks; + iEvent.getByToken(edGetTokenTracks_, handleTracks); + const StreamsTrack& streamsTracks = *handleTracks.product(); + for (int region = 0; region < numRegions; region++) { + // calculate track quality + const int offsetLayer = region * numLayers; + const StreamTrack& streamTrack = streamsTracks[region]; + const int nTracks = accumulate(streamTrack.begin(), streamTrack.end(), 0, valid); + vector tracks; + tracks.reserve(nTracks); + vector stream; + stream.reserve(streamTrack.size()); + for (int frame = 0; frame < (int)streamTrack.size(); frame++) { + const FrameTrack& frameTrack = streamTrack[frame]; + if (frameTrack.first.isNull()) { + stream.push_back(nullptr); + continue; + } + StreamStub streamStub; + streamStub.reserve(numLayers); + for (int layer = 0; layer < numLayers; layer++) + streamStub.push_back(streamsStubs[offsetLayer + layer][frame]); + tracks.emplace_back(frameTrack, streamStub, trackQuality_); + stream.push_back(&tracks.back()); + } + // fill TQ product + const int offsetChannel = region * numChannel; + for (int channel = 0; channel < numChannel; channel++) + output[offsetChannel + channel].reserve(stream.size()); + for (Track* track : stream) { + if (!track) { + for (int channel = 0; channel < numChannel; channel++) + output[offsetChannel + channel].emplace_back(FrameTrack()); + continue; + } + for (int channel = 0; channel < numChannel; channel++) + output[offsetChannel + channel].emplace_back(track->frame(channel)); + } + } + // store TQ product + iEvent.emplace(edPutTokenTracks_, std::move(output)); + } +} // namespace trackerTFP + +DEFINE_FWK_MODULE(trackerTFP::ProducerTQ); diff --git a/L1Trigger/TrackerTFP/plugins/ProducerTT.cc b/L1Trigger/TrackerTFP/plugins/ProducerTT.cc deleted file mode 100644 index dbb278cf9b068..0000000000000 --- a/L1Trigger/TrackerTFP/plugins/ProducerTT.cc +++ /dev/null @@ -1,124 +0,0 @@ -#include "FWCore/Framework/interface/stream/EDProducer.h" -#include "FWCore/Framework/interface/Run.h" -#include "FWCore/Framework/interface/EventSetup.h" -#include "FWCore/Framework/interface/Event.h" -#include "FWCore/Framework/interface/MakerMacros.h" -#include "FWCore/Utilities/interface/EDGetToken.h" -#include "FWCore/Utilities/interface/EDPutToken.h" -#include "FWCore/Utilities/interface/ESGetToken.h" -#include "FWCore/Utilities/interface/InputTag.h" -#include "FWCore/ParameterSet/interface/ParameterSet.h" -#include "DataFormats/Common/interface/Handle.h" - -#include "DataFormats/L1TrackTrigger/interface/TTTypes.h" -#include "L1Trigger/TrackTrigger/interface/Setup.h" -#include "L1Trigger/TrackerTFP/interface/DataFormats.h" - -#include -#include - -using namespace std; -using namespace edm; -using namespace tt; - -namespace trackerTFP { - - /*! \class trackerTFP::ProducerTT - * \brief Converts KF output into TTTracks - * \author Thomas Schuh - * \date 2020, Oct - */ - class ProducerTT : public stream::EDProducer<> { - public: - explicit ProducerTT(const ParameterSet&); - ~ProducerTT() override {} - - private: - void beginRun(const Run&, const EventSetup&) override; - void produce(Event&, const EventSetup&) override; - void endJob() {} - - // ED input token of kf stubs - EDGetTokenT edGetTokenStubs_; - // ED input token of kf tracks - EDGetTokenT edGetTokenTracks_; - // ED output token for TTTracks - EDPutTokenT edPutToken_; - // Setup token - ESGetToken esGetTokenSetup_; - // DataFormats token - ESGetToken esGetTokenDataFormats_; - // configuration - ParameterSet iConfig_; - // helper class to store configurations - const Setup* setup_ = nullptr; - // helper class to extract structured data from tt::Frames - const DataFormats* dataFormats_ = nullptr; - }; - - ProducerTT::ProducerTT(const ParameterSet& iConfig) : iConfig_(iConfig) { - const string& label = iConfig.getParameter("LabelKF"); - const string& branchStubs = iConfig.getParameter("BranchAcceptedStubs"); - const string& branchTracks = iConfig.getParameter("BranchAcceptedTracks"); - // book in- and output ED products - edGetTokenStubs_ = consumes(InputTag(label, branchStubs)); - edGetTokenTracks_ = consumes(InputTag(label, branchTracks)); - edPutToken_ = produces(branchTracks); - // book ES products - esGetTokenSetup_ = esConsumes(); - esGetTokenDataFormats_ = esConsumes(); - } - - void ProducerTT::beginRun(const Run& iRun, const EventSetup& iSetup) { - // helper class to store configurations - setup_ = &iSetup.getData(esGetTokenSetup_); - if (!setup_->configurationSupported()) - return; - // check process history if desired - if (iConfig_.getParameter("CheckHistory")) - setup_->checkHistory(iRun.processHistory()); - // helper class to extract structured data from tt::Frames - dataFormats_ = &iSetup.getData(esGetTokenDataFormats_); - } - - void ProducerTT::produce(Event& iEvent, const EventSetup& iSetup) { - // empty KFTTTrack product - TTTracks ttTracks; - // read in KF Product and produce KFTTTrack product - if (setup_->configurationSupported()) { - Handle handleTracks; - iEvent.getByToken(edGetTokenTracks_, handleTracks); - const StreamsTrack& streamsTracks = *handleTracks.product(); - Handle handleStubs; - iEvent.getByToken(edGetTokenTracks_, handleStubs); - const StreamsStub& streamsStubs = *handleStubs.product(); - int nTracks(0); - for (const StreamTrack& stream : streamsTracks) - nTracks += accumulate(stream.begin(), stream.end(), 0, [](int sum, const FrameTrack& frame) { - return sum + (frame.first.isNonnull() ? 1 : 0); - }); - ttTracks.reserve(nTracks); - for (int channel = 0; channel < dataFormats_->numStreamsTracks(Process::kf); channel++) { - int iTrk(0); - const int offset = channel * setup_->numLayers(); - for (const FrameTrack& frameTrack : streamsTracks[channel]) { - vector stubs; - stubs.reserve(setup_->numLayers()); - for (int layer = 0; layer < setup_->numLayers(); layer++) { - const FrameStub& frameStub = streamsStubs[offset + layer][iTrk]; - if (frameStub.first.isNonnull()) - stubs.emplace_back(frameStub, dataFormats_, layer); - } - TrackKF track(frameTrack, dataFormats_); - ttTracks.emplace_back(track.ttTrack(stubs)); - iTrk++; - } - } - } - // store products - iEvent.emplace(edPutToken_, std::move(ttTracks)); - } - -} // namespace trackerTFP - -DEFINE_FWK_MODULE(trackerTFP::ProducerTT); diff --git a/L1Trigger/TrackerTFP/plugins/ProducerTrackQuality.cc b/L1Trigger/TrackerTFP/plugins/ProducerTrackQuality.cc new file mode 100644 index 0000000000000..048e97a01d5df --- /dev/null +++ b/L1Trigger/TrackerTFP/plugins/ProducerTrackQuality.cc @@ -0,0 +1,39 @@ +#include "FWCore/Framework/interface/ESProducer.h" +#include "FWCore/Framework/interface/ModuleFactory.h" +#include "FWCore/Framework/interface/ESHandle.h" +#include "FWCore/ParameterSet/interface/ParameterSet.h" +#include "FWCore/Utilities/interface/ESGetToken.h" +#include "L1Trigger/TrackerTFP/interface/TrackQuality.h" + +#include + +using namespace std; +using namespace edm; + +namespace trackerTFP { + + /*! \class trackerTFP::ProducerTrackQuality + * \brief Class to produce TrackQuality of Track Trigger emulators + * \author Thomas Schuh + * \date 2024, July + */ + class ProducerTrackQuality : public ESProducer { + public: + ProducerTrackQuality(const ParameterSet& iConfig) : iConfig_(iConfig) { + auto cc = setWhatProduced(this); + esGetToken_ = cc.consumes(); + } + ~ProducerTrackQuality() override {} + unique_ptr produce(const TrackQualityRcd& trackQualityRcd) { + const DataFormats* dataFormats = &trackQualityRcd.get(esGetToken_); + return make_unique(iConfig_, dataFormats); + } + + private: + const ParameterSet iConfig_; + ESGetToken esGetToken_; + }; + +} // namespace trackerTFP + +DEFINE_FWK_EVENTSETUP_MODULE(trackerTFP::ProducerTrackQuality); diff --git a/L1Trigger/TrackerTFP/plugins/ProducerZHT.cc b/L1Trigger/TrackerTFP/plugins/ProducerZHT.cc deleted file mode 100644 index 4a07b157ea3a0..0000000000000 --- a/L1Trigger/TrackerTFP/plugins/ProducerZHT.cc +++ /dev/null @@ -1,110 +0,0 @@ -#include "FWCore/Framework/interface/stream/EDProducer.h" -#include "FWCore/Framework/interface/Run.h" -#include "FWCore/Framework/interface/EventSetup.h" -#include "FWCore/Framework/interface/Event.h" -#include "FWCore/Framework/interface/MakerMacros.h" -#include "FWCore/Utilities/interface/EDGetToken.h" -#include "FWCore/Utilities/interface/EDPutToken.h" -#include "FWCore/Utilities/interface/ESGetToken.h" -#include "FWCore/Utilities/interface/InputTag.h" -#include "FWCore/ParameterSet/interface/ParameterSet.h" -#include "DataFormats/Common/interface/Handle.h" - -#include "DataFormats/L1TrackTrigger/interface/TTTypes.h" -#include "L1Trigger/TrackTrigger/interface/Setup.h" -#include "L1Trigger/TrackerTFP/interface/DataFormats.h" -#include "L1Trigger/TrackerTFP/interface/ZHoughTransform.h" - -#include -#include - -using namespace std; -using namespace edm; -using namespace tt; - -namespace trackerTFP { - - /*! \class trackerTFP::ProducerZHT - * \brief L1TrackTrigger r-z Hough Transform emulator - * \author Thomas Schuh - * \date 2021, May - */ - class ProducerZHT : public stream::EDProducer<> { - public: - explicit ProducerZHT(const ParameterSet&); - ~ProducerZHT() override {} - - private: - void beginRun(const Run&, const EventSetup&) override; - void produce(Event&, const EventSetup&) override; - virtual void endJob() {} - - // ED input token of gp stubs - EDGetTokenT edGetToken_; - // ED output token for accepted stubs - EDPutTokenT edPutTokenAccepted_; - // ED output token for lost stubs - EDPutTokenT edPutTokenLost_; - // Setup token - ESGetToken esGetTokenSetup_; - // DataFormats token - ESGetToken esGetTokenDataFormats_; - // configuration - ParameterSet iConfig_; - // helper class to store configurations - const Setup* setup_ = nullptr; - // helper class to extract structured data from tt::Frames - const DataFormats* dataFormats_ = nullptr; - }; - - ProducerZHT::ProducerZHT(const ParameterSet& iConfig) : iConfig_(iConfig) { - const string& label = iConfig.getParameter("LabelMHT"); - const string& branchAccepted = iConfig.getParameter("BranchAcceptedStubs"); - const string& branchLost = iConfig.getParameter("BranchLostStubs"); - // book in- and output ED products - edGetToken_ = consumes(InputTag(label, branchAccepted)); - edPutTokenAccepted_ = produces(branchAccepted); - edPutTokenLost_ = produces(branchLost); - // book ES products - esGetTokenSetup_ = esConsumes(); - esGetTokenDataFormats_ = esConsumes(); - } - - void ProducerZHT::beginRun(const Run& iRun, const EventSetup& iSetup) { - // helper class to store configurations - setup_ = &iSetup.getData(esGetTokenSetup_); - if (!setup_->configurationSupported()) - return; - // check process history if desired - if (iConfig_.getParameter("CheckHistory")) - setup_->checkHistory(iRun.processHistory()); - // helper class to extract structured data from tt::Frames - dataFormats_ = &iSetup.getData(esGetTokenDataFormats_); - } - - void ProducerZHT::produce(Event& iEvent, const EventSetup& iSetup) { - // empty MHT products - StreamsStub accepted(dataFormats_->numStreams(Process::mht)); - StreamsStub lost(dataFormats_->numStreams(Process::mht)); - // read in HT Product and produce MHT product - if (setup_->configurationSupported()) { - Handle handle; - iEvent.getByToken(edGetToken_, handle); - const StreamsStub& streams = *handle.product(); - for (int region = 0; region < setup_->numRegions(); region++) { - // object to find in a region finer rough candidates in r-phi - ZHoughTransform zht(iConfig_, setup_, dataFormats_, region); - // read in and organize input product - zht.consume(streams); - // fill output products - zht.produce(accepted, lost); - } - } - // store products - iEvent.emplace(edPutTokenAccepted_, std::move(accepted)); - iEvent.emplace(edPutTokenLost_, std::move(lost)); - } - -} // namespace trackerTFP - -DEFINE_FWK_MODULE(trackerTFP::ProducerZHT); diff --git a/L1Trigger/TrackerTFP/plugins/ProducerZHTout.cc b/L1Trigger/TrackerTFP/plugins/ProducerZHTout.cc deleted file mode 100644 index 6abbba845d7fd..0000000000000 --- a/L1Trigger/TrackerTFP/plugins/ProducerZHTout.cc +++ /dev/null @@ -1,135 +0,0 @@ -#include "FWCore/Framework/interface/stream/EDProducer.h" -#include "FWCore/Framework/interface/Run.h" -#include "FWCore/Framework/interface/EventSetup.h" -#include "FWCore/Framework/interface/Event.h" -#include "FWCore/Framework/interface/MakerMacros.h" -#include "FWCore/Utilities/interface/EDGetToken.h" -#include "FWCore/Utilities/interface/EDPutToken.h" -#include "FWCore/Utilities/interface/ESGetToken.h" -#include "FWCore/Utilities/interface/InputTag.h" -#include "FWCore/ParameterSet/interface/ParameterSet.h" -#include "DataFormats/Common/interface/Handle.h" - -#include "DataFormats/L1TrackTrigger/interface/TTTypes.h" -#include "L1Trigger/TrackTrigger/interface/Setup.h" -#include "L1Trigger/TrackerTFP/interface/DataFormats.h" - -#include -#include -#include - -using namespace std; -using namespace edm; -using namespace tt; - -namespace trackerTFP { - - /*! \class trackerTFP::ProducerZHTout - * \brief transforms SF output into TTTracks - * \author Thomas Schuh - * \date 2020, July - */ - class ProducerZHTout : public stream::EDProducer<> { - public: - explicit ProducerZHTout(const ParameterSet&); - ~ProducerZHTout() override {} - - private: - void beginRun(const Run&, const EventSetup&) override; - void produce(Event&, const EventSetup&) override; - virtual void endJob() {} - - // ED input token of sf stubs - EDGetTokenT edGetToken_; - // ED output token of TTTracks - EDPutTokenT>> edPutToken_; - // Setup token - ESGetToken esGetTokenSetup_; - // DataFormats token - ESGetToken esGetTokenDataFormats_; - // configuration - ParameterSet iConfig_; - // helper class to store configurations - const Setup* setup_ = nullptr; - // helper class to extract structured data from tt::Frames - const DataFormats* dataFormats_ = nullptr; - }; - - ProducerZHTout::ProducerZHTout(const ParameterSet& iConfig) : iConfig_(iConfig) { - const string& label = iConfig.getParameter("LabelZHT"); - const string& branchAcceptedStubs = iConfig.getParameter("BranchAcceptedStubs"); - const string& branchAcceptedTracks = iConfig.getParameter("BranchAcceptedTracks"); - // book in- and output ED products - edGetToken_ = consumes(InputTag(label, branchAcceptedStubs)); - edPutToken_ = produces>>(branchAcceptedTracks); - // book ES products - esGetTokenSetup_ = esConsumes(); - esGetTokenDataFormats_ = esConsumes(); - } - - void ProducerZHTout::beginRun(const Run& iRun, const EventSetup& iSetup) { - // helper class to store configurations - setup_ = &iSetup.getData(esGetTokenSetup_); - if (!setup_->configurationSupported()) - return; - // check process history if desired - if (iConfig_.getParameter("CheckHistory")) - setup_->checkHistory(iRun.processHistory()); - // helper class to extract structured data from tt::Frames - dataFormats_ = &iSetup.getData(esGetTokenDataFormats_); - } - - void ProducerZHTout::produce(Event& iEvent, const EventSetup& iSetup) { - const DataFormat& dfCot = dataFormats_->format(Variable::cot, Process::zht); - const DataFormat& dfZT = dataFormats_->format(Variable::zT, Process::zht); - const DataFormat& dfPhiT = dataFormats_->format(Variable::phiT, Process::zht); - const DataFormat& dfinv2R = dataFormats_->format(Variable::inv2R, Process::zht); - // empty SFout product - deque> ttTracks; - // read in SF Product and produce SFout product - if (setup_->configurationSupported()) { - Handle handle; - iEvent.getByToken(edGetToken_, handle); - const StreamsStub& streams = *handle.product(); - for (int region = 0; region < setup_->numRegions(); region++) { - for (int channel = 0; channel < dataFormats_->numChannel(Process::zht); channel++) { - const int index = region * dataFormats_->numChannel(Process::zht) + channel; - // convert stream to stubs - const StreamStub& stream = streams[index]; - vector stubs; - stubs.reserve(stream.size()); - for (const FrameStub& frame : stream) - if (frame.first.isNonnull()) - stubs.emplace_back(frame, dataFormats_); - // form tracks - int i(0); - for (auto it = stubs.begin(); it != stubs.end();) { - const auto start = it; - const int id = it->trackId(); - auto different = [id](const StubZHT& stub) { return id != stub.trackId(); }; - it = find_if(it, stubs.end(), different); - vector ttStubRefs; - ttStubRefs.reserve(distance(start, it)); - transform(start, it, back_inserter(ttStubRefs), [](const StubZHT& stub) { return stub.ttStubRef(); }); - const double zT = dfZT.floating(start->zT()); - const double cot = dfCot.floating(start->cot()); - const double phiT = dfPhiT.floating(start->phiT()); - const double inv2R = dfinv2R.floating(start->inv2R()); - ttTracks.emplace_back(inv2R, phiT, cot, zT, 0., 0., 0., 0., 0., 0, 0, 0.); - ttTracks.back().setStubRefs(ttStubRefs); - ttTracks.back().setPhiSector(start->sectorPhi() + region * setup_->numSectorsPhi()); - ttTracks.back().setEtaSector(start->sectorEta()); - ttTracks.back().setTrackSeedType(start->trackId()); - if (i++ == setup_->zhtMaxTracks()) - break; - } - } - } - } - // store product - iEvent.emplace(edPutToken_, ttTracks.begin(), ttTracks.end()); - } - -} // namespace trackerTFP - -DEFINE_FWK_MODULE(trackerTFP::ProducerZHTout); diff --git a/L1Trigger/TrackerTFP/python/Analyzer_cff.py b/L1Trigger/TrackerTFP/python/Analyzer_cff.py index 8ecacab35aede..bf8fb9c10facd 100644 --- a/L1Trigger/TrackerTFP/python/Analyzer_cff.py +++ b/L1Trigger/TrackerTFP/python/Analyzer_cff.py @@ -1,12 +1,12 @@ +# EDAnalyzer for Track Trigger emulation steps + import FWCore.ParameterSet.Config as cms from L1Trigger.TrackerTFP.Analyzer_cfi import TrackerTFPAnalyzer_params from L1Trigger.TrackerTFP.Producer_cfi import TrackerTFPProducer_params -TrackerTFPAnalyzerGP = cms.EDAnalyzer( 'trackerTFP::AnalyzerGP', TrackerTFPAnalyzer_params, TrackerTFPProducer_params ) -TrackerTFPAnalyzerHT = cms.EDAnalyzer( 'trackerTFP::AnalyzerHT', TrackerTFPAnalyzer_params, TrackerTFPProducer_params ) -TrackerTFPAnalyzerMHT = cms.EDAnalyzer( 'trackerTFP::AnalyzerMHT', TrackerTFPAnalyzer_params, TrackerTFPProducer_params ) -TrackerTFPAnalyzerZHT = cms.EDAnalyzer( 'trackerTFP::AnalyzerZHT', TrackerTFPAnalyzer_params, TrackerTFPProducer_params ) -TrackerTFPAnalyzerKFin = cms.EDAnalyzer( 'trackerTFP::AnalyzerKFin', TrackerTFPAnalyzer_params, TrackerTFPProducer_params ) -TrackerTFPAnalyzerKF = cms.EDAnalyzer( 'trackerTFP::AnalyzerKF', TrackerTFPAnalyzer_params, TrackerTFPProducer_params ) -TrackerTFPAnalyzerTT = cms.EDAnalyzer( 'trackerTFP::AnalyzerTT', TrackerTFPAnalyzer_params, TrackerTFPProducer_params ) +AnalyzerGP = cms.EDAnalyzer( 'trackerTFP::AnalyzerGP' , TrackerTFPAnalyzer_params, TrackerTFPProducer_params ) +AnalyzerHT = cms.EDAnalyzer( 'trackerTFP::AnalyzerHT' , TrackerTFPAnalyzer_params, TrackerTFPProducer_params ) +AnalyzerCTB = cms.EDAnalyzer( 'trackerTFP::AnalyzerCTB', TrackerTFPAnalyzer_params, TrackerTFPProducer_params ) +AnalyzerKF = cms.EDAnalyzer( 'trackerTFP::AnalyzerKF' , TrackerTFPAnalyzer_params, TrackerTFPProducer_params ) +AnalyzerDR = cms.EDAnalyzer( 'trackerTFP::AnalyzerDR' , TrackerTFPAnalyzer_params, TrackerTFPProducer_params ) diff --git a/L1Trigger/TrackerTFP/python/Analyzer_cfi.py b/L1Trigger/TrackerTFP/python/Analyzer_cfi.py index cbcc20fd7b0e8..8b4e89f7fe5fa 100644 --- a/L1Trigger/TrackerTFP/python/Analyzer_cfi.py +++ b/L1Trigger/TrackerTFP/python/Analyzer_cfi.py @@ -1,3 +1,5 @@ +# configuration for Track Trigger emulation EDAnalyzer + import FWCore.ParameterSet.Config as cms TrackerTFPAnalyzer_params = cms.PSet ( @@ -5,6 +7,10 @@ UseMCTruth = cms.bool( True ), # enables analyze of TPs InputTagReconstructable = cms.InputTag("StubAssociator", "Reconstructable"), # InputTagSelection = cms.InputTag("StubAssociator", "UseForAlgEff"), # - + OutputLabelGP = cms.string( "ProducerGP" ), # + OutputLabelHT = cms.string( "ProducerHT" ), # + OutputLabelCTB = cms.string( "ProducerCTB" ), # + OutputLabelKF = cms.string( "ProducerKF" ), # + OutputLabelDR = cms.string( "ProducerDR" ), # ) diff --git a/L1Trigger/TrackerTFP/python/Customize_cff.py b/L1Trigger/TrackerTFP/python/Customize_cff.py index 9baa4073771f7..ba543b4b0c36f 100644 --- a/L1Trigger/TrackerTFP/python/Customize_cff.py +++ b/L1Trigger/TrackerTFP/python/Customize_cff.py @@ -1,7 +1,27 @@ +# function to alter TrackTriggerSetup to provide TMTT configuration + import FWCore.ParameterSet.Config as cms def setupUseTMTT(process): - process.TrackTriggerDataFormats.UseHybrid = False - process.TrackTriggerSetup.TrackingParticle.MinPt = 3.0 - process.TrackTriggerSetup.Firmware.MaxdPhi = 0.01 - return process + process.TrackTriggerSetup.TMTT.WidthR = 11 + process.TrackTriggerSetup.TMTT.WidthPhi = 14 + process.TrackTriggerSetup.TMTT.WidthZ = 13 + process.TrackTriggerSetup.TrackFinding.MinPt = 3.0 + process.TrackTriggerSetup.TrackFinding.MaxEta = 2.4 + process.TrackTriggerSetup.TrackFinding.ChosenRofPhi = 67.24 + process.TrackTriggerSetup.TrackFinding.NumLayers = 8 + process.TrackTriggerSetup.GeometricProcessor.ChosenRofZ = 57.76 + process.TrackTriggerSetup.HoughTransform.MinLayers = 5 + process.TrackTriggerSetup.CleanTrackBuilder.MaxStubs = 4 + process.TrackTriggerSetup.KalmanFilter.NumWorker = 4 + process.TrackTriggerSetup.KalmanFilter.MaxLayers = 8 + process.TrackTriggerSetup.KalmanFilter.MaxSeedingLayer = 3 + process.TrackTriggerSetup.KalmanFilter.MaxGaps = 2 + process.TrackTriggerSetup.KalmanFilter.ShiftChi20 = 0 + process.TrackTriggerSetup.KalmanFilter.ShiftChi21 = 0 + process.TrackTriggerSetup.KalmanFilter.CutChi2 = 2.0 + +def simUseTMTT(process): + process.StubAssociator.MinPt = 3.0 + + return process diff --git a/L1Trigger/TrackerTFP/python/DataFormats_cff.py b/L1Trigger/TrackerTFP/python/DataFormats_cff.py new file mode 100644 index 0000000000000..095cdd6771103 --- /dev/null +++ b/L1Trigger/TrackerTFP/python/DataFormats_cff.py @@ -0,0 +1,5 @@ +# ESProducer to provide and calculate and provide dataformats used by Track Trigger emulator + +import FWCore.ParameterSet.Config as cms + +TrackTriggerDataFormats = cms.ESProducer("trackerTFP::ProducerDataFormats") \ No newline at end of file diff --git a/L1Trigger/TrackerTFP/python/Demonstrator_cff.py b/L1Trigger/TrackerTFP/python/Demonstrator_cff.py index dd963873dad1d..f70ac8eab4dae 100644 --- a/L1Trigger/TrackerTFP/python/Demonstrator_cff.py +++ b/L1Trigger/TrackerTFP/python/Demonstrator_cff.py @@ -1,3 +1,6 @@ +# ESProducer providing the algorithm to run input data through modelsim and to compares results with expected output data +# and EDAnalyzer running the ESProduct produced by above ESProducer + import FWCore.ParameterSet.Config as cms from L1Trigger.TrackerTFP.Demonstrator_cfi import TrackTriggerDemonstrator_params diff --git a/L1Trigger/TrackerTFP/python/Demonstrator_cfi.py b/L1Trigger/TrackerTFP/python/Demonstrator_cfi.py index 720d2bbe1a640..7fc458b9cd7f9 100644 --- a/L1Trigger/TrackerTFP/python/Demonstrator_cfi.py +++ b/L1Trigger/TrackerTFP/python/Demonstrator_cfi.py @@ -1,11 +1,17 @@ -# configuration of Demonstrator. +# configuration of TrackTriggerDemonstrator. import FWCore.ParameterSet.Config as cms +# these parameters a for ModelSim runs of FW TrackTriggerDemonstrator_params = cms.PSet ( - LabelIn = cms.string( "TrackerTFPProducerKFin" ), # - LabelOut = cms.string( "TrackerTFPProducerKF" ), # - DirIPBB = cms.string( "/heplnw039/tschuh/work/proj/kf/" ), # path to ipbb proj area - RunTime = cms.double( 6.0 ) # runtime in us + LabelIn = cms.string( "ProducerCTB" ), # + LabelOut = cms.string( "ProducerKF" ), # + DirIPBB = cms.string( "/heplnw039/tschuh/work/proj/ctbkf/" ), # path to ipbb proj area + RunTime = cms.double( 6.50 ), # runtime in us + + #LinkMappingIn = cms.vint32( 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18, 19, 24, 25, 26, 27, 28, 29, 30, 31, 32, 33, 34, 35, 36, 37, 38, 39, 40, 41, 42, 43, 44, 45, 46, 47, 48, 49, 50, 51 ), + #LinkMappingOut = cms.vint32( 52, 53, 54, 55, 56, 57, 58, 59, 60 ) + LinkMappingIn = cms.vint32(), + LinkMappingOut = cms.vint32() ) diff --git a/L1Trigger/TrackerTFP/python/KalmanFilterFormats_cff.py b/L1Trigger/TrackerTFP/python/KalmanFilterFormats_cff.py deleted file mode 100644 index 3e0ac34332a90..0000000000000 --- a/L1Trigger/TrackerTFP/python/KalmanFilterFormats_cff.py +++ /dev/null @@ -1,5 +0,0 @@ -import FWCore.ParameterSet.Config as cms - -from L1Trigger.TrackerTFP.KalmanFilterFormats_cfi import TrackTriggerKalmanFilterFormats_params - -TrackTriggerKalmanFilterFormats = cms.ESProducer("trackerTFP::ProducerFormatsKF", TrackTriggerKalmanFilterFormats_params) diff --git a/L1Trigger/TrackerTFP/python/KalmanFilterFormats_cfi.py b/L1Trigger/TrackerTFP/python/KalmanFilterFormats_cfi.py index 4927e9605122c..fa695a4a0cdb2 100644 --- a/L1Trigger/TrackerTFP/python/KalmanFilterFormats_cfi.py +++ b/L1Trigger/TrackerTFP/python/KalmanFilterFormats_cfi.py @@ -6,70 +6,63 @@ TrackTriggerKalmanFilterFormats_params = cms.PSet ( - tmtt = cms.PSet ( - BaseShiftx0 = cms.int32( -3 ), - BaseShiftx1 = cms.int32( -9 ), - BaseShiftx2 = cms.int32( 0 ), - BaseShiftx3 = cms.int32( -2 ), - BaseShiftv0 = cms.int32( -6 ), - BaseShiftv1 = cms.int32( 8 ), - BaseShiftr0 = cms.int32( -8 ), - BaseShiftr1 = cms.int32( 0 ), - BaseShiftS00 = cms.int32( -2 ), - BaseShiftS01 = cms.int32( -7 ), - BaseShiftS12 = cms.int32( 3 ), - BaseShiftS13 = cms.int32( 1 ), - BaseShiftK00 = cms.int32( -16 ), - BaseShiftK10 = cms.int32( -22 ), - BaseShiftK21 = cms.int32( -22 ), - BaseShiftK31 = cms.int32( -23 ), - BaseShiftR00 = cms.int32( -5 ), - BaseShiftR11 = cms.int32( 7 ), - BaseShiftInvR00Approx = cms.int32( -26 ), - BaseShiftInvR11Approx = cms.int32( -38 ), - BaseShiftInvR00Cor = cms.int32( -15 ), - BaseShiftInvR11Cor = cms.int32( -15 ), - BaseShiftInvR00 = cms.int32( -24 ), - BaseShiftInvR11 = cms.int32( -33 ), - BaseShiftC00 = cms.int32( 5 ), - BaseShiftC01 = cms.int32( -3 ), - BaseShiftC11 = cms.int32( -7 ), - BaseShiftC22 = cms.int32( 3 ), - BaseShiftC23 = cms.int32( 0 ), - BaseShiftC33 = cms.int32( 1 ) - ), + EnableIntegerEmulation = cms.bool( True ), - hybrid = cms.PSet ( - BaseShiftx0 = cms.int32( -4 ), - BaseShiftx1 = cms.int32( -10 ), - BaseShiftx2 = cms.int32( -2 ), - BaseShiftx3 = cms.int32( -3 ), - BaseShiftv0 = cms.int32( -4 ), - BaseShiftv1 = cms.int32( 8 ), - BaseShiftr0 = cms.int32( -9 ), - BaseShiftr1 = cms.int32( -1 ), - BaseShiftS00 = cms.int32( 0 ), - BaseShiftS01 = cms.int32( -7 ), - BaseShiftS12 = cms.int32( 3 ), - BaseShiftS13 = cms.int32( 1 ), - BaseShiftK00 = cms.int32( -16 ), - BaseShiftK10 = cms.int32( -22 ), - BaseShiftK21 = cms.int32( -22 ), - BaseShiftK31 = cms.int32( -23 ), - BaseShiftR00 = cms.int32( -4 ), - BaseShiftR11 = cms.int32( 7 ), - BaseShiftInvR00Approx = cms.int32( -27 ), - BaseShiftInvR11Approx = cms.int32( -38 ), - BaseShiftInvR00Cor = cms.int32( -15 ), - BaseShiftInvR11Cor = cms.int32( -15 ), - BaseShiftInvR00 = cms.int32( -23 ), - BaseShiftInvR11 = cms.int32( -33 ), - BaseShiftC00 = cms.int32( 5 ), - BaseShiftC01 = cms.int32( -3 ), - BaseShiftC11 = cms.int32( -7 ), - BaseShiftC22 = cms.int32( 3 ), - BaseShiftC23 = cms.int32( 0 ), - BaseShiftC33 = cms.int32( 1 ) - ), + WidthR00 = cms.int32( 20 ), + WidthR11 = cms.int32( 20 ), + + WidthC00 = cms.int32( 20 ), + WidthC01 = cms.int32( 20 ), + WidthC11 = cms.int32( 20 ), + WidthC22 = cms.int32( 20 ), + WidthC23 = cms.int32( 20 ), + WidthC33 = cms.int32( 20 ), + + BaseShiftx0 = cms.int32( 0 ), + BaseShiftx1 = cms.int32( -7 ), + BaseShiftx2 = cms.int32( -1 ), + BaseShiftx3 = cms.int32( -1 ), + + BaseShiftr0 = cms.int32( -8 ), + BaseShiftr1 = cms.int32( 0 ), + + BaseShiftS00 = cms.int32( -4 ), + BaseShiftS01 = cms.int32( -12 ), + BaseShiftS12 = cms.int32( 0 ), + BaseShiftS13 = cms.int32( -1 ), + + BaseShiftR00 = cms.int32( -5 ), + BaseShiftR11 = cms.int32( 6 ), + + BaseShiftInvR00Approx = cms.int32( -30 ), + BaseShiftInvR11Approx = cms.int32( -41 ), + BaseShiftInvR00Cor = cms.int32( -24 ), + BaseShiftInvR11Cor = cms.int32( -24 ), + BaseShiftInvR00 = cms.int32( -30 ), + BaseShiftInvR11 = cms.int32( -41 ), + + BaseShiftS00Shifted = cms.int32( -1 ), + BaseShiftS01Shifted = cms.int32( -7 ), + BaseShiftS12Shifted = cms.int32( 4 ), + BaseShiftS13Shifted = cms.int32( 4 ), + + BaseShiftK00 = cms.int32( -7 ), + BaseShiftK10 = cms.int32( -13 ), + BaseShiftK21 = cms.int32( -13 ), + BaseShiftK31 = cms.int32( -13 ), + + BaseShiftC00 = cms.int32( 6 ), + BaseShiftC01 = cms.int32( 1 ), + BaseShiftC11 = cms.int32( -6 ), + BaseShiftC22 = cms.int32( 5 ), + BaseShiftC23 = cms.int32( 6 ), + BaseShiftC33 = cms.int32( 5 ), + + BaseShiftr0Shifted = cms.int32( 4 ), + BaseShiftr1Shifted = cms.int32( 7 ), + BaseShiftr02 = cms.int32( -1 ), + BaseShiftr12 = cms.int32( 10 ), + BaseShiftchi20 = cms.int32( -10 ), + BaseShiftchi21 = cms.int32( -10 ) ) diff --git a/L1Trigger/TrackerTFP/python/LayerEncoding_cff.py b/L1Trigger/TrackerTFP/python/LayerEncoding_cff.py new file mode 100644 index 0000000000000..7bc742aac993a --- /dev/null +++ b/L1Trigger/TrackerTFP/python/LayerEncoding_cff.py @@ -0,0 +1,5 @@ +# ESProducer providing layer id encoding (Layers consitent with rough r-z track parameters are counted from 0 onwards) used by Kalman Filter + +import FWCore.ParameterSet.Config as cms + +TrackTriggerLayerEncoding = cms.ESProducer("trackerTFP::ProducerLayerEncoding") \ No newline at end of file diff --git a/L1Trigger/TrackerTFP/python/ProducerES_cff.py b/L1Trigger/TrackerTFP/python/ProducerES_cff.py deleted file mode 100644 index a10a837ed809f..0000000000000 --- a/L1Trigger/TrackerTFP/python/ProducerES_cff.py +++ /dev/null @@ -1,5 +0,0 @@ -import FWCore.ParameterSet.Config as cms - -from L1Trigger.TrackerTFP.ProducerES_cfi import TrackTriggerDataFormats_params - -TrackTriggerDataFormats = cms.ESProducer("trackerTFP::ProducerES", TrackTriggerDataFormats_params) diff --git a/L1Trigger/TrackerTFP/python/ProducerES_cfi.py b/L1Trigger/TrackerTFP/python/ProducerES_cfi.py deleted file mode 100644 index 069a37dbeb5bc..0000000000000 --- a/L1Trigger/TrackerTFP/python/ProducerES_cfi.py +++ /dev/null @@ -1,28 +0,0 @@ -import FWCore.ParameterSet.Config as cms - -TrackTriggerDataFormats_params = cms.PSet ( - - UseHybrid = cms.bool( True ), - - ZHoughTransform = cms.PSet ( - - NumBinsZT = cms.int32( 2 ), - NumBinsCot = cms.int32( 2 ), - NumStages = cms.int32( 5 ) - - ), - - KalmanFilter = cms.PSet ( - - RangeFactor = cms.double( 2.0 ) # search window of each track parameter in initial uncertainties - - ), - - DuplicateRemoval = cms.PSet ( - WidthPhi0 = cms.int32( 12 ), # number of bist used for phi0 - WidthQoverPt = cms.int32( 15 ), # number of bist used for qOverPt - WidthCot = cms.int32( 16 ), # number of bist used for cot(theta) - WidthZ0 = cms.int32( 12 ) # number of bist used for z0 - ), - -) diff --git a/L1Trigger/TrackerTFP/python/ProducerLayerEncoding_cff.py b/L1Trigger/TrackerTFP/python/ProducerLayerEncoding_cff.py deleted file mode 100644 index 9fbd580d8b4af..0000000000000 --- a/L1Trigger/TrackerTFP/python/ProducerLayerEncoding_cff.py +++ /dev/null @@ -1,5 +0,0 @@ -import FWCore.ParameterSet.Config as cms - -from L1Trigger.TrackerTFP.ProducerLayerEncoding_cfi import TrackTriggerLayerEncoding_params - -TrackTriggerLayerEncoding = cms.ESProducer("trackerTFP::ProducerLayerEncoding", TrackTriggerLayerEncoding_params) diff --git a/L1Trigger/TrackerTFP/python/ProducerLayerEncoding_cfi.py b/L1Trigger/TrackerTFP/python/ProducerLayerEncoding_cfi.py deleted file mode 100644 index 3b098a398a863..0000000000000 --- a/L1Trigger/TrackerTFP/python/ProducerLayerEncoding_cfi.py +++ /dev/null @@ -1,7 +0,0 @@ -import FWCore.ParameterSet.Config as cms - -TrackTriggerLayerEncoding_params = cms.PSet ( - - - -) diff --git a/L1Trigger/TrackerTFP/python/Producer_cff.py b/L1Trigger/TrackerTFP/python/Producer_cff.py index 4276962e6d5f6..2aeaac76fe8e6 100644 --- a/L1Trigger/TrackerTFP/python/Producer_cff.py +++ b/L1Trigger/TrackerTFP/python/Producer_cff.py @@ -1,18 +1,18 @@ # Produce L1 tracks with TMTT C++ emulation import FWCore.ParameterSet.Config as cms -from L1Trigger.TrackTrigger.ProducerSetup_cff import TrackTriggerSetup +from L1Trigger.TrackTrigger.Setup_cff import TrackTriggerSetup from L1Trigger.TrackerTFP.Producer_cfi import TrackerTFPProducer_params -from L1Trigger.TrackerTFP.ProducerES_cff import TrackTriggerDataFormats -from L1Trigger.TrackerTFP.ProducerLayerEncoding_cff import TrackTriggerLayerEncoding -from L1Trigger.TrackerTFP.KalmanFilterFormats_cff import TrackTriggerKalmanFilterFormats +from L1Trigger.TrackerTFP.DataFormats_cff import TrackTriggerDataFormats +from L1Trigger.TrackerTFP.TrackQuality_cff import * +from L1Trigger.TrackerTFP.LayerEncoding_cff import TrackTriggerLayerEncoding +from L1Trigger.TrackerTFP.KalmanFilterFormats_cfi import TrackTriggerKalmanFilterFormats_params -TrackerTFPProducerGP = cms.EDProducer( 'trackerTFP::ProducerGP', TrackerTFPProducer_params ) -TrackerTFPProducerHT = cms.EDProducer( 'trackerTFP::ProducerHT', TrackerTFPProducer_params ) -TrackerTFPProducerMHT = cms.EDProducer( 'trackerTFP::ProducerMHT', TrackerTFPProducer_params ) -TrackerTFPProducerZHT = cms.EDProducer( 'trackerTFP::ProducerZHT', TrackerTFPProducer_params ) -TrackerTFPProducerZHTout = cms.EDProducer( 'trackerTFP::ProducerZHTout', TrackerTFPProducer_params ) -TrackerTFPProducerKFin = cms.EDProducer( 'trackerTFP::ProducerKFin', TrackerTFPProducer_params ) -TrackerTFPProducerKF = cms.EDProducer( 'trackerTFP::ProducerKF', TrackerTFPProducer_params ) -TrackerTFPProducerTT = cms.EDProducer( 'trackerTFP::ProducerTT', TrackerTFPProducer_params ) -TrackerTFPProducerAS = cms.EDProducer( 'trackerTFP::ProducerAS', TrackerTFPProducer_params ) +ProducerPP = cms.EDProducer( 'trackerTFP::ProducerPP' , TrackerTFPProducer_params ) +ProducerGP = cms.EDProducer( 'trackerTFP::ProducerGP' , TrackerTFPProducer_params ) +ProducerHT = cms.EDProducer( 'trackerTFP::ProducerHT' , TrackerTFPProducer_params ) +ProducerCTB = cms.EDProducer( 'trackerTFP::ProducerCTB', TrackerTFPProducer_params ) +ProducerKF = cms.EDProducer( 'trackerTFP::ProducerKF' , TrackerTFPProducer_params, TrackTriggerKalmanFilterFormats_params ) +ProducerDR = cms.EDProducer( 'trackerTFP::ProducerDR' , TrackerTFPProducer_params ) +ProducerTQ = cms.EDProducer( 'trackerTFP::ProducerTQ' , TrackerTFPProducer_params ) +ProducerTFP = cms.EDProducer( 'trackerTFP::ProducerTFP', TrackerTFPProducer_params ) diff --git a/L1Trigger/TrackerTFP/python/Producer_cfi.py b/L1Trigger/TrackerTFP/python/Producer_cfi.py index 351cab0337c78..7a6880dc42e4a 100644 --- a/L1Trigger/TrackerTFP/python/Producer_cfi.py +++ b/L1Trigger/TrackerTFP/python/Producer_cfi.py @@ -1,24 +1,22 @@ +# configuartion for L1 track Producer + import FWCore.ParameterSet.Config as cms TrackerTFPProducer_params = cms.PSet ( - LabelDTC = cms.string( "TrackerDTCProducer" ), # - LabelGP = cms.string( "TrackerTFPProducerGP" ), # - LabelHT = cms.string( "TrackerTFPProducerHT" ), # - LabelMHT = cms.string( "TrackerTFPProducerMHT" ), # - LabelZHT = cms.string( "TrackerTFPProducerZHT" ), # - LabelZHTout = cms.string( "TrackerTFPProducerZHTout" ), # - LabelKFin = cms.string( "TrackerTFPProducerKFin" ), # - LabelKF = cms.string( "TrackerTFPProducerKF" ), # - LabelDR = cms.string( "TrackerTFPProducerDR" ), # - LabelTT = cms.string( "TrackerTFPProducerTT" ), # - LabelAS = cms.string( "TrackerTFPProducerAS" ), # - BranchAcceptedStubs = cms.string( "StubAccepted" ), # branch for prodcut with passed stubs - BranchAcceptedTracks = cms.string( "TrackAccepted" ), # branch for prodcut with passed tracks - BranchLostStubs = cms.string( "StubLost" ), # branch for prodcut with lost stubs - BranchLostTracks = cms.string( "TracksLost" ), # branch for prodcut with lost tracks - CheckHistory = cms.bool ( False ), # checks if input sample production is configured as current process - EnableTruncation = cms.bool ( True ), # enable emulation of truncation, lost stubs are filled in BranchLost - PrintKFDebug = cms.bool ( False ) # print end job internal unused MSB + InputLabelPP = cms.string( "ProducerDTC" ), # + InputLabelGP = cms.string( "ProducerPP" ), # + InputLabelHT = cms.string( "ProducerGP" ), # + InputLabelCTB = cms.string( "ProducerHT" ), # + InputLabelKF = cms.string( "ProducerCTB" ), # + InputLabelDR = cms.string( "ProducerKF" ), # + InputLabelTQ = cms.string( "ProducerDR" ), # + InputLabelTFP = cms.string( "ProducerTQ" ), # + BranchStubs = cms.string( "StubAccepted" ), # branch for prodcut with passed stubs + BranchTracks = cms.string( "TrackAccepted" ), # branch for prodcut with passed tracks + BranchTTTracks = cms.string( "TrackAccepted" ), # branch for prodcut with passed TTTracks + BranchTruncated = cms.string( "Truncated" ), # branch for truncated prodcuts + EnableTruncation = cms.bool ( True ), # enable emulation of truncation, lost stubs are filled in BranchLost + PrintKFDebug = cms.bool ( True ) # print end job internal unused MSB ) diff --git a/L1Trigger/TrackerTFP/python/TrackQuality_cff.py b/L1Trigger/TrackerTFP/python/TrackQuality_cff.py new file mode 100644 index 0000000000000..133389c0e6a34 --- /dev/null +++ b/L1Trigger/TrackerTFP/python/TrackQuality_cff.py @@ -0,0 +1,12 @@ +# ESProducer providing Bit accurate emulation of the track quality BDT + +import FWCore.ParameterSet.Config as cms +from L1Trigger.TrackerTFP.TrackQuality_cfi import TrackQuality_params + +TrackTriggerTrackQuality = cms.ESProducer("trackerTFP::ProducerTrackQuality", TrackQuality_params) + +fakeTrackTriggerTrackQualitySource = cms.ESSource("EmptyESSource", + recordName = cms.string('trackerTFP::TrackQualityRcd'), + iovIsRunNotTime = cms.bool(True), + firstValid = cms.vuint32(1) +) diff --git a/L1Trigger/TrackerTFP/python/TrackQuality_cfi.py b/L1Trigger/TrackerTFP/python/TrackQuality_cfi.py new file mode 100644 index 0000000000000..f4b0210124c2d --- /dev/null +++ b/L1Trigger/TrackerTFP/python/TrackQuality_cfi.py @@ -0,0 +1,35 @@ +# configuration of TrackTriggerTrackQuality + +import FWCore.ParameterSet.Config as cms + +TrackQuality_params = cms.PSet( + # This emulation GBDT is optimised for the HYBRID_NEWKF emulation and works with the emulation of the KF out module + # It is compatible with the HYBRID simulation and will give equivilant performance with this workflow + Model = cms.FileInPath( "L1Trigger/TrackTrigger/data/L1_TrackQuality_GBDT_emulation_digitized.json" ), + #Vector of strings of training features, in the order that the model was trained with + FeatureNames = cms.vstring( ["tanl", + "z0_scaled", + "bendchi2_bin", + "nstub", + "nlaymiss_interior", + "chi2rphi_bin", + "chi2rz_bin" + ] ), + BaseShiftCot = cms.int32( -7 ), # + BaseShiftZ0 = cms.int32( -6 ), # + BaseShiftAPfixed = cms.int32( -5 ), # + Chi2rphiConv = cms.int32( 3 ), # Conversion factor between dphi^2/weight and chi2rphi + Chi2rzConv = cms.int32( 13 ), # Conversion factor between dz^2/weight and chi2rz + WeightBinFraction = cms.int32( 0 ), # Number of bits dropped from dphi and dz for v0 and v1 LUTs + DzTruncation = cms.int32( 262144 ), # Constant used in FW to prevent 32-bit int overflow + DphiTruncation = cms.int32( 16 ), # Constant used in FW to prevent 32-bit int overflow + WidthM20 = cms.int32( 16 ), # + WidthM21 = cms.int32( 16 ), # + WidthInvV0 = cms.int32( 16 ), # + WidthInvV1 = cms.int32( 16 ), # + Widthchi2rphi = cms.int32( 20 ), # + Widthchi2rz = cms.int32( 20 ), # + BaseShiftchi2rphi = cms.int32( -6 ), # + BaseShiftchi2rz = cms.int32( -6 ) # + +) diff --git a/L1Trigger/TrackerTFP/src/CleanTrackBuilder.cc b/L1Trigger/TrackerTFP/src/CleanTrackBuilder.cc new file mode 100644 index 0000000000000..a4ab33d38bf34 --- /dev/null +++ b/L1Trigger/TrackerTFP/src/CleanTrackBuilder.cc @@ -0,0 +1,489 @@ +#include "L1Trigger/TrackerTFP/interface/CleanTrackBuilder.h" + +#include +#include +#include +#include +#include +#include + +using namespace std; +using namespace edm; +using namespace tt; + +namespace trackerTFP { + + CleanTrackBuilder::CleanTrackBuilder(const ParameterSet& iConfig, + const Setup* setup, + const DataFormats* dataFormats, + const LayerEncoding* layerEncoding, + const DataFormat& cot, + vector& stubs, + vector& tracks) + : enableTruncation_(iConfig.getParameter("EnableTruncation")), + setup_(setup), + dataFormats_(dataFormats), + layerEncoding_(layerEncoding), + cot_(cot), + stubsCTB_(stubs), + tracksCTB_(tracks) { + stubs_.reserve(stubs.capacity()); + tracks_.reserve(tracks.capacity()); + } + + // fill output products + void CleanTrackBuilder::produce(const vector>& streamsIn, + vector>& regionTracks, + vector>>& regionStubs) { + static const int numChannelIn = dataFormats_->numChannel(Process::ht); + static const int numChannelOut = dataFormats_->numChannel(Process::ctb); + static const int numChannel = numChannelIn / numChannelOut; + static const int numLayers = setup_->numLayers(); + // loop over worker + for (int channelOut = 0; channelOut < numChannelOut; channelOut++) { + //if (channelOut != 3) + //continue; + // clean input tracks + vector> streamsT(numChannel); + vector> streamsS(numChannel); + for (int cin = 0; cin < numChannel; cin++) { + //if (cin != 1) + //continue; + const int index = numChannel * cin + channelOut; + cleanStream(streamsIn[index], streamsT[cin], streamsS[cin], index); + } + // route + deque tracks; + vector> stubs(numLayers); + route(streamsT, tracks); + route(streamsS, stubs); + // sort + sort(tracks, stubs); + // convert + deque& channelTracks = regionTracks[channelOut]; + vector>& channelStubs = regionStubs[channelOut]; + convert(tracks, stubs, channelTracks, channelStubs); + } + } + + // + void CleanTrackBuilder::cleanStream(const vector& input, + deque& tracks, + deque& stubs, + int channelId) { + static const DataFormat dfInv2R = dataFormats_->format(Variable::inv2R, Process::ht); + const double inv2R = dfInv2R.floating(dfInv2R.toSigned(channelId)); + const int offset = channelId * setup_->ctbMaxTracks(); + int trackId = offset; + // identify tracks in input container + int id; + auto toTrkId = [this](StubHT* stub) { + static const DataFormat& phiT = dataFormats_->format(Variable::phiT, Process::ht); + static const DataFormat& zT = dataFormats_->format(Variable::zT, Process::ht); + return (phiT.ttBV(stub->phiT()) + zT.ttBV(stub->zT())).val(); + }; + auto different = [&id, toTrkId](StubHT* stub) { return id != toTrkId(stub); }; + int delta = -setup_->htMinLayers() + 1; + int old = 0; + for (auto it = input.begin(); it != input.end();) { + id = toTrkId(*it); + const auto start = it; + const auto end = find_if(start, input.end(), different); + const vector track(start, end); + // restore clock accurancy + delta += (int)track.size() - old; + old = track.size(); + if (delta > 0) { + stubs.insert(stubs.end(), delta, nullptr); + tracks.insert(tracks.end(), delta, nullptr); + delta = 0; + } + // run single track through r-phi and r-z hough transform + cleanTrack(track, tracks, stubs, inv2R, (*start)->zT(), trackId++); + if (trackId - offset == setup_->ctbMaxTracks()) + break; + // set begin of next track + it = end; + } + } + + // run single track through r-phi and r-z hough transform + void CleanTrackBuilder::cleanTrack( + const vector& track, deque& tracks, deque& stubs, double inv2R, int zT, int trackId) { + static const int wlayer = dataFormats_->width(Variable::layer, Process::ctb); + static const DataFormat& dfR = dataFormats_->format(Variable::r, Process::ctb); + static const DataFormat& dfPhi = dataFormats_->format(Variable::phi, Process::ctb); + static const DataFormat& dfZ = dataFormats_->format(Variable::z, Process::ctb); + static const DataFormat& dfPhiT = dataFormats_->format(Variable::phiT, Process::ctb); + static const DataFormat& dfZT = dataFormats_->format(Variable::zT, Process::ctb); + static const int numBinsInv2R = setup_->ctbNumBinsInv2R(); + static const int numBinsPhiT = setup_->ctbNumBinsPhiT(); + static const int numBinsCot = setup_->ctbNumBinsCot(); + static const int numBinsZT = setup_->ctbNumBinsZT(); + static const double baseInv2R = dataFormats_->base(Variable::inv2R, Process::ctb) / numBinsInv2R; + static const double basePhiT = dfPhiT.base() / numBinsPhiT; + static const double baseCot = cot_.base(); + static const double baseZT = dfZT.base() / numBinsZT; + const TTBV& maybePattern = layerEncoding_->maybePattern(zT); + auto noTrack = [this, &maybePattern](const TTBV& pattern) { + // not enough seeding layer + if (pattern.count(0, setup_->kfMaxSeedingLayer()) < 2) + return true; + int nHits(0); + int nGaps(0); + bool doubleGap = false; + for (int layer = 0; layer < setup_->numLayers(); layer++) { + if (pattern.test(layer)) { + doubleGap = false; + if(++nHits == setup_->ctbMinLayers()) + return false; + } else if (!maybePattern.test(layer)) { + if (++nGaps == setup_->kfMaxGaps() || doubleGap) + break; + doubleGap = true; + } + } + return true; + }; + auto toLayerId = [](StubHT* stub) { return stub->layer().val(wlayer); }; + auto toDPhi = [this, inv2R](StubHT* stub) { + const bool barrel = stub->layer()[5]; + const bool ps = stub->layer()[4]; + const bool tilt = stub->layer()[3]; + const double pitchRow = ps ? setup_->pitchRowPS() : setup_->pitchRow2S(); + const double pitchCol = ps ? setup_->pitchColPS() : setup_->pitchCol2S(); + const double pitchColR = barrel ? (tilt ? setup_->tiltUncertaintyR() : 0.0) : pitchCol; + const double r = stub->r() + setup_->chosenRofPhi(); + const double dPhi = pitchRow / r + (setup_->scattering() + pitchColR) * abs(inv2R); + return dfPhi.digi(dPhi / 2.); + }; + auto toDZ = [this](StubHT* stub) { + static const double m = setup_->tiltApproxSlope(); + static const double c = setup_->tiltApproxIntercept(); + const bool barrel = stub->layer()[5]; + const bool ps = stub->layer()[4]; + const bool tilt = stub->layer()[3]; + const double pitchCol = ps ? setup_->pitchColPS() : setup_->pitchCol2S(); + const double zT = dfZT.floating(stub->zT()); + const double cot = abs(zT) / setup_->chosenRofZ(); + const double dZ = (barrel ? (tilt ? m * cot + c : 1.) : cot) * pitchCol; + return dfZ.digi(dZ / 2.); + }; + vector tStubs; + tStubs.reserve(track.size()); + vector hitPatternPhi(numBinsInv2R * numBinsPhiT, TTBV(0, setup_->numLayers())); + vector hitPatternZ(numBinsCot * numBinsZT, TTBV(0, setup_->numLayers())); + TTBV tracksPhi(0, numBinsInv2R * numBinsPhiT); + TTBV tracksZ(0, numBinsCot * numBinsZT); + // identify finer tracks each stub is consistent with + for (StubHT* stub : track) { + const int layerId = toLayerId(stub); + const double dPhi = toDPhi(stub); + const double dZ = toDZ(stub); + // r - phi HT + auto phiT = [stub](double inv2R, double dPhi) { return inv2R * stub->r() + stub->phi() + dPhi; }; + TTBV hitsPhi(0, numBinsInv2R * numBinsPhiT); + for (int binInv2R = 0; binInv2R < numBinsInv2R; binInv2R++) { + const int offset = binInv2R * numBinsPhiT; + const double inv2RMin = (binInv2R - numBinsInv2R / 2.) * baseInv2R; + const double inv2RMax = inv2RMin + baseInv2R; + const auto phiTs = {phiT(inv2RMin, -dPhi), phiT(inv2RMax, -dPhi), phiT(inv2RMin, dPhi), phiT(inv2RMax, dPhi)}; + const int binPhiTMin = floor(*min_element(phiTs.begin(), phiTs.end()) / basePhiT + 1.e-11) + numBinsPhiT / 2; + const int binPhiTMax = floor(*max_element(phiTs.begin(), phiTs.end()) / basePhiT + 1.e-11) + numBinsPhiT / 2; + for (int binPhiT = 0; binPhiT < numBinsPhiT; binPhiT++) + if (binPhiT >= binPhiTMin && binPhiT <= binPhiTMax) + hitsPhi.set(offset + binPhiT); + } + // check for tracks on the fly + for (int phi : hitsPhi.ids()) { + hitPatternPhi[phi].set(layerId); + if (!noTrack(hitPatternPhi[phi])) + tracksPhi.set(phi); + } + // r - z HT + auto zT = [this, stub](double cot, double dZ) { + const double r = dfR.digi(stub->r() + dfR.digi(setup_->chosenRofPhi() - setup_->chosenRofZ())); + return cot * r + stub->z() + dZ; + }; + TTBV hitsZ(0, numBinsCot * numBinsZT); + for (int binCot = 0; binCot < numBinsCot; binCot++) { + const int offset = binCot * numBinsZT; + const double cotMin = (binCot - numBinsCot / 2.) * baseCot; + const double cotMax = cotMin + baseCot; + const auto zTs = {zT(cotMin, -dZ), zT(cotMax, -dZ), zT(cotMin, dZ), zT(cotMax, dZ)}; + const int binZTMin = floor(*min_element(zTs.begin(), zTs.end()) / baseZT + 1.e-11) + numBinsZT / 2; + const int binZTMax = floor(*max_element(zTs.begin(), zTs.end()) / baseZT + 1.e-11) + numBinsZT / 2; + for (int binZT = 0; binZT < numBinsZT; binZT++) + if (binZT >= binZTMin && binZT <= binZTMax) + hitsZ.set(offset + binZT); + } + // check for tracks on the fly + for (int z : hitsZ.ids()) { + hitPatternZ[z].set(layerId); + if (!noTrack(hitPatternZ[z])) + tracksZ.set(z); + } + // store stubs consistent finer tracks + stubs_.emplace_back(stub, trackId, hitsPhi, hitsZ, layerId, dPhi, dZ); + tStubs.push_back(&stubs_.back()); + } + // clean + tracks.insert(tracks.end(), tStubs.size() - 1, nullptr); + tracks_.emplace_back(setup_, trackId, tracksPhi, tracksZ, tStubs, inv2R); + tracks.push_back(&tracks_.back()); + stubs.insert(stubs.end(), tStubs.begin(), tStubs.end()); + } + + // + void CleanTrackBuilder::Stub::update(const TTBV& phi, const TTBV& z, vector& ids, int max) { + auto consistent = [](const TTBV& stub, const TTBV& track) { + for (int id : track.ids()) + if (stub[id]) + return true; + return false; + }; + if (consistent(hitsPhi_, phi) && consistent(hitsZ_, z) && ids[layerId_] < max) + stubId_ = ids[layerId_]++; + else + valid_ = false; + } + + // construct Track from Stubs + CleanTrackBuilder::Track::Track(const Setup* setup, + int trackId, + const TTBV& hitsPhi, + const TTBV& hitsZ, + const std::vector& stubs, + double inv2R) + : valid_(true), stubs_(stubs), trackId_(trackId), hitsPhi_(hitsPhi), hitsZ_(hitsZ), inv2R_(inv2R) { + vector stubIds(setup->numLayers(), 0); + for (Stub* stub : stubs_) + stub->update(hitsPhi_, hitsZ_, stubIds, setup->ctbMaxStubs()); + const int nLayer = + accumulate(stubIds.begin(), stubIds.end(), 0, [](int& sum, int i) { return sum += (i > 0 ? 1 : 0); }); + if (nLayer < setup->ctbMinLayers()) + valid_ = false; + size_ = *max_element(stubIds.begin(), stubIds.end()); + } + + // + void CleanTrackBuilder::route(vector>& input, vector>& outputs) const { + for (int channelOut = 0; channelOut < (int)outputs.size(); channelOut++) { + deque& output = outputs[channelOut]; + vector> inputs(input); + for (deque& stream : inputs) { + for (Stub*& stub : stream) + if (stub && (!stub->valid_ || stub->layerId_ != channelOut)) + stub = nullptr; + for (auto it = stream.end(); it != stream.begin();) + it = (*--it) ? stream.begin() : stream.erase(it); + } + vector> stacks(input.size()); + // clock accurate firmware emulation, each while trip describes one clock tick, one stub in and one stub out per tick + while (!all_of(inputs.begin(), inputs.end(), [](const deque& stubs) { return stubs.empty(); }) or + !all_of(stacks.begin(), stacks.end(), [](const deque& stubs) { return stubs.empty(); })) { + // fill input fifos + for (int channel = 0; channel < (int)input.size(); channel++) { + deque& stack = stacks[channel]; + Stub* stub = pop_front(inputs[channel]); + if (stub) { + if (enableTruncation_ && (int)stack.size() == setup_->ctbDepthMemory() - 1) + pop_front(stack); + stack.push_back(stub); + } + } + // merge input fifos to one stream, prioritizing lower input channel over higher channel + bool nothingToRoute(true); + for (int channel = 0; channel < (int)input.size(); channel++) { + Stub* stub = pop_front(stacks[channel]); + if (stub) { + nothingToRoute = false; + output.push_back(stub); + break; + } + } + if (nothingToRoute) + output.push_back(nullptr); + } + } + } + + // + void CleanTrackBuilder::route(vector>& inputs, deque& output) const { + vector> stacks(inputs.size()); + // clock accurate firmware emulation, each while trip describes one clock tick, one stub in and one stub out per tick + while (!all_of(inputs.begin(), inputs.end(), [](const deque& tracks) { return tracks.empty(); }) or + !all_of(stacks.begin(), stacks.end(), [](const deque& tracks) { return tracks.empty(); })) { + // fill input fifos + for (int channel = 0; channel < (int)inputs.size(); channel++) { + deque& stack = stacks[channel]; + Track* track = pop_front(inputs[channel]); + if (track && track->valid_) { + if (enableTruncation_ && (int)stack.size() == setup_->ctbDepthMemory() - 1) + pop_front(stack); + stack.push_back(track); + } + } + // merge input fifos to one stream, prioritizing lower input channel over higher channel + bool nothingToRoute(true); + for (int channel = 0; channel < (int)inputs.size(); channel++) { + Track* track = pop_front(stacks[channel]); + if (track) { + nothingToRoute = false; + output.push_back(track); + break; + } + } + if (nothingToRoute) + output.push_back(nullptr); + } + } + + // sort + void CleanTrackBuilder::sort(deque& tracks, vector>& stubs) const { + // aplly truncation + if (enableTruncation_) { + if ((int)tracks.size() > setup_->numFramesHigh()) + tracks.resize(setup_->numFramesHigh()); + for (deque& stream : stubs) + if ((int)stream.size() > setup_->numFramesHigh()) + stream.resize(setup_->numFramesHigh()); + } + // cycle event, remove all gaps + tracks.erase(remove(tracks.begin(), tracks.end(), nullptr), tracks.end()); + for (deque& stream : stubs) + stream.erase(remove(stream.begin(), stream.end(), nullptr), stream.end()); + // prepare sort according to track id arrival order + vector trackIds; + trackIds.reserve(tracks.size()); + transform(tracks.begin(), tracks.end(), back_inserter(trackIds), [](Track* track) { return track->trackId_; }); + auto cleaned = [&trackIds](Stub* stub) { + return find(trackIds.begin(), trackIds.end(), stub->trackId_) == trackIds.end(); + }; + auto order = [&trackIds](auto lhs, auto rhs) { + const auto l = find(trackIds.begin(), trackIds.end(), lhs->trackId_); + const auto r = find(trackIds.begin(), trackIds.end(), rhs->trackId_); + return distance(r, l) < 0; + }; + for (deque& stream : stubs) { + // remove stubs from removed tracks + stream.erase(remove_if(stream.begin(), stream.end(), cleaned), stream.end()); + // sort according to stub id on layer + stable_sort(stream.begin(), stream.end(), [](Stub* lhs, Stub* rhs) { return lhs->stubId_ < rhs->stubId_; }); + // sort according to track id arrival order + stable_sort(stream.begin(), stream.end(), order); + } + // add all gaps + const int size = + accumulate(tracks.begin(), tracks.end(), 0, [](int& sum, Track* track) { return sum += track->size_; }); + for (int frame = 0; frame < size;) { + const int trackId = tracks[frame]->trackId_; + const int length = tracks[frame]->size_; + tracks.insert(next(tracks.begin(), frame + 1), length - 1, nullptr); + for (int layer = 0; layer < setup_->numLayers(); layer++) { + deque& stream = stubs[layer]; + if (frame >= (int)stream.size()) { + stream.insert(stream.end(), length, nullptr); + continue; + } + const auto begin = next(stream.begin(), frame); + const auto end = find_if(begin, stream.end(), [trackId](Stub* stub) { return stub->trackId_ != trackId; }); + stream.insert(end, length - distance(begin, end), nullptr); + } + frame += length; + } + } + + // + void CleanTrackBuilder::convert(const deque& iTracks, + const vector>& iStubs, + deque& oTracks, + vector>& oStubs) { + for (int iFrame = 0; iFrame < (int)iTracks.size();) { + Track* track = iTracks[iFrame]; + if (!track) { + oTracks.push_back(nullptr); + for (deque& stubs : oStubs) + stubs.push_back(nullptr); + iFrame++; + continue; + } + StubHT* s = nullptr; + for (int layer = 0; layer < setup_->numLayers(); layer++) { + for (int n = 0; n < track->size_; n++) { + Stub* stub = iStubs[layer][iFrame + n]; + if (!stub) { + oStubs[layer].push_back(nullptr); + continue; + } + s = stub->stubHT_; + const double r = s->r(); + const double phi = s->phi(); + const double z = s->z(); + const double dPhi = stub->dPhi_; + const double dZ = stub->dZ_; + stubsCTB_.emplace_back(*s, r, phi, z, dPhi, dZ); + oStubs[layer].push_back(&stubsCTB_.back()); + } + } + const double inv2R = track->inv2R_; + const double phiT = dataFormats_->format(Variable::phiT, Process::ctb).floating(s->phiT()); + const double zT = dataFormats_->format(Variable::zT, Process::ctb).floating(s->zT()); + tracksCTB_.emplace_back(TTTrackRef(), dataFormats_, inv2R, phiT, zT); + oTracks.push_back(&tracksCTB_.back()); + oTracks.insert(oTracks.end(), track->size_ - 1, nullptr); + iFrame += track->size_; + } + } + + // remove and return first element of deque, returns nullptr if empty + template + T* CleanTrackBuilder::pop_front(deque& ts) const { + T* t = nullptr; + if (!ts.empty()) { + t = ts.front(); + ts.pop_front(); + } + return t; + } + + void CleanTrackBuilder::put(TrackCTB* track, + const vector>& stubs, + int region, + TTTracks& ttTracks) const { + static const double dPhi = dataFormats_->format(Variable::phiT, Process::ctb).range(); + const double invR = -track->inv2R() * 2.; + const double phi0 = deltaPhi(track->phiT() - track->inv2R() * setup_->chosenRofPhi() + region * dPhi); + const double zT = track->zT(); + const double cot = zT / setup_->chosenRofZ(); + TTBV hits(0, setup_->numLayers()); + double chi2phi(0.); + double chi2z(0.); + const int nStubs = accumulate( + stubs.begin(), stubs.end(), 0, [](int& sum, const vector& layer) { return sum += layer.size(); }); + vector ttStubRefs; + ttStubRefs.reserve(nStubs); + for (int layer = 0; layer < setup_->numLayers(); layer++) { + for (StubCTB* stub : stubs[layer]) { + hits.set(layer); + chi2phi += pow(stub->phi(), 2) / pow(stub->dPhi(), 2); + chi2z += pow(stub->z(), 2) / pow(stub->dZ(), 2); + ttStubRefs.push_back(stub->frame().first); + } + } + static constexpr int nPar = 4; + static constexpr double d0 = 0.; + static constexpr double z0 = 0; + static constexpr double trkMVA1 = 0.; + static constexpr double trkMVA2 = 0.; + static constexpr double trkMVA3 = 0.; + const int hitPattern = hits.val(); + const double bField = setup_->bField(); + TTTrack ttTrack( + invR, phi0, cot, z0, d0, chi2phi, chi2z, trkMVA1, trkMVA2, trkMVA3, hitPattern, nPar, bField); + ttTrack.setStubRefs(ttStubRefs); + ttTrack.setPhiSector(region); + ttTracks.emplace_back(ttTrack); + } + +} // namespace trackerTFP diff --git a/L1Trigger/TrackerTFP/src/DataFormats.cc b/L1Trigger/TrackerTFP/src/DataFormats.cc index d58ba3d15b50b..044adc5d7a61b 100644 --- a/L1Trigger/TrackerTFP/src/DataFormats.cc +++ b/L1Trigger/TrackerTFP/src/DataFormats.cc @@ -1,5 +1,5 @@ #include "L1Trigger/TrackerTFP/interface/DataFormats.h" -#include "L1Trigger/TrackTrigger/interface/StubPtConsistency.h" +#include "DataFormats/L1TrackTrigger/interface/TTTrack_TrackWord.h" #include #include @@ -9,6 +9,7 @@ #include #include #include +#include using namespace std; using namespace edm; @@ -16,17 +17,16 @@ using namespace tt; namespace trackerTFP { - // default constructor, trying to needs space as proper constructed object + // default constructor, trying to need space as proper constructed object DataFormats::DataFormats() : numDataFormats_(0), formats_(+Variable::end, std::vector(+Process::end, nullptr)), - numUnusedBitsStubs_(+Process::end, TTBV::S_), - numUnusedBitsTracks_(+Process::end, TTBV::S_), + numUnusedBitsStubs_(+Process::end, TTBV::S_ - 1), + numUnusedBitsTracks_(+Process::end, TTBV::S_ - 1), numChannel_(+Process::end, 0) { setup_ = nullptr; countFormats(); dataFormats_.reserve(numDataFormats_); - numStreams_.reserve(+Process::end); numStreamsStubs_.reserve(+Process::end); numStreamsTracks_.reserve(+Process::end); } @@ -43,8 +43,7 @@ namespace trackerTFP { } // proper constructor - DataFormats::DataFormats(const ParameterSet& iConfig, const Setup* setup) : DataFormats() { - iConfig_ = iConfig; + DataFormats::DataFormats(const Setup* setup) : DataFormats() { setup_ = setup; fillDataFormats(); for (const Process p : Processes) @@ -57,31 +56,24 @@ namespace trackerTFP { numChannel_[+Process::pp] = setup_->numDTCsPerTFP(); numChannel_[+Process::gp] = setup_->numSectors(); numChannel_[+Process::ht] = setup_->htNumBinsInv2R(); - numChannel_[+Process::mht] = setup_->htNumBinsInv2R(); - numChannel_[+Process::zht] = setup_->htNumBinsInv2R(); - numChannel_[+Process::kfin] = setup_->kfNumWorker() * setup_->numLayers(); + numChannel_[+Process::ctb] = setup_->kfNumWorker(); numChannel_[+Process::kf] = setup_->kfNumWorker(); - transform(numChannel_.begin(), numChannel_.end(), back_inserter(numStreams_), [this](int channel) { - return channel * setup_->numRegions(); - }); - numStreamsStubs_ = numStreams_; - numStreamsStubs_[+Process::kf] = numStreams_[+Process::kfin]; - numStreamsTracks_ = vector(+Process::end, 0); - numStreamsTracks_[+Process::kfin] = numStreams_[+Process::kf]; - numStreamsTracks_[+Process::kf] = numStreams_[+Process::kf]; - // Print digi data format of all variables of any specified algo step - // (Look at DataFormat.h::tracks_ to see variable names). - //for (const Variable v : tracks_[+Process::kf]) { - // const DataFormat& f = format(v, Process::kf); - // cout <<" KF "<< f.base() << " " << f.range() << " " << f.width() << endl; - //} + numChannel_[+Process::dr] = 1; + for (const Process& p : {Process::dtc, Process::pp, Process::gp, Process::ht}) { + numStreamsStubs_.push_back(numChannel_[+p] * setup_->numRegions()); + numStreamsTracks_.push_back(0); + } + for (const Process& p : {Process::ctb, Process::kf, Process::dr}) { + numStreamsTracks_.emplace_back(numChannel_[+p] * setup_->numRegions()); + numStreamsStubs_.emplace_back(numStreamsTracks_.back() * setup_->numLayers()); + } } // constructs data formats of all unique used variables and flavours template void DataFormats::fillDataFormats() { if constexpr (config_[+v][+p] == p) { - dataFormats_.emplace_back(Format(iConfig_, setup_)); + dataFormats_.emplace_back(Format(setup_)); fillFormats(); } if constexpr (++p != Process::end) @@ -100,802 +92,207 @@ namespace trackerTFP { fillFormats(); } - // converts bits to ntuple of variables - template - void DataFormats::convertStub(Process p, const Frame& bv, tuple& data) const { - TTBV ttBV(bv); - extractStub(p, ttBV, data); - } - - // helper (loop) to convert bits to ntuple of variables - template - void DataFormats::extractStub(Process p, TTBV& ttBV, std::tuple& data) const { - Variable v = *next(stubs_[+p].begin(), sizeof...(Ts) - 1 - it); - formats_[+v][+p]->extract(ttBV, get(data)); - if constexpr (it + 1 != sizeof...(Ts)) - extractStub(p, ttBV, data); - } - - // converts ntuple of variables to bits - template - void DataFormats::convertStub(Process p, const std::tuple& data, Frame& bv) const { - TTBV ttBV(1, numUnusedBitsStubs_[+p]); - attachStub(p, data, ttBV); - bv = ttBV.bs(); - } - - // helper (loop) to convert ntuple of variables to bits - template - void DataFormats::attachStub(Process p, const tuple& data, TTBV& ttBV) const { - Variable v = *next(stubs_[+p].begin(), it); - formats_[+v][+p]->attach(get(data), ttBV); - if constexpr (it + 1 != sizeof...(Ts)) - attachStub(p, data, ttBV); - } - - // converts bits to ntuple of variables - template - void DataFormats::convertTrack(Process p, const Frame& bv, tuple& data) const { - TTBV ttBV(bv); - extractTrack(p, ttBV, data); - } - - // helper (loop) to convert bits to ntuple of variables - template - void DataFormats::extractTrack(Process p, TTBV& ttBV, std::tuple& data) const { - Variable v = *next(tracks_[+p].begin(), sizeof...(Ts) - 1 - it); - formats_[+v][+p]->extract(ttBV, get(data)); - if constexpr (it + 1 != sizeof...(Ts)) - extractTrack(p, ttBV, data); - } - - // converts ntuple of variables to bits - template - void DataFormats::convertTrack(Process p, const std::tuple& data, Frame& bv) const { - TTBV ttBV(1, numUnusedBitsTracks_[+p]); - attachTrack(p, data, ttBV); - bv = ttBV.bs(); - } - - // helper (loop) to convert ntuple of variables to bits - template - void DataFormats::attachTrack(Process p, const tuple& data, TTBV& ttBV) const { - Variable v = *next(tracks_[+p].begin(), it); - formats_[+v][+p]->attach(get(data), ttBV); - if constexpr (it + 1 != sizeof...(Ts)) - attachTrack(p, data, ttBV); - } - - // construct Stub from Frame - template - Stub::Stub(const FrameStub& frame, const DataFormats* dataFormats, Process p) - : dataFormats_(dataFormats), p_(p), frame_(frame), trackId_(0) { - dataFormats_->convertStub(p, frame.second, data_); - } - - // construct Stub from other Stub - template - template - Stub::Stub(const Stub& stub, Ts... data) - : dataFormats_(stub.dataFormats()), - p_(++stub.p()), - frame_(stub.frame().first, Frame()), - data_(data...), - trackId_(0) {} - - // construct Stub from TTStubRef - template - Stub::Stub(const TTStubRef& ttStubRef, const DataFormats* dataFormats, Process p, Ts... data) - : dataFormats_(dataFormats), p_(p), frame_(ttStubRef, Frame()), data_(data...), trackId_(0) {} - - // construct StubPP from Frame - StubPP::StubPP(const FrameStub& frame, const DataFormats* formats) : Stub(frame, formats, Process::pp) { - for (int sectorEta = sectorEtaMin(); sectorEta <= sectorEtaMax(); sectorEta++) - for (int sectorPhi = 0; sectorPhi < width(Variable::sectorsPhi); sectorPhi++) - sectors_[sectorEta * width(Variable::sectorsPhi) + sectorPhi] = sectorsPhi()[sectorPhi]; - } - - // construct StubGP from Frame - StubGP::StubGP(const FrameStub& frame, const DataFormats* formats, int sectorPhi, int sectorEta) - : Stub(frame, formats, Process::gp), sectorPhi_(sectorPhi), sectorEta_(sectorEta) { - const Setup* setup = dataFormats_->setup(); - inv2RBins_ = TTBV(0, setup->htNumBinsInv2R()); - for (int inv2R = inv2RMin(); inv2R <= inv2RMax(); inv2R++) - inv2RBins_.set(inv2R + inv2RBins_.size() / 2); - } - - // construct StubGP from StubPP - StubGP::StubGP(const StubPP& stub, int sectorPhi, int sectorEta) - : Stub(stub, stub.r(), stub.phi(), stub.z(), stub.layer(), stub.inv2RMin(), stub.inv2RMax()), - sectorPhi_(sectorPhi), - sectorEta_(sectorEta) { - const Setup* setup = dataFormats_->setup(); - get<1>(data_) -= (sectorPhi_ - .5) * setup->baseSector(); - get<2>(data_) -= (r() + dataFormats_->chosenRofPhi()) * setup->sectorCot(sectorEta_); - dataFormats_->convertStub(p_, data_, frame_.second); - } - - // construct StubHT from Frame - StubHT::StubHT(const FrameStub& frame, const DataFormats* formats, int inv2R) - : Stub(frame, formats, Process::ht), inv2R_(inv2R) { - fillTrackId(); - } - - // construct StubHT from StubGP and HT cell assignment - StubHT::StubHT(const StubGP& stub, int phiT, int inv2R) - : Stub(stub, stub.r(), stub.phi(), stub.z(), stub.layer(), stub.sectorPhi(), stub.sectorEta(), phiT), - inv2R_(inv2R) { - get<1>(data_) -= - format(Variable::inv2R).floating(this->inv2R()) * r() + format(Variable::phiT).floating(this->phiT()); - fillTrackId(); - dataFormats_->convertStub(p_, data_, frame_.second); - } - - void StubHT::fillTrackId() { - TTBV ttBV(bv()); - trackId_ = ttBV.extract(width(Variable::sectorPhi) + width(Variable::sectorEta) + width(Variable::phiT)); - } - - // construct StubMHT from Frame - StubMHT::StubMHT(const FrameStub& frame, const DataFormats* formats) : Stub(frame, formats, Process::mht) { - fillTrackId(); - } - - // construct StubMHT from StubHT and MHT cell assignment - StubMHT::StubMHT(const StubHT& stub, int phiT, int inv2R) - : Stub(stub, - stub.r(), - stub.phi(), - stub.z(), - stub.layer(), - stub.sectorPhi(), - stub.sectorEta(), - stub.phiT(), - stub.inv2R()) { - const Setup* setup = dataFormats_->setup(); - // update track (phIT, inv2R), and phi residuals w.r.t. track, to reflect MHT cell assignment. - get<6>(data_) = this->phiT() * setup->mhtNumBinsPhiT() + phiT; - get<7>(data_) = this->inv2R() * setup->mhtNumBinsInv2R() + inv2R; - get<1>(data_) -= base(Variable::inv2R) * (inv2R - .5) * r() + base(Variable::phiT) * (phiT - .5); - dataFormats_->convertStub(p_, data_, frame_.second); - fillTrackId(); - } - - // fills track id - void StubMHT::fillTrackId() { - TTBV ttBV(bv()); - trackId_ = ttBV.extract(width(Variable::sectorPhi) + width(Variable::sectorEta) + width(Variable::phiT) + - width(Variable::inv2R)); - } - - // construct StubZHT from Frame - StubZHT::StubZHT(const FrameStub& frame, const DataFormats* formats) : Stub(frame, formats, Process::zht) { - cot_ = 0.; - zT_ = 0.; - fillTrackId(); - } - - // construct StubZHT from StubMHT - StubZHT::StubZHT(const StubMHT& stub) - : Stub(stub, - stub.r(), - stub.phi(), - stub.z(), - stub.layer(), - stub.sectorPhi(), - stub.sectorEta(), - stub.phiT(), - stub.inv2R(), - 0, - 0) { - cot_ = 0.; - zT_ = 0.; - r_ = format(Variable::r).digi(this->r() + dataFormats_->chosenRofPhi() - dataFormats_->setup()->chosenRofZ()); - chi_ = stub.z(); - trackId_ = stub.trackId(); - } - - // - StubZHT::StubZHT(const StubZHT& stub, double zT, double cot, int id) - : Stub(stub.frame().first, - stub.dataFormats(), - Process::zht, - stub.r(), - stub.phi(), - stub.z(), - stub.layer(), - stub.sectorPhi(), - stub.sectorEta(), - stub.phiT(), - stub.inv2R(), - stub.zT(), - stub.cot()) { - // update track (zT, cot), and phi residuals w.r.t. track, to reflect ZHT cell assignment. - r_ = stub.r_; - cot_ = stub.cotf() + cot; - zT_ = stub.ztf() + zT; - chi_ = stub.z() - zT_ + r_ * cot_; - get<8>(data_) = format(Variable::zT).integer(zT_); - get<9>(data_) = format(Variable::cot).integer(cot_); - dataFormats_->convertStub(p_, data_, frame_.second); - trackId_ = stub.trackId() * 4 + id; - } - - // - StubZHT::StubZHT(const StubZHT& stub, int cot, int zT) - : Stub(stub.frame().first, - stub.dataFormats(), - Process::zht, - stub.r(), - stub.phi(), - 0., - stub.layer(), - stub.sectorPhi(), - stub.sectorEta(), - stub.phiT(), - stub.inv2R(), - zT, - cot) { - get<2>(data_) = - format(Variable::z) - .digi(stub.z() - (format(Variable::zT).floating(zT) - stub.r_ * format(Variable::cot).floating(cot))); - dataFormats_->convertStub(p_, data_, frame_.second); - fillTrackId(); - } - - // fills track id - void StubZHT::fillTrackId() { - TTBV ttBV(bv()); - trackId_ = ttBV.extract(width(Variable::sectorPhi) + width(Variable::sectorEta) + width(Variable::phiT) + - width(Variable::inv2R) + width(Variable::zT) + width(Variable::cot)); - } - - // construct StubKFin from Frame - StubKFin::StubKFin(const FrameStub& frame, const DataFormats* formats, int layer) - : Stub(frame, formats, Process::kfin), layer_(layer) {} - - // construct StubKFin from StubZHT - StubKFin::StubKFin(const StubZHT& stub, double dPhi, double dZ, int layer) - : Stub(stub, stub.r(), stub.phi(), stub.z(), dPhi, dZ), layer_(layer) { - dataFormats_->convertStub(p_, data_, frame_.second); - } - - // construct StubKFin from TTStubRef - StubKFin::StubKFin(const TTStubRef& ttStubRef, - const DataFormats* dataFormats, - double r, - double phi, - double z, - double dPhi, - double dZ, - int layer) - : Stub(ttStubRef, dataFormats, Process::kfin, r, phi, z, dPhi, dZ), layer_(layer) { - dataFormats_->convertStub(p_, data_, frame_.second); - } - - // construct StubKF from Frame - StubKF::StubKF(const FrameStub& frame, const DataFormats* formats, int layer) - : Stub(frame, formats, Process::kf), layer_(layer) {} - - // construct StubKF from StubKFin - StubKF::StubKF(const StubKFin& stub, double inv2R, double phiT, double cot, double zT) - : Stub(stub, stub.r(), 0., 0., stub.dPhi(), stub.dZ()), layer_(stub.layer()) { - const Setup* setup = dataFormats_->setup(); - get<1>(data_) = format(Variable::phi).digi(stub.phi() - (phiT + this->r() * inv2R)); - const double d = - (dataFormats_->hybrid() ? setup->hybridChosenRofPhi() : setup->chosenRofPhi()) - setup->chosenRofZ(); - const double rz = format(Variable::r).digi(this->r() + d); - get<2>(data_) = format(Variable::z).digi(stub.z() - (zT + rz * cot)); - dataFormats_->convertStub(p_, data_, frame_.second); - } - - // construct Track from Frame - template - Track::Track(const FrameTrack& frame, const DataFormats* dataFormats, Process p) - : dataFormats_(dataFormats), p_(p), frame_(frame) { - dataFormats_->convertTrack(p_, frame.second, data_); - } - - // construct Track from other Track - template - template - Track::Track(const Track& track, Ts... data) - : dataFormats_(track.dataFormats()), p_(++track.p()), frame_(track.frame().first, Frame()), data_(data...) {} - - // construct Track from Stub - template - template - Track::Track(const Stub& stub, const TTTrackRef& ttTrackRef, Ts... data) - : dataFormats_(stub.dataFormats()), p_(++stub.p()), frame_(ttTrackRef, Frame()), data_(data...) {} - - // construct Track from TTTrackRef - template - Track::Track(const TTTrackRef& ttTrackRef, const DataFormats* dataFormats, Process p, Ts... data) - : dataFormats_(dataFormats), p_(p), frame_(ttTrackRef, Frame()), data_(data...) {} - - // construct TrackKFin from Frame - TrackKFin::TrackKFin(const FrameTrack& frame, const DataFormats* dataFormats, const vector& stubs) - : Track(frame, dataFormats, Process::kfin), stubs_(setup()->numLayers()), hitPattern_(0, setup()->numLayers()) { - vector nStubs(stubs_.size(), 0); - for (StubKFin* stub : stubs) - nStubs[stub->layer()]++; - for (int layer = 0; layer < dataFormats->setup()->numLayers(); layer++) - stubs_[layer].reserve(nStubs[layer]); - for (StubKFin* stub : stubs) { - const int layer = stub->layer(); - stubs_[layer].push_back(stub); - hitPattern_.set(layer); - } - } - - // construct TrackKFin from StubZHT - TrackKFin::TrackKFin(const StubZHT& stub, const TTTrackRef& ttTrackRef, const TTBV& maybePattern) - : Track(stub, ttTrackRef, maybePattern, stub.sectorPhi(), stub.sectorEta(), 0., 0., 0., 0.), - stubs_(setup()->numLayers()), - hitPattern_(0, setup()->numLayers()) { - get<3>(data_) = format(Variable::phiT, Process::mht).floating(stub.phiT()); - get<4>(data_) = format(Variable::inv2R, Process::mht).floating(stub.inv2R()); - get<5>(data_) = format(Variable::zT, Process::zht).floating(stub.zT()); - get<6>(data_) = format(Variable::cot, Process::zht).floating(stub.cot()); - dataFormats_->convertTrack(p_, data_, frame_.second); - } - - // construct TrackKFin from TTTrackRef - TrackKFin::TrackKFin(const TTTrackRef& ttTrackRef, - const DataFormats* dataFormats, - const TTBV& maybePattern, - double phiT, - double inv2R, - double zT, - double cot, - int sectorPhi, - int sectorEta) - : Track(ttTrackRef, dataFormats, Process::kfin, maybePattern, sectorPhi, sectorEta, phiT, inv2R, zT, cot), - stubs_(setup()->numLayers()), - hitPattern_(0, setup()->numLayers()) { - dataFormats_->convertTrack(p_, data_, frame_.second); - } - - vector TrackKFin::ttStubRefs(const TTBV& hitPattern, const vector& layerMap) const { - vector stubs; - stubs.reserve(hitPattern.count()); - for (int layer = 0; layer < setup()->numLayers(); layer++) - if (hitPattern[layer]) - stubs.push_back(stubs_[layer][layerMap[layer]]->ttStubRef()); - return stubs; - } - - // construct TrackKF from Frame - TrackKF::TrackKF(const FrameTrack& frame, const DataFormats* dataFormats) : Track(frame, dataFormats, Process::kf) {} - - // construct TrackKF from TrackKfin - TrackKF::TrackKF(const TrackKFin& track, double phiT, double inv2R, double zT, double cot) - : Track( - track, false, track.sectorPhi(), track.sectorEta(), track.phiT(), track.inv2R(), track.cot(), track.zT()) { - get<0>(data_) = abs(inv2R) < dataFormats_->format(Variable::inv2R, Process::zht).base() / 2. && - abs(phiT) < dataFormats_->format(Variable::phiT, Process::zht).base() / 2.; - get<3>(data_) += phiT; - get<4>(data_) += inv2R; - get<5>(data_) += cot; - get<6>(data_) += zT; - dataFormats_->convertTrack(p_, data_, frame_.second); - } - - // conversion to TTTrack with given stubs - TTTrack TrackKF::ttTrack(const vector& stubs) const { - const double invR = -this->inv2R() * 2.; - const double phi0 = - deltaPhi(this->phiT() - this->inv2R() * dataFormats_->chosenRofPhi() + - setup()->baseSector() * (this->sectorPhi() - .5) + setup()->baseRegion() * frame_.first->phiSector()); - const double cot = this->cot() + setup()->sectorCot(this->sectorEta()); - const double z0 = this->zT() - this->cot() * setup()->chosenRofZ(); - TTBV hitVector(0, setup()->numLayers()); - double chi2phi(0.); - double chi2z(0.); - vector ttStubRefs; - const int nLayer = stubs.size(); - ttStubRefs.reserve(nLayer); - for (const StubKF& stub : stubs) { - hitVector.set(stub.layer()); - const TTStubRef& ttStubRef = stub.ttStubRef(); - chi2phi += pow(stub.phi(), 2) / setup()->v0(ttStubRef, this->inv2R()); - chi2z += pow(stub.z(), 2) / setup()->v1(ttStubRef, cot); - ttStubRefs.push_back(ttStubRef); - } - static constexpr int nPar = 4; - static constexpr double d0 = 0.; - static constexpr double trkMVA1 = 0.; - static constexpr double trkMVA2 = 0.; - static constexpr double trkMVA3 = 0.; - const int hitPattern = hitVector.val(); - const double bField = setup()->bField(); - TTTrack ttTrack( - invR, phi0, cot, z0, d0, chi2phi, chi2z, trkMVA1, trkMVA2, trkMVA3, hitPattern, nPar, bField); - ttTrack.setStubRefs(ttStubRefs); - ttTrack.setPhiSector(frame_.first->phiSector()); - ttTrack.setEtaSector(this->sectorEta()); - ttTrack.setTrackSeedType(frame_.first->trackSeedType()); - ttTrack.setStubPtConsistency(StubPtConsistency::getConsistency( - ttTrack, setup()->trackerGeometry(), setup()->trackerTopology(), bField, nPar)); - return ttTrack; - } - - // construct TrackDR from Frame - TrackDR::TrackDR(const FrameTrack& frame, const DataFormats* dataFormats) : Track(frame, dataFormats, Process::dr) {} - - // construct TrackDR from TrackKF - TrackDR::TrackDR(const TrackKF& track) : Track(track, 0., 0., 0., 0.) { - get<0>(data_) = track.phiT() + track.inv2R() * dataFormats_->chosenRofPhi() + - dataFormats_->format(Variable::phi, Process::gp).range() * (track.sectorPhi() - .5); - get<1>(data_) = track.inv2R(); - get<2>(data_) = track.zT() - track.cot() * setup()->chosenRofZ(); - get<3>(data_) = track.cot() + setup()->sectorCot(track.sectorEta()); - dataFormats_->convertTrack(p_, data_, frame_.second); - } - - // conversion to TTTrack - TTTrack TrackDR::ttTrack() const { - const double inv2R = this->inv2R(); - const double phi0 = this->phi0(); - const double cot = this->cot(); - const double z0 = this->z0(); - static constexpr double d0 = 0.; - static constexpr double chi2phi = 0.; - static constexpr double chi2z = 0; - static constexpr double trkMVA1 = 0.; - static constexpr double trkMVA2 = 0.; - static constexpr double trkMVA3 = 0.; - static constexpr int hitPattern = 0.; - static constexpr int nPar = 4; - static constexpr double bField = 0.; - const int sectorPhi = frame_.first->phiSector(); - const int sectorEta = frame_.first->etaSector(); - TTTrack ttTrack( - inv2R, phi0, cot, z0, d0, chi2phi, chi2z, trkMVA1, trkMVA2, trkMVA3, hitPattern, nPar, bField); - ttTrack.setPhiSector(sectorPhi); - ttTrack.setEtaSector(sectorEta); - return ttTrack; - } - template <> - Format::Format(const ParameterSet& iConfig, const Setup* setup) : DataFormat(true) { - range_ = 2. * M_PI / (double)(setup->numRegions() * setup->numSectorsPhi()); - base_ = range_ / (double)setup->htNumBinsPhiT(); - width_ = ceil(log2(setup->htNumBinsPhiT())); + Format::Format(const Setup* setup) : DataFormat(true) { + width_ = TTTrack_TrackWord::TrackBitWidths::kRinvSize; + range_ = -2. * TTTrack_TrackWord::minRinv; + base_ = range_ * pow(2, -width_); } - template <> - Format::Format(const ParameterSet& iConfig, const Setup* setup) : DataFormat(true) { - const Format ht(iConfig, setup); - range_ = ht.range(); - base_ = ht.base() / setup->mhtNumBinsPhiT(); - width_ = ceil(log2(setup->htNumBinsPhiT() * setup->mhtNumBinsPhiT())); + Format::Format(const Setup* setup) : DataFormat(true) { + width_ = TTTrack_TrackWord::TrackBitWidths::kPhiSize; + range_ = -2. * TTTrack_TrackWord::minPhi0; + base_ = range_ * pow(2, -width_); } - template <> - Format::Format(const ParameterSet& iConfig, const Setup* setup) : DataFormat(true) { - const double mintPt = iConfig.getParameter("UseHybrid") ? setup->hybridMinPtCand() : setup->minPt(); - range_ = 2. * setup->invPtToDphi() / mintPt; - base_ = range_ / (double)setup->htNumBinsInv2R(); - width_ = ceil(log2(setup->htNumBinsInv2R())); + Format::Format(const Setup* setup) : DataFormat(true) { + width_ = TTTrack_TrackWord::TrackBitWidths::kTanlSize; + range_ = -2. * TTTrack_TrackWord::minTanl; + base_ = range_ * pow(2, -width_); } - template <> - Format::Format(const ParameterSet& iConfig, const Setup* setup) : DataFormat(true) { - const Format ht(iConfig, setup); - range_ = ht.range(); - base_ = ht.base() / setup->mhtNumBinsInv2R(); - width_ = ceil(log2(setup->htNumBinsInv2R() * setup->mhtNumBinsInv2R())); + Format::Format(const Setup* setup) : DataFormat(true) { + width_ = TTTrack_TrackWord::TrackBitWidths::kZ0Size; + range_ = -2. * TTTrack_TrackWord::minZ0; + base_ = range_ * pow(2, -width_); } template <> - Format::Format(const ParameterSet& iConfig, const Setup* setup) : DataFormat(true) { - const double chosenRofPhi = - iConfig.getParameter("UseHybrid") ? setup->hybridChosenRofPhi() : setup->chosenRofPhi(); + Format::Format(const Setup* setup) : DataFormat(true) { + const Format phiT(setup); + const Format inv2R(setup); width_ = setup->tmttWidthR(); - range_ = 2. * max(abs(setup->outerRadius() - chosenRofPhi), abs(setup->innerRadius() - chosenRofPhi)); - const Format phiT(iConfig, setup); - const Format inv2R(iConfig, setup); + range_ = 2. * setup->maxRphi(); base_ = phiT.base() / inv2R.base(); - const int shift = ceil(log2(range_ / base_ / pow(2., width_))); + const int shift = ceil(log2(range_ / base_)) - width_; base_ *= pow(2., shift); } - - template <> - Format::Format(const ParameterSet& iConfig, const Setup* setup) : DataFormat(true) { - const Format phiT(iConfig, setup); - const Format inv2R(iConfig, setup); - const Format r(iConfig, setup); - range_ = phiT.range() + inv2R.range() * r.base() * pow(2., r.width()) / 4.; - const Format dtc(iConfig, setup); - base_ = dtc.base(); - width_ = ceil(log2(range_ / base_)); - } - template <> - Format::Format(const ParameterSet& iConfig, const Setup* setup) : DataFormat(true) { + Format::Format(const Setup* setup) : DataFormat(true) { + const Format phiT(setup); + const Format inv2R(setup); width_ = setup->tmttWidthPhi(); - const Format phiT(iConfig, setup); - const Format inv2R(iConfig, setup); - const Format r(iConfig, setup); - range_ = 2. * M_PI / (double)setup->numRegions() + inv2R.range() * r.base() * pow(2., r.width()) / 4.; - const int shift = ceil(log2(range_ / phiT.base() / pow(2., width_))); + range_ = phiT.range() + inv2R.range() * setup->maxRphi(); + const int shift = ceil(log2(range_ / phiT.base())) - width_; base_ = phiT.base() * pow(2., shift); } - template <> - Format::Format(const ParameterSet& iConfig, const Setup* setup) : DataFormat(true) { - const Format phiT(iConfig, setup); - range_ = 2. * phiT.base(); - const Format gp(iConfig, setup); - base_ = gp.base(); - width_ = ceil(log2(range_ / base_)); + Format::Format(const Setup* setup) : DataFormat(true) { + const Format zT(setup); + width_ = setup->tmttWidthZ(); + range_ = 2. * setup->halfLength(); + const int shift = ceil(log2(range_ / zT.base())) - width_; + base_ = zT.base() * pow(2., shift); } - template <> - Format::Format(const ParameterSet& iConfig, const Setup* setup) : DataFormat(true) { - const Format phiT(iConfig, setup); - range_ = 2. * phiT.base(); - const Format ht(iConfig, setup); - base_ = ht.base(); - width_ = ceil(log2(range_ / base_)); + Format::Format(const Setup* setup) : DataFormat(false) { + width_ = 5; } template <> - Format::Format(const ParameterSet& iConfig, const Setup* setup) : DataFormat(true) { - const Format phi(iConfig, setup); - const double rangeFactor = iConfig.getParameter("KalmanFilter").getParameter("RangeFactor"); - range_ = rangeFactor * phi.range(); + Format::Format(const Setup* setup) : DataFormat(true) { + const Format phi(setup); + const Format inv2R(setup); + const Format phiT(setup); base_ = phi.base(); + range_ = phiT.base() + inv2R.range() * setup->maxRphi(); width_ = ceil(log2(range_ / base_)); } - template <> - Format::Format(const ParameterSet& iConfig, const Setup* setup) : DataFormat(true) { - width_ = setup->tmttWidthZ(); - range_ = 2. * setup->halfLength(); - const Format r(iConfig, setup); - const int shift = ceil(log2(range_ / r.base())) - width_; - base_ = r.base() * pow(2., shift); + Format::Format(const Setup* setup) : DataFormat(true) { + const Format z(setup); + const Format zT(setup); + const double rangeCot = (zT.base() + 2. * setup->beamWindowZ()) / setup->chosenRofZ(); + base_ = z.base(); + range_ = zT.base() + rangeCot * setup->maxRz(); + width_ = ceil(log2(range_ / base_)); } - template <> - Format::Format(const ParameterSet& iConfig, const Setup* setup) : DataFormat(true) { - range_ = setup->neededRangeChiZ(); - const Format dtc(iConfig, setup); - base_ = dtc.base(); - width_ = ceil(log2(range_ / base_)); + Format::Format(const Setup* setup) : DataFormat(true) { + range_ = 2. * M_PI / setup->numRegions(); + width_ = ceil(log2(setup->gpNumBinsPhiT())); + base_ = range_ / pow(2., width_); } - template <> - Format::Format(const ParameterSet& iConfig, const Setup* setup) : DataFormat(true) { - const int numBinsZT = iConfig.getParameter("ZHoughTransform").getParameter("NumBinsZT"); - const int numStages = iConfig.getParameter("ZHoughTransform").getParameter("NumStages"); - width_ = ceil(log2(pow(numBinsZT, numStages))); - const Format z(iConfig, setup); - range_ = -1.; - for (int eta = 0; eta < setup->numSectorsEta(); eta++) - range_ = max(range_, (sinh(setup->boundarieEta(eta + 1)) - sinh(setup->boundarieEta(eta)))); - range_ *= setup->chosenRofZ(); - const int shift = ceil(log2(range_ / z.base() / pow(2., width_))); - base_ = z.base() * pow(2., shift); + Format::Format(const Setup* setup) : DataFormat(true) { + range_ = 2. * sinh(setup->maxEta()) * setup->chosenRofZ(); + base_ = range_ / setup->gpNumBinsZT(); + width_ = ceil(log2(setup->gpNumBinsZT())); } - template <> - Format::Format(const ParameterSet& iConfig, const Setup* setup) : DataFormat(true) { - const int numBinsCot = iConfig.getParameter("ZHoughTransform").getParameter("NumBinsCot"); - const int numStages = iConfig.getParameter("ZHoughTransform").getParameter("NumStages"); - width_ = ceil(log2(pow(numBinsCot, numStages))); - const Format zT(iConfig, setup); - range_ = (zT.range() + 2. * setup->beamWindowZ()) / setup->chosenRofZ(); - const int shift = ceil(log2(range_)) - width_; - base_ = pow(2., shift); + Format::Format(const Setup* setup) : DataFormat(true) { + const Format zT(setup); + const Format r(setup); + width_ = setup->widthDSPbb(); + range_ = (zT.range() - zT.base() + 2. * setup->beamWindowZ()) / setup->chosenRofZ(); + base_ = zT.base() / r.base(); + const int baseShift = ceil(log2(range_ / base_)) - width_; + base_ *= pow(2, baseShift); } - template <> - Format::Format(const ParameterSet& iConfig, const Setup* setup) : DataFormat(true) { - const Format zT(iConfig, setup); - const Format cot(iConfig, setup); - const double rangeR = - 2. * max(abs(setup->outerRadius() - setup->chosenRofZ()), abs(setup->innerRadius() - setup->chosenRofZ())); - range_ = zT.base() + cot.base() * rangeR + setup->maxdZ(); - const Format dtc(iConfig, setup); - base_ = dtc.base(); - width_ = ceil(log2(range_ / base_)); - /*const Format z(iConfig, setup); - width_ = z.width(); - range_ = z.range(); - base_ = z.base();*/ + Format::Format(const Setup* setup) : DataFormat(false) { + width_ = 6; } template <> - Format::Format(const ParameterSet& iConfig, const Setup* setup) : DataFormat(true) { - const Format zht(iConfig, setup); - range_ = zht.range() * pow(2, setup->kfinShiftRangeZ()); - base_ = zht.base(); + Format::Format(const Setup* setup) : DataFormat(true) { + const Format phi(setup); + const Format phiT(setup); + const Format inv2R(setup); + range_ = phiT.base() + setup->maxRphi() * inv2R.base(); + base_ = phi.base(); width_ = ceil(log2(range_ / base_)); } - template <> - Format::Format(const ParameterSet& iConfig, const Setup* setup) : DataFormat(true) { - const Format phiT(iConfig, setup); - const Format inv2R(iConfig, setup); - const double chosenRofPhi = - iConfig.getParameter("UseHybrid") ? setup->hybridChosenRofPhi() : setup->chosenRofPhi(); - const double rangeR = 2. * max(abs(setup->outerRadius() - chosenRofPhi), abs(setup->innerRadius() - chosenRofPhi)); - range_ = phiT.base() + inv2R.base() * rangeR + setup->maxdPhi(); - const Format dtc(iConfig, setup); - base_ = dtc.base(); + Format::Format(const Setup* setup) : DataFormat(true) { + range_ = 2. * setup->invPtToDphi() / setup->minPt(); + base_ = range_ / (double)setup->htNumBinsInv2R(); + width_ = ceil(log2(setup->htNumBinsInv2R())); + } + template <> + Format::Format(const Setup* setup) : DataFormat(true) { + const Format phiT(setup); + range_ = phiT.range(); + base_ = phiT.base() / (double)setup->htNumBinsPhiT(); width_ = ceil(log2(range_ / base_)); } template <> - Format::Format(const ParameterSet& iConfig, const Setup* setup) : DataFormat(true) { - const Format zht(iConfig, setup); - range_ = zht.range() * pow(2, setup->kfinShiftRangePhi()); - base_ = zht.base(); + Format::Format(const Setup* setup) : DataFormat(false) { + const Format phi(setup); + const Format inv2R(setup); + const double sigma = setup->pitchRowPS() / 2. / setup->innerRadius(); + const double pt = (setup->pitchCol2S() + setup->scattering()) / 2. * inv2R.range() / 2.; + range_ = sigma + pt; + base_ = phi.base(); width_ = ceil(log2(range_ / base_)); } - template <> - Format::Format(const ParameterSet& iConfig, const Setup* setup) : DataFormat(true) { - /*const Format z(iConfig, setup); - const double rangeFactor = iConfig.getParameter("KalmanFilter").getParameter("RangeFactor"); - range_ = rangeFactor * z.range(); + Format::Format(const Setup* setup) : DataFormat(false) { + const Format z(setup); + range_ = setup->pitchCol2S() / 2. * sinh(setup->maxEta()); base_ = z.base(); - width_ = ceil(log2(range_ / base_));*/ - const Format zT(iConfig, setup); - const Format cot(iConfig, setup); - const double rangeR = - 2. * max(abs(setup->outerRadius() - setup->chosenRofZ()), abs(setup->innerRadius() - setup->chosenRofZ())); - range_ = zT.base() + cot.base() * rangeR + setup->maxdZ(); - const Format dtc(iConfig, setup); - base_ = dtc.base(); - const double rangeFactor = iConfig.getParameter("KalmanFilter").getParameter("RangeFactor"); - range_ *= rangeFactor; width_ = ceil(log2(range_ / base_)); } - template <> - Format::Format(const ParameterSet& iConfig, const Setup* setup) : DataFormat(false) { + Format::Format(const Setup* setup) : DataFormat(false) { range_ = setup->numLayers(); width_ = ceil(log2(range_)); } template <> - Format::Format(const ParameterSet& iConfig, const Setup* setup) - : DataFormat(false) { - range_ = setup->numSectorsEta(); - width_ = ceil(log2(range_)); - } - - template <> - Format::Format(const ParameterSet& iConfig, const Setup* setup) - : DataFormat(false) { - range_ = setup->numSectorsPhi(); - width_ = ceil(log2(range_)); - } - - template <> - Format::Format(const ParameterSet& iConfig, const Setup* setup) - : DataFormat(false) { - range_ = setup->numSectorsPhi(); - width_ = setup->numSectorsPhi(); - } - - template <> - Format::Format(const edm::ParameterSet& iConfig, const Setup* setup) - : DataFormat(false) { - width_ = 1; - range_ = 1.; - } - - template <> - Format::Format(const edm::ParameterSet& iConfig, const Setup* setup) - : DataFormat(false) { - width_ = setup->numLayers(); - } - - template <> - Format::Format(const edm::ParameterSet& iConfig, const Setup* setup) : DataFormat(true) { - const Format inv2R(iConfig, setup); - const Format phiT(iConfig, setup); - const double chosenRofPhi = - iConfig.getParameter("UseHybrid") ? setup->hybridChosenRofPhi() : setup->chosenRofPhi(); - width_ = setup->tfpWidthPhi0(); - range_ = 2. * M_PI / (double)setup->numRegions() + inv2R.range() * chosenRofPhi; - base_ = phiT.base(); - const int shift = ceil(log2(range_ / base_ / pow(2., width_))); - base_ *= pow(2., shift); - } - - template <> - Format::Format(const edm::ParameterSet& iConfig, const Setup* setup) - : DataFormat(true) { - const Format inv2R(iConfig, setup); - width_ = setup->tfpWidthInv2R(); - range_ = inv2R.range(); - base_ = inv2R.base(); - const int shift = ceil(log2(range_ / base_ / pow(2., width_))); - base_ *= pow(2., shift); - } - - template <> - Format::Format(const edm::ParameterSet& iConfig, const Setup* setup) : DataFormat(true) { - const Format zT(iConfig, setup); - width_ = setup->tfpWidthZ0(); - range_ = 2. * setup->beamWindowZ(); - base_ = zT.base(); - const int shift = ceil(log2(range_ / base_ / pow(2., width_))); - base_ *= pow(2., shift); - } - - template <> - Format::Format(const edm::ParameterSet& iConfig, const Setup* setup) : DataFormat(true) { - const Format cot(iConfig, setup); - width_ = setup->tfpWidthCot(); - range_ = 2. * setup->maxCot(); - base_ = cot.base(); - const int shift = ceil(log2(range_ / base_ / pow(2., width_))); - base_ *= pow(2., shift); - } - - template <> - Format::Format(const edm::ParameterSet& iConfig, const Setup* setup) : DataFormat(true) { - const Format phi0(iConfig, setup); - const Format phiT(iConfig, setup); - const double rangeFactor = iConfig.getParameter("KalmanFilter").getParameter("RangeFactor"); - range_ = rangeFactor * phiT.range(); - base_ = phi0.base(); + Format::Format(const Setup* setup) : DataFormat(true) { + const Format tfp(setup); + const Format ht(setup); + range_ = ht.range() + 2. * ht.base(); + base_ = ht.base() * pow(2., floor(log2(.5 * tfp.base() / ht.base()))); width_ = ceil(log2(range_ / base_)); } - template <> - Format::Format(const edm::ParameterSet& iConfig, const Setup* setup) - : DataFormat(true) { - const Format dr(iConfig, setup); - const Format mht(iConfig, setup); - const double rangeFactor = iConfig.getParameter("KalmanFilter").getParameter("RangeFactor"); - range_ = mht.range() + rangeFactor * mht.base(); - base_ = dr.base(); + Format::Format(const Setup* setup) : DataFormat(true) { + const Format tfp(setup); + const Format ht(setup); + range_ = ht.range() + 2. * ht.base(); + base_ = ht.base() * pow(2., floor(log2(tfp.base() / ht.base()))); width_ = ceil(log2(range_ / base_)); } - template <> - Format::Format(const edm::ParameterSet& iConfig, const Setup* setup) : DataFormat(true) { - const Format z0(iConfig, setup); - const Format zT(iConfig, setup); - const double rangeFactor = iConfig.getParameter("KalmanFilter").getParameter("RangeFactor"); - range_ = zT.range() + rangeFactor * zT.base(); - base_ = z0.base(); + Format::Format(const Setup* setup) : DataFormat(true) { + const Format tfp(setup); + const Format zT(setup); + const Format r(setup); + range_ = (zT.base() + 2. * setup->beamWindowZ()) / setup->chosenRofZ(); + base_ = zT.base() / r.base() * pow(2., floor(log2(tfp.base() / zT.base() * r.base()))); width_ = ceil(log2(range_ / base_)); } - template <> - Format::Format(const edm::ParameterSet& iConfig, const Setup* setup) : DataFormat(true) { - const Format dr(iConfig, setup); - const Format zht(iConfig, setup); - const double rangeFactor = iConfig.getParameter("KalmanFilter").getParameter("RangeFactor"); - range_ = zht.range() + rangeFactor * zht.base(); - base_ = dr.base(); + Format::Format(const Setup* setup) : DataFormat(true) { + const Format tfp(setup); + const Format gp(setup); + range_ = gp.range(); + base_ = gp.base() * pow(2., floor(log2(tfp.base() / gp.base()))); width_ = ceil(log2(range_ / base_)); } - template <> - Format::Format(const edm::ParameterSet& iConfig, const Setup* setup) - : DataFormat(false) { - const Format phi(iConfig, setup); - range_ = setup->maxdPhi(); + Format::Format(const Setup* setup) : DataFormat(true) { + const Format phi(setup); + const Format phiT(setup); + const Format inv2R(setup); + range_ = phiT.base() + setup->maxRphi() * inv2R.base(); base_ = phi.base(); width_ = ceil(log2(range_ / base_)); } + template <> + Format::Format(const Setup* setup) : DataFormat(false) { + width_ = 1; + } template <> - Format::Format(const edm::ParameterSet& iConfig, const Setup* setup) - : DataFormat(false) { - const Format z(iConfig, setup); - range_ = setup->maxdZ(); - base_ = z.base(); + Format::Format(const Setup* setup) : DataFormat(true) { + const Format kf(setup); + const Format zT(setup); + range_ = (zT.range() + 2. * setup->beamWindowZ()) / setup->chosenRofZ(); + base_ = kf.base(); width_ = ceil(log2(range_ / base_)); } diff --git a/L1Trigger/TrackerTFP/src/Demonstrator.cc b/L1Trigger/TrackerTFP/src/Demonstrator.cc index 86a6e5d727246..a1c6433f60d76 100644 --- a/L1Trigger/TrackerTFP/src/Demonstrator.cc +++ b/L1Trigger/TrackerTFP/src/Demonstrator.cc @@ -1,9 +1,11 @@ #include "L1Trigger/TrackerTFP/interface/Demonstrator.h" #include +#include #include #include #include +#include using namespace std; using namespace edm; @@ -14,54 +16,79 @@ namespace trackerTFP { Demonstrator::Demonstrator(const ParameterSet& iConfig, const Setup* setup) : dirIPBB_(iConfig.getParameter("DirIPBB")), runTime_(iConfig.getParameter("RunTime")), + linkMappingIn_(iConfig.getParameter>("LinkMappingIn")), + linkMappingOut_(iConfig.getParameter>("LinkMappingOut")), dirIn_(dirIPBB_ + "in.txt"), dirOut_(dirIPBB_ + "out.txt"), dirPre_(dirIPBB_ + "pre.txt"), dirDiff_(dirIPBB_ + "diff.txt"), - numFrames_(setup->numFramesIO()), + numFrames_(setup->numFramesIOHigh()), numFramesInfra_(setup->numFramesInfra()), numRegions_(setup->numRegions()) {} // plays input through modelsim and compares result with output bool Demonstrator::analyze(const vector>& input, const vector>& output) const { - stringstream ss; + // default link mapping + auto linkMapping = [this](const vector& mapC, vector& map, const vector>& data) { + if (mapC.empty()) { + map.resize((int)data.size() / numRegions_); + iota(map.begin(), map.end(), 0); + } else + map = mapC; + }; // converts input into stringstream - convert(input, ss); + stringstream ss; + vector map; + linkMapping(linkMappingIn_, map, input); + convert(input, ss, map); // play input through modelsim sim(ss); // converts output into stringstream - convert(output, ss); + map.clear(); + linkMapping(linkMappingOut_, map, output); + convert(output, ss, map); // compares output with modelsim output return compare(ss); } // converts streams of bv into stringstream - void Demonstrator::convert(const vector>& bits, stringstream& ss) const { + void Demonstrator::convert(const vector>& bits, stringstream& ss, const vector& mapping) const { // reset ss ss.str(""); ss.clear(); // number of tranceiver in a quad static constexpr int quad = 4; - const int numChannel = bits.size() / numRegions_; - const int voidChannel = numChannel % quad == 0 ? 0 : quad - numChannel % quad; + set quads; + for (int channel : mapping) + quads.insert(channel / quad); + vector links; + links.reserve(quads.size() * quad); + for (int q : quads) { + const int offset = q * quad; + for (int c = 0; c < quad; c++) + links.push_back(offset + c); + } // start with header - ss << header(numChannel + voidChannel); + ss << header(links); int nFrame(0); // create one packet per region bool first = true; for (int region = 0; region < numRegions_; region++) { - const int offset = region * numChannel; + const int offset = region * mapping.size(); // start with emp 6 frame gap - ss << infraGap(nFrame, numChannel + voidChannel); + ss << infraGap(nFrame, links.size()); for (int frame = 0; frame < numFrames_; frame++) { // write one frame for all channel ss << this->frame(nFrame); - for (int channel = 0; channel < numChannel; channel++) { - const vector& bvs = bits[offset + channel]; - ss << (frame < (int)bvs.size() ? hex(bvs[frame], first) : hex(Frame(), first)); + for (int link : links) { + const auto channel = find(mapping.begin(), mapping.end(), link); + if (channel == mapping.end()) + ss << " 0000 " << string(TTBV::S_ / 4, '0'); + else { + const vector& bvs = bits[offset + distance(mapping.begin(), channel)]; + ss << (frame < (int)bvs.size() ? hex(bvs[frame], first) : hex(Frame(), first)); + } } - for (int channel = 0; channel < voidChannel; channel++) - ss << " 0000 " << string(TTBV::S_ / 4, '0'); ss << endl; first = false; } @@ -107,15 +134,15 @@ namespace trackerTFP { } // creates emp file header - string Demonstrator::header(int numLinks) const { + string Demonstrator::header(const vector& links) const { stringstream ss; // file header - ss << "Board CMSSW" << endl + ss << "Id: CMSSW" << endl << "Metadata: (strobe,) start of orbit, start of packet, end of packet, valid" << endl << endl; // link header ss << " Link "; - for (int link = 0; link < numLinks; link++) + for (int link : links) ss << " " << setfill('0') << setw(3) << link << " "; ss << endl; return ss.str(); diff --git a/L1Trigger/TrackerTFP/src/DuplicateRemoval.cc b/L1Trigger/TrackerTFP/src/DuplicateRemoval.cc new file mode 100644 index 0000000000000..e7393c3fe0ac4 --- /dev/null +++ b/L1Trigger/TrackerTFP/src/DuplicateRemoval.cc @@ -0,0 +1,142 @@ +#include "L1Trigger/TrackerTFP/interface/DuplicateRemoval.h" + +#include +#include +#include +#include +#include +#include +#include +#include + +using namespace std; +using namespace edm; +using namespace tt; + +namespace trackerTFP { + + DuplicateRemoval::DuplicateRemoval(const ParameterSet& iConfig, + const Setup* setup, + const DataFormats* dataFormats, + vector& tracks, + vector& stubs) + : enableTruncation_(iConfig.getParameter("EnableTruncation")), + setup_(setup), + dataFormats_(dataFormats), + tracks_(tracks), + stubs_(stubs) {} + + // fill output products + void DuplicateRemoval::produce(const vector>& tracksIn, + const vector>& stubsIn, + vector>& tracksOut, + vector>& stubsOut) { + static const int numChannel = dataFormats_->numChannel(Process::kf); + static const int numLayers = setup_->numLayers(); + static const int numInv2R = setup_->htNumBinsInv2R() + 2; + static const int numPhiT = setup_->htNumBinsPhiT() * setup_->gpNumBinsPhiT(); + static const int numZT = setup_->gpNumBinsZT(); + int nTracks(0); + for (const vector& tracks : tracksIn) + nTracks += + accumulate(tracks.begin(), tracks.end(), 0, [](int& sum, TrackKF* track) { return sum += track ? 1 : 0; }); + vector tracks; + tracks.reserve(nTracks); + deque stream; + // merge 4 channel to 1 + //for (int channel = 0; channel < numChannel; channel++ ) { + for (int channel = numChannel - 1; channel >= 0; channel--) { + const int offset = channel * numLayers; + const vector& tracksChannel = tracksIn[channel]; + for (int frame = 0; frame < (int)tracksChannel.size(); frame++) { + TrackKF* track = tracksChannel[frame]; + if (!track) { + stream.push_back(nullptr); + continue; + } + vector stubs; + stubs.reserve(numLayers); + for (int layer = 0; layer < numLayers; layer++) + stubs.push_back(stubsIn[offset + layer][frame]); + const bool match = track->match().val(); + const int inv2R = dataFormats_->format(Variable::inv2R, Process::ht).integer(track->inv2R()) + numInv2R / 2; + const int phiT = dataFormats_->format(Variable::phiT, Process::ht).integer(track->phiT()) + numPhiT / 2; + const int zT = dataFormats_->format(Variable::zT, Process::gp).integer(track->zT()) + numZT / 2; + tracks.emplace_back(track, stubs, match, inv2R, phiT, zT); + stream.push_back(&tracks.back()); + } + } + // truncate if desired + if (enableTruncation_ && (int)stream.size() > setup_->numFramesHigh()) { + const auto limit = next(stream.begin(), setup_->numFramesHigh()); + stream.erase(limit, stream.end()); + } + // remove duplicates + vector killed; + killed.reserve(stream.size()); + vector> hits(numZT, vector(numInv2R, TTBV(0, numPhiT))); + for (Track*& track : stream) { + if (!track) + continue; + if (track->match_) { + hits[track->zT_][track->inv2R_].set(track->phiT_); + } else { + killed.push_back(track); + track = nullptr; + } + } + // restore duplicates + for (Track* track : killed) { + if (hits[track->zT_][track->inv2R_][track->phiT_]) { + stream.push_back(nullptr); + continue; + } + hits[track->zT_][track->inv2R_].set(track->phiT_); + stream.push_back(track); + } + // truncate + if (enableTruncation_ && (int)stream.size() > setup_->numFramesHigh()) { + const auto limit = next(stream.begin(), setup_->numFramesHigh()); + stream.erase(limit, stream.end()); + } + // remove trailing nullptr + for (auto it = stream.end(); it != stream.begin();) + it = (*--it) ? stream.begin() : stream.erase(it); + // convert and store tracks + tracksOut[0].reserve(stream.size()); + for (vector& layer : stubsOut) + layer.reserve(stream.size()); + for (Track* track : stream) { + if (!track) { + tracksOut[0].push_back(nullptr); + for (vector& layer : stubsOut) + layer.push_back(nullptr); + continue; + } + static const DataFormat& gp = dataFormats_->format(Variable::zT, Process::gp); + TrackKF* trackKF = track->track_; + const double inv2R = trackKF->inv2R(); + const double phiT = trackKF->phiT(); + const double zT = trackKF->zT(); + const double cot = trackKF->cot() + gp.digi(zT) / setup_->chosenRofZ(); + tracks_.emplace_back(*trackKF, inv2R, phiT, cot, zT); + tracksOut[0].push_back(&tracks_.back()); + for (int layer = 0; layer < numLayers; layer++) { + vector& layerStubs = stubsOut[layer]; + StubKF* stub = track->stubs_[layer]; + if (!stub) { + layerStubs.push_back(nullptr); + continue; + } + const double r = stub->r(); + const double phi = stub->phi(); + const double z = stub->z(); + const double dPhi = stub->dPhi(); + const double dZ = stub->dZ(); + stubs_.emplace_back(*stub, r, phi, z, dPhi, dZ); + layerStubs.push_back(&stubs_.back()); + } + } + } + +} // namespace trackerTFP \ No newline at end of file diff --git a/L1Trigger/TrackerTFP/src/ES_KalmanFilterFormats.cc b/L1Trigger/TrackerTFP/src/ES_KalmanFilterFormats.cc deleted file mode 100644 index f6bd56a4e42b9..0000000000000 --- a/L1Trigger/TrackerTFP/src/ES_KalmanFilterFormats.cc +++ /dev/null @@ -1,4 +0,0 @@ -#include "FWCore/Utilities/interface/typelookup.h" -#include "L1Trigger/TrackerTFP/interface/KalmanFilterFormats.h" - -TYPELOOKUP_DATA_REG(trackerTFP::KalmanFilterFormats); diff --git a/L1Trigger/TrackerTFP/src/ES_TrackQuality.cc b/L1Trigger/TrackerTFP/src/ES_TrackQuality.cc new file mode 100644 index 0000000000000..ce4c6a786d677 --- /dev/null +++ b/L1Trigger/TrackerTFP/src/ES_TrackQuality.cc @@ -0,0 +1,5 @@ +#include "FWCore/Utilities/interface/typelookup.h" +#include "L1Trigger/TrackerTFP/interface/TrackQuality.h" + +TYPELOOKUP_DATA_REG(trackerTFP::TrackQuality); +TYPELOOKUP_DATA_REG(trackerTFP::TrackQualityRcd); diff --git a/L1Trigger/TrackerTFP/src/GeometricProcessor.cc b/L1Trigger/TrackerTFP/src/GeometricProcessor.cc index fd0ffaa44a084..f17d9177ade63 100644 --- a/L1Trigger/TrackerTFP/src/GeometricProcessor.cc +++ b/L1Trigger/TrackerTFP/src/GeometricProcessor.cc @@ -15,108 +15,108 @@ namespace trackerTFP { GeometricProcessor::GeometricProcessor(const ParameterSet& iConfig, const Setup* setup, const DataFormats* dataFormats, - int region) + const LayerEncoding* layerEncoding, + std::vector& stubs) : enableTruncation_(iConfig.getParameter("EnableTruncation")), setup_(setup), dataFormats_(dataFormats), - region_(region), - input_(dataFormats_->numChannel(Process::gp), vector>(dataFormats_->numChannel(Process::pp))) {} - - // read in and organize input product (fill vector input_) - void GeometricProcessor::consume(const TTDTC& ttDTC) { - auto validFrame = [](int sum, const FrameStub& frame) { return sum + (frame.first.isNonnull() ? 1 : 0); }; - int nStubsPP(0); - for (int channel = 0; channel < dataFormats_->numChannel(Process::pp); channel++) { - const StreamStub& stream = ttDTC.stream(region_, channel); - nStubsPP += accumulate(stream.begin(), stream.end(), 0, validFrame); - } - stubsPP_.reserve(nStubsPP); - for (int channel = 0; channel < dataFormats_->numChannel(Process::pp); channel++) { - for (const FrameStub& frame : ttDTC.stream(region_, channel)) { - StubPP* stub = nullptr; - if (frame.first.isNonnull()) { - stubsPP_.emplace_back(frame, dataFormats_); - stub = &stubsPP_.back(); - } - for (int sector = 0; sector < dataFormats_->numChannel(Process::gp); sector++) - // adding gaps (nullptr) if no stub available or not in sector to emulate f/w - input_[sector][channel].push_back(stub && stub->inSector(sector) ? stub : nullptr); - } - } - // remove all gaps between end and last stub - for (vector>& input : input_) - for (deque& stubs : input) - for (auto it = stubs.end(); it != stubs.begin();) - it = (*--it) ? stubs.begin() : stubs.erase(it); - auto validStub = [](int sum, StubPP* stub) { return sum + (stub ? 1 : 0); }; - int nStubsGP(0); - for (const vector>& sector : input_) - for (const deque& channel : sector) - nStubsGP += accumulate(channel.begin(), channel.end(), 0, validStub); - stubsGP_.reserve(nStubsGP); - } + layerEncoding_(layerEncoding), + stubs_(stubs) {} // fill output products - void GeometricProcessor::produce(StreamsStub& accepted, StreamsStub& lost) { - for (int sector = 0; sector < dataFormats_->numChannel(Process::gp); sector++) { - vector>& inputs = input_[sector]; - vector> stacks(dataFormats_->numChannel(Process::pp)); - const int sectorPhi = sector % setup_->numSectorsPhi(); - const int sectorEta = sector / setup_->numSectorsPhi(); - auto size = [](int sum, const deque& stubs) { return sum + stubs.size(); }; - const int nStubs = accumulate(inputs.begin(), inputs.end(), 0, size); - vector acceptedSector; - vector lostSector; - acceptedSector.reserve(nStubs); - lostSector.reserve(nStubs); + void GeometricProcessor::produce(const vector>& streamsIn, vector>& streamsOut) { + static const int numChannelIn = dataFormats_->numChannel(Process::pp); + static const int numChannelOut = dataFormats_->numChannel(Process::gp); + for (int channelOut = 0; channelOut < numChannelOut; channelOut++) { + // helper + const int phiT = channelOut % setup_->gpNumBinsPhiT() - setup_->gpNumBinsPhiT() / 2; + const int zT = channelOut / setup_->gpNumBinsPhiT() - setup_->gpNumBinsZT() / 2; + auto valid = [phiT, zT](StubPP* stub) { + const bool phiTValid = stub && phiT >= stub->phiTMin() && phiT <= stub->phiTMax(); + const bool zTValid = stub && zT >= stub->zTMin() && zT <= stub->zTMax(); + return (phiTValid && zTValid) ? stub : nullptr; + }; + // input streams of stubs + vector> inputs(numChannelIn); + for (int channelIn = 0; channelIn < numChannelIn; channelIn++) { + const vector& streamIn = streamsIn[channelIn]; + transform(streamIn.begin(), streamIn.end(), back_inserter(inputs[channelIn]), valid); + } + // fifo for each stream + vector> stacks(streamsIn.size()); + // output stream + deque& output = streamsOut[channelOut]; // clock accurate firmware emulation, each while trip describes one clock tick, one stub in and one stub out per tick while (!all_of(inputs.begin(), inputs.end(), [](const deque& stubs) { return stubs.empty(); }) or !all_of(stacks.begin(), stacks.end(), [](const deque& stubs) { return stubs.empty(); })) { // fill input fifos - for (int channel = 0; channel < dataFormats_->numChannel(Process::pp); channel++) { - deque& stack = stacks[channel]; - StubPP* stub = pop_front(inputs[channel]); + for (int channelIn = 0; channelIn < numChannelIn; channelIn++) { + deque& stack = stacks[channelIn]; + StubPP* stub = pop_front(inputs[channelIn]); if (stub) { - stubsGP_.emplace_back(*stub, sectorPhi, sectorEta); + // convert stub + StubGP* stubGP = produce(*stub, phiT, zT); + // buffer overflow if (enableTruncation_ && (int)stack.size() == setup_->gpDepthMemory() - 1) - lostSector.push_back(pop_front(stack)); - stack.push_back(&stubsGP_.back()); + pop_front(stack); + stack.push_back(stubGP); } } // merge input fifos to one stream, prioritizing higher input channel over lower channel bool nothingToRoute(true); - for (int channel = dataFormats_->numChannel(Process::pp) - 1; channel >= 0; channel--) { - StubGP* stub = pop_front(stacks[channel]); + //for (int channelIn = numChannelIn - 1; channelIn >= 0; channelIn--) { + for (int channelIn = 0; channelIn < numChannelIn; channelIn++) { + StubGP* stub = pop_front(stacks[channelIn]); if (stub) { nothingToRoute = false; - acceptedSector.push_back(stub); + output.push_back(stub); break; } } if (nothingToRoute) - acceptedSector.push_back(nullptr); + output.push_back(nullptr); } // truncate if desired - if (enableTruncation_ && (int)acceptedSector.size() > setup_->numFrames()) { - const auto limit = next(acceptedSector.begin(), setup_->numFrames()); - copy_if(limit, acceptedSector.end(), back_inserter(lostSector), [](const StubGP* stub) { return stub; }); - acceptedSector.erase(limit, acceptedSector.end()); - } + if (enableTruncation_ && (int)output.size() > setup_->numFramesHigh()) + output.resize(setup_->numFramesHigh()); // remove all gaps between end and last stub - for (auto it = acceptedSector.end(); it != acceptedSector.begin();) - it = (*--it) ? acceptedSector.begin() : acceptedSector.erase(it); - // fill products - auto put = [](const vector& stubs, StreamStub& stream) { - auto toFrame = [](StubGP* stub) { return stub ? stub->frame() : FrameStub(); }; - stream.reserve(stubs.size()); - transform(stubs.begin(), stubs.end(), back_inserter(stream), toFrame); - }; - const int index = region_ * dataFormats_->numChannel(Process::gp) + sector; - put(acceptedSector, accepted[index]); - put(lostSector, lost[index]); + for (auto it = output.end(); it != output.begin();) + it = (*--it) ? output.begin() : output.erase(it); } } + // convert stub + StubGP* GeometricProcessor::produce(const StubPP& stub, int phiT, int zT) { + static const DataFormat& dfPhiT = dataFormats_->format(Variable::phiT, Process::gp); + static const DataFormat& dfZT = dataFormats_->format(Variable::zT, Process::gp); + static const DataFormat& dfCot = dataFormats_->format(Variable::cot, Process::gp); + static const DataFormat& dfR = dataFormats_->format(Variable::r, Process::gp); + static const DataFormat& dfL = dataFormats_->format(Variable::layer, Process::gp); + const double cot = dfCot.digi(dfZT.floating(zT) / setup_->chosenRofZ()); + // determine kf layer id + const vector& le = layerEncoding_->layerEncoding(zT); + const int layerId = setup_->layerId(stub.frame().first); + const auto it = find(le.begin(), le.end(), layerId); + const int kfLayerId = min((int)distance(le.begin(), it), setup_->numLayers() - 1); + // create data fields + const double r = stub.r(); + const double phi = stub.phi() - dfPhiT.floating(phiT); + const double z = stub.z() - (stub.r() + dfR.digi(setup_->chosenRofPhi())) * cot; + TTBV layer(kfLayerId, dfL.width()); + if (stub.layer()[4]) { // barrel + layer.set(5); + if (stub.layer()[3]) // psTilt + layer.set(3); + if (stub.layer().val(3) < 3) // layerId < 3 + layer.set(4); + } else if (stub.layer()[3]) // psTilt + layer.set(4); + const int inv2RMin = stub.inv2RMin(); + const int inv2RMax = stub.inv2RMax(); + stubs_.emplace_back(stub, r, phi, z, layer, inv2RMin, inv2RMax); + return &stubs_.back(); + } + // remove and return first element of deque, returns nullptr if empty template T* GeometricProcessor::pop_front(deque& ts) const { diff --git a/L1Trigger/TrackerTFP/src/HoughTransform.cc b/L1Trigger/TrackerTFP/src/HoughTransform.cc index 5995285b06a9c..eb9c45801dd58 100644 --- a/L1Trigger/TrackerTFP/src/HoughTransform.cc +++ b/L1Trigger/TrackerTFP/src/HoughTransform.cc @@ -5,8 +5,6 @@ #include #include #include -#include -#include #include using namespace std; @@ -18,186 +16,196 @@ namespace trackerTFP { HoughTransform::HoughTransform(const ParameterSet& iConfig, const Setup* setup, const DataFormats* dataFormats, - int region) + const LayerEncoding* layerEncoding, + vector& stubs) : enableTruncation_(iConfig.getParameter("EnableTruncation")), setup_(setup), dataFormats_(dataFormats), - inv2R_(dataFormats_->format(Variable::inv2R, Process::ht)), - phiT_(dataFormats_->format(Variable::phiT, Process::ht)), - region_(region), - input_(dataFormats_->numChannel(Process::ht), vector>(dataFormats_->numChannel(Process::gp))) {} - - // read in and organize input product (fill vector input_) - void HoughTransform::consume(const StreamsStub& streams) { - const int offset = region_ * dataFormats_->numChannel(Process::gp); - auto validFrame = [](int sum, const FrameStub& frame) { return sum + (frame.first.isNonnull() ? 1 : 0); }; - int nStubsGP(0); - for (int sector = 0; sector < dataFormats_->numChannel(Process::gp); sector++) { - const StreamStub& stream = streams[offset + sector]; - nStubsGP += accumulate(stream.begin(), stream.end(), 0, validFrame); - } - stubsGP_.reserve(nStubsGP); - for (int sector = 0; sector < dataFormats_->numChannel(Process::gp); sector++) { - const int sectorPhi = sector % setup_->numSectorsPhi(); - const int sectorEta = sector / setup_->numSectorsPhi(); - for (const FrameStub& frame : streams[offset + sector]) { - // Store input stubs in vector, so rest of HT algo can work with pointers to them (saves CPU) - StubGP* stub = nullptr; - if (frame.first.isNonnull()) { - stubsGP_.emplace_back(frame, dataFormats_, sectorPhi, sectorEta); - stub = &stubsGP_.back(); - } - for (int binInv2R = 0; binInv2R < dataFormats_->numChannel(Process::ht); binInv2R++) - input_[binInv2R][sector].push_back(stub && stub->inInv2RBin(binInv2R) ? stub : nullptr); - } - } - // remove all gaps between end and last stub - for (vector>& input : input_) - for (deque& stubs : input) - for (auto it = stubs.end(); it != stubs.begin();) - it = (*--it) ? stubs.begin() : stubs.erase(it); - auto validStub = [](int sum, StubGP* stub) { return sum + (stub ? 1 : 0); }; - int nStubsHT(0); - for (const vector>& binInv2R : input_) - for (const deque& sector : binInv2R) - nStubsHT += accumulate(sector.begin(), sector.end(), 0, validStub); - stubsHT_.reserve(nStubsHT); - } + layerEncoding_(layerEncoding), + inv2R_(&dataFormats_->format(Variable::inv2R, Process::ht)), + phiT_(&dataFormats_->format(Variable::phiT, Process::ht)), + zT_(&dataFormats_->format(Variable::zT, Process::gp)), + phi_(&dataFormats_->format(Variable::phi, Process::ht)), + z_(&dataFormats_->format(Variable::z, Process::gp)), + stubs_(stubs) {} // fill output products - void HoughTransform::produce(StreamsStub& accepted, StreamsStub& lost) { - for (int binInv2R = 0; binInv2R < dataFormats_->numChannel(Process::ht); binInv2R++) { - const int inv2R = inv2R_.toSigned(binInv2R); - deque acceptedAll; - deque lostAll; - for (deque& inputSector : input_[binInv2R]) { - const int size = inputSector.size(); - vector acceptedSector; - vector lostSector; - acceptedSector.reserve(size); - lostSector.reserve(size); + void HoughTransform::produce(const vector>& streamsIn, vector>& streamsOut) { + static const int numChannelIn = dataFormats_->numChannel(Process::gp); + static const int numChannelOut = dataFormats_->numChannel(Process::ht); + static const int chan = setup_->kfNumWorker(); + static const int mux = numChannelOut / chan; + // count and reserve ht stubs + auto multiplicity = [](int& sum, StubGP* s) { return sum += s ? 1 + s->inv2RMax() - s->inv2RMin() : 0; }; + int nStubs(0); + for (const vector& input : streamsIn) + nStubs += accumulate(input.begin(), input.end(), 0, multiplicity); + stubs_.reserve(nStubs); + for (int channelOut = 0; channelOut < numChannelOut; channelOut++) { + const int inv2Ru = mux * (channelOut % chan) + channelOut / chan; + const int inv2R = inv2R_->toSigned(inv2Ru); + deque& output = streamsOut[channelOut]; + for (int channelIn = numChannelIn - 1; channelIn >= 0; channelIn--) { + const vector& input = streamsIn[channelIn]; + vector stubs; + stubs.reserve(2 * input.size()); // associate stubs with inv2R and phiT bins - fillIn(inv2R, inputSector, acceptedSector, lostSector); - // Process::ht collects all stubs before readout starts -> remove all gaps - acceptedSector.erase(remove(acceptedSector.begin(), acceptedSector.end(), nullptr), acceptedSector.end()); + fillIn(inv2R, channelIn, input, stubs); + // apply truncation + if (enableTruncation_ && (int)stubs.size() > setup_->numFramesHigh()) + stubs.resize(setup_->numFramesHigh()); + // ht collects all stubs before readout starts -> remove all gaps + stubs.erase(remove(stubs.begin(), stubs.end(), nullptr), stubs.end()); // identify tracks - readOut(acceptedSector, lostSector, acceptedAll, lostAll); + readOut(stubs, output); } - // truncate accepted stream - const auto limit = enableTruncation_ - ? next(acceptedAll.begin(), min(setup_->numFrames(), (int)acceptedAll.size())) - : acceptedAll.end(); - copy_if(limit, acceptedAll.end(), back_inserter(lostAll), [](StubHT* stub) { return stub; }); - acceptedAll.erase(limit, acceptedAll.end()); - // store found tracks - auto put = [](const deque& stubs, StreamStub& stream) { - stream.reserve(stubs.size()); - for (StubHT* stub : stubs) - stream.emplace_back(stub ? stub->frame() : FrameStub()); - }; - const int offset = region_ * dataFormats_->numChannel(Process::ht); - put(acceptedAll, accepted[offset + binInv2R]); - // store lost tracks - put(lostAll, lost[offset + binInv2R]); + // apply truncation + if (enableTruncation_ && (int)output.size() > setup_->numFramesHigh()) + output.resize(setup_->numFramesHigh()); + // remove trailing gaps + for (auto it = output.end(); it != output.begin();) + it = (*--it) ? output.begin() : output.erase(it); } } // associate stubs with phiT bins in this inv2R column - void HoughTransform::fillIn(int inv2R, - deque& inputSector, - vector& acceptedSector, - vector& lostSector) { + void HoughTransform::fillIn(int inv2R, int sector, const vector& input, vector& output) { + static const DataFormat& gp = dataFormats_->format(Variable::phiT, Process::gp); + auto inv2RrangeCheck = [inv2R](StubGP* stub) { + return (stub && stub->inv2RMin() <= inv2R && stub->inv2RMax() >= inv2R) ? stub : nullptr; + }; + const int gpPhiT = gp.toSigned(sector % setup_->gpNumBinsPhiT()); + const int zT = sector / setup_->gpNumBinsPhiT() - setup_->gpNumBinsZT() / 2; + const double inv2Rf = inv2R_->floating(inv2R); + const double zTf = zT_->floating(zT); + const double cotf = zTf / setup_->chosenRofZ(); + auto convert = [this, inv2Rf, gpPhiT, zT](StubGP* stub, int phiTht, double phi, double z) { + const double phiTf = phiT_->floating(phiTht); + const int phiT = phiT_->integer(gp.floating(gpPhiT) + phiTf); + const double htPhi = phi - (inv2Rf * stub->r() + phiTf); + stubs_.emplace_back(*stub, stub->r(), htPhi, z, stub->layer(), phiT, zT); + return &stubs_.back(); + }; + // Latency of ht fifo firmware + static constexpr int latency = 1; + // static delay container + deque delay(latency, nullptr); // fifo, used to store stubs which belongs to a second possible track deque stack; + // stream of incroming stubs + deque stream; + transform(input.begin(), input.end(), back_inserter(stream), inv2RrangeCheck); // clock accurate firmware emulation, each while trip describes one clock tick, one stub in and one stub out per tick - while (!inputSector.empty() || !stack.empty()) { + while (!stream.empty() || !stack.empty() || + !all_of(delay.begin(), delay.end(), [](const StubHT* stub) { return !stub; })) { StubHT* stubHT = nullptr; - StubGP* stubGP = pop_front(inputSector); + StubGP* stubGP = pop_front(stream); if (stubGP) { - const double phiT = stubGP->phi() - inv2R_.floating(inv2R) * stubGP->r(); - const int major = phiT_.integer(phiT); - if (phiT_.inRange(major)) { + double phi = stubGP->phi(); + double z = stubGP->z(); + if (false) { + const double d = inv2Rf * (stubGP->r() + setup_->chosenRofPhi()); + const double dPhi = asin(d) - d; + const double dZ = dPhi / inv2Rf * cotf; + phi = phi_->digi(phi - dPhi); + z = z_->digi(z - dZ); + } + const double phiT = phi - inv2Rf * stubGP->r(); + const int major = phiT_->integer(phiT); + if (major >= -setup_->htNumBinsPhiT() / 2 && major < setup_->htNumBinsPhiT() / 2) { // major candidate has pt > threshold (3 GeV) - // stubHT records which HT bin this stub is added to - stubsHT_.emplace_back(*stubGP, major, inv2R); - stubHT = &stubsHT_.back(); + stubHT = convert(stubGP, major, phi, z); } - const double chi = phiT - phiT_.floating(major); - if (abs(stubGP->r() * inv2R_.base()) + 2. * abs(chi) >= phiT_.base()) { + const double chi = phi_->digi(phiT - phiT_->floating(major)); + if (abs(stubGP->r() * inv2R_->base()) + 2. * abs(chi) >= phiT_->base()) { // stub belongs to two candidates const int minor = chi >= 0. ? major + 1 : major - 1; - if (phiT_.inRange(minor)) { + if (minor >= -setup_->htNumBinsPhiT() / 2 && minor < setup_->htNumBinsPhiT() / 2) { // second (minor) candidate has pt > threshold (3 GeV) - stubsHT_.emplace_back(*stubGP, minor, inv2R); - if (enableTruncation_ && (int)stack.size() == setup_->htDepthMemory() - 1) - // buffer overflow - lostSector.push_back(pop_front(stack)); - // store minor stub in fifo - stack.push_back(&stubsHT_.back()); + StubHT* stub = convert(stubGP, minor, phi, z); + delay.push_back(stub); } } } + // add nullptr to delay pipe if stub didn't fill any cell + if ((int)delay.size() == latency) + delay.push_back(nullptr); + // take fifo latency into account (read before write) + StubHT* stub = pop_front(delay); + if (stub) { + // buffer overflow + if (enableTruncation_ && (int)stack.size() == setup_->htDepthMemory() - 1) + pop_front(stack); + // store minor stub in fifo + stack.push_back(stub); + } // take a minor stub if no major stub available - acceptedSector.push_back(stubHT ? stubHT : pop_front(stack)); + output.push_back(stubHT ? stubHT : pop_front(stack)); } - // truncate to many input stubs - const auto limit = enableTruncation_ - ? next(acceptedSector.begin(), min(setup_->numFrames(), (int)acceptedSector.size())) - : acceptedSector.end(); - copy_if(limit, acceptedSector.end(), back_inserter(lostSector), [](StubHT* stub) { return stub; }); - acceptedSector.erase(limit, acceptedSector.end()); } // identify tracks - void HoughTransform::readOut(const vector& acceptedSector, - const vector& lostSector, - deque& acceptedAll, - deque& lostAll) const { + void HoughTransform::readOut(const vector& input, deque& output) const { + auto toBinPhiT = [this](StubHT* stub) { + static const DataFormat& gp = dataFormats_->format(Variable::phiT, Process::gp); + const double phiT = phiT_->floating(stub->phiT()); + const double local = phiT - gp.digi(phiT); + return phiT_->integer(local) + setup_->htNumBinsPhiT() / 2; + }; + auto toLayerId = [this](StubHT* stub) { + static const DataFormat& layer = dataFormats_->format(Variable::layer, Process::ctb); + return stub->layer().val(layer.width()); + }; // used to recognise in which order tracks are found TTBV trkFoundPhiTs(0, setup_->htNumBinsPhiT()); // hitPattern for all possible tracks, used to find tracks vector patternHits(setup_->htNumBinsPhiT(), TTBV(0, setup_->numLayers())); - // found unsigned phiTs, ordered in time - vector binsPhiT; - // stub container for all possible tracks - vector> tracks(setup_->htNumBinsPhiT()); - for (int binPhiT = 0; binPhiT < setup_->htNumBinsPhiT(); binPhiT++) { - const int phiT = phiT_.toSigned(binPhiT); - auto samePhiT = [phiT](int sum, StubHT* stub) { return sum + (stub->phiT() == phiT); }; - const int numAccepted = accumulate(acceptedSector.begin(), acceptedSector.end(), 0, samePhiT); - const int numLost = accumulate(lostSector.begin(), lostSector.end(), 0, samePhiT); - tracks[binPhiT].reserve(numAccepted + numLost); - } - for (StubHT* stub : acceptedSector) { - const int binPhiT = phiT_.toUnsigned(stub->phiT()); + // found phiTs, ordered in time + vector phiTs; + phiTs.reserve(setup_->htNumBinsPhiT()); + for (StubHT* stub : input) { + const int binPhiT = toBinPhiT(stub); + const int layerId = toLayerId(stub); TTBV& pattern = patternHits[binPhiT]; - pattern.set(stub->layer()); - tracks[binPhiT].push_back(stub); - if (pattern.count() >= setup_->htMinLayers() && !trkFoundPhiTs[binPhiT]) { - // first time track found - trkFoundPhiTs.set(binPhiT); - binsPhiT.push_back(binPhiT); - } + pattern.set(layerId); + if (trkFoundPhiTs[binPhiT] || noTrack(pattern, stub->zT())) + continue; + // first time track found + trkFoundPhiTs.set(binPhiT); + phiTs.push_back(binPhiT); } // read out found tracks ordered as found - for (int binPhiT : binsPhiT) { - const vector& track = tracks[binPhiT]; - acceptedAll.insert(acceptedAll.end(), track.begin(), track.end()); - } - // look for lost tracks - for (StubHT* stub : lostSector) { - const int binPhiT = phiT_.toUnsigned(stub->phiT()); - if (!trkFoundPhiTs[binPhiT]) - tracks[binPhiT].push_back(stub); + for (int phiT : phiTs) { + auto samePhiT = [phiT, toBinPhiT, this](StubHT* stub) { return toBinPhiT(stub) == phiT; }; + // read out stubs in reverse order to emulate f/w (backtracking linked list) + copy_if(input.rbegin(), input.rend(), back_inserter(output), samePhiT); } - for (int binPhiT : trkFoundPhiTs.ids(false)) { - const vector& track = tracks[binPhiT]; - set layers; - auto toLayer = [](StubHT* stub) { return stub->layer(); }; - transform(track.begin(), track.end(), inserter(layers, layers.begin()), toLayer); - if ((int)layers.size() >= setup_->htMinLayers()) - lostAll.insert(lostAll.end(), track.begin(), track.end()); + } + + // track identification + bool HoughTransform::noTrack(const TTBV& pattern, int zT) const { + // not enough seeding layer + if (pattern.count(0, setup_->kfMaxSeedingLayer()) < 2) + return true; + // check min layers req + const int minLayers = setup_->htMinLayers() - (((zT == -4 || zT == 3) && (!pattern.test(5) && !pattern.test(7))) ? 1 : 0); + // prepare pattern analysis + const TTBV& maybePattern = layerEncoding_->maybePattern(zT); + int nHits(0); + int nGaps(0); + bool doubleGap = false; + for (int layer = 0; layer < setup_->numLayers(); layer++) { + if (pattern.test(layer)) { + doubleGap = false; + if(++nHits == minLayers) + return false; + } else if (!maybePattern.test(layer)) { + if (++nGaps == setup_->kfMaxGaps() || doubleGap) + break; + doubleGap = true; + } } + return true; } // remove and return first element of deque, returns nullptr if empty @@ -211,4 +219,4 @@ namespace trackerTFP { return t; } -} // namespace trackerTFP +} // namespace trackerTFP \ No newline at end of file diff --git a/L1Trigger/TrackerTFP/src/KalmanFilter.cc b/L1Trigger/TrackerTFP/src/KalmanFilter.cc index 7b64f0a1f00e9..68da57d8d2483 100644 --- a/L1Trigger/TrackerTFP/src/KalmanFilter.cc +++ b/L1Trigger/TrackerTFP/src/KalmanFilter.cc @@ -18,192 +18,269 @@ namespace trackerTFP { KalmanFilter::KalmanFilter(const ParameterSet& iConfig, const Setup* setup, const DataFormats* dataFormats, + const LayerEncoding* layerEncoding, KalmanFilterFormats* kalmanFilterFormats, - int region) + vector& tracks, + vector& stubs) : enableTruncation_(iConfig.getParameter("EnableTruncation")), setup_(setup), dataFormats_(dataFormats), + layerEncoding_(layerEncoding), kalmanFilterFormats_(kalmanFilterFormats), - region_(region), - input_(dataFormats_->numChannel(Process::kf)), - layer_(0), - x0_(&kalmanFilterFormats_->format(VariableKF::x0)), - x1_(&kalmanFilterFormats_->format(VariableKF::x1)), - x2_(&kalmanFilterFormats_->format(VariableKF::x2)), - x3_(&kalmanFilterFormats_->format(VariableKF::x3)), - H00_(&kalmanFilterFormats_->format(VariableKF::H00)), - H12_(&kalmanFilterFormats_->format(VariableKF::H12)), - m0_(&kalmanFilterFormats_->format(VariableKF::m0)), - m1_(&kalmanFilterFormats_->format(VariableKF::m1)), - v0_(&kalmanFilterFormats_->format(VariableKF::v0)), - v1_(&kalmanFilterFormats_->format(VariableKF::v1)), - r0_(&kalmanFilterFormats_->format(VariableKF::r0)), - r1_(&kalmanFilterFormats_->format(VariableKF::r1)), - S00_(&kalmanFilterFormats_->format(VariableKF::S00)), - S01_(&kalmanFilterFormats_->format(VariableKF::S01)), - S12_(&kalmanFilterFormats_->format(VariableKF::S12)), - S13_(&kalmanFilterFormats_->format(VariableKF::S13)), - K00_(&kalmanFilterFormats_->format(VariableKF::K00)), - K10_(&kalmanFilterFormats_->format(VariableKF::K10)), - K21_(&kalmanFilterFormats_->format(VariableKF::K21)), - K31_(&kalmanFilterFormats_->format(VariableKF::K31)), - R00_(&kalmanFilterFormats_->format(VariableKF::R00)), - R11_(&kalmanFilterFormats_->format(VariableKF::R11)), - R00Rough_(&kalmanFilterFormats_->format(VariableKF::R00Rough)), - R11Rough_(&kalmanFilterFormats_->format(VariableKF::R11Rough)), - invR00Approx_(&kalmanFilterFormats_->format(VariableKF::invR00Approx)), - invR11Approx_(&kalmanFilterFormats_->format(VariableKF::invR11Approx)), - invR00Cor_(&kalmanFilterFormats_->format(VariableKF::invR00Cor)), - invR11Cor_(&kalmanFilterFormats_->format(VariableKF::invR11Cor)), - invR00_(&kalmanFilterFormats_->format(VariableKF::invR00)), - invR11_(&kalmanFilterFormats_->format(VariableKF::invR11)), - C00_(&kalmanFilterFormats_->format(VariableKF::C00)), - C01_(&kalmanFilterFormats_->format(VariableKF::C01)), - C11_(&kalmanFilterFormats_->format(VariableKF::C11)), - C22_(&kalmanFilterFormats_->format(VariableKF::C22)), - C23_(&kalmanFilterFormats_->format(VariableKF::C23)), - C33_(&kalmanFilterFormats_->format(VariableKF::C33)) { - C00_->updateRangeActual(pow(dataFormats_->base(Variable::inv2R, Process::kfin), 2)); - C11_->updateRangeActual(pow(dataFormats_->base(Variable::phiT, Process::kfin), 2)); - C22_->updateRangeActual(pow(dataFormats_->base(Variable::cot, Process::kfin), 2)); - C33_->updateRangeActual(pow(dataFormats_->base(Variable::zT, Process::kfin), 2)); - } - - // read in and organize input product (fill vector input_) - void KalmanFilter::consume(const StreamsTrack& streamsTrack, const StreamsStub& streamsStub) { - auto valid = [](const auto& frame) { return frame.first.isNonnull(); }; - auto acc = [](int sum, const auto& frame) { return sum + (frame.first.isNonnull() ? 1 : 0); }; - int nTracks(0); - int nStubs(0); - const int offset = region_ * dataFormats_->numChannel(Process::kf); - for (int channel = 0; channel < dataFormats_->numChannel(Process::kf); channel++) { - const int channelTrack = offset + channel; - const StreamTrack& streamTracks = streamsTrack[channelTrack]; - nTracks += accumulate(streamTracks.begin(), streamTracks.end(), 0, acc); - for (int layer = 0; layer < setup_->numLayers(); layer++) { - const int channelStub = channelTrack * setup_->numLayers() + layer; - const StreamStub& streamStubs = streamsStub[channelStub]; - nStubs += accumulate(streamStubs.begin(), streamStubs.end(), 0, acc); - } - } - tracks_.reserve(nTracks); - stubs_.reserve(nStubs); - // N.B. One input stream for track & one for its stubs in each layer. If a track has N stubs in one layer, and fewer in all other layers, then next valid track will be N frames later - for (int channel = 0; channel < dataFormats_->numChannel(Process::kf); channel++) { - const int channelTrack = offset + channel; - const StreamTrack& streamTracks = streamsTrack[channelTrack]; - vector& tracks = input_[channel]; - tracks.reserve(streamTracks.size()); - for (int frame = 0; frame < (int)streamTracks.size(); frame++) { - const FrameTrack& frameTrack = streamTracks[frame]; - // Select frames with valid track - if (frameTrack.first.isNull()) { - if (dataFormats_->hybrid()) - tracks.push_back(nullptr); - continue; - } - auto endOfTrk = find_if(next(streamTracks.begin(), frame + 1), streamTracks.end(), valid); - if (dataFormats_->hybrid()) - endOfTrk = next(streamTracks.begin(), frame + 1); - // No. of frames before next track indicates gives max. no. of stubs this track had in any layer - const int maxStubsPerLayer = distance(next(streamTracks.begin(), frame), endOfTrk); - tracks.insert(tracks.end(), maxStubsPerLayer - 1, nullptr); - deque stubs; - for (int layer = 0; layer < setup_->numLayers(); layer++) { - const int channelStub = channelTrack * setup_->numLayers() + layer; - const StreamStub& streamStubs = streamsStub[channelStub]; - // Get stubs on this track - for (int i = frame; i < frame + maxStubsPerLayer; i++) { - const FrameStub& frameStub = streamStubs[i]; - if (frameStub.first.isNull()) - break; - // Store input stubs, so remainder of KF algo can work with pointers to them (saves CPU) - stubs_.emplace_back(frameStub, dataFormats_, layer); - stubs.push_back(&stubs_.back()); - } - } - // Store input tracks, so remainder of KF algo can work with pointers to them (saves CPU) - tracks_.emplace_back(frameTrack, dataFormats_, vector(stubs.begin(), stubs.end())); - tracks.push_back(&tracks_.back()); - } - } - } + tracks_(tracks), + stubs_(stubs), + layer_(0) {} // fill output products - void KalmanFilter::produce(StreamsStub& acceptedStubs, - StreamsTrack& acceptedTracks, - StreamsStub& lostStubs, - StreamsTrack& lostTracks, + void KalmanFilter::produce(const vector>& tracksIn, + const vector>& stubsIn, + vector>& tracksOut, + vector>>& stubsOut, int& numAcceptedStates, - int& numLostStates) { - auto put = [this]( - const deque& states, StreamsStub& streamsStubs, StreamsTrack& streamsTracks, int channel) { - const int streamId = region_ * dataFormats_->numChannel(Process::kf) + channel; - const int offset = streamId * setup_->numLayers(); - StreamTrack& tracks = streamsTracks[streamId]; - tracks.reserve(states.size()); - for (int layer = 0; layer < setup_->numLayers(); layer++) - streamsStubs[offset + layer].reserve(states.size()); - for (State* state : states) { - tracks.emplace_back(state->frame()); - vector stubs; - state->fill(stubs); - for (const StubKF& stub : stubs) - streamsStubs[offset + stub.layer()].emplace_back(stub.frame()); - // adding a gap to all layer without a stub - for (int layer : state->hitPattern().ids(false)) - streamsStubs[offset + layer].emplace_back(FrameStub()); - } - }; - auto count = [this](int sum, const State* state) { - return sum + ((state && state->hitPattern().count() >= setup_->kfMinLayers()) ? 1 : 0); - }; + int& numLostStates, + deque>& chi2s) { for (int channel = 0; channel < dataFormats_->numChannel(Process::kf); channel++) { deque stream; - deque lost; // proto state creation - int trackId(0); - for (TrackKFin* track : input_[channel]) { - State* state = nullptr; - if (track) { - // Store states, so remainder of KF algo can work with pointers to them (saves CPU) - states_.emplace_back(dataFormats_, track, trackId++); - state = &states_.back(); - } - stream.push_back(state); - } + createProtoStates(tracksIn, stubsIn, channel, stream); + // seed building + for (layer_ = 0; layer_ < setup_->kfMaxSeedingLayer(); layer_++) + addSeedLayer(stream); + // calulcate seed parameter + calcSeeds(stream); // Propagate state to each layer in turn, updating it with all viable stub combinations there, using KF maths - for (layer_ = 0; layer_ < setup_->numLayers(); layer_++) + for (layer_ = setup_->kfNumSeedStubs(); layer_ < setup_->numLayers(); layer_++) addLayer(stream); - // calculate number of states before truncating - const int numUntruncatedStates = accumulate(stream.begin(), stream.end(), 0, count); - // untruncated best state selection - deque untruncatedStream = stream; - accumulator(untruncatedStream); + // count total number of final states + const int nStates = + accumulate(stream.begin(), stream.end(), 0, [](int& sum, State* state) { return sum += (state ? 1 : 0); }); // apply truncation - if (enableTruncation_ && (int)stream.size() > setup_->numFrames()) - stream.resize(setup_->numFrames()); - // calculate number of states after truncating - const int numTruncatedStates = accumulate(stream.begin(), stream.end(), 0, count); - // best state per candidate selection - accumulator(stream); - deque truncatedStream = stream; - // storing of best states missed due to truncation - sort(untruncatedStream.begin(), untruncatedStream.end()); - sort(truncatedStream.begin(), truncatedStream.end()); - set_difference(untruncatedStream.begin(), - untruncatedStream.end(), - truncatedStream.begin(), - truncatedStream.end(), - back_inserter(lost)); - // store found tracks - put(stream, acceptedStubs, acceptedTracks, channel); - // store lost tracks - put(lost, lostStubs, lostTracks, channel); + if (enableTruncation_ && (int)stream.size() > setup_->numFramesHigh()) + stream.resize(setup_->numFramesHigh()); + // cycle event, remove gaps + stream.erase(remove(stream.begin(), stream.end(), nullptr), stream.end()); // store number of states which got taken into account - numAcceptedStates += numTruncatedStates; + numAcceptedStates += (int)stream.size(); // store number of states which got not taken into account due to truncation - numLostStates += numUntruncatedStates - numTruncatedStates; + numLostStates += nStates - (int)stream.size(); + // apply final cuts + vector finals; + finals.reserve(stream.size()); + finalize(stream, finals); + // best track per candidate selection + vector best; + best.reserve(stream.size()); + accumulator(finals, best); + // store chi2s + for (Track* track : best) + chi2s.emplace_back(track->chi20_, track->chi21_); + // Transform States into Tracks + vector& tracks = tracksOut[channel]; + vector>& stubs = stubsOut[channel]; + conv(best, tracks, stubs); + } + } + + // create Proto States + void KalmanFilter::createProtoStates(const std::vector>& tracksIn, + const std::vector>& stubsIn, + int channel, + deque& stream) { + static const int numLayers = setup_->numLayers(); + const int offsetL = channel * numLayers; + const vector& tracksChannel = tracksIn[channel]; + int trackId(0); + for (int frame = 0; frame < (int)tracksChannel.size();) { + TrackCTB* track = tracksChannel[frame]; + if (!track) { + frame++; + continue; + } + const auto begin = next(tracksChannel.begin(), frame); + const auto end = find_if(begin + 1, tracksChannel.end(), [](TrackCTB* track) { return track; }); + const int size = distance(begin, end); + vector> stubs(numLayers); + for (vector& layer : stubs) + layer.reserve(size); + for (int layer = 0; layer < numLayers; layer++) { + const vector& layerAll = stubsIn[layer + offsetL]; + vector& layerTrack = stubs[layer]; + for (int frameS = 0; frameS < size; frameS++) { + Stub* stub = layerAll[frameS + frame]; + if (!stub) + break; + layerTrack.push_back(stub); + } + } + const TTBV& maybePattern = layerEncoding_->maybePattern(track->zT()); + states_.emplace_back(kalmanFilterFormats_, track, stubs, maybePattern, trackId++); + stream.insert(stream.end(), size - 1, nullptr); + stream.push_back(&states_.back()); + frame += size; + } + } + + // calulcate seed parameter + void KalmanFilter::calcSeeds(deque& stream) { + auto update = [this](State* s) { + updateRangeActual(VariableKF::m0, s->m0()); + updateRangeActual(VariableKF::m1, s->m1()); + updateRangeActual(VariableKF::v0, s->v0()); + updateRangeActual(VariableKF::v1, s->v1()); + updateRangeActual(VariableKF::H00, s->H00()); + updateRangeActual(VariableKF::H12, s->H12()); + }; + for (State*& state : stream) { + if (!state) + continue; + State* s1 = state->parent(); + State* s0 = s1->parent(); + update(s0); + update(s1); + const double dH = digi(VariableKF::dH, s1->H00() - s0->H00()); + const double invdH = digi(VariableKF::invdH, 1.0 / dH); + const double invdH2 = digi(VariableKF::invdH2, 1.0 / dH / dH); + const double H12 = digi(VariableKF::H2, s1->H00() * s1->H00()); + const double H02 = digi(VariableKF::H2, s0->H00() * s0->H00()); + const double H32 = digi(VariableKF::H2, s1->H12() * s1->H12()); + const double H22 = digi(VariableKF::H2, s0->H12() * s0->H12()); + const double H1m0 = digi(VariableKF::Hm0, s1->H00() * s0->m0()); + const double H0m1 = digi(VariableKF::Hm0, s0->H00() * s1->m0()); + const double H3m2 = digi(VariableKF::Hm1, s1->H12() * s0->m1()); + const double H2m3 = digi(VariableKF::Hm1, s0->H12() * s1->m1()); + const double H1v0 = digi(VariableKF::Hv0, s1->H00() * s0->v0()); + const double H0v1 = digi(VariableKF::Hv0, s0->H00() * s1->v0()); + const double H3v2 = digi(VariableKF::Hv1, s1->H12() * s0->v1()); + const double H2v3 = digi(VariableKF::Hv1, s0->H12() * s1->v1()); + const double H12v0 = digi(VariableKF::H2v0, H12 * s0->v0()); + const double H02v1 = digi(VariableKF::H2v0, H02 * s1->v0()); + const double H32v2 = digi(VariableKF::H2v1, H32 * s0->v1()); + const double H22v3 = digi(VariableKF::H2v1, H22 * s1->v1()); + const double x0 = digi(VariableKF::x0, (s1->m0() - s0->m0()) * invdH); + const double x2 = digi(VariableKF::x2, (s1->m1() - s0->m1()) * invdH); + const double x1 = digi(VariableKF::x1, (H1m0 - H0m1) * invdH); + const double x3 = digi(VariableKF::x3, (H3m2 - H2m3) * invdH); + const double C00 = digi(VariableKF::C00, (s1->v0() + s0->v0()) * invdH2); + const double C22 = digi(VariableKF::C22, (s1->v1() + s0->v1()) * invdH2); + const double C01 = -digi(VariableKF::C01, (H1v0 + H0v1) * invdH2); + const double C23 = -digi(VariableKF::C23, (H3v2 + H2v3) * invdH2); + const double C11 = digi(VariableKF::C11, (H12v0 + H02v1) * invdH2); + const double C33 = digi(VariableKF::C33, (H32v2 + H22v3) * invdH2); + // cut on eta sector boundaries + /*const bool invalidX3 = abs(x3) > zT.base() / 2.; + // cut on triple found inv2R window + const bool invalidX0 = abs(x0) > 1.5 * inv2R.base(); + // cut on triple found phiT window + const bool invalidX1 = abs(x1) > 1.5 * phiT.base(); + // cot cut + const bool invalidX2 = abs(x2) > maxCot; + if (invalidX3 || invalidX0 || invalidX1 || invalidX2) { + state = nullptr; + continue; + }*/ + // create updated state + static const double chi20 = digi(VariableKF::chi20, 0.); + static const double chi21 = digi(VariableKF::chi21, 0.); + states_.emplace_back(State(s1, {x0, x1, x2, x3, chi20, chi21, C00, C11, C22, C33, C01, C23})); + state = &states_.back(); + updateRangeActual(VariableKF::x0, x0); + updateRangeActual(VariableKF::x1, x1); + updateRangeActual(VariableKF::x2, x2); + updateRangeActual(VariableKF::x3, x3); + updateRangeActual(VariableKF::C00, C00); + updateRangeActual(VariableKF::C01, C01); + updateRangeActual(VariableKF::C11, C11); + updateRangeActual(VariableKF::C22, C22); + updateRangeActual(VariableKF::C23, C23); + updateRangeActual(VariableKF::C33, C33); + } + } + + // apply final cuts + void KalmanFilter::finalize(const deque& stream, vector& finals) { + for (State* state : stream) { + TrackCTB* track = state->track(); + int numConsistent(0); + int numConsistentPS(0); + TTBV hitPattern = state->hitPattern(); + vector stubs; + vector phis; + vector zs; + stubs.reserve(setup_->numLayers()); + phis.reserve(setup_->numLayers()); + zs.reserve(setup_->numLayers()); + double chi20(0.); + double chi21(0.); + // stub residual cut + State* s = state; + while ((s = s->parent())) { + const double dPhi = state->x1() + s->H00() * state->x0(); + const double dZ = state->x3() + s->H12() * state->x2(); + const double phi = digi(VariableKF::m0, s->m0() - dPhi); + const double z = digi(VariableKF::m1, s->m1() - dZ); + const bool validPhi = dataFormats_->format(Variable::phi, Process::kf).inRange(phi); + const bool validZ = dataFormats_->format(Variable::z, Process::kf).inRange(z); + StubCTB& stubCTB = s->stub()->stubCTB_; + if (validPhi && validZ) { + chi20 += pow(phi, 2); + chi21 += pow(z, 2); + stubs.push_back(&stubCTB); + phis.push_back(phi); + zs.push_back(z); + if (abs(phi) <= s->dPhi() && abs(z) <= s->dZ()) { + numConsistent++; + if (setup_->psModule(stubCTB.frame().first)) + numConsistentPS++; + } + } else + hitPattern.reset(s->layer()); + } + const double ndof = hitPattern.count() - 2; + chi20 /= ndof; + chi21 /= ndof; + // layer cut + bool invalidLayers = hitPattern.count() < setup_->kfMinLayers(); + // track parameter cuts + const double cotTrack = dataFormats_->format(Variable::cot, Process::kf).digi(track->zT() / setup_->chosenRofZ()); + const double inv2R = state->x0() + track->inv2R(); + const double phiT = state->x1() + track->phiT(); + const double cot = state->x2() + cotTrack; + const double zT = state->x3() + track->zT(); + // pt cut + const bool validX0 = dataFormats_->format(Variable::inv2R, Process::kf).inRange(inv2R); + // cut on phi sector boundaries + const bool validX1 = dataFormats_->format(Variable::phiT, Process::kf).inRange(phiT); + // cot cut + const bool validX2 = dataFormats_->format(Variable::cot, Process::kf).inRange(cot); + // zT cut + const bool validX3 = dataFormats_->format(Variable::zT, Process::kf).inRange(zT); + if (invalidLayers || !validX0 || !validX1 || !validX2 || !validX3) + continue; + const int trackId = state->trackId(); + finals_.emplace_back(trackId, numConsistent, numConsistentPS, inv2R, phiT, cot, zT, chi20, chi21, hitPattern, track, stubs, phis, zs); + } + } + + // Transform States into Tracks + void KalmanFilter::conv(const vector& best, vector& tracks, vector>& stubs) { + static const DataFormat& dfInv2R = dataFormats_->format(Variable::inv2R, Process::ht); + static const DataFormat& dfPhiT = dataFormats_->format(Variable::phiT, Process::ht); + tracks.reserve(best.size()); + for (vector& layer : stubs) + layer.reserve(best.size()); + for (Track* track : best) { + const vector layers = track->hitPattern_.ids(); + for (int iStub = 0; iStub < track->hitPattern_.count(); iStub++) { + StubCTB* s = track->stubs_[iStub]; + stubs_.emplace_back(*s, s->r(), track->phi_[iStub], track->z_[iStub], s->dPhi(), s->dZ()); + stubs[layers[iStub]].push_back(&stubs_.back()); + } + TrackCTB* trackCTB = track->track_; + const bool inInv2R = dfInv2R.integer(track->inv2R_) == dfInv2R.integer(trackCTB->inv2R()); + const bool inPhiT = dfPhiT.integer(track->phiT_) == dfPhiT.integer(trackCTB->phiT()); + const TTBV match(inInv2R && inPhiT, 1); + tracks_.emplace_back(*trackCTB, track->inv2R_, track->phiT_, track->cot_, track->zT_, match); + tracks.push_back(&tracks_.back()); } } @@ -216,7 +293,7 @@ namespace trackerTFP { // Memory stack used to handle combinatorics deque stack; // static delay container - vector delay(latency, nullptr); + deque delay(latency, nullptr); // each trip corresponds to a f/w clock tick // done if no states to process left, taking as much time as needed while (!stream.empty() || !stack.empty() || @@ -228,8 +305,7 @@ namespace trackerTFP { streamOutput.push_back(state); // The remainder of the code in this loop deals with combinatoric states. if (state) - // Assign next combinatoric stub to state - comb(state); + state = state->comb(states_, layer_); delay.push_back(state); state = pop_front(delay); if (state) @@ -238,195 +314,233 @@ namespace trackerTFP { stream = streamOutput; // Update state with next stub using KF maths for (State*& state : stream) - if (state && state->stub() && state->layer() == layer_) + if (state && state->hitPattern().pmEncode() == layer_) update(state); } - // Assign next combinatoric (i.e. not first in layer) stub to state - void KalmanFilter::comb(State*& state) { - const TrackKFin* track = state->track(); - const StubKFin* stub = state->stub(); - const vector& stubs = track->layerStubs(layer_); - const TTBV& hitPattern = state->hitPattern(); - StubKFin* stubNext = nullptr; - bool valid = state->stub() && state->layer() == layer_; - if (valid) { - // Get next unused stub on this layer - const int pos = distance(stubs.begin(), find(stubs.begin(), stubs.end(), stub)) + 1; - if (pos != (int)stubs.size()) - stubNext = stubs[pos]; - // picks next stub on different layer, nullifies state if skipping layer is not valid - else { - // having already maximum number of added layers - if (hitPattern.count() == setup_->kfMaxLayers()) - valid = false; - // Impossible for this state to ever get enough layers to form valid track - if (hitPattern.count() + track->hitPattern().count(stub->layer() + 1, setup_->numLayers()) < - setup_->kfMinLayers()) - valid = false; - // not diffrent layers left - if (layer_ == setup_->numLayers() - 1) - valid = false; - if (valid) { - // pick next stub on next populated layer - for (int nextLayer = layer_ + 1; nextLayer < setup_->numLayers(); nextLayer++) { - if (track->hitPattern(nextLayer)) { - stubNext = track->layerStub(nextLayer); - break; - } - } - } - } + // adds a layer to states to build seeds + void KalmanFilter::addSeedLayer(deque& stream) { + // Latency of KF Associator block firmware + static constexpr int latency = 5; + // dynamic state container for clock accurate emulation + deque streamOutput; + // Memory stack used to handle combinatorics + deque stack; + // static delay container + deque delay(latency, nullptr); + // each trip corresponds to a f/w clock tick + // done if no states to process left, taking as much time as needed + while (!stream.empty() || !stack.empty() || + !all_of(delay.begin(), delay.end(), [](const State* state) { return state == nullptr; })) { + State* state = pop_front(stream); + // Process a combinatoric state if no (non-combinatoric?) state available + if (!state) + state = pop_front(stack); + streamOutput.push_back(state); + // The remainder of the code in this loop deals with combinatoric states. + if (state) + state = state->combSeed(states_, layer_); + delay.push_back(state); + state = pop_front(delay); + if (state) + stack.push_back(state); } - if (valid) { - // create combinatoric state - states_.emplace_back(state, stubNext); - state = &states_.back(); - } else - state = nullptr; + stream = streamOutput; + // Update state with next stub using KF maths + for (State*& state : stream) + if (state) + state = state->update(states_, layer_); } // best state selection - void KalmanFilter::accumulator(deque& stream) { - // accumulator delivers contigious stream of best state per track - // remove gaps and not final states - stream.erase( - remove_if(stream.begin(), - stream.end(), - [this](State* state) { return !state || state->hitPattern().count() < setup_->kfMinLayers(); }), - stream.end()); - // Determine quality of completed state - for (State* state : stream) - state->finish(); - // sort in number of skipped layers - auto lessSkippedLayers = [](State* lhs, State* rhs) { return lhs->numSkippedLayers() < rhs->numSkippedLayers(); }; - stable_sort(stream.begin(), stream.end(), lessSkippedLayers); + void KalmanFilter::accumulator(vector& finals, vector& best) { + // create container of pointer to make sorts less CPU intense + //for (Track& track : finals) + //best.push_back(&track); + transform(finals.begin(), finals.end(), back_inserter(best), [](Track& track) { return &track; }); + // prepare arrival order + vector trackIds; + trackIds.reserve(best.size()); + for (Track* track : best) { + const int trackId = track->trackId_; + if (find_if(trackIds.begin(), trackIds.end(), [trackId](int id) { return id == trackId; }) == trackIds.end()) + trackIds.push_back(trackId); + } + // sort in chi2 + auto smallerChi2 = [](Track* lhs, Track* rhs) { return lhs->chi20_ + lhs->chi21_ < rhs->chi20_ + rhs->chi21_; }; + stable_sort(best.begin(), best.end(), smallerChi2); // sort in number of consistent stubs - auto moreConsistentLayers = [](State* lhs, State* rhs) { - return lhs->numConsistentLayers() > rhs->numConsistentLayers(); + auto moreConsistentLayers = [](Track* lhs, Track* rhs) { return lhs->numConsistent_ > rhs->numConsistent_; }; + stable_sort(best.begin(), best.end(), moreConsistentLayers); + // sort in number of consistent ps stubs + auto moreConsistentLayersPS = [](Track* lhs, Track* rhs) { return lhs->numConsistentPS_ > rhs->numConsistentPS_; }; + stable_sort(best.begin(), best.end(), moreConsistentLayersPS); + // sort in track id as arrived + auto order = [&trackIds](auto lhs, auto rhs) { + const auto l = find(trackIds.begin(), trackIds.end(), lhs->trackId_); + const auto r = find(trackIds.begin(), trackIds.end(), rhs->trackId_); + return distance(r, l) < 0; }; - stable_sort(stream.begin(), stream.end(), moreConsistentLayers); - // sort in track id - stable_sort(stream.begin(), stream.end(), [](State* lhs, State* rhs) { return lhs->trackId() < rhs->trackId(); }); + stable_sort(best.begin(), best.end(), order); // keep first state (best due to previous sorts) per track id - stream.erase( - unique(stream.begin(), stream.end(), [](State* lhs, State* rhs) { return lhs->track() == rhs->track(); }), - stream.end()); + auto same = [](Track* lhs, Track* rhs) { return lhs->trackId_ == rhs->trackId_; }; + best.erase(unique(best.begin(), best.end(), same), best.end()); } // updates state void KalmanFilter::update(State*& state) { // All variable names & equations come from Fruhwirth KF paper http://dx.doi.org/10.1016/0168-9002%2887%2990887-4", where F taken as unit matrix. Stub uncertainties projected onto (phi,z), assuming no correlations between r-phi & r-z planes. // stub phi residual wrt input helix - const double m0 = m0_->digi(state->m0()); + const double m0 = state->m0(); // stub z residual wrt input helix - const double m1 = m1_->digi(state->m1()); + const double m1 = state->m1(); // stub projected phi uncertainty squared); - const double v0 = v0_->digi(state->v0()); + const double v0 = state->v0(); // stub projected z uncertainty squared - const double v1 = v1_->digi(state->v1()); + const double v1 = state->v1(); + // Derivative of predicted stub coords wrt helix params: stub radius minus chosenRofPhi + const double H00 = state->H00(); + // Derivative of predicted stub coords wrt helix params: stub radius minus chosenRofZ + const double H12 = state->H12(); + updateRangeActual(VariableKF::m0, m0); + updateRangeActual(VariableKF::m1, m1); + updateRangeActual(VariableKF::v0, v0); + updateRangeActual(VariableKF::v1, v1); + updateRangeActual(VariableKF::H00, H00); + updateRangeActual(VariableKF::H12, H12); // helix inv2R wrt input helix - double x0 = x0_->digi(state->x0()); + double x0 = state->x0(); // helix phi at radius ChosenRofPhi wrt input helix - double x1 = x1_->digi(state->x1()); + double x1 = state->x1(); // helix cot(Theta) wrt input helix - double x2 = x2_->digi(state->x2()); + double x2 = state->x2(); // helix z at radius chosenRofZ wrt input helix - double x3 = x3_->digi(state->x3()); - // Derivative of predicted stub coords wrt helix params: stub radius minus chosenRofPhi - const double H00 = H00_->digi(state->H00()); - // Derivative of predicted stub coords wrt helix params: stub radius minus chosenRofZ - const double H12 = H12_->digi(state->H12()); + double x3 = state->x3(); // cov. matrix - double C00 = C00_->digi(state->C00()); - double C01 = C01_->digi(state->C01()); - double C11 = C11_->digi(state->C11()); - double C22 = C22_->digi(state->C22()); - double C23 = C23_->digi(state->C23()); - double C33 = C33_->digi(state->C33()); + double C00 = state->C00(); + double C01 = state->C01(); + double C11 = state->C11(); + double C22 = state->C22(); + double C23 = state->C23(); + double C33 = state->C33(); + // chi2s + double chi20 = state->chi20(); + double chi21 = state->chi21(); // stub phi residual wrt current state - const double r0C = x1_->digi(m0 - x1); - const double r0 = r0_->digi(r0C - x0 * H00); + const double r0C = digi(VariableKF::x1, m0 - x1); + const double r0 = digi(VariableKF::r0, r0C - x0 * H00); // stub z residual wrt current state - const double r1C = x3_->digi(m1 - x3); - const double r1 = r1_->digi(r1C - x2 * H12); + const double r1C = digi(VariableKF::x3, m1 - x3); + const double r1 = digi(VariableKF::r1, r1C - x2 * H12); // matrix S = H*C - const double S00 = S00_->digi(C01 + H00 * C00); - const double S01 = S01_->digi(C11 + H00 * C01); - const double S12 = S12_->digi(C23 + H12 * C22); - const double S13 = S13_->digi(C33 + H12 * C23); + const double S00 = digi(VariableKF::S00, C01 + H00 * C00); + const double S01 = digi(VariableKF::S01, C11 + H00 * C01); + const double S12 = digi(VariableKF::S12, C23 + H12 * C22); + const double S13 = digi(VariableKF::S13, C33 + H12 * C23); // Cov. matrix of predicted residuals R = V+HCHt = C+H*St - const double R00C = S01_->digi(v0 + S01); - const double R00 = R00_->digi(R00C + H00 * S00); - const double R11C = S13_->digi(v1 + S13); - const double R11 = R11_->digi(R11C + H12 * S12); - // imrpoved dynamic cancelling - const int msb0 = max(0, (int)ceil(log2(R00 / R00_->base()))); - const int msb1 = max(0, (int)ceil(log2(R11 / R11_->base()))); - const double R00Rough = R00Rough_->digi(R00 * pow(2., 16 - msb0)); - const double invR00Approx = invR00Approx_->digi(1. / R00Rough); - const double invR00Cor = invR00Cor_->digi(2. - invR00Approx * R00Rough); - const double invR00 = invR00_->digi(invR00Approx * invR00Cor * pow(2., 16 - msb0)); - const double R11Rough = R11Rough_->digi(R11 * pow(2., 16 - msb1)); - const double invR11Approx = invR11Approx_->digi(1. / R11Rough); - const double invR11Cor = invR11Cor_->digi(2. - invR11Approx * R11Rough); - const double invR11 = invR11_->digi(invR11Approx * invR11Cor * pow(2., 16 - msb1)); + const double R00 = digi(VariableKF::R00, v0 + S01 + H00 * S00); + const double R11 = digi(VariableKF::R11, v1 + S13 + H12 * S12); + // improved dynamic cancelling + const int msb0 = max(0, (int)ceil(log2(R00 / base(VariableKF::R00)))); + const int msb1 = max(0, (int)ceil(log2(R11 / base(VariableKF::R11)))); + const int shift0 = width(VariableKF::R00) - msb0; + const int shift1 = width(VariableKF::R11) - msb1; + const double R00Shifted = R00 * pow(2., shift0); + const double R11Shifted = R11 * pow(2., shift1); + const double R00Rough = digi(VariableKF::R00Rough, R00Shifted); + const double R11Rough = digi(VariableKF::R11Rough, R11Shifted); + const double invR00Approx = digi(VariableKF::invR00Approx, 1. / R00Rough); + const double invR11Approx = digi(VariableKF::invR11Approx, 1. / R11Rough); + const double invR00Cor = digi(VariableKF::invR00Cor, 2. - invR00Approx * R00Shifted); + const double invR11Cor = digi(VariableKF::invR11Cor, 2. - invR11Approx * R11Shifted); + const double invR00 = digi(VariableKF::invR00, invR00Approx * invR00Cor); + const double invR11 = digi(VariableKF::invR11, invR11Approx * invR11Cor); + // shift S to "undo" shifting of R + auto digiShifted = [](double val, double base) { return floor(val / base * 2. + 1.e-11) * base / 2.; }; + const double S00Shifted = digiShifted(S00 * pow(2., shift0), base(VariableKF::S00Shifted)); + const double S01Shifted = digiShifted(S01 * pow(2., shift0), base(VariableKF::S01Shifted)); + const double S12Shifted = digiShifted(S12 * pow(2., shift1), base(VariableKF::S12Shifted)); + const double S13Shifted = digiShifted(S13 * pow(2., shift1), base(VariableKF::S13Shifted)); // Kalman gain matrix K = S*R(inv) - const double K00 = K00_->digi(S00 * invR00); - const double K10 = K10_->digi(S01 * invR00); - const double K21 = K21_->digi(S12 * invR11); - const double K31 = K31_->digi(S13 * invR11); - // Updated helix params & their cov. matrix - x0 = x0_->digi(x0 + r0 * K00); - x1 = x1_->digi(x1 + r0 * K10); - x2 = x2_->digi(x2 + r1 * K21); - x3 = x3_->digi(x3 + r1 * K31); - C00 = C00_->digi(C00 - S00 * K00); - C01 = C01_->digi(C01 - S01 * K00); - C11 = C11_->digi(C11 - S01 * K10); - C22 = C22_->digi(C22 - S12 * K21); - C23 = C23_->digi(C23 - S13 * K21); - C33 = C33_->digi(C33 - S13 * K31); + const double K00 = digi(VariableKF::K00, S00Shifted * invR00); + const double K10 = digi(VariableKF::K10, S01Shifted * invR00); + const double K21 = digi(VariableKF::K21, S12Shifted * invR11); + const double K31 = digi(VariableKF::K31, S13Shifted * invR11); + // Updated helix params, their cov. matrix + x0 = digi(VariableKF::x0, x0 + r0 * K00); + x1 = digi(VariableKF::x1, x1 + r0 * K10); + x2 = digi(VariableKF::x2, x2 + r1 * K21); + x3 = digi(VariableKF::x3, x3 + r1 * K31); + C00 = digi(VariableKF::C00, C00 - S00 * K00); + C01 = digi(VariableKF::C01, C01 - S01 * K00); + C11 = digi(VariableKF::C11, C11 - S01 * K10); + C22 = digi(VariableKF::C22, C22 - S12 * K21); + C23 = digi(VariableKF::C23, C23 - S13 * K21); + C33 = digi(VariableKF::C33, C33 - S13 * K31); + // squared residuals + const double r0Shifted = digiShifted(r0 * pow(2., shift0), base(VariableKF::r0Shifted)); + const double r1Shifted = digiShifted(r1 * pow(2., shift1), base(VariableKF::r1Shifted)); + const double r02 = digi(VariableKF::r02, r0 * r0); + const double r12 = digi(VariableKF::r12, r1 * r1); + chi20 = digi(VariableKF::chi20, chi20 + r02 * invR00); + chi21 = digi(VariableKF::chi21, chi21 + r12 * invR11); + // update variable ranges to tune variable granularity + updateRangeActual(VariableKF::r0, r0); + updateRangeActual(VariableKF::r1, r1); + updateRangeActual(VariableKF::S00, S00); + updateRangeActual(VariableKF::S01, S01); + updateRangeActual(VariableKF::S12, S12); + updateRangeActual(VariableKF::S13, S13); + updateRangeActual(VariableKF::S00Shifted, S00Shifted); + updateRangeActual(VariableKF::S01Shifted, S01Shifted); + updateRangeActual(VariableKF::S12Shifted, S12Shifted); + updateRangeActual(VariableKF::S13Shifted, S13Shifted); + updateRangeActual(VariableKF::R00, R00); + updateRangeActual(VariableKF::R11, R11); + updateRangeActual(VariableKF::R00Rough, R00Rough); + updateRangeActual(VariableKF::R11Rough, R11Rough); + updateRangeActual(VariableKF::invR00Approx, invR00Approx); + updateRangeActual(VariableKF::invR11Approx, invR11Approx); + updateRangeActual(VariableKF::invR00Cor, invR00Cor); + updateRangeActual(VariableKF::invR11Cor, invR11Cor); + updateRangeActual(VariableKF::invR00, invR00); + updateRangeActual(VariableKF::invR11, invR11); + updateRangeActual(VariableKF::K00, K00); + updateRangeActual(VariableKF::K10, K10); + updateRangeActual(VariableKF::K21, K21); + updateRangeActual(VariableKF::K31, K31); + updateRangeActual(VariableKF::r0Shifted, r0Shifted); + updateRangeActual(VariableKF::r1Shifted, r1Shifted); + updateRangeActual(VariableKF::r02, r02); + updateRangeActual(VariableKF::r12, r12); + // range checks + const bool validX0 = inRange(VariableKF::x0, x0); + const bool validX1 = inRange(VariableKF::x1, x1); + const bool validX2 = inRange(VariableKF::x2, x2); + const bool validX3 = inRange(VariableKF::x3, x3); + // chi2 cut + const double dof = state->hitPattern().count() - 1; + const double chi2 = (chi20 + chi21) / 2.; + const bool validChi2 = chi2 < setup_->kfCutChi2() * dof; + if (!validX0 || !validX1 || !validX2 || !validX3 || !validChi2) { + state = nullptr; + return; + } // create updated state - states_.emplace_back(State(state, (initializer_list){x0, x1, x2, x3, C00, C11, C22, C33, C01, C23})); + states_.emplace_back(State(state, {x0, x1, x2, x3, chi20, chi21, C00, C11, C22, C33, C01, C23})); state = &states_.back(); - // update variable ranges to tune variable granularity - m0_->updateRangeActual(m0); - m1_->updateRangeActual(m1); - v0_->updateRangeActual(v0); - v1_->updateRangeActual(v1); - H00_->updateRangeActual(H00); - H12_->updateRangeActual(H12); - r0_->updateRangeActual(r0); - r1_->updateRangeActual(r1); - S00_->updateRangeActual(S00); - S01_->updateRangeActual(S01); - S12_->updateRangeActual(S12); - S13_->updateRangeActual(S13); - R00_->updateRangeActual(R00); - R11_->updateRangeActual(R11); - R00Rough_->updateRangeActual(R00Rough); - invR00Approx_->updateRangeActual(invR00Approx); - invR00Cor_->updateRangeActual(invR00Cor); - invR00_->updateRangeActual(invR00); - R11Rough_->updateRangeActual(R11Rough); - invR11Approx_->updateRangeActual(invR11Approx); - invR11Cor_->updateRangeActual(invR11Cor); - invR11_->updateRangeActual(invR11); - K00_->updateRangeActual(K00); - K10_->updateRangeActual(K10); - K21_->updateRangeActual(K21); - K31_->updateRangeActual(K31); - x0_->updateRangeActual(x0); - x1_->updateRangeActual(x1); - x2_->updateRangeActual(x2); - x3_->updateRangeActual(x3); - C00_->updateRangeActual(C00); - C01_->updateRangeActual(C01); - C11_->updateRangeActual(C11); - C22_->updateRangeActual(C22); - C23_->updateRangeActual(C23); - C33_->updateRangeActual(C33); + updateRangeActual(VariableKF::x0, x0); + updateRangeActual(VariableKF::x1, x1); + updateRangeActual(VariableKF::x2, x2); + updateRangeActual(VariableKF::x3, x3); + updateRangeActual(VariableKF::C00, C00); + updateRangeActual(VariableKF::C01, C01); + updateRangeActual(VariableKF::C11, C11); + updateRangeActual(VariableKF::C22, C22); + updateRangeActual(VariableKF::C23, C23); + updateRangeActual(VariableKF::C33, C33); + updateRangeActual(VariableKF::chi20, chi20); + updateRangeActual(VariableKF::chi21, chi21); } // remove and return first element of deque, returns nullptr if empty @@ -440,15 +554,4 @@ namespace trackerTFP { return t; } - // remove and return first element of vector, returns nullptr if empty - template - T* KalmanFilter::pop_front(vector& ts) const { - T* t = nullptr; - if (!ts.empty()) { - t = ts.front(); - ts.erase(ts.begin()); - } - return t; - } - } // namespace trackerTFP diff --git a/L1Trigger/TrackerTFP/src/KalmanFilterFormats.cc b/L1Trigger/TrackerTFP/src/KalmanFilterFormats.cc index fe3625a32a566..749e692d2d614 100644 --- a/L1Trigger/TrackerTFP/src/KalmanFilterFormats.cc +++ b/L1Trigger/TrackerTFP/src/KalmanFilterFormats.cc @@ -16,36 +16,33 @@ using namespace tt; namespace trackerTFP { constexpr auto variableKFstrs_ = { - "x0", "x1", "x2", "x3", "H00", "H12", "m0", "m1", "v0", - "v1", "r0", "r1", "S00", "S01", "S12", "S13", "K00", "K10", - "K21", "K31", "R00", "R11", "R00Rough", "R11Rough", "invR00Approx", "invR11Approx", "invR00Cor", - "invR11Cor", "invR00", "invR11", "C00", "C01", "C11", "C22", "C23", "C33"}; + "x0", "x1", "x2", "x3", "H00", "H12", "m0", "m1", + "v0", "v1", "r0", "r1", "S00", "S01", "S12", "S13", + "S00Shifted", "S01Shifted", "S12Shifted", "S13Shifted", "K00", "K10", "K21", "K31", + "R00", "R11", "R00Rough", "R11Rough", "invR00Approx", "invR11Approx", "invR00Cor", "invR11Cor", + "invR00", "invR11", "C00", "C01", "C11", "C22", "C23", "C33", + "r0Shifted", "r1Shifted", "r02", "r12", "chi20", "chi21"}; void KalmanFilterFormats::endJob() { const int wName = strlen(*max_element(variableKFstrs_.begin(), variableKFstrs_.end(), [](const auto& a, const auto& b) { return strlen(a) < strlen(b); })); - static constexpr int wWidth = 3; - for (VariableKF v = VariableKF::begin; v != VariableKF::end; v = VariableKF(+v + 1)) { - const pair& range = format(v).rangeActual(); - const double r = format(v).twos() ? max(abs(range.first), abs(range.second)) * 2. : range.second; - const int width = ceil(log2(r / format(v).base())); - cout << setw(wName) << *next(variableKFstrs_.begin(), +v) << ": " << setw(wWidth) << width << " " << setw(wWidth) - << format(v).width() << " | " << setw(wWidth) << format(v).width() - width << endl; + for (VariableKF v = VariableKF::begin; v != VariableKF::dH; v = VariableKF(+v + 1)) { + const double r = + format(v).twos() ? std::max(std::abs(format(v).min()), std::abs(format(v).max())) * 2. : format(v).max(); + const int delta = format(v).width() - ceil(log2(r / format(v).base())); + cout << setw(wName) << *next(variableKFstrs_.begin(), +v) << ": "; + cout << setw(3) << (delta == -2147483648 ? "-" : to_string(delta)) << endl; } } - KalmanFilterFormats::KalmanFilterFormats() : iConfig_(), dataFormats_(nullptr), setup_(nullptr) { + KalmanFilterFormats::KalmanFilterFormats(const ParameterSet& iConfig) : iConfig_(iConfig) { formats_.reserve(+VariableKF::end); } - KalmanFilterFormats::KalmanFilterFormats(const ParameterSet& iConfig, const DataFormats* dataFormats) - : iConfig_(dataFormats->hybrid() ? iConfig.getParameter("hybrid") - : iConfig.getParameter("tmtt")), - dataFormats_(dataFormats), - setup_(dataFormats_->setup()) { - formats_.reserve(+VariableKF::end); + void KalmanFilterFormats::beginRun(const DataFormats* dataFormats) { + dataFormats_ = dataFormats; fillFormats(); } @@ -56,13 +53,16 @@ namespace trackerTFP { fillFormats<++it>(); } - DataFormatKF::DataFormatKF(const VariableKF& v, bool twos) + DataFormatKF::DataFormatKF(const VariableKF& v, bool twos, const ParameterSet& iConfig) : v_(v), twos_(twos), + enableIntegerEmulation_(iConfig.getParameter("EnableIntegerEmulation")), width_(0), base_(1.), range_(0.), - rangeActual_(numeric_limits::max(), numeric_limits::lowest()) {} + min_(numeric_limits::max()), + abs_(numeric_limits::max()), + max_(numeric_limits::lowest()) {} // returns false if data format would oferflow for this double value bool DataFormatKF::inRange(double d) const { @@ -72,22 +72,25 @@ namespace trackerTFP { } void DataFormatKF::updateRangeActual(double d) { - rangeActual_ = make_pair(min(rangeActual_.first, d), max(rangeActual_.second, d)); - if (!inRange(d)) { + min_ = std::min(min_, d); + abs_ = std::min(abs_, std::abs(d)); + max_ = std::max(max_, d); + if (enableIntegerEmulation_ && !inRange(d)) { string v = *next(variableKFstrs_.begin(), +v_); cms::Exception exception("out_of_range"); exception.addContext("trackerTFP:DataFormatKF::updateRangeActual"); exception << "Variable " << v << " = " << d << " is out of range " << (twos_ ? -range_ / 2. : 0) << " to " << (twos_ ? range_ / 2. : range_) << "." << endl; if (twos_ || d >= 0.) - exception.addAdditionalInfo("Consider raising BaseShift" + v + " in KalmnaFilterFormats_cfi.py."); + exception.addAdditionalInfo("Consider raising BaseShift" + v + " in KalmanFilterFormats_cfi.py."); + exception.addAdditionalInfo("Consider disabling integer emulation in KalmanFilterFormats_cfi.py."); throw exception; } } template <> FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig) - : DataFormatKF(VariableKF::x0, true) { + : DataFormatKF(VariableKF::x0, true, iConfig) { const DataFormat& input = dataFormats->format(Variable::inv2R, Process::kf); const int baseShift = iConfig.getParameter("BaseShiftx0"); base_ = pow(2, baseShift) * input.base(); @@ -97,7 +100,7 @@ namespace trackerTFP { template <> FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig) - : DataFormatKF(VariableKF::x1, true) { + : DataFormatKF(VariableKF::x1, true, iConfig) { const DataFormat& input = dataFormats->format(Variable::phiT, Process::kf); const int baseShift = iConfig.getParameter("BaseShiftx1"); base_ = pow(2, baseShift) * input.base(); @@ -107,7 +110,7 @@ namespace trackerTFP { template <> FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig) - : DataFormatKF(VariableKF::x2, true) { + : DataFormatKF(VariableKF::x2, true, iConfig) { const DataFormat& input = dataFormats->format(Variable::cot, Process::kf); const int baseShift = iConfig.getParameter("BaseShiftx2"); base_ = pow(2, baseShift) * input.base(); @@ -117,7 +120,7 @@ namespace trackerTFP { template <> FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig) - : DataFormatKF(VariableKF::x3, true) { + : DataFormatKF(VariableKF::x3, true, iConfig) { const DataFormat& input = dataFormats->format(Variable::zT, Process::kf); const int baseShift = iConfig.getParameter("BaseShiftx3"); base_ = pow(2, baseShift) * input.base(); @@ -127,64 +130,65 @@ namespace trackerTFP { template <> FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig) - : DataFormatKF(VariableKF::H00, true) { - const DataFormat& kfin = dataFormats->format(Variable::r, Process::kfin); - base_ = kfin.base(); - width_ = kfin.width(); - range_ = kfin.range(); + : DataFormatKF(VariableKF::H00, true, iConfig) { + const DataFormat& ctb = dataFormats->format(Variable::r, Process::ctb); + base_ = ctb.base(); + width_ = ctb.width(); + calcRange(); } template <> FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig) - : DataFormatKF(VariableKF::H12, true) { + : DataFormatKF(VariableKF::H12, true, iConfig) { const Setup* setup = dataFormats->setup(); - const DataFormat& kfin = dataFormats->format(Variable::r, Process::kfin); - base_ = kfin.base(); - range_ = 2. * max(abs(setup->outerRadius() - setup->chosenRofZ()), abs(setup->innerRadius() - setup->chosenRofZ())); + const DataFormat& ctb = dataFormats->format(Variable::r, Process::ctb); + base_ = ctb.base(); + range_ = 2. * setup->maxRz(); width_ = ceil(log2(range_ / base_)); + calcRange(); } template <> FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig) - : DataFormatKF(VariableKF::m0, true) { - const DataFormat& kfin = dataFormats->format(Variable::phi, Process::kfin); - base_ = kfin.base(); - width_ = kfin.width(); - range_ = kfin.range(); + : DataFormatKF(VariableKF::m0, true, iConfig) { + const DataFormat& ctb = dataFormats->format(Variable::phi, Process::ctb); + base_ = ctb.base(); + width_ = ctb.width(); + calcRange(); } template <> FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig) - : DataFormatKF(VariableKF::m1, true) { - const DataFormat& kfin = dataFormats->format(Variable::z, Process::kfin); - base_ = kfin.base(); - width_ = kfin.width(); - range_ = kfin.range(); + : DataFormatKF(VariableKF::m1, true, iConfig) { + const DataFormat& ctb = dataFormats->format(Variable::z, Process::ctb); + base_ = ctb.base(); + width_ = ctb.width(); + calcRange(); } template <> FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig) - : DataFormatKF(VariableKF::v0, false) { - const DataFormat& x1 = dataFormats->format(Variable::phiT, Process::kf); - const int baseShift = iConfig.getParameter("BaseShiftv0"); - base_ = pow(2., baseShift) * x1.base() * x1.base(); - width_ = dataFormats->setup()->widthDSPbu(); - calcRange(); + : DataFormatKF(VariableKF::v0, false, iConfig) { + const DataFormat& dPhi = dataFormats->format(Variable::dPhi, Process::ctb); + const FormatKF S01(dataFormats, iConfig); + range_ = dPhi.range() * dPhi.range() * 4.; + base_ = S01.base(); + calcWidth(); } template <> FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig) - : DataFormatKF(VariableKF::v1, true) { - const DataFormat& x3 = dataFormats->format(Variable::zT, Process::kf); - const int baseShift = iConfig.getParameter("BaseShiftv1"); - base_ = pow(2., baseShift) * x3.base() * x3.base(); - width_ = dataFormats->setup()->widthDSPbu(); - calcRange(); + : DataFormatKF(VariableKF::v1, false, iConfig) { + const DataFormat& dZ = dataFormats->format(Variable::dZ, Process::ctb); + const FormatKF S13(dataFormats, iConfig); + range_ = dZ.range() * dZ.range() * 4.; + base_ = S13.base(); + calcWidth(); } template <> FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig) - : DataFormatKF(VariableKF::r0, true) { + : DataFormatKF(VariableKF::r0, true, iConfig) { const DataFormat& x1 = dataFormats->format(Variable::phiT, Process::kf); const int baseShift = iConfig.getParameter("BaseShiftr0"); base_ = pow(2., baseShift) * x1.base(); @@ -194,7 +198,7 @@ namespace trackerTFP { template <> FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig) - : DataFormatKF(VariableKF::r1, true) { + : DataFormatKF(VariableKF::r1, true, iConfig) { const DataFormat& x3 = dataFormats->format(Variable::zT, Process::kf); const int baseShift = iConfig.getParameter("BaseShiftr1"); base_ = pow(2., baseShift) * x3.base(); @@ -204,129 +208,171 @@ namespace trackerTFP { template <> FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig) - : DataFormatKF(VariableKF::S00, true) { + : DataFormatKF(VariableKF::S00, true, iConfig) { const DataFormat& x0 = dataFormats->format(Variable::inv2R, Process::kf); const DataFormat& x1 = dataFormats->format(Variable::phiT, Process::kf); const int baseShift = iConfig.getParameter("BaseShiftS00"); base_ = pow(2., baseShift) * x0.base() * x1.base(); - width_ = dataFormats->setup()->widthDSPbb(); + width_ = dataFormats->setup()->widthDSPab(); calcRange(); } template <> FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig) - : DataFormatKF(VariableKF::S01, true) { + : DataFormatKF(VariableKF::S01, true, iConfig) { const DataFormat& x1 = dataFormats->format(Variable::phiT, Process::kf); const int baseShift = iConfig.getParameter("BaseShiftS01"); base_ = pow(2., baseShift) * x1.base() * x1.base(); - width_ = dataFormats->setup()->widthDSPbb(); + width_ = dataFormats->setup()->widthDSPab(); calcRange(); } template <> FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig) - : DataFormatKF(VariableKF::S12, true) { + : DataFormatKF(VariableKF::S12, true, iConfig) { const DataFormat& x2 = dataFormats->format(Variable::cot, Process::kf); const DataFormat& x3 = dataFormats->format(Variable::zT, Process::kf); const int baseShift = iConfig.getParameter("BaseShiftS12"); base_ = pow(2., baseShift) * x2.base() * x3.base(); - width_ = dataFormats->setup()->widthDSPbb(); + width_ = dataFormats->setup()->widthDSPab(); calcRange(); } template <> FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig) - : DataFormatKF(VariableKF::S13, true) { + : DataFormatKF(VariableKF::S13, true, iConfig) { const DataFormat& x3 = dataFormats->format(Variable::zT, Process::kf); const int baseShift = iConfig.getParameter("BaseShiftS13"); base_ = pow(2., baseShift) * x3.base() * x3.base(); - width_ = dataFormats->setup()->widthDSPbb(); + width_ = dataFormats->setup()->widthDSPab(); + calcRange(); + } + + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig) + : DataFormatKF(VariableKF::S00Shifted, true, iConfig) { + const DataFormat& x0 = dataFormats->format(Variable::inv2R, Process::kf); + const DataFormat& x1 = dataFormats->format(Variable::phiT, Process::kf); + const int baseShift = iConfig.getParameter("BaseShiftS00Shifted"); + base_ = pow(2., baseShift) * x0.base() * x1.base(); + width_ = dataFormats->setup()->widthDSPab(); + calcRange(); + } + + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig) + : DataFormatKF(VariableKF::S01Shifted, true, iConfig) { + const DataFormat& x1 = dataFormats->format(Variable::phiT, Process::kf); + const int baseShift = iConfig.getParameter("BaseShiftS01Shifted"); + base_ = pow(2., baseShift) * x1.base() * x1.base(); + width_ = dataFormats->setup()->widthDSPab(); + calcRange(); + } + + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig) + : DataFormatKF(VariableKF::S12Shifted, true, iConfig) { + const DataFormat& x2 = dataFormats->format(Variable::cot, Process::kf); + const DataFormat& x3 = dataFormats->format(Variable::zT, Process::kf); + const int baseShift = iConfig.getParameter("BaseShiftS12Shifted"); + base_ = pow(2., baseShift) * x2.base() * x3.base(); + width_ = dataFormats->setup()->widthDSPab(); + calcRange(); + } + + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig) + : DataFormatKF(VariableKF::S13Shifted, true, iConfig) { + const DataFormat& x3 = dataFormats->format(Variable::zT, Process::kf); + const int baseShift = iConfig.getParameter("BaseShiftS13Shifted"); + base_ = pow(2., baseShift) * x3.base() * x3.base(); + width_ = dataFormats->setup()->widthDSPab(); calcRange(); } template <> FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig) - : DataFormatKF(VariableKF::K00, true) { + : DataFormatKF(VariableKF::K00, true, iConfig) { const DataFormat& x0 = dataFormats->format(Variable::inv2R, Process::kf); const DataFormat& x1 = dataFormats->format(Variable::phiT, Process::kf); const int baseShift = iConfig.getParameter("BaseShiftK00"); base_ = pow(2., baseShift) * x0.base() / x1.base(); - width_ = dataFormats->setup()->widthDSPab(); + width_ = dataFormats->setup()->widthDSPbb(); calcRange(); } template <> FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig) - : DataFormatKF(VariableKF::K10, true) { + : DataFormatKF(VariableKF::K10, true, iConfig) { const int baseShift = iConfig.getParameter("BaseShiftK10"); base_ = pow(2., baseShift); - width_ = dataFormats->setup()->widthDSPab(); + width_ = dataFormats->setup()->widthDSPbb(); calcRange(); } template <> FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig) - : DataFormatKF(VariableKF::K21, true) { + : DataFormatKF(VariableKF::K21, true, iConfig) { const DataFormat& x2 = dataFormats->format(Variable::cot, Process::kf); const DataFormat& x3 = dataFormats->format(Variable::zT, Process::kf); const int baseShift = iConfig.getParameter("BaseShiftK21"); base_ = pow(2., baseShift) * x2.base() / x3.base(); - width_ = dataFormats->setup()->widthDSPab(); + width_ = dataFormats->setup()->widthDSPbb(); calcRange(); } template <> FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig) - : DataFormatKF(VariableKF::K31, true) { + : DataFormatKF(VariableKF::K31, true, iConfig) { const int baseShift = iConfig.getParameter("BaseShiftK31"); base_ = pow(2., baseShift); - width_ = dataFormats->setup()->widthDSPab(); + width_ = dataFormats->setup()->widthDSPbb(); calcRange(); } template <> FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig) - : DataFormatKF(VariableKF::R00, false) { + : DataFormatKF(VariableKF::R00, false, iConfig) { const DataFormat& x1 = dataFormats->format(Variable::phiT, Process::kf); const int baseShift = iConfig.getParameter("BaseShiftR00"); + width_ = iConfig.getParameter("WidthR00"); base_ = pow(2., baseShift) * x1.base() * x1.base(); - width_ = dataFormats->setup()->widthDSPbu(); calcRange(); } template <> FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig) - : DataFormatKF(VariableKF::R11, false) { + : DataFormatKF(VariableKF::R11, false, iConfig) { const DataFormat& x3 = dataFormats->format(Variable::zT, Process::kf); const int baseShift = iConfig.getParameter("BaseShiftR11"); + width_ = iConfig.getParameter("WidthR11"); base_ = pow(2., baseShift) * x3.base() * x3.base(); - width_ = dataFormats->setup()->widthDSPbu(); calcRange(); } template <> FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig) - : DataFormatKF(VariableKF::R00Rough, false) { + : DataFormatKF(VariableKF::R00Rough, false, iConfig) { const FormatKF R00(dataFormats, iConfig); width_ = dataFormats->setup()->widthAddrBRAM18(); range_ = R00.range(); - const int baseShift = R00.width() - width_; + const int baseShift = R00.width() - width_ - 1; base_ = pow(2., baseShift) * R00.base(); } template <> FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig) - : DataFormatKF(VariableKF::R11Rough, false) { + : DataFormatKF(VariableKF::R11Rough, false, iConfig) { const FormatKF R11(dataFormats, iConfig); width_ = dataFormats->setup()->widthAddrBRAM18(); range_ = R11.range(); - const int baseShift = R11.width() - width_; + const int baseShift = R11.width() - width_ - 1; base_ = pow(2., baseShift) * R11.base(); } template <> FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig) - : DataFormatKF(VariableKF::invR00Approx, false) { + : DataFormatKF(VariableKF::invR00Approx, false, iConfig) { const DataFormat& x1 = dataFormats->format(Variable::phiT, Process::kf); const int baseShift = iConfig.getParameter("BaseShiftInvR00Approx"); base_ = pow(2., baseShift) / x1.base() / x1.base(); @@ -336,7 +382,7 @@ namespace trackerTFP { template <> FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig) - : DataFormatKF(VariableKF::invR11Approx, false) { + : DataFormatKF(VariableKF::invR11Approx, false, iConfig) { const DataFormat& x3 = dataFormats->format(Variable::zT, Process::kf); const int baseShift = iConfig.getParameter("BaseShiftInvR11Approx"); base_ = pow(2., baseShift) / x3.base() / x3.base(); @@ -346,102 +392,256 @@ namespace trackerTFP { template <> FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig) - : DataFormatKF(VariableKF::invR00Cor, false) { + : DataFormatKF(VariableKF::invR00Cor, false, iConfig) { const int baseShift = iConfig.getParameter("BaseShiftInvR00Cor"); base_ = pow(2., baseShift); - width_ = dataFormats->setup()->widthDSPbu(); + width_ = dataFormats->setup()->widthDSPau(); calcRange(); } template <> FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig) - : DataFormatKF(VariableKF::invR11Cor, false) { + : DataFormatKF(VariableKF::invR11Cor, false, iConfig) { const int baseShift = iConfig.getParameter("BaseShiftInvR11Cor"); base_ = pow(2., baseShift); - width_ = dataFormats->setup()->widthDSPbu(); + width_ = dataFormats->setup()->widthDSPau(); calcRange(); } template <> FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig) - : DataFormatKF(VariableKF::invR00, false) { + : DataFormatKF(VariableKF::invR00, false, iConfig) { const DataFormat& x1 = dataFormats->format(Variable::phiT, Process::kf); const int baseShift = iConfig.getParameter("BaseShiftInvR00"); base_ = pow(2., baseShift) / x1.base() / x1.base(); - width_ = dataFormats->setup()->widthDSPau(); + width_ = dataFormats->setup()->widthDSPbu(); calcRange(); } template <> FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig) - : DataFormatKF(VariableKF::invR11, false) { + : DataFormatKF(VariableKF::invR11, false, iConfig) { const DataFormat& x3 = dataFormats->format(Variable::zT, Process::kf); const int baseShift = iConfig.getParameter("BaseShiftInvR11"); base_ = pow(2., baseShift) / x3.base() / x3.base(); - width_ = dataFormats->setup()->widthDSPau(); + width_ = dataFormats->setup()->widthDSPbu(); calcRange(); } template <> FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig) - : DataFormatKF(VariableKF::C00, false) { + : DataFormatKF(VariableKF::C00, false, iConfig) { const DataFormat& x0 = dataFormats->format(Variable::inv2R, Process::kf); const int baseShift = iConfig.getParameter("BaseShiftC00"); + width_ = iConfig.getParameter("WidthC00"); base_ = pow(2., baseShift) * x0.base() * x0.base(); - width_ = dataFormats->setup()->widthDSPbu(); calcRange(); } template <> FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig) - : DataFormatKF(VariableKF::C01, true) { + : DataFormatKF(VariableKF::C01, true, iConfig) { const DataFormat& x0 = dataFormats->format(Variable::inv2R, Process::kf); const DataFormat& x1 = dataFormats->format(Variable::phiT, Process::kf); const int baseShift = iConfig.getParameter("BaseShiftC01"); + width_ = iConfig.getParameter("WidthC01"); base_ = pow(2., baseShift) * x0.base() * x1.base(); - width_ = dataFormats->setup()->widthDSPbb(); calcRange(); } template <> FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig) - : DataFormatKF(VariableKF::C11, false) { + : DataFormatKF(VariableKF::C11, false, iConfig) { const DataFormat& x1 = dataFormats->format(Variable::phiT, Process::kf); const int baseShift = iConfig.getParameter("BaseShiftC11"); + width_ = iConfig.getParameter("WidthC11"); base_ = pow(2., baseShift) * x1.base() * x1.base(); - width_ = dataFormats->setup()->widthDSPbu(); calcRange(); } template <> FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig) - : DataFormatKF(VariableKF::C22, false) { + : DataFormatKF(VariableKF::C22, false, iConfig) { const DataFormat& x2 = dataFormats->format(Variable::cot, Process::kf); const int baseShift = iConfig.getParameter("BaseShiftC22"); + width_ = iConfig.getParameter("WidthC22"); base_ = pow(2., baseShift) * x2.base() * x2.base(); - width_ = dataFormats->setup()->widthDSPbu(); calcRange(); } template <> FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig) - : DataFormatKF(VariableKF::C23, true) { + : DataFormatKF(VariableKF::C23, true, iConfig) { const DataFormat& x2 = dataFormats->format(Variable::cot, Process::kf); const DataFormat& x3 = dataFormats->format(Variable::zT, Process::kf); const int baseShift = iConfig.getParameter("BaseShiftC23"); + width_ = iConfig.getParameter("WidthC23"); base_ = pow(2., baseShift) * x2.base() * x3.base(); - width_ = dataFormats->setup()->widthDSPbb(); calcRange(); } template <> FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig) - : DataFormatKF(VariableKF::C33, false) { + : DataFormatKF(VariableKF::C33, false, iConfig) { const DataFormat& x3 = dataFormats->format(Variable::zT, Process::kf); const int baseShift = iConfig.getParameter("BaseShiftC33"); + width_ = iConfig.getParameter("WidthC33"); + base_ = pow(2., baseShift) * x3.base() * x3.base(); + calcRange(); + } + + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig) + : DataFormatKF(VariableKF::r0Shifted, true, iConfig) { + const DataFormat& x1 = dataFormats->format(Variable::phiT, Process::kf); + const int baseShift = iConfig.getParameter("BaseShiftr0Shifted"); + base_ = pow(2., baseShift) * x1.base(); + width_ = dataFormats->setup()->widthDSPbb(); + calcRange(); + } + + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig) + : DataFormatKF(VariableKF::r1Shifted, true, iConfig) { + const DataFormat& x3 = dataFormats->format(Variable::zT, Process::kf); + const int baseShift = iConfig.getParameter("BaseShiftr1Shifted"); + base_ = pow(2., baseShift) * x3.base(); + width_ = dataFormats->setup()->widthDSPbb(); + calcRange(); + } + + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig) + : DataFormatKF(VariableKF::r02, false, iConfig) { + const DataFormat& x1 = dataFormats->format(Variable::phiT, Process::kf); + const int baseShift = iConfig.getParameter("BaseShiftr02"); + base_ = pow(2., baseShift) * x1.base() * x1.base(); + width_ = dataFormats->setup()->widthDSPbu(); + calcRange(); + } + + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig) + : DataFormatKF(VariableKF::r12, false, iConfig) { + const DataFormat& x3 = dataFormats->format(Variable::zT, Process::kf); + const int baseShift = iConfig.getParameter("BaseShiftr12"); base_ = pow(2., baseShift) * x3.base() * x3.base(); width_ = dataFormats->setup()->widthDSPbu(); calcRange(); } + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig) + : DataFormatKF(VariableKF::chi20, false, iConfig) { + const int baseShift = iConfig.getParameter("BaseShiftchi20"); + base_ = pow(2., baseShift); + width_ = dataFormats->setup()->widthDSPbu(); + calcRange(); + } + + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const edm::ParameterSet& iConfig) + : DataFormatKF(VariableKF::chi21, false, iConfig) { + const int baseShift = iConfig.getParameter("BaseShiftchi21"); + base_ = pow(2., baseShift); + width_ = dataFormats->setup()->widthDSPbu(); + calcRange(); + } + + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const ParameterSet& iConfig) + : DataFormatKF(VariableKF::end, false, iConfig) { + const Setup* setup = dataFormats->setup(); + const DataFormat& ctb = dataFormats->format(Variable::r, Process::ctb); + width_ = setup->widthAddrBRAM18(); + range_ = setup->outerRadius() - setup->innerRadius(); + base_ = ctb.base() * pow(2, ceil(log2(range_ / ctb.base())) - width_); + } + + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const ParameterSet& iConfig) + : DataFormatKF(VariableKF::end, false, iConfig) { + const FormatKF H00(dataFormats, iConfig); + const Setup* setup = dataFormats->setup(); + width_ = setup->widthDSPbu(); + range_ = 1. / setup->kfMinSeedDeltaR(); + const int baseShift = ceil(log2(range_ * pow(2., -width_) * H00.base())); + base_ = pow(2., baseShift) / H00.base(); + } + + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const ParameterSet& iConfig) + : DataFormatKF(VariableKF::end, false, iConfig) { + const FormatKF H00(dataFormats, iConfig); + const Setup* setup = dataFormats->setup(); + width_ = setup->widthDSPbu(); + range_ = 1. / pow(setup->kfMinSeedDeltaR(), 2); + const double baseH2 = H00.base() * H00.base(); + const int baseShift = ceil(log2(range_ * pow(2., -width_) * baseH2)); + base_ = pow(2., baseShift) / baseH2; + } + + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const ParameterSet& iConfig) + : DataFormatKF(VariableKF::end, false, iConfig) { + const FormatKF H00(dataFormats, iConfig); + base_ = H00.base() * H00.base(); + } + + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const ParameterSet& iConfig) + : DataFormatKF(VariableKF::end, true, iConfig) { + const FormatKF H00(dataFormats, iConfig); + const FormatKF m0(dataFormats, iConfig); + base_ = H00.base() * m0.base(); + } + + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const ParameterSet& iConfig) + : DataFormatKF(VariableKF::end, true, iConfig) { + const FormatKF H12(dataFormats, iConfig); + const FormatKF m1(dataFormats, iConfig); + base_ = H12.base() * m1.base(); + } + + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const ParameterSet& iConfig) + : DataFormatKF(VariableKF::end, false, iConfig) { + const Setup* setup = dataFormats->setup(); + const FormatKF H00(dataFormats, iConfig); + const FormatKF v0(dataFormats, iConfig); + width_ = setup->widthDSPab(); + base_ = H00.base() * v0.base() * pow(2, H00.width() + v0.width() - width_); + } + + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const ParameterSet& iConfig) + : DataFormatKF(VariableKF::end, false, iConfig) { + const FormatKF H12(dataFormats, iConfig); + const Setup* setup = dataFormats->setup(); + const FormatKF v1(dataFormats, iConfig); + width_ = setup->widthDSPab(); + base_ = H12.base() * v1.base() * pow(2, H12.width() + v1.width() - width_); + } + + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const ParameterSet& iConfig) + : DataFormatKF(VariableKF::end, false, iConfig) { + const Setup* setup = dataFormats->setup(); + const FormatKF H00(dataFormats, iConfig); + const FormatKF v0(dataFormats, iConfig); + width_ = setup->widthDSPau(); + base_ = H00.base() * H00.base() * v0.base() * pow(2, 2 * H00.width() + v0.width() - width_); + } + + template <> + FormatKF::FormatKF(const DataFormats* dataFormats, const ParameterSet& iConfig) + : DataFormatKF(VariableKF::end, false, iConfig) { + const Setup* setup = dataFormats->setup(); + const FormatKF H12(dataFormats, iConfig); + const FormatKF v1(dataFormats, iConfig); + width_ = setup->widthDSPau(); + base_ = H12.base() * H12.base() * v1.base() * pow(2, 2 * H12.width() + v1.width() - width_); + } + } // namespace trackerTFP diff --git a/L1Trigger/TrackerTFP/src/KalmanFilterFormatsRcd.cc b/L1Trigger/TrackerTFP/src/KalmanFilterFormatsRcd.cc deleted file mode 100644 index 347bd9e9db443..0000000000000 --- a/L1Trigger/TrackerTFP/src/KalmanFilterFormatsRcd.cc +++ /dev/null @@ -1,4 +0,0 @@ -#include "L1Trigger/TrackerTFP/interface/KalmanFilterFormatsRcd.h" -#include "FWCore/Framework/interface/eventsetuprecord_registration_macro.h" - -EVENTSETUP_RECORD_REG(trackerTFP::KalmanFilterFormatsRcd); diff --git a/L1Trigger/TrackerTFP/src/LayerEncoding.cc b/L1Trigger/TrackerTFP/src/LayerEncoding.cc index 0687987c3bd25..9fb9da96bb725 100644 --- a/L1Trigger/TrackerTFP/src/LayerEncoding.cc +++ b/L1Trigger/TrackerTFP/src/LayerEncoding.cc @@ -15,17 +15,15 @@ namespace trackerTFP { LayerEncoding::LayerEncoding(const DataFormats* dataFormats) : setup_(dataFormats->setup()), dataFormats_(dataFormats), - zT_(&dataFormats->format(Variable::zT, Process::zht)), - cot_(&dataFormats->format(Variable::cot, Process::zht)), - layerEncoding_(setup_->numSectorsEta(), - vector>>(pow(2, zT_->width()), vector>(pow(2, cot_->width())))), - layerEncodingMap_(setup_->numSectorsEta(), - vector>>( - pow(2, zT_->width()), vector>(pow(2, cot_->width())))), - maybeLayer_(setup_->numSectorsEta(), - vector>>(pow(2, zT_->width()), vector>(pow(2, cot_->width())))) { + zT_(&dataFormats->format(Variable::zT, Process::gp)), + layerEncoding_(vector>(pow(2, zT_->width()))), + maybePattern_(vector(pow(2, zT_->width()), TTBV(0, setup_->numLayers()))), + nullLE_(setup_->numLayers(), 0), + nullMP_(0, setup_->numLayers()) { // number of boundaries of fiducial area in r-z plane for a given set of rough r-z track parameter static constexpr int boundaries = 2; + // z at radius chosenRofZ wrt zT of sectorZT of this bin boundaries + const vector z0s = {-setup_->beamWindowZ(), setup_->beamWindowZ()}; // find unique sensor mouldes in r-z // allowed distance in r and z in cm between modules to consider them not unique static constexpr double delta = 1.e-3; @@ -38,136 +36,85 @@ namespace trackerTFP { auto equalRZ = [](const SensorModule* lhs, const SensorModule* rhs) { return abs(lhs->r() - rhs->r()) < delta && abs(lhs->z() - rhs->z()) < delta; }; + stable_sort(sensorModules.begin(), sensorModules.end(), smallerZ); stable_sort(sensorModules.begin(), sensorModules.end(), smallerR); + sensorModules.erase(unique(sensorModules.begin(), sensorModules.end(), equalRZ), sensorModules.end()); stable_sort(sensorModules.begin(), sensorModules.end(), smallerZ); sensorModules.erase(unique(sensorModules.begin(), sensorModules.end(), equalRZ), sensorModules.end()); // find set of moudles for each set of rough r-z track parameter - // loop over eta sectors - for (int binEta = 0; binEta < setup_->numSectorsEta(); binEta++) { - // cotTheta of eta sector centre - const double sectorCot = (sinh(setup_->boundarieEta(binEta + 1)) + sinh(setup_->boundarieEta(binEta))) / 2.; - // z at radius choenRofZ of eta sector centre - const double sectorZT = setup_->chosenRofZ() * sectorCot; - // loop over bins in zT - for (int binZT = 0; binZT < pow(2, zT_->width()); binZT++) { - // z at radius chosenRofZ wrt zT of sectorZT of this bin centre - const double zT = zT_->floating(zT_->toSigned(binZT)); - // z at radius chosenRofZ wrt zT of sectorZT of this bin boundaries - const vector zTs = {sectorZT + zT - zT_->base() / 2., sectorZT + zT + zT_->base() / 2.}; - // loop over bins in cotTheta - for (int binCot = 0; binCot < pow(2, cot_->width()); binCot++) { - // cotTheta wrt sectorCot of this bin centre - const double cot = cot_->floating(cot_->toSigned(binCot)); - // layer ids crossed by left and right rough r-z parameter shape boundaries - vector> layers(boundaries); - map& layermaps = layerEncodingMap_[binEta][binZT][binCot]; - // cotTheta wrt sectorCot of this bin boundaries - const vector cots = {sectorCot + cot - cot_->base() / 2., sectorCot + cot + cot_->base() / 2.}; - // loop over all unique modules - for (const SensorModule* sm : sensorModules) { - // check if module is crossed by left and right rough r-z parameter shape boundaries - for (int i = 0; i < boundaries; i++) { - const int j = boundaries - i - 1; - const double zTi = zTs[sm->r() > setup_->chosenRofZ() ? i : j]; - const double coti = cots[sm->r() > setup_->chosenRofZ() ? j : i]; - // distance between module and boundary in moudle tilt angle direction - const double d = - (zTi - sm->z() + (sm->r() - setup_->chosenRofZ()) * coti) / (sm->cosTilt() - sm->sinTilt() * coti); - // compare distance with module size and add module layer id to layers if module is crossed - if (abs(d) < sm->numColumns() * sm->pitchCol() / 2.) { - layers[i].insert(sm->layerId()); - layermaps[sm->layerId()] = sm; - } - } - } - // mayber layers are given by layer ids crossed by only one booundary - set maybeLayer; - set_symmetric_difference(layers[0].begin(), - layers[0].end(), - layers[1].begin(), - layers[1].end(), - inserter(maybeLayer, maybeLayer.end())); - // layerEncoding is given by sorted layer ids crossed by any booundary - set layerEncoding; - set_union(layers[0].begin(), - layers[0].end(), - layers[1].begin(), - layers[1].end(), - inserter(layerEncoding, layerEncoding.end())); - vector& le = layerEncoding_[binEta][binZT][binCot]; - le = vector(layerEncoding.begin(), layerEncoding.end()); - vector& ml = maybeLayer_[binEta][binZT][binCot]; - ml.reserve(maybeLayer.size()); - for (int m : maybeLayer) { - int layer = distance(le.begin(), find(le.begin(), le.end(), m)); - if (layer >= setup_->numLayers()) - layer = setup_->numLayers() - 1; - ml.push_back(layer); - } - } - } - } - const bool print = false; - if (!print) - return; - static constexpr int widthLayer = 3; - static constexpr auto layerIds = {1, 2, 3, 4, 5, 6, 11, 12, 13, 14, 15}; - stringstream ss; - for (int layer : layerIds) { - auto encode = [layer, this](const vector& layers, int& l) { - const auto it = find(layers.begin(), layers.end(), layer); - if (it == layers.end()) - return false; - l = distance(layers.begin(), it); - if (l >= setup_->numLayers()) - l = setup_->numLayers() - 1; - return true; - }; - for (int binEta = 0; binEta < setup_->numSectorsEta(); binEta++) { - for (int binZT = 0; binZT < pow(2, zT_->width()); binZT++) { - for (int binCot = 0; binCot < pow(2, cot_->width()); binCot++) { - const int zT = - binZT < pow(2, zT_->width() - 1) ? binZT + pow(2, zT_->width() - 1) : binZT - pow(2, zT_->width() - 1); - const int cot = binCot < pow(2, cot_->width() - 1) ? binCot + pow(2, cot_->width() - 1) - : binCot - pow(2, cot_->width() - 1); - const vector& layers = layerEncoding_[binEta][zT][cot]; - const vector& maybes = maybeLayer_[binEta][zT][cot]; - int layerKF(-1); - if (encode(layers, layerKF)) - ss << "1" << TTBV(layerKF, widthLayer) << (encode(maybes, layerKF) ? "1" : "0"); - else - ss << "00000"; - ss << endl; - } + // loop over zT bins + for (int binZT = 0; binZT < pow(2, zT_->width()); binZT++) { + // z at radius chosenRofZ + const double zT = zT_->floating(zT_->toSigned(binZT)); + // z at radius chosenRofZ wrt zT of sectorZT of this bin boundaries + const vector zTs = {zT - zT_->base() / 2., zT + zT_->base() / 2.}; + vector> cots(boundaries); + for (int i = 0; i < boundaries; i++) + for (double z0 : z0s) + cots[i].push_back((zTs[i] - z0) / setup_->chosenRofZ()); + // layer ids crossed by left and right rough r-z parameter shape boundaries + vector> layers(boundaries); + // loop over all unique modules + for (const SensorModule* sm : sensorModules) { + // check if module is crossed by left and right rough r-z parameter shape boundaries + for (int i = 0; i < boundaries; i++) { + const double zTi = zTs[i]; + const double coti = sm->r() < setup_->chosenRofZ() ? cots[i][i == 0 ? 0 : 1] : cots[i][i == 0 ? 1 : 0]; + // distance between module and boundary in moudle tilt angle direction + const double d = + (zTi - sm->z() + (sm->r() - setup_->chosenRofZ()) * coti) / (sm->cosTilt() - sm->sinTilt() * coti); + // compare distance with module size and add module layer id to layers if module is crossed + if (abs(d) < sm->numColumns() * sm->pitchCol() / 2.) + layers[i].insert(sm->layerId()); } } + // mayber layers are given by layer ids crossed by only one boundary + set maybeLayer; + set_symmetric_difference(layers[0].begin(), + layers[0].end(), + layers[1].begin(), + layers[1].end(), + inserter(maybeLayer, maybeLayer.end())); + // layerEncoding is given by sorted layer ids crossed by any boundary + set layerEncoding; + set_union(layers[0].begin(), + layers[0].end(), + layers[1].begin(), + layers[1].end(), + inserter(layerEncoding, layerEncoding.end())); + // fill layerEncoding_ + vector& le = layerEncoding_[binZT]; + le = vector(layerEncoding.begin(), layerEncoding.end()); + le.resize(setup_->numLayers(), -1); + // fill maybePattern_ + TTBV& mp = maybePattern_[binZT]; + for (int m : maybeLayer) + mp.set(min((int)distance(le.begin(), find(le.begin(), le.end(), m)), setup_->numLayers() - 1)); } - fstream file; - file.open("layerEncoding.txt", ios::out); - file << ss.rdbuf(); - file.close(); } - // encoded layer id for given eta sector, bin in zT, bin in cotThea and decoed layer id, returns -1 if layer incositent with track - const int LayerEncoding::layerIdKF(int binEta, int binZT, int binCot, int layerId) const { - const vector& layers = layerEncoding_[binEta][binZT][binCot]; - const auto it = find(layers.begin(), layers.end(), layerId); - if (it == layers.end()) - return -1; - int layer = distance(layers.begin(), it); - if (layer >= setup_->numLayers()) - layer = setup_->numLayers() - 1; - return layer; + // Set of layers for given bin in zT + const vector& LayerEncoding::layerEncoding(int zT) const { + const int binZT = zT_->toUnsigned(zT); + return zT_->inRange(zT) ? layerEncoding_.at(binZT) : nullLE_; + } + + // Set of layers for given zT in cm + const vector& LayerEncoding::layerEncoding(double zT) const { + const int binZT = zT_->integer(zT); + return layerEncoding(binZT); + } + + // pattern of maybe layers for given bin in zT + const TTBV& LayerEncoding::maybePattern(int zT) const { + const int binZT = zT_->toUnsigned(zT); + return zT_->inRange(zT) ? maybePattern_[binZT] : nullMP_; } - // pattern of maybe layers for given eta sector, bin in zT and bin in cotThea - TTBV LayerEncoding::maybePattern(int binEta, int binZT, int binCot) const { - TTBV ttBV(0, setup_->numLayers()); - const vector& layers = layerEncoding_[binEta][binZT][binCot]; - const vector& maybes = maybeLayer_[binEta][binZT][binCot]; - for (int m : maybes) - ttBV.set(distance(layers.begin(), find(layers.begin(), layers.end(), m))); - return ttBV; + // pattern of maybe layers for given zT in cm + const TTBV& LayerEncoding::maybePattern(double zT) const { + const int binZT = zT_->integer(zT); + return maybePattern(binZT); } } // namespace trackerTFP diff --git a/L1Trigger/TrackerTFP/src/MiniHoughTransform.cc b/L1Trigger/TrackerTFP/src/MiniHoughTransform.cc deleted file mode 100644 index c81ed02da8eb1..0000000000000 --- a/L1Trigger/TrackerTFP/src/MiniHoughTransform.cc +++ /dev/null @@ -1,317 +0,0 @@ -#include "L1Trigger/TrackerTFP/interface/MiniHoughTransform.h" - -#include -#include -#include -#include -#include -#include -#include -#include - -using namespace std; -using namespace edm; -using namespace tt; - -namespace trackerTFP { - - MiniHoughTransform::MiniHoughTransform(const ParameterSet& iConfig, - const Setup* setup, - const DataFormats* dataFormats, - int region) - : enableTruncation_(iConfig.getParameter("EnableTruncation")), - setup_(setup), - dataFormats_(dataFormats), - inv2R_(dataFormats_->format(Variable::inv2R, Process::ht)), - phiT_(dataFormats_->format(Variable::phiT, Process::ht)), - region_(region), - numBinsInv2R_(setup_->htNumBinsInv2R()), - numCells_(setup_->mhtNumCells()), - numNodes_(setup_->mhtNumDLBNodes()), - numChannel_(setup_->mhtNumDLBChannel()), - input_(numBinsInv2R_) {} - - // read in and organize input product (fill vector input_) - void MiniHoughTransform::consume(const StreamsStub& streams) { - auto valid = [](int sum, const FrameStub& frame) { return sum + (frame.first.isNonnull() ? 1 : 0); }; - int nStubsHT(0); - for (int binInv2R = 0; binInv2R < numBinsInv2R_; binInv2R++) { - const StreamStub& stream = streams[region_ * numBinsInv2R_ + binInv2R]; - nStubsHT += accumulate(stream.begin(), stream.end(), 0, valid); - } - stubsHT_.reserve(nStubsHT); - stubsMHT_.reserve(nStubsHT * numCells_); - for (int binInv2R = 0; binInv2R < numBinsInv2R_; binInv2R++) { - const int inv2R = inv2R_.toSigned(binInv2R); - const StreamStub& stream = streams[region_ * numBinsInv2R_ + binInv2R]; - vector& stubs = input_[binInv2R]; - stubs.reserve(stream.size()); - // Store input stubs in vector, so rest of MHT algo can work with pointers to them (saves CPU) - for (const FrameStub& frame : stream) { - StubHT* stub = nullptr; - if (frame.first.isNonnull()) { - stubsHT_.emplace_back(frame, dataFormats_, inv2R); - stub = &stubsHT_.back(); - } - stubs.push_back(stub); - } - } - } - - // fill output products - void MiniHoughTransform::produce(StreamsStub& accepted, StreamsStub& lost) { - // fill MHT cells - vector> stubsCells(numBinsInv2R_ * numCells_); - for (int channel = 0; channel < numBinsInv2R_; channel++) - fill(channel, input_[channel], stubsCells); - // perform static load balancing - vector> streamsSLB(numBinsInv2R_); - for (int channel = 0; channel < numBinsInv2R_; channel++) { - vector> tmp(numCells_); - // gather streams to mux together: same MHT cell of 4 adjacent MHT input streams - for (int k = 0; k < numCells_; k++) - swap(tmp[k], stubsCells[(channel / numCells_) * numBinsInv2R_ + channel % numCells_ + k * numCells_]); - slb(tmp, streamsSLB[channel], lost[channel]); - } - // dynamic load balancing stage 1 - vector> streamsDLB(numBinsInv2R_); - for (int node = 0; node < numNodes_; node++) { - vector> tmp(numChannel_); - // gather streams to dynamically balance them - for (int k = 0; k < numChannel_; k++) - swap(tmp[k], streamsSLB[(node / numCells_) * numNodes_ + node % numCells_ + k * numCells_]); - dlb(tmp); - for (int k = 0; k < numChannel_; k++) - swap(tmp[k], streamsDLB[node * numChannel_ + k]); - } - // dynamic load balancing stage 2 - vector> streamsMHT(numBinsInv2R_); - for (int node = 0; node < numNodes_; node++) { - vector> tmp(numChannel_); - // gather streams to dynamically balance them - for (int k = 0; k < numChannel_; k++) - swap(tmp[k], streamsDLB[node + k * numNodes_]); - dlb(tmp); - for (int k = 0; k < numChannel_; k++) - swap(tmp[k], streamsMHT[node * numChannel_ + k]); - } - // fill output product - for (int channel = 0; channel < numBinsInv2R_; channel++) { - const vector& stubs = streamsMHT[channel]; - StreamStub& stream = accepted[region_ * numBinsInv2R_ + channel]; - stream.reserve(stubs.size()); - for (StubMHT* stub : stubs) - stream.emplace_back(stub ? stub->frame() : FrameStub()); - } - } - - // perform finer pattern recognition per track - void MiniHoughTransform::fill(int channel, const vector& stubs, vector>& streams) { - if (stubs.empty()) - return; - int id; - auto differentHT = [&id](StubHT* stub) { return id != stub->trackId(); }; - auto differentMHT = [&id](StubMHT* stub) { return !stub || id != stub->trackId(); }; - for (auto it = stubs.begin(); it != stubs.end();) { - const auto start = it; - id = (*it)->trackId(); - it = find_if(it, stubs.end(), differentHT); - const int size = distance(start, it); - // create finer track candidates stub container - vector> mhtCells(numCells_); - for (vector& mhtCell : mhtCells) - mhtCell.reserve(size); - // fill finer track candidates stub container - for (auto stub = start; stub != it; stub++) { - const double r = (*stub)->r(); - const double chi = (*stub)->phi(); - // identify finer track candidates for this stub - // 0 and 1 belong to the MHT cells with larger inv2R; 0 and 2 belong to those with smaller track PhiT - vector cells; - cells.reserve(numCells_); - const bool compA = 2. * abs(chi) < phiT_.base(); - const bool compB = 2. * abs(chi) < abs(r * inv2R_.base()); - const bool compAB = compA && compB; - if (chi >= 0. && r >= 0.) { - cells.push_back(3); - if (compA) - cells.push_back(1); - if (compAB) - cells.push_back(2); - } - if (chi >= 0. && r < 0.) { - cells.push_back(1); - if (compA) - cells.push_back(3); - if (compAB) - cells.push_back(0); - } - if (chi < 0. && r >= 0.) { - cells.push_back(0); - if (compA) - cells.push_back(2); - if (compAB) - cells.push_back(1); - } - if (chi < 0. && r < 0.) { - cells.push_back(2); - if (compA) - cells.push_back(0); - if (compAB) - cells.push_back(3); - } - // organise stubs in finer track candidates - for (int cell : cells) { - const int inv2R = cell / setup_->mhtNumBinsPhiT(); - const int phiT = cell % setup_->mhtNumBinsPhiT(); - stubsMHT_.emplace_back(**stub, phiT, inv2R); - mhtCells[cell].push_back(&stubsMHT_.back()); - } - } - // perform pattern recognition - for (int sel = 0; sel < numCells_; sel++) { - deque& stream = streams[channel * numCells_ + sel]; - vector& mhtCell = mhtCells[sel]; - set layers; - auto toLayer = [](StubMHT* stub) { return stub->layer(); }; - transform(mhtCell.begin(), mhtCell.end(), inserter(layers, layers.begin()), toLayer); - if ((int)layers.size() < setup_->mhtMinLayers()) - mhtCell.clear(); - for (StubMHT* stub : mhtCell) - stream.push_back(stub); - stream.insert(stream.end(), size - (int)mhtCell.size(), nullptr); - } - } - for (int sel = 0; sel < numCells_; sel++) { - deque& stream = streams[channel * numCells_ + sel]; - // remove all gaps between end and last stub - for (auto it = stream.end(); it != stream.begin();) - it = (*--it) ? stream.begin() : stream.erase(it); - // read out fine track cannot start before rough track has read in completely, add gaps to take this into account - int pos(0); - for (auto it = stream.begin(); it != stream.end();) { - if (!(*it)) { - it = stream.erase(it); - continue; - } - id = (*it)->trackId(); - const int s = distance(it, find_if(it, stream.end(), differentMHT)); - const int d = distance(stream.begin(), it); - pos += s; - if (d < pos) { - const int diff = pos - d; - it = stream.insert(it, diff, nullptr); - it = next(it, diff); - } else - it = stream.erase(remove(next(stream.begin(), pos), it, nullptr), it); - it = next(it, s); - } - // adjust stream start so that first output stub is in first place in case of quickest track - if (!stream.empty()) - stream.erase(stream.begin(), next(stream.begin(), setup_->mhtMinLayers())); - } - } - - // Static load balancing of inputs: mux 4 streams to 1 stream - void MiniHoughTransform::slb(vector>& inputs, vector& accepted, StreamStub& lost) const { - if (all_of(inputs.begin(), inputs.end(), [](const deque& stubs) { return stubs.empty(); })) - return; - auto size = [](int sum, const deque& stubs) { return sum = stubs.size(); }; - const int nFrames = accumulate(inputs.begin(), inputs.end(), 0, size); - accepted.reserve(nFrames); - // input fifos - vector> stacks(numCells_); - // helper for handshake - TTBV empty(-1, numCells_, true); - TTBV enable(0, numCells_); - // clock accurate firmware emulation, each while trip describes one clock tick, one stub in and one stub out per tick - while (!all_of(inputs.begin(), inputs.end(), [](const deque& d) { return d.empty(); }) or - !all_of(stacks.begin(), stacks.end(), [](const deque& d) { return d.empty(); })) { - // store stub in fifo - for (int channel = 0; channel < numCells_; channel++) { - StubMHT* stub = pop_front(inputs[channel]); - if (stub) - stacks[channel].push_back(stub); - } - // identify empty fifos - for (int channel = 0; channel < numCells_; channel++) - empty[channel] = stacks[channel].empty(); - // chose new fifo to read from if current fifo got empty - const int iEnableOld = enable.plEncode(); - if (enable.none() || empty[iEnableOld]) { - enable.reset(); - const int iNotEmpty = empty.plEncode(false); - if (iNotEmpty < numCells_) - enable.set(iNotEmpty); - } - // read from chosen fifo - const int iEnable = enable.plEncode(); - if (enable.any()) - accepted.push_back(pop_front(stacks[iEnable])); - else - // gap if no fifo has been chosen - accepted.push_back(nullptr); - } - // perform truncation if desired - if (enableTruncation_ && (int)accepted.size() > setup_->numFrames()) { - const auto limit = next(accepted.begin(), setup_->numFrames()); - auto valid = [](int sum, StubMHT* stub) { return sum + (stub ? 1 : 0); }; - const int nLost = accumulate(limit, accepted.end(), 0, valid); - lost.reserve(nLost); - for (auto it = limit; it != accepted.end(); it++) - if (*it) - lost.emplace_back((*it)->frame()); - accepted.erase(limit, accepted.end()); - } - // cosmetics -- remove gaps at the end of stream - for (auto it = accepted.end(); it != accepted.begin();) - it = (*--it) == nullptr ? accepted.erase(it) : accepted.begin(); - } - - // Dynamic load balancing of inputs: swapping parts of streams to balance the amount of tracks per stream - void MiniHoughTransform::dlb(vector>& streams) const { - if (all_of(streams.begin(), streams.end(), [](const vector& stubs) { return stubs.empty(); })) - return; - auto maxSize = [](int size, const vector& stream) { return size = max(size, (int)stream.size()); }; - const int nMax = accumulate(streams.begin(), streams.end(), 0, maxSize); - for (vector& stream : streams) - stream.resize(nMax, nullptr); - vector prevTrks(numChannel_, -1); - bool swapping(false); - vector loads(numChannel_, 0); - for (int i = 0; i < nMax; i++) { - TTBV newTrks(0, numChannel_); - for (int k = 0; k < numChannel_; k++) - if (!streams[numChannel_ - k - 1][i] && streams[k][i] && streams[k][i]->trackId() != prevTrks[k]) - newTrks.set(k); - for (int k = 0; k < numChannel_; k++) - if (newTrks[k]) - if ((swapping && loads[numChannel_ - k - 1] > loads[k]) || - (!swapping && loads[k] > loads[numChannel_ - k - 1])) - swapping = !swapping; - for (int k = 0; k < numChannel_; k++) { - if (streams[k][i]) - loads[swapping ? numChannel_ - k - 1 : k]++; - prevTrks[k] = streams[k][i] ? streams[k][i]->trackId() : -1; - } - if (swapping) - swap(streams[0][i], streams[1][i]); - } - // remove all gaps between end and last stub - for (vector& stream : streams) - for (auto it = stream.end(); it != stream.begin();) - it = (*--it) ? stream.begin() : stream.erase(it); - } - - // remove and return first element of deque, returns nullptr if empty - template - T* MiniHoughTransform::pop_front(deque& ts) const { - T* t = nullptr; - if (!ts.empty()) { - t = ts.front(); - ts.pop_front(); - } - return t; - } - -} // namespace trackerTFP diff --git a/L1Trigger/TrackerTFP/src/State.cc b/L1Trigger/TrackerTFP/src/State.cc index 739be28ddbe2a..c9d7a43ea2577 100644 --- a/L1Trigger/TrackerTFP/src/State.cc +++ b/L1Trigger/TrackerTFP/src/State.cc @@ -5,121 +5,183 @@ using namespace tt; namespace trackerTFP { - // default constructor - State::State(State* state) - : dataFormats_(state->dataFormats_), - setup_(state->setup_), - track_(state->track_), - trackId_(state->trackId_), - parent_(state->parent_), - stub_(state->stub_), - layerMap_(state->layerMap_), - hitPattern_(state->hitPattern_), - x0_(state->x0_), - x1_(state->x1_), - x2_(state->x2_), - x3_(state->x3_), - C00_(state->C00_), - C01_(state->C01_), - C11_(state->C11_), - C22_(state->C22_), - C23_(state->C23_), - C33_(state->C33_), - numSkippedLayers_(state->numSkippedLayers_), - numConsistentLayers_(state->numConsistentLayers_) {} + // + State::Stub::Stub(KalmanFilterFormats* formats, const FrameStub& frame) : stubCTB_(frame, formats->dataFormats()) { + const Setup* setup = formats->setup(); + H12_ = formats->format(VariableKF::H12).digi(stubCTB_.r() + setup->chosenRofPhi() - setup->chosenRofZ()); + v0_ = formats->format(VariableKF::v0).digi(pow(2. * stubCTB_.dPhi(), 2)); + v1_ = formats->format(VariableKF::v1).digi(pow(2. * stubCTB_.dZ(), 2)); + } // proto state constructor - State::State(const DataFormats* dataFormats, TrackKFin* track, int trackId) - : dataFormats_(dataFormats), - setup_(dataFormats->setup()), + State::State(KalmanFilterFormats* formats, + TrackCTB* track, + const vector>& stubs, + const TTBV& maybePattern, + int trackId) + : formats_(formats), + setup_(formats->setup()), track_(track), + stubs_(stubs), + maybePattern_(maybePattern), trackId_(trackId), - parent_(nullptr), - stub_(nullptr), - layerMap_(setup_->numLayers()), hitPattern_(0, setup_->numLayers()), - numSkippedLayers_(0), - numConsistentLayers_(0) { - // initial track parameter residuals w.r.t. found track - x0_ = 0.; - x1_ = 0.; - x2_ = 0.; - x3_ = 0.; - // initial uncertainties - C00_ = pow(dataFormats_->base(Variable::inv2R, Process::kfin), 2) * pow(2, setup_->kfShiftInitialC00()); - C11_ = pow(dataFormats_->base(Variable::phiT, Process::kfin), 2) * pow(2, setup_->kfShiftInitialC11()); - C22_ = pow(dataFormats_->base(Variable::cot, Process::kfin), 2) * pow(2, setup_->kfShiftInitialC22()); - C33_ = pow(dataFormats_->base(Variable::zT, Process::kfin), 2) * pow(2, setup_->kfShiftInitialC33()); - C01_ = 0.; - C23_ = 0.; - // first stub from first layer on input track with stubs - stub_ = track->layerStub(track->hitPattern().plEncode()); - } - - // combinatoric state constructor - State::State(State* state, StubKFin* stub) : State(state) { - parent_ = state->parent(); - stub_ = stub; + trackPattern_(0, setup_->numLayers()) { + for (const vector& stubs : stubs_) { + if (!stubs.empty()) + trackPattern_.set(layer_); + layer_++; + } + layer_ = trackPattern_.plEncode(); + stub_ = stubs_[layer_].front(); + hitPattern_.set(layer_); } // updated state constructor - State::State(State* state, const std::vector& doubles) : State(state) { + State::State(State* state, const vector& doubles) : State(state) { parent_ = state; // updated track parameter and uncertainties x0_ = doubles[0]; x1_ = doubles[1]; x2_ = doubles[2]; x3_ = doubles[3]; - C00_ = doubles[4]; - C11_ = doubles[5]; - C22_ = doubles[6]; - C33_ = doubles[7]; - C01_ = doubles[8]; - C23_ = doubles[9]; - // update maps - const int layer = stub_->layer(); - hitPattern_.set(layer); - const vector& stubs = track_->layerStubs(layer); - layerMap_[layer] = distance(stubs.begin(), find(stubs.begin(), stubs.end(), stub_)); + chi20_ = doubles[4]; + chi21_ = doubles[5]; + C00_ = doubles[6]; + C11_ = doubles[7]; + C22_ = doubles[8]; + C33_ = doubles[9]; + C01_ = doubles[10]; + C23_ = doubles[11]; // pick next stub (first stub in next layer with stub) stub_ = nullptr; - if (hitPattern_.count() == setup_->kfMaxLayers()) + if (hitPattern_.count() >= setup_->kfMinLayers() || hitPattern_.count() == setup_->kfMaxLayers()) { + layer_ = 0; return; - for (int nextLayer = layer + 1; nextLayer < setup_->numLayers(); nextLayer++) { - if (track_->hitPattern(nextLayer)) { - stub_ = track_->layerStub(nextLayer); - break; - } } + layer_ = trackPattern_.plEncode(layer_ + 1, setup_->numLayers()); + if (layer_ == setup_->numLayers()) + return; + stub_ = stubs_[layer_].front(); + hitPattern_.set(layer_); + } + + // combinatoric and seed building state constructor + State::State(State* state, State* parent, Stub* stub, int layer) : State(state) { + parent_ = parent; + stub_ = stub; + layer_ = layer; + hitPattern_ = parent ? parent->hitPattern() : TTBV(0, setup_->numLayers()); + hitPattern_.set(layer_); + } + + // + State* State::update(deque& states, int layer) { + if (!hitPattern_.test(layer) || hitPattern_.count() > setup_->kfNumSeedStubs()) + return this; + layer_ = trackPattern_.plEncode(layer_ + 1, setup_->numLayers()); + states.emplace_back(this, this, stubs_[layer_].front(), layer_); + return &states.back(); + } + + // + State* State::combSeed(deque& states, int layer) { + // handle trivial state + if (!hitPattern_.test(layer) || hitPattern_.count() > setup_->kfNumSeedStubs()) + return nullptr; + // pick next stub on layer + const vector& stubs = stubs_[layer]; + const int pos = distance(stubs.begin(), find(stubs.begin(), stubs.end(), stub_)) + 1; + if (pos < (int)stubs.size()) { + states.emplace_back(this, parent_, stubs[pos], layer); + return &states.back(); + } + // skip this layer + const int nextLayer = trackPattern_.plEncode(layer + 1, setup_->numLayers()); + if (gapCheck(nextLayer)) { + states.emplace_back(this, parent_, stubs_[nextLayer].front(), nextLayer); + return &states.back(); + } + return nullptr; } - // fills collection of stubs added so far to state - void State::fill(vector& stubs) const { - stubs.reserve(hitPattern_.count()); - State* s = parent_; - while (s) { - stubs.emplace_back(*(s->stub()), x0_, x1_, x2_, x3_); - s = s->parent(); + // + State* State::comb(deque& states, int layer) { + // handle skipping and min reached + if (!hitPattern_.test(layer)) { + if (!stub_ && trackPattern_[layer] && hitPattern_.count() < setup_->kfMaxLayers()) { + states.emplace_back(this, parent_, stubs_[layer].front(), layer); + return &states.back(); + } + return nullptr; + } + // handle part of seed + if (hitPattern_.pmEncode() != layer) + return nullptr; + // handle multiple stubs on layer + const vector& stubs = stubs_[layer]; + const int pos = distance(stubs.begin(), find(stubs.begin(), stubs.end(), stub_)) + 1; + if (pos < (int)stubs.size()) { + states.emplace_back(this, parent_, stubs[pos], layer); + return &states.back(); } + // handle skip + const int nextLayer = trackPattern_.plEncode(layer + 1, setup_->numLayers()); + if (gapCheck(nextLayer)) { + states.emplace_back(this, parent_, stubs_[nextLayer].front(), nextLayer); + return &states.back(); + } + return nullptr; } - // Determine quality of completed state - void State::finish() { - auto consistent = [this](int sum, const StubKF& stub) { - static const DataFormat& phi = dataFormats_->format(Variable::phi, Process::kf); - static const DataFormat& z = dataFormats_->format(Variable::z, Process::kf); - // Check stub consistent with helix, allowing for stub uncertainty - const bool inRange0 = 2. * abs(stub.phi()) - stub.dPhi() < phi.base(); - const bool inRange1 = 2. * abs(stub.z()) - stub.dZ() < z.base(); - return sum + (inRange0 && inRange1 ? 1 : 0); - }; - vector stubs; - fill(stubs); - numConsistentLayers_ = accumulate(stubs.begin(), stubs.end(), 0, consistent); - TTBV pattern = hitPattern_; - pattern |= maybePattern(); - // Skipped layers before final stub on state - numSkippedLayers_ = pattern.count(0, hitPattern_.pmEncode(), false); + // + bool State::gapCheck(int layer) const { + if (layer >= setup_->numLayers()) + return false; + bool gap(false); + int hits(0); + int gaps(0); + for (int k = 0; k < setup_->numLayers(); k++) { + if (k == setup_->kfMaxSeedingLayer()) + if (hits < setup_->kfNumSeedStubs()) + return false; + if (hitPattern_[k]) { + gap = false; + if (++hits >= setup_->kfMinLayers() && k >= layer) + return true; + } else if (!maybePattern_[k]) { + if (gap || ++ gaps > setup_->kfMaxGaps()) + return false; + gap = true; + } + } + return false; } + // copy constructor + State::State(State* state) + : formats_(state->formats_), + setup_(state->setup_), + track_(state->track_), + stubs_(state->stubs_), + maybePattern_(state->maybePattern_), + trackId_(state->trackId_), + parent_(state->parent_), + stub_(state->stub_), + layer_(state->layer_), + hitPattern_(state->hitPattern_), + trackPattern_(state->trackPattern_), + x0_(state->x0_), + x1_(state->x1_), + x2_(state->x2_), + x3_(state->x3_), + chi20_(state->chi20_), + chi21_(state->chi21_), + C00_(state->C00_), + C01_(state->C01_), + C11_(state->C11_), + C22_(state->C22_), + C23_(state->C23_), + C33_(state->C33_) {} + } // namespace trackerTFP diff --git a/L1Trigger/TrackerTFP/src/TrackFindingProcessor.cc b/L1Trigger/TrackerTFP/src/TrackFindingProcessor.cc new file mode 100644 index 0000000000000..c278009b8c447 --- /dev/null +++ b/L1Trigger/TrackerTFP/src/TrackFindingProcessor.cc @@ -0,0 +1,278 @@ +#include "L1Trigger/TrackerTFP/interface/TrackFindingProcessor.h" +#include "L1Trigger/TrackTrigger/interface/StubPtConsistency.h" + +#include +#include +#include +#include +#include + +using namespace std; +using namespace edm; +using namespace tt; + +namespace trackerTFP { + + TrackFindingProcessor::TrackFindingProcessor(const ParameterSet& iConfig, + const Setup* setup, + const DataFormats* dataFormats, + const TrackQuality* trackQuality) + : enableTruncation_(iConfig.getParameter("EnableTruncation")), + setup_(setup), + dataFormats_(dataFormats), + trackQuality_(trackQuality) {} + + // + TrackFindingProcessor::Track::Track(const FrameTrack& frameTrack, + const Frame& frameTQ, + const vector& ttStubRefs, + const TrackQuality* tq) + : ttTrackRef_(frameTrack.first), ttStubRefs_(ttStubRefs), valid_(true) { + partials_.reserve(partial_in); + // convert bits into nice formats + const DataFormats* df = tq->dataFormats(); + const Setup* setup = df->setup(); + const TrackDR trackDR(frameTrack, df); + inv2R_ = trackDR.inv2R(); + phiT_ = trackDR.phiT(); + cot_ = trackDR.cot(); + zT_ = trackDR.zT(); + const double d0 = max(min(ttTrackRef_->d0(), -TTTrack_TrackWord::minD0), TTTrack_TrackWord::minD0); + TTBV ttBV = TTBV(frameTQ); + tq->format(VariableTQ::chi2rz).extract(ttBV, chi2rz_); + tq->format(VariableTQ::chi2rphi).extract(ttBV, chi2rphi_); + mva_ = TTBV(ttBV, numBinsMVA_).val(); + ttBV >>= numBinsMVA_; + hitPattern_ = ttBV; + channel_ = cot_ < 0. ? 0 : 1; + // convert nice formats into bits + const double z0 = zT_ - cot_ * setup->chosenRofZ(); + const double phi0 = phiT_ - inv2R_ * setup->chosenRofPhi(); + double invR = -2. * inv2R_; + if (invR < TTTrack_TrackWord::minRinv) + invR = TTTrack_TrackWord::minRinv + df->format(Variable::inv2R, Process::dr).base(); + else if (invR > -TTTrack_TrackWord::minRinv) + invR = -TTTrack_TrackWord::minRinv - df->format(Variable::inv2R, Process::dr).base(); + const double chi2rphi = chi2rphi_ / (hitPattern_.count() - 2); + const double chi2rz = chi2rz_ / (hitPattern_.count() - 2); + int chi2rphiBin(-1); + for (double d : TTTrack_TrackWord::chi2RPhiBins) + if (chi2rphi >= d) + chi2rphiBin++; + else + break; + int chi2rzBin(-1); + for (double d : TTTrack_TrackWord::chi2RZBins) + if (chi2rz >= d) + chi2rzBin++; + else + break; + static const double rangeInvR = -2. * TTTrack_TrackWord::minRinv; + static const double rangePhi0 = -2. * TTTrack_TrackWord::minPhi0; + static const double rangeCot = -2. * TTTrack_TrackWord::minTanl; + static const double rangeZ0 = -2. * TTTrack_TrackWord::minZ0; + static const double rangeD0 = -2. * TTTrack_TrackWord::minD0; + if (abs(invR) > rangeInvR / 2.) + valid_ = false; + if (abs(phi0) > rangePhi0 / 2.) + valid_ = false; + if (abs(cot_) > rangeCot / 2.) + valid_ = false; + if (abs(z0) > rangeZ0 / 2.) + valid_ = false; + if (abs(d0) > rangeD0 / 2.) + valid_ = false; + if (!valid_) + return; + static const double baseInvR = rangeInvR / pow(2., TTTrack_TrackWord::TrackBitWidths::kRinvSize); + static const double basePhi0 = rangePhi0 / pow(2., TTTrack_TrackWord::TrackBitWidths::kPhiSize); + static const double baseCot = rangeCot / pow(2., TTTrack_TrackWord::TrackBitWidths::kTanlSize); + static const double baseZ0 = rangeZ0 / pow(2., TTTrack_TrackWord::TrackBitWidths::kZ0Size); + static const double baseD0 = rangeD0 / pow(2., TTTrack_TrackWord::TrackBitWidths::kD0Size); + static constexpr int nLayers = TTTrack_TrackWord::TrackBitWidths::kHitPatternSize; + static const TTBV Other_MVAs(0, 2 * TTTrack_TrackWord::TrackBitWidths::kMVAQualitySize); + const TTBV MVA_quality(mva_, TTTrack_TrackWord::TrackBitWidths::kMVAQualitySize); + const TTBV hit_pattern(hitPattern_.resize(nLayers).val(), nLayers); + static const TTBV chi2bend(0, TTTrack_TrackWord::TrackBitWidths::kBendChi2Size); + static const TTBV D0(d0, baseD0, TTTrack_TrackWord::TrackBitWidths::kD0Size, true); + const TTBV Chi2rz(chi2rzBin, TTTrack_TrackWord::TrackBitWidths::kChi2RZSize); + const TTBV Z0(z0, baseZ0, TTTrack_TrackWord::TrackBitWidths::kZ0Size, true); + const TTBV tanL(cot_, baseCot, TTTrack_TrackWord::TrackBitWidths::kTanlSize, true); + const TTBV Chi2rphi(chi2rphiBin, TTTrack_TrackWord::TrackBitWidths::kChi2RPhiSize); + const TTBV Phi0(phi0, basePhi0, TTTrack_TrackWord::TrackBitWidths::kPhiSize, true); + const TTBV InvR(invR, baseInvR, TTTrack_TrackWord::TrackBitWidths::kRinvSize, true); + static const TTBV valid(1, TTTrack_TrackWord::TrackBitWidths::kValidSize); + partials_.emplace_back((valid + InvR + Phi0 + Chi2rphi).str()); + partials_.emplace_back((tanL + Z0 + Chi2rz).str()); + partials_.emplace_back((D0 + chi2bend + hit_pattern + MVA_quality + Other_MVAs).str()); + } + + // fill output products + void TrackFindingProcessor::produce(const StreamsTrack& inputs, + const StreamsStub& stubs, + TTTracks& ttTracks, + StreamsTrack& outputs) { + // organize input tracks + vector> streams(outputs.size()); + consume(inputs, stubs, streams); + // emualte data format f/w + produce(streams, outputs); + // produce TTTracks + produce(outputs, ttTracks); + } + + // + void TrackFindingProcessor::consume(const StreamsTrack& inputs, + const StreamsStub& stubs, + vector>& outputs) { + // count input objects + int nTracks(0); + auto valid = [](int& sum, const FrameTrack& frame) { return sum += (frame.first.isNonnull() ? 1 : 0); }; + for (const StreamTrack& tracks : inputs) + nTracks += accumulate(tracks.begin(), tracks.end(), 0, valid); + tracks_.reserve(nTracks); + // convert input data + for (int region = 0; region < setup_->numRegions(); region++) { + const int offsetTQ = region * setup_->tqNumChannel(); + const int offsetTFP = region * setup_->tfpNumChannel(); + const int offsetStub = region * setup_->numLayers(); + const StreamTrack& streamDR = inputs[offsetTQ]; + const StreamTrack& streamTQ = inputs[offsetTQ + 1]; + for (int channel = 0; channel < setup_->tfpNumChannel(); channel++) + outputs[offsetTFP + channel] = deque(streamDR.size(), nullptr); + for (int frame = 0; frame < (int)streamDR.size(); frame++) { + const FrameTrack& frameTrack = streamDR[frame]; + const Frame& frameTQ = streamTQ[frame].second; + if (frameTrack.first.isNull()) + continue; + vector ttStubRefs; + ttStubRefs.reserve(setup_->numLayers()); + for (int layer = 0; layer < setup_->numLayers(); layer++) { + const TTStubRef& ttStubRef = stubs[offsetStub + layer][frame].first; + if (ttStubRef.isNonnull()) + ttStubRefs.push_back(ttStubRef); + } + tracks_.emplace_back(frameTrack, frameTQ, ttStubRefs, trackQuality_); + Track& track = tracks_.back(); + outputs[offsetTFP + track.channel_][frame] = track.valid_ ? &track : nullptr; + } + // remove all gaps between end and last track + for (int channel = 0; channel < setup_->tfpNumChannel(); channel++) { + deque input = outputs[offsetTFP + channel]; + for (auto it = input.end(); it != input.begin();) + it = (*--it) ? input.begin() : input.erase(it); + } + } + } + + // emualte data format f/w + void TrackFindingProcessor::produce(vector>& inputs, StreamsTrack& outputs) const { + for (int channel = 0; channel < (int)inputs.size(); channel++) { + deque& input = inputs[channel]; + deque stack; + deque output; + // clock accurate firmware emulation, each while trip describes one clock tick, one stub in and one stub out per tick + while (!input.empty() || !stack.empty()) { + output.emplace_back(FrameTrack()); + FrameTrack& frame = output.back(); + Track* track = pop_front(input); + if (track) + for (const PartialFrame& pf : track->partials_) + stack.emplace_back(track->ttTrackRef_, pf); + TTBV ttBV; + for (int i = 0; i < partial_out; i++) { + if (stack.empty()) { + ttBV += TTBV(0, partial_width); + continue; + } + const PartialFrameTrack& pft = stack.front(); + frame.first = pft.first; + ttBV += TTBV(pft.second.to_string()); + stack.pop_front(); + } + frame.second = ttBV.bs(); + } + // perorm truncation + if (enableTruncation_ && (int)output.size() > setup_->numFramesIOHigh()) + output.resize(setup_->numFramesIOHigh()); + outputs[channel] = StreamTrack(output.begin(), output.end()); + } + } + + // produce TTTracks + void TrackFindingProcessor::produce(const StreamsTrack& inputs, TTTracks& outputs) const { + // collect input TTTrackRefs + vector ttTrackRefs; + ttTrackRefs.reserve(tracks_.size()); + const TTTrack* last = nullptr; + for (const StreamTrack& stream : inputs) { + for (const FrameTrack& frame : stream) { + const TTTrackRef& ttTrackRef = frame.first; + if (frame.first.isNull() || last == ttTrackRef.get()) + continue; + last = ttTrackRef.get(); + ttTrackRefs.push_back(ttTrackRef); + } + } + // convert input TTTrackRefs into output TTTracks + static const DataFormat& dfZT = dataFormats_->format(Variable::zT, Process::gp); + outputs.reserve(ttTrackRefs.size()); + for (const TTTrackRef& ttTrackRef : ttTrackRefs) { + auto match = [&ttTrackRef](const Track& track) { return track.ttTrackRef_ == ttTrackRef; }; + const auto it = find_if(tracks_.begin(), tracks_.end(), match); + // TTTrack conversion + const int region = ttTrackRef->phiSector(); + const double aRinv = -2. * it->inv2R_; + const double aphi = deltaPhi(it->phiT_ - it->inv2R_ * setup_->chosenRofPhi() + region * setup_->baseRegion()); + const double aTanLambda = it->cot_; + const double az0 = it->zT_ - it->cot_ * setup_->chosenRofZ(); + const double ad0 = -ttTrackRef->d0(); + const double aChi2xyfit = it->chi2rphi_; + const double aChi2zfit = it->chi2rz_; + const double trkMVA1 = (TTTrack_TrackWord::tqMVABins[it->mva_]); + static constexpr double trkMVA2 = 0.; + static constexpr double trkMVA3 = 0.; + const unsigned int aHitpattern = it->hitPattern_.val(); + const unsigned int nPar = ttTrackRef->nFitPars(); + static const double Bfield = setup_->bField(); + outputs.emplace_back( + aRinv, aphi, aTanLambda, az0, ad0, aChi2xyfit, aChi2zfit, trkMVA1, trkMVA2, trkMVA3, aHitpattern, nPar, Bfield); + TTTrack& ttTrack = outputs.back(); + ttTrack.setPhiSector(region); + ttTrack.setEtaSector(dfZT.toUnsigned(dfZT.integer(it->zT_))); + ttTrack.setTrackSeedType(ttTrackRef->trackSeedType()); + ttTrack.setStubRefs(it->ttStubRefs_); + ttTrack.setStubPtConsistency(StubPtConsistency::getConsistency( + ttTrack, setup_->trackerGeometry(), setup_->trackerTopology(), Bfield, nPar)); + } + } + + // produce StreamsTrack + void TrackFindingProcessor::produce(const vector& inputs, StreamsTrack& outputs) const { + int iTrk(-1); + const TTTrack* last = nullptr; + for (StreamTrack& stream : outputs) { + for (FrameTrack& frame : stream) { + const TTTrackRef& ttTrackRef = frame.first; + if (ttTrackRef.isNull()) + continue; + if (last != ttTrackRef.get()) + iTrk++; + last = ttTrackRef.get(); + frame.first = inputs[iTrk]; + } + } + } + + // remove and return first element of deque, returns nullptr if empty + template + T* TrackFindingProcessor::pop_front(deque& ts) const { + T* t = nullptr; + if (!ts.empty()) { + t = ts.front(); + ts.pop_front(); + } + return t; + } + +} // namespace trackerTFP diff --git a/L1Trigger/TrackerTFP/src/TrackQuality.cc b/L1Trigger/TrackerTFP/src/TrackQuality.cc new file mode 100644 index 0000000000000..8ec26d809906e --- /dev/null +++ b/L1Trigger/TrackerTFP/src/TrackQuality.cc @@ -0,0 +1,284 @@ +/* +Track Quality Body file +C.Brown & C.Savard 07/2020 +*/ + +#include "L1Trigger/TrackerTFP/interface/TrackQuality.h" +#include "L1Trigger/TrackTrigger/interface/StubPtConsistency.h" + +#include +#include +#include +#include "conifer.h" +#include "ap_fixed.h" + +using namespace std; +using namespace edm; +using namespace tt; + +namespace trackerTFP { + + TrackQuality::TrackQuality(const ParameterSet& iConfig, const DataFormats* dataFormats) + : dataFormats_(dataFormats), + model_(iConfig.getParameter("Model")), + featureNames_(iConfig.getParameter>("FeatureNames")), + baseShiftCot_(iConfig.getParameter("BaseShiftCot")), + baseShiftZ0_(iConfig.getParameter("BaseShiftZ0")), + baseShiftAPfixed_(iConfig.getParameter("BaseShiftAPfixed")), + chi2rphiConv_(iConfig.getParameter("Chi2rphiConv")), + chi2rzConv_(iConfig.getParameter("Chi2rzConv")), + weightBinFraction_(iConfig.getParameter("WeightBinFraction")), + dzTruncation_(iConfig.getParameter("DzTruncation")), + dphiTruncation_(iConfig.getParameter("DphiTruncation")) { + dataFormatsTQ_.reserve(+VariableTQ::end); + fillDataFormats(iConfig); + } + + // constructs TQ data formats + template + void TrackQuality::fillDataFormats(const ParameterSet& iConfig) { + dataFormatsTQ_.emplace_back(FormatTQ(dataFormats_, iConfig)); + if constexpr (++v != VariableTQ::end) + fillDataFormats<++v>(iConfig); + } + + // TQ MVA bin conversion LUT + constexpr array TrackQuality::mvaPreSigBins() const { + array lut = {}; + lut[0] = -16.; + for (int i = 1; i < numBinsMVA_; i++) + lut[i] = invSigmoid(TTTrack_TrackWord::tqMVABins[i]); + return lut; + } + + // + template + int TrackQuality::toBin(const T& bins, double d) const { + int bin = 0; + for (; bin < (int)bins.size() - 1; bin++) + if (d < bins[bin + 1]) + break; + return bin; + } + + // Helper function to convert mvaPreSig to bin + int TrackQuality::toBinMVA(double mva) const { + static const array bins = mvaPreSigBins(); + return toBin(bins, mva); + } + + // Helper function to convert chi2B to bin + int TrackQuality::toBinChi2B(double chi2B) const { + static const array bins = TTTrack_TrackWord::bendChi2Bins; + return toBin(bins, chi2B); + } + + // Helper function to convert chi2rphi to bin + int TrackQuality::toBinchi2rphi(double chi2rphi) const { + static const array bins = TTTrack_TrackWord::chi2RPhiBins; + double chi2 = chi2rphi * chi2rphiConv_; + return toBin(bins, chi2); + } + + // Helper function to convert chi2rz to bin + int TrackQuality::toBinchi2rz(double chi2rz) const { + static const array bins = TTTrack_TrackWord::chi2RZBins; + double chi2 = chi2rz * chi2rzConv_; + return toBin(bins, chi2); + } + + TrackQuality::Track::Track(const FrameTrack& frameTrack, const StreamStub& streamStub, const TrackQuality* tq) { + static const DataFormats* df = tq->dataFormats(); + static const Setup* setup = df->setup(); + frames_.reserve(setup->tqNumChannel()); + const TrackDR track(frameTrack, df); + double trackchi2rphi(0.); + double trackchi2rz(0.); + TTBV hitPattern(0, streamStub.size()); + vector ttStubRefs; + ttStubRefs.reserve(setup->numLayers()); + for (int layer = 0; layer < (int)streamStub.size(); layer++) { + const FrameStub& frameStub = streamStub[layer]; + if (frameStub.first.isNull()) + continue; + const StubKF stub(frameStub, df); + hitPattern.set(layer); + ttStubRefs.push_back(frameStub.first); + const double m20 = tq->format(VariableTQ::m20).digi(pow(stub.phi(), 2)); + const double m21 = tq->format(VariableTQ::m21).digi(pow(stub.z(), 2)); + const double invV0 = tq->format(VariableTQ::invV0).digi(1. / pow(stub.dPhi(), 2)); + const double invV1 = tq->format(VariableTQ::invV1).digi(1. / pow(stub.dZ(), 2)); + const double stubchi2rphi = tq->format(VariableTQ::chi2rphi).digi(m20 * invV0); + const double stubchi2rz = tq->format(VariableTQ::chi2rz).digi(m21 * invV1); + trackchi2rphi += stubchi2rphi; + trackchi2rz += stubchi2rz; + } + if (trackchi2rphi > tq->range(VariableTQ::chi2rphi)) + trackchi2rphi = tq->range(VariableTQ::chi2rphi) - tq->base(VariableTQ::chi2rphi) / 2.; + if (trackchi2rz > tq->range(VariableTQ::chi2rz)) + trackchi2rz = tq->range(VariableTQ::chi2rz) - tq->base(VariableTQ::chi2rz) / 2.; + // calc bdt inputs + const double cot = tq->scaleCot(df->format(Variable::cot, Process::dr).integer(track.cot())); + const double z0 = + tq->scaleZ0(df->format(Variable::zT, Process::kf).integer(track.zT() - setup->chosenRofZ() * track.cot())); + const int nstub = hitPattern.count(); + const int n_missint = hitPattern.count(hitPattern.plEncode() + 1, setup->numLayers(), false); + // use simulation for bendchi2 + const TTTrackRef& ttTrackRef = frameTrack.first; + const int region = ttTrackRef->phiSector(); + const double aRinv = -.5 * track.inv2R(); + const double aphi = deltaPhi(track.phiT() - track.inv2R() * setup->chosenRofPhi() + region * setup->baseRegion()); + const double aTanLambda = track.cot(); + const double az0 = track.zT() - track.cot() * setup->chosenRofZ(); + const double ad0 = ttTrackRef->d0(); + static constexpr double aChi2xyfit = 0.; + static constexpr double aChi2zfit = 0.; + static constexpr double trkMVA1 = 0.; + static constexpr double trkMVA2 = 0.; + static constexpr double trkMVA3 = 0.; + static constexpr unsigned int aHitpattern = 0; + const unsigned int nPar = ttTrackRef->nFitPars(); + static const double Bfield = setup->bField(); + TTTrack ttTrack( + aRinv, aphi, aTanLambda, az0, ad0, aChi2xyfit, aChi2zfit, trkMVA1, trkMVA2, trkMVA3, aHitpattern, nPar, Bfield); + ttTrack.setStubRefs(ttStubRefs); + ttTrack.setStubPtConsistency( + StubPtConsistency::getConsistency(ttTrack, setup->trackerGeometry(), setup->trackerTopology(), Bfield, nPar)); + const int chi2B = tq->toBinChi2B(ttTrack.chi2Bend()); + const int chi2rphi = tq->toBinchi2rphi(trackchi2rphi); + const int chi2rz = tq->toBinchi2rz(trackchi2rz); + // load in bdt + conifer::BDT, ap_fixed<10, 5>> bdt(tq->model().fullPath()); + // collect features and classify using bdt + const vector>& output = bdt.decision_function({cot, z0, chi2B, nstub, n_missint, chi2rphi, chi2rz}); + const float mva = output[0].to_float(); + // fill frames + TTBV ttBV = hitPattern; + ttBV += TTBV(tq->toBinMVA(mva), numBinsMVA_); + tq->format(VariableTQ::chi2rphi).attach(trackchi2rphi, ttBV); + tq->format(VariableTQ::chi2rz).attach(trackchi2rz, ttBV); + frames_.push_back(frameTrack); + frames_.emplace_back(frameTrack.first, ttBV.bs()); + } + + template <> + FormatTQ::FormatTQ(const DataFormats* dataFormats, const edm::ParameterSet& iConfig) + : DataFormat(false) { + const Format phi(dataFormats->setup()); + width_ = iConfig.getParameter("WidthM20"); + base_ = pow(phi.base(), 2) * pow(2., width_ - phi.width()); + calcRange(); + } + template <> + FormatTQ::FormatTQ(const DataFormats* dataFormats, const edm::ParameterSet& iConfig) + : DataFormat(false) { + const Format z(dataFormats->setup()); + width_ = iConfig.getParameter("WidthM21"); + base_ = pow(z.base(), 2) * pow(2., width_ - z.width()); + calcRange(); + } + template <> + FormatTQ::FormatTQ(const DataFormats* dataFormats, const edm::ParameterSet& iConfig) + : DataFormat(false) { + const Format dPhi(dataFormats->setup()); + width_ = iConfig.getParameter("WidthInvV0"); + range_ = 4.0 / pow(dPhi.base(), 2); + calcBase(); + } + template <> + FormatTQ::FormatTQ(const DataFormats* dataFormats, const edm::ParameterSet& iConfig) + : DataFormat(false) { + const Format dZ(dataFormats->setup()); + width_ = iConfig.getParameter("WidthInvV1"); + range_ = 4.0 / pow(dZ.base(), 2); + calcBase(); + } + template <> + FormatTQ::FormatTQ(const DataFormats* dataFormats, const edm::ParameterSet& iConfig) + : DataFormat(false) { + const FormatTQ m20(dataFormats, iConfig); + const FormatTQ invV0(dataFormats, iConfig); + const int shift = iConfig.getParameter("BaseShiftchi2rphi"); + width_ = iConfig.getParameter("Widthchi2rphi"); + base_ = pow(2., shift); + calcRange(); + } + template <> + FormatTQ::FormatTQ(const DataFormats* dataFormats, const edm::ParameterSet& iConfig) + : DataFormat(false) { + const FormatTQ m21(dataFormats, iConfig); + const FormatTQ invV1(dataFormats, iConfig); + const int shift = iConfig.getParameter("BaseShiftchi2rz"); + width_ = iConfig.getParameter("Widthchi2rz"); + base_ = pow(2., shift); + calcRange(); + } + + // Controls the conversion between TTTrack features and ML model training features + vector TrackQuality::featureTransform(TTTrack& aTrack, + vector const& featureNames) const { + // List input features for MVA in proper order below, the current features options are + // {"phi", "eta", "z0", "bendchi2_bin", "nstub", "nlaymiss_interior", "chi2rphi_bin", + // "chi2rz_bin"} + // + // To use more features, they must be created here and added to feature_map below + vector transformedFeatures; + // Define feature map, filled as features are generated + map feature_map; + // -------- calculate feature variables -------- + // calculate number of missed interior layers from hitpattern + int tmp_trk_hitpattern = aTrack.hitPattern(); + int nbits = floor(log2(tmp_trk_hitpattern)) + 1; + int lay_i = 0; + int tmp_trk_nlaymiss_interior = 0; + bool seq = false; + for (int i = 0; i < nbits; i++) { + lay_i = ((1 << i) & tmp_trk_hitpattern) >> i; //0 or 1 in ith bit (right to left) + + if (lay_i && !seq) + seq = true; //sequence starts when first 1 found + if (!lay_i && seq) + tmp_trk_nlaymiss_interior++; + } + // binned chi2 variables + int tmp_trk_bendchi2_bin = aTrack.getBendChi2Bits(); + int tmp_trk_chi2rphi_bin = aTrack.getChi2RPhiBits(); + int tmp_trk_chi2rz_bin = aTrack.getChi2RZBits(); + // get the nstub + vector stubRefs = aTrack.getStubRefs(); + int tmp_trk_nstub = stubRefs.size(); + // get other variables directly from TTTrack + float tmp_trk_z0 = aTrack.z0(); + float tmp_trk_z0_scaled = tmp_trk_z0 / abs(aTrack.minZ0); + float tmp_trk_phi = aTrack.phi(); + float tmp_trk_eta = aTrack.eta(); + float tmp_trk_tanl = aTrack.tanL(); + // -------- fill the feature map --------- + feature_map["nstub"] = float(tmp_trk_nstub); + feature_map["z0"] = tmp_trk_z0; + feature_map["z0_scaled"] = tmp_trk_z0_scaled; + feature_map["phi"] = tmp_trk_phi; + feature_map["eta"] = tmp_trk_eta; + feature_map["nlaymiss_interior"] = float(tmp_trk_nlaymiss_interior); + feature_map["bendchi2_bin"] = tmp_trk_bendchi2_bin; + feature_map["chi2rphi_bin"] = tmp_trk_chi2rphi_bin; + feature_map["chi2rz_bin"] = tmp_trk_chi2rz_bin; + feature_map["tanl"] = tmp_trk_tanl; + // fill tensor with track params + transformedFeatures.reserve(featureNames.size()); + for (const string& feature : featureNames) + transformedFeatures.push_back(feature_map[feature]); + return transformedFeatures; + } + + // Passed by reference a track without MVA filled, method fills the track's MVA field + void TrackQuality::setL1TrackQuality(TTTrack& aTrack) const { + // load in bdt + conifer::BDT bdt(this->model_.fullPath()); + // collect features and classify using bdt + vector inputs = featureTransform(aTrack, this->featureNames_); + vector output = bdt.decision_function(inputs); + aTrack.settrkMVA1(1. / (1. + exp(-output.at(0)))); + } + +} // namespace trackerTFP diff --git a/L1Trigger/TrackerTFP/src/ZHoughTransform.cc b/L1Trigger/TrackerTFP/src/ZHoughTransform.cc deleted file mode 100644 index dde6aa0e0bc8e..0000000000000 --- a/L1Trigger/TrackerTFP/src/ZHoughTransform.cc +++ /dev/null @@ -1,322 +0,0 @@ -#include "L1Trigger/TrackerTFP/interface/ZHoughTransform.h" - -#include -#include -#include -#include -#include -#include -#include -#include - -using namespace std; -using namespace edm; -using namespace tt; - -namespace trackerTFP { - - ZHoughTransform::ZHoughTransform(const ParameterSet& iConfig, - const Setup* setup, - const DataFormats* dataFormats, - int region) - : enableTruncation_(iConfig.getParameter("EnableTruncation")), - setup_(setup), - dataFormats_(dataFormats), - region_(region), - input_(dataFormats->numChannel(Process::mht)), - stage_(0) {} - - // read in and organize input product (fill vector input_) - void ZHoughTransform::consume(const StreamsStub& streams) { - auto valid = [](int sum, const FrameStub& frame) { return sum + (frame.first.isNonnull() ? 1 : 0); }; - const int offset = region_ * dataFormats_->numChannel(Process::mht); - int nStubsMHT(0); - for (int channel = 0; channel < dataFormats_->numChannel(Process::mht); channel++) { - const StreamStub& stream = streams[offset + channel]; - nStubsMHT += accumulate(stream.begin(), stream.end(), 0, valid); - } - stubsZHT_.reserve(nStubsMHT * (setup_->zhtNumCells() * setup_->zhtNumStages())); - for (int channel = 0; channel < dataFormats_->numChannel(Process::mht); channel++) { - const StreamStub& stream = streams[offset + channel]; - vector& stubs = input_[channel]; - stubs.reserve(stream.size()); - // Store input stubs in vector, so rest of ZHT algo can work with pointers to them (saves CPU) - for (const FrameStub& frame : stream) { - StubZHT* stub = nullptr; - if (frame.first.isNonnull()) { - StubMHT stubMHT(frame, dataFormats_); - stubsZHT_.emplace_back(stubMHT); - stub = &stubsZHT_.back(); - } - stubs.push_back(stub); - } - } - } - - // fill output products - void ZHoughTransform::produce(StreamsStub& accepted, StreamsStub& lost) { - vector> streams(dataFormats_->numChannel(Process::mht)); - for (int channel = 0; channel < dataFormats_->numChannel(Process::mht); channel++) - streams[channel] = deque(input_[channel].begin(), input_[channel].end()); - vector> stubsCells(dataFormats_->numChannel(Process::mht) * setup_->zhtNumCells()); - for (stage_ = 0; stage_ < setup_->zhtNumStages(); stage_++) { - // fill ZHT cells - for (int channel = 0; channel < dataFormats_->numChannel(Process::mht); channel++) - fill(channel, streams[channel], stubsCells); - // perform static load balancing - for (int channel = 0; channel < dataFormats_->numChannel(Process::mht); channel++) { - vector> tmp(setup_->zhtNumCells()); - // gather streams to mux together: same ZHT cell of 4 adjacent ZHT input streams - for (int k = 0; k < setup_->zhtNumCells(); k++) - //swap(tmp[k], stubsCells[(channel / setup_->zhtNumCells()) * dataFormats_->numChannel(Process::mht) + channel % setup_->zhtNumCells() + k * setup_->zhtNumCells()]); - swap(tmp[k], stubsCells[channel * setup_->zhtNumCells() + k]); - slb(tmp, streams[channel], lost[channel]); - } - } - // fill output product - for (int channel = 0; channel < dataFormats_->numChannel(Process::mht); channel++) { - deque& stubs = streams[channel]; - StreamStub& stream = accepted[region_ * dataFormats_->numChannel(Process::mht) + channel]; - merge(stubs, stream); - } - } - - // perform finer pattern recognition per track - void ZHoughTransform::fill(int channel, const deque& stubs, vector>& streams) { - if (stubs.empty()) - return; - const double baseZT = - dataFormats_->format(Variable::zT, Process::zht).base() * pow(2, setup_->zhtNumStages() - stage_); - const double baseCot = - dataFormats_->format(Variable::cot, Process::zht).base() * pow(2, setup_->zhtNumStages() - stage_); - int id; - auto different = [&id](StubZHT* stub) { return !stub || id != stub->trackId(); }; - for (auto it = stubs.begin(); it != stubs.end();) { - if (!*it) { - const auto begin = find_if(it, stubs.end(), [](StubZHT* stub) { return stub; }); - const int nGaps = distance(it, begin); - for (deque& stream : streams) - stream.insert(stream.end(), nGaps, nullptr); - it = begin; - continue; - } - const auto start = it; - const double cotGlobal = (*start)->cotf() + setup_->sectorCot((*start)->sectorEta()); - id = (*it)->trackId(); - it = find_if(it, stubs.end(), different); - const int size = distance(start, it); - // create finer track candidates stub container - vector> mhtCells(setup_->zhtNumCells()); - for (vector& mhtCell : mhtCells) - mhtCell.reserve(size); - // fill finer track candidates stub container - for (auto stub = start; stub != it; stub++) { - const double r = (*stub)->r() + setup_->chosenRofPhi() - setup_->chosenRofZ(); - const double chi = (*stub)->chi(); - const double dChi = setup_->dZ((*stub)->ttStubRef(), cotGlobal); - // identify finer track candidates for this stub - // 0 and 1 belong to the ZHT cells with smaller cot; 0 and 2 belong to those with smaller zT - vector cells; - cells.reserve(setup_->zhtNumCells()); - const bool compA = 2. * abs(chi) < baseZT + dChi; - const bool compB = 2. * abs(chi) < abs(r) * baseCot + dChi; - const bool compC = 2. * abs(chi) < dChi; - if (chi >= 0. && r >= 0.) { - cells.push_back(1); - if (compA) - cells.push_back(3); - if (compB) - cells.push_back(0); - if (compC) - cells.push_back(2); - } - if (chi >= 0. && r < 0.) { - cells.push_back(3); - if (compA) - cells.push_back(1); - if (compB) - cells.push_back(2); - if (compC) - cells.push_back(0); - } - if (chi < 0. && r >= 0.) { - cells.push_back(2); - if (compA) - cells.push_back(0); - if (compB) - cells.push_back(3); - if (compC) - cells.push_back(1); - } - if (chi < 0. && r < 0.) { - cells.push_back(0); - if (compA) - cells.push_back(2); - if (compB) - cells.push_back(1); - if (compC) - cells.push_back(3); - } - for (int cell : cells) { - const double cot = (cell / setup_->zhtNumBinsZT() - .5) * baseCot / 2.; - const double zT = (cell % setup_->zhtNumBinsZT() - .5) * baseZT / 2.; - stubsZHT_.emplace_back(**stub, zT, cot, cell); - mhtCells[cell].push_back(&stubsZHT_.back()); - } - } - // perform pattern recognition - for (int sel = 0; sel < setup_->zhtNumCells(); sel++) { - deque& stream = streams[channel * setup_->zhtNumCells() + sel]; - vector& mhtCell = mhtCells[sel]; - set layers; - auto toLayer = [](StubZHT* stub) { return stub->layer(); }; - transform(mhtCell.begin(), mhtCell.end(), inserter(layers, layers.begin()), toLayer); - if ((int)layers.size() < setup_->mhtMinLayers()) - mhtCell.clear(); - for (StubZHT* stub : mhtCell) - stream.push_back(stub); - stream.insert(stream.end(), size - (int)mhtCell.size(), nullptr); - } - } - for (int sel = 0; sel < setup_->zhtNumCells(); sel++) { - deque& stream = streams[channel * setup_->zhtNumCells() + sel]; - // remove all gaps between end and last stub - for (auto it = stream.end(); it != stream.begin();) - it = (*--it) ? stream.begin() : stream.erase(it); - // read out fine track cannot start before rough track has read in completely, add gaps to take this into account - int pos(0); - for (auto it = stream.begin(); it != stream.end();) { - if (!(*it)) { - it = stream.erase(it); - continue; - } - id = (*it)->trackId(); - const int s = distance(it, find_if(it, stream.end(), different)); - const int d = distance(stream.begin(), it); - pos += s; - if (d < pos) { - const int diff = pos - d; - it = stream.insert(it, diff, nullptr); - it = next(it, diff); - } else - it = stream.erase(remove(next(stream.begin(), pos), it, nullptr), it); - it = next(it, s); - } - // adjust stream start so that first output stub is in first place in case of quickest track - if (!stream.empty()) - stream.erase(stream.begin(), next(stream.begin(), setup_->mhtMinLayers())); - } - } - - // Static load balancing of inputs: mux 4 streams to 1 stream - void ZHoughTransform::slb(vector>& inputs, deque& accepted, StreamStub& lost) const { - accepted.clear(); - if (all_of(inputs.begin(), inputs.end(), [](const deque& stubs) { return stubs.empty(); })) - return; - // input fifos - vector> stacks(setup_->zhtNumCells()); - // helper for handshake - TTBV empty(-1, setup_->zhtNumCells(), true); - TTBV enable(0, setup_->zhtNumCells()); - // clock accurate firmware emulation, each while trip describes one clock tick, one stub in and one stub out per tick - while (!all_of(inputs.begin(), inputs.end(), [](const deque& d) { return d.empty(); }) or - !all_of(stacks.begin(), stacks.end(), [](const deque& d) { return d.empty(); })) { - // store stub in fifo - for (int channel = 0; channel < setup_->zhtNumCells(); channel++) { - StubZHT* stub = pop_front(inputs[channel]); - if (stub) - stacks[channel].push_back(stub); - } - // identify empty fifos - for (int channel = 0; channel < setup_->zhtNumCells(); channel++) - empty[channel] = stacks[channel].empty(); - // chose new fifo to read from if current fifo got empty - const int iEnableOld = enable.plEncode(); - if (enable.none() || empty[iEnableOld]) { - enable.reset(); - const int iNotEmpty = empty.plEncode(false); - if (iNotEmpty < setup_->zhtNumCells()) - enable.set(iNotEmpty); - } - // read from chosen fifo - const int iEnable = enable.plEncode(); - if (enable.any()) - accepted.push_back(pop_front(stacks[iEnable])); - else - // gap if no fifo has been chosen - accepted.push_back(nullptr); - } - // perform truncation if desired - if (enableTruncation_ && (int)accepted.size() > setup_->numFrames()) { - const auto limit = next(accepted.begin(), setup_->numFrames()); - auto valid = [](int sum, StubZHT* stub) { return sum + (stub ? 1 : 0); }; - const int nLost = accumulate(limit, accepted.end(), 0, valid); - lost.reserve(nLost); - for (auto it = limit; it != accepted.end(); it++) - if (*it) - lost.emplace_back((*it)->frame()); - accepted.erase(limit, accepted.end()); - } - // cosmetics -- remove gaps at the end of stream - for (auto it = accepted.end(); it != accepted.begin();) - it = (*--it) == nullptr ? accepted.erase(it) : accepted.begin(); - } - - // - void ZHoughTransform::merge(deque& stubs, StreamStub& stream) const { - stubs.erase(remove(stubs.begin(), stubs.end(), nullptr), stubs.end()); - /*stream.reserve(stubs.size()); - transform(stubs.begin(), stubs.end(), back_inserter(stream), [](StubZHT* stub){ return stub->frame(); }); - return;*/ - map>> candidates; - const int weight = setup_->zhtNumCells() * pow(2, setup_->zhtNumStages()); - for (const StubZHT* stub : stubs) - candidates[stub->trackId() / weight].emplace(stub->cot(), stub->zT()); - vector> tracks(candidates.size()); - for (auto it = stubs.begin(); it != stubs.end();) { - const auto start = it; - const int id = (*it)->trackId(); - const int candId = id / weight; - const auto m = candidates.find(candId); - pair cotp(9e9, -9e9); - pair zTp(9e9, -9e9); - for (const pair& para : m->second) { - cotp = {min(cotp.first, para.first), max(cotp.second, para.first)}; - zTp = {min(zTp.first, para.second), max(zTp.second, para.second)}; - } - const int cot = (cotp.first + cotp.second) / 2; - const int zT = (cotp.first + cotp.second) / 2; - const int pos = distance(candidates.begin(), m); - deque& track = tracks[pos]; - auto different = [id](const StubZHT* stub) { return id != stub->trackId(); }; - it = find_if(it, stubs.end(), different); - for (auto s = start; s != it; s++) { - if (find_if(track.begin(), track.end(), [s](const FrameStub& stub) { - return (*s)->ttStubRef() == stub.first; - }) != track.end()) - continue; - const StubZHT stub(**s, cot, zT); - track.push_back(stub.frame()); - } - } - const int size = accumulate(tracks.begin(), tracks.end(), 0, [](int sum, const deque& stubs) { - return sum + (int)stubs.size(); - }); - stream.reserve(size); - for (deque& track : tracks) - for (const FrameStub& stub : track) - stream.push_back(stub); - } - - // remove and return first element of deque, returns nullptr if empty - template - T* ZHoughTransform::pop_front(deque& ts) const { - T* t = nullptr; - if (!ts.empty()) { - t = ts.front(); - ts.pop_front(); - } - return t; - } - -} // namespace trackerTFP diff --git a/L1Trigger/TrackerTFP/test/AnalyzerKFin.cc b/L1Trigger/TrackerTFP/test/AnalyzerCTB.cc similarity index 58% rename from L1Trigger/TrackerTFP/test/AnalyzerKFin.cc rename to L1Trigger/TrackerTFP/test/AnalyzerCTB.cc index 14a821ff226fc..12c8050bdebfb 100644 --- a/L1Trigger/TrackerTFP/test/AnalyzerKFin.cc +++ b/L1Trigger/TrackerTFP/test/AnalyzerCTB.cc @@ -18,6 +18,7 @@ #include #include +#include #include #include @@ -33,14 +34,14 @@ using namespace tt; namespace trackerTFP { - /*! \class trackerTFP::AnalyzerKFin - * \brief Class to analyze hardware like structured TTStub Collection generated by Seed Filter + /*! \class trackerTFP::AnalyzerCTB + * \brief Class to analyze hardware like structured TTStub Collection generated by Clean Track Builder * \author Thomas Schuh * \date 2020, April */ - class AnalyzerKFin : public one::EDAnalyzer { + class AnalyzerCTB : public one::EDAnalyzer { public: - AnalyzerKFin(const ParameterSet& iConfig); + AnalyzerCTB(const ParameterSet& iConfig); void beginJob() override {} void beginRun(const Run& iEvent, const EventSetup& iSetup) override; void analyze(const Event& iEvent, const EventSetup& iSetup) override; @@ -55,15 +56,10 @@ namespace trackerTFP { int channel) const; // void associate(const vector>& tracks, const StubAssociation* ass, set& tps, int& sum) const; - // ED input token of stubs - EDGetTokenT edGetTokenAcceptedStubs_; + EDGetTokenT edGetTokenStubs_; // ED input token of tracks - EDGetTokenT edGetTokenAcceptedTracks_; - // ED input token of lost stubs - EDGetTokenT edGetTokenLostStubs_; - // ED input token of lost tracks - EDGetTokenT edGetTokenLostTracks_; + EDGetTokenT edGetTokenTracks_; // ED input token of TTStubRef to TPPtr association for tracking efficiency EDGetTokenT edGetTokenSelection_; // ED input token of TTStubRef to recontructable TPPtr association @@ -84,25 +80,34 @@ namespace trackerTFP { // Histograms TProfile* prof_; - TProfile* profChannel_; - TH1F* hisChannel_; + TProfile* profChan_; + TProfile* profStubs_; + TProfile* profTracks_; + TH1F* hisChan_; + TH1F* hisStubs_; + TH1F* hisTracks_; + TH1F* hisLayers_; + TH1F* hisNumLayers_; + TProfile* profNumLayers_; + TH1F* hisEffZT_; + TH1F* hisEffZTTotal_; + TEfficiency* effZT_; + TH1F* hisEffInv2R_; + TH1F* hisEffInv2RTotal_; + TEfficiency* effInv2R_; // printout stringstream log_; }; - AnalyzerKFin::AnalyzerKFin(const ParameterSet& iConfig) : useMCTruth_(iConfig.getParameter("UseMCTruth")) { + AnalyzerCTB::AnalyzerCTB(const ParameterSet& iConfig) : useMCTruth_(iConfig.getParameter("UseMCTruth")) { usesResource("TFileService"); // book in- and output ED products - const string& label = iConfig.getParameter("LabelKFin"); - const string& branchAcceptedStubs = iConfig.getParameter("BranchAcceptedStubs"); - const string& branchAcceptedTracks = iConfig.getParameter("BranchAcceptedTracks"); - const string& branchLostStubs = iConfig.getParameter("BranchLostStubs"); - const string& branchLostTracks = iConfig.getParameter("BranchLostTracks"); - edGetTokenAcceptedStubs_ = consumes(InputTag(label, branchAcceptedStubs)); - edGetTokenAcceptedTracks_ = consumes(InputTag(label, branchAcceptedTracks)); - edGetTokenLostStubs_ = consumes(InputTag(label, branchLostStubs)); - edGetTokenLostTracks_ = consumes(InputTag(label, branchLostTracks)); + const string& label = iConfig.getParameter("OutputLabelCTB"); + const string& branchStubs = iConfig.getParameter("BranchStubs"); + const string& branchTracks = iConfig.getParameter("BranchTracks"); + edGetTokenStubs_ = consumes(InputTag(label, branchStubs)); + edGetTokenTracks_ = consumes(InputTag(label, branchTracks)); if (useMCTruth_) { const auto& inputTagSelecttion = iConfig.getParameter("InputTagSelection"); const auto& inputTagReconstructable = iConfig.getParameter("InputTagReconstructable"); @@ -117,7 +122,7 @@ namespace trackerTFP { log_.precision(4); } - void AnalyzerKFin::beginRun(const Run& iEvent, const EventSetup& iSetup) { + void AnalyzerCTB::beginRun(const Run& iEvent, const EventSetup& iSetup) { // helper class to store configurations setup_ = &iSetup.getData(esGetTokenSetup_); // helper class to extract structured data from tt::Frames @@ -125,38 +130,62 @@ namespace trackerTFP { // book histograms Service fs; TFileDirectory dir; - dir = fs->mkdir("KFin"); + dir = fs->mkdir("CTB"); prof_ = dir.make("Counts", ";", 9, 0.5, 9.5); prof_->GetXaxis()->SetBinLabel(1, "Stubs"); prof_->GetXaxis()->SetBinLabel(2, "Tracks"); - prof_->GetXaxis()->SetBinLabel(3, "Lost Tracks"); prof_->GetXaxis()->SetBinLabel(4, "Matched Tracks"); prof_->GetXaxis()->SetBinLabel(5, "All Tracks"); prof_->GetXaxis()->SetBinLabel(6, "Found TPs"); prof_->GetXaxis()->SetBinLabel(7, "Found selected TPs"); - prof_->GetXaxis()->SetBinLabel(8, "Lost TPs"); prof_->GetXaxis()->SetBinLabel(9, "All TPs"); // channel occupancy constexpr int maxOcc = 180; - const int numChannels = dataFormats_->numChannel(Process::kfin); - hisChannel_ = dir.make("His Channel Occupancy", ";", maxOcc, -.5, maxOcc - .5); - profChannel_ = dir.make("Prof Channel Occupancy", ";", numChannels, -.5, numChannels - .5); + const int numChannelsTracks = dataFormats_->numChannel(Process::ctb); + const int numChannelsStubs = numChannelsTracks * setup_->numLayers(); + hisChan_ = dir.make("His Channel Occupancy", ";", maxOcc, -.5, maxOcc - .5); + profChan_ = dir.make("Prof Channel Occupancy", ";", numChannelsTracks, -.5, numChannelsTracks - .5); + // stub occupancy + hisStubs_ = dir.make("His Channel Occupancy", ";", maxOcc, -.5, maxOcc - .5); + profStubs_ = dir.make("Prof Channel Occupancy", ";", numChannelsStubs, -.5, numChannelsStubs - .5); + // track occupancy + hisTracks_ = dir.make("His Track Occupancy", ";", maxOcc, -.5, maxOcc - .5); + profTracks_ = dir.make("Prof Track Occupancy", ";", numChannelsTracks, -.5, numChannelsTracks - .5); + // layers + hisLayers_ = dir.make("HisLayers", ";", 8, 0, 8); + hisNumLayers_ = dir.make("HisNumLayers", ";", 9, 0, 9); + profNumLayers_ = dir.make("Prof NumLayers", ";", 32, 0, 2.4); + // Efficiencies + const double rangeZT = dataFormats_->format(Variable::zT, Process::dr).range(); + const int zTBins = setup_->gpNumBinsZT(); + hisEffZTTotal_ = dir.make("HisTPZTTotal", ";", zTBins, -rangeZT / 2, rangeZT / 2); + hisEffZT_ = dir.make("HisTPZT", ";", zTBins, -rangeZT / 2, rangeZT / 2); + effZT_ = dir.make("EffZT", ";", zTBins, -rangeZT / 2, rangeZT / 2); + const double rangeInv2R = dataFormats_->format(Variable::inv2R, Process::dr).range(); + const int inv2RBins = (setup_->htNumBinsInv2R() + 2) * 2; + hisEffInv2R_ = dir.make("HisTPInv2R", ";", inv2RBins, -rangeInv2R / 2., rangeInv2R / 2.); + hisEffInv2RTotal_ = dir.make("HisTPInv2RTotal", ";", inv2RBins, -rangeInv2R / 2., rangeInv2R / 2.); + effInv2R_ = dir.make("EffInv2R", ";", inv2RBins, -rangeInv2R / 2., rangeInv2R / 2.); } - void AnalyzerKFin::analyze(const Event& iEvent, const EventSetup& iSetup) { + void AnalyzerCTB::analyze(const Event& iEvent, const EventSetup& iSetup) { + auto fill = [this](const TPPtr& tpPtr, TH1F* hisZT, TH1F* hisInv2R) { + const double tpPhi0 = tpPtr->phi(); + const double tpCot = sinh(tpPtr->eta()); + const math::XYZPointD& v = tpPtr->vertex(); + const double tpZ0 = v.z() - tpCot * (v.x() * cos(tpPhi0) + v.y() * sin(tpPhi0)); + const double tpZT = tpZ0 + tpCot * setup_->chosenRofZ(); + const double tpInv2R = tpPtr->charge() / tpPtr->pt() * setup_->invPtToDphi(); + hisZT->Fill(tpZT); + hisInv2R->Fill(tpInv2R); + }; // read in ht products - Handle handleAcceptedStubs; - iEvent.getByToken(edGetTokenAcceptedStubs_, handleAcceptedStubs); - const StreamsStub& acceptedStubs = *handleAcceptedStubs; - Handle handleAcceptedTracks; - iEvent.getByToken(edGetTokenAcceptedTracks_, handleAcceptedTracks); - const StreamsTrack& acceptedTracks = *handleAcceptedTracks; - Handle handleLostStubs; - iEvent.getByToken(edGetTokenLostStubs_, handleLostStubs); - const StreamsStub& lostStubs = *handleLostStubs; - Handle handleLostTracks; - iEvent.getByToken(edGetTokenLostTracks_, handleLostTracks); - const StreamsTrack& lostTracks = *handleLostTracks; + Handle handleStubs; + iEvent.getByToken(edGetTokenStubs_, handleStubs); + const StreamsStub& acceptedStubs = *handleStubs; + Handle handleTracks; + iEvent.getByToken(edGetTokenTracks_, handleTracks); + const StreamsTrack& acceptedTracks = *handleTracks; // read in MCTruth const StubAssociation* selection = nullptr; const StubAssociation* reconstructable = nullptr; @@ -168,98 +197,101 @@ namespace trackerTFP { Handle handleReconstructable; iEvent.getByToken(edGetTokenReconstructable_, handleReconstructable); reconstructable = handleReconstructable.product(); + for (const auto& p : selection->getTrackingParticleToTTStubsMap()) + fill(p.first, hisEffZTTotal_, hisEffInv2RTotal_); } // analyze ht products and associate found tracks with reconstrucable TrackingParticles set tpPtrs; set tpPtrsSelection; - set tpPtrsLost; int allMatched(0); int allTracks(0); for (int region = 0; region < setup_->numRegions(); region++) { - const int offset = region * dataFormats_->numChannel(Process::kf); + const int offsetTrack = region * dataFormats_->numChannel(Process::ctb); int nStubs(0); int nTracks(0); - int nLost(0); - for (int channel = 0; channel < dataFormats_->numChannel(Process::kf); channel++) { + for (int channel = 0; channel < dataFormats_->numChannel(Process::ctb); channel++) { + const int indexTrack = offsetTrack + channel; + const int size = acceptedTracks[indexTrack].size(); + hisChan_->Fill(size); + profChan_->Fill(channel, size); + const int offsetStub = indexTrack * setup_->numLayers(); + for (int layer = 0; layer < setup_->numLayers(); layer++) { + const StreamStub& stream = acceptedStubs[offsetStub + layer]; + const int nStubs = accumulate(stream.begin(), stream.end(), 0, [](int& sum, const FrameStub& frame) { + return sum += (frame.first.isNonnull() ? 1 : 0); + }); + hisStubs_->Fill(nStubs); + profStubs_->Fill(channel * setup_->numLayers() + layer, nStubs); + } vector> tracks; - formTracks(acceptedTracks, acceptedStubs, tracks, offset + channel); - vector> lost; - formTracks(lostTracks, lostStubs, lost, offset + channel); + formTracks(acceptedTracks, acceptedStubs, tracks, indexTrack); + hisTracks_->Fill(tracks.size()); + profTracks_->Fill(channel, tracks.size()); nTracks += tracks.size(); nStubs += accumulate(tracks.begin(), tracks.end(), 0, [](int sum, const vector& track) { return sum + (int)track.size(); }); - nLost += lost.size(); allTracks += tracks.size(); if (!useMCTruth_) continue; int tmp(0); associate(tracks, selection, tpPtrsSelection, tmp); - associate(lost, selection, tpPtrsLost, tmp); associate(tracks, reconstructable, tpPtrs, allMatched); } prof_->Fill(1, nStubs); prof_->Fill(2, nTracks); - prof_->Fill(3, nLost); } - vector recovered; - recovered.reserve(tpPtrsLost.size()); - set_intersection(tpPtrsLost.begin(), tpPtrsLost.end(), tpPtrs.begin(), tpPtrs.end(), back_inserter(recovered)); - for (const TPPtr& tpPtr : recovered) - tpPtrsLost.erase(tpPtr); + for (const TPPtr& tpPtr : tpPtrsSelection) + fill(tpPtr, hisEffZT_, hisEffInv2R_); prof_->Fill(4, allMatched); prof_->Fill(5, allTracks); prof_->Fill(6, tpPtrs.size()); prof_->Fill(7, tpPtrsSelection.size()); - prof_->Fill(8, tpPtrsLost.size()); nEvents_++; } - void AnalyzerKFin::endJob() { + void AnalyzerCTB::endJob() { if (nEvents_ == 0) return; - // printout SF summary + // effi + effZT_->SetPassedHistogram(*hisEffZT_, "f"); + effZT_->SetTotalHistogram(*hisEffZTTotal_, "f"); + effInv2R_->SetPassedHistogram(*hisEffInv2R_, "f"); + effInv2R_->SetTotalHistogram(*hisEffInv2RTotal_, "f"); + // printout TB summary const double totalTPs = prof_->GetBinContent(9); const double numStubs = prof_->GetBinContent(1); const double numTracks = prof_->GetBinContent(2); - const double numTracksLost = prof_->GetBinContent(3); const double totalTracks = prof_->GetBinContent(5); const double numTracksMatched = prof_->GetBinContent(4); const double numTPsAll = prof_->GetBinContent(6); const double numTPsEff = prof_->GetBinContent(7); - const double numTPsLost = prof_->GetBinContent(8); const double errStubs = prof_->GetBinError(1); const double errTracks = prof_->GetBinError(2); - const double errTracksLost = prof_->GetBinError(3); const double fracFake = (totalTracks - numTracksMatched) / totalTracks; const double fracDup = (numTracksMatched - numTPsAll) / totalTracks; const double eff = numTPsEff / totalTPs; const double errEff = sqrt(eff * (1. - eff) / totalTPs / nEvents_); - const double effLoss = numTPsLost / totalTPs; - const double errEffLoss = sqrt(effLoss * (1. - effLoss) / totalTPs / nEvents_); - const vector nums = {numStubs, numTracks, numTracksLost}; - const vector errs = {errStubs, errTracks, errTracksLost}; + const vector nums = {numStubs, numTracks}; + const vector errs = {errStubs, errTracks}; const int wNums = ceil(log10(*max_element(nums.begin(), nums.end()))) + 5; const int wErrs = ceil(log10(*max_element(errs.begin(), errs.end()))) + 5; - log_ << " KFin SUMMARY " << endl; + log_ << " CTB SUMMARY " << endl; log_ << "number of stubs per TFP = " << setw(wNums) << numStubs << " +- " << setw(wErrs) << errStubs << endl; log_ << "number of tracks per TFP = " << setw(wNums) << numTracks << " +- " << setw(wErrs) << errTracks << endl; - log_ << "number of lost tracks per TFP = " << setw(wNums) << numTracksLost << " +- " << setw(wErrs) << errTracksLost - << endl; log_ << " max tracking efficiency = " << setw(wNums) << eff << " +- " << setw(wErrs) << errEff << endl; - log_ << " lost tracking efficiency = " << setw(wNums) << effLoss << " +- " << setw(wErrs) << errEffLoss << endl; log_ << " fake rate = " << setw(wNums) << fracFake << endl; log_ << " duplicate rate = " << setw(wNums) << fracDup << endl; log_ << "============================================================="; - LogPrint("L1Trigger/TrackerTFP") << log_.str(); + LogPrint(moduleDescription().moduleName()) << log_.str(); } // - void AnalyzerKFin::formTracks(const StreamsTrack& streamsTrack, - const StreamsStub& streamsStubs, - vector>& tracks, - int channel) const { + void AnalyzerCTB::formTracks(const StreamsTrack& streamsTrack, + const StreamsStub& streamsStubs, + vector>& tracks, + int channel) const { const int offset = channel * setup_->numLayers(); const StreamTrack& streamTrack = streamsTrack[channel]; const int numTracks = accumulate(streamTrack.begin(), streamTrack.end(), 0, [](int sum, const FrameTrack& frame) { @@ -284,22 +316,33 @@ namespace trackerTFP { } vector stubs; stubs.reserve(numStubs); + int numLayers(0); for (int layer = 0; layer < setup_->numLayers(); layer++) { + bool any(false); for (int f = frame; f < frame + size; f++) { const FrameStub& stub = streamsStubs[offset + layer][f]; - if (stub.first.isNonnull()) + if (stub.first.isNonnull()) { + any = true; stubs.push_back(stub.first); + } + } + if (any) { + hisLayers_->Fill(layer); + numLayers++; } } + const double cot = TrackCTB(frameTrack, dataFormats_).zT() / setup_->chosenRofZ(); + hisNumLayers_->Fill(numLayers); + profNumLayers_->Fill(abs(sinh(cot)), numLayers); tracks.push_back(stubs); } } // - void AnalyzerKFin::associate(const vector>& tracks, - const StubAssociation* ass, - set& tps, - int& sum) const { + void AnalyzerCTB::associate(const vector>& tracks, + const StubAssociation* ass, + set& tps, + int& sum) const { for (const vector& ttStubRefs : tracks) { const vector& tpPtrs = ass->associate(ttStubRefs); if (tpPtrs.empty()) @@ -311,4 +354,4 @@ namespace trackerTFP { } // namespace trackerTFP -DEFINE_FWK_MODULE(trackerTFP::AnalyzerKFin); +DEFINE_FWK_MODULE(trackerTFP::AnalyzerCTB); diff --git a/L1Trigger/TrackFindingTracklet/test/AnalyzerKFin.cc b/L1Trigger/TrackerTFP/test/AnalyzerDR.cc similarity index 57% rename from L1Trigger/TrackFindingTracklet/test/AnalyzerKFin.cc rename to L1Trigger/TrackerTFP/test/AnalyzerDR.cc index bc9503631e16e..7c7594ec0cead 100644 --- a/L1Trigger/TrackFindingTracklet/test/AnalyzerKFin.cc +++ b/L1Trigger/TrackerTFP/test/AnalyzerDR.cc @@ -15,7 +15,6 @@ #include "SimTracker/TrackTriggerAssociation/interface/StubAssociation.h" #include "L1Trigger/TrackTrigger/interface/Setup.h" #include "L1Trigger/TrackerTFP/interface/DataFormats.h" -#include "L1Trigger/TrackFindingTracklet/interface/ChannelAssignment.h" #include #include @@ -32,16 +31,16 @@ using namespace edm; using namespace trackerTFP; using namespace tt; -namespace trklet { +namespace trackerTFP { - /*! \class trklet::AnalyzerKFin - * \brief Class to analyze hardware like structured TTStub Collection generated by Tracklet + /*! \class trackerTFP::AnalyzerDR + * \brief Class to analyze hardware like structured track Collection generated by Duplicate Removal * \author Thomas Schuh - * \date 2020, Nov + * \date 2023, Feb */ - class AnalyzerKFin : public one::EDAnalyzer { + class AnalyzerDR : public one::EDAnalyzer { public: - AnalyzerKFin(const ParameterSet& iConfig); + AnalyzerDR(const ParameterSet& iConfig); void beginJob() override {} void beginRun(const Run& iEvent, const EventSetup& iSetup) override; void analyze(const Event& iEvent, const EventSetup& iSetup) override; @@ -59,15 +58,11 @@ namespace trklet { const StubAssociation* ass, set& tps, int& sum, - bool perfect = false) const; + bool perfect = true) const; // ED input token of stubs - EDGetTokenT edGetTokenAcceptedStubs_; + EDGetTokenT edGetTokenStubs_; // ED input token of tracks - EDGetTokenT edGetTokenAcceptedTracks_; - // ED input token of lost stubs - EDGetTokenT edGetTokenLostStubs_; - // ED input token of lost tracks - EDGetTokenT edGetTokenLostTracks_; + EDGetTokenT edGetTokenTracks_; // ED input token of TTStubRef to TPPtr association for tracking efficiency EDGetTokenT edGetTokenSelection_; // ED input token of TTStubRef to recontructable TPPtr association @@ -89,24 +84,22 @@ namespace trklet { TProfile* prof_; TProfile* profChannel_; + TProfile* profTracks_; TH1F* hisChannel_; + TH1F* hisTracks_; // printout stringstream log_; }; - AnalyzerKFin::AnalyzerKFin(const ParameterSet& iConfig) : useMCTruth_(iConfig.getParameter("UseMCTruth")) { + AnalyzerDR::AnalyzerDR(const ParameterSet& iConfig) : useMCTruth_(iConfig.getParameter("UseMCTruth")) { usesResource("TFileService"); // book in- and output ED products - const string& label = iConfig.getParameter("LabelKFin"); - const string& branchAcceptedStubs = iConfig.getParameter("BranchAcceptedStubs"); - const string& branchAcceptedTracks = iConfig.getParameter("BranchAcceptedTracks"); - const string& branchLostStubs = iConfig.getParameter("BranchLostStubs"); - const string& branchLostTracks = iConfig.getParameter("BranchLostTracks"); - edGetTokenAcceptedStubs_ = consumes(InputTag(label, branchAcceptedStubs)); - edGetTokenAcceptedTracks_ = consumes(InputTag(label, branchAcceptedTracks)); - edGetTokenLostStubs_ = consumes(InputTag(label, branchLostStubs)); - edGetTokenLostTracks_ = consumes(InputTag(label, branchLostTracks)); + const string& label = iConfig.getParameter("OutputLabelDR"); + const string& branchStubs = iConfig.getParameter("BranchStubs"); + const string& branchTracks = iConfig.getParameter("BranchTracks"); + edGetTokenStubs_ = consumes(InputTag(label, branchStubs)); + edGetTokenTracks_ = consumes(InputTag(label, branchTracks)); if (useMCTruth_) { const auto& inputTagSelecttion = iConfig.getParameter("InputTagSelection"); const auto& inputTagReconstructable = iConfig.getParameter("InputTagReconstructable"); @@ -121,7 +114,7 @@ namespace trklet { log_.precision(4); } - void AnalyzerKFin::beginRun(const Run& iEvent, const EventSetup& iSetup) { + void AnalyzerDR::beginRun(const Run& iEvent, const EventSetup& iSetup) { // helper class to store configurations setup_ = &iSetup.getData(esGetTokenSetup_); // helper class to extract structured data from tt::Frames @@ -129,39 +122,35 @@ namespace trklet { // book histograms Service fs; TFileDirectory dir; - dir = fs->mkdir("KFin"); - prof_ = dir.make("Counts", ";", 10, 0.5, 10.5); + dir = fs->mkdir("DR"); + prof_ = dir.make("Counts", ";", 12, 0.5, 12.5); prof_->GetXaxis()->SetBinLabel(1, "Stubs"); prof_->GetXaxis()->SetBinLabel(2, "Tracks"); - prof_->GetXaxis()->SetBinLabel(3, "Lost Tracks"); prof_->GetXaxis()->SetBinLabel(4, "Matched Tracks"); prof_->GetXaxis()->SetBinLabel(5, "All Tracks"); prof_->GetXaxis()->SetBinLabel(6, "Found TPs"); prof_->GetXaxis()->SetBinLabel(7, "Found selected TPs"); - prof_->GetXaxis()->SetBinLabel(8, "Lost TPs"); prof_->GetXaxis()->SetBinLabel(9, "All TPs"); - prof_->GetXaxis()->SetBinLabel(10, "Perfect TPs"); + prof_->GetXaxis()->SetBinLabel(10, "states"); + prof_->GetXaxis()->SetBinLabel(12, "max tp"); // channel occupancy constexpr int maxOcc = 180; - const int numChannels = setup_->kfNumWorker(); + const int numChannels = dataFormats_->numChannel(Process::dr); hisChannel_ = dir.make("His Channel Occupancy", ";", maxOcc, -.5, maxOcc - .5); profChannel_ = dir.make("Prof Channel Occupancy", ";", numChannels, -.5, numChannels - .5); + // track occupancy + hisTracks_ = dir.make("His Track Occupancy", ";", maxOcc, -.5, maxOcc - .5); + profTracks_ = dir.make("Prof Track Occupancy", ";", numChannels, -.5, numChannels - .5); } - void AnalyzerKFin::analyze(const Event& iEvent, const EventSetup& iSetup) { + void AnalyzerDR::analyze(const Event& iEvent, const EventSetup& iSetup) { // read in ht products - Handle handleAcceptedStubs; - iEvent.getByToken(edGetTokenAcceptedStubs_, handleAcceptedStubs); - const StreamsStub& acceptedStubs = *handleAcceptedStubs; - Handle handleAcceptedTracks; - iEvent.getByToken(edGetTokenAcceptedTracks_, handleAcceptedTracks); - const StreamsTrack& acceptedTracks = *handleAcceptedTracks; - Handle handleLostStubs; - iEvent.getByToken(edGetTokenLostStubs_, handleLostStubs); - const StreamsStub& lostStubs = *handleLostStubs; - Handle handleLostTracks; - iEvent.getByToken(edGetTokenLostTracks_, handleLostTracks); - const StreamsTrack& lostTracks = *handleLostTracks; + Handle handleStubs; + iEvent.getByToken(edGetTokenStubs_, handleStubs); + const StreamsStub& acceptedStubs = *handleStubs; + Handle handleTracks; + iEvent.getByToken(edGetTokenTracks_, handleTracks); + const StreamsTrack& acceptedTracks = *handleTracks; // read in MCTruth const StubAssociation* selection = nullptr; const StubAssociation* reconstructable = nullptr; @@ -177,135 +166,112 @@ namespace trklet { // analyze ht products and associate found tracks with reconstrucable TrackingParticles set tpPtrs; set tpPtrsSelection; - set tpPtrsPerfect; - set tpPtrsLost; + set tpPtrsMax; int allMatched(0); int allTracks(0); for (int region = 0; region < setup_->numRegions(); region++) { - const int offset = region * setup_->kfNumWorker(); + const int offset = region * dataFormats_->numChannel(Process::dr); int nStubs(0); int nTracks(0); - int nLost(0); - for (int channel = 0; channel < setup_->kfNumWorker(); channel++) { + for (int channel = 0; channel < dataFormats_->numChannel(Process::dr); channel++) { vector> tracks; formTracks(acceptedTracks, acceptedStubs, tracks, offset + channel); - vector> lost; - formTracks(lostTracks, lostStubs, lost, offset + channel); + hisTracks_->Fill(tracks.size()); + profTracks_->Fill(channel, tracks.size()); nTracks += tracks.size(); - nStubs += accumulate(tracks.begin(), tracks.end(), 0, [](int sum, const vector& track) { - return sum + (int)track.size(); + nStubs += accumulate(tracks.begin(), tracks.end(), 0, [](int& sum, const vector& track) { + return sum += (int)track.size(); }); - nLost += lost.size(); allTracks += tracks.size(); if (!useMCTruth_) continue; int tmp(0); associate(tracks, selection, tpPtrsSelection, tmp); - associate(tracks, selection, tpPtrsPerfect, tmp, true); - associate(lost, selection, tpPtrsLost, tmp); - associate(tracks, reconstructable, tpPtrs, allMatched); - const StreamTrack& stream = acceptedTracks[offset + channel]; - const auto end = - find_if(stream.rbegin(), stream.rend(), [](const FrameTrack& frame) { return frame.first.isNonnull(); }); - const int size = distance(stream.begin(), end.base()) - 1; + associate(tracks, reconstructable, tpPtrs, allMatched, false); + associate(tracks, selection, tpPtrsMax, tmp, false); + //cout << "DR " << tpPtrsSelection.size() << " " << tpPtrsMax.size() << endl; + const int size = acceptedTracks[offset + channel].size(); hisChannel_->Fill(size); profChannel_->Fill(channel, size); } prof_->Fill(1, nStubs); prof_->Fill(2, nTracks); - prof_->Fill(3, nLost); } - vector recovered; - recovered.reserve(tpPtrsLost.size()); - set_intersection(tpPtrsLost.begin(), tpPtrsLost.end(), tpPtrs.begin(), tpPtrs.end(), back_inserter(recovered)); - for (const TPPtr& tpPtr : recovered) - tpPtrsLost.erase(tpPtr); prof_->Fill(4, allMatched); prof_->Fill(5, allTracks); prof_->Fill(6, tpPtrs.size()); prof_->Fill(7, tpPtrsSelection.size()); - prof_->Fill(8, tpPtrsLost.size()); - prof_->Fill(10, tpPtrsPerfect.size()); + prof_->Fill(12, tpPtrsMax.size()); nEvents_++; } - void AnalyzerKFin::endJob() { + void AnalyzerDR::endJob() { if (nEvents_ == 0) return; - // printout SF summary + // printout DR summary const double totalTPs = prof_->GetBinContent(9); const double numStubs = prof_->GetBinContent(1); const double numTracks = prof_->GetBinContent(2); - const double numTracksLost = prof_->GetBinContent(3); const double totalTracks = prof_->GetBinContent(5); const double numTracksMatched = prof_->GetBinContent(4); const double numTPsAll = prof_->GetBinContent(6); const double numTPsEff = prof_->GetBinContent(7); - const double numTPsLost = prof_->GetBinContent(8); - const double numTPsEffPerfect = prof_->GetBinContent(10); + const double numTPsEffMax = prof_->GetBinContent(12); const double errStubs = prof_->GetBinError(1); const double errTracks = prof_->GetBinError(2); - const double errTracksLost = prof_->GetBinError(3); const double fracFake = (totalTracks - numTracksMatched) / totalTracks; const double fracDup = (numTracksMatched - numTPsAll) / totalTracks; const double eff = numTPsEff / totalTPs; const double errEff = sqrt(eff * (1. - eff) / totalTPs / nEvents_); - const double effLoss = numTPsLost / totalTPs; - const double errEffLoss = sqrt(effLoss * (1. - effLoss) / totalTPs / nEvents_); - const double effPerfect = numTPsEffPerfect / totalTPs; - const double errEffPerfect = sqrt(effPerfect * (1. - effPerfect) / totalTPs / nEvents_); - const vector nums = {numStubs, numTracks, numTracksLost}; - const vector errs = {errStubs, errTracks, errTracksLost}; + const double effMax = numTPsEffMax / totalTPs; + const double errEffMax = sqrt(effMax * (1. - effMax) / totalTPs / nEvents_); + const vector nums = {numStubs, numTracks}; + const vector errs = {errStubs, errTracks}; const int wNums = ceil(log10(*max_element(nums.begin(), nums.end()))) + 5; const int wErrs = ceil(log10(*max_element(errs.begin(), errs.end()))) + 5; - log_ << " KFin SUMMARY " << endl; + log_ << " DR SUMMARY " << endl; log_ << "number of stubs per TFP = " << setw(wNums) << numStubs << " +- " << setw(wErrs) << errStubs << endl; log_ << "number of tracks per TFP = " << setw(wNums) << numTracks << " +- " << setw(wErrs) << errTracks << endl; - log_ << "number of lost tracks per TFP = " << setw(wNums) << numTracksLost << " +- " << setw(wErrs) << errTracksLost - << endl; - log_ << " current tracking efficiency = " << setw(wNums) << effPerfect << " +- " << setw(wErrs) << errEffPerfect - << endl; - log_ << " max tracking efficiency = " << setw(wNums) << eff << " +- " << setw(wErrs) << errEff << endl; - log_ << " lost tracking efficiency = " << setw(wNums) << effLoss << " +- " << setw(wErrs) << errEffLoss << endl; + log_ << " tracking efficiency = " << setw(wNums) << eff << " +- " << setw(wErrs) << errEff << endl; + log_ << " max tracking efficiency = " << setw(wNums) << effMax << " +- " << setw(wErrs) << errEffMax << endl; log_ << " fake rate = " << setw(wNums) << fracFake << endl; log_ << " duplicate rate = " << setw(wNums) << fracDup << endl; log_ << "============================================================="; - LogPrint("L1Trigger/TrackerTFP") << log_.str(); + LogPrint(moduleDescription().moduleName()) << log_.str(); } // - void AnalyzerKFin::formTracks(const StreamsTrack& streamsTrack, - const StreamsStub& streamsStubs, - vector>& tracks, - int channel) const { + void AnalyzerDR::formTracks(const StreamsTrack& streamsTrack, + const StreamsStub& streamsStubs, + vector>& tracks, + int channel) const { const int offset = channel * setup_->numLayers(); const StreamTrack& streamTrack = streamsTrack[channel]; - const int numTracks = accumulate(streamTrack.begin(), streamTrack.end(), 0, [](int sum, const FrameTrack& frame) { - return sum + (frame.first.isNonnull() ? 1 : 0); + const int numTracks = accumulate(streamTrack.begin(), streamTrack.end(), 0, [](int& sum, const FrameTrack& frame) { + return sum += (frame.first.isNonnull() ? 1 : 0); }); tracks.reserve(numTracks); for (int frame = 0; frame < (int)streamTrack.size(); frame++) { const FrameTrack& frameTrack = streamTrack[frame]; if (frameTrack.first.isNull()) continue; - vector ttStubRefs; - ttStubRefs.reserve(setup_->numLayers()); + deque stubs; for (int layer = 0; layer < setup_->numLayers(); layer++) { const FrameStub& stub = streamsStubs[offset + layer][frame]; if (stub.first.isNonnull()) - ttStubRefs.push_back(stub.first); + stubs.push_back(stub.first); } - tracks.push_back(ttStubRefs); + tracks.emplace_back(stubs.begin(), stubs.end()); } } // - void AnalyzerKFin::associate(const vector>& tracks, - const StubAssociation* ass, - set& tps, - int& sum, - bool perfect) const { + void AnalyzerDR::associate(const vector>& tracks, + const StubAssociation* ass, + set& tps, + int& sum, + bool perfect) const { for (const vector& ttStubRefs : tracks) { const vector& tpPtrs = perfect ? ass->associateFinal(ttStubRefs) : ass->associate(ttStubRefs); if (tpPtrs.empty()) @@ -315,6 +281,6 @@ namespace trklet { } } -} // namespace trklet +} // namespace trackerTFP -DEFINE_FWK_MODULE(trklet::AnalyzerKFin); +DEFINE_FWK_MODULE(trackerTFP::AnalyzerDR); diff --git a/L1Trigger/TrackerTFP/test/AnalyzerDemonstrator.cc b/L1Trigger/TrackerTFP/test/AnalyzerDemonstrator.cc index 6b65410d03e0f..c2c5761252607 100644 --- a/L1Trigger/TrackerTFP/test/AnalyzerDemonstrator.cc +++ b/L1Trigger/TrackerTFP/test/AnalyzerDemonstrator.cc @@ -21,8 +21,8 @@ using namespace tt; namespace trackerTFP { /*! \class trackerTFP::AnalyzerDemonstrator - * \brief Class to demontrate correctness of track trigger emulators - * by comparing FW with SW + * \brief calls questasim to simulate the f/w and compares the results with clock-and-bit-accurate emulation. + * A single bit error interrupts the run. * \author Thomas Schuh * \date 2020, Nov */ @@ -64,13 +64,14 @@ namespace trackerTFP { // book in- and output ED products const string& labelIn = iConfig.getParameter("LabelIn"); const string& labelOut = iConfig.getParameter("LabelOut"); - const string& branchStubs = iConfig.getParameter("BranchAcceptedStubs"); - const string& branchTracks = iConfig.getParameter("BranchAcceptedTracks"); + const string& branchStubs = iConfig.getParameter("BranchStubs"); + const string& branchTracks = iConfig.getParameter("BranchTracks"); edGetTokenStubsIn_ = consumes(InputTag(labelIn, branchStubs)); - edGetTokenStubsOut_ = consumes(InputTag(labelOut, branchStubs)); - if (labelIn == "TrackerTFPProducerKFin" || labelIn == "TrackerTFPProducerKF") + if (labelOut != "ProducerTFP") + edGetTokenStubsOut_ = consumes(InputTag(labelOut, branchStubs)); + if (labelIn == "ProducerCTB" || labelIn == "ProducerKF" || labelIn == "ProducerDR") edGetTokenTracksIn_ = consumes(InputTag(labelIn, branchTracks)); - if (labelOut == "TrackerTFPProducerKF" || labelOut == "TrackerTFPProducerDR") + if (labelOut == "ProducerCTB" || labelOut == "ProducerKF" || labelOut == "ProducerDR" || labelOut == "ProducerTFP") edGetTokenTracksOut_ = consumes(InputTag(labelOut, branchTracks)); // book ES products esGetTokenSetup_ = esConsumes(); @@ -85,6 +86,8 @@ namespace trackerTFP { } void AnalyzerDemonstrator::analyze(const Event& iEvent, const EventSetup& iSetup) { + Handle handle; + iEvent.getByToken(edGetTokenStubsIn_, handle); vector> input; vector> output; convert(iEvent, edGetTokenTracksIn_, edGetTokenStubsIn_, input); diff --git a/L1Trigger/TrackerTFP/test/AnalyzerGP.cc b/L1Trigger/TrackerTFP/test/AnalyzerGP.cc index 7a38af978a155..cbec621da5505 100644 --- a/L1Trigger/TrackerTFP/test/AnalyzerGP.cc +++ b/L1Trigger/TrackerTFP/test/AnalyzerGP.cc @@ -14,9 +14,11 @@ #include "SimTracker/TrackTriggerAssociation/interface/StubAssociation.h" #include "L1Trigger/TrackTrigger/interface/Setup.h" +#include "L1Trigger/TrackerTFP/interface/DataFormats.h" #include #include +#include #include #include @@ -45,15 +47,17 @@ namespace trackerTFP { private: // ED input token of stubs - EDGetTokenT edGetTokenAccepted_; - // ED input token of lost stubs - EDGetTokenT edGetTokenLost_; + EDGetTokenT edGetToken_; // ED input token of TTStubRef to TPPtr association for tracking efficiency EDGetTokenT edGetTokenAss_; // Setup token - ESGetToken esGetToken_; + ESGetToken esGetTokenSetup_; + // DataFormats token + ESGetToken esGetTokenDataFormats_; // stores, calculates and provides run-time constants const Setup* setup_ = nullptr; + // helper class to extract structured data from tt::Frames + const DataFormats* dataFormats_ = nullptr; // enables analyze of TPs bool useMCTruth_; // @@ -64,6 +68,18 @@ namespace trackerTFP { TProfile* prof_; TProfile* profChannel_; TH1F* hisChannel_; + TH1F* hisEffEta_; + TH1F* hisEffEtaTotal_; + TEfficiency* effEta_; + TH1F* hisEffZT_; + TH1F* hisEffZTTotal_; + TEfficiency* effZT_; + TH1F* hisEffInv2R_; + TH1F* hisEffInv2RTotal_; + TEfficiency* effInv2R_; + TH1F* hisEffPT_; + TH1F* hisEffPTTotal_; + TEfficiency* effPT_; // printout stringstream log_; @@ -72,17 +88,16 @@ namespace trackerTFP { AnalyzerGP::AnalyzerGP(const ParameterSet& iConfig) : useMCTruth_(iConfig.getParameter("UseMCTruth")) { usesResource("TFileService"); // book in- and output ED products - const string& label = iConfig.getParameter("LabelGP"); - const string& branchAccepted = iConfig.getParameter("BranchAcceptedStubs"); - const string& branchLost = iConfig.getParameter("BranchLostStubs"); - edGetTokenAccepted_ = consumes(InputTag(label, branchAccepted)); - edGetTokenLost_ = consumes(InputTag(label, branchLost)); + const string& label = iConfig.getParameter("OutputLabelGP"); + const string& branch = iConfig.getParameter("BranchStubs"); + edGetToken_ = consumes(InputTag(label, branch)); if (useMCTruth_) { const auto& inputTagAss = iConfig.getParameter("InputTagSelection"); edGetTokenAss_ = consumes(inputTagAss); } - // book ES product - esGetToken_ = esConsumes(); + // book ES products + esGetTokenSetup_ = esConsumes(); + esGetTokenDataFormats_ = esConsumes(); // log config log_.setf(ios::fixed, ios::floatfield); log_.precision(4); @@ -90,16 +105,33 @@ namespace trackerTFP { void AnalyzerGP::beginRun(const Run& iEvent, const EventSetup& iSetup) { // helper class to store configurations - setup_ = &iSetup.getData(esGetToken_); + setup_ = &iSetup.getData(esGetTokenSetup_); + // helper class to extract structured data from tt::Frames + dataFormats_ = &iSetup.getData(esGetTokenDataFormats_); // book histograms Service fs; TFileDirectory dir; dir = fs->mkdir("GP"); prof_ = dir.make("Counts", ";", 4, 0.5, 4.5); - prof_->GetXaxis()->SetBinLabel(1, "Stubs"); - prof_->GetXaxis()->SetBinLabel(2, "Lost Stubs"); + prof_->GetXaxis()->SetBinLabel(1, "Accepted Stubs"); + prof_->GetXaxis()->SetBinLabel(2, "Truncated Stubs"); prof_->GetXaxis()->SetBinLabel(3, "Found TPs"); prof_->GetXaxis()->SetBinLabel(4, "Selected TPs"); + // Efficiencies + hisEffEtaTotal_ = dir.make("HisTPEtaTotal", ";", 48, -2.4, 2.4); + hisEffEta_ = dir.make("HisTPEta", ";", 48, -2.4, 2.4); + effEta_ = dir.make("EffEta", ";", 48, -2.4, 2.4); + const int zTBins = setup_->gpNumBinsZT(); + hisEffZTTotal_ = dir.make("HisTPZTTotal", ";", zTBins, -zTBins / 2, zTBins / 2); + hisEffZT_ = dir.make("HisTPZT", ";", zTBins, -zTBins / 2, zTBins / 2); + effZT_ = dir.make("EffZT", ";", zTBins, -zTBins / 2, zTBins / 2); + const double rangeInv2R = dataFormats_->format(Variable::inv2R, Process::dr).range(); + hisEffInv2R_ = dir.make("HisTPInv2R", ";", 32, -rangeInv2R / 2., rangeInv2R / 2.); + hisEffInv2RTotal_ = dir.make("HisTPInv2RTotal", ";", 32, -rangeInv2R / 2., rangeInv2R / 2.); + effInv2R_ = dir.make("EffInv2R", ";", 32, -rangeInv2R / 2., rangeInv2R / 2.); + hisEffPT_ = dir.make("HisTPPT", ";", 100, 0, 100); + hisEffPTTotal_ = dir.make("HisTPPTTotal", ";", 100, 0, 100); + effPT_ = dir.make("EffPT", ";", 100, 0, 100); // channel occupancy constexpr int maxOcc = 180; const int numChannels = setup_->numSectors(); @@ -108,11 +140,20 @@ namespace trackerTFP { } void AnalyzerGP::analyze(const Event& iEvent, const EventSetup& iSetup) { + auto fill = [this](const TPPtr& tpPtr, TH1F* hisEta, TH1F* hisZT, TH1F* hisInv2R, TH1F* hisPT) { + const double tpPhi0 = tpPtr->phi(); + const double tpCot = sinh(tpPtr->eta()); + const math::XYZPointD& v = tpPtr->vertex(); + const double tpZ0 = v.z() - tpCot * (v.x() * cos(tpPhi0) + v.y() * sin(tpPhi0)); + const double tpZT = tpZ0 + tpCot * setup_->chosenRofZ(); + hisEta->Fill(tpPtr->eta()); + hisZT->Fill(dataFormats_->format(Variable::zT, Process::gp).integer(tpZT)); + hisInv2R->Fill(tpPtr->charge() / tpPtr->pt() * setup_->invPtToDphi()); + hisPT->Fill(tpPtr->pt()); + }; // read in gp products - Handle handleAccepted; - iEvent.getByToken(edGetTokenAccepted_, handleAccepted); - Handle handleLost; - iEvent.getByToken(edGetTokenLost_, handleLost); + Handle handleStubs; + iEvent.getByToken(edGetToken_, handleStubs); // read in MCTruth const StubAssociation* stubAssociation = nullptr; if (useMCTruth_) { @@ -120,16 +161,17 @@ namespace trackerTFP { iEvent.getByToken(edGetTokenAss_, handleAss); stubAssociation = handleAss.product(); prof_->Fill(4, stubAssociation->numTPs()); + for (const auto& p : stubAssociation->getTrackingParticleToTTStubsMap()) + fill(p.first, hisEffEtaTotal_, hisEffZTTotal_, hisEffInv2RTotal_, hisEffPTTotal_); } // analyze gp products and find still reconstrucable TrackingParticles set setTPPtr; for (int region = 0; region < setup_->numRegions(); region++) { int nStubs(0); - int nLost(0); map> mapTPsTTStubs; for (int channel = 0; channel < setup_->numSectors(); channel++) { const int index = region * setup_->numSectors() + channel; - const StreamStub& accepted = handleAccepted->at(index); + const StreamStub& accepted = handleStubs->at(index); hisChannel_->Fill(accepted.size()); profChannel_->Fill(channel, accepted.size()); for (const FrameStub& frame : accepted) { @@ -148,14 +190,14 @@ namespace trackerTFP { it->second.push_back(frame.first); } } - nLost += handleLost->at(index).size(); } for (const auto& p : mapTPsTTStubs) if (setup_->reconstructable(p.second)) setTPPtr.insert(p.first); prof_->Fill(1, nStubs); - prof_->Fill(2, nLost); } + for (const TPPtr& tpPtr : setTPPtr) + fill(tpPtr, hisEffEta_, hisEffZT_, hisEffInv2R_, hisEffPT_); prof_->Fill(3, setTPPtr.size()); nEvents_++; } @@ -163,26 +205,31 @@ namespace trackerTFP { void AnalyzerGP::endJob() { if (nEvents_ == 0) return; + // effi + effEta_->SetPassedHistogram(*hisEffEta_, "f"); + effEta_->SetTotalHistogram(*hisEffEtaTotal_, "f"); + effZT_->SetPassedHistogram(*hisEffZT_, "f"); + effZT_->SetTotalHistogram(*hisEffZTTotal_, "f"); + effInv2R_->SetPassedHistogram(*hisEffInv2R_, "f"); + effInv2R_->SetTotalHistogram(*hisEffInv2RTotal_, "f"); + effPT_->SetPassedHistogram(*hisEffPT_, "f"); + effPT_->SetTotalHistogram(*hisEffPTTotal_, "f"); // printout GP summary const double numStubs = prof_->GetBinContent(1); - const double numStubsLost = prof_->GetBinContent(2); const double errStubs = prof_->GetBinError(1); - const double errStubsLost = prof_->GetBinError(2); const double numTPs = prof_->GetBinContent(3); const double totalTPs = prof_->GetBinContent(4); const double eff = numTPs / totalTPs; const double errEff = sqrt(eff * (1. - eff) / totalTPs / nEvents_); - const vector nums = {numStubs, numStubsLost}; - const vector errs = {errStubs, errStubsLost}; + const vector nums = {numStubs}; + const vector errs = {errStubs}; const int wNums = ceil(log10(*max_element(nums.begin(), nums.end()))) + 5; const int wErrs = ceil(log10(*max_element(errs.begin(), errs.end()))) + 5; log_ << " GP SUMMARY " << endl; log_ << "number of stubs per TFP = " << setw(wNums) << numStubs << " +- " << setw(wErrs) << errStubs << endl; - log_ << "number of lost stubs per TFP = " << setw(wNums) << numStubsLost << " +- " << setw(wErrs) << errStubsLost - << endl; log_ << " max tracking efficiency = " << setw(wNums) << eff << " +- " << setw(wErrs) << errEff << endl; log_ << "============================================================="; - LogPrint("L1Trigger/TrackerTFP") << log_.str(); + LogPrint(moduleDescription().moduleName()) << log_.str(); } } // namespace trackerTFP diff --git a/L1Trigger/TrackerTFP/test/AnalyzerHT.cc b/L1Trigger/TrackerTFP/test/AnalyzerHT.cc index 62b14e8f5aab8..1f337045f973b 100644 --- a/L1Trigger/TrackerTFP/test/AnalyzerHT.cc +++ b/L1Trigger/TrackerTFP/test/AnalyzerHT.cc @@ -18,6 +18,7 @@ #include #include +#include #include #include @@ -48,14 +49,12 @@ namespace trackerTFP { private: // - void formTracks(const StreamStub& stream, vector>& tracks, int qOverPt) const; + void formTracks(const StreamStub& stream, vector>& tracks) const; // void associate(const vector>& tracks, const StubAssociation* ass, set& tps, int& sum) const; // ED input token of stubs - EDGetTokenT edGetTokenAccepted_; - // ED input token of lost stubs - EDGetTokenT edGetTokenLost_; + EDGetTokenT edGetToken_; // ED input token of TTStubRef to TPPtr association for tracking efficiency EDGetTokenT edGetTokenSelection_; // ED input token of TTStubRef to recontructable TPPtr association @@ -78,6 +77,21 @@ namespace trackerTFP { TProfile* prof_; TProfile* profChannel_; TH1F* hisChannel_; + TH1F* hisLayers_; + TH1F* hisNumLayers_; + TProfile* profNumLayers_; + TH1F* hisEffEta_; + TH1F* hisEffEtaTotal_; + TEfficiency* effEta_; + TH1F* hisEffZT_; + TH1F* hisEffZTTotal_; + TEfficiency* effZT_; + TH1F* hisEffInv2R_; + TH1F* hisEffInv2RTotal_; + TEfficiency* effInv2R_; + TH1F* hisEffPT_; + TH1F* hisEffPTTotal_; + TEfficiency* effPT_; // printout stringstream log_; @@ -86,11 +100,9 @@ namespace trackerTFP { AnalyzerHT::AnalyzerHT(const ParameterSet& iConfig) : useMCTruth_(iConfig.getParameter("UseMCTruth")) { usesResource("TFileService"); // book in- and output ED products - const string& label = iConfig.getParameter("LabelHT"); - const string& branchAccepted = iConfig.getParameter("BranchAcceptedStubs"); - const string& branchLost = iConfig.getParameter("BranchLostStubs"); - edGetTokenAccepted_ = consumes(InputTag(label, branchAccepted)); - edGetTokenLost_ = consumes(InputTag(label, branchLost)); + const string& label = iConfig.getParameter("OutputLabelHT"); + const string& branch = iConfig.getParameter("BranchStubs"); + edGetToken_ = consumes(InputTag(label, branch)); if (useMCTruth_) { const auto& inputTagSelecttion = iConfig.getParameter("InputTagSelection"); const auto& inputTagReconstructable = iConfig.getParameter("InputTagReconstructable"); @@ -117,26 +129,56 @@ namespace trackerTFP { prof_ = dir.make("Counts", ";", 9, 0.5, 9.5); prof_->GetXaxis()->SetBinLabel(1, "Stubs"); prof_->GetXaxis()->SetBinLabel(2, "Tracks"); - prof_->GetXaxis()->SetBinLabel(3, "Lost Tracks"); + prof_->GetXaxis()->SetBinLabel(3, "Truncated Tracks"); prof_->GetXaxis()->SetBinLabel(4, "Matched Tracks"); prof_->GetXaxis()->SetBinLabel(5, "All Tracks"); prof_->GetXaxis()->SetBinLabel(6, "Found TPs"); prof_->GetXaxis()->SetBinLabel(7, "Found selected TPs"); - prof_->GetXaxis()->SetBinLabel(8, "Lost TPs"); + prof_->GetXaxis()->SetBinLabel(8, "Truncated TPs"); prof_->GetXaxis()->SetBinLabel(9, "All TPs"); + // Efficiencies + hisEffEtaTotal_ = dir.make("HisTPEtaTotal", ";", 48, -2.4, 2.4); + hisEffEta_ = dir.make("HisTPEta", ";", 48, -2.4, 2.4); + effEta_ = dir.make("EffEta", ";", 48, -2.4, 2.4); + const double rangeZT = dataFormats_->format(Variable::zT, Process::dr).range(); + const int zTBins = setup_->gpNumBinsZT(); + hisEffZTTotal_ = dir.make("HisTPZTTotal", ";", zTBins, -rangeZT / 2, rangeZT / 2); + hisEffZT_ = dir.make("HisTPZT", ";", zTBins, -rangeZT / 2, rangeZT / 2); + effZT_ = dir.make("EffZT", ";", zTBins, -rangeZT / 2, rangeZT / 2); + const double rangeInv2R = dataFormats_->format(Variable::inv2R, Process::dr).range(); + const int inv2RBins = (setup_->htNumBinsInv2R() + 2) * 2; + hisEffInv2R_ = dir.make("HisTPInv2R", ";", inv2RBins, -rangeInv2R / 2., rangeInv2R / 2.); + hisEffInv2RTotal_ = dir.make("HisTPInv2RTotal", ";", inv2RBins, -rangeInv2R / 2., rangeInv2R / 2.); + effInv2R_ = dir.make("EffInv2R", ";", inv2RBins, -rangeInv2R / 2., rangeInv2R / 2.); + hisEffPT_ = dir.make("HisTPPT", ";", 100, 0, 100); + hisEffPTTotal_ = dir.make("HisTPPTTotal", ";", 100, 0, 100); + effPT_ = dir.make("EffPT", ";", 100, 0, 100); // binInv2R occupancy constexpr int maxOcc = 180; const int numChannel = dataFormats_->numChannel(Process::ht); hisChannel_ = dir.make("His binInv2R Occupancy", ";", maxOcc, -.5, maxOcc - .5); profChannel_ = dir.make("Prof binInv2R Occupancy", ";", numChannel, -.5, numChannel - .5); + // layers + hisLayers_ = dir.make("HisLayers", ";", 8, 0, 8); + hisNumLayers_ = dir.make("HisNumLayers", ";", 9, 0, 9); + profNumLayers_ = dir.make("Prof NumLayers", ";", 32, 0, 2.4); } void AnalyzerHT::analyze(const Event& iEvent, const EventSetup& iSetup) { + auto fill = [this](const TPPtr& tpPtr, TH1F* hisEta, TH1F* hisZT, TH1F* hisInv2R, TH1F* hisPT) { + const double tpPhi0 = tpPtr->phi(); + const double tpCot = sinh(tpPtr->eta()); + const math::XYZPointD& v = tpPtr->vertex(); + const double tpZ0 = v.z() - tpCot * (v.x() * cos(tpPhi0) + v.y() * sin(tpPhi0)); + const double tpZT = tpZ0 + tpCot * setup_->chosenRofZ(); + hisEta->Fill(tpPtr->eta()); + hisZT->Fill(tpZT); + hisInv2R->Fill(tpPtr->charge() / tpPtr->pt() * setup_->invPtToDphi()); + hisPT->Fill(tpPtr->pt()); + }; // read in ht products - Handle handleAccepted; - iEvent.getByToken(edGetTokenAccepted_, handleAccepted); - Handle handleLost; - iEvent.getByToken(edGetTokenLost_, handleLost); + Handle handleStubs; + iEvent.getByToken(edGetToken_, handleStubs); // read in MCTruth const StubAssociation* selection = nullptr; const StubAssociation* reconstructable = nullptr; @@ -148,110 +190,115 @@ namespace trackerTFP { Handle handleReconstructable; iEvent.getByToken(edGetTokenReconstructable_, handleReconstructable); reconstructable = handleReconstructable.product(); + for (const auto& p : selection->getTrackingParticleToTTStubsMap()) + fill(p.first, hisEffEtaTotal_, hisEffZTTotal_, hisEffInv2RTotal_, hisEffPTTotal_); } // analyze ht products and associate found tracks with reconstrucable TrackingParticles set tpPtrs; set tpPtrsSelection; - set tpPtrsLost; int allMatched(0); int allTracks(0); for (int region = 0; region < setup_->numRegions(); region++) { int nStubs(0); int nTracks(0); - int nLost(0); for (int channel = 0; channel < dataFormats_->numChannel(Process::ht); channel++) { - const int inv2R = dataFormats_->format(Variable::inv2R, Process::ht).toSigned(channel); const int index = region * dataFormats_->numChannel(Process::ht) + channel; - const StreamStub& accepted = handleAccepted->at(index); + const StreamStub& accepted = handleStubs->at(index); hisChannel_->Fill(accepted.size()); profChannel_->Fill(channel, accepted.size()); nStubs += accepted.size(); vector> tracks; - vector> lost; - formTracks(accepted, tracks, inv2R); - formTracks(handleLost->at(index), lost, inv2R); + formTracks(accepted, tracks); nTracks += tracks.size(); allTracks += tracks.size(); - nLost += lost.size(); if (!useMCTruth_) continue; int tmp(0); associate(tracks, selection, tpPtrsSelection, tmp); - associate(lost, selection, tpPtrsLost, tmp); associate(tracks, reconstructable, tpPtrs, allMatched); } prof_->Fill(1, nStubs); prof_->Fill(2, nTracks); - prof_->Fill(3, nLost); } - vector recovered; - recovered.reserve(tpPtrsLost.size()); - set_intersection(tpPtrsLost.begin(), tpPtrsLost.end(), tpPtrs.begin(), tpPtrs.end(), back_inserter(recovered)); - for (const TPPtr& tpPtr : recovered) - tpPtrsLost.erase(tpPtr); + for (const TPPtr& tpPtr : tpPtrsSelection) + fill(tpPtr, hisEffEta_, hisEffZT_, hisEffInv2R_, hisEffPT_); prof_->Fill(4, allMatched); prof_->Fill(5, allTracks); prof_->Fill(6, tpPtrs.size()); prof_->Fill(7, tpPtrsSelection.size()); - prof_->Fill(8, tpPtrsLost.size()); nEvents_++; } void AnalyzerHT::endJob() { if (nEvents_ == 0) return; + // effi + effEta_->SetPassedHistogram(*hisEffEta_, "f"); + effEta_->SetTotalHistogram(*hisEffEtaTotal_, "f"); + effZT_->SetPassedHistogram(*hisEffZT_, "f"); + effZT_->SetTotalHistogram(*hisEffZTTotal_, "f"); + effInv2R_->SetPassedHistogram(*hisEffInv2R_, "f"); + effInv2R_->SetTotalHistogram(*hisEffInv2RTotal_, "f"); + effPT_->SetPassedHistogram(*hisEffPT_, "f"); + effPT_->SetTotalHistogram(*hisEffPTTotal_, "f"); // printout HT summary const double totalTPs = prof_->GetBinContent(9); const double numStubs = prof_->GetBinContent(1); const double numTracks = prof_->GetBinContent(2); - const double numTracksLost = prof_->GetBinContent(3); const double totalTracks = prof_->GetBinContent(5); const double numTracksMatched = prof_->GetBinContent(4); const double numTPsAll = prof_->GetBinContent(6); const double numTPsEff = prof_->GetBinContent(7); - const double numTPsLost = prof_->GetBinContent(8); const double errStubs = prof_->GetBinError(1); const double errTracks = prof_->GetBinError(2); - const double errTracksLost = prof_->GetBinError(3); const double fracFake = (totalTracks - numTracksMatched) / totalTracks; const double fracDup = (numTracksMatched - numTPsAll) / totalTracks; const double eff = numTPsEff / totalTPs; const double errEff = sqrt(eff * (1. - eff) / totalTPs / nEvents_); - const double effLoss = numTPsLost / totalTPs; - const double errEffLoss = sqrt(effLoss * (1. - effLoss) / totalTPs / nEvents_); - const vector nums = {numStubs, numTracks, numTracksLost}; - const vector errs = {errStubs, errTracks, errTracksLost}; + const vector nums = {numStubs, numTracks}; + const vector errs = {errStubs, errTracks}; const int wNums = ceil(log10(*max_element(nums.begin(), nums.end()))) + 5; const int wErrs = ceil(log10(*max_element(errs.begin(), errs.end()))) + 5; log_ << " HT SUMMARY " << endl; log_ << "number of stubs per TFP = " << setw(wNums) << numStubs << " +- " << setw(wErrs) << errStubs << endl; log_ << "number of tracks per TFP = " << setw(wNums) << numTracks << " +- " << setw(wErrs) << errTracks << endl; - log_ << "number of lost tracks per TFP = " << setw(wNums) << numTracksLost << " +- " << setw(wErrs) << errTracksLost - << endl; log_ << " max tracking efficiency = " << setw(wNums) << eff << " +- " << setw(wErrs) << errEff << endl; - log_ << " lost tracking efficiency = " << setw(wNums) << effLoss << " +- " << setw(wErrs) << errEffLoss << endl; log_ << " fake rate = " << setw(wNums) << fracFake << endl; log_ << " duplicate rate = " << setw(wNums) << fracDup << endl; log_ << "============================================================="; - LogPrint("L1Trigger/TrackerTFP") << log_.str(); + LogPrint(moduleDescription().moduleName()) << log_.str(); } // - void AnalyzerHT::formTracks(const StreamStub& stream, vector>& tracks, int inv2R) const { + void AnalyzerHT::formTracks(const StreamStub& stream, vector>& tracks) const { + static const DataFormat& layer = dataFormats_->format(Variable::layer, Process::ctb); + auto toTrkId = [this](const StubHT& stub) { + static const DataFormat& phiT = dataFormats_->format(Variable::phiT, Process::ht); + static const DataFormat& zT = dataFormats_->format(Variable::zT, Process::ht); + return (phiT.ttBV(stub.phiT()) + zT.ttBV(stub.zT())).val(); + }; vector stubs; stubs.reserve(stream.size()); for (const FrameStub& frame : stream) - stubs.emplace_back(frame, dataFormats_, inv2R); + stubs.emplace_back(frame, dataFormats_); for (auto it = stubs.begin(); it != stubs.end();) { const auto start = it; - const int id = it->trackId(); - auto different = [id](const StubHT& stub) { return id != stub.trackId(); }; + const int id = toTrkId(*it); + auto different = [id, toTrkId](const StubHT& stub) { return id != toTrkId(stub); }; it = find_if(it, stubs.end(), different); vector ttStubRefs; ttStubRefs.reserve(distance(start, it)); - transform(start, it, back_inserter(ttStubRefs), [](const StubHT& stub) { return stub.ttStubRef(); }); + transform(start, it, back_inserter(ttStubRefs), [](const StubHT& stub) { return stub.frame().first; }); tracks.push_back(ttStubRefs); + TTBV hitPattern(0, setup_->numLayers()); + for (auto iter = start; iter != it; iter++) + hitPattern.set(iter->layer().val(layer.width())); + const double cot = dataFormats_->format(Variable::zT, Process::ht).floating(start->zT()) / setup_->chosenRofZ(); + hisNumLayers_->Fill(hitPattern.count()); + profNumLayers_->Fill(abs(sinh(cot)), hitPattern.count()); + for (int layer : hitPattern.ids()) + hisLayers_->Fill(layer); } } diff --git a/L1Trigger/TrackerTFP/test/AnalyzerKF.cc b/L1Trigger/TrackerTFP/test/AnalyzerKF.cc index 3f0b215080aca..06ed2f943123b 100644 --- a/L1Trigger/TrackerTFP/test/AnalyzerKF.cc +++ b/L1Trigger/TrackerTFP/test/AnalyzerKF.cc @@ -15,7 +15,6 @@ #include "SimTracker/TrackTriggerAssociation/interface/StubAssociation.h" #include "L1Trigger/TrackTrigger/interface/Setup.h" #include "L1Trigger/TrackerTFP/interface/DataFormats.h" -#include "L1Trigger/TrackerTFP/interface/LayerEncoding.h" #include "L1Trigger/TrackerTFP/interface/KalmanFilterFormats.h" #include @@ -28,6 +27,7 @@ #include #include #include +#include using namespace std; using namespace edm; @@ -51,25 +51,25 @@ namespace trackerTFP { private: // - void associate(const TTTracks& ttTracks, + void associate(const vector& tracks, + const vector>& stubs, + int region, const StubAssociation* ass, set& tps, int& sum, const vector& his, - TProfile* prof) const; - + const vector& prof, + bool perfect = true) const; // ED input token of accepted Tracks - EDGetTokenT edGetTokenAcceptedStubs_; + EDGetTokenT edGetTokenStubs_; // ED input token of accepted Stubs - EDGetTokenT edGetTokenAcceptedTracks_; - // ED input token of lost Stubs - EDGetTokenT edGetTokenLostStubs_; - // ED input token of lost Tracks - EDGetTokenT edGetTokenLostTracks_; + EDGetTokenT edGetTokenTracks_; // ED input token for number of accepted States - EDGetTokenT edGetTokenNumAcceptedStates_; + EDGetTokenT edGetTokenNumStatesAccepted_; // ED input token for number of lost States - EDGetTokenT edGetTokenNumLostStates_; + EDGetTokenT edGetTokenNumStatesTruncated_; + // ED input token for r-phi and r-z plane chi2s + EDGetTokenT>> edGetTokenChi2s_; // ED input token of TTStubRef to TPPtr association for tracking efficiency EDGetTokenT edGetTokenSelection_; // ED input token of TTStubRef to recontructable TPPtr association @@ -78,14 +78,10 @@ namespace trackerTFP { ESGetToken esGetTokenSetup_; // DataFormats token ESGetToken esGetTokenDataFormats_; - // LayerEncoding token - ESGetToken esGetTokenLayerEncoding_; // stores, calculates and provides run-time constants const Setup* setup_ = nullptr; // const DataFormats* dataFormats_ = nullptr; - // - const LayerEncoding* layerEncoding_ = nullptr; // enables analyze of TPs bool useMCTruth_; // @@ -97,37 +93,44 @@ namespace trackerTFP { TProfile* profChannel_; TH1F* hisChannel_; vector hisRes_; - TProfile* profResZ0_; + vector profRes_; TH1F* hisEffEta_; TH1F* hisEffEtaTotal_; TEfficiency* effEta_; + TH1F* hisEffZT_; + TH1F* hisEffZTTotal_; + TEfficiency* effZT_; TH1F* hisEffInv2R_; TH1F* hisEffInv2RTotal_; TEfficiency* effInv2R_; - TH1F* hisChi2_; - TH1F* hisPhi_; + TH1F* hisEffPT_; + TH1F* hisEffPTTotal_; + TEfficiency* effPT_; + TH1F* hisChi20s_; + TH1F* hisChi21s_; + TH1F* hisChi2s_; + TH1F* hisTracks_; + TH1F* hisLayers_; + TH1F* hisNumLayers_; + TProfile* profNumLayers_; // printout stringstream log_; }; AnalyzerKF::AnalyzerKF(const ParameterSet& iConfig) - : useMCTruth_(iConfig.getParameter("UseMCTruth")), hisRes_(4) { + : useMCTruth_(iConfig.getParameter("UseMCTruth")), nEvents_(0), hisRes_(4), profRes_(4) { usesResource("TFileService"); // book in- and output ED products - const string& label = iConfig.getParameter("LabelKF"); - const string& branchAcceptedStubs = iConfig.getParameter("BranchAcceptedStubs"); - const string& branchAcceptedTracks = iConfig.getParameter("BranchAcceptedTracks"); - const string& branchLostStubs = iConfig.getParameter("BranchLostStubs"); - const string& branchLostTracks = iConfig.getParameter("BranchLostTracks"); - edGetTokenAcceptedStubs_ = consumes(InputTag(label, branchAcceptedStubs)); - edGetTokenAcceptedTracks_ = consumes(InputTag(label, branchAcceptedTracks)); - edGetTokenLostStubs_ = consumes(InputTag(label, branchLostStubs)); - edGetTokenLostTracks_ = consumes(InputTag(label, branchLostTracks)); - edGetTokenNumAcceptedStates_ = consumes(InputTag(label, branchAcceptedTracks)); - ; - edGetTokenNumLostStates_ = consumes(InputTag(label, branchLostTracks)); - ; + const string& label = iConfig.getParameter("OutputLabelKF"); + const string& branchStubs = iConfig.getParameter("BranchStubs"); + const string& branchTracks = iConfig.getParameter("BranchTracks"); + const string& branchTruncated = iConfig.getParameter("BranchTruncated"); + edGetTokenStubs_ = consumes(InputTag(label, branchStubs)); + edGetTokenTracks_ = consumes(InputTag(label, branchTracks)); + edGetTokenNumStatesAccepted_ = consumes(InputTag(label, branchTracks)); + edGetTokenNumStatesTruncated_ = consumes(InputTag(label, branchTruncated)); + edGetTokenChi2s_ = consumes>>(InputTag(label, branchTracks)); if (useMCTruth_) { const auto& inputTagSelecttion = iConfig.getParameter("InputTagSelection"); const auto& inputTagReconstructable = iConfig.getParameter("InputTagReconstructable"); @@ -137,7 +140,6 @@ namespace trackerTFP { // book ES products esGetTokenSetup_ = esConsumes(); esGetTokenDataFormats_ = esConsumes(); - esGetTokenLayerEncoding_ = esConsumes(); // log config log_.setf(ios::fixed, ios::floatfield); log_.precision(4); @@ -147,70 +149,87 @@ namespace trackerTFP { // helper class to store configurations setup_ = &iSetup.getData(esGetTokenSetup_); dataFormats_ = &iSetup.getData(esGetTokenDataFormats_); - layerEncoding_ = &iSetup.getData(esGetTokenLayerEncoding_); // book histograms Service fs; TFileDirectory dir; dir = fs->mkdir("KF"); - prof_ = dir.make("Counts", ";", 11, 0.5, 11.5); + prof_ = dir.make("Counts", ";", 12, 0.5, 12.5); prof_->GetXaxis()->SetBinLabel(1, "Stubs"); prof_->GetXaxis()->SetBinLabel(2, "Tracks"); - prof_->GetXaxis()->SetBinLabel(3, "Lost Tracks"); prof_->GetXaxis()->SetBinLabel(4, "Matched Tracks"); prof_->GetXaxis()->SetBinLabel(5, "All Tracks"); prof_->GetXaxis()->SetBinLabel(6, "Found TPs"); prof_->GetXaxis()->SetBinLabel(7, "Found selected TPs"); - prof_->GetXaxis()->SetBinLabel(8, "Lost TPs"); prof_->GetXaxis()->SetBinLabel(9, "All TPs"); prof_->GetXaxis()->SetBinLabel(10, "states"); - prof_->GetXaxis()->SetBinLabel(11, "lost states"); + prof_->GetXaxis()->SetBinLabel(12, "max tp"); // channel occupancy constexpr int maxOcc = 180; const int numChannels = dataFormats_->numChannel(Process::kf); hisChannel_ = dir.make("His Channel Occupancy", ";", maxOcc, -.5, maxOcc - .5); profChannel_ = dir.make("Prof Channel Occupancy", ";", numChannels, -.5, numChannels - .5); // resoultions - static const vector names = {"phiT", "inv2R", "zT", "cot"}; - static const vector ranges = {.01, .1, 5, .1}; + static const vector names = {"phi0", "inv2R", "z0", "cot"}; + static const vector ranges = {.01, .004, 20., .4}; for (int i = 0; i < 4; i++) { const double range = ranges[i]; hisRes_[i] = dir.make(("HisRes" + names[i]).c_str(), ";", 100, -range, range); + profRes_[i] = dir.make(("ProfRes" + names[i]).c_str(), ";", 32, 0, 2.4); } - profResZ0_ = dir.make("ProfResZ0", ";", 32, 0, 2.5); // Efficiencies - hisEffEtaTotal_ = dir.make("HisTPEtaTotal", ";", 128, -2.5, 2.5); - hisEffEta_ = dir.make("HisTPEta", ";", 128, -2.5, 2.5); - effEta_ = dir.make("EffEta", ";", 128, -2.5, 2.5); + hisEffEtaTotal_ = dir.make("HisTPEtaTotal", ";", 48, -2.4, 2.4); + hisEffEta_ = dir.make("HisTPEta", ";", 48, -2.4, 2.4); + effEta_ = dir.make("EffEta", ";", 48, -2.4, 2.4); + const int zTBins = setup_->gpNumBinsZT(); + hisEffZTTotal_ = dir.make("HisTPZTTotal", ";", zTBins, -zTBins / 2, zTBins / 2); + hisEffZT_ = dir.make("HisTPZT", ";", zTBins, -zTBins / 2, zTBins / 2); + effZT_ = dir.make("EffZT", ";", zTBins, -zTBins / 2, zTBins / 2); const double rangeInv2R = dataFormats_->format(Variable::inv2R, Process::dr).range(); hisEffInv2R_ = dir.make("HisTPInv2R", ";", 32, -rangeInv2R / 2., rangeInv2R / 2.); hisEffInv2RTotal_ = dir.make("HisTPInv2RTotal", ";", 32, -rangeInv2R / 2., rangeInv2R / 2.); effInv2R_ = dir.make("EffInv2R", ";", 32, -rangeInv2R / 2., rangeInv2R / 2.); - // chi2 - hisChi2_ = dir.make("HisChi2", ";", 100, -.5, 99.5); - const double rangePhi = dataFormats_->format(Variable::phi0, Process::dr).range(); - hisPhi_ = dir.make("HisPhi", ";", 100, -rangePhi, rangePhi); + hisEffPT_ = dir.make("HisTPPT", ";", 100, 0, 100); + hisEffPTTotal_ = dir.make("HisTPPTTotal", ";", 100, 0, 100); + effPT_ = dir.make("EffPT", ";", 100, 0, 100); + // chi2s + hisChi20s_ = dir.make("HisChi20", ";", 128, 0., 10); + hisChi21s_ = dir.make("HisChi21", ";", 128, 0., 10); + hisChi2s_ = dir.make("HisChi2", ";", 128, 0., 10); + // tracks + hisTracks_ = dir.make("HisTracks", ";", 40, 0., 400); + // layers + hisLayers_ = dir.make("HisLayers", ";", 8, 0, 8); + hisNumLayers_ = dir.make("HisNumLayers", ";", 9, 0, 9); + profNumLayers_ = dir.make("Prof NumLayers", ";", 32, 0, 2.4); } void AnalyzerKF::analyze(const Event& iEvent, const EventSetup& iSetup) { - auto fill = [this](const TPPtr& tpPtr, TH1F* hisEta, TH1F* hisInv2R) { + static const int numChannel = dataFormats_->numChannel(Process::kf); + static const int numLayers = setup_->numLayers(); + auto fill = [this](const TPPtr& tpPtr, TH1F* hisEta, TH1F* hisZT, TH1F* hisInv2R, TH1F* hisPT) { + const double tpPhi0 = tpPtr->phi(); + const double tpCot = sinh(tpPtr->eta()); + const math::XYZPointD& v = tpPtr->vertex(); + const double tpZ0 = v.z() - tpCot * (v.x() * cos(tpPhi0) + v.y() * sin(tpPhi0)); + const double tpZT = tpZ0 + tpCot * setup_->chosenRofZ(); hisEta->Fill(tpPtr->eta()); + hisZT->Fill(dataFormats_->format(Variable::zT, Process::gp).integer(tpZT)); hisInv2R->Fill(tpPtr->charge() / tpPtr->pt() * setup_->invPtToDphi()); + hisPT->Fill(tpPtr->pt()); }; // read in kf products - Handle handleAcceptedStubs; - iEvent.getByToken(edGetTokenAcceptedStubs_, handleAcceptedStubs); - const StreamsStub& acceptedStubs = *handleAcceptedStubs; - Handle handleAcceptedTracks; - iEvent.getByToken(edGetTokenAcceptedTracks_, handleAcceptedTracks); - Handle handleLostStubs; - iEvent.getByToken(edGetTokenLostStubs_, handleLostStubs); - const StreamsStub& lostStubs = *handleLostStubs; - Handle handleLostTracks; - iEvent.getByToken(edGetTokenLostTracks_, handleLostTracks); - Handle handleNumAcceptedStates; - iEvent.getByToken(edGetTokenNumAcceptedStates_, handleNumAcceptedStates); - Handle handleNumLostStates; - iEvent.getByToken(edGetTokenNumLostStates_, handleNumLostStates); + Handle handleStubs; + iEvent.getByToken(edGetTokenStubs_, handleStubs); + const StreamsStub& allStubs = *handleStubs; + Handle handleTracks; + iEvent.getByToken(edGetTokenTracks_, handleTracks); + const StreamsTrack& allTracks = *handleTracks; + Handle handleNumStatesAccepted; + iEvent.getByToken(edGetTokenNumStatesAccepted_, handleNumStatesAccepted); + Handle handleNumStatesTruncated; + iEvent.getByToken(edGetTokenNumStatesTruncated_, handleNumStatesTruncated); + Handle>> handleChi2s; + iEvent.getByToken>>(edGetTokenChi2s_, handleChi2s); // read in MCTruth const StubAssociation* selection = nullptr; const StubAssociation* reconstructable = nullptr; @@ -223,82 +242,82 @@ namespace trackerTFP { iEvent.getByToken(edGetTokenReconstructable_, handleReconstructable); reconstructable = handleReconstructable.product(); for (const auto& p : selection->getTrackingParticleToTTStubsMap()) - fill(p.first, hisEffEtaTotal_, hisEffInv2RTotal_); + fill(p.first, hisEffEtaTotal_, hisEffZTTotal_, hisEffInv2RTotal_, hisEffPTTotal_); + } + // chi2s + for (const pair& chi2s : *handleChi2s) { + hisChi20s_->Fill(chi2s.first * 2.); + hisChi21s_->Fill(chi2s.second * 2.); + hisChi2s_->Fill(chi2s.first + chi2s.second); } // analyze kf products and associate found tracks with reconstrucable TrackingParticles set tpPtrs; set tpPtrsSelection; - set tpPtrsLost; - int allMatched(0); - int allTracks(0); - auto consume = [this](const StreamTrack& tracks, const StreamsStub& streams, int channel, TTTracks& ttTracks) { - const int offset = channel * setup_->numLayers(); - int pos(0); - for (const FrameTrack& frameTrack : tracks) { + set tpPtrsMax; + int numMatched(0); + int numTracks(0); + for (int region = 0; region < setup_->numRegions(); region++) { + int nRegionStubs(0); + int nRegionTracks(0); + for (int channel = 0; channel < numChannel; channel++) { + const int index = region * numChannel + channel; + const int offset = index * numLayers; + const StreamTrack& channelTracks = allTracks[index]; + hisChannel_->Fill(channelTracks.size()); + profChannel_->Fill(channel, channelTracks.size()); + vector tracks; vector stubs; - stubs.reserve(setup_->numLayers()); - for (int layer = 0; layer < setup_->numLayers(); layer++) { - const FrameStub& frameStub = streams[offset + layer][pos]; - if (frameStub.first.isNonnull()) - stubs.emplace_back(frameStub, dataFormats_, layer); + vector> tracksStubs(channelTracks.size(), vector(numLayers, nullptr)); + tracks.reserve(channelTracks.size()); + stubs.reserve(channelTracks.size() * numLayers); + for (int frame = 0; frame < (int)channelTracks.size(); frame++) { + tracks.emplace_back(channelTracks[frame], dataFormats_); + const double cot = tracks.back().zT() / setup_->chosenRofZ(); + int nLs(0); + for (int layer = 0; layer < numLayers; layer++) { + const FrameStub& fs = allStubs[offset + layer][frame]; + if (fs.first.isNull()) + continue; + stubs.emplace_back(fs, dataFormats_); + tracksStubs[frame][layer] = &stubs.back(); + hisLayers_->Fill(layer); + nLs++; + } + hisNumLayers_->Fill(nLs); + profNumLayers_->Fill(abs(sinh(cot)), nLs); } - TrackKF track(frameTrack, dataFormats_); - ttTracks.emplace_back(track.ttTrack(stubs)); - pos++; - } - }; - for (int region = 0; region < setup_->numRegions(); region++) { - int nStubsRegion(0); - int nTracksRegion(0); - int nLostRegion(0); - for (int channel = 0; channel < dataFormats_->numChannel(Process::kf); channel++) { - const int index = region * dataFormats_->numChannel(Process::kf) + channel; - const StreamTrack& accepted = handleAcceptedTracks->at(index); - const StreamTrack& lost = handleLostTracks->at(index); - hisChannel_->Fill(accepted.size()); - profChannel_->Fill(channel, accepted.size()); - TTTracks tracks; - const int nTracks = accumulate(accepted.begin(), accepted.end(), 0, [](int sum, const FrameTrack& frame) { - return sum + (frame.first.isNonnull() ? 1 : 0); - }); - nTracksRegion += nTracks; - tracks.reserve(nTracks); - consume(accepted, acceptedStubs, index, tracks); - for (const TTTrack& ttTrack : tracks) - hisPhi_->Fill(ttTrack.momentum().phi()); - nStubsRegion += accumulate(tracks.begin(), tracks.end(), 0, [](int sum, const auto& ttTrack) { - return sum + (int)ttTrack.getStubRefs().size(); - }); - TTTracks tracksLost; - const int nLost = accumulate(lost.begin(), lost.end(), 0, [](int sum, const FrameTrack& frame) { - return sum + (frame.first.isNonnull() ? 1 : 0); - }); - nLostRegion += nLost; - tracksLost.reserve(nLost); - consume(lost, lostStubs, index, tracksLost); - allTracks += nTracks; + nRegionStubs += stubs.size(); + nRegionTracks += tracks.size(); if (!useMCTruth_) continue; int tmp(0); - associate(tracks, selection, tpPtrsSelection, tmp, hisRes_, profResZ0_); - associate(tracksLost, selection, tpPtrsLost, tmp, vector(), nullptr); - associate(tracks, reconstructable, tpPtrs, allMatched, vector(), nullptr); + associate(tracks, tracksStubs, region, selection, tpPtrsSelection, tmp, hisRes_, profRes_); + associate(tracks, + tracksStubs, + region, + reconstructable, + tpPtrs, + numMatched, + vector(), + vector(), + false); + associate(tracks, tracksStubs, region, selection, tpPtrsMax, tmp, vector(), vector(), false); + //cout << "KF " << tpPtrsSelection.size() << " " << tpPtrsMax.size() << endl; } - prof_->Fill(1, nStubsRegion); - prof_->Fill(2, nTracksRegion); - prof_->Fill(3, nLostRegion); + numTracks += nRegionTracks; + prof_->Fill(1, nRegionStubs); + prof_->Fill(2, nRegionTracks); } for (const TPPtr& tpPtr : tpPtrsSelection) - fill(tpPtr, hisEffEta_, hisEffInv2R_); - deque tpPtrsRealLost; - set_difference(tpPtrsLost.begin(), tpPtrsLost.end(), tpPtrs.begin(), tpPtrs.end(), back_inserter(tpPtrsRealLost)); - prof_->Fill(4, allMatched); - prof_->Fill(5, allTracks); + fill(tpPtr, hisEffEta_, hisEffZT_, hisEffInv2R_, hisEffPT_); + prof_->Fill(4, numMatched); + prof_->Fill(5, numTracks); prof_->Fill(6, tpPtrs.size()); prof_->Fill(7, tpPtrsSelection.size()); - prof_->Fill(8, tpPtrsRealLost.size()); - prof_->Fill(10, *handleNumAcceptedStates); - prof_->Fill(11, *handleNumLostStates); + prof_->Fill(10, *handleNumStatesAccepted); + prof_->Fill(11, *handleNumStatesTruncated); + prof_->Fill(12, tpPtrsMax.size()); + hisTracks_->Fill(numTracks); nEvents_++; } @@ -308,79 +327,103 @@ namespace trackerTFP { // effi effEta_->SetPassedHistogram(*hisEffEta_, "f"); effEta_->SetTotalHistogram(*hisEffEtaTotal_, "f"); + effZT_->SetPassedHistogram(*hisEffZT_, "f"); + effZT_->SetTotalHistogram(*hisEffZTTotal_, "f"); effInv2R_->SetPassedHistogram(*hisEffInv2R_, "f"); effInv2R_->SetTotalHistogram(*hisEffInv2RTotal_, "f"); + effPT_->SetPassedHistogram(*hisEffPT_, "f"); + effPT_->SetTotalHistogram(*hisEffPTTotal_, "f"); // printout SF summary const double totalTPs = prof_->GetBinContent(9); const double numStubs = prof_->GetBinContent(1); const double numTracks = prof_->GetBinContent(2); - const double numTracksLost = prof_->GetBinContent(3); const double totalTracks = prof_->GetBinContent(5); const double numTracksMatched = prof_->GetBinContent(4); const double numTPsAll = prof_->GetBinContent(6); const double numTPsEff = prof_->GetBinContent(7); - const double numTPsLost = prof_->GetBinContent(8); + const double numTPsEffMax = prof_->GetBinContent(12); const double errStubs = prof_->GetBinError(1); const double errTracks = prof_->GetBinError(2); - const double errTracksLost = prof_->GetBinError(3); const double fracFake = (totalTracks - numTracksMatched) / totalTracks; const double fracDup = (numTracksMatched - numTPsAll) / totalTracks; const double eff = numTPsEff / totalTPs; const double errEff = sqrt(eff * (1. - eff) / totalTPs / nEvents_); - const double effLoss = numTPsLost / totalTPs; - const double errEffLoss = sqrt(effLoss * (1. - effLoss) / totalTPs / nEvents_); + const double effMax = numTPsEffMax / totalTPs; + const double errEffMax = sqrt(effMax * (1. - effMax) / totalTPs / nEvents_); const int numStates = prof_->GetBinContent(10); const int numStatesLost = prof_->GetBinContent(11); const double fracSatest = numStates / (double)(numStates + numStatesLost); - const vector nums = {numStubs, numTracks, numTracksLost}; - const vector errs = {errStubs, errTracks, errTracksLost}; + const vector nums = {numStubs, numTracks}; + const vector errs = {errStubs, errTracks}; const int wNums = ceil(log10(*max_element(nums.begin(), nums.end()))) + 5; const int wErrs = ceil(log10(*max_element(errs.begin(), errs.end()))) + 5; log_ << " KF SUMMARY " << endl; log_ << "number of stubs per TFP = " << setw(wNums) << numStubs << " +- " << setw(wErrs) << errStubs << endl; log_ << "number of tracks per TFP = " << setw(wNums) << numTracks << " +- " << setw(wErrs) << errTracks << endl; - log_ << "number of lost tracks per TFP = " << setw(wNums) << numTracksLost << " +- " << setw(wErrs) << errTracksLost - << endl; log_ << " tracking efficiency = " << setw(wNums) << eff << " +- " << setw(wErrs) << errEff << endl; - log_ << " lost tracking efficiency = " << setw(wNums) << effLoss << " +- " << setw(wErrs) << errEffLoss << endl; + log_ << " max tracking efficiency = " << setw(wNums) << effMax << " +- " << setw(wErrs) << errEffMax << endl; log_ << " fake rate = " << setw(wNums) << fracFake << endl; log_ << " duplicate rate = " << setw(wNums) << fracDup << endl; log_ << " state assessment fraction = " << setw(wNums) << fracSatest << endl; + log_ << " number of states per TFP = " << setw(wNums) << (numStates + numStatesLost) / setup_->numRegions() + << endl; log_ << "============================================================="; - LogPrint("L1Trigger/TrackerTFP") << log_.str(); + LogPrint(moduleDescription().moduleName()) << log_.str(); } // - void AnalyzerKF::associate(const TTTracks& ttTracks, + void AnalyzerKF::associate(const vector& tracks, + const vector>& tracksStubs, + int region, const StubAssociation* ass, set& tps, int& sum, const vector& his, - TProfile* prof) const { - for (const TTTrack& ttTrack : ttTracks) { - const vector& ttStubRefs = ttTrack.getStubRefs(); - const vector& tpPtrs = ass->associateFinal(ttStubRefs); + const vector& prof, + bool perfect) const { + for (int frame = 0; frame < (int)tracks.size(); frame++) { + const TrackKF& track = tracks[frame]; + const vector& stubs = tracksStubs[frame]; + vector ttStubRefs; + ttStubRefs.reserve(stubs.size()); + TTBV hitPattern(0, setup_->numLayers()); + int layer(-1); + for (StubKF* stub : stubs) { + layer++; + if (!stub) + continue; + hitPattern.set(layer); + ttStubRefs.push_back(stub->frame().first); + } + const vector& tpPtrs = perfect ? ass->associateFinal(ttStubRefs) : ass->associate(ttStubRefs); if (tpPtrs.empty()) continue; sum++; copy(tpPtrs.begin(), tpPtrs.end(), inserter(tps, tps.begin())); if (his.empty()) continue; + const double zT = dataFormats_->format(Variable::zT, Process::gp).digi(track.zT()); + const double cot = zT / setup_->chosenRofZ() + track.cot(); + const double z0 = track.zT() - setup_->chosenRofZ() * cot; + const double inv2R = track.inv2R(); + const double phi0 = deltaPhi(track.phiT() - setup_->chosenRofPhi() * inv2R + + region * dataFormats_->format(Variable::phiT, Process::kf).range()); for (const TPPtr& tpPtr : tpPtrs) { - const double phi0 = tpPtr->phi(); - const double cot = sinh(tpPtr->eta()); - const double inv2R = setup_->invPtToDphi() * tpPtr->charge() / tpPtr->pt(); + const double tpPhi0 = tpPtr->phi(); + const double tpCot = sinh(tpPtr->eta()); + const double tpInv2R = -setup_->invPtToDphi() * tpPtr->charge() / tpPtr->pt(); const math::XYZPointD& v = tpPtr->vertex(); - const double z0 = v.z() - cot * (v.x() * cos(phi0) + v.y() * sin(phi0)); - const double dCot = cot - ttTrack.tanL(); - const double dZ0 = z0 - ttTrack.z0(); - const double dInv2R = inv2R - ttTrack.rInv(); - const double dPhi0 = deltaPhi(phi0 - ttTrack.phi()); - const vector ds = {dPhi0, dInv2R, dZ0, dCot}; - for (int i = 0; i < (int)ds.size(); i++) + const double tpZ0 = v.z() - tpCot * (v.x() * cos(tpPhi0) + v.y() * sin(tpPhi0)); + const double dCot = tpCot - cot; + const double dZ0 = tpZ0 - z0; + const double dInv2R = tpInv2R - inv2R; + const double dPhi0 = deltaPhi(tpPhi0 - phi0); + const vector ds = {dPhi0, dInv2R / setup_->invPtToDphi(), dZ0, dCot}; + for (int i = 0; i < (int)ds.size(); i++) { his[i]->Fill(ds[i]); - prof->Fill(abs(tpPtr->eta()), abs(dZ0)); + prof[i]->Fill(abs(tpPtr->eta()), abs(ds[i])); + } } } } diff --git a/L1Trigger/TrackerTFP/test/AnalyzerTT.cc b/L1Trigger/TrackerTFP/test/AnalyzerTT.cc deleted file mode 100644 index 7104041824574..0000000000000 --- a/L1Trigger/TrackerTFP/test/AnalyzerTT.cc +++ /dev/null @@ -1,131 +0,0 @@ -#include "FWCore/Framework/interface/one/EDAnalyzer.h" -#include "FWCore/Framework/interface/Run.h" -#include "FWCore/Framework/interface/Event.h" -#include "FWCore/Framework/interface/EventSetup.h" -#include "FWCore/Framework/interface/MakerMacros.h" -#include "FWCore/ParameterSet/interface/ParameterSet.h" -#include "FWCore/Utilities/interface/EDGetToken.h" -#include "FWCore/Utilities/interface/InputTag.h" -#include "FWCore/Utilities/interface/Exception.h" -#include "DataFormats/Common/interface/Handle.h" - -#include "SimTracker/TrackTriggerAssociation/interface/StubAssociation.h" -#include "L1Trigger/TrackTrigger/interface/Setup.h" - -using namespace std; -using namespace edm; -using namespace tt; - -namespace trackerTFP { - - /*! \class trackerTFP::AnalyzerTT - * \brief Class to analyze TTTracks - * \author Thomas Schuh - * \date 2020, Oct - */ - class AnalyzerTT : public one::EDAnalyzer { - public: - AnalyzerTT(const ParameterSet& iConfig); - void beginJob() override {} - void beginRun(const Run& iEvent, const EventSetup& iSetup) override; - void analyze(const Event& iEvent, const EventSetup& iSetup) override; - void endRun(const Run& iEvent, const EventSetup& iSetup) override {} - void endJob() override {} - - private: - // ED input token of tt::TTTrackRefMap - EDGetTokenT edGetTokenTTTrackMap_; - // ED input token of TTStubRef to TPPtr association for tracking efficiency - EDGetTokenT edGetTokenStubAssociation_; - // Setup token - ESGetToken esGetTokenSetup_; - // stores, calculates and provides run-time constants - const Setup* setup_ = nullptr; - }; - - AnalyzerTT::AnalyzerTT(const ParameterSet& iConfig) { - // book in- and output ED products - const auto& label = iConfig.getParameter("LabelAS"); - const auto& branch = iConfig.getParameter("BranchAcceptedTracks"); - const auto& inputTag = iConfig.getParameter("InputTagSelection"); - edGetTokenTTTrackMap_ = consumes(InputTag(label, branch)); - edGetTokenStubAssociation_ = consumes(inputTag); - // book ES products - esGetTokenSetup_ = esConsumes(); - } - - void AnalyzerTT::beginRun(const Run& iEvent, const EventSetup& iSetup) { - // helper class to store configurations - setup_ = &iSetup.getData(esGetTokenSetup_); - } - - void AnalyzerTT::analyze(const Event& iEvent, const EventSetup& iSetup) { - Handle handleTTTrackMap; - iEvent.getByToken(edGetTokenTTTrackMap_, handleTTTrackMap); - Handle handleStubAssociation; - iEvent.getByToken(edGetTokenStubAssociation_, handleStubAssociation); - if (false) - return; - for (const auto& p : *handleTTTrackMap) { - const TTTrackRef& found = p.second; - const TTTrackRef& fitted = p.first; - const vector& ttStubRefsFound = found->getStubRefs(); - const vector& tpPtrsFound = handleStubAssociation->associate(ttStubRefsFound); - if (tpPtrsFound.empty()) - continue; - const vector& ttStubRefsFitted = fitted->getStubRefs(); - const vector& tpPtrsFitted = handleStubAssociation->associate(ttStubRefsFitted); - if (!tpPtrsFitted.empty()) - continue; - const TPPtr& tpPtr = tpPtrsFound.front(); - const double off = (found->phiSector() - .5) * 2. * M_PI / setup_->numRegions() / setup_->numSectorsPhi(); - cout << setprecision(8); - cout << found->phiSector() << " " << found->etaSector() << " " << endl; - cout << "Found" << endl; - for (const TTStubRef& ttStubRef : ttStubRefsFound) { - const GlobalPoint& gp = setup_->stubPos(ttStubRef); - cout << gp.perp() << " " << gp.phi() << " " << gp.z() << " " << setup_->layerId(ttStubRef) << endl; - } - cout << "Fitted" << endl; - for (const TTStubRef& ttStubRef : ttStubRefsFitted) { - const GlobalPoint& gp = setup_->stubPos(ttStubRef); - cout << gp.perp() << " " << gp.phi() << " " << gp.z() << " " << setup_->layerId(ttStubRef) << endl; - } - cout << "TP" << endl; - for (const TTStubRef& ttStubRef : handleStubAssociation->findTTStubRefs(tpPtr)) { - const GlobalPoint& gp = setup_->stubPos(ttStubRef); - cout << gp.perp() << " " << gp.phi() << " " << gp.z() << " " << setup_->layerId(ttStubRef) << endl; - } - cout << found->hitPattern() << " " << found->trackSeedType() << endl; - cout << "m0SF = " - << " " << -found->rInv() << endl; - cout << "c0SF = " - << " " << deltaPhi(found->phi() + found->rInv() * setup_->chosenRofPhi() + off) << endl; - cout << "m1SF = " - << " " << found->tanL() + setup_->sectorCot(found->etaSector()) << endl; - cout << "c1SF = " - << " " << found->z0() - found->tanL() * setup_->chosenRofZ() << endl; - cout << "m0KF = " - << " " << -fitted->rInv() * setup_->invPtToDphi() << endl; - cout << "c0KF = " - << " " << fitted->phi() << endl; - cout << "m1KF = " - << " " << fitted->tanL() << endl; - cout << "c1KF = " - << " " << fitted->z0() << endl; - cout << "m0TP = " - << " " << -tpPtr->charge() / tpPtr->pt() * setup_->invPtToDphi() << endl; - cout << "c0TP = " - << " " << tpPtr->phi() << endl; - cout << "m1TP = " - << " " << sinh(tpPtr->eta()) << endl; - const math::XYZPointD& v = tpPtr->vertex(); - cout << "c1TP = " - << " " << v.z() - sinh(tpPtr->eta()) * (v.x() * cos(tpPtr->phi()) + v.y() * sin(tpPtr->phi())) << endl; - throw cms::Exception("..."); - } - } - -} // namespace trackerTFP - -DEFINE_FWK_MODULE(trackerTFP::AnalyzerTT); diff --git a/L1Trigger/TrackerTFP/test/AnalyzerZHT.cc b/L1Trigger/TrackerTFP/test/AnalyzerZHT.cc deleted file mode 100644 index 6dc5c5a556a29..0000000000000 --- a/L1Trigger/TrackerTFP/test/AnalyzerZHT.cc +++ /dev/null @@ -1,294 +0,0 @@ -#include "FWCore/Framework/interface/one/EDAnalyzer.h" -#include "FWCore/Framework/interface/Run.h" -#include "FWCore/Framework/interface/Event.h" -#include "FWCore/Framework/interface/EventSetup.h" -#include "FWCore/Framework/interface/MakerMacros.h" -#include "FWCore/ParameterSet/interface/ParameterSet.h" -#include "FWCore/ServiceRegistry/interface/Service.h" -#include "FWCore/MessageLogger/interface/MessageLogger.h" -#include "FWCore/Utilities/interface/EDGetToken.h" -#include "FWCore/Utilities/interface/InputTag.h" -#include "FWCore/Utilities/interface/Exception.h" -#include "CommonTools/UtilAlgos/interface/TFileService.h" -#include "DataFormats/Common/interface/Handle.h" - -#include "SimTracker/TrackTriggerAssociation/interface/StubAssociation.h" -#include "L1Trigger/TrackTrigger/interface/Setup.h" -#include "L1Trigger/TrackerTFP/interface/DataFormats.h" - -#include -#include -#include - -#include -#include -#include -#include -#include -#include - -using namespace std; -using namespace edm; -using namespace tt; - -namespace trackerTFP { - - /*! \class trackerTFP::AnalyzerZHT - * \brief Class to analyze hardware like structured TTStub Collection generated by Z Hough Transform - * \author Thomas Schuh - * \date 2021, May - */ - class AnalyzerZHT : public one::EDAnalyzer { - public: - AnalyzerZHT(const ParameterSet& iConfig); - void beginJob() override {} - void beginRun(const Run& iEvent, const EventSetup& iSetup) override; - void analyze(const Event& iEvent, const EventSetup& iSetup) override; - void endRun(const Run& iEvent, const EventSetup& iSetup) override {} - void endJob() override; - - private: - // - void formTracks(const StreamStub& stream, vector>& tracks) const; - // - void associate(const vector>& tracks, const StubAssociation* ass, set& tps, int& sum) const; - - // ED input token of stubs - EDGetTokenT edGetTokenAccepted_; - // ED input token of lost stubs - EDGetTokenT edGetTokenLost_; - // ED input token of TTStubRef to TPPtr association for tracking efficiency - EDGetTokenT edGetTokenSelection_; - // ED input token of TTStubRef to recontructable TPPtr association - EDGetTokenT edGetTokenReconstructable_; - // Setup token - ESGetToken esGetTokenSetup_; - // DataFormats token - ESGetToken esGetTokenDataFormats_; - // stores, calculates and provides run-time constants - const Setup* setup_ = nullptr; - // helper class to extract structured data from tt::Frames - const DataFormats* dataFormats_ = nullptr; - // enables analyze of TPs - bool useMCTruth_; - // - int nEvents_ = 0; - - // Histograms - - TProfile* prof_; - TProfile* profChannel_; - TH1F* hisChannel_; - TH1F* hisEff_; - TH1F* hisEffTotal_; - TEfficiency* eff_; - - // printout - stringstream log_; - }; - - AnalyzerZHT::AnalyzerZHT(const ParameterSet& iConfig) : useMCTruth_(iConfig.getParameter("UseMCTruth")) { - usesResource("TFileService"); - // book in- and output ED products - const string& label = iConfig.getParameter("LabelZHT"); - const string& branchAccepted = iConfig.getParameter("BranchAcceptedStubs"); - const string& branchLost = iConfig.getParameter("BranchLostStubs"); - edGetTokenAccepted_ = consumes(InputTag(label, branchAccepted)); - edGetTokenLost_ = consumes(InputTag(label, branchLost)); - if (useMCTruth_) { - const auto& inputTagSelecttion = iConfig.getParameter("InputTagSelection"); - const auto& inputTagReconstructable = iConfig.getParameter("InputTagReconstructable"); - edGetTokenSelection_ = consumes(inputTagSelecttion); - edGetTokenReconstructable_ = consumes(inputTagReconstructable); - } - // book ES products - esGetTokenSetup_ = esConsumes(); - esGetTokenDataFormats_ = esConsumes(); - // log config - log_.setf(ios::fixed, ios::floatfield); - log_.precision(4); - } - - void AnalyzerZHT::beginRun(const Run& iEvent, const EventSetup& iSetup) { - // helper class to store configurations - setup_ = &iSetup.getData(esGetTokenSetup_); - // helper class to extract structured data from tt::Frames - dataFormats_ = &iSetup.getData(esGetTokenDataFormats_); - // book histograms - Service fs; - TFileDirectory dir; - dir = fs->mkdir("ZHT"); - prof_ = dir.make("Counts", ";", 9, 0.5, 9.5); - prof_->GetXaxis()->SetBinLabel(1, "Stubs"); - prof_->GetXaxis()->SetBinLabel(2, "Tracks"); - prof_->GetXaxis()->SetBinLabel(3, "Lost Tracks"); - prof_->GetXaxis()->SetBinLabel(4, "Matched Tracks"); - prof_->GetXaxis()->SetBinLabel(5, "All Tracks"); - prof_->GetXaxis()->SetBinLabel(6, "Found TPs"); - prof_->GetXaxis()->SetBinLabel(7, "Found selected TPs"); - prof_->GetXaxis()->SetBinLabel(8, "Lost TPs"); - prof_->GetXaxis()->SetBinLabel(9, "All TPs"); - // channel occupancy - constexpr int maxOcc = 180; - const int numChannels = dataFormats_->numChannel(Process::zht); - hisChannel_ = dir.make("His Channel Occupancy", ";", maxOcc, -.5, maxOcc - .5); - profChannel_ = dir.make("Prof Channel Occupancy", ";", numChannels, -.5, numChannels - .5); - // Efficiencies - hisEffTotal_ = dir.make("HisTPEtaTotal", ";", 128, -2.5, 2.5); - hisEff_ = dir.make("HisTPEta", ";", 128, -2.5, 2.5); - eff_ = dir.make("EffEta", ";", 128, -2.5, 2.5); - } - - void AnalyzerZHT::analyze(const Event& iEvent, const EventSetup& iSetup) { - auto fill = [](const TPPtr& tpPtr, TH1F* his) { his->Fill(tpPtr->eta()); }; - // read in ht products - Handle handleAccepted; - iEvent.getByToken(edGetTokenAccepted_, handleAccepted); - Handle handleLost; - iEvent.getByToken(edGetTokenLost_, handleLost); - // read in MCTruth - const StubAssociation* selection = nullptr; - const StubAssociation* reconstructable = nullptr; - if (useMCTruth_) { - Handle handleSelection; - iEvent.getByToken(edGetTokenSelection_, handleSelection); - selection = handleSelection.product(); - prof_->Fill(9, selection->numTPs()); - Handle handleReconstructable; - iEvent.getByToken(edGetTokenReconstructable_, handleReconstructable); - reconstructable = handleReconstructable.product(); - for (const auto& p : selection->getTrackingParticleToTTStubsMap()) - fill(p.first, hisEffTotal_); - } - // analyze ht products and associate found tracks with reconstrucable TrackingParticles - set tpPtrs; - set tpPtrsSelection; - set tpPtrsLost; - int allMatched(0); - int allTracks(0); - for (int region = 0; region < setup_->numRegions(); region++) { - int nStubs(0); - int nTracks(0); - int nLost(0); - for (int channel = 0; channel < dataFormats_->numChannel(Process::mht); channel++) { - const int index = region * dataFormats_->numChannel(Process::mht) + channel; - const StreamStub& accepted = handleAccepted->at(index); - hisChannel_->Fill(accepted.size()); - profChannel_->Fill(channel, accepted.size()); - nStubs += accumulate(accepted.begin(), accepted.end(), 0, [](int sum, const FrameStub& frame) { - return sum + (frame.first.isNonnull() ? 1 : 0); - }); - vector> tracks; - vector> lost; - formTracks(accepted, tracks); - formTracks(handleLost->at(index), lost); - nTracks += tracks.size(); - allTracks += tracks.size(); - nLost += lost.size(); - if (!useMCTruth_) - continue; - int tmp(0); - associate(tracks, selection, tpPtrsSelection, tmp); - associate(lost, selection, tpPtrsLost, tmp); - associate(tracks, reconstructable, tpPtrs, allMatched); - } - prof_->Fill(1, nStubs); - prof_->Fill(2, nTracks); - prof_->Fill(3, nLost); - } - for (const TPPtr& tpPtr : tpPtrsSelection) - fill(tpPtr, hisEff_); - vector recovered; - recovered.reserve(tpPtrsLost.size()); - set_intersection(tpPtrsLost.begin(), tpPtrsLost.end(), tpPtrs.begin(), tpPtrs.end(), back_inserter(recovered)); - for (const TPPtr& tpPtr : recovered) - tpPtrsLost.erase(tpPtr); - prof_->Fill(4, allMatched); - prof_->Fill(5, allTracks); - prof_->Fill(6, tpPtrs.size()); - prof_->Fill(7, tpPtrsSelection.size()); - prof_->Fill(8, tpPtrsLost.size()); - nEvents_++; - //if ((int)tpPtrsSelection.size() != selection->numTPs()) - //throw cms::Exception("..."); - } - - void AnalyzerZHT::endJob() { - if (nEvents_ == 0) - return; - // effi - eff_->SetPassedHistogram(*hisEff_, "f"); - eff_->SetTotalHistogram(*hisEffTotal_, "f"); - // printout SF summary - const double totalTPs = prof_->GetBinContent(9); - const double numStubs = prof_->GetBinContent(1); - const double numTracks = prof_->GetBinContent(2); - const double numTracksLost = prof_->GetBinContent(3); - const double totalTracks = prof_->GetBinContent(5); - const double numTracksMatched = prof_->GetBinContent(4); - const double numTPsAll = prof_->GetBinContent(6); - const double numTPsEff = prof_->GetBinContent(7); - const double numTPsLost = prof_->GetBinContent(8); - const double errStubs = prof_->GetBinError(1); - const double errTracks = prof_->GetBinError(2); - const double errTracksLost = prof_->GetBinError(3); - const double fracFake = (totalTracks - numTracksMatched) / totalTracks; - const double fracDup = (numTracksMatched - numTPsAll) / totalTracks; - const double eff = numTPsEff / totalTPs; - const double errEff = sqrt(eff * (1. - eff) / totalTPs / nEvents_); - const double effLoss = numTPsLost / totalTPs; - const double errEffLoss = sqrt(effLoss * (1. - effLoss) / totalTPs / nEvents_); - const vector nums = {numStubs, numTracks, numTracksLost}; - const vector errs = {errStubs, errTracks, errTracksLost}; - const int wNums = ceil(log10(*max_element(nums.begin(), nums.end()))) + 5; - const int wErrs = ceil(log10(*max_element(errs.begin(), errs.end()))) + 5; - log_ << " ZHT SUMMARY " << endl; - log_ << "number of stubs per TFP = " << setw(wNums) << numStubs << " +- " << setw(wErrs) << errStubs << endl; - log_ << "number of tracks per TFP = " << setw(wNums) << numTracks << " +- " << setw(wErrs) << errTracks - << endl; - log_ << "number of lost tracks per TFP = " << setw(wNums) << numTracksLost << " +- " << setw(wErrs) << errTracksLost - << endl; - log_ << " max tracking efficiency = " << setw(wNums) << eff << " +- " << setw(wErrs) << errEff << endl; - log_ << " lost tracking efficiency = " << setw(wNums) << effLoss << " +- " << setw(wErrs) << errEffLoss << endl; - log_ << " fake rate = " << setw(wNums) << fracFake << endl; - log_ << " duplicate rate = " << setw(wNums) << fracDup << endl; - log_ << "============================================================="; - LogPrint("L1Trigger/TrackerTFP") << log_.str(); - } - - // - void AnalyzerZHT::formTracks(const StreamStub& stream, vector>& tracks) const { - vector stubs; - stubs.reserve(stream.size()); - for (const FrameStub& frame : stream) - if (frame.first.isNonnull()) - stubs.emplace_back(frame, dataFormats_); - for (auto it = stubs.begin(); it != stubs.end();) { - const auto start = it; - const int id = it->trackId(); - auto different = [id](const StubZHT& stub) { return id != stub.trackId(); }; - it = find_if(it, stubs.end(), different); - vector ttStubRefs; - ttStubRefs.reserve(distance(start, it)); - transform(start, it, back_inserter(ttStubRefs), [](const StubZHT& stub) { return stub.ttStubRef(); }); - tracks.push_back(ttStubRefs); - } - } - - // - void AnalyzerZHT::associate(const vector>& tracks, - const StubAssociation* ass, - set& tps, - int& sum) const { - for (const vector& ttStubRefs : tracks) { - const vector& tpPtrs = ass->associate(ttStubRefs); - if (tpPtrs.empty()) - continue; - sum++; - copy(tpPtrs.begin(), tpPtrs.end(), inserter(tps, tps.begin())); - } - } - -} // namespace trackerTFP - -DEFINE_FWK_MODULE(trackerTFP::AnalyzerZHT); diff --git a/L1Trigger/TrackerTFP/test/ProducerAS.cc b/L1Trigger/TrackerTFP/test/ProducerAS.cc deleted file mode 100644 index 685f992330619..0000000000000 --- a/L1Trigger/TrackerTFP/test/ProducerAS.cc +++ /dev/null @@ -1,97 +0,0 @@ -#include "FWCore/Framework/interface/stream/EDProducer.h" -#include "FWCore/Framework/interface/Run.h" -#include "FWCore/Framework/interface/EventSetup.h" -#include "FWCore/Framework/interface/Event.h" -#include "FWCore/Framework/interface/MakerMacros.h" -#include "FWCore/Utilities/interface/EDGetToken.h" -#include "FWCore/Utilities/interface/EDPutToken.h" -#include "FWCore/Utilities/interface/ESGetToken.h" -#include "FWCore/Utilities/interface/InputTag.h" -#include "FWCore/ParameterSet/interface/ParameterSet.h" -#include "DataFormats/Common/interface/Handle.h" - -#include "DataFormats/L1TrackTrigger/interface/TTTypes.h" -#include "L1Trigger/TrackTrigger/interface/Setup.h" - -#include - -using namespace std; -using namespace edm; -using namespace tt; - -namespace trackerTFP { - - /*! \class trackerTFP::ProducerAS - * \brief Associate the TTTracks output by KF fitter with the tracks input to KF fitter. - * \author Thomas Schuh - * \date 2020, Oct - */ - class ProducerAS : public stream::EDProducer<> { - public: - explicit ProducerAS(const ParameterSet&); - ~ProducerAS() override {} - - private: - void beginRun(const Run&, const EventSetup&) override; - void produce(Event&, const EventSetup&) override; - void endJob() {} - - // ED input token of kf tracks - EDGetTokenT edGetTokenKF_; - // ED input token of kf TTtracks - EDGetTokenT edGetTokenTT_; - // ED output token for TTTrackRefMap - EDPutTokenT edPutToken_; - // Setup token - ESGetToken esGetTokenSetup_; - // configuration - ParameterSet iConfig_; - // helper class to store configurations - const Setup* setup_ = nullptr; - }; - - ProducerAS::ProducerAS(const ParameterSet& iConfig) : iConfig_(iConfig) { - const string& labelKF = iConfig.getParameter("LabelKF"); - const string& labelTT = iConfig.getParameter("LabelTT"); - const string& branch = iConfig.getParameter("BranchAcceptedTracks"); - // book in- and output ED products - edGetTokenKF_ = consumes(InputTag(labelKF, branch)); - edGetTokenTT_ = consumes(InputTag(labelTT, branch)); - edPutToken_ = produces(branch); - // book ES products - esGetTokenSetup_ = esConsumes(); - } - - void ProducerAS::beginRun(const Run& iRun, const EventSetup& iSetup) { - // helper class to store configurations - setup_ = &iSetup.getData(esGetTokenSetup_); - if (!setup_->configurationSupported()) - return; - // check process history if desired - if (iConfig_.getParameter("CheckHistory")) - setup_->checkHistory(iRun.processHistory()); - } - - void ProducerAS::produce(Event& iEvent, const EventSetup& iSetup) { - // empty KFTTTrack product - TTTrackRefMap ttTrackMap; - // read in KF Product and produce AssociatorKF product - if (setup_->configurationSupported()) { - Handle handleKF; - iEvent.getByToken(edGetTokenKF_, handleKF); - const StreamsTrack& streams = *handleKF.product(); - Handle handleTT; - iEvent.getByToken(edGetTokenTT_, handleTT); - int i(0); - for (const StreamTrack& stream : streams) - for (const FrameTrack& frame : stream) - if (frame.first.isNonnull()) - ttTrackMap.emplace(TTTrackRef(handleTT, i++), frame.first); - } - // store products - iEvent.emplace(edPutToken_, std::move(ttTrackMap)); - } - -} // namespace trackerTFP - -DEFINE_FWK_MODULE(trackerTFP::ProducerAS); diff --git a/L1Trigger/TrackerTFP/test/demonstrator_cfg.py b/L1Trigger/TrackerTFP/test/demonstrator_cfg.py index 7d8f493cb7307..367c617f78dba 100644 --- a/L1Trigger/TrackerTFP/test/demonstrator_cfg.py +++ b/L1Trigger/TrackerTFP/test/demonstrator_cfg.py @@ -3,8 +3,8 @@ process = cms.Process( "Demo" ) process.load( 'FWCore.MessageService.MessageLogger_cfi' ) -process.load( 'Configuration.Geometry.GeometryExtendedRun4D88Reco_cff' ) -process.load( 'Configuration.Geometry.GeometryExtendedRun4D88_cff' ) +process.load( 'Configuration.Geometry.GeometryExtended2026D98Reco_cff' ) +process.load( 'Configuration.Geometry.GeometryExtended2026D98_cff' ) process.load( 'Configuration.StandardSequences.MagneticField_cff' ) process.load( 'Configuration.StandardSequences.FrontierConditions_GlobalTag_cff' ) process.load( 'L1Trigger.TrackTrigger.TrackTrigger_cff' ) @@ -13,25 +13,28 @@ process.GlobalTag = GlobalTag(process.GlobalTag, 'auto:phase2_realistic', '') # load code that produces DTCStubs -process.load( 'L1Trigger.TrackerDTC.ProducerED_cff' ) +process.load( 'L1Trigger.TrackerDTC.DTC_cff' ) # cosutmize TT algorithm -#from L1Trigger.TrackerDTC.Customize_cff import * -#producerUseTMTT(process) -#analyzerUseTMTT(process) +from L1Trigger.TrackerDTC.Customize_cff import * +producerUseTMTT(process) +analyzerUseTMTT(process) #--- Load code that produces tfp Stubs process.load( 'L1Trigger.TrackerTFP.Producer_cff' ) +from L1Trigger.TrackerTFP.Customize_cff import * +setupUseTMTT( process ) #--- Load code that demonstrates tfp Stubs process.load( 'L1Trigger.TrackerTFP.Demonstrator_cff' ) # build schedule -process.tt = cms.Sequence ( process.TrackerDTCProducer - + process.TrackerTFPProducerGP - + process.TrackerTFPProducerHT - + process.TrackerTFPProducerMHT - + process.TrackerTFPProducerZHT - + process.TrackerTFPProducerZHTout - + process.TrackerTFPProducerKFin - + process.TrackerTFPProducerKF +process.tt = cms.Sequence ( process.ProducerDTC + + process.ProducerPP + + process.ProducerGP + + process.ProducerHT + + process.ProducerCTB + + process.ProducerKF + + process.ProducerDR + + process.ProducerTQ + + process.ProducerTFP ) process.demo = cms.Path( process.tt + process.TrackerTFPDemonstrator ) process.schedule = cms.Schedule( process.demo ) @@ -41,7 +44,16 @@ options = VarParsing.VarParsing( 'analysis' ) # specify input MC Samples = [ -'/store/mc/CMSSW_12_6_0/RelValTTbar_14TeV/GEN-SIM-DIGI-RAW/PU_125X_mcRun4_realistic_v5_2026D88PU200RV183v2-v1/30000/0959f326-3f52-48d8-9fcf-65fc41de4e27.root' +'/store/relval/CMSSW_14_0_0_pre2/RelValTTbar_14TeV/GEN-SIM-DIGI-RAW/PU_133X_mcRun4_realistic_v1_STD_2026D98_PU200_RV229-v1/2580000/0b2b0b0b-f312-48a8-9d46-ccbadc69bbfd.root', +'/store/relval/CMSSW_14_0_0_pre2/RelValTTbar_14TeV/GEN-SIM-DIGI-RAW/PU_133X_mcRun4_realistic_v1_STD_2026D98_PU200_RV229-v1/2580000/0c3cb20d-8556-450d-b4f0-e5c754818f74.root', +'/store/relval/CMSSW_14_0_0_pre2/RelValTTbar_14TeV/GEN-SIM-DIGI-RAW/PU_133X_mcRun4_realistic_v1_STD_2026D98_PU200_RV229-v1/2580000/0eafa2b4-711a-43ec-be1c-7e564c294a9a.root', +'/store/relval/CMSSW_14_0_0_pre2/RelValTTbar_14TeV/GEN-SIM-DIGI-RAW/PU_133X_mcRun4_realistic_v1_STD_2026D98_PU200_RV229-v1/2580000/1450b1bb-171e-495e-a767-68e2796d95c2.root', +'/store/relval/CMSSW_14_0_0_pre2/RelValTTbar_14TeV/GEN-SIM-DIGI-RAW/PU_133X_mcRun4_realistic_v1_STD_2026D98_PU200_RV229-v1/2580000/15498564-9cf0-4219-aab7-f97b3484b122.root', +'/store/relval/CMSSW_14_0_0_pre2/RelValTTbar_14TeV/GEN-SIM-DIGI-RAW/PU_133X_mcRun4_realistic_v1_STD_2026D98_PU200_RV229-v1/2580000/1838a806-316b-4f53-9d22-5b3856019623.root', +'/store/relval/CMSSW_14_0_0_pre2/RelValTTbar_14TeV/GEN-SIM-DIGI-RAW/PU_133X_mcRun4_realistic_v1_STD_2026D98_PU200_RV229-v1/2580000/1a34eb87-b9a3-47fb-b945-57e6f775fcac.root', +'/store/relval/CMSSW_14_0_0_pre2/RelValTTbar_14TeV/GEN-SIM-DIGI-RAW/PU_133X_mcRun4_realistic_v1_STD_2026D98_PU200_RV229-v1/2580000/1add5b2e-19cb-4581-956d-271907d03b72.root', +'/store/relval/CMSSW_14_0_0_pre2/RelValTTbar_14TeV/GEN-SIM-DIGI-RAW/PU_133X_mcRun4_realistic_v1_STD_2026D98_PU200_RV229-v1/2580000/1bed1837-ef65-4e07-a2ac-13c705b20fc1.root', +'/store/relval/CMSSW_14_0_0_pre2/RelValTTbar_14TeV/GEN-SIM-DIGI-RAW/PU_133X_mcRun4_realistic_v1_STD_2026D98_PU200_RV229-v1/2580000/1d057884-72bd-4353-8375-ec4616c00a33.root' ] options.register( 'inputMC', Samples, VarParsing.VarParsing.multiplicity.singleton, VarParsing.VarParsing.varType.string, "Files to be processed" ) # specify number of events to process. @@ -53,7 +65,7 @@ process.source = cms.Source( "PoolSource", fileNames = cms.untracked.vstring( options.inputMC ), - #skipEvents = cms.untracked.uint32( 914 ), + #skipEvents = cms.untracked.uint32( 2 ), secondaryFileNames = cms.untracked.vstring(), duplicateCheckMode = cms.untracked.string( 'noDuplicateCheck' ) ) diff --git a/L1Trigger/TrackerTFP/test/test_cfg.py b/L1Trigger/TrackerTFP/test/test_cfg.py index b4836e0ff0e9e..2529c53e6d895 100644 --- a/L1Trigger/TrackerTFP/test/test_cfg.py +++ b/L1Trigger/TrackerTFP/test/test_cfg.py @@ -10,8 +10,8 @@ process = cms.Process( "Demo" ) process.load( 'FWCore.MessageService.MessageLogger_cfi' ) -process.load( 'Configuration.Geometry.GeometryExtendedRun4D88Reco_cff' ) -process.load( 'Configuration.Geometry.GeometryExtendedRun4D88_cff' ) +process.load( 'Configuration.Geometry.GeometryExtended2026D98Reco_cff' ) +process.load( 'Configuration.Geometry.GeometryExtended2026D98_cff' ) process.load( 'Configuration.StandardSequences.MagneticField_cff' ) process.load( 'Configuration.StandardSequences.FrontierConditions_GlobalTag_cff' ) process.load( 'L1Trigger.TrackTrigger.TrackTrigger_cff' ) @@ -22,7 +22,7 @@ # load code that associates stubs with mctruth process.load( 'SimTracker.TrackTriggerAssociation.StubAssociator_cff' ) # load code that produces DTCStubs -process.load( 'L1Trigger.TrackerDTC.ProducerED_cff' ) +process.load( 'L1Trigger.TrackerDTC.DTC_cff' ) # load code that analyzes DTCStubs process.load( 'L1Trigger.TrackerDTC.Analyzer_cff' ) # cosutmize TT algorithm @@ -33,20 +33,22 @@ process.load( 'L1Trigger.TrackerTFP.Producer_cff' ) from L1Trigger.TrackerTFP.Customize_cff import * setupUseTMTT( process ) +simUseTMTT( process ) #--- Load code that analyzes tfp Stubs process.load( 'L1Trigger.TrackerTFP.Analyzer_cff' ) # build schedule -process.mc = cms.Sequence( process.StubAssociator ) -process.dtc = cms.Sequence( process.TrackerDTCProducer + process.TrackerDTCAnalyzer ) -process.gp = cms.Sequence( process.TrackerTFPProducerGP + process.TrackerTFPAnalyzerGP ) -process.ht = cms.Sequence( process.TrackerTFPProducerHT + process.TrackerTFPAnalyzerHT ) -process.mht = cms.Sequence( process.TrackerTFPProducerMHT + process.TrackerTFPAnalyzerMHT ) -process.zht = cms.Sequence( process.TrackerTFPProducerZHT + process.TrackerTFPAnalyzerZHT ) -process.interIn = cms.Sequence( process.TrackerTFPProducerZHTout + process.TrackerTFPProducerKFin + process.TrackerTFPAnalyzerKFin ) -process.kf = cms.Sequence( process.TrackerTFPProducerKF + process.TrackerTFPAnalyzerKF ) -process.interOut = cms.Sequence( process.TrackerTFPProducerTT + process.TrackerTFPProducerAS )#+ process.TrackerTFPAnalyzerTT ) -process.tt = cms.Path( process.mc + process.dtc + process.gp + process.ht + process.mht + process.zht + process.interIn + process.kf )#+ process.interOut ) +process.mc = cms.Sequence( process.StubAssociator ) +process.dtc = cms.Sequence( process.ProducerDTC + process.AnalyzerDTC ) +process.pp = cms.Sequence( process.ProducerPP ) +process.gp = cms.Sequence( process.ProducerGP + process.AnalyzerGP ) +process.ht = cms.Sequence( process.ProducerHT + process.AnalyzerHT ) +process.ctb = cms.Sequence( process.ProducerCTB + process.AnalyzerCTB ) +process.kf = cms.Sequence( process.ProducerKF + process.AnalyzerKF ) +process.dr = cms.Sequence( process.ProducerDR + process.AnalyzerDR ) +process.tq = cms.Sequence( process.ProducerTQ ) +process.tfp = cms.Sequence( process.ProducerTFP ) +process.tt = cms.Path( process.mc + process.dtc + process.pp + process.gp + process.ht + process.ctb + process.kf )#+ process.dr + process.tq )# + process.tfp ) process.schedule = cms.Schedule( process.tt ) # create options @@ -54,7 +56,16 @@ options = VarParsing.VarParsing( 'analysis' ) # specify input MC Samples = [ -'/store/mc/CMSSW_12_6_0/RelValTTbar_14TeV/GEN-SIM-DIGI-RAW/PU_125X_mcRun4_realistic_v5_2026D88PU200RV183v2-v1/30000/0959f326-3f52-48d8-9fcf-65fc41de4e27.root' +'/store/relval/CMSSW_14_0_0_pre2/RelValTTbar_14TeV/GEN-SIM-DIGI-RAW/PU_133X_mcRun4_realistic_v1_STD_2026D98_PU200_RV229-v1/2580000/0b2b0b0b-f312-48a8-9d46-ccbadc69bbfd.root', +'/store/relval/CMSSW_14_0_0_pre2/RelValTTbar_14TeV/GEN-SIM-DIGI-RAW/PU_133X_mcRun4_realistic_v1_STD_2026D98_PU200_RV229-v1/2580000/0c3cb20d-8556-450d-b4f0-e5c754818f74.root', +'/store/relval/CMSSW_14_0_0_pre2/RelValTTbar_14TeV/GEN-SIM-DIGI-RAW/PU_133X_mcRun4_realistic_v1_STD_2026D98_PU200_RV229-v1/2580000/0eafa2b4-711a-43ec-be1c-7e564c294a9a.root', +'/store/relval/CMSSW_14_0_0_pre2/RelValTTbar_14TeV/GEN-SIM-DIGI-RAW/PU_133X_mcRun4_realistic_v1_STD_2026D98_PU200_RV229-v1/2580000/1450b1bb-171e-495e-a767-68e2796d95c2.root', +'/store/relval/CMSSW_14_0_0_pre2/RelValTTbar_14TeV/GEN-SIM-DIGI-RAW/PU_133X_mcRun4_realistic_v1_STD_2026D98_PU200_RV229-v1/2580000/15498564-9cf0-4219-aab7-f97b3484b122.root', +'/store/relval/CMSSW_14_0_0_pre2/RelValTTbar_14TeV/GEN-SIM-DIGI-RAW/PU_133X_mcRun4_realistic_v1_STD_2026D98_PU200_RV229-v1/2580000/1838a806-316b-4f53-9d22-5b3856019623.root', +'/store/relval/CMSSW_14_0_0_pre2/RelValTTbar_14TeV/GEN-SIM-DIGI-RAW/PU_133X_mcRun4_realistic_v1_STD_2026D98_PU200_RV229-v1/2580000/1a34eb87-b9a3-47fb-b945-57e6f775fcac.root', +'/store/relval/CMSSW_14_0_0_pre2/RelValTTbar_14TeV/GEN-SIM-DIGI-RAW/PU_133X_mcRun4_realistic_v1_STD_2026D98_PU200_RV229-v1/2580000/1add5b2e-19cb-4581-956d-271907d03b72.root', +'/store/relval/CMSSW_14_0_0_pre2/RelValTTbar_14TeV/GEN-SIM-DIGI-RAW/PU_133X_mcRun4_realistic_v1_STD_2026D98_PU200_RV229-v1/2580000/1bed1837-ef65-4e07-a2ac-13c705b20fc1.root', +'/store/relval/CMSSW_14_0_0_pre2/RelValTTbar_14TeV/GEN-SIM-DIGI-RAW/PU_133X_mcRun4_realistic_v1_STD_2026D98_PU200_RV229-v1/2580000/1d057884-72bd-4353-8375-ec4616c00a33.root' ] options.register( 'inputMC', Samples, VarParsing.VarParsing.multiplicity.singleton, VarParsing.VarParsing.varType.string, "Files to be processed" ) # specify number of events to process. @@ -66,11 +77,11 @@ process.source = cms.Source( "PoolSource", fileNames = cms.untracked.vstring( options.inputMC ), - #skipEvents = cms.untracked.uint32( 3 + 8 ), + #skipEvents = cms.untracked.uint32( 30 ), noEventSort = cms.untracked.bool( True ), secondaryFileNames = cms.untracked.vstring(), - duplicateCheckMode = cms.untracked.string( 'noDuplicateCheck' ) + duplicateCheckMode = cms.untracked.string( 'noDuplicateCheck' ), ) -#process.Timing = cms.Service( "Timing", summaryOnly = cms.untracked.bool( True ) ) +process.Timing = cms.Service( "Timing", summaryOnly = cms.untracked.bool( True ) ) process.MessageLogger.cerr.enableStatistics = False process.TFileService = cms.Service( "TFileService", fileName = cms.string( "Hist.root" ) ) diff --git a/SimTracker/TrackTriggerAssociation/interface/StubAssociation.h b/SimTracker/TrackTriggerAssociation/interface/StubAssociation.h index e13c12c982556..b32343fc57e86 100644 --- a/SimTracker/TrackTriggerAssociation/interface/StubAssociation.h +++ b/SimTracker/TrackTriggerAssociation/interface/StubAssociation.h @@ -1,7 +1,8 @@ #ifndef SimTracker_TrackTriggerAssociation_StubAssociation_h #define SimTracker_TrackTriggerAssociation_StubAssociation_h -#include "SimDataFormats/Associations/interface/TTTypes.h" +#include "FWCore/ParameterSet/interface/ParameterSet.h" +#include "SimTracker/TrackTriggerAssociation/interface/TTTypes.h" #include "L1Trigger/TrackTrigger/interface/Setup.h" #include @@ -19,7 +20,7 @@ namespace tt { class StubAssociation { public: StubAssociation() { setup_ = nullptr; } - StubAssociation(const Setup* setup) : setup_(setup) {} + StubAssociation(const edm::ParameterSet& pSet, const Setup* setup); ~StubAssociation() {} // insert a TPPtr and its associated collection of TTstubRefs into the underlayering maps void insert(const TPPtr& tpPtr, const std::vector& ttSTubRefs); @@ -47,6 +48,14 @@ namespace tt { private: // stores, calculates and provides run-time constants const Setup* setup_; + // required number of layers a found track has to have in common with a TP to consider it matched + int minLayersGood_; + // required number of ps layers a found track has to have in common with a TP to consider it matched + int minLayersGoodPS_; + // max number of unassociated 2S stubs allowed to still associate TTTrack with TP + int maxLayersBad_; + // max number of unassociated PS stubs allowed to still associate TTTrack with TP + int maxLayersBadPS_; // map containing TTStubRef and their associated collection of TPPtrs std::map> mapTTStubRefsTPPtrs_; // map containing TPPtr and their associated collection of TTStubRefs diff --git a/SimTracker/TrackTriggerAssociation/plugins/StubAssociator.cc b/SimTracker/TrackTriggerAssociation/plugins/StubAssociator.cc index f631345a24373..f68d8b66b1c44 100644 --- a/SimTracker/TrackTriggerAssociation/plugins/StubAssociator.cc +++ b/SimTracker/TrackTriggerAssociation/plugins/StubAssociator.cc @@ -1,4 +1,4 @@ -#include "FWCore/Framework/interface/global/EDProducer.h" +#include "FWCore/Framework/interface/stream/EDProducer.h" #include "FWCore/Framework/interface/Run.h" #include "FWCore/Framework/interface/EventSetup.h" #include "FWCore/Framework/interface/Event.h" @@ -9,16 +9,16 @@ #include "FWCore/Utilities/interface/EDPutToken.h" #include "DataFormats/Common/interface/Handle.h" -#include "SimDataFormats/Associations/interface/TTTypes.h" +#include "SimTracker/TrackTriggerAssociation/interface/TTTypes.h" #include "SimTracker/TrackTriggerAssociation/interface/StubAssociation.h" #include "L1Trigger/TrackTrigger/interface/Setup.h" #include #include +#include #include #include #include -#include using namespace std; using namespace edm; @@ -34,12 +34,17 @@ namespace tt { * \author Thomas Schuh * \date 2020, Apr */ - class StubAssociator : public global::EDProducer<> { + class StubAssociator : public stream::EDProducer<> { public: explicit StubAssociator(const ParameterSet&); + ~StubAssociator() override {} private: - void produce(StreamID, Event&, const EventSetup&) const override; + void beginRun(const Run&, const EventSetup&) override; + void produce(Event&, const EventSetup&) override; + void endJob() {} + // helper classe to store configurations + const Setup* setup_; // ED input token of TTStubs EDGetTokenT getTokenTTStubDetSetVec_; // ED input token of TTClusterAssociation @@ -50,25 +55,70 @@ namespace tt { EDPutTokenT putTokenSelection_; // Setup token ESGetToken esGetTokenSetup_; + // + ParameterSet pSet_; + // required number of associated stub layers to a TP to consider it reconstruct-able + int minLayers_; + // required number of associated ps stub layers to a TP to consider it reconstruct-able + int minLayersPS_; + // pt cut in GeV + double minPt_; + // max eta for TP with z0 = 0 + double maxEta0_; + // half lumi region size in cm + double maxZ0_; + // cut on impact parameter in cm + double maxD0_; + // cut on vertex pos r in cm + double maxVertR_; + // cut on vertex pos z in cm + double maxVertZ_; + // cut on TP zT + double maxZT_; + // selector to partly select TPs for efficiency measurements + TrackingParticleSelector tpSelector_; }; - StubAssociator::StubAssociator(const ParameterSet& iConfig) { + StubAssociator::StubAssociator(const ParameterSet& iConfig) + : pSet_(iConfig), + minLayers_(iConfig.getParameter("MinLayers")), + minLayersPS_(iConfig.getParameter("MinLayersPS")), + minPt_(iConfig.getParameter("MinPt")), + maxEta0_(iConfig.getParameter("MaxEta0")), + maxZ0_(iConfig.getParameter("MaxZ0")), + maxD0_(iConfig.getParameter("MaxD0")), + maxVertR_(iConfig.getParameter("MaxVertR")), + maxVertZ_(iConfig.getParameter("MaxVertZ")) { // book in- and output ed products getTokenTTStubDetSetVec_ = consumes(iConfig.getParameter("InputTagTTStubDetSetVec")); getTokenTTClusterAssMap_ = consumes(iConfig.getParameter("InputTagTTClusterAssMap")); putTokenReconstructable_ = produces(iConfig.getParameter("BranchReconstructable")); putTokenSelection_ = produces(iConfig.getParameter("BranchSelection")); // book ES product - esGetTokenSetup_ = esConsumes(); + esGetTokenSetup_ = esConsumes(); } - void StubAssociator::produce(StreamID, Event& iEvent, const EventSetup& iSetup) const { - auto const& setup = iSetup.getData(esGetTokenSetup_); + void StubAssociator::beginRun(const Run& iRun, const EventSetup& iSetup) { + setup_ = &iSetup.getData(esGetTokenSetup_); + maxZT_ = sinh(maxEta0_) * setup_->chosenRofZ(); + // configure TrackingParticleSelector + static constexpr double ptMax = 9.e9; + static constexpr int minHit = 0; + static constexpr bool signalOnly = true; + static constexpr bool intimeOnly = true; + static constexpr bool chargedOnly = true; + static constexpr bool stableOnly = false; + static const double maxEta = asinh((maxZT_ + maxZ0_) / setup_->chosenRofZ()); + tpSelector_ = TrackingParticleSelector( + minPt_, ptMax, -maxEta, maxEta, maxVertR_, maxVertZ_, minHit, signalOnly, intimeOnly, chargedOnly, stableOnly); + } + void StubAssociator::produce(Event& iEvent, const EventSetup& iSetup) { // associate TTStubs with TrackingParticles - Handle handleTTStubDetSetVec = iEvent.getHandle(getTokenTTStubDetSetVec_); - auto const& ttClusterAssMap = iEvent.get(getTokenTTClusterAssMap_); - + Handle handleTTStubDetSetVec; + iEvent.getByToken(getTokenTTStubDetSetVec_, handleTTStubDetSetVec); + Handle handleTTClusterAssMap; + iEvent.getByToken(getTokenTTClusterAssMap_, handleTTClusterAssMap); map> mapTPPtrsTTStubRefs; auto isNonnull = [](const TPPtr& tpPtr) { return tpPtr.isNonnull(); }; for (TTStubDetSetVec::const_iterator ttModule = handleTTStubDetSetVec->begin(); @@ -78,7 +128,8 @@ namespace tt { const TTStubRef ttStubRef = makeRefTo(handleTTStubDetSetVec, ttStub); set tpPtrs; for (unsigned int iClus = 0; iClus < 2; iClus++) { - const vector& assocPtrs = ttClusterAssMap.findTrackingParticlePtrs(ttStubRef->clusterRef(iClus)); + const vector& assocPtrs = + handleTTClusterAssMap->findTrackingParticlePtrs(ttStubRef->clusterRef(iClus)); copy_if(assocPtrs.begin(), assocPtrs.end(), inserter(tpPtrs, tpPtrs.begin()), isNonnull); } for (const TPPtr& tpPtr : tpPtrs) @@ -86,17 +137,30 @@ namespace tt { } } // associate reconstructable TrackingParticles with TTStubs - StubAssociation reconstructable(&setup); - StubAssociation selection(&setup); + StubAssociation reconstructable(pSet_, setup_); + StubAssociation selection(pSet_, setup_); for (const auto& p : mapTPPtrsTTStubRefs) { - if (!setup.useForReconstructable(*p.first) || !setup.reconstructable(p.second)) + // require min layers + set hitPattern, hitPatternPS; + for (const TTStubRef& ttStubRef : p.second) { + const int layerId = setup_->layerId(ttStubRef); + hitPattern.insert(layerId); + if (setup_->psModule(ttStubRef)) + hitPatternPS.insert(layerId); + } + if ((int)hitPattern.size() < minLayers_ || (int)hitPatternPS.size() < minLayersPS_) continue; reconstructable.insert(p.first, p.second); - if (setup.useForAlgEff(*p.first)) + // require parameter space + const double zT = p.first->z0() + p.first->tanl() * setup_->chosenRofZ(); + if ((abs(p.first->d0()) > maxD0_) || (abs(p.first->z0()) > maxZ0_) || (abs(zT) > maxZT_)) + continue; + // require signal only and min pt + if (tpSelector_(*p.first)) selection.insert(p.first, p.second); } - iEvent.emplace(putTokenReconstructable_, std::move(reconstructable)); - iEvent.emplace(putTokenSelection_, std::move(selection)); + iEvent.emplace(putTokenReconstructable_, move(reconstructable)); + iEvent.emplace(putTokenSelection_, move(selection)); } } // namespace tt diff --git a/SimTracker/TrackTriggerAssociation/python/StubAssociator_cff.py b/SimTracker/TrackTriggerAssociation/python/StubAssociator_cff.py index 4f7c6d9c1f9d8..7ed70928f5f58 100644 --- a/SimTracker/TrackTriggerAssociation/python/StubAssociator_cff.py +++ b/SimTracker/TrackTriggerAssociation/python/StubAssociator_cff.py @@ -7,7 +7,7 @@ import FWCore.ParameterSet.Config as cms -from L1Trigger.TrackTrigger.ProducerSetup_cff import TrackTriggerSetup +from L1Trigger.TrackTrigger.Setup_cff import TrackTriggerSetup from SimTracker.TrackTriggerAssociation.StubAssociator_cfi import StubAssociator_params StubAssociator = cms.EDProducer('tt::StubAssociator', StubAssociator_params) diff --git a/SimTracker/TrackTriggerAssociation/python/StubAssociator_cfi.py b/SimTracker/TrackTriggerAssociation/python/StubAssociator_cfi.py index 1e5a68cbab1fc..fd726425abf29 100644 --- a/SimTracker/TrackTriggerAssociation/python/StubAssociator_cfi.py +++ b/SimTracker/TrackTriggerAssociation/python/StubAssociator_cfi.py @@ -1,9 +1,23 @@ import FWCore.ParameterSet.Config as cms StubAssociator_params = cms.PSet ( - InputTagTTStubDetSetVec = cms.InputTag( "TTStubsFromPhase2TrackerDigis", "StubAccepted" ), # - InputTagTTClusterAssMap = cms.InputTag( "TTClusterAssociatorFromPixelDigis", "ClusterAccepted" ), # - InputTagTTStubAssMap = cms.InputTag( "TTStubAssociatorFromPixelDigis", "StubAccepted" ), # - BranchReconstructable = cms.string ( "Reconstructable" ), # name of StubAssociation collection made with reconstractable TPs - BranchSelection = cms.string ( "UseForAlgEff" ) # name of StubAssociation collection used for tracking efficiency + InputTagTTStubDetSetVec = cms.InputTag( "TTStubsFromPhase2TrackerDigis", "StubAccepted" ), # + InputTagTTClusterAssMap = cms.InputTag( "TTClusterAssociatorFromPixelDigis", "ClusterAccepted" ), # + #InputTagTTClusterAssMap = cms.InputTag( "CleanAssoc", "AtLeastOneCluster" ), + BranchReconstructable = cms.string ( "Reconstructable" ), # name of StubAssociation collection made with reconstractable TPs + BranchSelection = cms.string ( "UseForAlgEff" ), # name of StubAssociation collection used for tracking efficiency + + MinPt = cms.double( 2. ), # pt cut in GeV + MaxEta0 = cms.double( 2.4 ), # max eta for TP with z0 = 0 + MaxZ0 = cms.double( 15. ), # half lumi region size in cm + MaxD0 = cms.double( 5. ), # cut on impact parameter in cm + MaxVertR = cms.double( 1. ), # cut on vertex pos r in cm + MaxVertZ = cms.double( 30. ), # cut on vertex pos z in cm + MinLayers = cms.int32 ( 4 ), # required number of associated stub layers to a TP to consider it reconstruct-able + MinLayersPS = cms.int32 ( 0 ), # required number of associated ps stub layers to a TP to consider it reconstruct-able + MinLayersGood = cms.int32 ( 4 ), # required number of layers a found track has to have in common with a TP to consider it matched + MinLayersGoodPS = cms.int32 ( 0 ), # required number of ps layers a found track has to have in common with a TP to consider it matched + MaxLayersBad = cms.int32 ( 1 ), # max number of unassociated 2S stubs allowed to still associate TTTrack with TP + MaxLayersBadPS = cms.int32 ( 0 ) # max number of unassociated PS stubs allowed to still associate TTTrack with TP + ) diff --git a/SimTracker/TrackTriggerAssociation/src/StubAssociation.cc b/SimTracker/TrackTriggerAssociation/src/StubAssociation.cc index c6805b25b7019..d8ccdf762211a 100644 --- a/SimTracker/TrackTriggerAssociation/src/StubAssociation.cc +++ b/SimTracker/TrackTriggerAssociation/src/StubAssociation.cc @@ -6,9 +6,17 @@ #include using namespace std; +using namespace edm; namespace tt { + StubAssociation::StubAssociation(const edm::ParameterSet& pSet, const Setup* setup) + : setup_(setup), + minLayersGood_(pSet.getParameter("MinLayersGood")), + minLayersGoodPS_(pSet.getParameter("MinLayersGoodPS")), + maxLayersBad_(pSet.getParameter("MaxLayersBad")), + maxLayersBadPS_(pSet.getParameter("MaxLayersBadPS")) {} + // insert a TPPtr and its associated collection of TTstubRefs into the underlayering maps void StubAssociation::insert(const TPPtr& tpPtr, const vector& ttSTubRefs) { mapTPPtrsTTStubRefs_.insert({tpPtr, ttSTubRefs}); @@ -32,32 +40,32 @@ namespace tt { // Get all TPs that are matched to these stubs in at least 'tpMinLayers' layers and 'tpMinLayersPS' ps layers vector StubAssociation::associate(const vector& ttStubRefs) const { // count associated layer for each TP - map> m; - map> mPS; + map, set>> m; for (const TTStubRef& ttStubRef : ttStubRefs) { for (const TPPtr& tpPtr : findTrackingParticlePtrs(ttStubRef)) { const int layerId = setup_->layerId(ttStubRef); - m[tpPtr].insert(layerId); + m[tpPtr].first.insert(layerId); if (setup_->psModule(ttStubRef)) - mPS[tpPtr].insert(layerId); + m[tpPtr].second.insert(layerId); } } // count matched TPs - auto acc = [this](int sum, const pair>& p) { - return sum + ((int)p.second.size() < setup_->tpMinLayers() ? 0 : 1); + auto acc = [this](int sum, const pair, set>>& p) { + return sum + + ((int)p.second.first.size() < minLayersGood_ || (int)p.second.second.size() < minLayersGoodPS_ ? 0 : 1); }; const int nTPs = accumulate(m.begin(), m.end(), 0, acc); vector tpPtrs; tpPtrs.reserve(nTPs); // fill and return matched TPs for (const auto& p : m) - if ((int)p.second.size() >= setup_->tpMinLayers() && (int)mPS[p.first].size() >= setup_->tpMinLayersPS()) + if ((int)p.second.first.size() >= minLayersGood_ && (int)p.second.second.size() >= minLayersGoodPS_) tpPtrs.push_back(p.first); return tpPtrs; } // Get all TPs that are matched to these stubs in at least 'tpMinLayers' layers and 'tpMinLayersPS' ps layers with not more then 'tpMaxBadStubs2S' not associated 2S stubs and not more then 'tpMaxBadStubsPS' associated PS stubs - std::vector StubAssociation::associateFinal(const std::vector& ttStubRefs) const { + vector StubAssociation::associateFinal(const vector& ttStubRefs) const { // Get all TPs that are matched to these stubs in at least 'tpMinLayers' layers and 'tpMinLayersPS' ps layers vector tpPtrs = associate(ttStubRefs); // remove TPs with more then 'tpMaxBadStubs2S' not associated 2S stubs and more then 'tpMaxBadStubsPS' not associated PS stubs @@ -69,9 +77,7 @@ namespace tt { if (find(tpPtrs.begin(), tpPtrs.end(), tpPtr) == tpPtrs.end()) setup_->psModule(ttStubRef) ? badPS++ : bad2S++; } - if (badPS > setup_->tpMaxBadStubsPS() || bad2S > setup_->tpMaxBadStubs2S()) - return true; - return false; + return (badPS > maxLayersBadPS_ || badPS + bad2S > maxLayersBad_); }; tpPtrs.erase(remove_if(tpPtrs.begin(), tpPtrs.end(), check), tpPtrs.end()); return tpPtrs; From 1ae5ffb457d7a07f6255824b99d848fdc6c4f010 Mon Sep 17 00:00:00 2001 From: Thomas Schuh Date: Fri, 22 Nov 2024 16:34:41 +0000 Subject: [PATCH 04/14] c++20 and merge fixes --- L1Trigger/TrackFindingTracklet/plugins/ProducerDR.cc | 2 +- L1Trigger/TrackFindingTracklet/plugins/ProducerKF.cc | 2 +- L1Trigger/TrackFindingTracklet/plugins/ProducerTM.cc | 2 +- L1Trigger/TrackFindingTracklet/src/DuplicateRemoval.cc | 4 ++-- L1Trigger/TrackFindingTracklet/src/KalmanFilter.cc | 6 +++--- L1Trigger/TrackFindingTracklet/test/AnalyzerDR.cc | 2 +- L1Trigger/TrackFindingTracklet/test/AnalyzerTM.cc | 8 +------- L1Trigger/TrackFindingTracklet/test/AnalyzerTracklet.cc | 2 +- L1Trigger/TrackerDTC/interface/Stub.h | 1 - L1Trigger/TrackerDTC/test/Analyzer.cc | 1 - L1Trigger/TrackerTFP/plugins/ProducerCTB.cc | 4 ++-- L1Trigger/TrackerTFP/plugins/ProducerDR.cc | 4 ++-- L1Trigger/TrackerTFP/plugins/ProducerGP.cc | 4 ++-- L1Trigger/TrackerTFP/plugins/ProducerHT.cc | 2 +- L1Trigger/TrackerTFP/plugins/ProducerKF.cc | 4 ++-- L1Trigger/TrackerTFP/src/CleanTrackBuilder.cc | 6 +++--- L1Trigger/TrackerTFP/src/DuplicateRemoval.cc | 2 +- L1Trigger/TrackerTFP/src/HoughTransform.cc | 2 +- L1Trigger/TrackerTFP/src/KalmanFilter.cc | 2 +- L1Trigger/TrackerTFP/src/TrackFindingProcessor.cc | 2 +- L1Trigger/TrackerTFP/test/AnalyzerCTB.cc | 2 +- L1Trigger/TrackerTFP/test/AnalyzerDR.cc | 4 ++-- .../TrackTriggerAssociation/interface/StubAssociation.h | 5 ++++- .../TrackTriggerAssociation/plugins/StubAssociator.cc | 1 - 24 files changed, 34 insertions(+), 40 deletions(-) diff --git a/L1Trigger/TrackFindingTracklet/plugins/ProducerDR.cc b/L1Trigger/TrackFindingTracklet/plugins/ProducerDR.cc index 225acc4c1aeee..5df86920d3ad0 100644 --- a/L1Trigger/TrackFindingTracklet/plugins/ProducerDR.cc +++ b/L1Trigger/TrackFindingTracklet/plugins/ProducerDR.cc @@ -14,7 +14,7 @@ #include "L1Trigger/TrackerTFP/interface/LayerEncoding.h" #include "L1Trigger/TrackFindingTracklet/interface/DataFormats.h" #include "L1Trigger/TrackFindingTracklet/interface/ChannelAssignment.h" -#include "L1Trigger/TrackFindingTracklet/interface/DR.h" +#include "L1Trigger/TrackFindingTracklet/interface/DuplicateRemoval.h" #include "SimDataFormats/Associations/interface/TTTypes.h" #include diff --git a/L1Trigger/TrackFindingTracklet/plugins/ProducerKF.cc b/L1Trigger/TrackFindingTracklet/plugins/ProducerKF.cc index 68b7ad13ca971..dc21db1ec1ef6 100644 --- a/L1Trigger/TrackFindingTracklet/plugins/ProducerKF.cc +++ b/L1Trigger/TrackFindingTracklet/plugins/ProducerKF.cc @@ -119,7 +119,7 @@ namespace trklet { } void ProducerKF::produce(Event& iEvent, const EventSetup& iSetup) { - auto valid = [](int& sum, const FrameTrack& f) { return sum += f.first.isNull() ? 0 : 1; }; + auto valid = [](int sum, const FrameTrack& f) { return sum += (f.first.isNull() ? 0 : 1); }; static const int numRegions = setup_->numRegions(); static const int numLayers = setup_->numLayers(); // empty KF products diff --git a/L1Trigger/TrackFindingTracklet/plugins/ProducerTM.cc b/L1Trigger/TrackFindingTracklet/plugins/ProducerTM.cc index bc487ae8c766c..0a9d79bac3b27 100644 --- a/L1Trigger/TrackFindingTracklet/plugins/ProducerTM.cc +++ b/L1Trigger/TrackFindingTracklet/plugins/ProducerTM.cc @@ -15,7 +15,7 @@ #include "L1Trigger/TrackFindingTracklet/interface/DataFormats.h" #include "L1Trigger/TrackFindingTracklet/interface/Settings.h" #include "L1Trigger/TrackFindingTracklet/interface/TrackMultiplexer.h" -#include "SimTracker/TrackTriggerAssociation/interface/TTTypes.h" +#include "DataFormats/L1TrackTrigger/interface/TTTypes.h" #include #include diff --git a/L1Trigger/TrackFindingTracklet/src/DuplicateRemoval.cc b/L1Trigger/TrackFindingTracklet/src/DuplicateRemoval.cc index 2f4192a0c2ac1..0cebcde1731b1 100644 --- a/L1Trigger/TrackFindingTracklet/src/DuplicateRemoval.cc +++ b/L1Trigger/TrackFindingTracklet/src/DuplicateRemoval.cc @@ -32,8 +32,8 @@ namespace trklet { // read in and organize input tracks and stubs void DuplicateRemoval::consume(const StreamsTrack& streamsTrack, const StreamsStub& streamsStub) { - auto nonNullTrack = [](int& sum, const FrameTrack& frame) { return sum += (frame.first.isNonnull() ? 1 : 0); }; - auto nonNullStub = [](int& sum, const FrameStub& frame) { return sum += (frame.first.isNonnull() ? 1 : 0); }; + auto nonNullTrack = [](int sum, const FrameTrack& frame) { return sum += (frame.first.isNonnull() ? 1 : 0); }; + auto nonNullStub = [](int sum, const FrameStub& frame) { return sum += (frame.first.isNonnull() ? 1 : 0); }; // count tracks and stubs and reserve corresponding vectors int sizeStubs(0); static const int numLayers = channelAssignment_->tmNumLayers(); diff --git a/L1Trigger/TrackFindingTracklet/src/KalmanFilter.cc b/L1Trigger/TrackFindingTracklet/src/KalmanFilter.cc index 27ff45ac6793a..55dd021595bc5 100644 --- a/L1Trigger/TrackFindingTracklet/src/KalmanFilter.cc +++ b/L1Trigger/TrackFindingTracklet/src/KalmanFilter.cc @@ -44,13 +44,13 @@ namespace trklet { static const int numLayers = setup_->numLayers(); const int offset = region_ * numLayers; const StreamTrack& streamTrack = streamsTrack[region_]; - const int numTracks = accumulate(streamTrack.begin(), streamTrack.end(), 0, [](int& sum, const FrameTrack& f) { + const int numTracks = accumulate(streamTrack.begin(), streamTrack.end(), 0, [](int sum, const FrameTrack& f) { return sum += (f.first.isNull() ? 0 : 1); }); int numStubs(0); for (int layer = 0; layer < numLayers; layer++) { const StreamStub& streamStub = streamsStub[offset + layer]; - numStubs += accumulate(streamStub.begin(), streamStub.end(), 0, [](int& sum, const FrameStub& f) { + numStubs += accumulate(streamStub.begin(), streamStub.end(), 0, [](int sum, const FrameStub& f) { return sum += (f.first.isNull() ? 0 : 1); }); } @@ -241,7 +241,7 @@ namespace trklet { } // count total number of final states const int nStates = - accumulate(stream_.begin(), stream_.end(), 0, [](int& sum, State* state) { return sum += (state ? 1 : 0); }); + accumulate(stream_.begin(), stream_.end(), 0, [](int sum, State* state) { return sum += (state ? 1 : 0); }); // apply truncation if (enableTruncation_ && (int)stream_.size() > setup_->numFramesHigh()) stream_.resize(setup_->numFramesHigh()); diff --git a/L1Trigger/TrackFindingTracklet/test/AnalyzerDR.cc b/L1Trigger/TrackFindingTracklet/test/AnalyzerDR.cc index d48e360bb2c7d..7308ade969ea0 100644 --- a/L1Trigger/TrackFindingTracklet/test/AnalyzerDR.cc +++ b/L1Trigger/TrackFindingTracklet/test/AnalyzerDR.cc @@ -180,7 +180,7 @@ namespace trklet { vector> tracks; formTracks(streamsTrack, streamsStub, tracks, region); nTracks += tracks.size(); - nStubs += accumulate(tracks.begin(), tracks.end(), 0, [](int& sum, const vector& track) { + nStubs += accumulate(tracks.begin(), tracks.end(), 0, [](int sum, const vector& track) { return sum += (int)track.size(); }); allTracks += tracks.size(); diff --git a/L1Trigger/TrackFindingTracklet/test/AnalyzerTM.cc b/L1Trigger/TrackFindingTracklet/test/AnalyzerTM.cc index 4ade077b3a398..2afa92a8386e6 100644 --- a/L1Trigger/TrackFindingTracklet/test/AnalyzerTM.cc +++ b/L1Trigger/TrackFindingTracklet/test/AnalyzerTM.cc @@ -179,7 +179,7 @@ namespace trklet { vector> tracks; formTracks(streamsTrack, streamsStub, tracks, region); nTracks += tracks.size(); - nStubs += accumulate(tracks.begin(), tracks.end(), 0, [](int& sum, const vector& track) { + nStubs += accumulate(tracks.begin(), tracks.end(), 0, [](int sum, const vector& track) { return sum += (int)track.size(); }); allTracks += tracks.size(); @@ -286,9 +286,3 @@ namespace trklet { } } // namespace trklet - -<<<<<<< HEAD:L1Trigger/TrackFindingTracklet/test/AnalyzerDRin.cc -DEFINE_FWK_MODULE(trklet::AnalyzerDRin); -======= -DEFINE_FWK_MODULE(trklet::AnalyzerTM); ->>>>>>> c9a89dd5637 (squash):L1Trigger/TrackFindingTracklet/test/AnalyzerTM.cc diff --git a/L1Trigger/TrackFindingTracklet/test/AnalyzerTracklet.cc b/L1Trigger/TrackFindingTracklet/test/AnalyzerTracklet.cc index 59711d55880c2..897838737b89b 100644 --- a/L1Trigger/TrackFindingTracklet/test/AnalyzerTracklet.cc +++ b/L1Trigger/TrackFindingTracklet/test/AnalyzerTracklet.cc @@ -192,7 +192,7 @@ namespace trklet { prof_->Fill(3, 0); for (int seedType = 0; seedType < channelAssignment_->numSeedTypes(); seedType++) { const int nTracks = - accumulate(ttTrackRefs.begin(), ttTrackRefs.end(), 0, [seedType](int& sum, const TTTrackRef& ttTrackRef) { + accumulate(ttTrackRefs.begin(), ttTrackRefs.end(), 0, [seedType](int sum, const TTTrackRef& ttTrackRef) { return sum += ((int)ttTrackRef->trackSeedType() == seedType ? 1 : 0); }); hisChannel_->Fill(nTracks); diff --git a/L1Trigger/TrackerDTC/interface/Stub.h b/L1Trigger/TrackerDTC/interface/Stub.h index 0b64f0c0d645b..37d06f774ccd4 100644 --- a/L1Trigger/TrackerDTC/interface/Stub.h +++ b/L1Trigger/TrackerDTC/interface/Stub.h @@ -3,7 +3,6 @@ #include "L1Trigger/TrackTrigger/interface/Setup.h" #include "L1Trigger/TrackerDTC/interface/LayerEncoding.h" -#include "SimTracker/TrackTriggerAssociation/interface/TTTypes.h" #include "L1Trigger/TrackerTFP/interface/DataFormats.h" #include diff --git a/L1Trigger/TrackerDTC/test/Analyzer.cc b/L1Trigger/TrackerDTC/test/Analyzer.cc index 4ef94a47cb0f4..e59a0edc60534 100644 --- a/L1Trigger/TrackerDTC/test/Analyzer.cc +++ b/L1Trigger/TrackerDTC/test/Analyzer.cc @@ -19,7 +19,6 @@ #include "DataFormats/SiStripDetId/interface/StripSubdetector.h" #include "SimTracker/TrackTriggerAssociation/interface/StubAssociation.h" -#include "SimTracker/TrackTriggerAssociation/interface/TTTypes.h" #include "L1Trigger/TrackTrigger/interface/Setup.h" #include "L1Trigger/TrackerDTC/interface/LayerEncoding.h" diff --git a/L1Trigger/TrackerTFP/plugins/ProducerCTB.cc b/L1Trigger/TrackerTFP/plugins/ProducerCTB.cc index 517a4dd5525cc..5e0aacd4fdaf1 100644 --- a/L1Trigger/TrackerTFP/plugins/ProducerCTB.cc +++ b/L1Trigger/TrackerTFP/plugins/ProducerCTB.cc @@ -123,7 +123,7 @@ namespace trackerTFP { const StreamsStub& streamsStub = *handleGet.product(); // count stubs int nStubsHT(0); - auto validFrame = [](int& sum, const FrameStub& frame) { return sum += (frame.first.isNonnull() ? 1 : 0); }; + auto validFrame = [](int sum, const FrameStub& frame) { return sum += (frame.first.isNonnull() ? 1 : 0); }; for (const StreamStub& stream : streamsStub) nStubsHT += accumulate(stream.begin(), stream.end(), 0, validFrame); // create input objects and count tracks @@ -183,7 +183,7 @@ namespace trackerTFP { } // store TTTracks int nTracks(0); - auto valid = [](int& sum, TrackCTB* track) { return sum += (track ? 1 : 0); }; + auto valid = [](int sum, TrackCTB* track) { return sum += (track ? 1 : 0); }; for (const vector>& region : streamsTracks) for (const deque& channel : region) nTracks += accumulate(channel.begin(), channel.end(), 0, valid); diff --git a/L1Trigger/TrackerTFP/plugins/ProducerDR.cc b/L1Trigger/TrackerTFP/plugins/ProducerDR.cc index 3661adcf49c3c..3f4e07d945f11 100644 --- a/L1Trigger/TrackerTFP/plugins/ProducerDR.cc +++ b/L1Trigger/TrackerTFP/plugins/ProducerDR.cc @@ -93,8 +93,8 @@ namespace trackerTFP { iEvent.getByToken(edGetTokenTracks_, handleTracks); const StreamsTrack& allTracks = *handleTracks; // helper - auto validFrameT = [](int& sum, const FrameTrack& frame) { return sum += (frame.first.isNonnull() ? 1 : 0); }; - auto validFrameS = [](int& sum, const FrameStub& frame) { return sum += (frame.first.isNonnull() ? 1 : 0); }; + auto validFrameT = [](int sum, const FrameTrack& frame) { return sum += (frame.first.isNonnull() ? 1 : 0); }; + auto validFrameS = [](int sum, const FrameStub& frame) { return sum += (frame.first.isNonnull() ? 1 : 0); }; auto putT = [](const vector& objects, StreamTrack& stream) { auto toFrame = [](TrackDR* object) { return object ? object->frame() : FrameTrack(); }; stream.reserve(objects.size()); diff --git a/L1Trigger/TrackerTFP/plugins/ProducerGP.cc b/L1Trigger/TrackerTFP/plugins/ProducerGP.cc index 41c96b5a598c5..0cb5e0ada264a 100644 --- a/L1Trigger/TrackerTFP/plugins/ProducerGP.cc +++ b/L1Trigger/TrackerTFP/plugins/ProducerGP.cc @@ -93,8 +93,8 @@ namespace trackerTFP { iEvent.getByToken(edGetToken_, handle); const StreamsStub& streamsStub = *handle.product(); // helper - auto validFrame = [](int& sum, const FrameStub& frame) { return sum += (frame.first.isNonnull() ? 1 : 0); }; - auto nSectors = [](int& sum, const StubPP& object) { + auto validFrame = [](int sum, const FrameStub& frame) { return sum += (frame.first.isNonnull() ? 1 : 0); }; + auto nSectors = [](int sum, const StubPP& object) { const int nPhiT = object.phiTMax() - object.phiTMin() + 1; const int nZT = object.zTMax() - object.zTMin() + 1; return sum += nPhiT * nZT; diff --git a/L1Trigger/TrackerTFP/plugins/ProducerHT.cc b/L1Trigger/TrackerTFP/plugins/ProducerHT.cc index c5089c103df72..c509f04679078 100644 --- a/L1Trigger/TrackerTFP/plugins/ProducerHT.cc +++ b/L1Trigger/TrackerTFP/plugins/ProducerHT.cc @@ -95,7 +95,7 @@ namespace trackerTFP { iEvent.getByToken(edGetToken_, handle); const StreamsStub& streamsStub = *handle.product(); // helper - auto validFrame = [](int& sum, const FrameStub& frame) { return sum += (frame.first.isNonnull() ? 1 : 0); }; + auto validFrame = [](int sum, const FrameStub& frame) { return sum += (frame.first.isNonnull() ? 1 : 0); }; auto toFrame = [](StubHT* object) { return object ? object->frame() : FrameStub(); }; // produce HT output per region for (int region = 0; region < numRegions; region++) { diff --git a/L1Trigger/TrackerTFP/plugins/ProducerKF.cc b/L1Trigger/TrackerTFP/plugins/ProducerKF.cc index c7d74f0b4ad16..1f603c11c32cd 100644 --- a/L1Trigger/TrackerTFP/plugins/ProducerKF.cc +++ b/L1Trigger/TrackerTFP/plugins/ProducerKF.cc @@ -125,8 +125,8 @@ namespace trackerTFP { iEvent.getByToken(edGetTokenTracks_, handleTracks); const StreamsTrack& allTracks = *handleTracks; // helper - auto validFrameT = [](int& sum, const FrameTrack& frame) { return sum += (frame.first.isNonnull() ? 1 : 0); }; - auto validFrameS = [](int& sum, const FrameStub& frame) { return sum += (frame.first.isNonnull() ? 1 : 0); }; + auto validFrameT = [](int sum, const FrameTrack& frame) { return sum += (frame.first.isNonnull() ? 1 : 0); }; + auto validFrameS = [](int sum, const FrameStub& frame) { return sum += (frame.first.isNonnull() ? 1 : 0); }; auto putT = [](const vector& objects, StreamTrack& stream) { auto toFrame = [](TrackKF* object) { return object ? object->frame() : FrameTrack(); }; stream.reserve(objects.size()); diff --git a/L1Trigger/TrackerTFP/src/CleanTrackBuilder.cc b/L1Trigger/TrackerTFP/src/CleanTrackBuilder.cc index a4ab33d38bf34..11610e8468467 100644 --- a/L1Trigger/TrackerTFP/src/CleanTrackBuilder.cc +++ b/L1Trigger/TrackerTFP/src/CleanTrackBuilder.cc @@ -260,7 +260,7 @@ namespace trackerTFP { for (Stub* stub : stubs_) stub->update(hitsPhi_, hitsZ_, stubIds, setup->ctbMaxStubs()); const int nLayer = - accumulate(stubIds.begin(), stubIds.end(), 0, [](int& sum, int i) { return sum += (i > 0 ? 1 : 0); }); + accumulate(stubIds.begin(), stubIds.end(), 0, [](int sum, int i) { return sum += (i > 0 ? 1 : 0); }); if (nLayer < setup->ctbMinLayers()) valid_ = false; size_ = *max_element(stubIds.begin(), stubIds.end()); @@ -375,7 +375,7 @@ namespace trackerTFP { } // add all gaps const int size = - accumulate(tracks.begin(), tracks.end(), 0, [](int& sum, Track* track) { return sum += track->size_; }); + accumulate(tracks.begin(), tracks.end(), 0, [](int sum, Track* track) { return sum += track->size_; }); for (int frame = 0; frame < size;) { const int trackId = tracks[frame]->trackId_; const int length = tracks[frame]->size_; @@ -460,7 +460,7 @@ namespace trackerTFP { double chi2phi(0.); double chi2z(0.); const int nStubs = accumulate( - stubs.begin(), stubs.end(), 0, [](int& sum, const vector& layer) { return sum += layer.size(); }); + stubs.begin(), stubs.end(), 0, [](int sum, const vector& layer) { return sum += layer.size(); }); vector ttStubRefs; ttStubRefs.reserve(nStubs); for (int layer = 0; layer < setup_->numLayers(); layer++) { diff --git a/L1Trigger/TrackerTFP/src/DuplicateRemoval.cc b/L1Trigger/TrackerTFP/src/DuplicateRemoval.cc index e7393c3fe0ac4..14cec5751399e 100644 --- a/L1Trigger/TrackerTFP/src/DuplicateRemoval.cc +++ b/L1Trigger/TrackerTFP/src/DuplicateRemoval.cc @@ -39,7 +39,7 @@ namespace trackerTFP { int nTracks(0); for (const vector& tracks : tracksIn) nTracks += - accumulate(tracks.begin(), tracks.end(), 0, [](int& sum, TrackKF* track) { return sum += track ? 1 : 0; }); + accumulate(tracks.begin(), tracks.end(), 0, [](int sum, TrackKF* track) { return sum += track ? 1 : 0; }); vector tracks; tracks.reserve(nTracks); deque stream; diff --git a/L1Trigger/TrackerTFP/src/HoughTransform.cc b/L1Trigger/TrackerTFP/src/HoughTransform.cc index eb9c45801dd58..cb90f0c0c694c 100644 --- a/L1Trigger/TrackerTFP/src/HoughTransform.cc +++ b/L1Trigger/TrackerTFP/src/HoughTransform.cc @@ -36,7 +36,7 @@ namespace trackerTFP { static const int chan = setup_->kfNumWorker(); static const int mux = numChannelOut / chan; // count and reserve ht stubs - auto multiplicity = [](int& sum, StubGP* s) { return sum += s ? 1 + s->inv2RMax() - s->inv2RMin() : 0; }; + auto multiplicity = [](int sum, StubGP* s) { return sum += s ? 1 + s->inv2RMax() - s->inv2RMin() : 0; }; int nStubs(0); for (const vector& input : streamsIn) nStubs += accumulate(input.begin(), input.end(), 0, multiplicity); diff --git a/L1Trigger/TrackerTFP/src/KalmanFilter.cc b/L1Trigger/TrackerTFP/src/KalmanFilter.cc index 68da57d8d2483..0a55934eff361 100644 --- a/L1Trigger/TrackerTFP/src/KalmanFilter.cc +++ b/L1Trigger/TrackerTFP/src/KalmanFilter.cc @@ -53,7 +53,7 @@ namespace trackerTFP { addLayer(stream); // count total number of final states const int nStates = - accumulate(stream.begin(), stream.end(), 0, [](int& sum, State* state) { return sum += (state ? 1 : 0); }); + accumulate(stream.begin(), stream.end(), 0, [](int sum, State* state) { return sum += (state ? 1 : 0); }); // apply truncation if (enableTruncation_ && (int)stream.size() > setup_->numFramesHigh()) stream.resize(setup_->numFramesHigh()); diff --git a/L1Trigger/TrackerTFP/src/TrackFindingProcessor.cc b/L1Trigger/TrackerTFP/src/TrackFindingProcessor.cc index c278009b8c447..4e99ea42035e5 100644 --- a/L1Trigger/TrackerTFP/src/TrackFindingProcessor.cc +++ b/L1Trigger/TrackerTFP/src/TrackFindingProcessor.cc @@ -127,7 +127,7 @@ namespace trackerTFP { vector>& outputs) { // count input objects int nTracks(0); - auto valid = [](int& sum, const FrameTrack& frame) { return sum += (frame.first.isNonnull() ? 1 : 0); }; + auto valid = [](int sum, const FrameTrack& frame) { return sum += (frame.first.isNonnull() ? 1 : 0); }; for (const StreamTrack& tracks : inputs) nTracks += accumulate(tracks.begin(), tracks.end(), 0, valid); tracks_.reserve(nTracks); diff --git a/L1Trigger/TrackerTFP/test/AnalyzerCTB.cc b/L1Trigger/TrackerTFP/test/AnalyzerCTB.cc index 12c8050bdebfb..c9c81bd842b12 100644 --- a/L1Trigger/TrackerTFP/test/AnalyzerCTB.cc +++ b/L1Trigger/TrackerTFP/test/AnalyzerCTB.cc @@ -217,7 +217,7 @@ namespace trackerTFP { const int offsetStub = indexTrack * setup_->numLayers(); for (int layer = 0; layer < setup_->numLayers(); layer++) { const StreamStub& stream = acceptedStubs[offsetStub + layer]; - const int nStubs = accumulate(stream.begin(), stream.end(), 0, [](int& sum, const FrameStub& frame) { + const int nStubs = accumulate(stream.begin(), stream.end(), 0, [](int sum, const FrameStub& frame) { return sum += (frame.first.isNonnull() ? 1 : 0); }); hisStubs_->Fill(nStubs); diff --git a/L1Trigger/TrackerTFP/test/AnalyzerDR.cc b/L1Trigger/TrackerTFP/test/AnalyzerDR.cc index 7c7594ec0cead..29fe8c64f3088 100644 --- a/L1Trigger/TrackerTFP/test/AnalyzerDR.cc +++ b/L1Trigger/TrackerTFP/test/AnalyzerDR.cc @@ -179,7 +179,7 @@ namespace trackerTFP { hisTracks_->Fill(tracks.size()); profTracks_->Fill(channel, tracks.size()); nTracks += tracks.size(); - nStubs += accumulate(tracks.begin(), tracks.end(), 0, [](int& sum, const vector& track) { + nStubs += accumulate(tracks.begin(), tracks.end(), 0, [](int sum, const vector& track) { return sum += (int)track.size(); }); allTracks += tracks.size(); @@ -248,7 +248,7 @@ namespace trackerTFP { int channel) const { const int offset = channel * setup_->numLayers(); const StreamTrack& streamTrack = streamsTrack[channel]; - const int numTracks = accumulate(streamTrack.begin(), streamTrack.end(), 0, [](int& sum, const FrameTrack& frame) { + const int numTracks = accumulate(streamTrack.begin(), streamTrack.end(), 0, [](int sum, const FrameTrack& frame) { return sum += (frame.first.isNonnull() ? 1 : 0); }); tracks.reserve(numTracks); diff --git a/SimTracker/TrackTriggerAssociation/interface/StubAssociation.h b/SimTracker/TrackTriggerAssociation/interface/StubAssociation.h index b32343fc57e86..1b29b3a946a10 100644 --- a/SimTracker/TrackTriggerAssociation/interface/StubAssociation.h +++ b/SimTracker/TrackTriggerAssociation/interface/StubAssociation.h @@ -2,7 +2,8 @@ #define SimTracker_TrackTriggerAssociation_StubAssociation_h #include "FWCore/ParameterSet/interface/ParameterSet.h" -#include "SimTracker/TrackTriggerAssociation/interface/TTTypes.h" +#include "DataFormats/Common/interface/Ptr.h" +#include "SimDataFormats/TrackingAnalysis/interface/TrackingParticle.h" #include "L1Trigger/TrackTrigger/interface/Setup.h" #include @@ -10,6 +11,8 @@ namespace tt { + typedef edm::Ptr TPPtr; + /*! \class tt::StubAssociation * \brief Class to associate reconstrucable TrackingParticles with TTStubs and vice versa. * It may associate multiple TPs with a TTStub and can therefore be used to associate diff --git a/SimTracker/TrackTriggerAssociation/plugins/StubAssociator.cc b/SimTracker/TrackTriggerAssociation/plugins/StubAssociator.cc index f68d8b66b1c44..5c06c77956b4c 100644 --- a/SimTracker/TrackTriggerAssociation/plugins/StubAssociator.cc +++ b/SimTracker/TrackTriggerAssociation/plugins/StubAssociator.cc @@ -9,7 +9,6 @@ #include "FWCore/Utilities/interface/EDPutToken.h" #include "DataFormats/Common/interface/Handle.h" -#include "SimTracker/TrackTriggerAssociation/interface/TTTypes.h" #include "SimTracker/TrackTriggerAssociation/interface/StubAssociation.h" #include "L1Trigger/TrackTrigger/interface/Setup.h" From 8e43955a2cbab84d3bf8b8f7b3d81a1a11ce7145 Mon Sep 17 00:00:00 2001 From: Thomas Schuh Date: Fri, 22 Nov 2024 16:35:45 +0000 Subject: [PATCH 05/14] code-format --- L1Trigger/TrackerTFP/interface/DataFormats.h | 3 ++- L1Trigger/TrackerTFP/interface/KalmanFilter.h | 4 +--- L1Trigger/TrackerTFP/plugins/ProducerCTB.cc | 2 +- L1Trigger/TrackerTFP/src/CleanTrackBuilder.cc | 6 +++--- L1Trigger/TrackerTFP/src/HoughTransform.cc | 5 +++-- L1Trigger/TrackerTFP/src/KalmanFilter.cc | 17 +++++++++++++++-- L1Trigger/TrackerTFP/src/KalmanFilterFormats.cc | 2 +- L1Trigger/TrackerTFP/src/State.cc | 2 +- 8 files changed, 27 insertions(+), 14 deletions(-) diff --git a/L1Trigger/TrackerTFP/interface/DataFormats.h b/L1Trigger/TrackerTFP/interface/DataFormats.h index 86ed628598515..5efe028546e24 100644 --- a/L1Trigger/TrackerTFP/interface/DataFormats.h +++ b/L1Trigger/TrackerTFP/interface/DataFormats.h @@ -44,7 +44,8 @@ namespace trackerTFP { public: DataFormat() {} DataFormat(bool twos, bool biased = true) : twos_(twos), width_(0), base_(1.), range_(0.) {} - DataFormat(bool twos, int width, double base, double range) : twos_(twos), width_(width), base_(base), range_(range) {} + DataFormat(bool twos, int width, double base, double range) + : twos_(twos), width_(width), base_(base), range_(range) {} ~DataFormat() {} // converts int to bitvector TTBV ttBV(int i) const { return TTBV(i, width_, twos_); } diff --git a/L1Trigger/TrackerTFP/interface/KalmanFilter.h b/L1Trigger/TrackerTFP/interface/KalmanFilter.h index 445b1342dd6cc..8d148ce2515f8 100644 --- a/L1Trigger/TrackerTFP/interface/KalmanFilter.h +++ b/L1Trigger/TrackerTFP/interface/KalmanFilter.h @@ -111,9 +111,7 @@ namespace trackerTFP { // apply final cuts void finalize(const std::deque& stream, std::vector& finals); // Transform States into Tracks - void conv(const std::vector& best, - std::vector& tracks, - std::vector>& stubs); + void conv(const std::vector& best, std::vector& tracks, std::vector>& stubs); // adds a layer to states void addLayer(std::deque& stream); // adds a layer to states to build seeds diff --git a/L1Trigger/TrackerTFP/plugins/ProducerCTB.cc b/L1Trigger/TrackerTFP/plugins/ProducerCTB.cc index 5e0aacd4fdaf1..0f866014e1a4b 100644 --- a/L1Trigger/TrackerTFP/plugins/ProducerCTB.cc +++ b/L1Trigger/TrackerTFP/plugins/ProducerCTB.cc @@ -98,7 +98,7 @@ namespace trackerTFP { layerEncoding_ = &iSetup.getData(esGetTokenLayerEncoding_); // const double baseZ = dataFormats_->base(Variable::z, Process::ctb); - const double baseR = dataFormats_->base(Variable::r, Process::ctb); + const double baseR = dataFormats_->base(Variable::r, Process::ctb); const double range = dataFormats_->range(Variable::cot, Process::kf); const int baseShift = ceil(log2(range / baseZ * baseR / setup_->ctbNumBinsCot())); const int width = ceil(log2(setup_->ctbNumBinsCot())); diff --git a/L1Trigger/TrackerTFP/src/CleanTrackBuilder.cc b/L1Trigger/TrackerTFP/src/CleanTrackBuilder.cc index 11610e8468467..e798decedcd31 100644 --- a/L1Trigger/TrackerTFP/src/CleanTrackBuilder.cc +++ b/L1Trigger/TrackerTFP/src/CleanTrackBuilder.cc @@ -42,13 +42,13 @@ namespace trackerTFP { // loop over worker for (int channelOut = 0; channelOut < numChannelOut; channelOut++) { //if (channelOut != 3) - //continue; + //continue; // clean input tracks vector> streamsT(numChannel); vector> streamsS(numChannel); for (int cin = 0; cin < numChannel; cin++) { //if (cin != 1) - //continue; + //continue; const int index = numChannel * cin + channelOut; cleanStream(streamsIn[index], streamsT[cin], streamsS[cin], index); } @@ -135,7 +135,7 @@ namespace trackerTFP { for (int layer = 0; layer < setup_->numLayers(); layer++) { if (pattern.test(layer)) { doubleGap = false; - if(++nHits == setup_->ctbMinLayers()) + if (++nHits == setup_->ctbMinLayers()) return false; } else if (!maybePattern.test(layer)) { if (++nGaps == setup_->kfMaxGaps() || doubleGap) diff --git a/L1Trigger/TrackerTFP/src/HoughTransform.cc b/L1Trigger/TrackerTFP/src/HoughTransform.cc index cb90f0c0c694c..932503f20d7b6 100644 --- a/L1Trigger/TrackerTFP/src/HoughTransform.cc +++ b/L1Trigger/TrackerTFP/src/HoughTransform.cc @@ -188,7 +188,8 @@ namespace trackerTFP { if (pattern.count(0, setup_->kfMaxSeedingLayer()) < 2) return true; // check min layers req - const int minLayers = setup_->htMinLayers() - (((zT == -4 || zT == 3) && (!pattern.test(5) && !pattern.test(7))) ? 1 : 0); + const int minLayers = + setup_->htMinLayers() - (((zT == -4 || zT == 3) && (!pattern.test(5) && !pattern.test(7))) ? 1 : 0); // prepare pattern analysis const TTBV& maybePattern = layerEncoding_->maybePattern(zT); int nHits(0); @@ -197,7 +198,7 @@ namespace trackerTFP { for (int layer = 0; layer < setup_->numLayers(); layer++) { if (pattern.test(layer)) { doubleGap = false; - if(++nHits == minLayers) + if (++nHits == minLayers) return false; } else if (!maybePattern.test(layer)) { if (++nGaps == setup_->kfMaxGaps() || doubleGap) diff --git a/L1Trigger/TrackerTFP/src/KalmanFilter.cc b/L1Trigger/TrackerTFP/src/KalmanFilter.cc index 0a55934eff361..853da1da4f123 100644 --- a/L1Trigger/TrackerTFP/src/KalmanFilter.cc +++ b/L1Trigger/TrackerTFP/src/KalmanFilter.cc @@ -257,7 +257,20 @@ namespace trackerTFP { if (invalidLayers || !validX0 || !validX1 || !validX2 || !validX3) continue; const int trackId = state->trackId(); - finals_.emplace_back(trackId, numConsistent, numConsistentPS, inv2R, phiT, cot, zT, chi20, chi21, hitPattern, track, stubs, phis, zs); + finals_.emplace_back(trackId, + numConsistent, + numConsistentPS, + inv2R, + phiT, + cot, + zT, + chi20, + chi21, + hitPattern, + track, + stubs, + phis, + zs); } } @@ -356,7 +369,7 @@ namespace trackerTFP { void KalmanFilter::accumulator(vector& finals, vector& best) { // create container of pointer to make sorts less CPU intense //for (Track& track : finals) - //best.push_back(&track); + //best.push_back(&track); transform(finals.begin(), finals.end(), back_inserter(best), [](Track& track) { return &track; }); // prepare arrival order vector trackIds; diff --git a/L1Trigger/TrackerTFP/src/KalmanFilterFormats.cc b/L1Trigger/TrackerTFP/src/KalmanFilterFormats.cc index 749e692d2d614..ee765f153c8d2 100644 --- a/L1Trigger/TrackerTFP/src/KalmanFilterFormats.cc +++ b/L1Trigger/TrackerTFP/src/KalmanFilterFormats.cc @@ -21,7 +21,7 @@ namespace trackerTFP { "S00Shifted", "S01Shifted", "S12Shifted", "S13Shifted", "K00", "K10", "K21", "K31", "R00", "R11", "R00Rough", "R11Rough", "invR00Approx", "invR11Approx", "invR00Cor", "invR11Cor", "invR00", "invR11", "C00", "C01", "C11", "C22", "C23", "C33", - "r0Shifted", "r1Shifted", "r02", "r12", "chi20", "chi21"}; + "r0Shifted", "r1Shifted", "r02", "r12", "chi20", "chi21"}; void KalmanFilterFormats::endJob() { const int wName = diff --git a/L1Trigger/TrackerTFP/src/State.cc b/L1Trigger/TrackerTFP/src/State.cc index c9d7a43ea2577..11ff8b0bef933 100644 --- a/L1Trigger/TrackerTFP/src/State.cc +++ b/L1Trigger/TrackerTFP/src/State.cc @@ -150,7 +150,7 @@ namespace trackerTFP { if (++hits >= setup_->kfMinLayers() && k >= layer) return true; } else if (!maybePattern_[k]) { - if (gap || ++ gaps > setup_->kfMaxGaps()) + if (gap || ++gaps > setup_->kfMaxGaps()) return false; gap = true; } From c89124e778d414484ae8cf00dd8b5aba0c5c8f19 Mon Sep 17 00:00:00 2001 From: Thomas Schuh Date: Fri, 29 Nov 2024 12:59:12 +0000 Subject: [PATCH 06/14] run scripts update. --- .../TrackFindingTracklet/test/HybridTracksNewKF_cfg.py | 4 ++-- L1Trigger/TrackFindingTracklet/test/HybridTracks_cfg.py | 4 ++-- L1Trigger/TrackerTFP/test/demonstrator_cfg.py | 4 ++-- L1Trigger/TrackerTFP/test/test_cfg.py | 9 ++++++--- 4 files changed, 12 insertions(+), 9 deletions(-) diff --git a/L1Trigger/TrackFindingTracklet/test/HybridTracksNewKF_cfg.py b/L1Trigger/TrackFindingTracklet/test/HybridTracksNewKF_cfg.py index f60550c3ada70..6055dadd0061d 100644 --- a/L1Trigger/TrackFindingTracklet/test/HybridTracksNewKF_cfg.py +++ b/L1Trigger/TrackFindingTracklet/test/HybridTracksNewKF_cfg.py @@ -12,8 +12,8 @@ process = cms.Process( "Demo" ) process.load( 'FWCore.MessageService.MessageLogger_cfi' ) process.load( 'Configuration.EventContent.EventContent_cff' ) -process.load( 'Configuration.Geometry.GeometryExtended2026D98Reco_cff' ) -process.load( 'Configuration.Geometry.GeometryExtended2026D98_cff' ) +process.load( 'Configuration.Geometry.GeometryExtendedRun4D98Reco_cff' ) +process.load( 'Configuration.Geometry.GeometryExtendedRun4D98_cff' ) process.load( 'Configuration.StandardSequences.MagneticField_cff' ) process.load( 'Configuration.StandardSequences.FrontierConditions_GlobalTag_cff' ) process.load( 'L1Trigger.TrackTrigger.TrackTrigger_cff' ) diff --git a/L1Trigger/TrackFindingTracklet/test/HybridTracks_cfg.py b/L1Trigger/TrackFindingTracklet/test/HybridTracks_cfg.py index f794985786064..1a6726a9cd36a 100644 --- a/L1Trigger/TrackFindingTracklet/test/HybridTracks_cfg.py +++ b/L1Trigger/TrackFindingTracklet/test/HybridTracks_cfg.py @@ -15,8 +15,8 @@ process.load('Configuration.EventContent.EventContent_cff') process.load('Configuration.StandardSequences.MagneticField_cff') -process.load('Configuration.Geometry.GeometryExtended2026D98Reco_cff') -process.load('Configuration.Geometry.GeometryExtended2026D98_cff') +process.load( 'Configuration.Geometry.GeometryExtendedRun4D98Reco_cff' ) +process.load( 'Configuration.Geometry.GeometryExtendedRun4D98_cff' ) process.load('Configuration.StandardSequences.EndOfProcess_cff') process.load('Configuration.StandardSequences.FrontierConditions_GlobalTag_cff') diff --git a/L1Trigger/TrackerTFP/test/demonstrator_cfg.py b/L1Trigger/TrackerTFP/test/demonstrator_cfg.py index 367c617f78dba..5509a1c1f5244 100644 --- a/L1Trigger/TrackerTFP/test/demonstrator_cfg.py +++ b/L1Trigger/TrackerTFP/test/demonstrator_cfg.py @@ -3,8 +3,8 @@ process = cms.Process( "Demo" ) process.load( 'FWCore.MessageService.MessageLogger_cfi' ) -process.load( 'Configuration.Geometry.GeometryExtended2026D98Reco_cff' ) -process.load( 'Configuration.Geometry.GeometryExtended2026D98_cff' ) +process.load( 'Configuration.Geometry.GeometryExtendedRun4D98Reco_cff' ) +process.load( 'Configuration.Geometry.GeometryExtendedRun4D98_cff' ) process.load( 'Configuration.StandardSequences.MagneticField_cff' ) process.load( 'Configuration.StandardSequences.FrontierConditions_GlobalTag_cff' ) process.load( 'L1Trigger.TrackTrigger.TrackTrigger_cff' ) diff --git a/L1Trigger/TrackerTFP/test/test_cfg.py b/L1Trigger/TrackerTFP/test/test_cfg.py index 2529c53e6d895..e4d3ee70d0710 100644 --- a/L1Trigger/TrackerTFP/test/test_cfg.py +++ b/L1Trigger/TrackerTFP/test/test_cfg.py @@ -10,14 +10,17 @@ process = cms.Process( "Demo" ) process.load( 'FWCore.MessageService.MessageLogger_cfi' ) -process.load( 'Configuration.Geometry.GeometryExtended2026D98Reco_cff' ) -process.load( 'Configuration.Geometry.GeometryExtended2026D98_cff' ) +process.load( 'Configuration.Geometry.GeometryExtendedRun4D98Reco_cff' ) +process.load( 'Configuration.Geometry.GeometryExtendedRun4D98_cff' ) process.load( 'Configuration.StandardSequences.MagneticField_cff' ) +process.load( 'Configuration.StandardSequences.Services_cff' ) +process.load( 'Configuration.EventContent.EventContent_cff' ) +process.load( 'Configuration.StandardSequences.EndOfProcess_cff' ) process.load( 'Configuration.StandardSequences.FrontierConditions_GlobalTag_cff' ) process.load( 'L1Trigger.TrackTrigger.TrackTrigger_cff' ) from Configuration.AlCa.GlobalTag import GlobalTag -process.GlobalTag = GlobalTag(process.GlobalTag, 'auto:phase2_realistic', '') +process.GlobalTag = GlobalTag(process.GlobalTag, '133X_mcRun4_realistic_v1', '') # load code that associates stubs with mctruth process.load( 'SimTracker.TrackTriggerAssociation.StubAssociator_cff' ) From 2db59a77a1171e5fa59e0602a7d00fcd4061d8db Mon Sep 17 00:00:00 2001 From: Thomas Schuh Date: Fri, 29 Nov 2024 17:32:35 +0000 Subject: [PATCH 07/14] run script clean up. --- L1Trigger/TrackFindingTMTT/src/Settings.cc | 4 ++-- L1Trigger/TrackFindingTracklet/python/Customize_cff.py | 8 +++----- L1Trigger/TrackFindingTracklet/python/Producer_cff.py | 1 - L1Trigger/TrackFindingTracklet/test/AnalyzerTM.cc | 2 ++ .../TrackFindingTracklet/test/L1TrackNtupleMaker_cfg.py | 2 -- L1Trigger/TrackTrigger/python/Setup_cfi.py | 4 ++-- 6 files changed, 9 insertions(+), 12 deletions(-) diff --git a/L1Trigger/TrackFindingTMTT/src/Settings.cc b/L1Trigger/TrackFindingTMTT/src/Settings.cc index d1a246cb51e74..2a782e407840b 100644 --- a/L1Trigger/TrackFindingTMTT/src/Settings.cc +++ b/L1Trigger/TrackFindingTMTT/src/Settings.cc @@ -38,8 +38,8 @@ namespace tmtt { stubMatchStrict_(false), // Kalman filter track fit cfg - //kalmanDebugLevel_(0), - kalmanDebugLevel_(2), // Good for debugging + kalmanDebugLevel_(0), + //kalmanDebugLevel_(2), // Good for debugging kalmanMinNumStubs_(4), kalmanMaxNumStubs_(6), kalmanAddBeamConstr_(false), // Apply post-fit beam-spot constraint to 5-param fit diff --git a/L1Trigger/TrackFindingTracklet/python/Customize_cff.py b/L1Trigger/TrackFindingTracklet/python/Customize_cff.py index 430e52bea0ba7..921055a241081 100644 --- a/L1Trigger/TrackFindingTracklet/python/Customize_cff.py +++ b/L1Trigger/TrackFindingTracklet/python/Customize_cff.py @@ -2,13 +2,11 @@ import FWCore.ParameterSet.Config as cms -# configures track finding s/w to use KF emulator instead of KF simulator -def newKFConfig(process): - process.l1tTTTracksFromTrackletEmulation.Fakefit = True - # configures track finding s/w to behave as track finding f/w def fwConfig(process): - newKFConfig(process) + process.l1tTTTracksFromTrackletEmulation.Fakefit = True + process.TrackTriggerSetup.TrackFinding.MaxEta = 2.5 + process.TrackTriggerSetup.GeometricProcessor.ChosenRofZ = 57.76 process.l1tTTTracksFromTrackletEmulation.RemovalType = "" process.l1tTTTracksFromTrackletEmulation.DoMultipleMatches = False process.l1tTTTracksFromTrackletEmulation.StoreTrackBuilderOutput = True diff --git a/L1Trigger/TrackFindingTracklet/python/Producer_cff.py b/L1Trigger/TrackFindingTracklet/python/Producer_cff.py index f51299b301958..53af5b206b8eb 100644 --- a/L1Trigger/TrackFindingTracklet/python/Producer_cff.py +++ b/L1Trigger/TrackFindingTracklet/python/Producer_cff.py @@ -4,7 +4,6 @@ from L1Trigger.TrackTrigger.Setup_cff import TrackTriggerSetup from L1Trigger.TrackerTFP.LayerEncoding_cff import TrackTriggerLayerEncoding -from L1Trigger.TrackerTFP.KalmanFilterFormats_cff import TrackTriggerKalmanFilterFormats from L1Trigger.TrackerTFP.TrackQuality_cff import * from L1Trigger.TrackFindingTracklet.ChannelAssignment_cff import ChannelAssignment from L1Trigger.TrackFindingTracklet.DataFormats_cff import * diff --git a/L1Trigger/TrackFindingTracklet/test/AnalyzerTM.cc b/L1Trigger/TrackFindingTracklet/test/AnalyzerTM.cc index 2afa92a8386e6..00931d48591c4 100644 --- a/L1Trigger/TrackFindingTracklet/test/AnalyzerTM.cc +++ b/L1Trigger/TrackFindingTracklet/test/AnalyzerTM.cc @@ -286,3 +286,5 @@ namespace trklet { } } // namespace trklet + +DEFINE_FWK_MODULE(trklet::AnalyzerTM); \ No newline at end of file diff --git a/L1Trigger/TrackFindingTracklet/test/L1TrackNtupleMaker_cfg.py b/L1Trigger/TrackFindingTracklet/test/L1TrackNtupleMaker_cfg.py index dabf98787df05..5490521df5cd6 100644 --- a/L1Trigger/TrackFindingTracklet/test/L1TrackNtupleMaker_cfg.py +++ b/L1Trigger/TrackFindingTracklet/test/L1TrackNtupleMaker_cfg.py @@ -151,8 +151,6 @@ # HYBRID: prompt tracking if (L1TRKALGO == 'HYBRID'): - process.TrackTriggerSetup.GeometricProcessor.ChosenRofZ = 50.0 - process.TrackTriggerSetup.TrackFinding.MaxEta = 2.4 process.TTTracksEmulation = cms.Path(process.L1THybridTracks) process.TTTracksEmulationWithTruth = cms.Path(process.L1THybridTracksWithAssociators) NHELIXPAR = 4 diff --git a/L1Trigger/TrackTrigger/python/Setup_cfi.py b/L1Trigger/TrackTrigger/python/Setup_cfi.py index b070040786aa6..5fc3c2288a8ad 100644 --- a/L1Trigger/TrackTrigger/python/Setup_cfi.py +++ b/L1Trigger/TrackTrigger/python/Setup_cfi.py @@ -17,7 +17,7 @@ MinLayers = cms.int32 ( 4 ), # required number of stub layers to form a track MinPt = cms.double( 2.0 ), # min track pt in GeV, also defines region overlap shape MinPtCand = cms.double( 1.34 ), # min candiate pt in GeV - MaxEta = cms.double( 2.5 ), # cut on stub eta + MaxEta = cms.double( 2.4 ), # cut on stub eta MaxD0 = cms.double( 5.0 ), # in cm, constraints track reconstruction phase space ChosenRofPhi = cms.double( 55. ), # critical radius defining region overlap shape in cm ), @@ -140,7 +140,7 @@ GeometricProcessor = cms.PSet ( NumBinsPhiT = cms.int32 ( 2 ), # number of phi sectors used in hough transform NumBinsZT = cms.int32 ( 32 ), # number of eta sectors used in hough transform - ChosenRofZ = cms.double( 57.76 ), # critical radius defining r-z sector shape in cm + ChosenRofZ = cms.double( 50 ), # critical radius defining r-z sector shape in cm DepthMemory = cms.int32 ( 32 ), # fifo depth in stub router firmware WidthModule = cms.int32 ( 3 ), # PosPS = cms.int32 ( 2 ), # From 82090610d7bee99f4c463bb7013ce133b05a9ba9 Mon Sep 17 00:00:00 2001 From: Thomas Schuh Date: Fri, 29 Nov 2024 17:44:41 +0000 Subject: [PATCH 08/14] rebase fix --- L1Trigger/TrackFindingTracklet/plugins/L1FPGATrackProducer.cc | 4 ---- 1 file changed, 4 deletions(-) diff --git a/L1Trigger/TrackFindingTracklet/plugins/L1FPGATrackProducer.cc b/L1Trigger/TrackFindingTracklet/plugins/L1FPGATrackProducer.cc index 88b79adc0b268..6cd0d241241ce 100644 --- a/L1Trigger/TrackFindingTracklet/plugins/L1FPGATrackProducer.cc +++ b/L1Trigger/TrackFindingTracklet/plugins/L1FPGATrackProducer.cc @@ -284,10 +284,6 @@ L1FPGATrackProducer::L1FPGATrackProducer(edm::ParameterSet const& iConfig) settings_.setExtended(extended_); settings_.setReduced(reduced_); settings_.setNHelixPar(nHelixPar_); - // combined_ must be false for extended tracking, regardless of whether - // combined modules are used or not. - if (extended_) - settings_.setCombined(false); #ifndef USEHYBRID fitPatternFile = iConfig.getParameter("fitPatternFile"); From 4a4b758a104fdc69a81e2f947a998fcd6a2d1894 Mon Sep 17 00:00:00 2001 From: Thomas Schuh Date: Wed, 4 Dec 2024 15:57:09 +0000 Subject: [PATCH 09/14] TrackFindingProcessor for hybrid adapted. --- .../interface/TrackFindingProcessor.h | 90 ++++++ .../plugins/ProducerTFP.cc | 121 ++++++++ .../python/Analyzer_cff.py | 3 +- .../python/Producer_cff.py | 2 +- .../TrackFindingTracklet/src/KalmanFilter.cc | 3 - .../src/TrackFindingProcessor.cc | 278 +++++++++++++++++ .../test/HybridTracksNewKF_cfg.py | 4 +- .../test/L1TrackNtupleMaker_cfg.py | 4 +- .../test/L1TrkNtuple.root | Bin 0 -> 316333 bytes L1Trigger/TrackerTFP/python/Analyzer_cff.py | 2 + L1Trigger/TrackerTFP/python/Analyzer_cfi.py | 2 + L1Trigger/TrackerTFP/src/HoughTransform.cc | 2 +- L1Trigger/TrackerTFP/test/AnalyzerCTB.cc | 1 - L1Trigger/TrackerTFP/test/AnalyzerDR.cc | 2 - L1Trigger/TrackerTFP/test/AnalyzerKF.cc | 1 - .../test/AnalyzerTFP.cc | 8 +- L1Trigger/TrackerTFP/test/AnalyzerTQ.cc | 284 ++++++++++++++++++ L1Trigger/TrackerTFP/test/test_cfg.py | 2 +- 18 files changed, 790 insertions(+), 19 deletions(-) create mode 100644 L1Trigger/TrackFindingTracklet/interface/TrackFindingProcessor.h create mode 100644 L1Trigger/TrackFindingTracklet/plugins/ProducerTFP.cc create mode 100644 L1Trigger/TrackFindingTracklet/src/TrackFindingProcessor.cc create mode 100644 L1Trigger/TrackFindingTracklet/test/L1TrkNtuple.root rename L1Trigger/{TrackFindingTracklet => TrackerTFP}/test/AnalyzerTFP.cc (98%) create mode 100644 L1Trigger/TrackerTFP/test/AnalyzerTQ.cc diff --git a/L1Trigger/TrackFindingTracklet/interface/TrackFindingProcessor.h b/L1Trigger/TrackFindingTracklet/interface/TrackFindingProcessor.h new file mode 100644 index 0000000000000..13b07dcdc8cb0 --- /dev/null +++ b/L1Trigger/TrackFindingTracklet/interface/TrackFindingProcessor.h @@ -0,0 +1,90 @@ +#ifndef L1Trigger_TrackFindingTracklet_TrackFindingProcessor_h +#define L1Trigger_TrackFindingTracklet_TrackFindingProcessor_h + +#include "FWCore/ParameterSet/interface/ParameterSet.h" +#include "L1Trigger/TrackTrigger/interface/Setup.h" +#include "L1Trigger/TrackFindingTracklet/interface/DataFormats.h" +#include "L1Trigger/TrackerTFP/interface/TrackQuality.h" +#include "DataFormats/L1TrackTrigger/interface/TTTypes.h" + +#include +#include +#include + +namespace trklet { + + // Class to format final tfp output and to prodcue final TTTrackCollection + class TrackFindingProcessor { + public: + TrackFindingProcessor(const edm::ParameterSet& iConfig, + const tt::Setup* setup_, + const DataFormats* dataFormats, + const trackerTFP::TrackQuality* trackQuality); + ~TrackFindingProcessor() {} + + // produce TTTracks + void produce(const tt::StreamsTrack& inputs, + const tt::StreamsStub& stubs, + tt::TTTracks& ttTracks, + tt::StreamsTrack& outputs); + // produce StreamsTrack + void produce(const std::vector& inputs, tt::StreamsTrack& outputs) const; + + private: + // + static constexpr int partial_width = 32; + // + static constexpr int partial_in = 3; + // + static constexpr int partial_out = 2; + // + typedef std::bitset PartialFrame; + // + typedef std::pair PartialFrameTrack; + // + struct Track { + Track(const tt::FrameTrack& frameTrack, + const tt::Frame& frameTQ, + const std::vector& ttStubRefs, + const DataFormats* df, + const trackerTFP::TrackQuality* tq); + const TTTrackRef& ttTrackRef_; + const std::vector ttStubRefs_; + bool valid_; + std::vector partials_; + TTBV hitPattern_; + int channel_; + int mva_; + double inv2R_; + double phiT_; + double cot_; + double zT_; + double chi2rphi_; + double chi2rz_; + }; + // remove and return first element of deque, returns nullptr if empty + template + T* pop_front(std::deque& ts) const; + // + void consume(const tt::StreamsTrack& inputs, + const tt::StreamsStub& stubs, + std::vector>& outputs); + // emualte data format f/w + void produce(std::vector>& inputs, tt::StreamsTrack& outputs) const; + // produce TTTracks + void produce(const tt::StreamsTrack& inputs, tt::TTTracks& ouputs) const; + // true if truncation is enbaled + bool enableTruncation_; + // provides run-time constants + const tt::Setup* setup_; + // provides data formats + const DataFormats* dataFormats_; + // provides Track Quality algo and formats + const trackerTFP::TrackQuality* trackQuality_; + // storage of tracks + std::vector tracks_; + }; + +} // namespace trklet + +#endif \ No newline at end of file diff --git a/L1Trigger/TrackFindingTracklet/plugins/ProducerTFP.cc b/L1Trigger/TrackFindingTracklet/plugins/ProducerTFP.cc new file mode 100644 index 0000000000000..2c68231a201aa --- /dev/null +++ b/L1Trigger/TrackFindingTracklet/plugins/ProducerTFP.cc @@ -0,0 +1,121 @@ +#include "FWCore/Framework/interface/stream/EDProducer.h" +#include "FWCore/Framework/interface/Run.h" +#include "FWCore/Framework/interface/EventSetup.h" +#include "FWCore/Framework/interface/Event.h" +#include "FWCore/Framework/interface/MakerMacros.h" +#include "FWCore/Utilities/interface/EDGetToken.h" +#include "FWCore/Utilities/interface/EDPutToken.h" +#include "FWCore/Utilities/interface/ESGetToken.h" +#include "FWCore/Utilities/interface/InputTag.h" +#include "FWCore/ParameterSet/interface/ParameterSet.h" +#include "DataFormats/Common/interface/Handle.h" +#include "DataFormats/Common/interface/OrphanHandle.h" + +#include "DataFormats/L1TrackTrigger/interface/TTTypes.h" +#include "L1Trigger/TrackTrigger/interface/Setup.h" +#include "L1Trigger/TrackFindingTracklet/interface/DataFormats.h" +#include "L1Trigger/TrackerTFP/interface/TrackQuality.h" +#include "L1Trigger/TrackFindingTracklet/interface/TrackFindingProcessor.h" + +#include +#include +#include +#include + +using namespace std; +using namespace edm; +using namespace tt; +using namespace trackerTFP; + +namespace trklet { + + /*! \class trklet::ProducerTFP + * \brief L1TrackTrigger final TFP output formatter + * \author Thomas Schuh + * \date 2023, June + */ + class ProducerTFP : public stream::EDProducer<> { + public: + explicit ProducerTFP(const ParameterSet&); + ~ProducerTFP() override {} + + private: + void beginRun(const Run&, const EventSetup&) override; + void produce(Event&, const EventSetup&) override; + void endStream() override {} + // ED input token of stubs and tracks + EDGetTokenT edGetTokenTracks_; + EDGetTokenT edGetTokenStubs_; + // ED output token for accepted stubs and tracks + EDPutTokenT edPutTokenTTTracks_; + EDPutTokenT edPutTokenTracks_; + // Setup token + ESGetToken esGetTokenSetup_; + // DataFormats token + ESGetToken esGetTokenDataFormats_; + // TrackQuality token + ESGetToken esGetTokenTrackQuality_; + // configuration + ParameterSet iConfig_; + // helper class to store configurations + const Setup* setup_ = nullptr; + // helper class to extract structured data from tt::Frames + const DataFormats* dataFormats_ = nullptr; + // helper class to determine track quality + const TrackQuality* trackQuality_ = nullptr; + }; + + ProducerTFP::ProducerTFP(const ParameterSet& iConfig) : iConfig_(iConfig) { + const string& labelTracks = iConfig.getParameter("InputLabelTFP"); + const string& labelStubs = iConfig.getParameter("InputLabelTQ"); + const string& branchTracks = iConfig.getParameter("BranchTracks"); + const string& branchTTTracks = iConfig.getParameter("BranchTTTracks"); + const string& branchStubs = iConfig.getParameter("BranchStubs"); + // book in- and output ED products + edGetTokenTracks_ = consumes(InputTag(labelTracks, branchTracks)); + edGetTokenStubs_ = consumes(InputTag(labelStubs, branchStubs)); + edPutTokenTTTracks_ = produces(branchTTTracks); + edPutTokenTracks_ = produces(branchTracks); + // book ES products + esGetTokenSetup_ = esConsumes(); + esGetTokenDataFormats_ = esConsumes(); + esGetTokenTrackQuality_ = esConsumes(); + } + + void ProducerTFP::beginRun(const Run& iRun, const EventSetup& iSetup) { + // helper class to store configurations + setup_ = &iSetup.getData(esGetTokenSetup_); + // helper class to extract structured data from tt::Frames + dataFormats_ = &iSetup.getData(esGetTokenDataFormats_); + // helper class to determine track quality + trackQuality_ = &iSetup.getData(esGetTokenTrackQuality_); + } + + void ProducerTFP::produce(Event& iEvent, const EventSetup& iSetup) { + // empty TFP products + TTTracks ttTracks; + StreamsTrack streamsTrack(setup_->numRegions() * setup_->tfpNumChannel()); + // read in TQ Products + Handle handleTracks; + iEvent.getByToken(edGetTokenTracks_, handleTracks); + Handle handleStubs; + iEvent.getByToken(edGetTokenStubs_, handleStubs); + // produce TTTracks + TrackFindingProcessor tfp(iConfig_, setup_, dataFormats_, trackQuality_); + tfp.produce(*handleTracks, *handleStubs, ttTracks, streamsTrack); + // put TTTRacks and produce TTTRackRefs + const int nTrks = ttTracks.size(); + const OrphanHandle oh = iEvent.emplace(edPutTokenTTTracks_, move(ttTracks)); + vector ttTrackRefs; + ttTrackRefs.reserve(nTrks); + for (int iTrk = 0; iTrk < nTrks; iTrk++) + ttTrackRefs.emplace_back(oh, iTrk); + // replace old TTTrackRefs in streamsTrack with new TTTrackRefs + tfp.produce(ttTrackRefs, streamsTrack); + // put StreamsTrack + iEvent.emplace(edPutTokenTracks_, move(streamsTrack)); + } + +} // namespace trklet + +DEFINE_FWK_MODULE(trklet::ProducerTFP); \ No newline at end of file diff --git a/L1Trigger/TrackFindingTracklet/python/Analyzer_cff.py b/L1Trigger/TrackFindingTracklet/python/Analyzer_cff.py index f38d6311b0a57..6324e4684a24e 100644 --- a/L1Trigger/TrackFindingTracklet/python/Analyzer_cff.py +++ b/L1Trigger/TrackFindingTracklet/python/Analyzer_cff.py @@ -9,4 +9,5 @@ AnalyzerTM = cms.EDAnalyzer( 'trklet::AnalyzerTM', TrackFindingTrackletAnalyzer_params, TrackFindingTrackletProducer_params ) AnalyzerDR = cms.EDAnalyzer( 'trklet::AnalyzerDR', TrackFindingTrackletAnalyzer_params, TrackFindingTrackletProducer_params ) AnalyzerKF = cms.EDAnalyzer( 'trklet::AnalyzerKF', TrackFindingTrackletAnalyzer_params, TrackFindingTrackletProducer_params ) -AnalyzerTFP = cms.EDAnalyzer( 'trklet::AnalyzerTFP', TrackFindingTrackletAnalyzer_params, TrackFindingTrackletProducer_params ) +AnalyzerTQ = cms.EDAnalyzer( 'trackerTFP::AnalyzerTQ', TrackFindingTrackletAnalyzer_params, TrackFindingTrackletProducer_params ) +AnalyzerTFP = cms.EDAnalyzer( 'trackerTFP::AnalyzerTFP', TrackFindingTrackletAnalyzer_params, TrackFindingTrackletProducer_params ) diff --git a/L1Trigger/TrackFindingTracklet/python/Producer_cff.py b/L1Trigger/TrackFindingTracklet/python/Producer_cff.py index 53af5b206b8eb..fb594be6d95b8 100644 --- a/L1Trigger/TrackFindingTracklet/python/Producer_cff.py +++ b/L1Trigger/TrackFindingTracklet/python/Producer_cff.py @@ -16,4 +16,4 @@ ProducerDR = cms.EDProducer( 'trklet::ProducerDR', TrackFindingTrackletProducer_params ) ProducerKF = cms.EDProducer( 'trklet::ProducerKF', TrackFindingTrackletProducer_params, HybridKalmanFilterFormats_params, TMTrackProducer_params ) ProducerTQ = cms.EDProducer( 'trackerTFP::ProducerTQ', TrackFindingTrackletProducer_params ) -ProducerTFP = cms.EDProducer( 'trackerTFP::ProducerTFP', TrackFindingTrackletProducer_params ) +ProducerTFP = cms.EDProducer( 'trklet::ProducerTFP', TrackFindingTrackletProducer_params ) diff --git a/L1Trigger/TrackFindingTracklet/src/KalmanFilter.cc b/L1Trigger/TrackFindingTracklet/src/KalmanFilter.cc index 55dd021595bc5..8799cdf101f3c 100644 --- a/L1Trigger/TrackFindingTracklet/src/KalmanFilter.cc +++ b/L1Trigger/TrackFindingTracklet/src/KalmanFilter.cc @@ -157,11 +157,8 @@ namespace trklet { if (zTtrack < zTs[iEtaReg + 1]) break; const L1track3D l1track3D(settings_, stubsFound, qOverPt, phi0, z0, tanLambda, helixD0, iPhiSec, iEtaReg); - //cout << "In " << qOverPt << " " << phi0 << " " << tanLambda << " " << z0 << endl; const L1fittedTrack trackFitted(tmtt_->fit(l1track3D)); - //cout << "Out " << trackFitted.qOverPt() << " " << trackFitted.phi0() << " " << trackFitted.tanLambda() << " " << trackFitted.z0() << endl; if (!trackFitted.accepted()) - //throw cms::Exception("..."); continue; static constexpr int trackId = 0; static constexpr int numConsistent = 0; diff --git a/L1Trigger/TrackFindingTracklet/src/TrackFindingProcessor.cc b/L1Trigger/TrackFindingTracklet/src/TrackFindingProcessor.cc new file mode 100644 index 0000000000000..813275293fce8 --- /dev/null +++ b/L1Trigger/TrackFindingTracklet/src/TrackFindingProcessor.cc @@ -0,0 +1,278 @@ +#include "L1Trigger/TrackFindingTracklet/interface/TrackFindingProcessor.h" +#include "L1Trigger/TrackTrigger/interface/StubPtConsistency.h" + +#include +#include +#include +#include +#include + +using namespace std; +using namespace edm; +using namespace tt; +using namespace trackerTFP; + +namespace trklet { + + TrackFindingProcessor::TrackFindingProcessor(const ParameterSet& iConfig, + const Setup* setup, + const DataFormats* dataFormats, + const TrackQuality* trackQuality) + : enableTruncation_(iConfig.getParameter("EnableTruncation")), + setup_(setup), + dataFormats_(dataFormats), + trackQuality_(trackQuality) {} + + // + TrackFindingProcessor::Track::Track(const FrameTrack& frameTrack, + const Frame& frameTQ, + const vector& ttStubRefs, + const DataFormats* df, + const TrackQuality* tq) + : ttTrackRef_(frameTrack.first), ttStubRefs_(ttStubRefs), valid_(true) { + partials_.reserve(partial_in); + // convert bits into nice formats + const Setup* setup = df->setup(); + const TrackKF track(frameTrack, df); + inv2R_ = track.inv2R(); + phiT_ = track.phiT(); + cot_ = track.cot(); + zT_ = track.zT(); + const double d0 = max(min(ttTrackRef_->d0(), -TTTrack_TrackWord::minD0), TTTrack_TrackWord::minD0); + TTBV ttBV = TTBV(frameTQ); + tq->format(VariableTQ::chi2rz).extract(ttBV, chi2rz_); + tq->format(VariableTQ::chi2rphi).extract(ttBV, chi2rphi_); + mva_ = TTBV(ttBV, numBinsMVA_).val(); + ttBV >>= numBinsMVA_; + hitPattern_ = ttBV; + channel_ = cot_ < 0. ? 0 : 1; + // convert nice formats into bits + const double z0 = zT_ - cot_ * setup->chosenRofZ(); + const double phi0 = phiT_ - inv2R_ * setup->chosenRofPhi(); + double invR = -2. * inv2R_; + if (invR < TTTrack_TrackWord::minRinv) + invR = TTTrack_TrackWord::minRinv + df->format(Variable::inv2R, Process::dr).base(); + else if (invR > -TTTrack_TrackWord::minRinv) + invR = -TTTrack_TrackWord::minRinv - df->format(Variable::inv2R, Process::dr).base(); + const double chi2rphi = chi2rphi_ / (hitPattern_.count() - 2); + const double chi2rz = chi2rz_ / (hitPattern_.count() - 2); + int chi2rphiBin(-1); + for (double d : TTTrack_TrackWord::chi2RPhiBins) + if (chi2rphi >= d) + chi2rphiBin++; + else + break; + int chi2rzBin(-1); + for (double d : TTTrack_TrackWord::chi2RZBins) + if (chi2rz >= d) + chi2rzBin++; + else + break; + static const double rangeInvR = -2. * TTTrack_TrackWord::minRinv; + static const double rangePhi0 = -2. * TTTrack_TrackWord::minPhi0; + static const double rangeCot = -2. * TTTrack_TrackWord::minTanl; + static const double rangeZ0 = -2. * TTTrack_TrackWord::minZ0; + static const double rangeD0 = -2. * TTTrack_TrackWord::minD0; + if (abs(invR) > rangeInvR / 2.) + valid_ = false; + if (abs(phi0) > rangePhi0 / 2.) + valid_ = false; + if (abs(cot_) > rangeCot / 2.) + valid_ = false; + if (abs(z0) > rangeZ0 / 2.) + valid_ = false; + if (abs(d0) > rangeD0 / 2.) + valid_ = false; + if (!valid_) + return; + static const double baseInvR = rangeInvR / pow(2., TTTrack_TrackWord::TrackBitWidths::kRinvSize); + static const double basePhi0 = rangePhi0 / pow(2., TTTrack_TrackWord::TrackBitWidths::kPhiSize); + static const double baseCot = rangeCot / pow(2., TTTrack_TrackWord::TrackBitWidths::kTanlSize); + static const double baseZ0 = rangeZ0 / pow(2., TTTrack_TrackWord::TrackBitWidths::kZ0Size); + static const double baseD0 = rangeD0 / pow(2., TTTrack_TrackWord::TrackBitWidths::kD0Size); + static constexpr int nLayers = TTTrack_TrackWord::TrackBitWidths::kHitPatternSize; + static const TTBV Other_MVAs(0, 2 * TTTrack_TrackWord::TrackBitWidths::kMVAQualitySize); + const TTBV MVA_quality(mva_, TTTrack_TrackWord::TrackBitWidths::kMVAQualitySize); + const TTBV hit_pattern(hitPattern_.resize(nLayers).val(), nLayers); + static const TTBV chi2bend(0, TTTrack_TrackWord::TrackBitWidths::kBendChi2Size); + static const TTBV D0(d0, baseD0, TTTrack_TrackWord::TrackBitWidths::kD0Size, true); + const TTBV Chi2rz(chi2rzBin, TTTrack_TrackWord::TrackBitWidths::kChi2RZSize); + const TTBV Z0(z0, baseZ0, TTTrack_TrackWord::TrackBitWidths::kZ0Size, true); + const TTBV tanL(cot_, baseCot, TTTrack_TrackWord::TrackBitWidths::kTanlSize, true); + const TTBV Chi2rphi(chi2rphiBin, TTTrack_TrackWord::TrackBitWidths::kChi2RPhiSize); + const TTBV Phi0(phi0, basePhi0, TTTrack_TrackWord::TrackBitWidths::kPhiSize, true); + const TTBV InvR(invR, baseInvR, TTTrack_TrackWord::TrackBitWidths::kRinvSize, true); + static const TTBV valid(1, TTTrack_TrackWord::TrackBitWidths::kValidSize); + partials_.emplace_back((valid + InvR + Phi0 + Chi2rphi).str()); + partials_.emplace_back((tanL + Z0 + Chi2rz).str()); + partials_.emplace_back((D0 + chi2bend + hit_pattern + MVA_quality + Other_MVAs).str()); + } + + // fill output products + void TrackFindingProcessor::produce(const StreamsTrack& inputs, + const StreamsStub& stubs, + TTTracks& ttTracks, + StreamsTrack& outputs) { + // organize input tracks + vector> streams(outputs.size()); + consume(inputs, stubs, streams); + // emualte data format f/w + produce(streams, outputs); + // produce TTTracks + produce(outputs, ttTracks); + } + + // + void TrackFindingProcessor::consume(const StreamsTrack& inputs, + const StreamsStub& stubs, + vector>& outputs) { + // count input objects + int nTracks(0); + auto valid = [](int sum, const FrameTrack& frame) { return sum += (frame.first.isNonnull() ? 1 : 0); }; + for (const StreamTrack& tracks : inputs) + nTracks += accumulate(tracks.begin(), tracks.end(), 0, valid); + tracks_.reserve(nTracks); + // convert input data + for (int region = 0; region < setup_->numRegions(); region++) { + const int offsetTQ = region * setup_->tqNumChannel(); + const int offsetTFP = region * setup_->tfpNumChannel(); + const int offsetStub = region * setup_->numLayers(); + const StreamTrack& streamDR = inputs[offsetTQ]; + const StreamTrack& streamTQ = inputs[offsetTQ + 1]; + for (int channel = 0; channel < setup_->tfpNumChannel(); channel++) + outputs[offsetTFP + channel] = deque(streamDR.size(), nullptr); + for (int frame = 0; frame < (int)streamDR.size(); frame++) { + const FrameTrack& frameTrack = streamDR[frame]; + const Frame& frameTQ = streamTQ[frame].second; + if (frameTrack.first.isNull()) + continue; + vector ttStubRefs; + ttStubRefs.reserve(setup_->numLayers()); + for (int layer = 0; layer < setup_->numLayers(); layer++) { + const TTStubRef& ttStubRef = stubs[offsetStub + layer][frame].first; + if (ttStubRef.isNonnull()) + ttStubRefs.push_back(ttStubRef); + } + tracks_.emplace_back(frameTrack, frameTQ, ttStubRefs, dataFormats_, trackQuality_); + Track& track = tracks_.back(); + outputs[offsetTFP + track.channel_][frame] = track.valid_ ? &track : nullptr; + } + // remove all gaps between end and last track + for (int channel = 0; channel < setup_->tfpNumChannel(); channel++) { + deque input = outputs[offsetTFP + channel]; + for (auto it = input.end(); it != input.begin();) + it = (*--it) ? input.begin() : input.erase(it); + } + } + } + + // emualte data format f/w + void TrackFindingProcessor::produce(vector>& inputs, StreamsTrack& outputs) const { + for (int channel = 0; channel < (int)inputs.size(); channel++) { + deque& input = inputs[channel]; + deque stack; + deque output; + // clock accurate firmware emulation, each while trip describes one clock tick, one stub in and one stub out per tick + while (!input.empty() || !stack.empty()) { + output.emplace_back(FrameTrack()); + FrameTrack& frame = output.back(); + Track* track = pop_front(input); + if (track) + for (const PartialFrame& pf : track->partials_) + stack.emplace_back(track->ttTrackRef_, pf); + TTBV ttBV; + for (int i = 0; i < partial_out; i++) { + if (stack.empty()) { + ttBV += TTBV(0, partial_width); + continue; + } + const PartialFrameTrack& pft = stack.front(); + frame.first = pft.first; + ttBV += TTBV(pft.second.to_string()); + stack.pop_front(); + } + frame.second = ttBV.bs(); + } + // perorm truncation + if (enableTruncation_ && (int)output.size() > setup_->numFramesIOHigh()) + output.resize(setup_->numFramesIOHigh()); + outputs[channel] = StreamTrack(output.begin(), output.end()); + } + } + + // produce TTTracks + void TrackFindingProcessor::produce(const StreamsTrack& inputs, TTTracks& outputs) const { + // collect input TTTrackRefs + vector ttTrackRefs; + ttTrackRefs.reserve(tracks_.size()); + const TTTrack* last = nullptr; + for (const StreamTrack& stream : inputs) { + for (const FrameTrack& frame : stream) { + const TTTrackRef& ttTrackRef = frame.first; + if (frame.first.isNull() || last == ttTrackRef.get()) + continue; + last = ttTrackRef.get(); + ttTrackRefs.push_back(ttTrackRef); + } + } + // convert input TTTrackRefs into output TTTracks + outputs.reserve(ttTrackRefs.size()); + for (const TTTrackRef& ttTrackRef : ttTrackRefs) { + auto match = [&ttTrackRef](const Track& track) { return track.ttTrackRef_ == ttTrackRef; }; + const auto it = find_if(tracks_.begin(), tracks_.end(), match); + // TTTrack conversion + const int region = ttTrackRef->phiSector(); + const double aRinv = -2. * it->inv2R_; + const double aphi = deltaPhi(it->phiT_ - it->inv2R_ * setup_->chosenRofPhi() + region * setup_->baseRegion()); + const double aTanLambda = it->cot_; + const double az0 = it->zT_ - it->cot_ * setup_->chosenRofZ(); + const double ad0 = -ttTrackRef->d0(); + const double aChi2xyfit = it->chi2rphi_; + const double aChi2zfit = it->chi2rz_; + const double trkMVA1 = (TTTrack_TrackWord::tqMVABins[it->mva_]); + static constexpr double trkMVA2 = 0.; + static constexpr double trkMVA3 = 0.; + const unsigned int aHitpattern = it->hitPattern_.val(); + const unsigned int nPar = ttTrackRef->nFitPars(); + static const double Bfield = setup_->bField(); + outputs.emplace_back( + aRinv, aphi, aTanLambda, az0, ad0, aChi2xyfit, aChi2zfit, trkMVA1, trkMVA2, trkMVA3, aHitpattern, nPar, Bfield); + TTTrack& ttTrack = outputs.back(); + ttTrack.setPhiSector(region); + ttTrack.setEtaSector(ttTrackRef->etaSector()); + ttTrack.setTrackSeedType(ttTrackRef->trackSeedType()); + ttTrack.setStubRefs(it->ttStubRefs_); + ttTrack.setStubPtConsistency(StubPtConsistency::getConsistency( + ttTrack, setup_->trackerGeometry(), setup_->trackerTopology(), Bfield, nPar)); + } + } + + // produce StreamsTrack + void TrackFindingProcessor::produce(const vector& inputs, StreamsTrack& outputs) const { + int iTrk(-1); + const TTTrack* last = nullptr; + for (StreamTrack& stream : outputs) { + for (FrameTrack& frame : stream) { + const TTTrackRef& ttTrackRef = frame.first; + if (ttTrackRef.isNull()) + continue; + if (last != ttTrackRef.get()) + iTrk++; + last = ttTrackRef.get(); + frame.first = inputs[iTrk]; + } + } + } + + // remove and return first element of deque, returns nullptr if empty + template + T* TrackFindingProcessor::pop_front(deque& ts) const { + T* t = nullptr; + if (!ts.empty()) { + t = ts.front(); + ts.pop_front(); + } + return t; + } + +} // namespace trklet diff --git a/L1Trigger/TrackFindingTracklet/test/HybridTracksNewKF_cfg.py b/L1Trigger/TrackFindingTracklet/test/HybridTracksNewKF_cfg.py index 6055dadd0061d..80796c9611458 100644 --- a/L1Trigger/TrackFindingTracklet/test/HybridTracksNewKF_cfg.py +++ b/L1Trigger/TrackFindingTracklet/test/HybridTracksNewKF_cfg.py @@ -19,7 +19,7 @@ process.load( 'L1Trigger.TrackTrigger.TrackTrigger_cff' ) from Configuration.AlCa.GlobalTag import GlobalTag -process.GlobalTag = GlobalTag(process.GlobalTag, 'auto:phase2_realistic', '') +process.GlobalTag = GlobalTag(process.GlobalTag, '133X_mcRun4_realistic_v1', '') # load code that associates stubs with mctruth process.load( 'SimTracker.TrackTriggerAssociation.StubAssociator_cff' ) @@ -45,7 +45,7 @@ process.tm = cms.Sequence( process.ProducerTM + process.AnalyzerTM ) process.dr = cms.Sequence( process.ProducerDR + process.AnalyzerDR ) process.kf = cms.Sequence( process.ProducerKF + process.AnalyzerKF ) -process.tq = cms.Sequence( process.ProducerTQ ) +process.tq = cms.Sequence( process.ProducerTQ + process.AnalyzerTQ ) process.tfp = cms.Sequence( process.ProducerTFP + process.AnalyzerTFP ) process.tt = cms.Path( process.mc + process.dtc + process.tracklet + process.tm + process.dr + process.kf + process.tq + process.tfp ) process.schedule = cms.Schedule( process.tt ) diff --git a/L1Trigger/TrackFindingTracklet/test/L1TrackNtupleMaker_cfg.py b/L1Trigger/TrackFindingTracklet/test/L1TrackNtupleMaker_cfg.py index 5490521df5cd6..a20c01161ddc2 100644 --- a/L1Trigger/TrackFindingTracklet/test/L1TrackNtupleMaker_cfg.py +++ b/L1Trigger/TrackFindingTracklet/test/L1TrackNtupleMaker_cfg.py @@ -23,7 +23,7 @@ # 'HYBRID_NEWKF' (baseline, 4par fit, with bit-accurate KF emulation), # 'HYBRID_REDUCED' to use the "Summer Chain" configuration with reduced inputs. # (Or legacy algos 'TMTT' or 'TRACKLET'). -L1TRKALGO = 'HYBRID' +L1TRKALGO = 'HYBRID_NEWKF' WRITE_DATA = False @@ -181,7 +181,7 @@ #process.TTTracksEmulationWithTruth = cms.Path(process.HybridNewKF + process.TrackTriggerAssociatorTracks) # Optionally include code producing performance plots & end-of-job summary. process.load( 'SimTracker.TrackTriggerAssociation.StubAssociator_cff' ) - process.TTTracksEmulationWithTruth = cms.Path(process.HybridNewKF + process.TrackTriggerAssociatorTracks + process.StubAssociator + process.AnalyzerTracklet + process.AnalyzerTM + process.AnalyzerDR + process.AnalyzerKF ) + process.TTTracksEmulationWithTruth = cms.Path(process.HybridNewKF + process.TrackTriggerAssociatorTracks + process.StubAssociator + process.AnalyzerTracklet + process.AnalyzerTM + process.AnalyzerDR + process.AnalyzerTQ + process.AnalyzerKF + process.AnalyzerTFP ) from L1Trigger.TrackFindingTracklet.Customize_cff import * if (L1TRKALGO == 'HYBRID_NEWKF'): fwConfig( process ) diff --git a/L1Trigger/TrackFindingTracklet/test/L1TrkNtuple.root b/L1Trigger/TrackFindingTracklet/test/L1TrkNtuple.root new file mode 100644 index 0000000000000000000000000000000000000000..559fa240c811307e93a61e51f2657ff4e4ab8900 GIT binary patch literal 316333 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%0AFlcLzBRs8MU1ANihhA7?FveWMnz z!mg{TZ^JjUy0yn2JnZZSbSRi+KN^v8lOAfM7so{=_7JQogQ3F}Hujfn8n=#MAC z9h}gruprM0zFtJoSgfHhShO*>jU8hXBg4C(vIYYH1;g+m3?BjBAvc(krb(82AC8%|cI9goVkPRfqP z&y~u0#1OlD9SK`6GdgLk8)p2EU7a2H9RlU}B-e`GWoOLZGu70<0iIRyFMPwrBJpN= zkFE(<`rl8EH5Q+o<3(sS+Q+#>04H~4TE=&EdHO%r({rrz&f)U;_s4fM$Hv3=8hq}< zyqyB~Stq4SNrxIcR^2w?4`|GUM%0>6ocT0NcCUChIK`K?9Dj7Pn~e?rEHe?i zaea)^!LHa)93t$5XWdVOB=M%ga@cDaAJn4bqZM~4J!=r3Qr&A}EUnBS9^cjL!M-ES zUfTY?n6_N)qi3kcy~Ir9yBmeD`uGFZdyEwh7ZOAs<|?LZEgnX15H!^;Zr6+ZoW5qb zj3p-@L4<`>I*!Y%U{3dH_R?~#tIno?56i2+n>g≺Jh2-VzOK7H2Tq%dzH3))?J9 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
SZ0ub-YC0PM~lP-UK-Y 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 E0LVIZ!vFvP literal 0 HcmV?d00001 diff --git a/L1Trigger/TrackerTFP/python/Analyzer_cff.py b/L1Trigger/TrackerTFP/python/Analyzer_cff.py index bf8fb9c10facd..fbc9959e108d9 100644 --- a/L1Trigger/TrackerTFP/python/Analyzer_cff.py +++ b/L1Trigger/TrackerTFP/python/Analyzer_cff.py @@ -10,3 +10,5 @@ AnalyzerCTB = cms.EDAnalyzer( 'trackerTFP::AnalyzerCTB', TrackerTFPAnalyzer_params, TrackerTFPProducer_params ) AnalyzerKF = cms.EDAnalyzer( 'trackerTFP::AnalyzerKF' , TrackerTFPAnalyzer_params, TrackerTFPProducer_params ) AnalyzerDR = cms.EDAnalyzer( 'trackerTFP::AnalyzerDR' , TrackerTFPAnalyzer_params, TrackerTFPProducer_params ) +AnalyzerTQ = cms.EDAnalyzer( 'trackerTFP::AnalyzerTQ', TrackerTFPAnalyzer_params, TrackerTFPProducer_params ) +AnalyzerTFP = cms.EDAnalyzer( 'trackerTFP::AnalyzerTFP', TrackerTFPAnalyzer_params, TrackerTFPProducer_params ) diff --git a/L1Trigger/TrackerTFP/python/Analyzer_cfi.py b/L1Trigger/TrackerTFP/python/Analyzer_cfi.py index 8b4e89f7fe5fa..2c1fe9b588dec 100644 --- a/L1Trigger/TrackerTFP/python/Analyzer_cfi.py +++ b/L1Trigger/TrackerTFP/python/Analyzer_cfi.py @@ -12,5 +12,7 @@ OutputLabelCTB = cms.string( "ProducerCTB" ), # OutputLabelKF = cms.string( "ProducerKF" ), # OutputLabelDR = cms.string( "ProducerDR" ), # + OutputLabelTQ = cms.string( "ProducerTQ" ), # + OutputLabelTFP = cms.string( "ProducerTFP" ), # ) diff --git a/L1Trigger/TrackerTFP/src/HoughTransform.cc b/L1Trigger/TrackerTFP/src/HoughTransform.cc index 932503f20d7b6..c6da54733705b 100644 --- a/L1Trigger/TrackerTFP/src/HoughTransform.cc +++ b/L1Trigger/TrackerTFP/src/HoughTransform.cc @@ -200,7 +200,7 @@ namespace trackerTFP { doubleGap = false; if (++nHits == minLayers) return false; - } else if (!maybePattern.test(layer)) { + } else if (nHits < setup_->kfMinLayers() && !maybePattern.test(layer)) { if (++nGaps == setup_->kfMaxGaps() || doubleGap) break; doubleGap = true; diff --git a/L1Trigger/TrackerTFP/test/AnalyzerCTB.cc b/L1Trigger/TrackerTFP/test/AnalyzerCTB.cc index c9c81bd842b12..374c5f7c5f92e 100644 --- a/L1Trigger/TrackerTFP/test/AnalyzerCTB.cc +++ b/L1Trigger/TrackerTFP/test/AnalyzerCTB.cc @@ -29,7 +29,6 @@ using namespace std; using namespace edm; -using namespace trackerTFP; using namespace tt; namespace trackerTFP { diff --git a/L1Trigger/TrackerTFP/test/AnalyzerDR.cc b/L1Trigger/TrackerTFP/test/AnalyzerDR.cc index 29fe8c64f3088..4ad0fdda6c133 100644 --- a/L1Trigger/TrackerTFP/test/AnalyzerDR.cc +++ b/L1Trigger/TrackerTFP/test/AnalyzerDR.cc @@ -28,7 +28,6 @@ using namespace std; using namespace edm; -using namespace trackerTFP; using namespace tt; namespace trackerTFP { @@ -189,7 +188,6 @@ namespace trackerTFP { associate(tracks, selection, tpPtrsSelection, tmp); associate(tracks, reconstructable, tpPtrs, allMatched, false); associate(tracks, selection, tpPtrsMax, tmp, false); - //cout << "DR " << tpPtrsSelection.size() << " " << tpPtrsMax.size() << endl; const int size = acceptedTracks[offset + channel].size(); hisChannel_->Fill(size); profChannel_->Fill(channel, size); diff --git a/L1Trigger/TrackerTFP/test/AnalyzerKF.cc b/L1Trigger/TrackerTFP/test/AnalyzerKF.cc index 06ed2f943123b..39e5ee99da9c4 100644 --- a/L1Trigger/TrackerTFP/test/AnalyzerKF.cc +++ b/L1Trigger/TrackerTFP/test/AnalyzerKF.cc @@ -302,7 +302,6 @@ namespace trackerTFP { vector(), false); associate(tracks, tracksStubs, region, selection, tpPtrsMax, tmp, vector(), vector(), false); - //cout << "KF " << tpPtrsSelection.size() << " " << tpPtrsMax.size() << endl; } numTracks += nRegionTracks; prof_->Fill(1, nRegionStubs); diff --git a/L1Trigger/TrackFindingTracklet/test/AnalyzerTFP.cc b/L1Trigger/TrackerTFP/test/AnalyzerTFP.cc similarity index 98% rename from L1Trigger/TrackFindingTracklet/test/AnalyzerTFP.cc rename to L1Trigger/TrackerTFP/test/AnalyzerTFP.cc index 2e79e64bb6fae..9a1483884fcd9 100644 --- a/L1Trigger/TrackFindingTracklet/test/AnalyzerTFP.cc +++ b/L1Trigger/TrackerTFP/test/AnalyzerTFP.cc @@ -30,9 +30,9 @@ using namespace std; using namespace edm; using namespace tt; -namespace trklet { +namespace trackerTFP { - /*! \class trklet::AnalyzerTFP + /*! \class trackerTFP::AnalyzerTFP * \brief Class to analyze TTTracks found by tfp * \author Thomas Schuh * \date 20204 Aug @@ -260,6 +260,6 @@ namespace trklet { } } -} // namespace trklet +} // namespace trackerTFP -DEFINE_FWK_MODULE(trklet::AnalyzerTFP); +DEFINE_FWK_MODULE(trackerTFP::AnalyzerTFP); diff --git a/L1Trigger/TrackerTFP/test/AnalyzerTQ.cc b/L1Trigger/TrackerTFP/test/AnalyzerTQ.cc new file mode 100644 index 0000000000000..213cd9a7ec1c2 --- /dev/null +++ b/L1Trigger/TrackerTFP/test/AnalyzerTQ.cc @@ -0,0 +1,284 @@ +#include "FWCore/Framework/interface/one/EDAnalyzer.h" +#include "FWCore/Framework/interface/Run.h" +#include "FWCore/Framework/interface/Event.h" +#include "FWCore/Framework/interface/EventSetup.h" +#include "FWCore/Framework/interface/MakerMacros.h" +#include "FWCore/ParameterSet/interface/ParameterSet.h" +#include "FWCore/ServiceRegistry/interface/Service.h" +#include "FWCore/MessageLogger/interface/MessageLogger.h" +#include "FWCore/Utilities/interface/EDGetToken.h" +#include "FWCore/Utilities/interface/InputTag.h" +#include "FWCore/Utilities/interface/Exception.h" +#include "CommonTools/UtilAlgos/interface/TFileService.h" +#include "DataFormats/Common/interface/Handle.h" + +#include "SimTracker/TrackTriggerAssociation/interface/StubAssociation.h" +#include "L1Trigger/TrackTrigger/interface/Setup.h" +#include "L1Trigger/TrackerTFP/interface/DataFormats.h" + +#include +#include + +#include +#include +#include +#include +#include +#include + +using namespace std; +using namespace edm; +using namespace tt; + +namespace trackerTFP { + + /*! \class trackerTFP::AnalyzerTQ + * \brief Class to analyze hardware like structured track Collection generated by Duplicate Removal + * \author Thomas Schuh + * \date 2023, Feb + */ + class AnalyzerTQ : public one::EDAnalyzer { + public: + AnalyzerTQ(const ParameterSet& iConfig); + void beginJob() override {} + void beginRun(const Run& iEvent, const EventSetup& iSetup) override; + void analyze(const Event& iEvent, const EventSetup& iSetup) override; + void endRun(const Run& iEvent, const EventSetup& iSetup) override {} + void endJob() override; + + private: + // + void formTracks(const StreamsTrack& streamsTrack, + const StreamsStub& streamsStubs, + vector>& tracks, + int channel) const; + // + void associate(const vector>& tracks, + const StubAssociation* ass, + set& tps, + int& sum, + bool perfect = true) const; + // ED input token of stubs + EDGetTokenT edGetTokenStubs_; + // ED input token of tracks + EDGetTokenT edGetTokenTracks_; + // ED input token of TTStubRef to TPPtr association for tracking efficiency + EDGetTokenT edGetTokenSelection_; + // ED input token of TTStubRef to recontructable TPPtr association + EDGetTokenT edGetTokenReconstructable_; + // Setup token + ESGetToken esGetTokenSetup_; + // DataFormats token + ESGetToken esGetTokenDataFormats_; + // stores, calculates and provides run-time constants + const Setup* setup_ = nullptr; + // helper class to extract structured data from tt::Frames + const DataFormats* dataFormats_ = nullptr; + // enables analyze of TPs + bool useMCTruth_; + // + int nEvents_ = 0; + + // Histograms + + TProfile* prof_; + TProfile* profChannel_; + TProfile* profTracks_; + TH1F* hisChannel_; + TH1F* hisTracks_; + + // printout + stringstream log_; + }; + + AnalyzerTQ::AnalyzerTQ(const ParameterSet& iConfig) : useMCTruth_(iConfig.getParameter("UseMCTruth")) { + usesResource("TFileService"); + // book in- and output ED products + const string& label = iConfig.getParameter("OutputLabelDR"); + const string& branchStubs = iConfig.getParameter("BranchStubs"); + const string& branchTracks = iConfig.getParameter("BranchTracks"); + edGetTokenStubs_ = consumes(InputTag(label, branchStubs)); + edGetTokenTracks_ = consumes(InputTag(label, branchTracks)); + if (useMCTruth_) { + const auto& inputTagSelecttion = iConfig.getParameter("InputTagSelection"); + const auto& inputTagReconstructable = iConfig.getParameter("InputTagReconstructable"); + edGetTokenSelection_ = consumes(inputTagSelecttion); + edGetTokenReconstructable_ = consumes(inputTagReconstructable); + } + // book ES products + esGetTokenSetup_ = esConsumes(); + esGetTokenDataFormats_ = esConsumes(); + // log config + log_.setf(ios::fixed, ios::floatfield); + log_.precision(4); + } + + void AnalyzerTQ::beginRun(const Run& iEvent, const EventSetup& iSetup) { + // helper class to store configurations + setup_ = &iSetup.getData(esGetTokenSetup_); + // helper class to extract structured data from tt::Frames + dataFormats_ = &iSetup.getData(esGetTokenDataFormats_); + // book histograms + Service fs; + TFileDirectory dir; + dir = fs->mkdir("DR"); + prof_ = dir.make("Counts", ";", 12, 0.5, 12.5); + prof_->GetXaxis()->SetBinLabel(1, "Stubs"); + prof_->GetXaxis()->SetBinLabel(2, "Tracks"); + prof_->GetXaxis()->SetBinLabel(4, "Matched Tracks"); + prof_->GetXaxis()->SetBinLabel(5, "All Tracks"); + prof_->GetXaxis()->SetBinLabel(6, "Found TPs"); + prof_->GetXaxis()->SetBinLabel(7, "Found selected TPs"); + prof_->GetXaxis()->SetBinLabel(9, "All TPs"); + prof_->GetXaxis()->SetBinLabel(10, "states"); + prof_->GetXaxis()->SetBinLabel(12, "max tp"); + // channel occupancy + constexpr int maxOcc = 180; + const int numChannels = dataFormats_->numChannel(Process::dr); + hisChannel_ = dir.make("His Channel Occupancy", ";", maxOcc, -.5, maxOcc - .5); + profChannel_ = dir.make("Prof Channel Occupancy", ";", numChannels, -.5, numChannels - .5); + // track occupancy + hisTracks_ = dir.make("His Track Occupancy", ";", maxOcc, -.5, maxOcc - .5); + profTracks_ = dir.make("Prof Track Occupancy", ";", numChannels, -.5, numChannels - .5); + } + + void AnalyzerTQ::analyze(const Event& iEvent, const EventSetup& iSetup) { + // read in ht products + Handle handleStubs; + iEvent.getByToken(edGetTokenStubs_, handleStubs); + const StreamsStub& acceptedStubs = *handleStubs; + Handle handleTracks; + iEvent.getByToken(edGetTokenTracks_, handleTracks); + const StreamsTrack& acceptedTracks = *handleTracks; + // read in MCTruth + const StubAssociation* selection = nullptr; + const StubAssociation* reconstructable = nullptr; + if (useMCTruth_) { + Handle handleSelection; + iEvent.getByToken(edGetTokenSelection_, handleSelection); + selection = handleSelection.product(); + prof_->Fill(9, selection->numTPs()); + Handle handleReconstructable; + iEvent.getByToken(edGetTokenReconstructable_, handleReconstructable); + reconstructable = handleReconstructable.product(); + } + // analyze ht products and associate found tracks with reconstrucable TrackingParticles + set tpPtrs; + set tpPtrsSelection; + set tpPtrsMax; + int allMatched(0); + int allTracks(0); + for (int region = 0; region < setup_->numRegions(); region++) { + const int offset = region * dataFormats_->numChannel(Process::dr); + int nStubs(0); + int nTracks(0); + for (int channel = 0; channel < dataFormats_->numChannel(Process::dr); channel++) { + vector> tracks; + formTracks(acceptedTracks, acceptedStubs, tracks, offset + channel); + hisTracks_->Fill(tracks.size()); + profTracks_->Fill(channel, tracks.size()); + nTracks += tracks.size(); + nStubs += accumulate(tracks.begin(), tracks.end(), 0, [](int sum, const vector& track) { + return sum += (int)track.size(); + }); + allTracks += tracks.size(); + if (!useMCTruth_) + continue; + int tmp(0); + associate(tracks, selection, tpPtrsSelection, tmp); + associate(tracks, reconstructable, tpPtrs, allMatched, false); + associate(tracks, selection, tpPtrsMax, tmp, false); + const int size = acceptedTracks[offset + channel].size(); + hisChannel_->Fill(size); + profChannel_->Fill(channel, size); + } + prof_->Fill(1, nStubs); + prof_->Fill(2, nTracks); + } + prof_->Fill(4, allMatched); + prof_->Fill(5, allTracks); + prof_->Fill(6, tpPtrs.size()); + prof_->Fill(7, tpPtrsSelection.size()); + prof_->Fill(12, tpPtrsMax.size()); + nEvents_++; + } + + void AnalyzerTQ::endJob() { + if (nEvents_ == 0) + return; + // printout DR summary + const double totalTPs = prof_->GetBinContent(9); + const double numStubs = prof_->GetBinContent(1); + const double numTracks = prof_->GetBinContent(2); + const double totalTracks = prof_->GetBinContent(5); + const double numTracksMatched = prof_->GetBinContent(4); + const double numTPsAll = prof_->GetBinContent(6); + const double numTPsEff = prof_->GetBinContent(7); + const double numTPsEffMax = prof_->GetBinContent(12); + const double errStubs = prof_->GetBinError(1); + const double errTracks = prof_->GetBinError(2); + const double fracFake = (totalTracks - numTracksMatched) / totalTracks; + const double fracDup = (numTracksMatched - numTPsAll) / totalTracks; + const double eff = numTPsEff / totalTPs; + const double errEff = sqrt(eff * (1. - eff) / totalTPs / nEvents_); + const double effMax = numTPsEffMax / totalTPs; + const double errEffMax = sqrt(effMax * (1. - effMax) / totalTPs / nEvents_); + const vector nums = {numStubs, numTracks}; + const vector errs = {errStubs, errTracks}; + const int wNums = ceil(log10(*max_element(nums.begin(), nums.end()))) + 5; + const int wErrs = ceil(log10(*max_element(errs.begin(), errs.end()))) + 5; + log_ << " TQ SUMMARY " << endl; + log_ << "number of stubs per TFP = " << setw(wNums) << numStubs << " +- " << setw(wErrs) << errStubs << endl; + log_ << "number of tracks per TFP = " << setw(wNums) << numTracks << " +- " << setw(wErrs) << errTracks + << endl; + log_ << " tracking efficiency = " << setw(wNums) << eff << " +- " << setw(wErrs) << errEff << endl; + log_ << " max tracking efficiency = " << setw(wNums) << effMax << " +- " << setw(wErrs) << errEffMax << endl; + log_ << " fake rate = " << setw(wNums) << fracFake << endl; + log_ << " duplicate rate = " << setw(wNums) << fracDup << endl; + log_ << "============================================================="; + LogPrint(moduleDescription().moduleName()) << log_.str(); + } + + // + void AnalyzerTQ::formTracks(const StreamsTrack& streamsTrack, + const StreamsStub& streamsStubs, + vector>& tracks, + int channel) const { + const int offset = channel * setup_->numLayers(); + const StreamTrack& streamTrack = streamsTrack[channel]; + const int numTracks = accumulate(streamTrack.begin(), streamTrack.end(), 0, [](int sum, const FrameTrack& frame) { + return sum += (frame.first.isNonnull() ? 1 : 0); + }); + tracks.reserve(numTracks); + for (int frame = 0; frame < (int)streamTrack.size(); frame++) { + const FrameTrack& frameTrack = streamTrack[frame]; + if (frameTrack.first.isNull()) + continue; + deque stubs; + for (int layer = 0; layer < setup_->numLayers(); layer++) { + const FrameStub& stub = streamsStubs[offset + layer][frame]; + if (stub.first.isNonnull()) + stubs.push_back(stub.first); + } + tracks.emplace_back(stubs.begin(), stubs.end()); + } + } + + // + void AnalyzerTQ::associate(const vector>& tracks, + const StubAssociation* ass, + set& tps, + int& sum, + bool perfect) const { + for (const vector& ttStubRefs : tracks) { + const vector& tpPtrs = perfect ? ass->associateFinal(ttStubRefs) : ass->associate(ttStubRefs); + if (tpPtrs.empty()) + continue; + sum++; + copy(tpPtrs.begin(), tpPtrs.end(), inserter(tps, tps.begin())); + } + } + +} // namespace trackerTFP + +DEFINE_FWK_MODULE(trackerTFP::AnalyzerTQ); diff --git a/L1Trigger/TrackerTFP/test/test_cfg.py b/L1Trigger/TrackerTFP/test/test_cfg.py index e4d3ee70d0710..a873444928982 100644 --- a/L1Trigger/TrackerTFP/test/test_cfg.py +++ b/L1Trigger/TrackerTFP/test/test_cfg.py @@ -51,7 +51,7 @@ process.dr = cms.Sequence( process.ProducerDR + process.AnalyzerDR ) process.tq = cms.Sequence( process.ProducerTQ ) process.tfp = cms.Sequence( process.ProducerTFP ) -process.tt = cms.Path( process.mc + process.dtc + process.pp + process.gp + process.ht + process.ctb + process.kf )#+ process.dr + process.tq )# + process.tfp ) +process.tt = cms.Path( process.mc + process.dtc + process.pp + process.gp + process.ht + process.ctb)# + process.kf )#+ process.dr + process.tq )# + process.tfp ) process.schedule = cms.Schedule( process.tt ) # create options From bf44bedc6720ae340d4ad1fd76863a71e2980a1d Mon Sep 17 00:00:00 2001 From: Thomas Schuh Date: Wed, 4 Dec 2024 16:48:58 +0000 Subject: [PATCH 10/14] script fix --- L1Trigger/TrackFindingTracklet/test/L1TrackNtupleMaker_cfg.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/L1Trigger/TrackFindingTracklet/test/L1TrackNtupleMaker_cfg.py b/L1Trigger/TrackFindingTracklet/test/L1TrackNtupleMaker_cfg.py index a20c01161ddc2..30905bbef40cb 100644 --- a/L1Trigger/TrackFindingTracklet/test/L1TrackNtupleMaker_cfg.py +++ b/L1Trigger/TrackFindingTracklet/test/L1TrackNtupleMaker_cfg.py @@ -23,7 +23,7 @@ # 'HYBRID_NEWKF' (baseline, 4par fit, with bit-accurate KF emulation), # 'HYBRID_REDUCED' to use the "Summer Chain" configuration with reduced inputs. # (Or legacy algos 'TMTT' or 'TRACKLET'). -L1TRKALGO = 'HYBRID_NEWKF' +L1TRKALGO = 'HYBRID' WRITE_DATA = False From dd57a5a013eb33ccf101444dd8d91d9e16e6c0fd Mon Sep 17 00:00:00 2001 From: Thomas Schuh Date: Wed, 4 Dec 2024 18:27:23 +0000 Subject: [PATCH 11/14] little fixes. --- L1Trigger/TrackFindingTracklet/test/L1TrackNtupleMaker_cfg.py | 2 +- L1Trigger/TrackerTFP/test/AnalyzerTQ.cc | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/L1Trigger/TrackFindingTracklet/test/L1TrackNtupleMaker_cfg.py b/L1Trigger/TrackFindingTracklet/test/L1TrackNtupleMaker_cfg.py index 30905bbef40cb..4a9717c407f3c 100644 --- a/L1Trigger/TrackFindingTracklet/test/L1TrackNtupleMaker_cfg.py +++ b/L1Trigger/TrackFindingTracklet/test/L1TrackNtupleMaker_cfg.py @@ -181,7 +181,7 @@ #process.TTTracksEmulationWithTruth = cms.Path(process.HybridNewKF + process.TrackTriggerAssociatorTracks) # Optionally include code producing performance plots & end-of-job summary. process.load( 'SimTracker.TrackTriggerAssociation.StubAssociator_cff' ) - process.TTTracksEmulationWithTruth = cms.Path(process.HybridNewKF + process.TrackTriggerAssociatorTracks + process.StubAssociator + process.AnalyzerTracklet + process.AnalyzerTM + process.AnalyzerDR + process.AnalyzerTQ + process.AnalyzerKF + process.AnalyzerTFP ) + process.TTTracksEmulationWithTruth = cms.Path(process.HybridNewKF + process.TrackTriggerAssociatorTracks + process.StubAssociator + process.AnalyzerTracklet + process.AnalyzerTM + process.AnalyzerDR + process.AnalyzerKF + process.AnalyzerTQ + process.AnalyzerTFP ) from L1Trigger.TrackFindingTracklet.Customize_cff import * if (L1TRKALGO == 'HYBRID_NEWKF'): fwConfig( process ) diff --git a/L1Trigger/TrackerTFP/test/AnalyzerTQ.cc b/L1Trigger/TrackerTFP/test/AnalyzerTQ.cc index 213cd9a7ec1c2..fa782a814c170 100644 --- a/L1Trigger/TrackerTFP/test/AnalyzerTQ.cc +++ b/L1Trigger/TrackerTFP/test/AnalyzerTQ.cc @@ -94,7 +94,7 @@ namespace trackerTFP { AnalyzerTQ::AnalyzerTQ(const ParameterSet& iConfig) : useMCTruth_(iConfig.getParameter("UseMCTruth")) { usesResource("TFileService"); // book in- and output ED products - const string& label = iConfig.getParameter("OutputLabelDR"); + const string& label = iConfig.getParameter("OutputLabelTQ"); const string& branchStubs = iConfig.getParameter("BranchStubs"); const string& branchTracks = iConfig.getParameter("BranchTracks"); edGetTokenStubs_ = consumes(InputTag(label, branchStubs)); From d0f1dcdbb9b0207af06b73b3f70586c0632e5c99 Mon Sep 17 00:00:00 2001 From: Thomas Schuh Date: Thu, 5 Dec 2024 14:04:49 +0000 Subject: [PATCH 12/14] furthrt little fixes --- .../TrackFindingTracklet/test/L1TrkNtuple.root | Bin 316333 -> 0 bytes .../TrackFindingTracklet/test/ProducerIRin.cc | 2 +- L1Trigger/TrackerDTC/plugins/ProducerDTC.cc | 4 ++-- L1Trigger/TrackerTFP/plugins/ProducerTQ.cc | 8 ++++++-- L1Trigger/TrackerTFP/test/demonstrator_cfg.py | 2 +- 5 files changed, 10 insertions(+), 6 deletions(-) delete mode 100644 L1Trigger/TrackFindingTracklet/test/L1TrkNtuple.root diff --git a/L1Trigger/TrackFindingTracklet/test/L1TrkNtuple.root b/L1Trigger/TrackFindingTracklet/test/L1TrkNtuple.root deleted file mode 100644 index 559fa240c811307e93a61e51f2657ff4e4ab8900..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 316333 zcmeEvby!qe+xN^cbO=ZZNS7krAl)6Jl*o_*QUcQ5DF`S?H!7eYB_Q1?Axd{iE7JAt z89j=po^zh>ulK!PFZNtpSZm*F{q}vYJJ%X}TU$pEXqOrU0-1oIgN49l4BRUK7Z|uS zpumSI2wHXw0$sif0^67c+q_o8+brp1WZw0w-30#nkN!Ob!5^gJjJ9zE5<@)M1m5qW z<_$+%d&9e?GKR*$jjN1>l_{@ZtUOHy~vAlbq4EE&#*a10JLS zT!<%vpry$FahRTn(Od`mqk_Z{qxk~wkD)1LVQ-2U2sV5WhQ{BA@$3KK#J5nt5eK;e z9mRnQ@c>{#*&wh(8p!BxFmnAt92+DICl0p#9dS%?8$&BsH&c5}8MX8G2*TLk3EKdj zgn#P{0#WsVz)n@Z3V%b`;v8WX1mR_P|8uiO0tmvd^Mt@Y03jBjA=e$lg6imKi~|9p3*O zS%{(x@N=FFN+Upq8E6JHh0jjU5kY4EZ^-;%c1Uc0H9IMdAG4FvI8XMs*?sE@0$Iev z$(GRk4cR};4k;G|LSj2VJ1GswuVmm744)mmDSUS3-^eun4Vgd84vPI(v(r@mF*{A= zKbzgRt{{-D2Yhy=eSbst53_>?fIv{}^Rv@bhWw69*SFcho5E*jQw%3lcJ*(__@84D zSvZS8SARD<+}}78A^ujTj(_;YFEjN6y379d7!b&l2R?*yl)o8*!?_{w!H0khNHTQw z+)UAaBmLv%JV|gEK$7oUH~6gG^ARL1|Ayqx6$j_rii14!yIKEOWu!DE&)*{m<9uIn zo&ueK73W)L5VXV!1a`Jk-uN5BKi?`2z=ptFQFP(`&&?I0ED8LaCj(Xk$N)SGZwjAZ zP#&C&>g?Z;`MI#+04o)oOvCTUAOZ7&U%wF21_%HP8@w?<2>O};PDq3OZwTG{@$PUy zY;Z!TSKHxpa69NkYtpNDojp6VWjlv1(efVoa zm_l&)G71pv1PBVy*8=GK1zA(m-qaM`)XCJw5q^gu4Zovu1a5Ve&`vNw1sNb0@Op~1 zIeFWqwlhJ-uxnlA5exCd57h$ik$-ezeZf*)uEr~IpFmbUa5f~JzczXELsj@fb(e){ z7e*^iP|Sxod@OoTc?ku~^{xi?7R;8~->6dN~mAj`I=1}KZjuG|w zYg+nxW*|G#B3tcuL);!2S*uRWPEpKlz4v@aEUcSv{iaQC^W;$Wq+pri`9Xs}>V;F@ z{j0;Dy|awVrpg~QLat5Y!QKU{er;+*ttf7|Y1GfBQv8lcqixtZHR+j|Vr3(j;x1wJ ze9>ZC-@S}gE~|1M;hlVTOh=)iuBTy57bG}AWN*Of6znGySFm1`FeA;o2z-Hog{0aC z;&sG2y!}q4h_>)8o|c!y2jFXe4ZbAanFaNt_i^_eH)!=DISRpQp>H|~YByeEi#SHF zy{LTVJM1ykE3mLKoRee!;Swsvo#6vvN3Mf4t!i05EX?Pg6@6t>5|%z<2S~oJRZ$l3 zP{P(tCbEb$?bV+p*95*iB4QVJB&QXW+6RpnO=Qhv5qfB1NLSTAmMFe_5Z%b<@$7SP z(vhs_L0cZ)Wn~d%;3^D!XcgzKbbrr$yyCq-pH-<}WEm;hJ?jn`VV9;LQZmVLKBrjQ zRt}1M?C^xlDOviByfGij(t&@qDduD z1tcZkYk8_*VRvEee7K|=8>ai_ip+hkuF9u0HY#|05>U_;`{(A`>c$Incj=W2j^nij z=t^Eo7d4KzY7I4Zlc{&V?SA@{-i#sS5;ckso14K8`H(2oL1hG0-Me9T?md^g3Wdp4X6`plgJ$ zS?|BA5=(uMaQ&XdZh`Q;&ftP6OiUsxBs`}*3naP_}o4Qn+_%vW{wSnn4>O$#k zM2@~S%@WhyzH@v&DpnzeRpY8(_E;hX1%L>tm;&G(n)Q% zyr+BYYT#`6SA~R%+`wn{ao)uR|+>;5KM+mf23hZQ69%#BxNv?4px=xJ_+O`p}1>Fh#oa4EeXXB-@1OX_7X z6AX8<$kBei@S4boRrXRZIbU*#3BR!kzXS?Mz_HTGT>Vp~k;;`T@$u`Uarn0eBorD$ zWfkwB2jLSc%(`|ee6oM;Q7Cm5@wOl^E+7@3rl~JeqKkjiSH?n)^5C`1L1&B6mDRZX zmfPzSD9^<)`Q!~gM9*G8FsQdoe zGK7+_XYiSMn2fXpzE8RjOflFYn?Lp8-c{`snqV`WuopyL_$vMc{YTA?`~sag+^v_F zNd$1)eQ~;QN^d$*jddKZ>MrX%v`<6c5OJ6niJcdbzyW#hpH|HT+b0hz);<4x;|hT= z@$HX(=*qKsw{>>l9A z&$W?_{t5dau*TyUPhw0k)(=kT}g|Ziv`aI@oz+fw-o# z%BssR&#B+bmfmzpJ-Hd*IMJK_My4wz+2xAtz(;h0#}oB|P)T$F`R>QG#s_FD5?vkv z7`i0#i?nyZLAZnxI9t`L{*KPVj@Ty(K~9_3G*G(2b(!R!G@wY6XlTlX4$aMDjI3n# ztX16_Xq10V(Z<%GmMPI;xWRbTq1t%oR$-5HPKq6$yeUgIolCJ2hW?ZOnNlXv7Cmp< zz{9}u6sxq`{Srzsq2RA~B&_cm0E;P>fYO?LDMokEN$X1YvfIE%EVE+?EH2A#rmu?k zEDV`a`|?`~_r%21VrHJ@kQXc8tRdDP zvw7Py%XB+tMlO=J+?sdnlX2p)IbTn?$dz~HlQRcP5BQd_ZHHzR`LfK6OIpfD!h(s- zd3Ue%v=R+~0>aU3d2T(YH!CkTk-uJlZi{OohZ!+`qsgU20`nEYjk;h? z^W|ygI@H!(@En&<8HUm^DB<|>mKEjJxccE-?1 zx%z5>>FQ1J)EV|x!^=?~jI(3!cj4Uk0VCNB;duxkOlL&I3J5d-rt>+^4D` zUfC783IU5)9SO9dZNUJ*A&}v*^@(gYPs+ToO||upv;B&(4lS!Yz2$G-Ic2U@cW$b}M^$o5F@hf4d+%yEsqJ zef7o1mABJb^U}#8xz#TZ^>oMAs0$<%s(X0ey1nWj|M)qx*;b*d>9b;)O?DXV@8J>gnV37kG}HB0~LOH?G3Z{YgYJ4$jFV`c&ck*T8en8r*Im<7aF|-V>(;p)rMPLu6ZseiW zzBet2I_dBtM=C_)o1t92ua|&MHnn{=I#?(}Vd?pv>@Y?(*&A%K^Oz9e4M8}^Ko1Zc z14QjWAR@q}2QNc4=v*csXwyTY002n0HS0?gw8gB^6% zuhrMnb=S|-*Qa;ax7W9Wz~q?}Pg^j0FNM0gJ{Q0^?dZrs4t4E#jD>@RMUJJUq)wFs z+*x5MVPSQ%T_z!ULy{n7w2KWp86zhlvA^^N_zZZ2WeO{lD`>xlaw)U1zOlZ3|H8HB zrZxM5@hsjExji2xPpFW@tWac7SWr-cSdJ*g$mgSPfgX@LVFU+0hO$5}1)ES`1CgwW z!Ya@h!kQYd`VnOwM4Mw?B1tgdXA&=HOIOaA6^Y&wW#>S-2Ffyg7&His$q%cbZ#hT^ z%-RDA(+aX?uSG&a^7luAN*a?e@F+wD)kkB3K_$qd1^`B8aO99~BalM`PKkkF&pBBm zg3H{W!$$)K-9`f7ONl8~Xi_5xsvFmi{FF|^ul31@k(lu@g(IOO!6$r)7KzAAw8B_h zDiQw+nfsuZFCRLs&R(^5*t}9^@hoAX?qIXyEXBbga4P`0;Vm9Ut|FZpBUQa{|LEtP z1)cTj>m9G1DxBhSE41XF?Ba#6FTVeP(QQmcj@C@F%Couc^P(QPxPC?Z>f<{DtK-Ab zf)2PIOG%L_d3!Fl&&w*0b`D`kYl%K6Mf=Ayf-w)KbVK7nM_j(Vwph;?#T>LHzBr+8 z*)Z4{HHsopAID+KvMtx{32hu3Q0P@=thnELz9JDEXSzgSKcqErsGm1=FZHC1S@T$! zG?I6;b&@#D#enKjl-5myn`2u?$j5P3g$J+PzG^|tyesADrKFnG$8V2z=qtXX z;pb-}z!;!6&Y11P8G8N|xtS05HnIWrJ_^n(GZQPTUu?U){Q?6|b|6Y)9KtTRs&r#gY1&T?HinnQ z5z6TQC~OVSXD^mj4|VUAx_img^<%dxb+GVUKO|Q^8u`)N#mYs6z?qwc8CL~@c4r+$ zmab#?V>GSmHVt4C^Fq&*M|FbimFE{WW}2>dj_hrVFxJeMr$MY)=swE8!evgMp3(5L zBXuWJM6zs%-K6uTo{0Yt$a}_xQDREE@Xn33_c9SnwR3^Re5 z_<(S5qPl0JK_6qJP*71&a8MX>nW75-u^~gch=jDQ2>gx}g&{h@6G>Y!770m`6)7GN zDfSovtOCMN!lx}%lA0viBqStfzVu}U#0E>R7oz%058R$nnt(!j>f6&>>g)ULe2`~R zP?@kF;G+aKH7fWK`6)@f#$CLNa&&>jDIIOPs3Hyh0f;{Ex*%Q*=k=kk|;K}uf4_m<+wcuta^_PN7wh#d2FDA)A+r)V#JtY-GNu&Xs5=6Hjty30lRM=xS~mZyQSmis5oxF_}wd~{uml3;Msfk zukc?Ikhh5p{Uo`Pc}K3dX775f!^WH<+6s#aB#27fZ{)q$wUmboH>kt{t<32lkZ8Ow z!uYt{cgtwfYZ6o>@cU$iL6*c?1^pW~yNV7Af?)qGK6WKWKhBKRs??Bo*L>0=>Sd#m z)X+)F?`F4hRW=XEs(iV6uP}MK8@-+vtOG?y8=0*)kR#+2V6`I0psKmlxn)&J6oEsR zN3+wV&l+|Vjfg!3e;&cMKK&xB!cQbZ5_TifjRJ-A^bjd!!aM9P2U2m zWDTLZw1VEdX{7_b{WBdh*{|cduLU|1e*ScwTchY{$J_Ao&uJ@Ph+NCe1liK2*x#9` zLwnIW6B;hopr}TSZd8o2xzH4+sut=%cU6g|AMc0Zcqd%TT&$!GF6i57_*~+l(DiD_ z;@V5J7~Fvex%Y*HKC1#b{%L3bp4qP!Ty#^f+xQJJWCFO0F@DdbL4}dhH#Fk;VJ9( zKXZ;aARr%va}-x+JE(@+R_A2I$P*nz)X*nEYaXbJ9(qaF9vr~w1EMz)yKHwOJa=A} zP)I`z>#_D#6t8O`Dagf~)V`V=UoI5ArXp;pZ)_Cnn{a<8FDcRBzqwGG=Dg?GI|5x2 z(FmAFr9+dKNc52`f{E)}>`9bNgh=2<2U$;ksK?8%KyKYjQp>9JEkwQaS)WQX|Ai=`_|}EU)yjuUOHfQ}u*R<#O=7 z_V;&foTmv*^9_Ak2rK|m4~0=GIzrgl12k~MwTWPNo^_@v^GM# zBvx;@xyS5u6N{`u#wgRJbWq|7=9%NB3$aR!)_Agnm<&nMcJ9|Qv=Kho;$u@SHNdon5 zPsl=IZVit%nlft-H)m!El`k#gG-TKPq(s<9!p*Ndu;GrApr&UFrz9naY znR3m^`mbs`%=0&h1+@4L;-J`eK~c0|)H&j7UeIE!Q-<`r+x<`EG_~5e$%o!|&Rd*3 znI*!`*rC6icBT63@mDpQJ=TMvZM)ka`}&K_RAgGt7O>7L6ll=#O3K{%A-=A{0;6qRDH0t{a>7uckc?c`rfUmE3v@ap)gdHQ* z6_^nWHXPKPmw>K(iD|LEnLIs{ij)Bbzp+Jz!z7N26hapSgh;MP)B+eHQjd^igrFik zE4uk#1odw`t341X83Nc~k$?k$Bu)@m_msnIM8*$xS%Oq@vOcW__Uc27nGQa_9IrSY zKJygS@W>ItEfj;s4;n&bW4LC{s896g%n}{$E_OR_)DwQ?dLV9K1f6KPXr85(-6d32 zn}U-_+{FIyXm0T)!Nj{peqE~RjGf3GjJ2!=56pdYccSYIAX@eYV*YOR=RHja%e@k# zYreXS@hP$K7tYqwojneZYD+a_v>z?)cK=BX51-9en>mzcGT{_ElFmy1w)$J-$gnZ}pHCmSP`o$89=UjevRm zb+1nV&~d@p59#I4HI9Mu2m4_FRy~jxwllY2hnuTsAka&=6AJk7cUs^+d~*Zbho@@z z?!(^(e0cHaEy$OiqWpB+T>*QZ7li?^v*Q6X`Zksl6~_@uQ7$(sO3-Z-R^X3@bg?Ab z%=pZl;o?Eh*o)ZN*-w!aEp^yxdriA2vn;lE)I{4sxJkZ`#KV1ksZG5Vha~qN&s(3HeJ4t1x&afe-1ZI2*?$#8vxu`OkvJ27>60$D|%uA_llwc zp9tX z#ssq4at(WnRX`p`{Vwb=EJ^MwUd{5UV!*1@0qrEkw&|nyXE<&WB#|qQDD)&#HhcZn zs?gY&wIzLw81#o7Wog>Z1F6CqAciQFF$0hL3LT%PO^#ZsA7CRN1+MMcrW}=6KF{;) zuIA}K+^$*M&dodEl1~=C9djmi!{jYsRSubRGfrM@WBywF|LqK`j;N%L!c^n>wXD z2&z~Iv10GGB^Z|o-9j4{<%v_B#&tD4>ljj3u|D-ka67uoW%hC}_6~F|D*&CluKo*l zgp5Xbqoa>o6Zk5|jSE?ZvrLK6{Ok|dN#(=n7>alK#kd0f(O!IB_R!QpdnZoU=;Ar{ zC3ZLMOpdnDB1;+$secu}rzPUVMGMh#5*>3Hm4$~(};r$;b^xl-b*O@wXd;0;&F+vtf z9@QJQYaqH=H%Go*XqMhwWMJ`(Q7358YBuUp6hz&K{~5OIl>9^7O*W zm8$hNWmm5REY!KAUQJk0QKbs>mic_zafKiK4KZhD0pl+QWt}?`)$bnWS4DnE&*yt9 zUa_(GQe}yk(MB?LED>Lx6Q4UCbm=c8~6Z%h>})hHw3O6a9o-Bei3r4??V0q zTtLd*ev@)zgp?a1q}&K0pm9|6xaKQJSHZM$eS6A#*Pc&>^xYRsx-EcO0z{%Mv8OP#w zn?+O&CvnznS(L@DN%x&){h#$D;*tM%U3!LaAM;R&qNP3PCJ|4pkvsx*qy1%Wl(*U97QLMKPXb- z*fSJy#QTPO#JQmQxiNK`jc7-{RRPE`h^EP# zRDz+M7qjJGsL_bssfr6hK_Bp;(g@SEiP0qZ8WKhn8q^lzf_?}5%-kGr^eD7fTYY&t zaU_LovwM_ifKj7?y926ANB1Vy^hDj&<7TfLnhA-+iwcM`R<Fp(|qz^OEPNhScKK!mn- zMhG$@9t^MNJ*TbFzYFp={8S(y=ij8*3?aoPa49Cbi-=&}{Vv75-=+BKIZa*qryIj(!g8ujswo^$aX1a#81GRL6tGNx6HM^$1hALM?;$dF^ zy{0!?{^Z`e+b+vldFp{1Q}@PVZgzw!m+FYBn_KuoLjmP3*=RYNw)YEUeaZ-1=MMj! z*`AK)yD5!?VWJo88k*&E*5eB`aN;qR>vP`KPT6)^9+h#eQMP-<)-pX96QAZ!+ZY%t zW8cBC&m;PHx%8#~XK{HAz|>EipHYXzn}Kp1vSDKtU#e9sk0D3O65x+?Cb;89w3Yh)(z*OT8QHvY?t9NLj3Uzq+fnJgngqW`BawAbMf#)uPfMw^dW2sCwtz z@`?+}k#IxXNwxlUH7rJ7WAEq2K?Z?fR92SVqxcl#{3_mt(?S0UCyg!HujE@;UP@tNx@rR_3 z`t-^^4%{N=ZaKal|8CH}kK5`g1;u3$przxOYNG-96-Q|4LIe|Dg=-Uzz#sC9{!M<3 zfvpgLj>Z9l3gH4g6vLqi=;%N|N4wp**xV=zGrr1<7qK?1qo+Wsj&#MZMJM@|1XZDS zsD&g3rnjWuv@LGZdtIoNmGt7qEnal$CuZTEwoYbCHHKPfl<3GM5_Co4Vq&$KLrpUF zj;YCaZqwizb{FR&KjFaj0unsJC43_(RbsK)u!(z8SA^xsLU~6_(ET>A>5Qz_wjM2v z-r89`xJAaqm<~zanWAufVC5T_cO`+^g-1(G$korrd~GSNb+`emVYSll;>A-P+(zf( z%3%4m3&BD|6HZ;biF>M9@s4&{T^N@i3ValUMQVI*t|NNOiS)i^-=J^S+4K%Mas7Q| z{tY!KPPsTr`uMAktd=ov&_!EiAVLH2LrAd;0zgDBu^Q zmT(u|)1XAM8wg$OO!-Bc@4DB$z1?jZ)1`NK?DsV++%JW?dlZwgP7mB=bL7+78oHq8 zUh;nMapy?pF4ubYX~^lk&-043t+H94#C4(HOm2(wCUKkdEA4sSi7GpgWR7ia}6h5F<=&EtcRt!%l4Xm)q!fOGwt1J|0>xJHm?U zh6S0WrfiXv{7(Wa9>(D5Jrc%HRVX_4Q)&8|If&EOZs((rzUm*Ewd()5f2avqwE!Iq zUb%;gr%Bd#@a#2i>dBljP7LAH@ml<%iowyYIkFr9C~CbHsL=p3Aw>9`3g!Vk!tZgK z?<&{?p@Mfo-&F8WOpPT39#Ke%*`BC%|$0oLG{VXioSc}YCg@VU{9sDLvLj_+6KOsnnh{7Fb+V|*sfiiM7cW(ewL#Kyoa}<9;+Oi%{9{Er&sl2U%H-a7X)AK~NqsB7 zrjJ*2)OERt;+%{8wEJ%AeVp`AXCdPcjHIZ6Tz;|PezIhDSkN|JCA2qjB*ISBuihG4 zEKEA+54^cGiH_>*GCj@!(b4#fxyeG;O295R&v!Yjevq%ZZadexc<~H;bKyE%QUiXk zZ+Y#0k<{Px?KiNL|CPQa_#dQi`KFe}W{0kQaAN!r-=efJX5{QKIK}8)U2O;`+#2+~ zMb0#AX{V*tUb59s6tDGi-;2(iHORQfCPjl0$%o9d<#yuDHomU!isHz1b z8$sd;l-L5BckKSW;HD$RofYTsaHMec9@ioTn%1Ur5WCarBy9t zXWo01TJ!OCTaRa__-?EV&R+7Pn-x=Jq?FuxyBES3=pNRr27h`eFTuR7pGCXF$XXSG zu$k}9u8$H%#m^3yP%?9_-(SEXJHk$y zp@rSh7L__^WUUF1O7K;LtQH}(1tLwu|MiEgV*O?=+gUw8)HMNrD*7#~)r1IZwIIS; zE#Jdh*T03eZlWLyD=#iNC>{0d+30cIeVSH%=N@og` z&0tx`9aDEfkt>>_9|$`IBY}2EMVtYoWDrOdBalK^$AUop*Y{NoUM+!eEUYbTzCm>X z3@ZNSU5LUH)-B$y#rQNdnbUn`ib5gPk0*<#7n$^-EGEwsaV;6^eD|@Td=33I`}E0B z+3JyooAb4zUyPU@45H4AyF1wTR>}u__0?p*X97{i#~<(ba3t;OcuQqhrsY67R4j<2 zO#x54MNIHwJBQ$^{>V}ggx21-P%ff%-{iLTBO>N#kI|%*lj@~om~r2R*LD4TpAU47 zK7*giF=;3~Uy@#9e|y{VnZhGO?c_=Ko-9sn7{89?zHyTQnZzN3etbrdP_V1Ro)_!- zhvWOZ18I|3*EnGGmAsH7?&G_s4{`gk*dq?T4Gu+B`ATHf@49v5x9J5dpdG8Vc_!#k z@0DLYWLACcj4!Fid`9xC78P6yb5yE(F8%AQ%p)UO?1Bz5T{; z9S8Kkavc4ia-8RRjtlx19Cw<;B@TNc92ZoD>ECE*NH2~l&Ooh$w$1N3HqpsI-KLGuOXlC- zSwk~vhJ@eYvz$n=u}r$gh1?LY{Ndxv)Y%=NAZKp7whyHf{TWjANxXwTv3%g;VmuOB zm+R^Xo7g@k3NO;vpTm#wB62XpaT!&B?EsT7mhst6#@=`>Ys{Tn1tJ?0lPNsb8BO3N zQnYJo`I852ep+Te4~fsvbx&AVTkeC&KL*Ucuc>Z?b^~ys{!r5y@sKKn~g7)%% z8jiv_AC5Bq7vZQGAGBt493Ws%^Gm?K>n{TKR#Y}(dFvJV-N&T8JVRc7!!rbF{^2-E z5n$$kekNAb`#`xL6bSR;VL3x$IDymx3O${Oj*={J(}rwfA>9uJ=_LrM0tj70Ae4?k z2%!f=fQ|0oS2wsGfFtyy6b!K2(tm^Luk-*MCh7zFP=Wub2NeFO2Z;Q!{-YxBZ~S+P zU|0fB1Y@=p!>O)vj@2j7;@B}9%n2c&MzH`_ChHA(^L?Q3$^;hJ$~lfM*O>swVqEmT z3CHD2BqZco(K&?D7eVy?B3J-c)o`qkM}J}^#stUeeCZN0P`dPur2@gAQMiA#uF0X; zBt~#*&!a0TURv%SD@QE!iDB5DEq$P&A%jXT-j4tyO!D%J8)v=`qG=qXKQs)rSZR!P z3^fQ=qE}Tq^4M3>HD~AOE}I2;IB!l=Z`Qa^T^9F5p{2q=#+WC*#fKRfNr;P!QbLXa zj)jE;iJwXYc&8sMs2 z(0t{|V2N+cBnQ#9>KS`J%5=Bk4AdB#5n#l8A!86}59Y5q!0EH)$dvJ_(cmH^!&zZu zUSwWo_H{+R{+8w2lDd;V^OU86Kpd2?eQH!bt1 zEUni)_2p;F9R{(}yZ+m_-bk4%F@Q4p-KzsCF2v{l@74X!cy-c<6>0_{bqM=QGWEv_ z^~Y*eKoQO$lGTrXt5&t3mkJBX$`4Ob5E>mQLBqkpKo+kuxuz{xD*50Gw!)N{9pNEH zhV3+!%0^gt_%z+ZIGug0?m$U$)X^D#FmnQ*N5dLE8aa@@kx~;Pi*P5-i+#qID8+{c zC2=Uk^}UR>PYI!xxjW(Bbq@mz9T|t5V15LM;|LHD(QzpR|07B$eyL1F0fWZBI=A)moQ~Ok8it+nKq)xqYgF&drdHUK+dAoXEd<>*|}!e%l)2reB~He1m69 z<|Y_IP~sfFO3X7Bw3+;`1DC&uJlbiPU06|Mor*6EZmTijDXlMjXZ12v%>TvF)+Po; zH|TOWLAVUQ-BcO7=)0zs#SBqAbk}0ntxLmZiRVP+-*bgPsBQw5O#vM}}l3m1(OF*xjC?-}c7naa^ zj8u#-#8C({WDAZg^J9oK-O4XU*A8KHh>u)NrM75pWwEglahgg>svxI+0~Qy9qF}jn zVK8hTm|Q`HTtnkC#>EW~8}^`~nwP$Klj`$|b<7hVLsF<(-J2Pu%n|w! zXRK*La_9I}7UL${Ib8>bO)n76oI?CiTA1F|+Xv<@pjTo@~$Wc(B_(*Xzwv94WzZ!jD zsO80x$f0h!R?RV?2gCs7S<)ZgEua%*fl8NO)%pLu+y9~ORsyl!r6S}OQI{w`_(N&H zYj!l%K(x;MJl_(}?38)(QE@z4u!aia^FCf9fNgw_X3 zB88ze5dv!ZFbbrZc(Dle-pTCv6$oQ9L@`ia6GhhljzMq%Cu#!|Az9B+g^N3fj_t`? ztOq_D=;olZyt$3hg%30S50({_>*MOTpZW?u$6YpvJV|XQq0NQf>BhQNvo+<5>!rZF zswN}kaXc~NQQ>s3VoY7>5K`- zc`Kf`ttz;wdQ~*L-l_4fotD=Dq(c>fU={*FLDkJ_o_dxnjo%WRT?m_D}r~RL~2d4$FGRyG&oACN4&R^WA z)xUD5sNwDuBg@`I73X11jg&>s=Z_9;f8tJI|8S>BDCbSgyBjSM$U}lU1o*|I#0lci zwNNyWVo}i>-`DTZg!<0IW~s_RLko)A0GtzWIFaXm!l`8P`^5$g@Ev||Wi}Wz_sxCy zu1YxUgXE&c>wxSz8^6lPK9DWTR-vIYWiX0qZN#eKbv z44(?53QOfzJ0>Snms)Vi8X8$PdV0Gzl14n;p_@n~U~2O|s^FJVd4sXcWJp|8VyIlk zq~Jmrn%f<;EqOLJ(6KV>4LP5*QIICS3widYfZ9ZeZ?2wR%)`w5*F^HBlk`Az)Z3= zKes?SYR%WHm`3q=xurmk`mDXXHrQ;*c9^E?J5wkD2f2VRftpc;eq^=!JUb6OV|GJA zD_E2Pdi@Z&h)NK zY&)6oKQ^swy(au7dh>B!@53bnR~KPX?docE&YRCR##`%Hw5UzRqwaaUYwy zwfPR&^W_G1rzcVt+#jQH@r|CDM=N-OC7UqSwn&@f)63rXHme!6k5Am;MH6>teCn}n zyOcD7cYLW929fYJzJ6T;9aF1DJb%QB8rP^hg#j6dW7-={iX@VUB9>U+SL9Lsqro89 zFvTv)=zIjyR&A_F4zIoxanAnRS`W{D6w5hvu;2)^y|Ao&4yQ4=CSbIex|Dfve_X{R zO9)NY9i5N&n1_0QQ$zKWH!M%rqe{oKWB&AD{czq<K>#C-tP&x?&dqVL((GBc01Er~})+eu*flY5vd} z$l(8+S)c}kseh+3{&{BMpQsE;jQ5wYa2~M}Ig$R{=-NF6D2pOJ=*5=b)sa|He=zR5 zBCUtKW7{I6lwaxndRFhNZzu;WkezhVoPQlC_ieAoh%|k3ae4qN-NjT+a+Zr__X4~!dd-n{7RE#Hdu*!vo#*{<)F??R6ECSx^GU#<4Rm!hBjx2`;!rg{ap(ykL$5iU58o_FnR~ z`)lLxuc5VaVENh5DAgA!FvaC-uD_L)H!u;8$3Kn(#i48aFLYs=6>8^e>fD(l0hKyG ze3W2QUhq}TkY=cM^sYd$05=^d4yx`$G+$7C0& zKT`n)&xLCEsnkD-c{FiQp=DXHCfpqvp-(}uFJplu?_r4=<@Z|oZpE|xb_Oub=dITK zZPG24W>V$PBQ)DXGkZhY-ly9b?CT$VZ}{DBG`)9!3$%i zCUjeT;WP9qKmYZITFVh;6Q5xGsnpXMnyOLA_DpuF5D|sFQN?Wr+=OY&xEoR?Oa2G# z(V87T4qQ#?#RB_`)oe0j*2sQy*Wl(>W?FLcdCn&Ol$y~@q%np*2d~~XzL+?vxPtrm+4POdI z0-Q~Wz(e>a9$GBF;GtybYHE+bV+}~Rd~>W`*#HVLxNZ!|Et#6Yw~!ynRxI6cpwOI@ z06hqXiw1->(xaK=rhEeO>hbPC^hN}+%IT}ed9+TUGTlIdmUC@NM7qKoKezge?Q_$1 z+-xTwF25hmQt|fj3D!`)K_r(^e?qjS&Y+bt%vURRVa}mghAw#B@FOp}oX@s#c*__2 zRnfb;ft_AleAr46l6Xj%%Bh*lukU~#+CY@6N!UpfS2j4AI)VS&LR1;vlA&$HcjrFw z&J920s5>mHIyFDJ;T|I7d#%H+#oHicLMYc=R{8-CJfEWS!1^abiy)BHcb~}~u?>9o*HUPTsBWXWf!k@#o$oV^e%4?8 zMJcq+xg8JROQ9S7^L`QEiz`4SG(8EXP~^9L;1QC*K5$!LANVkfU+C=b_JLQT*6$kk zd0xi-yrT>|1ok-)kamkRe9tx&!?R5o@NAO}ok;R0^3M8c7f5d)7igwVND4p!3yux4 z2Lc-|cs5SQ9FEP8&4M`a3N@-<5b6ViJm79nq^7bWnL2^+Qw|=eF3ZhjnsBO&(wFbc zjW`d-Rc`oiRJIfk;IVQJ(oqu05GghJ)fk;E2l@yXMKYkIYUJ{UYcyG_xC@|ogN z1QL@yuJ*p5&aB8}f04Uw)9|xb8I$A=btgm?Ub0P;d%RyKw;Hg0w^}pra0W_0X>IDy z5C7uY^w4t3OE~p(Tu4WCw(;vS$;Ar7iju+}6XUdHX3rK+)guYdE>GNxRlG$+JLOt+ z3bo?nyR%+(JiRyQy>e_*-Zr@Lu__(RDlg`5;v7)ij~5yEL^idW*ZyVl?2L|8z!gPo zs<PyFMfzU$^gYI3L0m(s|?d_Wtdv zW5@E(w&*_SIfEai2UU+g1r?R-lUD9rSaNj*&WlB9g!2MQJc1Vx0UF@oVSpFTZ4>@} zRxt=f5B$Lt!8-qWpD7@L|HW=Wpx1x9KMv^nPxprb)A(2WqJT&Lv1{k50--Qo>_Wvko z0kr$)&N|TeSNHVqoSHv021fjszAVuA9~#*}%YSHo0xkc!B?gGmzco#O4KkK+?f|hN zG#Fw>HE=@zucz(*I351iL;iuW0UduH4UYZoXmI!m{C{>>KL|n#rv&}BFd%fLQ9Arc z`yY1GxfAF%fjThw!9tO-LW;)l6X^CB_@eUP)DjY4SCdJJ`obg_(2Qmd>osE-TC4B9 z=8&+bwKToLyJ2LJHLM*Hm|{hHL|E(Ai%Sz4FZcbU23{s!Y%5)y=*%>KX^SZ%tGvmugqE%FbWIiS zEOXY=vN~;!Ws1|rX+$UNt(||^s(Nxua>_LV%LH(XBQrz2l8aH9#_h0dGwDj|K&-2D z?Q+#pohl6-Rkv@xJ_#FY!sH?&pr^-G7ki~(#?{BY8doc%qh(^~>+{&Twc_*4zKcW8 z{?`-n*@v`|btuSuqSv&Y4Nj}6rmJ(+NBYQw^1kv|uzBSLW)*7Ic$e(Pe5iSI>Uf$? zZ}`bfv=L__H>XBH(MGS92z#(Mdw$iCLN>`cr-pXL8dN9YV%;~%_B5C2g)0dm&Ye$j zbsC5|pM^CnuW#ScZ1i1i|#0St8m!6N65 zoHJ>Hv&erqB7xw4!4V01-oOzFlbt?ItAIDcPAStwj-3Y-RKly@`hj%K?L;uaE3C2T zegB8($`+c=#IGyctvrY_@+Fsy2QhCTW%UnDc+}7eF-%JZkDh< z!3XzQC&o{x!@L{Ij4wmuCGu$E=c(vExO5#&0Oy+=P2BhM%Du9DF(UoK78F7dX5~_S z*ZHLD!{?iEbi{mxgZy43Qu5a&>LphlRtblt{0|7W`kh$F^FXwA;cP`W(g#RnnpWL0 z(T7203VcGrY?NNk`4!%!E@|36h3bx=9l5w1Y-?3X!fE zodl0aAz&t(`2!HUO%MklLdpKw`G@0k z@beEZoj?Du`|K~m!f{kUSlATplQSykRzy@nb06X8RC7lpT0|cFf3|z_Am*F4tH^O9tQ^`pr?jcCMmC(DC|Bt=*fNDD3`bGgmSBfAg9g$w7L+FS~?@I4gx`5Qs6%eUP?+OAU z1f&J&NbjMC9uy(;9zy2^X1+NyXU;kAyzjmDyX)RNYq2D{cK#;2Ji9!5|F-}6_kK7V z^d~e1?urHZZblKLduS*2J_9n$1h6ff-tbf=OguOq@fFAT*3Co;;gNvGldb!BSTa8w zOE2|K&~#GkP$>pveijsIB(~)muImN9%hSn&uzUDF>F#Te-%xWGGBikXL)}V!f`%{pBs&$*T8eEn_JyoxYVihd`nz{H zT^!Hl<9)7Q=Q#ljHBl%~c#?eaXg~HWHKwPJDTdC`ldA!TFV?i5Z%u_rq86J7vC824 zgM$09jT#pYx$;}R7J5Q++ai|WyZ+?y=YFE)QsY0K6mp22vG*%A_3WxFH`i124o_qA zfzIk<-fk;}Gfi2dJ?tx|LD&{cfBJVktij;aQ#em(}31;Gof8+dJebUlu+phvgGI(huT=Kh0wJ^ZtKr7HhKp7HWU*;kJzSKd|sfdst~HN1&pi@lDiP1NH>fBYF+!)?Q4@gGd~QVG>ch8Hb|_CIA&fLH(YcJ$7y%e7e;SaO$4`#p}g3%dM{?fkC}d19>p zh2ssq!trjp#PNoiRqaq#P7NBg&F$xyDcg{KG;4q0uIZ*Y3|5WCw-OojE$du;1lalc z2mqZ!YMipx``&Dbm>=t^7jc=5=B@%w_=xYNrfyANB6+jjB*3S6z!;?7%&zz4Jw3c` zlKMsbZNTA|jT0KC=9f1_;rMuF6{WSgHN;X?PO`gdP9w&uII7IfZn!$YTr^6Z2^juN^2yGdHmD*R%G}v0IRWdw!%p!PG zKJ!!JbKU+d%7e;FEbsBs2zJK4>rs4~PjUpGKf;h=Jx61CYjbJ_O4>7LZuCe+r!veB zNm-WpFY7&bv*GG9$~k_?ge#3J=;!yW^)2vpD=cU)`T_ScJ;+5Jd$}9m8f7x-u49|G z^-e!@-++4s!D%dm^<4$r-TV9?TI|8Gfi5i86Ui-jwC|{hh&?}jgV|PfAR;)H_v9ja~+aP6y+J+08e+FWM&vvWZ0MZLh6M|L^P^{fw^L8=#GBO ztE%m_UvAE>>M8DM^t#{K=YMbrGME?-f1OW7Baxut!=`X8c=LFlqenbH@$&NX^2=a^ z>&a*aq=mRYwlFcVj|mBMbF-7RFmYpkiN-R(y+)=@t^od5s0I@lXaF`e3>yi1loiJ{ zhesBVZ1}oJZsAbi;B*9!d?x{v(Y_oAe^uA|^M)BV+RMHl z)DUT9SGnBP{Dr6GZ;buJ`~?l>FUa8F zcaoRrM4{59nO$x#4@rMu2;$2#u#GTihP-0aNK0VX3cdT0u?L%kS%|Xi*;h=O;OsYH zM2vz5JO<(r6+lw)WGZT@p}IWuS-1wTlSQx-7S=9nOzh+Mk%iVsMl8OL3{H;04w*xa zII|by*|wa7!F&1FD_??5AQZe$kpV#)d_*5P7>m)iyFc^qxWs% zNR6+6)-#jNTBJ!O(l9AZVml}T-*Z@)c5UA@dbW|RnF1bO4u}t1IFv}PAt>7On&Fs% zm9(5qau@|}l_D+<&Tz~o6n~=C^z4$GzsC^wLZ8mq&c7eT`(5XL$8Q+ol9wadl8PfY z(dZFq4Dl5+;}?)T*a!3=-UH1z9%BOR_7Al7lrIBiGTKzvIYhrIKX=s`TqP!HqDT&T zdD}@Ads>*aeDhKF%X zXb^Eqajl9)r2u3BtIx|1p6bsJ0A{g4+z;_zi~rB_a8GRUsKk?ILDF4GsMO=6jaaEH1bulZwvZcXogD%s3)5AQ*b+biB23T*d6HKGOfM<`5m)lYi=(1(s8cM-1C01(b!~1(I+Yf zRES#W-0Q6~aSb;x@n~48T`42uKG*$8t!Yz&-W2*z!bjgi_>}xD{QtiQ|IMZ4|B5bt zwLBKR{S|Hbzsr1n+Zp@Y3v`-ri8KCJZAVE=#~)K>+{z(tXwYmx>iX<)?Vh8A8JN9 zZt6M7M*#%@J6>pzD;~*v?nguv96nXovB+QlAnGM$kOO=a!O9@f=h9ffCXns=rWn-Q zf&~&X>36$H?-;b*8js${sjhaR;23~h+sq*4rN^Sc=nA$~#jNpQ*I^3Ax39Q$qrEp3 z>8Nl0)@wp_h2}mPZ?AcJ#LL$PJxi!N&MK6hrFOBO5&@JV+ArttHiHWub@0`b>c*#z zX2D*mu6@C$W#LQFQeq~YTc%*$UB)}DdctOP#-e~H6d z=&6NNU7vZZ5B-e{5l%?owD)s>l&A|Ftlqj1e4~!;5Pk!5C6QbeN-ti}*_!5L()QI- zkK2AzEYe4q)}EZwrAXU+cQ$ZSSdC{z&}vDH2J3OZw2jF(EOXQR!1O9B|CPK(6*+G8 zvPJf{aIMLegGJuCf@#krl+=)CJnOQ}?#Xf4Q8Q6fG6cl+zZCu2?|KabQ!gcam2X}3 zUTSFk3cO5>i6MEJ0YtxT_;qYm_r+w`Z9+#O;kXnM@jyInJrWKAjt3U&Ga+wqz+4+q z#4?L=dJp{Y{Q(;)QHc?J73+z;G_FYn``G^{EQ3J0pLP_EkJp0NV1+Th@dNX zJ=XX4w_jIZY@tJ{vv^g6%knS(c<0UyfUxJ8p4xt+7Lbv5s7gQQxEqOo-#7@EI(%m_mmoj?V(HYx1N71xgotfEly`Xz9!|^ec3&hbiAWXq4#+dx$APGSqO24Yv zZ^MEbSG+-=S0=yow3b~Cd7$xu$=qumzv{sK@>a=#S=>`EPD26n!w1G1Eb6G7L>DtZ z9D0C-og+T?tzZIb152BlcyYXMiYyi47kw8O>9KK9j}LH)o8Ie#)gdMe zg?q;V2GQ4pBg#Hu&Fum!8Au<;l+1_C&zHP6SHjCue@tI0s1-^bBJG8JpT|#+WW7P~ z0devt%Bs*N4(R??uT3M*X3~NhuXjg+dpsWb^Bm%sOPsej^`EX2fV2@&RqTr=isF|m zpBS#V6aNoWdxge7|5c-Y;f}!-?#vbjM*Od)w)*c)jqP`sZvWcU!bAQiriQ+o^pYX| z0k~WPMf!&|P}cuu4HSe#QUnENC^u&hBkE@qfUb!M-P@s1ip>OCR?U zyT}6LnVn=Mnms1)?(U zJWhF~k;3tI{&i50M9%W(>83in8)A5OqkO&?%ig;7y_%t8!oYpGaTFkF@y%Q@TxnRb z)}Pa@g$@u{qTl45*%p0vu(G(nFi9{Wg6T0YN(eRX%B8o%7QQXzEL5q)+-hN)*2GCi z75b|;f2<()3_kYSh;Pc5YIQy%ng*IMP%cGFhf7S6I9G4RT-^o zW~V6RCJ%fWdhtzZU3*$jk3(-l4?nAKbI#`zqe+>7_})@~`Xp1=jVO{f&U>;mEzD*F z6kN;Dey~^V271<-=heh_8cZObOe4v>?DM{@Lagv0VsF4t0j}pck4Lrc5q`m3(`F{* zgb|y#7B^NN2LyHl)>|K72JL58Vo?ccl|K%_Hg0pJf%Qb2l@1d;h{^b-n8#Uv&rI_h zB<*Bu`|H@8^~^`of{quT&8rLB`VY(&@l%BDzR{TEk-c(fx9KqJxbc|(@i^q#SZvTs z=?>{TVp6)c9qI?f7pgl%5;%eqv{T!ijTf^danQ{DZ4#MsdOi<_n7 zlOo~W*ch3~0wP2hWGf^2S^~EN3KL?ERnJ7IFm4GbYy?;$Jc6QVAuf-#FBEPi-aXOO zy=LlFYj}Cn3b~Za@2l()v6piBFRJWbzgF4D&{cLhS&~QUCO7fzo`g)`RPsIbP3*I< zEU5aZfcGW@NA}f2YGv^S`q$?nAB7Habs+3j6^fkvC!Q}N@XL_j>Qggo1W*Gdanx^b zy}OAWe2d>op{Zr~U`XF*?aQR|7Jagp0eEl3rf%%fdzp9Z>`%;M0Oea)k8bvlmG}jk z&cDi$VW(1R99gHyT#0+?7I(DY-gDgQ;OgGk2;uW>9KxiE|9~YSbwgX$?K#y)RZnfa zIstqSqleEZWi~AN-&z-tnDpJFncjxJit~@}O*l^uTsaCU;(>Mh1$p!NG;6q*rVF4w%okWP}PUKz+|fv#zoZ7vEmS> zN&mf(FaEGaN*F%Z@pYl1ebp(yBkU=C;6c8tGYuK+9CV6(7!-^K$HLmblo}`C%9^gq z_mSvvnBQ5`e<XQQY5loZtAlbDAL{C;vzfhT3F(kp2bp2ipcqCne*<^)51^OqvI(rA^~>T zZ}(;$lU$mPhpR_S>wG#3-c~qcdZ`j6k^%kYw4b-7-v8lb*FD19;3!u;QyM0Xud&#f zM?LA76_#pEeIC#n=%Vm5^ZUZHi)1NpXW`FEJlHes+eZduhV6z2D^?qC-~*Lp%e>|h@L@PNA6wX zk(_{iA6Lw&K-{A>vcYxx)#{>%GjJl-KOj(6sRYTqO8c+V70u^f&{R{SKo)1=mWcvCvu zFNc__G$`LH3K&)Pa;-1n!Hg9og2@xEBGN{yRUqF{>D;1y(XqWmS( zet#>Adx>80Ysoem6NCB+zv9U+IhB_~A8v?3+ysX(&Z}n#`7VGUpoqNifsA7ad-3qY zq6AM1YirAZmmdYbWt+vHV#JUpS(ARjqgF(!$viN0=Vm~C$c?&Z7V!h)SpEKbcLUKT-R=e>Cg*GK zdua4@f|v!-#UE;z0QA+sm)zH+aqpPhBax45?!77JT&KH*9jJ++hWAw_BCP+N53d@@&QnP=ToP#Co>-v-gqE#@YhQf!6j=6Wok#1bSnTkmT@jIjYdtt77ctqP&IigO)~vx4=G%?y=}N(;p0ig1^Jjx!iTs>u`wzZz2A_ONic=mWcgJrRWMrCS~ZKg?NP{b5+uQfQi9;g)|o5bO}`4DP40r zS~gDW^csM&?sY{{J&`?bIFa#a2%gsRyajEJ=)N=wrE&W?Hc^SzjFB-1k6MT`kB6X} zBALbT&5NM9-MQJix~*6ct6n5q@Le37?_a>tHWi*BLD2*RVaUdM3Fz=9wu8zZ-Jd_#QksAKnde&>P1Fc62ia`<~(CL1+v zCq~v1*6P~LpdsqEj+ijH$lH0_T$H?d+wdT>00rR3_=xaMRA?g1zeX$&wA+aCO^!CR zBp0}QUMZ)-Sf;w7JG!St{F6nqTM@fzEVxcleC0eE9DR-uMX_uoc(ClQQ#!B3X`qeg zk9XeT(yp$~UOBXpnaX zHoPv3j$n(}!8EdZr>{u-5(k$IhvpuanAzlJ&z-8wREAxn3X^0z<9tFn@U+8cI4n_erNZM(s@}-UJ)m7GEV-fmI5d#?m8w zJ|MkPgRrmX`e2R!I2~$Zaa9byO3)V{rJ?ffJ&^=-PbqYrC4?iMBq5sn3wdcb`{-(( zMLoyU@$>f_I7L_#03uhwa9O&7@Bvj-t+P`q`l&|?Y*^p!ShbU{n{E=RBFww;gya?iM?D@%njmB)xCOX2R z%{5OaRH1Q0Q}H3@2jQWzida%j{4$J17foM^Y!}u2-JDVcHt8Ur!D$MENH&3!eG|S2 zSw8k?Og>$si9o@3_m;}@)YJCL=SfY5xS!_G#_&g>pp79@=Wn6>KP?oWfA$PnYyT{i ze?ZvF{pK0a_y01tI|3WDZr{KU=99n=V}E|{%f3@fCjg7@kqxghhlaqlstXJaZyP+K zRvP`V7K_K*_vU#k5@x1Ml+vwFF~wcCV8+=9pDplw@dv;gJ-vrT^Xnp1pmaQPjoc7s z-pHJ1f%soF#~ogxXlE6Rn?w4py)4i*-jSMfc0L=>FvIVZd6*3FAP+OwD{`h~U?Vov zAi@sTN*gp1dNv+&gV`e|l8?DOVUT?Ox=1czfyWdB?gogYgIiDBVL3zx?k>5Dqj#K8KwXM+B^;lo>**0A#0~+ z-<#H=N_0%=vXkK;7;@-a+tS0+JGgR;9x5xkD(xLe%!`3nKzWUDC-0P)_$$_hnh?{ZY^UJMfsmwS8v=0INnTtt8~#fvSN1_x)(Hnk37JS)@upoU1! zjOr239%CF*Es~opz!w;ND)UF z*vOR4_>`XI8@W6=kW>2nIa7-HM?#Qwuw^-)aY}!#O~VHdFfVpTb>v(4=8f{uqp8(_ z?MATBiV%*ESz*(rHWFN|e6Ee5{upbu=x&fKaZ3x!hx&N(=8?BSe4&9julnl9{kBs! zTqLCeGihhKrXHb&rTmt&VKdFmEDukq!s8y?Q(578d|=64)i%7Q$6oj?WVLeMmp|)R z@)ek14rDOPvXH8C=dg6=y6^V28>b2Ax&r1OMS#}*T(iGL@V{3C30GE-a%BZqOB6mX z{Idvt*D-C3iIMV)8T__k^E=uM9=4%x-QVPeMTNd=dBDh^kR}*Ru2c3}2c{RvFo82d zKyh&M6qn$pGPP_O8phtIyt*3U!`RlsuRSxJnL3NI@l@-|$OYKAKIdJs7Fk2~w^Cuz zaYTt?>681QR2gjRndz12m*O4;v!?=ltBJ+5e0^%a&Mfa^+jOq7xVVC~gdAS1{LOrhQvt@W zrCL#4+7pSL2V-@$Nwd%sYY3{=c8quEhYV@c5@S-Fg7)z(q{$QmlqDz6F#Z{1WA~8Z zMg@?#+c!OIUVSe_&u>psRv^^%4k8`KRWIe|5ouru$R# z#J{Qcqe=hNJn^r}+-SPLRPaU9{i%84Ulu;mg#Vp7(VxiCO~3!4Hi#zw>xuNAh|$mf zx77|b@qa5T`4c_*;r}VMMAQHEIPXuy=!gH?XdF%a*S+YUh|y2}w^0w8_`h)w{fQd= z=zs4@Ffc?*&<2GM|E0<1s?)0o-HM8vh`u7_e~hPPz@YuY!{Q1p)K3&c3676$0?3r^ zl2lHEoy=w4=}@H6qyW6+Bv`e0E^saJBuiFBg@*vA_b%)P;F|e1ami!K+&U`JbX9NJ zxM(Y^m!#JNsGROTV@P`~V#K5hxJK%;>Xn5$%R)SPs_XR>d9)WmbQ&;gj65w;e=0dP z>w*H_51enhSX>0%?=1-QJUZ{)T_H}QZ{+GN<;mvC=AxIDKK*WbhJ>C^!7p%|j!;R- z)<@rWR|=Y(tM2#izFZu!K1#gUDd&DLFLs!?xxcl@Lm@}qKlo+t-TbD4@?>lO?~8D@|t!}?q7Ttxv_n- zn*Mw%@ECfrW7@=35Qr>jJk1SUl)u<*-8y}&wYQNTy*Jlee)?hY#1=6J4AqF_zu?oE5}^F!j|M)6OGAeT;W1d8zjYs>_eDaC0JUZNE_!Jm#rhfY9EJub3VV%a|KKbX47m-H7?J zasY9zK1}*jI#W4OxKi|u!jJ=Quj?@pjse(=4Vnc_?(Xbu3uPu{XjMGz;j7urC}pFJ zh5cwMI5t|Z&G#t=66x5&h?O;)EVAVsP9c7|Sr+f1bV%MD$H?nrTW3+%%wW8^w(vNF1 zt@IfaER3y{55Vvo-Lh^n{7Ef)P17H~$^(=#hzl1X-!q@h0Rx{Dllc3O5uK*anD?N#XKg zu>;0!6a0_$IT6-OuZ=c6$PBj}s(KJ5q%#FB%0AFlcLzBRs8MU1ANihhA7?FveWMnz z!mg{TZ^JjUy0yn2JnZZSbSRi+KN^v8lOAfM7so{=_7JQogQ3F}Hujfn8n=#MAC z9h}gruprM0zFtJoSgfHhShO*>jU8hXBg4C(vIYYH1;g+m3?BjBAvc(krb(82AC8%|cI9goVkPRfqP z&y~u0#1OlD9SK`6GdgLk8)p2EU7a2H9RlU}B-e`GWoOLZGu70<0iIRyFMPwrBJpN= zkFE(<`rl8EH5Q+o<3(sS+Q+#>04H~4TE=&EdHO%r({rrz&f)U;_s4fM$Hv3=8hq}< zyqyB~Stq4SNrxIcR^2w?4`|GUM%0>6ocT0NcCUChIK`K?9Dj7Pn~e?rEHe?i zaea)^!LHa)93t$5XWdVOB=M%ga@cDaAJn4bqZM~4J!=r3Qr&A}EUnBS9^cjL!M-ES zUfTY?n6_N)qi3kcy~Ir9yBmeD`uGFZdyEwh7ZOAs<|?LZEgnX15H!^;Zr6+ZoW5qb zj3p-@L4<`>I*!Y%U{3dH_R?~#tIno?56i2+n>g≺Jh2-VzOK7H2Tq%dzH3))?J9 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

SZ0ub-YC0PM~lP-UK-Y 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 E0LVIZ!vFvP diff --git a/L1Trigger/TrackFindingTracklet/test/ProducerIRin.cc b/L1Trigger/TrackFindingTracklet/test/ProducerIRin.cc index d695c7a28668a..f798a2e2b7cd4 100644 --- a/L1Trigger/TrackFindingTracklet/test/ProducerIRin.cc +++ b/L1Trigger/TrackFindingTracklet/test/ProducerIRin.cc @@ -91,7 +91,7 @@ namespace trklet { for (int tfpChannel : channelEncoding_) streamStubs.emplace_back(handleTTDTC->stream(tfpRegion, tfpChannel)); // store products - iEvent.emplace(edPutTokenStubs_, std::move(streamStubs)); + iEvent.emplace(edPutTokenStubs_, move(streamStubs)); } } // namespace trklet diff --git a/L1Trigger/TrackerDTC/plugins/ProducerDTC.cc b/L1Trigger/TrackerDTC/plugins/ProducerDTC.cc index 8f50d1c28604a..299b2aec86c29 100644 --- a/L1Trigger/TrackerDTC/plugins/ProducerDTC.cc +++ b/L1Trigger/TrackerDTC/plugins/ProducerDTC.cc @@ -116,8 +116,8 @@ namespace trackerDTC { dtc.produce(productAccepted, productLost); } // store ED products - iEvent.emplace(edPutTokenAccepted_, std::move(productAccepted)); - iEvent.emplace(edPutTokenLost_, std::move(productLost)); + iEvent.emplace(edPutTokenAccepted_, move(productAccepted)); + iEvent.emplace(edPutTokenLost_, move(productLost)); } } // namespace trackerDTC diff --git a/L1Trigger/TrackerTFP/plugins/ProducerTQ.cc b/L1Trigger/TrackerTFP/plugins/ProducerTQ.cc index 6f6059ca100cd..f5bc9f948e8b3 100644 --- a/L1Trigger/TrackerTFP/plugins/ProducerTQ.cc +++ b/L1Trigger/TrackerTFP/plugins/ProducerTQ.cc @@ -43,8 +43,10 @@ namespace trackerTFP { EDGetTokenT edGetTokenStubs_; // ED input token of kf tracks EDGetTokenT edGetTokenTracks_; - // ED output token for accepted kfout tracks + // ED output token for tracks EDPutTokenT edPutTokenTracks_; + // ED output token for stubs + EDPutTokenT edPutTokenStubs_; // Setup token ESGetToken esGetTokenSetup_; // DataFormats token @@ -67,6 +69,7 @@ namespace trackerTFP { edGetTokenStubs_ = consumes(InputTag(label, branchStubs)); edGetTokenTracks_ = consumes(InputTag(label, branchTracks)); edPutTokenTracks_ = produces(branchTracks); + edPutTokenStubs_ = produces(branchStubs); // book ES products esGetTokenSetup_ = esConsumes(); esGetTokenDataFormats_ = esConsumes(); @@ -133,7 +136,8 @@ namespace trackerTFP { } } // store TQ product - iEvent.emplace(edPutTokenTracks_, std::move(output)); + iEvent.emplace(edPutTokenTracks_, move(output)); + iEvent.emplace(edPutTokenStubs_, streamsStubs); } } // namespace trackerTFP diff --git a/L1Trigger/TrackerTFP/test/demonstrator_cfg.py b/L1Trigger/TrackerTFP/test/demonstrator_cfg.py index 5509a1c1f5244..ebe77c67db9de 100644 --- a/L1Trigger/TrackerTFP/test/demonstrator_cfg.py +++ b/L1Trigger/TrackerTFP/test/demonstrator_cfg.py @@ -10,7 +10,7 @@ process.load( 'L1Trigger.TrackTrigger.TrackTrigger_cff' ) from Configuration.AlCa.GlobalTag import GlobalTag -process.GlobalTag = GlobalTag(process.GlobalTag, 'auto:phase2_realistic', '') +process.GlobalTag = GlobalTag(process.GlobalTag, '133X_mcRun4_realistic_v1', '') # load code that produces DTCStubs process.load( 'L1Trigger.TrackerDTC.DTC_cff' ) From 457abe9b2a1897ef21d4666dfbe162adad3e4857 Mon Sep 17 00:00:00 2001 From: Thomas Schuh Date: Thu, 5 Dec 2024 17:35:08 +0000 Subject: [PATCH 13/14] TQ outputs altered. --- .../interface/TrackFindingProcessor.h | 2 + .../plugins/ProducerTFP.cc | 6 ++- .../src/TrackFindingProcessor.cc | 23 ++++----- L1Trigger/TrackerTFP/interface/TrackQuality.h | 14 ++++-- L1Trigger/TrackerTFP/plugins/ProducerTQ.cc | 30 ++++++++---- L1Trigger/TrackerTFP/src/TrackQuality.cc | 11 ++--- L1Trigger/TrackerTFP/test/AnalyzerTQ.cc | 49 +++++++++---------- 7 files changed, 75 insertions(+), 60 deletions(-) diff --git a/L1Trigger/TrackFindingTracklet/interface/TrackFindingProcessor.h b/L1Trigger/TrackFindingTracklet/interface/TrackFindingProcessor.h index 13b07dcdc8cb0..ef3cde5b97f28 100644 --- a/L1Trigger/TrackFindingTracklet/interface/TrackFindingProcessor.h +++ b/L1Trigger/TrackFindingTracklet/interface/TrackFindingProcessor.h @@ -24,6 +24,7 @@ namespace trklet { // produce TTTracks void produce(const tt::StreamsTrack& inputs, + const tt::Streams& inputsAdd, const tt::StreamsStub& stubs, tt::TTTracks& ttTracks, tt::StreamsTrack& outputs); @@ -67,6 +68,7 @@ namespace trklet { T* pop_front(std::deque& ts) const; // void consume(const tt::StreamsTrack& inputs, + const tt::Streams& inputsAdd, const tt::StreamsStub& stubs, std::vector>& outputs); // emualte data format f/w diff --git a/L1Trigger/TrackFindingTracklet/plugins/ProducerTFP.cc b/L1Trigger/TrackFindingTracklet/plugins/ProducerTFP.cc index 2c68231a201aa..3e76db01a7a1e 100644 --- a/L1Trigger/TrackFindingTracklet/plugins/ProducerTFP.cc +++ b/L1Trigger/TrackFindingTracklet/plugins/ProducerTFP.cc @@ -45,6 +45,7 @@ namespace trklet { void endStream() override {} // ED input token of stubs and tracks EDGetTokenT edGetTokenTracks_; + EDGetTokenT edGetTokenTracksAdd_; EDGetTokenT edGetTokenStubs_; // ED output token for accepted stubs and tracks EDPutTokenT edPutTokenTTTracks_; @@ -73,6 +74,7 @@ namespace trklet { const string& branchStubs = iConfig.getParameter("BranchStubs"); // book in- and output ED products edGetTokenTracks_ = consumes(InputTag(labelTracks, branchTracks)); + edGetTokenTracksAdd_ = consumes(InputTag(labelTracks, branchTracks)); edGetTokenStubs_ = consumes(InputTag(labelStubs, branchStubs)); edPutTokenTTTracks_ = produces(branchTTTracks); edPutTokenTracks_ = produces(branchTracks); @@ -98,11 +100,13 @@ namespace trklet { // read in TQ Products Handle handleTracks; iEvent.getByToken(edGetTokenTracks_, handleTracks); + Handle handleTracksAdd; + iEvent.getByToken(edGetTokenTracksAdd_, handleTracksAdd); Handle handleStubs; iEvent.getByToken(edGetTokenStubs_, handleStubs); // produce TTTracks TrackFindingProcessor tfp(iConfig_, setup_, dataFormats_, trackQuality_); - tfp.produce(*handleTracks, *handleStubs, ttTracks, streamsTrack); + tfp.produce(*handleTracks, *handleTracksAdd, *handleStubs, ttTracks, streamsTrack); // put TTTRacks and produce TTTRackRefs const int nTrks = ttTracks.size(); const OrphanHandle oh = iEvent.emplace(edPutTokenTTTracks_, move(ttTracks)); diff --git a/L1Trigger/TrackFindingTracklet/src/TrackFindingProcessor.cc b/L1Trigger/TrackFindingTracklet/src/TrackFindingProcessor.cc index 813275293fce8..380f060cf5205 100644 --- a/L1Trigger/TrackFindingTracklet/src/TrackFindingProcessor.cc +++ b/L1Trigger/TrackFindingTracklet/src/TrackFindingProcessor.cc @@ -42,9 +42,9 @@ namespace trklet { TTBV ttBV = TTBV(frameTQ); tq->format(VariableTQ::chi2rz).extract(ttBV, chi2rz_); tq->format(VariableTQ::chi2rphi).extract(ttBV, chi2rphi_); - mva_ = TTBV(ttBV, numBinsMVA_).val(); - ttBV >>= numBinsMVA_; - hitPattern_ = ttBV; + mva_ = TTBV(ttBV, widthMVA_).val(); + ttBV >>= widthMVA_; + hitPattern_ = TTBV(ttBV, setup->numLayers()); channel_ = cot_ < 0. ? 0 : 1; // convert nice formats into bits const double z0 = zT_ - cot_ * setup->chosenRofZ(); @@ -110,12 +110,13 @@ namespace trklet { // fill output products void TrackFindingProcessor::produce(const StreamsTrack& inputs, + const Streams& inputsAdd, const StreamsStub& stubs, TTTracks& ttTracks, StreamsTrack& outputs) { // organize input tracks vector> streams(outputs.size()); - consume(inputs, stubs, streams); + consume(inputs, inputsAdd, stubs, streams); // emualte data format f/w produce(streams, outputs); // produce TTTracks @@ -124,6 +125,7 @@ namespace trklet { // void TrackFindingProcessor::consume(const StreamsTrack& inputs, + const Streams& inputsAdd, const StreamsStub& stubs, vector>& outputs) { // count input objects @@ -134,16 +136,15 @@ namespace trklet { tracks_.reserve(nTracks); // convert input data for (int region = 0; region < setup_->numRegions(); region++) { - const int offsetTQ = region * setup_->tqNumChannel(); const int offsetTFP = region * setup_->tfpNumChannel(); const int offsetStub = region * setup_->numLayers(); - const StreamTrack& streamDR = inputs[offsetTQ]; - const StreamTrack& streamTQ = inputs[offsetTQ + 1]; + const StreamTrack& streamKF = inputs[region]; + const Stream& streamTQ = inputsAdd[region]; for (int channel = 0; channel < setup_->tfpNumChannel(); channel++) - outputs[offsetTFP + channel] = deque(streamDR.size(), nullptr); - for (int frame = 0; frame < (int)streamDR.size(); frame++) { - const FrameTrack& frameTrack = streamDR[frame]; - const Frame& frameTQ = streamTQ[frame].second; + outputs[offsetTFP + channel] = deque(streamKF.size(), nullptr); + for (int frame = 0; frame < (int)streamKF.size(); frame++) { + const FrameTrack& frameTrack = streamKF[frame]; + const Frame& frameTQ = streamTQ[frame]; if (frameTrack.first.isNull()) continue; vector ttStubRefs; diff --git a/L1Trigger/TrackerTFP/interface/TrackQuality.h b/L1Trigger/TrackerTFP/interface/TrackQuality.h index cfc32f8bfc8be..276edae490a11 100644 --- a/L1Trigger/TrackerTFP/interface/TrackQuality.h +++ b/L1Trigger/TrackerTFP/interface/TrackQuality.h @@ -19,8 +19,10 @@ C.Brown 28/07/20 namespace trackerTFP { + // number of mva bit + static constexpr int widthMVA_ = TTTrack_TrackWord::TrackBitWidths::kMVAQualitySize; // number of mva bins - static constexpr int numBinsMVA_ = 1 << TTTrack_TrackWord::TrackBitWidths::kMVAQualitySize; + static constexpr int numBinsMVA_ = 1 << widthMVA_; // number of chi2B bins static constexpr int numBinsChi2B_ = 1 << TTTrack_TrackWord::TrackBitWidths::kBendChi2Size; // number of chi2rphi bins @@ -75,10 +77,12 @@ namespace trackerTFP { // object to represent tracks struct Track { Track(const tt::FrameTrack& frameTrack, const tt::StreamStub& streamStub, const TrackQuality* tq); - // return the iths frame of this track - const tt::FrameTrack& frame(int i) const { return frames_.at(i); } - // collection of frames forming track - tt::StreamTrack frames_; + // track frame + tt::FrameTrack frameTrack_; + // additional track variables + tt::Frame frame_; + // collection of stubs forming track + tt::StreamStub streamStub_; }; // provides dataformats const DataFormats* dataFormats() const { return dataFormats_; } diff --git a/L1Trigger/TrackerTFP/plugins/ProducerTQ.cc b/L1Trigger/TrackerTFP/plugins/ProducerTQ.cc index f5bc9f948e8b3..59f09e078ce93 100644 --- a/L1Trigger/TrackerTFP/plugins/ProducerTQ.cc +++ b/L1Trigger/TrackerTFP/plugins/ProducerTQ.cc @@ -45,6 +45,8 @@ namespace trackerTFP { EDGetTokenT edGetTokenTracks_; // ED output token for tracks EDPutTokenT edPutTokenTracks_; + // ED output token for additional track variables + EDPutTokenT edPutTokenTracksAdd_; // ED output token for stubs EDPutTokenT edPutTokenStubs_; // Setup token @@ -69,6 +71,7 @@ namespace trackerTFP { edGetTokenStubs_ = consumes(InputTag(label, branchStubs)); edGetTokenTracks_ = consumes(InputTag(label, branchTracks)); edPutTokenTracks_ = produces(branchTracks); + edPutTokenTracksAdd_ = produces(branchTracks); edPutTokenStubs_ = produces(branchStubs); // book ES products esGetTokenSetup_ = esConsumes(); @@ -88,10 +91,11 @@ namespace trackerTFP { void ProducerTQ::produce(Event& iEvent, const EventSetup& iSetup) { static const int numRegions = setup_->numRegions(); static const int numLayers = setup_->numLayers(); - static const int numChannel = setup_->tqNumChannel(); auto valid = [](int sum, const FrameTrack& frame) { return sum += (frame.first.isNull() ? 0 : 1); }; // empty TQ product - StreamsTrack output(numRegions * numChannel); + StreamsTrack outputTracks(numRegions); + Streams outputTracksAdd(numRegions); + StreamsStub outputStubs(numRegions * numLayers); // read in KF Product and produce TQ product Handle handleStubs; iEvent.getByToken(edGetTokenStubs_, handleStubs); @@ -122,21 +126,27 @@ namespace trackerTFP { stream.push_back(&tracks.back()); } // fill TQ product - const int offsetChannel = region * numChannel; - for (int channel = 0; channel < numChannel; channel++) - output[offsetChannel + channel].reserve(stream.size()); + outputTracks[region].reserve(stream.size()); + outputTracksAdd[region].reserve(stream.size()); + for (int layer = 0; layer < setup_->numLayers(); layer++) + outputStubs[offsetLayer + layer].reserve(stream.size()); for (Track* track : stream) { if (!track) { - for (int channel = 0; channel < numChannel; channel++) - output[offsetChannel + channel].emplace_back(FrameTrack()); + outputTracks[region].emplace_back(FrameTrack()); + outputTracksAdd[region].emplace_back(Frame()); + for (int layer = 0; layer < setup_->numLayers(); layer++) + outputStubs[offsetLayer + layer].emplace_back(FrameStub()); continue; } - for (int channel = 0; channel < numChannel; channel++) - output[offsetChannel + channel].emplace_back(track->frame(channel)); + outputTracks[region].emplace_back(track->frameTrack_); + outputTracksAdd[region].emplace_back(track->frame_); + for (int layer = 0; layer < setup_->numLayers(); layer++) + outputStubs[offsetLayer + layer].emplace_back(track->streamStub_[layer]); } } // store TQ product - iEvent.emplace(edPutTokenTracks_, move(output)); + iEvent.emplace(edPutTokenTracks_, move(outputTracks)); + iEvent.emplace(edPutTokenTracksAdd_, move(outputTracksAdd)); iEvent.emplace(edPutTokenStubs_, streamsStubs); } } // namespace trackerTFP diff --git a/L1Trigger/TrackerTFP/src/TrackQuality.cc b/L1Trigger/TrackerTFP/src/TrackQuality.cc index 8ec26d809906e..2b85ed39bd90c 100644 --- a/L1Trigger/TrackerTFP/src/TrackQuality.cc +++ b/L1Trigger/TrackerTFP/src/TrackQuality.cc @@ -87,10 +87,10 @@ namespace trackerTFP { return toBin(bins, chi2); } - TrackQuality::Track::Track(const FrameTrack& frameTrack, const StreamStub& streamStub, const TrackQuality* tq) { + TrackQuality::Track::Track(const FrameTrack& frameTrack, const StreamStub& streamStub, const TrackQuality* tq) + : frameTrack_(frameTrack), streamStub_(streamStub) { static const DataFormats* df = tq->dataFormats(); static const Setup* setup = df->setup(); - frames_.reserve(setup->tqNumChannel()); const TrackDR track(frameTrack, df); double trackchi2rphi(0.); double trackchi2rz(0.); @@ -152,13 +152,12 @@ namespace trackerTFP { // collect features and classify using bdt const vector>& output = bdt.decision_function({cot, z0, chi2B, nstub, n_missint, chi2rphi, chi2rz}); const float mva = output[0].to_float(); - // fill frames + // fill frame TTBV ttBV = hitPattern; - ttBV += TTBV(tq->toBinMVA(mva), numBinsMVA_); + ttBV += TTBV(tq->toBinMVA(mva), widthMVA_); tq->format(VariableTQ::chi2rphi).attach(trackchi2rphi, ttBV); tq->format(VariableTQ::chi2rz).attach(trackchi2rz, ttBV); - frames_.push_back(frameTrack); - frames_.emplace_back(frameTrack.first, ttBV.bs()); + frame_ = ttBV.bs(); } template <> diff --git a/L1Trigger/TrackerTFP/test/AnalyzerTQ.cc b/L1Trigger/TrackerTFP/test/AnalyzerTQ.cc index fa782a814c170..35da10dcecba5 100644 --- a/L1Trigger/TrackerTFP/test/AnalyzerTQ.cc +++ b/L1Trigger/TrackerTFP/test/AnalyzerTQ.cc @@ -121,7 +121,7 @@ namespace trackerTFP { // book histograms Service fs; TFileDirectory dir; - dir = fs->mkdir("DR"); + dir = fs->mkdir("TQ"); prof_ = dir.make("Counts", ";", 12, 0.5, 12.5); prof_->GetXaxis()->SetBinLabel(1, "Stubs"); prof_->GetXaxis()->SetBinLabel(2, "Tracks"); @@ -169,29 +169,24 @@ namespace trackerTFP { int allMatched(0); int allTracks(0); for (int region = 0; region < setup_->numRegions(); region++) { - const int offset = region * dataFormats_->numChannel(Process::dr); - int nStubs(0); - int nTracks(0); - for (int channel = 0; channel < dataFormats_->numChannel(Process::dr); channel++) { - vector> tracks; - formTracks(acceptedTracks, acceptedStubs, tracks, offset + channel); - hisTracks_->Fill(tracks.size()); - profTracks_->Fill(channel, tracks.size()); - nTracks += tracks.size(); - nStubs += accumulate(tracks.begin(), tracks.end(), 0, [](int sum, const vector& track) { - return sum += (int)track.size(); - }); - allTracks += tracks.size(); - if (!useMCTruth_) - continue; - int tmp(0); - associate(tracks, selection, tpPtrsSelection, tmp); - associate(tracks, reconstructable, tpPtrs, allMatched, false); - associate(tracks, selection, tpPtrsMax, tmp, false); - const int size = acceptedTracks[offset + channel].size(); - hisChannel_->Fill(size); - profChannel_->Fill(channel, size); - } + vector> tracks; + formTracks(acceptedTracks, acceptedStubs, tracks, region); + hisTracks_->Fill(tracks.size()); + profTracks_->Fill(region, tracks.size()); + const int nTracks = tracks.size(); + const int nStubs = accumulate(tracks.begin(), tracks.end(), 0, [](int sum, const vector& track) { + return sum += (int)track.size(); + }); + allTracks += tracks.size(); + if (!useMCTruth_) + continue; + int tmp(0); + associate(tracks, selection, tpPtrsSelection, tmp); + associate(tracks, reconstructable, tpPtrs, allMatched, false); + associate(tracks, selection, tpPtrsMax, tmp, false); + const int size = acceptedTracks[region].size(); + hisChannel_->Fill(size); + profChannel_->Fill(region, size); prof_->Fill(1, nStubs); prof_->Fill(2, nTracks); } @@ -243,9 +238,9 @@ namespace trackerTFP { void AnalyzerTQ::formTracks(const StreamsTrack& streamsTrack, const StreamsStub& streamsStubs, vector>& tracks, - int channel) const { - const int offset = channel * setup_->numLayers(); - const StreamTrack& streamTrack = streamsTrack[channel]; + int region) const { + const int offset = region * setup_->numLayers(); + const StreamTrack& streamTrack = streamsTrack[region]; const int numTracks = accumulate(streamTrack.begin(), streamTrack.end(), 0, [](int sum, const FrameTrack& frame) { return sum += (frame.first.isNonnull() ? 1 : 0); }); From 4566010471f0e625518951fd55fc325c251c9817 Mon Sep 17 00:00:00 2001 From: Thomas Schuh Date: Wed, 11 Dec 2024 14:01:55 +0000 Subject: [PATCH 14/14] TM mux order configurable --- .../interface/ChannelAssignment.h | 6 ++++++ .../python/ChannelAssignment_cfi.py | 1 + .../src/ChannelAssignment.cc | 4 ++++ .../src/TrackMultiplexer.cc | 21 +++++++++++-------- 4 files changed, 23 insertions(+), 9 deletions(-) diff --git a/L1Trigger/TrackFindingTracklet/interface/ChannelAssignment.h b/L1Trigger/TrackFindingTracklet/interface/ChannelAssignment.h index d3cab07b1e3a1..7b5d5572dfe11 100644 --- a/L1Trigger/TrackFindingTracklet/interface/ChannelAssignment.h +++ b/L1Trigger/TrackFindingTracklet/interface/ChannelAssignment.h @@ -30,6 +30,8 @@ namespace trklet { int numChannelsTrack() const { return numChannelsTrack_; } // number of used TB channels for stubs int numChannelsStub() const { return numChannelsStub_; } + // + std::vector tmMuxOrder() const { return tmMuxOrderInt_; } // number of layers per rtack int tmNumLayers() const { return tmNumLayers_; } // number of bits used to represent stub id for projected stubs @@ -68,6 +70,8 @@ namespace trklet { const tt::Setup* setup_; // TM parameter edm::ParameterSet pSetTM_; + // + std::vector tmMuxOrder_; // number of layers per rtack int tmNumLayers_; // number of bits used to represent stub id for projected stubs @@ -100,6 +104,8 @@ namespace trklet { std::vector offsetsStubs_; // max number of seeding layers int numSeedingLayers_; + // + std::vector tmMuxOrderInt_; }; } // namespace trklet diff --git a/L1Trigger/TrackFindingTracklet/python/ChannelAssignment_cfi.py b/L1Trigger/TrackFindingTracklet/python/ChannelAssignment_cfi.py index 91023650a2e3e..9396453ef60ec 100644 --- a/L1Trigger/TrackFindingTracklet/python/ChannelAssignment_cfi.py +++ b/L1Trigger/TrackFindingTracklet/python/ChannelAssignment_cfi.py @@ -6,6 +6,7 @@ # TM parameter TM = cms.PSet ( + MuxOrder = cms.vstring( "L1L2", "L2L3", "L1D1", "D1D2", "D3D4", "L2D1", "L3L4", "L5L6" ), NumLayers = cms.int32( 11 ), # number of layers per track WidthStubId = cms.int32( 10 ), # number of bits used to represent stub id for projected stubs WidthCot = cms.int32( 14 ) diff --git a/L1Trigger/TrackFindingTracklet/src/ChannelAssignment.cc b/L1Trigger/TrackFindingTracklet/src/ChannelAssignment.cc index 0479e3f3702ce..57c66a53e1278 100644 --- a/L1Trigger/TrackFindingTracklet/src/ChannelAssignment.cc +++ b/L1Trigger/TrackFindingTracklet/src/ChannelAssignment.cc @@ -17,6 +17,7 @@ namespace trklet { ChannelAssignment::ChannelAssignment(const edm::ParameterSet& iConfig, const Setup* setup) : setup_(setup), pSetTM_(iConfig.getParameter("TM")), + tmMuxOrder_((pSetTM_.getParameter>("MuxOrder"))), tmNumLayers_((pSetTM_.getParameter("NumLayers"))), tmWidthStubId_(pSetTM_.getParameter("WidthStubId")), tmWidthCot_(pSetTM_.getParameter("WidthCot")), @@ -27,6 +28,9 @@ namespace trklet { numSeedTypes_(seedTypeNames_.size()), numChannelsTrack_(numSeedTypes_), channelEncoding_(iConfig.getParameter>("IRChannelsIn")) { + tmMuxOrderInt_.reserve(tmMuxOrder_.size()); + for (const string& s : tmMuxOrder_) + tmMuxOrderInt_.push_back(distance(tmMuxOrder_.begin(), find(tmMuxOrder_.begin(), tmMuxOrder_.end(), s))); const ParameterSet& pSetSeedTypesSeedLayers = iConfig.getParameter("SeedTypesSeedLayers"); const ParameterSet& pSetSeedTypesProjectionLayers = iConfig.getParameter("SeedTypesProjectionLayers"); seedTypesSeedLayers_.reserve(numSeedTypes_); diff --git a/L1Trigger/TrackFindingTracklet/src/TrackMultiplexer.cc b/L1Trigger/TrackFindingTracklet/src/TrackMultiplexer.cc index 5dd59d8d691f2..aa1f67d2a9dfc 100644 --- a/L1Trigger/TrackFindingTracklet/src/TrackMultiplexer.cc +++ b/L1Trigger/TrackFindingTracklet/src/TrackMultiplexer.cc @@ -64,20 +64,22 @@ namespace trklet { // read in and organize input tracks and stubs void TrackMultiplexer::consume(const StreamsTrack& streamsTrack, const StreamsStub& streamsStub) { static const int numS = channelAssignment_->numSeedingLayers(); - const int offsetTrack = region_ * channelAssignment_->numChannelsTrack(); + static const int numChannel = channelAssignment_->numChannelsTrack(); + const int offsetTrack = region_ * numChannel; // count tracks and stubs to reserve container int nTracks(0); int nStubs(0); - for (int channel = 0; channel < channelAssignment_->numChannelsTrack(); channel++) { + for (int channel = 0; channel < numChannel; channel++) { const int channelTrack = offsetTrack + channel; const int offsetStub = channelAssignment_->offsetStub(channelTrack); + const int numProjectionLayers = channelAssignment_->numProjectionLayers(channel); const StreamTrack& streamTrack = streamsTrack[channelTrack]; input_[channel].reserve(streamTrack.size()); for (int frame = 0; frame < (int)streamTrack.size(); frame++) { if (streamTrack[frame].first.isNull()) continue; nTracks++; - for (int layer = 0; layer < channelAssignment_->numProjectionLayers(channel); layer++) + for (int layer = 0; layer < numProjectionLayers; layer++) if (streamsStub[offsetStub + layer][frame].first.isNonnull()) nStubs++; } @@ -85,7 +87,7 @@ namespace trklet { stubs_.reserve(nStubs + nTracks * numS); tracks_.reserve(nTracks); // store tracks and stubs - for (int channel = 0; channel < channelAssignment_->numChannelsTrack(); channel++) { + for (int channel = 0; channel < numChannel; channel++) { const int numP = channelAssignment_->numProjectionLayers(channel); const int channelTrack = offsetTrack + channel; const int offsetStub = channelAssignment_->offsetStub(channelTrack); @@ -331,8 +333,9 @@ namespace trklet { // emualte clock domain crossing static constexpr int ticksPerGap = 3; static constexpr int gapPos = 1; - vector> streams(channelAssignment_->numChannelsTrack()); - for (int channel = 0; channel < channelAssignment_->numChannelsTrack(); channel++) { + static const int numChannel = channelAssignment_->numChannelsTrack(); + vector> streams(numChannel); + for (int channel = 0; channel < numChannel; channel++) { int iTrack(0); deque& stream = streams[channel]; const vector& intput = input_[channel]; @@ -347,19 +350,19 @@ namespace trklet { it = (*--it) ? stream.begin() : stream.erase(it); // route into single channel deque accepted; - vector> stacks(channelAssignment_->numChannelsTrack()); + vector> stacks(numChannel); // clock accurate firmware emulation, each while trip describes one clock tick, one stub in and one stub out per tick while (!all_of(streams.begin(), streams.end(), [](const deque& tracks) { return tracks.empty(); }) or !all_of(stacks.begin(), stacks.end(), [](const deque& tracks) { return tracks.empty(); })) { // fill input fifos - for (int channel = 0; channel < channelAssignment_->numChannelsTrack(); channel++) { + for (int channel = 0; channel < numChannel; channel++) { Track* track = pop_front(streams[channel]); if (track) stacks[channel].push_back(track); } // merge input fifos to one stream, prioritizing lower input channel over higher channel, affects DR bool nothingToRoute(true); - for (int channel = 0; channel < channelAssignment_->numChannelsTrack(); channel++) { + for (int channel : channelAssignment_->tmMuxOrder()) { Track* track = pop_front(stacks[channel]); if (track) { nothingToRoute = false;