File indexing completed on 2025-10-24 08:23:01
0001
0002
0003
0004
0005
0006
0007
0008
0009
0010
0011
0012
0013
0014
0015 #include "PhotoMultiplierHitDigi.h"
0016
0017 #include <Evaluator/DD4hepUnits.h>
0018 #include <algorithm>
0019 #include <algorithms/logger.h>
0020 #include <cmath>
0021 #include <edm4hep/Vector3d.h>
0022 #include <fmt/core.h>
0023 #include <gsl/pointers>
0024 #include <iterator>
0025 #include <podio/ObjectID.h>
0026
0027 #include "algorithms/digi/PhotoMultiplierHitDigiConfig.h"
0028
0029 namespace eicrecon {
0030
0031
0032
0033
0034 void PhotoMultiplierHitDigi::init() {
0035
0036 debug() << m_cfg << endmsg;
0037
0038
0039 qe_init();
0040 }
0041
0042
0043
0044
0045 void PhotoMultiplierHitDigi::process(const PhotoMultiplierHitDigi::Input& input,
0046 const PhotoMultiplierHitDigi::Output& output) const {
0047 const auto [headers, sim_hits] = input;
0048 auto [raw_hits, hit_assocs] = output;
0049
0050
0051 auto seed = m_uid.getUniqueID(*headers, name());
0052 std::default_random_engine generator(seed);
0053 std::normal_distribution<double> gaussian;
0054 std::uniform_real_distribution<double> uniform;
0055
0056 trace("{:=^70}", " call PhotoMultiplierHitDigi::process ");
0057 std::unordered_map<CellIDType, std::vector<HitData>> hit_groups;
0058
0059
0060 trace("{:-<70}", "Loop over simulated hits ");
0061 for (std::size_t sim_hit_index = 0; sim_hit_index < sim_hits->size(); sim_hit_index++) {
0062 const auto& sim_hit = sim_hits->at(sim_hit_index);
0063 auto edep_eV = sim_hit.getEDep() *
0064 1e9;
0065 auto id = sim_hit.getCellID();
0066 trace("hit: pixel id={:#018X} edep = {} eV", id, edep_eV);
0067
0068
0069 if (uniform(generator) > m_cfg.safetyFactor) {
0070 continue;
0071 }
0072
0073
0074 if (m_cfg.enableQuantumEfficiency and !qe_pass(edep_eV, uniform(generator))) {
0075 continue;
0076 }
0077
0078
0079 if (m_cfg.enablePixelGaps) {
0080 auto pos = sim_hit.getPosition();
0081 if (!m_PixelGapMask(
0082 id, dd4hep::Position(pos.x * dd4hep::mm, pos.y * dd4hep::mm, pos.z * dd4hep::mm))) {
0083 continue;
0084 }
0085 }
0086
0087
0088 trace(" -> hit accepted");
0089 trace(" -> MC hit id={}", sim_hit.getObjectID().index);
0090 auto time = sim_hit.getTime();
0091 double amp = m_cfg.speMean + gaussian(generator) * m_cfg.speError;
0092
0093
0094 InsertHit(hit_groups, id, amp, time, sim_hit_index, generator, gaussian);
0095 }
0096
0097
0098 if (level() <= algorithms::LogLevel::kTrace) {
0099 trace("{:-<70}", "Accepted hit groups ");
0100 for (auto& [id, hitVec] : hit_groups) {
0101 for (auto& hit : hitVec) {
0102 trace("hit_group: pixel id={:#018X} -> npe={} signal={} time={}", id, hit.npe, hit.signal,
0103 hit.time);
0104 for (auto i : hit.sim_hit_indices) {
0105 trace(" - MC hit: EDep={}, id={}", sim_hits->at(i).getEDep(),
0106 sim_hits->at(i).getObjectID().index);
0107 }
0108 }
0109 }
0110 }
0111
0112
0113 if (m_cfg.enableNoise) {
0114 trace("{:=^70}", " BEGIN NOISE INJECTION ");
0115 float p = m_cfg.noiseRate * m_cfg.noiseTimeWindow;
0116 auto cellID_action = [this, &gaussian, &generator, &hit_groups, &uniform](auto id) {
0117
0118 double amp = m_cfg.speMean + gaussian(generator) * m_cfg.speError;
0119 TimeType time = m_cfg.noiseTimeWindow * uniform(generator) / dd4hep::ns;
0120
0121
0122 this->InsertHit(hit_groups, id, amp, time,
0123 0,
0124 generator, gaussian, true);
0125 };
0126 m_VisitRngCellIDs(cellID_action, p);
0127 }
0128
0129
0130 trace("{:-<70}", "Digitized raw hits ");
0131 for (auto& it : hit_groups) {
0132 for (auto& data : it.second) {
0133
0134
0135 auto raw_hit = raw_hits->create();
0136 raw_hit.setCellID(it.first);
0137 raw_hit.setCharge(static_cast<decltype(edm4eic::RawTrackerHitData::charge)>(data.signal));
0138 raw_hit.setTimeStamp(static_cast<decltype(edm4eic::RawTrackerHitData::timeStamp)>(
0139 data.time / m_cfg.timeResolution));
0140 trace("raw_hit: cellID={:#018X} -> charge={} timeStamp={}", raw_hit.getCellID(),
0141 raw_hit.getCharge(), raw_hit.getTimeStamp());
0142
0143
0144 if (!data.sim_hit_indices.empty()) {
0145 for (auto i : data.sim_hit_indices) {
0146 auto hit_assoc = hit_assocs->create();
0147 hit_assoc.setWeight(1.0 / data.sim_hit_indices.size());
0148 hit_assoc.setRawHit(raw_hit);
0149 hit_assoc.setSimHit(sim_hits->at(i));
0150 }
0151 }
0152 }
0153 }
0154 }
0155
0156 void PhotoMultiplierHitDigi::qe_init() {
0157
0158 qeff.clear();
0159 auto hc = dd4hep::h_Planck * dd4hep::c_light / (dd4hep::eV * dd4hep::nm);
0160 for (const auto& [wl, qe] : m_cfg.quantumEfficiency) {
0161 qeff.emplace_back(hc / wl, qe);
0162 }
0163
0164
0165 std::ranges::sort(qeff, [](const std::pair<double, double>& v1,
0166 const std::pair<double, double>& v2) { return v1.first < v2.first; });
0167
0168
0169 debug("{:-^60}", " Quantum Efficiency vs. Energy ");
0170 for (auto& [en, qe] : qeff) {
0171 debug(" {:>10.4} {:<}", en, qe);
0172 }
0173 trace("{:=^60}", "");
0174
0175 if (m_cfg.enableQuantumEfficiency) {
0176
0177 if (qeff.empty()) {
0178 qeff = {{2.6, 0.3}, {7.0, 0.3}};
0179 warning("Invalid quantum efficiency data provided, using default values {} {:.2f} {} {:.2f} "
0180 "{} {:.2f} {} {:.2f} {}",
0181 "{{", qeff.front().first, ",", qeff.front().second, "},{", qeff.back().first, ",",
0182 qeff.back().second, "}}");
0183 }
0184 if (qeff.front().first > 3.0) {
0185 warning("Quantum efficiency data start from {:.2f} {}", qeff.front().first,
0186 " eV, maybe you are using wrong units?");
0187 }
0188 if (qeff.back().first < 3.0) {
0189 warning("Quantum efficiency data end at {:.2f} {}", qeff.back().first,
0190 " eV, maybe you are using wrong units?");
0191 }
0192 }
0193 }
0194
0195 template <class RndmIter, typename T, class Compare>
0196 RndmIter PhotoMultiplierHitDigi::interval_search(RndmIter beg, RndmIter end, const T& val,
0197 Compare comp) const {
0198
0199 auto dist = std::distance(beg, end);
0200 if ((dist < 2) || (comp(*beg, val) > 0) || (comp(*std::prev(end), val) < 0)) {
0201 return end;
0202 }
0203 auto mid = std::next(beg, dist / 2);
0204
0205 while (mid != end) {
0206 if (comp(*mid, val) == 0) {
0207 return mid;
0208 }
0209 if (comp(*mid, val) > 0) {
0210 end = mid;
0211 } else {
0212 beg = std::next(mid);
0213 }
0214 mid = std::next(beg, std::distance(beg, end) / 2);
0215 }
0216
0217 if (mid == end || comp(*mid, val) > 0) {
0218 return std::prev(mid);
0219 }
0220 return mid;
0221 }
0222
0223 bool PhotoMultiplierHitDigi::qe_pass(double ev, double rand) const {
0224 auto it = interval_search(
0225 qeff.begin(), qeff.end(), ev,
0226 [](const std::pair<double, double>& vals, double val) { return vals.first - val; });
0227
0228 if (it == qeff.end()) {
0229
0230 return false;
0231 }
0232
0233 double prob = it->second;
0234 auto itn = std::next(it);
0235 if (itn != qeff.end() && (itn->first - it->first != 0)) {
0236 prob = (it->second * (itn->first - ev) + itn->second * (ev - it->first)) /
0237 (itn->first - it->first);
0238 }
0239
0240
0241 return rand <= prob;
0242 }
0243
0244
0245
0246 void PhotoMultiplierHitDigi::InsertHit(
0247 std::unordered_map<CellIDType, std::vector<HitData>>& hit_groups, CellIDType id, double amp,
0248 TimeType time, std::size_t sim_hit_index, std::default_random_engine& generator,
0249 std::normal_distribution<double>& gaussian,
0250 bool is_noise_hit) const
0251 {
0252 auto it = hit_groups.find(id);
0253 if (it != hit_groups.end()) {
0254 std::size_t i = 0;
0255 for (auto ghit = it->second.begin(); ghit != it->second.end(); ++ghit, ++i) {
0256 if (std::abs(time - ghit->time) <= (m_cfg.hitTimeWindow)) {
0257
0258 ghit->npe += 1;
0259 ghit->signal += amp;
0260 if (!is_noise_hit) {
0261 ghit->sim_hit_indices.push_back(sim_hit_index);
0262 }
0263 trace(" -> add to group @ {:#018X}: signal={}", id, ghit->signal);
0264 break;
0265 }
0266 }
0267
0268 if (i >= it->second.size()) {
0269 auto sig = amp + m_cfg.pedMean + m_cfg.pedError * gaussian(generator);
0270 decltype(HitData::sim_hit_indices) indices;
0271 if (!is_noise_hit) {
0272 indices.push_back(sim_hit_index);
0273 }
0274 hit_groups.insert(
0275 {id, {HitData{.npe = 1, .signal = sig, .time = time, .sim_hit_indices = indices}}});
0276 trace(" -> no group found,");
0277 trace(" so new group @ {:#018X}: signal={}", id, sig);
0278 }
0279 } else {
0280 auto sig = amp + m_cfg.pedMean + m_cfg.pedError * gaussian(generator);
0281 decltype(HitData::sim_hit_indices) indices;
0282 if (!is_noise_hit) {
0283 indices.push_back(sim_hit_index);
0284 }
0285 hit_groups.insert(
0286 {id, {HitData{.npe = 1, .signal = sig, .time = time, .sim_hit_indices = indices}}});
0287 trace(" -> new group @ {:#018X}: signal={}", id, sig);
0288 }
0289 }
0290
0291 }