Version: SMASH-3.2.2
customnucleus.cc
Go to the documentation of this file.
1 /*
2  * Copyright (c) 2019-2022,2024
3  * SMASH Team
4  *
5  * GNU General Public License (GPLv3 or later)
6  */
7 #include "smash/customnucleus.h"
8 
9 #include <cmath>
10 #include <fstream>
11 #include <map>
12 #include <string>
13 #include <vector>
14 
15 #include "smash/constants.h"
16 #include "smash/input_keys.h"
17 #include "smash/logging.h"
18 #include "smash/particletype.h"
19 #include "smash/pdgcode.h"
20 
21 namespace smash {
22 static constexpr int LCollider = LogArea::Collider::id;
23 
24 std::unique_ptr<std::ifstream> CustomNucleus::filestream_shared_ = nullptr;
25 
26 CustomNucleus::CustomNucleus(Configuration& config, int testparticles,
27  bool same_file) {
28  assert(has_projectile_or_target(config));
29  const bool is_projectile = is_about_projectile(config);
30  const auto& [file_dir_key, filename_key, particles_key] = [&is_projectile]() {
31  return is_projectile
32  ? std::make_tuple(
36  : std::make_tuple(
40  }();
41  // Read in file directory from config
42  const std::string particle_list_file_directory = config.take(file_dir_key);
43  // Read in file name from config
44  const std::string particle_list_file_name = config.take(filename_key);
45 
46  if (particles_.size() != 0) {
47  throw std::runtime_error(
48  "Your Particle List is already filled before reading in from the "
49  "external file."
50  "Something went wrong. Please check your config.");
51  }
52  /*
53  * Counts number of nucleons in one nucleus as it is specialized
54  * by the user in the config file.
55  * It is needed to read in the proper number of nucleons for one
56  * nucleus and to restart at the listreading for the following
57  * nucleus as one does not want to read configurations twice.
58  */
59  std::map<PdgCode, int> particle_list = config.take(particles_key);
60  for (const auto& particle : particle_list) {
61  if (particle.first == pdg::p) {
62  number_of_protons_ = particle.second * testparticles;
63  } else if (particle.first == pdg::n) {
64  number_of_neutrons_ = particle.second * testparticles;
65  } else {
66  throw std::runtime_error(
67  "Your nucleus can only contain protons and/or neutrons."
68  "Please check what particles you have specified in the config");
69  }
71  }
72  /*
73  * "if" statement makes sure the streams to the file are initialized
74  * properly.
75  */
76  const std::string path =
77  file_path(particle_list_file_directory, particle_list_file_name);
78  if (same_file && !filestream_shared_) {
79  filestream_shared_ = std::make_unique<std::ifstream>(path);
81  } else if (!same_file) {
82  filestream_ = std::make_unique<std::ifstream>(path);
84  } else {
86  }
87 
90  // Inherited from nucleus class (see nucleus.h)
92 }
93 
94 void CustomNucleus::fill_from_list(const std::vector<Nucleoncustom>& vec) {
95  particles_.clear();
96  index_ = 0;
97  // checking if particle is proton or neutron
98  for (const auto& it : vec) {
99  PdgCode pdgcode;
100  if (it.isospin == 1) {
101  pdgcode = pdg::p;
102  } else if (it.isospin == 0) {
103  pdgcode = pdg::n;
104  } else {
105  throw std::runtime_error(
106  "Your particles charges are not 1 = proton or 0 = neutron.\n"
107  "Check whether your list is correct or there is an error.");
108  }
109  // setting parameters for the particles in the particlelist in smash
110  const ParticleType& current_type = ParticleType::find(pdgcode);
111  double current_mass = current_type.mass();
112  particles_.emplace_back(current_type);
113  particles_.back().set_4momentum(current_mass, 0.0, 0.0, 0.0);
114  }
115 }
116 
118  /*
119  * As only arrange_nucleons is called at the beginning of every
120  * event it is important to have readfile and fill from list
121  * called again when a new event starts. The constructor is only
122  * called twice to initialize the first target and projectile.
123  * Therefore this if statement is implemented.
124  */
125  if (index_ >= custom_nucleus_.size()) {
128  }
129  const auto& pos = custom_nucleus_.at(index_);
130  index_++;
131  ThreeVector nucleon_position(pos.x, pos.y, pos.z);
132  // rotate nucleon about euler angle
133  nucleon_position.rotate(euler_phi_, euler_theta_, euler_psi_);
134 
135  return nucleon_position;
136 }
137 
139  /* Randomly generate Euler angles for rotation everytime a new
140  * custom nucleus is initialized. Therefore this is done 2 times per
141  * event.
142  */
144 
145  for (auto i = begin(); i != end(); i++) {
146  // Initialize momentum
147  i->set_4momentum(i->pole_mass(), 0.0, 0.0, 0.0);
148  /* Sampling the Woods-Saxon, get the radial
149  * position and solid angle for the nucleon. */
151  // Set the position of the nucleon.
152  i->set_4position(FourVector(0.0, pos));
153  }
154  // Recenter
155  align_center();
156 }
157 
160  logg[LCollider].warn() << "Fermi motion activated with a custom nucleus.\n";
161  logg[LCollider].warn() << "Be aware that generating the Fermi momenta\n"
162  << "assumes nucleons distributed according to a\n"
163  << "Woods-Saxon distribution.";
164 }
165 
166 std::string CustomNucleus::file_path(const std::string& file_directory,
167  const std::string& file_name) {
168  if (file_directory.back() == '/') {
169  return file_directory + file_name;
170  } else {
171  return file_directory + '/' + file_name;
172  }
173 }
174 
175 std::vector<Nucleoncustom> CustomNucleus::readfile(
176  std::ifstream& infile) const {
177  int proton_counter = 0;
178  int neutron_counter = 0;
179  std::string line;
180  std::vector<Nucleoncustom> custom_nucleus;
181  // read in only A particles for one nucleus
182  for (int i = 0; i < number_of_nucleons_; ++i) {
183  std::getline(infile, line);
184  // make sure the stream goes back to the beginning when it hits end of file
185  if (infile.eof()) {
186  infile.clear();
187  infile.seekg(0, infile.beg);
188  std::getline(infile, line);
189  }
190  Nucleoncustom nucleon;
191  std::istringstream iss(line);
192  if (!(iss >> nucleon.x >> nucleon.y >> nucleon.z >>
193  nucleon.spinprojection >> nucleon.isospin)) {
194  throw std::runtime_error(
195  "SMASH could not read in a line from your initial nuclei input file."
196  "\nCheck if your file has the following format: x y z "
197  "spinprojection isospin");
198  }
199  if (nucleon.isospin == 1) {
200  proton_counter++;
201  } else if (nucleon.isospin == 0) {
202  neutron_counter++;
203  }
204  custom_nucleus.push_back(nucleon);
205  }
206  if (proton_counter != number_of_protons_ ||
207  neutron_counter != number_of_neutrons_) {
208  throw std::runtime_error(
209  "Number of protons and/or neutrons in the nuclei input file does not "
210  "correspond to the number specified in the config.\nCheck the config "
211  "and your input file.");
212  } else {
213  return custom_nucleus;
214  }
215 }
216 
217 } // namespace smash
Interface to the SMASH configuration files.
T take(const Key< T > &key)
The default interface for SMASH to read configuration values.
size_t index_
Index needed to read out vector in distribute nucleon.
std::string file_path(const std::string &file_directory, const std::string &file_name)
Generates the name of the stream file.
static std::unique_ptr< std::ifstream > filestream_shared_
Filestream variable used if projectile and target are read in from the same file and they use the sam...
Definition: customnucleus.h:98
int number_of_protons_
Number of protons per nucleus.
ThreeVector distribute_nucleon() override
Returns position of a nucleon as given in the external file.
void arrange_nucleons() override
Sets the positions of the nucleons inside a nucleus.
CustomNucleus(Configuration &config, int testparticles, bool same_file)
Constructor that needs configuration parameters from input file and the number of testparticles.
std::vector< Nucleoncustom > custom_nucleus_
Vector contianing Data for one nucleus given in the particlelist.
int number_of_neutrons_
Number of neutrons per nucleus.
int number_of_nucleons_
Number of nucleons per nucleus Set initally to zero to be modified in the constructor.
std::unique_ptr< std::ifstream > filestream_
Filestream variable used if projectile and target are read in from different files and they therefore...
std::vector< Nucleoncustom > readfile(std::ifstream &infile) const
The returned vector contains Data for one nucleus given in the particlelist.
void generate_fermi_momenta() override
Generates Fermi momenta as it is done in the mother class but in addition prints a warning that the F...
void fill_from_list(const std::vector< Nucleoncustom > &vec)
Fills Particlelist from vector containing data for one nucleus.
std::unique_ptr< std::ifstream > * used_filestream_
Pointer to the used filestream pointer.
The FourVector class holds relevant values in Minkowski spacetime with (+, −, −, −) metric signature.
Definition: fourvector.h:33
double euler_theta_
Euler angle theta.
Definition: nucleus.h:298
void random_euler_angles()
Randomly generate Euler angles.
Definition: nucleus.cc:543
virtual void generate_fermi_momenta()
Generates momenta according to Fermi motion for the nucleons.
Definition: nucleus.cc:411
virtual void set_parameters_automatic()
Sets the deformation parameters of the Woods-Saxon distribution according to the current mass number.
Definition: nucleus.cc:302
double euler_phi_
The Euler angle phi of the three Euler angles used to apply rotations to the nucleus.
Definition: nucleus.h:296
std::vector< ParticleData > particles_
Particles associated with this nucleus.
Definition: nucleus.h:279
double euler_psi_
Euler angle psi.
Definition: nucleus.h:300
void align_center()
Shifts the nucleus so that its center is at (0,0,0)
Definition: nucleus.h:214
std::vector< ParticleData >::iterator begin()
For iterators over the particle list:
Definition: nucleus.h:306
std::vector< ParticleData >::iterator end()
For iterators over the particle list:
Definition: nucleus.h:310
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
double mass() const
Definition: particletype.h:145
PdgCode stores a Particle Data Group Particle Numbering Scheme particle type number.
Definition: pdgcode.h:111
The ThreeVector class represents a physical three-vector with the components .
Definition: threevector.h:31
void rotate(double phi, double theta, double psi)
Rotate vector by the given Euler angles phi, theta, psi.
Definition: threevector.h:283
Collection of useful constants that are known at compile time.
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.
Definition: action.h:24
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 constexpr int LCollider
static const Key< std::string > modi_collider_projectile_custom_fileDirectory
See user guide description for more information.
Definition: input_keys.h:3399
static const Key< std::map< PdgCode, int > > modi_collider_target_particles
See user guide description for more information.
Definition: input_keys.h:3259
static const Key< std::map< PdgCode, int > > modi_collider_projectile_particles
See user guide description for more information.
Definition: input_keys.h:3253
static const Key< std::string > modi_collider_target_custom_fileName
See user guide description for more information.
Definition: input_keys.h:3422
static const Key< std::string > modi_collider_projectile_custom_fileName
See user guide description for more information.
Definition: input_keys.h:3417
static const Key< std::string > modi_collider_target_custom_fileDirectory
See user guide description for more information.
Definition: input_keys.h:3405
Contains data for one nucleon that is read in from the list.
Definition: customnucleus.h:24
double z
z-coordinate
Definition: customnucleus.h:30
double x
x-coordinate
Definition: customnucleus.h:26
bool isospin
to differentiate between protons isospin=1 and neutrons isospin=0
Definition: customnucleus.h:34
bool spinprojection
spinprojection of the nucleon
Definition: customnucleus.h:32
double y
y-coordinate
Definition: customnucleus.h:28