LDMX Software
GreedyAmbiguitySolver.cxx
1#include "Tracking/Reco/GreedyAmbiguitySolver.h"
2
3#include <algorithm>
4
5#include "Acts/EventData/SourceLink.hpp"
6#include "Acts/EventData/TrackHelpers.hpp"
7
8namespace tracking {
9namespace reco {
10
12 framework::Process& process)
13 : TrackingGeometryUser(name, process) {}
14
15// Helper Functions
16
17/*
18std::size_t GreedyAmbiguitySolver::sourceLinkHash(const Acts::SourceLink& a) {
19 return static_cast<std::size_t>(
20 a.get<ActsExamples::IndexSourceLink>().index());
21 }
22
23bool GreedyAmbiguitySolver::sourceLinkEquality(const Acts::SourceLink& a, const
24Acts::SourceLink& b) { return a.get<ActsExamples::IndexSourceLink>().index() ==
25 b.get<ActsExamples::IndexSourceLink>().index();
26}
27*/
28
30 std::size_t iTrack) const {
31 for (auto i_measurement : state.measurements_per_track_[iTrack]) {
32 state.tracks_per_measurement_[i_measurement].erase(iTrack);
33 if (state.tracks_per_measurement_[i_measurement].size() == 1) {
34 auto j_track = *state.tracks_per_measurement_[i_measurement].begin();
35 --state.shared_measurements_per_track_[j_track];
36 }
37 }
38 state.selected_tracks_.erase(iTrack);
39}
40
41template <typename geometry_t, typename source_link_hash_t,
42 typename source_link_equality_t>
44 std::vector<ldmx::Track> tracks, std::vector<ldmx::Measurement> meas_coll,
45 State& state, geometry_t& tg, source_link_hash_t&& sourceLinkHash,
46 source_link_equality_t&& sourceLinkEquality) const {
47 auto measurement_index_map =
48 std::unordered_map<Acts::SourceLink, std::size_t, source_link_hash_t,
49 source_link_equality_t>(0, sourceLinkHash,
50 sourceLinkEquality);
51
52 // auto tg{geometry()};
53 // Iterate through all input tracks, collect their properties like measurement
54 // count and chi2 and fill the measurement map in order to relate tracks to
55 // each other if they have shared hits_.
56 state.number_of_tracks_ = 0;
57 for (const auto& track : tracks) {
58 // Kick out tracks that do not fulfill our initial requirements
59 if (track.getNhits() < n_meas_min_) {
60 continue;
61 }
62
63 std::vector<std::size_t> measurements;
64 for (auto imeas : track.getMeasurementsIdxs()) {
65 auto meas = meas_coll.at(imeas);
66 const Acts::Surface* hit_surface = tg.getSurface(meas.getLayerID());
67 // Store the index_ source link
68 acts_examples::IndexSourceLink idx_sl(hit_surface->geometryId(), imeas);
69 Acts::SourceLink source_link = Acts::SourceLink(idx_sl);
70
71 auto emplace = measurement_index_map.try_emplace(
72 source_link, measurement_index_map.size());
73 measurements.push_back(emplace.first->second);
74 }
75
76 state.track_tips_.push_back(state.number_of_tracks_);
77 state.track_chi2_.push_back(track.getChi2() / track.getNdf());
78 state.measurements_per_track_.push_back(std::move(measurements));
79 state.selected_tracks_.insert(state.number_of_tracks_);
80
81 ++state.number_of_tracks_;
82 }
83
84 // Now we relate measurements to tracks
85 for (std::size_t i_track = 0; i_track < state.number_of_tracks_; ++i_track) {
86 for (auto i_measurement : state.measurements_per_track_[i_track]) {
87 state.tracks_per_measurement_[i_measurement].insert(i_track);
88 }
89 }
90
91 // Finally, we can accumulate the number of shared measurements per track
92 state.shared_measurements_per_track_ =
93 std::vector<std::size_t>(state.track_tips_.size(), 0);
94 for (std::size_t i_track = 0; i_track < state.number_of_tracks_; ++i_track) {
95 for (auto i_measurement : state.measurements_per_track_[i_track]) {
96 if (state.tracks_per_measurement_[i_measurement].size() > 1) {
97 ++state.shared_measurements_per_track_[i_track];
98 }
99 }
100 }
101}
102
106 auto shared_measurements_comperator = [&state](std::size_t a, std::size_t b) {
107 return state.shared_measurements_per_track_[a] <
108 state.shared_measurements_per_track_[b];
109 };
110
114 auto track_comperator = [&state](std::size_t a, std::size_t b) {
116 auto relative_shared_measurements = [&state](std::size_t i) {
117 return 1.0 * state.shared_measurements_per_track_[i] /
118 state.measurements_per_track_[i].size();
119 };
120
121 if (relative_shared_measurements(a) != relative_shared_measurements(b)) {
122 return relative_shared_measurements(a) < relative_shared_measurements(b);
123 }
124 return state.track_chi2_[a] < state.track_chi2_[b];
125 };
126
127 for (std::size_t i = 0; i < maximum_iterations_; ++i) {
128 // Lazy out if there is nothing to filter on.
129 if (state.selected_tracks_.empty()) {
130 ldmx_log(trace) << "No tracks left - exit loop";
131 break;
132 }
133
134 // Find the maximum amount of shared measurements per track to decide if we
135 // are done or not.
136 auto maximum_shared_measurements = *std::max_element(
137 state.selected_tracks_.begin(), state.selected_tracks_.end(),
138 shared_measurements_comperator);
139 // ldmx_log(debug) <<
140 // "maximum shared measurements "
141 // << state.sharedMeasurementsPerTrack[maximumSharedMeasurements];
142 if (state.shared_measurements_per_track_[maximum_shared_measurements] <
144 break;
145 }
146
147 // Find the "worst" track by comparing them to each other
148 auto bad_track =
149 *std::max_element(state.selected_tracks_.begin(),
150 state.selected_tracks_.end(), track_comperator);
151 ldmx_log(trace) << "Remove track " << bad_track << ", nMeas = "
152 << state.measurements_per_track_[bad_track].size()
153 << ", nShared = "
154 << state.shared_measurements_per_track_[bad_track]
155 << ", chi2 =" << state.track_chi2_[bad_track];
156 removeTrack(state, bad_track);
157 }
158}
159
160// Processor Functions
161
163
165 framework::config::Parameters& parameters) {
166 out_trk_collection_ =
167 parameters.get<std::string>("out_trk_collection", "TaggerTracksClean");
168
169 track_collection_ =
170 parameters.get<std::string>("trackCollection", "TaggerTracks");
171
172 meas_collection_ =
173 parameters.get<std::string>("measCollection", "DigiTaggerSimHits");
174 input_pass_name_ = parameters.get<std::string>("input_pass_name");
175 n_meas_min_ = parameters.get<int>("nMeasurementsMin", 5);
176 maximum_shared_hits_ = parameters.get<int>("maximumSharedHits", 1);
177}
178
181 std::vector<ldmx::Track> out_tracks;
182
183 auto tg{geometry()};
184
185 if (!event.exists(track_collection_, input_pass_name_)) {
186 ldmx_log(debug) << "Track collection not found, exiting";
187 return;
188 }
189 auto tracks{
190 event.getCollection<ldmx::Track>(track_collection_, input_pass_name_)};
191
192 if (!event.exists(meas_collection_, input_pass_name_)) {
193 ldmx_log(debug) << "Measurement collection not found, exiting";
194 return;
195 }
196 auto measurements{event.getCollection<ldmx::Measurement>(meas_collection_,
197 input_pass_name_)};
198
199 computeInitialState(tracks, measurements, state, tg,
200 tracking::sim::utils::sourceLinkHash,
201 tracking::sim::utils::sourceLinkEquality);
202 resolve(state);
203
204 for (auto i_track : state.selected_tracks_) {
205 auto clean_trk = tracks[state.track_tips_.at(i_track)];
206 if ((clean_trk.getNhits() > n_meas_min_) &&
207 (std::abs(1. / clean_trk.getQoP()) > 0.05)) {
208 out_tracks.push_back(clean_trk);
209 }
210 }
211
212 event.add(out_trk_collection_, out_tracks);
213
214 // for (auto iTrack : initial_state.selectedTracks) {
215 // std::cout << event.getEventNumber() << " " << iTrack << " " <<
216 // initial_state.trackChi2[iTrack] << " " <<
217 // initial_state.measurementsPerTrack[iTrack].size() << std::endl;
218 // }
219
220 ldmx_log(info) << " Resolved to " << state.selected_tracks_.size()
221 << " tracks from " << " " << tracks.size();
222}
223
224} // namespace reco
225} // namespace tracking
226
#define DECLARE_PRODUCER(CLASS)
Macro which allows the framework to construct a producer given its name during configuration.
Implements an event buffer system for storing event data.
Definition Event.h:42
bool exists(const std::string &name, const std::string &passName, bool unique=true) const
Check for the existence of an object or collection with the given name and pass name in the event.
Definition Event.cxx:92
Class which represents the process under execution.
Definition Process.h:36
Class encapsulating parameters for configuring a processor.
Definition Parameters.h:29
const T & get(const std::string &name) const
Retrieve the parameter of the given name.
Definition Parameters.h:78
Run-specific configuration and data stored in its own output TTree alongside the event TTree in the o...
Definition RunHeader.h:57
Implementation of a track object.
Definition Track.h:53
Minimal example of a processor.
GreedyAmbiguitySolver(const std::string &name, framework::Process &process)
Constructor.
std::size_t n_meas_min_
Minimum number of measurement to form a track.
void removeTrack(State &state, std::size_t iTrack) const
void configure(framework::config::Parameters &parameters) override
Configure the processor using the given user specified parameters.
void computeInitialState(std::vector< ldmx::Track > tracks, std::vector< ldmx::Measurement > measurements, State &state, geometry_t &tg, source_link_hash_t &&sourceLinkHash, source_link_equality_t &&sourceLinkEquality) const
void resolve(State &state)
Updates the state iteratively by evicting one track after the other until the final state conditions ...
void onNewRun(const ldmx::RunHeader &rh) override
onNewRun is the first function called for each processor after the conditions are fully configured an...
void produce(framework::Event &event) override
Process the event and put new data products into it.
std::uint32_t maximum_iterations_
Maximum number of iterations.
std::uint32_t maximum_shared_hits_
Maximum amount of shared hits per track.
a helper base class providing some methods to shorten access to common conditions used within the tra...
The measurement calibrator can be a function or a class/struct able to retrieve the sim hits containe...