Geant4 11.3.0
Toolkit for the simulation of the passage of particles through matter
Loading...
Searching...
No Matches
G4VMultipleScattering.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: G4VMultipleScattering
33//
34// Author: Vladimir Ivanchenko on base of Laszlo Urban code
35//
36// Creation date: 25.03.2003
37//
38// Modifications:
39//
40// 16-07-03 Use G4VMscModel interface (V.Ivanchenko)
41// 03-11-03 Fix initialisation problem in RetrievePhysicsTable (V.Ivanchenko)
42// 04-11-03 Update PrintInfoDefinition (V.Ivanchenko)
43// 01-03-04 SampleCosineTheta signature changed
44// 22-04-04 SampleCosineTheta signature changed back to original
45// 27-08-04 Add InitialiseForRun method (V.Ivanchneko)
46// 08-11-04 Migration to new interface of Store/Retrieve tables (V.Ivantchenko)
47// 11-03-05 Shift verbose level by 1 (V.Ivantchenko)
48// 15-04-05 optimize internal interface (V.Ivanchenko)
49// 15-04-05 remove boundary flag (V.Ivanchenko)
50// 27-10-05 introduce virtual function MscStepLimitation() (V.Ivanchenko)
51// 12-04-07 Add verbosity at destruction (V.Ivanchenko)
52// 27-10-07 Virtual functions moved to source (V.Ivanchenko)
53// 11-03-08 Set skin value does not effect step limit type (V.Ivanchenko)
54// 24-06-09 Removed hidden bin in G4PhysicsVector (V.Ivanchenko)
55// 04-06-13 Adoptation to MT mode (V.Ivanchenko)
56//
57
58// -------------------------------------------------------------------
59//
60//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo....
61//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo....
62
65#include "G4SystemOfUnits.hh"
66#include "G4LossTableManager.hh"
68#include "G4Step.hh"
71#include "G4UnitsTable.hh"
73#include "G4Electron.hh"
74#include "G4GenericIon.hh"
76#include "G4SafetyHelper.hh"
77#include "G4ParticleTable.hh"
78#include "G4ProcessVector.hh"
79#include "G4ProcessManager.hh"
80#include "G4LossTableBuilder.hh"
81#include "G4EmTableUtil.hh"
82#include <iostream>
83
84//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo....
85
88 fNewPosition(0.,0.,0.),
89 fNewDirection(0.,0.,1.)
90{
91 theParameters = G4EmParameters::Instance();
94
95 lowestKinEnergy = 10*CLHEP::eV;
96
97 geomMin = 0.05*CLHEP::nm;
98 minDisplacement2 = geomMin*geomMin;
99
101
102 modelManager = new G4EmModelManager();
103 emManager = G4LossTableManager::Instance();
104 mscModels.reserve(2);
105 emManager->Register(this);
106}
107
108//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo....
109
111{
112 delete modelManager;
113 emManager->DeRegister(this);
114}
115
116//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo....
117
119 const G4Region* region)
120{
121 if(nullptr == ptr) { return; }
122 G4VEmFluctuationModel* fm = nullptr;
123 modelManager->AddEmModel(order, ptr, fm, region);
125}
126
127//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo....
128
130{
131 if(nullptr == ptr) { return; }
132 if(!mscModels.empty()) {
133 for(auto & msc : mscModels) { if(msc == ptr) { return; } }
134 }
135 mscModels.push_back(ptr);
136}
137
138//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo....
139
140void
142{
143 G4bool master = emManager->IsMaster();
144 if (nullptr == firstParticle) { firstParticle = &part; }
145
146 emManager->PreparePhysicsTable(&part, this);
147 currParticle = nullptr;
148
149 if(firstParticle == &part) {
150 baseMat = emManager->GetTableBuilder()->GetBaseMaterialFlag();
151 G4EmTableUtil::PrepareMscProcess(this, part, modelManager,
152 stepLimit, facrange,
153 latDisplacement, master,
154 isIon, baseMat);
155
156 numberOfModels = modelManager->NumberOfModels();
157 currentModel = GetModelByIndex(0);
158
159 if (nullptr == safetyHelper) {
161 ->GetSafetyHelper();
162 safetyHelper->InitialiseHelper();
163 }
164 }
165}
166
167//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo....
168
170{
171 G4bool master = emManager->IsMaster();
172
173 if(firstParticle == &part) {
174 emManager->BuildPhysicsTable(&part);
175 }
176 const G4VMultipleScattering* ptr = this;
177 if(!master) {
178 ptr = static_cast<const G4VMultipleScattering*>(GetMasterProcess());
179 }
180
181 G4EmTableUtil::BuildMscProcess(this, ptr, part, firstParticle,
182 numberOfModels, master);
183}
184
185//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo....
186
187void G4VMultipleScattering::StreamInfo(std::ostream& outFile,
188 const G4ParticleDefinition& part, G4bool rst) const
189{
190 G4String indent = (rst ? " " : "");
191 outFile << G4endl << indent << GetProcessName() << ": ";
192 if (!rst) outFile << " for " << part.GetParticleName();
193 outFile << " SubType= " << GetProcessSubType() << G4endl;
194 modelManager->DumpModelList(outFile, verboseLevel);
195}
196
197//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo....
198
200{
201 G4VEnergyLossProcess* eloss = nullptr;
202 if(track->GetParticleDefinition() != currParticle) {
203 currParticle = track->GetParticleDefinition();
204 fIonisation = emManager->GetEnergyLossProcess(currParticle);
205 eloss = fIonisation;
206 }
207 for(G4int i=0; i<numberOfModels; ++i) {
209 msc->StartTracking(track);
210 if(nullptr != eloss) {
211 msc->SetIonisation(eloss, currParticle);
212 }
213 }
214}
215
216//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo......
217
219 const G4Track& track,
220 G4double,
221 G4double currentMinimalStep,
222 G4double&,
223 G4GPILSelection* selection)
224{
225 // get Step limit proposed by the process
226 *selection = NotCandidateForSelection;
227 physStepLimit = gPathLength = tPathLength = currentMinimalStep;
228
229 G4double ekin = track.GetKineticEnergy();
230 /*
231 G4cout << "MSC::AlongStepGPIL: Ekin= " << ekin
232 << " " << currParticle->GetParticleName()
233 << " currMod " << currentModel
234 << G4endl;
235 */
236 // isIon flag is used only to select a model
237 if(isIon) {
238 ekin *= proton_mass_c2/track.GetParticleDefinition()->GetPDGMass();
239 }
240 const G4MaterialCutsCouple* couple = track.GetMaterialCutsCouple();
241
242 // select new model, static cast is possible in this class
243 if(1 < numberOfModels) {
244 currentModel =
245 static_cast<G4VMscModel*>(SelectModel(ekin,couple->GetIndex()));
246 }
247 currentModel->SetCurrentCouple(couple);
248 // msc is active is model is active, energy above the limit,
249 // and step size is above the limit;
250 // if it is active msc may limit the step
251 if(currentModel->IsActive(ekin) && tPathLength > geomMin
252 && ekin >= lowestKinEnergy) {
253 isActive = true;
254 tPathLength =
255 currentModel->ComputeTruePathLengthLimit(track, gPathLength);
256 if (tPathLength < physStepLimit) {
257 *selection = CandidateForSelection;
258 }
259 } else {
260 isActive = false;
261 gPathLength = DBL_MAX;
262 }
263
264 //if(currParticle->GetPDGMass() > GeV)
265 /*
266 G4cout << "MSC::AlongStepGPIL: Ekin= " << ekin
267 << " " << currParticle->GetParticleName()
268 << " gPathLength= " << gPathLength
269 << " tPathLength= " << tPathLength
270 << " currentMinimalStep= " << currentMinimalStep
271 << " isActive " << isActive << G4endl;
272 */
273 return gPathLength;
274}
275
276//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo....
277
285
286//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo....
287
290{
291 fParticleChange.InitialiseMSC(track, step);
292 fNewPosition = fParticleChange.GetProposedPosition();
293 fPositionChanged = false;
294
295 G4double geomLength = step.GetStepLength();
296
297 // very small step - no msc
298 if(!isActive) {
299 tPathLength = geomLength;
300
301 // sample msc
302 } else {
303 G4double range =
304 currentModel->GetRange(currParticle,track.GetKineticEnergy(),
305 track.GetMaterialCutsCouple());
306
307 tPathLength = currentModel->ComputeTrueStepLength(geomLength);
308
309 /*
310 if(currParticle->GetPDGMass() > 0.9*GeV)
311 G4cout << "G4VMsc::AlongStepDoIt: GeomLength= "
312 << geomLength
313 << " tPathLength= " << tPathLength
314 << " physStepLimit= " << physStepLimit
315 << " dr= " << range - tPathLength
316 << " ekin= " << track.GetKineticEnergy() << G4endl;
317 */
318 // protection against wrong t->g->t conversion
319 tPathLength = std::min(tPathLength, physStepLimit);
320
321 // do not sample scattering at the last or at a small step
322 if(tPathLength < range && tPathLength > geomMin) {
323
324 static const G4double minSafety = 1.20*CLHEP::nm;
325 static const G4double sFact = 0.99;
326
327 G4ThreeVector displacement = currentModel->SampleScattering(
328 step.GetPostStepPoint()->GetMomentumDirection(),minSafety);
329
330 G4double r2 = displacement.mag2();
331 //G4cout << " R= " << sqrt(r2) << " Rmin= " << sqrt(minDisplacement2)
332 // << " flag= " << fDispBeyondSafety << G4endl;
333 if(r2 > minDisplacement2) {
334
335 fPositionChanged = true;
336 G4double dispR = std::sqrt(r2);
337 G4double postSafety =
338 sFact*safetyHelper->ComputeSafety(fNewPosition, dispR);
339 //G4cout<<" R= "<< dispR<<" postSafety= "<<postSafety<<G4endl;
340
341 // far away from geometry boundary
342 if(postSafety > 0.0 && dispR <= postSafety) {
343 fNewPosition += displacement;
344
345 //near the boundary
346 } else {
347 // displaced point is definitely within the volume
348 //G4cout<<" R= "<<dispR<<" postSafety= "<<postSafety<<G4endl;
349 if(dispR < postSafety) {
350 fNewPosition += displacement;
351
352 // reduced displacement
353 } else if(postSafety > geomMin) {
354 fNewPosition += displacement*(postSafety/dispR);
355
356 // very small postSafety
357 } else {
358 fPositionChanged = false;
359 }
360 }
361 if(fPositionChanged) {
362 safetyHelper->ReLocateWithinVolume(fNewPosition);
363 fParticleChange.ProposePosition(fNewPosition);
364 }
365 }
366 }
367 }
368 fParticleChange.ProposeTrueStepLength(tPathLength);
369 return &fParticleChange;
370}
371
372//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo....
373
375 const G4Track& track,
376 G4double previousStepSize,
377 G4double currentMinimalStep,
378 G4double& currentSafety)
379{
381 G4double x = AlongStepGetPhysicalInteractionLength(track,previousStepSize,
382 currentMinimalStep,
383 currentSafety,
384 &selection);
385 return x;
386}
387
388//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo....
389
391 const G4Track& track,
392 G4double previousStepSize,
393 G4double currentMinimalStep,
394 G4double& currentSafety)
395{
396 return GetContinuousStepLimit(track,previousStepSize,currentMinimalStep,
397 currentSafety);
398}
399
400//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo....
401
408
409//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo......
410
411G4bool
413 const G4String& directory,
414 G4bool ascii)
415{
416 G4bool yes = true;
417 if(part != firstParticle || !emManager->IsMaster()) { return yes; }
418
419 return G4EmTableUtil::StoreMscTable(this, part, directory,
420 numberOfModels, verboseLevel,
421 ascii);
422}
423
424//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo......
425
426G4bool
428 const G4String&,
429 G4bool)
430{
431 return true;
432}
433
434//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo....
435
436void G4VMultipleScattering::ProcessDescription(std::ostream& outFile) const
437{
438 if(nullptr != firstParticle) {
439 StreamInfo(outFile, *firstParticle, true);
440 }
441}
442
443//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo....
444
@ fMultipleScattering
G4double condition(const G4ErrorSymMatrix &m)
G4ForceCondition
@ NotForced
@ Forced
G4GPILSelection
@ CandidateForSelection
@ NotCandidateForSelection
G4ProcessType
@ fElectromagnetic
CLHEP::Hep3Vector G4ThreeVector
double G4double
Definition G4Types.hh:83
bool G4bool
Definition G4Types.hh:86
int G4int
Definition G4Types.hh:85
#define G4endl
Definition G4ios.hh:67
double mag2() const
static G4EmParameters * Instance()
static void PrepareMscProcess(G4VMultipleScattering *proc, const G4ParticleDefinition &part, G4EmModelManager *modelManager, G4MscStepLimitType &stepLimit, G4double &facrange, G4bool &latDisplacement, G4bool &master, G4bool &isIon, G4bool &baseMat)
static void BuildMscProcess(G4VMultipleScattering *proc, const G4VMultipleScattering *masterProc, const G4ParticleDefinition &part, const G4ParticleDefinition *firstPart, G4int nModels, G4bool master)
static G4bool StoreMscTable(G4VMultipleScattering *proc, const G4ParticleDefinition *part, const G4String &directory, const G4int nModels, const G4int verb, const G4bool ascii)
static G4LossTableManager * Instance()
const G4String & GetParticleName() const
const G4ThreeVector & GetMomentumDirection() const
G4double GetStepLength() const
G4StepPoint * GetPostStepPoint() const
const G4ParticleDefinition * GetParticleDefinition() const
G4double GetKineticEnergy() const
const G4MaterialCutsCouple * GetMaterialCutsCouple() const
static G4TransportationManager * GetTransportationManager()
G4SafetyHelper * GetSafetyHelper() const
G4VContinuousDiscreteProcess(const G4String &, G4ProcessType aType=fNotDefined)
void SetParticleChange(G4VParticleChange *, G4VEmFluctuationModel *f=nullptr)
virtual void StartTracking(G4Track *)
void SetIonisation(G4VEnergyLossProcess *, const G4ParticleDefinition *part)
void AddEmModel(G4int order, G4VMscModel *, const G4Region *region=nullptr)
G4double GetMeanFreePath(const G4Track &track, G4double, G4ForceCondition *condition) override
G4ParticleChangeForMSC fParticleChange
G4VMscModel * GetModelByIndex(G4int idx, G4bool ver=false) const
void StartTracking(G4Track *) override
void PreparePhysicsTable(const G4ParticleDefinition &) override
void BuildPhysicsTable(const G4ParticleDefinition &) override
G4double ContinuousStepLimit(const G4Track &track, G4double previousStepSize, G4double currentMinimalStep, G4double &currentSafety)
G4VMultipleScattering(const G4String &name="msc", G4ProcessType type=fElectromagnetic)
void ProcessDescription(std::ostream &outFile) const override
G4bool RetrievePhysicsTable(const G4ParticleDefinition *, const G4String &directory, G4bool ascii) override
void SetEmModel(G4VMscModel *, G4int idx=0)
G4double AlongStepGetPhysicalInteractionLength(const G4Track &, G4double previousStepSize, G4double currentMinimalStep, G4double &currentSafety, G4GPILSelection *selection) override
void StreamInfo(std::ostream &outFile, const G4ParticleDefinition &, G4bool rst=false) const
G4double PostStepGetPhysicalInteractionLength(const G4Track &, G4double previousStepSize, G4ForceCondition *condition) override
G4VParticleChange * AlongStepDoIt(const G4Track &, const G4Step &) override
G4bool StorePhysicsTable(const G4ParticleDefinition *, const G4String &directory, G4bool ascii=false) override
G4VEmModel * SelectModel(G4double kinEnergy, size_t idx)
G4double GetContinuousStepLimit(const G4Track &track, G4double previousStepSize, G4double currentMinimalStep, G4double &currentSafety) override
void SetVerboseLevel(G4int value)
const G4VProcess * GetMasterProcess() const
G4int verboseLevel
void SetProcessSubType(G4int)
G4VParticleChange * pParticleChange
G4int GetProcessSubType() const
const G4String & GetProcessName() const
#define DBL_MAX
Definition templates.hh:62