6 #include <Geant4/G4ParticleDefinition.hh>
7 #include <Geant4/G4ParticleTable.hh>
18 #include <HepMC/GenEvent.h>
20 #include <gsl/gsl_const.h>
21 #include <gsl/gsl_randist.h>
22 #include <gsl/gsl_rng.h>
45 bool operator()(
const HepMC::GenParticle *p)
47 if (!p->end_vertex() && p->status() == 1)
return 1;
67 , _particle_filter_on(false)
68 , _position_filter_on(false)
69 , _pos_filter_x_min(0.0)
70 , _pos_filter_x_max(0.0)
71 , _pos_filter_y_min(0.0)
72 , _pos_filter_y_max(0.0)
73 , _pos_filter_z_min(0.0)
74 , _pos_filter_z_max(0.0)
76 RandomGenerator = gsl_rng_alloc(gsl_rng_mt19937);
77 _particle_filter_pid.clear();
83 gsl_rng_free(RandomGenerator);
88 PHG4InEvent *ineve = findNode::getClass<PHG4InEvent>(topNode,
"PHG4INEVENT");
104 cout <<
Name() <<
" override random seed: " << phseed << endl;
106 gsl_rng_set(RandomGenerator, phseed);
113 PHHepMCGenEventMap *genevtmap = findNode::getClass<PHHepMCGenEventMap>(topNode,
"PHHepMCGenEventMap");
117 static bool once =
true;
123 cout <<
"HepMCNodeReader::process_event - No PHHepMCGenEventMap node. Do not perform HepMC->Geant4 input" << endl;
129 PHG4InEvent *ineve = findNode::getClass<PHG4InEvent>(topNode,
"PHG4INEVENT");
132 cout <<
PHWHERE <<
"no PHG4INEVENT node" << endl;
150 if (worldshape ==
"G4Tubs")
152 ishape = ShapeG4Tubs;
154 else if (worldshape ==
"G4Box")
160 cout <<
PHWHERE <<
" unknown world shape " << worldshape << endl;
176 cout <<
"HepMCNodeReader::process_event - this event is already simulated. Move on: ";
183 HepMC::GenEvent *evt = genevt->
getEvent();
186 cout <<
PHWHERE <<
" no evt pointer under HEPMC Node found:";
201 xshift += smeargauss(width_vx);
202 else if (width_vx < 0.0)
203 xshift += smearflat(width_vx);
206 yshift += smeargauss(width_vy);
207 else if (width_vy < 0.0)
208 yshift += smearflat(width_vy);
211 zshift += smeargauss(width_vz);
212 else if (width_vz < 0.0)
213 zshift += smearflat(width_vz);
215 std::list<HepMC::GenParticle *> finalstateparticles;
216 std::list<HepMC::GenParticle *>::const_iterator fiter;
219 const double mom_factor = HepMC::Units::conversion_factor(evt->momentum_unit(), HepMC::Units::GEV);
220 const double length_factor = HepMC::Units::conversion_factor(evt->length_unit(), HepMC::Units::CM);
221 const double time_factor = HepMC::Units::conversion_factor(evt->length_unit(), HepMC::Units::CM) / GSL_CONST_CGS_SPEED_OF_LIGHT * 1e9;
223 for (HepMC::GenEvent::vertex_iterator v = evt->vertices_begin();
224 v != evt->vertices_end();
227 finalstateparticles.clear();
228 for (HepMC::GenVertex::particle_iterator p =
229 (*v)->particles_begin(HepMC::children);
230 p != (*v)->particles_end(HepMC::children); ++p)
234 if(_particle_filter_on) {
237 finalstateparticles.push_back(*p);
241 if (!finalstateparticles.empty())
243 double xpos = (*v)->position().x() * length_factor + xshift;
244 double ypos = (*v)->position().y() * length_factor + yshift;
245 double zpos = (*v)->position().z() * length_factor + zshift;
246 double time = (*v)->position().t() * time_factor + tshift;
250 cout <<
"Vertex : " << endl;
252 cout <<
"id: " << (*v)->barcode() << endl;
253 cout <<
"x: " << xpos << endl;
254 cout <<
"y: " << ypos << endl;
255 cout <<
"z: " << zpos << endl;
256 cout <<
"t: " << time << endl;
257 cout <<
"Particles" << endl;
260 if (_position_filter_on &&
261 (xpos < _pos_filter_x_min || xpos > _pos_filter_x_max ||
262 ypos < _pos_filter_y_min || ypos > _pos_filter_y_max ||
263 zpos < _pos_filter_z_min || zpos > _pos_filter_z_max ))
continue;
265 if (ishape == ShapeG4Tubs)
267 if (sqrt(xpos * xpos + ypos * ypos) > worldsizey / 2 ||
268 fabs(zpos) > worldsizez / 2)
272 cout <<
"vertex x/y/z" << xpos <<
"/" << ypos <<
"/" << zpos
273 <<
" outside world volume radius/z (+-) " << worldsizex / 2
274 <<
"/" << worldsizez / 2 <<
", dropping it and its particles"
280 else if (ishape == ShapeG4Box)
282 if (fabs(xpos) > worldsizex / 2 || fabs(ypos) > worldsizey / 2 ||
283 fabs(zpos) > worldsizez / 2)
287 cout <<
"Vertex x/y/z " << xpos <<
"/" << ypos <<
"/" << zpos
288 <<
" outside world volume x/y/z (+-) " << worldsizex / 2 <<
"/"
289 << worldsizey / 2 <<
"/" << worldsizez / 2
290 <<
", dropping it and its particles" << endl;
297 cout <<
PHWHERE <<
" shape " << ishape <<
" not implemented. exiting"
303 vtxindex = ineve->
AddVtx(xpos, ypos, zpos, time);
305 for (fiter = finalstateparticles.begin();
306 fiter != finalstateparticles.end();
314 particle->
set_pid((*fiter)->pdg_id());
315 particle->
set_px((*fiter)->momentum().px() * mom_factor);
316 particle->
set_py((*fiter)->momentum().py() * mom_factor);
317 particle->
set_pz((*fiter)->momentum().pz() * mom_factor);
324 double FMAG_hole_R = 2.5;
325 double FMAG_hole_L = 25.;
327 double Transverse_pos = sqrt(pow(xpos,2)+pow(ypos,2));
328 double FMAG_X = 160.;
329 double FMAG_Y = 160.;
330 double FMAG_Z = 503.;
332 double pion_lambda_Fe = 20.42;
333 double pion_lambda_Cu = 18.51;
334 double pion_lambda_B = 48.75;
335 double pion_lambda_sh = 55.92;
338 if(fabs((*fiter)->momentum().px()* mom_factor/((*fiter)->momentum().pz()* mom_factor))>_pxy2pz_rat)
continue;
339 if(fabs((*fiter)->momentum().py()* mom_factor/((*fiter)->momentum().pz()* mom_factor))>_pxy2pz_rat)
continue;
340 if((*fiter)->momentum().pz()<0)
continue;
343 if(fabs(xpos)>FMAG_X/2.)
continue;
344 if(fabs(ypos)>FMAG_Y/2.)
continue;
349 double coll_y = 3.48;
350 double totMom =sqrt(pow((*fiter)->momentum().px() * mom_factor,2)+pow((*fiter)->momentum().py() * mom_factor,2)+pow((*fiter)->momentum().pz() * mom_factor,2));
358 if (gRandom->Uniform(0,1) > exp (-(-genevt->
get_collision_vertex().z()+zpos)/(pion_lambda_Cu)))
continue;
362 if (gRandom->Uniform(0,1) > exp (-(-genevt->
get_collision_vertex().z()-542.)/(pion_lambda_Cu)))
continue;
373 if (gRandom->Uniform(0,1) > exp (-(-genevt->
get_collision_vertex().z()+zpos)/pion_lambda_sh))
continue;
377 if (gRandom->Uniform(0,1) > exp (-(171.5)/pion_lambda_sh))
continue;
385 if(zpos>-171.5 && zpos<-0.001)
387 if (gRandom->Uniform(0,1) > exp (-(171.5+zpos)/pion_lambda_sh))
continue;
391 if (gRandom->Uniform(0,1) > exp (-(171.5)/pion_lambda_sh))
continue;
401 if (gRandom->Uniform(0,1)> exp (-(zpos-genevt->
get_collision_vertex().z())/pion_lambda_Fe))
continue;
407 if (Transverse_pos> FMAG_hole_R)
409 if (gRandom->Uniform(0,1) > exp (-zpos/pion_lambda_Fe))
continue;
413 if (gRandom->Uniform(0,1) > exp (-(zpos-FMAG_hole_L)/pion_lambda_Fe))
continue;
419 if (zpos<0. && totMom/(FMAG_Z)<0.01)
continue;
420 if (zpos>0. && totMom/(FMAG_Z-zpos)<0.01)
continue;
439 double HepMCNodeReader::smeargauss(
const double width)
441 if (width == 0)
return 0;
442 return gsl_ran_gaussian(RandomGenerator, width);
448 _position_filter_on =
true;
449 _pos_filter_x_min = x_min;
450 _pos_filter_x_max = x_max;
451 _pos_filter_y_min = y_min;
452 _pos_filter_y_max = y_max;
453 _pos_filter_z_min = z_min;
454 _pos_filter_z_max = z_max;
458 for(
int pid : _particle_filter_pid) {
459 if(p->pdg_id() == pid)
return true;
464 double HepMCNodeReader::smearflat(
const double width)
466 if (width == 0)
return 0;
467 return 2.0 * width * (gsl_rng_uniform_pos(RandomGenerator) - 0.5);
473 cout <<
"HepMCNodeReader::VertexPosition - WARNING - this function is depreciated. "
474 <<
"HepMCNodeReader::VertexPosition() move all HEPMC subevents to a new vertex location. "
475 <<
"This also leads to a different vertex is used for HepMC subevent in Geant4 than that recorded in the HepMCEvent Node."
476 <<
"Recommendation: the vertex shifts are better controlled for individually HEPMC subevents in Fun4AllHepMCInputManagers and event generators."
488 cout <<
"HepMCNodeReader::SmearVertex - WARNING - this function is depreciated. "
489 <<
"HepMCNodeReader::SmearVertex() smear each HEPMC subevents to a new vertex location. "
490 <<
"This also leads to a different vertex is used for HepMC subevent in Geant4 than that recorded in the HepMCEvent Node."
491 <<
"Recommendation: the vertex smears are better controlled for individually HEPMC subevents in Fun4AllHepMCInputManagers and event generators."
502 cout <<
"HepMCNodeReader::Embed - WARNING - this function is depreciated. "
503 <<
"Embedding IDs are controlled for individually HEPMC subevents in Fun4AllHepMCInputManagers and event generators."
static IsStateFinal isfinal
int verbosity
The verbosity level. 0 means not verbose at all.
virtual const std::string Name() const
Returns the name of this module.
virtual int Verbosity() const
Gets the verbosity of this module.
void VertexPosition(const double v_x, const double v_y, const double v_z)
void enable_position_filter(const double x_min, const double x_max, const double y_min, const double y_max, const double z_min, const double z_max)
Enable and define a position filter. The x,y,z limits are in cm.
void SmearVertex(const double s_x, const double s_y, const double s_z)
int Init(PHCompositeNode *topNode)
bool PassParticleFilter(HepMC::GenParticle *p)
void Embed(const int i=1)
virtual ~HepMCNodeReader()
int process_event(PHCompositeNode *topNode)
HepMCNodeReader(const std::string &name="HEPMCREADER")
PHBoolean addNode(PHNode *)
virtual const std::string get_CharFlag(const std::string &flag) const
virtual float get_FloatFlag(const std::string &name) const
int AddParticle(const int vtxid, PHG4Particle *particle)
int AddVtx(const double x, const double y, const double z, const double t)
virtual void identify(std::ostream &os=std::cout) const
void AddEmbeddedParticle(PHG4Particle *particle, int flag)
virtual void set_pid(const int i)
virtual void set_barcode(const int barcode)
virtual void set_py(const double x)
virtual void set_px(const double x)
virtual void set_pz(const double x)
PHHepMCGenEventMap is collection of HEPMC events input into this simulation map of embedding ID -> PH...
ConstReverseIter rend() const
ConstReverseIter rbegin() const
iterator from lowest ID to highest, i.e. signal to background
std::map< int, PHHepMCGenEvent * >::reverse_iterator ReverseIter
virtual HepMC::GenEvent * getEvent()
const HepMC::FourVector & get_collision_vertex() const
collision vertex position in the Hall coordinate system, use PHENIX units of cm, ns
int get_embedding_id() const
virtual void identify(std::ostream &os=std::cout) const
bool is_simulated() const
whether this event has been processed in Geant4 simulation
PHNode * findFirst(const std::string &, const std::string &)
static recoConsts * instance()