Geant4 11.3.0
Toolkit for the simulation of the passage of particles through matter
Loading...
Searching...
No Matches
G4EventManager.cc
Go to the documentation of this file.
1//
2// ********************************************************************
3// * License and Disclaimer *
4// * *
5// * The Geant4 software is copyright of the Copyright Holders of *
6// * the Geant4 Collaboration. It is provided under the terms and *
7// * conditions of the Geant4 Software License, included in the file *
8// * LICENSE and available at http://cern.ch/geant4/license . These *
9// * include a list of copyright holders. *
10// * *
11// * Neither the authors of this software system, nor their employing *
12// * institutes,nor the agencies providing financial support for this *
13// * work make any representation or warranty, express or implied, *
14// * regarding this software system or assume any liability for its *
15// * use. Please see the license in the file LICENSE and URL above *
16// * for the full disclaimer and the limitation of liability. *
17// * *
18// * This code implementation is the result of the scientific and *
19// * technical work of the GEANT4 collaboration. *
20// * By using, copying, modifying or distributing the software (or *
21// * any work based on the software) you agree to acknowledge its *
22// * use in resulting scientific publications, and indicate your *
23// * acceptance of all terms of the Geant4 Software license. *
24// ********************************************************************
25//
26// G4EventManager class implementation
27//
28// Author: M.Asai, SLAC
29// Adding sub-event parallelism: M.Asai, JLAB
30// --------------------------------------------------------------------
31
32#include "G4EventManager.hh"
33#include "G4ios.hh"
34#include "G4EvManMessenger.hh"
35#include "G4Event.hh"
37#include "G4VTrackingManager.hh"
38#include "G4UserEventAction.hh"
40#include "G4SDManager.hh"
41#include "G4StateManager.hh"
42#include "G4ApplicationState.hh"
44#include "G4Navigator.hh"
45#include "Randomize.hh"
47#include "G4AutoLock.hh"
48
49namespace {
50 G4Mutex EventMgrMutex = G4MUTEX_INITIALIZER;
51}
52
53#include <unordered_set>
54
55G4ThreadLocal G4EventManager* G4EventManager::fpEventManager = nullptr;
56
58{
59 return fpEventManager;
60}
61
63{
64 if(fpEventManager != nullptr)
65 {
66 G4Exception("G4EventManager::G4EventManager", "Event0001", FatalException,
67 "G4EventManager::G4EventManager() has already been made.");
68 }
69 else
70 {
71 trackManager = new G4TrackingManager;
72 transformer = new G4PrimaryTransformer;
73 trackContainer = new G4StackManager;
74 theMessenger = new G4EvManMessenger(this);
76 stateManager = G4StateManager::GetStateManager();
77 fpEventManager = this;
78 }
79}
80
82{
83 delete trackContainer;
84 delete transformer;
85 delete trackManager;
86 delete theMessenger;
87 delete userEventAction;
88 fpEventManager = nullptr;
89}
90
91void G4EventManager::DoProcessing(G4Event* anEvent,
92 G4TrackVector* trackVector, G4bool IDhasAlreadySet)
93{
94 abortRequested = false;
95 G4ApplicationState currentState = stateManager->GetCurrentState();
96 if(currentState != G4State_GeomClosed)
97 {
98 G4Exception("G4EventManager::ProcessOneEvent", "Event0002", JustWarning,
99 "IllegalState -- Geometry not closed: cannot process an event.");
100 return;
101 }
102 currentEvent = anEvent;
103 if(!subEventParaWorker) stateManager->SetNewState(G4State_EventProc);
104 if(storetRandomNumberStatusToG4Event > 1)
105 {
106 std::ostringstream oss;
108 randomNumberStatusToG4Event = oss.str();
109 currentEvent->SetRandomNumberStatusForProcessing(randomNumberStatusToG4Event);
110 }
111
112 // Resetting Navigator has been moved to G4EventManager,
113 // so that resetting is now done for every event.
114 G4ThreeVector center(0,0,0);
117 navigator->LocateGlobalPointAndSetup(center,nullptr,false);
118
119 G4Track* track = nullptr;
120 G4TrackStatus istop = fAlive;
121
122#ifdef G4VERBOSE
123 if ( verboseLevel > 0 )
124 {
125 G4cout << "=====================================" << G4endl;
126 G4cout << " G4EventManager::ProcessOneEvent() " << G4endl;
127 if(trackVector!=nullptr) G4cout << " for a sub-event" << G4endl;
128 G4cout << "=====================================" << G4endl;
129 }
130#endif
131
132 trackContainer->PrepareNewEvent(currentEvent);
133
134#ifdef G4_STORE_TRAJECTORY
135 trajectoryContainer = nullptr;
136// trajectoryContainer = currentEvent->GetTrajectoryContainer();
137// if(trajectoryContainer==nullptr) {
138// trajectoryContainer = new G4TrajectoryContainer;
139// currentEvent->SetTrajectoryContainer(trajectoryContainer);
140// }
141#endif
142
144 if(sdManager != nullptr)
145 { currentEvent->SetHCofThisEvent(sdManager->PrepareNewEvent()); }
146
147 if(!subEventParaWorker && userEventAction != nullptr) userEventAction->BeginOfEventAction(currentEvent);
148
149#ifdef G4VERBOSE
150 if ( verboseLevel > 1 )
151 {
152 G4cout << currentEvent->GetNumberOfPrimaryVertex()
153 << " vertices passed from G4Event." << G4endl;
154 }
155#endif
156
157 if(trackVector!=nullptr)
158 {
159 StackTracks(trackVector,IDhasAlreadySet);
160 }
161 if(!abortRequested)
162 {
163 StackTracks(transformer->GimmePrimaries(currentEvent,trackIDCounter), true);
164 }
165
166#ifdef G4VERBOSE
167 if ( verboseLevel > 0 )
168 {
169 G4cout << trackContainer->GetNTotalTrack() << " primary tracks "
170 << "are passed to the stack." << G4endl;
171 G4cout << "!!!!!!! Now start processing an event !!!!!!!" << G4endl;
172 }
173#endif
174
175 std::unordered_set<G4VTrackingManager *> trackingManagersToFlush;
176
177 do
178 {
179 G4VTrajectory* previousTrajectory;
180 while( (track=trackContainer->PopNextTrack(&previousTrajectory)) != nullptr )
181 { // Loop checking 12.28.2015 M.Asai
182
183 const G4ParticleDefinition* partDef = track->GetParticleDefinition();
184 G4VTrackingManager* particleTrackingManager = partDef->GetTrackingManager();
185
186 if (particleTrackingManager != nullptr)
187 {
188#ifdef G4VERBOSE
189 if ( verboseLevel > 1 )
190 {
191 G4cout << "Track " << track << " (trackID " << track->GetTrackID()
192 << ", parentID " << track->GetParentID()
193 << ") is handed over to custom TrackingManager." << G4endl;
194 }
195#endif
196
197 particleTrackingManager->HandOverOneTrack(track);
198 // The particle's tracking manager may either track immediately or
199 // defer processing until FlushEvent is called. Thus, we must neither
200 // check the track's status nor stack secondaries.
201
202 // Remember this tracking manager to later call FlushEvent.
203 trackingManagersToFlush.insert(particleTrackingManager);
204
205 } else {
206#ifdef G4VERBOSE
207 if ( verboseLevel > 1 )
208 {
209 G4cout << "Track " << track << " (trackID " << track->GetTrackID()
210 << ", parentID " << track->GetParentID()
211 << ") is passed to G4TrackingManager." << G4endl;
212 }
213#endif
214
215 tracking = true;
216 trackManager->ProcessOneTrack( track );
217 istop = track->GetTrackStatus();
218 tracking = false;
219
220#ifdef G4VERBOSE
221 if ( verboseLevel > 0 )
222 {
223 G4cout << "Track (trackID " << track->GetTrackID()
224 << ", parentID " << track->GetParentID()
225 << ") is processed with stopping code " << istop << G4endl;
226 }
227#endif
228
229 G4VTrajectory* aTrajectory = nullptr;
230#ifdef G4_STORE_TRAJECTORY
231 aTrajectory = trackManager->GimmeTrajectory();
232
233 if(previousTrajectory != nullptr)
234 {
235 previousTrajectory->MergeTrajectory(aTrajectory);
236 delete aTrajectory;
237 aTrajectory = previousTrajectory;
238 }
239 if((aTrajectory != nullptr)&&(istop!=fStopButAlive)
240 &&(istop!=fSuspend)&&(istop!=fSuspendAndWait))
241 {
242 if(trajectoryContainer == nullptr)
243 {
244 trajectoryContainer = new G4TrajectoryContainer;
245 currentEvent->SetTrajectoryContainer(trajectoryContainer);
246 }
247 trajectoryContainer->insert(aTrajectory);
248 }
249#endif
250
251 G4TrackVector* secondaries = trackManager->GimmeSecondaries();
252 switch (istop)
253 {
254 case fStopButAlive:
255 case fSuspend:
256 case fSuspendAndWait:
257 trackContainer->PushOneTrack( track, aTrajectory );
258 StackTracks( secondaries );
259 break;
260
262 trackContainer->PushOneTrack( track );
263 StackTracks( secondaries );
264 break;
265
266 case fStopAndKill:
267 StackTracks( secondaries );
268 delete track;
269 break;
270
271 case fAlive:
272 G4Exception("G4EventManager::DoProcessing", "Event004", JustWarning,
273 "Illegal track status returned from G4TrackingManager."\
274 " Continue with simulation.");
275 break;
276
278 if( secondaries != nullptr )
279 {
280 for(auto & secondarie : *secondaries)
281 { delete secondarie; }
282 secondaries->clear();
283 }
284 delete track;
285 break;
286 }
287 }
288 }
289
290 // Flush all tracking managers, which may have deferred processing until now.
291 for (G4VTrackingManager *tm : trackingManagersToFlush)
292 {
293 tm->FlushEvent();
294 }
295 trackingManagersToFlush.clear();
296
297 // flush any fast simulation models
299
300 // Check if flushing one of the tracking managers or a fast simulation model
301 // stacked new secondaries.
302 } while (trackContainer->GetNUrgentTrack() > 0);
303
304#ifdef G4VERBOSE
305 if ( verboseLevel > 0 )
306 {
307 G4cout << "NULL returned from G4StackManager." << G4endl;
308 G4cout << "Terminate current event processing." << G4endl;
309 }
310#endif
311
312 if(sdManager != nullptr)
313 {
314 sdManager->TerminateCurrentEvent(currentEvent->GetHCofThisEvent());
315 }
316
317// In case of sub-event parallelism, an event may not be completed at
318// this point but results of sus-events may be merged later. Thus
319// userEventAction->EndOfEventAction() is invoked by G4RunManager
320// immediately prior to deleting the event.
321 if(!subEventPara && (userEventAction != nullptr))
322 {
323 userEventAction->EndOfEventAction(currentEvent);
324 }
325
326 // Store remaining sub-events to the current event
327 auto nses = trackContainer->GetNSubEventTypes();
328 if(nses>0)
329 {
330#ifdef G4VERBOSE
331 if ( verboseLevel > 2 )
332 {
333 G4cout<<"## End of processing an event --- "
334 <<nses<<" sub-event types registered."<<G4endl;
335 }
336#endif
337 for(std::size_t i=0;i<nses;i++)
338 {
339 auto ty = trackContainer->GetSubEventType(i);
340 trackContainer->ReleaseSubEvent(ty);
341 }
342 }
343
344 if(!subEventParaWorker) stateManager->SetNewState(G4State_GeomClosed);
345 currentEvent = nullptr;
346 abortRequested = false;
347}
348
350{
351 G4AutoLock lock(&EventMgrMutex);
352 if(currentEvent==nullptr) return nullptr;
353 return currentEvent->PopSubEvent(ty);
354}
355
357{
358 G4AutoLock lock(&EventMgrMutex);
359 auto ev = se->GetEvent();
360 ev->MergeSubEventResults(evt);
361 if(!subEventParaWorker && userEventAction!=nullptr) userEventAction->MergeSubEvent(ev,evt);
362#ifdef G4VERBOSE
363 // Capture this here because termination will delete subevent...
364 G4int seType = se->GetSubEventType();
365#endif
366 ev->TerminateSubEvent(const_cast<G4SubEvent*>(se));
367#ifdef G4VERBOSE
368 if ( verboseLevel > 1 )
369 {
370 G4cout << "A sub-event of type " << seType
371 << " is merged to the event " << ev->GetEventID() << G4endl;
372 if(ev->GetNumberOfRemainingSubEvents()>0)
373 {
374 G4cout << " ---- This event still has " << ev->GetNumberOfRemainingSubEvents()
375 << " sub-events to be processed." << G4endl;
376 }
377 else
378 { G4cout << " ---- This event has no more sub-event remaining." << G4endl; }
379 }
380#endif
381}
382
384{
385 G4AutoLock lock(&EventMgrMutex);
386 if(evt != currentEvent) {
387 G4Exception("G4EventManager::StoreSubEvent","SubEvt1011", FatalException,
388 "StoreSubEvent is invoked with a G4Event that is not the current event. PANIC!");
389 }
390 return evt->StoreSubEvent(subEvtType,se);
391}
392
394 G4bool IDhasAlreadySet)
395{
396 if( trackVector != nullptr )
397 {
398 if( trackVector->empty() ) return;
399 for( auto newTrack : *trackVector )
400 {
401 ++trackIDCounter;
402 if(!IDhasAlreadySet)
403 {
404 newTrack->SetTrackID( trackIDCounter );
405 if(newTrack->GetDynamicParticle()->GetPrimaryParticle() != nullptr)
406 {
407 auto* pp
408 = (G4PrimaryParticle*)(newTrack->GetDynamicParticle()->GetPrimaryParticle());
409 pp->SetTrackID(trackIDCounter);
410 }
411 }
412 newTrack->SetOriginTouchableHandle(newTrack->GetTouchableHandle());
413 trackContainer->PushOneTrack( newTrack );
414#ifdef G4VERBOSE
415 if ( verboseLevel > 1 )
416 {
417 G4cout << "A new track " << newTrack
418 << " (trackID " << newTrack->GetTrackID()
419 << ", parentID " << newTrack->GetParentID()
420 << ") is passed to G4StackManager." << G4endl;
421 }
422#endif
423 }
424 trackVector->clear();
425 }
426}
427
429{
430 userEventAction = userAction;
431 if(userEventAction != nullptr)
432 {
433 userEventAction->SetEventManager(this);
434 }
435}
436
438{
439 userStackingAction = userAction;
440 trackContainer->SetUserStackingAction(userAction);
441}
442
444{
445 userTrackingAction = userAction;
446 trackManager->SetUserAction(userAction);
447}
448
450{
451 userSteppingAction = userAction;
452 trackManager->SetUserAction(userAction);
453}
454
456{
457 trackIDCounter = 0;
458 DoProcessing(anEvent);
459}
460
462 G4Event* anEvent)
463{
464 static G4ThreadLocal G4String* randStat = nullptr;
465 if (randStat == nullptr) randStat = new G4String;
466 G4bool tempEvent = false;
467 G4bool newEvent = false;
468 if(anEvent == nullptr)
469 {
470 anEvent = new G4Event();
471 tempEvent = true;
472 newEvent = true;
473 } else {
474 if(evID_inSubEv != anEvent->GetEventID()) {
475 evID_inSubEv = anEvent->GetEventID();
476 newEvent = true;
477 }
478 }
479 if(newEvent) trackIDCounter = 0;
480
481 if (storetRandomNumberStatusToG4Event==1
482 || storetRandomNumberStatusToG4Event==3)
483 {
484 std::ostringstream oss;
486 (*randStat) = oss.str();
487 anEvent->SetRandomNumberStatus(*randStat);
488 }
489 DoProcessing(anEvent,trackVector,false);
490 if(tempEvent) { delete anEvent; }
491}
492
494{
495 G4ApplicationState currentState = stateManager->GetCurrentState();
496 if(currentState != G4State_EventProc || currentEvent == nullptr)
497 {
498 G4Exception("G4EventManager::SetUserInformation",
499 "Event0003", JustWarning,
500 "G4VUserEventInformation cannot be set because of absence "\
501 "of G4Event.");
502 return;
503 }
504
505 currentEvent->SetUserInformation(anInfo);
506}
507
509{
510 G4ApplicationState currentState = stateManager->GetCurrentState();
511 if(currentState != G4State_EventProc || currentEvent == nullptr)
512 {
513 return nullptr;
514 }
515
516 return currentEvent->GetUserInformation();
517}
518
520{
521 if(currentEvent != nullptr) { currentEvent->KeepTheEvent(); }
522}
523
525{
526 abortRequested = true;
527 trackContainer->clear();
528 if(tracking) trackManager->EventAborted();
529}
G4ApplicationState
@ G4State_EventProc
@ G4State_GeomClosed
G4TemplateAutoLock< G4Mutex > G4AutoLock
@ JustWarning
@ FatalException
void G4Exception(const char *originOfException, const char *exceptionCode, G4ExceptionSeverity severity, const char *description)
#define G4MUTEX_INITIALIZER
std::mutex G4Mutex
CLHEP::Hep3Vector G4ThreeVector
G4TrackStatus
@ fKillTrackAndSecondaries
@ fSuspend
@ fAlive
@ fSuspendAndWait
@ fStopAndKill
@ fStopButAlive
@ fPostponeToNextEvent
std::vector< G4Track * > G4TrackVector
bool G4bool
Definition G4Types.hh:86
int G4int
Definition G4Types.hh:85
#define G4endl
Definition G4ios.hh:67
G4GLOB_DLL std::ostream G4cout
static std::ostream & saveFullState(std::ostream &os)
Definition Random.cc:288
void SetUserAction(G4UserEventAction *userAction)
void SetUserInformation(G4VUserEventInformation *anInfo)
G4int StoreSubEvent(G4Event *, G4int &, G4SubEvent *)
static G4EventManager * GetEventManager()
void StackTracks(G4TrackVector *trackVector, G4bool IDhasAlreadySet=false)
G4SubEvent * PopSubEvent(G4int ty)
void ProcessOneEvent(G4Event *anEvent)
G4VUserEventInformation * GetUserInformation()
void TerminateSubEvent(const G4SubEvent *se, const G4Event *evt)
G4int StoreSubEvent(G4int, G4SubEvent *)
Definition G4Event.cc:174
void SetRandomNumberStatus(G4String &st)
Definition G4Event.hh:91
void MergeSubEventResults(const G4Event *se)
Definition G4Event.cc:225
G4int GetEventID() const
Definition G4Event.hh:126
static G4GlobalFastSimulationManager * GetGlobalFastSimulationManager()
virtual G4VPhysicalVolume * LocateGlobalPointAndSetup(const G4ThreeVector &point, const G4ThreeVector *direction=nullptr, const G4bool pRelativeSearch=true, const G4bool ignoreDirection=true)
G4VTrackingManager * GetTrackingManager() const
static G4SDManager * GetSDMpointerIfExist()
const G4ApplicationState & GetCurrentState() const
static G4StateManager * GetStateManager()
G4int GetSubEventType() const
Definition G4SubEvent.hh:68
G4Event * GetEvent() const
Definition G4SubEvent.hh:76
G4TrackStatus GetTrackStatus() const
G4int GetTrackID() const
const G4ParticleDefinition * GetParticleDefinition() const
G4int GetParentID() const
static G4TransportationManager * GetTransportationManager()
G4Navigator * GetNavigatorForTracking() const
virtual void HandOverOneTrack(G4Track *aTrack)=0
virtual void MergeTrajectory(G4VTrajectory *secondTrajectory)=0
#define G4ThreadLocal
Definition tls.hh:77