70 emaxT(fManagerHP->GetMaxEnergyDoppler()),
71 elimit(1.0e-11*
CLHEP::eV),
72 logElimit(
G4Log(elimit)),
76 fDataDirectory(nameDir)
80 G4cout <<
"G4CrossSectionHP::G4CrossSectionHP: Initialise for "
81 << fDataName <<
" " << minZ <<
" < Z < " << maxZ
82 <<
" EmaxT(MeV)=" << emaxT <<
G4endl;
83 G4cout <<
"Data directory: " << fDataDirectory <<
G4endl;
86 auto data = ptr->GetElementDataByName(fDataName);
87 if (
nullptr == data) {
89 data->SetName(fDataName);
114 if (mat != fCurrentMat) { PrepareCache(mat); }
133 if (mat != fCurrentMat) { PrepareCache(mat); }
146 if (ekin > emax || Z > maxZ || Z < minZ || ekin < elimit) {
return xs; }
148 auto pv0 = fData->GetElementData(Z - minZ);
149 if (
nullptr == pv0) {
151 pv0 = fData->GetElementData(Z - minZ);
155 if (
nullptr == pv0) {
return xs; }
157 const auto pv = fData->GetComponentDataByID(Z - minZ,
A);
158 if (
nullptr == pv) {
return xs; }
162 if (ekin >= emaxT*T/CLHEP::STP_Temperature || fManagerHP->GetNeglectDoppler()) {
163 xs = pv->LogFreeVectorValue(ekin, logek);
169 G4double mass = fParticle->GetPDGMass();
171 G4double sig = std::sqrt(2.0*e0/(3.0*massTarget));
174 G4LorentzVector lv(0., 0., std::sqrt(ekin*(ekin + 2*mass)), mass + ekin);
178 const G4int nmin = 3;
184 G4double vx = G4RandGauss::shoot(0., sig);
185 G4double vy = G4RandGauss::shoot(0., sig);
186 G4double vz = G4RandGauss::shoot(0., sig);
187 fLV.set(massTarget*vx, massTarget*vy, massTarget*vz, massTarget*(1.0 + 0.5*(vx*vx + vy*vy + vz*vz)));
188 fBoost = fLV.boostVector();
189 fLV = lv.boost(-fBoost);
190 if (fLV.pz() <= 0.0) {
continue; }
196 if (ii >= nmin && ii*xs2 <= lim*xs*xs) {
break; }
198 if (ii > 0) { xs /= (
G4double)(ii); }
202 G4cout <<
"G4CrossSectionHP::IsoXS " << fDataName
203 <<
" Z= " << Z <<
" A= " <<
A
204 <<
" Ekin(MeV)= " << ekin/MeV <<
" xs(b)= " << xs/barn
205 <<
" size=" << fZA.size() <<
G4endl;
210 for (std::size_t i=0; i<fZA.size(); ++i) {
211 if (Z == fZA[i].first &&
A == fZA[i].second) {
226 if(1 == nIso) {
return iso; }
230 if (Z >= minZ && Z <= maxZ && nullptr == fData->GetElementData(Z - minZ)) {
240 if (Z < minZ || Z > maxZ || !CheckCache(Z) ||
241 0 == fData->GetNumberOfComponents(Z - minZ)) {
242 for (j = 0; j<nIso; ++j) {
243 sum += abundVector[j];
251 std::size_t nn = fTemp.size();
252 if (nn < nIso) { fTemp.resize(nIso, 0.); }
255 for (j=0; j<nIso; ++j) {
256 sum += abundVector[j]*
261 for (j = 0; j<nIso; ++j) {
262 if (fTemp[j] >= sum) {
273 G4cout <<
"G4CrossSectionHP::BuildPhysicsTable for "
281 for (
auto const & elm : *table ) {
282 G4int Z = elm->GetZasInt();
283 if (Z >= minZ && Z <= maxZ && nullptr == fData->GetElementData(Z - minZ)) {
289 std::size_t nmax = 0;
290 std::size_t imax = 0;
293 for (
auto const & elm : *mat->GetElementVector() ) {
294 std::size_t niso = elm->GetNumberOfIsotopes();
296 if(niso > imax) { imax = niso; }
298 if (n > nmax) { nmax = n; }
300 fTemp.resize(imax, 0.0);
303 fIsoXS.resize(nmax, 0.0);
308 if (fManagerHP->GetVerboseLevel() == 0 || fPrinted)
319 G4cout <<
"HP Cross Section " << fDataName <<
" for "
320 << fParticle->GetParticleName() <<
G4endl;
321 G4cout <<
"(Pointwise cross-section at 0 Kelvin.)" <<
G4endl;
328 for (
auto const & elm : *table ) {
329 G4int Z = elm->GetZasInt();
330 if (Z >= minZ && Z <= maxZ && nullptr != fData->GetElementData(Z - minZ)) {
331 G4cout <<
"---------------------------------------------------" <<
G4endl;
333 std::size_t n = fData->GetNumberOfComponents(Z);
334 for (
size_t i=0; i<n; ++i) {
335 G4cout <<
"## Z=" << Z <<
" A=" << fData->GetComponentID(Z - minZ, i);
336 G4cout << *(fData->GetComponentDataByIndex(Z - minZ, i)) <<
G4endl;
342void G4CrossSectionHP::PrepareCache(
const G4Material* mat)
347 G4int Z = elm->GetZasInt();
348 for (
auto const & iso : *(elm->GetIsotopeVector()) ) {
350 fZA.emplace_back(Z,
A);
353 fIsoXS.resize(fZA.size(), 0.0);
356void G4CrossSectionHP::Initialise(
const G4int Z)
358 if (fManagerHP->GetVerboseLevel() > 1) {
359 G4cout <<
" G4CrossSectionHP::Initialise: Z=" << Z
360 <<
" for " << fDataName
361 <<
" minZ=" << minZ <<
" maxZ=" << maxZ <<
G4endl;
363 if (Z < minZ || Z > maxZ ||
nullptr != fData->GetElementData(Z - minZ)) {
367 if (
nullptr != fData->GetElementData(Z - minZ)) {
373 fData->InitialiseForElement(Z - minZ,
new G4PhysicsVector());
375 G4String tnam =
"temp";
377 for (
G4int A=amin[Z];
A<=amax[Z]; ++
A) {
378 std::ostringstream ost;
379 ost << fDataDirectory;
381 if (6 == Z && 12 ==
A && fParticle == fNeutron) {
382 ost << Z <<
"_nat_" << elementName[Z];
383 }
else if (18 == Z && 40 !=
A) {
385 }
else if (27 == Z && 62 ==
A) {
386 ost << Z <<
"_62m1_" << elementName[Z];
387 }
else if (47 == Z && 106 ==
A) {
388 ost << Z <<
"_106m1_" << elementName[Z];
389 }
else if (48 == Z && 115 ==
A) {
390 ost << Z <<
"_115m1_" << elementName[Z];
391 }
else if (52 == Z && 127 ==
A) {
392 ost << Z <<
"_127m1_" << elementName[Z];
393 }
else if (52 == Z && 129 ==
A) {
394 ost << Z <<
"_129m1_" << elementName[Z];
395 }
else if (52 == Z && 131 ==
A) {
396 ost << Z <<
"_131m1_" << elementName[Z];
397 }
else if (61 == Z && 145 ==
A) {
398 ost << Z <<
"_147_" << elementName[Z];
399 }
else if (67 == Z && 166 ==
A) {
400 ost << Z <<
"_166m1_" << elementName[Z];
401 }
else if (73 == Z && 180 ==
A) {
402 ost << Z <<
"_180m1_" << elementName[Z];
403 }
else if ((Z == 85 &&
A == 210) || (Z == 86 &&
A == 222) || (Z == 87 &&
A == 223)) {
404 ost <<
"84_209_" << elementName[84];
407 ost << Z <<
"_" <<
A <<
"_" << elementName[Z];
409 std::istringstream theXSData(tnam, std::ios::in);
410 fManagerHP->GetDataStream(ost.str().c_str(), theXSData);
413 theXSData >> i1 >> i2 >>
n;
414 if (fManagerHP->GetVerboseLevel() > 1) {
415 G4cout <<
"## G4CrossSectionHP::Initialise for Z=" << Z
416 <<
" A=" <<
A <<
" Npoints=" <<
n <<
G4endl;
419 G4PhysicsFreeVector* v =
new G4PhysicsFreeVector(n);
420 for (
G4int i=0; i<
n; ++i) {
429 G4int nmax = amax[Z] -
A + 1;
430 fData->InitialiseForComponent(Z - minZ, nmax);
433 fData->AddComponent(Z - minZ,
A, v);
436 if (noComp) { fData->InitialiseForComponent(Z - minZ, 0); }
G4TemplateAutoLock< G4Mutex > G4AutoLock
std::vector< G4Element * > G4ElementTable
G4double G4Log(G4double x)
CLHEP::HepLorentzVector G4LorentzVector
#define G4MUTEX_INITIALIZER
G4GLOB_DLL std::ostream G4cout
G4double ComputeIsoCrossSection(G4double kinEnergy, G4double loge, const G4ParticleDefinition *, G4int Z, G4int A, const G4Isotope *iso, const G4Element *elm, const G4Material *mat) override
G4double GetIsoCrossSection(const G4DynamicParticle *, G4int Z, G4int A, const G4Isotope *, const G4Element *, const G4Material *) override
void BuildPhysicsTable(const G4ParticleDefinition &) override
G4CrossSectionHP(const G4ParticleDefinition *, const G4String &nameData, const G4String &nameDir, G4double emaxHP, G4int zmin, G4int zmax)
void DumpPhysicsTable(const G4ParticleDefinition &) override
G4bool IsIsoApplicable(const G4DynamicParticle *, G4int Z, G4int A, const G4Element *, const G4Material *) override
const G4Isotope * SelectIsotope(const G4Element *, G4double kinEnergy, G4double logE) override
G4double GetLogKineticEnergy() const
G4double GetKineticEnergy() const
static G4ElementDataRegistry * Instance()
G4double * GetRelativeAbundanceVector() const
std::size_t GetNumberOfIsotopes() const
const G4Isotope * GetIsotope(G4int iso) const
static const G4ElementTable * GetElementTable()
const G4ElementVector * GetElementVector() const
G4double GetTemperature() const
static G4MaterialTable * GetMaterialTable()
static G4double GetNuclearMass(const G4double A, const G4double Z)
const G4String & GetParticleName() const
void PutValues(const std::size_t index, const G4double energy, const G4double value)
void EnableLogBinSearch(const G4int n=1)
G4VCrossSectionDataSet(const G4String &nam="")