//************************************************************************ // DTrackFitterKalman.cc //************************************************************************ #include "DTrackFitterKalman.h" #include "CDC/DCDCTrackHit.h" #include "HDGEOMETRY/DLorentzDeflections.h" #include "HDGEOMETRY/DMaterialMap.h" #include "HDGEOMETRY/DRootGeom.h" #include "DANA/DApplication.h" #include #include #include #include #include #define qBr2p 0.003 // conversion for converting q*B*r to GeV/c #define EPS 3.0e-8 #define EPS2 1.e-4 #define BEAM_RADIUS 0.1 #define MAX_ITER 25 #define STEP_SIZE 0.5 // 0.25 #define CDC_STEP_SIZE 0.4 // 0.25 #define CDC_FORWARD_STEP_SIZE 0.4 #define CDC_BACKWARD_STEP_SIZE 0.5 #define NUM_ITER 10 #define Z_MIN 15. #define Z_MAX 175. #define R_MAX 60.0 #ifndef SPEED_OF_LIGHT #define SPEED_OF_LIGHT 29.98 #endif #define CDC_DRIFT_SPEED 55e-4 #define VAR_S 0.09 #define Q_OVER_P_MAX 100. // 10 MeV/c #define PT_MIN 0.01 // 10 MeV/c #define MAX_PATH_LENGTH 500. #define TAN_MAX 10. #define CDC_GAS_DENSITY 0.001706 #define FDC_GAS_DENSITY 0.001899 #define ONE_THIRD 0.33333333333333333 #define ONE_SIXTH 0.16666666666666667 #define CHISQ_DIFF_CUT 20. #define MAX_DEDX 40. // Local boolean routines for sorting //bool static DKalmanHit_cmp(DKalmanHit_t *a, DKalmanHit_t *b){ // return a->zz; //} bool static DKalman_dedx_cmp(DKalman_dedx_t a, DKalman_dedx_t b){ double dEdx1=a.dE/a.ds; double dEdx2=b.dE/b.ds; return dEdx1zz; } bool static DKalmanCDCHit_cmp(DKalmanCDCHit_t *a, DKalmanCDCHit_t *b){ if (a==NULL || b==NULL){ cout << "Null pointer in CDC hit list??" << endl; return false; } if(b->ring == a->ring){ return b->straw < a->straw; } return (b->ring>a->ring); } int grkuta_(double *CHARGE, double *STEP, double *VECT, double *VOUT,const DMagneticFieldMap *bfield); // Variance for position along wire using PHENIX angle dependence, transverse // diffusion, and an intrinsic resolution of 127 microns. #define DIFFUSION_COEFF 1.1e-6 // cm^2/s --> 200 microns at 1 cm #define DRIFT_SPEED .0055 inline double fdc_y_variance(double alpha,double x){ double diffusion=2.*DIFFUSION_COEFF*fabs(x)/DRIFT_SPEED; return diffusion+0.00040+0.0064*tan(alpha)*tan(alpha); } DTrackFitterKalman::DTrackFitterKalman(JEventLoop *loop):DTrackFitter(loop){ hits.clear(); // Get the position of the CDC downstream endplate from DGeometry geom->GetCDCEndplate(endplate_z,endplate_dz,endplate_rmin,endplate_rmax); endplate_z-=endplate_dz; // Beginning of the cdc geom->Get("//posXYZ[@volume='CentralDC']/@X_Y_Z",cdc_origin); // Beginning of the FDC geom->Get("//posXYZ[@volume='ForwardDC']/@X_Y_Z",fdc_origin); vectorfdc_z1; geom->Get("//composition[@name='ForwardDC']/posXYZ[@volume='forwardDC']/@X_Y_Z", fdc_z1); fdc_origin[2]+=fdc_z1[2]; geom->Get("//posXYZ[@volume='forwardDC_package_1']/@X_Y_Z",fdc_z1); fdc_origin[2]+=fdc_z1[2]; geom->Get("//posXYZ[@volume='forwardDC_chamber_1']/@X_Y_Z/layer[@value='1']", fdc_z1); fdc_origin[2]+=fdc_z1[2]-1.; // Number degrees of freedom ndf=0; // Energy loss track_dedx=0.; num_dedx=0; // Mass hypothesis MASS=0.13957; //charged pion mass2=MASS*MASS; //DEBUG_HISTS=true; DEBUG_HISTS=false; DEBUG_LEVEL=0; //DEBUG_LEVEL=2; if(DEBUG_HISTS){ DApplication* dapp = dynamic_cast(loop->GetJApplication()); dapp->Lock(); cdc_residuals=(TH2F*)gROOT->FindObject("cdc_residuals"); if (!cdc_residuals){ cdc_residuals=new TH2F("cdc_residuals","residuals vs ring", 30,0.5,30.5,1000,-0.1,0.1); cdc_residuals->SetXTitle("ring number"); cdc_residuals->SetYTitle("#Deltad (cm)"); } fdc_xresiduals=(TH2F*)gROOT->FindObject("fdc_xresiduals"); if (!fdc_xresiduals){ fdc_xresiduals=new TH2F("fdc_xresiduals","x residuals vs z", 200,170.,370.,100,-1,1.); fdc_xresiduals->SetXTitle("z (cm)"); fdc_xresiduals->SetYTitle("#Deltax (cm)"); } fdc_yresiduals=(TH2F*)gROOT->FindObject("fdc_yresiduals"); if (!fdc_yresiduals){ fdc_yresiduals=new TH2F("fdc_yresiduals","y residuals vs z", 200,170.,370.,100,-1,1.); fdc_yresiduals->SetXTitle("z (cm)"); fdc_yresiduals->SetYTitle("#Deltay (cm)"); } thetay_vs_thetax=(TH2F*)gROOT->FindObject("thetay_vs_thetax"); if (!thetay_vs_thetax){ thetay_vs_thetax=new TH2F("thetay_vs_thetax","#thetay vs. #thetax", 360,-90.,90.,360,-90,90.); thetay_vs_thetax->SetXTitle("z (cm)"); thetay_vs_thetax->SetYTitle("#Deltay (cm)"); } dapp->Unlock(); } } //----------------- // ResetKalman //----------------- void DTrackFitterKalman::ResetKalman(void) { for (unsigned int i=0;iMASS=input_params.mass(); this->mass2=MASS*MASS; // Do fit error = KalmanLoop(); if (error!=NOERROR) return kFitFailed; // Copy fit results into DTrackFitter base-class data membe rs DVector3 mom,pos; GetPosition(pos); GetMomentum(mom); double charge = GetCharge(); fit_params.setMass(0.13957); // Assume pion to start fit_params.setPosition(pos); fit_params.setMomentum(mom); fit_params.setCharge(charge); fit_params.setdEdx(track_dedx); // Set the mass of the particle based on dEdx for low-momentum tracks if (track_dedx>0 && mom.Mag()<0.6){ double one_over_p=1./mom.Mag(); double dedx_proton=GetdEdx(one_over_p,0.93827); double dedx_pion=GetdEdx(one_over_p,0.13957); double dedx_kaon=GetdEdx(one_over_p,0.493677); double dedx_sigma=GetdEdxSigma(); double proton_ratio=(track_dedx/dedx_proton-1.)/dedx_sigma; double pion_ratio=(track_dedx/dedx_pion-1.)/dedx_sigma; double kaon_ratio=(track_dedx/dedx_kaon-1.)/dedx_sigma; if (pion_ratio>3.){ if (charge==+1. && proton_ratio>-3.) fit_params.setMass(0.93827); else if (fabs(kaon_ratio)<3) fit_params.setMass(0.493677); } } // Convert error matrix from internal representation to the type expected // by the DKinematicData class DMatrixDSym errMatrix(5); for (unsigned int i=0;i<5;i++){ for (unsigned int j=i;j<5;j++){ errMatrix(i,j)=cov[i][j]; } } fit_params.setTrackingErrorMatrix(errMatrix); this->chisq = GetChiSq(); this->Ndof = GetNDF(); fit_status = kFitSuccess; cdchits_used_in_fit = cdchits; // this should be changed to reflect hits dropped by the filter fdchits_used_in_fit = fdchits; // this should be changed to reflect hits dropped by the filter return fit_status; } //----------------- // ChiSq //----------------- double DTrackFitterKalman::ChiSq(fit_type_t fit_type, DReferenceTrajectory *rt, double *chisq_ptr, int *dof_ptr) { // This simply returns whatever was left in for the chisq/NDF from the last fit. // Using a DReferenceTrajectory is not really appropriate here so the base class' // requirement of it should be reviewed. double chisq = GetChiSq(); double ndf = GetNDF(); if(chisq_ptr)*chisq_ptr = chisq; if(dof_ptr)*dof_ptr = ndf; return chisq/ndf; } // Initialize the state vector jerror_t DTrackFitterKalman::SetSeed(double q,DVector3 pos, DVector3 mom){ if (!isfinite(pos.Mag()) || !isfinite(mom.Mag())){ _DBG_ << "Invalid seed data." <t=fdchit->time; hit->uwire=fdchit->w; hit->vstrip=fdchit->s; hit->z=fdchit->wire->origin.z(); hit->cosa=fdchit->wire->udir(1); hit->sina=fdchit->wire->udir(0); hit->nr=0.; hit->nz=0.; hit->covu=hit->covv=0.0004; hit->dE=fdchit->dE; my_fdchits.push_back(hit); return NOERROR; } /// Add a hit to the list of hits using Cartesian coordinates jerror_t DTrackFitterKalman::AddHit(double x,double y, double z,double covx, double covy, double covxy, double dE){ DKalmanHit_t *hit = new DKalmanHit_t; hit->x = x; hit->y = y; hit->z = z; hit->covx=covx; hit->covy=covy; hit->covxy=covxy; hit->dE=dE; hits.push_back(hit); return NOERROR; } // Add CDC hits jerror_t DTrackFitterKalman::AddCDCHit (const DCDCTrackHit *cdchit){ DKalmanCDCHit_t *hit= new DKalmanCDCHit_t; hit->origin=cdchit->wire->origin; hit->t=cdchit->tdrift; hit->d=cdchit->dist; hit->stereo=cdchit->wire->stereo; //hit->dir=(1./cdchit->wire->udir.z())*cdchit->wire->udir; hit->dir=cdchit->wire->udir; hit->ring=cdchit->wire->ring; hit->straw=cdchit->wire->straw; hit->status=0; hit->is_stereo=false; if (fabs(hit->stereo)>0.0) hit->is_stereo=true; hit->dE=cdchit->dE; my_cdchits.push_back(hit); return NOERROR; } jerror_t DTrackFitterKalman::AddVertex(DVector3 vertex){ DKalmanCDCHit_t *hit= new DKalmanCDCHit_t; hit->origin=vertex; hit->t=0.; hit->d=0.; hit->stereo=0.; hit->dir.SetXYZ(0.,0.,1.); my_cdchits.push_back(hit); return NOERROR; } // Calculate the derivative of the state vector with respect to z jerror_t DTrackFitterKalman::CalcDeriv(double z,double dz,DMatrix S, double dEdx, DMatrix &D){ double x=S(state_x,0), y=S(state_y,0),tx=S(state_tx,0),ty=S(state_ty,0); double q_over_p=S(state_q_over_p,0); //B-field at (x,y,z) double Bx=0.,By=0.,Bz=-2.; //bfield->GetField(x,y,z, Bx, By, Bz); bfield->GetFieldBicubic(x,y,z, Bx, By, Bz); // Don't let the magnitude of the momentum drop below some cutoff if (fabs(q_over_p)>Q_OVER_P_MAX) q_over_p=Q_OVER_P_MAX*(q_over_p>0?1.:-1.); // Try to keep the direction tangents from heading towards 90 degrees if (fabs(tx)>TAN_MAX) tx=TAN_MAX*(tx>0?1.:-1.); if (fabs(ty)>TAN_MAX) ty=TAN_MAX*(ty>0?1.:-1.); // useful combinations of terms double qpfactor=qBr2p*q_over_p; double factor=sqrt(1.+tx*tx+ty*ty); double qpfactor2=qpfactor*factor; double qpfactor_ds=0.5*dz*qpfactor2; // Derivative of S with respect to z D(state_x,0)=tx + qpfactor_ds*(ty*Bz+tx*ty*Bx-(1.+tx*tx)*By); D(state_y,0)=ty + qpfactor_ds*(Bx*(1.+ty*ty)-tx*ty*By-tx*Bz); D(state_tx,0)=qpfactor2*(tx*ty*Bx-(1.+tx*tx)*By+ty*Bz); D(state_ty,0)=qpfactor2*((1.+ty*ty)*Bx-tx*ty*By-tx*Bz); D(state_q_over_p,0)=0.; if (fabs(dEdx)>0. && fabs(q_over_p)GetField(x,y,z, Bx, By, Bz); bfield->GetFieldGradient(x,y,z,dBxdx,dBxdy,dBxdz,dBydx,dBydy,dBydz,dBzdx, dBzdy,dBzdz); */ bfield->GetFieldAndGradient(x,y,z,Bx,By,Bz,dBxdx,dBxdy,dBxdz,dBydx,dBydy, dBydz,dBzdx,dBzdy,dBzdz); // Don't let the magnitude of the momentum drop below some cutoff if (fabs(q_over_p)>Q_OVER_P_MAX) q_over_p=Q_OVER_P_MAX*(q_over_p>0?1.:-1.); // Try to keep the direction tangents from heading towards 90 degrees if (fabs(tx)>TAN_MAX) tx=TAN_MAX*(tx>0?1.:-1.); if (fabs(ty)>TAN_MAX) ty=TAN_MAX*(ty>0?1.:-1.); // useful combinations of terms double qpfactor=qBr2p*q_over_p; double tx2=tx*tx; double ty2=ty*ty; double txty=tx*ty; double one_plus_tx2=1.+tx2; double one_plus_ty2=1.+ty2; double factor=sqrt(1.+tx2+ty2); double qpfactor2=qpfactor*factor; double qpfactor_ds=0.5*dz*qpfactor2; // Derivative of S with respect to z D(state_x,0)=tx + qpfactor_ds*(ty*Bz+txty*Bx-one_plus_tx2*By); D(state_y,0)=ty + qpfactor_ds*(Bx*one_plus_ty2-txty*By-tx*Bz); D(state_tx,0)=qpfactor2*(txty*Bx-one_plus_tx2*By+ty*Bz); D(state_ty,0)=qpfactor2*(one_plus_ty2*Bx-txty*By-tx*Bz); // Jacobian J(state_x,state_tx)=J(state_y,state_ty)=1.; J(state_tx,state_q_over_p)=qBr2p*factor*(Bx*txty-By*one_plus_tx2+Bz*ty); J(state_ty,state_q_over_p)=qBr2p*factor*(Bx*one_plus_ty2-By*txty-Bz*tx); J(state_tx,state_tx)=qpfactor*(Bx*ty*(1.+2.*tx2+ty2) -By*tx*(3.+3.*tx2+2.*ty2) +Bz*txty)/factor; J(state_tx,state_x)=qpfactor2*(ty*dBzdx+txty*dBxdx -one_plus_tx2*dBydx); J(state_ty,state_ty)=qpfactor*(Bx*ty*(3.+2.*tx2+3.*ty2) -By*tx*one_plus_tx2+2.*ty2 -Bz*txty)/factor; J(state_ty,state_y)= qpfactor2*(one_plus_ty2*dBxdy -txty*dBydy-tx*dBzdy); J(state_tx,state_ty)=qpfactor*((Bx*tx+Bz)*(one_plus_tx2+2.*ty2) -By*ty*one_plus_tx2)/factor; J(state_tx,state_y)= qpfactor2*(tx*dBzdy+txty*dBxdy -one_plus_tx2*dBydy); J(state_ty,state_tx)=-qpfactor*((By*ty+Bz)*(1.+2.*tx2+ty2) -Bx*tx*one_plus_ty2)/factor; J(state_ty,state_x)=qpfactor2*(one_plus_ty2*dBxdx-txty*dBydx -tx*dBzdx); J(state_q_over_p,state_tx)=D(state_q_over_p,0)*tx/factor/factor; J(state_q_over_p,state_ty)=D(state_q_over_p,0)*ty/factor/factor; D(state_q_over_p,0)=0.; J(state_q_over_p,state_q_over_p)=0.; if (fabs(dEdx)>0.){ double p2=1./q_over_p/q_over_p; double E=sqrt(p2+mass2); D(state_q_over_p,0)=-q_over_p/p2*E*dEdx*factor; J(state_q_over_p,state_q_over_p)=-dEdx*factor/E *(2.+3.*mass2/p2); } return NOERROR; } // Reference trajectory for forward tracks in CDC region // At each point we store the state vector and the Jacobian needed to get to //this state along z from the previous state. jerror_t DTrackFitterKalman::SetCDCForwardReferenceTrajectory(DMatrix &S){ int i=0,forward_traj_cdc_length=forward_traj_cdc.size(); double z=z_; double r=0.; double r_outer_hit=R_MAX; if (my_fdchits.size()==0) my_cdchits[0]->origin.Perp(); // Continue adding to the trajectory until we have reached the endplate // or the maximum radius while(zr_outer_hit) do_energy_loss=false; if (PropagateForwardCDC(forward_traj_cdc_length,i,z,step_size,S)!=NOERROR) return UNRECOVERABLE_ERROR; z+=step_size; } r=sqrt(S(state_x,0)*S(state_x,0)+S(state_y,0)*S(state_y,0)); if (rr_outer_hit) do_energy_loss=false; if (PropagateForwardCDC(forward_traj_cdc_length,i,z,endplate_z-z,S) !=NOERROR) return UNRECOVERABLE_ERROR; z=endplate_z; } if (my_fdchits.size()>0){ // Propagate through endplate with smaller steps r=sqrt(S(state_x,0)*S(state_x,0)+S(state_y,0)*S(state_y,0)); while(z " << p <<" s: " << setprecision(3) << forward_traj_cdc[m].s <<" t: " << setprecision(3) << forward_traj_cdc[m].t << endl; } } // Current state vector S=*(forward_traj_cdc[0].S); // position at the end of the swim z_=forward_traj_cdc[0].pos.Z(); x_=forward_traj_cdc[0].pos.X(); y_=forward_traj_cdc[0].pos.Y(); return NOERROR; } // Reference trajectory for backward tracks in CDC region using the "forward // parameterization". // At each point we store the state vector and the Jacobian needed to get to //this state along z from the previous state. jerror_t DTrackFitterKalman::SetCDCBackwardReferenceTrajectory(DMatrix &S){ int i=0,forward_traj_cdc_length=forward_traj_cdc.size(); double z=z_; // Continue adding to the trajectory until we have reached the endplate while(z>cdc_origin[2]+CDC_BACKWARD_STEP_SIZE){ double step_size=-CDC_BACKWARD_STEP_SIZE; double r2=S(state_x,0)*S(state_x,0)+S(state_y,0)*S(state_y,0); if (r2<81.) step_size=-CDC_BACKWARD_STEP_SIZE/2.; if (PropagateForwardCDC(forward_traj_cdc_length,i,z,step_size,S)!=NOERROR) return UNRECOVERABLE_ERROR; z+=step_size; } if (PropagateForwardCDC(forward_traj_cdc_length,i,z,cdc_origin[2],S) !=NOERROR) return UNRECOVERABLE_ERROR; z=cdc_origin[2]; printf("i %d size %d\n",i,(int)forward_traj_cdc.size()); if (i<(int)forward_traj_cdc.size()){ forward_traj_cdc_length=forward_traj_cdc.size(); for (int j=0;joperator()(state_q_over_p,0))); } // Current state vector S=*(forward_traj_cdc[0].S); // position at the end of the swim z_=forward_traj_cdc[0].pos.Z(); x_=forward_traj_cdc[0].pos.X(); y_=forward_traj_cdc[0].pos.Y(); return NOERROR; } // Routine that extracts the state vector propagation part out of the reference // trajectory loop jerror_t DTrackFitterKalman::PropagateForwardCDC(int length,int &index,double z, double step,DMatrix &S){ DMatrix J(5,5),Q(5,5); DKalmanState_t temp; int my_i=0; double newz=z+step; // Initialize some variables temp.h_id=0; temp.num_hits=0; double dEdx=0.; double ds=0.; double beta2=1.,q_over_p=1.,q_over_p_sq=1.,varE=0.; // State at current position temp.pos.SetXYZ(S(state_x,0),S(state_y,0),z); temp.s=len; temp.t=ftime; temp.density=temp.A=temp.Z=temp.X0=0.; //initialize // get material properties from the Root Geometry if (do_energy_loss){ if (geom->FindMat(temp.pos,temp.density,temp.A,temp.Z,temp.X0) !=NOERROR){ _DBG_<< " q/p " << S(state_q_over_p,0) << endl; _DBG_<<"Material error!"<operator()(j,0)=S(j,0); } } else{ temp.S= new DMatrix(S); } // Get dEdx for the upcoming step if (temp.density>0. && do_energy_loss){ dEdx=GetdEdx(S(state_q_over_p,0),temp.Z,temp.A,temp.density); } // Step through field ds=Step(z,newz,dEdx,S); len+=fabs(ds); q_over_p=S(state_q_over_p,0); q_over_p_sq=q_over_p*q_over_p; ftime+=ds*sqrt(1.+mass2*q_over_p_sq)/SPEED_OF_LIGHT; // Get the contribution to the covariance matrix due to multiple // scattering if (do_multiple_scattering) GetProcessNoise(ds,newz,temp.X0,S,Q); // Energy loss straggling in the approximation of thick absorbers if (temp.density>0. /* && do_energy_loss */){ beta2=1./(1.+mass2*q_over_p_sq); varE=GetEnergyVariance(ds,q_over_p,temp.Z,temp.A,temp.density); Q(state_q_over_p,state_q_over_p) =varE*q_over_p_sq*q_over_p_sq/beta2; } // Compute the Jacobian matrix StepJacobian(newz,z,S,dEdx,J); // update the trajectory if (index<=length){ for (unsigned int j=0;j<5;j++){ for (unsigned int k=0;k<5;k++){ forward_traj_cdc[my_i].Q->operator()(j,k)=Q(j,k); forward_traj_cdc[my_i].J->operator()(j,k)=J(j,k); } } } else{ temp.Q= new DMatrix(Q); temp.J= new DMatrix(J); forward_traj_cdc.push_front(temp); } return NOERROR; } // Swim the state vector through the field from the start of the reference // trajectory to the end jerror_t DTrackFitterKalman::SwimCentral(DVector3 &pos,DMatrix &Sc){ double ds=CDC_STEP_SIZE; double r_outer_hit=my_cdchits[0]->origin.Perp(); central_traj[0].h_id=0; for (int m=central_traj.size()-1;m>0;m--){ double q_over_p=Sc(state_q_over_pt,0)*cos(atan(Sc(state_tanl,0))); double dedx=0.; // Clear out cdc hit id tags central_traj[m].h_id=0; // Turn off energy loss correction if the trajectory passes throught the // cdc end plate but there are no more measurements if (my_fdchits.size()==0 && (pos.z()>endplate_z || pos.z()0)?1.:-1.; double qpt=1./Sc(state_q_over_pt,0); double sinphi=sin(Sc(state_phi,0)); double cosphi=cos(Sc(state_phi,0)); // Magnetic field //DVector3 B(0.,0.,-2.); //bfield->GetField(pos.x(),pos.y(),pos.z(), B(0), B(1), B(2)); //bfield->GetFieldBicubic(pos.x(),pos.y(),pos.z(), B(0), B(1), B(2)); double Bz_=-2.; DVector3 dpos1=central_traj[m-1].pos-central_traj[m].pos; // Propagate the state through the field ds=central_traj[m-1].s-central_traj[m].s; FixedStep(pos,ds,Sc,dedx,Bz_); // update D double qrc_old=qpt/qBr2p/Bz_; double qrc_plus_D=D+qrc_old; double rc=sqrt(dpos1.Perp2() +2.*qrc_plus_D*(dpos1.x()*sinphi-dpos1.y()*cosphi) +qrc_plus_D*qrc_plus_D); Sc(state_D,0)=q*rc-qrc_old; } // Turn energy loss back on do_energy_loss=true; return NOERROR; } // Reference trajectory for central tracks // At each point we store the state vector and the Jacobian needed to get to this state // along s from the previous state. // The tricky part is that we swim out from the target to find Sc and pos along the trajectory // but we need the Jacobians for the opposite direction, because the filter proceeds from // the outer hits toward the target. jerror_t DTrackFitterKalman::SetCDCReferenceTrajectory(DVector3 pos,DMatrix &Sc){ DKalmanState_t temp; DMatrix J(5,5); // State vector Jacobian matrix DMatrix Q(5,5); // Process noise covariance matrix // Position, step, radius, etc. variables DVector3 oldpos; double dedx=0; double ds=CDC_STEP_SIZE; double beta2=1.,varE=0.,q_over_p=1.,q_over_p_sq=1.; len=0.; int i=0; double t=0.; // Coordinates for outermost cdc hit unsigned int id=my_cdchits.size()-1; DVector3 origin=my_cdchits[id]->origin; DVector3 dir=my_cdchits[id]->dir; double r_outer_hit=origin.Perp(); if (central_traj.size()>0){ // reuse existing deque // Reset D to zero Sc(state_D,0)=0.; for (int m=central_traj.size()-1;m>=0;m--){ i++; central_traj[m].s=len; central_traj[m].t=t; central_traj[m].pos=pos; central_traj[m].h_id=0; for (unsigned int j=0;j<5;j++){ central_traj[m].S->operator()(j,0)=Sc(j,0); } // Make sure D is zero central_traj[m].S->operator()(state_D,0)=0.; // Check if we are within start counter outer radius double r=pos.Perp(); if (r<9.0) ds=0.1; else ds=CDC_STEP_SIZE; // Check if we are outside the radius of the last measurement if (r>r_outer_hit || pos.z()Q_OVER_P_MAX) do_energy_loss=false; // Initialize energy loss dedx=0.; if (do_energy_loss){ // get material properties from the Root Geometry if(geom->FindMat(pos,central_traj[m].density,central_traj[m].A, central_traj[m].Z,central_traj[m].X0)!=NOERROR){ _DBG_ << "Material error! " << endl; break; } dedx=GetdEdx(q_over_p,central_traj[m].Z,central_traj[m].A, central_traj[m].density); } // Propagate the state through the field FixedStep(pos,ds,Sc,dedx); if (do_energy_loss && central_traj[m].density>0.){ // Energy loss straggling in the approximation of thick absorbers beta2=1./(1.+mass2*q_over_p_sq); varE=GetEnergyVariance(ds,q_over_p,central_traj[m].Z, central_traj[m].A,central_traj[m].density); Q(state_q_over_pt,state_q_over_pt) =varE*Sc(state_q_over_pt,0)*Sc(state_q_over_pt,0)/beta2 *q_over_p_sq; } // Multiple scattering if (do_multiple_scattering) GetProcessNoiseCentral(ds,pos,central_traj[m].X0,Sc,Q); // Compute the Jacobian matrix for back-tracking towards target StepJacobian(pos,origin,dir,-ds,Sc,dedx,J); // Fill the deque with the Jacobian and Process Noise matrices for (unsigned int k=0;k<5;k++){ for (unsigned int j=0;j<5;j++){ central_traj[m].J->operator()(k,j)=J(k,j); central_traj[m].Q->operator()(k,j)=Q(k,j); } } } } // reset energy loss flag do_energy_loss=true; // Swim out double r=pos.Perp(); while(rZ_MIN && lenr_outer_hit || pos.z()Q_OVER_P_MAX) do_energy_loss=false; // Initialize energy loss dedx=0.; if (do_energy_loss){ // get material properties from the Root Geometry if(geom->FindMat(pos,temp.density,temp.A,temp.Z,temp.X0)!=NOERROR){ _DBG_ << "Material error! " << endl; break; } dedx=GetdEdx(q_over_p,temp.Z,temp.A,temp.density); } // Propagate the state through the field FixedStep(pos,ds,Sc,dedx); // Multiple scattering if (do_multiple_scattering) GetProcessNoiseCentral(ds,pos,temp.X0,Sc,Q); // Energy loss straggling in the approximation of thick absorbers if (temp.density>0. && do_energy_loss){ beta2=1./(1.+mass2*q_over_p_sq); varE=GetEnergyVariance(ds,q_over_p,temp.Z,temp.A,temp.density); Q(state_q_over_pt,state_q_over_pt) =varE*Sc(state_q_over_pt,0)*Sc(state_q_over_pt,0)/beta2 *q_over_p_sq; } // Compute the Jacobian matrix StepJacobian(pos,origin,dir,-ds,Sc,dedx,J); // update the radius relative to the beam line r=pos.Perp(); // Update the trajectory info temp.Q= new DMatrix(Q); temp.J= new DMatrix(J); central_traj.push_front(temp); } // reset energy loss flag do_energy_loss=true; // return an error if there are still no entries in the trajectory if (central_traj.size()==0) return RESOURCE_UNAVAILABLE; if (DEBUG_LEVEL>1) { for (unsigned int m=0;m " << pt/cos(atan(tanl)) <<" s: " << setprecision(3) << central_traj[m].s <<" t: " << setprecision(3) << central_traj[m].t << endl; } } // State at end of swim Sc=*(central_traj[0].S); // Position at the end of the swim x_=pos.x(); y_=pos.y(); z_=pos.z(); return NOERROR; } // Reference trajectory for trajectories with hits in the forward direction // At each point we store the state vector and the Jacobian needed to get to this state // along z from the previous state. jerror_t DTrackFitterKalman::SetReferenceTrajectory(DMatrix &S){ DMatrix J(5,5),Q(5,5); DKalmanState_t temp; double ds=0.; // path length increment // Initialize some variables temp.h_id=0; temp.num_hits=0; double dEdx=0.; double beta2=1.,q_over_p=1.,q_over_p_sq=1.,varE=0.; // progress in z from hit to hit double z=z_; double newz=z; double r=0.; int i=0,my_i=0; int forward_traj_length=forward_traj.size(); for (unsigned int m=0;mz-z)/STEP_SIZE); newz=my_fdchits[m]->z-STEP_SIZE*double(num); if (newz-z>0.){ temp.s=len; temp.t=ftime; temp.h_id=0; temp.pos.SetXYZ(S(state_x,0),S(state_y,0),z); temp.density=temp.A=temp.Z=temp.X0=0.; //initialize // get material properties from the Root Geometry if (do_energy_loss){ if (geom->FindMat(temp.pos,temp.density,temp.A,temp.Z,temp.X0) !=NOERROR){ _DBG_<<"Material error!"<operator()(j,0)=S(j,0); } } else{ temp.S=new DMatrix(S); } // Get dEdx for the upcoming step r=temp.pos.Perp(); if (do_energy_loss){ dEdx=GetdEdx(S(state_q_over_p,0),temp.Z,temp.A,temp.density); } // Step through field ds=Step(z,newz,dEdx,S); len+=ds; q_over_p=S(state_q_over_p,0); q_over_p_sq=q_over_p*q_over_p; ftime+=ds*sqrt(1.+mass2*q_over_p_sq)/SPEED_OF_LIGHT; // Get the contribution to the covariance matrix due to multiple // scattering if (do_multiple_scattering) GetProcessNoise(ds,newz,temp.X0,S,Q); // Energy loss straggling in the approximation of thick absorbers if (temp.density>0. && do_energy_loss){ beta2=1./(1.+mass2*q_over_p_sq); varE=GetEnergyVariance(ds,q_over_p,temp.Z,temp.A,temp.density); Q(state_q_over_p,state_q_over_p) =varE*q_over_p_sq*q_over_p_sq/beta2; } // Compute the Jacobian matrix StepJacobian(newz,z,S,dEdx,J); // update the trajectory data if (i<=forward_traj_length){ for (unsigned int j=0;j<5;j++){ for (unsigned int k=0;k<5;k++){ forward_traj[my_i].Q->operator()(j,k)=Q(j,k); forward_traj[my_i].J->operator()(j,k)=J(j,k); } } } else{ temp.Q=new DMatrix(Q); temp.J=new DMatrix(J); forward_traj.push_front(temp); } // update z z=newz; } for (int k=0;kFindMat(temp.pos,temp.density,temp.A,temp.Z,temp.X0) !=NOERROR){ _DBG_<<"Material error!"<operator()(j,0)=S(j,0); } } else{ temp.S=new DMatrix(S); } // Get dEdx for the upcoming step if (do_energy_loss //&& (pass==kTimeBased || r<9.0) ){ dEdx=GetdEdx(S(state_q_over_p,0),temp.Z,temp.A,temp.density); } // Step through field ds=Step(z,newz,dEdx,S); len+=ds; q_over_p=S(state_q_over_p,0); q_over_p_sq=q_over_p*q_over_p; ftime+=ds*sqrt(1.+mass2*q_over_p_sq)/SPEED_OF_LIGHT; // Get the contribution to the covariance matrix due to multiple // scattering if (do_multiple_scattering) GetProcessNoise(ds,newz,temp.X0,S,Q); // Energy loss straggling in the approximation of thick absorbers if (temp.density>0. && do_energy_loss){ beta2=1./(1.+mass2*q_over_p_sq); varE=GetEnergyVariance(ds,q_over_p,temp.Z,temp.A,temp.density); Q(state_q_over_p,state_q_over_p) =varE*q_over_p_sq*q_over_p_sq/beta2; } // Compute the Jacobian matrix StepJacobian(newz,z,S,dEdx,J); // update the trajectory data if (i<=forward_traj_length){ for (unsigned int j=0;j<5;j++){ for (unsigned int k=0;k<5;k++){ forward_traj[my_i].Q->operator()(j,k)=Q(j,k); forward_traj[my_i].J->operator()(j,k)=J(j,k); } } } else{ temp.Q=new DMatrix(Q); temp.J=new DMatrix(J); forward_traj.push_front(temp); } temp.h_id=0; //update z z=newz; } // Lorentz correction slope parameters double tanr=0.,tanz=0.; lorentz_def->GetLorentzCorrectionParameters(S(state_x,0),S(state_y,0),z, tanz,tanr); my_fdchits[m]->nr=tanr; my_fdchits[m]->nz=tanz; temp.h_id=m+1; } // Final point //i++; //my_i=forward_traj_length-i; temp.s=len; temp.t=ftime; temp.pos.SetXYZ(S(state_x,0),S(state_y,0),newz); temp.density=temp.A=temp.Z=temp.X0=0.; //initialize // get material properties from the Root Geometry if(do_energy_loss && geom->FindMat(temp.pos,temp.density,temp.A,temp.Z,temp.X0) ==NOERROR){ i++; my_i=forward_traj_length-i; // Get dEdx for the upcoming step r=temp.pos.Perp(); dEdx=GetdEdx(S(state_q_over_p,0),temp.Z,temp.A,temp.density); // Store next-to-final point if (i<=forward_traj_length){ forward_traj[my_i].s=temp.s; forward_traj[my_i].t=temp.t; forward_traj[my_i].h_id=temp.h_id; forward_traj[my_i].pos=temp.pos; forward_traj[my_i].A=temp.A; forward_traj[my_i].Z=temp.Z; forward_traj[my_i].density=temp.density; forward_traj[my_i].X0=temp.X0; for (unsigned int j=0;j<5;j++){ forward_traj[my_i].S->operator()(j,0)=S(j,0); } } else{ temp.S=new DMatrix(S); } // One more step to last hit point newz=z+STEP_SIZE; ds=Step(z,newz,dEdx,S); len+=ds; q_over_p=S(state_q_over_p,0); q_over_p_sq=q_over_p*q_over_p; ftime+=ds*sqrt(1.+mass2*q_over_p_sq)/SPEED_OF_LIGHT; // Get the contribution to the covariance matrix due to multiple // scattering if (do_multiple_scattering) GetProcessNoise(ds,newz,temp.X0,S,Q); // Energy loss straggling in the approximation of thick absorbers if (temp.density>0. && do_energy_loss){ beta2=1./(1.+mass2*q_over_p_sq); varE=GetEnergyVariance(ds,q_over_p,temp.Z,temp.A,temp.density); Q(state_q_over_p,state_q_over_p) =varE*q_over_p_sq*q_over_p_sq/beta2; } // Compute the Jacobian matrix StepJacobian(newz,z,S,dEdx,J); // update the trajectory data if (i<=forward_traj_length){ for (unsigned int j=0;j<5;j++){ for (unsigned int k=0;k<5;k++){ forward_traj[my_i].Q->operator()(j,k)=Q(j,k); forward_traj[my_i].J->operator()(j,k)=J(j,k); } } } else{ temp.Q=new DMatrix(Q); temp.J=new DMatrix(J); forward_traj.push_front(temp); } // Make sure the ref trajectory goes one step beyond the most downstream //hit //i++; //my_i=forward_traj_length-i; temp.s=len; temp.t=ftime; temp.pos.SetXYZ(S(state_x,0),S(state_y,0),newz); temp.h_id=0; temp.density=temp.A=temp.Z=temp.X0=0.; //initialize // get material properties from the Root Geometry if(do_energy_loss && geom->FindMat(temp.pos,temp.density,temp.A,temp.Z,temp.X0) ==NOERROR){ i++; my_i=forward_traj_length-i; // Get dEdx for the upcoming step r=temp.pos.Perp(); dEdx=GetdEdx(S(state_q_over_p,0),temp.Z,temp.A,temp.density); // Store final point if (i<=forward_traj_length){ forward_traj[my_i].s=temp.s; forward_traj[my_i].t=temp.t; forward_traj[my_i].h_id=temp.h_id; forward_traj[my_i].pos=temp.pos; forward_traj[my_i].A=temp.A; forward_traj[my_i].Z=temp.Z; forward_traj[my_i].density=temp.density; forward_traj[my_i].X0=temp.X0; for (unsigned int j=0;j<5;j++){ forward_traj[my_i].S->operator()(j,0)=S(j,0); } } else{ temp.S=new DMatrix(S); } // One more step after last hit point newz=z+STEP_SIZE; ds=Step(z,newz,dEdx,S); //len+=ds; // Get the contribution to the covariance matrix due to multiple // scattering if (do_multiple_scattering) GetProcessNoise(ds,newz,temp.X0,S,Q); // Energy loss straggling in the approximation of thick absorbers if (temp.density>0. && do_energy_loss){ q_over_p=S(state_q_over_p,0); q_over_p_sq=q_over_p*q_over_p; beta2=1./(1.+mass2*q_over_p_sq); varE=GetEnergyVariance(ds,q_over_p,temp.Z,temp.A,temp.density); Q(state_q_over_p,state_q_over_p) =varE*q_over_p_sq*q_over_p_sq/beta2; } // Compute the Jacobian matrix StepJacobian(newz,z,S,dEdx,J); // update the trajectory data if (i<=forward_traj_length){ for (unsigned int j=0;j<5;j++){ for (unsigned int k=0;k<5;k++){ forward_traj[my_i].Q->operator()(j,k)=Q(j,k); forward_traj[my_i].J->operator()(j,k)=J(j,k); } } } else{ temp.Q=new DMatrix(Q); temp.J=new DMatrix(J); forward_traj.push_front(temp); } } } else{ _DBG_ << "Material error!" < " << p <<" s: " << setprecision(3) << forward_traj[m].s <<" t: " << setprecision(3) << forward_traj[m].t << endl; } } // position at the end of the swim z_=z; x_=S(state_x,0); y_=S(state_y,0); return NOERROR; } // Step the state vector through the field from oldz to newz. // Uses the 4th-order Runga-Kutte algorithm. double DTrackFitterKalman::Step(double oldz,double newz, double dEdx,DMatrix &S){ double delta_z=newz-oldz; DMatrix D1(5,1),D2(5,1),D3(5,1),D4(5,1); CalcDeriv(oldz,delta_z,S,dEdx,D1); CalcDeriv(oldz+delta_z/2.,delta_z/2.,S+0.5*delta_z*D1,dEdx,D2); CalcDeriv(oldz+delta_z/2.,delta_z/2.,S+0.5*delta_z*D2,dEdx,D3); CalcDeriv(oldz+delta_z,delta_z,S+delta_z*D3,dEdx,D4); S+=delta_z*(ONE_SIXTH*D1+ONE_THIRD*D2+ONE_THIRD*D3+ONE_SIXTH*D4); // Don't let the magnitude of the momentum drop below some cutoff if (fabs(S(state_q_over_p,0))>Q_OVER_P_MAX) S(state_q_over_p,0)=Q_OVER_P_MAX*(S(state_q_over_p,0)>0?1.:-1.); // Try to keep the direction tangents from heading towards 90 degrees if (fabs(S(state_tx,0))>TAN_MAX) S(state_tx,0)=TAN_MAX*(S(state_tx,0)>0?1.:-1.); if (fabs(S(state_ty,0))>TAN_MAX) S(state_ty,0)=TAN_MAX*(S(state_ty,0)>0?1.:-1.); double s=sqrt(1.+S(state_tx,0)*S(state_tx,0)+S(state_ty,0)*S(state_ty,0)) *delta_z; return s; } // Step the state vector through the magnetic field and compute the Jacobian // matrix. Uses the 4th-order Runga-Kutte algorithm. jerror_t DTrackFitterKalman::StepJacobian(double oldz,double newz,DMatrix S, double dEdx,DMatrix &J){ // Initialize the Jacobian matrix J.Zero(); for (int i=0;i<5;i++) J(i,i)=1.; // Matrices for intermediate steps DMatrix J1(5,5),J2(5,5),J3(5,5),J4(5,5); DMatrix D1(5,1),D2(5,1),D3(5,1),D4(5,1); double delta_z=newz-oldz; CalcDerivAndJacobian(oldz,delta_z,S,dEdx,J1,D1); CalcDerivAndJacobian(oldz+delta_z/2.,delta_z/2.,S+0.5*delta_z*D1,dEdx,J2,D2); J2=J2+0.5*(J2*J1); CalcDerivAndJacobian(oldz+delta_z/2.,delta_z/2.,S+0.5*delta_z*D2,dEdx,J3,D3); J3=J3+0.5*(J3*J2); CalcDerivAndJacobian(oldz+delta_z,delta_z,S+delta_z*D3,dEdx,J4,D4); J4=J4+J4*J3; J+=delta_z*(ONE_SIXTH*J1+ONE_THIRD*J2+ONE_THIRD*J3+ONE_SIXTH*J4); return NOERROR; } // Calculate the derivative for the alternate set of parameters {q/pT, phi, // tan(lambda),D,z} jerror_t DTrackFitterKalman::CalcDeriv(double ds,DVector3 pos,DVector3 &dpos, DVector3 B,DMatrix S,double dEdx, DMatrix &D1){ //Direction at current point double tanl=S(state_tanl,0); // Don't let tanl exceed some maximum if (fabs(tanl)>TAN_MAX) tanl=TAN_MAX*(tanl>0?1.:-1.); double phi=S(state_phi,0); double cosphi=cos(phi); double sinphi=sin(phi); double lambda=atan(tanl); double sinl=sin(lambda); double cosl=cos(lambda); // Other parameters double q_over_pt=S(state_q_over_pt,0); double pt=fabs(1./q_over_pt); // Don't let the pt drop below some minimum if (pt0?1.:-1.); } double qpfactor=qBr2p*q_over_pt; // Derivative of S with respect to s double temp1=B.y()*cosphi-B.x()*sinphi; D1(state_q_over_pt,0) =qpfactor*q_over_pt*sinl*temp1; if (fabs(dEdx)>0){ double p=pt/cosl; double E=sqrt(p*p+mass2); if (1./pTAN_MAX) tanl=TAN_MAX*(tanl>0?1.:-1.); double phi=S(state_phi,0); double cosphi=cos(phi); double sinphi=sin(phi); double lambda=atan(tanl); double sinl=sin(lambda); double cosl=cos(lambda); // Other parameters double q_over_pt=S(state_q_over_pt,0); double pt=fabs(1./q_over_pt); double q=pt*q_over_pt; // Don't let the pt drop below some minimum if (ptGetFieldGradient(pos.x(),pos.y(),pos.z(),dBxdx,dBxdy,dBxdz,dBydx, dBydy,dBydz,dBzdx,dBzdy,dBzdz); */ bfield->GetFieldAndGradient(pos.x(),pos.y(),pos.z(),B(0),B(1),B(2), dBxdx,dBxdy,dBxdz,dBydx, dBydy,dBydz,dBzdx,dBzdy,dBzdz); // Derivative of S with respect to s double temp1=B.y()*cosphi-B.x()*sinphi; D1(state_q_over_pt,0)=qpfactor*q_over_pt*sinl*temp1; D1(state_phi,0)=qpfactor*(B.x()*cosphi*sinl+B.y()*sinphi*sinl-B.z()*cosl); D1(state_tanl,0)=qpfactor*temp1/cosl; double factor=qpfactor*ds/2.*cosl; //D1(state_z,0)=sinl+q_over_pt*qBr2p*ds/2.*cosl*cosl*temp1; D1(state_z,0)=sinl+factor*cosl*temp1; // New direction /* dpos.SetXYZ(cosl*cosphi +q_over_pt*cosl*qBr2p*ds/2.*(B.z()*cosl*sinphi-B.y()*sinl), cosl*sinphi +q_over_pt*cosl*qBr2p*ds/2.*(B.x()*sinl-B.z()*cosl*cosphi), D1(state_z,0)); */ dpos.SetXYZ(cosl*cosphi+factor*(B.z()*cosl*sinphi-B.y()*sinl), cosl*sinphi+factor*(B.x()*sinl-B.z()*cosl*cosphi), D1(state_z,0)); // Jacobian matrix elements J1(state_phi,state_phi)=qpfactor*sinl*(B.y()*cosphi-B.x()*sinphi); J1(state_phi,state_q_over_pt)=qBr2p*(B.x()*cosphi*sinl+B.y()*sinphi*sinl -B.z()*cosl); J1(state_phi,state_tanl)=qpfactor*(B.x()*cosphi*cosl+B.y()*sinphi*cosl +B.z()*sinl)/(1.+tanl*tanl); J1(state_phi,state_z) =qpfactor*(dBxdz*cosphi*sinl+dBydz*sinphi*sinl-dBzdz*cosl); J1(state_tanl,state_phi)=-qpfactor*(B.y()*sinphi+B.x()*cosphi)/cosl; J1(state_tanl,state_q_over_pt)=qBr2p*(B.y()*cosphi-B.x()*sinphi)/cosl; J1(state_tanl,state_tanl)=qpfactor*sinl*(B.y()*cosphi-B.x()*sinphi); J1(state_tanl,state_z)=qpfactor*(dBydz*cosphi-dBxdz*sinphi)/cosl; J1(state_q_over_pt,state_phi) =-qpfactor*q_over_pt*sinl*(B.y()*sinphi+B.x()*cosphi); double temp=sqrt(1.+cosl*cosl*q_over_pt*q_over_pt*mass2); J1(state_q_over_pt,state_q_over_pt) =2.*qpfactor*sinl*(B.y()*cosphi-B.x()*sinphi); if (fabs(dEdx)>0){ double p=pt/cosl; double E=sqrt(p*p+mass2); D1(state_q_over_pt,0)+=-q_over_pt*E/p/p*dEdx; J1(state_q_over_pt,state_q_over_pt)+=-q*dEdx*cosl*q_over_pt*(2.+3.*cosl*cosl*q_over_pt*q_over_pt*mass2) /temp; } J1(state_q_over_pt,state_tanl) =qpfactor*q_over_pt*cosl*cosl*cosl*(B.y()*cosphi-B.x()*sinphi) +q*dEdx*sinl*cosl*cosl/temp*q_over_pt*q_over_pt *(1.+2.*cosl*cosl*mass2*q_over_pt*q_over_pt); J1(state_q_over_pt,state_z) =qpfactor*q_over_pt*sinl*(dBydz*cosphi-dBxdz*sinphi); J1(state_z,state_tanl)=cosl/(1.+tanl*tanl); return NOERROR; } // Convert between the forward parameter set {x,y,tx,ty,q/p} and the central // parameter set {q/pT,phi,tan(lambda),D,z} jerror_t DTrackFitterKalman::ConvertStateVector(double z,double wire_x, double wire_y, DMatrix S, DMatrix C, DMatrix &Sc, DMatrix &Cc){ double x=S(state_x,0),y=S(state_y,0); double tx=S(state_tx,0),ty=S(state_ty,0),q_over_p=S(state_q_over_p,0); double tsquare=tx*tx+ty*ty; double factor=1./sqrt(1.+tsquare); double tanl=1./sqrt(tsquare); double cosl=cos(atan(tanl)); Sc(state_q_over_pt,0)=q_over_p/cosl; Sc(state_phi,0)=atan2(ty,tx); Sc(state_tanl,0)=tanl; // Sc(state_D,0)=sqrt((x-wire_x)*(x-wire_x)+(y-wire_y)*(y-wire_y)); Sc(state_D,0)=sqrt(x*x+y*y); Sc(state_z,0)=z; DMatrix J(5,5); double tanl3=tanl*tanl*tanl; J(state_tanl,state_tx)=-tx*tanl3; J(state_tanl,state_ty)=-ty*tanl3; J(state_z,state_x)=1./tx; J(state_z,state_y)=1./ty; J(state_q_over_pt,state_q_over_p)=1./cosl; J(state_q_over_pt,state_tx)=-tx*q_over_p*tanl3*factor; J(state_q_over_pt,state_ty)=-ty*q_over_p*tanl3*factor; J(state_phi,state_tx)=-ty/tsquare; J(state_phi,state_ty)=tx/tsquare; //J(state_D,state_x)=(x-wire_x)/S(state_D,0); //J(state_D,state_y)=(y-wire_y)/S(state_D,0); J(state_D,state_x)=x/Sc(state_D,0); J(state_D,state_y)=y/Sc(state_D,0); Cc=J*(C*DMatrix(DMatrix::kTransposed,J)); return NOERROR; } // Convert between the forward parameter set {x,y,tx,ty,q/p} and the central // parameter set {q/pT,phi,tan(lambda),D,z} jerror_t DTrackFitterKalman::ConvertStateVector(double z,double wire_x, double wire_y, DMatrix S,DMatrix &Sc){ double x=S(state_x,0),y=S(state_y,0); double tx=S(state_tx,0),ty=S(state_ty,0),q_over_p=S(state_q_over_p,0); double tsquare=tx*tx+ty*ty; // double factor=1./sqrt(1.+tsquare); double tanl=1./sqrt(tsquare); double cosl=cos(atan(tanl)); Sc(state_q_over_pt,0)=q_over_p/cosl; Sc(state_phi,0)=atan2(ty,tx); Sc(state_tanl,0)=tanl; // Sc(state_D,0)=sqrt((x-wire_x)*(x-wire_x)+(y-wire_y)*(y-wire_y)); Sc(state_D,0)=sqrt(x*x+y*y); Sc(state_z,0)=z; return NOERROR; } // Runga-Kutte for alternate parameter set {q/pT,phi,tanl(lambda),D,z} jerror_t DTrackFitterKalman::FixedStep(DVector3 &pos,double ds,DMatrix &S, double dEdx){ double Bz=0.; FixedStep(pos,ds,S,dEdx,Bz); return NOERROR; } // Runga-Kutte for alternate parameter set {q/pT,phi,tanl(lambda),D,z} jerror_t DTrackFitterKalman::FixedStep(DVector3 &pos,double ds,DMatrix &S, double dEdx,double &Bz){ // Matrices for intermediate steps DMatrix D1(5,1),D2(5,1),D3(5,1),D4(5,1); DMatrix S1(5,1),S2(5,1),S3(5,1),S4(5,1); DVector3 dpos1,dpos2,dpos3,dpos4; // Magnetic field DVector3 B(0.,0.,-2.); bfield->GetFieldBicubic(pos.x(),pos.y(),pos.z(), B(0), B(1), B(2)); Bz=fabs(B(2)); CalcDeriv(0.,pos,dpos1,B,S,dEdx,D1); DVector3 mypos=pos+(ds/2.)*dpos1; bfield->GetFieldBicubic(mypos.x(),mypos.y(),mypos.z(), B(0), B(1), B(2)); S1=S+(0.5*ds)*D1; CalcDeriv(ds/2.,mypos,dpos2,B,S1,dEdx,D2); mypos=pos+(ds/2.)*dpos2; bfield->GetFieldBicubic(mypos.x(),mypos.y(),mypos.z(), B(0), B(1), B(2)); S2=S+(0.5*ds)*D2; CalcDeriv(ds/2.,mypos,dpos3,B,S2,dEdx,D3); mypos=pos+ds*dpos3; bfield->GetFieldBicubic(mypos.x(),mypos.y(),mypos.z(), B(0), B(1), B(2)); S3=S+ds*D3; CalcDeriv(ds,mypos,dpos4,B,S3,dEdx,D4); // New state vector S+=ds*(ONE_SIXTH*D1+ONE_THIRD*D2+ONE_THIRD*D3+ONE_SIXTH*D4); // Don't let the pt drop below some minimum if (fabs(1./S(state_q_over_pt,0))0?1.:-1.); } // Don't let tanl exceed some maximum if (fabs(S(state_tanl,0))>TAN_MAX){ S(state_tanl,0)=TAN_MAX*(S(state_tanl,0)>0?1.:-1.); } // New position pos+= (dpos1=ds*(ONE_SIXTH*dpos1+ONE_THIRD*dpos2+ONE_THIRD*dpos3+ONE_SIXTH*dpos4)); return NOERROR; } // Runga-Kutte for alternate parameter set {q/pT,phi,tanl(lambda),D,z} jerror_t DTrackFitterKalman::StepJacobian(DVector3 pos, DVector3 wire_orig, DVector3 wiredir, double ds,DMatrix S, double dEdx,DMatrix &J){ // Initialize the Jacobian matrix J.Zero(); for (int i=0;i<5;i++) J(i,i)=1.; // Matrices for intermediate steps DMatrix J1(5,5),J2(5,5),J3(5,5),J4(5,5); DMatrix D1(5,1),D2(5,1),D3(5,1),D4(5,1); DMatrix S1(5,1),S2(5,1),S3(5,1),S4(5,1); DVector3 dpos1,dpos2,dpos3,dpos4; // charge double q=(S(state_q_over_pt,0)>0)?1.:-1.; // Magnetic field DVector3 B(0.,0.,-2.); //bfield->GetField(pos.x(),pos.y(),pos.z(), B(0), B(1), B(2)); //bfield->GetFieldBicubic(pos.x(),pos.y(),pos.z(), B(0), B(1), B(2)); double qpt=1./S(state_q_over_pt,0); double sinphi=sin(S(state_phi,0)); double cosphi=cos(S(state_phi,0)); double D=S(state_D,0); CalcDerivAndJacobian(0.,pos,dpos1,B,S,dEdx,J1,D1); double Bz_=fabs(B.z()); // needed for computing D DVector3 mypos=pos+(ds/2.)*dpos1; //bfield->GetField(mypos.x(),mypos.y(),mypos.z(), B(0), B(1), B(2)); S1=S+(0.5*ds)*D1; CalcDerivAndJacobian(ds/2.,mypos,dpos2,B,S1,dEdx,J2,D2); J2=J2+0.5*(J2*J1); mypos=pos+(ds/2.)*dpos2; //bfield->GetField(mypos.x(),mypos.y(),mypos.z(), B(0), B(1), B(2)); S2=S+(0.5*ds)*D2; CalcDerivAndJacobian(ds/2.,mypos,dpos3,B,S2,dEdx,J3,D3); J3=J3+0.5*(J3*J2); mypos=pos+ds*dpos3; //bfield->GetField(mypos.x(),mypos.y(),mypos.z(), B(0), B(1), B(2)); S3=S+ds*D3; CalcDerivAndJacobian(ds,mypos,dpos4,B,S3,dEdx,J4,D4); J4=J4+J4*J3; // New Jacobian matrix J+=ds*(ONE_SIXTH*J1+ONE_THIRD*J2+ONE_THIRD*J3+ONE_SIXTH*J4); // change in position DVector3 dpos =ds*(ONE_SIXTH*dpos1+ONE_THIRD*dpos2+ONE_THIRD*dpos3+ONE_SIXTH*dpos4); // Deal with changes in D double qrc_old=qpt/qBr2p/Bz_; double qrc_plus_D=D+qrc_old; double rc=sqrt(dpos.Perp2() +2.*qrc_plus_D*(dpos.x()*sinphi-dpos.y()*cosphi) +qrc_plus_D*qrc_plus_D); J(state_D,state_D)=q*(dpos.x()*sinphi-dpos.y()*cosphi+qrc_plus_D)/rc; J(state_D,state_q_over_pt)=qpt*qrc_old*(J(state_D,state_D)-1.); J(state_D,state_phi)=q*qrc_plus_D*(dpos.x()*cosphi+dpos.y()*sinphi)/rc; return NOERROR; } // Multiple scattering covariance matrix for central track parameters jerror_t DTrackFitterKalman::GetProcessNoiseCentral(double ds, DVector3 pos,double X0, DMatrix Sc,DMatrix &Q){ Q.Zero(); if (isfinite(X0) && X0<1e8 && X0>0.){ double tanl=Sc(state_tanl,0); double tanl2=tanl*tanl; double one_plus_tanl2=1.+tanl2; double q_over_pt=Sc(state_q_over_pt,0); // double X0=material->GetRadLen(pos.x(),pos.y(),pos.z()); double my_ds=fabs(ds); Q(state_phi,state_phi)=one_plus_tanl2; Q(state_tanl,state_tanl)=one_plus_tanl2*one_plus_tanl2; Q(state_q_over_pt,state_q_over_pt)=q_over_pt*q_over_pt*tanl2; Q(state_q_over_pt,state_tanl)=Q(state_tanl,state_q_over_pt) =q_over_pt*tanl*one_plus_tanl2; Q(state_D,state_D)=ds*ds/3.; Q(state_D,state_tanl)=Q(state_tanl,state_D) //=-my_ds/2.*cos(Sc(state_phi,0))*(one_plus_tanl2); =my_ds/2.*one_plus_tanl2; Q(state_D,state_phi)=Q(state_phi,state_D) //my_ds/2.*sin(Sc(state_phi,0))*sqrt(one_plus_tanl2); =my_ds/2.*sqrt(one_plus_tanl2); Q(state_D,state_q_over_pt)=Q(state_q_over_pt,state_D) // -my_ds/2.*cos(Sc(state_phi,0))*tanl; =my_ds/2.*q_over_pt*tanl; double p2=one_plus_tanl2/q_over_pt/q_over_pt; double log_correction=1.+0.038*log(my_ds/X0*(1.+mass2/p2)); double sig2_ms=0.0136*0.0136*(1.+mass2/p2)*my_ds/X0/p2 *log_correction*log_correction; //sig2_ms=0.; Q=sig2_ms*Q; } //Q.Print(); return NOERROR; } // Compute contributions to the covariance matrix due to multiple scattering jerror_t DTrackFitterKalman::GetProcessNoise(double ds,double z, double X0,DMatrix S,DMatrix &Q){ Q.Zero(); if (isfinite(X0) && X0<1e8 && X0>0.){ double tx=S(state_tx,0),ty=S(state_ty,0); double one_over_p_sq=S(state_q_over_p,0)*S(state_q_over_p,0); double my_ds=fabs(ds); double tx2=tx*tx; double ty2=ty*ty; double one_plus_tx2=1.+tx2; double one_plus_ty2=1.+ty2; double tsquare=tx2+ty2; double one_plus_tsquare=1.+tsquare; // double X0=material->GetRadLen(S(state_x,0),S(state_y,0),z); Q(state_tx,state_tx)=one_plus_tx2*one_plus_tsquare; Q(state_ty,state_ty)=one_plus_ty2*one_plus_tsquare; Q(state_tx,state_ty)=Q(state_ty,state_tx)=tx*ty*one_plus_tsquare; Q(state_x,state_x)=ds*ds/3.; Q(state_y,state_y)=ds*ds/3.; Q(state_y,state_ty)=Q(state_ty,state_y) // =my_ds/2.*tx*(one_plus_tsquare)/sqrt(tsquare); = my_ds/2.*sqrt(one_plus_tsquare*one_plus_ty2); Q(state_x,state_tx)=Q(state_tx,state_x) // =my_ds/2.*ty*sqrt((one_plus_tsquare)/(tsquare)); = my_ds/2.*sqrt(one_plus_tsquare*one_plus_tx2); double log_correction=1.+0.038*log(my_ds/X0*(1.+one_over_p_sq*mass2)); double sig2_ms= 0.0136*0.0136*(1.+one_over_p_sq*mass2) *one_over_p_sq*my_ds/X0*log_correction*log_correction; // sig2_ms=0.; Q=sig2_ms*Q; } return NOERROR; } // Calculate the energy loss per unit length given properties of the material // through which a particle of momentum p is passing double DTrackFitterKalman::GetdEdx(double q_over_p,double Z, double A, double rho){ if (rho<=0.) return 0.; //return 0; double betagamma=1./MASS/fabs(q_over_p); double betagamma2=betagamma*betagamma; double beta2=1./(1.+mass2*q_over_p*q_over_p); if (beta212) I0=9.76*Z+58.8*pow(Z,-0.19); // The next block of code approximates the "density effect" using the // prescription employed by GEANT 3.21. double delta=0.; //density effect double X=log10(betagamma); double X0,X1; double C=-2.*log(I0/28.816/sqrt(rho*Z/A))-1.; double Cbar=-C; if (rho>0.01){ // not a gas if (I0<100){ if (Cbar<=3.681) X0=0.2; else X0=-0.326*C-1.; X1=2.; } else{ if (Cbar<=5.215) X0=0.2; else X0=0.326*Cbar-1.5; X1=3.; } } else{ // gases X1=4.; if (Cbar<=9.5) X0=1.6; else if (Cbar>9.5 && Cbar<=10.) X0=1.7; else if (Cbar>10 && Cbar<=10.5) X0=1.8; else if (Cbar>10.5 && Cbar<=11.) X0=1.9; else if (Cbar>11.0 && Cbar<=12.25) X0=2.; else if (Cbar>12.25 && Cbar<=13.804){ X0=2.; X1=5.; } else { X0=0.326*Cbar-2.5; X1=5.; } } if (X>=X0 && X=X1) delta=4.606*X+C; I0*=1e-9; // convert to GeV return -0.0001535*Z/A*rho/beta2*(log(2.*Me*betagamma2*Tmax/I0/I0) -2.*beta2-delta); } // Calculate the most probable energy loss per unit length in units of // MeV cm^2/g in the FDC or CDC gas for a particle of momentum p. double DTrackFitterKalman::GetdEdx(double q_over_p,double mass_hyp){ double betagamma=1./mass_hyp/fabs(q_over_p); double beta2=1./(1.+mass_hyp*mass_hyp*q_over_p*q_over_p); if (beta2<0.001) return 0.; // Electron mass double Me=0.000511; //GeV // Material properties for gas double Z_over_A=0.85*0.45059+0.15*0.49989; double I0=(0.85*188.+0.15*85.)*1e-9; double rho=CDC_GAS_DENSITY; // First (non-logarithmic) term in Bethe-Block formula double mean_dedx=0.1535*Z_over_A/beta2; // Variables for calculating the density effect double X=log10(betagamma); double X0,X1; double C=-2.*log(I0/28.816/sqrt(rho*Z_over_A))-1.; double Cbar=-C; double delta=0.; X1=4.; if (Cbar<=9.5) X0=1.6; else if (Cbar>9.5 && Cbar<=10.) X0=1.7; else if (Cbar>10 && Cbar<=10.5) X0=1.8; else if (Cbar>10.5 && Cbar<=11.) X0=1.9; else if (Cbar>11.0 && Cbar<=12.25) X0=2.; else if (Cbar>12.25 && Cbar<=13.804){ X0=2.; X1=5.; } else { X0=0.326*Cbar-2.5; X1=5.; } if (X>=X0 && X=X1) delta=4.606*X+C; // Most probable energy loss from Landau theory (see Leo, pp. 51-52) return mean_dedx*(log(2.*Me*beta2*mean_dedx*rho*path_length*1e-3 /(1.-beta2)/I0/I0)-beta2+0.198-delta); } // Empirical form for sigma/mean for gaseous detectors with num_dedx samples // and sampling thickness path_length double DTrackFitterKalman::GetdEdxSigma(){ return 0.41*pow(double(num_dedx),-0.43)*pow(path_length,-0.32); } // Calculate the variance in the energy loss in a Gaussian approximation double DTrackFitterKalman::GetEnergyVariance(double ds,double q_over_p,double Z, double A, double rho){ if (rho<=0.) return 0.; //return 0; double betagamma=1./MASS/fabs(q_over_p); double betagamma2=betagamma*betagamma; double beta2=1./(1.+mass2*q_over_p*q_over_p); if (beta20 && my_fdchits.size()==0) r_outer_hit=my_cdchits[0]->origin.Perp(); // If we have trajectory entries for the CDC, start there if (max>1){ max--; forward_traj_cdc[0].h_id=forward_traj_cdc[max].h_id=0; z=forward_traj_cdc[max].pos.Z(); for (unsigned int m=max-1;m>0;m--){ forward_traj_cdc[m].h_id=0; newz=forward_traj_cdc[m].pos.Z(); r=forward_traj_cdc[m].pos.Perp(); // Turn off energy loss correction if the trajectory passes throught the // cdc end plate but there are no more measurements if (my_fdchits.size()==0 && newz>endplate_z) do_energy_loss=false; if (do_energy_loss && r1){ z=forward_traj[max].pos.Z(); for (unsigned int m=max-1;m>0;m--){ newz=forward_traj[m].pos.z(); r=forward_traj[m].pos.Perp(); if (do_energy_loss){ dedx=GetdEdx(S(state_q_over_p,0),forward_traj[m].Z, forward_traj[m].A,forward_traj[m].density); } Step(z,newz,dedx,S); z=newz; } if (do_energy_loss){ dedx=GetdEdx(S(state_q_over_p,0),forward_traj[1].Z, forward_traj[1].A,forward_traj[1].density); } newz=forward_traj[0].pos.Z(); Step(z,newz,dedx,S); } z_=newz; // turn energy loss back on do_energy_loss=true; return NOERROR; } // Interface routine for Kalman filter jerror_t DTrackFitterKalman::KalmanLoop(void){ if (z_0){ // Order the hits sort(my_fdchits.begin(),my_fdchits.end(),DKalmanFDCHit_cmp); if (my_cdchits.size()>0){ // Order the CDC hits by ring number sort(my_cdchits.begin(),my_cdchits.end(),DKalmanCDCHit_cmp); // For 2 adjacent hits in a single ring, swap hits from the default // ordering according to the phi values relative to the phi of the // innermost hit. if (my_cdchits.size()>1){ double phi0=my_cdchits[0]->origin.Phi(); for (unsigned int i=0;iring==my_cdchits[i+1]->ring){ double phi1=my_cdchits[i]->origin.Phi(); double phi2=my_cdchits[i+1]->origin.Phi(); if (fabs(phi1-phi0)>fabs(phi2-phi0)){ DKalmanCDCHit_t a=*my_cdchits[i]; DKalmanCDCHit_t b=*my_cdchits[i+1]; *my_cdchits[i]=b; *my_cdchits[i+1]=a; } my_cdchits[i+1]->status=1; } } } } // Initialize the state vector and covariance matrix S(state_x,0)=x_; S(state_y,0)=y_; S(state_tx,0)=tx_; S(state_ty,0)=ty_; S(state_q_over_p,0)=q_over_p_; // Initial guess for forward representation covariance matrix C0(state_x,state_x)=1.; C0(state_y,state_y)=1.; C0(state_tx,state_tx)=0.010*0.010; C0(state_ty,state_ty)=0.010*0.010; C0(state_q_over_p,state_q_over_p)=0.04*q_over_p_*q_over_p_; DMatrix Slast(S); DMatrix Clast(C0); double chisq_iter=chisq; double zvertex=65.; double scale_factor=50.,anneal_factor=1.; // Iterate over reference trajectories for (int iter2=0;iter2<(fit_type==kTimeBased?20:5);iter2++){ // Abort if momentum is too low if (fabs(S(state_q_over_p,0))>Q_OVER_P_MAX) break; double f=2.5; if (fit_type==kTimeBased) anneal_factor=scale_factor/pow(f,iter2)+1.; // Initialize path length variable and flight time len=0; ftime=0.; // If we have cdc hits, swim through the field past these measurements // first if (my_cdchits.size()>0){ SetCDCForwardReferenceTrajectory(S); } // Swim once through the field out to the most upstream FDC hit SetReferenceTrajectory(S); //C0=C; //printf("forward iteration %d cdc size %d\n",iter2,forward_traj_cdc.size()); if (forward_traj.size()> 1){ chisq_forward=1.e16; for (unsigned int iter=0;iter<20;iter++) { if (iter>0){ // Swim back to the first (most downstream) plane and use the new // values of S and C as the seed data to the Kalman filter SwimToPlane(S); } C=C0; // perform the kalman filter KalmanForward(anneal_factor,S,C,chisq); //KalmanForward(1.,S,C,chisq); //printf("forward chi2 %f p %f \n",chisq,1./S(state_q_over_p,0)); // include any hits from the CDC on the trajectory if (my_cdchits.size()>0 && forward_traj_cdc.size()>0){ // Proceed into CDC KalmanForwardCDC(anneal_factor,S,C,chisq); } //printf("iter %d chi2 %f %f\n",iter2,chisq,chisq_forward); if (!isfinite(chisq)) return VALUE_OUT_OF_RANGE; if (fabs(chisq-chisq_forward)<0.1 || chisq>chisq_forward) break; chisq_forward=chisq; Slast=S; Clast=C; } //iteration } //printf("iter2: %d chi2 %f %f\n",iter2,chisq_forward,chisq_iter); // Abort loop if the chisq is increasing if (fit_type==kWireBased && chisq_forward-chisq_iter>0.) break; if (fit_type==kTimeBased){ if (chisq_forward-chisq_iter>CHISQ_DIFF_CUT) break; if (iter2>7 && (fabs(chisq_forward-chisq_iter)<0.1 || chisq_forward-chisq_iter>0.)) break; } chisq_iter=chisq_forward; // Find the state at the so-called "vertex" position ExtrapolateToVertex(Slast,Clast); C=Clast; S=Slast; zvertex=z_; } // Convert from forward rep. to central rep. ConvertStateVector(zvertex,0.,0.,Slast,Clast,Sc,Cc); // Track Parameters at "vertex" phi_=Sc(state_phi,0); q_over_pt_=Sc(state_q_over_pt,0); tanl_=Sc(state_tanl,0); x_=Slast(state_x,0); y_=Slast(state_y,0); z_=zvertex; // Calculate the dEdx for the CDC and FDC hits CalcTrackdEdx(); if (DEBUG_LEVEL>0) cout << "Vertex: p " << 1./q_over_pt_/cos(atan(tanl_)) << " theta " << 90.0-180./M_PI*atan(tanl_) << " vertex " << x_ << " " << y_ << " " << z_ <dummy; for (unsigned int i=0;i<5;i++){ dummy.clear(); for(unsigned int j=0;j<5;j++){ dummy.push_back(Cc(i,j)); } cov.push_back(dummy); } // ... forward parameterization for (unsigned int i=0;i<5;i++){ dummy.clear(); for(unsigned int j=0;j<5;j++){ dummy.push_back(Clast(i,j)); } fcov.push_back(dummy); } // total chisq and ndf chisq_=chisq_iter; ndf=2*my_fdchits.size()+my_cdchits.size()-5; if (DEBUG_HISTS && fit_type==kTimeBased){ TH2F *fdc_xresiduals=(TH2F*)gROOT->FindObject("fdc_xresiduals"); if (fdc_xresiduals){ for (unsigned int i=0;iFill(my_fdchits[i]->z,my_fdchits[i]->xres); } } TH2F *fdc_yresiduals=(TH2F*)gROOT->FindObject("fdc_yresiduals"); if (fdc_yresiduals){ for (unsigned int i=0;iFill(my_fdchits[i]->z,my_fdchits[i]->yres); } } /* TH2F *fdc_ypulls=(TH2F*)gROOT->FindObject("fdc_ypulls"); if (fdc_ypulls) fdc_ypulls->Fill(my_fdchits[id]->z,R(state_y,0)/sqrt(RC(1,1))); TH2F *fdc_xpulls=(TH2F*)gROOT->FindObject("fdc_xpulls"); if (fdc_xpulls) fdc_xpulls->Fill(my_fdchits[id]->z,R(state_y,0)/sqrt(RC(0,0))); */ } return NOERROR; } // Deal with CDC-only tracks with theta<50 degrees using forward parameters if (my_cdchits.size()>0 && tanl_>0.84){ // Order the CDC hits by ring number sort(my_cdchits.begin(),my_cdchits.end(),DKalmanCDCHit_cmp); // For 2 adjacent hits in a single ring, swap hits from the default // ordering according to the phi values relative to the phi of the // innermost hit. if (my_cdchits.size()>1){ double phi0=my_cdchits[0]->origin.Phi(); for (unsigned int i=0;iring==my_cdchits[i+1]->ring){ double phi1=my_cdchits[i]->origin.Phi(); double phi2=my_cdchits[i+1]->origin.Phi(); if (fabs(phi1-phi0)>fabs(phi2-phi0)){ DKalmanCDCHit_t a=*my_cdchits[i]; DKalmanCDCHit_t b=*my_cdchits[i+1]; *my_cdchits[i]=b; *my_cdchits[i+1]=a; } if (my_cdchits[i+1]->stereo==0.) my_cdchits[i+1]->status=1; } } } // Initialize the state vector and covariance matrix S(state_x,0)=x_; S(state_y,0)=y_; S(state_tx,0)=tx_; S(state_ty,0)=ty_; S(state_q_over_p,0)=q_over_p_; // Initial guess for forward representation covariance matrix C0(state_x,state_x)=1; C0(state_y,state_y)=1; C0(state_tx,state_tx)=0.010*0.010; C0(state_ty,state_ty)=0.010*0.010; C0(state_q_over_p,state_q_over_p)=0.04*q_over_p_*q_over_p_; DMatrix Slast(S); DMatrix Clast(C0); double chisq_iter=chisq; double zvertex=65.; double scale_factor=99.,anneal_factor=1.; // Iterate over reference trajectories for (int iter2=0;iter2<(fit_type==kTimeBased?20:5);iter2++){ // Abort if momentum is too low if (fabs(S(state_q_over_p,0))>Q_OVER_P_MAX) break; if (fit_type==kTimeBased){ double f=2.75; anneal_factor=scale_factor/pow(f,iter2)+1.; } // Initialize path length variable and flight time len=0; ftime=0.; SetCDCForwardReferenceTrajectory(S); if (forward_traj_cdc.size()> 1){ chisq_forward=1.e16; for (unsigned int iter=0;iter<20;iter++) { // perform the kalman filter if (iter>0){ // Swim back to the first (most downstream) plane and use the new // values of S as the seed data to the Kalman filter SwimToPlane(S); } C=C0; chisq=0.; KalmanForwardCDC(anneal_factor,S,C,chisq); if (chisq==0.){ chisq=1.e16; break; } //printf("iter %d chi2 %f %f\n",iter,chisq,chisq_forward); if (!isfinite(chisq)) return VALUE_OUT_OF_RANGE; if (fabs(chisq-chisq_forward)<0.1 || chisq>chisq_forward) break; chisq_forward=chisq; Slast=S; Clast=C; } //iteration } //printf("iter2: %d factor %f chi2 %f %f\n",iter2,anneal_factor,chisq_forward,chisq_iter); // Abort loop if the chisq is increasing if (fit_type==kWireBased && chisq_forward-chisq_iter>0.) break; if (fit_type==kTimeBased){ if (chisq_forward-chisq_iter>CHISQ_DIFF_CUT) break; if (iter2>7 && (fabs(chisq_forward-chisq_iter)<0.1 || chisq_forward-chisq_iter>0.)) break; } chisq_iter=chisq_forward; // Find the state at the so-called "vertex" position ExtrapolateToVertex(Slast,Clast); C=Clast; S=Slast; zvertex=z_; } // Convert from forward rep. to central rep. ConvertStateVector(zvertex,0.,0.,Slast,Clast,Sc,Cc); // Track Parameters at "vertex" phi_=Sc(state_phi,0); q_over_pt_=Sc(state_q_over_pt,0); tanl_=Sc(state_tanl,0); x_=Slast(state_x,0); y_=Slast(state_y,0); z_=zvertex; // Calculate the dEdx for the CDC hits CalcTrackdEdx(); if (DEBUG_LEVEL>0) cout << "Vertex: p " << 1./q_over_pt_/cos(atan(tanl_)) << " theta " << 90.0-180./M_PI*atan(tanl_) << " vertex " << x_ << " " << y_ << " " << z_ <dummy; // ... forward parameterization for (unsigned int i=0;i<5;i++){ dummy.clear(); for(unsigned int j=0;j<5;j++){ dummy.push_back(Clast(i,j)); } fcov.push_back(dummy); } // ... central parameterization for (unsigned int i=0;i<5;i++){ dummy.clear(); for(unsigned int j=0;j<5;j++){ dummy.push_back(Cc(i,j)); } cov.push_back(dummy); } // total chisq and ndf chisq_=chisq_iter; ndf=my_cdchits.size()-5; return NOERROR; } /* // Deal with CDC-only tracks with theta>130 degrees using forward parameters if (my_cdchits.size()>0 && tanl_<-0.84){ // Order the CDC hits by ring number sort(my_cdchits.begin(),my_cdchits.end(),DKalmanCDCHit_cmp); // Initialize the state vector and covariance matrix S(state_x,0)=x_; S(state_y,0)=y_; S(state_tx,0)=tx_; S(state_ty,0)=ty_; S(state_q_over_p,0)=q_over_p_; // Initial guess for forward representation covariance matrix C0(state_x,state_x)=1; C0(state_y,state_y)=1; C0(state_tx,state_tx)=0.010*0.010; C0(state_ty,state_ty)=0.010*0.010; C0(state_q_over_p,state_q_over_p)=0.04*q_over_p_*q_over_p_; DMatrix Slast(S); DMatrix Clast(C0); double chisq_iter=chisq; double zvertex=65.; double scale_factor=200.,anneal_factor=1.; // Iterate over reference trajectories for (int iter2=0;iter2<20;iter2++){ //for (int iter2=0;iter2<1;iter2++){ //if (iter2>0) do_energy_loss=true; //if (iter2>0) do_multiple_scattering=true; if (fit_type==kTimeBased){ double f=1.75; anneal_factor=scale_factor/pow(f,iter2)+1.; } // Initialize path length variable len=0; SetCDCBackwardReferenceTrajectory(S); if (forward_traj_cdc.size()> 0){ unsigned int num_iter=NUM_ITER; //num_iter=1; //if (my_cdchits.size()==0) num_iter=3; chisq_backward=1.e16; for (unsigned int iter=0;iter<20;iter++) { // perform the kalman filter if (iter>0){ // Swim back to the first (most downstream) plane and use the new // values of S and C as the seed data to the Kalman filter SwimToPlane(S); } C=C0; chisq=0.; KalmanForwardCDC(anneal_factor,S,C,chisq); //printf("iter %d chi2 %f %f\n",iter,chisq,chisq_backward); if (isnan(chisq) || (fabs(chisq-chisq_backward)<0.1 || chisq>chisq_backward)) break; chisq_backward=chisq; Slast=S; Clast=C; } //iteration } // printf("iter2: %d factor %f chi2 %f %f\n",iter2,anneal_factor,chisq_backward,chisq_iter); // Abort loop if the chisq is not changing much or increasing too much if ( isnan(chisq_backward) || (//iter2>12 && (fabs(chisq_backward-chisq_iter)<0.1 || chisq_backward-chisq_iter>10.))) break; chisq_iter=chisq_backward; // Find the state at the so-called "vertex" position ExtrapolateToVertex(Slast,Clast); C=Clast; S=Slast; zvertex=z_; } // Convert from forward rep. to central rep. ConvertStateVector(zvertex,0.,0.,Slast,Clast,Sc,Cc); // Track Parameters at "vertex" phi_=Sc(state_phi,0); q_over_pt_=Sc(state_q_over_pt,0); tanl_=Sc(state_tanl,0); x_=Slast(state_x,0); y_=Slast(state_y,0); z_=zvertex; if (DEBUG_LEVEL>0) cout << "Vertex: p " << 1./q_over_pt_/cos(atan(tanl_)) << " theta " << 90.0-180./M_PI*atan(tanl_) << " vertex " << x_ << " " << y_ << " " << z_ <dummy; // ... forward parameterization for (unsigned int i=0;i<5;i++){ dummy.clear(); for(unsigned int j=0;j<5;j++){ dummy.push_back(Clast(i,j)); } fcov.push_back(dummy); } // ... central parameterization for (unsigned int i=0;i<5;i++){ dummy.clear(); for(unsigned int j=0;j<5;j++){ dummy.push_back(Cc(i,j)); } cov.push_back(dummy); } // total chisq and ndf chisq_=chisq_iter; ndf=my_cdchits.size()-5; return NOERROR; } */ // Fit in Central region: deal with hits in the CDC if (my_cdchits.size()>0){ // Order the CDC hits by radius sort(my_cdchits.begin(),my_cdchits.end(),DKalmanCDCHit_cmp); // For 2 adjacent hits in a single ring, swap hits from the default // ordering according to the phi values relative to the phi of the // innermost hit. if (my_cdchits.size()>1){ double phi0=my_cdchits[0]->origin.Phi(); for (unsigned int i=0;iring==my_cdchits[i+1]->ring){ double phi1=my_cdchits[i]->origin.Phi(); double phi2=my_cdchits[i+1]->origin.Phi(); if (fabs(phi1-phi0)>fabs(phi2-phi0)){ DKalmanCDCHit_t a=*my_cdchits[i]; DKalmanCDCHit_t b=*my_cdchits[i+1]; *my_cdchits[i]=b; *my_cdchits[i+1]=a; } } } } // Initialize the state vector and covariance matrix Sc(state_q_over_pt,0)=q_over_pt_; Sc(state_phi,0)=phi_; Sc(state_tanl,0)=tanl_; Sc(state_z,0)=z_; Sc(state_D,0)=0.; //C0(state_z,state_z)=1.; C0(state_z,state_z)=2.0; C0(state_q_over_pt,state_q_over_pt)=0.20*0.20*q_over_pt_*q_over_pt_; C0(state_phi,state_phi)=0.01*0.01; C0(state_D,state_D)=1.0; double dlambda=0.05; //dlambda=0.1; C0(state_tanl,state_tanl)=(1.+tanl_*tanl_)*(1.+tanl_*tanl_) *dlambda*dlambda; // Initialization Cc=C0; DMatrix Sclast(Sc); DMatrix Cclast(Cc); DVector3 pos0=pos; DVector3 best_pos=pos; // Make sure the energy loss and multiple scattering flags are set do_energy_loss=true; do_multiple_scattering=true; // iteration double scale_factor=100., anneal_factor=1.; double chisq_iter=chisq; for (int iter2=0;iter2<(fit_type==kTimeBased?20:5);iter2++){ // Break out of loop if p is too small double q_over_p=Sc(state_q_over_pt,0)*cos(atan(Sc(state_tanl,0))); if (fabs(q_over_p)>Q_OVER_P_MAX) break; // Initialize path length variable and flight time len=0.; ftime=0.; // Abort if the chisq for the previous iteration is junk if (chisq_central==0.) break; // Calculate an annealing factor for the measurement errors that depends // on the iteration,so that we approach the "true' measurement errors // by the last iteration. double f=3.5; if (fit_type==kTimeBased) anneal_factor=scale_factor/pow(f,iter2)+1.; // Initialize trajectory deque and position SetCDCReferenceTrajectory(pos0,Sc); if (central_traj.size()>1){ // Iteration for given reference trajectory chisq=1.e16; for (int iter=0;iter<20;iter++){ Cc=C0; if (iter>0){ SwimCentral(pos,Sc); //anneal_factor=scale_factor/pow(f,iter)+1.; } //anneal_factor=1.; jerror_t error=NOERROR; error=KalmanCentral(anneal_factor,Sc,Cc,pos,chisq_central); if (error!=NOERROR) break; if (chisq_central==0.) break; //fom=anneal_factor*chisq_central; if (chisq_central>=1e16 ){ cout << "-- central fit failed --" <0) cout << "iteration " << iter+1 << " factor " << anneal_factor << " chi2 " << chisq_central << " p " << 1./Sc(state_q_over_pt,0)/cos(atan(Sc(state_tanl,0))) << " theta " << 90.-180./M_PI*atan(Sc(state_tanl,0)) << " vertex " << x_ << " " << y_ << " " << z_ <0) break; return VALUE_OUT_OF_RANGE; } if (fabs(chisq_central-chisq)<0.1 || (chisq_central>chisq )) break; // Save the current "best" state vector and covariance matrix Cclast=Cc; Sclast=Sc; pos0=pos; chisq=chisq_central; } //iteration } // Abort loop if the chisq is increasing if (fit_type==kWireBased && chisq-chisq_iter>0.) break; if (!isfinite(chisq_central)) break; if (fit_type==kTimeBased){ if (chisq-chisq_iter>CHISQ_DIFF_CUT) break; if (iter2>10 && (fabs(chisq-chisq_iter)<0.1 || chisq-chisq_iter>0.)) break; } chisq_iter=chisq; // Find track parameters where track crosses beam line ExtrapolateToVertex(pos0,Sclast,Cclast); Cc=Cclast; Sc=Sclast; best_pos=pos0; } if (chisq_iter==1.e16) { _DBG_ << "Central fit failed!" <0) cout << "Vertex: p " << 1./Sclast(state_q_over_pt,0)/cos(atan(Sclast(state_tanl,0))) << " theta " << 90.-180./M_PI*atan(Sclast(state_tanl,0)) << " vertex " << x_<< " " << y_<< " " << z_<dummy; for (unsigned int i=0;i<5;i++){ dummy.clear(); for(unsigned int j=0;j<5;j++){ dummy.push_back(Cclast(i,j)); } cov.push_back(dummy); } // total chisq and ndf chisq_=chisq_iter; ndf=my_cdchits.size()-5; } if (DEBUG_HISTS && fit_type==kTimeBased){ TH2F *cdc_residuals=(TH2F*)gROOT->FindObject("cdc_residuals"); if (cdc_residuals){ for (unsigned int i=0;iFill(my_cdchits[i]->ring,my_cdchits[i]->residual); } /* TH2F *cdc_pulls=(TH2F*)gROOT->FindObject("cdc_pulls"); if (cdc_pulls){ //cdc_pulls->Fill(my_cdchits[cdc_index]->ring,dm/sqrt(var)); } */ } return NOERROR; } #define ITMAX 100 #define CGOLD 0.3819660 #define ZEPS 1.0e-10 #define SHFT(a,b,c,d) (a)=(b);(b)=(c);(c)=(d); #define SIGN(a,b) ((b)>=0.0?fabs(a):-fabs(a)) // Routine for finding the minimum of a function bracketed between two values // (see Numerical Recipes in C, pp. 404-405). double DTrackFitterKalman::BrentsAlgorithm(double ds1,double ds2, double dedx,DVector3 &pos,DVector3 origin, DVector3 dir, DMatrix &Sc){ int iter; double a,b,d=0.,etemp,fu,fv,fw,fx,p,q,r,tol1,tol2,u,v,w,x,xm; double e=0.0; // will be distance moved on step before last double ax=0.; double bx=-ds1; double cx=-ds1-ds2; a=(axcx?ax:cx); x=w=v=bx; // Save the starting position // DVector3 pos0=pos; // DMatrix S0(Sc); // Step to intermediate point FixedStep(pos,x,Sc,dedx); DVector3 wirepos=origin+((pos.z()-origin.z())/dir.z())*dir; double u_old=x; // initialization fw=fv=fx=(pos-wirepos).Mag(); // main loop for (iter=1;iter<=ITMAX;iter++){ xm=0.5*(a+b); tol2=2.0*(tol1=EPS*fabs(x)+ZEPS); if (fabs(x-xm)<=(tol2-0.5*(b-a))){ if (pos.z()>=endplate_z-endplate_dz){ double my_endz=endplate_z-endplate_dz; // Check if the minimum doca would occur outside the straw and if so, bring the state // vector back to the end of the straw int iter=0; while (fabs(pos.z()-my_endz)>EPS && iter<20){ u=x-(my_endz-pos.z())*sin(atan(Sc(state_tanl,0))); x=u; // Function evaluation FixedStep(pos,u_old-u,Sc,dedx); u_old=u; iter++; } //printf("new z %f ds %f \n",pos.z(),x); } if (pos.z()<=cdc_origin[2]){ int iter=0; while (fabs(pos.z()-cdc_origin[2])>EPS && iter<20){ u=x-(cdc_origin[2]-pos.z())*sin(atan(Sc(state_tanl,0))); x=u; // Function evaluation FixedStep(pos,u_old-u,Sc,dedx); u_old=u; iter++; } //printf("new z %f ds %f \n",pos.z(),x); } return cx-x; } // trial parabolic fit if (fabs(e)>tol1){ r=(x-w)*(fx-fv); q=(x-v)*(fx-fw); p=(x-v)*q-(x-w)*r; q=2.0*(q-r); if (q>0.0) p=-p; q=fabs(q); etemp=e; e=d; if (fabs(p)>=fabs(0.5*q*etemp) || p<=q*(a-x) || p>=q*(b-x)) // fall back on the Golden Section technique d=CGOLD*(e=(x>=xm?a-x:b-x)); else{ // parabolic step d=p/q; u=x+d; if (u-a=xm?a-x:b-x)); } u=(fabs(d)>=tol1 ? x+d: x+SIGN(tol1,d)); // Function evaluation FixedStep(pos,u_old-u,Sc,dedx); u_old=u; wirepos=origin+((pos.z()-origin.z())/dir.z())*dir; fu=(pos-wirepos).Mag(); //printf("Brent: z %f d %f\n",pos.z(),fu); if (fu<=fx){ if (u>=x) a=x; else b=x; SHFT(v,w,x,u); SHFT(fv,fw,fx,fu); } else { if (ucx?ax:cx); x=w=v=bx; // Save the state vector after the last step DMatrix S0(5,1); S0=S; // Step to intermediate point Step(z,z+x,dedx,S0); DVector3 wirepos=origin+((z+x-origin.z())/dir.z())*dir; DVector3 pos(S0(state_x,0),S0(state_y,0),z+x); // initialization fw=fv=fx=(pos-wirepos).Mag(); // main loop for (iter=1;iter<=ITMAX;iter++){ xm=0.5*(a+b); tol2=2.0*(tol1=EPS*fabs(x)+ZEPS); if (fabs(x-xm)<=(tol2-0.5*(b-a))){ return x; } // trial parabolic fit if (fabs(e)>tol1){ r=(x-w)*(fx-fv); q=(x-v)*(fx-fw); p=(x-v)*q-(x-w)*r; q=2.0*(q-r); if (q>0.0) p=-p; q=fabs(q); etemp=e; e=d; if (fabs(p)>=fabs(0.5*q*etemp) || p<=q*(a-x) || p>=q*(b-x)) // fall back on the Golden Section technique d=CGOLD*(e=(x>=xm?a-x:b-x)); else{ // parabolic step d=p/q; u=x+d; if (u-a=xm?a-x:b-x)); } u=(fabs(d)>=tol1 ? x+d: x+SIGN(tol1,d)); // Function evaluation S0=S; Step(z,z+u,dedx,S0); wirepos=origin+((z+u-origin.z())/dir.z())*dir; pos.SetXYZ(S0(state_x,0),S0(state_y,0),z+u); fu=(pos-wirepos).Mag(); if (fu<=fx){ if (u>=x) a=x; else b=x; SHFT(v,w,x,u); SHFT(fv,fw,fx,fu); } else { if (uorigin; DVector3 dir=my_cdchits[cid]->dir; // Track direction parameters double cosphi=cos(S(state_phi,0)); double sinphi=sin(S(state_phi,0)); double tanl=S(state_tanl,0); //Position relative to wire origin double dz=pos.z()-origin.z(); double dx=pos.x()-origin.x(); double dy=pos.y()-origin.y(); // square of straw radius double rs2=0.8*0.8; // Useful temporary variables related to the direction of the wire double A=1.-dir.x()*dir.x(); double B=-2.*dir.x()*dir.y(); double C=-2.*dir.x()*dir.z(); double D=-2.*dir.y()*dir.z(); double E=1.-dir.y()*dir.y(); double F=1.-dir.z()*dir.z(); // The path length in the straw is given by s=sqrt(b*b-4*a*c)/a/cosl. // a, b, and c follow. double a=A*cosphi*cosphi+B*cosphi*sinphi+C*cosphi*tanl+D*sinphi*tanl +E*sinphi*sinphi+F*tanl*tanl; double b=2.*A*dx*cosphi+B*dx*sinphi+B*dy*cosphi+C*dx*tanl+C*cosphi*dz +D*dy*tanl+D*sinphi*dz+2.*E*dy*sinphi+2.*F*dz*tanl; double c=A*dx*dx+B*dx*dy+C*dx*dz+D*dy*dz+E*dy*dy+F*dz*dz-rs2; // Check for valid arc length and compute dEdx double temp=b*b-4.*a*c; if (temp>0){ double cosl=fabs(cos(atan(tanl))); // arc length and energy deposition ds=CDC_GAS_DENSITY*sqrt(temp)/a/cosl; // g/cm^2 dE=1000.*my_cdchits[cid]->dE; //MeV return NOERROR; } return VALUE_OUT_OF_RANGE; } // Routine for estimating the energy loss per unit path length within the // active volume of the drift chambers jerror_t DTrackFitterKalman::CalcTrackdEdx(){ DMatrix S(5,1); double ds=0.; double dE=0.; // Vectors of dE and ds measurements vectordEdx_list; // Hits in central part of CDC for (unsigned int m=0;m0){ unsigned int cdc_index=central_traj[m].h_id-1; // Get the state vector from the reference trajectory S=*central_traj[m].S; if (CalcdEdxHit(S,central_traj[m].pos,cdc_index,dE,ds)==NOERROR){ DKalman_dedx_t temp; temp.dE=dE; temp.ds=ds; dEdx_list.push_back(temp); } // Reset the h_id to zero. Only do this because for tracks near 50 // degrees, the wire-based and time-based passes do not necessarily use // the same parameterization, and the trajectory deques are currently not // deleted between wire- and time-based passes. central_traj[m].h_id=0; } } // Hits in CDC with theta<50 degrees for (unsigned int m=0;m0){ unsigned int cdc_index=forward_traj_cdc[m].h_id-1; // Get the track parameters from the reference trajectory S=*forward_traj_cdc[m].S; DMatrix Sc(5,1); ConvertStateVector(forward_traj_cdc[m].pos.z(),0.,0.,S,Sc); if (CalcdEdxHit(Sc,forward_traj_cdc[m].pos,cdc_index,dE,ds)==NOERROR){ DKalman_dedx_t temp; temp.dE=dE; temp.ds=ds; dEdx_list.push_back(temp); } // Reset the h_id to zero. Only do this because for tracks near 50 // degrees, the wire-based and time-based passes do not necessarily use // the same parameterization, and the trajectory deques are currently not // deleted between wire- and time-based passes. forward_traj_cdc[m].h_id=0; } } // Hits in FDC for (unsigned int m=0;m0){ // Get the track parameters from the reference trajectory S=*forward_traj[m].S; double tx=S(state_tx,0); double ty=S(state_ty,0); if (DEBUG_HISTS && fit_type==kTimeBased){ TH2F *thetay_vs_thetax=(TH2F*)gROOT->FindObject("thetay_vs_thetax"); if (thetay_vs_thetax){ for (unsigned int i=0;iFill(180./M_PI*atan(tx),180./M_PI*atan(ty)); } } } DKalman_dedx_t temp; // Straight line approximation for arc length temp.ds=FDC_GAS_DENSITY*1.0*sqrt(1.+tx*tx+ty*ty); // g/cm^2 temp.dE=1000.*my_fdchits[forward_traj[m].h_id-1]->dE; // MeV dEdx_list.push_back(temp); } } //printf("dEsum %f dssum %f\n",1.e6*sum_dE,sum_ds); // Sort the dEdx entries from smallest to largest sort(dEdx_list.begin(),dEdx_list.end(),DKalman_dedx_cmp); // Compute the dEdx in the active volume for the track using a truncated // mean to minimize the effect of the long Landau tail unsigned int imax =(dEdx_list.size()>5)?int(0.6*dEdx_list.size()):dEdx_list.size(); double sum_dE=0.; double sum_ds=0.; for (unsigned int i=0;i0){ num_dedx=imax; path_length=sum_ds/double(imax); track_dedx=sum_dE/sum_ds; // MeV cm^2/g } return NOERROR; } // Routine for finding the minimum of a function bracketed between two values double DTrackFitterKalman::GoldenSection(double ds1,double ds2, double dedx,DVector3 pos,DVector3 origin, DVector3 dir, DMatrix Sc){ double ds=ds1+ds2; double s0=0.; double s3=-ds; // need to backtrack double golden_ratio1=0.61803399; double golden_ratio2=1.-golden_ratio1; unsigned int iter=0; double d1,d2; double s1,s2; if (fabs(s3+ds1)EPS*(fabs(s1)+fabs(s2))&& iterEPS*(fabs(s1)+fabs(s2))&& iterEPS2){ iter++; S0=S; double stemp; if(d2EPS*(fabs(s1)+fabs(s2))){ // Check for exit condition near zero if (fabs(s3)origin; DVector3 dir=my_cdchits[cdc_index]->dir; DVector3 wirepos=origin+((pos.z()-origin.z())/dir.z())*dir; // doca variables double doca,old_doca=(pos-wirepos).Mag(); // energy loss double dedx=0.; // Boolean for flagging when we are done with measurements bool more_measurements=true; // Initialize S0_ and perform the loop over the trajectory S0_=(*central_traj[0].S); for (unsigned int k=1;koperator()(i,j)=Cc(i,j); } } */ // update position based on new doca to reference trajectory pos(0)=central_traj[k].pos.x()-Sc(state_D,0)*sin(Sc(state_phi,0)); pos(1)=central_traj[k].pos.y()+Sc(state_D,0)*cos(Sc(state_phi,0)); pos(2)=Sc(state_z,0); // Save the current state of the reference trajectory S0_=S0; // new wire position wirepos=origin+((pos.z()-origin.z())/dir.z())*dir; // new doca doca=(pos-wirepos).Mag(); // Check if the doca is no longer decreasing if ((doca>old_doca) && (pos.z()cdc_origin[2]) && more_measurements){ if (my_cdchits[cdc_index]->status==0){ // Mark previous point on ref trajectory with a hit id for the straw central_traj[k-1].h_id=cdc_index+1; // Save values at end of current step DVector3 pos0=central_traj[k].pos; // dEdx for current position along trajectory double q_over_p=Sc(state_q_over_pt,0)*cos(atan(Sc(state_tanl,0))); if (do_energy_loss){ dedx=GetdEdx(q_over_p,central_traj[k].Z, central_traj[k].A,central_traj[k].density); } // Variables for the computation of D at the doca to the wire double D=Sc(state_D,0); double q=(Sc(state_q_over_pt,0)>0)?1.:-1.; double qpt=1./Sc(state_q_over_pt,0); double sinphi=sin(Sc(state_phi,0)); double cosphi=cos(Sc(state_phi,0)); double Bx=0.,By=0.,Bz=-2.; //bfield->GetField(pos.x(),pos.y(),pos.z(), Bx, By, Bz); bfield->GetFieldBicubic(pos.x(),pos.y(),pos.z(), Bx, By, Bz); double Bz_=fabs(Bz); // We've passed the true minimum; now use Brent's algorithm // to find the best doca. See Numerical Recipes in C, pp 404-405 ds2=BrentsAlgorithm(ds,ds,dedx,pos,origin,dir,Sc); int numstep=(int)(ds2/CDC_STEP_SIZE); double myds=CDC_STEP_SIZE; if (ds2<0) myds*=-1.; double ds3=ds2-CDC_STEP_SIZE*numstep; // propagate covariance matrix along the reference trajectory. // We ignore the tiny amount of multiple scattering for these small // steps. for (int j=0;jEPS2){ // Compute the Jacobian matrix StepJacobian(pos0,origin,dir,ds3,S0,dedx,J); // Step along reference trajectory FixedStep(pos0,ds3,S0,dedx); // Update covariance matrix JT=DMatrix(DMatrix::kTransposed,J); Cc=J*(Cc*JT); } // Compute the value of D (signed distance to the reference trajectory) // at the doca to the wire DVector3 dpos1=pos0-central_traj[k].pos; double qrc_old=qpt/qBr2p/Bz_; double qrc_plus_D=D+qrc_old; double rc=sqrt(dpos1.Perp2() +2.*qrc_plus_D*(dpos1.x()*sinphi-dpos1.y()*cosphi) +qrc_plus_D*qrc_plus_D); Sc(state_D,0)=q*rc-qrc_old; // wire position wirepos=origin+((pos.z()-origin.z())/dir.z())*dir; //doca doca=(pos-wirepos).Perp(); // Measurement double measurement=0.; if (fit_type==kTimeBased){ measurement=CDC_DRIFT_SPEED*(my_cdchits[cdc_index]->t -central_traj[k].t); // Measurement error V=anneal_factor*0.000225; } // prediction for measurement DVector3 diff=pos-wirepos; DVector3 mdir=pos-wirepos-(diff.Dot(dir))*dir; mdir.SetMag(1.0); double prediction=diff.Dot(mdir); // Projection matrix sinphi=sin(Sc(state_phi,0)); cosphi=cos(Sc(state_phi,0)); double ux=dir.x(); double uy=dir.y(); double uxuy=ux*uy; double ux2=ux*ux; double uy2=uy*uy; double dx=diff.x(); double dy=diff.y(); if (prediction>0.){ H(0,state_D)=H_T(state_D,0) =(dy*(uxuy*sinphi+(1.-uy2)*cosphi)-dx*((1.-ux2)*sinphi+uxuy*cosphi))/prediction; H(0,state_phi)=H_T(state_phi,0) =-Sc(state_D,0)*(dx*((1.-ux2)*cosphi-uxuy*sinphi)+dy*((1.-uy2)*sinphi-uxuy*cosphi))/prediction; H(0,state_z)=H_T(state_z,0) =-dir.z()*(dx*ux+dy*uy)/prediction; } // Difference and variance double var=V,var_pred=0.; if (prediction>0.){ var_pred=(H*(Cc*H_T))(0,0); var+=var_pred; } double dm=measurement-prediction; if (var_pred<0.){ /* Cc.Print(); cout << "Negative variance???" << var_pred << endl; H.Print(); */ return VALUE_OUT_OF_RANGE; } // probability double p=exp(-0.5*dm*dm/var)/sqrt(2.*M_PI*var); p=1.; //if (fabs(dm)/sqrt(var)>3.) dm=3.*sqrt(var)*(dm>0?1.:-1.); if (DEBUG_LEVEL>0) cout << "ring " << my_cdchits[cdc_index]->ring << " Dm " << measurement << " Dm-Dpred " << dm << " sigma " << sqrt(var) << " p " << 1./(Sc(state_q_over_pt,0)*cos(atan(Sc(state_tanl,0)))) << " theta " << 90.-180./M_PI*atan(Sc(state_tanl,0)) << " x " << pos.x() << " y " << pos.y() << " z " << pos.z() << endl; // Inverse of variance InvV=1./(p*V+var_pred); // Compute Kalman gain matrix K=InvV*(Cc*H_T); // Update the state vector dS=p*dm*K; //dS.Zero(); Sc=Sc+dS; // Update state vector covariance matrix Cc=Cc-(K*(H*Cc)); /* This is more accurate, but is it worth it?? // update position on current trajectory based on corrected doca to // reference trajectory pos=pos0; pos(0)+=-Sc(state_D,0)*sin(Sc(state_phi,0)); pos(1)+= Sc(state_D,0)*cos(Sc(state_phi,0)); pos(2)=Sc(state_z,0); // wire position wirepos=origin+((pos.z()-origin.z())/dir.z())*dir; // revised prediction diff=pos-wirepos; mdir=pos-wirepos-(diff.Dot(dir))*dir; mdir.SetMag(1.0); prediction=diff.Dot(mdir); */ // calculate the residual dm*=(1.-(H*K)(0,0)); //dm=measurement-prediction; my_cdchits[cdc_index]->residual=dm; // Update chi2 for this hit var=V*(1.-(H*K)(0,0)); chisq+=dm*dm/var; // propagate the covariance matrix to the next point on the trajectory for (int j=0;jEPS){ // Compute the Jacobian matrix StepJacobian(pos0,origin,dir,-ds3,S0,dedx,J); // Update covariance matrix JT=DMatrix(DMatrix::kTransposed,J); Cc=J*(Cc*JT); } /* for (unsigned int i=0;i<5;i++){ for (unsigned int j=0;j<5;j++){ central_traj[k].C->operator()(i,j)=Cc(i,j); } } */ // Step to the next point on the trajectory Sc=S0_+J*(Sc-S0); // update position on current trajectory based on corrected doca to // reference trajectory pos=central_traj[k].pos; pos(0)+=-Sc(state_D,0)*sin(Sc(state_phi,0)); pos(1)+= Sc(state_D,0)*cos(Sc(state_phi,0)); pos(2)=Sc(state_z,0); } else { if (cdc_index>0) cdc_index--; else cdc_index=0; } // new wire origin and direction if (cdc_index>0){ cdc_index--; origin=my_cdchits[cdc_index]->origin; dir=my_cdchits[cdc_index]->dir; } else{ origin.SetXYZ(0.,0.,65.); dir.SetXYZ(0,0,1.); more_measurements=false; } // Update the wire position wirepos=origin+((pos.z()-origin.z())/dir.z())*dir; //s+=ds2; // new doca doca=(pos-wirepos).Mag(); } old_doca=doca; } if (DEBUG_LEVEL>0) cout << " p " << 1./(Sc(state_q_over_pt,0)*cos(atan(Sc(state_tanl,0)))) << " theta " << 90.-180./M_PI*atan(Sc(state_tanl,0)) << " vertex " << pos.x() << " " << pos.y() <<" " << pos.z() <0){ unsigned int id=forward_traj[k].h_id-1; double cosa=my_fdchits[id]->cosa; double sina=my_fdchits[id]->sina; double u=my_fdchits[id]->uwire; double v=my_fdchits[id]->vstrip; double x=S(state_x,0); double y=S(state_y,0); double tx=S(state_tx,0); double ty=S(state_ty,0); double du=x*cosa-y*sina-u; double tu=tx*cosa-ty*sina; double alpha=atan(tu); double cosalpha=cos(alpha); double sinalpha=sin(alpha); // The next measurement M(0,0)=0.; M(1,0)=v; if (fit_type==kTimeBased){ // Compute drift distance double tflight=forward_traj[k].t; double drift=DRIFT_SPEED*(my_fdchits[id]->t-tflight); drift*=(du>0?1.:-1.); // Angles of incidence to the measurement plane double phi=atan2(S(state_y,0),S(state_x,0)); double cosphi=cos(phi); double sinphi=sin(phi); // Drift distance M(0,0)=drift; // Correction for lorentz effect double nz=my_fdchits[id]->nz; double nr=my_fdchits[id]->nr; double dv=nz*drift*sinalpha*cosphi-nr*drift*cosalpha; M(1,0)=v+dv;// with correction for Lorentz effect // ... and its covariance matrix V(0,0)=anneal_factor*0.000225; V(1,1)=fdc_y_variance(alpha,drift); // Variance due to Lorentz correction double var_alpha =(C(state_tx,state_tx)*cosa*cosa+C(state_ty,state_ty)*sina*sina -2.*sina*cosa*C(state_tx,state_ty))/(1.+tu*tu)/(1.+tu*tu); double var_phi=(y*y*C(state_x,state_x)+x*x*C(state_y,state_y) -2*x*y*C(state_x,state_y))/(x*x+y*y)/(x*x+y*y); V(1,1)+=dv*dv*V(0,0)/drift/drift +drift*drift*(nz*cosalpha*cosphi+nr*sinalpha) *(nz*cosalpha*cosphi+nr*sinalpha)*var_alpha +drift*drift*nz*nz*sinalpha*sinalpha*sinphi*sinphi*var_phi; V(1,1)*=anneal_factor; } // To transform from (x,y) to (u,v), need to do a rotation: // u = x*cosa-y*sina // v = y*cosa+x*sina H(0,state_x)=H_T(state_x,0)=cosa*cosalpha; H(1,state_x)=H_T(state_x,1)=sina; H(0,state_y)=H_T(state_y,0)=-sina*cosalpha; H(1,state_y)=H_T(state_y,1)=cosa; double temp=tx*cosa-ty*sina; H(0,state_ty)=H_T(state_ty,0) =du*sina*temp/sqrt(1.+temp*temp)/(1.+temp*temp); H(0,state_tx)=H_T(state_tx,0) =-du*cosa*temp/sqrt(1.+temp*temp)/(1.+temp*temp); // Updated error matrix V1=V+H*(C*H_T); // Calculate the inverse of V double det=V1(0,0)*V1(1,1)-V1(0,1)*V1(1,0); if (det!=0){ InvV(0,0)=V1(1,1)/det; InvV(1,0)=-V1(1,0)/det; InvV(0,1)=-V1(0,1)/det; InvV(1,1)=V1(0,0)/det; } else{ cout << "Kalman filter: Singular matrix..." << endl; return UNRECOVERABLE_ERROR; } // Compute Kalman gain matrix K=C*(H_T*InvV); // Update the state vector //Mpred(0,0)=du*cos(alpha); //Mpred(1,0)= y*cosa+x*sina; Mdiff(0,0)=M(0,0)-du*cos(alpha); Mdiff(1,0)=M(1,0)-(y*cosa+x*sina); //S=S+K*(M-Mpred); S=S+K*Mdiff; //. printf("z %f Diff\n",forward_traj[k].pos.z()); //Mdiff.Print(); // Update state vector covariance matrix C=C-K*(H*C); // Residuals /* x=S(state_x,0); y=S(state_y,0); tx=S(state_tx,0); ty=S(state_ty,0); du=x*cosa-y*sina-u; R(0,0)=M(0,0)-du*cos(atan(tx*cosa-ty*sina)); R(1,0)=M(1,0)-(y*cosa+x*sina); */ R=Mdiff-(H*K)*Mdiff; R_T(0,0)=R(0,0); R_T(0,1)=R(1,0); RC=V-H*(C*H_T); my_fdchits[id]->xres=R(0,0); my_fdchits[id]->yres=R(1,0); // Calculate the inverse of RC det=RC(0,0)*RC(1,1)-RC(0,1)*RC(1,0); if (det!=0){ InvRC(0,0)=RC(1,1)/det; InvRC(1,0)=-RC(1,0)/det; InvRC(0,1)=-RC(0,1)/det; InvRC(1,1)=RC(0,0)/det; } else{ cout << "Kalman filter: Singular matrix RC..." << endl; return UNRECOVERABLE_ERROR; } // Update chi2 for this segment chisq+=(R_T*(InvRC*R))(0,0); } } chisq*=anneal_factor; // Final position for this leg x_=S(state_x,0); y_=S(state_y,0); z_=forward_traj[forward_traj.size()-1].pos.Z(); if (DEBUG_LEVEL>0) cout << "Position after forward filter: " << x_ << ", " << y_ << ", " << z_ <origin; DVector3 dir=my_cdchits[cdc_index]->dir; DVector3 wirepos=origin+((z-origin.z())/dir.z())*dir; bool more_measurements=true; // doca variables double doca=0.,old_doca=sqrt((x-wirepos.x())*(x-wirepos.x()) +(y-wirepos.y())*(y-wirepos.y())); /* printf("p %f theta %f\n",1./S(state_q_over_p,0), 90-180/M_PI*atan(1./sqrt(S(state_tx,0)*S(state_tx,0) +S(state_ty,0)*S(state_ty,0)))); */ //C.Print(); // loop over entries in the trajectory S0_=(*forward_traj_cdc[0].S); for (unsigned int k=1;kold_doca) && zstatus==0){ // Mark previous point on ref trajectory with a hit id for the straw forward_traj_cdc[k-1].h_id=cdc_index+1; // Get energy loss double dedx=0.; if (do_energy_loss){ dedx=GetdEdx(S(state_q_over_p,0),forward_traj_cdc[k].Z, forward_traj_cdc[k].A,forward_traj_cdc[k].density); } // We have bracketed the minimum doca double step_size =forward_traj_cdc[k].pos.z()-forward_traj_cdc[k-1].pos.z(); double dz=BrentsAlgorithm(z,step_size,dedx,origin,dir,S); double newz=z+dz; double ds=Step(z,newz,dedx,S); // Step reference trajectory by dz Step(z,newz,dedx,S0); // propagate error matrix to z-position of hit StepJacobian(z,newz,S0,dedx,J); C=J*(C*DMatrix(DMatrix::kTransposed,J)); // Wire position at current z wirepos=origin+((newz-origin.z())/dir.z())*dir; double xw=wirepos.x(); double yw=wirepos.y(); // predicted doca taking into account the orientation of the wire double ux=dir.x(); double uy=dir.y(); double uxuy=ux*uy; double ux2=ux*ux; double uy2=uy*uy; double dy=S(state_y,0)-yw; double dx=S(state_x,0)-xw; double d=sqrt(dx*dx*(1.-ux2)+dy*dy*(1.-uy2)-2.*dx*dy*uxuy); // Track projection if (d>0.){ H(0,state_x)=H_T(state_x,0)=(dx*(1.-ux2)-dy*uxuy)/d; H(0,state_y)=H_T(state_y,0)=(dy*(1.-uy2)-dx*uxuy)/d; } //H.Print(); // The next measurement double dm=0.; double V=1.6*1.6/12.; if (fit_type==kTimeBased){ dm=CDC_DRIFT_SPEED*(my_cdchits[cdc_index]->t-forward_traj_cdc[k].t); // variance V=0.000225*anneal; } // variance including prediction double var=V,var_pred=0.; if (d>0.){ var_pred=(H*(C*H_T))(0,0); var+=var_pred; } if (var<0.){ //cout << "Negative variance???" << endl; return VALUE_OUT_OF_RANGE; } if (DEBUG_LEVEL==2) printf("Ring %d straw %d pred %f meas %f V %f %f sig %f\n", my_cdchits[cdc_index]->ring,my_cdchits[cdc_index]->straw, d,dm,V,var,sqrt(var)); // probability double p=exp(-0.5*(dm-d)*(dm-d)/var); p=1.; //if (fabs(dm-d)/sqrt(var)>3.) dm=d+3.*sqrt(var)*(dm-d>0?1.:-1.); // Inverse of covariance matrix InvV=p/(p*V+var_pred); // Compute Kalman gain matrix K=InvV*(C*H_T); //printf("invV %f\n",InvV); //C.Print(); //K.Print(); // Update the state vector S=S+(dm-d)*K; //printf("State\n"); //S.Print(); //printf("correction to C\n"); //(K*(H*C)).Print(); // Update state vector covariance matrix C=C-K*(H*C); // doca after correction //dy=S(state_y,0)-yw; //dx=S(state_x,0)-xw; //d=sqrt(dx*dx*(1.-ux2)+dy*dy*(1.-uy2)-2.*dx*dy*uxuy); // Residual //double res=dm-d; double res=(dm-d)*(1.-(H*K)(0,0)); my_cdchits[cdc_index]->residual=res; // Update chi2 for this segment chisq+=anneal*res*res/(V-(H*(C*H_T))(0,0)); /* printf("res %f chisq contrib %f varpred %f\n",res, anneal*res*res/(V-(H*(C*H_T))(0,0)), ( H*(C*H_T))(0,0) ); */ // multiple scattering if (do_multiple_scattering){ GetProcessNoise(ds,newz,forward_traj_cdc[k].X0,S0,Q); } // Step C back to the z-position on the reference trajectory StepJacobian(newz,z,S0,dedx,J); C=J*(C*DMatrix(DMatrix::kTransposed,J))+Q; // Step S to current position on the reference trajectory Step(newz,z,dedx,S); } else { if (cdc_index>0) cdc_index--; else cdc_index=0; } // new wire origin and direction if (cdc_index>0){ cdc_index--; origin=my_cdchits[cdc_index]->origin; dir=my_cdchits[cdc_index]->dir; } else{ origin.SetXYZ(0.,0.,65.); dir.SetXYZ(0,0,1.); more_measurements=false; } // Update the wire position wirepos=origin+((z-origin.z())/dir.z())*dir; // new doca x=S(state_x,0); y=S(state_y,0); doca=sqrt((x-wirepos.x())*(x-wirepos.x()) +(y-wirepos.y())*(y-wirepos.y())); } old_doca=doca; do_energy_loss=true; } // Final position for this leg x_=S(state_x,0); y_=S(state_y,0); z_=forward_traj_cdc[forward_traj_cdc.size()-1].pos.Z(); if (DEBUG_LEVEL>0) cout << "Position after forward cdc filter: " << x_ << ", " << y_ << ", " << z_ <r2_old) dz*=-1.; // material properties double Z=0.,A=0.,density=0.,X0=0.; DVector3 pos; // current position along trajectory while (z>Z_MIN && sqrt(r2_old)<65. && zFindMat(pos,density,A,Z,X0)!=NOERROR){ _DBG_ << "Material error in ExtrapolateToVertex! " << endl; break; } // Get dEdx for the upcoming step if (do_energy_loss){ dEdx=GetdEdx(S(state_q_over_p,0),Z,A,density); } // Get the contribution to the covariance matrix due to multiple // scattering if (do_multiple_scattering) GetProcessNoise(ds,z,X0,S,Q); newz=z+dz; // Compute the Jacobian matrix StepJacobian(z,newz,S,dEdx,J); // Propagate the covariance matrix JT=DMatrix(DMatrix::kTransposed,J); C=J*(C*JT)+Q; // Step through field ds=Step(z,newz,dEdx,S); r2=S(state_x,0)*S(state_x,0)+S(state_y,0)*S(state_y,0); if (r2>r2_old){ DVector3 origin(0,0,65.); DVector3 dir(0,0,1.); // Find position of best doca to beam line //GoldenSection(newz,dz,dEdx,origin,dir,S); dz=BrentsAlgorithm(newz,dz,dEdx,origin,dir,S); Step(newz,newz+dz,dEdx,S); newz+=dz; break; } r2_old=r2; z=newz; } // update internal variables x_=S(state_x,0); y_=S(state_y,0); z_=newz; return NOERROR; } // Propagate track to point of distance of closest approach to origin jerror_t DTrackFitterKalman::ExtrapolateToVertex(DVector3 &pos, DMatrix &Sc,DMatrix &Cc){ DMatrix Jc(5,5); //.Jacobian matrix DMatrix JcT(5,5); // and its transpose DMatrix Q(5,5); // multiple scattering matrix // Initialize the beam position = center of target, and the direction DVector3 origin(0,0,65.); DVector3 dir(0,0,1.); // Position and step variables double r=pos.Perp(); // Check if we are outside the nominal beam radius if (r>BEAM_RADIUS){ double ds=-CDC_STEP_SIZE; // step along path in cm double r_old=r; Sc(state_D,0)=r; // Energy loss double dedx=0.; // Check direction of propagation DMatrix S0(5,1); S0=Sc; DVector3 pos0=pos; FixedStep(pos0,ds,S0,dedx); r=pos0.Perp(); if (r>r_old) ds*=-1.; // Track propagation loop while (Sc(state_z,0)>Z_MIN && Sc(state_z,0)FindMat(pos,density,A,Z,X0)!=NOERROR){ _DBG_ << "Material error in ExtrapolateToVertex! " << endl; break; } // Get dEdx for the upcoming step double q_over_p=Sc(state_q_over_pt,0)*cos(atan(Sc(state_tanl,0))); if (do_energy_loss){ dedx=GetdEdx(q_over_p,Z,A,density); } // Compute the Jacobian matrix StepJacobian(pos,origin,dir,ds,Sc,dedx,Jc); // Multiple scattering if (do_multiple_scattering) GetProcessNoiseCentral(ds,pos,X0,Sc,Q); // Propagate the covariance matrix JcT=DMatrix(DMatrix::kTransposed,Jc); Cc=Jc*(Cc*JcT)+Q; // Propagate the state through the field FixedStep(pos,ds,Sc,dedx); r=pos.Perp(); if (r>r_old) { // We've passed the true minimum; now use the Brent's algorithm // to find the best doca. See Numerical Recipes in C, pp 404-405 ds=BrentsAlgorithm(ds,ds,dedx,pos,origin,dir,Sc); break; } r_old=r; } } // if (r>BEAM_RADIUS) return NOERROR; } // Swim the state vector from current position (x_,y_,z_) to z_end // through the field jerror_t DTrackFitterKalman::SwimToPlane(double z_end, DMatrix &S){ int num_inc=int(fabs((z_end-z_)/STEP_SIZE)); double z=z_; double dz=(z_end>z_)?STEP_SIZE:-STEP_SIZE; double dedx=0.; double A=0.,Z=0.,density=0.,X0=0.; DVector3 pos(x_,y_,z_); for (int i=0;iFindMat(pos,density,A,Z,X0)==NOERROR)){ dedx=GetdEdx(S(state_q_over_p,0),Z,A,density); } Step(z,z_,dedx,S); z=z_; } // Final step if (do_energy_loss && (geom->FindMat(pos,density,A,Z,X0)==NOERROR)){ dedx=GetdEdx(S(state_q_over_p,0),Z,A,density); } Step(z_,z_end,dedx,S); // Update internal variables z_=z_end; x_=S(state_x,0); y_=S(state_y,0); return NOERROR; } // Swim the state vector and the covariance matrix from the current position // to the position corresponding to the radius R jerror_t DTrackFitterKalman::SwimToRadius(DVector3 &pos,double Rf,DMatrix &Sc) { double R=pos.Perp(); double ds=CDC_STEP_SIZE*((Rf>R)?1.0:-1.0); double dEdx=0.; if (ds>0){ while (RFindMat(pos,density,A,Z,X0)==NOERROR)){ double q_over_p=Sc(state_q_over_pt,0)*cos(atan(Sc(state_tanl,0))); dEdx=GetdEdx(q_over_p,Z,A,density); } // Propagate through field FixedStep(pos,ds,Sc,dEdx); // New radius R=pos.Perp(); } } else{ double R_old=R; DMatrix Sc_old(Sc); DVector3 pos_old=pos; while (R>Rf && R_old>=R){ R_old=R; Sc_old=Sc; pos_old=pos; // Get dEdx for this step double A=0.,Z=0.,density=0.,X0=0.; if (do_energy_loss && (geom->FindMat(pos,density,A,Z,X0)==NOERROR)){ double q_over_p=Sc(state_q_over_pt,0)*cos(atan(Sc(state_tanl,0))); dEdx=GetdEdx(q_over_p,Z,A,density); } // Propagate through field FixedStep(pos,ds,Sc,dEdx); // New radius R=pos.Perp(); } Sc=Sc_old; pos=pos_old; } // Update the internal variables x_=pos.x(); y_=pos.y(); z_=pos.z(); q_over_pt_=Sc(state_q_over_pt,0); phi_=Sc(state_phi,0); tanl_=Sc(state_tanl,0); tx_=cos(phi_)/tanl_; ty_=sin(phi_)/tanl_; q_over_p_=q_over_pt_*cos(atan(tanl_)); return NOERROR; } /*---------------------------------------------------------------------------- These methods are currently not being used. ----------------------------------------------------------------------------*/ // Reference trajectory for forward tracks in CDC region // At each point we store the state vector and the Jacobian needed to get to this state // along z from the previous state. C is also propagated. jerror_t DTrackFitterKalman::SetCDCForwardReferenceTrajectory(DMatrix &S,DMatrix &C){ DMatrix J(5,5),Q(5,5); DKalmanState_t temp; DMatrix S0(5,1); // Initialize some variables temp.h_id=0; temp.num_hits=0; int i=0,my_i=0,forward_traj_cdc_length=forward_traj_cdc.size(); double dEdx=0.; double z=z_,newz,ds=0.; //double my_endz=endplate_z+STEP_SIZE; double my_endz=fdc_origin[2]; if (my_fdchits.size()==0) my_endz=endplate_z; double beta2=1.,q_over_p=1.,varE=0.; // doca variables double doca,old_doca; // Loop over CDC hits for (unsigned int m=0;morigin; DVector3 dir=my_cdchits[m]->dir; // Position along wire DVector3 wirepos=origin+((z-origin.z())/dir.z())*dir; // doca old_doca=sqrt((S(state_x,0)-wirepos.x())*(S(state_x,0)-wirepos.x()) +(S(state_y,0)-wirepos.y())*(S(state_y,0)-wirepos.y())); while(zoperator()(j,0)=S(j,0); } } else{ temp.S= new DMatrix(S); } // get material properties from the Root Geometry if (geom->FindMat(temp.pos,temp.density,temp.A,temp.Z,temp.X0) !=NOERROR){ _DBG_<< "Material error!"<0. && do_energy_loss){ q_over_p=S(state_q_over_p,0); beta2=1./(1.+mass2*q_over_p*q_over_p); varE=GetEnergyVariance(ds,q_over_p,temp.Z,temp.A,temp.density); Q(state_q_over_p,state_q_over_p) =varE*q_over_p*q_over_p*q_over_p*q_over_p/beta2; } // Compute the Jacobian matrix for swimming away from target StepJacobian(z,newz,S,dEdx,J); // Update covariance matrix for swimming away from target C=J*(C*DMatrix(DMatrix::kTransposed,J))+Q; // Step through field ds=Step(z,newz,dEdx,S); len+=ds; // Get the contribution to the covariance matrix due to multiple // scattering if (do_multiple_scattering) GetProcessNoise(ds,newz,temp.X0,S,Q); // Energy loss straggling in the approximation of thick absorbers if (temp.density>0. && do_energy_loss){ q_over_p=S(state_q_over_p,0); beta2=1./(1.+mass2*q_over_p*q_over_p); varE=GetEnergyVariance(ds,q_over_p,temp.Z,temp.A,temp.density); Q(state_q_over_p,state_q_over_p) =varE*q_over_p*q_over_p*q_over_p*q_over_p/beta2; } // Compute the Jacobian matrix StepJacobian(newz,z,S,dEdx,J); // update the trajectory data if (i<=forward_traj_cdc_length){ for (unsigned int j=0;j<5;j++){ for (unsigned int k=0;k<5;k++){ forward_traj_cdc[my_i].Q->operator()(j,k)=Q(j,k); forward_traj_cdc[my_i].J->operator()(j,k)=J(j,k); } } } else{ temp.Q= new DMatrix(Q); temp.J= new DMatrix(J); forward_traj_cdc.push_front(temp); } z=newz; // update position along wire wirepos=origin+((z-my_cdchits[m]->origin.z())/dir.z())*dir; // new doca doca=sqrt((S(state_x,0)-wirepos.x())*(S(state_x,0)-wirepos.x()) +(S(state_y,0)-wirepos.y())*(S(state_y,0)-wirepos.y())); // Check if we've passed the true minimum doca... if (doca>old_doca){ // if the doca is increasing, mark the previous point as being close to the true minumum doca. // I am using this tag to indicate where the start the search in the kalman routine. if (i<=forward_traj_cdc_length){ forward_traj_cdc[my_i].h_id=m+1; } else{ forward_traj_cdc[0].h_id=m+1; } break; } old_doca=doca; } } // Continue adding to the trajectory until we have passed the endplate while(zoperator()(j,0)=S(j,0); } } else{ temp.S= new DMatrix(S); } // get material properties from the Root Geometry if (geom->FindMat(temp.pos,temp.density,temp.A,temp.Z,temp.X0) !=NOERROR){ _DBG_<<"Material error!"<0. && do_energy_loss){ q_over_p=S(state_q_over_p,0); beta2=1./(1.+mass2*q_over_p*q_over_p); varE=GetEnergyVariance(ds,q_over_p,temp.Z,temp.A,temp.density); Q(state_q_over_p,state_q_over_p) =varE*q_over_p*q_over_p*q_over_p*q_over_p/beta2; } // Compute the Jacobian matrix for swimming away from target StepJacobian(z,newz,S,dEdx,J); // Update covariance matrix for swimming away from target C=J*(C*DMatrix(DMatrix::kTransposed,J))+Q; // Step through field ds=Step(z,newz,dEdx,S); len+=ds; // Get the contribution to the covariance matrix due to multiple // scattering if (do_multiple_scattering) GetProcessNoise(ds,newz,temp.X0,S,Q); // Energy loss straggling in the approximation of thick absorbers if (temp.density>0. && do_energy_loss){ q_over_p=S(state_q_over_p,0); beta2=1./(1.+mass2*q_over_p*q_over_p); varE=GetEnergyVariance(ds,q_over_p,temp.Z,temp.A,temp.density); Q(state_q_over_p,state_q_over_p) =varE*q_over_p*q_over_p*q_over_p*q_over_p/beta2; } // Compute the Jacobian matrix StepJacobian(newz,z,S,dEdx,J); // update the trajectory if (i<=forward_traj_cdc_length){ for (unsigned int j=0;j<5;j++){ for (unsigned int k=0;k<5;k++){ forward_traj_cdc[my_i].Q->operator()(j,k)=Q(j,k); forward_traj_cdc[my_i].J->operator()(j,k)=J(j,k); } } } else{ temp.Q= new DMatrix(Q); temp.J= new DMatrix(J); forward_traj_cdc.push_front(temp); } z=newz; } /* printf("================== %d %d\n",forward_traj_cdc_length,forward_traj_cdc.size()); for (unsigned int m=0;morigin; DVector3 dir=my_cdchits[id]->dir; if (central_traj.size()>0){ // reuse existing deque // Reset D to zero Sc(state_D,0)=0.; for (int m=central_traj.size()-1;m>=0;m--){ central_traj[m].s=len; central_traj[m].pos=pos; for (unsigned int i=0;i<5;i++){ central_traj[m].S->operator()(i,0)=Sc(i,0); } // update path length len+=ds; // get material properties from the Root Geometry if(geom->FindMat(pos,central_traj[m].density,central_traj[m].A,central_traj[m].Z,central_traj[m].X0) !=NOERROR){ /* for (unsigned int m=0;m " << pt/cos(atan(tanl)) <<" s: " << setprecision(3) << central_traj[m].s << endl; } */ _DBG_<< "Material error!" <cdc_origin[2] && Sc(state_z,0)cdc_origin[2]) GetProcessNoiseCentral(ds,pos,central_traj[m].X0,Sc,Q); // Energy loss straggling in the approximation of thick absorbers if (central_traj[m].density>0. && do_energy_loss){ beta2=1./(1.+mass2*q_over_p*q_over_p); varE=GetEnergyVariance(ds,q_over_p,central_traj[m].Z, central_traj[m].A,central_traj[m].density); Q(state_q_over_pt,state_q_over_pt) =varE*Sc(state_q_over_pt,0)*Sc(state_q_over_pt,0)/beta2 *q_over_p*q_over_p; } } // Compute the Jacobian matrix for forward-tracking away from target StepJacobian(pos,origin,dir,ds,Sc,dedx,J); // Propagate the coviarance matrix Cc=J*(Cc*DMatrix(DMatrix::kTransposed,J))+Q; // Propagate the state through the field FixedStep(pos,ds,Sc,dedx); if (Sc(state_z,0)>cdc_origin[2] && Sc(state_z,0)0. && do_energy_loss){ q_over_p=Sc(state_q_over_pt,0)*cos(atan(Sc(state_tanl,0))); beta2=1./(1.+mass2*q_over_p*q_over_p); varE=GetEnergyVariance(ds,q_over_p,central_traj[m].Z, central_traj[m].A,central_traj[m].density); Q(state_q_over_pt,state_q_over_pt) =varE*Sc(state_q_over_pt,0)*Sc(state_q_over_pt,0)/beta2 *q_over_p*q_over_p; } // Multiple scattering if (do_multiple_scattering) GetProcessNoiseCentral(ds,pos,central_traj[m].X0,Sc,Q); } // Compute the Jacobian matrix for back-tracking towards target StepJacobian(pos,origin,dir,-ds,Sc,dedx,J); //central_traj[m].s=len; //central_traj[m].pos=pos; for (unsigned int i=0;i<5;i++){ //central_traj[m].S->operator()(i,0)=Sc(i,0); for (unsigned int j=0;j<5;j++){ central_traj[m].J->operator()(i,j)=J(i,j); central_traj[m].Q->operator()(i,j)=Q(i,j); } } } } // Swim out double r=pos.Perp(); while(rZ_MIN){ // Reset D to zero Sc(state_D,0)=0.; // store old position oldpos=pos; temp.pos=pos; temp.s=len; temp.S= new DMatrix(Sc); temp.density=temp.A=temp.Z=temp.X0=0.; //initialize // update path length len+=ds; // get material properties from the Root Geometry if (geom->FindMat(pos,temp.density,temp.A,temp.Z,temp.X0)!=NOERROR){ _DBG_<<"Material error!"<cdc_origin[2] && Sc(state_z,0)0. && do_energy_loss){ beta2=1./(1.+mass2*q_over_p*q_over_p); varE=GetEnergyVariance(ds,q_over_p,temp.Z,temp.A,temp.density); Q(state_q_over_pt,state_q_over_pt) =varE*Sc(state_q_over_pt,0)*Sc(state_q_over_pt,0)/beta2 *q_over_p*q_over_p; } } // Compute the Jacobian matrix for forward-tracking away from target StepJacobian(pos,origin,dir,ds,Sc,dedx,J); // Propagate the covariance matrix Cc=J*(Cc*DMatrix(DMatrix::kTransposed,J))+Q; // Propagate the state through the field FixedStep(pos,ds,Sc,dedx); if (Sc(state_z,0)>cdc_origin[2] && Sc(state_z,0)0. && do_energy_loss){ q_over_p=Sc(state_q_over_pt,0)*cos(atan(Sc(state_tanl,0))); beta2=1./(1.+mass2*q_over_p*q_over_p); varE=GetEnergyVariance(ds,q_over_p,temp.Z,temp.A,temp.density); Q(state_q_over_pt,state_q_over_pt) =varE*Sc(state_q_over_pt,0)*Sc(state_q_over_pt,0)/beta2 *q_over_p*q_over_p; } } // Compute the Jacobian matrix StepJacobian(pos,origin,dir,-ds,Sc,dedx,J); // update the radius relative to the beam line r=pos.Perp(); // Update the trajectory info temp.Q= new DMatrix(Q); temp.J= new DMatrix(J); temp.C= new DMatrix(C); central_traj.push_front(temp); } if (DEBUG_LEVEL>1) { for (unsigned int m=0;m " << pt/cos(atan(tanl)) <<" s: " << setprecision(3) << central_traj[m].s << endl; } } // State at end of swim Sc=*(central_traj[0].S); // Position at the end of the swim x_=pos.x(); y_=pos.y(); z_=pos.z(); return NOERROR; } // Reference trajectory for trajectories with hits in the forward direction // At each point we store the state vector and the Jacobian needed to get to this state // along z from the previous state. jerror_t DTrackFitterKalman::SetReferenceTrajectory(DMatrix &S,DMatrix &C){ DMatrix J(5,5),Q(5,5); DKalmanState_t temp; double ds=0.; // path length increment // Initialize some variables temp.h_id=0; temp.num_hits=0; double dEdx=0.; double beta2=1.,q_over_p=1.,varE=0.; // progress in z from hit to hit double z=z_; double newz=z; int i=0,my_i=0; int forward_traj_length=forward_traj.size(); for (unsigned int m=0;mz-z)/STEP_SIZE); newz=my_fdchits[m]->z-STEP_SIZE*double(num); if (newz-z>0.){ temp.s=len; temp.h_id=0; temp.pos.SetXYZ(S(state_x,0),S(state_y,0),z); temp.density=temp.A=temp.Z=temp.X0=0.; //initialize i++; my_i=forward_traj_length-i; if (i<=forward_traj_length){ forward_traj[my_i].s=temp.s; forward_traj[my_i].h_id=temp.h_id; forward_traj[my_i].pos=temp.pos; for (unsigned int j=0;j<5;j++){ forward_traj[my_i].S->operator()(j,0)=S(j,0); } } else{ temp.S=new DMatrix(S); } // get material properties from the Root Geometry if (geom->FindMat(temp.pos,temp.density,temp.A,temp.Z,temp.X0) !=NOERROR){ _DBG_ << "Material error! " << endl; break; } // Get dEdx for the upcoming step if (do_energy_loss){ dEdx=GetdEdx(S(state_q_over_p,0),temp.Z,temp.A,temp.density); } // scattering if (do_multiple_scattering) GetProcessNoise(ds,z,temp.X0,S,Q); // Energy loss straggling in the approximation of thick absorbers if (temp.density>0. && do_energy_loss){ q_over_p=S(state_q_over_p,0); beta2=1./(1.+mass2*q_over_p*q_over_p); varE=GetEnergyVariance(ds,q_over_p,temp.Z,temp.A,temp.density); Q(state_q_over_p,state_q_over_p) =varE*q_over_p*q_over_p*q_over_p*q_over_p/beta2; } // Compute the Jacobian matrix for swimming away from target StepJacobian(z,newz,S,dEdx,J); // Update covariance matrix for swimming away from target C=J*(C*DMatrix(DMatrix::kTransposed,J))+Q; // Step through field ds=Step(z,newz,dEdx,S); len+=ds; // Get the contribution to the covariance matrix due to multiple // scattering if (do_multiple_scattering) GetProcessNoise(ds,newz,temp.X0,S,Q); // Energy loss straggling in the approximation of thick absorbers if (temp.density>0. && do_energy_loss){ q_over_p=S(state_q_over_p,0); beta2=1./(1.+mass2*q_over_p*q_over_p); varE=GetEnergyVariance(ds,q_over_p,temp.Z,temp.A,temp.density); Q(state_q_over_p,state_q_over_p) =varE*q_over_p*q_over_p*q_over_p*q_over_p/beta2; } // Compute the Jacobian matrix StepJacobian(newz,z,S,dEdx,J); // update the trajectory data if (i<=forward_traj_length){ for (unsigned int j=0;j<5;j++){ for (unsigned int k=0;k<5;k++){ forward_traj[my_i].Q->operator()(j,k)=Q(j,k); forward_traj[my_i].J->operator()(j,k)=J(j,k); } } } else{ temp.Q=new DMatrix(Q); temp.J=new DMatrix(J); forward_traj.push_front(temp); } // update z z=newz; } for (int k=0;koperator()(j,0)=S(j,0); } } else{ temp.S=new DMatrix(S); } // get material properties from the Root Geometry if(geom->FindMat(temp.pos,temp.density,temp.A,temp.Z,temp.X0) !=NOERROR){ _DBG_<<"Material error!"<0. && do_energy_loss){ q_over_p=S(state_q_over_p,0); beta2=1./(1.+mass2*q_over_p*q_over_p); varE=GetEnergyVariance(ds,q_over_p,temp.Z,temp.A,temp.density); Q(state_q_over_p,state_q_over_p) =varE*q_over_p*q_over_p*q_over_p*q_over_p/beta2; } // Compute the Jacobian matrix for swimming away from target StepJacobian(z,newz,S,dEdx,J); // Update covariance matrix for swimming away from target C=J*(C*DMatrix(DMatrix::kTransposed,J))+Q; // Step through field ds=Step(z,newz,dEdx,S); len+=ds; // Get the contribution to the covariance matrix due to multiple // scattering if (do_multiple_scattering) GetProcessNoise(ds,newz,temp.X0,S,Q); // Energy loss straggling in the approximation of thick absorbers if (temp.density>0. && do_energy_loss){ q_over_p=S(state_q_over_p,0); beta2=1./(1.+mass2*q_over_p*q_over_p); varE=GetEnergyVariance(ds,q_over_p,temp.Z,temp.A,temp.density); Q(state_q_over_p,state_q_over_p) =varE*q_over_p*q_over_p*q_over_p*q_over_p/beta2; } // Compute the Jacobian matrix StepJacobian(newz,z,S,dEdx,J); // update the trajectory data if (i<=forward_traj_length){ for (unsigned int j=0;j<5;j++){ for (unsigned int k=0;k<5;k++){ forward_traj[my_i].Q->operator()(j,k)=Q(j,k); forward_traj[my_i].J->operator()(j,k)=J(j,k); } } } else{ temp.Q=new DMatrix(Q); temp.J=new DMatrix(J); forward_traj.push_front(temp); } temp.h_id=0; //update z z=newz; } // Lorentz correction slope parameters double tanr=0.,tanz=0.; lorentz_def->GetLorentzCorrectionParameters(S(state_x,0),S(state_y,0),z, tanz,tanr); my_fdchits[m]->nr=tanr; my_fdchits[m]->nz=tanz; temp.h_id=m+1; } // Store final point i++; my_i=forward_traj_length-i; temp.s=len; temp.pos.SetXYZ(S(state_x,0),S(state_y,0),newz); temp.density=temp.A=temp.Z=temp.X0=0.; //initialize if (i<=forward_traj_length){ forward_traj[my_i].s=temp.s; forward_traj[my_i].h_id=temp.h_id; forward_traj[my_i].pos=temp.pos; for (unsigned int j=0;j<5;j++){ forward_traj[my_i].S->operator()(j,0)=S(j,0); } } else{ temp.S=new DMatrix(S); } // get material properties from the Root Geometry if(geom->FindMat(temp.pos,temp.density,temp.A,temp.Z,temp.X0)==NOERROR){ // Get dEdx for the upcoming step if (do_energy_loss){ dEdx=GetdEdx(S(state_q_over_p,0),temp.Z,temp.A,temp.density); } // multiple scattering if (do_multiple_scattering) GetProcessNoise(ds,z,temp.X0,S,Q); // Energy loss straggling in the approximation of thick absorbers if (temp.density>0. && do_energy_loss){ q_over_p=S(state_q_over_p,0); beta2=1./(1.+mass2*q_over_p*q_over_p); varE=GetEnergyVariance(ds,q_over_p,temp.Z,temp.A,temp.density); Q(state_q_over_p,state_q_over_p) =varE*q_over_p*q_over_p*q_over_p*q_over_p/beta2; } // Compute the Jacobian matrix for swimming away from target StepJacobian(z,newz,S,dEdx,J); // Update covariance matrix for swimming away from target C=J*(C*DMatrix(DMatrix::kTransposed,J))+Q; // One more step after last hit point newz=z+STEP_SIZE; ds=Step(z,newz,dEdx,S); len+=ds; // Get the contribution to the covariance matrix due to multiple // scattering if (do_multiple_scattering) GetProcessNoise(ds,newz,temp.X0,S,Q); // Energy loss straggling in the approximation of thick absorbers if (temp.density>0. && do_energy_loss){ q_over_p=S(state_q_over_p,0); beta2=1./(1.+mass2*q_over_p*q_over_p); varE=GetEnergyVariance(ds,q_over_p,temp.Z,temp.A,temp.density); Q(state_q_over_p,state_q_over_p) =varE*q_over_p*q_over_p*q_over_p*q_over_p/beta2; } // Compute the Jacobian matrix StepJacobian(newz,z,S,dEdx,J); // update the trajectory data if (i<=forward_traj_length){ for (unsigned int j=0;j<5;j++){ for (unsigned int k=0;k<5;k++){ forward_traj[my_i].Q->operator()(j,k)=Q(j,k); forward_traj[my_i].J->operator()(j,k)=J(j,k); } } } else{ temp.Q=new DMatrix(Q); temp.J=new DMatrix(J); forward_traj.push_front(temp); } } /* printf("--------------- %d %d\n",forward_traj_length,forward_traj.size()); for (unsigned int m=0;mGetField(pos.x(),pos.y(),pos.z(), Bx, By, Bz); // Change in momentum dmom.SetXYZ(q*qBr2p*(Bz*pdir.y()-By*pdir.z()), q*qBr2p*(Bx*pdir.z()-Bz*pdir.x()), q*qBr2p*(By*pdir.x()-Bx*pdir.y())); // change in position dpos.SetXYZ(pdir.x() +q*qBr2p*ds/2.*(Bz*pdir.y()-By*pdir.z()), pdir.y() +q*qBr2p*ds/2.*(Bx*pdir.z()-Bz*pdir.x()), pdir.z() +q*qBr2p*ds/2.*(By*pdir.x()-Bx*pdir.y()) ); return NOERROR; } // Runga-Kutte for alternate parameter set {q/pT,phi,tanl(lambda),D,z} double DTrackFitterKalman::Step(DVector3 &pos, DVector3 wire_orig,DVector3 wiredir, double ds,DMatrix &S,double dEdx){ DVector3 dpos1,dpos2,dpos3,dpos4; DVector3 dmom1,dmom2,dmom3,dmom4; //Direction at current point double cosphi=cos(S(state_phi,0)); double sinphi=sin(S(state_phi,0)); // Other parameters double q=S(state_q_over_pt,0)>0?1.:-1.; double pt=fabs(1./S(state_q_over_pt,0)); DVector3 mom(pt*cosphi,pt*sinphi,pt*S(state_tanl,0)); DVector3 pdir=mom.Unit(); /* CalcDirMom(0.,dEdx,q,mom,pos,dmom1,dpos1); CalcDirMom(ds/2.,dEdx,q,mom+(ds/2.)*dmom1, pos+ds/2.*pdir+ds*ds/8.*dmom1, dmom2,dpos2); CalcDirMom(ds/2.,dEdx,q,mom+(ds/2.)*dmom2, pos+ds/2*pdir+ds*ds/8.*dmom1, dmom3,dpos3); CalcDirMom(ds,dEdx,q,mom+ds*dmom3, pos+ds*pdir+ds*ds/2.*dmom3, dmom4,dpos4);*/ double VECT[7], VOUT[7]; VECT[0] = pos.x(); VECT[1] = pos.y(); VECT[2] = pos.z(); VECT[6] = mom.Mag(); VECT[3] = fabs(1./S(state_q_over_pt,0))*cos(S(state_phi,0))/VECT[6]; VECT[4] = fabs(1./S(state_q_over_pt,0))*sin(S(state_phi,0))/VECT[6]; VECT[5] = fabs(1./S(state_q_over_pt,0))*S(state_tanl,0)/VECT[6]; grkuta_(&q, &ds, VECT, VOUT, bfield); pos.SetXYZ(VOUT[0],VOUT[1],VOUT[2]); mom.SetXYZ(VOUT[3]*VECT[6],VOUT[4]*VECT[6],VOUT[5]*VECT[6]); // New state vector pt=mom.Perp(); S(state_q_over_pt,0)=q/pt; S(state_tanl,0)=mom.z()/pt; S(state_phi,0)=mom.Phi(); S(state_z,0)=pos.z(); return ds; } // Swim the state vector and the covariance matrix from the current position // to the position corresponding to the radius R jerror_t DTrackFitterKalman::SwimToRadius(DVector3 &pos,double Rf,DMatrix &Sc, DMatrix &Cc){ DMatrix Jc(5,5),Jc_T(5,5),Q(5,5); double R=pos.Perp(); double ds=0.3; double dedx=0.; unsigned int m=my_cdchits.size()-1; while (Rorigin,my_cdchits[m]->dir,ds,Sc,dedx,Jc); Cc=Jc*(Cc*DMatrix(DMatrix::kTransposed,Jc)); FixedStep(pos,ds,Sc,dedx); // New radius R=pos.Perp(); } if (pos.z()>0 && pos.z()<200.){ double ds1 =0.5,s1=0.; // Save the current state vector and position DMatrix S0(5,1); DVector3 newpos=pos; S0=Sc; int num_iter=0; // We've gone too far, so we back track... while (fabs(R-Rf)>EPS && num_iter<10){ ds1/=2.; if (R>Rf) ds=-ds1; else ds=ds1; FixedStep(newpos,ds,S0,dedx); R=newpos.Perp(); s1+=ds; num_iter++; } // Final step StepJacobian(pos,my_cdchits[m]->origin,my_cdchits[m]->dir,s1,Sc,dedx,Jc); Cc=Jc*(Cc*DMatrix(DMatrix::kTransposed,Jc)); FixedStep(pos,s1,Sc,dedx); } // Wire position for wire#0 double x=my_cdchits[0]->origin.x() +(pos.z()-my_cdchits[0]->origin.z())*my_cdchits[0]->dir.x(); double y=my_cdchits[0]->origin.y() +(pos.z()-my_cdchits[0]->origin.z())*my_cdchits[0]->dir.y(); double dx=pos.x()-x; double dy=pos.y()-y; //B-field at (x,y,z) double Bx,By,Bz; //bfield->GetField(pos.x(),pos.y(),pos.z(), Bx, By, Bz); bfield->GetFieldBicubic(pos.x(),pos.y(),pos.z(), Bx, By, Bz); double B=sqrt(Bx*Bx+By*By+Bz*Bz); double Rc=1./qBr2p/B/Sc(state_q_over_pt,0); double cosphi=cos(Sc(state_phi,0)); double sinphi=sin(Sc(state_phi,0)); double xc=pos.x()-Rc*sinphi; double yc=pos.y()+Rc*cosphi; double rc2=(x-xc)*(x-xc)+(y-yc)*(y-yc); double sign=-Sc(state_q_over_pt,0)/fabs(Sc(state_q_over_pt,0)); if (rc2