File indexing completed on 2025-10-15 08:04:40
0001
0002
0003
0004
0005
0006
0007
0008
0009 #include "ActsExamples/Io/Root/RootParticleWriter.hpp"
0010
0011 #include "Acts/Definitions/TrackParametrization.hpp"
0012 #include "Acts/Definitions/Units.hpp"
0013 #include "Acts/EventData/TrackParameters.hpp"
0014 #include "Acts/Propagator/Propagator.hpp"
0015 #include "Acts/Propagator/SympyStepper.hpp"
0016 #include "Acts/Surfaces/PerigeeSurface.hpp"
0017 #include "Acts/Surfaces/Surface.hpp"
0018 #include "Acts/Utilities/Helpers.hpp"
0019 #include "Acts/Utilities/VectorHelpers.hpp"
0020 #include "ActsExamples/EventData/SimParticle.hpp"
0021 #include "ActsExamples/Framework/AlgorithmContext.hpp"
0022
0023 #include <cstdint>
0024 #include <ios>
0025 #include <ostream>
0026 #include <stdexcept>
0027
0028 #include <TFile.h>
0029 #include <TTree.h>
0030
0031 ActsExamples::RootParticleWriter::RootParticleWriter(
0032 const ActsExamples::RootParticleWriter::Config& cfg,
0033 Acts::Logging::Level lvl)
0034 : WriterT(cfg.inputParticles, "RootParticleWriter", lvl), m_cfg(cfg) {
0035
0036 if (m_cfg.filePath.empty()) {
0037 throw std::invalid_argument("Missing file path");
0038 }
0039 if (m_cfg.treeName.empty()) {
0040 throw std::invalid_argument("Missing tree name");
0041 }
0042
0043
0044 m_outputFile = TFile::Open(m_cfg.filePath.c_str(), m_cfg.fileMode.c_str());
0045 if (m_outputFile == nullptr) {
0046 throw std::ios_base::failure("Could not open '" + m_cfg.filePath + "'");
0047 }
0048 m_outputFile->cd();
0049 m_outputTree = new TTree(m_cfg.treeName.c_str(), m_cfg.treeName.c_str());
0050 if (m_outputTree == nullptr) {
0051 throw std::bad_alloc();
0052 }
0053
0054
0055 m_outputTree->Branch("event_id", &m_eventId);
0056 m_outputTree->Branch("particle_hash", &m_particleHash);
0057 m_outputTree->Branch("particle_type", &m_particleType);
0058 m_outputTree->Branch("process", &m_process);
0059 m_outputTree->Branch("vx", &m_vx);
0060 m_outputTree->Branch("vy", &m_vy);
0061 m_outputTree->Branch("vz", &m_vz);
0062 m_outputTree->Branch("vt", &m_vt);
0063 m_outputTree->Branch("px", &m_px);
0064 m_outputTree->Branch("py", &m_py);
0065 m_outputTree->Branch("pz", &m_pz);
0066 m_outputTree->Branch("m", &m_m);
0067 m_outputTree->Branch("q", &m_q);
0068 m_outputTree->Branch("eta", &m_eta);
0069 m_outputTree->Branch("phi", &m_phi);
0070 m_outputTree->Branch("pt", &m_pt);
0071 m_outputTree->Branch("p", &m_p);
0072 m_outputTree->Branch("q_over_p", &m_qop);
0073 m_outputTree->Branch("theta", &m_theta);
0074 m_outputTree->Branch("vertex_primary", &m_vertexPrimary);
0075 m_outputTree->Branch("vertex_secondary", &m_vertexSecondary);
0076 m_outputTree->Branch("particle", &m_particle);
0077 m_outputTree->Branch("generation", &m_generation);
0078 m_outputTree->Branch("sub_particle", &m_subParticle);
0079
0080 if (m_cfg.writeHelixParameters) {
0081 m_outputTree->Branch("perigee_d0", &m_perigeeD0);
0082 m_outputTree->Branch("perigee_z0", &m_perigeeZ0);
0083 m_outputTree->Branch("perigee_phi", &m_perigeePhi);
0084 m_outputTree->Branch("perigee_theta", &m_perigeeTheta);
0085 m_outputTree->Branch("perigee_q_over_p", &m_perigeeQop);
0086 m_outputTree->Branch("perigee_p", &m_perigeeP);
0087 m_outputTree->Branch("perigee_px", &m_perigeePx);
0088 m_outputTree->Branch("perigee_py", &m_perigeePy);
0089 m_outputTree->Branch("perigee_pz", &m_perigeePz);
0090 m_outputTree->Branch("perigee_eta", &m_perigeeEta);
0091 m_outputTree->Branch("perigee_pt", &m_perigeePt);
0092 }
0093
0094 m_outputTree->Branch("e_loss", &m_eLoss);
0095 m_outputTree->Branch("total_x0", &m_pathInX0);
0096 m_outputTree->Branch("total_l0", &m_pathInL0);
0097 m_outputTree->Branch("number_of_hits", &m_numberOfHits);
0098 m_outputTree->Branch("outcome", &m_outcome);
0099 }
0100
0101 ActsExamples::RootParticleWriter::~RootParticleWriter() {
0102 if (m_outputFile != nullptr) {
0103 m_outputFile->Close();
0104 }
0105 }
0106
0107 ActsExamples::ProcessCode ActsExamples::RootParticleWriter::finalize() {
0108 m_outputFile->cd();
0109 m_outputTree->Write();
0110 m_outputFile->Close();
0111
0112 ACTS_INFO("Wrote particles to tree '" << m_cfg.treeName << "' in '"
0113 << m_cfg.filePath << "'");
0114
0115 return ProcessCode::SUCCESS;
0116 }
0117
0118 ActsExamples::ProcessCode ActsExamples::RootParticleWriter::writeT(
0119 const AlgorithmContext& ctx, const SimParticleContainer& particles) {
0120
0121 std::lock_guard<std::mutex> lock(m_writeMutex);
0122
0123 m_eventId = ctx.eventNumber;
0124 for (const auto& particle : particles) {
0125 m_particleHash.push_back(particle.particleId().hash());
0126 m_particleType.push_back(particle.pdg());
0127 m_process.push_back(static_cast<std::uint32_t>(particle.process()));
0128
0129 m_vx.push_back(Acts::clampValue<float>(particle.fourPosition().x() /
0130 Acts::UnitConstants::mm));
0131 m_vy.push_back(Acts::clampValue<float>(particle.fourPosition().y() /
0132 Acts::UnitConstants::mm));
0133 m_vz.push_back(Acts::clampValue<float>(particle.fourPosition().z() /
0134 Acts::UnitConstants::mm));
0135 m_vt.push_back(Acts::clampValue<float>(particle.fourPosition().w() /
0136 Acts::UnitConstants::mm));
0137
0138
0139 if (!std::isfinite(particle.mass()) || !std::isfinite(particle.charge())) {
0140 ACTS_WARNING("Particle mass or charge is not finite, can't write it");
0141 }
0142
0143 m_m.push_back(
0144 Acts::clampValue<float>(particle.mass() / Acts::UnitConstants::GeV));
0145 m_q.push_back(
0146 Acts::clampValue<float>(particle.charge() / Acts::UnitConstants::e));
0147
0148 m_vertexPrimary.push_back(particle.particleId().vertexPrimary());
0149 m_vertexSecondary.push_back(particle.particleId().vertexSecondary());
0150 m_particle.push_back(particle.particleId().particle());
0151 m_generation.push_back(particle.particleId().generation());
0152 m_subParticle.push_back(particle.particleId().subParticle());
0153
0154 m_eLoss.push_back(Acts::clampValue<float>(particle.energyLoss() /
0155 Acts::UnitConstants::GeV));
0156 m_pathInX0.push_back(
0157 Acts::clampValue<float>(particle.pathInX0() / Acts::UnitConstants::mm));
0158 m_pathInL0.push_back(
0159 Acts::clampValue<float>(particle.pathInL0() / Acts::UnitConstants::mm));
0160 m_numberOfHits.push_back(particle.numberOfHits());
0161 m_outcome.push_back(static_cast<std::uint32_t>(particle.outcome()));
0162
0163
0164 const auto p = particle.absoluteMomentum() / Acts::UnitConstants::GeV;
0165 m_p.push_back(Acts::clampValue<float>(p));
0166 m_px.push_back(Acts::clampValue<float>(p * particle.direction().x()));
0167 m_py.push_back(Acts::clampValue<float>(p * particle.direction().y()));
0168 m_pz.push_back(Acts::clampValue<float>(p * particle.direction().z()));
0169
0170 m_eta.push_back(Acts::clampValue<float>(
0171 Acts::VectorHelpers::eta(particle.direction())));
0172 m_pt.push_back(Acts::clampValue<float>(
0173 p * Acts::VectorHelpers::perp(particle.direction())));
0174 m_phi.push_back(Acts::clampValue<float>(
0175 Acts::VectorHelpers::phi(particle.direction())));
0176 m_theta.push_back(Acts::clampValue<float>(
0177 Acts::VectorHelpers::theta(particle.direction())));
0178 m_qop.push_back(Acts::clampValue<float>(
0179 particle.qOverP() * Acts::UnitConstants::GeV / Acts::UnitConstants::e));
0180
0181 if (!m_cfg.writeHelixParameters) {
0182
0183 continue;
0184 }
0185
0186
0187 auto pSurface =
0188 Acts::Surface::makeShared<Acts::PerigeeSurface>(m_cfg.referencePoint);
0189
0190
0191 const Acts::Vector3 startDir = particle.direction();
0192 const auto qOverP = particle.qOverP();
0193 auto intersection =
0194 pSurface
0195 ->intersect(ctx.geoContext, particle.position(), startDir,
0196 Acts::BoundaryTolerance::Infinite())
0197 .closest();
0198
0199
0200 if (particle.charge() == 0) {
0201 ACTS_WARNING(
0202 "Particle has zero charge, linearly extrapolating to perigee");
0203
0204 auto perigeeD0 = NaNfloat;
0205 auto perigeeZ0 = NaNfloat;
0206
0207 const auto position = intersection.position();
0208
0209
0210 auto lpResult =
0211 pSurface->globalToLocal(ctx.geoContext, position, startDir);
0212 if (lpResult.ok()) {
0213 perigeeD0 = lpResult.value()[Acts::BoundIndices::eBoundLoc0];
0214 perigeeZ0 = lpResult.value()[Acts::BoundIndices::eBoundLoc1];
0215 } else {
0216 ACTS_ERROR("Global to local transformation did not succeed.");
0217 }
0218
0219 m_perigeePhi.push_back(Acts::clampValue<float>(particle.phi()));
0220 m_perigeeTheta.push_back(Acts::clampValue<float>(particle.theta()));
0221 m_perigeeQop.push_back(Acts::clampValue<float>(
0222 qOverP * Acts::UnitConstants::GeV / Acts::UnitConstants::e));
0223 m_perigeeP.push_back(Acts::clampValue<float>(particle.absoluteMomentum() /
0224 Acts::UnitConstants::GeV));
0225 m_perigeePx.push_back(Acts::clampValue<float>(m_p.back() * startDir.x()));
0226 m_perigeePy.push_back(Acts::clampValue<float>(m_p.back() * startDir.y()));
0227 m_perigeePz.push_back(Acts::clampValue<float>(m_p.back() * startDir.z()));
0228 m_perigeeEta.push_back(Acts::clampValue<float>(
0229 Acts::VectorHelpers::eta(particle.direction())));
0230 m_perigeePt.push_back(Acts::clampValue<float>(
0231 m_p.back() * Acts::VectorHelpers::perp(particle.direction())));
0232
0233
0234 m_perigeeD0.push_back(
0235 Acts::clampValue<float>(perigeeD0 / Acts::UnitConstants::mm));
0236 m_perigeeZ0.push_back(
0237 Acts::clampValue<float>(perigeeZ0 / Acts::UnitConstants::mm));
0238 continue;
0239 }
0240
0241
0242
0243
0244 using Stepper = Acts::SympyStepper;
0245 Stepper stepper(m_cfg.bField);
0246 using PropagatorT = Acts::Propagator<Stepper>;
0247 auto propagator = std::make_shared<PropagatorT>(stepper);
0248
0249 Acts::BoundTrackParameters startParams =
0250 Acts::BoundTrackParameters::createCurvilinear(
0251 particle.fourPosition(), startDir, qOverP, std::nullopt,
0252 Acts::ParticleHypothesis::pion());
0253
0254
0255 using PropOptions = PropagatorT::Options<>;
0256 PropOptions pOptions(ctx.geoContext, ctx.magFieldContext);
0257
0258
0259 pOptions.direction =
0260 Acts::Direction::fromScalarZeroAsPositive(intersection.pathLength());
0261
0262
0263 auto propRes = propagator->propagate(startParams, *pSurface, pOptions);
0264 if (!propRes.ok() || !propRes->endParameters.has_value()) {
0265 ACTS_ERROR("Propagation to perigee surface failed.");
0266 m_perigeePhi.push_back(NaNfloat);
0267 m_perigeeTheta.push_back(NaNfloat);
0268 m_perigeeQop.push_back(NaNfloat);
0269 m_perigeeD0.push_back(NaNfloat);
0270 m_perigeeZ0.push_back(NaNfloat);
0271 m_perigeeP.push_back(NaNfloat);
0272 m_perigeePx.push_back(NaNfloat);
0273 m_perigeePy.push_back(NaNfloat);
0274 m_perigeePz.push_back(NaNfloat);
0275 m_perigeeEta.push_back(NaNfloat);
0276 m_perigeePt.push_back(NaNfloat);
0277 continue;
0278 }
0279 const Acts::BoundTrackParameters& atPerigee = *propRes->endParameters;
0280
0281
0282
0283 const auto& perigee_pars = atPerigee.parameters();
0284
0285 const auto perigeeD0 = perigee_pars[Acts::BoundIndices::eBoundLoc0];
0286 const auto perigeeZ0 = perigee_pars[Acts::BoundIndices::eBoundLoc1];
0287
0288
0289 const auto perigeePhi = perigee_pars[Acts::BoundIndices::eBoundPhi];
0290 const auto perigeeTheta = perigee_pars[Acts::BoundIndices::eBoundTheta];
0291 const auto perigeeQop = perigee_pars[Acts::BoundIndices::eBoundQOverP];
0292
0293 m_perigeePhi.push_back(Acts::clampValue<float>(perigeePhi));
0294 m_perigeeTheta.push_back(Acts::clampValue<float>(perigeeTheta));
0295 m_perigeeQop.push_back(Acts::clampValue<float>(
0296 perigeeQop * Acts::UnitConstants::GeV / Acts::UnitConstants::e));
0297
0298 const auto perigeeP =
0299 atPerigee.absoluteMomentum() / Acts::UnitConstants::GeV;
0300 m_perigeeP.push_back(Acts::clampValue<float>(perigeeP));
0301 const auto dir = atPerigee.direction();
0302 m_perigeePx.push_back(Acts::clampValue<float>(perigeeP * dir.x()));
0303 m_perigeePy.push_back(Acts::clampValue<float>(perigeeP * dir.y()));
0304 m_perigeePz.push_back(Acts::clampValue<float>(perigeeP * dir.z()));
0305 m_perigeeEta.push_back(Acts::clampValue<float>(
0306 Acts::VectorHelpers::eta(atPerigee.direction())));
0307 m_perigeePt.push_back(Acts::clampValue<float>(
0308 perigeeP * Acts::VectorHelpers::perp(atPerigee.direction())));
0309
0310
0311 m_perigeeD0.push_back(
0312 Acts::clampValue<float>(perigeeD0 / Acts::UnitConstants::mm));
0313 m_perigeeZ0.push_back(
0314 Acts::clampValue<float>(perigeeZ0 / Acts::UnitConstants::mm));
0315 }
0316
0317 m_outputTree->Fill();
0318
0319 m_particleHash.clear();
0320 m_particleType.clear();
0321 m_process.clear();
0322 m_vx.clear();
0323 m_vy.clear();
0324 m_vz.clear();
0325 m_vt.clear();
0326 m_p.clear();
0327 m_px.clear();
0328 m_py.clear();
0329 m_pz.clear();
0330 m_m.clear();
0331 m_q.clear();
0332 m_eta.clear();
0333 m_phi.clear();
0334 m_pt.clear();
0335 m_theta.clear();
0336 m_qop.clear();
0337 m_vertexPrimary.clear();
0338 m_vertexSecondary.clear();
0339 m_particle.clear();
0340 m_generation.clear();
0341 m_subParticle.clear();
0342 m_eLoss.clear();
0343 m_numberOfHits.clear();
0344 m_outcome.clear();
0345 m_pathInX0.clear();
0346 m_pathInL0.clear();
0347
0348 if (m_cfg.writeHelixParameters) {
0349 m_perigeeD0.clear();
0350 m_perigeeZ0.clear();
0351 m_perigeePhi.clear();
0352 m_perigeeTheta.clear();
0353 m_perigeeQop.clear();
0354 m_perigeeP.clear();
0355 m_perigeePx.clear();
0356 m_perigeePy.clear();
0357 m_perigeePz.clear();
0358 m_perigeeEta.clear();
0359 m_perigeePt.clear();
0360 }
0361
0362 return ProcessCode::SUCCESS;
0363 }