LDMX Software
SampleValidation.cxx
1#include "DQM/SampleValidation.h"
2
3#include <algorithm>
4#include <fstream>
5#include <iostream>
6
7#include "Framework/NtupleManager.h"
8#include "SimCore/Event/SimParticle.h"
10
11namespace dqm {
12
14 target_scoring_plane_passname_ =
15 ps.get<std::string>("target_scoring_plane_passname");
16 sim_particles_passname_ = ps.get<std::string>("sim_particles_passname");
17}
18
20 // Grab the SimParticle Map and Target Scoring Plane Hits
21 auto target_sp_hits(event.getCollection<ldmx::SimTrackerHit>(
22 "TargetScoringPlaneHits", target_scoring_plane_passname_));
23 auto particle_map{event.getMap<int, ldmx::SimParticle>(
24 "SimParticles", sim_particles_passname_)};
25
26 std::vector<int> primary_daughters;
27
28 double hard_thresh{9999.0};
29
30 // Loop over all SimParticles
31 for (auto const& it : particle_map) {
32 ldmx::SimParticle p = it.second;
33 std::vector<int> parents_track_ids = p.getParents();
34 std::vector<int> daughters = p.getDaughters();
35 const auto& pdgid = p.getPdgID();
36 const auto& vertex = p.getVertex();
37 const auto& energy = p.getEnergy();
38 const auto& momentum = p.getMomentum();
39 for (auto const& parent_track_id : parents_track_ids) {
40 if (parent_track_id == 0) {
41 ROOT::Math::XYZVector momentum_vec(momentum[0], momentum[1],
42 momentum[2]);
43 histograms_.fill("primaries_pdgid", pdgidLabel(pdgid));
44 histograms_.fill("primaries_energy", energy);
45 histograms_.fill("primaries_theta",
46 momentum_vec.Theta() * (180 / 3.14159));
47 histograms_.fill("primaries_pt", std::sqrt(momentum_vec.Perp2()));
48 hard_thresh = (2500. / 4000.) * energy;
49 primary_daughters = daughters;
50 for (const ldmx::SimTrackerHit& sphit : target_sp_hits) {
51 if (sphit.getTrackID() == it.first && sphit.getPosition()[2] < 0) {
52 histograms_.fill("beam_smear", vertex[0], vertex[1]);
53 }
54 }
55 }
56 } // end loop over parents
57 } // end loop over SimParticles (1st time)
58
59 std::vector<std::vector<int>> hardbrem_daughters;
60
61 for (auto const& it : particle_map) {
62 int trackid = it.first;
63 ldmx::SimParticle p = it.second;
64 const auto& momentum = p.getMomentum();
65 ROOT::Math::XYZVector momentum_vec(momentum[0], momentum[1], momentum[2]);
66 for (auto const& primary_daughter : primary_daughters) {
67 if (trackid == primary_daughter) {
68 histograms_.fill("primarydaughters_pdgid", pdgidLabel(p.getPdgID()));
69 if (p.getPdgID() == 22) {
70 histograms_.fill("daughterphoton_energy", p.getEnergy());
71 }
72 if (p.getEnergy() >= hard_thresh) {
73 histograms_.fill("harddaughters_pdgid", pdgidLabel(p.getPdgID()));
74 histograms_.fill("harddaughters_startZ", p.getVertex()[2]);
75 histograms_.fill("harddaughters_endZ", p.getEndPoint()[2]);
76 histograms_.fill("harddaughters_energy", p.getEnergy());
77 histograms_.fill("harddaughters_theta",
78 momentum_vec.Theta() * (180 / 3.14159));
79 histograms_.fill("harddaughters_pt", std::sqrt(momentum_vec.Perp2()));
80 hardbrem_daughters.push_back(p.getDaughters());
81 }
82 }
83 } // end loop over primary daughters
84 } // end loop over SimParticles (2nd time)
85
86 for (auto const& it : particle_map) {
87 int trackid = it.first;
88 ldmx::SimParticle p = it.second;
89 const auto& momentum = p.getMomentum();
90 ROOT::Math::XYZVector momentum_vec(momentum[0], momentum[1], momentum[2]);
91 for (const std::vector<int>& daughter_track_id : hardbrem_daughters) {
92 for (const int& daughter_id : daughter_track_id) {
93 if (trackid == daughter_id) {
94 histograms_.fill("hardbremdaughters_pdgid", pdgidLabel(p.getPdgID()));
95 histograms_.fill("hardbremdaughters_startZ", p.getVertex()[2]);
96 histograms_.fill("hardbremdaughters_endZ", p.getEndPoint()[2]);
97 histograms_.fill("hardbremdaughters_energy", p.getEnergy());
98 histograms_.fill("hardbremdaughters_theta",
99 momentum_vec.Theta() * (180 / 3.14159));
100 histograms_.fill("hardbremdaughters_pt",
101 std::sqrt(momentum_vec.Perp2()));
102 }
103 }
104 } // end loop over hardbrem daughters
105 } // end loop over SimParticles (3x time)
106
107 return;
108}
109
110int SampleValidation::pdgidLabel(const int pdgid) {
111 // initially assign label as "anything else"/overflow value,
112 // only change if the pdg id is something of interest
113 int label = 18;
114 if (pdgid == -11) label = 0; // e+
115 if (pdgid == 11) label = 1; // e-
116 if (pdgid == -13) label = 2; // μ+
117 if (pdgid == 13) label = 3; // μ-
118 if (pdgid == 22) label = 4; // γ
119 if (pdgid == 2212) label = 5; // proton
120 if (pdgid == 2112) label = 6; // neutron
121 if (pdgid == 211) label = 7; // π+
122 if (pdgid == -211) label = 8; // π-
123 if (pdgid == 111) label = 9; // π0
124 if (pdgid == 321) label = 10; // K+
125 if (pdgid == -321) label = 11; // K-
126 if (pdgid == 130) label = 12; // K-Long
127 if (pdgid == 310) label = 13; // K-Short
128 if (pdgid == 3122 || pdgid == 3222 || pdgid == 3212 || pdgid == 3112 ||
129 pdgid == 3322 || pdgid == 3312)
130 label = 16; // strange baryon
131 /*
132 * Nuclear PDG codes are given by ±10LZZZAAAI so to find the atomic
133 * number, we divide by 10 (to lose I) and then take the modulo
134 * with 1000.
135 */
136 if (pdgid > 1000000000) { // nuclei
137 if (((pdgid / 10) % 1000) <= 4) {
138 label = 14; // light nuclei
139 } else {
140 label = 15; // heavy nuclei
141 }
142 }
143 // dark photon, need pdg id for other models like ALPs and SIMPs
144 if (pdgid == 622) label = 17;
145
146 return label;
147}
148
149} // namespace dqm
#define DECLARE_ANALYZER(CLASS)
Macro which allows the framework to construct an analyzer given its name during configuration.
Class which encapsulates information from a hit in a simulated tracking detector.
virtual void configure(framework::config::Parameters &ps) override
Callback for the EventProcessor to configure itself from the given set of parameters.
virtual void analyze(const framework::Event &event) override
Process the event and make histograms or summaries.
HistogramPool histograms_
helper object for making and filling histograms
Implements an event buffer system for storing event data.
Definition Event.h:42
const std::vector< ContentType > & getCollection(const std::string &collectionName, const std::string &passName) const
Get a collection (std::vector) of objects from the event bus.
Definition Event.h:400
void fill(const std::string &name, const T &val)
Fill a 1D histogram.
Class encapsulating parameters for configuring a processor.
Definition Parameters.h:29
const T & get(const std::string &name) const
Retrieve the parameter of the given name.
Definition Parameters.h:78
Class representing a simulated particle.
Definition SimParticle.h:23
double getEnergy() const
Get the energy of this particle [MeV].
Definition SimParticle.h:72
std::vector< int > getParents() const
Get a vector containing the track IDs of the parent particles.
std::vector< double > getVertex() const
Get a vector containing the vertex of this particle in mm.
std::vector< int > getDaughters() const
Get a vector containing the track IDs of all daughter particles.
int getPdgID() const
Get the PDG ID of this particle.
Definition SimParticle.h:85
std::vector< double > getEndPoint() const
Get the end_point of this particle where it was destroyed or left the world volume [mm].
std::vector< double > getMomentum() const
Get a vector containing the momentum of this particle [MeV].
Represents a simulated tracker hit in the simulation.