// $Id$ // // File: DTrackTimeBased_factory.cc // Created: Thu Sep 4 14:02:44 EDT 2008 // Creator: davidl (on Darwin harriet.jlab.org 8.11.1 i386) // #include #include using namespace std; #include "DTrackTimeBased_factory.h" #include #include #include #include using namespace jana; //------------------ // init //------------------ jerror_t DTrackTimeBased_factory::init(void) { fitter = NULL; DEBUG_LEVEL = 0; MOMENTUM_CUT_FOR_DEDX=0.5; gPARMS->SetDefaultParameter("TRKFIT:DEBUG_LEVEL", DEBUG_LEVEL); gPARMS->SetDefaultParameter("TRKFIT:MOMENTUM_CUT_FOR_DEDX",MOMENTUM_CUT_FOR_DEDX); return NOERROR; } //------------------ // brun //------------------ jerror_t DTrackTimeBased_factory::brun(jana::JEventLoop *loop, int runnumber) { // Get pointer to TrackFitter object that actually fits a track vector fitters; loop->Get(fitters); if(fitters.size()<1){ _DBG_<<"Unable to get a DTrackFitter object! NO Charged track fitting will be done!"<(fitters[0]); // Warn user if something happened that caused us NOT to get a fitter object pointer if(!fitter){ _DBG_<<"Unable to get a DTrackFitter object! NO Charged track fitting will be done!"<SetDefaultParameter("TRKFIT:MASS_HYPOTHESES", MASS_HYPOTHESES); // Reserve the first mass hypothesis slot for the one decided on by the wire-base fit mass_hypotheses.push_back(0.0); // Parse MASS_HYPOTHESES string to make list of masses to try if(MASS_HYPOTHESES.length()>0){ string &str = MASS_HYPOTHESES; unsigned int cutAt; while( (cutAt = str.find(",")) != (unsigned int)str.npos ){ if(cutAt > 0)mass_hypotheses.push_back(atof(str.substr(0,cutAt).c_str())); str = str.substr(cutAt+1); } if(str.length() > 0)mass_hypotheses.push_back(atof(str.c_str())); } return NOERROR; } //------------------ // evnt //------------------ jerror_t DTrackTimeBased_factory::evnt(JEventLoop *loop, int eventnumber) { if(!fitter)return NOERROR; // Get candidates and hits vector tracks; loop->Get(tracks); // Deallocate some reference trajectories occasionally unsigned int rts_to_keep = 5; if(tracks.size()>rts_to_keep)rts_to_keep=tracks.size(); for(unsigned int i=rts_to_keep; iGetDMagneticFieldMap())); DReferenceTrajectory *rt = rtv[_data.size()]; // Copy in the mass from the wire-based fit as the first one to try mass_hypotheses[0] = track->mass(); // Loop over potential particle masses until one is found that gives a chisq/Ndof<3.0 // If none does, then use the one with the smallest chisq DTrackTimeBased *best_track = NULL; double best_fom = 0.0; for(unsigned int j=0; j1.0E-5)break; // Don't try the same mass twice! if(j!=0 && mass_hypotheses[j]==mass_hypotheses[0])continue; if(DEBUG_LEVEL>1){_DBG__;_DBG_<<"---- Starting time based fit with mass: "<SetFitType(DTrackFitter::kTimeBased); DTrackFitter::fit_status_t status = fitter->FindHitsAndFitTrack(*track, rt, loop, mass_hypotheses[j]); DTrackTimeBased *dtrack = NULL; switch(status){ case DTrackFitter::kFitNotDone: _DBG_<<"Fitter returned kFitNotDone. This should never happen!!"<Ndof < 1){ if(DEBUG_LEVEL>1)_DBG_<<"-- new track with mass "<1)_DBG_<<"-- first successful fit this candidate with mass: "< best_fom){ if(DEBUG_LEVEL>1)_DBG_<<"-- new best track with mass "<1)_DBG_<<"-- keeping best track with mass "<mass()<<" (old chisq/Ndof="<<(best_track->chisq/best_track->Ndof)<<" , new chisq/Ndof="<<(dtrack->chisq/dtrack->Ndof)<<") (old fom="<GetDMagneticFieldMap())); DReferenceTrajectory *rt = rtv[_data.size()]; DTrackTimeBased *timebased_track = new DTrackTimeBased; // Copy over DKinematicData part DKinematicData *track_kd = timebased_track; *track_kd = fitter->GetFitParameters(); rt->SetMass(track_kd->mass()); rt->Swim(timebased_track->position(), timebased_track->momentum(), timebased_track->charge()); timebased_track->rt = rt; timebased_track->chisq = fitter->GetChisq(); timebased_track->Ndof = fitter->GetNdof(); timebased_track->trackid = track->id; // Add hits used as associated objects const vector &cdchits = fitter->GetCDCFitHits(); const vector &fdchits = fitter->GetFDCFitHits(); for(unsigned int i=0; iAddAssociatedObject(cdchits[i]); for(unsigned int i=0; iAddAssociatedObject(fdchits[i]); // Add DTrack object as associate object timebased_track->AddAssociatedObject(track); return timebased_track; } //------------------ // GetFOM //------------------ double DTrackTimeBased_factory::GetFOM(DTrackTimeBased *dtrack) { double dedx,mean_path_length,p_avg; unsigned int num_hits=0; if (fitter->GetdEdx(dtrack->rt,dedx,mean_path_length,p_avg,num_hits) ==NOERROR){ dtrack->setdEdx(dedx); double dedx_sigma=fitter->GetdEdxSigma(num_hits,mean_path_length); double dedx_most_probable=fitter->GetdEdx(p_avg,dtrack->rt->GetMass(),mean_path_length); //figure of merit return ( dedx_sigma/fabs(dedx/dedx_most_probable-1.) ); } // If we got here, GetdEdx failed for this track dtrack->setdEdx(0.); return 0.; } //------------------ // GetRangeOutFOM //------------------ double DTrackTimeBased_factory::GetRangeOutFOM(DTrackTimeBased *dtrack) { /// Calculate a figure of merit for the track ranging out within the /// detector. If the particle does range out (lose all of its energy) /// then the value returned would be essentially zero. If it does not, /// then the value will be greater than zero. /// /// The FOM is ratio of the total pathlength of the track to the /// pathlength to the outermost wire associated with the track. /// Therefore, FOM=1 corresponds to a track that goes just /// as far after it hit the last wire as before, before finally /// hitting the BCAL, FCAL, etc... // We want the pathlength to the last wire that is on this track // (as determined by the DParticleHitSelector??? class). Since the // tracks can curl back in toward the beamline, we have to check every // wire to see what the pathlength to it is // Need the reference trajectory to find pathlengths DReferenceTrajectory *rt = const_cast(dtrack->rt); // Look first at FDC hits vector fdchits; dtrack->Get(fdchits); const DCoordinateSystem *outermost_wire = NULL; double s_to_outermost_wire=-1.0; for(unsigned int i=0; iFindClosestSwimStep(fdchits[i]->wire); if(step->s > s_to_outermost_wire){ s_to_outermost_wire = step->s; outermost_wire = fdchits[i]->wire; } } // Check CDC if no FDC wire was found if(!outermost_wire){ vector cdchits; dtrack->Get(cdchits); for(unsigned int i=0; iFindClosestSwimStep(cdchits[i]->wire); if(step->s > s_to_outermost_wire){ s_to_outermost_wire = step->s; outermost_wire = cdchits[i]->wire; } } } // Make sure *a* wire was found. (This is just a dummy check) if(!outermost_wire){ _DBG_<<"ERROR: No outermost wire found for track!"<rt->swim_steps[dtrack->rt->Nswim_steps-1].s; // Calculate figure of merit double fom = (total_s - s_to_outermost_wire)/s_to_outermost_wire; return fom; }