gsd
Loading...
Searching...
No Matches
gsd.cc
Go to the documentation of this file.
1// gemc
2#include "gsd.h"
3
4// geant4
5#include "G4SDManager.hh"
6
7// Thread-local sensitive detector instance.
8// Constructor: initializes base logging, Geant4 SD name, and the hits collection name used for this detector.
9GSensitiveDetector::GSensitiveDetector(const std::string& sdName,
10 const std::shared_ptr<GOptions>& goptions) :
12 G4VSensitiveDetector(sdName) {
13 log->info(2, FUNCTION_NAME, " for " + sdName);
14
15 // collectionName is a G4VSensitiveDetector G4CollectionNameVector.
16 // GEMC uses its own handling of hits; this is set primarily for compatibility with Geant4 conventions/examples.
17 std::string hitCollectionName = sdName + "__HitCollection";
18 collectionName.insert(hitCollectionName);
19
20 log->debug(CONSTRUCTOR, FUNCTION_NAME, " for " + sdName);
21}
22
23
24// Thread-local; called at the beginning of each event.
25// Allocates and registers the per-event hits collection and resets per-event caches.
26void GSensitiveDetector::Initialize(G4HCofThisEvent* g4hc) {
27 std::string sdName = GetName();
28 log->info(1, FUNCTION_NAME, sdName);
29
30 // Hit content definition is read once per event from the digitization routine.
31 gHitBitSet = digitization_routine->readoutSpecs->getHitBitSet();
32
33 // Clearing touchableVector at the start of the event (per-event hit identity cache).
34 touchableVector.clear();
35
36 // Initializing gHitsCollection using the Geant4 G4THitsCollection constructor (expects detector and collection names).
37 gHitsCollection = new GHitsCollection(sdName, collectionName[0]);
38
39 // Add hits collection to the Geant4 event container.
40 // hcID is assigned by Geant4 and can be retrieved later by collection name.
41 auto hcID = G4SDManager::GetSDMpointer()->GetCollectionID(collectionName[0]);
42 g4hc->AddHitsCollection(hcID, gHitsCollection);
43
44 log->info(2, "Added collection id ", hcID, " to G4HCofThisEvent");
45}
46
47
48// Thread-local; called for each step in sensitive volumes.
49// Applies plugin filtering and touchable transformation, then creates new hits or updates existing hits.
50G4bool GSensitiveDetector::ProcessHits(G4Step* thisStep, [[maybe_unused]] G4TouchableHistory* g4th) {
51 // If there is a decision to skip this hit based on deposited energy, return immediately.
52
53 double depe = thisStep->GetTotalEnergyDeposit();
54 if (digitization_routine->decisionToSkipHit(depe)) { return true; }
55
56 // The hits collection should have been created in Initialize().
57 if (gHitsCollection == nullptr) { log->error(ERR_NOCOLLECTION, "No hit collection found"); }
58
59 // Get the vector of processed touchables returned by the digitization routine.
60 // If not overridden by the plugin, the base implementation typically returns a vector with the input touchable only.
61 std::vector<std::shared_ptr<GTouchable>> thisStepProcessedTouchables =
62 digitization_routine->processTouchable(getGTouchable(thisStep), thisStep);
63
64 auto hcsize = gHitsCollection->GetSize();
65
66 log->info(2, FUNCTION_NAME, " for ", GetName(),
67 " with ", std::to_string(thisStepProcessedTouchables.size()), " touchable(s), edep: ",
68 std::to_string(depe), ", Hit collection size: ", hcsize);
69
70 // If a new touchable is created, create a new GHit.
71 // Otherwise, retrieve the existing hit and append step information according to gHitBitSet.
72 for (auto thisGTouchable : thisStepProcessedTouchables) {
73 // Track id is attached to the touchable to keep hit identity consistent across updates.
74 thisGTouchable->assignTrackId(thisStep->GetTrack()->GetTrackID());
75
76 if (isThisANewTouchable(thisGTouchable)) {
77 // New hit: constructor fills initial step information based on gHitBitSet.
78 gHitsCollection->insert(new GHit(thisGTouchable, gHitBitSet, thisStep));
79 }
80 else {
81 // Existing hit: locate it and add additional information from this step according to gHitBitSet.
82 GHit* existingHit = getHitInHitCollectionUsingTouchable(thisGTouchable);
83 if (existingHit != nullptr) {
84 existingHit->addHitInfosForBitset(gHitBitSet, thisStep);
85 }
86 }
87 }
88
89 return true;
90}
91
92
93// Checks whether a touchable has already been seen in this event.
94// If not present, it is appended to the per-event cache.
95bool GSensitiveDetector::isThisANewTouchable(const std::shared_ptr<GTouchable>& thisTouchable) {
96 log->info(2, "GSensitiveDetector::isThisANewTouchable for " + GetName(),
97 " with touchable: " + thisTouchable->getIdentityString());
98
99 if (thisTouchable->exists_in_vector(touchableVector)) {
100 log->info(2, " ❌ not a new GTouchable, it is found, retrieving hit...");
101 return false; // Not a new touchable.
102 }
103 else {
104 log->info(2, " ✅ yes, new GTouchable. Adding it to touchableVector.");
105 touchableVector.push_back(*thisTouchable);
106 return true; // It's a new touchable.
107 }
108}
109
110
111// Linear search for the hit matching the provided touchable.
112// The collection is expected to contain a matching entry when called from ProcessHits().
113GHit* GSensitiveDetector::getHitInHitCollectionUsingTouchable(const std::shared_ptr<GTouchable>& gtouchable) {
114 for (unsigned int i = 0; i < gHitsCollection->GetSize(); i++) {
115 GHit* thisHit = (*gHitsCollection)[i];
116 const std::shared_ptr<GTouchable>& thisHitGTouchable = thisHit->getGTouchable();
117
118 if (*gtouchable == *thisHitGTouchable) {
119 log->info(2, "GSensitiveDetector::getHitInHitCollectionUsingTouchable for " + GetName() +
120 " found existing hit in collection for touchable: " + gtouchable->getIdentityString() +
121 " at index: " + std::to_string(i));
122 return thisHit;
123 }
124 }
125
127 "GHit not found in hit collection for touchable: " + gtouchable->getIdentityString(),
128 "in ", GetName());
129}
130
131
132// Thread-local end-of-event hook.
133// At this stage, Geant4 owns the event hit container that references the hits collection.
134void GSensitiveDetector::EndOfEvent([[maybe_unused]] G4HCofThisEvent* g4hc) {
135 log->info(1, "GSensitiveDetector::EndOfEvent for " + GetName());
136}
std::shared_ptr< GLogger > log
std::vector< std::shared_ptr< GTouchable > > processTouchable(std::shared_ptr< GTouchable > gtouchable, G4Step *thisStep)
std::shared_ptr< const GReadoutSpecs > readoutSpecs
virtual bool decisionToSkipHit(double energy)
void addHitInfosForBitset(HitBitSet hbs, const G4Step *thisStep)
std::shared_ptr< GTouchable > getGTouchable() const
void debug(debug_type type, Args &&... args) const
void info(int level, Args &&... args) const
void error(int exit_code, Args &&... args) const
HitBitSet getHitBitSet() const
GSensitiveDetector(const std::string &sdName, const std::shared_ptr< GOptions > &goptions)
Constructs a sensitive detector instance for a given detector name.
Definition gsd.cc:9
void EndOfEvent(G4HCofThisEvent *g4HitCollection) override
End-of-event hook called by Geant4.
Definition gsd.cc:134
void Initialize(G4HCofThisEvent *g4hc) override
Per-event initialization hook called by Geant4.
Definition gsd.cc:26
G4bool ProcessHits(G4Step *thisStep, G4TouchableHistory *g4th) override
Processes a Geant4 step and creates or updates hits in the current hits collection.
Definition gsd.cc:50
#define FUNCTION_NAME
CONSTRUCTOR
G4THitsCollection< GHit > GHitsCollection
Convenience alias for the Geant4 hits collection used by this module.
Definition gsd.h:73
Declares the GSensitiveDetector class and module-level constants.
constexpr int ERR_NOCOLLECTION
Error code used when the Geant4 hits collection is unexpectedly missing.
Definition gsd.h:40
constexpr int ERR_HITNOTFOUNDINCOLLECTION
Error code used when a hit is expected to exist but cannot be found in the current hit collection.
Definition gsd.h:35
constexpr const char * GSENSITIVE_LOGGER
Logger name used by this module.
Definition gsd.h:47