Version: SMASH-3.2
hepmcinterface.cc
Go to the documentation of this file.
1 /*
2  *
3  * Copyright (c) 2021 Christian Holm Christensen
4  * Copyright (c) 2021-2022,2024
5  * SMASH Team
6  *
7  * GNU General Public License (GPLv3 or later)
8  *
9  */
10 
11 #include "smash/hepmcinterface.h"
12 
13 #include "HepMC3/GenRunInfo.h"
14 #include "HepMC3/Print.h"
15 #include "HepMC3/Setup.h"
16 
17 #include "smash/config.h"
18 #include "smash/experiment.h"
19 #include "smash/logging.h"
20 
21 namespace smash {
22 
23 HepMcInterface::HepMcInterface(const std::string& name, const bool full_event)
24  : OutputInterface(name),
25  event_(HepMC3::Units::GEV, HepMC3::Units::MM),
26  ion_(),
27  xs_(),
28  ip_(),
29  full_event_(full_event) {
30  logg[LOutput].debug() << "Name of output: " << name << " "
31  << (full_event_ ? "full event" : "final state only")
32  << " output" << std::endl;
33  ion_ = std::make_shared<HepMC3::GenHeavyIon>();
34  ion_->set(-1, -1, -1, -1, -1, -1, -1, -1, -1, -1.0, -1.0, -1.0, -1.0, -1.0,
35  -1.0);
36 
37  xs_ = std::make_shared<HepMC3::GenCrossSection>();
38  // GenRunInfo: HepMC3 class to store run-related information
39  std::shared_ptr<HepMC3::GenRunInfo> run_info =
40  std::make_shared<HepMC3::GenRunInfo>();
41  std::vector<std::string> weightnames;
42  weightnames.push_back("Default");
43  run_info->set_weight_names(weightnames);
44  HepMC3::GenRunInfo::ToolInfo tool;
45  tool.name = "SMASH";
46  tool.version = SMASH_VERSION;
47 #ifdef GIT_BRANCH
48  tool.version = tool.version + GIT_BRANCH;
49 #endif
50  tool.description = "";
51  run_info->tools().push_back(tool);
52  event_.set_run_info(run_info);
53  HepMC3::Setup::set_debug_level(logg[LOutput].isEnabled<einhard::DEBUG>() ? 5
54  : 0);
55 }
56 
58  const EventLabel& event_label,
59  const EventInfo& event) {
60  if (event.n_ensembles != 1) {
61  throw std::logic_error(
62  "HepMcInterface shouldn't be used with multiple parallel ensembles.");
63  }
64 
65  // Clear event and mapping and set event number
66  clear();
67 
68  // Set header stuff on event
69  ion_->impact_parameter = event.impact_parameter;
70  xs_->set_cross_section(1, 1); // Dummy values
71  event_.set_event_number(event_label.event_number);
72  event_.set_heavy_ion(ion_);
73 
74  // Create IP only if final state
75  ip_ = std::make_shared<HepMC3::GenVertex>();
76  event_.add_vertex(ip_);
77 
78  // Count up projectile and target
79  smash::FourVector p_proj;
80  smash::FourVector p_targ;
81  AZ az_proj{0, 0};
82  AZ az_targ{0, 0};
83  bool is_coll = (event.impact_parameter >= 0.0);
84 
85  for (auto& data : particles) {
86  if (is_coll) {
87  if (!data.is_neutron() && !data.is_proton()) {
88  throw std::runtime_error(
89  "Particle of PID=" + std::to_string(data.pdgcode().get_decimal()) +
90  " is not a valid HepMC beam particle!");
91  }
92 
93  if (data.belongs_to() == BelongsTo::Projectile) {
94  p_proj += data.momentum();
95  az_proj.first++;
96  az_proj.second += data.type().charge();
97  } else if (data.belongs_to() == BelongsTo::Target) {
98  p_targ += data.momentum();
99  az_targ.first++;
100  az_targ.second += data.type().charge();
101  }
102  }
103 
104  if (!full_event_ && is_coll) {
105  continue;
106  }
107 
108  // If we are not only adding final state particles, but all
109  // interactions, or not in collider modus, then we add all the
110  // incoming particles as outgoing (incoming for non-collider
111  // modus) of the vertex. In that way, we will keep track of
112  // participants and spectators as well as make the event
113  // structure consistent.
114  auto op = make_register(data, Status::fnal);
115  if (is_coll) {
116  ip_->add_particle_out(op);
117  } else {
118  ip_->add_particle_in(op);
119  }
120  }
121 
122  coll_.resize(az_proj.first + az_targ.first);
123  // Make beam particles
124  if (is_coll) {
125  auto proj = make_gen(ion_pdg(az_proj), Status::beam, p_proj);
126  auto targ = make_gen(ion_pdg(az_targ), Status::beam, p_targ);
127 
128  // Add to interaction point if we need to
129  ip_->add_particle_in(proj);
130  ip_->add_particle_in(targ);
131  }
132 }
133 
135  const double /* density */) {
136  HepMC3::GenVertexPtr vp;
137  auto type = action.get_type();
138  int status = get_status(type);
139 
140  if (full_event_) {
141  FourVector v = action.get_interaction_point();
142  vp = std::make_shared<HepMC3::GenVertex>(
143  HepMC3::FourVector(v.x1(), v.x2(), v.x3(), v.x0()));
144  event_.add_vertex(vp);
145  vp->add_attribute("weight", std::make_shared<HepMC3::FloatAttribute>(
146  action.get_total_weight()));
147  vp->add_attribute(
148  "partial_weight",
149  std::make_shared<HepMC3::FloatAttribute>(action.get_partial_weight()));
150  }
151 
152  // Now mark participants
153  if (full_event_) {
154  for (auto& i : action.incoming_particles()) {
155  // Create tree
156  HepMC3::GenParticlePtr ip = find_or_make(i, status);
157  ip->set_status(status);
158  vp->add_particle_in(ip);
159  }
160  } else {
161  return;
162  }
163 
164  // Add outgoing particles
165  for (auto& o : action.outgoing_particles()) {
166  vp->add_particle_out(make_register(o, Status::fnal));
167  }
168 }
169 
170 void HepMcInterface::at_eventend(const Particles& particles, const EventLabel&,
171  const EventInfo& event) {
172  if (event.n_ensembles != 1) {
173  throw std::logic_error(
174  "HepMcInterface shouldn't be used with multiple parallel ensembles.");
175  }
176  // We evaluate if it is a heavy ion collision event
177  bool is_coll = (event.impact_parameter >= 0.0);
178  // In case this was an empty event
179  if (event.empty_event && is_coll) {
180  clear();
181  return;
182  }
183  // Set the weights
184  event_.weights() = std::vector<double>(1, 1);
185  /* since the number of collisions is not an experimental obervable, we set it
186  * to -1.
187  */
188  ion_->Ncoll_hard = -1;
189  ion_->Ncoll = -1;
190  /* This should be the number of participants in the projectile nucleus.
191  * However, to avoid confusion with the Glauber model, we prefer to set it -1.
192  */
193  ion_->Npart_proj = -1;
194  /* This should be the number of participants in the target nucleus.
195  * However, to avoid confusion with the Glauber model, we prefer to set it -1.
196  */
197  ion_->Npart_targ = -1;
198  // If we only do final state events, then take particle
199  // Note, we should already have the particles if not only final
200  // state
201  // Take all passed particles and add as outgoing particles to event
202  for (auto& p : particles) {
203  if (!full_event_) {
204  auto h = make_register(p, Status::fnal);
205  ip_->add_particle_out(h);
206  } else if (map_.find(p.id()) == map_.end()) {
207  throw std::runtime_error("Dangling particle " + std::to_string(p.id()));
208  }
209  }
210 }
211 
213  event_.clear();
214  map_.clear();
215  ip_ = 0;
216  coll_ = 0;
217  ncoll_ = 0;
218  ncoll_hard_ = 0;
219 }
220 
222  if (t == ProcessType::Decay)
223  return Status::dcy;
224 
225  // Make all other codes SMASH specific
226  return static_cast<int>(t) + static_cast<int>(Status::off);
227 }
228 
229 HepMC3::GenParticlePtr HepMcInterface::make_gen(int pid, int status,
230  const smash::FourVector& mom,
231  double mass) {
232  auto p = std::make_shared<HepMC3::GenParticle>(
233  HepMC3::FourVector(mom.x1(), mom.x2(), mom.x3(), mom.x0()), pid, status);
234  if (mass > 0)
235  p->set_generated_mass(mass);
236  return p;
237 }
238 
239 HepMC3::GenParticlePtr HepMcInterface::make_register(const ParticleData& p,
240  int status) {
241  int id = p.id();
242  auto h = make_gen(p.pdgcode().get_decimal(), status, p.momentum(),
243  p.type().mass());
244  map_[id] = h;
245 
246  return h;
247 }
248 
249 HepMC3::GenParticlePtr HepMcInterface::find_or_make(const ParticleData& p,
250  int status,
251  bool force_new) {
252  int id = p.id();
253  if (!force_new) {
254  auto it = map_.find(id);
255  if (it != map_.end()) {
256  return it->second;
257  }
258  }
259 
260  return make_register(p, status);
261 }
262 
263 int HepMcInterface::ion_pdg(const AZ& az) const {
264  if (az.second == 1 && az.first == 1)
265  return 2212; // Proton
266  if (az.second == 1 && az.first == 0)
267  return 2112; // Neutron
268 
269  return 1000 * 1000 * 1000 + az.second * 10 * 1000 + az.first * 10;
270 }
271 } // namespace smash
Action is the base class for a generic process that takes a number of incoming particles and transfor...
Definition: action.h:35
virtual ProcessType get_type() const
Get the process type.
Definition: action.h:131
virtual double get_total_weight() const =0
Return the total weight value, which is mainly used for the weight output entry.
const ParticleList & incoming_particles() const
Get the list of particles that go into the action.
Definition: action.cc:58
virtual double get_partial_weight() const =0
Return the specific weight for the chosen outgoing channel, which is mainly used for the partial weig...
FourVector get_interaction_point() const
Get the interaction point.
Definition: action.cc:68
const ParticleList & outgoing_particles() const
Get the list of particles that resulted from the action.
Definition: action.h:247
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
double x0() const
Definition: fourvector.h:313
double x1() const
Definition: fourvector.h:317
std::pair< int, int > AZ
Pair of Atomic weight and number.
HepMC3::GenVertexPtr ip_
The interaction point.
int ncoll_
counter of binary collisions (e.g., where both incoming particles are from the beams.
bool full_event_
Whether the full event or only final-state particles are in the output.
void at_eventend(const Particles &particles, const EventLabel &event_label, const EventInfo &event) override
Add the final particles information of an event to the central vertex.
HepMC3::GenParticlePtr make_gen(int pid, int status, const smash::FourVector &mom, double mass=-1)
Make an HepMC particle.
int ion_pdg(const AZ &az) const
Encode ion PDG.
int get_status(const ProcessType &t) const
Convert SMASH process type to HepMC status.
HepMC3::GenParticlePtr make_register(const ParticleData &p, int status=Status::fnal)
Find particle in mapping or generate it.
int ncoll_hard_
counter of hard binary collisions (e.g., where both incoming particles are from the beams.
void clear()
Clear before an event.
IdMap map_
Mapping from ID to particle.
void at_interaction(const Action &action, const double density) override
Writes collisions to event.
HepMC3::GenParticlePtr find_or_make(const ParticleData &p, int status=Status::fnal, bool force_new=false)
Find particle in mapping or generate it.
HepMC3::GenEvent event_
The event.
HepMC3::GenHeavyIonPtr ion_
The heavy-ion structure.
void at_eventstart(const Particles &particles, const EventLabel &event_label, const EventInfo &event) override
Add the initial particles information of an event to the central vertex.
CollCounter coll_
Collision counter.
HepMcInterface(const std::string &name, const bool full_event)
Create HepMC particle event in memory.
HepMC3::GenCrossSectionPtr xs_
Dummy cross-section.
Abstraction of generic output.
ParticleData contains the dynamic information of a certain particle.
Definition: particledata.h:58
The Particles class abstracts the storage and manipulation of particles.
Definition: particles.h:33
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.
Definition: action.h:24
ProcessType
ProcessTypes are used to identify the type of the process.
Definition: processbranch.h:39
@ Decay
See here for a short description.
static constexpr int LOutput
Structure to contain custom data for output.
bool empty_event
True if no collisions happened.
int n_ensembles
Number of ensembles.
Structure to contain information about the event and ensemble numbers.
int32_t event_number
The number of the event.