85static std::once_flag applyOnce;
90 if(
nullptr == instance) {
101 for (
auto const & p : loss_vector) {
delete p; }
102 for (
auto const & p : msc_vector) {
delete p; }
103 for (
auto const & p : emp_vector) {
delete p; }
104 for (
auto const & p : p_vector) {
delete p; }
106 std::size_t mod = mod_vector.size();
107 std::size_t fmod = fmod_vector.size();
108 for (std::size_t a=0; a<mod; ++a) {
109 if(
nullptr != mod_vector[a] ) {
110 for (std::size_t b=0; b<fmod; ++b) {
111 if((
G4VEmModel*)(fmod_vector[b]) == mod_vector[a]) {
112 fmod_vector[b] =
nullptr;
115 delete mod_vector[a];
116 mod_vector[a] =
nullptr;
119 for (
auto const & p : fmod_vector) {
delete p; }
123 delete emCorrections;
124 delete emConfigurator;
125 delete emElectronIonPair;
126 delete nielCalculator;
127 delete atomDeexcitation;
128 delete subcutProducer;
139 std::call_once(applyOnce, [
this]() { isMaster =
true; });
140 verbose = isMaster ? theParameters->Verbose() : theParameters->WorkerVerbose();
142 tableBuilder =
new G4LossTableBuilder(isMaster);
143 emCorrections =
new G4EmCorrections(verbose);
146 loss_vector.reserve(n);
147 part_vector.reserve(n);
148 base_part_vector.reserve(n);
149 dedx_vector.reserve(n);
150 range_vector.reserve(n);
151 inv_range_vector.reserve(n);
152 tables_are_built.reserve(n);
154 msc_vector.reserve(10);
155 emp_vector.reserve(16);
156 mod_vector.reserve(150);
157 fmod_vector.reserve(60);
162void G4LossTableManager::Clear()
164 all_tables_are_built =
false;
165 currentLoss =
nullptr;
166 currentParticle =
nullptr;
169 range_vector.clear();
170 inv_range_vector.clear();
174 base_part_vector.clear();
175 tables_are_built.clear();
185 if (
nullptr == p) {
return; }
186 for (
G4int i=0; i<n_loss; ++i) {
187 if(loss_vector[i] == p) {
return; }
190 G4cout <<
"G4LossTableManager::Register G4VEnergyLossProcess : "
194 loss_vector.push_back(p);
195 part_vector.push_back(
nullptr);
196 base_part_vector.push_back(
nullptr);
197 dedx_vector.push_back(
nullptr);
198 range_vector.push_back(
nullptr);
199 inv_range_vector.push_back(
nullptr);
200 tables_are_built.push_back(
false);
201 isActive.push_back(
true);
202 all_tables_are_built =
false;
210 if (!resetParam) {
return; }
212 startInitialisation =
true;
213 verbose = theParameters->Verbose();
215 verbose = theParameters->WorkerVerbose();
217 if(verbose > 0) { theParameters->Dump(); }
220 tableBuilder->InitialiseBaseMaterials();
221 if (
nullptr != nielCalculator) { nielCalculator->Initialise(); }
223 emCorrections->SetVerbose(verbose);
224 if(
nullptr != emConfigurator) { emConfigurator->SetVerbose(verbose); };
225 if(
nullptr != emElectronIonPair) { emElectronIonPair->SetVerbose(verbose); };
226 if(
nullptr != atomDeexcitation) {
227 atomDeexcitation->SetVerboseLevel(verbose);
228 atomDeexcitation->InitialiseAtomicDeexcitation();
231 G4cout <<
"====== G4LossTableManager::ResetParameters "
232 <<
" Nloss=" << loss_vector.size()
233 <<
" run=" << run <<
" master=" << isMaster
242 if (
nullptr == p) {
return; }
243 for (
G4int i=0; i<n_loss; ++i) {
244 if(loss_vector[i] == p) {
245 loss_vector[i] =
nullptr;
255 if (
nullptr == p) {
return; }
256 std::size_t n = msc_vector.size();
257 for (std::size_t i=0; i<n; ++i) {
258 if(msc_vector[i] == p) {
return; }
261 G4cout <<
"G4LossTableManager::Register G4VMultipleScattering : "
264 msc_vector.push_back(p);
271 if (
nullptr == p) {
return; }
272 std::size_t msc = msc_vector.size();
273 for (std::size_t i=0; i<msc; ++i) {
274 if(msc_vector[i] == p) {
275 msc_vector[i] =
nullptr;
285 if (
nullptr == p) {
return; }
286 std::size_t n = emp_vector.size();
287 for (std::size_t i=0; i<n; ++i) {
288 if(emp_vector[i] == p) {
return; }
291 G4cout <<
"G4LossTableManager::Register G4VEmProcess : "
294 emp_vector.push_back(p);
301 if (
nullptr == p) {
return; }
302 std::size_t emp = emp_vector.size();
303 for (std::size_t i=0; i<emp; ++i) {
304 if(emp_vector[i] == p) {
305 emp_vector[i] =
nullptr;
315 if (
nullptr == p) {
return; }
316 std::size_t n = p_vector.size();
317 for (std::size_t i=0; i<n; ++i) {
318 if(p_vector[i] == p) {
return; }
321 G4cout <<
"G4LossTableManager::Register G4VProcess : "
324 p_vector.push_back(p);
331 if (
nullptr == p) {
return; }
332 std::size_t emp = p_vector.size();
333 for (std::size_t i=0; i<emp; ++i) {
334 if(p_vector[i] == p) {
335 p_vector[i] =
nullptr;
345 mod_vector.push_back(p);
347 G4cout <<
"G4LossTableManager::Register G4VEmModel : "
348 << p->
GetName() <<
" " << p <<
" " << mod_vector.size() <<
G4endl;
357 std::size_t n = mod_vector.size();
358 for (std::size_t i=0; i<n; ++i) {
359 if(mod_vector[i] == p) {
360 mod_vector[i] =
nullptr;
370 fmod_vector.push_back(p);
372 G4cout <<
"G4LossTableManager::Register G4VEmFluctuationModel : "
381 std::size_t n = fmod_vector.size();
382 for (std::size_t i=0; i<n; ++i) {
383 if(fmod_vector[i] == p) { fmod_vector[i] =
nullptr; }
393 if (
nullptr == p ||
nullptr == part) {
return; }
394 for (
G4int i=0; i<n_loss; ++i) {
395 if(loss_vector[i] == p) {
return; }
398 G4cout <<
"G4LossTableManager::RegisterExtraParticle "
403 loss_vector.push_back(p);
404 part_vector.push_back(part);
406 dedx_vector.push_back(
nullptr);
407 range_vector.push_back(
nullptr);
408 inv_range_vector.push_back(
nullptr);
409 tables_are_built.push_back(
false);
410 all_tables_are_built =
false;
418 if(aParticle != currentParticle) {
419 currentParticle = aParticle;
420 std::map<PD,G4VEnergyLossProcess*,std::less<PD> >::const_iterator pos;
421 if ((pos = loss_map.find(aParticle)) != loss_map.end()) {
422 currentLoss = (*pos).second;
424 currentLoss =
nullptr;
426 (pos = loss_map.find(theGenericIon)) != loss_map.end()) {
427 currentLoss = (*pos).second;
441 G4cout <<
"G4LossTableManager::PreparePhysicsTable for "
444 <<
" loss_vector " << loss_vector.size()
445 <<
" run=" << run <<
" master=" << isMaster
451 if (
nullptr != emConfigurator) { emConfigurator->PrepareModels(particle, p); }
454 for (
G4int j=0; j<n_loss; ++j) {
455 if (p == loss_vector[j] &&
nullptr == part_vector[j]) {
456 part_vector[j] = particle;
458 theGenericIon = particle;
473 G4cout <<
"G4LossTableManager::PreparePhysicsTable for "
476 <<
" run=" << run <<
" master=" << isMaster
482 if (
nullptr != emConfigurator) { emConfigurator->PrepareModels(particle, p); }
495 G4cout <<
"G4LossTableManager::PreparePhysicsTable for "
498 <<
" run=" << run <<
" master=" << isMaster
504 if (
nullptr != emConfigurator) { emConfigurator->PrepareModels(particle, p); }
515 if(-1 == run && startInitialisation) {
516 if (
nullptr != emConfigurator) { emConfigurator->Clear(); }
518 if (startInitialisation) { resetParam =
true; }
528 G4cout <<
"### G4LossTableManager::LocalPhysicsTable() for "
534 if(-1 == run && startInitialisation) {
535 if (
nullptr != emConfigurator) { emConfigurator->Clear(); }
536 firstParticle = aParticle;
539 if (startInitialisation) {
542 G4cout <<
"===== G4LossTableManager::LocalPhysicsTable() for run "
543 << run <<
" =====" <<
G4endl;
545 currentParticle =
nullptr;
546 startInitialisation =
false;
548 for (
G4int i=0; i<n_loss; ++i) {
549 if (
nullptr != loss_vector[i]) {
550 tables_are_built[i] =
false;
552 tables_are_built[i] =
true;
553 part_vector[i] =
nullptr;
558 all_tables_are_built=
true;
559 for (
G4int i=0; i<n_loss; ++i) {
560 if(p == loss_vector[i]) {
561 tables_are_built[i] =
true;
569 loss_map[part_vector[i]] = p;
575 G4cout <<
" for " << part_vector[i]->GetParticleName();
577 G4cout <<
" active= " << isActive[i]
578 <<
" table= " << tables_are_built[i]
583 }
else if(!tables_are_built[i]) {
584 all_tables_are_built =
false;
589 G4cout <<
"### G4LossTableManager::LocalPhysicsTable end"
592 if(all_tables_are_built) {
594 G4cout <<
"%%%%% All dEdx and Range tables for worker are ready for run "
595 << run <<
" %%%%%" <<
G4endl;
607 G4cout <<
"### G4LossTableManager::BuildPhysicsTable() for "
612 if(-1 == run && startInitialisation) {
613 if(
nullptr != emConfigurator) { emConfigurator->Clear(); }
614 firstParticle = aParticle;
616 if(startInitialisation) {
619 startInitialisation =
false;
621 G4cout <<
"===== G4LossTableManager::BuildPhysicsTable() for run "
622 << run <<
" ===== " << atomDeexcitation <<
G4endl;
624 currentParticle =
nullptr;
625 all_tables_are_built =
false;
627 for (
G4int i=0; i<n_loss; ++i) {
634 tables_are_built[i] =
false;
640 G4cout <<
" active= " << isActive[i]
641 <<
" table= " << tables_are_built[i]
643 if(base_part_vector[i]) {
644 G4cout <<
" base particle "
645 << base_part_vector[i]->GetParticleName();
650 tables_are_built[i] =
true;
651 part_vector[i] =
nullptr;
657 if (all_tables_are_built) {
658 theParameters->SetIsPrintedFlag(
true);
663 all_tables_are_built =
true;
665 for(
G4int i=0; i<n_loss; ++i) {
666 if(p == loss_vector[i] && !tables_are_built[i] &&
nullptr == base_part_vector[i]) {
671 <<
" " << tables_are_built[i] <<
" " << base_part_vector[i]
676 CopyTables(curr_part, curr_proc);
678 loss_map[aParticle] = p;
685 if ( !tables_are_built[i] ) { all_tables_are_built =
false; }
688 G4cout <<
"### G4LossTableManager::BuildPhysicsTable end: "
689 <<
"all_tables_are_built= " << all_tables_are_built <<
" "
699 for (
G4int j=0; j<n_loss; ++j) {
703 if (!tables_are_built[j] && part == base_part_vector[j]) {
704 tables_are_built[j] =
true;
715 loss_map[part_vector[j]] = proc;
722 <<
" for " << part_vector[j]->GetParticleName()
724 <<
" tables are assigned"
737 G4cout <<
" G4LossTableManager::BuildTables(part) for "
741 std::vector<G4PhysicsTable*> t_list;
742 std::vector<G4VEnergyLossProcess*> loss_list;
743 std::vector<G4bool> build_flags;
744 G4VEnergyLossProcess* em =
nullptr;
745 G4VEnergyLossProcess* p =
nullptr;
747 G4PhysicsTable* dedx =
nullptr;
750 G4ProcessVector* pvec =
754 for (i=0; i<n_loss; ++i) {
757 G4bool yes = (aParticle == part_vector[i]);
761 auto ptr =
static_cast<G4VProcess*
>(p);
762 for(
G4int j=0; j<nvec; ++j) {
764 if(ptr == (*pvec)[j]) {
771 if(yes && isActive[i]) {
778 if (!tables_are_built[i]) {
784 tables_are_built[i] =
true;
788 t_list.push_back(dedx);
789 loss_list.push_back(p);
790 build_flags.push_back(val);
796 if (0 == n_dedx || !em) {
797 G4cout <<
"G4LossTableManager WARNING: no DEDX processes for "
804 G4cout <<
" Start to build the sum of " << n_dedx <<
" processes"
806 <<
" buildCSDARange= " << theParameters->BuildCSDARange()
807 <<
" nSubRegions= " << nSubRegions;
809 G4cout <<
" SubCutProducer " << subcutProducer->GetName();
814 if(subcutProducer) { nSubRegions = 0; }
822 tableBuilder->BuildDEDXTable(dedx, t_list);
826 dedx_vector[iem] = dedx;
830 range_vector[iem] = range;
834 inv_range_vector[iem] = invrange;
836 tableBuilder->BuildRangeTable(dedx, range);
837 tableBuilder->BuildInverseRangeTable(range, invrange);
842 std::vector<G4PhysicsTable*> listCSDA;
844 for (i=0; i<n_dedx; ++i) {
849 if(theParameters->BuildCSDARange()) {
852 listCSDA.push_back(dedx);
856 if(theParameters->BuildCSDARange()) {
860 tableBuilder->BuildDEDXTable(dedxCSDA, listCSDA);
865 tableBuilder->BuildRangeTable(dedxCSDA, rCSDA);
870 G4cout <<
"G4LossTableManager::BuildTables: Tables are built for "
881void G4LossTableManager::ParticleHaveNoLoss(
885 ed <<
"Energy loss process not found for " << aParticle->
GetParticleName()
887 G4Exception(
"G4LossTableManager::ParticleHaveNoLoss",
"em0001",
900const std::vector<G4VEnergyLossProcess*>&
915const std::vector<G4VMultipleScattering*>&
925 return theParameters->GetEmSaturation();
932 if(!emConfigurator) {
935 return emConfigurator;
942 if(!emElectronIonPair) {
945 return emElectronIonPair;
952 if(
nullptr != ptr && ptr != nielCalculator) {
953 delete nielCalculator;
954 nielCalculator = ptr;
962 if(!nielCalculator) {
965 return nielCalculator;
972 if(atomDeexcitation != p) {
973 delete atomDeexcitation;
974 atomDeexcitation = p;
982 if(subcutProducer != p) {
983 delete subcutProducer;
992 G4String ss =
"G4LossTableManager::" + tit;
1015 char* dirName = std::getenv(
"G4PhysListDocDir");
1016 char* physList = std::getenv(
"G4PhysListName");
1017 if (dirName && physList) {
1021 std::ofstream outFile;
1022 outFile.open(pathName);
1024 outFile << physListName <<
G4endl;
1025 outFile << std::string(physListName.length(),
'=') <<
G4endl;
1027 std::vector<G4ParticleDefinition*> particles {
1037 std::vector<G4VEnergyLossProcess*> enloss_vector =
1039 std::vector<G4VMultipleScattering*> mscat_vector =
1042 for (
auto theParticle : particles) {
1043 outFile <<
G4endl <<
"**" << theParticle->GetParticleName()
1050 for (
auto emproc : emproc_vector) {
1051 for (
G4int i = 0; i < plen; ++i) {
1053 if (proc == emproc) {
1061 for (
auto mscproc : mscat_vector) {
1062 for (
G4int i = 0; i < plen; ++i) {
1064 if (proc == mscproc) {
1072 for (
auto enlossproc : enloss_vector) {
1073 for (
G4int i = 0; i < plen; ++i) {
1075 if (proc == enlossproc) {
void G4Exception(const char *originOfException, const char *exceptionCode, G4ExceptionSeverity severity, const char *description)
std::ostringstream G4ExceptionDescription
G4GLOB_DLL std::ostream G4cout
static G4Electron * Electron()
static G4EmParameters * Instance()
void SetAtomDeexcitation(G4VAtomDeexcitation *)
static G4LossTableManager * Instance()
const std::vector< G4VEmProcess * > & GetEmProcessVector()
G4VEnergyLossProcess * GetEnergyLossProcess(const G4ParticleDefinition *)
void LocalPhysicsTables(const G4ParticleDefinition *aParticle, G4VEnergyLossProcess *p)
void BuildPhysicsTable(const G4ParticleDefinition *aParticle)
const std::vector< G4VMultipleScattering * > & GetMultipleScatteringVector()
void SetVerbose(G4int val)
void DeRegister(G4VEnergyLossProcess *p)
G4NIELCalculator * NIELCalculator()
void SetNIELCalculator(G4NIELCalculator *)
G4EmConfigurator * EmConfigurator()
void Register(G4VEnergyLossProcess *p)
G4ElectronIonPair * ElectronIonPair()
G4EmSaturation * EmSaturation()
const std::vector< G4VEnergyLossProcess * > & GetEnergyLossProcessVector()
void PreparePhysicsTable(const G4ParticleDefinition *aParticle, G4VEnergyLossProcess *p)
void SetSubCutProducer(G4VSubCutProducer *)
void RegisterExtraParticle(const G4ParticleDefinition *aParticle, G4VEnergyLossProcess *p)
G4LossTableManager(G4LossTableManager &)=delete
static G4MuonMinus * MuonMinusDefinition()
static G4MuonPlus * MuonPlusDefinition()
G4ProcessManager * GetProcessManager() const
G4double GetPDGCharge() const
const G4String & GetParticleName() const
static G4PhysicsTable * PreparePhysicsTable(G4PhysicsTable *physTable)
static G4Positron * Positron()
G4int GetProcessListLength() const
G4ProcessVector * GetProcessList() const
static G4Proton * ProtonDefinition()
const G4String & GetName() const
const G4String & GetName() const
const G4ParticleDefinition * BaseParticle() const
G4PhysicsTable * RangeTableForLoss() const
G4PhysicsTable * InverseRangeTable() const
G4PhysicsTable * CSDARangeTable() const
void SetRangeTableForLoss(G4PhysicsTable *p)
G4int NumberOfSubCutoffRegions() const
G4PhysicsTable * BuildDEDXTable(G4EmTableType tType=fRestricted)
const G4ParticleDefinition * Particle() const
void SetInverseRangeTable(G4PhysicsTable *p)
G4bool IsIonisationProcess() const
void SetDEDXTable(G4PhysicsTable *p, G4EmTableType tType)
G4PhysicsTable * BuildLambdaTable(G4EmTableType tType=fRestricted)
void SetLambdaTable(G4PhysicsTable *p)
G4PhysicsTable * IonisationTable() const
G4PhysicsTable * LambdaTable() const
void SetCSDARangeTable(G4PhysicsTable *pRange)
G4PhysicsTable * DEDXunRestrictedTable() const
G4PhysicsTable * DEDXTable() const
virtual void ProcessDescription(std::ostream &outfile) const
const G4String & GetProcessName() const