// $Id$ // // File: DParticleCombo_factory_PreKinFit.cc // Created: Tue Aug 9 14:29:24 EST 2011 // Creator: pmatt // #include "DParticleCombo_factory_PreKinFit.h" using namespace std; using namespace jana; //------------------ // init //------------------ jerror_t DParticleCombo_factory_PreKinFit::init(void) { MAX_DParticleComboStepPoolSize = 40; MAX_DKinematicDataPoolSize = 1; MAX_DBeamPhotonPoolSize = 1; dMaxPhotonRFDeltaT = pair(false, -1.0); dMinIndividualChargedPIDFOM = pair(false, -1.0); dMinIndividualTrackingFOM = pair(false, -1.0); dMinCombinedChargedPIDFOM = pair(false, -1.0); dMinCombinedTrackingFOM = pair(false, -1.0); return NOERROR; } //------------------ // brun //------------------ jerror_t DParticleCombo_factory_PreKinFit::brun(jana::JEventLoop *locEventLoop, int runnumber) { DApplication* locApplication = dynamic_cast(locEventLoop->GetJApplication()); DGeometry* locGeometry = locApplication->GetDGeometry(runnumber); locGeometry->GetTargetZ(dTargetCenterZ); double locParamValue; locParamValue = numeric_limits::quiet_NaN(); gPARMS->SetDefaultParameter("COMBO:MAXPHOTONRFDELTAT", locParamValue); if((locParamValue > -1.0) || (locParamValue < 1.0)) { //not NaN: user set the value on the command line dMaxPhotonRFDeltaT.first = true; dMaxPhotonRFDeltaT.second = locParamValue; } locParamValue = numeric_limits::quiet_NaN(); gPARMS->SetDefaultParameter("COMBO:MININDIVIDUALCHARGEDPIDFOM", locParamValue); if((locParamValue > -1.0) || (locParamValue < 1.0)) { //not NaN: user set the value on the command line dMinIndividualChargedPIDFOM.first = true; dMinIndividualChargedPIDFOM.second = locParamValue; } locParamValue = numeric_limits::quiet_NaN(); gPARMS->SetDefaultParameter("COMBO:MININDIVIDUALTRACKINGFOM", locParamValue); if((locParamValue > -1.0) || (locParamValue < 1.0)) { //not NaN: user set the value on the command line dMinIndividualTrackingFOM.first = true; dMinIndividualTrackingFOM.second = locParamValue; } locParamValue = numeric_limits::quiet_NaN(); gPARMS->SetDefaultParameter("COMBO:MINCOMBINEDCHARGEDPIDFOM", locParamValue); if((locParamValue > -1.0) || (locParamValue < 1.0)) { //not NaN: user set the value on the command line dMinCombinedChargedPIDFOM.first = true; dMinCombinedChargedPIDFOM.second = locParamValue; } locParamValue = numeric_limits::quiet_NaN(); gPARMS->SetDefaultParameter("COMBO:MINCOMBINEDTRACKINGFOM", locParamValue); if((locParamValue > -1.0) || (locParamValue < 1.0)) { //not NaN: user set the value on the command line dMinCombinedTrackingFOM.first = true; dMinCombinedTrackingFOM.second = locParamValue; } return NOERROR; } //------------------ // evnt //------------------ jerror_t DParticleCombo_factory_PreKinFit::evnt(jana::JEventLoop *locEventLoop, int eventnumber) { dComboBlueprintStepMap.clear(); Reset_Pools(); vector locParticleComboBlueprints; locEventLoop->Get(locParticleComboBlueprints); vector locChargedTrackHypotheses_Reaction; locEventLoop->Get(locChargedTrackHypotheses_Reaction, "Reaction"); vector locNeutralParticleHypotheses; locEventLoop->Get(locNeutralParticleHypotheses); vector locBeamPhotons; locEventLoop->Get(locBeamPhotons); vector locEventRFBunches; locEventLoop->Get(locEventRFBunches); const DEventRFBunch* locEventRFBunch = locEventRFBunches[0]; DParticleCombo* locParticleCombo; DParticleComboStep* locParticleComboStep; DKinematicData* locTarget; const JObject* locSourceObject; map locTargetParticleMap; Particle_t locPID; vector locCandidatePhotons; map > locStepCloneForBeamMap; map >::iterator locIterator; for(size_t loc_i = 0; loc_i < locParticleComboBlueprints.size(); ++loc_i) { const DParticleComboBlueprint* locParticleComboBlueprint = locParticleComboBlueprints[loc_i]; locParticleCombo = new DParticleCombo(); const DReaction* locReaction = locParticleComboBlueprint->Get_Reaction(); locParticleCombo->Set_Reaction(locReaction); locParticleCombo->AddAssociatedObject(locParticleComboBlueprint); locParticleCombo->Set_KinFitResults(NULL); bool locBadComboFlag = false; for(size_t loc_j = 0; loc_j < locParticleComboBlueprint->Get_NumParticleComboBlueprintSteps(); ++loc_j) { const DParticleComboBlueprintStep* locParticleComboBlueprintStep = locParticleComboBlueprint->Get_ParticleComboBlueprintStep(loc_j); //search to see if blueprint step is a duplicate of a previous one. if so, combo step will be too! map::iterator locIterator = dComboBlueprintStepMap.find(locParticleComboBlueprintStep); if(locIterator != dComboBlueprintStepMap.end()) //identical! save it and continue { locParticleCombo->Add_ParticleComboStep(locIterator->second); continue; } locParticleComboStep = Get_ParticleComboStepResource(); locParticleComboStep->Set_ParticleComboBlueprintStep(locParticleComboBlueprintStep); //initial particle locPID = locParticleComboBlueprintStep->Get_InitialParticleID(); if(locParticleComboBlueprintStep->Get_InitialParticleID() == Gamma) //else decaying particle: nothing to set { //beam photon: will later create additional combo for each one that's within the time window, just set the first one for now if(locCandidatePhotons.empty()) { //compare photon time to RF time (at center of target) //if RF time not matched to tracks: don't cut on photon time pair locMaxPhotonRFDeltaT = dMaxPhotonRFDeltaT.first ? dMaxPhotonRFDeltaT : locReaction->Get_MaxPhotonRFDeltaT(); for(size_t loc_k = 0; loc_k < locBeamPhotons.size(); ++loc_k) { if((fabs(locBeamPhotons[loc_k]->time() - locEventRFBunch->dTime) < locMaxPhotonRFDeltaT.second) || (!locEventRFBunch->dMatchedToTracksFlag) || (!locMaxPhotonRFDeltaT.first)) locCandidatePhotons.push_back(locBeamPhotons[loc_k]); } if(locBeamPhotons.empty()) //e.g. genr8 locCandidatePhotons.push_back(Create_BeamPhoton()); } if(locCandidatePhotons.empty()) { locBadComboFlag = true; //no photons match the RF time break; } locParticleComboStep->Set_InitialParticle(locCandidatePhotons[0]); locParticleComboStep->Set_InitialParticle_Measured(locCandidatePhotons[0]); } //setup target locPID = locParticleComboBlueprintStep->Get_TargetParticleID(); if(locPID != Unknown) { if(locTargetParticleMap.find(locPID) != locTargetParticleMap.end()) locParticleComboStep->Set_TargetParticle(locTargetParticleMap[locPID]); else { locTarget = Create_Target(locPID); locParticleComboStep->Set_TargetParticle(locTarget); locTargetParticleMap[locPID] = locTarget; } } //final particles for(size_t loc_k = 0; loc_k < locParticleComboBlueprintStep->Get_NumFinalParticleSourceObjects(); ++loc_k) { const DKinematicData* locParticleData = NULL; if(locParticleComboBlueprintStep->Is_FinalParticleDetected(loc_k)) { locParticleData = Get_DetectedParticle(locReaction, locParticleComboBlueprintStep, loc_k, locChargedTrackHypotheses_Reaction, locNeutralParticleHypotheses, locSourceObject); if(locParticleData == NULL) //e.g. bad vertex-z { locBadComboFlag = true; break; } } locParticleComboStep->Add_FinalParticle(locParticleData); locParticleComboStep->Add_FinalParticle_Measured(locParticleData); } if(locBadComboFlag) //e.g. bad PID FOM break; locParticleCombo->Add_ParticleComboStep(locParticleComboStep); dComboBlueprintStepMap[locParticleComboBlueprintStep] = locParticleComboStep; } if(!locBadComboFlag) { if((!Cut_CombinedTrackingFOM(locParticleCombo)) || (!Cut_CombinedPIDFOM(locParticleCombo))) locBadComboFlag = true; } if(locBadComboFlag) //e.g. bad PID FOM { delete locParticleCombo; continue; } _data.push_back(locParticleCombo); //clone combos for additional beam photons (if needed) if((locParticleCombo->Get_ParticleComboStep(0)->Get_InitialParticleID() == Gamma) && (locCandidatePhotons.size() > 1)) { deque locNewComboSteps; locIterator = locStepCloneForBeamMap.find(locParticleCombo->Get_ParticleComboStep(0)); if(locIterator != locStepCloneForBeamMap.end()) locNewComboSteps = locIterator->second; //step cloned previously for beam photons: just use the old objects else { for(size_t loc_j = 1; loc_j < locCandidatePhotons.size(); ++loc_j) { locParticleComboStep = Clone_ParticleComboStep(locParticleCombo->Get_ParticleComboStep(0)); locParticleComboStep->Set_InitialParticle(locCandidatePhotons[loc_j]); locParticleComboStep->Set_InitialParticle_Measured(locCandidatePhotons[loc_j]); locNewComboSteps.push_back(locParticleComboStep); } locStepCloneForBeamMap[locParticleCombo->Get_ParticleComboStep(0)] = locNewComboSteps; } for(size_t loc_j = 0; loc_j < locNewComboSteps.size(); ++loc_j) { DParticleCombo* locNewParticleCombo = new DParticleCombo(*locParticleCombo); locNewParticleCombo->Set_ParticleComboStep(locNewComboSteps[loc_j], 0); _data.push_back(locNewParticleCombo); } } } return NOERROR; } DParticleComboStep* DParticleCombo_factory_PreKinFit::Clone_ParticleComboStep(const DParticleComboStep* locParticleComboStep) { DParticleComboStep* locNewParticleComboStep = Get_ParticleComboStepResource(); *locNewParticleComboStep = *locParticleComboStep; return locNewParticleComboStep; } DKinematicData* DParticleCombo_factory_PreKinFit::Create_Target(Particle_t locPID) { DKinematicData* locTarget = Get_KinematicDataResource(); locTarget->setPID(locPID); locTarget->setCharge(ParticleCharge(locPID)); locTarget->setMomentum(DVector3(0.0, 0.0, 0.0)); locTarget->setMass(ParticleMass(locPID)); return locTarget; } DBeamPhoton* DParticleCombo_factory_PreKinFit::Create_BeamPhoton(void) //for MC only! { DBeamPhoton* locBeamPhoton = Get_BeamPhotonResource(); double locPhotonEnergy = 9.0; Particle_t locPID = Gamma; locBeamPhoton->setPID(locPID); locBeamPhoton->setCharge(ParticleCharge(locPID)); locBeamPhoton->setMomentum(DVector3(0.0, 0.0, locPhotonEnergy)); locBeamPhoton->setPosition(DVector3(0.0, 0.0, dTargetCenterZ)); locBeamPhoton->setMass(ParticleMass(locPID)); return locBeamPhoton; } const DKinematicData* DParticleCombo_factory_PreKinFit::Get_DetectedParticle(const DReaction* locReaction, const DParticleComboBlueprintStep* locParticleComboBlueprintStep, size_t locParticleIndex, vector& locChargedTrackHypotheses_Reaction, vector& locNeutralParticleHypotheses, const JObject*& locSourceObject) { locSourceObject = NULL; Particle_t locPID = locParticleComboBlueprintStep->Get_FinalParticleID(locParticleIndex); int locCharge = ParticleCharge(locPID); locSourceObject = locParticleComboBlueprintStep->Get_FinalParticle_SourceObject(locParticleIndex); if(locSourceObject == NULL) return NULL; //decaying or missing if(locCharge == 0) { const DNeutralShower* locNeutralShower = static_cast(locSourceObject); vector locNeutralShowers; for(size_t loc_i = 0; loc_i < locNeutralParticleHypotheses.size(); ++loc_i) { if(locNeutralParticleHypotheses[loc_i]->PID() != locPID) continue; locNeutralParticleHypotheses[loc_i]->GetT(locNeutralShowers); if(locNeutralShowers[0] != locNeutralShower) continue; return static_cast(locNeutralParticleHypotheses[loc_i]); } return NULL; //uh oh! } const DChargedTrack* locChargedTrack = static_cast(locSourceObject); //check if pid already stored for this track const DChargedTrackHypothesis* locChargedTrackHypothesis = locChargedTrack->Get_Hypothesis(locPID); if(locChargedTrackHypothesis != NULL) return static_cast(locChargedTrackHypothesis); //check to see if the charged track was generated by the tag="Reaction" hypo factory vector locChargedTracks; for(size_t loc_i = 0; loc_i < locChargedTrackHypotheses_Reaction.size(); ++loc_i) { locChargedTrackHypothesis = locChargedTrackHypotheses_Reaction[loc_i]; if(locChargedTrackHypothesis->PID() != locPID) continue; locChargedTrackHypothesis->GetT(locChargedTracks); if(locChargedTracks[0] != locChargedTrack) continue; //check to make sure the PID and/or tracking FOM isn't way off (save time/mem) if((!Cut_PIDFOM(locReaction, locChargedTrackHypothesis)) || (!Cut_TrackingFOM(locReaction, locChargedTrackHypothesis))) return NULL; return static_cast(locChargedTrackHypothesis); } return NULL; //uh oh! } void DParticleCombo_factory_PreKinFit::Reset_Pools(void) { // delete pool sizes if too large, preventing memory-leakage-like behavor. if(dParticleComboStepPool_All.size() > MAX_DParticleComboStepPoolSize){ for(size_t loc_i = MAX_DParticleComboStepPoolSize; loc_i < dParticleComboStepPool_All.size(); ++loc_i) delete dParticleComboStepPool_All[loc_i]; dParticleComboStepPool_All.resize(MAX_DParticleComboStepPoolSize); } dParticleComboStepPool_Available = dParticleComboStepPool_All; if(dKinematicDataPool_All.size() > MAX_DKinematicDataPoolSize){ for(size_t loc_i = MAX_DKinematicDataPoolSize; loc_i < dKinematicDataPool_All.size(); ++loc_i) delete dKinematicDataPool_All[loc_i]; dKinematicDataPool_All.resize(MAX_DKinematicDataPoolSize); } dKinematicDataPool_Available = dKinematicDataPool_All; if(dBeamPhotonPool_All.size() > MAX_DBeamPhotonPoolSize){ for(size_t loc_i = MAX_DBeamPhotonPoolSize; loc_i < dBeamPhotonPool_All.size(); ++loc_i) delete dBeamPhotonPool_All[loc_i]; dBeamPhotonPool_All.resize(MAX_DBeamPhotonPoolSize); } dBeamPhotonPool_Available = dBeamPhotonPool_All; } DParticleComboStep* DParticleCombo_factory_PreKinFit::Get_ParticleComboStepResource(void) { DParticleComboStep* locParticleComboStep; if(dParticleComboStepPool_Available.empty()) { locParticleComboStep = new DParticleComboStep; dParticleComboStepPool_All.push_back(locParticleComboStep); } else { locParticleComboStep = dParticleComboStepPool_Available.back(); locParticleComboStep->Reset(); dParticleComboStepPool_Available.pop_back(); } return locParticleComboStep; } void DParticleCombo_factory_PreKinFit::Reset_KinematicData(DKinematicData* locKinematicData) { locKinematicData->setPID(Unknown); locKinematicData->setMassFixed(); locKinematicData->setCharge(0); locKinematicData->setMass(0.0); locKinematicData->setMomentum(DVector3()); locKinematicData->setPosition(DVector3()); locKinematicData->setTime(0.0); locKinematicData->setdEdx(0.0); locKinematicData->setPathLength(0.0, 0.0); locKinematicData->setTrackingStateVector(0.0, 0.0, 0.0, 0.0, 0.0); locKinematicData->setT0(0.0, 0.0, SYS_NULL); locKinematicData->setT1(0.0, 0.0, SYS_NULL); locKinematicData->clearErrorMatrix(); locKinematicData->clearTrackingErrorMatrix(); } DBeamPhoton* DParticleCombo_factory_PreKinFit::Get_BeamPhotonResource(void) { DBeamPhoton* locBeamPhoton; if(dBeamPhotonPool_Available.empty()) { locBeamPhoton = new DBeamPhoton; dBeamPhotonPool_All.push_back(locBeamPhoton); } else { locBeamPhoton = dBeamPhotonPool_Available.back(); Reset_KinematicData(static_cast(locBeamPhoton)); locBeamPhoton->ClearAssociatedObjects(); dBeamPhotonPool_Available.pop_back(); } return locBeamPhoton; } DKinematicData* DParticleCombo_factory_PreKinFit::Get_KinematicDataResource(void) { DKinematicData* locKinematicData; if(dKinematicDataPool_Available.empty()) { locKinematicData = new DKinematicData; dKinematicDataPool_All.push_back(locKinematicData); } else { locKinematicData = dKinematicDataPool_Available.back(); Reset_KinematicData(locKinematicData); locKinematicData->ClearAssociatedObjects(); dKinematicDataPool_Available.pop_back(); } return locKinematicData; } bool DParticleCombo_factory_PreKinFit::Cut_TrackingFOM(const DReaction* locReaction, const DChargedTrackHypothesis* locChargedTrackHypothesis) const { pair locMinIndividualTrackingFOM = dMinIndividualTrackingFOM.first ? dMinIndividualTrackingFOM : locReaction->Get_MinIndividualTrackingFOM(); if(!locMinIndividualTrackingFOM.first) return true; double locFOM = TMath::Prob(locChargedTrackHypothesis->dChiSq_Track, locChargedTrackHypothesis->dNDF_Track); return ((locChargedTrackHypothesis->dNDF_Track == 0) ? true : (locFOM >= locMinIndividualTrackingFOM.second)); } bool DParticleCombo_factory_PreKinFit::Cut_PIDFOM(const DReaction* locReaction, const DChargedTrackHypothesis* locChargedTrackHypothesis) const { pair locMinIndividualChargedPIDFOM = dMinIndividualChargedPIDFOM.first ? dMinIndividualChargedPIDFOM : locReaction->Get_MinIndividualChargedPIDFOM(); if(!locMinIndividualChargedPIDFOM.first) return true; return ((locChargedTrackHypothesis->dNDF == 0) ? true : (locChargedTrackHypothesis->dFOM >= locMinIndividualChargedPIDFOM.second)); } bool DParticleCombo_factory_PreKinFit::Cut_CombinedPIDFOM(const DParticleCombo* locParticleCombo) const { const DReaction* locReaction = locParticleCombo->Get_Reaction(); pair locMinCombinedChargedPIDFOM = dMinCombinedChargedPIDFOM.first ? dMinCombinedChargedPIDFOM : locReaction->Get_MinCombinedChargedPIDFOM(); if(!locMinCombinedChargedPIDFOM.first) return true; unsigned int locTotalPIDNDF = 0; double locTotalPIDChiSq = 0.0; deque locDetectedChargedParticles; locParticleCombo->Get_DetectedFinalChargedParticles_Measured(locDetectedChargedParticles); for(size_t loc_i = 0; loc_i < locDetectedChargedParticles.size(); ++loc_i) { const DChargedTrackHypothesis* locChargedTrackHypothesis = dynamic_cast(locDetectedChargedParticles[loc_i]); locTotalPIDNDF += locChargedTrackHypothesis->dNDF; locTotalPIDChiSq += locChargedTrackHypothesis->dChiSq; } double locFOM = TMath::Prob(locTotalPIDChiSq, locTotalPIDNDF); return (locFOM >= locMinCombinedChargedPIDFOM.second); } bool DParticleCombo_factory_PreKinFit::Cut_CombinedTrackingFOM(const DParticleCombo* locParticleCombo) const { const DReaction* locReaction = locParticleCombo->Get_Reaction(); pair locMinCombinedTrackingFOM = dMinCombinedTrackingFOM.first ? dMinCombinedTrackingFOM : locReaction->Get_MinCombinedTrackingFOM(); if(!locMinCombinedTrackingFOM.first) return true; unsigned int locTotalTrackingNDF = 0; double locTotalTrackingChiSq = 0.0; deque locDetectedChargedParticles; locParticleCombo->Get_DetectedFinalChargedParticles_Measured(locDetectedChargedParticles); for(size_t loc_i = 0; loc_i < locDetectedChargedParticles.size(); ++loc_i) { const DChargedTrackHypothesis* locChargedTrackHypothesis = dynamic_cast(locDetectedChargedParticles[loc_i]); locTotalTrackingNDF += locChargedTrackHypothesis->dNDF_Track; locTotalTrackingChiSq += locChargedTrackHypothesis->dChiSq_Track; } double locFOM = TMath::Prob(locTotalTrackingChiSq, locTotalTrackingNDF); return (locFOM >= locMinCombinedTrackingFOM.second); } //------------------ // erun //------------------ jerror_t DParticleCombo_factory_PreKinFit::erun(void) { return NOERROR; } //------------------ // fini //------------------ jerror_t DParticleCombo_factory_PreKinFit::fini(void) { for(size_t loc_i = 0; loc_i < dParticleComboStepPool_All.size(); ++loc_i) delete dParticleComboStepPool_All[loc_i]; for(size_t loc_i = 0; loc_i < dKinematicDataPool_All.size(); ++loc_i) delete dKinematicDataPool_All[loc_i]; for(size_t loc_i = 0; loc_i < dBeamPhotonPool_All.size(); ++loc_i) delete dBeamPhotonPool_All[loc_i]; return NOERROR; }