31 std::string defaultGeomList{
"ALICE2"};
32 if (isUpgrade ==
true) {
33 defaultGeomList =
"ALICE3";
36 int nsimworkersdefault = std::max(1u, std::thread::hardware_concurrency() / 2);
37 options.add_options()(
38 "mcEngine,e", bpo::value<std::string>()->default_value(
"TGeant4"),
"VMC backend to be used.")(
39 "generator,g", bpo::value<std::string>()->default_value(
"boxgen"),
"Event generator to be used.")(
40 "trigger,t", bpo::value<std::string>()->default_value(
""),
"Event generator trigger to be used.")(
41 "modules,m", bpo::value<std::vector<std::string>>()->multitoken()->default_value(std::vector<std::string>({
"all"}),
"all modules"),
"list of modules included in geometry")(
42 "skipModules", bpo::value<std::vector<std::string>>()->multitoken()->default_value(std::vector<std::string>({
""}),
""),
"list of modules excluded in geometry (precendence over -m")(
43 "readoutDetectors", bpo::value<std::vector<std::string>>()->multitoken()->default_value(std::vector<std::string>(),
""),
"list of detectors creating hits, all if not given; added to to active modules")(
44 "skipReadoutDetectors", bpo::value<std::vector<std::string>>()->multitoken()->default_value(std::vector<std::string>(),
""),
"list of detectors to skip hit creation (precendence over --readoutDetectors")(
45 "detectorList", bpo::value<std::string>()->default_value(defaultGeomList),
46 "Use a specific version of ALICE, e.g., a predefined list."
47 "There is an 'official' list provided with:"
48 "\nALICE2 : The default configuration for Run 3"
49 "\nALICE2.1: The future configuration for Run 4"
50 "\nALICE3 : The far-future configuration for Run 5-6"
51 "\nAdditionally one can provide their own custom list of modules which should be included in the geometry."
52 "\nBy specifiying LIST:JSONFILE where LIST is a list present in JSONFILE.")(
53 "nEvents,n", bpo::value<unsigned int>()->default_value(0),
"number of events")(
54 "startEvent", bpo::value<unsigned int>()->default_value(0),
"index of first event to be used (when applicable)")(
55 "extKinFile", bpo::value<std::string>()->default_value(
"Kinematics.root"),
56 "name of kinematics file for event generator from file (when applicable)")(
57 "embedIntoFile", bpo::value<std::string>()->default_value(
""),
58 "filename containing the reference events to be used for the embedding")(
59 "bMax,b", bpo::value<float>()->default_value(0.),
"maximum value for impact parameter sampling (when applicable)")(
60 "isMT", bpo::value<bool>()->default_value(
false),
"multi-threaded mode (Geant4 only")(
61 "outPrefix,o", bpo::value<std::string>()->default_value(
"o2sim"),
"prefix of output files")(
62 "logseverity", bpo::value<std::string>()->default_value(
"INFO"),
"severity level for FairLogger")(
63 "logverbosity", bpo::value<std::string>()->default_value(
"medium"),
"level of verbosity for FairLogger (low, medium, high, veryhigh)")(
64 "configKeyValues", bpo::value<std::string>()->default_value(
""),
"semicolon separated key=value strings (e.g.: 'TPC.gasDensity=1;...")(
65 "configFile", bpo::value<std::string>()->default_value(
""),
"Path to an INI or JSON configuration file")(
66 "chunkSize", bpo::value<unsigned int>()->default_value(500),
"max size of primary chunk (subevent) distributed by server")(
67 "chunkSizeI", bpo::value<int>()->default_value(-1),
"internalChunkSize")(
68 "seed", bpo::value<ULong_t>()->default_value(0),
"initial seed as ULong_t (default: 0 == random)")(
69 "field", bpo::value<std::string>()->default_value(
"-5"),
"L3 field rounded to kGauss, allowed values +-2,+-5 and 0; +-<intKGaus>U for uniform field; \"ccdb\" for taking it from CCDB ")(
"vertexMode", bpo::value<std::string>()->default_value(
"kDiamondParam"),
"Where the beam-spot vertex should come from. Must be one of kNoVertex, kDiamondParam, kCCDB")(
70 "nworkers,j", bpo::value<int>()->default_value(nsimworkersdefault),
"number of parallel simulation workers (only for parallel mode)")(
71 "noemptyevents",
"only writes events with at least one hit")(
72 "CCDBUrl", bpo::value<std::string>()->default_value(
"http://alice-ccdb.cern.ch"),
"URL for CCDB to be used.")(
73 "timestamp", bpo::value<uint64_t>(),
"global timestamp value in ms (for anchoring) - default is now ... or beginning of run if ALICE run number was given")(
74 "run", bpo::value<int>()->default_value(-1),
"ALICE run number")(
75 "asservice", bpo::value<bool>()->default_value(
false),
"run in service/server mode")(
76 "noGeant", bpo::bool_switch(),
"prohibits any Geant transport/physics (by using tight cuts)")(
77 "forwardKine", bpo::bool_switch(),
"forward kinematics on a FairMQ channel")(
78 "noDiscOutput", bpo::bool_switch(),
"switch off writing sim results to disc (useful in combination with forwardKine)");
79 options.add_options()(
"fromCollContext", bpo::value<std::string>()->default_value(
""),
"Use a pregenerated collision context to infer number of events to simulate, how to embedd them, the vertex position etc. Takes precedence of other options such as \"--nEvents\". The format is COLLISIONCONTEXTFILE.root[:SIGNALNAME] where SIGNALNAME is the event part in the context which is relevant.");
82void SimConfig::determineActiveModules(std::vector<std::string>
const& inputargs, std::vector<std::string>
const& skippedModules, std::vector<std::string>& activeModules,
bool isUpgrade)
88 activeModules = inputargs;
90 if (activeModules[0] !=
"all") {
92 for (
int i = 0;
i < activeModules.size(); ++
i) {
93 if (activeModules[
i] !=
"A3IP" &&
94 activeModules[
i] !=
"IT3" &&
95 activeModules[
i] !=
"TRK" &&
96 activeModules[
i] !=
"FT3" &&
97 activeModules[
i] !=
"FCT" &&
98 activeModules[
i] !=
"TF3" &&
99 activeModules[
i] !=
"RCH" &&
100 activeModules[
i] !=
"MI3" &&
101 activeModules[
i] !=
"ECL") {
102 LOGP(fatal,
"List of active modules contains {}, which is not a module from the upgrades.", activeModules[
i]);
107 for (
int i = 0;
i < activeModules.size(); ++
i) {
108 if (activeModules[
i] ==
"A3IP" ||
109 activeModules[
i] ==
"TRK" ||
110 activeModules[
i] ==
"FT3" ||
111 activeModules[
i] ==
"FCT" ||
112 activeModules[
i] ==
"TF3" ||
113 activeModules[
i] ==
"RCH" ||
114 activeModules[
i] ==
"MI3" ||
115 activeModules[
i] ==
"ECL") {
116 LOGP(fatal,
"List of active modules contains {}, which is not a run 3 module", activeModules[
i]);
122 if (activeModules.size() == 1 && activeModules[0] ==
"all") {
123 activeModules.clear();
124#ifdef ENABLE_UPGRADES
127 if (d == DetID::TRK ||
137 activeModules.emplace_back(
"A3IP");
138 activeModules.emplace_back(
"A3ABSO");
139 activeModules.emplace_back(
"A3MAG");
143 activeModules.emplace_back(
"HALL");
144 activeModules.emplace_back(
"MAG");
145 activeModules.emplace_back(
"DIPO");
146 activeModules.emplace_back(
"COMP");
147 activeModules.emplace_back(
"PIPE");
148 activeModules.emplace_back(
"ABSO");
149 activeModules.emplace_back(
"SHIL");
151#ifdef ENABLE_UPGRADES
152 if (d != DetID::IT3 && d != DetID::TRK && d != DetID::FT3 && d != DetID::FCT && d != DetID::TF3 && d != DetID::RCH && d != DetID::ECL && d != DetID::MI3) {
161 filterSkippedElements(activeModules, skippedModules);
168 if (
auto pos =
version.find(
':');
pos != std::string::npos) {
172 LOGP(error,
"Could not parse {}; check errors above!", ppath);
175 if (map.find(pversion) == map.end()) {
176 LOGP(error,
"List {} is not defined in custom JSON file!", pversion);
180 modules = map[pversion];
181 LOGP(info,
"Running with version {} from custom detector list '{}'", pversion, ppath);
184 auto o2env = std::getenv(
"O2_ROOT");
186 LOGP(error,
"O2_ROOT environment not defined");
189 const std::string rootpath(fmt::format(
"{}/share/config/o2simdefaultdetectorlist.json", o2env));
191 LOGP(error,
"Could not parse {} -> check errors above!", rootpath);
194 if (map.find(
version) == map.end()) {
195 LOGP(error,
"List {} is not defined in 'official' JSON file!",
version);
200 LOGP(info,
"Running with official detector version '{}'",
version);
203 if (inputargs.size() != 1 || inputargs[0] !=
"all") {
204 std::vector<std::string> diff;
205 for (
const auto& in : inputargs) {
206 if (std::find(modules.begin(), modules.end(), in) == std::end(modules)) {
207 diff.emplace_back(in);
211 LOGP(error,
"Modules specified that are not present in detector list {}",
version);
212 for (
int j{0};
const auto&
m : diff) {
213 LOGP(info,
" - {: <2}. {}",
j++,
m);
220 std::copy_if(modules.begin(), modules.end(), std::back_inserter(activeModules),
221 [&inputargs](
const auto& e) { return (inputargs.size() == 1 && inputargs[0] ==
"all") || (std::find(inputargs.begin(), inputargs.end(), e) != inputargs.end()); });
222 return filterSkippedElements(activeModules, skippedModules);
225void SimConfig::determineReadoutDetectors(std::vector<std::string>
const& activeModules, std::vector<std::string>
const& enableReadout, std::vector<std::string>
const& disableReadout, std::vector<std::string>& readoutDetectors)
229 readoutDetectors.clear();
231 auto isDet = [](std::string
const& s) {
235 if (enableReadout.empty()) {
237 for (
auto& am : activeModules) {
242 readoutDetectors.emplace_back(am);
245 for (
auto& er : enableReadout) {
248 LOG(fatal) <<
"Enabled readout for " << er <<
" which is not a detector.";
250 if (std::find(activeModules.begin(), activeModules.end(), er) == activeModules.end()) {
252 LOG(fatal) <<
"Module " << er <<
" not constructed and cannot be used for readout (make sure it is contained in -m option).";
254 readoutDetectors.emplace_back(er);
257 for (
auto& dr : disableReadout) {
260 LOG(fatal) <<
"Disabled readout for " << dr <<
" which is not a detector.";
262 if (std::find(activeModules.begin(), activeModules.end(), dr) == activeModules.end()) {
264 LOG(fatal) <<
"Module " << dr <<
" not constructed, makes no sense to disable its readout (make sure it is contained in -m option).";
266 auto iter = std::find(readoutDetectors.begin(), readoutDetectors.end(), dr);
267 if (iter != readoutDetectors.end()) {
268 readoutDetectors.erase(iter);
291 mConfigData.
mMCEngine = vm[
"mcEngine"].as<std::string>();
292 mConfigData.
mNoGeant = vm[
"noGeant"].as<
bool>();
299 if (!
determineActiveModulesList(vm[
"detectorList"].as<std::string>(), vm[
"modules"].as<std::vector<std::string>>(), vm[
"skipModules"].as<std::vector<std::string>>(), mConfigData.
mActiveModules)) {
307 mConfigData.
mMCEngine =
"O2TrivialMCEngine";
308 }
else if (mConfigData.
mMCEngine.compare(
"O2TrivialMCEngine") == 0) {
309 LOG(error) <<
"The O2TrivialMCEngine engine can only be used with --noGeant option";
318 mConfigData.
mGenerator = vm[
"generator"].as<std::string>();
319 mConfigData.
mTrigger = vm[
"trigger"].as<std::string>();
320 mConfigData.
mNEvents = vm[
"nEvents"].as<
unsigned int>();
323 mConfigData.
mStartEvent = vm[
"startEvent"].as<
unsigned int>();
324 mConfigData.
mBMax = vm[
"bMax"].as<
float>();
325 mConfigData.
mIsMT = vm[
"isMT"].as<
bool>();
326 mConfigData.
mOutputPrefix = vm[
"outPrefix"].as<std::string>();
327 mConfigData.
mLogSeverity = vm[
"logseverity"].as<std::string>();
328 mConfigData.
mLogVerbosity = vm[
"logverbosity"].as<std::string>();
330 mConfigData.
mConfigFile = vm[
"configFile"].as<std::string>();
333 mConfigData.
mStartSeed = vm[
"seed"].as<ULong_t>();
334 mConfigData.
mSimWorkers = vm[
"nworkers"].as<
int>();
335 if (vm.count(
"timestamp")) {
336 mConfigData.
mTimestamp = vm[
"timestamp"].as<uint64_t>();
339 mConfigData.
mTimestamp = std::chrono::time_point_cast<std::chrono::milliseconds>(std::chrono::system_clock::now()).time_since_epoch().count();
343 mConfigData.
mCCDBUrl = vm[
"CCDBUrl"].as<std::string>();
344 mConfigData.
mAsService = vm[
"asservice"].as<
bool>();
345 mConfigData.
mForwardKine = vm[
"forwardKine"].as<
bool>();
346 mConfigData.
mWriteToDisc = !vm[
"noDiscOutput"].as<
bool>();
347 if (vm.count(
"noemptyevents")) {
352 adjustFromCollContext(collcontext_simprefix.first, collcontext_simprefix.second);
361 auto& fieldstring = vm[
"field"].as<std::string>();
362 std::regex re(
"(ccdb)|([+-]?(0|[2-9]|[12][0-9]|20)U?)");
363 if (!std::regex_match(fieldstring, re)) {
364 LOG(error) <<
"Invalid field option " << fieldstring;
367 if (fieldstring ==
"ccdb") {
369 }
else if (fieldstring.find(
"U") != std::string::npos) {
372 if (fieldstring !=
"ccdb") {
373 mConfigData.
mField = std::stoi((vm[
"field"].as<std::string>()).substr(0, (vm[
"field"].as<std::string>()).rfind(
"U")));
539 bpo::options_description options(
"Allowed options");
541 options.add_options()(
542 "nEvents,n", bpo::value<unsigned int>(&
data.nEvents)->default_value(0),
"number of events")(
543 "generator,g", bpo::value<std::string>(&
data.generator)->default_value(
"boxgen"),
"Event generator to be used.")(
544 "trigger,t", bpo::value<std::string>(&
data.trigger)->default_value(
""),
"Event generator trigger to be used.")(
545 "startEvent", bpo::value<unsigned int>(&
data.startEvent)->default_value(0),
"index of first event to be used (when applicable)")(
546 "extKinFile", bpo::value<std::string>(&
data.extKinfileName)->default_value(
"Kinematics.root"),
547 "name of kinematics file for event generator from file (when applicable)")(
548 "embedIntoFile", bpo::value<std::string>(&
data.embedIntoFileName)->default_value(
""),
549 "filename containing the reference events to be used for the embedding")(
550 "bMax,b", bpo::value<float>(&
data.mBMax)->default_value(0.),
"maximum value for impact parameter sampling (when applicable)")(
551 "outPrefix,o", bpo::value<std::string>(&
data.outputPrefix)->default_value(
"o2sim"),
"prefix of output files")(
552 "outDir,d", bpo::value<std::string>(&
data.outputDir),
"directory where to put simulation output (created when non-existant")(
553 "configKeyValues", bpo::value<std::string>(&
data.keyValueTokens)->default_value(
""),
"semicolon separated key=value strings (e.g.: 'TPC.gasDensity=1;...")(
554 "configFile", bpo::value<std::string>(&
data.configFile)->default_value(
""),
"Path to an INI or JSON configuration file")(
555 "chunkSize", bpo::value<unsigned int>(&
data.primaryChunkSize)->default_value(500),
"max size of primary chunk (subevent) distributed by server")(
556 "seed", bpo::value<ULong_t>(&
data.startSeed)->default_value(0L),
"initial seed as ULong_t (default: 0 == random)")(
557 "stop", bpo::value<bool>(&
data.stop)->default_value(
false),
"control command to shut down daemon");
559 bpo::variables_map vm;
561 bpo::store(bpo::command_line_parser(bpo::split_unix(argumentstring))
566 }
catch (
const bpo::error& e) {
567 std::cerr << e.what() <<
"\n\n";
568 std::cerr <<
"Error parsing ReConfig data; Available options:\n";
569 std::cerr << options << std::endl;
static DigitizationContext * loadFromFile(std::string_view filename="")