13#include <fairlogger/Logger.h>
15#include <tbb/concurrent_queue.h>
16#include <tbb/task_arena.h>
17#include <tbb/parallel_for.h>
43 if (inputgens.starts_with(
"alien://")) {
45 TGrid::Connect(
"alien://");
47 LOG(fatal) <<
"AliEn connection failed, check token.";
51 TString aliencp = Form(
"alien_cp %s file:./%s",
52 inputgens.c_str(),
"hybridAlien.json");
53 if (gSystem->Exec(aliencp.Data()) != 0) {
54 LOG(fatal) <<
"Error: Issues in fetching file" << inputgens;
60 if (!
parseJSON(isAlien ?
"hybridAlien.json" : inputgens)) {
61 LOG(fatal) <<
"Failed to parse JSON configuration from input generators";
65 if (mConfigs.size() != mInputGens.size()) {
66 LOG(fatal) <<
"Number of configurations does not match the number of generators";
69 if (mConfigs.size() == 0) {
70 for (
auto gen : mInputGens) {
71 mConfigs.push_back(
"");
75 if (!(mRandomize || mGenerationMode == GenMode::kParallel)) {
77 if (mGroups.size() != mFractions.size()) {
78 LOG(fatal) <<
"Number of groups does not match the number of fractions";
82 if (mFractions.size() != mInputGens.size()) {
83 LOG(fatal) <<
"Number of fractions does not match the number of generators";
88 if (std::all_of(mFractions.begin(), mFractions.end(), [](
int i) { return i == 0; })) {
89 LOG(fatal) <<
"All fractions provided are 0, no simulation will be performed";
93 for (
auto gen : mInputGens) {
95 LOG(info) <<
"Checking if generator " <<
gen <<
" is in the list of available generators \n";
96 if (std::find(generatorNames.begin(), generatorNames.end(),
gen) != generatorNames.end()) {
97 LOG(info) <<
"Found generator " <<
gen <<
" in the list of available generators \n";
98 if (
gen.compare(
"boxgen") == 0) {
100 gens.push_back(std::make_shared<o2::eventgen::BoxGenerator>());
103 int confBoxIndex = std::stoi(mConfigs[
index].substr(7));
104 gens.push_back(std::make_shared<o2::eventgen::BoxGenerator>(*mBoxGenConfigs[confBoxIndex]));
106 mGens.push_back(
gen);
107 }
else if (
gen.compare(0, 7,
"pythia8") == 0) {
110 auto pars = Pythia8GenConfig();
111 gens.push_back(std::make_shared<o2::eventgen::GeneratorPythia8>(pars));
114 int confPythia8Index = std::stoi(mConfigs[
index].substr(8));
115 gens.push_back(std::make_shared<o2::eventgen::GeneratorPythia8>(*mPythia8GenConfigs[confPythia8Index]));
117 mConfsPythia8.push_back(mConfigs[
index]);
118 mGens.push_back(
gen);
119 }
else if (
gen.compare(
"evtpool") == 0) {
120 int confEvtPoolIndex = std::stoi(mConfigs[
index].substr(8));
121 gens.push_back(std::make_shared<o2::eventgen::GeneratorFromEventPool>(mEventPoolConfigs[confEvtPoolIndex]));
122 mGens.push_back(
gen);
123 }
else if (
gen.compare(
"external") == 0) {
124 int confextIndex = std::stoi(mConfigs[
index].substr(9));
126 if (mExternalGenConfigs[confextIndex]->iniFile.size() > 0) {
127 LOG(info) <<
"Setting up external gen using the given INI file";
134 std::string tmp_config_file =
"configkey_tmp_backup_" +
std::to_string(getpid()) + std::string(
".ini");
142 LOG(info) <<
"Setting up external generator with following parameters";
144 auto extgen_filename =
params.fileName;
145 auto extgen_func =
params.funcName;
146 auto extgen = std::shared_ptr<o2::eventgen::Generator>(o2::conf::GetFromMacro<o2::eventgen::Generator*>(extgen_filename, extgen_func,
"FairGenerator*",
"extgen"));
148 LOG(fatal) <<
"Failed to retrieve \'extgen\': problem with configuration ";
153 std::filesystem::remove(tmp_config_file);
155 gens.push_back(std::move(extgen));
156 mGens.push_back(
gen);
158 LOG(info) <<
"Setting up external gen using the given fileName and funcName";
160 auto& extgen_filename = mExternalGenConfigs[confextIndex]->fileName;
161 auto& extgen_func = mExternalGenConfigs[confextIndex]->funcName;
162 auto extGen = std::shared_ptr<o2::eventgen::Generator>(o2::conf::GetFromMacro<o2::eventgen::Generator*>(extgen_filename, extgen_func,
"FairGenerator*",
"extgen"));
164 LOG(fatal) <<
"Failed to load external generator from " << extgen_filename <<
" with function " << extgen_func;
166 gens.push_back(std::move(extGen));
167 mGens.push_back(
gen);
169 }
else if (
gen.compare(
"hepmc") == 0) {
170 int confHepMCIndex = std::stoi(mConfigs[
index].substr(6));
171 gens.push_back(std::make_shared<o2::eventgen::GeneratorHepMC>());
173 dynamic_cast<o2::eventgen::GeneratorHepMC*
>(gens.back().get())->setup(*mFileOrCmdGenConfigs[confHepMCIndex], *mHepMCGenConfigs[confHepMCIndex], globalConfig);
174 mGens.push_back(
gen);
177 LOG(fatal) <<
"Generator " <<
gen <<
" not found in the list of available generators \n";
184GeneratorHybrid::~GeneratorHybrid()
186 LOG(info) <<
"Destructor of generator hybrid called";
194 for (
auto&
gen : mGens) {
195 if (
gen ==
"pythia8pp") {
196 auto config = std::string(std::getenv(
"O2_ROOT")) +
"/share/Generators/egconfig/pythia8_inel.cfg";
197 LOG(info) <<
"Setting \'Pythia8\' base configuration: " << config << std::endl;
199 }
else if (
gen ==
"pythia8hf") {
200 auto config = std::string(std::getenv(
"O2_ROOT")) +
"/share/Generators/egconfig/pythia8_hf.cfg";
201 LOG(info) <<
"Setting \'Pythia8\' base configuration: " << config << std::endl;
203 }
else if (
gen ==
"pythia8hi") {
204 auto config = std::string(std::getenv(
"O2_ROOT")) +
"/share/Generators/egconfig/pythia8_hi.cfg";
205 LOG(info) <<
"Setting \'Pythia8\' base configuration: " << config << std::endl;
207 }
else if (
gen ==
"pythia8powheg") {
208 auto config = std::string(std::getenv(
"O2_ROOT")) +
"/share/Generators/egconfig/pythia8_powheg.cfg";
209 LOG(info) <<
"Setting \'Pythia8\' base configuration: " << config << std::endl;
215 gens[
count]->setTriggerMode(mTriggerModes[
count]);
216 LOG(info) <<
"Setting Trigger mode of generator " <<
gen <<
" to: " << mTriggerModes[
count];
219 for (
int trg = 0; trg < mTriggerMacros[
count].size(); trg++) {
224 LOG(info) <<
"Setting trigger " << trg <<
" of generator " <<
gen <<
" with following parameters";
225 LOG(info) <<
"Macro filename: " << expandedMacro;
226 LOG(info) <<
"Function name: " << mTriggerFuncs[
count][trg];
227 trigger = o2::conf::GetFromMacro<o2::eventgen::Trigger>(expandedMacro, mTriggerFuncs[
count][trg],
"o2::eventgen::Trigger",
"trigger");
229 LOG(info) <<
"Trying to retrieve a \'o2::eventgen::DeepTrigger\' type";
230 deeptrigger = o2::conf::GetFromMacro<o2::eventgen::DeepTrigger>(expandedMacro, mTriggerFuncs[
count][trg],
"o2::eventgen::DeepTrigger",
"deeptrigger");
232 if (!trigger && !deeptrigger) {
233 LOG(warn) <<
"Failed to retrieve \'external trigger\': problem with configuration";
234 LOG(warn) <<
"Trigger " << trg <<
" of generator " <<
gen <<
" will not be included";
237 LOG(info) <<
"Trigger " << trg <<
" of generator " <<
gen <<
" successfully set";
240 gens[
count]->addTrigger(trigger);
242 gens[
count]->addDeepTrigger(deeptrigger);
249 if (std::all_of(mFractions.begin(), mFractions.end(), [](
int i) { return i == 1; })) {
250 LOG(info) <<
"Full randomisation of generators order";
252 LOG(info) <<
"Randomisation based on fractions";
254 for (
auto&
f : mFractions) {
260 for (
int k = 0; k < mFractions.size(); k++) {
261 if (mFractions[k] == 0) {
263 mRngFractions.push_back(-1);
264 LOG(info) <<
"Generator " << mGens[k] <<
" will not be used";
266 chance =
static_cast<float>(mFractions[k]) / allfracs;
268 mRngFractions.push_back(
sum);
269 LOG(info) <<
"Generator " << (mConfigs[k] ==
"" ? mGens[k] : mConfigs[k]) <<
" has a " << chance * 100 <<
"% chance of being used";
274 LOG(info) <<
"Generators will be used in sequence, following provided fractions";
277 mGenIsInitialized.resize(gens.size(),
false);
278 if (mGenerationMode == GenMode::kParallel) {
280 mResultQueue.resize(1);
283 mResultQueue.resize(gens.size());
291 auto process_generator_task = [
this](std::vector<std::shared_ptr<o2::eventgen::Generator>>
const& generatorvec,
int task) {
292 LOG(
debug) <<
"Starting eventgen for task " << task;
293 auto& generator = generatorvec[task];
303 bool isTriggered =
false;
304 while (!isTriggered) {
305 generator->clearParticles();
306 generator->generateEvent();
307 generator->importParticles();
308 isTriggered = generator->triggerEvent();
310 LOG(
debug) <<
"eventgen finished for task " << task;
312 if (mGenerationMode == GenMode::kParallel) {
313 mResultQueue[0].push(task);
315 mResultQueue[task].push(task);
321 auto worker_function = [
this, process_generator_task]() {
325 auto generators_copy = gens;
329 if (mInputTaskQueue.try_pop(task)) {
330 process_generator_task(generators_copy, task);
332 std::this_thread::sleep_for(std::chrono::milliseconds(10));
338 mTBBTaskPoolRunner = std::thread([
this, worker_function]() { mTaskArena.execute([&]() { tbb::parallel_for(0, mTaskArena.max_concurrency(), [&](
int) { worker_function(); }); }); });
339 mTBBTaskPoolRunner.detach();
343 for (
size_t genindex = 0; genindex < gens.size(); ++genindex) {
344 mInputTaskQueue.push(genindex);
347 mIsInitialized =
true;
353 if (!mIsInitialized) {
356 if (mGenerationMode == GenMode::kParallel) {
365 if (mRngFractions.size() != 0) {
367 float rnum = gRandom->Rndm();
369 for (
int k = 0; k < mRngFractions.size(); k++) {
370 if (rnum <= mRngFractions[k]) {
376 mIndex = gRandom->Integer(mFractions.size());
379 while (mFractions[mCurrentFraction] == 0 || mseqCounter == mFractions[mCurrentFraction]) {
380 if (mFractions[mCurrentFraction] != 0) {
383 mCurrentFraction = (mCurrentFraction + 1) % mFractions.size();
385 mIndex = mCurrentFraction;
395 std::vector<int> subGenIndex = {};
398 mResultQueue[0].pop(genIndex);
401 if (!mCocktailMode) {
402 mResultQueue[mIndex].pop(genIndex);
405 subGenIndex.resize(mGroups[mIndex].
size());
406 for (
size_t pos = 0;
pos < mGroups[mIndex].size(); ++
pos) {
407 int subIndex = mGroups[mIndex][
pos];
408 LOG(info) <<
"Getting generator " << mGens[subIndex] <<
" from cocktail group " << mIndex;
409 mResultQueue[subIndex].pop(subGenIndex[
pos]);
414 auto unit_transformer = [](
auto& p,
auto pos_unit,
auto time_unit,
auto en_unit,
auto mom_unit) {
415 p.SetMomentum(p.Px() * mom_unit, p.Py() * mom_unit, p.Pz() * mom_unit, p.Energy() * en_unit);
416 p.SetProductionVertex(p.Vx() * pos_unit, p.Vy() * pos_unit, p.Vz() * pos_unit, p.T() * time_unit);
419 auto index_transformer = [](
auto& p,
int offset) {
420 for (
int i = 0;
i < 2; ++
i) {
421 if (p.GetMother(
i) != -1) {
422 const auto newindex = p.GetMother(
i) +
offset;
423 p.SetMother(
i, newindex);
426 if (p.GetNDaughters() > 0) {
427 for (
int i = 0;
i < 2; ++
i) {
428 const auto newindex = p.GetDaughter(
i) +
offset;
429 p.SetDaughter(
i, newindex);
439 mMCEventHeader.
Reset();
442 for (
auto subIndex : subGenIndex) {
443 LOG(info) <<
"Importing particles for task " << subIndex;
444 auto subParticles = gens[subIndex]->getParticles();
446 auto time_unit = gens[subIndex]->getTimeUnit();
447 auto pos_unit = gens[subIndex]->getPositionUnit();
448 auto mom_unit = gens[subIndex]->getMomentumUnit();
449 auto energy_unit = gens[subIndex]->getEnergyUnit();
455 for (
auto& p : subParticles) {
457 index_transformer(p,
offset);
459 unit_transformer(p, pos_unit, time_unit, energy_unit, mom_unit);
464 if (mHeaderGeneratorIndex == -1) {
465 gens[subIndex]->updateHeader(&mMCEventHeader);
466 mHeaderGeneratorIndex = subIndex;
468 mInputTaskQueue.push(subIndex);
472 LOG(info) <<
"Importing particles for task " << genIndex;
476 auto time_unit = gens[genIndex]->getTimeUnit();
477 auto pos_unit = gens[genIndex]->getPositionUnit();
478 auto mom_unit = gens[genIndex]->getMomentumUnit();
479 auto energy_unit = gens[genIndex]->getEnergyUnit();
484 unit_transformer(p, pos_unit, time_unit, energy_unit, mom_unit);
488 gens[genIndex]->updateHeader(&mMCEventHeader);
489 mHeaderGeneratorIndex = genIndex;
490 mInputTaskQueue.push(genIndex);
497 LOG(info) <<
"HybridGen: Stopping TBB task pool";
509 mMCEventHeader.SetVertex(eventHeader->GetX(), eventHeader->GetY(), eventHeader->GetZ());
510 mHeaderGeneratorIndex = -1;
517 eventHeader->
putInfo<std::string>(
"forwarding-generator",
"HybridGen");
524 rapidjson::StringBuffer
buffer;
525 rapidjson::Writer<rapidjson::StringBuffer> writer(
buffer);
526 value.Accept(writer);
527 return buffer.GetString();
532 std::string
name =
gen[
"name"].GetString();
533 mInputGens.push_back(
name);
534 if (
gen.HasMember(
"config")) {
535 if (
name ==
"boxgen") {
536 const auto& boxconf =
gen[
"config"];
537 auto boxConfig = TBufferJSON::FromJSON<o2::eventgen::BoxGenConfig>(
jsonValueToString(boxconf).c_str());
538 mBoxGenConfigs.push_back(std::move(boxConfig));
539 mConfigs.push_back(
"boxgen_" +
std::to_string(mBoxGenConfigs.size() - 1));
540 }
else if (
name ==
"pythia8") {
541 const auto& pythia8conf =
gen[
"config"];
542 auto pythia8Config = TBufferJSON::FromJSON<o2::eventgen::Pythia8GenConfig>(
jsonValueToString(pythia8conf).c_str());
543 mPythia8GenConfigs.push_back(std::move(pythia8Config));
544 mConfigs.push_back(
"pythia8_" +
std::to_string(mPythia8GenConfigs.size() - 1));
545 }
else if (
name ==
"evtpool") {
546 const auto& o2kineconf =
gen[
"config"];
547 auto poolConfig = TBufferJSON::FromJSON<o2::eventgen::EventPoolGenConfig>(
jsonValueToString(o2kineconf).c_str());
548 mEventPoolConfigs.push_back(*poolConfig);
549 mConfigs.push_back(
"evtpool_" +
std::to_string(mEventPoolConfigs.size() - 1));
550 }
else if (
name ==
"external") {
551 const auto& extconf =
gen[
"config"];
552 auto extConfig = TBufferJSON::FromJSON<o2::eventgen::ExternalGenConfig>(
jsonValueToString(extconf).c_str());
553 mExternalGenConfigs.push_back(std::move(extConfig));
554 mConfigs.push_back(
"external_" +
std::to_string(mExternalGenConfigs.size() - 1));
555 }
else if (
name ==
"hepmc") {
556 const auto& genconf =
gen[
"config"];
557 const auto& cmdconf = genconf[
"configcmd"];
558 const auto& hepmcconf = genconf[
"confighepmc"];
559 auto cmdConfig = TBufferJSON::FromJSON<o2::eventgen::FileOrCmdGenConfig>(
jsonValueToString(cmdconf).c_str());
560 auto hepmcConfig = TBufferJSON::FromJSON<o2::eventgen::HepMCGenConfig>(
jsonValueToString(hepmcconf).c_str());
561 mFileOrCmdGenConfigs.push_back(std::move(cmdConfig));
562 mHepMCGenConfigs.push_back(std::move(hepmcConfig));
563 mConfigs.push_back(
"hepmc_" +
std::to_string(mFileOrCmdGenConfigs.size() - 1));
565 mConfigs.push_back(
"");
568 if (
name ==
"boxgen" ||
name ==
"pythia8" ||
name ==
"external" ||
name ==
"hepmc") {
569 LOG(fatal) <<
"No configuration provided for generator " <<
name;
572 mConfigs.push_back(
"");
575 if (
gen.HasMember(
"triggers")) {
576 const auto& trigger =
gen[
"triggers"];
577 auto trigger_specs = [
this, &trigger]() {
578 mTriggerMacros.push_back({});
579 mTriggerFuncs.push_back({});
580 if (trigger.HasMember(
"specs")) {
581 for (
auto& spec : trigger[
"specs"].GetArray()) {
582 if (spec.HasMember(
"macro")) {
583 const auto& macro = spec[
"macro"].GetString();
584 if (!(strcmp(macro,
"") == 0)) {
585 mTriggerMacros.back().push_back(macro);
587 mTriggerMacros.back().push_back(
"");
590 mTriggerMacros.back().push_back(
"");
592 if (spec.HasMember(
"function")) {
593 const auto& function = spec[
"function"].GetString();
594 if (!(strcmp(function,
"") == 0)) {
595 mTriggerFuncs.back().push_back(function);
597 mTriggerFuncs.back().push_back(
"");
600 mTriggerFuncs.back().push_back(
"");
604 mTriggerMacros.back().push_back(
"");
605 mTriggerFuncs.back().push_back(
"");
608 if (trigger.HasMember(
"mode")) {
609 const auto& trmode = trigger[
"mode"].GetString();
610 if (strcmp(trmode,
"or") == 0) {
613 }
else if (strcmp(trmode,
"and") == 0) {
616 }
else if (strcmp(trmode,
"off") == 0) {
618 mTriggerMacros.push_back({
""});
619 mTriggerFuncs.push_back({
""});
621 LOG(warn) <<
"Wrong trigger mode provided for generator " <<
name <<
", keeping trigger OFF";
623 mTriggerMacros.push_back({
""});
624 mTriggerFuncs.push_back({
""});
627 LOG(warn) <<
"No trigger mode provided for generator " <<
name <<
", turning trigger OFF";
629 mTriggerMacros.push_back({
""});
630 mTriggerFuncs.push_back({
""});
634 mTriggerMacros.push_back({
""});
635 mTriggerFuncs.push_back({
""});
644 if (gSystem->AccessPathName(expandedPath.c_str())) {
645 LOG(fatal) <<
"Configuration file " << expandedPath <<
" for hybrid generator does not exist";
649 std::ifstream fileStream(expandedPath, std::ios::in);
650 if (!fileStream.is_open()) {
651 LOG(error) <<
"Cannot open " << expandedPath;
654 rapidjson::IStreamWrapper isw(fileStream);
655 rapidjson::Document doc;
656 doc.ParseStream(isw);
657 if (doc.HasParseError()) {
658 LOG(error) <<
"Error parsing provided json file " << expandedPath;
659 LOG(error) <<
" - Error -> " << rapidjson::GetParseError_En(doc.GetParseError());
664 if (doc.HasMember(
"mode")) {
665 const auto&
mode = doc[
"mode"].GetString();
666 if (strcmp(
mode,
"sequential") == 0) {
668 mGenerationMode = GenMode::kSeq;
670 if (strcmp(
mode,
"parallel") == 0) {
673 mGenerationMode = GenMode::kParallel;
674 LOG(info) <<
"Setting mode to parallel";
679 if (doc.HasMember(
"generators")) {
680 const auto& gens = doc[
"generators"];
681 for (
const auto&
gen : gens.GetArray()) {
682 mGroups.push_back({});
684 if (
gen.HasMember(
"cocktail")) {
685 mCocktailMode =
true;
686 for (
const auto& subgen :
gen[
"cocktail"].GetArray()) {
688 mGroups.back().push_back(mInputGens.size() - 1);
701 mGroups.back().push_back(mInputGens.size() - 1);
707 if (doc.HasMember(
"fractions")) {
708 const auto& fractions = doc[
"fractions"];
709 for (
const auto& frac : fractions.GetArray()) {
710 mFractions.push_back(frac.GetInt());
714 const auto& gens = doc[
"generators"];
715 for (
const auto&
gen : gens.GetArray()) {
716 mFractions.push_back(1);
default_random_engine gen(dev())
ClassImp(o2::eventgen::GeneratorHybrid)
static const GeneratorHybridParam & Instance()
static void writeINI(std::string const &filename, std::string const &keyOnly="")
static void updateFromFile(std::string const &, std::string const ¶msList="", bool unchangedOnly=false)
static SimConfig & Instance()
GeneratorHybrid(const GeneratorHybrid &)=delete
Bool_t parseJSON(const std::string &path)
std::string jsonValueToString(const T &value)
Bool_t generateEvent() override
void updateHeader(o2::dataformats::MCEventHeader *eventHeader) override
static GeneratorHybrid & Instance(const std::string &inputgens="")
Bool_t confSetter(const auto &gen)
Bool_t importParticles() override
void setPositionUnit(double val)
void setEnergyUnit(double val)
void notifySubGenerator(int subGeneratorId)
static unsigned int getTotalNEvents()
void setTimeUnit(double val)
void addSubGenerator(int subGeneratorId, std::string const &subGeneratorDescription)
std::vector< TParticle > mParticles
void setMomentumUnit(double val)
float sum(float s, o2::dcs::DataPointValue v)
GLuint const GLchar * name
GLsizei const GLfloat * value
GLenum const GLfloat * params
GLsizei const GLchar *const * path
std::function< bool(void *, std::string)> DeepTrigger
std::function< bool(const std::vector< TParticle > &)> Trigger
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)
void compare(std::string_view s1, std::string_view s2)
LOG(info)<< "Compressed in "<< sw.CpuTime()<< " s"