Version: SMASH-3.2
rootoutput.cc
Go to the documentation of this file.
1 /*
2  *
3  * Copyright (c) 2014-2024
4  * SMASH Team
5  *
6  * GNU General Public License (GPLv3 or later)
7  *
8  */
9 
10 #include "smash/rootoutput.h"
11 
12 #include "TFile.h"
13 #include "TTree.h"
14 #include "smash/action.h"
15 #include "smash/clock.h"
17 #include "smash/particles.h"
18 
19 namespace smash {
20 static constexpr int LHyperSurfaceCrossing = LogArea::HyperSurfaceCrossing::id;
21 
22 const int RootOutput::max_buffer_size_ = 500000;
23 
233 RootOutput::RootOutput(const std::filesystem::path &path,
234  const std::string &name, const OutputParameters &out_par)
235  : OutputInterface(name),
236  filename_(path / (name + ".root")),
237  write_collisions_(name == "Collisions" || name == "Dileptons" ||
238  name == "Photons"),
239  write_particles_(name == "Particles"),
240  write_initial_conditions_(name == "SMASH_IC"),
241  particles_only_final_(out_par.part_only_final),
242  part_extended_(out_par.part_extended),
243  coll_extended_(out_par.coll_extended),
244  ic_extended_(out_par.ic_extended) {
246  filename_unfinished_ += ".unfinished";
248  std::make_unique<TFile>(filename_unfinished_.native().c_str(), "NEW");
249  init_trees();
250 }
251 
254  particles_tree_ = new TTree("particles", "particles");
255 
256  particles_tree_->Branch("ev", &ev_, "ev/I");
257  particles_tree_->Branch("ens", &ens_, "ens/I");
258  particles_tree_->Branch("tcounter", &tcounter_, "tcounter/I");
259  particles_tree_->Branch("npart", &npart_, "npart/I");
260  particles_tree_->Branch("test_p", &test_p_, "test_p/I");
261  particles_tree_->Branch("modus_l", &modus_l_, "modus_l/D");
262  particles_tree_->Branch("current_t", &current_t_, "current_t/D");
263  particles_tree_->Branch("impact_b", &impact_b_, "impact_b/D");
264  particles_tree_->Branch("empty_event", &empty_event_, "empty_event/O");
265 
266  particles_tree_->Branch("pdgcode", &pdgcode_[0], "pdgcode[npart]/I");
267  particles_tree_->Branch("charge", &charge_[0], "charge[npart]/I");
268 
269  particles_tree_->Branch("p0", &p0_[0], "p0[npart]/D");
270  particles_tree_->Branch("px", &px_[0], "px[npart]/D");
271  particles_tree_->Branch("py", &py_[0], "py[npart]/D");
272  particles_tree_->Branch("pz", &pz_[0], "pz[npart]/D");
273 
274  particles_tree_->Branch("t", &t_[0], "t[npart]/D");
275  particles_tree_->Branch("x", &x_[0], "x[npart]/D");
276  particles_tree_->Branch("y", &y_[0], "y[npart]/D");
277  particles_tree_->Branch("z", &z_[0], "z[npart]/D");
278 
279  particles_tree_->Branch("E_kinetic_tot", &E_kinetic_tot_,
280  "E_kinetic_tot/D");
281  particles_tree_->Branch("E_fields_tot", &E_fields_tot_, "E_fields_tot/D");
282  particles_tree_->Branch("E_tot", &E_tot_, "E_tot/D");
283 
284  if (part_extended_ || ic_extended_) {
285  particles_tree_->Branch("ncoll", &coll_per_part_[0], "ncoll[npart]/I");
286  particles_tree_->Branch("form_time", &formation_time_[0],
287  "form_time[npart]/D");
288  particles_tree_->Branch("xsecfac", &xsec_factor_[0], "xsecfac[npart]/D");
289  particles_tree_->Branch("proc_id_origin", &proc_id_origin_[0],
290  "proc_id_origin[npart]/I");
291  particles_tree_->Branch("proc_type_origin", &proc_type_origin_[0],
292  "proc_type_origin[npart]/I");
293  particles_tree_->Branch("time_last_coll", &time_last_coll_[0],
294  "time_last_coll[npart]/D");
295  particles_tree_->Branch("pdg_mother1", &pdg_mother1_[0],
296  "pdg_mother1[npart]/I");
297  particles_tree_->Branch("pdg_mother2", &pdg_mother2_[0],
298  "pdg_mother2[npart]/I");
299  particles_tree_->Branch("baryon_number", &baryon_number_[0],
300  "baryon_number[npart]/I");
301  particles_tree_->Branch("strangeness", &strangeness_[0],
302  "strangeness[npart]/I");
303  }
304  }
305 
306  if (write_collisions_) {
307  collisions_tree_ = new TTree("collisions", "collisions");
308 
309  collisions_tree_->Branch("nin", &nin_, "nin/I");
310  collisions_tree_->Branch("nout", &nout_, "nout/I");
311  collisions_tree_->Branch("npart", &npart_, "npart/I");
312  collisions_tree_->Branch("ev", &ev_, "ev/I");
313  collisions_tree_->Branch("ens", &ens_, "ens/I");
314  collisions_tree_->Branch("weight", &wgt_, "weight/D");
315  collisions_tree_->Branch("partial_weight", &par_wgt_, "partial_weight/D");
316 
317  collisions_tree_->Branch("pdgcode", &pdgcode_[0], "pdgcode[npart]/I");
318  collisions_tree_->Branch("charge", &charge_[0], "charge[npart]/I");
319 
320  collisions_tree_->Branch("p0", &p0_[0], "p0[npart]/D");
321  collisions_tree_->Branch("px", &px_[0], "px[npart]/D");
322  collisions_tree_->Branch("py", &py_[0], "py[npart]/D");
323  collisions_tree_->Branch("pz", &pz_[0], "pz[npart]/D");
324 
325  collisions_tree_->Branch("t", &t_[0], "t[npart]/D");
326  collisions_tree_->Branch("x", &x_[0], "x[npart]/D");
327  collisions_tree_->Branch("y", &y_[0], "y[npart]/D");
328  collisions_tree_->Branch("z", &z_[0], "z[npart]/D");
329 
330  if (coll_extended_) {
331  collisions_tree_->Branch("ncoll", &coll_per_part_[0], "ncoll[npart]/I");
332  collisions_tree_->Branch("form_time", &formation_time_[0],
333  "form_time[npart]/D");
334  collisions_tree_->Branch("xsecfac", &xsec_factor_[0], "xsecfac[npart]/D");
335  collisions_tree_->Branch("proc_id_origin", &proc_id_origin_[0],
336  "proc_id_origin[npart]/I");
337  collisions_tree_->Branch("proc_type_origin", &proc_type_origin_[0],
338  "proc_type_origin[npart]/I");
339  collisions_tree_->Branch("time_last_coll", &time_last_coll_[0],
340  "time_last_coll[npart]/D");
341  collisions_tree_->Branch("pdg_mother1", &pdg_mother1_[0],
342  "pdg_mother1[npart]/I");
343  collisions_tree_->Branch("pdg_mother2", &pdg_mother2_[0],
344  "pdg_mother2[npart]/I");
345  collisions_tree_->Branch("baryon_number", &baryon_number_[0],
346  "baryon_number[npart]/I");
347  collisions_tree_->Branch("strangeness", &strangeness_[0],
348  "strangeness[npart]/I");
349  }
350  }
351 }
352 
358  // kOverwrite option prevents from writing extra TKey objects into root file
359  root_out_file_->Write("", TObject::kOverwrite);
360  root_out_file_->Close();
361  std::filesystem::rename(filename_unfinished_, filename_);
362 }
363 
364 void RootOutput::at_eventstart(const Particles &particles,
365  const EventLabel &event_label,
366  const EventInfo &event) {
367  // save event and ensemble numbers
368  current_event_ = event_label.event_number;
369  current_ensemble_ = event_label.ensemble_number;
370 
371  modus_l_ = event.modus_length;
372  test_p_ = event.test_particles;
373  current_t_ = event.current_time;
374  E_kinetic_tot_ = event.total_kinetic_energy;
375  E_fields_tot_ = event.total_mean_field_energy;
376  E_tot_ = event.total_energy;
377 
379  output_counter_ = 0;
380  // This is to have only one output of positive impact parameter per event
381  impact_b_ = -1.0;
382  empty_event_ = false;
383  particles_to_tree(particles);
384  output_counter_++;
385  }
386 }
387 
389  const std::unique_ptr<Clock> &,
390  const DensityParameters &,
391  const EventLabel &event_label,
392  const EventInfo &event) {
393  current_ensemble_ = event_label.ensemble_number;
394  modus_l_ = event.modus_length;
395  test_p_ = event.test_particles;
396  current_t_ = event.current_time;
397  E_kinetic_tot_ = event.total_kinetic_energy;
398  E_fields_tot_ = event.total_mean_field_energy;
399  E_tot_ = event.total_energy;
400 
402  particles_to_tree(particles);
403  output_counter_++;
404  }
405 }
406 
407 void RootOutput::at_eventend(const Particles &particles,
408  const EventLabel &event_label,
409  const EventInfo &event) {
410  current_ensemble_ = event_label.ensemble_number;
411  modus_l_ = event.modus_length;
412  test_p_ = event.test_particles;
413  current_t_ = event.current_time;
414  E_kinetic_tot_ = event.total_kinetic_energy;
415  E_fields_tot_ = event.total_mean_field_energy;
416  E_tot_ = event.total_energy;
417 
418  impact_b_ = event.impact_parameter;
419  empty_event_ = event.empty_event;
420  if (write_particles_ &&
421  !(event.empty_event &&
423  particles_to_tree(particles);
424  }
425  /* Calculate the autosave frequency to be used. In case multiple ensembles are
426  * used this has to be at least 1, so set it to 1 if the number of ensembles
427  * is larger than the initial value of the autosave frequency. The evaluation
428  * of the actual frequency to be used has to be done only once, i.e. when the
429  * autosave_frequency_ has still its meaningless initial negative value. */
430  if (autosave_frequency_ < 0) {
431  autosave_frequency_ = 1000 / event.n_ensembles;
432  if (autosave_frequency_ == 0) {
434  }
435  }
436  /* Forced regular dump from operational memory to disk. Very demanding!
437  * If program crashes written data will NOT be lost. */
440  particles_tree_->AutoSave("SaveSelf");
441  }
442  if (write_collisions_) {
443  collisions_tree_->AutoSave("SaveSelf");
444  }
445  }
446 
448  // If the runtime is too short some particles might not yet have
449  // reached the hypersurface. Warning is printed.
450  if (particles.size() != 0 && !event.impose_kinematic_cut_for_SMASH_IC) {
452  "End time might be too small for initial conditions output. "
453  "Hypersurface has not yet been crossed by ",
454  particles.size(), " particle(s).");
455  }
456  }
457 }
458 
460  const double /*density*/) {
461  if (write_collisions_) {
463  action.get_total_weight(), action.get_partial_weight());
464  }
465 
469  }
470 }
471 
472 template <typename T>
473 void RootOutput::particles_to_tree(T &particles) {
474  int i = 0;
475 
479  bool exceeded_buffer_message = true;
480 
481  for (const auto &p : particles) {
482  // Buffer full - flush to tree, else fill with particles
483  if (i >= max_buffer_size_) {
484  if (exceeded_buffer_message) {
485  logg[LOutput].warn()
486  << "\nThe number of particles N = " << particles.size()
487  << " exceeds the maximum buffer size B = " << max_buffer_size_
488  << ".\nceil(N/B) = "
489  << std::ceil(particles.size() /
490  static_cast<double>(max_buffer_size_))
491  << " separate ROOT Tree entries will be created at this output."
492  << "\nMaximum buffer size (max_buffer_size_) can be changed in "
493  << "rootoutput.h\n\n";
494  exceeded_buffer_message = false;
495  }
497  i = 0;
498  particles_tree_->Fill();
499  } else {
500  pdgcode_[i] = p.pdgcode().get_decimal();
501  charge_[i] = p.type().charge();
502 
503  p0_[i] = p.momentum().x0();
504  px_[i] = p.momentum().x1();
505  py_[i] = p.momentum().x2();
506  pz_[i] = p.momentum().x3();
507 
508  t_[i] = p.position().x0();
509  x_[i] = p.position().x1();
510  y_[i] = p.position().x2();
511  z_[i] = p.position().x3();
512 
513  if (part_extended_ || ic_extended_) {
514  const auto h = p.get_history();
515  formation_time_[i] = p.formation_time();
516  xsec_factor_[i] = p.xsec_scaling_factor();
517  time_last_coll_[i] = h.time_last_collision;
518  coll_per_part_[i] = h.collisions_per_particle;
519  proc_id_origin_[i] = h.id_process;
520  proc_type_origin_[i] = static_cast<int>(h.process_type);
521  pdg_mother1_[i] = h.p1.get_decimal();
522  pdg_mother2_[i] = h.p2.get_decimal();
523  baryon_number_[i] = p.type().baryon_number();
524  strangeness_[i] = p.type().strangeness();
525  }
526 
527  i++;
528  }
529  }
530  // Flush rest to tree
531  if (i > 0) {
532  npart_ = i;
533  particles_tree_->Fill();
534  }
535 }
536 
537 void RootOutput::collisions_to_tree(const ParticleList &incoming,
538  const ParticleList &outgoing,
539  const double weight,
540  const double partial_weight) {
543  nin_ = incoming.size();
544  nout_ = outgoing.size();
545  npart_ = nin_ + nout_;
546  wgt_ = weight;
547  par_wgt_ = partial_weight;
548 
549  int i = 0;
550 
551  /* It is assumed that nin + nout < max_buffer_size_
552  * This is true for any possible reaction for current buffer size
553  * But if one wants initial/final particles written to collisions
554  * then implementation should be updated. */
555 
556  for (const ParticleList &plist : {incoming, outgoing}) {
557  for (const auto &p : plist) {
558  pdgcode_[i] = p.pdgcode().get_decimal();
559  charge_[i] = p.type().charge();
560 
561  p0_[i] = p.momentum().x0();
562  px_[i] = p.momentum().x1();
563  py_[i] = p.momentum().x2();
564  pz_[i] = p.momentum().x3();
565 
566  t_[i] = p.position().x0();
567  x_[i] = p.position().x1();
568  y_[i] = p.position().x2();
569  z_[i] = p.position().x3();
570 
571  if (coll_extended_) {
572  const auto h = p.get_history();
573  formation_time_[i] = p.formation_time();
574  xsec_factor_[i] = p.xsec_scaling_factor();
575  time_last_coll_[i] = h.time_last_collision;
576  coll_per_part_[i] = h.collisions_per_particle;
577  proc_id_origin_[i] = h.id_process;
578  proc_type_origin_[i] = static_cast<int>(h.process_type);
579  pdg_mother1_[i] = h.p1.get_decimal();
580  pdg_mother2_[i] = h.p2.get_decimal();
581  baryon_number_[i] = p.type().baryon_number();
582  }
583 
584  i++;
585  }
586  }
587 
588  collisions_tree_->Fill();
589 }
590 } // 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...
const ParticleList & outgoing_particles() const
Get the list of particles that resulted from the action.
Definition: action.h:247
A class to pre-calculate and store parameters relevant for density calculation.
Definition: density.h:92
Abstraction of generic output.
The Particles class abstracts the storage and manipulation of particles.
Definition: particles.h:33
size_t size() const
Definition: particles.h:87
void at_interaction(const Action &action, const double density) override
Writes collisions to a tree defined by treename.
Definition: rootoutput.cc:459
std::vector< double > t_
Property that is written to ROOT output.
Definition: rootoutput.h:193
std::vector< double > y_
Property that is written to ROOT output.
Definition: rootoutput.h:195
void at_intermediate_time(const Particles &particles, const std::unique_ptr< Clock > &clock, const DensityParameters &dens_param, const EventLabel &event_label, const EventInfo &event) override
Writes intermediate particles to a tree defined by treename, if it is allowed (i.e....
Definition: rootoutput.cc:388
double current_t_
Property that is written to ROOT output.
Definition: rootoutput.h:212
OutputOnlyFinal particles_only_final_
Print only final particles in the event, no intermediate output.
Definition: rootoutput.h:227
int current_event_
Number of current event.
Definition: rootoutput.h:172
TTree * collisions_tree_
TTree for collision output.
Definition: rootoutput.h:152
~RootOutput()
Destructor.
Definition: rootoutput.cc:357
double E_kinetic_tot_
Property that is written to ROOT output.
Definition: rootoutput.h:213
int test_p_
Property that is written to ROOT output.
Definition: rootoutput.h:211
void collisions_to_tree(const ParticleList &incoming, const ParticleList &outgoing, const double weight, const double partial_weight)
Writes collisions to a tree defined by treename.
Definition: rootoutput.cc:537
void at_eventstart(const Particles &particles, const EventLabel &event_label, const EventInfo &event) override
update event number and writes intermediate particles to a tree.
Definition: rootoutput.cc:364
int autosave_frequency_
ROOT file cannot be read if it was not properly closed and finalized.
Definition: rootoutput.h:241
std::vector< int > proc_type_origin_
Property that is written to ROOT output.
Definition: rootoutput.h:206
double modus_l_
Property that is written to ROOT output.
Definition: rootoutput.h:212
std::vector< double > time_last_coll_
Property that is written to ROOT output.
Definition: rootoutput.h:200
std::vector< int > coll_per_part_
Property that is written to ROOT output.
Definition: rootoutput.h:204
int tcounter_
Property that is written to ROOT output.
Definition: rootoutput.h:211
std::vector< int > baryon_number_
Property that is written to ROOT output.
Definition: rootoutput.h:209
int npart_
Property that is written to ROOT output.
Definition: rootoutput.h:211
std::vector< int > pdg_mother1_
Property that is written to ROOT output.
Definition: rootoutput.h:207
int nin_
Property that is written to ROOT output.
Definition: rootoutput.h:211
int ev_
Property that is written to ROOT output.
Definition: rootoutput.h:211
bool write_collisions_
Option to write collisions tree.
Definition: rootoutput.h:218
std::vector< int > charge_
Property that is written to ROOT output.
Definition: rootoutput.h:203
std::vector< double > px_
Property that is written to ROOT output.
Definition: rootoutput.h:190
const std::filesystem::path filename_
Filename of output.
Definition: rootoutput.h:134
std::vector< int > strangeness_
Property that is written to ROOT output.
Definition: rootoutput.h:210
bool write_initial_conditions_
Option to write particles tree for initial conditions.
Definition: rootoutput.h:224
double wgt_
Property that is written to ROOT output.
Definition: rootoutput.h:212
std::vector< double > py_
Property that is written to ROOT output.
Definition: rootoutput.h:191
std::vector< int > proc_id_origin_
Property that is written to ROOT output.
Definition: rootoutput.h:205
void particles_to_tree(T &particles)
Writes particles to a tree defined by treename.
Definition: rootoutput.cc:473
const bool part_extended_
Whether extended particle output is on.
Definition: rootoutput.h:244
double impact_b_
Property that is written to ROOT output.
Definition: rootoutput.h:212
RootOutput(const std::filesystem::path &path, const std::string &name, const OutputParameters &out_par)
Construct ROOT output.
Definition: rootoutput.cc:233
std::vector< double > z_
Property that is written to ROOT output.
Definition: rootoutput.h:196
std::vector< double > formation_time_
Property that is written to ROOT output.
Definition: rootoutput.h:197
int current_ensemble_
Number of current ensemble.
Definition: rootoutput.h:174
void at_eventend(const Particles &particles, const EventLabel &event_label, const EventInfo &event) override
update event number and impact parameter, and writes intermediate particles to a tree.
Definition: rootoutput.cc:407
std::vector< double > x_
Property that is written to ROOT output.
Definition: rootoutput.h:194
std::vector< double > p0_
Property that is written to ROOT output.
Definition: rootoutput.h:189
bool empty_event_
Property that is written to ROOT output.
Definition: rootoutput.h:214
static const int max_buffer_size_
Maximal buffer size.
Definition: rootoutput.h:182
std::unique_ptr< TFile > root_out_file_
Pointer to root output file.
Definition: rootoutput.h:138
std::vector< double > xsec_factor_
Property that is written to ROOT output.
Definition: rootoutput.h:199
double E_fields_tot_
Property that is written to ROOT output.
Definition: rootoutput.h:213
std::filesystem::path filename_unfinished_
Filename of output as long as simulation is still running.
Definition: rootoutput.h:136
const bool coll_extended_
Whether extended collisions output is on.
Definition: rootoutput.h:246
double par_wgt_
Property that is written to ROOT output.
Definition: rootoutput.h:212
void init_trees()
Basic initialization routine, creating the TTree objects for particles and collisions.
Definition: rootoutput.cc:252
int output_counter_
Number of output in a given event.
Definition: rootoutput.h:170
double E_tot_
Property that is written to ROOT output.
Definition: rootoutput.h:213
TTree * particles_tree_
TTree for particles output.
Definition: rootoutput.h:145
int ens_
Property that is written to ROOT output.
Definition: rootoutput.h:211
std::vector< double > pz_
Property that is written to ROOT output.
Definition: rootoutput.h:192
int nout_
Property that is written to ROOT output.
Definition: rootoutput.h:211
const bool ic_extended_
Whether extended ic output is on.
Definition: rootoutput.h:248
bool write_particles_
Option to write particles tree.
Definition: rootoutput.h:221
std::vector< int > pdgcode_
Property that is written to ROOT output.
Definition: rootoutput.h:202
std::vector< int > pdg_mother2_
Property that is written to ROOT output.
Definition: rootoutput.h:208
@ IfNotEmpty
Print only final-state particles, and those only if the event is not empty.
@ No
Print initial, intermediate and final-state particles.
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
static constexpr int LHyperSurfaceCrossing
Definition: binaryoutput.cc:22
@ HyperSurfaceCrossing
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.
bool impose_kinematic_cut_for_SMASH_IC
Whether or not kinematic cuts are employed for SMASH IC.
Structure to contain information about the event and ensemble numbers.
int32_t ensemble_number
The number of the ensemble.
int32_t event_number
The number of the event.
Helper structure for Experiment to hold output options and parameters.