utilities
Loading...
Searching...
No Matches
gemcUtilities.cc
Go to the documentation of this file.
1#include "gemcUtilities.h"
2#include "gemcConventions.h"
3
4// Implementation notes:
5// - Doxygen documentation is kept authoritative in gemcUtilities.h.
6// - This file only provides brief, non-Doxygen comments to clarify intent and flow.
7
8// geant4 headers
9#include "G4Threading.hh"
10#include "G4UImanager.hh"
11#include "G4UnitsTable.hh"
12
13// gemc
14#include "glogger.h"
15#include "gtouchable.h"
16#include "gutilities.h"
17#include "g4display_options.h"
18
19namespace gemc {
20 // return the number of cores from options.
21 // if 0 is given, returns max number of available cores
22 int get_nthreads(const std::shared_ptr<GOptions>& gopts, const std::shared_ptr<GLogger>& log) {
23 int useThreads = gopts->getScalarInt("nthreads");
24
25 // Geant4 provides a platform-specific core count helper.
26 int ncores = G4Threading::G4GetNumberOfCores();
27
28 // Clamp user request:
29 // - 0 means "use all available cores"
30 // - values larger than available cores are clamped
31 if (useThreads == 0 || useThreads > ncores) useThreads = ncores;
32
33 log->info(0, "Using ", useThreads, " threads out of ", ncores, " available cores.");
34
35 return useThreads;
36 }
37
38 std::vector<std::string> verbosity_commands([[maybe_unused]] const std::shared_ptr<GOptions>& gopts,
39 [[maybe_unused]] const std::shared_ptr<GLogger>& log) {
40 std::vector<std::string> cmds;
41
42 // --- Always-quiet commands ---
43 // These commands reduce Geant4 output noise for typical production runs.
44 cmds.emplace_back("/control/verbose 0");
45 cmds.emplace_back("/hit/verbose 0");
46
47 cmds.emplace_back("/process/verbose 0");
48 cmds.emplace_back("/process/setVerbose 0 all");
49 cmds.emplace_back("/process/had/verbose 0");
50 cmds.emplace_back("/process/had/deex/verbose 0");
51 cmds.emplace_back("/process/had/cascade 0");
52 cmds.emplace_back("/process/em/verbose 0");
53 cmds.emplace_back("/process/eLoss/verbose 0");
54
55 cmds.emplace_back("/tracking/verbose 0");
56 cmds.emplace_back("/geometry/navigator/verbose 0");
57
58 cmds.emplace_back("/event/verbose 0");
59 cmds.emplace_back("/event/stack/verbose 0");
60
61 cmds.emplace_back("/cuts/verbose 0");
62
63 cmds.emplace_back("/run/particle/verbose 0");
64 cmds.emplace_back("/run/verbose 0");
65
66 cmds.emplace_back("/material/verbose 0");
67
68 cmds.emplace_back("/vis/verbose 0");
69 cmds.emplace_back("/particle/verbose 0");
70
71 // cmds.emplace_back("/control/cout/ignoreInitializationCout 1");
72 // cmds.emplace_back("/control/cout/useBuffer 1"); // keep MT output tidy?
73
74 return cmds;
75 }
76
77 std::vector<std::string> initial_commands(const std::shared_ptr<GOptions>& gopts,
78 [[maybe_unused]] const std::shared_ptr<GLogger>& log) {
79 // check_overlaps is typically provided by the Geant4 system options set.
80 auto check_overlaps = gopts->getScalarInt("check_overlaps"); // notice: from g4system options
81 auto gui = gopts->getSwitch("gui");
82 auto g4view = g4display::getG4View(gopts);
83
84 std::vector<std::string> cmds;
85
86 // Batch mode: optionally schedule geometry overlap checks before initialization.
87 // Geant4 overlap checks use the current "/geometry/test/..." configuration.
88 if (check_overlaps == 2) {
89 log->info(0, "Running /geometry/test/run with 50 points.");
90 cmds.emplace_back("/geometry/test/resolution 50");
91 cmds.emplace_back("/geometry/test/run");
92 }
93 else if (check_overlaps >= 100) {
94 log->info(0, "Running /geometry/test/run with ", check_overlaps, " points.");
95 cmds.emplace_back("/geometry/test/resolution " + std::to_string(check_overlaps));
96 cmds.emplace_back("/geometry/test/run");
97 }
98
99 // A re-initialize is required when:
100 // - physics changes
101 // - geometry changes
102 cmds.emplace_back("/run/initialize");
103
104
105
106 // If there is no GUI, or no need for batch screenshot, initialization commands are enough.
107 if (!gui && g4view.driver != "TOOLSSG_OFFSCREEN") return cmds;
108
109 // do not draw volumes in batch screenshot
110 if (g4view.driver != "TOOLSSG_OFFSCREEN") {
111 cmds.emplace_back("/vis/drawVolume");
112 // Disable auto refresh and quieten vis messages whilst scene and trajectories are established.
113 cmds.emplace_back("/vis/viewer/set/autoRefresh false");
114 cmds.emplace_back("/vis/viewer/set/viewpointVector -1 0 0");
115 cmds.emplace_back("/vis/viewer/set/lightsVector -1 0 0");
116 }
117
118 // GUI / batch screenshot mode: set up a minimal visualization scene with trajectories and hits.
119 cmds.emplace_back("/vis/scene/add/trajectories smooth");
120 cmds.emplace_back("/vis/modeling/trajectories/create/drawByCharge");
121 cmds.emplace_back("/vis/modeling/trajectories/drawByCharge-0/default/setDrawStepPts true");
122 cmds.emplace_back("/vis/modeling/trajectories/drawByCharge-0/default/setStepPtsSize 2");
123 cmds.emplace_back("/vis/scene/add/hits");
124 cmds.emplace_back("/vis/scene/endOfEventAction accumulate 10000");
125
126 // Background colors and root volume are the same.
127 // Chosen to be the same as pyvistab for now
128 cmds.push_back("/vis/viewer/set/background 0.05 0.05 0.26");
129
130 // do not draw volumes in batch screenshot
131 if (g4view.driver != "TOOLSSG_OFFSCREEN") {
132 // Re-enable refresh and flush once configuration is complete.
133 cmds.emplace_back("/vis/viewer/set/autoRefresh true");
134 cmds.emplace_back("/vis/viewer/flush");
135 }
136
137
138 // cmds.emplace_back("/tracking/verbose 2");
139 return cmds;
140 }
141
142 // initialize G4MTRunManager
143 void run_manager_commands([[maybe_unused]] const std::shared_ptr<GOptions>& gopts,
144 const std::shared_ptr<GLogger>& log, const std::vector<std::string>& commands) {
145 auto* g4uim = G4UImanager::GetUIpointer();
146
147 // Apply commands sequentially so the UI manager sees the same order as a macro file.
148 for (const auto& cmd : commands) {
149 log->info(2, "Executing UIManager command: ", cmd);
150 g4uim->ApplyCommand(cmd);
151 }
152 }
153
155 new G4UnitDefinition("milligray", "milliGy", "Dose", gemc_units::milligray);
156 new G4UnitDefinition("microgray", "microGy", "Dose", gemc_units::microgray);
157 new G4UnitDefinition("nanogray", "nanoGy", "Dose", gemc_units::nanogray);
158 new G4UnitDefinition("picogray", "picoGy", "Dose", gemc_units::picogray);
159 }
160
161
162#include <unistd.h> // needed for get_pid
163
164 void start_random_engine(const std::shared_ptr<GOptions>& gopts, const std::shared_ptr<GLogger>& log) {
165 auto randomEngineName = gopts->getScalarString("randomEngine");
166 auto seed = gopts->getScalarInt("seed");
167
168 // If the user did not set a seed, derive one using several fast-changing sources.
169 // This helps reduce accidental seed reuse across runs.
170 if (seed == SEEDNOTSET) {
171 auto timed = time(NULL);
172 auto clockd = clock();
173 auto getpidi = getpid();
174 seed = (G4int)( timed - clockd - getpidi );
175 log->info(1, "Using random seed ", seed);
176 } else {
177 log->info(1, "User defined seed ", seed);
178 }
179
180
181 // The names come from the CLHEP library, can be found with
182 // grep ": public HepRandomEngine" *.h $CLHEP_BASE_DIR/include/CLHEP/Random/* | awk -Fclass '{print $2}' | awk -F: '{print $1}'
183 //
184 // Select the engine implementation based on the configured string.
185 if (randomEngineName == "DRand48Engine")
186 G4Random::setTheEngine(new CLHEP::DRand48Engine(seed));
187 else if (randomEngineName == "DualRand")
188 G4Random::setTheEngine(new CLHEP::DualRand(seed));
189 else if (randomEngineName == "Hurd160Engine")
190 G4Random::setTheEngine(new CLHEP::Hurd160Engine(seed));
191 else if (randomEngineName == "HepJamesRandom")
192 G4Random::setTheEngine(new CLHEP::HepJamesRandom(seed));
193 else if (randomEngineName == "MTwistEngine")
194 G4Random::setTheEngine(new CLHEP::MTwistEngine(seed));
195 else if (randomEngineName == "MixMaxRng")
196 G4Random::setTheEngine(new CLHEP::MixMaxRng(seed));
197 else if (randomEngineName == "RandEngine")
198 G4Random::setTheEngine(new CLHEP::RandEngine(seed));
199 else if (randomEngineName == "RanecuEngine")
200 G4Random::setTheEngine(new CLHEP::RanecuEngine(seed));
201 else if (randomEngineName == "Ranlux64Engine")
202 G4Random::setTheEngine(new CLHEP::Ranlux64Engine(seed));
203 else if (randomEngineName == "RanluxEngine")
204 G4Random::setTheEngine(new CLHEP::RanluxEngine(seed));
205 else if (randomEngineName == "RanshiEngine")
206 G4Random::setTheEngine(new CLHEP::RanshiEngine(seed));
207 else if (randomEngineName == "Hurd288Engine")
208 G4Random::setTheEngine(new CLHEP::Hurd288Engine(seed));
209 else if (randomEngineName == "TripleRand")
210 G4Random::setTheEngine(new CLHEP::TripleRand(seed));
211 else { log->error(EC__RANDOMENGINENOTFOUND, "Random engine >", randomEngineName, "< not found. Exiting."); }
212
213 // Apply the seed after selecting the engine so the engine instance is active.
214 log->info(0, "Starting random engine ", randomEngineName, " with seed ", seed);
215 G4Random::setTheSeed(seed);
216 }
217}
Conventional constants used by GEMC utility helpers.
Utility helpers for runtime setup, command preparation, and random-engine initialization.
#define SEEDNOTSET
Sentinel value used to mark that the random seed has not been explicitly set.
#define EC__RANDOMENGINENOTFOUND
Error code used when the configured random engine name is not recognized.
void define_new_gemc_units()
void start_random_engine(const std::shared_ptr< GOptions > &gopts, const std::shared_ptr< GLogger > &log)
Select and start the random engine, then seed it.
std::vector< std::string > verbosity_commands(const std::shared_ptr< GOptions > &gopts, const std::shared_ptr< GLogger > &log)
Build a list of Geant4 UI commands that reduce verbosity across subsystems.
std::vector< std::string > initial_commands(const std::shared_ptr< GOptions > &gopts, const std::shared_ptr< GLogger > &log)
Build a list of Geant4 UI commands needed at startup.
void run_manager_commands(const std::shared_ptr< GOptions > &gopts, const std::shared_ptr< GLogger > &log, const std::vector< std::string > &commands)
Execute a sequence of Geant4 UI commands through the UI manager.
int get_nthreads(const std::shared_ptr< GOptions > &gopts, const std::shared_ptr< GLogger > &log)
Determine the number of worker threads to use for the run.
G4View getG4View(const std::shared_ptr< GOptions > &gopts)
constexpr G4double milligray
constexpr G4double nanogray
constexpr G4double microgray
constexpr G4double picogray