Class Reference for E1039 Core & Analysis Software
TruthNodeMaker.cc
Go to the documentation of this file.
1 #include <iomanip>
2 #include <list>
6 #include <g4main/PHG4Hit.h>
7 #include <g4main/PHG4Particle.h>
8 #include <g4main/PHG4VtxPoint.h>
18 #include <phool/PHNodeIterator.h>
19 #include <phool/PHIODataNode.h>
20 #include <phool/getClass.h>
21 #include <geom_svc/GeomSvc.h>
22 #include <ktracker/SRecEvent.h>
23 #include <GlobalConsts.h>
24 #include "TruthNodeMaker.h"
25 using namespace std;
26 
28  : SubsysReco("TruthNodeMaker")
29  , genevtmap(nullptr)
30  , g4true(nullptr)
31  , m_vec_hit(nullptr)
32  , m_rec_evt(nullptr)
33  , m_vec_rec_trk(nullptr)
34  , m_evt(nullptr)
35  , m_mcevt(nullptr)
36  , m_vec_trk(nullptr)
37  , m_vec_dim(nullptr)
38  , m_job_id(-1)
39  , m_legacy_rec_container(true)
40  , m_do_evt_header(true)
41  , m_do_truthtrk_tagging(true)
42  , m_matching_threshold(0.75)
43 {
44  for(int i = 0; i <= nChamberPlanes; ++i) {
45  m_g4hc[i] = nullptr;
46  }
47 }
48 
50 {
51  if (! m_evt ) delete m_evt;
52  if (! m_mcevt ) delete m_mcevt;
53  if (! m_vec_trk) delete m_vec_trk;
54  if (! m_vec_dim) delete m_vec_dim;
55 }
56 
58 {
60 }
61 
63 {
64  int ret = GetNodes(topNode);
65  if (ret != Fun4AllReturnCodes::EVENT_OK) return ret;
66  ret = MakeNodes(topNode);
67  if (ret != Fun4AllReturnCodes::EVENT_OK) return ret;
69 }
70 
72 {
74  if(m_do_evt_header) {
75  if (genevtmap->size() != 1) {
76  cout << "TruthNodeMaker::process_event(): size != 1 unexpectedly." << endl;
77  }
78 
79  for (PHHepMCGenEventMap::Iter iter = genevtmap->begin(); iter != genevtmap->end(); ++iter) {
80  PHHepMCGenEvent *genevt = iter->second;
81  if (! genevt) {
82  cout << "No PHHepMCGenEvent object." << endl;
83  //return Fun4AllReturnCodes::ABORTEVENT;
84  }
85  HepMC::GenEvent *evt = genevt->getEvent();
86  if (! evt) {
87  cout << "No HepMC::GenEvent object." << endl;
88  //return Fun4AllReturnCodes::ABORTEVENT;
89  }
90  m_evt->set_run_id (0);
91  m_evt->set_spill_id(0);
92  m_evt->set_event_id(evt->event_number());
93  m_mcevt->set_process_id(evt->signal_process_id());
94 
95  //HepMC::GenVertex* vtx = evt->signal_process_vertex(); // Return 0 as of 2019-11-19.
96  HepMC::GenEvent::particle_const_iterator it = evt->particles_begin();
97  it++; // Skip the 1st beam particle.
98  for (int iii = 0; iii < 4; iii++) {
99  it++;
100  const HepMC::GenParticle* par = *it;
101  const HepMC::FourVector * mom = &par->momentum();
102  TLorentzVector lvec;
103  lvec.SetPxPyPzE(mom->px(), mom->py(), mom->pz(), mom->e());
104  m_mcevt->set_particle_id(iii, par->pdg_id());
105  m_mcevt->set_particle_momentum(iii, lvec);
106  }
107  }
108  }
109 
110  if (m_job_id >= 0) {
111  m_evt->set_spill_id(m_job_id); // Divert the spill-ID field
112  //m_mcevt->set_job_id(m_job_id); // Better way in a future.
113  }
114 
116  m_vec_trk->clear();
117  map<int, int> trkid_idx;
118  vector<int> vec_vtx_id;
119  for (auto it = g4true->GetPrimaryParticleRange().first; it != g4true->GetPrimaryParticleRange().second; ++it) {
120  PHG4Particle* par = it->second;
121  int pid = par->get_pid();
122  if (abs(pid) != 13) continue; // not muon
123  int trk_id = par->get_track_id();
124  int vtx_id = par->get_vtx_id();
125  PHG4VtxPoint* vtx = g4true->GetVtx(vtx_id);
126 
127  SQTrack_v1 trk;
128  trk.set_track_id(trk_id);
129  trk.set_charge(pid < 0 ? +1: -1); // -13 = mu+
130  trk.set_pos_vtx(TVector3 (vtx->get_x (), vtx->get_y (), vtx->get_z ()));
131  trk.set_mom_vtx(TLorentzVector(par->get_px(), par->get_py(), par->get_pz(), par->get_e()));
132 
133  trkid_idx[trk_id] = m_vec_trk->size();
134  m_vec_trk->push_back(&trk);
135  vec_vtx_id.push_back(vtx_id);
136  }
137 
139  m_vec_dim->clear();
140  int id_dim = 0; // to be incremented
141  unsigned int n_trk = m_vec_trk->size();
142  for (unsigned int i1 = 0; i1 < n_trk; i1++) {
143  SQTrack* trk1 = m_vec_trk->at(i1);
144  if (trk1->get_charge() <= 0) continue;
145  for (unsigned int i2 = 0; i2 < n_trk; i2++) {
146  SQTrack* trk2 = m_vec_trk->at(i2);
147  if (trk2->get_charge() >= 0) continue;
148  if (vec_vtx_id[i1] != vec_vtx_id[i2]) continue;
149 
150  SQDimuon_v1 dim;
151  dim.set_dimuon_id(++id_dim);
152  //dim.set_pdg_id(par_dim->pdg_id()); // PDG ID is not accessible via PHG4Particle
153  dim.set_pos (trk1->get_pos_vtx());
154  dim.set_mom (trk1->get_mom_vtx() + trk2->get_mom_vtx());
155  dim.set_mom_pos (trk1->get_mom_vtx());
156  dim.set_mom_neg (trk2->get_mom_vtx());
157  dim.set_track_id_pos(trk1->get_track_id());
158  dim.set_track_id_neg(trk2->get_track_id());
159  m_vec_dim->push_back(&dim);
160  }
161  }
162 
164  map<int, vector<SQHit*> > trkid_hitvec;
165  int n_digihits = m_vec_hit->size();
166  for(int i = 0; i < n_digihits; ++i) {
167  SQHit* hit = m_vec_hit->at(i);
168 
169  int detid = hit->get_detector_id();
170  if(detid > nChamberPlanes || (detid >= 7 && detid <= 12)) continue;
171 
172  int trkid = hit->get_track_id();
173  if(trkid_idx.find(trkid) == trkid_idx.end()) continue;
174 
175  if(trkid_hitvec.find(trkid) == trkid_hitvec.end()) {
176  trkid_hitvec[trkid] = vector<SQHit*>();
177  }
178  trkid_hitvec[trkid].push_back(hit);
179  }
180 
182  for(unsigned int i = 0; i < n_trk; ++i) {
183  SQTrack* trk = m_vec_trk->at(i);
184 
185  int trkid = trk->get_track_id();
186  if(trkid_hitvec.find(trkid) == trkid_hitvec.end()) {
187  //this track does not have hits in detector
188  continue;
189  }
190  trk->set_num_hits(trkid_hitvec[trkid].size());
191 
192  TVector3 pos;
193  TLorentzVector mom;
194  int detIDs_st1[6] = {3, 4, 2, 5, 1, 6};
195  if(FindHitAtStation(detIDs_st1, trkid, trkid_hitvec[trkid], pos, mom)) {
196  trk->set_pos_st1(pos);
197  trk->set_mom_st1(mom);
198  }
199 
200  int detIDs_st3p[6] = {21, 22, 20, 23, 19, 24};
201  int detIDs_st3m[6] = {27, 28, 26, 29, 25, 30};
202  if(FindHitAtStation(detIDs_st3p, trkid, trkid_hitvec[trkid], pos, mom) ||
203  FindHitAtStation(detIDs_st3m, trkid, trkid_hitvec[trkid], pos, mom)) {
204  trk->set_pos_st3(pos);
205  trk->set_mom_st3(mom);
206  }
207  }
208 
210  if(!m_do_truthtrk_tagging) return Fun4AllReturnCodes::EVENT_OK;
211 
213  int n_rtrks = 0;
214  map<int, vector<int> > rtrkid_hitidvec;
215  if(m_legacy_rec_container) {
216  n_rtrks = m_rec_evt->getNTracks();
217  for(int i = 0; i < n_rtrks; i++) {
218  SRecTrack& recTrack = m_rec_evt->getTrack(i);
219  rtrkid_hitidvec[i] = vector<int>();
220 
221  int n_rhits = recTrack.getNHits();
222  for(int j = 0; j < n_rhits; ++j) {
223  rtrkid_hitidvec[i].push_back(recTrack.getHitIndex(j));
224  }
225 
226  sort(rtrkid_hitidvec[i].begin(), rtrkid_hitidvec[i].end());
227  }
228  } else {
229  n_rtrks = m_vec_rec_trk->size();
230  for(int i = 0; i < n_rtrks; ++i) {
231  SRecTrack* recTrack = dynamic_cast<SRecTrack*>(m_vec_rec_trk->at(i));
232  rtrkid_hitidvec[i] = vector<int>();
233 
234  int n_rhits = recTrack->getNHits();
235  for(int j = 0; j < n_rhits; ++j) {
236  rtrkid_hitidvec[i].push_back(recTrack->getHitIndex(j));
237  }
238 
239  sort(rtrkid_hitidvec[i].begin(), rtrkid_hitidvec[i].end());
240  }
241  }
242  if(n_rtrks <= 0) return Fun4AllReturnCodes::EVENT_OK;
243 
245  map<int, vector<int> > ttrkid_hitidvec;
246  for(auto it = trkid_hitvec.begin(); it != trkid_hitvec.end(); ++it) {
247  ttrkid_hitidvec[it->first] = vector<int>();
248  for(auto jt = it->second.begin(); jt != it->second.end(); ++jt) {
249  ttrkid_hitidvec[it->first].push_back((*jt)->get_hit_id());
250  }
251  sort(ttrkid_hitidvec[it->first].begin(), ttrkid_hitidvec[it->first].end());
252  }
253 
255  for(unsigned int i = 0; i < n_trk; ++i) {
256  SQTrack* trk = m_vec_trk->at(i);
257  int ttrkid = trk->get_track_id();
258 
259  int rtrkid = -1;
260  unsigned int n_match = 0;
261  for(auto it = rtrkid_hitidvec.begin(); it != rtrkid_hitidvec.end(); ++it) {
262  int n_match_new = FindCommonHitIDs(ttrkid_hitidvec[ttrkid], it->second);
263  if(n_match < n_match_new) {
264  n_match = n_match_new;
265  rtrkid = it->first;
266  }
267  }
268 
269  if(rtrkid >= 0 && double(n_match)/double(ttrkid_hitidvec[ttrkid].size()) > m_matching_threshold) {
270  trk->set_rec_track_id(rtrkid);
271  }
272  }
273 
275 }
276 
278 {
280 }
281 
282 int TruthNodeMaker::GetNodes(PHCompositeNode* topNode)
283 {
284  g4true = findNode::getClass<PHG4TruthInfoContainer>(topNode, "G4TruthInfo");
285  if(!g4true) {
286  cerr << Name() << ": failed locating G4TruthInfo, abort" << endl;
288  }
289 
290  m_vec_hit = findNode::getClass<SQHitVector>(topNode, "SQHitVector");
291  if(!m_vec_hit) {
292  cerr << Name() << ": failed locating the digitized hit vector, abort" << endl;
294  }
295 
296  genevtmap = findNode::getClass<PHHepMCGenEventMap>(topNode, "PHHepMCGenEventMap");
297  if(!genevtmap) {
298  m_do_evt_header = false;
299  cout << Name() << ": failed locating HepMCGenEvent, event process info will be missing" << endl;
300  }
301 
302  GeomSvc* p_geomSvc = GeomSvc::instance();
303  for(int i = 1; i <= nChamberPlanes; ++i)
304  {
305  string g4hitNodeName = "G4HIT_" + p_geomSvc->getDetectorName(i);
306  m_g4hc[i] = findNode::getClass<PHG4HitContainer>(topNode, g4hitNodeName);
307  }
308 
309  if(m_legacy_rec_container) {
310  m_rec_evt = findNode::getClass<SRecEvent>(topNode, "SRecEvent");
311  if(!m_rec_evt) {
312  m_do_truthtrk_tagging = false;
313  cout << Name() << ": failed locating rec event, no truth track tagging." << endl;
314  }
315  } else {
316  m_vec_rec_trk = findNode::getClass<SQTrackVector>(topNode, "SQRecTrackVector");
317  if(!m_vec_rec_trk) {
318  m_do_truthtrk_tagging = false;
319  cout << Name() << ": failed locating rec track vector, no truth track tagging." << endl;
320  }
321  }
322 
324 }
325 
326 int TruthNodeMaker::MakeNodes(PHCompositeNode* topNode)
327 {
328  PHNodeIterator iter(topNode);
329  PHCompositeNode* node_dst = static_cast<PHCompositeNode*>(iter.findFirst("PHCompositeNode", "DST"));
330  if (! node_dst) {
331  cout << "No DST node!?" << endl;
333  }
334 
335  m_evt = findNode::getClass<SQEvent>(topNode, "SQEvent");
336  if (! m_evt) {
337  m_evt = new SQEvent_v1();
338  node_dst->addNode(new PHIODataNode<PHObject>(m_evt, "SQEvent", "PHObject"));
339  }
340 
341  m_mcevt = findNode::getClass<SQMCEvent>(topNode, "SQMCEvent");
342  if (! m_mcevt) {
343  m_mcevt = new SQMCEvent_v1();
344  node_dst->addNode(new PHIODataNode<PHObject>(m_mcevt, "SQMCEvent", "PHObject"));
345  }
346 
347  m_vec_trk = findNode::getClass<SQTrackVector>(topNode, "SQTruthTrackVector");
348  if (! m_vec_trk) {
349  m_vec_trk = new SQTrackVector_v1();
350  node_dst->addNode(new PHIODataNode<PHObject>(m_vec_trk, "SQTruthTrackVector", "PHObject"));
351  }
352 
353  m_vec_dim = findNode::getClass<SQDimuonVector>(topNode, "SQTruthDimuonVector");
354  if (! m_vec_dim) {
355  m_vec_dim = new SQDimuonVector_v1();
356  node_dst->addNode(new PHIODataNode<PHObject>(m_vec_dim, "SQTruthDimuonVector", "PHObject"));
357  }
358 
360 }
361 
362 bool TruthNodeMaker::FindHitAtStation(int target_detIDs[], int trkid, const vector<SQHit*>& hitvec, TVector3& pos, TLorentzVector& mom)
363 {
364  for(int i = 0; i < 6; ++i) {
365  //We try to find the wanted hit from SQHitVector first
366  for(unsigned int j = 0; j < hitvec.size(); ++j) {
367  if(hitvec[j]->get_detector_id() == target_detIDs[i]) {
368  pos.SetXYZ(hitvec[j]->get_truth_x(), hitvec[j]->get_truth_y(), hitvec[j]->get_truth_z());
369  mom.SetXYZM(hitvec[j]->get_truth_px(), hitvec[j]->get_truth_py(), hitvec[j]->get_truth_pz(), M_MU);
370  return true;
371  }
372  }
373 
374  //if failed, we resort to the default solution of G4HIT list
375  if(!m_g4hc[target_detIDs[i]]) continue;
376  PHG4HitContainer::ConstRange range = m_g4hc[target_detIDs[i]]->getHits();
377  for(PHG4HitContainer::ConstIterator it = range.first; it != range.second; it++) {
378  PHG4Hit* hit = it->second;
379  if(hit->get_trkid() == trkid) {
380  pos.SetXYZ (hit->get_x(0) , hit->get_y(0) , hit->get_z(0) );
381  mom.SetXYZM(hit->get_px(0), hit->get_py(0), hit->get_pz(0), M_MU);
382  return true;
383  }
384  }
385  }
386  return false;
387 }
388 
389 int TruthNodeMaker::FindCommonHitIDs(vector<int>& hitidvec1, vector<int>& hitidvec2)
390 {
391  //This function assumes the input vectors have been sorted
392  auto iter = hitidvec1.begin();
393  auto jter = hitidvec2.begin();
394 
395  int nCommon = 0;
396  while(iter != hitidvec1.end() && jter != hitidvec2.end()) {
397  if(*iter < *jter) {
398  ++iter;
399  } else {
400  if(!(*jter < *iter)) {
401  ++nCommon;
402  ++iter;
403  }
404  ++jter;
405  }
406  }
407 
408  return nCommon;
409 }
#define nChamberPlanes
Definition: GlobalConsts.h:6
#define M_MU
Definition: GlobalConsts.h:12
virtual const std::string Name() const
Returns the name of this module.
Definition: Fun4AllBase.h:23
User interface class about the geometry of detector planes.
Definition: GeomSvc.h:164
static GeomSvc * instance()
singlton instance
Definition: GeomSvc.cxx:212
std::string getDetectorName(const int &detectorID) const
Definition: GeomSvc.h:223
PHBoolean addNode(PHNode *)
Map::const_iterator ConstIterator
std::pair< ConstIterator, ConstIterator > ConstRange
virtual float get_py(const int i) const
Definition: PHG4Hit.h:25
virtual float get_z(const int i) const
Definition: PHG4Hit.h:23
virtual float get_pz(const int i) const
Definition: PHG4Hit.h:26
virtual float get_px(const int i) const
Definition: PHG4Hit.h:24
virtual float get_y(const int i) const
Definition: PHG4Hit.h:22
virtual float get_x(const int i) const
Definition: PHG4Hit.h:21
virtual int get_trkid() const
Definition: PHG4Hit.h:41
virtual int get_track_id() const
Definition: PHG4Particle.h:21
virtual int get_pid() const
Definition: PHG4Particle.h:14
virtual double get_px() const
Definition: PHG4Particle.h:16
virtual int get_vtx_id() const
Definition: PHG4Particle.h:22
virtual double get_py() const
Definition: PHG4Particle.h:17
virtual double get_e() const
Definition: PHG4Particle.h:19
virtual double get_pz() const
Definition: PHG4Particle.h:18
PHG4VtxPoint * GetVtx(const int vtxid)
virtual double get_y() const
Definition: PHG4VtxPoint.h:21
virtual double get_x() const
Definition: PHG4VtxPoint.h:20
virtual double get_z() const
Definition: PHG4VtxPoint.h:22
std::map< int, PHHepMCGenEvent * >::iterator Iter
ConstIter end() const
ConstIter begin() const
iterator from lowest ID to highest, i.e. background to signal
virtual HepMC::GenEvent * getEvent()
virtual void clear()=0
virtual void push_back(const SQDimuon *dim)=0
void set_dimuon_id(const int a)
Definition: SQDimuon_v1.h:16
void set_track_id_pos(const int a)
Definition: SQDimuon_v1.h:25
void set_mom_pos(const TLorentzVector a)
Definition: SQDimuon_v1.h:37
void set_mom_neg(const TLorentzVector a)
Definition: SQDimuon_v1.h:40
void set_track_id_neg(const int a)
Definition: SQDimuon_v1.h:28
void set_pos(const TVector3 a)
Definition: SQDimuon_v1.h:31
void set_mom(const TLorentzVector a)
Definition: SQDimuon_v1.h:34
virtual void set_spill_id(const int a)=0
virtual void set_event_id(const int a)=0
virtual void set_run_id(const int 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 short get_detector_id() const
Return the detector ID of this hit.
Definition: SQHit.h:42
virtual int get_track_id() const
Return the track ID associated with this hit. Probably the value is not properly set at present.
Definition: SQHit.h:66
virtual void set_particle_momentum(const int i, const TLorentzVector a)=0
virtual void set_particle_id(const int i, const int a)=0
virtual void set_process_id(const int a)=0
virtual void push_back(const SQTrack *trk)=0
virtual const SQTrack * at(const size_t id) const =0
virtual size_t size() const =0
virtual void clear()=0
virtual void set_pos_vtx(const TVector3 a)
Definition: SQTrack_v1.h:28
virtual void set_charge(const int a)
Definition: SQTrack_v1.h:22
virtual void set_track_id(const int a)
Definition: SQTrack_v1.h:16
virtual void set_mom_vtx(const TLorentzVector a)
Definition: SQTrack_v1.h:37
An SQ interface class to hold one true or reconstructed track.
Definition: SQTrack.h:8
virtual int get_track_id() const =0
Return the track ID, which is unique per event(?).
virtual TVector3 get_pos_vtx() const =0
Return the track position at vertex.
virtual int get_charge() const =0
Return the charge, i.e. +1 or -1.
virtual void set_num_hits(const int a)=0
virtual TLorentzVector get_mom_vtx() const =0
Return the track momentum at vertex.
virtual void set_pos_st1(const TVector3 a)=0
virtual void set_pos_st3(const TVector3 a)=0
virtual void set_mom_st1(const TLorentzVector a)=0
virtual void set_rec_track_id(const int a)=0
virtual void set_mom_st3(const TLorentzVector a)=0
Int_t getNTracks()
Get tracks.
Definition: SRecEvent.h:455
SRecTrack & getTrack(Int_t i)
Definition: SRecEvent.h:456
Int_t getNHits() const
Definition: SRecEvent.h:102
Int_t getHitIndex(Int_t i)
Definition: SRecEvent.h:108
int process_event(PHCompositeNode *topNode)
int InitRun(PHCompositeNode *topNode)
int End(PHCompositeNode *topNode)
Called at the end of all processing.
virtual ~TruthNodeMaker()
int Init(PHCompositeNode *topNode)