Project
Loading...
Searching...
No Matches
GeneratorHybrid.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
13#include <fairlogger/Logger.h>
14#include <algorithm>
15#include <tbb/concurrent_queue.h>
16#include <tbb/task_arena.h>
17#include <tbb/parallel_for.h>
19#include <filesystem>
20
21namespace o2
22{
23namespace eventgen
24{
25
26GeneratorHybrid::GeneratorHybrid(const std::string& inputgens)
27{
28 // This generator has trivial unit conversions
29 setTimeUnit(1.);
32 setEnergyUnit(1.);
33
34 if (!parseJSON(inputgens)) {
35 LOG(fatal) << "Failed to parse JSON configuration from input generators";
36 exit(1);
37 }
38 mRandomize = GeneratorHybridParam::Instance().randomize;
39 if (mConfigs.size() != mInputGens.size()) {
40 LOG(fatal) << "Number of configurations does not match the number of generators";
41 exit(1);
42 }
43 if (mConfigs.size() == 0) {
44 for (auto gen : mInputGens) {
45 mConfigs.push_back("");
46 }
47 }
48 int index = 0;
49 if (!(mRandomize || mGenerationMode == GenMode::kParallel)) {
50 if (mCocktailMode) {
51 if (mGroups.size() != mFractions.size()) {
52 LOG(fatal) << "Number of groups does not match the number of fractions";
53 return;
54 }
55 } else {
56 if (mFractions.size() != mInputGens.size()) {
57 LOG(fatal) << "Number of fractions does not match the number of generators";
58 return;
59 }
60 }
61 // Check if all elements of mFractions are 0
62 if (std::all_of(mFractions.begin(), mFractions.end(), [](int i) { return i == 0; })) {
63 LOG(fatal) << "All fractions provided are 0, no simulation will be performed";
64 return;
65 }
66 }
67 for (auto gen : mInputGens) {
68 // Search if the generator name is inside generatorNames (which is a vector of strings)
69 LOG(info) << "Checking if generator " << gen << " is in the list of available generators \n";
70 if (std::find(generatorNames.begin(), generatorNames.end(), gen) != generatorNames.end()) {
71 LOG(info) << "Found generator " << gen << " in the list of available generators \n";
72 if (gen.compare("boxgen") == 0) {
73 if (mConfigs[index].compare("") == 0) {
74 gens.push_back(std::make_shared<o2::eventgen::BoxGenerator>());
75 } else {
76 // Get the index of boxgen configuration
77 int confBoxIndex = std::stoi(mConfigs[index].substr(7));
78 gens.push_back(std::make_shared<o2::eventgen::BoxGenerator>(*mBoxGenConfigs[confBoxIndex]));
79 }
80 mGens.push_back(gen);
81 } else if (gen.compare(0, 7, "pythia8") == 0) {
82 // Check if mConfigs[index] contains pythia8_ and a number
83 if (mConfigs[index].compare("") == 0) {
84 auto pars = Pythia8GenConfig();
85 gens.push_back(std::make_shared<o2::eventgen::GeneratorPythia8>(pars));
86 } else {
87 // Get the index of pythia8 configuration
88 int confPythia8Index = std::stoi(mConfigs[index].substr(8));
89 gens.push_back(std::make_shared<o2::eventgen::GeneratorPythia8>(*mPythia8GenConfigs[confPythia8Index]));
90 }
91 mConfsPythia8.push_back(mConfigs[index]);
92 mGens.push_back(gen);
93 } else if (gen.compare("extkinO2") == 0) {
94 int confO2KineIndex = std::stoi(mConfigs[index].substr(9));
95 gens.push_back(std::make_shared<o2::eventgen::GeneratorFromO2Kine>(*mO2KineGenConfigs[confO2KineIndex]));
96 mGens.push_back(gen);
97 } else if (gen.compare("evtpool") == 0) {
98 int confEvtPoolIndex = std::stoi(mConfigs[index].substr(8));
99 gens.push_back(std::make_shared<o2::eventgen::GeneratorFromEventPool>(mEventPoolConfigs[confEvtPoolIndex]));
100 mGens.push_back(gen);
101 } else if (gen.compare("external") == 0) {
102 int confextIndex = std::stoi(mConfigs[index].substr(9));
103 // we need analyse the ini file to update the config key param
104 if (mExternalGenConfigs[confextIndex]->iniFile.size() > 0) {
105 LOG(info) << "Setting up external gen using the given INI file";
106
107 // this means that we go via the ConfigurableParam system ---> in order not to interfere with other
108 // generators we use an approach with backup and restore of the system
109
110 // we write the current state to a file
111 // create a tmp file name
112 std::string tmp_config_file = "configkey_tmp_backup_" + std::to_string(getpid()) + std::string(".ini");
114
115 auto expandedFileName = o2::utils::expandShellVarsInFileName(mExternalGenConfigs[confextIndex]->iniFile);
117 // toDo: check that this INI file makes sense
118
120 LOG(info) << "Setting up external generator with following parameters";
121 LOG(info) << params;
122 auto extgen_filename = params.fileName;
123 auto extgen_func = params.funcName;
124 auto extgen = std::shared_ptr<o2::eventgen::Generator>(o2::conf::GetFromMacro<o2::eventgen::Generator*>(extgen_filename, extgen_func, "FairGenerator*", "extgen"));
125 if (!extgen) {
126 LOG(fatal) << "Failed to retrieve \'extgen\': problem with configuration ";
127 }
128 // restore old state
130 // delete tmp file
131 std::filesystem::remove(tmp_config_file);
132
133 gens.push_back(std::move(extgen));
134 mGens.push_back(gen);
135 } else {
136 LOG(info) << "Setting up external gen using the given fileName and funcName";
137 // we need to restore the config key param system to what is was before
138 auto& extgen_filename = mExternalGenConfigs[confextIndex]->fileName;
139 auto& extgen_func = mExternalGenConfigs[confextIndex]->funcName;
140 auto extGen = std::shared_ptr<o2::eventgen::Generator>(o2::conf::GetFromMacro<o2::eventgen::Generator*>(extgen_filename, extgen_func, "FairGenerator*", "extgen"));
141 if (!extGen) {
142 LOG(fatal) << "Failed to load external generator from " << extgen_filename << " with function " << extgen_func;
143 }
144 gens.push_back(std::move(extGen));
145 mGens.push_back(gen);
146 }
147 } else if (gen.compare("hepmc") == 0) {
148 int confHepMCIndex = std::stoi(mConfigs[index].substr(6));
149 gens.push_back(std::make_shared<o2::eventgen::GeneratorHepMC>());
150 auto& globalConfig = o2::conf::SimConfig::Instance();
151 dynamic_cast<o2::eventgen::GeneratorHepMC*>(gens.back().get())->setup(*mFileOrCmdGenConfigs[confHepMCIndex], *mHepMCGenConfigs[confHepMCIndex], globalConfig);
152 mGens.push_back(gen);
153 }
154 } else {
155 LOG(fatal) << "Generator " << gen << " not found in the list of available generators \n";
156 exit(1);
157 }
158 index++;
159 }
160}
161
163{
164 LOG(info) << "Destructor of generator hybrid called";
165 mStopFlag = true;
166}
167
169{
170 // init all sub-gens
171 int count = 0;
172 for (auto& gen : mGens) {
173 if (gen == "pythia8pp") {
174 auto config = std::string(std::getenv("O2_ROOT")) + "/share/Generators/egconfig/pythia8_inel.cfg";
175 LOG(info) << "Setting \'Pythia8\' base configuration: " << config << std::endl;
176 dynamic_cast<o2::eventgen::GeneratorPythia8*>(gens[count].get())->setConfig(config);
177 } else if (gen == "pythia8hf") {
178 auto config = std::string(std::getenv("O2_ROOT")) + "/share/Generators/egconfig/pythia8_hf.cfg";
179 LOG(info) << "Setting \'Pythia8\' base configuration: " << config << std::endl;
180 dynamic_cast<o2::eventgen::GeneratorPythia8*>(gens[count].get())->setConfig(config);
181 } else if (gen == "pythia8hi") {
182 auto config = std::string(std::getenv("O2_ROOT")) + "/share/Generators/egconfig/pythia8_hi.cfg";
183 LOG(info) << "Setting \'Pythia8\' base configuration: " << config << std::endl;
184 dynamic_cast<o2::eventgen::GeneratorPythia8*>(gens[count].get())->setConfig(config);
185 } else if (gen == "pythia8powheg") {
186 auto config = std::string(std::getenv("O2_ROOT")) + "/share/Generators/egconfig/pythia8_powheg.cfg";
187 LOG(info) << "Setting \'Pythia8\' base configuration: " << config << std::endl;
188 dynamic_cast<o2::eventgen::GeneratorPythia8*>(gens[count].get())->setConfig(config);
189 }
190 gens[count]->Init(); // TODO: move this to multi-threaded
192 if (mTriggerModes[count] != o2::eventgen::Generator::kTriggerOFF) {
193 gens[count]->setTriggerMode(mTriggerModes[count]);
194 LOG(info) << "Setting Trigger mode of generator " << gen << " to: " << mTriggerModes[count];
195 o2::eventgen::Trigger trigger = nullptr;
196 o2::eventgen::DeepTrigger deeptrigger = nullptr;
197 for (int trg = 0; trg < mTriggerMacros[count].size(); trg++) {
198 if (mTriggerMacros[count][trg].empty() || mTriggerFuncs[count][trg].empty()) {
199 continue;
200 }
201 std::string expandedMacro = o2::utils::expandShellVarsInFileName(mTriggerMacros[count][trg]);
202 LOG(info) << "Setting trigger " << trg << " of generator " << gen << " with following parameters";
203 LOG(info) << "Macro filename: " << expandedMacro;
204 LOG(info) << "Function name: " << mTriggerFuncs[count][trg];
205 trigger = o2::conf::GetFromMacro<o2::eventgen::Trigger>(expandedMacro, mTriggerFuncs[count][trg], "o2::eventgen::Trigger", "trigger");
206 if (!trigger) {
207 LOG(info) << "Trying to retrieve a \'o2::eventgen::DeepTrigger\' type";
208 deeptrigger = o2::conf::GetFromMacro<o2::eventgen::DeepTrigger>(expandedMacro, mTriggerFuncs[count][trg], "o2::eventgen::DeepTrigger", "deeptrigger");
209 }
210 if (!trigger && !deeptrigger) {
211 LOG(warn) << "Failed to retrieve \'external trigger\': problem with configuration";
212 LOG(warn) << "Trigger " << trg << " of generator " << gen << " will not be included";
213 continue;
214 } else {
215 LOG(info) << "Trigger " << trg << " of generator " << gen << " successfully set";
216 }
217 if (trigger) {
218 gens[count]->addTrigger(trigger);
219 } else {
220 gens[count]->addDeepTrigger(deeptrigger);
221 }
222 }
223 }
224 count++;
225 }
226 if (mRandomize) {
227 if (std::all_of(mFractions.begin(), mFractions.end(), [](int i) { return i == 1; })) {
228 LOG(info) << "Full randomisation of generators order";
229 } else {
230 LOG(info) << "Randomisation based on fractions";
231 int allfracs = 0;
232 for (auto& f : mFractions) {
233 allfracs += f;
234 }
235 // Assign new rng fractions
236 float sum = 0;
237 float chance = 0;
238 for (int k = 0; k < mFractions.size(); k++) {
239 if (mFractions[k] == 0) {
240 // Generator will not be used if fraction is 0
241 mRngFractions.push_back(-1);
242 LOG(info) << "Generator " << mGens[k] << " will not be used";
243 } else {
244 chance = static_cast<float>(mFractions[k]) / allfracs;
245 sum += chance;
246 mRngFractions.push_back(sum);
247 LOG(info) << "Generator " << (mConfigs[k] == "" ? mGens[k] : mConfigs[k]) << " has a " << chance * 100 << "% chance of being used";
248 }
249 }
250 }
251 } else {
252 LOG(info) << "Generators will be used in sequence, following provided fractions";
253 }
254
255 mGenIsInitialized.resize(gens.size(), false);
256 if (mGenerationMode == GenMode::kParallel) {
257 // in parallel mode we just use one queue --> collaboration
258 mResultQueue.resize(1);
259 } else {
260 // in sequential mode we have one queue per generator
261 mResultQueue.resize(gens.size());
262 }
263 // Create a task arena with a specified number of threads
264 mTaskArena.initialize(GeneratorHybridParam::Instance().num_workers);
265
266 // the process task function actually calls event generation
267 // when it is done it notifies the outside world by pushing it's index into an appropriate queue
268 // This should be a lambda, which can be given at TaskPool creation time
269 auto process_generator_task = [this](std::vector<std::shared_ptr<o2::eventgen::Generator>> const& generatorvec, int task) {
270 LOG(debug) << "Starting eventgen for task " << task;
271 auto& generator = generatorvec[task];
272 if (!mStopFlag) {
273 // TODO: activate this once we are use Init is threadsafe
274 // if (!mGenIsInitialized[task]) {
275 // if(!generator->Init()) {
276 // LOG(error) << "failed to init generator " << task;
277 // }
278 // mGenIsInitialized[task] = true;
279 // }
280 }
281 bool isTriggered = false;
282 while (!isTriggered) {
283 generator->clearParticles();
284 generator->generateEvent();
285 generator->importParticles();
286 isTriggered = generator->triggerEvent();
287 }
288 LOG(debug) << "eventgen finished for task " << task;
289 if (!mStopFlag) {
290 if (mGenerationMode == GenMode::kParallel) {
291 mResultQueue[0].push(task);
292 } else {
293 mResultQueue[task].push(task);
294 }
295 }
296 };
297
298 // fundamental tbb thread-worker function
299 auto worker_function = [this, process_generator_task]() {
300 // we increase the reference count in the generator pointers
301 // by making a copy of the vector. In this way we ensure that the lifetime
302 // of the generators is no shorter than the lifetime of the thread for this worker function
303 auto generators_copy = gens;
304
305 while (!mStopFlag) {
306 int task;
307 if (mInputTaskQueue.try_pop(task)) {
308 process_generator_task(generators_copy, task); // Process the task
309 } else {
310 std::this_thread::sleep_for(std::chrono::milliseconds(10)); // Wait if no task
311 }
312 }
313 };
314
315 // let the TBB task system run in it's own thread
316 mTBBTaskPoolRunner = std::thread([this, worker_function]() { mTaskArena.execute([&]() { tbb::parallel_for(0, mTaskArena.max_concurrency(), [&](int) { worker_function(); }); }); });
317 mTBBTaskPoolRunner.detach(); // detaching because we don't like to wait on the thread to finish
318 // some of the generators might still be generating when we are done
319
320 // let's also push initial generation tasks for each event generator
321 for (size_t genindex = 0; genindex < gens.size(); ++genindex) {
322 mInputTaskQueue.push(genindex);
323 mTasksStarted++;
324 }
325 mIsInitialized = true;
326 return Generator::Init();
327}
328
330{
331 if (!mIsInitialized) {
332 Init();
333 }
334 if (mGenerationMode == GenMode::kParallel) {
335 mIndex = -1; // this means any index is welcome
336 notifySubGenerator(0); // we shouldn't distinguish the sub-gen ids
337 } else {
338 // Order randomisation or sequence of generators
339 // following provided fractions, if not generators are used in proper sequence
340 // Order randomisation or sequence of generators
341 // following provided fractions. If not available generators will be used sequentially
342 if (mRandomize) {
343 if (mRngFractions.size() != 0) {
344 // Generate number between 0 and 1
345 float rnum = gRandom->Rndm();
346 // Find generator index
347 for (int k = 0; k < mRngFractions.size(); k++) {
348 if (rnum <= mRngFractions[k]) {
349 mIndex = k;
350 break;
351 }
352 }
353 } else {
354 mIndex = gRandom->Integer(mFractions.size());
355 }
356 } else {
357 while (mFractions[mCurrentFraction] == 0 || mseqCounter == mFractions[mCurrentFraction]) {
358 if (mFractions[mCurrentFraction] != 0) {
359 mseqCounter = 0;
360 }
361 mCurrentFraction = (mCurrentFraction + 1) % mFractions.size();
362 }
363 mIndex = mCurrentFraction;
364 }
365 notifySubGenerator(mIndex);
366 }
367 return true;
368}
369
371{
372 int genIndex = -1;
373 std::vector<int> subGenIndex = {};
374 if (mIndex == -1) {
375 // this means parallel mode ---> we have a common queue
376 mResultQueue[0].pop(genIndex);
377 } else {
378 // need to pop from a particular queue
379 if (!mCocktailMode) {
380 mResultQueue[mIndex].pop(genIndex);
381 } else {
382 // in cocktail mode we need to pop from the group queue
383 subGenIndex.resize(mGroups[mIndex].size());
384 for (size_t pos = 0; pos < mGroups[mIndex].size(); ++pos) {
385 int subIndex = mGroups[mIndex][pos];
386 LOG(info) << "Getting generator " << mGens[subIndex] << " from cocktail group " << mIndex;
387 mResultQueue[subIndex].pop(subGenIndex[pos]);
388 }
389 }
390 }
391
392 auto unit_transformer = [](auto& p, auto pos_unit, auto time_unit, auto en_unit, auto mom_unit) {
393 p.SetMomentum(p.Px() * mom_unit, p.Py() * mom_unit, p.Pz() * mom_unit, p.Energy() * en_unit);
394 p.SetProductionVertex(p.Vx() * pos_unit, p.Vy() * pos_unit, p.Vz() * pos_unit, p.T() * time_unit);
395 };
396
397 auto index_transformer = [](auto& p, int offset) {
398 for (int i = 0; i < 2; ++i) {
399 if (p.GetMother(i) != -1) {
400 const auto newindex = p.GetMother(i) + offset;
401 p.SetMother(i, newindex);
402 }
403 }
404 if (p.GetNDaughters() > 0) {
405 for (int i = 0; i < 2; ++i) {
406 const auto newindex = p.GetDaughter(i) + offset;
407 p.SetDaughter(i, newindex);
408 }
409 }
410 };
411
412 // Clear particles and event header
413 mParticles.clear();
414 mMCEventHeader.clearInfo();
415 if (mCocktailMode) {
416 // in cocktail mode we need to merge the particles from the different generators
417 for (auto subIndex : subGenIndex) {
418 LOG(info) << "Importing particles for task " << subIndex;
419 auto subParticles = gens[subIndex]->getParticles();
420
421 auto time_unit = gens[subIndex]->getTimeUnit();
422 auto pos_unit = gens[subIndex]->getPositionUnit();
423 auto mom_unit = gens[subIndex]->getMomentumUnit();
424 auto energy_unit = gens[subIndex]->getEnergyUnit();
425
426 // The particles carry mother and daughter indices, which are relative
427 // to the sub-generator. We need to adjust these indices to reflect that particles
428 // are now embedded into a cocktail.
429 auto offset = mParticles.size();
430 for (auto& p : subParticles) {
431 // apply the mother-daugher index transformation
432 index_transformer(p, offset);
433 // apply unit transformation of sub-generator
434 unit_transformer(p, pos_unit, time_unit, energy_unit, mom_unit);
435 }
436
437 mParticles.insert(mParticles.end(), subParticles.begin(), subParticles.end());
438 // fetch the event Header information from the underlying generator
439 gens[subIndex]->updateHeader(&mMCEventHeader);
440 mInputTaskQueue.push(subIndex);
441 mTasksStarted++;
442 }
443 } else {
444 LOG(info) << "Importing particles for task " << genIndex;
445 // at this moment the mIndex-th generator is ready to be used
446 mParticles = gens[genIndex]->getParticles();
447
448 auto time_unit = gens[genIndex]->getTimeUnit();
449 auto pos_unit = gens[genIndex]->getPositionUnit();
450 auto mom_unit = gens[genIndex]->getMomentumUnit();
451 auto energy_unit = gens[genIndex]->getEnergyUnit();
452
453 // transform units to units of the hybrid generator
454 for (auto& p : mParticles) {
455 // apply unit transformation
456 unit_transformer(p, pos_unit, time_unit, energy_unit, mom_unit);
457 }
458
459 // fetch the event Header information from the underlying generator
460 gens[genIndex]->updateHeader(&mMCEventHeader);
461 mInputTaskQueue.push(genIndex);
462 mTasksStarted++;
463 }
464
465 mseqCounter++;
466 mEventCounter++;
467 if (mEventCounter == getTotalNEvents()) {
468 LOG(info) << "HybridGen: Stopping TBB task pool";
469 mStopFlag = true;
470 }
471
472 return true;
473}
474
476{
477 if (eventHeader) {
478 // we forward the original header information if any
479 eventHeader->copyInfoFrom(mMCEventHeader);
480
481 // put additional information about
482 eventHeader->putInfo<std::string>("forwarding-generator", "HybridGen");
483 }
484}
485
486template <typename T>
488{
489 rapidjson::StringBuffer buffer;
490 rapidjson::Writer<rapidjson::StringBuffer> writer(buffer);
491 value.Accept(writer);
492 return buffer.GetString();
493}
494
496{
497 std::string name = gen["name"].GetString();
498 mInputGens.push_back(name);
499 if (gen.HasMember("config")) {
500 if (name == "boxgen") {
501 const auto& boxconf = gen["config"];
502 auto boxConfig = TBufferJSON::FromJSON<o2::eventgen::BoxGenConfig>(jsonValueToString(boxconf).c_str());
503 mBoxGenConfigs.push_back(std::move(boxConfig));
504 mConfigs.push_back("boxgen_" + std::to_string(mBoxGenConfigs.size() - 1));
505 } else if (name == "pythia8") {
506 const auto& pythia8conf = gen["config"];
507 auto pythia8Config = TBufferJSON::FromJSON<o2::eventgen::Pythia8GenConfig>(jsonValueToString(pythia8conf).c_str());
508 mPythia8GenConfigs.push_back(std::move(pythia8Config));
509 mConfigs.push_back("pythia8_" + std::to_string(mPythia8GenConfigs.size() - 1));
510 } else if (name == "extkinO2") {
511 const auto& o2kineconf = gen["config"];
512 auto o2kineConfig = TBufferJSON::FromJSON<o2::eventgen::O2KineGenConfig>(jsonValueToString(o2kineconf).c_str());
513 mO2KineGenConfigs.push_back(std::move(o2kineConfig));
514 mConfigs.push_back("extkinO2_" + std::to_string(mO2KineGenConfigs.size() - 1));
515 } else if (name == "evtpool") {
516 const auto& o2kineconf = gen["config"];
517 auto poolConfig = TBufferJSON::FromJSON<o2::eventgen::EventPoolGenConfig>(jsonValueToString(o2kineconf).c_str());
518 mEventPoolConfigs.push_back(*poolConfig);
519 mConfigs.push_back("evtpool_" + std::to_string(mEventPoolConfigs.size() - 1));
520 } else if (name == "external") {
521 const auto& extconf = gen["config"];
522 auto extConfig = TBufferJSON::FromJSON<o2::eventgen::ExternalGenConfig>(jsonValueToString(extconf).c_str());
523 mExternalGenConfigs.push_back(std::move(extConfig));
524 mConfigs.push_back("external_" + std::to_string(mExternalGenConfigs.size() - 1));
525 } else if (name == "hepmc") {
526 const auto& genconf = gen["config"];
527 const auto& cmdconf = genconf["configcmd"];
528 const auto& hepmcconf = genconf["confighepmc"];
529 auto cmdConfig = TBufferJSON::FromJSON<o2::eventgen::FileOrCmdGenConfig>(jsonValueToString(cmdconf).c_str());
530 auto hepmcConfig = TBufferJSON::FromJSON<o2::eventgen::HepMCGenConfig>(jsonValueToString(hepmcconf).c_str());
531 mFileOrCmdGenConfigs.push_back(std::move(cmdConfig));
532 mHepMCGenConfigs.push_back(std::move(hepmcConfig));
533 mConfigs.push_back("hepmc_" + std::to_string(mFileOrCmdGenConfigs.size() - 1));
534 } else {
535 mConfigs.push_back("");
536 }
537 } else {
538 if (name == "boxgen" || name == "pythia8" || name == "extkinO2" || name == "external" || name == "hepmc") {
539 LOG(fatal) << "No configuration provided for generator " << name;
540 return false;
541 } else {
542 mConfigs.push_back("");
543 }
544 }
545 if (gen.HasMember("triggers")) {
546 const auto& trigger = gen["triggers"];
547 auto trigger_specs = [this, &trigger]() {
548 mTriggerMacros.push_back({});
549 mTriggerFuncs.push_back({});
550 if (trigger.HasMember("specs")) {
551 for (auto& spec : trigger["specs"].GetArray()) {
552 if (spec.HasMember("macro")) {
553 const auto& macro = spec["macro"].GetString();
554 if (!(strcmp(macro, "") == 0)) {
555 mTriggerMacros.back().push_back(macro);
556 } else {
557 mTriggerMacros.back().push_back("");
558 }
559 } else {
560 mTriggerMacros.back().push_back("");
561 }
562 if (spec.HasMember("function")) {
563 const auto& function = spec["function"].GetString();
564 if (!(strcmp(function, "") == 0)) {
565 mTriggerFuncs.back().push_back(function);
566 } else {
567 mTriggerFuncs.back().push_back("");
568 }
569 } else {
570 mTriggerFuncs.back().push_back("");
571 }
572 }
573 } else {
574 mTriggerMacros.back().push_back("");
575 mTriggerFuncs.back().push_back("");
576 }
577 };
578 if (trigger.HasMember("mode")) {
579 const auto& trmode = trigger["mode"].GetString();
580 if (strcmp(trmode, "or") == 0) {
581 mTriggerModes.push_back(o2::eventgen::Generator::kTriggerOR);
582 trigger_specs();
583 } else if (strcmp(trmode, "and") == 0) {
584 mTriggerModes.push_back(o2::eventgen::Generator::kTriggerAND);
585 trigger_specs();
586 } else if (strcmp(trmode, "off") == 0) {
587 mTriggerModes.push_back(o2::eventgen::Generator::kTriggerOFF);
588 mTriggerMacros.push_back({""});
589 mTriggerFuncs.push_back({""});
590 } else {
591 LOG(warn) << "Wrong trigger mode provided for generator " << name << ", keeping trigger OFF";
592 mTriggerModes.push_back(o2::eventgen::Generator::kTriggerOFF);
593 mTriggerMacros.push_back({""});
594 mTriggerFuncs.push_back({""});
595 }
596 } else {
597 LOG(warn) << "No trigger mode provided for generator " << name << ", turning trigger OFF";
598 mTriggerModes.push_back(o2::eventgen::Generator::kTriggerOFF);
599 mTriggerMacros.push_back({""});
600 mTriggerFuncs.push_back({""});
601 }
602 } else {
603 mTriggerModes.push_back(o2::eventgen::Generator::kTriggerOFF);
604 mTriggerMacros.push_back({""});
605 mTriggerFuncs.push_back({""});
606 }
607 return true;
608}
609
610Bool_t GeneratorHybrid::parseJSON(const std::string& path)
611{
612 // Parse JSON file to build map
613 std::ifstream fileStream(path, std::ios::in);
614 if (!fileStream.is_open()) {
615 LOG(error) << "Cannot open " << path;
616 return false;
617 }
618 rapidjson::IStreamWrapper isw(fileStream);
619 rapidjson::Document doc;
620 doc.ParseStream(isw);
621 if (doc.HasParseError()) {
622 LOG(error) << "Error parsing provided json file " << path;
623 LOG(error) << " - Error -> " << rapidjson::GetParseError_En(doc.GetParseError());
624 return false;
625 }
626
627 // check if there is a mode field
628 if (doc.HasMember("mode")) {
629 const auto& mode = doc["mode"].GetString();
630 if (strcmp(mode, "sequential") == 0) {
631 // events are generated in the order given by fractions or random weight
632 mGenerationMode = GenMode::kSeq;
633 }
634 if (strcmp(mode, "parallel") == 0) {
635 // events are generated fully in parallel and the order will be random
636 // this is mainly for event pool generation or mono-type generators
637 mGenerationMode = GenMode::kParallel;
638 LOG(info) << "Setting mode to parallel";
639 }
640 }
641
642 // Put the generator names in mInputGens
643 if (doc.HasMember("generators")) {
644 const auto& gens = doc["generators"];
645 for (const auto& gen : gens.GetArray()) {
646 mGroups.push_back({});
647 // Check if gen is an array (cocktail mode)
648 if (gen.HasMember("cocktail")) {
649 mCocktailMode = true;
650 for (const auto& subgen : gen["cocktail"].GetArray()) {
651 if (confSetter(subgen)) {
652 mGroups.back().push_back(mInputGens.size() - 1);
653 } else {
654 return false;
655 }
656 }
657 } else {
658 if (!confSetter(gen)) {
659 return false;
660 }
661 // Groups are created in case cocktail mode is activated, this way
662 // cocktails can be declared anywhere in the JSON file, without the need
663 // of grouping single generators. If no cocktail is defined
664 // groups will be ignored nonetheless.
665 mGroups.back().push_back(mInputGens.size() - 1);
666 }
667 }
668 }
669
670 // Get fractions and put them in mFractions
671 if (doc.HasMember("fractions")) {
672 const auto& fractions = doc["fractions"];
673 for (const auto& frac : fractions.GetArray()) {
674 mFractions.push_back(frac.GetInt());
675 }
676 } else {
677 // Set fractions to unity for all generators in case they are not provided
678 const auto& gens = doc["generators"];
679 for (const auto& gen : gens.GetArray()) {
680 mFractions.push_back(1);
681 }
682 }
683 return true;
684}
685
686} // namespace eventgen
687} // namespace o2
688
default_random_engine gen(dev())
int32_t i
ClassImp(o2::eventgen::GeneratorHybrid)
uint16_t pos
Definition RawData.h:3
std::ostringstream debug
static void writeINI(std::string const &filename, std::string const &keyOnly="")
static void updateFromFile(std::string const &, std::string const &paramsList="", bool unchangedOnly=false)
static SimConfig & Instance()
Definition SimConfig.h:111
void copyInfoFrom(MCEventHeader const &other)
inits info fields from another Event header
void putInfo(std::string const &key, T const &value)
Bool_t parseJSON(const std::string &path)
std::string jsonValueToString(const T &value)
void updateHeader(o2::dataformats::MCEventHeader *eventHeader) override
GeneratorHybrid(const std::string &inputgens)
Bool_t confSetter(const auto &gen)
void setPositionUnit(double val)
Definition Generator.h:84
void setEnergyUnit(double val)
Definition Generator.h:82
void notifySubGenerator(int subGeneratorId)
Definition Generator.h:120
static unsigned int getTotalNEvents()
Definition Generator.h:97
void setTimeUnit(double val)
Definition Generator.h:86
void addSubGenerator(int subGeneratorId, std::string const &subGeneratorDescription)
std::vector< TParticle > mParticles
Definition Generator.h:144
void setMomentumUnit(double val)
Definition Generator.h:80
Bool_t Init() override
Definition Generator.cxx:57
float sum(float s, o2::dcs::DataPointValue v)
Definition dcs-ccdb.cxx:39
GLenum mode
Definition glcorearb.h:266
GLint GLsizei count
Definition glcorearb.h:399
GLuint buffer
Definition glcorearb.h:655
GLsizeiptr size
Definition glcorearb.h:659
GLuint index
Definition glcorearb.h:781
GLuint const GLchar * name
Definition glcorearb.h:781
GLdouble f
Definition glcorearb.h:310
GLsizei const GLfloat * value
Definition glcorearb.h:819
GLenum const GLfloat * params
Definition glcorearb.h:272
GLintptr offset
Definition glcorearb.h:660
GLsizei const GLchar *const * path
Definition glcorearb.h:3591
std::function< bool(void *, std::string)> DeepTrigger
Definition Trigger.h:26
std::function< bool(const std::vector< TParticle > &)> Trigger
Definition Trigger.h:25
std::string expandShellVarsInFileName(std::string const &input)
a couple of static helper functions to create timestamp values for CCDB queries or override obsolete ...
std::string to_string(gsl::span< T, Size > span)
Definition common.h:52
void empty(int)
void compare(std::string_view s1, std::string_view s2)
LOG(info)<< "Compressed in "<< sw.CpuTime()<< " s"