Version: SMASH-3.2
potentials.cc
Go to the documentation of this file.
1 /*
2  *
3  * Copyright (c) 2014-2015,2017-2024
4  * SMASH Team
5  *
6  * GNU General Public License (GPLv3 or later)
7  *
8  */
9 
10 #include "smash/potentials.h"
11 
12 #include "smash/constants.h"
13 #include "smash/density.h"
14 #include "smash/input_keys.h"
15 
16 namespace smash {
17 
19  : param_(param),
20  use_skyrme_(conf.has_section(InputSections::p_skyrme)),
21  use_symmetry_(conf.has_section(InputSections::p_symmetry)),
22  use_coulomb_(conf.has_section(InputSections::p_coulomb)),
23  use_vdf_(conf.has_section(InputSections::p_vdf)),
24  use_momentum_dependence_(
25  conf.has_section(InputSections::p_momentumDependence)),
26  use_potentials_outside_lattice_(
27  conf.take(InputKeys::potentials_use_potentials_outside_lattice)) {
28  if (use_skyrme_) {
32  }
37  }
38 
39  if (use_symmetry_) {
44  }
45  }
46  if (use_coulomb_) {
48  }
49  if (use_vdf_) {
51  std::vector<double> aux_coeffs =
53  std::vector<double> aux_powers =
55  if (aux_coeffs.size() != aux_powers.size()) {
56  throw std::invalid_argument(
57  "The number of coefficients should equal the number of powers.");
58  }
59  const int n_terms = aux_powers.size();
60  for (int i = 0; i < n_terms; i++) {
61  if (aux_powers[i] < 0.0) {
62  throw std::invalid_argument("Powers need to be positive real numbers.");
63  }
64  // coefficients are provided in MeV, but the code uses GeV
65  coeffs_.push_back(aux_coeffs[i] * mev_to_gev);
66  powers_.push_back(aux_powers[i]);
67  }
68  }
69 }
70 
72 
73 double Potentials::skyrme_pot(const double baryon_density, const double A,
74  const double B, const double tau) {
75  const double tmp = baryon_density / nuclear_density;
76  /* U = U(|rho|) * sgn , because the sign of the potential changes
77  * under a charge reversal transformation. */
78  const int sgn = tmp > 0 ? 1 : -1;
79  // Return in GeV
80  return mev_to_gev * sgn *
81  (A * std::abs(tmp) + B * std::pow(std::abs(tmp), tau));
82 }
83 
84 double Potentials::symmetry_S(const double baryon_density) const {
86  return 12.3 * std::pow(baryon_density / nuclear_density, 2. / 3.) +
87  20.0 * std::pow(baryon_density / nuclear_density, symmetry_gamma_);
88  } else {
89  return 0.;
90  }
91 }
92 double Potentials::symmetry_pot(const double baryon_isospin_density,
93  const double baryon_density) const {
94  double pot = mev_to_gev * 2. * symmetry_S_Pot_ * baryon_isospin_density /
97  pot += mev_to_gev * symmetry_S(baryon_density) * baryon_isospin_density *
98  baryon_isospin_density / (baryon_density * baryon_density);
99  }
100  return pot;
101 }
102 
103 FourVector Potentials::vdf_pot(double rhoB, const FourVector jmuB_net) const {
104  // this needs to be used in order to prevent trying to calculate something
105  // like
106  // (-rho_B)^{3.4}
107  const int sgn = rhoB > 0 ? 1 : -1;
108  double abs_rhoB = std::abs(rhoB);
109  // to prevent NAN expressions
110  if (abs_rhoB < very_small_double) {
111  abs_rhoB = very_small_double;
112  }
113  // F_2 is a multiplicative factor in front of the baryon current
114  // in the VDF potential
115  double F_2 = 0.0;
116  for (int i = 0; i < number_of_terms(); i++) {
117  F_2 += coeffs_[i] * std::pow(abs_rhoB, powers_[i] - 2.0) /
118  std::pow(saturation_density_, powers_[i] - 1.0);
119  }
120  F_2 = F_2 * sgn;
121  // Return in GeV
122  return F_2 * jmuB_net;
123 }
124 
125 double Potentials::potential(const ThreeVector &r, const ParticleList &plist,
126  const ParticleType &acts_on) const {
127  double total_potential = 0.0;
128  const bool compute_gradient = false;
129  const bool smearing = true;
130  const auto scale = force_scale(acts_on);
131 
132  if (!(acts_on.is_baryon() || acts_on.is_nucleus())) {
133  return total_potential;
134  }
135  const auto baryon_density_and_gradient = current_eckart(
136  r, plist, param_, DensityType::Baryon, compute_gradient, smearing);
137  const double rhoB = std::get<0>(baryon_density_and_gradient);
138  if (use_skyrme_) {
139  total_potential += scale.first * skyrme_pot(rhoB);
140  }
141  if (use_symmetry_) {
142  const double rho_iso = std::get<0>(
144  compute_gradient, smearing));
145  const double sym_pot = symmetry_pot(rho_iso, rhoB) * acts_on.isospin3_rel();
146  total_potential += scale.second * sym_pot;
147  }
148 
149  if (use_vdf_) {
150  const FourVector jmuB = std::get<1>(baryon_density_and_gradient);
151  const FourVector VDF_potential = vdf_pot(rhoB, jmuB);
152  total_potential += scale.first * VDF_potential.x0();
153  }
154 
155  return total_potential;
156 }
157 
158 std::pair<double, int> Potentials::force_scale(const ParticleType &data) {
159  const auto &pdg = data.pdgcode();
160  const double skyrme_or_VDF_scale =
161  (3 - std::abs(pdg.strangeness())) / 3. * pdg.baryon_number();
162  const int symmetry_scale = pdg.baryon_number();
163  return std::make_pair(skyrme_or_VDF_scale, symmetry_scale);
164 }
165 
166 std::pair<ThreeVector, ThreeVector> Potentials::skyrme_force(
167  const double rhoB, const ThreeVector grad_j0B, const ThreeVector dvecjB_dt,
168  const ThreeVector curl_vecjB) const {
169  ThreeVector E_component(0.0, 0.0, 0.0), B_component(0.0, 0.0, 0.0);
170  if (use_skyrme_) {
171  const int sgn = rhoB > 0 ? 1 : -1;
172  const double abs_rhoB = std::abs(rhoB);
173  const double dV_drho = sgn *
175  std::pow(abs_rhoB / nuclear_density,
176  skyrme_tau_ - 1)) *
178  E_component -= dV_drho * (grad_j0B + dvecjB_dt);
179  B_component += dV_drho * curl_vecjB;
180  }
181  return std::make_pair(E_component, B_component);
182 }
183 
184 std::pair<ThreeVector, ThreeVector> Potentials::symmetry_force(
185  const double rhoI3, const ThreeVector grad_j0I3,
186  const ThreeVector dvecjI3_dt, const ThreeVector curl_vecjI3,
187  const double rhoB, const ThreeVector grad_j0B, const ThreeVector dvecjB_dt,
188  const ThreeVector curl_vecjB) const {
189  ThreeVector E_component(0.0, 0.0, 0.0), B_component(0.0, 0.0, 0.0);
190  if (use_symmetry_) {
191  E_component -= dVsym_drhoI3(rhoB, rhoI3) * (grad_j0I3 + dvecjI3_dt) +
192  dVsym_drhoB(rhoB, rhoI3) * (grad_j0B + dvecjB_dt);
193  B_component += dVsym_drhoI3(rhoB, rhoI3) * curl_vecjI3 +
194  dVsym_drhoB(rhoB, rhoI3) * curl_vecjB;
195  }
196  return std::make_pair(E_component, B_component);
197 }
198 
199 std::pair<ThreeVector, ThreeVector> Potentials::vdf_force(
200  double rhoB, const double drhoB_dt, const ThreeVector grad_rhoB,
201  const ThreeVector gradrhoB_cross_vecjB, const double j0B,
202  const ThreeVector grad_j0B, const ThreeVector vecjB,
203  const ThreeVector dvecjB_dt, const ThreeVector curl_vecjB) const {
204  // this needs to be used to prevent trying to calculate something like
205  // (-rhoB)^{3.4}
206  const int sgn = rhoB > 0 ? 1 : -1;
207  ThreeVector E_component(0.0, 0.0, 0.0), B_component(0.0, 0.0, 0.0);
208  // to prevent NAN expressions
209  double abs_rhoB = std::abs(rhoB);
210  if (abs_rhoB < very_small_double) {
211  abs_rhoB = very_small_double;
212  }
213  if (use_vdf_) {
214  // F_1 and F_2 are multiplicative factors in front of the baryon current
215  // in the VDF potential
216  double F_1 = 0.0;
217  for (int i = 0; i < number_of_terms(); i++) {
218  F_1 += coeffs_[i] * (powers_[i] - 2.0) *
219  std::pow(abs_rhoB, powers_[i] - 3.0) /
220  std::pow(saturation_density_, powers_[i] - 1.0);
221  }
222  F_1 = F_1 * sgn;
223 
224  double F_2 = 0.0;
225  for (int i = 0; i < number_of_terms(); i++) {
226  F_2 += coeffs_[i] * std::pow(abs_rhoB, powers_[i] - 2.0) /
227  std::pow(saturation_density_, powers_[i] - 1.0);
228  }
229  F_2 = F_2 * sgn;
230 
231  E_component -= (F_1 * (grad_rhoB * j0B + drhoB_dt * vecjB) +
232  F_2 * (grad_j0B + dvecjB_dt));
233  B_component += F_1 * gradrhoB_cross_vecjB + F_2 * curl_vecjB;
234  }
235  return std::make_pair(E_component, B_component);
236 }
237 
238 // overload of the above
239 std::pair<ThreeVector, ThreeVector> Potentials::vdf_force(
240  const ThreeVector grad_A_0, const ThreeVector dA_dt,
241  const ThreeVector curl_A) const {
242  ThreeVector E_component(0.0, 0.0, 0.0), B_component(0.0, 0.0, 0.0);
243  if (use_vdf_) {
244  E_component -= (grad_A_0 + dA_dt);
245  B_component += curl_A;
246  }
247  return std::make_pair(E_component, B_component);
248 }
249 
250 double Potentials::dVsym_drhoI3(const double rhoB, const double rhoI3) const {
251  double term1 = 2. * symmetry_S_Pot_ / nuclear_density;
253  double term2 = 2. * rhoI3 * symmetry_S(rhoB) / (rhoB * rhoB);
254  return mev_to_gev * (term1 + term2);
255  } else {
256  return mev_to_gev * term1;
257  }
258 }
259 
260 double Potentials::dVsym_drhoB(const double rhoB, const double rhoI3) const {
262  double rhoB_over_rho0 = rhoB / nuclear_density;
263  double term1 = 8.2 * std::pow(rhoB_over_rho0, -1. / 3.) / nuclear_density +
264  20. * symmetry_gamma_ *
265  std::pow(rhoB_over_rho0, symmetry_gamma_) / rhoB;
266  double term2 = -2. * symmetry_S(rhoB) / rhoB;
267  return mev_to_gev * (term1 + term2) * rhoI3 * rhoI3 / (rhoB * rhoB);
268  } else {
269  return 0.;
270  }
271 }
272 
273 std::tuple<ThreeVector, ThreeVector, ThreeVector, ThreeVector>
274 Potentials::all_forces(const ThreeVector &r, const ParticleList &plist) const {
275  const bool compute_gradient = true;
276  const bool smearing = true;
277  auto F_skyrme_or_VDF =
278  std::make_pair(ThreeVector(0., 0., 0.), ThreeVector(0., 0., 0.));
279  auto F_symmetry =
280  std::make_pair(ThreeVector(0., 0., 0.), ThreeVector(0., 0., 0.));
281 
282  const auto baryon_density_and_gradient = current_eckart(
283  r, plist, param_, DensityType::Baryon, compute_gradient, smearing);
284  double rhoB = std::get<0>(baryon_density_and_gradient);
285  const ThreeVector grad_j0B = std::get<2>(baryon_density_and_gradient);
286  const ThreeVector curl_vecjB = std::get<3>(baryon_density_and_gradient);
287  const FourVector djmuB_dt = std::get<4>(baryon_density_and_gradient);
288  if (use_skyrme_) {
289  F_skyrme_or_VDF =
290  skyrme_force(rhoB, grad_j0B, djmuB_dt.threevec(), curl_vecjB);
291  }
292 
293  if (use_symmetry_) {
294  const auto density_and_gradient =
296  compute_gradient, smearing);
297  const double rhoI3 = std::get<0>(density_and_gradient);
298  const ThreeVector grad_j0I3 = std::get<2>(density_and_gradient);
299  const ThreeVector curl_vecjI3 = std::get<3>(density_and_gradient);
300  const FourVector dvecjI3_dt = std::get<4>(density_and_gradient);
301  F_symmetry =
302  symmetry_force(rhoI3, grad_j0I3, dvecjI3_dt.threevec(), curl_vecjI3,
303  rhoB, grad_j0B, djmuB_dt.threevec(), curl_vecjB);
304  }
305 
306  if (use_vdf_) {
307  const FourVector jmuB = std::get<1>(baryon_density_and_gradient);
308  const FourVector djmuB_dx = std::get<5>(baryon_density_and_gradient);
309  const FourVector djmuB_dy = std::get<6>(baryon_density_and_gradient);
310  const FourVector djmuB_dz = std::get<7>(baryon_density_and_gradient);
311 
312  // safety check to not divide by zero
313  const int sgn = rhoB > 0 ? 1 : -1;
314  if (std::abs(rhoB) < very_small_double) {
315  rhoB = sgn * very_small_double;
316  }
317 
318  const double drhoB_dt =
319  (1 / rhoB) * (jmuB.x0() * djmuB_dt.x0() - jmuB.x1() * djmuB_dt.x1() -
320  jmuB.x2() * djmuB_dt.x2() - jmuB.x3() * djmuB_dt.x3());
321 
322  const double drhoB_dx =
323  (1 / rhoB) * (jmuB.x0() * djmuB_dx.x0() - jmuB.x1() * djmuB_dx.x1() -
324  jmuB.x2() * djmuB_dx.x2() - jmuB.x3() * djmuB_dx.x3());
325 
326  const double drhoB_dy =
327  (1 / rhoB) * (jmuB.x0() * djmuB_dy.x0() - jmuB.x1() * djmuB_dy.x1() -
328  jmuB.x2() * djmuB_dy.x2() - jmuB.x3() * djmuB_dy.x3());
329 
330  const double drhoB_dz =
331  (1 / rhoB) * (jmuB.x0() * djmuB_dz.x0() - jmuB.x1() * djmuB_dz.x1() -
332  jmuB.x2() * djmuB_dz.x2() - jmuB.x3() * djmuB_dz.x3());
333 
334  const FourVector drhoB_dxnu = {drhoB_dt, drhoB_dx, drhoB_dy, drhoB_dz};
335 
336  const ThreeVector grad_rhoB = drhoB_dxnu.threevec();
337  const ThreeVector vecjB = jmuB.threevec();
338  const ThreeVector Drho_cross_vecj = grad_rhoB.cross_product(vecjB);
339 
340  F_skyrme_or_VDF = vdf_force(
341  rhoB, drhoB_dt, drhoB_dxnu.threevec(), Drho_cross_vecj, jmuB.x0(),
342  grad_j0B, jmuB.threevec(), djmuB_dt.threevec(), curl_vecjB);
343  }
344 
345  return std::make_tuple(F_skyrme_or_VDF.first, F_skyrme_or_VDF.second,
346  F_symmetry.first, F_symmetry.second);
347 }
348 
349 } // namespace smash
Interface to the SMASH configuration files.
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...
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.
Definition: density.h:92
The FourVector class holds relevant values in Minkowski spacetime with (+, −, −, −) metric signature.
Definition: fourvector.h:33
double x3() const
Definition: fourvector.h:325
double x2() const
Definition: fourvector.h:321
ThreeVector threevec() const
Definition: fourvector.h:329
double x0() const
Definition: fourvector.h:313
double x1() const
Definition: fourvector.h:317
Particle type contains the static properties of a particle species.
Definition: particletype.h:98
bool is_baryon() const
Definition: particletype.h:204
bool is_nucleus() const
Definition: particletype.h:249
PdgCode pdgcode() const
Definition: particletype.h:157
double isospin3_rel() const
Definition: particletype.h:180
int baryon_number() const
Definition: pdgcode.h:391
bool use_skyrme_
Skyrme potential on/off.
Definition: potentials.h:483
bool use_symmetry_
Symmetry potential on/off.
Definition: potentials.h:486
double skyrme_pot(const double baryon_density) const
Evaluates Skyrme potential given a baryon density.
Definition: potentials.h:157
double symmetry_pot(const double baryon_isospin_density, const double baryon_density) const
Evaluates symmetry potential given baryon isospin density.
Definition: potentials.cc:92
double mom_dependence_C_
Parameter C of the momentum-dependent part of the potentials given in MeV.
Definition: potentials.h:525
std::pair< ThreeVector, ThreeVector > symmetry_force(const double rhoI3, const ThreeVector grad_j0I3, const ThreeVector dvecjI3_dt, const ThreeVector curl_vecjI3, const double rhoB, const ThreeVector grad_j0B, const ThreeVector dvecjB_dt, const ThreeVector curl_vecjB) const
Evaluates the electric and magnetic components of the symmetry force.
Definition: potentials.cc:184
double potential(const ThreeVector &r, const ParticleList &plist, const ParticleType &acts_on) const
Evaluates potential (Skyrme with optional Symmetry or VDF) at point r.
Definition: potentials.cc:125
FourVector vdf_pot(double rhoB, const FourVector jmuB_net) const
Evaluates the FourVector potential in the VDF model given the rest frame density and the computationa...
Definition: potentials.cc:103
std::vector< double > powers_
Parameters of the VDF potential: exponents .
Definition: potentials.h:557
double dVsym_drhoB(const double rhoB, const double rhoI3) const
Calculate the derivative of the symmetry potential with respect to the net baryon density in GeV * fm...
Definition: potentials.cc:260
double symmetry_S(const double baryon_density) const
Calculate the factor in the symmetry potential.
Definition: potentials.cc:84
double symmetry_S_Pot_
Parameter S_Pot in the symmetry potential in MeV.
Definition: potentials.h:528
double skyrme_a_
Parameter of skyrme potentials: the coefficient in front of in GeV.
Definition: potentials.h:501
std::vector< double > coeffs_
Parameters of the VDF potential: coefficients , in GeV.
Definition: potentials.h:555
std::pair< ThreeVector, ThreeVector > vdf_force(double rhoB, const double drhoB_dt, const ThreeVector grad_rhoB, const ThreeVector gradrhoB_cross_vecjB, const double j0B, const ThreeVector grad_j0B, const ThreeVector vecjB, const ThreeVector dvecjB_dt, const ThreeVector curl_vecjB) const
Evaluates the electric and magnetic components of force in the VDF model given the derivatives of the...
Definition: potentials.cc:199
double mom_dependence_Lambda_
Parameter Lambda of the momentum-dependent part of the potentials given in 1/fm.
Definition: potentials.h:519
bool use_momentum_dependence_
Momentum-dependent part on/off.
Definition: potentials.h:495
double skyrme_tau_
Parameters of skyrme potentials: the power index.
Definition: potentials.h:513
Potentials(Configuration conf, const DensityParameters &parameters)
Potentials constructor.
Definition: potentials.cc:18
double dVsym_drhoI3(const double rhoB, const double rhoI3) const
Calculate the derivative of the symmetry potential with respect to the isospin density in GeV * fm^3.
Definition: potentials.cc:250
int number_of_terms() const
Definition: potentials.h:461
static std::pair< double, int > force_scale(const ParticleType &data)
Evaluates the scaling factor of the forces acting on the particles.
Definition: potentials.cc:158
std::pair< ThreeVector, ThreeVector > skyrme_force(const double rhoB, const ThreeVector grad_j0B, const ThreeVector dvecjB_dt, const ThreeVector curl_vecjB) const
Evaluates the electric and magnetic components of the skyrme force.
Definition: potentials.cc:166
virtual std::tuple< ThreeVector, ThreeVector, ThreeVector, ThreeVector > all_forces(const ThreeVector &r, const ParticleList &plist) const
Evaluates the electric and magnetic components of the forces at point r.
Definition: potentials.cc:274
bool use_coulomb_
Coulomb potential on/Off.
Definition: potentials.h:489
double symmetry_gamma_
Power in formula for :
Definition: potentials.h:541
bool symmetry_is_rhoB_dependent_
Whether the baryon density dependence of the symmetry potential is included.
Definition: potentials.h:534
double coulomb_r_cut_
Cutoff in integration for coulomb potential.
Definition: potentials.h:544
const DensityParameters param_
Struct that contains the gaussian smearing width , the distance cutoff and the testparticle number n...
Definition: potentials.h:480
bool use_vdf_
VDF potential on/off.
Definition: potentials.h:492
double saturation_density_
Saturation density of nuclear matter used in the VDF potential; it may vary between different paramet...
Definition: potentials.h:553
virtual ~Potentials()
Standard destructor.
Definition: potentials.cc:71
double skyrme_b_
Parameters of skyrme potentials: the coefficient in front of in GeV.
Definition: potentials.h:507
The ThreeVector class represents a physical three-vector with the components .
Definition: threevector.h:31
ThreeVector cross_product(const ThreeVector &b) const
Definition: threevector.h:246
Collection of useful constants that are known at compile time.
int sgn(T val)
Signum function.
Definition: random.h:190
Definition: action.h:24
constexpr double mev_to_gev
MeV to GeV conversion factor.
Definition: constants.h:34
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...
Definition: density.cc:171
constexpr double very_small_double
A very small double, used to avoid division by zero.
Definition: constants.h:40
constexpr double nuclear_density
Ground state density of symmetric nuclear matter [fm^-3].
Definition: constants.h:48
A container to keep track of all ever existed input keys.
Definition: input_keys.h:1067
static const Key< double > potentials_momentum_dependence_C
See user guide description for more information.
Definition: input_keys.h:5527
static const Key< std::vector< double > > potentials_vdf_powers
See user guide description for more information.
Definition: input_keys.h:5490
static const Key< double > potentials_coulomb_rCut
See user guide description for more information.
Definition: input_keys.h:5514
static const Key< double > potentials_skyrme_skyrmeB
See user guide description for more information.
Definition: input_keys.h:5423
static const Key< double > potentials_momentum_dependence_Lambda
See user guide description for more information.
Definition: input_keys.h:5540
static const Key< double > potentials_skyrme_skyrmeA
See user guide description for more information.
Definition: input_keys.h:5411
static const Key< double > potentials_symmetry_gamma
See user guide description for more information.
Definition: input_keys.h:5451
static const Key< double > potentials_skyrme_skyrmeTau
See user guide description for more information.
Definition: input_keys.h:5436
static const Key< double > potentials_symmetry_sPot
See user guide description for more information.
Definition: input_keys.h:5463
static const Key< double > potentials_vdf_satRhoB
See user guide description for more information.
Definition: input_keys.h:5502
static const Key< std::vector< double > > potentials_vdf_coeffs
See user guide description for more information.
Definition: input_keys.h:5475
A container to keep track of all ever existed sections in the input file.
Definition: input_keys.h:48