// $Id$ // // File: DMCThrownMatching_factory.cc // Created: Tue Aug 9 14:29:24 EST 2011 // Creator: pmatt (on Linux ifarml6 2.6.18-128.el5 x86_64) // #include "DMCThrownMatching_factory.h" #include "TMath.h" //------------------ // init //------------------ jerror_t DMCThrownMatching_factory::init(void) { dMinimumMatchFOM = -1.0; dDebugLevel = 0; dMaximumTOFMatchDistance = 10.0; //cm dMaximumFCALMatchDistance = 10.0; //cm dMaximumBCALMatchAngleDegrees = 5.0; dTargetCenter = 65.0; //set me from geometry! return NOERROR; } //------------------ // brun //------------------ jerror_t DMCThrownMatching_factory::brun(jana::JEventLoop* locEventLoop, int runnumber) { gPARMS->SetDefaultParameter("MCMATCH:MINMATCHFOM", dMinimumMatchFOM); gPARMS->SetDefaultParameter("MCMATCH:DEBUGLEVEL", dDebugLevel); gPARMS->SetDefaultParameter("MCMATCH:MAXMATCHTOFDISTANCE", dMaximumTOFMatchDistance); gPARMS->SetDefaultParameter("MCMATCH:MAXMATCHFCALDISTANCE", dMaximumFCALMatchDistance); gPARMS->SetDefaultParameter("MCMATCH:MAXMATCHBCALANGLE", dMaximumBCALMatchAngleDegrees); return NOERROR; } //------------------ // evnt //------------------ jerror_t DMCThrownMatching_factory::evnt(jana::JEventLoop* locEventLoop, int eventnumber) { vector locMCThrowns; locEventLoop->Get(locMCThrowns, "FinalState"); vector locOriginalMCThrowns_Charged; vector locOriginalMCThrowns_Neutral; for(size_t loc_i = 0; loc_i < locMCThrowns.size(); ++loc_i) { if(ParticleCharge((Particle_t)locMCThrowns[loc_i]->type) == 0) locOriginalMCThrowns_Neutral.push_back(locMCThrowns[loc_i]); else locOriginalMCThrowns_Charged.push_back(locMCThrowns[loc_i]); } if(dDebugLevel > 0) cout << "input #thrown, ok charged # thrown, ok neutral # thrown = " << locMCThrowns.size() << ", " << locOriginalMCThrowns_Charged.size() << ", " << locOriginalMCThrowns_Neutral.size() << endl; if(locMCThrowns.empty()) return NOERROR; vector locChargedTracks; locEventLoop->Get(locChargedTracks); vector locNeutralParticles; locEventLoop->Get(locNeutralParticles); vector locChargedTrackHypotheses; locEventLoop->Get(locChargedTrackHypotheses); vector locNeutralParticleHypotheses; locEventLoop->Get(locNeutralParticleHypotheses); DMCThrownMatching* locMCThrownMatching = new DMCThrownMatching(); Find_GenReconMatches_ChargedHypo(locOriginalMCThrowns_Charged, locChargedTrackHypotheses, locMCThrownMatching); Find_GenReconMatches_ChargedTrack(locOriginalMCThrowns_Charged, locChargedTracks, locMCThrownMatching); Find_GenReconMatches_NeutralHypo(locOriginalMCThrowns_Neutral, locNeutralParticleHypotheses, locMCThrownMatching); Find_GenReconMatches_NeutralParticle(locOriginalMCThrowns_Neutral, locNeutralParticles, locMCThrownMatching); Find_GenReconMatches_TOFPoints(locEventLoop, locMCThrownMatching); Find_GenReconMatches_BCALShowers(locEventLoop, locMCThrownMatching); Find_GenReconMatches_FCALShowers(locEventLoop, locMCThrownMatching); if(dDebugLevel > 0) { cout << "Charged Track Matching Summary:" << endl; for(size_t loc_i = 0; loc_i < locChargedTrackHypotheses.size(); ++loc_i) { double locP = locChargedTrackHypotheses[loc_i]->momentum().Mag(); double locTheta = locChargedTrackHypotheses[loc_i]->momentum().Theta()*180.0/TMath::Pi(); double locPhi = locChargedTrackHypotheses[loc_i]->momentum().Phi()*180.0/TMath::Pi(); Particle_t locPID = locChargedTrackHypotheses[loc_i]->PID(); cout << "charged info: " << locChargedTrackHypotheses[loc_i]->candidateid << ", " << ParticleType(locPID) << ", " << locP << ", " << locTheta << ", " << locPhi << endl; cout << "matched info: "; const DMCThrown* locMCThrown = locMCThrownMatching->Get_MatchingMCThrown(locChargedTrackHypotheses[loc_i]); if(locMCThrown == NULL) { cout << "NO MATCH." << endl; continue; } locP = locMCThrown->momentum().Mag(); locTheta = locMCThrown->momentum().Theta()*180.0/TMath::Pi(); locPhi = locMCThrown->momentum().Phi()*180.0/TMath::Pi(); locPID = locMCThrown->PID(); cout << ParticleType(locPID) << ", " << locP << ", " << locTheta << ", " << locPhi << endl; } cout << "Unmatched Charged DMCThrowns:" << endl; for(size_t loc_i = 0; loc_i < locMCThrowns.size(); ++loc_i) { if(ParticleCharge(locMCThrowns[loc_i]->PID()) == 0) continue; if(locMCThrownMatching->Get_MatchingChargedTrack(locMCThrowns[loc_i]) != NULL) continue; double locP = locMCThrowns[loc_i]->momentum().Mag(); double locTheta = locMCThrowns[loc_i]->momentum().Theta()*180.0/TMath::Pi(); double locPhi = locMCThrowns[loc_i]->momentum().Phi()*180.0/TMath::Pi(); Particle_t locPID = locMCThrowns[loc_i]->PID(); cout << "thrown info: " << ParticleType(locPID) << ", " << locP << ", " << locTheta << ", " << locPhi << endl; } } _data.push_back(locMCThrownMatching); return NOERROR; } void DMCThrownMatching_factory::Find_GenReconMatches_FCALShowers(JEventLoop* locEventLoop, DMCThrownMatching* locMCThrownMatching) const { vector locFCALTruthShowers; const DFCALTruthShower* locFCALTruthShower; locEventLoop->Get(locFCALTruthShowers); vector locFCALShowers; const DFCALShower* locFCALShower; locEventLoop->Get(locFCALShowers); map locFCALShowerToTruthMap; map locFCALTruthToShowerMap; size_t locBestFCALShowerIndex = 0, locBestFCALTruthShowerIndex = 0; double locMatchDistance; while((!locFCALShowers.empty()) && (!locFCALTruthShowers.empty())) { bool locMatchFoundFlag = false; double locLargestEnergy = 0.0; for(size_t loc_i = 0; loc_i < locFCALTruthShowers.size(); ++loc_i) { locFCALTruthShower = locFCALTruthShowers[loc_i]; for(size_t loc_j = 0; loc_j < locFCALShowers.size(); ++loc_j) { locFCALShower = locFCALShowers[loc_j]; DVector3 locFCALShowerPosition = locFCALShower->getPosition(); //propagate truth shower from the FCAL face to the depth of the measured shower double locDeltaZ = locFCALShowerPosition.Z() - locFCALTruthShower->z(); DVector3 locTrueMomentum(locFCALTruthShower->px(), locFCALTruthShower->py(), locFCALTruthShower->pz()); double locDeltaPathLength = locDeltaZ/cos(locTrueMomentum.Theta()); double locTrueX = locFCALTruthShower->x() + locDeltaPathLength*sin(locTrueMomentum.Theta())*cos(locTrueMomentum.Phi()); double locTrueY = locFCALTruthShower->y() + locDeltaPathLength*sin(locTrueMomentum.Theta())*sin(locTrueMomentum.Phi()); DVector3 locFCALTruthShowerPosition(locTrueX, locTrueY, locFCALShowerPosition.Z()); locMatchDistance = (locFCALShowerPosition - locFCALTruthShowerPosition).Mag(); if(locMatchDistance > dMaximumFCALMatchDistance) continue; //keep the shower with the largest energy if(locFCALShower->getEnergy() > locLargestEnergy) { locMatchFoundFlag = true; locLargestEnergy = locFCALShower->getEnergy(); locBestFCALTruthShowerIndex = loc_i; locBestFCALShowerIndex = loc_j; } } } if(!locMatchFoundFlag) //no more good matches!1 break; locFCALTruthShower = locFCALTruthShowers[locBestFCALTruthShowerIndex]; locFCALShower = locFCALShowers[locBestFCALShowerIndex]; locFCALShowerToTruthMap[locFCALShower] = locFCALTruthShower; locFCALTruthToShowerMap[locFCALTruthShower] = locFCALShower; locFCALShowers.erase(locFCALShowers.begin() + locBestFCALShowerIndex); locFCALTruthShowers.erase(locFCALTruthShowers.begin() + locBestFCALTruthShowerIndex); } locMCThrownMatching->Set_FCALShowerToTruthMap(locFCALShowerToTruthMap); locMCThrownMatching->Set_FCALTruthToShowerMap(locFCALTruthToShowerMap); } void DMCThrownMatching_factory::Find_GenReconMatches_BCALShowers(JEventLoop* locEventLoop, DMCThrownMatching* locMCThrownMatching) const { vector locBCALTruthShowers; const DBCALTruthShower* locBCALTruthShower; locEventLoop->Get(locBCALTruthShowers); vector locBCALShowers; const DBCALShower* locBCALShower; locEventLoop->Get(locBCALShowers); map locBCALShowerToTruthMap; map locBCALTruthToShowerMap; vector locMCThrowns; locEventLoop->Get(locMCThrowns); //map of truth shower to dmcthrown map locBCALTruthToMCThrownMap; for(size_t loc_i = 0; loc_i < locBCALTruthShowers.size(); ++loc_i) { for(size_t loc_j = 0; loc_j < locMCThrowns.size(); ++loc_j) { if(locBCALTruthShowers[loc_i]->track != locMCThrowns[loc_j]->myid) continue; locBCALTruthToMCThrownMap[locBCALTruthShowers[loc_i]] = locMCThrowns[loc_j]; break; } } size_t locBestBCALShowerIndex = 0, locBestBCALTruthShowerIndex = 0; while((!locBCALShowers.empty()) && (!locBCALTruthShowers.empty())) { bool locMatchFoundFlag = false; double locLargestEnergy = 0.0; for(size_t loc_i = 0; loc_i < locBCALTruthShowers.size(); ++loc_i) { locBCALTruthShower = locBCALTruthShowers[loc_i]; for(size_t loc_j = 0; loc_j < locBCALShowers.size(); ++loc_j) { locBCALShower = locBCALShowers[loc_j]; DVector3 locBCALShowerPosition(locBCALShower->x, locBCALShower->y, locBCALShower->z); DVector3 locBCALTruthShowerPosition(locBCALTruthShower->r*cos(locBCALTruthShower->phi), locBCALTruthShower->r*sin(locBCALTruthShower->phi), locBCALTruthShower->z); //compare theta & phi map::const_iterator locIterator = locBCALTruthToMCThrownMap.find(locBCALTruthShower); DVector3 locProductionVertex(0.0, 0.0, dTargetCenter); //target center if(locIterator != locBCALTruthToMCThrownMap.end()) locProductionVertex = locIterator->second->position(); double locTrueTheta = (locBCALTruthShowerPosition - locProductionVertex).Theta(); double locReconstructedTheta = (locBCALShowerPosition - locProductionVertex).Theta(); double locDeltaPhi = locBCALShowerPosition.Phi() - locBCALTruthShower->phi; while(locDeltaPhi > TMath::Pi()) locDeltaPhi -= 2.0*TMath::Pi(); while(locDeltaPhi < -1.0*TMath::Pi()) locDeltaPhi += 2.0*TMath::Pi(); double locUnitCircleArcLength = acos(sin(locTrueTheta)*sin(locReconstructedTheta)*cos(locDeltaPhi) + cos(locTrueTheta)*cos(locReconstructedTheta)); //x = r*cos(phi)*sin(theta) //y = r*sin(phi)*sin(theta) //z = r*cos(theta) //unit arclength = acos(unit dot product) //unit dot product //cos(phi1)*sin(theta1)*cos(phi2)*sin(theta2) + sin(phi1)*sin(theta1)*sin(phi2)*sin(theta2) + cos(theta1)*cos(theta2) //sin(theta1)*sin(theta2)*(cos(phi1)*cos(phi2) + sin(phi1)*sin(phi2)) + cos(theta1)*cos(theta2) //unit arclength = acos(sin(theta1)*sin(theta2)*cos(phi1 - phi2) + cos(theta1)*cos(theta2)); if((locUnitCircleArcLength*180.0/TMath::Pi()) > dMaximumBCALMatchAngleDegrees) continue; //keep the shower with the largest energy if(locBCALShower->E > locLargestEnergy) { locMatchFoundFlag = true; locLargestEnergy = locBCALShower->E; locBestBCALTruthShowerIndex = loc_i; locBestBCALShowerIndex = loc_j; } } } if(!locMatchFoundFlag) //no more good matches!1 break; locBCALTruthShower = locBCALTruthShowers[locBestBCALTruthShowerIndex]; locBCALShower = locBCALShowers[locBestBCALShowerIndex]; locBCALShowerToTruthMap[locBCALShower] = locBCALTruthShower; locBCALTruthToShowerMap[locBCALTruthShower] = locBCALShower; locBCALShowers.erase(locBCALShowers.begin() + locBestBCALShowerIndex); locBCALTruthShowers.erase(locBCALTruthShowers.begin() + locBestBCALTruthShowerIndex); } locMCThrownMatching->Set_BCALShowerToTruthMap(locBCALShowerToTruthMap); locMCThrownMatching->Set_BCALTruthToShowerMap(locBCALTruthToShowerMap); } void DMCThrownMatching_factory::Find_GenReconMatches_TOFPoints(JEventLoop* locEventLoop, DMCThrownMatching* locMCThrownMatching) const { vector locTOFTruths; const DTOFTruth* locTOFTruth; locEventLoop->Get(locTOFTruths); vector locTOFPoints; const DTOFPoint* locTOFPoint; locEventLoop->Get(locTOFPoints); map locTOFPointToTruthMap; map locTOFTruthToPointMap; size_t locBestTOFPointIndex = 0, locBestTOFTruthIndex = 0; double locMatchDistance; while((!locTOFPoints.empty()) && (!locTOFTruths.empty())) { bool locMatchFoundFlag = false; double locLargestEnergy = 0.0; for(size_t loc_i = 0; loc_i < locTOFTruths.size(); ++loc_i) { locTOFTruth = locTOFTruths[loc_i]; for(size_t loc_j = 0; loc_j < locTOFPoints.size(); ++loc_j) { locTOFPoint = locTOFPoints[loc_j]; DVector3 locTOFPointPosition = locTOFPoint->pos; //DTOFPoint and DTOFTruth reported at different z's (I think center vs. detector face): propagate truth information to the reconstructed z double locDeltaZ = locTOFPointPosition.Z() - locTOFTruth->z; DVector3 locTrueMomentum(locTOFTruth->px, locTOFTruth->py, locTOFTruth->pz); double locDeltaPathLength = locDeltaZ/cos(locTrueMomentum.Theta()); double locTrueX = locTOFTruth->x + locDeltaPathLength*sin(locTrueMomentum.Theta())*cos(locTrueMomentum.Phi()); double locTrueY = locTOFTruth->y + locDeltaPathLength*sin(locTrueMomentum.Theta())*sin(locTrueMomentum.Phi()); DVector3 locTOFTruthPosition(locTrueX, locTrueY, locTOFPointPosition.Z()); locMatchDistance = (locTOFTruthPosition - locTOFPointPosition).Mag(); if(locMatchDistance > dMaximumTOFMatchDistance) continue; //keep the shower with the largest energy if(locTOFTruth->E > locLargestEnergy) { locMatchFoundFlag = true; locLargestEnergy = locTOFTruth->E; locBestTOFTruthIndex = loc_i; locBestTOFPointIndex = loc_j; } } } if(!locMatchFoundFlag) //no more good matches! break; locTOFTruth = locTOFTruths[locBestTOFTruthIndex]; locTOFPoint = locTOFPoints[locBestTOFPointIndex]; locTOFPointToTruthMap[locTOFPoint] = locTOFTruth; locTOFTruthToPointMap[locTOFTruth] = locTOFPoint; locTOFPoints.erase(locTOFPoints.begin() + locBestTOFPointIndex); locTOFTruths.erase(locTOFTruths.begin() + locBestTOFTruthIndex); } locMCThrownMatching->Set_TOFPointToTruthMap(locTOFPointToTruthMap); locMCThrownMatching->Set_TOFTruthToPointMap(locTOFTruthToPointMap); } void DMCThrownMatching_factory::Find_GenReconMatches_ChargedTrack(const vector& locInputMCThrownVector, const vector& locChargedTracks, DMCThrownMatching* locMCThrownMatching) const { //assumes Find_GenReconMatches_ChargedHypo has been called first! map locChargedHypoToThrownMap; map > locThrownToChargedHypoMap; locMCThrownMatching->Get_ChargedHypoToThrownMap(locChargedHypoToThrownMap); locMCThrownMatching->Get_ThrownToChargedHypoMap(locThrownToChargedHypoMap); map locChargedToThrownMap; map locThrownToChargedMap; map >::iterator locIterator; for(locIterator = locThrownToChargedHypoMap.begin(); locIterator != locThrownToChargedHypoMap.end(); ++locIterator) { for(size_t loc_j = 0; loc_j < locChargedTracks.size(); ++loc_j) { if(locChargedTracks[loc_j]->Get_BestFOM()->candidateid == (locIterator->second)[0]->candidateid) //match { locChargedToThrownMap[locChargedTracks[loc_j]] = locIterator->first; locThrownToChargedMap[locIterator->first] = locChargedTracks[loc_j]; } } } locMCThrownMatching->Set_ChargedToThrownMap(locChargedToThrownMap); locMCThrownMatching->Set_ThrownToChargedMap(locThrownToChargedMap); } void DMCThrownMatching_factory::Find_GenReconMatches_ChargedHypo(const vector& locInputMCThrownVector, const vector& locInputChargedTrackHypothesisVector, DMCThrownMatching* locMCThrownMatching) const { map locChargedToThrownMap; map > locThrownToChargedMap; const DChargedTrackHypothesis* locChargedTrackHypothesis; const DMCThrown* locMCThrown; size_t locBestChargedTrackHypothesisIndex = 0, locBestMCThrownIndex = 0; double locMatchFOM, locBestMatchFOM; vector locMCThrownVector = locInputMCThrownVector; vector locChargedTrackHypothesisVector = locInputChargedTrackHypothesisVector; if(dDebugLevel > 0) cout << "START IT!" << endl; while((!locMCThrownVector.empty()) && (!locChargedTrackHypothesisVector.empty())) { if(dDebugLevel > 0) cout << "Begin loop!" << endl; bool locMatchFoundFlag = false; locBestMatchFOM = dMinimumMatchFOM; for(size_t loc_i = 0; loc_i < locMCThrownVector.size(); ++loc_i) { locMCThrown = locMCThrownVector[loc_i]; for(size_t loc_j = 0; loc_j < locChargedTrackHypothesisVector.size(); ++loc_j) { locChargedTrackHypothesis = locChargedTrackHypothesisVector[loc_j]; if(ParticleCharge(locChargedTrackHypothesis->PID()) != ParticleCharge((Particle_t)(locMCThrown->type))) continue; //wrong charge locMatchFOM = Calc_MatchFOM(locMCThrown->momentum(), locChargedTrackHypothesis->momentum()); if(dDebugLevel > 0) { cout << "MATCHING: MCTHROWN: "; cout << ParticleType((Particle_t)(locMCThrown->type)) << ", " << locMCThrown->momentum().Mag() << ", " << locMCThrown->momentum().Theta()*180.0/TMath::Pi() << ", " << locMCThrown->momentum().Phi()*180.0/TMath::Pi() << endl; cout << "MATCHING: CHARGEDHYPO: "; cout << ParticleType(locChargedTrackHypothesis->PID()) << ", " << locChargedTrackHypothesis->momentum().Mag() << ", " << locChargedTrackHypothesis->momentum().Theta()*180.0/TMath::Pi() << ", " << locChargedTrackHypothesis->momentum().Phi()*180.0/TMath::Pi() << endl; cout << "MATCHING: FOM, candidate id: " << locMatchFOM << ", " << locChargedTrackHypothesis->candidateid << endl; } if(locMatchFOM >= locBestMatchFOM) { locMatchFoundFlag = true; locBestMatchFOM = locMatchFOM; locBestMCThrownIndex = loc_i; locBestChargedTrackHypothesisIndex = loc_j; } } } if(!locMatchFoundFlag) //no more good matches!1 break; locMCThrown = locMCThrownVector[locBestMCThrownIndex]; locChargedTrackHypothesis = locChargedTrackHypothesisVector[locBestChargedTrackHypothesisIndex]; locChargedToThrownMap[locChargedTrackHypothesis] = locMCThrown; locChargedTrackHypothesisVector.erase(locChargedTrackHypothesisVector.begin() + locBestChargedTrackHypothesisIndex); locMCThrownVector.erase(locMCThrownVector.begin() + locBestMCThrownIndex); //automatically add all other DChargedTrackHypothesis objects from the same DChargedTrack to this match. deque locMatchedChargedHypos(1, locChargedTrackHypothesis); for(int loc_i = locChargedTrackHypothesisVector.size() - 1; loc_i >= 0; --loc_i) { if(dDebugLevel > 0) { cout << "CHECKING: CHARGEDHYPO: "; cout << ParticleType(locChargedTrackHypothesisVector[loc_i]->PID()) << ", " << locChargedTrackHypothesisVector[loc_i]->momentum().Mag() << ", " << locChargedTrackHypothesisVector[loc_i]->momentum().Theta()*180.0/TMath::Pi() << ", " << locChargedTrackHypothesisVector[loc_i]->momentum().Phi()*180.0/TMath::Pi() << endl; cout << "best id, test id = " << locChargedTrackHypothesis->candidateid << ", " << locChargedTrackHypothesisVector[loc_i]->candidateid << endl; } if(locChargedTrackHypothesisVector[loc_i]->candidateid == locChargedTrackHypothesis->candidateid) { if(dDebugLevel > 0) cout << "save!" << endl; locChargedToThrownMap[locChargedTrackHypothesisVector[loc_i]] = locMCThrown; locMatchedChargedHypos.push_back(locChargedTrackHypothesisVector[loc_i]); locChargedTrackHypothesisVector.erase(locChargedTrackHypothesisVector.begin() + loc_i); } } locThrownToChargedMap[locMCThrown] = locMatchedChargedHypos; } locMCThrownMatching->Set_ChargedHypoToThrownMap(locChargedToThrownMap); locMCThrownMatching->Set_ThrownToChargedHypoMap(locThrownToChargedMap); } void DMCThrownMatching_factory::Find_GenReconMatches_NeutralParticle(const vector& locInputMCThrownVector, const vector& locNeutralParticles, DMCThrownMatching* locMCThrownMatching) const { //assumes Find_GenReconMatches_NeutralHypo has been called first! map locNeutralHypoToThrownMap; map > locThrownToNeutralHypoMap; locMCThrownMatching->Get_NeutralHypoToThrownMap(locNeutralHypoToThrownMap); locMCThrownMatching->Get_ThrownToNeutralHypoMap(locThrownToNeutralHypoMap); vector locNeutralShowerVector_Matched; vector locNeutralShowerVector_Check; map locNeutralToThrownMap; map locThrownToNeutralMap; map >::iterator locIterator; for(locIterator = locThrownToNeutralHypoMap.begin(); locIterator != locThrownToNeutralHypoMap.end(); ++locIterator) { (locIterator->second)[0]->GetT(locNeutralShowerVector_Matched); for(size_t loc_j = 0; loc_j < locNeutralParticles.size(); ++loc_j) { locNeutralParticles[loc_j]->GetT(locNeutralShowerVector_Check); if(locNeutralShowerVector_Check[0] == locNeutralShowerVector_Matched[0]) { locNeutralToThrownMap[locNeutralParticles[loc_j]] = locIterator->first; locThrownToNeutralMap[locIterator->first] = locNeutralParticles[loc_j]; } } } locMCThrownMatching->Set_NeutralToThrownMap(locNeutralToThrownMap); locMCThrownMatching->Set_ThrownToNeutralMap(locThrownToNeutralMap); } void DMCThrownMatching_factory::Find_GenReconMatches_NeutralHypo(const vector& locInputMCThrownVector, const vector& locInputNeutralParticleHypothesisVector, DMCThrownMatching* locMCThrownMatching) const { map locNeutralToThrownMap; map > locThrownToNeutralMap; const DNeutralParticleHypothesis* locNeutralParticleHypothesis; const DMCThrown* locMCThrown; size_t locBestNeutralParticleHypothesisIndex = 0, locBestMCThrownIndex = 0; double locMatchFOM, locBestMatchFOM; vector locMCThrownVector = locInputMCThrownVector; vector locNeutralParticleHypothesisVector = locInputNeutralParticleHypothesisVector; while((!locMCThrownVector.empty()) && (!locNeutralParticleHypothesisVector.empty())) { bool locMatchFoundFlag = false; locBestMatchFOM = dMinimumMatchFOM; for(size_t loc_i = 0; loc_i < locMCThrownVector.size(); ++loc_i) { locMCThrown = locMCThrownVector[loc_i]; for(size_t loc_j = 0; loc_j < locNeutralParticleHypothesisVector.size(); ++loc_j) { locNeutralParticleHypothesis = locNeutralParticleHypothesisVector[loc_j]; locMatchFOM = Calc_MatchFOM(locMCThrown->momentum(), locNeutralParticleHypothesis->momentum()); if(locMatchFOM >= locBestMatchFOM) { locMatchFoundFlag = true; locBestMatchFOM = locMatchFOM; locBestMCThrownIndex = loc_i; locBestNeutralParticleHypothesisIndex = loc_j; } } } if(!locMatchFoundFlag) //no more good matches!1 break; locMCThrown = locMCThrownVector[locBestMCThrownIndex]; locNeutralParticleHypothesis = locNeutralParticleHypothesisVector[locBestNeutralParticleHypothesisIndex]; locNeutralToThrownMap[locNeutralParticleHypothesis] = locMCThrown; locNeutralParticleHypothesisVector.erase(locNeutralParticleHypothesisVector.begin() + locBestNeutralParticleHypothesisIndex); locMCThrownVector.erase(locMCThrownVector.begin() + locBestMCThrownIndex); //automatically add all other DNeutralParticleHypothesis objects from the same DNeutralShower to this match. deque locMatchedNeutralHypos(1, locNeutralParticleHypothesis); vector locNeutralShowerVector_Matched; vector locNeutralShowerVector_Check; locNeutralParticleHypothesis->GetT(locNeutralShowerVector_Matched); for(int loc_i = locNeutralParticleHypothesisVector.size() - 1; loc_i >= 0; --loc_i) { locNeutralParticleHypothesisVector[loc_i]->GetT(locNeutralShowerVector_Check); if(locNeutralShowerVector_Check[0] == locNeutralShowerVector_Matched[0]) { locNeutralToThrownMap[locNeutralParticleHypothesisVector[loc_i]] = locMCThrown; locMatchedNeutralHypos.push_back(locNeutralParticleHypothesisVector[loc_i]); locNeutralParticleHypothesisVector.erase(locNeutralParticleHypothesisVector.begin() + loc_i); } } locThrownToNeutralMap[locMCThrown] = locMatchedNeutralHypos; } locMCThrownMatching->Set_NeutralHypoToThrownMap(locNeutralToThrownMap); locMCThrownMatching->Set_ThrownToNeutralHypoMap(locThrownToNeutralMap); } double DMCThrownMatching_factory::Calc_MatchFOM(const DVector3& locMomentum_Thrown, const DVector3& locMomentum_Detected) const { double locDeltaPOverP = (locMomentum_Thrown.Mag() > 0.0) ? (locMomentum_Thrown.Mag() - locMomentum_Detected.Mag())/locMomentum_Thrown.Mag() : 9.9E9; //mean is zero double locDeltaPOverPSigma = 0.03; double locDeltaPOverPChiSq = locDeltaPOverP*locDeltaPOverP/(locDeltaPOverPSigma*locDeltaPOverPSigma); double locDeltaTheta = locMomentum_Thrown.Angle(locMomentum_Detected); //mean is zero double locDeltaThetaSigma = 0.5*TMath::Pi()/180.0; //0.5 degrees: phi dominates (theta is ~0.03 degrees) double locDeltaThetaChiSq = locDeltaTheta*locDeltaTheta/(locDeltaThetaSigma*locDeltaThetaSigma); double locFOM = TMath::Prob(locDeltaPOverPChiSq + locDeltaThetaChiSq, 2); if(dDebugLevel > 10) cout << "delta p over p, delta theta, total FOM = " << locDeltaPOverP << ", " << locDeltaTheta << ", " << locFOM << endl; return locFOM; } //------------------ // erun //------------------ jerror_t DMCThrownMatching_factory::erun(void) { return NOERROR; } //------------------ // fini //------------------ jerror_t DMCThrownMatching_factory::fini(void) { return NOERROR; }