7 #ifndef SRC_INCLUDE_SMASH_EXPERIMENT_H_
8 #define SRC_INCLUDE_SMASH_EXPERIMENT_H_
45 #ifdef SMASH_USE_HEPMC
48 #ifdef SMASH_USE_RIVET
73 template <
typename T,
typename Ratio>
75 const chrono::duration<T, Ratio> &seconds) {
76 using Seconds = chrono::duration<double>;
77 using Minutes = chrono::duration<double, std::ratio<60>>;
78 using Hours = chrono::duration<double, std::ratio<60 * 60>>;
79 constexpr Minutes threshold_for_minutes{10};
80 constexpr Hours threshold_for_hours{3};
81 if (seconds < threshold_for_minutes) {
82 return out << Seconds(seconds).count() <<
" [s]";
84 if (seconds < threshold_for_hours) {
85 return out << Minutes(seconds).count() <<
" [min]";
87 return out << Hours(seconds).count() <<
" [h]";
92 static constexpr
int LMain = LogArea::Main::id;
131 static std::unique_ptr<ExperimentBase>
create(
132 Configuration &config,
const std::filesystem::path &output_path);
148 using std::invalid_argument::invalid_argument;
157 using std::invalid_argument::invalid_argument;
161 template <
typename Modus>
163 template <
typename Modus>
190 template <
typename Modus>
217 const std::filesystem::path &output_path);
244 ParticleList &&remove_plist = {});
290 bool include_pauli_blocking =
true);
300 const std::filesystem::path &output_path,
325 const double end_time_propagation);
465 std::unique_ptr<RectangularLattice<FourVector>>
UB_lat_ =
nullptr;
471 std::unique_ptr<RectangularLattice<FourVector>>
UI3_lat_ =
nullptr;
477 std::unique_ptr<RectangularLattice<std::pair<ThreeVector, ThreeVector>>>
481 std::unique_ptr<RectangularLattice<std::pair<ThreeVector, ThreeVector>>>
485 std::unique_ptr<RectangularLattice<std::pair<ThreeVector, ThreeVector>>>
489 std::unique_ptr<RectangularLattice<EnergyMomentumTensor>>
Tmn_;
496 std::unique_ptr<RectangularLattice<std::array<FourVector, 4>>>
504 std::unique_ptr<RectangularLattice<std::array<FourVector, 4>>>
717 friend std::ostream &operator<<<>(std::ostream &out,
const Experiment &e);
721 template <
typename Modus>
723 out <<
"End time: " << e.
end_time_ <<
" fm\n";
739 template <
typename Modus>
741 const std::string &content,
742 const std::filesystem::path &output_path,
745 if (ensembles_.size() > 1) {
746 auto abort_because_of = [](
const std::string &s) {
747 throw std::invalid_argument(
748 s +
" output is not available with multiple parallel ensembles.");
750 if (content ==
"Initial_Conditions") {
751 abort_because_of(
"Initial_Conditions");
753 if ((
format ==
"HepMC") || (
format ==
"HepMC_asciiv3") ||
754 (
format ==
"HepMC_treeroot")) {
755 abort_because_of(
"HepMC");
757 if (content ==
"Rivet") {
758 abort_because_of(
"Rivet");
760 if (content ==
"Collisions") {
762 "Information coming from different ensembles in 'Collisions' output "
763 "is not distinguishable.\nSuch an output with multiple parallel "
764 "ensembles should only be used if later in the data analysis\nit is "
765 "not necessary to trace back which data belongs to which ensemble.");
769 if (
format ==
"VTK" && content ==
"Particles") {
770 outputs_.emplace_back(
771 std::make_unique<VtkOutput>(output_path, content, out_par));
772 }
else if (
format ==
"Root") {
773 #ifdef SMASH_USE_ROOT
774 if (content ==
"Initial_Conditions") {
775 outputs_.emplace_back(
776 std::make_unique<RootOutput>(output_path,
"SMASH_IC", out_par));
778 outputs_.emplace_back(
779 std::make_unique<RootOutput>(output_path, content, out_par));
783 "Root output requested, but Root support not compiled in");
785 }
else if (
format ==
"Binary" &&
786 (content ==
"Collisions" || content ==
"Dileptons" ||
787 content ==
"Photons" || content ==
"Particles" ||
788 content ==
"Initial_Conditions")) {
789 outputs_.emplace_back(
791 }
else if (
format ==
"Oscar2013_bin" &&
792 (content ==
"Collisions" || content ==
"Particles")) {
793 outputs_.emplace_back(
795 }
else if (
format ==
"Oscar1999" ||
format ==
"Oscar2013") {
796 outputs_.emplace_back(
798 }
else if (
format ==
"ASCII" &&
799 (content ==
"Particles" || content ==
"Collisions")) {
800 outputs_.emplace_back(
802 }
else if (content ==
"Thermodynamics" &&
format ==
"ASCII") {
803 outputs_.emplace_back(
804 std::make_unique<ThermodynamicOutput>(output_path, content, out_par));
805 }
else if (content ==
"Thermodynamics" &&
806 (
format ==
"Lattice_ASCII" ||
format ==
"Lattice_Binary")) {
807 printout_full_lattice_any_td_ =
true;
808 outputs_.emplace_back(std::make_unique<ThermodynamicLatticeOutput>(
809 output_path, content, out_par,
format ==
"Lattice_ASCII",
810 format ==
"Lattice_Binary"));
811 }
else if (content ==
"Thermodynamics" &&
format ==
"VTK") {
812 printout_lattice_td_ =
true;
813 outputs_.emplace_back(
814 std::make_unique<VtkOutput>(output_path, content, out_par));
815 }
else if (content ==
"Initial_Conditions" &&
format ==
"ASCII") {
817 throw std::invalid_argument(
818 "Dynamic initial conditions are only available in Oscar2013 and "
821 outputs_.emplace_back(
822 std::make_unique<ICOutput>(output_path,
"SMASH_IC", out_par));
823 }
else if ((
format ==
"HepMC") || (
format ==
"HepMC_asciiv3") ||
824 (
format ==
"HepMC_treeroot")) {
825 #ifdef SMASH_USE_HEPMC
826 if (content ==
"Particles") {
827 if ((
format ==
"HepMC") || (
format ==
"HepMC_asciiv3")) {
828 outputs_.emplace_back(std::make_unique<HepMcOutput>(
829 output_path,
"SMASH_HepMC_particles",
false,
"asciiv3"));
830 }
else if (
format ==
"HepMC_treeroot") {
831 #ifdef SMASH_USE_HEPMC_ROOTIO
832 outputs_.emplace_back(std::make_unique<HepMcOutput>(
833 output_path,
"SMASH_HepMC_particles",
false,
"root"));
836 "Requested HepMC_treeroot output not available, "
837 "ROOT or HepMC3_ROOTIO missing or not found by cmake.");
840 }
else if (content ==
"Collisions") {
841 if ((
format ==
"HepMC") || (
format ==
"HepMC_asciiv3")) {
842 outputs_.emplace_back(std::make_unique<HepMcOutput>(
843 output_path,
"SMASH_HepMC_collisions",
true,
"asciiv3"));
844 }
else if (
format ==
"HepMC_treeroot") {
845 #ifdef SMASH_USE_HEPMC_ROOTIO
846 outputs_.emplace_back(std::make_unique<HepMcOutput>(
847 output_path,
"SMASH_HepMC_collisions",
true,
"root"));
850 "Requested HepMC_treeroot output not available, "
851 "ROOT or HepMC3_ROOTIO missing or not found by cmake.");
856 "HepMC only available for Particles and "
857 "Collisions content. Requested for " +
862 "HepMC output requested, but HepMC support not compiled in");
864 }
else if (content ==
"Coulomb" &&
format ==
"VTK") {
865 outputs_.emplace_back(
866 std::make_unique<VtkOutput>(output_path,
"Fields", out_par));
867 }
else if (content ==
"Rivet") {
868 #ifdef SMASH_USE_RIVET
870 static bool rivet_format_already_selected =
false;
872 if (rivet_format_already_selected) {
874 "Rivet output format can only be one, either YODA or YODA-full. "
875 "Only your first valid choice will be used.");
879 outputs_.emplace_back(std::make_unique<RivetOutput>(
881 rivet_format_already_selected =
true;
882 }
else if (
format ==
"YODA-full") {
883 outputs_.emplace_back(std::make_unique<RivetOutput>(
885 rivet_format_already_selected =
true;
888 "not one of YODA or YODA-full");
892 "Rivet output requested, but Rivet support not compiled in");
896 <<
"Unknown combination of format (" <<
format <<
") and content ("
897 << content <<
"). Fix the config.";
900 logg[
LExperiment].info() <<
"Added output " << content <<
" of format "
913 template <
typename Modus>
915 const std::filesystem::path &output_path)
918 modus_(std::invoke([&]() {
928 const bool restore_key = config.
has_value(key);
929 const bool temporary_taken_key = config.
take(key);
933 config.
set_value(key, temporary_taken_key);
935 return Modus{std::move(modus_config),
parameters_};
937 ensembles_(parameters_.n_ensembles),
939 delta_time_startup_(parameters_.labclock->timestep_duration()),
947 bremsstrahlung_switch_(
950 modus_.is_IC_for_hybrid()),
951 IC_dynamic_(IC_switch_ ? (modus_.IC_parameters().type ==
958 const bool user_wants_min_nonempty =
960 if (user_wants_nevents == user_wants_min_nonempty) {
961 throw std::invalid_argument(
962 "Please specify either Nevents or Minimum_Nonempty_Ensembles.");
964 if (user_wants_nevents) {
969 minimum_nonempty_ensembles_ =
973 max_events_ = numeric_cast<int>(std::ceil(
974 static_cast<double>(max_ensembles) / parameters_.n_ensembles));
980 throw std::invalid_argument(
981 "Covariant Gaussian derivatives only make sense for Covariant Gaussian "
989 parameters_.discrete_weight < (1. / 7.)) {
990 throw std::invalid_argument(
991 "The central weight for discrete smearing should be >= 1./7.");
996 throw std::invalid_argument(
997 "The stochastic criterion can only be employed for fixed time step "
998 "mode and with a grid!");
1002 throw std::invalid_argument(
1003 "The box modus can only be used with the fixed time step mode!");
1007 " testparticles per particle.");
1009 " parallel ensembles.");
1014 "To preserve detailed balance in a box simulation, it is recommended\n"
1015 "to use the bottom-up strategy for evaluating total cross sections.\n"
1016 "Consider adding the following line to the 'Collision_Term' section "
1017 "in your configuration file:\n"
1018 " Total_Cross_Section_Strategy: \"BottomUp\"");
1023 "To preserve detailed balance in a box simulation, it is recommended "
1024 "to not include the pseudoresonances,\nas they artificially increase "
1025 "the resonance production without changing the corresponding "
1026 "decay.\nConsider adding the following line to the 'Collision_Term' "
1027 "section in your configuration file:\n Pseudoresonance: \"None\"");
1031 if (IC_output != modus_.is_IC_for_hybrid()) {
1032 throw std::invalid_argument(
1033 "The 'Initial_Conditions' subsection must be present in both 'Output' "
1034 "and 'Modi: Collider' sections.");
1047 modus_.sqrt_s_NN() >= 200. ? -1. : 1.);
1050 if (dileptons_switch_) {
1051 dilepton_finder_ = std::make_unique<DecayActionsFinderDilepton>();
1053 if (photons_switch_ || bremsstrahlung_switch_) {
1054 n_fractional_photons_ =
1057 if (parameters_.two_to_one) {
1058 if (parameters_.res_lifetime_factor < 0.) {
1059 throw std::invalid_argument(
1060 "Resonance lifetime modifier cannot be negative!");
1064 "Resonance lifetime set to zero. Make sure resonances cannot "
1066 "inelastically (e.g. resonance chains), else SMASH is known to "
1069 action_finders_.emplace_back(std::make_unique<DecayActionsFinder>(
1070 parameters_.res_lifetime_factor, parameters_.do_non_strong_decays));
1073 if ((parameters_.two_to_one || parameters_.included_2to2.any() ||
1074 parameters_.included_multi.any() || parameters_.strings_switch) &&
1076 parameters_.use_monash_tune_default =
1077 (modus_.is_collider() && modus_.sqrt_s_NN() >= 200.);
1079 std::make_unique<ScatterActionsFinder>(config, parameters_);
1080 max_transverse_distance_sqr_ =
1081 scat_finder->max_transverse_distance_sqr(parameters_.testparticles);
1082 process_string_ptr_ = scat_finder->get_process_string_ptr();
1083 action_finders_.emplace_back(std::move(scat_finder));
1085 max_transverse_distance_sqr_ =
1086 parameters_.maximum_cross_section / M_PI *
fm2_mb;
1087 process_string_ptr_ = NULL;
1089 if (modus_.is_box()) {
1090 action_finders_.emplace_back(
1091 std::make_unique<WallCrossActionsFinder>(parameters_.box_length));
1095 const InitialConditionParameters &IC_parameters = modus_.IC_parameters();
1098 action_finders_.emplace_back(std::make_unique<DynamicFluidizationFinder>(
1099 modus_.fluid_lattice(), modus_.fluid_background(), IC_parameters));
1110 double rapidity_cut = IC_parameters.rapidity_cut.has_value()
1111 ? IC_parameters.rapidity_cut.value()
1120 if (rapidity_cut < 0.0) {
1122 <<
"Rapidity cut for initial conditions configured as abs(y) < "
1123 << rapidity_cut <<
" is unreasonable. \nPlease choose a positive, "
1124 <<
"non-zero value or employ SMASH without rapidity cut.";
1125 throw std::runtime_error(
1126 "Kinematic cut for initial conditions malconfigured.");
1129 if (modus_.calculation_frame_is_fixed_target() && rapidity_cut != 0.0) {
1130 throw std::runtime_error(
1131 "Rapidity cut for initial conditions output is not implemented "
1132 "in the fixed target calculation frame. \nPlease use "
1133 "\"center of velocity\" or \"center of mass\" as a "
1134 "\"Calculation_Frame\" instead.");
1138 IC_parameters.pT_cut.has_value() ? IC_parameters.pT_cut.value() : 0.0;
1145 <<
"transverse momentum cut for initial conditions configured as "
1146 <<
"pT < " << pT_cut <<
" is unreasonable. \nPlease choose a "
1147 <<
"positive, non-zero value or employ SMASH without pT cut.";
1148 throw std::runtime_error(
1149 "Kinematic cut for initial conditions misconfigured.");
1152 if (rapidity_cut > 0.0 || pT_cut > 0.0) {
1153 kinematic_cuts_for_IC_output_ =
true;
1156 if (rapidity_cut > 0.0 && pT_cut > 0.0) {
1158 <<
"Extracting initial conditions in kinematic range: "
1159 << -rapidity_cut <<
" <= y <= " << rapidity_cut
1160 <<
"; pT <= " << pT_cut <<
" GeV.";
1161 }
else if (rapidity_cut > 0.0) {
1163 <<
"Extracting initial conditions in kinematic range: "
1164 << -rapidity_cut <<
" <= y <= " << rapidity_cut <<
".";
1165 }
else if (pT_cut > 0.0) {
1167 <<
"Extracting initial conditions in kinematic range: pT <= "
1168 << pT_cut <<
" GeV.";
1171 <<
"Extracting initial conditions without kinematic cuts.";
1174 double proper_time = std::numeric_limits<double>::quiet_NaN();
1181 }
else if (IC_parameters.proper_time.has_value()) {
1182 proper_time = IC_parameters.proper_time.value();
1184 double lower_bound =
1186 if (IC_parameters.lower_bound.has_value())
1191 double default_proper_time = modus_.nuclei_passing_time();
1192 if (default_proper_time >= lower_bound) {
1193 proper_time = default_proper_time;
1196 <<
"Nuclei passing time is too short, hypersurface proper time "
1197 <<
"set to tau = " << lower_bound <<
" fm.";
1198 proper_time = lower_bound;
1202 action_finders_.emplace_back(
1203 std::make_unique<HyperSurfaceCrossActionsFinder>(
1204 proper_time, rapidity_cut, pT_cut));
1210 pauli_blocker_ = std::make_unique<PauliBlocker>(
1469 " create OutputInterface objects");
1472 <<
"Density type printed to headers: " << dens_type_;
1483 if (output_path ==
"") {
1484 throw std::invalid_argument(
1485 "Invalid empty output path provided to Experiment constructor.");
1486 }
else if (!std::filesystem::exists(output_path)) {
1488 "Output path \"" + output_path.string() +
1489 "\" used to create an Experiment object does not exist.");
1490 throw NonExistingOutputPathRequest(
"Attempt to use not existing path.");
1491 }
else if (!std::filesystem::is_directory(output_path)) {
1493 "\" used to create an Experiment object "
1494 "exists, but it is not a directory.");
1495 throw std::logic_error(
"Attempt to use invalid existing path.");
1497 const std::vector<std::string> output_contents =
1498 output_conf.list_upmost_nodes();
1499 if (output_conf.is_empty()) {
1500 logg[
LExperiment].warn() <<
"No \"Output\" section found in the input "
1501 "file. No output file will be produced.";
1505 std::vector<std::vector<std::string>> list_of_formats(output_contents.size());
1507 output_contents.cbegin(), output_contents.cend(), list_of_formats.begin(),
1508 [&output_conf](std::string content) -> std::vector<std::string> {
1512 return output_conf.take(InputKeys::get_output_format_key(content));
1514 auto abort_because_of_invalid_input_file = []() {
1515 throw std::invalid_argument(
"Invalid configuration input file.");
1517 const OutputParameters output_parameters(std::move(output_conf));
1518 for (std::size_t i = 0; i < output_contents.size(); ++i) {
1519 if (output_contents[i] ==
"Particles" ||
1520 output_contents[i] ==
"Collisions") {
1521 assert(output_parameters.quantities.count(output_contents[i]) > 0);
1522 const bool quantities_given_nonempty =
1523 !output_parameters.quantities.at(output_contents[i]).empty();
1524 auto formats_contains = [&list_of_formats, &i](
const std::string &label) {
1525 return std::find(list_of_formats[i].begin(), list_of_formats[i].end(),
1526 label) != list_of_formats[i].end();
1528 const bool custom_ascii_requested = formats_contains(
"ASCII");
1529 const bool custom_binary_requested = formats_contains(
"Binary");
1530 const bool custom_requested =
1531 custom_ascii_requested || custom_binary_requested;
1532 const bool oscar2013_requested = formats_contains(
"Oscar2013");
1533 const bool oscar2013_bin_requested = formats_contains(
"Oscar2013_bin");
1534 const bool is_extended = (output_contents[i] ==
"Particles")
1535 ? output_parameters.part_extended
1536 : output_parameters.coll_extended;
1537 const auto &default_quantities =
1540 const bool are_given_quantities_oscar2013_ones =
1541 output_parameters.quantities.at(output_contents[i]) ==
1543 if (quantities_given_nonempty != custom_requested) {
1545 <<
"Non-empty \"Quantities\" and \"ASCII\"/\"Binary\" format have "
1546 <<
"not been specified both for " << std::quoted(output_contents[i])
1547 <<
" in config file.";
1548 abort_because_of_invalid_input_file();
1550 if (custom_ascii_requested && oscar2013_requested &&
1551 are_given_quantities_oscar2013_ones) {
1553 <<
"The specified \"Quantities\" for the ASCII format are the same "
1554 "as those of the requested \"Oscar2013\"\nformat for "
1555 << std::quoted(output_contents[i])
1556 <<
" and this would produce the same output file twice.";
1557 abort_because_of_invalid_input_file();
1559 if (custom_binary_requested && oscar2013_bin_requested &&
1560 are_given_quantities_oscar2013_ones) {
1562 <<
"The specified \"Quantities\" for the binary format are the "
1563 "same as those of the requested \"Oscar2013_bin\"\nformat for "
1564 << std::quoted(output_contents[i])
1565 <<
" and this would produce the same output file twice.";
1566 abort_because_of_invalid_input_file();
1570 if (list_of_formats[i].empty()) {
1572 <<
"Empty or unspecified list of formats for "
1573 << std::quoted(output_contents[i]) <<
" content.";
1574 abort_because_of_invalid_input_file();
1575 }
else if (std::find(list_of_formats[i].begin(), list_of_formats[i].end(),
1576 "None") != list_of_formats[i].end()) {
1577 if (list_of_formats[i].size() > 1) {
1579 <<
"Use of \"None\" output format together with other formats is "
1580 "not allowed.\nInvalid \"Format\" key for "
1581 << std::quoted(output_contents[i]) <<
" content.";
1582 abort_because_of_invalid_input_file();
1585 list_of_formats[i].clear();
1587 }
else if (std::set<std::string> tmp_set(list_of_formats[i].begin(),
1588 list_of_formats[i].end());
1589 list_of_formats[i].size() != tmp_set.size()) {
1590 auto join_container = [](
const auto &container) {
1591 std::string result{};
1593 [&result](
const std::string s) {
1594 result += (result ==
"") ? s :
", " + s;
1598 const std::string old_formats = join_container(list_of_formats[i]),
1599 new_formats = join_container(tmp_set);
1601 <<
"Found the same output format multiple times for "
1602 << std::quoted(output_contents[i])
1603 <<
" content. Duplicates will be ignored:\n 'Format: [" << old_formats
1604 <<
"] -> [" << new_formats <<
"]'";
1605 list_of_formats[i].assign(tmp_set.begin(), tmp_set.end());
1611 std::size_t total_number_of_requested_formats = 0;
1612 for (std::size_t i = 0; i < output_contents.size(); ++i) {
1613 for (
const auto &
format : list_of_formats[i]) {
1614 create_output(
format, output_contents[i], output_path, output_parameters);
1615 ++total_number_of_requested_formats;
1619 if (outputs_.size() != total_number_of_requested_formats) {
1621 <<
"At least one invalid output format has been provided.";
1622 abort_because_of_invalid_input_file();
1633 throw std::invalid_argument(
"Can't use potentials without time steps!");
1637 <<
"Potentials don't work with frozen Fermi momenta! "
1638 "Use normal Fermi motion instead.";
1639 throw std::invalid_argument(
1640 "Can't use potentials "
1641 "with frozen Fermi momenta!");
1644 << parameters_.labclock->timestep_duration();
1646 potentials_ = std::make_unique<Potentials>(
1651 if (potentials_->use_skyrme() && potentials_->use_vdf()) {
1652 throw std::runtime_error(
1653 "Can't use Skyrme and VDF potentials at the same time!");
1655 if (potentials_->use_symmetry() && potentials_->use_vdf()) {
1656 throw std::runtime_error(
1657 "Can't use symmetry and VDF potentials at the same time!");
1659 if (potentials_->use_skyrme()) {
1662 <<
"\t\tSkyrme_A [MeV] = " << potentials_->skyrme_a() <<
"\n";
1664 <<
"\t\tSkyrme_B [MeV] = " << potentials_->skyrme_b() <<
"\n";
1666 <<
"\t\t Skyrme_tau = " << potentials_->skyrme_tau() <<
"\n";
1668 if (potentials_->use_symmetry()) {
1670 <<
"Symmetry potential is:"
1671 <<
"\n S_pot [MeV] = " << potentials_->symmetry_S_pot() <<
"\n";
1673 if (potentials_->use_vdf()) {
1676 << potentials_->saturation_density() <<
"\n";
1677 for (
int i = 0; i < potentials_->number_of_terms(); i++) {
1679 <<
"\t\tCoefficient_" << i + 1 <<
" = "
1680 << 1000.0 * (potentials_->coeffs())[i] <<
" [MeV] \t Power_"
1681 << i + 1 <<
" = " << (potentials_->powers())[i] <<
"\n";
1687 throw std::invalid_argument(
1688 "Derivatives are necessary for running with potentials.\n"
1689 "Derivatives_Mode: \"Off\" only makes sense for "
1690 "Field_Derivatives_Mode: \"Direct\"!\nUse \"Covariant Gaussian\" or "
1691 "\"Finite difference\".");
1699 switch (parameters_.derivatives_mode) {
1710 switch (parameters_.rho_derivatives_mode) {
1719 if (potentials_->use_vdf()) {
1720 switch (parameters_.field_derivatives_mode) {
1733 if (potentials_->use_vdf() && (parameters_.rho_derivatives_mode ==
1735 parameters_.field_derivatives_mode ==
1737 throw std::runtime_error(
1738 "Can't use VDF potentials without rest frame density derivatives or "
1739 "direct field derivatives!");
1744 throw std::runtime_error(
1745 "Can't use potentials without gradients of baryon current (Skyrme, "
1747 " or direct field derivatives (VDF)!");
1750 if (!(potentials_->use_vdf()) &&
1752 throw std::invalid_argument(
1753 "Field_Derivatives_Mode: \"Direct\" only makes sense for the VDF "
1754 "potentials!\nUse Field_Derivatives_Mode: \"Chain Rule\" or comment "
1755 "this option out (Chain Rule is default)");
1760 switch (parameters_.smearing_mode) {
1766 << parameters_.discrete_weight;
1770 << parameters_.triangular_range;
1777 bool all_geometrical_properties_specified =
1781 if (!automatic && !all_geometrical_properties_specified) {
1782 throw std::invalid_argument(
1783 "The lattice was requested to be manually generated, but some\n"
1784 "lattice geometrical property was not specified. Be sure to provide\n"
1785 "both \"Cell_Number\" and \"Origin\" and \"Sizes\".");
1787 if (automatic && all_geometrical_properties_specified) {
1788 throw std::invalid_argument(
1789 "The lattice was requested to be automatically generated, but all\n"
1790 "lattice geometrical properties were specified. In this case you\n"
1791 "need to set \"Automatic: False\".");
1794 const auto [l,
n, origin] = [&config, automatic,
this]() {
1796 return std::make_tuple<std::array<double, 3>, std::array<int, 3>,
1797 std::array<double, 3>>(
1802 std::array<double, 3> l_default{20., 20., 20.};
1803 std::array<int, 3> n_default{10, 10, 10};
1804 std::array<double, 3> origin_default{-20., -20., -20.};
1805 if (modus_.is_collider() || (modus_.is_list() && !modus_.is_box())) {
1808 const double gam = modus_.is_collider()
1811 const double max_z = 5.0 / gam + end_time_;
1812 const double estimated_max_transverse_velocity = 0.7;
1813 const double max_xy =
1814 5.0 + estimated_max_transverse_velocity * end_time_;
1815 origin_default = {-max_xy, -max_xy, -max_z};
1816 l_default = {2 * max_xy, 2 * max_xy, 2 * max_z};
1819 const int n_xy = numeric_cast<int>(std::ceil(2 * max_xy / 0.8));
1820 int nz = numeric_cast<int>(std::ceil(2 * max_z / 0.8));
1825 nz = numeric_cast<int>(std::ceil(2 * max_z / 0.8 * gam));
1827 n_default = {n_xy, n_xy, nz};
1828 }
else if (modus_.is_box()) {
1829 origin_default = {0., 0., 0.};
1830 const double bl = modus_.length();
1831 l_default = {bl, bl, bl};
1832 const int n_xyz = numeric_cast<int>(std::ceil(bl / 0.5));
1833 n_default = {n_xyz, n_xyz, n_xyz};
1834 }
else if (modus_.is_sphere()) {
1837 const double max_d = modus_.radius() + end_time_;
1838 origin_default = {-max_d, -max_d, -max_d};
1839 l_default = {2 * max_d, 2 * max_d, 2 * max_d};
1841 const int n_xyz = numeric_cast<int>(std::ceil(2 * max_d / 0.8));
1842 n_default = {n_xyz, n_xyz, n_xyz};
1845 return std::make_tuple<std::array<double, 3>, std::array<int, 3>,
1846 std::array<double, 3>>(
1854 <<
"Lattice is ON. Origin = (" << origin[0] <<
"," << origin[1] <<
","
1855 << origin[2] <<
"), sizes = (" << l[0] <<
"," << l[1] <<
"," << l[2]
1856 <<
"), number of cells = (" <<
n[0] <<
"," <<
n[1] <<
"," <<
n[2]
1857 <<
"), periodic = " << std::boolalpha << periodic;
1859 if (printout_lattice_td_ || printout_full_lattice_any_td_) {
1860 dens_type_lattice_printout_ = output_parameters.td_dens_type;
1861 printout_rho_eckart_ = output_parameters.td_rho_eckart;
1862 printout_tmn_ = output_parameters.td_tmn;
1863 printout_tmn_landau_ = output_parameters.td_tmn_landau;
1864 printout_v_landau_ = output_parameters.td_v_landau;
1865 printout_j_QBS_ = output_parameters.td_jQBS;
1867 if (printout_tmn_ || printout_tmn_landau_ || printout_v_landau_) {
1868 Tmn_ = std::make_unique<RectangularLattice<EnergyMomentumTensor>>(
1871 if (printout_j_QBS_) {
1872 j_QBS_lat_ = std::make_unique<DensityLattice>(l,
n, origin, periodic,
1880 old_jmu_auxiliary_ = std::make_unique<RectangularLattice<FourVector>>(
1882 new_jmu_auxiliary_ = std::make_unique<RectangularLattice<FourVector>>(
1884 four_gradient_auxiliary_ =
1885 std::make_unique<RectangularLattice<std::array<FourVector, 4>>>(
1888 if (potentials_->use_skyrme()) {
1889 jmu_B_lat_ = std::make_unique<DensityLattice>(
1891 UB_lat_ = std::make_unique<RectangularLattice<FourVector>>(
1893 FB_lat_ = std::make_unique<
1894 RectangularLattice<std::pair<ThreeVector, ThreeVector>>>(
1897 if (potentials_->use_symmetry()) {
1898 jmu_I3_lat_ = std::make_unique<DensityLattice>(
1900 UI3_lat_ = std::make_unique<RectangularLattice<FourVector>>(
1902 FI3_lat_ = std::make_unique<
1903 RectangularLattice<std::pair<ThreeVector, ThreeVector>>>(
1906 if (potentials_->use_coulomb()) {
1907 jmu_el_lat_ = std::make_unique<DensityLattice>(
1909 EM_lat_ = std::make_unique<
1910 RectangularLattice<std::pair<ThreeVector, ThreeVector>>>(
1913 if (potentials_->use_vdf()) {
1914 jmu_B_lat_ = std::make_unique<DensityLattice>(
1916 UB_lat_ = std::make_unique<RectangularLattice<FourVector>>(
1918 FB_lat_ = std::make_unique<
1919 RectangularLattice<std::pair<ThreeVector, ThreeVector>>>(
1924 old_fields_auxiliary_ =
1925 std::make_unique<RectangularLattice<FourVector>>(
1927 new_fields_auxiliary_ =
1928 std::make_unique<RectangularLattice<FourVector>>(
1930 fields_four_gradient_auxiliary_ =
1931 std::make_unique<RectangularLattice<std::array<FourVector, 4>>>(
1935 fields_lat_ = std::make_unique<FieldsLattice>(
1940 jmu_B_lat_ = std::make_unique<DensityLattice>(l,
n, origin, periodic,
1945 jmu_I3_lat_ = std::make_unique<DensityLattice>(l,
n, origin, periodic,
1951 jmu_custom_lat_ = std::make_unique<DensityLattice>(
1954 }
else if (printout_lattice_td_ || printout_full_lattice_any_td_) {
1956 "If you want Therm. VTK or Lattice output, configure a lattice for "
1958 }
else if (potentials_ && potentials_->use_coulomb()) {
1960 "Coulomb potential requires a lattice. Please add one to the "
1965 if ((potentials_ !=
nullptr) && (jmu_B_lat_ ==
nullptr)) {
1966 logg[
LExperiment].warn() <<
"Lattice is NOT used. Mean-field energy is "
1967 <<
"not going to be calculated.";
1971 if (parameters_.potential_affect_threshold) {
1979 (jmu_B_lat_ ==
nullptr)) {
1980 throw std::runtime_error(
1981 "Lattice is necessary to calculate finite difference gradients.");
1988 thermalizer_ = modus_.create_grandcan_thermalizer(th_conf);
2024 uint64_t scatterings_this_interval,
2027 double E_mean_field,
2028 double E_mean_field_initial);
2065 double E_mean_field,
double modus_impact_parameter,
2067 bool projectile_target_interact,
2068 bool kinematic_cut_for_SMASH_IC);
2070 template <
typename Modus>
2080 while (r == INT64_MIN) {
2083 seed_ = std::abs(r);
2088 if (process_string_ptr_ != NULL) {
2089 process_string_ptr_->init_pythia_hadron_rndm();
2092 for (
Particles &particles : ensembles_) {
2097 double start_time = -1.0;
2101 if (modus_.is_collider()) {
2102 modus_.sample_impact();
2103 logg[
LExperiment].info(
"Impact parameter = ", modus_.impact_parameter(),
2106 for (
Particles &particles : ensembles_) {
2107 start_time = modus_.initial_conditions(&particles, parameters_);
2112 for (
Particles &particles : ensembles_) {
2113 modus_.impose_boundary_conditions(&particles, outputs_);
2116 double timestep = delta_time_startup_;
2118 switch (time_step_mode_) {
2122 timestep = end_time_ - start_time;
2124 const double max_dt = modus_.max_timestep(max_transverse_distance_sqr_);
2125 if (max_dt > 0. && max_dt < timestep) {
2130 std::unique_ptr<UniformClock> clock_for_this_event;
2131 if (modus_.is_list() && (timestep < 0.0)) {
2132 throw std::runtime_error(
2133 "Timestep for the given event is negative. \n"
2134 "This might happen if the formation times of the input particles are "
2135 "larger than the specified end time of the simulation.");
2137 clock_for_this_event =
2138 std::make_unique<UniformClock>(start_time, timestep, end_time_);
2139 parameters_.labclock = std::move(clock_for_this_event);
2142 parameters_.outputclock->reset(start_time,
true);
2144 parameters_.outputclock->remove_times_in_past(start_time);
2147 "Lab clock: t_start = ", parameters_.labclock->current_time(),
2148 ", dt = ", parameters_.labclock->timestep_duration());
2153 wall_actions_total_ = 0;
2154 previous_wall_actions_total_ = 0;
2155 interactions_total_ = 0;
2156 previous_interactions_total_ = 0;
2157 discarded_interactions_total_ = 0;
2158 total_pauli_blocked_ = 0;
2159 projectile_target_interact_.assign(parameters_.n_ensembles,
false);
2160 total_hypersurface_crossing_actions_ = 0;
2161 total_energy_removed_ = 0.0;
2162 total_energy_violated_by_Pythia_ = 0.0;
2165 logg[
LExperiment].info() <<
"Time[fm] Ekin[GeV] E_MF[GeV] ETotal[GeV] "
2166 <<
"ETot/N[GeV] D(ETot/N)[GeV] Scatt&Decays "
2167 <<
"Particles Comp.Time";
2169 double E_mean_field = 0.0;
2174 if ((jmu_B_lat_ !=
nullptr)) {
2176 new_jmu_auxiliary_.get(), four_gradient_auxiliary_.get(),
2178 density_param_, ensembles_,
2179 parameters_.labclock->timestep_duration(),
true);
2183 for (
auto &node : *jmu_B_lat_) {
2184 node.overwrite_drho_dt_to_zero();
2185 node.overwrite_djmu_dt_to_zero();
2188 EM_lat_.get(), parameters_);
2191 initial_mean_field_energy_ = E_mean_field;
2193 ensembles_, 0u, conserved_initial_, time_start_,
2194 parameters_.labclock->current_time(), E_mean_field,
2195 initial_mean_field_energy_);
2198 for (
const auto &output : outputs_) {
2199 for (
int i_ens = 0; i_ens < parameters_.n_ensembles; i_ens++) {
2201 ensembles_, E_mean_field, modus_.impact_parameter(), parameters_,
2202 projectile_target_interact_[i_ens], kinematic_cuts_for_IC_output_);
2203 output->at_eventstart(ensembles_[i_ens], {event_, i_ens}, event_info);
2206 output->at_eventstart(ensembles_, event_);
2208 if (printout_full_lattice_any_td_) {
2209 if (printout_rho_eckart_) {
2210 switch (dens_type_lattice_printout_) {
2227 if (printout_tmn_) {
2229 dens_type_lattice_printout_, *Tmn_);
2231 if (printout_tmn_landau_) {
2233 dens_type_lattice_printout_, *Tmn_);
2235 if (printout_v_landau_) {
2237 dens_type_lattice_printout_, *Tmn_);
2239 if (printout_j_QBS_) {
2241 dens_type_lattice_printout_, *j_QBS_lat_);
2253 const double m = particle.effective_mass();
2254 double v_beam = 0.0;
2256 v_beam = modus_.velocity_projectile();
2258 v_beam = modus_.velocity_target();
2260 const double gamma = 1.0 / std::sqrt(1.0 - v_beam * v_beam);
2261 beam_momentum_.emplace_back(
2262 FourVector(gamma * m, 0.0, 0.0, gamma * v_beam * m));
2267 template <
typename Modus>
2269 bool include_pauli_blocking) {
2270 Particles &particles = ensembles_[i_ensemble];
2273 discarded_interactions_total_++;
2275 " (discarded: invalid)");
2280 }
catch (Action::StochasticBelowEnergyThreshold &) {
2284 if (include_pauli_blocking && pauli_blocker_ &&
2286 total_pauli_blocked_++;
2292 if (modus_.is_collider()) {
2293 int count_target = 0, count_projectile = 0;
2301 if (count_target > 0 && count_projectile > 0) {
2302 projectile_target_interact_[i_ensemble] =
true;
2308 const auto id_process =
static_cast<uint32_t
>(interactions_total_ + 1);
2310 total_energy_violated_by_Pythia_ += action.
perform(&particles, id_process);
2312 interactions_total_++;
2314 wall_actions_total_++;
2317 total_hypersurface_crossing_actions_++;
2324 constexpr
bool compute_grad =
false;
2325 const bool smearing =
true;
2328 rho = std::get<0>(
current_eckart(r_interaction.threevec(), particles,
2329 density_param_, dens_type_, compute_grad,
2347 for (
const auto &output : outputs_) {
2348 if (!output->is_dilepton_output() && !output->is_photon_output()) {
2349 if (output->is_IC_output() &&
2351 output->at_interaction(action, rho);
2352 }
else if (!output->is_IC_output()) {
2353 output->at_interaction(action, rho);
2363 if (photons_switch_ &&
2369 constexpr
double action_time = 0.;
2371 n_fractional_photons_,
2387 photon_act.add_single_process();
2389 photon_act.perform_photons(outputs_);
2392 if (bremsstrahlung_switch_ &&
2397 constexpr
double action_time = 0.;
2400 n_fractional_photons_,
2417 brems_act.add_single_process();
2419 brems_act.perform_bremsstrahlung(outputs_);
2438 template <
typename Modus>
2440 ParticleList &&add_plist,
2441 ParticleList &&remove_plist) {
2442 if (!add_plist.empty() || !remove_plist.empty()) {
2443 if (ensembles_.size() > 1) {
2444 throw std::runtime_error(
2445 "Adding or removing particles from SMASH is only possible when one "
2446 "ensemble is used.");
2448 const double action_time = parameters_.labclock->current_time();
2452 if (!add_plist.empty()) {
2455 if (!add_plist.empty()) {
2457 auto action_add_particles = std::make_unique<FreeforallAction>(
2458 ParticleList{}, add_plist, action_time);
2459 perform_action(*action_add_particles, 0);
2462 if (!remove_plist.empty()) {
2465 if (!remove_plist.empty()) {
2466 ParticleList found_particles_to_remove;
2467 for (
const auto &particle_to_remove : remove_plist) {
2468 const auto iterator_to_particle_to_be_removed_in_ensemble =
2470 ensembles_[0].begin(), ensembles_[0].end(),
2471 [&particle_to_remove, &action_time](
const ParticleData &
p) {
2473 particle_to_remove,
p, action_time);
2475 if (iterator_to_particle_to_be_removed_in_ensemble !=
2476 ensembles_[0].end())
2477 found_particles_to_remove.push_back(
2478 *iterator_to_particle_to_be_removed_in_ensemble);
2482 std::sort(found_particles_to_remove.begin(),
2483 found_particles_to_remove.end(),
2485 return p1.id() < p2.id();
2487 const auto iterator_to_first_duplicate = std::adjacent_find(
2488 found_particles_to_remove.begin(), found_particles_to_remove.end(),
2490 return p1.id() == p2.id();
2492 if (iterator_to_first_duplicate != found_particles_to_remove.end()) {
2493 logg[
LExperiment].error() <<
"The same particle has been asked to be "
2494 "removed multiple times:\n"
2495 << *iterator_to_first_duplicate;
2496 throw std::logic_error(
"Particle cannot be removed twice!");
2498 if (
auto delta = remove_plist.size() - found_particles_to_remove.size();
2501 "When trying to remove particle(s) at the beginning ",
2502 "of the system evolution,\n", delta,
2503 " particle(s) could not be found and will be ignored.");
2505 if (!found_particles_to_remove.empty()) {
2506 [[maybe_unused]]
const auto number_particles_before_removal =
2507 ensembles_[0].size();
2509 auto action_remove_particles = std::make_unique<FreeforallAction>(
2510 found_particles_to_remove, ParticleList{}, action_time);
2511 perform_action(*action_remove_particles, 0);
2513 assert(number_particles_before_removal -
2514 found_particles_to_remove.size() ==
2515 ensembles_[0].size());
2520 if (t_end > end_time_) {
2522 <<
"Evolution asked to be run until " << t_end <<
" > " << end_time_
2523 <<
" and this cannot be done (because of how the clock works).";
2524 throw std::logic_error(
2525 "Experiment cannot evolve the system beyond End_Time.");
2527 while (*(parameters_.labclock) < t_end) {
2528 const double dt = parameters_.labclock->timestep_duration();
2529 logg[
LExperiment].debug(
"Timestepless propagation for next ", dt,
" fm.");
2533 thermalizer_->is_time_to_thermalize(parameters_.labclock)) {
2534 const bool ignore_cells_under_treshold =
true;
2537 thermalizer_->update_thermalizer_lattice(ensembles_, density_param_,
2538 ignore_cells_under_treshold);
2539 const double current_t = parameters_.labclock->current_time();
2540 for (
int i_ens = 0; i_ens < parameters_.n_ensembles; i_ens++) {
2541 thermalizer_->thermalize(ensembles_[i_ens], current_t,
2542 parameters_.testparticles);
2545 perform_action(th_act, i_ens);
2551 modus_.build_fluidization_lattice(parameters_.labclock->current_time(),
2552 ensembles_, density_param_);
2555 std::vector<Actions> actions(parameters_.n_ensembles);
2556 for (
int i_ens = 0; i_ens < parameters_.n_ensembles; i_ens++) {
2557 actions[i_ens].clear();
2558 if (ensembles_[i_ens].size() > 0 && action_finders_.size() > 0) {
2560 const double min_cell_length = compute_min_cell_length(dt);
2565 const bool include_unformed_particles = IC_switch_;
2567 use_grid_ ? modus_.create_grid(ensembles_[i_ens], min_cell_length,
2568 dt, parameters_.coll_crit,
2569 include_unformed_particles)
2570 : modus_.create_grid(ensembles_[i_ens], min_cell_length,
2571 dt, parameters_.coll_crit,
2572 include_unformed_particles,
2575 const double gcell_vol = grid.cell_volume();
2578 [&](
const ParticleList &search_list) {
2579 for (
const auto &finder : action_finders_) {
2580 actions[i_ens].insert(finder->find_actions_in_cell(
2581 search_list, dt, gcell_vol, beam_momentum_));
2584 [&](
const ParticleList &search_list,
2585 const ParticleList &neighbors_list) {
2586 for (
const auto &finder : action_finders_) {
2587 actions[i_ens].insert(finder->find_actions_with_neighbors(
2588 search_list, neighbors_list, dt, beam_momentum_));
2597 const double end_timestep_time = parameters_.labclock->next_time();
2598 while (next_output_time() < end_timestep_time) {
2599 for (
int i_ens = 0; i_ens < parameters_.n_ensembles; i_ens++) {
2600 run_time_evolution_timestepless(actions[i_ens], i_ens,
2601 next_output_time());
2603 ++(*parameters_.outputclock);
2605 intermediate_output();
2607 for (
int i_ens = 0; i_ens < parameters_.n_ensembles; i_ens++) {
2608 run_time_evolution_timestepless(actions[i_ens], i_ens, end_timestep_time);
2614 update_potentials();
2615 update_momenta(ensembles_, parameters_.labclock->timestep_duration(),
2616 *potentials_, FB_lat_.get(), FI3_lat_.get(), EM_lat_.get(),
2623 for (
Particles &particles : ensembles_) {
2628 ++(*parameters_.labclock);
2636 if (!potentials_ && !parameters_.strings_switch &&
2638 std::string err_msg = conserved_initial_.report_deviations(ensembles_);
2639 if (!err_msg.empty()) {
2641 throw std::runtime_error(
"Violation of conserved quantities!");
2654 ++(*parameters_.outputclock);
2656 if (pauli_blocker_) {
2658 "Interactions: Pauli-blocked/performed = ", total_pauli_blocked_,
"/",
2659 interactions_total_ - wall_actions_total_);
2663 template <
typename Modus>
2668 if (dilepton_finder_ !=
nullptr) {
2669 for (
const auto &output : outputs_) {
2670 dilepton_finder_->shine(particles, output.get(), dt);
2683 constexpr uint64_t max_uint32 = std::numeric_limits<uint32_t>::max();
2684 if (interactions_total >= max_uint32) {
2685 throw std::runtime_error(
"Integer overflow in total interaction number!");
2689 template <
typename Modus>
2691 Actions &actions,
int i_ensemble,
const double end_time_propagation) {
2692 Particles &particles = ensembles_[i_ensemble];
2694 "Timestepless propagation: ",
"Actions size = ", actions.
size(),
2695 ", end time = ", end_time_propagation);
2703 ActionPtr act = actions.
pop();
2704 if (!act->is_valid(particles)) {
2705 discarded_interactions_total_++;
2707 " (discarded: invalid)");
2711 ", action time = ", act->time_of_execution());
2714 propagate_and_shine(act->time_of_execution(), particles);
2721 act->update_incoming(particles);
2722 const bool performed = perform_action(*act, i_ensemble);
2732 const double end_time_timestep = parameters_.labclock->next_time();
2734 const double time_left = end_time_timestep - act->time_of_execution();
2735 const ParticleList &outgoing_particles = act->outgoing_particles();
2737 const double gcell_vol = 0.0;
2738 for (
const auto &finder : action_finders_) {
2740 actions.
insert(finder->find_actions_in_cell(outgoing_particles, time_left,
2741 gcell_vol, beam_momentum_));
2743 actions.
insert(finder->find_actions_with_surrounding_particles(
2744 outgoing_particles, particles, time_left, beam_momentum_));
2750 propagate_and_shine(end_time_propagation, particles);
2753 template <
typename Modus>
2755 const uint64_t wall_actions_this_interval =
2756 wall_actions_total_ - previous_wall_actions_total_;
2757 previous_wall_actions_total_ = wall_actions_total_;
2758 const uint64_t interactions_this_interval = interactions_total_ -
2759 previous_interactions_total_ -
2760 wall_actions_this_interval;
2761 previous_interactions_total_ = interactions_total_;
2762 double E_mean_field = 0.0;
2765 double computational_frame_time = 0.0;
2768 if ((jmu_B_lat_ !=
nullptr)) {
2770 EM_lat_.get(), parameters_);
2777 if (modus_.is_box()) {
2778 double tmp = (E_mean_field - initial_mean_field_energy_) /
2779 (E_mean_field + initial_mean_field_energy_);
2785 if (std::abs(tmp) > 0.01) {
2787 <<
"\n\n\n\t The mean field at t = "
2788 << parameters_.outputclock->current_time()
2789 <<
" [fm] differs from the mean field at t = 0:"
2790 <<
"\n\t\t initial_mean_field_energy_ = "
2791 << initial_mean_field_energy_ <<
" [GeV]"
2792 <<
"\n\t\t abs[(E_MF - E_MF(t=0))/(E_MF + E_MF(t=0))] = "
2794 <<
"\n\t\t E_MF/E_MF(t=0) = "
2795 << E_mean_field / initial_mean_field_energy_ <<
"\n\n";
2802 ensembles_, interactions_this_interval, conserved_initial_, time_start_,
2803 parameters_.outputclock->current_time(), E_mean_field,
2804 initial_mean_field_energy_);
2808 if (!(modus_.is_box() && parameters_.outputclock->current_time() <
2809 modus_.equilibration_time())) {
2810 for (
const auto &output : outputs_) {
2811 if (output->is_dilepton_output() || output->is_photon_output() ||
2812 output->is_IC_output()) {
2815 for (
int i_ens = 0; i_ens < parameters_.n_ensembles; i_ens++) {
2817 ensembles_, E_mean_field, modus_.impact_parameter(), parameters_,
2818 projectile_target_interact_[i_ens], kinematic_cuts_for_IC_output_);
2820 output->at_intermediate_time(ensembles_[i_ens], parameters_.outputclock,
2821 density_param_, {event_, i_ens},
2823 computational_frame_time = event_info.current_time;
2826 output->at_intermediate_time(ensembles_, parameters_.outputclock,
2830 if (printout_rho_eckart_) {
2831 switch (dens_type_lattice_printout_) {
2838 output->thermodynamics_lattice_output(*jmu_B_lat_,
2839 computational_frame_time);
2844 density_param_, ensembles_,
false);
2848 output->thermodynamics_lattice_output(*jmu_I3_lat_,
2849 computational_frame_time);
2855 jmu_custom_lat_.get(), lat_upd, dens_type_lattice_printout_,
2856 density_param_, ensembles_,
false);
2858 dens_type_lattice_printout_,
2860 output->thermodynamics_lattice_output(*jmu_custom_lat_,
2861 computational_frame_time);
2864 if (printout_tmn_ || printout_tmn_landau_ || printout_v_landau_) {
2866 Tmn_.get(), lat_upd, dens_type_lattice_printout_, density_param_,
2868 if (printout_tmn_) {
2870 dens_type_lattice_printout_, *Tmn_);
2871 output->thermodynamics_lattice_output(
2874 if (printout_tmn_landau_) {
2876 dens_type_lattice_printout_, *Tmn_);
2877 output->thermodynamics_lattice_output(
2879 computational_frame_time);
2881 if (printout_v_landau_) {
2883 dens_type_lattice_printout_, *Tmn_);
2884 output->thermodynamics_lattice_output(
2886 computational_frame_time);
2890 output->fields_output(
"Efield",
"Bfield", *EM_lat_);
2892 if (printout_j_QBS_) {
2893 output->thermodynamics_lattice_output(
2894 *j_QBS_lat_, computational_frame_time, ensembles_, density_param_);
2898 output->thermodynamics_output(*thermalizer_);
2904 template <
typename Modus>
2907 if (potentials_->use_symmetry() && jmu_I3_lat_ !=
nullptr) {
2909 new_jmu_auxiliary_.get(), four_gradient_auxiliary_.get(),
2911 density_param_, ensembles_,
2912 parameters_.labclock->timestep_duration(),
true);
2914 if ((potentials_->use_skyrme() || potentials_->use_symmetry()) &&
2915 jmu_B_lat_ !=
nullptr) {
2917 new_jmu_auxiliary_.get(), four_gradient_auxiliary_.get(),
2919 density_param_, ensembles_,
2920 parameters_.labclock->timestep_duration(),
true);
2921 const size_t UBlattice_size = UB_lat_->size();
2922 for (
size_t i = 0; i < UBlattice_size; i++) {
2923 auto jB = (*jmu_B_lat_)[i];
2927 double baryon_density = jB.rho();
2931 if (potentials_->use_skyrme()) {
2933 flow_four_velocity_B * potentials_->skyrme_pot(baryon_density);
2935 potentials_->skyrme_force(baryon_density, baryon_grad_j0,
2936 baryon_dvecj_dt, baryon_curl_vecj);
2938 if (potentials_->use_symmetry() && jmu_I3_lat_ !=
nullptr) {
2939 auto jI3 = (*jmu_I3_lat_)[i];
2942 ? jI3.jmu_net() / jI3.rho()
2944 (*UI3_lat_)[i] = flow_four_velocity_I3 *
2945 potentials_->symmetry_pot(jI3.rho(), baryon_density);
2946 (*FI3_lat_)[i] = potentials_->symmetry_force(
2947 jI3.rho(), jI3.grad_j0(), jI3.dvecj_dt(), jI3.curl_vecj(),
2948 baryon_density, baryon_grad_j0, baryon_dvecj_dt,
2953 if (potentials_->use_coulomb()) {
2956 density_param_, ensembles_,
true);
2957 for (
size_t i = 0; i < EM_lat_->size(); i++) {
2959 ThreeVector position = jmu_el_lat_->cell_center(i);
2960 jmu_el_lat_->integrate_volume(electric_field,
2962 potentials_->coulomb_r_cut(), position);
2964 jmu_el_lat_->integrate_volume(magnetic_field,
2966 potentials_->coulomb_r_cut(), position);
2967 (*EM_lat_)[i] = std::make_pair(electric_field, magnetic_field);
2970 if (potentials_->use_vdf() && jmu_B_lat_ !=
nullptr) {
2972 new_jmu_auxiliary_.get(), four_gradient_auxiliary_.get(),
2974 density_param_, ensembles_,
2975 parameters_.labclock->timestep_duration(),
true);
2978 fields_lat_.get(), old_fields_auxiliary_.get(),
2979 new_fields_auxiliary_.get(), fields_four_gradient_auxiliary_.get(),
2981 parameters_.labclock->timestep_duration());
2983 const size_t UBlattice_size = UB_lat_->size();
2984 for (
size_t i = 0; i < UBlattice_size; i++) {
2985 auto jB = (*jmu_B_lat_)[i];
2986 (*UB_lat_)[i] = potentials_->vdf_pot(jB.rho(), jB.jmu_net());
2987 switch (parameters_.field_derivatives_mode) {
2989 (*FB_lat_)[i] = potentials_->vdf_force(
2990 jB.rho(), jB.drho_dxnu().x0(), jB.drho_dxnu().threevec(),
2991 jB.grad_rho_cross_vecj(), jB.jmu_net().x0(), jB.grad_j0(),
2992 jB.jmu_net().threevec(), jB.dvecj_dt(), jB.curl_vecj());
2995 auto Amu = (*fields_lat_)[i];
2996 (*FB_lat_)[i] = potentials_->vdf_force(
2997 Amu.grad_A0(), Amu.dvecA_dt(), Amu.curl_vecA());
3005 template <
typename Modus>
3009 bool actions_performed, decays_found;
3010 uint64_t interactions_old;
3012 decays_found =
false;
3013 interactions_old = interactions_total_;
3014 for (
int i_ens = 0; i_ens < parameters_.n_ensembles; i_ens++) {
3018 if (dilepton_finder_ !=
nullptr) {
3019 for (
const auto &output : outputs_) {
3020 dilepton_finder_->shine_final(ensembles_[i_ens], output.get(),
true);
3024 for (
const auto &finder : action_finders_) {
3025 auto found_actions = finder->find_final_actions(ensembles_[i_ens]);
3026 if (!found_actions.empty()) {
3027 actions.
insert(std::move(found_actions));
3028 decays_found =
true;
3033 perform_action(*actions.
pop(), i_ens,
false);
3036 actions_performed = interactions_total_ > interactions_old;
3038 if (decays_found && !actions_performed) {
3039 throw std::runtime_error(
"Final decays were found but not performed.");
3042 }
while (actions_performed);
3045 if (dilepton_finder_ !=
nullptr) {
3046 for (
const auto &output : outputs_) {
3047 for (
Particles &particles : ensembles_) {
3048 dilepton_finder_->shine_final(particles, output.get(),
false);
3054 template <
typename Modus>
3059 double E_mean_field = 0.0;
3060 if (
likely(parameters_.labclock > 0)) {
3061 const uint64_t wall_actions_this_interval =
3062 wall_actions_total_ - previous_wall_actions_total_;
3063 const uint64_t interactions_this_interval = interactions_total_ -
3064 previous_interactions_total_ -
3065 wall_actions_this_interval;
3068 if ((jmu_B_lat_ !=
nullptr)) {
3070 EM_lat_.get(), parameters_);
3073 if (std::abs(parameters_.labclock->current_time() - end_time_) >
3076 <<
"SMASH not propagated until configured end time. Current time = "
3077 << parameters_.labclock->current_time()
3078 <<
"fm. End time = " << end_time_ <<
"fm.";
3081 ensembles_, interactions_this_interval, conserved_initial_,
3082 time_start_, end_time_, E_mean_field, initial_mean_field_energy_);
3084 int total_particles = 0;
3085 for (
const Particles &particles : ensembles_) {
3086 total_particles += particles.
size();
3088 if (IC_switch_ && (total_particles == 0)) {
3089 const double initial_system_energy_plus_Pythia_violations =
3090 conserved_initial_.momentum().x0() + total_energy_violated_by_Pythia_;
3091 const double fraction_of_total_system_energy_removed =
3092 initial_system_energy_plus_Pythia_violations / total_energy_removed_;
3095 if (std::fabs(fraction_of_total_system_energy_removed - 1.) >
3097 throw std::runtime_error(
3098 "There is remaining energy in the system although all particles "
3101 std::to_string((initial_system_energy_plus_Pythia_violations -
3102 total_energy_removed_)) +
3107 <<
"Time real: " << SystemClock::now() - time_start_;
3109 <<
"Interactions before reaching hypersurface: "
3110 << interactions_total_ - wall_actions_total_ -
3111 total_hypersurface_crossing_actions_;
3113 <<
"Total number of particles removed on hypersurface: "
3114 << total_hypersurface_crossing_actions_;
3117 const double precent_discarded =
3118 interactions_total_ > 0
3119 ?
static_cast<double>(discarded_interactions_total_) * 100.0 /
3122 std::stringstream msg_discarded;
3124 <<
"Discarded interaction number: " << discarded_interactions_total_
3125 <<
" (" << precent_discarded
3126 <<
"% of the total interaction number including wall crossings)";
3130 <<
"Time real: " << SystemClock::now() - time_start_;
3134 precent_discarded > 1.0) {
3137 << msg_discarded.str()
3138 <<
"\nThe number of discarded interactions is large, which means "
3139 "the assumption for the stochastic criterion of\n1 interaction "
3140 "per particle per timestep is probably violated. Consider "
3141 "reducing the timestep size.";
3145 << interactions_total_ - wall_actions_total_;
3149 int unformed_particles_count = 0;
3150 for (
const Particles &particles : ensembles_) {
3152 if (particle.formation_time() > end_time_) {
3153 unformed_particles_count++;
3157 if (unformed_particles_count > 0) {
3159 "End time might be too small. ", unformed_particles_count,
3160 " unformed particles were found at the end of the evolution.");
3165 count_nonempty_ensembles();
3167 for (
const auto &output : outputs_) {
3168 for (
int i_ens = 0; i_ens < parameters_.n_ensembles; i_ens++) {
3170 ensembles_, E_mean_field, modus_.impact_parameter(), parameters_,
3171 projectile_target_interact_[i_ens], kinematic_cuts_for_IC_output_);
3172 output->at_eventend(ensembles_[i_ens], {event_, i_ens}, event_info);
3175 output->at_eventend(ensembles_, event_);
3178 if (printout_rho_eckart_) {
3183 if (printout_tmn_) {
3186 if (printout_tmn_landau_) {
3189 if (printout_v_landau_) {
3192 if (printout_j_QBS_) {
3198 template <
typename Modus>
3200 for (
bool has_interaction : projectile_target_interact_) {
3201 if (has_interaction) {
3202 nonempty_ensembles_++;
3207 template <
typename Modus>
3210 return event_ >= nevents_;
3213 if (nonempty_ensembles_ >= minimum_nonempty_ensembles_) {
3216 if (event_ >= max_events_) {
3218 <<
"Maximum number of events (" << max_events_
3219 <<
") exceeded. Stopping calculation. "
3220 <<
"The fraction of empty ensembles is "
3221 << (1.0 -
static_cast<double>(nonempty_ensembles_) /
3222 (event_ * parameters_.n_ensembles))
3223 <<
". If this fraction is expected, try increasing the "
3224 "Maximum_Ensembles_Run.";
3229 throw std::runtime_error(
"Event counting option is invalid");
3233 template <
typename Modus>
3238 template <
typename Modus>
3241 for (event_ = 0; !is_finished(); event_++) {
3242 mainlog.info() <<
"Event " << event_;
3245 initialize_new_event();
3247 run_time_evolution(end_time_);
3249 if (force_decays_) {
Collection of useful type aliases to measure and output the (real) runtime.
A stream modifier that allows to colorize the log output.
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.
virtual double perform(Particles *particles, uint32_t id_process)
Actually perform the action, e.g.
const ParticleList & incoming_particles() const
Get the list of particles that go into the action.
virtual void generate_final_state()=0
Generate the final state for this action.
double sqrt_s() const
Determine the total energy in the center-of-mass frame [GeV].
FourVector get_interaction_point() const
Get the interaction point.
bool is_valid(const Particles &particles) const
Check whether the action still applies.
bool is_pauli_blocked(const std::vector< Particles > &ensembles, const PauliBlocker &p_bl) const
Check if the action is Pauli-blocked.
The Actions class abstracts the storage and manipulation of actions.
ActionPtr pop()
Return the first action in the list and removes it from the list.
double earliest_time() const
Return time of execution of earliest action.
ActionList::size_type size() const
void insert(ActionList &&new_acts)
Insert a list of actions into this object.
static bool is_bremsstrahlung_reaction(const ParticleList &in)
Check if particles can undergo an implemented photon process.
Interface to the SMASH configuration files.
void set_value(Key< U > key, T &&value)
Overwrite the value of the YAML node corresponding to the specified key.
Configuration extract_sub_configuration(KeyLabels section, Configuration::GetEmpty empty_if_not_existing=Configuration::GetEmpty::No)
Create a new configuration from a then-removed section of the present object.
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.
Configuration extract_complete_sub_configuration(KeyLabels section, Configuration::GetEmpty empty_if_not_existing=Configuration::GetEmpty::No)
Alternative method to extract a sub-configuration, which retains the labels from the top-level in the...
T take(const Key< T > &key)
The default interface for SMASH to read configuration values.
A class to pre-calculate and store parameters relevant for density calculation.
Non-template interface to Experiment<Modus>.
static std::unique_ptr< ExperimentBase > create(Configuration &config, const std::filesystem::path &output_path)
Factory method that creates and initializes a new Experiment<Modus>.
virtual ~ExperimentBase()=default
The virtual destructor avoids undefined behavior when destroying derived objects.
virtual void run()=0
Runs the experiment.
The main class, where the simulation of an experiment is executed.
const bool bremsstrahlung_switch_
This indicates whether bremsstrahlung is switched on.
void propagate_and_shine(double to_time, Particles &particles)
Propagate all particles until time to_time without any interactions and shine dileptons.
Experiment(Configuration &config, const std::filesystem::path &output_path)
Create a new Experiment.
const ExpansionProperties metric_
This struct contains information on the metric to be used.
std::unique_ptr< ActionFinderInterface > photon_finder_
The (Scatter) Actions Finder for Direct Photons.
const bool force_decays_
This indicates whether we force all resonances to decay in the last timestep.
double initial_mean_field_energy_
The initial total mean field energy in the system.
void create_output(const std::string &format, const std::string &content, const std::filesystem::path &output_path, const OutputParameters &par)
Create a list of output files.
std::vector< std::unique_ptr< ActionFinderInterface > > action_finders_
The Action finder objects.
bool printout_tmn_
Whether to print the energy-momentum tensor.
QuantumNumbers conserved_initial_
The conserved quantities of the system.
DensityParameters density_param_
Structure to precalculate and hold parameters for density computations.
double next_output_time() const
Shortcut for next output time.
double total_energy_removed_
Total energy removed from the system in hypersurface crossing actions.
DensityType dens_type_lattice_printout_
Type of density for lattice printout.
double max_transverse_distance_sqr_
Maximal distance at which particles can interact in case of the geometric criterion,...
const bool IC_dynamic_
This indicates if the IC is dynamic.
bool printout_j_QBS_
Whether to print the Q, B, S 4-currents.
std::unique_ptr< GrandCanThermalizer > thermalizer_
Instance of class used for forced thermalization.
void count_nonempty_ensembles()
Counts the number of ensembles in wich interactions took place at the end of an event.
const TimeStepMode time_step_mode_
This indicates whether to use time steps.
std::unique_ptr< DensityLattice > jmu_custom_lat_
Custom density on the lattices.
bool printout_full_lattice_any_td_
Whether to print the thermodynamics quantities evaluated on the lattices, point by point,...
void increase_event_number()
Increases the event number by one.
void run_time_evolution_timestepless(Actions &actions, int i_ensemble, const double end_time_propagation)
Performs all the propagations and actions during a certain time interval neglecting the influence of ...
Particles * first_ensemble()
Provides external access to SMASH particles.
void run_time_evolution(const double t_end, ParticleList &&add_plist={}, ParticleList &&remove_plist={})
Runs the time evolution of an event with fixed-size time steps or without timesteps,...
int minimum_nonempty_ensembles_
The number of ensembles, in which interactions take place, to be calculated.
std::unique_ptr< DensityLattice > jmu_B_lat_
Baryon density on the lattice.
DensityType dens_type_
Type of density to be written to collision headers.
void intermediate_output()
Intermediate output during an event.
std::vector< FourVector > beam_momentum_
The initial nucleons in the ColliderModus propagate with beam_momentum_, if Fermi motion is frozen.
std::unique_ptr< RectangularLattice< std::pair< ThreeVector, ThreeVector > > > FI3_lat_
Lattices for the electric and magnetic component of the symmetry force.
std::unique_ptr< RectangularLattice< FourVector > > new_jmu_auxiliary_
Auxiliary lattice for values of jmu at a time step t0 + dt.
double compute_min_cell_length(double dt) const
Calculate the minimal size for the grid cells such that the ScatterActionsFinder will find all collis...
void initialize_new_event()
This is called in the beginning of each event.
bool is_finished()
Checks wether the desired number events have been calculated.
int n_fractional_photons_
Number of fractional photons produced per single reaction.
const double delta_time_startup_
The clock's timestep size at start up.
std::unique_ptr< RectangularLattice< EnergyMomentumTensor > > Tmn_
Lattices of energy-momentum tensors for printout.
std::vector< Particles > ensembles_
Complete particle list, all ensembles in one vector.
std::unique_ptr< RectangularLattice< FourVector > > new_fields_auxiliary_
Auxiliary lattice for values of Amu at a time step t0 + dt.
SystemTimePoint time_start_
system starting time of the simulation
uint64_t previous_wall_actions_total_
Total number of wall-crossings for previous timestep.
const bool IC_switch_
This indicates whether the experiment will be used as initial condition for hydrodynamics.
std::unique_ptr< RectangularLattice< std::pair< ThreeVector, ThreeVector > > > FB_lat_
Lattices for the electric and magnetic components of the Skyrme or VDF force.
OutputsList outputs_
A list of output formaters.
bool printout_rho_eckart_
Whether to print the Eckart rest frame density.
std::unique_ptr< RectangularLattice< std::array< FourVector, 4 > > > fields_four_gradient_auxiliary_
Auxiliary lattice for calculating the four-gradient of Amu.
void do_final_decays()
Performs the final decays of an event.
std::unique_ptr< PauliBlocker > pauli_blocker_
An instance of PauliBlocker class that stores parameters needed for Pauli blocking calculations and c...
bool printout_v_landau_
Whether to print the 4-velocity in Landau frame.
bool perform_action(Action &action, int i_ensemble, bool include_pauli_blocking=true)
Perform the given action.
std::unique_ptr< RectangularLattice< FourVector > > UI3_lat_
Lattices for symmetry potentials (evaluated in the local rest frame) times the isospin flow 4-velocit...
bool printout_lattice_td_
Whether to print the thermodynamics quantities evaluated on the lattices.
std::unique_ptr< RectangularLattice< std::pair< ThreeVector, ThreeVector > > > EM_lat_
Lattices for electric and magnetic field in fm^-2.
std::unique_ptr< FieldsLattice > fields_lat_
Mean-field A^mu on the lattice.
int max_events_
Maximum number of events to be calculated in order obtain the desired number of non-empty events usin...
int nonempty_ensembles_
Number of ensembles containing an interaction.
OutputPtr photon_output_
The Photon output.
EventCounting event_counting_
The way in which the number of calculated events is specified.
const bool dileptons_switch_
This indicates whether dileptons are switched on.
Modus modus_
Instance of the Modus template parameter.
std::unique_ptr< RectangularLattice< FourVector > > old_jmu_auxiliary_
Auxiliary lattice for values of jmu at a time step t0.
std::unique_ptr< DecayActionsFinderDilepton > dilepton_finder_
The Dilepton Action Finder.
std::unique_ptr< DensityLattice > j_QBS_lat_
4-current for j_QBS lattice output
bool printout_tmn_landau_
Whether to print the energy-momentum tensor in Landau frame.
std::unique_ptr< RectangularLattice< std::array< FourVector, 4 > > > four_gradient_auxiliary_
Auxiliary lattice for calculating the four-gradient of jmu.
const bool photons_switch_
This indicates whether photons are switched on.
StringProcess * process_string_ptr_
Pointer to the string process class object, which is used to set the random seed for PYTHIA objects i...
std::unique_ptr< RectangularLattice< FourVector > > UB_lat_
Lattices for Skyrme or VDF potentials (evaluated in the local rest frame) times the baryon flow 4-vel...
ExperimentParameters parameters_
Struct of several member variables.
std::unique_ptr< RectangularLattice< FourVector > > old_fields_auxiliary_
Auxiliary lattice for values of Amu at a time step t0.
std::unique_ptr< DensityLattice > jmu_el_lat_
Electric charge density on the lattice.
std::unique_ptr< Potentials > potentials_
An instance of potentials class, that stores parameters of potentials, calculates them and their grad...
uint64_t wall_actions_total_
Total number of wall-crossings for current timestep.
uint64_t total_hypersurface_crossing_actions_
Total number of particles removed from the evolution in hypersurface crossing actions.
uint64_t interactions_total_
Total number of interactions for current timestep.
const bool use_grid_
This indicates whether to use the grid.
std::unique_ptr< DensityLattice > jmu_I3_lat_
Isospin projection density on the lattice.
int nevents_
Number of events.
uint64_t total_pauli_blocked_
Total number of Pauli-blockings for current timestep.
void final_output()
Output at the end of an event.
std::vector< bool > projectile_target_interact_
Whether the projectile and the target collided.
Modus * modus()
Provides external access to SMASH calculation modus.
void update_potentials()
Recompute potentials on lattices if necessary.
OutputPtr dilepton_output_
The Dilepton output.
int64_t seed_
random seed for the next event.
std::vector< Particles > * all_ensembles()
Getter for all ensembles.
uint64_t previous_interactions_total_
Total number of interactions for previous timestep.
uint64_t discarded_interactions_total_
Total number of discarded interactions, because they were invalidated before they could be performed.
void run() override
Runs the experiment.
bool kinematic_cuts_for_IC_output_
This indicates whether kinematic cuts are enabled for the IC output.
const double end_time_
simulation time at which the evolution is stopped.
double total_energy_violated_by_Pythia_
Total energy violation introduced by Pythia.
The FourVector class holds relevant values in Minkowski spacetime with (+, −, −, −) metric signature.
ParticleData contains the dynamic information of a certain particle.
static double formation_power_
Power with which the cross section scaling factor grows in time.
The Particles class abstracts the storage and manipulation of particles.
A class that stores parameters of potentials, calculates potentials and their gradients.
static ThreeVector B_field_integrand(ThreeVector pos, DensityOnLattice &charge_density, ThreeVector point)
Integrand for calculating the magnetic field using the Biot-Savart formula.
static ThreeVector E_field_integrand(ThreeVector pos, DensityOnLattice &charge_density, ThreeVector point)
Integrand for calculating the electric field.
A container for storing conserved values.
A container class to hold all the arrays on the lattice and access them.
static bool is_photon_reaction(const ParticleList &in)
Check if particles can undergo an implemented photon process.
static bool is_kinematically_possible(const double s_sqrt, const ParticleList &in)
Check if CM-energy is sufficient to produce hadron in final state.
String excitation processes used in SMASH.
ThermalizationAction implements forced thermalization as an Action class.
bool any_particles_thermalized() const
This method checks, if there are particles in the region to be thermalized.
The ThreeVector class represents a physical three-vector with the components .
@ Frozen
Use fermi motion without potentials.
@ Dynamic
Dynamic fluidization based on local densities.
TimeStepMode
The time step mode.
@ Fixed
Use fixed time step.
@ None
Don't use time steps; propagate from action to action.
@ EckartDensity
Density in the Eckart frame.
@ Tmn
Energy-momentum tensor in lab frame.
@ LandauVelocity
Velocity of the Landau rest frame.
@ j_QBS
Electric (Q), baryonic (B) and strange (S) currents.
@ TmnLandau
Energy-momentum tensor in Landau rest frame.
@ BottomUp
Sum the existing partial contributions.
@ Stochastic
Stochastic Criteiron.
@ None
No pseudo-resonance is created.
EventCounting
Defines how the number of events is determined.
@ Invalid
Unused, only in the code for internal logic.
@ FixedNumber
The desired number of events is simulated disregarding of whether an interaction took place.
@ MinimumNonEmpty
Events are simulated until there are at least a given number of ensembles in which an interaction too...
DensityType
Allows to choose which kind of density to calculate.
#define SMASH_SOURCE_LOCATION
Hackery that is required to output the location in the source code where the log statement occurs.
std::ostream & operator<<(std::ostream &out, const ActionPtr &action)
Convenience: dereferences the ActionPtr to Action.
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...
std::unique_ptr< OutputInterface > create_oscar_output(const std::string &format, const std::string &content, const std::filesystem::path &path, const OutputParameters &out_par)
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.
#define likely(x)
Tell the branch predictor that this expression is likely true.
Engine::result_type advance()
Advance the engine's state and return the generated value.
void set_seed(T &&seed)
Sets the seed of the random number engine.
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
void update_momenta(std::vector< Particles > &particles, double dt, const Potentials &pot, RectangularLattice< std::pair< ThreeVector, ThreeVector >> *FB_lat, RectangularLattice< std::pair< ThreeVector, ThreeVector >> *FI3_lat, RectangularLattice< std::pair< ThreeVector, ThreeVector >> *EM_lat, DensityLattice *jB_lat)
Updates the momenta of all particles at the current time step according to the equations of motion:
UnaryFunction for_each(Container &&c, UnaryFunction &&f)
Convenience wrapper for std::for_each that operates on a complete container.
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_.
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.
void expand_space_time(Particles *particles, const ExperimentParameters ¶meters, const ExpansionProperties &metric)
Modifies positions and momentum of all particles to account for space-time deformation.
void check_interactions_total(uint64_t interactions_total)
Make sure interactions_total can be represented as a 32-bit integer.
@ Wall
See here for a short description.
@ HyperSurfaceCrossing
See here for a short description.
std::tuple< double, FourVector, ThreeVector, ThreeVector, FourVector, FourVector, FourVector, FourVector > current_eckart(const ThreeVector &r, const ParticleList &plist, const DensityParameters &par, DensityType dens_type, bool compute_gradient, bool smearing)
Calculates Eckart rest frame density and 4-current of a given density type and optionally the gradien...
double propagate_straight_line(Particles *particles, double to_time, const std::vector< FourVector > &beam_momentum)
Propagates the positions of all particles on a straight line to a given moment.
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 update_fields_lattice(RectangularLattice< FieldsOnLattice > *fields_lat, RectangularLattice< FourVector > *old_fields, RectangularLattice< FourVector > *new_fields, RectangularLattice< std::array< FourVector, 4 >> *fields_four_grad_lattice, DensityLattice *jmu_B_lat, const LatticeUpdate fields_lat_update, const Potentials &potentials, const double time_step)
Updates the contents on the lattice of FieldsOnLattice type.
bool are_particles_identical_at_given_time(const ParticleData &p1, const ParticleData &p2, double time)
Utility function to compare two ParticleData instances with respect to their PDG code,...
void update_lattice_accumulating_ensembles(RectangularLattice< T > *lat, const LatticeUpdate update, const DensityType dens_type, const DensityParameters &par, const std::vector< Particles > &ensembles, const bool compute_gradient)
Updates the contents on the lattice when ensembles are used.
void validate_and_adjust_particle_list(ParticleList &particle_list)
Validate a particle list adjusting each particle to be a valid SMASH particle.
constexpr double nucleon_mass
Nucleon mass in GeV.
void update_lattice(RectangularLattice< DensityOnLattice > *lat, RectangularLattice< FourVector > *old_jmu, RectangularLattice< FourVector > *new_jmu, RectangularLattice< std::array< FourVector, 4 >> *four_grad_lattice, const LatticeUpdate update, const DensityType dens_type, const DensityParameters &par, const std::vector< Particles > &ensembles, const double time_step, const bool compute_gradient)
Updates the contents on the lattice of DensityOnLattice type.
LatticeUpdate
Enumerator option for lattice updates.
std::ostream & operator<<(std::ostream &out, const Experiment< Modus > &e)
Creates a verbose textual description of the setup of the Experiment.
Potentials * pot_pointer
Pointer to a Potential class.
static constexpr int LMain
constexpr double really_small
Numerical error tolerance.
RectangularLattice< FourVector > * UB_lat_pointer
Pointer to the skyrme potential on the lattice.
@ Largest
Make cells as large as possible.
std::chrono::time_point< std::chrono::system_clock > SystemTimePoint
Type (alias) that is used to store the current time.
const std::string hline(113, '-')
String representing a horizontal line.
RectangularLattice< FourVector > * UI3_lat_pointer
Pointer to the symmmetry potential on the lattice.
constexpr double fm2_mb
mb <-> fm^2 conversion factor.
Structure to contain custom data for output.
Struct containing the type of the metric and the expansion parameter of the metric.
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 fixed_min_cell_length
Fixed minimal grid cell length (in fm).
const CollisionCriterion coll_crit
Employed collision criterion.
std::unique_ptr< Clock > outputclock
Output clock to keep track of the next output time.
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.
RivetOutputParameters rivet_parameters
Rivet specfic parameters.