// $Id$ // // File: DMagneticFieldMapCalibDB.cc // Created: Thu Jul 19 13:58:21 EDT 2007 // Creator: davidl (on Darwin fwing-dhcp61.jlab.org 8.10.1 i386) // #include using namespace std; #include "DMagneticFieldMapCalibDB.h" //--------------------------------- // DMagneticFieldMapCalibDB (Constructor) //--------------------------------- DMagneticFieldMapCalibDB::DMagneticFieldMapCalibDB(JApplication *japp) { int runnumber = 1; jcalib = japp->GetJCalibration(runnumber); int Npoints = ReadMap(runnumber); if(Npoints==0){ _DBG_<<"Error getting JCalibration object for magnetic field!"<Quit(); } } //--------------------------------- // DMagneticFieldMapCalibDB (Constructor) //--------------------------------- DMagneticFieldMapCalibDB::DMagneticFieldMapCalibDB(JCalibration *jcalib) { this->jcalib = jcalib; if(ReadMap()==0){ _DBG_<<"Error getting JCalibration object for magnetic field!"< > Bmap; jcalib->Get(namepath, Bmap); cout< to make a // histogram of the entries by using the key to hold the extent // so that the number of entries will be equal to the number of // different values. map xvals; map yvals; map zvals; xmin = ymin = zmin = 1.0E6; xmax = ymax = zmax = -1.0E6; for(unsigned int i=0; i &a = Bmap[i]; float &x = a[0]; float &y = a[1]; float &z = a[2]; // Convert from inches to cm and shift in z by 26 inches to align // the origin with the GEANT defined one. x *= 2.54; y *= 2.54; z = (z-26.0)*2.54; xvals[x] = 1; yvals[y] = 1; zvals[z] = 1; if(xxmax)xmax=x; if(y>ymax)ymax=y; if(z>zmax)zmax=z; } Nx = xvals.size(); Ny = yvals.size(); Nz = zvals.size(); cout<<" Nx="< zvec(Nz); vector< vector > yvec; for(int i=0; i &a = Bmap[i]; int xindex = (int)floor((a[0]-xmin+dx/2.0)/dx); // the +dx/2.0 guarantees against round-off errors int yindex = (int)(Ny<2 ? 0:floor((a[1]-ymin+dy/2.0)/dy)); int zindex = (int)floor((a[2]-zmin+dz/2.0)/dz); DBfieldPoint_t *b = &Btable[xindex][yindex][zindex]; b->x = a[0]; b->y = a[1]; b->z = a[2]; b->Bx = a[3]; b->By = a[4]; b->Bz = a[5]; } // Calculate gradient at every point in the map. // The derivatives are calculated wrt to the *index* of the // field map, not the physical unit. This is because // the fractions used in interpolation are fractions // between indices. for(int index_x=0; index_x0 ? 1:0); int index_x1 = index_x + (index_x0 ? 1:0); int index_z1 = index_z + (index_z1){ _DBG_<<"Field map appears to be 3 dimensional. Code is currently"<dBxdx = (Bx1->Bx - Bx0->Bx)/(double)(index_x1-index_x0); g->dBxdy = (By1->Bx - By0->Bx)/(double)(index_y1-index_y0); g->dBxdz = (Bz1->Bx - Bz0->Bx)/(double)(index_z1-index_z0); g->dBydx = (Bx1->By - Bx0->By)/(double)(index_x1-index_x0); g->dBydy = (By1->By - By0->By)/(double)(index_y1-index_y0); g->dBydz = (Bz1->By - Bz0->By)/(double)(index_z1-index_z0); g->dBzdx = (Bx1->Bz - Bx0->Bz)/(double)(index_x1-index_x0); g->dBzdy = (By1->Bz - By0->Bz)/(double)(index_y1-index_y0); g->dBzdz = (Bz1->Bz - Bz0->Bz)/(double)(index_z1-index_z0); } } } return Bmap.size(); } //------------- // GetFieldGradient //------------- void DMagneticFieldMapCalibDB::GetFieldGradient(double x, double y, double z, double &dBxdx, double &dBxdy, double &dBxdz, double &dBydx, double &dBydy, double &dBydz, double &dBzdx, double &dBzdy, double &dBzdz) const{ // Get closest indices for this point double r = sqrt(x*x + y*y); int index_x = (int)floor((r-xmin)/dx + 0.5); if(index_x<0 || index_x>=Nx)return; int index_z = (int)floor((z-zmin)/dz + 0.5); if(index_z<0 || index_z>=Nz)return; int index_y = 0; const DBfieldPoint_t *B = &Btable[index_x][index_y][index_z]; // Convert r back to x,y components double cos_theta = x/r; double sin_theta = y/r; if(r==0.0){ cos_theta=1.0; sin_theta=0.0; } // Rotate back into phi direction dBxdx = B->dBxdx*cos_theta*cos_theta; dBxdy = B->dBxdx*cos_theta*sin_theta; dBxdz = B->dBxdz; dBydx = B->dBydx*sin_theta*cos_theta; dBydy = B->dBydx*sin_theta*sin_theta; dBydz = B->dBydz; dBzdx = B->dBzdx*cos_theta; dBzdy = B->dBzdx*sin_theta; dBzdz = B->dBzdz; } //--------------------------------- // GetField //--------------------------------- void DMagneticFieldMapCalibDB::GetField(double x, double y, double z, double &Bx, double &By, double &Bz, int method) const { /// This calculates the magnetic field at an arbitrary point /// in space using the field map read from the calibaration /// database. It interpolates between grid points using the /// gradient values calculated in ReadMap (called from the /// constructor). Bx = By = Bz = 0.0; if(Ny>1){ _DBG_<<"Field map appears to be 3 dimensional. Code is currently"<=Nx)return; int index_z = (int)floor((z-zmin)/dz + 0.5); if(index_z<0 || index_z>=Nz)return; int index_y = 0; const DBfieldPoint_t *B = &Btable[index_x][index_y][index_z]; // Fractional distance between map points. double ur = (r - B->x)/dx; double uz = (z - B->z)/dz; // Use gradient to project grid point to requested position double Br; Br = B->Bx + B->dBxdx*ur + B->dBxdz*uz; Bz = B->Bz + B->dBzdx*ur + B->dBzdz*uz; // Convert r back to x,y components double cos_theta = x/r; double sin_theta = y/r; if(r==0.0){ cos_theta=1.0; sin_theta=0.0; } // Rotate back into phi direction Bx = Br*cos_theta; By = Br*sin_theta; }