// $Id$ // // File: DMagneticFieldMapFineMesh.cc #include #include using namespace std; #ifdef HAVE_EVIO #include #include using namespace evio; #endif #include "DMagneticFieldMapFineMesh.h" #include "JANA/JException.h" //--------------------------------- // DMagneticFieldMapFineMesh (Constructor) //--------------------------------- DMagneticFieldMapFineMesh::DMagneticFieldMapFineMesh(JApplication *japp, unsigned int runnumber, string namepath) { jcalib = japp->GetJCalibration(runnumber); jresman = japp->GetJResourceManager(runnumber); JParameterManager *jparms = japp->GetJParameterManager(); jparms->SetDefaultParameter("BFIELD_MAP", namepath); int Npoints = ReadMap(namepath, runnumber); if(Npoints==0){ _DBG_<<"Error getting JCalibration object for magnetic field!"<Quit(); } GetFineMeshMap(namepath,runnumber); } //--------------------------------- // DMagneticFieldMapFineMesh (Constructor) //--------------------------------- DMagneticFieldMapFineMesh::DMagneticFieldMapFineMesh(JCalibration *jcalib, string namepath,int runnumber) { this->jcalib = jcalib; GetFineMeshMap(namepath,runnumber); } //--------------------------------- // ~DMagneticFieldMapFineMesh (Destructor) //--------------------------------- DMagneticFieldMapFineMesh::~DMagneticFieldMapFineMesh() { } //--------------------------------- // ReadMap //--------------------------------- int DMagneticFieldMapFineMesh::ReadMap(string namepath, int runnumber, string context) { /// Read the magnetic field map in from the calibration database. /// This will read in the map and figure out the number of grid /// points in each direction (x,y, and z) and the range in each. /// The gradiant of the field is calculated for all but the most /// exterior points and saved to use in later calls to GetField(...). // Read in map from calibration database. This should really be // built into a run-dependent geometry framework, but for now // we do it this way. if(!jcalib){ jerr << "ERROR: jcalib pointer is NULL in DMagneticFieldMapFineMesh::ReadMap() !" << endl; return 0; } jout<<"Reading Magnetic field map from "< > Bmap; // Newer maps are stored as resources while older maps were stored // directly in the calib DB. Here, we hardwire in the names of the old // maps that are stored directly in the DB so we know whether to // get them via the resource manager or not. vector old_maps; old_maps.push_back("Magnets/Solenoid/solenoid_1500_poisson_20090814_01"); old_maps.push_back("Magnets/Solenoid/solenoid_1050_poisson_20091123_02"); old_maps.push_back("Magnets/Solenoid/solenoid_1500_poisson_20090814_01"); old_maps.push_back("Magnets/Solenoid/solenoid_1500_20090312-2"); old_maps.push_back("Magnets/Solenoid/solenoid_1600_20090312-2"); old_maps.push_back("Magnets/Solenoid/solenoid_1500kinked"); bool is_old_map = false; for(unsigned int i=0; iGet(namepath, Bmap); is_old_map = true; break; } } if(!is_old_map){ // This map is NOT stored directly in the calibration DB. // Try getting it as a JANA resource. jresman->Get(namepath, Bmap); } jout< 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-0.0)*2.54; // Removed 26" offset 6/22/2009 DL 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)/d_index_x; //g->dBxdy = (By1->Bx - By0->Bx)/(double)(index_y1-index_y0); g->dBxdz = (Bz1->Bx - Bz0->Bx)/d_index_z; g->dBydx = (Bx1->By - Bx0->By)/d_index_x; //g->dBydy = (By1->By - By0->By)/(double)(index_y1-index_y0); g->dBydz = (Bz1->By - Bz0->By)/d_index_z; g->dBzdx = (Bx1->Bz - Bx0->Bz)/d_index_x; //g->dBzdy = (By1->Bz - By0->Bz)/(double)(index_y1-index_y0); g->dBzdz = (Bz1->Bz - Bz0->Bz)/d_index_z; DBfieldPoint_t *B11 = &Btable[index_x1][index_y][index_z1]; DBfieldPoint_t *B01 = &Btable[index_x0][index_y][index_z1]; DBfieldPoint_t *B10 = &Btable[index_x1][index_y][index_z0]; DBfieldPoint_t *B00 = &Btable[index_x0][index_y][index_z0]; g->dBxdxdz=(B11->Bx - B01->Bx - B10->Bx + B00->Bx)/d_index_x/d_index_z; g->dBzdxdz=(B11->Bz - B01->Bz - B10->Bz + B00->Bz)/d_index_x/d_index_z; } } } return Bmap.size(); } // Use bicubic interpolation to find the field at the point (x,y). //See Numerical Recipes in C (2nd ed.), pp.125-127. void DMagneticFieldMapFineMesh::GetFieldBicubic(double x,double y,double z, double &Bx_,double &By_, double &Bz_) const{ // table of weight factors for bicubic interpolation static const int wt[16][16]= { {1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}, {0, 0, 0, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 0, 0}, {-3, 0, 0, 3, 0, 0, 0, 0,-2, 0, 0,-1, 0, 0, 0, 0}, {2, 0, 0,-2, 0, 0, 0, 0, 1, 0, 0, 1, 0, 0, 0, 0}, {0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}, {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 0, 0, 0}, {0, 0, 0, 0,-3, 0, 0, 3, 0, 0, 0, 0,-2, 0, 0,-1}, {0, 0, 0, 0, 2, 0, 0,-2, 0, 0, 0, 0, 1, 0, 0, 1}, {-3,3, 0, 0,-2,-1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}, {0, 0, 0, 0, 0, 0, 0, 0,-3, 3, 0, 0,-2,-1, 0, 0}, {9,-9, 9,-9, 6, 3,-3,-6, 6,-6,-3, 3, 4, 2, 1, 2}, {-6,6,-6, 6,-4,-2, 2, 4,-3, 3, 3,-3,-2,-1,-1,-2}, {2,-2, 0, 0, 1, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}, {0, 0, 0, 0, 0, 0, 0, 0, 2,-2, 0, 0, 1, 1, 0, 0}, {-6, 6,-6,6,-3,-3, 3, 3,-4, 4, 2,-2,-2,-2,-1,-1}, {4,-4, 4,-4, 2, 2,-2,-2, 2,-2,-2, 2, 1, 1, 1, 1}}; double temp[16]; double cl[16],coeff[4][4]; // radial distance and radial component of bfield double r = sqrt(x*x + y*y); double Br_=0.; // If the point (x,y,z) is outside the fine-mesh grid, interpolate // on the coarse grid //if (true){ if (z=zmaxFine || r>=rmaxFine){ // Get closest indices for this point int index_x = (int)floor((r-xmin)*one_over_dx + 0.5); //if(index_x<0 || index_x>=Nx)return; if (index_x>=Nx) return; else if (index_x<0) index_x=0; int index_z = (int)floor((z-zmin)*one_over_dz + 0.5); if(index_z<0 || index_z>=Nz)return; int index_y = 0; int i, j, k, m=0; int index_x1 = index_x + (index_xBx; temp[1]=B01->Bx; temp[2]=B11->Bx; temp[3]=B10->Bx; temp[8]=B00->dBxdx; temp[9]=B01->dBxdx; temp[10]=B11->dBxdx; temp[11]=B10->dBxdx; temp[4]=B00->dBxdz; temp[5]=B01->dBxdz; temp[6]=B11->dBxdz; temp[7]=B10->dBxdz; temp[12]=B00->dBxdxdz; temp[13]=B01->dBxdxdz; temp[14]=B11->dBxdxdz; temp[15]=B10->dBxdxdz; for (i=0;i<16;i++){ double tmp2=0.0; for (k=0;k<16;k++) tmp2+=wt[i][k]*temp[k]; cl[i]=tmp2; } for (i=0;i<4;i++) for (j=0;j<4;j++) coeff[i][j]=cl[m++]; double t=(z - B00->z)*one_over_dz; double u=(r - B00->x)*one_over_dx; for (i=3;i>=0;i--){ Br_=t*Br_+((coeff[i][3]*u+coeff[i][2])*u+coeff[i][1])*u+coeff[i][0]; } // Next compute the interpolation for Bz temp[0]=B00->Bz; temp[1]=B01->Bz; temp[2]=B11->Bz; temp[3]=B10->Bz; temp[8]=B00->dBzdx; temp[9]=B01->dBzdx; temp[10]=B11->dBzdx; temp[11]=B10->dBzdx; temp[4]=B00->dBzdz; temp[5]=B01->dBzdz; temp[6]=B11->dBzdz; temp[7]=B10->dBzdz; temp[12]=B00->dBzdxdz; temp[13]=B01->dBzdxdz; temp[14]=B11->dBzdxdz; temp[15]=B10->dBzdxdz; for (i=0;i<16;i++){ double tmp2=0.0; for (k=0;k<16;k++) tmp2+=wt[i][k]*temp[k]; cl[i]=tmp2; } m=0; for (i=0;i<4;i++) for (j=0;j<4;j++) coeff[i][j]=cl[m++]; Bz_=0.; for (i=3;i>=0;i--){ Bz_=t*Bz_+((coeff[i][3]*u+coeff[i][2])*u+coeff[i][1])*u+coeff[i][0]; } } else{ // otherwise do a simple lookup in the fine-mesh table unsigned int indr=(unsigned int)floor((r-rminFine)*rscale); unsigned int indz=(unsigned int)floor((z-zminFine)*zscale); Bz_=mBfine[indr][indz].Bz; Br_=mBfine[indr][indz].Br; // printf("Bz Br %f %f\n",Bz,Br); } // 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; } // Use bicubic interpolation to find the field and field gradient at the point // (x,y). See Numerical Recipes in C (2nd ed.), pp.125-127. void DMagneticFieldMapFineMesh::InterpolateField(double r,double z,double &Br, double &Bz,double &dBrdr, double &dBrdz,double &dBzdr, double &dBzdz) const{ // table of weight factors for bicubic interpolation static const int wt[16][16]= { {1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}, {0, 0, 0, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 0, 0}, {-3, 0, 0, 3, 0, 0, 0, 0,-2, 0, 0,-1, 0, 0, 0, 0}, {2, 0, 0,-2, 0, 0, 0, 0, 1, 0, 0, 1, 0, 0, 0, 0}, {0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}, {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 0, 0, 0}, {0, 0, 0, 0,-3, 0, 0, 3, 0, 0, 0, 0,-2, 0, 0,-1}, {0, 0, 0, 0, 2, 0, 0,-2, 0, 0, 0, 0, 1, 0, 0, 1}, {-3,3, 0, 0,-2,-1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}, {0, 0, 0, 0, 0, 0, 0, 0,-3, 3, 0, 0,-2,-1, 0, 0}, {9,-9, 9,-9, 6, 3,-3,-6, 6,-6,-3, 3, 4, 2, 1, 2}, {-6,6,-6, 6,-4,-2, 2, 4,-3, 3, 3,-3,-2,-1,-1,-2}, {2,-2, 0, 0, 1, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}, {0, 0, 0, 0, 0, 0, 0, 0, 2,-2, 0, 0, 1, 1, 0, 0}, {-6, 6,-6,6,-3,-3, 3, 3,-4, 4, 2,-2,-2,-2,-1,-1}, {4,-4, 4,-4, 2, 2,-2,-2, 2,-2,-2, 2, 1, 1, 1, 1}}; double temp[16]; double cl[16],coeff[4][4]; // Get closest indices for the point (r,z) int index_x = (int)floor((r-xmin)*one_over_dx + 0.5); if (index_x>=Nx) return; else if (index_x<0) index_x=0; int index_z = (int)floor((z-zmin)*one_over_dz + 0.5); if(index_z<0 || index_z>=Nz)return; int index_y = 0; int i, j, k, m=0; int index_x1 = index_x + (index_xBx; temp[1]=B01->Bx; temp[2]=B11->Bx; temp[3]=B10->Bx; temp[8]=B00->dBxdx; temp[9]=B01->dBxdx; temp[10]=B11->dBxdx; temp[11]=B10->dBxdx; temp[4]=B00->dBxdz; temp[5]=B01->dBxdz; temp[6]=B11->dBxdz; temp[7]=B10->dBxdz; temp[12]=B00->dBxdxdz; temp[13]=B01->dBxdxdz; temp[14]=B11->dBxdxdz; temp[15]=B10->dBxdxdz; for (i=0;i<16;i++){ double tmp2=0.0; for (k=0;k<16;k++) tmp2+=wt[i][k]*temp[k]; cl[i]=tmp2; } for (i=0;i<4;i++) for (j=0;j<4;j++) coeff[i][j]=cl[m++]; double t=(z - B00->z)*one_over_dz; double u=(r - B00->x)*one_over_dx; Br=dBrdr=dBrdz=0.; for (i=3;i>=0;i--){ double c3u=coeff[i][3]*u; Br=t*Br+((c3u+coeff[i][2])*u+coeff[i][1])*u+coeff[i][0]; dBrdr=t*dBrdr+(3.*c3u+2.*coeff[i][2])*u+coeff[i][1]; dBrdz=u*dBrdz+(3.*coeff[3][i]*t+2.*coeff[2][i])*t+coeff[1][i]; } dBrdr/=dx; dBrdz/=dz; // Next compute the interpolation for Bz temp[0]=B00->Bz; temp[1]=B01->Bz; temp[2]=B11->Bz; temp[3]=B10->Bz; temp[8]=B00->dBzdx; temp[9]=B01->dBzdx; temp[10]=B11->dBzdx; temp[11]=B10->dBzdx; temp[4]=B00->dBzdz; temp[5]=B01->dBzdz; temp[6]=B11->dBzdz; temp[7]=B10->dBzdz; temp[12]=B00->dBzdxdz; temp[13]=B01->dBzdxdz; temp[14]=B11->dBzdxdz; temp[15]=B10->dBzdxdz; for (i=0;i<16;i++){ double tmp2=0.0; for (k=0;k<16;k++) tmp2+=wt[i][k]*temp[k]; cl[i]=tmp2; } m=0; for (i=0;i<4;i++) for (j=0;j<4;j++) coeff[i][j]=cl[m++]; Bz=dBzdr=dBzdz=0.; for (i=3;i>=0;i--){ double c3u=coeff[i][3]*u; Bz=t*Bz+((c3u+coeff[i][2])*u+coeff[i][1])*u+coeff[i][0]; dBzdr=t*dBzdr+(3.*c3u+2.*coeff[i][2])*u+coeff[i][1]; dBzdz=u*dBzdz+(3.*coeff[3][i]*t+2.*coeff[2][i])*t+coeff[1][i]; } dBzdr/=dx; dBzdz/=dz; } // Find the field and field gradient at the point (x,y,z). void DMagneticFieldMapFineMesh::GetFieldAndGradient(double x,double y,double z, double &Bx_,double &By_, double &Bz_, double &dBxdx_, double &dBxdy_, double &dBxdz_, double &dBydx_, double &dBydy_, double &dBydz_, double &dBzdx_, double &dBzdy_, double &dBzdz_) const{ // radial distance double r = sqrt(x*x + y*y); if (r>xmax || z>zmax || z=zmaxFine || r>=rmaxFine){ //InterpolateField(r,z,Br_,Bz_,dBrdx_,dBrdz_,dBzdx_,dBzdz_); // Get closest indices for this point int index_x = static_cast(r*one_over_dx); int index_z = static_cast((z-zmin)*one_over_dz); int index_y = 0; if(index_x=0 && index_zx)*one_over_dx; double uz = (z - B->z)*one_over_dz; // Use gradient to project grid point to requested position Br_ = B->Bx+B->dBxdx*ur+B->dBxdz*uz; Bz_ = B->Bz+B->dBzdx*ur+B->dBzdz*uz; dBrdx_=B->dBxdx; dBrdz_=B->dBxdz; dBzdx_=B->dBzdx; dBzdz_=B->dBzdz; } } else{ // otherwise do a simple lookup in the fine-mesh table unsigned int indr=static_cast(r*rscale); unsigned int indz=static_cast((z-zminFine)*zscale); const DBfieldCylindrical_t *field=&mBfine[indr][indz]; Bz_=field->Bz; Br_=field->Br; dBrdx_=field->dBrdr; dBrdz_=field->dBrdz; dBzdz_=field->dBzdz; dBzdx_=field->dBzdr; // printf("Bz Br %f %f\n",Bz,Br); } // 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; dBxdx_ =dBrdx_*cos_theta*cos_theta; dBxdy_ =dBrdx_*cos_theta*sin_theta; dBxdz_ =dBrdz_*cos_theta; dBydx_=dBxdy_; dBydy_ = dBrdx_*sin_theta*sin_theta; dBydz_ = dBrdz_*sin_theta; dBzdx_ = dBzdx_*cos_theta; dBzdy_ = dBzdx_*sin_theta; /* printf("Grad %f %f %f %f %f %f %f %f %f\n",dBxdx_,dBxdy_,dBxdz_, dBydx_,dBydy_,dBydz_,dBzdx_,dBzdy_,dBzdz_); */ } //------------- // GetFieldGradient //------------- void DMagneticFieldMapFineMesh::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)*one_over_dx + 0.5); //if(index_x<0 || index_x>=Nx)return; if (index_x>=Nx) return; else if (index_x<0) index_x=0; int index_z = (int)floor((z-zmin)*one_over_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*one_over_dx; dBxdy = B->dBxdx*cos_theta*sin_theta*one_over_dx; dBxdz = B->dBxdz*cos_theta*one_over_dz; dBydx = B->dBxdx*sin_theta*cos_theta*one_over_dx; dBydy = B->dBxdx*sin_theta*sin_theta*one_over_dx; dBydz = B->dBxdz*sin_theta*one_over_dz; dBzdx = B->dBzdx*cos_theta*one_over_dx; dBzdy = B->dBzdx*sin_theta*one_over_dx; dBzdz = B->dBzdz*one_over_dz; /* printf("old Grad %f %f %f %f %f %f %f %f %f\n",dBxdx,dBxdy,dBxdz, dBydx,dBydy,dBydz,dBzdx,dBzdy,dBzdz); */ } //--------------------------------- // GetField //--------------------------------- void DMagneticFieldMapFineMesh::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; double Br=0.0; if(Ny>1){ _DBG_<<"Field map appears to be 3 dimensional. Code is currently"<xmax || z>zmax || z=zmaxFine || r>=rmaxFine){ // Get closest indices for this point int index_x = static_cast(r*one_over_dx); //if(index_x<0 || index_x>=Nx)return; if (index_x>=Nx) return; int index_z = static_cast((z-zmin)*one_over_dz); 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)*one_over_dx; double uz = (z - B->z)*one_over_dz; // Use gradient to project grid point to requested position Br = B->Bx+B->dBxdx*ur+B->dBxdz*uz; Bz = B->Bz+B->dBzdx*ur+B->dBzdz*uz; } else{ // otherwise do a simple lookup in the fine-mesh table unsigned int indr=static_cast(r*rscale); unsigned int indz=static_cast((z-zminFine)*zscale); const DBfieldCylindrical_t *field=&mBfine[indr][indz]; Bz=field->Bz; Br=field->Br; // printf("Bz Br %f %f\n",Bz,Br); } // Rotate back into phi direction Bx = Br*cos_theta; By = Br*sin_theta; } //--------------------------------- // GetField //--------------------------------- void DMagneticFieldMapFineMesh::GetField(const DVector3 &pos,DVector3 &Bout) 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). double Bz=0.,Br=0.0; if(Ny>1){ _DBG_<<"Field map appears to be 3 dimensional. Code is currently"<xmax || z>zmax || z=zmaxFine || r>=rmaxFine){ // Get closest indices for this point int index_x = static_cast(r*one_over_dx); //if(index_x<0 || index_x>=Nx)return; if (index_x>=Nx) return; int index_z = static_cast((z-zmin)*one_over_dz); 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)*one_over_dx; double uz = (z - B->z)*one_over_dz; // Use gradient to project grid point to requested position Br = B->Bx+B->dBxdx*ur+B->dBxdz*uz; Bz = B->Bz+B->dBzdx*ur+B->dBzdz*uz; } else{ // otherwise do a simple lookup in the fine-mesh table unsigned int indr=static_cast(r*rscale); unsigned int indz=static_cast((z-zminFine)*zscale); const DBfieldCylindrical_t *field=&mBfine[indr][indz]; Bz=field->Bz; Br=field->Br; // printf("Bz Br %f %f\n",Bz,Br); } // Rotate back into phi direction Bout.SetXYZ(Br*cos_theta,Br*sin_theta,Bz); } // Get the z-component of the magnetic field double DMagneticFieldMapFineMesh::GetBz(double x, double y, double z) const{ // radial position double r = sqrt(x*x + y*y); if (r>xmax || z>zmax || z=zmaxFine || r>=rmaxFine){ // Get closest indices for this point int index_x = static_cast(r*one_over_dx); //if(index_x<0 || index_x>=Nx)return 0.; if (index_x>=Nx) return 0.; int index_z = static_cast((z-zmin)*one_over_dz); if(index_z<0 || index_z>=Nz)return 0.; 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)*one_over_dx; double uz = (z - B->z)*one_over_dz; // Use gradient to project grid point to requested position return (B->Bz + B->dBzdx*ur + B->dBzdz*uz); } // otherwise do a simple lookup in the fine-mesh table unsigned int indr=static_cast(r*rscale); unsigned int indz=static_cast((z-zminFine)*zscale); return mBfine[indr][indz].Bz; } // Read a fine-mesh B-field map from an evio file void DMagneticFieldMapFineMesh::GetFineMeshMap(string namepath,int runnumber){ #ifdef HAVE_EVIO // The solenoid field map files are stored in CCDB as /Magnets/Solenoid/BFIELD_MAP_NAME // The fine-mesh files are now stored as /Magnets/Solenoid/finemeshes/BFIELD_MAP_NAME size_t ipos = namepath.rfind("/"); if(ipos == string::npos) throw JException("Could not parse field map: "+namepath); string finemesh_namepath = namepath.substr(0,ipos) + "/finemeshes" + namepath.substr(ipos); string evioFileName = ""; string evioFileNameToWrite = ""; // see if we can get the EVIO file as a resource try { evioFileName = jresman->GetResource(finemesh_namepath); } catch ( JException e ) { // if we can't get it as a resource, try to get it from a local file size_t ipos=namepath.find("/"); size_t ipos2=namepath.find("/",ipos+1); evioFileName = namepath.substr(ipos2+1)+".evio"; // see if the file exists struct stat stFileInfo; int intStat = stat(evioFileName.c_str(),&stFileInfo); if (intStat != 0) { evioFileNameToWrite = evioFileName; evioFileName = ""; } } if(evioFileName != "") { ReadEvioFile(evioFileName); } else{ #endif cout << "Fine-mesh evio file does not exist." <Br_; vectorBz_; vectordBrdr_; vectordBrdz_; vectordBzdr_; vectordBzdz_; for (unsigned int i=0;iopen(); while (chan->read()){ // create event tree from channel contents evioDOMTree tree(chan); // Loop over the nodes in the evio file evioDOMNodeListP fullList = tree.getNodeList(typeIs()); evioDOMNodeList::const_iterator iter; for(iter=fullList->begin(); iter!=fullList->end(); iter++) { const evioDOMNodeP np = *iter; const vector *vec = NULL; vec=np->getVector(); if (vec!=NULL){ if (np->tag==2){ rminFine=(*vec)[0]; rmaxFine=(*vec)[1]; drFine=(*vec)[2]; zminFine=(*vec)[3]; zmaxFine=(*vec)[4]; dzFine=(*vec)[5]; zscale=1./dzFine; rscale=1./drFine; NrFine=(unsigned int)floor((rmaxFine-rminFine)/drFine+0.5); NzFine=(unsigned int)floor((zmaxFine-zminFine)/dzFine+0.5); vector temp(NzFine); for (unsigned int m=0;mtag==3){// actual B-field data switch(np->num){ case 0: // Br for (unsigned int k=0;ksize();k++){ unsigned int indr=k/NzFine; unsigned int indz=k%NzFine; mBfine[indr][indz].Br=(*vec)[k]; } break; case 1: // Bz for (unsigned int k=0;ksize();k++){ unsigned int indr=k/NzFine; unsigned int indz=k%NzFine; mBfine[indr][indz].Bz=(*vec)[k]; } break; case 2: // dBrdr for (unsigned int k=0;ksize();k++){ unsigned int indr=k/NzFine; unsigned int indz=k%NzFine; mBfine[indr][indz].dBrdr=(*vec)[k]; } break; case 3: // dBrdz for (unsigned int k=0;ksize();k++){ unsigned int indr=k/NzFine; unsigned int indz=k%NzFine; mBfine[indr][indz].dBrdz=(*vec)[k]; } break; case 4: // dBzdr for (unsigned int k=0;ksize();k++){ unsigned int indr=k/NzFine; unsigned int indz=k%NzFine; mBfine[indr][indz].dBzdr=(*vec)[k]; } break; case 5: // dBzdz for (unsigned int k=0;ksize();k++){ unsigned int indr=k/NzFine; unsigned int indz=k%NzFine; mBfine[indr][indz].dBzdz=(*vec)[k]; } break; default: break; } } } } } chan->close(); delete chan; } #endif