94 const G4double timeThresholdForRadioactiveDecays)
99 G4cout <<
"G4RadioactiveDecay constructor: processName = " << processName
119 theRadioactivityTables.push_back(rTable);
123 halflifethreshold = 1000.*nanosecond;
128 outFile <<
"The G4RadioactiveDecay process performs radioactive decay of\n"
129 <<
"nuclides (G4GenericIon) in biased mode which includes nucleus\n"
130 <<
"duplication, branching ratio biasing, source time convolution\n"
131 <<
"and detector time convolution. It is designed for use in\n"
132 <<
"activation physics.\n"
133 <<
"The required half-lives and decay schemes are retrieved from\n"
134 <<
"the RadioactiveDecay database which was derived from ENSDF.\n";
140 delete theRadioactivationMessenger;
149 for (std::size_t i = 0; i < theParentChainTable.size(); ++i) {
150 if (theParentChainTable[i].GetIonName() == aParticleName)
return true;
161 for (std::size_t i = 0; i < theParentChainTable.size(); ++i) {
162 if (theParentChainTable[i].GetIonName() == aParticleName) {
163 theDecayRateVector = theParentChainTable[i].GetItsRates();
168 G4cout <<
"The DecayRate Table for " << aParticleName <<
" is selected."
185 if ( t > SBin[NSourceBin]) {
191 while (t > SBin[nbin]) {
194 G4Exception(
"G4RadioactiveDecay::ConvolveSourceTimeProfile()",
195 "HAD_RDM_100",
JustWarning,
"While loop count exceeded");
207 for (
G4int i = 0; i < nbin; ++i) {
208 earg = (SBin[i+1] - SBin[i])/tau;
210 convolvedTime += SProfile[i] * std::exp((SBin[i] - t)/tau) *
213 convolvedTime += SProfile[i] *
214 (std::exp(-(t-SBin[i+1])/tau)-std::exp(-(t-SBin[i])/tau));
218 convolvedTime -= SProfile[nbin] * std::expm1((SBin[nbin] - t)/tau);
221 if (convolvedTime < 0.) {
222 G4cout <<
" Convolved time =: " << convolvedTime <<
" reset to zero! " <<
G4endl;
229 G4cout <<
" Convolved time: " << convolvedTime <<
G4endl;
231 return convolvedTime;
250 while (DProfile[i] < rand) {
256 G4Exception(
"G4RadioactiveDecay::GetDecayTime()",
"HAD_RDM_100",
263 decaytime = DBin[i] + rand*(DBin[i+1]-DBin[i]);
266 G4cout <<
" Decay time: " <<decaytime/s <<
"[s]" <<
G4endl;
277 while (aDecayTime > DBin[i] ) {
281 G4Exception(
"G4RadioactiveDecay::GetDecayTimeBin()",
"HAD_RDM_100",
310 G4int theG, std::vector<G4double>& theCoefficients,
311 std::vector<G4double>& theTaos)
315 ratesToDaughter.SetZ(theZ);
316 ratesToDaughter.SetA(theA);
317 ratesToDaughter.SetE(theE);
318 ratesToDaughter.SetGeneration(theG);
319 ratesToDaughter.SetDecayRateC(theCoefficients);
320 ratesToDaughter.SetTaos(theTaos);
334 theDecayRateVector.clear();
336 G4int nGeneration = 0;
338 std::vector<G4double> taos;
341 std::vector<G4double> Acoeffs;
344 Acoeffs.push_back(-1.);
346 const G4Ions* ion =
static_cast<const G4Ions*
>(&theParentNucleus);
351 if (tao < 0.) tao = 1e-100;
360 theDecayRateVector.push_back(ratesToDaughter);
385 std::vector<G4double> TP;
386 std::vector<G4double> RP;
390 G4int nearestLevelIndex = 0;
407 G4Exception(
"G4RadioactiveDecay::CalculateChainsFromParent()",
"HAD_RDM_100",
412 for (j = nS; j < nT; ++j) {
414 ZP = theDecayRateVector[j].GetZ();
415 AP = theDecayRateVector[j].GetA();
416 EP = theDecayRateVector[j].GetE();
417 RP = theDecayRateVector[j].GetDecayRateC();
418 TP = theDecayRateVector[j].GetTaos();
420 G4cout <<
"G4RadioactiveDecay::CalculateChainsFromParent: daughters of ("
421 << ZP <<
", " << AP <<
", " << EP
422 <<
") are being calculated, generation = " << nGeneration
426 aParentNucleus = theIonTable->
GetIon(ZP,AP,EP);
428 if (
nullptr == parentDecayTable) {
continue; }
440 for (
G4int k = 0; k < nMode; ++k) brs[k] = 0.0;
443 for (
G4int i = 0; i < parentDecayTable->
entries(); ++i) {
445 theNuclearDecayChannel =
static_cast<G4NuclearDecay*
>(theChannel);
450 ZD = ((
const G4Ions*)(theDaughterNucleus))->GetAtomicNumber();
459 if ((std::abs(daughterExcitation - nearestEnergy) <
levelTolerance) &&
460 (std::abs(daughterExcitation - nearestEnergy) >
DBL_EPSILON)) {
464 if (levelManager->
LifeTime(nearestLevelIndex)*
ns >= halflifethreshold){
466 summedDecayTable->
Insert(theChannel);
468 brs[theDecayMode] += theChannel->
GetBR();
471 brs[theDecayMode] += theChannel->
GetBR();
474 brs[theDecayMode] += theChannel->
GetBR();
481 for (
G4int i = 0; i < nMode; ++i) {
486 theITChannel =
new G4ITDecay(aParentNucleus, brs[
IT], 0.0, 0.0);
488 summedDecayTable->
Insert(theITChannel);
496 summedDecayTable->
Insert(theBetaMinusChannel);
504 summedDecayTable->
Insert(theBetaPlusChannel);
511 summedDecayTable->
Insert(theAlphaChannel);
518 summedDecayTable->
Insert(theProtonChannel);
525 summedDecayTable->
Insert(theNeutronChannel);
532 summedDecayTable->
Insert(theFissionChannel);
563 summedDecayTable->
Insert(theTritonChannel);
574 for (
G4int i = 0; i < summedDecayTable->
entries(); ++i){
576 theNuclearDecayChannel =
static_cast<G4NuclearDecay*
>(theChannel);
577 theBR = theChannel->
GetBR();
582 if (theNuclearDecayChannel->
GetDecayMode() ==
IT && nGeneration == 1) {
584 A = ((
const G4Ions*)(theDaughterNucleus))->GetAtomicMass();
585 Z = ((
const G4Ions*)(theDaughterNucleus))->GetAtomicNumber();
586 theDaughterNucleus=theIonTable->
GetIon(Z,
A,0.);
589 aParentNucleus != theDaughterNucleus) {
592 if (
nullptr != parentDecayTable && parentDecayTable->
entries() > 0) {
593 A = ((
const G4Ions*)(theDaughterNucleus))->GetAtomicMass();
594 Z = ((
const G4Ions*)(theDaughterNucleus))->GetAtomicNumber();
595 E = ((
const G4Ions*)(theDaughterNucleus))->GetExcitationEnergy();
598 if (TaoPlus <= 0.) TaoPlus = 1e-100;
608 taos.push_back(TaoPlus);
615 ta2 = (
long double)TaoPlus;
616 for (k = 0; k < RP.size(); ++k){
617 ta1 = (
long double)TP[k];
621 theRate = ta1/(ta1-ta2);
623 theRate = theRate * theBR * RP[k];
624 Acoeffs.push_back(theRate);
631 long double aRate, aRate1;
633 for (k = 0; k < RP.size(); ++k){
634 ta1 = (
long double)TP[k];
638 aRate = ta2/(ta1-ta2);
640 aRate = aRate * (
long double)(theBR * RP[k]);
644 Acoeffs.push_back(theRate);
646 theDecayRateVector.push_back(ratesToDaughter);
652 delete summedDecayTable;
657 if (nS == nT) stable =
true;
669 chainsFromParent.SetItsRates(theDecayRateVector);
672 theParentChainTable.push_back(chainsFromParent);
684 std::ifstream infile ( filename, std::ios::in );
687 ed <<
" Could not open file " << filename <<
G4endl;
688 G4Exception(
"G4RadioactiveDecay::SetSourceTimeProfile()",
"HAD_RDM_001",
696 while (infile >> bin >> flux) {
699 G4Exception(
"G4RadioactiveDecay::SetSourceTimeProfile()",
"HAD_RDM_100",
705 if (NSourceBin > 99) {
706 G4Exception(
"G4RadioactiveDecay::SetSourceTimeProfile()",
"HAD_RDM_002",
710 SBin[NSourceBin] = bin * s;
711 SProfile[NSourceBin] = flux;
720 G4cout <<
" Source Timeprofile Nbin = " << NSourceBin <<
G4endl;
733 std::ifstream infile(filename, std::ios::in);
734 if (!infile)
G4Exception(
"G4RadioactiveDecay::SetDecayBias()",
"HAD_RDM_001",
741 theRadioactivityTables.clear();
746 while (infile >> bin >> flux ) {
750 G4Exception(
"G4RadioactiveDecay::SetDecayBias()",
"HAD_RDM_100",
755 if (NDecayBin > 99) {
756 G4Exception(
"G4RadioactiveDecay::SetDecayBias()",
"HAD_RDM_002",
759 DBin[NDecayBin] = bin * s;
760 DProfile[NDecayBin] = flux;
762 decayWindows[NDecayBin] = dWindows;
765 theRadioactivityTables.push_back(rTable);
769 for ( i = 1; i<= NDecayBin; ++i) DProfile[i] += DProfile[i-1];
770 for ( i = 0; i<= NDecayBin; ++i) DProfile[i] /= DProfile[NDecayBin];
779 G4cout <<
" Decay Bias Profile Nbin = " << NDecayBin <<
G4endl;
804 G4cout <<
"G4RadioactiveDecay::DecayIt : "
806 <<
" is not selected for the RDM"<<
G4endl;
828 G4cout <<
"G4RadioactiveDecay::DecayIt : "
830 <<
" is not an ion or is outside (Z,A) limits set for the decay. "
831 <<
" Set particle change accordingly. "
846 if (theDecayTable ==
nullptr || theDecayTable->
entries() == 0) {
851 G4cout <<
"G4RadioactiveDecay::DecayIt : "
852 <<
"decay table not defined for "
854 <<
". Set particle change accordingly. "
893 std::vector<G4double> PT;
894 std::vector<G4double> PR;
896 long double decayRate;
899 G4int numberOfSecondaries;
900 G4int totalNumberOfSecondaries = 0;
904 std::vector<G4DynamicParticle*> secondaryparticles;
905 std::vector<G4double> pw;
906 std::vector<G4double> ptime;
911 for (
G4int n = 0; n < NSplit; ++n) {
920 weight1 = 1./DProfile[nbin-1]
921 *(DBin[nbin]-DBin[nbin-1])/NSplit;
922 }
else if (nbin > 1) {
924 weight1 = 1./(DProfile[nbin]-DProfile[nbin-2])
925 *(DBin[nbin]-DBin[nbin-1])/NSplit;
933 for (i = 0; i < theDecayRateVector.size(); ++i) {
934 PZ = theDecayRateVector[i].GetZ();
935 PA = theDecayRateVector[i].GetA();
936 PE = theDecayRateVector[i].GetE();
937 PT = theDecayRateVector[i].GetTaos();
938 PR = theDecayRateVector[i].GetDecayRateC();
986 for (
G4int j = 0; j <
G4int(PT.size() ); ++j) {
988 decayRate -= PR[j] * (
long double)taotime;
1000 if (decayRate < 0.0) decayRate = 0.0;
1009 theRadioactivityTables[decayWindows[nbin-1]]
1010 ->AddIsotope(PZ,PA,PE,weight1*decayRate,theTrack.
GetWeight());
1019 parentNucleus = theIonTable->
GetIon(PZ,PA,PE);
1029 if (
nullptr != decayTable) {
1034 if (theDecayChannel ==
nullptr) {
1038 G4cout <<
" G4RadioactiveDecay::DoIt : cannot determine decay channel ";
1039 G4cout <<
" for this nucleus; decay as if no biasing active. ";
1041 if (
nullptr != decayTable) { decayTable ->
DumpInfo(); }
1044 tempprods =
DoDecay(*parentNucleus, theDecayTable);
1048 tempprods = theDecayChannel->
DecayIt(tempmass);
1049 weight *= (theDecayChannel->
GetBR())*(decayTable->
entries());
1052 tempprods =
DoDecay(*parentNucleus, theDecayTable);
1056 numberOfSecondaries = tempprods->
entries();
1057 currentTime = finalGlobalTime + theDecayTime;
1058 for (index = 0; index < numberOfSecondaries; ++index) {
1061 pw.push_back(weight);
1062 ptime.push_back(currentTime);
1063 secondaryparticles.push_back(asecondaryparticle);
1067 ->GetExcitationEnergy() > 0. && weight > 0.) {
1070 ptime,secondaryparticles);
1080 totalNumberOfSecondaries = (
G4int)pw.size();
1082 for (index=0; index < totalNumberOfSecondaries; ++index) {
1084 ptime[index], currentPosition);
1108 std::vector<double>& weights_v,
1109 std::vector<double>& times_v,
1110 std::vector<G4DynamicParticle*>& secondaries_v)
1112 G4double elevel=((
const G4Ions*)(apartDef))->GetExcitationEnergy();
1116 while (life_time < halflifethreshold && elevel > 0.) {
1117 decayIT->SetupDecay(apartDef);
1123 for (
G4int ind = 0; ind < nb_pevapSecondaries; ind++) {
1128 elevel = ((
const G4Ions*)(secDef))->GetExcitationEnergy();
1132 weights_v.push_back(weight);
1133 times_v.push_back(currentTime);
1134 secondaries_v.push_back(a_pevap_secondary);
1137 weights_v.push_back(weight);
1138 times_v.push_back(currentTime);
1139 secondaries_v.push_back(a_pevap_secondary);
1144 delete pevap_products;
void G4Exception(const char *originOfException, const char *exceptionCode, G4ExceptionSeverity severity, const char *description)
std::ostringstream G4ExceptionDescription
@ G4RadioactiveDecayModeSize
CLHEP::Hep3Vector G4ThreeVector
G4GLOB_DLL std::ostream G4cout
G4DynamicParticle * PopProducts()
G4VDecayChannel * GetDecayChannel(G4int index) const
void Insert(G4VDecayChannel *aChannel)
G4ParticleDefinition * GetDefinition() const
G4ParticleDefinition * GetIon(G4int Z, G4int A, G4int lvl=0)
G4double GetExcitationEnergy() const
std::size_t NearestLevelIndex(const G4double energy, const std::size_t index=0) const
std::size_t NumberOfTransitions() const
G4double LifeTime(const std::size_t i) const
G4double NearestLevelEnergy(const G4double energy, const std::size_t index=0) const
const G4String & GetName() const
G4RadioactiveDecayMode GetDecayMode() const
G4double GetDaughterExcitation() const
G4ParticleDefinition * GetDaughterNucleus()
const G4LevelManager * GetLevelManager(G4int Z, G4int A)
static G4NuclearLevelData * GetInstance()
G4bool GetPDGStable() const
G4int GetAtomicNumber() const
G4double GetPDGMass() const
G4int GetAtomicMass() const
G4int GetBaryonNumber() const
G4double GetPDGLifeTime() const
const G4String & GetParticleName() const
G4IonTable * GetIonTable() const
static G4ParticleTable * GetParticleTable()
G4VParticleChange * DecayIt(const G4Track &theTrack, const G4Step &theStep) override
~G4RadioactiveDecay() override
void SetDecayBias(const G4String &filename)
void ProcessDescription(std::ostream &outFile) const override
G4int GetDecayTimeBin(const G4double aDecayTime)
void CalculateChainsFromParent(const G4ParticleDefinition &)
G4RadioactiveDecay(const G4String &processName="RadioactiveDecay", const G4double timeThresholdForRadioactiveDecays=-1.0)
G4double GetMeanLifeTime(const G4Track &theTrack, G4ForceCondition *condition) override
void SetSourceTimeProfile(const G4String &filename)
G4bool IsRateTableReady(const G4ParticleDefinition &)
G4double ConvolveSourceTimeProfile(const G4double, const G4double)
void AddDeexcitationSpectrumForBiasMode(G4ParticleDefinition *apartDef, G4double weight, G4double currenTime, std::vector< double > &weights_v, std::vector< double > ×_v, std::vector< G4DynamicParticle * > &secondaries_v)
void SetDecayRate(G4int, G4int, G4double, G4int, std::vector< G4double > &, std::vector< G4double > &)
void GetChainsFromParent(const G4ParticleDefinition &)
G4VPhysicalVolume * GetVolume() const
G4double GetWeight() const
void SetWeight(G4double aValue)
const G4ThreeVector & GetPosition() const
void SetTouchableHandle(const G4TouchableHandle &apValue)
G4double GetGlobalTime() const
G4double GetLocalTime() const
const G4DynamicParticle * GetDynamicParticle() const
const G4TouchableHandle & GetTouchableHandle() const
void SetGoodForTrackingFlag(G4bool value=true)
virtual G4DecayProducts * DecayIt(G4double parentMass=-1.0)=0
G4LogicalVolume * GetLogicalVolume() const
G4int GetVerboseLevel() const
void ClearNumberOfInteractionLengthLeft()
G4ParticleChangeForRadDecay fParticleChangeForRadDecay
G4VRadioactiveDecay(const G4String &processName="RadioactiveDecay", const G4double timeThreshold=-1.0)
void DecayAnalog(const G4Track &theTrack, G4DecayTable *)
G4DecayProducts * DoDecay(const G4ParticleDefinition &, G4DecayTable *)
std::vector< G4String > ValidVolumes
G4double GetMeanLifeTime(const G4Track &theTrack, G4ForceCondition *condition) override
G4bool IsApplicable(const G4ParticleDefinition &) override
static const G4double levelTolerance
G4DecayTable * GetDecayTable(const G4ParticleDefinition *)