Geant4 11.3.0
Toolkit for the simulation of the passage of particles through matter
Loading...
Searching...
No Matches
G4DynamicParticleIonisation.cc
Go to the documentation of this file.
1//
2// ********************************************************************
3// * License and Disclaimer *
4// * *
5// * The Geant4 software is copyright of the Copyright Holders of *
6// * the Geant4 Collaboration. It is provided under the terms and *
7// * conditions of the Geant4 Software License, included in the file *
8// * LICENSE and available at http://cern.ch/geant4/license . These *
9// * include a list of copyright holders. *
10// * *
11// * Neither the authors of this software system, nor their employing *
12// * institutes,nor the agencies providing financial support for this *
13// * work make any representation or warranty, express or implied, *
14// * regarding this software system or assume any liability for its *
15// * use. Please see the license in the file LICENSE and URL above *
16// * for the full disclaimer and the limitation of liability. *
17// * *
18// * This code implementation is the result of the scientific and *
19// * technical work of the GEANT4 collaboration. *
20// * By using, copying, modifying or distributing the software (or *
21// * any work based on the software) you agree to acknowledge its *
22// * use in resulting scientific publications, and indicate your *
23// * acceptance of all terms of the Geant4 Software license. *
24// ********************************************************************
25//
26//
27// -------------------------------------------------------------------
28//
29// GEANT4 Class file
30//
31//
32// File name: G4DynamicParticleIonisation
33//
34// Author: Vladimir Ivanchenko
35//
36// Creation date: 17.08.2024
37//
38// -------------------------------------------------------------------
39//
40//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo....
41//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo....
42
45#include "G4SystemOfUnits.hh"
48#include "G4Electron.hh"
49#include "G4EmParameters.hh"
50#include "G4EmProcessSubType.hh"
51#include "G4LossTableManager.hh"
54#include "G4Material.hh"
55#include "G4Step.hh"
56#include "G4Track.hh"
57#include "G4Log.hh"
58
59namespace
60{
61 constexpr G4double ekinLimit = 0.2*CLHEP::MeV;
62 const G4double twoln10 = 2*G4Log(10.0);
63}
64
65//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo....
66
68 : G4VContinuousDiscreteProcess("dynPartIoni")
69{
72 theElectron = G4Electron::Electron();
73
74 lManager = G4LossTableManager::Instance();
75 lManager->Register(this);
76
77 fUrban = new G4DynamicParticleFluctuation();
78
79 // define these flags only once
80 auto param = G4EmParameters::Instance();
81 fFluct = param->LossFluctuation();
82 fLinLimit = 5*param->LinearLossLimit();
83}
84
85//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo....
86
88{
89 lManager->DeRegister(this);
90}
91
92//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo....
93
94void
96{
97 auto theCoupleTable = G4ProductionCutsTable::GetProductionCutsTable();
98 fCuts = theCoupleTable->GetEnergyCutsVector(1);
99}
100
101//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo....
102
103void G4DynamicParticleIonisation::PreStepInitialisation(const G4Track& track)
104{
105 fCouple = track.GetMaterialCutsCouple();
106 fMaterial = fCouple->GetMaterial();
107 auto dpart = track.GetDynamicParticle();
108 fEkinPreStep = dpart->GetKineticEnergy();
109 fMass = std::max(dpart->GetMass(), CLHEP::electron_mass_c2);
110 fCharge = dpart->GetCharge()/CLHEP::eplus;
111 fRatio = fMass/CLHEP::proton_mass_c2;
112 fLowestEkin = ekinLimit*fRatio;
113 G4double tau = fEkinPreStep/fMass;
114 G4double ratio = CLHEP::electron_mass_c2/fMass;
115 fTmax = 2.0*CLHEP::electron_mass_c2*tau*(tau + 2.) /
116 (1. + 2.0*(tau + 1.)*ratio + ratio*ratio);
117 fCut = (*fCuts)[fCouple->GetIndex()];
118 fCut = std::max(fCut, fMaterial->GetIonisation()->GetMeanExcitationEnergy());
119 fCut = std::min(fCut, fTmax);
120}
121
122//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo....
123
126 G4GPILSelection* selection)
127{
128 *selection = CandidateForSelection;
129
130 // no step limit for the time being
131 return DBL_MAX;
132}
133
134//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo....
135
137 const G4Track& track, G4double previousStepSize,
139{
141 G4double x = DBL_MAX;
142 G4double xsec = 0.0;
143
144 PreStepInitialisation(track);
145
146 if (fCharge != 0.0) {
147 xsec = ComputeCrossSection(fEkinPreStep);
148 }
149
150 if (xsec <= 0.0) {
153
154 } else {
156
159
160 } else if(currentInteractionLength < DBL_MAX) {
161
162 // subtract NumberOfInteractionLengthLeft using previous step
164 previousStepSize/currentInteractionLength;
165
168 }
169 currentInteractionLength = 1.0/xsec;
171 }
172#ifdef G4VERBOSE
173 if (verboseLevel>2) {
174 G4cout << "G4DynamicParticleIonisation::PostStepGetPhysicalInteractionLength ";
175 G4cout << " Process: " << GetProcessName()
176 << " for unknown particle Mass(GeV)=" << fMass/CLHEP::GeV
177 << " charge=" << fCharge
178 << " Material " << fMaterial->GetName()
179 << " Ekin(MeV)=" << fEkinPreStep/CLHEP::MeV
180 << " MFP(cm)=" << currentInteractionLength/CLHEP::cm
181 << " ProposedLength(cm)=" << x/CLHEP::cm <<G4endl;
182 }
183#endif
184 return x;
185}
186
187//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo....
188
191 const G4Step& step)
192{
193 fParticleChange.InitializeForAlongStep(track);
194
195 // no energy loss
196 if (fCharge == 0.0) { return &fParticleChange; }
197
198 // stop low-energy object
199 if (fEkinPreStep <= fLowestEkin) {
200 fParticleChange.SetProposedKineticEnergy(0.0);
201 fParticleChange.ProposeLocalEnergyDeposit(fEkinPreStep);
202 return &fParticleChange;
203 }
204
205 G4double length = step.GetStepLength();
206 G4double dedxPre = ComputeDEDX(fEkinPreStep);
207 G4double eloss = dedxPre*length;
208 G4double ekinPostStep = fEkinPreStep - eloss;
209
210 // correction for large step if it is not the last step
211 if (fEkinPreStep*fLinLimit < eloss && ekinPostStep > fLowestEkin) {
212 G4double dedxPost = ComputeDEDX(ekinPostStep);
213 eloss = (eloss + dedxPost*length)*0.5;
214 }
215
216 // do not sample fluctuations at the last step
217 if (fFluct && fEkinPreStep > eloss) {
218 eloss = fUrban->SampleFluctuations(fCouple, track.GetDynamicParticle(),
219 fCut, fTmax, length, eloss);
220 }
221
222 ekinPostStep = fEkinPreStep - eloss;
223
224 // stop low-energy object
225 if (ekinPostStep <= fLowestEkin) {
226 fParticleChange.SetProposedKineticEnergy(0.0);
227 fParticleChange.ProposeLocalEnergyDeposit(fEkinPreStep);
228 } else {
229 fParticleChange.SetProposedKineticEnergy(ekinPostStep);
230 fParticleChange.ProposeLocalEnergyDeposit(eloss);
231 }
232 return &fParticleChange;
233}
234
235//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo....
236
239{
241 fParticleChange.InitializeForPostStep(track);
242
243 auto dp = track.GetDynamicParticle();
244 G4double kinEnergy = dp->GetKineticEnergy();
245 const G4double totEnergy = kinEnergy + fMass;
246 const G4double beta2 = kinEnergy*(kinEnergy + 2.0*fMass)/(totEnergy*totEnergy);
247
248 G4double deltaKinEnergy, f;
249
250 CLHEP::HepRandomEngine* rndmEngineMod = G4Random::getTheEngine();
251 G4double rndm[2];
252
253 // sampling without nuclear size effect
254 do {
255 rndmEngineMod->flatArray(2, rndm);
256 deltaKinEnergy = fCut*fTmax/(fCut*(1.0 - rndm[0]) + fTmax*rndm[0]);
257 f = 1.0 - beta2*deltaKinEnergy/fTmax;
258 // Loop checking, 14-Aug-2024, Vladimir Ivanchenko
259 } while( rndm[1] > f);
260
261 G4double deltaMomentum =
262 std::sqrt(deltaKinEnergy * (deltaKinEnergy + 2.0*CLHEP::electron_mass_c2));
263 G4double cost = deltaKinEnergy * (totEnergy + CLHEP::electron_mass_c2) /
264 (deltaMomentum * dp->GetTotalMomentum());
265 cost = std::min(cost, 1.0);
266 const G4double sint = std::sqrt((1.0 - cost)*(1.0 + cost));
267 const G4double phi = CLHEP::twopi*rndmEngineMod->flat();
268
269 G4ThreeVector deltaDirection(sint*std::cos(phi), sint*std::sin(phi), cost);
270 deltaDirection.rotateUz(dp->GetMomentumDirection());
271
272 // create G4DynamicParticle object for delta ray
273 auto delta = new G4DynamicParticle(theElectron, deltaDirection, deltaKinEnergy);
274 auto t = new G4Track(delta, track.GetGlobalTime(), track.GetPosition());
275 t->SetTouchableHandle(track.GetTouchableHandle());
276 t->SetCreatorModelID(fSecID);
277 fParticleChange.AddSecondary(t);
278
279 // Change kinematics of primary particle
280 kinEnergy -= deltaKinEnergy;
281 G4ThreeVector finalP = dp->GetMomentum() - delta->GetMomentum();
282 finalP = finalP.unit();
283
284 fParticleChange.SetProposedKineticEnergy(kinEnergy);
285 fParticleChange.SetProposedMomentumDirection(finalP);
286 return &fParticleChange;
287}
288
289//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo....
290
291G4double G4DynamicParticleIonisation::ComputeDEDX(G4double ekin)
292{
293 G4double tau = ekin/fMass;
294 G4double gam = tau + 1.0;
295 G4double bg2 = tau * (tau + 2.0);
296 G4double beta2 = bg2/(gam*gam);
297 G4double xc = fCut/fTmax;
298
299 G4double exc = fMaterial->GetIonisation()->GetMeanExcitationEnergy();
300 G4double exc2 = exc*exc;
301
302 // general Bethe-Bloch formula
303 G4double dedx = G4Log(2.0*CLHEP::electron_mass_c2*bg2*fCut/exc2) - (1.0 + xc)*beta2;
304
305 // density correction
306 G4double x = G4Log(bg2)/twoln10;
307 dedx -= fMaterial->GetIonisation()->DensityCorrection(x);
308
309 // now compute the total ionization loss per volume
310 dedx *= CLHEP::twopi_mc2_rcl2*fCharge*fCharge*fMaterial->GetElectronDensity()/beta2;
311 dedx = std::max(dedx, 0.0);
312 return dedx;
313}
314
315//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo....
316
317G4double G4DynamicParticleIonisation::ComputeCrossSection(G4double ekin)
318{
319 G4double cross = 0.0;
320 if (fCut < fTmax) {
321
322 G4double totEnergy = ekin + fMass;
323 G4double energy2 = totEnergy*totEnergy;
324 G4double beta2 = ekin*(ekin + 2.0*fMass)/energy2;
325
326 cross = (fTmax - fCut)/(fCut*fTmax*beta2) - G4Log(fTmax/fCut)/fTmax;
327 cross *= CLHEP::twopi_mc2_rcl2*fCharge*fCharge*fMaterial->GetElectronDensity();
328 }
329 return cross;
330}
331
332//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo....
333
336{
337 // Note: this method is not used at run-time, so its implementation is simplified.
338 // It might be eventually refined later.
340 return DBL_MAX;
341}
342
343//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo....
344
350
351//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo....
352
354{
355 out << "G4DynamicParticleIonisation: dynamic ionisation" << G4endl;
356}
357
358//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo....
@ fDynamicIonisation
G4double condition(const G4ErrorSymMatrix &m)
G4ForceCondition
@ NotForced
G4GPILSelection
@ CandidateForSelection
G4double G4Log(G4double x)
Definition G4Log.hh:227
CLHEP::Hep3Vector G4ThreeVector
double G4double
Definition G4Types.hh:83
#define G4endl
Definition G4ios.hh:67
G4GLOB_DLL std::ostream G4cout
#define G4UniformRand()
Definition Randomize.hh:52
Hep3Vector unit() const
Hep3Vector & rotateUz(const Hep3Vector &)
virtual double flat()=0
virtual void flatArray(const int size, double *vect)=0
G4double GetMeanFreePath(const G4Track &track, G4double previousStepSize, G4ForceCondition *condition) override
G4VParticleChange * AlongStepDoIt(const G4Track &, const G4Step &) override
G4double GetContinuousStepLimit(const G4Track &track, G4double previousStepSize, G4double currentMinimumStep, G4double &currentSafety) override
void BuildPhysicsTable(const G4ParticleDefinition &) override
void ProcessDescription(std::ostream &) const override
G4double AlongStepGetPhysicalInteractionLength(const G4Track &, G4double previousStepSize, G4double currentMinimumStep, G4double &currentSafety, G4GPILSelection *selection) override
G4VParticleChange * PostStepDoIt(const G4Track &, const G4Step &) override
G4double PostStepGetPhysicalInteractionLength(const G4Track &track, G4double previousStepSize, G4ForceCondition *condition) override
static G4Electron * Electron()
Definition G4Electron.cc:91
static G4EmParameters * Instance()
G4double DensityCorrection(G4double x) const
G4double GetMeanExcitationEnergy() const
static G4LossTableManager * Instance()
const G4Material * GetMaterial() const
G4IonisParamMat * GetIonisation() const
G4double GetElectronDensity() const
static G4ProductionCutsTable * GetProductionCutsTable()
G4double GetStepLength() const
const G4ThreeVector & GetPosition() const
G4double GetGlobalTime() const
const G4DynamicParticle * GetDynamicParticle() const
const G4TouchableHandle & GetTouchableHandle() const
const G4MaterialCutsCouple * GetMaterialCutsCouple() const
G4VContinuousDiscreteProcess(const G4String &, G4ProcessType aType=fNotDefined)
G4double currentInteractionLength
G4double theInitialNumberOfInteractionLength
void SetVerboseLevel(G4int value)
G4int verboseLevel
G4double theNumberOfInteractionLengthLeft
void SetProcessSubType(G4int)
const G4String & GetProcessName() const
#define DBL_MAX
Definition templates.hh:62