LDMX Software
LinearTruthTracking.cxx
1#include "Tracking/Reco/LinearTruthTracking.h"
2
3#include "Eigen/Dense"
4
5namespace tracking {
6namespace reco {
7
9 framework::Process& process)
10 : TrackingGeometryUser(name, process) {}
11
13 // Output seed name
14 out_trk_collection_ = parameters.get<std::string>("out_trk_collection",
15 "LinearRecoilTruthTracks");
16
17 // Input strip hits_
19 parameters.get<std::string>("input_hits_collection", "RecoilSimHits");
21 parameters.get<std::string>("input_recHits_collection", "EcalRecHits");
22
23 input_pass_name_ = parameters.get<std::string>("input_pass_name");
24
25 ecal_first_layer_z_threshold_ =
26 parameters.get<double>("ecal_first_layer_z_threshold");
27} // configure
28
30 auto start = std::chrono::high_resolution_clock::now();
31 std::vector<ldmx::StraightTrack> straight_truth_tracks;
32 n_events_++;
33
34 const std::vector<ldmx::SimTrackerHit> recoil_hits =
36 input_pass_name_);
37 const std::vector<ldmx::EcalHit> ecal_rec_hit =
38 event.getCollection<ldmx::EcalHit>(input_rec_hits_collection_,
39 input_pass_name_);
40
41 std::vector<std::array<double, 3>> first_layer_ecal_rec_hits;
42
43 for (const auto& x_ecal : ecal_rec_hit) {
44 if (x_ecal.getZPos() < ecal_first_layer_z_threshold_) {
45 first_layer_ecal_rec_hits.push_back(
46 {x_ecal.getZPos(), x_ecal.getXPos(), x_ecal.getYPos()});
47 } // if first layer_ of Ecal
48 } // for positions in ecalRecHit
49
50 // Check if we would fit empty seeds.
51 if (recoil_hits.size() < 2) {
52 n_missing_++;
53 n_truth_ += straight_truth_tracks.size();
54 event.add(out_trk_collection_, straight_truth_tracks);
55 return;
56 } // if not enough hits_
57
58 std::vector<ldmx::SimTrackerHit> truth_points;
59
60 // Find only recoil electron out of the hit collection
61 for (const auto& point : recoil_hits) {
62 if (point.getTrackID() == 1) {
63 truth_points.push_back(point);
64 } // if trackID == 1
65 } // for point in hit collection
66
67 // should only find 1 track (recoil e-) or 0 track (no recoil e-)
68 if (truth_points.size() > 1) {
69 straight_truth_tracks.push_back(
70 truthTracker(truth_points, first_layer_ecal_rec_hits));
71 } // if we have truth points
72 else {
73 n_empty_++;
74 n_truth_ += straight_truth_tracks.size();
75 return;
76 } // else no truth points
77
78 n_truth_ += straight_truth_tracks.size();
79 event.add(out_trk_collection_, straight_truth_tracks);
80
81 auto end = std::chrono::high_resolution_clock::now();
82 auto diff = end - start;
83 processing_time_ += std::chrono::duration<double, std::milli>(diff).count();
84
85 first_layer_ecal_rec_hits.clear();
86 straight_truth_tracks.clear();
87
88} // produce
89
90ldmx::StraightTrack LinearTruthTracking::truthTracker(
91 const std::vector<ldmx::SimTrackerHit>& points,
92 std::vector<std::array<double, 3>>& ecal_points) {
94
95 // We have pre-selected hits_ corresponding to the Recoil e-, so we only fit
96 // these (no RecHit)
97 auto [m_x, b_x, m_y, b_y] = fit3DLine(points);
98
99 trk.setSlopeX(m_x);
100 trk.setInterceptX(b_x);
101 trk.setSlopeY(m_y);
102 trk.setInterceptY(b_y);
103 trk.setTheta(std::atan2(m_y, std::sqrt(1 + m_x * m_x)));
104 trk.setPhi(std::atan2(m_x, 1.0));
105 trk.setTargetLocation(0.0, b_x, b_y);
106 trk.setNhits(points.size());
107 trk.setNdf(points.size() - 2);
108
109 trk.setTrackID(points[0].getTrackID());
110 trk.setPdgID(points[0].getPdgID());
111 trk.setTruthProb(1.0);
112
113 trk.setAllTruthSensorPoints(points);
114
115 trk.setChi2(globalChiSquare(points, m_x, m_y, b_x, b_y));
116
117 // Z position from the first point in ecalPoints
118 if (ecal_points.size() > 0) {
119 double ecal_first_layer_z = ecal_points[0][0];
120
121 // Extrapolate track to first layer_ of Ecal
122 std::array<double, 3> extrapolated_point = {ecal_first_layer_z,
123 m_x * ecal_first_layer_z + b_x,
124 m_y * ecal_first_layer_z + b_y};
125
126 const std::array<double, 3>* closest_rec_hit = nullptr;
127 double min_distance = std::numeric_limits<double>::max();
128
129 // Loop through ecalPoints to find the closest recHit to our track
130 for (const auto& ecal_rec_hit : ecal_points) {
131 double temp_distance =
132 calculateDistance(extrapolated_point, ecal_rec_hit);
133
134 if (temp_distance < min_distance) {
135 min_distance = temp_distance;
136 closest_rec_hit = &ecal_rec_hit;
137 } // if compare min distance
138 } // for recHits
139
140 if (closest_rec_hit != nullptr) {
141 trk.setFirstLayerEcalRecHit(*closest_rec_hit);
142 trk.setDistancetoRecHit(min_distance);
143 trk.setEcalLayer1Location(extrapolated_point);
144 } // if set trk info
145 } // make sure there are ecalPoints to work with
146
147 return trk;
148} // truthTracker
149
151 ldmx_log(info) << "AVG Time / Event: " << std::fixed << std::setprecision(1)
152 << processing_time_ / n_events_ << " ms";
153 ldmx_log(info) << "Number of Events without enough seed points: "
154 << n_missing_;
155 ldmx_log(info) << "Number of Events with only 1 trackID == 1 point: "
156 << n_empty_;
157 ldmx_log(info) << "Total Truth Tracks / Event: " << n_truth_ << "/"
158 << n_events_;
159} // onProcessEnd
160
161double LinearTruthTracking::calculateDistance(
162 const std::array<double, 3>& point1, const std::array<double, 3>& point2) {
163 return sqrt(pow(point1[1] - point2[1], 2) + pow(point1[2] - point2[2], 2));
164} // calculateDistance
165
166std::tuple<double, double, double, double> LinearTruthTracking::fit3DLine(
167 const std::vector<ldmx::SimTrackerHit>& points) {
168 std::vector<double> z_vals, x_vals, y_vals;
169
170 for (const auto& point : points) {
171 const auto& position = point.getPosition();
172 x_vals.push_back(position[0]);
173 y_vals.push_back(position[1]);
174 z_vals.push_back(position[2]);
175 }
176
177 size_t num_points = points.size();
178 Eigen::MatrixXd a_mat(num_points, 2);
179 Eigen::VectorXd x_vec(num_points), y_vec(num_points);
180
181 for (size_t i = 0; i < num_points; ++i) {
182 a_mat(i, 0) = z_vals[i];
183 a_mat(i, 1) = 1;
184 x_vec(i) = x_vals[i];
185 y_vec(i) = y_vals[i];
186 }
187
188 Eigen::Matrix2d at_a = a_mat.transpose() * a_mat;
189 Eigen::Vector2d at_x = a_mat.transpose() * x_vec;
190 Eigen::Vector2d at_y = a_mat.transpose() * y_vec;
191
192 Eigen::Vector2d x_params = at_a.ldlt().solve(at_x);
193 Eigen::Vector2d y_params = at_a.ldlt().solve(at_y);
194
195 // Return (mx, bx, my, by)
196 return {x_params(0), x_params(1), y_params(0), y_params(1)};
197}
198
199double LinearTruthTracking::globalChiSquare(
200 const std::vector<ldmx::SimTrackerHit>& points, double m_x, double m_y,
201 double b_x, double b_y) {
202 double chi2_x = 0, chi2_y = 0;
203
204 for (const auto& point : points) {
205 const auto& position = point.getPosition();
206 double x_pos = position[0];
207 double y_pos = position[1];
208 double z_pos = position[2];
209
210 double x_fit = m_x * z_pos + b_x;
211 double y_fit = m_y * z_pos + b_y;
212
213 chi2_x += std::pow((x_pos - x_fit), 2);
214 chi2_y += std::pow((y_pos - y_fit), 2);
215 }
216
217 return chi2_x + chi2_y;
218} // globalChiSquare
219
220} // namespace reco
221} // namespace tracking
222
#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
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
Stores reconstructed hit information from the ECAL.
Definition EcalHit.h:19
Represents a simulated tracker hit in the simulation.
std::string input_rec_hits_collection_
The name of the tagger Tracks (only for Recoil Seeding)
void produce(framework::Event &event) override
Run the processor and create a collection of results which indicate if a charge particle can be found...
std::string input_hits_collection_
The name of the input hits collection to use in finding seeds..
LinearTruthTracking(const std::string &name, framework::Process &process)
Constructor.
std::string out_trk_collection_
The name of the output collection of seeds to be stored.
void onProcessEnd() override
Output event statistics.
void configure(framework::config::Parameters &parameters) override
Configure the processor using the given user specified parameters.
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...