// $Id$ // // File: DEventRFBunch_factory_Combo.cc // Created: Tue Aug 9 14:29:24 EST 2011 // Creator: pmatt // #include "DEventRFBunch_factory_Combo.h" //------------------ // init //------------------ jerror_t DEventRFBunch_factory_Combo::init(void) { dRFBunchFrequency = 2.004; return NOERROR; } //------------------ // brun //------------------ jerror_t DEventRFBunch_factory_Combo::brun(jana::JEventLoop *locEventLoop, int runnumber) { vector locParticleIDVector; locEventLoop->Get(locParticleIDVector); dParticleID = locParticleIDVector[0]; // Get Target parameters from XML DApplication *locApplication = dynamic_cast (locEventLoop->GetJApplication()); DGeometry *locGeometry = locApplication ? locApplication->GetDGeometry(runnumber):NULL; dTargetCenterZ = 65.0; dTargetLength = 30.0; dTargetRadius = 1.5; //FIX: grab from database!!! if(locGeometry) { locGeometry->GetTargetZ(dTargetCenterZ); locGeometry->GetTargetLength(dTargetLength); } return NOERROR; } //------------------ // evnt //------------------ jerror_t DEventRFBunch_factory_Combo::evnt(jana::JEventLoop *locEventLoop, int eventnumber) { vector locParticleComboBlueprints; locEventLoop->Get(locParticleComboBlueprints); vector locEventRFBunches; locEventLoop->Get(locEventRFBunches); //the below disables this routine until everything is working DEventRFBunch* locEventRFBunch = new DEventRFBunch(*locEventRFBunches[0]); locEventRFBunch->dMatchedToTracksFlag = false; for(size_t loc_i = 0; loc_i < locParticleComboBlueprints.size(); ++loc_i) locEventRFBunch->AddAssociatedObject(locParticleComboBlueprints[loc_i]); _data.push_back(locEventRFBunch); return NOERROR; //note that this routine ignores any detached vertices: without kinematic fitting, probably bogus anyway, and probably won't affect the end result regardless //it uses all charged particles in the combo to find the vertex: if one is very forward, hopefully the others will avg it out (and it's variance will be large...) vector locTrackTimeBasedVector(0); locEventLoop->Get(locTrackTimeBasedVector, "Combo"); map locComboRFBunchMap; for(size_t loc_i = 0; loc_i < locParticleComboBlueprints.size(); ++loc_i) { const DParticleComboBlueprint* locParticleComboBlueprint = locParticleComboBlueprints[loc_i]; double locWeightedAverageStartTime_Numerator = 0.0, locWeightedAverageStartTime_Denominator = 0.0; double locWeightedAverageVertexZ_Numerator = 0.0, locWeightedAverageVertexZ_Denominator = 0.0; //do charged particles first (need vertex-z for neutrals), then neutrals for(size_t loc_j = 0; loc_j < locParticleComboBlueprint->Get_NumParticleComboBlueprintSteps(); ++loc_j) { const DParticleComboBlueprintStep* locParticleComboBlueprintStep = locParticleComboBlueprint->Get_ParticleComboBlueprintStep(loc_j); for(size_t loc_k = 0; loc_k < locParticleComboBlueprintStep->Get_NumFinalParticleSourceObjects(); ++loc_k) { const JObject* locSourceObject = locParticleComboBlueprintStep->Get_FinalParticle_SourceObject(loc_k); if(locSourceObject == NULL) continue; //missing or decaying const DChargedTrack* locChargedTrack = dynamic_cast(locSourceObject); if(locChargedTrack == NULL) continue; //neutral, will do later Particle_t locPID = locParticleComboBlueprintStep->Get_FinalParticleID(loc_k); const DChargedTrackHypothesis* locChargedTrackHypothesis = locChargedTrack->Get_Hypothesis(locPID); if(locChargedTrackHypothesis != NULL) { double locStartTime = locChargedTrackHypothesis->time(); double locStartTimeVariance = (locChargedTrackHypothesis->errorMatrix())(6, 6); if(locStartTimeVariance > 0.0) { locWeightedAverageStartTime_Numerator += locStartTime/locStartTimeVariance; locWeightedAverageStartTime_Denominator += 1.0/locStartTimeVariance; } double locVertexZ = locChargedTrackHypothesis->position().Z(); double locVertexZVariance = (locChargedTrackHypothesis->errorMatrix())(5, 5); locWeightedAverageVertexZ_Numerator += locVertexZ/locVertexZVariance; locWeightedAverageVertexZ_Denominator += 1.0/locVertexZVariance; continue; } //get from locTrackTimeBasedVector for(size_t loc_l = 0; loc_l < locTrackTimeBasedVector.size(); ++loc_l) { const DTrackTimeBased* locTrackTimeBased = locTrackTimeBasedVector[loc_l]; if(locTrackTimeBased->PID() != locPID) continue; const DChargedTrack* locTimeBasedSourceChargedTrack = NULL; locTrackTimeBased->GetSingleT(locTimeBasedSourceChargedTrack); if(locTimeBasedSourceChargedTrack != locChargedTrack) continue; double locStartTime, locStartTimeVariance; Get_StartTime(locEventLoop, locTrackTimeBased, locStartTime, locStartTimeVariance); if(locStartTimeVariance > 0.0) { locWeightedAverageStartTime_Numerator += locStartTime/locStartTimeVariance; locWeightedAverageStartTime_Denominator += 1.0/locStartTimeVariance; } double locVertexZ = locTrackTimeBased->position().Z(); double locVertexZVariance = (locTrackTimeBased->errorMatrix())(5, 5); locWeightedAverageVertexZ_Numerator += locVertexZ/locVertexZVariance; locWeightedAverageVertexZ_Denominator += 1.0/locVertexZVariance; break; } } } //now do neutrals double locVertexZ = (locWeightedAverageVertexZ_Denominator > 0.0) ? locWeightedAverageVertexZ_Numerator/locWeightedAverageVertexZ_Denominator : dTargetCenterZ; DVector3 locVertex(0.0, 0.0, locVertexZ); for(size_t loc_j = 0; loc_j < locParticleComboBlueprint->Get_NumParticleComboBlueprintSteps(); ++loc_j) { const DParticleComboBlueprintStep* locParticleComboBlueprintStep = locParticleComboBlueprint->Get_ParticleComboBlueprintStep(loc_j); for(size_t loc_k = 0; loc_k < locParticleComboBlueprintStep->Get_NumFinalParticleSourceObjects(); ++loc_k) { const JObject* locSourceObject = locParticleComboBlueprintStep->Get_FinalParticle_SourceObject(loc_k); if(locSourceObject == NULL) continue; //missing or decaying const DNeutralShower* locNeutralShower = dynamic_cast(locSourceObject); if(locNeutralShower == NULL) continue; //charged, did earlier Particle_t locPID = locParticleComboBlueprintStep->Get_FinalParticleID(loc_k); double locStartTime, locStartTimeVariance; Calc_StartTime(locNeutralShower, locPID, locVertex, locStartTime, locStartTimeVariance); if(locStartTimeVariance > 0.0) { locWeightedAverageStartTime_Numerator += locStartTime/locStartTimeVariance; locWeightedAverageStartTime_Denominator += 1.0/locStartTimeVariance; } } } if(!(locWeightedAverageStartTime_Denominator > 0.0)) { //no timing information somehow: use the pre-existing value DEventRFBunch* locEventRFBunch = new DEventRFBunch(*locEventRFBunches[0]); locEventRFBunch->dMatchedToTracksFlag = false; locEventRFBunch->AddAssociatedObject(locParticleComboBlueprint); _data.push_back(locEventRFBunch); continue; } // Find # RF Bunch Shifts double locStartTime = locWeightedAverageStartTime_Numerator/locWeightedAverageStartTime_Denominator; double locPropagatedRFTime = locEventRFBunches[0]->dTime + (locVertexZ - dTargetCenterZ)/SPEED_OF_LIGHT; int locNumBunchShifts = 0; while((locPropagatedRFTime - locStartTime) > (0.5*dRFBunchFrequency)) { --locNumBunchShifts; locPropagatedRFTime -= dRFBunchFrequency; } while((locPropagatedRFTime - locStartTime) < (-0.5*dRFBunchFrequency)) { ++locNumBunchShifts; locPropagatedRFTime += dRFBunchFrequency; } // Create new RF Bunch if doesn't already exist if(locComboRFBunchMap.find(locNumBunchShifts) != locComboRFBunchMap.end()) //already created, don't recreate identical object! locComboRFBunchMap[locNumBunchShifts]->AddAssociatedObject(locParticleComboBlueprint); else { DEventRFBunch* locEventRFBunch = new DEventRFBunch(); locEventRFBunch->dMatchedToTracksFlag = true; locEventRFBunch->dTime = locEventRFBunches[0]->dTime + (double)(locNumBunchShifts)*dRFBunchFrequency; locEventRFBunch->dTimeVariance = locEventRFBunches[0]->dTimeVariance; locEventRFBunch->AddAssociatedObject(locParticleComboBlueprint); _data.push_back(locEventRFBunch); locComboRFBunchMap[locNumBunchShifts] = locEventRFBunch; } } return NOERROR; } void DEventRFBunch_factory_Combo::Get_StartTime(JEventLoop* locEventLoop, const DTrackTimeBased* locTrackTimeBased, double& locStartTime, double& locStartTimeVariance) { vector locTOFPoints(0); locEventLoop->Get(locTOFPoints); vector locBCALShowers(0); locEventLoop->Get(locBCALShowers); vector locFCALShowers(0); locEventLoop->Get(locFCALShowers); vector locSCHits(0); locEventLoop->Get(locSCHits); deque locMatchedBCALShowers(0); deque locMatchedFCALShowers(0); // Use time-based tracking time as initial guess locStartTime = 0.0; locStartTimeVariance = 0.0; double locPathLength = 0.0, locTempProjectedTime = 0.0, locFlightTime = 0.0; //BCAL locTempProjectedTime = 0.0; // to reject hits that are not in time with the track if (dParticleID->MatchToBCAL(locTrackTimeBased->rt, locBCALShowers, locMatchedBCALShowers, locTempProjectedTime, locPathLength, locFlightTime) == NOERROR) { locStartTime = locTempProjectedTime; double locStartTimeUncetainty = 0.00255*pow(locTrackTimeBased->momentum().Mag(), -2.52) + 0.220; locStartTimeVariance = locStartTimeUncetainty*locStartTimeUncetainty; return; } //TOF locTempProjectedTime = 0.0; // to reject hits that are not in time with the track pair locTOFdEdx(0.0, 0.0); unsigned int locTOFIndex = 0; if (dParticleID->MatchToTOF(locTrackTimeBased->rt, DTrackFitter::kTimeBased, locTOFPoints, locTempProjectedTime, locTOFIndex, locPathLength, locFlightTime,&locTOFdEdx) == NOERROR) { locStartTime = locTempProjectedTime; locStartTimeVariance = 0.08*0.08; return; } //FCAL locTempProjectedTime = 0.0; // to reject hits that are not in time with the track double locFCALdEdx=0.; if (dParticleID->MatchToFCAL(locTrackTimeBased->rt, locFCALShowers, locMatchedFCALShowers, locTempProjectedTime, locPathLength, locFlightTime,&locFCALdEdx) == NOERROR) { locStartTime = locTempProjectedTime; locStartTimeVariance = 0.6*0.6; return; } /* //SC // DON'T ENABLE UNTIL START COUNTER RECONSTRUCTION READY // Match to the start counter using the result of the time-based fit locTempProjectedTime = 0.0; // to reject hits that are not in time with the track unsigned int locSCIndex; double locTempSCdEdx; if (dParticleID->MatchToSC(locTrackTimeBased->rt, DTrackFitter::kTimeBased, locSCHits, locTempProjectedTime, locSCIndex, locPathLength, locFlightTime,&locTempSCdEdx) == NOERROR) { locStartTime = locTempProjectedTime; locStartTimeVariance = 0.3*0.3; //guess for now } */ } void DEventRFBunch_factory_Combo::Calc_StartTime(const DNeutralShower* locNeutralShower, Particle_t locPID, DVector3 locVertex, double& locStartTime, double& locStartTimeVariance) { locStartTime = 0.0; locStartTimeVariance = 0.0; //doesn't work for neutrons!! double locHitTime = locNeutralShower->dSpacetimeVertex.T(); double locShowerEnergy = locNeutralShower->dEnergy; DVector3 locHitPoint = locNeutralShower->dSpacetimeVertex.Vect(); // Calculate DNeutralParticleHypothesis Quantities (projected time at vertex for given id, etc.) double locMass = ParticleMass(locPID); double locParticleEnergy = locShowerEnergy; //need to correct this for neutrons! if (locParticleEnergy < locMass) return; //not enough energy for PID hypothesis DVector3 locPath = locHitPoint - locVertex; double locPathLength = locPath.Mag(); if(!(locPathLength > 0.0)) return; double locPMag = sqrt(locParticleEnergy*locParticleEnergy - locMass*locMass); double locFlightTime = locPathLength*locParticleEnergy/(locPMag*SPEED_OF_LIGHT); locStartTime = locHitTime - locFlightTime; DVector3 locMomentum(locPath); locMomentum.SetMag(locPMag); locStartTimeVariance = Calc_StartTimeVariance(locNeutralShower, locMass, locMomentum, locPath); } double DEventRFBunch_factory_Combo::Calc_StartTimeVariance(const DNeutralShower* locNeutralShower, double locMass, const DVector3& locMomentum, const DVector3& locPathVector) { //build 8x8 matrix: 5x5 shower, 3x3 vertex position DMatrixDSym locShowerPlusVertCovariance(8); for(unsigned int loc_l = 0; loc_l < 5; ++loc_l) //shower: e, x, y, z, t { for(unsigned int loc_m = 0; loc_m < 5; ++loc_m) locShowerPlusVertCovariance(loc_l, loc_m) = locNeutralShower->dCovarianceMatrix(loc_l, loc_m); } if(!(locNeutralShower->dCovarianceMatrix(4, 4) > 0.0)) return 0.0; locShowerPlusVertCovariance(5, 5) = 0.25*dTargetRadius*dTargetRadius/12.0; //vertex position x locShowerPlusVertCovariance(6, 6) = 0.25*dTargetRadius*dTargetRadius/12.0; //vertex position y locShowerPlusVertCovariance(7, 7) = dTargetLength*dTargetLength/12.0; //vertex position z DVector3 locDeltaX = -1.0*locPathVector; //defined oppositely in document! DVector3 locDeltaXOverDeltaXSq = (1.0/locDeltaX.Mag2())*locDeltaX; DVector3 locEPVecOverPSq = (locNeutralShower->dEnergy/locMomentum.Mag2())*locMomentum; DVector3 locEPVecOverCPMagDeltaXMag = (locNeutralShower->dEnergy/(SPEED_OF_LIGHT*locDeltaX.Mag()*locMomentum.Mag()))*locDeltaX; //build transform matrix DMatrix locTransformMatrix(1, 8); locTransformMatrix(0, 0) = locMass*locMass*locDeltaX.Mag()/(SPEED_OF_LIGHT*locMomentum.Mag()*locMomentum.Mag2()); //partial deriv of t wrst shower-e locTransformMatrix(0, 1) = locEPVecOverCPMagDeltaXMag.X(); //partial deriv of t wrst shower-x locTransformMatrix(0, 2) = locEPVecOverCPMagDeltaXMag.Y(); //partial deriv of t wrst shower-y locTransformMatrix(0, 3) = locEPVecOverCPMagDeltaXMag.Z(); //partial deriv of t wrst shower-z locTransformMatrix(0, 4) = 1.0; //partial deriv of t wrst shower-t locTransformMatrix(0, 5) = -1.0*locTransformMatrix(0, 1); //partial deriv of t wrst vert-x locTransformMatrix(0, 6) = -1.0*locTransformMatrix(0, 2); //partial deriv of t wrst vert-y locTransformMatrix(0, 7) = -1.0*locTransformMatrix(0, 3); //partial deriv of t wrst vert-z //convert DMatrixDSym locParticleCovariance(1); locParticleCovariance = locShowerPlusVertCovariance.Similarity(locTransformMatrix); return locParticleCovariance(0, 0); } //------------------ // erun //------------------ jerror_t DEventRFBunch_factory_Combo::erun(void) { return NOERROR; } //------------------ // fini //------------------ jerror_t DEventRFBunch_factory_Combo::fini(void) { return NOERROR; }