6#include "G4ParticleTable.hh"
16constexpr size_t LUND_MIN_HEADER_COLUMNS = 10;
17constexpr size_t LUND_MAX_HEADER_COLUMNS = 100;
18constexpr size_t LUND_MIN_PARTICLE_COLUMNS = 14;
19constexpr size_t LUND_MAX_PARTICLE_COLUMNS = 100;
20constexpr int LUND_PROPAGATED_TYPE = 1;
22bool is_blank_line(
const std::string& line) {
23 return line.find_first_not_of(
" \t\r\n") == std::string::npos;
26std::vector<double> parse_lund_header(
const std::string& line) {
27 std::istringstream stream(line);
28 std::vector<double> values;
31 while (stream >> value) { values.emplace_back(value); }
33 if (!stream.eof()) { values.clear(); }
38struct LundParticleLine
56bool parse_lund_particle_line(
const std::string& line, LundParticleLine& particle) {
57 std::istringstream stream(line);
58 std::vector<double> values;
61 while (stream >> value) { values.emplace_back(value); }
64 values.size() < LUND_MIN_PARTICLE_COLUMNS ||
65 values.size() > LUND_MAX_PARTICLE_COLUMNS) {
69 particle.index =
static_cast<int>(values[0]);
70 particle.lifetime = values[1];
71 particle.type =
static_cast<int>(values[2]);
72 particle.pid =
static_cast<int>(values[3]);
73 particle.parent =
static_cast<int>(values[4]);
74 particle.daughter =
static_cast<int>(values[5]);
75 particle.px = values[6];
76 particle.py = values[7];
77 particle.pz = values[8];
78 particle.energy = values[9];
79 particle.mass = values[10];
80 particle.vx = values[11];
81 particle.vy = values[12];
82 particle.vz = values[13];
84 return particle.index == values[0] &&
85 particle.type == values[2] &&
86 particle.pid == values[3] &&
87 particle.parent == values[4] &&
88 particle.daughter == values[5];
91std::string particle_name_from_pid(
int pid,
const std::shared_ptr<GLogger>& logger) {
92 auto particle_table = G4ParticleTable::GetParticleTable();
93 if (particle_table ==
nullptr) {
98 auto particle_definition = particle_table->FindParticle(pid);
99 if (particle_definition ==
nullptr) {
104 return particle_definition->GetParticleName();
107GparticlePtr make_gparticle_from_lund(
const LundParticleLine& particle,
const std::shared_ptr<GLogger>& logger) {
108 const auto particle_name = particle_name_from_pid(particle.pid, logger);
109 if (particle_name.empty()) {
return nullptr; }
111 const double momentum = std::sqrt(
112 particle.px * particle.px +
113 particle.py * particle.py +
114 particle.pz * particle.pz
118 if (momentum > 0) { theta = std::acos(particle.pz / momentum); }
120 const double phi = std::atan2(particle.py, particle.px);
122 return std::make_shared<Gparticle>(
146 const std::shared_ptr<GLogger>& logger) {
149 const double momentum = std::sqrt(
150 particle.px * particle.px +
151 particle.py * particle.py +
152 particle.pz * particle.pz
156 if (momentum > 0) { theta = std::acos(particle.pz / momentum); }
158 std::string particle_name;
159 auto particle_table = G4ParticleTable::GetParticleTable();
160 if (particle_table !=
nullptr) {
161 auto particle_definition = particle_table->FindParticle(particle.pid);
162 if (particle_definition !=
nullptr) { particle_name = particle_definition->GetParticleName(); }
164 if (particle_name.empty()) { particle_name = std::to_string(particle.pid); }
173 std::atan2(particle.py, particle.px),
183 const std::shared_ptr<GLogger>& logger,
184 bool propagated_only) {
186 std::ifstream input(
source.filename);
188 if (!input.is_open()) {
194 while (std::getline(input,
line)) {
195 if (is_blank_line(
line)) {
continue; }
197 const auto header_values = parse_lund_header(
line);
199 if (header_values.size() < LUND_MIN_HEADER_COLUMNS || header_values.size() > LUND_MAX_HEADER_COLUMNS) {
204 const auto particle_count =
static_cast<int>(header_values.front());
205 if (particle_count < 0 || particle_count != header_values.front()) {
207 "Lund event header first column must be a non-negative integer in <",
212 bool have_first_particle_line =
false;
213 if (!std::getline(input,
line)) {
215 "Lund event header must be followed by a blank line in <",
source.filename,
">");
218 if (!is_blank_line(
line)) { have_first_particle_line =
true; }
221 for (
int i = 0; i < particle_count; i++) {
222 if (have_first_particle_line) {
223 have_first_particle_line =
false;
225 else if (!std::getline(input,
line)) {
227 "Lund event declared ", particle_count,
" particles but ended after ",
228 i,
" particle lines in <",
source.filename,
">");
232 if (is_blank_line(
line)) {
234 "Unexpected blank line inside Lund particle block in <",
source.filename,
">");
238 LundParticleLine lund_particle;
239 if (!parse_lund_particle_line(
line, lund_particle)) {
244 if (lund_particle.index != i + 1) {
246 "Lund particle index must start from 1 and follow particle order in <",
251 if (propagated_only && lund_particle.type != LUND_PROPAGATED_TYPE) {
continue; }
253 auto particle = make_gparticle_from_lund(lund_particle, logger);
254 if (particle !=
nullptr) { event_particles.emplace_back(particle); }
257 events.emplace_back(std::move(event_particles));
260 logger->info(1,
"Loaded ", events.size(),
" Lund events from <",
source.filename,
">");
265 const std::shared_ptr<GLogger>& logger) {
266 std::vector<GparticlePtr> particles;
268 particles.insert(particles.end(),
event.begin(),
event.end());
271 logger->info(1,
"Loaded ", particles.size(),
" propagated particles from Lund file <",
source.filename,
">");
276 const std::shared_ptr<GLogger>& logger) {
278 std::ifstream input(
source.filename);
280 if (!input.is_open()) {
286 while (std::getline(input,
line)) {
287 if (is_blank_line(
line)) {
continue; }
289 const auto header_values = parse_lund_header(
line);
290 if (header_values.size() < LUND_MIN_HEADER_COLUMNS || header_values.size() > LUND_MAX_HEADER_COLUMNS) {
295 const auto particle_count =
static_cast<int>(header_values.front());
296 if (particle_count < 0 || particle_count != header_values.front()) {
298 "Lund event header first column must be a non-negative integer in <",
303 bool have_first_particle_line =
false;
304 if (!std::getline(input,
line)) {
306 "Lund event header must be followed by a blank line in <",
source.filename,
">");
309 if (!is_blank_line(
line)) { have_first_particle_line =
true; }
312 for (
int i = 0; i < particle_count; i++) {
313 if (have_first_particle_line) {
314 have_first_particle_line =
false;
316 else if (!std::getline(input,
line)) {
318 "Lund event declared ", particle_count,
" particles but ended after ",
319 i,
" particle lines in <",
source.filename,
">");
323 if (is_blank_line(
line)) {
325 "Unexpected blank line inside Lund particle block in <",
source.filename,
">");
329 LundParticleLine lund_particle;
330 if (!parse_lund_particle_line(
line, lund_particle)) {
335 if (lund_particle.index != i + 1) {
337 "Lund particle index must start from 1 and follow particle order in <",
342 event_particles.emplace_back(make_record_from_lund(lund_particle, logger));
345 events.emplace_back(std::move(event_particles));
348 logger->info(1,
"Loaded ", events.size(),
" Lund generated-particle record events from <",
source.filename,
">");
GParticleRecordEvents loadParticleRecordEvents(const GParticleSourceDefinition &source, const std::shared_ptr< GLogger > &logger) override
Loads Lund events as generated-particle records.
GParticleEvents loadParticleEvents(const GParticleSourceDefinition &source, const std::shared_ptr< GLogger > &logger, bool propagated_only=true) override
Loads Lund events as Geant4-shootable particles.
std::vector< GparticlePtr > loadParticles(const GParticleSourceDefinition &source, const std::shared_ptr< GLogger > &logger) override
Loads all propagated Lund particles as one flattened list.
Conventions and error codes for the gparticle module.
#define ERR_GPARTICLEFILEFORMAT
Particle input file contained malformed or unsupported data.
#define ERR_GPARTICLEFILEOPEN
Particle input file could not be opened.
#define ERR_GPARTICLETABLENOTFOUND
G4ParticleTable could not be obtained (unexpected runtime state).
#define ERR_GPARTICLENOTFOUND
Requested particle name was not found in the G4ParticleTable.
std::vector< GParticleEvent > GParticleEvents
Sequence of file-backed generated-particle events.
std::vector< GparticlePtr > GParticleEvent
Event-local list of Geant4-propagated generator particles.
std::vector< GParticleRecordEvent > GParticleRecordEvents
Sequence of generated-particle record events indexed by event number.
std::vector< GParticleRecord > GParticleRecordEvent
Event-local list of generated-particle records for output.
std::shared_ptr< Gparticle > GparticlePtr
Shared pointer type used for Gparticle instances.
double getG4Number(const string &v, bool warnIfNotUnit=false)
Immutable generated-particle metadata used for output banks.
One configured -gparticlefile source.