Geant4 11.3.0
Toolkit for the simulation of the passage of particles through matter
Loading...
Searching...
No Matches
G4FieldManager Class Reference

#include <G4FieldManager.hh>

Public Member Functions

 G4FieldManager (G4Field *detectorField=nullptr, G4ChordFinder *pChordFinder=nullptr, G4bool b=true)
 
 G4FieldManager (G4MagneticField *detectorMagneticField)
 
virtual ~G4FieldManager ()
 
 G4FieldManager (const G4FieldManager &)=delete
 
G4FieldManageroperator= (const G4FieldManager &)=delete
 
G4bool SetDetectorField (G4Field *detectorField, G4int failMode=0)
 
void ProposeDetectorField (G4Field *detectorField)
 
void ChangeDetectorField (G4Field *detectorField)
 
const G4FieldGetDetectorField () const
 
G4bool DoesFieldExist () const
 
void CreateChordFinder (G4MagneticField *detectorMagField)
 
void SetChordFinder (G4ChordFinder *aChordFinder)
 
G4ChordFinderGetChordFinder ()
 
const G4ChordFinderGetChordFinder () const
 
virtual void ConfigureForTrack (const G4Track *)
 
G4double GetDeltaIntersection () const
 
G4double GetDeltaOneStep () const
 
void SetAccuraciesWithDeltaOneStep (G4double valDeltaOneStep)
 
void SetDeltaOneStep (G4double valueD1step)
 
void SetDeltaIntersection (G4double valueDintersection)
 
G4double GetMinimumEpsilonStep () const
 
G4bool SetMinimumEpsilonStep (G4double newEpsMin)
 
G4double GetMaximumEpsilonStep () const
 
G4bool SetMaximumEpsilonStep (G4double newEpsMax)
 
G4bool DoesFieldChangeEnergy () const
 
void SetFieldChangesEnergy (G4bool value)
 
virtual G4FieldManagerClone () const
 

Static Public Member Functions

static void SetGlobalFieldManager (G4FieldManager *fieldManager)
 
static G4FieldManagerGetGlobalFieldManager ()
 
static G4double GetMaxAcceptedEpsilon ()
 
static G4bool SetMaxAcceptedEpsilon (G4double maxEps, G4bool softFail=false)
 

Protected Member Functions

void ReportBadEpsilonValue (G4ExceptionDescription &erm, G4double value, G4String &name) const
 

Static Protected Attributes

static G4double fMaxAcceptedEpsilon = 0.01
 
static constexpr G4double fMinAcceptedEpsilon = 1000.0 * std::numeric_limits<G4double>::epsilon()
 
static constexpr G4double fMaxWarningEpsilon = 0.001
 
static constexpr G4double fMaxFinalEpsilon = 0.02
 
static G4bool fVerboseConstruction = false
 

Detailed Description

Definition at line 84 of file G4FieldManager.hh.

Constructor & Destructor Documentation

◆ G4FieldManager() [1/3]

G4FieldManager::G4FieldManager ( G4Field * detectorField = nullptr,
G4ChordFinder * pChordFinder = nullptr,
G4bool b = true )

Definition at line 50 of file G4FieldManager.cc.

53 : fDetectorField(detectorField),
54 fChordFinder(pChordFinder),
55 fDelta_One_Step_Value( fDefault_Delta_One_Step_Value ),
56 fDelta_Intersection_Val( fDefault_Delta_Intersection_Val ),
57 fEpsilonMin( fEpsilonMinDefault ),
58 fEpsilonMax( fEpsilonMaxDefault)
59{
60 if ( detectorField != nullptr )
61 {
62 fFieldChangesEnergy = detectorField->DoesFieldChangeEnergy();
63 }
64 else
65 {
66 fFieldChangesEnergy = fieldChangesEnergy;
67 }
68
70 {
71 G4cout << "G4FieldManager/ctor#1 fEpsilon Min/Max: eps_min = " << fEpsilonMin << " eps_max=" << fEpsilonMax << G4endl;
72 }
73
74 // Add to store
75 //
77}
#define G4endl
Definition G4ios.hh:67
G4GLOB_DLL std::ostream G4cout
static void Register(G4FieldManager *pVolume)
static G4bool fVerboseConstruction
virtual G4bool DoesFieldChangeEnergy() const =0

Referenced by Clone(), G4FieldManager(), GetGlobalFieldManager(), operator=(), SetFieldChangesEnergy(), and SetGlobalFieldManager().

◆ G4FieldManager() [2/3]

G4FieldManager::G4FieldManager ( G4MagneticField * detectorMagneticField)

Definition at line 79 of file G4FieldManager.cc.

80 : fDetectorField(detectorField), fAllocatedChordFinder(true),
81 fDelta_One_Step_Value( fDefault_Delta_One_Step_Value ),
82 fDelta_Intersection_Val( fDefault_Delta_Intersection_Val ),
83 fEpsilonMin( fEpsilonMinDefault ),
84 fEpsilonMax( fEpsilonMaxDefault )
85{
86 fChordFinder = new G4ChordFinder( detectorField );
87
89 {
90 G4cout << "G4FieldManager/ctor#2 fEpsilon Min/Max: eps_min = " << fEpsilonMin << " eps_max=" << fEpsilonMax << G4endl;
91 }
92 // Add to store
93 //
95}

◆ ~G4FieldManager()

G4FieldManager::~G4FieldManager ( )
virtual

Definition at line 159 of file G4FieldManager.cc.

160{
161 if( fAllocatedChordFinder )
162 {
163 delete fChordFinder;
164 }
166}
static void DeRegister(G4FieldManager *pVolume)

◆ G4FieldManager() [3/3]

G4FieldManager::G4FieldManager ( const G4FieldManager & )
delete

Member Function Documentation

◆ ChangeDetectorField()

void G4FieldManager::ChangeDetectorField ( G4Field * detectorField)
inline

◆ Clone()

G4FieldManager * G4FieldManager::Clone ( ) const
virtual

Definition at line 97 of file G4FieldManager.cc.

98{
99 G4Field* aField = nullptr;
100 G4FieldManager* aFM = nullptr;
101 G4ChordFinder* aCF = nullptr;
102 try {
103 if ( fDetectorField != nullptr )
104 {
105 aField = fDetectorField->Clone();
106 }
107
108 // Create a new field manager, note that we do not set
109 // any chordfinder now
110 //
111 aFM = new G4FieldManager( aField , nullptr , fFieldChangesEnergy );
112
113 // Check if originally we have the fAllocatedChordFinder variable
114 // set, in case, call chord constructor
115 //
116 if ( fAllocatedChordFinder )
117 {
118 aFM->CreateChordFinder( dynamic_cast<G4MagneticField*>(aField) );
119 }
120 else
121 {
122 // Chord was specified by user, should we clone?
123 // TODO: For the moment copy pointer, to be understood
124 // if cloning of ChordFinder is needed
125 //
126 aCF = fChordFinder; /*->Clone*/
127 aFM->fChordFinder = aCF;
128 }
129
130 // Copy values of other variables
131
132 aFM->fEpsilonMax = fEpsilonMax;
133 aFM->fEpsilonMin = fEpsilonMin;
134 aFM->fDelta_Intersection_Val = fDelta_Intersection_Val;
135 aFM->fDelta_One_Step_Value = fDelta_One_Step_Value;
136 // TODO: Should we really add to the store the cloned FM?
137 // Who will use this?
138 }
139 catch ( ... )
140 {
141 // Failed creating clone: probably user did not implement Clone method
142 // in derived classes?
143 // Perform clean-up after ourselves...
144 delete aField;
145 delete aFM;
146 delete aCF;
147 throw;
148 }
149
150 G4cout << "G4FieldManager/clone fEpsilon Min/Max: eps_min = " << fEpsilonMin << " eps_max=" << fEpsilonMax << G4endl;
151 return aFM;
152}
void CreateChordFinder(G4MagneticField *detectorMagField)
G4FieldManager(G4Field *detectorField=nullptr, G4ChordFinder *pChordFinder=nullptr, G4bool b=true)
virtual G4Field * Clone() const
Definition G4Field.cc:49

Referenced by G4VUserDetectorConstruction::CloneF().

◆ ConfigureForTrack()

void G4FieldManager::ConfigureForTrack ( const G4Track * )
virtual

Definition at line 154 of file G4FieldManager.cc.

155{
156 // Default is to do nothing!
157}

Referenced by G4CoupledTransportation::AlongStepGetPhysicalInteractionLength(), and G4ITTransportation::AlongStepGetPhysicalInteractionLength().

◆ CreateChordFinder()

void G4FieldManager::CreateChordFinder ( G4MagneticField * detectorMagField)

Definition at line 169 of file G4FieldManager.cc.

170{
171 if ( fAllocatedChordFinder )
172 {
173 delete fChordFinder;
174 }
175 fAllocatedChordFinder = false;
176
177 if( detectorMagField != nullptr )
178 {
179 fChordFinder = new G4ChordFinder( detectorMagField );
180 fAllocatedChordFinder = true;
181 }
182 else
183 {
184 fChordFinder = nullptr;
185 }
186}

Referenced by Clone().

◆ DoesFieldChangeEnergy()

G4bool G4FieldManager::DoesFieldChangeEnergy ( ) const
inline

◆ DoesFieldExist()

G4bool G4FieldManager::DoesFieldExist ( ) const
inline

◆ GetChordFinder() [1/2]

G4ChordFinder * G4FieldManager::GetChordFinder ( )
inline

◆ GetChordFinder() [2/2]

const G4ChordFinder * G4FieldManager::GetChordFinder ( ) const
inline

◆ GetDeltaIntersection()

G4double G4FieldManager::GetDeltaIntersection ( ) const
inline

◆ GetDeltaOneStep()

G4double G4FieldManager::GetDeltaOneStep ( ) const
inline

◆ GetDetectorField()

◆ GetGlobalFieldManager()

static G4FieldManager * G4FieldManager::GetGlobalFieldManager ( )
static

◆ GetMaxAcceptedEpsilon()

G4double G4FieldManager::GetMaxAcceptedEpsilon ( )
static

Definition at line 343 of file G4FieldManager.cc.

344{
345 return fMaxAcceptedEpsilon;
346}
static G4double fMaxAcceptedEpsilon

◆ GetMaximumEpsilonStep()

G4double G4FieldManager::GetMaximumEpsilonStep ( ) const
inline

◆ GetMinimumEpsilonStep()

G4double G4FieldManager::GetMinimumEpsilonStep ( ) const
inline

◆ operator=()

G4FieldManager & G4FieldManager::operator= ( const G4FieldManager & )
delete

◆ ProposeDetectorField()

void G4FieldManager::ProposeDetectorField ( G4Field * detectorField)
inline

◆ ReportBadEpsilonValue()

void G4FieldManager::ReportBadEpsilonValue ( G4ExceptionDescription & erm,
G4double value,
G4String & name ) const
protected

Definition at line 422 of file G4FieldManager.cc.

424{
425 erm << "Incorrect proposed value of " << name << " = " << value << G4endl
426 << " Its value is outside the permitted range from "
428 << " Clarification: " << G4endl;
429 G4long oldPrec = erm.precision();
430 if(value < fMinAcceptedEpsilon )
431 {
432 erm << " a) The value must be positive and enough larger than the accuracy limit"
433 << " of the (G4)double type - ("
434 << (value < fMinAcceptedEpsilon ? "FAILED" : "OK" ) << ")" << G4endl
435 << " i.e. std::numeric_limits<G4double>::epsilon()= "
436 << std::numeric_limits<G4double>::epsilon()
437 << " to ensure that integration " << G4endl
438 << " could potentially achieve this acccuracy." << G4endl
439 << " Minimum accepted eps_min/max value = " << fMinAcceptedEpsilon << G4endl;
440 }
441 else if( value > fMaxAcceptedEpsilon)
442 {
443 erm << " b) It must be smaller than (or equal) " << std::setw(8)
444 << std::setprecision(4) << fMaxAcceptedEpsilon
445 << " to ensure robustness of integration - ("
446 << (( value < fMaxAcceptedEpsilon) ? "OK" : "FAILED" ) << ")" << G4endl;
447 }
448 else
449 {
450 G4bool badRoundoff = (std::fabs(1.0+value) == 1.0);
451 erm << " Unknown ERROR case -- extra check: " << G4endl;
452 erm << " c) as a floating point number (of type G4double) the sum (1+" << name
453 << " ) must be > 1 , ("
454 << (badRoundoff ? "FAILED" : "OK" ) << ")" << G4endl
455 << " Now 1+eps_min = " << std::setw(20)
456 << std::setprecision(17) << (1+value) << G4endl
457 << " and (1.0+" << name << ") - 1.0 = " << std::setw(20)
458 << std::setprecision(9) << (1.0+value)-1.0;
459 }
460 erm.precision(oldPrec);
461}
long G4long
Definition G4Types.hh:87
bool G4bool
Definition G4Types.hh:86
static constexpr G4double fMinAcceptedEpsilon
const char * name(G4int ptype)

Referenced by SetMaximumEpsilonStep(), and SetMinimumEpsilonStep().

◆ SetAccuraciesWithDeltaOneStep()

void G4FieldManager::SetAccuraciesWithDeltaOneStep ( G4double valDeltaOneStep)
inline

◆ SetChordFinder()

void G4FieldManager::SetChordFinder ( G4ChordFinder * aChordFinder)
inline

◆ SetDeltaIntersection()

void G4FieldManager::SetDeltaIntersection ( G4double valueDintersection)
inline

◆ SetDeltaOneStep()

void G4FieldManager::SetDeltaOneStep ( G4double valueD1step)
inline

◆ SetDetectorField()

G4bool G4FieldManager::SetDetectorField ( G4Field * detectorField,
G4int failMode = 0 )

Definition at line 200 of file G4FieldManager.cc.

202{
203 G4VIntegrationDriver* driver = nullptr;
204 G4EquationOfMotion* equation = nullptr;
205 // G4bool compatibleField = false;
206 G4bool ableToSet = false;
207
208 fDetectorField = pDetectorField;
209 InitialiseFieldChangesEnergy();
210
211 // Must 'propagate' the field to the dependent classes
212 //
213 if( fChordFinder != nullptr )
214 {
215 failMode= std::max( failMode, 1) ;
216 // If a chord finder exists, warn in case of error!
217
218 driver = fChordFinder->GetIntegrationDriver();
219 if( driver != nullptr )
220 {
221 equation = driver->GetEquationOfMotion();
222
223 // Should check the compatibility between the
224 // field and the equation HERE
225
226 if( equation != nullptr )
227 {
228 equation->SetFieldObj(pDetectorField);
229 ableToSet = true;
230 }
231 }
232 }
233
234 if( !ableToSet && (failMode > 0) )
235 {
236 // If this fails, report the issue !
237
239 msg << "Unable to set the field in the dependent objects of G4FieldManager"
240 << G4endl;
241 msg << "All the dependent classes must be fully initialised,"
242 << "before it is possible to call this method." << G4endl;
243 msg << "The problem encountered was the following: " << G4endl;
244 if( fChordFinder == nullptr ) { msg << " No ChordFinder. " ; }
245 else if( driver == nullptr ) { msg << " No Integration Driver set. ";}
246 else if( equation == nullptr ) { msg << " No Equation found. " ; }
247 // else if( !compatibleField ) { msg << " Field not compatible. ";}
248 else { msg << " Can NOT find reason for failure. ";}
249 msg << G4endl;
250 G4ExceptionSeverity severity = (failMode != 1)
252 G4Exception("G4FieldManager::SetDetectorField", "Geometry001",
253 severity, msg);
254 }
255 return ableToSet;
256}
G4ExceptionSeverity
@ JustWarning
@ FatalException
void G4Exception(const char *originOfException, const char *exceptionCode, G4ExceptionSeverity severity, const char *description)
std::ostringstream G4ExceptionDescription
void SetFieldObj(G4Field *pField)
virtual G4EquationOfMotion * GetEquationOfMotion()=0

◆ SetFieldChangesEnergy()

void G4FieldManager::SetFieldChangesEnergy ( G4bool value)
inline

◆ SetGlobalFieldManager()

static void G4FieldManager::SetGlobalFieldManager ( G4FieldManager * fieldManager)
static

◆ SetMaxAcceptedEpsilon()

G4bool G4FieldManager::SetMaxAcceptedEpsilon ( G4double maxEps,
G4bool softFail = false )
static

Definition at line 350 of file G4FieldManager.cc.

352{
353 G4bool success= false;
354 // Limit for warning and absolute limit chosen from experience in and
355 // investigation of integration with G4DormandPrince745 in HEP-type setups.
356 if( maxAcceptValue <= fMaxWarningEpsilon )
357 {
358 fMaxAcceptedEpsilon= maxAcceptValue;
359 success= true;
360 }
361 else
362 {
364 G4ExceptionSeverity severity;
365
366 G4cout << "G4FieldManager::" << __func__
367 << " Parameters: fMaxAcceptedEpsilon = " << fMaxAcceptedEpsilon
368 << " fMaxFinalEpsilon = " << fMaxFinalEpsilon << G4endl;
369
370 if( maxAcceptValue <= fMaxFinalEpsilon )
371 {
372 success= true;
373 fMaxAcceptedEpsilon = maxAcceptValue;
374 // Integration is poor, and robustness will likely suffer
375 erm << "Proposed value for maximum-accepted-epsilon = " << maxAcceptValue
376 << " is larger than the recommended = " << fMaxWarningEpsilon
377 << G4endl
378 << "This may impact the robustness of integration of tracks in field."
379 << G4endl
380 << "The request was accepted and the value = " << fMaxAcceptedEpsilon
381 << " , but future releases are expected " << G4endl
382 << " to tighten the limit of acceptable values to "
384 << "Suggestion: If you need better performance investigate using "
385 << "alternative, low-order RK integration methods or " << G4endl
386 << " helix-based methods (for pure B-fields) for low(er) energy tracks, "
387 << " especially electrons if you need better performance." << G4endl;
388 severity= JustWarning;
389 }
390 else
391 {
393 erm << " Proposed value for maximum accepted epsilon " << maxAcceptValue
394 << " is larger than the top of the range = " << fMaxFinalEpsilon
395 << G4endl;
396 if( softFailure )
397 {
398 erm << " Using the latter value instead." << G4endl;
399 }
400 erm << G4endl;
401 erm << " Please adjust to request maxAccepted <= " << fMaxFinalEpsilon
402 << G4endl << G4endl;
403 if( !softFailure )
404 {
405 erm << " NOTE: you can accept the ceiling value and turn this into a "
406 << " warning by using a 2nd argument " << G4endl
407 << " in your call to SetMaxAcceptedEpsilon: softFailure = true ";
408 }
409 severity = softFailure ? JustWarning : FatalException;
410 // if( softFailure ) severity= JustWarning;
411 // else severity= FatalException;
412 success = false;
413 }
414 G4String methodName = G4String("G4FieldManager::")+ G4String(__func__);
415 G4Exception(methodName.c_str(), "Geometry003", severity, erm);
416 }
417 return success;
418}
static constexpr G4double fMaxWarningEpsilon
static constexpr G4double fMaxFinalEpsilon

◆ SetMaximumEpsilonStep()

G4bool G4FieldManager::SetMaximumEpsilonStep ( G4double newEpsMax)

Definition at line 258 of file G4FieldManager.cc.

259{
260 G4bool succeeded= false;
261 if( (newEpsMax > 0.0) && ( newEpsMax <= fMaxAcceptedEpsilon)
262 && (fMinAcceptedEpsilon <= newEpsMax ) ) // (std::fabs(1.0+newEpsMax)>1.0) )
263 {
264 if(newEpsMax >= fEpsilonMin)
265 {
266 fEpsilonMax = newEpsMax;
267 succeeded = true;
269 {
270 G4cout << "G4FieldManager/SetEpsMax : eps_max = " << std::setw(10) << fEpsilonMax
271 << " ( Note: unchanged eps_min=" << std::setw(10) << fEpsilonMin << " )" << G4endl;
272 }
273 }
274 else
275 {
277 erm << " Call to set eps_max = " << newEpsMax << " . The problem is that"
278 << " its value must be at larger or equal to eps_min= " << fEpsilonMin << G4endl;
279 erm << " Modifying both to the same value " << newEpsMax << " to ensure consistency."
280 << G4endl
281 << " To avoid this warning, please set eps_min first, and ensure that "
282 << " 0 < eps_min <= eps_max <= " << fMaxAcceptedEpsilon << G4endl;
283
284 fEpsilonMax = newEpsMax;
285 fEpsilonMin = newEpsMax;
286 G4String methodName = G4String("G4FieldManager::")+ G4String(__func__);
287 G4Exception(methodName.c_str(), "Geometry003", JustWarning, erm);
288 }
289 }
290 else
291 {
293 G4String paramName("eps_max");
294 ReportBadEpsilonValue(erm, newEpsMax, paramName );
295 G4String methodName = G4String("G4FieldManager::")+ G4String(__func__);
296 G4Exception(methodName.c_str(), "Geometry001", FatalException, erm);
297 }
298 return succeeded;
299}
void ReportBadEpsilonValue(G4ExceptionDescription &erm, G4double value, G4String &name) const

◆ SetMinimumEpsilonStep()

G4bool G4FieldManager::SetMinimumEpsilonStep ( G4double newEpsMin)

Definition at line 303 of file G4FieldManager.cc.

304{
305 G4bool succeeded= false;
306
307 if( fMinAcceptedEpsilon <= newEpsMin && newEpsMin <= fMaxAcceptedEpsilon )
308 {
309 fEpsilonMin = newEpsMin;
310 //*********
311 succeeded= true;
312
314 {
315 G4cout << "G4FieldManager/SetEpsMin : eps_min = "
316 << std::setw(10) << fEpsilonMin << G4endl;
317 }
318 if( fEpsilonMax < fEpsilonMin )
319 {
320 // Ensure consistency
322 erm << "Setting eps_min = " << newEpsMin
323 << " For consistency set eps_max= " << fEpsilonMin
324 << " ( Old value = " << fEpsilonMax << " )" << G4endl;
325 fEpsilonMax = fEpsilonMin;
326 G4String methodName = G4String("G4FieldManager::")+ G4String(__func__);
327 G4Exception(methodName.c_str(), "Geometry003", JustWarning, erm);
328 }
329 }
330 else
331 {
333 G4String paramName("eps_min");
334 ReportBadEpsilonValue(erm, newEpsMin, paramName );
335 G4String methodName = G4String("G4FieldManager::")+ G4String(__func__);
336 G4Exception(methodName.c_str(), "Geometry001", FatalException, erm);
337 }
338 return succeeded;
339}

Member Data Documentation

◆ fMaxAcceptedEpsilon

G4double G4FieldManager::fMaxAcceptedEpsilon = 0.01
staticprotected

◆ fMaxFinalEpsilon

G4double G4FieldManager::fMaxFinalEpsilon = 0.02
staticconstexprprotected

Definition at line 189 of file G4FieldManager.hh.

Referenced by SetMaxAcceptedEpsilon().

◆ fMaxWarningEpsilon

G4double G4FieldManager::fMaxWarningEpsilon = 0.001
staticconstexprprotected

Definition at line 188 of file G4FieldManager.hh.

Referenced by SetMaxAcceptedEpsilon().

◆ fMinAcceptedEpsilon

G4double G4FieldManager::fMinAcceptedEpsilon = 1000.0 * std::numeric_limits<G4double>::epsilon()
staticconstexprprotected

◆ fVerboseConstruction

G4bool G4FieldManager::fVerboseConstruction = false
staticprotected

The documentation for this class was generated from the following files: