Class Reference for E1039 Core & Analysis Software
G4TBMagneticFieldSetup.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 //
27 // $Id: G4TBMagneticFieldSetup.cc,v 1.24 2014/11/14 21:47:38 mccumber Exp $
28 // GEANT4 tag $Name: $
29 //
30 //
31 // User Field class implementation.
32 //
33 
35 #include "G4TBFieldMessenger.hh"
36 #include "PHG4MagneticField.h"
37 
38 #include <fun4all/Fun4AllServer.h>
39 #include <phool/getClass.h>
40 
41 #include <phool/PHCompositeNode.h>
42 #include <phool/PHIODataNode.h>
43 #include <phool/PHNodeIterator.h>
44 
45 #include <Geant4/G4SystemOfUnits.hh>
46 #include <Geant4/G4UniformMagField.hh>
47 #include <Geant4/G4MagneticField.hh>
48 #include <Geant4/G4FieldManager.hh>
49 #include <Geant4/G4TransportationManager.hh>
50 #include <Geant4/G4EquationOfMotion.hh>
51 #include <Geant4/G4EqMagElectricField.hh>
52 #include <Geant4/G4Mag_UsualEqRhs.hh>
53 #include <Geant4/G4MagIntegratorStepper.hh>
54 #include <Geant4/G4MagIntegratorDriver.hh>
55 #include <Geant4/G4ChordFinder.hh>
56 
57 #include <Geant4/G4ExplicitEuler.hh>
58 #include <Geant4/G4ImplicitEuler.hh>
59 #include <Geant4/G4SimpleRunge.hh>
60 #include <Geant4/G4SimpleHeum.hh>
61 #include <Geant4/G4ClassicalRK4.hh>
62 #include <Geant4/G4HelixExplicitEuler.hh>
63 #include <Geant4/G4HelixImplicitEuler.hh>
64 #include <Geant4/G4HelixSimpleRunge.hh>
65 #include <Geant4/G4CashKarpRKF45.hh>
66 #include <Geant4/G4RKG3_Stepper.hh>
67 
68 
69 #include <sstream>
70 #include <cassert>
71 
72 using namespace std;
73 
75 //
76 // Constructors:
77 
79  : verbosity(0),
80  fChordFinder(0),
81  fStepper(0),
82  fIntgrDriver(0)
83 {
84  assert(phfield);
85 
86  fEMfield = new PHG4MagneticField(phfield);
87  fFieldMessenger = new G4TBFieldMessenger(this) ;
88  fEquation = new G4Mag_UsualEqRhs(fEMfield);
89  fMinStep = 0.005*mm ; // minimal step of 10 microns
90  fStepperType = 4 ; // ClassicalRK4 -- the default stepper
91 
92  fFieldManager = GetGlobalFieldManager();
93  UpdateField();
94  double point[4] = {0,0,0,0};
95  fEMfield->GetFieldValue(&point[0],&magfield_at_000[0]);
96  for (size_t i=0; i<sizeof(magfield_at_000)/sizeof(double);i++)
97  {
98  magfield_at_000[i] = magfield_at_000[i]/tesla;
99  }
100  if (verbosity > 0)
101  {
102  cout << "field: x" << magfield_at_000[0]
103  << ", y: " << magfield_at_000[1]
104  << ", z: " << magfield_at_000[2]
105  << endl;
106  }
107 
108  fDummyFieldManager = new G4FieldManager();
109  fDummyFieldManager->SetDetectorField(0);
110 }
111 
112 //G4TBMagneticFieldSetup::G4TBMagneticFieldSetup(const float magfield)
113 // : verbosity(0), fChordFinder(0), fStepper(0), fIntgrDriver(0)
114 //{
115 // //solenoidal field along the axis of the cyclinders?
116 // fEMfield = new G4UniformMagField(G4ThreeVector(0.0, 0.0, magfield*tesla));
117 // fFieldMessenger = new G4TBFieldMessenger(this) ;
118 // fEquation = new G4Mag_UsualEqRhs(fEMfield);
119 // fMinStep = 0.005 * mm ; // minimal step of 10 microns
120 // fStepperType = 4 ; // ClassicalRK4 -- the default stepper
121 //
122 // fFieldManager = GetGlobalFieldManager();
123 // UpdateField();
124 //
125 // double point[4] = {0,0,0,0};
126 // fEMfield->GetFieldValue(&point[0],&magfield_at_000[0]);
127 // for (size_t i=0; i<sizeof(magfield_at_000)/sizeof(double);i++)
128 // {
129 // magfield_at_000[i] = magfield_at_000[i]/tesla;
130 // }
131 //}
132 //
134 //
135 //G4TBMagneticFieldSetup::G4TBMagneticFieldSetup(const string &fieldmapname, const int dim, const float magfield_rescale)
136 // : verbosity(0),
137 // fChordFinder(0),
138 // fStepper(0),
139 // fIntgrDriver(0)
140 //{
141 //
142 // switch(dim)
143 // {
144 // case 1:
145 // fEMfield = new PHG4FieldsPHENIX(fieldmapname,magfield_rescale);
146 // break;
147 // case 2:
148 // fEMfield = new PHG4Field2D(fieldmapname,0,magfield_rescale);
149 // break;
150 // case 3:
151 // fEMfield = new PHG4Field3D(fieldmapname,0,magfield_rescale);
152 // break;
153 // default:
154 // cout << "Invalid dimension, valid is 2 for 2D, 3 for 3D" << endl;
155 // exit(1);
156 // }
157 // fFieldMessenger = new G4TBFieldMessenger(this) ;
158 // fEquation = new G4Mag_UsualEqRhs(fEMfield);
159 // fMinStep = 0.005*mm ; // minimal step of 10 microns
160 // fStepperType = 4 ; // ClassicalRK4 -- the default stepper
161 //
162 // fFieldManager = GetGlobalFieldManager();
163 // UpdateField();
164 // double point[4] = {0,0,0,0};
165 // fEMfield->GetFieldValue(&point[0],&magfield_at_000[0]);
166 // for (size_t i=0; i<sizeof(magfield_at_000)/sizeof(double);i++)
167 // {
168 // magfield_at_000[i] = magfield_at_000[i]/tesla;
169 // }
170 // if (verbosity > 0)
171 // {
172 // cout << "field: x" << magfield_at_000[0]
173 // << ", y: " << magfield_at_000[1]
174 // << ", z: " << magfield_at_000[2]
175 // << endl;
176 // }
177 //}
178 
179 
181 
183 {
184  delete fChordFinder;
185  delete fStepper;
186  delete fFieldMessenger;
187  delete fEquation;
188  delete fEMfield;
189 }
190 
192 //
193 // Register this field to 'global' Field Manager and
194 // Create Stepper and Chord Finder with predefined type, minstep (resp.)
195 //
196 
198 {
199  SetStepper();
200 
201  fFieldManager->SetDetectorField(fEMfield );
202 
203  delete fChordFinder;
204 
205  fIntgrDriver = new G4MagInt_Driver(fMinStep,
206  fStepper,
207  fStepper->GetNumberOfVariables() );
208 
209  fChordFinder = new G4ChordFinder(fIntgrDriver);
210 
211  fFieldManager->SetChordFinder( fChordFinder );
212 }
213 
215 //
216 // Set stepper according to the stepper type
217 //
218 
220 {
221  G4int nvar = 8;
222 
223  delete fStepper;
224 
225  std::stringstream message;
226 
227  switch ( fStepperType )
228  {
229  case 0:
230  fStepper = new G4ExplicitEuler( fEquation, nvar );
231  message << "Stepper in use: G4ExplicitEuler";
232  break;
233  case 1:
234  fStepper = new G4ImplicitEuler( fEquation, nvar );
235  message << "Stepper in use: G4ImplicitEuler";
236  break;
237  case 2:
238  fStepper = new G4SimpleRunge( fEquation, nvar );
239  message << "Stepper in use: G4SimpleRunge";
240  break;
241  case 3:
242  fStepper = new G4SimpleHeum( fEquation, nvar );
243  message << "Stepper in use: G4SimpleHeum";
244  break;
245  case 4:
246  fStepper = new G4ClassicalRK4( fEquation, nvar );
247  message << "Stepper in use: G4ClassicalRK4 (default)";
248  break;
249  case 5:
250  fStepper = new G4CashKarpRKF45( fEquation, nvar );
251  message << "Stepper in use: G4CashKarpRKF45";
252  break;
253  case 6:
254  fStepper = NULL; // new G4RKG3_Stepper( fEquation, nvar );
255  message << "G4RKG3_Stepper is not currently working for Magnetic Field";
256  break;
257  case 7:
258  fStepper = NULL; // new G4HelixExplicitEuler( fEquation );
259  message << "G4HelixExplicitEuler is not valid for Magnetic Field";
260  break;
261  case 8:
262  fStepper = NULL; // new G4HelixImplicitEuler( fEquation );
263  message << "G4HelixImplicitEuler is not valid for Magnetic Field";
264  break;
265  case 9:
266  fStepper = NULL; // new G4HelixSimpleRunge( fEquation );
267  message << "G4HelixSimpleRunge is not valid for Magnetic Field";
268  break;
269  default: fStepper = NULL;
270  }
271 
272  if (verbosity > 0) {
273  G4cout << " ---------- G4TBMagneticFieldSetup::SetStepper() -----------" << G4endl;
274  G4cout << " " << message.str() << endl;
275  G4cout << " Minimum step size: " << fMinStep/mm << " mm" << G4endl;
276  G4cout << " -----------------------------------------------------------" << G4endl;
277  }
278 
279  if (!fStepper)
280  {
281  cout << "no stepper set, edxiting now" << endl;
282  exit(1);
283  }
284 
285  return;
286 }
287 
289 //
290 // Set the value of the Global Field to fieldValue along Z
291 //
292 
293 void G4TBMagneticFieldSetup::SetFieldValue(const G4double fieldValue)
294 {
295  G4ThreeVector fieldVector( 0.0, 0.0, fieldValue );
296 
297  SetFieldValue( fieldVector );
298 }
299 
301 //
302 // Set the value of the Global Field value to fieldVector
303 //
304 
305 void G4TBMagneticFieldSetup::SetFieldValue(const G4ThreeVector fieldVector)
306 {
307  // Find the Field Manager for the global field
308  G4FieldManager* fieldMgr= GetGlobalFieldManager();
309 
310  if(fieldVector != G4ThreeVector(0.,0.,0.))
311  {
312  if(fEMfield) delete fEMfield;
313  fEMfield = new G4UniformMagField(fieldVector);
314 
315  fEquation->SetFieldObj(fEMfield); // must now point to the new field
316 
317  // UpdateField();
318 
319  fieldMgr->SetDetectorField(fEMfield);
320  }
321  else
322  {
323  // If the new field's value is Zero, then it is best to
324  // insure that it is not used for propagation.
325  if(fEMfield) delete fEMfield;
326  fEMfield = 0;
327  fEquation->SetFieldObj(fEMfield); // As a double check ...
328 
329  G4MagneticField* fEMfield = 0;
330  fieldMgr->SetDetectorField(fEMfield);
331  }
332 }
333 
335 //
336 // Utility method
337 
339 {
340  return G4TransportationManager::GetTransportationManager()->GetFieldManager();
341 }
#define NULL
Definition: Pdb.h:9
G4FieldManager * GetGlobalFieldManager()
void SetFieldValue(const G4ThreeVector fieldVector)
G4TBMagneticFieldSetup(PHField *phfield)
transient DST object for field storage and access
Definition: PHField.h:14
PHG4MagneticField interfaces with Geant4.