Project
Loading...
Searching...
No Matches
SimConfig.cxx
Go to the documentation of this file.
1// Copyright 2019-2020 CERN and copyright holders of ALICE O2.
2// See https://alice-o2.web.cern.ch/copyright for details of the copyright holders.
3// All rights not expressly granted are reserved.
4//
5// This software is distributed under the terms of the GNU General Public
6// License v3 (GPL Version 3), copied verbatim in the file "COPYING".
7//
8// In applying this license CERN does not waive the privileges and immunities
9// granted to it by virtue of its status as an Intergovernmental Organization
10// or submit itself to any jurisdiction.
11
12#include <SimConfig/SimConfig.h>
16#include <boost/program_options.hpp>
17#include <iostream>
18#include <fairlogger/Logger.h>
19#include <thread>
20#include <cmath>
21#include <chrono>
22#include <regex>
23
24using namespace o2::conf;
25namespace bpo = boost::program_options;
26
27void SimConfig::initOptions(boost::program_options::options_description& options, bool isUpgrade)
28{
29 // some default args might depend on whether Run3 or Run5
30 // can be updated here:
31 std::string defaultGeomList{"ALICE2"};
32 if (isUpgrade == true) {
33 defaultGeomList = "ALICE3";
34 }
35
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.");
80}
81
82void SimConfig::determineActiveModules(std::vector<std::string> const& inputargs, std::vector<std::string> const& skippedModules, std::vector<std::string>& activeModules, bool isUpgrade)
83{
85
86 // input args is a vector of module strings as obtained from the -m,--modules options
87 // of SimConfig
88 activeModules = inputargs;
89#ifdef ENABLE_UPGRADES
90 if (activeModules[0] != "all") {
91 if (isUpgrade) {
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]);
103 }
104 }
105 }
106 if (!isUpgrade) {
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]);
117 }
118 }
119 }
120 }
121#endif
122 if (activeModules.size() == 1 && activeModules[0] == "all") {
123 activeModules.clear();
124#ifdef ENABLE_UPGRADES
125 if (isUpgrade) {
126 for (int d = DetID::First; d <= DetID::Last; ++d) {
127 if (d == DetID::TRK ||
128 d == DetID::FT3 ||
129 d == DetID::FCT ||
130 d == DetID::TF3 ||
131 d == DetID::RCH ||
132 d == DetID::ECL ||
133 d == DetID::MI3) {
134 activeModules.emplace_back(DetID::getName(d));
135 }
136 }
137 activeModules.emplace_back("A3IP");
138 activeModules.emplace_back("A3ABSO");
139 activeModules.emplace_back("A3MAG");
140 } else {
141#endif
142 // add passive components manually (make a PassiveDetID for them!)
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");
150 for (int d = DetID::First; d <= DetID::Last; ++d) {
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) {
153 activeModules.emplace_back(DetID::getName(d));
154 }
155 }
156#else
157 activeModules.emplace_back(DetID::getName(d));
158#endif
159 }
160 }
161 filterSkippedElements(activeModules, skippedModules);
162}
163
164bool SimConfig::determineActiveModulesList(const std::string& version, std::vector<std::string> const& inputargs, std::vector<std::string> const& skippedModules, std::vector<std::string>& activeModules)
165{
166 DetectorList_t modules;
167 DetectorMap_t map;
168 if (auto pos = version.find(':'); pos != std::string::npos) {
169 auto pversion = version.substr(0, pos);
170 auto ppath = version.substr(pos + 1);
171 if (!parseDetectorMapfromJSON(ppath, map)) {
172 LOGP(error, "Could not parse {}; check errors above!", ppath);
173 return false;
174 }
175 if (map.find(pversion) == map.end()) {
176 LOGP(error, "List {} is not defined in custom JSON file!", pversion);
177 printDetMap(map);
178 return false;
179 }
180 modules = map[pversion];
181 LOGP(info, "Running with version {} from custom detector list '{}'", pversion, ppath);
182 } else {
183 // Otherwise check 'official' versions which provided in config
184 auto o2env = std::getenv("O2_ROOT");
185 if (!o2env) {
186 LOGP(error, "O2_ROOT environment not defined");
187 return false;
188 }
189 const std::string rootpath(fmt::format("{}/share/config/o2simdefaultdetectorlist.json", o2env));
190 if (!parseDetectorMapfromJSON(rootpath, map)) {
191 LOGP(error, "Could not parse {} -> check errors above!", rootpath);
192 return false;
193 }
194 if (map.find(version) == map.end()) {
195 LOGP(error, "List {} is not defined in 'official' JSON file!", version);
196 printDetMap(map);
197 return false;
198 }
199 modules = map[version];
200 LOGP(info, "Running with official detector version '{}'", version);
201 }
202 // check if specified modules are in list
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);
208 }
209 }
210 if (!diff.empty()) {
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);
214 }
215 printDetMap(map, version);
216 return false;
217 }
218 }
219 // Insert into active modules if module is built buy -m or insert all if default for -m is used ("all")
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);
223}
224
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)
226{
228
229 readoutDetectors.clear();
230
231 auto isDet = [](std::string const& s) {
232 return DetID::nameToID(s.c_str()) >= DetID::First;
233 };
234
235 if (enableReadout.empty()) {
236 // if no readout explicitly given, use all detectors from active modules
237 for (auto& am : activeModules) {
238 if (!isDet(am)) {
239 // either we found a passive module or one with disabled readout ==> skip
240 continue;
241 }
242 readoutDetectors.emplace_back(am);
243 }
244 } else {
245 for (auto& er : enableReadout) {
246 if (!isDet(er)) {
247 // either we found a passive module or one with disabled readout ==> skip
248 LOG(fatal) << "Enabled readout for " << er << " which is not a detector.";
249 }
250 if (std::find(activeModules.begin(), activeModules.end(), er) == activeModules.end()) {
251 // add to active modules if not yet there
252 LOG(fatal) << "Module " << er << " not constructed and cannot be used for readout (make sure it is contained in -m option).";
253 }
254 readoutDetectors.emplace_back(er);
255 }
256 }
257 for (auto& dr : disableReadout) {
258 if (!isDet(dr)) {
259 // either we found a passive module or one with disabled readout ==> skip
260 LOG(fatal) << "Disabled readout for " << dr << " which is not a detector.";
261 }
262 if (std::find(activeModules.begin(), activeModules.end(), dr) == activeModules.end()) {
263 // add to active modules if not yet there
264 LOG(fatal) << "Module " << dr << " not constructed, makes no sense to disable its readout (make sure it is contained in -m option).";
265 }
266 auto iter = std::find(readoutDetectors.begin(), readoutDetectors.end(), dr);
267 if (iter != readoutDetectors.end()) {
268 readoutDetectors.erase(iter);
269 }
270 }
271}
272
273std::pair<std::string, std::string> SimConfig::getCollContextFilenameAndEventPrefix() const
274{
275 // we decompose the argument to fetch
276 // (a) collision contextfilename
277 // (b) sim prefix to use from the context
278 auto pos = mConfigData.mFromCollisionContext.find(':');
279 std::string collcontextfile{mConfigData.mFromCollisionContext};
280 std::string simprefix{mConfigData.mOutputPrefix};
281 if (pos != std::string::npos) {
282 collcontextfile = mConfigData.mFromCollisionContext.substr(0, pos);
283 simprefix = mConfigData.mFromCollisionContext.substr(pos + 1);
284 }
285 return std::make_pair(collcontextfile, simprefix);
286}
287
288bool SimConfig::resetFromParsedMap(boost::program_options::variables_map const& vm)
289{
291 mConfigData.mMCEngine = vm["mcEngine"].as<std::string>();
292 mConfigData.mNoGeant = vm["noGeant"].as<bool>();
293
294 // Reset modules and detectors as they are anyway re-parsed
295 mConfigData.mReadoutDetectors.clear();
296 mConfigData.mActiveModules.clear();
297
298 // Get final set of active Modules
299 if (!determineActiveModulesList(vm["detectorList"].as<std::string>(), vm["modules"].as<std::vector<std::string>>(), vm["skipModules"].as<std::vector<std::string>>(), mConfigData.mActiveModules)) {
300 return false;
301 }
302
303 if (mConfigData.mNoGeant) {
304 // CAVE is all that's needed (and that will be built either way), so clear all modules and set the O2TrivialMCEngine
305 mConfigData.mActiveModules.clear();
306 // force usage of O2TrivialMCEngine, no overhead from actual transport engine initialisation
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";
310 return false;
311 }
312
313 const auto& activeModules = mConfigData.mActiveModules;
314
315 // get final set of detectors which are readout
316 determineReadoutDetectors(activeModules, vm["readoutDetectors"].as<std::vector<std::string>>(), vm["skipReadoutDetectors"].as<std::vector<std::string>>(), mConfigData.mReadoutDetectors);
317
318 mConfigData.mGenerator = vm["generator"].as<std::string>();
319 mConfigData.mTrigger = vm["trigger"].as<std::string>();
320 mConfigData.mNEvents = vm["nEvents"].as<unsigned int>();
321 mConfigData.mExtKinFileName = vm["extKinFile"].as<std::string>();
322 mConfigData.mEmbedIntoFileName = vm["embedIntoFile"].as<std::string>();
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>();
329 mConfigData.mKeyValueTokens = vm["configKeyValues"].as<std::string>();
330 mConfigData.mConfigFile = vm["configFile"].as<std::string>();
331 mConfigData.mPrimaryChunkSize = vm["chunkSize"].as<unsigned int>();
332 mConfigData.mInternalChunkSize = vm["chunkSizeI"].as<int>();
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>();
338 } else {
339 mConfigData.mTimestamp = std::chrono::time_point_cast<std::chrono::milliseconds>(std::chrono::system_clock::now()).time_since_epoch().count();
341 }
342 mConfigData.mRunNumber = vm["run"].as<int>();
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")) {
348 mConfigData.mFilterNoHitEvents = true;
349 }
350 mConfigData.mFromCollisionContext = vm["fromCollContext"].as<std::string>();
351 auto collcontext_simprefix = getCollContextFilenameAndEventPrefix();
352 adjustFromCollContext(collcontext_simprefix.first, collcontext_simprefix.second);
353
354 // analyse vertex options
355 if (!parseVertexModeString(vm["vertexMode"].as<std::string>(), mConfigData.mVertexMode)) {
356 return false;
357 }
358
359 // analyse field options
360 // either: "ccdb" or +-2[U],+-5[U] and 0[U]; +-<intKGaus>U
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;
365 return false;
366 }
367 if (fieldstring == "ccdb") {
368 mConfigData.mFieldMode = SimFieldMode::kCCDB;
369 } else if (fieldstring.find("U") != std::string::npos) {
371 }
372 if (fieldstring != "ccdb") {
373 mConfigData.mField = std::stoi((vm["field"].as<std::string>()).substr(0, (vm["field"].as<std::string>()).rfind("U")));
374 }
375 if (!parseFieldString(fieldstring, mConfigData.mField, mConfigData.mFieldMode)) {
376 return false;
377 }
378
379 return true;
380}
381
382bool SimConfig::parseVertexModeString(std::string const& vertexstring, VertexMode& mode)
383{
384 // vertexstring must be either kNoVertex, kDiamondParam, kCCDB
385 if (vertexstring == "kNoVertex") {
387 return true;
388 } else if (vertexstring == "kDiamondParam") {
390 return true;
391 } else if (vertexstring == "kCCDB") {
393 return true;
394 } else if (vertexstring == "kCollContext") {
396 return true;
397 }
398 LOG(error) << "Vertex mode must be one of kNoVertex, kDiamondParam, kCCDB, kCollContext";
399 return false;
400}
401
402bool SimConfig::parseFieldString(std::string const& fieldstring, int& fieldvalue, SimFieldMode& mode)
403{
404 // analyse field options
405 // either: "ccdb" or +-2[U],+-5[U] and 0[U]; +-<intKGaus>U
406 std::regex re("(ccdb)|([+-]?(0|[2-9]|[12][0-9]|20)U?)");
407 if (!std::regex_match(fieldstring, re)) {
408 LOG(error) << "Invalid field option " << fieldstring;
409 return false;
410 }
411 if (fieldstring == "ccdb") {
413 } else if (fieldstring.find("U") != std::string::npos) {
415 }
416 if (fieldstring != "ccdb") {
417 fieldvalue = std::stoi(fieldstring.substr(0, fieldstring.rfind("U")));
418 }
419 return true;
420}
421
422bool SimConfig::filterSkippedElements(std::vector<std::string>& elements, std::vector<std::string> const& skipped)
423{
424 for (auto& s : skipped) {
425 if (s.empty()) { // nothing to skip here
426 continue;
427 }
428 auto iter = std::find(elements.begin(), elements.end(), s);
429 if (iter != elements.end()) {
430 // take it out
431 elements.erase(iter);
432 } else {
433 LOGP(error, "Skipped modules specified that are not present in built modules!");
434 LOGP(error, "Built modules:");
435 for (int j{0}; const auto& m : elements) {
436 LOGP(error, " + {: <2}. {}", j++, m);
437 }
438 std::vector<std::string> diff;
439 for (const auto& skip : skipped) {
440 if (std::find(elements.begin(), elements.end(), skip) == std::end(elements)) {
441 diff.emplace_back(skip);
442 }
443 }
444 LOGP(error, "Specified skipped modules not present in built modules:");
445 for (int j{0}; const auto& m : diff) {
446 LOGP(error, " - {: <2}. {}", j++, m);
447 }
448 return false;
449 }
450 }
451 return true;
452}
453
454void SimConfig::adjustFromCollContext(std::string const& collcontextfile, std::string const& prefix)
455{
456 // When we use pregenerated collision contexts, some options
457 // need to be auto-adjusted. Do so and inform about this in the logs.
458 if (collcontextfile == "") {
459 return;
460 }
461
462 auto context = o2::steer::DigitizationContext::loadFromFile(collcontextfile);
463 if (context) {
464 // find the events belonging to a source that corresponds to a sim prefix
465 LOG(info) << "Looking up simprefixes " << prefix;
466 int sourceid = context->findSimPrefix(prefix);
467 if (sourceid == -1) {
468 LOG(error) << "Could not find collisions with sim prefix " << prefix << " in the collision context. The collision context specifies the following prefixes:";
469 for (auto& sp : context->getSimPrefixes()) {
470 LOG(info) << sp;
471 }
472 LOG(fatal) << "Aborting due to prefix error";
473 } else {
474 auto collisionmap = context->getCollisionIndicesForSource(sourceid);
475 LOG(info) << "Found " << collisionmap.size() << " events in the collisioncontext for prefix " << prefix;
476
477 // check if collisionmap is dense (otherwise it will get screwed up with order/indexing in ROOT output)
478 bool good = true;
479 for (auto index = 0; index < collisionmap.size(); ++index) {
480 if (collisionmap.find(index) == collisionmap.end()) {
481 good = false;
482 }
483 }
484 if (!good) {
485 LOG(fatal) << "events in collisioncontext are non-compact ";
486 }
487
488 // do some adjustments based on the number of events to be simulated
489 if (mConfigData.mNEvents == 0 || mConfigData.mNEvents == collisionmap.size()) {
490 // we take what is specified in the context
491 mConfigData.mNEvents = collisionmap.size();
492 } else {
493 LOG(warning) << "The number of events on the command line " << mConfigData.mNEvents << " and in the collision context differ. We take the one from collision context " << collisionmap.size();
494 mConfigData.mNEvents = collisionmap.size();
495 }
496 LOG(info) << "Setting number of events to simulate to " << mConfigData.mNEvents;
497 }
498 } else {
499 LOG(fatal) << "Could not open collision context file " << collcontextfile;
500 }
501}
502
503bool SimConfig::resetFromArguments(int argc, char* argv[])
504{
505 // Arguments parsing
506 bpo::variables_map vm;
507 bpo::options_description desc("Allowed options");
508 desc.add_options()("help,h", "Produce help message.");
509 initOptions(desc, mConfigData.mIsUpgrade);
510
511 try {
512 bpo::store(parse_command_line(argc, argv, desc), vm);
513
514 // help
515 if (vm.count("help")) {
516 std::cout << desc << std::endl;
517 return false;
518 }
519
520 bpo::notify(vm);
521 } catch (const bpo::error& e) {
522 std::cerr << e.what() << "\n\n";
523 std::cerr << "Error parsing command line arguments; Available options:\n";
524
525 std::cerr << desc << std::endl;
526 return false;
527 }
528
529 return resetFromParsedMap(vm);
530}
531
532namespace o2::conf
533{
534// returns a reconfig struct given a configuration string (boost program options format)
535bool parseSimReconfigFromString(std::string const& argumentstring, SimReconfigData& data)
536{
537 namespace bpo = boost::program_options;
538
539 bpo::options_description options("Allowed options");
540
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");
558
559 bpo::variables_map vm;
560 try {
561 bpo::store(bpo::command_line_parser(bpo::split_unix(argumentstring))
562 .options(options)
563 .run(),
564 vm);
565 bpo::notify(vm);
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;
570 return false;
571 }
572 return true;
573}
574
575} // namespace o2::conf
576
int32_t i
ClassImp(IdPath)
uint16_t pos
Definition RawData.h:3
uint32_t j
Definition RawData.h:0
uint32_t version
Definition RawData.h:8
bool resetFromParsedMap(boost::program_options::variables_map const &)
bool resetFromArguments(int argc, char *argv[])
static void determineActiveModules(std::vector< std::string > const &input, std::vector< std::string > const &skipped, std::vector< std::string > &active, bool isUpgrade=false)
Definition SimConfig.cxx:82
static bool parseVertexModeString(std::string const &vertexstring, o2::conf::VertexMode &mode)
static bool parseFieldString(std::string const &fieldstring, int &fieldvalue, o2::conf::SimFieldMode &mode)
static void determineReadoutDetectors(std::vector< std::string > const &active, std::vector< std::string > const &enabledRO, std::vector< std::string > const &skippedRO, std::vector< std::string > &finalRO)
static bool determineActiveModulesList(const std::string &version, std::vector< std::string > const &input, std::vector< std::string > const &skipped, std::vector< std::string > &active)
std::pair< std::string, std::string > getCollContextFilenameAndEventPrefix() const
static void initOptions(boost::program_options::options_description &, bool isUpgrade=false)
Definition SimConfig.cxx:27
Static class with identifiers, bitmasks and names for ALICE detectors.
Definition DetID.h:58
static constexpr const char * getName(ID id)
names of defined detectors
Definition DetID.h:145
static constexpr ID First
Definition DetID.h:94
static constexpr int nameToID(char const *name, int id=First)
Definition DetID.h:154
static constexpr ID Last
if extra detectors added, update this !!!
Definition DetID.h:92
static DigitizationContext * loadFromFile(std::string_view filename="")
const GLfloat * m
Definition glcorearb.h:4066
GLenum mode
Definition glcorearb.h:266
GLuint index
Definition glcorearb.h:781
GLboolean * data
Definition glcorearb.h:298
std::vector< std::string > DetectorList_t
bool parseSimReconfigFromString(std::string const &argumentstring, SimReconfigData &config)
void printDetMap(const DetectorMap_t &map, const std::string &list="")
std::unordered_map< std::string, DetectorList_t > DetectorMap_t
bool parseDetectorMapfromJSON(const std::string &path, DetectorMap_t &map)
std::string mFromCollisionContext
Definition SimConfig.h:86
std::vector< std::string > mActiveModules
Definition SimConfig.h:52
std::string mOutputPrefix
Definition SimConfig.h:63
std::string mExtKinFileName
Definition SimConfig.h:58
unsigned int mPrimaryChunkSize
Definition SimConfig.h:72
std::vector< std::string > mReadoutDetectors
Definition SimConfig.h:53
std::string mKeyValueTokens
Definition SimConfig.h:66
TimeStampMode mTimestampMode
Definition SimConfig.h:79
unsigned int mNEvents
Definition SimConfig.h:57
std::string mGenerator
Definition SimConfig.h:55
SimFieldMode mFieldMode
Definition SimConfig.h:82
std::string mLogSeverity
Definition SimConfig.h:64
std::string mConfigFile
Definition SimConfig.h:69
std::string mEmbedIntoFileName
Definition SimConfig.h:59
std::string mLogVerbosity
Definition SimConfig.h:65
unsigned int mStartEvent
Definition SimConfig.h:60
std::string mMCEngine
Definition SimConfig.h:54
TODO: Make this a base class of SimConfigData?
Definition SimConfig.h:203
LOG(info)<< "Compressed in "<< sw.CpuTime()<< " s"