23 const std::filesystem::path &output_path) {
24 if (!std::filesystem::exists(output_path)) {
26 output_path.string() +
32 logg[
LExperiment].debug() <<
"Modus for this calculation: " << modus_chooser;
34 if (modus_chooser ==
"Box") {
35 return std::make_unique<Experiment<BoxModus>>(config, output_path);
36 }
else if (modus_chooser ==
"List") {
37 return std::make_unique<Experiment<ListModus>>(config, output_path);
38 }
else if (modus_chooser ==
"ListBox") {
39 return std::make_unique<Experiment<ListBoxModus>>(config, output_path);
40 }
else if (modus_chooser ==
"Collider") {
41 return std::make_unique<Experiment<ColliderModus>>(config, output_path);
42 }
else if (modus_chooser ==
"Sphere") {
43 return std::make_unique<Experiment<SphereModus>>(config, output_path);
46 ") requested from ExperimentBase::create.");
137 throw std::invalid_argument(
"Testparticle number should be positive!");
141 const bool only_participants =
145 throw std::invalid_argument(
146 "Only_Participants option cannot be "
147 "set to True when using Potentials.");
154 double box_length = -1.0;
167 throw std::invalid_argument(
"Delta_Time cannot be zero or negative.");
172 throw std::invalid_argument(
"End_Time cannot be zero or negative.");
176 if (box_length > 0.0 && dt > box_length / 10.0) {
177 throw std::invalid_argument(
178 "Please decrease the timestep size. "
179 "A value of (dt <= l_box / 10) is necessary in the box modus.");
183 std::unique_ptr<Clock> output_clock =
nullptr;
186 throw std::invalid_argument(
187 "Please specify either Output_Interval or Output_Times");
189 std::vector<double> output_times =
193 output_times.push_back(t_end + 1.);
194 output_clock = std::make_unique<CustomClock>(output_times);
196 const double output_dt =
198 if (output_dt <= 0.) {
199 throw std::invalid_argument(
200 "Output_Interval cannot be zero or negative.");
202 output_clock = std::make_unique<UniformClock>(0.0, output_dt, t_end);
209 throw std::invalid_argument(
210 "Photon output is enabled although photon production is disabled. "
211 "Photon production can be configured in the \"Photon\" subsection "
212 "of the \"Collision_Term\".");
217 const bool missing_output_2to2 =
219 missing_output_brems =
221 if (missing_output_2to2 || missing_output_brems) {
222 throw std::invalid_argument(
223 "Photon output is disabled although photon production is enabled. "
224 "Please enable the photon output.");
232 throw std::invalid_argument(
233 "Dilepton output is enabled although dilepton production is disabled. "
234 "Dilepton production can be configured in the \"Dileptons\" subsection "
235 "of the \"Collision_Term\".");
240 const bool missing_output_decays =
242 if (missing_output_decays) {
243 throw std::invalid_argument(
244 "Dilepton output is disabled although dilepton production is "
246 "Please enable the dilepton output.");
251 const double low_snn_cut =
255 if (proton && pion &&
256 low_snn_cut > proton->mass() + proton->mass() + pion->mass()) {
257 logg[
LExperiment].warn(
"The cut-off should be below the threshold energy",
258 " of the process: NN to NNpi");
260 const bool potential_affect_threshold =
268 throw std::invalid_argument(
269 "Only use a fixed minimal cell length with the stochastic collision "
274 throw std::invalid_argument(
275 "Only use maximum cross section with the geometric collision "
276 "criterion. Use Fixed_Min_Cell_Length to change the grid size for the "
277 "stochastic criterion.");
288 const double maximum_cross_section_default =
291 double maximum_cross_section = config.
take(
293 maximum_cross_section *= scale_xs;
294 return {std::make_unique<UniformClock>(0.0, dt, t_end),
295 std::move(output_clock),
316 potential_affect_threshold,
318 maximum_cross_section,
328 uint64_t scatterings_this_interval,
332 double E_mean_field_initial) {
333 const SystemTimeSpan elapsed_seconds = SystemClock::now() - time_start;
336 const QuantumNumbers difference = current_values - conserved_initial;
337 int total_particles = 0;
338 for (
const Particles &particles : ensembles) {
339 total_particles += particles.size();
344 const double current_energy = current_values.
momentum().
x0();
345 const double energy_per_part =
346 (total_particles > 0) ? (current_energy + E_mean_field) / total_particles
349 std::ostringstream ss;
351 ss << field<7, 3> << time
353 << field<11, 3> << current_energy
355 << field<11, 3> << E_mean_field
357 << field<12, 3> << current_energy + E_mean_field
359 << field<12, 6> << energy_per_part;
361 if (total_particles == 0) {
362 ss << field<13, 6> <<
"N/A";
364 ss << field<13, 6> << (difference.
momentum().
x0()
365 + E_mean_field - E_mean_field_initial)
368 ss << field<14, 3> << scatterings_this_interval
369 << field<10, 3> << total_particles
370 << field<9, 3> << elapsed_seconds;
381 const double V_cell = (jmuB_lat.
cell_sizes())[0] *
384 double E_mean_field = 0.0;
385 double density_mean = 0.0;
386 double density_variance = 0.0;
401 <<
"\nSymmetry energy is not included in the mean field calculation."
417 double C1GeV = (potentials.
skyrme_a()) / 1000.0;
418 double C2GeV = (potentials.
skyrme_b()) / 1000.0;
427 int number_of_nodes = 0;
428 double lattice_mean_field_total = 0.0;
430 for (
auto &node : jmuB_lat) {
433 double rhoB = node.rho();
435 const double j0B = node.jmu_net().x0();
437 const double abs_rhoB = std::abs(rhoB);
442 density_variance += j0B * j0B;
452 double mean_field_contribution_1 = (C1GeV / b1) * std::pow(abs_rhoB, b1) /
454 double mean_field_contribution_2 = (C2GeV / b2) * std::pow(abs_rhoB, b2) /
457 lattice_mean_field_total +=
458 V_cell * (mean_field_contribution_1 + mean_field_contribution_2);
462 density_mean = density_mean / number_of_nodes;
463 density_variance = density_variance / number_of_nodes;
464 double density_scaled_variance =
465 std::sqrt(density_variance - density_mean * density_mean) /
469 <<
"\n\t\t\t\t\t density mean = " << density_mean;
471 <<
"\n\t\t\t\t\t density scaled variance = " << density_scaled_variance;
473 <<
"\n\t\t\t\t\t total mean_field = "
478 E_mean_field = lattice_mean_field_total;
490 <<
"\nSymmetry energy is not included in the VDF mean-field "
492 <<
"\nas VDF potentials haven't been fitted with symmetry energy."
512 int number_of_nodes = 0;
513 double lattice_mean_field_total = 0.0;
515 for (
auto &node : jmuB_lat) {
518 double rhoB = node.rho();
520 const double j0B = node.jmu_net().x0();
521 double abs_rhoB = std::abs(rhoB);
523 density_variance += j0B * j0B;
534 double mean_field_contribution = 0.0;
536 mean_field_contribution +=
538 std::pow(abs_rhoB, potentials.
powers()[i] - 2.0) *
540 ((potentials.
powers()[i] - 1.0) / potentials.
powers()[i]) *
541 abs_rhoB * abs_rhoB) /
542 std::pow(rhoB_0, potentials.
powers()[i] - 1.0);
544 lattice_mean_field_total += V_cell * mean_field_contribution;
548 density_mean = density_mean / number_of_nodes;
549 density_variance = density_variance / number_of_nodes;
550 double density_scaled_variance =
551 std::sqrt(density_variance - density_mean * density_mean) /
555 <<
"\n\t\t\t\t\t density mean = " << density_mean;
557 <<
"\n\t\t\t\t\t density scaled variance = " << density_scaled_variance;
559 <<
"\n\t\t\t\t\t total mean_field = "
564 E_mean_field = lattice_mean_field_total;
567 double electromagnetic_potential = 0.0;
571 double V_cell_em = em_lattice->cell_sizes()[0] *
572 em_lattice->cell_sizes()[1] *
573 em_lattice->cell_sizes()[2];
574 for (
auto &fields : *em_lattice) {
576 electromagnetic_potential +=
577 hbarc * 0.5 * V_cell_em * (fields.first.sqr() + fields.second.sqr());
580 logg[
LExperiment].debug() <<
"Total energy in electromagnetic field = "
581 << electromagnetic_potential;
582 E_mean_field += electromagnetic_potential;
596 double E_mean_field,
double modus_impact_parameter,
598 bool projectile_target_interact,
599 bool kinematic_cut_for_SMASH_IC) {
601 const double E_kinetic_total = current_values.
momentum().
x0();
602 const double E_total = E_kinetic_total + E_mean_field;
604 EventInfo event_info{modus_impact_parameter,
612 !projectile_target_interact,
613 kinematic_cut_for_SMASH_IC};
618 static bool warn_mass_discrepancy =
true;
619 static bool warn_off_shell_particle =
true;
620 for (
auto it = particle_list.begin(); it != particle_list.end();) {
621 auto &particle = *it;
622 auto pdgcode = particle.pdgcode();
625 if (pdgcode == 0x310 || pdgcode == 0x130) {
635 auto valid_smash_particle =
637 pdgcode, particle.effective_mass(), particle.position(),
638 particle.momentum(),
LExperiment, warn_mass_discrepancy,
639 warn_off_shell_particle);
640 particle.set_4position(valid_smash_particle.position());
641 particle.set_4momentum(valid_smash_particle.momentum());
642 particle.set_cross_section_scaling_factor(
643 valid_smash_particle.xsec_scaling_factor());
647 <<
"SMASH does not recognize pdg code " << pdgcode
648 <<
" obtained from hadron list. This particle will be ignored.\n";
649 it = particle_list.erase(it);
656 const std::string deprecated_message =
657 "Configuration key for initial conditions was provided twice and "
658 "inconsistently.\nSome parameters in the Initial_Conditions section of "
659 "Output are deprecated.\nPlease use the corresponding values in the "
660 "Initial_Conditions subsection under Collider.";
661 if (collider.has_value()) {
662 if (output != collider) {
664 " in configuration.");
665 throw std::invalid_argument(deprecated_message);
Interface to the SMASH configuration files.
T read(const Key< T > &key) const
Additional interface for SMASH to read configuration values without removing them.
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...
bool has_section(const KeyLabels &labels) const
Return whether there is a (possibly empty) section with the given labels.
T take(const Key< T > &key)
The default interface for SMASH to read configuration values.
void remove_all_entries_in_section_but_one(const std::string &key, KeyLabels section={})
Remove all entries in the given section except for key.
static std::unique_ptr< ExperimentBase > create(Configuration &config, const std::filesystem::path &output_path)
Factory method that creates and initializes a new Experiment<Modus>.
static const ParticleTypePtr try_find(PdgCode pdgcode)
Returns the ParticleTypePtr for the given pdgcode.
static bool exists(PdgCode pdgcode)
The Particles class abstracts the storage and manipulation of particles.
A class that stores parameters of potentials, calculates potentials and their gradients.
const std::vector< double > & powers() const
virtual bool use_symmetry() const
const std::vector< double > & coeffs() const
virtual bool use_skyrme() const
virtual bool use_coulomb() const
double skyrme_tau() const
int number_of_terms() const
virtual bool use_vdf() const
double saturation_density() const
A container for storing conserved values.
FourVector momentum() const
A container class to hold all the arrays on the lattice and access them.
const std::array< double, 3 > & cell_sizes() const
@ Stochastic
Stochastic Criteiron.
#define SMASH_SOURCE_LOCATION
Hackery that is required to output the location in the source code where the log statement occurs.
std::array< einhard::Logger<>, std::tuple_size< LogArea::AreaTuple >::value > logg
An array that stores all pre-configured Logger objects.
T uniform_int(T min, T max)
void validate_duplicate_IC_config(double, std::optional< double >, std::string)
The physics inputs for Initial Conditions are currently duplicated in both Output and Collider sectio...
static constexpr int LInitialConditions
EventInfo fill_event_info(const std::vector< Particles > &ensembles, double E_mean_field, double modus_impact_parameter, const ExperimentParameters ¶meters, bool projectile_target_interact, bool kinematic_cut_for_SMASH_IC)
Generate the EventInfo object which is passed to outputs_.
SystemClock::duration SystemTimeSpan
The time duration type (alias) used for measuring run times.
std::string format_measurements(const std::vector< Particles > &ensembles, uint64_t scatterings_this_interval, const QuantumNumbers &conserved_initial, SystemTimePoint time_start, double time, double E_mean_field, double E_mean_field_initial)
Generate a string which will be printed to the screen when SMASH is running.
ExperimentParameters create_experiment_parameters(Configuration &config)
Gathers all general Experiment parameters.
constexpr double very_small_double
A very small double, used to avoid division by zero.
double calculate_mean_field_energy(const Potentials &potentials, RectangularLattice< smash::DensityOnLattice > &jmu_B_lat, RectangularLattice< std::pair< ThreeVector, ThreeVector >> *em_lattice, const ExperimentParameters ¶meters)
Calculate the total mean field energy of the system; this will be printed to the screen when SMASH is...
static constexpr int LExperiment
void validate_and_adjust_particle_list(ParticleList &particle_list)
Validate a particle list adjusting each particle to be a valid SMASH particle.
constexpr double nuclear_density
Ground state density of symmetric nuclear matter [fm^-3].
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 hbarc
GeV <-> fm conversion factor.
std::chrono::time_point< std::chrono::system_clock > SystemTimePoint
Type (alias) that is used to store the current time.
Structure to contain custom data for output.
Exception class that is thrown if an invalid modus is requested from the Experiment factory.
Exception class that is thrown if the requested output path in the Experiment factory is not existing...
Helper structure for Experiment.
double box_length
Length of the box in fm in case of box modus, otherwise -1.
int n_ensembles
Number of parallel ensembles.
std::unique_ptr< Clock > outputclock
Output clock to keep track of the next output time.
int testparticles
Number of test-particles.