17 #include "smash/config.h"
23 template <OscarOutputFormat Format,
int Contents>
25 const std::filesystem::path &path,
const std::string &name,
26 const std::vector<std::string> quantities)
28 file_{path / (name + ((Format ==
ASCII) ?
".dat" :
".oscar") +
31 formatter_{Format ==
ASCII ? quantities
86 if (Format !=
ASCII && !quantities.empty()) {
87 throw std::logic_error(
88 "Non-empty Quantities given alongside format other than ASCII.");
90 std::string format_name;
91 if (Format ==
ASCII) {
92 format_name =
"ASCII";
94 format_name =
"OSCAR2013";
96 format_name =
"OSCAR2013Extended";
98 format_name =
"OSC1999A";
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);
107 const std::string &oscar_name =
108 name ==
"particle_lists" ?
"final_id_p_x" : name;
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");
119 std::fprintf(file_.get(),
"# nin nout event_number ensemble_number\n");
121 std::fprintf(file_.get(),
"# %s\n", formatter_.quantities_line().c_str());
124 "# End of event: 0 0 event_number ensemble_number impact_parameter\n");
125 std::fprintf(file_.get(),
"#\n");
129 template <OscarOutputFormat Format,
int Contents>
132 write_particledata(data);
136 template <OscarOutputFormat Format,
int Contents>
145 std::fprintf(file_.get(),
"# event %i ensemble %i in %zu\n",
152 const size_t zero = 0;
153 std::fprintf(file_.get(),
"%zu %zu %i %i\n", zero, particles.
size(),
160 std::fprintf(file_.get(),
"# event %i ensemble %i start\n",
163 const size_t zero = 0;
164 std::fprintf(file_.get(),
"%zu %zu %i %i\n", zero, zero,
170 template <OscarOutputFormat Format,
int Contents>
178 std::fprintf(file_.get(),
"# event %i ensemble %i out %zu\n",
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",
192 std::fprintf(file_.get(),
"# event %i ensemble %i end\n",
199 const size_t zero = 0;
202 std::fprintf(file_.get(),
"%zu %zu %i %i\n", particles.
size(), zero,
207 std::fprintf(file_.get(),
"%zu %zu %i %i %7.3f\n", zero, zero,
212 std::fflush(file_.get());
219 "End time might be too small for initial conditions output. "
220 "Hypersurface has not yet been crossed by ",
221 particles.
size(),
" particle(s).");
226 template <OscarOutputFormat Format,
int Contents>
228 const double density) {
232 std::fprintf(file_.get(),
233 "# interaction in %zu out %zu rho %12.7f weight %12.7g"
234 " partial %12.7f type %5i\n",
238 static_cast<int>(action.
get_type()));
246 std::fprintf(file_.get(),
"%zu %zu %12.7f %12.7f %12.7f %5i\n",
250 static_cast<int>(action.
get_type()));
253 write_particledata(
p);
256 write_particledata(
p);
260 write_particledata(
p);
265 template <OscarOutputFormat Format,
int Contents>
267 const Particles &particles,
const std::unique_ptr<Clock> &,
273 std::fprintf(file_.get(),
"# event %i ensemble %i out %zu\n",
277 const size_t zero = 0;
278 std::fprintf(file_.get(),
"%zu %zu %i %i\n", particles.
size(), zero,
841 template <OscarOutputFormat Format,
int Contents>
844 std::fprintf(file_.get(),
"%s\n", formatter_.data_line(data).c_str());
860 template <
int Contents>
862 bool modern_format,
const std::filesystem::path &path,
864 const bool custom_format =
false) {
871 return std::make_unique<OscarOutput<ASCII, Contents>>(path, name,
873 }
else if (modern_format && extended_format) {
874 return std::make_unique<OscarOutput<OscarFormat2013Extended, Contents>>(
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);
883 <<
"There is no extended Oscar1999 format.";
884 return std::make_unique<OscarOutput<OscarFormat1999, Contents>>(path, name);
890 const std::string &
format,
const std::string &content,
893 throw std::invalid_argument(
"Creating Oscar output: unknown format");
895 const bool modern_format = (
format ==
"Oscar2013");
896 const bool custom_format = (
format ==
"ASCII");
897 if (content ==
"Particles") {
899 return create_select_format<OscarParticlesAtEventend>(
900 modern_format, path, out_par,
"particle_lists", custom_format);
902 return create_select_format<OscarParticlesAtEventendIfNotEmpty>(
903 modern_format, path, out_par,
"particle_lists", custom_format);
908 modern_format, path, out_par,
"particle_lists", custom_format);
910 }
else if (content ==
"Collisions") {
914 modern_format, path, out_par,
"full_event_history", custom_format);
916 return create_select_format<OscarInteractions>(
917 modern_format, path, out_par,
"full_event_history", custom_format);
919 }
else if (content ==
"Dileptons") {
921 return std::make_unique<
925 return std::make_unique<OscarOutput<OscarFormat2013, OscarInteractions>>(
928 return std::make_unique<OscarOutput<OscarFormat1999, OscarInteractions>>(
930 }
else if (!custom_format && !modern_format && out_par.
dil_extended) {
932 <<
"Creating Oscar output: "
933 <<
"There is no extended Oscar1999 (dileptons) format.";
935 }
else if (content ==
"Photons") {
937 return std::make_unique<OscarOutput<OscarFormat2013, OscarInteractions>>(
940 return std::make_unique<
944 return std::make_unique<OscarOutput<OscarFormat1999, OscarInteractions>>(
948 <<
"Creating Oscar output: "
949 <<
"There is no extended Oscar1999 (photons) format.";
951 }
else if (content ==
"Initial_Conditions") {
953 return std::make_unique<
960 }
else if (!modern_format && !out_par.
ic_extended) {
961 return std::make_unique<
964 }
else if (!custom_format && !modern_format && out_par.
ic_extended) {
966 <<
"Creating Oscar output: "
967 <<
"There is no extended Oscar1999 (initial conditions) format.";
971 throw std::invalid_argument(
"Create_oscar_output got unknown content.");
Action is the base class for a generic process that takes a number of incoming particles and transfor...
virtual ProcessType get_type() const
Get the process type.
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.
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.
A class to pre-calculate and store parameters relevant for density calculation.
Abstraction of generic output.
ParticleData contains the dynamic information of a certain particle.
The Particles class abstracts the storage and manipulation of particles.
@ 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.
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...
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)
@ OscarParticlesAtEventend
store the state at the end of each event (at_eventend)
@ OscarParticlesAtEventendIfNotEmpty
store the state at the end of each event if it is not empty (at_eventend)
@ OscarAtEventstart
store the state at the start of each event (at_eventstart)
@ OscarInteractions
store interaction information (write_interaction)
@ OscarParticlesIC
store the particles that are removed on the hypersurface
@ OscarTimesteps
store the state after N timesteps (after_Nth_timestep)
@ OscarFormat2013Extended
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...
static constexpr int LHyperSurfaceCrossing
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.