source: trunk/examples/extended/electromagnetic/TestEm7/include/G4ScreenedNuclearRecoil.hh@ 1307

Last change on this file since 1307 was 1230, checked in by garnier, 16 years ago

update to geant4.9.3

File size: 17.7 KB
RevLine 
[807]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//
[1230]27// G4ScreenedNuclearRecoil.hh,v 1.24 2008/05/01 19:58:59 marcus Exp
28// GEANT4 tag
[807]29//
30//
31//
32// Class Description
33// Process for screened electromagnetic nuclear elastic scattering;
34// Physics comes from:
35// Marcus H. Mendenhall and Robert A. Weller,
36// "Algorithms for the rapid computation of classical cross sections
37// for screened Coulomb collisions "
38// Nuclear Instruments and Methods in Physics Research B58 (1991) 11-17
39// The only input required is a screening function phi(r/a) which is the ratio
40// of the actual interatomic potential for two atoms with atomic numbers Z1 and Z2,
41// to the unscreened potential Z1*Z2*e^2/r where e^2 is elm_coupling in Geant4 units
42// the actual screening tables are computed externally in a python module "screened_scattering.py"
43// to allow very specific screening functions to be added if desired, without messing
44// with the insides of this code.
45//
46// First version, April 2004, Marcus H. Mendenhall, Vanderbilt University
[1230]47// May 1, 2008 -- Added code to allow process to have zero cross section above max energy, to coordinate with G4MSC. -- mhm
[807]48//
49// Class Description - End
50
51
52#ifndef G4ScreenedNuclearRecoil_h
53#define G4ScreenedNuclearRecoil_h 1
54
55#include "globals.hh"
56#include "G4VDiscreteProcess.hh"
[1230]57#include "G4ParticleChange.hh"
[807]58#include "c2_function.hh"
59
60#include <map>
61#include <vector>
62
[1230]63class G4VNIELPartition;
[807]64
[1230]65typedef c2_const_ptr<G4double> G4_c2_const_ptr;
66typedef c2_ptr<G4double> G4_c2_ptr;
67typedef c2_function<G4double> G4_c2_function;
68
[807]69typedef struct G4ScreeningTables {
70 G4double z1, z2, m1, m2, au, emin;
[1230]71 G4_c2_const_ptr EMphiData;
[807]72} G4ScreeningTables;
73
74// A class for loading ScreenedCoulombCrossSections
75class G4ScreenedCoulombCrossSectionInfo
76{
77public:
[1230]78 G4ScreenedCoulombCrossSectionInfo() { }
79 ~G4ScreenedCoulombCrossSectionInfo() { }
80
81 static const char* CVSHeaderVers() { return
82 "G4ScreenedNuclearRecoil.hh,v 1.24 2008/05/01 19:58:59 marcus Exp GEANT4 tag ";
83 }
84 static const char* CVSFileVers();
[807]85};
86
87// A class for loading ScreenedCoulombCrossSections
88class G4ScreenedCoulombCrossSection : public G4ScreenedCoulombCrossSectionInfo
89{
90public:
91
92 G4ScreenedCoulombCrossSection() : verbosity(1) { }
93 G4ScreenedCoulombCrossSection(const G4ScreenedCoulombCrossSection &src) :
94 G4ScreenedCoulombCrossSectionInfo(),verbosity(src.verbosity) { }
95 virtual ~G4ScreenedCoulombCrossSection();
96
97 typedef std::map<G4int, G4ScreeningTables> ScreeningMap;
98
99 // a local, fast-access mapping of a particle's Z to its full definition
100 typedef std::map<G4int, class G4ParticleDefinition *> ParticleCache;
101
102 // LoadData is called by G4ScreenedNuclearRecoil::GetMeanFreePath
103 // It loads the data tables, builds the elemental cross-section tables.
104 virtual void LoadData(G4String screeningKey, G4int z1, G4double m1, G4double recoilCutoff) = 0;
105
106 // BuildMFPTables is called by G4ScreenedNuclearRecoil::GetMeanFreePath to build the MFP tables for each material
107 void BuildMFPTables(void); // scan the MaterialsTable and construct MFP tables
108
109 virtual G4ScreenedCoulombCrossSection *create() = 0; // a 'virtual constructor' which clones the class
110 const G4ScreeningTables *GetScreening(G4int Z) { return &(screeningData[Z]); }
111 void SetVerbosity(G4int v) { verbosity=v; }
112
113 // this process needs element selection weighted only by number density
114 G4ParticleDefinition* SelectRandomUnweightedTarget(const G4MaterialCutsCouple* couple);
115
116 enum { nMassMapElements=116 };
117
118 G4double standardmass(G4int z1) { return z1 <= nMassMapElements ? massmap[z1] : 2.5*z1; }
119
120 // get the mean-free-path table for the indexed material
[1230]121 const G4_c2_function * operator [] (G4int materialIndex) {
122 return MFPTables.find(materialIndex)!=MFPTables.end() ? &(MFPTables[materialIndex].get()) : (G4_c2_function *)0;
[807]123 }
124
125protected:
126 ScreeningMap screeningData; // screening tables for each element
127 ParticleCache targetMap;
128 G4int verbosity;
[1230]129 std::map<G4int, G4_c2_const_ptr > sigmaMap; // total cross section for each element
130 std::map<G4int, G4_c2_const_ptr > MFPTables; // MFP for each material
[807]131
132private:
133 static const G4double massmap[nMassMapElements+1];
134
135};
136
137typedef struct G4CoulombKinematicsInfo {
138 G4double impactParameter;
139 G4ScreenedCoulombCrossSection *crossSection;
140 G4double a1, a2, sinTheta, cosTheta, sinZeta, cosZeta, eRecoil;
[1230]141 G4ParticleDefinition *recoilIon;
142 const G4Material *targetMaterial;
143} G4CoulombKinematicsInfo;
[807]144
145class G4ScreenedCollisionStage {
146public:
147 virtual void DoCollisionStep(class G4ScreenedNuclearRecoil *master,
148 const class G4Track& aTrack, const class G4Step& aStep)=0;
149 virtual ~G4ScreenedCollisionStage() {}
150};
151
152class G4ScreenedCoulombClassicalKinematics: public G4ScreenedCoulombCrossSectionInfo, public G4ScreenedCollisionStage {
153
154public:
[1230]155 G4ScreenedCoulombClassicalKinematics();
[807]156 virtual void DoCollisionStep(class G4ScreenedNuclearRecoil *master,
157 const class G4Track& aTrack, const class G4Step& aStep);
158
159 G4bool DoScreeningComputation(class G4ScreenedNuclearRecoil *master,
160 const G4ScreeningTables *screen,
161 G4double eps, G4double beta);
[1230]162 virtual ~G4ScreenedCoulombClassicalKinematics() { }
163protected:
164 // the c2_functions we need to do the work.
165 c2_const_plugin_function_p<G4double> &phifunc;
166 c2_linear_p<G4double> &xovereps;
167 G4_c2_ptr diff;
168
[807]169};
170
171class G4SingleScatter: public G4ScreenedCoulombCrossSectionInfo, public G4ScreenedCollisionStage {
172
173public:
174 G4SingleScatter() { }
175 virtual void DoCollisionStep(class G4ScreenedNuclearRecoil *master,
176 const class G4Track& aTrack, const class G4Step& aStep);
177 virtual ~G4SingleScatter() {}
178};
179
[1230]180/**
181 \brief A process which handles screened Coulomb collisions between nuclei
182
183*/
184
[807]185class G4ScreenedNuclearRecoil : public G4ScreenedCoulombCrossSectionInfo, public G4VDiscreteProcess
186{
187public:
188
189 friend class G4ScreenedCollisionStage;
190
[1230]191 /// \brief Construct the process and set some physics parameters for it.
192 /// \param processName the name to assign the process
193 /// \param ScreeningKey the name of a screening function to use.
194 /// The default functions are "zbl" (recommended for soft scattering),
195 /// "lj" (recommended for backscattering) and "mol" (Moliere potential)
196 /// \param GenerateRecoils if frue, ions struck by primary are converted into new moving particles.
197 /// If false, energy is deposited, but no new moving ions are created.
198 /// \param RecoilCutoff energy below which no new moving particles will be created,
199 /// even if \a GenerateRecoils is true.
200 /// Also, a moving primary particle will be stopped if its energy falls below this limit.
201 /// \param PhysicsCutoff the energy transfer to which screening tables are calucalted.
202 /// There is no really
203 /// compelling reason to change it from the 10.0 eV default. However, see the paper on running this
204 /// in thin targets for further discussion, and its interaction with SetMFPScaling()
[807]205 G4ScreenedNuclearRecoil(const G4String& processName = "ScreenedElastic",
206 const G4String &ScreeningKey="zbl", G4bool GenerateRecoils=1,
207 G4double RecoilCutoff=100.0*eV, G4double PhysicsCutoff=10.0*eV);
[1230]208 /// \brief destructor
[807]209 virtual ~G4ScreenedNuclearRecoil();
[1230]210 /// \brief used internally by Geant4 machinery
[807]211 virtual G4double GetMeanFreePath(const G4Track&, G4double, G4ForceCondition* );
[1230]212 /// \brief used internally by Geant4 machinery
[807]213 virtual G4VParticleChange* PostStepDoIt(const G4Track& aTrack, const G4Step& aStep);
[1230]214 /// \brief test if a prticle of type \a aParticleType can use this process
215 /// \param aParticleType the particle to test
[807]216 virtual G4bool IsApplicable(const G4ParticleDefinition& aParticleType);
[1230]217 /// \brief Build physics tables in advance. Not Implemented.
218 /// \param aParticleType the type of particle to build tables for
[807]219 virtual void BuildPhysicsTable(const G4ParticleDefinition&) { }
[1230]220 /// \brief Export physics tables for persistency. Not Implemented.
221 /// \param aParticleType the type of particle to build tables for
[807]222 virtual void DumpPhysicsTable(const G4ParticleDefinition& aParticleType);
[1230]223 /// \brief deterine if the moving particle is within the strong force range of the selected nucleus
224 /// \param A the nucleon number of the beam
225 /// \param A1 the nucleon number of the target
226 /// \param apsis the distance of closest approach
[807]227 virtual G4bool CheckNuclearCollision(G4double A, G4double A1, G4double apsis); // return true if hard collision
228
229 virtual G4ScreenedCoulombCrossSection *GetNewCrossSectionHandler(void);
230
[1230]231 /// \brief Get non-ionizing energy loss for last step
232 G4double GetNIEL() const { return NIEL; }
[807]233
[1230]234 /// \brief clear precomputed screening tables
[807]235 void ResetTables(); // clear all data tables to allow changing energy cutoff, materials, etc.
[1230]236
237 /// \brief set the upper energy beyond which this process has no cross section
238 ///
239 /// This funciton is used to coordinate this process with G4MSC. Typically, G4MSC should
240 /// not be allowed to operate in a range which overlaps that of this process. The criterion which is most reasonable
241 /// is that the transition should be somewhere in the modestly relativistic regime (500 MeV/u for example).
242 /// \param energy energy per nucleon for the cutoff
243 void SetMaxEnergyForScattering(G4double energy) { processMaxEnergy=energy; }
244 /// \brief find out what screening funciton we are using
[807]245 std::string GetScreeningKey() const { return screeningKey; }
[1230]246 /// \brief enable or disable all energy deposition by this process
247 /// \param flag if true, enable deposition of energy (the default). If false, disable deposition.
[807]248 void AllowEnergyDeposition(G4bool flag) { registerDepositedEnergy=flag; }
[1230]249 /// \brief get flag indicating whether deposition is enabled
[807]250 G4bool GetAllowEnergyDeposition() const { return registerDepositedEnergy; }
[1230]251 /// \brief enable or disable the generation of recoils.
252 /// If recoils are disabled, the energy they would have received is just deposited.
253 /// \param flag if true, create recoil ions in cases in which the energy is above the recoilCutoff.
254 /// If false, just deposit the energy.
[807]255 void EnableRecoils(G4bool flag) { generateRecoils=flag; }
[1230]256 /// \brief find out if generation of recoils is enabled.
[807]257 G4bool GetEnableRecoils() const { return generateRecoils; }
[1230]258 /// \brief set the mean free path scaling as specified
259 /// \param scale the factor by which the default MFP will be scaled.
260 /// Set to less than 1 for very thin films, typically, to sample multiple scattering,
261 /// or to greater than 1 for quick simulaitons with a very long flight path.
[807]262 void SetMFPScaling(G4double scale) { MFPScale=scale; }
[1230]263 /// \brief get the MFPScaling parameter
[807]264 G4double GetMFPScaling() const { return MFPScale; }
[1230]265 /// \brief enable or disable whether this process will skip collisions
266 /// which are close enough they need hadronic phsyics. Default is true (skip close collisions).
267 /// Disabling this results in excess nuclear stopping power.
268 /// \param flag true results in hard collisions being skipped. false allows hard collisions.
[807]269 void AvoidNuclearReactions(G4bool flag) { avoidReactions=flag; }
[1230]270 /// \brief get the flag indicating whether hadronic collisions are ignored.
[807]271 G4bool GetAvoidNuclearReactions() const { return avoidReactions; }
[1230]272 /// \brief set the minimum energy (per nucleon) at which recoils can be generated,
273 /// and the energy (per nucleon) below which all ions are stopped.
274 /// \param energy energy per nucleon
[807]275 void SetRecoilCutoff(G4double energy) { recoilCutoff=energy; }
[1230]276 /// \brief get the recoil cutoff
[807]277 G4double GetRecoilCutoff() const { return recoilCutoff; }
[1230]278 /// \brief set the energy to which screening tables are computed. Typically, this is 10 eV or so, and not often changed.
279 /// \param energy the cutoff energy
[807]280 void SetPhysicsCutoff(G4double energy) { physicsCutoff=energy; ResetTables(); }
[1230]281 /// \brief get the physics cutoff energy.
[807]282 G4double GetPhysicsCutoff() const { return physicsCutoff; }
[1230]283 /// \brief set the pointer to a class for paritioning energy into NIEL
284 /// \brief part the pointer to the class.
285 void SetNIELPartitionFunction(const G4VNIELPartition *part);
286 /// \brief set the cross section boost to provide faster computation of backscattering
287 /// \param fraction the fraction of particles to have their cross section boosted.
288 /// \param HardeningFactor the factor by which to boost the scattering cross section.
[807]289 void SetCrossSectionHardening(G4double fraction, G4double HardeningFactor) {
290 hardeningFraction=fraction;
291 hardeningFactor=HardeningFactor;
292 }
[1230]293 /// \brief get the fraction of particles which will have boosted scattering
[807]294 G4double GetHardeningFraction() const { return hardeningFraction; }
[1230]295 /// \brief get the boost factor in use.
[807]296 G4double GetHardeningFactor() const { return hardeningFactor; }
[1230]297 /// \brief the the interaciton length used in the last scattering.
[807]298 G4double GetCurrentInteractionLength() const { return currentInteractionLength; }
[1230]299 /// \brief set a function to compute screening tables, if the user needs non-standard behavior.
300 /// \param cs a class which constructs the screening tables.
[807]301 void SetExternalCrossSectionHandler(G4ScreenedCoulombCrossSection *cs) {
302 externalCrossSectionConstructor=cs;
303 }
[1230]304 /// \brief get the verbosity.
[807]305 G4int GetVerboseLevel() const { return verboseLevel; }
[1230]306
[807]307 std::map<G4int, G4ScreenedCoulombCrossSection*> &GetCrossSectionHandlers()
308 { return crossSectionHandlers; }
309 void ClearStages(void);
310 void AddStage(G4ScreenedCollisionStage *stage) { collisionStages.push_back(stage); }
311 G4CoulombKinematicsInfo &GetKinematics() { return kinematics; }
312 void SetValidCollision(G4bool flag) { validCollision=flag; }
313 G4bool GetValidCollision() const { return validCollision; }
[1230]314
315 /// \brief get the pointer to our ParticleChange object. for internal use, primarily.
316 class G4ParticleChange &GetParticleChange() { return static_cast<G4ParticleChange &>(*pParticleChange); }
317 /// \brief take the given energy, and use the material information to partition it into NIEL and ionizing energy.
318 void DepositEnergy(G4int z1, G4double a1, const G4Material *material, G4double energy);
[807]319
320protected:
[1230]321 /// \brief the energy per nucleon above which the MFP is constant
322 G4double highEnergyLimit;
323 /// \brief the energy per nucleon below which the MFP is zero
324 G4double lowEnergyLimit;
325 /// \brief the energy per nucleon beyond which the cross section is zero, to cross over to G4MSC
326 G4double processMaxEnergy;
[807]327 G4String screeningKey;
328 G4bool generateRecoils, avoidReactions;
329 G4double recoilCutoff, physicsCutoff;
330 G4bool registerDepositedEnergy;
[1230]331 G4double IonizingLoss, NIEL;
[807]332 G4double MFPScale;
333 G4double hardeningFraction, hardeningFactor;
334
335 G4ScreenedCoulombCrossSection *externalCrossSectionConstructor;
336 std::vector<G4ScreenedCollisionStage *> collisionStages;
337
338 std::map<G4int, G4ScreenedCoulombCrossSection*> crossSectionHandlers;
339
340 G4bool validCollision;
341 G4CoulombKinematicsInfo kinematics;
[1230]342 const G4VNIELPartition *NIELPartitionFunction;
[807]343};
344
345// A customized G4CrossSectionHandler which gets its data from an external program
346class G4NativeScreenedCoulombCrossSection: public G4ScreenedCoulombCrossSection
347{
348public:
349 G4NativeScreenedCoulombCrossSection();
350
351 G4NativeScreenedCoulombCrossSection(const G4NativeScreenedCoulombCrossSection &src)
352 : G4ScreenedCoulombCrossSection(src), phiMap(src.phiMap) { }
353
354 G4NativeScreenedCoulombCrossSection(const G4ScreenedCoulombCrossSection &src) : G4ScreenedCoulombCrossSection(src) { }
355 virtual ~G4NativeScreenedCoulombCrossSection();
356
357 virtual void LoadData(G4String screeningKey, G4int z1, G4double m1, G4double recoilCutoff);
358 virtual G4ScreenedCoulombCrossSection *create()
359 { return new G4NativeScreenedCoulombCrossSection(*this); }
360 // get a list of available keys
361 std::vector<G4String> GetScreeningKeys() const;
362
[1230]363 typedef G4_c2_function &(*ScreeningFunc)(G4int z1, G4int z2, size_t nPoints, G4double rMax, G4double *au);
[807]364
365 void AddScreeningFunction(G4String name, ScreeningFunc fn) {
366 phiMap[name]=fn;
367 }
368
369private:
370 // this is a map used to look up screening function generators
371 std::map<std::string, ScreeningFunc> phiMap;
372};
373
[1230]374G4_c2_function &ZBLScreening(G4int z1, G4int z2, size_t npoints, G4double rMax, G4double *auval);
375G4_c2_function &MoliereScreening(G4int z1, G4int z2, size_t npoints, G4double rMax, G4double *auval);
376G4_c2_function &LJScreening(G4int z1, G4int z2, size_t npoints, G4double rMax, G4double *auval);
377G4_c2_function &LJZBLScreening(G4int z1, G4int z2, size_t npoints, G4double rMax, G4double *auval);
[807]378
379#endif
Note: See TracBrowser for help on using the repository browser.