Version: SMASH-3.2
spheremodus.cc
Go to the documentation of this file.
1 /*
2  *
3  * Copyright (c) 2012-2024
4  * SMASH Team
5  *
6  * GNU General Public License (GPLv3 or later)
7  *
8  */
9 
10 #include "smash/spheremodus.h"
11 
12 #include <cmath>
13 #include <cstdio>
14 #include <cstdlib>
15 #include <list>
16 #include <map>
17 #include <utility>
18 #include <vector>
19 
20 #include "smash/angles.h"
22 #include "smash/configuration.h"
23 #include "smash/constants.h"
24 #include "smash/cxx17compat.h"
26 #include "smash/fourvector.h"
27 #include "smash/hadgas_eos.h"
28 #include "smash/logging.h"
29 #include "smash/particles.h"
30 #include "smash/quantumsampling.h"
31 #include "smash/random.h"
32 #include "smash/threevector.h"
33 
34 namespace smash {
35 static constexpr int LSphere = LogArea::Sphere::id;
36 
38  const ExperimentParameters &)
39  : radius_(modus_config.take(InputKeys::modi_sphere_radius)),
40  sphere_temperature_(
41  modus_config.take(InputKeys::modi_sphere_temperature)),
42  start_time_(modus_config.take(InputKeys::modi_sphere_startTime)),
43  use_thermal_(
44  modus_config.take(InputKeys::modi_sphere_useThermalMultiplicities)),
45  mub_(modus_config.take(InputKeys::modi_sphere_baryonChemicalPotential)),
46  mus_(modus_config.take(InputKeys::modi_sphere_strangeChemicalPotential)),
47  muq_(modus_config.take(InputKeys::modi_sphere_chargeChemicalPotential)),
48  account_for_resonance_widths_(
49  modus_config.take(InputKeys::modi_sphere_accountResonanceWidths)),
50  init_multipl_(use_thermal_
51  ? std::map<PdgCode, int>()
52  : modus_config.take(
53  InputKeys::modi_sphere_initialMultiplicities)),
54  init_distr_(modus_config.take(InputKeys::modi_sphere_initialCondition)),
55  radial_velocity_(
56  modus_config.take(InputKeys::modi_sphere_addRadialVelocity)),
57  /* Note that it is crucial not to take other keys from the Jet section
58  * before Jet_PDG, since we want here the take to throw in case the user
59  * had a Jet section without the mandatory Jet_PDG key. If all other keys
60  * are taken first, the section is removed from the config because empty,
61  * and has_section(InputSections::m_s_jet) method would return false.
62  */
63  jet_pdg_(modus_config.has_section(InputSections::m_s_jet)
64  ? make_optional<PdgCode>(
65  modus_config.take(InputKeys::modi_sphere_jet_jetPdg))
66  : std::nullopt),
67 
68  jet_mom_(modus_config.take(InputKeys::modi_sphere_jet_jetMomentum)) {}
69 
70 /* console output on startup of sphere specific parameters */
71 std::ostream &operator<<(std::ostream &out, const SphereModus &m) {
72  out << "-- Sphere Modus:\nRadius of the sphere: " << m.radius_ << " fm\n";
73  if (m.use_thermal_) {
74  out << "Thermal multiplicities (T = " << m.sphere_temperature_
75  << " GeV, muB = " << m.mub_ << " GeV, muS = " << m.mus_
76  << " GeV, muQ = " << m.muq_ << " GeV)\n";
77  } else {
78  for (const auto &p : m.init_multipl_) {
79  ParticleTypePtr ptype = &ParticleType::find(p.first);
80  out << ptype->name() << " initial multiplicity " << p.second << '\n';
81  }
82  }
83  switch (m.init_distr_) {
85  out << "Boltzmann momentum distribution with T = "
86  << m.sphere_temperature_ << " GeV.\n";
87  break;
89  out << "Fermi/Bose momentum distribution with T = "
90  << m.sphere_temperature_ << " GeV.\n";
91  break;
93  out << "Sphere Initial Condition is IC_ES";
94  break;
96  out << "Sphere Initial Condition is IC_1M";
97  break;
99  out << "Sphere Initial Condition is IC_2M";
100  break;
102  out << "Sphere Initial Condition is IC_Massive";
103  break;
104  }
105  if (m.jet_pdg_) {
106  ParticleTypePtr ptype = &ParticleType::find(m.jet_pdg_.value());
107  out << "Adding a " << ptype->name() << " as a jet in the middle "
108  << "of the sphere with " << m.jet_mom_ << " GeV initial momentum.\n";
109  }
110  return out;
111 }
112 
113 /* initial_conditions - sets particle data for @particles */
115  const ExperimentParameters &parameters) {
116  FourVector momentum_total(0, 0, 0, 0);
117  const double T = this->sphere_temperature_;
118  const double V = 4.0 / 3.0 * M_PI * radius_ * radius_ * radius_;
119  /* Create NUMBER OF PARTICLES according to configuration */
120  if (use_thermal_) {
121  if (average_multipl_.empty()) {
122  for (const ParticleType &ptype : ParticleType::list_all()) {
123  if (HadronGasEos::is_eos_particle(ptype)) {
124  const double n = HadronGasEos::partial_density(
126  average_multipl_[ptype.pdgcode()] = n * V * parameters.testparticles;
127  }
128  }
129  }
130  double nb_init = 0.0, ns_init = 0.0, nq_init = 0.0;
131  for (const auto &mult : average_multipl_) {
132  const int thermal_mult_int = random::poisson(mult.second);
133  particles->create(thermal_mult_int, mult.first);
134  nb_init += mult.second * mult.first.baryon_number();
135  ns_init += mult.second * mult.first.strangeness();
136  nq_init += mult.second * mult.first.charge();
137  logg[LSphere].debug(mult.first, " initial multiplicity ",
138  thermal_mult_int);
139  }
140  logg[LSphere].info("Initial hadron gas baryon density ", nb_init);
141  logg[LSphere].info("Initial hadron gas strange density ", ns_init);
142  logg[LSphere].info("Initial hadron gas charge density ", nq_init);
143  } else {
144  for (const auto &p : init_multipl_) {
145  particles->create(p.second * parameters.testparticles, p.first);
146  logg[LSphere].debug("Particle ", p.first, " initial multiplicity ",
147  p.second);
148  }
149  }
150  std::unique_ptr<QuantumSampling> quantum_sampling;
152  quantum_sampling = std::make_unique<QuantumSampling>(init_multipl_, V, T);
153  }
154  /* loop over particle data to fill in momentum and position information */
155  for (ParticleData &data : *particles) {
156  Angles phitheta;
157  /* thermal momentum according Maxwell-Boltzmann distribution */
158  double momentum_radial = 0.0, mass = data.pole_mass();
159  /* assign momentum_radial according to requested distribution */
160  switch (init_distr_) {
162  momentum_radial = sample_momenta_IC_ES(T);
163  break;
165  momentum_radial = sample_momenta_1M_IC(T, mass);
166  break;
168  momentum_radial = sample_momenta_2M_IC(T, mass);
169  break;
171  momentum_radial = sample_momenta_non_eq_mass(T, mass);
172  break;
174  default:
176  ? data.type().mass()
177  : HadronGasEos::sample_mass_thermal(data.type(), 1.0 / T);
178  momentum_radial = sample_momenta_from_thermal(T, mass);
179  break;
181  /*
182  * **********************************************************************
183  * Sampling the thermal momentum according Bose/Fermi/Boltzmann
184  * distribution.
185  * We take the pole mass as the mass.
186  * **********************************************************************
187  */
188  mass = data.type().mass();
189  momentum_radial = quantum_sampling->sample(data.pdgcode());
190  break;
191  }
192  phitheta.distribute_isotropically();
193  logg[LSphere].debug(data.type().name(), "(id ", data.id(),
194  ") radial momentum ", momentum_radial, ", direction",
195  phitheta);
196  data.set_4momentum(mass, phitheta.threevec() * momentum_radial);
197  momentum_total += data.momentum();
198  /* uniform sampling in a sphere with radius r */
199  double position_radial;
200  position_radial = std::cbrt(random::canonical()) * radius_;
201  Angles pos_phitheta;
202  pos_phitheta.distribute_isotropically();
203  data.set_4position(
204  FourVector(start_time_, pos_phitheta.threevec() * position_radial));
205  data.set_formation_time(start_time_);
206  }
207 
208  /* boost in radial direction with an underlying velocity field of the form u_r
209  * = u_0 * r / R */
210  if (radial_velocity_ > 0.0) {
211  if (radial_velocity_ > 1.0) {
212  throw std::invalid_argument(
213  "Additional velocity cannot be greater than 1!");
214  }
215  for (ParticleData &data : *particles) {
216  double particle_radius = std::sqrt(data.position().sqr3());
217  auto e_r = data.position().threevec() / particle_radius;
218  auto radial_velocity =
219  -1.0 * radial_velocity_ * e_r * particle_radius / radius_;
220  data.set_4momentum(data.momentum().lorentz_boost(radial_velocity));
221  momentum_total += data.momentum();
222  }
223  }
224 
225  /* Make total 3-momentum 0 */
226  for (ParticleData &data : *particles) {
227  data.set_4momentum(data.momentum().abs(),
228  data.momentum().threevec() -
229  momentum_total.threevec() / particles->size());
230  }
231 
232  /* Add a single highly energetic particle in the center of the sphere (jet) */
233  if (jet_pdg_) {
234  auto &jet_particle = particles->create(jet_pdg_.value());
235  jet_particle.set_formation_time(start_time_);
236  jet_particle.set_4position(FourVector(start_time_, 0., 0., 0.));
237  jet_particle.set_4momentum(ParticleType::find(jet_pdg_.value()).mass(),
238  ThreeVector(jet_mom_, 0., 0.));
239  }
240 
241  /* Recalculate total momentum */
242  momentum_total = FourVector(0, 0, 0, 0);
243  for (ParticleData &data : *particles) {
244  momentum_total += data.momentum();
245  /* IC: debug checks */
246  logg[LSphere].debug() << data;
247  }
248  /* allows to check energy conservation */
249  logg[LSphere].debug() << "Sphere initial total 4-momentum [GeV]: "
250  << momentum_total;
251  return start_time_;
252 }
253 } // namespace smash
Angles provides a common interface for generating directions: i.e., two angles that should be interpr...
Definition: angles.h:59
ThreeVector threevec() const
Definition: angles.h:288
void distribute_isotropically()
Populate the object with a new direction.
Definition: angles.h:199
Interface to the SMASH configuration files.
The FourVector class holds relevant values in Minkowski spacetime with (+, −, −, −) metric signature.
Definition: fourvector.h:33
ThreeVector threevec() const
Definition: fourvector.h:329
static double partial_density(const ParticleType &ptype, double T, double mub, double mus, double muq, bool account_for_resonance_widths=false)
Compute partial density of one hadron sort.
Definition: hadgas_eos.cc:270
static double sample_mass_thermal(const ParticleType &ptype, double beta)
Sample resonance mass in a thermal medium.
Definition: hadgas_eos.cc:385
static bool is_eos_particle(const ParticleType &ptype)
Check if a particle belongs to the EoS.
Definition: hadgas_eos.h:355
ParticleData contains the dynamic information of a certain particle.
Definition: particledata.h:58
A pointer-like interface to global references to ParticleType objects.
Definition: particletype.h:679
Particle type contains the static properties of a particle species.
Definition: particletype.h:98
static const ParticleType & find(PdgCode pdgcode)
Returns the ParticleType object for the given pdgcode.
Definition: particletype.cc:99
const std::string & name() const
Definition: particletype.h:142
static const ParticleTypeList & list_all()
Definition: particletype.cc:51
double mass() const
Definition: particletype.h:145
The Particles class abstracts the storage and manipulation of particles.
Definition: particles.h:33
size_t size() const
Definition: particles.h:87
void create(size_t n, PdgCode pdg)
Add n particles of the same type (pdg) to the list.
Definition: particles.cc:66
PdgCode stores a Particle Data Group Particle Numbering Scheme particle type number.
Definition: pdgcode.h:111
SphereModus: Provides a modus for expanding matter calculations.
Definition: spheremodus.h:49
const bool account_for_resonance_widths_
In case of thermal initialization: true – account for resonance spectral functions,...
Definition: spheremodus.h:115
const bool use_thermal_
Whether to use a thermal initialization for all particles instead of specific numbers.
Definition: spheremodus.h:94
const double muq_
Charge chemical potential for thermal initialization; only used if use_thermal_ is true.
Definition: spheremodus.h:109
double sphere_temperature_
Temperature for momentum distribution (in GeV)
Definition: spheremodus.h:87
const double start_time_
Starting time for the Sphere.
Definition: spheremodus.h:89
const SphereInitialCondition init_distr_
Initialization scheme for momenta in the sphere; used for expanding metric setup.
Definition: spheremodus.h:130
const std::optional< PdgCode > jet_pdg_
Optional PDG code of the particle to use as a jet, i.e.
Definition: spheremodus.h:144
SphereModus(Configuration modus_config, const ExperimentParameters &parameters)
Constructor.
Definition: spheremodus.cc:37
double initial_conditions(Particles *particles, const ExperimentParameters &parameters)
Generates initial state of the particles in the system according to specified parameters: number of p...
Definition: spheremodus.cc:114
std::map< PdgCode, double > average_multipl_
Average multiplicities in case of thermal initialization.
Definition: spheremodus.h:125
const double mub_
Baryon chemical potential for thermal initialization; only used if use_thermal_ is true.
Definition: spheremodus.h:99
const double jet_mom_
Initial momentum of the jet particle; only used if jet_pdg_ is not nullopt.
Definition: spheremodus.h:148
const std::map< PdgCode, int > init_multipl_
Particle multiplicities at initialization; required if use_thermal_ is false.
Definition: spheremodus.h:120
const double mus_
Strange chemical potential for thermal initialization; only used if use_thermal_ is true.
Definition: spheremodus.h:104
const double radial_velocity_
Wether to add a constant radial velocity profile to the momenta of the particles in the sphere.
Definition: spheremodus.h:136
double radius_
Sphere radius (in fm)
Definition: spheremodus.h:85
The ThreeVector class represents a physical three-vector with the components .
Definition: threevector.h:31
Collection of useful constants that are known at compile time.
@ ThermalMomentaBoltzmann
A thermalized ensemble is generated, with momenta sampled from a Maxwell-Boltzmann distribution.
@ IC_ES
Off-equilibrium distribution used in massless comparisons of SMASH to the extended universe metric.
@ ThermalMomentaQuantum
A thermalized ensemble is generated, with momenta of baryons(mesons) sampled from a Fermi(Bose) distr...
@ IC_Massive
A generalization of IC_ES for the non-zero mass case; note that there is currently no analytical comp...
@ IC_2M
Off-equilibrium distribution used in massless comparisons of SMASH to the extended universe metric.
@ IC_1M
Off-equilibrium distribution used in massless comparisons of SMASH to the extended universe metric.
std::ostream & operator<<(std::ostream &out, const ActionPtr &action)
Convenience: dereferences the ActionPtr to Action.
Definition: action.h:546
std::array< einhard::Logger<>, std::tuple_size< LogArea::AreaTuple >::value > logg
An array that stores all pre-configured Logger objects.
Definition: logging.cc:40
constexpr int p
Proton.
constexpr int n
Neutron.
int poisson(const T &lam)
Returns a Poisson distributed random number.
Definition: random.h:226
T canonical()
Definition: random.h:113
Definition: action.h:24
static constexpr int LSphere
Definition: spheremodus.cc:35
double sample_momenta_from_thermal(const double temperature, const double mass)
Samples a momentum from the Maxwell-Boltzmann (thermal) distribution in a faster way,...
double sample_momenta_IC_ES(const double temperature)
Sample momenta according to the momentum distribution in Bazow:2016oky .
double sample_momenta_non_eq_mass(const double temperature, const double mass)
Samples a momentum via rejection method from the non-equilibrium distribution.
double sample_momenta_1M_IC(const double temperature, const double mass)
Samples a momentum from the non-equilibrium distribution 1M_IC from Bazow:2016oky .
double sample_momenta_2M_IC(const double temperature, const double mass)
Samples a momentum from the non-equilibrium distribution 2M_IC from Bazow:2016oky .
Helper structure for Experiment.
int testparticles
Number of test-particles.
A container to keep track of all ever existed input keys.
Definition: input_keys.h:1067
A container to keep track of all ever existed sections in the input file.
Definition: input_keys.h:48