// $Id$ // // File: DParticleID.cc // Created: Mon Feb 28 14:48:56 EST 2011 // Creator: staylor (on Linux ifarml1 2.6.18-128.el5 x86_64) // #include "DParticleID.h" #include #include #include "FCAL/DFCALGeometry.h" #include "BCAL/DBCALHit.h" #define C_EFFECTIVE 15. // start counter light propagation speed #define OUT_OF_TIME_CUT 200. // Routine for sorting dEdx data bool static DParticleID_dedx_cmp(DParticleID::dedx_t a,DParticleID::dedx_t b){ return a.dEdx < b.dEdx; } // Routine for sorting hypotheses according to FOM bool static DParticleID_hypothesis_cmp(const DTrackTimeBased *a, const DTrackTimeBased *b){ return (a->FOM>b->FOM); } bool static BCALShowerE_cmp(const DBCALShower* shower1, const DBCALShower* shower2){ return shower1->E > shower2->E; } //--------------------------------- // DParticleID (Constructor) //--------------------------------- DParticleID::DParticleID(JEventLoop *loop) { DApplication* dapp = dynamic_cast(loop->GetJApplication()); if(!dapp){ _DBG_<<"Cannot get DApplication from JEventLoop! (are you using a JApplication based program?)"<SetDefaultParameter( "PID:BCALMATCH_DZ_CUT", BCALMATCH_DZ_CUT, "maximum distance in z for track-BCAL cluster association"); gPARMS->SetDefaultParameter( "PID:BCALMATCH_SIGPHI_CUT", BCALMATCH_SIGPHI_CUT, "maximum azim. separation (in phi sigmas) for track-BCAL cluster association"); gPARMS->SetDefaultParameter( "PID:BCAL_CHSHOWER_ENVFACT", BCAL_CHSHOWER_ENVFACT, "mult. factor for phi/z acceptance of deep showers for ch. track assoc."); RootGeom = dapp->GetRootGeom(); bfield = dapp->GetBfield(); stepper= new DMagneticFieldStepper(bfield); // Get material properties for chamber gas double rho_Z_over_A_LnI=0,radlen=0; RootGeom->FindMat("CDchamberGas", dRhoZoverA_CDC, rho_Z_over_A_LnI, radlen); dLnI_CDC = rho_Z_over_A_LnI/dRhoZoverA_CDC; dKRhoZoverA_CDC = 0.1535E-3*dRhoZoverA_CDC; RootGeom->FindMat("FDchamberGas", dRhoZoverA_FDC, rho_Z_over_A_LnI, radlen); dLnI_FDC = rho_Z_over_A_LnI/dRhoZoverA_FDC; dKRhoZoverA_FDC = 0.1535E-3*dRhoZoverA_FDC; // Get the geometry geom = dapp->GetDGeometry(0); // should have run number argument... vectorsc_origin; geom->Get("//posXYZ[@volume='StartCntr']/@X_Y_Z",sc_origin); vectorsc_light_guide(3); //geom->Get("//tubs[@name='STLG']/@Rio_Z",sc_light_guide); //sc_light_guide_length=sc_light_guide[2]; vector > sc_rioz; geom->GetMultiple("//pgon[@name='STRC']/polyplane/@Rio_Z", sc_rioz); for (unsigned int k=0;k locPIDParams; if ( !loop->GetCalib("PID/photon_track_matching", locPIDParams)){ cout<<"DParticleID: loading values from PID data base"< &tracks, vector >&grouped_tracks){ if (tracks.size()==0) return RESOURCE_UNAVAILABLE; JObject::oid_t old_id=tracks[0]->candidateid; vectorhypotheses; for (unsigned int i=0;icandidateid){ // Sort according to FOM sort(hypotheses.begin(),hypotheses.end(),DParticleID_hypothesis_cmp); // Add to the grouped_tracks vector grouped_tracks.push_back(hypotheses); // Clear the hypothesis list for the next track hypotheses.clear(); } hypotheses.push_back(track); old_id=track->candidateid; } // Final set sort(hypotheses.begin(),hypotheses.end(),DParticleID_hypothesis_cmp); grouped_tracks.push_back(hypotheses); return NOERROR; } // Compute the energy losses and the path lengths in the chambers for each hit // on the track. Returns a list of dE and dx pairs with the momentum at the // hit. jerror_t DParticleID::GetDCdEdxHits(const DTrackTimeBased *track, vector& dEdxHits_CDC, vector& dEdxHits_FDC){ // Position and momentum DVector3 pos,mom; //dE and dx pairs pairde_and_dx; // We cast away the const-ness of the reference trajectory so that we can use the DisToRT method DReferenceTrajectory *my_rt=const_cast(track->rt); //Get the list of cdc hits used in the fit vectorcdchits; track->GetT(cdchits); // Loop over cdc hits for (unsigned int i=0;iDistToRT(cdchits[i]->wire); if(!((locReturnValue >= 0.0) || (locReturnValue <= 0.0))) continue; //NaN my_rt->GetLastDOCAPoint(pos, mom); // Create the dE,dx pair from the position and momentum using a helical approximation for the path // in the straw and keep track of the momentum in the active region of the detector if (CalcdEdxHit(mom,pos,cdchits[i],de_and_dx)==NOERROR) dEdxHits_CDC.push_back(dedx_t(de_and_dx.first, de_and_dx.second, mom.Mag())); } //Get the list of fdc hits used in the fit vectorfdchits; track->GetT(fdchits); // loop over fdc hits for (unsigned int i=0;iDistToRT(fdchits[i]->wire); if(!((locReturnValue >= 0.0) || (locReturnValue <= 0.0))) continue; //NaN my_rt->GetLastDOCAPoint(pos, mom); double gas_thickness = 1.0; // cm dEdxHits_FDC.push_back(dedx_t(fdchits[i]->dE, gas_thickness/cos(mom.Theta()), mom.Mag())); } // Sort the dEdx entries from smallest to largest sort(dEdxHits_FDC.begin(),dEdxHits_FDC.end(),DParticleID_dedx_cmp); sort(dEdxHits_CDC.begin(),dEdxHits_CDC.end(),DParticleID_dedx_cmp); return NOERROR; } jerror_t DParticleID::CalcDCdEdx(const DTrackTimeBased *locTrackTimeBased, double& locdEdx_FDC, double& locdx_FDC, double& locdEdx_CDC, double& locdx_CDC, unsigned int& locNumHitsUsedFordEdx_FDC, unsigned int& locNumHitsUsedFordEdx_CDC){ vector locdEdxHits_CDC, locdEdxHits_FDC; jerror_t locReturnStatus = GetDCdEdxHits(locTrackTimeBased, locdEdxHits_CDC, locdEdxHits_FDC); if(locReturnStatus != NOERROR){ locdEdx_FDC = NaN; locdx_FDC = NaN; locNumHitsUsedFordEdx_FDC = 0; locdEdx_CDC = NaN; locdx_CDC = NaN; locNumHitsUsedFordEdx_CDC = 0; return locReturnStatus; } return CalcDCdEdx(locTrackTimeBased, locdEdxHits_CDC, locdEdxHits_FDC, locdEdx_FDC, locdx_FDC, locdEdx_CDC, locdx_CDC, locNumHitsUsedFordEdx_FDC, locNumHitsUsedFordEdx_CDC); } jerror_t DParticleID::CalcDCdEdx(const DTrackTimeBased *locTrackTimeBased, const vector& locdEdxHits_CDC, const vector& locdEdxHits_FDC, double& locdEdx_FDC, double& locdx_FDC, double& locdEdx_CDC, double& locdx_CDC, unsigned int& locNumHitsUsedFordEdx_FDC, unsigned int& locNumHitsUsedFordEdx_CDC){ locdx_CDC = 0.0; locdEdx_CDC = 0.0; locNumHitsUsedFordEdx_CDC = locdEdxHits_CDC.size()/2; if(locNumHitsUsedFordEdx_CDC > 0){ for(unsigned int loc_i = 0; loc_i < locNumHitsUsedFordEdx_CDC; ++loc_i){ locdEdx_CDC += locdEdxHits_CDC[loc_i].dE; //weight is ~ #e- (scattering sites): dx! locdx_CDC += locdEdxHits_CDC[loc_i].dx; } locdEdx_CDC /= locdx_CDC; } locdx_FDC = 0.0; locdEdx_FDC = 0.0; locNumHitsUsedFordEdx_FDC = locdEdxHits_FDC.size()/2; if(locNumHitsUsedFordEdx_FDC > 0){ for(unsigned int loc_i = 0; loc_i < locNumHitsUsedFordEdx_FDC; ++loc_i){ locdEdx_FDC += locdEdxHits_FDC[loc_i].dE; //weight is ~ #e- (scattering sites): dx! locdx_FDC += locdEdxHits_FDC[loc_i].dx; } locdEdx_FDC /= locdx_FDC; //weight is dx/dx_total } return NOERROR; } // Calculate the path length for a single hit in a straw and return ds and the // energy deposition in the straw. It returns dE as the first element in the // dedx pair and ds as the second element in the dedx pair. jerror_t DParticleID::CalcdEdxHit(const DVector3 &mom, const DVector3 &pos, const DCDCTrackHit *hit, pair &dedx){ if (hit==NULL || hit->wire==NULL) return RESOURCE_UNAVAILABLE; // Track direction parameters double phi=mom.Phi(); double lambda=M_PI_2-mom.Theta(); double cosphi=cos(phi); double sinphi=sin(phi); double tanl=tan(lambda); //Position relative to wire origin double dz=pos.z()-hit->wire->origin.z(); double dx=pos.x()-hit->wire->origin.x(); double dy=pos.y()-hit->wire->origin.y(); // square of straw radius double rs2=0.776*0.776; // Useful temporary variables related to the direction of the wire double ux=hit->wire->udir.x(); double uy=hit->wire->udir.y(); double uz=hit->wire->udir.z(); double A=1.-ux*ux; double B=-2.*ux*uy; double C=-2.*ux*uz; double D=-2.*uy*uz; double E=1.-uy*uy; double F=1.-uz*uz; // 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(lambda)); // double gas_density=0.0018; // arc length and energy deposition //dedx.second=gas_density*sqrt(temp)/a/cosl; // g/cm^2 dedx.second=sqrt(temp)/a/cosl; dedx.first=hit->dE; //GeV return NOERROR; } return VALUE_OUT_OF_RANGE; } //------------------ // MatchToTOF //------------------ // Loop over TOF points, looking for minimum distance of closest approach // of track to a point in the TOF and using this to check for a match. // // NOTE: an initial guess for tproj is expected as input so that out-of-time // hits can be skipped jerror_t DParticleID::MatchToTOF(const DReferenceTrajectory *rt, DTrackFitter::fit_type_t fit_type, vector&tof_points, double &tproj, unsigned int &tof_match_id, double &locPathLength, double &locFlightTime){ //tproj=NaN; tof_match_id=0; if (tof_points.size()==0){ tproj=NaN; return RESOURCE_UNAVAILABLE; } double dmin=10000.; // loop over tof points double locTempPathLength; for (unsigned int k=0;kpos; DVector3 norm(0,0,1); DVector3 proj_pos,dir; // Find the distance of closest approach between the track trajectory // and the tof cluster position, looking for the minimum double locTempFlightTime=0.; rt->GetIntersectionWithPlane(tof_pos,norm,proj_pos,dir,&locTempPathLength,&locTempFlightTime); double d=(tof_pos-proj_pos).Mag(); // Check that the hit is not out of time with respect to the track if (fabs(tof_points[k]->t - locTempFlightTime - tproj) > OUT_OF_TIME_CUT) continue; if (dswim_steps[0].mom.Mag(); double match_cut=0.; match_cut = 6.15; //current dPositionMatchCut_DoubleEnded variable in DTOFPoint_factory.cc // if (fit_type==DTrackFitter::kTimeBased) match_cut=3.624+0.488/p; // else match_cut=4.0+0.488/p; if (dmint - locFlightTime; return NOERROR; } tproj=NaN; return VALUE_OUT_OF_RANGE; } //------------------ // MatchToBCAL //------------------ // Loop over bcal clusters, looking for minimum distance of closest approach // of track to a cluster and using this to check for a match. // // NOTE: an initial guess for tproj is expected as input so that out-of-time // hits can be skipped jerror_t DParticleID:: MatchToBCAL(const DReferenceTrajectory *rt, const vector& locInputBCALShowers, vector& locMatchedBCALShowers, double& locProjectedTime, double& locPathLength, double& locFlightTime) { if (locInputBCALShowers.size() == 0){ locProjectedTime = NaN; return RESOURCE_UNAVAILABLE; } //Loop over bcal showers double locTempPathLength, dphi, dz; double locLargestShowerEnergy = -1.0; // these not necessary using the shower sort approach below -IS //int locBestShowerMatchIndex = -1, locBestShowerMatchIndexInMatchVector = -1; for (unsigned int k = 0; k < locInputBCALShowers.size(); ++k){ // Get the BCAL cluster position and normal const DBCALShower *shower = locInputBCALShowers[k]; DVector3 bcal_pos(shower->x, shower->y, shower->z); DVector3 proj_pos; // and the bcal cluster position, looking for the minimum double locTempFlightTime = 0.0; double d = rt->DistToRTwithTime(bcal_pos, &locTempPathLength, &locTempFlightTime); if(!finite(d)) continue; // Check that the hit is not out of time with respect to the track if (fabs(locInputBCALShowers[k]->t - locTempFlightTime - locProjectedTime) > OUT_OF_TIME_CUT) continue; proj_pos = rt->GetLastDOCAPoint(); dz = proj_pos.z() - bcal_pos.z(); dphi = proj_pos.Phi() - bcal_pos.Phi(); double p = rt->swim_steps[0].mom.Mag(); double phi_sigma = BCALshowerMatch_phi_sigma(p); // The following broadens z and phi windows for showers // that don't start on BCAL entry - possible broad splashes // from center of main shower. -IS vector BCALshowerHits; shower->GetT(BCALshowerHits); bool showerOnEntry=false; for(unsigned int ind_hits=0; ind_hitslayer == 1; double ZbracketFactor=1.0; if(!showerOnEntry){ phi_sigma *= BCAL_CHSHOWER_ENVFACT; ZbracketFactor=BCAL_CHSHOWER_ENVFACT; } // Commented out in favor of a separate function -IS //dphi += 0.002 + 8.314e-3/(p + 0.3788)/(p + 0.3788); BCALshowerMatch_dphiCorr(dphi, p, rt->q); while(dphi > TMath::Pi()) dphi -= 2.0*TMath::Pi(); while(dphi < -1.0*TMath::Pi()) dphi += 2.0*TMath::Pi(); if (fabs(dz) < BCALMATCH_DZ_CUT*ZbracketFactor && fabs(dphi) < BCALMATCH_SIGPHI_CUT * phi_sigma){ locMatchedBCALShowers.push_back(shower); if(shower->E > locLargestShowerEnergy){ locLargestShowerEnergy = shower->E; locPathLength = locTempPathLength; locFlightTime = locTempFlightTime; // Unnecessary witht the sort procedure below -IS //locBestShowerMatchIndex = k; //locBestShowerMatchIndexInMatchVector = locMatchedBCALShowers.size() - 1; } } } // Opting for a (likely) more efficient sort and counting on perhaps // needing an ordered shower list later -IS sort(locMatchedBCALShowers.begin(), locMatchedBCALShowers.end(), BCALShowerE_cmp); /*if(locBestShowerMatchIndexInMatchVector > 0){ //move highest energy shower to the front of the list locMatchedBCALShowers.erase(locMatchedBCALShowers.begin() + locBestShowerMatchIndexInMatchVector); locMatchedBCALShowers.insert(locMatchedBCALShowers.begin(), locInputBCALShowers[locBestShowerMatchIndex]); }*/ if(locMatchedBCALShowers.size() > 0){ // Projected time at the target locProjectedTime = locMatchedBCALShowers[0]->t - locFlightTime; // locProjectedTime += 0.03; // correction determined from observation of simulated data //Paul Mattione return NOERROR; } locProjectedTime = NaN; return VALUE_OUT_OF_RANGE; } /** * This function computes an azimuthal span width * for matching showers to reconstructed tracks * the function: phi_sigma = a + b/p^3 has hard-coded * a and b parameters presumably optimized emperically */ inline double DParticleID::BCALshowerMatch_phi_sigma(double p){ return 0.025 + 5.8e-4/(p*p*p); } /** * This function issues a correction to the delta phi * (difference between track-projected point and BCAL shower) * presumably to correct for the magnetic field. * The charge factor been applied based on this assumption. -IS */ inline void DParticleID::BCALshowerMatch_dphiCorr(double &dphi, double p, double charge){ p += 0.3788; dphi += -charge*(0.002 + 8.314e-3/(p*p)); } //------------------ // MatchToFCAL //------------------ // Loop over fcal clusters, looking for minimum distance of closest approach // of track to a cluster and using this to check for a match. // // NOTE: an initial guess for locProjectedTime is expected as input so that out-of-time // hits can be skipped jerror_t DParticleID::MatchToFCAL(const DReferenceTrajectory *rt, const vector& locInputFCALShowers, vector& locMatchedFCALShowers, double& locProjectedTime, double& locPathLength, double& locFlightTime){ if (locInputFCALShowers.size() == 0){ locProjectedTime = NaN; return RESOURCE_UNAVAILABLE; } // Set minimum matching distance to a large default value double locLargestShowerEnergy = -1.0; // loop over clusters double locTempPathLength; int locBestShowerMatchIndex = -1, locBestShowerMatchIndexInMatchVector = -1; for (unsigned int k = 0;k < locInputFCALShowers.size(); k++){ // Get the FCAL cluster position and normal vector for the FCAL plane DVector3 fcal_pos = locInputFCALShowers[k]->getPosition(); // This is a bit of a kludge... //fcal_pos.SetZ(DFCALGeometry::fcalFaceZ()); DVector3 norm(0,0,1); DVector3 proj_pos, dir; // Find the distance of closest approach between the track trajectory // and the tof cluster position, looking for the minimum double locTempFlightTime = 0.; rt->GetIntersectionWithPlane(fcal_pos, norm, proj_pos, dir, &locTempPathLength, &locTempFlightTime); double d = (fcal_pos - proj_pos).Mag(); // Check that the hit is not out of time with respect to the track if (fabs(locInputFCALShowers[k]->getTime() - locTempFlightTime - locProjectedTime) > OUT_OF_TIME_CUT) continue; if(d < DELTA_R_FCAL){ locMatchedFCALShowers.push_back(locInputFCALShowers[k]); if(locInputFCALShowers[k]->getEnergy() > locLargestShowerEnergy){ locLargestShowerEnergy = locInputFCALShowers[k]->getEnergy(); locPathLength = locTempPathLength; locFlightTime = locTempFlightTime; locBestShowerMatchIndex = k; locBestShowerMatchIndexInMatchVector = locMatchedFCALShowers.size() - 1; } } } if(locBestShowerMatchIndexInMatchVector > 0){ //move highest energy shower to the front of the list locMatchedFCALShowers.erase(locMatchedFCALShowers.begin() + locBestShowerMatchIndexInMatchVector); locMatchedFCALShowers.insert(locMatchedFCALShowers.begin(), locInputFCALShowers[locBestShowerMatchIndex]); } // Check for a match if(locMatchedFCALShowers.size() > 0){ locProjectedTime = locMatchedFCALShowers[0]->getTime() - locFlightTime; locProjectedTime -= 2.218; // correction determined from fit to simulated data return NOERROR; } locProjectedTime = NaN; return VALUE_OUT_OF_RANGE; } //------------------ // MatchToSC //------------------ // Match track to the start counter paddles with hits. If a match // is found, use the z-position of the track projection to the start counter // planes to correct for the light propagation within the scintillator and // estimate the "vertex" time. // // Unlike the next method, this method does not use the reference trajectory. // // NOTE: an initial guess for tproj is expected as input so that out-of-time // hits can be skipped jerror_t DParticleID::MatchToSC(const DKinematicData &parms, vector&sc_hits, double &tproj,unsigned int &sc_match_id){ sc_match_id=0; if (sc_hits.size()==0){ tproj=NaN; return RESOURCE_UNAVAILABLE; } double myz=0.; double dphi_min=10000.,myphi=0.; unsigned int num=sc_norm.size()-1; DVector3 pos(parms.position()); DVector3 mom(parms.momentum()); stepper->SetCharge(parms.charge()); // Swim to barrel representing the start counter straight portion double ds=0.,myds=0; if (stepper->SwimToRadius(pos,mom,sc_pos[1].x(),&ds)){ tproj=NaN; return VALUE_OUT_OF_RANGE; } // Change the sign of ds -- most of the time we will need to add a small // amount of time to the time calculated at the start counter ds*=-1.; // Position along z and phi at intersection double proj_z=pos.z(); double proj_phi=pos.Phi(); if (proj_phi<0) proj_phi+=2.*M_PI; // Position of transition between start counter nose and leg double sc_pos1=sc_pos[1].z(); // position in z at the end of the nose double sc_posn=sc_pos[num].z(); // loop over sc hits, looking for the one with closest phi value to the // projected phi value for (unsigned int i=0;it)>OUT_OF_TIME_CUT) continue; double phi=(6.0+12.*(sc_hits[i]->sector-1))*M_PI/180.; double dphi=phi-proj_phi; // If the z position is in the nose region, match to the appropriate start // counter plane if (proj_z>sc_pos1){ double cosphi=cos(phi); double sinphi=sin(phi); for (unsigned int i=1;iSwimToPlane(mypos,mymom,plane,norm,&ds2)) continue; proj_z=mypos.z(); if (proj_zt-sc_leg_tcor; // Check position along z if (myz&sc_hits, double &tproj,unsigned int &sc_match_id, double &locPathLength, double &locFlightTime){ //tproj=NaN; sc_match_id=0; if (sc_hits.size()==0){ tproj=NaN; return RESOURCE_UNAVAILABLE; } double myz=0.; double dphi_min=10000.,myphi=0.; double locTempPathLength, locTempFlightTime = 0.0; DVector3 proj_pos; // Find intersection with a "barrel" approximation for the start counter rt->GetIntersectionWithRadius(sc_pos[1].x(),proj_pos,&locTempPathLength,&locTempFlightTime); double proj_phi=proj_pos.Phi(); if (proj_phi<0) proj_phi+=2.*M_PI; // loop over sc hits, looking for the one with closest phi value to the // projected phi value for (unsigned int i=0;it)>OUT_OF_TIME_CUT) continue; double phi=(6.0+12.*(sc_hits[i]->sector-1))*M_PI/180.; double dphi=phi-proj_phi; if (fabs(dphi)t-sc_leg_tcor; if (myzsc_pos[1].z()){ unsigned int num=sc_norm.size()-1; for (unsigned int i=1;iGetIntersectionWithPlane(pos,norm,proj_pos,&locTempPathLength,&locTempFlightTime); myz=proj_pos.z(); if (myzsc_pos[num].z()){ // Assume that the particle hit the most downstream z position of the // start counter double costheta=rt->swim_steps[0].mom.CosTheta(); double s=(sc_pos[num].z()-rt->swim_steps[0].origin.z())/costheta; double mass=rt->GetMass(); double p2=rt->swim_steps[0].mom.Mag2(); double one_over_beta=sqrt(1.+mass*mass/p2); tproj -= s*one_over_beta/SPEED_OF_LIGHT + ((sc_pos[num].z()-sc_pos1)*sc_angle_cor+sc_pos1)/C_EFFECTIVE; locFlightTime = s*one_over_beta/SPEED_OF_LIGHT; locPathLength = s; }else{ tproj-=locTempFlightTime+((myz-sc_pos1)*sc_angle_cor+sc_pos1)/C_EFFECTIVE; locFlightTime = locTempFlightTime; locPathLength = locTempPathLength; } }else{ tproj-=locTempFlightTime+myz/C_EFFECTIVE; locFlightTime = locTempFlightTime; locPathLength = locTempPathLength; } return NOERROR; } tproj=NaN; return VALUE_OUT_OF_RANGE; } Particle_t DParticleID::IDTrack(float locCharge, float locMass){ float locMassTolerance = 0.010; if (locCharge > 0.0){ // Positive particles if (fabs(locMass - ParticleMass(Proton)) < locMassTolerance) return Proton; if (fabs(locMass - ParticleMass(PiPlus)) < locMassTolerance) return PiPlus; if (fabs(locMass - ParticleMass(KPlus)) < locMassTolerance) return KPlus; if (fabs(locMass - ParticleMass(Positron)) < locMassTolerance) return Positron; if (fabs(locMass - ParticleMass(MuonPlus)) < locMassTolerance) return MuonPlus; } else if (locCharge < 0.0){ // Negative particles if (fabs(locMass - ParticleMass(PiMinus)) < locMassTolerance) return PiMinus; if (fabs(locMass - ParticleMass(KMinus)) < locMassTolerance) return KMinus; if (fabs(locMass - ParticleMass(MuonMinus)) < locMassTolerance) return MuonMinus; if (fabs(locMass - ParticleMass(Electron)) < locMassTolerance) return Electron; if (fabs(locMass - ParticleMass(AntiProton)) < locMassTolerance) return AntiProton; } else { //Neutral Track if (fabs(locMass - ParticleMass(Gamma)) < locMassTolerance) return Gamma; if (fabs(locMass - ParticleMass(Neutron)) < locMassTolerance) return Neutron; } return Unknown; }