Class Reference for E1039 Core & Analysis Software
PHG4SquareTubeSteppingAction.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>
12 
13 #include <phool/getClass.h>
14 
15 #include <Geant4/G4Step.hh>
16 #include <Geant4/G4SystemOfUnits.hh>
17 
18 #include <iostream>
19 
20 using namespace std;
21 //____________________________________________________________________________..
23  : detector_(detector)
24  , params(parameters)
25  , hits_(nullptr)
26  , hit(nullptr)
27  , saveshower(nullptr)
28  , savevolpre(nullptr)
29  , savevolpost(nullptr)
30  , savetrackid(-1)
31  , saveprestepstatus(-1)
32  , savepoststepstatus(-1)
33  , active(params->get_int_param("active"))
34  , IsBlackHole(params->get_int_param("blackhole"))
35  , use_g4_steps(params->get_int_param("use_g4steps"))
36 {
37 }
38 
40 {
41  // if the last hit was a zero energie deposit hit, it is just reset
42  // and the memory is still allocated, so we need to delete it here
43  // if the last hit was saved, hit is a nullptr pointer which are
44  // legal to delete (it results in a no operation)
45  delete hit;
46 }
47 
48 //____________________________________________________________________________..
49 bool PHG4SquareTubeSteppingAction::UserSteppingAction(const G4Step* aStep, bool)
50 {
51  G4TouchableHandle touch = aStep->GetPreStepPoint()->GetTouchableHandle();
52  G4TouchableHandle touchpost = aStep->GetPostStepPoint()->GetTouchableHandle();
53  // get volume of the current step
54  G4VPhysicalVolume* volume = touch->GetVolume();
55 
56  if (!detector_->IsInBlock(volume))
57  {
58  return false;
59  }
60 
61  // collect energy and track length step by step
62  G4double edep = aStep->GetTotalEnergyDeposit() / GeV;
63  G4double eion = (aStep->GetTotalEnergyDeposit() - aStep->GetNonIonizingEnergyDeposit()) / GeV;
64  const G4Track* aTrack = aStep->GetTrack();
65 
66  // if this block stops everything, just put all kinetic energy into edep
67  if (IsBlackHole)
68  {
69  edep = aTrack->GetKineticEnergy() / GeV;
70  G4Track* killtrack = const_cast<G4Track*>(aTrack);
71  killtrack->SetTrackStatus(fStopAndKill);
72  }
73 
74  // make sure we are in a volume
75  if (active)
76  {
77  int layer_id = detector_->get_Layer();
78  bool geantino = false;
79  // the check for the pdg code speeds things up, I do not want to make
80  // an expensive string compare for every track when we know
81  // geantino or chargedgeantino has pid=0
82  if (aTrack->GetParticleDefinition()->GetPDGEncoding() == 0 &&
83  aTrack->GetParticleDefinition()->GetParticleName().find("geantino") != string::npos)
84  {
85  geantino = true;
86  }
87  G4StepPoint* prePoint = aStep->GetPreStepPoint();
88  G4StepPoint* postPoint = aStep->GetPostStepPoint();
89  // cout << "track id " << aTrack->GetTrackID() << endl;
90  // cout << "time prepoint: " << prePoint->GetGlobalTime() << endl;
91  // cout << "time postpoint: " << postPoint->GetGlobalTime() << endl;
92  int prepointstatus = prePoint->GetStepStatus();
93  if (prepointstatus == fGeomBoundary ||
94  prepointstatus == fUndefined ||
95  (prepointstatus == fPostStepDoItProc && savepoststepstatus == fGeomBoundary) ||
96  use_g4_steps > 0)
97  {
98  if (prepointstatus == fPostStepDoItProc && savepoststepstatus == fGeomBoundary)
99  {
100  cout << GetName() << ": New Hit for " << endl;
101  cout << "prestep status: " << PHG4StepStatusDecode::GetStepStatus(prePoint->GetStepStatus())
102  << ", poststep status: " << PHG4StepStatusDecode::GetStepStatus(postPoint->GetStepStatus())
103  << ", last pre step status: " << PHG4StepStatusDecode::GetStepStatus(saveprestepstatus)
104  << ", last post step status: " << PHG4StepStatusDecode::GetStepStatus(savepoststepstatus) << endl;
105  cout << "last track: " << savetrackid
106  << ", current trackid: " << aTrack->GetTrackID() << endl;
107  cout << "phys pre vol: " << volume->GetName()
108  << " post vol : " << touchpost->GetVolume()->GetName() << endl;
109  cout << " previous phys pre vol: " << savevolpre->GetName()
110  << " previous phys post vol: " << savevolpost->GetName() << endl;
111  }
112  if (!hit)
113  {
114  hit = new PHG4Hitv1();
115  }
116  hit->set_layer(layer_id);
117  //here we set the entrance values in cm
118  hit->set_x(0, prePoint->GetPosition().x() / cm);
119  hit->set_y(0, prePoint->GetPosition().y() / cm);
120  hit->set_z(0, prePoint->GetPosition().z() / cm);
121  // time in ns
122  hit->set_t(0, prePoint->GetGlobalTime() / nanosecond);
123  //set the track ID
124  hit->set_trkid(aTrack->GetTrackID());
125  savetrackid = aTrack->GetTrackID();
126 
127  //set the initial energy deposit
128  hit->set_edep(0);
129  if (!geantino && !IsBlackHole && active)
130  {
131  hit->set_eion(0);
132  }
133  if (G4VUserTrackInformation* p = aTrack->GetUserInformation())
134  {
135  if (PHG4TrackUserInfoV1* pp = dynamic_cast<PHG4TrackUserInfoV1*>(p))
136  {
137  hit->set_trkid(pp->GetUserTrackId());
138  hit->set_shower_id(pp->GetShower()->get_id());
139  saveshower = pp->GetShower();
140  }
141  }
142  }
143 
144  // some sanity checks for inconsistencies
145  // check if this hit was created, if not print out last post step status
146  if (!hit || !isfinite(hit->get_x(0)))
147  {
148  cout << GetName() << ": hit was not created" << endl;
149  cout << "prestep status: " << PHG4StepStatusDecode::GetStepStatus(prePoint->GetStepStatus())
150  << ", poststep status: " << PHG4StepStatusDecode::GetStepStatus(postPoint->GetStepStatus())
151  << ", last pre step status: " << PHG4StepStatusDecode::GetStepStatus(saveprestepstatus)
152  << ", last post step status: " << PHG4StepStatusDecode::GetStepStatus(savepoststepstatus) << endl;
153  cout << "last track: " << savetrackid
154  << ", current trackid: " << aTrack->GetTrackID() << endl;
155  cout << "phys pre vol: " << volume->GetName()
156  << " post vol : " << touchpost->GetVolume()->GetName() << endl;
157  cout << " previous phys pre vol: " << savevolpre->GetName()
158  << " previous phys post vol: " << savevolpost->GetName() << endl;
159  exit(1);
160  }
161  savepoststepstatus = postPoint->GetStepStatus();
162  // check if track id matches the initial one when the hit was created
163  if (aTrack->GetTrackID() != savetrackid)
164  {
165  cout << "hits do not belong to the same track" << endl;
166  cout << "saved track: " << savetrackid
167  << ", current trackid: " << aTrack->GetTrackID()
168  << endl;
169  exit(1);
170  }
171  saveprestepstatus = prePoint->GetStepStatus();
172  savepoststepstatus = postPoint->GetStepStatus();
173  savevolpre = volume;
174  savevolpost = touchpost->GetVolume();
175 
176  // here we just update the exit values, it will be overwritten
177  // for every step until we leave the volume or the particle
178  // ceases to exist
179  hit->set_x(1, postPoint->GetPosition().x() / cm);
180  hit->set_y(1, postPoint->GetPosition().y() / cm);
181  hit->set_z(1, postPoint->GetPosition().z() / cm);
182 
183  hit->set_t(1, postPoint->GetGlobalTime() / nanosecond);
184  //sum up the energy to get total deposited
185  hit->set_edep(hit->get_edep() + edep);
186  // update ionization energy only for active volumes, not for black holes or geantinos
187  // if the hit is created without eion, get_eion() returns NAN
188  // if you see NANs check the creation of the hit
189  if (active && !IsBlackHole && !geantino)
190  {
191  hit->set_eion(hit->get_eion() + eion);
192  }
193  if (geantino)
194  {
195  hit->set_edep(-1); // only energy=0 g4hits get dropped, this way geantinos survive the g4hit compression
196  }
197  if (edep > 0)
198  {
199  if (G4VUserTrackInformation* p = aTrack->GetUserInformation())
200  {
201  if (PHG4TrackUserInfoV1* pp = dynamic_cast<PHG4TrackUserInfoV1*>(p))
202  {
203  pp->SetKeep(1); // we want to keep the track
204  }
205  }
206  }
207  // if any of these conditions is true this is the last step in
208  // this volume and we need to save the hit
209  // postPoint->GetStepStatus() == fGeomBoundary: track leaves this volume
210  // postPoint->GetStepStatus() == fWorldBoundary: track leaves this world
211  // (happens when your detector goes outside world volume)
212  // postPoint->GetStepStatus() == fAtRestDoItProc: track stops (typically
213  // aTrack->GetTrackStatus() == fStopAndKill is also set)
214  // aTrack->GetTrackStatus() == fStopAndKill: track ends
215  if (postPoint->GetStepStatus() == fGeomBoundary ||
216  postPoint->GetStepStatus() == fWorldBoundary ||
217  postPoint->GetStepStatus() == fAtRestDoItProc ||
218  aTrack->GetTrackStatus() == fStopAndKill ||
219  use_g4_steps > 0)
220  {
221  // save only hits with energy deposit (or -1 for geantino)
222  if (hit->get_edep())
223  {
224  hits_->AddHit(layer_id, hit);
225  if (saveshower)
226  {
227  saveshower->add_g4hit_id(hits_->GetID(), hit->get_hit_id());
228  }
229  // ownership has been transferred to container, set to null
230  // so we will create a new hit for the next track
231  hit = nullptr;
232  }
233  else
234  {
235  // if this hit has no energy deposit, just reset it for reuse
236  // this means we have to delete it in the dtor. If this was
237  // the last hit we processed the memory is still allocated
238  hit->Reset();
239  }
240  }
241  // return true to indicate the hit was used
242  return true;
243  }
244  else
245  {
246  return false;
247  }
248 }
249 
250 //____________________________________________________________________________..
252 {
253  string hitnodename;
254  if (detector_->SuperDetector() != "NONE")
255  {
256  hitnodename = "G4HIT_" + detector_->SuperDetector();
257  }
258  else
259  {
260  hitnodename = "G4HIT_" + detector_->GetName();
261  }
262 
263  //now look for the map and grab a pointer to it.
264  hits_ = findNode::getClass<PHG4HitContainer>(topNode, hitnodename.c_str());
265 
266  // if we do not find the node we need to make it.
267  if (!hits_)
268  {
269  std::cout << "PHG4SquareTubeSteppingAction::SetTopNode - unable to find " << hitnodename << std::endl;
270  }
271 }
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_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 float get_edep() const
Definition: PHG4Hit.h:31
virtual void set_t(const int i, const float f)
Definition: PHG4Hit.h:61
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 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
void SuperDetector(const std::string &name)
bool IsInBlock(G4VPhysicalVolume *) const
PHG4SquareTubeSteppingAction(PHG4SquareTubeDetector *, const PHParameters *parameters)
constructor
virtual bool UserSteppingAction(const G4Step *, bool)
stepping action
virtual void SetInterfacePointers(PHCompositeNode *)
reimplemented from base class
std::string GetName() const
std::string GetStepStatus(const int istatus)