Version: SMASH-3.2
deformednucleus.cc
Go to the documentation of this file.
1 /*
2  * Copyright (c) 2014-2022,2024
3  * SMASH Team
4  *
5  * GNU General Public License (GPLv3 or later)
6  */
8 
9 #include <cmath>
10 #include <map>
11 #include <stdexcept>
12 
13 #include "smash/configuration.h"
14 #include "smash/constants.h"
15 #include "smash/fourvector.h"
16 #include "smash/input_keys.h"
17 #include "smash/random.h"
18 #include "smash/threevector.h"
19 
20 namespace smash {
21 
22 DeformedNucleus::DeformedNucleus(const std::map<PdgCode, int> &particle_list,
23  int nTest)
24  : Nucleus(particle_list, nTest) {}
25 
27  bool auto_deformation)
28  : Nucleus(config, nTest) {
29  if (auto_deformation) {
32  } else {
34  }
35  /* If the config does not contain (anymore) a target or a projectile
36  sub-section, this code should not be executed, because e.g. the
37  is_about_projectile function would fail. */
38  if (has_projectile_or_target(config)) {
39  const auto &orientation_section = [&config]() {
42  }();
43  if (config.has_section(orientation_section)) {
44  Configuration sub_conf =
45  config.extract_complete_sub_configuration(orientation_section);
47  }
48  }
49 }
50 
52  double a_radius;
53  Angles a_direction;
54  // Set a sensible maximum bound for radial sampling.
55  double radius_max =
58 
59  // Sample the distribution.
60  do {
61  a_direction.distribute_isotropically();
62  // sample r**2 dr
63  a_radius = radius_max * std::cbrt(random::canonical());
64  } while (random::canonical() > nucleon_density(a_radius,
65  a_direction.costheta(),
66  a_direction.phi()) /
68 
69  // Update (x, y, z) positions.
70  return a_direction.threevec() * a_radius;
71 }
72 
74  // Set the deformation parameters
75  // reference for U, Pb, Au, Cu: \iref{Moller:1993ed}
76  // reference for Zr and Ru: \iref{Schenke:2019ruo}
77  // reference for Xe: \iref{Moller:2015fba}
78  bool listed = 0;
79  const std::map<int, std::string> A_map = {{238, "Uranium"},
80  {208, "Lead"},
81  {197, "Gold"},
82  {63, "Copper"},
83  {129, "Xenon"}};
84  const std::map<std::string, std::string> Z_map = {{"Uranium", "92"},
85  {"Lead", "82"},
86  {"Gold", "79"},
87  {"Copper", "29"},
88  {"Xenon", "54"}};
91  switch (A) {
92  case 238: // Uranium
93  if (Z == 92) {
94  set_beta_2(0.28);
95  set_beta_4(0.093);
96  } else {
97  listed = true;
98  }
99  break;
100  case 208: // Lead
101  if (Z == 82) {
102  set_beta_2(0.0);
103  set_beta_4(0.0);
104  } else {
105  listed = true;
106  }
107  break;
108  case 197: // Gold
109  if (Z == 79) {
110  set_beta_2(-0.131);
111  set_beta_4(-0.031);
112  } else {
113  listed = true;
114  }
115  break;
116  case 129: // Xenon
117  if (Z == 54) {
118  set_beta_2(0.162);
119  set_beta_4(-0.003);
120  } else {
121  listed = true;
122  }
123  break;
124  case 63: // Copper
125  if (Z == 29) {
126  set_beta_2(0.162);
127  set_beta_4(-0.006);
128  } else {
129  listed = true;
130  }
131  break;
132  case 96:
133  if (Z == 40) { // Zirconium
134  set_beta_2(0.0);
135  set_beta_4(0.0);
136  } else if (Z == 44) { // Ruthenium
137  set_beta_2(0.158);
138  set_beta_4(0.0);
139  } else {
140  throw std::domain_error(
141  "Number of protons for nuclei with mass number A = 96 does not "
142  "match that of Zirconium or Ruthenium. The deformation parameters "
143  "for additional isobars are currently not implemented."
144  " Please specify at least \"Beta_2\" and \"Beta_4\" "
145  "manually and set \"Automatic: False.\" ");
146  }
147  break;
148  default:
149  throw std::domain_error(
150  "Mass number not listed for automatically setting deformation "
151  "parameters. Please specify at least \"Beta_2\" and \"Beta_4\" "
152  "manually and set \"Automatic: False.\" ");
153  }
154  if (listed) {
155  throw std::domain_error("Mass number is listed under " + A_map.at(A) +
156  " but the proton "
157  "number of " +
158  std::to_string(Z) +
159  " does not match "
160  "its " +
161  Z_map.at(A_map.at(A)) +
162  " protons."
163  "Please specify at least \"Beta_2\" and \"Beta_4\" "
164  "manually and set \"Automatic: False.\" ");
165  }
166 }
167 
169  Configuration &config) {
170  if (has_projectile_or_target(config)) {
171  const bool is_projectile = is_about_projectile(config);
172  const auto &[beta2_key, beta3_key, beta4_key,
173  gamma_key] = [&is_projectile]() {
174  return is_projectile
175  ? std::make_tuple(
180  : std::make_tuple(
185  }();
186  // Deformation parameters
187  if (config.has_value(beta2_key)) {
188  beta2_ = config.take(beta2_key);
189  }
190  if (config.has_value(gamma_key)) {
191  gamma_ = config.take(gamma_key);
192  }
193  if (config.has_value(beta3_key)) {
194  beta3_ = config.take(beta3_key);
195  }
196  if (config.has_value(beta4_key)) {
197  beta4_ = config.take(beta4_key);
198  }
199  }
200 }
201 
202 double y_l_m(int l, int m, double cosx, double phi) {
203  if (l == 2 && m == 0) {
204  return (1. / 4) * std::sqrt(5 / M_PI) * (3. * (cosx * cosx) - 1);
205  } else if (l == 2 && m == 2) {
206  double sinx2 = 1. - cosx * cosx;
207  return (1. / 4) * std::sqrt(15 / (2. * M_PI)) * sinx2 * std::cos(2. * phi);
208  } else if (l == 3 && m == 0) {
209  return (1. / 4) * std::sqrt(7 / M_PI) *
210  (5. * cosx * (cosx * cosx) - 3. * cosx);
211  } else if (l == 4 && m == 0) {
212  return (3. / 16) * std::sqrt(1 / M_PI) *
213  (35. * (cosx * cosx) * (cosx * cosx) - 30. * (cosx * cosx) + 3);
214  } else {
215  throw std::domain_error(
216  "Not a valid angular momentum quantum number in y_l_m.");
217  }
218 }
219 
220 double DeformedNucleus::nucleon_density(double r, double cosx,
221  double phi) const {
223  (1 + std::exp((r - Nucleus::get_nuclear_radius() *
224  (1 +
225  beta2_ * (std::cos(gamma_) *
226  y_l_m(2, 0, cosx, phi) +
227  std::sqrt(2) * std::sin(gamma_) *
228  y_l_m(2, 2, cosx, phi)) +
229  beta3_ * y_l_m(3, 0, cosx, phi) +
230  beta4_ * y_l_m(4, 0, cosx, phi))) /
232 }
233 
235  double phi) const {
236  return 1.0 /
237  (1 + std::exp((r - Nucleus::get_nuclear_radius() *
238  (1 +
239  beta2_ * (std::cos(gamma_) *
240  y_l_m(2, 0, cosx, phi) +
241  std::sqrt(2) * std::sin(gamma_) *
242  y_l_m(2, 2, cosx, phi)) +
243  beta3_ * y_l_m(3, 0, cosx, phi) +
244  beta4_ * y_l_m(4, 0, cosx, phi))) /
246 }
247 
249  double cosx) const {
251  // Perform the phi integration. This is needed if the triaxiality coefficient
252  // gamma is included, which includes a dependency around the phi axis.
253  // Unfortunately the Integrator class does not support 3d integration which is
254  // why this intermediate integral is needed. It has been checked that the
255  // integral factorizes.
256  const auto result = integrate(0.0, 2.0 * M_PI, [&](double phi) {
257  return nucleon_density_unnormalized(r, cosx, phi);
258  });
259  return result.value();
260 }
261 
264  // Transform integral from (0, oo) to (0, 1) via r = (1 - t) / t.
265  // To prevent overflow, the integration is only performed to t = 0.01 which
266  // corresponds to r = 99fm. Additionally the precision settings in the
267  // Integrator2d scheme are equally important. However both these point affect
268  // the result only after the seventh digit which should not be relevant here.
269  if (gamma_ == 0.0) {
270  const auto result = integrate(0.01, 1, -1, 1, [&](double t, double cosx) {
271  const double r = (1 - t) / t;
272  return twopi * std::pow(r, 2.0) *
273  nucleon_density_unnormalized(r, cosx, 0.0) / std::pow(t, 2.0);
274  });
275  const auto rho0 = number_of_particles() / result.value();
276  return rho0;
277  } else {
278  const auto result = integrate(0.01, 1, -1, 1, [&](double t, double cosx) {
279  const double r = (1 - t) / t;
280  return std::pow(r, 2.0) * integrant_nucleon_density_phi(r, cosx) /
281  std::pow(t, 2.0);
282  });
283  const auto rho0 = number_of_particles() / result.value();
284  return rho0;
285  }
286 }
287 
288 } // 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
double phi() const
Definition: angles.h:279
double costheta() const
Definition: angles.h:278
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...
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.
ThreeVector distribute_nucleon() override
Deformed Woods-Saxon sampling routine.
double nucleon_density_unnormalized(double r, double cosx, double phi) const override
Return the unnormalized deformed Woods-Saxon distribution for the given position.
double integrant_nucleon_density_phi(double r, double cosx) const
Return the integral over the azimuthal angle phi.
double beta4_
Deformation parameter for angular momentum l=4.
double gamma_
Triaxiality parameter for angular momentum l=2.
void set_deformation_parameters_from_config(Configuration &config)
Set parameters for spherical deformation of the nucleus from the values specified in the configuratio...
void set_beta_2(double b2)
Set deformation coefficient for Y_2_0.
double beta3_
Deformation parameter for angular momentum l=3.
double nucleon_density(double r, double cosx, double phi) const override
Return the deformed Woods-Saxon probability density for the given position.
DeformedNucleus(const std::map< PdgCode, int > &particle_list, int nTest)
Constructor for DeformedNucles which takes a particle list and the number of testparticles.
void set_beta_4(double b4)
Set deformation coefficient for Y_4_0.
double beta2_
Deformation parameter for angular momentum l=2.
double calculate_saturation_density() const override
void set_deformation_parameters_automatic()
Sets the deformation parameters of the radius according to the current mass number.
A C++ interface for numerical integration in two dimensions with the Cuba Cuhre integration function.
Definition: integrate.h:219
A C++ interface for numerical integration in one dimension with the GSL CQUAD integration functions.
Definition: integrate.h:106
A nucleus is a collection of particles that are initialized, before the beginning of the simulation a...
Definition: nucleus.h:27
double get_diffusiveness() const
Definition: nucleus.h:328
double get_saturation_density() const
Definition: nucleus.h:333
double get_nuclear_radius() const
Definition: nucleus.h:361
virtual void set_saturation_density(double density)
Sets the saturation density of the nucleus.
Definition: nucleus.h:252
size_t number_of_protons() const
Number of physical protons in the nucleus:
Definition: nucleus.h:175
size_t number_of_particles() const
Number of physical particles in the nucleus:
Definition: nucleus.h:156
void set_orientation_from_config(Configuration &orientation_config)
Set angles for rotation of the nucleus from config file.
Definition: nucleus.cc:360
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.
T canonical()
Definition: random.h:113
Definition: action.h:24
static Integrator integrate
Definition: decaytype.cc:143
double y_l_m(int l, int m, double cosx, double phi)
Spherical harmonics Y_2_0, Y_2_2, Y_3_0 and Y_4_0.
constexpr double twopi
.
Definition: constants.h:45
bool has_projectile_or_target(const Configuration &config)
Find out whether a configuration has a projectile or a target sub-section.
Definition: nucleus.cc:584
bool is_about_projectile(const Configuration &config)
Find out whether a configuration is about projectile or target.
Definition: nucleus.cc:590
static const Key< double > modi_collider_projectile_deformed_beta3
See user guide description for more information.
Definition: input_keys.h:3487
static const Key< double > modi_collider_projectile_deformed_beta4
See user guide description for more information.
Definition: input_keys.h:3507
static const Key< double > modi_collider_target_deformed_beta2
See user guide description for more information.
Definition: input_keys.h:3472
static const Key< double > modi_collider_target_deformed_beta4
See user guide description for more information.
Definition: input_keys.h:3512
static const Key< double > modi_collider_projectile_deformed_beta2
See user guide description for more information.
Definition: input_keys.h:3467
static const Key< double > modi_collider_projectile_deformed_gamma
See user guide description for more information.
Definition: input_keys.h:3527
static const Key< double > modi_collider_target_deformed_gamma
See user guide description for more information.
Definition: input_keys.h:3532
static const Key< double > modi_collider_target_deformed_beta3
See user guide description for more information.
Definition: input_keys.h:3492
static const Section m_c_p_orientation
Subsection for the projectile orientation in collider modus.
Definition: input_keys.h:114
static const Section m_c_t_orientation
Subsection for the target orientation in collider modus.
Definition: input_keys.h:128