Class Reference for E1039 Core & Analysis Software
PHG4EMCalSteppingAction.cc
Go to the documentation of this file.
2 #include "PHG4EMCalDetector.h"
3 
4 #include <phparameter/PHParameters.h>
5 
6 #include <g4main/PHG4Hit.h>
8 #include <g4main/PHG4Hitv1.h>
9 #include <g4main/PHG4Shower.h>
10 
11 #include <g4main/PHG4SteppingAction.h> // for PHG4SteppingAction
13 
14 #include <phool/getClass.h>
15 
16 #include <Geant4/G4ReferenceCountedHandle.hh> // for G4ReferenceCountedHandle
17 #include <Geant4/G4Step.hh>
18 #include <Geant4/G4StepPoint.hh> // for G4StepPoint
19 #include <Geant4/G4StepStatus.hh> // for fGeomBoundary, fAtRest...
20 #include <Geant4/G4String.hh> // for G4String
21 #include <Geant4/G4SystemOfUnits.hh>
22 #include <Geant4/G4ThreeVector.hh> // for G4ThreeVector
23 #include <Geant4/G4TouchableHandle.hh>
24 #include <Geant4/G4Track.hh> // for G4Track
25 #include <Geant4/G4TrackStatus.hh> // for fStopAndKill
26 #include <Geant4/G4Types.hh> // for G4double
27 #include <Geant4/G4VPhysicalVolume.hh> // for G4VPhysicalVolume
28 #include <Geant4/G4VTouchable.hh> // for G4VTouchable
29 #include <Geant4/G4VUserTrackInformation.hh> // for G4VUserTrackInformation
30 
31 #include <iostream>
32 #include <string> // for basic_string, operator+
33 
34 class PHCompositeNode;
35 
36 using namespace std;
37 
38 //____________________________________________________________________________..
40  : m_Detector(detector)
41  , m_SignalHitContainer(nullptr)
42  , m_AbsorberHitContainer(nullptr)
43  , m_Params(parameters)
44  , m_CurrentHitContainer(nullptr)
45  , m_Hit(nullptr)
46  , m_CurrentShower(nullptr)
47  , m_IsActive(m_Params->get_int_param("active"))
48  , m_IsAbsorberActive(m_Params->get_int_param("absorberactive"))
49  , m_AbsorberTruth(0)
50  , m_LightScintModel(1)
51  , m_IsBlackHole(m_Params->get_int_param("blackhole"))
52  , m_nTowersX(m_Params->get_int_param("n_towers_x"))
53  , m_nTowersY(m_Params->get_int_param("n_towers_y"))
54 {
55 }
56 
58 {
59  // if the last hit was a zero energie deposit hit, it is just reset
60  // and the memory is still allocated, so we need to delete it here
61  // if the last hit was saved, hit is a nullptr pointer which are
62  // legal to delete (it results in a no operation)
63  delete m_Hit;
64 }
65 
66 //____________________________________________________________________________..
67 bool PHG4EMCalSteppingAction::UserSteppingAction(const G4Step* aStep, bool)
68 {
69  G4TouchableHandle touch = aStep->GetPreStepPoint()->GetTouchableHandle();
70  G4VPhysicalVolume* volume = touch->GetVolume();
71 
72  int whichactive = m_Detector->IsInForwardEcal(volume);
73  if(whichactive == 0) return false;
74 
75  int towerID = -1;
76  int plateID = -1;
77  int towerIDX = -1;
78  int towerIDY = -1;
79  int depth = touch->GetHistoryDepth();
80  if(depth >= 2)
81  {
82  towerID = touch->GetCopyNumber(1);
83  plateID = touch->GetCopyNumber(0);
84 
85  towerIDY = towerID % m_nTowersY;
86  towerIDX = (towerID - towerIDY)/m_nTowersY;
87  }
88  else //should not happen, just put here for sanity check
89  {
90  std::cerr << "EMCal volumn history is wrong for " << name << " with history depth = " << depth << std::endl;
91  }
92 
93  // Get energy deposited by this step
94  double edep = aStep->GetTotalEnergyDeposit()/GeV;
95  double eion = (aStep->GetTotalEnergyDeposit() - aStep->GetNonIonizingEnergyDeposit())/GeV;
96  double light_yield = 0;
97 
98  // Get pointer to associated Geant4 track
99  const G4Track* aTrack = aStep->GetTrack();
100 
101  // if this block stops everything, just put all kinetic energy into edep
102  if(m_IsBlackHole > 0)
103  {
104  edep = aTrack->GetKineticEnergy()/GeV;
105  G4Track* killtrack = const_cast<G4Track*>(aTrack);
106  killtrack->SetTrackStatus(fStopAndKill);
107  }
108 
109  if(m_IsActive > 0)
110  {
111  //Check if the particle is geantino
112  bool geantino = (aTrack->GetParticleDefinition()->GetPDGEncoding() == 0 && aTrack->GetParticleDefinition()->GetParticleName().find("geantino") != string::npos);
113 
114  G4StepPoint* prePoint = aStep->GetPreStepPoint();
115  if(prePoint->GetStepStatus() == fGeomBoundary || prePoint->GetStepStatus() == fUndefined)
116  {
117  if(m_Hit == nullptr)
118  {
119  m_Hit = new PHG4Hitv1();
120  }
121 
122  m_Hit->set_scint_id(towerID);
123 
124  // Set hit location (tower index)
125  m_Hit->set_index_j(towerIDX);
126  m_Hit->set_index_k(towerIDY);
127  m_Hit->set_index_l(plateID);
128 
129  // Set hit location (space point)
130  m_Hit->set_x(0, prePoint->GetPosition().x()/cm);
131  m_Hit->set_y(0, prePoint->GetPosition().y()/cm);
132  m_Hit->set_z(0, prePoint->GetPosition().z()/cm);
133 
134  // Set hit momentum
135  m_Hit->set_px(0, prePoint->GetMomentum().x()/GeV);
136  m_Hit->set_py(0, prePoint->GetMomentum().y()/GeV);
137  m_Hit->set_pz(0, prePoint->GetMomentum().z()/GeV);
138 
139  // Set hit time
140  m_Hit->set_t(0, prePoint->GetGlobalTime()/nanosecond);
141 
142  //set the track ID
143  m_Hit->set_trkid(aTrack->GetTrackID());
144 
145  // set intial energy deposit
146  m_Hit->set_edep(0.);
147  m_Hit->set_eion(0.);
148 
149  // Now add the hit to the hit collection
150  // here we do things which are different between scintillator and absorber hits
151  if(whichactive > 0)
152  {
153  m_CurrentHitContainer = m_SignalHitContainer;
154  m_Hit->set_light_yield(0.); // for scintillator only, initialize light yields
155  }
156  else
157  {
158  m_CurrentHitContainer = m_AbsorberHitContainer;
159  }
160 
161  // here we set what is common for scintillator and absorber hits
162  if(G4VUserTrackInformation* p = aTrack->GetUserInformation())
163  {
164  if(PHG4TrackUserInfoV1* pp = dynamic_cast<PHG4TrackUserInfoV1*>(p))
165  {
166  m_Hit->set_trkid(pp->GetUserTrackId());
167  m_Hit->set_shower_id(pp->GetShower()->get_id());
168  m_CurrentShower = pp->GetShower();
169  }
170  }
171  }
172 
173  if(whichactive > 0)
174  {
175  light_yield = eion;
176  if(m_LightScintModel > 0)
177  {
178  light_yield = GetVisibleEnergyDeposition(aStep); //for scintillator only, calculate light yields
179  }
180  }
181 
182  // Update exit values- will be overwritten with every step until
183  // we leave the volume or the particle ceases to exist
184  G4StepPoint* postPoint = aStep->GetPostStepPoint();
185  m_Hit->set_x(1, postPoint->GetPosition().x()/cm);
186  m_Hit->set_y(1, postPoint->GetPosition().y()/cm);
187  m_Hit->set_z(1, postPoint->GetPosition().z()/cm);
188  m_Hit->set_t(1, postPoint->GetGlobalTime()/nanosecond);
189 
190  m_Hit->set_px(1, postPoint->GetMomentum().x()/GeV);
191  m_Hit->set_py(1, postPoint->GetMomentum().y()/GeV);
192  m_Hit->set_pz(1, postPoint->GetMomentum().z()/GeV);
193 
194  // sum up the energy to get total deposited
195  m_Hit->set_edep(m_Hit->get_edep() + edep);
196  m_Hit->set_eion(m_Hit->get_eion() + eion);
197  if(whichactive > 0)
198  {
199  m_Hit->set_light_yield(m_Hit->get_light_yield() + light_yield);
200  }
201 
202  if(geantino)
203  {
204  m_Hit->set_edep(-1.); // only energy=0 g4hits get dropped, this way geantinos survive the g4hit compression
205  m_Hit->set_eion(-1.);
206  if(whichactive > 0)
207  {
208  m_Hit->set_light_yield(-1.);
209  }
210  }
211 
212  if(edep > 0 && (whichactive > 0 || m_AbsorberTruth > 0))
213  {
214  if(G4VUserTrackInformation* p = aTrack->GetUserInformation())
215  {
216  if(PHG4TrackUserInfoV1* pp = dynamic_cast<PHG4TrackUserInfoV1*>(p))
217  {
218  pp->SetKeep(1); // we want to keep the track
219  }
220  }
221  }
222 
223  if(postPoint->GetStepStatus() == fGeomBoundary || postPoint->GetStepStatus() == fWorldBoundary ||
224  postPoint->GetStepStatus() == fAtRestDoItProc || aTrack->GetTrackStatus() == fStopAndKill)
225  {
226  // save only hits with energy deposit (or -1 for geantino)
227  if(m_Hit->get_edep() > 0.)
228  {
229  m_CurrentHitContainer->AddHit(m_Detector->get_Layer(), m_Hit);
230  if(m_CurrentShower != nullptr)
231  {
232  m_CurrentShower->add_g4hit_id(m_CurrentHitContainer->GetID(), m_Hit->get_hit_id());
233  }
234  // ownership has been transferred to container, set to null
235  // so we will create a new hit for the next track
236  m_Hit = nullptr;
237  }
238  else
239  {
240  // if this hit has no energy deposit, just reset it for reuse
241  // this means we have to delete it in the dtor. If this was
242  // the last hit we processed the memory is still allocated
243  m_Hit->Reset();
244  }
245  }
246 
247  return true;
248  }
249 
250  return false;
251 }
252 
253 //____________________________________________________________________________..
255 {
256  std::string hitnodename;
257  std::string absorbernodename;
258 
259  if(m_Detector->SuperDetector() != "NONE")
260  {
261  hitnodename = "G4HIT_" + m_Detector->SuperDetector();
262  absorbernodename = "G4HIT_ABSORBER_" + m_Detector->SuperDetector();
263  }
264  else
265  {
266  hitnodename = "G4HIT_" + m_Detector->GetName();
267  absorbernodename = "G4HIT_ABSORBER_" + m_Detector->GetName();
268  }
269 
270  //now look for the map and grab a pointer to it.
271  m_SignalHitContainer = findNode::getClass<PHG4HitContainer>(topNode, hitnodename);
272  m_AbsorberHitContainer = findNode::getClass<PHG4HitContainer>(topNode, absorbernodename);
273 
274  // if we do not find the node it's messed up.
275  if(m_IsActive > 0 && !m_SignalHitContainer)
276  {
277  std::cerr << "PHG4EMCalSteppingAction::SetTopNode - unable to find " << hitnodename << std::endl;
278  }
279 
280  // this is perfectly fine if absorber hits are disabled
281  if(m_IsAbsorberActive > 0 && !m_AbsorberHitContainer)
282  {
283  if(Verbosity() > 0) std::cout << "PHG4EMCalSteppingAction::SetTopNode - unable to find " << absorbernodename << std::endl;
284  }
285 }
virtual std::string GetName() const
Definition: PHG4Detector.h:51
int IsInForwardEcal(G4VPhysicalVolume *) const
void SuperDetector(const std::string &name)
virtual bool UserSteppingAction(const G4Step *, bool)
stepping action
virtual void SetInterfacePointers(PHCompositeNode *)
reimplemented from base class
PHG4EMCalSteppingAction(PHG4EMCalDetector *, const PHParameters *parameters)
constructor
virtual ~PHG4EMCalSteppingAction()
destroctor
ConstIterator AddHit(PHG4Hit *newhit)
virtual void set_index_l(const int i)
Definition: PHG4Hit.h:79
virtual void set_index_k(const int i)
Definition: PHG4Hit.h:78
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 void set_trkid(const int i)
Definition: PHG4Hit.h:71
virtual void set_scint_id(const int i)
Definition: PHG4Hit.h:69
virtual void set_edep(const float f)
Definition: PHG4Hit.h:62
virtual void set_index_j(const int i)
Definition: PHG4Hit.h:77
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.