LArSoft  v10_04_05
Liquid Argon Software toolkit - https://larsoft.org/
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Modules Pages
EvaluateROIEff_module.cc
Go to the documentation of this file.
1 // Class: EvaluateROIEff
3 // Plugin Type: analyzer (art v3_05_00)
4 // File: EvaluateROIEff_module.cc
5 //
6 // Generated at Sun May 3 23:16:14 2020 by Tingjun Yang using cetskelgen
7 // from cetlib version v3_10_00.
8 //
9 // Author: Tingjun Yang, tjyang@fnal.gov
10 // Wanwei Wu, wwu@fnal.gov
11 //
12 // About: ROI efficiency and purity evaluation using "signal". Here, "signal"
13 // is consecutive energy deposits based on tdc ticks.
15 
23 #include "larevt/CalibrationDBI/Interface/ChannelStatusProvider.h"
24 #include "larevt/CalibrationDBI/Interface/ChannelStatusService.h"
25 
31 #include "art_root_io/TFileService.h"
34 #include "fhiclcpp/ParameterSet.h"
36 
37 #include "TEfficiency.h"
38 #include "TH1D.h"
39 
40 #include <cstdlib> // std::abs()
41 #include <string>
42 #include <vector>
43 
44 namespace nnet {
45  class EvaluateROIEff;
46 }
47 
49 public:
50  explicit EvaluateROIEff(fhicl::ParameterSet const& p);
51  // The compiler-generated destructor is fine for non-base
52  // classes without bare pointers or other resource use.
53 
54  // Plugins should not be copied or assigned.
55  EvaluateROIEff(EvaluateROIEff const&) = delete;
56  EvaluateROIEff(EvaluateROIEff&&) = delete;
57  EvaluateROIEff& operator=(EvaluateROIEff const&) = delete;
59 
60  // Required functions.
61  void analyze(art::Event const& e) override;
62 
63 private:
64  void beginJob() override;
65  void endJob() override;
66  bool isSignalInROI(int starttick, int endtick, int maxtick, int roistart, int roiend);
67  // Declare member data here.
70  fSimulationProducerLabel; // The name of the producer that tracked simulated particles through the detector
71 
72  TH1D* h_energy[3];
73  TH1D* h_energy_roi[3];
74 
75  TEfficiency* eff_energy[3];
76 
77  TH1D* h_purity[3];
78  TH1D* h_purity_all;
79 
80  TH1D* h_roi[3];
81  TH1D* h1_roi_max[3];
82  TH1D* h1_roi_max_sim[3];
83 
84  TH1D* h1_tickdiff_max[3];
85 
86  int fCount_Roi_sig[3] = {0, 0, 0};
87  int fCount_Roi_total[3] = {0, 0, 0};
88 };
89 
91  : EDAnalyzer{p}
92  , fWireProducerLabel(p.get<art::InputTag>("WireProducerLabel", ""))
93  , fSimulationProducerLabel(p.get<art::InputTag>("SimulationProducerLabel", "largeant"))
94 // More initializers here.
95 {
96  // Call appropriate consumes<>() for any products to be retrieved by this module.
97 }
98 
100 {
101  auto const& wireReadoutGeom = art::ServiceHandle<geo::WireReadout const>()->Get();
102  auto const clockData = art::ServiceHandle<detinfo::DetectorClocksService>()->DataFor(e);
103  auto const detProp =
105  auto const& chStatus = art::ServiceHandle<lariov::ChannelStatusService>()->GetProvider();
106 
107  art::Handle<std::vector<recob::Wire>> wireListHandle;
108  std::vector<art::Ptr<recob::Wire>> wires;
109  if (e.getByLabel(fWireProducerLabel, wireListHandle)) {
110  art::fill_ptr_vector(wires, wireListHandle);
111  }
112 
113  auto simChannelHandle = e.getValidHandle<std::vector<sim::SimChannel>>(fSimulationProducerLabel);
114 
115  // efficiency: according to each simulated energy deposit
116  // ... Loop over simChannels
117  for (auto const& channel : *simChannelHandle) {
118 
119  // .. get simChannel channel number
120  const raw::ChannelID_t ch1 = channel.Channel();
121  if (chStatus.IsBad(ch1)) continue;
122 
123  if (ch1 % 1000 == 0) mf::LogInfo("EvaluateROIEFF") << ch1;
124  int view = wireReadoutGeom.View(ch1);
125  auto const& timeSlices = channel.TDCIDEMap();
126 
127  // time slice from simChannel is for individual tick
128  // group neighboring time slices into a "signal"
129  int tdctick_previous = -999;
130  double totalE = 0.;
131  double maxE = -999.;
132  int maxEtick = -999;
133  std::vector<int> signal_starttick;
134  std::vector<int> signal_endtick;
135  std::vector<double> signal_energydeposits;
136  std::vector<double> signal_energy_max;
137  std::vector<double> signal_max_tdctick;
138 
139  for (auto const& timeSlice : timeSlices) {
140  auto const tpctime = timeSlice.first;
141  auto const& energyDeposits = timeSlice.second;
142  int tdctick = static_cast<int>(clockData.TPCTDC2Tick(double(tpctime)));
143  if (tdctick < 0 || tdctick > int(detProp.ReadOutWindowSize()) - 1) continue;
144 
145  // for a time slice, there may exist more than one energy deposit.
146  double e_deposit = 0;
147  for (auto const& energyDeposit : energyDeposits) {
148  e_deposit += energyDeposit.energy;
149  }
150 
151  if (tdctick_previous == -999) {
152  signal_starttick.push_back(tdctick);
153  totalE += e_deposit;
154  maxE = e_deposit;
155  maxEtick = tdctick;
156  }
157  else if (tdctick - tdctick_previous != 1) {
158  signal_starttick.push_back(tdctick);
159  signal_endtick.push_back(tdctick_previous);
160  signal_energydeposits.push_back(totalE);
161  signal_energy_max.push_back(maxE);
162  signal_max_tdctick.push_back(maxEtick);
163  totalE = e_deposit;
164  maxE = e_deposit;
165  maxEtick = tdctick;
166  }
167  else if (tdctick - tdctick_previous == 1) {
168  totalE += e_deposit;
169  if (maxE < e_deposit) {
170  maxE = e_deposit;
171  maxEtick = tdctick;
172  }
173  }
174 
175  tdctick_previous = tdctick;
176 
177  } // loop over timeSlices timeSlice
178 
179  signal_endtick.push_back(tdctick_previous); // for last one
180  signal_energydeposits.push_back(totalE); // for last one
181  signal_energy_max.push_back(maxE); // for last one
182  signal_max_tdctick.push_back(maxEtick); // for last one
183 
184  if (signal_starttick.size() == 0 ||
185  (signal_endtick.size() == 1 && signal_endtick.back() == -999))
186  continue;
187 
188  for (auto& wire : wires) {
189  if (wire->Channel() != ch1) continue;
190 
191  const recob::Wire::RegionsOfInterest_t& signalROI = wire->SignalROI();
192 
193  std::vector<float> vecADC =
194  wire->Signal(); // a zero-padded full length vector filled with RoI signal.
195 
196  // loop over signals:
197  // a) if signal s is not in any ROI (including the case no ROI), fill h_energy
198  // with signal_energy_max[s];
199  // b) if signal s is in a ROI, check following signals that are also in this ROI,
200  // then use the maximum of signal_energy_max to fill h_energy and h_energy_roi.
201  // After this, the loop will skip to the signal that is not in this ROI.
202  for (size_t s = 0; s < signal_starttick.size(); s++) {
203  // case a: signal is not in any ROI
204  bool IsSignalOutside = true;
205  for (const auto& range : signalROI.get_ranges()) {
206  //const auto& waveform = range.data();
207  raw::TDCtick_t roiFirstBinTick = range.begin_index();
208  raw::TDCtick_t roiLastBinTick = range.end_index();
209 
210  if (isSignalInROI(signal_starttick[s],
211  signal_endtick[s],
212  signal_max_tdctick[s],
213  roiFirstBinTick,
214  roiLastBinTick)) {
215  IsSignalOutside = false;
216  break;
217  }
218  } // loop over range
219 
220  if (IsSignalOutside) {
221  h_energy[view]->Fill(signal_energy_max[s]);
222  h1_tickdiff_max[view]->Fill(-99);
223  continue;
224  }
225 
226  // signal is in one ROI
227  for (const auto& range : signalROI.get_ranges()) {
228  raw::TDCtick_t roiFirstBinTick = range.begin_index();
229  raw::TDCtick_t roiLastBinTick = range.end_index();
230 
231  if (isSignalInROI(signal_starttick[s],
232  signal_endtick[s],
233  signal_max_tdctick[s],
234  roiFirstBinTick,
235  roiLastBinTick)) {
236  // maximum pulse height and postion for this ROI
237  double maxadc_sig = 0;
238  int maxadc_tick = -99;
239  for (int k = roiFirstBinTick; k < roiLastBinTick; k++) {
240  if (vecADC[k] > maxadc_sig) {
241  maxadc_sig = vecADC[k];
242  maxadc_tick = k;
243  }
244  }
245 
246  double maxE_roi = -999.;
247  double maxE_roi_tick = -999.;
248 
249  if (maxE_roi < signal_energy_max[s]) {
250  maxE_roi = signal_energy_max[s];
251  maxE_roi_tick = signal_max_tdctick[s];
252  }
253 
254  // check the following signals in the same ROI
255  for (size_t s2 = s + 1; s2 < signal_starttick.size(); s2++) {
256  if (isSignalInROI(signal_starttick[s2],
257  signal_endtick[s2],
258  signal_max_tdctick[s2],
259  roiFirstBinTick,
260  roiLastBinTick)) {
261  if (maxE_roi < signal_energy_max[s2]) {
262  maxE_roi = signal_energy_max[s2];
263  maxE_roi_tick = signal_max_tdctick[s2];
264  }
265  if (s2 == signal_starttick.size() - 1) { s = s2; }
266  }
267  else {
268  s = s2 - 1;
269  break;
270  }
271  } // loop over s2
272 
273  // finish this ROI
274  h_energy[view]->Fill(maxE_roi);
275  h_energy_roi[view]->Fill(maxE_roi);
276  h1_tickdiff_max[view]->Fill(maxE_roi_tick - maxadc_tick);
277  break;
278  } // isSignalInROI
279  } // loop over range
280 
281  } // loop over signals s
282  } // loop over wires wire
283  } // loop simChannels
284 
285  // purity: # signals in ROI / (#signals in ROI + #non-signals in ROI). Because we only consider the maximum signal in the ROI, this is quivalent to purity of ( number of ROIs with signal / number of ROIs)
286  double roi_sig[3] = {0., 0., 0.}; // number of roi contains signal in an event
287  double roi_total[3] = {0., 0., 0.}; // number of roi in an event
288 
289  for (auto& wire : wires) {
290  const raw::ChannelID_t wirechannel = wire->Channel();
291  if (chStatus.IsBad(wirechannel)) continue;
292 
293  int view = wire->View();
294 
295  const recob::Wire::RegionsOfInterest_t& signalROI = wire->SignalROI();
296 
297  if (signalROI.get_ranges().empty()) continue;
298 
299  std::vector<float> vecADC =
300  wire->Signal(); // a zero-padded full length vector filled with RoI signal.
301 
302  roi_total[view] += signalROI.get_ranges().size();
303  fCount_Roi_total[view] += signalROI.get_ranges().size();
304 
305  for (const auto& range : signalROI.get_ranges()) {
306  bool IsSigROI = false;
307 
308  raw::TDCtick_t roiFirstBinTick = range.begin_index();
309  raw::TDCtick_t roiLastBinTick = range.end_index();
310 
311  double maxadc_sig = -99.;
312  for (int k = roiFirstBinTick; k < roiLastBinTick; k++) {
313  if (std::abs(vecADC[k]) > maxadc_sig) maxadc_sig = std::abs(vecADC[k]);
314  }
315  h1_roi_max[view]->Fill(maxadc_sig);
316 
317  // check simulation: ideally we could put this part outside loop over range to make the algorithm more efficient. However, as there are only one or two ROIs in a channel for most of cases, we keep this style to make the algorithm more readable. Also, signal information are kept here.
318  for (auto const& channel : (*simChannelHandle)) {
319  if (wirechannel != channel.Channel()) continue;
320 
321  int tdctick_previous = -999;
322  double totalE = 0.;
323  double maxE = -999.;
324  int maxEtick = -999;
325  std::vector<int> signal_starttick;
326  std::vector<int> signal_endtick;
327  std::vector<double> signal_energydeposits;
328  std::vector<double> signal_energy_max;
329  std::vector<double> signal_max_tdctick;
330 
331  auto const& timeSlices = channel.TDCIDEMap();
332 
333  for (auto const& timeSlice : timeSlices) {
334  auto const tpctime = timeSlice.first;
335  auto const& energyDeposits = timeSlice.second;
336  int tdctick = static_cast<int>(clockData.TPCTDC2Tick(double(tpctime)));
337  if (tdctick < 0 || tdctick > int(detProp.ReadOutWindowSize()) - 1) continue;
338 
339  double e_deposit = 0;
340  for (auto const& energyDeposit : energyDeposits) {
341  e_deposit += energyDeposit.energy;
342  }
343 
344  if (tdctick_previous == -999) {
345  signal_starttick.push_back(tdctick);
346  totalE += e_deposit;
347  maxE = e_deposit;
348  maxEtick = tdctick;
349  }
350  else if (tdctick - tdctick_previous != 1) {
351  signal_starttick.push_back(tdctick);
352  signal_endtick.push_back(tdctick_previous);
353  signal_energydeposits.push_back(totalE);
354  signal_energy_max.push_back(maxE);
355  signal_max_tdctick.push_back(maxEtick);
356  totalE = e_deposit;
357  maxE = e_deposit;
358  maxEtick = tdctick;
359  }
360  else if (tdctick - tdctick_previous == 1) {
361  totalE += e_deposit;
362  if (maxE < e_deposit) {
363  maxE = e_deposit;
364  maxEtick = tdctick;
365  }
366  }
367 
368  tdctick_previous = tdctick;
369 
370  } // loop over timeSlices timeSlice
371 
372  signal_endtick.push_back(tdctick_previous); // for last one
373  signal_energydeposits.push_back(totalE); // for last one
374  signal_energy_max.push_back(maxE); // for last one
375  signal_max_tdctick.push_back(maxEtick); // for last one
376 
377  if (signal_starttick.size() == 0 ||
378  (signal_endtick.size() == 1 && signal_endtick.back() == -999))
379  continue;
380 
381  // check if signal/signals in this ROI
382  for (size_t s = 0; s < signal_starttick.size(); s++) {
383  if (isSignalInROI(signal_starttick[s],
384  signal_endtick[s],
385  signal_max_tdctick[s],
386  roiFirstBinTick,
387  roiLastBinTick)) {
388  IsSigROI = true;
389  break;
390  }
391  } // loop over s
392  } // loop simChannels
393 
394  h_roi[view]->Fill(0); // total roi
395  if (IsSigROI) {
396  roi_sig[view] += 1.;
397  fCount_Roi_sig[view] += 1;
398  h_roi[view]->Fill(1); // sig roi
399  h1_roi_max_sim[view]->Fill(maxadc_sig);
400  }
401  } // loop ranges of signalROI
402  } // loop wires
403 
404  // purity of each plane for each event
405  for (int i = 0; i < 3; i++) {
406  if (roi_total[i]) {
407  double purity = roi_sig[i] / roi_total[i];
408  if (purity == 1) purity = 1. - 1.e-6;
409  h_purity[i]->Fill(purity);
410  }
411  }
412 
413  // combined purity for each event
414  if (roi_total[0] + roi_total[1] + roi_total[2]) {
415  double purity =
416  (roi_sig[0] + roi_sig[1] + roi_sig[2]) / (roi_total[0] + roi_total[1] + roi_total[2]);
417  if (purity == 1) purity = 1. - 1.e-6;
418  h_purity_all->Fill(purity);
419  }
420 }
421 
423 {
424 
426 
427  for (int i = 0; i < 3; ++i) {
428  h_energy[i] =
429  tfs->make<TH1D>(Form("h_energy_%d", i), Form("Plane %d; Energy (MeV);", i), 100, 0, 1);
430  h_energy_roi[i] =
431  tfs->make<TH1D>(Form("h_energy_roi_%d", i), Form("Plane %d; Energy (MeV);", i), 100, 0, 1);
432 
433  h_purity[i] = tfs->make<TH1D>(Form("h_purity_%d", i), Form("Plane %d; Purity;", i), 20, 0, 1);
434  h_roi[i] = tfs->make<TH1D>(Form("h_roi_%d", i),
435  Form("Plane %d; Purity;", i),
436  5,
437  0,
438  5); // index 0: total rois; index 1: total sig rois
439 
440  h1_roi_max[i] =
441  tfs->make<TH1D>(Form("h1_roi_max_%d", i), Form("Plane %d; Max adc;", i), 50, 0, 50);
442  h1_roi_max_sim[i] =
443  tfs->make<TH1D>(Form("h1_roi_max_sim_%d", i), Form("Plane %d; Max adc;", i), 50, 0, 50);
444 
445  h1_tickdiff_max[i] = tfs->make<TH1D>(Form("h1_tickdiff_max_%d", i),
446  Form("Plane %d; tick diff (maxE - max pulse);", i),
447  100,
448  -50,
449  50);
450  }
451 
452  h_purity_all = tfs->make<TH1D>("h_purity_all", "All Planes; Purity;", 20, 0, 1);
453 }
454 
456 {
458 
459  for (int i = 0; i < 3; ++i) {
460  if (TEfficiency::CheckConsistency(*(h_energy_roi[i]), *(h_energy[i]))) {
461  eff_energy[i] = tfs->make<TEfficiency>(*(h_energy_roi[i]), *(h_energy[i]));
462  eff_energy[i]->Write(Form("eff_energy_%d", i));
463  }
464  }
465 }
466 
468  int endtick,
469  int maxtick,
470  int roistart,
471  int roiend)
472 {
473  // For a signal in the ROI, two cases are considered:
474  // (i) signal is totally in the ROI
475  // (ii) signal is partially in the ROI
476  // Equivalently, we can just check whether the maxtick is in the ROI (simple one).
477  return roistart <= maxtick && maxtick < roiend;
478 }
479 
art::InputTag fSimulationProducerLabel
Utilities related to art service access.
MaybeLogger_< ELseverityLevel::ELsev_info, false > LogInfo
constexpr auto abs(T v)
Returns the absolute value of the argument.
const range_list_t & get_ranges() const
Returns the internal list of non-void ranges.
cout<< "Opened file "<< fin<< " ixs= "<< ixs<< endl;if(ixs==0) hhh=(TH1F *) fff-> Get("h1")
Definition: AddMC.C:8
EvaluateROIEff(fhicl::ParameterSet const &p)
EDAnalyzer(fhicl::ParameterSet const &pset)
Definition: EDAnalyzer.cc:6
int TDCtick_t
Type representing a TDC tick.
Definition: RawTypes.h:25
#define DEFINE_ART_MODULE(klass)
Definition: ModuleMacros.h:65
void analyze(art::Event const &e) override
Definition: EmTrack.h:40
EvaluateROIEff & operator=(EvaluateROIEff const &)=delete
ValidHandle< PROD > getValidHandle(InputTag const &tag) const
bool getByLabel(std::string const &label, std::string const &instance, Handle< PROD > &result) const
bool isSignalInROI(int starttick, int endtick, int maxtick, int roistart, int roiend)
object containing MC truth information necessary for making RawDigits and doing back tracking ...
Declaration of basic channel signal object.
unsigned int ChannelID_t
Type representing the ID of a readout channel.
Definition: RawTypes.h:28
void fill_ptr_vector(std::vector< Ptr< T >> &ptrs, H const &h)
Definition: Ptr.h:306
double maxE
Definition: plot_hist.C:8
Float_t e
Definition: plot.C:35