File indexing completed on 2025-04-03 08:04:51
0001
0002
0003
0004
0005
0006
0007
0008
0009
0010
0011
0012
0013 #include "DDG4/Geant4SteppingAction.h"
0014
0015 #include <G4Step.hh>
0016 #include <G4RunManager.hh>
0017 #include <G4TrackStatus.hh>
0018 #include <G4Run.hh>
0019 #include <G4Event.hh>
0020 #include <G4UnitsTable.hh>
0021 #include <stack>
0022 #include <CLHEP/Units/SystemOfUnits.h>
0023
0024 #include <fmt/core.h>
0025
0026
0027 namespace dd4hep {
0028
0029
0030 namespace sim {
0031
0032
0033
0034
0035
0036
0037
0038
0039 class TextDumpingSteppingAction : public Geant4SteppingAction {
0040 std::size_t m_total_steps {0UL };
0041 std::size_t m_total_tracks {0UL };
0042 std::size_t m_total_events {0UL };
0043 std::string m_file_name {"events_stepping.txt"};
0044 std::string m_class_name;
0045 std::ofstream m_output_file;
0046 std::once_flag m_init_flag;
0047 double m_vertex_z_min = -4500;
0048 double m_vertex_z_max = 4500;
0049 bool m_vertex_cut = true;
0050
0051 bool m_save_optical = false;
0052 bool m_only_primary = true;
0053 int m_prev_event = -1;
0054 int m_prev_track_id = -1;
0055
0056 bool m_skipping_track = false;
0057 double m_min_momentum = 300 * CLHEP::MeV;
0058 double m_max_momentum = 10000 * CLHEP::TeV;
0059
0060 public:
0061
0062 TextDumpingSteppingAction(Geant4Context* context, const std::string& nam)
0063 : Geant4SteppingAction(context, nam)
0064 {
0065 declareProperty("OutputFile", m_file_name);
0066 declareProperty("MomentumMin", m_min_momentum);
0067 declareProperty("SaveOptical", m_save_optical);
0068 declareProperty("OnlyPrimary", m_only_primary);
0069 declareProperty("VertexCut", m_vertex_cut);
0070 declareProperty("VertexZMin", m_vertex_z_min);
0071 declareProperty("VertexZMax", m_vertex_z_max);
0072 m_class_name = __func__;
0073 }
0074
0075 ~TextDumpingSteppingAction() override
0076 {
0077 info("+++ Total Steps: %ld", m_total_steps);
0078 info("+++ Total Tracks written: %ld", m_total_tracks);
0079 info("+++ Total Events written: %ld", m_total_events);
0080
0081
0082
0083 if(m_output_file.is_open() && m_output_file.good()) {
0084 m_output_file.close();
0085 }
0086 }
0087
0088
0089 void ensure_output_writable() {
0090 if(!m_output_file.is_open() || !m_output_file.good()) {
0091 auto err_msg = fmt::format( "Failed to open the file or file stream is in a bad state. File name: '{}'", m_file_name);
0092 error(err_msg.c_str());
0093 throw std::runtime_error(err_msg);
0094 }
0095 }
0096
0097 void write_point(G4StepPoint *point){
0098 auto row = fmt::format("P {} {} {} {}",
0099 point->GetPosition().x(),
0100 point->GetPosition().y(),
0101 point->GetPosition().z(),
0102 point->GetGlobalTime()
0103 );
0104 m_output_file<<row<<std::endl;
0105 }
0106
0107
0108 void operator()(const G4Step* step, G4SteppingManager*) override
0109 {
0110
0111
0112 auto run_num = context()->run().run().GetRunID();
0113 auto event_num = context()->event().event().GetEventID();
0114
0115
0116
0117 std::call_once(m_init_flag, [this](){
0118
0119 fmt::print("Plugin {} info: \n", m_class_name);
0120 fmt::print(" OutputFile {}\n", m_file_name);
0121 fmt::print(" MomentumMin {}\n", m_min_momentum);
0122 fmt::print(" MomentumMax {}\n", m_max_momentum);
0123 fmt::print(" SaveOptical {}\n", m_save_optical);
0124 fmt::print(" OnlyPrimary {}\n", m_only_primary);
0125 fmt::print(" VertexCut {}\n", m_vertex_cut);
0126 fmt::print(" VertexZMin {}\n", m_vertex_z_min);
0127 fmt::print(" VertexZMax {}\n", m_vertex_z_max);
0128
0129 m_output_file = std::ofstream(m_file_name);
0130
0131 ensure_output_writable();
0132 m_output_file<<"#Format description" << std::endl;
0133 m_output_file<<"# E - event: run_num event_num" << std::endl;
0134 m_output_file<<"# T - track: id, status, pdg, pdg_name, eta, phi, qOverP, px, py, pz, vx, vy, vz" << std::endl;
0135 m_output_file<<"# P - point: x, y, z, t" << std::endl;
0136 });
0137
0138
0139 ensure_output_writable();
0140
0141 if(event_num != m_prev_event) {
0142
0143 m_prev_event = event_num;
0144 std::string evt_str = fmt::format("E {} {} ", run_num, event_num);
0145 m_prev_track_id = -1;
0146 m_output_file << evt_str << std::endl;
0147 m_total_events++;
0148 fmt::print("+\n+\n+\n+\n+\n=====================================================\n");
0149 }
0150
0151
0152 auto track = step->GetTrack();
0153 if(track->GetTrackID() != m_prev_track_id)
0154 {
0155 m_prev_track_id = track->GetTrackID();
0156
0157
0158 auto id = track->GetTrackID();
0159 auto pdg = track->GetParticleDefinition()->GetPDGEncoding();
0160 auto pdg_name = track->GetParticleDefinition()->GetParticleName();
0161 auto charge = track->GetParticleDefinition()->GetPDGCharge();
0162 auto eta = track->GetMomentum().eta();
0163 auto phi = track->GetMomentum().phi();
0164 auto qOverP = track->GetParticleDefinition()->GetPDGCharge()/track->GetMomentum().mag();
0165 auto px = track->GetMomentum().x();
0166 auto py = track->GetMomentum().y();
0167 auto pz = track->GetMomentum().z();
0168 auto p = track->GetMomentum().mag();
0169 auto vx = track->GetVertexPosition().x();
0170 auto vy = track->GetVertexPosition().y();
0171 auto vz = track->GetVertexPosition().z();
0172
0173 bool filter = m_vertex_cut && (vz < m_vertex_z_min || vz > m_vertex_z_max);
0174
0175 if(m_prev_track_id < 1000)
0176 {
0177 fmt::print("track: {:<5}, {:<10} vtx: {:12.5f} {:12.5f} {:12.5f} {:5} {:12.5f} {:12.5f} {:5}\n",
0178 id, pdg_name, vx,vy,vz, filter, m_vertex_z_min, m_vertex_z_max, m_vertex_cut);
0179 }
0180
0181
0182
0183
0184
0185 bool track_is_ok = true;
0186
0187 if(m_only_primary)
0188 {
0189
0190 track_is_ok = track->GetParentID() == 0;
0191 }
0192
0193 if(track_is_ok && (p < m_min_momentum || p > m_max_momentum))
0194 {
0195 track_is_ok = false;
0196 }
0197
0198
0199 if(m_vertex_cut && (vz < m_vertex_z_min || vz > m_vertex_z_max))
0200 {
0201 track_is_ok = false;
0202 }
0203
0204 if(m_save_optical && pdg_name == "opticalphoton")
0205 {
0206 track_is_ok = true;
0207 }
0208
0209 m_skipping_track = !track_is_ok;
0210
0211
0212 if(m_skipping_track) return;
0213
0214 std::string trk_str = fmt::format("T {} {} {} {} {} {} {} {} {} {} {} {} {}",
0215 id, pdg, pdg_name, charge, eta, phi, qOverP, px, py, pz, vx, vy, vz);
0216 m_output_file << trk_str << std::endl;
0217 m_total_tracks++;
0218
0219
0220 write_point(step->GetPreStepPoint());
0221
0222 }
0223
0224
0225 if(m_skipping_track) return;
0226
0227
0228 write_point(step->GetPostStepPoint());
0229
0230
0231 m_total_steps++;
0232 }
0233 };
0234 }
0235 }
0236
0237 #include "DDG4/Factories.h"
0238 DECLARE_GEANT4ACTION_NS(dd4hep::sim,TextDumpingSteppingAction)