Class Reference for E1039 Core & Analysis Software
PHG4TargetCoilV2SteppingAction.cc
Go to the documentation of this file.
3 #include "PHG4StepStatusDecode.h"
4 
5 #include <phparameter/PHParameters.h>
6 
7 #include <g4main/PHG4Hit.h>
9 #include <g4main/PHG4Hitv1.h>
10 #include <g4main/PHG4Shower.h>
11 
13 
14 #include <phool/getClass.h>
15 
16 #include <Geant4/G4Step.hh>
17 #include <Geant4/G4SystemOfUnits.hh>
18 #include <Geant4/G4UserLimits.hh>
19 
20 #include <iomanip>
21 #include <iostream>
22 
23 using namespace std;
24 //____________________________________________________________________________..
26  : detector_(detector)
27  , params(parameters)
28  , hits_(nullptr)
29  , hit(nullptr)
30  , saveshower(nullptr)
31  , savevolpre(nullptr)
32  , savevolpost(nullptr)
33  , save_light_yield(params->get_int_param("lightyield"))
34  , savetrackid(-1)
35  , saveprestepstatus(-1)
36  , savepoststepstatus(-1)
37  , active(params->get_int_param("active"))
38  , IsBlackHole(params->get_int_param("blackhole"))
39  , use_g4_steps(params->get_int_param("use_g4steps"))
40  , zmin(params->get_double_param("place_z") * cm - params->get_double_param("length") * cm / 2.)
41  , zmax(params->get_double_param("place_z") * cm + params->get_double_param("length") * cm / 2.)
42  , tmin(params->get_double_param("tmin") * ns)
43  , tmax(params->get_double_param("tmax") * ns)
44 {
45  // G4 seems to have issues in the um range
46  zmin -= copysign(zmin, 1. / 1e6 * cm);
47  zmax += copysign(zmax, 1. / 1e6 * cm);
48 }
49 
51 {
52  // if the last hit was a zero energie deposit hit, it is just reset
53  // and the memory is still allocated, so we need to delete it here
54  // if the last hit was saved, hit is a nullptr pointer which are
55  // legal to delete (it results in a no operation)
56  delete hit;
57 }
58 
59 //____________________________________________________________________________..
61 {
62  // get volume of the current step
63  G4TouchableHandle touch = aStep->GetPreStepPoint()->GetTouchableHandle();
64  G4TouchableHandle touchpost = aStep->GetPostStepPoint()->GetTouchableHandle();
65 
66  G4VPhysicalVolume* volume = touch->GetVolume();
67  // G4 just calls UserSteppingAction for every step (and we loop then over all our
68  // steppingactions. First we have to check if we are actually in our volume
69  if (!detector_->IsInCylinder(volume))
70  {
71  return false;
72  }
73 
74  // collect energy and track length step by step
75  G4double edep = aStep->GetTotalEnergyDeposit() / GeV;
76 
77  const G4Track* aTrack = aStep->GetTrack();
78 
79  // if this cylinder stops everything, just put all kinetic energy into edep
80  if (IsBlackHole)
81  {
82  if ((!isfinite(tmin) && !isfinite(tmax)) ||
83  aTrack->GetGlobalTime() < tmin ||
84  aTrack->GetGlobalTime() > tmax)
85  {
86  edep = aTrack->GetKineticEnergy() / GeV;
87  G4Track* killtrack = const_cast<G4Track*>(aTrack);
88  killtrack->SetTrackStatus(fStopAndKill);
89  }
90  }
91 
92  int layer_id = detector_->get_Layer();
93  // test if we are active
94  if (active)
95  {
96  bool geantino = false;
97  // the check for the pdg code speeds things up, I do not want to make
98  // an expensive string compare for every track when we know
99  // geantino or chargedgeantino has pid=0
100  if (aTrack->GetParticleDefinition()->GetPDGEncoding() == 0 &&
101  aTrack->GetParticleDefinition()->GetParticleName().find("geantino") != string::npos)
102  {
103  geantino = true;
104  }
105  G4StepPoint* prePoint = aStep->GetPreStepPoint();
106  G4StepPoint* postPoint = aStep->GetPostStepPoint();
107  // cout << "time prepoint: " << prePoint->GetGlobalTime()/ns << endl;
108  // cout << "time postpoint: " << postPoint->GetGlobalTime()/ns << endl;
109  // cout << "kinetic energy: " << aTrack->GetKineticEnergy()/GeV << endl;
110  // G4ParticleDefinition* def = aTrack->GetDefinition();
111  // cout << "Particle: " << def->GetParticleName() << endl;
112  int prepointstatus = prePoint->GetStepStatus();
113  if (prepointstatus == fGeomBoundary ||
114  prepointstatus == fUndefined ||
115  (prepointstatus == fPostStepDoItProc && savepoststepstatus == fGeomBoundary) ||
116  use_g4_steps > 0)
117  {
118  if (prepointstatus == fPostStepDoItProc && savepoststepstatus == fGeomBoundary)
119  {
120  cout << GetName() << ": New Hit for " << endl;
121  cout << "prestep status: " << PHG4StepStatusDecode::GetStepStatus(prePoint->GetStepStatus())
122  << ", poststep status: " << PHG4StepStatusDecode::GetStepStatus(postPoint->GetStepStatus())
123  << ", last pre step status: " << PHG4StepStatusDecode::GetStepStatus(saveprestepstatus)
124  << ", last post step status: " << PHG4StepStatusDecode::GetStepStatus(savepoststepstatus) << endl;
125  cout << "last track: " << savetrackid
126  << ", current trackid: " << aTrack->GetTrackID() << endl;
127  cout << "phys pre vol: " << volume->GetName()
128  << " post vol : " << touchpost->GetVolume()->GetName() << endl;
129  cout << " previous phys pre vol: " << savevolpre->GetName()
130  << " previous phys post vol: " << savevolpost->GetName() << endl;
131  }
132 
133  if (!hit)
134  {
135  hit = new PHG4Hitv1();
136  }
137 
138  int elem_id = detector_->get_elem_id(volume->GetName());
139 
140  hit->set_layer((unsigned int) layer_id*100 + elem_id);
141 
142  //here we set the entrance values in cm
143  hit->set_x(0, prePoint->GetPosition().x() / cm);
144  hit->set_y(0, prePoint->GetPosition().y() / cm);
145  hit->set_z(0, prePoint->GetPosition().z() / cm);
146 
147  hit->set_px(0, prePoint->GetMomentum().x() / GeV);
148  hit->set_py(0, prePoint->GetMomentum().y() / GeV);
149  hit->set_pz(0, prePoint->GetMomentum().z() / GeV);
150 
151  // time in ns
152  hit->set_t(0, prePoint->GetGlobalTime() / nanosecond);
153  //set and save the track ID
154  hit->set_trkid(aTrack->GetTrackID());
155  savetrackid = aTrack->GetTrackID();
156  //set the initial energy deposit
157  hit->set_edep(0);
158  if (!geantino && !IsBlackHole)
159  {
160  hit->set_eion(0);
161  }
162  if (save_light_yield)
163  {
164  hit->set_light_yield(0);
165  }
166  if (G4VUserTrackInformation* p = aTrack->GetUserInformation())
167  {
168  if (PHG4TrackUserInfoV1* pp = dynamic_cast<PHG4TrackUserInfoV1*>(p))
169  {
170  hit->set_trkid(pp->GetUserTrackId());
171  hit->set_shower_id(pp->GetShower()->get_id());
172  saveshower = pp->GetShower();
173  }
174  }
175 
176  if (hit->get_z(0) * cm > zmax || hit->get_z(0) * cm < zmin)
177  {
178  cout << detector_->SuperDetector() << std::setprecision(9)
179  << "PHG4TargetCoilV2SteppingAction: Entry hit z " << hit->get_z(0) * cm
180  << " outside acceptance, zmin " << zmin
181  << ", zmax " << zmax << ", layer: " << layer_id << endl;
182  }
183  }
184  // here we just update the exit values, it will be overwritten
185  // for every step until we leave the volume or the particle
186  // ceases to exist
187  // some sanity checks for inconsistencies
188  // check if this hit was created, if not print out last post step status
189  if (!hit || !isfinite(hit->get_x(0)))
190  {
191  cout << GetName() << ": hit was not created" << endl;
192  cout << "prestep status: " << PHG4StepStatusDecode::GetStepStatus(prePoint->GetStepStatus())
193  << ", poststep status: " << PHG4StepStatusDecode::GetStepStatus(postPoint->GetStepStatus())
194  << ", last pre step status: " << PHG4StepStatusDecode::GetStepStatus(saveprestepstatus)
195  << ", last post step status: " << PHG4StepStatusDecode::GetStepStatus(savepoststepstatus) << endl;
196  cout << "last track: " << savetrackid
197  << ", current trackid: " << aTrack->GetTrackID() << endl;
198  cout << "phys pre vol: " << volume->GetName()
199  << " post vol : " << touchpost->GetVolume()->GetName() << endl;
200  cout << " previous phys pre vol: " << savevolpre->GetName()
201  << " previous phys post vol: " << savevolpost->GetName() << endl;
202  exit(1);
203  }
204  savepoststepstatus = postPoint->GetStepStatus();
205  // check if track id matches the initial one when the hit was created
206  if (aTrack->GetTrackID() != savetrackid)
207  {
208  cout << "hits do not belong to the same track" << endl;
209  cout << "saved track: " << savetrackid
210  << ", current trackid: " << aTrack->GetTrackID()
211  << endl;
212  exit(1);
213  }
214  saveprestepstatus = prePoint->GetStepStatus();
215  savepoststepstatus = postPoint->GetStepStatus();
216  savevolpre = volume;
217  savevolpost = touchpost->GetVolume();
218 
219  hit->set_x(1, postPoint->GetPosition().x() / cm);
220  hit->set_y(1, postPoint->GetPosition().y() / cm);
221  hit->set_z(1, postPoint->GetPosition().z() / cm);
222 
223  hit->set_px(1, postPoint->GetMomentum().x() / GeV);
224  hit->set_py(1, postPoint->GetMomentum().y() / GeV);
225  hit->set_pz(1, postPoint->GetMomentum().z() / GeV);
226 
227  hit->set_t(1, postPoint->GetGlobalTime() / nanosecond);
228  //sum up the energy to get total deposited
229  hit->set_edep(hit->get_edep() + edep);
230  if (hit->get_z(1) * cm > zmax || hit->get_z(1) * cm < zmin)
231  {
232  cout << detector_->SuperDetector() << std::setprecision(9)
233  << " PHG4TargetCoilV2SteppingAction: Exit hit z " << hit->get_z(1) * cm
234  << " outside acceptance zmin " << zmin
235  << ", zmax " << zmax << ", layer: " << layer_id << endl;
236  }
237  if (geantino)
238  {
239  hit->set_edep(-1); // only energy=0 g4hits get dropped, this way geantinos survive the g4hit compression
240  }
241  else
242  {
243  if (!IsBlackHole)
244  {
245  double eion = edep - aStep->GetNonIonizingEnergyDeposit() / GeV;
246  hit->set_eion(hit->get_eion() + eion);
247  }
248  }
249  if (save_light_yield)
250  {
251  double light_yield = GetVisibleEnergyDeposition(aStep);
252  hit->set_light_yield(hit->get_light_yield() + light_yield);
253  }
254  if (edep > 0)
255  {
256  if (G4VUserTrackInformation* p = aTrack->GetUserInformation())
257  {
258  if (PHG4TrackUserInfoV1* pp = dynamic_cast<PHG4TrackUserInfoV1*>(p))
259  {
260  pp->SetKeep(1); // we want to keep the track
261  }
262  }
263  }
264  // if any of these conditions is true this is the last step in
265  // this volume and we need to save the hit
266  // postPoint->GetStepStatus() == fGeomBoundary: track leaves this volume
267  // postPoint->GetStepStatus() == fWorldBoundary: track leaves this world
268  // (happens when your detector goes outside world volume)
269  // postPoint->GetStepStatus() == fAtRestDoItProc: track stops (typically
270  // aTrack->GetTrackStatus() == fStopAndKill is also set)
271  // aTrack->GetTrackStatus() == fStopAndKill: track ends
272  if (postPoint->GetStepStatus() == fGeomBoundary ||
273  postPoint->GetStepStatus() == fWorldBoundary ||
274  postPoint->GetStepStatus() == fAtRestDoItProc ||
275  aTrack->GetTrackStatus() == fStopAndKill ||
276  use_g4_steps > 0)
277  {
278  // save only hits with energy deposit (or -1 for geantino)
279  if (hit->get_edep())
280  {
281  hits_->AddHit(layer_id, hit);
282  if (saveshower)
283  {
284  saveshower->add_g4hit_id(hits_->GetID(), hit->get_hit_id());
285  }
286  // ownership has been transferred to container, set to null
287  // so we will create a new hit for the next track
288  hit = nullptr;
289  }
290  else
291  {
292  // if this hit has no energy deposit, just reset it for reuse
293  // this means we have to delete it in the dtor. If this was
294  // the last hit we processed the memory is still allocated
295  hit->Reset();
296  }
297  }
298  // return true to indicate the hit was used
299  return true;
300  }
301  else
302  {
303  return false;
304  }
305 }
306 
307 //____________________________________________________________________________..
309 {
310  string hitnodename;
311  if (detector_->SuperDetector() != "NONE")
312  {
313  hitnodename = "G4HIT_" + detector_->SuperDetector();
314  }
315  else
316  {
317  hitnodename = "G4HIT_" + detector_->GetName();
318  }
319 
320  //now look for the map and grab a pointer to it.
321  hits_ = findNode::getClass<PHG4HitContainer>(topNode, hitnodename.c_str());
322 
323  // if we do not find the node we need to make it.
324  if (!hits_ && !IsBlackHole)
325  {
326  std::cout << "PHG4TargetCoilV2SteppingAction::SetTopNode - unable to find " << hitnodename << std::endl;
327  }
328 }
virtual std::string GetName() const
Definition: PHG4Detector.h:51
ConstIterator AddHit(PHG4Hit *newhit)
virtual void set_layer(const unsigned int i)
Definition: PHG4Hit.h:66
virtual float get_z(const int i) const
Definition: PHG4Hit.h:23
virtual float get_eion() const
Definition: PHG4Hit.h:32
virtual void set_y(const int i, const float f)
Definition: PHG4Hit.h:53
virtual void set_shower_id(const int i)
Definition: PHG4Hit.h:68
virtual void set_light_yield(const float lightYield)
Definition: PHG4Hit.h:64
virtual float get_edep() const
Definition: PHG4Hit.h:31
virtual void set_px(const int i, const float f)
Definition: PHG4Hit.h:55
virtual void set_py(const int i, const float f)
Definition: PHG4Hit.h:56
virtual void set_t(const int i, const float f)
Definition: PHG4Hit.h:61
virtual void set_pz(const int i, const float f)
Definition: PHG4Hit.h:57
virtual void set_eion(const float f)
Definition: PHG4Hit.h:63
virtual void Reset()
Clear Event.
Definition: PHG4Hit.cc:63
virtual void set_z(const int i, const float f)
Definition: PHG4Hit.h:54
virtual float get_light_yield() const
Definition: PHG4Hit.h:33
virtual void set_x(const int i, const float f)
Definition: PHG4Hit.h:52
virtual PHG4HitDefs::keytype get_hit_id() const
Definition: PHG4Hit.h:36
virtual float get_x(const int i) const
Definition: PHG4Hit.h:21
virtual void set_trkid(const int i)
Definition: PHG4Hit.h:71
virtual void set_edep(const float f)
Definition: PHG4Hit.h:62
virtual void add_g4hit_id(int volume, PHG4HitDefs::keytype id)
Definition: PHG4Shower.h:100
virtual double GetVisibleEnergyDeposition(const G4Step *step)
get amount of energy that can make scintillation light, in Unit of GeV.
std::string GetName() const
int get_elem_id(const std::string &name) const
bool IsInCylinder(const G4VPhysicalVolume *) const
void SuperDetector(const std::string &name)
bool UserSteppingAction(const G4Step *, bool)
stepping action
PHG4TargetCoilV2SteppingAction(PHG4TargetCoilV2Detector *, const PHParameters *parameters)
constructor
void SetInterfacePointers(PHCompositeNode *)
reimplemented from base class
std::string GetStepStatus(const int istatus)