61 constexpr G4double ekinLimit = 0.2*CLHEP::MeV;
75 lManager->Register(
this);
81 fFluct = param->LossFluctuation();
82 fLinLimit = 5*param->LinearLossLimit();
89 lManager->DeRegister(
this);
98 fCuts = theCoupleTable->GetEnergyCutsVector(1);
103void G4DynamicParticleIonisation::PreStepInitialisation(
const G4Track& track)
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;
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()];
119 fCut = std::min(fCut, fTmax);
144 PreStepInitialisation(track);
146 if (fCharge != 0.0) {
147 xsec = ComputeCrossSection(fEkinPreStep);
174 G4cout <<
"G4DynamicParticleIonisation::PostStepGetPhysicalInteractionLength ";
176 <<
" for unknown particle Mass(GeV)=" << fMass/CLHEP::GeV
177 <<
" charge=" << fCharge
178 <<
" Material " << fMaterial->GetName()
179 <<
" Ekin(MeV)=" << fEkinPreStep/CLHEP::MeV
181 <<
" ProposedLength(cm)=" << x/CLHEP::cm <<
G4endl;
193 fParticleChange.InitializeForAlongStep(track);
196 if (fCharge == 0.0) {
return &fParticleChange; }
199 if (fEkinPreStep <= fLowestEkin) {
200 fParticleChange.SetProposedKineticEnergy(0.0);
201 fParticleChange.ProposeLocalEnergyDeposit(fEkinPreStep);
202 return &fParticleChange;
206 G4double dedxPre = ComputeDEDX(fEkinPreStep);
208 G4double ekinPostStep = fEkinPreStep - eloss;
211 if (fEkinPreStep*fLinLimit < eloss && ekinPostStep > fLowestEkin) {
212 G4double dedxPost = ComputeDEDX(ekinPostStep);
213 eloss = (eloss + dedxPost*length)*0.5;
217 if (fFluct && fEkinPreStep > eloss) {
219 fCut, fTmax, length, eloss);
222 ekinPostStep = fEkinPreStep - eloss;
225 if (ekinPostStep <= fLowestEkin) {
226 fParticleChange.SetProposedKineticEnergy(0.0);
227 fParticleChange.ProposeLocalEnergyDeposit(fEkinPreStep);
229 fParticleChange.SetProposedKineticEnergy(ekinPostStep);
230 fParticleChange.ProposeLocalEnergyDeposit(eloss);
232 return &fParticleChange;
241 fParticleChange.InitializeForPostStep(track);
244 G4double kinEnergy = dp->GetKineticEnergy();
245 const G4double totEnergy = kinEnergy + fMass;
246 const G4double beta2 = kinEnergy*(kinEnergy + 2.0*fMass)/(totEnergy*totEnergy);
256 deltaKinEnergy = fCut*fTmax/(fCut*(1.0 - rndm[0]) + fTmax*rndm[0]);
257 f = 1.0 - beta2*deltaKinEnergy/fTmax;
259 }
while( rndm[1] > f);
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();
269 G4ThreeVector deltaDirection(sint*std::cos(phi), sint*std::sin(phi), cost);
270 deltaDirection.
rotateUz(dp->GetMomentumDirection());
276 t->SetCreatorModelID(fSecID);
277 fParticleChange.AddSecondary(t);
280 kinEnergy -= deltaKinEnergy;
281 G4ThreeVector finalP = dp->GetMomentum() - delta->GetMomentum();
282 finalP = finalP.
unit();
284 fParticleChange.SetProposedKineticEnergy(kinEnergy);
285 fParticleChange.SetProposedMomentumDirection(finalP);
286 return &fParticleChange;
303 G4double dedx =
G4Log(2.0*CLHEP::electron_mass_c2*bg2*fCut/exc2) - (1.0 + xc)*beta2;
311 dedx = std::max(dedx, 0.0);
323 G4double energy2 = totEnergy*totEnergy;
324 G4double beta2 = ekin*(ekin + 2.0*fMass)/energy2;
326 cross = (fTmax - fCut)/(fCut*fTmax*beta2) -
G4Log(fTmax/fCut)/fTmax;
327 cross *= CLHEP::twopi_mc2_rcl2*fCharge*fCharge*fMaterial->GetElectronDensity();
355 out <<
"G4DynamicParticleIonisation: dynamic ionisation" <<
G4endl;
G4double condition(const G4ErrorSymMatrix &m)
G4double G4Log(G4double x)
CLHEP::Hep3Vector G4ThreeVector
G4GLOB_DLL std::ostream G4cout
Hep3Vector & rotateUz(const Hep3Vector &)
virtual void flatArray(const int size, double *vect)=0
~G4DynamicParticleIonisation() override
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 ¤tSafety) override
void BuildPhysicsTable(const G4ParticleDefinition &) override
void ProcessDescription(std::ostream &) const override
G4double AlongStepGetPhysicalInteractionLength(const G4Track &, G4double previousStepSize, G4double currentMinimumStep, G4double ¤tSafety, G4GPILSelection *selection) override
G4VParticleChange * PostStepDoIt(const G4Track &, const G4Step &) override
G4DynamicParticleIonisation()
G4double PostStepGetPhysicalInteractionLength(const G4Track &track, G4double previousStepSize, G4ForceCondition *condition) override
static G4Electron * Electron()
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)
G4double theNumberOfInteractionLengthLeft
void SetProcessSubType(G4int)
const G4String & GetProcessName() const