24 profiling_map_[
"setup"] = 0.;
25 profiling_map_[
"hits"] = 0.;
26 profiling_map_[
"seeds"] = 0.;
27 profiling_map_[
"ckf_setup"] = 0.;
28 profiling_map_[
"ckf_run"] = 0.;
29 profiling_map_[
"result_loop"] = 0.;
37 Acts::Vector3 b_field(0., 0., bfield_ * Acts::UnitConstants::T);
40 const auto const_b_field = std::make_shared<Acts::ConstantBField>(b_field);
48 surf_rotation_ = Acts::RotationMatrix3::Zero();
50 surf_rotation_(1, 0) = 1;
52 surf_rotation_(2, 1) = 1;
54 surf_rotation_(0, 2) = 1;
56 Acts::Vector3 target_pos(0., 0., 0.);
57 Acts::Translation3 target_translation(target_pos);
58 Acts::Transform3 target_transform(target_translation * surf_rotation_);
62 Acts::Surface::makeShared<Acts::PlaneSurface>(target_transform);
65 if (field_map_.empty())
70 std::static_pointer_cast<InterpolatedMagneticField3>(
bField());
72 auto acts_logging_level = Acts::Logging::FATAL;
73 if (debug_acts_) acts_logging_level = Acts::Logging::VERBOSE;
76 const auto stepper = Acts::EigenStepper<>{map};
77 const auto const_stepper = Acts::EigenStepper<>{const_b_field};
78 const auto multi_stepper = Acts::MultiEigenStepperLoop{map};
81 Acts::Navigator::Config nav_cfg{geometry().getTG()};
82 nav_cfg.resolveMaterial =
true;
83 nav_cfg.resolvePassive =
true;
84 nav_cfg.resolveSensitive =
true;
85 const Acts::Navigator navigator(nav_cfg);
87 propagator_ = std::make_unique<CkfPropagator>(
89 Acts::getDefaultLogger(
"CKF_PROP", acts_logging_level));
92 ckf_ = std::make_unique<std::decay_t<
decltype(*ckf_)>>(
93 *propagator_, Acts::getDefaultLogger(
"CKF", acts_logging_level));
94 trk_extrap_ = std::make_shared<std::decay_t<
decltype(*trk_extrap_)>>(
95 *propagator_, geometryContext(), magneticFieldContext());
98 Acts::ConstantBField zero_b_field(Acts::Vector3(0., 0., 0.));
99 const auto zero_b_stepper = Acts::EigenStepper<>{
100 std::make_shared<Acts::ConstantBField>(zero_b_field)};
102 std::make_unique<CkfPropagator>(zero_b_stepper, navigator);
103 ckf_zero_b_ = std::make_unique<std::decay_t<
decltype(*ckf_zero_b_)>>(
105 Acts::getDefaultLogger(
"CKF_ZERO_B", acts_logging_level));
107 std::make_shared<std::decay_t<
decltype(*trk_extrap_zero_b_)>>(
108 *propagator_zero_b_, geometryContext(), magneticFieldContext());
111 propagator_const_b_ =
112 std::make_unique<CkfPropagator>(const_stepper, navigator);
113 ckf_const_b_ = std::make_unique<std::decay_t<
decltype(*ckf_const_b_)>>(
114 *propagator_const_b_,
115 Acts::getDefaultLogger(
"CKF_CONST_B", acts_logging_level));
116 trk_extrap_const_b_ =
117 std::make_shared<std::decay_t<
decltype(*trk_extrap_const_b_)>>(
118 *propagator_const_b_, geometryContext(), magneticFieldContext());
128 std::vector<ldmx::Track> tracks;
130 auto start = std::chrono::high_resolution_clock::now();
134 ACTS_LOCAL_LOGGER(Acts::getDefaultLogger(
"LDMX Tracking Geometry Maker",
135 Acts::Logging::DEBUG));
138 Acts::PropagatorOptions<Acts::StepperPlainOptions,
139 Acts::NavigatorPlainOptions, ActionList, AbortList>
140 propagator_options(geometryContext(), magneticFieldContext());
142 propagator_options.pathLimit = std::numeric_limits<double>::max();
144 propagator_options.loopProtection =
false;
149 propagator_options.actionList.get<Acts::MaterialInteractor>();
150 m_interactor.multipleScattering =
true;
151 m_interactor.energyLoss =
true;
152 m_interactor.recordInteractions =
false;
156 propagator_options.actionList.get<Acts::detail::SteppingLogger>();
157 s_logger.sterile =
true;
159 propagator_options.stepping.maxStepSize =
160 propagator_step_size_ * Acts::UnitConstants::mm;
161 propagator_options.maxSteps = propagator_max_steps_;
171 auto setup = std::chrono::high_resolution_clock::now();
172 profiling_map_[
"setup"] +=
173 std::chrono::duration<double, std::milli>(setup - start).count();
176 measurement_collection_, input_pass_name_);
179 std::shared_ptr<tracking::sim::TruthMatchingTool> truth_matching_tool =
181 std::map<int, ldmx::SimParticle> particle_map;
183 if (event.
exists(sim_particles_coll_name_, sim_particles_event_passname_)) {
184 ldmx_log(debug) <<
"Setting up track truth matching tool";
186 sim_particles_coll_name_, sim_particles_event_passname_);
187 truth_matching_tool = std::make_shared<tracking::sim::TruthMatchingTool>(
188 particle_map, measurements);
193 const auto geo_id_sl_map = makeGeoIdSourceLinkMap(tg, measurements);
195 auto hits = std::chrono::high_resolution_clock::now();
196 profiling_map_[
"hits"] +=
197 std::chrono::duration<double, std::milli>(hits - setup).count();
202 const auto& seed_tracks =
203 event.getCollection<
ldmx::Track>(seed_coll_name_, input_pass_name_);
205 ldmx_log(info) <<
"Number of " << seed_coll_name_
206 <<
" seed tracks = " << seed_tracks.size();
208 if (seed_tracks.empty()) {
209 std::vector<ldmx::Track> empty;
210 ldmx_log(warn) <<
"No seed tracks, returning...";
211 event.add(out_trk_collection_, empty);
216 std::vector<Acts::BoundTrackParameters> start_parameters;
218 ldmx_log(debug) <<
"Transform the seed track to bound parameters";
219 int seed_track_index{0};
220 for (
auto& seed : seed_tracks) {
224 Acts::Vector3 perigee_acts = tracking::sim::utils::ldmx2Acts(Acts::Vector3(
225 seed.getPerigeeX(), seed.getPerigeeY(), seed.getPerigeeZ()));
226 std::shared_ptr<Acts::PerigeeSurface> perigee_surface =
227 Acts::Surface::makeShared<Acts::PerigeeSurface>(perigee_acts);
229 Acts::BoundVector param_vec;
230 param_vec << seed.getD0(), seed.getZ0(), seed.getPhi(), seed.getTheta(),
231 seed.getQoP(), seed.getT();
233 Acts::BoundSquareMatrix cov_mat =
234 tracking::sim::utils::unpackCov(seed.getPerigeeCov());
236 ldmx_log(debug) <<
" For seed index_ = " << seed_track_index
237 <<
": Perigee X / Y / Z = " << seed.getPerigeeX() <<
" / "
238 << seed.getPerigeeY() <<
" / " << seed.getPerigeeZ()
239 <<
", D0 = " << param_vec[0] <<
", Z0 = " << param_vec[1]
240 <<
", Phi = " << param_vec[2]
241 <<
", Theta = " << param_vec[3]
242 <<
", QoP = " << param_vec[4]
243 <<
", Time = " << param_vec[5];
245 ldmx_log(debug) <<
" Cov matrix diagonal (" << cov_mat(0, 0) <<
", "
246 << cov_mat(1, 1) <<
", " << cov_mat(2, 2) <<
")";
249 auto part_hypo{Acts::SinglyChargedParticleHypothesis::electron()};
250 start_parameters.push_back(Acts::BoundTrackParameters(
251 perigee_surface, param_vec, cov_mat, part_hypo));
259 auto seeds = std::chrono::high_resolution_clock::now();
260 profiling_map_[
"seeds"] +=
261 std::chrono::duration<double, std::milli>(seeds - hits).count();
263 Acts::GainMatrixUpdater kf_updater;
268 Acts::MeasurementSelector::Config measurement_selector_cfg = {
270 {Acts::GeometryIdentifier(), {{}, {outlier_pval_}, {1u}}},
273 Acts::MeasurementSelector meas_sel{measurement_selector_cfg};
277 Acts::CombinatorialKalmanFilterExtensions<TrackContainer> ckf_extensions;
279 if (use1_dmeasurements_) {
280 ckf_extensions.calibrator
282 Acts::VectorMultiTrajectory>>(&calibrator);
284 ckf_extensions.calibrator
286 Acts::VectorMultiTrajectory>>(&calibrator);
289 ckf_extensions.updater.connect<
290 &Acts::GainMatrixUpdater::operator()<Acts::VectorMultiTrajectory>>(
293 ckf_extensions.measurementSelector
294 .connect<&Acts::MeasurementSelector::select<Acts::VectorMultiTrajectory>>(
297 ldmx_log(debug) <<
"SourceLinkAccessor...";
300 struct SourceLinkAccIt {
301 using BaseIt =
decltype(geo_id_sl_map.begin());
304#pragma GCC diagnostic push
305#pragma GCC diagnostic ignored "-Wunused-local-typedefs"
307 using difference_type =
typename BaseIt::difference_type;
308 using iterator_category =
typename BaseIt::iterator_category;
310 using value_type = Acts::SourceLink;
311 using pointer =
typename BaseIt::pointer;
312 using reference = value_type&;
313#pragma GCC diagnostic pop
315 SourceLinkAccIt& operator++() {
319 bool operator==(
const SourceLinkAccIt& other)
const {
320 return it_ == other.it_;
322 bool operator!=(
const SourceLinkAccIt& other)
const {
323 return !(*
this == other);
328 value_type operator*()
const {
return value_type{it_->second}; }
331 auto source_link_accessor = [&](
const Acts::Surface& surface)
332 -> std::pair<SourceLinkAccIt, SourceLinkAccIt> {
333 auto [begin, end] = geo_id_sl_map.equal_range(surface.geometryId());
334 return {SourceLinkAccIt{begin}, SourceLinkAccIt{end}};
337 Acts::SourceLinkAccessorDelegate<SourceLinkAccIt>
338 source_link_accessor_delegate;
339 source_link_accessor_delegate
340 .connect<&
decltype(source_link_accessor)::operator(),
341 decltype(source_link_accessor)>(&source_link_accessor);
343 ldmx_log(debug) <<
"Setting up surfaces...";
345 std::shared_ptr<const Acts::PerigeeSurface> origin_surface =
346 Acts::Surface::makeShared<Acts::PerigeeSurface>(
347 Acts::Vector3(0., 0., 0.));
349 ldmx_log(debug) <<
"About to run CKF...";
352 auto ckf_setup = std::chrono::high_resolution_clock::now();
353 profiling_map_[
"ckf_setup"] +=
354 std::chrono::duration<double, std::milli>(ckf_setup - seeds).count();
356 Acts::VectorTrackContainer vtc;
357 Acts::VectorMultiTrajectory mtj;
358 Acts::TrackContainer tc{vtc, mtj};
362 ldmx_log(debug) <<
"Loop on the track candidates";
363 for (
size_t track_id = 0u; track_id < start_parameters.size(); ++track_id) {
364 ldmx_log(debug) <<
"---------------------------";
365 ldmx_log(debug) <<
"Candidate Track ID = " << track_id;
367 const Acts::CombinatorialKalmanFilterOptions<SourceLinkAccIt,
369 ckf_options(TrackingGeometryUser::geometryContext(),
370 TrackingGeometryUser::magneticFieldContext(),
371 TrackingGeometryUser::calibrationContext(),
372 source_link_accessor_delegate, ckf_extensions,
373 propagator_options,
true ,
376 ldmx_log(debug) <<
" Checking options: multiple scattering = "
377 << ckf_options.multipleScattering
378 <<
" energy loss = " << ckf_options.energyLoss;
382 ckf_->findTracks(start_parameters.at(track_id), ckf_options, tc);
384 auto start_params = start_parameters.at(track_id).parameters().transpose();
388 if (!tagger_tracking_) {
390 n_fieldmap_ckf_failed_recoil_++;
392 <<
" Field-map CKF failed, trying zero-B CKF fallback";
393 results = ckf_zero_b_->findTracks(start_parameters.at(track_id),
396 n_zerob_ckf_recovered_recoil_++;
397 ldmx_log(debug) <<
" Yay! Zero-B CKF succeeded as fallback!";
399 ldmx_log(debug) <<
" Zero-B CKF also failed!";
403 n_fieldmap_ckf_failed_tagger_++;
405 <<
" Field-map CKF failed, trying const-B (1.5T) CKF fallback";
406 results = ckf_const_b_->findTracks(start_parameters.at(track_id),
409 n_constb_ckf_recovered_tagger_++;
410 ldmx_log(debug) <<
" Yay! Const-B CKF succeeded as fallback!";
412 ldmx_log(debug) <<
" Const-B CKF also failed!";
418 <<
" Checking CKF success for track candidate with params: "
419 <<
" D0 = " << start_params[0] <<
" Z0 = " << start_params[1]
420 <<
", Phi = " << start_params[2] <<
" Theta = " << start_params[3]
421 <<
", QoP = " << start_params[4] <<
" Time = " << start_params[5];
422 if (not results.ok()) {
423 ldmx_log(debug) <<
" CKF failed!";
426 ldmx_log(debug) <<
" CKF succeded!";
429 auto& tracks_from_seed = results.value();
430 if (tracks_from_seed.size() != 1) {
431 ldmx_log(info) <<
" tracksFromSeed.size = " << tracks_from_seed.size();
434 for (
auto& track : tracks_from_seed) {
436 Acts::smoothTrack(geometryContext(), track);
441 auto opt_target = trk_extrap_->extrapolate(track, target_surface_);
444 if (tagger_tracking_) {
445 n_fieldmap_target_extrap_failed_tagger_++;
446 ldmx_log(debug) <<
" Field-map target extrapolation failed, "
447 "trying const-B (1.5T) fallback";
448 opt_target = trk_extrap_const_b_->extrapolate(track, target_surface_);
450 n_constb_target_extrap_recovered_tagger_++;
452 ldmx_log(debug) <<
" Both field-map and Const-B target "
453 "extrapolation failed!";
455 n_fieldmap_target_extrap_failed_recoil_++;
456 ldmx_log(debug) <<
" Field-map target extrapolation failed, "
457 "trying zero-B fallback";
458 opt_target = trk_extrap_zero_b_->extrapolate(track, target_surface_);
460 n_zerob_target_extrap_recovered_recoil_++;
463 <<
" Both field-map and Zero-B target extrapolation failed!";
468 ldmx_log(debug) <<
" Could not extrapolate to target! nhits = "
469 << track.nMeasurements() <<
" Printing track states:";
470 for (
const auto ts : track.trackStatesReversed()) {
471 if (ts.hasSmoothed())
472 ldmx_log(debug) <<
" Parameters: " << ts.smoothed().transpose();
474 ldmx_log(debug) <<
" Track state not smoothed!";
476 ldmx_log(debug) <<
" ...skipping this track candidate...";
480 ldmx_log(debug) <<
" Successfully obtained TrackState at target";
483 auto ts_at_target = tracking::sim::utils::makeTrackState(
484 geometryContext(), *opt_target, ldmx::AtTarget);
485 trk.addTrackState(ts_at_target);
487 ldmx_log(debug) <<
" Position at target (LDMX): ("
488 << ts_at_target.pos_[0] <<
", " << ts_at_target.pos_[1]
489 <<
", " << ts_at_target.pos_[2] <<
") mm"
490 <<
" Momentum: (" << ts_at_target.mom_[0] <<
", "
491 << ts_at_target.mom_[1] <<
", " << ts_at_target.mom_[2]
495 track.setReferenceSurface(target_surface_);
496 track.parameters() = opt_target->parameters();
499 trk.setPerigeeParameters(tracking::sim::utils::convertActsToLdmxPars(
500 opt_target->parameters()));
501 if (opt_target->covariance()) {
502 std::vector<double> cov_vec;
503 tracking::sim::utils::flatCov(*(opt_target->covariance()), cov_vec);
504 trk.setPerigeeCov(cov_vec);
507 Acts::Vector3 target_loc_ldmx = tracking::sim::utils::acts2Ldmx(
508 target_surface_->transform(geometryContext()).translation());
509 trk.setPerigeeLocation(target_loc_ldmx[0], target_loc_ldmx[1],
512 trk.setChi2(track.chi2());
513 trk.setNhits(track.nMeasurements());
514 trk.setNdf(track.nMeasurements() - 5);
515 trk.setNsharedHits(track.nSharedHits());
516 trk.setCharge(opt_target->parameters()[Acts::eBoundQOverP] > 0 ? 1 : -1);
519 if ((trk.getNhits() <= min_hits_) ||
520 (std::abs(1. / trk.getQoP()) <= 0.05)) {
522 <<
" > Track candidate did NOT meet the requirements: Nhits = "
523 << trk.getNhits() <<
" and p = " << std::abs(1. / trk.getQoP())
529 ldmx_log(debug) <<
" Add measurements to the final track from "
530 << track.nTrackStates() <<
" TrackStates with "
531 << track.nMeasurements() <<
" measurements";
533 int trk_state_index{0};
534 for (
const auto ts : track.trackStatesReversed()) {
536 ldmx_log(debug) <<
" Checking Track State index_ = "
537 << trk_state_index <<
" at location "
538 << ts.referenceSurface()
539 .transform(geometryContext())
543 if (ts.hasSmoothed()) {
544 ldmx_log(debug) <<
" Smoothed track parameters: "
545 << ts.smoothed().transpose();
551 auto type_flags = ts.typeFlags();
553 if (type_flags.test(Acts::TrackStateFlag::MeasurementFlag) &&
554 ts.hasUncalibratedSourceLink()) {
556 ts.getUncalibratedSourceLink()
557 .template get<acts_examples::IndexSourceLink>();
560 ldmx_log(debug) <<
" Adding measurement to ldmx::track with "
561 "source link index_ = "
563 ldmx_log(trace) <<
" Measurement:\n" << ldmx_meas;
564 trk.addMeasurementIndex(sl.
index());
572 if (ts.hasSmoothed()) {
574 static_cast<float>(ts.smoothed()[Acts::eBoundLoc0]),
575 static_cast<float>(ts.smoothedCovariance()(Acts::eBoundLoc0,
580 if (ts.hasSmoothed()) {
581 const auto& meas_surface = ts.referenceSurface();
582 const auto& smoothed_params = ts.smoothed();
587 float p_inv = smoothed_params[Acts::eBoundQOverP];
588 float p = 1.0f / std::abs(p_inv);
589 float theta = smoothed_params[Acts::eBoundTheta];
590 float phi = smoothed_params[Acts::eBoundPhi];
592 Acts::Vector3 global_momentum(p * std::sin(theta) * std::cos(phi),
593 p * std::sin(theta) * std::sin(phi),
594 p * std::cos(theta));
597 auto local_frame_transform =
598 meas_surface.transform(geometryContext());
599 Acts::Vector3 local_momentum =
600 local_frame_transform.rotation().transpose() * global_momentum;
603 float phi_u = (local_momentum.z() != 0)
604 ? local_momentum.x() / local_momentum.z()
606 float phi_v = (local_momentum.z() != 0)
607 ? local_momentum.y() / local_momentum.z()
614 float sensor_thickness = 0.0f;
615 if (
const auto* det_el = meas_surface.associatedDetectorElement()) {
616 sensor_thickness =
static_cast<float>(det_el->thickness());
618 ldmx_log(warn) <<
"No detector element for measurement surface"
619 <<
" — skipping dE/dx for this hit";
622 float tan_angle_sq = phi_u * phi_u + phi_v * phi_v;
623 float cos_angle = 1.0f / std::sqrt(1.0f + tan_angle_sq);
624 float path_length = sensor_thickness / cos_angle;
626 ldmx_log(debug) <<
" Local angles: phi_u = " << phi_u
627 <<
", phi_v = " << phi_v
628 <<
"; Path length = " << path_length <<
" mm";
631 float edep = ldmx_meas.
getEdep();
632 float dedx = edep / path_length;
633 trk.addDedxMeasurement(dedx);
635 ldmx_log(debug) <<
" Edep = " << edep
636 <<
" MeV, dE/dx = " << dedx <<
" MeV/mm";
639 ldmx_log(debug) <<
" This TrackState is not a measurement";
644 ldmx_log(debug) <<
" Starting extrapolations";
647 const double ecal_scoring_plane = 240.5;
648 Acts::Vector3 pos(ecal_scoring_plane, 0., 0.);
649 Acts::Translation3 surf_translation(pos);
650 Acts::Transform3 surf_transform(surf_translation * surf_rotation_);
651 const std::shared_ptr<Acts::PlaneSurface> ecal_surface =
652 Acts::Surface::makeShared<Acts::PlaneSurface>(surf_transform);
655 const std::shared_ptr<Acts::Surface> beam_origin_surface =
656 tracking::sim::utils::unboundSurface(-700);
658 if (tagger_tracking_) {
659 ldmx_log(debug) <<
" Beam Origin Extrapolation";
660 auto opt_beam_origin =
661 trk_extrap_->extrapolate(track, beam_origin_surface);
662 if (opt_beam_origin) {
663 trk.addTrackState(tracking::sim::utils::makeTrackState(
664 geometryContext(), *opt_beam_origin, ldmx::AtBeamOrigin));
666 <<
" Successfully obtained TrackState at beam origin";
671 if (!tagger_tracking_) {
672 ldmx_log(debug) <<
" Ecal Extrapolation";
673 auto opt_ecal = trk_extrap_->extrapolate(track, ecal_surface);
676 n_fieldmap_ecal_extrap_failed_recoil_++;
677 ldmx_log(debug) <<
" Field-map ECAL extrapolation failed, trying "
679 opt_ecal = trk_extrap_zero_b_->extrapolate(track, ecal_surface);
681 n_zerob_ecal_extrap_recovered_recoil_++;
684 <<
" Both field-map and Zero-B ECAL extrapolation failed!";
688 auto ts_at_ecal = tracking::sim::utils::makeTrackState(
689 geometryContext(), *opt_ecal, ldmx::AtECAL);
690 trk.addTrackState(ts_at_ecal);
691 ldmx_log(debug) <<
" Successfully obtained TrackState at ECAL";
692 ldmx_log(debug) <<
" Position at ECAL (LDMX): ("
693 << ts_at_ecal.pos_[0] <<
", " << ts_at_ecal.pos_[1]
694 <<
", " << ts_at_ecal.pos_[2] <<
") mm";
699 if (truth_matching_tool) {
700 auto truth_info = truth_matching_tool->truthMatch(trk);
701 trk.setTrackID(truth_info.track_id_);
702 trk.setPdgID(truth_info.pdg_id_);
703 trk.setTruthProb(truth_info.truth_prob_);
708 <<
" > Adding the track candidate to the track collection";
709 tracks.push_back(trk);
714 ldmx_log(info) <<
"Number of CKF tracks " << tracks.size();
716 auto ckf_run = std::chrono::high_resolution_clock::now();
717 profiling_map_[
"ckf_run"] +=
718 std::chrono::duration<double, std::milli>(ckf_run - ckf_setup).count();
721 auto shared_hits = computeSharedHits(
722 tracks, measurements, tg, tracking::sim::utils::sourceLinkHash,
723 tracking::sim::utils::sourceLinkEquality);
724 for (std::size_t i_track = 0; i_track < shared_hits.size(); ++i_track) {
725 tracks[i_track].setNsharedHits(shared_hits[i_track].size());
726 for (
auto idx : shared_hits[i_track]) {
727 tracks[i_track].addSharedIndex(idx);
731 auto result_loop = std::chrono::high_resolution_clock::now();
732 profiling_map_[
"result_loop"] +=
733 std::chrono::duration<double, std::milli>(result_loop - ckf_run).count();
736 event.add(out_trk_collection_, tracks);
738 auto end = std::chrono::high_resolution_clock::now();
741 auto diff = end - start;
742 processing_time_ += std::chrono::duration<double, std::milli>(diff).count();