12 double cluster_threshold,
double noise_sigma_adc,
13 double mean_time_ns,
double time_window_ns,
14 double neighbor_delta_t_ns,
double max_chi2_ndf)
15 : seed_threshold_(seed_threshold),
16 neighbor_threshold_(neighbor_threshold),
17 cluster_threshold_(cluster_threshold),
18 noise_sigma_adc_(noise_sigma_adc),
19 mean_time_ns_(mean_time_ns),
20 time_window_ns_(time_window_ns),
21 neighbor_delta_t_ns_(neighbor_delta_t_ns),
22 max_chi2_ndf_(max_chi2_ndf) {}
51 const std::vector<ldmx::FittedSiStripHit>& hits)
const {
56 std::map<int, const ldmx::FittedSiStripHit*> channel_map;
57 for (
const auto& h : hits) {
58 const int ch = h.getStripID();
59 auto it = channel_map.find(ch);
60 if (it == channel_map.end()) {
64 if (std::abs(h.
getT0() - mean_time_ns_) <
65 std::abs(it->second->getT0() - mean_time_ns_)) {
75 const double snr_neighbor = neighbor_threshold_ * noise_sigma_adc_;
76 const double snr_seed = seed_threshold_ * noise_sigma_adc_;
78 std::set<int> clusterable_set;
79 std::vector<int> seed_channels;
81 for (
const auto& [ch, hp] : channel_map) {
82 const double amp = hp->getAmplitude();
83 if (amp >= snr_neighbor) {
84 clusterable_set.insert(ch);
86 if (amp >= snr_seed && passesSeedCuts(*hp)) {
87 seed_channels.push_back(ch);
92 std::sort(seed_channels.begin(), seed_channels.end(), [&](
int a,
int b) {
93 return channel_map.at(a)->getAmplitude() >
94 channel_map.at(b)->getAmplitude();
100 std::vector<ClusterCandidate> clusters;
102 for (
int seed_ch : seed_channels) {
104 if (clusterable_set.find(seed_ch) == clusterable_set.end())
continue;
107 double cluster_weighted_t = 0.0;
108 double cluster_total_amp = 0.0;
109 double cluster_noise_sq = 0.0;
111 std::deque<int> unchecked;
112 unchecked.push_back(seed_ch);
113 clusterable_set.erase(seed_ch);
115 while (!unchecked.empty()) {
116 const int cur_ch = unchecked.front();
117 unchecked.pop_front();
123 cand.strip_ids.push_back(cur_ch);
124 cluster_total_amp += amp;
125 cluster_weighted_t += amp * hit.
getT0();
126 cluster_noise_sq += noise_sigma_adc_ * noise_sigma_adc_;
129 for (
int delta : {-1, +1}) {
130 const int nb_ch = cur_ch + delta;
131 if (clusterable_set.find(nb_ch) == clusterable_set.end())
continue;
134 if (!passesNeighborCuts(*channel_map.at(nb_ch), cluster_weighted_t,
135 cluster_total_amp)) {
139 unchecked.push_back(nb_ch);
140 clusterable_set.erase(nb_ch);
145 if (cluster_noise_sq <= 0.0)
continue;
146 if (cluster_total_amp / std::sqrt(cluster_noise_sq) < cluster_threshold_) {
153 double sum_amp_strip = 0.0;
154 for (
int ch : cand.strip_ids) {
155 sum_amp_strip += channel_map.at(ch)->getAmplitude() * ch;
160 cand.
time_ns = cluster_weighted_t / cluster_total_amp;
161 cand.n_strips =
static_cast<int>(cand.strip_ids.size());
168 double sum_amp_dsq = 0.0;
169 for (
int ch : cand.strip_ids) {
171 sum_amp_dsq += channel_map.at(ch)->getAmplitude() * d * d;
173 constexpr double k_single_strip_sigma = 1.0 / 3.4641;
174 const double rms = std::sqrt(sum_amp_dsq / cluster_total_amp);
175 cand.
sigma_strip = (rms > 0.0) ? rms : k_single_strip_sigma;
176 cand.layer_id = channel_map.at(seed_ch)->getLayerID();
178 clusters.push_back(std::move(cand));
StripClusterer(double seed_threshold=4.0, double neighbor_threshold=3.0, double cluster_threshold=4.0, double noise_sigma_adc=5.0, double mean_time_ns=0.0, double time_window_ns=-1.0, double neighbor_delta_t_ns=-1.0, double max_chi2_ndf=-1.0)