13 using namespace CLHEP;
24 cout <<
" ------------- PHField2D::PHField2D() ------------------" << endl;
27 TFile *rootinput = TFile::Open(filename.c_str());
30 cout <<
" could not open " << filename <<
" exiting now" << endl;
33 if (
verb_ > 0) cout <<
" Field grid file: " << filename << endl;
36 Float_t ROOT_Z, ROOT_R;
37 Float_t ROOT_BZ, ROOT_BR;
39 TNtuple *field_map = (TNtuple*)gDirectory->Get(
"fieldmap");
46 field_map = (TNtuple*)gDirectory->Get(
"map");
49 cout <<
"PHField2D: could not locate ntuple of name map or fieldmap, exiting now" << endl;
54 field_map->SetBranchAddress(
"z", &ROOT_Z);
55 field_map->SetBranchAddress(
"r", &ROOT_R);
56 field_map->SetBranchAddress(
"bz", &ROOT_BZ);
57 field_map->SetBranchAddress(
"br", &ROOT_BR);
61 nz = field_map->GetEntries(
"z>-1e6");
62 nr = field_map->GetEntries(
"r>-1e6");
63 static const int NENTRIES = field_map->GetEntries();
66 if (
verb_ > 0) cout <<
" The field grid contained " << NENTRIES <<
" entries" << endl;
69 cout <<
"\n NENTRIES should be the same as the following values:"
70 <<
"\n [ Number of values r,z: "
71 << nr <<
" " << nz <<
" ]! " << endl;
76 cout <<
"!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!"
77 <<
"\n The file you entered is not a \"table\" of values"
78 <<
"\n Something very likely went oh so wrong"
79 <<
"!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!" << endl;
83 std::set<float> z_set, r_set;
92 cout <<
" --> Sorting Entries..." << endl;
94 std::map<trio, trio> sorted_map;
95 for (
int i = 0; i < field_map->GetEntries(); i++)
97 field_map->GetEntry(i);
98 trio coord_key( ROOT_Z, ROOT_R );
99 trio field_val( ROOT_BZ, ROOT_BR );
100 sorted_map[coord_key] = field_val;
102 z_set.insert(ROOT_Z*cm);
103 r_set.insert(ROOT_R*cm);
109 map<trio, trio>::iterator it = sorted_map.begin();
111 float last_z = it->first.get<0>();
112 for ( it = sorted_map.begin(); it != sorted_map.end(); ++it )
114 if ( it->first.get<0>() != last_z )
116 last_z = it->first.get<0>();
124 cout <<
" --> Putting entries into containers... " << endl;
128 minz_ = *(z_set.begin());
129 std::set<float>::iterator ziter = z_set.end();
139 std::copy(z_set.begin(), z_set.end(),
z_map_.begin());
140 std::copy(r_set.begin(), r_set.end(),
r_map_.begin());
143 BFieldR_.resize( nz, vector<float>(nr, 0) );
144 BFieldZ_.resize( nz, vector<float>(nr, 0) );
147 unsigned int ir = 0, iz = 0;
148 map<trio, trio>::iterator iter = sorted_map.begin();
149 for ( ; iter != sorted_map.end(); ++iter )
153 float z = iter-> first.get<0>() * cm;
154 float r = iter-> first.get<1>() * cm;
168 else if ( r !=
r_map_[ir] )
174 if ( iz > 0 && z <
z_map_[iz-1] )
176 cout <<
"!!!!!!!!! Your map isn't ordered.... z: " << z <<
" zprev: " <<
z_map_[iz-1] << endl;
179 BFieldR_[iz][ir] = Br * magfield_rescale;
180 BFieldZ_[iz][ir] = Bz * magfield_rescale;
185 if ( fabs(z) < 10 && ir < 10 &&
verb_ > 3)
201 if (
verb_ > 0) cout <<
" Mag field z boundaries (min,max): (" <<
minz_ / cm <<
", " <<
maxz_ / cm <<
") cm" << endl;
202 if (
verb_ > 0) cout <<
" Mag field r max boundary: " <<
r_map_.back()/ cm <<
" cm" << endl;
205 cout <<
" -----------------------------------------------------------" << endl;
211 cout <<
"\nPHField2D::GetFieldValue" << endl;
215 double r = sqrt(x * x + y * y);
218 if ( phi < 0 ) phi += 2 * M_PI;
225 double cylpoint[4] = { z, r, phi, 0 };
231 Bfield[0] = cos(phi) * BFieldCyl[1] - sin(phi) * BFieldCyl[2];
234 Bfield[1] = sin(phi) * BFieldCyl[1] + cos(phi) * BFieldCyl[2];
237 Bfield[2] = BFieldCyl[0];
247 cout <<
"!!!!!!!!!! Field point not in defined region (outside of z bounds)" << endl;
252 cout <<
"END PHField2D::GetFieldValue\n"
253 <<
" ---> {Bx, By, Bz} : "
254 <<
"< " << Bfield[0] <<
", " << Bfield[1] <<
", " << Bfield[2] <<
" >" << endl;
262 float z = CylPoint[0];
263 float r = CylPoint[1];
270 cout <<
"GetFieldCyl@ <z,r>: {" << z <<
"," << r <<
"}" << endl;
275 cout <<
"!!!! Point not in defined region (radius too large in specific z-plane)" << endl;
283 unsigned int r_index0 = r_index0_cache;
284 unsigned int r_index1 = r_index1_cache;
286 if (!((r >
r_map_[r_index0])&&(r <
r_map_[r_index1]))) {
289 vector<float>::const_iterator riter = upper_bound(
r_map_.begin(),
r_map_.end(), r);
290 r_index0 = distance(
r_map_.begin(), riter) - 1;
291 if (r_index0 >=
r_map_.size()) {
293 cout <<
"!!!! Point not in defined region (radius too large in specific z-plane)" << endl;
297 r_index1 = r_index0 + 1;
298 if (r_index1 >=
r_map_.size()) {
300 cout <<
"!!!! Point not in defined region (radius too large in specific z-plane)" << endl;
305 r_index0_cache = r_index0;
306 r_index1_cache = r_index1;
309 unsigned int z_index0 = z_index0_cache;
310 unsigned int z_index1 = z_index1_cache;
312 if (!((z >
z_map_[z_index0])&&(z <
z_map_[z_index1]))) {
315 vector<float>::const_iterator ziter = upper_bound(
z_map_.begin(),
z_map_.end(), z);
316 z_index0 = distance(
z_map_.begin(), ziter) - 1;
317 z_index1 = z_index0 + 1;
318 if (z_index1 >=
z_map_.size()) {
320 cout <<
"!!!! Point not in defined region (z too large in specific r-plane)" << endl;
325 z_index0_cache = z_index0;
326 z_index1_cache = z_index1;
329 double Br000 =
BFieldR_[z_index0][r_index0];
330 double Br010 =
BFieldR_[z_index0][r_index1];
331 double Br100 =
BFieldR_[z_index1][r_index0];
332 double Br110 =
BFieldR_[z_index1][r_index1];
334 double Bz000 =
BFieldZ_[z_index0][r_index0];
335 double Bz100 =
BFieldZ_[z_index1][r_index0];
336 double Bz010 =
BFieldZ_[z_index0][r_index1];
337 double Bz110 =
BFieldZ_[z_index1][r_index1];
339 double zweight = z -
z_map_[z_index0];
343 double rweight = r -
r_map_[r_index0];
349 (1 - zweight) * ((1 - rweight) * Bz000 +
351 zweight * ((1 - rweight) * Bz100 +
356 (1 - zweight) * ((1 - rweight) * Br000 +
358 zweight * ((1 - rweight) * Br100 +
366 cout <<
"End GFCyl Call: <bz,br,bphi> : {"
367 << BfieldCyl[0] / gauss <<
"," << BfieldCyl[1] / gauss <<
"," << BfieldCyl[2] / gauss <<
"}"
375 void PHField2D::print_map( map<trio, trio>::iterator& it )
const
379 << it->first.get<0>() / cm <<
","
380 << it->first.get<1>() / cm <<
">"
std::vector< std::vector< float > > BFieldR_
void GetFieldCyl(const double CylPoint[4], double *Bfield) const
PHField2D(const std::string &filename, const int verb=0, const float magfield_rescale=1.0)
std::vector< float > r_map_
std::vector< float > z_map_
std::vector< std::vector< float > > BFieldZ_
void GetFieldValue(const double Point[4], double *Bfield) const
transient DST object for field storage and access