File indexing completed on 2025-04-04 08:02:27
0001
0002
0003
0004 #include <DD4hep/Detector.h>
0005 #include <DD4hep/IDDescriptor.h>
0006 #include <DD4hep/Readout.h>
0007 #include <Evaluator/DD4hepUnits.h>
0008 #include <algorithms/geo.h>
0009 #include <catch2/catch_test_macros.hpp> // for AssertionHandler, operator""_catch_sr, StringRef, REQUIRE, operator<, operator==, operator>, TEST_CASE
0010 #include <edm4hep/RawTimeSeriesCollection.h>
0011 #include <edm4hep/SimTrackerHitCollection.h>
0012 #include <fmt/core.h>
0013 #include <podio/RelationRange.h>
0014 #include <spdlog/common.h> // for level_enum
0015 #include <spdlog/logger.h> // for logger
0016 #include <spdlog/spdlog.h> // for default_logger
0017 #include <algorithm>
0018 #include <cmath>
0019 #include <gsl/pointers>
0020 #include <limits>
0021 #include <memory> // for allocator, unique_ptr, make_unique, shared_ptr, __shared_ptr_access
0022 #include <tuple>
0023 #include <utility>
0024 #include <vector>
0025
0026 #include "TF1.h"
0027 #include "TGraphErrors.h"
0028 #include "algorithms/digi/LGADPulseGeneration.h"
0029 #include "algorithms/digi/LGADPulseGenerationConfig.h"
0030
0031 TEST_CASE("the LGAD charge sharing algorithm runs", "[LGADPulseGeneration]") {
0032 const float EPSILON = 1e-5;
0033 eicrecon::LGADPulseGenerationConfig cfg;
0034 cfg.gain = 113;
0035 cfg.Vm = 1e-4 * dd4hep::GeV;
0036 cfg.ignore_thres = 1e-4 / 5;
0037 cfg.sigma_analog = 0.293951 * dd4hep::ns;
0038 cfg.adc_bit = 8;
0039 cfg.adc_range = pow(2, cfg.adc_bit);
0040
0041 eicrecon::LGADPulseGeneration algo("LGADPulseGeneration");
0042
0043 std::shared_ptr<spdlog::logger> logger = spdlog::default_logger()->clone("LGADPulseGeneration");
0044 logger->set_level(spdlog::level::trace);
0045
0046
0047 auto detector = algorithms::GeoSvc::instance().detector();
0048 auto id_desc = detector->readout("MockLGADHits").idSpec();
0049
0050 SECTION("Pulse height linearlity test") {
0051
0052 algo.applyConfig(cfg);
0053 algo.init();
0054
0055 TGraphErrors graph;
0056 for (double edep = 0; edep <= 1e-4; edep += 1e-4 / 9) {
0057 edm4hep::SimTrackerHitCollection rawhits_coll;
0058 auto time_series_coll = std::make_unique<edm4hep::RawTimeSeriesCollection>();
0059
0060 auto hit = rawhits_coll.create();
0061 auto cellID =
0062 id_desc.encode({{"system", 0}, {"module", 0}, {"sensor", 1}, {"x", 1}, {"y", 1}});
0063
0064 hit.setCellID(cellID);
0065 hit.setEDep(
0066 edep);
0067 hit.setTime(1.5 * dd4hep::ns);
0068
0069
0070 auto input = std::make_tuple(&rawhits_coll);
0071 auto output = std::make_tuple(time_series_coll.get());
0072
0073 algo.process(input, output);
0074
0075 if (edep < 1e-4 / 5)
0076 REQUIRE(time_series_coll->size() == 0);
0077 else {
0078 REQUIRE(time_series_coll->size() == 1);
0079 auto min_adc = std::numeric_limits<int>::max();
0080 for(const auto& pulse : (*time_series_coll)) {
0081 REQUIRE(pulse.getCellID() == cellID);
0082 auto adcs = pulse.getAdcCounts();
0083 for (const auto adc : adcs)
0084 min_adc = std::min(min_adc, adc);
0085 }
0086 int npt = graph.GetN();
0087 graph.SetPoint(npt, edep, min_adc);
0088 graph.SetPointError(npt, 0, 0.5);
0089
0090 if (edep == 1e-4)
0091 REQUIRE(min_adc == -cfg.adc_range + 1);
0092 }
0093 }
0094
0095
0096 TF1 tf1("tf1", "pol1", 0, 1e-4);
0097 graph.Fit(&tf1, "R0");
0098
0099 REQUIRE(!(tf1.GetParameter(1) - tf1.GetParError(1) < 0 && 0 < tf1.GetParameter(1) + tf1.GetParError(1) ));
0100 double chi2_dof = tf1.GetChisquare() / tf1.GetNDF();
0101 logger->info("Chi-square/dof value for Edep vs min-adc = {}", chi2_dof);
0102 REQUIRE(chi2_dof < 2);
0103 }
0104
0105 SECTION("Pulse timing linearlity test") {
0106
0107 algo.applyConfig(cfg);
0108 algo.init();
0109
0110 TGraphErrors graph;
0111 std::vector<double> times;
0112
0113
0114 for (double time = 0; time < 12; time += 1.) times.push_back(time);
0115
0116 for (double time = 10; time < 101; time += 25.) times.push_back(time);
0117
0118 times.push_back(-10);
0119
0120 for (double time : times) {
0121 edm4hep::SimTrackerHitCollection rawhits_coll;
0122 auto time_series_coll = std::make_unique<edm4hep::RawTimeSeriesCollection>();
0123
0124 auto hit = rawhits_coll.create();
0125 auto cellID =
0126 id_desc.encode({{"system", 0}, {"module", 0}, {"sensor", 1}, {"x", 1}, {"y", 1}});
0127
0128 hit.setCellID(cellID);
0129 hit.setEDep(
0130 0.5e-4 * dd4hep::GeV);
0131 hit.setTime(time);
0132
0133
0134 auto input = std::make_tuple(&rawhits_coll);
0135 auto output = std::make_tuple(time_series_coll.get());
0136
0137 algo.process(input, output);
0138
0139 REQUIRE(time_series_coll->size() == 1);
0140 auto min_adc = std::numeric_limits<int>::max();
0141 int time_bin = 0;
0142 for(const auto& pulse: *time_series_coll) {
0143
0144 REQUIRE(pulse.getCellID() == cellID);
0145
0146 auto adcs = pulse.getAdcCounts();
0147 for (unsigned int i = 0; i < adcs.size(); ++i) {
0148 auto adc = adcs[i];
0149 if (adc < min_adc)
0150 time_bin = i + pulse.getTime()/cfg.tMax*cfg.tdc_range;
0151 min_adc = std::min(min_adc, adc);
0152 }
0153 }
0154 int npt = graph.GetN();
0155 graph.SetPoint(npt, time, time_bin);
0156 graph.SetPointError(npt, 0, 0.5);
0157 }
0158
0159
0160 TF1 tf1("tf1", "pol1", *std::min_element(times.begin(), times.end()), *std::max_element(times.begin(), times.end()));
0161 graph.Fit(&tf1, "R0");
0162
0163 REQUIRE(!(tf1.GetParameter(1) - tf1.GetParError(1) < 0 && 0 < tf1.GetParameter(1) + tf1.GetParError(1) ));
0164 double chi2_dof = tf1.GetChisquare() / tf1.GetNDF();
0165 logger->info("Chi-square/dof value for time vs TDC-bin = {}", chi2_dof);
0166 REQUIRE(chi2_dof < 2);
0167 }
0168 }