Version: SMASH-3.2
particletype.cc
Go to the documentation of this file.
1 /*
2  *
3  * Copyright (c) 2014-2020,2022-2024
4  * SMASH Team
5  *
6  * GNU General Public License (GPLv3 or later)
7  *
8  */
9 
10 #include "smash/particletype.h"
11 
12 #include <assert.h>
13 
14 #include <algorithm>
15 #include <map>
16 #include <vector>
17 
18 #include "smash/constants.h"
19 #include "smash/decaymodes.h"
20 #include "smash/distributions.h"
21 #include "smash/formfactors.h"
22 #include "smash/inputfunctions.h"
23 #include "smash/integrate.h"
24 #include "smash/iomanipulators.h"
25 #include "smash/isoparticletype.h"
26 #include "smash/logging.h"
28 #include "smash/stringfunctions.h"
29 
30 namespace smash {
31 static constexpr int LParticleType = LogArea::ParticleType::id;
32 static constexpr int LResonances = LogArea::Resonances::id;
33 
34 namespace {
36 const ParticleTypeList *all_particle_types = nullptr;
38 ParticleTypePtrList nucleons_list;
40 ParticleTypePtrList anti_nucs_list;
42 ParticleTypePtrList deltas_list;
44 ParticleTypePtrList anti_deltas_list;
46 ParticleTypePtrList baryon_resonances_list;
48 ParticleTypePtrList light_nuclei_list;
49 } // unnamed namespace
50 
51 const ParticleTypeList &ParticleType::list_all() {
52  assert(all_particle_types);
53  return *all_particle_types;
54 }
55 
56 // For performance reasons one might want to inline this function.
58  // Calculate the offset via pointer subtraction:
59  const auto offset = this - std::addressof(list_all()[0]);
60  // Since we're using uint16_t for storing the index better be safe than sorry:
61  // The offset must fit into the data type. If this ever fails we got a lot
62  // more particle types than initially expected and you have to increase the
63  // ParticleTypePtr storage to uint32_t.
64  assert(offset >= 0 && offset < 0xffff);
65  // After the assertion above the down-cast to uint16_t is safe:
66  return ParticleTypePtr(static_cast<uint16_t>(offset));
67 }
68 
69 ParticleTypePtrList &ParticleType::list_nucleons() { return nucleons_list; }
70 
71 ParticleTypePtrList &ParticleType::list_anti_nucleons() {
72  return anti_nucs_list;
73 }
74 
75 ParticleTypePtrList &ParticleType::list_Deltas() { return deltas_list; }
76 
77 ParticleTypePtrList &ParticleType::list_anti_Deltas() {
78  return anti_deltas_list;
79 }
80 
81 ParticleTypePtrList &ParticleType::list_baryon_resonances() {
83 }
84 
85 ParticleTypePtrList &ParticleType::list_light_nuclei() {
86  return light_nuclei_list;
87 }
88 
90  const auto found = std::lower_bound(
92  [](const ParticleType &l, const PdgCode &r) { return l.pdgcode() < r; });
93  if (found == all_particle_types->end() || found->pdgcode() != pdgcode) {
94  return {}; // The default constructor creates an invalid pointer.
95  }
96  return &*found;
97 }
98 
100  const auto found = ParticleType::try_find(pdgcode);
101  if (!found) {
102  throw PdgNotFoundFailure("PDG code " + pdgcode.string() + " not found!");
103  }
104  return *found;
105 }
106 
108  const auto found = ParticleType::try_find(pdgcode);
109  return found;
110 }
111 
112 bool ParticleType::exists(const std::string &name) {
113  const auto found =
114  std::find_if(all_particle_types->begin(), all_particle_types->end(),
115  [&](const ParticleType &p) { return p.name() == name; });
116  if (found == all_particle_types->end()) {
117  return false;
118  }
119  return true;
120 }
121 
122 ParticleType::ParticleType(std::string n, double m, double w, Parity p,
123  PdgCode id)
124  : name_(n),
125  mass_(m),
126  width_(w),
127  parity_(p),
128  pdgcode_(id),
129  min_mass_kinematic_(-1.),
130  min_mass_spectral_(-1.),
131  charge_(pdgcode_.charge()),
132  isospin_(-1),
133  I3_(pdgcode_.isospin3()) {}
134 
143 static std::string antiname(const std::string &name, PdgCode code) {
144  std::string basename, charge;
145 
146  if (name.find("⁺⁺") != std::string::npos) {
147  basename = name.substr(0, name.length() - sizeof("⁺⁺") + 1);
148  charge = "⁻⁻";
149  } else if (name.find("⁺") != std::string::npos) {
150  basename = name.substr(0, name.length() - sizeof("⁺") + 1);
151  charge = "⁻";
152  } else if (name.find("⁻⁻") != std::string::npos) {
153  basename = name.substr(0, name.length() - sizeof("⁻⁻") + 1);
154  charge = "⁺⁺";
155  } else if (name.find("⁻") != std::string::npos) {
156  basename = name.substr(0, name.length() - sizeof("⁻") + 1);
157  charge = "⁺";
158  } else if (name.find("⁰") != std::string::npos) {
159  basename = name.substr(0, name.length() - sizeof("⁰") + 1);
160  charge = "⁰";
161  } else {
162  basename = name;
163  charge = "";
164  }
165 
166  // baryons & strange mesons: insert a bar
167  if (code.baryon_number() != 0 || code.strangeness() != 0 ||
168  code.charmness() != 0 || code.is_neutrino()) {
169  constexpr char bar[] = "\u0305";
170  basename.insert(utf8::sequence_length(basename.begin()), bar);
171  }
172 
173  return basename + charge;
174 }
175 
183 static std::string chargestr(int charge) {
184  switch (charge) {
185  case 2:
186  return "⁺⁺";
187  case 1:
188  return "⁺";
189  case 0:
190  return "⁰";
191  case -1:
192  return "⁻";
193  case -2:
194  return "⁻⁻";
195  default:
196  throw std::runtime_error("Invalid charge " + std::to_string(charge));
197  }
198 }
199 
200 void ParticleType::create_type_list(const std::string &input) { // {{{
201  static ParticleTypeList type_list;
202  type_list.clear(); // in case LoadFailure was thrown and caught and we should
203  // try again
204  for (const Line &line : line_parser(input)) {
205  std::istringstream lineinput(line.text);
206  std::string name;
207  double mass, width;
208  std::string parity_string;
209  std::vector<std::string> pdgcode_strings;
210  // We expect at most 4 PDG codes per multiplet.
211  pdgcode_strings.reserve(4);
212  lineinput >> name >> mass >> width >> parity_string;
213  Parity parity;
214  bool fail = false;
215  if (parity_string == "+") {
217  } else if (parity_string == "-") {
219  } else {
220  fail = true;
221  }
222  if (lineinput.fail() || fail) {
224  "While loading the ParticleType data:\nFailed to convert the input "
225  "string to the expected data types.",
226  line));
227  }
228  // read additional PDG codes (if present)
229  while (!lineinput.eof()) {
230  pdgcode_strings.push_back("");
231  lineinput >> pdgcode_strings.back();
232  if (lineinput.fail()) {
234  "While loading the ParticleType data:\nFailed to convert the input "
235  "string to the expected data types.",
236  line));
237  }
238  }
239  if (pdgcode_strings.size() < 1) {
241  "While loading the ParticleType data:\nFailed to convert the input "
242  "string due to missing PDG code.",
243  line));
244  }
245  std::vector<PdgCode> pdgcode;
246  pdgcode.resize(pdgcode_strings.size());
247  std::transform(pdgcode_strings.begin(), pdgcode_strings.end(),
248  pdgcode.begin(),
249  [](const std::string &s) { return PdgCode(s); });
250  ensure_all_read(lineinput, line);
251 
252  // Check if provided masses are the same as hardcoded ones, if present
254  throw std::runtime_error("Nucleon mass in input file different from " +
255  std::to_string(nucleon_mass));
256  }
257  if (pdgcode[0].is_pion() && !almost_equal(mass, pion_mass)) {
258  throw std::runtime_error("Pion mass in input file different from " +
259  std::to_string(pion_mass));
260  }
261  if (pdgcode[0].is_kaon() && !almost_equal(mass, kaon_mass)) {
262  throw std::runtime_error("Kaon mass in input file different from " +
263  std::to_string(kaon_mass));
264  }
265  if (pdgcode[0].is_omega() && !almost_equal(mass, omega_mass)) {
266  throw std::runtime_error("Omega mass in input file different from " +
267  std::to_string(omega_mass));
268  }
269  if (pdgcode[0].is_Delta() && !almost_equal(mass, delta_mass)) {
270  throw std::runtime_error("Delta mass in input file different from " +
271  std::to_string(delta_mass));
272  }
274  throw std::runtime_error("Deuteron mass in input file different from " +
275  std::to_string(deuteron_mass));
276  }
277 
278  // add all states to type list
279  for (size_t i = 0; i < pdgcode.size(); i++) {
280  std::string full_name = name;
281  if (pdgcode.size() > 1) {
282  // for multiplets: add charge string to name
283  full_name += chargestr(pdgcode[i].charge());
284  }
285  type_list.emplace_back(full_name, mass, width, parity, pdgcode[i]);
286  logg[LParticleType].debug()
287  << "Setting particle type: " << type_list.back();
288  if (pdgcode[i].has_antiparticle()) {
289  /* add corresponding antiparticle */
290  PdgCode anti = pdgcode[i].get_antiparticle();
291  // For bosons the parity does not change, for fermions it gets inverted.
292  const auto anti_parity = (anti.spin() % 2 == 0) ? parity : -parity;
293  full_name = antiname(full_name, pdgcode[i]);
294  type_list.emplace_back(full_name, mass, width, anti_parity, anti);
295  logg[LParticleType].debug()
296  << "Setting antiparticle type: " << type_list.back();
297  }
298  }
299  }
300  type_list.shrink_to_fit();
301 
302  /* Sort the type list by PDG code. */
303  std::sort(type_list.begin(), type_list.end());
304 
305  /* Look for duplicates. */
306  PdgCode prev_pdg = 0;
307  for (const auto &t : type_list) {
308  if (t.pdgcode() == prev_pdg) {
309  throw ParticleType::LoadFailure("Duplicate PdgCode in particles.txt: " +
310  t.pdgcode().string());
311  }
312  prev_pdg = t.pdgcode();
313  }
314 
315  if (all_particle_types != nullptr) {
316  throw std::runtime_error("Error: Type list was already built!");
317  }
318  all_particle_types = &type_list; // note that type_list is a function-local
319  // static and thus will live on until after
320  // main().
321 
322  // create all isospin multiplets
323  for (const auto &t : type_list) {
325  }
326  // link the multiplets to the types
327  for (auto &t : type_list) {
328  t.iso_multiplet_ = IsoParticleType::find(t);
329  }
330 
331  // Create nucleons/anti-nucleons list
332  if (IsoParticleType::exists("N")) {
333  for (const auto &state : IsoParticleType::find("N").get_states()) {
334  nucleons_list.push_back(state);
335  anti_nucs_list.push_back(state->get_antiparticle());
336  }
337  }
338 
339  // Create deltas list
340  if (IsoParticleType::exists("Δ")) {
341  for (const auto &state : IsoParticleType::find("Δ").get_states()) {
342  deltas_list.push_back(state);
343  anti_deltas_list.push_back(state->get_antiparticle());
344  }
345  }
346 
347  // Create baryon resonances list
348  for (const ParticleType &type_resonance : ParticleType::list_all()) {
349  /* Only loop over baryon resonances. */
350  if (type_resonance.is_stable() ||
351  type_resonance.pdgcode().baryon_number() != 1) {
352  continue;
353  }
354  baryon_resonances_list.push_back(&type_resonance);
355  baryon_resonances_list.push_back(type_resonance.get_antiparticle());
356  }
357 
358  for (const ParticleType &type : ParticleType::list_all()) {
359  if (type.is_nucleus()) {
360  light_nuclei_list.push_back(&type);
361  }
362  }
363 } /*}}}*/
364 
366  if (unlikely(min_mass_kinematic_ < 0.)) {
367  /* If the particle is stable, min. mass is just the mass. */
369  /* Otherwise, find the lowest mass value needed in any decay mode */
370  if (!is_stable()) {
371  for (const auto &mode : decay_modes().decay_mode_list()) {
372  min_mass_kinematic_ = std::min(min_mass_kinematic_, mode->threshold());
373  }
374  }
375  }
376  return min_mass_kinematic_;
377 }
378 
380  if (unlikely(min_mass_spectral_ < 0.)) {
381  /* If the particle is stable or it has a non-zero spectral function value at
382  * the minimum mass that is allowed by kinematics, min_mass_spectral is just
383  * the min_mass_kinetic. */
385  /* Otherwise, find the lowest mass value where spectral function has a
386  * non-zero value by bisection.*/
387  if (!is_stable() &&
389  // find a right bound that has non-zero spectral function for bisection
390  const double m_step = 0.01;
391  double right_bound_bis;
392  for (unsigned int i = 0;; i++) {
393  right_bound_bis = min_mass_kinematic() + m_step * i;
394  if (this->spectral_function(right_bound_bis) > really_small) {
395  break;
396  }
397  }
398  // bisection
399  const double precision = 1E-6;
400  double left_bound_bis = right_bound_bis - m_step;
401  while (right_bound_bis - left_bound_bis > precision) {
402  const double mid = (left_bound_bis + right_bound_bis) / 2.0;
403  if (this->spectral_function(mid) > really_small) {
404  right_bound_bis = mid;
405  } else {
406  left_bound_bis = mid;
407  }
408  }
409  min_mass_spectral_ = right_bound_bis;
410  }
411  }
412  return min_mass_spectral_;
413 }
414 
416  if (isospin_ < 0) {
419  : 0;
420  }
421  return isospin_;
422 }
423 
424 double ParticleType::partial_width(const double m,
425  const DecayBranch *mode) const {
426  if (m < mode->threshold()) {
427  return 0.;
428  }
429  double partial_width_at_pole = width_at_pole() * mode->weight();
430  return mode->type().width(mass(), partial_width_at_pole, m);
431 }
432 
434  const auto offset = this - std::addressof(list_all()[0]);
435  const auto &modes = (*DecayModes::all_decay_modes)[offset];
436  assert(is_stable() || !modes.is_empty());
437  return modes;
438 }
439 
440 double ParticleType::total_width(const double m) const {
441  double w = 0.;
442  if (is_stable()) {
443  return w;
444  }
445  /* Loop over decay modes and sum up all partial widths. */
446  const auto &modes = decay_modes().decay_mode_list();
447  for (unsigned int i = 0; i < modes.size(); i++) {
448  w = w + partial_width(m, modes[i].get());
449  }
450  if (w < width_cutoff) {
451  return 0.;
452  }
453  return w;
454 }
455 
457  for (const ParticleType &ptype : ParticleType::list_all()) {
458  if (!ptype.is_stable() && ptype.decay_modes().is_empty()) {
459  throw std::runtime_error(
460  "Unstable particle " + ptype.name() +
461  " has no decay chanels! Either add one to it in decaymodes file or "
462  "set it's width to 0 in particles file.");
463  }
464  if (ptype.is_dprime() && !ParticleType::try_find(pdg::deuteron)) {
465  throw std::runtime_error(
466  "d' cannot be used without deuteron. Modify input particles file "
467  "accordingly.");
468  }
469  }
470 }
471 
473  WhichDecaymodes wh) const {
474  switch (wh) {
475  case WhichDecaymodes::All: {
476  return true;
477  }
479  return !t.is_dilepton_decay();
480  }
482  return t.is_dilepton_decay();
483  }
484  default:
485  throw std::runtime_error(
486  "Problem in selecting decaymodes in wanted_decaymode()");
487  }
488 }
489 
491  const ThreeVector x,
492  WhichDecaymodes wh) const {
493  const auto &decay_mode_list = decay_modes().decay_mode_list();
494  /* Determine whether the decay is affected by the potentials. If it's
495  * affected, read the values of the potentials at the position of the
496  * particle */
497  FourVector UB = FourVector();
498  FourVector UI3 = FourVector();
499  if (UB_lat_pointer != nullptr) {
500  UB_lat_pointer->value_at(x, UB);
501  }
502  if (UI3_lat_pointer != nullptr) {
503  UI3_lat_pointer->value_at(x, UI3);
504  }
505  /* Loop over decay modes and calculate all partial widths. */
506  DecayBranchList partial;
507  partial.reserve(decay_mode_list.size());
508  for (unsigned int i = 0; i < decay_mode_list.size(); i++) {
509  /* Calculate the sqare root s of the final state particles. */
510  const auto FinalTypes = decay_mode_list[i]->type().particle_types();
511  double scale_B = 0.0;
512  double scale_I3 = 0.0;
513  if (pot_pointer != nullptr) {
514  scale_B += pot_pointer->force_scale(*this).first;
515  scale_I3 += pot_pointer->force_scale(*this).second * isospin3_rel();
516  for (const auto &finaltype : FinalTypes) {
517  scale_B -= pot_pointer->force_scale(*finaltype).first;
518  scale_I3 -= pot_pointer->force_scale(*finaltype).second *
519  finaltype->isospin3_rel();
520  }
521  }
522  double sqrt_s = (p + UB * scale_B + UI3 * scale_I3).abs();
523 
524  const double w = partial_width(sqrt_s, decay_mode_list[i].get());
525  if (w > 0.) {
526  if (wanted_decaymode(decay_mode_list[i]->type(), wh)) {
527  partial.push_back(
528  std::make_unique<DecayBranch>(decay_mode_list[i]->type(), w));
529  }
530  }
531  }
532  return partial;
533 }
534 
535 double ParticleType::get_partial_width(const double m,
536  const ParticleTypePtrList dlist) const {
537  /* Get all decay modes. */
538  const auto &decaymodes = decay_modes().decay_mode_list();
539 
540  /* Find the right one(s) and add up corresponding widths. */
541  double w = 0.;
542  for (const auto &mode : decaymodes) {
543  double partial_width_at_pole = width_at_pole() * mode->weight();
544  if (mode->type().has_particles(dlist)) {
545  w += mode->type().width(mass(), partial_width_at_pole, m);
546  }
547  }
548  return w;
549 }
550 
552  const ParticleData &p_a,
553  const ParticleData &p_b) const {
554  /* Get all decay modes. */
555  const auto &decaymodes = decay_modes().decay_mode_list();
556 
557  /* Find the right one(s) and add up corresponding widths. */
558  double w = 0.;
559  for (const auto &mode : decaymodes) {
560  double partial_width_at_pole = width_at_pole() * mode->weight();
561  const ParticleTypePtrList l = {&p_a.type(), &p_b.type()};
562  if (mode->type().has_particles(l)) {
563  w += mode->type().in_width(mass(), partial_width_at_pole, m,
564  p_a.effective_mass(), p_b.effective_mass());
565  }
566  }
567  return w;
568 }
569 
570 double ParticleType::spectral_function(double m) const {
571  if (norm_factor_ < 0.) {
572  /* Initialize the normalization factor
573  * by integrating over the unnormalized spectral function. */
574  static /*thread_local (see #3075)*/ Integrator integrate;
575  const double width = width_at_pole();
576  const double m_pole = mass();
577  // We transform the integral using m = m_min + width_pole * tan(x), to
578  // make it definite and to avoid numerical issues.
579  const double x_min = std::atan((min_mass_kinematic() - m_pole) / width);
580  norm_factor_ = 1. / integrate(x_min, M_PI / 2., [&](double x) {
581  const double tanx = std::tan(x);
582  const double m_x = m_pole + width * tanx;
583  const double jacobian = width * (1.0 + tanx * tanx);
584  return spectral_function_no_norm(m_x) * jacobian;
585  });
586  }
588 }
589 
591  /* The spectral function is a relativistic Breit-Wigner function
592  * with mass-dependent width. Here: without normalization factor. */
593  const double resonance_width = total_width(m);
594  if (resonance_width < ParticleType::width_cutoff) {
595  return 0.;
596  }
597  return breit_wigner(m, mass(), resonance_width);
598 }
599 
601  /* The spectral function is a relativistic Breit-Wigner function.
602  * This variant is using a constant width (evaluated at the pole mass). */
603  const double resonance_width = width_at_pole();
604  if (resonance_width < ParticleType::width_cutoff) {
605  return 0.;
606  }
607  return breit_wigner(m, mass(), resonance_width);
608 }
609 
611  return breit_wigner_nonrel(m, mass(), width_at_pole());
612 }
613 
614 /* Resonance mass sampling for 2-particle final state */
615 double ParticleType::sample_resonance_mass(const double mass_stable,
616  const double cms_energy,
617  int L) const {
618  /* largest possible mass: Use 'nextafter' to make sure it is not above the
619  * physical limit by numerical error. */
620  const double max_mass = std::nextafter(cms_energy - mass_stable, 0.);
621 
622  // smallest possible mass to find non-zero spectral function contributions
623  const double min_mass = this->min_mass_spectral();
624 
625  // largest possible cm momentum (from smallest mass)
626  const double pcm_max = pCM(cms_energy, mass_stable, min_mass);
627  const double blw_max = pcm_max * blatt_weisskopf_sqr(pcm_max, L);
628  /* The maximum of the spectral-function ratio 'usually' happens at the
629  * largest mass. However, this is not always the case, therefore we need
630  * and additional fudge factor (determined automatically). Additionally,
631  * a heuristic knowledge is used that usually such mass exist that
632  * spectral_function(m) > spectral_function_simple(m). */
633  const double sf_ratio_max =
634  std::max(1., this->spectral_function(max_mass) /
635  this->spectral_function_simple(max_mass));
636 
637  double mass_res, val;
638  // outer loop: repeat if maximum is too small
639  do {
640  const double q_max = sf_ratio_max * this->max_factor1_;
641  const double max = blw_max * q_max; // maximum value for rejection sampling
642  // inner loop: rejection sampling
643  do {
644  // sample mass from a simple Breit-Wigner (aka Cauchy) distribution
645  mass_res = random::cauchy(this->mass(), this->width_at_pole() / 2.,
646  this->min_mass_spectral(), max_mass);
647  // determine cm momentum for this case
648  const double pcm = pCM(cms_energy, mass_stable, mass_res);
649  const double blw = pcm * blatt_weisskopf_sqr(pcm, L);
650  // determine ratio of full to simple spectral function
651  const double q = this->spectral_function(mass_res) /
652  this->spectral_function_simple(mass_res);
653  val = q * blw;
654  } while (val < random::uniform(0., max));
655 
656  // check that we are using the proper maximum value
657  if (val > max) {
658  logg[LResonances].debug(
659  "maximum is being increased in sample_resonance_mass: ",
660  this->max_factor1_, " ", val / max, " ", this->pdgcode(), " ",
661  mass_stable, " ", cms_energy, " ", mass_res);
662  this->max_factor1_ *= val / max;
663  } else {
664  break; // maximum ok, exit loop
665  }
666  } while (true);
667 
668  return mass_res;
669 }
670 
671 /* Resonance mass sampling for 2-particle final state with two resonances. */
672 std::pair<double, double> ParticleType::sample_resonance_masses(
673  const ParticleType &t2, const double cms_energy, int L) const {
674  const ParticleType &t1 = *this;
675  /* Sample resonance mass from the distribution
676  * used for calculating the cross section. */
677  const double max_mass_1 =
678  std::nextafter(cms_energy - t2.min_mass_spectral(), 0.);
679  const double max_mass_2 =
680  std::nextafter(cms_energy - t1.min_mass_spectral(), 0.);
681  // largest possible cm momentum (from smallest mass)
682  const double pcm_max =
683  pCM(cms_energy, t1.min_mass_spectral(), t2.min_mass_spectral());
684  const double blw_max = pcm_max * blatt_weisskopf_sqr(pcm_max, L);
685 
686  double mass_1, mass_2, val;
687  // outer loop: repeat if maximum is too small
688  do {
689  // maximum value for rejection sampling (determined automatically)
690  const double max = blw_max * t1.max_factor2_;
691  // inner loop: rejection sampling
692  do {
693  // sample mass from a simple Breit-Wigner (aka Cauchy) distribution
694  mass_1 = random::cauchy(t1.mass(), t1.width_at_pole() / 2.,
695  t1.min_mass_spectral(), max_mass_1);
696  mass_2 = random::cauchy(t2.mass(), t2.width_at_pole() / 2.,
697  t2.min_mass_spectral(), max_mass_2);
698  // determine cm momentum for this case
699  const double pcm = pCM(cms_energy, mass_1, mass_2);
700  const double blw = pcm * blatt_weisskopf_sqr(pcm, L);
701  // determine ratios of full to simple spectral function
702  const double q1 =
703  t1.spectral_function(mass_1) / t1.spectral_function_simple(mass_1);
704  const double q2 =
705  t2.spectral_function(mass_2) / t2.spectral_function_simple(mass_2);
706  val = q1 * q2 * blw;
707  } while (val < random::uniform(0., max));
708 
709  if (val > max) {
710  logg[LResonances].debug(
711  "maximum is being increased in sample_resonance_masses: ",
712  t1.max_factor2_, " ", val / max, " ", t1.pdgcode(), " ", t2.pdgcode(),
713  " ", cms_energy, " ", mass_1, " ", mass_2);
714  t1.max_factor2_ *= val / max;
715  } else {
716  break; // maximum ok, exit loop
717  }
718  } while (true);
719 
720  return {mass_1, mass_2};
721 }
722 
724  if (is_stable()) {
725  std::stringstream err;
726  err << "Particle " << *this << " is stable, so it makes no"
727  << " sense to print its spectral function, etc.";
728  throw std::runtime_error(err.str());
729  }
730 
731  double rightmost_pole = 0.0;
732  const auto &decaymodes = decay_modes().decay_mode_list();
733  for (const auto &mode : decaymodes) {
734  double pole_mass_sum = 0.0;
735  for (const ParticleTypePtr p : mode->type().particle_types()) {
736  pole_mass_sum += p->mass();
737  }
738  if (pole_mass_sum > rightmost_pole) {
739  rightmost_pole = pole_mass_sum;
740  }
741  }
742 
743  std::cout << "# mass m[GeV], width w(m) [GeV],"
744  << " spectral function(m^2)*m [GeV^-1] of " << *this << std::endl;
745  constexpr double m_step = 0.02;
746  const double m_min = min_mass_spectral();
747  // An empirical value used to stop the printout. Assumes that spectral
748  // function decays at high mass, which is true for all known resonances.
749  constexpr double spectral_function_threshold = 8.e-3;
750  std::cout << std::fixed << std::setprecision(5);
751  for (unsigned int i = 0;; i++) {
752  const double m = m_min + m_step * i;
753  const double w = total_width(m), sf = spectral_function(m);
754  if (m > rightmost_pole * 2 && sf < spectral_function_threshold) {
755  break;
756  }
757  std::cout << m << " " << w << " " << sf << std::endl;
758  }
759 }
760 
761 std::ostream &operator<<(std::ostream &out, const ParticleType &type) {
762  const PdgCode &pdg = type.pdgcode();
763  return out << type.name() << std::setfill(' ') << std::right
764  << "[ mass:" << field<6> << type.mass()
765  << ", width:" << field<6> << type.width_at_pole()
766  << ", PDG:" << field<6> << pdg
767  << ", charge:" << field<3> << pdg.charge()
768  << ", spin:" << field<2> << pdg.spin() << "/2 ]";
769 }
770 
771 /*
772  * This is valid for two particles of the same species because the comparison
773  * operator for smart pointers compares the pointed object. In this case, the
774  * `std::set<ParticleTypePtr> incoming` will contain one element instead of two.
775  */
776 ParticleTypePtrList list_possible_resonances(const ParticleTypePtr type_a,
777  const ParticleTypePtr type_b) {
778  static std::map<std::set<ParticleTypePtr>, ParticleTypePtrList>
779  map_possible_resonances_of;
780  std::set<ParticleTypePtr> incoming{type_a, type_b};
781  const ParticleTypePtrList incoming_types = {type_a, type_b};
782  // Fill map if set is not yet present
783  if (map_possible_resonances_of.count(incoming) == 0) {
784  logg[LResonances].debug()
785  << "Filling map of compatible resonances for ptypes " << type_a->name()
786  << " " << type_b->name();
787  ParticleTypePtrList resonance_list{};
788  // The tests below are redundant as the decay modes already obey them, but
789  // they are quicker to check and so improve performance.
790  for (const ParticleType &resonance : ParticleType::list_all()) {
791  /* Not a resonance, go to next type of particle */
792  if (resonance.is_stable()) {
793  continue;
794  }
795  // Same resonance as in the beginning, ignore
796  if ((resonance.pdgcode() == type_a->pdgcode()) ||
797  (resonance.pdgcode() == type_b->pdgcode())) {
798  continue;
799  }
800  // Check for charge conservation.
801  if (resonance.charge() != type_a->charge() + type_b->charge()) {
802  continue;
803  }
804  // Check for baryon-number conservation.
805  if (resonance.baryon_number() !=
806  type_a->baryon_number() + type_b->baryon_number()) {
807  continue;
808  }
809  // Check for strangeness conservation.
810  if (resonance.strangeness() !=
811  type_a->strangeness() + type_b->strangeness()) {
812  continue;
813  }
814  const auto &decaymodes = resonance.decay_modes().decay_mode_list();
815  for (const auto &mode : decaymodes) {
816  if (mode->type().has_particles(incoming_types)) {
817  resonance_list.push_back(&resonance);
818  break;
819  }
820  }
821  }
822  // Here `resonance_list` can be empty, corresponding to the case where there
823  // are no possible resonances.
824  map_possible_resonances_of[incoming] = resonance_list;
825  }
826 
827  return map_possible_resonances_of[incoming];
828 }
829 
830 } // namespace smash
DecayBranch is a derivative of ProcessBranch, which is used to represent decay channels.
const DecayType & type() const
The DecayModes class is used to store and update information about decay branches (i....
Definition: decaymodes.h:29
const DecayBranchList & decay_mode_list() const
Definition: decaymodes.h:63
DecayType is the abstract base class for all decay types.
Definition: decaytype.h:23
virtual bool is_dilepton_decay() const
Definition: decaytype.h:82
virtual double width(double m0, double G0, double m) const =0
The FourVector class holds relevant values in Minkowski spacetime with (+, −, −, −) metric signature.
Definition: fourvector.h:33
A C++ interface for numerical integration in one dimension with the GSL CQUAD integration functions.
Definition: integrate.h:106
static bool exists(const std::string &name)
Returns whether the ParticleType with the given pdgcode exists.
static const IsoParticleType & find(const std::string &name)
Returns the IsoParticleType object for the given name.
int isospin() const
Returns twice the total isospin of the multiplet.
static void create_multiplet(const ParticleType &type)
Add a new multiplet to the global list of IsoParticleTypes, which contains type.
ParticleData contains the dynamic information of a certain particle.
Definition: particledata.h:58
const ParticleType & type() const
Get the type of the particle.
Definition: particledata.h:128
double effective_mass() const
Get the particle's effective mass.
Definition: particledata.cc:24
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
double min_mass_spectral_
minimum mass, where the spectral function is non-zero Mutable, because it is initialized at first cal...
Definition: particletype.h:644
bool is_pion() const
Definition: particletype.h:219
double min_mass_spectral() const
The minimum mass of the resonance, where the spectral function is non-zero.
IsoParticleType * iso_multiplet_
Container for the isospin multiplet information.
Definition: particletype.h:656
const DecayModes & decay_modes() const
double spectral_function(double m) const
Full spectral function of the resonance (relativistic Breit-Wigner distribution with mass-dependent ...
double sample_resonance_mass(const double mass_stable, const double cms_energy, int L=0) const
Resonance mass sampling for 2-particle final state with one resonance (type given by 'this') and one ...
double get_partial_width(const double m, const ParticleTypePtrList dlist) const
Get the mass-dependent partial width of a resonance with mass m, decaying into two given daughter par...
int strangeness() const
Definition: particletype.h:213
static const ParticleTypePtr try_find(PdgCode pdgcode)
Returns the ParticleTypePtr for the given pdgcode.
Definition: particletype.cc:89
void dump_width_and_spectral_function() const
Prints out width and spectral function versus mass to the standard output.
double total_width(const double m) const
Get the mass-dependent total width of a particle with mass m.
bool wanted_decaymode(const DecayType &t, WhichDecaymodes wh) const
Helper Function that containes the if-statement logic that decides if a decay mode is either a hadron...
static const ParticleType & find(PdgCode pdgcode)
Returns the ParticleType object for the given pdgcode.
Definition: particletype.cc:99
double min_mass_kinematic() const
The minimum mass of the resonance that is kinematically allowed.
PdgCode pdgcode() const
Definition: particletype.h:157
static bool exists(PdgCode pdgcode)
static void check_consistency()
double max_factor2_
Maximum factor for double-res mass sampling, cf. sample_resonance_masses.
Definition: particletype.h:661
const std::string & name() const
Definition: particletype.h:142
double min_mass_kinematic_
minimum kinematically allowed mass of the particle Mutable, because it is initialized at first call o...
Definition: particletype.h:637
int32_t charge() const
The charge of the particle.
Definition: particletype.h:189
double spectral_function_no_norm(double m) const
Full spectral function without normalization factor.
static ParticleTypePtrList & list_nucleons()
Definition: particletype.cc:69
static ParticleTypePtrList & list_anti_nucleons()
Definition: particletype.cc:71
static const ParticleTypeList & list_all()
Definition: particletype.cc:51
bool is_stable() const
Definition: particletype.h:246
int isospin_
Isospin of the particle; filled automatically from pdgcode_.
Definition: particletype.h:651
double isospin3_rel() const
Definition: particletype.h:180
bool is_Delta() const
Definition: particletype.h:225
double spectral_function_simple(double m) const
This one is the most simple form of the spectral function, using a Cauchy distribution (non-relativis...
double width_at_pole() const
Definition: particletype.h:151
static ParticleTypePtrList & list_anti_Deltas()
Definition: particletype.cc:77
PdgCode pdgcode_
PDG Code of the particle.
Definition: particletype.h:630
bool has_antiparticle() const
Definition: particletype.h:160
ParticleType(std::string n, double m, double w, Parity p, PdgCode id)
Creates a fully initialized ParticleType object.
bool is_nucleon() const
Definition: particletype.h:216
double mass() const
Definition: particletype.h:145
static ParticleTypePtrList & list_baryon_resonances()
Definition: particletype.cc:81
static constexpr double width_cutoff
Decay width cutoff for considering a particle as stable.
Definition: particletype.h:108
DecayBranchList get_partial_widths(const FourVector p, const ThreeVector x, WhichDecaymodes wh) const
Get all the mass-dependent partial decay widths of a particle with mass m.
static ParticleTypePtrList & list_Deltas()
Definition: particletype.cc:75
bool is_deuteron() const
Definition: particletype.h:252
double get_partial_in_width(const double m, const ParticleData &p_a, const ParticleData &p_b) const
Get the mass-dependent partial in-width of a resonance with mass m, decaying into two given daughter ...
bool is_kaon() const
Definition: particletype.h:222
static void create_type_list(const std::string &particles)
Initialize the global ParticleType list (list_all) from the given input data.
int isospin() const
Returns twice the isospin vector length .
double norm_factor_
This normalization factor ensures that the spectral function is normalized to unity,...
Definition: particletype.h:647
int baryon_number() const
Definition: particletype.h:210
ParticleTypePtr operator&() const
Returns an object that acts like a pointer, except that it requires only 2 bytes and inhibits pointer...
Definition: particletype.cc:57
double max_factor1_
Maximum factor for single-res mass sampling, cf. sample_resonance_mass.
Definition: particletype.h:659
double spectral_function_const_width(double m) const
std::pair< double, double > sample_resonance_masses(const ParticleType &t2, const double cms_energy, int L=0) const
Resonance mass sampling for 2-particle final state with two resonances.
static ParticleTypePtrList & list_light_nuclei()
Definition: particletype.cc:85
double partial_width(const double m, const DecayBranch *mode) const
Get the mass-dependent partial decay width of a particle with mass m in a particular decay mode.
Parity parity() const
Definition: particletype.h:154
double mass_
pole mass of the particle
Definition: particletype.h:624
PdgCode stores a Particle Data Group Particle Numbering Scheme particle type number.
Definition: pdgcode.h:111
int baryon_number() const
Definition: pdgcode.h:391
unsigned int spin() const
Definition: pdgcode.h:670
int charmness() const
Definition: pdgcode.h:612
int strangeness() const
Definition: pdgcode.h:605
std::string string() const
Definition: pdgcode.h:325
PdgCode get_antiparticle() const
Construct the antiparticle to a given PDG code.
Definition: pdgcode.h:332
bool is_hadron() const
Definition: pdgcode.h:370
bool is_neutrino() const
Definition: pdgcode.h:381
int charge() const
The charge of the particle.
Definition: pdgcode.h:629
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
double weight() const
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.
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
#define unlikely(x)
Tell the branch predictor that this expression is likely false.
Definition: macros.h:16
ParticleTypePtrList deltas_list
Global pointer to the Particle Type list of deltas.
Definition: particletype.cc:42
ParticleTypePtrList baryon_resonances_list
Global pointer to the Particle Type list of baryon resonances.
Definition: particletype.cc:46
const ParticleTypeList * all_particle_types
Global pointer to the Particle Type list.
Definition: particletype.cc:36
ParticleTypePtrList anti_deltas_list
Global pointer to the Particle Type list of anti-deltas.
Definition: particletype.cc:44
ParticleTypePtrList light_nuclei_list
Global pointer to the Particle Type list of light nuclei.
Definition: particletype.cc:48
ParticleTypePtrList anti_nucs_list
Global pointer to the Particle Type list of anti-nucleons.
Definition: particletype.cc:40
ParticleTypePtrList nucleons_list
Global pointer to the Particle Type list of nucleons.
Definition: particletype.cc:38
constexpr int p
Proton.
constexpr int n
Neutron.
constexpr int64_t deuteron
Deuteron.
T uniform(T min, T max)
Definition: random.h:88
T cauchy(T pole, T width, T min, T max)
Draws a random number from a Cauchy distribution (sometimes also called Lorentz or non-relativistic B...
Definition: random.h:307
std::iterator_traits< octet_iterator >::difference_type sequence_length(octet_iterator lead_it)
Given an iterator to the beginning of a UTF-8 sequence, return the length of the next UTF-8 code poin...
Definition: action.h:24
constexpr double delta_mass
Delta mass in GeV.
Definition: constants.h:86
T pCM(const T sqrts, const T mass_a, const T mass_b) noexcept
Definition: kinematics.h:79
ParticleTypePtrList list_possible_resonances(const ParticleTypePtr type_a, const ParticleTypePtr type_b)
Lists the possible resonances that decay into two particles.
void ensure_all_read(std::istream &input, const Line &line)
Makes sure that nothing is left to read from this line.
static Integrator integrate
Definition: decaytype.cc:143
double breit_wigner(double m, double pole, double width)
Returns a relativistic Breit-Wigner distribution.
static std::string antiname(const std::string &name, PdgCode code)
Construct an antiparticle name-string from the given name-string for the particle and its PDG code.
build_vector_< Line > line_parser(const std::string &input)
Helper function for parsing particles.txt and decaymodes.txt.
Parity
Represent the parity of a particle type.
Definition: particletype.h:25
@ Neg
Negative parity.
@ Pos
Positive parity.
constexpr double deuteron_mass
Deuteron mass in GeV.
Definition: constants.h:92
constexpr double nucleon_mass
Nucleon mass in GeV.
Definition: constants.h:58
double blatt_weisskopf_sqr(const double p_ab, const int L)
Definition: formfactors.h:35
static constexpr int LResonances
Potentials * pot_pointer
Pointer to a Potential class.
constexpr double pion_mass
Pion mass in GeV.
Definition: constants.h:65
constexpr double really_small
Numerical error tolerance.
Definition: constants.h:37
std::string build_error_string(std::string message, const Line &line)
Builds a meaningful error message.
constexpr double kaon_mass
Kaon mass in GeV.
Definition: constants.h:72
RectangularLattice< FourVector > * UB_lat_pointer
Pointer to the skyrme potential on the lattice.
double breit_wigner_nonrel(double m, double pole, double width)
Returns a non-relativistic Breit-Wigner distribution, which is essentially a Cauchy distribution with...
static std::string chargestr(int charge)
Construct a charge string, given the charge as integer.
bool almost_equal(const N x, const N y)
Checks two numbers for relative approximate equality.
Definition: numerics.h:44
constexpr double omega_mass
omega mass in GeV.
Definition: constants.h:79
RectangularLattice< FourVector > * UI3_lat_pointer
Pointer to the symmmetry potential on the lattice.
WhichDecaymodes
Decide which decay mode widths are returned in get partical widths.
Definition: particletype.h:33
@ Hadronic
Ignore dilepton decay modes widths.
@ Dileptons
Only return dilepton decays widths.
@ All
All decay mode widths.
static constexpr int LParticleType
Line consists of a line number and the contents of that line.