Version: SMASH-3.2
binaryoutput.cc
Go to the documentation of this file.
1 /*
2  *
3  * Copyright (c) 2014-2020,2022-2024
4  * SMASH Team
5  *
6  * GNU General Public License (GPLv3 or later)
7  *
8  */
9 
10 #include "smash/binaryoutput.h"
11 
12 #include <cstdint>
13 #include <filesystem>
14 #include <string>
15 
16 #include "smash/action.h"
17 #include "smash/clock.h"
18 #include "smash/config.h"
19 
20 namespace smash {
21 
22 static constexpr int LHyperSurfaceCrossing = LogArea::HyperSurfaceCrossing::id;
23 
24 static auto get_list_of_binary_quantities(const std::string &content,
25  const std::string &format,
26  const OutputParameters &parameters);
27 
28 static auto get_binary_filename(const std::string &content,
29  const std::vector<std::string> &quantities) {
30  std::string filename = content;
31  if (content == "Particles" || content == "Collisions") {
32  std::transform(filename.begin(), filename.end(), filename.begin(),
33  [](unsigned char c) { return std::tolower(c); });
34  if (quantities == OutputDefaultQuantities::oscar2013) {
35  filename += "_oscar2013";
36  } else if (quantities == OutputDefaultQuantities::oscar2013extended) {
37  filename += "_oscar2013_extended";
38  } else {
39  filename += "_binary";
40  }
41  } else if (content == "Photons" || content == "Dileptons") {
42  // Nothing to be done here
43  } else if (content == "Initial_Conditions") {
44  filename = "SMASH_IC";
45  } else {
46  throw std::invalid_argument(
47  "Unknown content to get the binary output filename.");
48  }
49  return filename + ".bin";
50 }
51 
204 BinaryOutputBase::BinaryOutputBase(const std::filesystem::path &path,
205  const std::string &mode,
206  const std::string &name,
207  const std::vector<std::string> &quantities)
208  : OutputInterface(name), file_{path, mode}, formatter_(quantities) {
209  if (quantities.empty()) {
210  throw std::invalid_argument(
211  "Empty quantities list passed to 'BinaryOutputBase' cconstructor.");
212  }
213  std::fwrite("SMSH", 4, 1, file_.get()); // magic number
214  write(format_version_); // file format version number
215  std::uint16_t format_variant{};
216  if (quantities == OutputDefaultQuantities::oscar2013) {
217  format_variant = 0;
218  } else if (quantities == OutputDefaultQuantities::oscar2013extended) {
219  format_variant = 1;
220  } else {
221  format_variant = format_custom_;
222  }
223  write(format_variant);
224  write(SMASH_VERSION);
225 }
226 
227 // write functions:
228 void BinaryOutputBase::write(const char c) {
229  std::fwrite(&c, sizeof(char), 1, file_.get());
230 }
232  std::fwrite(chunk.data(), sizeof(char), chunk.size(), file_.get());
233 }
234 
235 void BinaryOutputBase::write(const std::string &s) {
236  const auto size = smash::numeric_cast<uint32_t>(s.size());
237  std::fwrite(&size, sizeof(std::uint32_t), 1, file_.get());
238  std::fwrite(s.c_str(), s.size(), 1, file_.get());
239 }
240 
241 void BinaryOutputBase::write(const double x) {
242  std::fwrite(&x, sizeof(x), 1, file_.get());
243 }
244 
246  std::fwrite(v.begin(), sizeof(*v.begin()), 4, file_.get());
247 }
248 
249 void BinaryOutputBase::write(const Particles &particles) {
250  for (const auto &p : particles) {
252  }
253 }
254 
255 void BinaryOutputBase::write(const ParticleList &particles) {
256  for (const auto &p : particles) {
258  }
259 }
260 
262  write(formatter_.binary_chunk(p));
263 }
264 
266  const std::filesystem::path &path, std::string name,
267  const OutputParameters &out_par, const std::vector<std::string> &quantities)
268  : BinaryOutputBase(path / get_binary_filename(name, quantities), "wb", name,
269  quantities),
270  print_start_end_(out_par.coll_printstartend) {}
271 
273  const EventLabel &event_label,
274  const EventInfo &) {
275  const char pchar = 'p';
276  if (print_start_end_) {
277  std::fwrite(&pchar, sizeof(char), 1, file_.get());
278  write(event_label.event_number);
279  write(event_label.ensemble_number);
280  write(particles.size());
281  write(particles);
282  }
283 }
284 
286  const EventLabel &event_label,
287  const EventInfo &event) {
288  const char pchar = 'p';
289  if (print_start_end_) {
290  std::fwrite(&pchar, sizeof(char), 1, file_.get());
291  write(event_label.event_number);
292  write(event_label.ensemble_number);
293  write(particles.size());
294  write(particles);
295  }
296 
297  // Event end line
298  const char fchar = 'f';
299  std::fwrite(&fchar, sizeof(char), 1, file_.get());
300  write(event_label.event_number);
301  write(event_label.ensemble_number);
302  write(event.impact_parameter);
303  const char empty = event.empty_event;
304  write(empty);
305 
306  // Flush to disk
307  std::fflush(file_.get());
308 }
309 
311  const double density) {
312  const char ichar = 'i';
313  std::fwrite(&ichar, sizeof(char), 1, file_.get());
314  write(action.incoming_particles().size());
315  write(action.outgoing_particles().size());
316  std::fwrite(&density, sizeof(double), 1, file_.get());
317  const double weight = action.get_total_weight();
318  std::fwrite(&weight, sizeof(double), 1, file_.get());
319  const double partial_weight = action.get_partial_weight();
320  std::fwrite(&partial_weight, sizeof(double), 1, file_.get());
321  const auto type = static_cast<uint32_t>(action.get_type());
322  std::fwrite(&type, sizeof(uint32_t), 1, file_.get());
323  write(action.incoming_particles());
324  write(action.outgoing_particles());
325 }
326 
328  const std::filesystem::path &path, std::string name,
329  const OutputParameters &out_par, const std::vector<std::string> &quantities)
330  : BinaryOutputBase(path / get_binary_filename(name, quantities), "wb", name,
331  quantities),
332  only_final_(out_par.part_only_final) {}
333 
335  const EventLabel &event_label,
336  const EventInfo &) {
337  const char pchar = 'p';
339  std::fwrite(&pchar, sizeof(char), 1, file_.get());
340  write(event_label.event_number);
341  write(event_label.ensemble_number);
342  write(particles.size());
343  write(particles);
344  }
345 }
346 
348  const EventLabel &event_label,
349  const EventInfo &event) {
350  const char pchar = 'p';
352  std::fwrite(&pchar, sizeof(char), 1, file_.get());
353  write(event_label.event_number);
354  write(event_label.ensemble_number);
355  write(particles.size());
356  write(particles);
357  }
358 
359  // Event end line
360  const char fchar = 'f';
361  std::fwrite(&fchar, sizeof(char), 1, file_.get());
362  write(event_label.event_number);
363  write(event_label.ensemble_number);
364  write(event.impact_parameter);
365  const char empty = event.empty_event;
366  write(empty);
367 
368  // Flush to disk
369  std::fflush(file_.get());
370 }
371 
373  const std::unique_ptr<Clock> &,
374  const DensityParameters &,
375  const EventLabel &event_label,
376  const EventInfo &) {
377  const char pchar = 'p';
379  std::fwrite(&pchar, sizeof(char), 1, file_.get());
380  write(event_label.event_number);
381  write(event_label.ensemble_number);
382  write(particles.size());
383  write(particles);
384  }
385 }
386 
388  const std::filesystem::path &path, std::string name,
389  const std::vector<std::string> &quantities)
390  : BinaryOutputBase(path / get_binary_filename(name, quantities), "wb", name,
391  quantities) {}
392 
394  const EventLabel &,
395  const EventInfo &) {}
396 
398  const EventLabel &event_label,
399  const EventInfo &event) {
400  // Event end line
401  const char fchar = 'f';
402  std::fwrite(&fchar, sizeof(char), 1, file_.get());
403  write(event_label.event_number);
404  write(event_label.ensemble_number);
405  write(event.impact_parameter);
406  const char empty = event.empty_event;
407  write(empty);
408 
409  // Flush to disk
410  std::fflush(file_.get());
411 
412  // If the runtime is too short some particles might not yet have
413  // reached the hypersurface. Warning is printed.
414  if (particles.size() != 0 && !event.impose_kinematic_cut_for_SMASH_IC) {
416  "End time might be too small for initial conditions output. "
417  "Hypersurface has not yet been crossed by ",
418  particles.size(), " particle(s).");
419  }
420 }
421 
423  const double) {
425  const char pchar = 'p';
426  std::fwrite(&pchar, sizeof(char), 1, file_.get());
427  write(action.incoming_particles().size());
428  write(action.incoming_particles());
429  }
430 }
431 
432 static auto get_list_of_binary_quantities(const std::string &content,
433  const std::string &format,
434  const OutputParameters &parameters) {
435  const bool is_extended = std::invoke([&content, &parameters]() {
436  if (content == "Particles")
437  return parameters.part_extended;
438  else if (content == "Collisions")
439  return parameters.coll_extended;
440  else if (content == "Dileptons")
441  return parameters.dil_extended;
442  else if (content == "Photons")
443  return parameters.photons_extended;
444  else if (content == "Initial_Conditions")
445  return parameters.ic_extended;
446  else
447  return false;
448  });
449  const auto default_quantities =
452  if (format == "Oscar2013_bin" || content == "Dileptons" ||
453  content == "Photons" || content == "Initial_Conditions") {
454  return default_quantities;
455  } else if (format == "Binary") {
456  if (content == "Particles" || content == "Collisions") {
457  auto list_of_quantities = parameters.quantities.at(content);
458  if (list_of_quantities.empty()) {
459  return default_quantities;
460  } else {
461  return list_of_quantities;
462  }
463  } else {
464  /* Note that this function should not be called with "Binary" format for
465  * output contents which do not support custom binary quantities. Hence we
466  * throw here to prevent such a case.*/
467  throw std::invalid_argument(
468  "Unknown content to get the list of quantities for binary output.");
469  }
470  } else {
471  throw std::invalid_argument(
472  "Unknown format to get the list of quantities for binary output.");
473  }
474 }
475 
476 std::unique_ptr<OutputInterface> create_binary_output(
477  const std::string &format, const std::string &content,
478  const std::filesystem::path &path, const OutputParameters &out_par) {
479  if (content == "Particles") {
480  return std::make_unique<BinaryOutputParticles>(
481  path, content, out_par,
482  get_list_of_binary_quantities(content, format, out_par));
483  } else if (content == "Collisions" || content == "Dileptons" ||
484  content == "Photons") {
485  return std::make_unique<BinaryOutputCollisions>(
486  path, content, out_par,
487  get_list_of_binary_quantities(content, format, out_par));
488  } else if (content == "Initial_Conditions") {
489  return std::make_unique<BinaryOutputInitialConditions>(
490  path, content, get_list_of_binary_quantities(content, format, out_par));
491  } else {
492  throw std::invalid_argument("Binary output not available for '" + content +
493  "' content.");
494  }
495 }
496 
497 } // 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
Base class for SMASH binary output.
Definition: binaryoutput.h:30
void write(const ToBinary::type &chunk)
Write several bytes to the binary output.
void write_particledata(const ParticleData &p)
Write particle data to binary output.
const uint16_t format_version_
Binary file format version number.
Definition: binaryoutput.h:131
const uint16_t format_custom_
Format variant number associated to the custom quantities case.
Definition: binaryoutput.h:133
OutputFormatter< ToBinary > formatter_
The output formatter.
Definition: binaryoutput.h:135
RenamingFilePtr file_
Binary particles output file path.
Definition: binaryoutput.h:127
BinaryOutputBase(const std::filesystem::path &path, const std::string &mode, const std::string &name, const std::vector< std::string > &quantities)
Create binary output base.
void at_interaction(const Action &action, const double density) override
Writes an interaction block, including information about the incoming and outgoing particles,...
void at_eventstart(const Particles &particles, const EventLabel &event_label, const EventInfo &event) override
Writes the initial particle information list of an event to the binary output.
void at_eventend(const Particles &particles, const EventLabel &event_label, const EventInfo &event) override
Writes the final particle information list of an event to the binary output.
BinaryOutputCollisions(const std::filesystem::path &path, std::string name, const OutputParameters &out_par, const std::vector< std::string > &quantities)
Create binary particle output.
bool print_start_end_
Write initial and final particles additonally to collisions?
Definition: binaryoutput.h:194
void at_eventend(const Particles &particles, const EventLabel &event_label, const EventInfo &event) override
Writes the final particle information of an event to the binary output.
BinaryOutputInitialConditions(const std::filesystem::path &path, std::string name, const std::vector< std::string > &quantities)
Create binary initial conditions particle output.
void at_interaction(const Action &action, const double) override
Writes particles that are removed when crossing the hypersurface to the output.
void at_eventstart(const Particles &, const EventLabel &, const EventInfo &) override
Writes the initial particle information of an event to the binary output.
void at_eventstart(const Particles &particles, const EventLabel &event_label, const EventInfo &event) override
Writes the initial particle information of an event to the binary output.
void at_intermediate_time(const Particles &particles, const std::unique_ptr< Clock > &clock, const DensityParameters &dens_param, const EventLabel &event_label, const EventInfo &event) override
Writes particles at each time interval; fixed by option OUTPUT_INTERVAL.
OutputOnlyFinal only_final_
Whether final- or initial-state particles should be written.
Definition: binaryoutput.h:260
BinaryOutputParticles(const std::filesystem::path &path, std::string name, const OutputParameters &out_par, const std::vector< std::string > &quantities)
Create binary particle output.
void at_eventend(const Particles &particles, const EventLabel &event_label, const EventInfo &event) override
Writes the final particle information of an event to the binary output.
A class to pre-calculate and store parameters relevant for density calculation.
Definition: density.h:92
The FourVector class holds relevant values in Minkowski spacetime with (+, −, −, −) metric signature.
Definition: fourvector.h:33
iterator begin()
Definition: fourvector.h:291
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
FILE * get()
Get the underlying FILE* pointer.
Definition: file.cc:27
std::vector< char > type
Return type of this converter.
@ IfNotEmpty
Print only final-state particles, and those only if the event is not empty.
@ No
Print initial, intermediate and 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
std::unique_ptr< OutputInterface > create_binary_output(const std::string &format, const std::string &content, const std::filesystem::path &path, const OutputParameters &out_par)
Create a binary output object.
constexpr int p
Proton.
Definition: action.h:24
static constexpr int LHyperSurfaceCrossing
Definition: binaryoutput.cc:22
@ HyperSurfaceCrossing
See here for a short description.
static auto get_list_of_binary_quantities(const std::string &content, const std::string &format, const OutputParameters &parameters)
static auto get_binary_filename(const std::string &content, const std::vector< std::string > &quantities)
Definition: binaryoutput.cc:28
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.
static const std::vector< std::string > oscar2013
Quantities output in OSCAR2013 format.
static const std::vector< std::string > oscar2013extended
Quantities output in Extended OSCAR2013 format.
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.
bool ic_extended
Extended initial conditions output.