Class Reference for E1039 Core & Analysis Software
EventReducer.cxx
Go to the documentation of this file.
1 #include <iostream>
2 #include <stdlib.h>
3 
4 #include "EventReducer.h"
5 
6 EventReducer::EventReducer(TString options) : afterhit(false), hodomask(false), outoftime(false), decluster(false), mergehodo(false), triggermask(false), sagitta(false), hough(false), realization(false), difnim(false)
7 {
8  //parse the reducer setup
9  options.ToLower();
10  if(options.Contains("a")) afterhit = true;
11  if(options.Contains("h")) hodomask = true;
12  if(options.Contains("o")) outoftime = true;
13  if(options.Contains("c")) decluster = true;
14  if(options.Contains("m")) mergehodo = true;
15  if(options.Contains("t")) triggermask = true;
16  if(options.Contains("s")) sagitta = true;
17  if(options.Contains("g")) hough = true;
18  if(options.Contains("r")) realization = true;
19  if(options.Contains("n")) difnim = true;
20 
21  if(options.Contains("e")) {
22  std::cout << "EventReducer: !!ERROR!! Option 'e' is no longer available. Use 'CalibDriftDist'. Abort." << std::endl;
23  exit(1);
24  }
25 
26  rc = recoConsts::instance();
27  timeOffset = rc->get_DoubleFlag("TDCTimeOffset");
28  SAGITTA_TARGET_CENTER = rc->get_DoubleFlag("SAGITTA_TARGET_CENTER");
29  SAGITTA_TARGET_WIDTH = rc->get_DoubleFlag("SAGITTA_TARGET_WIDTH");
30  SAGITTA_DUMP_CENTER = rc->get_DoubleFlag("SAGITTA_DUMP_CENTER");
31  SAGITTA_TARGET_WIDTH = rc->get_DoubleFlag("SAGITTA_DUMP_WIDTH");
32  Z_TARGET = rc->get_DoubleFlag("Z_TARGET");
33  Z_DUMP = rc->get_DoubleFlag("Z_DUMP");
34 
35  TX_MAX = rc->get_DoubleFlag("TX_MAX");
36  TY_MAX = rc->get_DoubleFlag("TY_MAX");
37  USE_V1495_HIT = rc->get_BoolFlag("USE_V1495_HIT");
38  USE_TWTDC_HIT = rc->get_BoolFlag("USE_TWTDC_HIT");
39 
40  chamEff = 0.94;
41  chamResol = 0.04;
42 
43  //Screen output for all the methods enabled
44  if(afterhit) std::cout << "EventReducer: after-pulse removal enabled. " << std::endl;
45  if(hodomask) std::cout << "EventReducer: hodoscope masking enabled. " << std::endl;
46  if(outoftime) std::cout << "EventReducer: out-of-time hits removal enabled. " << std::endl;
47  if(decluster) std::cout << "EventReducer: hit cluster removal enabled. " << std::endl;
48  if(mergehodo) std::cout << "EventReducer: v1495 hits will be merged with TW-TDC hits. " << std::endl;
49  if(triggermask) std::cout << "EventReducer: trigger road masking enabled. " << std::endl;
50  if(sagitta) std::cout << "EventReducer: sagitta reducer enabled. " << std::endl;
51  if(hough) std::cout << "EventReducer: hough transform reducer enabled. " << std::endl;
52  if(realization) std::cout << "EventReducer: realization enabled. " << std::endl;
53  if(difnim) std::cout << "EventReducer: trigger masking will be disabled in NIM events. " << std::endl;
54  if(fabs(timeOffset) > 0.01) std::cout << "EventReducer: " << timeOffset << " ns will be added to tdcTime. " << std::endl;
55 
56  //initialize services
57  p_geomSvc = GeomSvc::instance();
58  if(triggermask)
59  {
60  p_triggerAna = new TriggerAnalyzer();
61  p_triggerAna->init();
62  p_triggerAna->buildTriggerTree();
63  }
64 
65  if(hodomask) initHodoMaskLUT();
66 
67  //set random seed
68  rndm.SetSeed(0);
69 }
70 
72 {
73  if(triggermask)
74  {
75  delete p_triggerAna;
76  }
77 }
78 
80 {
81  int nHits_before = rawEvent->getNChamberHitsAll();
82 
83  //temporarily disable trigger road masking if this event is not fired by any MATRIX triggers
84  bool triggermask_local = triggermask;
85  if(!(rc->get_BoolFlag("MC_MODE") || rawEvent->isFPGATriggered()))
86  {
87  triggermask_local = false;
88  }
89 
90  //dump the vector of hits from SRawEvent to a list first
91  hitlist.clear();
92  hodohitlist.clear();
93  for(std::vector<Hit>::iterator iter = rawEvent->fAllHits.begin(); iter != rawEvent->fAllHits.end(); ++iter)
94  {
95  if(outoftime && (!iter->isInTime())) continue;
96 
97  if(iter->detectorID <= nChamberPlanes) //chamber hits
98  {
99  if(realization && rndm.Rndm() > chamEff) continue;
100  //if(hodomask && (!iter->isHodoMask())) continue;
101  //if(triggermask && (!iter->isTriggerMask())) continue;
102  }
103  else if(iter->detectorID > nChamberPlanes && iter->detectorID <= nChamberPlanes+nHodoPlanes)
104  {
105  // if trigger masking is enabled, all the X hodos are discarded
106  if(triggermask_local && p_geomSvc->getPlaneType(iter->detectorID) == 1) continue;
107  }
108 
109  if(realization && iter->detectorID <= nChamberPlanes) iter->driftDistance += rndm.Gaus(0., chamResol);
110 
111  if(iter->detectorID >= nChamberPlanes+1 && iter->detectorID <= nChamberPlanes+nHodoPlanes)
112  {
113  hodohitlist.push_back(*iter);
114  }
115  else
116  {
117  hitlist.push_back(*iter);
118  }
119  }
120 
121  if(mergehodo)
122  {
123  for(std::vector<Hit>::iterator iter = rawEvent->fTriggerHits.begin(); iter != rawEvent->fTriggerHits.end(); ++iter)
124  {
125  if(triggermask_local && p_geomSvc->getPlaneType(iter->detectorID) == 1) continue;
126  hodohitlist.push_back(*iter);
127  }
128  }
129 
130  // manully create the X-hodo hits by the trigger roads
131  if(triggermask_local) p_triggerAna->trimEvent(rawEvent, hodohitlist, mergehodo || USE_V1495_HIT, mergehodo || USE_TWTDC_HIT);
132 
133  //apply hodoscope mask
134  hodohitlist.sort();
135  if(hodomask) hodoscopeMask(hitlist, hodohitlist);
136 
137  //Remove after hits
138  hitlist.sort();
139  hitlist.merge(hodohitlist);
140  if(afterhit) hitlist.unique();
141 
142  //Remove hit clusters
143  if(decluster) deClusterize();
144 
145  //Remove the hits by sagitta ratio
146  if(sagitta) sagittaReducer();
147 
148  //Push everything back to SRawEvent
149  rawEvent->fAllHits.clear();
150  rawEvent->fAllHits.assign(hitlist.begin(), hitlist.end());
151 
152  rawEvent->reIndex();
153  return nHits_before - rawEvent->getNChamberHitsAll();
154 }
155 
157 {
158  //find index for D1, D2, and D3
159  int nHits_D1 = 0;
160  int nHits_D2 = 0;
161  int nHits_D3 = 0;
162  int detectorID_st1_max = 12;
163  int detectorID_st2_max = 18;
164 
165  //hitlist here is assumed to be sorted of course
166  for(std::list<Hit>::iterator iter = hitlist.begin(); iter != hitlist.end(); ++iter)
167  {
168  if(iter->detectorID > nChamberPlanes) break;
169  if(iter->detectorID <= detectorID_st1_max)
170  {
171  ++nHits_D1;
172  }
173  else if(iter->detectorID <= detectorID_st2_max)
174  {
175  ++nHits_D2;
176  }
177  else
178  {
179  ++nHits_D3;
180  }
181  }
182  int idx_D1 = nHits_D1;
183  int idx_D2 = nHits_D1 + nHits_D2;
184  int idx_D3 = nHits_D1 + nHits_D2 + nHits_D3;
185 
186  //Loop over all hits
187  std::vector<Hit> hitTemp;
188  hitTemp.assign(hitlist.begin(), hitlist.end());
189 
190  std::vector<int> flag(hitTemp.size(), -1);
191  for(int i = idx_D2; i < idx_D3; ++i)
192  {
193  double z3 = p_geomSvc->getPlanePosition(hitTemp[i].detectorID);
194  double slope_target = hitTemp[i].pos/(z3 - Z_TARGET);
195  double slope_dump = hitTemp[i].pos/(z3 - Z_DUMP);
196  for(int j = idx_D1; j < idx_D2; ++j)
197  {
198  if(p_geomSvc->getPlaneType(hitTemp[i].detectorID) != p_geomSvc->getPlaneType(hitTemp[j].detectorID)) continue;
199 
200  double z2 = p_geomSvc->getPlanePosition(hitTemp[j].detectorID);
201  if(fabs((hitTemp[i].pos - hitTemp[j].pos)/(z2 - z3)) > TX_MAX) continue;
202  double s2_target = hitTemp[j].pos - slope_target*(z2 - Z_TARGET);
203  double s2_dump = hitTemp[j].pos - slope_dump*(z2 - Z_DUMP);
204 
205  for(int k = 0; k < idx_D1; ++k)
206  {
207  if(p_geomSvc->getPlaneType(hitTemp[i].detectorID) != p_geomSvc->getPlaneType(hitTemp[k].detectorID)) continue;
208  if(flag[i] > 0 && flag[j] > 0 && flag[k] > 0) continue;
209 
210  double z1 = p_geomSvc->getPlanePosition(hitTemp[k].detectorID);
211  double pos_exp_target = SAGITTA_TARGET_CENTER*s2_target + slope_target*(z1 - Z_TARGET);
212  double pos_exp_dump = SAGITTA_DUMP_CENTER*s2_dump + slope_dump*(z1 - Z_DUMP);
213  double win_target = fabs(s2_target*SAGITTA_TARGET_WIDTH);
214  double win_dump = fabs(s2_dump*SAGITTA_DUMP_WIDTH);
215 
216  double p_min = std::min(pos_exp_target - win_target, pos_exp_dump - win_dump);
217  double p_max = std::max(pos_exp_target + win_target, pos_exp_dump + win_dump);
218 
219  if(hitTemp[k].pos > p_min && hitTemp[k].pos < p_max)
220  {
221  flag[i] = 1;
222  flag[j] = 1;
223  flag[k] = 1;
224  }
225  }
226  }
227  }
228 
229  int idx = 0;
230  for(std::list<Hit>::iterator iter = hitlist.begin(); iter != hitlist.end(); )
231  {
232  if(flag[idx] < 0)
233  {
234  iter = hitlist.erase(iter);
235  }
236  else
237  {
238  ++iter;
239  }
240 
241  ++idx;
242  if(idx >= idx_D3) break;
243  }
244 }
245 
247 {
248  std::vector<std::list<Hit>::iterator> cluster;
249  cluster.clear();
250  for(std::list<Hit>::iterator hit = hitlist.begin(); hit != hitlist.end(); ++hit)
251  {
252  //if we already reached the hodo part, stop
253  if(hit->detectorID > nChamberPlanes) break;
254 
255  if(cluster.size() == 0)
256  {
257  cluster.push_back(hit);
258  }
259  else
260  {
261  if(hit->detectorID != cluster.back()->detectorID)
262  {
263  processCluster(cluster);
264  cluster.push_back(hit);
265  }
266  else if(hit->elementID - cluster.back()->elementID > 1)
267  {
268  processCluster(cluster);
269  cluster.push_back(hit);
270  }
271  else
272  {
273  cluster.push_back(hit);
274  }
275  }
276  }
277 }
278 
279 void EventReducer::processCluster(std::vector<std::list<Hit>::iterator>& cluster)
280 {
281  unsigned int clusterSize = cluster.size();
282 
283  //size-2 clusters, retain the hit with smaller driftDistance
284  if(clusterSize == 2)
285  {
286  double w_max = 0.9*0.5*(cluster.back()->pos - cluster.front()->pos);
287  double w_min = w_max/9.*4.; //double w_min = 0.6*0.5*(cluster.back()->pos - cluster.front()->pos);
288 
289  if((cluster.front()->driftDistance > w_max && cluster.back()->driftDistance > w_min) || (cluster.front()->driftDistance > w_min && cluster.back()->driftDistance > w_max))
290  {
291  cluster.front()->driftDistance > cluster.back()->driftDistance ? hitlist.erase(cluster.front()) : hitlist.erase(cluster.back());
292  }
293  else if(fabs(cluster.front()->tdcTime - cluster.back()->tdcTime) < 8. && cluster.front()->detectorID >= 19 && cluster.front()->detectorID <= 24)
294  {
295  hitlist.erase(cluster.front());
296  hitlist.erase(cluster.back());
297  }
298  }
299 
300  //size-larger-than-3, discard entirely
301  if(clusterSize >= 3)
302  {
303  double dt_mean = 0.;
304  for(unsigned int i = 1; i < clusterSize; ++i)
305  {
306  dt_mean += fabs(cluster[i]->tdcTime - cluster[i-1]->tdcTime);
307  }
308  dt_mean = dt_mean/(clusterSize - 1);
309 
310  if(dt_mean < 10.)
311  {
312  //electric noise, discard them all
313  for(unsigned int i = 0; i < clusterSize; ++i)
314  {
315  hitlist.erase(cluster[i]);
316  }
317  }
318  else
319  {
320  /*
321  double dt_rms = 0.;
322  for(unsigned int i = 1; i < clusterSize; ++i)
323  {
324  double dt = fabs(cluster[i]->tdcTime - cluster[i-1]->tdcTime);
325  dt_rms += ((dt - dt_mean)*(dt - dt_mean));
326  }
327  dt_rms = sqrt(dt_rms/(clusterSize - 1));
328 
329  //delta ray, keep the first and last
330  if(dt_rms < 5.)*/
331  {
332  for(unsigned int i = 1; i < clusterSize - 1; ++i)
333  {
334  hitlist.erase(cluster[i]);
335  }
336  }
337  }
338  }
339 
340  cluster.clear();
341 }
342 
344 {
345  h2celementID_lo.clear();
346  h2celementID_hi.clear();
347 
348  TString hodoNames[8] = {"H1B", "H1T", "H2B", "H2T", "H3B", "H3T", "H4B", "H4T"};
349  int hodoIDs[8];
350  for(int i = 0; i < 8; ++i) hodoIDs[i] = p_geomSvc->getDetectorID(hodoNames[i].Data());
351 
352  int chamIDs[8][12] = { {1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12},
353  {1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12},
354  {13, 14, 15, 16, 17, 18, 0, 0, 0, 0, 0, 0},
355  {13, 14, 15, 16, 17, 18, 0, 0, 0, 0, 0, 0},
356  {0, 0, 0, 0, 0, 0, 25, 26, 27, 28, 29, 30},
357  {19, 20, 21, 22, 23, 24, 0, 0, 0, 0, 0, 0},
358  {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0},
359  {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}};
360 
361  for(int i = 0; i < 8; ++i)
362  {
363  int nPaddles = p_geomSvc->getPlaneNElements(hodoIDs[i]);
364  for(int j = 1; j <= nPaddles; ++j)
365  {
366  //for each paddle, there is a group of 6/12 chamber planes to work with
367  int uniqueID = hodoIDs[i]*1000 + j;
368 
369  //get the four corners of the paddle
370  double z0, x0_min, x0_max, y0_min, y0_max;
371  z0 = p_geomSvc->getPlanePosition(hodoIDs[i]);
372  p_geomSvc->get2DBoxSize(hodoIDs[i], j, x0_min, x0_max, y0_min, y0_max);
373 
374  for(int k = 0; k < 12; ++k)
375  {
376  if(chamIDs[i][k] < 1) continue;
377 
378  //project the box to the chamber plane with maximum xz/yz slope
379  double z = p_geomSvc->getPlanePosition(chamIDs[i][k]);
380  double x_min = x0_min - fabs(TX_MAX*(z - z0));
381  double x_max = x0_max + fabs(TX_MAX*(z - z0));
382  double y_min = y0_min - fabs(TY_MAX*(z - z0));
383  double y_max = y0_max + fabs(TY_MAX*(z - z0));
384 
385  int elementID_lo = p_geomSvc->getPlaneNElements(chamIDs[i][k]);;
386  int elementID_hi = 0;
387  if(p_geomSvc->getPlaneType(chamIDs[i][k]) == 1)
388  {
389  elementID_lo = p_geomSvc->getExpElementID(chamIDs[i][k], x_min);
390  elementID_hi = p_geomSvc->getExpElementID(chamIDs[i][k], x_max);
391  }
392  else
393  {
394  for(int m = 1; m <= p_geomSvc->getPlaneNElements(chamIDs[i][k]); ++m)
395  {
396  double x1, y1, x2, y2;
397  p_geomSvc->getWireEndPoints(chamIDs[i][k], m, x1, x2, y1, y2);
398 
399  if(!lineCrossing(x_min, y_min, x_min, y_max, x1, y1, x2, y2) &&
400  !lineCrossing(x_max, y_min, x_max, y_max, x1, y1, x2, y2)) continue;
401 
402  if(m < elementID_lo) elementID_lo = m;
403  if(m > elementID_hi) elementID_hi = m;
404  }
405  }
406 
407  elementID_lo -= 2;
408  if(elementID_lo < 1) elementID_lo = 1;
409  elementID_hi += 2;
410  if(elementID_hi > p_geomSvc->getPlaneNElements(chamIDs[i][k])) elementID_hi = p_geomSvc->getPlaneNElements(chamIDs[i][k]);
411 
412  h2celementID_lo[uniqueID].push_back(chamIDs[i][k]*1000 + elementID_lo);
413  h2celementID_hi[uniqueID].push_back(chamIDs[i][k]*1000 + elementID_hi);
414  }
415  }
416  }
417 
418  //reverse the LUT
419  c2helementIDs.clear();
420  for(LUT::iterator iter = h2celementID_lo.begin(); iter != h2celementID_lo.end(); ++iter)
421  {
422  int hodoUID = iter->first;
423  for(unsigned int i = 0; i < h2celementID_lo[hodoUID].size(); ++i)
424  {
425  int chamUID_lo = iter->second[i];
426  int chamUID_hi = h2celementID_hi[hodoUID][i];
427  for(int j = chamUID_lo; j <= chamUID_hi; ++j)
428  {
429  c2helementIDs[j].push_back(hodoUID);
430  }
431  }
432  }
433 }
434 
435 void EventReducer::hodoscopeMask(std::list<Hit>& chamberhits, std::list<Hit>& hodohits)
436 {
437  for(std::list<Hit>::iterator iter = chamberhits.begin(); iter != chamberhits.end(); )
438  {
439  if(iter->detectorID > nChamberPlanes)
440  {
441  ++iter;
442  continue;
443  }
444 
445  int uniqueID = iter->uniqueID();
446 
447  bool masked = false;
448  for(std::vector<int>::iterator jter = c2helementIDs[uniqueID].begin(); jter != c2helementIDs[uniqueID].end(); ++jter)
449  {
450  if(std::find(hodohits.begin(), hodohits.end(), Hit(*jter)) != hodohits.end())
451  {
452  masked = true;
453  break;
454  }
455  }
456 
457  if(masked)
458  {
459  ++iter;
460  }
461  else
462  {
463  iter = chamberhits.erase(iter);
464  }
465  }
466 }
467 
468 bool EventReducer::lineCrossing(double x1, double y1, double x2, double y2,
469  double x3, double y3, double x4, double y4)
470 {
471  double tc = (x1 - x2)*(y3 - y1) + (y1 - y2)*(x1 - x3);
472  double td = (x1 - x2)*(y4 - y1) + (y1 - y2)*(x1 - x4);
473 
474  return tc*td < 0;
475 }
#define nChamberPlanes
Definition: GlobalConsts.h:6
#define nHodoPlanes
Definition: GlobalConsts.h:7
void sagittaReducer()
EventReducer(TString options)
Definition: EventReducer.cxx:6
bool lineCrossing(double x1, double y1, double x2, double y2, double x3, double y3, double x4, double y4)
void deClusterize()
void processCluster(std::vector< std::list< Hit >::iterator > &cluster)
void initHodoMaskLUT()
void hodoscopeMask(std::list< Hit > &chamberhits, std::list< Hit > &hodohits)
int reduceEvent(SRawEvent *rawEvent)
int getDetectorID(const std::string &detectorName) const
Get the plane position.
Definition: GeomSvc.h:219
double getPlanePosition(int detectorID) const
Definition: GeomSvc.h:235
void getWireEndPoints(int detectorID, int elementID, double &x_min, double &x_max, double &y_min, double &y_max)
Definition: GeomSvc.cxx:725
static GeomSvc * instance()
singlton instance
Definition: GeomSvc.cxx:212
int getPlaneType(int detectorID) const
Definition: GeomSvc.h:261
int getPlaneNElements(int detectorID) const
Definition: GeomSvc.h:260
int getExpElementID(int detectorID, double pos_exp)
Definition: GeomSvc.cxx:677
void get2DBoxSize(int detectorID, int elementID, double &x_min, double &x_max, double &y_min, double &y_max)
Definition: GeomSvc.cxx:700
Definition of hit structure.
Definition: SRawEvent.h:35
virtual double get_DoubleFlag(const std::string &name) const
Definition: PHFlag.cc:49
virtual bool get_BoolFlag(const std::string &name) const
Definition: PHFlag.cc:151
void reIndex(bool doSort=false)
Reset the number hits on each plane.
Definition: SRawEvent.cxx:559
bool isFPGATriggered()
Definition: SRawEvent.cxx:640
Int_t getNChamberHitsAll()
Definition: SRawEvent.cxx:381
bool init(std::string fileName, double cut_td=0., double cut_gun=1E8)
void trimEvent(SRawEvent *rawEvent, std::list< Hit > &hitlist, bool USE_TRIGGER_HIT, bool USE_HIT)
static recoConsts * instance()
Definition: recoConsts.cc:7