56 #include "CLHEP/Units/GlobalPhysicalConstants.h"
70 class MTDHitMatchingInfo {
72 MTDHitMatchingInfo() {
79 inline bool operator<(
const MTDHitMatchingInfo&
m2)
const {
81 constexpr
float chi2_cut = 10.f;
82 constexpr
float low_weight = 3.f;
83 constexpr
float high_weight = 8.f;
84 if (timeChi2 < chi2_cut && m2.timeChi2 < chi2_cut)
85 return chi2(low_weight) < m2.chi2(low_weight);
87 return chi2(high_weight) < m2.chi2(high_weight);
90 inline float chi2(
float timeWeight = 1.
f)
const {
return estChi2 + timeWeight * timeChi2; }
99 TrackSegments() =
default;
101 inline uint32_t addSegment(
float tPath,
float tMom2) {
102 segmentPathOvc_.emplace_back(tPath * c_inv);
103 segmentMom2_.emplace_back(tMom2);
106 LogTrace(
"TrackExtenderWithMTD") <<
"addSegment # " << nSegment_ <<
" s = " << tPath
112 inline float computeTof(
float mass_inv2)
const {
114 for (uint32_t iSeg = 0; iSeg < nSegment_; iSeg++) {
115 float gammasq = 1.f + segmentMom2_[iSeg] * mass_inv2;
117 tof += segmentPathOvc_[iSeg] /
beta;
119 LogTrace(
"TrackExtenderWithMTD") <<
" TOF Segment # " << iSeg + 1 <<
" p = " <<
std::sqrt(segmentMom2_[iSeg])
126 inline uint32_t
size()
const {
return nSegment_; }
128 inline uint32_t removeFirstSegment() {
130 segmentPathOvc_.erase(segmentPathOvc_.begin());
131 segmentMom2_.erase(segmentMom2_.begin());
137 inline std::pair<float, float> getSegmentPathAndMom2(uint32_t iSegment)
const {
138 if (iSegment >= nSegment_) {
139 throw cms::Exception(
"TrackExtenderWithMTD") <<
"Requesting non existing track segment #" << iSegment;
141 return std::make_pair(segmentPathOvc_[iSegment], segmentMom2_[iSegment]);
144 uint32_t nSegment_ = 0;
145 std::vector<float> segmentPathOvc_;
146 std::vector<float> segmentMom2_;
149 struct TrackTofPidInfo {
181 enum class TofCalc { kCost = 1, kSegm = 2, kMixd = 3 };
183 const TrackTofPidInfo computeTrackTofPidInfo(
float magp2,
190 bool addPIDError =
true,
191 TofCalc choice = TofCalc::kCost) {
192 constexpr
float m_pi = 0.13957018f;
193 constexpr
float m_pi_inv2 = 1.0f / m_pi /
m_pi;
194 constexpr
float m_k = 0.493677f;
195 constexpr
float m_k_inv2 = 1.0f / m_k / m_k;
196 constexpr
float m_p = 0.9382720813f;
197 constexpr
float m_p_inv2 = 1.0f / m_p / m_p;
199 TrackTofPidInfo tofpid;
202 tofpid.tmtderror = t_mtderr;
203 tofpid.pathlength = length;
205 auto deltat = [&](
const float mass_inv2,
const float betatmp) {
209 res = tofpid.pathlength / betatmp *
c_inv;
212 res = trs.computeTof(mass_inv2);
215 res = trs.computeTof(mass_inv2) + tofpid.pathlength / betatmp *
c_inv;
221 tofpid.gammasq_pi = 1.f + magp2 * m_pi_inv2;
222 tofpid.beta_pi =
std::sqrt(1.
f - 1.
f / tofpid.gammasq_pi);
223 tofpid.dt_pi = deltat(m_pi_inv2, tofpid.beta_pi);
225 tofpid.gammasq_k = 1.f + magp2 * m_k_inv2;
226 tofpid.beta_k =
std::sqrt(1.
f - 1.
f / tofpid.gammasq_k);
227 tofpid.dt_k = deltat(m_k_inv2, tofpid.beta_k);
229 tofpid.gammasq_p = 1.f + magp2 * m_p_inv2;
230 tofpid.beta_p =
std::sqrt(1.
f - 1.
f / tofpid.gammasq_p);
231 tofpid.dt_p = deltat(m_p_inv2, tofpid.beta_p);
233 tofpid.dt = tofpid.tmtd - tofpid.dt_pi - t_vtx;
234 tofpid.dterror =
sqrt(tofpid.tmtderror * tofpid.tmtderror + t_vtx_err * t_vtx_err);
235 tofpid.betaerror = 0.f;
238 sqrt(tofpid.dterror * tofpid.dterror + (tofpid.dt_p - tofpid.dt_pi) * (tofpid.dt_p - tofpid.dt_pi));
239 tofpid.betaerror = tofpid.beta_p - tofpid.beta_pi;
242 tofpid.dtchi2 = (tofpid.dt * tofpid.dt) / (tofpid.dterror * tofpid.dterror);
244 tofpid.dt_best = tofpid.dt;
245 tofpid.dterror_best = tofpid.dterror;
246 tofpid.dtchi2_best = tofpid.dtchi2;
248 tofpid.prob_pi = -1.f;
249 tofpid.prob_k = -1.f;
250 tofpid.prob_p = -1.f;
254 float chi2_pi = tofpid.dtchi2;
256 (tofpid.tmtd - tofpid.dt_k - t_vtx) * (tofpid.tmtd - tofpid.dt_k - t_vtx) / (tofpid.dterror * tofpid.dterror);
258 (tofpid.tmtd - tofpid.dt_p - t_vtx) * (tofpid.tmtd - tofpid.dt_p - t_vtx) / (tofpid.dterror * tofpid.dterror);
260 float rawprob_pi =
exp(-0.5
f * chi2_pi);
261 float rawprob_k =
exp(-0.5
f * chi2_k);
262 float rawprob_p =
exp(-0.5
f * chi2_p);
263 float normprob = 1.f / (rawprob_pi + rawprob_k + rawprob_p);
265 tofpid.prob_pi = rawprob_pi * normprob;
266 tofpid.prob_k = rawprob_k * normprob;
267 tofpid.prob_p = rawprob_p * normprob;
269 float prob_heavy = 1.f - tofpid.prob_pi;
270 constexpr
float heavy_threshold = 0.75f;
272 if (prob_heavy > heavy_threshold) {
273 if (chi2_k < chi2_p) {
274 tofpid.dt_best = (tofpid.tmtd - tofpid.dt_k - t_vtx);
275 tofpid.dtchi2_best = chi2_k;
277 tofpid.dt_best = (tofpid.tmtd - tofpid.dt_p - t_vtx);
278 tofpid.dtchi2_best = chi2_p;
285 bool getTrajectoryStateClosestToBeamLine(
const Trajectory& traj,
293 if (!stateForProjectionToBeamLineOnSurface.
isValid()) {
294 edm::LogError(
"CannotPropagateToBeamLine") <<
"the state on the closest measurement isnot valid. skipping track.";
301 tscbl = tscblBuilder(stateForProjectionToBeamLine, bs);
310 TrackSegments& trs) {
313 bool validpropagation =
true;
314 float oldp = traj.
measurements().begin()->updatedState().globalMomentum().mag();
315 float pathlength1 = 0.f;
316 float pathlength2 = 0.f;
320 const auto& propresult = thePropagator->
propagateWithPath(it->updatedState(), (it + 1)->updatedState().surface());
321 float layerpathlength =
std::abs(propresult.second);
322 if (layerpathlength == 0.
f) {
323 validpropagation =
false;
325 pathlength1 += layerpathlength;
326 trs.addSegment(layerpathlength, (it + 1)->updatedState().globalMomentum().
mag2());
327 LogTrace(
"TrackExtenderWithMTD") <<
"TSOS " << std::fixed << std::setw(4) << trs.size() <<
" R_i " << std::fixed
328 << std::setw(14) << it->updatedState().globalPosition().perp() <<
" z_i "
329 << std::fixed << std::setw(14) << it->updatedState().globalPosition().z()
330 <<
" R_e " << std::fixed << std::setw(14)
331 << (it + 1)->updatedState().globalPosition().perp() <<
" z_e " << std::fixed
332 << std::setw(14) << (it + 1)->updatedState().globalPosition().z() <<
" p "
333 << std::fixed << std::setw(14) << (it + 1)->updatedState().globalMomentum().mag()
334 <<
" dp " << std::fixed << std::setw(14)
335 << (it + 1)->updatedState().globalMomentum().mag() - oldp;
336 oldp = (it + 1)->updatedState().globalMomentum().mag();
344 if (pathlength2 == 0.
f) {
345 validpropagation =
false;
347 pathlength = pathlength1 + pathlength2;
348 trs.addSegment(pathlength2, tscblPCA.momentum().mag2());
349 LogTrace(
"TrackExtenderWithMTD") <<
"TSOS " << std::fixed << std::setw(4) << trs.size() <<
" R_e " << std::fixed
350 << std::setw(14) << tscblPCA.position().perp() <<
" z_e " << std::fixed
351 << std::setw(14) << tscblPCA.position().z() <<
" p " << std::fixed << std::setw(14)
352 << tscblPCA.momentum().mag() <<
" dp " << std::fixed << std::setw(14)
353 << tscblPCA.momentum().mag() - oldp;
354 return validpropagation;
361 TrackSegments& trs) {
365 bool tscbl_status = getTrajectoryStateClosestToBeamLine(traj, bs, thePropagator, tscbl);
370 return trackPathLength(traj, tscbl, thePropagator, pathlength, trs);
375 template <
class TrackCollection>
383 template <
class H,
class T>
394 const TrackSegments&,
401 const bool matchVertex,
402 MTDHitMatchingInfo& bestHit)
const;
408 const TrackSegments&,
415 const bool matchVertex,
416 MTDHitMatchingInfo& bestHit)
const;
418 void fillMatchingHits(
const DetLayer*,
423 const TrackSegments&,
430 MTDHitMatchingInfo&)
const;
434 if (!recHits.empty()) {
439 auto rFirst = first.
mag2();
440 auto rLast = last.
mag2();
446 LogDebug(
"TrackExtenderWithMTD") <<
"Impossible to determine the rechits order" << endl;
523 template <
class TrackCollection>
529 updateTraj_(iConfig.getParameter<bool>(
"updateTrackTrajectory")),
530 updateExtra_(iConfig.getParameter<bool>(
"updateTrackExtra")),
531 updatePattern_(iConfig.getParameter<bool>(
"updateTrackHitPattern")),
532 mtdRecHitBuilder_(iConfig.getParameter<std::
string>(
"MTDRecHitBuilder")),
533 propagator_(iConfig.getParameter<std::
string>(
"Propagator")),
534 transientTrackBuilder_(iConfig.getParameter<std::
string>(
"TransientTrackBuilder")),
535 estMaxChi2_(iConfig.getParameter<double>(
"estimatorMaxChi2")),
536 estMaxNSigma_(iConfig.getParameter<double>(
"estimatorMaxNSigma")),
537 btlChi2Cut_(iConfig.getParameter<double>(
"btlChi2Cut")),
538 btlTimeChi2Cut_(iConfig.getParameter<double>(
"btlTimeChi2Cut")),
539 etlChi2Cut_(iConfig.getParameter<double>(
"etlChi2Cut")),
540 etlTimeChi2Cut_(iConfig.getParameter<double>(
"etlTimeChi2Cut")),
541 useVertex_(iConfig.getParameter<bool>(
"useVertex")),
542 useSimVertex_(iConfig.getParameter<bool>(
"useSimVertex")),
543 dzCut_(iConfig.getParameter<double>(
"dZCut")),
544 bsTimeSpread_(iConfig.getParameter<double>(
"bsTimeSpread")) {
577 gtgToken_ = esConsumes<GlobalTrackingGeometry, GlobalTrackingGeometryRecord>();
578 dlgeoToken_ = esConsumes<MTDDetLayerGeometry, MTDRecoGeometryRecord>();
579 magfldToken_ = esConsumes<MagneticField, IdealMagneticFieldRecord>();
581 ttopoToken_ = esConsumes<TrackerTopology, TrackerTopologyRcd>();
583 produces<edm::OwnVector<TrackingRecHit>>();
584 produces<reco::TrackExtraCollection>();
585 produces<TrackCollection>();
588 template <
class TrackCollection>
598 desc.
add<
bool>(
"updateTrackTrajectory",
true);
599 desc.
add<
bool>(
"updateTrackExtra",
true);
600 desc.
add<
bool>(
"updateTrackHitPattern",
true);
601 desc.
add<
std::string>(
"TransientTrackBuilder",
"TransientTrackBuilder");
603 desc.
add<
std::string>(
"Propagator",
"PropagatorWithMaterialForMTD");
606 "KFFitterForRefitInsideOut",
607 "KFSmootherForRefitInsideOut",
608 "PropagatorWithMaterialForMTD",
615 desc.
add<
double>(
"estimatorMaxChi2", 500.);
616 desc.
add<
double>(
"estimatorMaxNSigma", 10.);
617 desc.
add<
double>(
"btlChi2Cut", 50.);
618 desc.
add<
double>(
"btlTimeChi2Cut", 10.);
619 desc.
add<
double>(
"etlChi2Cut", 50.);
620 desc.
add<
double>(
"etlTimeChi2Cut", 10.);
621 desc.
add<
bool>(
"useVertex",
false);
622 desc.
add<
bool>(
"useSimVertex",
false);
623 desc.
add<
double>(
"dZCut", 0.1);
624 desc.
add<
double>(
"bsTimeSpread", 0.2);
625 descriptions.
add(
"trackExtenderWithMTDBase", desc);
628 template <
class TrackCollection>
629 template <
class H,
class T>
632 const std::vector<T>& vec,
634 auto out = std::make_unique<edm::ValueMap<T>>();
636 filler.insert(handle, vec.begin(), vec.end());
641 template <
class TrackCollection>
646 theTransformer->setServices(es);
658 hitbuilder_ = es.
getHandle(hitbuilderToken_);
666 auto output = std::make_unique<TrackCollection>();
667 auto extras = std::make_unique<reco::TrackExtraCollection>();
668 auto outhits = std::make_unique<edm::OwnVector<TrackingRecHit>>();
670 std::vector<float> btlMatchChi2;
671 std::vector<float> etlMatchChi2;
672 std::vector<float> btlMatchTimeChi2;
673 std::vector<float> etlMatchTimeChi2;
674 std::vector<int> npixBarrel;
675 std::vector<int> npixEndcap;
676 std::vector<float> pOrigTrkRaw;
677 std::vector<float> betaOrigTrkRaw;
678 std::vector<float> t0OrigTrkRaw;
679 std::vector<float> sigmat0OrigTrkRaw;
680 std::vector<float> pathLengthsOrigTrkRaw;
681 std::vector<float> tmtdOrigTrkRaw;
682 std::vector<float> sigmatmtdOrigTrkRaw;
683 std::vector<float> tofpiOrigTrkRaw;
684 std::vector<float> tofkOrigTrkRaw;
685 std::vector<float> tofpOrigTrkRaw;
686 std::vector<int> assocOrigTrkRaw;
688 auto const tracksH = ev.
getHandle(tracksToken_);
690 const auto& trjtrks = ev.
get(trajTrackAToken_);
693 const auto& hits = ev.
get(hitsToken_);
696 const auto& bs = ev.
get(bsToken_);
699 if (useVertex_ && !useSimVertex_) {
700 auto const& vtxs = ev.
get(vtxToken_);
705 std::unique_ptr<math::XYZTLorentzVectorF> genPV(
nullptr);
706 if (useVertex_ && useSimVertex_) {
707 const auto& genVtxPosition = ev.
get(genVtxPositionToken_);
708 const auto& genVtxTime = ev.
get(genVtxTimeToken_);
709 genPV = std::make_unique<math::XYZTLorentzVectorF>(
710 genVtxPosition.x(), genVtxPosition.y(), genVtxPosition.z(), genVtxTime);
715 if (useSimVertex_ && genPV) {
716 vtxTime = genPV->t();
721 std::vector<unsigned> track_indices;
724 for (
const auto& trjtrk : trjtrks) {
728 float trackVtxTime = 0.f;
737 trackVtxTime = vtxTime;
741 auto thits = theTransformer->getTransientRecHits(ttrack);
743 MTDHitMatchingInfo mBTL, mETL;
749 bool tscbl_status = getTrajectoryStateClosestToBeamLine(trajs, bs, prop, tscbl);
755 trackPathLength(trajs, tscbl, prop, pathlength0, trs0);
757 const auto& btlhits = tryBTLLayers(tsos,
770 mtdthits.insert(mtdthits.end(), btlhits.begin(), btlhits.end());
774 const auto& etlhits = tryETLLayers(tsos,
787 mtdthits.insert(mtdthits.end(), etlhits.begin(), etlhits.end());
791 auto ordering = checkRecHitsOrdering(thits);
793 thits.insert(thits.end(), mtdthits.begin(), mtdthits.end());
795 std::reverse(mtdthits.begin(), mtdthits.end());
796 mtdthits.insert(mtdthits.end(), thits.begin(), thits.end());
797 thits.swap(mtdthits);
800 const auto& trajwithmtd =
801 mtdthits.empty() ? std::vector<Trajectory>(1, trajs) : theTransformer->transform(ttrack, thits);
802 float pMap = 0.f, betaMap = 0.f, t0Map = 0.f, sigmat0Map = -1.f, pathLengthMap = -1.f, tmtdMap = 0.f,
803 sigmatmtdMap = -1.f, tofpiMap = 0.f, tofkMap = 0.f, tofpMap = 0.f;
806 for (
const auto& trj : trajwithmtd) {
807 const auto& thetrj = (updateTraj_ ? trj : trajs);
808 float pathLength = 0.f, tmtd = 0.f, sigmatmtd = -1.f, tofpi = 0.f, tofk = 0.f, tofp = 0.f;
809 LogTrace(
"TrackExtenderWithMTD") <<
"Refit track " << itrack <<
" p/pT = " << track->p() <<
" " << track->pt()
810 <<
" eta = " << track->eta();
817 !trajwithmtd.empty() && !mtdthits.empty(),
824 if (result.
ndof() >= 0) {
828 size_t hitsstart = outhits->size();
829 if (updatePattern_) {
830 t2t(trj, *outhits, trajParams, chi2s);
832 t2t(thetrj, *outhits, trajParams, chi2s);
834 size_t hitsend = outhits->size();
835 extras->push_back(buildTrackExtra(trj));
836 extras->back().setHits(hitsRefProd, hitsstart, hitsend - hitsstart);
837 extras->back().setTrajParams(trajParams, chi2s);
839 output->push_back(result);
840 btlMatchChi2.push_back(mBTL.hit ? mBTL.estChi2 : -1.f);
841 etlMatchChi2.push_back(mETL.hit ? mETL.estChi2 : -1.f);
842 btlMatchTimeChi2.push_back(mBTL.hit ? mBTL.timeChi2 : -1.f);
843 etlMatchTimeChi2.push_back(mETL.hit ? mETL.timeChi2 : -1.f);
844 pathLengthMap = pathLength;
846 sigmatmtdMap = sigmatmtd;
847 auto& backtrack =
output->back();
848 iMap =
output->size() - 1;
849 pMap = backtrack.p();
850 betaMap = backtrack.beta();
851 t0Map = backtrack.t0();
852 sigmat0Map = std::copysign(
std::sqrt(
std::abs(backtrack.covt0t0())), backtrack.covt0t0());
857 backtrack.setExtra((updateExtra_ ? extraRef : track->extra()));
858 for (
unsigned ihit = hitsstart; ihit < hitsend; ++ihit) {
859 backtrack.appendHitPattern((*outhits)[ihit], ttopo);
861 npixBarrel.push_back(backtrack.hitPattern().numberOfValidPixelBarrelHits());
862 npixEndcap.push_back(backtrack.hitPattern().numberOfValidPixelEndcapHits());
863 LogTrace(
"TrackExtenderWithMTD") <<
"tmtd " << tmtdMap <<
" +/- " << sigmatmtdMap <<
" t0 " << t0Map <<
" +/- "
864 << sigmat0Map <<
" tof pi/K/p " << tofpiMap <<
" " << tofkMap <<
" "
867 LogTrace(
"TrackExtenderWithMTD") <<
"Error in the MTD track refitting. This should not happen";
871 pOrigTrkRaw.push_back(pMap);
872 betaOrigTrkRaw.push_back(betaMap);
873 t0OrigTrkRaw.push_back(t0Map);
874 sigmat0OrigTrkRaw.push_back(sigmat0Map);
875 pathLengthsOrigTrkRaw.push_back(pathLengthMap);
876 tmtdOrigTrkRaw.push_back(tmtdMap);
877 sigmatmtdOrigTrkRaw.push_back(sigmatmtdMap);
878 tofpiOrigTrkRaw.push_back(tofpiMap);
879 tofkOrigTrkRaw.push_back(tofkMap);
880 tofpOrigTrkRaw.push_back(tofpMap);
881 assocOrigTrkRaw.push_back(iMap);
884 btlMatchChi2.push_back(-1.
f);
885 etlMatchChi2.push_back(-1.
f);
886 btlMatchTimeChi2.push_back(-1.
f);
887 etlMatchTimeChi2.push_back(-1.
f);
888 npixBarrel.push_back(-1.
f);
889 npixEndcap.push_back(-1.
f);
899 fillValueMap(ev, tracksH, btlMatchChi2, btlMatchChi2Token_);
900 fillValueMap(ev, tracksH, etlMatchChi2, etlMatchChi2Token_);
901 fillValueMap(ev, tracksH, btlMatchTimeChi2, btlMatchTimeChi2Token_);
902 fillValueMap(ev, tracksH, etlMatchTimeChi2, etlMatchTimeChi2Token_);
903 fillValueMap(ev, tracksH, npixBarrel, npixBarrelToken_);
904 fillValueMap(ev, tracksH, npixEndcap, npixEndcapToken_);
905 fillValueMap(ev, tracksH, pOrigTrkRaw, pOrigTrkToken_);
906 fillValueMap(ev, tracksH, betaOrigTrkRaw, betaOrigTrkToken_);
907 fillValueMap(ev, tracksH, t0OrigTrkRaw, t0OrigTrkToken_);
908 fillValueMap(ev, tracksH, sigmat0OrigTrkRaw, sigmat0OrigTrkToken_);
909 fillValueMap(ev, tracksH, pathLengthsOrigTrkRaw, pathLengthOrigTrkToken_);
910 fillValueMap(ev, tracksH, tmtdOrigTrkRaw, tmtdOrigTrkToken_);
911 fillValueMap(ev, tracksH, sigmatmtdOrigTrkRaw, sigmatmtdOrigTrkToken_);
912 fillValueMap(ev, tracksH, tofpiOrigTrkRaw, tofpiOrigTrkToken_);
913 fillValueMap(ev, tracksH, tofkOrigTrkRaw, tofkOrigTrkToken_);
914 fillValueMap(ev, tracksH, tofpOrigTrkRaw, tofpOrigTrkToken_);
915 fillValueMap(ev, tracksH, assocOrigTrkRaw, assocOrigTrkToken_);
919 bool cmp_for_detset(
const unsigned one,
const unsigned two) {
return one <
two; };
926 const float pathlength0,
927 const TrackSegments& trs0,
930 const float bsTimeSpread,
933 bool useVtxConstraint,
934 std::set<MTDHitMatchingInfo>&
out) {
935 pair<bool, TrajectoryStateOnSurface>
comp = layer->
compatible(tsos, *prop, *estimator);
937 const vector<DetLayer::DetWithState> compDets = layer->
compatibleDets(tsos, *prop, *estimator);
938 if (!compDets.empty()) {
939 for (
const auto& detWithState : compDets) {
940 auto range = hits.
equal_range(detWithState.first->geographicalId(), cmp_for_detset);
948 const float t_vtx = useVtxConstraint ? vtxTime : 0.f;
950 constexpr
float vtx_res = 0.008f;
951 const float t_vtx_err = useVtxConstraint ? vtx_res : bsTimeSpread;
953 float lastpmag2 = trs0.getSegmentPathAndMom2(0).second;
955 for (
auto detitr =
range.first; detitr !=
range.second; ++detitr) {
956 for (
const auto&
hit : *detitr) {
957 auto est = estimator->
estimate(detWithState.second,
hit);
961 TrackTofPidInfo tof = computeTrackTofPidInfo(lastpmag2,
970 MTDHitMatchingInfo mi;
972 mi.estChi2 = est.second;
973 mi.timeChi2 = tof.dtchi2_best;
984 template <
class TrackCollection>
989 const float pathlength0,
990 const TrackSegments& trs0,
997 const bool matchVertex,
998 MTDHitMatchingInfo& bestHit)
const {
1002 bestHit = MTDHitMatchingInfo();
1003 for (
const DetLayer* ilay : layers)
1004 fillMatchingHits(ilay, tsos, traj, pmag2, pathlength0, trs0, hits, prop, bs, vtxTime, matchVertex, output, bestHit);
1008 template <
class TrackCollection>
1013 const float pathlength0,
1014 const TrackSegments& trs0,
1020 const float vtxTime,
1021 const bool matchVertex,
1022 MTDHitMatchingInfo& bestHit)
const {
1026 bestHit = MTDHitMatchingInfo();
1027 for (
const DetLayer* ilay : layers) {
1029 const float diskZ = disk.position().z();
1034 fillMatchingHits(ilay, tsos, traj, pmag2, pathlength0, trs0, hits, prop, bs, vtxTime, matchVertex, output, bestHit);
1039 if (output.size() == 2) {
1040 if (
std::abs(output[0]->globalPosition().
z()) >
std::abs(output[1]->globalPosition().
z())) {
1041 std::reverse(output.begin(), output.end());
1047 template <
class TrackCollection>
1052 const float pathlength0,
1053 const TrackSegments& trs0,
1057 const float& vtxTime,
1058 const bool matchVertex,
1060 MTDHitMatchingInfo& bestHit)
const {
1061 std::set<MTDHitMatchingInfo> hitsInLayer;
1062 bool hitMatched =
false;
1064 using namespace std::placeholders;
1065 auto find_hits = std::bind(find_hits_in_dets,
1079 std::ref(hitsInLayer));
1081 if (useVertex_ && matchVertex)
1082 find_hits(vtxTime,
true);
1084 find_hits(0,
false);
1086 float spaceChi2Cut = ilay->
isBarrel() ? btlChi2Cut_ : etlChi2Cut_;
1087 float timeChi2Cut = ilay->
isBarrel() ? btlTimeChi2Cut_ : etlTimeChi2Cut_;
1090 if (!hitsInLayer.empty()) {
1092 auto const& firstHit = *hitsInLayer.begin();
1093 if (firstHit.estChi2 < spaceChi2Cut && firstHit.timeChi2 < timeChi2Cut) {
1095 output.push_back(hitbuilder_->build(firstHit.hit));
1096 if (firstHit < bestHit)
1101 if (useVertex_ && matchVertex && !hitMatched) {
1103 hitsInLayer.clear();
1104 find_hits(0,
false);
1105 if (!hitsInLayer.empty()) {
1106 auto const& firstHit = *hitsInLayer.begin();
1107 if (firstHit.timeChi2 < timeChi2Cut) {
1108 if (firstHit.estChi2 < spaceChi2Cut) {
1110 output.push_back(hitbuilder_->build(firstHit.hit));
1111 if (firstHit < bestHit)
1121 template <
class TrackCollection>
1129 float& pathLengthOut,
1131 float& sigmatmtdOut,
1134 float& tofp)
const {
1136 bool tsbcl_status = getTrajectoryStateClosestToBeamLine(traj, bs, thePropagator, tscbl);
1149 float covt0t0 = -1.f;
1150 pathLengthOut = -1.f;
1152 sigmatmtdOut = -1.f;
1153 float betaOut = 0.f;
1154 float covbetabeta = -1.f;
1156 auto routput = [&]() {
1175 bool validpropagation = trackPathLength(trajWithMtd, bs, thePropagator, pathlength, trs);
1177 float thiterror = -1.f;
1178 bool validmtd =
false;
1180 if (!validpropagation) {
1184 uint32_t ihitcount(0), ietlcount(0);
1189 if (
MTDDetId(
hit.recHit()->geographicalId()).mtdSubDetector() == MTDDetId::MTDType::ETL) {
1196 if (ihitcount == 1) {
1198 thit = mtdhit->
time();
1201 }
else if (ihitcount == 2 && ietlcount == 2) {
1202 std::pair<float, float> lastStep = trs.getSegmentPathAndMom2(0);
1203 float etlpathlength =
std::abs(lastStep.first * c_cm_ns);
1207 if (etlpathlength == 0.
f) {
1208 validpropagation =
false;
1210 pathlength -= etlpathlength;
1211 trs.removeFirstSegment();
1214 TrackTofPidInfo tofInfo = computeTrackTofPidInfo(
1215 lastStep.second, etlpathlength, trs, mtdhit1->
time(), mtdhit1->
timeError(), 0.f, 0.f,
true, TofCalc::kCost);
1219 float err1 = tofInfo.dterror * tofInfo.dterror;
1223 <<
"MTD tracking hits with zero time uncertainty: " << err1 <<
" " << err2;
1225 if ((tofInfo.dt - mtdhit2->
time()) * (tofInfo.dt - mtdhit2->
time()) < (err1 + err2) * etlTimeChi2Cut_) {
1232 thiterror = 1.f / (err1 + err2);
1233 thit = (tofInfo.dt * err1 + mtdhit2->
time() * err2) * thiterror;
1235 LogDebug(
"TrackExtenderWithMTD") <<
"p trk = " << p.mag() <<
" ETL hits times/errors: " << mtdhit1->
time()
1236 <<
" +/- " << mtdhit1->
timeError() <<
" , " << mtdhit2->
time() <<
" +/- "
1237 << mtdhit2->
timeError() <<
" extrapolated time1: " << tofInfo.dt <<
" +/- "
1238 << tofInfo.dterror <<
" average = " << thit <<
" +/- " << thiterror;
1245 <<
"MTD hits #" << ihitcount <<
"ETL hits #" << ietlcount <<
" anomalous pattern, skipping...";
1248 if (validmtd && validpropagation) {
1250 TrackTofPidInfo tofInfo =
1251 computeTrackTofPidInfo(p.mag2(), pathlength, trs, thit, thiterror, 0.f, 0.f,
true, TofCalc::kSegm);
1252 pathLengthOut = pathlength;
1254 sigmatmtdOut = thiterror;
1256 covt0t0 = tofInfo.dterror * tofInfo.dterror;
1257 betaOut = tofInfo.beta_pi;
1258 covbetabeta = tofInfo.betaerror * tofInfo.betaerror;
1259 tofpi = tofInfo.dt_pi;
1260 tofk = tofInfo.dt_k;
1261 tofp = tofInfo.dt_p;
1268 template <
class TrackCollection>
1270 static const string metname =
"TrackExtenderWithMTD";
1280 unsigned int innerId = 0, outerId = 0;
1285 LogTrace(metname) <<
"alongMomentum";
1293 LogTrace(metname) <<
"oppositeToMomentum";
1301 LogError(metname) <<
"Wrong propagation direction!";
1303 const GeomDet* outerDet = gtg_->idToDet(outerDetId);
1311 LogTrace(metname) <<
"The Global Muon outerMostMeasurementState is not compatible with the recHit detector!"
1312 <<
" Setting outerMost postition to recHit position if recHit isValid: "
1313 << outerRecHit->isValid();
1314 LogTrace(metname) <<
"From " << outerTSOSPos <<
" to " << hitPos;
1344 template <
class TrackCollection>
1353 if ((bc = dynamic_cast<const BoundCylinder*>(sur))) {
1354 output <<
" Cylinder of radius: " << bc->radius() << endl;
1355 }
else if ((bd = dynamic_cast<const BoundDisk*>(sur))) {
1356 output <<
" Disk at: " << bd->position().z() << endl;
1358 return output.str();
static void fillDescriptions(edm::ConfigurationDescriptions &descriptions)
virtual std::pair< bool, TrajectoryStateOnSurface > compatible(const TrajectoryStateOnSurface &ts, const Propagator &, const MeasurementEstimator &) const =0
double z0() const
z coordinate
virtual const BoundSurface & surface() const =0
The surface of the GeometricSearchDet.
TrackExtenderWithMTDT< reco::TrackCollection > TrackExtenderWithMTD
edm::EDPutToken btlMatchTimeChi2Token_
OrphanHandle< PROD > put(std::unique_ptr< PROD > product)
Put a new product.
edm::EDPutToken pOrigTrkToken_
std::vector< LayerSetAndLayers > layers(const SeedingLayerSetsHits &sets)
edm::ESGetToken< TransientTrackingRecHitBuilder, TransientRecHitRecord > hitbuilderToken_
edm::EDPutToken t0OrigTrkToken_
ConstRecHitPointer const & recHit() const
const float bsTimeSpread_
edm::EDPutToken btlMatchChi2Token_
Range equal_range(id_type i, CMP cmp, bool update=false) const
reco::TrackExtra buildTrackExtra(const Trajectory &trajectory) const
virtual bool inside(const Local3DPoint &) const =0
Determine if the point is inside the bounds.
TransientTrackingRecHit::ConstRecHitContainer tryETLLayers(const TrajectoryStateOnSurface &, const Trajectory &traj, const float, const float, const TrackSegments &, const MTDTrackingDetSetVector &, const MTDDetLayerGeometry *, const MagneticField *field, const Propagator *prop, const reco::BeamSpot &bs, const float vtxTime, const bool matchVertex, MTDHitMatchingInfo &bestHit) const
edm::EDPutToken tofpOrigTrkToken_
const float btlTimeChi2Cut_
edm::ESGetToken< GlobalTrackingGeometry, GlobalTrackingGeometryRecord > gtgToken_
const std::string metname
edm::EDGetTokenT< reco::BeamSpot > bsToken_
std::unique_ptr< MeasurementEstimator > theEstimator
#define DEFINE_FWK_MODULE(type)
TrackExtenderWithMTDT(const ParameterSet &pset)
const CurvilinearTrajectoryError & curvilinearError() const
edm::ESGetToken< TrackerTopology, TrackerTopologyRcd > ttopoToken_
void fillMatchingHits(const DetLayer *, const TrajectoryStateOnSurface &, const Trajectory &, const float, const float, const TrackSegments &, const MTDTrackingDetSetVector &, const Propagator *, const reco::BeamSpot &, const float &, const bool, TransientTrackingRecHit::ConstRecHitContainer &, MTDHitMatchingInfo &) const
Global3DPoint GlobalPoint
string dumpLayer(const DetLayer *layer) const
edm::EDPutToken etlMatchChi2Token_
edm::View< TrackType > InputCollection
edm::EDGetTokenT< GlobalPoint > genVtxPositionToken_
GlobalPoint globalPosition() const
LocalPoint toLocal(const GlobalPoint &gp) const
Conversion to the R.F. of the GeomDet.
ConstRecHitContainer recHits() const
const Bounds & bounds() const
const std::vector< const DetLayer * > & allBTLLayers() const
return the BTL DetLayers (barrel), inside-out
Exp< T >::type exp(const T &t)
TrackCharge charge() const
Log< level::Error, false > LogError
edm::EDPutToken tofkOrigTrkToken_
TrajectoryMeasurement const & closestMeasurement(GlobalPoint) const
edm::EDPutToken etlMatchTimeChi2Token_
edm::EDPutToken sigmat0OrigTrkToken_
const Plane & surface() const
The nominal surface of the GeomDet.
const Point & position() const
position
const CurvilinearTrajectoryError & curvilinearError() const
virtual std::vector< DetWithState > compatibleDets(const TrajectoryStateOnSurface &startingState, const Propagator &prop, const MeasurementEstimator &est) const
edm::EDPutToken tmtdOrigTrkToken_
TransientTrackingRecHit::ConstRecHitContainer tryBTLLayers(const TrajectoryStateOnSurface &, const Trajectory &traj, const float, const float, const TrackSegments &, const MTDTrackingDetSetVector &, const MTDDetLayerGeometry *, const MagneticField *field, const Propagator *prop, const reco::BeamSpot &bs, const float vtxTime, const bool matchVertex, MTDHitMatchingInfo &bestHit) const
edm::EDGetTokenT< MTDTrackingDetSetVector > hitsToken_
Detector identifier base class for the MIP Timing Layer.
constexpr std::array< uint8_t, layerIndexSize > layer
const uint16_t range(const Frame &aFrame)
PropagationDirection const & direction() const
DataContainer const & measurements() const
Handle< PROD > getHandle(EDGetTokenT< PROD > token) const
const SurfaceType & surface() const
Container::value_type value_type
const float estMaxNSigma_
edm::ESGetToken< TransientTrackBuilder, TransientTrackRecord > builderToken_
const std::vector< const DetLayer * > & allETLLayers() const
return the ETL DetLayers (endcap), -Z to +Z
const std::string mtdRecHitBuilder_
double ndof() const
number of degrees of freedom of the fit
TrajectoryMeasurement const & lastMeasurement() const
FreeTrajectoryState const * freeState(bool withErrors=true) const
GlobalVector momentum() const
T mag2() const
The vector magnitude squared. Equivalent to vec.dot(vec)
RefitDirection::GeometricalDirection checkRecHitsOrdering(TransientTrackingRecHit::ConstRecHitContainer const &recHits) const
Abs< T >::type abs(const T &t)
bool get(ProductID const &oid, Handle< PROD > &result) const
edm::EDPutToken assocOrigTrkToken_
const std::string propagator_
edm::ESGetToken< Propagator, TrackingComponentsRecord > propToken_
reco::Track buildTrack(const reco::TrackRef &, const Trajectory &, const Trajectory &, const reco::BeamSpot &, const MagneticField *field, const Propagator *prop, bool hasMTD, float &pathLength, float &tmtdOut, float &sigmatmtdOut, float &tofpi, float &tofk, float &tofp) const
virtual HitReturnType estimate(const TrajectoryStateOnSurface &ts, const TrackingRecHit &hit) const =0
ParameterDescriptionBase * add(U const &iLabel, T const &value)
edm::RefToBase< TrajectorySeed > seedRef(void) const
virtual std::pair< TrajectoryStateOnSurface, double > propagateWithPath(const FreeTrajectoryState &, const Surface &) const final
constexpr valType roundIfNear0(valType value, double tolerance=1.e-7)
edm::ESHandle< TransientTrackingRecHitBuilder > hitbuilder_
GlobalVector momentum() const
RefProd< PROD > getRefBeforePut()
FTS const & trackStateAtPCA() const
GlobalPoint position() const
std::vector< ConstRecHitPointer > ConstRecHitContainer
const std::string transientTrackBuilder_
Log< level::Info, false > LogInfo
GlobalPoint position() const
int ndof(bool bon=true) const
std::vector< TrackExtra > TrackExtraCollection
collection of TrackExtra objects
TrajectoryMeasurement const & firstMeasurement() const
ConstRecHitContainer RecHitContainer
constexpr NumType convertMmToCm(NumType millimeters)
static void fillDescriptions(edm::ConfigurationDescriptions &descriptions)
const GlobalTrajectoryParameters & globalParameters() const
ParameterSet const & getParameterSet(std::string const &) const
XYZVectorD XYZVector
spatial vector with cartesian internal representation
XYZPointD XYZPoint
point in space with cartesian internal representation
A 2D TrackerRecHit with time and time error information.
T getParameter(std::string const &) const
void add(std::string const &label, ParameterSetDescription const &psetDescription)
TrackingRecHit::ConstRecHitPointer ConstRecHitPointer
edm::ESHandle< TransientTrackBuilder > builder_
std::unique_ptr< TrackTransformer > theTransformer
edm::EDPutToken npixBarrelToken_
edm::ESGetToken< MagneticField, IdealMagneticFieldRecord > magfldToken_
bool operator<(DTCELinkId const &lhs, DTCELinkId const &rhs)
edm::EDPutToken npixEndcapToken_
static int position[264][3]
ESTransientHandle< T > getTransientHandle(const ESGetToken< T, R > &iToken) const
edm::EDPutToken betaOrigTrkToken_
double y0() const
y coordinate
edm::EDGetTokenT< TrajTrackAssociationCollection > trajTrackAToken_
edm::EDPutToken pathLengthOrigTrkToken_
TrajectoryStateOnSurface const & updatedState() const
ESHandle< T > getHandle(const ESGetToken< T, R > &iToken) const
edm::ESHandle< GlobalTrackingGeometry > gtg_
edm::EDPutToken sigmatmtdOrigTrkToken_
edm::EDPutToken tofpiOrigTrkToken_
edm::EDGetTokenT< float > genVtxTimeToken_
edm::EDGetTokenT< VertexCollection > vtxToken_
void produce(edm::Event &ev, const edm::EventSetup &es) final
tuple size
Write out results.
TrackCollection::value_type TrackType
edm::EDGetTokenT< InputCollection > tracksToken_
const float etlTimeChi2Cut_
double t() const
t coordinate
void fillValueMap(edm::Event &iEvent, const H &handle, const std::vector< T > &vec, const edm::EDPutToken &token) const
edm::ESGetToken< MTDDetLayerGeometry, MTDRecoGeometryRecord > dlgeoToken_
double x0() const
x coordinate