Class Reference for E1039 Core & Analysis Software
SQSingleMuonGen.cxx
Go to the documentation of this file.
1 #include "SQSingleMuonGen.h"
2 
4 #include <g4main/PHG4InEvent.h>
7 
9 #include <phool/getClass.h>
10 
11 #include <phool/PHCompositeNode.h>
12 #include <phool/PHIODataNode.h>
13 #include <phool/PHRandomSeed.h>
14 
15 #include <Geant4/G4ParticleTable.hh>
16 #include <Geant4/G4ParticleDefinition.hh>
17 
18 #include <TMath.h>
19 #include <TGenPhaseSpace.h>
20 
21 #include <cstdlib>
22 #include <cmath>
23 #include <cassert>
24 
25 SQSingleMuonGen::SQSingleMuonGen(const std::string& name):
27  _ineve(nullptr),
28  _truth(nullptr),
29  _target_only(true),
30  _enable_geom_cut(true),
31  _enable_real_mom_dist(true),
32  _ctau(2.6E-8*29979245800.),
33  _mom_min(1.),
34  _mom_max(100.),
35  _pt_max(4.),
36  _charge_ratio(0.5),
37  _m_muon(0.105658),
38  _m_pion(0.13957),
39  _pt_dist(nullptr),
40  _rndm(PHRandomSeed())
41 {
42  for(int i = 0; i < 6; ++i) _pz_dist[i] = nullptr;
43 }
44 
46 {
47  PHNodeIterator iter(topNode);
48  PHCompositeNode* dstNode = dynamic_cast<PHCompositeNode*>(iter.findFirst("PHCompositeNode", "DST"));
49 
50  _truth = new SQSingleMuonTruthInfo();
51  PHIODataNode<PHObject>* truthNode = new PHIODataNode<PHObject>(_truth, "TruthInfo", "PHObject");
52  dstNode->addNode(truthNode);
53 
54  _ineve = findNode::getClass<PHG4InEvent>(topNode, "PHG4INEVENT");
55  if(!_ineve)
56  {
57  _ineve = new PHG4InEvent();
58  PHIODataNode<PHObject>* eveNode = new PHIODataNode<PHObject>(_ineve, "PHG4INEVENT", "PHObject");
59  dstNode->addNode(eveNode);
60  }
61 
62  _pt_dist = new TF1("PtDistr", "TMath::Landau(x, 0.416742, 0.174227)", 0., _pt_max);
63  _pz_dist[0] = new TF1("PzDistr1", "0.161*(x^-2.396)", _mom_min, _mom_max);
64  _pz_dist[1] = new TF1("PzDistr2", "0.527*(x^-1.996)", _mom_min, _mom_max);
65  _pz_dist[2] = new TF1("PzDistr3", "0.489*(x^-2.200)", _mom_min, _mom_max);
66  _pz_dist[3] = new TF1("PzDistr4", "0.302*(x^-1.900)", _mom_min, _mom_max);
67  _pz_dist[4] = new TF1("PzDistr5", "0.174*(x^-1.750)", _mom_min, _mom_max);
68  _pz_dist[5] = new TF1("PzDistr6", "0.909*(x^-1.850)", _mom_min > 8. ? _mom_min : 8., _mom_max);
69 
71 }
72 
74 {
75  for(int i = 0; i < 6; ++i)
76  {
77  if(_pz_dist[i] != nullptr) delete _pz_dist[i];
78  }
79  if(_pt_dist != nullptr) delete _pt_dist;
80 }
81 
83 {
84  int nTries = 0;
85  bool accepted = false;
86  while(!accepted)
87  {
88  //Generate mother particle first
89  generatePrimaryVtx();
90  generateMotherParticle();
91 
92  //Decay the mother particle, and get the muon's vtx and mom
93  decayMotherParticle();
94 
95  accepted = _enable_geom_cut ? geometryCut() : true;
96  ++nTries;
97  }
98 
99  if(Verbosity() > 10) std::cout << "Generated single muon within acceptance after " << nTries << " tries." << std::endl;
100 
101  int pdgcode = _truth->motherPid > 0 ? 13 : -13;
102  std::string pdgname = get_pdgname(pdgcode);
103 
104  //Generate a muon and set common features
105  int vtxID = _ineve->AddVtx(_truth->muVtx.X(), _truth->muVtx.Y(), _truth->muVtx.Z(), 0.);
106  PHG4Particle* particle = new PHG4Particlev2();
107  particle->set_track_id(0);
108  particle->set_vtx_id(vtxID);
109  particle->set_parent_id(0);
110  particle->set_pid(pdgcode);
111  particle->set_name(pdgname);
112 
113  double e = TMath::Sqrt(_truth->muMom.Mag2() + _m_muon*_m_muon);
114  particle->set_px(_truth->muMom.X());
115  particle->set_py(_truth->muMom.Y());
116  particle->set_pz(_truth->muMom.Z());
117  particle->set_e(e);
118  _ineve->AddParticle(vtxID, particle); //ownership of particle is transferred to _ineve and released in its Reset action
119 
121  {
122  _ineve->identify();
123  }
124 
126 }
127 
128 void SQSingleMuonGen::generateMotherParticle()
129 {
130  int pid = _rndm.Rndm() < _charge_ratio ? 211 : -211;
131  std::string name = get_pdgname(pid);
132 
133  double pz, pt;
134  if(_enable_real_mom_dist)
135  {
136  pt = _pt_dist->GetRandom();
137  if(pt < 0.1)
138  {
139  pz = _pz_dist[0]->GetRandom();
140  }
141  else if(pt < 0.2)
142  {
143  pz = _pz_dist[1]->GetRandom();
144  }
145  else if(pt < 0.3)
146  {
147  pz = _pz_dist[2]->GetRandom();
148  }
149  else if(pt < 0.4)
150  {
151  pz = _pz_dist[3]->GetRandom();
152  }
153  else if(pt < 0.5)
154  {
155  pz = _pz_dist[4]->GetRandom();
156  }
157  else
158  {
159  pz = _pz_dist[5]->GetRandom();
160  }
161  }
162  else
163  {
164  pz = _mom_min + (_mom_max - _mom_min)*_rndm.Rndm();
165  pt = _rndm.Rndm()*_pt_max;
166  }
167 
168  double phi = _rndm.Rndm()*TMath::TwoPi();
169  double px = pt*TMath::Cos(phi);
170  double py = pt*TMath::Sin(phi);
171 
172  _truth->motherPid = pid;
173  _truth->motherMom = TVector3(px, py, pz);
174 }
175 
176 void SQSingleMuonGen::decayMotherParticle()
177 {
178  TGenPhaseSpace evtgen;
179  double m[2] = {_m_muon, 0.};
180 
181  TLorentzVector W;
182  W.SetVectM(_truth->motherMom, _m_pion);
183  evtgen.SetDecay(W, 2, m);
184  evtgen.Generate();
185 
186  TLorentzVector* p = evtgen.GetDecay(0);
187  _truth->muMom = p->Vect();
188 
189  // Get the decay vector
190  double ctau1 = _ctau*W.Gamma();
191  _truth->decayLength = _rndm.Exp(ctau1);
192  _truth->muVtx = _truth->decayLength*_truth->muMom.Unit() + _truth->motherVtx;
193 }
194 
195 bool SQSingleMuonGen::geometryCut()
196 {
197  if(fabs(_truth->muMom.Px()/_truth->muMom.Pz()) > 0.15) return false;
198  if(fabs(_truth->muMom.Py()/_truth->muMom.Pz()) > 0.15) return false;
199  if(_truth->muMom.Mag() < 10.) return false;
200 
201  if(_truth->decayLength > 300.) return false;
202  return true;
203 }
204 
205 bool SQSingleMuonGen::generatePrimaryVtx()
206 {
207  double x = 0.;
208  double y = 0.;
209  double z = -300.;
210 
211  _truth->motherVtx.SetXYZ(x, y, z);
212  return true;
213 }
@ VERBOSITY_A_LOT
Output a lot of messages.
Definition: Fun4AllBase.h:48
virtual int Verbosity() const
Gets the verbosity of this module.
Definition: Fun4AllBase.h:64
PHBoolean addNode(PHNode *)
int AddParticle(const int vtxid, PHG4Particle *particle)
Definition: PHG4InEvent.cc:63
int AddVtx(const double x, const double y, const double z, const double t)
Definition: PHG4InEvent.cc:38
virtual void identify(std::ostream &os=std::cout) const
Definition: PHG4InEvent.cc:134
std::string get_pdgname(const int pdgcode) const
virtual void set_parent_id(const int i)
Definition: PHG4Particle.h:30
virtual void set_vtx_id(const int i)
Definition: PHG4Particle.h:29
virtual void set_pid(const int i)
Definition: PHG4Particle.h:33
virtual void set_e(const double e)
Definition: PHG4Particle.h:37
virtual void set_py(const double x)
Definition: PHG4Particle.h:35
virtual void set_track_id(const int i)
Definition: PHG4Particle.h:28
virtual void set_name(const std::string &name)
Definition: PHG4Particle.h:32
virtual void set_px(const double x)
Definition: PHG4Particle.h:34
virtual void set_pz(const double x)
Definition: PHG4Particle.h:36
PHNode * findFirst(const std::string &, const std::string &)
virtual ~SQSingleMuonGen()
SQSingleMuonGen(const std::string &name="SINGLEGEN")
int InitRun(PHCompositeNode *topNode)
int process_event(PHCompositeNode *topNode)