Version: SMASH-3.2
oscaroutput.cc
Go to the documentation of this file.
1 /*
2  *
3  * Copyright (c) 2014-2024
4  * SMASH Team
5  *
6  * GNU General Public License (GPLv3 or later)
7  *
8  */
9 
10 #include "smash/oscaroutput.h"
11 
12 #include <filesystem>
13 #include <string>
14 
15 #include "smash/action.h"
16 #include "smash/clock.h"
17 #include "smash/config.h"
19 
20 namespace smash {
21 static constexpr int LHyperSurfaceCrossing = LogArea::HyperSurfaceCrossing::id;
22 
23 template <OscarOutputFormat Format, int Contents>
25  const std::filesystem::path &path, const std::string &name,
26  const std::vector<std::string> quantities)
27  : OutputInterface(name),
28  file_{path / (name + ((Format == ASCII) ? ".dat" : ".oscar") +
29  ((Format == OscarFormat1999) ? "1999" : "")),
30  "w"},
31  formatter_{Format == ASCII ? quantities
32  : (Format == OscarFormat2013)
33  ? OutputDefaultQuantities::oscar2013
34  : (Format == OscarFormat2013Extended)
35  ? OutputDefaultQuantities::oscar2013extended
36  : OutputDefaultQuantities::oscar1999} {
86  if (Format != ASCII && !quantities.empty()) {
87  throw std::logic_error(
88  "Non-empty Quantities given alongside format other than ASCII.");
89  }
90  std::string format_name;
91  if (Format == ASCII) {
92  format_name = "ASCII";
93  } else if (Format == OscarFormat2013) {
94  format_name = "OSCAR2013";
95  } else if (Format == OscarFormat2013Extended) {
96  format_name = "OSCAR2013Extended";
97  } else {
98  format_name = "OSC1999A";
99  }
100  if (Format == ASCII || Format == OscarFormat2013 ||
101  Format == OscarFormat2013Extended) {
102  std::fprintf(file_.get(), "#!%s %s %s\n", format_name.c_str(), name.c_str(),
103  formatter_.quantities_line().c_str());
104  std::fprintf(file_.get(), "# Units: %s\n", formatter_.unit_line().c_str());
105  std::fprintf(file_.get(), "# %s\n", SMASH_VERSION);
106  } else {
107  const std::string &oscar_name =
108  name == "particle_lists" ? "final_id_p_x" : name;
109  // This is necessary because OSCAR1999A requires
110  // this particular string for particle output.
111 
112  std::fprintf(file_.get(), "# %s\n# %s\n# %s\n", format_name.c_str(),
113  oscar_name.c_str(), SMASH_VERSION);
114  std::fprintf(file_.get(), "# Block format:\n");
115  if (oscar_name == "full_event_history") {
116  std::fprintf(file_.get(),
117  "# nin nout density tot_weight part_weight proc_type\n");
118  } else {
119  std::fprintf(file_.get(), "# nin nout event_number ensemble_number\n");
120  }
121  std::fprintf(file_.get(), "# %s\n", formatter_.quantities_line().c_str());
122  std::fprintf(
123  file_.get(),
124  "# End of event: 0 0 event_number ensemble_number impact_parameter\n");
125  std::fprintf(file_.get(), "#\n");
126  }
127 }
128 
129 template <OscarOutputFormat Format, int Contents>
130 inline void OscarOutput<Format, Contents>::write(const Particles &particles) {
131  for (const ParticleData &data : particles) {
132  write_particledata(data);
133  }
134 }
135 
136 template <OscarOutputFormat Format, int Contents>
138  const EventLabel &event_label,
139  const EventInfo &) {
140  // We do not want the inital particle list or number to be printed in case of
141  // IC output
142  if (Contents & OscarAtEventstart && !(Contents & OscarParticlesIC)) {
143  if (Format == ASCII || Format == OscarFormat2013 ||
144  Format == OscarFormat2013Extended) {
145  std::fprintf(file_.get(), "# event %i ensemble %i in %zu\n",
146  event_label.event_number, event_label.ensemble_number,
147  particles.size());
148  } else {
149  /* OSCAR line prefix : initial particles; final particles; event id
150  * First block of an event: initial = 0, final = number of particles
151  */
152  const size_t zero = 0;
153  std::fprintf(file_.get(), "%zu %zu %i %i\n", zero, particles.size(),
154  event_label.event_number, event_label.ensemble_number);
155  }
156  write(particles);
157  } else if (Contents & OscarParticlesIC) {
158  if (Format == ASCII || Format == OscarFormat2013 ||
159  Format == OscarFormat2013Extended) {
160  std::fprintf(file_.get(), "# event %i ensemble %i start\n",
161  event_label.event_number, event_label.ensemble_number);
162  } else if (Format == OscarFormat1999) {
163  const size_t zero = 0;
164  std::fprintf(file_.get(), "%zu %zu %i %i\n", zero, zero,
165  event_label.event_number, event_label.ensemble_number);
166  }
167  }
168 }
169 
170 template <OscarOutputFormat Format, int Contents>
172  const EventLabel &event_label,
173  const EventInfo &event) {
174  if (Format == ASCII || Format == OscarFormat2013 ||
175  Format == OscarFormat2013Extended) {
176  if (Contents & OscarParticlesAtEventend ||
177  (Contents & OscarParticlesAtEventendIfNotEmpty && !event.empty_event)) {
178  std::fprintf(file_.get(), "# event %i ensemble %i out %zu\n",
179  event_label.event_number, event_label.ensemble_number,
180  particles.size());
181  write(particles);
182  }
183  // Comment end of an event
184  if (!(Contents & OscarParticlesIC)) {
185  const char *empty_event_str = event.empty_event ? "no" : "yes";
186  std::fprintf(file_.get(),
187  "# event %i ensemble %i end 0 impact %7.3f "
188  "scattering_projectile_target %s\n",
189  event_label.event_number, event_label.ensemble_number,
190  event.impact_parameter, empty_event_str);
191  } else {
192  std::fprintf(file_.get(), "# event %i ensemble %i end\n",
193  event_label.event_number, event_label.ensemble_number);
194  }
195  } else {
196  /* OSCAR line prefix : initial particles; final particles; event id
197  * Last block of an event: initial = number of particles, final = 0
198  * Block ends with null interaction. */
199  const size_t zero = 0;
200  if (Contents & OscarParticlesAtEventend ||
201  (Contents & OscarParticlesAtEventendIfNotEmpty && !event.empty_event)) {
202  std::fprintf(file_.get(), "%zu %zu %i %i\n", particles.size(), zero,
203  event_label.event_number, event_label.ensemble_number);
204  write(particles);
205  }
206  // Null interaction marks the end of an event
207  std::fprintf(file_.get(), "%zu %zu %i %i %7.3f\n", zero, zero,
208  event_label.event_number, event_label.ensemble_number,
209  event.impact_parameter);
210  }
211  // Flush to disk
212  std::fflush(file_.get());
213 
214  if (Contents & OscarParticlesIC) {
215  // If the runtime is too short some particles might not yet have
216  // reached the hypersurface. Warning is printed.
217  if (particles.size() != 0 && !event.impose_kinematic_cut_for_SMASH_IC) {
219  "End time might be too small for initial conditions output. "
220  "Hypersurface has not yet been crossed by ",
221  particles.size(), " particle(s).");
222  }
223  }
224 }
225 
226 template <OscarOutputFormat Format, int Contents>
228  const double density) {
229  if (Contents & OscarInteractions) {
230  if (Format == ASCII || Format == OscarFormat2013 ||
231  Format == OscarFormat2013Extended) {
232  std::fprintf(file_.get(),
233  "# interaction in %zu out %zu rho %12.7f weight %12.7g"
234  " partial %12.7f type %5i\n",
235  action.incoming_particles().size(),
236  action.outgoing_particles().size(), density,
237  action.get_total_weight(), action.get_partial_weight(),
238  static_cast<int>(action.get_type()));
239  } else {
240  /* OSCAR line prefix : initial final
241  * particle creation: 0 1
242  * particle 2<->2 collision: 2 2
243  * resonance formation: 2 1
244  * resonance decay: 1 2
245  * etc.*/
246  std::fprintf(file_.get(), "%zu %zu %12.7f %12.7f %12.7f %5i\n",
247  action.incoming_particles().size(),
248  action.outgoing_particles().size(), density,
249  action.get_total_weight(), action.get_partial_weight(),
250  static_cast<int>(action.get_type()));
251  }
252  for (const auto &p : action.incoming_particles()) {
253  write_particledata(p);
254  }
255  for (const auto &p : action.outgoing_particles()) {
256  write_particledata(p);
257  }
258  } else if (Contents & OscarParticlesIC) {
259  for (const auto &p : action.incoming_particles()) {
260  write_particledata(p);
261  }
262  }
263 }
264 
265 template <OscarOutputFormat Format, int Contents>
267  const Particles &particles, const std::unique_ptr<Clock> &,
268  const DensityParameters &, const EventLabel &event_label,
269  const EventInfo &) {
270  if (Contents & OscarTimesteps) {
271  if (Format == ASCII || Format == OscarFormat2013 ||
272  Format == OscarFormat2013Extended) {
273  std::fprintf(file_.get(), "# event %i ensemble %i out %zu\n",
274  event_label.event_number, event_label.ensemble_number,
275  particles.size());
276  } else {
277  const size_t zero = 0;
278  std::fprintf(file_.get(), "%zu %zu %i %i\n", particles.size(), zero,
279  event_label.event_number, event_label.ensemble_number);
280  }
281  write(particles);
282  }
283 }
284 
841 template <OscarOutputFormat Format, int Contents>
843  const ParticleData &data) {
844  std::fprintf(file_.get(), "%s\n", formatter_.data_line(data).c_str());
845 }
846 
847 namespace {
860 template <int Contents>
861 std::unique_ptr<OutputInterface> create_select_format(
862  bool modern_format, const std::filesystem::path &path,
863  const OutputParameters &out_par, const std::string &name,
864  const bool custom_format = false) {
865  bool extended_format = (Contents & OscarInteractions) ? out_par.coll_extended
866  : out_par.part_extended;
867  if (custom_format) {
868  const auto &quantities = (Contents & OscarInteractions)
869  ? out_par.quantities.at("Collisions")
870  : out_par.quantities.at("Particles");
871  return std::make_unique<OscarOutput<ASCII, Contents>>(path, name,
872  quantities);
873  } else if (modern_format && extended_format) {
874  return std::make_unique<OscarOutput<OscarFormat2013Extended, Contents>>(
875  path, name);
876  } else if (modern_format && !extended_format) {
877  return std::make_unique<OscarOutput<OscarFormat2013, Contents>>(path, name);
878  } else if (!modern_format && !extended_format) {
879  return std::make_unique<OscarOutput<OscarFormat1999, Contents>>(path, name);
880  } else {
881  // Only remaining possibility: (!modern_format && extended_format)
882  logg[LOutput].warn() << "Creating Oscar output: "
883  << "There is no extended Oscar1999 format.";
884  return std::make_unique<OscarOutput<OscarFormat1999, Contents>>(path, name);
885  }
886 }
887 } // unnamed namespace
888 
889 std::unique_ptr<OutputInterface> create_oscar_output(
890  const std::string &format, const std::string &content,
891  const std::filesystem::path &path, const OutputParameters &out_par) {
892  if (format != "Oscar2013" && format != "Oscar1999" && format != "ASCII") {
893  throw std::invalid_argument("Creating Oscar output: unknown format");
894  }
895  const bool modern_format = (format == "Oscar2013");
896  const bool custom_format = (format == "ASCII");
897  if (content == "Particles") {
898  if (out_par.part_only_final == OutputOnlyFinal::Yes) {
899  return create_select_format<OscarParticlesAtEventend>(
900  modern_format, path, out_par, "particle_lists", custom_format);
901  } else if (out_par.part_only_final == OutputOnlyFinal::IfNotEmpty) {
902  return create_select_format<OscarParticlesAtEventendIfNotEmpty>(
903  modern_format, path, out_par, "particle_lists", custom_format);
904 
905  } else { // out_par.part_only_final == OutputOnlyFinal::No
908  modern_format, path, out_par, "particle_lists", custom_format);
909  }
910  } else if (content == "Collisions") {
911  if (out_par.coll_printstartend) {
914  modern_format, path, out_par, "full_event_history", custom_format);
915  } else {
916  return create_select_format<OscarInteractions>(
917  modern_format, path, out_par, "full_event_history", custom_format);
918  }
919  } else if (content == "Dileptons") {
920  if (modern_format && out_par.dil_extended) {
921  return std::make_unique<
923  "Dileptons");
924  } else if (modern_format && !out_par.dil_extended) {
925  return std::make_unique<OscarOutput<OscarFormat2013, OscarInteractions>>(
926  path, "Dileptons");
927  } else if (!modern_format && !out_par.dil_extended) {
928  return std::make_unique<OscarOutput<OscarFormat1999, OscarInteractions>>(
929  path, "Dileptons");
930  } else if (!custom_format && !modern_format && out_par.dil_extended) {
931  logg[LOutput].warn()
932  << "Creating Oscar output: "
933  << "There is no extended Oscar1999 (dileptons) format.";
934  }
935  } else if (content == "Photons") {
936  if (modern_format && !out_par.photons_extended) {
937  return std::make_unique<OscarOutput<OscarFormat2013, OscarInteractions>>(
938  path, "Photons");
939  } else if (modern_format && out_par.photons_extended) {
940  return std::make_unique<
942  "Photons");
943  } else if (!modern_format && !out_par.photons_extended) {
944  return std::make_unique<OscarOutput<OscarFormat1999, OscarInteractions>>(
945  path, "Photons");
946  } else if (!custom_format && !modern_format && out_par.photons_extended) {
947  logg[LOutput].warn()
948  << "Creating Oscar output: "
949  << "There is no extended Oscar1999 (photons) format.";
950  }
951  } else if (content == "Initial_Conditions") {
952  if (modern_format && !out_par.ic_extended) {
953  return std::make_unique<
955  path, "SMASH_IC");
956  } else if (modern_format && out_par.ic_extended) {
957  return std::make_unique<OscarOutput<
959  path, "SMASH_IC");
960  } else if (!modern_format && !out_par.ic_extended) {
961  return std::make_unique<
963  path, "SMASH_IC");
964  } else if (!custom_format && !modern_format && out_par.ic_extended) {
965  logg[LOutput].warn()
966  << "Creating Oscar output: "
967  << "There is no extended Oscar1999 (initial conditions) format.";
968  }
969  }
970 
971  throw std::invalid_argument("Create_oscar_output got unknown content.");
972 }
973 
974 } // namespace smash
Action is the base class for a generic process that takes a number of incoming particles and transfor...
Definition: action.h:35
virtual ProcessType get_type() const
Get the process type.
Definition: action.h:131
virtual double get_total_weight() const =0
Return the total weight value, which is mainly used for the weight output entry.
const ParticleList & incoming_particles() const
Get the list of particles that go into the action.
Definition: action.cc:58
virtual double get_partial_weight() const =0
Return the specific weight for the chosen outgoing channel, which is mainly used for the partial weig...
const ParticleList & outgoing_particles() const
Get the list of particles that resulted from the action.
Definition: action.h:247
A class to pre-calculate and store parameters relevant for density calculation.
Definition: density.h:92
Abstraction of generic output.
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
@ IfNotEmpty
Print only final-state particles, and those only if the event is not empty.
@ Yes
Print only final-state particles.
std::array< einhard::Logger<>, std::tuple_size< LogArea::AreaTuple >::value > logg
An array that stores all pre-configured Logger objects.
Definition: logging.cc:40
FormattingHelper< T > format(const T &value, const char *unit, int width=-1, int precision=-1)
Acts as a stream modifier for std::ostream to output an object with an optional suffix string and wit...
Definition: logging.h:217
OscarOutput(const std::filesystem::path &path, const std::string &name, const std::vector< std::string > quantities={})
Create oscar output.
std::unique_ptr< OutputInterface > create_oscar_output(const std::string &format, const std::string &content, const std::filesystem::path &path, const OutputParameters &out_par)
Definition: oscaroutput.cc:889
@ OscarParticlesAtEventend
store the state at the end of each event (at_eventend)
Definition: oscaroutput.h:53
@ OscarParticlesAtEventendIfNotEmpty
store the state at the end of each event if it is not empty (at_eventend)
Definition: oscaroutput.h:55
@ OscarAtEventstart
store the state at the start of each event (at_eventstart)
Definition: oscaroutput.h:51
@ OscarInteractions
store interaction information (write_interaction)
Definition: oscaroutput.h:47
@ OscarParticlesIC
store the particles that are removed on the hypersurface
Definition: oscaroutput.h:57
@ OscarTimesteps
store the state after N timesteps (after_Nth_timestep)
Definition: oscaroutput.h:49
@ OscarFormat1999
Definition: oscaroutput.h:34
@ OscarFormat2013Extended
Definition: oscaroutput.h:33
@ ASCII
Definition: oscaroutput.h:35
@ OscarFormat2013
Definition: oscaroutput.h:32
std::unique_ptr< OutputInterface > create_select_format(bool modern_format, const std::filesystem::path &path, const OutputParameters &out_par, const std::string &name, const bool custom_format=false)
Helper function that creates the oscar output with the format selected by create_oscar_output (except...
Definition: oscaroutput.cc:861
constexpr int p
Proton.
Definition: action.h:24
static constexpr int LHyperSurfaceCrossing
Definition: binaryoutput.cc:22
static constexpr int LOutput
Structure to contain custom data for output.
bool empty_event
True if no collisions happened.
bool impose_kinematic_cut_for_SMASH_IC
Whether or not kinematic cuts are employed for SMASH IC.
double impact_parameter
Impact parameter for collider modus, otherwise dummy.
Structure to contain information about the event and ensemble numbers.
int32_t ensemble_number
The number of the ensemble.
int32_t event_number
The number of the event.
Struct that holds quantities required by default output standards.
Helper structure for Experiment to hold output options and parameters.
bool dil_extended
Extended format for dilepton output.
bool coll_extended
Extended format for collisions output.
bool part_extended
Extended format for particles output.
bool photons_extended
Extended format for photon output.
std::map< std::string, std::vector< std::string > > quantities
Map of quantities to be printed in the output.
OutputOnlyFinal part_only_final
Print only final particles in event.
bool ic_extended
Extended initial conditions output.
bool coll_printstartend
Print initial and final particles in event into collision output.