From 0922c11483a0d5a6607005fa77ca8cfd1823ff22 Mon Sep 17 00:00:00 2001 From: Eric Torrence <eric.torrence@cern.ch> Date: Tue, 5 Dec 2023 05:34:48 +0100 Subject: [PATCH] Add hit timing to waveform dititization --- .../src/ScintWaveformDigiAlg.cxx | 213 +++++++++++++----- .../ScintDigiAlgs/src/ScintWaveformDigiAlg.h | 8 +- .../WaveDigiTools/IWaveformDigitisationTool.h | 9 + .../IWaveformDigitisationTool.icc | 2 + .../src/WaveformDigitisationTool.cxx | 27 +++ .../src/WaveformDigitisationTool.h | 7 +- .../IWaveformDigiConditionsTool.h | 5 + .../src/WaveformDigiConditionsTool.cxx | 11 + .../src/WaveformDigiConditionsTool.h | 3 + 9 files changed, 229 insertions(+), 56 deletions(-) diff --git a/Scintillator/ScintDigiAlgs/src/ScintWaveformDigiAlg.cxx b/Scintillator/ScintDigiAlgs/src/ScintWaveformDigiAlg.cxx index 1869860e8..33bd45d7d 100644 --- a/Scintillator/ScintDigiAlgs/src/ScintWaveformDigiAlg.cxx +++ b/Scintillator/ScintDigiAlgs/src/ScintWaveformDigiAlg.cxx @@ -95,11 +95,19 @@ ScintWaveformDigiAlg::execute(const EventContext& ctx) const { // Also save the baseline parameters m_base_mean = m_digiCondTool->base_mean(ctx); m_base_rms = m_digiCondTool->base_rms(ctx); + + // And print out our timing simulation + if (m_advancedTiming) { + ATH_MSG_INFO("Using improved digitization timing"); + } else { + ATH_MSG_INFO("Using simple digitization timing"); + } } // Create structure to store pulse for each channel std::map<Identifier, std::vector<uint16_t>> waveforms; + // This gives us empty vectors auto first = *scintHitHandle->begin(); if (first.isVeto()) { waveforms = m_digiTool->create_waveform_map(m_vetoID); @@ -109,79 +117,168 @@ ScintWaveformDigiAlg::execute(const EventContext& ctx) const { waveforms = m_digiTool->create_waveform_map(m_triggerID); } else if (first.isPreshower()) { waveforms = m_digiTool->create_waveform_map(m_preshowerID); - } + } + + // Identifiers we have seen hits from + std::set<Identifier> id_set; // Sum energy for each channel (i.e. identifier) std::map<Identifier, float> esum; + // Histograms of E vs. t, make sure this is empty + m_digiTool->clear_evst_hist(m_evst_hist); + Identifier id; + + // Loop over all hits and make histograms of E vs. t for each channel for (const auto& hit : *scintHitHandle) { + + float dtime; // Time offset due to scint. geometry + + // Trigger plane, horizontal readout if (first.isTrigger()) { Identifier plate_id = m_triggerID->plate_id(hit.getIdentifier()); + + // Time delay based on horizontal distance + // Don't know if I have the sign right here + // Assume +x is further from PMT 0 + dtime = m_digiCondTool->time_slope(ctx) * (hit.localStartPosition().x()+hit.localEndPosition().x()) / 2.; + // Need to do something better here, but for now just fill both id = m_triggerID->pmt_id(plate_id, 0); // ID for PMT 0 + id_set.insert(id); esum[id] += hit.energyLoss(); + m_digiTool->fill_evst_hist(m_evst_hist, id, hit.meanTime()+dtime, hit.energyLoss()); + id_set.insert(id); + id = m_triggerID->pmt_id(plate_id, 1); // ID for PMT 1 + id_set.insert(id); esum[id] += hit.energyLoss(); + m_digiTool->fill_evst_hist(m_evst_hist, id, hit.meanTime()-dtime, hit.energyLoss()); + } else { // All others have only 1 PMT // Use detector id (not hit id) throughout id = hit.getIdentifier(); + id_set.insert(id); esum[id] += hit.energyLoss(); - } - } - // Loop over time samples - for (const auto& tk : m_timekernel) { - std::map<Identifier, float> counts; - -// // Convolve hit energy with evaluated kernel and sum for each hit id (i.e. channel) -// Identifier id; -// for (const auto& hit : *scintHitHandle) { -// -// // Special handling for trigger scintillator -// if (first.isTrigger()) { -// // Hits are created per plate -// // Need to split between 2 PMTs -// -// // Identifier from hit is plate ID -// Identifier plate_id = m_triggerID->plate_id(hit.getIdentifier()); -// -// // Need to do something better here, but for now just fill both -// id = m_triggerID->pmt_id(plate_id, 0); // ID for PMT 0 -// counts[id] += tk * hit.energyLoss(); -// id = m_triggerID->pmt_id(plate_id, 1); // ID for PMT 1 -// counts[id] += tk * hit.energyLoss(); -// -// } else { -// // All others have only 1 PMT -// // Use detector id (not hit id) throughout -// id = hit.getIdentifier(); -// counts[id] += tk * hit.energyLoss(); -// } -// } - - // Convolve summed energy with evaluated kernel for each hit id (i.e. channel) - for (const auto& e : esum) { - // Convert hit id to Identifier and store - id = e.first; - counts[id] = tk * e.second; + // +y means shorter times + dtime = -m_digiCondTool->time_slope(ctx) * (hit.localStartPosition().y() + hit.localEndPosition().y()) / 2.; + m_digiTool->fill_evst_hist(m_evst_hist, id, hit.meanTime()+dtime, hit.energyLoss()); } - // Subtract count from basleine and add result to correct waveform vector - for (const auto& c : counts) { + // Print out hit times + ATH_MSG_DEBUG("Scint Hit: " << id << " E: " << hit.energyLoss() + << " t: " << hit.meanTime() + << " x: " << hit.localStartPosition().x() + << " y: " << hit.localStartPosition().y() + << " dt: " << dtime + ); - float baseline = m_digiTool->generate_baseline(m_base_mean, m_base_rms); - int value = std::round(baseline - c.second); + } // Done with loop over hits + + // Choose how we construct the waveform + if (m_advancedTiming) { + + // Loop over seen identifiers + for (const auto& id : id_set) { + + // Convolve histogram with time kernel + + // Get the histogram + auto it = m_evst_hist.find(id); + if (it == m_evst_hist.end()) { + ATH_MSG_WARNING("Didn't find EvsT hist for id " << id <<"!"); + continue; + } + auto h = it->second; + + // Use this vector to sum our values + std::vector<double> dwave; + + // Loop over bins and fill with baseline values + for(unsigned int ibin=1; ibin <= h->GetNbinsX(); ibin++) { + + // Push back baseline value so we can index this + float baseline = m_digiTool->generate_baseline(m_base_mean, m_base_rms); + // Starts empty, push back each element here + dwave.push_back(baseline); + + } // End of loop over bins + + // Now, do it again and convolute any signals with the time kernal + // Because root histos start with bin=1, subtract 1 from ibin + for(unsigned int ibin=1; ibin <= h->GetNbinsX(); ibin++) { + + float value = h->GetBinContent(ibin); + + // If bin is empty, move on + if (value <= 0.) continue; + + // Convolute the histogram value with our time kernel + unsigned int ik = 0; + for (const auto& tk : m_timekernel) { + // Don't write off the end + if ((ibin + ik - 1) >= dwave.size()) break; + + dwave[ibin+ik-1] -= (value * tk); + ik++; // Iterate + } + } // End of loop over histogram bins + + // Finally protect against underflows and 'digitize' the values + for(unsigned int i=0; i<dwave.size(); i++) { + + // waveforms is initialized empty, must push back each element here + if (dwave[i] < 0.) { + waveforms[id].push_back(0); + } else { + waveforms[id].push_back(std::round(dwave[i])); + } + + // Debug weird overflow values + // if (waveforms[id][i] > m_base_mean + 5*m_base_rms) { + // ATH_MSG_WARNING("Found waveform id " << id << " bin " << i << " value " << waveforms[id][i] << "!"); + // } - if (value < 0) { - ATH_MSG_WARNING("Found pulse " << c.second << " larger than baseline " << c.first); - value = 0; // Protect against scaling signal above baseline } - // Convert hit id to Identifier and store - id = c.first; - waveforms[id].push_back(value); + // Check what we have done + ATH_MSG_DEBUG("Waveform for ID " << id); + for(unsigned int i=400; i<450; i++) + ATH_MSG_DEBUG(2*i << "ns -> " << waveforms[id][i]); + + } // End of loop over ids + + } else { + + // Loop over time samples + for (const auto& tk : m_timekernel) { + std::map<Identifier, float> counts; + + // Convolve summed energy with evaluated kernel for each hit id (i.e. channel) + for (const auto& e : esum) { + // Convert hit id to Identifier and store + id = e.first; + counts[id] = tk * e.second; + } + + // Subtract count from basleine and add result to correct waveform vector + for (const auto& c : counts) { + + float baseline = m_digiTool->generate_baseline(m_base_mean, m_base_rms); + int value = std::round(baseline - c.second); + + if (value < 0) { + ATH_MSG_WARNING("Found pulse " << c.second << " larger than baseline " << c.first); + value = 0; // Protect against scaling signal above baseline + } + + // Convert hit id to Identifier and store + id = c.first; + waveforms[id].push_back(value); + } } } @@ -196,12 +293,12 @@ ScintWaveformDigiAlg::execute(const EventContext& ctx) const { ATH_MSG_DEBUG("Writing baseline into empty waveform in channel "<< channel); int i = m_digiTool->nsamples(); while(i--) { // Use while to avoid unused variable warning with for - int baseline = m_digiTool->generate_baseline(m_base_mean, m_base_rms); - waveforms[w.first].push_back(baseline); + float baseline = m_digiTool->generate_baseline(m_base_mean, m_base_rms); + waveforms[w.first].push_back(std::round(baseline)); } } - // Loop over wavefrom vectors to make and store raw waveform + // Loop over waveform vectors to make and store raw waveform unsigned int nsamples = m_digiTool->nsamples(); for (const auto& w : waveforms) { RawWaveform* wfm = new RawWaveform(); @@ -221,5 +318,19 @@ ScintWaveformDigiAlg::execute(const EventContext& ctx) const { ATH_MSG_DEBUG("WaveformsHitContainer " << waveformContainerHandle.name() << "' filled with "<< waveformContainerHandle->size() <<" items"); + // Debugging printout + for (auto it : m_evst_hist) { + ATH_MSG_DEBUG("Nonzero bins Evst hist id = " << it.first); + for (unsigned int ibin=0; ibin <= it.second->GetNbinsX(); ibin++) { + if (it.second->GetBinContent(ibin) == 0) continue; + ATH_MSG_DEBUG(" bin: " << ibin << " val: " << it.second->GetBinContent(ibin)); + } + } + + // Cleanup any histograms we made + m_digiTool->clear_evst_hist(m_evst_hist); + return StatusCode::SUCCESS; } + + diff --git a/Scintillator/ScintDigiAlgs/src/ScintWaveformDigiAlg.h b/Scintillator/ScintDigiAlgs/src/ScintWaveformDigiAlg.h index 99c374f55..7f76328a3 100644 --- a/Scintillator/ScintDigiAlgs/src/ScintWaveformDigiAlg.h +++ b/Scintillator/ScintDigiAlgs/src/ScintWaveformDigiAlg.h @@ -33,7 +33,6 @@ // ROOT #include "TF1.h" - // STL #include <string> #include <vector> @@ -52,6 +51,9 @@ class ScintWaveformDigiAlg : public AthReentrantAlgorithm { virtual StatusCode finalize() override; //@} + // + // Simulate detailed timing of waveforms + BooleanProperty m_advancedTiming{this, "AdvancedTiming", true}; private: @@ -68,13 +70,13 @@ class ScintWaveformDigiAlg : public AthReentrantAlgorithm { mutable float m_base_mean; mutable float m_base_rms; - /** Kernel PDF and evaluated values **/ //@{ mutable TF1* m_kernel; mutable std::vector<float> m_timekernel; + mutable EvstHistMap m_evst_hist; + //mutable std::map<Identifier, TH1D*> m_evst_hist; //@} - /// Detector ID helpers const VetoID* m_vetoID{nullptr}; diff --git a/Waveform/WaveDigiTools/WaveDigiTools/IWaveformDigitisationTool.h b/Waveform/WaveDigiTools/WaveDigiTools/IWaveformDigitisationTool.h index 37cb7af18..2009b495d 100644 --- a/Waveform/WaveDigiTools/WaveDigiTools/IWaveformDigitisationTool.h +++ b/Waveform/WaveDigiTools/WaveDigiTools/IWaveformDigitisationTool.h @@ -27,11 +27,13 @@ #include "TF1.h" #include "TRandom3.h" +#include "TH1D.h" #include <utility> #include <map> #include <vector> +typedef std::map<Identifier, TH1D*> EvstHistMap; ///Interface for waveform digitisation tools class IWaveformDigitisationTool : virtual public IAlgTool @@ -60,12 +62,19 @@ public: /// Number of time samples unsigned int nsamples() const { return m_nsamples; } + /// Fill histograms + virtual void fill_evst_hist(EvstHistMap& map, const Identifier& id, float time, float energy) const = 0; + + // Remove old histograms + virtual void clear_evst_hist(EvstHistMap& map) const = 0; + private: ServiceHandle<IMessageSvc> m_msgSvc; protected: TRandom3* m_random; unsigned int m_nsamples; + }; #include "WaveDigiTools/IWaveformDigitisationTool.icc" diff --git a/Waveform/WaveDigiTools/WaveDigiTools/IWaveformDigitisationTool.icc b/Waveform/WaveDigiTools/WaveDigiTools/IWaveformDigitisationTool.icc index 41b8c2650..fd50ecfcf 100644 --- a/Waveform/WaveDigiTools/WaveDigiTools/IWaveformDigitisationTool.icc +++ b/Waveform/WaveDigiTools/WaveDigiTools/IWaveformDigitisationTool.icc @@ -15,3 +15,5 @@ std::map<Identifier, std::vector<uint16_t>> IWaveformDigitisationTool::create_w return waveforms; } + + diff --git a/Waveform/WaveDigiTools/src/WaveformDigitisationTool.cxx b/Waveform/WaveDigiTools/src/WaveformDigitisationTool.cxx index 3a48027d5..1b9ad0f44 100644 --- a/Waveform/WaveDigiTools/src/WaveformDigitisationTool.cxx +++ b/Waveform/WaveDigiTools/src/WaveformDigitisationTool.cxx @@ -45,3 +45,30 @@ float WaveformDigitisationTool::generate_baseline(float mean, float rms) const { return m_random->Gaus(mean, rms); } + + +/// Fill EvsT histograms +void +WaveformDigitisationTool::fill_evst_hist(EvstHistMap& map, const Identifier& id, float time, float energy) const { + + if (map.find(id) == map.end()) { + ATH_MSG_DEBUG("Creating EvsT hist for channel" << id); + map[id] = new TH1D("", "", m_nsamples, 0., 2.*m_nsamples); + } + + map[id]->Fill(time, energy); +} + +/// Delete histograms and clear map +void +WaveformDigitisationTool::clear_evst_hist(EvstHistMap& map) const { + + // Delete histogram objects + for (auto& it : map) + delete it.second; + + // Also remove map elements (now null pointers) + map.clear(); +} + + diff --git a/Waveform/WaveDigiTools/src/WaveformDigitisationTool.h b/Waveform/WaveDigiTools/src/WaveformDigitisationTool.h index b6554c337..ba73b3c8d 100644 --- a/Waveform/WaveDigiTools/src/WaveformDigitisationTool.h +++ b/Waveform/WaveDigiTools/src/WaveformDigitisationTool.h @@ -34,10 +34,13 @@ class WaveformDigitisationTool: public extends<AthAlgTool, IWaveformDigitisation /// Generate random baseline float generate_baseline(float mean, float rms) const; + /// Fill EvsT histograms + void fill_evst_hist(EvstHistMap& map, const Identifier& id, float time, float energy) const; - private: - // None + void clear_evst_hist(EvstHistMap& map) const; + private: + /// None }; #endif // WAVEDIGITOOLS_WAVEFORMDIGITISATIONTOOL_H diff --git a/Waveform/WaveformConditions/WaveformConditionsTools/WaveformConditionsTools/IWaveformDigiConditionsTool.h b/Waveform/WaveformConditions/WaveformConditionsTools/WaveformConditionsTools/IWaveformDigiConditionsTool.h index 1bfaf0dd9..3f531beca 100644 --- a/Waveform/WaveformConditions/WaveformConditionsTools/WaveformConditionsTools/IWaveformDigiConditionsTool.h +++ b/Waveform/WaveformConditions/WaveformConditionsTools/WaveformConditionsTools/IWaveformDigiConditionsTool.h @@ -57,6 +57,11 @@ class IWaveformDigiConditionsTool: virtual public IAlgTool { virtual float cb_n(void) const = 0; virtual float cb_n(const EventContext& ctx) const = 0; + + // Time dependence parameters + virtual float time_slope(void) const = 0; + virtual float time_slope(const EventContext& ctx) const = 0; + }; //---------------------------------------------------------------------- diff --git a/Waveform/WaveformConditions/WaveformConditionsTools/src/WaveformDigiConditionsTool.cxx b/Waveform/WaveformConditions/WaveformConditionsTools/src/WaveformDigiConditionsTool.cxx index 4e2fedd97..d16f72242 100644 --- a/Waveform/WaveformConditions/WaveformConditionsTools/src/WaveformDigiConditionsTool.cxx +++ b/Waveform/WaveformConditions/WaveformConditionsTools/src/WaveformDigiConditionsTool.cxx @@ -93,6 +93,12 @@ WaveformDigiConditionsTool::cb_n(const EventContext& ctx) const { return get_value(ctx, "cb_n"); } +float +WaveformDigiConditionsTool::time_slope(const EventContext& ctx) const { + return 1/230.; // 23 cm/ns + // return get_value(ctx, "time_slope"); +} + //---------------------------------------------------------------------- float WaveformDigiConditionsTool::base_mean(void) const { @@ -136,6 +142,11 @@ WaveformDigiConditionsTool::cb_n(void) const { return cb_n(ctx); } +float +WaveformDigiConditionsTool::time_slope(void) const { + const EventContext& ctx(Gaudi::Hive::currentContext()); + return time_slope(ctx); +} //---------------------------------------------------------------------- float diff --git a/Waveform/WaveformConditions/WaveformConditionsTools/src/WaveformDigiConditionsTool.h b/Waveform/WaveformConditions/WaveformConditionsTools/src/WaveformDigiConditionsTool.h index e5e823208..eab1a65d7 100644 --- a/Waveform/WaveformConditions/WaveformConditionsTools/src/WaveformDigiConditionsTool.h +++ b/Waveform/WaveformConditions/WaveformConditionsTools/src/WaveformDigiConditionsTool.h @@ -67,6 +67,9 @@ class WaveformDigiConditionsTool: public extends<AthAlgTool, IWaveformDigiCondit virtual float cb_n(void) const override; virtual float cb_n(const EventContext& ctx) const override; + // Time dependence parameters + virtual float time_slope(void) const override; + virtual float time_slope(const EventContext& ctx) const override; private: -- GitLab