Geant4.10
 All Data Structures Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Groups Pages
G4MicroElecInelasticModel.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 // G4MicroElecInelasticModel.cc, 2011/08/29 A.Valentin, M. Raine
28 //
29 // Based on the following publications
30 //
31 // - Inelastic cross-sections of low energy electrons in silicon
32 // for the simulation of heavy ion tracks with theGeant4-DNA toolkit,
33 // NSS Conf. Record 2010, pp. 80-85.
34 // - Geant4 physics processes for microdosimetry simulation:
35 // very low energy electromagnetic models for electrons in Si,
36 // NIM B, vol. 288, pp. 66 - 73, 2012.
37 // - Geant4 physics processes for microdosimetry simulation:
38 // very low energy electromagnetic models for protons and
39 // heavy ions in Si, NIM B, vol. 287, pp. 124 - 129, 2012.
40 //
41 //....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo......
42 
44 
45 #include "globals.hh"
46 #include "G4PhysicalConstants.hh"
47 #include "G4SystemOfUnits.hh"
48 #include "G4ios.hh"
49 #include "G4UnitsTable.hh"
50 #include "G4UAtomicDeexcitation.hh"
51 #include "G4LossTableManager.hh"
52 #include "G4ionEffectiveCharge.hh"
53 
54 //....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo....
55 
56 using namespace std;
57 
58 //....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo....
59 
61  const G4String& nam)
62 :G4VEmModel(nam),fAtomDeexcitation(0),isInitialised(false)
63 {
64  nistSi = G4NistManager::Instance()->FindOrBuildMaterial("G4_Si");
65 
66  verboseLevel= 0;
67  // Verbosity scale:
68  // 0 = nothing
69  // 1 = warning for energy non-conservation
70  // 2 = details of energy budget
71  // 3 = calculation of cross sections, file openings, sampling of atoms
72  // 4 = entering in methods
73 
74  if( verboseLevel>0 )
75  {
76  G4cout << "MicroElec inelastic model is constructed " << G4endl;
77  }
78 
79  //Mark this model as "applicable" for atomic deexcitation
80  SetDeexcitationFlag(true);
82 }
83 
84 //....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo....
85 
87 {
88  // Cross section
89 
90  std::map< G4String,G4MicroElecCrossSectionDataSet*,std::less<G4String> >::iterator pos;
91  for (pos = tableData.begin(); pos != tableData.end(); ++pos)
92  {
93  G4MicroElecCrossSectionDataSet* table = pos->second;
94  delete table;
95  }
96 
97  // Final state
98 
99  eVecm.clear();
100  pVecm.clear();
101 
102 }
103 
104 //....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo....
105 
107  const G4DataVector& /*cuts*/)
108 {
109 
110  if (verboseLevel > 3)
111  G4cout << "Calling G4MicroElecInelasticModel::Initialise()" << G4endl;
112 
113  // Energy limits
114 
115  G4String fileElectron("microelec/sigma_inelastic_e_Si");
116  G4String fileProton("microelec/sigma_inelastic_p_Si");
117 
120 
123 
124  G4double scaleFactor = 1e-18 * cm *cm;
125 
126  char *path = getenv("G4LEDATA");
127 
128  // *** ELECTRON
129  electron = electronDef->GetParticleName();
130 
131  tableFile[electron] = fileElectron;
132 
133  lowEnergyLimit[electron] = 16.7 * eV;
134  highEnergyLimit[electron] = 100.0 * MeV;
135 
136  // Cross section
137 
139  tableE->LoadData(fileElectron);
140 
141  tableData[electron] = tableE;
142 
143  // Final state
144 
145  std::ostringstream eFullFileName;
146  eFullFileName << path << "/microelec/sigmadiff_inelastic_e_Si.dat";
147  std::ifstream eDiffCrossSection(eFullFileName.str().c_str());
148 
149  if (!eDiffCrossSection)
150  {
151  G4Exception("G4MicroElecInelasticModel::Initialise","em0003",FatalException,"Missing data file:/microelec/sigmadiff_inelastic_e_Si.dat");
152  }
153 
154  eTdummyVec.push_back(0.);
155  while(!eDiffCrossSection.eof())
156  {
157  double tDummy;
158  double eDummy;
159  eDiffCrossSection>>tDummy>>eDummy;
160  if (tDummy != eTdummyVec.back()) eTdummyVec.push_back(tDummy);
161  for (int j=0; j<6; j++)
162  {
163  eDiffCrossSection>>eDiffCrossSectionData[j][tDummy][eDummy];
164 
165  // SI - only if eof is not reached !
166  if (!eDiffCrossSection.eof()) eDiffCrossSectionData[j][tDummy][eDummy]*=scaleFactor;
167 
168  eVecm[tDummy].push_back(eDummy);
169 
170  }
171  }
172  //
173 
174  // *** PROTON
175 
176  proton = protonDef->GetParticleName();
177 
178  tableFile[proton] = fileProton;
179 
180  lowEnergyLimit[proton] = 50. * keV;
181  highEnergyLimit[proton] = 10. * GeV;
182 
183  // Cross section
184 
186  tableP->LoadData(fileProton);
187 
188  tableData[proton] = tableP;
189 
190  // Final state
191 
192  std::ostringstream pFullFileName;
193  pFullFileName << path << "/microelec/sigmadiff_inelastic_p_Si.dat";
194  std::ifstream pDiffCrossSection(pFullFileName.str().c_str());
195 
196  if (!pDiffCrossSection)
197  {
198  G4Exception("G4MicroElecInelasticModel::Initialise","em0003",FatalException,"Missing data file:/microelec/sigmadiff_inelastic_p_Si.dat");
199  }
200 
201  pTdummyVec.push_back(0.);
202  while(!pDiffCrossSection.eof())
203  {
204  double tDummy;
205  double eDummy;
206  pDiffCrossSection>>tDummy>>eDummy;
207  if (tDummy != pTdummyVec.back()) pTdummyVec.push_back(tDummy);
208  for (int j=0; j<6; j++)
209  {
210  pDiffCrossSection>>pDiffCrossSectionData[j][tDummy][eDummy];
211 
212  // SI - only if eof is not reached !
213  if (!pDiffCrossSection.eof()) pDiffCrossSectionData[j][tDummy][eDummy]*=scaleFactor;
214 
215  pVecm[tDummy].push_back(eDummy);
216  }
217  }
218 
219 
220  if (particle==electronDef)
221  {
222  SetLowEnergyLimit(lowEnergyLimit[electron]);
223  SetHighEnergyLimit(highEnergyLimit[electron]);
224  }
225 
226  if (particle==protonDef)
227  {
228  SetLowEnergyLimit(lowEnergyLimit[proton]);
229  SetHighEnergyLimit(highEnergyLimit[proton]);
230  }
231 
232  if( verboseLevel>0 )
233  {
234  G4cout << "MicroElec Inelastic model is initialized " << G4endl
235  << "Energy range: "
236  << LowEnergyLimit() / eV << " eV - "
237  << HighEnergyLimit() / keV << " keV for "
238  << particle->GetParticleName()
239  << " with mass (amu) " << particle->GetPDGMass()/proton_mass_c2
240  << " and charge " << particle->GetPDGCharge()
241  << G4endl << G4endl ;
242  }
243 
244  //
245 
246  fAtomDeexcitation = G4LossTableManager::Instance()->AtomDeexcitation();
247 
248  if (isInitialised) { return; }
250  isInitialised = true;
251 
252 }
253 
254 //....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo....
255 
257  const G4ParticleDefinition* particleDefinition,
258  G4double ekin,
259  G4double,
260  G4double)
261 {
262  if (verboseLevel > 3)
263  G4cout << "Calling CrossSectionPerVolume() of G4MicroElecInelasticModel" << G4endl;
264 
266 
267  /* if (
268  particleDefinition != G4Proton::ProtonDefinition()
269  &&
270  particleDefinition != G4Electron::ElectronDefinition()
271  &&
272  particleDefinition != G4GenericIon::GenericIonDefinition()
273  )
274 
275  return 0;*/
276 
277  // Calculate total cross section for model
278 
279  G4double lowLim = 0;
280  G4double highLim = 0;
281  G4double sigma=0;
282 
283  const G4String& particleName = particleDefinition->GetParticleName();
284  G4String nameLocal = particleName ;
285 
286  G4double Zeff2 = 1.0;
287  G4double Mion_c2 = particleDefinition->GetPDGMass();
288 
289  if (Mion_c2 > proton_mass_c2)
290  {
291  G4ionEffectiveCharge EffCharge ;
292  G4double Zeff = EffCharge.EffectiveCharge(particleDefinition, material,ekin);
293  Zeff2 = Zeff*Zeff;
294 
295  if (verboseLevel > 3)
296  G4cout << "Before scaling : " << G4endl
297  << "Particle : " << nameLocal << ", mass : " << Mion_c2/proton_mass_c2 << "*mp, charge " << Zeff
298  << ", Ekin (eV) = " << ekin/eV << G4endl ;
299 
300  ekin *= proton_mass_c2/Mion_c2 ;
301  nameLocal = "proton" ;
302 
303  if (verboseLevel > 3)
304  G4cout << "After scaling : " << G4endl
305  << "Particle : " << nameLocal << ", Ekin (eV) = " << ekin/eV << G4endl ;
306  }
307 
308  if (material == nistSi || material->GetBaseMaterial() == nistSi)
309  {
310 
311  std::map< G4String,G4double,std::less<G4String> >::iterator pos1;
312  pos1 = lowEnergyLimit.find(nameLocal);
313  if (pos1 != lowEnergyLimit.end())
314  {
315  lowLim = pos1->second;
316  }
317 
318  std::map< G4String,G4double,std::less<G4String> >::iterator pos2;
319  pos2 = highEnergyLimit.find(nameLocal);
320  if (pos2 != highEnergyLimit.end())
321  {
322  highLim = pos2->second;
323  }
324 
325  if (ekin >= lowLim && ekin < highLim)
326  {
327  std::map< G4String,G4MicroElecCrossSectionDataSet*,std::less<G4String> >::iterator pos;
328  pos = tableData.find(nameLocal);
329 
330  if (pos != tableData.end())
331  {
332  G4MicroElecCrossSectionDataSet* table = pos->second;
333  if (table != 0)
334  {
335  sigma = table->FindValue(ekin);
336  }
337  }
338  else
339  {
340  G4Exception("G4MicroElecInelasticModel::CrossSectionPerVolume","em0002",FatalException,"Model not applicable to particle type.");
341  }
342  }
343  else
344  {
345  if (nameLocal!="e-")
346  {
347  // G4cout << "Particle : " << nameLocal << ", Ekin (eV) = " << ekin/eV << G4endl;
348  // G4cout << "### Warning: particle energy out of bounds! ###" << G4endl;
349  }
350  }
351 
352  if (verboseLevel > 3)
353  {
354  G4cout << "---> Kinetic energy (eV)=" << ekin/eV << G4endl;
355  G4cout << " - Cross section per Si atom (cm^2)=" << sigma*Zeff2/cm2 << G4endl;
356  G4cout << " - Cross section per Si atom (cm^-1)=" << sigma*density*Zeff2/(1./cm) << G4endl;
357  }
358 
359  } // if (SiMaterial)
360  return sigma*density*Zeff2;
361 
362 
363 }
364 
365 //....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo....
366 
367 void G4MicroElecInelasticModel::SampleSecondaries(std::vector<G4DynamicParticle*>* fvect,
368  const G4MaterialCutsCouple* /*couple*/,
369  const G4DynamicParticle* particle,
370  G4double,
371  G4double)
372 {
373 
374  if (verboseLevel > 3)
375  G4cout << "Calling SampleSecondaries() of G4MicroElecInelasticModel" << G4endl;
376 
377  G4double lowLim = 0;
378  G4double highLim = 0;
379 
380  G4double ekin = particle->GetKineticEnergy();
381  G4double k = ekin ;
382 
383  G4ParticleDefinition* PartDef = particle->GetDefinition();
384  const G4String& particleName = PartDef->GetParticleName();
385  G4String nameLocal2 = particleName ;
386  G4double particleMass = particle->GetDefinition()->GetPDGMass();
387 
388  if (particleMass > proton_mass_c2)
389  {
390  k *= proton_mass_c2/particleMass ;
391  PartDef = G4Proton::ProtonDefinition();
392  nameLocal2 = "proton" ;
393  }
394 
395  std::map< G4String,G4double,std::less<G4String> >::iterator pos1;
396  pos1 = lowEnergyLimit.find(nameLocal2);
397 
398  if (pos1 != lowEnergyLimit.end())
399  {
400  lowLim = pos1->second;
401  }
402 
403  std::map< G4String,G4double,std::less<G4String> >::iterator pos2;
404  pos2 = highEnergyLimit.find(nameLocal2);
405 
406  if (pos2 != highEnergyLimit.end())
407  {
408  highLim = pos2->second;
409  }
410 
411  if (k >= lowLim && k < highLim)
412  {
413  G4ParticleMomentum primaryDirection = particle->GetMomentumDirection();
414  G4double totalEnergy = ekin + particleMass;
415  G4double pSquare = ekin * (totalEnergy + particleMass);
416  G4double totalMomentum = std::sqrt(pSquare);
417 
418  G4int Shell = RandomSelect(k,nameLocal2);
419  G4double bindingEnergy = SiStructure.Energy(Shell);
420  if (verboseLevel > 3)
421  {
422  G4cout << "---> Kinetic energy (eV)=" << k/eV << G4endl ;
423  G4cout << "Shell: " << Shell << ", energy: " << bindingEnergy/eV << G4endl;
424  }
425 
426  // sample deexcitation
427 
428  G4int secNumberInit = 0; // need to know at a certain point the energy of secondaries
429  G4int secNumberFinal = 0; // So I'll make the difference and then sum the energies
430 
431  if(fAtomDeexcitation && Shell > 2) {
432  G4int Z = 14;
434 
435  if (Shell == 4)
436  {
437  as = G4AtomicShellEnumerator(1);
438  }
439  else if (Shell == 3)
440  {
441  as = G4AtomicShellEnumerator(3);
442  }
443 
444  const G4AtomicShell* shell = fAtomDeexcitation->GetAtomicShell(Z, as);
445  secNumberInit = fvect->size();
446  fAtomDeexcitation->GenerateParticles(fvect, shell, Z, 0, 0);
447  secNumberFinal = fvect->size();
448  }
449 
450  G4double secondaryKinetic = RandomizeEjectedElectronEnergy(PartDef,k,Shell);
451 
452  if (verboseLevel > 3)
453  {
454  G4cout << "Ionisation process" << G4endl;
455  G4cout << "Shell: " << Shell << " Kin. energy (eV)=" << k/eV
456  << " Sec. energy (eV)=" << secondaryKinetic/eV << G4endl;
457  }
458 
459  G4double cosTheta = 0.;
460  G4double phi = 0.;
461  RandomizeEjectedElectronDirection(PartDef, k, secondaryKinetic, cosTheta, phi);
462 
463  G4double sinTheta = std::sqrt(1.-cosTheta*cosTheta);
464  G4double dirX = sinTheta*std::cos(phi);
465  G4double dirY = sinTheta*std::sin(phi);
466  G4double dirZ = cosTheta;
467  G4ThreeVector deltaDirection(dirX,dirY,dirZ);
468  deltaDirection.rotateUz(primaryDirection);
469 
470  //if (particle->GetDefinition() == G4Electron::ElectronDefinition())
471  //{
472  G4double deltaTotalMomentum = std::sqrt(secondaryKinetic*(secondaryKinetic + 2.*electron_mass_c2 ));
473 
474  G4double finalPx = totalMomentum*primaryDirection.x() - deltaTotalMomentum*deltaDirection.x();
475  G4double finalPy = totalMomentum*primaryDirection.y() - deltaTotalMomentum*deltaDirection.y();
476  G4double finalPz = totalMomentum*primaryDirection.z() - deltaTotalMomentum*deltaDirection.z();
477  G4double finalMomentum = std::sqrt(finalPx*finalPx + finalPy*finalPy + finalPz*finalPz);
478  finalPx /= finalMomentum;
479  finalPy /= finalMomentum;
480  finalPz /= finalMomentum;
481 
482  G4ThreeVector direction;
483  direction.set(finalPx,finalPy,finalPz);
484 
486  //}
487  //else fParticleChangeForGamma->ProposeMomentumDirection(primaryDirection) ;
488 
489  // note that secondaryKinetic is the energy of the delta ray, not of all secondaries.
490  G4double deexSecEnergy = 0;
491  for (G4int j=secNumberInit; j < secNumberFinal; j++) {
492  deexSecEnergy = deexSecEnergy + (*fvect)[j]->GetKineticEnergy();}
493 
494  fParticleChangeForGamma->SetProposedKineticEnergy(ekin-bindingEnergy-secondaryKinetic);
495  fParticleChangeForGamma->ProposeLocalEnergyDeposit(bindingEnergy-deexSecEnergy);
496 
497  G4DynamicParticle* dp = new G4DynamicParticle (G4Electron::Electron(),deltaDirection,secondaryKinetic) ;
498  fvect->push_back(dp);
499 
500  }
501 
502 }
503 
504 //....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo......
505 
506 G4double G4MicroElecInelasticModel::RandomizeEjectedElectronEnergy(G4ParticleDefinition* particleDefinition,
507 G4double k, G4int shell)
508 {
509  if (particleDefinition == G4Electron::ElectronDefinition())
510  {
511  G4double maximumEnergyTransfer=0.;
512  if ((k+SiStructure.Energy(shell))/2. > k) maximumEnergyTransfer=k;
513  else maximumEnergyTransfer = (k+SiStructure.Energy(shell))/2.;
514 
515  G4double crossSectionMaximum = 0.;
516 
517  G4double minEnergy = SiStructure.Energy(shell);
518  G4double maxEnergy = maximumEnergyTransfer;
519  G4int nEnergySteps = 100;
520 
521  G4double value(minEnergy);
522  G4double stpEnergy(std::pow(maxEnergy/value, 1./static_cast<G4double>(nEnergySteps-1)));
523  G4int step(nEnergySteps);
524  while (step>0)
525  {
526  step--;
527  G4double differentialCrossSection = DifferentialCrossSection(particleDefinition, k/eV, value/eV, shell);
528  if(differentialCrossSection >= crossSectionMaximum) crossSectionMaximum = differentialCrossSection;
529  value*=stpEnergy;
530  }
531 
532 
533  G4double secondaryElectronKineticEnergy=0.;
534  do
535  {
536  secondaryElectronKineticEnergy = G4UniformRand() * (maximumEnergyTransfer-SiStructure.Energy(shell));
537  } while(G4UniformRand()*crossSectionMaximum >
538  DifferentialCrossSection(particleDefinition, k/eV,(secondaryElectronKineticEnergy+SiStructure.Energy(shell))/eV,shell));
539 
540  return secondaryElectronKineticEnergy;
541 
542  }
543 
544  if (particleDefinition == G4Proton::ProtonDefinition())
545  {
546  G4double maximumEnergyTransfer = 4.* (electron_mass_c2 / proton_mass_c2) * k;
547  G4double crossSectionMaximum = 0.;
548 
549  G4double minEnergy = SiStructure.Energy(shell);
550  G4double maxEnergy = maximumEnergyTransfer;
551  G4int nEnergySteps = 100;
552 
553  G4double value(minEnergy);
554  G4double stpEnergy(std::pow(maxEnergy/value, 1./static_cast<G4double>(nEnergySteps-1)));
555  G4int step(nEnergySteps);
556  while (step>0)
557  {
558  step--;
559  G4double differentialCrossSection = DifferentialCrossSection(particleDefinition, k/eV, value/eV, shell);
560  if(differentialCrossSection >= crossSectionMaximum) crossSectionMaximum = differentialCrossSection;
561  value*=stpEnergy;
562  }
563  G4double secondaryElectronKineticEnergy = 0.;
564  do
565  {
566  secondaryElectronKineticEnergy = G4UniformRand() * (maximumEnergyTransfer-SiStructure.Energy(shell));
567 
568  } while(G4UniformRand()*crossSectionMaximum >=
569  DifferentialCrossSection(particleDefinition, k/eV,(secondaryElectronKineticEnergy+SiStructure.Energy(shell))/eV,shell));
570  return secondaryElectronKineticEnergy;
571  }
572 
573  return 0;
574 }
575 
576 //....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo......
577 
578 void G4MicroElecInelasticModel::RandomizeEjectedElectronDirection(G4ParticleDefinition* particleDefinition,
579  G4double k,
580  G4double secKinetic,
581  G4double & cosTheta,
582  G4double & phi )
583 {
584  if (particleDefinition == G4Electron::ElectronDefinition())
585  {
586  phi = twopi * G4UniformRand();
587  G4double sin2O = (1.-secKinetic/k) / (1.+secKinetic/(2.*electron_mass_c2));
588  cosTheta = std::sqrt(1.-sin2O);
589  }
590 
591  if (particleDefinition == G4Proton::ProtonDefinition())
592  {
593  G4double maxSecKinetic = 4.* (electron_mass_c2 / proton_mass_c2) * k;
594  phi = twopi * G4UniformRand();
595  cosTheta = std::sqrt(secKinetic / maxSecKinetic);
596  }
597 
598  else
599  {
600  G4double maxSecKinetic = 4.* (electron_mass_c2 / particleDefinition->GetPDGMass()) * k;
601  phi = twopi * G4UniformRand();
602  cosTheta = std::sqrt(secKinetic / maxSecKinetic);
603  }
604 }
605 
606 //....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo......
607 
609  G4double k,
610  G4double energyTransfer,
611  G4int LevelIndex)
612 {
613  G4double sigma = 0.;
614 
615  if (energyTransfer >= SiStructure.Energy(LevelIndex))
616  {
617  G4double valueT1 = 0;
618  G4double valueT2 = 0;
619  G4double valueE21 = 0;
620  G4double valueE22 = 0;
621  G4double valueE12 = 0;
622  G4double valueE11 = 0;
623 
624  G4double xs11 = 0;
625  G4double xs12 = 0;
626  G4double xs21 = 0;
627  G4double xs22 = 0;
628 
629  if (particleDefinition == G4Electron::ElectronDefinition())
630  {
631  // k should be in eV and energy transfer eV also
632 
633  std::vector<double>::iterator t2 = std::upper_bound(eTdummyVec.begin(),eTdummyVec.end(), k);
634  std::vector<double>::iterator t1 = t2-1;
635  // SI : the following condition avoids situations where energyTransfer >last vector element
636  if (energyTransfer <= eVecm[(*t1)].back() && energyTransfer <= eVecm[(*t2)].back() )
637  {
638  std::vector<double>::iterator e12 = std::upper_bound(eVecm[(*t1)].begin(),eVecm[(*t1)].end(), energyTransfer);
639  std::vector<double>::iterator e11 = e12-1;
640 
641  std::vector<double>::iterator e22 = std::upper_bound(eVecm[(*t2)].begin(),eVecm[(*t2)].end(), energyTransfer);
642  std::vector<double>::iterator e21 = e22-1;
643 
644  valueT1 =*t1;
645  valueT2 =*t2;
646  valueE21 =*e21;
647  valueE22 =*e22;
648  valueE12 =*e12;
649  valueE11 =*e11;
650 
651  xs11 = eDiffCrossSectionData[LevelIndex][valueT1][valueE11];
652  xs12 = eDiffCrossSectionData[LevelIndex][valueT1][valueE12];
653  xs21 = eDiffCrossSectionData[LevelIndex][valueT2][valueE21];
654  xs22 = eDiffCrossSectionData[LevelIndex][valueT2][valueE22];
655  }
656 
657  }
658 
659  if (particleDefinition == G4Proton::ProtonDefinition())
660  {
661  // k should be in eV and energy transfer eV also
662  std::vector<double>::iterator t2 = std::upper_bound(pTdummyVec.begin(),pTdummyVec.end(), k);
663  std::vector<double>::iterator t1 = t2-1;
664  if (energyTransfer <= pVecm[(*t1)].back() && energyTransfer <= pVecm[(*t2)].back() )
665  {
666  std::vector<double>::iterator e12 = std::upper_bound(pVecm[(*t1)].begin(),pVecm[(*t1)].end(), energyTransfer);
667  std::vector<double>::iterator e11 = e12-1;
668 
669  std::vector<double>::iterator e22 = std::upper_bound(pVecm[(*t2)].begin(),pVecm[(*t2)].end(), energyTransfer);
670  std::vector<double>::iterator e21 = e22-1;
671 
672  valueT1 =*t1;
673  valueT2 =*t2;
674  valueE21 =*e21;
675  valueE22 =*e22;
676  valueE12 =*e12;
677  valueE11 =*e11;
678 
679  xs11 = pDiffCrossSectionData[LevelIndex][valueT1][valueE11];
680  xs12 = pDiffCrossSectionData[LevelIndex][valueT1][valueE12];
681  xs21 = pDiffCrossSectionData[LevelIndex][valueT2][valueE21];
682  xs22 = pDiffCrossSectionData[LevelIndex][valueT2][valueE22];
683  }
684  }
685 
686  G4double xsProduct = xs11 * xs12 * xs21 * xs22;
687  if (xsProduct != 0.)
688  {
689  sigma = QuadInterpolator( valueE11, valueE12,
690  valueE21, valueE22,
691  xs11, xs12,
692  xs21, xs22,
693  valueT1, valueT2,
694  k, energyTransfer);
695  }
696 
697  }
698 
699  return sigma;
700 }
701 
702 //....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo......
703 
704 G4double G4MicroElecInelasticModel::LogLogInterpolate(G4double e1,
705  G4double e2,
706  G4double e,
707  G4double xs1,
708  G4double xs2)
709 {
710  G4double a = (std::log10(xs2)-std::log10(xs1)) / (std::log10(e2)-std::log10(e1));
711  G4double b = std::log10(xs2) - a*std::log10(e2);
712  G4double sigma = a*std::log10(e) + b;
713  G4double value = (std::pow(10.,sigma));
714  return value;
715 }
716 
717 //....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo......
718 
719 G4double G4MicroElecInelasticModel::QuadInterpolator(G4double e11, G4double e12,
720  G4double e21, G4double e22,
721  G4double xs11, G4double xs12,
722  G4double xs21, G4double xs22,
723  G4double t1, G4double t2,
724  G4double t, G4double e)
725 {
726  G4double interpolatedvalue1 = LogLogInterpolate(e11, e12, e, xs11, xs12);
727  G4double interpolatedvalue2 = LogLogInterpolate(e21, e22, e, xs21, xs22);
728  G4double value = LogLogInterpolate(t1, t2, t, interpolatedvalue1, interpolatedvalue2);
729  return value;
730 }
731 
732 //....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo......
733 
734 G4int G4MicroElecInelasticModel::RandomSelect(G4double k, const G4String& particle )
735 {
736  G4int level = 0;
737 
738  std::map< G4String,G4MicroElecCrossSectionDataSet*,std::less<G4String> >::iterator pos;
739  pos = tableData.find(particle);
740 
741  if (pos != tableData.end())
742  {
743  G4MicroElecCrossSectionDataSet* table = pos->second;
744 
745  if (table != 0)
746  {
747  G4double* valuesBuffer = new G4double[table->NumberOfComponents()];
748  const size_t n(table->NumberOfComponents());
749  size_t i(n);
750  G4double value = 0.;
751 
752  while (i>0)
753  {
754  i--;
755  valuesBuffer[i] = table->GetComponent(i)->FindValue(k);
756  value += valuesBuffer[i];
757  }
758 
759  value *= G4UniformRand();
760 
761  i = n;
762 
763  while (i > 0)
764  {
765  i--;
766 
767  if (valuesBuffer[i] > value)
768  {
769  delete[] valuesBuffer;
770  return i;
771  }
772  value -= valuesBuffer[i];
773  }
774 
775  if (valuesBuffer) delete[] valuesBuffer;
776 
777  }
778  }
779  else
780  {
781  G4Exception("G4MicroElecInelasticModel::RandomSelect","em0002",FatalException,"Model not applicable to particle type.");
782  }
783 
784  return level;
785 }
786 
787 //....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo......
788 
789 
void set(double x, double y, double z)
static G4Electron * ElectronDefinition()
Definition: G4Electron.cc:89
G4double LowEnergyLimit() const
Definition: G4VEmModel.hh:599
virtual G4double CrossSectionPerVolume(const G4Material *material, const G4ParticleDefinition *p, G4double ekin, G4double emin, G4double emax)
G4Material * FindOrBuildMaterial(const G4String &name, G4bool isotopes=true, G4bool warning=false)
virtual const G4VEMDataSet * GetComponent(G4int componentId) const
virtual G4double FindValue(G4double x, G4int componentId=0) const =0
static G4LossTableManager * Instance()
G4double GetKineticEnergy() const
G4double HighEnergyLimit() const
Definition: G4VEmModel.hh:592
double x() const
G4MicroElecInelasticModel(const G4ParticleDefinition *p=0, const G4String &nam="MicroElecInelasticModel")
virtual void Initialise(const G4ParticleDefinition *, const G4DataVector &)
static G4Proton * ProtonDefinition()
Definition: G4Proton.cc:88
G4ParticleDefinition * GetDefinition() const
double DifferentialCrossSection(G4ParticleDefinition *aParticleDefinition, G4double k, G4double energyTransfer, G4int shell)
int G4int
Definition: G4Types.hh:78
static G4NistManager * Instance()
void ProposeMomentumDirection(G4double Px, G4double Py, G4double Pz)
const G4String & GetParticleName() const
double z() const
void ProposeLocalEnergyDeposit(G4double anEnergyPart)
G4double Energy(G4int level)
void SetHighEnergyLimit(G4double)
Definition: G4VEmModel.hh:683
string material
Definition: eplot.py:19
virtual const G4AtomicShell * GetAtomicShell(G4int Z, G4AtomicShellEnumerator shell)=0
G4double density
Definition: TRTMaterials.hh:39
#define G4UniformRand()
Definition: Randomize.hh:87
G4GLOB_DLL std::ostream G4cout
virtual size_t NumberOfComponents(void) const
const G4ThreeVector & GetMomentumDirection() const
Hep3Vector & rotateUz(const Hep3Vector &)
Definition: ThreeVector.cc:72
float proton_mass_c2
Definition: hepunit.py:275
float electron_mass_c2
Definition: hepunit.py:274
const G4int n
G4ParticleChangeForGamma * fParticleChangeForGamma
virtual G4double FindValue(G4double e, G4int componentId=0) const
virtual G4bool LoadData(const G4String &argFileName)
void G4Exception(const char *originOfException, const char *exceptionCode, G4ExceptionSeverity severity, const char *comments)
Definition: G4Exception.cc:41
G4double EffectiveCharge(const G4ParticleDefinition *p, const G4Material *material, G4double kineticEnergy)
G4double GetTotNbOfAtomsPerVolume() const
Definition: G4Material.hh:207
G4double GetPDGMass() const
Hep3Vector unit() const
tuple t1
Definition: plottest35.py:33
double y() const
virtual void SampleSecondaries(std::vector< G4DynamicParticle * > *, const G4MaterialCutsCouple *, const G4DynamicParticle *, G4double tmin, G4double maxEnergy)
const G4Material * GetBaseMaterial() const
Definition: G4Material.hh:231
const XML_Char int const XML_Char * value
static G4Electron * Electron()
Definition: G4Electron.cc:94
void SetProposedKineticEnergy(G4double proposedKinEnergy)
#define G4endl
Definition: G4ios.hh:61
G4VAtomDeexcitation * AtomDeexcitation()
double G4double
Definition: G4Types.hh:76
void SetLowEnergyLimit(G4double)
Definition: G4VEmModel.hh:690
void GenerateParticles(std::vector< G4DynamicParticle * > *secVect, const G4AtomicShell *, G4int Z, G4int coupleIndex)
void SetDeexcitationFlag(G4bool val)
Definition: G4VEmModel.hh:739
G4double GetPDGCharge() const
G4double bindingEnergy(G4int A, G4int Z)
G4AtomicShellEnumerator
G4ParticleChangeForGamma * GetParticleChangeForGamma()
Definition: G4VEmModel.cc:121