Class Reference for E1039 Core & Analysis Software
DPTriggerAnalyzer.cxx
Go to the documentation of this file.
1 #include "DPTriggerAnalyzer.h"
2 #include <interface_main/SQRun.h>
5 #include <geom_svc/GeomSvc.h>
6 #include <fun4all/Fun4AllBase.h>
8 #include <phool/recoConsts.h>
9 #include <phool/PHNodeIterator.h>
10 #include <phool/PHCompositeNode.h>
11 #include <phool/PHIODataNode.h>
12 #include <phool/getClass.h>
13 #include <TSystem.h>
14 #include <iomanip>
15 #include <fstream>
16 #include <sstream>
17 #include <string>
18 using namespace std;
19 
20 DPTriggerRoad::DPTriggerRoad() : roadID(0), sigWeight(0.), bkgRate(0.), pXmin(0.)
21 {
22  uniqueTrIDs.clear();
23 }
24 
25 DPTriggerRoad::DPTriggerRoad(const std::list<int>& path) : roadID(0), sigWeight(0.), bkgRate(0.), pXmin(0.)
26 {
27  uniqueTrIDs.clear();
28  for(std::list<int>::const_iterator iter = path.begin(); iter != path.end(); ++iter)
29  {
30  if(*iter < 0) continue;
31  uniqueTrIDs.push_back(*iter);
32  }
33 }
34 
36 {
37  int TB = 0;
38  for(unsigned int i = 0; i < NTRPLANES; ++i)
39  {
40  int detectorID = getTrDetectorID(i);
41  TB = TB + (((detectorID & 1) == 0) ? 1 : -1);
42  }
43 
44  if(TB == 4) return 1;
45  if(TB == -4) return -1;
46 
47  return 0;
48 }
49 
51 {
52  int corr = getTB();
53  for(unsigned int i = 0; i < NTRPLANES; ++i)
54  {
55  int detectorID = getTrDetectorID(i);
56  int elementID = getTrElementID(i);
57 
58  detectorID -= corr;
59  uniqueTrIDs[i] = detectorID*1000 + elementID;
60  }
61 }
62 
64 {
65  for(unsigned int i = 0; i < NTRPLANES; ++i)
66  {
67  if(uniqueTrIDs[i] != elem.uniqueTrIDs[i]) return false;
68  }
69 
70  return true;
71 }
72 
74 {
75  return sigWeight < elem.sigWeight;
76 }
77 
79 {
80  TString sid;
81  for(unsigned int i = 0; i < uniqueTrIDs.size(); ++i)
82  {
83  sid += TString::Format("%06d", uniqueTrIDs[i]);
84  }
85 
86  return sid;
87 }
88 
89 std::ostream& operator << (std::ostream& os, const DPTriggerRoad& road)
90 {
91  os << "Trigger Road ID = " << road.roadID << ", signal = " << road.sigWeight << ", bkg = " << road.bkgRate << "\n ";
92  for(unsigned int i = 0; i < NTRPLANES; ++i)
93  {
94  os << road.uniqueTrIDs[i] << " == ";
95  }
96 
97  return os;
98 }
99 
101 
102 DPTriggerAnalyzer::DPTriggerAnalyzer(const std::string& name) :
103  SubsysReco(name),
104  _road_set_file_name("trigger_67.txt"),
105  _output_node_name(""),
106  _use_trig_hit(false),
107  _req_intime(false),
108  _mode_nim1(NIM_AND),
109  _mode_nim2(NIM_AND),
110  _run_header(nullptr),
111  _event_header(nullptr),
112  _event_header_out(nullptr),
113  _hit_vector(nullptr)
114 {
115 }
116 
118 {
119  deleteMatrix(matrix[0]);
120  deleteMatrix(matrix[1]);
121 }
122 
123 void DPTriggerAnalyzer::set_nim_mode(const NimMode nim1, const NimMode nim2)
124 {
125  _mode_nim1 = nim1;
126  _mode_nim2 = nim2;
127 }
128 
135 {
136  //Load the trigger roads
137  int charge = (int) -1e3, roadID = -1;
138  int uIDs[NTRPLANES] = { -1, -1, -1, -1 };
139  double pXmin = -1.e6, sigWeight = -1.e6, bkgRate = -1.e6;
140  string line = "";
141  std::ifstream fin(gSystem->ExpandPathName(_road_set_file_name.c_str()), std::ifstream::in);
142  if (fin.is_open()) {
143  recoConsts::instance()->set_CharFlag("TRIGGER_EMU_L1", _road_set_file_name);
144  while (getline(fin, line)) {
145  stringstream ss(line);
146 
147  ss >> charge >> roadID;
148  for (int i = 0; i < NTRPLANES; ++i)
149  ss >> uIDs[i];
150  ss >> pXmin >> sigWeight >> bkgRate;
151 
152  DPTriggerRoad road;
153  road.setRoadID(roadID);
154  road.setSigWeight(sigWeight);
155  road.setBkgRate(bkgRate);
156  road.setPxMin(pXmin);
157 
158  for (int i = 0; i < NTRPLANES; ++i)
159  road.addTrElement(uIDs[i]);
160  roads[(-charge + 1) / 2].insert(
161  std::map<TString, DPTriggerRoad>::value_type(road.getStringID(),
162  road));
163  }
164  } else {
165  NIMONLY = true;
166  LogInfo("NIM Only mode NOT supported now.");
168  }
169  //build the search matrix
171 
173 }
174 
176 {
177  return GetNodes(topNode);
178 }
179 
181  GeomSvc* geomSvc = GeomSvc::instance();
182 
183  if (_event_header_out != _event_header) {
184  *_event_header_out = *_event_header; // Set the contents of the output node
185  }
186 
187  //NIM trigger first
189  map<int, int> nhit_det; // [det_id] = counts
190  for (Int_t ihit = 0; ihit < _hit_vector->size(); ++ihit) {
191  if (_req_intime && ! _hit_vector->at(ihit)->is_in_time()) continue;
192  nhit_det[ _hit_vector->at(ihit)->get_detector_id() ]++;
193  // Note: GeomSvc::getTriggerLv() was used in the past version, but it should not be used since the "trigger level" was not a standard variable.
194  }
195 
196  bool HXB = nhit_det[ geomSvc->getDetectorID("H1B" ) ] > 0 &&
197  nhit_det[ geomSvc->getDetectorID("H2B" ) ] > 0 &&
198  nhit_det[ geomSvc->getDetectorID("H3B" ) ] > 0 &&
199  nhit_det[ geomSvc->getDetectorID("H4B" ) ] > 0 ;
200  bool HXT = nhit_det[ geomSvc->getDetectorID("H1T" ) ] > 0 &&
201  nhit_det[ geomSvc->getDetectorID("H2T" ) ] > 0 &&
202  nhit_det[ geomSvc->getDetectorID("H3T" ) ] > 0 &&
203  nhit_det[ geomSvc->getDetectorID("H4T" ) ] > 0 ;
204  bool HYL = nhit_det[ geomSvc->getDetectorID("H1L" ) ] > 0 &&
205  nhit_det[ geomSvc->getDetectorID("H2L" ) ] > 0 &&
206  nhit_det[ geomSvc->getDetectorID("H4Y1L") ] > 0 &&
207  nhit_det[ geomSvc->getDetectorID("H4Y2L") ] > 0 ;
208  bool HYR = nhit_det[ geomSvc->getDetectorID("H1R" ) ] > 0 &&
209  nhit_det[ geomSvc->getDetectorID("H2R" ) ] > 0 &&
210  nhit_det[ geomSvc->getDetectorID("H4Y1R") ] > 0 &&
211  nhit_det[ geomSvc->getDetectorID("H4Y2R") ] > 0 ;
212  bool nim1_on = _mode_nim1 == NIM_AND ? HYL && HYR : HYL || HYR;
213  bool nim2_on = _mode_nim2 == NIM_AND ? HXT && HXB : HXT || HXB;
214  _event_header_out->set_trigger(SQEvent::NIM1, nim1_on);
215  _event_header_out->set_trigger(SQEvent::NIM2, nim2_on);
216 
217  //For FPGA trigger, build the internal hit pattern first
218  data.clear();
219  data.resize(NTRPLANES);
220  for(Int_t ihit = 0; ihit < _hit_vector->size(); ++ihit) {
221  SQHit *hit=_hit_vector->at(ihit);
222  if (_req_intime && ! hit->is_in_time()) continue;
223  int index = geomSvc->getHodoStation(hit->get_detector_id()) - 1;
224  if (0 <= index && index < NTRPLANES) {
225  data[index].insert( hit->get_detector_id()*1000 + hit->get_element_id() );
226  }
227  }
228  bool all_planes_have_hits = true;
229  for (int i = 0; i < NTRPLANES; ++i) {
230  if (data[i].empty()) {
231  all_planes_have_hits = false;
232  break;
233  }
234  }
235 
236  //do the tree DFS search
237  for(int i = 0; i < 2; ++i) {
238  path.clear();
239  roads_found[i].clear();
240  if (all_planes_have_hits) searchMatrix(matrix[i], 0, i);
241  }
242 
243  //FPGA singles trigger
244  int nPlusTop = 0;
245  int nPlusBot = 0;
246  int nMinusTop = 0;
247  int nMinusBot = 0;
248  int nHiPxPlusTop = 0;
249  int nHiPxPlusBot = 0;
250  int nHiPxMinusTop = 0;
251  int nHiPxMinusBot = 0;
252 
253  for (std::list<DPTriggerRoad>::iterator iter = roads_found[0].begin();
254  iter != roads_found[0].end(); ++iter) {
255  if (iter->getTB() > 0) {
256  ++nPlusTop;
257  if (iter->getPxMin() > 3.) ++nHiPxPlusTop;
258  } else {
259  ++nPlusBot;
260  if (iter->getPxMin() > 3.) ++nHiPxPlusBot;
261  }
262  }
263 
264  for (std::list<DPTriggerRoad>::iterator iter = roads_found[1].begin();
265  iter != roads_found[1].end(); ++iter) {
266  if (iter->getTB() > 0) {
267  ++nMinusTop;
268  if (iter->getPxMin() > 3.) ++nHiPxMinusTop;
269  } else {
270  ++nMinusBot;
271  if (iter->getPxMin() > 3.) ++nHiPxMinusBot;
272  }
273  }
274 
275  bool fpga1_on = (nPlusTop > 0 && nMinusBot > 0) || (nPlusBot > 0 && nMinusTop > 0);
276  bool fpga2_on = (nPlusTop > 0 && nMinusTop > 0) || (nPlusBot > 0 && nMinusBot > 0);
277  bool fpga3_on = (nPlusTop > 0 && nPlusBot > 0) || (nMinusTop > 0 && nMinusBot > 0);
278  bool fpga4_on = nPlusTop > 0 || nMinusTop > 0 || nPlusBot > 0 || nMinusBot > 0 ;
279  bool fpga5_on = nHiPxPlusTop > 0 || nHiPxMinusTop > 0 || nHiPxPlusBot > 0 || nHiPxMinusBot > 0;
280  _event_header_out->set_trigger(SQEvent::MATRIX1, fpga1_on);
281  _event_header_out->set_trigger(SQEvent::MATRIX2, fpga2_on);
282  _event_header_out->set_trigger(SQEvent::MATRIX3, fpga3_on);
283  _event_header_out->set_trigger(SQEvent::MATRIX4, fpga4_on);
284  _event_header_out->set_trigger(SQEvent::MATRIX5, fpga5_on);
285 
287 }
288 
291 }
292 
293 int DPTriggerAnalyzer::GetNodes(PHCompositeNode* topNode) {
294 
295  _event_header = findNode::getClass<SQEvent>(topNode, "SQEvent");
296  if (!_event_header) {
297  LogInfo("!_event_header");
299  }
300 
301  string name_hit_vec = _use_trig_hit ? "SQTriggerHitVector" : "SQHitVector";
302  _hit_vector = findNode::getClass<SQHitVector>(topNode, name_hit_vec);
303  if (!_hit_vector) {
304  LogInfo("!_hit_vector");
306  }
307 
308  if (_output_node_name == "") {
309  _event_header_out = _event_header;
310  } else { // Create and use a new node for output
311  _event_header_out = findNode::getClass<SQEvent>(topNode, _output_node_name);
312  if (!_event_header_out) {
313  PHNodeIterator iter(topNode);
314  PHCompositeNode* eventNode = static_cast<PHCompositeNode*>(iter.findFirst("PHCompositeNode", "DST"));
315  if(!eventNode) {
316  LogInfo("!eventNode (DST)");
318  }
319  _event_header_out = (SQEvent*)_event_header->Clone();
320  _event_header_out->Reset();
321  PHIODataNode<PHObject>* node = new PHIODataNode<PHObject>(_event_header_out, _output_node_name, "PHObject");
322  eventNode->addNode(node);
323  }
324  }
325 
327 }
328 
330 
334  for (int i = 0; i < 2; ++i) {
335  matrix[i] = new MatrixNode(-1);
336  for (std::map<TString, DPTriggerRoad>::iterator iter = roads[i].begin(); iter != roads[i].end(); ++iter) {
337  MatrixNode* parentNode[NTRPLANES + 1]; //NOTE: the last entry is useless, just to keep the following code simpler
338  parentNode[0] = matrix[i];
339  for (int j = 0; j < NTRPLANES; ++j) {
340  int uniqueID = iter->second.getTrID(j);
341  bool isNewNode = true;
342  for (std::list<MatrixNode*>::iterator jter = parentNode[j]->children.begin(); jter != parentNode[j]->children.end(); ++jter) {
343  if (uniqueID == (*jter)->uniqueID) {
344  parentNode[j + 1] = *jter;
345  isNewNode = false;
346 
347  break;
348  }
349  }
350 
351  if (isNewNode) {
352  MatrixNode* newNode = new MatrixNode(uniqueID);
353  parentNode[j]->add(newNode);
354  parentNode[j + 1] = newNode;
355  }
356  }
357  }
358  }
359 }
360 
363 
364 void DPTriggerAnalyzer::searchMatrix(MatrixNode* node, int level, int index) {
365  path.push_back(node->uniqueID);
366  if (node->children.empty()) {
367  //printPath();
368  DPTriggerRoad road_found(path);
369  if (roads[index].find(road_found.getStringID()) != roads[index].end())
370  roads_found[index].push_back(road_found);
371  path.pop_back();
372 
373  return;
374  }
375 
376  for (std::list<MatrixNode*>::iterator iter = node->children.begin(); iter != node->children.end(); ++iter) {
377  if (data[level].find((*iter)->uniqueID) == data[level].end())
378  continue;
379  searchMatrix(*iter, level + 1, index);
380  }
381  path.pop_back();
382 }
383 
385  if (node == NULL) return;
386 
387  for (std::list<MatrixNode*>::iterator iter = node->children.begin(); iter != node->children.end(); ++iter) {
388  deleteMatrix(*iter);
389  }
390 
391  delete node;
392 }
393 
395  for (unsigned int i = 1; i < NTRPLANES; ++i) {
396  std::cout << "Lv. " << i << ": ";
397  for (std::set<int>::iterator iter = data[i].begin(); iter != data[i].end(); ++iter) {
398  std::cout << *iter << " ";
399  }
400  std::cout << std::endl;
401  }
402 }
403 
405  std::cout << "Found one road: " << std::endl;
406  for (std::list<int>::iterator iter = path.begin(); iter != path.end(); ++iter) {
407  std::cout << *iter << " === ";
408  }
409  std::cout << std::endl;
410 }
411 
413 
415 {
416  children.clear();
417 }
418 
420 {
421  children.push_back(child);
422 }
std::ostream & operator<<(std::ostream &os, const DPTriggerRoad &road)
#define NTRPLANES
#define LogInfo(message)
Definition: DbSvc.cc:15
#define NULL
Definition: Pdb.h:9
void add(MatrixNode *child)
add a child
std::list< MatrixNode * > children
DPTriggerAnalyzer(const std::string &name="DPTriggerAnalyzer")
void deleteMatrix(MatrixNode *node)
Tree deletion.
int InitRun(PHCompositeNode *topNode)
void buildTriggerMatrix()
Build the trigger matrix by the input roads list.
void printHitPattern()
Helper functions to print various things.
void set_nim_mode(const NimMode nim1, const NimMode nim2)
int Init(PHCompositeNode *topNode)
int End(PHCompositeNode *topNode)
Called at the end of all processing.
int process_event(PHCompositeNode *topNode)
void searchMatrix(MatrixNode *node, int level, int index)
Test the trigger pattern.
void setPxMin(double pxmin)
int getTrDetectorID(unsigned int i) const
void setSigWeight(double weight)
int getTrElementID(unsigned int i) const
int getTB()
Get the sign of LR or TB.
void addTrElement(int uniqueID)
add one hit into the road
bool operator==(const DPTriggerRoad &elem) const
comparison
void flipTB()
flip the LR or TB
void setBkgRate(double rate)
void setRoadID(int id)
Sets.
bool operator<(const DPTriggerRoad &elem) const
User interface class about the geometry of detector planes.
Definition: GeomSvc.h:164
int getDetectorID(const std::string &detectorName) const
Get the plane position.
Definition: GeomSvc.h:219
static GeomSvc * instance()
singlton instance
Definition: GeomSvc.cxx:212
int getHodoStation(const int detectorID) const
Return a station number (1-4) for hodo planes or "0" for others.
Definition: GeomSvc.cxx:822
PHBoolean addNode(PHNode *)
An SQ interface class to hold one event header.
Definition: SQEvent.h:17
void Reset()
Clear Event.
Definition: SQEvent.h:36
@ MATRIX2
Definition: SQEvent.h:28
@ MATRIX1
Definition: SQEvent.h:27
@ NIM2
Definition: SQEvent.h:23
@ NIM1
Definition: SQEvent.h:22
@ MATRIX3
Definition: SQEvent.h:29
@ MATRIX4
Definition: SQEvent.h:30
@ MATRIX5
Definition: SQEvent.h:31
virtual void set_trigger(const SQEvent::TriggerMask i, const bool a)=0
virtual const SQHit * at(const size_t idkey) const =0
virtual size_t size() const =0
An SQ interface class to hold one detector hit.
Definition: SQHit.h:20
virtual bool is_in_time() const
Return 'true' if this hit is in the time window.
Definition: SQHit.h:90
virtual short get_element_id() const
Return the element ID of this hit.
Definition: SQHit.h:45
virtual short get_detector_id() const
Return the detector ID of this hit.
Definition: SQHit.h:42
static recoConsts * instance()
Definition: recoConsts.cc:7
virtual void set_CharFlag(const std::string &name, const std::string &flag)
overide the virtual function to expand the environmental variables
Definition: recoConsts.cc:21