103std::map<G4int, G4String>* G4VRadioactiveDecay::theUserRDataFiles =
nullptr;
104G4String G4VRadioactiveDecay::dirPath =
"";
114 fThresholdForVeryLongDecayTime( 1.0*
CLHEP::year )
117 G4cout <<
"G4VRadioactiveDecay constructor: processName = " << processName
127 if (dirPath.empty()) {
129 if (
nullptr == path_var) {
131 "Environment variable G4RADIOACTIVEDATA is not set");
134 std::ostringstream os;
135 os << dirPath <<
"/z1.a3";
136 std::ifstream testFile;
137 testFile.open(os.str() );
138 if ( !testFile.is_open() )
140 "Environment variable G4RADIOACTIVEDATA is set, but does not point to correct directory");
153 if (
nullptr == theUserRDataFiles) {
154 theUserRDataFiles =
new std::map<G4int, G4String>;
170 if ( timeThreshold > 0.0 || timeThresholdBis > 0.0 ) {
171 if ( timeThreshold > timeThresholdBis ) timeThresholdBis = timeThreshold;
172 fThresholdForVeryLongDecayTime = timeThresholdBis;
191 delete theUserRDataFiles;
192 theUserRDataFiles =
nullptr;
201 if (pname ==
"GenericIon" || pname ==
"triton") {
return true; }
203 const G4Ions* p =
dynamic_cast<const G4Ions*
>(&aParticle);
204 if (
nullptr == p) {
return false; }
211 if (lifeTime < 0.0 || lifeTime > fThresholdForVeryLongDecayTime) {
219 if (
A > theNucleusLimits.GetAMax() ||
A < theNucleusLimits.GetAMin() ||
220 Z > theNucleusLimits.GetZMax() || Z < theNucleusLimits.GetZMin()) {
231 return DecayIt(theTrack, theStep);
238 return DecayIt(theTrack, theStep);
244 outFile <<
"The radioactive decay process (G4VRadioactiveDecay) handles the\n"
245 <<
"alpha, beta+, beta-, electron capture and isomeric transition\n"
246 <<
"decays of nuclei (G4GenericIon) with masses A > 4.\n"
247 <<
"The required half-lives and decay schemes are retrieved from\n"
248 <<
"the RadioactiveDecay database which was derived from ENSDF.\n";
262 const G4Ions* ion =
dynamic_cast<const G4Ions*
>(aNucleus);
263 if (
nullptr != ion) {
267 theDecayTable = ptr->second;
269 return theDecayTable;
277 volume = theLogicalVolumes->
GetVolume(aVolume);
278 if (volume !=
nullptr)
285 G4cout <<
" Radioactive decay applied to " << aVolume <<
G4endl;
290 ed << aVolume <<
" is not a valid logical volume name."
291 <<
" Decay not activated for it."
293 G4Exception(
"G4VRadioactiveDecay::SelectAVolume()",
"HAD_RDM_300",
303 volume = theLogicalVolumes->
GetVolume(aVolume);
304 if (volume !=
nullptr)
313 G4cout <<
" G4VRadioactiveDecay::DeselectAVolume: " << aVolume
314 <<
" is removed from list " <<
G4endl;
319 ed << aVolume <<
" is not in the list. No action taken." <<
G4endl;
320 G4Exception(
"G4VRadioactiveDecay::DeselectAVolume()",
"HAD_RDM_300",
327 ed << aVolume <<
" is not a valid logical volume name. No action taken."
329 G4Exception(
"G4VRadioactiveDecay::DeselectAVolume()",
"HAD_RDM_300",
344 for (std::size_t i = 0; i < theLogicalVolumes->size(); ++i){
345 volume = (*theLogicalVolumes)[i];
374 if (!
IsApplicable(*theParticleDef)) {
return meanlife; }
378 G4cout <<
"G4VRadioactiveDecay::GetMeanLifeTime() for "
381 <<
" Mass(GeV)=" << theParticleDef->
GetPDGMass()/CLHEP::GeV
382 <<
" LifeTime(ns)=" << theLife/CLHEP::ns <<
G4endl;
385 if (theLife >= 0.0 && theLife <= fThresholdForVeryLongDecayTime) {
390 const G4Ions* ion =
dynamic_cast<const G4Ions*
>(theParticleDef);
398 G4cout <<
"G4VRadioactiveDecay::GetMeanLifeTime: "
399 << meanlife/CLHEP::s <<
" second " <<
G4endl;
417 if (lifeTime > 0.0 && lifeTime <
DBL_MAX) {
426 G4cout <<
"G4VRadioactiveDecay::GetMeanFreePath() for "
429 <<
" lifeTime(ns)=" << lifeTime
430 <<
" mean free path(cm)=" << res/CLHEP::cm <<
G4endl;
444 if (isInitialised) {
return; }
445 isInitialised =
true;
473 G4long prec = os.precision(5);
474 os <<
"======================================================================"
476 os <<
"====== Radioactive Decay Physics Parameters ======="
478 os <<
"======================================================================"
480 os <<
"min MeanLife (from G4NuclideTable) "
481 <<
G4BestUnit(minMeanLife,
"Time") << endline;
482 os <<
"Max life time (from G4DeexPrecoParameters) "
484 os <<
"Internal e- conversion flag "
486 os <<
"Stored internal conversion coefficients "
488 os <<
"Enabled atomic relaxation mode "
489 << applyARM << endline;
490 os <<
"Enable correlated gamma emission "
492 os <<
"Max 2J for sampling of angular correlations "
494 os <<
"Atomic de-excitation enabled "
495 << emparam->
Fluo() << endline;
496 os <<
"Auger electron emission enabled "
497 << emparam->
Auger() << endline;
498 os <<
"Check EM cuts disabled for atomic de-excitation "
500 os <<
"Use Bearden atomic level energies "
502 os <<
"Use ANSTO fluorescence model "
504 os <<
"Threshold for very long decay time at rest "
505 <<
G4BestUnit(fThresholdForVeryLongDecayTime,
"Time") << endline;
506 os <<
"======================================================================"
526 return dtptr->second;
542 auto ptr = theUserRDataFiles->find(ke);
543 if (ptr != theUserRDataFiles->end()) {
546 std::ostringstream os;
547 os << dirPath <<
"/z" << Z <<
".a" <<
A <<
'\0';
554 std::ifstream DecaySchemeFile;
555 DecaySchemeFile.open(file);
557 if (DecaySchemeFile.good()) {
561 G4double modeTotalBR[nMode] = {0.0};
563 for (
G4int i = 0; i < nMode; ++i) {
567 char inputChars[120]={
' '};
589 while (!complete && !DecaySchemeFile.getline(inputChars, 120).eof()) {
592 G4Exception(
"G4VRadioactiveDecay::LoadDecayTable()",
"HAD_RDM_100",
597 inputLine = inputChars;
598 G4StrUtil::rstrip(inputLine);
599 if (inputChars[0] !=
'#' && inputLine.length() != 0) {
600 std::istringstream tmpStream(inputLine);
602 if (inputChars[0] ==
'P') {
605 tmpStream >> recordType >> parentExcitation >> floatingFlag >> dummy;
613 found = (std::abs(parentExcitation*keV - levelEnergy) <
levelTolerance);
614 if (floatingLevel !=
noFloat) {
617 if (!floatMatch) found =
false;
626 if (inputLine.length() < 72) {
627 tmpStream >> theDecayMode >> dummy >> decayModeTotal;
629 switch (theDecayMode) {
633 theDecayTable->
Insert(anITChannel);
637 modeTotalBR[
BetaMinus] = decayModeTotal;
break;
639 modeTotalBR[
BetaPlus] = decayModeTotal;
break;
641 modeTotalBR[
KshellEC] = decayModeTotal;
break;
643 modeTotalBR[
LshellEC] = decayModeTotal;
break;
645 modeTotalBR[
MshellEC] = decayModeTotal;
break;
647 modeTotalBR[
NshellEC] = decayModeTotal;
break;
649 modeTotalBR[
Alpha] = decayModeTotal;
break;
651 modeTotalBR[
Proton] = decayModeTotal;
break;
653 modeTotalBR[
Neutron] = decayModeTotal;
break;
655 modeTotalBR[
SpFission] = decayModeTotal;
break;
669 modeTotalBR[
Triton] = decayModeTotal;
break;
673 G4Exception(
"G4VRadioactiveDecay::LoadDecayTable()",
"HAD_RDM_000",
678 if (inputLine.length() < 84) {
679 tmpStream >> theDecayMode >> a >> daughterFloatFlag >> b >> c;
682 tmpStream >> theDecayMode >> a >> daughterFloatFlag >> b >> c >> betaType;
692 switch (theDecayMode) {
697 daughterFloatLevel, betaType);
699 theDecayTable->
Insert(aBetaMinusChannel);
708 daughterFloatLevel, betaType);
710 theDecayTable->
Insert(aBetaPlusChannel);
721 aKECChannel->
SetARM(applyARM);
722 theDecayTable->
Insert(aKECChannel);
733 aLECChannel->
SetARM(applyARM);
734 theDecayTable->
Insert(aLECChannel);
745 aMECChannel->
SetARM(applyARM);
746 theDecayTable->
Insert(aMECChannel);
757 aNECChannel->
SetARM(applyARM);
758 theDecayTable->
Insert(aNECChannel);
769 theDecayTable->
Insert(anAlphaChannel);
770 modeSumBR[
Alpha] += b;
780 theDecayTable->
Insert(aProtonChannel);
791 theDecayTable->
Insert(aNeutronChannel);
799 new G4SFDecay(theIon, b, c*MeV, a*MeV, daughterFloatLevel);
800 theDecayTable->
Insert(aSpontFissChannel);
838 new G4TritonDecay(theIon, b, c*MeV, a*MeV, daughterFloatLevel);
839 theDecayTable->
Insert(aTritonChannel);
847 G4Exception(
"G4VRadioactiveDecay::LoadDecayTable()",
"HAD_RDM_000",
865 theNuclearDecayChannel =
static_cast<G4NuclearDecay*
>(theChannel);
868 if (theDecayMode !=
IT) {
869 theBR = theChannel->
GetBR();
870 theChannel->
SetBR(theBR*modeTotalBR[theDecayMode]/modeSumBR[theDecayMode]);
875 DecaySchemeFile.close();
877 if (!found && levelEnergy > 0) {
881 theDecayTable->
Insert(anITChannel);
891 return theDecayTable;
897 if (Z < 1 ||
A < 2)
G4cout <<
"Z and A not valid!" <<
G4endl;
899 std::ifstream DecaySchemeFile(filename);
900 if (DecaySchemeFile) {
901 G4int ID_ion =
A*1000 + Z;
902 (*theUserRDataFiles)[ID_ion] = filename;
905 ed << filename <<
" does not exist! " <<
G4endl;
906 G4Exception(
"G4VRadioactiveDecay::AddUserDecayDataFile()",
"HAD_RDM_001",
932 G4cout <<
"G4VRadioactiveDecay::DecayIt : "
934 <<
" is not selected for the RDM"<<
G4endl;
936 G4cout <<
" The Valid volumes are: ";
953 if ( theDecayTable ==
nullptr || theDecayTable->
entries() == 0) {
957 G4cout <<
"G4VRadioactiveDecay::DecayIt : "
959 <<
" is outside (Z,A) limits set for the decay or has no decays."
990 if (
nullptr == products || products->
entries() == 1) {
1023 if (temptime < 0.) temptime = 0.;
1024 finalGlobalTime += temptime;
1025 finalLocalTime += temptime;
1038 if ( finalGlobalTime > fThresholdForVeryLongDecayTime ) {
1047 products->
Boost(ParentEnergy, ParentDirection);
1054 G4cout <<
"G4VRadioactiveDecay::DecayAnalog: Decay vertex :";
1055 G4cout <<
" Time: " << finalGlobalTime/
ns <<
"[ns]";
1060 G4cout <<
"G4Decay::DecayIt : decay products in Lab. Frame" <<
G4endl;
1066 G4int modelID = modelID_forIT + 10*theRadDecayMode;
1067 const G4int modelID_forAtomicRelaxation =
1069 for (
G4int index = 0; index < numberOfSecondaries; ++index ) {
1075 if ( theRadDecayMode ==
IT && index > 0 ) {
1076 if ( index == numberOfSecondaries-1 ) {
1082 index < numberOfSecondaries-1 ) {
1115 if (theDecayChannel ==
nullptr) {
1119 G4Exception(
"G4VRadioactiveDecay::DoDecay",
"HAD_RDM_013",
1125 G4cout <<
"G4VRadioactiveDecay::DoIt : selected decay channel addr: "
1126 << theDecayChannel <<
G4endl;
1129 theRadDecayMode =
static_cast<G4NuclearDecay*
>(theDecayChannel)->GetDecayMode();
1132 if (theRadDecayMode ==
IT) {
1133 decayIT->SetupDecay(&theParticleDef);
1134 products =
decayIT->DecayIt(0.0);
1151 if (origin == forceDecayDirection)
return;
1152 if (180.*deg == forceDecayHalfAngle)
return;
1153 if (0 == products || 0 == products->
entries())
return;
1173 if (daughterType == electron || daughterType == positron ||
1174 daughterType == neutron || daughterType == gamma ||
1175 daughterType == alpha || daughterType == triton || daughterType == proton) {
1184 G4cout <<
"CollimateDecayProduct for daughter "
1197 if (origin == forceDecayDirection)
return origin;
1198 if (forceDecayHalfAngle == 180.*deg)
return origin;
1203 if (forceDecayHalfAngle > 0.) {
1206 G4double cosMin = std::cos(forceDecayHalfAngle);
1215 G4cout <<
" ChooseCollimationDirection returns " << dir <<
G4endl;
G4TemplateAutoLock< G4Mutex > G4AutoLock
const char * G4FindDataDir(const char *)
void G4Exception(const char *originOfException, const char *exceptionCode, G4ExceptionSeverity severity, const char *description)
std::ostringstream G4ExceptionDescription
@ G4RadioactiveDecayModeSize
std::map< G4String, G4DecayTable * > DecayTableMap
#define G4MUTEX_INITIALIZER
CLHEP::Hep3Vector G4ThreeVector
std::map< G4String, G4DecayTable * > DecayTableMap
G4GLOB_DLL std::ostream G4cout
static G4Alpha * Definition()
G4DynamicParticle * PopProducts()
void Boost(G4double totalEnergy, const G4ThreeVector &momentumDirection)
G4VDecayChannel * GetDecayChannel(G4int index) const
void Insert(G4VDecayChannel *aChannel)
G4VDecayChannel * SelectADecayChannel(G4double parentMass=-1.)
G4bool CorrelatedGamma() const
G4bool StoreICLevelData() const
G4double GetMaxLifeTime() const
G4bool GetInternalConversionFlag() const
void SetMomentumDirection(const G4ThreeVector &aDirection)
const G4ThreeVector & GetMomentumDirection() const
const G4ParticleDefinition * GetParticleDefinition() const
G4ParticleDefinition * GetDefinition() const
G4double GetKineticEnergy() const
G4double GetTotalEnergy() const
void SetARM(G4bool onoff)
static G4Electron * Definition()
static G4EmParameters * Instance()
G4bool DeexcitationIgnoreCut() const
static G4Gamma * Definition()
static G4HadronicParameters * Instance()
G4double GetTimeThresholdForRadioactiveDecay() const
void RegisterExtraProcess(G4VProcess *)
void RegisterParticleForExtraProcess(G4VProcess *, const G4ParticleDefinition *)
static G4HadronicProcessStore * Instance()
void PrintInfo(const G4ParticleDefinition *)
G4double GetExcitationEnergy() const
G4Ions::G4FloatLevelBase GetFloatLevelBase() const
static G4Ions::G4FloatLevelBase FloatLevelBase(char flbChar)
G4LogicalVolume * GetVolume(const G4String &name, G4bool verbose=true, G4bool reverseSearch=false) const
static G4LogicalVolumeStore * GetInstance()
const G4String & GetName() const
static G4Neutron * Definition()
G4RadioactiveDecayMode GetDecayMode() const
G4DeexPrecoParameters * GetParameters()
static G4NuclearLevelData * GetInstance()
G4double GetMeanLifeThreshold()
static G4NuclideTable * GetInstance()
G4int GetAtomicNumber() const
G4double GetPDGMass() const
G4int GetAtomicMass() const
G4double GetPDGLifeTime() const
const G4String & GetParticleName() const
static G4int GetModelID(const G4int modelIndex)
static G4Positron * Definition()
static G4Proton * Definition()
G4TrackStatus GetTrackStatus() const
G4double GetVelocity() const
const G4ParticleDefinition * GetParticleDefinition() const
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
G4ParticleDefinition * GetDefinition() const
const G4DynamicParticle * GetDynamicParticle() const
const G4TouchableHandle & GetTouchableHandle() const
G4double GetKineticEnergy() const
void SetCreatorModelID(const G4int id)
void SetGoodForTrackingFlag(G4bool value=true)
static G4Triton * Definition()
void SetBR(G4double value)
virtual G4DecayProducts * DecayIt(G4double parentMass=-1.0)=0
G4LogicalVolume * GetLogicalVolume() const
G4int GetVerboseLevel() const
void ClearNumberOfInteractionLengthLeft()
void SetProcessSubType(G4int)
G4VParticleChange * pParticleChange
virtual G4VParticleChange * DecayIt(const G4Track &theTrack, const G4Step &theStep)
~G4VRadioactiveDecay() override
G4ParticleChangeForRadDecay fParticleChangeForRadDecay
G4VRadioactiveDecay(const G4String &processName="RadioactiveDecay", const G4double timeThreshold=-1.0)
void DecayAnalog(const G4Track &theTrack, G4DecayTable *)
void AddUserDecayDataFile(G4int Z, G4int A, const G4String &filename)
G4PhotonEvaporation * photonEvaporation
void ProcessDescription(std::ostream &outFile) const override
G4DecayProducts * DoDecay(const G4ParticleDefinition &, G4DecayTable *)
void BuildPhysicsTable(const G4ParticleDefinition &) override
std::vector< G4String > ValidVolumes
G4VParticleChange * AtRestDoIt(const G4Track &theTrack, const G4Step &theStep) override
void SelectAVolume(const G4String &aVolume)
G4double GetMeanFreePath(const G4Track &theTrack, G4double previousStepSize, G4ForceCondition *condition) override
G4RadioactiveDecayMessenger * theRadioactiveDecayMessenger
void StreamInfo(std::ostream &os, const G4String &endline)
void CollimateDecay(G4DecayProducts *products)
G4DecayTable * LoadDecayTable(const G4Ions *)
G4double GetMeanLifeTime(const G4Track &theTrack, G4ForceCondition *condition) override
G4bool IsApplicable(const G4ParticleDefinition &) override
G4ThreeVector ChooseCollimationDirection() const
static DecayTableMap * master_dkmap
static const G4double levelTolerance
void DeselectAVolume(const G4String &aVolume)
void DeselectAllVolumes()
G4DecayTable * GetDecayTable(const G4ParticleDefinition *)
G4VParticleChange * PostStepDoIt(const G4Track &theTrack, const G4Step &theStep) override
void CollimateDecayProduct(G4DynamicParticle *product)
G4VRestDiscreteProcess(const G4String &, G4ProcessType aType=fNotDefined)