LDMX Software
ecal::EcalDigiProducer Class Reference

Performs basic ECal digitization. More...

#include <EcalDigiProducer.h>

Public Member Functions

 EcalDigiProducer (const std::string &name, framework::Process &process)
 Constructor.
 
virtual ~EcalDigiProducer ()=default
 Destructor.
 
virtual void configure (framework::config::Parameters &) override
 Configure this producer from the python configuration.
 
virtual void produce (framework::Event &event) override
 Simulates measurement of pulse and creates digi collection for input event.
 
virtual void onNewRun (const ldmx::RunHeader &runHeader) override
 Set up random number / noise generation.
 
- Public Member Functions inherited from framework::Producer
 Producer (const std::string &name, Process &process)
 Class constructor.
 
virtual void process (Event &event) final
 Processing an event for a Producer is calling produce.
 
- Public Member Functions inherited from framework::EventProcessor
 DECLARE_FACTORY (EventProcessor, EventProcessor *, const std::string &, Process &)
 declare that we have a factory for this class
 
 EventProcessor (const std::string &name, Process &process)
 Class constructor.
 
virtual ~EventProcessor ()=default
 Class destructor.
 
virtual void beforeNewRun (ldmx::RunHeader &run_header)
 Callback for Producers to add parameters to the run header before conditions are initialized.
 
virtual void onFileOpen (EventFile &event_file)
 Callback for the EventProcessor to take any necessary action when a new event input ROOT file is opened.
 
virtual void onFileClose (EventFile &event_file)
 Callback for the EventProcessor to take any necessary action when a event input ROOT file is closed.
 
virtual void onProcessStart ()
 Callback for the EventProcessor to take any necessary action when the processing of events starts, such as creating histograms.
 
virtual void onProcessEnd ()
 Callback for the EventProcessor to take any necessary action when the processing of events finishes, such as calculating job-summary quantities.
 
template<class T >
const T & getCondition (const std::string &condition_name)
 Access a conditions object for the current event.
 
TDirectory * getHistoDirectory ()
 Access/create a directory in the histogram file for this event processor to create histograms and analysis tuples.
 
void setStorageHint (framework::StorageControl::Hint hint)
 Mark the current event as having the given storage control hint from this module_.
 
void setStorageHint (framework::StorageControl::Hint hint, const std::string &purposeString)
 Mark the current event as having the given storage control hint from this module and the given purpose string.
 
int getLogFrequency () const
 Get the current logging frequency from the process.
 
int getRunNumber () const
 Get the run number from the process.
 
std::string getName () const
 Get the processor name.
 
void createHistograms (const std::vector< framework::config::Parameters > &histos)
 Internal function which is used to create histograms passed from the python configuration @parma histos vector of Parameters that configure histograms to create.
 

Private Attributes

std::string input_coll_name_
 input hit collection name
 
std::string input_pass_name_
 input pass name
 
std::string digi_coll_name_
 output hit collection name
 
double clock_cycle_
 Time interval for chip clock in ns.
 
int n_adcs_
 Depth of ADC buffer.
 
int i_soi_
 Index for the Sample Of Interest in the list of digi samples.
 
double mev_
 Conversion from energy in MeV to voltage in mV.
 
bool zero_suppression_
 When emulating noise in empty channels, do we zero suppress?
 
bool noise_ {true}
 Put noise into empty channels, not configurable, only helpful in development.
 
std::unique_ptr< ldmx::HgcrocEmulatorhgcroc_
 Hgcroc Emulator to digitize analog voltage signals.
 
double ns_
 Conversion from time in ns to ticks of the internal clock.
 
double readout_threshold_
 Read out threshold.
 
double pedestal_
 Read out pedestal.
 
double noise_rms_
 Noise RMS.
 
std::unique_ptr< ldmx::NoiseGeneratornoise_generator_
 Generates noise hits based off of number of cells that are not hit.
 
std::mt19937 rng_
 Generates random numbers for which channels to fill up with noise.
 

Additional Inherited Members

- Protected Member Functions inherited from framework::EventProcessor
void abortEvent ()
 Abort the event immediately.
 
- Protected Attributes inherited from framework::EventProcessor
HistogramPool histograms_
 helper object for making and filling histograms
 
NtupleManagerntuple_ {NtupleManager::getInstance()}
 Manager for any ntuples.
 
logging::logger the_log_
 The logger for this EventProcessor.
 

Detailed Description

Performs basic ECal digitization.

Definition at line 38 of file EcalDigiProducer.h.

Constructor & Destructor Documentation

◆ EcalDigiProducer()

ecal::EcalDigiProducer::EcalDigiProducer ( const std::string & name,
framework::Process & process )
inline

Constructor.

Definition at line 43 of file EcalDigiProducer.h.

44 : Producer(name, process) {}
Producer(const std::string &name, Process &process)
Class constructor.
virtual void process(Event &event) final
Processing an event for a Producer is calling produce.

Member Function Documentation

◆ configure()

void ecal::EcalDigiProducer::configure ( framework::config::Parameters & ps)
overridevirtual

Configure this producer from the python configuration.

Sets event constants and configures the noise generator, noise injector, and pulse function. Creates digi collection

Reimplemented from framework::EventProcessor.

Definition at line 16 of file EcalDigiProducer.cxx.

16 {
17 // settings of readout chip
18 // used in actual digitization
19 auto hgcroc_params = ps.get<framework::config::Parameters>("hgcroc");
20 hgcroc_ = std::make_unique<ldmx::HgcrocEmulator>(hgcroc_params);
21 clock_cycle_ = hgcroc_params.get<double>("clockCycle");
22 n_adcs_ = hgcroc_params.get<int>("nADCs");
23 i_soi_ = hgcroc_params.get<int>("iSOI");
24 noise_ = hgcroc_params.get<bool>("noise");
25
26 // collection names
27 input_coll_name_ = ps.get<std::string>("inputCollName");
28 input_pass_name_ = ps.get<std::string>("inputPassName");
29 digi_coll_name_ = ps.get<std::string>("digiCollName");
30
31 zero_suppression_ = ps.get<bool>("zero_suppression");
32
33 // physical constants
34 // used to calculate unit conversions
35 mev_ = ps.get<double>("MeV");
36
37 // Time -> clock counts conversion
38 // time [ns] * ( 2^10 / max time in ns ) = clock counts
39 ns_ = 1024. / clock_cycle_;
40
41 readout_threshold_ = ps.get<double>("avgReadoutThreshold");
42 pedestal_ = ps.get<double>("avgPedestal");
43 noise_rms_ = ps.get<double>("avgNoiseRMS");
44}
std::string input_pass_name_
input pass name
double noise_rms_
Noise RMS.
bool noise_
Put noise into empty channels, not configurable, only helpful in development.
double readout_threshold_
Read out threshold.
double pedestal_
Read out pedestal.
double mev_
Conversion from energy in MeV to voltage in mV.
int i_soi_
Index for the Sample Of Interest in the list of digi samples.
double clock_cycle_
Time interval for chip clock in ns.
double ns_
Conversion from time in ns to ticks of the internal clock.
std::string digi_coll_name_
output hit collection name
int n_adcs_
Depth of ADC buffer.
std::string input_coll_name_
input hit collection name
std::unique_ptr< ldmx::HgcrocEmulator > hgcroc_
Hgcroc Emulator to digitize analog voltage signals.
bool zero_suppression_
When emulating noise in empty channels, do we zero suppress?
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

References clock_cycle_, digi_coll_name_, framework::config::Parameters::get(), hgcroc_, i_soi_, input_coll_name_, input_pass_name_, mev_, n_adcs_, noise_, noise_rms_, ns_, pedestal_, readout_threshold_, and zero_suppression_.

◆ onNewRun()

void ecal::EcalDigiProducer::onNewRun ( const ldmx::RunHeader & runHeader)
overridevirtual

Set up random number / noise generation.

Reimplemented from framework::EventProcessor.

Definition at line 46 of file EcalDigiProducer.cxx.

46 {
47 // noise generator by default uses a Gausian model for noise
48 // i.e. It assumes the noise is distributed around a mean (setPedestal)
49 // with a certain RMS (setNoise) and then calculates
50 // how many hits_ should be generated for a given number of empty
51 // channels and a minimum readout value (setNoiseThreshold)
52 noise_generator_ = std::make_unique<ldmx::NoiseGenerator>();
53 // Configure generator that will produce noise hits_ in empty channels
54 // rms noise in mV
56 // mean noise amplitude (if using Gaussian Model for the noise) in mV
57 noise_generator_->setPedestal(pedestal_);
58 // threshold for readout in mV
59 noise_generator_->setNoiseThreshold(readout_threshold_);
60 // Set up seeds
63 noise_generator_->seedGenerator(
64 rseed.getSeed("EcalDigiProducer::NoiseGenerator"));
65 // Random number generator for layer_ / module_ / cell
66 rng_.seed(rseed.getSeed("EcalDigiProducer"));
67 // Setting up the read-out chip
68 hgcroc_->seedGenerator(rseed.getSeed("EcalDigiProducer::HgcrocEmulator"));
69 hgcroc_->condition(
71}
std::unique_ptr< ldmx::NoiseGenerator > noise_generator_
Generates noise hits based off of number of cells that are not hit.
std::mt19937 rng_
Generates random numbers for which channels to fill up with noise.
const T & getCondition(const std::string &condition_name)
Access a conditions object for the current event.
static const std::string CONDITIONS_OBJECT_NAME
Conditions object name.

References framework::RandomNumberSeedService::CONDITIONS_OBJECT_NAME, framework::EventProcessor::getCondition(), hgcroc_, noise_generator_, noise_rms_, pedestal_, readout_threshold_, and rng_.

◆ produce()

void ecal::EcalDigiProducer::produce ( framework::Event & event)
overridevirtual

Simulates measurement of pulse and creates digi collection for input event.

HACK ALERT The shifting of the time should not be done this sloppily. In reality, each chip has a set time phase that it samples at (relative to target), so the time shifting should be at the emulator level.

Implements framework::Producer.

Definition at line 73 of file EcalDigiProducer.cxx.

73 {
74 // Empty collection to be filled
76 ecal_digis.setNumSamplesPerDigi(n_adcs_);
78
79 // detector IDs that already have a hit in them
80 std::set<unsigned int> filled_det_i_ds;
81
82 /******************************************************************************************
83 * HGCROC Emulation on Simulated Hits
84 *****************************************************************************************/
85 // std::cout << "Sim Hits" << std::endl;
86 // get simulated ecal hits_ from Geant4
87 // the class EcalHitIO in the SimApplication module_ handles the translation
88 // from G4CalorimeterHits to SimCalorimeterHits this class ensures that only
89 // one SimCalorimeterHit is generated per cell, but multiple "contributions"
90 // are still handled within SimCalorimeterHit
91 auto ecal_sim_hits{event.getCollection<ldmx::SimCalorimeterHit>(
93
94 /* debug printout
95 std::cout << "Energy to Voltage Conversion: " << mev_ << " mV/MeV" <<
96 std::endl;
97 */
98
99 for (auto const& sim_hit : ecal_sim_hits) {
100 std::vector<std::pair<double, double>> pulses_at_chip;
101 for (int i_contrib = 0; i_contrib < sim_hit.getNumberOfContribs();
102 i_contrib++) {
103 /* debug printout
104 std::cout << simHit.getContrib(iContrib).edep << " MeV" << std::endl;
105 */
112 pulses_at_chip.emplace_back(
113 sim_hit.getContrib(i_contrib).edep_ * mev_,
114 sim_hit.getContrib(i_contrib).time_ // global time (t=0ns at target)
115 - sim_hit.getPosition().at(2) /
116 299.702547 // shift light-speed particle traveling along z_
117 );
118 }
119
120 unsigned int hit_id = sim_hit.getID();
121 filled_det_i_ds.insert(hit_id);
122
123 ldmx_log(debug) << " Emulation of hitID = " << hit_id
124 << " with energy = " << sim_hit.getEdep()
125 << " MeV at time = "
126 << sim_hit.getTime() -
127 sim_hit.getPosition().at(2) / 299.702547;
128
129 // container emulator uses to write out samples and
130 // transfer samples into the digi collection
131 std::vector<ldmx::HgcrocDigiCollection::Sample> digi_to_add;
132 if (hgcroc_->digitize(hit_id, pulses_at_chip, digi_to_add)) {
133 ldmx_log(debug) << " --> The HGCROC will read-out this hit!";
134 ecal_digis.addDigi(hit_id, digi_to_add);
135 }
136 }
137
138 /******************************************************************************************
139 * Noise Simulation on Empty Channels
140 *****************************************************************************************/
141 if (noise_) {
142 // put noise into some empty channels
143
144 // geometry constants
145 // These are used in the noise generation so that we can randomly
146 // distribute the noise uniformly throughout the ECal channels.
147 const auto& geom = getCondition<ldmx::EcalGeometry>(
148 ldmx::EcalGeometry::CONDITIONS_OBJECT_NAME);
149 int n_ecal_layers = geom.getNumLayers();
150 int n_modules_per_layer = geom.getNumModulesPerLayer();
151 int n_cells_per_module = geom.getNumCellsPerModule();
152 int num_empty_channels =
153 n_ecal_layers * n_modules_per_layer * n_cells_per_module -
154 ecal_digis.getNumDigis();
155
156 // Uniform distributions for integer generation
157 std::uniform_int_distribution<int> layer_dist(0, n_ecal_layers - 1);
158 std::uniform_int_distribution<int> module_dist(0, n_modules_per_layer - 1);
159 std::uniform_int_distribution<int> cell_dist(0, n_cells_per_module - 1);
160
161 if (zero_suppression_) {
162 // noise generator gives us a list of noise amplitudes [mV] that randomly
163 // populate the empty channels and are above the readout threshold
164 auto noise_hit_amplitudes{
165 noise_generator_->generateNoiseHits(num_empty_channels)};
166 std::vector<std::pair<double, double>> fake_pulse(1, {0., 0.});
167 for (double noise_hit : noise_hit_amplitudes) {
168 // generate detector ID for noise hit
169 // making sure that it is in an empty channel
170 unsigned int noise_id;
171 do {
172 int layer_id = layer_dist(rng_);
173 int module_id = module_dist(rng_);
174 int cell_id = cell_dist(rng_);
175 auto det_id = ldmx::EcalID(layer_id, module_id, cell_id);
176 noise_id = det_id.raw();
177 } while (filled_det_i_ds.find(noise_id) != filled_det_i_ds.end());
178 filled_det_i_ds.insert(noise_id);
179
180 // noise generator gives the amplitude above the readout threshold
181 // we need to convert it to the amplitude above the pedestal
182 noise_hit +=
183 hgcroc_->gain(noise_id) *
184 (hgcroc_->readoutThreshold(noise_id) - hgcroc_->pedestal(noise_id));
185
186 // create a digi as put it into the collection
187 ecal_digis.addDigi(noise_id, hgcroc_->noiseDigi(noise_id, noise_hit));
188 } // loop over noise amplitudes
189 } else {
190 // no zero suppression, put some noise emulation in **all** empty channels
191 // loop through all channels
192 for (int layer{0}; layer < n_ecal_layers; layer++) {
193 for (int module{0}; module < n_modules_per_layer; module++) {
194 for (int cell{0}; cell < n_cells_per_module; cell++) {
195 unsigned int channel{ldmx::EcalID(layer, module, cell).raw()};
196 // check if channel already has a (real) hit in it
197 if (filled_det_i_ds.find(channel) != filled_det_i_ds.end())
198 continue;
199 // create a digi as put it into the collection
200 ecal_digis.addDigi(channel, hgcroc_->noiseDigi(channel));
201 } // cells in each module_
202 } // modules in each layer_
203 } // layers in ECal
204 } // yes or no zero suppression
205 } // if we should do the noise
206
207 event.add(digi_coll_name_, ecal_digis);
208
209 return;
210} // produce
RawValue raw() const
Definition DetectorID.h:68
Extension of DetectorID providing access to ECal layers and cell numbers in a hex grid.
Definition EcalID.h:20
Represents a collection of the digi hits readout by an HGCROC.
void setNumSamplesPerDigi(unsigned int n)
Set number of samples for each digi.
void setSampleOfInterestIndex(unsigned int n)
Set index of sample of interest.
void addDigi(unsigned int id, const std::vector< Sample > &digi)
Add samples to collection.
unsigned int getNumDigis() const
Get total number of digis.
Stores simulated calorimeter hit information.

References ldmx::HgcrocDigiCollection::addDigi(), framework::EventProcessor::getCondition(), ldmx::HgcrocDigiCollection::getNumDigis(), hgcroc_, i_soi_, input_coll_name_, input_pass_name_, mev_, n_adcs_, noise_, noise_generator_, ldmx::DetectorID::raw(), rng_, ldmx::HgcrocDigiCollection::setNumSamplesPerDigi(), ldmx::HgcrocDigiCollection::setSampleOfInterestIndex(), and zero_suppression_.

Member Data Documentation

◆ clock_cycle_

double ecal::EcalDigiProducer::clock_cycle_
private

Time interval for chip clock in ns.

Definition at line 82 of file EcalDigiProducer.h.

Referenced by configure().

◆ digi_coll_name_

std::string ecal::EcalDigiProducer::digi_coll_name_
private

output hit collection name

Definition at line 79 of file EcalDigiProducer.h.

Referenced by configure().

◆ hgcroc_

std::unique_ptr<ldmx::HgcrocEmulator> ecal::EcalDigiProducer::hgcroc_
private

Hgcroc Emulator to digitize analog voltage signals.

Definition at line 115 of file EcalDigiProducer.h.

Referenced by configure(), onNewRun(), and produce().

◆ i_soi_

int ecal::EcalDigiProducer::i_soi_
private

Index for the Sample Of Interest in the list of digi samples.

Definition at line 88 of file EcalDigiProducer.h.

Referenced by configure(), and produce().

◆ input_coll_name_

std::string ecal::EcalDigiProducer::input_coll_name_
private

input hit collection name

Definition at line 73 of file EcalDigiProducer.h.

Referenced by configure(), and produce().

◆ input_pass_name_

std::string ecal::EcalDigiProducer::input_pass_name_
private

input pass name

Definition at line 76 of file EcalDigiProducer.h.

Referenced by configure(), and produce().

◆ mev_

double ecal::EcalDigiProducer::mev_
private

Conversion from energy in MeV to voltage in mV.

Definition at line 91 of file EcalDigiProducer.h.

Referenced by configure(), and produce().

◆ n_adcs_

int ecal::EcalDigiProducer::n_adcs_
private

Depth of ADC buffer.

Definition at line 85 of file EcalDigiProducer.h.

Referenced by configure(), and produce().

◆ noise_

bool ecal::EcalDigiProducer::noise_ {true}
private

Put noise into empty channels, not configurable, only helpful in development.

Definition at line 112 of file EcalDigiProducer.h.

112{true};

Referenced by configure(), and produce().

◆ noise_generator_

std::unique_ptr<ldmx::NoiseGenerator> ecal::EcalDigiProducer::noise_generator_
private

Generates noise hits based off of number of cells that are not hit.

Definition at line 128 of file EcalDigiProducer.h.

Referenced by onNewRun(), and produce().

◆ noise_rms_

double ecal::EcalDigiProducer::noise_rms_
private

Noise RMS.

Definition at line 125 of file EcalDigiProducer.h.

Referenced by configure(), and onNewRun().

◆ ns_

double ecal::EcalDigiProducer::ns_
private

Conversion from time in ns to ticks of the internal clock.

Definition at line 118 of file EcalDigiProducer.h.

Referenced by configure().

◆ pedestal_

double ecal::EcalDigiProducer::pedestal_
private

Read out pedestal.

Definition at line 123 of file EcalDigiProducer.h.

Referenced by configure(), and onNewRun().

◆ readout_threshold_

double ecal::EcalDigiProducer::readout_threshold_
private

Read out threshold.

Definition at line 121 of file EcalDigiProducer.h.

Referenced by configure(), and onNewRun().

◆ rng_

std::mt19937 ecal::EcalDigiProducer::rng_
private

Generates random numbers for which channels to fill up with noise.

Definition at line 131 of file EcalDigiProducer.h.

Referenced by onNewRun(), and produce().

◆ zero_suppression_

bool ecal::EcalDigiProducer::zero_suppression_
private

When emulating noise in empty channels, do we zero suppress?

There are two ways to emulate the noise in the chip: 1) (with zero suppression) Use the NoiseGenerator to get a list of amplitudes (about 3-5 on avg) and put those noise hits which are above the readout threshold in random empty channels. 2) (without zero suppresion) Go through all channels and put noise in each channel that doesn't have a real hit in it. This will mean many channels with have a DIGI below readout threshold.

Definition at line 105 of file EcalDigiProducer.h.

Referenced by configure(), and produce().


The documentation for this class was generated from the following files: