Version: SMASH-3.2
listmodus.cc
Go to the documentation of this file.
1 /*
2  *
3  * Copyright (c) 2015-2025
4  * SMASH Team
5  *
6  * GNU General Public License (GPLv3 or later)
7  *
8  */
9 
10 #include "smash/listmodus.h"
11 
12 #include <cmath>
13 #include <cstdio>
14 #include <cstdlib>
15 #include <filesystem>
16 #include <fstream>
17 #include <list>
18 #include <map>
19 #include <sstream>
20 #include <utility>
21 #include <vector>
22 
23 #include "smash/algorithms.h"
24 #include "smash/boxmodus.h"
25 #include "smash/configuration.h"
26 #include "smash/constants.h"
28 #include "smash/fourvector.h"
29 #include "smash/inputfunctions.h"
30 #include "smash/logging.h"
31 #include "smash/particledata.h"
32 #include "smash/threevector.h"
34 
35 namespace smash {
36 static constexpr int LList = LogArea::List::id;
37 
38 static bool is_list_of_particles_invalid(const Particles &, int);
39 
41  const ExperimentParameters &param)
42  : file_id_{std::nullopt}, event_id_{0} {
43  /*
44  * Make some logic on config to understand whether to extract parent or
45  * children keys. These have all the same keys but have a different section
46  * name (like for instance 'ListBox' instead of 'List'). At the moment there
47  * is only one child and this approach is fine enough, although ugly.
48  */
49  const bool is_list =
51  const bool is_list_box =
53  if (is_list == is_list_box) {
54  throw std::logic_error(
55  "Unexpected error in ListModus constructor. Either List or ListBox "
56  "sections must be present in configuration.");
57  }
59  file_directory_key = InputKeys::modi_list_fileDirectory,
60  filename_key = InputKeys::modi_list_filename;
62  if (is_list_box) {
63  file_prefix_key = InputKeys::modi_listBox_filePrefix;
64  file_directory_key = InputKeys::modi_listBox_fileDirectory;
65  filename_key = InputKeys::modi_listBox_filename;
66  shift_id_key = InputKeys::modi_listBox_shiftId;
67  }
68  // Impose strict requirement on possible keys present in configuration file
69  const bool file_prefix_used = modus_config.has_value(file_prefix_key);
70  const bool filename_used = modus_config.has_value(filename_key);
71  if (file_prefix_used == filename_used) {
72  throw std::invalid_argument(
73  "Either 'Filename' or 'File_Prefix' key must be used in 'Modi' section "
74  "in configuration file. Please, adjust your configuration file.");
75  }
76  if (file_prefix_used) {
77  particle_list_filename_or_prefix_ = modus_config.take(file_prefix_key);
78  file_id_ = modus_config.take(shift_id_key);
79  } else {
80  particle_list_filename_or_prefix_ = modus_config.take(filename_key);
81  }
82  particle_list_file_directory_ = modus_config.take(file_directory_key);
83  if (param.n_ensembles > 1) {
84  throw std::runtime_error("ListModus only makes sense with one ensemble");
85  }
87 }
88 
89 /* console output on startup of List specific parameters */
90 std::ostream &operator<<(std::ostream &out, const ListModus &m) {
91  out << "-- List Modus\nInput directory for external particle lists:\n"
92  << m.particle_list_file_directory_ << "\n";
93  return out;
94 }
95 
97  /* (1) If particles are already at the same time - don't touch them
98  AND start at the start_time_ from the config. */
99  double earliest_formation_time = DBL_MAX;
100  double formation_time_difference = 0.0;
101  double reference_formation_time = 0.0; // avoid compiler warning
102  bool first_particle = true;
103  for (const auto &particle : particles) {
104  const double t = particle.position().x0();
105  if (t < earliest_formation_time) {
106  earliest_formation_time = t;
107  }
108  if (first_particle) {
109  reference_formation_time = t;
110  first_particle = false;
111  } else {
112  formation_time_difference += std::abs(t - reference_formation_time);
113  }
114  }
115  /* (2) If particles are NOT at the same time -> anti-stream them to
116  the earliest time (Note: not to the start_time_ set by config) */
117  bool anti_streaming_needed = (formation_time_difference > really_small);
118  start_time_ = earliest_formation_time;
119  if (anti_streaming_needed) {
120  for (auto &particle : particles) {
121  /* for hydro output where formation time is different */
122  const double t = particle.position().x0();
123  const double delta_t = t - start_time_;
124  const ThreeVector r =
125  particle.position().threevec() - delta_t * particle.velocity();
126  particle.set_4position(FourVector(start_time_, r));
127  particle.set_formation_time(t);
128  particle.set_cross_section_scaling_factor(0.0);
129  }
130  }
131 }
132 
134  double t, double x, double y, double z,
135  double mass, double E, double px, double py,
136  double pz) {
137  try {
138  ParticleData new_particle =
140  pdgcode, mass, {t, x, y, z}, {E, px, py, pz}, LList,
142  particles.insert(new_particle);
144  logg[LList].warn() << "SMASH does not recognize pdg code " << pdgcode
145  << " loaded from file. This particle will be ignored.\n";
146  }
147 }
148 
149 /* initial_conditions - sets particle data for @particles */
151  const ExperimentParameters &) {
153  if (particles->size() > 0) {
154  backpropagate_to_same_time(*particles);
155  } else {
156  start_time_ = 0.0;
157  }
158  event_id_++;
159 
160  return start_time_;
161 }
162 
164  std::string particle_list = next_event_();
165  for (const Line &line : line_parser(particle_list)) {
166  std::istringstream lineinput(line.text);
167  double t, x, y, z, mass, E, px, py, pz;
168  int id, charge;
169  std::string pdg_string;
170  lineinput >> t >> x >> y >> z >> mass >> E >> px >> py >> pz >>
171  pdg_string >> id >> charge;
172  if (lineinput.fail()) {
173  throw LoadFailure(
174  build_error_string("While loading external particle lists data:\n"
175  "Failed to convert the input string to the "
176  "expected data types.",
177  line));
178  }
179  PdgCode pdgcode(pdg_string);
180  logg[LList].debug("Particle ", pdgcode, " (x,y,z)= (", x, ", ", y, ", ", z,
181  ")");
182 
183  // Charge consistency check
184  if (pdgcode.charge() != charge) {
185  if (verbose_) {
186  logg[LList].error()
187  << "Charge of pdg = " << pdgcode << " != " << charge;
188  }
189  throw std::invalid_argument("Inconsistent input (charge).");
190  }
191  try_create_particle(particles, pdgcode, t, x, y, z, mass, E, px, py, pz);
192  }
193 }
194 
195 std::filesystem::path ListModus::file_path_(std::optional<int> file_id) {
196  std::string fname = particle_list_filename_or_prefix_ +
197  ((file_id) ? std::to_string(*file_id) : "");
198 
199  const std::filesystem::path default_path =
200  std::filesystem::absolute(particle_list_file_directory_);
201 
202  const std::filesystem::path fpath = default_path / fname;
203 
204  logg[LList].debug() << "File: " << std::filesystem::absolute(fpath) << '\n';
205 
206  if (!std::filesystem::exists(fpath)) {
207  if (verbose_) {
208  logg[LList].fatal()
209  << fpath.filename().native() << " does not exist! \n\n"
210  << "Usage of smash with external particle lists:\n"
211  << " 1. Put the external particle lists in one or more files\n"
212  << " according to the user guide instructions.\n"
213  << " 2. Particles info: t x y z mass p0 px py pz pdg ID charge\n"
214  << " in units of: fm fm fm fm GeV GeV GeV GeV GeV none none e\n";
215  }
216  throw std::runtime_error("External particle list does not exist!");
217  }
218 
219  return fpath;
220 }
221 
222 std::string ListModus::next_event_() {
223  const std::filesystem::path fpath = file_path_(file_id_);
224  std::ifstream ifs{fpath};
225  ifs.seekg(last_read_position_);
226 
227  if (!file_has_events_(fpath, last_read_position_)) {
228  if (file_id_) {
229  // Get next file and call this function recursively
230  (*file_id_)++;
232  ifs.close();
233  return next_event_();
234  } else {
235  throw std::runtime_error(
236  "Attempt to read in next event in ListModus object but no further "
237  "data found in single provided file. Please, check your setup.");
238  }
239  }
240 
241  // read one event. events marked by line # event end i in case of Oscar
242  // output. Assume one event per file for all other output formats
243  std::string event_string;
244  const std::string needle = "end";
245  std::string line;
246  while (getline(ifs, line)) {
247  if (line.find(needle) == std::string::npos) {
248  event_string += line + "\n";
249  } else {
250  break;
251  }
252  }
253 
254  if (!ifs.eof() && (ifs.fail() || ifs.bad())) {
255  if (verbose_) {
256  logg[LList].fatal() << "Error while reading "
257  << fpath.filename().native();
258  }
259  throw std::runtime_error("Error while reading external particle list");
260  }
261  // save position for next event read
262  last_read_position_ = ifs.tellg();
263  ifs.close();
264 
265  return event_string;
266 }
267 
268 bool ListModus::file_has_events_(std::filesystem::path filepath,
269  std::streampos last_position) {
270  std::ifstream ifs{filepath};
271  std::string line;
272 
273  // last event read read at end of file. we know this because errors are
274  // handled in next_event
275  if (last_position == -1) {
276  return false;
277  }
278  ifs.seekg(last_position);
279  // skip over comment lines, assume that a max. of four consecutive comment
280  // lines can occur
281  int skipped_lines = 0;
282  const int max_comment_lines = 4;
283  while (std::getline(ifs, line) && line[0] != '#' &&
284  skipped_lines++ < max_comment_lines) {
285  }
286 
287  if (ifs.eof()) {
288  return false;
289  }
290 
291  if (!ifs.good()) {
292  if (verbose_) {
293  logg[LList].fatal() << "Error while reading "
294  << filepath.filename().native();
295  }
296  throw std::runtime_error("Error while reading external particle list");
297  }
298 
299  ifs.close();
300  return true;
301 }
302 
303 /* In this method, which is called from the constructor only, we "abuse" of the
304  * class functionality to read in all events and validate them. In order not to
305  * modify the original object we work on an utility copy. Note that the copy
306  * constructor provided by the compiler is enough as the class has only STL or
307  * builtin members.
308  */
310  ListModus utility_copy{*this};
311  utility_copy.verbose_ = false;
312  utility_copy.warn_about_mass_discrepancy_ = false;
313  utility_copy.warn_about_off_shell_particles_ = false;
314  bool are_there_faulty_events = false;
315  while (true) {
316  try {
317  Particles particles{};
318  utility_copy.read_particles_from_next_event_(particles);
319  if (is_list_of_particles_invalid(particles, utility_copy.event_id_)) {
320  are_there_faulty_events = true;
321  }
322  utility_copy.event_id_++;
323  } catch (const std::exception &) {
324  break;
325  }
326  }
327  if (are_there_faulty_events) {
328  throw InvalidEvents(
329  "More than 2 particles with the same 4-position have been found in the "
330  "same event.\nPlease, check your particles list file.");
331  }
332 }
333 
335  const ExperimentParameters &param)
336  : ListModus(), length_(modus_config.take(InputKeys::modi_listBox_length)) {
337  /*
338  * ATTENTION: In a child class initialization list nothing can be done before
339  * calling the base constructor. However, here we cannot hand over the
340  * configuration to the base class as there are child-specific keys to be
341  * taken before. This cannot be done after having moved the configuration and
342  * changing the constructor signature would be a big change as all modus
343  * classes should have the same constructor signature to allow Experiment to
344  * template on it. Therefore, we abuse C++ here by default-initializing the
345  * parent class, then taking the child-specific key(s) and then assigning a
346  * parent instance to the child using the parent assignment operator. In
347  * general this would be risky as it would open up the possibility to leave
348  * part of the children uninitialized, but here we should have under control
349  * what exactly happens at initialization time.
350  */
351  this->ListModus::operator=(ListModus(std::move(modus_config), param));
352 }
353 
355  const OutputsList &output_list) {
356  int wraps = 0;
357  for (ParticleData &data : *particles) {
358  FourVector position = data.position();
359  bool wall_hit = enforce_periodic_boundaries(position.begin() + 1,
360  position.end(), length_);
361  if (wall_hit) {
362  const ParticleData incoming_particle(data);
363  data.set_4position(position);
364  ++wraps;
365  ActionPtr action =
366  std::make_unique<WallcrossingAction>(incoming_particle, data);
367  for (const auto &output : output_list) {
368  if (!output->is_dilepton_output() && !output->is_photon_output()) {
369  output->at_interaction(*action, 0.);
370  }
371  }
372  }
373  }
374 
375  logg[LList].debug("Moved ", wraps, " particles back into the box.");
376  return wraps;
377 }
378 
379 /* This function is meant to throw if there are more than two particles at the
380  * same position. To be more user-friendly we first check all particles and then
381  * report about all faulty groups of particles with their position. Only
382  * afterwards the simulation is aborted. */
383 static bool is_list_of_particles_invalid(const Particles &particles,
384  int event) {
385  /* In order to make the desired check, particles are classified in an std::map
386  * using their position as a key. However, to do so, the operator< of the
387  * FourVector class is not suitable since in a std::map, by default, two keys
388  * a and b are considered equivalent if !(a<b) && !(b<a). Therefore we convert
389  * the 4-postion to a string and use this as key. Note that this should also
390  * work in the case in which the file contains apparently different positions,
391  * i.e. with differences in the decimals beyond double precision. At this
392  * point the file has been already read and the 4-positions are stored in
393  * double position. */
394  auto to_string = [](const FourVector &v) {
395  return "(" + std::to_string(v[0]) + ", " + std::to_string(v[1]) + ", " +
396  std::to_string(v[2]) + ", " + std::to_string(v[3]) + ")";
397  };
398  std::map<std::string, int> checker{};
399  for (const auto &p : particles) {
400  checker[to_string(p.position())]++;
401  }
402  bool error_found = false;
403  for (const auto &[key, value] : checker) {
404  if (value > 2) {
405  logg[LList].error() << "Event " << event << ": Found " << value
406  << " particles at same position " << key;
407  error_found = true;
408  }
409  }
410  return error_found;
411 }
412 } // namespace smash
Generic algorithms on containers and ranges.
Interface to the SMASH configuration files.
bool has_value(const Key< T > &key) const
Return whether there is a non-empty value behind the requested key (which is supposed not to refer to...
T take(const Key< T > &key)
The default interface for SMASH to read configuration values.
The FourVector class holds relevant values in Minkowski spacetime with (+, −, −, −) metric signature.
Definition: fourvector.h:33
iterator end()
Definition: fourvector.h:294
iterator begin()
Definition: fourvector.h:291
const double length_
Length of the cube's edge in fm.
Definition: listmodus.h:342
int impose_boundary_conditions(Particles *particles, const OutputsList &output_list={})
Enforces that all particles are inside the box at the beginning of an event.
Definition: listmodus.cc:354
ListBoxModus(Configuration modus_config, const ExperimentParameters &parameters)
Constructor (This is the same as for the ListModus)
Definition: listmodus.cc:334
ListModus: Provides a modus for running SMASH on an external particle list, for example as an afterbu...
Definition: listmodus.h:56
void backpropagate_to_same_time(Particles &particles)
Judge whether formation times are the same for all the particles; Don't do anti-freestreaming if all ...
Definition: listmodus.cc:96
bool file_has_events_(std::filesystem::path filepath, std::streampos last_position)
Check if the given file has events left after the given position.
Definition: listmodus.cc:268
std::optional< int > file_id_
The id of the current file.
Definition: listmodus.h:231
bool is_list() const
Definition: listmodus.h:156
std::string particle_list_filename_or_prefix_
Prefix of the file(s) containing the particle list.
Definition: listmodus.h:228
std::filesystem::path file_path_(std::optional< int > file_id)
Return the absolute path of the data file.
Definition: listmodus.cc:195
std::string particle_list_file_directory_
File directory of the particle list.
Definition: listmodus.h:222
void validate_list_of_particles_of_all_events_() const
Read and validate all events particles.
Definition: listmodus.cc:309
void read_particles_from_next_event_(Particles &particles)
Read the next event from file.
Definition: listmodus.cc:163
double start_time_
Starting time for the List; changed to the earliest formation time.
Definition: listmodus.h:160
ListModus()=default
Construct an empty list.
void try_create_particle(Particles &particles, PdgCode pdgcode, double t, double x, double y, double z, double mass, double E, double px, double py, double pz)
Tries to add a new particle to particles and performs consistency checks: (i) The PDG code is legal a...
Definition: listmodus.cc:133
bool verbose_
Flag to suppress some error messages.
Definition: listmodus.h:250
std::streampos last_read_position_
Last read position in current file.
Definition: listmodus.h:237
bool warn_about_mass_discrepancy_
Auxiliary flag to warn about mass-discrepancies only once per instance.
Definition: listmodus.h:240
bool warn_about_off_shell_particles_
Auxiliary flag to warn about off-shell particles only once per instance.
Definition: listmodus.h:242
std::string next_event_()
Read the next event.
Definition: listmodus.cc:222
double initial_conditions(Particles *particles, const ExperimentParameters &parameters)
Generates initial state of the particles in the system according to a list.
Definition: listmodus.cc:150
int event_id_
The unique id of the current event.
Definition: listmodus.h:234
ParticleData contains the dynamic information of a certain particle.
Definition: particledata.h:58
The Particles class abstracts the storage and manipulation of particles.
Definition: particles.h:33
size_t size() const
Definition: particles.h:87
const ParticleData & insert(const ParticleData &p)
Inserts the particle into the list of particles.
Definition: particles.cc:50
PdgCode stores a Particle Data Group Particle Numbering Scheme particle type number.
Definition: pdgcode.h:111
int charge() const
The charge of the particle.
Definition: pdgcode.h:629
The ThreeVector class represents a physical three-vector with the components .
Definition: threevector.h:31
Collection of useful constants that are known at compile time.
std::ostream & operator<<(std::ostream &out, const ActionPtr &action)
Convenience: dereferences the ActionPtr to Action.
Definition: action.h:546
std::array< einhard::Logger<>, std::tuple_size< LogArea::AreaTuple >::value > logg
An array that stores all pre-configured Logger objects.
Definition: logging.cc:40
constexpr int p
Proton.
Definition: action.h:24
static bool enforce_periodic_boundaries(Iterator begin, const Iterator &end, typename std::iterator_traits< Iterator >::value_type length)
Enforces periodic boundaries on the given collection of values.
Definition: algorithms.h:53
build_vector_< Line > line_parser(const std::string &input)
Helper function for parsing particles.txt and decaymodes.txt.
static constexpr int LList
Definition: listmodus.cc:36
ParticleData create_valid_smash_particle_matching_provided_quantities(PdgCode pdgcode, double mass, const FourVector &four_position, const FourVector &four_momentum, int log_area, bool &mass_warning, bool &on_shell_warning)
This function creates a SMASH particle validating the provided information.
constexpr double really_small
Numerical error tolerance.
Definition: constants.h:37
std::string build_error_string(std::string message, const Line &line)
Builds a meaningful error message.
static bool is_list_of_particles_invalid(const Particles &, int)
Definition: listmodus.cc:383
Helper structure for Experiment.
int n_ensembles
Number of parallel ensembles.
A container to keep track of all ever existed input keys.
Definition: input_keys.h:1067
static const Key< std::string > modi_list_fileDirectory
See user guide description for more information.
Definition: input_keys.h:4441
static const Key< std::string > modi_listBox_filePrefix
See user guide description for more information.
Definition: input_keys.h:4525
static const Key< std::string > modi_list_filename
See user guide description for more information.
Definition: input_keys.h:4456
static const Key< int > modi_listBox_shiftId
See user guide description for more information.
Definition: input_keys.h:4551
static const Key< int > modi_list_shiftId
See user guide description for more information.
Definition: input_keys.h:4486
static const Key< std::string > modi_list_filePrefix
See user guide description for more information.
Definition: input_keys.h:4469
static const Key< std::string > modi_listBox_filename
See user guide description for more information.
Definition: input_keys.h:4512
static const Key< std::string > modi_listBox_fileDirectory
See user guide description for more information.
Definition: input_keys.h:4499
Line consists of a line number and the contents of that line.
Used when external particle list is invalid.
Definition: listmodus.h:151
Used when external particle list cannot be found.
Definition: listmodus.h:144