Geant4 11.1.1
Toolkit for the simulation of the passage of particles through matter
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Pages
G4EmSaturation.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: G4EmSaturation
33//
34// Author: Vladimir Ivanchenko
35//
36// Creation date: 18.02.2008
37//
38// Modifications:
39//
40// -------------------------------------------------------------
41
42//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo....
43//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo....
44
45#include "G4EmSaturation.hh"
47#include "G4SystemOfUnits.hh"
48#include "G4LossTableManager.hh"
49#include "G4NistManager.hh"
50#include "G4Material.hh"
52#include "G4ParticleTable.hh"
53
54//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo....
55
56std::size_t G4EmSaturation::nMaterials = 0;
57std::vector<G4double> G4EmSaturation::massFactors;
58std::vector<G4double> G4EmSaturation::effCharges;
59std::vector<G4double> G4EmSaturation::g4MatData;
60std::vector<G4String> G4EmSaturation::g4MatNames;
61
63{
64 verbose = verb;
65 nWarnings = nG4Birks = 0;
66
67 electron = nullptr;
68 proton = nullptr;
71}
72
73//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo....
74
76
77//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo....
78
80 const G4ParticleDefinition* p,
81 const G4MaterialCutsCouple* couple,
82 G4double length,
83 G4double edep,
84 G4double niel) const
85{
86 // no energy deposition
87 if(edep <= 0.0) { return 0.0; }
88
89 // zero step length may happens only if step limiter process
90 // is applied, in that case saturation should not be applied
91 if(length <= 0.0) { return edep; }
92
93 G4double evis = edep;
94 G4double bfactor = couple->GetMaterial()->GetIonisation()->GetBirksConstant();
95
96 if(bfactor > 0.0) {
97
98 // atomic relaxations for gamma incident
99 if(22 == p->GetPDGEncoding()) {
100 //G4cout << "%% gamma edep= " << edep/keV << " keV " << G4endl;
101 evis /= (1.0 + bfactor*edep/
102 G4LossTableManager::Instance()->GetRange(electron,edep,couple));
103
104 // energy loss
105 } else {
106
107 // protections
108 G4double nloss = std::max(niel, 0.0);
109 G4double eloss = edep - nloss;
110
111 // neutrons and neutral hadrons
112 if(0.0 == p->GetPDGCharge() || eloss < 0.0) {
113 nloss = edep;
114 eloss = 0.0;
115 } else {
116
117 // continues energy loss
118 eloss /= (1.0 + bfactor*eloss/length);
119 }
120 // non-ionizing energy loss
121 if(nloss > 0.0) {
122 std::size_t idx = couple->GetMaterial()->GetIndex();
123 G4double escaled = nloss*massFactors[idx];
124 /*
125 G4cout << "%% p edep= " << nloss/keV << " keV Escaled= "
126 << escaled << " MeV in " << couple->GetMaterial()->GetName()
127 << " " << p->GetParticleName()
128 << G4endl;
129 G4cout << proton->GetParticleName() << G4endl;
130 */
132 ->GetRange(proton,escaled,couple)/effCharges[idx];
133 nloss /= (1.0 + bfactor*nloss/range);
134 }
135 evis = eloss + nloss;
136 }
137 }
138 return evis;
139}
140
141//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo....
142
144{
145 if(nMaterials == G4Material::GetNumberOfMaterials()) { return; }
147 massFactors.resize(nMaterials, 1.0);
148 effCharges.resize(nMaterials, 1.0);
149
150 if(0 == nG4Birks) { InitialiseG4materials(); }
151
152 for(std::size_t i=0; i<nMaterials; ++i) {
153 InitialiseBirksCoefficient((*G4Material::GetMaterialTable())[i]);
154 }
155 if(verbose > 0) { DumpBirksCoefficients(); }
156}
157
158//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo....
159
161{
162 if(0 == nG4Birks) { InitialiseG4materials(); }
163
164 G4String name = mat->GetName();
165 // is this material in the vector?
166
167 for(G4int j=0; j<nG4Birks; ++j) {
168 if(name == g4MatNames[j]) {
169 if(verbose > 0)
170 G4cout << "### G4EmSaturation::FindG4BirksCoefficient for "
171 << name << " is " << g4MatData[j]*MeV/mm << " mm/MeV "
172 << G4endl;
173 return g4MatData[j];
174 }
175 }
176 return 0.0;
177}
178
179//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo....
180
181void G4EmSaturation::InitialiseBirksCoefficient(const G4Material* mat)
182{
183 // electron and proton should exist in any case
184 if(nullptr == electron) {
187 if(nullptr == electron) {
188 G4Exception("G4EmSaturation::InitialiseBirksCoefficient", "em0001",
189 FatalException, "electron should exist");
190 }
191 }
192
193 G4double curBirks = mat->GetIonisation()->GetBirksConstant();
194
195 G4String name = mat->GetName();
196
197 // material has no Birks coeffitient defined
198 // seach in the Geant4 list
199 if(curBirks == 0.0) {
200 for(G4int j=0; j<nG4Birks; ++j) {
201 if(name == g4MatNames[j]) {
202 mat->GetIonisation()->SetBirksConstant(g4MatData[j]);
203 curBirks = g4MatData[j];
204 break;
205 }
206 }
207 }
208
209 if(curBirks == 0.0) { return; }
210
211 // compute mean mass ratio
212 G4double curRatio = 0.0;
213 G4double curChargeSq = 0.0;
214 G4double norm = 0.0;
215 const G4ElementVector* theElementVector = mat->GetElementVector();
216 const G4double* theAtomNumDensityVector = mat->GetVecNbOfAtomsPerVolume();
217 std::size_t nelm = mat->GetNumberOfElements();
218 for (std::size_t i=0; i<nelm; ++i) {
219 const G4Element* elm = (*theElementVector)[i];
220 G4double Z = elm->GetZ();
221 G4double w = Z*Z*theAtomNumDensityVector[i];
222 curRatio += w/nist->GetAtomicMassAmu(G4int(Z));
223 curChargeSq = Z*Z*w;
224 norm += w;
225 }
226 curRatio *= proton_mass_c2/norm;
227 curChargeSq /= norm;
228
229 // store results
230 std::size_t idx = mat->GetIndex();
231 massFactors[idx] = curRatio;
232 effCharges[idx] = curChargeSq;
233}
234
235//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo....
236
238{
239 G4cout << "### Birks coefficients used in run time" << G4endl;
241 for(std::size_t i=0; i<nMaterials; ++i) {
242 const G4Material* mat = (*mtable)[i];
244 if(br > 0.0) {
245 G4cout << " " << mat->GetName() << " "
246 << br*MeV/mm << " mm/MeV" << " "
247 << br*mat->GetDensity()*MeV*cm2/g
248 << " g/cm^2/MeV massFactor= " << massFactors[i]
249 << " effCharge= " << effCharges[i] << G4endl;
250 }
251 }
252}
253
254//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo....
255
257{
258 if(nG4Birks > 0) {
259 G4cout << "### Birks coefficients for Geant4 materials" << G4endl;
260 for(G4int i=0; i<nG4Birks; ++i) {
261 G4cout << " " << g4MatNames[i] << " "
262 << g4MatData[i]*MeV/mm << " mm/MeV" << G4endl;
263 }
264 }
265}
266
267//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo....
268
269void G4EmSaturation::InitialiseG4materials()
270{
271 nG4Birks = 4;
272 g4MatData.reserve(nG4Birks);
273
274 // M.Hirschberg et al., IEEE Trans. Nuc. Sci. 39 (1992) 511
275 // SCSN-38 kB = 0.00842 g/cm^2/MeV; rho = 1.06 g/cm^3
276 g4MatNames.push_back("G4_POLYSTYRENE");
277 g4MatData.push_back(0.07943*mm/MeV);
278
279 // C.Fabjan (private communication)
280 // kB = 0.006 g/cm^2/MeV; rho = 7.13 g/cm^3
281 g4MatNames.push_back("G4_BGO");
282 g4MatData.push_back(0.008415*mm/MeV);
283
284 // A.Ribon analysis of publications
285 // Scallettar et al., Phys. Rev. A25 (1982) 2419.
286 // NIM A 523 (2004) 275.
287 // kB = 0.022 g/cm^2/MeV; rho = 1.396 g/cm^3;
288 // ATLAS Efield = 10 kV/cm provide the strongest effect
289 // kB = 0.1576*mm/MeV
290 // A. Kiryunin and P.Strizenec "Geant4 hadronic
291 // working group meeting " kB = 0.041/9.13 g/cm^2/MeV
292 g4MatNames.push_back("G4_lAr");
293 g4MatData.push_back(0.032*mm/MeV);
294
295 //G4_BARIUM_FLUORIDE
296 //G4_CESIUM_IODIDE
297 //G4_GEL_PHOTO_EMULSION
298 //G4_PHOTO_EMULSION
299 //G4_PLASTIC_SC_VINYLTOLUENE
300 //G4_SODIUM_IODIDE
301 //G4_STILBENE
302 //G4_lAr
303
304 //G4_PbWO4 - CMS value
305 g4MatNames.push_back("G4_PbWO4");
306 g4MatData.push_back(0.0333333*mm/MeV);
307
308 //G4_Lucite
309
310}
311
312//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo....
std::vector< const G4Element * > G4ElementVector
@ FatalException
void G4Exception(const char *originOfException, const char *exceptionCode, G4ExceptionSeverity severity, const char *description)
Definition: G4Exception.cc:59
std::vector< G4Material * > G4MaterialTable
double G4double
Definition: G4Types.hh:83
int G4int
Definition: G4Types.hh:85
const G4int Z[17]
#define G4endl
Definition: G4ios.hh:57
G4GLOB_DLL std::ostream G4cout
G4double GetZ() const
Definition: G4Element.hh:131
virtual ~G4EmSaturation()
virtual G4double VisibleEnergyDeposition(const G4ParticleDefinition *, const G4MaterialCutsCouple *, G4double length, G4double edepTotal, G4double edepNIEL=0.0) const
G4double FindG4BirksCoefficient(const G4Material *)
void DumpBirksCoefficients()
void InitialiseG4Saturation()
void DumpG4BirksCoefficients()
G4EmSaturation(G4int verb)
void SetBirksConstant(G4double value)
G4double GetBirksConstant() const
static G4LossTableManager * Instance()
G4double GetRange(const G4ParticleDefinition *aParticle, G4double kineticEnergy, const G4MaterialCutsCouple *couple)
const G4Material * GetMaterial() const
G4double GetDensity() const
Definition: G4Material.hh:175
const G4ElementVector * GetElementVector() const
Definition: G4Material.hh:185
static size_t GetNumberOfMaterials()
Definition: G4Material.cc:684
G4IonisParamMat * GetIonisation() const
Definition: G4Material.hh:221
size_t GetNumberOfElements() const
Definition: G4Material.hh:181
const G4double * GetVecNbOfAtomsPerVolume() const
Definition: G4Material.hh:201
static G4MaterialTable * GetMaterialTable()
Definition: G4Material.cc:677
const G4String & GetName() const
Definition: G4Material.hh:172
size_t GetIndex() const
Definition: G4Material.hh:255
static G4NistManager * Instance()
G4double GetAtomicMassAmu(const G4String &symb) const
G4double GetPDGCharge() const
G4ParticleDefinition * FindParticle(G4int PDGEncoding)
static G4ParticleTable * GetParticleTable()
const char * name(G4int ptype)