68 ATH_MSG_FATAL(
"Incorrect number of Psi daughters (should be 3 or 4)");
69 return StatusCode::FAILURE;
72 constexpr int topoN = 3;
75 return StatusCode::FAILURE;
77 std::array<SG::WriteHandle<xAOD::VertexContainer>, topoN> VtxWriteHandles;
int ikey(0);
80 ATH_CHECK( VtxWriteHandles[ikey].record(std::make_unique<xAOD::VertexContainer>(), std::make_unique<xAOD::VertexAuxContainer>()) );
89 if (pvContainer.
cptr()->size()==0) {
91 return StatusCode::RECOVERABLE;
100 ATH_CHECK( refPvContainer.
record(std::make_unique<xAOD::VertexContainer>(), std::make_unique<xAOD::VertexAuxContainer>()) );
103 std::vector<Trk::VxCascadeInfo*> cascadeinfoContainer;
104 std::vector<Trk::VxCascadeInfo*> cascadeinfoContainer_noConstr;
149 for(
size_t ic=0; ic<cascadeinfoContainer.size(); ic++) {
151 if(cascade_info==
nullptr) {
158 const std::vector<xAOD::Vertex*> &cascadeVertices = cascade_info->
vertices();
159 if(cascadeVertices.size() != topoN)
ATH_MSG_ERROR(
"Incorrect number of vertices");
160 if(cascadeVertices[0]==
nullptr || cascadeVertices[1]==
nullptr || cascadeVertices[2]==
nullptr)
ATH_MSG_ERROR(
"Error null vertex");
162 for(
int i=0; i<topoN; i++) VtxWriteHandles[i].ptr()->push_back(cascadeVertices[i]);
165 const auto mainVertex = cascadeVertices[2];
166 const std::vector< std::vector<TLorentzVector> > &moms = cascade_info->
getParticleMoms();
169 std::vector<VertexLink> precedingVertexLinks;
173 if( vertexLink1.
isValid() ) precedingVertexLinks.push_back( vertexLink1 );
177 if( vertexLink2.
isValid() ) precedingVertexLinks.push_back( vertexLink2 );
178 CascadeLinksDecor(*mainVertex) = precedingVertexLinks;
190 std::vector<const xAOD::Vertex*> psi2VerticestoLink;
191 if(psi2Vertex) psi2VerticestoLink.push_back(psi2Vertex);
195 std::vector<const xAOD::Vertex*> psi1VerticestoLink;
196 if(psi1Vertex) psi1VerticestoLink.push_back(psi1Vertex);
217 chi2_decor(*mainVertex) = cascade_info->
fitChi2();
218 ndof_decor(*mainVertex) = cascade_info->
nDoF();
219 chi2_nc_decor(*mainVertex) = cascade_info_noConstr ? cascade_info_noConstr->
fitChi2() : -999999.;
220 ndof_nc_decor(*mainVertex) = cascade_info_noConstr ? cascade_info_noConstr->
nDoF() : -1;
223 chi2_SV1_decor(*cascadeVertices[0]) =
m_V0Tools->chisq(cascadeVertices[0]);
224 chi2_nc_SV1_decor(*cascadeVertices[0]) = cascade_info_noConstr ?
m_V0Tools->chisq(cascade_info_noConstr->
vertices()[0]) : -999999.;
225 chi2_V1_decor(*cascadeVertices[0]) =
m_V0Tools->chisq(psi1Vertex);
226 ndof_V1_decor(*cascadeVertices[0]) =
m_V0Tools->ndof(psi1Vertex);
227 lxy_SV1_decor(*cascadeVertices[0]) =
m_CascadeTools->lxy(moms[0],cascadeVertices[0],mainVertex);
228 lxyErr_SV1_decor(*cascadeVertices[0]) =
m_CascadeTools->lxyError(moms[0],cascade_info->
getCovariance()[0],cascadeVertices[0],mainVertex);
229 a0z_SV1_decor(*cascadeVertices[0]) =
m_CascadeTools->a0z(moms[0],cascadeVertices[0],mainVertex);
230 a0zErr_SV1_decor(*cascadeVertices[0]) =
m_CascadeTools->a0zError(moms[0],cascade_info->
getCovariance()[0],cascadeVertices[0],mainVertex);
231 a0xy_SV1_decor(*cascadeVertices[0]) =
m_CascadeTools->a0xy(moms[0],cascadeVertices[0],mainVertex);
232 a0xyErr_SV1_decor(*cascadeVertices[0]) =
m_CascadeTools->a0xyError(moms[0],cascade_info->
getCovariance()[0],cascadeVertices[0],mainVertex);
235 chi2_SV2_decor(*cascadeVertices[1]) =
m_V0Tools->chisq(cascadeVertices[1]);
236 chi2_nc_SV2_decor(*cascadeVertices[1]) = cascade_info_noConstr ?
m_V0Tools->chisq(cascade_info_noConstr->
vertices()[1]) : -999999.;
237 chi2_V2_decor(*cascadeVertices[1]) =
m_V0Tools->chisq(psi2Vertex);
238 ndof_V2_decor(*cascadeVertices[1]) =
m_V0Tools->ndof(psi2Vertex);
239 lxy_SV2_decor(*cascadeVertices[1]) =
m_CascadeTools->lxy(moms[1],cascadeVertices[1],mainVertex);
240 lxyErr_SV2_decor(*cascadeVertices[1]) =
m_CascadeTools->lxyError(moms[1],cascade_info->
getCovariance()[1],cascadeVertices[1],mainVertex);
241 a0z_SV2_decor(*cascadeVertices[1]) =
m_CascadeTools->a0z(moms[1],cascadeVertices[1],mainVertex);
242 a0zErr_SV2_decor(*cascadeVertices[1]) =
m_CascadeTools->a0zError(moms[1],cascade_info->
getCovariance()[1],cascadeVertices[1],mainVertex);
243 a0xy_SV2_decor(*cascadeVertices[1]) =
m_CascadeTools->a0xy(moms[1],cascadeVertices[1],mainVertex);
244 a0xyErr_SV2_decor(*cascadeVertices[1]) =
m_CascadeTools->a0xyError(moms[1],cascade_info->
getCovariance()[1],cascadeVertices[1],mainVertex);
252 for (
auto cascade_info : cascadeinfoContainer)
delete cascade_info;
253 for (
auto cascade_info_noConstr : cascadeinfoContainer_noConstr)
delete cascade_info_noConstr;
255 return StatusCode::SUCCESS;
371 StatusCode
PsiPlusPsiCascade::performSearch(std::vector<Trk::VxCascadeInfo*> *cascadeinfoContainer, std::vector<Trk::VxCascadeInfo*> *cascadeinfoContainer_noConstr,
const EventContext& ctx)
const {
373 assert(cascadeinfoContainer!=
nullptr && cascadeinfoContainer_noConstr!=
nullptr);
379 std::vector<const xAOD::TrackParticle*> tracksJpsi1;
380 std::vector<const xAOD::TrackParticle*> tracksJpsi2;
381 std::vector<const xAOD::TrackParticle*> tracksDiTrk1;
382 std::vector<const xAOD::TrackParticle*> tracksDiTrk2;
383 std::vector<const xAOD::TrackParticle*> tracksPsi1;
384 std::vector<const xAOD::TrackParticle*> tracksPsi2;
385 std::vector<double> massesPsi1;
390 std::vector<double> massesPsi2;
405 std::vector<const xAOD::Vertex*> selectedPsi2Candidates;
406 for(
auto vxcItr=psi2Container.
cptr()->cbegin(); vxcItr!=psi2Container.
cptr()->cend(); ++vxcItr) {
419 double mass_psi2 =
m_V0Tools->invariantMass(*vxcItr, massesPsi2);
420 if (mass_psi2 < m_psi2MassLower || mass_psi2 >
m_psi2MassUpper)
continue;
423 TLorentzVector p4_mu1, p4_mu2;
424 p4_mu1.SetPtEtaPhiM( (*vxcItr)->trackParticle(0)->pt(),
425 (*vxcItr)->trackParticle(0)->eta(),
427 p4_mu2.SetPtEtaPhiM( (*vxcItr)->trackParticle(1)->pt(),
428 (*vxcItr)->trackParticle(1)->eta(),
430 double mass_jpsi2 = (p4_mu1 + p4_mu2).M();
431 if (mass_jpsi2 < m_jpsi2MassLower || mass_jpsi2 >
m_jpsi2MassUpper)
continue;
434 TLorentzVector p4_trk1, p4_trk2;
435 p4_trk1.SetPtEtaPhiM( (*vxcItr)->trackParticle(2)->pt(),
436 (*vxcItr)->trackParticle(2)->eta(),
438 p4_trk2.SetPtEtaPhiM( (*vxcItr)->trackParticle(3)->pt(),
439 (*vxcItr)->trackParticle(3)->eta(),
441 double mass_diTrk2 = (p4_trk1 + p4_trk2).M();
445 double chi2DOF = (*vxcItr)->chiSquared()/(*vxcItr)->numberDoF();
448 selectedPsi2Candidates.push_back(*vxcItr);
450 if(selectedPsi2Candidates.size()==0)
return StatusCode::SUCCESS;
453 std::vector<const xAOD::Vertex*> selectedPsi1Candidates;
454 for(
auto vxcItr=psi1Container.
cptr()->cbegin(); vxcItr!=psi1Container.
cptr()->cend(); ++vxcItr) {
467 double mass_psi1 =
m_V0Tools->invariantMass(*vxcItr,massesPsi1);
468 if(mass_psi1 < m_psi1MassLower || mass_psi1 >
m_psi1MassUpper)
continue;
471 TLorentzVector p4_mu1, p4_mu2;
472 p4_mu1.SetPtEtaPhiM( (*vxcItr)->trackParticle(0)->pt(),
473 (*vxcItr)->trackParticle(0)->eta(),
475 p4_mu2.SetPtEtaPhiM( (*vxcItr)->trackParticle(1)->pt(),
476 (*vxcItr)->trackParticle(1)->eta(),
478 double mass_jpsi1 = (p4_mu1 + p4_mu2).M();
479 if (mass_jpsi1 < m_jpsi1MassLower || mass_jpsi1 >
m_jpsi1MassUpper)
continue;
482 TLorentzVector p4_trk1, p4_trk2;
483 p4_trk1.SetPtEtaPhiM( (*vxcItr)->trackParticle(2)->pt(),
484 (*vxcItr)->trackParticle(2)->eta(),
486 p4_trk2.SetPtEtaPhiM( (*vxcItr)->trackParticle(3)->pt(),
487 (*vxcItr)->trackParticle(3)->eta(),
489 double mass_diTrk1 = (p4_trk1 + p4_trk2).M();
493 double chi2DOF = (*vxcItr)->chiSquared()/(*vxcItr)->numberDoF();
496 selectedPsi1Candidates.push_back(*vxcItr);
498 if(selectedPsi1Candidates.size()==0)
return StatusCode::SUCCESS;
500 std::vector<std::pair<const xAOD::Vertex*, const xAOD::Vertex*> > candidatePairs;
501 for(
auto psi1Itr=selectedPsi1Candidates.cbegin(); psi1Itr!=selectedPsi1Candidates.cend(); ++psi1Itr) {
503 for(
size_t i=0; i<(*psi1Itr)->nTrackParticles(); i++) tracksPsi1.push_back((*psi1Itr)->trackParticle(i));
504 for(
auto psi2Itr=selectedPsi2Candidates.cbegin(); psi2Itr!=selectedPsi2Candidates.cend(); ++psi2Itr) {
506 for(
size_t j=0; j<(*psi2Itr)->nTrackParticles(); j++) {
507 if(std::find(tracksPsi1.cbegin(), tracksPsi1.cend(), (*psi2Itr)->trackParticle(j)) != tracksPsi1.cend()) {
skip =
true;
break; }
511 for(
size_t ic=0; ic<candidatePairs.size(); ic++) {
512 const xAOD::Vertex* psi1Vertex = candidatePairs[ic].first;
513 const xAOD::Vertex* psi2Vertex = candidatePairs[ic].second;
514 if((psi1Vertex == *psi1Itr && psi2Vertex == *psi2Itr) || (psi1Vertex == *psi2Itr && psi2Vertex == *psi1Itr)) {
skip =
true;
break; }
518 candidatePairs.push_back(std::pair<const xAOD::Vertex*, const xAOD::Vertex*>(*psi1Itr,*psi2Itr));
522 std::sort( candidatePairs.begin(), candidatePairs.end(), [](std::pair<const xAOD::Vertex*, const xAOD::Vertex*>
a, std::pair<const xAOD::Vertex*, const xAOD::Vertex*> b) { return a.first->chiSquared()/a.first->numberDoF()+a.second->chiSquared()/a.second->numberDoF() < b.first->chiSquared()/b.first->numberDoF()+b.second->chiSquared()/b.second->numberDoF(); } );
524 candidatePairs.erase(candidatePairs.begin()+
m_maxCandidates, candidatePairs.end());
527 for(
size_t ic=0; ic<candidatePairs.size(); ic++) {
528 const xAOD::Vertex* psi1Vertex = candidatePairs[ic].first;
529 const xAOD::Vertex* psi2Vertex = candidatePairs[ic].second;
533 if (tracksPsi1.size() != massesPsi1.size()) {
534 ATH_MSG_ERROR(
"Problems with Psi1 input: number of tracks or track mass inputs is not correct!");
538 if (tracksPsi2.size() != massesPsi2.size()) {
539 ATH_MSG_ERROR(
"Problems with Psi2 input: number of tracks or track mass inputs is not correct!");
545 tracksDiTrk1.clear();
553 tracksDiTrk2.clear();
559 TLorentzVector p4_moth;
578 std::vector<Trk::VertexID> vrtList;
587 vrtList.push_back(vID1);
595 vrtList.push_back(vID2);
597 std::vector<const xAOD::TrackParticle*> tp; tp.clear();
598 std::vector<double> tp_masses; tp_masses.clear();
601 std::vector<Trk::VertexID> cnstV; cnstV.clear();
607 std::vector<Trk::VertexID> cnstV; cnstV.clear();
613 std::vector<Trk::VertexID> cnstV; cnstV.clear();
619 std::vector<Trk::VertexID> cnstV; cnstV.clear();
629 for(
auto v :
result->vertices()) {
630 if(v->nTrackParticles()==0) {
631 std::vector<ElementLink<xAOD::TrackParticleContainer> > nullLinkVector;
632 v->setTrackParticleLinks(nullLinkVector);
639 result->setSVOwnership(
true);
646 cascadeinfoContainer->push_back(
result.release());
656 std::vector<Trk::VertexID> vrtList_nc;
659 vrtList_nc.push_back(vID1_nc);
662 vrtList_nc.push_back(vID2_nc);
664 std::vector<const xAOD::TrackParticle*> tp; tp.clear();
665 std::vector<double> tp_masses; tp_masses.clear();
668 std::unique_ptr<Trk::VxCascadeInfo> result_nc(
m_iVertexFitter->fitCascade(*state));
670 if (result_nc !=
nullptr) {
671 for(
auto v : result_nc->vertices()) {
672 if(v->nTrackParticles()==0) {
673 std::vector<ElementLink<xAOD::TrackParticleContainer> > nullLinkVector;
674 v->setTrackParticleLinks(nullLinkVector);
681 result_nc->setSVOwnership(
true);
682 cascadeinfoContainer_noConstr->push_back(result_nc.release());
684 else cascadeinfoContainer_noConstr->push_back(0);
686 else cascadeinfoContainer_noConstr->push_back(0);
690 return StatusCode::SUCCESS;