From d1b9add7790e65dab501d1b0970df514045e25a2 Mon Sep 17 00:00:00 2001 From: Adam Edward Barton <adam.edward.barton@cern.ch> Date: Wed, 16 Dec 2020 10:16:12 +0000 Subject: [PATCH] Initial migration of BPHY Derivations to master --- .../BPhysTools/BPhysTools/BPhysBlindingTool.h | 278 +++ .../BPhysTools/BPhysTools/BPhysToolsDict.h | 16 + .../BPhysTools/BPhysTrackVertexMapTool.h | 196 ++ .../BPhysTools/IBPhysBlindingTool.h | 79 + .../BPhysTools/IBPhysTrackVertexMapTool.h | 109 + .../BPhysTools/BPhysTools/SimpleEncrypter.h | 244 ++ .../BPhys/BPhysTools/BPhysTools/selection.xml | 10 + .../BPhys/BPhysTools/CMakeLists.txt | 84 + .../BPhysTools/Root/BPhysBlindingTool.cxx | 464 ++++ .../Root/BPhysTrackVertexMapTool.cxx | 664 +++++ .../BPhys/BPhysTools/Root/SimpleEncrypter.cxx | 516 ++++ .../src/components/BPhysTools_entries.cxx | 4 + .../BPhysTools/util/createBlindingKeys.cxx | 83 + .../DerivationFrameworkBPhys/CMakeLists.txt | 26 + .../AugOriginalCounts.h | 91 + .../BMuonTrackIsoTool.h | 111 + .../BPhysAddMuonBasedInvMass.h | 304 +++ .../BPhysConversionFinder.h | 76 + .../BPhysMetadataBase.h | 100 + .../BPhysPVCascadeTools.h | 159 ++ .../BPhysPVThinningTool.h | 53 + .../DerivationFrameworkBPhys/BPhysPVTools.h | 108 + .../BPhysVarBlinder.h | 76 + .../BPhysVertexTrackBase.h | 303 +++ .../BTrackVertexMapLogger.h | 49 + .../BVertexClosestTrackTool.h | 149 ++ .../BVertexTrackIsoTool.h | 119 + .../BmumuThinningTool.h | 441 ++++ .../DerivationFrameworkBPhys/Bmumu_metadata.h | 42 + .../Bmumu_reco_mumu.h | 73 + .../DerivationFrameworkBPhys/Cascade3Plus1.h | 99 + .../DerivationFrameworkBPhys/CascadeTools.h | 87 + .../DerivationFrameworkBPhys/CfAthAlgTool.h | 86 + .../DerivationFrameworkBPhys/FourMuonTool.h | 188 ++ .../JpsiPlusDpstCascade.h | 96 + .../JpsiPlusDs1Cascade.h | 105 + .../JpsiPlusDsCascade.h | 91 + .../JpsiPlusV0Cascade.h | 94 + .../DerivationFrameworkBPhys/LocalVector.h | 72 + .../MuonExtrapolationTool.h | 59 + .../DerivationFrameworkBPhys/ReVertex.h | 98 + .../DerivationFrameworkBPhys/Reco_4mu.h | 63 + .../DerivationFrameworkBPhys/Reco_V0Finder.h | 63 + .../DerivationFrameworkBPhys/Reco_Vertex.h | 54 + .../DerivationFrameworkBPhys/Select_Bmumu.h | 116 + .../Select_onia2mumu.h | 71 + .../Thin_vtxDuplicates.h | 37 + .../DerivationFrameworkBPhys/Thin_vtxTrk.h | 54 + .../TriggerCountToMetadata.h | 51 + .../VertexCaloIsolation.h | 89 + .../VertexPlus1TrackCascade.h | 74 + .../VertexTrackIsolation.h | 52 + .../python/BPhysPyHelpers.py | 84 + .../python/__init__.py | 2 + .../DerivationFrameworkBPhys/share/BPHY1.py | 274 ++ .../DerivationFrameworkBPhys/share/BPHY10.py | 477 ++++ .../DerivationFrameworkBPhys/share/BPHY11.py | 506 ++++ .../DerivationFrameworkBPhys/share/BPHY12.py | 389 +++ .../DerivationFrameworkBPhys/share/BPHY13.py | 483 ++++ .../DerivationFrameworkBPhys/share/BPHY14.py | 329 +++ .../DerivationFrameworkBPhys/share/BPHY15.py | 855 +++++++ .../DerivationFrameworkBPhys/share/BPHY16.py | 288 +++ .../DerivationFrameworkBPhys/share/BPHY17.py | 316 +++ .../DerivationFrameworkBPhys/share/BPHY18.py | 506 ++++ .../DerivationFrameworkBPhys/share/BPHY19.py | 313 +++ .../DerivationFrameworkBPhys/share/BPHY2.py | 332 +++ .../DerivationFrameworkBPhys/share/BPHY20.py | 775 ++++++ .../DerivationFrameworkBPhys/share/BPHY21.py | 329 +++ .../DerivationFrameworkBPhys/share/BPHY22.py | 22 + .../DerivationFrameworkBPhys/share/BPHY3.py | 284 +++ .../DerivationFrameworkBPhys/share/BPHY4.py | 155 ++ .../DerivationFrameworkBPhys/share/BPHY5.py | 558 +++++ .../DerivationFrameworkBPhys/share/BPHY6.py | 344 +++ .../DerivationFrameworkBPhys/share/BPHY7.py | 648 +++++ .../DerivationFrameworkBPhys/share/BPHY8.py | 2200 +++++++++++++++++ .../DerivationFrameworkBPhys/share/BPHY9.py | 398 +++ .../SaveExtraMetadataInMerge_jobOFragment.py | 53 + .../share/configureConversionFinder.py | 69 + .../share/configureSimpleV0Finder.py | 201 ++ .../share/configureV0Finder.py | 202 ++ .../share/configureVertexing.py | 148 ++ .../src/AugOriginalCounts.cxx | 150 ++ .../src/BMuonTrackIsoTool.cxx | 448 ++++ .../src/BPhysAddMuonBasedInvMass.cxx | 702 ++++++ .../src/BPhysConversionFinder.cxx | 589 +++++ .../src/BPhysMetadataBase.cxx | 270 ++ .../src/BPhysPVCascadeTools.cxx | 475 ++++ .../src/BPhysPVThinningTool.cxx | 134 + .../src/BPhysPVTools.cxx | 545 ++++ .../src/BPhysVarBlinder.cxxNoCompile | 72 + .../src/BPhysVertexTrackBase.cxx | 1480 +++++++++++ .../src/BTrackVertexMapLogger.cxx | 104 + .../src/BVertexClosestTrackTool.cxx | 672 +++++ .../src/BVertexTrackIsoTool.cxx | 511 ++++ .../src/BmumuThinningTool.cxx_NoCompile | 1167 +++++++++ .../src/Bmumu_metadata.cxx | 238 ++ .../src/Bmumu_reco_mumu.cxx | 177 ++ .../src/Cascade3Plus1.cxx | 524 ++++ .../src/CascadeTools.cxx | 608 +++++ .../src/CfAthAlgTool.cxx | 178 ++ .../src/FourMuonTool.cxx | 449 ++++ .../src/JpsiPlusDpstCascade.cxx | 724 ++++++ .../src/JpsiPlusDs1Cascade.cxx | 905 +++++++ .../src/JpsiPlusDsCascade.cxx | 702 ++++++ .../src/JpsiPlusV0Cascade.cxx | 576 +++++ .../src/MuonExtrapolationTool.cxx | 182 ++ .../DerivationFrameworkBPhys/src/ReVertex.cxx | 284 +++ .../DerivationFrameworkBPhys/src/Reco_4mu.cxx | 269 ++ .../src/Reco_V0Finder.cxx | 307 +++ .../src/Reco_Vertex.cxx | 163 ++ .../src/Select_Bmumu.cxx | 563 +++++ .../src/Select_onia2mumu.cxx | 197 ++ .../src/Thin_vtxDuplicates.cxx | 176 ++ .../src/Thin_vtxTrk.cxx | 200 ++ .../src/TriggerCountToMetadata.cxx | 62 + .../src/VertexCaloIsolation.cxx | 583 +++++ .../src/VertexPlus1TrackCascade.cxx | 215 ++ .../src/VertexTrackIsolation.cxx | 274 ++ .../DerivationFrameworkBPhys_entries.cxx | 74 + 119 files changed, 34315 insertions(+) create mode 100644 PhysicsAnalysis/BPhys/BPhysTools/BPhysTools/BPhysBlindingTool.h create mode 100644 PhysicsAnalysis/BPhys/BPhysTools/BPhysTools/BPhysToolsDict.h create mode 100644 PhysicsAnalysis/BPhys/BPhysTools/BPhysTools/BPhysTrackVertexMapTool.h create mode 100644 PhysicsAnalysis/BPhys/BPhysTools/BPhysTools/IBPhysBlindingTool.h create mode 100644 PhysicsAnalysis/BPhys/BPhysTools/BPhysTools/IBPhysTrackVertexMapTool.h create mode 100644 PhysicsAnalysis/BPhys/BPhysTools/BPhysTools/SimpleEncrypter.h create mode 100644 PhysicsAnalysis/BPhys/BPhysTools/BPhysTools/selection.xml create mode 100644 PhysicsAnalysis/BPhys/BPhysTools/CMakeLists.txt create mode 100644 PhysicsAnalysis/BPhys/BPhysTools/Root/BPhysBlindingTool.cxx create mode 100644 PhysicsAnalysis/BPhys/BPhysTools/Root/BPhysTrackVertexMapTool.cxx create mode 100644 PhysicsAnalysis/BPhys/BPhysTools/Root/SimpleEncrypter.cxx create mode 100644 PhysicsAnalysis/BPhys/BPhysTools/src/components/BPhysTools_entries.cxx create mode 100644 PhysicsAnalysis/BPhys/BPhysTools/util/createBlindingKeys.cxx create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/CMakeLists.txt create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/AugOriginalCounts.h create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/BMuonTrackIsoTool.h create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/BPhysAddMuonBasedInvMass.h create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/BPhysConversionFinder.h create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/BPhysMetadataBase.h create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/BPhysPVCascadeTools.h create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/BPhysPVThinningTool.h create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/BPhysPVTools.h create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/BPhysVarBlinder.h create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/BPhysVertexTrackBase.h create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/BTrackVertexMapLogger.h create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/BVertexClosestTrackTool.h create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/BVertexTrackIsoTool.h create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/BmumuThinningTool.h create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/Bmumu_metadata.h create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/Bmumu_reco_mumu.h create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/Cascade3Plus1.h create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/CascadeTools.h create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/CfAthAlgTool.h create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/FourMuonTool.h create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/JpsiPlusDpstCascade.h create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/JpsiPlusDs1Cascade.h create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/JpsiPlusDsCascade.h create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/JpsiPlusV0Cascade.h create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/LocalVector.h create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/MuonExtrapolationTool.h create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/ReVertex.h create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/Reco_4mu.h create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/Reco_V0Finder.h create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/Reco_Vertex.h create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/Select_Bmumu.h create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/Select_onia2mumu.h create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/Thin_vtxDuplicates.h create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/Thin_vtxTrk.h create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/TriggerCountToMetadata.h create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/VertexCaloIsolation.h create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/VertexPlus1TrackCascade.h create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/VertexTrackIsolation.h create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/python/BPhysPyHelpers.py create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/python/__init__.py create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/BPHY1.py create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/BPHY10.py create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/BPHY11.py create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/BPHY12.py create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/BPHY13.py create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/BPHY14.py create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/BPHY15.py create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/BPHY16.py create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/BPHY17.py create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/BPHY18.py create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/BPHY19.py create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/BPHY2.py create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/BPHY20.py create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/BPHY21.py create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/BPHY22.py create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/BPHY3.py create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/BPHY4.py create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/BPHY5.py create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/BPHY6.py create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/BPHY7.py create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/BPHY8.py create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/BPHY9.py create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/SaveExtraMetadataInMerge_jobOFragment.py create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/configureConversionFinder.py create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/configureSimpleV0Finder.py create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/configureV0Finder.py create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/configureVertexing.py create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/AugOriginalCounts.cxx create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/BMuonTrackIsoTool.cxx create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/BPhysAddMuonBasedInvMass.cxx create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/BPhysConversionFinder.cxx create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/BPhysMetadataBase.cxx create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/BPhysPVCascadeTools.cxx create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/BPhysPVThinningTool.cxx create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/BPhysPVTools.cxx create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/BPhysVarBlinder.cxxNoCompile create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/BPhysVertexTrackBase.cxx create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/BTrackVertexMapLogger.cxx create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/BVertexClosestTrackTool.cxx create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/BVertexTrackIsoTool.cxx create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/BmumuThinningTool.cxx_NoCompile create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/Bmumu_metadata.cxx create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/Bmumu_reco_mumu.cxx create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/Cascade3Plus1.cxx create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/CascadeTools.cxx create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/CfAthAlgTool.cxx create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/FourMuonTool.cxx create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/JpsiPlusDpstCascade.cxx create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/JpsiPlusDs1Cascade.cxx create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/JpsiPlusDsCascade.cxx create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/JpsiPlusV0Cascade.cxx create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/MuonExtrapolationTool.cxx create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/ReVertex.cxx create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/Reco_4mu.cxx create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/Reco_V0Finder.cxx create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/Reco_Vertex.cxx create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/Select_Bmumu.cxx create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/Select_onia2mumu.cxx create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/Thin_vtxDuplicates.cxx create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/Thin_vtxTrk.cxx create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/TriggerCountToMetadata.cxx create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/VertexCaloIsolation.cxx create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/VertexPlus1TrackCascade.cxx create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/VertexTrackIsolation.cxx create mode 100644 PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/components/DerivationFrameworkBPhys_entries.cxx diff --git a/PhysicsAnalysis/BPhys/BPhysTools/BPhysTools/BPhysBlindingTool.h b/PhysicsAnalysis/BPhys/BPhysTools/BPhysTools/BPhysBlindingTool.h new file mode 100644 index 00000000000..c7aed623cec --- /dev/null +++ b/PhysicsAnalysis/BPhys/BPhysTools/BPhysTools/BPhysBlindingTool.h @@ -0,0 +1,278 @@ +// Dear emacs, this is -*- c++ -*- + +/* + Copyright (C) 2002-2018 CERN for the benefit of the ATLAS collaboration +*/ + +/** + * @file BPhysBlindingTool.h + * @author Wolfgang Walkowiak <Wolfgang.Walkowiak@cern.ch> + * + * @brief Dual-use tool for blinding and unblinding certain float values + */ + +// $Id: $ +#ifndef BPHYSTOOLS_BPHYSBLINDINGTOOL_H +#define BPHYSTOOLS_BPHYSBLINDINGTOOL_H + +// Framework includes +#include "AsgTools/AsgTool.h" + +// System include(s): +#include <memory> + +// Local includes +#include "BPhysTools/IBPhysBlindingTool.h" +#include "BPhysTools/SimpleEncrypter.h" + + +// EDM includes +#include "xAODTracking/VertexAuxContainer.h" + +namespace xAOD { + /// + /// @class BPhysBlindingToll + /// @author Wolfgang Walkowiak <Wolfgang.Walkowiak@cern.ch> + /// + /// Dual-use tool for blinding and unblinding certain float values + /// provided as variables in a container. + /// + /// This tool can be used in two ways: + /// 1. As a tool to blind or unblind arbitrary positive float values + /// using doBlind(float val) or doUnblind(float val). + /// For this mode to work only the corresponding key + /// (JO BlindingKey or UnblindingKey) needs to be set. + /// 2. As a tool to blind a list of variables (VarToBlindNames) + /// for a certain xAOD::VertexContainer's vertices. + /// Since this only works for positive float values, + /// an the values may be scaled by a factor and an + /// offset may be added to the values, specified separately for + /// each variable (BlindingFactors, BlindingOffsets). + /// In addition a negative sign may be added to the resulting blinded + /// value (NegativeSigns) as a convenience. + /// If this tool is used for unblinding the same values for + /// BlindingFactors, BlindingOffsets and NegativeSigns need to be + /// provided. + /// Depending on the mode, the BlindingKey or Unblindingkey need + /// to be set. + /// + /// @Note: Key pairs may be produced using the createBlindingKeys + /// utility. + /// + /// Job options: + /// - BlindingKey + /// Hex string providing the (public) blinding key. + /// - UnblindingKey + /// Hex string providing the (private) unblinding key. + /// - VertexContainerName + /// Name of the vertex container to be used + /// - BlindingFlag + /// Flag to indicate candidates for blinding ("pass_XXXX") + /// Blind values for all candidates if empty. + /// - VarToBlindNames + /// String with list of float variables to blind (delimiter: .) + /// - BlindingOffsets + /// Offsets applied to values before blinding + /// List must have same length as VarToBlindNames or zero. + /// - BlindingFactors + /// Scale factors applied before blinding + /// List must have same length as VarToBlindNames or zero. + /// - NegativeSigns + /// Flip signs to negative range? + /// List must have same length as VarToBlindNames or zero. + /// + /// + class BPhysBlindingTool : + public asg::AsgTool, virtual public xAOD::IBPhysBlindingTool { + + /// Declare the correct constructor for Athena + ASG_TOOL_CLASS( BPhysBlindingTool, xAOD::IBPhysBlindingTool ) + + public: + /// + /// @brief Regular AsgTool constructor + BPhysBlindingTool(const std::string& name = "BPhysBlindingTool"); + /// + /// @brief Method initialising the tool + virtual StatusCode initialize() override; + + /// + /// @brief Method finalizing the tool + virtual StatusCode finalize() override; + + /// + /// @brief Simply blind one positive float value + /// + /// @param[in] val : positive float value to blind. + /// + /// @returns Blinded positive float value; same value as input on error. + virtual float doBlind(const float& val) override; + + /// + /// @name Methods to be called by user classes + /// @{ + /// + /// @brief Simply unblind one positive float value + /// + /// @param[in] val : Blinded positive float value. + /// + /// @returns Unblinded positive float value; same value as input on error. + virtual float doUnblind(const float& val) override; + + /// + /// @brief Simply blind one (positive) float value with corrections + /// + /// @param[in] val : float value to blind. + /// @param[in] negativeSign : flip sign after blinding + /// @param[in] offset : before blinding, shift by offset + /// @param[in] factor : before blinding, stretch by factor + /// + /// @returns Blinded float value; same value as input on error. + virtual float doBlind(const float& val, + const bool& negativeSign, + const float& offset, + const float& factor) override; + + /// + /// @name Methods to be called by user classes + /// @{ + /// + /// @brief Simply unblind one (positive) float value with corrections + /// + /// @param[in] val : Blinded float value. + /// @param[in] negativeSign : flip sign before unblinding + /// @param[in] offset : after unblinding, shift by offset + /// @param[in] factor : after unblinding, stretch by factor + /// + /// @returns Unblinded positive float value; same value as input on error. + virtual float doUnblind(const float& val, + const bool& negativeSign, + const float& offset, + const float& factor) override; + + /// + /// @brief Perform blinding of requested variables + virtual StatusCode doBlind() override; + + /// + /// @brief Perform unblinding of requested variables + virtual StatusCode doUnblind() override; + + protected: + /// + /// @name Perform blinding or unblinding action + /// + virtual StatusCode doBlindingAction(bool unblind=false); + /// + /// @name Utility methods + /// @{ + /// + /// @brief Check whether an element is marked as passing a hypothesis. + /// + virtual bool pass(const SG::AuxElement& em, std::string hypo); + + /// + /// @brief Tokenize a string using certain separators + /// + virtual std::vector<std::string> getTokens(std::string input, + std::string seperators); + /// + /// @brief Convert vector of floats to string + /// + virtual std::string vecToString(const std::vector<float>& v) const; + /// + /// @brief Convert vector of bools to string + /// + virtual std::string vecToString(const std::vector<bool>& v) const; + /// + /// @name Cache current event. + /// + virtual StatusCode cacheEvent(); + /// @} + + protected: + /// + /// @name Job options + /// @{ + /// + /// @brief Vertex container name + std::string m_vertexContainerName; + /// + /// @brief List of variables to blind + /// + /// (as concatenated string using . as delimiter) + std::string m_varToBlindNames; + /// + /// @brief Flag to indicate candidates for blinding + /// + /// Left empty: Blind values for all candidates. + std::string m_blindingFlag; + /// + /// @brief Offsets applied to values before blinding + /// + /// List must have same length as VarToBlindNames or zero. + /// Applied before blinding/after unblinding. + std::vector<float> m_vOffsets; + /// + /// @brief Scale factors applied before blinding + /// + /// List must have same length as VarToBlindNames or zero. + /// Applied before blinding/after unblinding. + std::vector<float> m_vFactors; + /// + /// @brief Flip signs to negative range? + /// + /// List must have same length as VarToBlindNames or zero. + /// Applied after blinding/before unblinding. + std::vector<bool> m_vNegSigns; + /// + /// @brief Key for blinding + std::string m_blindKey; + /// + /// @brief Key for unblinding + std::string m_unblindKey; + /// @} + + /// + /// @name Containers + /// @{ + xAOD::VertexContainer* m_vtxContainer; //! + xAOD::VertexAuxContainer* m_vtxAuxContainer; //! + /// @} + + /// + /// @name Event caching + /// @{ + int m_cachedRun; //! + int m_cachedEvent; //! + /// @} + + /// + /// + /// @name Counters + /// @{ + long m_eventsForBlindingSeen; //! + long m_candidatesForBlindingSeen; //! + long m_eventsForUnblindingSeen; //! + long m_candidatesForUnblindingSeen; //! + long m_eventsBlinded; //! + long m_candidatesBlinded; //! + long m_eventsUnblinded; //! + long m_candidatesUnblinded; //! + /// @} + private: + /// + /// @brief Vector of variable names + /// + std::vector<std::string> m_vVarNames; //! + + /// + /// @brief Instance of SimpleEncrypter + /// + SimpleEncrypter m_senc; //! + + }; // class BPhysBlindingTool + +} // namespace xAOD + +#endif // BPHYSTOOLS_BPHYSBLINDINGTOOL_H diff --git a/PhysicsAnalysis/BPhys/BPhysTools/BPhysTools/BPhysToolsDict.h b/PhysicsAnalysis/BPhys/BPhysTools/BPhysTools/BPhysToolsDict.h new file mode 100644 index 00000000000..bf760bccff1 --- /dev/null +++ b/PhysicsAnalysis/BPhys/BPhysTools/BPhysTools/BPhysToolsDict.h @@ -0,0 +1,16 @@ +// This file's extension implies that it's C, but it's really -*- C++ -*-. + +/* + Copyright (C) 2002-2018 CERN for the benefit of the ATLAS collaboration +*/ + +/** + * @file BPhysTools/BPhysToolsDict.h + * @author Wolfgang Walkowiak <wolfgang.walkowiak@cern.ch> + * @date Mar 2018 + * @brief Dictionary header for BPhysTools. + */ + +#include "BPhysTools/BPhysBlindingTool.h" +#include "BPhysTools/BPhysTrackVertexMapTool.h" +#include "BPhysTools/SimpleEncrypter.h" diff --git a/PhysicsAnalysis/BPhys/BPhysTools/BPhysTools/BPhysTrackVertexMapTool.h b/PhysicsAnalysis/BPhys/BPhysTools/BPhysTools/BPhysTrackVertexMapTool.h new file mode 100644 index 00000000000..597539c71b9 --- /dev/null +++ b/PhysicsAnalysis/BPhys/BPhysTools/BPhysTools/BPhysTrackVertexMapTool.h @@ -0,0 +1,196 @@ +// Dear emacs, this is -*- c++ -*- + +/* + Copyright (C) 2002-2017 CERN for the benefit of the ATLAS collaboration +*/ + +// $Id: $ +#ifndef BPHYSTOOLS_BPHYSTRACKVERTEXMAPTOOL_H +#define BPHYSTOOLS_BPHYSTRACKVERTEXMAPTOOL_H + +// Framework includes +#include "BPhysTools/IBPhysTrackVertexMapTool.h" +#include "AsgTools/AsgTool.h" + +// System include(s): +#include <memory> + +// EDM includes +#include "xAODTracking/TrackParticleAuxContainer.h" +#include "xAODTracking/VertexAuxContainer.h" + +namespace xAOD { + /// + /// Dual-use tool createing a track-to-vertex map from + /// the vertex-to-track information. + /// + /// Job options provided by this class: + /// - VertexContainerName -- name of container for secondary vertices + /// - RefPVContainerName -- name of container for refitted PVs + /// - PVContainerName -- name of container for primary vertices + /// - TrackParticleContainerName -- name of container for TrackParticles + /// - DebugTrkToVtxMaxEvents -- Maximum number of events to produce + /// detailed log output for the + /// track-to-vertex association maps. + /// Set to -1 for infinity. + /// - DumpPrefix -- Line prefix for log dump lines. + /// - HypoName -- Hypothesis name + /// (for picking up inv. mass values) + /// May be a set of hypo names to be + /// tested, delimited by '|'. + /// + /// @author Wolfgang Walkowiak <Wolfgang.Walkowiak@cern.ch> + /// + /// $Revision:$ + /// $Date: $ + /// + class BPhysTrackVertexMapTool : + public asg::AsgTool, virtual public xAOD::IBPhysTrackVertexMapTool { + + /// Declare the correct constructor for Athena + ASG_TOOL_CLASS( BPhysTrackVertexMapTool, xAOD::IBPhysTrackVertexMapTool ) + + public: + /// Regular AsgTool constructor + BPhysTrackVertexMapTool(const std::string& name = + "BPhysTrackVertexMapTool"); + + /// Function initialising the tool + virtual StatusCode initialize() override; + + /// Function being excuted for each event + virtual StatusCode logEvent() override; + + /// Function finalizing the tool + virtual StatusCode finalize() override; + + /// Function indicating whether log counter allows logging of current event + virtual bool doLog() const override; + + /// convenience method to wrap output lines by a prefix + static std::string wrapLines(std::string lines, std::string prefix); + + protected: + /// @name Functions to be called by user classes + /// @{ + /// fill cache for current event + virtual StatusCode cacheEvent() override; + + /// obtain primary vertices for a given ID track (may return empty vector) + virtual std::vector<const xAOD::Vertex*> + pvsForIDTrack(const xAOD::TrackParticle* track) const override; + + /// obtain refitted primary vertices for a given ID track + /// (may return empty vector) + virtual std::vector<const xAOD::Vertex*> + refPVsForIDTrack(const xAOD::TrackParticle* track) const override; + + /// obtain secondary vertices for a given ID track (may return empty vector) + virtual std::vector<const xAOD::Vertex*> + svsForIDTrack(const xAOD::TrackParticle* track) const override; + + // track-vertex association related + virtual std::string idTrackToString(const xAOD::TrackParticle* track, + unsigned int indent=0, + bool withPV=false, + bool withRefPV=false, + bool withSV=false) override; + + virtual std::string pvToString(const xAOD::Vertex* vtx, + unsigned int indent=0, + bool withTracks=false) override; + + virtual std::string refPVToString(const xAOD::Vertex* vtx, + unsigned int indent=0, + bool withTracks=false) override; + virtual std::string svToString(const xAOD::Vertex* vtx, + unsigned int indent=0, + bool withTracks=false, + bool withMasses=false) override; + virtual std::string idTracksToString(const xAOD::TrackParticleContainer* + tpc, + unsigned int indent=0, + bool withPV=false, + bool withRefPV=false, + bool withSV=false) override; + + virtual std::string pvsToString(const xAOD::VertexContainer* pvc, + unsigned int indent=0, + bool withTracks=false) override; + virtual std::string refPVsToString(const xAOD::VertexContainer* rpvc, + unsigned int indent=0, + bool withTracks=false) override; + virtual std::string svsToString(const xAOD::VertexContainer* svc, + unsigned int indent=0, + bool withTracks=false, + bool withMasses=false) override; + virtual std::string summaryToString(std::string prefix) override; + /// @} + + protected: + virtual float getFloat(std::string name, const xAOD::Vertex* b); + + virtual std::vector<std::string> getTokens(std::string input, + std::string seperators); + + + private: + // track-vertex association related + typedef std::map<const xAOD::TrackParticle*, + std::vector<const xAOD::Vertex*> > TrackToVertexMap_t; + + virtual void initTrackVertexMaps(const xAOD::TrackParticleContainer* tpc, + const xAOD::VertexContainer* pvc, + const xAOD::VertexContainer* rpvc, + const xAOD::VertexContainer* svc); + virtual void addVertexToTrackVertexMap(TrackToVertexMap_t& map, + const xAOD::TrackParticle* track, + const xAOD::Vertex* vtx); + virtual std::string pvName(const xAOD::Vertex* vtx); + virtual std::string refPVName(const xAOD::Vertex* vtx); + virtual std::string svName(const xAOD::Vertex* vtx); + virtual std::string idTrackName(const xAOD::TrackParticle* vtx); + + protected: + // job options + std::string m_vertexContainerName; + std::string m_refPVContainerName; + std::string m_pvContainerName; + std::string m_trackParticleContainerName; + int m_debugTrkToVtxMaxEvents; + std::string m_dumpPrefix; + std::string m_hypoName; + + // containers + const xAOD::TrackParticleContainer* m_tracks; + const xAOD::TrackParticleAuxContainer* m_tracksAux; + const xAOD::VertexContainer* m_pvtxContainer; + const xAOD::VertexContainer* m_svtxContainer; + const xAOD::VertexAuxContainer* m_svtxAuxContainer; + const xAOD::VertexContainer* m_refPVContainer; + const xAOD::VertexAuxContainer* m_refPVAuxContainer; + + unsigned int m_nEvtsSeen; + + int m_cachedRun; + int m_cachedEvent; + + private: + // track-vertex association related + typedef std::map<const xAOD::Vertex*, std::string> VertexNameMap_t; + VertexNameMap_t m_pvNameMap; + VertexNameMap_t m_refPVNameMap; + VertexNameMap_t m_svNameMap; + + typedef std::map<const xAOD::TrackParticle*, std::string> TrackNameMap_t; + TrackNameMap_t m_idTrackNameMap; + + TrackToVertexMap_t m_idTrackToPVMap; + TrackToVertexMap_t m_idTrackToRefPVMap; + TrackToVertexMap_t m_idTrackToSVMap; + + }; // class BPhysTrackVertexMapTool + +} // namespace xAOD + +#endif // BPHYSTOOLS_BPHYSTRACKVERTEXMAPTOOL_H diff --git a/PhysicsAnalysis/BPhys/BPhysTools/BPhysTools/IBPhysBlindingTool.h b/PhysicsAnalysis/BPhys/BPhysTools/BPhysTools/IBPhysBlindingTool.h new file mode 100644 index 00000000000..8891bb12cd7 --- /dev/null +++ b/PhysicsAnalysis/BPhys/BPhysTools/BPhysTools/IBPhysBlindingTool.h @@ -0,0 +1,79 @@ +// Dear emacs, this is -*- c++ -*- + +/* + Copyright (C) 2002-2018 CERN for the benefit of the ATLAS collaboration +*/ + +/** + * @file IBPhysBlindingTool.h + * @author Wolfgang Walkowiak <Wolfgang.Walkowiak@cern.ch> + * + * @brief Interface for dual-use tool for (un-)blinding of float values. + */ + +#ifndef BPHYSTOOLS_IBPHYSBLINDINGTOOL_H +#define BPHYSTOOLS_IBPHYSBLINDINGTOOL_H + +// Framework includes +#include "AsgTools/IAsgTool.h" + +// System include(s): +#include <string> +#include <vector> + +// EDM includes +#include "xAODTracking/VertexContainer.h" + +namespace xAOD { + /// + /// Interface for dual-use tool for blinding and unblinding + /// certain float values provided as variables in a container. + /// + /// @author Wolfgang Walkowiak <Wolfgang.Walkowiak@cern.ch> + /// + /// + class IBPhysBlindingTool : virtual public asg::IAsgTool { + + public: + /// Declare the correct interface for Athena + ASG_TOOL_INTERFACE( xAOD::IBPhysBlindingTool ) + + /// @ brief Function finalizing the tool + virtual StatusCode finalize() = 0; + + /// + /// @name Methods to be called by user classes + /// @{ + /// + /// @brief Simply blind one positive float value + virtual float doBlind(const float& val) = 0; + + /// + /// @brief Simply unblind one positive float value + virtual float doUnblind(const float& val) = 0; + + /// + /// @brief Simply blind one (positive) float value with corretions + virtual float doBlind(const float& val, const bool& negativeSign, + const float& offset, const float& factor) = 0; + + /// + /// @brief Simply unblind one (positive) float value with corrections + virtual float doUnblind(const float& val, const bool& negativeSign, + const float& offset, const float& factor) = 0; + + /// + /// @brief Perform blinding of requested variables + virtual StatusCode doBlind() = 0; + + /// + /// @brief Perform unblinding of requested variables + virtual StatusCode doUnblind() = 0; + + /// @} + + }; // class IBPhysBlindingTool + +} // namespace xAOD + +#endif // BPHYSTOOLS_IBPHYSBLINDINGTOOL_H diff --git a/PhysicsAnalysis/BPhys/BPhysTools/BPhysTools/IBPhysTrackVertexMapTool.h b/PhysicsAnalysis/BPhys/BPhysTools/BPhysTools/IBPhysTrackVertexMapTool.h new file mode 100644 index 00000000000..bb0857657df --- /dev/null +++ b/PhysicsAnalysis/BPhys/BPhysTools/BPhysTools/IBPhysTrackVertexMapTool.h @@ -0,0 +1,109 @@ +// Dear emacs, this is -*- c++ -*- + +/* + Copyright (C) 2002-2017 CERN for the benefit of the ATLAS collaboration +*/ + +// $Id: $ +#ifndef BPHYSTOOLS_IBPHYSTRACKVERTEXMAPTOOL_H +#define BPHYSTOOLS_IBPHYSTRACKVERTEXMAPTOOL_H + +// Framework includes +#include "AsgTools/IAsgTool.h" + +// System include(s): +#include <string> +#include <memory> +#include <vector> + +// EDM includes +#include "xAODTracking/TrackParticleContainer.h" +#include "xAODTracking/VertexContainer.h" + +namespace xAOD { + /// + /// Interface for dual-use tool createing a track-to-vertex map from + /// the vertex-to-track information. + /// + /// @author Wolfgang Walkowiak <Wolfgang.Walkowiak@cern.ch> + /// + /// $Revision:$ + /// $Date: $ + /// + class IBPhysTrackVertexMapTool : virtual public asg::IAsgTool { + + public: + /// Declare the correct interface for Athena + ASG_TOOL_INTERFACE( xAOD::IBPhysTrackVertexMapTool ) + + /// Function being excuted for each event + virtual StatusCode logEvent() = 0; + + /// Function finalizing the tool + virtual StatusCode finalize() = 0; + + /// Function indicating whether log counter allows logging of current event + virtual bool doLog() const = 0; + + public: + /// @name Functions to be called by user classes + /// @{ + /// fill cache for current event + virtual StatusCode cacheEvent() = 0; + + /// obtain primary vertices for a given ID track (may return empty vector) + virtual std::vector<const xAOD::Vertex*> + pvsForIDTrack(const xAOD::TrackParticle* track) const = 0; + + /// obtain refitted primary vertices for a given ID track + /// (may return empty vector) + virtual std::vector<const xAOD::Vertex*> + refPVsForIDTrack(const xAOD::TrackParticle* track) const = 0; + + /// obtain secondary vertices for a given ID track (may return empty vector) + virtual std::vector<const xAOD::Vertex*> + svsForIDTrack(const xAOD::TrackParticle* track) const = 0; + + // track-vertex association related + virtual std::string idTrackToString(const xAOD::TrackParticle* track, + unsigned int indent=0, + bool withPV=false, + bool withRefPV=false, + bool withSV=false) = 0; + + virtual std::string pvToString(const xAOD::Vertex* vtx, + unsigned int indent=0, + bool withTracks=false) = 0; + + virtual std::string refPVToString(const xAOD::Vertex* vtx, + unsigned int indent=0, + bool withTracks=false) = 0; + virtual std::string svToString(const xAOD::Vertex* vtx, + unsigned int indent=0, + bool withTracks=false, + bool withMasses=false) = 0; + virtual std::string idTracksToString(const xAOD::TrackParticleContainer* + tpc, + unsigned int indent=0, + bool withPV=false, + bool withRefPV=false, + bool withSV=false) = 0; + + virtual std::string pvsToString(const xAOD::VertexContainer* pvc, + unsigned int indent=0, + bool withTracks=false) = 0; + virtual std::string refPVsToString(const xAOD::VertexContainer* rpvc, + unsigned int indent=0, + bool withTracks=false) =0; + virtual std::string svsToString(const xAOD::VertexContainer* svc, + unsigned int indent=0, + bool withTracks=false, + bool withMasses=false) = 0; + virtual std::string summaryToString(std::string prefix) = 0; + /// @} + + }; // class IBPhysTrackVertexMapTool + +} // namespace xAOD + +#endif // BPHYSTOOLS_IBPHYSTRACKVERTEXMAPTOOL_H diff --git a/PhysicsAnalysis/BPhys/BPhysTools/BPhysTools/SimpleEncrypter.h b/PhysicsAnalysis/BPhys/BPhysTools/BPhysTools/SimpleEncrypter.h new file mode 100644 index 00000000000..59c351aa638 --- /dev/null +++ b/PhysicsAnalysis/BPhys/BPhysTools/BPhysTools/SimpleEncrypter.h @@ -0,0 +1,244 @@ +// Dear emacs, this is -*- c++ -*- + +/* + Copyright (C) 2002-2018 CERN for the benefit of the ATLAS collaboration +*/ + +/** + * @file SimpleEncrypter.h + * @author Wolfgang Walkowiak <Wolfgang.Walkowiak@cern.ch> + * + * @brief Provide simple asymmetric encryption for blinding of float values. + */ + +#ifndef BPHYSTOOLS_SIMPLEENCRYPTER_H +#define BPHYSTOOLS_SIMPLEENCRYPTER_H + +// Framework includes +#include "AsgMessaging/AsgMessaging.h" + +// System includes +#include <string> +#include <set> + +namespace xAOD { + /// + /// @class SimpleEncrypter + /// @author Wolfgang Walkowiak <Wolfgang.Walkowiak@cern.ch.> + /// + /// @brief Provide simple asymmetric encryption for blinding of float values. + /// + /// Provides asymmetric key encryption for blinding of positive float + /// values. Internally it uses a simple RSA encryption of bits in + /// the floating point numbers. + /// This class is used by the BPhysBlindingTool. + /// + class SimpleEncrypter : public asg::AsgMessaging { + + public: + /// @brief Useful typedefs + typedef long long int LLI_t; + typedef unsigned long long int ULLI_t; + + /// + /// @brief Main constructor + /// + /// @param[in] name of instance + /// + SimpleEncrypter(const std::string& name = "SimpleEncrypter"); + + /// + /// @brief Default destructor + /// + virtual ~SimpleEncrypter(); + + /// + /// @brief Generate private and public keys + /// + /// @returns key pair as string: [private key, public key] + /// + virtual std::pair<std::string, std::string> genKeyPair(); + + /// + /// @brief Set private key + /// + /// @param[in] hex string with private key + /// + virtual void setPrivKey(std::string keystr); + + /// + /// @brief Set public key + /// + /// @param[in] hex string with public key + /// + virtual void setPubKey(std::string keystr); + + /// + /// @brief Get private key + /// + /// @returns hex string with private key + /// + virtual std::string getPrivKey() const; + + /// + /// @brief Get public key + /// + /// @returns hex string with public key + /// + virtual std::string getPubKey() const; + + /// + /// @brief Encrypt a positive integer value + /// + /// @param[in] unsigned integer value to be encrypted + /// + /// @returns encrypted unsigned integer value + /// + virtual ULLI_t encrypt(ULLI_t x); + + /// + /// @brief Decrypt a positive integer value + /// + /// @param[in] unsigned integer value to be decrypted + /// + /// @returns encrypted unsigned integer value + /// + virtual ULLI_t decrypt(ULLI_t x); + + /// + /// @brief Encrypt a positive float value + /// + /// @param[in] positive float value to be encrypted + /// + /// @returns encrypted float value + /// + virtual float encrypt(float x); + + /// + /// @brief Decrypt a positive float value + /// + /// @param[in] positive float value to be decrypted + /// + /// @returns encrypted float value + /// + virtual float decrypt(float x); + + + private: + /// + /// @name Key generation utilities + /// @{ + /// + /// @brief Internally generate numeric representation of key pair + /// + virtual void genKeyPairInternal(); + /// + /// @brief Find a prime number + /// + virtual ULLI_t genPrime() const; + /// + /// @brief Check for being a prime number + /// + virtual bool isPrime(ULLI_t n) const; + /// + /// @brief Find greatest common denominator + /// + virtual ULLI_t greatestCommonDenominator(ULLI_t n1, ULLI_t n2) const; + /// + /// @brief Find a coprime number + /// + virtual ULLI_t genCoprime(ULLI_t n) const; + /// + /// @brief Find decryption exponent + /// + virtual ULLI_t genDecryptionExponent(ULLI_t phi, ULLI_t e) const; + /// + /// @} + /// + /// @name Key conversion utilities + /// @{ + /// + /// @brief Convert key to hex string + /// + virtual std::string keyToString(ULLI_t a, ULLI_t b) const; + /// + /// @brief Decode hex string to two integers + /// + virtual std::pair<ULLI_t, ULLI_t> decodeKeyString(std::string str) const; + /// @} + /// + /// @name float <-> int conversion utilities + /// @{ + /// + /// @brief Interpret bits of floating point number as integer + /// + virtual ULLI_t floatBitsToInt(float val) const; + /// + /// @brief Interpret bits of integer as floating point number + /// + virtual float intBitsToFloat(ULLI_t val) const; + /// @} + /// + /// @name Internal en-/decryption methods + /// @{ + /// + /// @brief Encrypt using format preserving encryption w.r.t. RSA modulus + /// + ULLI_t encryptFPECycle(ULLI_t a) const; + /// + /// @brief Decrypt using format preserving encryption w.r.t. RSA modulus + /// + ULLI_t decryptFPECycle(ULLI_t a) const; + /// + /// @brief Encrypt integer (internal) + /// + ULLI_t encryptInternal(ULLI_t x) const; + /// + /// @brief Decrypt integer (internal) + /// + ULLI_t decryptInternal(ULLI_t x) const; + /// + /// @brief Exponentiate a with d observing modulus n + /// + ULLI_t powerMod(ULLI_t a, ULLI_t d, ULLI_t n) const; + /// + /// @brief Check setup readiness for encryption + /// + bool isOkForEnc(); + /// + /// @brief Check setup readiness for decryption + /// + bool isOkForDec(); + /// + /// @} + + private: + /// + /// @name Internal static consts + /// + /// @brief Approximate range for prime numbers to be generated in + static const ULLI_t m_MAXRANGE; + static const ULLI_t m_MINRANGE; + /// @brief maximum number of hex digits for key parts + static const unsigned int m_MAXHEXDIGITS; + + /// + /// @name Internal member variables + /// + /// RSA modulus: common part of both keys + ULLI_t m_n; + /// encryption exponent: public key part II + ULLI_t m_e; + /// decryption exponent: private key part II + ULLI_t m_d; + + /// indicates that keys are set and range checks are ok + bool m_isOkForEnc; + bool m_isOkForDec; + + }; // class + +} // namespace xAOD + +#endif // BPHYSTOOLS_SIMPLEENCRYPTER_H + diff --git a/PhysicsAnalysis/BPhys/BPhysTools/BPhysTools/selection.xml b/PhysicsAnalysis/BPhys/BPhysTools/BPhysTools/selection.xml new file mode 100644 index 00000000000..a6539c15eec --- /dev/null +++ b/PhysicsAnalysis/BPhys/BPhysTools/BPhysTools/selection.xml @@ -0,0 +1,10 @@ +<!-- + Copyright (C) 2002-2018 CERN for the benefit of the ATLAS collaboration +--> + +<lcgdict> + <!-- BPhysTools --> + <class name="xAOD::BPhysBlindingTool" /> + <class name="xAOD::BPhysTrackVertexMapTool" /> + <class name="xAOD::SimpleEncrypter" /> +</lcgdict> diff --git a/PhysicsAnalysis/BPhys/BPhysTools/CMakeLists.txt b/PhysicsAnalysis/BPhys/BPhysTools/CMakeLists.txt new file mode 100644 index 00000000000..4229cef3b69 --- /dev/null +++ b/PhysicsAnalysis/BPhys/BPhysTools/CMakeLists.txt @@ -0,0 +1,84 @@ +# +# Copyright (C) 2002-2018 CERN for the benefit of the ATLAS collaboration +# + +# $Id: CMakeLists.txt 805745 2017-05-31 17:23:48Z wwalko $ +# +# Build configuration for the package. +# +#******************************************** +set(extra_dep) +if( XAOD_STANDALONE ) + set(extra_dep) +else() + set( extra_dep + GaudiKernel + Control/AthenaKernel + ) +endif() +#******************************************** +set(extra_libs) +if( XAOD_STANDALONE ) + set(extra_libs) +else() + set( extra_libs + GaudiKernel + AthenaKernel + ) +endif() +#******************************************** +# The name of the package: +atlas_subdir( BPhysTools ) + + +# Used external(s): +find_package( ROOT COMPONENTS Core Physics Matrix ) +find_package( Boost ) + +# Build the main library of the package: +atlas_add_library( BPhysToolsLib + BPhysTools/*.h Root/*.cxx src/*.cxx + PUBLIC_HEADERS BPhysTools + INCLUDE_DIRS ${ROOT_INCLUDE_DIRS} + PRIVATE_INCLUDE_DIRS ${Boost_INCLUDE_DIRS} + LINK_LIBRARIES ${ROOT_LIBRARIES} + xAODTracking + xAODBPhysLib + AsgTools + ${extra_libs} + PRIVATE_LINK_LIBRARIES ${Boost_LIBRARIES} + xAODEventInfo + ) + +if(NOT XAOD_STANDALONE) +atlas_add_component( BPhysTools + src/components/*.cxx + INCLUDE_DIRS ${ROOT_INCLUDE_DIRS} + PRIVATE_INCLUDE_DIRS ${Boost_INCLUDE_DIRS} + LINK_LIBRARIES ${ROOT_LIBRARIES} + xAODTracking + xAODBPhysLib + AsgTools + ${extra_libs} + BPhysToolsLib + PRIVATE_LINK_LIBRARIES ${Boost_LIBRARIES} + xAODEventInfo + ) +endif() + +# Build the dictionary +atlas_add_dictionary( BPhysToolsDict + BPhysTools/BPhysToolsDict.h + BPhysTools/selection.xml + LINK_LIBRARIES BPhysToolsLib + ) + + +# Executables in util subdirectory +file (GLOB util_sources RELATIVE "${CMAKE_CURRENT_SOURCE_DIR}" + "${CMAKE_CURRENT_SOURCE_DIR}/util/*.cxx") +foreach (source ${util_sources}) + string (REGEX REPLACE "util/(.*).cxx" "\\1" util ${source}) + atlas_add_executable (${util} ${source} LINK_LIBRARIES BPhysToolsLib) +endforeach (source ${util_sources}) + diff --git a/PhysicsAnalysis/BPhys/BPhysTools/Root/BPhysBlindingTool.cxx b/PhysicsAnalysis/BPhys/BPhysTools/Root/BPhysBlindingTool.cxx new file mode 100644 index 00000000000..269f6699792 --- /dev/null +++ b/PhysicsAnalysis/BPhys/BPhysTools/Root/BPhysBlindingTool.cxx @@ -0,0 +1,464 @@ +/* + Copyright (C) 2002-2018 CERN for the benefit of the ATLAS collaboration +*/ + +// system include: +#include "boost/tokenizer.hpp" +#include <boost/algorithm/string.hpp> +#include <set> +#include <cmath> + +// EDM includes: +#include "xAODEventInfo/EventInfo.h" + +// ROOT includes +#include "TString.h" + +// Local include(s): +#include "BPhysTools/BPhysBlindingTool.h" + +namespace xAOD { + + //-------------------------------------------------------------------------- + // Constructor + //-------------------------------------------------------------------------- + BPhysBlindingTool::BPhysBlindingTool( const std::string& name ) + : asg::AsgTool( name ), + m_vtxContainer(nullptr), m_vtxAuxContainer(nullptr), + m_cachedRun(-1), m_cachedEvent(-1), + m_eventsForBlindingSeen(0), + m_candidatesForBlindingSeen(0), + m_eventsForUnblindingSeen(0), + m_candidatesForUnblindingSeen(0), + m_eventsBlinded(0), + m_candidatesBlinded(0), + m_eventsUnblinded(0), + m_candidatesUnblinded(0) { + +#ifdef ASGTOOL_ATHENA + declareInterface< IBPhysBlindingTool >( this ); +#endif // ASGTOOL_ATHENA + + // Vertex container + declareProperty("VertexContainerName", m_vertexContainerName = ""); + + // List of variables to blind + // (as concatenated string using . as delimiter) + declareProperty("VarToBlindNames", m_varToBlindNames = ""); + + // Flag to indicate candidates for blinding + // Left empty: Blind values for all candidates. + declareProperty("BlindingFlag" , m_blindingFlag = ""); + + // Offsets applied to values before blinding + // List must have same length as VarToBlindNames or zero. + declareProperty("BlindingOffsets", m_vOffsets); + + // Scale factors applied before blinding + // List must have same length as VarToBlindNames or zero. + declareProperty("BlindingFactors", m_vFactors); + + // Flip signs to negative range? + declareProperty("NegativeSigns" , m_vNegSigns); + + // Key for blinding + declareProperty("BlindingKey" , m_blindKey = ""); + + // Key for unblinding + declareProperty("UnblindingKey" , m_unblindKey = ""); + + } + //-------------------------------------------------------------------------- + StatusCode BPhysBlindingTool::initialize() { + + // Greet the user: + ATH_MSG_DEBUG( "Initializing xAOD::BPhysBlindingTool" ); + + // Setup of variables + if ( m_vertexContainerName == "" ) { + ATH_MSG_INFO("No vertex container name provided."); + } + + if ( m_varToBlindNames != "" ) { + m_vVarNames = getTokens(m_varToBlindNames, ".,:;|"); + } + + // Blinding and unblinding keys + if ( m_blindKey == "" && m_unblindKey == "" ) { + ATH_MSG_ERROR("You must at least set a key for blinding or unblinding!"); + } else { + if ( m_blindKey != "" ) { + m_senc.setPubKey(m_blindKey); + ATH_MSG_INFO("Setting blinding key."); + } + if ( m_unblindKey != "" ) { + m_senc.setPrivKey(m_unblindKey); + ATH_MSG_INFO("Setting unblinding key."); + } + } + + // make sure offsets vector is of correct length + if ( m_vOffsets.size() < m_vVarNames.size() ) { + for (uint i=m_vOffsets.size(); i<m_vVarNames.size(); ++i) { + m_vOffsets.push_back(0.); + } + ATH_MSG_INFO("Extending BlindingOffsets list ..."); + } else if ( m_vOffsets.size() > m_vVarNames.size() ) { + ATH_MSG_WARNING("BlindingOffsets list longer than VarToBlindNames."); + } + + // make sure scale factors vector is of correct length + if ( m_vFactors.size() < m_vVarNames.size() ) { + for (uint i=m_vFactors.size(); i<m_vVarNames.size(); ++i) { + m_vFactors.push_back(1.); + } + ATH_MSG_INFO("Extending BlindingOffsets list ..."); + } else if ( m_vFactors.size() > m_vVarNames.size() ) { + ATH_MSG_WARNING("BlindingFactors list longer than VarToBlindNames."); + } + + // make sure negative signs vector is of correct length + if ( m_vNegSigns.size() < m_vVarNames.size() ) { + for (uint i=m_vNegSigns.size(); i<m_vVarNames.size(); ++i) { + m_vNegSigns.push_back(1.); + } + ATH_MSG_INFO("Extending NegativeSigns list ..."); + } else if ( m_vNegSigns.size() > m_vVarNames.size() ) { + ATH_MSG_WARNING("NegativeSigns list longer than VarToBlindNames."); + } + + // some info for the job log + ATH_MSG_INFO("VertexContainerName : " << m_vertexContainerName); + ATH_MSG_INFO("BlindingFlag : " << m_blindingFlag); + ATH_MSG_INFO("VarToBlindNames : " << m_varToBlindNames); + ATH_MSG_INFO("BlindingOffsets : " << vecToString(m_vOffsets)); + ATH_MSG_INFO("BlindingFactors : " << vecToString(m_vFactors)); + ATH_MSG_INFO("NegativeSigns : " << vecToString(m_vNegSigns)); + ATH_MSG_INFO("BlindingKey : " << m_blindKey); + ATH_MSG_INFO("UnblindingKey : " << m_unblindKey); + + // Return gracefully: + return StatusCode::SUCCESS; + } + //-------------------------------------------------------------------------- + StatusCode BPhysBlindingTool::finalize() { + + ATH_MSG_DEBUG( "Finalizing xAOD::BPhysBlindingTool" ); + + ATH_MSG_INFO("Statistics for " << name() << ":"); + ATH_MSG_INFO(Form("N_eventsForBlindingSeen : %10ld", + m_eventsForBlindingSeen)); + ATH_MSG_INFO(Form("N_eventsBlinded : %10ld", + m_eventsBlinded)); + ATH_MSG_INFO(Form("N_eventsForUnblindingSeen : %10ld", + m_eventsForUnblindingSeen)); + ATH_MSG_INFO(Form("N_eventsUnblinded : %10ld", + m_eventsUnblinded)); + ATH_MSG_INFO(Form("N_candidatesForBlindingSeen : %10ld", + m_candidatesForBlindingSeen)); + ATH_MSG_INFO(Form("N_candidatesBlinded : %10ld", + m_candidatesBlinded)); + ATH_MSG_INFO(Form("N_candidatesForUnblindingSeen : %10ld", + m_candidatesForUnblindingSeen)); + ATH_MSG_INFO(Form("N_candidatesUnblinded : %10ld", + m_candidatesUnblinded)); + + // Return gracefully: + return StatusCode::SUCCESS; + } + //-------------------------------------------------------------------------- + // Simply blind one positive float value + //-------------------------------------------------------------------------- + float BPhysBlindingTool::doBlind(const float& val) { + + return m_senc.encrypt(val); + } + //-------------------------------------------------------------------------- + // Simply unblind one positive float value + //-------------------------------------------------------------------------- + float BPhysBlindingTool::doUnblind(const float& val) { + + return m_senc.decrypt(val); + } + //-------------------------------------------------------------------------- + // Simply blind one (positive) float value + //-------------------------------------------------------------------------- + float BPhysBlindingTool::doBlind(const float& val, + const bool& negativeSign, + const float& offset, + const float& factor) { + + // adjustment if requested + float bval(val); + float cval = val*factor + offset; + if ( cval > 0. ) { + // perform actual blinding + bval = m_senc.encrypt(cval); + if (negativeSign) bval *= -1.; + } else { + ATH_MSG_WARNING("Blinding: Corrected value not positive: " + << val << Form(" (%a) -> ", val) + << cval << Form(" (%a)", cval)); + } // if cval > 0 + + return bval; + } + //-------------------------------------------------------------------------- + // Simply unblind one (positive) float value + //-------------------------------------------------------------------------- + float BPhysBlindingTool::doUnblind(const float& val, + const bool& negativeSign, + const float& offset, + const float& factor) { + + float bval(val), cval(val); + if (negativeSign) bval *= -1.; + // if ( bval > 0. || isnan(bval) ) { + if ( bval > 0. || !std::isnormal(bval) ) { + // perform actual unblinding + cval = m_senc.decrypt(bval); + if ( factor != 0. ) { + cval = (cval - offset)/factor; + } else { + ATH_MSG_WARNING("Unblinding: BlindingFactor == 0!: " + << val << Form(" (%a)", val)); + } // if m_vFactors[ivtx] != 0 + } else { + ATH_MSG_WARNING("Unblinding: Corrected value not positive: " + << val << Form(" (%a) -> ", val) + << bval << Form(" (%a)", bval)); + } // if bval > 0 + + return cval; + } + //-------------------------------------------------------------------------- + // Perform blinding of requested variables + //-------------------------------------------------------------------------- + StatusCode BPhysBlindingTool::doBlind() { + + if ( m_blindKey == "" ) { + ATH_MSG_WARNING("Can not blind without blinding key!"); + } else { + ATH_CHECK( doBlindingAction(false) ); + } + + // Return gracefully: + return StatusCode::SUCCESS; + } + //-------------------------------------------------------------------------- + // Perform unblinding of requested variables + //-------------------------------------------------------------------------- + StatusCode BPhysBlindingTool::doUnblind() { + + if ( m_unblindKey == "" ) { + ATH_MSG_WARNING("Can not unblind without unblinding key!"); + } else { + ATH_CHECK( doBlindingAction(true) ); + } + + // Return gracefully: + return StatusCode::SUCCESS; + } + //-------------------------------------------------------------------------- + + //-------------------------------------------------------------------------- + // Protected methods + //-------------------------------------------------------------------------- + + //-------------------------------------------------------------------------- + // Perform blinding or unblinding action + //-------------------------------------------------------------------------- + StatusCode BPhysBlindingTool::doBlindingAction(bool unblind) { + + ATH_CHECK(cacheEvent()); + + // counters + if ( unblind ) { + ++m_eventsForUnblindingSeen; + } else { + ++m_eventsForBlindingSeen; + } + + if ( m_vVarNames.size() > 0 ) { + long candidatesBlinded(0); + long candidatesUnblinded(0); + // loop over vertices + // int ivtx(0); + for (xAOD::VertexContainer::const_iterator + vtxItr = m_vtxContainer->begin(); + vtxItr != m_vtxContainer->end(); ++vtxItr) { + // counters + if ( unblind ) { + ++m_candidatesForUnblindingSeen; + } else { + ++m_candidatesForBlindingSeen; + } + const xAOD::Vertex* vtx = *vtxItr; + // check whether to apply (un-)blinding to this candidate + if ( m_blindingFlag == "" || pass(*vtx, m_blindingFlag) ) { + // counters + if ( unblind ) { + ++candidatesUnblinded; + } else { + ++candidatesBlinded; + } + // loop over variable names + for (size_t iv=0; iv<m_vVarNames.size(); ++iv) { + SG::AuxElement::Decorator<float> floatDec(m_vVarNames[iv]); + // check for variable + if ( floatDec.isAvailable(*vtx) ) { + float val = floatDec(*vtx); + if ( unblind ) { + // unblinding + floatDec(*vtx) = doUnblind(val, m_vNegSigns[iv], + m_vOffsets[iv], m_vFactors[iv]); + ATH_MSG_DEBUG("Unblind: " << val << Form(" (%a) -> ", val) + << floatDec(*vtx) + << Form(" (%a)", floatDec(*vtx))); + } else { + // blinding + floatDec(*vtx) = doBlind(val, m_vNegSigns[iv], + m_vOffsets[iv], m_vFactors[iv]); + ATH_MSG_DEBUG("Blind: " << val << Form(" (%a) -> ", val) + << floatDec(*vtx) + << Form(" (%a)", floatDec(*vtx))); + } // if unblind + } else { + ATH_MSG_WARNING("Missing variable " << m_vVarNames[iv]); + } // if isAvailable + } // for m_vVarNames + } // if blinding + } // for iv + // counters + if ( unblind ) { + m_candidatesUnblinded += candidatesUnblinded; + if ( candidatesUnblinded > 0 ) ++m_eventsUnblinded; + } else { + m_candidatesBlinded += candidatesBlinded; + if ( candidatesBlinded > 0 ) ++m_eventsBlinded; + } + } // if m_vVarNames.size() + + // Return gracefully: + return StatusCode::SUCCESS; + } + //-------------------------------------------------------------------------- + // Cache current event. + // + // Call this once per event. + // Repeated calls for the same run/event are not updating the cache again. + //-------------------------------------------------------------------------- + StatusCode BPhysBlindingTool::cacheEvent() { + + ATH_MSG_DEBUG("BPhysBlindingTool::cacheEvent -- begin"); + + const xAOD::EventInfo* eventInfo = NULL; + ATH_CHECK(evtStore()->retrieve(eventInfo, "EventInfo")); + + if ( m_cachedRun != (int)eventInfo->runNumber() || + m_cachedEvent != (int)eventInfo->eventNumber() ) { + + // note update + m_cachedRun = eventInfo->runNumber(); + m_cachedEvent = eventInfo->eventNumber(); + + ATH_MSG_DEBUG("BPhysBlindingTool::cacheEvent: caching now: " + << "run " << m_cachedRun << " event " << m_cachedEvent); + + // retrieve vertices container + m_vtxContainer = nullptr; + m_vtxAuxContainer = nullptr; + + if ( evtStore()->transientContains<xAOD::VertexContainer>(m_vertexContainerName) ) { + ATH_MSG_DEBUG("In transient store: " << m_vertexContainerName); + ATH_CHECK(evtStore()->retrieve(m_vtxContainer, + m_vertexContainerName)); + ATH_CHECK(evtStore()->retrieve(m_vtxAuxContainer, + m_vertexContainerName+"Aux.")); + } else { + ATH_MSG_DEBUG("Not in transient store: " << m_vertexContainerName); + const xAOD::VertexContainer* constVtxContainer = nullptr; + const xAOD::VertexAuxContainer* constVtxAuxContainer = nullptr; + ATH_CHECK(evtStore()->retrieve(constVtxContainer, + m_vertexContainerName)); + ATH_CHECK(evtStore()->retrieve(constVtxAuxContainer, + m_vertexContainerName+"Aux.")); + // create a copy + m_vtxContainer = new xAOD::VertexContainer(); + m_vtxAuxContainer = new xAOD::VertexAuxContainer(); + m_vtxContainer->setStore(m_vtxAuxContainer); + for (const xAOD::Vertex* constVtx : *constVtxContainer) { + xAOD::Vertex* vtx = new xAOD::Vertex(); + m_vtxContainer->push_back(vtx); + *vtx = *constVtx; + } + ATH_CHECK(evtStore()->record(m_vtxContainer, + m_vertexContainerName)); + ATH_CHECK(evtStore()->record(m_vtxAuxContainer, + m_vertexContainerName+"Aux.")); + } + + ATH_MSG_DEBUG("Found vertex collection with key " + << m_vertexContainerName); + + } // if new run/event + + ATH_MSG_DEBUG("BPhysBlindingTool::cacheEvent -- end"); + + // Return gracefully: + return StatusCode::SUCCESS; + } + //-------------------------------------------------------------------------- + // Helper to check whether an element is marked as passing a specific + // hypothesis. + //-------------------------------------------------------------------------- + bool BPhysBlindingTool::pass(const SG::AuxElement& em, std::string hypo) { + + if ( !boost::algorithm::starts_with(hypo, "passed_") ) + hypo = "passed_" + hypo; + SG::AuxElement::Accessor<Char_t> flagAcc(hypo); + return flagAcc.isAvailable(em) && flagAcc(em) != 0; + } + //-------------------------------------------------------------------------- + // Tokenize a string using certain separators + //-------------------------------------------------------------------------- + std::vector<std::string> + BPhysBlindingTool::getTokens(std::string input, std::string seperators) { + + std::vector<std::string> tokens; + boost::char_separator<char> sep(seperators.c_str()); + typedef boost::tokenizer<boost::char_separator<char> > Tokenizer_t; + Tokenizer_t tokenizer(input, sep); + for (auto& token : tokenizer) { + tokens.push_back(token); + } + return tokens; + } + //-------------------------------------------------------------------------- + // Format vector of floats as string + //-------------------------------------------------------------------------- + std::string BPhysBlindingTool::vecToString(const std::vector<float>& v) + const { + std::string str("["); + for (unsigned int i=0; i<v.size(); ++i) { + str += std::to_string(v[i]); + if ( i < v.size()-1 ) str += ","; + } + str += "]"; + return str; + } + //-------------------------------------------------------------------------- + // Format vector of bools as string + //-------------------------------------------------------------------------- + std::string BPhysBlindingTool::vecToString(const std::vector<bool>& v) + const { + std::string str("["); + for (unsigned int i=0; i<v.size(); ++i) { + str += std::to_string(v[i]); + if ( i < v.size()-1 ) str += ","; + } + str += "]"; + return str; + } + //-------------------------------------------------------------------------- +} // namespace xAOD diff --git a/PhysicsAnalysis/BPhys/BPhysTools/Root/BPhysTrackVertexMapTool.cxx b/PhysicsAnalysis/BPhys/BPhysTools/Root/BPhysTrackVertexMapTool.cxx new file mode 100644 index 00000000000..c0cf97b32de --- /dev/null +++ b/PhysicsAnalysis/BPhys/BPhysTools/Root/BPhysTrackVertexMapTool.cxx @@ -0,0 +1,664 @@ +/* + Copyright (C) 2002-2017 CERN for the benefit of the ATLAS collaboration +*/ + +// $Id: $ + +// system include: +#include "boost/format.hpp" +#include "boost/tokenizer.hpp" + +// EDM includes: +#include "xAODEventInfo/EventInfo.h" +#include "xAODBPhys/BPhysHypoHelper.h" + +// Local include(s): +#include "BPhysTools/BPhysTrackVertexMapTool.h" + +namespace xAOD { + + //-------------------------------------------------------------------------- + // Static utility method to prefix every line by a certain string + //-------------------------------------------------------------------------- + std::string BPhysTrackVertexMapTool::wrapLines(std::string lines, + std::string prefix) { + + std::string ostr; + std::istringstream stream(lines); + std::string line; + while ( std::getline(stream, line) ) { + if ( !ostr.empty() ) ostr += "\n"; + ostr += prefix + line; + } + return ostr; + } + //-------------------------------------------------------------------------- + // Constructor + //-------------------------------------------------------------------------- + BPhysTrackVertexMapTool::BPhysTrackVertexMapTool( const std::string& name ) + : asg::AsgTool( name ), + m_tracks(NULL), m_tracksAux(NULL), m_pvtxContainer(NULL), + m_svtxContainer(NULL), m_svtxAuxContainer(NULL), m_refPVContainer(NULL), + m_refPVAuxContainer(NULL), + m_nEvtsSeen (0), m_cachedRun(-1), m_cachedEvent(-1) { + +#ifdef ASGTOOL_ATHENA + declareInterface< IBPhysTrackVertexMapTool >( this ); +#endif // ASGTOOL_ATHENA + + // Necessary containers + declareProperty("VertexContainerName", m_vertexContainerName); + declareProperty("TrackParticleContainerName", + m_trackParticleContainerName="InDetTrackParticles"); + declareProperty("PVContainerName", m_pvContainerName = "PrimaryVertices"); + declareProperty("RefPVContainerName", m_refPVContainerName); + + // Maximum number of events to dump maps to log file for + declareProperty("DebugTrkToVtxMaxEvents", m_debugTrkToVtxMaxEvents = 0); + + // Prefix for log dump lines + declareProperty("DumpPrefix", m_dumpPrefix="TTV> "); + + // Hypothesis name (for mass value pickup) + declareProperty("HypoName", m_hypoName="__NONE__"); + } + //-------------------------------------------------------------------------- + StatusCode BPhysTrackVertexMapTool::initialize() { + + // Greet the user: + ATH_MSG_DEBUG( "Initializing xAOD::BPhysTrackVertexMapTool" ); + + if ( m_vertexContainerName == "" ) { + ATH_MSG_ERROR("No vertex container name provided!"); + } + if ( m_refPVContainerName == "" ) { + ATH_MSG_ERROR("No refitted PV container name provided!"); + } + if ( m_trackParticleContainerName == "" ) { + ATH_MSG_ERROR("No track particle container name provided!"); + } + if ( m_pvContainerName == "" ) { + ATH_MSG_ERROR("No PV container name provided!"); + } + // some info for the job log + ATH_MSG_INFO("VertexContainerName : " << m_vertexContainerName); + ATH_MSG_INFO("PVContainerName : " << m_pvContainerName); + ATH_MSG_INFO("RefPVContainerName : " << m_refPVContainerName); + ATH_MSG_INFO("TrackParticleContainerName : " + << m_trackParticleContainerName); + ATH_MSG_INFO("DebugTrkToVtxMaxEvents : " << m_debugTrkToVtxMaxEvents); + ATH_MSG_INFO("DumpPrefix : " << m_dumpPrefix); + ATH_MSG_INFO("HypoName : " << m_hypoName); + + // Return gracefully: + return StatusCode::SUCCESS; + } + //-------------------------------------------------------------------------- + StatusCode BPhysTrackVertexMapTool::finalize() { + + ATH_MSG_DEBUG( "Finalizing xAOD::BPhysTrackVertexMapTool" ); + + // Return gracefully: + return StatusCode::SUCCESS; + } + //-------------------------------------------------------------------------- + StatusCode BPhysTrackVertexMapTool::logEvent() { + + ATH_MSG_DEBUG( "logEvent in xAOD::BPhysTrackVertexMapTool" ); + + // read info into maps cache + ATH_CHECK(cacheEvent()); + + // dump info from maps if requested + if ( doLog() ) { + + ATH_MSG_INFO("Track-to-vertex association map:"); + + std::cout << summaryToString(m_dumpPrefix) << std::endl; + + } // if requested + + // increment counter + m_nEvtsSeen++; + + // Return gracefully: + return StatusCode::SUCCESS; + } + //-------------------------------------------------------------------------- + bool BPhysTrackVertexMapTool::doLog() const { + + return ( m_debugTrkToVtxMaxEvents < 0 || + m_debugTrkToVtxMaxEvents > (int)m_nEvtsSeen ); + } + //-------------------------------------------------------------------------- + // + // Cache maps for current event. + // + // Call this once per event. + // Repeated calls for the same run/event are not updating the cache again. + // + //-------------------------------------------------------------------------- + StatusCode BPhysTrackVertexMapTool::cacheEvent() { + + ATH_MSG_DEBUG("BPhysTrackVertexMapTool::cacheEvent -- begin"); + + const xAOD::EventInfo* eventInfo = NULL; + ATH_CHECK(evtStore()->retrieve(eventInfo, "EventInfo")); + + if ( m_cachedRun != (int)eventInfo->runNumber() || + m_cachedEvent != (int)eventInfo->eventNumber() ) { + + // note update + m_cachedRun = eventInfo->runNumber(); + m_cachedEvent = eventInfo->eventNumber(); + + ATH_MSG_DEBUG("BPhysTrackVertexMapTool::cacheEvent: caching now: " + << "run " << m_cachedRun << " event " << m_cachedEvent); + + // retrieve primary vertices container + m_pvtxContainer = NULL; + ATH_CHECK(evtStore()->retrieve(m_pvtxContainer, m_pvContainerName)); + ATH_MSG_DEBUG("Found PV collection with key " << m_pvContainerName); + + // retrieve ID track container + m_tracks = NULL; + m_tracksAux = NULL; + ATH_CHECK(evtStore()->retrieve(m_tracks, m_trackParticleContainerName)); + if (evtStore()->contains<xAOD:: + TrackParticleAuxContainer>(m_trackParticleContainerName+"Aux.")) { + ATH_CHECK(evtStore()->retrieve(m_tracksAux, + m_trackParticleContainerName+"Aux.")); + } else { + ATH_MSG_DEBUG("No aux track collection with key " + << m_trackParticleContainerName+"Aux."); + } + ATH_MSG_DEBUG("Found track collection with key " + << m_trackParticleContainerName); + + // vertex container and its auxilliary store + m_svtxContainer = NULL; + m_svtxAuxContainer = NULL; + ATH_CHECK(evtStore()->retrieve(m_svtxContainer, m_vertexContainerName)); + ATH_CHECK(evtStore()->retrieve(m_svtxAuxContainer, + m_vertexContainerName+"Aux.")); + ATH_MSG_DEBUG("Found SV collection with key " << m_vertexContainerName); + + // refitted primary vertex container and its auxilliary store + m_refPVContainer = NULL; + m_refPVAuxContainer = NULL; + ATH_CHECK(evtStore()->retrieve(m_refPVContainer, m_refPVContainerName)); + ATH_CHECK(evtStore()->retrieve(m_refPVAuxContainer, + m_refPVContainerName+"Aux.")); + ATH_MSG_DEBUG("Found refitted PV collection with key " + << m_refPVContainerName); + + // initialize track, PV and refPV maps + initTrackVertexMaps(m_tracks, m_pvtxContainer, m_refPVContainer, + m_svtxContainer); + } // if new run/event + + ATH_MSG_DEBUG("BPhysTrackVertexMapTool::cacheEvent -- end"); + + // Return gracefully: + return StatusCode::SUCCESS; + } + //-------------------------------------------------------------------------- + // + // Retrieve primary vertices for ID track from map. + // + //-------------------------------------------------------------------------- + std::vector<const xAOD::Vertex*> + BPhysTrackVertexMapTool::pvsForIDTrack(const xAOD::TrackParticle* track) + const { + + TrackToVertexMap_t::const_iterator it = m_idTrackToPVMap.find(track); + + if ( it != m_idTrackToPVMap.end() ) { + return it->second; + } else { + std::vector<const xAOD::Vertex*> dummy; + return dummy; + } + } + //-------------------------------------------------------------------------- + // + // Retrieve refitted primary vertices for ID track from map. + // + //-------------------------------------------------------------------------- + std::vector<const xAOD::Vertex*> + BPhysTrackVertexMapTool::refPVsForIDTrack(const xAOD::TrackParticle* track) + const { + + TrackToVertexMap_t::const_iterator it = m_idTrackToRefPVMap.find(track); + + if ( it != m_idTrackToRefPVMap.end() ) { + return it->second; + } else { + std::vector<const xAOD::Vertex*> dummy; + return dummy; + } + } + //-------------------------------------------------------------------------- + // + // Retrieve secondary vertices for ID track from map. + // + //-------------------------------------------------------------------------- + std::vector<const xAOD::Vertex*> + BPhysTrackVertexMapTool::svsForIDTrack(const xAOD::TrackParticle* track) + const { + + TrackToVertexMap_t::const_iterator it = m_idTrackToSVMap.find(track); + + if ( it != m_idTrackToSVMap.end() ) { + return it->second; + } else { + std::vector<const xAOD::Vertex*> dummy; + return dummy; + } + } + //-------------------------------------------------------------------------- + // + // Initialize ID tracks, PV and refPV related maps. + // + //-------------------------------------------------------------------------- + void BPhysTrackVertexMapTool + ::initTrackVertexMaps(const xAOD::TrackParticleContainer* tpc, + const xAOD::VertexContainer* pvc, + const xAOD::VertexContainer* rpvc, + const xAOD::VertexContainer* svc) { + + // clear previous entries + m_pvNameMap.clear(); + m_refPVNameMap.clear(); + m_svNameMap.clear(); + m_idTrackNameMap.clear(); + m_idTrackToPVMap.clear(); + m_idTrackToRefPVMap.clear(); + m_idTrackToSVMap.clear(); + + // initialize maps for PVs + for (xAOD::VertexContainer::const_iterator vtxItr = pvc->begin(); + vtxItr != pvc->end(); ++vtxItr) { + const xAOD::Vertex* vtx = *vtxItr; + pvName(vtx); + for (size_t i = 0; i < vtx->nTrackParticles(); ++i) { + const xAOD::TrackParticle* track = vtx->trackParticle(i); + // m_idTrackToPVMap[track] = vtx; + addVertexToTrackVertexMap(m_idTrackToPVMap, track, vtx); + } + } + // initialize maps for refitted PVs + for (xAOD::VertexContainer::const_iterator vtxItr = rpvc->begin(); + vtxItr != rpvc->end(); ++vtxItr) { + const xAOD::Vertex* vtx = *vtxItr; + refPVName(vtx); + for (size_t i = 0; i < vtx->nTrackParticles(); ++i) { + const xAOD::TrackParticle* track = vtx->trackParticle(i); + // m_idTrackToRefPVMap[track] = vtx; + addVertexToTrackVertexMap(m_idTrackToRefPVMap, track, vtx); + } + } + + // initialize maps for SVs + for (xAOD::VertexContainer::const_iterator vtxItr = svc->begin(); + vtxItr != svc->end(); ++vtxItr) { + const xAOD::Vertex* vtx = *vtxItr; + svName(vtx); + for (size_t i = 0; i < vtx->nTrackParticles(); ++i) { + const xAOD::TrackParticle* track = vtx->trackParticle(i); + // m_idTrackToSVMap[track] = vtx; + addVertexToTrackVertexMap(m_idTrackToSVMap, track, vtx); + } + } + // initialize maps for ID tracks + for (xAOD::TrackParticleContainer::const_iterator trkItr = tpc->begin(); + trkItr != tpc->end(); ++trkItr) { + const xAOD::TrackParticle* track = *trkItr; + idTrackName(track); + } + } + //-------------------------------------------------------------------------- + // + // Add vertex to track-to-vertex map with vector of vertices. + // + //-------------------------------------------------------------------------- + void BPhysTrackVertexMapTool + ::addVertexToTrackVertexMap(TrackToVertexMap_t& map, + const xAOD::TrackParticle* track, + const xAOD::Vertex* vtx) { + + TrackToVertexMap_t::const_iterator it = map.find(track); + + if ( it == map.end() ) { + map[track] = std::vector<const xAOD::Vertex*>(); + } + map[track].push_back(vtx); + } + //-------------------------------------------------------------------------- + // Lookup name for PV -- add as next if not yet known + //-------------------------------------------------------------------------- + std::string BPhysTrackVertexMapTool::pvName(const xAOD::Vertex* vtx) { + + if ( m_pvNameMap.find(vtx) == m_pvNameMap.end() ) { + boost::format f("PV%03d"); + f % m_pvNameMap.size(); + m_pvNameMap[vtx] = f.str(); + } + return m_pvNameMap[vtx]; + } + //-------------------------------------------------------------------------- + // Lookup name for refitted PV -- add as next if not yet known + //-------------------------------------------------------------------------- + std::string BPhysTrackVertexMapTool::refPVName(const xAOD::Vertex* vtx) { + + if ( m_refPVNameMap.find(vtx) == m_refPVNameMap.end() ) { + boost::format f("RV%03d"); + f % m_refPVNameMap.size(); + m_refPVNameMap[vtx] = f.str(); + } + return m_refPVNameMap[vtx]; + } + //-------------------------------------------------------------------------- + // Lookup name for SV -- add as next if not yet known + //-------------------------------------------------------------------------- + std::string BPhysTrackVertexMapTool::svName(const xAOD::Vertex* vtx) { + + if ( m_svNameMap.find(vtx) == m_svNameMap.end() ) { + boost::format f("SV%03d"); + f % m_svNameMap.size(); + m_svNameMap[vtx] = f.str(); + } + return m_svNameMap[vtx]; + } + //-------------------------------------------------------------------------- + // Lookup name for ID track -- add as next if not yet known + //-------------------------------------------------------------------------- + std::string + BPhysTrackVertexMapTool::idTrackName(const xAOD::TrackParticle* track) { + + if ( m_idTrackNameMap.find(track) == m_idTrackNameMap.end() ) { + boost::format f("T%04d"); + f % m_idTrackNameMap.size(); + m_idTrackNameMap[track] = f.str(); + } + return m_idTrackNameMap[track]; + } + //-------------------------------------------------------------------------- + // Print Track information to string -- optionally adding PVs or refPVs + //-------------------------------------------------------------------------- + std::string + BPhysTrackVertexMapTool::idTrackToString(const xAOD::TrackParticle* track, + unsigned int indent, + bool withPV, bool withRefPV, + bool withSV) { + + std::string sind(indent, ' '); + boost::format f1("%s %-5s %p (%10.4f, %10.4f, %10.4f) VL %p"); + f1 % sind % idTrackName(track) % track + % track->pt() % track->eta() % track->phi(); //% track->vertex(); REMOVE IN MIGRATION TO MAKER + std::string str = f1.str(); + if ( withPV ) { + TrackToVertexMap_t::iterator it = m_idTrackToPVMap.find(track); + if ( it != m_idTrackToPVMap.end() ) { + for ( auto vtx : it->second ) { + str += "\n" + pvToString(vtx, indent+2, false); + } + } else { + boost::format f2("\n%s %s"); + f2 % sind % "NOPV"; + str += f2.str(); + + } + } + if ( withRefPV ) { + TrackToVertexMap_t::iterator it = m_idTrackToRefPVMap.find(track); + if ( it != m_idTrackToRefPVMap.end() ) { + for ( auto vtx : it->second ) { + str += "\n" + refPVToString(vtx, indent+2, false); + } + } else { + boost::format f2("\n%s %s"); + f2 % sind % "NORV"; + str += f2.str(); + } + } + if ( withSV ) { + TrackToVertexMap_t::iterator it = m_idTrackToSVMap.find(track); + if ( it != m_idTrackToSVMap.end() ) { + for ( auto vtx : it->second ) { + str += "\n" + svToString(vtx, indent+2, false); + } + } else { + boost::format f2("\n%s %s"); + f2 % sind % "NOSV"; + str += f2.str(); + } + } + return str; + } + //-------------------------------------------------------------------------- + // Print PV information to string -- optionally adding tracks + //-------------------------------------------------------------------------- + std::string + BPhysTrackVertexMapTool::pvToString(const xAOD::Vertex* vtx, + unsigned int indent, + bool withTracks) { + + std::string sind(indent, ' '); + boost::format f1("%s %-5s %p (%10.4f, %10.4f, %10.4f) NT %4d VT %d"); + f1 % sind % pvName(vtx) % vtx % vtx->x() % vtx->y() % vtx->z() + % vtx->nTrackParticles() % vtx->vertexType(); + std::string str = f1.str(); + if ( withTracks ) { + for (size_t i=0; i < vtx->nTrackParticles(); ++i) { + boost::format f2("\n%s %4d %s"); + f2 % sind % i + % idTrackToString(vtx->trackParticle(i), 0, false, false); + str += f2.str(); + } // for + } + + return str; + } + //-------------------------------------------------------------------------- + // Print refitted PV information to string -- optionally adding tracks + //-------------------------------------------------------------------------- + std::string + BPhysTrackVertexMapTool::refPVToString(const xAOD::Vertex* vtx, + unsigned int indent, + bool withTracks) { + + std::string sind(indent, ' '); + boost::format f1("%s %-5s %p (%10.4f, %10.4f, %10.4f) NT %4d VT %d"); + f1 % sind % refPVName(vtx) % vtx % vtx->x() % vtx->y() % vtx->z() + % vtx->nTrackParticles() % vtx->vertexType(); + std::string str = f1.str(); + if ( withTracks ) { + for (size_t i=0; i < vtx->nTrackParticles(); ++i) { + boost::format f2("\n%s %4d %s"); + f2 % sind % i + % idTrackToString(vtx->trackParticle(i), 0, false, false); + str += f2.str(); + } // for + } + + return str; + } + //-------------------------------------------------------------------------- + // Print SV information to string -- optionally adding tracks + //-------------------------------------------------------------------------- + std::string + BPhysTrackVertexMapTool::svToString(const xAOD::Vertex* vtx, + unsigned int indent, + bool withTracks, + bool withMasses) { + + std::string sind(indent, ' '); + boost::format f1("%s %-5s %p (%10.4f, %10.4f, %10.4f) NT %4d VT %d"); + f1 % sind % svName(vtx) % vtx % vtx->x() % vtx->y() % vtx->z() + % vtx->nTrackParticles() % vtx->vertexType(); + std::string str = f1.str(); + if ( withMasses && m_hypoName != "__NONE__" ) { + // vector of possible hypo names + std::vector<std::string> hypoNames = getTokens(m_hypoName, "|;/"); + for ( auto hypoName : hypoNames ) { + BPhysHypoHelper bhh(hypoName, vtx); + float bMass = bhh.mass(); + float bMassErr = bhh.massErr(); + float bMucMass = getFloat(hypoName+"_MUCALC_mass", vtx); + float bMucMassErr = getFloat(hypoName+"_MUCALC_massErr", vtx); + if ( bMass > 0. || bMassErr > 0. + || bMucMass > 0. || bMucMassErr > 0. ) { + boost::format f3("\n%s %-10s : mass : (%15.4f +/- %15.4f) MeV"); + + boost::format f4("\n%s %-10s : m(MUCALC): (%15.4f +/- %15.4f) MeV"); + f3 % sind % hypoName % bMass % bMassErr; + f4 % sind % hypoName % bMucMass % bMucMassErr; + str += f3.str() + f4.str(); + } // if one > 0. + } // for hypoNames + } // if withMasses + if ( withTracks ) { + for (size_t i=0; i < vtx->nTrackParticles(); ++i) { + boost::format f2("\n%s %4d %s"); + f2 % sind % i + % idTrackToString(vtx->trackParticle(i), 0, false, false); + str += f2.str(); + } // for + } + return str; + } + //-------------------------------------------------------------------------- + // Print track container information to string + // -- optionally adding PVs and refitted PVs + //-------------------------------------------------------------------------- + std::string + BPhysTrackVertexMapTool::idTracksToString(const xAOD::TrackParticleContainer* + tpc, + unsigned int indent, + bool withPV, + bool withRefPV, + bool withSV) { + + std::string str; + std::string sind(indent, ' '); + str += sind + "ID tracks: (" + std::to_string(tpc->size()) + ")\n"; + str += sind + std::string(80-indent, '-'); + // loop over ID tracks + for (xAOD::TrackParticleContainer::const_iterator trkItr = tpc->begin(); + trkItr != tpc->end(); ++trkItr) { + const xAOD::TrackParticle* track = *trkItr; + str += "\n" + + idTrackToString(track, indent+2, withPV, withRefPV, withSV); + } + return str; + } + //-------------------------------------------------------------------------- + // Print PV container information to string -- optionally adding tracks + //-------------------------------------------------------------------------- + std::string + BPhysTrackVertexMapTool::pvsToString(const xAOD::VertexContainer* pvc, + unsigned int indent, + bool withTracks) { + + std::string str; + std::string sind(indent, ' '); + str += sind + "Primary vertices: (" + std::to_string(pvc->size()) + ")\n"; + str += sind + std::string(80-indent, '-'); + for (xAOD::VertexContainer::const_iterator vtxItr = pvc->begin(); + vtxItr != pvc->end(); ++vtxItr) { + const xAOD::Vertex* vtx = *vtxItr; + str += "\n" + pvToString(vtx, indent+2, withTracks); + } // for + + return str; + } + //-------------------------------------------------------------------------- + // Print refitted PV container information to string + // -- optionally adding tracks + //-------------------------------------------------------------------------- + std::string + BPhysTrackVertexMapTool::refPVsToString(const xAOD::VertexContainer* rpvc, + unsigned int indent, + bool withTracks) { + + std::string str; + std::string sind(indent, ' '); + str += sind + "Refitted primary vertices: (" + std::to_string(rpvc->size()) + ")\n"; + str += sind + std::string(80-indent, '-'); + for (xAOD::VertexContainer::const_iterator vtxItr = rpvc->begin(); + vtxItr != rpvc->end(); ++vtxItr) { + const xAOD::Vertex* vtx = *vtxItr; + str += "\n" + refPVToString(vtx, indent+2, withTracks); + } // for + + return str; + } + //-------------------------------------------------------------------------- + // Print SV container information to string -- optionally adding tracks + //-------------------------------------------------------------------------- + std::string + BPhysTrackVertexMapTool::svsToString(const xAOD::VertexContainer* svc, + unsigned int indent, + bool withTracks, + bool withMasses) { + + std::string str; + std::string sind(indent, ' '); + str += sind + "Secondary vertices: (" + std::to_string(svc->size()) + ")\n"; + str += sind + std::string(80-indent, '-'); + for (xAOD::VertexContainer::const_iterator vtxItr = svc->begin(); + vtxItr != svc->end(); ++vtxItr) { + const xAOD::Vertex* vtx = *vtxItr; + str += "\n" + svToString(vtx, indent+2, withTracks, withMasses); + } // for + + return str; + } + //-------------------------------------------------------------------------- + // Print a summary of all maps to string -- optionally adding a prefix + //-------------------------------------------------------------------------- + std::string BPhysTrackVertexMapTool::summaryToString(std::string prefix) { + + boost::format form("%s\n\nRun: %d Event: %d\n\n"); + form % name() % m_cachedRun % m_cachedEvent; + std::string dstr = + wrapLines("\n"+form.str() + + pvsToString(m_pvtxContainer, 0, true) + "\n\n" + + refPVsToString(m_refPVContainer, 0, true) + "\n\n" + + svsToString(m_svtxContainer, 0, true, true) + "\n\n" + + idTracksToString(m_tracks, 0, true, true, true) + "\n", + prefix); + + return dstr; + } + //-------------------------------------------------------------------------- + // Pick up a float from StoreGate. + //-------------------------------------------------------------------------- + float BPhysTrackVertexMapTool::getFloat(std::string name, + const xAOD::Vertex* b) { + + float res = -999999.; + + SG::AuxElement::Accessor<float> floatAcc(name); + if ( floatAcc.isAvailable(*b) ) res = floatAcc(*b); + + return res; + } + //-------------------------------------------------------------------------- + // Tokenize a string using certain separators + //-------------------------------------------------------------------------- + std::vector<std::string> BPhysTrackVertexMapTool + ::getTokens(std::string input, std::string seperators) { + + std::vector<std::string> tokens; + boost::char_separator<char> sep(seperators.c_str()); + typedef boost::tokenizer<boost::char_separator<char> > Tokenizer_t; + Tokenizer_t tokenizer(input, sep); + for (auto& token : tokenizer) { + tokens.push_back(token); + } + return tokens; + } + //-------------------------------------------------------------------------- +} // namespace xAOD diff --git a/PhysicsAnalysis/BPhys/BPhysTools/Root/SimpleEncrypter.cxx b/PhysicsAnalysis/BPhys/BPhysTools/Root/SimpleEncrypter.cxx new file mode 100644 index 00000000000..77cc5475f19 --- /dev/null +++ b/PhysicsAnalysis/BPhys/BPhysTools/Root/SimpleEncrypter.cxx @@ -0,0 +1,516 @@ +/* + Copyright (C) 2002-2018 CERN for the benefit of the ATLAS collaboration +*/ + +// system include: +#include <climits> +#include <vector> +#include <algorithm> +#include <cstdlib> +#include <ctime> +#include <cmath> + +// ROOT includes +#include <TString.h> + +// Local include(s): +#include "BPhysTools/SimpleEncrypter.h" + +namespace xAOD { + + //-------------------------------------------------------------------------- + // Private static constants + //-------------------------------------------------------------------------- + const SimpleEncrypter::ULLI_t SimpleEncrypter::m_MAXRANGE = + (SimpleEncrypter::ULLI_t)pow(std::numeric_limits<ULLI_t>::max(), 0.25); + const SimpleEncrypter::ULLI_t SimpleEncrypter::m_MINRANGE = + (SimpleEncrypter::ULLI_t)SimpleEncrypter::m_MAXRANGE/10; + const unsigned int SimpleEncrypter::m_MAXHEXDIGITS = + (unsigned int)(log(pow(SimpleEncrypter::m_MAXRANGE,2))/log(16.))+3; + + //-------------------------------------------------------------------------- + // Public methods + //-------------------------------------------------------------------------- + + //-------------------------------------------------------------------------- + // Constructor + //-------------------------------------------------------------------------- + SimpleEncrypter::SimpleEncrypter(const std::string& name) : + asg::AsgMessaging(name), m_n(0), m_e(0), m_d(0), + m_isOkForEnc(false), m_isOkForDec(false) { + + // initialize random number generator + srand(static_cast<unsigned>(time(0))); + } + + //-------------------------------------------------------------------------- + // Destructor + //-------------------------------------------------------------------------- + SimpleEncrypter::~SimpleEncrypter() { + + } + + //-------------------------------------------------------------------------- + // Generation of key pair as pair of hex strings + //-------------------------------------------------------------------------- + std::pair<std::string, std::string> SimpleEncrypter::genKeyPair() { + + // default preset + std::pair<std::string, std::string> keys = + std::make_pair("__NO_PRIV_KEY__", "__NO_PUB_KEY__"); + + // generate keys + genKeyPairInternal(); + + if ( isOkForEnc() && isOkForDec() ) { + keys = std::make_pair(getPrivKey(), getPubKey()); + } + return keys; + } + + //-------------------------------------------------------------------------- + // Set private key + //-------------------------------------------------------------------------- + void SimpleEncrypter::setPrivKey(std::string keystr) { + + std::pair<ULLI_t, ULLI_t> keys = decodeKeyString(keystr); + + if ( m_n > 0 && m_n != keys.first ) { + ATH_MSG_WARNING("RSA module already set!"); + } + m_n = keys.first; + m_d = keys.second; + m_isOkForDec = false; + } + //-------------------------------------------------------------------------- + // Set public key + //-------------------------------------------------------------------------- + void SimpleEncrypter::setPubKey(std::string keystr) { + + std::pair<ULLI_t, ULLI_t> keys = decodeKeyString(keystr); + + if ( m_n > 0 && m_n != keys.second ) { + ATH_MSG_WARNING("RSA module already set!"); + } + m_e = keys.first; + m_n = keys.second; + m_isOkForEnc = false; + } + //-------------------------------------------------------------------------- + // Get private key + //-------------------------------------------------------------------------- + std::string SimpleEncrypter::getPrivKey() const { + + return keyToString(m_n, m_d); + } + //-------------------------------------------------------------------------- + // Get public key + //-------------------------------------------------------------------------- + std::string SimpleEncrypter::getPubKey() const { + + return keyToString(m_e, m_n); + } + //-------------------------------------------------------------------------- + // Encrypt unsigned integer value + //-------------------------------------------------------------------------- + SimpleEncrypter::ULLI_t SimpleEncrypter::encrypt(ULLI_t a) { + + ULLI_t b = a; + + if ( isOkForEnc() ) { + b = encryptFPECycle(a); + } + return b; + } + //-------------------------------------------------------------------------- + // Decrypt unsigned integer value + //-------------------------------------------------------------------------- + SimpleEncrypter::ULLI_t SimpleEncrypter::decrypt(ULLI_t a) { + + ULLI_t b = a; + + if ( isOkForDec() ) { + b = decryptFPECycle(a); + } + return b; + } + //-------------------------------------------------------------------------- + // Encrypt positive float value + //-------------------------------------------------------------------------- + float SimpleEncrypter::encrypt(float a) { + + float b = a; + + if ( a > 0. ) { + if ( isOkForEnc() ) { + ULLI_t ia = floatBitsToInt(a); + ULLI_t ib = encryptFPECycle(ia); + b = intBitsToFloat(ib); + } + } else { + ATH_MSG_WARNING("Encrypt: Float value not positive: " + << a << Form(" (%a) !", a)); + } // if a > 0 + return b; + } + + //-------------------------------------------------------------------------- + // Decrypt positive float value + //-------------------------------------------------------------------------- + float SimpleEncrypter::decrypt(float a) { + + float b = a; + + // As nan is a valid encrypted value, decrypt it as well. + if ( a > 0. || std::isnan(a) ) { + if ( isOkForDec() ) { + ULLI_t ia = floatBitsToInt(a); + ULLI_t ib = decryptFPECycle(ia); + b = intBitsToFloat(ib); + } + } else { + ATH_MSG_WARNING("Decrypt: Float value not positive: " + << a << Form(" (%a) !", a)); + } // if a > 0 + return b; + } + + //-------------------------------------------------------------------------- + // Private methods + //-------------------------------------------------------------------------- + + //-------------------------------------------------------------------------- + // Generate numeric representation of the keys + //-------------------------------------------------------------------------- + void SimpleEncrypter::genKeyPairInternal() { + + // Generate prime numbers p != q + ULLI_t p(1); + ULLI_t q(1); + // Euler's phi function + ULLI_t phi(1); + + // reset encryption and decryption exponent + m_e = 0; + m_d = 0; + while ( p == q || m_e < 2 || m_e >= phi || m_d < 2 + || m_e*m_d % phi != 1 ) { + double dlog2 = 0.; + while ( p == q || dlog2 < 0.1 || dlog2 > 30. ) { + p = genPrime(); + q = genPrime(); + dlog2 = fabs(log2(p)-log2(q)); + } // inner while loop + phi = (p-1)*(q-1); + m_n = p*q; + m_e = genCoprime(phi); + m_d = genDecryptionExponent(phi, m_e); + } // outer while loop + m_isOkForDec = false; + m_isOkForEnc = false; + } + //-------------------------------------------------------------------------- + // Find a prime number + //-------------------------------------------------------------------------- + SimpleEncrypter::ULLI_t SimpleEncrypter::genPrime() const { + + ULLI_t t = (m_MINRANGE + rand()) % (m_MAXRANGE-1); + do { + t++; + } while ( !isPrime(t) || t < m_MINRANGE ); + return t; + } + //-------------------------------------------------------------------------- + // Test for being a prime number + //-------------------------------------------------------------------------- + bool SimpleEncrypter::isPrime(ULLI_t n) const { + + bool isPrime = true; + if (n != 2) { + for (LLI_t i = 2; i < (LLI_t)sqrt(n) + 1; ++i) { + if (n % i == 0) { + isPrime = false; + break; + } + } + } + return isPrime; + } + //-------------------------------------------------------------------------- + // Greatest common denominator + //-------------------------------------------------------------------------- + SimpleEncrypter::ULLI_t + SimpleEncrypter::greatestCommonDenominator(ULLI_t n1, ULLI_t n2) const { + + std::vector<LLI_t> r; + LLI_t i = 1; + r.push_back(std::max(n1, n2)); + r.push_back(std::min(n1, n2)); + while (r[i] != 0) { + ++i; + r.push_back(r[i-2] % r[i-1]); + } + return r[i-1]; + } + //-------------------------------------------------------------------------- + // Find coprime number + //-------------------------------------------------------------------------- + SimpleEncrypter::ULLI_t SimpleEncrypter::genCoprime(ULLI_t n) const { + + // make sure coprime is larger than 5th Fermat number (2^16+1 = 65537) + ULLI_t i = (65537 + rand()) % (m_MAXRANGE -1); + do { + ++i; + } while (greatestCommonDenominator(n, i) != 1); + return i; + } + //-------------------------------------------------------------------------- + // Find decryption exponent + //-------------------------------------------------------------------------- + SimpleEncrypter::ULLI_t + SimpleEncrypter::genDecryptionExponent(ULLI_t phi, ULLI_t e) const { + + for (ULLI_t i=1; i<m_MAXRANGE; ++i) { + if ( ((phi * i + 1) % e) == 0 ) { + return (ULLI_t)((phi * i + 1) / e); + } + } + return 0; + } + //-------------------------------------------------------------------------- + // Convert key to a hex string + //-------------------------------------------------------------------------- + std::string SimpleEncrypter::keyToString(ULLI_t a, ULLI_t b) const { + + // length of keys w.r.t. hex digits + unsigned int ra = (unsigned int)(log(a)/log(16.))+1; + unsigned int rb = (unsigned int)(log(b)/log(16.))+1; + + // random numbers for padding + unsigned int r1 = rand() & ((1 << 4*(m_MAXHEXDIGITS-ra))-1); + unsigned int r2 = rand() & ((1 << 4*(m_MAXHEXDIGITS-rb))-1); + + // format string + TString tstr = Form("%02x%02x%02x%0*x%0*llx%0*x%0*llx", + m_MAXHEXDIGITS, ra, rb, + m_MAXHEXDIGITS-ra, r1, ra, a, + m_MAXHEXDIGITS-rb, r2, rb, b); + + return std::string(tstr.Data()); + } + //-------------------------------------------------------------------------- + // Convert hex string to two integers + //-------------------------------------------------------------------------- + std::pair<SimpleEncrypter::ULLI_t, SimpleEncrypter::ULLI_t> + SimpleEncrypter::decodeKeyString(std::string hstr) const { + + std::pair<ULLI_t, ULLI_t> keys(0,0); + + TString str(hstr); + if (str.IsHex() && str.Length() > 3) { + str.ToLower(); + unsigned int ndigits = strtoul(TString(str(0,2)).Data(), nullptr, 16); + unsigned int ra = strtoul(TString(str(2,2)).Data(), nullptr, 16); + unsigned int rb = strtoul(TString(str(4,2)).Data(), nullptr, 16); + if ( str.Length() == (int)(2*ndigits + 6) ) { + keys.first = strtoll(TString(str(ndigits+6-ra, ra)).Data(), + nullptr, 16); + keys.second = strtoll(TString(str(2*ndigits+6-rb, rb)).Data(), + nullptr, 16); + } else { + ATH_MSG_ERROR("Private/public key must be a hex string of " << + 2*m_MAXHEXDIGITS+6 << " digits!"); + } // if Length() + } else { + ATH_MSG_ERROR("Private/public key must be a hex string of " << + 2*m_MAXHEXDIGITS+6 << " digits!"); + } // if IsHex() ... + + return keys; + } + //-------------------------------------------------------------------------- + // Interpret bits of positive floating point number as integer + //-------------------------------------------------------------------------- + SimpleEncrypter::ULLI_t SimpleEncrypter::floatBitsToInt(float val) const { + + ULLI_t res(0); + + if ( val < 0. ) { + ATH_MSG_ERROR("Float value needs to be positive!"); + } else { + // convert floating point number to ULLI_t if size fits + if ( sizeof(float) <= sizeof(ULLI_t) ) { + // check whether a quick conversion is possible + if ( sizeof(float) == sizeof(int) ) { + int* p = reinterpret_cast<int*>(&val); + res = *p; + } else { + // do a slow conversion + char* pval = reinterpret_cast<char*>(&val); + // loop over bytes + for (unsigned int i=0; i<sizeof(float); ++i) { + // loop over bits + for (unsigned int j=0; j<CHAR_BIT; ++j) { + unsigned int n = i*CHAR_BIT + j; + unsigned int bit = (*(pval+i) >> j) & 1; + if ( bit > 0 ) res |= 1 << n; + } // for bits + } // for bytes + } // if sizeof + } else { + ATH_MSG_ERROR("sizeof(float) > sizeof(ULLI_t): " + << sizeof(float) << " > " << sizeof(LLI_t)); + } // if sizeof + } // if val < 0. + + return res; + } + //-------------------------------------------------------------------------- + // Interpret bits of positive integer as floating point number + //-------------------------------------------------------------------------- + float SimpleEncrypter::intBitsToFloat(ULLI_t val) const { + + float res(0.); + + // number of bits needed + unsigned int r = (int)(std::log2(val))+1; + + // convert ULLI_t to floating point number if size fits + if ( sizeof(float)*CHAR_BIT >= r ) { + // check whether a quick conversion is possible + if ( sizeof(float) == sizeof(int) ) { + float* p = reinterpret_cast<float*>(&val); + res = *p; + } else { + // do a slow conversion + char* pres = reinterpret_cast<char*>(&res); + // loop over bytes + for (unsigned int i=0; i<sizeof(float); ++i) { + // loop over bits + for (unsigned int j=0; j<CHAR_BIT; ++j) { + unsigned int n = i*CHAR_BIT + j; + unsigned int bit = (val >> n) & 1; + if ( bit > 0 ) *(pres+i) |= 1 << j; + } // for bits + } // for bytes + } // if sizeof + } else { + ATH_MSG_WARNING("sizeof(float)*CHAR_BIT < r: " + << sizeof(float)*CHAR_BIT << " < " << r); + } // if sizeof + + return res; + } + //-------------------------------------------------------------------------- + // Encrypt using format preserving encryption w.r.t. RSA modulus + // via cycling + //-------------------------------------------------------------------------- + SimpleEncrypter::ULLI_t SimpleEncrypter::encryptFPECycle(ULLI_t a) const { + + ULLI_t enc = 0; + if ( a > 0 ) { + ULLI_t r = (int)(std::log2(m_n)); + ULLI_t rmask = pow(2,r)-1; + ULLI_t c = a & rmask; + ULLI_t b = a - c; + do { + c = encryptInternal(c); + } while ( c > rmask ); + enc = b + c; + } // if + return enc; + } + //-------------------------------------------------------------------------- + // Decrypt using format preserving encryption w.r.t. RSA modulus + // via cycling + //-------------------------------------------------------------------------- + SimpleEncrypter::ULLI_t SimpleEncrypter::decryptFPECycle(ULLI_t enc) const { + + ULLI_t dec = 0; + if ( enc > 0 ) { + ULLI_t r = (int)(std::log2(m_n)); + ULLI_t rmask = pow(2,r)-1; + ULLI_t d = enc & rmask; + ULLI_t b = enc - d; + do { + d = decryptInternal(d); + } while ( d > rmask ); + dec = d + b; + } // if + return dec; + } + //-------------------------------------------------------------------------- + // Encrypt integer + //-------------------------------------------------------------------------- + SimpleEncrypter::ULLI_t SimpleEncrypter::encryptInternal(ULLI_t x) const { + + return powerMod(x, m_e, m_n); + } + //-------------------------------------------------------------------------- + // Decrypt integer + //-------------------------------------------------------------------------- + SimpleEncrypter::ULLI_t SimpleEncrypter::decryptInternal(ULLI_t x) const { + + return powerMod(x, m_d, m_n); + } + //-------------------------------------------------------------------------- + // Exponentiate a with d observing modulus n + //-------------------------------------------------------------------------- + SimpleEncrypter::ULLI_t + SimpleEncrypter::powerMod(ULLI_t a, ULLI_t d, ULLI_t n) const { + + int bin[sizeof(ULLI_t)*CHAR_BIT]; + ULLI_t dec[sizeof(ULLI_t)*CHAR_BIT]; + + ULLI_t r = (ULLI_t)(std::log2(d))+1; + ULLI_t tmp = d; + // decompose exponent into binary number (reverse order!) + for (ULLI_t i=0; i < r; ++i) { + bin[r-i-1] = tmp % 2; + tmp = (LLI_t)(tmp/2); + } // for i + + // perform the exponentiation taking modulus into account + dec[0] = a; + for (ULLI_t i=1; i < r; ++i) { + ULLI_t d2 = dec[i-1]*dec[i-1] % n; + if ( bin[i] > 0 ) d2 *= a; + dec[i] = d2 % n; + } // for i + + return dec[r-1]; + } + //-------------------------------------------------------------------------- + // Check setup readiness for encryption + //-------------------------------------------------------------------------- + bool SimpleEncrypter::isOkForEnc() { + + if ( !m_isOkForEnc ) { + if ( m_n > 0 && m_e > 1 && m_e < m_n ) { + m_isOkForEnc = true; + } else { + ATH_MSG_ERROR("Setup not OK for encryption: public key set?"); + } + } // if ! m_isOkForEnc + + return m_isOkForEnc; + } + + //-------------------------------------------------------------------------- + // Check setup readiness for decryption + //-------------------------------------------------------------------------- + bool SimpleEncrypter::isOkForDec() { + + if ( !m_isOkForDec ) { + if ( m_n > 0 && m_d > 1 && m_d < m_n ) { + m_isOkForDec = true; + } else { + ATH_MSG_ERROR("Setup not OK for decryption: private key set?"); + } + } // if ! m_isOkForDec + + return m_isOkForDec; + } + + //-------------------------------------------------------------------------- +} // namespace xAOD diff --git a/PhysicsAnalysis/BPhys/BPhysTools/src/components/BPhysTools_entries.cxx b/PhysicsAnalysis/BPhys/BPhysTools/src/components/BPhysTools_entries.cxx new file mode 100644 index 00000000000..2298acdeb10 --- /dev/null +++ b/PhysicsAnalysis/BPhys/BPhysTools/src/components/BPhysTools_entries.cxx @@ -0,0 +1,4 @@ +/* + Copyright (C) 2002-2017 CERN for the benefit of the ATLAS collaboration +*/ + diff --git a/PhysicsAnalysis/BPhys/BPhysTools/util/createBlindingKeys.cxx b/PhysicsAnalysis/BPhys/BPhysTools/util/createBlindingKeys.cxx new file mode 100644 index 00000000000..3be9cb51430 --- /dev/null +++ b/PhysicsAnalysis/BPhys/BPhysTools/util/createBlindingKeys.cxx @@ -0,0 +1,83 @@ +/* + Copyright (C) 2002-2017 CERN for the benefit of the ATLAS collaboration +*/ + +/** + * @file creteBlindingKeys.cxx + * @author Wolfgang Walkowiak <Wolfgang.Walkowiak@cern.ch> + * + * @brief Utility to create a set of blinding keys + * + * @param[in] option: -c : perform a quick encoding/decoding check + */ + +// system includes: +#include <iostream> +#include <iomanip> +#include <set> +#include <string> +#include <cstdlib> +#include <ctime> + +// Local include(s): +#include "BPhysTools/SimpleEncrypter.h" + +int main(int argc, char* argv[]) { + + // Enable check (-c flag) + bool doCheck(false); + int nChecks(1); + if ( argc > 1 ) { + std::string arg(argv[1]); + if ( arg == "-c" ) doCheck = true; + } + if ( argc > 2 ) { + nChecks = atoi(argv[2]); + } + + // Helper object + xAOD::SimpleEncrypter senc; + + // Create key pair + std::pair<std::string, std::string> keys = senc.genKeyPair(); + + std::cout << std::endl; + std::cout << "Blinding keys generated:" << std::endl; + std::cout << " Private key: " << keys.first << std::endl; + std::cout << " Public key: " << keys.second << std::endl; + std::cout << std::endl; + + // check that encryption works + if ( doCheck ) { + srand(static_cast<unsigned>(time(0))); + + std::cout << "Encryption test:" << std::endl; + int nOK(0); + for (int i=0; i<nChecks; ++i) { + float val = 10000.* + static_cast <float>(rand())/(static_cast <float> (RAND_MAX)); + // float val = 5267.23; + float enc = senc.encrypt(val); + float dec = senc.decrypt(enc); + if ( dec == val ) ++nOK; + if ( i == 0 || dec != val ) { + std::cout << " Test # " << i << std::endl; + std::cout << " val = " << val << std::endl; + std::cout << " enc = " << enc << std::endl; + std::cout << " dec = " << dec << std::endl; + if ( dec == val ) { + std::cout << " => worked!" << std::endl; + } else { + std::cout << " => FAILED!" << std::endl; + } + } // if + } // for + std::cout << std::endl; + std::cout << "Summary:" << std::endl; + std::cout << " nChecks: " << std::setw(12) << nChecks << std::endl; + std::cout << " nOK : " << std::setw(12) << nOK << std::endl; + std::cout << " nFailed: " << std::setw(12) << nChecks - nOK << std::endl; + } // if + + exit(0); +} diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/CMakeLists.txt b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/CMakeLists.txt new file mode 100644 index 00000000000..357dae19e36 --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/CMakeLists.txt @@ -0,0 +1,26 @@ +# Copyright (C) 2002-2020 CERN for the benefit of the ATLAS collaboration +################################################################################ +# Package: DerivationFrameworkBPhys +################################################################################ + +# Declare the package name: +atlas_subdir( DerivationFrameworkBPhys ) +find_package( ROOT COMPONENTS Core MathCore ) + +# Component(s) in the package: +atlas_add_component( DerivationFrameworkBPhys + DerivationFrameworkBPhys/*.h src/*.cxx src/components/*.cxx + INCLUDE_DIRS ${ROOT_INCLUDE_DIRS} + LINK_LIBRARIES xAODMuon AthenaBaseComps JpsiUpsilonToolsLib + MuonSelectorToolsLib ${ROOT_LIBRARIES} + xAODTracking xAODBPhysLib AthenaKernel RecoToolInterfaces EventPrimitives + DerivationFrameworkInterfaces BPhysToolsLib TrackVertexAssociationToolLib + xAODBase xAODMetaData AsgTools CaloInterfaceLib TrackToCaloLib + xAODEventInfo AthenaPoolUtilities xAODPrimitives TrigDecisionToolLib + BeamSpotConditionsData TrkVertexAnalysisUtilsLib ITrackToVertex + InDetTrackSelectionToolLib InDetV0FinderLib) + + +# Install files from the package: +atlas_install_python_modules( python/*.py POST_BUILD_CMD ${ATLAS_FLAKE8}) +atlas_install_joboptions( share/*.py ) diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/AugOriginalCounts.h b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/AugOriginalCounts.h new file mode 100644 index 00000000000..47c5b166c79 --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/AugOriginalCounts.h @@ -0,0 +1,91 @@ +/* + Copyright (C) 2002-2019 CERN for the benefit of the ATLAS collaboration +*/ + +/** + * @file AugOriginalCounts.h + * + * @brief Augmentation with primary vertex counts (before thinning) + */ + +#ifndef DERIVATIONFRAMEWORKBPHYS_AUGORIGINALCOUNTS_H +#define DERIVATIONFRAMEWORKBPHYS_AUGORIGINALCOUNTS_H + +#include "DerivationFrameworkInterfaces/IAugmentationTool.h" + + +#include "AthenaBaseComps/AthAlgTool.h" +#include "StoreGate/WriteDecorHandleKey.h" + +#include "xAODTracking/TrackParticleContainer.h" +#include "xAODTracking/VertexContainer.h" +#include "xAODEventInfo/EventInfo.h" + +#include <string> + + +#include "AthenaBaseComps/AthAlgTool.h" +#include "DerivationFrameworkInterfaces/IAugmentationTool.h" +#include "xAODTracking/TrackParticleContainer.h" +#include "xAODTracking/VertexContainer.h" +#include <string> +#include "xAODEventInfo/EventInfo.h" +#include <StoreGate/WriteDecorHandleKey.h> + +namespace DerivationFramework { + /// + /// @class AugOriginalCounts + /// + /// @brief Augmentation with primary vertex counts (before thinning) + /// + /// This tool adds primary vertex counts and track counts + /// to the EventInfo container in order to preserve them in + /// case the primary vertex or track collections are thinned. + /// + /// ### Job options + /// <table border="0"> + /// <tr><th align="left">Name</td> + /// <th align="left">Description</th></tr> + /// <tr><td>TrackContainer</td> + /// <td>name of the TrackParticle container to be used</td> + /// </tr> + /// <tr><td>VertexContainer</td> + /// <td>name of the Vertex container to be used</td> + /// </tr> + /// <tr><td>AddPVCountsByType</td> + /// <td>add PV counts by PV type (default: false)</td> + /// </td> + /// </table> + /// + class AugOriginalCounts : public AthAlgTool, public IAugmentationTool { + public: + /// @brief Main constructor + AugOriginalCounts(const std::string& t, const std::string& n, + const IInterface* p); + /// @brief Main method called for each event + virtual StatusCode addBranches() const override; + virtual StatusCode initialize() override; + private: + /// + /// @name job options + /// @{ + SG::WriteDecorHandleKey<xAOD::EventInfo> m_OrigPVNTracks{this, "DO_NOT_SET1", "", "internal property"}; + SG::WriteDecorHandleKey<xAOD::EventInfo> m_OrigNTracksKeys{this, "DO_NOT_SET2", "", "internal property"}; + SG::WriteDecorHandleKey<xAOD::EventInfo> m_OrigNtype0{this, "DO_NOT_SET3", "", "internal property"}; + SG::WriteDecorHandleKey<xAOD::EventInfo> m_OrigNtype1{this, "DO_NOT_SET4", "", "internal property"}; + SG::WriteDecorHandleKey<xAOD::EventInfo> m_OrigNtype2{this, "DO_NOT_SET5", "", "internal property"}; + SG::WriteDecorHandleKey<xAOD::EventInfo> m_OrigNtype3{this, "DO_NOT_SET6", "", "internal property"}; + SG::WriteDecorHandleKey<xAOD::EventInfo> m_OrigNtypeUnknown{this, "DO_NOT_SET7", "", "internal property"}; + + SG::WriteDecorHandleKey<xAOD::VertexContainer> m_OrigSqrtPt2Sum{this, "DO_NOT_SET8", "", "internal property"}; + SG::WriteDecorHandleKey<xAOD::VertexContainer> m_d_nPVTracks{this, "DO_NOT_SET9", "", "internal property"}; + SG::ReadHandleKey<xAOD::TrackParticleContainer> m_TrackContainername; + SG::ReadHandleKey<xAOD::VertexContainer> m_PVContainername; + bool m_addPVCountsByType; + bool m_addNTracksToPVs; + bool m_addSqrtPt2SumToPVs; + /// @} + }; +} + +#endif // DERIVATIONFRAMEWORKBPHYS_AUGORIGINALCOUNTS_H diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/BMuonTrackIsoTool.h b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/BMuonTrackIsoTool.h new file mode 100644 index 00000000000..b670df14553 --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/BMuonTrackIsoTool.h @@ -0,0 +1,111 @@ +/* + Copyright (C) 2002-2017 CERN for the benefit of the ATLAS collaboration +*/ + +//============================================================================ +// BMuonTrackIsoTool.h +//============================================================================ +// +// Author : Wolfgang Walkowiak <Wolfgang.Walkowiak@cern.ch.> +// Changes: +// +// Add muon track isolation information for different configurations, +// different track selections and different PV-to-SV association methods. +// +// For an usage example see BPHY8.py . +// +//============================================================================ +// +#ifndef DERIVATIONFRAMEWORK_BMuonTrackIsoTool_H +#define DERIVATIONFRAMEWORK_BMuonTrackIsoTool_H + +#include "DerivationFrameworkBPhys/BPhysVertexTrackBase.h" +#include "xAODMuon/MuonContainer.h" +#include "boost/multi_array.hpp" + +namespace InDet { + class IInDetTrackSelectionTool; +} + +namespace DerivationFramework { + + class BMuonTrackIsoTool : virtual public BPhysVertexTrackBase { + + private: + typedef BPhysVertexTrackBase super; + + // + // internal helper class + // + protected: + class MuIsoItem : public BaseItem { + + public: + MuIsoItem(std::string Name="_none_", std::string Bname="muiso", + std::string Prefix=""); + virtual ~MuIsoItem(); + + virtual void resetVals(); + virtual void copyVals(const BaseItem& item); + virtual void copyVals(const MuIsoItem& item); + virtual void fill(double isoValue=-2., int nTracks=-1, + const xAOD::Muon* muon=NULL); + virtual std::string muIsoName(); + virtual std::string nTracksName(); + virtual std::string muLinkName(); + + public: + mutable std::vector<float> vIsoValues; + mutable std::vector<int> vNTracks; + mutable MuonBag vMuons; + }; // MuIsoItem + + public: + BMuonTrackIsoTool(const std::string& t, const std::string& n, + const IInterface* p); + + protected: + // Hook methods + virtual StatusCode initializeHook(); + virtual StatusCode finalizeHook(); + + virtual StatusCode addBranchesVCSetupHook(size_t ivc) const; + + virtual StatusCode addBranchesSVLoopHook(const xAOD::Vertex* vtx) const; + + virtual StatusCode calcValuesHook(const xAOD::Vertex* vtx, + const unsigned int ipv, + const unsigned int its, + const unsigned int itt) const; + virtual bool fastFillHook(const xAOD::Vertex* vtx, + const int ipv) const; + + private: + virtual StatusCode saveIsolation(const xAOD::Vertex* vtx) const; + virtual void initResults(); + virtual void setResultsPrefix(std::string prefix) const; + + virtual std::string buildBranchName(unsigned int ic, + unsigned int its, + unsigned int ipv, + unsigned int itt) const; + + private: + // job options + std::string m_muonContainerName; + std::vector<double> m_isoConeSizes; + std::vector<double> m_isoTrkImpLogChi2Max; + std::vector<int> m_isoDoTrkImpLogChi2Cut; + + // containers + mutable const xAOD::MuonContainer* m_muons; + + + // results array + typedef boost::multi_array<MuIsoItem, 4> MuIsoItem4_t; + mutable MuIsoItem4_t m_results; + + }; // BMuonTrackIsoTool +} // namespace + +#endif // DERIVATIONFRAMEWORK_BMuonTrackIsoTool_H diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/BPhysAddMuonBasedInvMass.h b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/BPhysAddMuonBasedInvMass.h new file mode 100644 index 00000000000..3b87f752824 --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/BPhysAddMuonBasedInvMass.h @@ -0,0 +1,304 @@ +/* + Copyright (C) 2002-2017 CERN for the benefit of the ATLAS collaboration +*/ + +/** + * @file BPhysAddMuonBasedInvMass.h + * @author Wolfgang Walkowiak <wolfgang.walkowiak@cern.ch> + * + * @brief Augmentation with muon-information based invariant mass. + * + */ +// +#ifndef DERIVATIONFRAMEWORK_BPhysAddMuonBasedInvMass_H +#define DERIVATIONFRAMEWORK_BPhysAddMuonBasedInvMass_H + +#include <string> +#include <vector> +#include <set> +#include <map> + +#include "AthenaBaseComps/AthAlgTool.h" +#include "DerivationFrameworkInterfaces/IAugmentationTool.h" +#include "GaudiKernel/ToolHandle.h" +#include "xAODBPhys/BPhysHelper.h" +#include "EventPrimitives/EventPrimitives.h" +#include "ITrackToVertex/ITrackToVertex.h" + +namespace DerivationFramework { + // + // typedefs -- to abbreviate long lines + // + typedef std::vector<const xAOD::TrackParticle*> TrackBag; + typedef std::vector<const xAOD::Muon*> MuonBag; + + /// + /// @class BPhysAddMuonBasedInvMass + /// @author Wolfgang Walkowiak <wolfgang.walkowiak@cern.ch> + /// + /// @brief Augment secondary vertices with muon-information-based mass. + /// + /// Add muon-information based invarient mass to secondary vertices using + /// a four vector sum. Optionally, it also calculates the minimum + /// chi2 for all muon tracks of the secondary vertex candidate w.r.t. + /// any primary vertex matching the selection criteria. + /// + /// ### Job options provided by this class: + /// <table border="0"> + /// <tr><th align="left">Name</th> + /// <th align="left">Description</th> + /// </tr> + /// <tr><td>BranchPrefix</td> + /// <td>assign the prefix of added branches + /// (possibly the derivation format's name)</td> + /// </tr> + /// <tr><td>VertexContainerName</td> + /// <td>name of container for vertices</td> + /// </tr> + /// <tr><td>TrkMasses</td> + /// <td>ordered list of track masses + /// (Important to keep proper order: J/psi muons go first!)</td> + /// </tr> + /// <tr><td>TrackToVertexTool</td> + /// <td>ToolHandle for track-to-vertex tool</td> + /// </tr> + /// <tr><td>AdjustToMuonKinematics</td> + /// <td>Adjust the primary track particle's kinematics to the one of + /// the muon.</td> + /// </tr> + /// <tr><td valign="top">AddMinChi2ToAnyPVMode</td> + /// <td>mode of minLogChi2ToAnyPV calculation: (default: 0) + /// <table border="0"> + /// <tr><th align="left">Value</th> + /// <th align="left">Explanation<th></tr> + /// <tr><td> 0 </td><td>no such calculation</td></tr> + /// <tr><td> 1 </td><td>use all PVs of requested + /// type(s)</td></tr> + /// <tr><td> 2 </td><td>exclude PVs associated to SVs</td></tr> + /// <tr><td> 3 </td><td>replace PVs associated to SVs by + /// corresponding refitted PVs</td></tr> + /// </table></td> + /// </tr> + /// <tr><td>PrimaryVertexContainerName</td> + /// <td>name of container for primary vertices</td> + /// </tr> + /// <tr><td>MinNTracksInPV</td> + /// <td>minimum number of tracks in PV + /// for PV to be considered in calculation + /// of minChi2MuToAnyPV variable.</td> + /// </tr> + /// <tr><td>PVTypesToConsider</td> + /// <td>list of primary vertex types to consider + /// (default: {1, 3})</td> + /// </tr> + /// <tr><td>DoVertexType</td> + /// <td>PV-to-SV association types to be considered (bitwise variable, + /// see xAODBPhys::BPhysHelper)<br> + /// Note: only needed for AddMinChi2ToAnyPVMode > 1</td> + /// </tr> + /// </table> + /// + /// @note + /// + /// For a usage example see BPHY8.py . + /// + class BPhysAddMuonBasedInvMass : virtual public AthAlgTool, + virtual public IAugmentationTool { + + public: + /// + /// @brief Main contructor + /// + BPhysAddMuonBasedInvMass(const std::string& t, const std::string& n, + const IInterface* p); + + /// @brief Initialize augmentation tool. + virtual StatusCode initialize(); + /// @brief Finalize augmentation tool. + virtual StatusCode finalize(); + /// @brief Main method called for each event. + virtual StatusCode addBranches() const; + + protected: + /// + /// @name Internal protected methods + /// @{ + /// + /// @brief Calculate muon-information based mass values if available. + /// + /// @param[in] vtx secondary vertex + /// @param[in] trkMasses ordered vector of track mass values + /// @param[in] nMuRequested number of muons requested + /// @returns muon-information based invariant mass for secondary + /// vertex and the corresponding uncertainty + /// + std::pair<double, double> getMuCalcMass(xAOD::BPhysHelper& vtx, + std::vector<double> + trkMasses, + int nMuRequested) const; + /// + /// @brief Obtain a set of tracks with muon track information if available + /// + /// @param[in] vtx secondary vertex + /// @returns container of muon tracks, number of muons found + /// + std::pair<TrackBag, int> getTracksWithMuons(xAOD::BPhysHelper& vtx) const; + /// + /// @brief Calculate invariant mass and uncertainty from a set of tracks. + /// + /// Returns invariant mass and mass error given + /// a set of tracks, their mass hypotheses and a reference position. + /// Each track must have a separate mass hypothesis in + /// the vector, and they must be in the same order as the tracks in the + /// track vector. Otherwise it will go horribly wrong. + /// + /// @param[in] trksIn container with tracks to be considered + /// @param[in] massHypoTheses vector of mass hypotheses in the same + /// order as the tracks + /// @param[in] pos position of the vertex + /// @returns invariant mass value and uncertainty + /// + std::pair<double,double> + getInvariantMassWithError(TrackBag trksIn, + std::vector<double> massHypotheses, + const Amg::Vector3D& pos) const; + /// + /// @brief Determine minimum log chi2 of signal muon tracks w.r.t. + // any primary vertex. + /// + /// Find minimum log chi2 distance of signal muons w.r.t any primary + /// vertex of required types and with a minimum number of tracks cut. + /// It also depends on the mode w.r.t. the treatment of the associated + /// primary vertex and the type of PV-to-SV association. + /// Returns this minimum chi2. + /// + /// @param[in] vtx secondary vertex + /// @param[in] pvContainer container of primary vertices + /// @parma[in] pvtypes vector of primary vertex types to be considered + /// @param[in] minNTracksInPV minimum number of tracks in primary + /// vertex for it to be considered + /// @param[in] mode mode of operation (possible values: 0, 1, 2 ,3) + /// @param[in] pv_type type of PV-to-SV association + /// @returns minimum log chi2 = log(d0^2/d0e^+z0^2/z0e^2) w.r.t. + /// any primary vertex + /// + double getMinChi2ToAnyPV(xAOD::BPhysHelper& vtx, + const xAOD::VertexContainer* pvContainer, + const std::vector<int>& pvtypes, + const int minNTracksInPV, + const int mode, + const xAOD::BPhysHelper::pv_type& + pvAssocType) const; + /// + /// @brief Calculate log chi2 value of a track w.r.t. a position. + /// + /// Calculate the log chi2 ( = log((d0/d0e)^2+(z0/z0e)^2) contribution + /// of a track at the position closest to the given PV. + /// + /// @param[in] track track considered + /// @param[in] pos position considered + /// @returns log chi2 value + /// + double getTrackPVChi2(const xAOD::TrackParticle& track, + const Amg::Vector3D& pos) const; + /// + /// @brief Extract 3x3 momentum covariance matrix from a TrackParticle. + /// + /// Extract the 3x3 momentum covariance matrix in (x,y,z) notation + /// from the (phi, theta, qoverp) notation from a TrackParticle. + /// + /// @param[in] track TrackParticle considered + /// @returns 3x3 momentum covariance matrix + /// + AmgSymMatrix(3) getMomentumCov(const xAOD::TrackParticle* track) const; + /// + /// @brief Extract 3x3 momentum covariance matrix from a Perigee. + /// + /// Extract the 3x3 momentum covariance matrix in (x,y,z) notation + /// from the (phi, theta, qoverp) notation from a Perigee. + /// + /// @param[in] perigee Trk::Perigee considered + /// @returns 3x3 momentum covariance matrix + /// + AmgSymMatrix(3) getMomentumCov(const Trk::Perigee* perigee) const; + /// + /// @brief Extract 3x3 momentum covariance matrix from a track parameter + /// vector and 5x5 covariance matrix. + /// + /// Extract the 3x3 momentum covariance matrix in (x,y,z) notation + /// from the (phi, theta, qoverp) notation from a vector of + /// track parameters and the 5x5 error matrix. + /// + /// @param[in] pars 5-vector of track parameters + /// @param[in] cov 5x5 covariance matrix of track parameters + /// @returns 3x3 momentum covariance matrix + /// + AmgSymMatrix(3) getMomentumCov(const AmgVector(5)& pars, + const AmgSymMatrix(5)& cov) const; + /// + /// @brief Find all muons associated to secondary vertex. + /// + /// Returns a vector of xAOD::Muon objects found + /// in this vertex and subsequent decay vertices. + /// Recursively calls itself if necessary. + /// + /// @param[in] vtx secondary vertex + /// @returns container of muons found + /// + MuonBag findAllMuonsInDecay(xAOD::BPhysHelper& vtx) const; + /// + /// @brief Obtain a set of ID tracks for a set of muons. + /// + /// @param[in] muons container of muon objects + /// @returns container of associated ID tracks + /// + TrackBag getIdTracksForMuons(MuonBag& muons) const; + /// + /// @brief Extract TrackParticle for Muon and adjust kinematics. + /// + /// Extract primary track particle from muon; + /// if configured adjust pt, eta and phi of it before returning + /// a pointer to it. + /// + /// @param[in] muon pointer to muon + /// @returns TrackParticle pointer + /// + const xAOD::TrackParticle* adjustTrackParticle(const xAOD::Muon* muon) + const; + /// + /// @brief Initialize PV-to-SV association type vector. + /// + void initPvAssocTypeVec(); + /// + /// @brief Clear the cache of adjusted TrackParticles. + /// + void clearAdjTpCache() const; + /// @} + private: + /// @name job options + /// @{ + std::string m_branchPrefix; + std::string m_vertexContainerName; + std::vector<double> m_trkMasses; + ToolHandle<Reco::ITrackToVertex> m_trackToVertexTool; + bool m_adjustToMuonKinematics; + int m_addMinChi2ToAnyPVMode; + std::string m_pvContainerName; + int m_minNTracksInPV; + std::vector<int> m_pvTypesToConsider; + int m_doVertexType; + /// @} + /// + /// map original -> adjusted track particles + typedef std::map<const xAOD::TrackParticle*, const xAOD::TrackParticle*> + TpMap_t; + /// map of adjusted track particles as cache + mutable TpMap_t m_adjTpCache; + + /// cache for individual vertex types + std::vector<xAOD::BPhysHelper::pv_type> m_pvAssocTypes; + + }; // class +} // namespace + +#endif // DERIVATIONFRAMEWORK_BPhysAddMuonBasedInvMass_H diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/BPhysConversionFinder.h b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/BPhysConversionFinder.h new file mode 100644 index 00000000000..50a42966962 --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/BPhysConversionFinder.h @@ -0,0 +1,76 @@ +/* + Copyright (C) 2002-2019 CERN for the benefit of the ATLAS collaboration +*/ +/////////////////////////////////////////////////////////////////// +// BPhysConversionFinder.h, (c) ATLAS Detector software +/////////////////////////////////////////////////////////////////// +// Author: A. Chisholm <andrew.chisholm@cern.ch> +#ifndef DERIVATIONFRAMEWORK_BPHYSCONVERSIONFINDER_H +#define DERIVATIONFRAMEWORK_BPHYSCONVERSIONFINDER_H + +#include <string> + +#include "AthenaBaseComps/AthAlgTool.h" +#include "GaudiKernel/ToolHandle.h" +#include "DerivationFrameworkInterfaces/IAugmentationTool.h" + +#include "InDetConversionFinderTools/VertexPointEstimator.h" +#include "InDetConversionFinderTools/ConversionPostSelector.h" +#include "TrkVertexSeedFinderUtils/ITrkDistanceFinder.h" + +#include "TLorentzVector.h" + +namespace Trk +{ + class V0Tools; + class IVertexFitter; + class TrkVKalVrtFitter; +} + +namespace InDet +{ + class VertexPointEstimator; + class TrackPairsSelector; + class ConversionPostSelector; +} + +namespace DerivationFramework { + +class BPhysConversionFinder : public AthAlgTool, public IAugmentationTool { + + public: + + BPhysConversionFinder(const std::string& t, const std::string& n, const IInterface* p); + + StatusCode initialize() override; + StatusCode finalize() override; + + virtual StatusCode addBranches() const override; + + private: + + StatusCode doCascadeFit(const xAOD::Vertex * diMuonVertex, const xAOD::Vertex * convVertex, const double diMuonMassConstraint, TLorentzVector & fitMom, float & chiSq) const; + + std::string m_diMuonCollectionToCheck; + std::vector<std::string> m_passFlagsToCheck; + + ToolHandle <Trk::V0Tools> m_v0Tools; + ToolHandle <Trk::IVertexFitter> m_vertexFitter; + ToolHandle <InDet::VertexPointEstimator> m_vertexEstimator; + ToolHandle <Trk::ITrkDistanceFinder> m_distanceTool; + ToolHandle <InDet::ConversionPostSelector> m_postSelector; + ToolHandle <Trk::TrkVKalVrtFitter > m_cascadeFitter; + + std::string m_inputTrackParticleContainerName; + std::string m_conversionContainerName; + + float m_maxDistBetweenTracks; + float m_maxDeltaCotTheta; + + bool m_requireDeltaM; + float m_maxDeltaM; + + }; +} + +#endif // DERIVATIONFRAMEWORK_BPhysConversionFinder_H diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/BPhysMetadataBase.h b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/BPhysMetadataBase.h new file mode 100644 index 00000000000..d4a5ffd9085 --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/BPhysMetadataBase.h @@ -0,0 +1,100 @@ +/* + Copyright (C) 2002-2019 CERN for the benefit of the ATLAS collaboration +*/ + +//============================================================================ +// BPhysMetadataBase.h +//============================================================================ +// +// Author : Wolfgang Walkowiak <Wolfgang.Walkowiak@cern.ch.> +// Changes: +// - w.w., 2017-01-22: Added use of BPhysMetaDataTool. +// - w.w., 2019-12-05: Added long and vector<long> types +// +// Store JO metadata in the output file. +// +// It uses the BPhysMetaDataTool (default) or the IOVDbMetaDataTool to +// store job option information as metadata in a specific branch whose +// name needs to prefixed by the deriviation format name. +// Note: Metadata stored by the IOVDbMetaDataTool is not readable on +// 'RootCore' level. +// +// This is a base class. Inherit from it to add the job options you want +// to store. For a usage example, see +// Bmumu_metadata.h / Bmumu_metadata.cxx +// and +// BPHY8.py . +// +//============================================================================ +// +#ifndef DERIVATIONFRAMEWORK_BPhysMetadataBase_H +#define DERIVATIONFRAMEWORK_BPhysMetadataBase_H + +#include <string> +#include <map> +#include <vector> + +#include "AthenaBaseComps/AthAlgTool.h" +#include "DerivationFrameworkInterfaces/IAugmentationTool.h" +#include "GaudiKernel/ToolHandle.h" + +namespace DerivationFramework { + + class BPhysMetadataBase : virtual public AthAlgTool, + virtual public IAugmentationTool { + public: + BPhysMetadataBase(const std::string& t, const std::string& n, + const IInterface* p); + + virtual StatusCode initialize(); + virtual StatusCode finalize(); + + virtual StatusCode addBranches() const; + + protected: + virtual void recordPropertyI(const std::string& name, int val); + virtual void recordPropertyL(const std::string& name, long val); + virtual void recordPropertyD(const std::string& name, double val); + virtual void recordPropertyB(const std::string& name, bool val); + virtual void recordPropertyS(const std::string& name, const std::string& val); + + virtual void recordPropertyVI(const std::string& name, const std::vector<int>& val); + virtual void recordPropertyVL(const std::string& name, const std::vector<long>& val); + virtual void recordPropertyVD(const std::string& name, const std::vector<double>& val); + virtual void recordPropertyVB(const std::string& name, const std::vector<bool>& val); + virtual void recordPropertyVS(const std::string& name, + const std::vector<std::string>& val); + + private: + virtual StatusCode saveMetaDataBPhys() const; + virtual std::string buildFolderName(const std::string& fname="") const; + virtual std::string vecToString(const std::vector<int>& v) const; + virtual std::string vecToString(const std::vector<long>& v) const; + virtual std::string vecToString(const std::vector<double>& v) const; + virtual std::string vecToString(const std::vector<bool>& v) const; + virtual std::string vecToString(const std::vector<std::string>& v) const; + + private: + /// Object accessing the output metadata store + mutable ServiceHandle< StoreGateSvc > m_outputMetaStore; + + // job options + std::string m_derivationName; + std::string m_mdFolderName; + std::string m_prefix; + + // maps for different types of JOs + std::map<std::string, int> m_propInt; + std::map<std::string, long> m_propLong; + std::map<std::string, double> m_propDouble; + std::map<std::string, bool> m_propBool; + std::map<std::string, std::string> m_propString; + std::map<std::string, std::vector<int> > m_propVInt; + std::map<std::string, std::vector<long> > m_propVLong; + std::map<std::string, std::vector<double> > m_propVDouble; + std::map<std::string, std::vector<bool> > m_propVBool; + std::map<std::string, std::vector<std::string> > m_propVString; + }; // class +} // namespace + +#endif // DERIVATIONFRAMEWORK_BPhysMetadataBase_H diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/BPhysPVCascadeTools.h b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/BPhysPVCascadeTools.h new file mode 100644 index 00000000000..c1561102db7 --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/BPhysPVCascadeTools.h @@ -0,0 +1,159 @@ +/* + Copyright (C) 2002-2017 CERN for the benefit of the ATLAS collaboration +*/ + +#ifndef DERIVATIONFRAMEWORK_PVCASCADETOOLS_H +#define DERIVATIONFRAMEWORK_PVCASCADETOOLS_H + +#include "GaudiKernel/ToolHandle.h" +#include "xAODBPhys/BPhysHypoHelper.h" +#include "DerivationFrameworkBPhys/CascadeTools.h" +#include "EventKernel/PdtPdg.h" + +#include <vector> +// Authors: Adam Barton <abarton@SPAMMENOTTtttcern.ch> +// Eva Bouhova <bouhova@SPAMMENOTTtttcern.ch> + + +//class CascadeTools; +namespace Trk { + class V0Tools; + class VxCascadeInfo; +} + +namespace Analysis{ + class PrimaryVertexRefitter; +} +namespace InDet{ +class BeamSpotData; +} + +namespace HepPDT{ + class ParticleDataTable; +} + +namespace DerivationFramework { + + class BPhysPVCascadeTools { + typedef ElementLink<xAOD::VertexContainer> VertexLink; + typedef std::vector<VertexLink> VertexLinkVector; + private: + const Trk::V0Tools *m_v0Tools; + const CascadeTools *m_cascadeTools; + const InDet::BeamSpotData *m_beamSpotData; + + /// minimum number of tracks required in PVs considered + size_t m_PV_minNTracks; + + public: + bool m_copyAllVertices; + BPhysPVCascadeTools(const CascadeTools *cascadeTools); + BPhysPVCascadeTools(const CascadeTools *cascadeTools, + const InDet::BeamSpotData*); + + + void ProcessVertex(const std::vector<TLorentzVector> &mom, Amg::MatrixX cov, xAOD::BPhysHypoHelper &vtx, xAOD::BPhysHelper::pv_type pvtype, double mass) const; + + static void FillBPhysHelperNULL(xAOD::BPhysHelper &vtx, const xAOD::VertexContainer* PvContainer, + xAOD::BPhysHelper::pv_type pvtype); + + ///Fills the BPhysHelper object with the standard parameters + void FillBPhysHelper(const std::vector<TLorentzVector> &mom, Amg::MatrixX cov, xAOD::BPhysHelper &vtx, const xAOD::Vertex* refPV,const xAOD::VertexContainer* refPvContainer, + xAOD::BPhysHelper::pv_type pvtype, int) const; + + ///Returns the index integer of the vertex with the lowest Z in relation to the given vertex + size_t FindLowZIndex(const std::vector<TLorentzVector> &mom, const xAOD::BPhysHelper &Obj, + const std::vector<const xAOD::Vertex*> &PVlist, + const size_t PV_minNTracks=0) const; + ///Returns the index integer of the vertex with the lowest A0 in relation to the given vertex + size_t FindLowA0Index(const std::vector<TLorentzVector> &mom, const xAOD::BPhysHelper &Obj, + const std::vector<const xAOD::Vertex*> &PVlist, + const size_t PV_minNTracks=0) const; + + static size_t FindHighPtIndex(const std::vector<const xAOD::Vertex*> &PVlist); + + template< size_t NTracks> //NTracks = number of tracks in this type of vertex, if this is not known do not use this method + static bool VerticesMatchTracks(const xAOD::Vertex* v1, const xAOD::Vertex* v2); + + template< size_t NTracks> + static const xAOD::Vertex* FindVertex(const xAOD::VertexContainer* c, const xAOD::Vertex* v); + + /// Static method call with + /// DerivationFramework::BPhysDerHelpers::GetGoodPV + /// Returns a std::vector containing only PVs of type 1 and 3 - HighPt + /// and Pileup, which have at least PV_minNTracks tracks. + static std::vector<const xAOD::Vertex*> GetGoodPV(const xAOD::VertexContainer* pvContainer); + + /// Set the minimum number of tracks required for primary vertices to be + /// considered for primary vertex association to a secondary vertex. + /// Note that this requirement will not be applied for finding + /// the vertex with the highest pT sum (FindHighPtIndex()) since + /// it would possibly exclude this vertex which has been marked + /// earlier in the tool chain. + void SetMinNTracksInPV(size_t PV_minNTracks); + + /// Get the current beamspot position either from cache or from + /// BeamCondSvc. + /// Before processing a new event, make sure to call + /// GetBeamSpot(); + [[nodiscard]] const Amg::Vector3D& GetBeamSpot() const noexcept; + + /// Find the index for the PV with the lowest distance in z of + /// the SV's DOCA point w.r.t. the beamline and the PV. + size_t FindLowZ0BAIndex(const std::vector<TLorentzVector> &mom, const xAOD::BPhysHelper &obj, + const std::vector<const xAOD::Vertex*> &PVlist, + const size_t PV_minNTracks=0) const; + /// Calculate the distance along z axis between the PV and + /// SV's DOCA point w.r.t. the beamline. + double DistInZtoDOCA(const std::vector<TLorentzVector> &mom, const xAOD::BPhysHelper &obj, + const xAOD::Vertex* vertex) const; + /// Point of DOCA w.r.t. the beamline backward extrapolated + /// along the B candidate's momentum direction. + Amg::Vector3D DocaExtrapToBeamSpot(const std::vector<TLorentzVector> &mom, const xAOD::BPhysHelper &obj) const; + + static void PrepareVertexLinks(Trk::VxCascadeInfo *result, const xAOD::TrackParticleContainer* importedTrackCollection); + + StatusCode FillCandwithRefittedVertices( bool refitPV, + const xAOD::VertexContainer* pvContainer, xAOD::VertexContainer* refPvContainer, + const Analysis::PrimaryVertexRefitter *pvRefitter, size_t in_PV_max, int DoVertexType, + Trk::VxCascadeInfo* casc, int index, + double mass, xAOD::BPhysHypoHelper &vtx); + + static std::vector<const xAOD::TrackParticle*> CollectAllChargedTracks(const std::vector<xAOD::Vertex*> &cascadeVertices); + + static void SetVectorInfo(xAOD::BPhysHelper &, const Trk::VxCascadeInfo*); + static bool uniqueCollection(const std::vector<const xAOD::TrackParticle*>&); + static bool uniqueCollection(const std::vector<const xAOD::TrackParticle*>&, const std::vector<const xAOD::TrackParticle*>&); + static bool LinkVertices(SG::AuxElement::Decorator<VertexLinkVector> &decor, const std::vector<const xAOD::Vertex*>& vertices, + const xAOD::VertexContainer* vertexContainer, const xAOD::Vertex* vert); + static double getParticleMass(const HepPDT::ParticleDataTable* pdt, int pdg); + }; // class BPhysPVCascadeTools + +} // namespace DerivationFramework + + +//added by ab +template< size_t NTracks> +bool DerivationFramework::BPhysPVCascadeTools::VerticesMatchTracks(const xAOD::Vertex* v1, const xAOD::Vertex* v2) +{ + if(v1->nTrackParticles() != v2->nTrackParticles()) return false; + assert(v1->nTrackParticles() == NTracks); + std::array<const xAOD::TrackParticle*, NTracks> a1; + std::array<const xAOD::TrackParticle*, NTracks> a2; + for(size_t i=0;i<NTracks;i++){ + a1[i] = v1->trackParticle(i); + a2[i] = v2->trackParticle(i); + } + std::sort(a1.begin(), a1.end()); + std::sort(a2.begin(), a2.end()); + return a1 == a2; +} + +template< size_t NTracks> +const xAOD::Vertex* DerivationFramework::BPhysPVCascadeTools::FindVertex(const xAOD::VertexContainer* c, const xAOD::Vertex* v){ + for (const xAOD::Vertex* a : *c){ + if(VerticesMatchTracks<NTracks>(a,v)) return a; + } + return nullptr; +} +#endif // DERIVATIONFRAMEWORK_PVCASCADETOOLS_H diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/BPhysPVThinningTool.h b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/BPhysPVThinningTool.h new file mode 100644 index 00000000000..62d62d55943 --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/BPhysPVThinningTool.h @@ -0,0 +1,53 @@ +/* + Copyright (C) 2002-2017 CERN for the benefit of the ATLAS collaboration +*/ + +/////////////////////////////////////////////////////////////////// +// BPhysPVThinningTool.h +/////////////////////////////////////////////////////////////////// + +#ifndef DERIVATIONFRAMEWORK_BPhysPVThinningTool_H +#define DERIVATIONFRAMEWORK_BPhysPVThinningTool_H 1 + +#include "xAODTracking/VertexContainer.h" +// Gaudi & Athena basics +#include "AthenaBaseComps/AthAlgTool.h" +#include "StoreGate/ReadHandleKeyArray.h" +#include "StoreGate/ThinningHandleKey.h" +// DerivationFramework includes +#include "DerivationFrameworkInterfaces/IThinningTool.h" + +namespace DerivationFramework { + + + class BPhysPVThinningTool : public AthAlgTool, public IThinningTool { + + public: + /** Constructor with parameters */ + BPhysPVThinningTool( const std::string& t, const std::string& n, const IInterface* p ); + + /** Destructor */ + ~BPhysPVThinningTool(); + + // Athena algtool's Hooks + virtual StatusCode initialize() override; + virtual StatusCode finalize() override; + + /** Check that the current event passes this filter */ + virtual StatusCode doThinning() const override; + + private: + StringProperty m_streamName{ this, "StreamName", "", "Name of the stream being thinned" }; + SG::ReadHandleKeyArray<xAOD::VertexContainer> m_BPhyCandList; + SG::ThinningHandleKey<xAOD::TrackParticleContainer> m_TrackContainerName; + SG::ThinningHandleKey<xAOD::VertexContainer> m_PVContainerName; + mutable std::atomic<unsigned int> m_ntot; + mutable std::atomic<unsigned int> m_npass; + mutable std::atomic<unsigned int> m_tracks_kept; + bool m_keepTracks; + }; + +} + + +#endif diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/BPhysPVTools.h b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/BPhysPVTools.h new file mode 100644 index 00000000000..86412eb46d9 --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/BPhysPVTools.h @@ -0,0 +1,108 @@ +/* + Copyright (C) 2002-2017 CERN for the benefit of the ATLAS collaboration +*/ + +#ifndef DERIVATIONFRAMEWORK_PVTOOLS_H +#define DERIVATIONFRAMEWORK_PVTOOLS_H + +#include "GaudiKernel/ToolHandle.h" +#include "xAODBPhys/BPhysHelper.h" +#include <vector> + +// Author: Adam Barton <abarton@SPAMMENOTTtttcern.ch> +namespace InDet{ +class BeamSpotData; +} +namespace Trk { + class V0Tools; +} +namespace Analysis{ + class PrimaryVertexRefitter; +} + + +namespace DerivationFramework { + + class BPhysPVTools { + + private: + const Trk::V0Tools *m_v0Tools; + const InDet::BeamSpotData *m_beamSpotData; + + /// minimum number of tracks required in PVs considered + size_t m_PV_minNTracks; + + bool m_3dCalc; + + public: + + BPhysPVTools(const Trk::V0Tools *v0Tools); + BPhysPVTools(const Trk::V0Tools *v0Tools, const InDet::BeamSpotData*); + void SetSave3d(bool v) { m_3dCalc =v; } + StatusCode FillCandExistingVertices(xAOD::VertexContainer* vtxContainer, const xAOD::VertexContainer* pvContainer, int DoVertexType); + + static void FillBPhysHelperNULL(xAOD::BPhysHelper &vtx, const xAOD::VertexContainer* PvContainer, + xAOD::BPhysHelper::pv_type pvtype, bool do3d = false); + + StatusCode FillCandwithRefittedVertices(xAOD::VertexContainer* vtxContainer, const xAOD::VertexContainer* pvContainer,xAOD::VertexContainer* refPvContainer, const Analysis::PrimaryVertexRefitter* , size_t in_PV_max, int DoVertexType); + + void DecorateWithNULL(xAOD::VertexContainer* vtxContainer,const xAOD::VertexContainer* pvContainer, int DoVertexType) const; + + void DecorateWithDummyVertex(xAOD::VertexContainer* vtxContainer, const xAOD::VertexContainer* pvContainer, const xAOD::Vertex* Dummy, int DoVertexType, bool SetOrignal) const; + + ///Fills the BPhysHelper object with the standard parameters + void FillBPhysHelper(xAOD::BPhysHelper &vtx, const xAOD::Vertex* refPV,const xAOD::VertexContainer* refPvContainer, + xAOD::BPhysHelper::pv_type pvtype, int) const; + + ///Returns the index integer of the vertex with the lowest Z in relation to the given vertex + size_t FindLowZIndex(const xAOD::BPhysHelper &Obj, + const std::vector<const xAOD::Vertex*> &PVlist, + const size_t PV_minNTracks=0) const; + ///Returns the index integer of the vertex with the lowest A0 in relation to the given vertex + size_t FindLowA0Index(const xAOD::BPhysHelper &Obj, + const std::vector<const xAOD::Vertex*> &PVlist, + const size_t PV_minNTracks=0) const; + + static size_t FindHighPtIndex(const std::vector<const xAOD::Vertex*> &PVlist); + + /// Static method call with + /// DerivationFramework::BPhysDerHelpers::GetGoodPV + /// Returns a std::vector containing only PVs of type 1 and 3 - HighPt + /// and Pileup, which have at least PV_minNTracks tracks. + static std::vector<const xAOD::Vertex*> GetGoodPV(const xAOD::VertexContainer* pvContainer); + + /// Set the minimum number of tracks required for primary vertices to be + /// considered for primary vertex association to a secondary vertex. + /// Note that this requirement will not be applied for finding + /// the vertex with the highest pT sum (FindHighPtIndex()) since + /// it would possibly exclude this vertex which has been marked + /// earlier in the tool chain. + void SetMinNTracksInPV(size_t PV_minNTracks); + + /// Get the current beamspot position either from cache or from + /// BeamCondSvc. + /// Before processing a new event, make sure to call + /// GetBeamSpot(); + [[nodiscard]] const Amg::Vector3D& GetBeamSpot() const noexcept; + + /// Find the index for the PV with the lowest distance in z of + /// the SV's DOCA point w.r.t. the beamline and the PV. + size_t FindLowZ0BAIndex(const xAOD::BPhysHelper &obj, + const std::vector<const xAOD::Vertex*> &PVlist, + const size_t PV_minNTracks=0) const; + /// Calculate the distance along z axis between the PV and + /// SV's DOCA point w.r.t. the beamline. + double DistInZtoDOCA(const xAOD::BPhysHelper &obj, + const xAOD::Vertex* vertex) const; + /// Point of DOCA w.r.t. the beamline backward extrapolated + /// along the B candidate's momentum direction. + Amg::Vector3D DocaExtrapToBeamSpot(const xAOD::BPhysHelper &obj) const; + + static void PrepareVertexLinks(xAOD::Vertex* theResult, + const xAOD::TrackParticleContainer* importedTrackCollection); + }; // class BPhysPVTools + +} // namespace DerivationFramework + + +#endif // DERIVATIONFRAMEWORK_PVTOOLS_H diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/BPhysVarBlinder.h b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/BPhysVarBlinder.h new file mode 100644 index 00000000000..02e572a4f59 --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/BPhysVarBlinder.h @@ -0,0 +1,76 @@ +/* + Copyright (C) 2002-2018 CERN for the benefit of the ATLAS collaboration +*/ + +/// +/// @file BPhysVarBlinder.h +/// @author Wolfgang Walkowiak <Wolfgang.Walkowiak@cern.ch> +/// +/// @brief Vertex variable(s) blinding tool +/// +#ifndef DERIVATIONFRAMEWORK_BPhysVarBlinder_H +#define DERIVATIONFRAMEWORK_BPhysVarBlinder_H + +#include "DerivationFrameworkInterfaces/IAugmentationTool.h" +#include "DerivationFrameworkBPhys/CfAthAlgTool.h" +#include "GaudiKernel/ToolHandle.h" + +/// +/// forward declarations +/// +namespace xAOD { + class BPhysBlindingTool; +} +namespace DerivationFramework { + + /// + /// @class BPhysVarBlinder + /// @author Wolfgang Walkowiak <Wolfgang.Walkowiak@cern.ch> + /// + /// @ brief Vertex variable(s) blinding tool + /// + /// This is an AthAlgTool wrapper around the BPhysBlindingTool + /// from the package BPhysTools. + /// + /// Job options: + /// - BlindingTool : ToolHandle for xAOD::BPhysBlindingTool + /// - EnableBlinding : Switch to easily en-/disable this tool + /// + /// For an example configuration using this tool see BPHY8.py. + /// + class BPhysVarBlinder : public CfAthAlgTool, public IAugmentationTool { + + public: + /// + /// @brief Constructor + /// + BPhysVarBlinder(const std::string& t, const std::string& n, + const IInterface* p); + /// + /// @brief Initialization + /// + StatusCode initialize(); + /// + /// @brief Finalization + /// + StatusCode finalize(); + /// + /// @brief Perform blinding per event (if enabled) + /// + virtual StatusCode addBranches() const; + + private: + /// + /// @name Job options + /// @{ + /// + /// @brief ToolHandle for blinding tool + ToolHandle<xAOD::BPhysBlindingTool> m_blindingTool; + /// + /// @brief Switch for enabling blinding + bool m_enableBlinding; + /// @} + + }; +} +#endif // DERIVATIONFRAMEWORK_BPhysVarBlinder_H diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/BPhysVertexTrackBase.h b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/BPhysVertexTrackBase.h new file mode 100644 index 00000000000..28a8f683152 --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/BPhysVertexTrackBase.h @@ -0,0 +1,303 @@ +/* + Copyright (C) 2002-2019 CERN for the benefit of the ATLAS collaboration +*/ + +//============================================================================ +// BPhysVertexTrackBase.h +//============================================================================ +// +// Author : Wolfgang Walkowiak <Wolfgang.Walkowiak@cern.ch.> +// Changes: +// +// Base class for vertex-track related classes in need of +// track-to-vertex association handling. +// +// For an usage example see BVertexTrackIsoTool and BPHY8.py . +// +//============================================================================ +// +#ifndef DERIVATIONFRAMEWORK_BPhysVertexTrackBase_H +#define DERIVATIONFRAMEWORK_BPhysVertexTrackBase_H + +#include "AthenaBaseComps/AthAlgTool.h" +#include "DerivationFrameworkInterfaces/IAugmentationTool.h" +#include "GaudiKernel/ToolHandle.h" +#include "xAODEventInfo/EventInfo.h" +#include "xAODBPhys/BPhysHelper.h" +#include "EventPrimitives/EventPrimitives.h" +#include "ITrackToVertex/ITrackToVertex.h" +#include "TrackVertexAssociationTool/ITrackVertexAssociationTool.h" +#include "xAODTracking/TrackParticleContainer.h" +#include "xAODTracking/TrackParticleAuxContainer.h" +#include "xAODTracking/VertexContainer.h" +#include "xAODTracking/VertexAuxContainer.h" + +#include <string> +#include <vector> +#include <map> + +// forward declarations +namespace InDet { + class IInDetTrackSelectionTool; +} + +class TVector3; + +namespace DerivationFramework { + // + // typedefs -- to abbreviate long lines + // + typedef std::vector<const xAOD::TrackParticle*> TrackBag; + typedef std::vector<const xAOD::Muon*> MuonBag; + typedef InDet::IInDetTrackSelectionTool TrkSelTool; + + class BPhysVertexTrackBase : public AthAlgTool, + virtual public IAugmentationTool { + + protected: + class BaseItem { + + public: + BaseItem(std::string Name="_none_", std::string Bname="iso", + std::string Prefix=""); + virtual ~BaseItem(); + + virtual void setup(std::string Name, std::string Bname="iso", + std::string Prefix=""); + virtual void setPrefix(std::string Prefix); + virtual void resetVals(); + virtual void copyVals(const BaseItem& item) = 0; + virtual std::string buildName(std::string qualifier="", + std::string suffix=""); + virtual std::string toString() const; + + public: + std::string name; + std::string bname; + std::string prefix; + }; + + protected: + class TrackTypeCounter { + + public: + TrackTypeCounter(BPhysVertexTrackBase& Parent, std::string Name); + virtual ~TrackTypeCounter(); + + virtual void addToCounter(uint64_t atype, uint64_t rtype=0, + std::string prefix="", std::string suffix="", + uint64_t counts=1); + + virtual void addToCounter(std::string name, uint64_t atype=0, + uint64_t counts=1); + + virtual std::string countsToString(uint indent=0) const; + + public: + std::string name; + + private: + typedef std::map<std::string, std::pair<uint64_t, uint64_t> > + NameCountMap_t; + NameCountMap_t m_cnts; + BPhysVertexTrackBase& m_parent; + }; + + public: + // + // enumeration for types of tracks to be considered + // + enum track_type {ASSOCPV, PVTYPE0, PVTYPE1, PVTYPE2, PVTYPE3, NONE, + NULLVP, + CAPVRFN3U0, CAPVNRN3U0, CAPVRF3DU0, CAPVNR3DU0, + CAPVRFN3U1, CAPVNRN3U1, CAPVRF3DU1, CAPVNR3DU1, + CAPVRFN3U2, CAPVNRN3U2, CAPVRF3DU2, CAPVNR3DU2, + CAPVRFNNU3, CAPVNRNNU3, CAPVRFNNU4, CAPVNRNNU4, + CAPVRFNNU5, CAPVNRNNU5, CAPVRFNNU6, CAPVNRNNU6, + CAPVRFNNU7, CAPVNRNNU7, CAPVRFNNU8, CAPVNRNNU8, + CAPVRFNNU9, CAPVNRNNU9 }; + static const int n_track_types; + static const std::string track_type_str[]; + static const uint64_t track_type_bit[]; + private: + static uint64_t s_track_type_all_cached; + + public: + // + // convenience methods + // + static const std::string tts(track_type type); + static uint64_t ttb(track_type type); + static uint64_t ttall(); + static uint64_t ttallMin(); + static uint64_t rttor(const std::vector<uint64_t> &vtypes); + static std::string wrapLines(std::string lines, + std::string prefix); + static std::string trackToString(const xAOD::TrackParticle* track); + + public: + // + // public methods called by the framework + // + BPhysVertexTrackBase(const std::string& t, const std::string& n, + const IInterface* p); + + virtual StatusCode initialize(); + virtual StatusCode finalize(); + virtual StatusCode addBranches() const; + + protected: + // + // Hook methods -- need be be overwritten in the concrete class + // + virtual StatusCode initializeHook(); + virtual StatusCode finalizeHook(); + virtual StatusCode addBranchesHook() const; + virtual StatusCode addBranchesVCSetupHook(size_t ivc) const; + virtual StatusCode addBranchesSVLoopHook(const xAOD::Vertex* vtx) const; + virtual StatusCode calcValuesHook(const xAOD::Vertex* vtx, + const unsigned int ipv, + const unsigned int its, + const unsigned int itt) const; + virtual bool fastFillHook(const xAOD::Vertex* vtx, + const int ipv) const; + + // + // Methods to be called from within addBranchesSVLoopHook() + // + virtual StatusCode calculateValues(const xAOD::Vertex* vtx) const; + + // + // internal methods + // + // name string for vertex pointer and PV index + virtual std::string buildPvAssocCacheName(const xAOD::Vertex* vtx, + const int ipv) const; + + virtual void initPvAssocTypeVec(); + virtual TrackBag findAllTracksInDecay(xAOD::BPhysHelper& vtx) const; + virtual void findAllTracksInDecay(xAOD::BPhysHelper& vtx, + TrackBag& tracks) const; + virtual MuonBag findAllMuonsInDecay(xAOD::BPhysHelper& vtx) const; + virtual void findAllMuonsInDecay(xAOD::BPhysHelper& vtx, + MuonBag& muons) const; + virtual TrackBag findAllMuonIdTracksInDecay(xAOD::BPhysHelper& vtx, + MuonBag& muons) const; + virtual std::vector<TVector3> + findMuonRefTrackMomenta(xAOD::BPhysHelper& vtx, MuonBag& muons) const; + + virtual TrackBag selectTracks(const xAOD::TrackParticleContainer* + inpTracks, + xAOD::BPhysHelper& cand, + const unsigned int ipv, + const unsigned int its, + const unsigned int itt) const; + virtual TrackBag selectTracks(const xAOD::TrackParticleContainer* + inpTracks, + const TrackBag& exclTracks, + xAOD::BPhysHelper& cand, + const unsigned int ipv, + const unsigned int its, + const unsigned int itt) const; + virtual uint64_t detTrackTypes(const xAOD::TrackParticle* track, + const xAOD::Vertex* candPV, + const xAOD::Vertex* candRefPV) const; + virtual double getTrackCandPVLogChi2(const xAOD::TrackParticle* + track, + const xAOD::Vertex* vtx, + bool doDCAin3D=false, + int chi2DefToUse=0) const; + virtual std::vector<double> getTrackLogChi2DCA(const xAOD::TrackParticle* + track, + const xAOD::Vertex* vtx, + bool doDCAin3D=false, + int chi2DefToUse=0) + const; + virtual std::string buildBranchBaseName(unsigned int its, + unsigned int ipv, + unsigned int itt, + std::string preSuffix="") const; + + virtual std::pair<const xAOD::Vertex*, double> + findMinChi2PV(const xAOD::TrackParticle* track, + const xAOD::Vertex* candPV, + const xAOD::Vertex* candRefPV, + const std::vector<uint64_t>& pvtypes, + const int minNTracksInPV, + const bool useRefittedPvs, + const bool doDCAin3D, + const int chi2DefToUse) const; + + virtual const xAOD::Vertex* + findAssocPV(const xAOD::TrackParticle* track, + const xAOD::Vertex* candPV, + const xAOD::Vertex* candRefPV, + const std::vector<uint64_t>& pvtypes, + const int minNTracksInPV, + const bool useRefittedPvs) const; + + protected: + // job options + std::vector<std::string> m_branchPrefixes; + std::string m_branchBaseName; + std::string m_branchSuffix; + std::vector<std::string> m_vertexContainerNames; + std::string m_trackParticleContainerName; + ToolHandleArray<TrkSelTool> m_trackSelectionTools; + + ToolHandle<Reco::ITrackToVertex> m_trackToVertexTool; + + ToolHandle<CP::ITrackVertexAssociationTool> m_tvaTool; + + std::string m_pvContainerName; + std::vector<std::string> m_refPVContainerNames; + + int m_doVertexType; + std::vector<uint64_t> m_useTrackTypes; + bool m_incPrecVerticesInDecay; + int m_minNTracksInPV; + std::vector<uint64_t> m_pvTypesToConsider; + int m_debugTrackTypes; + std::vector<uint64_t> m_debugTracksInEvents; + + // working point of TVA tool + bool m_tvaToolHasWpLoose; + + // containers + mutable const xAOD::TrackParticleContainer* m_tracks; + mutable const xAOD::TrackParticleAuxContainer* m_tracksAux; + mutable const xAOD::VertexContainer* m_pvtxContainer; + mutable const xAOD::VertexContainer* m_svtxContainer; + mutable const xAOD::VertexAuxContainer* m_svtxAuxContainer; + mutable const xAOD::VertexContainer* m_refPVContainer; + mutable const xAOD::VertexAuxContainer* m_refPVAuxContainer; + + // cache for individual vertex types + std::vector<xAOD::BPhysHelper::pv_type> m_pvAssocTypes; + + mutable unsigned int m_nEvtsSeen; + + // event info + mutable const xAOD::EventInfo* m_eventInfo; + + // cache for similar PV-to-SV associations + typedef std::map<std::string, int> StringIntMap_t; + mutable StringIntMap_t m_pvAssocResMap; + + // track types considered + uint64_t m_trackTypesUsed; + + // track type counter map (for debugging) + std::unique_ptr<TrackTypeCounter> m_mttc; + + // run and event numbers (see EventIDBase.h for types) + mutable unsigned int m_runNumber; + mutable uint64_t m_evtNumber; + + // debug tracks in the current event? + mutable bool m_debugTracksInThisEvent; + + }; // class +} // namespace + +#endif // DERIVATIONFRAMEWORK_BPhysVertexTrackBase_H diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/BTrackVertexMapLogger.h b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/BTrackVertexMapLogger.h new file mode 100644 index 00000000000..0b20390c590 --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/BTrackVertexMapLogger.h @@ -0,0 +1,49 @@ +/* + Copyright (C) 2002-2017 CERN for the benefit of the ATLAS collaboration +*/ + +//============================================================================ +// BTrackVertexMapLogger.h +//============================================================================ +// +// Author : Wolfgang Walkowiak <Wolfgang.Walkowiak@cern.ch.> +// Changes: +// +// Initiate dumps of track-to-vertex maps to log file which are provided +// by BPhysTrackVertexMapTool instances. +// +// The BPhysTrackVertexMapTool instances need to be configured separately +// and handed to this tool. +// +//============================================================================ +// +#ifndef DERIVATIONFRAMEWORK_BTrackVertexMapLogger_H +#define DERIVATIONFRAMEWORK_BTrackVertexMapLogger_H + +#include "AthenaBaseComps/AthAlgTool.h" +#include "DerivationFrameworkInterfaces/IAugmentationTool.h" +#include "GaudiKernel/ToolHandle.h" +#include "BPhysTools/IBPhysTrackVertexMapTool.h" + +namespace DerivationFramework { + + class BTrackVertexMapLogger : virtual public AthAlgTool, + virtual public IAugmentationTool { + public: + BTrackVertexMapLogger(const std::string& t, const std::string& n, + const IInterface* p); + + virtual StatusCode initialize(); + virtual StatusCode finalize(); + + virtual StatusCode addBranches() const; + + private: + // job options + ToolHandleArray<xAOD::IBPhysTrackVertexMapTool> m_ttvmTools; + bool m_enable; + + }; // class +} // namespace + +#endif // DERIVATIONFRAMEWORK_BTrackVertexMapLogger_H diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/BVertexClosestTrackTool.h b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/BVertexClosestTrackTool.h new file mode 100644 index 00000000000..2b704516a23 --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/BVertexClosestTrackTool.h @@ -0,0 +1,149 @@ +/* + Copyright (C) 2002-2019 CERN for the benefit of the ATLAS collaboration +*/ + +//============================================================================ +// BVertexClosestTrackTool.h +//============================================================================ +// +// Author : Wolfgang Walkowiak <Wolfgang.Walkowiak@cern.ch.> +// Changes: +// +// Add B vertex closest track information for different configurations, +// different track selections and different PV-to-SV association methods. +// +// For an usage example see BPHY8.py . +// +//============================================================================ +// +#ifndef DERIVATIONFRAMEWORK_BVertexClosestTrackTool_H +#define DERIVATIONFRAMEWORK_BVertexClosestTrackTool_H + +#include "DerivationFrameworkBPhys/BPhysVertexTrackBase.h" +#include "boost/multi_array.hpp" + +namespace InDet { + class IInDetTrackSelectionTool; +} + +namespace DerivationFramework { + + class BVertexClosestTrackTool : virtual public BPhysVertexTrackBase { + + private: + typedef BPhysVertexTrackBase super; + + // + // internal helper class + // + protected: + class CtItem : public BaseItem { + + public: + CtItem(std::string Name="_none_", + std::string Bname = "ctrk", + std::string Prefix="", + double Dca=-999., double DcaErr=-99., + double Zca=-999., double ZcaErr=-99., + double VtxNDErr2=-99., double TrkNDErr2=-99., + double Phi0Used=-999., + int NTrksChi2=0, xAOD::TrackParticle* CloseTrack=NULL, + TrackBag Tracks = {}, + std::vector<std::vector<double> > Vtap = {}, + std::vector<unsigned short> Selpat = {}); + + virtual ~CtItem(); + + virtual void setup(std::string Name="_none_", + std::string Bname="ctrk", + std::string Prefix=""); + virtual void setup(std::string Name, std::string Bname, + std::string Prefix, + double Dca, double DcaErr, + double Zca, double ZcaErr, + double VtxNDErr2, double TrkNDErr2, + double Phi0Used, + int NTrksChi2, + xAOD::TrackParticle* CloseTrack=NULL, + TrackBag Tracks = {}, + std::vector<std::vector<double> > Vtap = {}, + std::vector<unsigned short> Selpat = {}); + virtual void resetVals(); + virtual void copyVals(const BaseItem& item); + virtual void copyVals(const CtItem& item); + virtual std::string dcaName(); + virtual std::string dcaErrName(); + virtual std::string zcaName(); + virtual std::string zcaErrName(); + virtual std::string vtxNDErr2Name(); + virtual std::string trkNDErr2Name(); + virtual std::string phi0UsedName(); + virtual std::string nTrksChi2Name(); + virtual std::string closeTrackName(); + virtual std::string toString() const; + + public: + mutable double dca; + mutable double dcaErr; + mutable double zca; + mutable double zcaErr; + mutable double vtxNDErr2; + mutable double trkNDErr2; + mutable double phi0Used; + mutable int nTrksChi2; + const xAOD::TrackParticle* closeTrack; + mutable TrackBag tracks; + mutable std::vector<std::vector<double> > vtap; + mutable std::vector<unsigned short> selpat; + + }; // CtItem + + public: + BVertexClosestTrackTool(const std::string& t, const std::string& n, + const IInterface* p); + + protected: + // Hook methods + virtual StatusCode initializeHook(); + virtual StatusCode finalizeHook(); + + virtual StatusCode addBranchesVCSetupHook(size_t ivc) const; + + virtual StatusCode addBranchesSVLoopHook(const xAOD::Vertex* vtx) const; + + virtual StatusCode calcValuesHook(const xAOD::Vertex* vtx, + const unsigned int ipv, + const unsigned int its, + const unsigned int itt) const; + virtual bool fastFillHook(const xAOD::Vertex* vtx, + const int ipv) const; + private: + virtual StatusCode saveClosestTrack(const xAOD::Vertex* vtx) const; + virtual void initResults(); + virtual void setResultsPrefix(std::string prefix) const; + virtual StatusCode logCloseTracksDebugInfo() const; + + private: + // job options + + std::vector<std::string> m_closeTrackChi2SetName; + std::vector<int> m_closeTrackCorrChi2; + std::vector<bool> m_minDCAin3D; + std::vector<double> m_closeTrackMaxLogChi2; + std::vector<double> m_nCloseTrackMaxLogChi2; + + // results array + typedef boost::multi_array<CtItem, 4> CtItem4_t; + mutable CtItem4_t m_results; + + // last run and event numbers seen + mutable unsigned int m_lastRunNumber; + mutable uint64_t m_lastEvtNumber; + + // last secondary vertex (candidate) index + mutable unsigned int m_svIdx; + + }; // BVertexClosestTrackTool +} // namespace + +#endif // DERIVATIONFRAMEWORK_BVertexClosestTrackTool_H diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/BVertexTrackIsoTool.h b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/BVertexTrackIsoTool.h new file mode 100644 index 00000000000..03e645b8514 --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/BVertexTrackIsoTool.h @@ -0,0 +1,119 @@ +/* + Copyright (C) 2002-2017 CERN for the benefit of the ATLAS collaboration +*/ + +//============================================================================ +// BVertexTrackIsoTool.h +//============================================================================ +// +// Author : Wolfgang Walkowiak <Wolfgang.Walkowiak@cern.ch.> +// Changes: +// +// Add B vertex track isolation information for different configurations, +// different track selections and different PV-to-SV association methods. +// +// For an usage example see BPHY8.py . +// +//============================================================================ +// +#ifndef DERIVATIONFRAMEWORK_BVertexTrackIsoTool_H +#define DERIVATIONFRAMEWORK_BVertexTrackIsoTool_H + +#include "DerivationFrameworkBPhys/BPhysVertexTrackBase.h" +#include "boost/multi_array.hpp" + +namespace InDet { + class IInDetTrackSelectionTool; +} + +namespace DerivationFramework { + + class BVertexTrackIsoTool : virtual public BPhysVertexTrackBase { + + private: + typedef BPhysVertexTrackBase super; + + // + // internal helper class + // + protected: + class IsoItem : public BaseItem { + + public: + IsoItem(std::string Name="_none_", std::string Bname="iso", + std::string Prefix="", + double IsoValue=-1., int NTracks=0); + virtual ~IsoItem(); + + virtual void setup(std::string Name, std::string Bname="iso", + std::string Prefix=""); + virtual void setup(std::string Name, std::string Bname, + std::string Prefix, + double IsoValue, int NTracks=0); + virtual void resetVals(); + virtual void copyVals(const BaseItem& item); + virtual void copyVals(const IsoItem& item); + virtual std::string isoName(); + virtual std::string nTracksName(); + + public: + mutable double isoValue; + mutable int nTracks; + }; // IsoItem + + public: + BVertexTrackIsoTool(const std::string& t, const std::string& n, + const IInterface* p); + + protected: + // Hook methods + virtual StatusCode initializeHook(); + virtual StatusCode finalizeHook(); + + virtual StatusCode addBranchesVCSetupHook(size_t ivc) const; + + virtual StatusCode addBranchesSVLoopHook(const xAOD::Vertex* vtx) const; + + virtual StatusCode calcValuesHook(const xAOD::Vertex* vtx, + const unsigned int ipv, + const unsigned int its, + const unsigned int itt) const; + virtual bool fastFillHook(const xAOD::Vertex* vtx, + const int ipv) const; + + private: + virtual StatusCode saveIsolation(const xAOD::Vertex* vtx) const; + virtual StatusCode calculateIsolation(const xAOD::Vertex* vtx) const; + virtual StatusCode calcIsolation(const IsoItem& iso, + const xAOD::Vertex* vtx, + const double coneSize, + const double logChi2Max, + const int doLogChi2, + const ToolHandle<TrkSelTool>& tSelTool, + const xAOD::BPhysHelper::pv_type + pvAssocType, + const int trackTypes ) const; + + virtual void initResults(); + virtual void setResultsPrefix(std::string prefix) const; + + virtual std::string buildBranchName(unsigned int ic, + unsigned int its, + unsigned int ipv, + unsigned int itt) const; + + private: + // job options + std::vector<double> m_isoConeSizes; + std::vector<double> m_isoTrkImpLogChi2Max; + std::vector<int> m_isoDoTrkImpLogChi2Cut; + bool m_useOptimizedAlgo; + + // results array + typedef boost::multi_array<IsoItem, 4> IsoItem4_t; + mutable IsoItem4_t m_results; + + }; // BVertexTrackIsoTool +} // namespace + +#endif // DERIVATIONFRAMEWORK_BVertexTrackIsoTool_H diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/BmumuThinningTool.h b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/BmumuThinningTool.h new file mode 100644 index 00000000000..bba64bb5fcb --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/BmumuThinningTool.h @@ -0,0 +1,441 @@ +/* + Copyright (C) 2002-2017 CERN for the benefit of the ATLAS collaboration +*/ + +// +/** + * @file BmumuThinningTool.h + * @author Wolfgang Walkowiak <wolfgang.walkowiak@cern.ch> + * + * @brief Primary vertex, track and muon thinning for Bmumu analysis. + */ + +#ifndef DERIVATIONFRAMEWORK_BmumuThinningTool_H +#define DERIVATIONFRAMEWORK_BmumuThinningTool_H + +#include "DerivationFrameworkBPhys/CfAthAlgTool.h" +#include "DerivationFrameworkInterfaces/IThinningTool.h" +#include "GaudiKernel/ToolHandle.h" +#include "xAODBPhys/BPhysHypoHelper.h" + +#include <string> + +class IThinningSvc; + +namespace SG { + class AuxElement; +} + +namespace xAOD { + class AuxContainerBase; +} + +namespace DerivationFramework { + /// + /// @class BmumuThinningTool + /// @author Wolfgang Walkowiak <wolfgang.walkowiak@cern.ch> + /// + /// @brief Primary vertex, track and muon thinning for Bmumu analysis. + /// + /// This class provides a thinning tool to thin the primary vertex, + /// the muon and calibrated muon collections as well as the inner + /// detector track selection depending on: + /// - the selected secondary vertices + /// - the selected PV-to-SV vertex association method + /// - additional track sub-collection (e.g. "closest tracks") + /// + /// This tool is primarily written for the B(s)->mumu analysis + /// and used by the BPHY8 derivation. + /// + /// ### Job options + /// <table border="0"> + /// <tr><th align="left">Name</td> + /// <th align="left">Description</th></tr> + /// <tr><td valign="top">ThinningService</td> + /// <td>Thinning service handle</td></tr> + /// <tr><td valign="top">TrackParticleContainerName</td> + /// <td>TrackParticle container name + /// (default: InDetTrackParticles)</td></tr> + /// <tr><td valign="top">VertexContainerNames</td> + /// <td>List of secondary vertex container names</td></tr> + /// <tr><td valign="top">VertexPassFlags</td> + /// <td>List of pass flags for the seconary vertices + /// empty list lets all vertices pass. + /// List length needs to be identical to length of + /// VertexContainerNames list if AlignPassToVertexList + /// is True</td></tr> + /// <tr><td valign="top">AlignPassToVertexList</td> + /// <td>Align VertexPassFlags to VertexContainerNames list? + /// This option causes a 1:1 correlation between the two lists, + /// i.e. a flag is only applied to the corresponding container + /// if this option is set to True. (default: false)</td></tr> + /// <tr><td valign="top">PVContainerName</td> + /// <td>Primary vertex container name</td></tr> + /// <tr><td valign="top">RefPVContainerNames</td> + /// <td>Refitted primary vertex container names. + /// This list must be of same length and order as the + /// m_vtxContNames list. + /// (or empty: no thinning of refitted primary + /// vertex containers)</td></tr> + /// <tr><td valign="top">MuonContainerName</td> + /// <td> Name of the used muon container (default: "")</td></tr> + /// <tr><td valign="top">CalibMuonContainerName</td> + /// <td>Name of the calibrated muons container (default: "")</td></tr> + /// <tr><td valign="top">CloseTrackBranchBaseName</td> + /// <td>Closest track branch base name</td></tr> + /// <tr><td valign="top">CloseTrackBranchPrefixes</td> + /// <td>Closest track branch prefixes</td></tr> + /// <tr><td valign="top">KeepTracksForSelectedPVs</td> + /// <td>Keep tracks for selected (refitted) primary vertices? + /// (default: false)</td></tr> + /// <tr><td valign="top">MatchCalibratedMuons</td> + /// <td>Match vertex muons with calibrated muons? + /// (default: false)</td></tr> + /// <tr><td valign="top">MarkMatchedMuons</td> + /// <td>Mark orginal muons for matched calibrated muons as well? + /// (only makes sense if MatchCalibratedMuons = True; + /// default: false)</td></tr> + /// <tr><td valign="top">MarkMatchedCalMuons</td> + /// <td>Mark calibrated muons for matched calibrated muons as well? + /// (only makes sense if MatchedCalibratedMuons = False; + /// default: false)</td></tr> + /// <tr><td valign="top">SyncMatchedMuonsBothWays</td> + /// <td> Force syncing marked muons both ways? + /// (default: false)</td></tr> + /// <tr><td valign="top">AllowFastMuonMaskSync</td> + /// <td>Allow fast sync of myon masks? + /// (Set to 'False' to force in-depth synchronization + /// of muon masks. Default: false)</td></tr> + /// <tr><td valign="top">KeepCloseTracks</td> + /// <td>Keep tracks for closest tracks? (default: false)</td></tr> + /// <tr><td valign="top">KeepTracksForMuons</td> + /// <td>Keep tracks for selected muons? (default: false)</td></tr> + /// <tr><td valign="top">KeepTracksForCalMuons</td> + /// <td>Keep tracks for selected calibrated muons? + /// (default: false)</td></tr> + /// <tr><td valign="top">KeepMuonsForTracks</td> + /// <td>Keep (original) muons for selected tracks? + /// (default: false)</td></tr> + /// <tr><td valign="top">KeepCalMuonsForTracks</td> + /// <td>Keep calibrated muons for selected tracks? + /// (default: false)</td></tr> + /// <tr><td valign="top">ApplyAndForVertices</td> + /// <td>Apply AND for mask matching for vertices? + /// (default: false)</td></tr> + /// <tr><td valign="top">ApplyAndForTracks</td> + /// <td>Apply AND for mask matching for tracks? + /// (default: false)</td></tr> + /// <tr><td valign="top">ApplyAndForMuons</td> + /// <td>Apply AND for mask matching for muons? + /// (default: false)</td></tr> + /// <tr><td valign="top">ThinPVs"</td> + /// <td>Thin primary vertex collection? (default: true)</td></tr> + /// <tr><td valign="top">ThinRefittedPVs"</td> + /// <td>Thin refitted primary vertex collection? + /// (default: true)</td></tr> + /// <tr><td valign="top">ThinTracks"</td> + /// <td>Thin ID track collection? + /// (default: true)</td></tr> + /// <tr><td valign="top">ThinMuons"</td> + /// <td>Thin muon collections? + /// (default: true)</td></tr> + /// </table> + /// + class BmumuThinningTool : public CfAthAlgTool, public IThinningTool { + + // useful typedefs + typedef xAOD::BPhysHelper::pv_type pv_type; + typedef ElementLink<xAOD::TrackParticleContainer> TrackParticleLink; + + public: + /// @name pv_type to string map + // Note: may later be migrated to xAODBPhys/BPhysHelper + static std::map<pv_type, std::string> PvTypeToVarNameMap; + + public: + /// @brief Main constructor + BmumuThinningTool(const std::string& t, const std::string& n, + const IInterface* p); + /// @brief Default destructor + ~BmumuThinningTool(); + /// @brief Initialize tool + StatusCode initialize(); + /// @brief Finalize tool + StatusCode finalize(); + /// @brief Main thinning method executed for each event + virtual StatusCode doThinning() const; + + protected: + /// + /// @brief Helper checking for hypothesis passing + /// + /// Helper to check whether an element is marked as passing a specific + /// hypothesis. + /// + /// @param[in] em auxillary storage element + /// @param[in] hypo name of the hypothesis + /// @returns true if hypothesis element contains true + /// + bool pass(const SG::AuxElement& em, std::string hypo) const; + /// + /// @brief Helper to get a TrackParticle link + /// + /// @param[in] vtx secondary vertex containing link + /// @param[in] name name of the link + /// @returns pointer to TrackParticle (NULL if not found) + /// + const xAOD::TrackParticle* getTrackParticle(const xAOD::Vertex* vtx, + std::string name) const; + + template<typename TYPE> + StatusCode applyThinMask(SG::ThinningHandle<TYPE> &trkCont, + const std::vector<bool>& trkMask, + bool doAnd) const; + + + /// + /// @brief Mark muons matched to secondary vertices + /// + /// @param[in] muCont pointer to MuonContainer + /// @param[in,out] muMask vector with mask per muon + /// @param[in] vtx secondary vertex + /// @param[in] counterName name of counter + /// @returns StatusCode + /// + StatusCode matchMuons(const xAOD::MuonContainer* muCont, + std::vector<bool>& muMask, + xAOD::BPhysHelper& vtx, + std::string counterName) const; + /// + /// @name Sync-mark methods + /// + /// @{ + /// + /// @brief Mark original muons for accepted calibrated muons + /// + /// @param[in] muCont pointer to MuonContainer of + /// (original) muons + /// @param[in] cmuCont pointer to MuonContainer of + /// calibrated muons + /// @param[in,out] muMask mask for (original) muons + /// @param[in] cmuMask mask for calibrated muons + /// @param[in] counterName base name for counters + /// @param[in] allowFastSync use fast synchronization method + /// @returns StatusCode + /// + StatusCode markOrigMuons(const xAOD::MuonContainer* muCont, + const xAOD::MuonContainer* cmuCont, + std::vector<bool>& muMask, + std::vector<bool>& cmuMask, + std::string counterName, + bool allowFastSync=true) const; + /// + /// @brief Mark calibrated muons for accepted (original) muons + /// + /// @param[in] muCont pointer to MuonContainer of + /// (original) muons + /// @param[in] cmuCont pointer to MuonContainer of + /// calibrated muons + /// @param[in] muMask mask for (original) muons + /// @param[in,out] cmuMask mask for calibrated muons + /// @param[in] counterName base name for counters + /// @param[in] allowFastSync use fast synchronization method + /// @returns StatusCode + /// + StatusCode markCalibMuons(const xAOD::MuonContainer* muCont, + const xAOD::MuonContainer* cmuCont, + std::vector<bool>& muMask, + std::vector<bool>& cmuMask, + std::string counterName, + bool allowFastSync) const; + /// + /// @brief Mark ID tracks of selected (original or calibrated) muons + /// + /// @param[in] trkPartCont pointer to TrackParticle container + /// @param[in,out] trkMask mask for tracks + /// @param[in] muCont pointer to MuonContainer + /// @param[in] muMask mask for muons + /// @param[in] counterName base name for counters + /// @returns StatusCode + /// + StatusCode markTrksForSelMuons(const xAOD::TrackParticleContainer* + trkPartCont, + std::vector<bool>& trkMask, + const xAOD::MuonContainer* muCont, + std::vector<bool>& muMask, + std::string counterName) const; + /// + /// @brief Mark muons for selected ID tracks + /// + /// @param[in] trkPartCont pointer to TrackParticle container + /// @param[in] trkMask mask for tracks + /// @param[in,out] muCont pointer to MuonContainer + /// @param[in] muMask mask for muons + /// @param[in] counterName base name for counters + /// @returns StatusCode + /// + StatusCode markMuonsForSelTracks(const xAOD::TrackParticleContainer* + trkPartCont, + std::vector<bool>& trkMask, + const xAOD::MuonContainer* muCont, + std::vector<bool>& muMask, + std::string counterName) const; + /// @} + /// + + /// + /// @brief Obtain all auxillary elements matching a certain pattern. + /// + /// Helper to filter all names of auxillary elements of an aux container + /// according to a certain pattern. The pattern must be a regular + /// expression pattern. + /// + /// @param[in] auxCont pointer to AuxContainer + /// @param[in] pattern regular expression pattern to be matched by names + /// @returns vector<string> of auxillary element names + /// + std::vector<std::string> + filterAuxElements(const xAOD::AuxContainerBase* auxCont, + std::string pattern) const; + /// + /// @brief Determine aux elements to be looked at -- for (refitted) PVs + /// + /// @param[in] auxCont pointer to AuxContainer + /// @param[out] vLinkNames vector of aux element names selected + /// @param[out] vLinkTypes vector of PV-to-SV types corresponding to + /// aux element names selected + /// @param[in] pattern regular expression pattern to be matched by names + /// + void selectAuxElements(const xAOD::AuxContainerBase* auxCont, + std::vector<std::string>& vLinkNames, + std::vector<pv_type>& vLinkTypes, + std::string pattern) const; + /// + /// @brief Determine aux elements to be looked at -- for closest tracks + /// + /// @param[in] auxCont pointer to AuxContainer + /// @param[out] vLinkNames vector of aux element names selected + /// @param[in] vPrefixes vector of prefixes to be concatenated with + /// pattern for search + /// @param[out] vLinkTypes vector of PV-to-SV types corresponding to + /// aux element names selected + /// @param[in] pattern regular expression pattern to be matched by names + /// + void selectAuxElements(const xAOD::AuxContainerBase* auxCont, + std::vector<std::string>& vLinkNames, + std::vector<std::string> vPrefixes, + std::vector<pv_type>& vLinkTypes, + std::string pattern) const; + /// + /// @brief Dump a vector<str> to a string + /// + /// @param[in] vs vector<string> to be dumped + /// @param[in] header header string to be prepended + /// @param[in] nBlanks number of blanks to prepend each line with + /// + std::string dumpVS(const std::vector<std::string>& vs, + const std::string header="", + size_t nBlanks=0) const; + /// + /// @brief Wrap string at line breaks and print with + /// appropriate message level + /// + /// @param[in] str string to be printed + /// @param[in] lvl MSG::Level chosen + /// + void logWrappedMsg(const std::string& str, const MSG::Level lvl) const; + /// + /// @brief Check two masks for consistency + /// + /// This is a method returning debugging information. + /// + /// @param[in] mask1 first mask vector to be checked + /// @param[in] mask2 second mask vector to be checked + /// @param[in] name1 name of first mask vector + /// @param[in] name2 name of second mask vector + /// @param[in] header text to be prepended to output string + /// @returns string with debugging information + /// + std::string checkMaskConsistency(const std::vector<bool>& mask1, + const std::vector<bool>& mask2, + const std::string name1, + const std::string name2, + const std::string header="") const; + + private: + /// + /// @name Job options + /// @{ + ServiceHandle<IThinningSvc> m_thinningSvc; + std::string m_trkPartContName; + std::vector<std::string> m_vtxContNames; + std::vector<std::string> m_vtxPassFlags; + std::string m_PVContName; + std::vector<std::string> m_refPVContNames; + std::string m_muonContName; + std::string m_calMuonContName; + std::string m_ctBranchBaseName; + std::vector<std::string> m_ctBranchPrefixes; + bool m_alignPassToVertexList; + bool m_keepPVTracks; + bool m_matchCalMuons; + bool m_markMuons; + bool m_markCalMuons; + bool m_syncMuonsBothWays; + bool m_keepCloseTracks; + bool m_keepSelMuonTracks; + bool m_keepSelCalMuonTracks; + bool m_keepSelTrackMuons; + bool m_keepSelTrackCalMuons; + bool m_allowFastMuonMaskSync; + bool m_thinPVs; + bool m_thinRefPVs; + bool m_thinTracks; + bool m_thinMuons; + bool m_vertexAnd; + bool m_trackAnd; + bool m_muonAnd; + /// @} + + /// + /// @name internal member variables + /// + /// process close tracks + bool m_doCloseTracks; + /// process primary vertices + bool m_doPVs; + /// process refitted primary vertices + bool m_doRefPVs; + /// process (original) muons + bool m_doMuons; + /// process refitted muons + bool m_doCalMuons; + /// process ID tracks + bool m_doTracks; + /// @} + + /// + /// @name aux element link name caches + /// + /// @{ + /// + /// caching aux element link names (and pv types) + /// for original and refitted PVs + /// + mutable std::vector<std::vector<std::string> > m_vvOrigPVLinkNames; + mutable std::vector<std::vector<pv_type> > m_vvOrigPVLinkTypes; + mutable std::vector<std::vector<std::string> > m_vvRefPVLinkNames; + mutable std::vector<std::vector<pv_type> > m_vvRefPVLinkTypes; + + /// + /// caching aux element link names (and pv types) + /// for closest tracks + mutable std::vector<std::vector<std::string> > m_vvCtLinkNames; + mutable std::vector<std::vector<pv_type> > m_vvCtLinkTypes; + /// @} + + }; + +} // namespace DerivationFramework + +#endif // DERIVATIONFRAMEWORK_BmumuThinningTool_H diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/Bmumu_metadata.h b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/Bmumu_metadata.h new file mode 100644 index 00000000000..4e040a8ad14 --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/Bmumu_metadata.h @@ -0,0 +1,42 @@ +/* + Copyright (C) 2002-2017 CERN for the benefit of the ATLAS collaboration +*/ + +/** + * @file Bmumu_metadata.h + * @author Wolfgang Walkowiak <Wolfgang.Walkowiak@cern.ch> + * + * @brief Store JO metadata specific to the Bmumu analysis. + */ + +#ifndef DERIVATIONFRAMEWORK_Bmumu_metadata_H +#define DERIVATIONFRAMEWORK_Bmumu_metadata_H + +#include <string> +#include <map> +#include <vector> + +#include "DerivationFrameworkBPhys/BPhysMetadataBase.h" +#include "AthenaBaseComps/AthAlgTool.h" +#include "DerivationFrameworkInterfaces/IAugmentationTool.h" +#include "GaudiKernel/ToolHandle.h" + +namespace DerivationFramework { + /// + /// @class Bmumu_metadata + /// @author Wolfgang Walkowiak <Wolfgang.Walkowiak@cern.ch.> + /// + /// @brief Store JO metadata specific to the Bmumu analysis. + /// + /// Store JO metadata specific to the Bmumu analysis in the output file. + /// This class inherits from BPhysMetadataBase. + /// + class Bmumu_metadata : virtual public BPhysMetadataBase { + public: + /// @brief Main constructor + Bmumu_metadata(const std::string& t, const std::string& n, + const IInterface* p); + }; // class +} // namespace + +#endif // DERIVATIONFRAMEWORK_Bmumu_metadata_H diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/Bmumu_reco_mumu.h b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/Bmumu_reco_mumu.h new file mode 100644 index 00000000000..efe1ab4ee91 --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/Bmumu_reco_mumu.h @@ -0,0 +1,73 @@ +/* + Copyright (C) 2002-2018 CERN for the benefit of the ATLAS collaboration +*/ + +/////////////////////////////////////////////////////////////////// +// Bmumu_reco_mumu.h +/////////////////////////////////////////////////////////////////// +// +// Author : Wolfgang Walkowiak <Wolfgang.Walkowiak@cern.ch.> +// Original author (Reco_mumu): +// Daniel Scheirich <daniel.scheirich@cern.ch> +// +// Changes: +// Basic dimuon reconstruction for the derivation framework. +// This class inherits from CfAthAlgTool instead of AthAlgTool in order +// to have access to the CutFlowSvc instance. +// +//============================================================================ +// +#ifndef DERIVATIONFRAMEWORK_Bmumu_reco_mumu_H +#define DERIVATIONFRAMEWORK_Bmumu_reco_mumu_H + +#include <string> + +#include "DerivationFrameworkBPhys/CfAthAlgTool.h" +#include "GaudiKernel/ToolHandle.h" +#include "DerivationFrameworkInterfaces/IAugmentationTool.h" +#include "JpsiUpsilonTools/JpsiFinder.h" +#include "JpsiUpsilonTools/PrimaryVertexRefitter.h" +#include "BeamSpotConditionsData/BeamSpotData.h" + +/** forward declarations + */ +namespace Trk { + class V0Tools; +} + +/** THE reconstruction tool + */ +namespace DerivationFramework { + + class Bmumu_reco_mumu : public CfAthAlgTool, public IAugmentationTool { + public: + Bmumu_reco_mumu(const std::string& t, const std::string& n, + const IInterface* p); + + StatusCode initialize(); + StatusCode finalize(); + + virtual StatusCode addBranches() const; + + private: + /** tools + */ + ToolHandle<Trk::V0Tools> m_v0Tools; + ToolHandle<Analysis::JpsiFinder> m_jpsiFinder; + ToolHandle<Analysis::PrimaryVertexRefitter> m_pvRefitter; + SG::ReadCondHandleKey<InDet::BeamSpotData> m_beamSpotKey { this, "BeamSpotKey", "BeamSpotData", "SG key for beam spot" }; + + /** job options + */ + std::string m_outputVtxContainerName; + std::string m_pvContainerName; + std::string m_refPVContainerName; + bool m_refitPV; + int m_PV_max; + int m_DoVertexType; + size_t m_PV_minNTracks; + bool m_do3d; + }; +} + +#endif // DERIVATIONFRAMEWORK_Bmumu_reco_mumu_H diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/Cascade3Plus1.h b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/Cascade3Plus1.h new file mode 100644 index 00000000000..7ed5f72f719 --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/Cascade3Plus1.h @@ -0,0 +1,99 @@ +/* + Copyright (C) 2002-2018 CERN for the benefit of the ATLAS collaboration +*/ +#ifndef DERIVATIONFRAMEWORKBPHYS_CASCADE3PLUS1_H +#define DERIVATIONFRAMEWORKBPHYS_CASCADE3PLUS1_H +//********************* +// Cascade3Plus1 header file +// +// Adam Barton <abarton@cern.ch> +#include "AthenaBaseComps/AthAlgTool.h" +#include "GaudiKernel/ToolHandle.h" +#include <vector> +#include "DerivationFrameworkInterfaces/IAugmentationTool.h" +#include "xAODTracking/TrackParticle.h" +#include "JpsiUpsilonTools/PrimaryVertexRefitter.h" +#include "InDetConversionFinderTools/VertexPointEstimator.h" +#include "BeamSpotConditionsData/BeamSpotData.h" + +namespace Trk { + class IVertexFitter; + class TrkVKalVrtFitter; + class VxTrackAtVertex; + class ITrackSelectorTool; + class VxCascadeInfo; + class V0Tools; +} + + +namespace DerivationFramework { + class CascadeTools; +} + +namespace DerivationFramework { + static const InterfaceID IID_Cascade3Plus1("Cascade3Plus1", 1, 0); +class Cascade3Plus1 : virtual public AthAlgTool, public IAugmentationTool +{ + +public: + static const InterfaceID& interfaceID() { return IID_Cascade3Plus1;} + Cascade3Plus1(const std::string& t, const std::string& n, const IInterface* p); + virtual ~Cascade3Plus1(); + virtual StatusCode initialize() override; + virtual StatusCode addBranches() const override; + +private: + static constexpr int s_topoN = 2; + + ToolHandle < Trk::ITrackSelectorTool > m_trkSelector; + ToolHandle < Trk::TrkVKalVrtFitter > m_iVertexFitter; + ToolHandle < Trk::V0Tools > m_V0Tools; + ToolHandle < DerivationFramework::CascadeTools > m_CascadeTools; + ToolHandle < Analysis::PrimaryVertexRefitter > m_pvRefitter; + ToolHandle < InDet::VertexPointEstimator > m_vertexEstimator; + SG::ReadCondHandleKey<InDet::BeamSpotData> m_beamSpotKey { this, "BeamSpotKey", "BeamSpotData", "SG key for beam spot" }; + + std::unique_ptr<Trk::VxCascadeInfo> CascadeFit(std::array<const xAOD::TrackParticle*, 4> &Track)const; + + std::vector<double> m_trackMasses; + std::vector<std::string> m_cascadeOutputsKeys; + double m_2trackmassMin = 979.45; + double m_2trackmassMax = 1059.45; + double m_3trackmassMin = 1800.47; + double m_3trackmassMax = 2168.47; + double m_4trackmassMin = 5200.0; + double m_4trackmassMax = 5450.0; + double m_3tracksMass = 1968.47; + double m_4tracksMass = 5366.79; + double m_2tracksMass = 0; + + double m_4trackmassFinalMin = 0; + double m_4trackmassFinalMax = 0; + std::string m_hypoName; //!< name of the mass hypothesis. E.g. Jpsi, Upsi, etc. Will be used as a prefix for decorations + std::string m_3TrackName; + int m_PV_max; + int m_DoVertexType; + size_t m_PV_minNTracks; + std::string m_VxPrimaryCandidateName; //!< Name of primary vertex container + std::string m_refPVContainerName; + double m_Chi2NDFCut=0.; + float m_3TrackChi2NDFCut=0.; + double m_tauCut = -999999; + bool m_refitPV; + bool m_3TrackMassConstraint = false; + bool m_2TrackMassConstraint = false; + bool m_eliminateBad3Tracksfrom4Track = false; + bool m_copyAllVertices = false; + std::bitset<4> m_muonTrackBit{0}; + std::vector<int> m_requireMuonsOnTrack; + std::string m_3TrackVertexOutput; + std::unique_ptr<xAOD::Vertex> StandardFit(const std::vector<const xAOD::TrackParticle*> &inputTracks, const xAOD::TrackParticleContainer* importedTrackCollection) const; + std::vector<double> m_ptCutPerTrack; + std::array<double, 3> m_ptCutPerVertex; + const std::vector<const xAOD::TrackParticle*>& ApplyAdditionalCuts(const std::vector<const xAOD::TrackParticle*>&, + const std::vector<const xAOD::TrackParticle*>&, + std::vector<const xAOD::TrackParticle*>&, size_t) const; +}; +} + +#endif diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/CascadeTools.h b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/CascadeTools.h new file mode 100644 index 00000000000..932b814d3d2 --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/CascadeTools.h @@ -0,0 +1,87 @@ +/* + Copyright (C) 2002-2017 CERN for the benefit of the ATLAS collaboration +*/ +#ifndef DERIVATIONFRAMEWORKBPHYS_CASCADETOOLS_H +#define DERIVATIONFRAMEWORKBPHYS_CASCADETOOLS_H +//********************* +// CascadeTools header file +// +// Eva Bouhova <e.bouhova@cern.ch> +// Adam Barton <abarton@cern.ch> + +#include "AthenaBaseComps/AthAlgTool.h" +#include "CLHEP/Vector/LorentzVector.h" +#include "xAODTracking/Vertex.h" + + +namespace DerivationFramework { + + static const InterfaceID IID_CascadeTools("CascadeTools", 1, 1); + + class CascadeTools : public AthAlgTool{ + + public: + +/** + * Default constructor due to Athena interface + */ + CascadeTools(const std::string& t, const std::string& n, const IInterface* p); + +/** + * Virtual destructor + */ + ~CascadeTools(); + +/** + * Standard AlgTool methods + */ + //Nothing done not needed + //StatusCode initialize() override; + //StatusCode finalize() override; + +/** + * AlgTool interface methods + */ + static const InterfaceID& interfaceID() + { + return IID_CascadeTools; + } + + + Amg::Vector3D momentum(const std::vector<TLorentzVector> &particleMom) const; + Amg::Vector3D pca(const std::vector<TLorentzVector> &particleMom, const xAOD::Vertex* SV, const xAOD::Vertex* PV) const; + + double invariantMass(const std::vector<TLorentzVector> &moms, const std::vector<double> &masses) const; + double invariantMass(const std::vector<TLorentzVector> &moms) const; + double invariantMassError(const std::vector<TLorentzVector> &moms, const Amg::MatrixX& cov, const std::vector<double> &masses) const; + double invariantMassError(const std::vector<TLorentzVector> &moms, const Amg::MatrixX& cov) const; + double pT(const std::vector<TLorentzVector> &moms) const; + double pTError(const std::vector<TLorentzVector> &moms, const Amg::MatrixX& cov) const; + double lxy(const std::vector<TLorentzVector> &particleMom, const xAOD::Vertex* SV, const xAOD::Vertex* PV) const; + double lxyError(const std::vector<TLorentzVector> &particleMom, const Amg::MatrixX& cov, const xAOD::Vertex* SV, const xAOD::Vertex* PV) const; + double tau(const std::vector<TLorentzVector> &particleMom, const xAOD::Vertex* SV, const xAOD::Vertex* PV) const; + double tauError(const std::vector<TLorentzVector> &particleMom, const Amg::MatrixX& cov, const xAOD::Vertex* SV, const xAOD::Vertex* PV) const; + double tau(const std::vector<TLorentzVector> &particleMom, const xAOD::Vertex* SV, const xAOD::Vertex* PV, double M) const; + double tauError(const std::vector<TLorentzVector> &particleMom, const Amg::MatrixX& cov, const xAOD::Vertex* SV, const xAOD::Vertex* PV, double M) const; + double a0z(const std::vector<TLorentzVector> &particleMom, const xAOD::Vertex* SV, const xAOD::Vertex* PV) const; + double a0zError(const std::vector<TLorentzVector> &particleMom, const Amg::MatrixX& cov, const xAOD::Vertex* SV, const xAOD::Vertex* PV) const; + double a0xy(const std::vector<TLorentzVector> &particleMom, const xAOD::Vertex* SV, const xAOD::Vertex* PV) const; + double a0xyError(const std::vector<TLorentzVector> &particleMom, const Amg::MatrixX& cov, const xAOD::Vertex* SV, const xAOD::Vertex* PV) const; + double a0(const std::vector<TLorentzVector> &particleMom, const xAOD::Vertex* SV, const xAOD::Vertex* PV) const; + double a0Error(const std::vector<TLorentzVector> &particleMom, const Amg::MatrixX& cov, const xAOD::Vertex* SV, const xAOD::Vertex* PV) const; + double cosTheta(const std::vector<TLorentzVector> &particleMom, const xAOD::Vertex* SV, const xAOD::Vertex* PV) const; + double cosTheta_xy(const std::vector<TLorentzVector> &particleMom, const xAOD::Vertex* SV, const xAOD::Vertex* PV) const; + double massProbability(double V0Mass, double mass, double massErr) const; + double vertexProbability(int ndf, double chi2) const; + + Amg::MatrixX * convertCovMatrix(const xAOD::Vertex * vxCandidate) const; + Amg::MatrixX SetFullMatrix(int NTrk, const std::vector<float> & Matrix) const; + + //private: + + }; //end of class definitions + +} //end of namespace definitions + + +#endif diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/CfAthAlgTool.h b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/CfAthAlgTool.h new file mode 100644 index 00000000000..f5280c5ccde --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/CfAthAlgTool.h @@ -0,0 +1,86 @@ +/* + Copyright (C) 2002-2017 CERN for the benefit of the ATLAS collaboration +*/ + +//============================================================================ +// CfAthAlgTool.h +//============================================================================ +// +// Author : Wolfgang Walkowiak <Wolfgang.Walkowiak@cern.ch.> +// Changes: +// +// Wrapper around AthAlgTool to provide easy access to CutFlowSvc +// and some utility methods for it. +// Methods for accessing the CutFlowSvc are modelled after +// AthFilterAlgorithm's implementation. +// +// This class inherits from AthAlgTool. It should be inherited from. +// +//============================================================================ +// +#ifndef DERIVATIONFRAMEWORK_CfAthAlgTool_H +#define DERIVATIONFRAMEWORK_CfAthAlgTool_H + +#include "GaudiKernel/ToolHandle.h" +#include "GaudiKernel/ServiceHandle.h" +#include "AthenaKernel/ICutFlowSvc.h" +#include "AthenaBaseComps/AthAlgTool.h" + +#include <string> +#include <map> + +namespace DerivationFramework { + + class CfAthAlgTool : public AthAlgTool { + public: + // constructor with parameters + CfAthAlgTool(const std::string& t, const std::string& n, + const IInterface* p); + // destructor + virtual ~CfAthAlgTool(); + + // return a handle to an ICutFlowSvc instance + ServiceHandle<ICutFlowSvc>& cutFlowSvc() const; + + // Initialization method invoked by the framework. + virtual StatusCode sysInitialize() override; + + // add event to a named counter -- returns counts after adding + virtual bool addEvent(const std::string &name, double weight=1.) const; + + // add to a named counter -- returns counts after adding + // if counts > 1 : same weight is added multiple times + virtual bool addToCounter(const std::string &name, uint64_t counts=1, + double weight=1.) const; + + protected: + // add a counter by name -- returns id if it already exists + virtual CutIdentifier getCounter(const std::string &name) const; + + // returns counter name by id + virtual std::string getCounterNameById(CutIdentifier id) const; + + // returns counter id by name + virtual CutIdentifier getCounterIdByName(const std::string &name) const; + + private: + // typedef for ServiceHandle<ICutFlowSvc> + typedef ServiceHandle<ICutFlowSvc> ICutFlowSvc_t; + // handle to the service holding tables of cut-flows for filtering algs. + mutable ICutFlowSvc_t m_cutFlowSvc; + + // base name for counters + std::string m_ctbasename; + + // map of counter names to counter ids + typedef std::map<std::string, CutIdentifier> NameIdMap_t; + mutable NameIdMap_t m_mctn; + + // base counter + mutable CutIdentifier m_bid; + mutable bool m_bidisset; + + }; // class +} // namespace + +#endif // DERIVATIONFRAMEWORK_CfAthAlgTool_H diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/FourMuonTool.h b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/FourMuonTool.h new file mode 100644 index 00000000000..88348740f02 --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/FourMuonTool.h @@ -0,0 +1,188 @@ +/* + Copyright (C) 2002-2017 CERN for the benefit of the ATLAS collaboration +*/ + +// **************************************************************************** +// ---------------------------------------------------------------------------- +// FourMuonTool header file +// +// James Catmore <James.Catmore@cern.ch> + +// ---------------------------------------------------------------------------- +// **************************************************************************** +#ifndef BPHY4TOOL_H +#define BPHY4TOOL_H +#include "AthenaBaseComps/AthAlgorithm.h" +#include "GaudiKernel/ToolHandle.h" +#include "TrkVKalVrtFitter/TrkVKalVrtFitter.h" +#include "InDetConversionFinderTools/InDetConversionFinderTools.h" +#include "BeamSpotConditionsData/BeamSpotData.h" +#include "xAODMuon/Muon.h" +#include "xAODMuon/MuonContainer.h" + +#include <vector> +#include <string> +///////////////////////////////////////////////////////////////////////////// + +namespace Trk { + class IVertexFitter; + // class VxCandidate; + // class TrackParticleBase; + // class VxTrackAtVertex; + // class RecVertex; + class TrkV0VertexFitter; + class ITrackSelectorTool; + class V0Tools; + // class ExtendedVxCandidate; +} + +namespace InDet { class VertexPointEstimator; class BeamSpotData;} + +namespace DerivationFramework { + + static const InterfaceID IID_FourMuonTool("FourMuonTool", 1, 0); + + // Struct and enum to associate muon pairs with track pairs + // and make the program flow more straightforward + struct Combination + { + std::vector<const xAOD::Muon*> muons; + std::vector<unsigned int> quadIndices; + std::pair<unsigned int, unsigned int> pairIndices; + + std::string combinationCharges() { + std::string chargeStr = ""; + if (muons.at(0)->charge() > 0) {chargeStr += "+";} + else {chargeStr += "-";} + if (muons.at(1)->charge() > 0) {chargeStr += "+";} + else {chargeStr += "-";} + if (muons.size()==4) { + if (muons.at(2)->charge() > 0) {chargeStr += "+";} + else {chargeStr += "-";} + if (muons.at(3)->charge() > 0) {chargeStr += "+";} + else {chargeStr += "-";} + } + return chargeStr; + } + + std::string combinationIndices() { + std::string indexStr = ""; + std::stringstream ss; + if (muons.size()==2) { + ss.str(""); ss.clear(); + ss << pairIndices.first; + indexStr+=ss.str(); + ss.str(""); ss.clear(); + ss << pairIndices.second; + indexStr+=ss.str(); + } + if (muons.size()==4) { + for (unsigned int i=0; i<4; ++i) { + ss.str(""); ss.clear(); + ss << quadIndices[i]; + indexStr+=ss.str(); + } + } + return indexStr; + } + + const xAOD::TrackParticle* GetMuonTrack(const xAOD::Muon* mu) const{ + auto& link = mu->inDetTrackParticleLink(); + return link.isValid() ? *link : nullptr; + } + + std::vector<const xAOD::TrackParticle*> trackParticles(std::string specify) { + std::vector<const xAOD::TrackParticle*> theTracks; + bool oppCh(false); + if (muons.at(0)->charge()*muons.at(1)->charge() < 0) oppCh=true; + if (specify=="pair1") { + theTracks.push_back(GetMuonTrack(muons.at(0))); + theTracks.push_back(GetMuonTrack(muons.at(1))); + } + if (specify=="pair2") { + theTracks.push_back(GetMuonTrack(muons.at(2))); + theTracks.push_back(GetMuonTrack(muons.at(3))); + } + if (specify=="DC") { + if (oppCh) { + theTracks.push_back(GetMuonTrack(muons.at(0))); + theTracks.push_back(GetMuonTrack(muons.at(1))); + theTracks.push_back(GetMuonTrack(muons.at(2))); + theTracks.push_back(GetMuonTrack(muons.at(3))); + } else { + theTracks.push_back(GetMuonTrack(muons.at(0))); + theTracks.push_back(GetMuonTrack(muons.at(2))); + theTracks.push_back(GetMuonTrack(muons.at(1))); + theTracks.push_back(GetMuonTrack(muons.at(3))); + } + } + if (specify=="AC") { + theTracks.push_back(GetMuonTrack(muons.at(0))); + theTracks.push_back(GetMuonTrack(muons.at(3))); + theTracks.push_back(GetMuonTrack(muons.at(1))); + theTracks.push_back(GetMuonTrack(muons.at(2))); + } + if (specify=="SS") { + if (oppCh) { + theTracks.push_back(GetMuonTrack(muons.at(0))); + theTracks.push_back(GetMuonTrack(muons.at(2))); + theTracks.push_back(GetMuonTrack(muons.at(1))); + theTracks.push_back(GetMuonTrack(muons.at(3))); + } else { + theTracks.push_back(GetMuonTrack(muons.at(0))); + theTracks.push_back(GetMuonTrack(muons.at(1))); + theTracks.push_back(GetMuonTrack(muons.at(2))); + theTracks.push_back(GetMuonTrack(muons.at(3))); + } + } + return theTracks; + } + + }; + + class FourMuonTool: virtual public AthAlgTool + { + public: + FourMuonTool(const std::string& t, const std::string& n, const IInterface* p); + ~FourMuonTool(); + StatusCode initialize(); + + static const InterfaceID& interfaceID() { return IID_FourMuonTool;} + + //------------------------------------------------------------------------------------- + //Doing Calculation and inline functions + StatusCode performSearch(xAOD::VertexContainer*& pairVxContainer, xAOD::VertexAuxContainer*& pairVxAuxContainer, + xAOD::VertexContainer*& quadVxContainer, xAOD::VertexAuxContainer*& quadVxAuxContainer, bool &acceptEvent) const; + xAOD::Vertex* fit(const std::vector<const xAOD::TrackParticle*>& ,const xAOD::TrackParticleContainer* importedTrackCollection, const Amg::Vector3D &beamSpot) const; + static std::vector<std::vector<unsigned int> > getQuadIndices(unsigned int length); + static std::vector<std::pair<unsigned int, unsigned int> > getPairIndices(unsigned int length); + static std::vector<std::vector<unsigned int> > mFromN(unsigned int m, unsigned int n); + static void combinatorics(unsigned int offset, + unsigned int k, + std::vector<unsigned int> &combination, + std::vector<unsigned int> &mainList, + std::vector<std::vector<unsigned int> > &allCombinations); + static void buildCombinations(const std::vector<const xAOD::Muon*> &muonsIn, + std::vector<Combination> &pairs, + std::vector<Combination> &quadruplets, + unsigned int nSelectedMuons); + static bool passesQuadSelection(const std::vector<const xAOD::Muon*> &muonsIn); + //------------------------------------------------------------------------------------- + + private: + double m_ptCut; + double m_etaCut; + bool m_useV0Fitter; + SG::ReadHandleKey<xAOD::MuonContainer> m_muonCollectionKey; + SG::ReadHandleKey<xAOD::TrackParticleContainer> m_TrkParticleCollection; + ToolHandle < Trk::IVertexFitter > m_iVertexFitter; + ToolHandle < Trk::IVertexFitter > m_iV0VertexFitter; + ToolHandle < Trk::V0Tools > m_V0Tools; + ToolHandle < Trk::ITrackSelectorTool > m_trkSelector; + ToolHandle < InDet::VertexPointEstimator > m_vertexEstimator; + SG::ReadCondHandleKey<InDet::BeamSpotData> m_beamSpotKey { this, "BeamSpotKey", "BeamSpotData", "SG key for beam spot" }; + SG::WriteDecorHandleKey<xAOD::MuonContainer> m_muonIndex{this, "muonIndexDec", "Muons.BPHY4MuonIndex"}; + + }; +} // end of namespace +#endif diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/JpsiPlusDpstCascade.h b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/JpsiPlusDpstCascade.h new file mode 100644 index 00000000000..7376be66ae1 --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/JpsiPlusDpstCascade.h @@ -0,0 +1,96 @@ +/* + Copyright (C) 2002-2018 CERN for the benefit of the ATLAS collaboration +*/ +#ifndef JPSIPLUSDPSTCASCADE_H +#define JPSIPLUSDPSTCASCADE_H +//********************* +// JpsiPlusDpstCascade header file +// +// Eva Bouhova <e.bouhova@cern.ch> +// Adam Barton <abarton@cern.ch> + +#include "AthenaBaseComps/AthAlgTool.h" +#include "GaudiKernel/ToolHandle.h" +#include "DerivationFrameworkInterfaces/IAugmentationTool.h" +#include "JpsiUpsilonTools/PrimaryVertexRefitter.h" +#include <vector> +#include "BeamSpotConditionsData/BeamSpotData.h" + +namespace Trk { + class IVertexFitter; + class TrkVKalVrtFitter; + class IVertexCascadeFitter; + class VxCascadeInfo; + class V0Tools; + class ParticleDataTable; +} + +namespace DerivationFramework { + class CascadeTools; +} + +namespace DerivationFramework { + + static const InterfaceID IID_JpsiPlusDpstCascade("JpsiPlusDpstCascade", 1, 0); + + class JpsiPlusDpstCascade : virtual public AthAlgTool, public IAugmentationTool + { + public: + static const InterfaceID& interfaceID() { return IID_JpsiPlusDpstCascade;} + JpsiPlusDpstCascade(const std::string& t, const std::string& n, const IInterface* p); + ~JpsiPlusDpstCascade(); + virtual StatusCode initialize() override; + StatusCode performSearch(std::vector<Trk::VxCascadeInfo*> *cascadeinfoContainer ) const; + virtual StatusCode addBranches() const override; + + private: + std::string m_vertexContainerKey; + std::string m_vertexD0ContainerKey; + std::vector<std::string> m_cascadeOutputsKeys; + + std::string m_VxPrimaryCandidateName; //!< Name of primary vertex container + + double m_jpsiMassLower; + double m_jpsiMassUpper; + double m_jpsipiMassLower; + double m_jpsipiMassUpper; + double m_D0MassLower; + double m_D0MassUpper; + double m_DstMassLower; + double m_DstMassUpper; + double m_MassLower; + double m_MassUpper; + double m_vtx0MassHypo; // mass hypothesis of vertex 0 + double m_vtx1MassHypo; // mass hypothesis of vertex 1 + double m_vtx0Daug1MassHypo; // mass hypothesis of 1st daughter from vertex 0 + double m_vtx0Daug2MassHypo; // mass hypothesis of 2nd daughter from vertex 0 + double m_vtx0Daug3MassHypo; // mass hypothesis of 3rd daughter from vertex 0 + double m_vtx1Daug1MassHypo; // mass hypothesis of 1st daughter from vertex 1 + double m_vtx1Daug2MassHypo; // mass hypothesis of 2nd daughter from vertex 1 + + + double m_mass_jpsi; + int m_Dx_pid; + bool m_constrD0; + bool m_constrJpsi; + double m_chi2cut; + + SG::ReadCondHandleKey<InDet::BeamSpotData> m_beamSpotKey { this, "BeamSpotKey", "BeamSpotData", "SG key for beam spot" }; + ToolHandle < Trk::TrkVKalVrtFitter > m_iVertexFitter; + ToolHandle < Analysis::PrimaryVertexRefitter > m_pvRefitter; + ToolHandle < Trk::V0Tools > m_V0Tools; + ToolHandle < DerivationFramework::CascadeTools > m_CascadeTools; + + bool m_refitPV; + std::string m_refPVContainerName; + std::string m_hypoName; //!< name of the mass hypothesis. E.g. Jpsi, Upsi, etc. Will be used as a prefix for decorations + //This parameter will allow us to optimize the number of PVs under consideration as the probability + //of a useful primary vertex drops significantly the higher you go + int m_PV_max; + int m_DoVertexType; + size_t m_PV_minNTracks; + + }; +} + +#endif diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/JpsiPlusDs1Cascade.h b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/JpsiPlusDs1Cascade.h new file mode 100644 index 00000000000..d923f5fe4f9 --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/JpsiPlusDs1Cascade.h @@ -0,0 +1,105 @@ +/* + Copyright (C) 2002-2018 CERN for the benefit of the ATLAS collaboration +*/ +#ifndef JPSIPLUSDS1CASCADE_H +#define JPSIPLUSDS1CASCADE_H +//********************* +// JpsiPlusDs1Cascade header file +// +// Eva Bouhova <e.bouhova@cern.ch> +// Adam Barton <abarton@cern.ch> + +#include "AthenaBaseComps/AthAlgTool.h" +#include "GaudiKernel/ToolHandle.h" +#include "HepPDT/ParticleDataTable.hh" +#include "DerivationFrameworkInterfaces/IAugmentationTool.h" +#include "JpsiUpsilonTools/PrimaryVertexRefitter.h" +#include <vector> +#include "BeamSpotConditionsData/BeamSpotData.h" + +namespace Trk { + class IVertexFitter; + class TrkVKalVrtFitter; + class IVertexCascadeFitter; + class VxCascadeInfo; + class V0Tools; + class ParticleDataTable; +} + +namespace DerivationFramework { + class CascadeTools; +} + +namespace DerivationFramework { + + static const InterfaceID IID_JpsiPlusDs1Cascade("JpsiPlusDs1Cascade", 1, 0); + + class JpsiPlusDs1Cascade : virtual public AthAlgTool, public IAugmentationTool + { + public: + static const InterfaceID& interfaceID() { return IID_JpsiPlusDs1Cascade;} + JpsiPlusDs1Cascade(const std::string& t, const std::string& n, const IInterface* p); + ~JpsiPlusDs1Cascade(); + virtual StatusCode initialize() override; + StatusCode performSearch(std::vector<Trk::VxCascadeInfo*> *cascadeinfoContainer ) const; + virtual StatusCode addBranches() const override; + + private: + std::string m_vertexContainerKey; + std::string m_vertexD0ContainerKey; + std::string m_vertexK0ContainerKey; + std::vector<std::string> m_cascadeOutputsKeys; + + std::string m_VxPrimaryCandidateName; //!< Name of primary vertex container + + double m_jpsiMassLower; + double m_jpsiMassUpper; + double m_jpsipiMassLower; + double m_jpsipiMassUpper; + double m_D0MassLower; + double m_D0MassUpper; + double m_K0MassLower; + double m_K0MassUpper; + double m_DstMassLower; + double m_DstMassUpper; + double m_MassLower; + double m_MassUpper; + double m_vtx0MassHypo; // mass hypothesis of vertex 0 + double m_vtx1MassHypo; // mass hypothesis of vertex 1 + double m_vtx2MassHypo; // mass hypothesis of vertex 2 + double m_vtx0Daug1MassHypo; // mass hypothesis of 1st daughter from vertex 0 + double m_vtx0Daug2MassHypo; // mass hypothesis of 2nd daughter from vertex 0 + double m_vtx0Daug3MassHypo; // mass hypothesis of 3rd daughter from vertex 0 + double m_vtx1Daug1MassHypo; // mass hypothesis of 1st daughter from vertex 1 + double m_vtx1Daug2MassHypo; // mass hypothesis of 2nd daughter from vertex 1 + double m_vtx2Daug1MassHypo; // mass hypothesis of 1st daughter from vertex 2 + double m_vtx2Daug2MassHypo; // mass hypothesis of 2nd daughter from vertex 2 + + const HepPDT::ParticleDataTable* m_particleDataTable; + double m_mass_jpsi; + int m_Dx_pid; + bool m_constrD0; + bool m_constrK0; + bool m_constrJpsi; + double m_chi2cut; + + SG::ReadCondHandleKey<InDet::BeamSpotData> m_beamSpotKey { this, "BeamSpotKey", "BeamSpotData", "SG key for beam spot" }; + ToolHandle < Trk::TrkVKalVrtFitter > m_iVertexFitter; + ToolHandle < Analysis::PrimaryVertexRefitter > m_pvRefitter; + ToolHandle < Trk::V0Tools > m_V0Tools; + ToolHandle < DerivationFramework::CascadeTools > m_CascadeTools; + + bool m_refitPV; + std::string m_refPVContainerName; + std::string m_hypoName; //!< name of the mass hypothesis. E.g. Jpsi, Upsi, etc. Will be used as a prefix for decorations + //This parameter will allow us to optimize the number of PVs under consideration as the probability + //of a useful primary vertex drops significantly the higher you go + int m_PV_max; + int m_DoVertexType; + size_t m_PV_minNTracks; + + double getParticleMass(int particlecode) const; + }; +} + +#endif diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/JpsiPlusDsCascade.h b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/JpsiPlusDsCascade.h new file mode 100644 index 00000000000..024fe6a7313 --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/JpsiPlusDsCascade.h @@ -0,0 +1,91 @@ +/* + Copyright (C) 2002-2018 CERN for the benefit of the ATLAS collaboration +*/ +#ifndef JPSIPLUSDSCASCADE_H +#define JPSIPLUSDSCASCADE_H +//********************* +// JpsiPlusDsCascade header file +// +// Eva Bouhova <e.bouhova@cern.ch> +// Adam Barton <abarton@cern.ch> + +#include "AthenaBaseComps/AthAlgTool.h" +#include "GaudiKernel/ToolHandle.h" +#include "DerivationFrameworkInterfaces/IAugmentationTool.h" +#include "JpsiUpsilonTools/PrimaryVertexRefitter.h" +#include <vector> +#include "BeamSpotConditionsData/BeamSpotData.h" + +namespace Trk { + class IVertexFitter; + class TrkVKalVrtFitter; + class IVertexCascadeFitter; + class VxCascadeInfo; + class V0Tools; + class ParticleDataTable; +} + +namespace DerivationFramework { + class CascadeTools; +} + +namespace DerivationFramework { + + static const InterfaceID IID_JpsiPlusDsCascade("JpsiPlusDsCascade", 1, 0); + + class JpsiPlusDsCascade : virtual public AthAlgTool, public IAugmentationTool + { + public: + static const InterfaceID& interfaceID() { return IID_JpsiPlusDsCascade;} + JpsiPlusDsCascade(const std::string& t, const std::string& n, const IInterface* p); + ~JpsiPlusDsCascade(); + virtual StatusCode initialize() override; + StatusCode performSearch(std::vector<Trk::VxCascadeInfo*> *cascadeinfoContainer ) const; + virtual StatusCode addBranches() const override; + + private: + std::string m_vertexContainerKey; + std::string m_vertexDxContainerKey; + std::vector<std::string> m_cascadeOutputsKeys; + + std::string m_VxPrimaryCandidateName; //!< Name of primary vertex container + + double m_jpsiMassLower; + double m_jpsiMassUpper; + double m_DxMassLower; + double m_DxMassUpper; + double m_MassLower; + double m_MassUpper; + double m_vtx0MassHypo; // 1st vertex mass hypothesis + double m_vtx1MassHypo; // 2nd vertex mass hypothesis + double m_vtx0Daug1MassHypo; // mass hypothesis of 1st daughter from vertex 0 + double m_vtx0Daug2MassHypo; // mass hypothesis of 2nd daughter from vertex 0 + double m_vtx1Daug1MassHypo; // mass hypothesis of 1st daughter from vertex 1 + double m_vtx1Daug2MassHypo; // mass hypothesis of 2nd daughter from vertex 1 + double m_vtx1Daug3MassHypo; // mass hypothesis of 3rd daughter from vertex 1 + + double m_mass_jpsi; + int m_Dx_pid; + bool m_constrDx; + bool m_constrJpsi; + double m_chi2cut; + + SG::ReadCondHandleKey<InDet::BeamSpotData> m_beamSpotKey { this, "BeamSpotKey", "BeamSpotData", "SG key for beam spot" }; + ToolHandle < Trk::TrkVKalVrtFitter > m_iVertexFitter; + ToolHandle < Analysis::PrimaryVertexRefitter > m_pvRefitter; + ToolHandle < Trk::V0Tools > m_V0Tools; + ToolHandle < DerivationFramework::CascadeTools > m_CascadeTools; + + bool m_refitPV; + std::string m_refPVContainerName; + std::string m_hypoName; //!< name of the mass hypothesis. E.g. Jpis, Upsi, etc. Will be used as a prefix for decorations + //This parameter will allow us to optimize the number of PVs under consideration as the probability + //of a useful primary vertex drops significantly the higher you go + int m_PV_max; + int m_DoVertexType; + size_t m_PV_minNTracks; + + }; +} + +#endif diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/JpsiPlusV0Cascade.h b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/JpsiPlusV0Cascade.h new file mode 100644 index 00000000000..9720b3d6ca3 --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/JpsiPlusV0Cascade.h @@ -0,0 +1,94 @@ +/* + Copyright (C) 2002-2017 CERN for the benefit of the ATLAS collaboration +*/ +#ifndef JPSIPLUSV0CASCADE_H +#define JPSIPLUSV0CASCADE_H +//********************* +// JpsiPlusV0Cascade header file +// +// Eva Bouhova <e.bouhova@cern.ch> +// Adam Barton <abarton@cern.ch> + +#include "AthenaBaseComps/AthAlgTool.h" +#include "GaudiKernel/ToolHandle.h" + +#include "DerivationFrameworkInterfaces/IAugmentationTool.h" +#include "JpsiUpsilonTools/PrimaryVertexRefitter.h" +#include <vector> +#include "BeamSpotConditionsData/BeamSpotData.h" + + +namespace Trk { + class IVertexFitter; + class TrkVKalVrtFitter; + class IVertexCascadeFitter; + class VxCascadeInfo; + class V0Tools; + class ParticleDataTable; +} + +namespace DerivationFramework { + class CascadeTools; +} + + +namespace DerivationFramework { + + static const InterfaceID IID_JpsiPlusV0Cascade("JpsiPlusV0Cascade", 1, 0); + + class JpsiPlusV0Cascade : virtual public AthAlgTool, public IAugmentationTool + { + + std::string m_vertexContainerKey; + std::string m_vertexV0ContainerKey; + std::vector<std::string> m_cascadeOutputsKeys; + + std::string m_VxPrimaryCandidateName; //!< Name of primary vertex container + + double m_jpsiMassLower; + double m_jpsiMassUpper; + double m_V0MassLower; + double m_V0MassUpper; + double m_MassLower; + double m_MassUpper; + + double m_mass_muon; + double m_mass_pion; + double m_mass_proton; + double m_mass_lambda; + double m_mass_ks; + double m_mass_jpsi; + double m_mass_b0; + double m_mass_lambdaB; + int m_v0_pid; + bool m_constrV0; + bool m_constrJpsi; + + SG::ReadCondHandleKey<InDet::BeamSpotData> m_beamSpotKey { this, "BeamSpotKey", "BeamSpotData", "SG key for beam spot" }; + ToolHandle < Trk::TrkVKalVrtFitter > m_iVertexFitter; + ToolHandle < Analysis::PrimaryVertexRefitter > m_pvRefitter; + ToolHandle < Trk::V0Tools > m_V0Tools; + ToolHandle < DerivationFramework::CascadeTools > m_CascadeTools; + + bool m_refitPV; + std::string m_refPVContainerName; + std::string m_hypoName; //!< name of the mass hypothesis. E.g. Jpis, Upsi, etc. Will be used as a prefix for decorations + //This parameter will allow us to optimize the number of PVs under consideration as the probability + //of a useful primary vertex drops significantly the higher you go + int m_PV_max; + int m_DoVertexType; + size_t m_PV_minNTracks; + + public: + static const InterfaceID& interfaceID() { return IID_JpsiPlusV0Cascade;} + JpsiPlusV0Cascade(const std::string& t, const std::string& n, const IInterface* p); + ~JpsiPlusV0Cascade(); + StatusCode initialize() override; + StatusCode performSearch(std::vector<Trk::VxCascadeInfo*> *cascadeinfoContainer ) const; + virtual StatusCode addBranches() const override; + }; +} + + +#endif + diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/LocalVector.h b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/LocalVector.h new file mode 100644 index 00000000000..1865c520e35 --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/LocalVector.h @@ -0,0 +1,72 @@ +/* + Copyright (C) 2002-2017 CERN for the benefit of the ATLAS collaboration +*/ +#ifndef LOCALVECTOR_H +#define LOCALVECTOR_H + +#include <array> +#include <algorithm> +#include <iterator> + +template <class A_Type, size_t SIZE> +class LocalVector +{ +protected: + std::array<A_Type, SIZE> m_memblock; + size_t m_x; +public: + LocalVector() : m_x(0) { } + size_t size() const noexcept { + return m_x; + } + constexpr size_t max_size() const noexcept { + return SIZE; + } + typename std::array<A_Type, SIZE>::iterator begin() noexcept { + return m_memblock.begin(); + } + typename std::array<A_Type, SIZE>::iterator end() noexcept { + auto it = m_memblock.begin(); + std::advance(it, m_x); + return it; + } + typename std::array<A_Type, SIZE>::const_iterator begin() const noexcept { + return m_memblock.cbegin(); + } + typename std::array<A_Type, SIZE>::const_iterator end() const noexcept { + auto it = m_memblock.cbegin(); + std::advance(it, m_x); + return it; + } + A_Type& operator[]( size_t pos ) { + return m_memblock[pos]; + } + const A_Type& operator[]( size_t pos ) const { + return m_memblock[pos]; + } + bool contains(const A_Type& a) const { + return std::find(begin(), end(), a) != end(); + } + void push_back(const A_Type& a) { + m_memblock[m_x++] = a; + } + void pop_back() { + --m_x; + } + void clear() noexcept { + m_x = 0; + } + A_Type& back( ) { + return m_memblock[m_x-1]; + } + const A_Type& back( ) const { + return m_memblock[m_x-1]; + } + A_Type& front( ) { + return m_memblock.front(); + } + const A_Type& front( ) const { + return m_memblock.front(); + } +}; +#endif diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/MuonExtrapolationTool.h b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/MuonExtrapolationTool.h new file mode 100644 index 00000000000..7464e520f10 --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/MuonExtrapolationTool.h @@ -0,0 +1,59 @@ +/* + Copyright (C) 2002-2017 CERN for the benefit of the ATLAS collaboration + */ + +////////////////////////////////////////////////////////////////////////////// +// MuonExtrapolationTool +////////////////////////////////////////////////////////////////////////////// +#ifndef MuonExtrapolationTool_H +#define MuonExtrapolationTool_H + +#include "AthenaBaseComps/AthAlgTool.h" +#include "DerivationFrameworkInterfaces/IAugmentationTool.h" +#include "xAODTracking/TrackParticle.h" +#include "xAODMuon/Muon.h" +#include "TrkParameters/TrackParameters.h" + +namespace Trk { + class IExtrapolator; +} + +namespace DerivationFramework { + class MuonExtrapolationTool : public AthAlgTool, public IAugmentationTool { + + public: + MuonExtrapolationTool(const std::string& t, const std::string& n, const IInterface *p); + + virtual StatusCode initialize(); + virtual StatusCode addBranches() const; + + ToolHandle<Trk::IExtrapolator> m_extrapolator; + private: + + /// run the extrapolation - only available in full athena + const Trk::TrackParameters* extrapolateToTriggerPivotPlane(const xAOD::TrackParticle& track) const; + + // Utility method to handle extrapolation and decoration for one TrackParticle. + // It looks for the decoration, and, if it is missing, runs track extrapolation, decorating the result + // to the particle to avoid repeating the process unnecessarily. + // Returns success (true) or failure (false) of the procedure, fills eta and phi coordinates via reference + // If the extrapolation fails or the decoration is missing in AthAnalysis, it will *not* change eta and phi + // So you can set them to defaults before calling this guy, and they will be preserved in case of failure. + // Will not run outside athena, because it requires the extrapolator + bool extrapolateAndDecorateTrackParticle(const xAOD::TrackParticle* particle, float & eta, float & phi) const; + + // utility method: Obtains the track particle which we want to extrapolate into the MS. + // Works for all kinds of probes. + const xAOD::TrackParticle* getPreferredTrackParticle (const xAOD::IParticle* probe) const; + + // these define the surfaces that we extrapolate to. + // We approximate the pivot plane in the form of a cylinder surface and two disks + double m_endcapPivotPlaneZ; + double m_endcapPivotPlaneMinimumRadius; + double m_endcapPivotPlaneMaximumRadius; + double m_barrelPivotPlaneRadius; + double m_barrelPivotPlaneHalfLength; + std::string m_muonContainerName; + }; +} +#endif // MuonExtrapolationTool_H diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/ReVertex.h b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/ReVertex.h new file mode 100644 index 00000000000..853f03bcf72 --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/ReVertex.h @@ -0,0 +1,98 @@ +/* + Copyright (C) 2002-2018 CERN for the benefit of the ATLAS collaboration +*/ +// **************************************************************************** +// ---------------------------------------------------------------------------- +// ReVertex header file +// +// Konstantin Beloborodov <Konstantin.Beloborodov@cern.ch> +// +// ---------------------------------------------------------------------------- +// **************************************************************************** + +#ifndef DERIVATIONFRAMEWORK_ReVertex_H +#define DERIVATIONFRAMEWORK_ReVertex_H + +#include "AthenaBaseComps/AthAlgTool.h" +#include "GaudiKernel/ToolHandle.h" +#include "DerivationFrameworkInterfaces/IAugmentationTool.h" +#include "xAODTracking/Vertex.h" +#include "BeamSpotConditionsData/BeamSpotData.h" +/** forward declarations + */ +namespace Trk { + class IVertexFitter; + class TrkV0VertexFitter; + class ITrackSelectorTool; + class V0Tools; + class TrkVKalVrtFitter; +} + +namespace Analysis { + class PrimaryVertexRefitter; +} + +namespace InDet { class VertexPointEstimator; } + +namespace DerivationFramework { + +class ReVertex : public AthAlgTool, public IAugmentationTool { +public: + + ReVertex(const std::string& t, const std::string& n, const IInterface* p); + + virtual StatusCode initialize() override; + //virtual StatusCode finalize() override; + + virtual StatusCode addBranches() const override; + + void fitAndStore(xAOD::VertexContainer* vtxContainer, + const xAOD::Vertex* v, + const xAOD::VertexContainer *InVtxContainer, + const std::vector<const xAOD::TrackParticle*> &inputTracks, + const xAOD::TrackParticleContainer* importedTrackCollection, + const xAOD::VertexContainer* pvContainer) const; + xAOD::Vertex* fit(const std::vector<const xAOD::TrackParticle*> &inputTracks, + const xAOD::Vertex* pv) const; +private: + std::vector<int> m_TrackIndices; + ToolHandle < InDet::VertexPointEstimator > m_vertexEstimator; + ToolHandle < Trk::IVertexFitter > m_iVertexFitter; + Trk::TrkVKalVrtFitter* m_VKVFitter; + SG::WriteHandleKey<xAOD::VertexContainer> m_OutputContainerName; + SG::ReadHandleKey<xAOD::VertexContainer> m_inputContainerName; + SG::ReadHandleKey<xAOD::TrackParticleContainer> m_trackContainer; + SG::WriteHandleKey<xAOD::VertexContainer> m_refPVContainerName; + SG::ReadHandleKey<xAOD::VertexContainer> m_pvContainerName; + + + std::vector<double> m_trkMasses; + std::vector<int> m_indices; + double m_massConst; + double m_totalMassConst; + std::vector<std::string> m_hypoNames; + + ToolHandle<Trk::V0Tools> m_v0Tools; + ToolHandle<Analysis::PrimaryVertexRefitter> m_pvRefitter; + SG::ReadCondHandleKey<InDet::BeamSpotData> m_beamSpotKey { this, "BeamSpotKey", "BeamSpotData", "SG key for beam spot" }; + int m_PV_max; + int m_DoVertexType; + size_t m_PV_minNTracks; + bool m_do3d; + bool m_AddPVData; + bool m_refitPV; + bool m_doMassConst; + bool m_startingpoint0; + + bool m_vertexFittingWithPV; + + double m_BMassUpper; + double m_BMassLower; + double m_chi2cut; // chi2/Ndof of the final veretx + double m_trkDeltaZ; // DeltaZ between the JPsi vertex and hadronic tracks Z0 + + bool m_useAdditionalTrack; +}; +} + +#endif diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/Reco_4mu.h b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/Reco_4mu.h new file mode 100644 index 00000000000..5b311a975fd --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/Reco_4mu.h @@ -0,0 +1,63 @@ +/* + Copyright (C) 2002-2017 CERN for the benefit of the ATLAS collaboration +*/ + +/////////////////////////////////////////////////////////////////// +// Reco_4mu.h +/////////////////////////////////////////////////////////////////// + +#ifndef DERIVATIONFRAMEWORK_Reco_4mu_H +#define DERIVATIONFRAMEWORK_Reco_4mu_H + +#include <string> + +#include "AthenaBaseComps/AthAlgTool.h" +#include "DerivationFrameworkInterfaces/ISkimmingTool.h" +#include "DerivationFrameworkBPhys/FourMuonTool.h" +#include "JpsiUpsilonTools/PrimaryVertexRefitter.h" +#include "xAODBPhys/BPhysHelper.h" + +/** forward declarations + */ +namespace Trk { + class V0Tools; +} + +namespace xAOD { + class BPhysHypoHelper; +} + +/** THE reconstruction tool + */ +namespace DerivationFramework { + + class Reco_4mu : public AthAlgTool, public ISkimmingTool { + public: + Reco_4mu(const std::string& t, const std::string& n, const IInterface* p); + + StatusCode initialize(); + StatusCode finalize(); + + virtual bool eventPassesFilter() const; + + private: + /** tools + */ + void ProcessVertex(xAOD::BPhysHypoHelper&, xAOD::BPhysHelper::pv_type, std::vector<double> trackMasses) const; + ToolHandle<Trk::V0Tools> m_v0Tools; + ToolHandle<DerivationFramework::FourMuonTool> m_fourMuonTool; + ToolHandle<Analysis::PrimaryVertexRefitter> m_pvRefitter; + + /** job options + */ + std::string m_pairName; + std::string m_quadName; + std::string m_pvContainerName; + std::string m_refPVContainerName; + bool m_refitPV; + int m_PV_max; + int m_DoVertexType; + }; +} + +#endif // DERIVATIONFRAMEWORK_Reco_4mu_H diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/Reco_V0Finder.h b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/Reco_V0Finder.h new file mode 100644 index 00000000000..ceb666c2393 --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/Reco_V0Finder.h @@ -0,0 +1,63 @@ +/* + Copyright (C) 2002-2017 CERN for the benefit of the ATLAS collaboration +*/ +/////////////////////////////////////////////////////////////////// +// Reco_V0Finder.h, (c) ATLAS Detector software +/////////////////////////////////////////////////////////////////// + +#ifndef DERIVATIONFRAMEWORK_V0FINDER_H +#define DERIVATIONFRAMEWORK_V0FINDER_H + +#include <string> + +#include "AthenaBaseComps/AthAlgTool.h" +#include "GaudiKernel/ToolHandle.h" +#include "DerivationFrameworkInterfaces/IAugmentationTool.h" +#include "InDetV0Finder/InDetV0FinderTool.h" +#include "HepPDT/ParticleDataTable.hh" + +/** forward declarations + */ +namespace Trk +{ + class V0Tools; + class ParticleDataTable; +} + +namespace DerivationFramework { + + class Reco_V0Finder : public AthAlgTool, public IAugmentationTool { + public: + Reco_V0Finder(const std::string& t, const std::string& n, const IInterface* p); + + StatusCode initialize() override; + StatusCode finalize() override; + + virtual StatusCode addBranches() const override; + + private: + + std::vector<std::string> m_CollectionsToCheck; + ToolHandle <InDet::InDetV0FinderTool> m_v0FinderTool; + ToolHandle <Trk::V0Tools> m_V0Tools; + const HepPDT::ParticleDataTable *m_particleDataTable; + + int m_masses; //!< = 1 if using PDG values, = 2 if user set (1) + double m_masspi; //!< pion mass (139.57 MeV) + double m_massp; //!< proton mass (938.272 MeV) + double m_masse; //!< electron mass (0.510999 MeV) + double m_massK0S; //!< Kshort mass (497.672 MeV) + double m_massLambda; //!< Lambda mass (1115.68 MeV) + + std::string m_VxPrimaryCandidateName; //!< Name of primary vertex container + + std::string m_v0ContainerName; + std::string m_ksContainerName; + std::string m_laContainerName; + std::string m_lbContainerName; + + }; +} + +#endif // DERIVATIONFRAMEWORK_Reco_dimuTrk_H + diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/Reco_Vertex.h b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/Reco_Vertex.h new file mode 100644 index 00000000000..57e82e2048e --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/Reco_Vertex.h @@ -0,0 +1,54 @@ +/* + Copyright (C) 2002-2019 CERN for the benefit of the ATLAS collaboration +*/ + +/////////////////////////////////////////////////////////////////// +// Reco_Vertex.h +/////////////////////////////////////////////////////////////////// + +#ifndef DERIVATIONFRAMEWORK_Reco_Vertex_H +#define DERIVATIONFRAMEWORK_Reco_Vertex_H + +#include "AthenaBaseComps/AthAlgTool.h" +#include "GaudiKernel/ToolHandle.h" +#include "TrkVertexAnalysisUtils/V0Tools.h" +#include "DerivationFrameworkInterfaces/IAugmentationTool.h" +#include "JpsiUpsilonTools/ICandidateSearch.h" +#include "JpsiUpsilonTools/PrimaryVertexRefitter.h" +#include "BeamSpotConditionsData/BeamSpotData.h" +#include "StoreGate/ReadHandleKeyArray.h" + +namespace DerivationFramework { + + class Reco_Vertex : public AthAlgTool, public IAugmentationTool { + public: + Reco_Vertex(const std::string& t, const std::string& n, const IInterface* p); + + virtual StatusCode initialize(); + + virtual StatusCode addBranches() const; + + private: + /** tools + */ + ToolHandle<Trk::V0Tools> m_v0Tools; + ToolHandle<Analysis::ICandidateSearch> m_SearchTool; + ToolHandle<Analysis::PrimaryVertexRefitter> m_pvRefitter; + SG::ReadCondHandleKey<InDet::BeamSpotData> m_beamSpotKey { this, "BeamSpotKey", "BeamSpotData", "SG key for beam spot" }; + + /** job options + */ + SG::WriteHandleKey<xAOD::VertexContainer> m_outputVtxContainerName; + SG::ReadHandleKey<xAOD::VertexContainer> m_pvContainerName; + SG::WriteHandleKey<xAOD::VertexContainer> m_refPVContainerName; + bool m_refitPV; + int m_PV_max; + int m_DoVertexType; + size_t m_PV_minNTracks; + bool m_do3d; + bool m_checkCollections; + SG::ReadHandleKeyArray<xAOD::VertexContainer> m_CollectionsToCheck; + }; +} + +#endif // DERIVATIONFRAMEWORK_Reco_Vertex_H diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/Select_Bmumu.h b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/Select_Bmumu.h new file mode 100644 index 00000000000..cf43240a3ac --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/Select_Bmumu.h @@ -0,0 +1,116 @@ +/* + Copyright (C) 2002-2018 CERN for the benefit of the ATLAS collaboration +*/ + +//============================================================================ +// Select_Bmumu.h +//============================================================================ +// +// Author : Wolfgang Walkowiak <Wolfgang.Walkowiak@cern.ch.> +// Changes: +// +// Based on Select_onia2mumu.h. +// Original author: Daniel Scheirich <daniel.scheirich@cern.ch> +// +// Select B candidates for the B(s)mumu analysis including for +// the reference channels used. +// +// For an example see BPHY8.py . +// +// Job options provided by this class: +// +//============================================================================ +// +#ifndef DERIVATIONFRAMEWORK_Select_Bmumu_H +#define DERIVATIONFRAMEWORK_Select_Bmumu_H + +#include <string> + +#include "GaudiKernel/ToolHandle.h" +#include "AthenaBaseComps/AthAlgTool.h" +#include "DerivationFrameworkBPhys/CfAthAlgTool.h" +#include "DerivationFrameworkInterfaces/IAugmentationTool.h" +#include "MuonAnalysisInterfaces/IMuonSelectionTool.h" +#include "JpsiUpsilonTools/JpsiFinder.h" +#include "xAODBPhys/BPhysHelper.h" + +/** forward declarations + */ +namespace Trk { + class V0Tools; +} + +namespace xAOD { + class BPhysHypoHelper; +} + +namespace SG { + class AuxElement; +} + +/** THE candidate selection tool + */ +namespace DerivationFramework { + + class Select_Bmumu : public CfAthAlgTool, public IAugmentationTool { + public: + Select_Bmumu(const std::string& t, const std::string& n, const IInterface* p); + + /** initialization and finalization + */ + StatusCode initialize() override; + StatusCode finalize() override; + + /** @brief: augmentation and selection + * Retrieved vertices are augmented with usual information. + * Selection is performed and each candidate is decorated with the + * Char_t flag named "passed_"+name() to indicate whether if the candidate + * passed the selection. This flag is then used by the event selection tool + * and by the vertex thinning tool. + */ + virtual StatusCode addBranches() const override; + + private: + void ProcessVertex(xAOD::BPhysHypoHelper&, xAOD::BPhysHelper::pv_type) const; + bool massCuts(float mass) const; + bool massInBlindedRegion(float mass) const; + + bool checkAllMuonsTight(const std::vector<const xAOD::Muon*>& muons, + int maxMuonsToCheck=-1) const; + + bool pass(const SG::AuxElement& em, std::string hypo) const; + bool setPass(const SG::AuxElement& em, std::string hypo, bool passVal) const; + bool setPassIfNotAvailable(SG::AuxElement& em, std::string hypo, + bool passVal) const; + // std::vector<xAOD::Vertex*> getPrecedingVertices(const xAOD::Vertex* vtx); + + /** tools + */ + ToolHandle<Trk::V0Tools> m_v0Tools; + ToolHandle<CP::IMuonSelectionTool> m_muSelectionTool; + + /** job options + */ + std::string m_hypoName; //!< name of the mass hypothesis. E.g. Jpis, Upsi, etc. Will be used as a prefix for decorations + std::string m_inputVtxContainerName; //!< name of the input container name + std::vector<double> m_trkMasses; //!< track mass hypotheses + double m_massHypo; //!< vertex mass hypothesis + double m_massMax; //!< invariant mass range + double m_massMin; //!< invariant mass range + double m_chi2Max; //!< max chi2 cut + int m_DoVertexType; //!< Allows user to skip certain vertexes - bitwise test 7==all(111) + bool m_do3d; //!< add 3d proper time + double m_blindMassMin; //!< blinding mass range + double m_blindMassMax; //!< blinding mass range + bool m_doBlinding; //!< enable blinding range + bool m_doCutBlinded; //!< enable cutting blinded vertices + bool m_blindOnlyAllMuonsTight; //!< only blind candidates with all tight muons + bool m_useMuCalcMass; //!< also check against MUCALC mass + + std::vector<std::string> m_subDecVtxContNames; //!< names of sub-decay vertex containers + std::vector<std::string> m_subDecVtxHypoCondNames; //!< hypo names for sub-decays to be considered + std::vector<std::string> m_subDecVtxHypoFlagNames; //!< names of hypo flags set on sub-decays if passing + }; +} + +#endif // DERIVATIONFRAMEWORK_Select_Bmumu_H diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/Select_onia2mumu.h b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/Select_onia2mumu.h new file mode 100644 index 00000000000..f2157f8b24e --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/Select_onia2mumu.h @@ -0,0 +1,71 @@ +/* + Copyright (C) 2002-2017 CERN for the benefit of the ATLAS collaboration +*/ + +/////////////////////////////////////////////////////////////////// +// Select_onia2mumu.h +/////////////////////////////////////////////////////////////////// + +#ifndef DERIVATIONFRAMEWORK_Select_onia2mumu_H +#define DERIVATIONFRAMEWORK_Select_onia2mumu_H + + +#include "AthenaBaseComps/AthAlgTool.h" +#include "DerivationFrameworkInterfaces/IAugmentationTool.h" +#include "JpsiUpsilonTools/JpsiFinder.h" +#include "xAODBPhys/BPhysHelper.h" +#include <string> + +/** forward declarations + */ +namespace Trk { + class V0Tools; +} + +namespace xAOD { + class BPhysHypoHelper; +} + +/** THE candidate selection tool + */ +namespace DerivationFramework { + + class Select_onia2mumu : public AthAlgTool, public IAugmentationTool { + public: + Select_onia2mumu(const std::string& t, const std::string& n, const IInterface* p); + + /** inirialization and finalization + */ + StatusCode initialize() override; + + /** @brief: augmentation and selection + * Retrieved vertices are augmented with usual information. + * Selection is performed and each candidate is decorated with the + * Char_t flag named "passed_"+name() to indicate whether if the candidate + * passed the selection. This flag is then used by the event selection tool + * and by the vertex thinning tool. + */ + virtual StatusCode addBranches() const override; + + private: + void ProcessVertex(xAOD::BPhysHypoHelper&, xAOD::BPhysHelper::pv_type) const; + /** tools + */ + ToolHandle<Trk::V0Tools> m_v0Tools; + + /** job options + */ + std::string m_hypoName; //!< name of the mass hypothesis. E.g. Jpis, Upsi, etc. Will be used as a prefix for decorations + SG::ReadHandleKey<xAOD::VertexContainer> m_inputVtxContainerName; //!< name of the input container name + std::vector<double> m_trkMasses; //!< track mass hypotheses + double m_massHypo; //!< vertex mass hypothesis + double m_massMax; //!< invariant mass range + double m_massMin; //!< invariant mass range + double m_chi2Max; //!< max chi2 cut + double m_lxyMin; //!< min lxy cut + int m_DoVertexType; //!< Allows user to skip certain vertexes - bitwise test 7==all(111) + bool m_do3d; + }; +} + +#endif // DERIVATIONFRAMEWORK_Select_onia2mumu_H diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/Thin_vtxDuplicates.h b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/Thin_vtxDuplicates.h new file mode 100644 index 00000000000..77aa7c81386 --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/Thin_vtxDuplicates.h @@ -0,0 +1,37 @@ +/* + Copyright (C) 2002-2017 CERN for the benefit of the ATLAS collaboration +*/ + +#ifndef DERIVATIONFRAMEWORK_Thin_vtxDuplicates_H +#define DERIVATIONFRAMEWORK_Thin_vtxDuplicates_H + +#include "xAODTracking/VertexContainer.h" +#include "AthenaBaseComps/AthAlgTool.h" +#include "DerivationFrameworkInterfaces/IThinningTool.h" +#include "GaudiKernel/ToolHandle.h" +#include "StoreGate/ThinningHandleKey.h" +#include "StoreGate/ReadDecorHandleKeyArray.h" +class IThinningSvc; + +namespace DerivationFramework { + + class Thin_vtxDuplicates : public AthAlgTool, public IThinningTool { + public: + Thin_vtxDuplicates(const std::string& t, const std::string& n, const IInterface* p); + ~Thin_vtxDuplicates(); + virtual StatusCode initialize(); + virtual StatusCode finalize(); + virtual StatusCode doThinning() const; + + private: + bool m_noFlags; + StringProperty m_streamName{ this, "StreamName", "", "Name of the stream being thinned" }; + mutable std::atomic<unsigned int> m_nVtxTot, m_nVtxPass; + + SG::ThinningHandleKey< xAOD::VertexContainer > m_vertexContainerNames; + SG::ReadDecorHandleKeyArray<xAOD::VertexContainer> m_passFlags; + bool m_and; + }; +} + +#endif diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/Thin_vtxTrk.h b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/Thin_vtxTrk.h new file mode 100644 index 00000000000..3222f0ffcfd --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/Thin_vtxTrk.h @@ -0,0 +1,54 @@ +/* + Copyright (C) 2002-2017 CERN for the benefit of the ATLAS collaboration +*/ + +/////////////////////////////////////////////////////////////////// +// Thin_vtxTrk.h +/////////////////////////////////////////////////////////////////// + +#ifndef DERIVATIONFRAMEWORK_Thin_vtxTrk_H +#define DERIVATIONFRAMEWORK_Thin_vtxTrk_H + +#include "xAODTracking/TrackParticleContainer.h" +#include "xAODTracking/VertexContainer.h" +#include "AthenaBaseComps/AthAlgTool.h" +#include "DerivationFrameworkInterfaces/IThinningTool.h" +#include "GaudiKernel/ToolHandle.h" +#include "StoreGate/ThinningHandleKey.h" +#include "StoreGate/HandleKeyArray.h" +#include "StoreGate/ReadDecorHandleKeyArray.h" +#include <string> + +namespace SG{ + template <class T> + using ThinningHandleKeyArray = HandleKeyArray<ReadHandle<T>, ThinningHandleKey<T>, Gaudi::DataHandle::Reader >; +} + +namespace DerivationFramework { + + class Thin_vtxTrk : public AthAlgTool, public IThinningTool { + public: + Thin_vtxTrk(const std::string& t, const std::string& n, const IInterface* p); + ~Thin_vtxTrk(); + StatusCode initialize(); + StatusCode finalize(); + virtual StatusCode doThinning() const; + + private: + StringProperty m_streamName{ this, "StreamName", "", "Name of the stream being thinned" }; + mutable std::atomic<unsigned int> m_ntot, m_npass; + double m_acceptanceR; + mutable std::atomic<unsigned int> m_nVtxTot, m_nVtxPass; + + SG::ThinningHandleKey<xAOD::TrackParticleContainer> m_trackParticleContainerName; + SG::ThinningHandleKeyArray<xAOD::VertexContainer> m_vertexContainerName; + std::vector<std::string> m_passFlags; + SG::ReadDecorHandleKeyArray<xAOD::VertexContainer> m_passArray{this, "INTERNALARRAY", {}}; + bool m_and; + bool m_trackAnd; + bool m_thinTracks; + bool m_noFlags; //To take all entries, regardless of flags + }; +} + +#endif diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/TriggerCountToMetadata.h b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/TriggerCountToMetadata.h new file mode 100644 index 00000000000..69290703ebd --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/TriggerCountToMetadata.h @@ -0,0 +1,51 @@ +/* +Copyright (C) 2002-2017 CERN for the benefit of the ATLAS collaboration +*/ +//============================================================================ +// +// Author : Matteo Bedognetti <matteo.bedognetti@cern.ch.> +// Changes: +// +// Store trigger counts for specific chains in the DAOD's MetaData. +// This allows it to store information about triggers upon which events are NOT selected during the derivation +// +// Job options: +// - TriggerList -- a vector containing all triggers to store as strings +// - TrigDecisionTool -- if one wants to pass this a specific TrigDecisionTool +// +//============================================================================ +// +#ifndef DERIVATIONFRAMEWORK_TriggerCountToMetadata_H +#define DERIVATIONFRAMEWORK_TriggerCountToMetadata_H + +#include "DerivationFrameworkInterfaces/IAugmentationTool.h" +#include "GaudiKernel/ToolHandle.h" +#include "DerivationFrameworkBPhys/CfAthAlgTool.h" +#include "TrigDecisionTool/TrigDecisionTool.h" + +#include <string> +#include <vector> + +namespace Trig{ + class TrigDecisionTool; +} + +namespace DerivationFramework { + + class TriggerCountToMetadata : virtual public CfAthAlgTool, virtual public IAugmentationTool { + + public: + TriggerCountToMetadata(const std::string& t, const std::string& n, const IInterface* p); + virtual StatusCode initialize() override; + virtual StatusCode addBranches() const override; + + private: //Don't use protected for this one! + + std::vector<std::string> m_triggerList; + ToolHandle<Trig::TrigDecisionTool> m_trigDecisionTool; + std::string m_folderName; + + }; // class +} // namespace + +#endif // DERIVATIONFRAMEWORK_TriggerCountToMetadata_H diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/VertexCaloIsolation.h b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/VertexCaloIsolation.h new file mode 100644 index 00000000000..50e7b4d252d --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/VertexCaloIsolation.h @@ -0,0 +1,89 @@ +/* + Copyright (C) 2002-2017 CERN for the benefit of the ATLAS collaboration +*/ + +/////////////////////////////////////////////////////////////////// +// VertexCaloIsolation.h by Matteo Bedognetti +/////////////////////////////////////////////////////////////////// +// +// This code is based on CaloIsolationTool of IsolationTools package +// +// Etcone is determined as a topoCluster-isolation value minus Energy Density (ED) correction and minus the energy depositions of the muons +// Muon's energy deposition is already stored in side the xAOD::Muon objects, but the muon-clusters are used to correct for the fact that they muons may have overlapping clusters +// The muon-clusters are stored as well in connection with the muons themselves +// +// The idea of comparing topoClusters with muon-clusters to decide what part of the muon's deposition is of +// importance had to be abandoned because topCluster cells are not present in xAOD +// +// It enforces the fact that for muons no core-surface is removed for the energy-density correction (thus the corrections are independent from each other) +// +// "isReliable" flag reports of each isolation value if all particles crossing the cone have been correctly corrected for. +// In the case of 2mu+ 1 track it mirrors the fact that the track does not extrapolate into the cone (as tracks have no muon-cluster from which to determine the core-correction) +// +#ifndef DERIVATIONFRAMEWORK_VertexCaloIsolation_H +#define DERIVATIONFRAMEWORK_VertexCaloIsolation_H + +#include <string> + +#include "AthenaBaseComps/AthAlgTool.h" +#include "DerivationFrameworkInterfaces/IAugmentationTool.h" +#include "JpsiUpsilonTools/PrimaryVertexRefitter.h" +//#include "MuonIdHelpers/MuonIdHelperTool.h" +//#include "IsolationTool/CaloIsolationTool.h" +//#include "TrkCaloExtension/CaloExtension.h" + +#include "RecoToolInterfaces/IParticleCaloExtensionTool.h" +#include "RecoToolInterfaces/ICaloTopoClusterIsolationTool.h" + +//#include "TrackToCalo/CaloCellCollector.h" +//#include "CaloInterface/ICaloNoiseTool.h" +#include "xAODBPhys/BPhysHelper.h" +//#include "xAODPrimitives/IsolationType.h" // + + +/** THE reconstruction tool + */ +namespace DerivationFramework { + + class VertexCaloIsolation : public AthAlgTool, public IAugmentationTool { + public: + VertexCaloIsolation(const std::string& t, const std::string& n, const IInterface* p); + + StatusCode initialize(); + StatusCode finalize(); + + StatusCode addBranches() const; + bool extrapolateTrack(TLorentzVector& extr_tp, const xAOD::IParticle& tp) const; + bool extrapolateMuon(TLorentzVector& extr_tp, const xAOD::CaloCluster* cluster) const; + xAOD::TrackParticle& makeSlyTrack(xAOD::TrackParticle&, const TLorentzVector& candidate, const xAOD::Vertex* vertex, xAOD::BPhysHelper::pv_type vertexType) const; + + + private: + + //ToolHandle<xAOD::ICaloCellIsolationTool> m_caloIsoTool; + ToolHandle<xAOD::ICaloTopoClusterIsolationTool> m_caloIsoTool; + std::string m_trackContainerName; + std::string m_vertexContainerName; + std::string m_caloClusterContainerName; + std::string m_cellContainerName; + std::string m_muonContainerName; + ToolHandle<Trk::IParticleCaloExtensionTool> m_caloExtTool; + std::vector<unsigned int> m_cones; //I cannot use xAOD::Iso::IsolationType as a type here, as it clashes with setProperty() + std::vector<std::string> m_passFlags; + + + // ToolHandle <ICaloNoiseTool> m_caloNoiseTool; //Removed to reduce requirements + //Rec::CaloCellCollector m_cellCollector; //Seems to be a plain class, so no need for handles + + /// Number of sigma for calo cell noise cut + float m_sigmaCaloNoiseCut; + + int m_vertexType; + + + + }; +} + +#endif + diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/VertexPlus1TrackCascade.h b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/VertexPlus1TrackCascade.h new file mode 100644 index 00000000000..416b42c5a07 --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/VertexPlus1TrackCascade.h @@ -0,0 +1,74 @@ +/* + Copyright (C) 2002-2017 CERN for the benefit of the ATLAS collaboration +*/ + +#ifndef VERTEXPLUS1TRACKCASCADE_H +#define VERTEXPLUS1TRACKCASCADE_H +//********************* +// VertexPlus1Cascade header file +// +// Adam Barton <abarton@cern.ch> + +#include "AthenaBaseComps/AthAlgTool.h" +#include "GaudiKernel/ToolHandle.h" +#include "xAODMuon/MuonContainer.h" +#include "xAODTracking/TrackParticle.h" +#include "xAODTracking/TrackParticleContainer.h" +#include "xAODTracking/VertexContainer.h" +#include <vector> + + +namespace Trk { + class IVertexFitter; + class ITrackSelectorTool; + class TrkVKalVrtFitter; + class IVertexCascadeFitter; + class VxCascadeInfo; +} + +namespace DerivationFramework { + + static const InterfaceID IID_VertexPlus1TrackCascade("VertexPlus1TrackCascade", 1, 0); + + class VertexPlus1TrackCascade : virtual public AthAlgTool + { + + + SG::ReadHandleKey<xAOD::VertexContainer> m_vertexContainerKey; + SG::ReadHandleKey<xAOD::TrackParticleContainer> m_TrackPContainerKey; + SG::ReadHandleKey<xAOD::MuonContainer> m_MuonsUsedInJpsiKey; + + std::vector<double> m_massHypothesis; + + std::vector<int> m_massConstraintTracksVtx1; + std::vector<int> m_massConstraintTracksVtx2; + double m_Vtx1MassConstraint; + double m_Vtx2MassConstraint; + + double m_trkThresholdPt; + double m_trkMaxEta; +// double m_BThresholdPt; +// double m_BMassUpper; +// double m_BMassLower; + + double m_roughMassLower; + double m_roughMassUpper; + ToolHandle < Trk::TrkVKalVrtFitter > m_iVertexFitter; + ToolHandle < Trk::ITrackSelectorTool > m_trkSelector; + + public: + static const InterfaceID& interfaceID() { return IID_VertexPlus1TrackCascade;} + VertexPlus1TrackCascade(const std::string& t, const std::string& n, const IInterface* p); + ~VertexPlus1TrackCascade(); + StatusCode initialize() override; + StatusCode finalize() override; + static double getInvariantMass(const std::vector<const xAOD::TrackParticle*> &Tracks, const std::vector<double> &massHypotheses); + static bool isContainedIn(const xAOD::TrackParticle* theTrack, const xAOD::MuonContainer* theColl); + StatusCode performSearch(std::vector<Trk::VxCascadeInfo*> *cascadeinfoContainer ) const; + + }; +} + + +#endif + diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/VertexTrackIsolation.h b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/VertexTrackIsolation.h new file mode 100644 index 00000000000..603da6cef65 --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/DerivationFrameworkBPhys/VertexTrackIsolation.h @@ -0,0 +1,52 @@ +/* + Copyright (C) 2002-2020 CERN for the benefit of the ATLAS collaboration +*/ + +/////////////////////////////////////////////////////////////////// +// VertexTrackIsolation.h, +/////////////////////////////////////////////////////////////////// + +#ifndef DERIVATIONFRAMEWORK_VertexTrackIsolation_H +#define DERIVATIONFRAMEWORK_VertexTrackIsolation_H + +#include <string> + +#include "AthenaBaseComps/AthAlgTool.h" +#include "DerivationFrameworkInterfaces/IAugmentationTool.h" +#include "JpsiUpsilonTools/PrimaryVertexRefitter.h" +//#include "MuonIdHelpers/MuonIdHelperTool.h" +//#include "IsolationTool/TrackIsolationTool.h" +#include "RecoToolInterfaces/ITrackIsolationTool.h" +//#include "xAODPrimitives/IsolationType.h" +#include <vector> +/** THE reconstruction tool + */ +namespace DerivationFramework { + + class VertexTrackIsolation : public AthAlgTool, public IAugmentationTool { + public: + VertexTrackIsolation(const std::string& t, const std::string& n, const IInterface* p); + + StatusCode initialize(); + StatusCode finalize(); + + virtual StatusCode addBranches() const; + + bool isSame(const xAOD::Vertex* theVtx1, const xAOD::Vertex* theVtx2) const; + bool isContainedIn(const xAOD::Vertex* theVtx, const std::vector<const xAOD::Vertex*> &theColl) const; + + private: + + ToolHandle<xAOD::ITrackIsolationTool> m_trackIsoTool; + std::string m_trackContainerName; + std::string m_vertexContainerName; + std::vector<unsigned int> m_cones; + std::vector<std::string> m_passFlags; + int m_vertexType; //Which type of primary vertices should be used? (7 = 0b111 are all at the moment) + + bool m_doIsoPerTrk; + int m_removeDuplicate; + }; +} + +#endif diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/python/BPhysPyHelpers.py b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/python/BPhysPyHelpers.py new file mode 100644 index 00000000000..c30320e4e2f --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/python/BPhysPyHelpers.py @@ -0,0 +1,84 @@ +# Copyright (C) 2002-2020 CERN for the benefit of the ATLAS collaboration + +#==================================================================== +# BPhysPyHelpers.py +# +# Python helper classes for BPHY derivations. +# +# authors: W. Walkowiak <wolfgang.walkowiak@cern.ch>, 2016-05-19 +# changed: +# +# This file contains a set of Python helper classes for the +# BPHY derivations. +# +# Available helper classes and methods: +# - BPyWrapper -- wrap configurable to ensure default contents of +# __slots__ dict are available as attributes +# Usage example: see BPHY8.py +# Note: Unfortunately, this doesn't quite work, since the +# wrapped class is no longer recognized as Configurable +# - BPhysEnsureAttributes(algtool) +# -- ensure default contents of __slots__ dict a +# are available as attributes to the class +# - BPhysFilterBranches(...) +# -- create list of isolation or closest track branches +# to be thinned +# +#==================================================================== + +#-------------------------------------------------------------------- +class BPyWrapper(object): + __slots__ = {'wclass' : object} + + def __init__(self, wclass, *args, **kargs): + object.__setattr__(self, 'wclass', wclass(*args, **kargs)) + # the important part: make __slot__ variables attributes + for n,v in self.wclass.__slots__.items(): + if not hasattr(self.wclass, n): + setattr(self.wclass, n, v) + + def __getattr__(self, attr): + return self.wclass.__getattribute__(attr) + + def __setattr__(self, attr, value): + setattr(self.wclass, attr, value) + + def __call__(self, *args, **kwargs): + return self.wclass(*args, **kwargs) +#-------------------------------------------------------------------- +# +# ensure default contents of __slots__ dict are available as attributes +# +def BPhysEnsureAttributes(algtool): + + for n,v in algtool.__slots__.items(): + if not hasattr(algtool, n): + setattr(algtool, n, v) + return algtool +#-------------------------------------------------------------------- +# +# create list of isolation or closest track branches to be thinned +# (used by BPHY8) +# +def BPhysFilterBranches(name, brPrefixList, brIncludeList, doVertexTypeList, + categoryList, trackTypeList, coneOrChi2SetList, + forCloseTrack=False): + res = "" + brIncludes = [tuple(x.split('|',3)) for x in brIncludeList] + for bntup in brPrefixList: + bn, sep, bnsuf = bntup.partition('+') + for i, cstr in enumerate(coneOrChi2SetList): + for itt in trackTypeList: + ittstr = "T%010d" % itt + for itcstr in categoryList: + if brIncludes and not (cstr,str(itt),itcstr) in brIncludes: + for dvs in doVertexTypeList: + if forCloseTrack: + fbn = '_'.join(filter(None, [name,bn,ittstr,itcstr, + dvs,cstr,bnsuf])) + else: + fbn = '_'.join(filter(None, [name,bn,cstr,ittstr, + itcstr,dvs,bnsuf])) + res += ".-"+fbn + return res +#-------------------------------------------------------------------- diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/python/__init__.py b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/python/__init__.py new file mode 100644 index 00000000000..74583d364ec --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/python/__init__.py @@ -0,0 +1,2 @@ +# Copyright (C) 2002-2017 CERN for the benefit of the ATLAS collaboration + diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/BPHY1.py b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/BPHY1.py new file mode 100644 index 00000000000..6b20ae9dacb --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/BPHY1.py @@ -0,0 +1,274 @@ +#==================================================================== +# BPHY1.py +# This an example job options script showing how to set up a +# derivation of the data using the derivation framework. +# It requires the reductionConf flag BPHY1 in Reco_tf.py +#==================================================================== + +# Set up common services and job object. +# This should appear in ALL derivation job options +from DerivationFrameworkCore.DerivationFrameworkMaster import * + +isSimulation = False +if globalflags.DataSource()=='geant4': + isSimulation = True + +print (isSimulation) + +#==================================================================== +# AUGMENTATION TOOLS +#==================================================================== +## 1/ setup vertexing tools and services +#include( "JpsiUpsilonTools/configureServices.py" ) + +include("DerivationFrameworkBPhys/configureVertexing.py") +BPHY1_VertexTools = BPHYVertexTools("BPHY1") + +#-------------------------------------------------------------------- +## 2/ Setup the vertex fitter tools (e.g. JpsiFinder, JpsiPlus1Track, etc). +## These are general tools independent of DerivationFramework that do the +## actual vertex fitting and some pre-selection. +from JpsiUpsilonTools.JpsiUpsilonToolsConf import Analysis__JpsiFinder +BPHY1JpsiFinder = Analysis__JpsiFinder( + name = "BPHY1JpsiFinder", + OutputLevel = INFO, + muAndMu = True, + muAndTrack = False, + TrackAndTrack = False, + assumeDiMuons = True, # If true, will assume dimu hypothesis and use PDG value for mu mass + invMassUpper = 100000.0, + invMassLower = 0.0, + Chi2Cut = 200., + oppChargesOnly = True, + atLeastOneComb = True, + useCombinedMeasurement = False, # Only takes effect if combOnly=True + muonCollectionKey = "Muons", + TrackParticleCollection = "InDetTrackParticles", + V0VertexFitterTool = BPHY1_VertexTools.TrkV0Fitter, # V0 vertex fitter + useV0Fitter = False, # if False a TrkVertexFitterTool will be used + TrkVertexFitterTool = BPHY1_VertexTools.TrkVKalVrtFitter, # VKalVrt vertex fitter + TrackSelectorTool = BPHY1_VertexTools.InDetTrackSelectorTool, + VertexPointEstimator = BPHY1_VertexTools.VtxPointEstimator, + useMCPCuts = False ) + +ToolSvc += BPHY1JpsiFinder +print (BPHY1JpsiFinder) + +#-------------------------------------------------------------------- +## 3/ setup the vertex reconstruction "call" tool(s). They are part of the derivation framework. +## These Augmentation tools add output vertex collection(s) into the StoreGate and add basic +## decorations which do not depend on the vertex mass hypothesis (e.g. lxy, ptError, etc). +## There should be one tool per topology, i.e. Jpsi and Psi(2S) do not need two instance of the +## Reco tool is the JpsiFinder mass window is wide enough. + +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__Reco_Vertex +BPHY1_Reco_mumu = DerivationFramework__Reco_Vertex( + name = "BPHY1_Reco_mumu", + VertexSearchTool = BPHY1JpsiFinder, + OutputVtxContainerName = "BPHY1OniaCandidates", + PVContainerName = "PrimaryVertices", + RefPVContainerName = "BPHY1RefittedPrimaryVertices", + RefitPV = True, + MaxPVrefit = 100000, + DoVertexType = 7) + +ToolSvc += BPHY1_Reco_mumu +print (BPHY1_Reco_mumu) + +#-------------------------------------------------------------------- +## 4/ setup the vertex selection and augmentation tool(s). These tools decorate the vertices with +## variables that depend on the vertex mass hypothesis, e.g. invariant mass, proper decay time, etc. +## Property HypothesisName is used as a prefix for these decorations. +## They also perform tighter selection, flagging the vertecis that passed. The flag is a Char_t branch +## named "passed_"+HypothesisName. It is used later by the "SelectEvent" and "Thin_vtxTrk" tools +## to determine which events and candidates should be kept in the output stream. +## Multiple instances of the Select_* tools can be used on a single input collection as long as they +## use different "HypothesisName" flags. + +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__Select_onia2mumu + +## a/ augment and select Jpsi->mumu candidates +BPHY1_Select_Jpsi2mumu = DerivationFramework__Select_onia2mumu( + name = "BPHY1_Select_Jpsi2mumu", + HypothesisName = "Jpsi", + InputVtxContainerName = "BPHY1OniaCandidates", + VtxMassHypo = 3096.916, + MassMin = 2000.0, + MassMax = 3600.0, + Chi2Max = 200, + DoVertexType = 7) + +ToolSvc += BPHY1_Select_Jpsi2mumu +print (BPHY1_Select_Jpsi2mumu) + +## b/ augment and select Psi(2S)->mumu candidates +BPHY1_Select_Psi2mumu = DerivationFramework__Select_onia2mumu( + name = "BPHY1_Select_Psi2mumu", + HypothesisName = "Psi", + InputVtxContainerName = "BPHY1OniaCandidates", + VtxMassHypo = 3686.09, + MassMin = 3300.0, + MassMax = 4500.0, + Chi2Max = 200, + DoVertexType = 7) + +ToolSvc += BPHY1_Select_Psi2mumu +print (BPHY1_Select_Psi2mumu) + +# Added by ASC +## c/ augment and select Upsilon(nS)->mumu candidates +BPHY1_Select_Upsi2mumu = DerivationFramework__Select_onia2mumu( + name = "BPHY1_Select_Upsi2mumu", + HypothesisName = "Upsi", + InputVtxContainerName = "BPHY1OniaCandidates", + VtxMassHypo = 9460.30, + MassMin = 7000.0, + MassMax = 12500.0, + Chi2Max = 200, + DoVertexType = 7) + +ToolSvc += BPHY1_Select_Upsi2mumu +print (BPHY1_Select_Upsi2mumu) + + + +#==================================================================== +# SET UP STREAM +#==================================================================== +streamName = derivationFlags.WriteDAOD_BPHY1Stream.StreamName +fileName = buildFileName( derivationFlags.WriteDAOD_BPHY1Stream ) +BPHY1Stream = MSMgr.NewPoolRootStream( streamName, fileName ) +BPHY1Stream.AcceptAlgs(["BPHY1Kernel"]) +# Special lines for thinning +# Thinning service name must match the one passed to the thinning tools +augStream = MSMgr.GetStream( streamName ) + + + +#-------------------------------------------------------------------- +## 5/ select the event. We only want to keep events that contain certain vertices which passed certain selection. +## This is specified by the "SelectionExpression" property, which contains the expression in the following format: +## +## "ContainerName.passed_HypoName > count" +## +## where "ContainerName" is output container form some Reco_* tool, "HypoName" is the hypothesis name setup in some "Select_*" +## tool and "count" is the number of candidates passing the selection you want to keep. + +expression = "count(BPHY1OniaCandidates.passed_Jpsi) > 0 || count(BPHY1OniaCandidates.passed_Psi) > 0 || count(BPHY1OniaCandidates.passed_Upsi) > 0" +from DerivationFrameworkTools.DerivationFrameworkToolsConf import DerivationFramework__xAODStringSkimmingTool +BPHY1_SelectEvent = DerivationFramework__xAODStringSkimmingTool(name = "BPHY1_SelectEvent", + expression = expression) +ToolSvc += BPHY1_SelectEvent +print (BPHY1_SelectEvent) + +#-------------------------------------------------------------------- +## 6/ track and vertex thinning. We want to remove all reconstructed secondary vertices +## which hasn't passed any of the selections defined by (Select_*) tools. +## We also want to keep only tracks which are associates with either muons or any of the +## vertices that passed the selection. Multiple thinning tools can perform the +## selection. The final thinning decision is based OR of all the decisions (by default, +## although it can be changed by the JO). + +## a) thining out vertices that didn't pass any selection and idetifying tracks associated with +## selected vertices. The "VertexContainerNames" is a list of the vertex containers, and "PassFlags" +## contains all pass flags for Select_* tools that must be satisfied. The vertex is kept is it +## satisfy any of the listed selections. + +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__Thin_vtxTrk +BPHY1Thin_vtxTrk = DerivationFramework__Thin_vtxTrk( + name = "BPHY1Thin_vtxTrk", + TrackParticleContainerName = "InDetTrackParticles", + StreamName = streamName, + VertexContainerNames = ["BPHY1OniaCandidates"], + PassFlags = ["passed_Jpsi", "passed_Psi", "passed_Upsi"] ) + +ToolSvc += BPHY1Thin_vtxTrk + +## b) thinning out tracks that are not attached to muons. The final thinning decision is based on the OR operation +## between decision from this and the previous tools. +from DerivationFrameworkInDet.DerivationFrameworkInDetConf import DerivationFramework__MuonTrackParticleThinning +BPHY1MuonTPThinningTool = DerivationFramework__MuonTrackParticleThinning(name = "BPHY1MuonTPThinningTool", + MuonKey = "Muons", + StreamName = streamName, + InDetTrackParticlesKey = "InDetTrackParticles") +ToolSvc += BPHY1MuonTPThinningTool + +# Added by ASC +# Only save truth informtion directly associated with Onia +from DerivationFrameworkMCTruth.DerivationFrameworkMCTruthConf import DerivationFramework__GenericTruthThinning +BPHY1TruthThinTool = DerivationFramework__GenericTruthThinning(name = "BPHY1TruthThinTool", + StreamName = streamName, + ParticleSelectionString = "TruthParticles.pdgId == 443 || TruthParticles.pdgId == 100443 || TruthParticles.pdgId == 553 || TruthParticles.pdgId == 100553 || TruthParticles.pdgId == 200553", + PreserveDescendants = True, + PreserveAncestors = True) +ToolSvc += BPHY1TruthThinTool +print (BPHY1TruthThinTool) + + +#==================================================================== +# CREATE THE DERIVATION KERNEL ALGORITHM AND PASS THE ABOVE TOOLS +#==================================================================== +## 7/ IMPORTANT bit. Don't forget to pass the tools to the DerivationKernel! If you don't do that, they will not be +## be executed! + +# Added by ASC +BPHY1ThinningTools = [BPHY1Thin_vtxTrk, BPHY1MuonTPThinningTool] +if globalflags.DataSource()=='geant4': + BPHY1ThinningTools.append(BPHY1TruthThinTool) + +# The name of the kernel (BPHY1Kernel in this case) must be unique to this derivation +from DerivationFrameworkCore.DerivationFrameworkCoreConf import DerivationFramework__DerivationKernel +DerivationFrameworkJob += CfgMgr.DerivationFramework__DerivationKernel( + "BPHY1Kernel", + AugmentationTools = [BPHY1_Reco_mumu, BPHY1_Select_Jpsi2mumu, BPHY1_Select_Psi2mumu, BPHY1_Select_Upsi2mumu], + SkimmingTools = [BPHY1_SelectEvent], + ThinningTools = BPHY1ThinningTools + ) + + + +#==================================================================== +# Slimming +#==================================================================== + +# Added by ASC +from DerivationFrameworkCore.SlimmingHelper import SlimmingHelper +BPHY1SlimmingHelper = SlimmingHelper("BPHY1SlimmingHelper") +AllVariables = [] +StaticContent = [] + +# Needed for trigger objects +BPHY1SlimmingHelper.IncludeMuonTriggerContent = True +BPHY1SlimmingHelper.IncludeBPhysTriggerContent = True + +## primary vertices +AllVariables += ["PrimaryVertices"] +StaticContent += ["xAOD::VertexContainer#BPHY1RefittedPrimaryVertices"] +StaticContent += ["xAOD::VertexAuxContainer#BPHY1RefittedPrimaryVerticesAux."] + +## ID track particles +AllVariables += ["InDetTrackParticles"] + +## combined / extrapolated muon track particles +## (note: for tagged muons there is no extra TrackParticle collection since the ID tracks +## are store in InDetTrackParticles collection) +AllVariables += ["CombinedMuonTrackParticles"] +AllVariables += ["ExtrapolatedMuonTrackParticles"] + +## muon container +AllVariables += ["Muons"] + +## Jpsi candidates +StaticContent += ["xAOD::VertexContainer#%s" % BPHY1_Reco_mumu.OutputVtxContainerName] +StaticContent += ["xAOD::VertexAuxContainer#%sAux." % BPHY1_Reco_mumu.OutputVtxContainerName] +## we have to disable vxTrackAtVertex branch since it is not xAOD compatible +StaticContent += ["xAOD::VertexAuxContainer#%sAux.-vxTrackAtVertex" % BPHY1_Reco_mumu.OutputVtxContainerName] + +# Added by ASC +# Truth information for MC only +if isSimulation: + AllVariables += ["TruthEvents","TruthParticles","TruthVertices","MuonTruthParticles"] + +BPHY1SlimmingHelper.AllVariables = AllVariables +BPHY1SlimmingHelper.StaticContent = StaticContent +BPHY1SlimmingHelper.AppendContentToStream(BPHY1Stream) diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/BPHY10.py b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/BPHY10.py new file mode 100644 index 00000000000..2f9dcb083a9 --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/BPHY10.py @@ -0,0 +1,477 @@ +#==================================================================== +# BPHY10.py +# Bs>J/psiKK +# It requires the reductionConf flag BPHY10 in Reco_tf.py +#==================================================================== + +# Set up common services and job object. +# This should appear in ALL derivation job options +from DerivationFrameworkCore.DerivationFrameworkMaster import * + +isSimulation = False +if globalflags.DataSource()=='geant4': + isSimulation = True + +print(isSimulation) + + +#==================================================================== +# AUGMENTATION TOOLS +#==================================================================== +## 1/ setup vertexing tools and services +#include( "JpsiUpsilonTools/configureServices.py" ) + +include("DerivationFrameworkBPhys/configureVertexing.py") +BPHY10_VertexTools = BPHYVertexTools("BPHY10") + + +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__AugOriginalCounts +BPHY10_AugOriginalCounts = DerivationFramework__AugOriginalCounts( + name = "BPHY10_AugOriginalCounts", + VertexContainer = "PrimaryVertices", + TrackContainer = "InDetTrackParticles" ) +ToolSvc += BPHY10_AugOriginalCounts + + +#-------------------------------------------------------------------- +## 2/ setup JpsiFinder tool +## These are general tools independent of DerivationFramework that do the +## actual vertex fitting and some pre-selection. +from JpsiUpsilonTools.JpsiUpsilonToolsConf import Analysis__JpsiFinder +BPHY10JpsiFinder = Analysis__JpsiFinder( + name = "BPHY10JpsiFinder", + OutputLevel = INFO, + muAndMu = True, + muAndTrack = False, + TrackAndTrack = False, + assumeDiMuons = True, + invMassUpper = 4000.0, + invMassLower = 2600.0, + Chi2Cut = 200., + oppChargesOnly = True, + combOnly = True, + atLeastOneComb = False, + useCombinedMeasurement = False, # Only takes effect if combOnly=True + muonCollectionKey = "Muons", + TrackParticleCollection = "InDetTrackParticles", + V0VertexFitterTool = BPHY10_VertexTools.TrkV0Fitter, # V0 vertex fitter + useV0Fitter = False, # if False a TrkVertexFitterTool will be used + TrkVertexFitterTool = BPHY10_VertexTools.TrkVKalVrtFitter, # VKalVrt vertex fitter + TrackSelectorTool = BPHY10_VertexTools.InDetTrackSelectorTool, + VertexPointEstimator = BPHY10_VertexTools.VtxPointEstimator, + useMCPCuts = False) + +ToolSvc += BPHY10JpsiFinder +print(BPHY10JpsiFinder) + +#-------------------------------------------------------------------- +## 3/ setup the vertex reconstruction "call" tool(s). They are part of the derivation framework. +## These Augmentation tools add output vertex collection(s) into the StoreGate and add basic +## decorations which do not depend on the vertex mass hypothesis (e.g. lxy, ptError, etc). +## There should be one tool per topology, i.e. Jpsi and Psi(2S) do not need two instance of the +## Reco tool is the JpsiFinder mass window is wide enough. +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__Reco_Vertex +BPHY10JpsiSelectAndWrite = DerivationFramework__Reco_Vertex( + name = "BPHY10JpsiSelectAndWrite", + VertexSearchTool = BPHY10JpsiFinder, + OutputVtxContainerName = "BPHY10JpsiCandidates", + PVContainerName = "PrimaryVertices", + RefPVContainerName = "SHOULDNOTBEUSED", + # RefPVContainerName = "BPHY10RefJpsiPrimaryVertices", + # RefitPV = True, + # MaxPVrefit = 10000, + DoVertexType = 1) + +ToolSvc += BPHY10JpsiSelectAndWrite +print(BPHY10JpsiSelectAndWrite) + +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__Select_onia2mumu + +## a/ augment and select Jpsi->mumu candidates +BPHY10_Select_Jpsi2mumu = DerivationFramework__Select_onia2mumu( + name = "BPHY10_Select_Jpsi2mumu", + HypothesisName = "Jpsi", + InputVtxContainerName = "BPHY10JpsiCandidates", + VtxMassHypo = 3096.916, + MassMin = 2600.0, + MassMax = 4000.0, + Chi2Max = 200, + DoVertexType =1) + + +ToolSvc += BPHY10_Select_Jpsi2mumu +print(BPHY10_Select_Jpsi2mumu) + + +## 4/ setup a new vertexing tool (necessary due to use of mass constraint) +from TrkVKalVrtFitter.TrkVKalVrtFitterConf import Trk__TrkVKalVrtFitter +BdKstVertexFit = Trk__TrkVKalVrtFitter( + name = "BdKstVertexFit", + Extrapolator = BPHY10_VertexTools.InDetExtrapolator, + FirstMeasuredPoint = True, + MakeExtendedVertex = True) + +ToolSvc += BdKstVertexFit +print(BdKstVertexFit) + +## 5/ setup the Jpsi+2 track finder +from JpsiUpsilonTools.JpsiUpsilonToolsConf import Analysis__JpsiPlus2Tracks +BPHY10BdJpsiKst = Analysis__JpsiPlus2Tracks( + name = "BPHY10BdJpsiKst", + OutputLevel = INFO, + kaonkaonHypothesis = False, + pionpionHypothesis = False, + kaonpionHypothesis = True, + trkThresholdPt = 500.0, + trkMaxEta = 3.0, + BThresholdPt = 5000., + BMassLower = 4300.0, + BMassUpper = 6300.0, + JpsiContainerKey = "BPHY10JpsiCandidates", + TrackParticleCollection = "InDetTrackParticles", + #MuonsUsedInJpsi = "Muons", #Don't remove all muons, just those in J/psi candidate (see the following cut) + ExcludeCrossJpsiTracks = False, #setting this to False rejects the muons from J/psi candidate + TrkVertexFitterTool = BdKstVertexFit, + TrackSelectorTool = BPHY10_VertexTools.InDetTrackSelectorTool, + VertexPointEstimator = BPHY10_VertexTools.VtxPointEstimator, + UseMassConstraint = True, + #DiTrackMassUpper = 1500., + #DiTrackMassLower = 500., + Chi2Cut = 10.0, + DiTrackPt = 500., + TrkQuadrupletMassLower = 3500.0, + TrkQuadrupletMassUpper = 6800.0, + #FinalDiTrackMassUpper = 1000., + #FinalDiTrackMassLower = 800., + #TrkDeltaZ = 20., #Normally, this cut should not be used since it is lifetime-dependent + FinalDiTrackPt = 500. + ) + +ToolSvc += BPHY10BdJpsiKst +print(BPHY10BdJpsiKst) + + +## 6/ setup the combined augmentation/skimming tool for the BdKst +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__Reco_Vertex +BPHY10BdKstSelectAndWrite = DerivationFramework__Reco_Vertex( + name = "BPHY10BdKstSelectAndWrite", + Jpsi2PlusTrackName = BPHY10BdJpsiKst, + OutputVtxContainerName = "BPHY10BdJpsiKstCandidates", + PVContainerName = "PrimaryVertices", + RefPVContainerName = "BPHY10RefittedPrimaryVertices", + RefitPV = True, + MaxPVrefit = 10000, + DoVertexType = 7) + +ToolSvc += BPHY10BdKstSelectAndWrite +print(BPHY10BdKstSelectAndWrite) + +## b/ augment and select Bd->JpsiKst candidates +# set mass hypothesis (K pi) +BPHY10_Select_Bd2JpsiKst = DerivationFramework__Select_onia2mumu( + name = "BPHY10_Select_Bd2JpsiKst", + HypothesisName = "Bd", + InputVtxContainerName = "BPHY10BdJpsiKstCandidates", + TrkMasses = [105.658, 105.658, 493.677, 139.570], + VtxMassHypo = 5279.6, + MassMin = 100.0, #no mass cuts here + MassMax = 100000.0, #no mass cuts here + Chi2Max = 200) + +ToolSvc += BPHY10_Select_Bd2JpsiKst +print(BPHY10_Select_Bd2JpsiKst) + +## c/ augment and select Bdbar->JpsiKstbar candidates +# set mass hypothesis (pi K) +BPHY10_Select_Bd2JpsiKstbar = DerivationFramework__Select_onia2mumu( + name = "BPHY10_Select_Bd2JpsiKstbar", + HypothesisName = "Bdbar", + InputVtxContainerName = "BPHY10BdJpsiKstCandidates", + TrkMasses = [105.658, 105.658, 139.570, 493.677], + VtxMassHypo = 5279.6, + MassMin = 100.0, #no mass cuts here + MassMax = 100000.0, #no mass cuts here + Chi2Max = 200) + +ToolSvc += BPHY10_Select_Bd2JpsiKstbar +print(BPHY10_Select_Bd2JpsiKstbar) + + +## 7/ call the V0Finder if a Jpsi has been found +doSimpleV0Finder = False +if doSimpleV0Finder: + include("DerivationFrameworkBPhys/configureSimpleV0Finder.py") +else: + include("DerivationFrameworkBPhys/configureV0Finder.py") + +BPHY10_V0FinderTools = BPHYV0FinderTools("BPHY10") +print(BPHY10_V0FinderTools) + +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__Reco_V0Finder +BPHY10_Reco_V0Finder = DerivationFramework__Reco_V0Finder( + name = "BPHY10_Reco_V0Finder", + V0FinderTool = BPHY10_V0FinderTools.V0FinderTool, + #OutputLevel = DEBUG, + V0ContainerName = "BPHY10RecoV0Candidates", + KshortContainerName = "BPHY10RecoKshortCandidates", + LambdaContainerName = "BPHY10RecoLambdaCandidates", + LambdabarContainerName = "BPHY10RecoLambdabarCandidates", + CheckVertexContainers = ['BPHY10JpsiCandidates']) + +ToolSvc += BPHY10_Reco_V0Finder +print(BPHY10_Reco_V0Finder) + +## 8/ setup the cascade vertexing tool +from TrkVKalVrtFitter.TrkVKalVrtFitterConf import Trk__TrkVKalVrtFitter +JpsiV0VertexFit = Trk__TrkVKalVrtFitter( + name = "JpsiV0VertexFit", + #OutputLevel = DEBUG, + Extrapolator = BPHY10_VertexTools.InDetExtrapolator, + #FirstMeasuredPoint = True, + FirstMeasuredPoint = False, + CascadeCnstPrecision = 1e-6, + MakeExtendedVertex = True) + +ToolSvc += JpsiV0VertexFit +print(JpsiV0VertexFit) + +## 9/ setup the Jpsi+V0 finder +## a/ Bd->JpsiKshort +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__JpsiPlusV0Cascade +BPHY10JpsiKshort = DerivationFramework__JpsiPlusV0Cascade( + name = "BPHY10JpsiKshort", + #OutputLevel = DEBUG, + HypothesisName = "Bd", + TrkVertexFitterTool = JpsiV0VertexFit, + V0Hypothesis = 310, + JpsiMassLowerCut = 2800., + JpsiMassUpperCut = 4000., + V0MassLowerCut = 400., + V0MassUpperCut = 600., + MassLowerCut = 4300., + MassUpperCut = 6300., + RefitPV = True, + RefPVContainerName = "BPHY10RefittedPrimaryVertices", + JpsiVertices = "BPHY10JpsiCandidates", + CascadeVertexCollections= ["BPHY10JpsiKshortCascadeSV2", "BPHY10JpsiKshortCascadeSV1"], + V0Vertices = "BPHY10RecoV0Candidates") + +ToolSvc += BPHY10JpsiKshort +print(BPHY10JpsiKshort) + +## b/ Lambda_b->JpsiLambda +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__JpsiPlusV0Cascade +BPHY10JpsiLambda = DerivationFramework__JpsiPlusV0Cascade( + name = "BPHY10JpsiLambda", + #OutputLevel = DEBUG, + HypothesisName = "Lambda_b", + TrkVertexFitterTool = JpsiV0VertexFit, + V0Hypothesis = 3122, + JpsiMassLowerCut = 2800., + JpsiMassUpperCut = 4000., + V0MassLowerCut = 1050., + V0MassUpperCut = 1250., + MassLowerCut = 4600., + MassUpperCut = 6600., + RefitPV = True, + RefPVContainerName = "BPHY10RefittedPrimaryVertices", + JpsiVertices = "BPHY10JpsiCandidates", + CascadeVertexCollections= ["BPHY10JpsiLambdaCascadeSV2", "BPHY10JpsiLambdaCascadeSV1"], + V0Vertices = "BPHY10RecoV0Candidates") + +ToolSvc += BPHY10JpsiLambda +print(BPHY10JpsiLambda) + +## c/ Lambda_bbar->JpsiLambdabar +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__JpsiPlusV0Cascade +BPHY10JpsiLambdabar = DerivationFramework__JpsiPlusV0Cascade( + name = "BPHY10JpsiLambdabar", + HypothesisName = "Lambda_bbar", + #OutputLevel = DEBUG, + TrkVertexFitterTool = JpsiV0VertexFit, + V0Hypothesis = -3122, + JpsiMassLowerCut = 2800., + JpsiMassUpperCut = 4000., + V0MassLowerCut = 1050., + V0MassUpperCut = 1250., + MassLowerCut = 4600., + MassUpperCut = 6600., + RefitPV = True, + RefPVContainerName = "BPHY10RefittedPrimaryVertices", + JpsiVertices = "BPHY10JpsiCandidates", + CascadeVertexCollections= ["BPHY10JpsiLambdabarCascadeSV2", "BPHY10JpsiLambdabarCascadeSV1"], + V0Vertices = "BPHY10RecoV0Candidates") + +ToolSvc += BPHY10JpsiLambdabar +print(BPHY10JpsiLambdabar) + +CascadeCollections = [] +CascadeCollections += BPHY10JpsiKshort.CascadeVertexCollections +CascadeCollections += BPHY10JpsiLambda.CascadeVertexCollections +CascadeCollections += BPHY10JpsiLambdabar.CascadeVertexCollections + + + +if not isSimulation: #Only Skim Data + from DerivationFrameworkTools.DerivationFrameworkToolsConf import DerivationFramework__xAODStringSkimmingTool + BPHY10_SelectBdJpsiKstEvent = DerivationFramework__xAODStringSkimmingTool( + name = "BPHY10_SelectBdJpsiKstEvent", + #expression = "(count(BPHY10BdJpsiKstCandidates.passed_Bd > 0) + count(BPHY10BdJpsiKstCandidates.passed_BdBar > 0) + count(BPHY10RecoV0Candidates) + count(RecoKshortContainerName) + count(RecoLambdaContainerName) + count(RecoLambdabarContainerName) ) > 0") + expression = "(count(BPHY10BdJpsiKstCandidates.passed_Bd > 0) + count(BPHY10BdJpsiKstCandidates.passed_Bdbar > 0) + count(BPHY10JpsiKshortCascadeSV1.x > -999) + count(BPHY10JpsiLambdaCascadeSV1.x > -999) + count(BPHY10JpsiLambdabarCascadeSV1.x > -999) ) > 0") + + ToolSvc += BPHY10_SelectBdJpsiKstEvent + print(BPHY10_SelectBdJpsiKstEvent) + + + + #==================================================================== + # Make event selection based on an OR of the input skimming tools + #==================================================================== + + from DerivationFrameworkTools.DerivationFrameworkToolsConf import DerivationFramework__FilterCombinationOR + BPHY10SkimmingOR = CfgMgr.DerivationFramework__FilterCombinationOR( + "BPHY10SkimmingOR", + FilterList = [BPHY10_SelectBdJpsiKstEvent],) + ToolSvc += BPHY10SkimmingOR + print(BPHY10SkimmingOR) + +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__Thin_vtxTrk +BPHY10_thinningTool_Tracks = DerivationFramework__Thin_vtxTrk( + name = "BPHY10_thinningTool_Tracks", + TrackParticleContainerName = "InDetTrackParticles", + VertexContainerNames = ["BPHY10BdJpsiKstCandidates","BPHY10JpsiKshortCascadeSV1","BPHY10JpsiKshortCascadeSV2","BPHY10JpsiLambdaCascadeSV1","BPHY10JpsiLambdaCascadeSV2","BPHY10JpsiLambdabarCascadeSV1","BPHY10JpsiLambdabarCascadeSV2"], + PassFlags = ["passed_Bd", "passed_Bdbar"] ) + +ToolSvc += BPHY10_thinningTool_Tracks + +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__BPhysPVThinningTool +BPHY10_thinningTool_PV = DerivationFramework__BPhysPVThinningTool( + name = "BPHY10_thinningTool_PV", + CandidateCollections = ["BPHY10BdJpsiKstCandidates","BPHY10JpsiKshortCascadeSV1","BPHY10JpsiKshortCascadeSV2","BPHY10JpsiLambdaCascadeSV1","BPHY10JpsiLambdaCascadeSV2","BPHY10JpsiLambdabarCascadeSV1","BPHY10JpsiLambdabarCascadeSV2"], + KeepPVTracks =True + ) + +ToolSvc += BPHY10_thinningTool_PV + + +## b) thinning out tracks that are not attached to muons. The final thinning decision is based on the OR operation +## between decision from this and the previous tools. +from DerivationFrameworkInDet.DerivationFrameworkInDetConf import DerivationFramework__MuonTrackParticleThinning +BPHY10MuonTPThinningTool = DerivationFramework__MuonTrackParticleThinning( + name = "BPHY10MuonTPThinningTool", + MuonKey = "Muons", + InDetTrackParticlesKey = "InDetTrackParticles") +ToolSvc += BPHY10MuonTPThinningTool + + +#==================================================================== +# CREATE THE DERIVATION KERNEL ALGORITHM AND PASS THE ABOVE TOOLS +#==================================================================== + +thiningCollection = [] + +print(thiningCollection) + + +# The name of the kernel (BPHY10Kernel in this case) must be unique to this derivation +from DerivationFrameworkCore.DerivationFrameworkCoreConf import DerivationFramework__DerivationKernel +DerivationFrameworkJob += CfgMgr.DerivationFramework__DerivationKernel( + "BPHY10Kernel", + AugmentationTools = [BPHY10JpsiSelectAndWrite, BPHY10_Select_Jpsi2mumu, + BPHY10BdKstSelectAndWrite, BPHY10_Select_Bd2JpsiKst, BPHY10_Select_Bd2JpsiKstbar, + BPHY10_Reco_V0Finder, BPHY10JpsiKshort, BPHY10JpsiLambda, BPHY10JpsiLambdabar, + BPHY10_AugOriginalCounts], + #Only skim if not MC + SkimmingTools = [BPHY10SkimmingOR] if not isSimulation else [], + ThinningTools = thiningCollection + ) + +#==================================================================== +# SET UP STREAM +#==================================================================== +streamName = derivationFlags.WriteDAOD_BPHY10Stream.StreamName +fileName = buildFileName( derivationFlags.WriteDAOD_BPHY10Stream ) +BPHY10Stream = MSMgr.NewPoolRootStream( streamName, fileName ) +BPHY10Stream.AcceptAlgs(["BPHY10Kernel"]) + +# Special lines for thinning +# Thinning service name must match the one passed to the thinning tools +from AthenaServices.Configurables import ThinningSvc, createThinningSvc +augStream = MSMgr.GetStream( streamName ) +evtStream = augStream.GetEventStream() + +BPHY10ThinningSvc = createThinningSvc( svcName="BPHY10ThinningSvc", outStreams=[evtStream] ) +svcMgr += BPHY10ThinningSvc + +#==================================================================== +# Slimming +#==================================================================== + +# Added by ASC +from DerivationFrameworkCore.SlimmingHelper import SlimmingHelper +BPHY10SlimmingHelper = SlimmingHelper("BPHY10SlimmingHelper") +AllVariables = [] +StaticContent = [] + +# Needed for trigger objects +BPHY10SlimmingHelper.IncludeMuonTriggerContent = TRUE +BPHY10SlimmingHelper.IncludeBPhysTriggerContent = TRUE + +## primary vertices +AllVariables += ["PrimaryVertices"] +StaticContent += ["xAOD::VertexContainer#BPHY10RefittedPrimaryVertices"] +StaticContent += ["xAOD::VertexAuxContainer#BPHY10RefittedPrimaryVerticesAux."] + +## ID track particles +AllVariables += ["InDetTrackParticles"] + +## combined / extrapolated muon track particles +## (note: for tagged muons there is no extra TrackParticle collection since the ID tracks +## are store in InDetTrackParticles collection) +AllVariables += ["CombinedMuonTrackParticles"] +AllVariables += ["ExtrapolatedMuonTrackParticles"] + +## muon container +AllVariables += ["Muons"] + + +## Jpsi candidates +StaticContent += ["xAOD::VertexContainer#%s" % BPHY10JpsiSelectAndWrite.OutputVtxContainerName] +## we have to disable vxTrackAtVertex branch since it is not xAOD compatible +StaticContent += ["xAOD::VertexAuxContainer#%sAux.-vxTrackAtVertex" % BPHY10JpsiSelectAndWrite.OutputVtxContainerName] + +StaticContent += ["xAOD::VertexContainer#%s" % BPHY10BdKstSelectAndWrite.OutputVtxContainerName] +StaticContent += ["xAOD::VertexAuxContainer#%sAux.-vxTrackAtVertex" % BPHY10BdKstSelectAndWrite.OutputVtxContainerName] + +StaticContent += ["xAOD::VertexContainer#%s" % 'BPHY10RecoV0Candidates'] +StaticContent += ["xAOD::VertexAuxContainer#%sAux.-vxTrackAtVertex" % 'BPHY10RecoV0Candidates'] +StaticContent += ["xAOD::VertexContainer#%s" % 'BPHY10RecoKshortCandidates'] +StaticContent += ["xAOD::VertexAuxContainer#%sAux.-vxTrackAtVertex" % 'BPHY10RecoKshortCandidates'] +StaticContent += ["xAOD::VertexContainer#%s" % 'BPHY10RecoLambdaCandidates'] +StaticContent += ["xAOD::VertexAuxContainer#%sAux.-vxTrackAtVertex" % 'BPHY10RecoLambdaCandidates'] +StaticContent += ["xAOD::VertexContainer#%s" % 'BPHY10RecoLambdabarCandidates'] +StaticContent += ["xAOD::VertexAuxContainer#%sAux.-vxTrackAtVertex" % 'BPHY10RecoLambdabarCandidates'] + +for cascades in CascadeCollections: + StaticContent += ["xAOD::VertexContainer#%s" % cascades] + StaticContent += ["xAOD::VertexAuxContainer#%sAux.-vxTrackAtVertex" % cascades] + +# Tagging information (in addition to that already requested by usual algorithms) +AllVariables += ["GSFTrackParticles", "MuonSpectrometerTrackParticles" ] + + + +# Added by ASC +# Truth information for MC only +if isSimulation: + AllVariables += ["TruthEvents","TruthParticles","TruthVertices","MuonTruthParticles"] + +AllVariables = list(set(AllVariables)) # remove duplicates + +BPHY10SlimmingHelper.AllVariables = AllVariables +BPHY10SlimmingHelper.StaticContent = StaticContent +BPHY10SlimmingHelper.SmartCollections = [] + +BPHY10SlimmingHelper.AppendContentToStream(BPHY10Stream) + + diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/BPHY11.py b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/BPHY11.py new file mode 100644 index 00000000000..a0ea555b5e4 --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/BPHY11.py @@ -0,0 +1,506 @@ +#==================================================================== +# BPHY11.py +# Lambda_b -> J/psi p K +# It requires the reductionConf flag BPHY11 in Reco_tf.py +#==================================================================== + +# Set up common services and job object. +# This should appear in ALL derivation job options +from DerivationFrameworkCore.DerivationFrameworkMaster import * + +isSimulation = False +if globalflags.DataSource()=='geant4': + isSimulation = True + +print(isSimulation) + + +#==================================================================== +# AUGMENTATION TOOLS +#==================================================================== +## 1/ setup vertexing tools and services +#include( "JpsiUpsilonTools/configureServices.py" ) + +include("DerivationFrameworkBPhys/configureVertexing.py") +BPHY11_VertexTools = BPHYVertexTools("BPHY11") + + +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__AugOriginalCounts +BPHY11_AugOriginalCounts = DerivationFramework__AugOriginalCounts( + name = "BPHY11_AugOriginalCounts", + VertexContainer = "PrimaryVertices", + TrackContainer = "InDetTrackParticles" +) + +ToolSvc += BPHY11_AugOriginalCounts + + +#-------------------------------------------------------------------- +## 2/ setup JpsiFinder tool +## These are general tools independent of DerivationFramework that do the +## actual vertex fitting and some pre-selection. +from JpsiUpsilonTools.JpsiUpsilonToolsConf import Analysis__JpsiFinder +BPHY11_JpsiFinder = Analysis__JpsiFinder( + name = "BPHY11_JpsiFinder", + OutputLevel = INFO, + muAndMu = True, + muAndTrack = False, + TrackAndTrack = False, + assumeDiMuons = True, + invMassUpper = 3600.0, + invMassLower = 2600.0, + Chi2Cut = 30.0, + oppChargesOnly = False, + allChargeCombinations = True, + combOnly = False, + atLeastOneComb = True, + useCombinedMeasurement = False, # Only takes effect if combOnly=True + muonCollectionKey = "Muons", + TrackParticleCollection = "InDetTrackParticles", + V0VertexFitterTool = BPHY11_VertexTools.TrkV0Fitter, # V0 vertex fitter + useV0Fitter = False, # if False a TrkVertexFitterTool will be used + TrkVertexFitterTool = BPHY11_VertexTools.TrkVKalVrtFitter, # VKalVrt vertex fitter + TrackSelectorTool = BPHY11_VertexTools.InDetTrackSelectorTool, + VertexPointEstimator = BPHY11_VertexTools.VtxPointEstimator, + useMCPCuts = False +) + +ToolSvc += BPHY11_JpsiFinder +print(BPHY11_JpsiFinder) + +#-------------------------------------------------------------------- +## 3/ setup the vertex reconstruction "call" tool(s). They are part of the derivation framework. +## These Augmentation tools add output vertex collection(s) into the StoreGate and add basic +## decorations which do not depend on the vertex mass hypothesis (e.g. lxy, ptError, etc). +## There should be one tool per topology, i.e. Jpsi and Psi(2S) do not need two instance of the +## Reco tool is the JpsiFinder mass window is wide enough. +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__Reco_Vertex +BPHY11_JpsiSelectAndWrite = DerivationFramework__Reco_Vertex( + name = "BPHY11_JpsiSelectAndWrite", + VertexSearchTool = BPHY11_JpsiFinder, + OutputVtxContainerName = "BPHY11_JpsiCandidates", + PVContainerName = "PrimaryVertices", + RefPVContainerName = "SHOULDNOTBEUSED", + DoVertexType = 1 +) + +ToolSvc += BPHY11_JpsiSelectAndWrite +print(BPHY11_JpsiSelectAndWrite) + + +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__Select_onia2mumu + +## a/ augment and select Jpsi->mumu candidates +BPHY11_Select_Jpsi2mumu = DerivationFramework__Select_onia2mumu( + name = "BPHY11_Select_Jpsi2mumu", + HypothesisName = "Jpsi", + InputVtxContainerName = "BPHY11_JpsiCandidates", + VtxMassHypo = 3096.900, + MassMin = 2600.0, + MassMax = 3600.0, + Chi2Max = 30.0, + DoVertexType = 1 +) + +ToolSvc += BPHY11_Select_Jpsi2mumu +print(BPHY11_Select_Jpsi2mumu) + + +## 4/ setup a new vertexing tool (necessary due to use of mass constraint) +from TrkVKalVrtFitter.TrkVKalVrtFitterConf import Trk__TrkVKalVrtFitter +LbJpsipKVertexFit = Trk__TrkVKalVrtFitter( + name = "LbJpsipKVertexFit", + Extrapolator = BPHY11_VertexTools.InDetExtrapolator, + FirstMeasuredPoint = False, + MakeExtendedVertex = True, + usePassWithTrkErrCnst = True +) + +ToolSvc += LbJpsipKVertexFit +print(LbJpsipKVertexFit) + + +## 5/ setup the Jpsi+2 track finder +from JpsiUpsilonTools.JpsiUpsilonToolsConf import Analysis__JpsiPlus2Tracks +BPHY11_LbJpsipK = Analysis__JpsiPlus2Tracks( + name = "BPHY11_LbJpsipK", + OutputLevel = INFO, + kaonkaonHypothesis = False, + pionpionHypothesis = False, + kaonpionHypothesis = False, + kaonprotonHypothesis = True, + oppChargesOnly = False, + trkThresholdPt = 1500.0, + trkMaxEta = 3.0, + BMassUpper = 6500.0, + BMassLower = 4000.0, +# DiTrackMassUpper = 10000., + DiTrackMassLower = 1000., + Chi2Cut = 200.0, + TrkQuadrupletMassUpper = 7000.0, + TrkQuadrupletMassLower = 4000.0, + JpsiContainerKey = "BPHY11_JpsiCandidates", + TrackParticleCollection = "InDetTrackParticles", + MuonsUsedInJpsi = "Muons", + TrkVertexFitterTool = LbJpsipKVertexFit, + TrackSelectorTool = BPHY11_VertexTools.InDetTrackSelectorTool, + UseMassConstraint = True, + UseVertexFittingWithPV = True, + VertexContainer = "PrimaryVertices" +) + +ToolSvc += BPHY11_LbJpsipK +print(BPHY11_LbJpsipK) + +## 6/ setup the combined augmentation/skimming tool for the Bpm +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__Reco_Vertex +BPHY11_LbJpsipKSelectAndWrite = DerivationFramework__Reco_Vertex( + name = "BPHY11_LbJpsipKSelectAndWrite", + Jpsi2PlusTrackName = BPHY11_LbJpsipK, + OutputVtxContainerName = "LbJpsipKCandidates", + PVContainerName = "PrimaryVertices", + RefPVContainerName = "BPHY11_RefittedPrimaryVertices", + RefitPV = True, + MaxPVrefit = 10000, + DoVertexType = 7 +) + +ToolSvc += BPHY11_LbJpsipKSelectAndWrite +print(BPHY11_LbJpsipKSelectAndWrite) + +## b/ augment and select Lb->JpsipK candidates +BPHY11_Select_Lb2JpsipK = DerivationFramework__Select_onia2mumu( + name = "BPHY11_Select_Lb2JpsipK", + HypothesisName = "Lb_pK", + InputVtxContainerName = "LbJpsipKCandidates", + TrkMasses = [105.658, 105.658, 938.272, 493.677], + VtxMassHypo = 5619.6, + MassMin = 4000.0, + MassMax = 6500.0, + Chi2Max = 200, + LxyMin = 0.3 +) + +ToolSvc += BPHY11_Select_Lb2JpsipK +print(BPHY11_Select_Lb2JpsipK) + + +## b/ augment and select Lb->JpsiKp candidates +BPHY11_Select_Lb2JpsiKp = DerivationFramework__Select_onia2mumu( + name = "BPHY11_Select_Lb2JpsiKp", + HypothesisName = "Lb_Kp", + InputVtxContainerName = "LbJpsipKCandidates", + TrkMasses = [105.658, 105.658, 493.677, 938.272], + VtxMassHypo = 5619.6, + MassMin = 4000.0, + MassMax = 6500.0, + Chi2Max = 200.0, + LxyMin = 0.3 +) + +ToolSvc += BPHY11_Select_Lb2JpsiKp +print(BPHY11_Select_Lb2JpsiKp) + +#------------------------------------------------------- +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__ReVertex +BPHY11_LbPlusTrk = DerivationFramework__ReVertex( + name = "BPHY11_LbPlusTrk", + InputVtxContainerName = "LbJpsipKCandidates", + HypothesisNames = [ BPHY11_Select_Lb2JpsipK.HypothesisName, BPHY11_Select_Lb2JpsiKp.HypothesisName ], + TrackIndices = [ 0, 1, 2, 3 ], + UseAdditionalTrack = True, + UseMassConstraint = True, + UseVertexFittingWithPV = True, +# VertexMass = 5619.6, + SubVertexMass = 3096.900, + MassInputParticles = [ 105.658, 105.658, 139.57, 139.57, 139.57 ], + SubVertexTrackIndices = [ 1, 2 ], + BMassUpper = 10000.0, + BMassLower = 4000.0, + Chi2Cut = 5.0, + TrkVertexFitterTool = LbJpsipKVertexFit, + OutputVtxContainerName = "LbJpsipKTrkCandidates" +) + +ToolSvc += BPHY11_LbPlusTrk +print(BPHY11_LbPlusTrk) + +BPHY11_Select_LbPlusTrk = DerivationFramework__Select_onia2mumu( + name = "BPHY11_Select_LbPlusTrk", + HypothesisName = "LbPlusTrk", + InputVtxContainerName = "LbJpsipKTrkCandidates", + TrkMasses = BPHY11_LbPlusTrk.MassInputParticles, + VtxMassHypo = 5619.6, + MassMin = 4000.0, + MassMax = 10000.0, + Chi2Max = 50.0 +) + +ToolSvc += BPHY11_Select_LbPlusTrk +print(BPHY11_Select_LbPlusTrk) + +#------------------------------------------------------- + +BPHY11_Lb_pK_ReFit = DerivationFramework__ReVertex( + name = "BPHY11_Lb_pK_ReFit", + InputVtxContainerName = "LbJpsipKCandidates", + HypothesisNames = [ BPHY11_Select_Lb2JpsipK.HypothesisName ], + TrackIndices = [ 0, 1, 2, 3 ], + UseMassConstraint = True, + UseVertexFittingWithPV = True, + VertexMass = 5619.6, + SubVertexMass = 3096.900, + MassInputParticles = [ 105.658, 105.658, 938.272, 493.677 ], + SubVertexTrackIndices = [ 1, 2 ], + TrkVertexFitterTool = LbJpsipKVertexFit, + OutputVtxContainerName = "LbJpsipKCandidatesReFit" +) + +ToolSvc += BPHY11_Lb_pK_ReFit +print(BPHY11_Lb_pK_ReFit) + +BPHY11_Select_Lb_pK_ReFit = DerivationFramework__Select_onia2mumu( + name = "BPHY11_Select_Lb_pK_ReFit", + HypothesisName = "Lb_pK_ReFit", + InputVtxContainerName = "LbJpsipKCandidatesReFit", + TrkMasses = BPHY11_Lb_pK_ReFit.MassInputParticles, + VtxMassHypo = 5619.6, + MassMin = 0.0, + MassMax = 1.0e10, + Chi2Max = 1.0e10 +) + +ToolSvc += BPHY11_Select_Lb_pK_ReFit +print(BPHY11_Select_Lb_pK_ReFit) + +BPHY11_Lb_Kp_ReFit = DerivationFramework__ReVertex( + name = "BPHY11_Lb_Kp_ReFit", + InputVtxContainerName = "LbJpsipKCandidates", + HypothesisNames = [ BPHY11_Select_Lb2JpsiKp.HypothesisName ], + TrackIndices = [ 0, 1, 2, 3 ], + UseMassConstraint = True, + UseVertexFittingWithPV = True, + VertexMass = 5619.6, + SubVertexMass = 3096.900, + MassInputParticles = [ 105.658, 105.658, 493.677, 938.272 ], + SubVertexTrackIndices = [ 1, 2 ], + TrkVertexFitterTool = LbJpsipKVertexFit, + OutputVtxContainerName = "LbJpsiKpCandidatesReFit" +) + +ToolSvc += BPHY11_Lb_Kp_ReFit +print(BPHY11_Lb_Kp_ReFit) + +BPHY11_Select_Lb_Kp_ReFit = DerivationFramework__Select_onia2mumu( + name = "BPHY11_Select_Lb_Kp_ReFit", + HypothesisName = "Lb_Kp_ReFit", + InputVtxContainerName = "LbJpsiKpCandidatesReFit", + TrkMasses = BPHY11_Lb_Kp_ReFit.MassInputParticles, + VtxMassHypo = 5619.6, + MassMin = 0.0, + MassMax = 1.0e10, + Chi2Max = 1.0e10 +) + +ToolSvc += BPHY11_Select_Lb_Kp_ReFit +print(BPHY11_Select_Lb_Kp_ReFit) + + + +#from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__SelectEvent + +if not isSimulation: #Only Skim Data + from DerivationFrameworkTools.DerivationFrameworkToolsConf import DerivationFramework__xAODStringSkimmingTool + BPHY11_SelectLdJpsipKEvent = DerivationFramework__xAODStringSkimmingTool( + name = "BPHY11_SelectLdJpsipKEvent", + expression = "count(LbJpsipKCandidates.passed_Lb_pK > 0) > 0" + ) + + ToolSvc += BPHY11_SelectLdJpsipKEvent + print(BPHY11_SelectLdJpsipKEvent) + + BPHY11_SelectLdJpsiKpEvent = DerivationFramework__xAODStringSkimmingTool( + name = "BPHY11_SelectLdJpsiKpEvent", + expression = "count(LbJpsipKCandidates.passed_Lb_Kp > 0) > 0" + ) + + ToolSvc += BPHY11_SelectLdJpsiKpEvent + print(BPHY11_SelectLdJpsiKpEvent) + + #==================================================================== + # Make event selection based on an OR of the input skimming tools + #==================================================================== + + from DerivationFrameworkTools.DerivationFrameworkToolsConf import DerivationFramework__FilterCombinationOR + BPHY11_SkimmingOR = CfgMgr.DerivationFramework__FilterCombinationOR( + "BPHY11_SkimmingOR", + FilterList = [BPHY11_SelectLdJpsipKEvent,BPHY11_SelectLdJpsiKpEvent],) + + ToolSvc += BPHY11_SkimmingOR + print(BPHY11_SkimmingOR) + +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__Thin_vtxTrk +BPHY11_thinningTool_Tracks = DerivationFramework__Thin_vtxTrk( + name = "BPHY11_thinningTool_Tracks", + TrackParticleContainerName = "InDetTrackParticles", + VertexContainerNames = ["LbJpsipKCandidates"], + PassFlags = ["passed_Lb_pK","passed_Lb_Kp"] +) + +ToolSvc += BPHY11_thinningTool_Tracks + +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__BPhysPVThinningTool +BPHY11_thinningTool_PV = DerivationFramework__BPhysPVThinningTool( + name = "BPHY11_thinningTool_PV", + CandidateCollections = ["LbJpsipKCandidates"], + KeepPVTracks = True +) + +ToolSvc += BPHY11_thinningTool_PV + + +## b) thinning out tracks that are not attached to muons. The final thinning decision is based on the OR operation +## between decision from this and the previous tools. +from DerivationFrameworkInDet.DerivationFrameworkInDetConf import DerivationFramework__MuonTrackParticleThinning +BPHY11_MuonTPThinningTool = DerivationFramework__MuonTrackParticleThinning( + name = "BPHY11_MuonTPThinningTool", + MuonKey = "Muons", + InDetTrackParticlesKey = "InDetTrackParticles" +) + +ToolSvc += BPHY11_MuonTPThinningTool + +from DerivationFrameworkInDet.DerivationFrameworkInDetConf import DerivationFramework__EgammaTrackParticleThinning +BPHY11_ElectronTPThinningTool = DerivationFramework__EgammaTrackParticleThinning( + name = "BPHY11_ElectronTPThinningTool", + SGKey = "Electrons", + GSFTrackParticlesKey = "GSFTrackParticles", + InDetTrackParticlesKey = "InDetTrackParticles", + SelectionString = "", + BestMatchOnly = True, + ConeSize = 0.3, + ApplyAnd = False +) + +ToolSvc+=BPHY11_ElectronTPThinningTool + +#==================================================================== +# CREATE THE DERIVATION KERNEL ALGORITHM AND PASS THE ABOVE TOOLS +#==================================================================== + +BPHY11_ThiningCollection = [BPHY11_thinningTool_Tracks, + BPHY11_thinningTool_PV, + BPHY11_MuonTPThinningTool, + BPHY11_ElectronTPThinningTool] +print(BPHY11_ThiningCollection) + + +# The name of the kernel (BPHY11_Kernel in this case) must be unique to this derivation +from DerivationFrameworkCore.DerivationFrameworkCoreConf import DerivationFramework__DerivationKernel +DerivationFrameworkJob += CfgMgr.DerivationFramework__DerivationKernel( + "BPHY11_Kernel", + AugmentationTools = [BPHY11_JpsiSelectAndWrite, BPHY11_Select_Jpsi2mumu, + BPHY11_LbJpsipKSelectAndWrite, BPHY11_Select_Lb2JpsipK, BPHY11_Select_Lb2JpsiKp, + BPHY11_LbPlusTrk, BPHY11_Select_LbPlusTrk, + BPHY11_Lb_pK_ReFit, BPHY11_Select_Lb_pK_ReFit, + BPHY11_Lb_Kp_ReFit, BPHY11_Select_Lb_Kp_ReFit, + BPHY11_AugOriginalCounts], + #Only skim if not MC + SkimmingTools = [BPHY11_SkimmingOR] if not isSimulation else [], + ThinningTools = BPHY11_ThiningCollection +) + +#==================================================================== +# SET UP STREAM +#==================================================================== +streamName = derivationFlags.WriteDAOD_BPHY11Stream.StreamName +fileName = buildFileName( derivationFlags.WriteDAOD_BPHY11Stream ) +BPHY11Stream = MSMgr.NewPoolRootStream( streamName, fileName ) +BPHY11Stream.AcceptAlgs(["BPHY11_Kernel"]) + +# Special lines for thinning +# Thinning service name must match the one passed to the thinning tools +from AthenaServices.Configurables import ThinningSvc, createThinningSvc +augStream = MSMgr.GetStream( streamName ) +evtStream = augStream.GetEventStream() + +BPHY11_ThinningSvc = createThinningSvc( svcName="BPHY11_ThinningSvc", outStreams=[evtStream] ) +svcMgr += BPHY11_ThinningSvc + +#==================================================================== +# Slimming +#==================================================================== + +# Added by ASC +from DerivationFrameworkCore.SlimmingHelper import SlimmingHelper +BPHY11_SlimmingHelper = SlimmingHelper("BPHY11_SlimmingHelper") +AllVariables = [] +StaticContent = [] + +# Needed for trigger objects +BPHY11_SlimmingHelper.IncludeMuonTriggerContent = TRUE +BPHY11_SlimmingHelper.IncludeBPhysTriggerContent = TRUE + +## primary vertices +AllVariables += ["PrimaryVertices"] +StaticContent += ["xAOD::VertexContainer#BPHY11_RefittedPrimaryVertices"] +StaticContent += ["xAOD::VertexAuxContainer#BPHY11_RefittedPrimaryVerticesAux."] + +## ID track particles +AllVariables += ["InDetTrackParticles"] + +## combined / extrapolated muon track particles +## (note: for tagged muons there is no extra TrackParticle collection since the ID tracks +## are store in InDetTrackParticles collection) +AllVariables += ["CombinedMuonTrackParticles"] +AllVariables += ["ExtrapolatedMuonTrackParticles"] + +## muon container +AllVariables += ["Muons"] + +## Jpsi candidates +StaticContent += ["xAOD::VertexContainer#%s" % BPHY11_JpsiSelectAndWrite.OutputVtxContainerName] +## we have to disable vxTrackAtVertex branch since it is not xAOD compatible +StaticContent += ["xAOD::VertexAuxContainer#%sAux.-vxTrackAtVertex" % BPHY11_JpsiSelectAndWrite.OutputVtxContainerName] + +StaticContent += ["xAOD::VertexContainer#%s" % BPHY11_LbJpsipKSelectAndWrite.OutputVtxContainerName] +StaticContent += ["xAOD::VertexAuxContainer#%sAux.-vxTrackAtVertex" % BPHY11_LbJpsipKSelectAndWrite.OutputVtxContainerName] + +StaticContent += ["xAOD::VertexContainer#%s" % BPHY11_LbPlusTrk.OutputVtxContainerName] +StaticContent += ["xAOD::VertexAuxContainer#%sAux.-vxTrackAtVertex" % BPHY11_LbPlusTrk.OutputVtxContainerName] + +StaticContent += ["xAOD::VertexContainer#%s" % BPHY11_Lb_pK_ReFit.OutputVtxContainerName] +StaticContent += ["xAOD::VertexAuxContainer#%sAux.-vxTrackAtVertex" % BPHY11_Lb_pK_ReFit.OutputVtxContainerName] + +StaticContent += ["xAOD::VertexContainer#%s" % BPHY11_Lb_Kp_ReFit.OutputVtxContainerName] +StaticContent += ["xAOD::VertexAuxContainer#%sAux.-vxTrackAtVertex" % BPHY11_Lb_Kp_ReFit.OutputVtxContainerName] + + + +# Tagging information (in addition to that already requested by usual algorithms) +#AllVariables += ["Electrons"] +AllVariables += ["GSFTrackParticles"] +tagJetCollections = ['AntiKt4LCTopoJets'] + +for jet_collection in tagJetCollections: + AllVariables += [jet_collection] + AllVariables += ["BTagging_%s" % (jet_collection[:-4]) ] + AllVariables += ["BTagging_%sJFVtx" % (jet_collection[:-4]) ] + AllVariables += ["BTagging_%sSecVtx" % (jet_collection[:-4]) ] + + + + +# Added by ASC +# Truth information for MC only +if isSimulation: + AllVariables += ["TruthEvents","TruthParticles","TruthVertices","MuonTruthParticles"] + AllVariables += ["AntiKt4TruthJets","egammaTruthParticles"] + +BPHY11_SlimmingHelper.AllVariables = AllVariables +BPHY11_SlimmingHelper.StaticContent = StaticContent +BPHY11_SlimmingHelper.SmartCollections = ["Electrons" , "Photons"] + +BPHY11_SlimmingHelper.AppendContentToStream(BPHY11Stream) + + diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/BPHY12.py b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/BPHY12.py new file mode 100644 index 00000000000..90f52acbc56 --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/BPHY12.py @@ -0,0 +1,389 @@ +#==================================================================== +# BPHY12.py +# This an example job options script showing how to set up a +# derivation of the data using the derivation framework. +# It requires the reductionConf flag BPHY12 in Reco_tf.py +#==================================================================== + +#==================================================================== +# FLAGS TO PERSONALIZE THE DERIVATION +#==================================================================== + +skimTruth = False + +# Set up common services and job object. +# This should appear in ALL derivation job options +from DerivationFrameworkCore.DerivationFrameworkMaster import * + +isSimulation = False +if globalflags.DataSource()=='geant4': + isSimulation = True + +print('is this simulation? ', isSimulation) + +#==================================================================== +# AUGMENTATION TOOLS +#==================================================================== +## 1/ setup vertexing tools and services +include("DerivationFrameworkBPhys/configureVertexing.py") +BPHY12_VertexTools = BPHYVertexTools("BPHY12") + +print('********************** VERTEX TOOLS ***********************') +print(BPHY12_VertexTools) +print(BPHY12_VertexTools.TrkV0Fitter) +print('********************** END VERTEX TOOLS ***********************') + +#==================================================================== +# TriggerCounting for Kernel1 #Added by Matteo +#==================================================================== +#List of trigggers to be counted (high Sig-eff*Lumi ones are in) +triggersToMetadata= [ +"HLT_mu11_mu6_bBmumuxv2", +"HLT_2mu10_bBmumuxv2", +"HLT_2mu6_bBmumuxv2_L1LFV-MU6", +"HLT_mu11_mu6_bBmumux_BpmumuKp", +"HLT_2mu6_bBmumux_BpmumuKp_L1BPH-2M9-2MU6_BPH-2DR15-2MU6", + +"HLT_mu11_mu6_bDimu", +"HLT_4mu4_bDimu6000" + ] + + +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__TriggerCountToMetadata +BPHY12TriggerCountToMetadata = DerivationFramework__TriggerCountToMetadata(name = "BPHY12TriggerCount", + TriggerList = triggersToMetadata, + FolderName = "BPHY12") + +ToolSvc += BPHY12TriggerCountToMetadata + +#==================================================================== +# PRESELECTION for Kernel1 #Added by Matteo +#==================================================================== +## 1/ Setup the skimming based on triggers +## + +triggerList = [ +"HLT_mu11_mu6_bBmumuxv2", +"HLT_2mu10_bBmumuxv2", +"HLT_2mu6_bBmumuxv2_L1LFV-MU6", +"HLT_mu11_mu6_bBmumux_BpmumuKp", +"HLT_2mu6_bBmumux_BpmumuKp_L1BPH-2M9-2MU6_BPH-2DR15-2MU6", +"HLT_mu11_mu6_bDimu", +"HLT_4mu4_bDimu6000" + ] + +from DerivationFrameworkTools.DerivationFrameworkToolsConf import DerivationFramework__TriggerSkimmingTool +BPHY12TriggerSkim = DerivationFramework__TriggerSkimmingTool(name = "BPHY12TriggerSkim", + TriggerListOR = triggerList, + TriggerListAND = [] ) + +ToolSvc += BPHY12TriggerSkim + +#-------------------------------------------------------------------- +## 2/ Setup the vertex fitter tools (e.g. JpsiFinder, JpsiPlus1Track, etc). +## These are general tools independent of DerivationFramework that do the +## actual vertex fitting and some pre-selection. +from JpsiUpsilonTools.JpsiUpsilonToolsConf import Analysis__JpsiFinder +BPHY12DiMuonFinder = Analysis__JpsiFinder( + name = "BPHY12DiMuonFinder", + OutputLevel = INFO, + muAndMu = True, + muAndTrack = False, + TrackAndTrack = False, + assumeDiMuons = True, # If true, will assume dimu hypothesis and use PDG value for mu mass + invMassUpper = 100000.0, + invMassLower = 0.0, + Chi2Cut = 200., + oppChargesOnly = True, + atLeastOneComb = True, + useCombinedMeasurement = False, # Only takes effect if combOnly=True + muonCollectionKey = "Muons", + TrackParticleCollection = "InDetTrackParticles", + V0VertexFitterTool = BPHY12_VertexTools.TrkV0Fitter, # V0 vertex fitter + useV0Fitter = False, # if False a TrkVertexFitterTool will be used + TrkVertexFitterTool = BPHY12_VertexTools.TrkVKalVrtFitter, # VKalVrt vertex fitter + TrackSelectorTool = BPHY12_VertexTools.InDetTrackSelectorTool, + VertexPointEstimator = BPHY12_VertexTools.VtxPointEstimator, + useMCPCuts = False ) + +ToolSvc += BPHY12DiMuonFinder +print(BPHY12DiMuonFinder) + +#-------------------------------------------------------------------- +## 3/ setup the vertex reconstruction "call" tool(s). They are part of the derivation framework. +## These Augmentation tools add output vertex collection(s) into the StoreGate and add basic +## decorations which do not depend on the vertex mass hypothesis (e.g. lxy, ptError, etc). +## There should be one tool per topology, i.e. Jpsi and Psi(2S) do not need two instance of the +## Reco tool is the JpsiFinder mass window is wide enough. + +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__Reco_Vertex +BPHY12_Reco_DiMuon = DerivationFramework__Reco_Vertex( + name = "BPHY12_Reco_DiMuon", + VertexSearchTool = BPHY12DiMuonFinder, + OutputVtxContainerName = "BPHY12DiMuonCandidates", + PVContainerName = "PrimaryVertices", + RefPVContainerName = "BPHY12RefittedPrimaryVertices", + RefitPV = True, + MaxPVrefit = 100000, + DoVertexType = 7) + +ToolSvc += BPHY12_Reco_DiMuon +print(BPHY12_Reco_DiMuon) + +#-------------------------------------------------------------------- +## 4/ setup the vertex selection and augmentation tool(s). These tools decorate the vertices with +## variables that depend on the vertex mass hypothesis, e.g. invariant mass, proper decay time, etc. +## Property HypothesisName is used as a prefix for these decorations. +## They also perform tighter selection, flagging the vertecis that passed. The flag is a Char_t branch +## named "passed_"+HypothesisName. It is used later by the "SelectEvent" and "Thin_vtxTrk" tools +## to determine which events and candidates should be kept in the output stream. +## Multiple instances of the Select_* tools can be used on a single input collection as long as they +## use different "HypothesisName" flags. + +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__Select_onia2mumu + +## a/ augment and select Jpsi->mumu candidates +BPHY12_Select_DiMuons = DerivationFramework__Select_onia2mumu( + name = "BPHY12_Select_DiMuons", + HypothesisName = "Jpsi", + InputVtxContainerName = "BPHY12DiMuonCandidates", + VtxMassHypo = 3096.916, + MassMin = 1.0, + MassMax = 7000.0, + Chi2Max = 200., + DoVertexType = 7) + +ToolSvc += BPHY12_Select_DiMuons +print(BPHY12_Select_DiMuons) + +## 4/ setup a new vertexing tool (necessary due to use of mass constraint) +from TrkVKalVrtFitter.TrkVKalVrtFitterConf import Trk__TrkVKalVrtFitter +BmumuKstVertexFit = Trk__TrkVKalVrtFitter( + name = "BmumuKstVertexFit", + Extrapolator = BPHY12_VertexTools.InDetExtrapolator, + FirstMeasuredPoint = True, + MakeExtendedVertex = True) + +ToolSvc += BmumuKstVertexFit +print(BmumuKstVertexFit) + +## 5/ setup the Jpsi+2 track finder +from JpsiUpsilonTools.JpsiUpsilonToolsConf import Analysis__JpsiPlus2Tracks +BPHY12BmumuKstFinder = Analysis__JpsiPlus2Tracks( + name = "BPHY12BmumuKstFinder", + OutputLevel = INFO, #can also be DEBUG, WARNING, VERBOSE + kaonkaonHypothesis = False, + pionpionHypothesis = False, + kaonpionHypothesis = True, + trkThresholdPt = 500.0, #minimum track pT in MeV + trkMaxEta = 3.0, + BThresholdPt = 1000., + BMassLower = 3000.0, #OI makes no sense below Jpsi mass #same values as BPHY18 (original) - Bs->JpsiKK + BMassUpper = 6500.0, + JpsiContainerKey = "BPHY12DiMuonCandidates", + TrackParticleCollection = "InDetTrackParticles", + #MuonsUsedInJpsi = "Muons", #Don't remove all muons, just those in J/psi candidate (see the following cut) + ExcludeCrossJpsiTracks = False, #setting this to False rejects the muons from J/psi candidate + TrkVertexFitterTool = BmumuKstVertexFit, + TrackSelectorTool = BPHY12_VertexTools.InDetTrackSelectorTool, + UseMassConstraint = False, #Set to True, according to Bs->JpsiKK DAOD + DiTrackMassUpper = 1110., #OI was 1500. Can eventually set these to be the K* mass? + DiTrackMassLower = 690., #OI was 500 + Chi2Cut = 15., #THIS IS CHI2/NDOF, checked the code!!! + DiTrackPt = 500., + TrkQuadrupletMassLower = 1000.0, #Two electrons + two tracks (one K, one pi) + TrkQuadrupletMassUpper = 100000.0, # same as BPHY18, original + #FinalDiTrackMassUpper = 1000., + #FinalDiTrackMassLower = 800., + #TrkDeltaZ = 20., #Normally, this cut should not be used since it is lifetime-dependent + FinalDiTrackPt = 500., +#OI DoElectrons = True, + #UseGSFTrackIndices = [0,1] + ) + +ToolSvc += BPHY12BmumuKstFinder +print(BPHY12BmumuKstFinder) +## 6/ setup the combined augmentation/skimming tool for the BeeKst +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__Reco_Vertex +BPHY12_Reco_BmumuKst = DerivationFramework__Reco_Vertex( + name = "BPHY12_Reco_BmumuKst", + Jpsi2PlusTrackName = BPHY12BmumuKstFinder, + OutputVtxContainerName = "BPHY12BmumuKstCandidates", + PVContainerName = "PrimaryVertices", + RefPVContainerName = "BPHY12RefittedPrimaryVertices", + RefitPV = True, + MaxPVrefit = 10000, + DoVertexType = 7) + +ToolSvc += BPHY12_Reco_BmumuKst +print(BPHY12_Reco_BmumuKst) + +## b/ augment and select B->eeKst candidates +# set mass hypothesis (K pi) +BPHY12_Select_BmumuKst = DerivationFramework__Select_onia2mumu( + name = "BPHY12_Select_BmumuKst", + HypothesisName = "Bd", #creates output variable pass_Bd + InputVtxContainerName = "BPHY12BmumuKstCandidates", + TrkMasses = [105.658, 105.658, 493.677, 139.570], + VtxMassHypo = 5279.6, #mass of B + MassMin = 1.0, #no mass cuts here + MassMax = 10000.0, #no mass cuts here + Chi2Max = 30.0) #THIS IS CHI2! NOT CHI2/NDOF! Careful! + +ToolSvc += BPHY12_Select_BmumuKst +print(BPHY12_Select_BmumuKst) + +#-------------------------------------------------------------------- +## 5/ select the event. We only want to keep events that contain certain vertices which passed certain selection. +## This is specified by the "SelectionExpression" property, which contains the expression in the following format: +## +## "ContainerName.passed_HypoName > count" +## +## where "ContainerName" is output container form some Reco_* tool, "HypoName" is the hypothesis name setup in some "Select_*" +## tool and "count" is the number of candidates passing the selection you want to keep. + +if skimTruth or not isSimulation: #Only Skim Data + expression = "count(BPHY12BmumuKstCandidates.passed_Bd) > 0" + from DerivationFrameworkTools.DerivationFrameworkToolsConf import DerivationFramework__xAODStringSkimmingTool + BPHY12_SelectEvent = DerivationFramework__xAODStringSkimmingTool(name = "BPHY12_SelectEvent", + expression = expression) + ToolSvc += BPHY12_SelectEvent + print(BPHY12_SelectEvent) + + #==================================================================== + # Make event selection based on an OR of the input skimming tools (though it seems we only have one here!) + #==================================================================== + from DerivationFrameworkTools.DerivationFrameworkToolsConf import DerivationFramework__FilterCombinationOR + BPHY12SkimmingOR = CfgMgr.DerivationFramework__FilterCombinationOR( + name = "BPHY12SkimmingOR", + FilterList = [BPHY12_SelectEvent, BPHY12TriggerSkim]) #OR of all your different filters + ToolSvc += BPHY12SkimmingOR + print(BPHY12SkimmingOR) + +#-------------------------------------------------------------------- +## 6/ track and vertex thinning. We want to remove all reconstructed secondary vertices +## which hasn't passed any of the selections defined by (Select_*) tools. +## We also want to keep only tracks which are associates with either muons or any of the +## vertices that passed the selection. Multiple thinning tools can perform the +## selection. The final thinning decision is based OR of all the decisions (by default, +## although it can be changed by the JO). + +## a) thining out vertices that didn't pass any selection and idetifying tracks associated with +## selected vertices. The "VertexContainerNames" is a list of the vertex containers, and "PassFlags" +## contains all pass flags for Select_* tools that must be satisfied. The vertex is kept is it +## satisfy any of the listed selections. + +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__Thin_vtxTrk +BPHY12Thin_vtxTrk = DerivationFramework__Thin_vtxTrk( + name = "BPHY12Thin_vtxTrk", + TrackParticleContainerName = "InDetTrackParticles", + VertexContainerNames = ["BPHY12BmumuKstCandidates"], + PassFlags = ["passed_Bd"] ) + +ToolSvc += BPHY12Thin_vtxTrk + +## b) thinning out tracks that are not attached to muons. The final thinning decision is based on the OR operation +## between decision from this and the previous tools. +from DerivationFrameworkInDet.DerivationFrameworkInDetConf import DerivationFramework__MuonTrackParticleThinning +BPHY12MuonTPThinningTool = DerivationFramework__MuonTrackParticleThinning(name = "BPHY12MuonTPThinningTool", + MuonKey = "Muons", + InDetTrackParticlesKey = "InDetTrackParticles") +ToolSvc += BPHY12MuonTPThinningTool + +# Added by ASC +# Only save truth informtion directly associated with Onia +from DerivationFrameworkMCTruth.DerivationFrameworkMCTruthConf import DerivationFramework__GenericTruthThinning +BPHY12TruthThinTool = DerivationFramework__GenericTruthThinning(name = "BPHY12TruthThinTool", + ParticleSelectionString = "TruthParticles.pdgId == 511 || TruthParticles.pdgId == -511 || TruthParticles.pdgId == 531 || TruthParticles.pdgId == -531", + PreserveDescendants = True, + PreserveAncestors = True) +ToolSvc += BPHY12TruthThinTool +print(BPHY12TruthThinTool) + + +#==================================================================== +# CREATE THE DERIVATION KERNEL ALGORITHM AND PASS THE ABOVE TOOLS +#==================================================================== +## 7/ IMPORTANT bit. Don't forget to pass the tools to the DerivationKernel! If you don't do that, they will not be +## be executed! + +# Added by ASC +BPHY12ThinningTools = [BPHY12Thin_vtxTrk, BPHY12MuonTPThinningTool] +if globalflags.DataSource()=='geant4': + BPHY12ThinningTools.append(BPHY12TruthThinTool) + +# The name of the kernel (BPHY12Kernel in this case) must be unique to this derivation +from DerivationFrameworkCore.DerivationFrameworkCoreConf import DerivationFramework__DerivationKernel +DerivationFrameworkJob += CfgMgr.DerivationFramework__DerivationKernel( + "BPHY12Kernel", + AugmentationTools = [BPHY12_Reco_DiMuon, BPHY12_Select_DiMuons, + BPHY12_Reco_BmumuKst, BPHY12_Select_BmumuKst], + SkimmingTools = [BPHY12SkimmingOR] if skimTruth or not isSimulation else [], + ThinningTools = BPHY12ThinningTools + ) + +#==================================================================== +# SET UP STREAM +#==================================================================== +streamName = derivationFlags.WriteDAOD_BPHY12Stream.StreamName +fileName = buildFileName( derivationFlags.WriteDAOD_BPHY12Stream ) +BPHY12Stream = MSMgr.NewPoolRootStream( streamName, fileName ) +BPHY12Stream.AcceptAlgs(["BPHY12Kernel"]) +# Special lines for thinning +# Thinning service name must match the one passed to the thinning tools +from AthenaServices.Configurables import ThinningSvc, createThinningSvc +augStream = MSMgr.GetStream( streamName ) +evtStream = augStream.GetEventStream() +svcMgr += createThinningSvc( svcName="BPHY12ThinningSvc", outStreams=[evtStream] ) + + +#==================================================================== +# Slimming +#==================================================================== + +# Added by ASC +from DerivationFrameworkCore.SlimmingHelper import SlimmingHelper +BPHY12SlimmingHelper = SlimmingHelper("BPHY12SlimmingHelper") +AllVariables = [] +StaticContent = [] + +# Needed for trigger objects +BPHY12SlimmingHelper.IncludeMuonTriggerContent = True +BPHY12SlimmingHelper.IncludeBPhysTriggerContent = True + +## primary vertices +AllVariables += ["PrimaryVertices"] +StaticContent += ["xAOD::VertexContainer#BPHY12RefittedPrimaryVertices"] +StaticContent += ["xAOD::VertexAuxContainer#BPHY12RefittedPrimaryVerticesAux."] + +## ID track particles +AllVariables += ["InDetTrackParticles"] + +## combined / extrapolated muon track particles +## (note: for tagged muons there is no extra TrackParticle collection since the ID tracks +## are store in InDetTrackParticles collection) +AllVariables += ["CombinedMuonTrackParticles"] +AllVariables += ["ExtrapolatedMuonTrackParticles"] + +## muon container +AllVariables += ["Muons"] + +## Jpsi candidates +StaticContent += ["xAOD::VertexContainer#%s" % BPHY12_Reco_DiMuon.OutputVtxContainerName] +StaticContent += ["xAOD::VertexAuxContainer#%sAux." % BPHY12_Reco_DiMuon.OutputVtxContainerName] +## we have to disable vxTrackAtVertex branch since it is not xAOD compatible +StaticContent += ["xAOD::VertexAuxContainer#%sAux.-vxTrackAtVertex" % BPHY12_Reco_DiMuon.OutputVtxContainerName] + +StaticContent += ["xAOD::VertexContainer#%s" % BPHY12_Reco_BmumuKst.OutputVtxContainerName] +StaticContent += ["xAOD::VertexAuxContainer#%sAux." % BPHY12_Reco_BmumuKst.OutputVtxContainerName] +## we have to disable vxTrackAtVertex branch since it is not xAOD compatible +StaticContent += ["xAOD::VertexAuxContainer#%sAux.-vxTrackAtVertex" % BPHY12_Reco_BmumuKst.OutputVtxContainerName] + +# Added by ASC +# Truth information for MC only +if isSimulation: + AllVariables += ["TruthEvents","TruthParticles","TruthVertices","MuonTruthParticles"] + +BPHY12SlimmingHelper.AllVariables = AllVariables +BPHY12SlimmingHelper.StaticContent = StaticContent +BPHY12SlimmingHelper.AppendContentToStream(BPHY12Stream) diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/BPHY13.py b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/BPHY13.py new file mode 100644 index 00000000000..b0549a61618 --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/BPHY13.py @@ -0,0 +1,483 @@ +#==================================================================== +# BPHY13.py (Based on BPHY8, BPHY16, and the old BPHY13) +# Contact: xin.chen@cern.ch +#==================================================================== + +# Set up common services and job object. +# This should appear in ALL derivation job options +from DerivationFrameworkCore.DerivationFrameworkMaster import * + +isSimulation = False +if globalflags.DataSource()=='geant4': + isSimulation = True + +print(isSimulation) + +#==================================================================== +# AUGMENTATION TOOLS +#==================================================================== +## 1/ setup vertexing tools and services +include("DerivationFrameworkBPhys/configureVertexing.py") +BPHY13_VertexTools = BPHYVertexTools("BPHY13") + +#-------------------------------------------------------------------- +## 2/ Setup the vertex fitter tools (e.g. JpsiFinder, JpsiPlus1Track, etc). +## These are general tools independent of DerivationFramework that do the +## actual vertex fitting and some pre-selection. +from JpsiUpsilonTools.JpsiUpsilonToolsConf import Analysis__JpsiFinder +BPHY13JpsiFinder = Analysis__JpsiFinder( + name = "BPHY13JpsiFinder", + OutputLevel = INFO, + muAndMu = True, + muAndTrack = False, + TrackAndTrack = False, + assumeDiMuons = True, # If true, will assume dimu hypothesis and use PDG value for mu mass + trackThresholdPt = 2500., + invMassUpper = 12500., + invMassLower = 2000., + Chi2Cut = 200., + oppChargesOnly = True, + atLeastOneComb = True, + useCombinedMeasurement = False, # Only takes effect if combOnly=True + muonCollectionKey = "Muons", + TrackParticleCollection = "InDetTrackParticles", + V0VertexFitterTool = BPHY13_VertexTools.TrkV0Fitter, # V0 vertex fitter + useV0Fitter = False, # if False a TrkVertexFitterTool will be used + TrkVertexFitterTool = BPHY13_VertexTools.TrkVKalVrtFitter, # VKalVrt vertex fitter + TrackSelectorTool = BPHY13_VertexTools.InDetTrackSelectorTool, + VertexPointEstimator = BPHY13_VertexTools.VtxPointEstimator, + useMCPCuts = False ) + +ToolSvc += BPHY13JpsiFinder +print(BPHY13JpsiFinder) + +#-------------------------------------------------------------------- +## 3/ setup the vertex reconstruction "call" tool(s). They are part of the derivation framework. +## These Augmentation tools add output vertex collection(s) into the StoreGate and add basic +## decorations which do not depend on the vertex mass hypothesis (e.g. lxy, ptError, etc). +## There should be one tool per topology, i.e. Jpsi and Psi(2S) do not need two instance of the +## Reco tool if the JpsiFinder mass window is wide enough. + +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__Reco_Vertex +BPHY13_Reco_mumu = DerivationFramework__Reco_Vertex( + name = "BPHY13_Reco_mumu", + VertexSearchTool = BPHY13JpsiFinder, + OutputVtxContainerName = "BPHY13OniaCandidates", + PVContainerName = "PrimaryVertices", + RefPVContainerName = "SHOULDNOTBEUSED", +# RefPVContainerName = "BPHY13RefittedPrimaryVertices", +# RefitPV = True, +# MaxPVrefit = 10000, +#https://gitlab.cern.ch/atlas/athena/-/blob/21.2/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/BPhysPVTools.cxx#L259 +# bit pattern: doZ0BA|doZ0|doA0|doPt + DoVertexType = 1) + +ToolSvc += BPHY13_Reco_mumu +print(BPHY13_Reco_mumu) + +## 4/ setup a new vertexing tool (necessary due to use of mass constraint) +from TrkVKalVrtFitter.TrkVKalVrtFitterConf import Trk__TrkVKalVrtFitter +BPHY13VertexFit = Trk__TrkVKalVrtFitter( + name = "BPHY13VertexFit", + Extrapolator = BPHY13_VertexTools.InDetExtrapolator, +# FirstMeasuredPoint = True, + FirstMeasuredPoint = False, + MakeExtendedVertex = True) +ToolSvc += BPHY13VertexFit +print(BPHY13VertexFit) + +## 5/ setup the Jpsi+2 track finder +# https://gitlab.cern.ch/atlas/athena/-/blob/21.2/PhysicsAnalysis/JpsiUpsilonTools/src/JpsiPlus2Tracks.cxx +from JpsiUpsilonTools.JpsiUpsilonToolsConf import Analysis__JpsiPlus2Tracks +BPHY13Plus2Tracks = Analysis__JpsiPlus2Tracks( + name = "BPHY13Plus2Tracks", + # OutputLevel = DEBUG, + kaonkaonHypothesis = False, + pionpionHypothesis = False, + kaonpionHypothesis = False, + ManualMassHypo = [ 105.658, 105.658, 105.658, 105.658 ], + trkThresholdPt = 1500., + trkMaxEta = 2.5, + oppChargesOnly = False, + DiTrackMassUpper = 12500., + DiTrackMassLower = 2000., + TrkQuadrupletMassUpper = 25000., + TrkQuadrupletMassLower = 0., + Chi2Cut = 200., + JpsiContainerKey = "BPHY13OniaCandidates", + TrackParticleCollection = "InDetTrackParticles", + MuonsUsedInJpsi = "Muons", + ExcludeJpsiMuonsOnly = True, + RequireNMuonTracks = 1, + TrkVertexFitterTool = BPHY13VertexFit, + TrackSelectorTool = BPHY13_VertexTools.InDetTrackSelectorTool, + UseMassConstraint = False) + +ToolSvc += BPHY13Plus2Tracks +print(BPHY13Plus2Tracks) + +## 6/ setup the combined augmentation/skimming tool +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__Reco_Vertex +BPHY13FourTrackSelectAndWrite = DerivationFramework__Reco_Vertex( + name = "BPHY13FourTrackSelectAndWrite", + Jpsi2PlusTrackName = BPHY13Plus2Tracks, + OutputVtxContainerName = "BPHY13FourTrack", + PVContainerName = "PrimaryVertices", + RefPVContainerName = "BPHY13RefittedPrimaryVertices", + RefitPV = True, + MaxPVrefit = 10000, + DoVertexType = 7) + +ToolSvc += BPHY13FourTrackSelectAndWrite +print(BPHY13FourTrackSelectAndWrite) + + +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__Select_onia2mumu + +BPHY13_Select_FourTrack = DerivationFramework__Select_onia2mumu( + name = "BPHY13_Select_FourTrack", + HypothesisName = "FourTracks", + InputVtxContainerName = "BPHY13FourTrack", + TrkMasses = [105.658, 105.658, 105.658, 105.658], + VtxMassHypo = 6900.0, # for decay time + MassMin = 0., + MassMax = 25000., + Chi2Max = 200.) + +ToolSvc += BPHY13_Select_FourTrack +print(BPHY13_Select_FourTrack) + + +#==================================================================== +# Isolation +#==================================================================== + +#Track isolation for candidates +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__VertexTrackIsolation +BPHY13TrackIsolationDecorator = DerivationFramework__VertexTrackIsolation( + name = "BPHY13TrackIsolationDecorator", + OutputLevel = INFO, + TrackIsoTool = "xAOD::TrackIsolationTool", + TrackContainer = "InDetTrackParticles", + InputVertexContainer = "BPHY13FourTrack", + PassFlags = ["passed_FourTracks"], + DoIsoPerTrk = True, + RemoveDuplicate = 2 +) + +ToolSvc += BPHY13TrackIsolationDecorator +print(BPHY13TrackIsolationDecorator) + + +#==================================================================== +# Revertex with mass constraint +#==================================================================== + +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__ReVertex +BPHY13_Revertex_2mu = DerivationFramework__ReVertex( + name = "BPHY13_Revertex_2mu", + InputVtxContainerName = "BPHY13FourTrack", + TrackIndices = [ 0, 1 ], + RefitPV = True, + RefPVContainerName = "BPHY13RefittedPrimaryVertices", # use existing refitted PVs + UseMassConstraint = True, + VertexMass = 3096.916, + MassInputParticles = [105.658, 105.658], + TrkVertexFitterTool = BPHY13VertexFit, + OutputVtxContainerName = "BPHY13TwoMuon") + +ToolSvc += BPHY13_Revertex_2mu +print(BPHY13_Revertex_2mu) + +BPHY13_Select_TwoMuon = DerivationFramework__Select_onia2mumu( + name = "BPHY13_Select_TwoMuon", + HypothesisName = "TwoMuons", + InputVtxContainerName = "BPHY13TwoMuon", + TrkMasses = [105.658, 105.658], + VtxMassHypo = 3096.916, + MassMin = 2000., + MassMax = 3600., + Chi2Max = 200) + +ToolSvc += BPHY13_Select_TwoMuon +print(BPHY13_Select_TwoMuon) + +BPHY13_Revertex_2trk = DerivationFramework__ReVertex( + name = "BPHY13_Revertex_2trk", + InputVtxContainerName = "BPHY13FourTrack", + TrackIndices = [ 2, 3 ], + RefitPV = True, + RefPVContainerName = "BPHY13RefittedPrimaryVertices", # use existing refitted PVs + UseMassConstraint = True, + VertexMass = 3096.916, + MassInputParticles = [105.658, 105.658], + TrkVertexFitterTool = BPHY13VertexFit, + OutputVtxContainerName = "BPHY13TwoTrack") + +ToolSvc += BPHY13_Revertex_2trk +print(BPHY13_Revertex_2trk) + +BPHY13_Select_TwoTrack = DerivationFramework__Select_onia2mumu( + name = "BPHY13_Select_TwoTrack", + HypothesisName = "TwoTracks", + InputVtxContainerName = "BPHY13TwoTrack", + TrkMasses = [105.658, 105.658], + VtxMassHypo = 3096.916, + MassMin = 2000., + MassMax = 3600., + Chi2Max = 200) + +ToolSvc += BPHY13_Select_TwoTrack +print(BPHY13_Select_TwoTrack) + + +BPHY13_Revertex_2muHi = DerivationFramework__ReVertex( + name = "BPHY13_Revertex_2muHi", + InputVtxContainerName = "BPHY13FourTrack", + TrackIndices = [ 0, 1 ], + RefitPV = True, + RefPVContainerName = "BPHY13RefittedPrimaryVertices", # use existing refitted PVs + UseMassConstraint = True, + VertexMass = 9460.30, + MassInputParticles = [105.658, 105.658], + TrkVertexFitterTool = BPHY13VertexFit, + OutputVtxContainerName = "BPHY13TwoMuonHi") + +ToolSvc += BPHY13_Revertex_2muHi +print(BPHY13_Revertex_2muHi) + +BPHY13_Select_TwoMuonHi = DerivationFramework__Select_onia2mumu( + name = "BPHY13_Select_TwoMuonHi", + HypothesisName = "TwoMuonsHi", + InputVtxContainerName = "BPHY13TwoMuonHi", + TrkMasses = [105.658, 105.658], + VtxMassHypo = 9460.30, + MassMin = 8500., + MassMax = 11000., + Chi2Max = 200) + +ToolSvc += BPHY13_Select_TwoMuonHi +print(BPHY13_Select_TwoMuonHi) + +BPHY13_Revertex_2trkHi = DerivationFramework__ReVertex( + name = "BPHY13_Revertex_2trkHi", + InputVtxContainerName = "BPHY13FourTrack", + TrackIndices = [ 2, 3 ], + RefitPV = True, + RefPVContainerName = "BPHY13RefittedPrimaryVertices", # use existing refitted PVs + UseMassConstraint = True, + VertexMass = 9460.30, + MassInputParticles = [105.658, 105.658], + TrkVertexFitterTool = BPHY13VertexFit, + OutputVtxContainerName = "BPHY13TwoTrackHi") + +ToolSvc += BPHY13_Revertex_2trkHi +print(BPHY13_Revertex_2trkHi) + +BPHY13_Select_TwoTrackHi = DerivationFramework__Select_onia2mumu( + name = "BPHY13_Select_TwoTrackHi", + HypothesisName = "TwoTracksHi", + InputVtxContainerName = "BPHY13TwoTrackHi", + TrkMasses = [105.658, 105.658], + VtxMassHypo = 9460.30, + MassMin = 8500., + MassMax = 11000., + Chi2Max = 200) + +ToolSvc += BPHY13_Select_TwoTrackHi +print(BPHY13_Select_TwoTrackHi) + + +BPHY13_Revertex_2muMed = DerivationFramework__ReVertex( + name = "BPHY13_Revertex_2muMed", + InputVtxContainerName = "BPHY13FourTrack", + TrackIndices = [ 0, 1 ], + RefitPV = True, + RefPVContainerName = "BPHY13RefittedPrimaryVertices", # use existing refitted PVs + UseMassConstraint = True, + VertexMass = 3686.10, + MassInputParticles = [105.658, 105.658], + TrkVertexFitterTool = BPHY13VertexFit, + OutputVtxContainerName = "BPHY13TwoMuonMed") + +ToolSvc += BPHY13_Revertex_2muMed +print(BPHY13_Revertex_2muMed) + +BPHY13_Select_TwoMuonMed = DerivationFramework__Select_onia2mumu( + name = "BPHY13_Select_TwoMuonMed", + HypothesisName = "TwoMuonsMed", + InputVtxContainerName = "BPHY13TwoMuonMed", + TrkMasses = [105.658, 105.658], + VtxMassHypo = 3686.10, + MassMin = 3300.0, + MassMax = 4500.0, + Chi2Max = 200) + +ToolSvc += BPHY13_Select_TwoMuonMed +print(BPHY13_Select_TwoMuonMed) + +BPHY13_Revertex_2trkMed = DerivationFramework__ReVertex( + name = "BPHY13_Revertex_2trkMed", + InputVtxContainerName = "BPHY13FourTrack", + TrackIndices = [ 2, 3 ], + RefitPV = True, + RefPVContainerName = "BPHY13RefittedPrimaryVertices", # use existing refitted PVs + UseMassConstraint = True, + VertexMass = 3686.10, + MassInputParticles = [105.658, 105.658], + TrkVertexFitterTool = BPHY13VertexFit, + OutputVtxContainerName = "BPHY13TwoTrackMed") + +ToolSvc += BPHY13_Revertex_2trkMed +print(BPHY13_Revertex_2trkMed) + +BPHY13_Select_TwoTrackMed = DerivationFramework__Select_onia2mumu( + name = "BPHY13_Select_TwoTrackMed", + HypothesisName = "TwoTracksMed", + InputVtxContainerName = "BPHY13TwoTrackMed", + TrkMasses = [105.658, 105.658], + VtxMassHypo = 3686.10, + MassMin = 3300., + MassMax = 4500., + Chi2Max = 200) + +ToolSvc += BPHY13_Select_TwoTrackMed +print(BPHY13_Select_TwoTrackMed) + +#-------------------------------------------------------------------- +## 7/ select the event. We only want to keep events that contain certain vertices which passed certain selection. +## This is specified by the "SelectionExpression" property, which contains the expression in the following format: +## +## "ContainerName.passed_HypoName > count" +## +## where "ContainerName" is output container from some Reco_* tool, "HypoName" is the hypothesis name setup in some "Select_*" +## tool and "count" is the number of candidates passing the selection you want to keep. + +#expression = "count(BPHY13FourTrack.passed_FourTracks) > 0" +expression = "count(BPHY13FourTrack.passed_FourTracks) > 0 && ( count(BPHY13TwoMuon.passed_TwoMuons) + count(BPHY13TwoTrack.passed_TwoTracks) > 1 || count(BPHY13TwoMuonMed.passed_TwoMuonsMed) + count(BPHY13TwoTrackMed.passed_TwoTracksMed) > 1 || count(BPHY13TwoMuon.passed_TwoMuons) + count(BPHY13TwoTrackMed.passed_TwoTracksMed) > 1 || count(BPHY13TwoMuonMed.passed_TwoMuonsMed) + count(BPHY13TwoTrack.passed_TwoTracks) > 1 || count(BPHY13TwoMuonHi.passed_TwoMuonsHi) + count(BPHY13TwoTrackHi.passed_TwoTracksHi) > 0 )" + +from DerivationFrameworkTools.DerivationFrameworkToolsConf import DerivationFramework__xAODStringSkimmingTool +BPHY13_SelectEvent = DerivationFramework__xAODStringSkimmingTool(name = "BPHY13_SelectEvent", + expression = expression) +ToolSvc += BPHY13_SelectEvent +print(BPHY13_SelectEvent) + +#-------------------------------------------------------------------- +## 8/ track and vertex thinning. We want to remove all reconstructed secondary vertices +## which hasn't passed any of the selections defined by (Select_*) tools. +## We also want to keep only tracks which are associates with either muons or any of the +## vertices that passed the selection. Multiple thinning tools can perform the +## selection. The final thinning decision is based OR of all the decisions (by default, +## although it can be changed by the JO). + +## a) thining out vertices that didn't pass any selection and idetifying tracks associated with +## selected vertices. The "VertexContainerNames" is a list of the vertex containers, and "PassFlags" +## contains all pass flags for Select_* tools that must be satisfied. The vertex is kept is it +## satisfy any of the listed selections. + + +#==================================================================== +# CREATE THE DERIVATION KERNEL ALGORITHM AND PASS THE ABOVE TOOLS +#==================================================================== +## 9/ IMPORTANT bit. Don't forget to pass the tools to the DerivationKernel! If you don't do that, they will not be +## be executed! + + +# The name of the kernel (BPHY13Kernel in this case) must be unique to this derivation +from DerivationFrameworkCore.DerivationFrameworkCoreConf import DerivationFramework__DerivationKernel +DerivationFrameworkJob += CfgMgr.DerivationFramework__DerivationKernel( + "BPHY13Kernel", + AugmentationTools = [BPHY13_Reco_mumu, BPHY13FourTrackSelectAndWrite, BPHY13_Select_FourTrack, BPHY13TrackIsolationDecorator, BPHY13_Revertex_2mu, BPHY13_Select_TwoMuon, BPHY13_Revertex_2trk, BPHY13_Select_TwoTrack, BPHY13_Revertex_2muHi, BPHY13_Select_TwoMuonHi, BPHY13_Revertex_2trkHi, BPHY13_Select_TwoTrackHi, BPHY13_Revertex_2muMed, BPHY13_Select_TwoMuonMed, BPHY13_Revertex_2trkMed, BPHY13_Select_TwoTrackMed], + SkimmingTools = [BPHY13_SelectEvent] + ) + +#==================================================================== +# SET UP STREAM +#==================================================================== +streamName = derivationFlags.WriteDAOD_BPHY13Stream.StreamName +fileName = buildFileName( derivationFlags.WriteDAOD_BPHY13Stream ) +BPHY13Stream = MSMgr.NewPoolRootStream( streamName, fileName ) +BPHY13Stream.AcceptAlgs(["BPHY13Kernel"]) +# Special lines for thinning +# Thinning service name must match the one passed to the thinning tools +from AthenaServices.Configurables import ThinningSvc, createThinningSvc +augStream = MSMgr.GetStream( streamName ) +evtStream = augStream.GetEventStream() +svcMgr += createThinningSvc( svcName="BPHY13ThinningSvc", outStreams=[evtStream] ) + + +#==================================================================== +# Slimming +#==================================================================== + +from DerivationFrameworkCore.SlimmingHelper import SlimmingHelper +BPHY13SlimmingHelper = SlimmingHelper("BPHY13SlimmingHelper") +BPHY13_AllVariables = [] +BPHY13_StaticContent = [] + +# Needed for trigger objects +BPHY13SlimmingHelper.IncludeMuonTriggerContent = True +BPHY13SlimmingHelper.IncludeBPhysTriggerContent = True + +## primary vertices +BPHY13_AllVariables += ["PrimaryVertices"] +BPHY13_StaticContent += ["xAOD::VertexContainer#BPHY13RefittedPrimaryVertices"] +BPHY13_StaticContent += ["xAOD::VertexAuxContainer#BPHY13RefittedPrimaryVerticesAux."] + +## ID track particles +BPHY13_AllVariables += ["InDetTrackParticles"] + +## combined / extrapolated muon track particles +## (note: for tagged muons there is no extra TrackParticle collection since the ID tracks +## are store in InDetTrackParticles collection) +BPHY13_AllVariables += ["CombinedMuonTrackParticles"] +BPHY13_AllVariables += ["ExtrapolatedMuonTrackParticles"] + +## muon container +BPHY13_AllVariables += ["Muons"] + + +BPHY13_StaticContent += ["xAOD::VertexContainer#%s" % BPHY13FourTrackSelectAndWrite.OutputVtxContainerName] +BPHY13_StaticContent += ["xAOD::VertexAuxContainer#%sAux." % BPHY13FourTrackSelectAndWrite.OutputVtxContainerName] +## we have to disable vxTrackAtVertex branch since it is not xAOD compatible +BPHY13_StaticContent += ["xAOD::VertexAuxContainer#%sAux.-vxTrackAtVertex" % BPHY13FourTrackSelectAndWrite.OutputVtxContainerName] + +BPHY13_StaticContent += ["xAOD::VertexContainer#%s" % BPHY13_Revertex_2mu.OutputVtxContainerName] +BPHY13_StaticContent += ["xAOD::VertexAuxContainer#%sAux." % BPHY13_Revertex_2mu.OutputVtxContainerName] +## we have to disable vxTrackAtVertex branch since it is not xAOD compatible +BPHY13_StaticContent += ["xAOD::VertexAuxContainer#%sAux.-vxTrackAtVertex" % BPHY13_Revertex_2mu.OutputVtxContainerName] + +BPHY13_StaticContent += ["xAOD::VertexContainer#%s" % BPHY13_Revertex_2trk.OutputVtxContainerName] +BPHY13_StaticContent += ["xAOD::VertexAuxContainer#%sAux." % BPHY13_Revertex_2trk.OutputVtxContainerName] +## we have to disable vxTrackAtVertex branch since it is not xAOD compatible +BPHY13_StaticContent += ["xAOD::VertexAuxContainer#%sAux.-vxTrackAtVertex" % BPHY13_Revertex_2trk.OutputVtxContainerName] + +BPHY13_StaticContent += ["xAOD::VertexContainer#%s" % BPHY13_Revertex_2muHi.OutputVtxContainerName] +BPHY13_StaticContent += ["xAOD::VertexAuxContainer#%sAux." % BPHY13_Revertex_2muHi.OutputVtxContainerName] +## we have to disable vxTrackAtVertex branch since it is not xAOD compatible +BPHY13_StaticContent += ["xAOD::VertexAuxContainer#%sAux.-vxTrackAtVertex" % BPHY13_Revertex_2muHi.OutputVtxContainerName] + +BPHY13_StaticContent += ["xAOD::VertexContainer#%s" % BPHY13_Revertex_2trkHi.OutputVtxContainerName] +BPHY13_StaticContent += ["xAOD::VertexAuxContainer#%sAux." % BPHY13_Revertex_2trkHi.OutputVtxContainerName] +## we have to disable vxTrackAtVertex branch since it is not xAOD compatible +BPHY13_StaticContent += ["xAOD::VertexAuxContainer#%sAux.-vxTrackAtVertex" % BPHY13_Revertex_2trkHi.OutputVtxContainerName] + +BPHY13_StaticContent += ["xAOD::VertexContainer#%s" % BPHY13_Revertex_2muMed.OutputVtxContainerName] +BPHY13_StaticContent += ["xAOD::VertexAuxContainer#%sAux." % BPHY13_Revertex_2muMed.OutputVtxContainerName] +## we have to disable vxTrackAtVertex branch since it is not xAOD compatible +BPHY13_StaticContent += ["xAOD::VertexAuxContainer#%sAux.-vxTrackAtVertex" % BPHY13_Revertex_2muMed.OutputVtxContainerName] + +BPHY13_StaticContent += ["xAOD::VertexContainer#%s" % BPHY13_Revertex_2trkMed.OutputVtxContainerName] +BPHY13_StaticContent += ["xAOD::VertexAuxContainer#%sAux." % BPHY13_Revertex_2trkMed.OutputVtxContainerName] +## we have to disable vxTrackAtVertex branch since it is not xAOD compatible +BPHY13_StaticContent += ["xAOD::VertexAuxContainer#%sAux.-vxTrackAtVertex" % BPHY13_Revertex_2trkMed.OutputVtxContainerName] + + +# Truth information for MC only +if isSimulation: + BPHY13_AllVariables += ["TruthEvents","TruthParticles","TruthVertices","MuonTruthParticles"] + +BPHY13SlimmingHelper.AllVariables = BPHY13_AllVariables +BPHY13SlimmingHelper.StaticContent = BPHY13_StaticContent +BPHY13SlimmingHelper.AppendContentToStream(BPHY13Stream) + diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/BPHY14.py b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/BPHY14.py new file mode 100644 index 00000000000..411fe0b4a3c --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/BPHY14.py @@ -0,0 +1,329 @@ +#==================================================================== +# BPHY14.py +# This an example job options script showing how to set up a +# derivation of the data using the derivation framework. +# It requires the reductionConf flag BPHY14 in Reco_tf.py +#==================================================================== + +# Set up common services and job object. +# This should appear in ALL derivation job options +from DerivationFrameworkCore.DerivationFrameworkMaster import * + +isSimulation = False +if globalflags.DataSource()=='geant4': + isSimulation = True + + +print(isSimulation) + + +#==================================================================== +# AUGMENTATION TOOLS +#==================================================================== +## 1/ setup vertexing tools and services +#include( "JpsiUpsilonTools/configureServices.py" ) + +include("DerivationFrameworkBPhys/configureVertexing.py") +BPHY14_VertexTools = BPHYVertexTools("BPHY14") + +#-------------------------------------------------------------------- +## 2/ Setup the vertex fitter tools (e.g. JpsiFinder, JpsiPlus1Track, etc). +## These are general tools independent of DerivationFramework that do the +## actual vertex fitting and some pre-selection. +from JpsiUpsilonTools.JpsiUpsilonToolsConf import Analysis__JpsiFinder +BPHY14JpsiFinder = Analysis__JpsiFinder( + name = "BPHY14JpsiFinder", + OutputLevel = INFO, + muAndMu = True, + muAndTrack = False, + TrackAndTrack = False, + assumeDiMuons = True, # If true, will assume dimu hypothesis and use PDG value for mu mass + invMassUpper = 15000.0, + invMassLower = 2000., + Chi2Cut = 200., + muonThresholdPt = 2500., + oppChargesOnly = True, + atLeastOneComb = False, + combOnly = True, + useCombinedMeasurement = False, # Only takes effect if combOnly=True + muonCollectionKey = "Muons", + TrackParticleCollection = "InDetTrackParticles", + V0VertexFitterTool = BPHY14_VertexTools.TrkV0Fitter, # V0 vertex fitter + useV0Fitter = False, # if False a TrkVertexFitterTool will be used + TrkVertexFitterTool = BPHY14_VertexTools.TrkVKalVrtFitter, # VKalVrt vertex fitter + TrackSelectorTool = BPHY14_VertexTools.InDetTrackSelectorTool, + VertexPointEstimator = BPHY14_VertexTools.VtxPointEstimator, + useMCPCuts = False ) + +ToolSvc += BPHY14JpsiFinder +print(BPHY14JpsiFinder) + +#-------------------------------------------------------------------- +## 3/ setup the vertex reconstruction "call" tool(s). They are part of the derivation framework. +## These Augmentation tools add output vertex collection(s) into the StoreGate and add basic +## decorations which do not depend on the vertex mass hypothesis (e.g. lxy, ptError, etc). +## There should be one tool per topology, i.e. Jpsi and Psi(2S) do not need two instance of the +## Reco tool is the JpsiFinder mass window is wide enough. + +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__Reco_Vertex +BPHY14_Reco_mumu = DerivationFramework__Reco_Vertex( + name = "BPHY14_Reco_mumu", + VertexSearchTool = BPHY14JpsiFinder, + OutputVtxContainerName = "BPHY14OniaCandidates", + PVContainerName = "PrimaryVertices", + RefPVContainerName = "BPHY14RefittedPrimaryVertices", + RefitPV = True, + MaxPVrefit = 100000, + DoVertexType = 7) + +ToolSvc += BPHY14_Reco_mumu +print(BPHY14_Reco_mumu) + +#-------------------------------------------------------------------- +## 4/ setup the vertex selection and augmentation tool(s). These tools decorate the vertices with +## variables that depend on the vertex mass hypothesis, e.g. invariant mass, proper decay time, etc. +## Property HypothesisName is used as a prefix for these decorations. +## They also perform tighter selection, flagging the vertecis that passed. The flag is a Char_t branch +## named "passed_"+HypothesisName. It is used later by the "SelectEvent" and "Thin_vtxTrk" tools +## to determine which events and candidates should be kept in the output stream. +## Multiple instances of the Select_* tools can be used on a single input collection as long as they +## use different "HypothesisName" flags. + +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__Select_onia2mumu + +## a/ augment and select Jpsi->mumu candidates +BPHY14_Select_Jpsi2mumu = DerivationFramework__Select_onia2mumu( + name = "BPHY14_Select_Jpsi2mumu", + HypothesisName = "Jpsi", + InputVtxContainerName = "BPHY14OniaCandidates", + VtxMassHypo = 3096.916, + MassMin = 2000.0, + MassMax = 3600.0, + Chi2Max = 200, + DoVertexType = 7) + +ToolSvc += BPHY14_Select_Jpsi2mumu +print(BPHY14_Select_Jpsi2mumu) + +## b/ augment and select Psi(2S)->mumu candidates +BPHY14_Select_Psi2mumu = DerivationFramework__Select_onia2mumu( + name = "BPHY14_Select_Psi2mumu", + HypothesisName = "Psi", + InputVtxContainerName = "BPHY14OniaCandidates", + VtxMassHypo = 3686.09, + MassMin = 3300.0, + MassMax = 4500.0, + Chi2Max = 200, + DoVertexType = 7) + +ToolSvc += BPHY14_Select_Psi2mumu +print(BPHY14_Select_Psi2mumu) + +# Added by ASC +## c/ augment and select Upsilon(nS)->mumu candidates +BPHY14_Select_Upsi2mumu = DerivationFramework__Select_onia2mumu( + name = "BPHY14_Select_Upsi2mumu", + HypothesisName = "Upsi", + InputVtxContainerName = "BPHY14OniaCandidates", + VtxMassHypo = 9460.30, + MassMin = 7000.0, + MassMax = 12500.0, + Chi2Max = 200, + DoVertexType = 7) + +ToolSvc += BPHY14_Select_Upsi2mumu +print(BPHY14_Select_Upsi2mumu) + +#-------------------------------------------------------------------- +## 5/ select the event. We only want to keep events that contain certain vertices which passed certain selection. +## This is specified by the "SelectionExpression" property, which contains the expression in the following format: +## +## "ContainerName.passed_HypoName > count" +## +## where "ContainerName" is output container form some Reco_* tool, "HypoName" is the hypothesis name setup in some "Select_*" +## tool and "count" is the number of candidates passing the selection you want to keep. + +#==================================================================== +# Photon things +#==================================================================== +from DerivationFrameworkCore.DerivationFrameworkMaster import * +from DerivationFrameworkInDet.InDetCommon import * +from DerivationFrameworkMuons.MuonsCommon import * +from DerivationFrameworkJetEtMiss.JetCommon import * +from DerivationFrameworkJetEtMiss.METCommon import * +from DerivationFrameworkEGamma.EGammaCommon import * + +#photonRequirements = '(DFCommonPhotons_et >= 5*GeV) && (abs(DFCommonPhotons_eta) < 2.6)'# && (Photons.Loose)' +photonRequirements = 'DFCommonPhotons_et > 5*GeV' + + +expression = "(count(BPHY14OniaCandidates.passed_Jpsi) > 0 || count(BPHY14OniaCandidates.passed_Psi) > 0 || count(BPHY14OniaCandidates.passed_Upsi) > 0) && count("+photonRequirements+") >0" +from DerivationFrameworkTools.DerivationFrameworkToolsConf import DerivationFramework__xAODStringSkimmingTool +BPHY14_SelectEvent = DerivationFramework__xAODStringSkimmingTool(name = "BPHY14_SelectEvent", + expression = expression) +ToolSvc += BPHY14_SelectEvent +print(BPHY14_SelectEvent) + +#-------------------------------------------------------------------- +## 6/ track and vertex thinning. We want to remove all reconstructed secondary vertices +## which hasn't passed any of the selections defined by (Select_*) tools. +## We also want to keep only tracks which are associates with either muons or any of the +## vertices that passed the selection. Multiple thinning tools can perform the +## selection. The final thinning decision is based OR of all the decisions (by default, +## although it can be changed by the JO). + +## a) thining out vertices that didn't pass any selection and idetifying tracks associated with +## selected vertices. The "VertexContainerNames" is a list of the vertex containers, and "PassFlags" +## contains all pass flags for Select_* tools that must be satisfied. The vertex is kept is it +## satisfy any of the listed selections. + +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__Thin_vtxTrk +BPHY14Thin_vtxTrk = DerivationFramework__Thin_vtxTrk( + name = "BPHY14Thin_vtxTrk", + TrackParticleContainerName = "InDetTrackParticles", + VertexContainerNames = ["BPHY14OniaCandidates"], + PassFlags = ["passed_Jpsi", "passed_Psi", "passed_Upsi"] ) + +ToolSvc += BPHY14Thin_vtxTrk + +## b) thinning out tracks that are not attached to muons. The final thinning decision is based on the OR operation +## between decision from this and the previous tools. +from DerivationFrameworkInDet.DerivationFrameworkInDetConf import DerivationFramework__MuonTrackParticleThinning +BPHY14MuonTPThinningTool = DerivationFramework__MuonTrackParticleThinning(name = "BPHY14MuonTPThinningTool", + MuonKey = "Muons", + InDetTrackParticlesKey = "InDetTrackParticles") +ToolSvc += BPHY14MuonTPThinningTool + + +# Tracks associated with Photons +from DerivationFrameworkInDet.DerivationFrameworkInDetConf import DerivationFramework__EgammaTrackParticleThinning +BPHY14PhotonTPThinningTool = DerivationFramework__EgammaTrackParticleThinning(name = "BPHY14PhotonTPThinningTool", + SGKey = "Photons", + GSFTrackParticlesKey = "GSFTrackParticles", + InDetTrackParticlesKey = "InDetTrackParticles", + SelectionString = photonRequirements, + BestMatchOnly = False, + ConeSize = 0.6, + ApplyAnd = False) +ToolSvc += BPHY14PhotonTPThinningTool + + + + +# Added by ASC +# Only save truth informtion directly associated with Onia +from DerivationFrameworkMCTruth.DerivationFrameworkMCTruthConf import DerivationFramework__GenericTruthThinning +BPHY14TruthThinTool = DerivationFramework__GenericTruthThinning(name = "BPHY14TruthThinTool", + ParticleSelectionString = "TruthParticles.pdgId == 22 || TruthParticles.pdgId == 443 || TruthParticles.pdgId == 100443 || TruthParticles.pdgId == 553 || TruthParticles.pdgId == 100553 || TruthParticles.pdgId == 200553", + PreserveDescendants = True, + PreserveAncestors = True) +ToolSvc += BPHY14TruthThinTool +print(BPHY14TruthThinTool) + + +#==================================================================== +# CREATE THE DERIVATION KERNEL ALGORITHM AND PASS THE ABOVE TOOLS +#==================================================================== +## 7/ IMPORTANT bit. Don't forget to pass the tools to the DerivationKernel! If you don't do that, they will not be +## be executed! + +# Added by ASC +BPHY14ThinningTools = [BPHY14Thin_vtxTrk, BPHY14MuonTPThinningTool,BPHY14PhotonTPThinningTool] +if globalflags.DataSource()=='geant4': + BPHY14ThinningTools.append(BPHY14TruthThinTool) + +# The name of the kernel (BPHY14Kernel in this case) must be unique to this derivation +from DerivationFrameworkCore.DerivationFrameworkCoreConf import DerivationFramework__DerivationKernel +DerivationFrameworkJob += CfgMgr.DerivationFramework__DerivationKernel( + "BPHY14Kernel", + AugmentationTools = [BPHY14_Reco_mumu, BPHY14_Select_Jpsi2mumu, BPHY14_Select_Psi2mumu, BPHY14_Select_Upsi2mumu], + SkimmingTools = [BPHY14_SelectEvent], + ThinningTools = BPHY14ThinningTools + ) + +#==================================================================== +# SET UP STREAM +#==================================================================== +streamName = derivationFlags.WriteDAOD_BPHY14Stream.StreamName +fileName = buildFileName( derivationFlags.WriteDAOD_BPHY14Stream ) +BPHY14Stream = MSMgr.NewPoolRootStream( streamName, fileName ) +BPHY14Stream.AcceptAlgs(["BPHY14Kernel"]) +# Special lines for thinning +# Thinning service name must match the one passed to the thinning tools +from AthenaServices.Configurables import ThinningSvc, createThinningSvc +augStream = MSMgr.GetStream( streamName ) +evtStream = augStream.GetEventStream() +svcMgr += createThinningSvc( svcName="BPHY14ThinningSvc", outStreams=[evtStream] ) + + +#==================================================================== +# Slimming +#==================================================================== + +# Added by ASC +from DerivationFrameworkCore.SlimmingHelper import SlimmingHelper +BPHY14SlimmingHelper = SlimmingHelper("BPHY14SlimmingHelper") +BPHY14_AllVariables = [] +BPHY14_StaticContent = [] +BPHY14_SmartCollections = [] +BPHY14_ExtraVariables = [] + +# Needed for trigger objects +BPHY14SlimmingHelper.IncludeMuonTriggerContent = True +BPHY14SlimmingHelper.IncludeBPhysTriggerContent = True +BPHY14SlimmingHelper.IncludeEGammaTriggerContent = True + +## primary vertices +BPHY14_SmartCollections += ["PrimaryVertices"] +BPHY14_StaticContent += ["xAOD::VertexContainer#BPHY14RefittedPrimaryVertices"] +BPHY14_StaticContent += ["xAOD::VertexAuxContainer#BPHY14RefittedPrimaryVerticesAux."] + +## ID track particles +BPHY14_SmartCollections += ["InDetTrackParticles"] +BPHY14_ExtraVariables += ["%s.vx.vy.vz" % "InDetTrackParticles"] +#BPHY14_AllVariables += ["InDetTrackParticles"] + + +## combined / extrapolated muon track particles +## (note: for tagged muons there is no extra TrackParticle collection since the ID tracks +## are store in InDetTrackParticles collection) +BPHY14_AllVariables += ["CombinedMuonTrackParticles"] +BPHY14_AllVariables += ["ExtrapolatedMuonTrackParticles"] + +## muon container +BPHY14_SmartCollections += ["Muons"] +BPHY14_ExtraVariables += ["%s.etcone30.etcone40" % "Muons" + +".momentumBalanceSignificance" + +".scatteringCurvatureSignificance" + +".scatteringNeighbourSignificance" + +".msInnerMatchDOF.msInnerMatchChi2" + +".msOuterMatchDOF.msOuterMatchChi2" + +".EnergyLoss.ParamEnergyLoss.MeasEnergyLoss" + +".ET_Core" ] +#BPHY14_AllVariables += ["Muons"] + +## Jpsi candidates +BPHY14_StaticContent += ["xAOD::VertexContainer#%s" % BPHY14_Reco_mumu.OutputVtxContainerName] +BPHY14_StaticContent += ["xAOD::VertexAuxContainer#%sAux." % BPHY14_Reco_mumu.OutputVtxContainerName] +## we have to disable vxTrackAtVertex branch since it is not xAOD compatible +BPHY14_StaticContent += ["xAOD::VertexAuxContainer#%sAux.-vxTrackAtVertex" % BPHY14_Reco_mumu.OutputVtxContainerName] + +# Truth information for MC only +if isSimulation: + BPHY14_StaticContent += ["xAOD::TruthParticleContainer#TruthMuons","xAOD::TruthParticleAuxContainer#TruthMuonsAux."] + BPHY14_StaticContent += ["xAOD::TruthParticleContainer#TruthPhotons","xAOD::TruthParticleAuxContainer#TruthPhotonsAux."] + BPHY14_AllVariables += ["TruthEvents","TruthParticles","TruthVertices","MuonTruthParticles"] + +#Photon Information +#AllVariables += ["Photons"] +BPHY14_SmartCollections += ["Photons"] #,"Muons","InDetTrackParticles","PrimaryVertices"] +from DerivationFrameworkSM.STDMExtraContent import * +BPHY14_ExtraVariables.extend(ExtraContentPhotons) + + +BPHY14SlimmingHelper.AllVariables = BPHY14_AllVariables +BPHY14SlimmingHelper.StaticContent = BPHY14_StaticContent +BPHY14SlimmingHelper.SmartCollections = BPHY14_SmartCollections +BPHY14SlimmingHelper.ExtraVariables = BPHY14_ExtraVariables +BPHY14SlimmingHelper.AppendContentToStream(BPHY14Stream) + diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/BPHY15.py b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/BPHY15.py new file mode 100644 index 00000000000..be74b9f1394 --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/BPHY15.py @@ -0,0 +1,855 @@ +#2018/11/24 +#==================================================================== +# BPHY15.py +# Bc+>J/psiD_s+, Bc+>J/psiD+, Bc+>J/psiD*+, Bc+>J/psiD_s1+ +# It requires the reductionConf flag BPHY15 in Reco_tf.py +#==================================================================== + +# Set up common services and job object. +# This should appear in ALL derivation job options +from DerivationFrameworkCore.DerivationFrameworkMaster import * + +isSimulation = False +if globalflags.DataSource()=='geant4': + isSimulation = True + +print(isSimulation) + + +#==================================================================== +# AUGMENTATION TOOLS +#==================================================================== +## 1/ setup vertexing tools and services +include("DerivationFrameworkBPhys/configureVertexing.py") +BPHY15_VertexTools = BPHYVertexTools("BPHY15") + + +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__AugOriginalCounts +BPHY15_AugOriginalCounts = DerivationFramework__AugOriginalCounts( + name = "BPHY15_AugOriginalCounts", + VertexContainer = "PrimaryVertices", + TrackContainer = "InDetTrackParticles" ) +ToolSvc += BPHY15_AugOriginalCounts + + +#-------------------------------------------------------------------- +# 2/ Select J/psi>mu+mu- +#-------------------------------------------------------------------- +## a/ setup JpsiFinder tool +## These are general tools independent of DerivationFramework that do the +## actual vertex fitting and some pre-selection. +from JpsiUpsilonTools.JpsiUpsilonToolsConf import Analysis__JpsiFinder +BPHY15JpsiFinder = Analysis__JpsiFinder( + name = "BPHY15JpsiFinder", + OutputLevel = INFO, + muAndMu = True, + muAndTrack = False, + TrackAndTrack = False, + assumeDiMuons = True, + muonThresholdPt = 2700, + invMassUpper = 3400.0, + invMassLower = 2800.0, + Chi2Cut = 10., + oppChargesOnly = True, + allMuons = True, + combOnly = False, + atLeastOneComb = False, + useCombinedMeasurement = False, # Only takes effect if combOnly=True + muonCollectionKey = "Muons", + TrackParticleCollection = "InDetTrackParticles", + V0VertexFitterTool = BPHY15_VertexTools.TrkV0Fitter, # V0 vertex fitter + useV0Fitter = False, # if False a TrkVertexFitterTool will be used + TrkVertexFitterTool = BPHY15_VertexTools.TrkVKalVrtFitter, # VKalVrt vertex fitter + TrackSelectorTool = BPHY15_VertexTools.InDetTrackSelectorTool, + VertexPointEstimator = BPHY15_VertexTools.VtxPointEstimator, + useMCPCuts = False) + +ToolSvc += BPHY15JpsiFinder +print(BPHY15JpsiFinder) + +#-------------------------------------------------------------------- +## b/ setup the vertex reconstruction "call" tool(s). They are part of the derivation framework. +## These Augmentation tools add output vertex collection(s) into the StoreGate and add basic +## decorations which do not depend on the vertex mass hypothesis (e.g. lxy, ptError, etc). +## There should be one tool per topology, i.e. Jpsi and Psi(2S) do not need two instance of the +## Reco tool is the JpsiFinder mass window is wide enough. +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__Reco_Vertex +BPHY15JpsiSelectAndWrite = DerivationFramework__Reco_Vertex( + name = "BPHY15JpsiSelectAndWrite", + VertexSearchTool = BPHY15JpsiFinder, + OutputVtxContainerName = "BPHY15JpsiCandidates", + PVContainerName = "PrimaryVertices", + RefPVContainerName = "SHOULDNOTBEUSED", + DoVertexType = 1) + +ToolSvc += BPHY15JpsiSelectAndWrite +print(BPHY15JpsiSelectAndWrite) + +#-------------------------------------------------------------------- +## c/ augment and select Jpsi->mumu candidates +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__Select_onia2mumu +BPHY15_Select_Jpsi2mumu = DerivationFramework__Select_onia2mumu( + name = "BPHY15_Select_Jpsi2mumu", + HypothesisName = "Jpsi", + InputVtxContainerName = "BPHY15JpsiCandidates", + VtxMassHypo = 3096.900, + MassMin = 2600.0, + MassMax = 3600.0, + Chi2Max = 200, + LxyMin = 0.1, + DoVertexType = 1) + +ToolSvc += BPHY15_Select_Jpsi2mumu +print(BPHY15_Select_Jpsi2mumu) + +#-------------------------------------------------------------------- +# 3/ select B_c+->J/psi pi+ +#-------------------------------------------------------------------- +## a/ setup a new vertexing tool (necessary due to use of mass constraint) +from TrkVKalVrtFitter.TrkVKalVrtFitterConf import Trk__TrkVKalVrtFitter +BcJpsipiVertexFit = Trk__TrkVKalVrtFitter( + name = "BcJpsipiVertexFit", + Extrapolator = BPHY15_VertexTools.InDetExtrapolator, + FirstMeasuredPoint = True, + MakeExtendedVertex = True) + +ToolSvc += BcJpsipiVertexFit +print(BcJpsipiVertexFit) + +#-------------------------------------------------------------------- +## b/ setup the Jpsi+1 track finder +from JpsiUpsilonTools.JpsiUpsilonToolsConf import Analysis__JpsiPlus1Track +BPHY15BcJpsipi = Analysis__JpsiPlus1Track( + name = "BPHY15BcJpsipi", + OutputLevel = INFO, #DEBUG, + pionHypothesis = True, #False, + kaonHypothesis = False,#True, + trkThresholdPt = 2700, + trkMaxEta = 2.7, + BThresholdPt = 100.0, + BMassUpper = 6900.0, + BMassLower = 5600.0, + JpsiContainerKey = "BPHY15JpsiCandidates", + TrackParticleCollection = "InDetTrackParticles", + MuonsUsedInJpsi = "Muons", + TrkVertexFitterTool = BcJpsipiVertexFit, + TrackSelectorTool = BPHY15_VertexTools.InDetTrackSelectorTool, + UseMassConstraint = True, + Chi2Cut = 5, + TrkTrippletMassUpper = 6900, + TrkTrippletMassLower = 5600) + +ToolSvc += BPHY15BcJpsipi +print(BPHY15BcJpsipi) + +#-------------------------------------------------------------------- +## c/ setup the combined augmentation/skimming tool for the Bc+>J/psi pi+ +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__Reco_Vertex +BPHY15BcJpsipiSelectAndWrite = DerivationFramework__Reco_Vertex( + name = "BPHY15BcJpsipiSelectAndWrite", + Jpsi1PlusTrackName = BPHY15BcJpsipi, + OutputVtxContainerName = "BPHY15BcJpsipiCandidates", + PVContainerName = "PrimaryVertices", + RefPVContainerName = "BPHY15RefittedPrimaryVertices", + RefitPV = True, + MaxPVrefit = 1000) + +ToolSvc += BPHY15BcJpsipiSelectAndWrite +print(BPHY15BcJpsipiSelectAndWrite) + +#-------------------------------------------------------------------- +## d/ augment and select B_c+>Jpsi pi+ candidates +BPHY15_Select_Bc2Jpsipi = DerivationFramework__Select_onia2mumu( + name = "BPHY15_Select_Bc2Jpsipi", + HypothesisName = "Bc", + InputVtxContainerName = "BPHY15BcJpsipiCandidates", + TrkMasses = [105.658, 105.658, 139.571], + VtxMassHypo = 6274.9, + MassMin = 5600.0, + MassMax = 6900.0, + Chi2Max = 200) + +ToolSvc += BPHY15_Select_Bc2Jpsipi +print(BPHY15_Select_Bc2Jpsipi) + +#-------------------------------------------------------------------- +# 4/ select J/psi pi+ +#-------------------------------------------------------------------- +## a/ setup the Jpsi+1 track finder +BPHY15JpsipiFinder = Analysis__JpsiPlus1Track( + name = "BPHY15JpsipiFinder", + OutputLevel = INFO, #DEBUG, + pionHypothesis = True, #False, + kaonHypothesis = False,#True, + trkThresholdPt = 350.0, + trkMaxEta = 2.7, + BThresholdPt = 5000.0, + BMassUpper = 3600.0, + BMassLower = 3200.0, + TrkDeltaZ = 20., + TrkQuadrupletPt = 5000, + JpsiContainerKey = "BPHY15JpsiCandidates", + TrackParticleCollection = "InDetTrackParticles", + MuonsUsedInJpsi = "Muons", + TrkVertexFitterTool = BcJpsipiVertexFit, + TrackSelectorTool = BPHY15_VertexTools.InDetTrackSelectorTool, + UseMassConstraint = True, + Chi2Cut = 5, + TrkTrippletMassUpper = 3600, + TrkTrippletMassLower = 3200) + +ToolSvc += BPHY15JpsipiFinder +print(BPHY15JpsipiFinder) + +#-------------------------------------------------------------------- +## b/ setup the combined augmentation/skimming tool for J/psi pi+ +BPHY15JpsipiSelectAndWrite = DerivationFramework__Reco_Vertex( + name = "BPHY15JpsipiSelectAndWrite", + Jpsi1PlusTrackName = BPHY15JpsipiFinder, + OutputVtxContainerName = "BPHY15JpsipiCandidates", + PVContainerName = "PrimaryVertices", + RefPVContainerName = "SHOULDNOTBEUSED", + #RefitPV = True, + MaxPVrefit = 1000) + +ToolSvc += BPHY15JpsipiSelectAndWrite +print(BPHY15JpsipiSelectAndWrite) + +#-------------------------------------------------------------------- +## c/ augment and select Jpsi pi+ candidates for the J/psi D*+ and J/psi D_s1+ modes +BPHY15_Select_Jpsipi = DerivationFramework__Select_onia2mumu( + name = "BPHY15_Select_Jpsipi", + HypothesisName = "Jpsipi", + TrkMasses = [105.658, 105.658, 139.571], + InputVtxContainerName = "BPHY15JpsipiCandidates", + VtxMassHypo = 3396.900, + MassMin = 3200.0, + MassMax = 3600.0, + Chi2Max = 200, + LxyMin = 0.1, + DoVertexType = 1) + +ToolSvc += BPHY15_Select_Jpsipi +print(BPHY15_Select_Jpsipi) + +#-------------------------------------------------------------------- +# 5/ Select K+K-, pi+K- and K+pi- +#-------------------------------------------------------------------- +## a/ Setup the vertex fitter tools +BPHY15DiTrkFinder = Analysis__JpsiFinder( + name = "BPHY15DiTrkFinder", + OutputLevel = INFO, + muAndMu = False, + muAndTrack = False, + TrackAndTrack = True, + assumeDiMuons = False, # If true, will assume dimu hypothesis and use PDG value for mu mass + trackThresholdPt = 900, + invMassUpper = 1900.0, + invMassLower = 280.0, + Chi2Cut = 10., + oppChargesOnly = True, + atLeastOneComb = False, + useCombinedMeasurement = False, # Only takes effect if combOnly=True + muonCollectionKey = "Muons", + TrackParticleCollection = "InDetTrackParticles", + V0VertexFitterTool = BPHY15_VertexTools.TrkV0Fitter, # V0 vertex fitter + useV0Fitter = False, # if False a TrkVertexFitterTool will be used + TrkVertexFitterTool = BPHY15_VertexTools.TrkVKalVrtFitter, # VKalVrt vertex fitter + TrackSelectorTool = BPHY15_VertexTools.InDetTrackSelectorTool, + VertexPointEstimator = BPHY15_VertexTools.VtxPointEstimator, + useMCPCuts = False, + track1Mass = 139.571, # Not very important, only used to calculate inv. mass cut, leave it loose here + track2Mass = 139.571) + +ToolSvc += BPHY15DiTrkFinder +print(BPHY15DiTrkFinder) + +#-------------------------------------------------------------------- +## b/ setup the vertex reconstruction "call" tool(s). +BPHY15DiTrkSelectAndWrite = DerivationFramework__Reco_Vertex( + name = "BPHY15DiTrkSelectAndWrite", + VertexSearchTool = BPHY15DiTrkFinder, + OutputVtxContainerName = "BPHY15DiTrkCandidates", + PVContainerName = "PrimaryVertices", + RefPVContainerName = "SHOULDNOTBEUSED", + CheckCollections = True, + CheckVertexContainers = ['BPHY15JpsiCandidates'], + DoVertexType = 1) + +ToolSvc += BPHY15DiTrkSelectAndWrite +print(BPHY15DiTrkSelectAndWrite) + +#-------------------------------------------------------------------- +## c/ augment and select D0 candidates +BPHY15_Select_D0 = DerivationFramework__Select_onia2mumu( + name = "BPHY15_Select_D0", + HypothesisName = "D0", + InputVtxContainerName = "BPHY15DiTrkCandidates", + TrkMasses = [139.571, 493.677], + VtxMassHypo = 1864.83, + MassMin = 1864.83-170, + MassMax = 1864.83+170, + LxyMin = 0.15, + Chi2Max = 200) + +ToolSvc += BPHY15_Select_D0 +print(BPHY15_Select_D0) + +#-------------------------------------------------------------------- +## d/ augment and select D0bar candidates +BPHY15_Select_D0b = DerivationFramework__Select_onia2mumu( + name = "BPHY15_Select_D0b", + HypothesisName = "D0b", + InputVtxContainerName = "BPHY15DiTrkCandidates", + TrkMasses = [493.677, 139.571], + VtxMassHypo = 1864.83, + MassMin = 1864.83-170, + MassMax = 1864.83+170, + LxyMin = 0.15, + Chi2Max = 200) + +ToolSvc += BPHY15_Select_D0b +print(BPHY15_Select_D0b) + +#-------------------------------------------------------------------- +# 6/ select D_s+>K+K-pi+ and D+>K+pi-pi- candidates +#-------------------------------------------------------------------- +## a/ setup a new vertexing tool (necessary due to use of mass constraint) +Dh3VertexFit = Trk__TrkVKalVrtFitter( + name = "Dh3VertexFit", + Extrapolator = BPHY15_VertexTools.InDetExtrapolator, + FirstMeasuredPoint = True, + MakeExtendedVertex = True) + +ToolSvc += Dh3VertexFit +print(Dh3VertexFit) + +#-------------------------------------------------------------------- +## b/ setup the Jpsi+1 track finder +BPHY15Dh3Finder = Analysis__JpsiPlus1Track( + name = "BPHY15Dh3Finder", + OutputLevel = INFO, + pionHypothesis = True, + kaonHypothesis = False, + trkThresholdPt = 900.0, + trkMaxEta = 2.7, # is this value fine?? default would be 102.5 + BThresholdPt = 2000.0, + #BThresholdPt = 3000.0, + BMassUpper = 1800.0, # What is this?? + BMassLower = 500.0, + TrkDeltaZ = 20., + TrkTrippletMassUpper = 1800, + TrkTrippletMassLower = 500, + TrkQuadrupletPt = 2000, + #TrkQuadrupletPt = 3000, + JpsiContainerKey = "BPHY15DiTrkCandidates", + TrackParticleCollection = "InDetTrackParticles", + MuonsUsedInJpsi = "NONE", # ? + ExcludeCrossJpsiTracks = False, + TrkVertexFitterTool = Dh3VertexFit, + TrackSelectorTool = BPHY15_VertexTools.InDetTrackSelectorTool, + UseMassConstraint = False, + Chi2Cut = 5) #Cut on chi2/Ndeg_of_freedom + +ToolSvc += BPHY15Dh3Finder +print(BPHY15Dh3Finder) + +#-------------------------------------------------------------------- +## c/ setup the combined augmentation/skimming tool for the D(s)+ +BPHY15Dh3SelectAndWrite = DerivationFramework__Reco_Vertex( + name = "BPHY15Dh3SelectAndWrite", + OutputLevel = INFO, + Jpsi1PlusTrackName = BPHY15Dh3Finder, + OutputVtxContainerName = "BPHY15Dh3Candidates", + PVContainerName = "PrimaryVertices", + RefPVContainerName = "SHOULDNOTBEUSED", + MaxPVrefit = 1000) + +ToolSvc += BPHY15Dh3SelectAndWrite +print(BPHY15Dh3SelectAndWrite) + + +#-------------------------------------------------------------------- +## d/ augment and select D_s+/- candidates +BPHY15_Select_Ds = DerivationFramework__Select_onia2mumu( + name = "BPHY15_Select_Ds", + HypothesisName = "Ds", + TrkMasses = [493.677, 493.677, 139.571], + InputVtxContainerName = "BPHY15Dh3Candidates", + VtxMassHypo = 1968.28, + MassMin = 1968.28-200, + MassMax = 1968.28+200, + Chi2Max = 200, + LxyMin = 0.1, + DoVertexType = 1) + +ToolSvc += BPHY15_Select_Ds +print(BPHY15_Select_Ds) + +#-------------------------------------------------------------------- +## e/ augment and select D+ candidates +BPHY15_Select_Dp = DerivationFramework__Select_onia2mumu( + name = "BPHY15_Select_Dp", + HypothesisName = "Dp", + TrkMasses = [139.571, 493.677, 139.571], + InputVtxContainerName = "BPHY15Dh3Candidates", + VtxMassHypo = 1869.59, + MassMin = 1869.59-200, + MassMax = 1869.59+200, + Chi2Max = 200, + LxyMin = 0.1, + DoVertexType = 1) + +ToolSvc += BPHY15_Select_Dp +print(BPHY15_Select_Dp) + +#-------------------------------------------------------------------- +## c/ augment and select D- candidates +BPHY15_Select_Dm = DerivationFramework__Select_onia2mumu( + name = "BPHY15_Select_Dm", + HypothesisName = "Dm", + TrkMasses = [493.677, 139.571, 139.571], + InputVtxContainerName = "BPHY15Dh3Candidates", + VtxMassHypo = 1869.59, + MassMin = 1869.59-200, + MassMax = 1869.59+200, + Chi2Max = 200, + LxyMin = 0.1, + DoVertexType = 1) + +ToolSvc += BPHY15_Select_Dm +print(BPHY15_Select_Dm) + + +#-------------------------------------------------------------------- +# 7/ select Bc+>J/psi D_(s)+/- +#-------------------------------------------------------------------- +## a/ setup the cascade vertexing tool +BcJpsiDxVertexFit = Trk__TrkVKalVrtFitter( + name = "BcJpsiDxVertexFit", + Extrapolator = BPHY15_VertexTools.InDetExtrapolator, + #FirstMeasuredPoint = True, + FirstMeasuredPoint = False, + CascadeCnstPrecision = 1e-6, + MakeExtendedVertex = True) + +ToolSvc += BcJpsiDxVertexFit +print(BcJpsiDxVertexFit) + +#-------------------------------------------------------------------- +## b/ setup the Jpsi Ds finder +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__JpsiPlusDsCascade +BPHY15JpsiDs = DerivationFramework__JpsiPlusDsCascade( + name = "BPHY15JpsiDs", + HypothesisName = "Bc", + TrkVertexFitterTool = BcJpsiDxVertexFit, + DxHypothesis = 431, + ApplyDxMassConstraint = True, + ApplyJpsiMassConstraint = True, + JpsiMassLowerCut = 2600., + JpsiMassUpperCut = 3600., + DxMassLowerCut = 1968.28 - 200., + DxMassUpperCut = 1968.28 + 200., + MassLowerCut = 6274.90 - 600., + MassUpperCut = 6274.90 + 600., + Chi2Cut = 10, + RefitPV = True, + RefPVContainerName = "BPHY15RefittedPrimaryVertices", + JpsiVertices = "BPHY15JpsiCandidates", + CascadeVertexCollections = ["BcJpsiDsCascadeSV2", "BcJpsiDsCascadeSV1"], + DxVertices = "BPHY15Dh3Candidates") + +ToolSvc += BPHY15JpsiDs +print(BPHY15JpsiDs) + +#-------------------------------------------------------------------- +## c/ setup the Jpsi D+ finder +BPHY15JpsiDp = DerivationFramework__JpsiPlusDsCascade( + name = "BPHY15JpsiDp", + HypothesisName = "Bc", + TrkVertexFitterTool = BcJpsiDxVertexFit, + DxHypothesis = 411, + ApplyDxMassConstraint = True, + ApplyJpsiMassConstraint = True, + JpsiMassLowerCut = 2600., + JpsiMassUpperCut = 3600., + DxMassLowerCut = 1869.59 - 180., + DxMassUpperCut = 1869.59 + 180., + MassLowerCut = 6274.90 - 600., + MassUpperCut = 6274.90 + 600., + Chi2Cut = 10, + RefitPV = True, + RefPVContainerName = "BPHY15RefittedPrimaryVertices", + JpsiVertices = "BPHY15JpsiCandidates", + CascadeVertexCollections = ["BcJpsiDpCascadeSV2", "BcJpsiDpCascadeSV1"], + DxVertices = "BPHY15Dh3Candidates") + +ToolSvc += BPHY15JpsiDp +print(BPHY15JpsiDp) + +#-------------------------------------------------------------------- +# 8/ select Bc+>J/psi D*+/- +#-------------------------------------------------------------------- +## a/ setup the cascade vertexing tool +BcJpsiDstVertexFit = Trk__TrkVKalVrtFitter( + name = "BcJpsiDstVertexFit", + Extrapolator = BPHY15_VertexTools.InDetExtrapolator, + #FirstMeasuredPoint = True, + FirstMeasuredPoint = False, + CascadeCnstPrecision = 1e-6, + MakeExtendedVertex = True) + +ToolSvc += BcJpsiDstVertexFit +print(BcJpsiDstVertexFit) + +#-------------------------------------------------------------------- +## b/ setup Jpsi D*+ finder +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__JpsiPlusDpstCascade +BPHY15JpsiDpst = DerivationFramework__JpsiPlusDpstCascade( + name = "BPHY15JpsiDpst", + HypothesisName = "Bc", + TrkVertexFitterTool = BcJpsiDstVertexFit, + DxHypothesis = 421, + ApplyD0MassConstraint = True, + ApplyJpsiMassConstraint = True, + JpsiMassLowerCut = 2600., + JpsiMassUpperCut = 3600., + JpsipiMassLowerCut = 2600., + JpsipiMassUpperCut = 6800., + D0MassLowerCut = 1864.83 - 200., + D0MassUpperCut = 1864.83 + 200., + DstMassLowerCut = 2010.26 - 300., + DstMassUpperCut = 2010.26 + 300., + MassLowerCut = 5400, + MassUpperCut = 6274.90 + 600., + Chi2Cut = 10, + RefitPV = True, + RefPVContainerName = "BPHY15RefittedPrimaryVertices", + JpsipiVertices = "BPHY15JpsipiCandidates", + CascadeVertexCollections = ["BcJpsiDpstCascadeSV2", "BcJpsiDpstCascadeSV1"], + D0Vertices = "BPHY15DiTrkCandidates") + +ToolSvc += BPHY15JpsiDpst +print(BPHY15JpsiDpst) + + + +#-------------------------------------------------------------------- +# 9/ select K_S0>pi+pi- +#-------------------------------------------------------------------- + +include("DerivationFrameworkBPhys/configureV0Finder.py") +BPHY15_K0FinderTools = BPHYV0FinderTools("BPHY15") +print(BPHY15_K0FinderTools) + +## a/ Setup the vertex fitter tools +BPHY15K0Finder = Analysis__JpsiFinder( + name = "BPHY15K0Finder", + OutputLevel = INFO, + muAndMu = False, + muAndTrack = False, + TrackAndTrack = True, + assumeDiMuons = False, # If true, will assume dimu hypothesis and use PDG value for mu mass + trackThresholdPt = 400, + #trackThresholdPt = 500, + invMassUpper = 600.0, + invMassLower = 400.0, + Chi2Cut = 20, + #Chi2Cut = 5., + oppChargesOnly = True, + atLeastOneComb = False, + useCombinedMeasurement = False, # Only takes effect if combOnly=True + muonCollectionKey = "Muons", + TrackParticleCollection = "InDetTrackParticles", + V0VertexFitterTool = BPHY15_VertexTools.TrkV0Fitter, # V0 vertex fitter + useV0Fitter = True, # if False a TrkVertexFitterTool will be used + #useV0Fitter = False, # if False a TrkVertexFitterTool will be used + TrkVertexFitterTool = BPHY15_VertexTools.TrkVKalVrtFitter, # VKalVrt vertex fitter + #TrackSelectorTool = BPHY15_VertexTools.InDetTrackSelectorTool, + TrackSelectorTool = BPHY15_K0FinderTools.InDetV0VxTrackSelector, + VertexPointEstimator = BPHY15_K0FinderTools.V0VtxPointEstimator, + #VertexPointEstimator = BPHY15_VertexTools.VtxPointEstimator, + useMCPCuts = False, + track1Mass = 139.571, # Not very important, only used to calculate inv. mass cut, leave it loose here + track2Mass = 139.571) + +ToolSvc += BPHY15K0Finder +print(BPHY15K0Finder) + +#-------------------------------------------------------------------- +## b/ setup the vertex reconstruction "call" tool(s). +BPHY15K0SelectAndWrite = DerivationFramework__Reco_Vertex( + name = "BPHY15K0SelectAndWrite", + VertexSearchTool = BPHY15K0Finder, + OutputVtxContainerName = "BPHY15K0Candidates", + PVContainerName = "PrimaryVertices", + RefPVContainerName = "SHOULDNOTBEUSED", + CheckCollections = True, + CheckVertexContainers = ['BPHY15JpsipiCandidates','BPHY15DiTrkCandidates','BcJpsiDpstCascadeSV1'], + DoVertexType = 1) + +ToolSvc += BPHY15K0SelectAndWrite +print(BPHY15K0SelectAndWrite) + +#-------------------------------------------------------------------- +## c/ augment and select K_S0 candidates +BPHY15_Select_K0 = DerivationFramework__Select_onia2mumu( + name = "BPHY15_Select_K0", + HypothesisName = "K0", + InputVtxContainerName = "BPHY15K0Candidates", + TrkMasses = [139.571, 139.571], + VtxMassHypo = 497.672, + MassMin = 400, + MassMax = 600, + LxyMin = 0.2, + Chi2Max = 200) + +ToolSvc += BPHY15_Select_K0 +print(BPHY15_Select_K0) + +#-------------------------------------------------------------------- +# 10/ select Bc+>J/psi D_s1+/- +#-------------------------------------------------------------------- +## a/ setup the cascade vertexing tool +BcJpsiDs1VertexFit = Trk__TrkVKalVrtFitter( + name = "BcJpsiDs1VertexFit", + Extrapolator = BPHY15_VertexTools.InDetExtrapolator, + #FirstMeasuredPoint = True, + FirstMeasuredPoint = False, + CascadeCnstPrecision = 1e-6, + MakeExtendedVertex = True) + +ToolSvc += BcJpsiDs1VertexFit +print(BcJpsiDs1VertexFit) + +#-------------------------------------------------------------------- +## b/ setup Jpsi D_s1+ finder +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__JpsiPlusDs1Cascade +BPHY15JpsiDps1 = DerivationFramework__JpsiPlusDs1Cascade( + name = "BPHY15JpsiDps1", + HypothesisName = "Bc", + TrkVertexFitterTool = BcJpsiDs1VertexFit, + DxHypothesis = 421, + ApplyD0MassConstraint = True, + ApplyK0MassConstraint = True, + ApplyJpsiMassConstraint = True, + JpsiMassLowerCut = 2600., + JpsiMassUpperCut = 3600., + JpsipiMassLowerCut = 2600., + JpsipiMassUpperCut = 6800., + D0MassLowerCut = 1864.83 - 180., + D0MassUpperCut = 1864.83 + 180., + K0MassLowerCut = 400., + K0MassUpperCut = 600., + DstMassLowerCut = 2010.26 - 300., + DstMassUpperCut = 2010.26 + 300., + MassLowerCut = 6274.90 - 600, + MassUpperCut = 6274.90 + 600., + Chi2Cut = 10, + RefitPV = True, + RefPVContainerName = "BPHY15RefittedPrimaryVertices", + JpsipiVertices = "BPHY15JpsipiCandidates", + CascadeVertexCollections = ["BcJpsiDps1CascadeSV3", "BcJpsiDps1CascadeSV2", "BcJpsiDps1CascadeSV1"], + K0Vertices = "BPHY15K0Candidates", + D0Vertices = "BPHY15DiTrkCandidates") + +ToolSvc += BPHY15JpsiDps1 +print(BPHY15JpsiDps1) + +#-------------------------------------------------------------------- + +CascadeCollections = [] + +CascadeCollections += BPHY15JpsiDs.CascadeVertexCollections +CascadeCollections += BPHY15JpsiDp.CascadeVertexCollections + +CascadeCollections += BPHY15JpsiDpst.CascadeVertexCollections +CascadeCollections += BPHY15JpsiDps1.CascadeVertexCollections + +#-------------------------------------------------------------------- + + +if not isSimulation: #Only Skim Data + from DerivationFrameworkTools.DerivationFrameworkToolsConf import DerivationFramework__xAODStringSkimmingTool + BPHY15_SelectBcJpsipiEvent = DerivationFramework__xAODStringSkimmingTool( + name = "BPHY15_SelectBcJpsipiEvent", + expression = "( count(BPHY15BcJpsipiCandidates.passed_Bc > 0) + count(BcJpsiDsCascadeSV1.x > -999) + count(BcJpsiDpCascadeSV1.x > -999) + count(BcJpsiDpstCascadeSV1.x > -999) + count(BcJpsiDps1CascadeSV1.x > -999) ) > 0") + + ToolSvc += BPHY15_SelectBcJpsipiEvent + print(BPHY15_SelectBcJpsipiEvent) + + #==================================================================== + # Make event selection based on an OR of the input skimming tools + #==================================================================== + + from DerivationFrameworkTools.DerivationFrameworkToolsConf import DerivationFramework__FilterCombinationOR + BPHY15SkimmingOR = CfgMgr.DerivationFramework__FilterCombinationOR( + "BPHY15SkimmingOR", + FilterList = [BPHY15_SelectBcJpsipiEvent] ) + ToolSvc += BPHY15SkimmingOR + print(BPHY15SkimmingOR) + +#-------------------------------------------------------------------- +##10/ track and vertex thinning. We want to remove all reconstructed secondary vertices +## which hasn't passed any of the selections defined by (Select_*) tools. +## We also want to keep only tracks which are associates with either muons or any of the +## vertices that passed the selection. Multiple thinning tools can perform the +## selection. The final thinning decision is based OR of all the decisions (by default, +## although it can be changed by the JO). + +## a) thining out vertices that didn't pass any selection and idetifying tracks associated with +## selected vertices. The "VertexContainerNames" is a list of the vertex containers, and "PassFlags" +## contains all pass flags for Select_* tools that must be satisfied. The vertex is kept is it +## satisfy any of the listed selections. +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__Thin_vtxTrk +BPHY15_thinningTool_Tracks = DerivationFramework__Thin_vtxTrk( + name = "BPHY15_thinningTool_Tracks", + TrackParticleContainerName = "InDetTrackParticles", + VertexContainerNames = ["BPHY15BcJpsipiCandidates", "BcJpsiDsCascadeSV1", "BcJpsiDsCascadeSV2", "BcJpsiDpCascadeSV1", "BcJpsiDpCascadeSV2", "BcJpsiDpstCascadeSV1", "BcJpsiDpstCascadeSV2", "BcJpsiDps1CascadeSV1", "BcJpsiDps1CascadeSV2", "BcJpsiDps1CascadeSV3"], + PassFlags = ["passed_Bc"]) + +ToolSvc += BPHY15_thinningTool_Tracks +print(BPHY15_thinningTool_Tracks) + +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__BPhysPVThinningTool +BPHY15_thinningTool_PV = DerivationFramework__BPhysPVThinningTool( + name = "BPHY15_thinningTool_PV", + CandidateCollections = ["BPHY15BcJpsipiCandidates", "BcJpsiDsCascadeSV1", "BcJpsiDsCascadeSV2", "BcJpsiDpCascadeSV1", "BcJpsiDpCascadeSV2", "BcJpsiDpstCascadeSV1", "BcJpsiDpstCascadeSV2", "BcJpsiDps1CascadeSV1", "BcJpsiDps1CascadeSV2", "BcJpsiDps1CascadeSV3"], + KeepPVTracks = True) + +ToolSvc += BPHY15_thinningTool_PV +print(BPHY15_thinningTool_PV) + +## b) thinning out tracks that are not attached to muons. The final thinning decision is based on the OR operation +## between decision from this and the previous tools. +from DerivationFrameworkInDet.DerivationFrameworkInDetConf import DerivationFramework__MuonTrackParticleThinning +BPHY15MuonTPThinningTool = DerivationFramework__MuonTrackParticleThinning( + name = "BPHY15MuonTPThinningTool", + MuonKey = "Muons", + InDetTrackParticlesKey = "InDetTrackParticles") + +ToolSvc += BPHY15MuonTPThinningTool +print(BPHY15MuonTPThinningTool) + +#==================================================================== +# CREATE THE DERIVATION KERNEL ALGORITHM AND PASS THE ABOVE TOOLS +#==================================================================== + +thiningCollection = [] + +print(thiningCollection) + +# The name of the kernel (BPHY15Kernel in this case) must be unique to this derivation +from DerivationFrameworkCore.DerivationFrameworkCoreConf import DerivationFramework__DerivationKernel +DerivationFrameworkJob += CfgMgr.DerivationFramework__DerivationKernel( + "BPHY15Kernel", + AugmentationTools = [BPHY15JpsiSelectAndWrite, BPHY15_Select_Jpsi2mumu, + BPHY15BcJpsipiSelectAndWrite, BPHY15_Select_Bc2Jpsipi, + BPHY15JpsipiSelectAndWrite, BPHY15_Select_Jpsipi, + BPHY15DiTrkSelectAndWrite, BPHY15_Select_D0, BPHY15_Select_D0b, + BPHY15Dh3SelectAndWrite, BPHY15_Select_Ds, BPHY15_Select_Dp, BPHY15_Select_Dm, + BPHY15JpsiDs, + BPHY15JpsiDp, + BPHY15JpsiDpst, + BPHY15K0SelectAndWrite, BPHY15_Select_K0, + BPHY15JpsiDps1, + BPHY15_AugOriginalCounts], + #Only skim if not MC + SkimmingTools = [BPHY15SkimmingOR] if not isSimulation else [], + ThinningTools = thiningCollection + ) + +#==================================================================== +# SET UP STREAM +#==================================================================== +streamName = derivationFlags.WriteDAOD_BPHY15Stream.StreamName +fileName = buildFileName( derivationFlags.WriteDAOD_BPHY15Stream ) +BPHY15Stream = MSMgr.NewPoolRootStream( streamName, fileName ) +BPHY15Stream.AcceptAlgs(["BPHY15Kernel"]) + +# Special lines for thinning +# Thinning service name must match the one passed to the thinning tools +from AthenaServices.Configurables import ThinningSvc, createThinningSvc +augStream = MSMgr.GetStream( streamName ) +evtStream = augStream.GetEventStream() + +BPHY15ThinningSvc = createThinningSvc( svcName="BPHY15ThinningSvc", outStreams=[evtStream] ) +svcMgr += BPHY15ThinningSvc + +#==================================================================== +# Slimming +#==================================================================== + +# Added by ASC +from DerivationFrameworkCore.SlimmingHelper import SlimmingHelper +BPHY15SlimmingHelper = SlimmingHelper("BPHY15SlimmingHelper") +AllVariables = [] +StaticContent = [] + +# Needed for trigger objects +BPHY15SlimmingHelper.IncludeMuonTriggerContent = TRUE +BPHY15SlimmingHelper.IncludeBPhysTriggerContent = TRUE + +## primary vertices +AllVariables += ["PrimaryVertices"] +StaticContent += ["xAOD::VertexContainer#BPHY15RefittedPrimaryVertices"] +StaticContent += ["xAOD::VertexAuxContainer#BPHY15RefittedPrimaryVerticesAux."] + +## ID track particles +AllVariables += ["InDetTrackParticles"] + +## combined / extrapolated muon track particles +## (note: for tagged muons there is no extra TrackParticle collection since the ID tracks +## are store in InDetTrackParticles collection) +AllVariables += ["CombinedMuonTrackParticles"] +AllVariables += ["ExtrapolatedMuonTrackParticles"] + +## muon container +AllVariables += ["Muons"] + + +## Jpsi candidates +StaticContent += ["xAOD::VertexContainer#%s" % BPHY15JpsiSelectAndWrite.OutputVtxContainerName] +## we have to disable vxTrackAtVertex branch since it is not xAOD compatible +StaticContent += ["xAOD::VertexAuxContainer#%sAux.-vxTrackAtVertex" % BPHY15JpsiSelectAndWrite.OutputVtxContainerName] + +## Bc+>J/psi pi+ candidates +StaticContent += ["xAOD::VertexContainer#%s" % BPHY15BcJpsipiSelectAndWrite.OutputVtxContainerName] +StaticContent += ["xAOD::VertexAuxContainer#%sAux.-vxTrackAtVertex" % BPHY15BcJpsipiSelectAndWrite.OutputVtxContainerName] + +## K+K-, Kpi, D0/D0bar candidates +#StaticContent += ["xAOD::VertexContainer#%s" % BPHY15DiTrkSelectAndWrite.OutputVtxContainerName] +#StaticContent += ["xAOD::VertexAuxContainer#%sAux.-vxTrackAtVertex" % BPHY15DiTrkSelectAndWrite.OutputVtxContainerName] + +## D_(s)+/- candidates +#StaticContent += ["xAOD::VertexContainer#%s" % BPHY15Dh3SelectAndWrite.OutputVtxContainerName] +#StaticContent += ["xAOD::VertexAuxContainer#%sAux.-vxTrackAtVertex" % BPHY15Dh3SelectAndWrite.OutputVtxContainerName] + +## Jpsi pi+ candidates +#StaticContent += ["xAOD::VertexContainer#%s" % BPHY15JpsipiSelectAndWrite.OutputVtxContainerName] +#StaticContent += ["xAOD::VertexAuxContainer#%sAux.-vxTrackAtVertex" % BPHY15JpsipiSelectAndWrite.OutputVtxContainerName] + +## K_S0 candidates +#StaticContent += ["xAOD::VertexContainer#%s" % BPHY15K0SelectAndWrite.OutputVtxContainerName] +#StaticContent += ["xAOD::VertexAuxContainer#%sAux.-vxTrackAtVertex" % BPHY15K0SelectAndWrite.OutputVtxContainerName] + +## Bc+>J/psi D_(s)+/-, J/psi D*+/- and J/psi D_s1+/- candidates +for cascades in CascadeCollections: + StaticContent += ["xAOD::VertexContainer#%s" % cascades] + StaticContent += ["xAOD::VertexAuxContainer#%sAux.-vxTrackAtVertex" % cascades] + +# Tagging information (in addition to that already requested by usual algorithms) +AllVariables += ["GSFTrackParticles", "MuonSpectrometerTrackParticles" ] + +# Added by ASC +# Truth information for MC only +if isSimulation: + AllVariables += ["TruthEvents","TruthParticles","TruthVertices","MuonTruthParticles"] + +AllVariables = list(set(AllVariables)) # remove duplicates + +BPHY15SlimmingHelper.AllVariables = AllVariables +BPHY15SlimmingHelper.StaticContent = StaticContent +BPHY15SlimmingHelper.SmartCollections = [] + +BPHY15SlimmingHelper.AppendContentToStream(BPHY15Stream) diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/BPHY16.py b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/BPHY16.py new file mode 100644 index 00000000000..44d8383d943 --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/BPHY16.py @@ -0,0 +1,288 @@ +#==================================================================== +# BPHY16.py +#==================================================================== + +# Set up common services and job object. +# This should appear in ALL derivation job options +from DerivationFrameworkCore.DerivationFrameworkMaster import * + +isSimulation = False +if globalflags.DataSource()=='geant4': + isSimulation = True + +print(isSimulation) + +#==================================================================== +# AUGMENTATION TOOLS +#==================================================================== +## 1/ setup vertexing tools and services +include("DerivationFrameworkBPhys/configureVertexing.py") +BPHY16_VertexTools = BPHYVertexTools("BPHY16") + +#-------------------------------------------------------------------- +## 2/ Setup the vertex fitter tools (e.g. JpsiFinder, JpsiPlus1Track, etc). +## These are general tools independent of DerivationFramework that do the +## actual vertex fitting and some pre-selection. +from JpsiUpsilonTools.JpsiUpsilonToolsConf import Analysis__JpsiFinder +BPHY16JpsiFinder = Analysis__JpsiFinder( + name = "BPHY16JpsiFinder", + OutputLevel = INFO, + muAndMu = True, + muAndTrack = False, + TrackAndTrack = False, + assumeDiMuons = True, # If true, will assume dimu hypothesis and use PDG value for mu mass + invMassUpper = 12000.0, + invMassLower = 8000., + Chi2Cut = 20., + oppChargesOnly = True, + atLeastOneComb = True, + useCombinedMeasurement = False, # Only takes effect if combOnly=True + muonCollectionKey = "Muons", + TrackParticleCollection = "InDetTrackParticles", + V0VertexFitterTool = BPHY16_VertexTools.TrkV0Fitter, # V0 vertex fitter + useV0Fitter = False, # if False a TrkVertexFitterTool will be used + TrkVertexFitterTool = BPHY16_VertexTools.TrkVKalVrtFitter, # VKalVrt vertex fitter + TrackSelectorTool = BPHY16_VertexTools.InDetTrackSelectorTool, + VertexPointEstimator = BPHY16_VertexTools.VtxPointEstimator, + useMCPCuts = False ) + +ToolSvc += BPHY16JpsiFinder +print(BPHY16JpsiFinder) + +#-------------------------------------------------------------------- +## 3/ setup the vertex reconstruction "call" tool(s). They are part of the derivation framework. +## These Augmentation tools add output vertex collection(s) into the StoreGate and add basic +## decorations which do not depend on the vertex mass hypothesis (e.g. lxy, ptError, etc). +## There should be one tool per topology, i.e. Jpsi and Psi(2S) do not need two instance of the +## Reco tool is the JpsiFinder mass window is wide enough. + +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__Reco_Vertex +BPHY16_Reco_mumu = DerivationFramework__Reco_Vertex( + name = "BPHY16_Reco_mumu", + VertexSearchTool = BPHY16JpsiFinder, + OutputVtxContainerName = "BPHY16OniaCandidates", + PVContainerName = "PrimaryVertices", + RefPVContainerName = "BPHY16RefittedPrimaryVertices", + RefitPV = True, + MaxPVrefit = 100000, + DoVertexType = 7) + +ToolSvc += BPHY16_Reco_mumu +print(BPHY16_Reco_mumu) + + +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__Select_onia2mumu + +## a/ augment and select Jpsi->mumu candidates +BPHY16_Select_Upsi = DerivationFramework__Select_onia2mumu( + name = "BPHY16_Select_Upsi", + HypothesisName = "Upsilon", + InputVtxContainerName = "BPHY16OniaCandidates", + VtxMassHypo = 9460.30, + MassMin = 8000., + MassMax = 12000., + Chi2Max = 200, + DoVertexType = 7) + +ToolSvc += BPHY16_Select_Upsi +print(BPHY16_Select_Upsi) + + +## 4/ setup a new vertexing tool (necessary due to use of mass constraint) +from TrkVKalVrtFitter.TrkVKalVrtFitterConf import Trk__TrkVKalVrtFitter +BPHY16VertexFit = Trk__TrkVKalVrtFitter( + name = "BPHY16VertexFit", + Extrapolator = BPHY16_VertexTools.InDetExtrapolator, + FirstMeasuredPoint = True, + MakeExtendedVertex = True) +ToolSvc += BPHY16VertexFit +print(BPHY16VertexFit) + +## 5/ setup the Jpsi+2 track finder +from JpsiUpsilonTools.JpsiUpsilonToolsConf import Analysis__JpsiPlus2Tracks +BPHY16Plus2Tracks = Analysis__JpsiPlus2Tracks(name = "BPHY16Plus2Tracks", +# OutputLevel = DEBUG, +kaonkaonHypothesis = False, +pionpionHypothesis = False, +kaonpionHypothesis = False, +ManualMassHypo = [ 105.658, 105.658, 105.658, 105.658 ], +trkThresholdPt = 0.0, +trkMaxEta = 3.0, +BMassUpper = 50000.0, +BMassLower = 0, +oppChargesOnly = False, +#DiTrackMassUpper = 1019.445 + 100., +#DiTrackMassLower = 1019.445 - 100., +Chi2Cut = 100.0, +#TrkQuadrupletMassUpper = 60000.0, +#TrkQuadrupletMassLower = 0.01, +JpsiContainerKey = "BPHY16OniaCandidates", +TrackParticleCollection = "InDetTrackParticles", +MuonsUsedInJpsi = "Muons", +ExcludeJpsiMuonsOnly = True, +RequireNMuonTracks = 1, +TrkVertexFitterTool = BPHY16VertexFit, +TrackSelectorTool = BPHY16_VertexTools.InDetTrackSelectorTool, +UseMassConstraint = False) + +ToolSvc += BPHY16Plus2Tracks +print(BPHY16Plus2Tracks) + +## 6/ setup the combined augmentation/skimming tool for the Bpm +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__Reco_Vertex +BPHY16FourTrackSelectAndWrite = DerivationFramework__Reco_Vertex(name = "BPHY16FourTrackSelectAndWrite", + Jpsi2PlusTrackName = BPHY16Plus2Tracks, + OutputVtxContainerName = "BPHY16FourTrack", + PVContainerName = "PrimaryVertices", + RefPVContainerName = "BPHY16RefittedPrimaryVertices", + RefitPV = True, + MaxPVrefit = 10000, DoVertexType = 7) +ToolSvc += BPHY16FourTrackSelectAndWrite +print(BPHY16FourTrackSelectAndWrite) + + +BPHY16_Select_FourTrack = DerivationFramework__Select_onia2mumu( + name = "BPHY16_Select_FourTracks", + HypothesisName = "FourTracks", + InputVtxContainerName = "BPHY16FourTrack", + TrkMasses = [105.658, 105.658, 105.658, 105.658], + VtxMassHypo = 18100.0, + MassMin = 0, + MassMax = 500000, + Chi2Max = BPHY16Plus2Tracks.Chi2Cut) + +ToolSvc += BPHY16_Select_FourTrack +print(BPHY16_Select_FourTrack) + +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__ReVertex +BPHY16_Revertex = DerivationFramework__ReVertex( + name = "BPHY16_ReVertex", + InputVtxContainerName = "BPHY16FourTrack", + TrackIndices = [ 2, 3 ], + TrkVertexFitterTool = BPHY16VertexFit, + OutputVtxContainerName = "BPHY16TwoTrack" +) + +ToolSvc += BPHY16_Revertex +print(BPHY16_Revertex) + +BPHY16_Select_TwoTrack = DerivationFramework__Select_onia2mumu( + name = "BPHY16_Select_TwoTracks", + HypothesisName = "TwoTracks", + InputVtxContainerName = "BPHY16TwoTrack", + TrkMasses = [105.658, 105.658], + VtxMassHypo = 18100.0, + MassMin = 1, + MassMax = 500000, + Chi2Max = 90) + +ToolSvc += BPHY16_Select_TwoTrack +print(BPHY16_Select_TwoTrack) + +expression = "count(BPHY16FourTrack.passed_FourTracks) > 0" + +from DerivationFrameworkTools.DerivationFrameworkToolsConf import DerivationFramework__xAODStringSkimmingTool +BPHY16_SelectEvent = DerivationFramework__xAODStringSkimmingTool(name = "BPHY16_SelectEvent", + expression = expression) +ToolSvc += BPHY16_SelectEvent +print(BPHY16_SelectEvent) + +#-------------------------------------------------------------------- +## 6/ track and vertex thinning. We want to remove all reconstructed secondary vertices +## which hasn't passed any of the selections defined by (Select_*) tools. +## We also want to keep only tracks which are associates with either muons or any of the +## vertices that passed the selection. Multiple thinning tools can perform the +## selection. The final thinning decision is based OR of all the decisions (by default, +## although it can be changed by the JO). + +## a) thining out vertices that didn't pass any selection and idetifying tracks associated with +## selected vertices. The "VertexContainerNames" is a list of the vertex containers, and "PassFlags" +## contains all pass flags for Select_* tools that must be satisfied. The vertex is kept is it +## satisfy any of the listed selections. + + +#==================================================================== +# CREATE THE DERIVATION KERNEL ALGORITHM AND PASS THE ABOVE TOOLS +#==================================================================== +## 7/ IMPORTANT bit. Don't forget to pass the tools to the DerivationKernel! If you don't do that, they will not be +## be executed! + + +# The name of the kernel (BPHY16Kernel in this case) must be unique to this derivation +from DerivationFrameworkCore.DerivationFrameworkCoreConf import DerivationFramework__DerivationKernel +DerivationFrameworkJob += CfgMgr.DerivationFramework__DerivationKernel( + "BPHY16Kernel", + AugmentationTools = [BPHY16_Reco_mumu, BPHY16_Select_Upsi, BPHY16FourTrackSelectAndWrite, BPHY16_Select_FourTrack, BPHY16_Revertex, BPHY16_Select_TwoTrack], + SkimmingTools = [BPHY16_SelectEvent] + ) + +#==================================================================== +# SET UP STREAM +#==================================================================== +streamName = derivationFlags.WriteDAOD_BPHY16Stream.StreamName +fileName = buildFileName( derivationFlags.WriteDAOD_BPHY16Stream ) +BPHY16Stream = MSMgr.NewPoolRootStream( streamName, fileName ) +BPHY16Stream.AcceptAlgs(["BPHY16Kernel"]) +# Special lines for thinning +# Thinning service name must match the one passed to the thinning tools +from AthenaServices.Configurables import ThinningSvc, createThinningSvc +augStream = MSMgr.GetStream( streamName ) +evtStream = augStream.GetEventStream() +svcMgr += createThinningSvc( svcName="BPHY16ThinningSvc", outStreams=[evtStream] ) + + +#==================================================================== +# Slimming +#==================================================================== + +from DerivationFrameworkCore.SlimmingHelper import SlimmingHelper +BPHY16SlimmingHelper = SlimmingHelper("BPHY16SlimmingHelper") +AllVariables = [] +StaticContent = [] + +# Needed for trigger objects +BPHY16SlimmingHelper.IncludeMuonTriggerContent = True +BPHY16SlimmingHelper.IncludeBPhysTriggerContent = True + +## primary vertices +AllVariables += ["PrimaryVertices"] +StaticContent += ["xAOD::VertexContainer#BPHY16RefittedPrimaryVertices"] +StaticContent += ["xAOD::VertexAuxContainer#BPHY16RefittedPrimaryVerticesAux."] + +## ID track particles +AllVariables += ["InDetTrackParticles"] + +## combined / extrapolated muon track particles +## (note: for tagged muons there is no extra TrackParticle collection since the ID tracks +## are store in InDetTrackParticles collection) +AllVariables += ["CombinedMuonTrackParticles"] +AllVariables += ["ExtrapolatedMuonTrackParticles"] + +## muon container +AllVariables += ["Muons"] + + +StaticContent += ["xAOD::VertexContainer#%s" % BPHY16_Reco_mumu.OutputVtxContainerName] +StaticContent += ["xAOD::VertexAuxContainer#%sAux." % BPHY16_Reco_mumu.OutputVtxContainerName] +## we have to disable vxTrackAtVertex branch since it is not xAOD compatible +StaticContent += ["xAOD::VertexAuxContainer#%sAux.-vxTrackAtVertex" % BPHY16_Reco_mumu.OutputVtxContainerName] + +StaticContent += ["xAOD::VertexContainer#%s" % BPHY16FourTrackSelectAndWrite.OutputVtxContainerName] +StaticContent += ["xAOD::VertexAuxContainer#%sAux." % BPHY16FourTrackSelectAndWrite.OutputVtxContainerName] +## we have to disable vxTrackAtVertex branch since it is not xAOD compatible +StaticContent += ["xAOD::VertexAuxContainer#%sAux.-vxTrackAtVertex" % BPHY16FourTrackSelectAndWrite.OutputVtxContainerName] + +StaticContent += ["xAOD::VertexContainer#%s" % BPHY16_Revertex.OutputVtxContainerName] +StaticContent += ["xAOD::VertexAuxContainer#%sAux." % BPHY16_Revertex.OutputVtxContainerName] +## we have to disable vxTrackAtVertex branch since it is not xAOD compatible +StaticContent += ["xAOD::VertexAuxContainer#%sAux.-vxTrackAtVertex" % BPHY16_Revertex.OutputVtxContainerName] + + +# Truth information for MC only +if isSimulation: + AllVariables += ["TruthEvents","TruthParticles","TruthVertices","MuonTruthParticles"] + +BPHY16SlimmingHelper.AllVariables = AllVariables +BPHY16SlimmingHelper.StaticContent = StaticContent +BPHY16SlimmingHelper.AppendContentToStream(BPHY16Stream) + diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/BPHY17.py b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/BPHY17.py new file mode 100644 index 00000000000..fcd3b47a2e6 --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/BPHY17.py @@ -0,0 +1,316 @@ +#==================================================================== +# BPHY17.py +# This an example job options script showing how to set up a +# derivation of the data using the derivation framework. +# It requires the reductionConf flag BPHY17 in Reco_tf.py +#==================================================================== + +# Set up common services and job object. +# This should appear in ALL derivation job options +from DerivationFrameworkCore.DerivationFrameworkMaster import * + +isSimulation = globalflags.DataSource()=='geant4' + +print(isSimulation) + + +#==================================================================== +# AUGMENTATION TOOLS +#==================================================================== +## 1/ setup vertexing tools and services +include("DerivationFrameworkBPhys/configureVertexing.py") +BPHY17_VertexTools = BPHYVertexTools("BPHY17") + +from InDetTrackSelectorTool.InDetTrackSelectorToolConf import InDet__InDetDetailedTrackSelectorTool +InDetTrackSelectorTool = InDet__InDetDetailedTrackSelectorTool(name = "BPHY17_CascadeTrackSelectorTool", + pTMin = 1000.0, + IPd0Max = 10000.0, + IPz0Max = 10000.0, + z0Max = 10000.0, + sigIPd0Max = 10000.0, + sigIPz0Max = 10000.0, + d0significanceMax = -1., + z0significanceMax = -1., + etaMax = 9999., + useTrackSummaryInfo = True, + nHitBLayer = 0, + nHitPix = 1, + nHitBLayerPlusPix = 1, + nHitSct = 2, + nHitSi = 3, + nHitTrt = 0, + nHitTrtHighEFractionMax = 10000.0, + useSharedHitInfo = False, + useTrackQualityInfo = True, + fitChi2OnNdfMax = 10000.0, + TrtMaxEtaAcceptance = 1.9, + TrackSummaryTool = BPHY17_VertexTools.InDetTrackSummaryTool, + Extrapolator = BPHY17_VertexTools.InDetExtrapolator + ) + +ToolSvc += InDetTrackSelectorTool +print(InDetTrackSelectorTool) + +#BPHY17_VertexTools.TrkVKalVrtFitter.OutputLevel = DEBUG + +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__Cascade3Plus1 +BPHY17BsDsPi = DerivationFramework__Cascade3Plus1( + name = "BPHY17BsDsPi", + TrackMassHyp = [ 493.677, 493.677, 139.57018, 139.57018 ], + PTCutPerTrack = [ 0,0,1500, 1500], + HypothesisName = "Bs", + Track3Name = "Ds", + TrackSelectorTool = InDetTrackSelectorTool, + TrkVertexFitterTool = BPHY17_VertexTools.TrkVKalVrtFitter, + PVContainerName = "PrimaryVertices", + RefitPV = True, + #OutputLevel = DEBUG, + RefPVContainerName = "BPHY17RefittedPrimaryVertices", + CascadeVertexCollections= ["BPHY17CascadeVtx1", "BPHY17CascadeVtx2"], + ThreeVertexOutputContainer = "BPHY17DsKaonVertexes", EliminateBad3Tracksfrom4Track = True, + ThreeTrackChi2NDF = 6, + TwoTrackMassMin = 1009.0, + TwoTrackMassMax = 1031.0, + ThreeTrackMassMin = 1800.47, + ThreeTrackMassMax = 2100.0, + FourTrackMassMin = 5100.0, + FourTrackMassMax = 5600.0, + ThreeTracksMass = 1968.47, + FourTracksMass = 5366.79, + Chi2NDFCut = 10, + FourTrackMassFinalMin = 5150., + FourTrackMassFinalMax = 5650.0, + ThreeTrackMassConstraint = False, + CopyAllVertices = False + ) + +ToolSvc += BPHY17BsDsPi +print(BPHY17BsDsPi) + +### +BPHY17BsDsPiMuons = DerivationFramework__Cascade3Plus1( + name = "BPHY17BsDsPiMuons", + TrackMassHyp = [ 105.658374, 105.658374, 139.57018, 139.57018 ], + PTCutPerTrack = [ 0,0,1500, 1500], + HypothesisName = "Bs", + Track3Name = "Ds", + TrackSelectorTool = InDetTrackSelectorTool, + TrkVertexFitterTool = BPHY17_VertexTools.TrkVKalVrtFitter, + PVContainerName = "PrimaryVertices", + RefitPV = True, + #OutputLevel = DEBUG, + RefPVContainerName = "BPHY17RefittedPrimaryVerticesMuons", + CascadeVertexCollections= ["BPHY17CascadeMuonVtx1", "BPHY17CascadeMuonVtx2"], + ThreeVertexOutputContainer = "BPHY17DsMuonVertexes", EliminateBad3Tracksfrom4Track = True, + ThreeTrackChi2NDF = 10, + TwoTrackMassMin = 860.0, + TwoTrackMassMax = 1180.0, + ThreeTrackMassMin = 1800.47, + ThreeTrackMassMax = 2100.0, + FourTrackMassMin = 5100.0, + FourTrackMassMax = 5550.0, + ThreeTracksMass = 1968.47, + FourTracksMass = 5366.79, + Chi2NDFCut = 10, + FourTrackMassFinalMin = 5150., + FourTrackMassFinalMax = 5500.0, + ThreeTrackMassConstraint = False, + UseMuonsForTracks = [0, 1], + CopyAllVertices = False + ) + +ToolSvc += BPHY17BsDsPiMuons +print(BPHY17BsDsPiMuons) + +### +BPHY17BsDsMuSemiLepMuons = DerivationFramework__Cascade3Plus1( + name = "BPHY17BsDsMuSemiLepMuons", + TrackMassHyp = [ 105.658374, 105.658374, 139.57018, 105.658374 ], + PTCutPerTrack = [ 0,0,1500, 0], + HypothesisName = "Bs", + Track3Name = "Ds", + TrackSelectorTool = InDetTrackSelectorTool, + TrkVertexFitterTool = BPHY17_VertexTools.TrkVKalVrtFitter, + PVContainerName = "PrimaryVertices", + RefitPV = True, + #OutputLevel = DEBUG, + RefPVContainerName = "BPHY17RefittedPrimaryVerticesMuonsSemiLep", + CascadeVertexCollections= ["BPHY17CascadeMuonSemiLepVtx1", "BPHY17CascadeMuonSemiLepVtx2"], + ThreeVertexOutputContainer = "BPHY17DsMuonSemiLepVertexes", EliminateBad3Tracksfrom4Track = True, + ThreeTrackChi2NDF = 10, + TwoTrackMassMin = 860.0, + TwoTrackMassMax = 1180.0, + ThreeTrackMassMin = 1800.47, + ThreeTrackMassMax = 2100.0, + FourTrackMassMin = 0., + FourTrackMassMax = 999999.0, + ThreeTracksMass = 1968.47, + FourTracksMass = 5366.79, + Chi2NDFCut = 10, + FourTrackMassFinalMin = 0, + FourTrackMassFinalMax = 999999.0, + ThreeTrackMassConstraint = False, + UseMuonsForTracks = [0, 1, 3], + CopyAllVertices = False + ) + +ToolSvc += BPHY17BsDsMuSemiLepMuons +print(BPHY17BsDsMuSemiLepMuons) + + +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__Select_onia2mumu +BPHY17_Select_DsPhiKK = DerivationFramework__Select_onia2mumu( + name = "BPHY17_Select_DsPhiKK", + HypothesisName = "DsPhiKK", + InputVtxContainerName = BPHY17BsDsPi.ThreeVertexOutputContainer, + TrkMasses = [493.677, 493.677, 139.57018], + VtxMassHypo = 1968.47, + MassMin = 1000.0, + MassMax = 3000.0, Do3d = False, DoVertexType = 1, + Chi2Max = 200) + +ToolSvc += BPHY17_Select_DsPhiKK +print(BPHY17_Select_DsPhiKK) + +BPHY17_Select_DsPhiMM = DerivationFramework__Select_onia2mumu( + name = "BPHY17_Select_DsPhiMM", + HypothesisName = "DsPhiMM", + InputVtxContainerName = BPHY17BsDsPiMuons.ThreeVertexOutputContainer, + TrkMasses = [105.658374, 105.658374, 139.57018], + VtxMassHypo = 1968.47, + MassMin = 1000.0, + MassMax = 3000.0, Do3d = False, DoVertexType = 1, + Chi2Max = 200) + +ToolSvc += BPHY17_Select_DsPhiMM +print(BPHY17_Select_DsPhiMM) + + +BPHY17_Select_DsPhiMMSemi = DerivationFramework__Select_onia2mumu( + name = "BPHY17_Select_DsPhiMMSemi", + HypothesisName = "DsPhiMMSemiLep", + InputVtxContainerName = BPHY17BsDsMuSemiLepMuons.ThreeVertexOutputContainer, + TrkMasses = [105.658374, 105.658374, 139.57018], + VtxMassHypo = 1968.47, + MassMin = 1000.0, + MassMax = 3000.0, Do3d = False, DoVertexType = 1, + Chi2Max = 200) + +ToolSvc += BPHY17_Select_DsPhiMMSemi +print(BPHY17_Select_DsPhiMMSemi) + +### + +#-------------------------------------------------------------------- +## 5/ select the event. We only want to keep events that contain certain vertices which passed certain selection. +## This is specified by the "SelectionExpression" property, which contains the expression in the following format: +## +## "ContainerName.passed_HypoName > count" +## +## where "ContainerName" is output container form some Reco_* tool, "HypoName" is the hypothesis name setup in some "Select_*" +## tool and "count" is the number of candidates passing the selection you want to keep. + +expression = "count(%s.x > -999) > 0" % BPHY17BsDsPi.CascadeVertexCollections[0] +expression += " || count(%s.x > -999) > 0" % BPHY17BsDsPiMuons.CascadeVertexCollections[0] +expression += " || count(%s.x > -999) > 0" % BPHY17BsDsPi.ThreeVertexOutputContainer +expression += " || count(%s.x > -999) > 0" % BPHY17BsDsPiMuons.ThreeVertexOutputContainer +expression += " || count(%s.x > -999) > 0" % BPHY17BsDsMuSemiLepMuons.CascadeVertexCollections[0] +expression += " || count(%s.x > -999) > 0" % BPHY17BsDsMuSemiLepMuons.ThreeVertexOutputContainer + +from DerivationFrameworkTools.DerivationFrameworkToolsConf import DerivationFramework__xAODStringSkimmingTool +BPHY17_SelectEvent = DerivationFramework__xAODStringSkimmingTool(name = "BPHY17_SelectEvent", + expression = expression) +ToolSvc += BPHY17_SelectEvent +print(BPHY17_SelectEvent) + +MyVertexCollections = BPHY17BsDsPi.CascadeVertexCollections + BPHY17BsDsPiMuons.CascadeVertexCollections + BPHY17BsDsMuSemiLepMuons.CascadeVertexCollections + \ + [ BPHY17BsDsPi.ThreeVertexOutputContainer, BPHY17BsDsPiMuons.ThreeVertexOutputContainer, BPHY17BsDsMuSemiLepMuons.ThreeVertexOutputContainer ] + +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__Thin_vtxTrk +BPHY17_thinningTool_Tracks = DerivationFramework__Thin_vtxTrk( + name = "BPHY17_thinningTool_Tracks", + TrackParticleContainerName = "InDetTrackParticles", + IgnoreFlags = True, + VertexContainerNames = MyVertexCollections, + PassFlags = ["passed_DsPhiMM", "passed_DsPhiKK"] ) + +ToolSvc += BPHY17_thinningTool_Tracks + + +# Only save truth informtion directly associated with Onia +from DerivationFrameworkMCTruth.DerivationFrameworkMCTruthConf import DerivationFramework__GenericTruthThinning +BPHY17TruthThinTool = DerivationFramework__GenericTruthThinning(name = "BPHY17TruthThinTool", + ParticleSelectionString = "TruthParticles.pdgId == 511 || TruthParticles.pdgId == -511 || TruthParticles.pdgId == 531 || TruthParticles.pdgId == -531", + PreserveDescendants = True, + PreserveAncestors = True) +ToolSvc += BPHY17TruthThinTool +print(BPHY17TruthThinTool) + + +#==================================================================== +# CREATE THE DERIVATION KERNEL ALGORITHM AND PASS THE ABOVE TOOLS +#==================================================================== +## 7/ IMPORTANT bit. Don't forget to pass the tools to the DerivationKernel! If you don't do that, they will not be +## be executed! + +# Added by ASC +BPHY17ThinningTools = [BPHY17_thinningTool_Tracks] +if globalflags.DataSource()=='geant4': + BPHY17ThinningTools.append(BPHY17TruthThinTool) + +# The name of the kernel (BPHY17Kernel in this case) must be unique to this derivation +from DerivationFrameworkCore.DerivationFrameworkCoreConf import DerivationFramework__DerivationKernel +DerivationFrameworkJob += CfgMgr.DerivationFramework__DerivationKernel( + "BPHY17Kernel", + AugmentationTools = [BPHY17BsDsPi, BPHY17BsDsPiMuons, BPHY17BsDsMuSemiLepMuons, BPHY17_Select_DsPhiKK, BPHY17_Select_DsPhiMM, BPHY17_Select_DsPhiMMSemi], + SkimmingTools = [BPHY17_SelectEvent], + ThinningTools = BPHY17ThinningTools, +# OutputLevel = DEBUG + ) + +#==================================================================== +# SET UP STREAM +#==================================================================== +streamName = derivationFlags.WriteDAOD_BPHY17Stream.StreamName +fileName = buildFileName( derivationFlags.WriteDAOD_BPHY17Stream ) +BPHY17Stream = MSMgr.NewPoolRootStream( streamName, fileName ) +BPHY17Stream.AcceptAlgs(["BPHY17Kernel"]) +# Special lines for thinning +# Thinning service name must match the one passed to the thinning tools +from AthenaServices.Configurables import ThinningSvc, createThinningSvc +augStream = MSMgr.GetStream( streamName ) +evtStream = augStream.GetEventStream() +svcMgr += createThinningSvc( svcName="BPHY17ThinningSvc", outStreams=[evtStream] ) + + +#==================================================================== +# Slimming +#==================================================================== + +# Added by ASC +from DerivationFrameworkCore.SlimmingHelper import SlimmingHelper +BPHY17SlimmingHelper = SlimmingHelper("BPHY17SlimmingHelper") +AllVariables = [] +StaticContent = [] + +# Needed for trigger objects +BPHY17SlimmingHelper.IncludeBPhysTriggerContent = False +BPHY17SlimmingHelper.IncludeMuonTriggerContent = False +## primary vertices +AllVariables += ["PrimaryVertices"] + +for f in MyVertexCollections + [BPHY17BsDsPi.RefPVContainerName, BPHY17BsDsPiMuons.RefPVContainerName + BPHY17BsDsMuSemiLepMuons.RefPVContainerName ]: + StaticContent += ["xAOD::VertexContainer#%s" % f] + StaticContent += ["xAOD::VertexAuxContainer#%sAux.-vxTrackAtVertex" % f] + +## ID track particles +AllVariables += ["InDetTrackParticles"] +AllVariables += ["Muons"] + +# Truth information for MC only +if isSimulation: + AllVariables += ["TruthEvents","TruthParticles","TruthVertices"] + +BPHY17SlimmingHelper.AllVariables = AllVariables +BPHY17SlimmingHelper.StaticContent = StaticContent +BPHY17SlimmingHelper.AppendContentToStream(BPHY17Stream) diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/BPHY18.py b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/BPHY18.py new file mode 100644 index 00000000000..fdabaaacdfb --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/BPHY18.py @@ -0,0 +1,506 @@ +#==================================================================== +# BPHY18.py +# B0 -> K*ee +# It requires the reductionConf flag BPHY18 in Reco_tf.py +#==================================================================== + + +#==================================================================== +# FLAGS TO PERSONALIZE THE DERIVATION +#==================================================================== + +onlyAugmentations = False +thinTruth = False +skimTruth = False + + +# Set up common services and job object. +from DerivationFrameworkCore.DerivationFrameworkMaster import * +from DerivationFrameworkMuons.MuonsCommon import * +from DerivationFrameworkJetEtMiss.JetCommon import * +from DerivationFrameworkEGamma.EGammaCommon import * +from DerivationFrameworkEGamma.ElectronsCPContent import * + +isSimulation = False +if globalflags.DataSource()=='geant4': + isSimulation = True + +print(isSimulation) + + +#==================================================================== +# AUGMENTATION TOOLS +#==================================================================== +## 1/ setup vertexing tools and services +include("DerivationFrameworkBPhys/configureVertexing.py") +BPHY18_VertexTools = BPHYVertexTools("BPHY18") + + +print('********************** VERTEX TOOLS ***********************') +print(BPHY18_VertexTools) +print(BPHY18_VertexTools.TrkV0Fitter) +print('********************** END VERTEX TOOLS ***********************') + +#==================================================================== +# TriggerCounting for Kernel1 #Added by Matteo +#==================================================================== +#List of trigggers to be counted (high Sig-eff*Lumi ones are in) +triggersToMetadata = [ +"HLT_2e5_lhvloose_nod0_bBeexM6000t", #37,143,877 inb +"HLT_e5_lhvloose_nod0_bBeexM6000t", #37,143,877 +"HLT_e5_lhvloose_nod0_bBeexM6000t_2mu4_nomucomb_L1BPH-0DR3-EM7J15_2MU4", #37,312,506 +"HLT_e5_lhvloose_nod0_bBeexM6000t_mu6_nomucomb_L1BPH-0DR3-EM7J15_MU6", #27,041,892 +"HLT_e5_lhvloose_nod0_bBeexM6000_mu6_nomucomb_L1BPH-0DR3-EM7J15_MU6", #149,100 +"HLT_e9_lhloose_bBeexM2700_2mu4_nomucomb_L1BPH-0DR3-EM7J15_2MU4", #2,681,764 +"HLT_e9_lhloose_bBeexM2700_mu6_nomucomb_L1BPH-0DR3-EM7J15_MU6", #1,979,362 +"HLT_e9_lhloose_bBeexM6000_2mu4_nomucomb_L1BPH-0DR3-EM7J15_2MU4", #3,359,105 +"HLT_e9_lhloose_bBeexM6000_mu6_nomucomb_L1BPH-0DR3-EM7J15_MU6", #2,426,663 +"HLT_e9_lhloose_e5_lhloose_bBeexM2700_2mu4_nomucomb_L1BPH-0M9-EM7-EM5_2MU4", #2,950,935 +"HLT_e9_lhloose_e5_lhloose_bBeexM2700_mu6_nomucomb_L1BPH-0M9-EM7-EM5_MU6", #2,928,030 +"HLT_e9_lhloose_e5_lhloose_bBeexM6000_2mu4_nomucomb_L1BPH-0M9-EM7-EM5_2MU4", #3,647,507 +"HLT_e9_lhloose_e5_lhloose_bBeexM6000_mu6_nomucomb_L1BPH-0M9-EM7-EM5_MU6", #3,605,371 +"HLT_e9_lhvloose_nod0_e5_lhvloose_nod0_bBeexM6000t_2mu4_nomucomb_L1BPH-0M9-EM7-EM5_2MU4", #40,169,436 +"HLT_e9_lhvloose_nod0_e5_lhvloose_nod0_bBeexM6000t_mu6_nomucomb_L1BPH-0M9-EM7-EM5_MU6", #37,312,506 +"HLT_e9_lhvloose_nod0_e5_lhvloose_nod0_bBeexM6000_mu6_nomucomb_L1BPH-0M9-EM7-EM5_MU6", #677,340 + +'HLT_3mu4_bDimu2700', 'HLT_3mu6_bDimu', 'HLT_mu6_2mu4_bDimu2700','HLT_mu11_mu6_bJpsimumu', 'HLT_3mu4_bJpsi', 'HLT_mu11_mu6_bDimu', 'HLT_2mu6_bBmumu_Lxy0_L1BPH-2M9-2MU6_BPH-2DR15-2MU6','HLT_2mu6_bJpsimumu_L1BPH-2M9-2MU6_BPH-2DR15-2MU6', 'HLT_2mu10_bBmumuxv2', 'HLT_mu6_mu4_bDimu', 'HLT_2mu6_bBmumuxv2_L1LFV-MU6', 'HLT_mu11_mu6_bJpsimumu_Lxy0', 'HLT_mu11_mu6_bDimu2700', 'HLT_mu11_mu6_bBmumux_BpmumuKp', 'HLT_mu6_mu4_bJpsimumu_Lxy0_L1BPH-2M9-MU6MU4_BPH-0DR15-MU6MU4', 'HLT_mu11_mu6_bBmumuxv2', 'HLT_mu6_2mu4_bJpsi', 'HLT_2mu6_bBmumux_BpmumuKp_L1BPH-2M9-2MU6_BPH-2DR15-2MU6', 'HLT_2mu6_bJpsimumu_Lxy0_L1BPH-2M9-2MU6_BPH-2DR15-2MU6', 'HLT_3mu6_bJpsi', 'HLT_mu11_mu6_bBmumu'] + +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__TriggerCountToMetadata +BPHY18TriggerCountToMetadata = DerivationFramework__TriggerCountToMetadata(name = "BPHY18TriggerCount", + TriggerList = triggersToMetadata, + FolderName = "BPHY18") + +ToolSvc += BPHY18TriggerCountToMetadata + +#==================================================================== +# PRESELECTION for Kernel1 #Added by Matteo +#==================================================================== +## 1/ Setup the skimming based on triggers +## + +triggerList = ["HLT_e5_lhvloose_nod0_bBeexM6000t_2mu4_nomucomb_L1BPH-0DR3-EM7J15_2MU4", #37,312,506 +"HLT_e5_lhvloose_nod0_bBeexM6000t_mu6_nomucomb_L1BPH-0DR3-EM7J15_MU6", #27,041,892 +"HLT_e5_lhvloose_nod0_bBeexM6000_mu6_nomucomb_L1BPH-0DR3-EM7J15_MU6", #149,100 +"HLT_e9_lhloose_bBeexM2700_2mu4_nomucomb_L1BPH-0DR3-EM7J15_2MU4", #2,681,764 +"HLT_e9_lhloose_bBeexM2700_mu6_nomucomb_L1BPH-0DR3-EM7J15_MU6", #1,979,362 +"HLT_e9_lhloose_bBeexM6000_2mu4_nomucomb_L1BPH-0DR3-EM7J15_2MU4", #3,359,105 +"HLT_e9_lhloose_bBeexM6000_mu6_nomucomb_L1BPH-0DR3-EM7J15_MU6", #2,426,663 +"HLT_e9_lhloose_e5_lhloose_bBeexM2700_2mu4_nomucomb_L1BPH-0M9-EM7-EM5_2MU4", #2,950,935 +"HLT_e9_lhloose_e5_lhloose_bBeexM2700_mu6_nomucomb_L1BPH-0M9-EM7-EM5_MU6", #2,928,030 +"HLT_e9_lhloose_e5_lhloose_bBeexM6000_2mu4_nomucomb_L1BPH-0M9-EM7-EM5_2MU4", #3,647,507 +"HLT_e9_lhloose_e5_lhloose_bBeexM6000_mu6_nomucomb_L1BPH-0M9-EM7-EM5_MU6", #3,605,371 +"HLT_e9_lhvloose_nod0_e5_lhvloose_nod0_bBeexM6000t_2mu4_nomucomb_L1BPH-0M9-EM7-EM5_2MU4", #40,169,436 +"HLT_e9_lhvloose_nod0_e5_lhvloose_nod0_bBeexM6000t_mu6_nomucomb_L1BPH-0M9-EM7-EM5_MU6", #37,312,506 +"HLT_e9_lhvloose_nod0_e5_lhvloose_nod0_bBeexM6000_mu6_nomucomb_L1BPH-0M9-EM7-EM5_MU6", #677,340 +"HLT_2e5_lhvloose_nod0_bBeexM6000t", #37,143,877 inb +"HLT_e5_lhvloose_nod0_bBeexM6000t" #37,143,877 +] # Seeded + Unseeded BeeX triggers + +triggerList_unseeded = ["HLT_2e5_lhvloose_nod0_bBeexM6000t", #37,143,877 inb +"HLT_e5_lhvloose_nod0_bBeexM6000t" #37,143,877 +] + +from DerivationFrameworkTools.DerivationFrameworkToolsConf import DerivationFramework__TriggerSkimmingTool +BPHY18TriggerSkim = DerivationFramework__TriggerSkimmingTool(name = "BPHY18TriggerSkim", + TriggerListOR = triggerList, + TriggerListORHLTOnly = triggerList_unseeded ) + +ToolSvc += BPHY18TriggerSkim +print(BPHY18TriggerSkim) + +#do not know what this does, but let's leave it for now, until we see if it's useful or not! +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__AugOriginalCounts +BPHY18_AugOriginalCounts = DerivationFramework__AugOriginalCounts(name = "BPHY18_AugOriginalCounts", + VertexContainer = "PrimaryVertices", + TrackContainer = "InDetTrackParticles" ) +ToolSvc += BPHY18_AugOriginalCounts + +#lhvloose_nod0 +from ElectronPhotonSelectorTools.ElectronPhotonSelectorToolsConf import AsgElectronLikelihoodTool +ElectronLHSelectorLHvloose_nod0 = AsgElectronLikelihoodTool("ElectronLHSelectorLHvloosenod0", +ConfigFile="ElectronPhotonSelectorTools/offline/mc16_20190328_nod0/ElectronLikelihoodVeryLooseOfflineConfig2017_Smooth_nod0.conf") +ElectronLHSelectorLHvloose_nod0.primaryVertexContainer = "PrimaryVertices" +ToolSvc += ElectronLHSelectorLHvloose_nod0 +print(ElectronLHSelectorLHvloose_nod0) + +# decorate electrons with the output of LH vloose nod0 +ElectronPassLHvloosenod0 = DerivationFramework__EGSelectionToolWrapper(name = "ElectronPassLHvloosenod0", + EGammaSelectionTool = ElectronLHSelectorLHvloose_nod0, + EGammaFudgeMCTool = "", + CutType = "", + StoreGateEntryName = "DFCommonElectronsLHVeryLoosenod0", + ContainerName = "Electrons") +ToolSvc += ElectronPassLHvloosenod0 +print(ElectronPassLHvloosenod0) + +#-------------------------------------------------------------------- +## 2/ setup JpsiFinder tool +from JpsiUpsilonTools.JpsiUpsilonToolsConf import Analysis__JpsiFinder_ee +BPHY18DiElectronFinder = Analysis__JpsiFinder_ee( + name = "BPHY18DiElectronFinder", + OutputLevel = INFO, + elAndEl = True, + elAndTrack = False, + TrackAndTrack = False, + assumeDiElectrons = True, + elThresholdPt = 4000.0, + invMassUpper = 7000.0, + invMassLower = 1.0, + Chi2Cut = 30., + oppChargesOnly = False, + allChargeCombinations = True, + useElectronTrackMeasurement = True, + electronCollectionKey = "Electrons", + TrackParticleCollection = "GSFTrackParticles", + useEgammaCuts = True, + V0VertexFitterTool = BPHY18_VertexTools.TrkV0Fitter, + useV0Fitter = False, + TrkVertexFitterTool = BPHY18_VertexTools.TrkVKalVrtFitter, + TrackSelectorTool = BPHY18_VertexTools.InDetTrackSelectorTool, + VertexPointEstimator = BPHY18_VertexTools.VtxPointEstimator, + ElectronSelection = "d0_or_nod0" + ) + +ToolSvc += BPHY18DiElectronFinder +print(BPHY18DiElectronFinder) + +#-------------------------------------------------------------------- +## 3/ setup the vertex reconstruction "call" tool(s). +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__Reco_Vertex +BPHY18DiElectronSelectAndWrite = DerivationFramework__Reco_Vertex( + name = "BPHY18DiElectronSelectAndWrite", + VertexSearchTool = BPHY18DiElectronFinder, + OutputVtxContainerName = "BPHY18DiElectronCandidates", + PVContainerName = "PrimaryVertices", + RefPVContainerName = "SHOULDNOTBEUSED", + DoVertexType = 7 + ) + +ToolSvc += BPHY18DiElectronSelectAndWrite +print(BPHY18DiElectronSelectAndWrite) + +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__Select_onia2mumu +BPHY18_Select_DiElectrons = DerivationFramework__Select_onia2mumu( + name = "BPHY18_Select_DiElectrons", + HypothesisName = "Jpsi", + InputVtxContainerName = "BPHY18DiElectronCandidates", + VtxMassHypo = 3096.916, + MassMin = 1.0, + MassMax = 7000.0, + Chi2Max = 30, + DoVertexType = 7 + ) + +ToolSvc += BPHY18_Select_DiElectrons +print(BPHY18_Select_DiElectrons) + +## 4/ setup a new vertexing tool (necessary due to use of mass constraint) +from TrkVKalVrtFitter.TrkVKalVrtFitterConf import Trk__TrkVKalVrtFitter +BeeKstVertexFit = Trk__TrkVKalVrtFitter( + name = "BeeKstVertexFit", + Extrapolator = BPHY18_VertexTools.InDetExtrapolator, + FirstMeasuredPoint = True, + MakeExtendedVertex = True + ) + +ToolSvc += BeeKstVertexFit +print(BeeKstVertexFit) + +## 5/ setup the Jpsi+2 track finder +from JpsiUpsilonTools.JpsiUpsilonToolsConf import Analysis__JpsiPlus2Tracks +BPHY18BeeKst = Analysis__JpsiPlus2Tracks( + name = "BPHY18BeeKstFinder", + OutputLevel = INFO, + kaonkaonHypothesis = False, + pionpionHypothesis = False, + kaonpionHypothesis = True, + oppChargesOnly = False, + SameChargesOnly = False, + trkThresholdPt = 500.0, + trkMaxEta = 3.0, + BThresholdPt = 1000., + BMassLower = 3000.0, + BMassUpper = 6500.0, + JpsiContainerKey = "BPHY18DiElectronCandidates", + TrackParticleCollection = "InDetTrackParticles", + ExcludeCrossJpsiTracks = False, + TrkVertexFitterTool = BeeKstVertexFit, + TrackSelectorTool = BPHY18_VertexTools.InDetTrackSelectorTool, + UseMassConstraint = False, + DiTrackMassUpper = 1110., + DiTrackMassLower = 690., + Chi2Cut = 15.0, + DiTrackPt = 500., + TrkQuadrupletMassLower = 1000.0, + TrkQuadrupletMassUpper = 10000.0, + FinalDiTrackPt = 500., + UseGSFTrackIndices = [0,1] + ) + +ToolSvc += BPHY18BeeKst +print(BPHY18BeeKst) + +## 6/ setup the combined augmentation/skimming tool for the BeeKst +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__Reco_Vertex +BPHY18BeeKstSelectAndWrite = DerivationFramework__Reco_Vertex( + name = "BPHY18BeeKstSelectAndWrite", + Jpsi2PlusTrackName = BPHY18BeeKst, + OutputVtxContainerName = "BeeKstCandidates", + PVContainerName = "PrimaryVertices", + RefPVContainerName = "BPHY18RefittedPrimaryVertices", + RefitPV = True, + MaxPVrefit = 10000, + DoVertexType = 7 + ) + +ToolSvc += BPHY18BeeKstSelectAndWrite +print(BPHY18BeeKstSelectAndWrite) + +## b/ augment and select B->eeKst candidates +BPHY18_Select_BeeKst = DerivationFramework__Select_onia2mumu( + name = "BPHY18_Select_BeeKst", + HypothesisName = "Bd", + InputVtxContainerName = "BeeKstCandidates", + TrkMasses = [0.511, 0.511, 493.677, 139.570], + VtxMassHypo = 5279.6, + MassMin = 1.0, + MassMax = 10000.0, + Chi2Max = 30.0 + ) + +ToolSvc += BPHY18_Select_BeeKst +print(BPHY18_Select_BeeKst) + +## c/ augment and select Bdbar->eeKstbar candidates +BPHY18_Select_BeeKstbar = DerivationFramework__Select_onia2mumu( + name = "BPHY18_Select_Bd2JpsiKstbar", + HypothesisName = "Bdbar", + InputVtxContainerName = "BeeKstCandidates", + TrkMasses = [0.511, 0.511, 139.570, 493.677], + VtxMassHypo = 5279.6, + MassMin = 1.0, + MassMax = 10000.0, + Chi2Max = 30.0 + ) + +ToolSvc += BPHY18_Select_BeeKstbar +print(BPHY18_Select_BeeKstbar) + +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__ReVertex +BPHY18_diMeson_revertex = DerivationFramework__ReVertex( + name = "BPHY18_diMeson_revertex", + InputVtxContainerName = "BeeKstCandidates", + TrackIndices = [ 2, 3 ], + TrkVertexFitterTool = BeeKstVertexFit, + OutputVtxContainerName = "BPHY18DiMeson" + ) + +ToolSvc += BPHY18_diMeson_revertex +print(BPHY18_diMeson_revertex) + +BPHY18_Select_Kpi = DerivationFramework__Select_onia2mumu( + name = "BPHY18_Select_Kpi", + HypothesisName = "Kpi", + InputVtxContainerName = "BPHY18DiMeson", + TrkMasses = [ 493.677, 139.570 ], + VtxMassHypo = 891.66, + MassMin = 1.0, + MassMax = 100000.0, + Chi2Max = 100.0 + ) + +ToolSvc += BPHY18_Select_Kpi +print(BPHY18_Select_Kpi) + +BPHY18_Select_piK = DerivationFramework__Select_onia2mumu( + name = "BPHY18_Select_piK", + HypothesisName = "piK", + InputVtxContainerName = "BPHY18DiMeson", + TrkMasses = [ 139.570, 493.677 ], + VtxMassHypo = 891.66, + MassMin = 1.0, + MassMax = 100000.0, + Chi2Max = 100.0 + ) + +ToolSvc += BPHY18_Select_piK +print(BPHY18_Select_piK) + +if True: + from DerivationFrameworkTools.DerivationFrameworkToolsConf import DerivationFramework__xAODStringSkimmingTool + BPHY18_SelectBeeKstEvent = DerivationFramework__xAODStringSkimmingTool( + name = "BPHY18_SelectBeeKstEvent", + expression = "(count(BeeKstCandidates.passed_Bd > 0) + count(BeeKstCandidates.passed_Bdbar > 0)) > 0") + ToolSvc += BPHY18_SelectBeeKstEvent + print(BPHY18_SelectBeeKstEvent) + + #==================================================================== + # Make event selection based on an OR of the input skimming tools + #==================================================================== + from DerivationFrameworkTools.DerivationFrameworkToolsConf import DerivationFramework__FilterCombinationAND + BPHY18SkimmingAND = CfgMgr.DerivationFramework__FilterCombinationAND( + "BPHY18SkimmingAND", + FilterList = [BPHY18_SelectBeeKstEvent, BPHY18TriggerSkim]) + ToolSvc += BPHY18SkimmingAND + print(BPHY18SkimmingAND) + +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__Thin_vtxTrk +BPHY18_thinningTool_Tracks = DerivationFramework__Thin_vtxTrk( + name = "BPHY18_thinningTool_Tracks", + TrackParticleContainerName = "InDetTrackParticles", + VertexContainerNames = ["BeeKstCandidates"], + PassFlags = ["passed_Bd", "passed_Bdbar"] ) + +BPHY18_thinningTool_GSFTracks = DerivationFramework__Thin_vtxTrk( + name = "BPHY18_thinningTool_GSFTracks", + TrackParticleContainerName = "GSFTrackParticles", + VertexContainerNames = ["BeeKstCandidates"], + PassFlags = ["passed_Bd", "passed_Bdbar"] ) + +ToolSvc += BPHY18_thinningTool_Tracks +ToolSvc += BPHY18_thinningTool_GSFTracks + +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__BPhysPVThinningTool +BPHY18_thinningTool_PV = DerivationFramework__BPhysPVThinningTool( + name = "BPHY18_thinningTool_PV", + CandidateCollections = ["BeeKstCandidates"], + KeepPVTracks = True + ) + +ToolSvc += BPHY18_thinningTool_PV + +## b) thinning out tracks that are not attached to muons. +from DerivationFrameworkInDet.DerivationFrameworkInDetConf import DerivationFramework__MuonTrackParticleThinning +BPHY18MuonTPThinningTool = DerivationFramework__MuonTrackParticleThinning( + name = "BPHY18MuonTPThinningTool", + MuonKey = "Muons", + InDetTrackParticlesKey = "InDetTrackParticles") +ToolSvc += BPHY18MuonTPThinningTool + +from DerivationFrameworkInDet.DerivationFrameworkInDetConf import DerivationFramework__EgammaTrackParticleThinning +BPHY18EgammaTPThinningTool = DerivationFramework__EgammaTrackParticleThinning( + name = "BPHY18EgammaTPThinningTool", + SGKey = "Electrons", + InDetTrackParticlesKey = "InDetTrackParticles") +ToolSvc += BPHY18EgammaTPThinningTool + +# Only save truth informtion directly associated with: mu Ds+ D+ D*+ Ds*+ D0 D*0 B+ B*+ B0 B*0 +from DerivationFrameworkMCTruth.DerivationFrameworkMCTruthConf import DerivationFramework__GenericTruthThinning +BPHY18TruthThinTool = DerivationFramework__GenericTruthThinning(name = "BPHY18TruthThinTool", + ParticleSelectionString = "abs(TruthParticles.pdgId) == 11 || abs(TruthParticles.pdgId) == 13 || abs(TruthParticles.pdgId) == 10311 || abs(TruthParticles.pdgId) == 521 || abs(TruthParticles.pdgId) == 523 || TruthParticles.pdgId == 511 || TruthParticles.pdgId == 513", + PreserveDescendants = True, + PreserveAncestors = True) +ToolSvc += BPHY18TruthThinTool + +# Only save truth neutrino and b/c quarks information +from DerivationFrameworkMCTruth.DerivationFrameworkMCTruthConf import DerivationFramework__GenericTruthThinning +BPHY18TruthThinNoChainTool = DerivationFramework__GenericTruthThinning(name = "BPHY18TruthThinNoChainTool", + ParticleSelectionString = "abs(TruthParticles.pdgId) == 5 || abs(TruthParticles.pdgId) == 12 || abs(TruthParticles.pdgId) == 14", + PreserveDescendants = False, + PreserveAncestors = False) +ToolSvc += BPHY18TruthThinNoChainTool + +#==================================================================== +# CREATE THE DERIVATION KERNEL ALGORITHM AND PASS THE ABOVE TOOLS +#==================================================================== + +thinningCollection = [ BPHY18_thinningTool_Tracks, BPHY18_thinningTool_GSFTracks, + BPHY18_thinningTool_PV, #BPHY18_thinningTool_PV_GSF, + BPHY18EgammaTPThinningTool, BPHY18MuonTPThinningTool + ] + +#if we're doing truth, add these [BPHY18TruthThinTool,BPHY18TruthThinNoChainTool] +if isSimulation: + thinningCollection += [BPHY18TruthThinTool,BPHY18TruthThinNoChainTool] + +print(thinningCollection) + +from DerivationFrameworkCore.DerivationFrameworkCoreConf import DerivationFramework__DerivationKernel +DerivationFrameworkJob += CfgMgr.DerivationFramework__DerivationKernel( + "BPHY18Kernel", + + AugmentationTools = [ ElectronPassLHvloosenod0,BPHY18DiElectronSelectAndWrite, + BPHY18_Select_DiElectrons, + BPHY18BeeKstSelectAndWrite, BPHY18_Select_BeeKst, BPHY18_Select_BeeKstbar, + BPHY18_diMeson_revertex, BPHY18_Select_Kpi, BPHY18_Select_piK ], + + #Only skim if not MC + SkimmingTools = [BPHY18SkimmingAND], + ThinningTools = thinningCollection + ) + + +#==================================================================== +# SET UP STREAM +#==================================================================== +streamName = derivationFlags.WriteDAOD_BPHY18Stream.StreamName +fileName = buildFileName( derivationFlags.WriteDAOD_BPHY18Stream ) +BPHY18Stream = MSMgr.NewPoolRootStream( streamName, fileName ) +BPHY18Stream.AcceptAlgs(["BPHY18Kernel"]) + +# Special lines for thinning +from AthenaServices.Configurables import ThinningSvc, createThinningSvc +augStream = MSMgr.GetStream( streamName ) +evtStream = augStream.GetEventStream() + +BPHY18ThinningSvc = createThinningSvc( svcName="BPHY18ThinningSvc", outStreams=[evtStream] ) +svcMgr += BPHY18ThinningSvc + +#==================================================================== +# Slimming +#==================================================================== + +# Added by ASC +from DerivationFrameworkCore.SlimmingHelper import SlimmingHelper +BPHY18SlimmingHelper = SlimmingHelper("BPHY18SlimmingHelper") +AllVariables = [] +StaticContent = [] +ExtraVariables = [] +BPHY18SlimmingHelper.SmartCollections = ["Electrons", "Muons", "InDetTrackParticles" ] + +# Needed for trigger objects +BPHY18SlimmingHelper.IncludeMuonTriggerContent = False +BPHY18SlimmingHelper.IncludeBPhysTriggerContent = False +BPHY18SlimmingHelper.IncludeEGammaTriggerContent = True + +## primary vertices +AllVariables += ["PrimaryVertices"] +StaticContent += ["xAOD::VertexContainer#BPHY18RefittedPrimaryVertices"] +StaticContent += ["xAOD::VertexAuxContainer#BPHY18RefittedPrimaryVerticesAux."] + +ExtraVariables += ["Muons.etaLayer1Hits.etaLayer2Hits.etaLayer3Hits.etaLayer4Hits.phiLayer1Hits.phiLayer2Hits.phiLayer3Hits.phiLayer4Hits", + "Muons.numberOfTriggerEtaLayers.numberOfPhiLayers", + "InDetTrackParticles.numberOfTRTHits.numberOfTRTHighThresholdHits.vx.vy.vz", + "PrimaryVertices.chiSquared.covariance", "Electrons.deltaEta1.DFCommonElectronsLHVeryLoosenod0","egammaClusters.calE.calEta.calPhi.e_sampl.eta_sampl.etaCalo.phiCalo.ETACALOFRAME.PHICALOFRAME","HLT_xAOD__ElectronContainer_egamma_ElectronsAuxDyn.charge"] + +## Jpsi candidates +StaticContent += ["xAOD::VertexContainer#%s" % BPHY18DiElectronSelectAndWrite.OutputVtxContainerName] +StaticContent += ["xAOD::VertexAuxContainer#%sAux.-vxTrackAtVertex" % BPHY18DiElectronSelectAndWrite.OutputVtxContainerName] + +StaticContent += ["xAOD::VertexContainer#%s" % BPHY18BeeKstSelectAndWrite.OutputVtxContainerName] +StaticContent += ["xAOD::VertexAuxContainer#%sAux.-vxTrackAtVertex" % BPHY18BeeKstSelectAndWrite.OutputVtxContainerName] + +StaticContent += ["xAOD::VertexContainer#%s" % BPHY18_diMeson_revertex.OutputVtxContainerName] +StaticContent += ["xAOD::VertexAuxContainer#%sAux." % BPHY18_diMeson_revertex.OutputVtxContainerName] +StaticContent += ["xAOD::VertexAuxContainer#%sAux.-vxTrackAtVertex" % BPHY18_diMeson_revertex.OutputVtxContainerName] + +AllVariables += [ "GSFTrackParticles"] + + +# Added by ASC +# Truth information for MC only +if isSimulation: + AllVariables += ["TruthEvents","TruthParticles","TruthVertices", "ElectronTruthParticles"] + +AllVariables = list(set(AllVariables)) # remove duplicates + +BPHY18SlimmingHelper.AllVariables = AllVariables +BPHY18SlimmingHelper.ExtraVariables = ExtraVariables + +BPHY18SlimmingHelper.StaticContent = StaticContent + +from DerivationFrameworkEGamma.ElectronsCPDetailedContent import * +BPHY18SlimmingHelper.ExtraVariables += ElectronsCPDetailedContent +BPHY18SlimmingHelper.ExtraVariables += GSFTracksCPDetailedContent + +BPHY18SlimmingHelper.AppendContentToStream(BPHY18Stream) diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/BPHY19.py b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/BPHY19.py new file mode 100644 index 00000000000..a8679449c86 --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/BPHY19.py @@ -0,0 +1,313 @@ +# +# Copyright (C) 2002-2019 CERN for the benefit of the ATLAS collaboration +# +#==================================================================== +# BPHY19.py +# Derivation for dimuon + photon conversion (chi_c/b) +# Contact: A. Chisholm <andrew.chisholm@cern.ch> +#==================================================================== + +# Set up common services and job object. +# This should appear in ALL derivation job options +from DerivationFrameworkCore.DerivationFrameworkMaster import * + +isSimulation = False +if globalflags.DataSource()=='geant4': + isSimulation = True + +print(isSimulation) + +#-------------------------------------------------------------------- +# Setup the JpsiFinder vertex fitter tools +#-------------------------------------------------------------------- + +include("DerivationFrameworkBPhys/configureVertexing.py") +BPHY19_VertexTools = BPHYVertexTools("BPHY19") + +from JpsiUpsilonTools.JpsiUpsilonToolsConf import Analysis__JpsiFinder +BPHY19JpsiFinder = Analysis__JpsiFinder( + name = "BPHY19JpsiFinder", + OutputLevel = INFO, + muAndMu = True, + muAndTrack = False, + TrackAndTrack = False, + assumeDiMuons = True, # If true, will assume dimu hypothesis and use PDG value for mu mass + invMassUpper = 100000.0, + invMassLower = 0.0, + Chi2Cut = 200., + oppChargesOnly = True, + atLeastOneComb = True, + useCombinedMeasurement = False, # Only takes effect if combOnly=True + muonCollectionKey = "Muons", + TrackParticleCollection = "InDetTrackParticles", + V0VertexFitterTool = BPHY19_VertexTools.TrkV0Fitter, # V0 vertex fitter + useV0Fitter = False, # if False a TrkVertexFitterTool will be used + TrkVertexFitterTool = BPHY19_VertexTools.TrkVKalVrtFitter, # VKalVrt vertex fitter + TrackSelectorTool = BPHY19_VertexTools.InDetTrackSelectorTool, + VertexPointEstimator = BPHY19_VertexTools.VtxPointEstimator, + useMCPCuts = False ) + +ToolSvc += BPHY19JpsiFinder +print(BPHY19JpsiFinder) + +#-------------------------------------------------------------------- +# Setup the vertex reconstruction tools +#-------------------------------------------------------------------- + +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__Reco_Vertex + +BPHY19_Reco_mumu = DerivationFramework__Reco_Vertex( + name = "BPHY19_Reco_mumu", + VertexSearchTool = BPHY19JpsiFinder, + OutputVtxContainerName = "BPHY19OniaCandidates", + PVContainerName = "PrimaryVertices", + RefPVContainerName = "BPHY19RefittedPrimaryVertices", + RefitPV = True, + MaxPVrefit = 100000, + DoVertexType = 7) + +ToolSvc += BPHY19_Reco_mumu +print(BPHY19_Reco_mumu) + +#-------------------------------------------------------------------- +# Setup the vertex selection and augmentation tools +#-------------------------------------------------------------------- + +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__Select_onia2mumu + +# psi(nS)->mu+mu- candidates +BPHY19_Select_Psi2mumu = DerivationFramework__Select_onia2mumu( + name = "BPHY19_Select_Psi2mumu", + HypothesisName = "Psi", + InputVtxContainerName = "BPHY19OniaCandidates", + VtxMassHypo = 3096.916, + MassMin = 2000.0, + MassMax = 4500.0, + Chi2Max = 200, + DoVertexType = 7) + +ToolSvc += BPHY19_Select_Psi2mumu +print(BPHY19_Select_Psi2mumu) + +# Y(nS)->mu+mu- candidates +BPHY19_Select_Upsi2mumu = DerivationFramework__Select_onia2mumu( + name = "BPHY19_Select_Upsi2mumu", + HypothesisName = "Upsi", + InputVtxContainerName = "BPHY19OniaCandidates", + VtxMassHypo = 9460.30, + MassMin = 8000.0, + MassMax = 12000.0, + Chi2Max = 200, + DoVertexType = 7) + +ToolSvc += BPHY19_Select_Upsi2mumu +print(BPHY19_Select_Upsi2mumu) + + +#-------------------------------------------------------------------- +# Configure the conversion finder +#-------------------------------------------------------------------- + +from TrkVKalVrtFitter.TrkVKalVrtFitterConf import Trk__TrkVKalVrtFitter +BPHY19_CascadeVertexFitter = Trk__TrkVKalVrtFitter( + name = "BPHY19_CascadeVertexFit", + Extrapolator = BPHY19_VertexTools.InDetExtrapolator, + #FirstMeasuredPoint = True, + FirstMeasuredPoint = False, + CascadeCnstPrecision = 1e-6, + MakeExtendedVertex = True) + +ToolSvc += BPHY19_CascadeVertexFitter +print(BPHY19_CascadeVertexFitter) + +include("DerivationFrameworkBPhys/configureConversionFinder.py") +BPHY19_ConvTools = BPHYConversionFinderTools("BPHY19") + +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__BPhysConversionFinder +BPHY19_ConversionFinder = DerivationFramework__BPhysConversionFinder( + name = "BPHY19_ConversionFinder", + VertexFitterTool = BPHY19_ConvTools.InDetSecVxFitterTool, + VertexEstimator = BPHY19_ConvTools.InDetSecVtxPointEstimator, + DistanceTool = BPHY19_ConvTools.InDetSecVxTrkDistanceFinder, + ConversionPostSelector = BPHY19_ConvTools.InDetSecVtxPostSelector, + CascadeFitter = BPHY19_CascadeVertexFitter, + InputTrackParticleContainerName = "InDetTrackParticles", + ConversionContainerName = "BPhysConversionCandidates", + DiMuonVertexContainer = "BPHY19OniaCandidates", + PassFlagsToCheck = ["passed_Psi","passed_Upsi"], + RequireDeltaM = True, # Only save conversion if it's a chi_c,b candidate (passes "MaxDeltaM" cut w.r.t. any di-muon candidate) + MaxDeltaM = 2000.0) + +ToolSvc += BPHY19_ConversionFinder +print(BPHY19_ConversionFinder) + +#-------------------------------------------------------------------- +# Select the events to save +#-------------------------------------------------------------------- + +# Require at least one conversion AND one di-muon +BPHY19_expression = "count(BPhysConversionCandidates.passed) > 0 && (count(BPHY19OniaCandidates.passed_Psi) > 0 || count(BPHY19OniaCandidates.passed_Upsi) > 0)" + +from DerivationFrameworkTools.DerivationFrameworkToolsConf import DerivationFramework__xAODStringSkimmingTool +BPHY19_SelectEvent = DerivationFramework__xAODStringSkimmingTool(name = "BPHY19_SelectEvent", + expression = BPHY19_expression) +ToolSvc += BPHY19_SelectEvent +print(BPHY19_SelectEvent) + +#-------------------------------------------------------------------- +# Thin Collections +#-------------------------------------------------------------------- + +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__Thin_vtxTrk + +# Keep tracks from di-muon vertices +BPHY19Thin_vtxTrk = DerivationFramework__Thin_vtxTrk( + name = "BPHY19Thin_vtxTrk", + TrackParticleContainerName = "InDetTrackParticles", + VertexContainerNames = ["BPHY19OniaCandidates"], + PassFlags = ["passed_Psi", "passed_Upsi"] ) + +ToolSvc += BPHY19Thin_vtxTrk + +# Keep tracks from conversions +BPHY19Thin_ConvTrk = DerivationFramework__Thin_vtxTrk( + name = "BPHY19Thin_ConvTrk", + TrackParticleContainerName = "InDetTrackParticles", + VertexContainerNames = ["BPhysConversionCandidates"], + PassFlags = ["passed"] ) + +ToolSvc += BPHY19Thin_ConvTrk + +# Keep tracks from all muons +from DerivationFrameworkInDet.DerivationFrameworkInDetConf import DerivationFramework__MuonTrackParticleThinning +BPHY19MuonTPThinningTool = DerivationFramework__MuonTrackParticleThinning(name = "BPHY19MuonTPThinningTool", + MuonKey = "Muons", + InDetTrackParticlesKey = "InDetTrackParticles") + +ToolSvc += BPHY19MuonTPThinningTool + +#-------------------------------------------------------------------- +# Truth Particle Thinning +#-------------------------------------------------------------------- + +BPHY19_TruthIDString = "" +BPHY19_TruthIDString += "TruthParticles.pdgId == 443" #J/psi +BPHY19_TruthIDString += " || " +BPHY19_TruthIDString += "TruthParticles.pdgId == 100443" #psi(2S) +BPHY19_TruthIDString += " || " +BPHY19_TruthIDString += "TruthParticles.pdgId == 10441" #chi_c0(1P) +BPHY19_TruthIDString += " || " +BPHY19_TruthIDString += "TruthParticles.pdgId == 20443" #chi_c1(1P) +BPHY19_TruthIDString += " || " +BPHY19_TruthIDString += "TruthParticles.pdgId == 445" #chi_c2(1P) +BPHY19_TruthIDString += " || " +BPHY19_TruthIDString += "TruthParticles.pdgId == 553" #Y(1S) +BPHY19_TruthIDString += " || " +BPHY19_TruthIDString += "TruthParticles.pdgId == 100553" #Y(2S) +BPHY19_TruthIDString += " || " +BPHY19_TruthIDString += "TruthParticles.pdgId == 200553" #Y(3S) +BPHY19_TruthIDString += " || " +BPHY19_TruthIDString += "TruthParticles.pdgId == 10551" #chi_b0(1P) +BPHY19_TruthIDString += " || " +BPHY19_TruthIDString += "TruthParticles.pdgId == 110551" #chi_b0(2P) +BPHY19_TruthIDString += " || " +BPHY19_TruthIDString += "TruthParticles.pdgId == 210551" #chi_b0(3P) +BPHY19_TruthIDString += " || " +BPHY19_TruthIDString += "TruthParticles.pdgId == 20553" #chi_b1(1P) +BPHY19_TruthIDString += " || " +BPHY19_TruthIDString += "TruthParticles.pdgId == 120553" #chi_b1(2P) +BPHY19_TruthIDString += " || " +BPHY19_TruthIDString += "TruthParticles.pdgId == 220553" #chi_b1(3P) +BPHY19_TruthIDString += " || " +BPHY19_TruthIDString += "TruthParticles.pdgId == 555" #chi_b2(1P) +BPHY19_TruthIDString += " || " +BPHY19_TruthIDString += "TruthParticles.pdgId == 100555" #chi_b2(2P) +BPHY19_TruthIDString += " || " +BPHY19_TruthIDString += "TruthParticles.pdgId == 200555" #chi_b2(3P) + +print("PDG IDs to save:") +print(BPHY19_TruthIDString) + +# Only save truth informtion directly associated with Onia +from DerivationFrameworkMCTruth.DerivationFrameworkMCTruthConf import DerivationFramework__GenericTruthThinning +BPHY19TruthThinTool = DerivationFramework__GenericTruthThinning(name = "BPHY19TruthThinTool", + ParticleSelectionString = BPHY19_TruthIDString, + PreserveDescendants = True, + PreserveAncestors = True) +ToolSvc += BPHY19TruthThinTool +print(BPHY19TruthThinTool) + +#-------------------------------------------------------------------- +# Create the derivation kernel +#-------------------------------------------------------------------- + +BPHY19ThinningTools = [BPHY19Thin_vtxTrk, BPHY19MuonTPThinningTool, BPHY19Thin_ConvTrk] +if isSimulation: + BPHY19ThinningTools.append(BPHY19TruthThinTool) + +from DerivationFrameworkCore.DerivationFrameworkCoreConf import DerivationFramework__DerivationKernel +DerivationFrameworkJob += CfgMgr.DerivationFramework__DerivationKernel( + "BPHY19Kernel", + AugmentationTools = [BPHY19_Reco_mumu, BPHY19_Select_Psi2mumu, BPHY19_Select_Upsi2mumu,BPHY19_ConversionFinder], + SkimmingTools = [BPHY19_SelectEvent], + ThinningTools = BPHY19ThinningTools + ) + +#-------------------------------------------------------------------- +# Create the stream +#-------------------------------------------------------------------- + +streamName = derivationFlags.WriteDAOD_BPHY19Stream.StreamName +fileName = buildFileName( derivationFlags.WriteDAOD_BPHY19Stream ) +BPHY19Stream = MSMgr.NewPoolRootStream( streamName, fileName ) +BPHY19Stream.AcceptAlgs(["BPHY19Kernel"]) +# Special lines for thinning +# Thinning service name must match the one passed to the thinning tools +from AthenaServices.Configurables import ThinningSvc, createThinningSvc +augStream = MSMgr.GetStream( streamName ) +evtStream = augStream.GetEventStream() +svcMgr += createThinningSvc( svcName="BPHY19ThinningSvc", outStreams=[evtStream] ) + +#-------------------------------------------------------------------- +# Generic Collection Slimming +#-------------------------------------------------------------------- + +from DerivationFrameworkCore.SlimmingHelper import SlimmingHelper +BPHY19SlimmingHelper = SlimmingHelper("BPHY19SlimmingHelper") +AllVariables = [] +StaticContent = [] + +# Trigger informtion +BPHY19SlimmingHelper.IncludeMuonTriggerContent = True +BPHY19SlimmingHelper.IncludeBPhysTriggerContent = True + +## Primary vertices +AllVariables += ["PrimaryVertices"] +StaticContent += ["xAOD::VertexContainer#BPHY19RefittedPrimaryVertices"] +StaticContent += ["xAOD::VertexAuxContainer#BPHY19RefittedPrimaryVerticesAux."] + +## ID track particles +AllVariables += ["InDetTrackParticles"] +AllVariables += ["CombinedMuonTrackParticles"] +AllVariables += ["ExtrapolatedMuonTrackParticles"] + +## Muon container +AllVariables += ["Muons"] + +## Di-muon candidates +StaticContent += ["xAOD::VertexContainer#%s" % BPHY19_Reco_mumu.OutputVtxContainerName] +## we have to disable vxTrackAtVertex branch since it is not xAOD compatible +StaticContent += ["xAOD::VertexAuxContainer#%sAux.-vxTrackAtVertex" % BPHY19_Reco_mumu.OutputVtxContainerName] + +# Conversions +StaticContent += ["xAOD::VertexContainer#%s" % BPHY19_ConversionFinder.ConversionContainerName] +## we have to disable vxTrackAtVertex branch since it is not xAOD compatible +StaticContent += ["xAOD::VertexAuxContainer#%sAux.-vxTrackAtVertex" % BPHY19_ConversionFinder.ConversionContainerName] + +# Truth information for MC only +if isSimulation: + AllVariables += ["TruthEvents","TruthParticles","TruthVertices","MuonTruthParticles"] + +BPHY19SlimmingHelper.AllVariables = AllVariables +BPHY19SlimmingHelper.StaticContent = StaticContent +BPHY19SlimmingHelper.AppendContentToStream(BPHY19Stream) diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/BPHY2.py b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/BPHY2.py new file mode 100644 index 00000000000..ea9966593f4 --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/BPHY2.py @@ -0,0 +1,332 @@ +#==================================================================== +# BPHY2.py +# BPHY2 repurposed for students analysis of Bs->Upsilon phi Analysis +# It requires the reductionConf flag BPHY2 in Reco_tf.py +#==================================================================== + +# Set up common services and job object. +# This should appear in ALL derivation job options +from DerivationFrameworkCore.DerivationFrameworkMaster import * + +isSimulation = False +if globalflags.DataSource()=='geant4': + isSimulation = True + +print(isSimulation) + + +#==================================================================== +# AUGMENTATION TOOLS +#==================================================================== + +include("DerivationFrameworkBPhys/configureVertexing.py") +BPHY2_VertexTools = BPHYVertexTools("BPHY2") + + +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__AugOriginalCounts +BPHY2_AugOriginalCounts = DerivationFramework__AugOriginalCounts( + name = "BPHY2_AugOriginalCounts", + VertexContainer = "PrimaryVertices", + TrackContainer = "InDetTrackParticles" ) +ToolSvc += BPHY2_AugOriginalCounts + + +#-------------------------------------------------------------------- +## 2/ setup JpsiFinder tool +## These are general tools independent of DerivationFramework that do the +## actual vertex fitting and some pre-selection. +from JpsiUpsilonTools.JpsiUpsilonToolsConf import Analysis__JpsiFinder +BPHY2JpsiFinder = Analysis__JpsiFinder(name = "BPHY2JpsiFinder", + OutputLevel = INFO, + muAndMu = True, + muAndTrack = False, + TrackAndTrack = False, + assumeDiMuons = True, + invMassUpper = 4700.0, + invMassLower = 2600.0, + Chi2Cut = 15., + oppChargesOnly = True, + combOnly = True, + atLeastOneComb = False, + useCombinedMeasurement = False, # Only takes effect if combOnly=True + muonCollectionKey = "Muons", + TrackParticleCollection = "InDetTrackParticles", + V0VertexFitterTool = BPHY2_VertexTools.TrkV0Fitter, # V0 vertex fitter + useV0Fitter = False, # if False a TrkVertexFitterTool will be used + TrkVertexFitterTool = BPHY2_VertexTools.TrkVKalVrtFitter, # VKalVrt vertex fitter + TrackSelectorTool = BPHY2_VertexTools.InDetTrackSelectorTool, + VertexPointEstimator = BPHY2_VertexTools.VtxPointEstimator, + useMCPCuts = False) +ToolSvc += BPHY2JpsiFinder +print(BPHY2JpsiFinder) + +#-------------------------------------------------------------------- +## 3/ setup the vertex reconstruction "call" tool(s). They are part of the derivation framework. +## These Augmentation tools add output vertex collection(s) into the StoreGate and add basic +## decorations which do not depend on the vertex mass hypothesis (e.g. lxy, ptError, etc). +## There should be one tool per topology, i.e. Jpsi and Psi(2S) do not need two instance of the +## Reco tool is the JpsiFinder mass window is wide enough. +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__Reco_Vertex +BPHY2JpsiSelectAndWrite = DerivationFramework__Reco_Vertex(name = "BPHY2JpsiSelectAndWrite", + VertexSearchTool = BPHY2JpsiFinder, + OutputVtxContainerName = "BPHY2JpsiCandidates", + PVContainerName = "PrimaryVertices", + RefPVContainerName = "SHOULDNOTBEUSED", + DoVertexType =1) +ToolSvc += BPHY2JpsiSelectAndWrite +print(BPHY2JpsiSelectAndWrite) + + +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__Select_onia2mumu + + + +## 4/ setup a new vertexing tool (necessary due to use of mass constraint) +from TrkVKalVrtFitter.TrkVKalVrtFitterConf import Trk__TrkVKalVrtFitter +BPHY2BsKKVertexFit = Trk__TrkVKalVrtFitter( + name = "BPHY2BsKKVertexFit", + Extrapolator = BPHY2_VertexTools.InDetExtrapolator, + FirstMeasuredPoint = True, + MakeExtendedVertex = True) +ToolSvc += BPHY2BsKKVertexFit +print(BPHY2BsKKVertexFit) + + + +## 5/ setup the Jpsi+2 track finder +from JpsiUpsilonTools.JpsiUpsilonToolsConf import Analysis__JpsiPlus2Tracks +BPHY2BsJpsiKK = Analysis__JpsiPlus2Tracks(name = "BPHY2BsJpsiKK", + OutputLevel = INFO, +kaonkaonHypothesis = True, +pionpionHypothesis = False, +kaonpionHypothesis = False, +trkThresholdPt = 800.0, +trkMaxEta = 3.0, +BMassUpper = 5800.0, +BMassLower = 5000.0, +DiTrackMassUpper = 1019.445 + 100., +DiTrackMassLower = 1019.445 - 100., +Chi2Cut = 8.0, +TrkQuadrupletMassUpper = 6000.0, +TrkQuadrupletMassLower = 4800.0, +JpsiContainerKey = "BPHY2JpsiCandidates", +TrackParticleCollection = "InDetTrackParticles", +MuonsUsedInJpsi = "Muons", +TrkVertexFitterTool = BPHY2BsKKVertexFit, +TrackSelectorTool = BPHY2_VertexTools.InDetTrackSelectorTool, +UseMassConstraint = False) + +ToolSvc += BPHY2BsJpsiKK +print(BPHY2BsJpsiKK) + + + + +## 6/ setup the combined augmentation/skimming tool for the Bpm +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__Reco_Vertex +BPHY2BsKKSelectAndWrite = DerivationFramework__Reco_Vertex(name = "BPHY2BsKKSelectAndWrite", + VertexSearchTool = BPHY2BsJpsiKK, + OutputVtxContainerName = "BPHY2BsJpsiKKCandidates", + PVContainerName = "PrimaryVertices", + RefPVContainerName = "BPHY2RefittedPrimaryVertices", + RefitPV = True, + MaxPVrefit = 10000, DoVertexType = 7) +ToolSvc += BPHY2BsKKSelectAndWrite +print(BPHY2BsKKSelectAndWrite) + + +## b/ augment and select Psi(2S)->mumu candidates +BPHY2_Select_Psi2mumu = DerivationFramework__Select_onia2mumu( + name = "BPHY2_Select_Psi2mumu", + HypothesisName = "Psi", + InputVtxContainerName = "BPHY2JpsiCandidates", + VtxMassHypo = 3686.09, + MassMin = 3300.0, + MassMax = 4500.0, + Chi2Max = 200, + DoVertexType = 7) + +ToolSvc += BPHY2_Select_Psi2mumu +print(BPHY2_Select_Psi2mumu) + + +## a/ augment and select Jpsi->mumu candidates +BPHY2_Select_Jpsi2mumu = DerivationFramework__Select_onia2mumu( + name = "BPHY2_Select_Jpsi2mumu", + HypothesisName = "Jpsi", + InputVtxContainerName = "BPHY2JpsiCandidates", + VtxMassHypo = 3096.916, + MassMin = 2000.0, + MassMax = 3600.0, + Chi2Max = 200, + DoVertexType = 7) + +ToolSvc += BPHY2_Select_Jpsi2mumu +print(BPHY2_Select_Jpsi2mumu) + +## b/ augment and select Bs->JpsiKK candidates +BPHY2_Select_Bs2JpsiKK = DerivationFramework__Select_onia2mumu( + name = "BPHY2_Select_Bs2JpsiKK", + HypothesisName = "Bs", + InputVtxContainerName = "BPHY2BsJpsiKKCandidates", + TrkMasses = [105.658, 105.658, 493.677, 493.677], + VtxMassHypo = 5366.3, + MassMin = 5000.0, + MassMax = 5800.0, + Chi2Max = 200) + +ToolSvc += BPHY2_Select_Bs2JpsiKK +print(BPHY2_Select_Bs2JpsiKK) + +#==================================================================== +# SET UP STREAM +#==================================================================== +streamName = derivationFlags.WriteDAOD_BPHY2Stream.StreamName +fileName = buildFileName( derivationFlags.WriteDAOD_BPHY2Stream ) +BPHY2Stream = MSMgr.NewPoolRootStream( streamName, fileName ) +BPHY2Stream.AcceptAlgs(["BPHY2Kernel"]) + +# Special lines for thinning +# Thinning service name must match the one passed to the thinning tools +augStream = MSMgr.GetStream( streamName ) + + +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__Thin_vtxTrk +BPHY2_thinningTool_Tracks = DerivationFramework__Thin_vtxTrk( + name = "BPHY2_thinningTool_Tracks", + TrackParticleContainerName = "InDetTrackParticles", + StreamName = streamName, + VertexContainerNames = ["BPHY2BsJpsiKKCandidates", "BPHY2JpsiCandidates"], + PassFlags = ["passed_Bs", "passed_Psi", "passed_Jpsi"] ) + +ToolSvc += BPHY2_thinningTool_Tracks + +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__BPhysPVThinningTool +BPHY2_thinningTool_PV = DerivationFramework__BPhysPVThinningTool( + name = "BPHY2_thinningTool_PV", + CandidateCollections = ["BPHY2BsJpsiKKCandidates"], + StreamName = streamName, + KeepPVTracks =True) + +ToolSvc += BPHY2_thinningTool_PV + +if not isSimulation: #Only Skim Data + from DerivationFrameworkTools.DerivationFrameworkToolsConf import DerivationFramework__xAODStringSkimmingTool + + BPHY2_SelectBsJpsiKKEvent = DerivationFramework__xAODStringSkimmingTool( + name = "BPHY2_SelectBsJpsiKKEvent", + expression = "count(BPHY2BsJpsiKKCandidates.passed_Bs > 0) > 0") + ToolSvc += BPHY2_SelectBsJpsiKKEvent + print(BPHY2_SelectBsJpsiKKEvent) + + + #==================================================================== + # Make event selection based on an OR of the input skimming tools + #==================================================================== + from DerivationFrameworkTools.DerivationFrameworkToolsConf import DerivationFramework__FilterCombinationOR + BPHY2SkimmingOR = CfgMgr.DerivationFramework__FilterCombinationOR("BPHY2SkimmingOR", + FilterList = [BPHY2_SelectBsJpsiKKEvent ])#, BPHY2_SelectBplJpsiKplEventBc + PassFlags = ["passed_Bs"] + ToolSvc += BPHY2SkimmingOR + print(BPHY2SkimmingOR) + + + +## b) thinning out tracks that are not attached to muons. The final thinning decision is based on the OR operation +## between decision from this and the previous tools. +from DerivationFrameworkInDet.DerivationFrameworkInDetConf import DerivationFramework__MuonTrackParticleThinning +BPHY2MuonTPThinningTool = DerivationFramework__MuonTrackParticleThinning(name = "BPHY2MuonTPThinningTool", + MuonKey = "Muons", + StreamName = streamName, + InDetTrackParticlesKey = "InDetTrackParticles") +ToolSvc += BPHY2MuonTPThinningTool + + +#==================================================================== +# CREATE THE DERIVATION KERNEL ALGORITHM AND PASS THE ABOVE TOOLS +#==================================================================== + +thiningCollection = [BPHY2_thinningTool_Tracks, BPHY2_thinningTool_PV, BPHY2MuonTPThinningTool] +print(thiningCollection) + +BPHY2Seq = CfgMgr.AthSequencer("BPHY2Sequence") +DerivationFrameworkJob += BPHY2Seq + +# The name of the kernel (BPHY2Kernel in this case) must be unique to this derivation +from DerivationFrameworkCore.DerivationFrameworkCoreConf import DerivationFramework__DerivationKernel +BPHY2Seq += CfgMgr.DerivationFramework__DerivationKernel("BPHY2Kernel", + AugmentationTools = [ BPHY2JpsiSelectAndWrite, + BPHY2BsKKSelectAndWrite, + BPHY2_Select_Psi2mumu, + BPHY2_Select_Jpsi2mumu, BPHY2_Select_Bs2JpsiKK, + BPHY2_AugOriginalCounts], + #Only skim if not MC + SkimmingTools = [BPHY2SkimmingOR] if not isSimulation else [], + ThinningTools = thiningCollection + + ) + + +#==================================================================== +# Slimming +#==================================================================== + +# Added by ASC +from DerivationFrameworkCore.SlimmingHelper import SlimmingHelper +BPHY2SlimmingHelper = SlimmingHelper("BPHY2SlimmingHelper") +AllVariables = [] +StaticContent = [] + +# Needed for trigger objects +BPHY2SlimmingHelper.IncludeMuonTriggerContent = TRUE +BPHY2SlimmingHelper.IncludeBPhysTriggerContent = TRUE +SmartVar = [] +## primary vertices +SmartVar += ["PrimaryVertices"] +StaticContent += ["xAOD::VertexContainer#BPHY2RefittedPrimaryVertices"] +StaticContent += ["xAOD::VertexAuxContainer#BPHY2RefittedPrimaryVerticesAux."] + + + +## ID track particles +AllVariables += ["InDetTrackParticles"] + +## combined / extrapolated muon track particles +## (note: for tagged muons there is no extra TrackParticle collection since the ID tracks +## are store in InDetTrackParticles collection) +AllVariables += ["CombinedMuonTrackParticles"] + + +## muon container +SmartVar += ["Muons"] + + +## Jpsi candidates +StaticContent += ["xAOD::VertexContainer#%s" % BPHY2JpsiSelectAndWrite.OutputVtxContainerName] +## we have to disable vxTrackAtVertex branch since it is not xAOD compatible +StaticContent += ["xAOD::VertexAuxContainer#%sAux.-vxTrackAtVertex" % BPHY2JpsiSelectAndWrite.OutputVtxContainerName] + +StaticContent += ["xAOD::VertexContainer#%s" % BPHY2BsKKSelectAndWrite.OutputVtxContainerName] +StaticContent += ["xAOD::VertexAuxContainer#%sAux.-vxTrackAtVertex" % BPHY2BsKKSelectAndWrite.OutputVtxContainerName] + + +# Tagging information (in addition to that already requested by usual algorithms) +AllVariables += ["MuonSpectrometerTrackParticles" ] + + + + + +# Added by ASC +# Truth information for MC only +if isSimulation: + AllVariables += ["TruthEvents","TruthParticles","TruthVertices","MuonTruthParticles" ] + + +AllVariables = list(set(AllVariables)) # remove duplicates + +BPHY2SlimmingHelper.AllVariables = AllVariables +BPHY2SlimmingHelper.StaticContent = StaticContent +BPHY2SlimmingHelper.SmartCollections = SmartVar + +BPHY2SlimmingHelper.AppendContentToStream(BPHY2Stream) diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/BPHY20.py b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/BPHY20.py new file mode 100644 index 00000000000..1e8f6f8fd31 --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/BPHY20.py @@ -0,0 +1,775 @@ +#==================================================================== +# +# Copyright (C) 2002-2019 CERN for the benefit of the ATLAS collaboration +# +# BPHY20.py +# +# R_Jpsi analysis (currently muon channel only) +# +#==================================================================== +from AthenaCommon.AppMgr import ServiceMgr as svcMgr +if not hasattr(svcMgr, 'ItemListSvc'): svcMgr += CfgMgr.ItemListSvc() +svcMgr.ItemListSvc.OutputLevel = DEBUG + +#==================================================================== +# FLAGS TO PERSONALIZE THE DERIVATION +#==================================================================== + +onlyAugmentations = False # Set to True to deactivate thinning and skimming, and only keep augmentations (to generate a sample with full xAOD plus all the extra) +thinTruth = True +addMuExtrapolationForTrigger = True + +from DerivationFrameworkCore.DerivationFrameworkMaster import * +isSimulation = False +if globalflags.DataSource()=='geant4': + isSimulation = True + +print(isSimulation) + + +from DerivationFrameworkJetEtMiss.JetCommon import * +from DerivationFrameworkJetEtMiss.METCommon import * + +#==================================================================== +# SET UP STREAM +#==================================================================== +streamName = derivationFlags.WriteDAOD_BPHY20Stream.StreamName +fileName = buildFileName( derivationFlags.WriteDAOD_BPHY20Stream ) + +BPHY20Stream = MSMgr.NewPoolRootStream( streamName, fileName ) +BPHY20Stream.AcceptAlgs(["BPHY20Kernel"]) + +#==================================================================== +# AUGMENTATION TOOLS +#==================================================================== +## 1/ setup vertexing tools and services + +include("DerivationFrameworkBPhys/configureVertexing.py") +BPHY20_VertexTools = BPHYVertexTools("BPHY20") + +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__AugOriginalCounts +BPHY20_AugOriginalCounts = DerivationFramework__AugOriginalCounts( + name = "BPHY20_AugOriginalCounts", + VertexContainer = "PrimaryVertices", + TrackContainer = "InDetTrackParticles" ) +ToolSvc += BPHY20_AugOriginalCounts + +#==================================================================== +# TriggerCounting for Kernel1 (from BPHY7) +#==================================================================== +#List of trigggers to be counted (high Sig-eff*Lumi ones are in) +triggersToMetadata= [ +### 2018 +"HLT_mu11_mu6_bJpsimumu", +"HLT_2mu10_bJpsimumu", +"HLT_mu6_2mu4_bJpsi", +"HLT_3mu4_bJpsi", +"HLT_mu11_mu6_bJpsimumu_Lxy0", +"HLT_3mu6_bJpsi", +"HLT_mu11_mu6_bJpsimumu_L1LFV-MU11", +"HLT_2mu6_bJpsimumu_L1BPH-2M9-2MU6_BPH-2DR15-2MU6", +"HLT_mu11_mu6_bJpsimumu_Lxy0_L1LFV-MU11", +"HLT_2mu6_bJpsimumu_Lxy0_L1BPH-2M9-2MU6_BPH-2DR15-2MU6", +"HLT_mu6_mu4_bJpsimumu_Lxy0_L1BPH-2M9-MU6MU4_BPH-0DR15-MU6MU4", +"HLT_2mu4_bJpsimumu_Lxy0_L1BPH-2M9-2MU4_BPH-0DR15-2MU4", +"HLT_mu6_mu4_bJpsimumu_L1BPH-2M9-MU6MU4_BPH-0DR15-MU6MU4", +"HLT_mu10_bJpsi_TrkPEB", +"HLT_mu6_bJpsi_TrkPEB", +"HLT_2mu6_bJpsimumu", +"HLT_mu6_mu2noL1_msonly_bJpsimumu_noid_PEB", + +### 2017 +#HLT_2mu10_bJpsimumu +"HLT_2mu10_bJpsimumu_noL2", +"HLT_2mu6_bJpsimumu", +#HLT_2mu6_bJpsimumu_L1BPH-2M9-2MU6_BPH-2DR15-2MU6 +"HLT_2mu6_bJpsimumu_Lxy0", +#HLT_2mu6_bJpsimumu_Lxy0_L1BPH-2M9-2MU6_BPH-2DR15-2MU6 +#HLT_3mu4_bJpsi +#HLT_3mu6_bJpsi +#HLT_mu10_bJpsi_TrkPEB +"HLT_mu10_mu6_bJpsimumu", +"HLT_mu10_mu6_bJpsimumu_Lxy0", +#HLT_mu11_mu6_bJpsimumu +#HLT_mu11_mu6_bJpsimumu_Lxy0 +"HLT_mu14_bJpsi_Trkloose", +"HLT_mu14_bJpsi_TrkPEB", +"HLT_mu20_2mu2noL1_JpsimumuFS", +"HLT_mu20_2mu4_JpsimumuL2", +"HLT_mu20_bJpsi_Trkloose", +"HLT_mu20_bJpsi_TrkPEB", +#HLT_mu6_bJpsi_TrkPEB +#HLT_mu6_mu4_bJpsimumu_L1BPH-2M9-MU6MU4_BPH-0DR15-MU6MU4 +#HLT_mu6_mu4_bJpsimumu_Lxy0_L1BPH-2M9-MU6MU4_BPH-0DR15-MU6MU4 + +### 2016 +#HLT_2mu10_bJpsimumu +"HLT_2mu10_bJpsimumu_delayed", +"HLT_2mu10_bJpsimumu_noL2", +"HLT_2mu4_bJpsimumu_delayed_L1BPH-2M8-2MU4", +"HLT_2mu4_bJpsimumu_L1BPH-2M8-2MU4", +"HLT_2mu4_bJpsimumu_Lxy0_delayed_L1BPH-2M8-2MU4", +"HLT_2mu6_bJpsimumu", +"HLT_2mu6_bJpsimumu_delayed", +"HLT_2mu6_bJpsimumu_delayed_L1BPH-2M9-2MU6_BPH-2DR15-2MU6", +#HLT_2mu6_bJpsimumu_L1BPH-2M9-2MU6_BPH-2DR15-2MU6 +"HLT_2mu6_bJpsimumu_Lxy0", +"HLT_2mu6_bJpsimumu_Lxy0_delayed", +"HLT_2mu6_bJpsimumu_Lxy0_delayed_L1BPH-2M9-2MU6_BPH-2DR15-2MU6", +#HLT_2mu6_bJpsimumu_Lxy0_L1BPH-2M9-2MU6_BPH-2DR15-2MU6 +#HLT_3mu4_bJpsi +"HLT_3mu4_bJpsi_delayed", +#HLT_3mu6_bJpsi +"HLT_mu10_mu6_bJpsimumu", +"HLT_mu10_mu6_bJpsimumu_delayed", +"HLT_mu10_mu6_bJpsimumu_Lxy0", +"HLT_mu10_mu6_bJpsimumu_Lxy0_delayed", +"HLT_mu18_bJpsi_Trkloose", +"HLT_mu20_2mu0noL1_JpsimumuFS", +"HLT_mu20_2mu4_JpsimumuL2", +#HLT_mu6_2mu4_bJpsi +"HLT_mu6_2mu4_bJpsi_delayed", +"HLT_mu6_mu4_bJpsimumu", +"HLT_mu6_mu4_bJpsimumu_delayed", +"HLT_mu6_mu4_bJpsimumu_delayed_L1BPH-2M8-MU6MU4_BPH-0DR15-MU6MU4", +"HLT_mu6_mu4_bJpsimumu_delayed_L1MU6MU4-BO", +"HLT_mu6_mu4_bJpsimumu_L12MU4-B", +"HLT_mu6_mu4_bJpsimumu_L1BPH-2M8-MU6MU4_BPH-0DR15-MU6MU4", +"HLT_mu6_mu4_bJpsimumu_L1MU6MU4-BO", +"HLT_mu6_mu4_bJpsimumu_Lxy0", +"HLT_mu6_mu4_bJpsimumu_Lxy0_delayed", +"HLT_mu6_mu4_bJpsimumu_Lxy0_delayed_L1BPH-2M8-MU6MU4_BPH-0DR15-MU6MU4", +"HLT_mu6_mu4_bJpsimumu_Lxy0_L1BPH-2M8-MU6MU4_BPH-0DR15-MU6MU4", + +### 2015 +#HLT_2mu10_bJpsimumu +#HLT_2mu10_bJpsimumu_noL2 +"HLT_2mu10_l2msonly_bJpsimumu_noL2", +"HLT_2mu4_bJpsimumu", +"HLT_2mu4_bJpsimumu_noL2", +"HLT_2mu4_l2msonly_bJpsimumu_noL2", +#HLT_2mu6_bJpsimumu +"HLT_2mu6_bJpsimumu_noL2", +"HLT_2mu6_l2msonly_bJpsimumu_noL2", +#HLT_3mu4_bJpsi +#HLT_3mu6_bJpsi +"HLT_mu10_mu10_l2msonly_bJpsimumu_noL2", +"HLT_mu18_2mu0noL1_JpsimumuFS", +"HLT_mu18_2mu4_JpsimumuL2", +#HLT_mu18_bJpsi_Trkloose +#HLT_mu20_2mu0noL1_JpsimumuFS +#HLT_mu20_2mu4_JpsimumuL2 +"HLT_mu4_mu4_l2msonly_bJpsimumu_noL2", +"HLT_mu6_l2msonly_mu4_bJpsimumu_noL2", +"HLT_mu6_l2msonly_mu4_l2msonly_bJpsimumu_noL2", +#HLT_mu6_mu4_bJpsimumu +"HLT_mu6_mu4_bJpsimumu_noL2", +"HLT_mu6_mu4_l2msonly_bJpsimumu_noL2", +"HLT_mu6_mu6_l2msonly_bJpsimumu_noL2", + +"HLT_mu4","HLT_mu6","HLT_mu10","HLT_mu14","HLT_mu18","HLT_mu24" #5% + + ] + +triggersToMetadata_filter = list( set(triggersToMetadata) ) + +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__TriggerCountToMetadata +BPHY20TriggerCountToMetadata = DerivationFramework__TriggerCountToMetadata(name = "BPHY20TriggerCount", + TriggerList = triggersToMetadata_filter, + FolderName = "BPHY20") + +ToolSvc += BPHY20TriggerCountToMetadata + +#==================================================================== +# PRESELECTION for Kernel1 #Added by Matteo +#==================================================================== +## 1/ Setup the skimming based on triggers +## + +triggerList = [ +## 2018 +"HLT_mu11_mu6_bJpsimumu", +"HLT_2mu10_bJpsimumu", +"HLT_mu6_2mu4_bJpsi", +"HLT_3mu4_bJpsi", +"HLT_mu11_mu6_bJpsimumu_Lxy0", +"HLT_3mu6_bJpsi", +"HLT_mu11_mu6_bJpsimumu_L1LFV-MU11", +"HLT_2mu6_bJpsimumu_L1BPH-2M9-2MU6_BPH-2DR15-2MU6", +"HLT_mu11_mu6_bJpsimumu_Lxy0_L1LFV-MU11", +"HLT_2mu6_bJpsimumu_Lxy0_L1BPH-2M9-2MU6_BPH-2DR15-2MU6", +"HLT_mu6_mu4_bJpsimumu_Lxy0_L1BPH-2M9-MU6MU4_BPH-0DR15-MU6MU4", +"HLT_2mu4_bJpsimumu_Lxy0_L1BPH-2M9-2MU4_BPH-0DR15-2MU4", +"HLT_mu6_mu4_bJpsimumu_L1BPH-2M9-MU6MU4_BPH-0DR15-MU6MU4", +"HLT_mu10_bJpsi_TrkPEB", +"HLT_mu6_bJpsi_TrkPEB", +"HLT_2mu6_bJpsimumu", +"HLT_mu6_mu2noL1_msonly_bJpsimumu_noid_PEB", + + +"HLT_mu22_mu8noL1_TagandProbe", + +### 2017 +#HLT_2mu10_bJpsimumu +"HLT_2mu10_bJpsimumu_noL2", +"HLT_2mu6_bJpsimumu", +#HLT_2mu6_bJpsimumu_L1BPH-2M9-2MU6_BPH-2DR15-2MU6 +"HLT_2mu6_bJpsimumu_Lxy0", +#HLT_2mu6_bJpsimumu_Lxy0_L1BPH-2M9-2MU6_BPH-2DR15-2MU6 +#HLT_3mu4_bJpsi +#HLT_3mu6_bJpsi +#HLT_mu10_bJpsi_TrkPEB +"HLT_mu10_mu6_bJpsimumu", +"HLT_mu10_mu6_bJpsimumu_Lxy0", +#HLT_mu11_mu6_bJpsimumu +#HLT_mu11_mu6_bJpsimumu_Lxy0 +"HLT_mu14_bJpsi_Trkloose", +"HLT_mu14_bJpsi_TrkPEB", +"HLT_mu20_2mu2noL1_JpsimumuFS", +"HLT_mu20_2mu4_JpsimumuL2", +"HLT_mu20_bJpsi_Trkloose", +"HLT_mu20_bJpsi_TrkPEB", +#HLT_mu6_bJpsi_TrkPEB +#HLT_mu6_mu4_bJpsimumu_L1BPH-2M9-MU6MU4_BPH-0DR15-MU6MU4 +#HLT_mu6_mu4_bJpsimumu_Lxy0_L1BPH-2M9-MU6MU4_BPH-0DR15-MU6MU4 + +### 2016 +#HLT_2mu10_bJpsimumu +"HLT_2mu10_bJpsimumu_delayed", +"HLT_2mu10_bJpsimumu_noL2", +"HLT_2mu4_bJpsimumu_delayed_L1BPH-2M8-2MU4", +"HLT_2mu4_bJpsimumu_L1BPH-2M8-2MU4", +"HLT_2mu4_bJpsimumu_Lxy0_delayed_L1BPH-2M8-2MU4", +"HLT_2mu6_bJpsimumu", +"HLT_2mu6_bJpsimumu_delayed", +"HLT_2mu6_bJpsimumu_delayed_L1BPH-2M9-2MU6_BPH-2DR15-2MU6", +#HLT_2mu6_bJpsimumu_L1BPH-2M9-2MU6_BPH-2DR15-2MU6 +"HLT_2mu6_bJpsimumu_Lxy0", +"HLT_2mu6_bJpsimumu_Lxy0_delayed", +"HLT_2mu6_bJpsimumu_Lxy0_delayed_L1BPH-2M9-2MU6_BPH-2DR15-2MU6", +#HLT_2mu6_bJpsimumu_Lxy0_L1BPH-2M9-2MU6_BPH-2DR15-2MU6 +#HLT_3mu4_bJpsi +"HLT_3mu4_bJpsi_delayed", +#HLT_3mu6_bJpsi +"HLT_mu10_mu6_bJpsimumu", +"HLT_mu10_mu6_bJpsimumu_delayed", +"HLT_mu10_mu6_bJpsimumu_Lxy0", +"HLT_mu10_mu6_bJpsimumu_Lxy0_delayed", +"HLT_mu18_bJpsi_Trkloose", +"HLT_mu20_2mu0noL1_JpsimumuFS", +"HLT_mu20_2mu4_JpsimumuL2", +#HLT_mu6_2mu4_bJpsi +"HLT_mu6_2mu4_bJpsi_delayed", +"HLT_mu6_mu4_bJpsimumu", +"HLT_mu6_mu4_bJpsimumu_delayed", +"HLT_mu6_mu4_bJpsimumu_delayed_L1BPH-2M8-MU6MU4_BPH-0DR15-MU6MU4", +"HLT_mu6_mu4_bJpsimumu_delayed_L1MU6MU4-BO", +"HLT_mu6_mu4_bJpsimumu_L12MU4-B", +"HLT_mu6_mu4_bJpsimumu_L1BPH-2M8-MU6MU4_BPH-0DR15-MU6MU4", +"HLT_mu6_mu4_bJpsimumu_L1MU6MU4-BO", +"HLT_mu6_mu4_bJpsimumu_Lxy0", +"HLT_mu6_mu4_bJpsimumu_Lxy0_delayed", +"HLT_mu6_mu4_bJpsimumu_Lxy0_delayed_L1BPH-2M8-MU6MU4_BPH-0DR15-MU6MU4", +"HLT_mu6_mu4_bJpsimumu_Lxy0_L1BPH-2M8-MU6MU4_BPH-0DR15-MU6MU4", + +### 2015 +#HLT_2mu10_bJpsimumu +#HLT_2mu10_bJpsimumu_noL2 +"HLT_2mu10_l2msonly_bJpsimumu_noL2", +"HLT_2mu4_bJpsimumu", +"HLT_2mu4_bJpsimumu_noL2", +"HLT_2mu4_l2msonly_bJpsimumu_noL2", +#HLT_2mu6_bJpsimumu +"HLT_2mu6_bJpsimumu_noL2", +"HLT_2mu6_l2msonly_bJpsimumu_noL2", +#HLT_3mu4_bJpsi +#HLT_3mu6_bJpsi +"HLT_mu10_mu10_l2msonly_bJpsimumu_noL2", +"HLT_mu18_2mu0noL1_JpsimumuFS", +"HLT_mu18_2mu4_JpsimumuL2", +#HLT_mu18_bJpsi_Trkloose +#HLT_mu20_2mu0noL1_JpsimumuFS +#HLT_mu20_2mu4_JpsimumuL2 +"HLT_mu4_mu4_l2msonly_bJpsimumu_noL2", +"HLT_mu6_l2msonly_mu4_bJpsimumu_noL2", +"HLT_mu6_l2msonly_mu4_l2msonly_bJpsimumu_noL2", +#HLT_mu6_mu4_bJpsimumu +"HLT_mu6_mu4_bJpsimumu_noL2", +"HLT_mu6_mu4_l2msonly_bJpsimumu_noL2", +"HLT_mu6_mu6_l2msonly_bJpsimumu_noL2" , + + + +"HLT_mu4","HLT_mu6","HLT_mu10","HLT_mu14","HLT_mu18","HLT_mu24", #5% + + +"HLT_.*mu11_mu6.*", # Recent triggers +"HLT_.*mu.*imedium.*", # Trigger with looser isolation selection +"HLT_.*mu.*iloose.*", +"HLT_.*mu6.*2mu4.*", +"HLT_.*2mu.*", +"HLT_.*mu11.*2mu4noL1.*", +"HLT_.*2mu14_nomucomb.*", +"HLT_.*bTau.*", # Our tau triggers +"HLT_.*bDimu2700.*", +"HLT_.*bPhi.*" + ] + +from DerivationFrameworkTools.DerivationFrameworkToolsConf import DerivationFramework__TriggerSkimmingTool +BPHY20TriggerSkim = DerivationFramework__TriggerSkimmingTool(name = "BPHY20TriggerSkim", + TriggerListOR = triggerList, + TriggerListAND = [] ) + +ToolSvc += BPHY20TriggerSkim + +#-------------------------------------------------------------------- +# 2/ Select J/psi>mu+mu- +#-------------------------------------------------------------------- +## a/ setup JpsiFinder tool +## These are general tools independent of DerivationFramework that do the +## actual vertex fitting and some pre-selection. +from JpsiUpsilonTools.JpsiUpsilonToolsConf import Analysis__JpsiFinder +BPHY20JpsiFinder = Analysis__JpsiFinder( + name = "BPHY20JpsiFinder", + OutputLevel = INFO, + muAndMu = True, + muAndTrack = False, + TrackAndTrack = False, + assumeDiMuons = True, + muonThresholdPt = 1000, + invMassUpper = 4500.0, + invMassLower = 2000.0, + Chi2Cut = 20., + oppChargesOnly = True, +# allChargeCombinations = True, + combOnly = False, + atLeastOneComb = True, + useCombinedMeasurement = False, # Only takes effect if combOnly=True + muonCollectionKey = "Muons", + TrackParticleCollection = "InDetTrackParticles", + V0VertexFitterTool = BPHY20_VertexTools.TrkV0Fitter, # V0 vertex fitter + useV0Fitter = False, # if False a TrkVertexFitterTool will be used + TrkVertexFitterTool = BPHY20_VertexTools.TrkVKalVrtFitter, # VKalVrt vertex fitter + TrackSelectorTool = BPHY20_VertexTools.InDetTrackSelectorTool, + VertexPointEstimator = BPHY20_VertexTools.VtxPointEstimator, + useMCPCuts = False) + +ToolSvc += BPHY20JpsiFinder +print(BPHY20JpsiFinder) + +#-------------------------------------------------------------------- +## b/ setup the vertex reconstruction "call" tool(s). They are part of the derivation framework. +## These Augmentation tools add output vertex collection(s) into the StoreGate and add basic +## decorations which do not depend on the vertex mass hypothesis (e.g. lxy, ptError, etc). +## There should be one tool per topology, i.e. Jpsi and Psi(2S) do not need two instance of the +## Reco tool is the JpsiFinder mass window is wide enough. +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__Reco_Vertex +BPHY20JpsiSelectAndWrite = DerivationFramework__Reco_Vertex( + name = "BPHY20JpsiSelectAndWrite", + VertexSearchTool = BPHY20JpsiFinder, + OutputVtxContainerName = "BPHY20JpsiCandidates", + PVContainerName = "PrimaryVertices", + RefPVContainerName = "SHOULDNOTBEUSED", + DoVertexType = 1) + +ToolSvc += BPHY20JpsiSelectAndWrite +print(BPHY20JpsiSelectAndWrite) + +#-------------------------------------------------------------------- +## c/ augment and select Jpsi->mumu candidates +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__Select_onia2mumu +BPHY20_Select_Jpsi2mumu = DerivationFramework__Select_onia2mumu( + name = "BPHY20_Select_Jpsi2mumu", + HypothesisName = "Jpsi", + InputVtxContainerName = "BPHY20JpsiCandidates", + VtxMassHypo = 3096.900, + MassMin = 2000.0, + MassMax = 4500.0, + Chi2Max = 20, + DoVertexType = 1) + +ToolSvc += BPHY20_Select_Jpsi2mumu +print(BPHY20_Select_Jpsi2mumu) + +#-------------------------------------------------------------------- +# 3/ select B_c+->J/psi mu +#-------------------------------------------------------------------- +## a/ setup a new vertexing tool (necessary due to use of mass constraint) +from TrkVKalVrtFitter.TrkVKalVrtFitterConf import Trk__TrkVKalVrtFitter +BcJpsiMuVertexFit = Trk__TrkVKalVrtFitter( + name = "BcJpsiMuVertexFit", + Extrapolator = BPHY20_VertexTools.InDetExtrapolator, + FirstMeasuredPoint = True, + MakeExtendedVertex = True) + +ToolSvc += BcJpsiMuVertexFit +print(BcJpsiMuVertexFit) + +#-------------------------------------------------------------------- +## b/ setup the Jpsi+1 track finder +from JpsiUpsilonTools.JpsiUpsilonToolsConf import Analysis__JpsiPlus1Track +BPHY20BcJpsiMu = Analysis__JpsiPlus1Track( + name = "BPHY20BcJpsiMu", + OutputLevel = INFO, #DEBUG, + pionHypothesis = True, #False, + kaonHypothesis = False,#True, + trkThresholdPt = 1000, + trkMaxEta = 3.0, + BThresholdPt = 1000.0, + BMassUpper = 6900.0, + BMassLower = 2000.0, + JpsiContainerKey = "BPHY20JpsiCandidates", + TrackParticleCollection = "InDetTrackParticles", + MuonsUsedInJpsi = "Muons", + ExcludeCrossJpsiTracks = False, + TrkVertexFitterTool = BcJpsiMuVertexFit, + TrackSelectorTool = BPHY20_VertexTools.InDetTrackSelectorTool, + UseMassConstraint = True, + RequireNMuonTracks = 1, + Chi2Cut = 1000, #5 + TrkTrippletMassUpper = 6900, + TrkTrippletMassLower = 2000.0) + +ToolSvc += BPHY20BcJpsiMu +print(BPHY20BcJpsiMu) + +#-------------------------------------------------------------------- +## c/ setup the combined augmentation/skimming tool for the Bc+>J/psi mu +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__Reco_Vertex +BPHY20BcJpsiMuSelectAndWrite = DerivationFramework__Reco_Vertex( + name = "BPHY20BcJpsiMuSelectAndWrite", + OutputLevel = INFO, + VertexSearchTool = BPHY20BcJpsiMu, + OutputVtxContainerName = "BPHY20BcJpsiMuCandidates", + PVContainerName = "PrimaryVertices", + RefPVContainerName = "BPHY20RefittedPrimaryVertices", + RefitPV = True, + MaxPVrefit = 1000) + +ToolSvc += BPHY20BcJpsiMuSelectAndWrite +print(BPHY20BcJpsiMuSelectAndWrite) + +#-------------------------------------------------------------------- +## c/ augment and select B_c+>Jpsi mu candidates +BPHY20_Select_Bc2JpsiMu = DerivationFramework__Select_onia2mumu( + name = "BPHY20_Select_Bc2JpsiMu", + HypothesisName = "Bc", + InputVtxContainerName = "BPHY20BcJpsiMuCandidates", + TrkMasses = [105.658, 105.658, 105.658], + VtxMassHypo = 6274.9, + MassMin = 2000.0, + MassMax = 6900.0, + Chi2Max = 1000) + +ToolSvc += BPHY20_Select_Bc2JpsiMu +print(BPHY20_Select_Bc2JpsiMu) + + + +#==================================================================== +# Isolation +#==================================================================== + + +#Track isolation for candidates +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__VertexTrackIsolation +BPHY20TrackIsolationDecorator = DerivationFramework__VertexTrackIsolation( + name = "BPHY20TrackIsolationDecorator", + OutputLevel = INFO, + TrackIsoTool = "xAOD::TrackIsolationTool", + TrackContainer = "InDetTrackParticles", + InputVertexContainer = "BPHY20BcJpsiMuCandidates", + PassFlags = ["passed_Bc"] ) + +ToolSvc += BPHY20TrackIsolationDecorator + +#CaloIsolationTool explicitly declared to avoid pointless warnings (it works!!!) +from IsolationTool.IsolationToolConf import xAOD__CaloIsolationTool +BPHY20CaloIsolationTool = xAOD__CaloIsolationTool( + name = "BPHY20CaloIsolationTool", + OutputLevel = WARNING, + saveOnlyRequestedCorrections = True, + IsoLeakCorrectionTool = "" ) #Workaround for a bug in older versions + +ToolSvc += BPHY20CaloIsolationTool + +#Calo isolation for candidates +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__VertexCaloIsolation +BPHY20CaloIsolationDecorator = DerivationFramework__VertexCaloIsolation( + name = "BPHY20CaloIsolationDecorator", + OutputLevel = INFO, + CaloIsoTool = BPHY20CaloIsolationTool, #"xAOD::CaloIsolationTool", + TrackContainer = "InDetTrackParticles", + InputVertexContainer = "BPHY20BcJpsiMuCandidates", + CaloClusterContainer = "CaloCalTopoClusters", + ParticleCaloExtensionTool = "Trk::ParticleCaloExtensionTool/ParticleCaloExtensionTool", + PassFlags = ["passed_Bc"] ) + +ToolSvc += BPHY20CaloIsolationDecorator + + + +#==================================================================== +# Skimming tool to select only events with the correct vertices +#==================================================================== + +#-------------------------------------------------------------------- +## 9/ select the event. We only want to keep events that contain certain three-mu vertices which passed certain selection. +## Exactly like in the preselection, where only 2mu vertices are treated. + +# +#expression = "count(BPHY20JpsiCandidates.x > -999.0)+count(BPHY20BcJpsiMuCandidates.x > -999.0)+ count(BPHY20BcJpsiMuCandidates.passed_Bc) >0 " + +from DerivationFrameworkTools.DerivationFrameworkToolsConf import DerivationFramework__xAODStringSkimmingTool +BPHY20_SelectBcJpsiMuEvent = DerivationFramework__xAODStringSkimmingTool( + name = "BPHY20_SelectBcJpsiMuEvent", + expression = "count(BPHY20BcJpsiMuCandidates.passed_Bc) >= 1 ") + +ToolSvc += BPHY20_SelectBcJpsiMuEvent +print(BPHY20_SelectBcJpsiMuEvent) + + #==================================================================== + # Make event selection based on an OR of the input skimming tools + #==================================================================== + +# from DerivationFrameworkTools.DerivationFrameworkToolsConf import DerivationFramework__FilterCombinationOR +# BPHY20SkimmingOR = CfgMgr.DerivationFramework__FilterCombinationOR( +# "BPHY20SkimmingOR", +# FilterList = [BPHY20_SelectBcJpsiMuEvent] ) +# ToolSvc += BPHY20SkimmingOR +# print BPHY20SkimmingOR + + +#==================================================================== +# Add Extrapolation of muons to trigger layers +#==================================================================== + +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__MuonExtrapolationTool +BPHY20_Extrap_Tool = DerivationFramework__MuonExtrapolationTool( name = "BPHY20_ExtrapolationTool", OutputLevel = INFO ) + +ToolSvc += BPHY20_Extrap_Tool + + + + + +#==================================================================== +# Thinning Helper and various thinning tools +#==================================================================== + +#-------------------------------------------------------------------- +## 10/ Setup the thinning helper, only tool able to perform thinning of trigger navigation information + +from DerivationFrameworkCore.ThinningHelper import ThinningHelper +BPHY20ThinningHelper = ThinningHelper( "BPHY20ThinningHelper" ) +BPHY20ThinningHelper.TriggerChains = 'HLT_.*mu.*' #triggerList # . = any character; * = 0 or more times; + = 1 or more times; ? 0 or 1 times "Regular_Expression" +BPHY20ThinningHelper.AppendToStream( BPHY20Stream ) + + +#-------------------------------------------------------------------- +## 11/ track and vertex thinning. We want to remove all reconstructed secondary vertices +## which haven't passed any of the selections defined by (Select_*) tools. +## We also want to keep only tracks which are associates with either muons or any of the +## vertices that passed the selection. Multiple thinning tools can perform the +## selection. The final thinning decision is based OR of all the decisions (by default, +## although it can be changed by the JO). + +## 12/ Cleans up, removing duplicate vertices. An issue caused by the logic of Jpsi+1 track in the case of 3-muon candidates + +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__Thin_vtxDuplicates +BPHY20Thin_vtxDuplicates = DerivationFramework__Thin_vtxDuplicates(name = "BPHY20Thin_vtxDuplicates", + VertexContainerName = "BPHY20BcJpsiMuCandidates", + PassFlags = ["passed_Bc"]) + +ToolSvc += BPHY20Thin_vtxDuplicates + +## a) thining out vertices that didn't pass any selection and idetifying tracks associated with +## selected vertices. The "VertexContainerNames" is a list of the vertex containers, and "PassFlags" +## contains all pass flags for Select_* tools that must be satisfied. The vertex is kept is it +## satisfy any of the listed selections. + +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__Thin_vtxTrk +BPHY20Thin_vtxTrk = DerivationFramework__Thin_vtxTrk( + name = "BPHY20Thin_vtxTrk", + OutputLevel = INFO, + TrackParticleContainerName = "InDetTrackParticles", + AcceptanceRadius = 1., + VertexContainerNames = ["BPHY20BcJpsiMuCandidates"], + PassFlags = ["passed_Bc"], + ApplyAnd = True ) # "and" requirement for Vertices + +ToolSvc += BPHY20Thin_vtxTrk + + +## 13/ thinning out tracks that are not attached to muons. The final thinning decision is based on the OR operation +## between decision from this and the previous tools. +from DerivationFrameworkInDet.DerivationFrameworkInDetConf import DerivationFramework__MuonTrackParticleThinning +BPHY20MuonTPThinningTool = DerivationFramework__MuonTrackParticleThinning(name = "BPHY20MuonTPThinningTool", + MuonKey = "Muons", + InDetTrackParticlesKey = "InDetTrackParticles") +ToolSvc += BPHY20MuonTPThinningTool + +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__BPhysPVThinningTool +BPHY20_thinningTool_PV = DerivationFramework__BPhysPVThinningTool(name = "BPHY20_thinningTool_PV", + CandidateCollections = ["BPHY20BcJpsiMuCandidates"], + KeepPVTracks =True) + +ToolSvc += BPHY20_thinningTool_PV + +from DerivationFrameworkInDet.DerivationFrameworkInDetConf import DerivationFramework__TauTrackParticleThinning +BPHY20TauTPThinningTool = DerivationFramework__TauTrackParticleThinning(name = "BPHY20TauTPThinningTool", + TauKey = "TauJets", + InDetTrackParticlesKey = "InDetTrackParticles") +ToolSvc += BPHY20TauTPThinningTool + +# Only save truth informtion directly associated with: mu Ds+ D+ D*+ Ds*+ D0 D*0 B+ B*+ B0 B*0 +from DerivationFrameworkMCTruth.DerivationFrameworkMCTruthConf import DerivationFramework__GenericTruthThinning +BPHY20TruthThinTool = DerivationFramework__GenericTruthThinning(name = "BPHY20TruthThinTool", + ParticleSelectionString = "abs(TruthParticles.pdgId) == 13 || abs(TruthParticles.pdgId) == 15 || abs(TruthParticles.pdgId) == 541 || abs(TruthParticles.pdgId) == 431 || abs(TruthParticles.pdgId) == 411 || abs(TruthParticles.pdgId) == 413 || abs(TruthParticles.pdgId) == 433 || TruthParticles.pdgId == 421 || TruthParticles.pdgId == 423 || abs(TruthParticles.pdgId) == 521 || abs(TruthParticles.pdgId) == 523 || TruthParticles.pdgId == 511 || TruthParticles.pdgId == 513", + PreserveDescendants = True, + PreserveAncestors = True) +ToolSvc += BPHY20TruthThinTool + +# Only save truth neutrino and b/c quarks information +from DerivationFrameworkMCTruth.DerivationFrameworkMCTruthConf import DerivationFramework__GenericTruthThinning +BPHY20TruthThinNoChainTool = DerivationFramework__GenericTruthThinning(name = "BPHY20TruthThinNoChainTool", + ParticleSelectionString = "abs(TruthParticles.pdgId) == 4 || abs(TruthParticles.pdgId) == 5 || abs(TruthParticles.pdgId) == 12 || abs(TruthParticles.pdgId) == 14 || abs(TruthParticles.pdgId) == 16", + PreserveDescendants = False, + PreserveAncestors = False) +ToolSvc += BPHY20TruthThinNoChainTool + + +#==================================================================== +# CREATE THE DERIVATION KERNEL ALGORITHM AND PASS THE ABOVE TOOLS +#==================================================================== + +BPHY20ThinningTools = [ BPHY20MuonTPThinningTool, BPHY20Thin_vtxDuplicates, + BPHY20Thin_vtxTrk, BPHY20_thinningTool_PV, + BPHY20TauTPThinningTool] + +BPHY20SkimmingTools = [BPHY20_SelectBcJpsiMuEvent] + +BPHY20AugmentationTools = [BPHY20JpsiSelectAndWrite, BPHY20_Select_Jpsi2mumu, + BPHY20BcJpsiMuSelectAndWrite, BPHY20_Select_Bc2JpsiMu, + BPHY20_AugOriginalCounts, + BPHY20TrackIsolationDecorator, BPHY20CaloIsolationDecorator] + +if addMuExtrapolationForTrigger: + BPHY20AugmentationTools.append(BPHY20_Extrap_Tool) + +Kernel1Tools = [BPHY20TriggerSkim] + +if isSimulation: + #BPHY20AugmentationTools.append(DFCommonTauTruthMatchingWrapper) + if thinTruth: + BPHY20ThinningTools.append(BPHY20TruthThinTool) + BPHY20ThinningTools.append(BPHY20TruthThinNoChainTool) + +#The sequence object. Is in principle just a wrapper which allows to run two kernels in sequence +BPHY20_Sequence = CfgMgr.AthSequencer("BPHY20_Sequence") +from DerivationFrameworkFlavourTag.FlavourTagCommon import FlavorTagInit +FlavorTagInit(JetCollections=['AntiKt4EMPFlowJets'], Sequencer=BPHY20_Sequence) + +#onlyAugmentations implementation +if onlyAugmentations: + Kernel1Tools = [] + BPHY20SkimmingTools = [] + BPHY20ThinningTools = [] + +# Kernel n1 PRESELECTION +# The name of the kernel (BPHY20Kernel1 in this case) must be unique to this derivation +from DerivationFrameworkCore.DerivationFrameworkCoreConf import DerivationFramework__DerivationKernel +BPHY20_Sequence += CfgMgr.DerivationFramework__DerivationKernel("BPHY20Kernel_trigPresel", + AugmentationTools = [BPHY20TriggerCountToMetadata] , + SkimmingTools = Kernel1Tools) +# Kernel n2 deep Derivation +# The name of the kernel (BPHY20Kernel2 in this case) must be unique to this derivation +from DerivationFrameworkCore.DerivationFrameworkCoreConf import DerivationFramework__DerivationKernel +BPHY20_Sequence += CfgMgr.DerivationFramework__DerivationKernel("BPHY20Kernel", + AugmentationTools = BPHY20AugmentationTools, + SkimmingTools = BPHY20SkimmingTools, + ThinningTools = BPHY20ThinningTools) + +#Vital, replaces the adding of kernels directly +DerivationFrameworkJob += BPHY20_Sequence + +#==================================================================== +# Slimming +#==================================================================== + +from DerivationFrameworkCore.SlimmingHelper import SlimmingHelper +BPHY20SlimmingHelper = SlimmingHelper("BPHY20SlimmingHelper") +AllVariables = [] +StaticContent = [] + + + +SmartCollections = [ + "Photons", + "TauJets", + "AntiKt4EMTopoJets_BTagging201810", + "BTagging_AntiKt4EMTopo_201810", + "PrimaryVertices", + "Muons", + "InDetTrackParticles", + "MET_Reference_AntiKt4EMTopo" + ] + + +AllVariables = ["METAssoc_AntiKt4EMTopo", + "MET_Core_AntiKt4EMTopo", + "MET_Truth", + "MET_Track", + "MET_LocHadTopo"] + +AllVariables += ["Kt4EMTopoOriginEventShape", + "Kt4EMTopoEventShape"] + +AllVariables += ["CombinedMuonTrackParticles", + "ExtrapolatedMuonTrackParticles", + "MuonSpectrometerTrackParticles"] + + +ExtraVariables = ["Photons.pt.eta.phi.m", + "Electrons.pt.eta.phi.m","TauJets.pt.eta.phi.m.IsTruthMatched.truthJetLink.truthParticleLink", + "AntiKt4EMTopoJets_BTagging201810.JetPileupScaleMomentum_pt.JetPileupScaleMomentum_eta.JetPileupScaleMomentum_phi.JetPileupScaleMomentum_m", + "AntiKt4EMTopoJets_BTagging201810.JvtJvfcorr.HECFrac.LArQuality.HECQuality.NegativeE.AverageLArQF", + "AntiKt4EMTopoJets_BTagging201810.JetEtaJESScaleMomentum_pt.JetEtaJESScaleMomentum_eta.JetEtaJESScaleMomentum_phi.JetEtaJESScaleMomentum_m"] + +ExtraVariables += ["Muons.etaLayer1Hits.etaLayer2Hits.etaLayer3Hits.etaLayer4Hits.phiLayer1Hits.phiLayer2Hits.phiLayer3Hits.phiLayer4Hits", + "Muons.numberOfTriggerEtaLayers.numberOfPhiLayers", + "CombinedMuonTrackParticles.numberOfTRTHits.numberOfTRTHighThresholdHits", + "InDetTrackParticles.numberOfTRTHits.numberOfTRTHighThresholdHits.vx.vy.vz", + "PrimaryVertices.chiSquared.covariance"] + + +StaticContent = ["xAOD::VertexContainer#BPHY20RefittedPrimaryVertices", + "xAOD::VertexAuxContainer#BPHY20RefittedPrimaryVerticesAux."] + + +## Jpsi candidates +StaticContent += ["xAOD::VertexContainer#%s" % BPHY20JpsiSelectAndWrite.OutputVtxContainerName] +## we have to disable vxTrackAtVertex branch since it is not xAOD compatible +StaticContent += ["xAOD::VertexAuxContainer#%sAux.-vxTrackAtVertex" % BPHY20JpsiSelectAndWrite.OutputVtxContainerName] + +## Bc+>J/psi Mu+ candidates +StaticContent += ["xAOD::VertexContainer#%s" % BPHY20BcJpsiMuSelectAndWrite.OutputVtxContainerName] +StaticContent += ["xAOD::VertexAuxContainer#%sAux.-vxTrackAtVertex" % BPHY20BcJpsiMuSelectAndWrite.OutputVtxContainerName] + + +# Truth information for MC only +if isSimulation: + AllVariables += ["TruthEvents","TruthParticles","TruthVertices","MuonTruthParticles", "METMap_Truth"] + SmartCollections += ["AntiKt4TruthJets"] + +# Needed for trigger objects +BPHY20SlimmingHelper.IncludeMuonTriggerContent = True +BPHY20SlimmingHelper.IncludeBPhysTriggerContent = True + +# Pass all lists to the SlimmingHelper +BPHY20SlimmingHelper.ExtraVariables = ExtraVariables +BPHY20SlimmingHelper.AllVariables = AllVariables +BPHY20SlimmingHelper.StaticContent = StaticContent +BPHY20SlimmingHelper.SmartCollections = SmartCollections +BPHY20SlimmingHelper.AppendContentToStream(BPHY20Stream) + diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/BPHY21.py b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/BPHY21.py new file mode 100644 index 00000000000..def21d405a7 --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/BPHY21.py @@ -0,0 +1,329 @@ +#2019/11/18 +#==================================================================== +# BPHY21.py +# W -> (J/psi + D_s) +# It requires the reductionConf flag BPHY21 in Reco_tf.py +#==================================================================== + +# Set up common services and job object. +# This should appear in ALL derivation job options +from DerivationFrameworkCore.DerivationFrameworkMaster import * + +isSimulation = False +if globalflags.DataSource()=='geant4': + isSimulation = True + +print(isSimulation) + + +#==================================================================== +# AUGMENTATION TOOLS +#==================================================================== +## 1/ setup vertexing tools and services +include("DerivationFrameworkBPhys/configureVertexing.py") +BPHY21_VertexTools = BPHYVertexTools("BPHY21") + + +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__AugOriginalCounts +BPHY21_AugOriginalCounts = DerivationFramework__AugOriginalCounts( + name = "BPHY21_AugOriginalCounts", + VertexContainer = "PrimaryVertices", + TrackContainer = "InDetTrackParticles" ) +ToolSvc += BPHY21_AugOriginalCounts + +#==================================================================== +# TriggerCounting for Kernel1 +#==================================================================== +#List of trigggers to be counted +BPHY21_triggersToMetadata= [ + "HLT_2mu10", + "HLT_2mu10_nomucomb", + "HLT_2mu14", + "HLT_2mu14_nomucomb", + "HLT_mu18_mu8noL1", + "HLT_mu18_nomucomb_mu8noL1", + "HLT_mu20_mu8noL1", + "HLT_mu20_nomucomb_mu8noL1", + "HLT_mu22_mu8noL1", + "HLT_mu22_nomucomb_mu8noL1", + "HLT_mu20_mu8noL1", + "HLT_mu24_mu8noL1", + "HLT_mu10_mu6_bJpsimumu", + "HLT_mu22_mu8noL1_calotag_0eta010_L1MU1" + ] + + +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__TriggerCountToMetadata +BPHY21_TriggerCountToMetadata = DerivationFramework__TriggerCountToMetadata(name = "BPHY21_TriggerCount", + TriggerList = BPHY21_triggersToMetadata, + FolderName = "BPHY21") + +ToolSvc += BPHY21_TriggerCountToMetadata + +#==================================================================== + + +#==================================================================== +#==================================================================== +## 1/ Setup the skimming based on triggers +## + +BPHY21_triggerList = [ + "HLT_2mu10", + "HLT_2mu10_nomucomb", + "HLT_2mu14", + "HLT_2mu14_nomucomb", + "HLT_mu18_mu8noL1", + "HLT_mu18_nomucomb_mu8noL1", + "HLT_mu20_mu8noL1", + "HLT_mu20_nomucomb_mu8noL1", + "HLT_mu22_mu8noL1", + "HLT_mu22_nomucomb_mu8noL1", + "HLT_mu20_mu8noL1", + "HLT_mu24_mu8noL1", + "HLT_mu10_mu6_bJpsimumu", + "HLT_mu22_mu8noL1_calotag_0eta010_L1MU1" + ] + +from DerivationFrameworkTools.DerivationFrameworkToolsConf import DerivationFramework__TriggerSkimmingTool +BPHY21_TriggerSkim = DerivationFramework__TriggerSkimmingTool(name = "BPHY21_TriggerSkim", + TriggerListOR = BPHY21_triggerList) + +ToolSvc += BPHY21_TriggerSkim + + +#-------------------------------------------------------------------- +# 2/ Select J/psi>mu+mu- +#-------------------------------------------------------------------- +## a/ setup JpsiFinder tool +## These are general tools independent of DerivationFramework that do the +## actual vertex fitting and some pre-selection. +from JpsiUpsilonTools.JpsiUpsilonToolsConf import Analysis__JpsiFinder +BPHY21_JpsiFinder = Analysis__JpsiFinder( + name = "BPHY21_JpsiFinder", + OutputLevel = INFO, + muAndMu = True, + muAndTrack = False, + TrackAndTrack = False, + assumeDiMuons = True, + muonThresholdPt = 2700, + invMassUpper = 3400.0, + invMassLower = 2800.0, + Chi2Cut = 10., + oppChargesOnly = True, + combOnly = True, + atLeastOneComb = False, + useCombinedMeasurement = False, # Only takes effect if combOnly=True + muonCollectionKey = "Muons", + TrackParticleCollection = "InDetTrackParticles", + V0VertexFitterTool = BPHY21_VertexTools.TrkV0Fitter, # V0 vertex fitter + useV0Fitter = False, # if False a TrkVertexFitterTool will be used + TrkVertexFitterTool = BPHY21_VertexTools.TrkVKalVrtFitter, # VKalVrt vertex fitter + TrackSelectorTool = BPHY21_VertexTools.InDetTrackSelectorTool, + VertexPointEstimator = BPHY21_VertexTools.VtxPointEstimator, + useMCPCuts = False) + +ToolSvc += BPHY21_JpsiFinder +print(BPHY21_JpsiFinder) + +#-------------------------------------------------------------------- +## b/ setup the vertex reconstruction "call" tool(s). They are part of the derivation framework. +## These Augmentation tools add output vertex collection(s) into the StoreGate and add basic +## decorations which do not depend on the vertex mass hypothesis (e.g. lxy, ptError, etc). +## There should be one tool per topology, i.e. Jpsi and Psi(2S) do not need two instance of the +## Reco tool is the JpsiFinder mass window is wide enough. +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__Reco_Vertex +BPHY21_JpsiSelectAndWrite = DerivationFramework__Reco_Vertex( + name = "BPHY21_JpsiSelectAndWrite", + VertexSearchTool = BPHY21_JpsiFinder, + OutputVtxContainerName = "BPHY21_JpsiCandidates", + PVContainerName = "PrimaryVertices", + RefPVContainerName = "SHOULDNOTBEUSED", + DoVertexType = 1) + +ToolSvc += BPHY21_JpsiSelectAndWrite +print(BPHY21_JpsiSelectAndWrite) + +#-------------------------------------------------------------------- +## c/ augment and select Jpsi->mumu candidates +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__Select_onia2mumu +BPHY21_Select_Jpsi2mumu = DerivationFramework__Select_onia2mumu( + name = "BPHY21_Select_Jpsi2mumu", + HypothesisName = "Jpsi", + InputVtxContainerName = "BPHY21_JpsiCandidates", + VtxMassHypo = 3096.900, + MassMin = 2600.0, + MassMax = 3600.0, + Chi2Max = 200, + LxyMin = 0.1, + DoVertexType = 1) + +ToolSvc += BPHY21_Select_Jpsi2mumu +print(BPHY21_Select_Jpsi2mumu) + + +#-------------------------------------------------------------------- + +BPHY21_CascadeCollections = [] + + +#-------------------------------------------------------------------- + + +if not isSimulation: #Only Skim Data + from DerivationFrameworkTools.DerivationFrameworkToolsConf import DerivationFramework__xAODStringSkimmingTool + BPHY21_SelectJpsiEvent = DerivationFramework__xAODStringSkimmingTool( + name = "BPHY21_SelectJpsiEvent", + expression = "count(BPHY21_JpsiCandidates.passed_Jpsi) > 0") + + ToolSvc += BPHY21_SelectJpsiEvent + print(BPHY21_SelectJpsiEvent) + + #==================================================================== + # Make event selection based on an OR of the input skimming tools + #==================================================================== + + from DerivationFrameworkTools.DerivationFrameworkToolsConf import DerivationFramework__FilterCombinationOR + BPHY21_SkimmingOR = CfgMgr.DerivationFramework__FilterCombinationOR( + "BPHY21_SkimmingOR", + FilterList = [ BPHY21_TriggerSkim, BPHY21_SelectJpsiEvent] ) + ToolSvc += BPHY21_SkimmingOR + print(BPHY21_SkimmingOR) + +#-------------------------------------------------------------------- +##10/ track and vertex thinning. We want to remove all reconstructed secondary vertices +## which hasn't passed any of the selections defined by (Select_*) tools. +## We also want to keep only tracks which are associates with either muons or any of the +## vertices that passed the selection. Multiple thinning tools can perform the +## selection. The final thinning decision is based OR of all the decisions (by default, +## although it can be changed by the JO). + +## a) thining out vertices that didn't pass any selection and idetifying tracks associated with +## selected vertices. The "VertexContainerNames" is a list of the vertex containers, and "PassFlags" +## contains all pass flags for Select_* tools that must be satisfied. The vertex is kept is it +## satisfy any of the listed selections. +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__Thin_vtxTrk +BPHY21_thinningTool_Tracks = DerivationFramework__Thin_vtxTrk( + name = "BPHY21_thinningTool_Tracks", + TrackParticleContainerName = "InDetTrackParticles", + VertexContainerNames = ["BPHY21_JpsiCandidates"], + PassFlags = ["passed_Jpsi"]) + +ToolSvc += BPHY21_thinningTool_Tracks +print(BPHY21_thinningTool_Tracks) +''' +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__BPhysPVThinningTool +BPHY21_thinningTool_PV = DerivationFramework__BPhysPVThinningTool( + name = "BPHY21_thinningTool_PV", + CandidateCollections = ["BPHY21_JpsiCandidates"], + KeepPVTracks = True) + +ToolSvc += BPHY21_thinningTool_PV +print BPHY21_thinningTool_PV +''' +## b) thinning out tracks that are not attached to muons. The final thinning decision is based on the OR operation +## between decision from this and the previous tools. +from DerivationFrameworkInDet.DerivationFrameworkInDetConf import DerivationFramework__MuonTrackParticleThinning +BPHY21_MuonTPThinningTool = DerivationFramework__MuonTrackParticleThinning( + name = "BPHY21_MuonTPThinningTool", + MuonKey = "Muons", + InDetTrackParticlesKey = "InDetTrackParticles") + +ToolSvc += BPHY21_MuonTPThinningTool +print(BPHY21_MuonTPThinningTool) + +#==================================================================== +# CREATE THE DERIVATION KERNEL ALGORITHM AND PASS THE ABOVE TOOLS +#==================================================================== + +BPHY21_thiningCollection = [] + +print(BPHY21_thiningCollection) + +# The name of the kernel (BPHY21_Kernel in this case) must be unique to this derivation +from DerivationFrameworkCore.DerivationFrameworkCoreConf import DerivationFramework__DerivationKernel +DerivationFrameworkJob += CfgMgr.DerivationFramework__DerivationKernel( + "BPHY21_Kernel", + AugmentationTools = [BPHY21_JpsiSelectAndWrite, BPHY21_Select_Jpsi2mumu, + + BPHY21_AugOriginalCounts], + #Only skim if not MC + SkimmingTools = [BPHY21_SkimmingOR] if not isSimulation else [], + ThinningTools = BPHY21_thiningCollection + ) + +#==================================================================== +# SET UP STREAM +#==================================================================== +BPHY21_streamName = derivationFlags.WriteDAOD_BPHY21Stream.StreamName +BPHY21_fileName = buildFileName( derivationFlags.WriteDAOD_BPHY21Stream ) +BPHY21_Stream = MSMgr.NewPoolRootStream( BPHY21_streamName, BPHY21_fileName ) +BPHY21_Stream.AcceptAlgs(["BPHY21_Kernel"]) + +# Special lines for thinning +# Thinning service name must match the one passed to the thinning tools +from AthenaServices.Configurables import ThinningSvc, createThinningSvc +BPHY21_augStream = MSMgr.GetStream( BPHY21_streamName ) +BPHY21_evtStream = BPHY21_augStream.GetEventStream() + +BPHY21_ThinningSvc = createThinningSvc( svcName="BPHY21_ThinningSvc", outStreams=[BPHY21_evtStream] ) +svcMgr += BPHY21_ThinningSvc + +#==================================================================== +# Slimming +#==================================================================== + +# Added by ASC +from DerivationFrameworkCore.SlimmingHelper import SlimmingHelper +BPHY21_SlimmingHelper = SlimmingHelper("BPHY21_SlimmingHelper") +BPHY21_AllVariables = [] +BPHY21_StaticContent = [] + +# Needed for trigger objects +BPHY21_SlimmingHelper.IncludeMuonTriggerContent = TRUE +BPHY21_SlimmingHelper.IncludeBPhysTriggerContent = TRUE + +## primary vertices +BPHY21_AllVariables += ["PrimaryVertices"] +BPHY21_StaticContent += ["xAOD::VertexContainer#BPHY21_RefittedPrimaryVertices"] +BPHY21_StaticContent += ["xAOD::VertexAuxContainer#BPHY21_RefittedPrimaryVerticesAux."] + +## ID track particles +BPHY21_AllVariables += ["InDetTrackParticles"] + +## combined / extrapolated muon track particles +## (note: for tagged muons there is no extra TrackParticle collection since the ID tracks +## are store in InDetTrackParticles collection) +BPHY21_AllVariables += ["CombinedMuonTrackParticles"] +BPHY21_AllVariables += ["ExtrapolatedMuonTrackParticles"] + +## muon container +BPHY21_AllVariables += ["Muons"] + + +## Jpsi candidates +BPHY21_StaticContent += ["xAOD::VertexContainer#%s" % BPHY21_JpsiSelectAndWrite.OutputVtxContainerName] +## we have to disable vxTrackAtVertex branch since it is not xAOD compatible +BPHY21_StaticContent += ["xAOD::VertexAuxContainer#%sAux.-vxTrackAtVertex" % BPHY21_JpsiSelectAndWrite.OutputVtxContainerName] + + +# Tagging information (in addition to that already requested by usual algorithms) +#AllVariables += ["GSFTrackParticles", "MuonSpectrometerTrackParticles" ] + +# Added by ASC +# Truth information for MC only +if isSimulation: + BPHY21_AllVariables += ["TruthEvents","TruthParticles","TruthVertices","MuonTruthParticles"] + + +BPHY21_AllVariables = list(set(BPHY21_AllVariables)) # remove duplicates + +BPHY21_SlimmingHelper.AllVariables = BPHY21_AllVariables +BPHY21_SlimmingHelper.StaticContent = BPHY21_StaticContent +BPHY21_SlimmingHelper.SmartCollections = [] + +BPHY21_SlimmingHelper.AppendContentToStream(BPHY21_Stream) + +#==================================================================== +# END OF BPHY21.py +#==================================================================== \ No newline at end of file diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/BPHY22.py b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/BPHY22.py new file mode 100644 index 00000000000..791a4f5b028 --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/BPHY22.py @@ -0,0 +1,22 @@ +#==================================================================== +# BPHY22.py +# This an example job options script showing how to set up a +# derivation of the data using the derivation framework. +# It requires the reductionConf flag BPHY12 in Reco_tf.py +#==================================================================== + +# Set up common services and job object. +# This should appear in ALL derivation job options +from DerivationFrameworkCore.DerivationFrameworkMaster import * + +# This is a dummy file thus just printing something + +print("") +print("BPHY22 dummy file ... doing nothing.") +print("") +print("Please make sure that all local variables in this python") +print("script are prefixed by BPHY22_ in order to avoid collisions") +print("in case this derivation format is run in a train with others.") +print("Please ensure that it is python3 compatible e.g. by using") +print("print() instead of just print withouth parentheses.") +print("") diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/BPHY3.py b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/BPHY3.py new file mode 100644 index 00000000000..20f69848280 --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/BPHY3.py @@ -0,0 +1,284 @@ +#==================================================================== +# BPHY3.py +# This an example job options script showing how to set up a +# derivation of the data using the derivation framework. +# It requires the reductionConf flag BPHY3 in Reco_tf.py +#==================================================================== + +# Set up common services and job object. +# This should appear in ALL derivation job options +from DerivationFrameworkCore.DerivationFrameworkMaster import * + +isSimulation = False +if globalflags.DataSource()=='geant4': + isSimulation = True + +print(isSimulation) + +#==================================================================== +# AUGMENTATION TOOLS +#==================================================================== +## 1/ setup vertexing tools and services +#include( "JpsiUpsilonTools/configureServices.py" ) + +include("DerivationFrameworkBPhys/configureVertexing.py") +BPHY3_VertexTools = BPHYVertexTools("BPHY3") + +#-------------------------------------------------------------------- +## 2/ Setup the vertex fitter tools (e.g. JpsiFinder, JpsiPlus1Track, etc). +## These are general tools independent of DerivationFramework that do the +## actual vertex fitting and some pre-selection. +from JpsiUpsilonTools.JpsiUpsilonToolsConf import Analysis__JpsiFinder +BPHY3JpsiFinder = Analysis__JpsiFinder( + name = "BPHY3JpsiFinder", + OutputLevel = INFO, + muAndMu = False, + muAndTrack = False, + TrackAndTrack = True, + assumeDiMuons = False, # If true, will assume dimu hypothesis and use PDG value for mu mass + invMassUpper = 10000.0, + invMassLower = 0.0, + Chi2Cut = 100., + oppChargesOnly = True, + atLeastOneComb = False, + useCombinedMeasurement = False, # Only takes effect if combOnly=True + muonCollectionKey = "Muons", + TrackParticleCollection = "InDetTrackParticles", + V0VertexFitterTool = BPHY3_VertexTools.TrkV0Fitter, # V0 vertex fitter + useV0Fitter = False, # if False a TrkVertexFitterTool will be used + TrkVertexFitterTool = BPHY3_VertexTools.TrkVKalVrtFitter, # VKalVrt vertex fitter + TrackSelectorTool = BPHY3_VertexTools.InDetTrackSelectorTool, + VertexPointEstimator = BPHY3_VertexTools.VtxPointEstimator, + useMCPCuts = False, + track1Mass = 139.57, # Not very important, only used to calculate inv. mass cut, leave it loose here + track2Mass = 139.57) + +ToolSvc += BPHY3JpsiFinder +print(BPHY3JpsiFinder) + +#-------------------------------------------------------------------- +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__Reco_Vertex +BPHY3_Reco_diTrk = DerivationFramework__Reco_Vertex( + name = "BPHY3_Reco_diTrk", + VertexSearchTool = BPHY3JpsiFinder, + OutputVtxContainerName = "BPHY3VertexCandidates", + PVContainerName = "PrimaryVertices", + RefPVContainerName = "BPHY3RefittedPrimaryVertices") + +ToolSvc += BPHY3_Reco_diTrk +print(BPHY3_Reco_diTrk) +#-------------------------------------------------------------------- + + +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__Select_onia2mumu + +#-------------------------------------------------------------------- +## a/ augment and select X->pi+pi- candidates +BPHY3_Select_PiPi = DerivationFramework__Select_onia2mumu( + name = "BPHY3_Select_PiPi", + HypothesisName = "PiPi", + InputVtxContainerName = "BPHY3VertexCandidates", + TrkMasses = [139.57,139.57], + VtxMassHypo = 497.614, + MassMin = 300.0, + MassMax = 700.0, + Chi2Max = 20) + +ToolSvc += BPHY3_Select_PiPi +print(BPHY3_Select_PiPi) +#-------------------------------------------------------------------- + +#-------------------------------------------------------------------- +## a/ augment and select X->piK candidates +BPHY3_Select_PiK = DerivationFramework__Select_onia2mumu( + name = "BPHY3_Select_PiK", + HypothesisName = "PiK", + InputVtxContainerName = "BPHY3VertexCandidates", + TrkMasses = [139.57,493.677], + VtxMassHypo = 892., + MassMin = 0.0, + MassMax = 3500.0, + Chi2Max = 10) + +ToolSvc += BPHY3_Select_PiK +print(BPHY3_Select_PiK) +#-------------------------------------------------------------------- + +#-------------------------------------------------------------------- +## a/ augment and select X->KPi candidates +BPHY3_Select_KPi = DerivationFramework__Select_onia2mumu( + name = "BPHY3_Select_KPi", + HypothesisName = "KPi", + InputVtxContainerName = "BPHY3VertexCandidates", + TrkMasses = [493.677,139.57], + VtxMassHypo = 892., + MassMin = 0.0, + MassMax = 3500.0, + Chi2Max = 10) + +ToolSvc += BPHY3_Select_KPi +print(BPHY3_Select_KPi) +#-------------------------------------------------------------------- + + +#-------------------------------------------------------------------- +## a/ augment and select X->K+K- candidates +BPHY3_Select_KK = DerivationFramework__Select_onia2mumu( + name = "BPHY3_Select_KK", + HypothesisName = "KK", + InputVtxContainerName = "BPHY3VertexCandidates", + TrkMasses = [493.677,493.677], + VtxMassHypo = 1019.461, + MassMin = 0.0, + MassMax = 1100.0, + Chi2Max = 20) + +ToolSvc += BPHY3_Select_KK +print(BPHY3_Select_KK) +#-------------------------------------------------------------------- + +#-------------------------------------------------------------------- +## a/ augment and select X->ppbar candidates +BPHY3_Select_PP = DerivationFramework__Select_onia2mumu( + name = "BPHY3_Select_PP", + HypothesisName = "PP", + InputVtxContainerName = "BPHY3VertexCandidates", + TrkMasses = [938.272,938.272], + VtxMassHypo = 3096.916, + MassMin = 2800.0, + MassMax = 3600.0, + Chi2Max = 1) + +ToolSvc += BPHY3_Select_PP +print(BPHY3_Select_PP) +#-------------------------------------------------------------------- + + +#-------------------------------------------------------------------- +## 5/ select the event. We only want to keep events that contain certain vertices which passed certain selection. +## This is specified by the "SelectionExpression" property, which contains the expression in the following format: +## +## "ContainerName.passed_HypoName > count" +## +## where "ContainerName" is output container form some Reco_* tool, "HypoName" is the hypothesis name setup in some "Select_*" +## tool and "count" is the number of candidates passing the selection you want to keep. + +expression = "count(BPHY3VertexCandidates.passed_PiPi) > 0 || count(BPHY3VertexCandidates.passed_KPi) > 0 || count(BPHY3VertexCandidates.passed_PiK) > 0 || count(BPHY3VertexCandidates.passed_KK) > 0 || count(BPHY3VertexCandidates.passed_PP) > 0" + +from DerivationFrameworkTools.DerivationFrameworkToolsConf import DerivationFramework__xAODStringSkimmingTool +BPHY3_SelectEvent = DerivationFramework__xAODStringSkimmingTool(name = "BPHY3_SelectEvent", + expression = expression) +ToolSvc += BPHY3_SelectEvent +print(BPHY3_SelectEvent) + + +#==================================================================== +# SET UP STREAM +#==================================================================== +streamName = derivationFlags.WriteDAOD_BPHY3Stream.StreamName +fileName = buildFileName( derivationFlags.WriteDAOD_BPHY3Stream ) +BPHY3Stream = MSMgr.NewPoolRootStream( streamName, fileName ) +BPHY3Stream.AcceptAlgs(["BPHY3Kernel"]) +# Special lines for thinning +# Thinning service name must match the one passed to the thinning tools +augStream = MSMgr.GetStream( streamName ) +evtStream = augStream.GetEventStream() + + + + +#-------------------------------------------------------------------- +## 6/ track and vertex thinning. We want to remove all reconstructed secondary vertices +## which hasn't passed any of the selections defined by (Select_*) tools. +## We also want to keep only tracks which are associates with either muons or any of the +## vertices that passed the selection. Multiple thinning tools can perform the +## selection. The final thinning decision is based OR of all the decisions (by default, +## although it can be changed by the JO). + +## a) thining out vertices that didn't pass any selection and idetifying tracks associated with +## selected vertices. The "VertexContainerNames" is a list of the vertex containers, and "PassFlags" +## contains all pass flags for Select_* tools that must be satisfied. The vertex is kept is it +## satisfy any of the listed selections. + +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__Thin_vtxTrk +BPHY3Thin_vtxTrk = DerivationFramework__Thin_vtxTrk( + name = "BPHY3Thin_vtxTrk", + TrackParticleContainerName = "InDetTrackParticles", + VertexContainerNames = ["BPHY3VertexCandidates"], + StreamName = streamName, + PassFlags = ["passed_PiPi","passed_KPi","passed_PiK","passed_KK","passed_PP"]) + +ToolSvc += BPHY3Thin_vtxTrk + + + +# Added by ASC +# Only save truth informtion directly associated with Onia +#from DerivationFrameworkMCTruth.DerivationFrameworkMCTruthConf import DerivationFramework__GenericTruthThinning +#BPHY1TruthThinTool = DerivationFramework__GenericTruthThinning(name = "BPHY1TruthThinTool", +# ParticleSelectionString = "TruthParticles.pdgId == 443 || TruthParticles.pdgId == 100443", +# PreserveDescendants = True, +# PreserveAncestors = True) +#ToolSvc += BPHY1TruthThinTool +#print BPHY1TruthThinTool + + +#==================================================================== +# CREATE THE DERIVATION KERNEL ALGORITHM AND PASS THE ABOVE TOOLS +#==================================================================== +## 7/ IMPORTANT bit. Don't forget to pass the tools to the DerivationKernel! If you don't do that, they will not be +## be executed! + +# Added by ASC +BPHY3ThinningTools = [BPHY3Thin_vtxTrk] +#if globalflags.DataSource()=='geant4': +# BPHY3ThinningTools.append(BPHY3TruthThinTool) + +# The name of the kernel (BPHY1Kernel in this case) must be unique to this derivation +from DerivationFrameworkCore.DerivationFrameworkCoreConf import DerivationFramework__DerivationKernel +DerivationFrameworkJob += CfgMgr.DerivationFramework__DerivationKernel( + "BPHY3Kernel", + AugmentationTools = [BPHY3_Reco_diTrk,BPHY3_Select_PiPi,BPHY3_Select_KPi,BPHY3_Select_PiK,BPHY3_Select_KK,BPHY3_Select_PP], + SkimmingTools = [BPHY3_SelectEvent], + ThinningTools = BPHY3ThinningTools + + ) + + + +#==================================================================== +# Slimming +#==================================================================== + +# Added by ASC +from DerivationFrameworkCore.SlimmingHelper import SlimmingHelper +BPHY3SlimmingHelper = SlimmingHelper("BPHY3SlimmingHelper") +AllVariables = [] +StaticContent = [] + +# Needed for trigger objects +#BPHY3SlimmingHelper.IncludeMuonTriggerContent = True +#BPHY3SlimmingHelper.IncludeBPhysTriggerContent = True + +## primary vertices +AllVariables += ["PrimaryVertices"] +#StaticContent += ["xAOD::VertexContainer#BPHY3RefittedPrimaryVertices"] +#StaticContent += ["xAOD::VertexAuxContainer#BPHY3RefittedPrimaryVerticesAux."] + +## ID track particles +AllVariables += ["InDetTrackParticles"] + +## Vertex candidates +StaticContent += ["xAOD::VertexContainer#%s" % BPHY3_Reco_diTrk.OutputVtxContainerName] +StaticContent += ["xAOD::VertexAuxContainer#%sAux." % BPHY3_Reco_diTrk.OutputVtxContainerName] +## we have to disable vxTrackAtVertex branch since it is not xAOD compatible +StaticContent += ["xAOD::VertexAuxContainer#%sAux.-vxTrackAtVertex" % BPHY3_Reco_diTrk.OutputVtxContainerName] + +# Added by ASC +# Truth information for MC only +#if isSimulation: +# AllVariables += ["TruthEvents","TruthParticles","TruthVertices","MuonTruthParticles"] + +BPHY3SlimmingHelper.AllVariables = AllVariables +BPHY3SlimmingHelper.StaticContent = StaticContent +BPHY3SlimmingHelper.AppendContentToStream(BPHY3Stream) diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/BPHY4.py b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/BPHY4.py new file mode 100644 index 00000000000..9a973b28e4d --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/BPHY4.py @@ -0,0 +1,155 @@ +#==================================================================== +# BPHY4.py +#==================================================================== +#ServiceMgr.MessageSvc.debugLimit=100000000 +from DerivationFrameworkCore.DerivationFrameworkMaster import * + +isSimulation = False +if globalflags.DataSource()=='geant4': + isSimulation = True + +print(isSimulation) + +#==================================================================== +# AUGMENTATION TOOLS +#==================================================================== +## 1/ setup vertexing tools and services +include("DerivationFrameworkBPhys/configureVertexing.py") +BPHY4_VertexTools = BPHYVertexTools("BPHY4") + +#-------------------------------------------------------------------- +## 2/ Setup the vertex fitter tools +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__FourMuonTool +BPHY4FourMuonTool = DerivationFramework__FourMuonTool( + name = "BPHY4FourMuonTool", + OutputLevel = INFO, + ptCut = 2500.0, + etaCut = 2.5, + muonCollectionKey = "Muons", + TrackParticleCollection = "InDetTrackParticles", + V0VertexFitterTool = BPHY4_VertexTools.TrkV0Fitter, # V0 vertex fitter + useV0Fitter = False, # if False a TrkVertexFitterTool will be used + TrkVertexFitterTool = BPHY4_VertexTools.TrkVKalVrtFitter, # VKalVrt vertex fitter + TrackSelectorTool = BPHY4_VertexTools.InDetTrackSelectorTool, + VertexPointEstimator = BPHY4_VertexTools.VtxPointEstimator) + +ToolSvc += BPHY4FourMuonTool +print(BPHY4FourMuonTool) + +#-------------------------------------------------------------------- +## 3/ setup the vertex reconstruction "call" tool(s). + +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__Reco_4mu +BPHY4_Reco_4mu = DerivationFramework__Reco_4mu( + name = "BPHY4_Reco_4mu", + OutputLevel = INFO, + FourMuonTool = BPHY4FourMuonTool, + PairContainerName = "BPHY4Pairs", + QuadrupletContainerName = "BPHY4Quads", + PVContainerName = "PrimaryVertices", + RefPVContainerName = "BPHY4RefittedPrimaryVertices", + RefitPV = True, + MaxPVrefit = 100000, + DoVertexType = 7) + +ToolSvc += BPHY4_Reco_4mu +print(BPHY4_Reco_4mu) + +#==================================================================== +# SET UP STREAM +#==================================================================== +streamName = derivationFlags.WriteDAOD_BPHY4Stream.StreamName +fileName = buildFileName( derivationFlags.WriteDAOD_BPHY4Stream ) +BPHY4Stream = MSMgr.NewPoolRootStream( streamName, fileName ) +BPHY4Stream.AcceptAlgs(["BPHY4Kernel"]) +# Special lines for thinning +# Thinning service name must match the one passed to the thinning tools + +augStream = MSMgr.GetStream( streamName ) + + +#-------------------------------------------------------------------- +## thinning out tracks that are not attached to muons/electrons. The final thinning decision is based on the OR operation +## between decision from this and the previous tools. +from DerivationFrameworkInDet.DerivationFrameworkInDetConf import DerivationFramework__MuonTrackParticleThinning +BPHY4MuonTPThinningTool = DerivationFramework__MuonTrackParticleThinning(name = "BPHY4MuonTPThinningTool", + MuonKey = "Muons", + StreamName = streamName, + InDetTrackParticlesKey = "InDetTrackParticles") +ToolSvc += BPHY4MuonTPThinningTool +BPHY4ThinningTools = [BPHY4MuonTPThinningTool] + +from DerivationFrameworkInDet.DerivationFrameworkInDetConf import DerivationFramework__EgammaTrackParticleThinning +BPHY4ElectronTPThinningTool = DerivationFramework__EgammaTrackParticleThinning(name = "BPHY4ElectronTPThinningTool", + SGKey = "Electrons", + GSFTrackParticlesKey = "GSFTrackParticles", + StreamName = streamName, + InDetTrackParticlesKey = "InDetTrackParticles") +ToolSvc += BPHY4ElectronTPThinningTool +BPHY4ThinningTools += [BPHY4ElectronTPThinningTool] + +#==================================================================== +# CREATE THE DERIVATION KERNEL ALGORITHM AND PASS THE ABOVE TOOLS +#==================================================================== + +# The name of the kernel (BPHY4Kernel in this case) must be unique to this derivation +from DerivationFrameworkCore.DerivationFrameworkCoreConf import DerivationFramework__DerivationKernel +DerivationFrameworkJob += CfgMgr.DerivationFramework__DerivationKernel( + "BPHY4Kernel", + SkimmingTools = [BPHY4_Reco_4mu], + ThinningTools = BPHY4ThinningTools + ) + + + +#==================================================================== +# Slimming +#==================================================================== + +# Added by ASC +from DerivationFrameworkCore.SlimmingHelper import SlimmingHelper +BPHY4SlimmingHelper = SlimmingHelper("BPHY4SlimmingHelper") +BPHY4AllVariables = [] +BPHY4SmartVariables = [] +BPHY4StaticContent = [] + +# Needed for trigger objects +BPHY4SlimmingHelper.IncludeMuonTriggerContent = True +BPHY4SlimmingHelper.IncludeBPhysTriggerContent = True + +## primary vertices +BPHY4AllVariables += ["PrimaryVertices"] +BPHY4StaticContent += ["xAOD::VertexContainer#BPHY4RefittedPrimaryVertices"] +BPHY4StaticContent += ["xAOD::VertexAuxContainer#BPHY4RefittedPrimaryVerticesAux."] + +## ID track particles +BPHY4AllVariables += ["InDetTrackParticles"] +BPHY4SmartVariables += ["InDetTrackParticles"] + +## combined / extrapolated muon track particles +## (note: for tagged muons there is no extra TrackParticle collection since the ID tracks +## are store in InDetTrackParticles collection) +BPHY4AllVariables += ["CombinedMuonTrackParticles"] +BPHY4AllVariables += ["ExtrapolatedMuonTrackParticles"] + +## muon container +BPHY4AllVariables += ["Muons"] +BPHY4SmartVariables += ["Muons"] + +## Electron container +BPHY4SmartVariables += ["Electrons"] + +## Pair/quad candidates +BPHY4StaticContent += ["xAOD::VertexContainer#%s" % BPHY4_Reco_4mu.PairContainerName] +BPHY4StaticContent += ["xAOD::VertexAuxContainer#%sAux." % BPHY4_Reco_4mu.PairContainerName] +BPHY4StaticContent += ["xAOD::VertexContainer#%s" % BPHY4_Reco_4mu.QuadrupletContainerName] +BPHY4StaticContent += ["xAOD::VertexAuxContainer#%sAux." % BPHY4_Reco_4mu.QuadrupletContainerName] + +## we have to disable vxTrackAtVertex branch since it is not xAOD compatible +BPHY4StaticContent += ["xAOD::VertexAuxContainer#%sAux.-vxTrackAtVertex" % BPHY4_Reco_4mu.PairContainerName] +BPHY4StaticContent += ["xAOD::VertexAuxContainer#%sAux.-vxTrackAtVertex" % BPHY4_Reco_4mu.QuadrupletContainerName] + +BPHY4SlimmingHelper.AllVariables = BPHY4AllVariables +BPHY4SlimmingHelper.StaticContent = BPHY4StaticContent +BPHY4SlimmingHelper.SmartCollections = BPHY4SmartVariables +BPHY4SlimmingHelper.AppendContentToStream(BPHY4Stream) diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/BPHY5.py b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/BPHY5.py new file mode 100644 index 00000000000..0b041d9fc3e --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/BPHY5.py @@ -0,0 +1,558 @@ +#==================================================================== +# BPHY5.py +# Bs>J/psiKK +# It requires the reductionConf flag BPHY5 in Reco_tf.py +#==================================================================== + +# Set up common services and job object. +# This should appear in ALL derivation job options +from DerivationFrameworkCore.DerivationFrameworkMaster import * + +isSimulation = False +if globalflags.DataSource()=='geant4': + isSimulation = True + +print( isSimulation ) + + +#==================================================================== +# AUGMENTATION TOOLS +#==================================================================== +## 1/ setup vertexing tools and services +#include( "JpsiUpsilonTools/configureServices.py" ) + +include("DerivationFrameworkBPhys/configureVertexing.py") +BPHY5_VertexTools = BPHYVertexTools("BPHY5") + + +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__AugOriginalCounts +BPHY5_AugOriginalCounts = DerivationFramework__AugOriginalCounts( + name = "BPHY5_AugOriginalCounts", + VertexContainer = "PrimaryVertices", + TrackContainer = "InDetTrackParticles" ) +ToolSvc += BPHY5_AugOriginalCounts + + +#-------------------------------------------------------------------- +## 2/ setup JpsiFinder tool +## These are general tools independent of DerivationFramework that do the +## actual vertex fitting and some pre-selection. +from JpsiUpsilonTools.JpsiUpsilonToolsConf import Analysis__JpsiFinder +BPHY5JpsiFinder = Analysis__JpsiFinder(name = "BPHY5JpsiFinder", + OutputLevel = INFO, + muAndMu = True, + muAndTrack = False, + TrackAndTrack = False, + assumeDiMuons = True, + invMassUpper = 3600.0, + invMassLower = 2600.0, + Chi2Cut = 30., + oppChargesOnly = True, + combOnly = True, + atLeastOneComb = False, + useCombinedMeasurement = False, # Only takes effect if combOnly=True + muonCollectionKey = "Muons", + TrackParticleCollection = "InDetTrackParticles", + V0VertexFitterTool = BPHY5_VertexTools.TrkV0Fitter, # V0 vertex fitter + useV0Fitter = False, # if False a TrkVertexFitterTool will be used + TrkVertexFitterTool = BPHY5_VertexTools.TrkVKalVrtFitter, # VKalVrt vertex fitter + TrackSelectorTool = BPHY5_VertexTools.InDetTrackSelectorTool, + VertexPointEstimator = BPHY5_VertexTools.VtxPointEstimator, + useMCPCuts = False) +ToolSvc += BPHY5JpsiFinder +print (BPHY5JpsiFinder) + +#-------------------------------------------------------------------- +## 3/ setup the vertex reconstruction "call" tool(s). They are part of the derivation framework. +## These Augmentation tools add output vertex collection(s) into the StoreGate and add basic +## decorations which do not depend on the vertex mass hypothesis (e.g. lxy, ptError, etc). +## There should be one tool per topology, i.e. Jpsi and Psi(2S) do not need two instance of the +## Reco tool is the JpsiFinder mass window is wide enough. +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__Reco_Vertex +BPHY5JpsiSelectAndWrite = DerivationFramework__Reco_Vertex(name = "BPHY5JpsiSelectAndWrite", + VertexSearchTool = BPHY5JpsiFinder, + OutputVtxContainerName = "BPHY5JpsiCandidates", + PVContainerName = "PrimaryVertices", + RefPVContainerName = "SHOULDNOTBEUSED", + DoVertexType =1) +ToolSvc += BPHY5JpsiSelectAndWrite +print (BPHY5JpsiSelectAndWrite) + + +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__Select_onia2mumu + +## a/ augment and select Jpsi->mumu candidates +BPHY5_Select_Jpsi2mumu = DerivationFramework__Select_onia2mumu( + name = "BPHY5_Select_Jpsi2mumu", + HypothesisName = "Jpsi", + InputVtxContainerName = "BPHY5JpsiCandidates", + VtxMassHypo = 3096.916, + MassMin = 2000.0, + MassMax = 3600.0, + Chi2Max = 200, Do3d = False, + DoVertexType =1) + + +ToolSvc += BPHY5_Select_Jpsi2mumu +print (BPHY5_Select_Jpsi2mumu) + + + + +## 4/ setup a new vertexing tool (necessary due to use of mass constraint) +from TrkVKalVrtFitter.TrkVKalVrtFitterConf import Trk__TrkVKalVrtFitter +BsKKVertexFit = Trk__TrkVKalVrtFitter( + name = "BsKKVertexFit", + Extrapolator = BPHY5_VertexTools.InDetExtrapolator, + FirstMeasuredPoint = False, + MakeExtendedVertex = True) +ToolSvc += BsKKVertexFit +print (BsKKVertexFit) + +from TrkVKalVrtFitter.TrkVKalVrtFitterConf import Trk__TrkVKalVrtFitter +BplKplVertexFit = Trk__TrkVKalVrtFitter( + name = "BplKplVertexFit", + Extrapolator = BPHY5_VertexTools.InDetExtrapolator, + FirstMeasuredPoint = False, + MakeExtendedVertex = True) +ToolSvc += BplKplVertexFit +print (BplKplVertexFit) + +#Add the B to pi pi Jpsi X final states +from TrkVKalVrtFitter.TrkVKalVrtFitterConf import Trk__TrkVKalVrtFitter +BpipiXVertexFit = Trk__TrkVKalVrtFitter( + name = "BpipiXVertexFit", + Extrapolator = BPHY5_VertexTools.InDetExtrapolator, + FirstMeasuredPoint = False, + MakeExtendedVertex = True) +ToolSvc += BpipiXVertexFit +print (BpipiXVertexFit) + +## 5/ setup the Jpsi+2 track finder +from JpsiUpsilonTools.JpsiUpsilonToolsConf import Analysis__JpsiPlus2Tracks +BPHY5BsJpsiKK = Analysis__JpsiPlus2Tracks(name = "BPHY5BsJpsiKK", + OutputLevel = INFO, +kaonkaonHypothesis = True, +pionpionHypothesis = False, +kaonpionHypothesis = False, +trkThresholdPt = 800.0, +trkMaxEta = 3.0, +BMassUpper = 5800.0, +BMassLower = 5000.0, +#DiTrackMassUpper = 1019.445 + 100., +#DiTrackMassLower = 1019.445 - 100., +Chi2Cut = 15.0, +TrkQuadrupletMassUpper = 6000.0, +TrkQuadrupletMassLower = 4800.0, +JpsiContainerKey = "BPHY5JpsiCandidates", +TrackParticleCollection = "InDetTrackParticles", +MuonsUsedInJpsi = "Muons", +TrkVertexFitterTool = BsKKVertexFit, +TrackSelectorTool = BPHY5_VertexTools.InDetTrackSelectorTool, +UseMassConstraint = True) + +ToolSvc += BPHY5BsJpsiKK +print (BPHY5BsJpsiKK) + +from JpsiUpsilonTools.JpsiUpsilonToolsConf import Analysis__JpsiPlus2Tracks +BPHY5BdJpsiKst = Analysis__JpsiPlus2Tracks( + name = "BPHY5BdJpsiKst", + OutputLevel = INFO, + kaonkaonHypothesis = False, + pionpionHypothesis = False, + kaonpionHypothesis = True, + trkThresholdPt = 800.0, + trkMaxEta = 3.0, + BThresholdPt = 5000., + BMassLower = 4300.0, + BMassUpper = 6300.0, + JpsiContainerKey = "BPHY5JpsiCandidates", + TrackParticleCollection = "InDetTrackParticles", + #MuonsUsedInJpsi = "Muons", #Don't remove all muons, just those in J/psi candidate (see the following cut) + ExcludeCrossJpsiTracks = False, #setting this to False rejects the muons from J/psi candidate + TrkVertexFitterTool = BsKKVertexFit, + TrackSelectorTool = BPHY5_VertexTools.InDetTrackSelectorTool, + VertexPointEstimator = BPHY5_VertexTools.VtxPointEstimator, + UseMassConstraint = True, + Chi2Cut = 15.0, + TrkQuadrupletMassLower = 3500.0, + TrkQuadrupletMassUpper = 6800.0, + ) + +ToolSvc += BPHY5BdJpsiKst +print (BPHY5BdJpsiKst) + + +## 5a/ setup the Jpsi+1 track finder +from JpsiUpsilonTools.JpsiUpsilonToolsConf import Analysis__JpsiPlus1Track +BPHY5BplJpsiKpl = Analysis__JpsiPlus1Track(name = "BPHY5BplJpsiKpl", +OutputLevel = INFO,#DEBUG, +pionHypothesis = True, +kaonHypothesis = True, +trkThresholdPt = 750.0, +trkMaxEta = 3.0, +BThresholdPt = 4000.0, +BMassUpper = 7000.0, +BMassLower = 4500.0, +Chi2Cut = 15.0, +TrkTrippletMassUpper = 8000, +TrkTrippletMassLower = 4000, +JpsiContainerKey = "BPHY5JpsiCandidates", +TrackParticleCollection = "InDetTrackParticles", +MuonsUsedInJpsi = "Muons", +TrkVertexFitterTool = BplKplVertexFit, +TrackSelectorTool = BPHY5_VertexTools.InDetTrackSelectorTool, +UseMassConstraint = True, +ExcludeCrossJpsiTracks = False, +ExcludeJpsiMuonsOnly = True) + +ToolSvc += BPHY5BplJpsiKpl +print (BPHY5BplJpsiKpl) + +## 5b/ setup the Jpsi+pi+pi+X track finder +from JpsiUpsilonTools.JpsiUpsilonToolsConf import Analysis__JpsiPlus2Tracks +BPHY5BJpsipipiX = Analysis__JpsiPlus2Tracks(name = "BPHY5BJpsipipiX", + OutputLevel = INFO, +kaonkaonHypothesis = False, +pionpionHypothesis = True, +kaonpionHypothesis = False, +trkThresholdPt = 800.0, +trkMaxEta = 3.0, +BMassUpper = 5800.0, +BMassLower = 3400.0, +#DiTrackMassUpper = 1019.445 + 100., +#DiTrackMassLower = 1019.445 - 100., +Chi2Cut = 15.0, +TrkQuadrupletMassUpper = 5800.0, +TrkQuadrupletMassLower = 3400.0, +JpsiContainerKey = "BPHY5JpsiCandidates", +TrackParticleCollection = "InDetTrackParticles", +MuonsUsedInJpsi = "Muons", +TrkVertexFitterTool = BpipiXVertexFit, +TrackSelectorTool = BPHY5_VertexTools.InDetTrackSelectorTool, +UseMassConstraint = True, +ExcludeCrossJpsiTracks = False, +ExcludeJpsiMuonsOnly = True) + +ToolSvc += BPHY5BJpsipipiX +print (BPHY5BJpsipipiX) + +## 6/ setup the combined augmentation/skimming tool for the Bpm +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__Reco_Vertex +BPHY5BsKKSelectAndWrite = DerivationFramework__Reco_Vertex(name = "BPHY5BsKKSelectAndWrite", + VertexSearchTool = BPHY5BsJpsiKK, + OutputVtxContainerName = "BPHY5BsJpsiKKCandidates", + PVContainerName = "PrimaryVertices", + RefPVContainerName = "BPHY5RefittedPrimaryVertices", + RefitPV = True, Do3d = False, + MaxPVrefit = 10000, DoVertexType = 7) +ToolSvc += BPHY5BsKKSelectAndWrite +print (BPHY5BsKKSelectAndWrite) + +BPHY5BplKplSelectAndWrite = DerivationFramework__Reco_Vertex(name = "BPHY5BplKplSelectAndWrite", + VertexSearchTool = BPHY5BplJpsiKpl, + OutputVtxContainerName = "BPHY5BpmJpsiKpmCandidates", + PVContainerName = "PrimaryVertices", + RefPVContainerName = "BPHY5RefBplJpsiKplPrimaryVertices", + RefitPV = True, + MaxPVrefit = 10000 ) +ToolSvc += BPHY5BplKplSelectAndWrite +print (BPHY5BplKplSelectAndWrite) + +BPHY5BpipiXSelectAndWrite = DerivationFramework__Reco_Vertex(name = "BPHY5BpipiXSelectAndWrite", + VertexSearchTool = BPHY5BJpsipipiX, + OutputVtxContainerName = "BPHY5BJpsipipiXCandidates", + PVContainerName = "PrimaryVertices", + RefPVContainerName = "BPHY5RefittedBPipiPrimaryVertices", + RefitPV = True, Do3d = False, + MaxPVrefit = 10000, DoVertexType = 7) +ToolSvc += BPHY5BpipiXSelectAndWrite +print (BPHY5BpipiXSelectAndWrite) + +BPHY5BdKstSelectAndWrite = DerivationFramework__Reco_Vertex( + name = "BPHY5BdKstSelectAndWrite", + VertexSearchTool = BPHY5BdJpsiKst, + OutputVtxContainerName = "BPHY5BdJpsiKstCandidates", + PVContainerName = "PrimaryVertices", + RefPVContainerName = "BPHY5RefittedKstPrimaryVertices", + RefitPV = True, + MaxPVrefit = 10000, + DoVertexType = 7) +ToolSvc += BPHY5BdKstSelectAndWrite +print (BPHY5BdKstSelectAndWrite) +## b/ augment and select Bd->JpsiKst candidates +# set mass hypothesis (K pi) +BPHY5_Select_Bd2JpsiKst = DerivationFramework__Select_onia2mumu( + name = "BPHY5_Select_Bd2JpsiKst", + HypothesisName = "Bd", + InputVtxContainerName = "BPHY5BdJpsiKstCandidates", + TrkMasses = [105.658, 105.658, 493.677, 139.570], + VtxMassHypo = 5279.6, + MassMin = 100.0, #no mass cuts here + MassMax = 100000.0, #no mass cuts here + Chi2Max = 200) + +ToolSvc += BPHY5_Select_Bd2JpsiKst +print (BPHY5_Select_Bd2JpsiKst) + +## c/ augment and select Bdbar->JpsiKstbar candidates +# set mass hypothesis (pi K) +BPHY5_Select_Bd2JpsiKstbar = DerivationFramework__Select_onia2mumu( + name = "BPHY5_Select_Bd2JpsiKstbar", + HypothesisName = "Bdbar", + InputVtxContainerName = "BPHY5BdJpsiKstCandidates", + TrkMasses = [105.658, 105.658, 139.570, 493.677], + VtxMassHypo = 5279.6, + MassMin = 100.0, #no mass cuts here + MassMax = 100000.0, #no mass cuts here + Chi2Max = 200) + +ToolSvc += BPHY5_Select_Bd2JpsiKstbar +print (BPHY5_Select_Bd2JpsiKstbar) + + +## b/ augment and select Bs->JpsiKK candidates +BPHY5_Select_Bs2JpsiKK = DerivationFramework__Select_onia2mumu( + name = "BPHY5_Select_Bs2JpsiKK", + HypothesisName = "Bs", + InputVtxContainerName = "BPHY5BsJpsiKKCandidates", + TrkMasses = [105.658, 105.658, 493.677, 493.677], + VtxMassHypo = 5366.3, + MassMin = 5000.0, + MassMax = 5800.0, Do3d = False, + Chi2Max = 200) + +ToolSvc += BPHY5_Select_Bs2JpsiKK +print (BPHY5_Select_Bs2JpsiKK) + +BPHY5_Select_Bpl2JpsiKpl = DerivationFramework__Select_onia2mumu( + name = "BPHY5_Select_Bpl2JpsiKpl", + HypothesisName = "Bplus", + InputVtxContainerName = "BPHY5BpmJpsiKpmCandidates", + TrkMasses = [105.658, 105.658, 493.677], + VtxMassHypo = 5279.26, + MassMin = 5279.26 - 500, Do3d = False, + MassMax = 5279.26 + 500, + Chi2Max = 200 ) + +ToolSvc += BPHY5_Select_Bpl2JpsiKpl +print (BPHY5_Select_Bpl2JpsiKpl) + +BPHY5_Select_Bpl2JpsiPi = DerivationFramework__Select_onia2mumu( + name = "BPHY5_Select_Bpl2JpsiPi", + HypothesisName = "Bc", + InputVtxContainerName = "BPHY5BpmJpsiKpmCandidates", + TrkMasses = [105.658, 105.658, 139.570], + VtxMassHypo = 6275.1, Do3d = False, + MassMin = 6275.1 - 500, + MassMax = 6275.1 + 500, + Chi2Max = 200 ) + +ToolSvc += BPHY5_Select_Bpl2JpsiPi +print (BPHY5_Select_Bpl2JpsiPi) + +BPHY5_Select_B2JpsipipiX = DerivationFramework__Select_onia2mumu( + name = "BPHY5_Select_B2JpsipipiX", + HypothesisName = "pipiJpsi", + InputVtxContainerName = "BPHY5BJpsipipiXCandidates", + TrkMasses = [105.658, 105.658, 139.570, 139.570], + VtxMassHypo = 4260, + MassMin = 3400.0, + MassMax = 5800.0, Do3d = False, + Chi2Max = 200) + +ToolSvc += BPHY5_Select_B2JpsipipiX +print (BPHY5_Select_B2JpsipipiX) + +#expression = "count(BPHY5BpmJpsiKpmCandidates.passed_Bplus) > 0" +#from DerivationFrameworkTools.DerivationFrameworkToolsConf import DerivationFramework__xAODStringSkimmingTool +#BPHY5_SelectEvent = DerivationFramework__xAODStringSkimmingTool(name = "BPHY5_SelectEvent", +# expression = expression) +#ToolSvc += BPHY5_SelectEvent +#print BPHY5_SelectEvent + + +#from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__SelectEvent + +if not isSimulation: #Only Skim Data + from DerivationFrameworkTools.DerivationFrameworkToolsConf import DerivationFramework__xAODStringSkimmingTool + BPHY5_SelectBsJpsiKKEvent = DerivationFramework__xAODStringSkimmingTool( + name = "BPHY5_SelectBsJpsiKKEvent", + expression = "count(BPHY5BsJpsiKKCandidates.passed_Bs > 0) > 0") + + ToolSvc += BPHY5_SelectBsJpsiKKEvent + print (BPHY5_SelectBsJpsiKKEvent) + + BPHY5_SelectBplJpsiKplEvent = DerivationFramework__xAODStringSkimmingTool(name = "BPHY5_SelectBplJpsiKplEvent", + expression = "count(BPHY5BpmJpsiKpmCandidates.passed_Bplus>0) > 0") + ToolSvc += BPHY5_SelectBplJpsiKplEvent + print (BPHY5_SelectBplJpsiKplEvent) + + BPHY5_SelectBplJpsiKplEventBc = DerivationFramework__xAODStringSkimmingTool(name = "BPHY5_SelectBplJpsiKplEventBc", + expression = "count(BPHY5BpmJpsiKpmCandidates.passed_Bc>0) > 0") + ToolSvc += BPHY5_SelectBplJpsiKplEventBc + print (BPHY5_SelectBplJpsiKplEventBc) + + BPHY5_SelectBdKstarEventBd = DerivationFramework__xAODStringSkimmingTool(name = "BPHY5_SelectBdKstarEventBd", + expression = "count(BPHY5BdJpsiKstCandidates.passed_Bd>0) > 0") + ToolSvc += BPHY5_SelectBdKstarEventBd + print (BPHY5_SelectBdKstarEventBd) + + BPHY5_SelectBdKstarEventBdBar = DerivationFramework__xAODStringSkimmingTool(name = "BPHY5_SelectBdKstarEventBdbar", + expression = "count(BPHY5BdJpsiKstCandidates.passed_Bdbar>0) > 0") + ToolSvc += BPHY5_SelectBdKstarEventBdBar + print (BPHY5_SelectBdKstarEventBdBar) + #==================================================================== + # Make event selection based on an OR of the input skimming tools + #==================================================================== + from DerivationFrameworkTools.DerivationFrameworkToolsConf import DerivationFramework__FilterCombinationOR + BPHY5SkimmingOR = CfgMgr.DerivationFramework__FilterCombinationOR("BPHY5SkimmingOR", + FilterList = [BPHY5_SelectBsJpsiKKEvent, BPHY5_SelectBplJpsiKplEvent, BPHY5_SelectBplJpsiKplEventBc, + BPHY5_SelectBdKstarEventBd, BPHY5_SelectBdKstarEventBdBar]) + ToolSvc += BPHY5SkimmingOR + print (BPHY5SkimmingOR) + + +## b) thinning out tracks that are not attached to muons. The final thinning decision is based on the OR operation +## between decision from this and the previous tools. + + +#==================================================================== +# CREATE THE DERIVATION KERNEL ALGORITHM AND PASS THE ABOVE TOOLS +#==================================================================== + +thiningCollection = [] +print (thiningCollection) + +import DerivationFrameworkJetEtMiss.JetCommon +bphy5Seq = CfgMgr.AthSequencer("BPHY5Sequence") +DerivationFrameworkJob += bphy5Seq + +# The name of the kernel (BPHY5Kernel in this case) must be unique to this derivation +from DerivationFrameworkCore.DerivationFrameworkCoreConf import DerivationFramework__DerivationKernel +bphy5Seq += CfgMgr.DerivationFramework__DerivationKernel("BPHY5Kernel", + AugmentationTools = [BPHY5JpsiSelectAndWrite, BPHY5_Select_Jpsi2mumu, + BPHY5BsKKSelectAndWrite, BPHY5_Select_Bs2JpsiKK, + BPHY5BplKplSelectAndWrite, BPHY5BpipiXSelectAndWrite, BPHY5_Select_Bpl2JpsiKpl, BPHY5_Select_Bpl2JpsiPi, BPHY5_Select_B2JpsipipiX, + BPHY5BdKstSelectAndWrite, BPHY5_Select_Bd2JpsiKst, BPHY5_Select_Bd2JpsiKstbar, + BPHY5_AugOriginalCounts], + #Only skim if not MC + SkimmingTools = [BPHY5SkimmingOR] if not isSimulation else [], + ThinningTools = thiningCollection + + ) + +#==================================================================== +# SET UP STREAM +#==================================================================== +streamName = derivationFlags.WriteDAOD_BPHY5Stream.StreamName +fileName = buildFileName( derivationFlags.WriteDAOD_BPHY5Stream ) +BPHY5Stream = MSMgr.NewPoolRootStream( streamName, fileName ) +BPHY5Stream.AcceptAlgs(["BPHY5Kernel"]) + +# Special lines for thinning +# Thinning service name must match the one passed to the thinning tools +#from AthenaServices.Configurables import ThinningSvc, createThinningSvc +augStream = MSMgr.GetStream( streamName ) +evtStream = augStream.GetEventStream() + +#BPHY5ThinningSvc = createThinningSvc( svcName="BPHY5ThinningSvc", outStreams=[evtStream] ) +#svcMgr += BPHY5ThinningSvc + +#==================================================================== +# Slimming +#==================================================================== + +# Added by ASC +from DerivationFrameworkCore.SlimmingHelper import SlimmingHelper +BPHY5SlimmingHelper = SlimmingHelper("BPHY5SlimmingHelper") +AllVariables = [] +StaticContent = [] + +# Needed for trigger objects +BPHY5SlimmingHelper.IncludeMuonTriggerContent = TRUE +BPHY5SlimmingHelper.IncludeBPhysTriggerContent = TRUE + +## primary vertices +AllVariables += ["PrimaryVertices"] +StaticContent += ["xAOD::VertexContainer#BPHY5RefittedPrimaryVertices"] +StaticContent += ["xAOD::VertexAuxContainer#BPHY5RefittedPrimaryVerticesAux."] +StaticContent += ["xAOD::VertexContainer#BPHY5RefBplJpsiKplPrimaryVertices"] +StaticContent += ["xAOD::VertexAuxContainer#BPHY5RefBplJpsiKplPrimaryVerticesAux."] +StaticContent += ["xAOD::VertexContainer#BPHY5RefittedBPipiPrimaryVertices"] +StaticContent += ["xAOD::VertexAuxContainer#BPHY5RefittedBPipiPrimaryVerticesAux."] +StaticContent += ["xAOD::VertexContainer#BPHY5RefittedKstPrimaryVertices"] +StaticContent += ["xAOD::VertexAuxContainer#BPHY5RefittedKstPrimaryVerticesAux."] + +## ID track particles +AllVariables += ["InDetTrackParticles"] + +## combined / extrapolated muon track particles +## (note: for tagged muons there is no extra TrackParticle collection since the ID tracks +## are store in InDetTrackParticles collection) +AllVariables += ["CombinedMuonTrackParticles"] +AllVariables += ["ExtrapolatedMuonTrackParticles"] + +## muon container +AllVariables += ["Muons"] + + +## Jpsi candidates +StaticContent += ["xAOD::VertexContainer#%s" % BPHY5JpsiSelectAndWrite.OutputVtxContainerName] +## we have to disable vxTrackAtVertex branch since it is not xAOD compatible +StaticContent += ["xAOD::VertexAuxContainer#%sAux.-vxTrackAtVertex" % BPHY5JpsiSelectAndWrite.OutputVtxContainerName] + +StaticContent += ["xAOD::VertexContainer#%s" % BPHY5BsKKSelectAndWrite.OutputVtxContainerName] +StaticContent += ["xAOD::VertexAuxContainer#%sAux.-vxTrackAtVertex" % BPHY5BsKKSelectAndWrite.OutputVtxContainerName] + +StaticContent += ["xAOD::VertexContainer#%s" % BPHY5BplKplSelectAndWrite.OutputVtxContainerName] +StaticContent += ["xAOD::VertexAuxContainer#%sAux.-vxTrackAtVertex" % BPHY5BplKplSelectAndWrite.OutputVtxContainerName] + +StaticContent += ["xAOD::VertexContainer#%s" % BPHY5BpipiXSelectAndWrite.OutputVtxContainerName] +StaticContent += ["xAOD::VertexAuxContainer#%sAux.-vxTrackAtVertex" % BPHY5BpipiXSelectAndWrite.OutputVtxContainerName] + +StaticContent += ["xAOD::VertexContainer#%s" % BPHY5BdKstSelectAndWrite.OutputVtxContainerName] +StaticContent += ["xAOD::VertexAuxContainer#%sAux.-vxTrackAtVertex" % BPHY5BdKstSelectAndWrite.OutputVtxContainerName] + +# Tagging information (in addition to that already requested by usual algorithms) +#AllVariables += ["Electrons"] +AllVariables += ["GSFTrackParticles", "Electrons" , "Photons", "MuonSpectrometerTrackParticles" ] +tagJetCollections = ['AntiKt4LCTopoJets', 'AntiKt4EMTopoJets', 'AntiKt4PV0TrackJets'] + +AllVariables += [ "Kt4LCTopoOriginEventShape", "Kt4EMTopoOriginEventShape" ] +SmartVar = ["Photons" ] #[ tagJetCollections ] + + + + +for jet_collection in tagJetCollections: + AllVariables += [jet_collection] + AllVariables += ["BTagging_%s" % (jet_collection[:-4]) ] + AllVariables += ["BTagging_%sJFVtx" % (jet_collection[:-4]) ] + AllVariables += ["BTagging_%sSecVtx" % (jet_collection[:-4]) ] + +#addStandardJets("AntiKt", 0.4, "PV0Track", 2000, mods="track_ungroomed", algseq=bphy5Seq, outputGroup="BPHY5") + + +# Added by ASC +# Truth information for MC only +if isSimulation: + AllVariables += ["TruthEvents","TruthParticles","TruthVertices","MuonTruthParticles", "egammaTruthParticles" ] + AllVariables += ["AntiKt4TruthJets", "AntiKt4TruthWZJets" ] +# addStandardJets("AntiKt", 0.4, "Truth", 5000, mods="truth_ungroomed", algseq=bphy5Seq, outputGroup="BPHY5") +# addStandardJets("AntiKt", 0.4, "TruthWZ", 5000, mods="truth_ungroomed", algseq=bphy5Seq, outputGroup="BPHY5") + tagJetCollections += [ "AntiKt4TruthJets", "AntiKt4TruthWZJets" ] + +from DerivationFrameworkJetEtMiss.ExtendedJetCommon import replaceAODReducedJets +replaceAODReducedJets(tagJetCollections, bphy5Seq , "BPHY5" ) + + +AllVariables = list(set(AllVariables)) # remove duplicates + +BPHY5SlimmingHelper.AllVariables = AllVariables +BPHY5SlimmingHelper.StaticContent = StaticContent +BPHY5SlimmingHelper.SmartCollections = SmartVar + +BPHY5SlimmingHelper.AppendContentToStream(BPHY5Stream) + + diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/BPHY6.py b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/BPHY6.py new file mode 100644 index 00000000000..23b5bb5095d --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/BPHY6.py @@ -0,0 +1,344 @@ +#==================================================================== +# BPHY6.py +# This an example job options script showing how to set up a +# derivation of the data using the derivation framework. +# It requires the reductionConf flag BPHY6 in Reco_tf.py +#==================================================================== + +# Set up common services and job object. +# This should appear in ALL derivation job options +from DerivationFrameworkCore.DerivationFrameworkMaster import * + +isSimulation = False +if globalflags.DataSource()=='geant4': + isSimulation = True + +print(isSimulation) +#==================================================================== +# AUGMENTATION TOOLS +#==================================================================== +## 1/ setup vertexing tools and services +#include( "JpsiUpsilonTools/configureServices.py" ) + +include("DerivationFrameworkBPhys/configureVertexing.py") +BPHY6_VertexTools = BPHYVertexTools("BPHY6") + + +# General Variables +dimuon_chi2_max = 50. +dimuon_mass_min = 100. +dimuon_mass_max = 150e3 + +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__MuonExtrapolationTool +BPHY6_Extrap_Tool = DerivationFramework__MuonExtrapolationTool( + name = "BPHY6_ExtrapolationTool", + OutputLevel = INFO ) +ToolSvc += BPHY6_Extrap_Tool + + + +#-------------------------------------------------------------------- +## 2/ Setup the vertex fitter tools (e.g. JpsiFinder, JpsiPlus1Track, etc). +## These are general tools independent of DerivationFramework that do the +## actual vertex fitting and some pre-selection. +from JpsiUpsilonTools.JpsiUpsilonToolsConf import Analysis__JpsiFinder +BPHY6JpsiFinder = Analysis__JpsiFinder( + name = "BPHY6JpsiFinder", + OutputLevel = INFO, + muAndMu = True, + muAndTrack = False, + TrackAndTrack = False, + assumeDiMuons = True, # If true, will assume dimu hypothesis and use PDG value for mu mass + invMassUpper = dimuon_mass_max, + invMassLower = dimuon_mass_min, + Chi2Cut = dimuon_chi2_max, + oppChargesOnly = True, + atLeastOneComb = True, + useCombinedMeasurement = False, # Only takes effect if combOnly=True + muonCollectionKey = "Muons", + TrackParticleCollection = "InDetTrackParticles", + V0VertexFitterTool = BPHY6_VertexTools.TrkV0Fitter, # V0 vertex fitter + useV0Fitter = False, # if False a TrkVertexFitterTool will be used + TrkVertexFitterTool = BPHY6_VertexTools.TrkVKalVrtFitter, # VKalVrt vertex fitter + TrackSelectorTool = BPHY6_VertexTools.InDetTrackSelectorTool, + VertexPointEstimator = BPHY6_VertexTools.VtxPointEstimator, + useMCPCuts = False ) + +ToolSvc += BPHY6JpsiFinder +print(BPHY6JpsiFinder) + + +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__Reco_Vertex + + + +BPHY6_Reco_mumu = DerivationFramework__Reco_Vertex( + name = "BPHY6_Reco_mumu", + VertexSearchTool = BPHY6JpsiFinder, + OutputVtxContainerName = "BPHY6OniaCandidates", + PVContainerName = "PrimaryVertices", + RefPVContainerName = "BPHY6RefittedPrimaryVertices") + +ToolSvc += BPHY6_Reco_mumu +print(BPHY6_Reco_mumu) + + +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__Select_onia2mumu + +## a/ augment and select Jpsi->mumu candidates +BPHY6_Select_Jpsi2mumu = DerivationFramework__Select_onia2mumu( + name = "BPHY6_Select_Jpsi2mumu", + HypothesisName = "Jpsi", + InputVtxContainerName = "BPHY6OniaCandidates", + VtxMassHypo = 3096.916, + MassMin = 2700.0, + MassMax = 3500.0, + Chi2Max = 20) + +ToolSvc += BPHY6_Select_Jpsi2mumu +print(BPHY6_Select_Jpsi2mumu) + + +## b/ augment and select Psi(2S)->mumu candidates +BPHY6_Select_Psi2mumu = DerivationFramework__Select_onia2mumu( + name = "BPHY6_Select_Psi2mumu", + HypothesisName = "Psi", + InputVtxContainerName = "BPHY6OniaCandidates", + VtxMassHypo = 3686.09, + MassMin = 3200.0, + MassMax = 4200.0, + Chi2Max = 20) + +ToolSvc += BPHY6_Select_Psi2mumu +print(BPHY6_Select_Psi2mumu) + +# Added by ASC +## c/ augment and select Upsilon(nS)->mumu candidates +BPHY6_Select_Upsi2mumu = DerivationFramework__Select_onia2mumu( + name = "BPHY6_Select_Upsi2mumu", + HypothesisName = "Upsi", + InputVtxContainerName = "BPHY6OniaCandidates", + VtxMassHypo = 9460.30, + MassMin = 8000.0, + MassMax = 12000.0, + Chi2Max = 20) + +ToolSvc += BPHY6_Select_Upsi2mumu +print(BPHY6_Select_Upsi2mumu) + +BPHY6_Select_Bmumu2mumu = DerivationFramework__Select_onia2mumu( + name = "BPHY6_Select_Bmumu2mumu", + HypothesisName = "Bmumu", + InputVtxContainerName = "BPHY6OniaCandidates", + VtxMassHypo = 5366.77, + MassMin = 4200.0, + MassMax = 8000.0, + Chi2Max = 20) + +ToolSvc += BPHY6_Select_Bmumu2mumu +print(BPHY6_Select_Bmumu2mumu) + +BPHY6_Select_Zmumu2mumu = DerivationFramework__Select_onia2mumu( + name = "BPHY6_Select_Zmumu2mumu", + HypothesisName = "Zmumu", + InputVtxContainerName = "BPHY6OniaCandidates", + VtxMassHypo = 91187.6, + MassMin = 60000.0, + MassMax = 120000.0, + Chi2Max = 20) + +ToolSvc += BPHY6_Select_Zmumu2mumu +print(BPHY6_Select_Zmumu2mumu) + +BPHY6_Select_Onia2mumu = DerivationFramework__Select_onia2mumu( + name = "BPHY6_Select_Onia2mumu", + HypothesisName = "Onia", + InputVtxContainerName = "BPHY6OniaCandidates", + VtxMassHypo = 3096.916, + MassMin = dimuon_mass_min, + MassMax = dimuon_mass_max, + Chi2Max = 20) + +ToolSvc += BPHY6_Select_Onia2mumu +print(BPHY6_Select_Onia2mumu) + + + +trigger_list = [r'HLT_\d?mu\d+'] + +from DerivationFrameworkTools.DerivationFrameworkToolsConf import DerivationFramework__TriggerSkimmingTool +BPHY6TrigSkimmingTool = DerivationFramework__TriggerSkimmingTool( name = "BPHY6TrigSkimmingTool", + TriggerListOR = trigger_list ) +ToolSvc += BPHY6TrigSkimmingTool + + + +expression = "count(BPHY6OniaCandidates.passed_Onia) > 0 " +from DerivationFrameworkTools.DerivationFrameworkToolsConf import DerivationFramework__xAODStringSkimmingTool +BPHY6_SelectEvent = DerivationFramework__xAODStringSkimmingTool(name = "BPHY6_SelectEvent", + expression = expression) +ToolSvc += BPHY6_SelectEvent +print(BPHY6_SelectEvent) + +#-------------------------------------------------------------------- +## 6/ track and vertex thinning. We want to remove all reconstructed secondary vertices +## which hasn't passed any of the selections defined by (Select_*) tools. +## We also want to keep only tracks which are associates with either muons or any of the +## vertices that passed the selection. Multiple thinning tools can perform the +## selection. The final thinning decision is based OR of all the decisions (by default, +## although it can be changed by the JO). + +## a) thining out vertices that didn't pass any selection and idetifying tracks associated with +## selected vertices. The "VertexContainerNames" is a list of the vertex containers, and "PassFlags" +## contains all pass flags for Select_* tools that must be satisfied. The vertex is kept is it +## satisfy any of the listed selections. + + +#==================================================================== +# SET UP STREAM +#==================================================================== +streamName = derivationFlags.WriteDAOD_BPHY6Stream.StreamName +fileName = buildFileName( derivationFlags.WriteDAOD_BPHY6Stream ) +BPHY6Stream = MSMgr.NewPoolRootStream( streamName, fileName ) +BPHY6Stream.AcceptAlgs(["BPHY6Kernel"]) +# Special lines for thinning +# Thinning service name must match the one passed to the thinning tools +augStream = MSMgr.GetStream( streamName ) +evtStream = augStream.GetEventStream() + + + +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__Thin_vtxTrk +BPHY6Thin_vtxTrk = DerivationFramework__Thin_vtxTrk( + name = "BPHY6Thin_vtxTrk", + TrackParticleContainerName = "InDetTrackParticles", + StreamName = streamName, + VertexContainerNames = ["BPHY6OniaCandidates"], + PassFlags = ["passed_Onia"], ) + +ToolSvc += BPHY6Thin_vtxTrk + + +## b) thinning out tracks that are not attached to muons. The final thinning decision is based on the OR operation +## between decision from this and the previous tools. +from DerivationFrameworkInDet.DerivationFrameworkInDetConf import DerivationFramework__MuonTrackParticleThinning +BPHY6MuonTPThinningTool = DerivationFramework__MuonTrackParticleThinning(name = "BPHY6MuonTPThinningTool", + MuonKey = "Muons", + StreamName = streamName, + InDetTrackParticlesKey = "InDetTrackParticles") +ToolSvc += BPHY6MuonTPThinningTool + +# Added by ASC +# Only save truth informtion directly associated with Onia +from DerivationFrameworkMCTruth.DerivationFrameworkMCTruthConf import DerivationFramework__GenericTruthThinning +BPHY6TruthThinTool = DerivationFramework__GenericTruthThinning(name = "BPHY6TruthThinTool", + ParticleSelectionString = "TruthParticles.pdgId == 443 || TruthParticles.pdgId == 100443 || TruthParticles.pdgId == 553 || TruthParticles.pdgId == 100553 || TruthParticles.pdgId == 200553 || TruthParticles.pdgId == 23 || TruthParticles.pdgId == 531 || TruthParticles.pdgId == 511 || TruthParticles.pdgId == 521 || TruthParticles.pdgId == 541", + PreserveDescendants = True, + PreserveAncestors = True) +ToolSvc += BPHY6TruthThinTool +print(BPHY6TruthThinTool) + + +#==================================================================== +# CREATE THE DERIVATION KERNEL ALGORITHM AND PASS THE ABOVE TOOLS +#==================================================================== +## 7/ IMPORTANT bit. Don't forget to pass the tools to the DerivationKernel! If you don't do that, they will not be +## be executed! + +# Added by ASC +BPHY6ThinningTools = [BPHY6Thin_vtxTrk, BPHY6MuonTPThinningTool] +if globalflags.DataSource()=='geant4': + BPHY6ThinningTools.append(BPHY6TruthThinTool) + + +# Build a tool to apply the OR combination of the String expression skimming tool, and the Trigger Skimming Tool +from DerivationFrameworkTools.DerivationFrameworkToolsConf import DerivationFramework__FilterCombinationOR +SkimmingORTool = CfgMgr.DerivationFramework__FilterCombinationOR("BPHY6SkimmingOR", + FilterList = [BPHY6_SelectEvent,BPHY6TrigSkimmingTool],) +ToolSvc += SkimmingORTool +print(SkimmingORTool) + +# The name of the kernel (BPHY6Kernel in this case) must be unique to this derivation +from DerivationFrameworkCore.DerivationFrameworkCoreConf import DerivationFramework__DerivationKernel +DerivationFrameworkJob += CfgMgr.DerivationFramework__DerivationKernel( + "BPHY6Kernel", + AugmentationTools = [BPHY6_Reco_mumu, + BPHY6_Select_Jpsi2mumu, BPHY6_Select_Psi2mumu, BPHY6_Select_Upsi2mumu,BPHY6_Select_Bmumu2mumu, + BPHY6_Select_Zmumu2mumu,BPHY6_Select_Onia2mumu, BPHY6_Extrap_Tool], + SkimmingTools = [SkimmingORTool], + # ThinningTools = [BPHY6Thin_vtxTrk, BPHY6MuonTPThinningTool] + ThinningTools = BPHY6ThinningTools + + ) + + + + +#==================================================================== +# Slimming +#==================================================================== + +# Added by ASC +from DerivationFrameworkCore.SlimmingHelper import SlimmingHelper +BPHY6SlimmingHelper = SlimmingHelper("BPHY6SlimmingHelper") +AllVariables = [] +StaticContent = [] + +# Needed for trigger objects +BPHY6SlimmingHelper.IncludeMuonTriggerContent = True +BPHY6SlimmingHelper.IncludeBPhysTriggerContent = True + +AllVariables += ["LVL1MuonRoIs"] + +## primary vertices +AllVariables += ["PrimaryVertices"] +StaticContent += ["xAOD::VertexContainer#BPHY6RefittedPrimaryVertices"] +StaticContent += ["xAOD::VertexAuxContainer#BPHY6RefittedPrimaryVerticesAux."] + +## ID track particles +AllVariables += ["InDetTrackParticles"] + +AllVariables += ["HLT_xAOD__TrackParticleContainer_InDetTrigTrackingxAODCnv_Muon_EFID"] +AllVariables += ["HLT_xAOD__TrackParticleContainer_InDetTrigTrackingxAODCnv_Muon_IDTrig"] +AllVariables += ["HLT_xAOD__TrackParticleContainer_InDetTrigTrackingxAODCnv_Muon_FTF"] +AllVariables += ["HLT_xAOD__TrackParticleContainer_InDetTrigTrackingxAODCnv_Bphysics_FTF"] +AllVariables += ["HLT_xAOD__TrackParticleContainer_InDetTrigTrackingxAODCnv_Bphysics_IDTrig"] + + + +## combined / extrapolated muon track particles +## (note: for tagged muons there is no extra TrackParticle collection since the ID tracks +## are store in InDetTrackParticles collection) +AllVariables += ["CombinedMuonTrackParticles"] +AllVariables += ["ExtrapolatedMuonTrackParticles"] +AllVariables += ["MuonSpectrometerTrackParticles"] + +## muon container +AllVariables += ["Muons"] +AllVariables += ["HLT_xAOD__L2StandAloneMuonContainer_MuonL2SAInfo"] +AllVariables += ["HLT_xAOD__L2CombinedMuonContainer_MuonL2CBInfo"] +AllVariables += ["HLT_xAOD__MuonContainer_MuonEFInfo"] + + +AllVariables += ["HLT_xAOD__TrigBphysContainer_L2BMuMuXFex" ] +AllVariables += ["HLT_xAOD__TrigBphysContainer_EFBMuMuXFex" ] +AllVariables += ["HLT_xAOD__TrigBphysContainer_L2BMuMuFex" ] +AllVariables += ["HLT_xAOD__TrigBphysContainer_EFBMuMuFex" ] +AllVariables += ["HLT_xAOD__TrigBphysContainer_L2TrackMass" ] +AllVariables += ["HLT_xAOD__TrigBphysContainer_EFTrackMass" ] +AllVariables += ["HLT_xAOD__TrigBphysContainer_L2MultiMuFex"] +AllVariables += ["HLT_xAOD__TrigBphysContainer_EFMultiMuFex"] + + +## Jpsi candidates +StaticContent += ["xAOD::VertexContainer#%s" % BPHY6_Reco_mumu.OutputVtxContainerName] +StaticContent += ["xAOD::VertexAuxContainer#%sAux.-vxTrackAtVertex" % BPHY6_Reco_mumu.OutputVtxContainerName] + +if isSimulation: + AllVariables += ["TruthEvents","TruthParticles","TruthVertices","MuonTruthParticles"] + +BPHY6SlimmingHelper.AllVariables = AllVariables +BPHY6SlimmingHelper.StaticContent = StaticContent +BPHY6SlimmingHelper.AppendContentToStream(BPHY6Stream) + + diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/BPHY7.py b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/BPHY7.py new file mode 100644 index 00000000000..b53b1a403bc --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/BPHY7.py @@ -0,0 +1,648 @@ +#==================================================================== +# BPHY7.py +# +# https://twiki.cern.ch/twiki/bin/view/AtlasProtected/LfvBphy7 +#==================================================================== + + +#==================================================================== +# FLAGS TO PERSONALIZE THE DERIVATION +#==================================================================== + +onlyAugmentations = False # Set to True to deactivate thinning and skimming, and only keep augmentations (to generate a sample with full xAOD plus all the extra) +thinTruth = True +addMuExtrapolationForTrigger = True + + +from DerivationFrameworkCore.DerivationFrameworkMaster import * +isSimulation = False +if globalflags.DataSource()=='geant4': + isSimulation = True + +from DerivationFrameworkJetEtMiss.JetCommon import * +from DerivationFrameworkJetEtMiss.METCommon import * + + +#==================================================================== +# SET UP STREAM +#==================================================================== +streamName = derivationFlags.WriteDAOD_BPHY7Stream.StreamName +fileName = buildFileName( derivationFlags.WriteDAOD_BPHY7Stream ) + +BPHY7Stream = MSMgr.NewPoolRootStream( streamName, fileName ) +BPHY7Stream.AcceptAlgs(["BPHY7Kernel2"]) + +## 0/ setup vertexing tools and services +#include( "JpsiUpsilonTools/configureServices.py" ) + +include("DerivationFrameworkBPhys/configureVertexing.py") +BPHY7_VertexTools = BPHYVertexTools("BPHY7") + + +#==================================================================== +# TriggerCounting for Kernel1 #Added by Matteo +#==================================================================== +#List of trigggers to be counted (high Sig-eff*Lumi ones are in) +triggersToMetadata= ["HLT_2mu10", + "HLT_2mu10_bJpsimumu", + "HLT_2mu10_bJpsimumu_delayed", + "HLT_2mu10_l2msonly", + "HLT_2mu10_nomucomb", + "HLT_2mu14", + "HLT_2mu14_nomucomb", + "HLT_2mu4", + "HLT_2mu4_bBmumuxv2", + "HLT_2mu4_bDimu_noinvm_novtx_ss", + "HLT_2mu6", + "HLT_2mu6_10invm30_pt2_z10", + "HLT_2mu6_bBmumu", + "HLT_2mu6_bBmumux_Taumumux", + "HLT_2mu6_bBmumuxv2", + "HLT_2mu6_bBmumuxv2_delayed", + "HLT_2mu6_bDimu_noinvm_novtx_ss", + "HLT_2mu6_bJpsimumu", + "HLT_2mu6_bJpsimumu_delayed", + "HLT_2mu6_bJpsimumu_Lxy0_delayed", + "HLT_2mu6_nomucomb_bPhi", + "HLT_2mu6_nomucomb_mu4_nomucomb_bTau_L12MU6_3MU4", + "HLT_3mu4", + "HLT_3mu4_bDimu", + "HLT_3mu4_bDimu2700", + "HLT_3mu4_bTau", + "HLT_3mu4_l2msonly", + "HLT_3mu4_nomucomb_bTau", + "HLT_3mu4_nomucomb_delayed", + "HLT_3mu6", + "HLT_3mu6_bTau", + "HLT_3mu6_msonly", + "HLT_mu10_mu6_bBmumux_BcmumuDsloose_delayed", + "HLT_mu10_mu6_bBmumux_Taumumux", + "HLT_mu10_mu6_bBmumux_Taumumux_noL2", + "HLT_mu10_mu6_bBmumuxv2", + "HLT_mu10_mu6_bBmumuxv2_delayed", + "HLT_mu10_mu6_bJpsimumu", + "HLT_mu10_mu6_bJpsimumu_Lxy0", + "HLT_mu10_mu6_bJpsimumu_Lxy0_delayed", + "HLT_mu10_mu6_bUpsimumu", + "HLT_mu11_mu6_bBmumu", + "HLT_mu11_mu6_bBmumux_BpmumuKp", + "HLT_mu11_mu6_bBmumuxv2", + "HLT_mu11_mu6_bDimu", + "HLT_mu11_mu6_bDimu2700", + "HLT_mu11_mu6_bDimu2700_Lxy0", + "HLT_mu11_mu6_bDimu_Lxy0", + "HLT_mu11_mu6_bJpsimumu", + "HLT_mu11_mu6_bJpsimumu_Lxy0", + "HLT_mu11_mu6_bPhi", + "HLT_mu11_mu6_bTau", + "HLT_mu11_mu6_bUpsimumu", + "HLT_mu11_mu6noL1_bPhi_L1MU11_2MU6", + "HLT_mu10_mu6_bDimu", + "HLT_2mu6_bBmumu_Lxy0_L1BPH-2M9-2MU6_BPH-2DR15-2MU6", + "HLT_2mu6_bJpsimumu_Lxy0_L1BPH-2M9-2MU6_BPH-2DR15-2MU6", + "HLT_2mu10_bDimu", + "HLT_mu11_2mu4noL1_nscan03_L1MU11_2MU6", + "HLT_mu11_L1MU10_2mu4noL1_nscan03_L1MU10_2MU6", + "HLT_mu11_nomucomb_2mu4noL1_nscan03_L1MU11_2MU6", + "HLT_mu11_nomucomb_2mu4noL1_nscan03_L1MU11_2MU6_bTau", + "HLT_mu11_nomucomb_mu6noL1_nscan03_L1MU11_2MU6", + "HLT_mu11_nomucomb_mu6noL1_nscan03_L1MU11_2MU6_bTau", + "HLT_mu11_nomucomb_mu6noL1_nscan03_L1MU11_2MU6_bTau_delayed", + "HLT_mu18_2mu4noL1", + "HLT_mu18_mu8noL1", + "HLT_mu20_2mu4noL1", + "HLT_mu20_l2idonly_mu6noL1_nscan03", + "HLT_mu20_l2idonly_mu6noL1_nscan03_bTau", + "HLT_mu20_msonly_mu6noL1_msonly_nscan05", + "HLT_mu20_mu8noL1", + "HLT_mu20_nomucomb_mu6noL1_nscan03", + "HLT_mu20_nomucomb_mu6noL1_nscan03_bTau", + "HLT_mu22_2mu4noL1", + "HLT_mu22_mu8noL1", + "HLT_mu24_2mu4noL1", + "HLT_mu24_imedium", + "HLT_mu24_mu8noL1", + "HLT_mu26_ivarmedium", + "HLT_mu26i", + "HLT_mu50", + "HLT_mu6_2mu4", + "HLT_mu6_2mu4_bJpsi_delayed", + "HLT_mu6_2mu4_bTau_noL2", + "HLT_mu6_l2msonly_2mu4_l2msonly_L1MU6_3MU4", + "HLT_mu6_mu4_bBmumuxv2", + "HLT_mu6_mu4_bBmumuxv2_delayed", + "HLT_mu6_mu4_bDimu_noinvm_novtx_ss", + "HLT_mu6_nomucomb_2mu4_nomucomb_bTau_L1MU6_3MU4", + "HLT_mu6_nomucomb_2mu4_nomucomb_delayed_L1MU6_3MU4", + "HLT_mu20_mu6noL1_bTau", + "HLT_2mu6_mu4_bTau_L12MU6_3MU4", + "HLT_mu6_2mu4_bTau_L1MU6_3MU4", + "HLT_mu11_2mu4noL1_bTau_L1MU11_2MU6", + "HLT_mu11_mu6noL1_bTau_L1MU11_2MU6", + "HLT_3mu4_bPhi", + "HLT_mu11_mu6_bPhi", + "HLT_mu11_nomucomb_mu6_nomucomb_bPhi", + "HLT_mu11_nomucomb_mu6noL1_nscan03_L1MU11_2MU6_bPhi", + "HLT_mu6_2mu4_bTau_L1MU6_3MU4", + "HLT_mu20_mu6btrk_bTauTight", + "HLT_mu20_2mu2btrk_bTauTight", + "HLT_mu11_2mu2btrk_bTauTight_L1MU11_2MU6", + "HLT_3mu4_bPhi", + "HLT_mu11_mu6_bPhi", + "HLT_mu11_mu6noL1_bPhi_L1MU11_2MU6", + "HLT_mu11_mu6_bPhi_L1LFV-MU11", + "HLT_2mu6_bPhi_L1LFV-MU6" ] + + + + +triggersToMetadata_filter = list( set(triggersToMetadata) ) + +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__TriggerCountToMetadata +BPHY7TriggerCountToMetadata = DerivationFramework__TriggerCountToMetadata(name = "BPHY7TriggerCount", + TriggerList = triggersToMetadata_filter, + FolderName = "BPHY7") + +ToolSvc += BPHY7TriggerCountToMetadata + +#==================================================================== +# PRESELECTION for Kernel1 #Added by Matteo +#==================================================================== +## 1/ Setup the skimming based on triggers +## + +triggerList = [ "HLT_2mu10", + "HLT_2mu10_l2msonly", + "HLT_2mu10_nomucomb", + "HLT_2mu14", + "HLT_mu50", + "HLT_2mu14_l2msonly", + "HLT_2mu14_nomucomb", + "HLT_2mu6_l2msonly_mu4_l2msonly_L12MU6_3MU4", + "HLT_2mu6_nomucomb_mu4_nomucomb_L12MU6_3MU4", + "HLT_mu6_2mu4", + "HLT_mu6_l2msonly_2mu4_l2msonly_L1MU6_3MU4", + "HLT_mu6_nomucomb_2mu4_nomucomb_L1MU6_3MU4", + "HLT_3mu6", + "HLT_3mu6_msonly", + "HLT_3mu6_nomucomb", + "HLT_mu4","HLT_mu6","HLT_mu10","HLT_mu18", + "HLT_mu14", + "HLT_mu24", + "HLT_mu24_L1MU15", + "HLT_2mu4", + "HLT_2mu6", + "HLT_mu20_L1MU15", + "HLT_mu18_2mu4noL1", + "HLT_mu18_nomucomb_2mu4noL1", + "HLT_mu20_2mu4noL1", + "HLT_mu20_l2idonly_2mu4noL1", + "HLT_mu20_nomucomb_2mu4noL1", + "HLT_mu18_mu8noL1", + "HLT_mu18_nomucomb_mu8noL1", + "HLT_mu20_mu8noL1", + "HLT_mu20_l2idonly_2mu4noL1", + "HLT_mu20_nomucomb_mu8noL1", + "HLT_mu22_mu8noL1", + "HLT_mu22_l2idonly_2mu4noL1", + "HLT_mu22_nomucomb_mu8noL1", + "HLT_mu22_2mu4noL1", + "HLT_mu22_nomucomb_2mu4noL1", + "HLT_mu20_2mu4noL1", "HLT_mu20_mu8noL1", + "HLT_mu14_tau25_medium1_tracktwo", + "HLT_mu14_tau35_medium1_tracktwo", + "HLT_mu14_tau25_medium1_tracktwo_xe50", + "HLT_mu14_tau35_medium1_tracktwo_L1TAU20", + "HLT_mu24_mu8noL1", + "HLT_mu6_nomucomb_2mu4_nomucomb_delayed_L1MU6_3MU4", + "HLT_2mu6_bBmumuxv2_delayed", + "HLT_2mu4_bDimu_noinvm_novtx_ss", + "HLT_2mu6_bDimu_noinvm_novtx_ss", + "HLT_mu24_2mu4noL1", + "HLT_mu10_mu6_bUpsimumu", + "HLT_mu10_mu6_bBmumuxv2", + "HLT_mu10_mu6_bJpsimumu", + "HLT_mu6_mu4_bBmumuxv2_delayed", + "HLT_2mu6_10invm30_pt2_z10", + "HLT_2mu6_nomucomb_bPhi", + "HLT_mu6_mu4_bDimu_noinvm_novtx_ss", + "HLT_mu11_mu6_bDimu2700", + "HLT_2mu6_bBmumux_Taumumux", + "HLT_mu10_mu6_bBmumux_Taumumux", + "HLT_mu10_mu6_bBmumux_Taumumux_noL2", + "HLT_.*mu11_mu6.*", # Recent triggers + "HLT_.*3mu4.*", + "HLT_.*mu.*imedium.*", # Trigger with looser isolation selection + "HLT_.*mu.*iloose.*", + "HLT_.*mu6.*2mu4.*", + "HLT_.*mu11.*2mu4noL1.*", + "HLT_.*2mu14_nomucomb.*", + "HLT_.*bTau.*", # Our tau triggers + "HLT_.*bDimu2700.*", + "HLT_.*bPhi.*", + "HLT_.*bBmumuxv2.*", + "HLT_.*nscan.*" ] # Narrow scan triggers + +from DerivationFrameworkTools.DerivationFrameworkToolsConf import DerivationFramework__TriggerSkimmingTool +BPHY7TriggerSkim = DerivationFramework__TriggerSkimmingTool(name = "BPHY7TriggerSkim", + TriggerListOR = triggerList, + TriggerListAND = [] ) + +ToolSvc += BPHY7TriggerSkim + + +#==================================================================== +# 2mu vertex for Kernel2 #Added by Matteo +#==================================================================== + +from JpsiUpsilonTools.JpsiUpsilonToolsConf import Analysis__JpsiFinder +BPHY7DiMuon_Finder = Analysis__JpsiFinder(name = "BPHY7DiMuon_Finder", + # OutputLevel = DEBUG, + muAndMu = True, + muAndTrack = False, + TrackAndTrack = False, + assumeDiMuons = True, + invMassUpper = 2900.0, # Cut just below the J/psi + invMassLower = 0.0, + Chi2Cut = 110., #CHANGED! Was 200 + oppChargesOnly = False, + allChargeCombinations = True, + combOnly = False, + atLeastOneComb = True, + useCombinedMeasurement = False, # Only takes effect if combOnly=True + muonCollectionKey = "Muons", + TrackParticleCollection = "InDetTrackParticles", + V0VertexFitterTool = BPHY7_VertexTools.TrkV0Fitter, # V0 vertex fitter + useV0Fitter = False, # if False a TrkVertexFitterTool will be used + TrkVertexFitterTool = BPHY7_VertexTools.TrkVKalVrtFitter, # VKalVrt vertex fitter + TrackSelectorTool = BPHY7_VertexTools.InDetTrackSelectorTool, + VertexPointEstimator = BPHY7_VertexTools.VtxPointEstimator, + useMCPCuts = False) +ToolSvc += BPHY7DiMuon_Finder + +#-------------------------------------------------------------------- +##Comment from BPHY2... +## 3/ setup the vertex reconstruction "call" tool(s). They are part of the derivation framework. +## These Augmentation tools add output vertex collection(s) into the StoreGate and add basic +## decorations which do not depend on the vertex mass hypothesis (e.g. lxy, ptError, etc). +## There should be one tool per topology, i.e. Jpsi and Psi(2S) do not need two instance of the +## Reco tool is the JpsiFinder mass window is wide enough. +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__Reco_Vertex +BPHY7DiMuon_SelectAndWrite = DerivationFramework__Reco_Vertex(name = "BPHY7DiMuon_SelectAndWrite", + VertexSearchTool = BPHY7DiMuon_Finder, + OutputVtxContainerName = "BPHY7TwoMuCandidates", + PVContainerName = "PrimaryVertices", + RefPVContainerName = "SHOULDNOTBEUSED_DiMuonRefittedPV") +ToolSvc += BPHY7DiMuon_SelectAndWrite + +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__Select_onia2mumu + +## a/ augment and select Jpsi->mumu candidates +BPHY7DiMuon_Decorator = DerivationFramework__Select_onia2mumu(name = "BPHY7DiMuon_Decorator", + HypothesisName = "Jpsi", + InputVtxContainerName = "BPHY7TwoMuCandidates", + VtxMassHypo = 1230, # used to determine time-of-flight and thus lifetime (deviations and sigmas are also added to the vertex) + MassMin = 0.0, + MassMax = 2900.0, + Chi2Max = 200, + DoVertexType =1) # 1 = Pt, 2 = A0, 4 = Z0 + +ToolSvc += BPHY7DiMuon_Decorator +#==================================================================== +# 3mu/2mu+trk vertex for Kernel2 #Added by Matteo +#==================================================================== +## 4/ setup a new vertexing tool (necessary due to use of mass constraint) +from TrkVKalVrtFitter.TrkVKalVrtFitterConf import Trk__TrkVKalVrtFitter +BpmVertexFit = Trk__TrkVKalVrtFitter(name = "BpmVertexFit", + Extrapolator = BPHY7_VertexTools.InDetExtrapolator, + FirstMeasuredPoint = True, + MakeExtendedVertex = True) +ToolSvc += BpmVertexFit + +## 5/ setup the Jpsi+1 track finder +from JpsiUpsilonTools.JpsiUpsilonToolsConf import Analysis__JpsiPlus1Track +BPHY7ThreeMuon_Finder = Analysis__JpsiPlus1Track(name = "BPHY7ThreeMuon_Finder", + OutputLevel = INFO, + pionHypothesis = True, + kaonHypothesis = False, + trkThresholdPt = 1000.0, + #trkMaxEta = 2.5, # is this value fine?? default would be 102.5 + BThresholdPt = 1000.0, + BMassUpper = 5000.0, # What is this?? + BMassLower = 0.0, + JpsiContainerKey = "BPHY7TwoMuCandidates", + TrackParticleCollection = "InDetTrackParticles", + MuonsUsedInJpsi = "NONE", #cannnot allow, would kill 3muons + ExcludeCrossJpsiTracks = False, + TrkVertexFitterTool = BpmVertexFit, + TrackSelectorTool = BPHY7_VertexTools.InDetTrackSelectorTool, + UseMassConstraint = False, + Chi2Cut = 150) #Cut on chi2/Ndeg_of_freedom, so is very loose + + +ToolSvc += BPHY7ThreeMuon_Finder + +## 6/ setup the combined augmentation/skimming tool for the Bpm +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__Reco_Vertex +BPHY7ThreeMuon_SelectAndWrite = DerivationFramework__Reco_Vertex(name = "BPHY7ThreeMuon_SelectAndWrite", + OutputLevel = INFO, + VertexSearchTool = BPHY7ThreeMuon_Finder, + OutputVtxContainerName = "BPHY7Tau3MuCandidates", + PVContainerName = "PrimaryVertices", + RefPVContainerName = "BPHY7RefittedPrimaryVertices", + RefitPV = True, + MaxPVrefit = 1000) +ToolSvc += BPHY7ThreeMuon_SelectAndWrite + +## b/ augment and select Bplus->JpsiKplus candidates +BPHY7ThreeMuon_Decorator = DerivationFramework__Select_onia2mumu( + name = "BPHY7ThreeMuon_Decorator", + OutputLevel = INFO, + HypothesisName = "Tau3MuLoose", + InputVtxContainerName = "BPHY7Tau3MuCandidates", + TrkMasses = [105.658, 105.658, 105.658], + VtxMassHypo = 1777., + MassMin = 0.0, + MassMax = 5000., # If the two selections start differing one might have to check that the tools below still run on the right vertices + Chi2Max = 100.) + +ToolSvc += BPHY7ThreeMuon_Decorator + +## b/ augment and select Bplus->JpsiKplus candidates +BPHY7ThreeMuon_Decorator2 = DerivationFramework__Select_onia2mumu( + name = "BPHY7ThreeMuon_Decorator2", + OutputLevel = INFO, + HypothesisName = "Ds2MuPi", + InputVtxContainerName = "BPHY7Tau3MuCandidates", + TrkMasses = [105.658, 105.658, 139.57], + VtxMassHypo = 1968.3, + MassMin = 0.0, + MassMax = 5000., # If the two selections start differing one might have to check that the tools below still run on the right vertices + Chi2Max = 100.) + +ToolSvc += BPHY7ThreeMuon_Decorator2 + +#Track isolation for candidates +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__VertexTrackIsolation +BPHY7TrackIsolationDecorator = DerivationFramework__VertexTrackIsolation( + name = "BPHY7TrackIsolationDecorator", + OutputLevel = INFO, + TrackIsoTool = "xAOD::TrackIsolationTool", + TrackContainer = "InDetTrackParticles", + InputVertexContainer = "BPHY7Tau3MuCandidates", + PassFlags = ["passed_Tau3MuLoose", "passed_Ds2MuPi"] ) + +ToolSvc += BPHY7TrackIsolationDecorator + +#CaloIsolationTool explicitly declared to avoid pointless warnings (it works!!!) +from IsolationTool.IsolationToolConf import xAOD__CaloIsolationTool +BPHY7CaloIsolationTool = xAOD__CaloIsolationTool( + name = "BPHY7CaloIsolationTool", + OutputLevel = WARNING, + saveOnlyRequestedCorrections = True, + IsoLeakCorrectionTool = "" ) #Workaround for a bug in older versions + +ToolSvc += BPHY7CaloIsolationTool + +#Calo isolation for candidates +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__VertexCaloIsolation +BPHY7CaloIsolationDecorator = DerivationFramework__VertexCaloIsolation( + name = "BPHY7CaloIsolationDecorator", + OutputLevel = INFO, + CaloIsoTool = BPHY7CaloIsolationTool, #"xAOD::CaloIsolationTool", + TrackContainer = "InDetTrackParticles", + InputVertexContainer = "BPHY7Tau3MuCandidates", + CaloClusterContainer = "CaloCalTopoClusters", + ParticleCaloExtensionTool = "Trk::ParticleCaloExtensionTool/ParticleCaloExtensionTool", + PassFlags = ["passed_Tau3MuLoose", "passed_Ds2MuPi"] ) + +ToolSvc += BPHY7CaloIsolationDecorator + +#==================================================================== +# Skimming tool to select only events with the correct vertices +#==================================================================== + +#-------------------------------------------------------------------- +## 9/ select the event. We only want to keep events that contain certain three-mu vertices which passed certain selection. +## Exactly like in the preselection, where only 2mu vertices are treated. + +expression = "count(BPHY7Tau3MuCandidates.passed_Tau3MuLoose) > 0 || count(BPHY7Tau3MuCandidates.passed_Ds2MuPi) > 0" + +from DerivationFrameworkTools.DerivationFrameworkToolsConf import DerivationFramework__xAODStringSkimmingTool +BPHY7_SelectEvent = DerivationFramework__xAODStringSkimmingTool(name = "BPHY7_SelectEvent", + OutputLevel = INFO, + expression = expression) + +ToolSvc += BPHY7_SelectEvent +print(BPHY7_SelectEvent) + +#==================================================================== +# Add Extrapolation of muons to trigger layers +#==================================================================== + +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__MuonExtrapolationTool +BPHY7_Extrap_Tool = DerivationFramework__MuonExtrapolationTool( name = "BPHY7_ExtrapolationTool", OutputLevel = INFO ) + +ToolSvc += BPHY7_Extrap_Tool + + +#==================================================================== +# Thinning Helper and various thinning tools +#==================================================================== + +#-------------------------------------------------------------------- +## 10/ Setup the thinning helper, only tool able to perform thinning of trigger navigation information + +from DerivationFrameworkCore.ThinningHelper import ThinningHelper +BPHY7ThinningHelper = ThinningHelper( "BPHY7ThinningHelper" ) +BPHY7ThinningHelper.TriggerChains = 'HLT_.*mu.*' #triggerList # . = any character; * = 0 or more times; + = 1 or more times; ? 0 or 1 times "Regular_Expression" +BPHY7ThinningHelper.AppendToStream( BPHY7Stream ) + + +#-------------------------------------------------------------------- +## 11/ track and vertex thinning. We want to remove all reconstructed secondary vertices +## which haven't passed any of the selections defined by (Select_*) tools. +## We also want to keep only tracks which are associates with either muons or any of the +## vertices that passed the selection. Multiple thinning tools can perform the +## selection. The final thinning decision is based OR of all the decisions (by default, +## although it can be changed by the JO). + +## 12/ Cleans up, removing duplicate vertices. An issue caused by the logic of Jpsi+1 track in the case of 3-muon candidates + +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__Thin_vtxDuplicates +BPHY7Thin_vtxDuplicates = DerivationFramework__Thin_vtxDuplicates(name = "BPHY7Thin_vtxDuplicates", + OutputLevel = INFO, + VertexContainerName = "BPHY7Tau3MuCandidates", + PassFlags = ["passed_Tau3MuLoose", "passed_Ds2MuPi"]) + +ToolSvc += BPHY7Thin_vtxDuplicates + +## a) thining out vertices that didn't pass any selection and idetifying tracks associated with +## selected vertices. The "VertexContainerNames" is a list of the vertex containers, and "PassFlags" +## contains all pass flags for Select_* tools that must be satisfied. The vertex is kept is it +## satisfy any of the listed selections. + +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__Thin_vtxTrk +BPHY7Thin_vtxTrk = DerivationFramework__Thin_vtxTrk( + name = "BPHY7Thin_vtxTrk", + OutputLevel = INFO, + TrackParticleContainerName = "InDetTrackParticles", + AcceptanceRadius = 1., + VertexContainerNames = ["BPHY7Tau3MuCandidates"], + PassFlags = ["passed_Tau3MuLoose", "passed_Ds2MuPi"], + ApplyAnd = True ) # "and" requirement for Vertices + +ToolSvc += BPHY7Thin_vtxTrk + + +## 13/ thinning out tracks that are not attached to muons. The final thinning decision is based on the OR operation +## between decision from this and the previous tools. +from DerivationFrameworkInDet.DerivationFrameworkInDetConf import DerivationFramework__MuonTrackParticleThinning +BPHY7MuonTPThinningTool = DerivationFramework__MuonTrackParticleThinning(name = "BPHY7MuonTPThinningTool", + MuonKey = "Muons", + InDetTrackParticlesKey = "InDetTrackParticles") +ToolSvc += BPHY7MuonTPThinningTool + +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__BPhysPVThinningTool +BPHY7_thinningTool_PV = DerivationFramework__BPhysPVThinningTool(name = "BPHY7_thinningTool_PV", + CandidateCollections = ["BPHY7Tau3MuCandidates"], + KeepPVTracks =True) + +ToolSvc += BPHY7_thinningTool_PV + +from DerivationFrameworkInDet.DerivationFrameworkInDetConf import DerivationFramework__TauTrackParticleThinning +BPHY7TauTPThinningTool = DerivationFramework__TauTrackParticleThinning(name = "BPHY7TauTPThinningTool", + TauKey = "TauJets", + InDetTrackParticlesKey = "InDetTrackParticles") +ToolSvc += BPHY7TauTPThinningTool + +# Only save truth informtion directly associated with: mu Ds+ D+ D*+ Ds*+ D0 D*0 B+ B*+ B0 B*0 +from DerivationFrameworkMCTruth.DerivationFrameworkMCTruthConf import DerivationFramework__GenericTruthThinning +BPHY7TruthThinTool = DerivationFramework__GenericTruthThinning(name = "BPHY7TruthThinTool", + ParticleSelectionString = "abs(TruthParticles.pdgId) == 13 || abs(TruthParticles.pdgId) == 431 || abs(TruthParticles.pdgId) == 411 || abs(TruthParticles.pdgId) == 413 || abs(TruthParticles.pdgId) == 433 || TruthParticles.pdgId == 421 || TruthParticles.pdgId == 423 || abs(TruthParticles.pdgId) == 521 || abs(TruthParticles.pdgId) == 523 || TruthParticles.pdgId == 511 || TruthParticles.pdgId == 513", + PreserveDescendants = True, + PreserveAncestors = True) +ToolSvc += BPHY7TruthThinTool + +# Only save truth neutrino and b/c quarks information +from DerivationFrameworkMCTruth.DerivationFrameworkMCTruthConf import DerivationFramework__GenericTruthThinning +BPHY7TruthThinNoChainTool = DerivationFramework__GenericTruthThinning(name = "BPHY7TruthThinNoChainTool", + ParticleSelectionString = "abs(TruthParticles.pdgId) == 4 || abs(TruthParticles.pdgId) == 5 || abs(TruthParticles.pdgId) == 12 || abs(TruthParticles.pdgId) == 14 || abs(TruthParticles.pdgId) == 16", + PreserveDescendants = False, + PreserveAncestors = False) +ToolSvc += BPHY7TruthThinNoChainTool + + +#==================================================================== +# CREATE THE DERIVATION KERNEL ALGORITHM AND PASS THE ABOVE TOOLS +#==================================================================== + +BPHY7ThinningTools = [ BPHY7MuonTPThinningTool, BPHY7Thin_vtxDuplicates, BPHY7Thin_vtxTrk, BPHY7_thinningTool_PV, BPHY7TauTPThinningTool] + +BPHY7SkimmingTools = [BPHY7_SelectEvent] + +BPHY7AugmentationTools = [BPHY7DiMuon_SelectAndWrite, BPHY7DiMuon_Decorator, BPHY7ThreeMuon_SelectAndWrite, BPHY7ThreeMuon_Decorator, BPHY7ThreeMuon_Decorator2, BPHY7TrackIsolationDecorator, BPHY7CaloIsolationDecorator] + +if addMuExtrapolationForTrigger: + BPHY7AugmentationTools.append(BPHY7_Extrap_Tool) + +Kernel1Tools = [BPHY7TriggerSkim] + +if isSimulation: + #BPHY7AugmentationTools.append(DFCommonTauTruthMatchingWrapper) + if thinTruth: + BPHY7ThinningTools.append(BPHY7TruthThinTool) + BPHY7ThinningTools.append(BPHY7TruthThinNoChainTool) + +#The sequence object. Is in principle just a wrapper which allows to run two kernels in sequence +BPHY7_Sequence = CfgMgr.AthSequencer("BPHY7_Sequence") +from DerivationFrameworkFlavourTag.FlavourTagCommon import FlavorTagInit +FlavorTagInit(JetCollections=['AntiKt4EMPFlowJets'], Sequencer=BPHY7_Sequence) + + +#onlyAugmentations implementation +if onlyAugmentations: + Kernel1Tools = [] + BPHY7SkimmingTools = [] + BPHY7ThinningTools = [] + +# Kernel n1 PRESELECTION +# The name of the kernel (BPHY7Kernel1 in this case) must be unique to this derivation +from DerivationFrameworkCore.DerivationFrameworkCoreConf import DerivationFramework__DerivationKernel +BPHY7_Sequence += CfgMgr.DerivationFramework__DerivationKernel("BPHY7Kernel1", + AugmentationTools = [BPHY7TriggerCountToMetadata] , + SkimmingTools = Kernel1Tools) +# Kernel n2 deep Derivation +# The name of the kernel (BPHY7Kernel2 in this case) must be unique to this derivation +from DerivationFrameworkCore.DerivationFrameworkCoreConf import DerivationFramework__DerivationKernel +BPHY7_Sequence += CfgMgr.DerivationFramework__DerivationKernel("BPHY7Kernel2", + AugmentationTools = BPHY7AugmentationTools, + SkimmingTools = BPHY7SkimmingTools, + ThinningTools = BPHY7ThinningTools) + +#Vital, replaces the adding of kernels directly +DerivationFrameworkJob += BPHY7_Sequence + +#==================================================================== +# Slimming +#==================================================================== + +from DerivationFrameworkCore.SlimmingHelper import SlimmingHelper +BPHY7SlimmingHelper = SlimmingHelper("BPHY7SlimmingHelper") + + +SmartCollections = ["Electrons", "Photons", "TauJets", "AntiKt4EMTopoJets_BTagging201810", "BTagging_AntiKt4EMTopo_201810", "PrimaryVertices", "Muons", "InDetTrackParticles", "MET_Reference_AntiKt4EMTopo"] + + +AllVariables = ["METAssoc_AntiKt4EMTopo", + "MET_Core_AntiKt4EMTopo", + "MET_Truth", + "MET_Track", + "MET_LocHadTopo"] + +AllVariables += ["Kt4EMTopoOriginEventShape", + "Kt4EMTopoEventShape"] + +AllVariables += ["CombinedMuonTrackParticles", + "ExtrapolatedMuonTrackParticles", + "MuonSpectrometerTrackParticles"] + + +ExtraVariables = ["Photons.pt.eta.phi.m", + "Electrons.pt.eta.phi.m","TauJets.pt.eta.phi.m.IsTruthMatched.truthJetLink.truthParticleLink", + "AntiKt4EMTopoJets_BTagging201810.JetPileupScaleMomentum_pt.JetPileupScaleMomentum_eta.JetPileupScaleMomentum_phi.JetPileupScaleMomentum_m", + "AntiKt4EMTopoJets_BTagging201810.JvtJvfcorr.HECFrac.LArQuality.HECQuality.NegativeE.AverageLArQF", + "AntiKt4EMTopoJets_BTagging201810.JetEtaJESScaleMomentum_pt.JetEtaJESScaleMomentum_eta.JetEtaJESScaleMomentum_phi.JetEtaJESScaleMomentum_m"] + +ExtraVariables += ["Muons.etaLayer1Hits.etaLayer2Hits.etaLayer3Hits.etaLayer4Hits.phiLayer1Hits.phiLayer2Hits.phiLayer3Hits.phiLayer4Hits", + "Muons.numberOfTriggerEtaLayers.numberOfPhiLayers", + "CombinedMuonTrackParticles.numberOfTRTHits.numberOfTRTHighThresholdHits", + "InDetTrackParticles.numberOfTRTHits.numberOfTRTHighThresholdHits.vx.vy.vz", + "PrimaryVertices.chiSquared.covariance"] + + +StaticContent = ["xAOD::VertexContainer#BPHY7RefittedPrimaryVertices", + "xAOD::VertexAuxContainer#BPHY7RefittedPrimaryVerticesAux."] + +# ThreeBody candidates (vertices) +StaticContent += ["xAOD::VertexContainer#%s" % BPHY7ThreeMuon_SelectAndWrite.OutputVtxContainerName] +StaticContent += ["xAOD::VertexAuxContainer#%sAux." % BPHY7ThreeMuon_SelectAndWrite.OutputVtxContainerName] +## we have to disable vxTrackAtVertex branch since it is not xAOD compatible +StaticContent += ["xAOD::VertexAuxContainer#%sAux.-vxTrackAtVertex" % BPHY7ThreeMuon_SelectAndWrite.OutputVtxContainerName] + +# Truth information for MC only +if isSimulation: + AllVariables += ["TruthEvents","TruthParticles","TruthVertices","MuonTruthParticles", "METMap_Truth"] + SmartCollections += ["AntiKt4TruthJets"] + +# Needed for trigger objects +BPHY7SlimmingHelper.IncludeMuonTriggerContent = True +BPHY7SlimmingHelper.IncludeBPhysTriggerContent = True + +# Pass all lists to the SlimmingHelper +BPHY7SlimmingHelper.ExtraVariables = ExtraVariables +BPHY7SlimmingHelper.AllVariables = AllVariables +BPHY7SlimmingHelper.StaticContent = StaticContent +BPHY7SlimmingHelper.SmartCollections = SmartCollections +BPHY7SlimmingHelper.AppendContentToStream(BPHY7Stream) + diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/BPHY8.py b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/BPHY8.py new file mode 100644 index 00000000000..5e3de69ae0d --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/BPHY8.py @@ -0,0 +1,2200 @@ +#==================================================================== +# +# Copyright (C) 2002-2020 CERN for the benefit of the ATLAS collaboration +# +# @file BPHY8.py +# +# @author W. Walkowiak, <wolfgang.walkowiak@cern.ch> +# +# Based on example derivation formats. +# It requires the reductionConf flag BPHY8 in Reco_tf.py +# +# Produces DxAODs for the B(s)->mu+mu- analysis including the reference +# channels B+->J/psiK+ and Bs->J/psiPhi: +# * For data vertex containers for all three channels are produced +# in parallel. +# * For signal or reference channel MC the appropriate configuration +# is set according to the dataset number (DSN). The list associating +# known dataset numbers to decay channels (below) needs to be adjusted +# in case there are new MC samples with new numbers. +# +#==================================================================== +# Set up common services and job object. +# This should appear in ALL derivation job options +from DerivationFrameworkCore.DerivationFrameworkMaster import * + +# for debugging output +from pprint import pprint +from egammaRec.Factories import getPropertyValue + +# more debug messages +## svcMgr.MessageSvc.debugLimit = 5000000 +## svcMgr.MessageSvc.debugLimit = 5000 + +# Set up Bmumu configuration (metadata) tracking tool. +# This tool imports our defaults from Bmumu_metadata.cxx. +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf \ + import DerivationFramework__Bmumu_metadata +BPHY8_MetaDataTool = DerivationFramework__Bmumu_metadata( + name = "BPHY8_metadata", + DerivationName = "BPHY8", + OutputLevel = WARNING, + # verbosity of python script (0 - 10) + verbose = 10 +) +# local shorthand and ensure default contents of __slots__ dict are +# available as attributes +from DerivationFrameworkBPhys.BPhysPyHelpers import BPhysEnsureAttributes +BPHY8cf = BPhysEnsureAttributes(BPHY8_MetaDataTool) + +# add it to the ToolSvc chain +ToolSvc += BPHY8_MetaDataTool + +print(BPHY8_MetaDataTool) +pprint(BPHY8_MetaDataTool.properties()) + +# data or simulation? +if globalflags.DataSource() == 'geant4': + BPHY8cf.isSimulation = True + +# project tag +BPHY8cf.projectTag = rec.projectName() + +# trigger stream name +from RecExConfig.InputFilePeeker import inputFileSummary +if inputFileSummary is not None: + BPHY8cf.triggerStream = inputFileSummary['tag_info']['triggerStreamOfFile'] + +# release 21 or newer? +from PyJobTransforms.trfUtils import releaseIsOlderThan +BPHY8cf.isRelease21 = not releaseIsOlderThan(21,0) + +# MC campaigns by MC run number +BPHY8MCcampaigns = {284500 : 'mc16a', + 300000 : 'mc16d', + 310000 : 'mc16e'} + +# run number and MC campaign by MC run number +from AthenaCommon.AthenaCommonFlags import athenaCommonFlags +import PyUtils.AthFile as BPHY8_af +BPHY8_f = BPHY8_af.fopen(athenaCommonFlags.PoolAODInput()[0]) +BPHY8cf.mcCampaign = 'unknown' +if len(BPHY8_f.run_numbers) > 0: + BPHY8cf.runNumber = int(BPHY8_f.run_numbers[0]) + if BPHY8cf.isSimulation and BPHY8cf.runNumber in BPHY8MCcampaigns: + BPHY8cf.mcCampaign = BPHY8MCcampaigns[BPHY8cf.runNumber] + +print("BPHY8: isSimulation = %s" % BPHY8cf.isSimulation) +print("BPHY8: project tag = %s" % BPHY8cf.projectTag) +print("BPHY8: MC campaign = %s" % BPHY8cf.mcCampaign) +print("BPHY8: DerivationFrameworkHasTruth = %s" % \ + DerivationFrameworkHasTruth) +print("BPHY8: release 21 or up: %s" % BPHY8cf.isRelease21) + +#==================================================================== +# MC dataset categories (lists of dataset numbers) +#==================================================================== +BPHY8cf.mcBsmumu = [300203,300306,300307,300308,300402,300426,300430,300446,300447] +BPHY8cf.mcBplusJpsiKplus = [300203,300306,300307,300308,300997,300999,300404,300405,300406,300437] +BPHY8cf.mcBsJpsiPhi = [300203,300306,300307,300308,300401,300438,300448,300449] +BPHY8cf.mcBplusJpsiPiplus = [300406,300437] +BPHY8cf.mcBhh = [300431,300432,300433,300434] +BPHY8cf.mcNoTrigger = [300446,300447,300448,300449] + +#==================================================================== +# Data datasets to receive special treatment +#==================================================================== +# Remember our special runs used during validation +## BPHY8cf.specDataRuns = [302393,339849,358096] +BPHY8cf.specDataRuns = [] +# +# for testing only +## BPHY8cf.specDataRuns += [337491] + +#==================================================================== +# MC datasets to receive special treatment +#==================================================================== +# Remember our special MC datasets used during validation +## BPHY8cf.specMcChannels = [300307,300404,300405,300426,300430,300438] +BPHY8cf.specMcChannels = [] +# +#==================================================================== +# Defaults for BPHY8 configuration +#==================================================================== +# +# Blind search setup +# +# Enable? +BPHY8cf.doBmumuBlinding = True +# Cut blinded values/vertices? +BPHY8cf.doCutBlinded = False +## BPHY8cf.doCutBlinded = True +# Blind only candidates where all muons are of quality tight +BPHY8cf.blindOnlyAllMuonsTight = True +# Variables to blind (of vertices) +BPHY8cf.BlindedVars = "Bsmumu_mass.Bsmumu_MUCALC_mass" +# Pass flag indicating blinded candidates +BPHY8cf.BlindingFlag = "Bsmumu_blinded" +# Blinding key for testing +## BPHY8cf.BlindingKey = "0b0408d1f5c4760e7d4b50e97095" +# Blinding key for production +# key for Run 2 - 2015/16 only analysis +## BPHY8cf.BlindingKey = "0b04087bdac4564252fd778ac351" +# keys for full Run 2 analysis +BPHY8_data15BlindingKey = "0b040820229968c09fec401ace33" +BPHY8_data16BlindingKey = "0b04083901ad3d2bb3881ffca6a7" +BPHY8_data17BlindingKey = "0b040831a13a9c83f9936cf5b703" +BPHY8_data18BlindingKey = "0b040893fc715e9b346759bf4f3b" +# default is data15 +BPHY8cf.BlindingKey = BPHY8_data15BlindingKey +# +# Thinning level +# 0 - simple vertex thinning using Thin_vtxTrk. +# 1 - thinning subdecay vertex candidates using Thin_vtxTrk. +# 2 - thinning subdecay vertex candidates using BmumuThinningTool. +# 3 - thinning subdecay vertex candidates using BmumuThinningTool, +# but keeping all PVs and refittedPVs +# 4 - thinning subdecay vertex candidates using BmumuThinningTool, +# but keeping all PVs and refittedPVs and all ID tracks +# 5 - thinning subdecay vertex candidates using BmumuThinningTool, +# but keeping all PVs, all refittedPVs, all ID tracks and all muons +# +BPHY8cf.thinLevel = 3 +# +# Track particle collection +BPHY8cf.TrkPartContName = "InDetTrackParticles" +# +# Primary vertex collection +BPHY8cf.PVContName = "PrimaryVertices" +# +# Trigger navigation object thinning +# +# Apply thinning? +BPHY8cf.doTrigNavThinning = True +# +# Keep muon based HLT items for now +BPHY8cf.TrigNavThinList = [ "HLT_[0-9]*mu[0-9]+.*" ] +# +# Add containers for soft B tagging vertices +BPHY8cf.doAddSoftBVertices = True +# +# Muon collection +BPHY8cf.MuonCollection = "Muons" +# +# Apply MCP calibration to muons? (only for MC) +# +# Options: +# 0 : none +# 1 : apply calibration on-the-fly inside the muon container +# (calibration effects B candidate building) +# 2 : apply calibration and store it in an extra container for +# calibrated muons +# (calibration does not effect B candidate building) +# 3 : apply calibration and store it in an extra container for +# calibrated muons and use this container for B candidate building +# +BPHY8cf.useCalibratedMuons = 3 +## BPHY8cf.useCalibratedMuons = 2 + +# make sure data are not affected +# Update: Needs to be applied to data as well +## if not BPHY8cf.isSimulation: +## BPHY8cf.useCalibratedMuons = 0 + +# +# Configuration for MuonCalibrationAndSmearingTool +# +# Set string values to "_READ_" to only read the default values +# from the tool. +# +# THE FOLLOWING COMMENT IS VOID AS OF 2017-07-30. +# Note: Several JOs of the MuonCalibrationAndSmearingTool are not +# (yet) available in AtlasDerivation-20.7.8.x caches which uses +# MuonMomentumCorrections-01-00-35 instead of the latest +# MuonMomentumCorrections-01-00-60. +# For now these options will be disabled further below. +# +# Note: (2018-05-05 -> 2018-11-29) +# Formerly used the rel. 21 pre-recommendations from +# https://twiki.cern.ch/twiki/bin/view/AtlasProtected/MCPAnalysisGuidelinesMC16 +# +# Note: (2018-11-29) +# Now updated to new rel. 21 pre-recommendations +# Page revision r18 (as of 2018-11-27) +# https://twiki.cern.ch/twiki/bin/view/AtlasProtected/MCPAnalysisConsolidationMC16 +# +# Note: (2020-01-15) +# Now updated to new release 21 recommendatons for full run 2 (winter update) +# Page revision r17 (as of 2019-11-13) +# https://twiki.cern.ch/twiki/bin/view/AtlasProtected/MCPAnalysisWinterMC16#Momentum_corrections +# +# Note: (2020-03-31) +# Now updated to new release 21 recommendations for full run 2 / setup 1 +# Page reivision r37 (as of 2020-03-23) +# https://twiki.cern.ch/twiki/bin/view/AtlasProtected/MCPAnalysisGuidelinesMC16#Momentum_corrections +# +# MC +if BPHY8cf.isSimulation: +# +# for MC16a + if BPHY8cf.mcCampaign == "mc16a": + BPHY8cf.McstYear = "Data16" + BPHY8cf.McstRelease = "Recs2020_03_03" + BPHY8cf.McstStatComb = False + BPHY8cf.McstSagittaCorr = True + BPHY8cf.McstSagittaRelease = "sagittaBiasDataAll_03_02_19_Data16" + BPHY8cf.McstDoSagittaMCDistortion = False + BPHY8cf.McstSagittaCorrPhaseSpace = True +# +# for MC16d + elif BPHY8cf.mcCampaign == "mc16d": + BPHY8cf.McstYear = "Data17" + BPHY8cf.McstRelease = "Recs2020_03_03" + BPHY8cf.McstStatComb = False + BPHY8cf.McstSagittaCorr = True + BPHY8cf.McstSagittaRelease = "sagittaBiasDataAll_03_02_19_Data17" + BPHY8cf.McstDoSagittaMCDistortion = False + BPHY8cf.McstSagittaCorrPhaseSpace = True +# +# for MC16e + elif BPHY8cf.mcCampaign == "mc16e": + BPHY8cf.McstYear = "Data18" + BPHY8cf.McstRelease = "Recs2020_03_03" + BPHY8cf.McstStatComb = False + BPHY8cf.McstSagittaCorr = True + BPHY8cf.McstSagittaRelease = "sagittaBiasDataAll_03_02_19_Data18" + BPHY8cf.McstDoSagittaMCDistortion = False + BPHY8cf.McstSagittaCorrPhaseSpace = True +# +# default (like for mc16a) + else: + BPHY8cf.McstYear = "Data16" + BPHY8cf.McstRelease = "Recs2020_03_03" + BPHY8cf.McstStatComb = False + BPHY8cf.McstSagittaCorr = True + BPHY8cf.McstSagittaRelease = "sagittaBiasDataAll_03_02_19_Data16" + BPHY8cf.McstDoSagittaMCDistortion = False + BPHY8cf.McstSagittaCorrPhaseSpace = True +# +# data 15 +else: + # Note: The recommendation page sets McstYear to 'Data16' + if BPHY8cf.projectTag.startswith("data15"): + BPHY8cf.McstYear = "Data16" + BPHY8cf.McstRelease = "Recs2020_03_03" + BPHY8cf.McstStatComb = False + BPHY8cf.McstSagittaCorr = True + BPHY8cf.McstSagittaRelease = "sagittaBiasDataAll_03_02_19_Data16" + BPHY8cf.McstDoSagittaMCDistortion = False + BPHY8cf.McstSagittaCorrPhaseSpace = True +# +# data 16 + if BPHY8cf.projectTag.startswith("data16"): + BPHY8cf.McstYear = "Data16" + BPHY8cf.McstRelease = "Recs2020_03_03" + BPHY8cf.McstStatComb = False + BPHY8cf.McstSagittaCorr = True + BPHY8cf.McstSagittaRelease = "sagittaBiasDataAll_03_02_19_Data16" + BPHY8cf.McstDoSagittaMCDistortion = False + BPHY8cf.McstSagittaCorrPhaseSpace = True +# +# data 17 + if BPHY8cf.projectTag.startswith("data17"): + BPHY8cf.McstYear = "Data17" + BPHY8cf.McstRelease = "Recs2020_03_03" + BPHY8cf.McstStatComb = False + BPHY8cf.McstSagittaCorr = True + BPHY8cf.McstSagittaRelease = "sagittaBiasDataAll_03_02_19_Data17" + BPHY8cf.McstDoSagittaMCDistortion = False + BPHY8cf.McstSagittaCorrPhaseSpace = True +# +# data 18 + if BPHY8cf.projectTag.startswith("data18"): + BPHY8cf.McstYear = "Data18"; + BPHY8cf.McstRelease = "Recs2020_03_03" + BPHY8cf.McstStatComb = False + BPHY8cf.McstSagittaCorr = True + BPHY8cf.McstSagittaRelease = "sagittaBiasDataAll_03_02_19_Data18" + BPHY8cf.McstDoSagittaMCDistortion = False + BPHY8cf.McstSagittaCorrPhaseSpace = True + +# wide mumu mass range? +BPHY8cf.doUseWideMuMuMassRange = False + +# other default settings +BPHY8cf.GlobalChi2CutBase = 15. + +# Global mass values (in MeV, from PDG 2015) +BPHY8cf.GlobalMuonMass = 105.6584 +BPHY8cf.GlobalPionMass = 139.57061 +BPHY8cf.GlobalKaonMass = 493.677 +BPHY8cf.GlobalJpsiMass = 3096.92 +BPHY8cf.GlobalBplusMass = 5279.29 +BPHY8cf.GlobalB0Mass = 5279.61 +BPHY8cf.GlobalBsMass = 5366.79 + +# Cut values for kaon candidates +BPHY8cf.GlobalKaonPtCut = 1000. +BPHY8cf.GlobalKaonEtaCut = 2.5 + +# primary vertex association types (interpreted bit-wise) +## BPHY8cf.doVertexType = 15 # ALL +BPHY8cf.doVertexType = 8 # only Z0_BA +## BPHY8cf.doVertexType = 9 # only SUM_PT2 and Z0_BA + +# minimum number of tracks in PV considered for PV association +BPHY8cf.minNTracksInPV = 3 + +# add 3-dimensional proper time information? +BPHY8cf.do3dProperTime = True + +# use invariant mass based on combined muon track information in mass cuts? +BPHY8cf.useMuCalcMass = True + +# add MUCALC mass from non-modified muons for debugging +BPHY8cf.addMucalcMassForDebug = False + +# PV types to be considered in calculation of minLogChi2ToAnyPV variable +BPHY8cf.MinChi2ToAnyPVTypes = [1, 3] + +# MCP cuts for JpsiFinder +BPHY8cf.useJpsiFinderMCPCuts = False + +# reject muons in JpsiPlus1Track or JpsiPlus2Track finders +BPHY8cf.GlobalMuonsUsedInJpsi = "NONE" # default to turn it off + +# mode of minLogChi2ToAnyPV calculation: +# 0 : no such calculation +# 1 : use all PVs of requested type(s) +# 2 : exclude PVs associated to SVs +# 3 : replace PVs associated to SVs by +# corresponding refitted PVs +BPHY8cf.AddMinChi2ToAnyPVMode = 3 + +# Vertex isolation -- track selection requirements +# (Sizes of all lists below need to be identical!) +# Set to "Custom" (for strings) or -1. (for numerics) to disable setting +BPHY8cf.IsoTrackCategoryName = ["LoosePt05", "LooSiHi1Pt05"] +BPHY8cf.IsoTrackCutLevel = ["Loose" , "Loose" ] +BPHY8cf.IsoTrackPtCut = [ 500. , 500. ] +BPHY8cf.IsoTrackEtaCut = [ -1. , -1. ] +BPHY8cf.IsoTrackPixelHits = [ -1 , 1 ] +BPHY8cf.IsoTrackSCTHits = [ -1 , 2 ] +BPHY8cf.IsoTrackbLayerHits = [ -1 , -1 ] +BPHY8cf.IsoTrackIBLHits = [ -1 , -1 ] +# Vertex isolation -- cone sizes +# (Sizes of all lists below need to be identical!) +# Note: IsoDoTrkImpLogChi2Cut = 2 implements old method used +# for the 2015/16 analysis +BPHY8cf.IsolationConeSizes = [ 0.7, 0.7, 1.0] +BPHY8cf.IsoTrkImpLogChi2Max = [ 5.0, 5.0, 0.0] +BPHY8cf.IsoDoTrkImpLogChi2Cut = [ 2 , 1 , 0 ] +# Track types to be used (bit pattern) +# track sets to consider: +# bit : meaning +# 0 : tracks close to PV associated +# with SV +# 1 : tracks associated with dummy PV +# ("type-0 PV tracks") +# 2 : tracks associated with PV of type 1 +# 3 : tracks associated with PV of type 2 +# 4 : tracks associated with PV of type 3 +# 5 : tracks associated with PV with types other than 0 to 4. +# 6 : tracks with missing pointer to PV (NULL pointer) +# 7-24: tracks being closest to assoc. PV +# decimal useRefittedPVs doDCAin3D chi2DefToUse +# 7 : 128 yes no 0 +# 8 : 256 no no 0 +# 9 : 512 yes yes 0 +# 10 : 1024 no yes 0 +# 11 : 2048 yes no 1 +# 12 : 4096 no no 1 +# 13 : 8192 yes yes 1 +# 14 : 16384 no yes 1 +# 15 : 32768 yes no 2 +# 16 : 65536 no no 2 +# 17 : 131072 yes yes 2 +# 18 : 262144 no yes 2 +# 19 : 524288 yes -- 3 +# 20 : 1048576 no -- 3 +# 21 : 2097152 yes -- 4 +# 22 : 4194304 no -- 4 +# 23 : 8388608 yes -- 5 +# 24 : 16777216 no -- 5 +# 25 : 33554432 yes yes 6 +# 26 : 67108864 no yes 6 +# 27 : 134217728 yes yes 7 +# 28 : 268435456 no yes 7 +# 29 : 536870912 yes yes 8 +# 30 : 1073741824 no yes 8 +# 31 : 2147483648 yes yes 9 +# 32 : 4294967296 no yes 9 +# useRefittedPVs: +# replace PV associated to decay candidate +# by the refitted PV +# doDCAin3D: +# use d0 and z0 in the determination of +# of the point of closest approach of +# a track to a vertex +# chi2DefToUse: +# PV uncertainties in the chi2 calculation +# in addition to track uncertainties +# 0 : from track perigee (old method) +# (only track uncertainties) +# 1 : from track perigee with +# uncertainties from track and vertex +# 2 : simple extrapolation from track +# parameters with uncertainties from +# track and vertex (extrapolation +# used for track swimming) +# 3 : CalcLogChi2toPV method from NtupleMaker +# using xAOD::TrackingHelpers. +# (only track uncertainties) +# 4 : CalcLogChi2toPV method from NtupleMaker +# using xAOD::TrackingHelpers. +# (track and vertex uncertainties) +# 5 : use TrackVertexAssociationTool +# 6 : full 3D chi2 from track perigee with uncertainties +# from track and vertex (sum of 3x3 covariance matrices) +# 7 : full 3D chi2 from track perigee with uncertainties +# from track and vertex (sum of 2x2 covariance matrices) +# 8 : simple extrapolation from track parameters with uncertainties +# from track and vertex (sum of 3x3 covariance matrices) +# 9 simple extrapolation from track parameters with uncertainties +# from track and vertex (sum of 2x2 covariance matrices) +# (E.g. 127 means to consider all tracks.) +BPHY8cf.useIsoTrackTypes = [ 35, 8388608, 134217728, 127] +# Working point for TrackVertexAssociationTool (for chi2DefToUse == 5) +BPHY8cf.IsoTvaWorkingPoint = "Loose" +# use of speed-optimized algorithm +BPHY8cf.IsoUseOptimizedAlgo = True +## BPHY8cf.IsoUseOptimizedAlgo = False +# +# Combinations to keep: save from removal as non-needed branches +# Tuples: (isolation settings|track types|ID track selection) +# Note: use an empty list to keep all +BPHY8cf.IsoIncludes = ['07_LC50d2|35|LoosePt05', # ACH + '10_LC00d0|134217728|LooSiHi1Pt05', # BEJ + '10_LC00d0|8388608|LooSiHi1Pt05', # BGJ + '07_LC50d1|127|LooSiHi1Pt05' ] # BDI + +# Isolation for muons from B candidate -- track selection requirements +# (Sizes of all lists below need to be identical!) +# Set to "Custom" (for strings) or -1. (for numerics) to disable setting +BPHY8cf.MuIsoTrackCategoryName = ["LoosePt05", "LooSiHi1Pt05"] +BPHY8cf.MuIsoTrackCutLevel = ["Loose" , "Loose" ] +BPHY8cf.MuIsoTrackPtCut = [ 500. , 500. ] +BPHY8cf.MuIsoTrackEtaCut = [ -1. , -1. ] +BPHY8cf.MuIsoTrackPixelHits = [ -1 , 1 ] +BPHY8cf.MuIsoTrackSCTHits = [ -1 , 2 ] +BPHY8cf.MuIsoTrackbLayerHits = [ -1 , -1 ] +BPHY8cf.MuIsoTrackIBLHits = [ -1 , -1 ] +# Muon isolation -- cone sizes +# (Sizes of all lists below need to be identical!) +# Note: MuIsoDoTrkImpLogChi2Cut = 2 implements old method used +# for the 2015/16 analysis +BPHY8cf.MuIsolationConeSizes = [ 0.7, 0.7, 1.0] +BPHY8cf.MuIsoTrkImpLogChi2Max = [ 5.0, 5.0, 0.0] +BPHY8cf.MuIsoDoTrkImpLogChi2Cut = [ 2 , 1 , 0 ] +# Track types to be used (bit pattern) +# track sets to consider: +# bit : meaning +# 0 : tracks close to PV associated +# with SV +# 1 : tracks associated with dummy PV +# ("type-0 PV tracks") +# 2 : tracks associated with PV of type 1 +# 3 : tracks associated with PV of type 2 +# 4 : tracks associated with PV of type 3 +# 5 : tracks associated with PV with types other than 0 to 4. +# 6 : tracks with missing pointer to PV (NULL pointer) +# 7-24: tracks being closest to assoc. PV +# decimal useRefittedPVs doDCAin3D chi2DefToUse +# 7 : 128 yes no 0 +# 8 : 256 no no 0 +# 9 : 512 yes yes 0 +# 10 : 1024 no yes 0 +# 11 : 2048 yes no 1 +# 12 : 4096 no no 1 +# 13 : 8192 yes yes 1 +# 14 : 16384 no yes 1 +# 15 : 32768 yes no 2 +# 16 : 65536 no no 2 +# 17 : 131072 yes yes 2 +# 18 : 262144 no yes 2 +# 19 : 524288 yes -- 3 +# 20 : 1048576 no -- 3 +# 21 : 2097152 yes -- 4 +# 22 : 4194304 no -- 4 +# 23 : 8388608 yes -- 5 +# 24 : 16777216 no -- 5 +# 25 : 33554432 yes yes 6 +# 26 : 67108864 no yes 6 +# 27 : 134217728 yes yes 7 +# 28 : 268435456 no yes 7 +# 29 : 536870912 yes yes 8 +# 30 : 1073741824 no yes 8 +# 31 : 2147483648 yes yes 9 +# 32 : 4294967296 no yes 9 +# useRefittedPVs: +# replace PV associated to decay candidate +# by the refitted PV +# doDCAin3D: +# use d0 and z0 in the determination of +# of the point of closest approach of +# a track to a vertex +# chi2DefToUse: +# PV uncertainties in the chi2 calculation +# in addition to track uncertainties +# 0 : from track perigee (old method) +# (only track uncertainties) +# 1 : from track perigee with +# uncertainties from track and vertex +# 2 : simple extrapolation from track +# parameters with uncertainties from +# track and vertex (extrapolation +# used for track swimming) +# 3 : CalcLogChi2toPV method from NtupleMaker +# using xAOD::TrackingHelpers. +# (only track uncertainties) +# 4 : CalcLogChi2toPV method from NtupleMaker +# using xAOD::TrackingHelpers. +# (track and vertex uncertainties) +# 5 : use TrackVertexAssociationTool +# 6 : full 3D chi2 from track perigee with uncertainties +# from track and vertex (sum of 3x3 covariance matrices) +# 7 : full 3D chi2 from track perigee with uncertainties +# from track and vertex (sum of 2x2 covariance matrices) +# 8 : simple extrapolation from track parameters with uncertainties +# from track and vertex (sum of 3x3 covariance matrices) +# 9 simple extrapolation from track parameters with uncertainties +# from track and vertex (sum of 2x2 covariance matrices) +# (E.g. 127 means to consider all tracks.) +BPHY8cf.useMuIsoTrackTypes = [ 35, 8388608, 134217728, 127] +# Working point for TrackVertexAssociationTool (for chi2DefToUse == 5) +BPHY8cf.MuIsoTvaWorkingPoint = "Loose" +# +# Combinations to keep: save from removal as non-needed branches +# Tuples: (isolation settings|track types|ID track selection) +# Note: use an empty list to keep all +BPHY8cf.MuIsoIncludes = ['07_LC50d2|35|LoosePt05', # ACH + '10_LC00d0|134217728|LooSiHi1Pt05', # BEJ + '10_LC00d0|8388608|LooSiHi1Pt05', # BGJB + '07_LC50d1|127|LooSiHi1Pt05' ] # BDI + +# Closest track finding -- track selection requirements +# Set to "Custom" (for strings) or -1. (for numerics) to disable setting +BPHY8cf.CloseTrackCategoryName = ["LoosePt05", "LooSiHi1Pt05"] +BPHY8cf.CloseTrackCutLevel = ["Loose" , "Loose" ] +BPHY8cf.CloseTrackPtCut = [ 500. , 500. ] +BPHY8cf.CloseTrackEtaCut = [ -1. , -1. ] +BPHY8cf.CloseTrackPixelHits = [ -1 , 1 ] +BPHY8cf.CloseTrackSCTHits = [ -1 , 2 ] +BPHY8cf.CloseTrackbLayerHits = [ -1 , -1 ] +BPHY8cf.CloseTrackIBLHits = [ -1 , -1 ] +# Track types to be used (bit pattern) +# track sets to consider: +# bit : meaning +# 0 : tracks close to PV associated +# with SV +# 1 : tracks associated with dummy PV +# ("type-0 PV tracks") +# 2 : tracks associated with PV of type 1 +# 3 : tracks associated with PV of type 2 +# 4 : tracks associated with PV of type 3 +# 5 : tracks associated with PV with types other than 0 to 4. +# 6 : tracks with missing pointer to PV (NULL pointer) +# 7-24: tracks being closest to assoc. PV +# decimal useRefittedPVs doDCAin3D chi2DefToUse +# 7 : 128 yes no 0 +# 8 : 256 no no 0 +# 9 : 512 yes yes 0 +# 10 : 1024 no yes 0 +# 11 : 2048 yes no 1 +# 12 : 4096 no no 1 +# 13 : 8192 yes yes 1 +# 14 : 16384 no yes 1 +# 15 : 32768 yes no 2 +# 16 : 65536 no no 2 +# 17 : 131072 yes yes 2 +# 18 : 262144 no yes 2 +# 19 : 524288 yes -- 3 +# 20 : 1048576 no -- 3 +# 21 : 2097152 yes -- 4 +# 22 : 4194304 no -- 4 +# 23 : 8388608 yes -- 5 +# 24 : 16777216 no -- 5 +# 25 : 33554432 yes yes 6 +# 26 : 67108864 no yes 6 +# 27 : 134217728 yes yes 7 +# 28 : 268435456 no yes 7 +# 29 : 536870912 yes yes 8 +# 30 : 1073741824 no yes 8 +# 31 : 2147483648 yes yes 9 +# 32 : 4294967296 no yes 9 +# useRefittedPVs: +# replace PV associated to decay candidate +# by the refitted PV +# doDCAin3D: +# use d0 and z0 in the determination of +# of the point of closest approach of +# a track to a vertex +# chi2DefToUse: +# PV uncertainties in the chi2 calculation +# in addition to track uncertainties +# 0 : from track perigee (old method) +# (only track uncertainties) +# 1 : from track perigee with +# uncertainties from track and vertex +# 2 : simple extrapolation from track +# parameters with uncertainties from +# track and vertex (extrapolation +# used for track swimming) +# 3 : CalcLogChi2toPV method from NtupleMaker +# using xAOD::TrackingHelpers. +# (only track uncertainties) +# 4 : CalcLogChi2toPV method from NtupleMaker +# using xAOD::TrackingHelpers. +# (track and vertex uncertainties) +# 5 : use TrackVertexAssociationTool +# 6 : full 3D chi2 from track perigee with uncertainties +# from track and vertex (sum of 3x3 covariance matrices) +# 7 : full 3D chi2 from track perigee with uncertainties +# from track and vertex (sum of 2x2 covariance matrices) +# 8 : simple extrapolation from track parameters with uncertainties +# from track and vertex (sum of 3x3 covariance matrices) +# 9 simple extrapolation from track parameters with uncertainties +# from track and vertex (sum of 2x2 covariance matrices) +# (E.g. 127 means to consider all tracks.) +# +# Correspondence to Run 1 settings: +# +# Option to only use tracks from specific primary vertices: +# (always excluding B decay tracks) +# CloseTrackOption: +# old new +# 0 : 63 : use all tracks (default) +# 1 : 1 : use only tracks from PV associated with B vertex +# 2 : 1 : use all tracks which are not from PVs other than +# PV associated with B vertex +# 3 : 35 : use all tracks which are not from PVs other than +# PV associated with B vertex but including those +# from the dummy vertex (type 0 vertex) +# 4 : 127: same as option 3 but using the vertex pointers +# for comparing in old setup; including tracks +# with broken (NULL) vertex pointers as well +BPHY8cf.useCloseTrackTypes = [ 35, 8388608, 134217728] +# Working point for TrackVertexAssociationTool (for chi2DefToUse == 5) +BPHY8cf.CloseTrackTvaWorkingPoint = "Loose" +# +# Close tracks chi2 related settings +# (The next five lists need to be exactly of the same length.) +BPHY8cf.CloseTrackChi2SetName = [ "201516", "f2dc2" ] +# use corrected chi2 calculation including SV uncertainties +# 0 : from track perigee (old method, only track uncertainties) +# 1 : from track perigee with uncertainties from track and vertex +# 2 : simple extrapolation from track parameters with uncertainties +# from track and vertex (extrapolation used for track swimming) +# 3 : CalcLogChi2toPV method from NtupleMaker using xAOD::TrackingHelpers. +# (only track uncertainties) +# 4 : CalcLogChi2toPV method from NtupleMaker using xAOD::TrackingHelpers. +# (track and vertex uncertainties) +# 5 : use TrackVertexAssociationTool +# 6 : full 3D chi2 from track perigee with uncertainties +# from track and vertex (sum of 3x3 covariance matrices) +# 7 : full 3D chi2 from track perigee with uncertainties +# from track and vertex (sum of 2x2 covariance matrices) +# 8 : simple extrapolation from track parameters with uncertainties +# from track and vertex (sum of 3x3 covariance matrices) +# 9 simple extrapolation from track parameters with uncertainties +# from track and vertex (sum of 2x2 covariance matrices) +# N.B.: Settings 3, 4 and 5 may be less reasonable here. Do not use. +BPHY8cf.CloseTrackCorrChi2 = [ 0 , 7 ] +# use 3-dimensional information in minimization +BPHY8cf.CloseTrackMinDCAin3D = [ True , True ] +# maximum chi2 distance of closest track to B vertex +BPHY8cf.CloseTrackMaxLogChi2 = [ 7. , 7. ] +# maximum chi2 distance of closest track to B vertex for track counting +BPHY8cf.NCloseTrackMaxLogChi2 = [ 1. , 2. ] +# +# Combinations to keep: save from removal as non-needed branches +# Tuples: (close track chi2 set|track types|ID track selection) +# Note: use an empty list to keep all +BPHY8cf.CloseTrackIncludes = ['201516|35|LoosePt05', # ACK + 'f2dc2|134217728|LooSiHi1Pt05', # BEL + 'f2dc2|8388608|LooSiHi1Pt05' ] # BGL + +# track/muon isolation and closest track tools +# debugging level for track types (output to log) +# (Set to 1 to enable, 0 otherwise.) +BPHY8cf.DebugTrackTypes = 0 + +# BTrackVertexMapLogger / BPhysTrackVertexMapTools +# maximum number of events to dump track-to-vertex assoc. maps for +# (Set to -1 for no limit, to 0 to disable.) +BPHY8cf.DebugTrkToVtxMaxEvents = 0 +#==================================================================== +# General job setup +#==================================================================== +# for MC run specific channel(s) only, +# for data run 2-, 3- and 4-prong algorithms in parallel +if BPHY8cf.isSimulation: + # MC channel number (ie dataset number for MC) + if len(BPHY8_f.infos['mc_channel_number']) > 0: + BPHY8cf.mcChNumber = int((BPHY8_f.infos['mc_channel_number'])[0]) + if (BPHY8cf.mcChNumber in BPHY8cf.mcBsmumu): + BPHY8cf.doChannels.append("Bsmumu") + if (BPHY8cf.mcChNumber in BPHY8cf.mcBplusJpsiKplus): + BPHY8cf.doChannels.append("BJpsiK") + if (BPHY8cf.mcChNumber in BPHY8cf.mcBsJpsiPhi): + BPHY8cf.doChannels.append("BsJpsiPhi") + if (BPHY8cf.mcChNumber in BPHY8cf.mcBplusJpsiPiplus): + BPHY8cf.doChannels.append("BJpsiPi") + if (BPHY8cf.mcChNumber in BPHY8cf.mcBhh): + BPHY8cf.doChannels.append("Bhh") + # use trigger? + if (BPHY8cf.mcChNumber in BPHY8cf.mcNoTrigger): + BPHY8cf.doTriggerInfo = False + # for special MC channels keep all ID tracks and all muons + if BPHY8cf.mcChNumber in BPHY8cf.specMcChannels: + BPHY8cf.thinLevel = 5 + # no blind search for MC + BPHY8cf.doBmumuBlinding = False +else: + # for data + BPHY8cf.doChannels += ["Bsmumu", "BJpsiK", "BsJpsiPhi"] + # for special data runs keep all ID tracks and all muons + if BPHY8cf.runNumber in BPHY8cf.specDataRuns: + BPHY8cf.thinLevel = 5 + # blinding key by year + if BPHY8cf.projectTag.startswith("data15"): + BPHY8cf.BlindingKey = BPHY8_data15BlindingKey + elif BPHY8cf.projectTag.startswith("data16"): + BPHY8cf.BlindingKey = BPHY8_data16BlindingKey + elif BPHY8cf.projectTag.startswith("data17"): + BPHY8cf.BlindingKey = BPHY8_data17BlindingKey + elif BPHY8cf.projectTag.startswith("data18"): + BPHY8cf.BlindingKey = BPHY8_data18BlindingKey + +# disable soft B tagging vertices if BJpsiK channel is not run +if not "BJpsiK" in BPHY8cf.doChannels: + BPHY8cf.doAddSoftBVertices = False + +print("BPHY8 job setup: run : %d" % BPHY8cf.runNumber) +print("BPHY8 job setup: MC channel number : %d" % BPHY8cf.mcChNumber) +print("BPHY8 job setup: isSimulation : %s" % BPHY8cf.isSimulation) +print("BPHY8 job setup: doChannels :", end=' ') +for BPHY8_channel in BPHY8cf.doChannels: + print("%s" % (BPHY8_channel), end=' ') +print() +print("BPHY8 job setup: thin level : %d" % BPHY8cf.thinLevel) +print("BPHY8 job setup: soft B vertices : %d" % BPHY8cf.doAddSoftBVertices) + +# abort if no channels are to be run on +assert len(BPHY8cf.doChannels) > 0 + +#==================================================================== +# Mass ranges +#==================================================================== +BPHY8cf.GlobalBMassUpperCut = 7000. +BPHY8cf.GlobalBMassLowerCut = 3500. +BPHY8cf.GlobalTrksMassUpperCut = 7500. +BPHY8cf.GlobalTrksMassLowerCut = 3000. +BPHY8cf.GlobalDiMuonMassUpperCut = 7000. +BPHY8cf.GlobalDiMuonMassLowerCut = 2000. +BPHY8cf.GlobalJpsiMassUpperCut = 7000. +BPHY8cf.GlobalJpsiMassLowerCut = 2000. +BPHY8cf.GlobalBlindLowerCut = 5166. +BPHY8cf.GlobalBlindUpperCut = 5526. + +if BPHY8cf.doUseWideMuMuMassRange: + BPHY8cf.GlobalBMassUpperCut = 10000. + BPHY8cf.GlobalBMassLowerCut = 3250. + BPHY8cf.GlobalTrksMassUpperCut = 10500. + BPHY8cf.GlobalTrksMassLowerCut = 2750. + BPHY8cf.GlobalDiMuonMassUpperCut = 10000. + BPHY8cf.GlobalDiMuonMassLowerCut = 2000. + BPHY8cf.GlobalJpsiMassUpperCut = 10000. + BPHY8cf.GlobalJpsiMassLowerCut = 2000. + +#==================================================================== +# Vertexing chi2 cuts for n-prong decays +#==================================================================== +# The global chi2 cut times NdF +BPHY8cf.Chi2Cut2Prong = BPHY8cf.GlobalChi2CutBase * 1. +BPHY8cf.Chi2Cut3Prong = BPHY8cf.GlobalChi2CutBase * 4. +BPHY8cf.Chi2Cut4Prong = BPHY8cf.GlobalChi2CutBase * 6. + +#==================================================================== +# Muons or tracks for JpsiFinder +#==================================================================== +BPHY8cf.JfTwoMuons = True +BPHY8cf.JfTwoTracks = False +BPHY8cf.JfTrackThresholdPt = 0. # MeV +if "Bhh" in BPHY8cf.doChannels: + BPHY8cf.JfTwoMuons = False + BPHY8cf.JfTwoTracks = True + BPHY8cf.JfTrackThresholdPt = 3500. # MeV + +#==================================================================== +# DEBUGGING SETUP +#==================================================================== +# +# Dump contents of this file to log +if BPHY8cf.verbose > 4: + import inspect + thisfile = inspect.getfile(inspect.currentframe()) + print("# >>>------------------ %s ------------------------" % thisfile) + with open (thisfile, 'r') as fin: + print(fin.read()) + print("# <<<------------------ %s ------------------------" % thisfile) + +# required for track jets for Soft B Tagging +if BPHY8cf.doAddSoftBVertices: + from DerivationFrameworkJetEtMiss.JetCommon import * + +#==================================================================== +# CALIBRATION SEQUENCES +#==================================================================== +# +# For parameters of the MuonCalibrationAndSmearingTool see: +# https://twiki.cern.ch/twiki/bin/view/AtlasProtected/MCPAnalysisGuidelinesMC15#Muon_momentum_scale_and_resoluti +# +# ordered dicts +from collections import OrderedDict +BPHY8_CalibrationAlgs = OrderedDict() + +# Create calibrated muons if requested +BPHY8cf.CalMuonCollection = BPHY8cf.MuonCollection +BPHY8cf.UsedMuonCollection = BPHY8cf.MuonCollection +BPHY8cf.AllMuonCollections = [ BPHY8cf.MuonCollection ] +if BPHY8cf.useCalibratedMuons > 0: + BPHY8cf.adjustMucalcKinematics = True + if BPHY8cf.useCalibratedMuons > 1: + BPHY8cf.CalMuonCollection = BPHY8cf.DerivationName+"_CalibratedMuons" + BPHY8cf.adjustMucalcKinematics = False + BPHY8cf.AllMuonCollections += [ BPHY8cf.CalMuonCollection ] + if BPHY8cf.useCalibratedMuons > 2: + BPHY8cf.UsedMuonCollection = BPHY8cf.CalMuonCollection + BPHY8cf.adjustMucalcKinematics = True + BPHY8_MuonCalTool = CfgMgr.CP__MuonCalibrationAndSmearingTool( + BPHY8cf.DerivationName+"_MCPTool", + OutputLevel = INFO ) + if BPHY8cf.McstYear != "_READ_": + BPHY8_MuonCalTool.Year = BPHY8cf.McstYear + if BPHY8cf.McstRelease != "_READ_": + BPHY8_MuonCalTool.Release = BPHY8cf.McstRelease + # read back string values + BPHY8cf.McstYear = getPropertyValue(BPHY8_MuonCalTool, "Year") + BPHY8cf.McstRelease = getPropertyValue(BPHY8_MuonCalTool, "Release") + # additional options for MuonMomentumCorrections-01-00-64 and up + # Don't decorate with Eigen (from MuonMomentumCorrections-01-00-62 + # onwards, see ATLASG-1126) + BPHY8_MuonCalTool.noEigenDecor = True + BPHY8_MuonCalTool.StatComb = BPHY8cf.McstStatComb + BPHY8_MuonCalTool.SagittaCorr = BPHY8cf.McstSagittaCorr + BPHY8_MuonCalTool.doSagittaMCDistortion = BPHY8cf.McstDoSagittaMCDistortion + BPHY8_MuonCalTool.SagittaCorrPhaseSpace = BPHY8cf.McstSagittaCorrPhaseSpace + if BPHY8cf.McstSagittaRelease != "_READ_": + BPHY8_MuonCalTool.SagittaRelease = BPHY8cf.McstSagittaRelease + # read back string value + BPHY8cf.McstSagittaRelease = getPropertyValue(BPHY8_MuonCalTool, + "SagittaRelease") + ToolSvc += BPHY8_MuonCalTool + print(BPHY8_MuonCalTool) + pprint(BPHY8_MuonCalTool.properties()) + BPHY8_CalibrationAlgs["CalMuonProvider"] = CfgMgr.CP__CalibratedMuonsProvider( + BPHY8cf.DerivationName+"_CalMuonProvider", + Input = BPHY8cf.MuonCollection, + Output = BPHY8cf.CalMuonCollection, + Tool = BPHY8_MuonCalTool, + OutputLevel = INFO ) # output only if set to VERBOSE + +# for quick debugging +## BPHY8cf.adjustMucalcKinematics = False + +for BPHY8_name in list(BPHY8_CalibrationAlgs.keys()): + print(BPHY8_CalibrationAlgs[BPHY8_name]) + pprint(BPHY8_CalibrationAlgs[BPHY8_name].properties()) + +#==================================================================== +# Muon extrapolation for trigger scaling +#==================================================================== +# Introduced by BPhys trigger group, see merge request 7857 +# (https://gitlab.cern.ch/atlas/athena/merge_requests/7857) +# +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf \ + import DerivationFramework__MuonExtrapolationTool + +BPHY8_MuonExtrapTool = DerivationFramework__MuonExtrapolationTool( + name = "BPHY8_MuonExtrapolationTool", + MuonCollection = BPHY8cf.UsedMuonCollection, + OutputLevel = INFO ) + +ToolSvc += BPHY8_MuonExtrapTool +print(BPHY8_MuonExtrapTool) +pprint(BPHY8_MuonExtrapTool.properties()) + +#==================================================================== +# AUGMENTATION TOOLS +#==================================================================== +# setup vertexing tools and services +# superseeded by BPHYVertexTools +## include( "JpsiUpsilonTools/configureServices.py" ) + +# we need to have the DiMuon finder running for channels with Jpsi as well +BPHY8_recoList = [] +if [BPHY8_i for BPHY8_i in BPHY8cf.doChannels \ + if BPHY8_i in ["Bsmumu", "BJpsiK", "BsJpsiPhi", "BJpsiPi", "Bhh"]]: + BPHY8_recoList += ["DiMuon"] +if "BJpsiK" in BPHY8cf.doChannels: BPHY8_recoList += [ "BJpsiK" ] +if "BsJpsiPhi" in BPHY8cf.doChannels: BPHY8_recoList += [ "BsJpsiPhi" ] +if "BJpsiPi" in BPHY8cf.doChannels: BPHY8_recoList += [ "BJpsiPi" ] + +# setup of vertexing tools per channel +include("DerivationFrameworkBPhys/configureVertexing.py") +BPHY8_VertexTools = OrderedDict() +for BPHY8_reco in BPHY8_recoList: + BPHY8_VertexTools[BPHY8_reco] = BPHYVertexTools(BPHY8cf.DerivationName+"_"+BPHY8_reco) + print(BPHY8_VertexTools[BPHY8_reco]) + +# setup of vertex finder tools +from JpsiUpsilonTools.JpsiUpsilonToolsConf import \ + Analysis__JpsiFinder, Analysis__JpsiPlus1Track, Analysis__JpsiPlus2Tracks + +BPHY8_FinderTools = OrderedDict() +for BPHY8_reco in BPHY8_recoList: +# a) for DiMuon + if BPHY8_reco == "DiMuon": + BPHY8_FinderTools[BPHY8_reco] = Analysis__JpsiFinder( + name = BPHY8cf.DerivationName+"_"+BPHY8_reco+"_Finder", + OutputLevel = INFO, + muAndMu = BPHY8cf.JfTwoMuons, + muAndTrack = False, + TrackAndTrack = BPHY8cf.JfTwoTracks, + doTagAndProbe = False, + assumeDiMuons = False, # If true, will assume dimu hypothesis and use PDG value for mu mass + track1Mass = BPHY8cf.GlobalMuonMass, + track2Mass = BPHY8cf.GlobalMuonMass, + muonThresholdPt = 0., + trackThresholdPt = BPHY8cf.JfTrackThresholdPt, + invMassUpper = BPHY8cf.GlobalDiMuonMassUpperCut, + invMassLower = BPHY8cf.GlobalDiMuonMassLowerCut, + # For JpsiFinder the cut is really on chi2 and not on chi2/ndf + Chi2Cut = BPHY8cf.Chi2Cut2Prong, + oppChargesOnly = True, + sameChargesOnly = False, + allChargeCombinations = False, + allMuons = True, + combOnly = False, + atLeastOneComb = False, + useCombinedMeasurement = False, # Only takes effect if combOnly=True + muonCollectionKey = BPHY8cf.UsedMuonCollection, + TrackParticleCollection = BPHY8cf.TrkPartContName, + V0VertexFitterTool = BPHY8_VertexTools[BPHY8_reco].TrkV0Fitter, # V0 vertex fitter + useV0Fitter = False, # if False a TrkVertexFitterTool will be used + TrkVertexFitterTool = BPHY8_VertexTools[BPHY8_reco].TrkVKalVrtFitter, # VKalVrt vertex fitter + TrackSelectorTool = BPHY8_VertexTools[BPHY8_reco].InDetTrackSelectorTool, + VertexPointEstimator = BPHY8_VertexTools[BPHY8_reco].VtxPointEstimator, + useMCPCuts = BPHY8cf.useJpsiFinderMCPCuts ) + +# b) for BJpsiK or BJpsiPi + if BPHY8_reco in [ "BJpsiK", "BJpsiPi" ] : + BPHY8_kaonHypo = (False if BPHY8_reco == "BJpsiPi" else True) + BPHY8_FinderTools[BPHY8_reco] = Analysis__JpsiPlus1Track( + name = BPHY8cf.DerivationName+"_"+BPHY8_reco+"_Finder", + OutputLevel = INFO, + pionHypothesis = not BPHY8_kaonHypo, + kaonHypothesis = BPHY8_kaonHypo, + trkThresholdPt = BPHY8cf.GlobalKaonPtCut, + trkMaxEta = BPHY8cf.GlobalKaonEtaCut, + BThresholdPt = 1000., + TrkTrippletMassUpper = BPHY8cf.GlobalTrksMassUpperCut, + TrkTrippletMassLower = BPHY8cf.GlobalTrksMassLowerCut, + BMassUpper = BPHY8cf.GlobalBMassUpperCut, + BMassLower = BPHY8cf.GlobalBMassLowerCut, + JpsiContainerKey = BPHY8cf.DerivationName+"DiMuonCandidates", + JpsiMassUpper = BPHY8cf.GlobalJpsiMassUpperCut, + JpsiMassLower = BPHY8cf.GlobalJpsiMassLowerCut, + MuonsUsedInJpsi = BPHY8cf.GlobalMuonsUsedInJpsi, + TrackParticleCollection = BPHY8cf.TrkPartContName, + TrkVertexFitterTool = BPHY8_VertexTools[BPHY8_reco].TrkVKalVrtFitter, # VKalVrt vertex fitter + TrackSelectorTool = BPHY8_VertexTools[BPHY8_reco].InDetTrackSelectorTool, + # This JO should rather be named Chi2byNdfCut + Chi2Cut = BPHY8cf.GlobalChi2CutBase, + UseMassConstraint = True, + ExcludeJpsiMuonsOnly = True, + ExcludeCrossJpsiTracks = False) + +# c) for BsJpsiPhi + if BPHY8_reco == "BsJpsiPhi": + BPHY8_FinderTools[BPHY8_reco] = Analysis__JpsiPlus2Tracks( + name = BPHY8cf.DerivationName+"_"+BPHY8_reco+"_Finder", + OutputLevel = INFO, + pionpionHypothesis = False, + kaonkaonHypothesis = True, + kaonpionHypothesis = False, + trkThresholdPt = BPHY8cf.GlobalKaonPtCut, + trkMaxEta = BPHY8cf.GlobalKaonEtaCut, + BThresholdPt = 1000., + TrkQuadrupletMassUpper = BPHY8cf.GlobalTrksMassUpperCut, + TrkQuadrupletMassLower = BPHY8cf.GlobalTrksMassLowerCut, + BMassUpper = BPHY8cf.GlobalBMassUpperCut, + BMassLower = BPHY8cf.GlobalBMassLowerCut, + JpsiContainerKey = BPHY8cf.DerivationName+"DiMuonCandidates", + JpsiMassUpper = BPHY8cf.GlobalJpsiMassUpperCut, + JpsiMassLower = BPHY8cf.GlobalJpsiMassLowerCut, + MuonsUsedInJpsi = BPHY8cf.GlobalMuonsUsedInJpsi, + TrackParticleCollection = BPHY8cf.TrkPartContName, + TrkVertexFitterTool = BPHY8_VertexTools[BPHY8_reco].TrkVKalVrtFitter, # VKalVrt vertex fitter + TrackSelectorTool = BPHY8_VertexTools[BPHY8_reco].InDetTrackSelectorTool, + # This JO should rather be named Chi2byNdfCut + Chi2Cut = BPHY8cf.GlobalChi2CutBase, + UseMassConstraint = True, + ExcludeJpsiMuonsOnly = True, + ExcludeCrossJpsiTracks = False) + +ToolSvc += list(BPHY8_FinderTools.values()) +for BPHY8_name in list(BPHY8_FinderTools.keys()): + print(BPHY8_FinderTools[BPHY8_name]) + pprint(BPHY8_FinderTools[BPHY8_name].properties()) + +#-------------------------------------------------------------------- +# Setup the vertex reconstruction "call" tool(s). They are part of the +# derivation framework. +# These Augmentation tools add output vertex collection(s) into the +# StoreGate and add basic decorations which do not depend on the vertex +# mass hypothesis (e.g. lxy, ptError, etc). +# There should be one tool per topology, i.e. Jpsi and Psi(2S) do not need +# two instance of the Reco tool if the JpsiFinder mass window is wide enough. +# +# The reconstruction tools must be interleaved with the vertex selection +# and augmentation tools as e.g. the Jpsimumu container ist needed for +# ???? +# + +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf \ + import DerivationFramework__Reco_Vertex +BPHY8_RecoTools = OrderedDict() +for BPHY8_reco in BPHY8_recoList: +# a) for DiMuon + if BPHY8_reco == "DiMuon": + BPHY8_RecoTools[BPHY8_reco] = DerivationFramework__Reco_Vertex( + name = BPHY8cf.DerivationName+"_"+BPHY8_reco+"_Reco", + JpsiFinder = BPHY8_FinderTools[BPHY8_reco], + OutputVtxContainerName = BPHY8cf.DerivationName+BPHY8_reco+"Candidates", + PVContainerName = BPHY8cf.PVContName, + RefPVContainerName = BPHY8cf.DerivationName+"DiMuonRefittedPrimaryVertices", + RefitPV = True, + Do3d = BPHY8cf.do3dProperTime, + MaxPVrefit = 100000, + MinNTracksInPV = BPHY8cf.minNTracksInPV, + DoVertexType = BPHY8cf.doVertexType) +# b) for BJpsiK + if BPHY8_reco in [ "BJpsiK", "BJpsiPi" ] : + BPHY8_RecoTools[BPHY8_reco] = DerivationFramework__Reco_Vertex( + name = BPHY8cf.DerivationName+"_"+BPHY8_reco+"_Reco", + Jpsi1PlusTrackName = BPHY8_FinderTools[BPHY8_reco], + OutputVtxContainerName = BPHY8cf.DerivationName+BPHY8_reco+"Candidates", + PVContainerName = BPHY8cf.PVContName, + RefPVContainerName = BPHY8cf.DerivationName+BPHY8_reco+"RefittedPrimaryVertices", + RefitPV = True, + Do3d = BPHY8cf.do3dProperTime, + MaxPVrefit = 100000, + MinNTracksInPV = BPHY8cf.minNTracksInPV, + DoVertexType = BPHY8cf.doVertexType) +# c) for BsJpsiPhi + if BPHY8_reco == "BsJpsiPhi": + BPHY8_RecoTools[BPHY8_reco] = DerivationFramework__Reco_Vertex( + name = BPHY8cf.DerivationName+"_"+BPHY8_reco+"_Reco", + Jpsi2PlusTrackName = BPHY8_FinderTools[BPHY8_reco], + OutputVtxContainerName = BPHY8cf.DerivationName+BPHY8_reco+"Candidates", + PVContainerName = BPHY8cf.PVContName, + RefPVContainerName = BPHY8cf.DerivationName+BPHY8_reco+"RefittedPrimaryVertices", + RefitPV = True, + Do3d = BPHY8cf.do3dProperTime, + MaxPVrefit = 100000, + MinNTracksInPV = BPHY8cf.minNTracksInPV, + DoVertexType = BPHY8cf.doVertexType) + +ToolSvc += list(BPHY8_RecoTools.values()) +for BPHY8_name in list(BPHY8_RecoTools.keys()): + print(BPHY8_RecoTools[BPHY8_name]) + pprint(BPHY8_RecoTools[BPHY8_name].properties()) + +#-------------------------------------------------------------------- +# Augmentation of vertices by MUCALC mass and it's error +# +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf \ + import DerivationFramework__BPhysAddMuonBasedInvMass + +BPHY8_MuMassTools = OrderedDict() +# a) for Bsmumu +if "Bsmumu" in BPHY8cf.doChannels: + # augment B(s)->mumu candidates + BPHY8_MuMassTools["Bsmumu"] = DerivationFramework__BPhysAddMuonBasedInvMass( + name = "BPHY8_MuMass_Bsmumu", + BranchPrefix = "Bsmumu", + OutputLevel = WARNING, + AdjustToMuonKinematics = BPHY8cf.adjustMucalcKinematics, + VertexContainerName = BPHY8cf.DerivationName+"DiMuonCandidates", + TrkMasses = [BPHY8cf.GlobalMuonMass, BPHY8cf.GlobalMuonMass], + AddMinChi2ToAnyPVMode = BPHY8cf.AddMinChi2ToAnyPVMode, + PrimaryVertexContainerName = BPHY8cf.PVContName, + MinNTracksInPV = BPHY8cf.minNTracksInPV, + PVTypesToConsider = BPHY8cf.MinChi2ToAnyPVTypes, + DoVertexType = BPHY8cf.doVertexType) +# b) for BJpsiK, BsJpsiPhi and BJpsiPi retain the Jpsi +if [i for i in BPHY8cf.doChannels if i in ["BJpsiK", "BsJpsiPhi", "BJpsiPi"]]: + BPHY8_MuMassTools["Jpsimumu"] = DerivationFramework__BPhysAddMuonBasedInvMass( + name = "BPHY8_MuMass_Jpsimumu", + BranchPrefix = "Jpsimumu", + OutputLevel = WARNING, + AdjustToMuonKinematics = BPHY8cf.adjustMucalcKinematics, + VertexContainerName = BPHY8cf.DerivationName+"DiMuonCandidates", + TrkMasses = [BPHY8cf.GlobalMuonMass, BPHY8cf.GlobalMuonMass], + AddMinChi2ToAnyPVMode = BPHY8cf.AddMinChi2ToAnyPVMode, + PrimaryVertexContainerName = BPHY8cf.PVContName, + MinNTracksInPV = BPHY8cf.minNTracksInPV, + PVTypesToConsider = BPHY8cf.MinChi2ToAnyPVTypes, + DoVertexType = BPHY8cf.doVertexType) +# c) for BJpsiK +if "BJpsiK" in BPHY8cf.doChannels: + # augment B+/- ->JpsiK+/- candidates + BPHY8_MuMassTools["BJpsiK"] = DerivationFramework__BPhysAddMuonBasedInvMass( + name = "BPHY8_MuMass_BJpsiK", + BranchPrefix = "BJpsiK", + OutputLevel = WARNING, + AdjustToMuonKinematics = BPHY8cf.adjustMucalcKinematics, + VertexContainerName = BPHY8cf.DerivationName+"BJpsiKCandidates", + TrkMasses = [BPHY8cf.GlobalMuonMass, BPHY8cf.GlobalMuonMass, BPHY8cf.GlobalKaonMass], + AddMinChi2ToAnyPVMode = BPHY8cf.AddMinChi2ToAnyPVMode, + PrimaryVertexContainerName = BPHY8cf.PVContName, + MinNTracksInPV = BPHY8cf.minNTracksInPV, + PVTypesToConsider = BPHY8cf.MinChi2ToAnyPVTypes, + DoVertexType = BPHY8cf.doVertexType) +# d) for BsJpsiPhi +if "BsJpsiPhi" in BPHY8cf.doChannels: + # augment Bs ->JpsiPhi candidates + BPHY8_MuMassTools["BsJpsiPhi"] = DerivationFramework__BPhysAddMuonBasedInvMass( + name = "BPHY8_MuMass_BsJpsiPhi", + BranchPrefix = "BsJpsiPhi", + OutputLevel = WARNING, + AdjustToMuonKinematics = BPHY8cf.adjustMucalcKinematics, + VertexContainerName = BPHY8cf.DerivationName+"BsJpsiPhiCandidates", + TrkMasses = [BPHY8cf.GlobalMuonMass, BPHY8cf.GlobalMuonMass, BPHY8cf.GlobalKaonMass, BPHY8cf.GlobalKaonMass], + AddMinChi2ToAnyPVMode = BPHY8cf.AddMinChi2ToAnyPVMode, + PrimaryVertexContainerName = BPHY8cf.PVContName, + MinNTracksInPV = BPHY8cf.minNTracksInPV, + PVTypesToConsider = BPHY8cf.MinChi2ToAnyPVTypes, + DoVertexType = BPHY8cf.doVertexType) +# e) for BJpsiPi +if "BJpsiPi" in BPHY8cf.doChannels: + # augment B+/- ->JpsiPi+/- candidates + BPHY8_MuMassTools["BJpsiPi"] = DerivationFramework__BPhysAddMuonBasedInvMass( + name = "BPHY8_MuMass_BJpsiPi", + BranchPrefix = "BJpsiPi", + OutputLevel = WARNING, + AdjustToMuonKinematics = BPHY8cf.adjustMucalcKinematics, + VertexContainerName = BPHY8cf.DerivationName+"BJpsiPiCandidates", + TrkMasses = [BPHY8cf.GlobalMuonMass, BPHY8cf.GlobalMuonMass, BPHY8cf.GlobalPionMass], + AddMinChi2ToAnyPVMode = BPHY8cf.AddMinChi2ToAnyPVMode, + PrimaryVertexContainerName = BPHY8cf.PVContName, + MinNTracksInPV = BPHY8cf.minNTracksInPV, + PVTypesToConsider = BPHY8cf.MinChi2ToAnyPVTypes, + DoVertexType = BPHY8cf.doVertexType) +# f) for Bhh +if "Bhh" in BPHY8cf.doChannels: + # augment B->hh candidates + BPHY8_MuMassTools["Bhh"] = DerivationFramework__BPhysAddMuonBasedInvMass( + name = "BPHY8_MuMass_Bhh", + BranchPrefix = "Bhh", + OutputLevel = WARNING, + AdjustToMuonKinematics = BPHY8cf.adjustMucalcKinematics, + VertexContainerName = BPHY8cf.DerivationName+"DiMuonCandidates", + TrkMasses = [BPHY8cf.GlobalMuonMass, BPHY8cf.GlobalMuonMass], + AddMinChi2ToAnyPVMode = BPHY8cf.AddMinChi2ToAnyPVMode, + PrimaryVertexContainerName = BPHY8cf.PVContName, + MinNTracksInPV = BPHY8cf.minNTracksInPV, + PVTypesToConsider = BPHY8cf.MinChi2ToAnyPVTypes, + DoVertexType = BPHY8cf.doVertexType) + +######################## duplication for debugging only ####################### + +if BPHY8cf.addMucalcMassForDebug: + # a) for Bsmumu + if "Bsmumu" in BPHY8cf.doChannels: + # augment B(s)->mumu candidates + BPHY8_MuMassTools["Bsmumu2"] = DerivationFramework__BPhysAddMuonBasedInvMass( + name = "BPHY8_MuMass_Bsmumu2", + BranchPrefix = "Bsmumu2", + OutputLevel = WARNING, + AdjustToMuonKinematics = False, + VertexContainerName = BPHY8cf.DerivationName+"DiMuonCandidates", + TrkMasses = [BPHY8cf.GlobalMuonMass, BPHY8cf.GlobalMuonMass], + AddMinChi2ToAnyPVMode = BPHY8cf.AddMinChi2ToAnyPVMode, + PrimaryVertexContainerName = BPHY8cf.PVContName, + MinNTracksInPV = BPHY8cf.minNTracksInPV, + PVTypesToConsider = BPHY8cf.MinChi2ToAnyPVTypes, + DoVertexType = BPHY8cf.doVertexType) + # b) for BJpsiK and BsJpsiPhi retain the Jpsi + if [i for i in BPHY8cf.doChannels if i in ["BJpsiK", "BsJpsiPhi"]]: + BPHY8_MuMassTools["Jpsimumu2"] = DerivationFramework__BPhysAddMuonBasedInvMass( + name = "BPHY8_MuMass_Jpsimumu2", + BranchPrefix = "Jpsimumu2", + OutputLevel = WARNING, + AdjustToMuonKinematics = False, + VertexContainerName = BPHY8cf.DerivationName+"DiMuonCandidates", + TrkMasses = [BPHY8cf.GlobalMuonMass, BPHY8cf.GlobalMuonMass], + AddMinChi2ToAnyPVMode = BPHY8cf.AddMinChi2ToAnyPVMode, + PrimaryVertexContainerName = BPHY8cf.PVContName, + MinNTracksInPV = BPHY8cf.minNTracksInPV, + PVTypesToConsider = BPHY8cf.MinChi2ToAnyPVTypes, + DoVertexType = BPHY8cf.doVertexType) + # c) for BJpsiK + if "BJpsiK" in BPHY8cf.doChannels: + # augment B+/- ->JpsiK+/- candidates + BPHY8_MuMassTools["BJpsiK2"] = DerivationFramework__BPhysAddMuonBasedInvMass( + name = "BPHY8_MuMass_BJpsiK2", + BranchPrefix = "BJpsiK2", + OutputLevel = WARNING, + AdjustToMuonKinematics = False, + VertexContainerName = BPHY8cf.DerivationName+"BJpsiKCandidates", + TrkMasses = [BPHY8cf.GlobalMuonMass, BPHY8cf.GlobalMuonMass, BPHY8cf.GlobalKaonMass], + AddMinChi2ToAnyPVMode = BPHY8cf.AddMinChi2ToAnyPVMode, + PrimaryVertexContainerName = BPHY8cf.PVContName, + MinNTracksInPV = BPHY8cf.minNTracksInPV, + PVTypesToConsider = BPHY8cf.MinChi2ToAnyPVTypes, + DoVertexType = BPHY8cf.doVertexType) + # d) for BsJpsiPhi + if "BsJpsiPhi" in BPHY8cf.doChannels: + # augment Bs ->JpsiPhi candidates + BPHY8_MuMassTools["BsJpsiPhi2"] = DerivationFramework__BPhysAddMuonBasedInvMass( + name = "BPHY8_MuMass_BsJpsiPhi2", + BranchPrefix = "BsJpsiPhi2", + OutputLevel = WARNING, + AdjustToMuonKinematics = False, + VertexContainerName = BPHY8cf.DerivationName+"BsJpsiPhiCandidates", + TrkMasses = [BPHY8cf.GlobalMuonMass, BPHY8cf.GlobalMuonMass, BPHY8cf.GlobalKaonMass, BPHY8cf.GlobalKaonMass], + AddMinChi2ToAnyPVMode = BPHY8cf.AddMinChi2ToAnyPVMode, + PrimaryVertexContainerName = BPHY8cf.PVContName, + MinNTracksInPV = BPHY8cf.minNTracksInPV, + PVTypesToConsider = BPHY8cf.MinChi2ToAnyPVTypes, + DoVertexType = BPHY8cf.doVertexType) + # e) for BJpsiPi + if "BJpsiPi" in BPHY8cf.doChannels: + # augment B+/- ->JpsiPi+/- candidates + BPHY8_MuMassTools["BJpsiPi2"] = DerivationFramework__BPhysAddMuonBasedInvMass( + name = "BPHY8_MuMass_BJpsiPi2", + BranchPrefix = "BJpsiPi2", + OutputLevel = WARNING, + AdjustToMuonKinematics = False, + VertexContainerName = BPHY8cf.DerivationName+"BJpsiPiCandidates", + TrkMasses = [BPHY8cf.GlobalMuonMass, BPHY8cf.GlobalMuonMass, BPHY8cf.GlobalPionMass], + AddMinChi2ToAnyPVMode = BPHY8cf.AddMinChi2ToAnyPVMode, + PrimaryVertexContainerName = BPHY8cf.PVContName, + MinNTracksInPV = BPHY8cf.minNTracksInPV, + PVTypesToConsider = BPHY8cf.MinChi2ToAnyPVTypes, + DoVertexType = BPHY8cf.doVertexType) + # f) for Bhh + if "Bhh" in BPHY8cf.doChannels: + # augment B->hh candidates + BPHY8_MuMassTools["Bhh2"] = DerivationFramework__BPhysAddMuonBasedInvMass( + name = "BPHY8_MuMass_Bhh2", + BranchPrefix = "Bhh2", + OutputLevel = WARNING, + AdjustToMuonKinematics = False, + VertexContainerName = BPHY8cf.DerivationName+"DiMuonCandidates", + TrkMasses = [BPHY8cf.GlobalMuonMass, BPHY8cf.GlobalMuonMass], + AddMinChi2ToAnyPVMode = BPHY8cf.AddMinChi2ToAnyPVMode, + PrimaryVertexContainerName = BPHY8cf.PVContName, + MinNTracksInPV = BPHY8cf.minNTracksInPV, + PVTypesToConsider = BPHY8cf.MinChi2ToAnyPVTypes, + DoVertexType = BPHY8cf.doVertexType) + +######################## duplication for debugging only ####################### + +ToolSvc += list(BPHY8_MuMassTools.values()) +for BPHY8_name in list(BPHY8_MuMassTools.keys()): + print(BPHY8_MuMassTools[BPHY8_name]) + pprint(BPHY8_MuMassTools[BPHY8_name].properties()) + +#-------------------------------------------------------------------- +# Augmentation of vertices by vertex track isolation values, +# closest track information and muon isolation values. +# +# First: Set up track selections +# +from InDetTrackSelectionTool.InDetTrackSelectionToolConf \ + import InDet__InDetTrackSelectionTool + +# a) for BVertexTrackIsoTool +BPHY8_IsoTrkSelTools = OrderedDict() +for i in range(len(BPHY8cf.IsoTrackCategoryName)): + BPHY8_name = BPHY8cf.DerivationName+"_iso_"+BPHY8cf.IsoTrackCategoryName[i] + BPHY8_Tool = InDet__InDetTrackSelectionTool( + name = BPHY8_name, + OutputLevel = INFO + ) + if BPHY8cf.IsoTrackCutLevel[i] != "Custom" : + BPHY8_Tool.CutLevel = BPHY8cf.IsoTrackCutLevel[i] + if BPHY8cf.IsoTrackPtCut[i] > -1. : + BPHY8_Tool.minPt = BPHY8cf.IsoTrackPtCut[i] + if BPHY8cf.IsoTrackEtaCut[i] > -1. : + BPHY8_Tool.maxAbsEta = BPHY8cf.IsoTrackEtaCut[i] + if BPHY8cf.IsoTrackPixelHits[i] > -1 : + BPHY8_Tool.minNPixelHits = BPHY8cf.IsoTrackPixelHits[i] + if BPHY8cf.IsoTrackSCTHits[i] > -1 : + BPHY8_Tool.minNSctHits = BPHY8cf.IsoTrackSCTHits[i] + if BPHY8cf.IsoTrackIBLHits[i] > -1 : + BPHY8_Tool.minNInnermostLayerHits = BPHY8cf.IsoTrackIBLHits[i] + if BPHY8cf.IsoTrackbLayerHits[i] > -1 : + BPHY8_Tool.minNNextToInnermostLayerHits = BPHY8cf.IsoTrackbLayerHits[i] + BPHY8_IsoTrkSelTools[BPHY8_name] = BPHY8_Tool + +ToolSvc += list(BPHY8_IsoTrkSelTools.values()) +for BPHY8_name in list(BPHY8_IsoTrkSelTools.keys()): + print(BPHY8_IsoTrkSelTools[BPHY8_name]) + pprint(BPHY8_IsoTrkSelTools[BPHY8_name].properties()) + +# b) for BMuonTrackIsoTool +BPHY8_MuIsoTrkSelTools = OrderedDict() +for i in range(len(BPHY8cf.MuIsoTrackCategoryName)): + BPHY8_name = BPHY8cf.DerivationName+"_muiso_" + \ + BPHY8cf.MuIsoTrackCategoryName[i] + BPHY8_Tool = InDet__InDetTrackSelectionTool( + name = BPHY8_name, + OutputLevel = INFO + ) + if BPHY8cf.MuIsoTrackCutLevel[i] != "Custom" : + BPHY8_Tool.CutLevel = BPHY8cf.MuIsoTrackCutLevel[i] + if BPHY8cf.MuIsoTrackPtCut[i] > -1. : + BPHY8_Tool.minPt = BPHY8cf.MuIsoTrackPtCut[i] + if BPHY8cf.MuIsoTrackEtaCut[i] > -1. : + BPHY8_Tool.maxAbsEta = BPHY8cf.MuIsoTrackEtaCut[i] + if BPHY8cf.MuIsoTrackPixelHits[i] > -1 : + BPHY8_Tool.minNPixelHits = BPHY8cf.MuIsoTrackPixelHits[i] + if BPHY8cf.MuIsoTrackSCTHits[i] > -1 : + BPHY8_Tool.minNSctHits = BPHY8cf.MuIsoTrackSCTHits[i] + if BPHY8cf.MuIsoTrackIBLHits[i] > -1 : + BPHY8_Tool.minNInnermostLayerHits = BPHY8cf.MuIsoTrackIBLHits[i] + if BPHY8cf.MuIsoTrackbLayerHits[i] > -1 : + BPHY8_Tool.minNNextToInnermostLayerHits \ + = BPHY8cf.MuIsoTrackbLayerHits[i] + BPHY8_MuIsoTrkSelTools[BPHY8_name] = BPHY8_Tool + +ToolSvc += list(BPHY8_MuIsoTrkSelTools.values()) +for BPHY8_name in list(BPHY8_MuIsoTrkSelTools.keys()): + print(BPHY8_MuIsoTrkSelTools[BPHY8_name]) + pprint(BPHY8_MuIsoTrkSelTools[BPHY8_name].properties()) + +# c) for ClosestTrackTool +BPHY8_CtTrkSelTools = OrderedDict() +for i in range(len(BPHY8cf.CloseTrackCategoryName)): + BPHY8_name = BPHY8cf.DerivationName+"_ct_"+BPHY8cf.CloseTrackCategoryName[i] + BPHY8_Tool = InDet__InDetTrackSelectionTool( + name = BPHY8_name, + OutputLevel = INFO + ) + if BPHY8cf.CloseTrackCutLevel[i] != "Custom" : + BPHY8_Tool.CutLevel = BPHY8cf.CloseTrackCutLevel[i] + if BPHY8cf.CloseTrackPtCut[i] > -1. : + BPHY8_Tool.minPt = BPHY8cf.CloseTrackPtCut[i] + if BPHY8cf.CloseTrackEtaCut[i] > -1. : + BPHY8_Tool.maxAbsEta = BPHY8cf.CloseTrackEtaCut[i] + if BPHY8cf.CloseTrackPixelHits[i] > -1 : + BPHY8_Tool.minNPixelHits = BPHY8cf.CloseTrackPixelHits[i] + if BPHY8cf.CloseTrackSCTHits[i] > -1 : + BPHY8_Tool.minNSctHits = BPHY8cf.CloseTrackSCTHits[i] + if BPHY8cf.CloseTrackIBLHits[i] > -1 : + BPHY8_Tool.minNInnermostLayerHits = BPHY8cf.CloseTrackIBLHits[i] + if BPHY8cf.CloseTrackbLayerHits[i] > -1 : + BPHY8_Tool.minNNextToInnermostLayerHits \ + = BPHY8cf.CloseTrackbLayerHits[i] + BPHY8_CtTrkSelTools[BPHY8_name] = BPHY8_Tool + +ToolSvc += list(BPHY8_CtTrkSelTools.values()) +for BPHY8_name in list(BPHY8_CtTrkSelTools.keys()): + print(BPHY8_CtTrkSelTools[BPHY8_name]) + pprint(BPHY8_CtTrkSelTools[BPHY8_name].properties()) + +# +# Step 1.5: Set up track vertex assocation tool +# +from TrackVertexAssociationTool.TrackVertexAssociationToolConf \ + import CP__TrackVertexAssociationTool + +BPHY8_TvaTools = OrderedDict() + +# a) for BVertexTrackIsoTool +BPHY8_TvaTools["TrackVtxIsoTva"] = CP__TrackVertexAssociationTool( + name = BPHY8cf.DerivationName+"_VtxIsoTvaTool", + WorkingPoint = BPHY8cf.IsoTvaWorkingPoint, + OutputLevel = WARNING) + +# b) for BMuonTrackIsoTool +BPHY8_TvaTools["TrackMuonIsoTva"] = CP__TrackVertexAssociationTool( + name = BPHY8cf.DerivationName+"_MuonIsoTvaTool", + WorkingPoint = BPHY8cf.MuIsoTvaWorkingPoint, + OutputLevel = WARNING) + +# c) for ClosestTrackTool +BPHY8_TvaTools["TrackVtxCtTva"] = CP__TrackVertexAssociationTool( + name = BPHY8cf.DerivationName+"_VtxCtTvaTool", + WorkingPoint = BPHY8cf.CloseTrackTvaWorkingPoint, + OutputLevel = WARNING) + +# attach to ToolSvc +ToolSvc += list(BPHY8_TvaTools.values()) +for BPHY8_name in list(BPHY8_TvaTools.keys()): + print(BPHY8_TvaTools[BPHY8_name]) + pprint(BPHY8_TvaTools[BPHY8_name].properties()) + +# +# Second: Set up the B candidate vertex container arrays +# +BPHY8cf.VtxContNames = []; +BPHY8cf.RefPVContNames = []; +BPHY8cf.BranchPrefixes = []; +if [i for i in BPHY8cf.doChannels \ + if i in ["Bsmumu", "BJpsiK", "BsJpsiPhi", "BJpsiPi", "Bhh"]]: + BPHY8cf.VtxContNames += [ BPHY8cf.DerivationName+"DiMuonCandidates" ] + BPHY8cf.RefPVContNames += [ BPHY8cf.DerivationName + +"DiMuonRefittedPrimaryVertices" ] + BPHY8cf.BranchPrefixes += [ "DiMuon" ]; +if "BJpsiK" in BPHY8cf.doChannels: + BPHY8cf.VtxContNames += [ BPHY8cf.DerivationName+"BJpsiKCandidates" ] + BPHY8cf.RefPVContNames += [ BPHY8cf.DerivationName + +"BJpsiKRefittedPrimaryVertices" ] + BPHY8cf.BranchPrefixes += [ "BJpsiK" ]; +if "BsJpsiPhi" in BPHY8cf.doChannels: + BPHY8cf.VtxContNames += [ BPHY8cf.DerivationName+"BsJpsiPhiCandidates" ] + BPHY8cf.RefPVContNames += [ BPHY8cf.DerivationName + +"BsJpsiPhiRefittedPrimaryVertices" ] + BPHY8cf.BranchPrefixes += [ "BsJpsiPhi" ]; +if "BJpsiPi" in BPHY8cf.doChannels: + BPHY8cf.VtxContNames += [ BPHY8cf.DerivationName+"BJpsiPiCandidates" ] + BPHY8cf.RefPVContNames += [ BPHY8cf.DerivationName + +"BJpsiPiRefittedPrimaryVertices" ] + BPHY8cf.BranchPrefixes += [ "BJpsiPi" ]; + +# +# Third: Set up the real tools +# +BPHY8_IsoTools = OrderedDict() + +# a) BVertexTrackIsoTool +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf \ + import DerivationFramework__BVertexTrackIsoTool + +BPHY8_IsoTools["TrackVtxIso"] = DerivationFramework__BVertexTrackIsoTool( + name = "BPHY8_VtxIsoTool", + BranchPrefixes = BPHY8cf.BranchPrefixes, + BranchBaseName = "iso", + OutputLevel = INFO, + VertexContainerNames = BPHY8cf.VtxContNames, + RefPVContainerNames = BPHY8cf.RefPVContNames, + TrackParticleContainerName = BPHY8cf.TrkPartContName, + PVContainerName = BPHY8cf.PVContName, + PVTypesToConsider = BPHY8cf.MinChi2ToAnyPVTypes, + TrackSelectionTools = list(BPHY8_IsoTrkSelTools.values()), + TVATool = BPHY8_TvaTools["TrackVtxIsoTva"], + IsolationConeSizes = BPHY8cf.IsolationConeSizes, + IsoTrkImpLogChi2Max = BPHY8cf.IsoTrkImpLogChi2Max, + IsoDoTrkImpLogChi2Cut = BPHY8cf.IsoDoTrkImpLogChi2Cut, + DoVertexType = BPHY8cf.doVertexType, + UseTrackTypes = BPHY8cf.useIsoTrackTypes, + UseOptimizedAlgo = BPHY8cf.IsoUseOptimizedAlgo, + DebugTrackTypes = BPHY8cf.DebugTrackTypes, + DebugTracksInEvents = []) + +# b) BMuonTrackIsoTool +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf \ + import DerivationFramework__BMuonTrackIsoTool + +BPHY8_IsoTools["TrackMuonIso"] = DerivationFramework__BMuonTrackIsoTool( + name = "BPHY8_MuonIsoTool", + BranchPrefixes = BPHY8cf.BranchPrefixes, + BranchBaseName = "muiso", + OutputLevel = INFO, + VertexContainerNames = BPHY8cf.VtxContNames, + RefPVContainerNames = BPHY8cf.RefPVContNames, + TrackParticleContainerName = BPHY8cf.TrkPartContName, + PVContainerName = BPHY8cf.PVContName, + PVTypesToConsider = BPHY8cf.MinChi2ToAnyPVTypes, + MuonContainerName = BPHY8cf.UsedMuonCollection, + TrackSelectionTools = list(BPHY8_MuIsoTrkSelTools.values()), + TVATool = BPHY8_TvaTools["TrackMuonIsoTva"], + IsolationConeSizes = BPHY8cf.MuIsolationConeSizes, + IsoTrkImpLogChi2Max = BPHY8cf.MuIsoTrkImpLogChi2Max, + IsoDoTrkImpLogChi2Cut = BPHY8cf.MuIsoDoTrkImpLogChi2Cut, + DoVertexType = BPHY8cf.doVertexType, + UseTrackTypes = BPHY8cf.useMuIsoTrackTypes, + DebugTrackTypes = BPHY8cf.DebugTrackTypes, + DebugTracksInEvents = []) + +# c) BVertexClosestTrackTool +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf \ + import DerivationFramework__BVertexClosestTrackTool + +BPHY8_IsoTools["TrackVtxCt"] = DerivationFramework__BVertexClosestTrackTool( + name = "BPHY8_VtxClosestTrkTool", + BranchPrefixes = BPHY8cf.BranchPrefixes, + BranchBaseName = "ct", + OutputLevel = INFO, + VertexContainerNames = BPHY8cf.VtxContNames, + RefPVContainerNames = BPHY8cf.RefPVContNames, + TrackParticleContainerName = BPHY8cf.TrkPartContName, + PVContainerName = BPHY8cf.PVContName, + TrackSelectionTools = list(BPHY8_CtTrkSelTools.values()), + TVATool = BPHY8_TvaTools["TrackVtxCtTva"], + PVTypesToConsider = BPHY8cf.MinChi2ToAnyPVTypes, + DoVertexType = BPHY8cf.doVertexType, + UseTrackTypes = BPHY8cf.useCloseTrackTypes, + CloseTrackChi2SetName = BPHY8cf.CloseTrackChi2SetName, + CloseTrackCorrChi2 = BPHY8cf.CloseTrackCorrChi2, + CloseTrackMinDCAin3D = BPHY8cf.CloseTrackMinDCAin3D, + CloseTrackMaxLogChi2 = BPHY8cf.CloseTrackMaxLogChi2, + NCloseTrackMaxLogChi2 = BPHY8cf.NCloseTrackMaxLogChi2, + DebugTrackTypes = BPHY8cf.DebugTrackTypes, + DebugTracksInEvents = []) + +# +# Fourth: Add track-to-vertex map debugging +# +BPHY8_TtvmTools = OrderedDict(); + +# a) configure BPhysTrackVertexMapTools +# Configure only if not 0 events requested +if BPHY8cf.DebugTrkToVtxMaxEvents != 0 : + for BPHY8_prefix, BPHY8_SVcont, BPHY8_refPVcont in \ + zip(BPHY8cf.BranchPrefixes, BPHY8cf.VtxContNames, + BPHY8cf.RefPVContNames): + BPHY8_hypos = BPHY8_prefix + if BPHY8_hypos == "DiMuon": + BPHY8_hypos += "|Bsmumu|Jpsimumu" + BPHY8_TtvmTools[BPHY8_prefix] = CfgMgr.xAOD__BPhysTrackVertexMapTool( + "BPHY8_ttvm_"+BPHY8_prefix, + OutputLevel = INFO, + VertexContainerName = BPHY8_SVcont, + RefPVContainerName = BPHY8_refPVcont, + PVContainerName = BPHY8cf.PVContName, + TrackParticleContainerName = BPHY8cf.TrkPartContName, + DebugTrkToVtxMaxEvents = BPHY8cf.DebugTrkToVtxMaxEvents, + DumpPrefix = "TTV2> ", + HypoName = BPHY8_hypos ) + + ToolSvc += list(BPHY8_TtvmTools.values()) + for BPHY8_name in list(BPHY8_TtvmTools.keys()): + print(BPHY8_TtvmTools[BPHY8_name]) + pprint(BPHY8_TtvmTools[BPHY8_name].properties()) + +# b) wrap into logger algorithm +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf \ + import DerivationFramework__BTrackVertexMapLogger + +# Configure only if not 0 events requested +if BPHY8cf.DebugTrkToVtxMaxEvents != 0 : + BPHY8_IsoTools["TvmLogger"] = DerivationFramework__BTrackVertexMapLogger( + name = "BPHY8_TrackVertexMapLogger", + OutputLevel = INFO, + TrackVertexMapTools = list(BPHY8_TtvmTools.values()), + Enable = True) + +# +# Fifth: Attach to ToolSvc +# +ToolSvc += list(BPHY8_IsoTools.values()) +for BPHY8_name in list(BPHY8_IsoTools.keys()): + print(BPHY8_IsoTools[BPHY8_name]) + pprint(BPHY8_IsoTools[BPHY8_name].properties()) + +#-------------------------------------------------------------------- +# Record the original counts for primary vertices and tracks +#-------------------------------------------------------------------- +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf \ + import DerivationFramework__AugOriginalCounts +BPHY8_AugOriginalCounts = DerivationFramework__AugOriginalCounts( + name = "BPHY8_AugOriginalCounts", + VertexContainer = BPHY8cf.PVContName, + TrackContainer = BPHY8cf.TrkPartContName, + AddPVCountsByType = True, + AddNTracksToPVs = True, + AddSqrtPt2SumToPVs = True) + +ToolSvc += BPHY8_AugOriginalCounts +pprint(BPHY8_AugOriginalCounts.properties()) + +#-------------------------------------------------------------------- +# Setup the vertex selection and augmentation tool(s). These tools decorate +# the vertices with variables that depend on the vertex mass hypothesis, +# e.g. invariant mass, proper decay time, etc. +# Property HypothesisName is used as a prefix for these decorations. +# They also perform tighter selection, flagging the vertecis that passed. +# The flag is a Char_t branch named "passed_"+HypothesisName. It is used +# later by the "SelectEvent" and "Thin_vtxTrk" tools to determine which +# events and candidates should be kept in the output stream. +# Multiple instances of the Select_* tools can be used on a single input +# collection as long as they use different "HypothesisName" flags. + +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf \ + import DerivationFramework__Select_Bmumu + +BPHY8_SelectTools = OrderedDict() +# a) for Bsmumu +if "Bsmumu" in BPHY8cf.doChannels: + # augment and select B(s)->mumu candidates + BPHY8_SelectTools["Bsmumu"] = DerivationFramework__Select_Bmumu( + name = "BPHY8_Select_Bsmumu", + HypothesisName = "Bsmumu", + InputVtxContainerName = BPHY8cf.DerivationName+"DiMuonCandidates", + TrkMasses = [BPHY8cf.GlobalMuonMass, BPHY8cf.GlobalMuonMass], + VtxMassHypo = BPHY8cf.GlobalBsMass, + MassMin = BPHY8cf.GlobalBMassLowerCut, + MassMax = BPHY8cf.GlobalBMassUpperCut, + Chi2Max = BPHY8cf.Chi2Cut2Prong, + DoVertexType = BPHY8cf.doVertexType, + Do3d = BPHY8cf.do3dProperTime, + BlindMassMin = BPHY8cf.GlobalBlindLowerCut, + BlindMassMax = BPHY8cf.GlobalBlindUpperCut, + DoBlinding = BPHY8cf.doBmumuBlinding, + DoCutBlinded = BPHY8cf.doCutBlinded, + BlindOnlyAllMuonsTight = BPHY8cf.blindOnlyAllMuonsTight, + UseMuCalcMass = BPHY8cf.useMuCalcMass, + OutputLevel = WARNING) +# b) for BJpsiK and BsJpsiPhi retain the Jpsi +if [i for i in BPHY8cf.doChannels if i in ["BJpsiK", "BsJpsiPhi"]]: + BPHY8_SelectTools["Jpsimumu"] = DerivationFramework__Select_Bmumu( + name = "BPHY8_Select_Jpsimumu", + HypothesisName = "Jpsimumu", + InputVtxContainerName = BPHY8cf.DerivationName+"DiMuonCandidates", + TrkMasses = [BPHY8cf.GlobalMuonMass, BPHY8cf.GlobalMuonMass], + VtxMassHypo = BPHY8cf.GlobalJpsiMass, + MassMin = BPHY8cf.GlobalJpsiMassLowerCut, + MassMax = BPHY8cf.GlobalJpsiMassUpperCut, + Chi2Max = BPHY8cf.Chi2Cut2Prong, + DoVertexType = BPHY8cf.doVertexType, + Do3d = BPHY8cf.do3dProperTime, + UseMuCalcMass = BPHY8cf.useMuCalcMass, + OutputLevel = WARNING) +# c) for BJpsiK +if "BJpsiK" in BPHY8cf.doChannels: + # augment and select B+/- ->JpsiK+/- candidates + BPHY8_SelectTools["BJpsiK"] = DerivationFramework__Select_Bmumu( + name = "BPHY8_Select_BJpsiK", + HypothesisName = "BJpsiK", + InputVtxContainerName = BPHY8cf.DerivationName+"BJpsiKCandidates", + TrkMasses = [BPHY8cf.GlobalMuonMass, BPHY8cf.GlobalMuonMass, BPHY8cf.GlobalKaonMass], + VtxMassHypo = BPHY8cf.GlobalBplusMass, + MassMin = BPHY8cf.GlobalBMassLowerCut, + MassMax = BPHY8cf.GlobalBMassUpperCut, + Chi2Max = BPHY8cf.Chi2Cut3Prong, + DoVertexType = BPHY8cf.doVertexType, + Do3d = BPHY8cf.do3dProperTime, + UseMuCalcMass = BPHY8cf.useMuCalcMass, + SubDecVtxContNames = [BPHY8cf.DerivationName+"DiMuonCandidates"], + SubDecVtxHypoCondNames = ["Jpsimumu"], + SubDecVtxHypoFlagNames = ["JpsimumuSubDecay"], + OutputLevel = WARNING) +# d) for BsJpsiPhi +if "BsJpsiPhi" in BPHY8cf.doChannels: + # augment and select Bs ->JpsiPhi candidates + BPHY8_SelectTools["BsJpsiPhi"] = DerivationFramework__Select_Bmumu( + name = "BPHY8_Select_BsJpsiPhi", + HypothesisName = "BsJpsiPhi", + InputVtxContainerName = BPHY8cf.DerivationName+"BsJpsiPhiCandidates", + TrkMasses = [BPHY8cf.GlobalMuonMass, BPHY8cf.GlobalMuonMass, BPHY8cf.GlobalKaonMass, BPHY8cf.GlobalKaonMass], + VtxMassHypo = BPHY8cf.GlobalBsMass, + MassMin = BPHY8cf.GlobalBMassLowerCut, + MassMax = BPHY8cf.GlobalBMassUpperCut, + Chi2Max = BPHY8cf.Chi2Cut4Prong, + DoVertexType = BPHY8cf.doVertexType, + Do3d = BPHY8cf.do3dProperTime, + UseMuCalcMass = BPHY8cf.useMuCalcMass, + SubDecVtxContNames = [BPHY8cf.DerivationName+"DiMuonCandidates"], + SubDecVtxHypoCondNames = ["Jpsimumu"], + SubDecVtxHypoFlagNames = ["JpsimumuSubDecay"], + OutputLevel = WARNING) +# e) for BJpsiPi +if "BJpsiPi" in BPHY8cf.doChannels: + # augment and select B+/- ->JpsiPi+/- candidates + BPHY8_SelectTools["BJpsiPi"] = DerivationFramework__Select_Bmumu( + name = "BPHY8_Select_BJpsiPi", + HypothesisName = "BJpsiPi", + InputVtxContainerName = BPHY8cf.DerivationName+"BJpsiPiCandidates", + TrkMasses = [BPHY8cf.GlobalMuonMass, BPHY8cf.GlobalMuonMass, BPHY8cf.GlobalPionMass], + VtxMassHypo = BPHY8cf.GlobalBplusMass, + MassMin = BPHY8cf.GlobalBMassLowerCut, + MassMax = BPHY8cf.GlobalBMassUpperCut, + Chi2Max = BPHY8cf.Chi2Cut3Prong, + DoVertexType = BPHY8cf.doVertexType, + Do3d = BPHY8cf.do3dProperTime, + UseMuCalcMass = BPHY8cf.useMuCalcMass, + SubDecVtxContNames = [BPHY8cf.DerivationName+"DiMuonCandidates"], + SubDecVtxHypoCondNames = ["Jpsimumu"], + SubDecVtxHypoFlagNames = ["JpsimumuSubDecay"], + OutputLevel = WARNING) +# f) for Bhh +if "Bhh" in BPHY8cf.doChannels: + # augment and select B->hh candidates + BPHY8_SelectTools["Bhh"] = DerivationFramework__Select_Bmumu( + name = "BPHY8_Select_Bhh", + HypothesisName = "Bhh", + InputVtxContainerName = BPHY8cf.DerivationName+"DiMuonCandidates", + TrkMasses = [BPHY8cf.GlobalMuonMass, BPHY8cf.GlobalMuonMass], + VtxMassHypo = BPHY8cf.GlobalBsMass, + MassMin = BPHY8cf.GlobalBMassLowerCut, + MassMax = BPHY8cf.GlobalBMassUpperCut, + Chi2Max = BPHY8cf.Chi2Cut2Prong, + DoVertexType = BPHY8cf.doVertexType, + Do3d = BPHY8cf.do3dProperTime, + UseMuCalcMass = BPHY8cf.useMuCalcMass, + OutputLevel = WARNING) + +ToolSvc += list(BPHY8_SelectTools.values()) +for BPHY8_name in list(BPHY8_SelectTools.keys()): + print(BPHY8_SelectTools[BPHY8_name]) + pprint(BPHY8_SelectTools[BPHY8_name].properties()) + +#-------------------------------------------------------------------- +# Setup the vertex variable blinding tools. +# These tools are only used by the Bsmumu channel in case +# blinding is enabled. + +BPHY8_BlindingTools = OrderedDict() +BPHY8_BlinderTools = OrderedDict() + +if BPHY8cf.doBmumuBlinding and not BPHY8cf.doCutBlinded: + # BlindingTools + from BPhysTools.BPhysToolsConf import xAOD__BPhysBlindingTool + # 1) for Bsmumu + if "Bsmumu" in BPHY8cf.doChannels : + # setup blinding tool + BPHY8_BlindingTools["Bsmumu"] = xAOD__BPhysBlindingTool( + name = "BPHY8_BlindingTool_Bsmumu", + VertexContainerName = BPHY8cf.DerivationName+"DiMuonCandidates", + VarToBlindNames = BPHY8cf.BlindedVars, + BlindingFlag = BPHY8cf.BlindingFlag, + NegativeSigns = [True, True], + BlindingKey = BPHY8cf.BlindingKey, + OutputLevel = INFO) + + ToolSvc += list(BPHY8_BlindingTools.values()) + for BPHY8_name in list(BPHY8_BlindingTools.keys()): + print(BPHY8_BlindingTools[BPHY8_name]) + pprint(BPHY8_BlindingTools[BPHY8_name].properties()) + + # Blinders + from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf \ + import DerivationFramework__BPhysVarBlinder + # a2) for Bsmumu + if "Bsmumu" in BPHY8cf.doChannels : + # blind mass values for B(s)->mumu candidates + BPHY8_BlinderTools["Bsmumu"] = DerivationFramework__BPhysVarBlinder( + name = "BPHY8_Blinder_Bsmumu", + BlindingTool = BPHY8_BlindingTools["Bsmumu"], + EnableBlinding = True, + OutputLevel = INFO) + + ToolSvc += list(BPHY8_BlinderTools.values()) + for BPHY8_name in list(BPHY8_BlinderTools.keys()): + print(BPHY8_BlinderTools[BPHY8_name]) + pprint(BPHY8_BlinderTools[BPHY8_name].properties()) + +#==================================================================== +# Skimming tool to select only events with the correct vertices +#==================================================================== +#-------------------------------------------------------------------- +# Select the event. We only want to keep events that contain certain +# vertices which passed certain selection. This is specified by the +# "SelectionExpression" property, which contains the expression in the +# following format: +# +# "ContainerName.passed_HypoName > count" +# +# where "ContainerName" is output container form some Reco_* tool, +# "HypoName" is the hypothesis name setup in some "Select_*" tool and +# "count" is the number of candidates passing the selection you want to keep. +# + +# Build expression depending on "select tools" used: +# If any of them marked any candidate passed, add expression for it. +BPHY8_expressions = [] +for BPHY8_tool in list(BPHY8_SelectTools.values()): + BPHY8_passName = BPHY8_tool.HypothesisName + if BPHY8_tool.HypothesisName == "Jpsimumu" and BPHY8cf.thinLevel > 0: + BPHY8_passName = "JpsimumuSubDecay" + BPHY8_expressions += [ "count(%s.passed_%s) > 0" % \ + (BPHY8_tool.InputVtxContainerName, + BPHY8_passName) ] +BPHY8cf.SelExpression = " || ".join(BPHY8_expressions) + +from DerivationFrameworkTools.DerivationFrameworkToolsConf \ + import DerivationFramework__xAODStringSkimmingTool +BPHY8_SkimmingTool = DerivationFramework__xAODStringSkimmingTool( + name = "BPHY8_SelectEvent", + expression = BPHY8cf.SelExpression) + +ToolSvc += BPHY8_SkimmingTool +print(BPHY8_SkimmingTool) +pprint(BPHY8_SkimmingTool.properties()) + +# Check for global ToolSvc: +print(">>> Checking ToolSvc tools: <<<") +for BPHY8_i in ToolSvc: + print(BPHY8_i) +print(">>> End of ToolSvc tools. <<<") + +#==================================================================== +# SET UP STREAM +#==================================================================== +BPHY8_streamName = derivationFlags.WriteDAOD_BPHY8Stream.StreamName +BPHY8_fileName = buildFileName( derivationFlags.WriteDAOD_BPHY8Stream ) +BPHY8Stream = MSMgr.NewPoolRootStream(BPHY8_streamName, BPHY8_fileName ) +BPHY8Stream.AcceptAlgs(["BPHY8Kernel"]) +# +# Special lines for thinning +# Thinning service name must match the one passed to the thinning tools +## +## NOTE: We use the ThinningHelper which already instantiates a ThinningSvc +## +## from AthenaServices.Configurables import ThinningSvc, createThinningSvc +## BPHY8_augStream = MSMgr.GetStream( BPHY8_streamName ) +## BPHY8_evtStream = BPHY8_augStream.GetEventStream() +## svcMgr += createThinningSvc(svcName=BPHY8cf.DerivationName+"ThinningSvc", +## outStreams=[BPHY8_evtStream] ) + +# Additional metadata output +BPHY8Stream.AddMetaDataItem([ "xAOD::FileMetaData#%s*" % + BPHY8cf.DerivationName, + "xAOD::FileMetaDataAuxInfo#%s*Aux." % + BPHY8cf.DerivationName] ) + +#==================================================================== +# Thinning Helper and various thinning tools +#==================================================================== +#-------------------------------------------------------------------- +# Setup the thinning helper, only tool able to perform thinning +# of trigger navigation information. +# +from DerivationFrameworkCore.ThinningHelper import ThinningHelper +BPHY8ThinningHelper = ThinningHelper( BPHY8cf.DerivationName+"ThinningHelper" ) + +if BPHY8cf.doTrigNavThinning and BPHY8cf.doTriggerInfo: + BPHY8ThinningHelper.TriggerChains = '|'.join(BPHY8cf.TrigNavThinList) + +BPHY8ThinningHelper.AppendToStream( BPHY8Stream ) + +#-------------------------------------------------------------------- +# Thinning tools +BPHY8ThinningTools = [] + +# +# MC Truth Thinning +# +if BPHY8cf.isSimulation: + # + # PDG-ID list of truth decay particles whose decay chains are to be recorded + # B mesons + BPHY8cf.TruthDecayParents = [511, 521, 10511, 10521, 513, 523, 10513, 10523, 20513, 20523, 515, 525, 531, 10531, 533, 10533, 20533, 535, 541, 10541, 543, 10543, 20543, 545] + # b bbar mesons + BPHY8cf.TruthDecayParents += [551,10551,100551,110551,200551,210551,553,10553,20553,30553,100553,110553,120553,130553,200553,210553,220553,300553,9000553,9010553,555,10555,20555,100555,110555,120555,200555,557,100557] + BPHY8cf.TruthDecayParents += [5122,5112,5212,5222,5114,5214,5224,5132,5232,5312,5322,5314,5324,5332,5334,5142,5242,5412,5422,5414,5424,5342,5432,5434,5442,5444,5512,5522,5514,5524,5532,5534,5542,5544,5554] + # Charmed mesons + ## BPHY8cf.TruthDecayParents += [411, 421, 10411, 10421, 413, 423, 10413, 10423, 20413, 20423, 415, 425, 431, 10431, 433, 10433, 20433, 435] + # c cbar mesons + ## BPHY8cf.TruthDecayParents += [441, 10441, 100441, 443, 10443, 20443, 100443, 30443, 9000443, 9010443, 9020443, 445, 100445] + # charmed baryons + ## BPHY8cf.TruthDecayParents += [4122, 4222, 4212, 4112, 4224, 4214, 4114, 4232, 4132, 4322, 4312, 4324, 4314, 4332, 4334, 4412, 4422, 4414, 4424, 4432, 4434, 4444] + + # compose ParticleSelectionString + BPHY8_ParticleSelConds = [] + for BPHY8_pdgid in BPHY8cf.TruthDecayParents: + BPHY8_ParticleSelConds.append("abs(TruthParticles.pdgId) == %d" % + BPHY8_pdgid) + BPHY8_ParticleSelection = " || ".join(BPHY8_ParticleSelConds) + + # Only save truth information directly associated with B decays. + # We'll skip the GEANT particles (barcode >= 200000). + from DerivationFrameworkMCTruth.DerivationFrameworkMCTruthConf \ + import DerivationFramework__GenericTruthThinning + BPHY8TruthThinTool = DerivationFramework__GenericTruthThinning( + name = BPHY8cf.DerivationName+"TruthThinTool", + ParticleSelectionString = BPHY8_ParticleSelection, + PreserveGeneratorDescendants = True, + PreserveDescendants = False, + PreserveAncestors = False) + + ToolSvc += BPHY8TruthThinTool + BPHY8ThinningTools.append(BPHY8TruthThinTool) + print(BPHY8TruthThinTool) + pprint(BPHY8TruthThinTool.properties()) + +# +# Vertex thinning +# +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf \ + import DerivationFramework__Thin_vtxTrk + +# Build list of input containers and passed flags depending on +# "select tools" used. +import re +BPHY8_vtxContainers = [] +BPHY8_refPVContainers = [] +BPHY8_passedFlags = [] +for BPHY8_tool in list(BPHY8_SelectTools.values()): + BPHY8_vtxContainers.append(BPHY8_tool.InputVtxContainerName) + BPHY8_refPVContainers.append(re.sub('Candidates$', + 'RefittedPrimaryVertices', + BPHY8_tool.InputVtxContainerName)) + BPHY8_passName = BPHY8_tool.HypothesisName + if BPHY8_tool.HypothesisName == "Jpsimumu" and BPHY8cf.thinLevel > 0: + BPHY8_passName = "JpsimumuSubDecay" + BPHY8_passedFlags.append("passed_%s" % BPHY8_passName) +# +# Use the general Thin_vtxTrk tool to thin the vertex containers only. +# +if BPHY8cf.thinLevel < 2: + BPHY8Thin_vtxTrk = DerivationFramework__Thin_vtxTrk( + name = BPHY8cf.DerivationName+"Thin_vtxTrk", + TrackParticleContainerName = BPHY8cf.TrkPartContName, + VertexContainerNames = BPHY8_vtxContainers, + PassFlags = BPHY8_passedFlags, + ThinTracks = False) + ToolSvc += BPHY8Thin_vtxTrk + BPHY8ThinningTools.append(BPHY8Thin_vtxTrk) + print(BPHY8Thin_vtxTrk) + pprint(BPHY8Thin_vtxTrk.properties()) + +# +# Bmumu PV, muon collections and ID track thinnning +# +if BPHY8cf.thinLevel > 1: + from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf \ + import DerivationFramework__BmumuThinningTool + BPHY8BmumuThinningTool = DerivationFramework__BmumuThinningTool( + name = BPHY8cf.DerivationName+"BmumuThinningTool", + TrackParticleContainerName = BPHY8cf.TrkPartContName, + VertexContainerNames = BPHY8_vtxContainers, + VertexPassFlags = BPHY8_passedFlags, + RefPVContainerNames = BPHY8_refPVContainers, + # RefPVContainerNames = BPHY8cf.RefPVContNames, + AlignPassToVertexList = True, + PVContainerName = BPHY8cf.PVContName, + MuonContainerName = BPHY8cf.MuonCollection, + CalibMuonContainerName = BPHY8cf.CalMuonCollection, + MatchCalibratedMuons = (BPHY8cf.useCalibratedMuons > 2), + MarkMatchedMuons = (BPHY8cf.useCalibratedMuons > 2), + MarkMatchedCalMuons = (BPHY8cf.useCalibratedMuons > 1 + and BPHY8cf.useCalibratedMuons < 3), + SyncMatchedMuonsBothWays = True, + AllowFastMuonMaskSync = True, + KeepTracksForMuons = True, + KeepTracksForCalMuons = True, + KeepMuonsForTracks = True, + KeepCalMuonsForTracks = True, + KeepCloseTracks = True, + ThinMuons = (BPHY8cf.thinLevel < 5), + CloseTrackBranchPrefixes = BPHY8cf.BranchPrefixes, + CloseTrackBranchBaseName = BPHY8_IsoTools["TrackVtxCt"].BranchBaseName, + ThinPVs = (BPHY8cf.thinLevel == 2), + ThinRefittedPVs = (BPHY8cf.thinLevel == 2), + ThinTracks = (BPHY8cf.thinLevel < 4), + KeepTracksForSelectedPVs = False, + OutputLevel = INFO) + ToolSvc += BPHY8BmumuThinningTool + BPHY8ThinningTools.append(BPHY8BmumuThinningTool) + print(BPHY8BmumuThinningTool) + pprint(BPHY8BmumuThinningTool.properties()) + +#==================================================================== +# CREATE THE DERIVATION KERNEL ALGORITHM AND PASS THE ABOVE TOOLS +#==================================================================== +# IMPORTANT bit. Don't forget to pass the tools to the DerivationKernel! +# If you don't do that, they will not be be executed! +# The name of the kernel (BPHY8Kernel in this case) must be unique to +# this derivation. +# Make use of a AthSequence in order to run the muon calibrations +# beforehand if requested. +from DerivationFrameworkCore.DerivationFrameworkCoreConf import DerivationFramework__DerivationKernel +BPHY8_seq = CfgMgr.AthSequencer("BPHY8Sequence") +DerivationFrameworkJob += BPHY8_seq +BPHY8_seq += list(BPHY8_CalibrationAlgs.values()) + +# required for Soft B tagging +if BPHY8cf.doAddSoftBVertices: + from DerivationFrameworkJetEtMiss.ExtendedJetCommon import replaceAODReducedJets + OutputJets["BPHY8"] = [] + reducedJetList = ["AntiKt4PV0TrackJets"] + replaceAODReducedJets(reducedJetList, BPHY8_seq, "BPHY8") + + from SoftBVrtClusterTool.SoftBVrtConfig import addSoftBVrt + addSoftBVrt(BPHY8_seq,'Loose') + addSoftBVrt(BPHY8_seq,'Medium') + addSoftBVrt(BPHY8_seq,'Tight') + +BPHY8_seq += CfgMgr.DerivationFramework__DerivationKernel( + "BPHY8Kernel", + OutputLevel = INFO, + AugmentationTools = [ BPHY8_MetaDataTool, BPHY8_AugOriginalCounts, + BPHY8_MuonExtrapTool] \ + + list(BPHY8_RecoTools.values()) + list(BPHY8_MuMassTools.values()) \ + + list(BPHY8_IsoTools.values()) \ + + list(BPHY8_SelectTools.values()) \ + + list(BPHY8_BlinderTools.values()), + SkimmingTools = [BPHY8_SkimmingTool], + ThinningTools = BPHY8ThinningTools + ) + +#==================================================================== +# Slimming +#==================================================================== +from DerivationFrameworkCore.SlimmingHelper import SlimmingHelper +BPHY8SlimmingHelper = SlimmingHelper("BPHY8SlimmingHelper") +BPHY8_AllVariables = [] +BPHY8_StaticContent = [] +BPHY8_SmartCollections = [] +BPHY8_ExtraVariables = [] + +# Needed for trigger objects +BPHY8SlimmingHelper.IncludeMuonTriggerContent = BPHY8cf.doTriggerInfo +BPHY8SlimmingHelper.IncludeBPhysTriggerContent = BPHY8cf.doTriggerInfo + +# primary vertices +BPHY8_SmartCollections += [BPHY8cf.PVContName] +# +# 2018-03-12: These extra variables are not used by NtupleMaker +# which are removed by using SmartCollection instead of +# AllVariables. +## BPHY8_ExtraVariables += ["%s.covariance" % BPHY8cf.PVContName +## + ".chiSquared.numberDoF.sumPt2" +## + ".trackParticleLinks.trackWeights.neutralWeights"] +# +# 2018-05-04: The covariance matrix is occasionally used by the NtupleMaker +# 2019-11-15: adding original number of tracks and sqrt(sum(pt^2)) +BPHY8_ExtraVariables += ["%s.covariance" % BPHY8cf.PVContName + +".OrigNTracks.OrigSqrtPt2Sum"] + +for BPHY8_reco in BPHY8_recoList: + BPHY8_StaticContent \ + += ["xAOD::VertexContainer#BPHY8"+BPHY8_reco+"RefittedPrimaryVertices"] + BPHY8_StaticContent \ + += ["xAOD::VertexAuxContainer#BPHY8"+BPHY8_reco+"RefittedPrimaryVerticesAux."] + +# combined / extrapolated muon track particles +# (note: for tagged muons there is no extra TrackParticle collection since +# the ID tracks are stored in InDetTrackParticles collection) +BPHY8_AllVariables += ["CombinedMuonTrackParticles"] + +BPHY8_AllVariables += ["ExtrapolatedMuonTrackParticles"] +# TODO: copy smart slimming for calibrated muons. +if BPHY8cf.useCalibratedMuons > 1: + BPHY8_AllVariables += [BPHY8cf.CalMuonCollection] + BPHY8SlimmingHelper.AppendToDictionary = { + '%s' % BPHY8cf.CalMuonCollection : 'xAOD::MuonContainer', + '%sAux' % BPHY8cf.CalMuonCollection : 'xAOD::ShallowAuxContainer' } + +# muon container +## AllVariables += [BPHY8cf.MuonCollection] +# smart collection adds info needed for CP tools +BPHY8_SmartCollections += [BPHY8cf.MuonCollection] +BPHY8_ExtraVariables += ["%s.etcone30.etcone40" % BPHY8cf.MuonCollection + +".momentumBalanceSignificance" + +".scatteringCurvatureSignificance" + +".scatteringNeighbourSignificance" + +".msInnerMatchDOF.msInnerMatchChi2" + +".msOuterMatchDOF.msOuterMatchChi2" + +".EnergyLoss.ParamEnergyLoss.MeasEnergyLoss" + +".ET_Core" ] + +# ID track particles +BPHY8_SmartCollections += [BPHY8cf.TrkPartContName] +BPHY8_ExtraVariables += ["%s.vx.vy" % BPHY8cf.TrkPartContName] + +# decay candidates +# we have to disable vxTrackAtVertex branch since it is not xAOD compatible +# also remove not needed isolation and close-track branches from DxAOD +BPHY8_DoVertexTypeStr = ['PV_MIN_Z0_BA'] +BPHY8_IsoBranches = ['iso', 'iso_Ntracks'] +BPHY8_MuIsoBranches = ['muiso', 'muiso_Ntracks', 'muiso+muLink'] +BPHY8_CtBranches = ['ct_DCA', 'ct_DCAError', 'ct_ZCA', 'ct_ZCAError', + 'ct_NTracksChi2','ct_CloseTrack+Link'] +from DerivationFrameworkBPhys.BPhysPyHelpers import BPhysFilterBranches +for BPHY8_name in list(BPHY8_RecoTools.keys()): + BPHY8_StaticContent += ["xAOD::VertexContainer#%s" % + BPHY8_RecoTools[BPHY8_name].OutputVtxContainerName] + BPHY8_StaticContent += ["xAOD::VertexAuxContainer#%sAux." % + BPHY8_RecoTools[BPHY8_name].OutputVtxContainerName] + BPHY8_str = "xAOD::VertexAuxContainer#%sAux" % \ + BPHY8_RecoTools[BPHY8_name].OutputVtxContainerName + BPHY8_str += ".-vxTrackAtVertex" + # isolation branches + BPHY8_cones = ["%02d_LC%02dd%01d" % \ + (int(cs*10), int(BPHY8cf.IsoTrkImpLogChi2Max[i]*10), + BPHY8cf.IsoDoTrkImpLogChi2Cut[i]) + for i,cs in enumerate(BPHY8cf.IsolationConeSizes)] + BPHY8_str += BPhysFilterBranches(BPHY8_name, + BPHY8_IsoBranches, + BPHY8cf.IsoIncludes, + BPHY8_DoVertexTypeStr, + BPHY8cf.IsoTrackCategoryName, + BPHY8cf.useIsoTrackTypes, + BPHY8_cones, + False) + # muon isolation branches + BPHY8_cones = ["%02d_LC%02dd%01d" % \ + (int(cs*10), int(BPHY8cf.MuIsoTrkImpLogChi2Max[i]*10), + BPHY8cf.MuIsoDoTrkImpLogChi2Cut[i]) + for i,cs in enumerate(BPHY8cf.MuIsolationConeSizes)] + BPHY8_str += BPhysFilterBranches(BPHY8_name, + BPHY8_MuIsoBranches, + BPHY8cf.MuIsoIncludes, + BPHY8_DoVertexTypeStr, + BPHY8cf.MuIsoTrackCategoryName, + BPHY8cf.useMuIsoTrackTypes, + BPHY8_cones, + False) + # close track branches + BPHY8_str += BPhysFilterBranches(BPHY8_name, + BPHY8_CtBranches, + BPHY8cf.CloseTrackIncludes, + BPHY8_DoVertexTypeStr, + BPHY8cf.CloseTrackCategoryName, + BPHY8cf.useCloseTrackTypes, + BPHY8cf.CloseTrackChi2SetName, + True) + print(("Branches to be removed: %s" % BPHY8_str)) + BPHY8_StaticContent += [ BPHY8_str ] + +# Truth information for MC only +if BPHY8cf.isSimulation: + BPHY8_AllVariables += ["TruthEvents","TruthParticles","TruthVertices"] + +# required for Soft B Tagging +if BPHY8cf.doAddSoftBVertices: + excludedVertexAuxData = "-vxTrackAtVertex.-MvfFitInfo.-isInitialized.-VTAV" + BPHY8_StaticContent += ["xAOD::VertexContainer#" + +"SoftBVrtClusterTool_Tight_Vertices"] + BPHY8_StaticContent += ["xAOD::VertexAuxContainer#" + +"SoftBVrtClusterTool_Tight_VerticesAux." + + excludedVertexAuxData] + BPHY8_StaticContent += ["xAOD::VertexContainer#" + +"SoftBVrtClusterTool_Medium_Vertices"] + BPHY8_StaticContent += ["xAOD::VertexAuxContainer#" + +"SoftBVrtClusterTool_Medium_VerticesAux." + + excludedVertexAuxData] + BPHY8_StaticContent += ["xAOD::VertexContainer#" + +"SoftBVrtClusterTool_Loose_Vertices"] + BPHY8_StaticContent += ["xAOD::VertexAuxContainer#" + +"SoftBVrtClusterTool_Loose_VerticesAux." + + excludedVertexAuxData] + +BPHY8SlimmingHelper.AllVariables = BPHY8_AllVariables +BPHY8SlimmingHelper.SmartCollections = BPHY8_SmartCollections +BPHY8SlimmingHelper.StaticContent = BPHY8_StaticContent +BPHY8SlimmingHelper.ExtraVariables = BPHY8_ExtraVariables +BPHY8SlimmingHelper.AppendContentToStream(BPHY8Stream) + +#==================================================================== +# END OF BPHY8.py +#==================================================================== diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/BPHY9.py b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/BPHY9.py new file mode 100644 index 00000000000..a617264e2ce --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/BPHY9.py @@ -0,0 +1,398 @@ +#******************************************************************** +# BPHY9.py +# reductionConf flag BPHY9 in Reco_tf.py +#******************************************************************** + +from DerivationFrameworkCore.DerivationFrameworkMaster import * +from DerivationFrameworkInDet.InDetCommon import * +from DerivationFrameworkJetEtMiss.JetCommon import * +from DerivationFrameworkEGamma.EGammaCommon import * +from DerivationFrameworkMuons.MuonsCommon import * +from DerivationFrameworkHiggs.TruthCategories import * +from AthenaCommon.GlobalFlags import globalflags + +is_MC = globalflags.DataSource()=='geant4' + +print('is_MC = ',is_MC) + + +if is_MC: + from DerivationFrameworkMCTruth.MCTruthCommon import addStandardTruthContents + addStandardTruthContents() + from DerivationFrameworkMCTruth.HFHadronsCommon import * + +#==================================================================== +# SET UP STREAM +#==================================================================== +streamName = derivationFlags.WriteDAOD_BPHY9Stream.StreamName +fileName = buildFileName( derivationFlags.WriteDAOD_BPHY9Stream ) +BPHY9Stream = MSMgr.NewPoolRootStream( streamName, fileName ) +BPHY9Stream.AcceptAlgs(['BPHY9Kernel']) + +#==================================================================== +# AUGMENTATION TOOLS +#==================================================================== +augmentationTools = [] + +#-------------------------------------------------------------------- +# Jpsi vertexing Tool +#-------------------------------------------------------------------- +# 1/ setup vertexing tools and services +include('DerivationFrameworkBPhys/configureVertexing.py') +BPHY9_VertexTools = BPHYVertexTools('BPHY9') + +# 2/ Setup the vertex fitter tools (e.g. JpsiFinder, JpsiPlus1Track, etc). +# These are general tools independent of DerivationFramework that do the +# actual vertex fitting and some pre-selection. +from JpsiUpsilonTools.JpsiUpsilonToolsConf import Analysis__JpsiFinder +BPHY9JpsiFinder = Analysis__JpsiFinder(name = 'BPHY9JpsiFinder', + OutputLevel = INFO, + muAndMu = True, + muAndTrack = False, + TrackAndTrack = False, + assumeDiMuons = True, # If true, will assume dimu hypothesis and use PDG value for mu mass + invMassUpper = 100000.0, + invMassLower = 0.0, + Chi2Cut = 200., + oppChargesOnly = True, + atLeastOneComb = True, + useCombinedMeasurement = False, # Only takes effect if combOnly=True + muonCollectionKey = 'Muons', + TrackParticleCollection = 'InDetTrackParticles', + V0VertexFitterTool = BPHY9_VertexTools.TrkV0Fitter, # V0 vertex fitter + useV0Fitter = False, # if False a TrkVertexFitterTool will be used + TrkVertexFitterTool = BPHY9_VertexTools.TrkVKalVrtFitter, # VKalVrt vertex fitter + TrackSelectorTool = BPHY9_VertexTools.InDetTrackSelectorTool, + VertexPointEstimator = BPHY9_VertexTools.VtxPointEstimator, + useMCPCuts = False) +ToolSvc += BPHY9JpsiFinder +print(BPHY9JpsiFinder) + +# 3/ Setup the vertex reconstruction 'call' tool(s). They are part of the derivation framework. These Augmentation tools add +# output vertex collection(s) into the StoreGate and add basic decorations which do not depend on the vertex mass hypothesis +# (e.g. lxy, ptError, etc). There should be one tool per topology, i.e. Jpsi and Psi(2S) do not need two instance of the +# Reco tool is the JpsiFinder mass window is wide enough. +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__Reco_Vertex +BPHY9_Reco_mumu = DerivationFramework__Reco_Vertex(name = 'BPHY9_Reco_mumu', + VertexSearchTool = BPHY9JpsiFinder, + OutputVtxContainerName = 'BPHY9OniaCandidates', + PVContainerName = 'PrimaryVertices', + RefPVContainerName = 'BPHY9RefittedPrimaryVertices', + RefitPV = True, + MaxPVrefit = 100000, + DoVertexType = 7) +ToolSvc += BPHY9_Reco_mumu +print(BPHY9_Reco_mumu) + +# 4/ Setup the vertex selection and augmentation tool(s). These tools decorate the vertices with variables that depend +# on the vertex mass hypothesis, e.g. invariant mass, proper decay time, etc. Property HypothesisName is used as a +# prefix for these decorations. They also perform tighter selection, flagging the vertecis that passed. The flag is +# a Char_t branch named 'passed_'+HypothesisName. It is used later by the 'SelectEvent' and 'Thin_vtxTrk' tools to +# determine which events and candidates should be kept in the output stream. Multiple instances of the Select_* tools +# can be used on a single input collection as long as they use different 'HypothesisName' flags. +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__Select_onia2mumu + +# 4a/ augment and select Jpsi->mumu candidates +BPHY9_Select_Jpsi2mumu = DerivationFramework__Select_onia2mumu(name = 'BPHY9_Select_Jpsi2mumu', + HypothesisName = 'Jpsi', + InputVtxContainerName = 'BPHY9OniaCandidates', + VtxMassHypo = 3096.916, + MassMin = 2000.0, + MassMax = 3600.0, + Chi2Max = 200, + DoVertexType = 7) +ToolSvc += BPHY9_Select_Jpsi2mumu +print(BPHY9_Select_Jpsi2mumu) + +# 4b/ augment and select Psi(2S)->mumu candidates +BPHY9_Select_Psi2mumu = DerivationFramework__Select_onia2mumu(name = 'BPHY9_Select_Psi2mumu', + HypothesisName = 'Psi', + InputVtxContainerName = 'BPHY9OniaCandidates', + VtxMassHypo = 3686.09, + MassMin = 3300.0, + MassMax = 4500.0, + Chi2Max = 200, + DoVertexType = 7) +ToolSvc += BPHY9_Select_Psi2mumu +print(BPHY9_Select_Psi2mumu) + +# 5/ select the event. We only want to keep events that contain certain vertices which passed certain selection. +# This is specified by the 'SelectionExpression' property, which contains the expression in the following format: +# 'ContainerName.passed_HypoName > count' +# where 'ContainerName' is output container form some Reco_* tool, 'HypoName' is the hypothesis name setup in some 'Select_*' +# tool and 'count' is the number of candidates passing the selection you want to keep. + +# Skimming +# a/ High pt lepton +ptSelection = '( count(Electrons.pt > 20*GeV) > 0 || count(Muons.pt > 20*GeV) > 0 )' +# b/ >3 total leptons +threelepSelection = '( count(Muons.pt > 0) + count(Electrons.pt > 0) >= 3 )' +# c/ di-muon vertex near Onia peak +oniaSelection = '( count(BPHY9OniaCandidates.passed_Jpsi) > 0 || count(BPHY9OniaCandidates.passed_Psi) > 0 )' +# & +expression = oniaSelection + ' && ' + threelepSelection + ' && ' + ptSelection +from DerivationFrameworkTools.DerivationFrameworkToolsConf import DerivationFramework__xAODStringSkimmingTool +BPHY9_SelectEvent = DerivationFramework__xAODStringSkimmingTool(name = 'BPHY9_SelectEvent', + expression = expression) +ToolSvc += BPHY9_SelectEvent + +# 6/ Track and vertex thinning. We want to remove all reconstructed secondary vertices which hasn't passed any of the +# selections defined by (Select_*) tools. We also want to keep only tracks which are associates with either muons or +# any of the vertices that passed the selection. Multiple thinning tools can perform the selection. The final thinning +# decision is based OR of all the decisions (by default, although it can be changed by the JO). + +# 6a/ Thining out vertices that didn't pass any selection and idetifying tracks associated with selected vertices. The +# 'VertexContainerNames' is a list of the vertex containers, and 'PassFlags' contains all pass flags for Select_* +# tools that must be satisfied. The vertex is kept is it satisfy any of the listed selections. + +from DerivationFrameworkBPhys.DerivationFrameworkBPhysConf import DerivationFramework__Thin_vtxTrk +BPHY9Thin_vtxTrk = DerivationFramework__Thin_vtxTrk(name = 'BPHY9Thin_vtxTrk', + TrackParticleContainerName = 'InDetTrackParticles', + VertexContainerNames = ['BPHY9OniaCandidates'], + PassFlags = ['passed_Jpsi', 'passed_Psi'] ) +ToolSvc += BPHY9Thin_vtxTrk + +# 6b/ thinning out tracks that are not attached to muons. The final thinning decision is based on the OR operation +# between decision from this and the previous tools. +from DerivationFrameworkInDet.DerivationFrameworkInDetConf import DerivationFramework__MuonTrackParticleThinning +BPHY9MuonTPThinningTool = DerivationFramework__MuonTrackParticleThinning(name = 'BPHY9MuonTPThinningTool', + MuonKey = 'Muons', + InDetTrackParticlesKey = 'InDetTrackParticles') +ToolSvc += BPHY9MuonTPThinningTool + +# 6c/ Only save truth informtion directly associated with Onia +from DerivationFrameworkMCTruth.DerivationFrameworkMCTruthConf import DerivationFramework__GenericTruthThinning +BPHY9TruthThinTool = DerivationFramework__GenericTruthThinning(name = 'BPHY9TruthThinTool', + ParticleSelectionString = 'TruthParticles.pdgId==443 || TruthParticles.pdgId==100443', + PreserveDescendants = True, + PreserveAncestors = True) +ToolSvc += BPHY9TruthThinTool +print(BPHY9TruthThinTool) + +#============================================================================== +# BACKGROUND ELECTRON DECORATION TYPE/ORIGIN +#============================================================================== +# PhysicsAnalysis/DerivationFramework/DerivationFrameworkEGamma/trunk/src/BkgElectronClassification.cxx + +if is_MC: + from MCTruthClassifier.MCTruthClassifierBase import MCTruthClassifier as BkgElectronMCTruthClassifier + from DerivationFrameworkEGamma.DerivationFrameworkEGammaConf import DerivationFramework__BkgElectronClassification + BPHY9BkgElectronClassificationTool = DerivationFramework__BkgElectronClassification (name = 'BkgElectronClassificationTool', + MCTruthClassifierTool = BkgElectronMCTruthClassifier) + ToolSvc += BPHY9BkgElectronClassificationTool + augmentationTools.append(BPHY9BkgElectronClassificationTool) + print('BkgElectronClassificationTool: ', BPHY9BkgElectronClassificationTool) + +#==================================================================== +# THINNING TOOLS +#==================================================================== +thinningTools=[] + +# Establish the thinning helper (which will set up the services behind the scenes) +from DerivationFrameworkCore.ThinningHelper import ThinningHelper +BPHY9ThinningHelper = ThinningHelper( 'BPHY9ThinningHelper' ) + +# Trigger Thinning Tool +elTrig = '^(?!.*_[0-9]*(mu|j|xe|tau|ht|xs|te))(?!HLT_e.*_[0-9]*e.*)HLT_e.*lhloose.*'\ + +'|^(?!.*_[0-9]*(mu|j|xe|tau|ht|xs|te))(?!HLT_e.*_[0-9]*e.*)HLT_e.*lhmedium.*'\ + +'|^(?!.*_[0-9]*(mu|j|xe|tau|ht|xs|te))(?!HLT_e.*_[0-9]*e.*)HLT_e.*lhtight.*'\ + +'|^(?!.*_[0-9]*(mu|j|xe|tau|ht|xs|te))(?!HLT_e.*_[0-9]*e.*)HLT_e.*lhvloose.*' +muTrig = '^(?!.*_[0-9]*(e|j|xe|tau|ht|xs|te))(?!HLT_mu.*_[0-9]*mu.*)HLT_mu.*' +BPHY9ThinningHelper.TriggerChains = elTrig + '|' + muTrig +BPHY9ThinningHelper.AppendToStream( BPHY9Stream ) + +# Jet tracks +from DerivationFrameworkInDet.DerivationFrameworkInDetConf import DerivationFramework__JetTrackParticleThinning +BPHY9JetTPThinningTool = DerivationFramework__JetTrackParticleThinning(name = 'BPHY9JetTPThinningTool', + JetKey = 'AntiKt4EMTopoJets', + InDetTrackParticlesKey = 'InDetTrackParticles', + ApplyAnd = True) +ToolSvc += BPHY9JetTPThinningTool +thinningTools.append(BPHY9JetTPThinningTool) + +# Tracks associated with Muons +from DerivationFrameworkInDet.DerivationFrameworkInDetConf import DerivationFramework__MuonTrackParticleThinning +BPHY9MuonTPThinningTool = DerivationFramework__MuonTrackParticleThinning(name = 'BPHY9MuonTPThinningTool', + MuonKey = 'Muons', + InDetTrackParticlesKey = 'InDetTrackParticles') +ToolSvc += BPHY9MuonTPThinningTool +thinningTools.append(BPHY9MuonTPThinningTool) + +# Tracks associated with Electrons +from DerivationFrameworkInDet.DerivationFrameworkInDetConf import DerivationFramework__EgammaTrackParticleThinning +BPHY9ElectronTPThinningTool = DerivationFramework__EgammaTrackParticleThinning(name = 'BPHY9ElectronTPThinningTool', + SGKey = 'Electrons', + InDetTrackParticlesKey = 'InDetTrackParticles') +ToolSvc += BPHY9ElectronTPThinningTool +thinningTools.append(BPHY9ElectronTPThinningTool) + +#==================================================================== +# Truth Thinning +#==================================================================== +# Truth selection strings +truth_cond_lep_list = [ +'(abs(TruthParticles.pdgId)>=11 && abs(TruthParticles.pdgId)<=14)', +'(TruthParticles.pt > 4.0*GeV)', +'(TruthParticles.status == 1)', +'(TruthParticles.barcode<200000)'] +truth_cond_lep = ' && '.join(truth_cond_lep_list) +truth_cond_photon = '(abs(TruthParticles.pdgId)==22) && (TruthParticles.pt>1*GeV)' +truth_cond_comb = '('+truth_cond_lep+') || ('+truth_cond_photon+')' + +# PreserveGeneratorDescendants only keeps particles that came directly from the event generator +# PreserveDescendants keeps all particles including those that come from Geant processes +BPHY9TruthTool = DerivationFramework__GenericTruthThinning(name = 'BPHY9TruthTool', + ParticleSelectionString = truth_cond_comb, + PreserveDescendants = True, + PreserveGeneratorDescendants = False, + PreserveAncestors = True, + TauHandling = False) + +from DerivationFrameworkMCTruth.DerivationFrameworkMCTruthConf import DerivationFramework__MenuTruthThinning +BPHY9TruthToolMenu = DerivationFramework__MenuTruthThinning(name = 'BPHY9TruthToolMenu', + WritePartons = False, + WriteHadrons = False, + WriteBHadrons = False, + WriteGeant = False, + GeantPhotonPtThresh = -1.0, + WriteTauHad = False, + PartonPtThresh = -1.0, + WriteBSM = False, + WriteBosons = True, + WriteBSMProducts = False, + WriteBosonProducts = True, + WriteTopAndDecays = True, + WriteEverything = False, + WriteAllLeptons = True, + WriteStatus3 = False, + WriteFirstN = -1) + +if is_MC: + ToolSvc += BPHY9TruthTool + thinningTools.append(BPHY9TruthTool) + ToolSvc += BPHY9TruthToolMenu + thinningTools.append(BPHY9TruthToolMenu) + +#======================================= +# CREATE PRIVATE SEQUENCE +#======================================= +BPHY9Seq = CfgMgr.AthSequencer('BPHY9Sequence') +from DerivationFrameworkFlavourTag.FlavourTagCommon import FlavorTagInit +FlavorTagInit(JetCollections=['AntiKt4EMPFlowJets'], Sequencer=BPHY9Seq) +#======================================= +# CREATE THE DERIVATION KERNEL ALGORITHM +#======================================= +BPHY9ThinningTools = [BPHY9Thin_vtxTrk, BPHY9MuonTPThinningTool] +if is_MC: + BPHY9ThinningTools.append(BPHY9TruthThinTool) + +from DerivationFrameworkCore.DerivationFrameworkCoreConf import DerivationFramework__DerivationKernel +DerivationFrameworkJob += CfgMgr.DerivationFramework__DerivationKernel( + 'BPHY9Kernel', + AugmentationTools = [BPHY9_Reco_mumu, BPHY9_Select_Jpsi2mumu, BPHY9_Select_Psi2mumu], + SkimmingTools = [BPHY9_SelectEvent]) + +#==================================================================== +# JetTagNonPromptLepton decorations +#==================================================================== +import JetTagNonPromptLepton.JetTagNonPromptLeptonConfig as JetTagConfig +# Build AntiKt4PV0TrackJets and run b-tagging +JetTagConfig.ConfigureAntiKt4PV0TrackJets(BPHY9Seq, 'BPHY9') +# Add BDT decoration algs +BPHY9Seq += JetTagConfig.GetDecoratePromptLeptonAlgs() +DerivationFrameworkJob += BPHY9Seq + +#==================================================================== +# SLIMMING TOOL +#==================================================================== +from DerivationFrameworkCore.SlimmingHelper import SlimmingHelper +BPHY9SlimmingHelper = SlimmingHelper('BPHY9SlimmingHelper') + +# Slimming for recontruction content +BPHY9SlimmingHelper.AllVariables = [] + +BPHY9SlimmingHelper.SmartCollections = ['AntiKt4EMPFlowJets', + 'AntiKt4EMPFlowJets_BTagging201903', + 'AntiKt4EMTopoJets', + 'AntiKt4EMTopoJets_BTagging201810', + 'BTagging_AntiKt4EMPFlow_201903', + 'BTagging_AntiKt4EMTopo_201810', + 'Electrons', + 'InDetTrackParticles', + 'MET_Reference_AntiKt4EMPFlow', + 'MET_Reference_AntiKt4EMTopo', + 'Muons', + 'PrimaryVertices'] + +extraJetVariables = '.JetEMScaleMomentum_pt.JetEMScaleMomentum_eta.JetEMScaleMomentum_phi.JetEMScaleMomentum_m'\ + +'.ConeTruthLabelID.PartonTruthLabelID.SumPtTrkPt1000.Jvt.JvtJvfcorr.JvtRpt'\ + +'.HECFrac.LArQuality.HECQuality.NegativeE.AverageLArQF' + +BPHY9SlimmingHelper.ExtraVariables = ['AntiKt4EMPFlowJets'+extraJetVariables, + 'AntiKt4EMTopoJets'+extraJetVariables, + 'CombinedMuonTrackParticles'+'.z0' + +'.vz' + +'.definingParametersCovMatrix', + 'Electrons'+'.author' + +'.charge', + 'ExtrapolatedMuonTrackParticles'+'.z0' + +'.vz' + +'.definingParametersCovMatrix', + 'GSFTrackParticles'+'.z0' + +'.vz' + +'.definingParametersCovMatrix', + 'Muons'+'.clusterLink' + +'.allAuthors' + +'.charge' + +'.extrapolatedMuonSpectrometerTrackParticleLink' + +'.scatteringCurvatureSignificance' + +'.scatteringNeighbourSignificance', + 'PrimaryVertices'+'.x' + +'.y'] + +BPHY9Stream.StaticContent = [] + +# Slimming for truth content +if is_MC: + BPHY9SlimmingHelper.AllVariables += ['TruthParticles', + 'TruthEvents', + 'TruthVertices'] + + BPHY9SlimmingHelper.SmartCollections += ['AntiKt4TruthJets', + 'AntiKt4TruthWZJets'] + + BPHY9SlimmingHelper.ExtraVariables += ['CombinedMuonTrackParticles'+'.truthOrigin' + +'.truthType' + +'.truthParticleLink', + 'Electrons'+'.truthOrigin' + +'.truthType' + +'.truthParticleLink' + +'.bkgTruthType' + +'.bkgTruthOrigin' + +'.bkgTruthParticleLink' + +'.bkgMotherPdgId' + +'.deltaPhi1', + 'InDetTrackParticles'+'.truthOrigin' + +'.truthType' + +'.truthParticleLink', + 'MuonTruthParticles'+'.truthOrigin' + +'.truthType' + +'.truthParticleLink'] + + BPHY9SlimmingHelper.StaticContent += ['xAOD::TruthParticleContainer#TruthMuons', + 'xAOD::TruthParticleAuxContainer#TruthMuonsAux.', + 'xAOD::TruthParticleContainer#TruthElectrons', + 'xAOD::TruthParticleAuxContainer#TruthElectronsAux.', + 'xAOD::TruthParticleContainer#TruthNeutrinos', + 'xAOD::TruthParticleAuxContainer#TruthNeutrinosAux.'] + +# Slimming for trigger content +BPHY9SlimmingHelper.IncludeEGammaTriggerContent = True +BPHY9SlimmingHelper.IncludeMuonTriggerContent = True + +# Slimming for charmonia content +BPHY9Stream.AddItem('xAOD::VertexContainer#BPHY9OniaCandidates') +BPHY9Stream.AddItem('xAOD::VertexAuxContainer#BPHY9OniaCandidatesAux.') +BPHY9Stream.AddItem('xAOD::VertexAuxContainer#BPHY9OniaCandidatesAux.-vxTrackAtVertex') + +BPHY9SlimmingHelper.AppendContentToStream(BPHY9Stream) \ No newline at end of file diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/SaveExtraMetadataInMerge_jobOFragment.py b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/SaveExtraMetadataInMerge_jobOFragment.py new file mode 100644 index 00000000000..0e7742c69c7 --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/SaveExtraMetadataInMerge_jobOFragment.py @@ -0,0 +1,53 @@ +# Copyright (C) 2002-2017 CERN for the benefit of the ATLAS collaboration +# +# JobOption fragment to be used during DAOD merging. It takes care of +# propagating all B-physics metadata objects to the output file. +# + +# Python import(s): +import re + +# Core import(s): +from RecExConfig.InputFilePeeker import inputFileSummary +from OutputStreamAthenaPool.MultipleStreamManager import MSMgr +from AthenaCommon.Logging import logging + +# Create a logger: +_logger = logging.getLogger( "SaveExtraMetadataInMerge_jobOFragment" ) + +# Find the exact name of xAOD::FileMetaData_vX in the inputFileSummary +# dictionary: +mdType = "" +for key in list(inputFileSummary[ 'metadata_itemsDic' ].keys()): + if re.match( 'xAOD::FileMetaData_v[0-9]+', key ): + mdType = key + break + pass + +# If there is, then let's do the rest of the setup: +if mdType != "": + + # Loop over the keys of all the xAOD::FileMetaData objects: + for key in inputFileSummary[ 'metadata_itemsDic' ][ mdType ]: + + # If it doesn't look like a b-physics metadata object, then leave + # it alone: + if not key.endswith( '_MetaData' ): + continue + + # Create the metadata tool for propagating this info: + toolName = "BPhysFileMetadataTool_%s" % key + ToolSvc += CfgMgr.xAODMaker__FileMetaDataTool( toolName, + InputKey = key, + OutputKey = key ) + svcMgr.MetaDataSvc.MetaDataTools += [ getattr( ToolSvc, toolName ) ] + _logger.info( "Created tool: %s" % toolName ) + + # Add the metadata to the output stream(s): + outputItems = [ 'xAOD::FileMetaData#%s' % key, + 'xAOD::FileMetaDataAuxInfo#%sAux.' % key ] + MSMgr.AddMetaDataItemToAllStreams( outputItems ) + _logger.info( "Added metatata items: %s" % str( outputItems ) ) + + pass + pass diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/configureConversionFinder.py b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/configureConversionFinder.py new file mode 100644 index 00000000000..67b775ec061 --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/configureConversionFinder.py @@ -0,0 +1,69 @@ +# +# Copyright (C) 2002-2019 CERN for the benefit of the ATLAS collaboration +# +#==================================================================== +# Collection of tools to run the conversion finder +#==================================================================== + +class BPHYConversionFinderTools: + + def __init__(self, derivation = ""): + + if derivation == "": + print('--------> FATAL: BPHYConversionFinderTools, "derivation" string not set!') + import sys + sys.exit() + + + prefix = derivation+"ConversionFinderTools" + + from AthenaCommon.AppMgr import ToolSvc + + # set up extrapolator + from TrkExTools.AtlasExtrapolator import AtlasExtrapolator + self.Extrapolator = AtlasExtrapolator(name = prefix+"_AtlasExtrapolator") + ToolSvc += self.Extrapolator + print(self.Extrapolator) + + from TrkVKalVrtFitter.TrkVKalVrtFitterConf import Trk__TrkVKalVrtFitter + self.InDetSecVxFitterTool = Trk__TrkVKalVrtFitter(name = prefix+"_Fitter", + Extrapolator = self.Extrapolator, + MakeExtendedVertex = True, + FirstMeasuredPoint = False, + Robustness = 6, + InputParticleMasses = [0.511,0.511], + VertexForConstraint = [0.,0.,0.], + CovVrtForConstraint = [0.015*0.015,0.,0.015*0.015,0.,0.,10000.*10000.], + FirstMeasuredPointLimit = True, + usePhiCnst = True, + useThetaCnst = True) + ToolSvc += self.InDetSecVxFitterTool + print(self.InDetSecVxFitterTool) + + + from TrkVertexSeedFinderUtils.TrkVertexSeedFinderUtilsConf import Trk__SeedNewtonTrkDistanceFinder + self.InDetSecVxTrkDistanceFinder = Trk__SeedNewtonTrkDistanceFinder(name = prefix+"_TrkDistanceFinder") + ToolSvc += self.InDetSecVxTrkDistanceFinder + print(self.InDetSecVxTrkDistanceFinder) + + + from InDetConversionFinderTools.InDetConversionFinderToolsConf import InDet__VertexPointEstimator + self.InDetSecVtxPointEstimator = InDet__VertexPointEstimator(name = prefix+"_PointEstimator", + MinDeltaR = [-5.,-25.,-50.], # D-R1-R2 min cut + MaxDeltaR = [5.,10.,10.] , # D-R1-R2 max cut + MaxPhi = [0.05, 0.5, 0.5]) # dphi cut at vertex + ToolSvc += self.InDetSecVtxPointEstimator + print(self.InDetSecVtxPointEstimator) + + + from InDetConversionFinderTools.InDetConversionFinderToolsConf import InDet__ConversionPostSelector + self.InDetSecVtxPostSelector = InDet__ConversionPostSelector(name = prefix+"_PostSelector", + MaxChi2Vtx = [10.,10.,10.], + MaxInvariantMass = [10000.,10000.,10000.], + MinFitMomentum = [0.,0.,0.], + MinRadius = [10.0,10.0,10.0], + MinPt = 0.0, + MaxdR = -10000.0, + MaxPhiVtxTrk = 10000.0) + ToolSvc += self.InDetSecVtxPostSelector + print(self.InDetSecVtxPostSelector) diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/configureSimpleV0Finder.py b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/configureSimpleV0Finder.py new file mode 100644 index 00000000000..bf1f8ca02ee --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/configureSimpleV0Finder.py @@ -0,0 +1,201 @@ +#==================================================================== +# Collection of tools required by V0Finder +#==================================================================== + +class BPHYV0FinderTools: + + def __init__(self, derivation = ""): + + if derivation == "": + print('--------> FATAL: BPHYV0FinderTools, "derivation" string not set!') + import sys + sys.exit() + + from AthenaCommon.AppMgr import ToolSvc + + # set up extrapolator + from TrkExTools.AtlasExtrapolator import AtlasExtrapolator + self.InDetExtrapolator = AtlasExtrapolator(name = derivation+"_AtlasExtrapolator") + ToolSvc += self.InDetExtrapolator + print(self.InDetExtrapolator) + + # Vertex point estimator + from InDetConversionFinderTools.InDetConversionFinderToolsConf import InDet__VertexPointEstimator + self.V0VtxPointEstimator = InDet__VertexPointEstimator(name = derivation+"_VtxPointEstimator", + MaxTrkXYDiffAtVtx = [20.,20.,20.], + MaxTrkZDiffAtVtx = [100.,100.,100.], + MaxTrkXYValue = [400.,400.,400.], + MinArcLength = [-800.,-800.,-800.], + MaxArcLength = [800.,800.,800.], + MinDeltaR = [-10000.,-10000.,-10000.], + MaxDeltaR = [10000.,10000.,10000.], + MaxPhi = [10000., 10000., 10000.], + MaxChi2OfVtxEstimation = 2000.) + ToolSvc += self.V0VtxPointEstimator + print(self.V0VtxPointEstimator) + + from InDetRecExample.InDetKeys import InDetKeys + + from InDetAssociationTools.InDetAssociationToolsConf import InDet__InDetPRD_AssociationToolGangedPixels + self.V0PrdAssociationTool = InDet__InDetPRD_AssociationToolGangedPixels(name = derivation+"_V0PrdAssociationTool", + PixelClusterAmbiguitiesMapName = InDetKeys.GangedPixelMap()) + ToolSvc += self.V0PrdAssociationTool + print(self.V0PrdAssociationTool) + + from RecExConfig.RecFlags import rec + CountDeadModulesAfterLastHit=False + #rec.Commissioning=False + + from InDetTrackHoleSearch.InDetTrackHoleSearchConf import InDet__InDetTrackHoleSearchTool + self.V0HoleSearchTool = InDet__InDetTrackHoleSearchTool(name = derivation+"_V0HoleSearchTool", + Extrapolator = self.InDetExtrapolator, + usePixel = DetFlags.haveRIO.pixel_on(), + useSCT = DetFlags.haveRIO.SCT_on(), + #Commissioning = rec.Commissioning()) + CountDeadModulesAfterLastHit = CountDeadModulesAfterLastHit) + ToolSvc += self.V0HoleSearchTool + print(self.V0HoleSearchTool) + + from InDetTrackSummaryHelperTool.InDetTrackSummaryHelperToolConf import InDet__InDetTrackSummaryHelperTool + self.V0TrackSummaryHelperTool = InDet__InDetTrackSummaryHelperTool(name = derivation+"_InDetSummaryHelper", + AssoTool = self.V0PrdAssociationTool, + DoSharedHits = False, + HoleSearch = self.V0HoleSearchTool, + usePixel = DetFlags.haveRIO.pixel_on(), + useSCT = DetFlags.haveRIO.SCT_on(), + useTRT = DetFlags.haveRIO.TRT_on()) + ToolSvc += self.V0TrackSummaryHelperTool + print(self.V0TrackSummaryHelperTool) + + from TrkTrackSummaryTool.TrkTrackSummaryToolConf import Trk__TrackSummaryTool + self.V0TrackSummaryTool = Trk__TrackSummaryTool(name = derivation+"_V0TrackSummaryTool", + InDetSummaryHelperTool = self.V0TrackSummaryHelperTool, + doSharedHits = False, + InDetHoleSearchTool = self.V0HoleSearchTool) + ToolSvc += self.V0TrackSummaryTool + print(self.V0TrackSummaryTool) + + + + from InDetTrackSelectorTool.InDetTrackSelectorToolConf import InDet__InDetConversionTrackSelectorTool + self.InDetV0VxTrackSelector = InDet__InDetConversionTrackSelectorTool(name = derivation+"InDetV0VxTrackSelector", + TrackSummaryTool = self.V0TrackSummaryTool, + Extrapolator = self.InDetExtrapolator, + # Extrapolator = "Trk::Extrapolator/InDetExtrapolator", + maxTrtD0 = 50., + maxSiZ0 = 250., + significanceD0_Si = 1., + significanceD0_Trt = 1., + significanceZ0_Trt = 3., + minPt = 400.0, + IsConversion = False) + ToolSvc += self.InDetV0VxTrackSelector + print(self.InDetV0VxTrackSelector) + + + # configure vertex fitters + + from TrkV0Fitter.TrkV0FitterConf import Trk__TrkV0VertexFitter + self.BPhysV0Fitter = Trk__TrkV0VertexFitter(name = derivation+'_BPhysV0Fitter', + MaxIterations = 10, + Use_deltaR = False, + FirstMeasuredPoint = True, + Extrapolator = self.InDetExtrapolator) + ToolSvc += self.BPhysV0Fitter + print(self.BPhysV0Fitter) +# + from TrkVKalVrtFitter.TrkVKalVrtFitterConf import Trk__TrkVKalVrtFitter + self.BPhysVKVertexFitter = Trk__TrkVKalVrtFitter(name = derivation+"_BPhysVKVFitter", + Extrapolator = self.InDetExtrapolator, + IterationNumber = 30, + FirstMeasuredPoint = True, + MakeExtendedVertex = True) + ToolSvc += self.BPhysVKVertexFitter + print(self.BPhysVKVertexFitter) + +# + from TrkVKalVrtFitter.TrkVKalVrtFitterConf import Trk__TrkVKalVrtFitter + self.BPhysKshortFitter = Trk__TrkVKalVrtFitter(name = derivation+"_BPhysVKKVFitter", + Extrapolator = self.InDetExtrapolator, + IterationNumber = 30, + FirstMeasuredPoint = True, + MakeExtendedVertex = True, + InputParticleMasses = [139.57,139.57], + MassForConstraint = 497.672) + ToolSvc += self.BPhysKshortFitter + print(self.BPhysKshortFitter) +# + from TrkVKalVrtFitter.TrkVKalVrtFitterConf import Trk__TrkVKalVrtFitter + self.BPhysLambdaFitter = Trk__TrkVKalVrtFitter(name = derivation+"_BPhysVKLFitter", + Extrapolator = self.InDetExtrapolator, + IterationNumber = 30, + FirstMeasuredPoint = True, + MakeExtendedVertex = True, + InputParticleMasses = [938.272,139.57], + MassForConstraint = 1115.68) + ToolSvc += self.BPhysLambdaFitter + print(self.BPhysLambdaFitter) +# + from TrkVKalVrtFitter.TrkVKalVrtFitterConf import Trk__TrkVKalVrtFitter + self.BPhysLambdabarFitter = Trk__TrkVKalVrtFitter(name = derivation+"_BPhysVKLbFitter", + Extrapolator = self.InDetExtrapolator, + IterationNumber = 30, + FirstMeasuredPoint = True, + MakeExtendedVertex = True, + InputParticleMasses = [139.57,938.272], + MassForConstraint = 1115.68) + ToolSvc += self.BPhysLambdabarFitter + print(self.BPhysLambdabarFitter) +# + from TrkVKalVrtFitter.TrkVKalVrtFitterConf import Trk__TrkVKalVrtFitter + self.BPhysGammaFitter = Trk__TrkVKalVrtFitter(name = derivation+"_BPhysVKGFitter", + Extrapolator = self.InDetExtrapolator, + IterationNumber = 30, + Robustness = 6, + FirstMeasuredPoint = True, + MakeExtendedVertex = True, + usePhiCnst = True, + useThetaCnst = True, + InputParticleMasses = [0.511,0.511]) + ToolSvc += self.BPhysGammaFitter + print(self.BPhysGammaFitter) + +##-------------------------------------------- +## Setup V0Finder +##-------------------------------------------- + from InDetV0Finder.InDetV0FinderConf import InDet__InDetV0FinderTool + self.V0FinderTool = InDet__InDetV0FinderTool(name = derivation+'_InDetV0FinderTool', + TrackParticleCollection = InDetKeys.xAODTrackParticleContainer(), + #TrackParticleCollection = "InDetTrackParticles", + useV0Fitter = True, + VertexFitterTool = self.BPhysV0Fitter, + VKVertexFitterTool = self.BPhysVKVertexFitter, + KshortFitterTool = self.BPhysKshortFitter, + LambdaFitterTool = self.BPhysLambdaFitter, + LambdabarFitterTool = self.BPhysLambdabarFitter, + GammaFitterTool = self.BPhysGammaFitter, + TrackSelectorTool = self.InDetV0VxTrackSelector, + VertexPointEstimator = self.V0VtxPointEstimator, + doSimpleV0 = True, + #useorigin = False, + #useTRTplusTRT = True, + #useTRTplusSi = True, + useVertexCollection = True, + #trkSelPV = True, + Extrapolator = self.InDetExtrapolator) + #Extrapolator = "Trk::Extrapolator/InDetExtrapolator") + ToolSvc += self.V0FinderTool + print(self.V0FinderTool) + + #from InDetV0Finder.InDetV0FinderConf import InDet__InDetV0Finder + #self.InDetV0Finder = InDet__InDetV0Finder(name = derivation+'InDetV0Finder', + # #decorateV0 = False, + # InDetV0FinderToolName = V0FinderTool, + # V0ContainerName = InDetKeys.xAODV0VertexContainer(), + # KshortContainerName = InDetKeys.xAODKshortVertexContainer(), + # LambdaContainerName = InDetKeys.xAODLambdaVertexContainer(), + # LambdabarContainerName = InDetKeys.xAODLambdabarVertexContainer()) + #topSequence += self.InDetV0Finder + #print self.InDetV0Finder + + diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/configureV0Finder.py b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/configureV0Finder.py new file mode 100644 index 00000000000..0add564e26b --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/configureV0Finder.py @@ -0,0 +1,202 @@ +#==================================================================== +# Collection of tools required by V0Finder +#==================================================================== + +class BPHYV0FinderTools: + + def __init__(self, derivation = ""): + + if derivation == "": + print('--------> FATAL: BPHYV0FinderTools, "derivation" string not set!') + import sys + sys.exit() + + from AthenaCommon.AppMgr import ToolSvc + + # set up extrapolator + from TrkExTools.AtlasExtrapolator import AtlasExtrapolator + self.InDetExtrapolator = AtlasExtrapolator(name = derivation+"_AtlasExtrapolator") + ToolSvc += self.InDetExtrapolator + print(self.InDetExtrapolator) + + # Vertex point estimator + from InDetConversionFinderTools.InDetConversionFinderToolsConf import InDet__VertexPointEstimator + self.V0VtxPointEstimator = InDet__VertexPointEstimator(name = derivation+"_VtxPointEstimator", + MaxTrkXYDiffAtVtx = [20.,20.,20.], + MaxTrkZDiffAtVtx = [100.,100.,100.], + MaxTrkXYValue = [400.,400.,400.], + MinArcLength = [-800.,-800.,-800.], + MaxArcLength = [800.,800.,800.], + MinDeltaR = [-10000.,-10000.,-10000.], + MaxDeltaR = [10000.,10000.,10000.], + MaxPhi = [10000., 10000., 10000.], + MaxChi2OfVtxEstimation = 2000.) + ToolSvc += self.V0VtxPointEstimator + print(self.V0VtxPointEstimator) + + from InDetRecExample.InDetKeys import InDetKeys + + from InDetAssociationTools.InDetAssociationToolsConf import InDet__InDetPRD_AssociationToolGangedPixels + self.V0PrdAssociationTool = InDet__InDetPRD_AssociationToolGangedPixels(name = derivation+"_V0PrdAssociationTool", + PixelClusterAmbiguitiesMapName = InDetKeys.GangedPixelMap()) + ToolSvc += self.V0PrdAssociationTool + print(self.V0PrdAssociationTool) + + from RecExConfig.RecFlags import rec + CountDeadModulesAfterLastHit=False + #rec.Commissioning=False + + from InDetTrackHoleSearch.InDetTrackHoleSearchConf import InDet__InDetTrackHoleSearchTool + self.V0HoleSearchTool = InDet__InDetTrackHoleSearchTool(name = derivation+"_V0HoleSearchTool", + Extrapolator = self.InDetExtrapolator, + usePixel = DetFlags.haveRIO.pixel_on(), + useSCT = DetFlags.haveRIO.SCT_on(), + #Commissioning = rec.Commissioning()) + CountDeadModulesAfterLastHit = CountDeadModulesAfterLastHit) + ToolSvc += self.V0HoleSearchTool + print(self.V0HoleSearchTool) + + from InDetTrackSummaryHelperTool.InDetTrackSummaryHelperToolConf import InDet__InDetTrackSummaryHelperTool + self.V0TrackSummaryHelperTool = InDet__InDetTrackSummaryHelperTool(name = derivation+"_InDetSummaryHelper", + AssoTool = self.V0PrdAssociationTool, + DoSharedHits = False, + HoleSearch = self.V0HoleSearchTool, + usePixel = DetFlags.haveRIO.pixel_on(), + useSCT = DetFlags.haveRIO.SCT_on(), + useTRT = DetFlags.haveRIO.TRT_on()) + ToolSvc += self.V0TrackSummaryHelperTool + print(self.V0TrackSummaryHelperTool) + + from TrkTrackSummaryTool.TrkTrackSummaryToolConf import Trk__TrackSummaryTool + self.V0TrackSummaryTool = Trk__TrackSummaryTool(name = derivation+"_V0TrackSummaryTool", + InDetSummaryHelperTool = self.V0TrackSummaryHelperTool, + doSharedHits = False, + InDetHoleSearchTool = self.V0HoleSearchTool) + ToolSvc += self.V0TrackSummaryTool + print(self.V0TrackSummaryTool) + + + + from InDetTrackSelectorTool.InDetTrackSelectorToolConf import InDet__InDetConversionTrackSelectorTool + self.InDetV0VxTrackSelector = InDet__InDetConversionTrackSelectorTool(name = derivation+"InDetV0VxTrackSelector", + TrackSummaryTool = self.V0TrackSummaryTool, + Extrapolator = self.InDetExtrapolator, + # Extrapolator = "Trk::Extrapolator/InDetExtrapolator", + maxTrtD0 = 50., + maxSiZ0 = 250., + significanceD0_Si = 1., + significanceD0_Trt = 1., + significanceZ0_Trt = 3., + minPt = 400.0, + IsConversion = False) + ToolSvc += self.InDetV0VxTrackSelector + print(self.InDetV0VxTrackSelector) + + + # configure vertex fitters + + from TrkV0Fitter.TrkV0FitterConf import Trk__TrkV0VertexFitter + self.BPhysV0Fitter = Trk__TrkV0VertexFitter(name = derivation+'_BPhysV0Fitter', + MaxIterations = 10, + Use_deltaR = False, + FirstMeasuredPoint = True, + Extrapolator = self.InDetExtrapolator) + ToolSvc += self.BPhysV0Fitter + print(self.BPhysV0Fitter) +# + from TrkVKalVrtFitter.TrkVKalVrtFitterConf import Trk__TrkVKalVrtFitter + self.BPhysVKVertexFitter = Trk__TrkVKalVrtFitter(name = derivation+"_BPhysVKVFitter", + Extrapolator = self.InDetExtrapolator, + IterationNumber = 30, + FirstMeasuredPoint = True, + MakeExtendedVertex = True) + ToolSvc += self.BPhysVKVertexFitter + print(self.BPhysVKVertexFitter) + +# + from TrkVKalVrtFitter.TrkVKalVrtFitterConf import Trk__TrkVKalVrtFitter + self.BPhysKshortFitter = Trk__TrkVKalVrtFitter(name = derivation+"_BPhysVKKVFitter", + Extrapolator = self.InDetExtrapolator, + IterationNumber = 30, + FirstMeasuredPoint = True, + MakeExtendedVertex = True, + InputParticleMasses = [139.57,139.57], + MassForConstraint = 497.672) + ToolSvc += self.BPhysKshortFitter + print(self.BPhysKshortFitter) +# + from TrkVKalVrtFitter.TrkVKalVrtFitterConf import Trk__TrkVKalVrtFitter + self.BPhysLambdaFitter = Trk__TrkVKalVrtFitter(name = derivation+"_BPhysVKLFitter", + Extrapolator = self.InDetExtrapolator, + IterationNumber = 30, + FirstMeasuredPoint = True, + MakeExtendedVertex = True, + InputParticleMasses = [938.272,139.57], + MassForConstraint = 1115.68) + ToolSvc += self.BPhysLambdaFitter + print(self.BPhysLambdaFitter) +# + from TrkVKalVrtFitter.TrkVKalVrtFitterConf import Trk__TrkVKalVrtFitter + self.BPhysLambdabarFitter = Trk__TrkVKalVrtFitter(name = derivation+"_BPhysVKLbFitter", + Extrapolator = self.InDetExtrapolator, + IterationNumber = 30, + FirstMeasuredPoint = True, + MakeExtendedVertex = True, + InputParticleMasses = [139.57,938.272], + MassForConstraint = 1115.68) + ToolSvc += self.BPhysLambdabarFitter + print(self.BPhysLambdabarFitter) +# + from TrkVKalVrtFitter.TrkVKalVrtFitterConf import Trk__TrkVKalVrtFitter + self.BPhysGammaFitter = Trk__TrkVKalVrtFitter(name = derivation+"_BPhysVKGFitter", + Extrapolator = self.InDetExtrapolator, + IterationNumber = 30, + Robustness = 6, + FirstMeasuredPoint = True, + MakeExtendedVertex = True, + usePhiCnst = True, + useThetaCnst = True, + InputParticleMasses = [0.511,0.511]) + ToolSvc += self.BPhysGammaFitter + print(self.BPhysGammaFitter) + +##-------------------------------------------- +## Setup V0Finder +##-------------------------------------------- + from InDetV0Finder.InDetV0FinderConf import InDet__InDetV0FinderTool + self.V0FinderTool = InDet__InDetV0FinderTool(name = derivation+'_InDetV0FinderTool', + TrackParticleCollection = InDetKeys.xAODTrackParticleContainer(), + #TrackParticleCollection = "InDetTrackParticles", + useV0Fitter = True, + VertexFitterTool = self.BPhysV0Fitter, + VKVertexFitterTool = self.BPhysVKVertexFitter, + KshortFitterTool = self.BPhysKshortFitter, + LambdaFitterTool = self.BPhysLambdaFitter, + LambdabarFitterTool = self.BPhysLambdabarFitter, + GammaFitterTool = self.BPhysGammaFitter, + TrackSelectorTool = self.InDetV0VxTrackSelector, + VertexPointEstimator = self.V0VtxPointEstimator, + doSimpleV0 = False, + #doSimpleV0 = True, + #useorigin = False, + #useTRTplusTRT = True, + #useTRTplusSi = True, + useVertexCollection = True, + #trkSelPV = True, + Extrapolator = self.InDetExtrapolator) + #Extrapolator = "Trk::Extrapolator/InDetExtrapolator") + ToolSvc += self.V0FinderTool + print(self.V0FinderTool) + + #from InDetV0Finder.InDetV0FinderConf import InDet__InDetV0Finder + #self.InDetV0Finder = InDet__InDetV0Finder(name = derivation+'InDetV0Finder', + # #decorateV0 = False, + # InDetV0FinderToolName = V0FinderTool, + # V0ContainerName = InDetKeys.xAODV0VertexContainer(), + # KshortContainerName = InDetKeys.xAODKshortVertexContainer(), + # LambdaContainerName = InDetKeys.xAODLambdaVertexContainer(), + # LambdabarContainerName = InDetKeys.xAODLambdabarVertexContainer()) + #topSequence += self.InDetV0Finder + #print self.InDetV0Finder + + diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/configureVertexing.py b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/configureVertexing.py new file mode 100644 index 00000000000..a3746d92509 --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/share/configureVertexing.py @@ -0,0 +1,148 @@ +#==================================================================== +# Collection of tools required by JpsiFinder +# Based on JpsiUpsilonTools/configureServices.py +#==================================================================== + +class BPHYVertexTools: + + def __init__(self, derivation = ""): + + if derivation == "": + print ('--------> FATAL: BPHYVertexTools, "derivation" string not set!') + import sys + sys.exit() + + from AthenaCommon.AppMgr import ToolSvc + + # set up extrapolator + from TrkExTools.AtlasExtrapolator import AtlasExtrapolator + self.InDetExtrapolator = AtlasExtrapolator(name = derivation+"_AtlasExtrapolator") + ToolSvc += self.InDetExtrapolator + print((self.InDetExtrapolator)) + + # Vertex point estimator + from InDetConversionFinderTools.InDetConversionFinderToolsConf import InDet__VertexPointEstimator + self.VtxPointEstimator = InDet__VertexPointEstimator(name = derivation+"_VtxPointEstimator", + MinDeltaR = [-10000.,-10000.,-10000.], + MaxDeltaR = [10000.,10000.,10000.], + MaxPhi = [10000., 10000., 10000.], + MaxChi2OfVtxEstimation = 2000.) + ToolSvc += self.VtxPointEstimator + print((self.VtxPointEstimator)) + + from InDetConversionFinderTools.InDetConversionFinderToolsConf import InDet__ConversionFinderUtils + self.InDetConversionHelper = InDet__ConversionFinderUtils(name = derivation+"_InDetConversionFinderUtils") + ToolSvc += self.InDetConversionHelper + print((self.InDetConversionHelper)) + + from InDetRecExample.InDetKeys import InDetKeys + + from InDetAssociationTools.InDetAssociationToolsConf import InDet__InDetPRD_AssociationToolGangedPixels + self.InDetPrdAssociationTool = InDet__InDetPRD_AssociationToolGangedPixels(name = derivation+"_InDetPrdAssociationTool", + PixelClusterAmbiguitiesMapName = InDetKeys.GangedPixelMap()) + ToolSvc += self.InDetPrdAssociationTool + print((self.InDetPrdAssociationTool)) + + from RecExConfig.RecFlags import rec + CountDeadModulesAfterLastHit=False + #rec.Commissioning=False + + from InDetTrackHoleSearch.InDetTrackHoleSearchConf import InDet__InDetTrackHoleSearchTool + self.InDetHoleSearchTool = InDet__InDetTrackHoleSearchTool(name = derivation+"_InDetHoleSearchTool", + Extrapolator = self.InDetExtrapolator, + #usePixel = DetFlags.haveRIO.pixel_on(), + #useSCT = DetFlags.haveRIO.SCT_on(), + #Commissioning = rec.Commissioning()) + CountDeadModulesAfterLastHit = CountDeadModulesAfterLastHit) + ToolSvc += self.InDetHoleSearchTool + print((self.InDetHoleSearchTool)) + + from InDetTrackSummaryHelperTool.InDetTrackSummaryHelperToolConf import InDet__InDetTrackSummaryHelperTool + self.InDetTrackSummaryHelperTool = InDet__InDetTrackSummaryHelperTool(name = derivation+"_InDetSummaryHelper", + AssoTool = self.InDetPrdAssociationTool, + DoSharedHits = False, + HoleSearch = self.InDetHoleSearchTool, + usePixel = DetFlags.haveRIO.pixel_on(), + useSCT = DetFlags.haveRIO.SCT_on(), + useTRT = DetFlags.haveRIO.TRT_on()) + ToolSvc += self.InDetTrackSummaryHelperTool + print((self.InDetTrackSummaryHelperTool)) + + from TrkTrackSummaryTool.TrkTrackSummaryToolConf import Trk__TrackSummaryTool + self.InDetTrackSummaryTool = Trk__TrackSummaryTool(name = derivation+"_InDetTrackSummaryTool", + InDetSummaryHelperTool = self.InDetTrackSummaryHelperTool, + doSharedHits = False, + #InDetHoleSearchTool = self.InDetHoleSearchTool + ) + ToolSvc += self.InDetTrackSummaryTool + print((self.InDetTrackSummaryTool)) + + # ===================================================== + # THIS IS WHERE THE USER CONTROLS MAIN TRACK SELECTIONS + # ===================================================== + from InDetTrackSelectorTool.InDetTrackSelectorToolConf import InDet__InDetDetailedTrackSelectorTool + self.InDetTrackSelectorTool = InDet__InDetDetailedTrackSelectorTool(name = derivation+"_InDetDetailedTrackSelectorTool", + pTMin = 400.0, + IPd0Max = 10000.0, + IPz0Max = 10000.0, + z0Max = 10000.0, + sigIPd0Max = 10000.0, + sigIPz0Max = 10000.0, + d0significanceMax = -1., + z0significanceMax = -1., + etaMax = 9999., + useTrackSummaryInfo = True, + nHitBLayer = 0, + nHitPix = 1, + nHitBLayerPlusPix = 1, + nHitSct = 2, + nHitSi = 3, + nHitTrt = 0, + nHitTrtHighEFractionMax = 10000.0, + useSharedHitInfo = False, + useTrackQualityInfo = True, + fitChi2OnNdfMax = 10000.0, + TrtMaxEtaAcceptance = 1.9, + TrackSummaryTool = self.InDetTrackSummaryTool, + Extrapolator = self.InDetExtrapolator + ) + + ToolSvc += self.InDetTrackSelectorTool + print((self.InDetTrackSelectorTool)) + + # configure vertex fitters + from TrkVKalVrtFitter.TrkVKalVrtFitterConf import Trk__TrkVKalVrtFitter + self.TrkVKalVrtFitter = Trk__TrkVKalVrtFitter( + name = derivation+"_VKalVrtFitter", + Extrapolator = self.InDetExtrapolator, + # MagFieldSvc = InDetMagField, + FirstMeasuredPoint = False, + #FirstMeasuredPointLimit = True, + MakeExtendedVertex = True) + ToolSvc += self.TrkVKalVrtFitter + print((self.TrkVKalVrtFitter)) + + from TrkVertexFitterUtils.TrkVertexFitterUtilsConf import Trk__FullLinearizedTrackFactory + self.InDetLinFactory = Trk__FullLinearizedTrackFactory(name = derivation+"_Trk::InDetFullLinearizedTrackFactory", + Extrapolator = self.InDetExtrapolator, + # MagneticFieldTool = InDetMagField + ) + ToolSvc += self.InDetLinFactory + print((self.InDetLinFactory)) + + + from TrkV0Fitter.TrkV0FitterConf import Trk__TrkV0VertexFitter + self.TrkV0Fitter = Trk__TrkV0VertexFitter(name = derivation+"_TrkV0FitterName", + MaxIterations = 10, + Use_deltaR = False, + Extrapolator = self.InDetExtrapolator, + # MagneticFieldTool = InDetMagField + ) + ToolSvc += self.TrkV0Fitter + print((self.TrkV0Fitter)) + + # Primary vertex refitting + from TrkVertexFitterUtils.TrkVertexFitterUtilsConf import Trk__KalmanVertexUpdator + self.VertexUpdator = Trk__KalmanVertexUpdator(name = derivation+"_KalmanVertexUpdator") + ToolSvc += self.VertexUpdator + print((self.VertexUpdator)) diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/AugOriginalCounts.cxx b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/AugOriginalCounts.cxx new file mode 100644 index 00000000000..68c294d70fb --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/AugOriginalCounts.cxx @@ -0,0 +1,150 @@ +/* + Copyright (C) 2002-2019 CERN for the benefit of the ATLAS collaboration +*/ + +/** + * @file AugOriginalCounts.cxx + * + */ + +#include "DerivationFrameworkBPhys/AugOriginalCounts.h" +#include <StoreGate/WriteDecorHandle.h> +#include "GaudiKernel/EventContext.h" + +using namespace xAOD; +namespace DerivationFramework { + + AugOriginalCounts::AugOriginalCounts(const std::string& t, + const std::string& n, + const IInterface* p) : + AthAlgTool(t,n,p), + m_TrackContainername("InDetTrackParticles"), + m_PVContainername("PrimaryVertices") + { + declareInterface<DerivationFramework::IAugmentationTool>(this); + + declareProperty("TrackContainer", m_TrackContainername); + declareProperty("VertexContainer", m_PVContainername); + declareProperty("AddPVCountsByType", m_addPVCountsByType = false); + // decorate PVs with track counts and/or sqrt(sum(pt^2)) + // (needed if track collection will be thinned) + declareProperty("AddNTracksToPVs", m_addNTracksToPVs = false); + declareProperty("AddSqrtPt2SumToPVs", m_addSqrtPt2SumToPVs = false); + } + + StatusCode AugOriginalCounts::initialize() + { + ATH_CHECK(m_TrackContainername.initialize(SG::AllowEmpty)); + ATH_CHECK(m_PVContainername.initialize(SG::AllowEmpty)); + + if(!m_PVContainername.empty()){ + std::string pvstring = "EventInfo.OriginalCount_"; + pvstring += m_PVContainername.key(); + m_OrigPVNTracks = std::move(pvstring); + ATH_CHECK(m_OrigPVNTracks.initialize()); + } + if ( m_addPVCountsByType ) { + std::string pv0string = "EventInfo.OriginalCount_type0_"+m_PVContainername.key(); + std::string pv1string = "EventInfo.OriginalCount_type1_"+m_PVContainername.key(); + std::string pv2string = "EventInfo.OriginalCount_type2_"+m_PVContainername.key(); + std::string pv3string = "EventInfo.OriginalCount_type3_"+m_PVContainername.key(); + std::string pvUstring = "EventInfo.OriginalCount_typeUnknown_"+m_PVContainername.key(); + m_OrigNtype0 = std::move(pv0string); + m_OrigNtype1 = std::move(pv1string); + m_OrigNtype2 = std::move(pv2string); + m_OrigNtype3 = std::move(pv3string); + m_OrigNtypeUnknown = std::move(pvUstring); + ATH_CHECK(m_OrigNtype0.initialize()); + ATH_CHECK(m_OrigNtype1.initialize()); + ATH_CHECK(m_OrigNtype2.initialize()); + ATH_CHECK(m_OrigNtype3.initialize()); + ATH_CHECK(m_OrigNtypeUnknown.initialize()); + } + if ( m_addSqrtPt2SumToPVs ) { + std::string trackcon = m_PVContainername.key(); + trackcon += ".OriginalCount_"; + trackcon += m_TrackContainername.key(); + m_OrigSqrtPt2Sum = std::move(trackcon); + ATH_CHECK(m_OrigSqrtPt2Sum.initialize()); + } + if ( m_addNTracksToPVs ) { + std::string name = m_PVContainername.key(); + name+= ".OrigNTracks"; + m_d_nPVTracks = std::move(name); + ATH_CHECK(m_d_nPVTracks.initialize()); + } + if(!m_TrackContainername.empty()){ + m_OrigNTracksKeys = "EventInfo.OriginalCount_" + m_TrackContainername.key(); + ATH_CHECK(m_OrigNTracksKeys.initialize()); + } + return StatusCode::SUCCESS; + } + + StatusCode AugOriginalCounts::addBranches() const + { + + const EventContext& ctx = Gaudi::Hive::currentContext(); + + if(!m_PVContainername.empty()){ + + SG::WriteDecorHandle<xAOD::EventInfo, int> PV_count(m_OrigPVNTracks, ctx); + SG::ReadHandle<xAOD::VertexContainer> vertices(m_PVContainername, ctx); + PV_count(0) = vertices->size(); + + if ( m_addPVCountsByType ) { + SG::WriteDecorHandle<xAOD::EventInfo, int> PV0_count(m_OrigNtype0, ctx); + SG::WriteDecorHandle<xAOD::EventInfo, int> PV1_count(m_OrigNtype1, ctx); + SG::WriteDecorHandle<xAOD::EventInfo, int> PV2_count(m_OrigNtype2, ctx); + SG::WriteDecorHandle<xAOD::EventInfo, int> PV3_count(m_OrigNtype3, ctx); + SG::WriteDecorHandle<xAOD::EventInfo, int> PVUnk_count(m_OrigNtypeUnknown, ctx); + + // now count + constexpr int nvtypes = 5; + int nvtc[] = {0, 0, 0, 0, 0}; + for (auto vtx : *vertices) { + VxType::VertexType vt = vtx->vertexType(); + if ( vt >=0 && vt < nvtypes ) { + nvtc[vt]++; // vertex types 0 - 3 + } else { + nvtc[nvtypes-1]++; // unknown + } + } + PV0_count(0) = nvtc[0]; + PV1_count(0) = nvtc[1]; + PV2_count(0) = nvtc[2]; + PV3_count(0) = nvtc[3]; + PVUnk_count(0) = nvtc[4]; + } // m_addPVCountsByType + + // decorate PVs with track counts + // (needed if track collection will be thinned) + if ( m_addNTracksToPVs ) { + SG::WriteDecorHandle<xAOD::VertexContainer, int> d_nPVTracks(m_d_nPVTracks, ctx); + for (auto vtx : *vertices) { + d_nPVTracks(*vtx) = (int)vtx->nTrackParticles(); + } + } // m_addNTracksToPVs + + // decorate PVs with sqrt(sum(pt^2)) of tracks + // (needed if track collection will be thinned) + if ( m_addSqrtPt2SumToPVs ) { + SG::WriteDecorHandle<xAOD::VertexContainer, float> d_pvSqrtPt2Sum(m_OrigSqrtPt2Sum, ctx); + for (auto vtx : *vertices) { + float sqrtPt2Sum(0.); + for (auto tp : vtx->trackParticleLinks()) { + sqrtPt2Sum += std::sqrt(pow((*tp)->pt(),2)); + } + d_pvSqrtPt2Sum(*vtx) = sqrtPt2Sum; + } + } // m_addSqrtPt2SumToPVs + } + + if(!m_TrackContainername.empty()){ + SG::ReadHandle<xAOD::TrackParticleContainer> tracks(m_TrackContainername, ctx); + SG::WriteDecorHandle<xAOD::EventInfo, int> track_count(m_OrigNTracksKeys, ctx); + track_count(0) = tracks->size(); + } + + return StatusCode::SUCCESS; + } +} diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/BMuonTrackIsoTool.cxx b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/BMuonTrackIsoTool.cxx new file mode 100644 index 00000000000..6cc8a053a1e --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/BMuonTrackIsoTool.cxx @@ -0,0 +1,448 @@ +/* + Copyright (C) 2002-2019 CERN for the benefit of the ATLAS collaboration +*/ + +//============================================================================ +// BMuonTrackIsoTool.cxx +//============================================================================ +// +// Author : Wolfgang Walkowiak <Wolfgang.Walkowiak@cern.ch.> +// Changes: +// +// Add muon track isolation information for different configurations, +// different track selections and different PV-to-SV association methods. +// +// For an usage example see BPHY8.py . +// +// Job options provided by this class: +// - MuonContainerName -- name of muon container +// - IsolationConeSizes -- List of isolation cone sizes +// - IsoTrkImpLogChi2Max -- List of maximum log(chi2) cuts for +// association of tracks to the primary +// vertex picked. +// - IsoDoTrkImpLogChi2Cut -- apply log(chi2) cuts +// 0 : don't apply log(chi2) cuts +// 1 : apply log(chi2) cuts +// 2 : apply log(chi2) cuts [former version] +// (The last two job options must +// contain the same number of elements +// as the IsolationConeSizes list.) +// +//============================================================================ +// +#include "DerivationFrameworkBPhys/BMuonTrackIsoTool.h" +#include "xAODMuon/MuonContainer.h" +#include "xAODEventInfo/EventInfo.h" +#include "xAODBPhys/BPhysHelper.h" +#include "InDetTrackSelectionTool/IInDetTrackSelectionTool.h" +#include "EventPrimitives/EventPrimitivesHelpers.h" + +#include "boost/format.hpp" +#include "TVector3.h" +#include <algorithm> +#include <sstream> +#include <string> + +namespace DerivationFramework { + + //------------------------------------------------------------------------- + // + // helper class + BMuonTrackIsoTool::MuIsoItem::MuIsoItem(std::string Name, + std::string Bname, + std::string Prefix) : + BaseItem(Name, Bname, Prefix) { + } + + BMuonTrackIsoTool::MuIsoItem::~MuIsoItem() { + } + + void BMuonTrackIsoTool::MuIsoItem::resetVals() { + vIsoValues.clear(); + vNTracks.clear(); + vMuons.clear(); + } + + void BMuonTrackIsoTool::MuIsoItem::copyVals(const BaseItem& item) { + copyVals((const MuIsoItem&)item); + } + + void BMuonTrackIsoTool::MuIsoItem::copyVals(const MuIsoItem& item) { + vIsoValues = item.vIsoValues; + vNTracks = item.vNTracks; + vMuons = item.vMuons; + } + + void BMuonTrackIsoTool::MuIsoItem::fill(double isoValue, int nTracks, + const xAOD::Muon* muon) { + vIsoValues.push_back(isoValue); + vNTracks.push_back(nTracks); + vMuons.push_back(muon); + } + + std::string BMuonTrackIsoTool::MuIsoItem::muIsoName() { + return buildName(); + } + + std::string BMuonTrackIsoTool::MuIsoItem::nTracksName() { + return buildName("Ntracks"); + } + + std::string BMuonTrackIsoTool::MuIsoItem::muLinkName() { + return buildName("", "_muLink"); + } + + //-------------------------------------------------------------------------- + BMuonTrackIsoTool::BMuonTrackIsoTool(const std::string& t, + const std::string& n, + const IInterface* p) + : BPhysVertexTrackBase(t,n,p) { + + declareInterface<DerivationFramework::IAugmentationTool>(this); + + declareProperty("MuonContainerName" , m_muonContainerName=""); + declareProperty("IsolationConeSizes" , m_isoConeSizes); + declareProperty("IsoTrkImpLogChi2Max" , m_isoTrkImpLogChi2Max); + declareProperty("IsoDoTrkImpLogChi2Cut" , m_isoDoTrkImpLogChi2Cut); + } + //-------------------------------------------------------------------------- + StatusCode BMuonTrackIsoTool::initializeHook() { + + ATH_MSG_DEBUG("BMuonTrackIsoTool::initializeHook() -- begin"); + + // check like-sized arrays + if ( m_isoConeSizes.size() != m_isoTrkImpLogChi2Max.size() || + m_isoConeSizes.size() != m_isoDoTrkImpLogChi2Cut.size() ) { + ATH_MSG_ERROR("Size mismatch of IsolationConeSizes (" + << m_isoConeSizes.size() + << "), IsoTrkImpChi2Max (" + << m_isoTrkImpLogChi2Max.size() + << ") and IsoDoTrkImpChi2Cut (" + << m_isoDoTrkImpLogChi2Cut.size() << ") lists!"); + } + + // check muon container name + if ( m_muonContainerName == "" ) { + ATH_MSG_ERROR("No muon container name provided!"); + } + + // initialize results array + initResults(); + + ATH_MSG_DEBUG("BMuonTrackIsoTool::initializeHook() -- end"); + + return StatusCode::SUCCESS; + } + //-------------------------------------------------------------------------- + StatusCode BMuonTrackIsoTool::finalizeHook() { + + ATH_MSG_DEBUG("BMuonTrackIsoTool::finalizeHook()"); + + // everything all right + return StatusCode::SUCCESS; + } + //-------------------------------------------------------------------------- + StatusCode + BMuonTrackIsoTool::addBranchesVCSetupHook(size_t ivc) const { + + ATH_MSG_DEBUG("BMuonTrackIsoTool::addBranchesVCLoopHook() -- begin"); + + ATH_MSG_DEBUG("BMuonTrackisoTool::addBranchesVCSetupHook: " + << "Vertex container index " << ivc + << " for collection " << m_vertexContainerNames[ivc] + << " with prefix " << m_branchPrefixes[ivc]); + + setResultsPrefix(m_branchPrefixes[ivc]); + + ATH_MSG_DEBUG("BMuonTrackIsoTool::addBranchesVCSetupHook() -- end"); + + // nothing to do here + return StatusCode::SUCCESS; + } + //-------------------------------------------------------------------------- + StatusCode + BMuonTrackIsoTool::addBranchesSVLoopHook(const xAOD::Vertex* vtx) const { + + ATH_MSG_DEBUG("BMuonTrackIsoTool::addBranchesSVLoopHook() -- begin"); + + // retrieve muon container + m_muons = NULL; + if ( m_muonContainerName != "" ) { + CHECK(evtStore()->retrieve(m_muons, m_muonContainerName)); + ATH_MSG_DEBUG("Found muon collection with key " << m_muonContainerName); + } + + ATH_MSG_DEBUG("BMuonTrackIsoTool::addBranchesSVLoopHook(): " + "calculate muon track isolation ..."); + CHECK(calculateValues(vtx)); + + ATH_MSG_DEBUG("BMuonTrackIsoTool::addBranchesSVLoopHook(): " + "save muon track isolation ..."); + // save the isolation values + CHECK(saveIsolation(vtx)); + + ATH_MSG_DEBUG("BMuonTrackIsoTool::addBranchesSVLoopHook() -- end"); + + // nothing to do here + return StatusCode::SUCCESS; + } + //-------------------------------------------------------------------------- + // Calculate track isolation variables -- faster method with caching + //-------------------------------------------------------------------------- + StatusCode + BMuonTrackIsoTool::calcValuesHook(const xAOD::Vertex* vtx, + const unsigned int ipv, + const unsigned int its, + const unsigned int itt) const { + + ATH_MSG_DEBUG("calcValuesHook: ipv: " << ipv + << ", its: " << its << ", itt: " << itt); + + // candidate tracks and momentum + xAOD::BPhysHelper cand(vtx); + TVector3 candP = cand.totalP(); + const xAOD::Vertex* candRefPV = cand.pv(m_pvAssocTypes[ipv]); + + MuonBag muons; + // TrackBag candMuTracks = findAllMuonIdTracksInDecay(cand, muons); + std::vector<TVector3> candMuTracks = findMuonRefTrackMomenta(cand, muons); + + TrackBag tracks = selectTracks(m_tracks, cand, ipv, its, itt); + + ATH_MSG_DEBUG("calcValuesHook: found " << muons.size() << + " muons and " << candMuTracks.size() << + " tracks from B cand; " << tracks.size() << + " tracks to check."); + + // loop over isolation cones (pt and deltaR) + unsigned int nCones = m_isoConeSizes.size(); + for (unsigned int ic = 0; ic < nCones; ++ic) { + MuIsoItem& iso = m_results[ic][its][ipv][itt]; + // reset + iso.resetVals(); + + // loop over refitted ID tracks for muons in candidate + unsigned int id(0); + // for (TrackBag::const_iterator muTrkItr = candMuTracks.begin(); + // muTrkItr != candMuTracks.end(); ++muTrkItr, ++id) { + for (id=0; id < candMuTracks.size(); ++id) { + + // make sure there was an ID track for the muon + // if ( *muTrkItr != NULL ) { + if ( candMuTracks[id].Mag() > 0. ) { + + const double& coneSize = m_isoConeSizes[ic]; + const double& logChi2Max = m_isoTrkImpLogChi2Max[ic]; + const int& doLogChi2 = m_isoDoTrkImpLogChi2Cut[ic]; + + double nTracksInCone = 0; + double ptSumInCone = 0.; + + double isoValue(-5.); + + // make sure candRefPV exists + if ( candRefPV != NULL ) { + + for (TrackBag::const_iterator trkItr = tracks.begin(); + trkItr != tracks.end(); ++trkItr) { + double deltaR = candMuTracks[id].DeltaR((*trkItr)->p4().Vect()); + if ( deltaR < coneSize ) { + double logChi2 = (doLogChi2 > 0) ? + getTrackCandPVLogChi2(*trkItr, candRefPV) : -9999.; + // next line needed exactly as is for backward validation + if ( doLogChi2 == 2 ) logChi2 = abs(logChi2); + if ( doLogChi2 == 0 || logChi2 < logChi2Max ) { + nTracksInCone++; + ptSumInCone += (*trkItr)->pt(); + } + } // deltaR + } + // calculate result + if ( ptSumInCone + candMuTracks[id].Pt() > 0. ) { + isoValue = candMuTracks[id].Pt() + / ( ptSumInCone + candMuTracks[id].Pt() ); + } + + } else { + isoValue = -10.; + } // if candRefPV != NULL + + const xAOD::Muon* muon = id < muons.size() ? muons.at(id) : NULL; + iso.fill(isoValue, nTracksInCone, muon); + } // if *muTrkItr != NULL + } // for muTrkItr + } // for ic + + return StatusCode::SUCCESS; + } + //-------------------------------------------------------------------------- + // Fill track isolation values from cache if found + //-------------------------------------------------------------------------- + bool BMuonTrackIsoTool::fastFillHook(const xAOD::Vertex* vtx, + const int ipv) const { + + ATH_MSG_DEBUG("fastFillHook: ipv: " << ipv); + + bool found(false); + + StringIntMap_t::iterator itpv = + m_pvAssocResMap.find(buildPvAssocCacheName(vtx, ipv)); + if ( itpv != m_pvAssocResMap.end() ) { + found = true; + unsigned int nCones = m_isoConeSizes.size(); + unsigned int nTrackSels = m_trackSelectionTools.size(); + unsigned int nTrackTypes = m_useTrackTypes.size(); + for (unsigned int its = 0; its < nTrackSels; ++its) { + for (unsigned int ic = 0; ic < nCones; ++ic) { + for (unsigned int itt = 0; itt < nTrackTypes; ++itt) { + m_results[ic][its][ipv][itt] + .copyVals(m_results[ic][its][itpv->second][itt]); + } // for its + } // for ic + } // for itt + } // if found + + ATH_MSG_DEBUG("fastFillHook: cache index: " + << buildPvAssocCacheName(vtx, ipv) + << ", found ? " << found + << ", ipv_ref: " + << (found ? itpv->second : -1)); + + return found; + } + //-------------------------------------------------------------------------- + StatusCode + BMuonTrackIsoTool::saveIsolation(const xAOD::Vertex* vtx) const { + + typedef ElementLink<xAOD::MuonContainer> MuonLink_t; + typedef std::vector<MuonLink_t> MuonLinkVector_t; + + unsigned int nCones = m_isoConeSizes.size(); + unsigned int nTrackSels = m_trackSelectionTools.size(); + unsigned int nPvAssocs = m_pvAssocTypes.size(); + unsigned int nTrackTypes = m_useTrackTypes.size(); + + for (unsigned int its = 0; its < nTrackSels; ++its) { + for (unsigned int ipv = 0; ipv < nPvAssocs; ++ipv) { + for (unsigned int ic = 0; ic < nCones; ++ic) { + for (unsigned int itt = 0; itt < nTrackTypes; ++itt) { + MuIsoItem result = m_results[ic][its][ipv][itt]; + SG::AuxElement::Decorator< std::vector<float> > + dv_iso_values(result.muIsoName()); + SG::AuxElement::Decorator< std::vector<int> > + dv_iso_ntracks(result.nTracksName()); + dv_iso_values(*vtx) = result.vIsoValues; + dv_iso_ntracks(*vtx) = result.vNTracks; + ATH_MSG_DEBUG("BMuonTrackIsoTool::saveIsolation() -- isobn: " + << result.muIsoName() << ", ntbn: " + << result.nTracksName()); + ATH_MSG_DEBUG("BMuonTrackIsoTool::saveIsolation() -- vertex: (" + << vtx->x() << ", " + << vtx->y() << ", " + << vtx->z() << "), N(iso): " + << result.vIsoValues.size() << ", N(nTracks): " + << result.vNTracks.size()); + MuonLinkVector_t links; + for (const xAOD::Muon* muon : result.vMuons) { + if ( muon != NULL ) { + MuonLink_t link(muon, *m_muons); + links.push_back(link); + } else { + ATH_MSG_WARNING("BMuonTrackIsoTool::saveIsolation(): " + << " *muon == NULL -- EL not saved!"); + } + } + vtx->auxdecor<MuonLinkVector_t>(result.muLinkName()) = links; + ATH_MSG_DEBUG("BMuonTrackIsoTool::saveIsolation() -- muLinks: " + << "N_saved = " << links.size() ); + } // for itt + } // for ic + } // for ipv + } // for its + + return StatusCode::SUCCESS; + } + //-------------------------------------------------------------------------- + void BMuonTrackIsoTool::setResultsPrefix(std::string prefix) const { + + ATH_MSG_DEBUG("BMuonTrackIsoTool::setResultsPrefix -- begin"); + + unsigned int nCones = m_isoConeSizes.size(); + unsigned int nTrackSels = m_trackSelectionTools.size(); + unsigned int nPvAssocs = m_pvAssocTypes.size(); + unsigned int nTrackTypes = m_useTrackTypes.size(); + + for (unsigned int its = 0; its < nTrackSels; ++its) { + for (unsigned int ipv = 0; ipv < nPvAssocs; ++ipv) { + for (unsigned int ic = 0; ic < nCones; ++ic) { + for (unsigned int itt = 0; itt < nTrackTypes; ++itt) { + m_results[ic][its][ipv][itt].setPrefix(prefix); + } // for itt + } // for ic + } // for ipv + } // for its + + ATH_MSG_DEBUG("BMuonTrackIsoTool::setResultsPrefix -- end"); + } + //-------------------------------------------------------------------------- + void BMuonTrackIsoTool::initResults() { + + unsigned int nCones = m_isoConeSizes.size(); + unsigned int nTrackSels = m_trackSelectionTools.size(); + unsigned int nPvAssocs = m_pvAssocTypes.size(); + unsigned int nTrackTypes = m_useTrackTypes.size(); + + ATH_MSG_DEBUG("BMuonTrackIsoTool::initResults -- begin"); + ATH_MSG_DEBUG("BMuonTrackIsoTool::initResults : nCones = " << nCones); + ATH_MSG_DEBUG("BMuonTrackIsoTool::initResults : nTrackSels = " + << nTrackSels); + ATH_MSG_DEBUG("BMuonTrackIsoTool::initResults : nPvAssocs = " + << nPvAssocs); + ATH_MSG_DEBUG("BMuonTrackIsoTool::initResults : nTrackTypes = " + << nTrackTypes); + m_results.resize(boost::extents[nCones][nTrackSels][nPvAssocs][nTrackTypes]); + for (unsigned int its = 0; its < nTrackSels; ++its) { + ATH_MSG_DEBUG("BMuonTrackIsoTool::initResults -- its = " << its); + for (unsigned int ipv = 0; ipv < nPvAssocs; ++ipv) { + ATH_MSG_DEBUG("BMuonTrackIsoTool::initResults -- ipv = " << ipv); + for (unsigned int ic = 0; ic < nCones; ++ic) { + ATH_MSG_DEBUG("BMuonTrackIsoTool::initResults -- ic = " << ic); + for (unsigned int itt = 0; itt < nTrackTypes; ++itt) { + ATH_MSG_DEBUG("BMuonTrackIsoTool::initResults -- itt = " << itt); + + ATH_MSG_DEBUG("BMuonTrackIsoTool::initResults :" + << m_branchBaseName << buildBranchName(ic, its, + ipv, itt)); + + m_results[ic][its][ipv][itt].setup(buildBranchName(ic, its, + ipv, itt), + m_branchBaseName); + } // for itt + } // for ic + } // for ipv + } // for its + + ATH_MSG_DEBUG("BMuonTrackIsoTool::initResults -- end"); + } + //-------------------------------------------------------------------------- + std::string BMuonTrackIsoTool::buildBranchName(unsigned int ic, + unsigned int its, + unsigned int ipv, + unsigned int itt) const { + ATH_MSG_DEBUG("BMuonTrackIsoTool::buildBranchName -- begin"); + + double coneSize = m_isoConeSizes[ic]; + double logChi2Max = m_isoTrkImpLogChi2Max[ic]; + int doLogChi2 = m_isoDoTrkImpLogChi2Cut[ic]; + + // format it nicely + boost::format f("%02d_LC%02dd%1d_%s"); + f % (int)(coneSize*10.) % (int)(logChi2Max*10.) % doLogChi2 + % buildBranchBaseName(its, ipv, itt); + + ATH_MSG_DEBUG("BMuonTrackIsoTool::buildBranchName: " << f.str()); + + return f.str(); + } + //-------------------------------------------------------------------------- +} diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/BPhysAddMuonBasedInvMass.cxx b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/BPhysAddMuonBasedInvMass.cxx new file mode 100644 index 00000000000..79a285ffcfe --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/BPhysAddMuonBasedInvMass.cxx @@ -0,0 +1,702 @@ +/* + Copyright (C) 2002-2017 CERN for the benefit of the ATLAS collaboration +*/ + +/** + * @file BPhysAddMuonBasedInvMass.cxx + * @author Wolfgang Walkowiak <wolfgang.walkowiak@cern.ch> + */ + +// includes +#include "DerivationFrameworkBPhys/BPhysAddMuonBasedInvMass.h" +#include "xAODTracking/VertexContainer.h" +#include "xAODTracking/VertexAuxContainer.h" +#include "xAODTracking/TrackParticleContainer.h" +#include "xAODMuon/MuonContainer.h" +#include "xAODBase/IParticleHelpers.h" +#include "xAODBPhys/BPhysHelper.h" +// for Amg::error(): +#include "EventPrimitives/EventPrimitivesHelpers.h" + +#include <sstream> +#include <limits> + +namespace DerivationFramework { + + //-------------------------------------------------------------------------- + BPhysAddMuonBasedInvMass::BPhysAddMuonBasedInvMass(const std::string& t, + const std::string& n, + const IInterface* p) + : AthAlgTool(t,n,p), m_trackToVertexTool("Reco::TrackToVertex") { + + declareInterface<DerivationFramework::IAugmentationTool>(this); + + // Declare branch prefix + declareProperty("BranchPrefix", m_branchPrefix = "_NONE_"); + // Necessary containers + declareProperty("VertexContainerName" , m_vertexContainerName = ""); + // track mass assignment + declareProperty("TrkMasses", m_trkMasses = std::vector<double>()); + // track-to-vertex tool + declareProperty("TrackToVertexTool" , m_trackToVertexTool); + // adjust track from muon kinematics? + declareProperty("AdjustToMuonKinematics", m_adjustToMuonKinematics = false); + // add minChi2ToAnyPV decoration + declareProperty("AddMinChi2ToAnyPVMode" , m_addMinChi2ToAnyPVMode = 0); + // name of container for primary vertices + declareProperty("PrimaryVertexContainerName", m_pvContainerName = ""); + // minimum number of tracks in PV for PV to be considered in calculation + // of minChi2MuToAnyPV variable + declareProperty("MinNTracksInPV" , m_minNTracksInPV = 0); + // list of primary vertex types to consider + declareProperty("PVTypesToConsider" , m_pvTypesToConsider = {1,3}); + // PV-to-SV association types to be considered + declareProperty("DoVertexType" , m_doVertexType = 63); + } + //-------------------------------------------------------------------------- + StatusCode BPhysAddMuonBasedInvMass::initialize() { + + ATH_MSG_DEBUG("BPhysAddMuonBasedInvMass::initialize() -- begin"); + + // candidate vertices container + if ( m_vertexContainerName == "" ) { + ATH_MSG_ERROR("No vertex container name provided!"); + } + + // TrackToVertexTool + CHECK(m_trackToVertexTool.retrieve()); + + // PV container if needed + if ( m_addMinChi2ToAnyPVMode > 0 && m_pvContainerName == "" ) { + ATH_MSG_ERROR("No primary vertex container name provided!"); + } + + // PV type list if needed + if ( m_addMinChi2ToAnyPVMode > 0 && m_pvTypesToConsider.size() == 0 ) { + ATH_MSG_ERROR("No primary vertex types to be considered provided!"); + } + + // PV-to-SV association type if needed + if ( m_addMinChi2ToAnyPVMode > 1 && m_doVertexType < 1 ) { + ATH_MSG_ERROR("No PV-to-SV association types to be considered provided!"); + } + + ATH_MSG_INFO("BPhysAddMuonBasedInvMass::initialize(): " + << "AdjustToMuonKinematics = " << m_adjustToMuonKinematics); + + ATH_MSG_INFO("BPhysAddMuonBasedInvMass::initialize(): " + << "AddMinChi2ToAnyPVMode = " << m_addMinChi2ToAnyPVMode); + + // initialize PV-to-SV association type vector + initPvAssocTypeVec(); + + ATH_MSG_DEBUG("BPhysAddMuonBasedInvMass::initialize() -- end"); + + return StatusCode::SUCCESS; + } + //-------------------------------------------------------------------------- + StatusCode BPhysAddMuonBasedInvMass::finalize() { + + ATH_MSG_DEBUG("BPhysAddMuonBasedInvMass::finalize()"); + + // everything all right + return StatusCode::SUCCESS; + } + //-------------------------------------------------------------------------- + StatusCode BPhysAddMuonBasedInvMass::addBranches() const { + + ATH_MSG_DEBUG("BPhysAddMuonBasedInvMass::addBranches() -- begin"); + + // vertex container and its auxilliary store + xAOD::VertexContainer* vtxContainer = NULL; + xAOD::VertexAuxContainer* vtxAuxContainer = NULL; + + // retrieve from StoreGate + CHECK(evtStore()->retrieve(vtxContainer , m_vertexContainerName)); + CHECK(evtStore()->retrieve(vtxAuxContainer, m_vertexContainerName+"Aux.")); + + const xAOD::VertexContainer* pvContainer = NULL; + if ( m_addMinChi2ToAnyPVMode > 0 ) { + CHECK(evtStore()->retrieve(pvContainer , m_pvContainerName)); + } + + // apply the decorations + std::string branchPrefix(""); + if ( m_branchPrefix != "" && m_branchPrefix != "_NONE_" ) { + branchPrefix = m_branchPrefix + "_"; + } + + // loop over secondary vertices + for (xAOD::VertexContainer::iterator vtxItr = vtxContainer->begin(); + vtxItr!=vtxContainer->end(); ++vtxItr) { + + xAOD::BPhysHelper vtx(*vtxItr); + + SG::AuxElement::Decorator< float > + d_mucalc_mass(branchPrefix+"MUCALC_mass"); + SG::AuxElement::Decorator< float > + d_mucalc_massErr(branchPrefix+"MUCALC_massErr"); + + // TODO: check number of muons requested! + std::pair<double,double> MuCalcCandMass = + getMuCalcMass(vtx, m_trkMasses, 2); + + // fill default values + d_mucalc_mass(**vtxItr) = MuCalcCandMass.first; + d_mucalc_massErr(**vtxItr) = MuCalcCandMass.second; + + // add MinChi2ToAnyPV information if requested + if ( m_addMinChi2ToAnyPVMode > 0 ) { + + if (m_addMinChi2ToAnyPVMode == 1) { + // w.r.t. to all PVs + SG::AuxElement::Decorator< float > + d_minChi2ToAnyPV(branchPrefix+"minLogChi2ToAnyPV"); + // fill it + d_minChi2ToAnyPV(**vtxItr) = + getMinChi2ToAnyPV(vtx, pvContainer, m_pvTypesToConsider, + m_minNTracksInPV, m_addMinChi2ToAnyPVMode, + xAOD::BPhysHelper::PV_MIN_A0); // dummy + } else if (m_addMinChi2ToAnyPVMode > 1 && m_addMinChi2ToAnyPVMode < 4) { + // skip or replace associated PVs + for (auto pvAssocType : m_pvAssocTypes) { + SG::AuxElement::Decorator< float > + d_minChi2ToAnyPV(branchPrefix+"minLogChi2ToAnyPV_" + +xAOD::BPhysHelper::pv_type_str[pvAssocType]); + // fill it + d_minChi2ToAnyPV(**vtxItr) = + getMinChi2ToAnyPV(vtx, pvContainer, m_pvTypesToConsider, + m_minNTracksInPV, m_addMinChi2ToAnyPVMode, + pvAssocType); + + } // for pvAssocType + } else { + ATH_MSG_WARNING("BPhysAddMuonBasedInvMass::addBranches():" + << " Undefined AddMinChi2ToAnyPVMode value: " + << m_addMinChi2ToAnyPVMode); + } + } // if m_addMinChi2ToAnyPVMode + } // end of loop over vertices + + // clean cache + clearAdjTpCache(); + + ATH_MSG_DEBUG("BPhysAddMuonBasedInvMass::addBranches() -- end"); + + // nothing to do here + return StatusCode::SUCCESS; + } + //-------------------------------------------------------------------------- + // Calculate invariant mass based on muon system information if available. + // + std::pair<double, double> + BPhysAddMuonBasedInvMass::getMuCalcMass(xAOD::BPhysHelper& vtx, + std::vector<double> trkMasses, + int nMuRequested) const { + + std::pair<double, double> mpe(0., -1.); + + std::pair<TrackBag, int> tracksWithMu = getTracksWithMuons(vtx); + + if ( tracksWithMu.second == nMuRequested ) { + if ( tracksWithMu.first.size() == trkMasses.size() ) { + mpe = getInvariantMassWithError(tracksWithMu.first, + trkMasses, + vtx.vtx()->position()); + } else { + ATH_MSG_WARNING("BPhysAddMuonBasedInvMass::getMuCalcMass:" + << " vector sizes disagree!" + << " tracksWithMu: " << tracksWithMu.first.size() + << " BtrkMasses: " << trkMasses.size()); + } + } else { + mpe.second = -10 - tracksWithMu.second; + ATH_MSG_DEBUG("BPhysAddMuonBasedInvMass::getMuCalcMass:" + << " muon number mismatch:" + << " tracksWithMu: " << tracksWithMu.second + << " requested: " << nMuRequested); + } + return mpe; + } + //-------------------------------------------------------------------------- + // Obtain a set of ID tracks for a set of muons + //-------------------------------------------------------------------------- + TrackBag BPhysAddMuonBasedInvMass::getIdTracksForMuons(MuonBag& muons) const { + + TrackBag muTracks; + + for (auto &muon : muons) { + if ( muon != nullptr ) { + const xAOD::TrackParticle* trk = + muon->trackParticle(xAOD::Muon::InnerDetectorTrackParticle); + if ( trk != nullptr ) { + muTracks.push_back(trk); + } else { + ATH_MSG_WARNING("BPhysAddMuonBasedInvMass::getIdTracksForMuon:" + << " no ID track for muon found."); + } + } else { + ATH_MSG_WARNING("BPhysAddMuonBasedInvMass::getIdTracksForMuon:" + << " muon pointer is NULL!"); + } + } // for muon + return muTracks; + } + //-------------------------------------------------------------------------- + // Obtain a set of tracks with muon track information if available + //-------------------------------------------------------------------------- + std::pair<TrackBag, int> + BPhysAddMuonBasedInvMass::getTracksWithMuons(xAOD::BPhysHelper& vtx) const { + + TrackBag tracksWithMu; + int nMuFound = 0; + std::vector<int> vnMuFound; + + MuonBag muons = findAllMuonsInDecay(vtx); + + if ( muons.size() > 0 ) { + for (int itrk=0; itrk<vtx.nRefTrks(); ++itrk) { + // only charged tracks are of interest + if ( abs(vtx.refTrkCharge(itrk)) > 0. ) { + const xAOD::TrackParticle* trkParticle = + (xAOD::TrackParticle*)vtx.refTrkOrigin(itrk); + for (unsigned int imu = 0; imu<muons.size(); ++imu) { + if ( vtx.refTrkOrigin(itrk) == + muons.at(imu)->trackParticle(xAOD::Muon::InnerDetectorTrackParticle) ) { + const xAOD::TrackParticle* trkMuon = + adjustTrackParticle(muons.at(imu)); + if ( trkMuon != NULL ) { + trkParticle = trkMuon; + nMuFound++; + break; + } + } + } // for imu + tracksWithMu.push_back(trkParticle); + vnMuFound.push_back(nMuFound); + } // for charged track + } // for itrk + } else { + ATH_MSG_DEBUG("BPhysAddMuonBasedInvMass::getTracksWithMuons: " + "vertex contains no muons, but " + << vtx.nRefTrks() << " refitted tracks ..."); + } + // debug output + std::string svnMuFound = "["; + for (unsigned int i=0; i<vnMuFound.size(); ++i) { + svnMuFound += std::to_string(vnMuFound[i]) + ','; + } + svnMuFound.back() = ']'; + ATH_MSG_DEBUG("BPhysAddMuonBasedInvMass::getTracksWithMuons: " + "nMuFound = " << nMuFound + << "\nvnMuFound = " << svnMuFound ); + + return std::pair<TrackBag, int>(tracksWithMu, nMuFound); + } + //-------------------------------------------------------------------------- + // adjustTrackParticle: extract primary track particle from muon + // if configured adjust pt, eta and phi of it before returning + // a pointer to it. + //-------------------------------------------------------------------------- + const xAOD::TrackParticle* BPhysAddMuonBasedInvMass + ::adjustTrackParticle(const xAOD::Muon* muon) const { + + const xAOD::TrackParticle* tp = NULL; + const xAOD::TrackParticle* org = muon->primaryTrackParticle(); + + if ( m_adjustToMuonKinematics ) { + if ( org != NULL ) { + TpMap_t::iterator it = m_adjTpCache.find(org); + if ( it != m_adjTpCache.end() ) { + // copy cached link + tp = it->second; + ATH_MSG_DEBUG("adjustTrackParticle(): from cache: tp = " << tp); + } else { + // copy object -- this does not work because of seg fault later + // xAOD::TrackParticle* newTp = new xAOD::TrackParticle(*org); + + // create new object and copy properties + xAOD::TrackParticle* newTp = new xAOD::TrackParticle(); + newTp->makePrivateStore(*org); + + // ajdust pt, eta and phi to the muon's properties + xAOD::IParticle::FourMom_t p4 = muon->p4(); + float qoverp = p4.P() > 0. ? 1./p4.P() : 10.e6; + if ( org->qOverP() < 0. ) qoverp *= -1.; + newTp->setDefiningParameters(org->d0(), org->z0(), + p4.Phi(), p4.Theta(), qoverp); + // cache new TrackParticle + m_adjTpCache[org] = newTp; + tp = newTp; + ATH_MSG_DEBUG("adjustTrackParticle(): new tp = " << tp + << " org = " << org); + } // if it != end() + } // if != NULL + } else { + // copy pointer + tp = org; + ATH_MSG_DEBUG("adjustTrackParticle(): copy: org: " << org + << " -> tp: " << tp); + } + + // debug output + if ( org != NULL ) { + ATH_MSG_DEBUG("adjustTrackParticle(): org: " << org << " (" + << org->d0() << "," << org->z0() << "," << org->phi0() + << "," << org->theta() << "," << org->qOverP() << ") pt: " + << org->pt()); + } else { + ATH_MSG_DEBUG("adjustTrackParticle(): org = NULL"); + } + if ( org != NULL ) { + ATH_MSG_DEBUG("adjustTrackParticle(): tp : " << tp << " (" + << tp->d0() << "," << tp->z0() << "," << tp->phi0() + << "," << tp->theta() << "," << tp->qOverP() << ") pt: " + << tp->pt()); + } else { + ATH_MSG_DEBUG("adjustTrackParticle(): tp = NULL"); + } + return tp; + } + //-------------------------------------------------------------------------- + // clearAdjTpCache: clear the cache of adjusted TrackParticles + //-------------------------------------------------------------------------- + void BPhysAddMuonBasedInvMass::clearAdjTpCache() const { + + for ( TpMap_t::iterator it = m_adjTpCache.begin(); it != m_adjTpCache.end(); + ++it) { + if ( it->second != NULL ) { + const_cast<xAOD::TrackParticle*>(it->second)->releasePrivateStore(); + delete(it->second); + it->second = NULL; + } + m_adjTpCache.clear(); + } + } + //-------------------------------------------------------------------------- + // findAllMuonsInDecay: returns a vector of xAOD::Muon objects found + // in this vertex and subsequent decay vertices. + // Recursively calls itself if necessary. + //-------------------------------------------------------------------------- + MuonBag BPhysAddMuonBasedInvMass::findAllMuonsInDecay(xAOD::BPhysHelper& vtx) + const { + + MuonBag muons = vtx.muons(); + + // loop over preceeding vertices + for (int ivtx = 0; ivtx < vtx.nPrecedingVertices(); ++ivtx) { + xAOD::BPhysHelper precVtx(vtx.precedingVertex(ivtx)); + MuonBag muonsForVtx = findAllMuonsInDecay(precVtx); + muons.insert(muons.end(), muonsForVtx.begin(), muonsForVtx.end()); + } + return muons; + } + //-------------------------------------------------------------------------- + // getMinChi2ToAnyPV: + // Find minimum chi2 distance of signal muons w.r.t any primary vertex + // of required types and with a minimum number of tracks cut. + // It also depends on the mode w.r.t. the treatment of the associated + // primary vertex and the type of PV-to-SV association. + // Returns this minimum chi2. + //-------------------------------------------------------------------------- + double + BPhysAddMuonBasedInvMass::getMinChi2ToAnyPV(xAOD::BPhysHelper& vtx, + const xAOD::VertexContainer* + pvContainer, + const std::vector<int>& pvtypes, + const int minNTracksInPV, + const int mode, + const xAOD::BPhysHelper::pv_type& + pvAssocType) const { + + MuonBag muons = findAllMuonsInDecay(vtx); + TrackBag tracks = getIdTracksForMuons(muons); + const xAOD::Vertex* origPV = nullptr; + const xAOD::Vertex* refPV = nullptr; + + if ( mode > 1 ) { + // need to obtain original PV + origPV = vtx.origPv(pvAssocType); + if ( origPV == nullptr ) { + ATH_MSG_WARNING("BPhysAddMuonBasedInvMass::getMinChi2ToAnyPV:" + << " origPV == NULL for pvAssocType = " + << pvAssocType); + } + if ( mode > 2 ) { + refPV = vtx.pv(pvAssocType); + if ( refPV == nullptr ) { + ATH_MSG_WARNING("BPhysAddMuonBasedInvMass::getMinChi2ToAnyPV:" + << " refPV == NULL for pvAssocType = " + << pvAssocType); + } + } + } + + double minChi2 = std::numeric_limits<double>::max(); + + for (const auto pvtx : *pvContainer) { + if ( pvtx != nullptr ) { + if ( std::find(pvtypes.begin(),pvtypes.end(),pvtx->vertexType()) + != pvtypes.end() ) { + const xAOD::Vertex* cvtx = pvtx; + // switch if PV matches original PV and replacement is requested + if ( mode > 1 && pvtx == origPV ) { + // mode 2 -- skip + switch(mode) { + case 2: // skip current PV + continue; + break; + case 3: // replace by refitted PV + if ( refPV != nullptr ) { + cvtx = refPV; + } else { + ATH_MSG_WARNING("BPhysAddMuonBasedInvMass::getMinChi2ToAnyPV:" + << " refPV == NULL!"); + continue; + } + break; + } + } + if ( (int)cvtx->nTrackParticles() >= minNTracksInPV ) { + for (auto &track : tracks) { + const Amg::Vector3D pos = cvtx->position(); + minChi2 = std::min(minChi2, getTrackPVChi2(*track, pos)); + } // for track + } // if minNTracksInPV + } // if pvTypes in pvtypes vector + } else { + ATH_MSG_WARNING("BPhysAddMuonBasedInvMass::getMinChi2ToAnyPV:" + << " pvtx == NULL!"); + } // if vtx != nullptr + } // for pvtx + + return minChi2; + } + //-------------------------------------------------------------------------- + // getTrackPVChi2: + // Calculate the chi2 ( = log((d0/d0e)^2+(z0/z0e)^2) contribution of + // a track at the position closest to the given PV. + //-------------------------------------------------------------------------- + double + BPhysAddMuonBasedInvMass::getTrackPVChi2(const xAOD::TrackParticle& track, + const Amg::Vector3D& pos) const { + + double chi2 = -100.; + + const Trk::Perigee* trkPerigee = + m_trackToVertexTool->perigeeAtVertex(track, pos); + if ( trkPerigee != NULL ) { + const AmgSymMatrix(5)* locError = trkPerigee->covariance(); + if ( locError != NULL ) { + double d0 = trkPerigee->parameters()[Trk::d0]; + double z0 = trkPerigee->parameters()[Trk::z0]; + double d0Err = Amg::error(*locError, Trk::d0); + double z0Err = Amg::error(*locError, Trk::z0); + if (fabs(d0Err) > 0. && fabs(z0Err) > 0.) { + chi2 = log( pow(d0/d0Err,2.0) + pow(z0/z0Err,2.0) ); + } else { + ATH_MSG_WARNING("BPhysAddMuonBasedInvMass::getTrackPVChi2():" + << " d0 = " << d0 << ", d0Err = " << d0Err + << ", z0 = " << z0 << ", z0Err = " << z0Err); + } + } // locError != NULL + delete trkPerigee; + trkPerigee = nullptr; + } else { + ATH_MSG_WARNING("getTrackPVChi2: Could not get perigee"); + } + + return chi2; +} + //-------------------------------------------------------------------------- + // getInvariantMassWithError: returns invariant mass and mass error given + // a set of tracks, their mass hypotheses and a reference position. + // Each track must have a separate mass hypothesis in + // the vector, and they must be in the same order as the tracks in the + // track vector. Otherwise it will go horribly wrong. + //-------------------------------------------------------------------------- + std::pair<double,double> BPhysAddMuonBasedInvMass:: + getInvariantMassWithError(TrackBag trksIn, + std::vector<double> massHypotheses, + const Amg::Vector3D& pos) const { + + std::pair<double, double> mass(0.,0.); + + // ensure there is a mass hypothesis for each track + if ( trksIn.size() == massHypotheses.size() ) { + std::vector<const xAOD::TrackParticle*>::iterator trItr = trksIn.begin(); + std::vector<const xAOD::TrackParticle*>::iterator trItrEnd =trksIn.end(); + std::vector<double>::iterator massHypItr = massHypotheses.begin(); + + double pxTmp,pyTmp,pzTmp,massTmp,eTmp; + + std::vector<TLorentzVector> trkMom; + TLorentzVector totMom; + std::vector<const Trk::Perigee*> trkPer; + + for (;trItr != trItrEnd; trItr++,massHypItr++){ + const Trk::Perigee* trkPerigee = + m_trackToVertexTool->perigeeAtVertex(*(*trItr), pos); + trkPer.push_back(trkPerigee); + if ( trkPerigee != NULL ) { + // try to get the correct momentum measurement + pxTmp = trkPerigee->momentum()[Trk::px]; + pyTmp = trkPerigee->momentum()[Trk::py]; + pzTmp = trkPerigee->momentum()[Trk::pz]; + ATH_MSG_DEBUG("getInvariantMassWithError(): pvec = (" + << pxTmp << "," << pyTmp << "," << pzTmp << ")"); + } else { + // otherwise default to this one + pxTmp = ((*trItr)->p4()).Px(); + pyTmp = ((*trItr)->p4()).Py(); + pzTmp = ((*trItr)->p4()).Pz(); + ATH_MSG_WARNING("BPhysAddMuonBasedInvMass::getInvariantMassError: " + "defaulting to simple momentum!"); + } + massTmp = *massHypItr; + eTmp = pxTmp*pxTmp+pyTmp*pyTmp+pzTmp*pzTmp+massTmp*massTmp; + eTmp = eTmp > 0. ? sqrt(eTmp) : 0.; + TLorentzVector tmpMom(pxTmp, pyTmp, pzTmp, eTmp); + trkMom.push_back(tmpMom); + totMom += tmpMom; + } + mass.first = totMom.M(); + double mErr2 = 0.; + // reset trItr + trItr = trksIn.begin(); + std::vector<TLorentzVector>::iterator tmItr = trkMom.begin(); + std::vector<const Trk::Perigee*>::iterator perItr = trkPer.begin(); + AmgVector(3) dMdP; + dMdP.setZero(); + for (; tmItr != trkMom.end(); ++tmItr, ++trItr, ++perItr) { + dMdP(0) = (totMom.E() * tmItr->Px()/tmItr->E() - totMom.Px())/totMom.M(); + dMdP(1) = (totMom.E() * tmItr->Py()/tmItr->E() - totMom.Py())/totMom.M(); + dMdP(2) = (totMom.E() * tmItr->Pz()/tmItr->E() - totMom.Pz())/totMom.M(); + if ( *perItr != NULL ) { + mErr2 += (dMdP.transpose() * getMomentumCov(*perItr) * dMdP)(0,0); + } else { + mErr2 += (dMdP.transpose() * getMomentumCov(*trItr ) * dMdP)(0,0); + } + } + mass.second = mErr2 > 0. ? sqrt(mErr2) : 0.; + // clean up + for ( perItr = trkPer.begin(); perItr != trkPer.end(); ++perItr) { + delete (*perItr); + } + } else { + ATH_MSG_WARNING("BPhysAddMuonBasedInvMass::getInvariantMassError: " + "size mismatch of tracks and mass hypotheses vectors!"); + } // if size comparison + + return mass; + } + //-------------------------------------------------------------------------- + // + // Extract the 3x3 momentum covariance matrix in (x,y,z) notation + // from the (phi, theta, qoverp) notation from a TrackParticle. + // + //-------------------------------------------------------------------------- + AmgSymMatrix(3) BPhysAddMuonBasedInvMass + ::getMomentumCov(const xAOD::TrackParticle* track) const { + + AmgSymMatrix(3) cov; + cov.setZero(); + + if ( track != NULL ) { + cov = getMomentumCov( &track->perigeeParameters() ); + } + return cov; + } + //-------------------------------------------------------------------------- + // + // Extract the 3x3 momentum covariance matrix in (x,y,z) notation + // from the (phi, theta, qoverp) notation from a Perigee. + // + //-------------------------------------------------------------------------- + AmgSymMatrix(3) BPhysAddMuonBasedInvMass + ::getMomentumCov(const Trk::Perigee* perigee) const { + + AmgSymMatrix(3) cov; + cov.setZero(); + + if ( perigee != NULL ) { + cov = getMomentumCov(perigee->parameters(), *perigee->covariance()); + } + return cov; + } + //-------------------------------------------------------------------------- + // Extract the 3x3 momentum covariance matrix in (x,y,z) notation + // from the (phi, theta, qoverp) notation from a vector of + // track parameters and the error matrix + // + // Coding ideas orignally taken from + // V0Tools::massErrorVKalVrt(...), + // Code converted from BPhysToolBox::getMomentumCov(...). + //-------------------------------------------------------------------------- + // + AmgSymMatrix(3) BPhysAddMuonBasedInvMass + ::getMomentumCov(const AmgVector(5)& pars, + const AmgSymMatrix(5)& cMatrix) const { + + AmgSymMatrix(3) cov; + cov.setZero(); + + AmgMatrix(3,3) der; + der.setZero(); + + double phi = pars[Trk::phi]; + double theta = pars[Trk::theta]; + double qoverp = pars[Trk::qOverP]; + + if ( qoverp != 0. ) { + AmgVector(3) p( cos(phi)*sin(theta)/fabs(qoverp), + sin(phi)*sin(theta)/fabs(qoverp), + cos(theta)/fabs(qoverp) ); + + // d(px,py,pz)/d(phi,theta,qoverp) + der(0,0) = - p.y(); + der(1,0) = p.x(); + der(2,0) = 0.; + der(0,1) = cos(phi) * p.z(); + der(1,1) = sin(phi) * p.z(); + der(2,1) = - sin(theta) / fabs(qoverp); + der(0,2) = - p.x()/qoverp; + der(1,2) = - p.y()/qoverp; + der(2,2) = - p.z()/qoverp; + + for (unsigned int i=0; i<3; i++) { + for (unsigned int j=0; j<3; j++) { + for (unsigned int k=0; k<3; k++) { + for (unsigned int l=0; l<3; l++) { + cov(i,j) += der(i,k)*cMatrix(k+2,l+2)*der(j,l); + } + } + } + } + + // debug output + ATH_MSG_DEBUG("BPhysAddMuonBasedInvMass::getTracksWithMuons:" + << "\nlocalErrCov:\n" + << std::setprecision(10) << cMatrix + << "\ncov:\n" + << std::setprecision(10) << cov + << "\np: " << std::setprecision(10) << p + << "\nder:\n" + << std::setprecision(10) << der); + } // if qoverp + + return cov; + } + //-------------------------------------------------------------------------- + // Initialize PV-to-SV association type vector + //-------------------------------------------------------------------------- + void BPhysAddMuonBasedInvMass::initPvAssocTypeVec() { + + m_pvAssocTypes.clear(); + for (unsigned int i=0; i<xAOD::BPhysHelper::n_pv_types; ++i) { + if ( (m_doVertexType & (1 << i)) > 0 ) + m_pvAssocTypes.push_back((xAOD::BPhysHelper::pv_type)i); + } + } + //-------------------------------------------------------------------------- +} diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/BPhysConversionFinder.cxx b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/BPhysConversionFinder.cxx new file mode 100644 index 00000000000..de43d2bf118 --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/BPhysConversionFinder.cxx @@ -0,0 +1,589 @@ +/* + Copyright (C) 2002-2019 CERN for the benefit of the ATLAS collaboration +*/ +///////////////////////////////////////////////////////////////// +// BPhysConversionFinder.cxx, (c) ATLAS Detector software +/////////////////////////////////////////////////////////////////// +// Author: A. Chisholm <andrew.chisholm@cern.ch> +#include "DerivationFrameworkBPhys/BPhysConversionFinder.h" +#include "xAODTracking/VertexContainer.h" +#include "xAODTracking/VertexAuxContainer.h" +#include "TrkVertexAnalysisUtils/V0Tools.h" +#include "GaudiKernel/IPartPropSvc.h" +#include "GeoPrimitives/GeoPrimitivesHelpers.h" +#include "TrkVertexFitterInterfaces/IVertexFitter.h" +#include "TrkVKalVrtFitter/TrkVKalVrtFitter.h" + +namespace DerivationFramework { + + BPhysConversionFinder::BPhysConversionFinder(const std::string& t, + const std::string& n, + const IInterface* p) : + AthAlgTool(t,n,p), + m_v0Tools("Trk::V0Tools"), + m_vertexFitter("Trk::TrkVKalVrtFitter"), + m_vertexEstimator("InDet::VertexPointEstimator"), + m_distanceTool("Trk::SeedNewtonDistanceFinder/InDetConversionTrkDistanceFinder"), + m_postSelector("InDet::ConversionPostSelector"), + m_cascadeFitter("Trk::TrkVKalVrtFitter"), + m_inputTrackParticleContainerName("InDetTrackParticles"), + m_conversionContainerName("BPhysConversionCandidates"), + m_maxDistBetweenTracks(10.0), + m_maxDeltaCotTheta(0.3), + m_requireDeltaM(true), + m_maxDeltaM(3000.0) + { + declareInterface<DerivationFramework::IAugmentationTool>(this); + + // Declare user-defined properties + declareProperty("DiMuonVertexContainer", m_diMuonCollectionToCheck); + declareProperty("PassFlagsToCheck", m_passFlagsToCheck); + declareProperty("V0Tools", m_v0Tools); + declareProperty("VertexFitterTool", m_vertexFitter); + declareProperty("VertexEstimator", m_vertexEstimator); + declareProperty("DistanceTool", m_distanceTool); + declareProperty("ConversionPostSelector", m_postSelector); + declareProperty("CascadeFitter", m_cascadeFitter); + declareProperty("InputTrackParticleContainerName", m_inputTrackParticleContainerName); + declareProperty("ConversionContainerName", m_conversionContainerName); + declareProperty("MaxDistBetweenTracks", m_maxDistBetweenTracks = 10.0); // Maximum allowed distance of minimum approach + declareProperty("MaxDeltaCotTheta", m_maxDeltaCotTheta = 0.3); // Maximum allowed dCotTheta between tracks + declareProperty("RequireDeltaM", m_requireDeltaM = true); // Only save a conversions if it's a chi_c,b candidate (must then pass "MaxDeltaM" requirement), if "False" all conversions in the event will be saved + declareProperty("MaxDeltaM", m_maxDeltaM = 3000.0); // Maximum mass difference between di-muon+conversion and di-muon + + } + + // * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * + + StatusCode BPhysConversionFinder::initialize() + { + + ATH_MSG_DEBUG("in initialize()"); + + ATH_CHECK( m_v0Tools.retrieve() ); + ATH_CHECK( m_vertexFitter.retrieve() ); + ATH_CHECK( m_vertexEstimator.retrieve() ); + ATH_CHECK( m_distanceTool.retrieve() ); + ATH_CHECK( m_postSelector.retrieve() ); + ATH_CHECK( m_cascadeFitter.retrieve() ); + + return StatusCode::SUCCESS; + + } + + // * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * + + StatusCode BPhysConversionFinder::finalize() + { + // everything all right + return StatusCode::SUCCESS; + } + + // * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * + + StatusCode BPhysConversionFinder::addBranches() const + { + + int nTrackPairs_Init = 0; + int nTrackPairs_Selected = 0; + int nConv_VertexFit = 0; + int nConv_Selected = 0; + int nConv_Selected_DeltaM = 0; + + std::vector<const xAOD::Vertex*> oniaVertices; + oniaVertices.clear(); + + //------------------------------------ + // Look for di-muons + //------------------------------------ + const xAOD::VertexContainer* diMuonContainer = NULL; + ATH_CHECK( evtStore()->retrieve(diMuonContainer, m_diMuonCollectionToCheck) ); + + if(diMuonContainer->size() == 0) { + + ATH_MSG_DEBUG("Vertex Container (" << m_diMuonCollectionToCheck << ") is empty"); + + } else { + + ATH_MSG_DEBUG("Vertex Container (" << m_diMuonCollectionToCheck << ") contains " << diMuonContainer->size() << " vertices"); + + for(xAOD::VertexContainer::const_iterator vtxItr = diMuonContainer->begin(); vtxItr != diMuonContainer->end(); ++vtxItr) { + + const xAOD::Vertex* vertex = (*vtxItr); + + bool passedHypothesis = false; + + for(const auto &flag : m_passFlagsToCheck) { + bool pass = vertex->auxdata<Char_t>(flag); + if(pass) passedHypothesis = true; + } + + if(passedHypothesis) { + oniaVertices.push_back(vertex); + } + + } + } + //------------------------------------ + + // Output conversion container + std::unique_ptr<xAOD::VertexContainer> conversionContainer( new xAOD::VertexContainer() ); + std::unique_ptr<xAOD::VertexAuxContainer> conversionAuxContainer( new xAOD::VertexAuxContainer() ); + conversionContainer->setStore(conversionAuxContainer.get()); + + // Only call conversion finder if we've found a di-muon candidate or + // we really want to look for conversions independently + const bool callConvFinder = !m_requireDeltaM || oniaVertices.size() > 0; + + if(callConvFinder) { + + // Retrieve track particles from StoreGate + const xAOD::TrackParticleContainer* inputTrackParticles = NULL; + ATH_CHECK( evtStore()->retrieve(inputTrackParticles,m_inputTrackParticleContainerName) ); + + ATH_MSG_DEBUG("Track particle container size " << inputTrackParticles->size()); + + // Track Selection + std::vector<const xAOD::TrackParticle*> posTracks; posTracks.clear(); + std::vector<const xAOD::TrackParticle*> negTracks; negTracks.clear(); + + // Track Loop + for(xAOD::TrackParticleContainer::const_iterator trkItr = inputTrackParticles->begin(); trkItr != inputTrackParticles->end(); ++trkItr) { + + const xAOD::TrackParticle* track = (*trkItr); + + uint8_t nSCT(0); + uint8_t nPIX(0); + + track->summaryValue(nPIX,xAOD::numberOfPixelHits); + track->summaryValue(nSCT,xAOD::numberOfSCTHits); + + // Don't want TRT-only tracks + // Require Si hits on all tracks + if( nSCT + nPIX < 1 ) continue; + + if( track->charge() > 0.0) { + posTracks.push_back(track); + } else { + negTracks.push_back(track); + } + + } // Track Loop + + ATH_MSG_DEBUG(posTracks.size() + negTracks.size() << " tracks pass pre-selection"); + + std::vector<const xAOD::TrackParticle*>::const_iterator tpIt1; + std::vector<const xAOD::TrackParticle*>::const_iterator tpIt2; + + // Pos Track Loop + for(tpIt1 = posTracks.begin(); tpIt1 != posTracks.end(); ++tpIt1) { + + const xAOD::TrackParticle* trackParticle1 = (*tpIt1); + + const Trk::Perigee& trackPerigee1 = trackParticle1->perigeeParameters(); + + // Neg Track Loop + for(tpIt2 = negTracks.begin(); tpIt2 != negTracks.end(); ++tpIt2) { + + if (*tpIt1 == *tpIt2) continue; + + const xAOD::TrackParticle* trackParticle2 = (*tpIt2); + + const Trk::Perigee& trackPerigee2 = trackParticle2->perigeeParameters(); + + nTrackPairs_Init++; + + //------------------------------------ + // Track pair selection + //------------------------------------ + const double deltaCotTheta = fabs(1./tan(trackPerigee1.parameters()[Trk::theta]) - 1./tan(trackPerigee2.parameters()[Trk::theta])); + if(deltaCotTheta > m_maxDeltaCotTheta) continue; + + double distance = 1000000.; + std::optional<std::pair<Amg::Vector3D,Amg::Vector3D>> result = m_distanceTool->CalculateMinimumDistance(trackParticle1->perigeeParameters(),trackParticle2->perigeeParameters() ); + bool gotDistance = result.has_value(); + if(gotDistance) distance = Amg::distance (result->first, result->second); + if(!gotDistance || (distance > m_maxDistBetweenTracks)) continue; + //------------------------------------ + + //------------------------------------ + // Estimate starting point + cuts on compatiblity of tracks + //------------------------------------ + int sflag = 0; + int errorcode = 0; + std::map<std::string, float> vertexOutput; + Amg::Vector3D startingPoint = m_vertexEstimator->getCirclesIntersectionPoint(&trackPerigee1,&trackPerigee2,sflag,errorcode, vertexOutput); + if(errorcode != 0) continue; + //------------------------------------ + + nTrackPairs_Selected++; + + std::vector<const xAOD::TrackParticle*> trackPair; + trackPair.clear(); + trackPair.push_back(trackParticle1); + trackPair.push_back(trackParticle2); + + // Do the vertex fit + std::unique_ptr<xAOD::Vertex> convVertexCandidate( m_vertexFitter->fit(trackPair, startingPoint) ); + + // Check for successful fit + if(convVertexCandidate != NULL) { + + ATH_MSG_DEBUG("Vertex Fit Succeeded"); + + convVertexCandidate->clearTracks(); + ElementLink<xAOD::TrackParticleContainer> newLink1; + newLink1.setElement(*tpIt1); + newLink1.setStorableObject(*inputTrackParticles); + ElementLink<xAOD::TrackParticleContainer> newLink2; + newLink2.setElement(*tpIt2); + newLink2.setStorableObject(*inputTrackParticles); + convVertexCandidate->addTrackAtVertex(newLink1); + convVertexCandidate->addTrackAtVertex(newLink2); + + nConv_VertexFit++; + + //------------------------------------ + // Post-vertexing cuts + //------------------------------------ + + // This is empty and only present for compatiblity. + // The cut this informtion pertains to is not used for Si-Si conversions so this is OK + std::vector<Amg::Vector3D> positionList; + + // Apply Si-Si converion post-selection + if( !m_postSelector->selectConversionCandidate(convVertexCandidate.get(),0,positionList) ) { + convVertexCandidate.reset(); + continue; + } + //------------------------------------ + + nConv_Selected++; + + // Get photon momentum 3-vector + const xAOD::Vertex * constConvVertex = convVertexCandidate.get(); + Amg::Vector3D momentum = m_v0Tools->V0Momentum(constConvVertex); + + TLorentzVector photon; + photon.SetXYZM(momentum.x(),momentum.y(),momentum.z(),0.0); + + //------------------------------------ + // Check if conversion is consistent with a chi_c,b candidate + // by requiring a small mass difference w.r.t. any di-muon in event + //------------------------------------ + bool passDeltaM = false; + + // Use to keep track of which dimuon(s) gave a chi_c/b candidate + std::vector<const xAOD::Vertex*> candidateOniaVertices; + candidateOniaVertices.clear(); + + for ( std::vector<const xAOD::Vertex*>::const_iterator vtxItr = oniaVertices.begin(); vtxItr != oniaVertices.end(); ++vtxItr ) { + + const xAOD::Vertex* oniaVertex = (*vtxItr); + + std::vector<float> diMuon_Px = oniaVertex->auxdata< std::vector<float> >("RefTrackPx"); + std::vector<float> diMuon_Py = oniaVertex->auxdata< std::vector<float> >("RefTrackPy"); + std::vector<float> diMuon_Pz = oniaVertex->auxdata< std::vector<float> >("RefTrackPz"); + + TLorentzVector muon1, muon2; + muon1.SetXYZM(diMuon_Px.at(0),diMuon_Py.at(0),diMuon_Pz.at(0),105.658); + muon2.SetXYZM(diMuon_Px.at(1),diMuon_Py.at(1),diMuon_Pz.at(1),105.658); + + TLorentzVector diMuon = muon1 + muon2; + + const double deltaM = (diMuon+photon).M() - diMuon.M(); + + ATH_MSG_DEBUG("Candidate DeltaM = " << deltaM << " MeV DiMuon " << oniaVertex->index() << " ( Mass = " << diMuon.M() << " MeV )"); + + // Did we find a one di-muon + photon candidate with a mass diff. consistent with chi_c/b? + if(deltaM < m_maxDeltaM) { + passDeltaM = true; + candidateOniaVertices.push_back(oniaVertex); + } + + } + + // Only keep the conversion candidate if it's consistent with a chi_c,b decay + if(m_requireDeltaM && !passDeltaM) { + convVertexCandidate.reset(); + continue; + } + //------------------------------------ + + //------------------------------------ + // Final conversion candidates + //------------------------------------ + nConv_Selected_DeltaM++; + + // Keep track of which dimuon(s) gave a chi_c/b candidate + std::vector< ElementLink<xAOD::VertexContainer> > diMuonLinks; + diMuonLinks.clear(); + + // Output of cascade fits with various di-muon mass hypotheses + std::vector<float> fit_Psi1S_Px, fit_Psi1S_Py, fit_Psi1S_Pz, fit_Psi1S_M, fit_Psi1S_ChiSq; + std::vector<float> fit_Psi2S_Px, fit_Psi2S_Py, fit_Psi2S_Pz, fit_Psi2S_M, fit_Psi2S_ChiSq; + std::vector<float> fit_Upsi1S_Px, fit_Upsi1S_Py, fit_Upsi1S_Pz, fit_Upsi1S_M, fit_Upsi1S_ChiSq; + std::vector<float> fit_Upsi2S_Px, fit_Upsi2S_Py, fit_Upsi2S_Pz, fit_Upsi2S_M, fit_Upsi2S_ChiSq; + std::vector<float> fit_Upsi3S_Px, fit_Upsi3S_Py, fit_Upsi3S_Pz, fit_Upsi3S_M, fit_Upsi3S_ChiSq; + + // Loop over di-muon vertices associated with a candidate + for(std::vector<const xAOD::Vertex*>::const_iterator vtxItr = candidateOniaVertices.begin(); vtxItr != candidateOniaVertices.end(); ++vtxItr ) { + + //------------------------------------ + // Add an element link to each dimuon which formed a + // candidate, leading to the decision to save this conversion + //------------------------------------ + ElementLink<xAOD::VertexContainer> myLink; + myLink.setElement(*vtxItr); + myLink.setStorableObject(*diMuonContainer); + + if(!myLink.isValid()) { + ATH_MSG_WARNING("Invalid DiMuon ElementLink!"); + } + + diMuonLinks.push_back(myLink); + //------------------------------------ + + // Check which mass window this di-muon passed + bool passed_Psi = (*vtxItr)->auxdata<Char_t>("passed_Psi"); + bool passed_Upsi = (*vtxItr)->auxdata<Char_t>("passed_Upsi"); + + //------------------------------------ + // Cascade fit with J/psi mass hypothesis + //------------------------------------ + float fitChiSq_Psi1S = 99999; + TLorentzVector fitResult_Psi1S; + + // Only bother with the fit if di-muon mass is within the relveant range, + // but still fill an dummy 4-vector to preserve one to one correspondance with "DiMuonLinks" + if(passed_Psi) { + ATH_CHECK( doCascadeFit(*vtxItr,constConvVertex,3096.916,fitResult_Psi1S,fitChiSq_Psi1S) ); + } + + fit_Psi1S_Px.push_back(fitResult_Psi1S.Px()); + fit_Psi1S_Py.push_back(fitResult_Psi1S.Py()); + fit_Psi1S_Pz.push_back(fitResult_Psi1S.Pz()); + fit_Psi1S_M.push_back(fitResult_Psi1S.M()); + fit_Psi1S_ChiSq.push_back(fitChiSq_Psi1S); + + //------------------------------------ + // Cascade fit with psi(2S) mass hypothesis + //------------------------------------ + float fitChiSq_Psi2S = 99999; + TLorentzVector fitResult_Psi2S; + + // Only bother with the fit if di-muon mass is within the relveant range, + // but still fill an dummy 4-vector to preserve one to one correspondance with "DiMuonLinks" + if(passed_Psi) { + ATH_CHECK( doCascadeFit(*vtxItr,constConvVertex,3686.097,fitResult_Psi2S,fitChiSq_Psi2S) ); + } + + fit_Psi2S_Px.push_back(fitResult_Psi2S.Px()); + fit_Psi2S_Py.push_back(fitResult_Psi2S.Py()); + fit_Psi2S_Pz.push_back(fitResult_Psi2S.Pz()); + fit_Psi2S_M.push_back(fitResult_Psi2S.M()); + fit_Psi2S_ChiSq.push_back(fitChiSq_Psi2S); + + //------------------------------------ + // Cascade fit with Upsi(1S) mass hypothesis + //------------------------------------ + float fitChiSq_Upsi1S = 99999; + TLorentzVector fitResult_Upsi1S; + + // Only bother with the fit if di-muon mass is within the relveant range, + // but still fill an dummy 4-vector to preserve one to one correspondance with "DiMuonLinks" + if(passed_Upsi) { + ATH_CHECK( doCascadeFit(*vtxItr,constConvVertex,9460.30,fitResult_Upsi1S,fitChiSq_Upsi1S) ); + } + + fit_Upsi1S_Px.push_back(fitResult_Upsi1S.Px()); + fit_Upsi1S_Py.push_back(fitResult_Upsi1S.Py()); + fit_Upsi1S_Pz.push_back(fitResult_Upsi1S.Pz()); + fit_Upsi1S_M.push_back(fitResult_Upsi1S.M()); + fit_Upsi1S_ChiSq.push_back(fitChiSq_Upsi1S); + + //------------------------------------ + // Cascade fit with Upsi(2S) mass hypothesis + //------------------------------------ + float fitChiSq_Upsi2S = 99999; + TLorentzVector fitResult_Upsi2S; + + // Only bother with the fit if di-muon mass is within the relveant range, + // but still fill an dummy 4-vector to preserve one to one correspondance with "DiMuonLinks" + if(passed_Upsi) { + ATH_CHECK( doCascadeFit(*vtxItr,constConvVertex,10023.26,fitResult_Upsi2S,fitChiSq_Upsi2S) ); + } + + fit_Upsi2S_Px.push_back(fitResult_Upsi2S.Px()); + fit_Upsi2S_Py.push_back(fitResult_Upsi2S.Py()); + fit_Upsi2S_Pz.push_back(fitResult_Upsi2S.Pz()); + fit_Upsi2S_M.push_back(fitResult_Upsi2S.M()); + fit_Upsi2S_ChiSq.push_back(fitChiSq_Upsi2S); + + //------------------------------------ + // Cascade fit with Upsi(3S) mass hypothesis + //------------------------------------ + float fitChiSq_Upsi3S = 99999; + TLorentzVector fitResult_Upsi3S; + + // Only bother with the fit if di-muon mass is within the relveant range, + // but still fill an dummy 4-vector to preserve one to one correspondance with "DiMuonLinks" + if(passed_Upsi) { + ATH_CHECK( doCascadeFit(*vtxItr,constConvVertex,10355.2,fitResult_Upsi3S,fitChiSq_Upsi3S) ); + } + + fit_Upsi3S_Px.push_back(fitResult_Upsi3S.Px()); + fit_Upsi3S_Py.push_back(fitResult_Upsi3S.Py()); + fit_Upsi3S_Pz.push_back(fitResult_Upsi3S.Pz()); + fit_Upsi3S_M.push_back(fitResult_Upsi3S.M()); + fit_Upsi3S_ChiSq.push_back(fitChiSq_Upsi3S); + + } + + //------------------------------------ + // Decorate selected conversions + //------------------------------------ + ATH_MSG_DEBUG("Decorating conversion vertices"); + + convVertexCandidate->auxdata< std::vector< ElementLink<xAOD::VertexContainer> > >("DiMuonLinks") = diMuonLinks; + + convVertexCandidate->auxdata< std::vector<float> >("CascadeFit_Psi1S_Px") = fit_Psi1S_Px; + convVertexCandidate->auxdata< std::vector<float> >("CascadeFit_Psi1S_Py") = fit_Psi1S_Py; + convVertexCandidate->auxdata< std::vector<float> >("CascadeFit_Psi1S_Pz") = fit_Psi1S_Pz; + convVertexCandidate->auxdata< std::vector<float> >("CascadeFit_Psi1S_M") = fit_Psi1S_M; + convVertexCandidate->auxdata< std::vector<float> >("CascadeFit_Psi1S_ChiSq") = fit_Psi1S_ChiSq; + + convVertexCandidate->auxdata< std::vector<float> >("CascadeFit_Psi2S_Px") = fit_Psi2S_Px; + convVertexCandidate->auxdata< std::vector<float> >("CascadeFit_Psi2S_Py") = fit_Psi2S_Py; + convVertexCandidate->auxdata< std::vector<float> >("CascadeFit_Psi2S_Pz") = fit_Psi2S_Pz; + convVertexCandidate->auxdata< std::vector<float> >("CascadeFit_Psi2S_M") = fit_Psi2S_M; + convVertexCandidate->auxdata< std::vector<float> >("CascadeFit_Psi2S_ChiSq") = fit_Psi2S_ChiSq; + + convVertexCandidate->auxdata< std::vector<float> >("CascadeFit_Upsi1S_Px") = fit_Upsi1S_Px; + convVertexCandidate->auxdata< std::vector<float> >("CascadeFit_Upsi1S_Py") = fit_Upsi1S_Py; + convVertexCandidate->auxdata< std::vector<float> >("CascadeFit_Upsi1S_Pz") = fit_Upsi1S_Pz; + convVertexCandidate->auxdata< std::vector<float> >("CascadeFit_Upsi1S_M") = fit_Upsi1S_M; + convVertexCandidate->auxdata< std::vector<float> >("CascadeFit_Upsi1S_ChiSq") = fit_Upsi1S_ChiSq; + + convVertexCandidate->auxdata< std::vector<float> >("CascadeFit_Upsi2S_Px") = fit_Upsi2S_Px; + convVertexCandidate->auxdata< std::vector<float> >("CascadeFit_Upsi2S_Py") = fit_Upsi2S_Py; + convVertexCandidate->auxdata< std::vector<float> >("CascadeFit_Upsi2S_Pz") = fit_Upsi2S_Pz; + convVertexCandidate->auxdata< std::vector<float> >("CascadeFit_Upsi2S_M") = fit_Upsi2S_M; + convVertexCandidate->auxdata< std::vector<float> >("CascadeFit_Upsi2S_ChiSq") = fit_Upsi2S_ChiSq; + + convVertexCandidate->auxdata< std::vector<float> >("CascadeFit_Upsi3S_Px") = fit_Upsi3S_Px; + convVertexCandidate->auxdata< std::vector<float> >("CascadeFit_Upsi3S_Py") = fit_Upsi3S_Py; + convVertexCandidate->auxdata< std::vector<float> >("CascadeFit_Upsi3S_Pz") = fit_Upsi3S_Pz; + convVertexCandidate->auxdata< std::vector<float> >("CascadeFit_Upsi3S_M") = fit_Upsi3S_M; + convVertexCandidate->auxdata< std::vector<float> >("CascadeFit_Upsi3S_ChiSq") = fit_Upsi3S_ChiSq; + + convVertexCandidate->auxdata<float>("px") = momentum.x(); + convVertexCandidate->auxdata<float>("py") = momentum.y(); + convVertexCandidate->auxdata<float>("pz") = momentum.z(); + + convVertexCandidate->auxdata<float>("deltaCotThetaTrk") = deltaCotTheta; + convVertexCandidate->auxdata<float>("minimumDistanceTrk") = distance; + + convVertexCandidate->auxdata<float>("deltaPhiTracks") = vertexOutput["deltaPhiTracks"]; + convVertexCandidate->auxdata<float>("DR1R2") = vertexOutput["DR1R2"]; + + convVertexCandidate->auxdata<Char_t>("passed") = true; // Used in event skimming + + conversionContainer->push_back(convVertexCandidate.release()); + + } else { + ATH_MSG_DEBUG("Vertex Fit Failed"); + } + + } // Neg Track Loop + + } // Pos Track Loop + + } // callConvFinder + + // Write the results to StoreGate + CHECK(evtStore()->record(conversionContainer.release(), m_conversionContainerName)); + CHECK(evtStore()->record(conversionAuxContainer.release(), m_conversionContainerName+"Aux.")); + + ATH_MSG_DEBUG("-------------------------"); + ATH_MSG_DEBUG("Number of track pairs: " << nTrackPairs_Init); + ATH_MSG_DEBUG("Number of track pairs selected: " << nTrackPairs_Selected); + ATH_MSG_DEBUG("Number of successful vertex fits: " << nConv_VertexFit); + ATH_MSG_DEBUG("Number of selected vertices: " << nConv_Selected); + ATH_MSG_DEBUG("Number of selected vertices (after DeltaM req.): " << nConv_Selected_DeltaM); + + return StatusCode::SUCCESS; + } + + StatusCode BPhysConversionFinder::doCascadeFit(const xAOD::Vertex * diMuonVertex, const xAOD::Vertex * convVertex, const double diMuonMassConstraint, TLorentzVector & fitMom, float & chiSq) const { + + std::vector<const xAOD::TrackParticle*> diMuonTracks; + diMuonTracks.push_back(diMuonVertex->trackParticle(0)); + diMuonTracks.push_back(diMuonVertex->trackParticle(1)); + + std::vector<double> diMuonTrackMasses; + diMuonTrackMasses.push_back(105.658); + diMuonTrackMasses.push_back(105.658); + + std::vector<const xAOD::TrackParticle*> convTracks; + convTracks.push_back(convVertex->trackParticle(0)); + convTracks.push_back(convVertex->trackParticle(1)); + + std::vector<double> convTrackMasses; + convTrackMasses.push_back(0.511); + convTrackMasses.push_back(0.511); + + // Reset + std::unique_ptr<Trk::IVKalState> state = m_cascadeFitter->makeState(); + + // Set Robustness + m_cascadeFitter->setRobustness(0, *state); + + // Build up the topology + + // Vertex list + std::vector<Trk::VertexID> vrtList; + // V0 vertex + Trk::VertexID vID; + vID = m_cascadeFitter->startVertex(convTracks,convTrackMasses,*state,0.0); // Constrain converision mass to zero + + vrtList.push_back(vID); + + // chi_c/b vertex + Trk::VertexID vID2 = m_cascadeFitter->nextVertex(diMuonTracks,diMuonTrackMasses,vrtList,*state); + + std::vector<Trk::VertexID> cnstV; + cnstV.clear(); + if ( !m_cascadeFitter->addMassConstraint(vID2,diMuonTracks,cnstV,*state,diMuonMassConstraint).isSuccess() ) { + ATH_MSG_WARNING("addMassConstraint failed"); + } + + // Do the fit + std::unique_ptr<Trk::VxCascadeInfo> result(m_cascadeFitter->fitCascade(*state)); + + const std::vector< std::vector<TLorentzVector> > &moms = result->getParticleMoms(); + + // Check for a successful fit + if(result != NULL) { + + if(moms.size() > 2) ATH_MSG_WARNING("DoCascadeFit - More than two output momentum!?"); + + TLorentzVector conv_Fit = moms.at(0).at(0) + moms.at(0).at(1); + TLorentzVector diMuon_Fit = moms.at(1).at(0) + moms.at(1).at(1); + + // Momentum of DiMuon + photon system + fitMom = diMuon_Fit + conv_Fit; + + chiSq = result->fitChi2()/result->nDoF(); + + // Done with the fit result + result.reset(); + + } + + return StatusCode::SUCCESS; + + } + + +} diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/BPhysMetadataBase.cxx b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/BPhysMetadataBase.cxx new file mode 100644 index 00000000000..cfabb3f2356 --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/BPhysMetadataBase.cxx @@ -0,0 +1,270 @@ +/* + Copyright (C) 2002-2019 CERN for the benefit of the ATLAS collaboration +*/ + +//============================================================================ +// BPhysMetadataBase.cxx +//============================================================================ +// +// Author : Wolfgang Walkowiak <Wolfgang.Walkowiak@cern.ch.> +// Changes: +// - w.w., 2017-01-22: Added use of BPhysMetaDataTool. +// - w.w., 2017-05-27: Removed use of BPhysMetaDataTool and +// IOVDbMetaDataTool. +// - w.w., 2019-12-05: Added long and vector<long> types +// +// Store JO metadata in the output file. +// +// It uses a FileMetaData object to store job option information +// as metadata in a specific branch whose name needs to prefixed by +// the derivation format name. +// +// This is a base class. Inherit from it to add the job options you want +// to store. For a usage example, see +// Bmumu_metadata.h / Bmumu_metadata.cxx +// and +// BPHY8.py . +// +// Job options provided by the base class: +// - DerivationName -- assign the name of the derivation format +// - MetadataFolderName -- assign the name of the metadata folder, +// should start with the derivation format name, +// defaults to DerivationName if not set. +// +//============================================================================ +// + +#include "DerivationFrameworkBPhys/BPhysMetadataBase.h" +#include "xAODMetaData/FileMetaData.h" +#include "xAODMetaData/FileMetaDataAuxInfo.h" + +namespace DerivationFramework { + + //-------------------------------------------------------------------------- + BPhysMetadataBase::BPhysMetadataBase(const std::string& t, + const std::string& n, + const IInterface* p) + : AthAlgTool(t,n,p), + m_outputMetaStore("StoreGateSvc/MetaDataStore", n) { + + declareInterface<DerivationFramework::IAugmentationTool>(this); + + // Declare derivation format name + declareProperty("DerivationName", m_derivationName = "_NOSUCHFORMAT_"); + + // Declare metadata folder name (should start with derivation name) + declareProperty("MetadataFolderName", m_mdFolderName = "_NONE_"); + + // Prefix would typically be the derivation format name + declareProperty("Prefix", m_prefix = ""); + + } + //-------------------------------------------------------------------------- + StatusCode BPhysMetadataBase::initialize() { + + ATH_MSG_DEBUG("BPhysMetaDataBase::initialize() -- begin"); + + // handle general prefix + if ( m_prefix == "" ) { + if ( m_derivationName == "_NOSUCHFORMAT_" ) { + m_prefix = name() +"_"; + } else { + m_prefix = m_derivationName + "_"; + } + } + + CHECK( saveMetaDataBPhys() ); + + ATH_MSG_DEBUG("BPhysMetaDataBase::initialize() -- end"); + + return StatusCode::SUCCESS; + } + //-------------------------------------------------------------------------- + StatusCode BPhysMetadataBase::finalize() { + + ATH_MSG_DEBUG("BPhysMetaDataBase::finalize()"); + + // everything all right + return StatusCode::SUCCESS; + } + //-------------------------------------------------------------------------- + StatusCode BPhysMetadataBase::addBranches() const { + + // nothing to do here + return StatusCode::SUCCESS; + } + //-------------------------------------------------------------------------- +#define SET_VALUES_IMP( TYPE, MAP ) \ + for (auto const &ent : MAP) { \ + fm->auxdata< TYPE >( m_prefix + ent.first ) = ent.second; \ + } + + StatusCode BPhysMetadataBase::saveMetaDataBPhys() const { + + ATH_MSG_DEBUG("BPhysMetaDataBase::saveMetaDataBPhys() -- begin"); + + std::string mdFolderKey = buildFolderName() + "_MetaData"; + // protection + if ( m_outputMetaStore->contains< xAOD::FileMetaData >( mdFolderKey ) ) { + ATH_MSG_WARNING("saveMetaDataBPhys2: " + "xAOD::FileMetaData already in output: " + << mdFolderKey + << " -- BPhys metadata will NOT be saved!"); + } else { + // create a FileMetaData object + auto fm = std::make_unique< xAOD::FileMetaData >(); + auto fmAux = std::make_unique< xAOD::FileMetaDataAuxInfo >(); + fm->setStore( fmAux.get() ); + + // fill it + fm->auxdata< std::string >(m_prefix+"DerivationName" ) = + m_derivationName; + fm->auxdata< std::string >(m_prefix+"MetaDataFolderName") = + m_mdFolderName; + + // fill it with contents of maps + SET_VALUES_IMP( int , m_propInt ); + SET_VALUES_IMP( long , m_propLong ); + SET_VALUES_IMP( double , m_propDouble ); + SET_VALUES_IMP( bool , m_propBool ); + SET_VALUES_IMP( std::string , m_propString ); + SET_VALUES_IMP( std::vector<int> , m_propVInt ); + SET_VALUES_IMP( std::vector<long> , m_propVLong ); + SET_VALUES_IMP( std::vector<double> , m_propVDouble ); + SET_VALUES_IMP( std::vector<bool> , m_propVBool ); + SET_VALUES_IMP( std::vector<std::string>, m_propVString ); + + // record it + ATH_CHECK( m_outputMetaStore->record( std::move(fm), mdFolderKey ) ); + ATH_CHECK( m_outputMetaStore->record( std::move(fmAux), + mdFolderKey+"Aux." ) ); + } + + return StatusCode::SUCCESS; + } +#undef SET_VALUES_IMP + //-------------------------------------------------------------------------- + std::string BPhysMetadataBase::buildFolderName(const std::string& fname) const { + + std::string result = fname; + if ( m_mdFolderName != "_NONE_" && m_mdFolderName != "" ) { + result += m_mdFolderName; + } else { + if ( m_derivationName != "_NOSUCHFORMAT_" && m_derivationName != "" ) { + result += m_derivationName; + } else { + // default to the tool's name + result += name(); + } + } + return result; + } + //-------------------------------------------------------------------------- + void BPhysMetadataBase::recordPropertyI(const std::string& name, int val) { + ATH_MSG_INFO("Calling recordProperty(int)"); + declareProperty(name, m_propInt[name] = val); + } + //-------------------------------------------------------------------------- + void BPhysMetadataBase::recordPropertyL(const std::string& name, long val) { + ATH_MSG_INFO("Calling recordProperty(long)"); + declareProperty(name, m_propLong[name] = val); + } + //-------------------------------------------------------------------------- + void BPhysMetadataBase::recordPropertyD(const std::string& name, double val) { + ATH_MSG_INFO("Calling recordProperty(double)"); + declareProperty(name, m_propDouble[name] = val); + } + //-------------------------------------------------------------------------- + void BPhysMetadataBase::recordPropertyB(const std::string& name, bool val) { + ATH_MSG_INFO("Calling recordProperty(bool)"); + declareProperty(name, m_propBool[name] = val); + } + //-------------------------------------------------------------------------- + void BPhysMetadataBase::recordPropertyS(const std::string& name, const std::string& val) { + ATH_MSG_INFO("Calling recordProperty(string)"); + declareProperty(name, m_propString[name] = val); + } + //-------------------------------------------------------------------------- + void BPhysMetadataBase::recordPropertyVI(const std::string& name, + const std::vector<int>& val) { + ATH_MSG_INFO("Calling recordProperty(vector<int>)"); + declareProperty(name, m_propVInt[name] = val); + } + //-------------------------------------------------------------------------- + void BPhysMetadataBase::recordPropertyVL(const std::string& name, + const std::vector<long>& val) { + ATH_MSG_INFO("Calling recordProperty(vector<long>)"); + declareProperty(name, m_propVLong[name] = val); + } + //-------------------------------------------------------------------------- + void BPhysMetadataBase::recordPropertyVD(const std::string& name, + const std::vector<double>& val) { + ATH_MSG_INFO("Calling recordProperty(vector<double>)"); + declareProperty(name, m_propVDouble[name] = val); + } + //-------------------------------------------------------------------------- + void BPhysMetadataBase::recordPropertyVB(const std::string& name, + const std::vector<bool>& val) { + ATH_MSG_INFO("Calling recordProperty(vector<bool>)"); + declareProperty(name, m_propVBool[name] = val); + } + //-------------------------------------------------------------------------- + void BPhysMetadataBase::recordPropertyVS(const std::string& name, + const std::vector<std::string>& val) { + ATH_MSG_INFO("Calling recordProperty(vector<string>)"); + declareProperty(name, m_propVString[name] = val); + } + //-------------------------------------------------------------------------- + std::string BPhysMetadataBase::vecToString(const std::vector<int>& v) const { + std::string str("["); + for (unsigned int i=0; i<v.size(); ++i) { + str += std::to_string(v[i]); + if ( i < v.size()-1 ) str += ","; + } + str += "]"; + return str; + } + //-------------------------------------------------------------------------- + std::string BPhysMetadataBase::vecToString(const std::vector<long>& v) const { + std::string str("["); + for (unsigned int i=0; i<v.size(); ++i) { + str += std::to_string(v[i]); + if ( i < v.size()-1 ) str += ","; + } + str += "]"; + return str; + } + //-------------------------------------------------------------------------- + std::string BPhysMetadataBase::vecToString(const std::vector<double>& v) const { + std::string str("["); + for (unsigned int i=0; i<v.size(); ++i) { + str += std::to_string(v[i]); + if ( i < v.size()-1 ) str += ","; + } + str += "]"; + return str; + } + //-------------------------------------------------------------------------- + std::string BPhysMetadataBase::vecToString(const std::vector<bool>& v) const { + std::string str("["); + for (unsigned int i=0; i<v.size(); ++i) { + str += std::to_string(v[i]); + if ( i < v.size()-1 ) str += ","; + } + str += "]"; + return str; + } + //-------------------------------------------------------------------------- + std::string BPhysMetadataBase::vecToString(const std::vector<std::string>& v) const { + std::string str("["); + for (unsigned int i=0; i<v.size(); ++i) { + str += "'"; + str += v[i]; + str += "'"; + if ( i < v.size()-1 ) str += ","; + } + str += "]"; + return str; + } + //-------------------------------------------------------------------------- +} diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/BPhysPVCascadeTools.cxx b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/BPhysPVCascadeTools.cxx new file mode 100644 index 00000000000..09a3822d6d7 --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/BPhysPVCascadeTools.cxx @@ -0,0 +1,475 @@ +/* + Copyright (C) 2002-2017 CERN for the benefit of the ATLAS collaboration +*/ + +#include "DerivationFrameworkBPhys/BPhysPVCascadeTools.h" +#include "xAODTracking/VertexContainer.h" +#include "xAODBPhys/BPhysHelper.h" +#include "TVector3.h" +#include "DerivationFrameworkBPhys/BPhysPVTools.h" +#include "TrkVKalVrtFitter/VxCascadeInfo.h" +#include "DerivationFrameworkBPhys/LocalVector.h" +#include "JpsiUpsilonTools/PrimaryVertexRefitter.h" +#include "HepPDT/ParticleDataTable.hh" +#include "BeamSpotConditionsData/BeamSpotData.h" +#include <limits> +#include <iostream> + +DerivationFramework::BPhysPVCascadeTools::BPhysPVCascadeTools(const CascadeTools *cascadeTools) : + m_cascadeTools(cascadeTools), m_beamSpotData(nullptr), m_PV_minNTracks(0), + m_copyAllVertices(true) +{ +} + +DerivationFramework::BPhysPVCascadeTools::BPhysPVCascadeTools(const CascadeTools *cascadeTools, + const InDet::BeamSpotData* beamSpotSvc) : + m_cascadeTools(cascadeTools), m_beamSpotData(beamSpotSvc), m_PV_minNTracks(0), + m_copyAllVertices(true) +{ +} + +void DerivationFramework::BPhysPVCascadeTools::FillBPhysHelper(const std::vector<TLorentzVector> &mom, Amg::MatrixX cov, xAOD::BPhysHelper &vtx, + const xAOD::Vertex* PV, const xAOD::VertexContainer* PvContainer, + xAOD::BPhysHelper::pv_type pvtype, int refitCode) const { + + BPHYS_CHECK( vtx.setPv ( PV, PvContainer, pvtype ) ); + + // cout << "BPhysPVCascadeTools::FillBPhysHelper for pvtype = " << pvtype << endl; + // cout << "lxy " << m_cascadeTools->lxy(mom, vtx.vtx(), PV) << " error " << m_cascadeTools->lxyError(mom, cov, vtx.vtx(), PV) << endl; + + // set variables calculated from PV + BPHYS_CHECK( vtx.setLxy ( m_cascadeTools->lxy (mom, vtx.vtx(), PV), pvtype ) ); + BPHYS_CHECK( vtx.setLxyErr ( m_cascadeTools->lxyError (mom, cov, vtx.vtx(), PV), pvtype ) ); + BPHYS_CHECK( vtx.setA0 ( m_cascadeTools->a0 (mom, vtx.vtx(), PV), pvtype ) ); + BPHYS_CHECK( vtx.setA0Err ( m_cascadeTools->a0Error (mom, cov, vtx.vtx(), PV), pvtype ) ); + BPHYS_CHECK( vtx.setA0xy ( m_cascadeTools->a0xy (mom, vtx.vtx(), PV), pvtype ) ); + BPHYS_CHECK( vtx.setA0xyErr( m_cascadeTools->a0xyError (mom, cov, vtx.vtx(), PV), pvtype ) ); + BPHYS_CHECK( vtx.setZ0 ( m_cascadeTools->a0z (mom, vtx.vtx(), PV), pvtype ) ); + BPHYS_CHECK( vtx.setZ0Err ( m_cascadeTools->a0zError (mom, cov, vtx.vtx(), PV), pvtype ) ); + BPHYS_CHECK( vtx.setRefitPVStatus ( refitCode, pvtype ) ); + +} + +void DerivationFramework::BPhysPVCascadeTools::ProcessVertex(const std::vector<TLorentzVector> &mom, Amg::MatrixX cov, xAOD::BPhysHypoHelper &vtx, + xAOD::BPhysHelper::pv_type pvtype, double mass) const { + + const xAOD::Vertex* pv = vtx.pv(pvtype); + if (pv) { + // decorate the vertex. + vtx.setTau( m_cascadeTools->tau(mom, vtx.vtx(), pv), pvtype, xAOD::BPhysHypoHelper::TAU_INV_MASS ); + vtx.setTauErr( m_cascadeTools->tauError(mom, cov, vtx.vtx(), pv), pvtype, xAOD::BPhysHypoHelper::TAU_INV_MASS ); + // Proper decay time assuming constant mass hypothesis + vtx.setTau( m_cascadeTools->tau(mom, vtx.vtx(), pv, mass), pvtype, xAOD::BPhysHypoHelper::TAU_CONST_MASS ); + vtx.setTauErr( m_cascadeTools->tauError(mom, cov, vtx.vtx(), pv, mass), pvtype, xAOD::BPhysHypoHelper::TAU_CONST_MASS ); + //enum pv_type {PV_MAX_SUM_PT2, PV_MIN_A0, PV_MIN_Z0, PV_MIN_Z0_BA}; + } else { + const float errConst = -9999999.; + BPHYS_CHECK( vtx.setTau( errConst, pvtype, xAOD::BPhysHypoHelper::TAU_INV_MASS) ); + BPHYS_CHECK( vtx.setTauErr( errConst, pvtype, xAOD::BPhysHypoHelper::TAU_INV_MASS) ); + BPHYS_CHECK( vtx.setTau(errConst, pvtype, xAOD::BPhysHypoHelper::TAU_CONST_MASS) ); + BPHYS_CHECK( vtx.setTauErr( errConst, pvtype, xAOD::BPhysHypoHelper::TAU_CONST_MASS) ); + } + +} + +void DerivationFramework::BPhysPVCascadeTools::FillBPhysHelperNULL(xAOD::BPhysHelper &vtx, + const xAOD::VertexContainer* PvContainer, + xAOD::BPhysHelper::pv_type pvtype) { + DerivationFramework::BPhysPVTools::FillBPhysHelperNULL(vtx, PvContainer, pvtype); +} + +size_t DerivationFramework::BPhysPVCascadeTools::FindLowZIndex(const std::vector<TLorentzVector> &mom, const xAOD::BPhysHelper &Obj, + const std::vector<const xAOD::Vertex*> &PVlist, + const size_t PV_minNTracks) const { + size_t lowZ = 0; + if(PVlist.empty()) { + lowZ=std::numeric_limits<std::size_t>::max(); + return lowZ; + } + size_t size = PVlist.size(); + double lowA0zcalc = fabs(m_cascadeTools->a0z (mom, Obj.vtx(), PVlist[0])); + for(size_t i =1; i<size; i++) { + if ( PVlist[i]->nTrackParticles() >= PV_minNTracks ) { + double a0z = fabs(m_cascadeTools->a0z(mom, Obj.vtx(), PVlist[i])); + if(a0z < lowA0zcalc) { + lowA0zcalc = a0z; + lowZ =i; + } + } + } + return lowZ; +} + +size_t DerivationFramework::BPhysPVCascadeTools::FindHighPtIndex(const std::vector<const xAOD::Vertex*> &PVlist) { + return DerivationFramework::BPhysPVTools::FindHighPtIndex(PVlist); +} + +size_t DerivationFramework::BPhysPVCascadeTools::FindLowA0Index(const std::vector<TLorentzVector> &mom, const xAOD::BPhysHelper &Obj, + const std::vector<const xAOD::Vertex*> &PVlist, + const size_t PV_minNTracks) const { + size_t lowA0 = 0; + if(PVlist.empty()) { + lowA0=std::numeric_limits<std::size_t>::max(); + return lowA0; + } + size_t size = PVlist.size(); + double lowA0calc = m_cascadeTools->a0(mom, Obj.vtx(), PVlist[0]); + for(size_t i =1; i<size; i++) { + if ( PVlist[i]->nTrackParticles() >= PV_minNTracks ) { + double a0 = m_cascadeTools->a0(mom, Obj.vtx(), PVlist[i]); + if(a0 < lowA0calc) { + lowA0calc = a0; + lowA0 =i; + } + } + } + return lowA0; +} + +std::vector<const xAOD::Vertex*> DerivationFramework::BPhysPVCascadeTools::GetGoodPV(const xAOD::VertexContainer* pvContainer) { + typedef xAOD::VxType::VertexType VertexType; + VertexType Pvtx = xAOD::VxType::PriVtx; + VertexType Pileupvtx = xAOD::VxType::PileUp; + std::vector<const xAOD::Vertex*> goodPrimaryVertices; + goodPrimaryVertices.reserve(pvContainer->size()); + + for (auto ptr = pvContainer->begin(); ptr!= pvContainer->end(); ++ptr) { + VertexType thistype = (*ptr)->vertexType(); + if ( thistype == Pileupvtx || thistype == Pvtx ) { + goodPrimaryVertices.push_back(*ptr); + } else { +// cout << "vertex type " << thistype << endl; + } + } + return goodPrimaryVertices; +} +//----------------------------------------------------------------------------- +// +void DerivationFramework::BPhysPVCascadeTools::SetMinNTracksInPV(size_t PV_minNTracks) +{ + + m_PV_minNTracks = PV_minNTracks; +} +//----------------------------------------------------------------------------- +// +const Amg::Vector3D& DerivationFramework::BPhysPVCascadeTools::GetBeamSpot() const noexcept { + if(m_beamSpotData) return m_beamSpotData->beamPos(); + else { + static const Amg::Vector3D defaultBS(-10000.,-10000.,-10000.); + return defaultBS; + } +} +//----------------------------------------------------------------------------- +// +size_t DerivationFramework::BPhysPVCascadeTools::FindLowZ0BAIndex(const std::vector<TLorentzVector> &mom, const xAOD::BPhysHelper &obj, + const std::vector<const xAOD::Vertex*> &PVlist, + const size_t PV_minNTracks) const { + + size_t ilowZ0BA = std::numeric_limits<std::size_t>::max(); + double lowZ0BAcalc = std::numeric_limits<double>::max(); + for (size_t i = 0; i<PVlist.size(); ++i) { + if ( PVlist[i]->nTrackParticles() >= PV_minNTracks ) { + double z0BA = m_cascadeTools->a0(mom, obj.vtx(), PVlist[i]); + if (z0BA < lowZ0BAcalc) { + lowZ0BAcalc = z0BA; + ilowZ0BA = i; + } + } + } + return ilowZ0BA; +} +//----------------------------------------------------------------------------- +// +double DerivationFramework::BPhysPVCascadeTools::DistInZtoDOCA(const std::vector<TLorentzVector> &mom, const xAOD::BPhysHelper &obj, const xAOD::Vertex* vertex) const { + + Amg::Vector3D pv = vertex->position(); + Amg::Vector3D xDOCA = DocaExtrapToBeamSpot(mom, obj); + Amg::Vector3D vec = pv - xDOCA; + return vec.z(); +} +//----------------------------------------------------------------------------- +// +Amg::Vector3D DerivationFramework::BPhysPVCascadeTools::DocaExtrapToBeamSpot(const std::vector<TLorentzVector> &mom, const xAOD::BPhysHelper& obj) const { + + Amg::Vector3D xDOCA(-99999., -99999., -99999.); + TLorentzVector totalMom; + unsigned int NTrk = mom.size(); + for( unsigned int it=0; it<NTrk; it++) totalMom += mom[it]; + TVector3 totP = totalMom.Vect(); + Amg::Vector3D pSV(totP.X(), totP.Y(), totP.Z()); + Amg::Vector3D pT(pSV.x(), pSV.y(), 0.); + if ( pT.mag2() > 0 ) { + Amg::Vector3D xBS = GetBeamSpot(); + Amg::Vector3D xSV = obj.vtx()->position(); + Amg::Vector3D xT(xSV.x()-xBS.x(), xSV.y()-xBS.y(), 0.); + xDOCA = xSV - pSV*pT.dot(xT)/pT.mag2(); + } else { + std::cout << "BPhysPVCascadeTools::DocaExtrapToBeamSpot: WARNING pT == 0." + << std::endl; + } + return xDOCA; +} + +void DerivationFramework::BPhysPVCascadeTools::PrepareVertexLinks(Trk::VxCascadeInfo *result, const xAOD::TrackParticleContainer* importedTrackCollection) +{ + auto &collection = result->vertices(); + for(auto v : collection) + { + std::vector<ElementLink<DataVector<xAOD::TrackParticle> > > newLinkVector; + for(unsigned int i=0; i< v->trackParticleLinks().size(); i++) + { + ElementLink<DataVector<xAOD::TrackParticle> > mylink=v->trackParticleLinks()[i]; // makes a copy (non-const) + mylink.setStorableObject(*importedTrackCollection, true); + newLinkVector.push_back( mylink ); + } + v->clearTracks(); + v->setTrackParticleLinks( newLinkVector ); + } +} + +std::vector<const xAOD::TrackParticle*> DerivationFramework::BPhysPVCascadeTools::CollectAllChargedTracks(const std::vector<xAOD::Vertex*> &cascadeVertices) +{ + std::vector<const xAOD::TrackParticle*> exclTrk; + for( size_t jt=0; jt<cascadeVertices.size(); jt++) { + for( size_t it=0; it<cascadeVertices[jt]->vxTrackAtVertex().size(); it++) { + if(cascadeVertices[jt]->trackParticle(it)->charge() != 0) exclTrk.push_back(cascadeVertices[jt]->trackParticle(it)); + } + } + return exclTrk; +} + +StatusCode DerivationFramework::BPhysPVCascadeTools::FillCandwithRefittedVertices( bool refitPV, + const xAOD::VertexContainer* pvContainer, xAOD::VertexContainer* refPvContainer, + const Analysis::PrimaryVertexRefitter *pvRefitter, size_t in_PV_max, int DoVertexType, + Trk::VxCascadeInfo* casc, int index, + double mass, xAOD::BPhysHypoHelper &vtx) +{ + const std::vector<TLorentzVector> &mom = casc->getParticleMoms()[index]; + const Amg::MatrixX &cov = casc->getCovariance()[index]; + const std::vector<xAOD::Vertex*> &cascadeVertices = casc->vertices(); + const bool doPt = (DoVertexType & 1) != 0; + const bool doA0 = (DoVertexType & 2) != 0; + const bool doZ0 = (DoVertexType & 4) != 0; + const bool doZ0BA = (DoVertexType & 8) != 0; + + // Collect the tracks that should be excluded from the PV + std::vector<const xAOD::TrackParticle*> exclTrk = CollectAllChargedTracks(cascadeVertices); + + + const std::vector<const xAOD::Vertex*> GoodPVs = GetGoodPV(pvContainer); + // 2) PV dependent variables + if (GoodPVs.empty() == false) { + if (refitPV) { + size_t pVmax =std::min((size_t)in_PV_max, GoodPVs.size()); + std::vector<const xAOD::Vertex*> refPVvertexes; + std::vector<const xAOD::Vertex*> refPVvertexes_toDelete; + std::vector<int> exitCode; + refPVvertexes.reserve(pVmax); + refPVvertexes_toDelete.reserve(pVmax); + exitCode.reserve(pVmax); + + // Refit the primary vertex and set the related decorations. + + for (size_t i =0; i < pVmax ; i++) { + const xAOD::Vertex* oldPV = GoodPVs.at(i); + // when set to false this will return null when a new vertex is not required +// ATH_MSG_DEBUG("old PV x " << oldPV->x() << " y " << oldPV->y() << " z " << oldPV->z()); + int exitcode = 0; + const xAOD::Vertex* refPV = pvRefitter->refitVertex(oldPV, exclTrk, m_copyAllVertices, &exitcode); +// if (refPV) ATH_MSG_DEBUG("ref PV x " << refPV->x() << " y " << refPV->y() << " z " << refPV->z()); + exitCode.push_back(exitcode); + // we want positioning to match the goodPrimaryVertices + if (refPV == nullptr) { + refPVvertexes.push_back(oldPV); + refPVvertexes_toDelete.push_back(nullptr); + } else { + refPVvertexes.push_back(refPV); + refPVvertexes_toDelete.push_back(refPV); + } + } + LocalVector<size_t, 4> indexesUsed; + LocalVector<std::pair<size_t, xAOD::BPhysHelper::pv_type>, 4> indexestoProcess; + + if(doPt){ + indexestoProcess.push_back(std::make_pair + (FindHighPtIndex(refPVvertexes), xAOD::BPhysHelper::PV_MAX_SUM_PT2)); + } + if(doA0) { + indexestoProcess.push_back(std::make_pair( FindLowA0Index(mom, vtx, refPVvertexes, m_PV_minNTracks), + xAOD::BPhysHelper::PV_MIN_A0)); + } + if(doZ0) { + indexestoProcess.push_back(std::make_pair(FindLowZIndex(mom, vtx, refPVvertexes, m_PV_minNTracks), + xAOD::BPhysHelper::PV_MIN_Z0)); + } + if(doZ0BA) { + size_t lowZBA = FindLowZ0BAIndex(mom, vtx, refPVvertexes, m_PV_minNTracks); + if( lowZBA < pVmax ) { + indexestoProcess.push_back(std::make_pair(lowZBA, xAOD::BPhysHelper::PV_MIN_Z0_BA)); + } + else FillBPhysHelperNULL(vtx, pvContainer, xAOD::BPhysHelper::PV_MIN_Z0_BA); + } + + for(size_t i =0 ; i<indexestoProcess.size(); i++){ + //if refitted add to refitted container + auto index = indexestoProcess[i].first; + auto pvtype = indexestoProcess[i].second; + const xAOD::VertexContainer* ParentContainer = + (refPVvertexes_toDelete.at(index)) ? refPvContainer : pvContainer; + if(ParentContainer == refPvContainer && !indexesUsed.contains(index)) { + // store the new vertex + refPvContainer->push_back(const_cast<xAOD::Vertex*>(refPVvertexes.at(index))); + indexesUsed.push_back(index); + } + FillBPhysHelper(mom, cov, vtx, refPVvertexes[index], + ParentContainer, pvtype, exitCode[index]); + vtx.setOrigPv(GoodPVs[index], pvContainer, pvtype); + } + //nullify ptrs we want to keep so these won't get deleted + //"delete null" is valid in C++ and does nothing so this is quicker than a lot of if statements + for(size_t x : indexesUsed) refPVvertexes_toDelete[x] = nullptr; + //Loop over toDELETE container, anything that is used or was not refitted is null + //This cleans up all extra vertices that were created and not used + for(const xAOD::Vertex* ptr : refPVvertexes_toDelete) delete ptr; + refPVvertexes.clear(); // Clear lists of now dangling ptrs + refPVvertexes_toDelete.clear(); + exitCode.clear(); + + } else { + // 2.a) the first PV with the largest sum pT. + if(doPt) { + size_t highPtindex = FindHighPtIndex(GoodPVs); // Should be 0 in PV ordering + FillBPhysHelper(mom, cov, vtx, GoodPVs[highPtindex], pvContainer, xAOD::BPhysHelper::PV_MAX_SUM_PT2, 0); + } + // 2.b) the closest in 3D: + if(doA0) { + size_t lowA0 = FindLowA0Index(mom, vtx, GoodPVs, m_PV_minNTracks); + FillBPhysHelper(mom, cov, vtx, GoodPVs[lowA0], pvContainer, xAOD::BPhysHelper::PV_MIN_A0, 0); + } + // 2.c) the closest in Z: + if(doZ0) { + size_t lowZ = FindLowZIndex(mom, vtx, GoodPVs, m_PV_minNTracks); + FillBPhysHelper(mom, cov, vtx, GoodPVs[lowZ], pvContainer, xAOD::BPhysHelper::PV_MIN_Z0, 0); + } + // 2.d) the closest in Z (DOCA w.r.t. beam axis): + if(doZ0BA) { + size_t lowZBA = FindLowZ0BAIndex(mom, vtx, GoodPVs, m_PV_minNTracks); + if ( lowZBA < GoodPVs.size() ) { // safety against vector index out-of-bounds + FillBPhysHelper(mom, cov, vtx, GoodPVs[lowZBA], pvContainer, xAOD::BPhysHelper::PV_MIN_Z0_BA, 0); + } else { + // nothing found -- fill NULL + FillBPhysHelperNULL(vtx, pvContainer, xAOD::BPhysHelper::PV_MIN_Z0_BA); + } + } + } // refitPV + } else { + + if(pvContainer->empty()) return StatusCode::FAILURE; + const xAOD::Vertex* Dummy = pvContainer->at(0); + + // 2.a) the first PV with the largest sum pT. + if(doPt) { + FillBPhysHelper(mom, cov, vtx, Dummy, pvContainer, xAOD::BPhysHelper::PV_MAX_SUM_PT2, 0); + if(refitPV) vtx.setOrigPv(Dummy, pvContainer, xAOD::BPhysHelper::PV_MAX_SUM_PT2); + } + // 2.b) the closest in 3D: + if(doA0) { + FillBPhysHelper(mom, cov, vtx, Dummy, pvContainer, xAOD::BPhysHelper::PV_MIN_A0, 0); + if(refitPV) vtx.setOrigPv(Dummy, pvContainer, xAOD::BPhysHelper::PV_MIN_A0); + } + // 2.c) the closest in Z: + if(doZ0) { + FillBPhysHelper(mom, cov, vtx, Dummy, pvContainer, xAOD::BPhysHelper::PV_MIN_Z0, 0); + if(refitPV) vtx.setOrigPv(Dummy, pvContainer, xAOD::BPhysHelper::PV_MIN_Z0); + } + // 2.d) the closest in Z (DOCA w.r.t. beam axis): + if(doZ0BA) { + FillBPhysHelper(mom, cov, vtx, Dummy, pvContainer, xAOD::BPhysHelper::PV_MIN_Z0_BA, 0); + if(refitPV) vtx.setOrigPv(Dummy, pvContainer, xAOD::BPhysHelper::PV_MIN_Z0_BA); + } + } // GoodPVs.empty() + + // 3) proper decay time and error: + // retrieve the refitted PV (or the original one, if the PV refitting was turned off) + if(doPt) ProcessVertex(mom, cov, vtx, xAOD::BPhysHelper::PV_MAX_SUM_PT2, mass); + if(doA0) ProcessVertex(mom, cov, vtx, xAOD::BPhysHelper::PV_MIN_A0, mass); + if(doZ0) ProcessVertex(mom, cov, vtx, xAOD::BPhysHelper::PV_MIN_Z0, mass); + if(doZ0BA) ProcessVertex(mom, cov, vtx, xAOD::BPhysHelper::PV_MIN_Z0_BA, mass); + + return StatusCode::SUCCESS; +} + +//----------------------------------------------------------------------------- + +void DerivationFramework::BPhysPVCascadeTools::SetVectorInfo(xAOD::BPhysHelper &vtx, const Trk::VxCascadeInfo* casc){ + + const std::vector< std::vector<TLorentzVector> > &moms = casc->getParticleMoms(); + const std::vector<xAOD::Vertex*> &cascadeVertices = casc->vertices(); + // Get refitted track momenta from all vertices, charged tracks only + std::vector<float> px; + std::vector<float> py; + std::vector<float> pz; + for( size_t jt=0; jt<moms.size(); jt++) { + for( size_t it=0; it<cascadeVertices[jt]->vxTrackAtVertex().size(); it++) { + px.push_back( moms[jt][it].Px() ); + py.push_back( moms[jt][it].Py() ); + pz.push_back( moms[jt][it].Pz() ); + } + } + vtx.setRefTrks(std::move(px),std::move(py),std::move(pz)); + +} + +bool DerivationFramework::BPhysPVCascadeTools::uniqueCollection(const std::vector<const xAOD::TrackParticle*>&col){ + for(auto p : col){ + if(std::count(col.begin(), col.end(), p) > 1) return false; + } + return true; +} + +bool DerivationFramework::BPhysPVCascadeTools::uniqueCollection(const std::vector<const xAOD::TrackParticle*>&col1, const std::vector<const xAOD::TrackParticle*>&col2){ + for(auto p : col1){ + if((std::count(col1.begin(), col1.end(), p) + std::count(col2.begin(), col2.end(), p)) > 1) return false; + } + for(auto p : col2){ + if((std::count(col1.begin(), col1.end(), p) + std::count(col2.begin(), col2.end(), p)) > 1) return false; + } + return true; +} + +bool DerivationFramework::BPhysPVCascadeTools::LinkVertices(SG::AuxElement::Decorator<VertexLinkVector> &decor, const std::vector<const xAOD::Vertex*>& vertices, + const xAOD::VertexContainer* vertexContainer, const xAOD::Vertex* vert){ + // create tmp vector of preceding vertex links + VertexLinkVector precedingVertexLinks; + + // loop over input precedingVertices + auto precedingVerticesItr = vertices.begin(); + for(; precedingVerticesItr!=vertices.end(); ++precedingVerticesItr) { + // sanity check 1: protect against null pointers + if( !(*precedingVerticesItr) ) + return false; + + // create element link + VertexLink vertexLink; + vertexLink.setElement(*precedingVerticesItr); + vertexLink.setStorableObject(*vertexContainer); + + // sanity check 2: is the link valid? + if( !vertexLink.isValid() ) + return false; + + // link is OK, store it in the tmp vector + precedingVertexLinks.push_back( vertexLink ); + + } // end of loop over preceding vertices + + // all OK: store preceding vertex links in the aux store + decor(*vert) = precedingVertexLinks; + return true; +} + +double DerivationFramework::BPhysPVCascadeTools::getParticleMass(const HepPDT::ParticleDataTable* pdt, int pdgcode){ + auto ptr = pdt->particle( pdgcode ); + return ptr ? ptr->mass() : 0.; +} + diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/BPhysPVThinningTool.cxx b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/BPhysPVThinningTool.cxx new file mode 100644 index 00000000000..33e5030c830 --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/BPhysPVThinningTool.cxx @@ -0,0 +1,134 @@ +/* + Copyright (C) 2002-2017 CERN for the benefit of the ATLAS collaboration +*/ + +///////////////////////////////////////////////////////////////// +// ThinningToolExample.cxx +/////////////////////////////////////////////////////////////////// +// Author: James Catmore (James.Catmore@cern.ch) +// This is a trivial example of an implementation of a thinning tool +// which removes all ID tracks which do not pass a user-defined cut + +#include "DerivationFrameworkBPhys/BPhysPVThinningTool.h" +#include "xAODTracking/TrackParticleContainer.h" +#include "xAODTracking/TrackParticle.h" +#include "StoreGate/ThinningHandle.h" +#include "xAODBPhys/BPhysHelper.h" +#include "StoreGate/ThinningHandle.h" +#include <algorithm> +#include <numeric> +#include <vector> +#include <string> + +using namespace std; +using namespace xAOD; +// Constructor +DerivationFramework::BPhysPVThinningTool::BPhysPVThinningTool(const std::string& t, + const std::string& n, + const IInterface* p ) : + AthAlgTool(t,n,p), + m_TrackContainerName("InDetTrackParticles"), + m_PVContainerName("PrimaryVertices"), + m_ntot(0), + m_npass(0), m_tracks_kept(0), m_keepTracks(false) +{ + declareInterface<DerivationFramework::IThinningTool>(this); + declareProperty("CandidateCollections" , m_BPhyCandList); + declareProperty("KeepPVTracks", m_keepTracks); + declareProperty("TrackParticleContainerName", m_TrackContainerName); + declareProperty("PrimaryVertexContainerName", m_PVContainerName); +} + +// Destructor +DerivationFramework::BPhysPVThinningTool::~BPhysPVThinningTool() { +} + +// Athena initialize and finalize +StatusCode DerivationFramework::BPhysPVThinningTool::initialize() +{ + ATH_MSG_VERBOSE("initialize() ..."); + ATH_CHECK(m_BPhyCandList.initialize()); + if(not m_TrackContainerName.key().empty()) ATH_CHECK(m_TrackContainerName.initialize(m_streamName)); + ATH_CHECK(m_PVContainerName.initialize(m_streamName)); + + return StatusCode::SUCCESS; +} +StatusCode DerivationFramework::BPhysPVThinningTool::finalize() +{ + ATH_MSG_VERBOSE("finalize() ..."); + ATH_MSG_INFO("Processed "<< m_ntot <<" PV, "<< m_npass<< " were retained "); + if(m_keepTracks) ATH_MSG_INFO("Additional tracks kept " << m_tracks_kept); + return StatusCode::SUCCESS; +} + +// The thinning itself +StatusCode DerivationFramework::BPhysPVThinningTool::doThinning() const +{ + + // Get the track container + SG::ThinningHandle<xAOD::VertexContainer> PV_col(m_PVContainerName); + if(!PV_col.isValid()) { + ATH_MSG_ERROR ("Couldn't retrieve VertexContainer with key PrimaryVertices"); + return StatusCode::FAILURE; + } + m_ntot+=PV_col->size(); + // Loop over tracks, see if they pass, set mask + std::vector<bool> mask(PV_col->size(), false); + + BPhysHelper::pv_type pvtypes[] = {BPhysHelper::PV_MAX_SUM_PT2, + BPhysHelper::PV_MIN_A0, + BPhysHelper::PV_MIN_Z0, + BPhysHelper::PV_MIN_Z0_BA}; + + + + for(auto &str : m_BPhyCandList) { + SG::ReadHandle<xAOD::VertexContainer> Container(str); + ATH_CHECK(Container.isValid()); + size_t s = Container->size(); + for(size_t i = 0; i<s; i++) { + xAOD::BPhysHelper vtx(Container->at(i)); + + for(size_t i =0; i < 4; i++) { + const xAOD::Vertex* origPv = vtx.origPv(pvtypes[i]); + if(origPv==nullptr) continue; + auto pvit = std::find (PV_col->begin(), PV_col->end(), origPv); + if(pvit == PV_col->end()) { + ATH_MSG_WARNING("PV not found in container"); + continue; + } + size_t x = std::distance(PV_col->begin(), pvit); + mask.at(x) = true; + } + + } + } + + m_npass += std::accumulate(mask.begin(), mask.end(), 0); + + if(m_keepTracks){ + SG::ThinningHandle<xAOD::TrackParticleContainer> importedTrackParticles(m_TrackContainerName); + std::vector<bool> trackmask(importedTrackParticles->size(), false); + size_t pvnum = mask.size(); + for(size_t i =0; i<pvnum;i++){ + if(mask[i] == false) continue; + auto vtx = PV_col->at(i); + size_t s = vtx->nTrackParticles(); + for(size_t j=0;j<s;j++){ + auto trackit = std::find(importedTrackParticles->begin(), importedTrackParticles->end(), vtx->trackParticle(j)); + if(trackit == importedTrackParticles->end()){ + ATH_MSG_WARNING("track not found in container"); + continue; + } + size_t x = std::distance(importedTrackParticles->begin(), trackit); + trackmask.at(x) = true; + } + } + importedTrackParticles.keep(trackmask, SG::ThinningHandleBase::Op::Or); + m_tracks_kept += std::accumulate(trackmask.begin(), trackmask.end(), 0); + } + PV_col.keep(mask, SG::ThinningHandleBase::Op::Or); + + return StatusCode::SUCCESS; +} + diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/BPhysPVTools.cxx b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/BPhysPVTools.cxx new file mode 100644 index 00000000000..6bb75c4d31d --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/BPhysPVTools.cxx @@ -0,0 +1,545 @@ +/* + Copyright (C) 2002-2017 CERN for the benefit of the ATLAS collaboration +*/ + +#include "DerivationFrameworkBPhys/BPhysPVTools.h" +#include "xAODTracking/VertexContainer.h" +#include "TrkVertexAnalysisUtils/V0Tools.h" +#include "xAODBPhys/BPhysHelper.h" +#include "JpsiUpsilonTools/PrimaryVertexRefitter.h" +#include "TVector3.h" +#include "BeamSpotConditionsData/BeamSpotData.h" + +#include <limits> +#include <iostream> +#include <cmath> +using namespace std; + +DerivationFramework::BPhysPVTools::BPhysPVTools(const Trk::V0Tools *v0Tools) : + m_v0Tools(v0Tools), m_beamSpotData(nullptr), m_PV_minNTracks(0), + m_3dCalc(false) +{ +} + +DerivationFramework::BPhysPVTools::BPhysPVTools(const Trk::V0Tools *v0Tools, const InDet::BeamSpotData *beamSpotSvc) : + m_v0Tools(v0Tools), m_beamSpotData(beamSpotSvc), m_PV_minNTracks(0), + m_3dCalc(false) +{ +} + +void DerivationFramework::BPhysPVTools::FillBPhysHelper(xAOD::BPhysHelper &vtx, + const xAOD::Vertex* PV, const xAOD::VertexContainer* PvContainer, + xAOD::BPhysHelper::pv_type pvtype, int refitCode) const { + + BPHYS_CHECK( vtx.setPv ( PV, PvContainer, pvtype ) ); + + // cout << "BPhysPVTools::FillBPhysHelper for pvtype = " << pvtype << endl; + + // set variables calculated from PV + if(m_3dCalc){ + BPHYS_CHECK( vtx.setLxyz ( m_v0Tools->lxyz (vtx.vtx(), PV), pvtype ) ); + BPHYS_CHECK( vtx.setLxyzErr ( m_v0Tools->lxyzError (vtx.vtx(), PV), pvtype ) ); + } + BPHYS_CHECK( vtx.setLxy ( m_v0Tools->lxy (vtx.vtx(), PV), pvtype ) ); + BPHYS_CHECK( vtx.setLxyErr ( m_v0Tools->lxyError (vtx.vtx(), PV), pvtype ) ); + BPHYS_CHECK( vtx.setA0 ( m_v0Tools->a0 (vtx.vtx(), PV), pvtype ) ); + BPHYS_CHECK( vtx.setA0Err ( m_v0Tools->a0Error (vtx.vtx(), PV), pvtype ) ); + BPHYS_CHECK( vtx.setA0xy ( m_v0Tools->a0xy (vtx.vtx(), PV), pvtype ) ); + BPHYS_CHECK( vtx.setA0xyErr( m_v0Tools->a0xyError (vtx.vtx(), PV), pvtype ) ); + BPHYS_CHECK( vtx.setZ0 ( m_v0Tools->a0z (vtx.vtx(), PV), pvtype ) ); + BPHYS_CHECK( vtx.setZ0Err ( m_v0Tools->a0zError (vtx.vtx(), PV), pvtype ) ); + BPHYS_CHECK( vtx.setRefitPVStatus ( refitCode, pvtype ) ); + +} + +void DerivationFramework::BPhysPVTools::FillBPhysHelperNULL(xAOD::BPhysHelper &vtx, + const xAOD::VertexContainer* PvContainer, + xAOD::BPhysHelper::pv_type pvtype, bool do3d) { + const xAOD::Vertex* PV = nullptr; + BPHYS_CHECK( vtx.setPv ( PV, PvContainer, pvtype ) ); + constexpr float errConst = std::numeric_limits<float>::lowest(); + // set variables claculated from PV + if(do3d){ + BPHYS_CHECK( vtx.setLxyz ( errConst, pvtype ) ); + BPHYS_CHECK( vtx.setLxyzErr ( errConst, pvtype ) ); + } + BPHYS_CHECK( vtx.setLxy ( errConst, pvtype ) ); + BPHYS_CHECK( vtx.setLxyErr ( errConst, pvtype ) ); + BPHYS_CHECK( vtx.setA0 ( errConst, pvtype ) ); + BPHYS_CHECK( vtx.setA0Err ( errConst, pvtype ) ); + BPHYS_CHECK( vtx.setA0xy ( errConst, pvtype ) ); + BPHYS_CHECK( vtx.setA0xyErr( errConst, pvtype ) ); + BPHYS_CHECK( vtx.setZ0 ( errConst, pvtype ) ); + BPHYS_CHECK( vtx.setZ0Err ( errConst, pvtype ) ); + BPHYS_CHECK( vtx.setRefitPVStatus ( 0, pvtype ) ); +} + +size_t DerivationFramework::BPhysPVTools::FindLowZIndex(const xAOD::BPhysHelper &Obj, + const std::vector<const xAOD::Vertex*> &PVlist, + const size_t PV_minNTracks) const { + size_t lowZ = 0; + if(PVlist.empty()) { + lowZ=std::numeric_limits<std::size_t>::max(); + return lowZ; + } + size_t size = PVlist.size(); + double lowA0zcalc = fabs(m_v0Tools->a0z (Obj.vtx(), PVlist[0])); + for(size_t i =1; i<size; i++) { + if ( PVlist[i]->nTrackParticles() >= PV_minNTracks ) { + double a0z = fabs(m_v0Tools->a0z(Obj.vtx(), PVlist[i])); + if(a0z < lowA0zcalc) { + lowA0zcalc = a0z; + lowZ =i; + } + } + } + return lowZ; +} + +void DerivationFramework::BPhysPVTools::DecorateWithDummyVertex(xAOD::VertexContainer* vtxContainer, + const xAOD::VertexContainer* pvContainer, const xAOD::Vertex* Dummy, + int DoVertexType, const bool SetOrignal) const { + const bool doPt = (DoVertexType & 1) != 0; + const bool doA0 = (DoVertexType & 2) != 0; + const bool doZ0 = (DoVertexType & 4) != 0; + const bool doZ0BA = (DoVertexType & 8) != 0; + + xAOD::VertexContainer::iterator vtxItr = vtxContainer->begin(); + for(; vtxItr!=vtxContainer->end(); ++vtxItr) { + xAOD::BPhysHelper vtx(*vtxItr); + + // 1) pT error + double ptErr = m_v0Tools->pTError( vtx.vtx() ); + BPHYS_CHECK( vtx.setPtErr(ptErr) ); + if(doPt) { + // 2.a) the first PV with the largest sum pT. + FillBPhysHelper(vtx, Dummy, pvContainer, xAOD::BPhysHelper::PV_MAX_SUM_PT2, 0); + if(SetOrignal) vtx.setOrigPv(Dummy, pvContainer, xAOD::BPhysHelper::PV_MAX_SUM_PT2); + } + + if(doA0) { + // 2.b) the closest in 3D: + FillBPhysHelper(vtx, Dummy, pvContainer, xAOD::BPhysHelper::PV_MIN_A0, 0); + if(SetOrignal) vtx.setOrigPv(Dummy, pvContainer, xAOD::BPhysHelper::PV_MIN_A0); + } + + if(doZ0) { + FillBPhysHelper(vtx, Dummy, pvContainer, xAOD::BPhysHelper::PV_MIN_Z0, 0); + if(SetOrignal) vtx.setOrigPv(Dummy, pvContainer, xAOD::BPhysHelper::PV_MIN_Z0); + } + + if(doZ0BA) { + FillBPhysHelper(vtx, Dummy, pvContainer, xAOD::BPhysHelper::PV_MIN_Z0_BA, 0); + if(SetOrignal) vtx.setOrigPv(Dummy, pvContainer, xAOD::BPhysHelper::PV_MIN_Z0_BA); + } + } +} + +void DerivationFramework::BPhysPVTools::DecorateWithNULL(xAOD::VertexContainer* vtxContainer, + const xAOD::VertexContainer* pvContainer, int DoVertexType) const { + const bool doPt = (DoVertexType & 1) != 0; + const bool doA0 = (DoVertexType & 2) != 0; + const bool doZ0 = (DoVertexType & 4) != 0; + const bool doZ0BA = (DoVertexType & 8) != 0; + xAOD::VertexContainer::iterator vtxItr = vtxContainer->begin(); + for(; vtxItr!=vtxContainer->end(); ++vtxItr) { + xAOD::BPhysHelper vtx(*vtxItr); + + // 1) pT error + double ptErr = m_v0Tools->pTError( vtx.vtx() ); + BPHYS_CHECK( vtx.setPtErr(ptErr) ); + if(doPt) { + // 2.a) the first PV with the largest sum pT. + FillBPhysHelperNULL(vtx, pvContainer, xAOD::BPhysHelper::PV_MAX_SUM_PT2, m_3dCalc); + } + + if(doA0) { + // 2.b) the closest in 3D: + FillBPhysHelperNULL(vtx, pvContainer, xAOD::BPhysHelper::PV_MIN_A0, m_3dCalc); + } + + if(doZ0) { + FillBPhysHelperNULL(vtx, pvContainer, xAOD::BPhysHelper::PV_MIN_Z0, m_3dCalc); + } + if(doZ0BA) { + FillBPhysHelperNULL(vtx, pvContainer, xAOD::BPhysHelper::PV_MIN_Z0_BA, m_3dCalc); + } + } +} + +StatusCode DerivationFramework::BPhysPVTools::FillCandExistingVertices(xAOD::VertexContainer* vtxContainer, + const xAOD::VertexContainer* pvContainer, int DoVertexType) { + + + + //---------------------------------------------------- + // decorate the vertex + //---------------------------------------------------- + // loop over candidates -- Don't apply PV_minNTracks requirement here + // because it may result in exclusion of the high-pt PV. + const std::vector<const xAOD::Vertex*> GoodPVs = GetGoodPV(pvContainer); + + + if(GoodPVs.empty() == false) { + + const bool doPt = (DoVertexType & 1) != 0; + const bool doA0 = (DoVertexType & 2) != 0; + const bool doZ0 = (DoVertexType & 4) != 0; + const bool doZ0BA = (DoVertexType & 8) != 0; + + xAOD::VertexContainer::iterator vtxItr = vtxContainer->begin(); + for(; vtxItr!=vtxContainer->end(); ++vtxItr) { + xAOD::BPhysHelper vtx(*vtxItr); + + // 1) pT error + double ptErr = m_v0Tools->pTError( vtx.vtx() ); + BPHYS_CHECK( vtx.setPtErr(ptErr) ); + + // 2) refit the primary vertex and set the related decorations. + if(doPt) { + size_t highPtindex = FindHighPtIndex(GoodPVs); //Should be 0 in PV ordering + // 2.a) the first PV with the largest sum pT. + FillBPhysHelper(vtx, GoodPVs[highPtindex], pvContainer, xAOD::BPhysHelper::PV_MAX_SUM_PT2, 0); + } + + if(doA0) { + // 2.b) the closest in 3D: + size_t lowA0 = FindLowA0Index(vtx, GoodPVs, m_PV_minNTracks); + FillBPhysHelper(vtx, GoodPVs[lowA0], pvContainer, xAOD::BPhysHelper::PV_MIN_A0, 0); + } + + if(doZ0) { + size_t lowZ = FindLowZIndex(vtx, GoodPVs, m_PV_minNTracks); + FillBPhysHelper(vtx, GoodPVs[lowZ], pvContainer, xAOD::BPhysHelper::PV_MIN_Z0, 0); + } + + if(doZ0BA) { + size_t lowZBA = FindLowZ0BAIndex(vtx, GoodPVs, m_PV_minNTracks); + if ( lowZBA < GoodPVs.size() ) { // safety against vector index out-of-bounds + FillBPhysHelper(vtx, GoodPVs[lowZBA], pvContainer, xAOD::BPhysHelper::PV_MIN_Z0_BA, 0); + } else { + // nothing found -- fill nullptr + FillBPhysHelperNULL(vtx, pvContainer, xAOD::BPhysHelper::PV_MIN_Z0_BA, m_3dCalc); + } + } + + }// end of loop over vertices + } //end of check for vertices + else { + // cout << "Warning: DerivationFramework::BPhysPVTools::FillCandExistingVertices No Primary Vertices Found trying to decorate wilth dummy \n"; + if(pvContainer->empty()) return StatusCode::FAILURE; + const xAOD::Vertex* dummy = pvContainer->at(0); //No good vertices so last vertex must be dummy + DecorateWithDummyVertex(vtxContainer, pvContainer, dummy, DoVertexType, false); + } + return StatusCode::SUCCESS; +} + + +StatusCode DerivationFramework::BPhysPVTools::FillCandwithRefittedVertices(xAOD::VertexContainer* vtxContainer, + const xAOD::VertexContainer* pvContainer, xAOD::VertexContainer* refPvContainer, + const Analysis::PrimaryVertexRefitter *pvRefitter, size_t in_PV_max, int DoVertexType) { + + + //---------------------------------------------------- + // decorate the vertex + //---------------------------------------------------- + // loop over candidates -- Don't apply PV_minNTracks requirement here + // because it may result in exclusion of the high-pt PV. + std::vector<const xAOD::Vertex*> goodPrimaryVertices = GetGoodPV(pvContainer); + + /* + cout << "BPhysPVTools::FillCandwithRefittedVertices: #PVs: all: " + << pvContainer->size() << " ref: " << refPvContainer->size() + << " good: " << goodPrimaryVertices.size() + << " PV_minNTracks: " << m_PV_minNTracks << endl; + */ + if(goodPrimaryVertices.empty() == false) { + + size_t pVmax =std::min(in_PV_max, goodPrimaryVertices.size()); + std::vector<const xAOD::Vertex*> refPVvertexes; + std::vector<const xAOD::Vertex*> refPVvertexes_toDelete; + std::vector<int> exitCode; + refPVvertexes.reserve(pVmax); + refPVvertexes_toDelete.reserve(pVmax); + exitCode.reserve(pVmax); + + bool doPt = (DoVertexType & 1) != 0; + bool doA0 = (DoVertexType & 2) != 0; + bool doZ0 = (DoVertexType & 4) != 0; + bool doZ0BA = (DoVertexType & 8) != 0; + + xAOD::VertexContainer::iterator vtxItr = vtxContainer->begin(); + for(; vtxItr!=vtxContainer->end(); ++vtxItr) { + xAOD::BPhysHelper vtx(*vtxItr); + + // 1) pT error + double ptErr = m_v0Tools->pTError( vtx.vtx() ); + BPHYS_CHECK( vtx.setPtErr(ptErr) ); + + for(size_t i =0; i < pVmax ; i++) { + const xAOD::Vertex* oldPV = goodPrimaryVertices.at(i); + //when set to false this will return nullptr when a new vertex is not required + int exit =0; + const xAOD::Vertex* refPV = pvRefitter->refitVertex(oldPV, vtx.vtx(), false, &exit); + exitCode.push_back(exit); + //I want positioning to match the goodPrimaryVertices + if(refPV == nullptr){ + refPVvertexes.push_back(oldPV); + refPVvertexes_toDelete.push_back(nullptr); + }else{ + refPVvertexes.push_back(refPV); + refPVvertexes_toDelete.push_back(refPV); + } + } + + // 2) refit the primary vertex and set the related decorations. + + size_t highPtindex = doPt ? FindHighPtIndex(refPVvertexes) : 9999999; //Should be 0 in PV ordering + size_t lowA0 = doA0 ? + FindLowA0Index(vtx, refPVvertexes, m_PV_minNTracks) : 9999998; + size_t lowZ = doZ0 ? + FindLowZIndex(vtx, refPVvertexes, m_PV_minNTracks) : 9999997; + size_t lowZBA = doZ0BA ? + FindLowZ0BAIndex(vtx, refPVvertexes, m_PV_minNTracks) : 9999996; + /* + cout << "BPhysPVTools::FillCandwithRefittedVertices: in_PV_max/pVMax = " + << in_PV_max << ", " << pVmax << endl; + cout << "BPhysPVTools::FillCandwithRefittedVertices: m_PV_minNTracks = " + << m_PV_minNTracks << endl; + cout << "BPhysPVTools::FillCandwithRefittedVertices: hPt,lowA0/Z/ZBA = " + << highPtindex << ", " + << lowA0 << ", " << lowZ << ", " << lowZBA << " " + << (lowA0 != lowZ ? "1!" : " ") + << (lowA0 != lowZBA ? "2!" : " ") + << (lowZ != lowZBA ? "3!" : " ") + << (highPtindex != lowA0 ? "4!" : " ") + << (highPtindex != lowZ ? "5!" : " ") + << (highPtindex != lowZBA ? "6!" : " ") + << endl; + */ + if(doPt) { + //Choose old PV container if not refitted + const xAOD::VertexContainer* ParentContainer = + (refPVvertexes_toDelete.at(highPtindex)) ? refPvContainer : pvContainer; + if(ParentContainer == refPvContainer) //if refitted add to refitted container + refPvContainer->push_back(const_cast<xAOD::Vertex*>(refPVvertexes.at(highPtindex))); // store the new vertex + + FillBPhysHelper(vtx, refPVvertexes[highPtindex], + ParentContainer, xAOD::BPhysHelper::PV_MAX_SUM_PT2, exitCode[highPtindex]); + vtx.setOrigPv(goodPrimaryVertices[highPtindex], pvContainer, xAOD::BPhysHelper::PV_MAX_SUM_PT2); + } + + if(doA0) { + const xAOD::VertexContainer* ParentContainer = + (refPVvertexes_toDelete.at(lowA0)) ? refPvContainer : pvContainer; + if(ParentContainer == refPvContainer && highPtindex!=lowA0) + refPvContainer->push_back(const_cast<xAOD::Vertex*>(refPVvertexes.at(lowA0))); // store the new vertex + + FillBPhysHelper(vtx, refPVvertexes[lowA0], + ParentContainer, xAOD::BPhysHelper::PV_MIN_A0, exitCode[lowA0]); + vtx.setOrigPv(goodPrimaryVertices[lowA0], pvContainer, xAOD::BPhysHelper::PV_MIN_A0); + } + + + // 2.c) the closest in Z: + if(doZ0) { + + const xAOD::VertexContainer* ParentContainer = + (refPVvertexes_toDelete.at(lowZ)) ? refPvContainer : pvContainer; + if(ParentContainer == refPvContainer && highPtindex!=lowZ && lowZ!=lowA0) + refPvContainer->push_back(const_cast<xAOD::Vertex*>(refPVvertexes.at(lowZ))); // store the new vertex + + FillBPhysHelper(vtx, refPVvertexes[lowZ], + ParentContainer, xAOD::BPhysHelper::PV_MIN_Z0, exitCode[lowZ]); + vtx.setOrigPv(goodPrimaryVertices[lowZ], pvContainer, xAOD::BPhysHelper::PV_MIN_Z0); + } + + // 2.d) the closest in Z (DOCA w.r.t. beam axis): + if (doZ0BA) { + if ( lowZBA < pVmax ) { // safety for vector indices + const xAOD::VertexContainer* ParentContainer = + (refPVvertexes_toDelete.at(lowZBA)) ? + refPvContainer : pvContainer; + if (ParentContainer == refPvContainer && highPtindex!=lowZBA + && lowZBA!=lowA0 && lowZBA != lowZ) { + // store the new vertex + refPvContainer->push_back(const_cast<xAOD::Vertex*> + (refPVvertexes.at(lowZBA))); + } + FillBPhysHelper(vtx, refPVvertexes[lowZBA], + ParentContainer, xAOD::BPhysHelper::PV_MIN_Z0_BA, + exitCode[lowZBA]); + vtx.setOrigPv(goodPrimaryVertices[lowZBA], pvContainer, + xAOD::BPhysHelper::PV_MIN_Z0_BA); + } else { + // nothing found -- fill nullptr + FillBPhysHelperNULL(vtx, pvContainer, xAOD::BPhysHelper::PV_MIN_Z0_BA, m_3dCalc); + // nothing found -- fill dummy vertex (type-0 vertex) + // if(pvContainer->empty()) return StatusCode::FAILURE; + // const xAOD::Vertex* dummy = pvContainer->at(pvContainer->size()-1); //No good vertices so last vertex must be dummy + // FillBPhysHelper(vtx, dummy, pvContainer, xAOD::BPhysHelper::PV_MIN_Z0_BA, 0); + // vtx.setOrigPv(Dummy, pvContainer, xAOD::BPhysHelper::PV_MIN_Z0_BA); + } + } + + //nullptrify ptrs we want to keep so these won't get deleted + //"delete nullptr" is valid in C++ and does nothing so this is quicker than a lot of if statements + if(doPt) refPVvertexes_toDelete[highPtindex] = nullptr; + if(doA0) refPVvertexes_toDelete[lowA0] = nullptr; + if(doZ0) refPVvertexes_toDelete[lowZ] = nullptr; + if(doZ0BA && lowZBA < pVmax) refPVvertexes_toDelete[lowZBA] = nullptr; + //Loop over toDELETE container, anything that is used or was not refitted is nullptr + //This cleans up all extra vertices that were created and not used + for(const xAOD::Vertex* ptr : refPVvertexes_toDelete) { + delete ptr; + } + refPVvertexes.clear();// Clear lists of now dangling ptrs + refPVvertexes_toDelete.clear(); + exitCode.clear(); + } // end of loop over vertices + + } else { +// cout << "Warning: DerivationFramework::BPhysPVTools::FillCandwithRefittedVertices No Primary Vertices Found trying to decorate wilth dummy \n"; + if(pvContainer->empty()) return StatusCode::FAILURE; + const xAOD::Vertex* dummy = pvContainer->at(0); //No good vertices so last vertex must be dummy + DecorateWithDummyVertex(vtxContainer, pvContainer, dummy, DoVertexType, true); + } + + return StatusCode::SUCCESS; +} + +size_t DerivationFramework::BPhysPVTools::FindHighPtIndex(const std::vector<const xAOD::Vertex*> &PVlist) { + // it SHOULD be the first one in the collection but it shouldn't take long to do a quick check + for(size_t i =0; i<PVlist.size(); i++) { + if(PVlist[i]->vertexType() == xAOD::VxType::PriVtx) return i; + } + cout << "FATAL ERROR High Pt Primary vertex not found - this should not happen\n"; + return std::numeric_limits<std::size_t>::max(); //This should not happen +} + +size_t DerivationFramework::BPhysPVTools::FindLowA0Index(const xAOD::BPhysHelper &Obj, + const std::vector<const xAOD::Vertex*> &PVlist, + const size_t PV_minNTracks) const { + size_t lowA0 = 0; + if(PVlist.empty()) { + lowA0=std::numeric_limits<std::size_t>::max(); + return lowA0; + } + size_t size = PVlist.size(); + double lowA0calc = m_v0Tools->a0(Obj.vtx(), PVlist[0]); + for(size_t i =1; i<size; i++) { + if ( PVlist[i]->nTrackParticles() >= PV_minNTracks ) { + double a0 = m_v0Tools->a0(Obj.vtx(), PVlist[i]); + if(a0 < lowA0calc) { + lowA0calc = a0; + lowA0 =i; + } + } + } + return lowA0; +} + +vector<const xAOD::Vertex*> DerivationFramework::BPhysPVTools::GetGoodPV(const xAOD::VertexContainer* pvContainer) { + typedef xAOD::VxType::VertexType VertexType; + VertexType Pvtx = xAOD::VxType::PriVtx; + VertexType Pileupvtx = xAOD::VxType::PileUp; + std::vector<const xAOD::Vertex*> goodPrimaryVertices; + goodPrimaryVertices.reserve(pvContainer->size()); + + for (auto ptr = pvContainer->begin(); ptr!= pvContainer->end(); ++ptr) { + VertexType thistype = (*ptr)->vertexType(); + if ( thistype == Pileupvtx || thistype == Pvtx ) { + goodPrimaryVertices.push_back(*ptr); + } else { +// cout << "vertex type " << thistype << endl; + } + } + return goodPrimaryVertices; +} +//----------------------------------------------------------------------------- +// added by WW: +// +void DerivationFramework::BPhysPVTools::SetMinNTracksInPV(size_t PV_minNTracks) +{ + + m_PV_minNTracks = PV_minNTracks; +} +//----------------------------------------------------------------------------- +// added by WW: +// +const Amg::Vector3D& DerivationFramework::BPhysPVTools::GetBeamSpot() const noexcept { + + if(m_beamSpotData) return m_beamSpotData->beamPos(); + else { + static const Amg::Vector3D defaultBS(-10000.,-10000.,-10000.); + return defaultBS; + } +} +//----------------------------------------------------------------------------- +// added by WW: +// +size_t DerivationFramework::BPhysPVTools::FindLowZ0BAIndex(const xAOD::BPhysHelper &obj, + const std::vector<const xAOD::Vertex*> &PVlist, + const size_t PV_minNTracks) const { + + size_t ilowZ0BA = std::numeric_limits<std::size_t>::max(); + double lowZ0BAcalc = std::numeric_limits<double>::max(); + for (size_t i = 0; i<PVlist.size(); ++i) { + if ( PVlist[i]->nTrackParticles() >= PV_minNTracks ) { + double z0BA = m_v0Tools->a0(obj.vtx(), PVlist[i]); + if (z0BA < lowZ0BAcalc) { + lowZ0BAcalc = z0BA; + ilowZ0BA = i; + } + } + } + return ilowZ0BA; +} +//----------------------------------------------------------------------------- +// added by WW: +// +double DerivationFramework::BPhysPVTools::DistInZtoDOCA(const xAOD::BPhysHelper &obj, const xAOD::Vertex* vertex) const { + + Amg::Vector3D pv = vertex->position(); + Amg::Vector3D xDOCA = DocaExtrapToBeamSpot(obj); + Amg::Vector3D vec = pv - xDOCA; + return vec.z(); +} +//----------------------------------------------------------------------------- +// added by WW: +// +Amg::Vector3D DerivationFramework::BPhysPVTools::DocaExtrapToBeamSpot(const xAOD::BPhysHelper& obj) const { + + Amg::Vector3D xDOCA(-99999., -99999., -99999.); + TVector3 totP(const_cast<xAOD::BPhysHelper&>(obj).totalP()); + Amg::Vector3D pSV(totP.X(), totP.Y(), totP.Z()); + Amg::Vector3D pT(pSV.x(), pSV.y(), 0.); + if ( pT.mag2() > 0 ) { + Amg::Vector3D xBS = GetBeamSpot(); + Amg::Vector3D xSV = m_v0Tools->vtx(obj.vtx()); + Amg::Vector3D xT(xSV.x()-xBS.x(), xSV.y()-xBS.y(), 0.); + xDOCA = xSV - pSV*pT.dot(xT)/pT.mag2(); + } else { + std::cout << "BPhysPVTools::DocaExtrapToBeamSpot: WARNING pT == 0." + << std::endl; + } + return xDOCA; +} + +void DerivationFramework::BPhysPVTools::PrepareVertexLinks(xAOD::Vertex* theResult, + const xAOD::TrackParticleContainer* importedTrackCollection) +{ + std::vector<ElementLink<DataVector<xAOD::TrackParticle> > > newLinkVector; + const auto &trkprtl = theResult->trackParticleLinks(); + for(unsigned int i=0; i< trkprtl.size(); i++) + { + ElementLink<DataVector<xAOD::TrackParticle> > mylink=trkprtl[i]; //makes a copy (non-const) + mylink.setStorableObject(*importedTrackCollection, true); + newLinkVector.push_back( mylink ); + } + theResult->clearTracks(); + theResult->setTrackParticleLinks( newLinkVector ); +} + + +//----------------------------------------------------------------------------- diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/BPhysVarBlinder.cxxNoCompile b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/BPhysVarBlinder.cxxNoCompile new file mode 100644 index 00000000000..13a63f881ba --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/BPhysVarBlinder.cxxNoCompile @@ -0,0 +1,72 @@ +/* + Copyright (C) 2002-2018 CERN for the benefit of the ATLAS collaboration +*/ + +//============================================================================ +// BPhysVarBlinder.cxx +//============================================================================ +// +// Author : Wolfgang Walkowiak <Wolfgang.Walkowiak@cern.ch.> +// Changes: +// +//============================================================================ +// +#include "BPhysTools/BPhysBlindingTool.h" +#include "DerivationFrameworkBPhys/BPhysVarBlinder.h" + +namespace DerivationFramework { + + //--------------------------------------------------------------------------- + // Constructor + //--------------------------------------------------------------------------- + BPhysVarBlinder::BPhysVarBlinder(const std::string& t, + const std::string& n, + const IInterface* p) : + CfAthAlgTool(t,n,p), + m_blindingTool("xAOD::BPhysBlindingTool") { + + declareInterface<DerivationFramework::IAugmentationTool>(this); + + // Declare tools + declareProperty("BlindingTool", m_blindingTool); + + // Declare user-defined properties + declareProperty("EnableBlinding", m_enableBlinding = true); + + } + //--------------------------------------------------------------------------- + // Initialization + //--------------------------------------------------------------------------- + StatusCode BPhysVarBlinder::initialize() { + + ATH_MSG_DEBUG("in initialize()"); + + // retrieve blinding tool + CHECK( m_blindingTool.retrieve() ); + + ATH_MSG_INFO("initialize(): EnableBlinding: " << m_enableBlinding); + + return StatusCode::SUCCESS; + } + //--------------------------------------------------------------------------- + // Finalization + //--------------------------------------------------------------------------- + StatusCode BPhysVarBlinder::finalize() { + + // everything all right + return StatusCode::SUCCESS; + } + //--------------------------------------------------------------------------- + // Perform blinding + //--------------------------------------------------------------------------- + StatusCode BPhysVarBlinder::addBranches() const { + + if ( m_enableBlinding ) { + CHECK( m_blindingTool->doBlind() ); + } + + return StatusCode::SUCCESS; + } + //--------------------------------------------------------------------------- + +} // namespace DerivationFramework diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/BPhysVertexTrackBase.cxx b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/BPhysVertexTrackBase.cxx new file mode 100644 index 00000000000..2f2cef79bfc --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/BPhysVertexTrackBase.cxx @@ -0,0 +1,1480 @@ +/* + Copyright (C) 2002-2019 CERN for the benefit of the ATLAS collaboration +*/ + +//============================================================================ +// BPhysVertexTrackBase.cxx +//============================================================================ +// +// Author : Wolfgang Walkowiak <Wolfgang.Walkowiak@cern.ch.> +// Changes: +// +// Base class for vertex-track related classes in need of +// track-to-vertex association handling. +// +// Derived classes should overwrite the following methods instead +// of the orginal initialize() etc methods: +// - initializeHook() +// - finalizeHook() +// - addBranchesHook() -- only called at end of addBranches() +// - addBranchesSVLoopHook() -- preferred (inside SV loop) +// +// From within addBranchesSVLoopHook() +// - calculateValues() +// should be called and the following hook methods should be overwritten: +// - calcValuesHook() -- called for the actual calculation of values +// - fastFillHook() -- called to check and apply caching of values +// +// For an usage example see BVertexTrackIsoTool and BPHY8.py . +// +// Job options provided by this class: +// - BranchPrefixes -- vector of prefixes to assign to +// added branches +// (Must be of same size as VertexContainerNames +// and in the same order.) +// - BranchBaseName -- assign the base name of added branches +// (default: iso) +// - BranchSuffix -- assign the suffix of added branches +// (default: empty = none) +// - VertexContainerNames -- names of containers for secondary vertices +// - TrackParticleContainerName -- name of container for TrackParticles +// - TrackToVertexTool -- ToolHandle for track-to-vertex tool +// - TrackSelectionTools -- Array of ToolHandles for track +// selection tools; each tool should +// be named XXXX_YYYY, where the YYYY +// suffix string which needs to be unique; +// will be used to name the isolation track +// category (part of the new attribute names) +// - RefPVContainerNames -- List of refitted PV container names. +// (Must be of same size as VertexContainerNames +// and in the same order.) +// - DoVertexType -- PV-to-SV association types to be +// considered (bitwise variable, see +// xAODBPhys::BPhysHelper) +// - UseTrackTypes -- List of or-ed bit-wise selection of +// track sets to consider: +// bit : meaning +// 0 : tracks close to PV associated +// with SV +// 1 : tracks associated with dummy PV +// ("type-0 PV tracks") +// 2 : tracks associated with PV of type 1 +// 3 : tracks associated with PV of type 2 +// 4 : tracks associated with PV of type 3 +// 5 : tracks associated with PV with types +// other than 0 to 4. +// 6 : tracks with missing pointer to +// PV (NULL pointer) +// 7-22: tracks being closest to assoc. PV +// useRefittedPVs doDCAin3D chi2DefToUse +// 7 : yes no 0 +// 8 : no no 0 +// 9 : yes yes 0 +// 10 : no yes 0 +// 11 : yes no 1 +// 12 : no no 1 +// 13 : yes yes 1 +// 14 : no yes 1 +// 15 : yes no 2 +// 16 : no no 2 +// 17 : yes yes 2 +// 18 : no yes 2 +// 19 : yes -- 3 +// 20 : no -- 3 +// 21 : yes -- 4 +// 22 : no -- 4 +// 23 : yes -- 5 +// 24 : no -- 5 +// 25 : yes yes 6 +// 26 : no yes 6 +// 27 : yes yes 7 +// 28 : no yes 7 +// 29 : yes yes 8 +// 30 : no yes 8 +// 31 : yes yes 9 +// 32 : no yes 9 +// useRefittedPVs: +// replace PV associated to decay candidate +// by the refitted PV +// doDCAin3D: +// use d0 and z0 in the determination of +// of the point of closest approach of +// a track to a vertex +// chi2DefToUse: +// PV uncertainties in the chi2 calculation +// in addition to track uncertainties +// 0 : use old method +// (only track uncertainties) +// 1 : from track perigee with +// uncertainties from track and vertex +// 2 : simple extrapolation from track +// parameters with uncertainties from +// track and vertex (extrapolation +// used for track swimming) +// 3 : CalcLogChi2toPV method from NtupleMaker +// using xAOD::TrackingHelpers. +// (only track uncertainties) +// 4 : CalcLogChi2toPV method from NtupleMaker +// using xAOD::TrackingHelpers. +// (track and vertex uncertainties) +// 5 : use TrackVertexAssociationTool +// 6 : full 3D chi2 from track perigee +// with uncertainties from track and +// vertex (sum of 3x3 covariance matrices) +// 7 : full 3D chi2 from track perigee with +// uncertainties from track and vertex +// (sum of 2x2 covariance matrices) +// 8 : simple extrapolation from track +// parameters with uncertainties +// from track and vertex +// (sum of 3x3 covariance matrices) +// 9 simple extrapolation from track +// parameters with uncertainties +// from track and vertex +// (sum of 2x2 covariance matrices) +// (E.g. 127 means to consider all tracks.) +// - IncPrecVerticesInDecay -- Include preceeding vertices in search +// for ID tracks and muons from decaying +// particle. (May be a bit slower but +// more accurate. Double-counting of track +// or muon objects is excluded. +// Default: True) +// - MinNTracksInPV -- Minimum number of tracks in PV for +// PV to be considered in calculation +// of closest PV to a track +// - PVTypesToConsider -- List of primary vertex types to consider +// in calculation of closest PV to a track +// - DebugTrackTypes -- Count tracks of specific types (bit +// patterns w.r.t. vertex association) +// and dump statistics to log file +// 0 : disabled +// 1 : count tracks of certain types +// - DebugTracksInEvents -- debug track selections in detail for +// a list of event numbers. +// +// +//============================================================================ +// +#include "DerivationFrameworkBPhys/BPhysVertexTrackBase.h" +#include "xAODTracking/TrackParticlexAODHelpers.h" +#include "xAODBPhys/BPhysHelper.h" +#include "InDetTrackSelectionTool/IInDetTrackSelectionTool.h" +#include "EventPrimitives/EventPrimitivesHelpers.h" +#include "TrackVertexAssociationTool/TrackVertexAssociationTool.h" +#include "TVector3.h" +#include "TString.h" +#include "boost/format.hpp" +#include <algorithm> +#include <sstream> +#include <limits> + +namespace DerivationFramework { + + //------------------------------------------------------------------------- + // + // helper class + BPhysVertexTrackBase::BaseItem::BaseItem(std::string Name, + std::string Bname, + std::string Prefix) : + name(std::move(Name)), bname(std::move(Bname)), prefix(std::move(Prefix)) { + } + + BPhysVertexTrackBase::BaseItem::~BaseItem() { + } + + void BPhysVertexTrackBase::BaseItem::setup(std::string Name, + std::string Bname, + std::string Prefix) { + name = std::move(Name); + bname = std::move(Bname); + prefix = std::move(Prefix); + } + + void BPhysVertexTrackBase::BaseItem::setPrefix(std::string Prefix) { + prefix = std::move(Prefix); + } + + void BPhysVertexTrackBase::BaseItem::resetVals() { + // needs to be implemented by derived class + } + + std::string BPhysVertexTrackBase::BaseItem::buildName(std::string qualifier, + std::string suffix) { + boost::format f("%s%s%s%s%s"); + f % (prefix.length() > 0 ? prefix+"_" : "") + % (bname.length() > 0 ? bname+"_" : "") + % (qualifier.length() > 0 ? qualifier+"_" : "") + % name + % suffix; + return f.str(); + } + + std::string BPhysVertexTrackBase::BaseItem::toString() const { + boost::format f("nm: %s\nbn: %s"); + f % name % bname; + return f.str(); + } + //------------------------------------------------------------------------- + // + // helper class (for track types) + // + BPhysVertexTrackBase::TrackTypeCounter:: + TrackTypeCounter(BPhysVertexTrackBase& Parent, std::string Name) + : name(std::move(Name)), m_parent(Parent) { + } + + BPhysVertexTrackBase::TrackTypeCounter::~TrackTypeCounter() { + } + + void BPhysVertexTrackBase::TrackTypeCounter::addToCounter(uint64_t atype, + uint64_t rtype, + std::string prefix, + std::string suffix, + uint64_t counts) { + boost::format f("%sT%010d_R%010d%s"); + f % (prefix.length() > 0 ? prefix+"_" : "") + % atype + % m_parent.m_useTrackTypes[rtype] + % (suffix.length() > 0 ? "_"+suffix : ""); + + addToCounter(f.str(), atype, counts); + } + + void BPhysVertexTrackBase::TrackTypeCounter::addToCounter(std::string name, + uint64_t atype, + uint64_t counts) { + + NameCountMap_t::const_iterator it = m_cnts.find(name); + + if ( it != m_cnts.end() ) { + m_cnts[name].first += counts; + } else { + m_cnts[name] = std::make_pair(counts, atype); + } + } + + std::string BPhysVertexTrackBase::TrackTypeCounter:: + countsToString(uint indent) const { + + boost::format f("%sCounters for %s:\n"); + f % boost::io::group(std::setw(indent), " ") % name; + std::string str = f.str(); + + int lmax(0); + for (NameCountMap_t::const_iterator it = m_cnts.begin(); + it != m_cnts.end(); ++it) { + lmax = std::max(lmax, (int)(it->first).length()); + } + + for (NameCountMap_t::const_iterator it = m_cnts.begin(); + it != m_cnts.end(); ++it) { + boost::format f("%s%-s : %10lld %33s"); + f % boost::io::group(std::setw(indent+4), " ") + % boost::io::group(std::setw(lmax), it->first) + % (it->second).first + % std::bitset<33>((it->second).second).to_string(); + str += f.str() + "\n"; + } + // clean up last newline + str.erase(str.length()-1); + + return str; + } + //-------------------------------------------------------------------------- + //------------------------------------------------------------------------- + // static members + const int BPhysVertexTrackBase::n_track_types = 33; + const std::string BPhysVertexTrackBase::track_type_str[] = + {"ASSOCPV", "PVTYPE0", "PVTYPE1", "PVTYPE2", "PVTYPE3", "NONE", "NULLVP", + "CAPVRFN3U0", "CAPVNRN3U0", "CAPVRF3DU0", "CAPVNR3DU0", + "CAPVRFN3U1", "CAPVNRN3U1", "CAPVRF3DU1", "CAPVNR3DU1", + "CAPVRFN3U2", "CAPVNRN3U2", "CAPVRF3DU2", "CAPVNR3DU2", + "CAPVRFNNU3", "CAPVNRNNU3", "CAPVRFNNU4", "CAPVNRNNU4", + "CAPVRFNNU5", "CAPVNRNNU5", "CAPVRFNNU6", "CAPVNRNNU6", + "CAPVRFNNU7", "CAPVNRNNU7", "CAPVRFNNU8", "CAPVNRNNU8", + "CAPVRFNNU9", "CAPVNRNNU9"}; + const uint64_t BPhysVertexTrackBase::track_type_bit[] = + {0x1, 0x2, 0x4, 0x8, 0x10, 0x20, 0x40, + 0x80, 0x100, 0x200, 0x400, + 0x800, 0x1000, 0x2000, 0x4000, + 0x8000, 0x10000, 0x20000, 0x40000, + 0x80000, 0x100000, 0x200000, 0x400000, + 0x800000, 0x1000000, 0x2000000, 0x4000000, + 0x8000000, 0x10000000, 0x20000000, 0x40000000, + 0x80000000, 0x100000000}; + uint64_t BPhysVertexTrackBase::s_track_type_all_cached = 0x0; + + // static methods + const std::string + BPhysVertexTrackBase::tts(BPhysVertexTrackBase::track_type type) { + return track_type_str[type]; + } + + uint64_t BPhysVertexTrackBase::ttb(BPhysVertexTrackBase::track_type type) { + return track_type_bit[type]; + } + + uint64_t BPhysVertexTrackBase::ttallMin() { + // only bits 0 - 6 + return 127; + } + + uint64_t BPhysVertexTrackBase::ttall() { + if ( s_track_type_all_cached == 0x0 ) { + for (unsigned int i=0; i<n_track_types; ++i) { + s_track_type_all_cached |= track_type_bit[i]; + } + } + return s_track_type_all_cached; + } + + uint64_t BPhysVertexTrackBase::rttor(const std::vector<uint64_t> &vtypes) { + // or of requested track types + uint64_t ttor(0); + for (size_t i=0; i<vtypes.size(); ++i) { + ttor |= vtypes[i]; + } + return ttor; + } + + // track to string + std::string + BPhysVertexTrackBase::trackToString(const xAOD::TrackParticle* track) { + std::string str; + if (track != nullptr) { + boost::format f("p(%10.4f,%10.4f,%10.4f)\n" + "d:(%10.5f,%10.5f,%10.5f,%10.5f,%10.6f)"); + f % (track->p4()).Px() % (track->p4()).Py() % (track->p4()).Pz(); + f % track->d0() % track->z0() % track->phi0() % track->theta(); + f % track->qOverP(); + str = f.str(); + } // if track + return str; + } + + //-------------------------------------------------------------------------- + // Static utility method to prefix every line by a certain string + //-------------------------------------------------------------------------- + std::string BPhysVertexTrackBase::wrapLines(std::string lines, + std::string prefix) { + + std::string ostr; + std::istringstream stream(lines); + std::string line; + while ( std::getline(stream, line) ) { + if ( !ostr.length() == 0 ) ostr += "\n"; + ostr += prefix + line; + } + return ostr; + } + //-------------------------------------------------------------------------- + //-------------------------------------------------------------------------- + BPhysVertexTrackBase::BPhysVertexTrackBase(const std::string& t, + const std::string& n, + const IInterface* p) + : AthAlgTool(t,n,p), m_trackToVertexTool("Reco::TrackToVertex"), + m_tvaTool("CP::TrackVertexAssociationTool"), + m_tvaToolHasWpLoose(false), + m_tracks(NULL), m_tracksAux(NULL), m_nEvtsSeen(0), m_eventInfo(nullptr), + m_trackTypesUsed(0), m_runNumber(0), m_evtNumber(0), + m_debugTracksInThisEvent(false) { + + declareInterface<DerivationFramework::IAugmentationTool>(this); + + // Declare branch prefix + declareProperty("BranchPrefixes", m_branchPrefixes); + declareProperty("BranchBaseName", m_branchBaseName = "iso"); + declareProperty("BranchSuffix" , m_branchSuffix = "" ); + + // Necessary containers + declareProperty("VertexContainerNames" , m_vertexContainerNames); + declareProperty("TrackParticleContainerName", + m_trackParticleContainerName); + declareProperty("TrackToVertexTool" , m_trackToVertexTool); + declareProperty("TrackSelectionTools" , m_trackSelectionTools); + declareProperty("TVATool" , m_tvaTool); + declareProperty("PVContainerName", m_pvContainerName = "PrimaryVertices"); + declareProperty("RefPVContainerNames" , m_refPVContainerNames); + declareProperty("DoVertexType" , m_doVertexType = 8); + declareProperty("UseTrackTypes" , m_useTrackTypes = {7}); + declareProperty("IncPrecVerticesInDecay", m_incPrecVerticesInDecay = true); + declareProperty("MinNTracksInPV" , m_minNTracksInPV = 0); + declareProperty("PVTypesToConsider" , m_pvTypesToConsider = {1,3}); + declareProperty("DebugTrackTypes" , m_debugTrackTypes=0); + declareProperty("DebugTracksInEvents" , m_debugTracksInEvents = {}); + } + //-------------------------------------------------------------------------- + StatusCode BPhysVertexTrackBase::initialize() { + + ATH_MSG_DEBUG("BPhysVertexTrackBase::initialize() -- begin"); + + if ( m_vertexContainerNames.size() == 0 ) { + ATH_MSG_ERROR("No vertex container names provided!"); + } + if ( m_refPVContainerNames.size() == 0 ) { + ATH_MSG_ERROR("No refitted PV container names provided!"); + } + if ( m_trackParticleContainerName == "" ) { + ATH_MSG_ERROR("No track particle container name provided!"); + } + if ( m_pvContainerName == "" ) { + ATH_MSG_ERROR("No PV container name provided!"); + } + if ( m_vertexContainerNames.size() != m_refPVContainerNames.size() ) { + ATH_MSG_ERROR("Size mismatch of VertexContainerNames (" + << m_vertexContainerNames.size() + << ") and RefPVContainerNames (" + << m_refPVContainerNames.size() << ") lists!"); + } + + if ( m_vertexContainerNames.size() != m_branchPrefixes.size() ) { + ATH_MSG_ERROR("Size mismatch of VertexContainerNames (" + << m_vertexContainerNames.size() + << ") and BranchPrefixes (" + << m_branchPrefixes.size() << ") lists!"); + } + + // TrackToVertexTool + ATH_CHECK(m_trackToVertexTool.retrieve()); + + // TrackSelectionTools + for (auto selTool : m_trackSelectionTools ) { + ATH_CHECK(selTool.retrieve()); + } + + // TrackVertexAssociationTool + ATH_CHECK(m_tvaTool.retrieve()); + // take note of working point + // const std::string tvaWp("Loose"); + const std::string tvaWp = + dynamic_cast<CP::TrackVertexAssociationTool*>(m_tvaTool.get())->getProperty("WorkingPoint").toString(); + m_tvaToolHasWpLoose = (tvaWp == "Loose"); + + // initialize PV-to-SV association type vector + initPvAssocTypeVec(); + + // initialize track type request pattern + m_trackTypesUsed = rttor(m_useTrackTypes); + + // initialize track type counters + if ( m_debugTrackTypes > 0 ) { + m_mttc = std::make_unique<TrackTypeCounter>(*this, name()); + } + + ATH_MSG_DEBUG("BPhysVertexTrackBase::initialize() -- end"); + + return initializeHook(); + } + //-------------------------------------------------------------------------- + StatusCode BPhysVertexTrackBase::finalize() { + + ATH_MSG_DEBUG("BPhysVertexTrackBase::finalize()"); + + // dump track type counters to log + if ( m_debugTrackTypes > 0 ) { + ATH_MSG_INFO("Track type counters:\n" << m_mttc->countsToString()); + } + + // everything all right + return finalizeHook(); + } + //-------------------------------------------------------------------------- + StatusCode BPhysVertexTrackBase::addBranches() const { + + ATH_MSG_DEBUG("BPhysVertexTrackBase::addBranches() -- begin"); + + // counter + m_nEvtsSeen++; + + // run and event numbers + CHECK(evtStore()->retrieve(m_eventInfo)); + m_runNumber = m_eventInfo->runNumber(); + m_evtNumber = m_eventInfo->eventNumber(); + + // debug tracks in current event? + m_debugTracksInThisEvent = (std::find(m_debugTracksInEvents.begin(), + m_debugTracksInEvents.end(), + m_evtNumber) + != m_debugTracksInEvents.end()); + + // retrieve primary vertices container + m_pvtxContainer = NULL; + CHECK(evtStore()->retrieve(m_pvtxContainer, m_pvContainerName)); + ATH_MSG_DEBUG("Found PV collection with key " << m_pvContainerName); + + + // retrieve ID track container + m_tracks = NULL; + m_tracksAux = NULL; + CHECK(evtStore()->retrieve(m_tracks, m_trackParticleContainerName)); + if (evtStore()->contains<xAOD:: + TrackParticleAuxContainer>(m_trackParticleContainerName+"Aux.")) { + CHECK(evtStore()->retrieve(m_tracksAux, + m_trackParticleContainerName+"Aux.")); + } else { + ATH_MSG_DEBUG("No aux track collection with key " + << m_trackParticleContainerName+"Aux."); + } + ATH_MSG_DEBUG("Found track collection with key " + << m_trackParticleContainerName); + + // Loop over all vertex containers + for (size_t i=0; i<m_vertexContainerNames.size(); ++i) { + // vertex container and its auxilliary store + const xAOD::VertexContainer* svtxContainer = NULL; + const xAOD::VertexAuxContainer* svtxAuxContainer = NULL; + // refitted primary vertex container and its auxilliary store + const xAOD::VertexContainer* refPVContainer = NULL; + const xAOD::VertexAuxContainer* refPVAuxContainer = NULL; + + // retrieve from StoreGate + CHECK(evtStore()->retrieve(svtxContainer, m_vertexContainerNames[i])); + CHECK(evtStore()->retrieve(svtxAuxContainer, + m_vertexContainerNames[i]+"Aux.")); + ATH_MSG_DEBUG("Found SV collection with key " + << m_vertexContainerNames[i]); + CHECK(evtStore()->retrieve(refPVContainer , + m_refPVContainerNames[i])); + CHECK(evtStore()->retrieve(refPVAuxContainer, + m_refPVContainerNames[i]+"Aux.")); + ATH_MSG_DEBUG("Found refitted PV collection with key " + << m_refPVContainerNames[i]); + + // vertex container depending setup in derived class + CHECK(addBranchesVCSetupHook(i)); + + // loop over secondary vertices + for (xAOD::VertexContainer::const_iterator vtxItr = + svtxContainer->begin(); vtxItr!=svtxContainer->end(); + ++vtxItr) { + + CHECK(addBranchesSVLoopHook(*vtxItr)); + + } // end of loop over vertices + } // end of loop over vertex container names + + ATH_MSG_DEBUG("BPhysVertexTrackBase::addBranches() -- end"); + + // nothing to do here + return addBranchesHook(); + } + + //-------------------------------------------------------------------------- + // Hook method for initialize() -- to be overwritten by derived class + //-------------------------------------------------------------------------- + StatusCode BPhysVertexTrackBase::initializeHook() { + + return StatusCode::SUCCESS; + } + //-------------------------------------------------------------------------- + // Hook method for finalize() -- to be overwritten by derived class + //-------------------------------------------------------------------------- + StatusCode BPhysVertexTrackBase::finalizeHook() { + + return StatusCode::SUCCESS; + } + //-------------------------------------------------------------------------- + // Hook method for addBranches() -- to be overwritten by derived class + //-------------------------------------------------------------------------- + StatusCode BPhysVertexTrackBase::addBranchesHook() const { + + return StatusCode::SUCCESS; + } + //-------------------------------------------------------------------------- + // Hook method for addBranches() VC setup + // -- to be overwritten by derived class + //-------------------------------------------------------------------------- + StatusCode BPhysVertexTrackBase::addBranchesVCSetupHook(size_t ivc) const { + + // just to avoid a compiler warning + ATH_MSG_DEBUG("addBranchesVCSetupHook: Vertex container index " << ivc + << " for collection " << m_vertexContainerNames[ivc] + << " with prefix " << m_branchPrefixes[ivc]); + + return StatusCode::SUCCESS; + } + //-------------------------------------------------------------------------- + // Hook method for addBranches() SV loop -- to be overwritten by derived class + //-------------------------------------------------------------------------- + StatusCode + BPhysVertexTrackBase::addBranchesSVLoopHook(const xAOD::Vertex* vtx) const { + + // just to avoid a compiler warning + ATH_MSG_DEBUG("addBranchesSVLoopHook: Vertex " << vtx); + + return StatusCode::SUCCESS; + } + //-------------------------------------------------------------------------- + // Calculate values -- used by calculateValues() + // -- to be overwritten by derived class + //-------------------------------------------------------------------------- + StatusCode + BPhysVertexTrackBase::calcValuesHook(const xAOD::Vertex* vtx, + const unsigned int ipv, + const unsigned int its, + const unsigned int itt) const { + + // just to avoid a compiler warning + ATH_MSG_DEBUG("calcIsolationOpti: vtx: " << vtx << ", ipv: " << ipv + << ", its: " << its << ", itt: " << itt); + + return StatusCode::SUCCESS; + } + //-------------------------------------------------------------------------- + // Fill values from cache if found -- used by calculateValues() + // -- to be overwritten by derived class + //-------------------------------------------------------------------------- + bool BPhysVertexTrackBase::fastFillHook(const xAOD::Vertex* vtx, + const int ipv) const { + + // just to avoid a compiler warning + ATH_MSG_DEBUG("fastIsoFill: vtx: " << vtx << ", ipv: " << ipv); + + return false; + } + //-------------------------------------------------------------------------- + // Calculation loops -- needs to be called from inside the implementation + // of addBranchesSVLoopHook() in the derived class. + // Derived class also needs to provide override methods for + // - fastFillHook -- needs to return true if cached value is used + // - calcValuesHook -- actually calculating value(s) + //-------------------------------------------------------------------------- + StatusCode + BPhysVertexTrackBase::calculateValues(const xAOD::Vertex* vtx) const { + + ATH_MSG_DEBUG("BPhysVertexTrackBase::calculateValues -- begin"); + + unsigned int nPvAssocs = m_pvAssocTypes.size(); + unsigned int nTrackSels = m_trackSelectionTools.size(); + unsigned int nTrackTypes = m_useTrackTypes.size(); + + m_pvAssocResMap.clear(); + + const xAOD::BPhysHelper cand(vtx); + for (unsigned int ipv = 0; ipv < nPvAssocs; ++ipv) { + if ( ipv == 0 || ! fastFillHook(vtx, ipv) ) { + for (unsigned int its = 0; its < nTrackSels; ++its) { + for (unsigned int itt = 0; itt < nTrackTypes; ++itt) { + ATH_MSG_DEBUG("Calling calcValuesHook with ipv: " << ipv + << ", its: " << its << ", itt: " << itt); + CHECK(calcValuesHook(vtx, ipv, its, itt)); + } // for itt + } // for its + // cache result index -- only needed once per ipv value + m_pvAssocResMap[buildPvAssocCacheName(vtx, ipv)] = ipv; + ATH_MSG_DEBUG("calculateValues: cache index: " + << buildPvAssocCacheName(vtx, ipv) + << " -- cached ipv: " << ipv); + } // if !fastFillHook() + } // for ipv + + return StatusCode::SUCCESS; + } + //------------------------------------------------------------------------- + // Build SV-to-PV association cache index string + //------------------------------------------------------------------------- + std::string + BPhysVertexTrackBase::buildPvAssocCacheName(const xAOD::Vertex* vtx, + const int ipv) const { + xAOD::BPhysHelper cand(vtx); + boost::format f("SV_%p_RPV_%p"); + f % cand.vtx() % cand.pv(m_pvAssocTypes[ipv]); + + return f.str(); + } + //-------------------------------------------------------------------------- + // getTrackCandPVLogChi2() + // Calculate the logChi2 (= log((d0/d0e)^2+(z0/z0e)^2) contribution of a + // track at the position closest to the PV associated with the SV. + //-------------------------------------------------------------------------- + double BPhysVertexTrackBase::getTrackCandPVLogChi2(const xAOD::TrackParticle* + track, + const xAOD::Vertex* vtx, + bool doDCAin3D, + int chi2DefToUse + ) const { + + return getTrackLogChi2DCA(track, vtx, doDCAin3D, chi2DefToUse)[4]; + } + //-------------------------------------------------------------------------- + // getTrackLogChi2DCA() + // Calculate logChi2 (= log((d0/d0e)^2+(z0/z0e)^2) contribution of a + // track at the position closest to a position and + // the distance of closest approach of a track w.r.t. + // a position. Either only in the transverse plane or in 3 dimensions. + // Option chi2DefToUse: + // 0 : from track perigee with uncertainties from track only + // 1 : from track perigee with uncertainties from track and vertex + // 2 : simple extrapolation from track parameters + // with uncertainties from track and vertex + // 3 : CalcLogChi2toPV method from NtupleMaker using xAOD::TrackingHelpers. + // (only track uncertainties) + // 4 : CalcLogChi2toPV method from NtupleMaker using xAOD::TrackingHelpers. + // (track and vertex uncertainties) + // 5 : use TrackVertexAssociationTool + // 6 : full 3D chi2 from track perigee with uncertainties + // from track and vertex (sum of 3x3 covariance matrices) + // 7 : full 3D chi2 from track perigee with uncertainties + // from track and vertex (sum of 2x2 covariance matrices) + // 8 : simple extrapolation from track parameters with uncertainties + // from track and vertex (sum of 3x3 covariance matrices) + // 9 simple extrapolation from track parameters with uncertainties + // from track and vertex (sum of 2x2 covariance matrices) + // Returned vector components: + // 0: d0, 1: d0Err, 2: z0, 3: z0Err, 4: logChi2, 5: dca, 6: okFlag + // 7: vtxErrPart2, 8: trkErrPart2, 9: phi0Used + //-------------------------------------------------------------------------- + std::vector<double> + BPhysVertexTrackBase::getTrackLogChi2DCA(const xAOD::TrackParticle* track, + const xAOD::Vertex* vtx, + bool doDCAin3D, + int chi2DefToUse) const { + // presets + std::vector<double> res = {-999., -99., -999., -99., -100., -100., -1., + -99., -99., -999.}; + + const Amg::Vector3D pos = vtx->position(); + const AmgSymMatrix(3) poscov = vtx->covariancePosition(); + + if ( track != NULL ) { + if ( chi2DefToUse < 2 || (chi2DefToUse > 5 && chi2DefToUse < 8) ) { + // use track perigee method + std::unique_ptr<const Trk::Perigee> + trkPerigee(m_trackToVertexTool->perigeeAtVertex(*track, pos)); + if ( trkPerigee != NULL ) { + res[0] = trkPerigee->parameters()[Trk::d0]; + res[2] = trkPerigee->parameters()[Trk::z0]; + const AmgSymMatrix(5)* locError = trkPerigee->covariance(); + if ( locError != NULL ) { + // uncertainties from track + res[1] = Amg::error(*locError, Trk::d0); + res[3] = Amg::error(*locError, Trk::z0); + if ( chi2DefToUse == 1 ) { + // add uncertainties from vertex + Amg::Vector3D perppt(trkPerigee->momentum().y()/trkPerigee->pT(), + -trkPerigee->momentum().x()/trkPerigee->pT(), + 0.); + double vtxD0Err2 = perppt.transpose()*poscov*perppt; + res[1] = sqrt( pow(res[1], 2.) + vtxD0Err2 ); + res[3] = sqrt( pow(res[3], 2.) + poscov(2,2) ); + } + if ( chi2DefToUse < 2 ) { + if ( fabs(res[1]) > 0. && fabs(res[3]) > 0. ) { + res[4] = log( pow(res[0]/res[1], 2.) + + pow(res[2]/res[3], 2.) ); + res[6] = 2.; // ok + } else { + ATH_MSG_WARNING("BPhysVertexTrackBase::getTrackLogChi2DCA():" + << " d0 = " << res[0] << ", d0Err = " + << res[1] << ", z0 = " << res[2] + << ", z0Err = " << res[3]); + } + } + // chi2DefToUse 6 or 7 + if ( chi2DefToUse > 5 && chi2DefToUse < 8 ) { + double phi0 = trkPerigee->parameters()[Trk::phi0]; + double doca = sqrt(pow(res[0],2.) + pow(res[2], 2.)); + res[9] = phi0; + if ( doca > 0. ) { + if ( chi2DefToUse == 6 ) { + AmgMatrix(5,3) dmat = AmgMatrix(5,3)::Zero(); + dmat(0,0) = -sin(phi0); + dmat(0,1) = cos(phi0); + dmat(1,2) = 1.; + dmat(2,0) = -res[0]*cos(phi0); + dmat(2,1) = -res[0]*sin(phi0); + AmgSymMatrix(3) mCovTrk3D = dmat.transpose()*(*locError)*dmat; + Amg::Vector3D dvec(-res[0]*sin(phi0), res[0]*cos(phi0), + res[2]); // (x,y,z) + Amg::Vector3D duvec = dvec.unit(); + // log(chi2) = log( docavec^T * V^-1 * docavec ) + res[4] = log( dvec.transpose() * (poscov+mCovTrk3D).inverse() + * dvec ); + res[7] = duvec.transpose()*poscov*duvec; + res[8] = duvec.transpose()*mCovTrk3D*duvec; + res[6] = 3.; // ok + } + if ( chi2DefToUse == 7 ) { + AmgMatrix(3,2) dmat = AmgMatrix(3,2)::Zero(); + dmat(0,0) = -sin(phi0); + dmat(1,0) = cos(phi0); + dmat(2,0) = 0.; + dmat(0,1) = 0.; + dmat(1,1) = 0.; + dmat(2,1) = 1.; + AmgSymMatrix(2) mCovVtx2D = dmat.transpose()*poscov*dmat; + AmgSymMatrix(2) mCovTrk2D = AmgSymMatrix(2)::Zero(); + mCovTrk2D(0,0) = (*locError)(Trk::d0,Trk::d0); + mCovTrk2D(0,1) = (*locError)(Trk::d0,Trk::z0); + mCovTrk2D(1,0) = (*locError)(Trk::d0,Trk::z0); + mCovTrk2D(1,1) = (*locError)(Trk::z0,Trk::z0); + Amg::Vector2D dvec(res[0], res[2]); // (d0, z0) + Amg::Vector2D duvec = dvec.unit(); + // log(chi2) = log( (d0, z0) * V^-1 * (d0, z0)^T ) + res[4] = log( dvec.transpose()*(mCovVtx2D+mCovTrk2D).inverse() + * dvec ); + res[7] = duvec.transpose()*mCovVtx2D*duvec; + res[8] = duvec.transpose()*mCovTrk2D*duvec; + res[6] = 4.; // ok + } + } else { + ATH_MSG_WARNING("BPhysVertexTrackBase::getTrackLogChi2DCA():" + << " doca == 0 !"); + } + } // if chi2DefToUse > 5 && chi2DefToUse < 8 + res[5] = doDCAin3D ? + sqrt( pow(res[0], 2.) + pow(res[2], 2.) ) : res[0]; + res[6] += 1.; // ok + } else { + ATH_MSG_WARNING("BPhysVertexTrackBase::getTrackLogChi2DCA():" + " locError pointer is NULL!"); + } + } else { + ATH_MSG_WARNING("BPhysVertexTrackBase::getTrackLogChi2DCA():" + " trkPerigee pointer is NULL!"); + } // if trkPerigee + + } else if ( chi2DefToUse == 2 + || (chi2DefToUse > 7 && chi2DefToUse < 10 )) { + // simple extrapolation method + // (directly taken from NtupleMaker for comparisons) + + // SV position and covariance matrix + TVector3 SV_def(vtx->x(), vtx->y(), vtx->z()); + const AmgSymMatrix(3)& SV_cov = poscov; + + // chi2 track to SV + double px = ( track->p4() ).Px(); + double py = ( track->p4() ).Py(); + double pt = ( track->p4() ).Pt(); + double d0 = track->d0(); + double d0Err2 = track->definingParametersCovMatrixVec()[0]; + double z0 = track->z0(); + double z0Err2 = track->definingParametersCovMatrixVec()[2]; + double theta = track->theta(); + double d0z0Cov = track->definingParametersCovMatrixVec()[1]; + double phi = track->phi(); + + TVector3 trk_origin( track->vx(), track->vy(), track->vz() ); + TVector3 SV = SV_def - trk_origin; + + // calc. error in direction perpendicular to pT (still x-y plane) + double upx = py/pt; + double upy = -px/pt; + double d0toSV = d0 + (SV[0]*upx + SV[1]*upy); + double d0toSVErr2 = upx*SV_cov(0, 0)*upx + 2*upx*SV_cov(1, 0)*upy + + upy*SV_cov(1, 1)*upy + d0Err2; + + upx = px/pt; + upy = py/pt; + double cot_theta = cos(theta)/sin(theta); + double z0corr = (SV[0]*upx + SV[1]*upy)*cot_theta; + double z0toSV = z0 + z0corr - SV[2]; + double z0toSVErr2 = SV_cov(2, 2) + z0Err2; + + double docaSV = sqrt( pow(d0toSV, 2) + pow(z0toSV, 2) ); + + double chi2testSV(999.); + if ( chi2DefToUse == 2 ) { + if (d0toSVErr2 !=0 && z0toSVErr2 != 0) + chi2testSV = log(pow( d0toSV, 2)/d0toSVErr2 + + pow( z0toSV, 2)/z0toSVErr2); + // set results + res = {d0toSV, sqrt(d0toSVErr2), z0toSV, sqrt(z0toSVErr2), + chi2testSV, (doDCAin3D ? docaSV : d0toSV), 4, + -99., -99., -999.}; + } + if ( chi2DefToUse > 7 && chi2DefToUse < 10 ) { + if ( docaSV > 0. ) { + if ( chi2DefToUse == 8 ) { + AmgMatrix(5,3) dmat = AmgMatrix(5,3)::Zero(); + dmat(0,0) = -sin(phi); + dmat(0,1) = cos(phi); + dmat(1,2) = 1.; + dmat(2,0) = -d0toSV*cos(phi); + dmat(2,1) = -d0toSV*sin(phi); + const AmgSymMatrix(5) mCovTrk5D = + track->definingParametersCovMatrix(); + AmgSymMatrix(3) mCovTrk3D = dmat.transpose()*mCovTrk5D*dmat; + Amg::Vector3D dvec(-d0toSV*sin(phi), d0toSV*cos(phi), + z0toSV); // (x,y,z) + Amg::Vector3D duvec = dvec.unit(); + // log(chi2) = log( docavec^T * V^-1 * docavec ) + double chi2testSV = log( dvec.transpose() + * (poscov+mCovTrk3D).inverse() + * dvec ); + double vtx3DErr2 = duvec.transpose()*poscov*duvec; + double trk3DErr2 = duvec.transpose()*mCovTrk3D*duvec; + // set results + res = {d0toSV, sqrt(d0Err2), z0toSV, sqrt(z0Err2), + chi2testSV, (doDCAin3D ? docaSV : d0toSV), 5, + vtx3DErr2, trk3DErr2, phi}; + } + if ( chi2DefToUse == 9 ) { + AmgMatrix(3,2) dmat = AmgMatrix(3,2)::Zero(); + dmat(0,0) = -sin(phi); + dmat(1,0) = cos(phi); + dmat(2,0) = 0.; + dmat(0,1) = 0.; + dmat(1,1) = 0.; + dmat(2,1) = 1.; + AmgSymMatrix(2) mCovVtx2D = dmat.transpose()*SV_cov*dmat; + AmgSymMatrix(2) mCovTrk2D = AmgSymMatrix(2)::Zero(); + mCovTrk2D(0,0) = d0Err2; + mCovTrk2D(0,1) = d0z0Cov; + mCovTrk2D(1,0) = d0z0Cov; + mCovTrk2D(1,1) = z0Err2; + Amg::Vector2D dvec(d0toSV, z0toSV); + Amg::Vector2D duvec = dvec.unit(); + // log(chi2) = log( (d0, z0) * V^-1 * (d0, z0)^T ) + chi2testSV = log( dvec.transpose()*(mCovVtx2D+mCovTrk2D).inverse() + * dvec ); + double vtx2DErr2 = duvec.transpose()*mCovVtx2D*duvec; + double trk2DErr2 = duvec.transpose()*mCovTrk2D*duvec; + + if ( vtx2DErr2 < 0. || trk2DErr2 < 0. ) { + ATH_MSG_WARNING("BPhysVertexTrackBase::" + "getTrackLogChi2DCA(): " + << "vtx2DErr2 = " << vtx2DErr2 + << " trk2DErr2 = " << trk2DErr2 + << " chi2testSV = " << chi2testSV); + ATH_MSG_WARNING("dvec = " << dvec); + ATH_MSG_WARNING("mCovVtx2D = " << mCovVtx2D); + ATH_MSG_WARNING("mCovTrk2D = " << mCovTrk2D); + ATH_MSG_WARNING("dmat = " << dmat); + ATH_MSG_WARNING("SV_cov = " << SV_cov); + ATH_MSG_WARNING("det(mCovVtx2D) = " << mCovVtx2D.determinant()); + ATH_MSG_WARNING("det(mCovTrk2D) = " << mCovTrk2D.determinant()); + ATH_MSG_WARNING("det(SV_cov) = " << SV_cov.determinant()); + ATH_MSG_WARNING("d0toSV = " << d0toSV + << " z0toSV = " << z0toSV + << " phi = " << phi + << " docaSV = " << docaSV); + } + + // set results + res = {d0toSV, sqrt(d0Err2), z0toSV, sqrt(z0Err2), + chi2testSV, (doDCAin3D ? docaSV : d0toSV), 6, + vtx2DErr2, trk2DErr2, phi}; + } + } else { + ATH_MSG_WARNING("BPhysVertexTrackBase::getTrackLogChi2DCA():" + << " docaSV == 0 !"); + } + } // if chi2DefToUse > 7 && chi2DefToUse < 10 + + } else if ( chi2DefToUse > 2 && chi2DefToUse < 5 ) { + // CalcLogChi2toPV method using xAOD::TrackingHelpers + // (simply taken from NtupleMaker for comparisons) + // N.B. z0significance method of the helper doesn't include pv_z0 + // uncertainty + double d0sign(0.); + if (chi2DefToUse == 4) { + d0sign = + xAOD::TrackingHelpers::d0significance(track, + m_eventInfo->beamPosSigmaX(), + m_eventInfo->beamPosSigmaY(), + m_eventInfo->beamPosSigmaXY() + ); + } else { + d0sign = xAOD::TrackingHelpers::d0significance( track ); + } + // trk z0 is expressed relative to the beamspot position along z-axis + // (trk->vz()) + // DCA always in 3D + double z0toPV = track->z0() + track->vz() - vtx->z(); + double z0Err2 = track->definingParametersCovMatrixVec()[2]; + if (chi2DefToUse == 4) z0Err2+= vtx->covariancePosition()(2,2); + double z0sign = z0toPV / sqrt( z0Err2 ); + double chi2 = log( pow(d0sign, 2.) + pow(z0sign, 2.) ); + // set results + res = {-999., -99., z0toPV, sqrt(z0Err2), chi2, -100., 4, -99., -99., + -999.}; + + } // if chi2DefToUse + } else { + ATH_MSG_WARNING("BPhysVertexTrackBase::getTrackLogChi2DCA():" + " track pointer is NULL!"); + res[6] = -2.; + } // if track != NULL + return res; + } + //-------------------------------------------------------------------------- + // detTrackTypes(): returns a bit pattern of the applicable + // track types from {ASSOCPV, PVTYPE0, PVTYPE1, PVTYPE2, PVTYPE3, NONE, + // NULLVP, CAPVXXXXXXX, ...} (or'd). + //-------------------------------------------------------------------------- + uint64_t BPhysVertexTrackBase::detTrackTypes(const xAOD::TrackParticle* track, + const xAOD::Vertex* candPV, + const xAOD::Vertex* candRefPV) const { + int bits = 0x0; + + // PVTYPE0 - PVTYPE3, NONE + ATH_MSG_ERROR("BPhysVertexTrackBase::detTrackTypes must be adjusted due to changes in TrackParticle"); + + // ASOCPV + if ( candPV != NULL ) { + bool found(false); + for (size_t i=0; i<candPV->nTrackParticles(); ++i) { + if ( track == candPV->trackParticle(i) ) { + found = true; + break; + } + } + if ( found ) bits |= track_type_bit[ASSOCPV]; + // + // CLOSEAPV + for (unsigned int i=7; i<n_track_types; ++i) { + if ( (track_type_bit[i] & m_trackTypesUsed) > 0x0 ) { + bool useRefittedPvs = ( i%2 == 1 ); + bool doDCAin3D = ( (i-7)%4 > 1 ); + int chi2DefToUse = (i-7)/4; + // adjustment above bit 20 + if ( i > 20 ) { + doDCAin3D = true; + chi2DefToUse = (i-13)/2; + } + const xAOD::Vertex* minChi2PV(nullptr); + if ( chi2DefToUse == 5 ) { + minChi2PV = + findAssocPV(track, candPV, candRefPV, m_pvTypesToConsider, + m_minNTracksInPV, useRefittedPvs); + } else { + minChi2PV = + findMinChi2PV(track, candPV, candRefPV, m_pvTypesToConsider, + m_minNTracksInPV, useRefittedPvs, + doDCAin3D, chi2DefToUse).first; + } // if chi2DefToUse + if ( candPV == minChi2PV + || (candRefPV != nullptr && candRefPV == minChi2PV) ) { + bits |= track_type_bit[i]; + } + } // if m_trackTypesUsed + } // for i + + } // if candPV != NULL + + return bits; + } + //-------------------------------------------------------------------------- + // findAllTracksInDecay: returns a vector of xAOD::TrackParticle objects + // found in this vertex and subsequent decay vertices (if chosen). + //-------------------------------------------------------------------------- + TrackBag BPhysVertexTrackBase::findAllTracksInDecay(xAOD::BPhysHelper& vtx) + const { + + TrackBag tracks; + findAllTracksInDecay(vtx, tracks); + + return tracks; + } + //-------------------------------------------------------------------------- + // findAllTracksInDecay: fills a vector of xAOD::TrackParticle objects + // found in this vertex and subsequent decay vertices (if chosen). + // Method avoids duplicate entries in vector. + // Recursively calls itself if necessary. + //-------------------------------------------------------------------------- + void BPhysVertexTrackBase::findAllTracksInDecay(xAOD::BPhysHelper& vtx, + TrackBag& tracks) + const { + + for (unsigned int i=0; i < vtx.vtx()->nTrackParticles(); ++i) { + const xAOD::TrackParticle* track = vtx.vtx()->trackParticle(i); + if ( std::find(tracks.begin(),tracks.end(),track) == tracks.end() ) { + tracks.push_back(track); + } // if + } // for + // loop over preceeding vertices + if ( m_incPrecVerticesInDecay ) { + for (int ivtx = 0; ivtx < vtx.nPrecedingVertices(); ++ivtx) { + xAOD::BPhysHelper precVtx(vtx.precedingVertex(ivtx)); + findAllTracksInDecay(precVtx, tracks); + } // if + } // for + } + //-------------------------------------------------------------------------- + // findAllMuonsInDecay: returns a vector of xAOD::Muon objects + // found in this vertex and subsequent decay vertices (if chosen). + //-------------------------------------------------------------------------- + MuonBag BPhysVertexTrackBase::findAllMuonsInDecay(xAOD::BPhysHelper& vtx) + const { + + MuonBag muons; + findAllMuonsInDecay(vtx, muons); + + return muons; + } + //-------------------------------------------------------------------------- + // findAllMuonsInDecay: fills vector of xAOD::Muon objects + // found in this vertex and subsequent decay vertices (if chosen). + // Method avoids duplicate entries in vector. + // Recursively calls itself if necessary. + //-------------------------------------------------------------------------- + void BPhysVertexTrackBase::findAllMuonsInDecay(xAOD::BPhysHelper& vtx, + MuonBag& muons) + const { + + for (int i=0; i < vtx.nMuons(); ++i) { + if ( std::find(muons.begin(),muons.end(),vtx.muon(i)) == muons.end() ) { + muons.push_back(vtx.muon(i)); + } // if + } // for + // loop over preceeding vertices + if ( m_incPrecVerticesInDecay ) { + for (int ivtx = 0; ivtx < vtx.nPrecedingVertices(); ++ivtx) { + xAOD::BPhysHelper precVtx(vtx.precedingVertex(ivtx)); + findAllMuonsInDecay(precVtx, muons); + } // for + } // if + } + //-------------------------------------------------------------------------- + // findAllMuonsIdTracksInDecay: returns a vector of xAOD::TrackParticle + // objects found in this vertex and subsequent decay vertices. + // Returns the tracks. + // The vector of track pointers reeturned may contain NULL elements. + //-------------------------------------------------------------------------- + TrackBag + BPhysVertexTrackBase::findAllMuonIdTracksInDecay(xAOD::BPhysHelper& vtx, + MuonBag& muons) const { + + TrackBag tracks; + muons = findAllMuonsInDecay(vtx); + + for (MuonBag::const_iterator muItr = muons.begin(); muItr != muons.end(); + ++muItr) { + const xAOD::TrackParticle* track = + (*muItr)->trackParticle(xAOD::Muon::InnerDetectorTrackParticle); + tracks.push_back(track); + } // for + + return tracks; + } + //-------------------------------------------------------------------------- + // findMuonRefTrackMomenta: returns a vector<TVector3> containing the + // three momenta of refitted tracks identified as muons. + // The vector may contain (0,0,0) elements indicating an error. + //-------------------------------------------------------------------------- + std::vector<TVector3> + BPhysVertexTrackBase::findMuonRefTrackMomenta(xAOD::BPhysHelper& vtx, + MuonBag& muons) const { + + std::vector<TVector3> refMuTracks; + + // quick solution if nRefTrks == nMuons: + if ( vtx.nRefTrks() == vtx.nMuons() && !m_incPrecVerticesInDecay ) { + muons = vtx.muons(); + for ( auto refMuTrack : vtx.refTrks() ) { + refMuTracks.push_back(refMuTrack); + } + } else { + TrackBag muonIdTracks = findAllMuonIdTracksInDecay(vtx, muons); + if ( vtx.nRefTrks() == (int)vtx.vtx()->nTrackParticles() ) { + for (int i=0; i<vtx.nRefTrks(); ++i) { + const xAOD::TrackParticle* otp = + (const xAOD::TrackParticle*)vtx.refTrkOrigin(i); + if ( otp != NULL ) { + if ( std::find(muonIdTracks.begin(), muonIdTracks.end(), otp) + != muonIdTracks.end() ) { + refMuTracks.push_back(vtx.refTrk(i)); + } + } else { + ATH_MSG_WARNING("BPhysVertexTrackBase::findMuonRefTrackMomenta():" + " refTrkOrigin == NULL for refTrk # " + << i << " !"); + } + } // for + } else { + ATH_MSG_WARNING("BPhysVertexTrackBase::findMuonRefTrackMomenta():" + " size mismatch #refTrks = " << vtx.nRefTrks() + << "#trackParticles = " << vtx.vtx()->nTrackParticles() + << " !"); + } // if nRefTracks == nTrackParticles + // loop over preceeding vertices -- only if not all refMuTrks found yet + if ( m_incPrecVerticesInDecay && muons.size() > refMuTracks.size() ) { + for (int ivtx = 0; ivtx < vtx.nPrecedingVertices(); ++ivtx) { + xAOD::BPhysHelper precVtx(vtx.precedingVertex(ivtx)); + std::vector<TVector3> precRefMuTracks = + findMuonRefTrackMomenta(precVtx, muons); + // append only if not yet contained in + for ( auto precRefMuTrack : precRefMuTracks ) { + if ( std::find(refMuTracks.begin(), refMuTracks.end(), + precRefMuTrack) == refMuTracks.end() ) { + refMuTracks.push_back(precRefMuTrack); + } // if + } // for + } // for ivtx + } // if + } // if (shortcut) + + // debug output + if ( msgLvl( MSG::DEBUG ) ) { + ATH_MSG_DEBUG("BPhysVertexTrackBase::findMuonRefTrackMomenta():" + << " #muons: " << muons.size() + << " #refMuTrks: " << refMuTracks.size()); + TString str = Form(">> refMuTracks(%d):\n", (int)refMuTracks.size()); + for (unsigned int i=0; i < refMuTracks.size(); ++i) { + str += Form("(%10.4f,%10.4f,%10.4f) ", + refMuTracks[i].x(), refMuTracks[i].y(), + refMuTracks[i].z()); + } + ATH_MSG_DEBUG(str.Data()); + } + + return refMuTracks; + } + + //-------------------------------------------------------------------------- + // selectTracks: returns a vector of xAOD::TrackParticle objects + // seleted from the input track collection according to the selection + // criteria and with respect to the B candidate vertex. + //-------------------------------------------------------------------------- + TrackBag BPhysVertexTrackBase::selectTracks(const + xAOD::TrackParticleContainer* + inpTracks, + xAOD::BPhysHelper& cand, + const unsigned int ipv, + const unsigned int its, + const unsigned int itt) const { + + return selectTracks(inpTracks, findAllTracksInDecay(cand), cand, + ipv, its, itt); + } + //-------------------------------------------------------------------------- + // selectTracks: returns a vector of xAOD::TrackParticle objects + // seleted from the input track collection according to the selection + // criteria and with respect to the B candidate vertex. + //-------------------------------------------------------------------------- + TrackBag BPhysVertexTrackBase::selectTracks(const + xAOD::TrackParticleContainer* + inpTracks, + const TrackBag& exclTracks, + xAOD::BPhysHelper& cand, + const unsigned int ipv, + const unsigned int its, + const unsigned int itt) const { + + const xAOD::Vertex* candRefPV = cand.pv(m_pvAssocTypes[ipv]); + const xAOD::Vertex* candPV = cand.origPv(m_pvAssocTypes[ipv]); + + ATH_MSG_DEBUG("selectTracks: Found " << exclTracks.size() + << " " << exclTracks + << " for decay candidate " << cand.vtx() + << "; candPV: " << candPV << " candRefPV: " << candRefPV); + + std::string bname(buildBranchBaseName(its, ipv, itt)); + + // tracks to be considered + TrackBag tracks; + for (xAOD::TrackParticleContainer::const_iterator trkItr = + inpTracks->begin(); trkItr != inpTracks->end(); ++trkItr) { + const xAOD::TrackParticle* track = *trkItr; + uint64_t trackTypesForTrack(0x0); + // debug track types (before any cuts) + if ( m_debugTrackTypes > 0 ) { + trackTypesForTrack = detTrackTypes(track, candPV, candRefPV); + m_mttc->addToCounter(trackTypesForTrack, itt, bname, "all"); + } + // track selection check + if ( ! m_trackSelectionTools[its]->accept(*track, candRefPV) ) continue; + // debug track types (after track selection cuts) + if ( m_debugTrackTypes > 0 ) { + m_mttc->addToCounter(trackTypesForTrack, itt, bname, "ats"); + } + + // calcluation of track type bits not necessary if all bits requested + if ( ! ((unsigned int)m_useTrackTypes[itt] == ttall() || + (unsigned int)m_useTrackTypes[itt] == ttallMin()) ) { + // track type check -- determination if not in debugging mode + // delayed for execution speed reasons + if ( trackTypesForTrack == 0x0 ) { + trackTypesForTrack = detTrackTypes(track, candPV, candRefPV); + } + if ( (trackTypesForTrack & m_useTrackTypes[itt]) == 0x0 ) { + continue; + } + } + // debug track types (after track type cuts) + if ( m_debugTrackTypes > 0 ) { + m_mttc->addToCounter(trackTypesForTrack, itt, bname, "att"); + } + // track not in list of tracks to exclude + if ( std::find(exclTracks.begin(), exclTracks.end(), track) + != exclTracks.end() ) continue; + // debug track types (after all cuts) + if ( m_debugTrackTypes > 0 ) { + m_mttc->addToCounter(trackTypesForTrack, itt, bname, "fin"); + } + // tracks that survived so far + tracks.push_back(track); + } // for + + return tracks; + } + //-------------------------------------------------------------------------- + // buildBranchBaseName: build branch name from track selection, primary + // vertex association and track type qualifiers. + //-------------------------------------------------------------------------- + std::string BPhysVertexTrackBase::buildBranchBaseName(unsigned int its, + unsigned int ipv, + unsigned int itt, + std::string preSuffix) + const { + + ATH_MSG_DEBUG("BPhysVertexTrackBase::buildBranchBaseName -- begin"); + + std::string tsName = m_trackSelectionTools[its].name(); + std::string pvAssoc = xAOD::BPhysHelper::pv_type_str[m_pvAssocTypes[ipv]]; + + // need to get part of tsname after last underscore + std::size_t ipos = tsName.find_last_of("_"); + if ( ipos != std::string::npos ) tsName = tsName.substr(ipos+1); + + // format it nicely + boost::format f("T%010d_%s_%s%s%s"); + f % m_useTrackTypes[itt] % tsName % pvAssoc; + f % (preSuffix.length() > 0 ? "_"+preSuffix : ""); + f % (m_branchSuffix.length() > 0 ? "_"+m_branchSuffix : ""); + + ATH_MSG_DEBUG("BPhysVertexBaseTrackBase::buildBranchBaseName: " << f.str()); + + return f.str(); + } + //-------------------------------------------------------------------------- + // + // Initialize PV-to-SV association type vector + // + //-------------------------------------------------------------------------- + void BPhysVertexTrackBase::initPvAssocTypeVec() { + + m_pvAssocTypes.clear(); + for (unsigned int i=0; i<xAOD::BPhysHelper::n_pv_types; ++i) { + if ( (m_doVertexType & (1 << i)) > 0 ) + m_pvAssocTypes.push_back((xAOD::BPhysHelper::pv_type)i); + } + } + //-------------------------------------------------------------------------- + // + // Find primary vertex to which a track is closest to in terms of minimum + // chi2 to any primary vertex. Replace primary vertex by refitted primary + // vertex (for B candidate associated primary vertices) + // if appropriate (and available). + // Only consider primary vertices of specified primary vertex types and + // with a minimum number of tracks. + // + //-------------------------------------------------------------------------- + std::pair<const xAOD::Vertex*, double> + BPhysVertexTrackBase::findMinChi2PV(const xAOD::TrackParticle* track, + const xAOD::Vertex* candPV, + const xAOD::Vertex* candRefPV, + const std::vector<uint64_t>& pvtypes, + const int minNTracksInPV, + const bool useRefittedPvs, + const bool doDCAin3D, + const int chi2DefToUse) const { + + double minChi2 = std::numeric_limits<double>::max(); + const xAOD::Vertex* minChi2PV(nullptr); + + for (auto pvtx: *m_pvtxContainer) { + if ( pvtx != nullptr ) { + if ( std::find(pvtypes.begin(),pvtypes.end(),pvtx->vertexType()) + != pvtypes.end() ) { + const xAOD::Vertex* cvtx = pvtx; + // replace by refitted PV if associated PV matches orignal PV + if ( useRefittedPvs && pvtx == candPV ) { + if ( candRefPV != nullptr ) { + cvtx = candRefPV; + } else { + ATH_MSG_WARNING(" BPhysVertexTrackBase::findMinChi2PV:" + << " candRefPV == NULL!"); + continue; + } + } // if pvtx == candPV + if ( (int)cvtx->nTrackParticles() >= minNTracksInPV ) { + double chi2 = getTrackLogChi2DCA(track, cvtx, doDCAin3D, + chi2DefToUse)[4]; + if ( chi2 < minChi2 ) { + minChi2 = chi2; + minChi2PV = cvtx; + } // if chi2 < minChi2 + } // if minNTracksInPV + } // if pvTypes in pvtypes vector + } // if pvtx != nullptr + } // for pvtx + + return std::make_pair(minChi2PV, minChi2); + } + //-------------------------------------------------------------------------- + // + // Find primary vertex to which a track is closest using the + // TrackVertexAssociationTool. Replace primary vertex by refitted primary + // vertex (for B candidate associated primary vertices) + // if appropriate (and available). + // Only consider primary vertices of specified primary vertex types and + // with a minimum number of tracks. + // + //-------------------------------------------------------------------------- + const xAOD::Vertex* + BPhysVertexTrackBase::findAssocPV(const xAOD::TrackParticle* track, + const xAOD::Vertex* candPV, + const xAOD::Vertex* candRefPV, + const std::vector<uint64_t>& pvtypes, + const int minNTracksInPV, + const bool useRefittedPvs) const { + + // select PVs to be considered/replace candPV by candRefPV if requested + std::vector<const xAOD::Vertex*> vpvtx; + for (auto pvtx: *m_pvtxContainer) { + if ( pvtx != nullptr ) { + if ( std::find(pvtypes.begin(),pvtypes.end(),pvtx->vertexType()) + != pvtypes.end() ) { + const xAOD::Vertex* cvtx = pvtx; + // replace by refitted PV if associated PV matches orignal PV + if ( useRefittedPvs && pvtx == candPV ) { + if ( candRefPV != nullptr ) { + cvtx = candRefPV; + } else { + ATH_MSG_WARNING("BPhysVertexTrackBase::findAssocPV:" + << " candRefPV == NULL!"); + continue; + } + } // if pvtx == candPV + if ( (int)cvtx->nTrackParticles() >= minNTracksInPV ) { + vpvtx.push_back(cvtx); + } // if minNTracksInPV + } // if pvTypes in pvtypes vector + } // if pvtx != nullptr + } // for pvtx + + const xAOD::Vertex* assocPV(NULL); + if ( useRefittedPvs && m_tvaToolHasWpLoose ) { + // check whether track is in refitted PV - if so accept + // Need to do this here as the TrackVertexAssociationTool + // with WP 'Loose' only checks the track->vertex() pointer + // which always points to the original PV. + for (const auto &tp : candRefPV->trackParticleLinks()) { + if ( *tp == track ) { + // track is part of refitted PV -- accept it + assocPV = candRefPV; + break; + } + } // for tp + // if not matching use the TrackVertexAssociationTool (other PVs etc) + if ( assocPV == nullptr ) { + assocPV = m_tvaTool->getUniqueMatchVertex(*track, vpvtx); + } + } else { + assocPV = m_tvaTool->getUniqueMatchVertex(*track, vpvtx); + } // if useRefittedPvs && m_tvaToolHasWpLoose + if ( assocPV == nullptr ) { + ATH_MSG_WARNING("BPhysVertexTrackBase::findAssocPV:" + << " assocPV == NULL for track!" + << " len(vpvtx) = " << vpvtx.size() + << " useRefittedPvs = " << useRefittedPvs + << " minNTracksInPV = " << minNTracksInPV); + } + + return assocPV; + } + //-------------------------------------------------------------------------- +} diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/BTrackVertexMapLogger.cxx b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/BTrackVertexMapLogger.cxx new file mode 100644 index 00000000000..aa3a40ab2a5 --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/BTrackVertexMapLogger.cxx @@ -0,0 +1,104 @@ +/* + Copyright (C) 2002-2017 CERN for the benefit of the ATLAS collaboration +*/ + +//============================================================================ +// BTrackVertexMapLogger.cxx +//============================================================================ +// +// Author : Wolfgang Walkowiak <Wolfgang.Walkowiak@cern.ch.> +// Changes: +// - w.w., 2017-01-22: Added use of BPhysMetaDataTool. +// +// Store JO metadata in the output file. +// +// It uses the BPhysMetaDataTool (default) or the IOVDbMetaDataTool to +// store job option information as metadata in a specific branch whose +// name needs to prefixed by the deriviation format name. +// Note: Metadata stored by the IOVDbMetaDataTool is not readable on +// 'RootCore' level. +// +// This is a base class. Inherit from it to add the job options you want +// to store. For a usage example, see +// Bmumu_metadata.h / Bmumu_metadata.cxx +// and +// BPHY8.py . +// +// Job options provided by the base class: +// - DerivationName -- assign the name of the derivation format +// - MetadataFolderName -- assign the name of the metadata folder, +// should start with the derivation format name, +// defaults to DerivationName if not set. +// - UseIOVDbMetaDataTool -- use the IOVDbMetaDataTool to store +// the additional metadata +// - UseBPhysMetaDataTool -- use the BPhysMetaDataTool to store +// the additional metadata +// +//============================================================================ +// + +#include "DerivationFrameworkBPhys/BTrackVertexMapLogger.h" +#include "AthenaPoolUtilities/CondAttrListCollection.h" + +namespace DerivationFramework { + + //-------------------------------------------------------------------------- + BTrackVertexMapLogger::BTrackVertexMapLogger(const std::string& t, + const std::string& n, + const IInterface* p) + : AthAlgTool(t,n,p) { + + declareInterface<DerivationFramework::IAugmentationTool>(this); + + // Declare BPhysTrackVertexMapTool handles + declareProperty("TrackVertexMapTools", m_ttvmTools); + + // Enable log output? + declareProperty("Enable", m_enable = true); + } + //-------------------------------------------------------------------------- + StatusCode BTrackVertexMapLogger::initialize() { + + ATH_MSG_DEBUG("BTrackVertexMapLogger::initialize() -- begin"); + + // get the BPhysTrackVertexMapTools + if ( m_enable ) { + for (auto ttvmTool : m_ttvmTools) { + ATH_CHECK( ttvmTool.retrieve() ); + ATH_MSG_INFO("initialize: Successfully retrieved " + << ttvmTool.name() << " ...."); + } + } // if m_enable + + ATH_MSG_DEBUG("BTrackVertexMapLogger::initialize() -- end"); + + return StatusCode::SUCCESS; + } + //-------------------------------------------------------------------------- + StatusCode BTrackVertexMapLogger::finalize() { + + ATH_MSG_DEBUG("BTrackVertexMapLogger::finalize()"); + + // everything all right + return StatusCode::SUCCESS; + } + //-------------------------------------------------------------------------- + StatusCode BTrackVertexMapLogger::addBranches() const { + + ATH_MSG_DEBUG("BTrackVertexMapLogger::addBranches()"); + + // call the BPhysTrackVertexMapTools + if ( m_enable ) { + for (auto ttvmTool : m_ttvmTools) { + if ( ttvmTool->doLog() ) { + ATH_MSG_INFO("addBranches: dump for " << ttvmTool.name() << ":"); + ATH_CHECK( ttvmTool->logEvent() ); + } // if doLog() + } // for + } // if m_enable + + // still everything is ok + return StatusCode::SUCCESS; + } + //-------------------------------------------------------------------------- +} diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/BVertexClosestTrackTool.cxx b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/BVertexClosestTrackTool.cxx new file mode 100644 index 00000000000..e6771bb459f --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/BVertexClosestTrackTool.cxx @@ -0,0 +1,672 @@ +/* + Copyright (C) 2002-2019 CERN for the benefit of the ATLAS collaboration +*/ + +//============================================================================ +// BVertexClosestTrackTool.cxx +//============================================================================ +// +// Author : Wolfgang Walkowiak <Wolfgang.Walkowiak@cern.ch.> +// Changes: +// +// Add B vertex closest track information for different configurations, +// different track selections and different PV-to-SV association methods. +// +// For an usage example see BPHY8.py . +// +// Job options provided by this class: +// - CloseTrackChi2SetName -- list with labels for the following +// four settings (all five lists must +// be of exactly same length) +// - CloseTrackCorrChi2 -- list with options for using the +// SV uncertainties in the chi2 calculation +// in addition to track uncertainties +// 0 : use old method +// (only track uncertainties) +// 1 : from track perigee with +// uncertainties from track and vertex +// 2 : simple extrapolation from track +// parameters with uncertainties from +// track and vertex (extrapolation +// used for track swimming) +// - CloseTrackMinDCAin3D -- use 3-dimensional information in +// minimization (list) +// - CloseTrackMaxLogChi2 -- maximum chi2 distance of closest track +// to B vertex (list) +// - NCloseTrackMaxLogChi2 -- maximum chi2 distance of track +// to B vertex for track counting (list) +// +//============================================================================ +// +#include "DerivationFrameworkBPhys/BVertexClosestTrackTool.h" +#include "xAODMuon/MuonContainer.h" +#include "xAODEventInfo/EventInfo.h" +#include "xAODBPhys/BPhysHelper.h" +#include "InDetTrackSelectionTool/IInDetTrackSelectionTool.h" +#include "EventPrimitives/EventPrimitivesHelpers.h" +#include "AthLinks/ElementLink.h" + +#include "boost/format.hpp" +#include "TVector3.h" +#include <algorithm> +#include <sstream> + +namespace DerivationFramework { + + //------------------------------------------------------------------------- + // + // helper class + BVertexClosestTrackTool::CtItem::CtItem(std::string Name, std::string Prefix, + std::string Bname, + double Dca, double DcaErr, + double Zca, double ZcaErr, + double VtxNDErr2, double TrkNDErr2, double Phi0Used, + int NTrksChi2, + xAOD::TrackParticle* CloseTrack, + TrackBag Tracks, + std::vector<std::vector<double> > Vtap, + std::vector<unsigned short> Selpat) + : BaseItem(Name, Bname, Prefix), dca(Dca), dcaErr(DcaErr), + zca(Zca), zcaErr(ZcaErr), vtxNDErr2(VtxNDErr2), trkNDErr2(TrkNDErr2), + phi0Used(Phi0Used), + nTrksChi2(NTrksChi2), closeTrack(CloseTrack), + tracks(Tracks), vtap(Vtap), selpat(Selpat) { + } + + BVertexClosestTrackTool::CtItem::~CtItem() { + } + + void BVertexClosestTrackTool::CtItem::setup(std::string Name, + std::string Bname, + std::string Prefix) { + BaseItem::setup(Name, Bname, Prefix); + dca = -999.; + dcaErr = -99.; + zca = -999.; + zcaErr = -99.; + vtxNDErr2 = -99.; + trkNDErr2 = -99.; + phi0Used = -999.; + nTrksChi2 = 0; + closeTrack = NULL; + tracks.clear(); + vtap.clear(); + selpat.clear(); + } + + void BVertexClosestTrackTool::CtItem::setup(std::string Name, + std::string Bname, + std::string Prefix, + double Dca, double DcaErr, + double Zca, double ZcaErr, + double VtxNDErr2, double TrkNDErr2, double Phi0Used, + int NTrksChi2, + xAOD::TrackParticle* + CloseTrack, + TrackBag Tracks, + std::vector<std::vector<double> > Vtap, + std::vector<unsigned short> Selpat) { + BaseItem::setup(Name, Bname, Prefix); + dca = Dca; + dcaErr = DcaErr; + zca = Zca; + zcaErr = ZcaErr; + vtxNDErr2 = VtxNDErr2; + trkNDErr2 = TrkNDErr2; + phi0Used = Phi0Used; + nTrksChi2 = NTrksChi2; + closeTrack = CloseTrack; + tracks = Tracks; + vtap = Vtap; + selpat = Selpat; + } + + void BVertexClosestTrackTool::CtItem::resetVals() { + dca = -999.; + dcaErr = -99.; + zca = -999.; + zcaErr = -99.; + vtxNDErr2 = -99.; + trkNDErr2 = -99.; + phi0Used = -999.; + nTrksChi2 = 0; + closeTrack = NULL; + tracks.clear(); + vtap.clear(); + selpat.clear(); + } + + void BVertexClosestTrackTool::CtItem::copyVals(const BaseItem& item) { + copyVals((const CtItem&)item); + } + + void BVertexClosestTrackTool::CtItem::copyVals(const CtItem& item) { + dca = item.dca; + dcaErr = item.dcaErr; + zca = item.zca; + zcaErr = item.zcaErr; + vtxNDErr2 = item.vtxNDErr2; + trkNDErr2 = item.trkNDErr2; + phi0Used = item.phi0Used; + nTrksChi2 = item.nTrksChi2; + closeTrack = item.closeTrack; + tracks = item.tracks; + vtap = item.vtap; + selpat = item.selpat; + } + + std::string BVertexClosestTrackTool::CtItem::dcaName() { + return buildName("DCA"); + } + + std::string BVertexClosestTrackTool::CtItem::dcaErrName() { + return buildName("DCAError"); + } + + std::string BVertexClosestTrackTool::CtItem::zcaName() { + return buildName("ZCA"); + } + + std::string BVertexClosestTrackTool::CtItem::zcaErrName() { + return buildName("ZCAError"); + } + + std::string BVertexClosestTrackTool::CtItem::vtxNDErr2Name() { + return buildName("VtxNDError2"); + } + + std::string BVertexClosestTrackTool::CtItem::trkNDErr2Name() { + return buildName("TrkNDError2"); + } + + std::string BVertexClosestTrackTool::CtItem::phi0UsedName() { + return buildName("Phi0Used"); + } + + std::string BVertexClosestTrackTool::CtItem::nTrksChi2Name() { + return buildName("NTracksChi2"); + } + + std::string BVertexClosestTrackTool::CtItem::closeTrackName() { + return buildName("CloseTrack", "_Link"); + } + + std::string BVertexClosestTrackTool::CtItem::toString() const { + boost::format f1("dca: %10.6f %10.6f zca: %10.6f %10.6f nt: %10d"); + f1 % dca % dcaErr % zca % zcaErr % nTrksChi2; + boost::format f2("%s\n %s\n"); + f2 % BPhysVertexTrackBase::BaseItem::toString(); + f2 % f1.str(); + std::string rstr = f2.str(); + rstr += "per track: p(px, py, pz)\n"; + rstr += " d(d0, z0, phi, theta, qoverp)\n"; + rstr += " d0, d0Err, z0, z0Err, logChi2, dca, okFlag\n"; + rstr += " vtxNDErr2, trkNDErr2, phi0Used\n"; + rstr += " vtxNDErr, trkNDErr, log(chi2Err2Sum)\n"; + // loop over tracks + if (tracks.size() == vtap.size() && vtap.size() == selpat.size()) { + for (unsigned int i=0; i<tracks.size(); ++i) { + boost::format f3(" %3d %2d "); + f3 % i % selpat[i]; + std::string f3str = f3.str(); + // 0: d0, 1: d0Err, 2: z0, 3: z0Err, 4: logChi2, 5: dca, 6: okFlag + // 7: vtxNDErr2, 8: trkNDErr2, 9: phi0Used + boost::format f4("%s\nd0: %10.4f %10.4f z0: %10.4f %10.4f " + "lc2: %10.4f dca: %10.4f ok: %3f\n" + "vtxNDErr2: %10.4f trkNDErr2: %10.4f " + "phi0Used: %10.4f\n" + "vtxNDErr: %10.4f trkNDErr2 %10.4f " + "logChi2Err2Sum: %10.4f"); + f4 % trackToString(tracks[i]); + f4 % vtap[i][0] % vtap[i][1] % vtap[i][2] % vtap[i][3]; + f4 % vtap[i][4] % vtap[i][5] % vtap[i][6]; + f4 % vtap[i][7] % vtap[i][8] % vtap[i][9]; + f4 % (vtap[i][7] < 0. ? -99. : sqrt(vtap[i][7])); + f4 % (vtap[i][8] < 0. ? -99. : sqrt(vtap[i][8])); + f4 % (vtap[i][7]+vtap[i][8] > 0. ? + log(vtap[i][5]*vtap[i][5]/(vtap[i][7]+vtap[i][8])) : -999.); + std::string tstr = wrapLines(f4.str(), + std::string(f3str.length(), ' ')); + tstr.replace(0,f3str.length(),f3str); + rstr.append(tstr+"\n"); + } // for i + } else { + boost::format f5("Mismatch: nTracks: %d nVtap: %d nSelpat: %d\n"); + f5 % tracks.size() % vtap.size() % selpat.size(); + rstr.append(f5.str()); + } // if sizes + + rstr.erase(rstr.length()-1); + return rstr; + } + + //-------------------------------------------------------------------------- + BVertexClosestTrackTool::BVertexClosestTrackTool(const std::string& t, + const std::string& n, + const IInterface* p) + : BPhysVertexTrackBase(t,n,p), m_lastRunNumber(0), m_lastEvtNumber(0), + m_svIdx(0) { + + declareInterface<DerivationFramework::IAugmentationTool>(this); + + declareProperty("CloseTrackChi2SetName", m_closeTrackChi2SetName = {"def"}); + declareProperty("CloseTrackCorrChi2" , m_closeTrackCorrChi2 = {0}); + declareProperty("CloseTrackMinDCAin3D" , m_minDCAin3D = {true}); + declareProperty("CloseTrackMaxLogChi2" , m_closeTrackMaxLogChi2 = {-1.}); + declareProperty("NCloseTrackMaxLogChi2", m_nCloseTrackMaxLogChi2 = {-1.}); + } + //-------------------------------------------------------------------------- + StatusCode BVertexClosestTrackTool::initializeHook() { + + ATH_MSG_DEBUG("BVertexClosestTrackTool::initializeHook() -- begin"); + + // check job options + if ( m_closeTrackChi2SetName.size() == 0 ) { + ATH_MSG_ERROR("No chi2 set names provided!"); + } + if ( m_closeTrackCorrChi2.size() == 0 ) { + ATH_MSG_ERROR("No use of corrected chi2 settings provided!"); + } + if ( m_minDCAin3D.size() == 0 ) { + ATH_MSG_ERROR("No use of min DCA in 3D settings provided!"); + } + if ( m_closeTrackMaxLogChi2.size() == 0 ) { + ATH_MSG_ERROR("No cuts on max log chi2 for DOCA calculation provided!"); + } + if ( m_nCloseTrackMaxLogChi2.size() == 0 ) { + ATH_MSG_ERROR("No cuts on max log chi2 for nClosetTracks calculation " + "provided!"); + } + if ( m_closeTrackCorrChi2.size() != m_closeTrackChi2SetName.size() || + m_minDCAin3D.size() != m_closeTrackChi2SetName.size() || + m_closeTrackMaxLogChi2.size() != m_closeTrackChi2SetName.size() || + m_nCloseTrackMaxLogChi2.size() != m_closeTrackChi2SetName.size() ) { + ATH_MSG_ERROR("Size mismatch of CloseTrackChi2SetName (" + << m_closeTrackChi2SetName.size() << "), " + << "CloseTrackCorrChi2 (" + << m_closeTrackCorrChi2 << "), " + << "CloseTrackMinDCAin3D (" + << m_minDCAin3D.size() << "), " + << "CloseTrackMaxLogChi2 (" + << m_closeTrackMaxLogChi2.size() << ") and/or " + << "NCloseTrackMaxLogChi2 (" + << m_nCloseTrackMaxLogChi2.size() << ")"); + } + + // initialize results array + initResults(); + + ATH_MSG_DEBUG("BVertexClosestTrackTool::initializeHook() -- end"); + + return StatusCode::SUCCESS; + } + //-------------------------------------------------------------------------- + StatusCode BVertexClosestTrackTool::finalizeHook() { + + ATH_MSG_DEBUG("BVertexClosestTrackTool::finalizeHook()"); + + // everything all right + return StatusCode::SUCCESS; + } + //-------------------------------------------------------------------------- + StatusCode + BVertexClosestTrackTool::addBranchesVCSetupHook(size_t ivc) const { + + ATH_MSG_DEBUG("BVertexClosestTrackTool::addBranchesVCLoopHook() -- begin"); + + ATH_MSG_DEBUG("BVertexClosestTrackTool::addBranchesVCSetupHook: " + << "Vertex container index " << ivc + << " for collection " << m_vertexContainerNames[ivc] + << " with prefix " << m_branchPrefixes[ivc]); + + setResultsPrefix(m_branchPrefixes[ivc]); + + ATH_MSG_DEBUG("BVertexClosestTrackTool::addBranchesVCSetupHook() -- end"); + + // nothing to do here + return StatusCode::SUCCESS; + } + + //-------------------------------------------------------------------------- + StatusCode + BVertexClosestTrackTool::addBranchesSVLoopHook(const xAOD::Vertex* vtx) const { + + ATH_MSG_DEBUG("BVertexClosestTrackTool::addBranchesSVLoopHook() -- begin"); + + // calculate closest track values + ATH_MSG_DEBUG("BVertexClosestTrackTool::addBranchesSVLoopHook(): " + "calculate closest track ..."); + CHECK(calculateValues(vtx)); + + // save the closest track values + ATH_MSG_DEBUG("BVertexClosestTrackTool::addBranchesSVLoopHook(): " + "save closest track ..."); + CHECK(saveClosestTrack(vtx)); + + // dump close track item debugging information + if (m_debugTracksInThisEvent) { + CHECK(logCloseTracksDebugInfo()); + } + + ATH_MSG_DEBUG("BVertexClosestTrackTool::addBranchesSVLoopHook() -- end"); + + // nothing to do here + return StatusCode::SUCCESS; + } + //-------------------------------------------------------------------------- + // Calculate closest track variables -- method called from caching loop + //-------------------------------------------------------------------------- + StatusCode + BVertexClosestTrackTool::calcValuesHook(const xAOD::Vertex* vtx, + const unsigned int ipv, + const unsigned int its, + const unsigned int itt) const { + + ATH_MSG_DEBUG("calcValuesHook: ipv: " << ipv + << ", its: " << its << ", itt: " << itt); + + // candidate tracks and momentum + xAOD::BPhysHelper cand(vtx); + + // tracks to be considered + TrackBag tracks = selectTracks(m_tracks, cand, ipv, its, itt); + + // loop over chi2 setting sets + unsigned int nChi2Sets = m_closeTrackChi2SetName.size(); + for (unsigned int ics = 0; ics < nChi2Sets; ++ics) { + + CtItem& cti = m_results[its][ipv][itt][ics]; + + // presets + cti.resetVals(); + + double closestTrkDCA = 9999.; + int closestTrkIdx(-1); + unsigned int trkIdx(0); + for (TrackBag::const_iterator trkItr = tracks.begin(); + trkItr != tracks.end(); ++trkItr, ++trkIdx) { + + // + // track selection bit pattern: + // bit 0 : included in number of close tracks + // bit 1 : chosen as closest track + // + unsigned short selpat(0); + + // Returned vector components: + // 0: d0, 1: d0Err, 2: z0, 3: z0Err, 4: logChi2, 5: dca, 6: okFlag + // 7: vtxErrPart2, 8: trkErrPart2, 9: phi0Used + std::vector<double> vtap = + getTrackLogChi2DCA(*trkItr, cand.vtx(), + m_minDCAin3D[ics], m_closeTrackCorrChi2[ics]); + ATH_MSG_DEBUG("calcValuesHook: track: " << *trkItr + << ", logChi2: " << vtap[4] << ", dca: " << vtap[5]); + + // track values at perigee found? + if ( vtap[6] >= 0. ) { + ATH_MSG_DEBUG("calcValuesHook: checking track count for " + "m_nCloseTrackMaxLogChi2[ics] = " + << m_nCloseTrackMaxLogChi2[ics]); + // count tracks + if ( vtap[4] < m_nCloseTrackMaxLogChi2[ics] ) { + cti.nTrksChi2++; + selpat |= 1; + ATH_MSG_DEBUG("calcValuesHook: nTrksChi2++ for track " << *trkItr); + } + // find closest track + ATH_MSG_DEBUG("calcValuesHook: log(chi2) check: " + "m_closeTrackMaxLogChi2[ics]: " + << m_closeTrackMaxLogChi2[ics] + << ", logChi2: " << vtap[4] + << ", closestTrkDCA: " << closestTrkDCA + << ", dca: " << fabs(vtap[5])); + if ( fabs(vtap[5]) < closestTrkDCA && + vtap[4] < m_closeTrackMaxLogChi2[ics] ) { + closestTrkDCA = fabs(vtap[5]); + cti.dca = vtap[0]; + cti.dcaErr = vtap[1]; + cti.zca = vtap[2]; + cti.zcaErr = vtap[3]; + cti.vtxNDErr2 = vtap[7]; + cti.trkNDErr2 = vtap[8]; + cti.phi0Used = vtap[9]; + cti.closeTrack = *trkItr; + closestTrkIdx = trkIdx; + ATH_MSG_DEBUG("calcValuesHook: closestTrkDCA now: " + << closestTrkDCA + << " for track " << *trkItr); + } + } // if ok + cti.tracks.push_back(*trkItr); + cti.vtap.push_back(vtap); + cti.selpat.push_back(selpat); + } // for tracks + // mark closest track + if (closestTrkIdx > -1 && closestTrkIdx < (int)cti.selpat.size()) { + cti.selpat[closestTrkIdx] |= 2; + } + } // for ics + + return StatusCode::SUCCESS; + } + //-------------------------------------------------------------------------- + // Fill closest track values from cache if found + //-------------------------------------------------------------------------- + bool BVertexClosestTrackTool::fastFillHook(const xAOD::Vertex* vtx, + const int ipv) const { + + ATH_MSG_DEBUG("fastFillHook: ipv: " << ipv); + + bool found(false); + + StringIntMap_t::iterator itpv = + m_pvAssocResMap.find(buildPvAssocCacheName(vtx, ipv)); + if ( itpv != m_pvAssocResMap.end() ) { + found = true; + unsigned int nTrackSels = m_trackSelectionTools.size(); + unsigned int nTrackTypes = m_useTrackTypes.size(); + unsigned int nChi2Sets = m_closeTrackChi2SetName.size(); + for (unsigned int its = 0; its < nTrackSels; ++its) { + for (unsigned int itt = 0; itt < nTrackTypes; ++itt) { + for (unsigned int ics = 0; ics < nChi2Sets; ++ics) { + m_results[its][ipv][itt][ics] + .copyVals(m_results[its][itpv->second][itt][ics]); + } // for ics + } // for its + } // for itt + } // if found + + ATH_MSG_DEBUG("fastFillHook: cache index: " + << buildPvAssocCacheName(vtx, ipv) + << ", found ? " << found + << ", ipv_ref: " + << (found ? itpv->second : -1)); + + return found; + } + //-------------------------------------------------------------------------- + StatusCode + BVertexClosestTrackTool::saveClosestTrack(const xAOD::Vertex* vtx) const { + + typedef ElementLink< xAOD::TrackParticleContainer > TrackParticleLink_t; + + unsigned int nTrackSels = m_trackSelectionTools.size(); + unsigned int nPvAssocs = m_pvAssocTypes.size(); + unsigned int nTrackTypes = m_useTrackTypes.size(); + unsigned int nChi2Sets = m_closeTrackChi2SetName.size(); + + for (unsigned int its = 0; its < nTrackSels; ++its) { + for (unsigned int ipv = 0; ipv < nPvAssocs; ++ipv) { + for (unsigned int itt = 0; itt < nTrackTypes; ++itt) { + for (unsigned int ics = 0; ics < nChi2Sets; ++ics) { + CtItem result = m_results[its][ipv][itt][ics]; + SG::AuxElement::Decorator< float > + d_dca_value(result.dcaName()); + SG::AuxElement::Decorator< float > + d_dcaErr_value(result.dcaErrName()); + SG::AuxElement::Decorator< float > + d_zca_value(result.zcaName()); + SG::AuxElement::Decorator< float > + d_zcaErr_value(result.zcaErrName()); + SG::AuxElement::Decorator< int > + d_nTrksChi2_value(result.nTrksChi2Name()); + d_dca_value(*vtx) = result.dca; + d_dcaErr_value(*vtx) = result.dcaErr; + d_zca_value(*vtx) = result.zca; + d_zcaErr_value(*vtx) = result.zcaErr; + d_nTrksChi2_value(*vtx) = result.nTrksChi2; + ATH_MSG_DEBUG("BVertexClosestTrackTool::saveClosestTrack() " + << "-- dca: " << result.dcaName() + << ", dcaErr: " << result.dcaErrName() + << ", zca: " << result.zcaName() + << ", zcaErr: " << result.zcaErrName() + << ", nTrksChi2: " << result.nTrksChi2Name()); + ATH_MSG_DEBUG("BVertexClosestTrackTool::saveClosestTrack() " + << "-- vertex: (" + << vtx->x() << ", " + << vtx->y() << ", " + << vtx->z() << ")" + << ", dca: " << result.dca + << ", dcaErr: " << result.dcaErr + << ", zca: " << result.zca + << ", zcaErr: " << result.zcaErr + << ", nTrksChi2: " << result.nTrksChi2); + // add ElementLink to closest track + std::string linkName = result.closeTrackName(); + SG::AuxElement::Decorator<TrackParticleLink_t> + tpLinkDecor(linkName); + TrackParticleLink_t tpLink; + if ( result.closeTrack != NULL ) { + tpLink.toContainedElement( *m_tracks, result.closeTrack ); + } + ATH_MSG_DEBUG("saveClosestTrack: Decorate vtx " + << vtx << " with " << linkName + << ", closeTrkPtr: " << result.closeTrack); + tpLinkDecor(*vtx) = tpLink; + if ( tpLink.isValid() ) { + ATH_MSG_DEBUG("saveClosestTrack: Decorated vtx " + << vtx << " with " << linkName + << ", closeTrkPtr: " << result.closeTrack); + } else { + ATH_MSG_VERBOSE("saveClosestTrack: Failed to decorate vtx " + << vtx << " with " << linkName + << ", closeTrkPtr: " + << result.closeTrack << " !"); + } + // if valid + } // for ics + } // for itt + } // for ipv + } // for its + + return StatusCode::SUCCESS; + } + //-------------------------------------------------------------------------- + void BVertexClosestTrackTool::setResultsPrefix(std::string prefix) const { + + ATH_MSG_DEBUG("BVertexClosestTrackTool::setResultsPrefix -- begin"); + + unsigned int nTrackSels = m_trackSelectionTools.size(); + unsigned int nPvAssocs = m_pvAssocTypes.size(); + unsigned int nTrackTypes = m_useTrackTypes.size(); + unsigned int nChi2Sets = m_closeTrackChi2SetName.size(); + + for (unsigned int its = 0; its < nTrackSels; ++its) { + for (unsigned int ipv = 0; ipv < nPvAssocs; ++ipv) { + for (unsigned int itt = 0; itt < nTrackTypes; ++itt) { + for (unsigned int ics = 0; ics < nChi2Sets; ++ics) { + m_results[its][ipv][itt][ics].setPrefix(prefix); + } // for ics + } // for itt + } // for ipv + } // for its + + ATH_MSG_DEBUG("BVertexClosestTrackTool::setResultsPrefix -- end"); + } + //-------------------------------------------------------------------------- + void BVertexClosestTrackTool::initResults() { + + ATH_MSG_DEBUG("BVertexClosestTrackTool::initResults -- begin"); + + unsigned int nTrackSels = m_trackSelectionTools.size(); + unsigned int nPvAssocs = m_pvAssocTypes.size(); + unsigned int nTrackTypes = m_useTrackTypes.size(); + unsigned int nChi2Sets = m_closeTrackChi2SetName.size(); + + ATH_MSG_DEBUG("BVertexClosestTrackTool::initResults : nTrackSels = " + << nTrackSels); + ATH_MSG_DEBUG("BVertexClosestTrackTool::initResults : nPvAssocs = " + << nPvAssocs); + ATH_MSG_DEBUG("BVertexClosestTrackTool::initResults : nTrackTypes = " + << nTrackTypes); + m_results.resize(boost::extents[nTrackSels][nPvAssocs][nTrackTypes][nChi2Sets]); + for (unsigned int its = 0; its < nTrackSels; ++its) { + ATH_MSG_DEBUG("BVertexClosestTrackTool::initResults -- its = " << its); + for (unsigned int ipv = 0; ipv < nPvAssocs; ++ipv) { + ATH_MSG_DEBUG("BVertexClosestTrackTool::initResults -- ipv = " << ipv); + for (unsigned int itt = 0; itt < nTrackTypes; ++itt) { + ATH_MSG_DEBUG("BVertexClosestTrackTool::initResults -- itt = " + << itt); + for (unsigned int ics = 0; ics < nChi2Sets; ++ics) { + ATH_MSG_DEBUG("BVertexClosestTrackTool::initResults -- ics = " + << ics); + std::string csname = m_closeTrackChi2SetName[ics]; + ATH_MSG_DEBUG("BVertexClosestTrackTool::initResults : " + << m_branchBaseName << ", " + << buildBranchBaseName(its, ipv, itt, csname)); + m_results[its][ipv][itt][ics].setup(buildBranchBaseName(its, ipv, + itt, + csname), + m_branchBaseName); + } // for ics + } // for itt + } // for ipv + } // for its + + ATH_MSG_DEBUG("BVertexClosestTrackTool::initResults -- end"); + } + //-------------------------------------------------------------------------- + // Dump CloseTracks debug information to log file + //-------------------------------------------------------------------------- + StatusCode BVertexClosestTrackTool::logCloseTracksDebugInfo() const { + + // Count candidates + if (m_runNumber != m_lastRunNumber || m_evtNumber != m_lastEvtNumber) { + m_lastRunNumber = m_runNumber; + m_lastEvtNumber = m_evtNumber; + m_svIdx = 0; + } else { + m_svIdx++; + } + + std::string str(">>>>> logCloseTracksDebugInfo:\n"); + boost::format f("Run %d Event %d SV %d\n"); + f % m_runNumber % m_evtNumber % m_svIdx; + str.append(f.str()); + + unsigned int nTrackSels = m_trackSelectionTools.size(); + unsigned int nPvAssocs = m_pvAssocTypes.size(); + unsigned int nTrackTypes = m_useTrackTypes.size(); + unsigned int nChi2Sets = m_closeTrackChi2SetName.size(); + + for (unsigned int its = 0; its < nTrackSels; ++its) { + for (unsigned int ipv = 0; ipv < nPvAssocs; ++ipv) { + for (unsigned int itt = 0; itt < nTrackTypes; ++itt) { + for (unsigned int ics = 0; ics < nChi2Sets; ++ics) { + boost::format f1("its: %d ipv: %d itt: %d ics: %d\n"); + f1 % its % ipv % itt % its; + str.append(f1.str()); + CtItem result = m_results[its][ipv][itt][ics]; + str.append(result.toString()+"\n"); + } // for ics + } // for itt + } // for ipv + } // for its + + str.append("<<<<< logCloseTracksDebugInfo"); + ATH_MSG_INFO(str); + + return StatusCode::SUCCESS; + } + //-------------------------------------------------------------------------- +} diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/BVertexTrackIsoTool.cxx b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/BVertexTrackIsoTool.cxx new file mode 100644 index 00000000000..6bb527ea80e --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/BVertexTrackIsoTool.cxx @@ -0,0 +1,511 @@ +/* + Copyright (C) 2002-2019 CERN for the benefit of the ATLAS collaboration +*/ + +//============================================================================ +// BVertexTrackIsoTool.cxx +//============================================================================ +// +// Author : Wolfgang Walkowiak <Wolfgang.Walkowiak@cern.ch.> +// Changes: +// +// Add B vertex track isolation information for different configurations, +// different track selections and different PV-to-SV association methods. +// +// For an usage example see BPHY8.py . +// +// Job options provided by this class: +// - IsolationConeSizes -- List of isolation cone sizes +// - IsoTrkImpLogChi2Max -- List of maximum log(chi2) cuts for +// association of tracks to the primary +// vertex picked. +// - IsoDoTrkImpLogChi2Cut -- apply log(chi2) cuts +// 0 : don't apply log(chi2) cuts +// 1 : apply log(chi2) cuts +// 2 : apply log(chi2) cuts [former version] +// (The last two job options must +// contain the same number of elements +// as the IsolationConeSizes list.) +// - UseOptimizedAlgo -- Use the speed-optimized algorithm. +// +//============================================================================ +// +#include "DerivationFrameworkBPhys/BVertexTrackIsoTool.h" +#include "xAODEventInfo/EventInfo.h" +#include "xAODBPhys/BPhysHelper.h" +#include "InDetTrackSelectionTool/IInDetTrackSelectionTool.h" +#include "EventPrimitives/EventPrimitivesHelpers.h" + +#include "boost/format.hpp" +#include "TVector3.h" +#include <algorithm> +#include <sstream> + +namespace DerivationFramework { + + //------------------------------------------------------------------------- + // + // helper class + BVertexTrackIsoTool::IsoItem::IsoItem(std::string Name, + std::string Bname, + std::string Prefix, + double IsoValue, + int NTracks) : + BaseItem(Name, Bname, Prefix), isoValue(IsoValue), nTracks(NTracks) { + } + + BVertexTrackIsoTool::IsoItem::~IsoItem() { + } + + void BVertexTrackIsoTool::IsoItem::setup(std::string Name, + std::string Bname, + std::string Prefix) { + BaseItem::setup(Name, Bname, Prefix); + isoValue = -1.; + nTracks = 0; + } + + void BVertexTrackIsoTool::IsoItem::setup(std::string Name, + std::string Bname, + std::string Prefix, + double IsoValue, + int NTracks) { + BaseItem::setup(Name, Bname, Prefix); + isoValue = IsoValue; + nTracks = NTracks; + } + + void BVertexTrackIsoTool::IsoItem::resetVals() { + isoValue = -2.; + nTracks = -1; + } + + void BVertexTrackIsoTool::IsoItem::copyVals(const BaseItem& item) { + copyVals((const IsoItem&)item); + } + + void BVertexTrackIsoTool::IsoItem::copyVals(const IsoItem& item) { + isoValue = item.isoValue; + nTracks = item.nTracks; + } + + std::string BVertexTrackIsoTool::IsoItem::isoName() { + return buildName(); + } + + std::string BVertexTrackIsoTool::IsoItem::nTracksName() { + return buildName("Ntracks"); + } + + //-------------------------------------------------------------------------- + BVertexTrackIsoTool::BVertexTrackIsoTool(const std::string& t, + const std::string& n, + const IInterface* p) + : BPhysVertexTrackBase(t,n,p) { + + declareInterface<DerivationFramework::IAugmentationTool>(this); + + declareProperty("IsolationConeSizes" , m_isoConeSizes); + declareProperty("IsoTrkImpLogChi2Max" , m_isoTrkImpLogChi2Max); + declareProperty("IsoDoTrkImpLogChi2Cut" , m_isoDoTrkImpLogChi2Cut); + declareProperty("UseOptimizedAlgo" , m_useOptimizedAlgo = true); + } + //-------------------------------------------------------------------------- + StatusCode BVertexTrackIsoTool::initializeHook() { + + ATH_MSG_DEBUG("BVertexTrackIsoTool::initializeHook() -- begin"); + + // check like-sized arrays + if ( m_isoConeSizes.size() != m_isoTrkImpLogChi2Max.size() || + m_isoConeSizes.size() != m_isoDoTrkImpLogChi2Cut.size() ) { + ATH_MSG_ERROR("Size mismatch of IsolationConeSizes (" + << m_isoConeSizes.size() + << "), IsoTrkImpChi2Max (" + << m_isoTrkImpLogChi2Max.size() + << ") and IsoDoTrkImpChi2Cut (" + << m_isoDoTrkImpLogChi2Cut.size() << ") lists!"); + } + + // initialize results array + initResults(); + + // info output + ATH_MSG_INFO("calculateIsolation: using " + << (m_useOptimizedAlgo ? + "optimized (faster)" : "regular (slower)") + << "track isolation calculation methd."); + + ATH_MSG_DEBUG("BVertexTrackIsoTool::initializeHook() -- end"); + + return StatusCode::SUCCESS; + } + //-------------------------------------------------------------------------- + StatusCode BVertexTrackIsoTool::finalizeHook() { + + ATH_MSG_DEBUG("BVertexTrackIsoTool::finalizeHook()"); + + // everything all right + return StatusCode::SUCCESS; + } + //-------------------------------------------------------------------------- + StatusCode + BVertexTrackIsoTool::addBranchesVCSetupHook(size_t ivc) const { + + ATH_MSG_DEBUG("BVertexTrackIsoTool::addBranchesVCLoopHook() -- begin"); + + ATH_MSG_DEBUG("BVertexTrackIsoTool::addBranchesVCSetupHook: " + << "Vertex container index " << ivc + << " for collection " << m_vertexContainerNames[ivc] + << " with prefix " << m_branchPrefixes[ivc]); + + setResultsPrefix(m_branchPrefixes[ivc]); + + ATH_MSG_DEBUG("BVertexTrackIsoTool::addBranchesVCSetupHook() -- end"); + + // nothing to do here + return StatusCode::SUCCESS; + } + //-------------------------------------------------------------------------- + StatusCode + BVertexTrackIsoTool::addBranchesSVLoopHook(const xAOD::Vertex* vtx) const { + + ATH_MSG_DEBUG("BVertexTrackIsoTool::addBranchesSVLoopHook() -- begin"); + + ATH_MSG_DEBUG("BVertexTrackIsoTool::addBranchesSVLoopHook(): " + "calculate isolation ..."); + if ( m_useOptimizedAlgo ) { + CHECK(calculateValues(vtx)); + } else { + CHECK(calculateIsolation(vtx)); + } + ATH_MSG_DEBUG("BVertexTrackIsoTool::addBranchesSVLoopHook(): " + "save isolation ..."); + // save the isolation values + CHECK(saveIsolation(vtx)); + + ATH_MSG_DEBUG("BVertexTrackIsoTool::addBranchesSVLoopHook() -- end"); + + // nothing to do here + return StatusCode::SUCCESS; + } + //-------------------------------------------------------------------------- + // Calculate track isolation variables -- faster method with caching + //-------------------------------------------------------------------------- + StatusCode + BVertexTrackIsoTool::calcValuesHook(const xAOD::Vertex* vtx, + const unsigned int ipv, + const unsigned int its, + const unsigned int itt) const { + + ATH_MSG_DEBUG("calcValuesHook: ipv: " << ipv + << ", its: " << its << ", itt: " << itt); + + // candidate tracks and momentum + xAOD::BPhysHelper cand(vtx); + TVector3 candP = cand.totalP(); + const xAOD::Vertex* candRefPV = cand.pv(m_pvAssocTypes[ipv]); + + TrackBag tracks = selectTracks(m_tracks, cand, ipv, its, itt); + + // loop over isolation cones (pt and deltaR) + unsigned int nCones = m_isoConeSizes.size(); + for (unsigned int ic = 0; ic < nCones; ++ic) { + + IsoItem& iso = m_results[ic][its][ipv][itt]; + const double& coneSize = m_isoConeSizes[ic]; + const double& logChi2Max = m_isoTrkImpLogChi2Max[ic]; + const int& doLogChi2 = m_isoDoTrkImpLogChi2Cut[ic]; + + // presets + iso.resetVals(); + + double nTracksInCone = 0; + double ptSumInCone = 0.; + + // make sure candRefPV exists + if ( candRefPV != NULL ) { + + for (TrackBag::const_iterator trkItr = tracks.begin(); + trkItr != tracks.end(); ++trkItr) { + double deltaR = candP.DeltaR((*trkItr)->p4().Vect()); + if ( deltaR < coneSize ) { + double logChi2 = (doLogChi2 > 0) ? + getTrackCandPVLogChi2(*trkItr, candRefPV) : -9999.; + // next line needed exactly as is for backward validation + if ( doLogChi2 == 2 ) logChi2 = abs(logChi2); + if ( doLogChi2 == 0 || logChi2 < logChi2Max ) { + nTracksInCone++; + ptSumInCone += (*trkItr)->pt(); + } // logChi2 + } // deltaR + } + // calculate result + if ( ptSumInCone + candP.Pt() > 0. ) { + iso.isoValue = candP.Pt() / ( ptSumInCone + candP.Pt() ); + } else { + iso.isoValue = -5.; + } + + } else { + iso.isoValue = -10.; + } // if candRefPV != NULL + + iso.nTracks = nTracksInCone; + } // for ic + + return StatusCode::SUCCESS; + } + //-------------------------------------------------------------------------- + // Fill track isolation values from cache if found + //-------------------------------------------------------------------------- + bool BVertexTrackIsoTool::fastFillHook(const xAOD::Vertex* vtx, + const int ipv) const { + + ATH_MSG_DEBUG("fastFillHook: ipv: " << ipv); + + bool found(false); + + StringIntMap_t::iterator itpv = + m_pvAssocResMap.find(buildPvAssocCacheName(vtx, ipv)); + if ( itpv != m_pvAssocResMap.end() ) { + found = true; + unsigned int nCones = m_isoConeSizes.size(); + unsigned int nTrackSels = m_trackSelectionTools.size(); + unsigned int nTrackTypes = m_useTrackTypes.size(); + for (unsigned int its = 0; its < nTrackSels; ++its) { + for (unsigned int ic = 0; ic < nCones; ++ic) { + for (unsigned int itt = 0; itt < nTrackTypes; ++itt) { + m_results[ic][its][ipv][itt] + .copyVals(m_results[ic][its][itpv->second][itt]); + } // for its + } // for ic + } // for itt + } // if found + + ATH_MSG_DEBUG("fastFillHook: cache index: " + << buildPvAssocCacheName(vtx, ipv) + << ", found ? " << found + << ", ipv_ref: " + << (found ? itpv->second : -1)); + + return found; + } + //-------------------------------------------------------------------------- + // Track isolation calculation loops -- slower method + //-------------------------------------------------------------------------- + StatusCode + BVertexTrackIsoTool::calculateIsolation(const xAOD::Vertex* vtx) const { + + ATH_MSG_DEBUG("BVertexTrackIsoTool::calculateIsolation -- begin"); + + unsigned int nCones = m_isoConeSizes.size(); + unsigned int nTrackSels = m_trackSelectionTools.size(); + unsigned int nPvAssocs = m_pvAssocTypes.size(); + unsigned int nTrackTypes = m_useTrackTypes.size(); + + for (unsigned int its = 0; its < nTrackSels; ++its) { + for (unsigned int ipv = 0; ipv < nPvAssocs; ++ipv) { + for (unsigned int ic = 0; ic < nCones; ++ic) { + for (unsigned int itt = 0; itt < nTrackTypes; ++itt) { + CHECK(calcIsolation(m_results[ic][its][ipv][itt], vtx, + m_isoConeSizes[ic], m_isoTrkImpLogChi2Max[ic], + m_isoDoTrkImpLogChi2Cut[ic], + m_trackSelectionTools[its], + m_pvAssocTypes[ipv], m_useTrackTypes[itt])); + } // for itt + } // for ic + } // for ipv + } // for its + + return StatusCode::SUCCESS; + } + //-------------------------------------------------------------------------- + // Calculate track isolation variables -- slower method + //-------------------------------------------------------------------------- + StatusCode BVertexTrackIsoTool:: + calcIsolation(const IsoItem& iso, + const xAOD::Vertex* vtx, + const double coneSize, + const double logChi2Max, + const int doLogChi2, + const ToolHandle<TrkSelTool>& tSelTool, + const xAOD::BPhysHelper::pv_type pvAssocType, + const int trackTypes ) const { + + // preset + iso.nTracks = -1; + iso.isoValue = -2.; + + // candidate tracks and momentum + xAOD::BPhysHelper cand(vtx); + TrackBag candTracks = findAllTracksInDecay(cand); + TVector3 candP = cand.totalP(); + const xAOD::Vertex* candRefPV = cand.pv(pvAssocType); + const xAOD::Vertex* candPV = cand.origPv(pvAssocType); + + // tracks to be considered + TrackBag tracks; + for (xAOD::TrackParticleContainer::const_iterator trkItr = + m_tracks->begin(); trkItr != m_tracks->end(); ++trkItr) { + const xAOD::TrackParticle* track = *trkItr; + // track selection check + if ( ! tSelTool->accept(*track, candRefPV) ) continue; + // track type check + if ( ! ((unsigned int)trackTypes == ttall() || + (unsigned int)trackTypes == ttallMin() || + (detTrackTypes(track, candPV, candRefPV) + & trackTypes) > 0x0) ) continue; + // track not in SV + if ( std::find(candTracks.begin(), candTracks.end(), track) + != candTracks.end() ) continue; + // tracks that survived so far + tracks.push_back(track); + } + + double nTracksInCone = 0; + double ptSumInCone = 0.; + for (TrackBag::const_iterator trkItr = tracks.begin(); + trkItr != tracks.end(); ++trkItr) { + double deltaR = candP.DeltaR((*trkItr)->p4().Vect()); + if ( deltaR < coneSize ) { + double logChi2 = (doLogChi2 > 0) ? + getTrackCandPVLogChi2(*trkItr, candRefPV) : -9999.; + // next line needed exactly as is for backward validation + if ( doLogChi2 == 2 ) logChi2 = abs(logChi2); + if ( doLogChi2 == 0 || logChi2 < logChi2Max ) { + nTracksInCone++; + ptSumInCone += (*trkItr)->pt(); + } + } // deltaR + } + // calculate result + if ( ptSumInCone + candP.Pt() > 0. ) { + iso.isoValue = candP.Pt() / ( ptSumInCone + candP.Pt() ); + } else { + iso.isoValue = -5; + } + iso.nTracks = nTracksInCone; + + return StatusCode::SUCCESS; + } + //-------------------------------------------------------------------------- + StatusCode + BVertexTrackIsoTool::saveIsolation(const xAOD::Vertex* vtx) const { + + unsigned int nCones = m_isoConeSizes.size(); + unsigned int nTrackSels = m_trackSelectionTools.size(); + unsigned int nPvAssocs = m_pvAssocTypes.size(); + unsigned int nTrackTypes = m_useTrackTypes.size(); + + for (unsigned int its = 0; its < nTrackSels; ++its) { + for (unsigned int ipv = 0; ipv < nPvAssocs; ++ipv) { + for (unsigned int ic = 0; ic < nCones; ++ic) { + for (unsigned int itt = 0; itt < nTrackTypes; ++itt) { + IsoItem result = m_results[ic][its][ipv][itt]; + SG::AuxElement::Decorator< float > + d_iso_value(result.isoName()); + SG::AuxElement::Decorator< int > + d_iso_ntracks(result.nTracksName()); + d_iso_value(*vtx) = result.isoValue; + d_iso_ntracks(*vtx) = result.nTracks; + ATH_MSG_DEBUG("BVertexTrackIsoTool::saveIsolation() -- isobn: " + << result.isoName() << ", ntbn: " + << result.nTracksName()); + ATH_MSG_DEBUG("BVertexTrackIsoTool::saveIsolation() -- vertex: (" + << vtx->x() << ", " + << vtx->y() << ", " + << vtx->z() << "), iso: " + << result.isoValue << ", nTracks: " + << result.nTracks); + } // for itt + } // for ic + } // for ipv + } // for its + + return StatusCode::SUCCESS; + } + //-------------------------------------------------------------------------- + void BVertexTrackIsoTool::setResultsPrefix(std::string prefix) const { + + ATH_MSG_DEBUG("BVertexTrackIsoTool::setResultsPrefix -- begin"); + + unsigned int nCones = m_isoConeSizes.size(); + unsigned int nTrackSels = m_trackSelectionTools.size(); + unsigned int nPvAssocs = m_pvAssocTypes.size(); + unsigned int nTrackTypes = m_useTrackTypes.size(); + + for (unsigned int its = 0; its < nTrackSels; ++its) { + for (unsigned int ipv = 0; ipv < nPvAssocs; ++ipv) { + for (unsigned int ic = 0; ic < nCones; ++ic) { + for (unsigned int itt = 0; itt < nTrackTypes; ++itt) { + m_results[ic][its][ipv][itt].setPrefix(prefix); + } // for itt + } // for ic + } // for ipv + } // for its + + ATH_MSG_DEBUG("BVertexTrackIsoTool::setResultsPrefix -- end"); + } + //-------------------------------------------------------------------------- + void BVertexTrackIsoTool::initResults() { + + unsigned int nCones = m_isoConeSizes.size(); + unsigned int nTrackSels = m_trackSelectionTools.size(); + unsigned int nPvAssocs = m_pvAssocTypes.size(); + unsigned int nTrackTypes = m_useTrackTypes.size(); + + ATH_MSG_DEBUG("BVertexTrackIsoTool::initResults -- begin"); + ATH_MSG_DEBUG("BVertexTrackIsoTool::initResults : nCones = " << nCones); + ATH_MSG_DEBUG("BVertexTrackIsoTool::initResults : nTrackSels = " + << nTrackSels); + ATH_MSG_DEBUG("BVertexTrackIsoTool::initResults : nPvAssocs = " + << nPvAssocs); + ATH_MSG_DEBUG("BVertexTrackIsoTool::initResults : nTrackTypes = " + << nTrackTypes); + m_results.resize(boost::extents[nCones][nTrackSels][nPvAssocs][nTrackTypes]); + for (unsigned int its = 0; its < nTrackSels; ++its) { + ATH_MSG_DEBUG("BVertexTrackIsoTool::initResults -- its = " << its); + for (unsigned int ipv = 0; ipv < nPvAssocs; ++ipv) { + ATH_MSG_DEBUG("BVertexTrackIsoTool::initResults -- ipv = " << ipv); + for (unsigned int ic = 0; ic < nCones; ++ic) { + ATH_MSG_DEBUG("BVertexTrackIsoTool::initResults -- ic = " << ic); + for (unsigned int itt = 0; itt < nTrackTypes; ++itt) { + ATH_MSG_DEBUG("BVertexTrackIsoTool::initResults -- itt = " << itt); + + ATH_MSG_DEBUG("BVertexTrackIsoTool::initResults :" + << m_branchBaseName << buildBranchName(ic, its, + ipv, itt)); + + m_results[ic][its][ipv][itt].setup(buildBranchName(ic, its, + ipv, itt), + m_branchBaseName); + } // for itt + } // for ic + } // for ipv + } // for its + + ATH_MSG_DEBUG("BVertexTrackIsoTool::initResults -- end"); + } + //-------------------------------------------------------------------------- + std::string BVertexTrackIsoTool::buildBranchName(unsigned int ic, + unsigned int its, + unsigned int ipv, + unsigned int itt) const { + ATH_MSG_DEBUG("BVertexTrackIsoTool::buildBranchName -- begin"); + + double coneSize = m_isoConeSizes[ic]; + double logChi2Max = m_isoTrkImpLogChi2Max[ic]; + int doLogChi2 = m_isoDoTrkImpLogChi2Cut[ic]; + + // format it nicely + boost::format f("%02d_LC%02dd%1d_%s"); + f % (int)(coneSize*10.) % (int)(logChi2Max*10.) % doLogChi2 + % buildBranchBaseName(its, ipv, itt); + + ATH_MSG_DEBUG("BVertexTrackIsoTool::buildBranchName: " << f.str()); + + return f.str(); + } + //-------------------------------------------------------------------------- +} + diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/BmumuThinningTool.cxx_NoCompile b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/BmumuThinningTool.cxx_NoCompile new file mode 100644 index 00000000000..c8a539972c6 --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/BmumuThinningTool.cxx_NoCompile @@ -0,0 +1,1167 @@ +/* + Copyright (C) 2002-2020 CERN for the benefit of the ATLAS collaboration +*/ + +/** + * @file BmumuThinningTool.cxx + * @author Wolfgang Walkowiak <wolfgang.walkowiak@cern.ch> + * + */ + +#include "DerivationFrameworkBPhys/BmumuThinningTool.h" +#include "AthenaKernel/IThinningSvc.h" +#include "xAODTracking/TrackParticleContainer.h" +#include "xAODTracking/VertexContainer.h" +#include "xAODTracking/VertexAuxContainer.h" +#include "xAODBase/IParticleHelpers.h" +#include "AthContainers/AuxElement.h" +#include "AthContainers/AuxTypeRegistry.h" + +#include <string> +#include <sstream> +#include <istream> +#include <vector> +#include <iomanip> +#include <algorithm> +#include <numeric> +#include <regex> +#include <boost/algorithm/string.hpp> + +namespace DerivationFramework { + + // static members + // Note: may later be migrated to xAODBPhys/BPhysHelper + std::map<xAOD::BPhysHelper::pv_type, std::string> + BmumuThinningTool::PvTypeToVarNameMap = + { {xAOD::BPhysHelper::PV_MAX_SUM_PT2, "MaxSumPt2"}, + {xAOD::BPhysHelper::PV_MIN_A0 , "MinA0" }, + {xAOD::BPhysHelper::PV_MIN_Z0 , "MinZ0" }, + {xAOD::BPhysHelper::PV_MIN_Z0_BA , "MinZ0BA" } }; + + //-------------------------------------------------------------------------- + // Constructor + //-------------------------------------------------------------------------- + BmumuThinningTool::BmumuThinningTool(const std::string& t, + const std::string& n, + const IInterface* p) : + CfAthAlgTool(t, n, p), + m_thinningSvc("ThinningSvc", n), + m_doCloseTracks(false), + m_doPVs(false), + m_doRefPVs(false), + m_doMuons(false), + m_doCalMuons(false), + m_doTracks(false) { + + declareInterface<DerivationFramework::IThinningTool>(this); + + // thinning service + declareProperty("ThinningService" , m_thinningSvc); + // TrackParticle container name + declareProperty("TrackParticleContainerName", + m_trkPartContName = "InDetTrackParticles"); + // list of secondary vertex container names + declareProperty("VertexContainerNames" , m_vtxContNames); + // list of pass flags for the seconary vertices + // empty list lets all vertices pass + // list length needs to be identical to length of + // VertexContainerNames list if AlignPassToVertexList is True + declareProperty("VertexPassFlags" , m_vtxPassFlags); + // align VertexPassFlags to VertexContainerNames list? + // This option causes a 1:1 correlation between the two lists, + // i.e. a flag is only applied to the corresponding container + // if this option is set to True. (default: false) + declareProperty("AlignPassToVertexList", + m_alignPassToVertexList = false); + // Primary vertex container name + declareProperty("PVContainerName" , m_PVContName); + // Refitted primary vertex container names + // This list must be of same length and order as the m_vtxContNames list + // (or empty => no thinning of refitted primary vertex containers) + declareProperty("RefPVContainerNames" , m_refPVContNames); + // name of the used muon container + declareProperty("MuonContainerName" , m_muonContName = ""); + // name of the calibrated muons container + declareProperty("CalibMuonContainerName" , m_calMuonContName = ""); + // closest track branch base name + declareProperty("CloseTrackBranchBaseName", m_ctBranchBaseName); + // closest track branch prefixes + declareProperty("CloseTrackBranchPrefixes", m_ctBranchPrefixes); + // keep tracks for selected (refitted) primary vertices + declareProperty("KeepTracksForSelectedPVs", m_keepPVTracks = false); + // match vertex muons with calibrated muons + declareProperty("MatchCalibratedMuons" , m_matchCalMuons = false); + // mark orginal muons for matched calibrated muons as well + // (only makes sense if MatchCalibratedMuons = True) + declareProperty("MarkMatchedMuons" , m_markMuons = false); + // mark calibrated muons for matched original muons as well + // (only makes sense if MatchCalibratedMuons = False) + declareProperty("MarkMatchedCalMuons" , m_markCalMuons = false); + // sync marked muons both ways (forces it) + declareProperty("SyncMatchedMuonsBothWays", m_syncMuonsBothWays = false); + // allow fast sync of myon masks + // (Set to 'False' to force in-depth synchronization of muon masks.) + declareProperty("AllowFastMuonMaskSync" , m_allowFastMuonMaskSync = true); + // keep tracks for closest tracks + declareProperty("KeepCloseTracks" , m_keepCloseTracks = false); + // keep tracks for selected muons + declareProperty("KeepTracksForMuons" , m_keepSelMuonTracks = false); + // keep tracks for selected calibrated muons + declareProperty("KeepTracksForCalMuons" , m_keepSelCalMuonTracks = false); + // keep (original) muons for selected tracks + declareProperty("KeepMuonsForTracks" , m_keepSelTrackMuons = false); + // keep calibrated muons for selected tracks + declareProperty("KeepCalMuonsForTracks" , m_keepSelTrackCalMuons = false); + // apply AND for mask matching for vertices (default: false) + declareProperty("ApplyAndForVertices" , m_vertexAnd = false); + // apply AND for mask matching for tracks (default: false) + declareProperty("ApplyAndForTracks" , m_trackAnd = false); + // apply AND for mask matching for muons (default: false) + declareProperty("ApplyAndForMuons" , m_muonAnd = false); + // thin primary vertex collection + declareProperty("ThinPVs" , m_thinPVs = true); + // thin refittd primary vertex collections + declareProperty("ThinRefittedPVs" , m_thinRefPVs = true); + // thin ID track collection + declareProperty("ThinTracks" , m_thinTracks = true); + // thin muon collections + declareProperty("ThinMuons" , m_thinMuons = true); + } + //-------------------------------------------------------------------------- + // Destructor + //-------------------------------------------------------------------------- + BmumuThinningTool::~BmumuThinningTool() { + } + //-------------------------------------------------------------------------- + // initialization + //-------------------------------------------------------------------------- + StatusCode BmumuThinningTool::initialize() { + + ATH_MSG_INFO("BmumuThinningTool::initialize()"); + + // check TrackParticle container name + if ( m_trkPartContName == "" ) { + ATH_MSG_INFO("No ID track collection provided for thinning."); + } else { + ATH_MSG_INFO("Using " << m_trkPartContName + << " as the source collection for ID track particles."); + m_doTracks = true; + } + + // check secondary vertex container names + if ( m_vtxContNames.empty() ) { + ATH_MSG_FATAL("No secondary vertex collections provided for thinning."); + return StatusCode::FAILURE; + } else { + for (std::vector<std::string>::iterator it = m_vtxContNames.begin(); + it != m_vtxContNames.end(); ++it) { + ATH_MSG_INFO("Using " << *it + << " as a source collection for secondary vertices."); + } + } + + // check vertex pass flags + if ( m_alignPassToVertexList ) { + if ( m_vtxPassFlags.size() != m_vtxContNames.size() ) { + ATH_MSG_FATAL("Size mismatch of VertexContainerNames (" + << m_vtxContNames.size() + << ") and VertexPassFlags (" + << m_vtxPassFlags.size() << ")"); + return StatusCode::FAILURE; + } else { + ATH_MSG_INFO(std::left << std::setw(35) << "VertexContainerNames" + << " : " << "VertexPassFlags"); + ATH_MSG_INFO(std::setfill('-') << std::setw(70) << "" + << std::setfill(' ')); + for (size_t i=0; i<m_vtxContNames.size(); ++i) { + ATH_MSG_INFO(std::left << std::setw(35) << m_vtxContNames[i] + << " : " << m_vtxPassFlags[i]); + } + } + } else { + if ( m_vtxPassFlags.empty() ) { + ATH_MSG_INFO("No VertexPassFlags: all secondary vertices will be " + << "accepted."); + } else { + std::string str; + for (size_t i=0; i < m_vtxPassFlags.size(); ++i) { + if (i > 0) str += ", "; + str += m_vtxPassFlags[i]; + } + ATH_MSG_INFO("VertexPassFlags applied to all vertices:"); + ATH_MSG_INFO(str); + } + } + + // check primary vertex container name + if ( m_PVContName == "" ) { + ATH_MSG_FATAL("No primary vertex collection provided for thinning."); + return StatusCode::FAILURE; + } else { + ATH_MSG_INFO("Using " << m_PVContName + << " as the source collection for primary vertices."); + m_doPVs = true; + } + + // check refitted primary vertex container names + if ( m_refPVContNames.empty() ) { + ATH_MSG_INFO("No refitted PV collections provided for thinning."); + } else { + if ( m_refPVContNames.size() != m_vtxContNames.size() ) { + ATH_MSG_FATAL("Size mismatch of VertexContainerNames (" + << m_vtxContNames.size() + << ") and RefPVContainerNames (" + << m_refPVContNames.size() << ")"); + return StatusCode::FAILURE; + } else { + for (std::vector<std::string>::iterator it = m_refPVContNames.begin(); + it != m_refPVContNames.end(); ++it) { + ATH_MSG_INFO("Using " << *it + << " as a source collection for refitted PVs."); + } + m_doRefPVs = true; + } + } + + // check muon container name + if ( m_muonContName == "" ) { + ATH_MSG_INFO("No (orginal) muon collection provided for thinning."); + } else { + ATH_MSG_INFO("Using " << m_muonContName + << " as a source collection for (original) muons."); + m_doMuons = true; + } + + // check calibrated muons container name + if ( m_calMuonContName == "" ) { + ATH_MSG_INFO("No calibrated muons collection provided for thinning."); + } else { + ATH_MSG_INFO("Using " << m_calMuonContName + << " as a source collection for calibrated muons."); + m_doCalMuons = true; + } + + // check muon thinning settings + if ( m_thinMuons ) { + if ( (m_matchCalMuons || m_markCalMuons) && !m_doCalMuons ) { + ATH_MSG_ERROR("No container for calibrated muons given!"); + } + if ( (!m_matchCalMuons || m_markMuons) && !m_doMuons ) { + ATH_MSG_ERROR("No container for (original) muons given!"); + } + if ( m_matchCalMuons && m_markCalMuons ) { + ATH_MSG_WARNING("Configuration issue: both MatchWithCalMuons and " + << "MarkMatchedCalMuons set to true! " + << "Ignoring the second setting."); + } + if ( !m_matchCalMuons && m_markMuons ) { + ATH_MSG_WARNING("Configuration issue: MatchWithCalMuons set to " + << "false and " + << "MarkMatchedMuons set to true! " + << "Ignoring the second setting."); + } + ATH_MSG_INFO("MatchWithCalMuons: " << m_matchCalMuons + << ", MarkMatchedMuons: " << m_markMuons + << ", MarkMatchedCalMuons: " << m_markCalMuons); + } + + // check closest track settings + m_doCloseTracks = (m_ctBranchBaseName != "" && !m_ctBranchPrefixes.empty()); + if ( m_doCloseTracks ) { + for (std::vector<std::string>::iterator it = m_ctBranchPrefixes.begin(); + it != m_ctBranchPrefixes.end(); ++it) { + + ATH_MSG_INFO("Keeping tracks for " + << *it << "_" << m_ctBranchBaseName << "_*"); + } + } else { + ATH_MSG_INFO("Not keeping anything for closest tracks in thinning."); + } + + // check track container for combination of track and muon thinning + if ( (m_thinTracks || m_thinMuons) && !m_doTracks) { + ATH_MSG_FATAL("Requested track or muon thinning but required " + "track container not provided"); + return StatusCode::FAILURE; + } + + // Output of options + ATH_MSG_INFO("=== Option settings - begin ==="); + ATH_MSG_INFO("KeepTracksForSelectedPVs : " << m_keepPVTracks); + ATH_MSG_INFO("MatchCalibratedMuons : " << m_matchCalMuons); + ATH_MSG_INFO("MarkMatchedMuons : " << m_markMuons); + ATH_MSG_INFO("MarkMatchedCalMuons : " << m_markCalMuons); + ATH_MSG_INFO("SyncMatchedMuonsBothWays : " << m_syncMuonsBothWays); + ATH_MSG_INFO("AllowFastMuonMaskSync : " << m_allowFastMuonMaskSync); + ATH_MSG_INFO("KeepCloseTracks : " << m_keepCloseTracks); + ATH_MSG_INFO("KeepTracksForMuons : " << m_keepSelMuonTracks); + ATH_MSG_INFO("KeepTracksForCalMuons : " << m_keepSelCalMuonTracks); + ATH_MSG_INFO("KeepMuonsForTracks : " << m_keepSelTrackMuons); + ATH_MSG_INFO("KeepCalMuonsForTracks : " << m_keepSelTrackCalMuons); + ATH_MSG_INFO("ApplyAndForVertices : " << m_vertexAnd); + ATH_MSG_INFO("ApplyAndForTracks : " << m_trackAnd); + ATH_MSG_INFO("ApplyAndForMuons : " << m_muonAnd); + ATH_MSG_INFO("ThinPVs : " << m_thinPVs); + ATH_MSG_INFO("ThinRefittedPVs : " << m_thinRefPVs); + ATH_MSG_INFO("ThinTracks : " << m_thinTracks); + ATH_MSG_INFO("ThinMuons : " << m_thinMuons); + ATH_MSG_INFO("=== Option settings - end ==="); + + + // initialize cache vector-of-vectors (one per vertex container) + for (size_t ivcname=0; ivcname < m_vtxContNames.size(); ++ivcname) { + m_vvOrigPVLinkNames.push_back(std::vector<std::string>()); + m_vvOrigPVLinkTypes.push_back(std::vector<pv_type>()); + m_vvRefPVLinkNames.push_back(std::vector<std::string>()); + m_vvRefPVLinkTypes.push_back(std::vector<pv_type>()); + m_vvCtLinkNames.push_back(std::vector<std::string>()); + m_vvCtLinkTypes.push_back(std::vector<pv_type>()); + } // for ivcname + + return StatusCode::SUCCESS; + } + //-------------------------------------------------------------------------- + // finalization + //-------------------------------------------------------------------------- + StatusCode BmumuThinningTool::finalize() { + + ATH_MSG_INFO("BmumuThinningTool::finalize()"); + + return StatusCode::SUCCESS; + } + //-------------------------------------------------------------------------- + // apply thinning + //-------------------------------------------------------------------------- + StatusCode BmumuThinningTool::doThinning() const { + + ATH_MSG_DEBUG("BmumuThinningTool::doThinning()"); + + // retrieve TrackParticle container + const xAOD::TrackParticleContainer* trkPartCont = nullptr; + std::vector<bool> trkMask; + if ( m_doTracks && (m_thinTracks || m_thinMuons) ) { + CHECK( evtStore()->retrieve(trkPartCont, m_trkPartContName) ); + // default: keep no track + trkMask.assign(trkPartCont->size(), false); + } + + // retrieve PV container + const xAOD::VertexContainer* pvCont = nullptr; + std::vector<bool> pvMask; + if ( m_doPVs && (m_thinPVs || m_keepPVTracks) ) { + CHECK( evtStore()->retrieve(pvCont, m_PVContName) ); + // default: keep no PV + pvMask.assign(pvCont->size(), false); + } + // retrieve refitted PV container + std::vector<const xAOD::VertexContainer*> vRefPvCont = + std::vector<const xAOD::VertexContainer*>(m_refPVContNames.size(), + nullptr); + std::vector<std::vector<bool> > vRefPvMasks; + if ( m_thinRefPVs && m_doRefPVs ) { + for (size_t irpv = 0; irpv < m_refPVContNames.size(); ++irpv) { + CHECK( evtStore()->retrieve(vRefPvCont[irpv], m_refPVContNames[irpv]) ); + // default: keep no refitted PV + vRefPvMasks.push_back(std::vector<bool>(vRefPvCont[irpv]->size(), + false)); + } + } + + // retrieve container of (original) muons + const xAOD::MuonContainer* muonCont = nullptr; + std::vector<bool> muonMask; + if ( m_thinMuons && m_doMuons && (!m_matchCalMuons || m_markMuons) ) { + CHECK( evtStore()->retrieve(muonCont, m_muonContName) ); + // default: keep no muon + muonMask.assign(muonCont->size(), false); + } + + // retrieve container of calibrated muons + const xAOD::MuonContainer* calMuonCont = nullptr; + std::vector<bool> calMuonMask; + if ( m_thinMuons && m_doCalMuons && (m_matchCalMuons || m_markCalMuons) ) { + CHECK( evtStore()->retrieve(calMuonCont, m_calMuonContName) ); + // default: keep no muon + calMuonMask.assign(calMuonCont->size(), false); + } + + // retrieve vertex containers + for (size_t ivcname=0; ivcname < m_vtxContNames.size(); ++ivcname) { + auto &vtxContName = m_vtxContNames[ivcname]; + ATH_MSG_DEBUG("doThinning(): vtxContName: " << vtxContName); + const xAOD::VertexContainer* vtxCont = nullptr; + const xAOD::VertexAuxContainer* vtxAuxCont = nullptr; + CHECK( evtStore()->retrieve(vtxCont , vtxContName) ); + CHECK( evtStore()->retrieve(vtxAuxCont, vtxContName+"Aux.") ); + size_t vtxContSize = vtxCont->size(); + std::vector<bool> vtxMask; + // default: keep no vertex + vtxMask.assign(vtxContSize, false); + ATH_MSG_DEBUG("doThinning(): vtxContSize: " << vtxContSize); + // loop over vertices + for (size_t ivtx = 0; ivtx < vtxContSize; ++ivtx) { + xAOD::BPhysHelper vtx(vtxCont->at(ivtx)); + bool vtxPassed = false; + addToCounter(vtxContName+"_allVertices"); + ATH_MSG_DEBUG("doThinning(): ivtx = " << ivtx); + if ( m_alignPassToVertexList ) { + ATH_MSG_DEBUG("doThinning(): 1: passFlag(" << ivcname << ") = " + << m_vtxPassFlags[ivcname]); + vtxPassed = pass(*vtx.vtx(), m_vtxPassFlags[ivcname]); + } else { + for (size_t ipass = 0; ipass < m_vtxPassFlags.size(); ++ipass) { + ATH_MSG_DEBUG("doThinning(): 2: passFlag(" << ipass << ") = " + << m_vtxPassFlags[ipass]); + if ( pass(*vtx.vtx(), m_vtxPassFlags[ipass]) ) { + vtxPassed = true; + break; + } + } // for ipass + } + if ( vtxPassed ) { + // + // vertex passed selection + // + ATH_MSG_DEBUG("doThinning(): ivtx " << ivtx << " passed selection"); + vtxMask[ivtx] = true; + addToCounter(vtxContName+"_passedVertices"); + + // keep tracks from secondary vertices + if ( m_doTracks && (m_thinTracks || m_thinMuons) ) { + for (size_t itrk=0; itrk < trkPartCont->size(); ++itrk) { + // only consider if not yet kept + if ( !trkMask[itrk] ) { + for (size_t ivt=0; ivt < vtx.vtx()->nTrackParticles(); ++ivt) { + if ( vtx.vtx()->trackParticle(ivt) + == trkPartCont->at(itrk) ) { + trkMask[itrk] = true; + addToCounter(vtxContName+"_accTracks"); + } + } // for trk + } + } // for itrk + } // if m_doTracks + + // find aux variable names for closest tracks + if ( (m_thinTracks || m_thinMuons) && m_keepCloseTracks + && m_doCloseTracks && m_vvCtLinkNames[ivcname].empty() ) { + std::string prefix = + m_vtxContNames[ivcname]+"_"+m_ctBranchBaseName; + ATH_MSG_DEBUG("doThinning(): CT basename: " << prefix); + selectAuxElements(vtxAuxCont, m_vvCtLinkNames[ivcname], + m_ctBranchPrefixes, + m_vvCtLinkTypes[ivcname], + "_"+m_ctBranchBaseName+".*._Link"); + if ( msgLvl(MSG::INFO) ) { + std::string sAuxVars = + dumpVS(m_vvCtLinkNames[ivcname], + "doThinning(): "+vtxContName+": CT aux vars: (" + +std::to_string(m_vvCtLinkNames[ivcname].size())+")", + 4); + logWrappedMsg(sAuxVars, MSG::INFO); + } + } + + // keep tracks identified as closest tracks + if ( (m_thinTracks || m_thinMuons) && m_doTracks && m_keepCloseTracks + && m_doCloseTracks ) { + for (size_t i = 0; i < m_vvCtLinkNames[ivcname].size(); ++i) { + const xAOD::TrackParticle* closeTrack = + getTrackParticle(vtx.vtx(), m_vvCtLinkNames[ivcname][i]); + if ( closeTrack == nullptr ) continue; + auto tpit = std::find(trkPartCont->begin(), trkPartCont->end(), + closeTrack); + if ( tpit == trkPartCont->end() ) { + ATH_MSG_WARNING("ClosestTrack not found in " + << m_trkPartContName + << " for " << m_vvCtLinkNames[ivcname][i]); + continue; + } + size_t x = std::distance(trkPartCont->begin(), tpit); + if ( !trkMask.at(x) ) { + trkMask.at(x) = true; + addToCounter(vtxContName+"_addTracksByCT"); + } + } // for i (CT link names) + } // if + + // find aux variable names for original PVs + if ( m_thinPVs && m_vvOrigPVLinkNames[ivcname].empty() ) { + selectAuxElements(vtxAuxCont, m_vvOrigPVLinkNames[ivcname], + m_vvOrigPVLinkTypes[ivcname], "OrigPv.*.Link"); + if ( msgLvl(MSG::INFO) ) { + std::string sAuxVars = + dumpVS(m_vvOrigPVLinkNames[ivcname], + "doThinning(): "+vtxContName+": OrigPV aux vars: (" + +std::to_string(m_vvOrigPVLinkNames[ivcname].size())+")", + 4); + logWrappedMsg(sAuxVars, MSG::INFO); + } + } + // find aux variable names for refitted PVs + if ( m_thinRefPVs && m_vvRefPVLinkNames[ivcname].empty() ) { + selectAuxElements(vtxAuxCont, m_vvRefPVLinkNames[ivcname], + m_vvRefPVLinkTypes[ivcname], "Pv.*.Link"); + if ( msgLvl(MSG::INFO) ) { + std::string sAuxVars = + dumpVS(m_vvRefPVLinkNames[ivcname], + "doThinning() :"+vtxContName+": RefPV aux vars: (" + +std::to_string(m_vvRefPVLinkNames[ivcname].size())+")", + 4); + logWrappedMsg(sAuxVars, MSG::INFO); + } + } + + // debug stuff + if ( msgLvl(MSG::VERBOSE) ) { + std::vector<std::string> auxVars = + filterAuxElements(vtxAuxCont, ".*Link"); + std::string sAuxVars = + dumpVS(auxVars, + "doThinning(): "+vtxContName+": vtxAuxContVarNames:", 4); + logWrappedMsg(sAuxVars, MSG::DEBUG); + } + + // now mark associated PVs + if ( m_thinPVs && m_doPVs ) { + for (size_t i = 0; i < m_vvOrigPVLinkTypes[ivcname].size(); ++i) { + const xAOD::Vertex* origPv = + vtx.origPv(m_vvOrigPVLinkTypes[ivcname][i]); + if ( origPv == nullptr ) continue; + auto pvit = std::find(pvCont->begin(), pvCont->end(), origPv); + if ( pvit == pvCont->end() ) { + ATH_MSG_WARNING("PV not found in " << m_PVContName + << " for " << m_vvOrigPVLinkNames[ivcname][i]); + continue; + } + size_t x = std::distance(pvCont->begin(), pvit); + if ( !pvMask.at(x) ) { + pvMask.at(x) = true; + addToCounter(vtxContName+"_accAssocPVs"); + } + // keep tracks for selected PVs + if ( (m_thinTracks || m_thinMuons) && m_doTracks && m_keepPVTracks ) { + for (size_t ipvt=0; ipvt < (*pvit)->nTrackParticles(); + ++ipvt) { + const xAOD::TrackParticle* tp = (*pvit)->trackParticle(ipvt); + if ( tp == nullptr ) continue; + auto tpit = std::find(trkPartCont->begin(), + trkPartCont->end(), tp); + if ( tpit == trkPartCont->end() ) { + ATH_MSG_WARNING("PV track not found in " + << m_trkPartContName << " for PV from " + << m_PVContName); + continue; + } + size_t x = std::distance(trkPartCont->begin(), tpit); + if ( !trkMask.at(x) ) { + trkMask.at(x) = true; + addToCounter(vtxContName+"_addTracksBySelPVs"); + } + } // for ipvt + } // if m_keepPVTracks + } // for i (PVs) + } // if m_thinPVs && m_doPVs + + // now mark associated refittedPVs + if ( m_thinRefPVs && m_doRefPVs ) { + for (size_t i = 0; i < m_vvRefPVLinkTypes[ivcname].size(); ++i) { + const xAOD::Vertex* refPv = + vtx.pv(m_vvRefPVLinkTypes[ivcname][i]); + if ( refPv == nullptr ) continue; + auto pvit = std::find(vRefPvCont[ivcname]->begin(), + vRefPvCont[ivcname]->end(), refPv); + if ( pvit == vRefPvCont[ivcname]->end() ) { + ATH_MSG_WARNING("PV not found in " << m_refPVContNames[ivcname] + << " for " << m_vvRefPVLinkNames[ivcname][i]); + continue; + } + size_t x = std::distance(vRefPvCont[ivcname]->begin(), pvit); + if ( !vRefPvMasks[ivcname].at(x) ) { + vRefPvMasks[ivcname].at(x) = true; + addToCounter(vtxContName+"_accAssocRefPVs"); + } + // keep tracks for associated refitted PVs + if ( (m_thinTracks || m_thinMuons) && m_doTracks + && m_keepPVTracks ) { + for (size_t ipvt=0; ipvt < (*pvit)->nTrackParticles(); + ++ipvt) { + const xAOD::TrackParticle* tp = (*pvit)->trackParticle(ipvt); + if ( tp == nullptr ) continue; + auto tpit = std::find(trkPartCont->begin(), + trkPartCont->end(), tp); + if ( tpit == trkPartCont->end() ) { + ATH_MSG_WARNING("Refitted PV track not found in " + << m_trkPartContName + << " for refitted PV from " + << m_refPVContNames[ivcname][i]); + continue; + } + size_t x = std::distance(trkPartCont->begin(), tpit); + if ( !trkMask.at(x) ) { + trkMask.at(x) = true; + addToCounter(vtxContName+"_addTracksByAssocRefPVs"); + } + } // for ipvt + } // if (m_thinTracks || m_thinMuons) && m_doTracks + // && m_keepPVTracks + } // for i (refPVs) + } // if m_thinRefPVs && m_doRefPVs + + // keep muons from secondary vertices + if ( m_thinMuons ) { + if ( m_matchCalMuons ) { // calibrated muons collection + if ( m_doCalMuons ) { + addToCounter(vtxContName+"_accCalMuons_calls"); + CHECK( matchMuons(calMuonCont, calMuonMask, vtx, + vtxContName+"_accCalMuons") ); + } + } else { // original muons collection + if ( m_doMuons ) { + addToCounter(vtxContName+"_accMuons_calls"); + CHECK( matchMuons(muonCont, muonMask, vtx, + vtxContName+"_accMuons") ); + } + } // if m_matchCalMuons + } // if m_thinMuons + } else { + ATH_MSG_DEBUG("doThinning(): ivtx " << ivtx + << " did not pass selection"); + } + } // for ivtx (B vertex) + + // Apply the thinning service based on vtxMask + CHECK( applyThinMask(vtxCont, vtxMask, m_vertexAnd, vtxContName) ); + + } // for m_vtxContNames + + // Keep tracks for all PVs (i.e. if all PVs are kept) + if ( (m_thinTracks || m_thinMuons) && m_doTracks && m_keepPVTracks && + !m_thinPVs && m_doPVs ) { + for (auto &pv : *pvCont) { + for (size_t ipvt=0; ipvt < pv->nTrackParticles(); ++ipvt) { + const xAOD::TrackParticle* tp = pv->trackParticle(ipvt); + if ( tp == nullptr ) continue; + auto tpit = std::find(trkPartCont->begin(), trkPartCont->end(), tp); + if ( tpit == trkPartCont->end() ) { + ATH_MSG_WARNING("PV track not found in " + << m_trkPartContName << " for PV from " + << m_PVContName); + continue; + } + size_t x = std::distance(trkPartCont->begin(), tpit); + if ( !trkMask.at(x) ) { + trkMask.at(x) = true; + addToCounter("addTracksByAllPVs"); + } + } // for ipvt + } // for pv + } // if m_keepPVTracks && m_doTracks && m_keepPVTracks && !m_thinPVs + + // Keep (original) muons for selected ID tracks + if (m_keepSelTrackMuons && m_thinMuons && m_doMuons && m_doTracks) { + CHECK( markMuonsForSelTracks(trkPartCont, trkMask, muonCont, muonMask, + "addMuonsBySelTracks") ); + } + + // Keep (original) muons for selected ID tracks + if (m_keepSelTrackCalMuons && m_thinMuons && m_doCalMuons && m_doTracks) { + CHECK( markMuonsForSelTracks(trkPartCont, trkMask, calMuonCont, + calMuonMask, "addCalMuonsBySelTracks") ); + } + + // mark 'other' muon container elements if requested + if ( m_thinMuons ) { + if ( m_syncMuonsBothWays || m_matchCalMuons ) { + // calibrated muons -> original muons + if ( (m_markMuons || m_syncMuonsBothWays) + && m_doMuons && m_doCalMuons) { + CHECK( markOrigMuons(muonCont, calMuonCont, muonMask, calMuonMask, + "addMarkedMuons", m_allowFastMuonMaskSync) ); + } + } // if m_syncMuonsBothWays || m_matchCalMuons ) { + if ( m_syncMuonsBothWays || !m_matchCalMuons ) { + // 'orignal' muons -> calibrated muons + if ( (m_markCalMuons || m_syncMuonsBothWays) + && m_doCalMuons && m_doMuons ) { + CHECK( markCalibMuons(muonCont, calMuonCont, muonMask, calMuonMask, + "addMarkedCalMuons", m_allowFastMuonMaskSync) ); + } + } // if m_syncMuonsBothWays || !m_matchCalMuons ) { + } // if m_thinMuons + + // Keep tracks for selected (original) muons + if ( (m_thinTracks || m_thinMuons) && m_doTracks && m_keepSelMuonTracks + && m_doMuons ) { + CHECK( markTrksForSelMuons(trkPartCont, trkMask, muonCont, muonMask, + "addTracksBySelMuons") ); + } // if (m_thinTracks || m_thinMuons) && m_doTracks && m_keepSelMuonTracks + // && m_doMuons + + // Keep tracks for selected (calibrated) muons + if ( (m_thinTracks || m_thinMuons) && m_doTracks && m_keepSelCalMuonTracks + && m_doCalMuons ) { + CHECK( markTrksForSelMuons(trkPartCont, trkMask, calMuonCont, + calMuonMask, "addTracksBySelCalMuons") ); + } // if (m_thinTracks || m_thinMuons) && m_doTracks + // && m_keepSelCalMuonTracks && m_doCalMuons + + // debug: check muon masks' consistency + if ( msgLvl(MSG::DEBUG) ) { + std::string msg = + checkMaskConsistency(muonMask, calMuonMask, + m_muonContName+"Mask", + m_calMuonContName+"Mask", + "Muon mask consistency check:"); + logWrappedMsg(msg, MSG::DEBUG); + } + + // Apply the thinning service for TrackParticles based on trkMask + if ( m_thinTracks && m_doTracks ) { + addToCounter(m_trkPartContName+"_allTracks", trkPartCont->size()); + addToCounter(m_trkPartContName+"_passedTracks", + std::accumulate(trkMask.begin(), trkMask.end(), 0)); + CHECK( applyThinMask(trkPartCont, trkMask, m_trackAnd, + m_trkPartContName) ); + } // if m_thinTracks && m_doTracks + + // Apply the thinning service for PVs based on pvMask + if ( m_thinPVs && m_doPVs ) { + addToCounter(m_PVContName+"_allVertices", pvCont->size()); + addToCounter(m_PVContName+"_passedVertices", + std::accumulate(pvMask.begin(), pvMask.end(), 0)); + CHECK( applyThinMask(pvCont, pvMask, m_vertexAnd, m_PVContName) ); + } // if m_doPVs + + // Apply the thinning service for refPVs based on vRefPvMasks + if ( m_thinRefPVs && m_doRefPVs ) { + for (size_t irpv = 0; irpv < m_refPVContNames.size(); ++irpv) { + addToCounter(m_refPVContNames[irpv]+"_allVertices", + vRefPvCont[irpv]->size()); + addToCounter(m_refPVContNames[irpv]+"_passedVertices", + std::accumulate(vRefPvMasks[irpv].begin(), + vRefPvMasks[irpv].end(), 0)); + CHECK( applyThinMask(vRefPvCont[irpv], vRefPvMasks[irpv], + m_vertexAnd, m_refPVContNames[irpv]) ); + } // for irpv + } // if m_doRefPVs + + // Apply the thinning service for (original) Muons based on muonMask + if ( m_thinMuons && m_doMuons ) { + addToCounter(m_muonContName+"_allMuons", muonCont->size()); + addToCounter(m_muonContName+"_passedMuons", + std::accumulate(muonMask.begin(), muonMask.end(), 0)); + CHECK( applyThinMask(muonCont, muonMask, m_muonAnd, + m_muonContName) ); + } // if m_thinMuons && m_doMuons + + // Apply the thinning service for calibrated Muons based on calMuonMask + if ( m_thinMuons && m_doCalMuons ) { + addToCounter(m_calMuonContName+"_allMuons", calMuonCont->size()); + addToCounter(m_calMuonContName+"_passedMuons", + std::accumulate(calMuonMask.begin(), calMuonMask.end(), + 0)); + CHECK( applyThinMask(calMuonCont, calMuonMask, m_muonAnd, + m_calMuonContName) ); + } // if m_thinMuons && m_doCalMuons + + return StatusCode::SUCCESS; + } + + //-------------------------------------------------------------------------- + // Helper to apply thinning service mask -- for all Containers + //-------------------------------------------------------------------------- + template<typename TYPE> + StatusCode + BmumuThinningTool::applyThinMask(SG::ThinningHandle<TYPE> &muCont, + const std::vector<bool>& muMask, + bool doAnd) const { + + if (doAnd) { + ATH_MSG_DEBUG("doThinning(): apply thinning (AND) for " << muCont.key()); + muCont.keep(muMask, SG::ThinningHandleBase::Op::And); + } else { + ATH_MSG_DEBUG("doThinning(): apply thinning (OR) for " << muCont.key()); + muCont.keep(muMask, SG::ThinningHandleBase::Op::Or); + } + return StatusCode::SUCCESS; + } + //-------------------------------------------------------------------------- + // Helper to check whether an element is marked as passing a specific + // hypothesis. + //-------------------------------------------------------------------------- + bool BmumuThinningTool::pass(const SG::AuxElement& em, std::string hypo) + const { + + if ( !boost::algorithm::starts_with(hypo, "passed_") ) + hypo = "passed_" + hypo; + SG::AuxElement::Accessor<Char_t> flagAcc(hypo); + return flagAcc.isAvailable(em) && flagAcc(em) != 0; + } + //-------------------------------------------------------------------------- + // Helper to get a TrackParticle link + //-------------------- ------------------------------------------------------ + const xAOD::TrackParticle* + BmumuThinningTool::getTrackParticle(const xAOD::Vertex* vtx, + std::string name) const { + SG::AuxElement::Accessor<TrackParticleLink> tpLinkAcc(name); + if (!tpLinkAcc.isAvailable(*vtx)) { + return nullptr; + } + const TrackParticleLink& tpLink = tpLinkAcc(*vtx); + if (!tpLink.isValid()) { + return nullptr; + } + return *tpLink; + } + //-------------------------------------------------------------------------- + // Helper to filter all names of auxillary elements of an aux container + // according to a certain pattern. The pattern must be a regular + // expression pattern. + //-------------------------------------------------------------------------- + std::vector<std::string> + BmumuThinningTool::filterAuxElements(const xAOD::AuxContainerBase* auxCont, + std::string pattern) const { + + SG::AuxTypeRegistry& reg = SG::AuxTypeRegistry::instance(); + + std::vector<std::string> vElNames; + std::regex re(pattern); + + const SG::auxid_set_t& auxids = auxCont->getAuxIDs(); + for ( SG::auxid_t auxid : auxids ) { + const std::string name = reg.getName(auxid); + if ( std::regex_match(name, re) ) { + vElNames.push_back(name); + } + } // for auxids + + return vElNames; + } + //-------------------------------------------------------------------------- + // Determine aux elements to be looked at -- for (refitted) PVs + //-------------------------------------------------------------------------- + void + BmumuThinningTool::selectAuxElements(const xAOD::AuxContainerBase* auxCont, + std::vector<std::string>& vLinkNames, + std::vector<pv_type>& vLinkTypes, + std::string pattern) const { + // find aux element names matching pattern + std::vector<std::string> vAuxNames = + filterAuxElements(auxCont, pattern); + + // select aux element names matching our PV-to-SV association types + for (auto &name : vAuxNames) { + for (size_t ipvt=0; ipvt < xAOD::BPhysHelper::n_pv_types; ++ipvt) { + if ( name.find(PvTypeToVarNameMap[(pv_type)ipvt] + +"Link") != std::string::npos) { + vLinkNames.push_back(name); + vLinkTypes.push_back((pv_type)ipvt); + } + } // for ipvt + } // for name + } + //-------------------------------------------------------------------------- + // Determine aux elements to be looked at -- for closest tracks + //-------------------------------------------------------------------------- + void + BmumuThinningTool::selectAuxElements(const xAOD::AuxContainerBase* auxCont, + std::vector<std::string>& vLinkNames, + std::vector<std::string> vPrefixes, + std::vector<pv_type>& vLinkTypes, + std::string pattern) const { + // find aux element names matching pattern + std::vector<std::string> vAuxNames; + for (auto &prefix : vPrefixes) { + std::string cpat = prefix+pattern; + std::vector<std::string> vMoreAuxNames = + filterAuxElements(auxCont, cpat); + vAuxNames.insert(vAuxNames.end(), vMoreAuxNames.begin(), + vMoreAuxNames.end()); + } // for prefix + + // select aux element names matching our PV-to-SV association types + for (auto &name : vAuxNames) { + for (size_t ipvt=0; ipvt < xAOD::BPhysHelper::n_pv_types; ++ipvt) { + std::regex re(".*"+xAOD::BPhysHelper::pv_type_str[ipvt]+".*_Link"); + if ( std::regex_match(name, re) ) { + vLinkNames.push_back(name); + vLinkTypes.push_back((pv_type)ipvt); + } + } // for ipvt + } // for name + } + //-------------------------------------------------------------------------- + // Mark muons matched to secondary vertices + //-------------------------------------------------------------------------- + StatusCode + BmumuThinningTool::matchMuons(const xAOD::MuonContainer* muCont, + std::vector<bool>& muMask, + xAOD::BPhysHelper& vtx, + std::string counterName) const { + + for (size_t imu=0; imu < muCont->size(); ++imu) { + // only consider if not yet kept + if ( !muMask[imu] ) { + for (int ivm=0; ivm < vtx.nMuons(); ++ivm) { + if ( vtx.muon(ivm) == muCont->at(imu) ) { + muMask[imu] = true; + addToCounter(counterName); + } + } // for ivm + } + } // for imu + return StatusCode::SUCCESS; + } + //-------------------------------------------------------------------------- + // Mark original muons for accepted calibrated muons + //-------------------------------------------------------------------------- + StatusCode + BmumuThinningTool::markOrigMuons(const xAOD::MuonContainer* muCont, + const xAOD::MuonContainer* cmuCont, + std::vector<bool>& muMask, + std::vector<bool>& cmuMask, + std::string counterName, + bool allowFastSync) const { + + bool fastSync = allowFastSync; + // go to slow sync if muon mask sizes do not match + if ( muMask.size() != cmuMask.size() ) { + fastSync = false; + addToCounter(counterName+"_maskSizeMismatches"); + } + + for (size_t imu=0; imu < cmuMask.size(); ++imu) { + if ( cmuMask[imu] ) { + if ( fastSync ) { + if ( !muMask[imu] ) { + muMask[imu] = true; + addToCounter(counterName); + } + } else { + const xAOD::Muon* cMuon = cmuCont->at(imu); + if ( cMuon != nullptr ) { + const xAOD::Muon* oMuon = + (const xAOD::Muon*)xAOD::getOriginalObject(*cMuon); + if ( oMuon != nullptr ) { + auto muit = std::find(muCont->begin(), muCont->end(), oMuon); + if ( muit == muCont->end() ) { + ATH_MSG_WARNING("Muon not found in " << m_muonContName + << " for calibrated muon index " << imu); + } else { + size_t x = std::distance(muCont->begin(), muit); + if ( !muMask.at(x) ) { + muMask.at(x) = true; + addToCounter(counterName); + } + } + } else { + ATH_MSG_WARNING("No orignal muon for calibrated muon index " + << imu); + } + } else { + ATH_MSG_WARNING("No calibrated muon for index " << imu); + } + } // if fastSync + } + } // for imu + return StatusCode::SUCCESS; + } + //-------------------------------------------------------------------------- + // Mark calibrated muons for accepted (original) muons + //-------------------------------------------------------------------------- + StatusCode + BmumuThinningTool::markCalibMuons(const xAOD::MuonContainer* muCont, + const xAOD::MuonContainer* cmuCont, + std::vector<bool>& muMask, + std::vector<bool>& cmuMask, + std::string counterName, + bool allowFastSync) const { + + bool fastSync = allowFastSync; + // go to slow sync if muon mask sizes do not match + if ( muMask.size() != cmuMask.size() ) { + fastSync = false; + addToCounter(counterName+"_maskSizeMismatches"); + } + + for (size_t imu=0; imu < muMask.size(); ++imu) { + if ( muMask[imu] ) { + if ( fastSync ) { + if ( !cmuMask[imu] ) { + cmuMask[imu] = true; + addToCounter(counterName); + } + } else { + const xAOD::Muon* oMuon = muCont->at(imu); + if ( oMuon != nullptr ) { + bool found = false; + for (size_t icmu = 0; icmu < cmuCont->size(); ++icmu) { + const xAOD::Muon* cMuon = cmuCont->at(icmu); + if ( cMuon != nullptr ) { + const xAOD::Muon* aMuon = + (const xAOD::Muon*)xAOD::getOriginalObject(*cMuon); + if ( aMuon == oMuon ) { + found = true; + if ( !cmuMask.at(icmu) ) { + cmuMask.at(icmu) = true; + addToCounter(counterName); + } + } + } else { + ATH_MSG_WARNING("No calibrated muon for calibrated " + << "muon index " << icmu); + } + } // for icmu + if ( !found ) { + ATH_MSG_WARNING("No calibrated muon found for orignal " + << "muon index " << imu); + } + } else { + ATH_MSG_WARNING("No (original) muon for index " << imu); + } + } // if fastSync + } + } // for imu + return StatusCode::SUCCESS; + } + //-------------------------------------------------------------------------- + // Mark ID tracks of selected (original or calibrated) muons + //-------------------------------------------------------------------------- + StatusCode + BmumuThinningTool::markTrksForSelMuons(const xAOD::TrackParticleContainer* + trkPartCont, + std::vector<bool>& trkMask, + const xAOD::MuonContainer* muCont, + std::vector<bool>& muMask, + std::string counterName) const { + + for (size_t itrk=0; itrk < trkPartCont->size(); ++itrk) { + if ( trkMask[itrk] ) continue; + const xAOD::TrackParticle* tp = trkPartCont->at(itrk); + if ( tp == nullptr ) continue; + for (size_t imu=0; imu < muCont->size(); ++imu) { + if ( muMask[imu] ) { + const xAOD::Muon* muon = muCont->at(imu); + if ( muon != nullptr ) { + const xAOD::TrackParticle* mutp = + muon->trackParticle(xAOD::Muon::InnerDetectorTrackParticle); + if ( mutp == tp ) { + trkMask[itrk] = true; + addToCounter(counterName); + } + } + } + } // for imu + } // for itrk + return StatusCode::SUCCESS; + } + //-------------------------------------------------------------------------- + // Mark muons for selected ID tracks + //-------------------------------------------------------------------------- + StatusCode + BmumuThinningTool::markMuonsForSelTracks(const xAOD::TrackParticleContainer* + trkPartCont, + std::vector<bool>& trkMask, + const xAOD::MuonContainer* muCont, + std::vector<bool>& muMask, + std::string counterName) const { + + for (size_t imu=0; imu < muCont->size(); ++imu) { + if ( muMask[imu] ) continue; + const xAOD::Muon* muon = muCont->at(imu); + if ( muon == nullptr ) continue; + const xAOD::TrackParticle* mutp = + muon->trackParticle(xAOD::Muon::InnerDetectorTrackParticle); + if ( mutp == nullptr) continue; + auto tpit = std::find(trkPartCont->begin(), trkPartCont->end(), mutp); + if ( tpit == trkPartCont->end() ) { + ATH_MSG_WARNING("Muon track not found in " << m_trkPartContName + << " for counter " << counterName); + addToCounter(counterName+"_missingTracksForMuons"); + continue; + } + size_t x = std::distance(trkPartCont->begin(), tpit); + if ( trkMask.at(x) ) { + muMask[imu] = true; + addToCounter(counterName); + } + } // for imu + return StatusCode::SUCCESS; + } + + //-------------------------------------------------------------------------- + // Check two masks for consistency + //-------------------------------------------------------------------------- + std::string + BmumuThinningTool::checkMaskConsistency(const std::vector<bool>& mask1, + const std::vector<bool>& mask2, + const std::string name1, + const std::string name2, + const std::string header) const { + + bool sizesMatch = (mask1.size() == mask2.size()); + int nTrueMask1 = std::accumulate(mask1.begin(), mask1.end(), 0); + int nTrueMask2 = std::accumulate(mask2.begin(), mask2.end(), 0); + + std::string basecname = name1+"_"+name2; + + int nEntryMismatches(0); + int nMoreTrueMask1(0); + int nMoreTrueMask2(0); + for (size_t i=0; i < std::min(mask1.size(), mask2.size()); ++i) { + if ( mask1[i] != mask2[i] ) nEntryMismatches++; + if ( mask1[i] && !mask2[i] ) nMoreTrueMask1++; + if ( !mask1[i] && mask2[i] ) nMoreTrueMask2++; + } + + std::string str(header); + if ( str.length() > 0 ) str += "\n"; + if ( sizesMatch && nTrueMask1 == nTrueMask2 && nEntryMismatches == 0 ) { + str += "Masks match OK: "+name1+" ("+mask1.size()+") : " + + name2+" ("+mask2.size()+")"; + addToCounter(basecname+"_matchedOK"); + } else { + str += "Masks do NOT match: "+name1+" ("+mask1.size()+") : " + + name2+" ("+mask2.size()+")"; + str += "\nnTrueMask1: "+std::to_string(nTrueMask1) + +", nTrueMask2: "+std::to_string(nTrueMask2); + str += "\nnEntryMismatches: "+std::to_string(nEntryMismatches) + +"nMoreTrueMas1: "+std::to_string(nMoreTrueMask1) + +", nMoreTrueMask2: "+std::to_string(nMoreTrueMask2); + addToCounter(basecname+"_NOTmatched"); + if (!sizesMatch) addToCounter(basecname+"_sizeMismatch"); + addToCounter(basecname+"_nEntryMismatches", nEntryMismatches); + addToCounter(basecname+"_nMoreTrueMask1", nMoreTrueMask1); + addToCounter(basecname+"_nMoreTrueMask2", nMoreTrueMask2); + } + return str; + } + //-------------------------------------------------------------------------- + // Dump a vector<string> to string. + //-------------------------------------------------------------------------- + std::string BmumuThinningTool::dumpVS(const std::vector<std::string>& vs, + const std::string header, + size_t nBlanks) const { + std::string str(header); + for (const std::string& s : vs) { + if ( str.length() > 0 ) str += "\n"; + str += std::string(nBlanks, ' ') + s; + } // for s + + return str; + } + //-------------------------------------------------------------------------- + // Wrap string at line breaks and print with appropriate message level + //-------------------------------------------------------------------------- + void BmumuThinningTool::logWrappedMsg(const std::string& str, + const MSG::Level lvl) const { + std::istringstream isstr(str); + std::string s; + while (std::getline(isstr, s)) { + msg(lvl) << s << endreq; + } + } + //-------------------------------------------------------------------------- + +} // namespace DerivationFramework diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/Bmumu_metadata.cxx b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/Bmumu_metadata.cxx new file mode 100644 index 00000000000..5d45a327e69 --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/Bmumu_metadata.cxx @@ -0,0 +1,238 @@ +/* + Copyright (C) 2002-2020 CERN for the benefit of the ATLAS collaboration +*/ + +/** + * @file Bmumu_metadata.cxx + * @author Wolfgang Walkowiak <wolfgang.walkowiak@cern.ch> + */ + +#include "DerivationFrameworkBPhys/Bmumu_metadata.h" + +namespace DerivationFramework { + + //-------------------------------------------------------------------------- + Bmumu_metadata::Bmumu_metadata(const std::string& t, + const std::string& n, + const IInterface* p) : + AthAlgTool(t,n,p), BPhysMetadataBase(t,n,p) { + + // configuration defaults etc. + recordPropertyI("verbose", 0); + recordPropertyB("isSimulation", false); + recordPropertyS("projectTag", "__NONE__"); + recordPropertyB("isRelease21" , true); + recordPropertyS("mcCampaign", "__NONE__"); + recordPropertyS("triggerStream", "__NONE__"); + + // MC dataset number lists + recordPropertyVI("mcBsmumu" , {}); + recordPropertyVI("mcBplusJpsiKplus" , {}); + recordPropertyVI("mcBsJpsiPhi" , {}); + recordPropertyVI("mcBplusJpsiPiplus", {}); + recordPropertyVI("mcBhh" , {}); + + // MC datasets without trigger information + recordPropertyVI("mcNoTrigger" , {}); + + // special data runs + recordPropertyVI("specDataRuns", {}); + + // special MC channels + recordPropertyVI("specMcChannels", {}); + + // blind search + recordPropertyB("doBmumuBlinding" , true); + recordPropertyB("doCutBlinded" , true); + recordPropertyB("blindOnlyAllMuonsTight", true); + recordPropertyS("BlindingKey" , ""); + recordPropertyS("BlindedVars" , ""); + recordPropertyS("BlindingFlag" , ""); + + // include trigger + recordPropertyB("doTriggerInfo" , true); + + // include soft B tagging vertex containers + recordPropertyB("doAddSoftBVertices" , true); + + // trigger navigation thinning + recordPropertyB("doTrigNavThinning", true); + recordPropertyVS("TrigNavThinList" , {}); + + // wide mumu mass range + recordPropertyB("doUseWideMuMuMassRange", false); + + // use mass calculated using the combined muon track information in cuts? + recordPropertyB("useMuCalcMass", true); + + // use calibrated muons instead of the original ones + recordPropertyI("useCalibratedMuons", 0); + + // adjust primary track to muon kinematics for MUCALC mass + recordPropertyB("adjustMucalcKinematics", false); + + // add MUCALC mass from non-modified muons for debugging + recordPropertyB("addMucalcMassForDebug", false); + + // primary vertex types to consider for MinChi2ToAnyPV + recordPropertyVI("MinChi2ToAnyPVTypes", {1, 3}); + + // JpsiFinder: muAndMu or TrackAndTrack option? + recordPropertyB("JfTwoMuons" , true ); + recordPropertyB("JfTwoTracks", false); + + // JpsiFinder: TrackThresholdPt + recordPropertyD("JfTrackThresholdPt", 0.); + + // muon calibration and smearing tool configuration + recordPropertyS("McstYear" , "Data16"); + recordPropertyS("McstRelease" , "_NONE_"); + recordPropertyB("McstStatComb" , true); + recordPropertyB("McstSagittaCorr" , true); + recordPropertyS("McstSagittaRelease" , "_NONE_"); + recordPropertyB("McstDoSagittaMCDistortion", false); + recordPropertyB("McstSagittaCorrPhaseSpace", true); + + // muon collections + recordPropertyS("MuonCollection" , "Muons"); + recordPropertyS("CalMuonCollection" , "Muons"); + recordPropertyS("UsedMuonCollection" , "Muons"); + recordPropertyVS("AllMuonCollections", {} ); + + // Global mass values (in MeV, from PDG 2015) + recordPropertyD("GlobalMuonMass" , 105.6584); + recordPropertyD("GlobalPionMass" , 139.57061); + recordPropertyD("GlobalKaonMass" , 493.677 ); + recordPropertyD("GlobalJpsiMass" , 3096.92 ); + recordPropertyD("GlobalBplusMass", 5279.29 ); + recordPropertyD("GlobalB0Mass" , 5279.61 ); + recordPropertyD("GlobalBsMass" , 5366.79 ); + + // mass ranges + recordPropertyD("GlobalBMassUpperCut" , 7000.); + recordPropertyD("GlobalBMassLowerCut" , 3500.); + recordPropertyD("GlobalDiMuonMassUpperCut", 7000.); + recordPropertyD("GlobalDiMuonMassLowerCut", 2000.); + recordPropertyD("GlobalJpsiMassUpperCut" , 7000.); + recordPropertyD("GlobalJpsiMassLowerCut" , 2000.); + recordPropertyD("GlobalBlindUpperCut" , 5166.); + recordPropertyD("GlobalBlindLowerCut" , 5526.); + recordPropertyD("GlobalTrksMassUpperCut" , 7500.); + recordPropertyD("GlobalTrksMassLowerCut" , 3000.); + + // Global chi2 cut for vertexing + recordPropertyD("GlobalChi2CutBase", 15.0); + // Different chi2 cuts for 2-, 3- and 4-prong vertices + recordPropertyD("Chi2Cut2Prong" , 30.0); + recordPropertyD("Chi2Cut3Prong" , 45.0); + recordPropertyD("Chi2Cut4Prong" , 60.0); + + // Cut values for kaon candidates + recordPropertyD("GlobalKaonPtCut" , 1000.); // MeV + recordPropertyD("GlobalKaonEtaCut", 2.5 ); + + // MCP cuts for JpsiFinder + recordPropertyB("useJpsiFinderMCPCuts", false); + + // reject muons in JpsiPlus1Track or JpsiPlus2Track finders + recordPropertyS("GlobalMuonsUsedInJpsi", "NONE"); // turn off by default + + // run number + recordPropertyI("runNumber", -1); + + // MC channel number + recordPropertyI("mcChNumber", -1); + + // channels to be processed + recordPropertyVS("doChannels", {}); + + // vertex types to be done + recordPropertyI("doVertexType", 7); + + // minimum number of tracks in PV considered for PV association + recordPropertyI("minNTracksInPV", 0); + + // mode of minLogChi2ToAnyPV calculation + recordPropertyI("AddMinChi2ToAnyPVMode", 0); + + // record 3-dimensional proper time in addition + recordPropertyB("do3dProperTime", false); + + // thinning level + recordPropertyI("thinLevel", 0); + + // selection expression + recordPropertyS("SelExpression", ""); + + // MC truth decay parents + recordPropertyVI("TruthDecayParents", {}); + + // vertex isolation properties + recordPropertyVS("IsoTrackCategoryName" , {}); + recordPropertyVS("IsoTrackCutLevel" , {}); + recordPropertyVD("IsoTrackPtCut" , {}); + recordPropertyVD("IsoTrackEtaCut" , {}); + recordPropertyVI("IsoTrackPixelHits" , {}); + recordPropertyVI("IsoTrackSCTHits" , {}); + recordPropertyVI("IsoTrackbLayerHits" , {}); + recordPropertyVI("IsoTrackIBLHits" , {}); + recordPropertyVD("IsolationConeSizes" , {}); + recordPropertyVD("IsoTrkImpLogChi2Max" , {}); + recordPropertyVI("IsoDoTrkImpLogChi2Cut", {}); + recordPropertyVL("useIsoTrackTypes" , {}); + recordPropertyB("IsoUseOptimizedAlgo" , true); + recordPropertyS("IsoTvaWorkingPoint" , "Nominal"); + recordPropertyVS("IsoIncludes" , {}); + + + // muon isolation properties (muons of B candidate) + recordPropertyVS("MuIsoTrackCategoryName" , {}); + recordPropertyVS("MuIsoTrackCutLevel" , {}); + recordPropertyVD("MuIsoTrackPtCut" , {}); + recordPropertyVD("MuIsoTrackEtaCut" , {}); + recordPropertyVI("MuIsoTrackPixelHits" , {}); + recordPropertyVI("MuIsoTrackSCTHits" , {}); + recordPropertyVI("MuIsoTrackbLayerHits" , {}); + recordPropertyVI("MuIsoTrackIBLHits" , {}); + recordPropertyVD("MuIsolationConeSizes" , {}); + recordPropertyVD("MuIsoTrkImpLogChi2Max" , {}); + recordPropertyVI("MuIsoDoTrkImpLogChi2Cut", {}); + recordPropertyVL("useMuIsoTrackTypes" , {}); + recordPropertyS("MuIsoTvaWorkingPoint" , "Nominal"); + recordPropertyVS("MuIsoIncludes" , {}); + + // closest track properties + recordPropertyVS("CloseTrackCategoryName" , {}); + recordPropertyVS("CloseTrackCutLevel" , {}); + recordPropertyVD("CloseTrackPtCut" , {}); + recordPropertyVD("CloseTrackEtaCut" , {}); + recordPropertyVI("CloseTrackPixelHits" , {}); + recordPropertyVI("CloseTrackSCTHits" , {}); + recordPropertyVI("CloseTrackbLayerHits" , {}); + recordPropertyVI("CloseTrackIBLHits" , {}); + recordPropertyVL("useCloseTrackTypes" , {}); + recordPropertyVS("CloseTrackChi2SetName" , {}); + recordPropertyVI("CloseTrackCorrChi2" , {}); + recordPropertyVB("CloseTrackMinDCAin3D" , {}); + recordPropertyVD("CloseTrackMaxLogChi2" , {}); + recordPropertyVD("NCloseTrackMaxLogChi2" , {}); + recordPropertyS("CloseTrackTvaWorkingPoint", "Nominal"); + recordPropertyVS("CloseTrackIncludes" , {}); + + // debug track types for isolation and closest track tools + recordPropertyI("DebugTrackTypes", 0); + + // track-to-vertex association check tool + recordPropertyI("DebugTrkToVtxMaxEvents" , 0); + + // output containers and branch prefixes + // (mostly used for isolation tools) + recordPropertyS("TrkPartContName", "InDetTrackParticles"); + recordPropertyS("PVContName" , "PrimaryVertices"); + recordPropertyVS("VtxContNames" , {} ); + recordPropertyVS("RefPVContNames", {} ); + recordPropertyVS("BranchPrefixes", {} ); + + } + //-------------------------------------------------------------------------- +} // namespace diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/Bmumu_reco_mumu.cxx b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/Bmumu_reco_mumu.cxx new file mode 100644 index 00000000000..a3c06874ffa --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/Bmumu_reco_mumu.cxx @@ -0,0 +1,177 @@ +/* + Copyright (C) 2002-2018 CERN for the benefit of the ATLAS collaboration +*/ + +///////////////////////////////////////////////////////////////// +// Bmumu_reco_mumu.cxx +/////////////////////////////////////////////////////////////////// +// +// Author : Wolfgang Walkowiak <Wolfgang.Walkowiak@cern.ch.> +// Original author (Reco_mumu): +// Daniel Scheirich <daniel.scheirich@cern.ch> +// +// Changes: +// Basic dimuon reconstruction for the derivation framework. +// This class inherits from CfAthAlgTool instead of AthAlgTool in order +// to have access to the CutFlowSvc instance. +// +//============================================================================ +// + +#include "DerivationFrameworkBPhys/Bmumu_reco_mumu.h" + +#include "xAODTracking/VertexContainer.h" +#include "xAODTracking/VertexAuxContainer.h" +#include "TrkVertexAnalysisUtils/V0Tools.h" +#include "BeamSpotConditionsData/BeamSpotData.h" +#include "DerivationFrameworkBPhys/BPhysPVTools.h" + + +namespace DerivationFramework { + + Bmumu_reco_mumu::Bmumu_reco_mumu(const std::string& t, + const std::string& n, + const IInterface* p) : + CfAthAlgTool(t,n,p), + m_v0Tools("Trk::V0Tools"), + m_jpsiFinder("Analysis::JpsiFinder"), + m_pvRefitter("Analysis::PrimaryVertexRefitter") + { + declareInterface<DerivationFramework::IAugmentationTool>(this); + + // Declare tools + declareProperty("V0Tools" , m_v0Tools); + declareProperty("JpsiFinder", m_jpsiFinder); + declareProperty("PVRefitter", m_pvRefitter); + + // Declare user-defined properties + declareProperty("OutputVtxContainerName", m_outputVtxContainerName = "OniaCandidates"); + declareProperty("PVContainerName" , m_pvContainerName = "PrimaryVertices"); + declareProperty("RefPVContainerName" , m_refPVContainerName = "RefittedPrimaryVertices"); + declareProperty("RefitPV" , m_refitPV = false); + declareProperty("MaxPVrefit" , m_PV_max = 1); + declareProperty("DoVertexType" , m_DoVertexType = 1); + // minimum number of tracks for PV to be considered for PV association + declareProperty("MinNTracksInPV" , m_PV_minNTracks = 0); + declareProperty("Do3d" , m_do3d = false); + } + + // * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * + + StatusCode Bmumu_reco_mumu::initialize() + { + + ATH_MSG_DEBUG("in initialize()"); + + // retrieve V0 tools + CHECK( m_v0Tools.retrieve() ); + + // get the JpsiFinder tool + CHECK( m_jpsiFinder.retrieve() ); + + // get the PrimaryVertexRefitter tool + CHECK( m_pvRefitter.retrieve() ); + + // Get the beam spot service + CHECK( m_beamSpotKey.initialize() ); + + return StatusCode::SUCCESS; + + } + + // * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * + + StatusCode Bmumu_reco_mumu::finalize() + { + // everything all right + return StatusCode::SUCCESS; + } + + // * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * + + StatusCode Bmumu_reco_mumu::addBranches() const + { + // Jpsi container and its auxilliary store + xAOD::VertexContainer* vtxContainer = NULL; + xAOD::VertexAuxContainer* vtxAuxContainer = NULL; + + //---------------------------------------------------- + // call Jpsi finder + //---------------------------------------------------- + if( !m_jpsiFinder->performSearch(vtxContainer, vtxAuxContainer).isSuccess() ) { + ATH_MSG_FATAL("Jpsi finder (" << m_jpsiFinder << ") failed."); + return StatusCode::FAILURE; + } + + //---------------------------------------------------- + // retrieve primary vertices + //---------------------------------------------------- + const xAOD::VertexContainer* pvContainer = NULL; + CHECK( evtStore()->retrieve(pvContainer, m_pvContainerName) ); + + //---------------------------------------------------- + // Try to retrieve refitted primary vertices + //---------------------------------------------------- + bool refPvExists = false; + xAOD::VertexContainer* refPvContainer = NULL; + xAOD::VertexAuxContainer* refPvAuxContainer = NULL; + if(m_refitPV) { + if(evtStore()->contains<xAOD::VertexContainer>(m_refPVContainerName)) { + // refitted PV container exists. Get it from the store gate + CHECK( evtStore()->retrieve(refPvContainer, m_refPVContainerName) ); + CHECK( evtStore()->retrieve(refPvAuxContainer, m_refPVContainerName+"Aux.") ); + refPvExists = true; + } else { + // refitted PV container does not exist. Create a new one. + refPvContainer = new xAOD::VertexContainer; + refPvAuxContainer = new xAOD::VertexAuxContainer; + refPvContainer->setStore(refPvAuxContainer); + } + } + + // Give the helper class the ptr to v0tools and beamSpotsSvc to use + SG::ReadCondHandle<InDet::BeamSpotData> beamSpotHandle { m_beamSpotKey }; + if(not beamSpotHandle.isValid()) ATH_MSG_ERROR("Cannot Retrieve " << m_beamSpotKey.key() ); + BPhysPVTools helper(&(*m_v0Tools), beamSpotHandle.cptr()); + helper.SetMinNTracksInPV(m_PV_minNTracks); + helper.SetSave3d(m_do3d); + + if(m_refitPV){ + if(vtxContainer->size() >0){ + StatusCode SC = helper.FillCandwithRefittedVertices(vtxContainer, pvContainer, refPvContainer, &(*m_pvRefitter) , m_PV_max, m_DoVertexType); + if(SC.isFailure()){ + ATH_MSG_FATAL("refitting failed - check the vertices you passed"); + return SC; + } + } + }else{ + if(vtxContainer->size() >0)CHECK(helper.FillCandExistingVertices(vtxContainer, pvContainer, m_DoVertexType)); + } + + + //---------------------------------------------------- + // save in the StoreGate + //---------------------------------------------------- + if (!evtStore()->contains<xAOD::VertexContainer>(m_outputVtxContainerName)) + CHECK(evtStore()->record(vtxContainer, m_outputVtxContainerName)); + + if (!evtStore()->contains<xAOD::VertexAuxContainer>(m_outputVtxContainerName+"Aux.")) + CHECK(evtStore()->record(vtxAuxContainer, m_outputVtxContainerName+"Aux.")); + + if(!refPvExists && m_refitPV) { + CHECK(evtStore()->record(refPvContainer , m_refPVContainerName)); + CHECK(evtStore()->record(refPvAuxContainer, m_refPVContainerName+"Aux.")); + } + + // add counter for number of events seen + addEvent("dimuEvents"); + // add counter for the number of events with >= 1 reco'd vertices + if ( vtxContainer->size() > 0 ) { + addEvent("dimuWithVertexCand"); + } + // add counter for the number of vertices + addToCounter("dimuNumVertices", vtxContainer->size()); + + return StatusCode::SUCCESS; + } +} diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/Cascade3Plus1.cxx b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/Cascade3Plus1.cxx new file mode 100644 index 00000000000..fbc231d3666 --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/Cascade3Plus1.cxx @@ -0,0 +1,524 @@ +/* + Copyright (C) 2002-2018 CERN for the benefit of the ATLAS collaboration +*/ +///////////////////////////////////////////////////////////////// +// Cascade3Plus1.cxx, (c) ATLAS Detector software +///////////////////////////////////////////////////////////////// +#include "DerivationFrameworkBPhys/Cascade3Plus1.h" +#include "xAODTracking/TrackParticle.h" +#include "xAODTracking/TrackParticleContainer.h" +#include "TrkToolInterfaces/ITrackSelectorTool.h" +#include "TrkVKalVrtFitter/TrkVKalVrtFitter.h" +#include "xAODTracking/VertexContainer.h" +#include "xAODTracking/VertexAuxContainer.h" +#include "DerivationFrameworkBPhys/BPhysPVCascadeTools.h" +#include "TrkVertexAnalysisUtils/V0Tools.h" +#include "DerivationFrameworkBPhys/BPhysPVTools.h" +#include "xAODBPhys/BPhysHelper.h" +#include "Math/Vector4D.h" + +namespace DerivationFramework { +typedef ElementLink<xAOD::VertexContainer> VertexLink; +typedef std::vector<VertexLink> VertexLinkVector; +typedef std::vector<const xAOD::TrackParticle*> TrackBag; +/// Base 4 Momentum type for TrackParticle +typedef ROOT::Math::LorentzVector<ROOT::Math::PtEtaPhiM4D<double> > GenVecFourMom_t; + +template<size_t N> +struct Candidate { + std::array<const xAOD::TrackParticle*, N> tracks; +}; + +struct VertexCand : Candidate<4> { + std::unique_ptr<Trk::VxCascadeInfo> cascVertex; +}; + +template<size_t N> +GenVecFourMom_t SumVector(const std::array<GenVecFourMom_t, N> &vectors) { + GenVecFourMom_t total = vectors[0]; + for(size_t i =1; i<N; i++) total+= vectors[i]; + return total; +} + +Cascade3Plus1::Cascade3Plus1(const std::string& t, const std::string& n, const IInterface* p) : AthAlgTool(t,n,p), + m_trkSelector("InDet::TrackSelectorTool"), + m_iVertexFitter("Trk::TrkVKalVrtFitter"), + m_V0Tools("Trk::V0Tools"), + m_CascadeTools("DerivationFramework::CascadeTools"), + m_pvRefitter("Analysis::PrimaryVertexRefitter"), + m_cascadeOutputsKeys{ "CascadeVtx1", "CascadeVtx2" } +{ + + declareProperty("TrackMassHyp", m_trackMasses); + declareProperty("CascadeVertexCollections", m_cascadeOutputsKeys); + declareProperty("TwoTrackMassMin", m_2trackmassMin); + declareProperty("TwoTrackMassMax", m_2trackmassMax); + declareProperty("ThreeTrackMassMin", m_3trackmassMin); + declareProperty("ThreeTrackMassMax", m_3trackmassMax); + declareProperty("FourTrackMassMin", m_4trackmassMin); + declareProperty("FourTrackMassMax", m_4trackmassMax); + declareProperty("TwoTracksMass", m_2tracksMass); + declareProperty("ThreeTracksMass", m_3tracksMass); + declareProperty("FourTracksMass", m_4tracksMass); + declareProperty("TrackSelectorTool",m_trkSelector); + declareProperty("CascadeTools", m_CascadeTools); + declareProperty("MinNTracksInPV", m_PV_minNTracks = 0); + declareProperty("HypothesisName", m_hypoName = "Bs"); + declareProperty("Track3Name", m_3TrackName = "Ds"); + declareProperty("MaxnPV", m_PV_max = 999); + declareProperty("DoVertexType", m_DoVertexType = 7); + declareProperty("RefitPV", m_refitPV = true); + declareProperty("RefPVContainerName", m_refPVContainerName = "RefittedPrimaryVertices"); + declareProperty("PVRefitter", m_pvRefitter); + declareProperty("TrkVertexFitterTool", m_iVertexFitter); + declareProperty("PVContainerName", m_VxPrimaryCandidateName); + declareProperty("ThreeTrackMassConstraint", m_3TrackMassConstraint); + declareProperty("TwoTrackMassConstraint", m_2TrackMassConstraint); + declareProperty("Chi2NDFCut", m_Chi2NDFCut); + + declareProperty("FourTrackMassFinalMin", m_4trackmassFinalMin); + declareProperty("FourTrackMassFinalMax", m_4trackmassFinalMax); + declareProperty("FourTrackTauCut", m_tauCut); + declareProperty("UseMuonsForTracks", m_requireMuonsOnTrack); + declareProperty("ThreeVertexOutputContainer", m_3TrackVertexOutput); + declareProperty("VertexEstimator", m_vertexEstimator); + declareProperty("ThreeTrackChi2NDF", m_3TrackChi2NDFCut); + declareProperty("EliminateBad3Tracksfrom4Track", m_eliminateBad3Tracksfrom4Track); + declareProperty("CopyAllVertices", m_copyAllVertices); + declareProperty("PTCutPerTrack", m_ptCutPerTrack); + m_ptCutPerVertex.fill(0); + declareProperty("PTCutVertex1", m_ptCutPerVertex[0]); + declareProperty("PTCutVertex2", m_ptCutPerVertex[1]); + declareProperty("PTCutVertex3", m_ptCutPerVertex[2]); +} + +StatusCode Cascade3Plus1::initialize() { + if(m_trackMasses.size()!=4) { + ATH_MSG_ERROR("4 mass hypotheses must be provided"); + return StatusCode::FAILURE; + } + if(m_cascadeOutputsKeys.size() !=s_topoN) { + ATH_MSG_FATAL("Incorrect number of VtxContainers"); + return StatusCode::FAILURE; + } + // retrieving vertex Fitter + ATH_CHECK( m_iVertexFitter.retrieve()); + + // retrieving the Cascade tools + ATH_CHECK( m_CascadeTools.retrieve()); + + // Get the beam spot service + CHECK( m_beamSpotKey.initialize() ); + + ATH_CHECK(m_vertexEstimator.retrieve()); + if(m_eliminateBad3Tracksfrom4Track && m_3TrackChi2NDFCut<=0.0) { + ATH_MSG_FATAL("Invalid configuration"); + return StatusCode::FAILURE; + } + + if(m_ptCutPerTrack.size() == 1 || m_ptCutPerTrack.size() > 4){ + ATH_MSG_FATAL("Invalid configuration"); + return StatusCode::FAILURE; + } + if(m_ptCutPerTrack.size() >=2 && m_ptCutPerTrack[0] != m_ptCutPerTrack[1]){ + ATH_MSG_FATAL("Invalid configuration"); + return StatusCode::FAILURE; + } + m_muonTrackBit.reset(); + for(int i : m_requireMuonsOnTrack) { + if(i>=4) { + ATH_MSG_FATAL("Invalid configuration" << " muon track " << i); + return StatusCode::FAILURE; + } + m_muonTrackBit[i] = true; + } + m_requireMuonsOnTrack.clear(); + m_requireMuonsOnTrack.shrink_to_fit(); + + if(m_muonTrackBit[0] != m_muonTrackBit[1]) + { + ATH_MSG_FATAL("Invalid configuration" << " variable is " << m_muonTrackBit.to_string()); + return StatusCode::FAILURE; + } + + + return StatusCode::SUCCESS; +} + +Cascade3Plus1::~Cascade3Plus1() { } + + +const TrackBag& Cascade3Plus1::ApplyAdditionalCuts(const TrackBag& alltracks, const TrackBag& muonTracks, TrackBag& cuttracks, size_t track) const { + const TrackBag& tracks = m_muonTrackBit[track] ? muonTracks : alltracks; + if(track >= m_ptCutPerTrack.size()) return tracks; + double ptCut = m_ptCutPerTrack.at(track); + if(ptCut <=0.0) return tracks; + cuttracks.clear();//reset any previous selections + for(auto ptr : tracks){ + if(ptr->pt() > ptCut) cuttracks.push_back(ptr); + } + return cuttracks; +} + +StatusCode Cascade3Plus1::addBranches() const +{ + const xAOD::TrackParticleContainer *trackContainer(nullptr); + ATH_CHECK(evtStore()->retrieve(trackContainer, "InDetTrackParticles" )); + + //---------------------------------------------------- + // Try to retrieve refitted primary vertices + //---------------------------------------------------- + xAOD::VertexContainer* refPvContainer = nullptr; + xAOD::VertexAuxContainer* refPvAuxContainer = nullptr; + if (m_refitPV) { + if (evtStore()->contains<xAOD::VertexContainer>(m_refPVContainerName)) { + // refitted PV container exists. Get it from the store gate + ATH_CHECK(evtStore()->retrieve(refPvContainer, m_refPVContainerName )); + ATH_CHECK(evtStore()->retrieve(refPvAuxContainer, m_refPVContainerName + "Aux.")); + } else { + // refitted PV container does not exist. Create a new one. + refPvContainer = new xAOD::VertexContainer; + refPvAuxContainer = new xAOD::VertexAuxContainer; + refPvContainer->setStore(refPvAuxContainer); + ATH_CHECK(evtStore()->record(refPvContainer, m_refPVContainerName)); + ATH_CHECK(evtStore()->record(refPvAuxContainer, m_refPVContainerName+"Aux.")); + } + } + + std::array<xAOD::VertexContainer*, s_topoN> Vtxwritehandles; + std::array<xAOD::VertexAuxContainer*, s_topoN> Vtxwritehandlesaux; + + for(int i =0; i<s_topoN; i++) { + Vtxwritehandles[i] = new xAOD::VertexContainer(); + Vtxwritehandlesaux[i] = new xAOD::VertexAuxContainer(); + Vtxwritehandles[i]->setStore(Vtxwritehandlesaux[i]); + ATH_CHECK(evtStore()->record(Vtxwritehandles[i], m_cascadeOutputsKeys[i] )); + ATH_CHECK(evtStore()->record(Vtxwritehandlesaux[i], m_cascadeOutputsKeys[i] + "Aux.")); + } + xAOD::VertexContainer *v3container = nullptr; + if(!m_3TrackVertexOutput.empty()) { + v3container = new xAOD::VertexContainer(); + auto vcontaineraux = new xAOD::VertexAuxContainer(); + v3container->setStore(vcontaineraux); + ATH_CHECK(evtStore()->record(v3container, m_3TrackVertexOutput )); + ATH_CHECK(evtStore()->record(vcontaineraux, m_3TrackVertexOutput + "Aux.")); + } + //---------------------------------------------------- + // retrieve primary vertices + //---------------------------------------------------- + + const xAOD::Vertex * primaryVertex(nullptr); + const xAOD::VertexContainer *pvContainer(nullptr); + ATH_CHECK(evtStore()->retrieve(pvContainer, m_VxPrimaryCandidateName)); + + if (pvContainer->size()==0) { + ATH_MSG_WARNING("You have no primary vertices: " << pvContainer->size()); + return StatusCode::RECOVERABLE; + } else { + primaryVertex = (*pvContainer)[0]; + } + + + TrackBag theIDTracksAfterSelection; + TrackBag theIDTracksAfterAdditionalSelection; + for(auto x : *trackContainer) { + if ( m_trkSelector->decision(*x, nullptr) ) theIDTracksAfterSelection.push_back(x); + } + ATH_MSG_DEBUG("Found good tracks N="<<theIDTracksAfterSelection.size()); + TrackBag theMuonsAfterSelection; + if(m_muonTrackBit.any()) { + const xAOD::MuonContainer* importedMuonCollection(0); + ATH_CHECK(evtStore()->retrieve(importedMuonCollection, "Muons")); + for(auto muon : *importedMuonCollection) { + if(muon->muonType() == xAOD::Muon::SiliconAssociatedForwardMuon) continue; + auto ptr = muon->trackParticle( xAOD::Muon::InnerDetectorTrackParticle ); + if(ptr) theMuonsAfterSelection.push_back(ptr); + } + } + + std::vector<Candidate<2>> Initialcandidates; + { //Isolate scope for safety + const TrackBag &Tracksfor2Vertex = ApplyAdditionalCuts(theIDTracksAfterSelection, theMuonsAfterSelection, theIDTracksAfterAdditionalSelection, 0); + for(auto track1itr = Tracksfor2Vertex.cbegin(); track1itr != Tracksfor2Vertex.cend(); ++track1itr) { + Candidate<2> cand; + std::array<GenVecFourMom_t, 2> vectors; + cand.tracks[0] = *track1itr; + vectors[0].SetCoordinates(cand.tracks[0]->pt(), cand.tracks[0]->eta(), cand.tracks[0]->phi(), m_trackMasses[0]); + for(auto track2itr = track1itr+1; track2itr != Tracksfor2Vertex.cend(); ++track2itr) { + cand.tracks[1] = *track2itr; + if(cand.tracks[0]->qOverP() * cand.tracks[1]->qOverP() >= 0.) continue; //Skip same signed + vectors[1].SetCoordinates(cand.tracks[1]->pt(), cand.tracks[1]->eta(), cand.tracks[1]->phi(), m_trackMasses[1]); + GenVecFourMom_t pair = SumVector(vectors); + if(pair.Pt() < m_ptCutPerVertex[0]) continue; + if(pair.M() >= m_2trackmassMin && pair.M() < m_2trackmassMax) { + ATH_MSG_VERBOSE("2 Track candidate found: " << pair.M() << " Within " << m_2trackmassMin << " and " << m_2trackmassMax); + Initialcandidates.push_back(cand); + } + } + } + } + ATH_MSG_DEBUG("2 Track candidates found: " << Initialcandidates.size()); + if(Initialcandidates.empty()) { + //No work to do Leave method early + return StatusCode::SUCCESS; + } + std::vector<Candidate<3>> Candidates3; + + {//isolate scope + const TrackBag &Tracksfor3Vertex = ApplyAdditionalCuts(theIDTracksAfterSelection, theMuonsAfterSelection, theIDTracksAfterAdditionalSelection, 2); + for(auto &c : Initialcandidates) { + Candidate<3> c3; + std::copy(c.tracks.begin(), c.tracks.end(), c3.tracks.begin()); + std::array<GenVecFourMom_t, 3> vectors; + vectors[0].SetCoordinates(c.tracks[0]->pt(), c.tracks[0]->eta(), c.tracks[0]->phi(), m_trackMasses[0]); + vectors[1].SetCoordinates(c.tracks[1]->pt(), c.tracks[1]->eta(), c.tracks[1]->phi(), m_trackMasses[1]); + for(auto track3itr = Tracksfor3Vertex.cbegin(); track3itr != Tracksfor3Vertex.cend(); ++track3itr) { + if(std::find(c3.tracks.begin(), c3.tracks.end(), *track3itr) != c3.tracks.end()) continue; + c3.tracks[2] = *track3itr; + vectors[2].SetCoordinates(c3.tracks[2]->pt(), c3.tracks[2]->eta(), c3.tracks[2]->phi(), m_trackMasses[2]); + GenVecFourMom_t tripple = SumVector(vectors); + if(tripple.Pt() < m_ptCutPerVertex[1]) continue; + if(tripple.M() >= m_3trackmassMin && tripple.M() < m_3trackmassMax) { + ATH_MSG_VERBOSE("3 Track candidate found: " << tripple.M() << " Within " << m_3trackmassMin << " and " << m_3trackmassMax); + Candidates3.push_back(c3); + } + } + } + } + Initialcandidates.clear(); + Initialcandidates.shrink_to_fit(); + + ATH_MSG_DEBUG("3 Track candidates found: " << Candidates3.size()); + std::map<const std::array<const xAOD::TrackParticle*, 3>, xAOD::Vertex* > threeVertexMap; + + if(!m_3TrackVertexOutput.empty()) { + SG::ReadCondHandle<InDet::BeamSpotData> beamSpotHandle { m_beamSpotKey }; + if(not beamSpotHandle.isValid()) ATH_MSG_ERROR("Cannot Retrieve " << m_beamSpotKey.key() ); + BPhysPVTools helper(&(*m_V0Tools), beamSpotHandle.cptr()); + helper.SetMinNTracksInPV(0); + helper.SetSave3d(false); + std::vector<const xAOD::TrackParticle*> tracksforfit; + std::vector<Candidate<3>> Candidates3PassCuts; + if(m_eliminateBad3Tracksfrom4Track) Candidates3PassCuts.reserve(Candidates3.size()); + for(const auto &c3 : Candidates3) { + tracksforfit.assign(c3.tracks.begin(), c3.tracks.end()); + auto v = StandardFit(tracksforfit, trackContainer); + if(v==nullptr) { + ATH_MSG_DEBUG("3Vertex fit returned null"); + continue; + } + if(m_3TrackChi2NDFCut > 0. && v->chiSquared() / v->numberDoF() > m_3TrackChi2NDFCut) { + ATH_MSG_DEBUG("Rejecting 3 track vertex because Chi " << v->chiSquared() / v->numberDoF() << " > " << m_3TrackChi2NDFCut); + continue; + } + if(m_eliminateBad3Tracksfrom4Track) Candidates3PassCuts.push_back(c3); + threeVertexMap[c3.tracks] = v.get(); + xAOD::BPhysHelper bHelper(v.get());//"get" does not "release" still automatically deleted + bHelper.setRefTrks(); + v3container->push_back(v.release()); + } + + if(v3container->size() >0) ATH_CHECK(helper.FillCandExistingVertices(v3container, pvContainer, 1)); + + if(m_eliminateBad3Tracksfrom4Track) { + ATH_MSG_DEBUG("Swapping container to N = "<< Candidates3PassCuts.size() << " from " << Candidates3.size()); + Candidates3PassCuts.swap(Candidates3);//Swap old container with one that passed cuts + } + + } + std::vector<VertexCand> Candidates4; + {//isolate scope + const TrackBag &Tracksfor4Vertex = ApplyAdditionalCuts(theIDTracksAfterSelection, theMuonsAfterSelection, theIDTracksAfterAdditionalSelection, 3); + for(auto &c : Candidates3) { + VertexCand c4; + std::copy(c.tracks.begin(), c.tracks.end(), c4.tracks.begin()); + std::array<GenVecFourMom_t, 4> vectors; + vectors[0].SetCoordinates(c.tracks[0]->pt(), c.tracks[0]->eta(), c.tracks[0]->phi(), m_trackMasses[0]); + vectors[1].SetCoordinates(c.tracks[1]->pt(), c.tracks[1]->eta(), c.tracks[1]->phi(), m_trackMasses[1]); + vectors[2].SetCoordinates(c.tracks[2]->pt(), c.tracks[2]->eta(), c.tracks[2]->phi(), m_trackMasses[2]); + for(auto track4itr = Tracksfor4Vertex.cbegin(); track4itr != Tracksfor4Vertex.cend(); ++track4itr) { + if(std::find(c4.tracks.begin(), c4.tracks.end(), *track4itr) != c4.tracks.end()) continue; + c4.tracks[3] = *track4itr; + if(c4.tracks[2]->qOverP() * c4.tracks[3]->qOverP() >= 0.) continue; //Skip same signed + vectors[3].SetCoordinates(c4.tracks[3]->pt(), c4.tracks[3]->eta(), c4.tracks[3]->phi(), m_trackMasses[3]); + GenVecFourMom_t fourtrack = SumVector(vectors); + if(fourtrack.Pt() < m_ptCutPerVertex[2]) continue; + if(fourtrack.M() >= m_4trackmassMin && fourtrack.M() < m_4trackmassMax) { + ATH_MSG_VERBOSE("3 Track candidate found: " << fourtrack.M() << " Within " << m_4trackmassMin << " and " << m_4trackmassMax); + Candidates4.push_back(std::move(c4)); + } + } + } + } + Candidates3.clear(); + Candidates3.shrink_to_fit(); + + ATH_MSG_DEBUG("4 Track candidates found: " << Candidates4.size() << " running cascade"); + for(auto &c : Candidates4) { + c.cascVertex = CascadeFit(c.tracks); + if(c.cascVertex!=nullptr) { + c.cascVertex->setSVOwnership(true); + } + } + + SG::AuxElement::Decorator<VertexLinkVector> CascadeLinksDecor("CascadeVertexLinks"); + SG::AuxElement::Decorator<VertexLink> Vertex3Decor(m_3TrackName+ "_VertexLink"); + SG::AuxElement::Decorator<float> chi2_decor("ChiSquared"); + SG::AuxElement::Decorator<float> ndof_decor("NumberDoF"); +// SG::AuxElement::Decorator<float> TotalProb_decor("TotalProb"); + SG::AuxElement::Decorator<float> Pt_decor("Pt"); + SG::AuxElement::Decorator<float> PtErr_decor("PtErr"); + SG::AuxElement::Decorator<float> Mass_svdecor(m_3TrackName+ "_mass"); + SG::AuxElement::Decorator<float> MassErr_svdecor(m_3TrackName+"_massErr"); + SG::AuxElement::Decorator<float> Pt_svdecor(m_3TrackName+"_Pt"); + SG::AuxElement::Decorator<float> PtErr_svdecor(m_3TrackName+"_PtErr"); + SG::AuxElement::Decorator<float> Lxy_svdecor(m_3TrackName+"_Lxy"); + SG::AuxElement::Decorator<float> LxyErr_svdecor(m_3TrackName+"_LxyErr"); + SG::AuxElement::Decorator<float> Tau_svdecor(m_3TrackName+"_Tau"); + SG::AuxElement::Decorator<float> TauErr_svdecor(m_3TrackName+"_TauErr"); + + + SG::ReadCondHandle<InDet::BeamSpotData> beamSpotHandle { m_beamSpotKey }; + if(not beamSpotHandle.isValid()) ATH_MSG_ERROR("Cannot Retrieve " << m_beamSpotKey.key() ); + BPhysPVCascadeTools helper(&(*m_CascadeTools), beamSpotHandle.cptr()); + helper.SetMinNTracksInPV(m_PV_minNTracks); + helper.m_copyAllVertices = this->m_copyAllVertices; + + + + int totalnotnull=0; + for(auto &c : Candidates4) { + if(c.cascVertex==nullptr) { + totalnotnull++; + continue; + } + auto x = c.cascVertex.get(); + const std::vector<xAOD::Vertex*> &cascadeVertices = x->vertices(); + if(cascadeVertices.size()!=s_topoN) { + ATH_MSG_ERROR("Incorrect number of vertices"); + continue; + } + if(cascadeVertices[0] == nullptr || cascadeVertices[1] == nullptr) { + ATH_MSG_ERROR("Error null vertex"); + continue; + } + if( m_Chi2NDFCut > 0.0 && (x->fitChi2() / x->nDoF()) > m_Chi2NDFCut) { + continue; + } + BPhysPVCascadeTools::PrepareVertexLinks(c.cascVertex.get(), trackContainer); + const std::vector< std::vector<TLorentzVector> > &moms = x->getParticleMoms(); + double mass1 = m_CascadeTools->invariantMass(moms[1]); + if(m_4trackmassFinalMin > 0. && mass1 < m_4trackmassFinalMin) continue; + if(m_4trackmassFinalMax > 0. && mass1 > m_4trackmassFinalMax) continue; + double tau = m_CascadeTools->tau(moms[1],cascadeVertices[1],primaryVertex); + if(tau < m_tauCut) continue; +// ATH_MSG_INFO("Total chi " << x->fitChi2()<< " sum chi2 " << cascadeVertices[0]->chiSquared() + cascadeVertices[1]->chiSquared() ); + // Keep vertices (bear in mind that they come in reverse order!) + for(int i =0; i<s_topoN; i++) Vtxwritehandles[i]->push_back(cascadeVertices[i]); + x->setSVOwnership(false); // Prevent Container from deleting vertices + const auto mainVertex = cascadeVertices[1]; // this is the B vertex + + // Set links to cascade vertices + std::vector<const xAOD::Vertex*> verticestoLink; + verticestoLink.push_back(cascadeVertices[0]); + if(!BPhysPVCascadeTools::LinkVertices(CascadeLinksDecor, verticestoLink, Vtxwritehandles[0], cascadeVertices[1])) { + ATH_MSG_ERROR("Error decorating with cascade vertices"); + } + + // loop over candidates -- Don't apply PV_minNTracks requirement here + // because it may result in exclusion of the high-pt PV. + // get good PVs + + xAOD::BPhysHypoHelper vtx(m_hypoName, mainVertex); + + // Get refitted track momenta from all vertices, charged tracks only + BPhysPVCascadeTools::SetVectorInfo(vtx, x); + + // Decorate main vertex + // + // 1.a) mass, mass error + + BPHYS_CHECK( vtx.setMass(m_CascadeTools->invariantMass(moms[1])) ); + BPHYS_CHECK( vtx.setMassErr(m_CascadeTools->invariantMassError(moms[1],x->getCovariance()[1])) ); + // 1.b) pt and pT error (the default pt of mainVertex is != the pt of the full cascade fit!) + Pt_decor(*mainVertex) = m_CascadeTools->pT(moms[1]); + PtErr_decor(*mainVertex) = m_CascadeTools->pTError(moms[1],x->getCovariance()[1]); + // 1.c) chi2 and ndof (the default chi2 of mainVertex is != the chi2 of the full cascade fit!) + chi2_decor(*mainVertex) = x->fitChi2(); + ndof_decor(*mainVertex) = x->nDoF(); + ATH_CHECK(helper.FillCandwithRefittedVertices(m_refitPV, pvContainer, + refPvContainer, &(*m_pvRefitter), m_PV_max, m_DoVertexType, x, 1, m_4tracksMass, vtx)); + // 4) decorate the main vertex with D0 vertex mass, pt, lifetime and lxy values (plus errors) + // D0 points to the main vertex, so lifetime and lxy are w.r.t the main vertex + Mass_svdecor(*mainVertex) = m_CascadeTools->invariantMass(moms[0]); + MassErr_svdecor(*mainVertex) = m_CascadeTools->invariantMassError(moms[0],x->getCovariance()[0]); + Pt_svdecor(*mainVertex) = m_CascadeTools->pT(moms[0]); + PtErr_svdecor(*mainVertex) = m_CascadeTools->pTError(moms[0],x->getCovariance()[0]); + Lxy_svdecor(*mainVertex) = m_CascadeTools->lxy(moms[0],cascadeVertices[0],cascadeVertices[1]); + LxyErr_svdecor(*mainVertex) = m_CascadeTools->lxyError(moms[0],x->getCovariance()[0], cascadeVertices[0],cascadeVertices[1]); + Tau_svdecor(*mainVertex) = m_CascadeTools->tau(moms[0],cascadeVertices[0],cascadeVertices[1]); + TauErr_svdecor(*mainVertex) = m_CascadeTools->tauError(moms[0],x->getCovariance()[0], cascadeVertices[0],cascadeVertices[1]); + if(!threeVertexMap.empty()) { + std::array<const xAOD::TrackParticle*, 3> lookuparray; + std::copy(c.tracks.begin(), c.tracks.begin()+3, lookuparray.begin()); + auto ptr = threeVertexMap[lookuparray]; + if(ptr == nullptr) ATH_MSG_WARNING("3Vertex lookup found null"); + Vertex3Decor(*mainVertex) = ptr ? VertexLink(ptr, *v3container) : VertexLink(); + } + } + + + ATH_MSG_DEBUG("Found " << Vtxwritehandles[0]->size() << " candidates " << totalnotnull << " were null"); + if(Vtxwritehandles[0]->size() > 200) ATH_MSG_WARNING("A lot of candidates N=" << Vtxwritehandles[0]->size()); + return StatusCode::SUCCESS; +} + +std::unique_ptr<Trk::VxCascadeInfo> Cascade3Plus1::CascadeFit(std::array<const xAOD::TrackParticle*, 4> &Track) const { +// ATH_MSG_DEBUG("Running Cascade Fit"); + std::vector<const xAOD::TrackParticle*> tracksDs(Track.begin(), Track.begin()+3); + std::vector<const xAOD::TrackParticle*> tracksBs(1, Track[3]); + + std::vector<double> massesDs(m_trackMasses.begin(), m_trackMasses.begin()+3); + std::vector<double> massesBs(1, m_trackMasses[3]); + + std::unique_ptr<Trk::IVKalState> state = m_iVertexFitter->makeState(); + int robustness = 0; + m_iVertexFitter->setRobustness(robustness, *state); +// if(tracksDs.size() != massesDs.size()) ATH_MSG_ERROR("Track sizes do not match"); +// for(int i =0;i < tracksDs.size();i++) ATH_MSG_DEBUG("Num " << i << " track " << tracksDs[i] << " mass " << massesDs[i]); + // Vertex list + std::vector<Trk::VertexID> vrtList; + // Ds vertex + auto vID = m_iVertexFitter->startVertex(tracksDs, massesDs, *state, m_3TrackMassConstraint ? m_3tracksMass : 0.0); + std::vector<Trk::VertexID> cnstV; + if (m_2TrackMassConstraint && !m_iVertexFitter->addMassConstraint(vID, tracksDs, cnstV, *state, m_2tracksMass).isSuccess() ) { + ATH_MSG_WARNING("addMassConstraint failed"); + } + vrtList.push_back(vID); + // Bs vertex + m_iVertexFitter->nextVertex(tracksBs,massesBs,vrtList, *state); + // Do the work + auto x = std::unique_ptr<Trk::VxCascadeInfo> (m_iVertexFitter->fitCascade(*state)); + if(x==nullptr) ATH_MSG_VERBOSE("Cascade returned null"); + return x; +} + +std::unique_ptr<xAOD::Vertex> Cascade3Plus1::StandardFit(const std::vector<const xAOD::TrackParticle*> &inputTracks, const xAOD::TrackParticleContainer* importedTrackCollection) const { + assert(inputTracks.size() >=2); + const Trk::Perigee& aPerigee1 = inputTracks[0]->perigeeParameters(); + const Trk::Perigee& aPerigee2 = inputTracks[1]->perigeeParameters(); + int sflag = 0; + int errorcode = 0; + Amg::Vector3D startingPoint = m_vertexEstimator->getCirclesIntersectionPoint(&aPerigee1,&aPerigee2,sflag,errorcode); + if (errorcode != 0) { + startingPoint(0) = 0.0; + startingPoint(1) = 0.0; + startingPoint(2) = 0.0; + } + std::unique_ptr<Trk::IVKalState> state = m_iVertexFitter->makeState(); + auto theResult = std::unique_ptr<xAOD::Vertex>( m_iVertexFitter->fit(inputTracks, startingPoint, *state)); + if(theResult) BPhysPVTools::PrepareVertexLinks(theResult.get(), importedTrackCollection); + return theResult; +} + +} diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/CascadeTools.cxx b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/CascadeTools.cxx new file mode 100644 index 00000000000..7721d36d9ee --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/CascadeTools.cxx @@ -0,0 +1,608 @@ +/* + Copyright (C) 2002-2017 CERN for the benefit of the ATLAS collaboration +*/ +///////////////////////////////////////////////////////////////// +// CascadeTools.cxx, (c) ATLAS Detector software +///////////////////////////////////////////////////////////////// +#include "DerivationFrameworkBPhys/CascadeTools.h" +#include "CLHEP/GenericFunctions/CumulativeChiSquare.hh" + + +namespace DerivationFramework { + + +CascadeTools::CascadeTools(const std::string& t, const std::string& n, const IInterface* p) : + AthAlgTool(t,n,p) +{ + declareInterface<CascadeTools>(this); +} + +CascadeTools::~CascadeTools() {} + +//Light speed constant for various calculations +constexpr double s_CONST = 1000./299.792; + +double CascadeTools::invariantMass(const std::vector<TLorentzVector> &particleMom) const +{ + if(particleMom.size() == 0) return -999999.; + TLorentzVector totalMom; + unsigned int NTrk = particleMom.size(); + for( unsigned int it=0; it<NTrk; it++) totalMom += particleMom[it]; + return totalMom.M(); +} + +double CascadeTools::invariantMass(const std::vector<TLorentzVector> &particleMom2, const std::vector<double> &masses) const +{ + if(particleMom2.size() == 0) return -999999.; + TLorentzVector totalMom; + unsigned int NTrk = particleMom2.size(); + if (masses.size() != NTrk) { + ATH_MSG_DEBUG("The provided number of masses does not match the number of tracks in the vertex"); + return -999999.; + } + TLorentzVector temp; + for( unsigned int it=0; it<NTrk; it++) { + double esq = particleMom2[it].Px()*particleMom2[it].Px() + particleMom2[it].Py()*particleMom2[it].Py() + + particleMom2[it].Pz()*particleMom2[it].Pz() + masses[it]*masses[it]; + double e = (esq>0.) ? sqrt(esq) : 0.; + + temp.SetPxPyPzE(particleMom2[it].Px(),particleMom2[it].Py(),particleMom2[it].Pz(),e); + totalMom += temp; + } + return totalMom.M(); +} + +double CascadeTools::invariantMassError(const std::vector<TLorentzVector> &particleMom, const Amg::MatrixX& cov) const +{ + if(particleMom.size() == 0) return -999999.; + unsigned int NTrk = particleMom.size(); + TLorentzVector totalMom; + for( unsigned int it=0; it<NTrk; it++) totalMom += particleMom[it]; + + Amg::MatrixX D_vec(3*NTrk+3,1); D_vec.setZero(); + for( unsigned int it=0; it<NTrk; it++) { + D_vec(3*it+3) = 2.*(totalMom.E()*particleMom[it].Px()/particleMom[it].E()-totalMom.Px()); + D_vec(3*it+4) = 2.*(totalMom.E()*particleMom[it].Py()/particleMom[it].E()-totalMom.Py()); + D_vec(3*it+5) = 2.*(totalMom.E()*particleMom[it].Pz()/particleMom[it].E()-totalMom.Pz()); + } + Amg::MatrixX merr = D_vec.transpose() * cov * D_vec; + double massVarsq = merr(0,0); + if (massVarsq <= 0.) ATH_MSG_DEBUG("massError: negative sqrt massVarsq " << massVarsq); + double massVar = (massVarsq>0.) ? sqrt(massVarsq) : 0.; + double massErr = massVar/(2.*totalMom.M()); + return massErr; +} + +double CascadeTools::invariantMassError(const std::vector<TLorentzVector> &particleMom2, const Amg::MatrixX& cov, const std::vector<double> &masses) const +{ + if(particleMom2.size() == 0) return -999999.; + unsigned int NTrk = particleMom2.size(); + if (masses.size() != NTrk) { + ATH_MSG_DEBUG("The provided number of masses does not match the number of tracks in the vertex"); + return -999999.; + } + std::vector<TLorentzVector> particleMom(NTrk); particleMom.clear(); + for( unsigned int it=0; it<NTrk; it++) { + double esq = particleMom2[it].Px()*particleMom2[it].Px() + particleMom2[it].Py()*particleMom2[it].Py() + + particleMom2[it].Pz()*particleMom2[it].Pz() + masses[it]*masses[it]; + double e = (esq>0.) ? sqrt(esq) : 0.; + particleMom[it].SetPxPyPzE(particleMom2[it].Px(),particleMom2[it].Py(),particleMom2[it].Pz(),e); + } + TLorentzVector totalMom; + for( unsigned int it=0; it<NTrk; it++) totalMom += particleMom[it]; + + std::vector<double>dm2dpx(NTrk), dm2dpy(NTrk), dm2dpz(NTrk); + for( unsigned int it=0; it<NTrk; it++) { + dm2dpx[it] = 2.*(totalMom.E()*particleMom[it].Px()/particleMom[it].E()-totalMom.Px()); + dm2dpy[it] = 2.*(totalMom.E()*particleMom[it].Py()/particleMom[it].E()-totalMom.Py()); + dm2dpz[it] = 2.*(totalMom.E()*particleMom[it].Pz()/particleMom[it].E()-totalMom.Pz()); + } + Amg::MatrixX D_vec(3*NTrk+3,1); D_vec.setZero(); + for( unsigned int it=0; it<NTrk; it++) { + D_vec(3*it+3) = dm2dpx[it]; + D_vec(3*it+4) = dm2dpy[it]; + D_vec(3*it+5) = dm2dpz[it]; + } + Amg::MatrixX merr = D_vec.transpose() * cov * D_vec; + double massVarsq = merr(0,0); + if (massVarsq <= 0.) ATH_MSG_DEBUG("massError: negative sqrt massVarsq " << massVarsq); + double massVar = (massVarsq>0.) ? sqrt(massVarsq) : 0.; + double massErr = massVar/(2.*totalMom.M()); + return massErr; +} + +double CascadeTools::pT(const std::vector<TLorentzVector> &particleMom) const +{ + if(particleMom.size() == 0) return -999999.; + Amg::Vector3D P = momentum(particleMom);; + return P.perp(); +} + +double CascadeTools::pTError(const std::vector<TLorentzVector> &particleMom, const Amg::MatrixX& cov) const +{ + if(particleMom.size() == 0) return -999999.; + Amg::Vector3D P = momentum(particleMom);; + double Px = P.x(); + double Py = P.y(); + double PT = P.perp(); + + unsigned int NTrk = particleMom.size(); + Amg::MatrixX D_vec(3*NTrk+3,1); D_vec.setZero(); + for( unsigned int it=0; it<NTrk; it++) { + D_vec(3*it+3) = Px/PT; + D_vec(3*it+4) = Py/PT; + D_vec(3*it+5) = 0.; + } + Amg::MatrixX PtErrSq = D_vec.transpose() * cov * D_vec; + double PtErrsq = PtErrSq(0,0); + if (PtErrsq <= 0.) ATH_MSG_DEBUG("ptError: negative sqrt PtErrsq " << PtErrsq); + return (PtErrsq>0.) ? sqrt(PtErrsq) : 0.; +} + +double CascadeTools::lxy(const std::vector<TLorentzVector> &particleMom, const xAOD::Vertex* SV, const xAOD::Vertex* PV) const +{ + if(particleMom.size() == 0) return -999999.; + auto vert = SV->position() - PV->position(); + double dx = vert.x(); + double dy = vert.y(); + Amg::Vector3D P = momentum(particleMom);; + double dxy = (P.x()*dx + P.y()*dy)/P.perp(); + return dxy; +} + +double CascadeTools::lxyError(const std::vector<TLorentzVector> &particleMom, const Amg::MatrixX& cov, const xAOD::Vertex* SV, const xAOD::Vertex* PV) const +{ + if(particleMom.size() == 0) return -999999.; + auto vert = SV->position() - PV->position(); + double dx = vert.x(); + double dy = vert.y(); + Amg::Vector3D P = momentum(particleMom);; + double Px = P.x(); + double Py = P.y(); + double PT = P.perp(); + double LXYoverPT = (Px*dx+Py*dy)/(PT*PT); + + unsigned int NTrk = particleMom.size(); + + double dLxydx = Px/PT; + double dLxydy = Py/PT; + double dLxydx0 = -dLxydx; + double dLxydy0 = -dLxydy; + + Amg::MatrixX D_vec(3*NTrk+6,1); D_vec.setZero(); + D_vec(0) = dLxydx; + D_vec(1) = dLxydy; + D_vec(2) = 0.; + for( unsigned int it=0; it<NTrk; it++) { + D_vec(3*it+3) = (dx - LXYoverPT*Px)/PT; + D_vec(3*it+4) = (dy - LXYoverPT*Py)/PT; + D_vec(3*it+5) = 0.; + } + D_vec(3*NTrk+3) = dLxydx0; + D_vec(3*NTrk+4) = dLxydy0; + D_vec(3*NTrk+5) = 0.; + + unsigned int ndim = 3*NTrk+3; + Amg::MatrixX W_mat(3*NTrk+6,3*NTrk+6); W_mat.setZero(); + W_mat.block(0,0,ndim,ndim) = cov; + W_mat.block(3*NTrk+3,3*NTrk+3,3,3) = PV->covariancePosition(); + Amg::MatrixX V_err = D_vec.transpose() * W_mat * D_vec; + + double LxyErrsq = V_err(0,0); + if (LxyErrsq <= 0.) ATH_MSG_DEBUG("lxyError: negative sqrt LxyErrsq " << LxyErrsq); + return (LxyErrsq>0.) ? sqrt(LxyErrsq) : 0.; +} + +double CascadeTools::tau(const std::vector<TLorentzVector> &particleMom, const xAOD::Vertex* SV, const xAOD::Vertex* PV) const +{ + if(particleMom.size() == 0) return -999999.; + double M = invariantMass(particleMom); + double LXY = lxy(particleMom,SV,PV); + double PT = pT(particleMom); + return s_CONST*M*LXY/PT; +} + +double CascadeTools::tauError(const std::vector<TLorentzVector> &particleMom, const Amg::MatrixX& cov, const xAOD::Vertex* SV, const xAOD::Vertex* PV) const +{ + if(particleMom.size() == 0) return -999999.; + double M = invariantMass(particleMom); + auto vert = SV->position() - PV->position(); + double dx = vert.x(); + double dy = vert.y(); + Amg::Vector3D P = momentum(particleMom);; + double Px = P.x(); + double Py = P.y(); + double PT = P.perp(); + double LXY = Px*dx+Py*dy; + + unsigned int NTrk = particleMom.size(); + TLorentzVector totalMom; + for( unsigned int it=0; it<NTrk; it++) totalMom += particleMom[it]; + + double dTaudx = (M*Px)/(PT*PT); + double dTaudy = (M*Py)/(PT*PT); + double dTaudx0 = -dTaudx; + double dTaudy0 = -dTaudy; + + Amg::MatrixX D_vec(3*NTrk+6,1); D_vec.setZero(); + D_vec(0) = dTaudx; + D_vec(1) = dTaudy; + D_vec(2) = 0.; + for( unsigned int it=0; it<NTrk; it++) { + D_vec(3*it+3) = (((totalMom.E()*particleMom[it].Px()*LXY)/(M*particleMom[it].E()))-Px*LXY/M+M*dx)/(PT*PT) - 2.*M*LXY*Px/(PT*PT*PT*PT); + D_vec(3*it+4) = (((totalMom.E()*particleMom[it].Py()*LXY)/(M*particleMom[it].E()))-Py*LXY/M+M*dy)/(PT*PT) - 2.*M*LXY*Py/(PT*PT*PT*PT); + D_vec(3*it+5) = 0.; + } + D_vec(3*NTrk+3) = dTaudx0; + D_vec(3*NTrk+4) = dTaudy0; + D_vec(3*NTrk+5) = 0.; + + unsigned int ndim = 3*NTrk+3; + Amg::MatrixX W_mat(3*NTrk+6,3*NTrk+6); W_mat.setZero(); + W_mat.block(0,0,ndim,ndim) = cov; + W_mat.block(3*NTrk+3,3*NTrk+3,3,3) = PV->covariancePosition(); + Amg::MatrixX V_err = D_vec.transpose() * W_mat * D_vec; + + double tauErrsq = V_err(0,0); + if (tauErrsq <= 0.) ATH_MSG_DEBUG("tauError: negative sqrt tauErrsq " << tauErrsq); + double tauErr = (tauErrsq>0.) ? sqrt(tauErrsq) : 0.; + return s_CONST*tauErr; +} + +double CascadeTools::tau(const std::vector<TLorentzVector> &particleMom, const xAOD::Vertex* SV, const xAOD::Vertex* PV, double M) const +{ + if(particleMom.size() == 0) return -999999.; + double LXY = lxy(particleMom,SV,PV); + double PT = pT(particleMom); + return s_CONST*M*LXY/PT; +} + +double CascadeTools::tauError(const std::vector<TLorentzVector> &particleMom, const Amg::MatrixX& cov, const xAOD::Vertex* SV, const xAOD::Vertex* PV, double M) const +{ + if(particleMom.size() == 0) return -999999.; + auto vert = SV->position() - PV->position(); + double dx = vert.x(); + double dy = vert.y(); + Amg::Vector3D P = momentum(particleMom);; + double Px = P.x(); + double Py = P.y(); + double PT = P.perp(); + double LXY = Px*dx+Py*dy; + + unsigned int NTrk = particleMom.size(); + TLorentzVector totalMom; + for( unsigned int it=0; it<NTrk; it++) totalMom += particleMom[it]; + + double dTaudx = (M*Px)/(PT*PT); + double dTaudy = (M*Py)/(PT*PT); + double dTaudx0 = -dTaudx; + double dTaudy0 = -dTaudy; + + Amg::MatrixX D_vec(3*NTrk+6,1); D_vec.setZero(); + D_vec(0) = dTaudx; + D_vec(1) = dTaudy; + D_vec(2) = 0.; + for( unsigned int it=0; it<NTrk; it++) { + D_vec(3*it+3) = (M*dx)/(PT*PT) - 2.*M*LXY*Px/(PT*PT*PT*PT); + D_vec(3*it+4) = (M*dy)/(PT*PT) - 2.*M*LXY*Py/(PT*PT*PT*PT); + D_vec(3*it+5) = 0.; + } + D_vec(3*NTrk+3) = dTaudx0; + D_vec(3*NTrk+4) = dTaudy0; + D_vec(3*NTrk+5) = 0.; + + unsigned int ndim = 3*NTrk+3; + Amg::MatrixX W_mat(3*NTrk+6,3*NTrk+6); W_mat.setZero(); + W_mat.block(0,0,ndim,ndim) = cov; + W_mat.block(3*NTrk+3,3*NTrk+3,3,3) = PV->covariancePosition(); + Amg::MatrixX V_err = D_vec.transpose() * W_mat * D_vec; + + double tauErrsq = V_err(0,0); + if (tauErrsq <= 0.) ATH_MSG_DEBUG("tauError: negative sqrt tauErrsq " << tauErrsq); + double tauErr = (tauErrsq>0.) ? sqrt(tauErrsq) : 0.; + return s_CONST*tauErr; +} + +Amg::Vector3D CascadeTools::pca(const std::vector<TLorentzVector> &particleMom, const xAOD::Vertex* SV, const xAOD::Vertex* PV) const +{ + if(particleMom.size() == 0) { + Amg::Vector3D p; p.setZero(); + return p; + } + Amg::Vector3D pv = PV->position(); + Amg::Vector3D sv = SV->position(); + Amg::Vector3D P = momentum(particleMom);; + double p2 = P.mag2(); + double pdr = P.dot((sv - pv)); + return sv - P*pdr/p2; +} + +double CascadeTools::cosTheta(const std::vector<TLorentzVector> &particleMom, const xAOD::Vertex* SV, const xAOD::Vertex* PV) const +{ + if(particleMom.size() == 0) return -999999.; + Amg::Vector3D P = momentum(particleMom);; + Amg::Vector3D vtx = SV->position(); + vtx -= PV->position(); + return (P.dot(vtx))/(P.mag()*vtx.mag()); +} + +double CascadeTools::cosTheta_xy(const std::vector<TLorentzVector> &particleMom, const xAOD::Vertex* SV, const xAOD::Vertex* PV) const +{ + if(particleMom.size() == 0) return -999999.; + Amg::Vector3D P = momentum(particleMom);; + Amg::Vector3D vtx = SV->position(); + vtx -= PV->position(); + double pT = P.perp(); + return (P.x()*vtx.x()+P.y()*vtx.y())/(pT*vtx.perp()); +} + +double CascadeTools::a0z(const std::vector<TLorentzVector> &particleMom, const xAOD::Vertex* SV, const xAOD::Vertex* PV) const +{ + if(particleMom.size() == 0) return -999999.; + Amg::Vector3D pv = PV->position(); + Amg::Vector3D ca_point = pca(particleMom,SV,PV); + Amg::Vector3D a0_vec = pv - ca_point; + return a0_vec.z(); +} + +double CascadeTools::a0zError(const std::vector<TLorentzVector> &particleMom, const Amg::MatrixX& cov, const xAOD::Vertex* SV, const xAOD::Vertex* PV) const +{ + if(particleMom.size() == 0) return -999999.; + auto vert = SV->position() - PV->position(); + double dx = vert.x(); + double dy = vert.y(); + double dz = vert.z(); + Amg::Vector3D P = momentum(particleMom);; + double Px = P.x(); + double Py = P.y(); + double Pz = P.z(); + double P2 = P.mag2(); + double L = Px*dx+Py*dy+Pz*dz; + + unsigned int NTrk = particleMom.size(); + + + double da0zdx = (Px*Pz)/P2; + double da0zdy = (Py*Pz)/P2; + double da0zdz = (Pz*Pz)/P2 - 1.; + double da0zdx0 = -da0zdx; + double da0zdy0 = -da0zdy; + double da0zdz0 = -da0zdz; + + Amg::MatrixX D_vec(3*NTrk+6,1); D_vec.setZero(); + D_vec(0) = da0zdx; + D_vec(1) = da0zdy; + D_vec(2) = da0zdz; + for( unsigned int it=0; it<NTrk; it++) { + D_vec(3*it+3) = (Pz*(P2*dx-2.*L*Px))/(P2*P2); + D_vec(3*it+4) = (Pz*(P2*dy-2.*L*Py))/(P2*P2); + D_vec(3*it+5) = (Pz*(P2*dz-2.*L*Pz))/(P2*P2)+L/P2; + } + D_vec(3*NTrk+3) = da0zdx0; + D_vec(3*NTrk+4) = da0zdy0; + D_vec(3*NTrk+5) = da0zdz0; + + unsigned int ndim = 3*NTrk+3; + Amg::MatrixX W_mat(3*NTrk+6,3*NTrk+6); W_mat.setZero(); + W_mat.block(0,0,ndim,ndim) = cov; + W_mat.block(3*NTrk+3,3*NTrk+3,3,3) = PV->covariancePosition(); + Amg::MatrixX V_err = D_vec.transpose() * W_mat * D_vec; + + double a0zErrsq = V_err(0,0); + if (a0zErrsq <= 0.) ATH_MSG_DEBUG("a0zError: negative sqrt a0zErrsq " << a0zErrsq); + return (a0zErrsq>0.) ? sqrt(a0zErrsq) : 0.; +} + +double CascadeTools::a0xy(const std::vector<TLorentzVector> &particleMom, const xAOD::Vertex* SV, const xAOD::Vertex* PV) const +{ + if(particleMom.size() == 0) return -999999.; + double cosineTheta_xy = cosTheta_xy(particleMom,SV,PV); + double sinTheta_xy = ((1.-cosineTheta_xy*cosineTheta_xy)>0.) ? sqrt((1.-cosineTheta_xy*cosineTheta_xy)) : 0.; + return (SV->position()-PV->position()).perp() * sinTheta_xy; +} + +double CascadeTools::a0xyError(const std::vector<TLorentzVector> &particleMom, const Amg::MatrixX& cov, const xAOD::Vertex* SV, const xAOD::Vertex* PV) const +{ + if(particleMom.size() == 0) return -999999.; + auto vert = SV->position() - PV->position(); + double dx = vert.x(); + double dy = vert.y(); + Amg::Vector3D P = momentum(particleMom);; + double Px = P.x(); + double Py = P.y(); + double P2 = P.perp()*P.perp(); + double L = Px*dx+Py*dy; + double dR2 = vert.perp()*vert.perp(); + double d = sqrt((P2*dR2-L*L)/P2); + + unsigned int NTrk = particleMom.size(); + + double da0dx = (P2*dx-L*Px)/(P2*d); + double da0dy = (P2*dy-L*Py)/(P2*d); + double da0dx0 = -da0dx; + double da0dy0 = -da0dy; + + Amg::MatrixX D_vec(3*NTrk+6,1); D_vec.setZero(); + D_vec(0) = da0dx; + D_vec(1) = da0dy; + D_vec(2) = 0.; + for( unsigned int it=0; it<NTrk; it++) { + D_vec(3*it+3) = (L*(L*Px-P2*dx))/(P2*P2*d); + D_vec(3*it+4) = (L*(L*Py-P2*dy))/(P2*P2*d); + D_vec(3*it+5) = 0.; + } + D_vec(3*NTrk+3) = da0dx0; + D_vec(3*NTrk+4) = da0dy0; + D_vec(3*NTrk+5) = 0.; + + unsigned int ndim = 3*NTrk+3; + Amg::MatrixX W_mat(3*NTrk+6,3*NTrk+6); W_mat.setZero(); + W_mat.block(0,0,ndim,ndim) = cov; + W_mat.block(3*NTrk+3,3*NTrk+3,3,3) = PV->covariancePosition(); + Amg::MatrixX V_err = D_vec.transpose() * W_mat * D_vec; + + double a0xyErrsq = V_err(0,0); + if (a0xyErrsq <= 0.) ATH_MSG_DEBUG("a0xyError: negative sqrt a0xyErrsq " << a0xyErrsq); + return (a0xyErrsq>0.) ? sqrt(a0xyErrsq) : 0.; +} + +double CascadeTools::a0(const std::vector<TLorentzVector> &particleMom, const xAOD::Vertex* SV, const xAOD::Vertex* PV) const +{ + if(particleMom.size() == 0) return -999999.; + double cosineTheta = cosTheta(particleMom,SV,PV); + double sinTheta = ((1.-cosineTheta*cosineTheta)>0.) ? sqrt((1.-cosineTheta*cosineTheta)) : 0.; + return (SV->position()-PV->position()).mag() * sinTheta; +} + +double CascadeTools::a0Error(const std::vector<TLorentzVector> &particleMom, const Amg::MatrixX& cov, const xAOD::Vertex* SV, const xAOD::Vertex* PV) const +{ + if(particleMom.size() == 0) return -999999.; + auto vert = SV->position() - PV->position(); + double dx = vert.x(); + double dy = vert.y(); + double dz = vert.z(); + Amg::Vector3D P = momentum(particleMom);; + double Px = P.x(); + double Py = P.y(); + double Pz = P.z(); + double P2 = P.mag2(); + double L = Px*dx+Py*dy+Pz*dz; + double dR2 = vert.mag2(); + double d = sqrt((P2*dR2-L*L)/P2); + + unsigned int NTrk = particleMom.size(); + + double da0dx = (P2*dx-L*Px)/(P2*d); + double da0dy = (P2*dy-L*Py)/(P2*d); + double da0dz = (P2*dz-L*Pz)/(P2*d); + double da0dx0 = -da0dx; + double da0dy0 = -da0dy; + double da0dz0 = -da0dz; + + Amg::MatrixX D_vec(3*NTrk+6,1); D_vec.setZero(); + D_vec(0) = da0dx; + D_vec(1) = da0dy; + D_vec(2) = da0dz; + for( unsigned int it=0; it<NTrk; it++) { + D_vec(3*it+3) = (L*(L*Px-P2*dx))/(P2*P2*d); + D_vec(3*it+4) = (L*(L*Py-P2*dy))/(P2*P2*d); + D_vec(3*it+5) = (L*(L*Pz-P2*dz))/(P2*P2*d); + } + D_vec(3*NTrk+3) = da0dx0; + D_vec(3*NTrk+4) = da0dy0; + D_vec(3*NTrk+5) = da0dz0; + + unsigned int ndim = 3*NTrk+3; + Amg::MatrixX W_mat(3*NTrk+6,3*NTrk+6); W_mat.setZero(); + W_mat.block(0,0,ndim,ndim) = cov; + W_mat.block(3*NTrk+3,3*NTrk+3,3,3) = PV->covariancePosition(); + Amg::MatrixX V_err = D_vec.transpose() * W_mat * D_vec; + + double a0Errsq = V_err(0,0); + if (a0Errsq <= 0.) ATH_MSG_DEBUG("a0Error: negative sqrt a0Errsq " << a0Errsq); + return (a0Errsq>0.) ? sqrt(a0Errsq) : 0.; +} + +Amg::Vector3D CascadeTools::momentum(const std::vector<TLorentzVector> &particleMom) const +{ + if(particleMom.size() == 0) { + Amg::Vector3D p; p.setZero(); + return p; + } + TLorentzVector totalMom; + unsigned int NTrk = particleMom.size(); + for( unsigned int it=0; it<NTrk; it++) totalMom += particleMom[it]; + TVector3 P3 = totalMom.Vect(); + Amg::Vector3D mom(P3.Px(),P3.Py(),P3.Pz()); + return mom; +} + +double CascadeTools::massProbability(double V0Mass, double mass, double massErr) const +{ + if(massErr > 0.) { + double chi2 = (V0Mass - mass)*(V0Mass - mass)/(massErr*massErr); + int ndf = 1; + Genfun::CumulativeChiSquare myCumulativeChiSquare(ndf); + if (chi2 > 0.) { + double achi2prob = 1.-myCumulativeChiSquare(chi2); + return achi2prob; + } else { + ATH_MSG_DEBUG("chi2 <= 0"); + return -1.; + } + } else { + return -1.; + } +} + +double CascadeTools::vertexProbability(int ndf, double chi2) const +{ + if (ndf > 0.) { + Genfun::CumulativeChiSquare myCumulativeChiSquare(ndf); + if (chi2 > 0.) { + double chi2prob = 1.-myCumulativeChiSquare(chi2); + return chi2prob; + } else { + ATH_MSG_DEBUG("chi2 <= 0"); + return -1.; + } + } else { + ATH_MSG_DEBUG("ndf <= 0"); + return -1.; + } +} + + +Amg::MatrixX * CascadeTools::convertCovMatrix(const xAOD::Vertex * vxCandidate) const +{ + unsigned int NTrk = vxCandidate->nTrackParticles(); + std::vector<float> matrix = vxCandidate->covariance(); + + int ndim = 0; + + if ( matrix.size() == (3*NTrk+3)*(3*NTrk+3+1)/2) { + ndim = 3*NTrk+3; + } else if (matrix.size() == (5*NTrk+3)*(5*NTrk+3+1)/2) { + ndim = 5*NTrk+3; + } else { + return nullptr; + } + + Amg::MatrixX* mtx = new Amg::MatrixX(ndim,ndim); + + long int ij=0; + for (int i=1; i<= ndim; i++) { + for (int j=1; j<=i; j++){ + if (i==j) { + (*mtx)(i-1,j-1)=matrix[ij]; + } else { + (*mtx).fillSymmetric(i-1,j-1,matrix[ij]); + } + ij++; + } + } + // NOTE: mtx is a pointer! Take care of deleting it after you do not need it anymore!!! + + return mtx; +} + +Amg::MatrixX CascadeTools::SetFullMatrix(int NTrk, const std::vector<float> & Matrix) const +{ + + Amg::MatrixX mtx(3+3*NTrk,3+3*NTrk); // Create identity matrix of needed size + + int ij=0; + + for(int i=0; i<(3+3*NTrk); i++){ + for(int j=0; j<=i; j++){ + mtx(i,j)=Matrix[ij]; + if(i!=j) mtx(j,i)=Matrix[ij]; + ij++; + } + } + + return mtx; +} + +} //end of namespace definitions + diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/CfAthAlgTool.cxx b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/CfAthAlgTool.cxx new file mode 100644 index 00000000000..299340b7b3a --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/CfAthAlgTool.cxx @@ -0,0 +1,178 @@ +/* + Copyright (C) 2002-2017 CERN for the benefit of the ATLAS collaboration +*/ + +//============================================================================ +// CfAthAlgTool.h +//============================================================================ +// +// Author : Wolfgang Walkowiak <Wolfgang.Walkowiak@cern.ch.> +// Changes: +// +// Wrapper around AthAlgTool to provide easy access to CutFlowSvc +// and some utility methods for it. +// Methods for accessing the CutFlowSvc are modelled after +// AthFilterAlgorithm's implementation. +// +// This class inherits from AthAlgTool. It should be inherited from. +// +//----------------------------------------------------------------------------- +// +// Usage information +// +// Example: +// +// // Bmumu_reco_mumu.h: +// class Bmumu_reco_mumu : public CfAthAlgTool, public IAugmentationTool { +// public: +// Bmumu_reco_mumu(const std::string& t, const std::string& n, +// const IInterface* p); +// ... +// +// // Bmumu_reco_mumu.cxx: +// Bmumu_reco_mumu::Bmumu_reco_mumu(const std::string& t, +// const std::string& n, +// const IInterface* p) : +// CfAthAlgTool(t,n,p), +// ... +// +// // inside a method like Bmumu_reco_mumu::addBranches(): +// ... +// // add counter for number of events seen +// addEvent("dimuEvents"); +// // add counter for the number of events with >= 1 reco'd vertices +// if ( vtxContainer->size() > 0 ) { +// addEvent("dimuWithVertexCand"); +// } +// // add counter for the number of vertices +// addToCounter("dimuNumVertices", vtxContainer->size()); +// ... +// +// Please note that a line for +// addEvent(nameString, weight=1.); +// or +// addToCounter(nameString, counts=1, weight=1.); +// is sufficient. +// In a case a counter with that name does not exist yet, it will be +// initialized automatically. +// +//============================================================================ +// +#include "DerivationFrameworkBPhys/CfAthAlgTool.h" + +namespace DerivationFramework { + + //-------------------------------------------------------------------------- + // Constructor + CfAthAlgTool::CfAthAlgTool(const std::string& t, + const std::string& n, + const IInterface* p) : + AthAlgTool(t,n,p), + m_cutFlowSvc("CutFlowSvc/CutFlowSvc", n), + m_ctbasename(n), + m_bid(0), m_bidisset(false) { + + ATH_MSG_DEBUG("Calling constructor with parameters"); + + // Declare counters base name + declareProperty("CountersBaseName", m_ctbasename); + + // clean-up counter base name + std::string fstr("ToolSvc."); + std::string::size_type ind = m_ctbasename.find(fstr); + if (ind != std::string::npos) m_ctbasename.erase(ind, fstr.length()); + + } + //-------------------------------------------------------------------------- + // Destructor + CfAthAlgTool::~CfAthAlgTool() { + + ATH_MSG_DEBUG("Calling destructor"); + } + //-------------------------------------------------------------------------- + // return a handle to an ICutFlowSvc instance + ServiceHandle<ICutFlowSvc>& CfAthAlgTool::cutFlowSvc() const { + + return m_cutFlowSvc; + } + //-------------------------------------------------------------------------- + // Initialization method invoked by the framework. + StatusCode CfAthAlgTool::sysInitialize() { + + // retrieve CutFlowSvc instance + CHECK( cutFlowSvc().retrieve() ); + + // re-direct to base class... + return AthAlgTool::sysInitialize(); + } + //-------------------------------------------------------------------------- + // add one event to a named counter -- returns true on success + bool CfAthAlgTool::addEvent(const std::string &name, double weight) const { + + CutIdentifier id = getCounter(name); + if ( id > 0 ) { + cutFlowSvc()->addEvent(id, weight); + } + return (id > 0); + } + //-------------------------------------------------------------------------- + // add to a named counter -- returns true on success + // if counts > 1 : same weight is added multiple times + bool CfAthAlgTool::addToCounter(const std::string &name, uint64_t counts, + double weight) const { + + CutIdentifier id = getCounter(name); + if ( id > 0 ) { + for (uint64_t i=0; i<counts; ++i) { + cutFlowSvc()->addEvent(id, weight); + } + } + return (id > 0); + } + //-------------------------------------------------------------------------- + // add a counter by name -- simply returns id if counter already exists + CutIdentifier CfAthAlgTool::getCounter(const std::string &name) const { + + CutIdentifier id = getCounterIdByName(name); + if ( id < 1 ) { + std::string fullname = m_ctbasename + "_" + name; + if ( ! m_bidisset ) { + throw std::runtime_error("cutFlowSvc()->registerFilter is no longer supported. code an alternative here"); + //id = cutFlowSvc()->registerFilter(fullname, "N/A"); + m_bid = id; + } else { + throw std::runtime_error("cutFlowSvc()->registerCut is no longer supported. code an alternative here"); + //id = cutFlowSvc()->registerCut(fullname, "N/A", m_bid); + } + m_mctn[name] = id; + } + return id; + } + //-------------------------------------------------------------------------- + // returns counter name by id + std::string CfAthAlgTool::getCounterNameById(CutIdentifier id) const { + + std::string res = "__NOT_FOUND__"; + + for (NameIdMap_t::iterator it = m_mctn.begin(); it != m_mctn.end(); ++it) { + if ( it->second == id ) { + res = it->first; + break; + } + } + return res; + } + //-------------------------------------------------------------------------- + // returns counter id by name + CutIdentifier CfAthAlgTool::getCounterIdByName(const std::string &name) const { + + CutIdentifier id = 0; + + NameIdMap_t::const_iterator it = m_mctn.find(name); + if ( it != m_mctn.end() ) { + id = it->second; + } + return id; + } + //-------------------------------------------------------------------------- +} // namespace diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/FourMuonTool.cxx b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/FourMuonTool.cxx new file mode 100644 index 00000000000..bdf1acabf9e --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/FourMuonTool.cxx @@ -0,0 +1,449 @@ +/* + Copyright (C) 2002-2017 CERN for the benefit of the ATLAS collaboration +*/ + +// **************************************************************************** +// ---------------------------------------------------------------------------- +// FourMuonTool +// James Catmore <James.Catmore@cern.ch> +// Evelina Bouhova-Thacker <e.bouhova@cern.ch> +// ---------------------------------------------------------------------------- +// **************************************************************************** + +#include "DerivationFrameworkBPhys/FourMuonTool.h" +#include "DerivationFrameworkBPhys/BPhysPVTools.h" +#include "xAODBPhys/BPhysHelper.h" +#include "TrkVertexAnalysisUtils/V0Tools.h" +#include "TrkVertexFitterInterfaces/IVertexFitter.h" +#include "TrkVKalVrtFitter/TrkVKalVrtFitter.h" +#include "TrkV0Fitter/TrkV0VertexFitter.h" +#include "InDetConversionFinderTools/ConversionFinderUtils.h" +#include "InDetConversionFinderTools/VertexPointEstimator.h" +#include "TrkToolInterfaces/ITrackSelectorTool.h" +#include "AthLinks/ElementLink.h" +#include "BeamSpotConditionsData/BeamSpotData.h" + +#include "xAODTracking/Vertex.h" +#include "xAODTracking/VertexContainer.h" +#include "xAODTracking/VertexAuxContainer.h" +#include "xAODTracking/TrackParticle.h" +#include "xAODTracking/TrackParticleContainer.h" +#include "StoreGate/WriteDecorHandle.h" + +#include <algorithm> + +namespace DerivationFramework { + + StatusCode FourMuonTool::initialize() { + + // retrieving vertex Fitter + if ( m_iVertexFitter.retrieve().isFailure() ) { + ATH_MSG_FATAL("Failed to retrieve tool " << m_iVertexFitter); + return StatusCode::FAILURE; + } else { + ATH_MSG_DEBUG("Retrieved tool " << m_iVertexFitter); + } + + // retrieving V0 Fitter + if ( m_iV0VertexFitter.retrieve().isFailure() ) { + ATH_MSG_FATAL("Failed to retrieve tool " << m_iV0VertexFitter); + return StatusCode::FAILURE; + } else { + ATH_MSG_DEBUG("Retrieved tool " << m_iV0VertexFitter); + } + + // Get the track selector tool from ToolSvc + if ( m_trkSelector.retrieve().isFailure() ) { + ATH_MSG_FATAL("Failed to retrieve tool " << m_trkSelector); + return StatusCode::FAILURE; + } else { + ATH_MSG_DEBUG("Retrieved tool " << m_trkSelector); + } + + // uploading the V0 tools + if ( m_V0Tools.retrieve().isFailure() ) { + ATH_MSG_FATAL("Failed to retrieve tool " << m_V0Tools); + return StatusCode::FAILURE; + } else { + ATH_MSG_DEBUG("Retrieved tool " << m_V0Tools); + } + + // Get the vertex point estimator tool from ToolSvc + if ( m_vertexEstimator.retrieve().isFailure() ) { + ATH_MSG_FATAL("Failed to retrieve tool " << m_vertexEstimator); + return StatusCode::FAILURE; + } else { + ATH_MSG_DEBUG("Retrieved tool " << m_vertexEstimator); + } + + // Get the beam spot service + CHECK( m_beamSpotKey.initialize() ); + + m_muonIndex = m_muonCollectionKey.key() + ".BPHY4MuonIndex"; + ATH_CHECK(m_muonIndex.initialize()); + ATH_MSG_DEBUG("Initialize successful"); + + return StatusCode::SUCCESS; + + } + + FourMuonTool::FourMuonTool(const std::string& t, const std::string& n, const IInterface* p) : AthAlgTool(t,n,p), + m_ptCut(0.0), + m_etaCut(0.0), + m_useV0Fitter(false), + m_muonCollectionKey("Muons"), + m_TrkParticleCollection("TrackParticleCandidate"), + m_iVertexFitter("Trk::TrkVKalVrtFitter"), + m_iV0VertexFitter("Trk::V0VertexFitter"), + m_V0Tools("Trk::V0Tools"), + m_trkSelector("InDet::TrackSelectorTool"), + m_vertexEstimator("InDet::VertexPointEstimator") + { + declareInterface<FourMuonTool>(this); + declareProperty("ptCut",m_ptCut); + declareProperty("etaCut",m_etaCut); + declareProperty("useV0Fitter",m_useV0Fitter); + declareProperty("muonCollectionKey",m_muonCollectionKey); + declareProperty("TrackParticleCollection",m_TrkParticleCollection); + declareProperty("TrkVertexFitterTool",m_iVertexFitter); + declareProperty("V0VertexFitterTool",m_iV0VertexFitter); + declareProperty("V0Tools",m_V0Tools); + declareProperty("TrackSelectorTool",m_trkSelector); + declareProperty("VertexPointEstimator",m_vertexEstimator); + } + + FourMuonTool::~FourMuonTool() { } + + //------------------------------------------------------------------------------------- + // Find the candidates + //------------------------------------------------------------------------------------- + StatusCode FourMuonTool::performSearch(xAOD::VertexContainer*& pairVxContainer, xAOD::VertexAuxContainer*& pairVxAuxContainer, + xAOD::VertexContainer*& quadVxContainer, xAOD::VertexAuxContainer*& quadVxAuxContainer, bool &selectEvent) const + { + ATH_MSG_DEBUG( "FourMuonTool::performSearch" ); + selectEvent = false; + + // pairs + pairVxContainer = new xAOD::VertexContainer; + pairVxAuxContainer = new xAOD::VertexAuxContainer; + pairVxContainer->setStore(pairVxAuxContainer); + // quads + quadVxContainer = new xAOD::VertexContainer; + quadVxAuxContainer = new xAOD::VertexAuxContainer; + quadVxContainer->setStore(quadVxAuxContainer); + + + // Get the muons from StoreGate + SG::ReadHandle<xAOD::MuonContainer> importedMuonCollection(m_muonCollectionKey); + ATH_CHECK(importedMuonCollection.isValid()); + ATH_MSG_DEBUG("Muon container size "<<importedMuonCollection->size()); + + // Get ID tracks + SG::ReadHandle<xAOD::TrackParticleContainer> importedTrackCollection(m_TrkParticleCollection); + ATH_CHECK(importedTrackCollection.isValid()); + ATH_MSG_DEBUG("ID TrackParticle container size "<< importedTrackCollection->size()); + + // Select the muons + std::vector<const xAOD::Muon*> theMuonsAfterSelection; + SG::WriteDecorHandle<xAOD::MuonContainer, int> muonDecorator(m_muonIndex); + unsigned int nCombMuons = 0; + unsigned int nSegmentTaggedMuons = 0; + + for (auto muItr=importedMuonCollection->begin(); muItr!=importedMuonCollection->end(); ++muItr) { + if ( *muItr == NULL ) continue; + muonDecorator(**muItr) = -1; // all muons must be decorated + if ( ((*muItr)->muonType() != xAOD::Muon::Combined ) && ((*muItr)->muonType() != xAOD::Muon::SegmentTagged ) ) continue; + if (!(*muItr)->inDetTrackParticleLink().isValid()) continue; // No muons without ID tracks + auto& link = (*muItr)->inDetTrackParticleLink(); + const xAOD::TrackParticle* muonTrk = *link; + if ( muonTrk==NULL) continue; + const xAOD::Vertex* vx = nullptr; + if ( !m_trkSelector->decision(*muonTrk, vx) ) continue; // all ID tracks must pass basic tracking cuts + if ( fabs(muonTrk->pt())<m_ptCut ) continue; // pt cut + if ( fabs(muonTrk->eta())>m_etaCut ) continue; // eta cut + if ( (*muItr)->muonType() == xAOD::Muon::Combined ) ++nCombMuons; + if ( (*muItr)->muonType() == xAOD::Muon::SegmentTagged ) ++nSegmentTaggedMuons; + theMuonsAfterSelection.push_back(*muItr); + } + unsigned int nSelectedMuons = theMuonsAfterSelection.size(); + ATH_MSG_DEBUG("Number of muons after selection: " << nSelectedMuons); + ATH_MSG_DEBUG("of which " << nCombMuons << " are combined"); + ATH_MSG_DEBUG("and " << nSegmentTaggedMuons << " are segment tagged"); + if ( (nSelectedMuons < 4) || (nCombMuons < 1) ) { + ATH_MSG_DEBUG("Muon criteria not met. Skipping event."); + return StatusCode::SUCCESS; + } + selectEvent = true; // if we got this far we should definitively accept the event + + // Decorators + SG::AuxElement::Decorator< std::string > indexDecorator("CombinationCode"); + SG::AuxElement::Decorator< std::string > chargeDecorator("ChargeCode"); + //SG::AuxElement::Decorator< double > acdcDecorator("ACminusDC"); + //SG::AuxElement::Decorator< double > ssdcDecorator("SSminusDC"); + + // Order by pT + std::sort(theMuonsAfterSelection.begin(), theMuonsAfterSelection.end(), [](const xAOD::Muon *a, const xAOD::Muon *b) { + return b->pt() < a->pt(); + }); + + // Decorate the selected muons (now pT ordered) with their index + unsigned int muonIndex(0); + for (auto selMuon : theMuonsAfterSelection) { + muonDecorator(*selMuon) = muonIndex; + ++muonIndex; + } + + // Quadruplet combinatorics + std::vector<Combination> quadruplets; + std::vector<Combination> pairs; + buildCombinations(theMuonsAfterSelection,pairs,quadruplets,nSelectedMuons); + if (quadruplets.size()==0) { + ATH_MSG_DEBUG("No acceptable quadruplets"); + return StatusCode::SUCCESS; + } + + // Get the beam spot (for the vertexing starting point) + SG::ReadCondHandle<InDet::BeamSpotData> beamSpotHandle { m_beamSpotKey }; + if(not beamSpotHandle.isValid()) ATH_MSG_ERROR("Cannot Retrieve " << m_beamSpotKey.key() ); + const Amg::Vector3D &beamSpot = beamSpotHandle->beamPos(); + + // fit pairs + ATH_MSG_DEBUG("Successful pairs....."); + for (std::vector<Combination>::iterator pairItr = pairs.begin(); pairItr!=pairs.end(); ++pairItr) { + std::vector<const xAOD::TrackParticle*> theTracks = (*pairItr).trackParticles("pair1"); + xAOD::Vertex* pairVxCandidate = fit(theTracks,importedTrackCollection.get(),beamSpot); // This line actually does the fitting and object making + if (pairVxCandidate != 0) { + // decorate the candidate with its codes + indexDecorator(*pairVxCandidate) = (*pairItr).combinationIndices(); + chargeDecorator(*pairVxCandidate) = (*pairItr).combinationCharges(); + // decorate the candidate with refitted tracks and muons via the BPhysHelper + xAOD::BPhysHelper helper(pairVxCandidate); + helper.setRefTrks(); + std::vector<const xAOD::Muon*> theStoredMuons; + theStoredMuons = (*pairItr).muons; + helper.setMuons(theStoredMuons,importedMuonCollection.get()); + // Retain the vertex + pairVxContainer->push_back(pairVxCandidate); + ATH_MSG_DEBUG("..... indices: " << (*pairItr).combinationIndices() << + " charges: " << (*pairItr).combinationCharges() << + " chi2: " << pairVxCandidate->chiSquared()); + } else { // fit failed + ATH_MSG_DEBUG("Fitter failed!"); + } + } + ATH_MSG_DEBUG("pairContainer size " << pairVxContainer->size()); + + // fit quadruplets + ATH_MSG_DEBUG("Successful quadruplets....."); + for (std::vector<Combination>::iterator quadItr = quadruplets.begin(); quadItr!=quadruplets.end(); ++quadItr) { + std::vector<const xAOD::TrackParticle*> theDCTracks; theDCTracks.clear(); + //std::vector<const xAOD::TrackParticle*> theACTracks; theACTracks.clear(); + //std::vector<const xAOD::TrackParticle*> theSSTracks; theSSTracks.clear(); + theDCTracks = (*quadItr).trackParticles("DC"); + //theACTracks = (*quadItr).trackParticles("AC"); + //theSSTracks = (*quadItr).trackParticles("SS"); + xAOD::Vertex* dcVxCandidate = fit(theDCTracks,importedTrackCollection.get(), beamSpot); + //xAOD::Vertex* acVxCandidate = fit(theACTracks,importedTrackCollection, beamSpot); + //xAOD::Vertex* ssVxCandidate = fit(theSSTracks,importedTrackCollection, beamSpot); + // Get the chi2 for each one + //double acChi2(0.0); + //double ssChi2(0.0); + //if (acVxCandidate != 0) {acChi2 = acVxCandidate->chiSquared();} + //if (ssVxCandidate != 0) {ssChi2 = ssVxCandidate->chiSquared();} + if (dcVxCandidate != 0) { + // decorate the candidate with its codes + indexDecorator(*dcVxCandidate) = (*quadItr).combinationIndices(); + chargeDecorator(*dcVxCandidate) = (*quadItr).combinationCharges(); + // Decorate the DC candidate with the differences between its chi2 and the other + double dcChi2 = dcVxCandidate->chiSquared(); + //acdcDecorator(*dcVxCandidate) = acChi2 - dcChi2; + //ssdcDecorator(*dcVxCandidate) = ssChi2 - dcChi2; + // decorate the candidate with refitted tracks and muons via the BPhysHelper + xAOD::BPhysHelper helper(dcVxCandidate); + helper.setRefTrks(); + const std::vector<const xAOD::Muon*> &theStoredMuons = (*quadItr).muons; + helper.setMuons(theStoredMuons,importedMuonCollection.get()); + // Retain the vertex + quadVxContainer->push_back(dcVxCandidate); + ATH_MSG_DEBUG("..... indices: " << (*quadItr).combinationIndices() << + " charges: " << (*quadItr).combinationCharges() << + " chi2(DC): " << dcChi2); + //" chi2(AC): " << acChi2 << + //" chi2(SS): " << ssChi2); + } else { // fit failed + ATH_MSG_DEBUG("Fitter failed!"); + } + } + ATH_MSG_DEBUG("quadruplet container size " << quadVxContainer->size()); + + return StatusCode::SUCCESS;; + } + + // ********************************************************************************* + + // --------------------------------------------------------------------------------- + // fit - does the fit + // --------------------------------------------------------------------------------- + + xAOD::Vertex* FourMuonTool::fit(const std::vector<const xAOD::TrackParticle*> &inputTracks, + const xAOD::TrackParticleContainer* importedTrackCollection, + const Amg::Vector3D &beamSpot) const { + + const Trk::TrkV0VertexFitter* concreteVertexFitter=0; + if (m_useV0Fitter) { + // making a concrete fitter for the V0Fitter + concreteVertexFitter = dynamic_cast<const Trk::TrkV0VertexFitter * >(&(*m_iV0VertexFitter)); + if(concreteVertexFitter == 0) { + ATH_MSG_FATAL("The vertex fitter passed is not a V0 Vertex Fitter"); + return NULL; + } + } + + //int sflag = 0; + //int errorcode = 0; + //Amg::Vector3D startingPoint = m_vertexEstimator->getCirclesIntersectionPoint(&aPerigee1,&aPerigee2,sflag,errorcode); + //startingPoint(0) = 0.0; startingPoint(1) = 0.0; startingPoint(2) = 0.0;} + //Trk::Vertex vertex(beamSpot); + + xAOD::Vertex* myVxCandidate = nullptr; + if (m_useV0Fitter) { + myVxCandidate = concreteVertexFitter->fit(inputTracks, beamSpot /*vertex startingPoint*/ ); + } else { + myVxCandidate = m_iVertexFitter->fit(inputTracks, beamSpot /*vertex startingPoint*/ ); + } + + if(myVxCandidate) BPhysPVTools::PrepareVertexLinks(myVxCandidate, importedTrackCollection); + + return myVxCandidate; + + } // End of fit method + + + // ********************************************************************************* + + // --------------------------------------------------------------------------------- + // getQuadIndices: forms up index lists + // --------------------------------------------------------------------------------- + + std::vector<std::vector<unsigned int> > FourMuonTool::getQuadIndices(unsigned int length) { + + std::vector<std::vector<unsigned int> > quadIndices = mFromN(4,length); + return(quadIndices); + } + + + // ********************************************************************************* + + // --------------------------------------------------------------------------------- + // mFromN and combinatorics + // --------------------------------------------------------------------------------- + std::vector<std::vector<unsigned int> > FourMuonTool::mFromN(unsigned int m, unsigned int N) { + + std::vector<std::vector<unsigned int> > allCombinations; + std::vector<unsigned int> mainList; + std::vector<unsigned int> combination; + for (unsigned int i=0; i<N; ++i) mainList.push_back(i); + combinatorics(0,m,combination,mainList,allCombinations); + return allCombinations; + } + + void FourMuonTool::combinatorics(unsigned int offset, + unsigned int k, + std::vector<unsigned int> &combination, + std::vector<unsigned int> &mainList, + std::vector<std::vector<unsigned int> > &allCombinations) { + if (k==0) { + allCombinations.push_back(combination); + return; + } + if (k>0) { + for (unsigned int i=offset; i<=mainList.size()-k; ++i) { + combination.push_back(mainList[i]); + combinatorics(i+1,k-1,combination,mainList,allCombinations); + combination.pop_back(); + } + } + } + + // --------------------------------------------------------------------------------- + // getPairIndices + // --------------------------------------------------------------------------------- + + std::vector<std::pair<unsigned int, unsigned int> > FourMuonTool::getPairIndices(unsigned int length){ + + std::vector<std::pair<unsigned int, unsigned int> > uniquePairs; + std::vector<std::vector<unsigned int> > doublets = mFromN(2,length); + for (std::vector<std::vector<unsigned int> >::iterator it=doublets.begin(); it!=doublets.end(); ++it) { + std::pair<unsigned int, unsigned int> tmpPair = std::make_pair((*it).at(0),(*it).at(1)); + uniquePairs.push_back(tmpPair); + } + + return(uniquePairs); + } + + + + // ********************************************************************************* + + // --------------------------------------------------------------------------------- + // buildCombinations: forms up the quadruplet of muons/tracks + // --------------------------------------------------------------------------------- + + void FourMuonTool::buildCombinations(const std::vector<const xAOD::Muon*> &muonsIn, + std::vector<Combination> &pairs, + std::vector<Combination> &quadruplets, + unsigned int nSelectedMuons) { + + std::vector<std::vector<unsigned int> > quadrupletIndices = getQuadIndices(nSelectedMuons); + std::vector<std::pair<unsigned int, unsigned int> > pairIndices = getPairIndices(nSelectedMuons); + + // Quadruplets + std::vector<std::vector<unsigned int> >::iterator quadItr; + for (quadItr=quadrupletIndices.begin(); quadItr!=quadrupletIndices.end(); ++quadItr) { + const std::vector<unsigned int> &quad = (*quadItr); + std::vector<const xAOD::Muon*> theMuons = {muonsIn[quad[0]],muonsIn[quad[1]],muonsIn[quad[2]],muonsIn[quad[3]]}; + if (!passesQuadSelection(theMuons)) continue; + Combination tmpQuad; + tmpQuad.muons = std::move(theMuons); + tmpQuad.quadIndices = quad; + quadruplets.emplace_back(std::move(tmpQuad)); + } + if (quadruplets.size() == 0) return; + + // pairs + std::vector<std::pair<unsigned int, unsigned int> >::iterator pairItr; + for (pairItr=pairIndices.begin(); pairItr!=pairIndices.end(); ++pairItr) { + std::pair<unsigned int, unsigned int> pair = (*pairItr); + Combination tmpPair; + std::vector<const xAOD::Muon*> theMuons = {muonsIn[pair.first],muonsIn[pair.second]}; + tmpPair.muons = std::move(theMuons); + tmpPair.pairIndices = pair; + pairs.emplace_back(std::move(tmpPair)); + } + + return; + + } + + // ********************************************************************************* + + // --------------------------------------------------------------------------------- + // passesQuadSelection: 4-muon selection + // --------------------------------------------------------------------------------- + + bool FourMuonTool::passesQuadSelection(const std::vector<const xAOD::Muon*> &muons) { + bool accept(false); + bool charges(true); + bool quality(false); + //unsigned int sumCharges = abs(mu0->charge() + mu1->charge() + mu2->charge() + mu3->charge()); + //if (sumCharges<4) charges = true; + if (( muons.at(0)->muonType() == xAOD::Muon::Combined ) || + ( muons.at(1)->muonType() == xAOD::Muon::Combined ) || + ( muons.at(2)->muonType() == xAOD::Muon::Combined ) || + ( muons.at(3)->muonType() == xAOD::Muon::Combined ) + ) quality = true; + if (charges && quality) accept = true; + return accept; + } + +} diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/JpsiPlusDpstCascade.cxx b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/JpsiPlusDpstCascade.cxx new file mode 100644 index 00000000000..04f187bf3d8 --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/JpsiPlusDpstCascade.cxx @@ -0,0 +1,724 @@ +/* + Copyright (C) 2002-2018 CERN for the benefit of the ATLAS collaboration +*/ +///////////////////////////////////////////////////////////////// +// JpsiPlusDpstCascade.cxx, (c) ATLAS Detector software +///////////////////////////////////////////////////////////////// +#include "DerivationFrameworkBPhys/JpsiPlusDpstCascade.h" +#include "TrkVertexFitterInterfaces/IVertexFitter.h" +#include "TrkVKalVrtFitter/TrkVKalVrtFitter.h" +#include "TrkVertexAnalysisUtils/V0Tools.h" +#include "GaudiKernel/IPartPropSvc.h" +#include "DerivationFrameworkBPhys/CascadeTools.h" +#include "DerivationFrameworkBPhys/BPhysPVCascadeTools.h" +#include "xAODTracking/VertexAuxContainer.h" +#include "xAODBPhys/BPhysHypoHelper.h" +#include <algorithm> +#include "xAODTracking/VertexContainer.h" +#include "DerivationFrameworkBPhys/LocalVector.h" +#include "HepPDT/ParticleDataTable.hh" + +namespace DerivationFramework { + typedef ElementLink<xAOD::VertexContainer> VertexLink; + typedef std::vector<VertexLink> VertexLinkVector; + typedef std::vector<const xAOD::TrackParticle*> TrackBag; + + + StatusCode JpsiPlusDpstCascade::initialize() { + + // retrieving vertex Fitter + ATH_CHECK( m_iVertexFitter.retrieve()); + + // retrieving the V0 tools + ATH_CHECK( m_V0Tools.retrieve()); + + // retrieving the Cascade tools + ATH_CHECK( m_CascadeTools.retrieve()); + + // Get the beam spot service + ATH_CHECK(m_beamSpotKey.initialize()); + + IPartPropSvc* partPropSvc = nullptr; + ATH_CHECK( service("PartPropSvc", partPropSvc, true) ); + auto pdt = partPropSvc->PDT(); + + // retrieve particle masses + if(m_mass_jpsi < 0. ) m_mass_jpsi = BPhysPVCascadeTools::getParticleMass(pdt, PDG::J_psi); + if(m_vtx0MassHypo < 0.) m_vtx0MassHypo = BPhysPVCascadeTools::getParticleMass(pdt, PDG::B_c_plus); + if(m_vtx1MassHypo < 0.) m_vtx1MassHypo = BPhysPVCascadeTools::getParticleMass(pdt, PDG::D0); + + if(m_vtx0Daug1MassHypo < 0.) m_vtx0Daug1MassHypo = BPhysPVCascadeTools::getParticleMass(pdt, PDG::mu_minus); + if(m_vtx0Daug2MassHypo < 0.) m_vtx0Daug2MassHypo = BPhysPVCascadeTools::getParticleMass(pdt, PDG::mu_minus); + if(m_vtx0Daug3MassHypo < 0.) m_vtx0Daug3MassHypo = BPhysPVCascadeTools::getParticleMass(pdt, PDG::pi_plus); + if(m_vtx1Daug1MassHypo < 0.) m_vtx1Daug1MassHypo = BPhysPVCascadeTools::getParticleMass(pdt, PDG::pi_plus); + if(m_vtx1Daug2MassHypo < 0.) m_vtx1Daug2MassHypo = BPhysPVCascadeTools::getParticleMass(pdt, PDG::K_plus); + + return StatusCode::SUCCESS; + } + + + StatusCode JpsiPlusDpstCascade::addBranches() const + { + std::vector<Trk::VxCascadeInfo*> cascadeinfoContainer; + constexpr int topoN = 2; + std::array<xAOD::VertexContainer*, topoN> Vtxwritehandles; + std::array<xAOD::VertexAuxContainer*, topoN> Vtxwritehandlesaux; + if(m_cascadeOutputsKeys.size() !=topoN) { ATH_MSG_FATAL("Incorrect number of VtxContainers"); return StatusCode::FAILURE; } + + for(int i =0; i<topoN;i++){ + Vtxwritehandles[i] = new xAOD::VertexContainer(); + Vtxwritehandlesaux[i] = new xAOD::VertexAuxContainer(); + Vtxwritehandles[i]->setStore(Vtxwritehandlesaux[i]); + ATH_CHECK(evtStore()->record(Vtxwritehandles[i] , m_cascadeOutputsKeys[i] )); + ATH_CHECK(evtStore()->record(Vtxwritehandlesaux[i], m_cascadeOutputsKeys[i] + "Aux.")); + } + + //---------------------------------------------------- + // retrieve primary vertices + //---------------------------------------------------- + const xAOD::Vertex * primaryVertex(nullptr); + const xAOD::VertexContainer *pvContainer(nullptr); + ATH_CHECK(evtStore()->retrieve(pvContainer, m_VxPrimaryCandidateName)); + ATH_MSG_DEBUG("Found " << m_VxPrimaryCandidateName << " in StoreGate!"); + + if (pvContainer->size()==0){ + ATH_MSG_WARNING("You have no primary vertices: " << pvContainer->size()); + return StatusCode::RECOVERABLE; + } else { + primaryVertex = (*pvContainer)[0]; + } + + //---------------------------------------------------- + // Try to retrieve refitted primary vertices + //---------------------------------------------------- + xAOD::VertexContainer* refPvContainer = nullptr; + xAOD::VertexAuxContainer* refPvAuxContainer = nullptr; + if (m_refitPV) { + if (evtStore()->contains<xAOD::VertexContainer>(m_refPVContainerName)) { + // refitted PV container exists. Get it from the store gate + ATH_CHECK(evtStore()->retrieve(refPvContainer , m_refPVContainerName )); + ATH_CHECK(evtStore()->retrieve(refPvAuxContainer, m_refPVContainerName + "Aux.")); + } else { + // refitted PV container does not exist. Create a new one. + refPvContainer = new xAOD::VertexContainer; + refPvAuxContainer = new xAOD::VertexAuxContainer; + refPvContainer->setStore(refPvAuxContainer); + ATH_CHECK(evtStore()->record(refPvContainer , m_refPVContainerName)); + ATH_CHECK(evtStore()->record(refPvAuxContainer, m_refPVContainerName+"Aux.")); + } + } + + ATH_CHECK(performSearch(&cascadeinfoContainer)); + + SG::ReadCondHandle<InDet::BeamSpotData> beamSpotHandle { m_beamSpotKey }; + if(not beamSpotHandle.isValid()) ATH_MSG_ERROR("Cannot Retrieve " << m_beamSpotKey.key() ); + BPhysPVCascadeTools helper(&(*m_CascadeTools), beamSpotHandle.cptr()); + helper.SetMinNTracksInPV(m_PV_minNTracks); + + // Decorators for the main vertex: chi2, ndf, pt and pt error, plus the D0 vertex variables + SG::AuxElement::Decorator<VertexLinkVector> CascadeLinksDecor("CascadeVertexLinks"); + SG::AuxElement::Decorator<VertexLinkVector> JpsipiLinksDecor("JpsipiVertexLinks"); + SG::AuxElement::Decorator<VertexLinkVector> D0LinksDecor("D0VertexLinks"); + SG::AuxElement::Decorator<float> chi2_decor("ChiSquared"); + SG::AuxElement::Decorator<float> ndof_decor("NumberDoF"); + SG::AuxElement::Decorator<float> Pt_decor("Pt"); + SG::AuxElement::Decorator<float> PtErr_decor("PtErr"); + SG::AuxElement::Decorator<float> Mass_svdecor("D0_mass"); + SG::AuxElement::Decorator<float> MassErr_svdecor("D0_massErr"); + SG::AuxElement::Decorator<float> Pt_svdecor("D0_Pt"); + SG::AuxElement::Decorator<float> PtErr_svdecor("D0_PtErr"); + SG::AuxElement::Decorator<float> Lxy_svdecor("D0_Lxy"); + SG::AuxElement::Decorator<float> LxyErr_svdecor("D0_LxyErr"); + SG::AuxElement::Decorator<float> Tau_svdecor("D0_Tau"); + SG::AuxElement::Decorator<float> TauErr_svdecor("D0_TauErr"); + + SG::AuxElement::Decorator<float> MassMumu_decor("Mumu_mass"); + SG::AuxElement::Decorator<float> MassKpi_svdecor("Kpi_mass"); + SG::AuxElement::Decorator<float> MassJpsi_decor("Jpsi_mass"); + SG::AuxElement::Decorator<float> MassPiD0_decor("PiD0_mass"); + + ATH_MSG_DEBUG("cascadeinfoContainer size " << cascadeinfoContainer.size()); + + // Get Jpsi+pi container and identify the input Jpsi+pi + const xAOD::VertexContainer *jpsipiContainer(nullptr); + ATH_CHECK(evtStore()->retrieve(jpsipiContainer , m_vertexContainerKey )); + const xAOD::VertexContainer *d0Container(nullptr); + ATH_CHECK(evtStore()->retrieve(d0Container , m_vertexD0ContainerKey )); + + for (Trk::VxCascadeInfo* x : cascadeinfoContainer) { + if(x==nullptr) ATH_MSG_ERROR("cascadeinfoContainer is null"); + + // the cascade fitter returns: + // std::vector<xAOD::Vertex*>, each xAOD::Vertex contains the refitted track parameters (perigee at the vertex position) + // vertices[iv] the links to the original TPs and a covariance of size 3+5*NTRK; the chi2 of the total fit + // is split between the cascade vertices as per track contribution + // std::vector< std::vector<TLorentzVector> >, each std::vector<TLorentzVector> contains the refitted momenta (TLorentzVector) + // momenta[iv][...] of all tracks in the corresponding vertex, including any pseudotracks (from cascade vertices) + // originating in this vertex; the masses are as assigned in the cascade fit + // std::vector<Amg::MatrixX>, the corresponding covariance matrices in momentum space + // covariance[iv] + // int nDoF, double Chi2 + // + // the invariant mass, pt, lifetime etc. errors should be calculated using the covariance matrices in momentum space as these + // take into account the full track-track and track-vertex correlations + // + // in the case of Jpsi+V0: vertices[0] is the V0 vertex, vertices[1] is the B/Lambda_b(bar) vertex, containing the 2 Jpsi tracks. + // The covariance terms between the two vertices are not stored. In momentum space momenta[0] contains the 2 V0 tracks, + // their momenta add up to the momentum of the 3rd track in momenta[1], the first two being the Jpsi tracks + + const std::vector<xAOD::Vertex*> &cascadeVertices = x->vertices(); + if(cascadeVertices.size()!=topoN) + ATH_MSG_ERROR("Incorrect number of vertices"); + if(cascadeVertices[0] == nullptr || cascadeVertices[1] == nullptr) ATH_MSG_ERROR("Error null vertex"); + // Keep vertices (bear in mind that they come in reverse order!) + for(int i =0;i<topoN;i++) Vtxwritehandles[i]->push_back(cascadeVertices[i]); + + x->setSVOwnership(false); // Prevent Container from deleting vertices + const auto mainVertex = cascadeVertices[1]; // this is the B_c+/- vertex + const std::vector< std::vector<TLorentzVector> > &moms = x->getParticleMoms(); + + // Set links to cascade vertices + std::vector<const xAOD::Vertex*> verticestoLink; + verticestoLink.push_back(cascadeVertices[0]); + if(Vtxwritehandles[1] == nullptr) ATH_MSG_ERROR("Vtxwritehandles[1] is null"); + if(!BPhysPVCascadeTools::LinkVertices(CascadeLinksDecor, verticestoLink, Vtxwritehandles[0], cascadeVertices[1])) + ATH_MSG_ERROR("Error decorating with cascade vertices"); + + // Identify the input Jpsi+pi + const xAOD::Vertex* jpsipiVertex = BPhysPVCascadeTools::FindVertex<3>(jpsipiContainer, cascadeVertices[1]); + ATH_MSG_DEBUG("1 pt Jpsi+pi tracks " << cascadeVertices[1]->trackParticle(0)->pt() << ", " << cascadeVertices[1]->trackParticle(1)->pt() << ", " << cascadeVertices[1]->trackParticle(2)->pt()); + if (jpsipiVertex) ATH_MSG_DEBUG("2 pt Jpsi+pi tracks " << jpsipiVertex->trackParticle(0)->pt() << ", " << jpsipiVertex->trackParticle(1)->pt() << ", " << jpsipiVertex->trackParticle(2)->pt()); + + // Identify the input D0 + const xAOD::Vertex* d0Vertex = BPhysPVCascadeTools::FindVertex<2>(d0Container, cascadeVertices[0]);; + ATH_MSG_DEBUG("1 pt D0 tracks " << cascadeVertices[0]->trackParticle(0)->pt() << ", " << cascadeVertices[0]->trackParticle(1)->pt()); + if (d0Vertex) ATH_MSG_DEBUG("2 pt D0 tracks " << d0Vertex->trackParticle(0)->pt() << ", " << d0Vertex->trackParticle(1)->pt()); + + // Set links to input vertices + std::vector<const xAOD::Vertex*> jpsipiVerticestoLink; + if (jpsipiVertex) jpsipiVerticestoLink.push_back(jpsipiVertex); + else ATH_MSG_WARNING("Could not find linking Jpsi+pi"); + if(!BPhysPVCascadeTools::LinkVertices(JpsipiLinksDecor, jpsipiVerticestoLink, jpsipiContainer, cascadeVertices[1])) + ATH_MSG_ERROR("Error decorating with Jpsi+pi vertices"); + + std::vector<const xAOD::Vertex*> d0VerticestoLink; + if (d0Vertex) d0VerticestoLink.push_back(d0Vertex); + else ATH_MSG_WARNING("Could not find linking D0"); + if(!BPhysPVCascadeTools::LinkVertices(D0LinksDecor, d0VerticestoLink, d0Container, cascadeVertices[1])) + ATH_MSG_ERROR("Error decorating with D0 vertices"); + + bool tagD0(true); + if (jpsipiVertex){ + if(abs(m_Dx_pid)==421 && (jpsipiVertex->trackParticle(2)->charge()==-1)) tagD0 = false; + } + + double mass_b = m_vtx0MassHypo; + double mass_d0 = m_vtx1MassHypo; + std::vector<double> massesJpsipi; + massesJpsipi.push_back(m_vtx0Daug1MassHypo); + massesJpsipi.push_back(m_vtx0Daug2MassHypo); + massesJpsipi.push_back(m_vtx0Daug3MassHypo); + std::vector<double> massesD0; + if(tagD0){ + massesD0.push_back(m_vtx1Daug1MassHypo); + massesD0.push_back(m_vtx1Daug2MassHypo); + }else{ // Change the oreder of masses for D*-->D0bar pi-, D0bar->K+pi- + massesD0.push_back(m_vtx1Daug2MassHypo); + massesD0.push_back(m_vtx1Daug1MassHypo); + } + std::vector<double> Masses; + Masses.push_back(m_vtx0Daug1MassHypo); + Masses.push_back(m_vtx0Daug2MassHypo); + Masses.push_back(m_vtx0Daug3MassHypo); + Masses.push_back(m_vtx1MassHypo); + + // loop over candidates -- Don't apply PV_minNTracks requirement here + // because it may result in exclusion of the high-pt PV. + // get good PVs + + xAOD::BPhysHypoHelper vtx(m_hypoName, mainVertex); + + // Get refitted track momenta from all vertices, charged tracks only + BPhysPVCascadeTools::SetVectorInfo(vtx, x); + + // Decorate main vertex + // + // 1.a) mass, mass error + BPHYS_CHECK( vtx.setMass(m_CascadeTools->invariantMass(moms[1])) ); + BPHYS_CHECK( vtx.setMassErr(m_CascadeTools->invariantMassError(moms[1],x->getCovariance()[1])) ); + // 1.b) pt and pT error (the default pt of mainVertex is != the pt of the full cascade fit!) + Pt_decor(*mainVertex) = m_CascadeTools->pT(moms[1]); + PtErr_decor(*mainVertex) = m_CascadeTools->pTError(moms[1],x->getCovariance()[1]); + // 1.c) chi2 and ndof (the default chi2 of mainVertex is != the chi2 of the full cascade fit!) + chi2_decor(*mainVertex) = x->fitChi2(); + ndof_decor(*mainVertex) = x->nDoF(); + + float massMumu = 0.; + if (jpsipiVertex) { + TLorentzVector p4_mu1, p4_mu2; + p4_mu1.SetPtEtaPhiM(jpsipiVertex->trackParticle(0)->pt(), + jpsipiVertex->trackParticle(0)->eta(), + jpsipiVertex->trackParticle(0)->phi(), m_vtx0Daug1MassHypo); + p4_mu2.SetPtEtaPhiM(jpsipiVertex->trackParticle(1)->pt(), + jpsipiVertex->trackParticle(1)->eta(), + jpsipiVertex->trackParticle(1)->phi(), m_vtx0Daug2MassHypo); + massMumu = (p4_mu1 + p4_mu2).M(); + } + MassMumu_decor(*mainVertex) = massMumu; + + float massKpi = 0.; + if (d0Vertex) { + TLorentzVector p4_ka, p4_pi; + if(tagD0){ + p4_pi.SetPtEtaPhiM(d0Vertex->trackParticle(0)->pt(), + d0Vertex->trackParticle(0)->eta(), + d0Vertex->trackParticle(0)->phi(), m_vtx1Daug1MassHypo); + p4_ka.SetPtEtaPhiM(d0Vertex->trackParticle(1)->pt(), + d0Vertex->trackParticle(1)->eta(), + d0Vertex->trackParticle(1)->phi(), m_vtx1Daug2MassHypo); + }else{ // Change the oreder of masses for D*-->D0bar pi-, D0bar->K+pi- + p4_pi.SetPtEtaPhiM(d0Vertex->trackParticle(1)->pt(), + d0Vertex->trackParticle(1)->eta(), + d0Vertex->trackParticle(1)->phi(), m_vtx1Daug1MassHypo); + p4_ka.SetPtEtaPhiM(d0Vertex->trackParticle(0)->pt(), + d0Vertex->trackParticle(0)->eta(), + d0Vertex->trackParticle(0)->phi(), m_vtx1Daug2MassHypo); + } + massKpi = (p4_ka + p4_pi).M(); + } + MassKpi_svdecor(*mainVertex) = massKpi; + MassJpsi_decor(*mainVertex) = (moms[1][0] + moms[1][1]).M(); + MassPiD0_decor(*mainVertex) = (moms[1][2] + moms[1][3]).M(); + + + ATH_CHECK(helper.FillCandwithRefittedVertices(m_refitPV, pvContainer, + refPvContainer, &(*m_pvRefitter), m_PV_max, m_DoVertexType, x, 1, mass_b, vtx)); + + // 4) decorate the main vertex with D0 vertex mass, pt, lifetime and lxy values (plus errors) + // D0 points to the main vertex, so lifetime and lxy are w.r.t the main vertex + Mass_svdecor(*mainVertex) = m_CascadeTools->invariantMass(moms[0]); + MassErr_svdecor(*mainVertex) = m_CascadeTools->invariantMassError(moms[0],x->getCovariance()[0]); + Pt_svdecor(*mainVertex) = m_CascadeTools->pT(moms[0]); + PtErr_svdecor(*mainVertex) = m_CascadeTools->pTError(moms[0],x->getCovariance()[0]); + Lxy_svdecor(*mainVertex) = m_CascadeTools->lxy(moms[0],cascadeVertices[0],cascadeVertices[1]); + LxyErr_svdecor(*mainVertex) = m_CascadeTools->lxyError(moms[0],x->getCovariance()[0],cascadeVertices[0],cascadeVertices[1]); + Tau_svdecor(*mainVertex) = m_CascadeTools->tau(moms[0],cascadeVertices[0],cascadeVertices[1]); + TauErr_svdecor(*mainVertex) = m_CascadeTools->tauError(moms[0],x->getCovariance()[0],cascadeVertices[0],cascadeVertices[1]); + + // Some checks in DEBUG mode + ATH_MSG_DEBUG("chi2 " << x->fitChi2() + << " chi2_1 " << m_V0Tools->chisq(cascadeVertices[0]) + << " chi2_2 " << m_V0Tools->chisq(cascadeVertices[1]) + << " vprob " << m_CascadeTools->vertexProbability(x->nDoF(),x->fitChi2())); + ATH_MSG_DEBUG("ndf " << x->nDoF() << " ndf_1 " << m_V0Tools->ndof(cascadeVertices[0]) << " ndf_2 " << m_V0Tools->ndof(cascadeVertices[1])); + ATH_MSG_DEBUG("V0Tools mass_d0 " << m_V0Tools->invariantMass(cascadeVertices[0],massesD0) + << " error " << m_V0Tools->invariantMassError(cascadeVertices[0],massesD0) + << " mass_J " << m_V0Tools->invariantMass(cascadeVertices[1],massesJpsipi) + << " error " << m_V0Tools->invariantMassError(cascadeVertices[1],massesJpsipi)); + // masses and errors, using track masses assigned in the fit + double Mass_B = m_CascadeTools->invariantMass(moms[1]); + double Mass_D0 = m_CascadeTools->invariantMass(moms[0]); + double Mass_B_err = m_CascadeTools->invariantMassError(moms[1],x->getCovariance()[1]); + double Mass_D0_err = m_CascadeTools->invariantMassError(moms[0],x->getCovariance()[0]); + ATH_MSG_DEBUG("Mass_B " << Mass_B << " Mass_D0 " << Mass_D0); + ATH_MSG_DEBUG("Mass_B_err " << Mass_B_err << " Mass_D0_err " << Mass_D0_err); + double mprob_B = m_CascadeTools->massProbability(mass_b,Mass_B,Mass_B_err); + double mprob_D0 = m_CascadeTools->massProbability(mass_d0,Mass_D0,Mass_D0_err); + ATH_MSG_DEBUG("mprob_B " << mprob_B << " mprob_D0 " << mprob_D0); + // masses and errors, assigning user defined track masses + ATH_MSG_DEBUG("Mass_b " << m_CascadeTools->invariantMass(moms[1],Masses) + << " Mass_d0 " << m_CascadeTools->invariantMass(moms[0],massesD0)); + ATH_MSG_DEBUG("Mass_b_err " << m_CascadeTools->invariantMassError(moms[1],x->getCovariance()[1],Masses) + << " Mass_d0_err " << m_CascadeTools->invariantMassError(moms[0],x->getCovariance()[0],massesD0)); + ATH_MSG_DEBUG("pt_b " << m_CascadeTools->pT(moms[1]) + << " pt_d " << m_CascadeTools->pT(moms[0]) + << " pt_d0 " << m_V0Tools->pT(cascadeVertices[0])); + ATH_MSG_DEBUG("ptErr_b " << m_CascadeTools->pTError(moms[1],x->getCovariance()[1]) + << " ptErr_d " << m_CascadeTools->pTError(moms[0],x->getCovariance()[0]) + << " ptErr_d0 " << m_V0Tools->pTError(cascadeVertices[0])); + ATH_MSG_DEBUG("lxy_B " << m_V0Tools->lxy(cascadeVertices[1],primaryVertex) << " lxy_D " << m_V0Tools->lxy(cascadeVertices[0],cascadeVertices[1])); + ATH_MSG_DEBUG("lxy_b " << m_CascadeTools->lxy(moms[1],cascadeVertices[1],primaryVertex) << " lxy_d " << m_CascadeTools->lxy(moms[0],cascadeVertices[0],cascadeVertices[1])); + ATH_MSG_DEBUG("lxyErr_b " << m_CascadeTools->lxyError(moms[1],x->getCovariance()[1],cascadeVertices[1],primaryVertex) + << " lxyErr_d " << m_CascadeTools->lxyError(moms[0],x->getCovariance()[0],cascadeVertices[0],cascadeVertices[1]) + << " lxyErr_d0 " << m_V0Tools->lxyError(cascadeVertices[0],cascadeVertices[1])); + ATH_MSG_DEBUG("tau_B " << m_CascadeTools->tau(moms[1],cascadeVertices[1],primaryVertex,mass_b) + << " tau_d0 " << m_V0Tools->tau(cascadeVertices[0],cascadeVertices[1],massesD0)); + ATH_MSG_DEBUG("tau_b " << m_CascadeTools->tau(moms[1],cascadeVertices[1],primaryVertex) + << " tau_d " << m_CascadeTools->tau(moms[0],cascadeVertices[0],cascadeVertices[1]) + << " tau_D " << m_CascadeTools->tau(moms[0],cascadeVertices[0],cascadeVertices[1],mass_d0)); + ATH_MSG_DEBUG("tauErr_b " << m_CascadeTools->tauError(moms[1],x->getCovariance()[1],cascadeVertices[1],primaryVertex) + << " tauErr_d " << m_CascadeTools->tauError(moms[0],x->getCovariance()[0],cascadeVertices[0],cascadeVertices[1]) + << " tauErr_d0 " << m_V0Tools->tauError(cascadeVertices[0],cascadeVertices[1],massesD0)); + ATH_MSG_DEBUG("TauErr_b " << m_CascadeTools->tauError(moms[1],x->getCovariance()[1],cascadeVertices[1],primaryVertex,mass_b) + << " TauErr_d " << m_CascadeTools->tauError(moms[0],x->getCovariance()[0],cascadeVertices[0],cascadeVertices[1],mass_d0) + << " TauErr_d0 " << m_V0Tools->tauError(cascadeVertices[0],cascadeVertices[1],massesD0,mass_d0)); + + ATH_MSG_DEBUG("CascadeTools main vert wrt PV " << " CascadeTools SV " << " V0Tools SV"); + ATH_MSG_DEBUG("a0z " << m_CascadeTools->a0z(moms[1],cascadeVertices[1],primaryVertex) + << ", " << m_CascadeTools->a0z(moms[0],cascadeVertices[0],cascadeVertices[1]) + << ", " << m_V0Tools->a0z(cascadeVertices[0],cascadeVertices[1])); + ATH_MSG_DEBUG("a0zErr " << m_CascadeTools->a0zError(moms[1],x->getCovariance()[1],cascadeVertices[1],primaryVertex) + << ", " << m_CascadeTools->a0zError(moms[0],x->getCovariance()[0],cascadeVertices[0],cascadeVertices[1]) + << ", " << m_V0Tools->a0zError(cascadeVertices[0],cascadeVertices[1])); + ATH_MSG_DEBUG("a0xy " << m_CascadeTools->a0xy(moms[1],cascadeVertices[1],primaryVertex) + << ", " << m_CascadeTools->a0xy(moms[0],cascadeVertices[0],cascadeVertices[1]) + << ", " << m_V0Tools->a0xy(cascadeVertices[0],cascadeVertices[1])); + ATH_MSG_DEBUG("a0xyErr " << m_CascadeTools->a0xyError(moms[1],x->getCovariance()[1],cascadeVertices[1],primaryVertex) + << ", " << m_CascadeTools->a0xyError(moms[0],x->getCovariance()[0],cascadeVertices[0],cascadeVertices[1]) + << ", " << m_V0Tools->a0xyError(cascadeVertices[0],cascadeVertices[1])); + ATH_MSG_DEBUG("a0 " << m_CascadeTools->a0(moms[1],cascadeVertices[1],primaryVertex) + << ", " << m_CascadeTools->a0(moms[0],cascadeVertices[0],cascadeVertices[1]) + << ", " << m_V0Tools->a0(cascadeVertices[0],cascadeVertices[1])); + ATH_MSG_DEBUG("a0Err " << m_CascadeTools->a0Error(moms[1],x->getCovariance()[1],cascadeVertices[1],primaryVertex) + << ", " << m_CascadeTools->a0Error(moms[0],x->getCovariance()[0],cascadeVertices[0],cascadeVertices[1]) + << ", " << m_V0Tools->a0Error(cascadeVertices[0],cascadeVertices[1])); + ATH_MSG_DEBUG("x0 " << m_V0Tools->vtx(cascadeVertices[0]).x() << " y0 " << m_V0Tools->vtx(cascadeVertices[0]).y() << " z0 " << m_V0Tools->vtx(cascadeVertices[0]).z()); + ATH_MSG_DEBUG("x1 " << m_V0Tools->vtx(cascadeVertices[1]).x() << " y1 " << m_V0Tools->vtx(cascadeVertices[1]).y() << " z1 " << m_V0Tools->vtx(cascadeVertices[1]).z()); + ATH_MSG_DEBUG("X0 " << primaryVertex->x() << " Y0 " << primaryVertex->y() << " Z0 " << primaryVertex->z()); + ATH_MSG_DEBUG("rxy0 " << m_V0Tools->rxy(cascadeVertices[0]) << " rxyErr0 " << m_V0Tools->rxyError(cascadeVertices[0])); + ATH_MSG_DEBUG("rxy1 " << m_V0Tools->rxy(cascadeVertices[1]) << " rxyErr1 " << m_V0Tools->rxyError(cascadeVertices[1])); + ATH_MSG_DEBUG("Rxy0 wrt PV " << m_V0Tools->rxy(cascadeVertices[0],primaryVertex) << " RxyErr0 wrt PV " << m_V0Tools->rxyError(cascadeVertices[0],primaryVertex)); + ATH_MSG_DEBUG("Rxy1 wrt PV " << m_V0Tools->rxy(cascadeVertices[1],primaryVertex) << " RxyErr1 wrt PV " << m_V0Tools->rxyError(cascadeVertices[1],primaryVertex)); + ATH_MSG_DEBUG("number of covariance matrices " << (x->getCovariance()).size()); + } // loop over cascadeinfoContainer + + // Deleting cascadeinfo since this won't be stored. + // Vertices have been kept in m_cascadeOutputs and should be owned by their container + for (auto x : cascadeinfoContainer) delete x; + + return StatusCode::SUCCESS; + } + + + JpsiPlusDpstCascade::JpsiPlusDpstCascade(const std::string& t, const std::string& n, const IInterface* p) : AthAlgTool(t,n,p), + m_vertexContainerKey(""), + m_vertexD0ContainerKey(""), + m_cascadeOutputsKeys{ "JpsiPlusDpstCascadeVtx1", "JpsiPlusDpstCascadeVtx2" }, + m_VxPrimaryCandidateName("PrimaryVertices"), + m_jpsiMassLower(0.0), + m_jpsiMassUpper(10000.0), + m_jpsipiMassLower(0.0), + m_jpsipiMassUpper(10000.0), + m_D0MassLower(0.0), + m_D0MassUpper(10000.0), + m_DstMassLower(0.0), + m_DstMassUpper(10000.0), + m_MassLower(0.0), + m_MassUpper(20000.0), + m_vtx0MassHypo(-1), + m_vtx1MassHypo(-1), + m_vtx0Daug1MassHypo(-1), + m_vtx0Daug2MassHypo(-1), + m_vtx0Daug3MassHypo(-1), + m_vtx1Daug1MassHypo(-1), + m_vtx1Daug2MassHypo(-1), + m_mass_jpsi(-1), + m_Dx_pid(421), + m_constrD0(true), + m_constrJpsi(true), + m_chi2cut(-1.0), + m_iVertexFitter("Trk::TrkVKalVrtFitter"), + m_pvRefitter("Analysis::PrimaryVertexRefitter"), + m_V0Tools("Trk::V0Tools"), + m_CascadeTools("DerivationFramework::CascadeTools") + { + declareProperty("JpsipiVertices", m_vertexContainerKey); + declareProperty("D0Vertices", m_vertexD0ContainerKey); + declareProperty("VxPrimaryCandidateName", m_VxPrimaryCandidateName); + declareProperty("RefPVContainerName", m_refPVContainerName = "RefittedPrimaryVertices"); + declareProperty("JpsiMassLowerCut", m_jpsiMassLower); + declareProperty("JpsiMassUpperCut", m_jpsiMassUpper); + declareProperty("JpsipiMassLowerCut", m_jpsipiMassLower); + declareProperty("JpsipiMassUpperCut", m_jpsipiMassUpper); + declareProperty("D0MassLowerCut", m_D0MassLower); + declareProperty("D0MassUpperCut", m_D0MassUpper); + declareProperty("DstMassLowerCut", m_DstMassLower); + declareProperty("DstMassUpperCut", m_DstMassUpper); + declareProperty("MassLowerCut", m_MassLower); + declareProperty("MassUpperCut", m_MassUpper); + declareProperty("HypothesisName", m_hypoName = "Bc"); + declareProperty("Vtx0MassHypo", m_vtx0MassHypo); + declareProperty("Vtx1MassHypo", m_vtx1MassHypo); + declareProperty("Vtx0Daug1MassHypo", m_vtx0Daug1MassHypo); + declareProperty("Vtx0Daug2MassHypo", m_vtx0Daug2MassHypo); + declareProperty("Vtx0Daug3MassHypo", m_vtx0Daug3MassHypo); + declareProperty("Vtx1Daug1MassHypo", m_vtx1Daug1MassHypo); + declareProperty("Vtx1Daug2MassHypo", m_vtx1Daug2MassHypo); + declareProperty("JpsiMass", m_mass_jpsi); + declareProperty("DxHypothesis", m_Dx_pid); + declareProperty("ApplyD0MassConstraint", m_constrD0); + declareProperty("ApplyJpsiMassConstraint", m_constrJpsi); + declareProperty("Chi2Cut", m_chi2cut); + declareProperty("RefitPV", m_refitPV = true); + declareProperty("MaxnPV", m_PV_max = 999); + declareProperty("MinNTracksInPV", m_PV_minNTracks = 0); + declareProperty("DoVertexType", m_DoVertexType = 7); + declareProperty("TrkVertexFitterTool", m_iVertexFitter); + declareProperty("PVRefitter", m_pvRefitter); + declareProperty("V0Tools", m_V0Tools); + declareProperty("CascadeTools", m_CascadeTools); + declareProperty("CascadeVertexCollections", m_cascadeOutputsKeys); + } + + JpsiPlusDpstCascade::~JpsiPlusDpstCascade(){ } + + StatusCode JpsiPlusDpstCascade::performSearch(std::vector<Trk::VxCascadeInfo*> *cascadeinfoContainer) const + { + ATH_MSG_DEBUG( "JpsiPlusDpstCascade::performSearch" ); + assert(cascadeinfoContainer!=nullptr); + + // Get TrackParticle container (for setting links to the original tracks) + const xAOD::TrackParticleContainer *trackContainer(nullptr); + ATH_CHECK(evtStore()->retrieve(trackContainer , "InDetTrackParticles" )); + + // Get Jpsi+pi container + const xAOD::VertexContainer *jpsipiContainer(nullptr); + ATH_CHECK(evtStore()->retrieve(jpsipiContainer , m_vertexContainerKey )); + + // Get D0 container + const xAOD::VertexContainer *d0Container(nullptr); + ATH_CHECK(evtStore()->retrieve(d0Container , m_vertexD0ContainerKey )); + + double mass_d0 = m_vtx1MassHypo; + std::vector<const xAOD::TrackParticle*> tracksJpsipi; + std::vector<const xAOD::TrackParticle*> tracksJpsi; + std::vector<const xAOD::TrackParticle*> tracksD0; + std::vector<const xAOD::TrackParticle*> tracksBc; + std::vector<double> massesJpsipi; + massesJpsipi.push_back(m_vtx0Daug1MassHypo); + massesJpsipi.push_back(m_vtx0Daug2MassHypo); + massesJpsipi.push_back(m_vtx0Daug3MassHypo); + std::vector<double> massesD0; + massesD0.push_back(m_vtx1Daug1MassHypo); + massesD0.push_back(m_vtx1Daug2MassHypo); + std::vector<double> massesD0b; // Change the oreder of masses for D*-->D0bar pi-, D0bar->K+pi- + massesD0b.push_back(m_vtx1Daug2MassHypo); + massesD0b.push_back(m_vtx1Daug1MassHypo); + std::vector<double> Masses; + Masses.push_back(m_vtx0Daug1MassHypo); + Masses.push_back(m_vtx0Daug2MassHypo); + Masses.push_back(m_vtx0Daug3MassHypo); + Masses.push_back(m_vtx1MassHypo); + + // Select J/psi pi+ candidates before calling cascade fit + std::vector<const xAOD::Vertex*> selectedJpsipiCandidates; + for(auto vxcItr=jpsipiContainer->cbegin(); vxcItr!=jpsipiContainer->cend(); ++vxcItr) { + + // Check the passed flag first + const xAOD::Vertex* vtx = *vxcItr; + SG::AuxElement::Accessor<Char_t> flagAcc1("passed_Jpsipi"); + if(flagAcc1.isAvailable(*vtx)){ + if(!flagAcc1(*vtx)) continue; + } + + // Check J/psi candidate invariant mass and skip if need be + TLorentzVector p4Mup_in, p4Mum_in; + p4Mup_in.SetPtEtaPhiM((*vxcItr)->trackParticle(0)->pt(), + (*vxcItr)->trackParticle(0)->eta(), + (*vxcItr)->trackParticle(0)->phi(), m_vtx0Daug1MassHypo); + p4Mum_in.SetPtEtaPhiM((*vxcItr)->trackParticle(1)->pt(), + (*vxcItr)->trackParticle(1)->eta(), + (*vxcItr)->trackParticle(1)->phi(), m_vtx0Daug2MassHypo); + double mass_Jpsi = (p4Mup_in + p4Mum_in).M(); + ATH_MSG_DEBUG("Jpsi mass " << mass_Jpsi); + if (mass_Jpsi < m_jpsiMassLower || mass_Jpsi > m_jpsiMassUpper) { + ATH_MSG_DEBUG(" Original Jpsi candidate rejected by the mass cut: mass = " + << mass_Jpsi << " != (" << m_jpsiMassLower << ", " << m_jpsiMassUpper << ")" ); + continue; + } + + // Check J/psi pi+ candidate invariant mass and skip if need be + double mass_Jpsipi = m_V0Tools->invariantMass(*vxcItr, massesJpsipi); + ATH_MSG_DEBUG("Jpsipi mass " << mass_Jpsipi); + if (mass_Jpsipi < m_jpsipiMassLower || mass_Jpsipi > m_jpsipiMassUpper) { + ATH_MSG_DEBUG(" Original Jpsipi candidate rejected by the mass cut: mass = " + << mass_Jpsipi << " != (" << m_jpsipiMassLower << ", " << m_jpsipiMassUpper << ")" ); + continue; + } + + selectedJpsipiCandidates.push_back(*vxcItr); + } + if(selectedJpsipiCandidates.size()<1) return StatusCode::SUCCESS; + + // Select the D0/D0b candidates before calling cascade fit + std::vector<const xAOD::Vertex*> selectedD0Candidates; + for(auto vxcItr=d0Container->cbegin(); vxcItr!=d0Container->cend(); ++vxcItr) { + + // Check the passed flag first + const xAOD::Vertex* vtx = *vxcItr; + SG::AuxElement::Accessor<Char_t> flagAcc1("passed_D0"); + SG::AuxElement::Accessor<Char_t> flagAcc2("passed_D0b"); + bool isD0(true); + bool isD0b(true); + if(flagAcc1.isAvailable(*vtx)){ + if(!flagAcc1(*vtx)) isD0 = false; + } + if(flagAcc2.isAvailable(*vtx)){ + if(!flagAcc2(*vtx)) isD0b = false; + } + if(!(isD0||isD0b)) continue; + + // Ensure the total charge is correct + if ((*vxcItr)->trackParticle(0)->charge() != 1 || (*vxcItr)->trackParticle(1)->charge() != -1) { + ATH_MSG_DEBUG(" Original D0/D0-bar candidate rejected by the charge requirement: " + << (*vxcItr)->trackParticle(0)->charge() << ", " << (*vxcItr)->trackParticle(1)->charge() ); + continue; + } + + // Check D0/D0bar candidate invariant mass and skip if need be + double mass_D0 = m_V0Tools->invariantMass(*vxcItr,massesD0); + double mass_D0b = m_V0Tools->invariantMass(*vxcItr,massesD0b); + ATH_MSG_DEBUG("D0 mass " << mass_D0 << ", D0b mass "<<mass_D0b); + if ((mass_D0 < m_D0MassLower || mass_D0 > m_D0MassUpper) && (mass_D0b < m_D0MassLower || mass_D0b > m_D0MassUpper)) { + ATH_MSG_DEBUG(" Original D0 candidate rejected by the mass cut: mass = " + << mass_D0 << " != (" << m_D0MassLower << ", " << m_D0MassUpper << ") " + << mass_D0b << " != (" << m_D0MassLower << ", " << m_D0MassUpper << ") " ); + continue; + } + + selectedD0Candidates.push_back(*vxcItr); + } + if(selectedD0Candidates.size()<1) return StatusCode::SUCCESS; + + // Select J/psi D*+ candidates + // Iterate over Jpsi+pi vertices + for(auto jpsipiItr=selectedJpsipiCandidates.cbegin(); jpsipiItr!=selectedJpsipiCandidates.cend(); ++jpsipiItr) { + + size_t jpsipiTrkNum = (*jpsipiItr)->nTrackParticles(); + tracksJpsipi.clear(); + tracksJpsi.clear(); + for( unsigned int it=0; it<jpsipiTrkNum; it++) tracksJpsipi.push_back((*jpsipiItr)->trackParticle(it)); + for( unsigned int it=0; it<jpsipiTrkNum-1; it++) tracksJpsi.push_back((*jpsipiItr)->trackParticle(it)); + + if (tracksJpsipi.size() != 3 || massesJpsipi.size() != 3 ) { + ATH_MSG_INFO("problems with Jpsi+pi input"); + } + + bool tagD0(true); + if(abs(m_Dx_pid)==421 && (*jpsipiItr)->trackParticle(2)->charge()==-1) tagD0 = false; + + TLorentzVector p4_pi1; // Momentum of soft pion + p4_pi1.SetPtEtaPhiM((*jpsipiItr)->trackParticle(2)->pt(), + (*jpsipiItr)->trackParticle(2)->eta(), + (*jpsipiItr)->trackParticle(2)->phi(), m_vtx0Daug3MassHypo); + + // Iterate over D0/D0bar vertices + for(auto d0Itr=selectedD0Candidates.cbegin(); d0Itr!=selectedD0Candidates.cend(); ++d0Itr) { + + // Check identical tracks in input + if(std::find(tracksJpsipi.cbegin(), tracksJpsipi.cend(), (*d0Itr)->trackParticle(0)) != tracksJpsipi.cend()) continue; + if(std::find(tracksJpsipi.cbegin(), tracksJpsipi.cend(), (*d0Itr)->trackParticle(1)) != tracksJpsipi.cend()) continue; + + + TLorentzVector p4_ka, p4_pi2; + if(tagD0){ // for D*+ + p4_pi2.SetPtEtaPhiM((*d0Itr)->trackParticle(0)->pt(), + (*d0Itr)->trackParticle(0)->eta(), + (*d0Itr)->trackParticle(0)->phi(), m_vtx1Daug1MassHypo); + p4_ka.SetPtEtaPhiM( (*d0Itr)->trackParticle(1)->pt(), + (*d0Itr)->trackParticle(1)->eta(), + (*d0Itr)->trackParticle(1)->phi(), m_vtx1Daug2MassHypo); + }else{ // change the order in the case of D*- + p4_pi2.SetPtEtaPhiM((*d0Itr)->trackParticle(1)->pt(), + (*d0Itr)->trackParticle(1)->eta(), + (*d0Itr)->trackParticle(1)->phi(), m_vtx1Daug1MassHypo); + p4_ka.SetPtEtaPhiM( (*d0Itr)->trackParticle(0)->pt(), + (*d0Itr)->trackParticle(0)->eta(), + (*d0Itr)->trackParticle(0)->phi(), m_vtx1Daug2MassHypo); + } + // Check D*+/- candidate invariant mass and skip if need be + double mass_Dst= (p4_pi1 + p4_ka + p4_pi2).M(); + ATH_MSG_DEBUG("D*+/- mass " << mass_Dst); + if (mass_Dst < m_DstMassLower || mass_Dst > m_DstMassUpper) { + ATH_MSG_DEBUG(" Original D*+/- candidate rejected by the mass cut: mass = " + << mass_Dst << " != (" << m_DstMassLower << ", " << m_DstMassUpper << ")" ); + continue; + } + + size_t d0TrkNum = (*d0Itr)->nTrackParticles(); + tracksD0.clear(); + for( unsigned int it=0; it<d0TrkNum; it++) tracksD0.push_back((*d0Itr)->trackParticle(it)); + if (tracksD0.size() != 2 || massesD0.size() != 2 ) { + ATH_MSG_INFO("problems with D0 input"); + } + + ATH_MSG_DEBUG("using tracks" << tracksJpsipi[0] << ", " << tracksJpsipi[1] << ", " << tracksJpsipi[2] << ", " << tracksD0[0] << ", " << tracksD0[1]); + ATH_MSG_DEBUG("Charge of Jpsi+pi tracks: "<<(*jpsipiItr)->trackParticle(0)->charge()<<", "<<(*jpsipiItr)->trackParticle(1)->charge()<<", "<<(*jpsipiItr)->trackParticle(2)->charge()); + ATH_MSG_DEBUG("Charge of D0 tracks: "<<(*d0Itr)->trackParticle(0)->charge()<<", "<<(*d0Itr)->trackParticle(1)->charge()); + + tracksBc.clear(); + for( unsigned int it=0; it<jpsipiTrkNum; it++) tracksBc.push_back((*jpsipiItr)->trackParticle(it)); + for( unsigned int it=0; it<d0TrkNum; it++) tracksBc.push_back((*d0Itr)->trackParticle(it)); + + + // Apply the user's settings to the fitter + // Reset + std::unique_ptr<Trk::IVKalState> state (m_iVertexFitter->makeState()); + // Robustness + int robustness = 0; + m_iVertexFitter->setRobustness(robustness, *state); + // Build up the topology + // Vertex list + std::vector<Trk::VertexID> vrtList; + // D0 vertex + Trk::VertexID vID; + if (m_constrD0) { + if(tagD0) vID = m_iVertexFitter->startVertex(tracksD0,massesD0,*state,mass_d0); + else vID = m_iVertexFitter->startVertex(tracksD0,massesD0b,*state,mass_d0); + } else { + if(tagD0) vID = m_iVertexFitter->startVertex(tracksD0,massesD0,*state); + else vID = m_iVertexFitter->startVertex(tracksD0,massesD0b,*state); + } + vrtList.push_back(vID); + // B vertex including Jpsi+pi + Trk::VertexID vID2 = m_iVertexFitter->nextVertex(tracksJpsipi,massesJpsipi,vrtList,*state); + if (m_constrJpsi) { + std::vector<Trk::VertexID> cnstV; + cnstV.clear(); + if ( !m_iVertexFitter->addMassConstraint(vID2,tracksJpsi,cnstV,*state,m_mass_jpsi).isSuccess() ) { + ATH_MSG_WARNING("addMassConstraint failed"); + //return StatusCode::FAILURE; + } + } + // Do the work + std::unique_ptr<Trk::VxCascadeInfo> result(m_iVertexFitter->fitCascade(*state)); + + if (result != nullptr) { + + // reset links to original tracks + BPhysPVCascadeTools::PrepareVertexLinks(result.get(), trackContainer); + ATH_MSG_DEBUG("storing tracks " << ((result->vertices())[0])->trackParticle(0) << ", " + << ((result->vertices())[0])->trackParticle(1) << ", " + << ((result->vertices())[1])->trackParticle(0) << ", " + << ((result->vertices())[1])->trackParticle(1) << ", " + << ((result->vertices())[1])->trackParticle(2)); + // necessary to prevent memory leak + result->setSVOwnership(true); + + // Chi2/DOF cut + double bChi2DOF = result->fitChi2()/result->nDoF(); + ATH_MSG_DEBUG("Candidate chi2/DOF is " << bChi2DOF); + bool chi2CutPassed = (m_chi2cut <= 0.0 || bChi2DOF < m_chi2cut); + + const std::vector< std::vector<TLorentzVector> > &moms = result->getParticleMoms(); + double mass = m_CascadeTools->invariantMass(moms[1]); + if(chi2CutPassed) { + if (mass >= m_MassLower && mass <= m_MassUpper) { + cascadeinfoContainer->push_back(result.release()); + } else { + ATH_MSG_DEBUG("Candidate rejected by the mass cut: mass = " + << mass << " != (" << m_MassLower << ", " << m_MassUpper << ")" ); + } + } + } + + } //Iterate over D0 vertices + + } //Iterate over Jpsi+pi vertices + + ATH_MSG_DEBUG("cascadeinfoContainer size " << cascadeinfoContainer->size()); + + return StatusCode::SUCCESS; + } + +} diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/JpsiPlusDs1Cascade.cxx b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/JpsiPlusDs1Cascade.cxx new file mode 100644 index 00000000000..e49a95911bc --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/JpsiPlusDs1Cascade.cxx @@ -0,0 +1,905 @@ +/* + Copyright (C) 2002-2018 CERN for the benefit of the ATLAS collaboration +*/ +///////////////////////////////////////////////////////////////// +// JpsiPlusDs1Cascade.cxx, (c) ATLAS Detector software +///////////////////////////////////////////////////////////////// +#include "DerivationFrameworkBPhys/JpsiPlusDs1Cascade.h" +#include "TrkVertexFitterInterfaces/IVertexFitter.h" +#include "TrkVKalVrtFitter/TrkVKalVrtFitter.h" +#include "TrkVertexAnalysisUtils/V0Tools.h" +#include "GaudiKernel/IPartPropSvc.h" +#include "DerivationFrameworkBPhys/CascadeTools.h" +#include "DerivationFrameworkBPhys/BPhysPVCascadeTools.h" +#include "xAODTracking/VertexAuxContainer.h" +#include "xAODBPhys/BPhysHypoHelper.h" +#include <algorithm> +#include "xAODTracking/VertexContainer.h" +#include "DerivationFrameworkBPhys/LocalVector.h" + +namespace DerivationFramework { + typedef ElementLink<xAOD::VertexContainer> VertexLink; + typedef std::vector<VertexLink> VertexLinkVector; + typedef std::vector<const xAOD::TrackParticle*> TrackBag; + + double JpsiPlusDs1Cascade::getParticleMass(int pdgcode) const{ + auto ptr = m_particleDataTable->particle( pdgcode ); + return ptr ? ptr->mass() : 0.; + } + + StatusCode JpsiPlusDs1Cascade::initialize() { + + // retrieving vertex Fitter + ATH_CHECK( m_iVertexFitter.retrieve()); + + // retrieving the V0 tools + ATH_CHECK( m_V0Tools.retrieve()); + + // retrieving the Cascade tools + ATH_CHECK( m_CascadeTools.retrieve()); + + // Get the beam spot service + ATH_CHECK(m_beamSpotKey.initialize()); + + IPartPropSvc* partPropSvc = nullptr; + ATH_CHECK( service("PartPropSvc", partPropSvc, true) ); + m_particleDataTable = partPropSvc->PDT(); + + // retrieve particle masses + if(m_mass_jpsi < 0. ) m_mass_jpsi = getParticleMass(PDG::J_psi); + if(m_vtx0MassHypo < 0.) m_vtx0MassHypo = getParticleMass(PDG::B_c_plus); + if(m_vtx1MassHypo < 0.) m_vtx1MassHypo = getParticleMass(PDG::D0); + if(m_vtx2MassHypo < 0.) m_vtx2MassHypo = getParticleMass(PDG::K_S0); + + if(m_vtx0Daug1MassHypo < 0.) m_vtx0Daug1MassHypo = getParticleMass(PDG::mu_minus); + if(m_vtx0Daug2MassHypo < 0.) m_vtx0Daug2MassHypo = getParticleMass(PDG::mu_minus); + if(m_vtx0Daug3MassHypo < 0.) m_vtx0Daug3MassHypo = getParticleMass(PDG::pi_plus); + if(m_vtx1Daug1MassHypo < 0.) m_vtx1Daug1MassHypo = getParticleMass(PDG::pi_plus); + if(m_vtx1Daug2MassHypo < 0.) m_vtx1Daug2MassHypo = getParticleMass(PDG::K_plus); + if(m_vtx2Daug1MassHypo < 0.) m_vtx2Daug1MassHypo = getParticleMass(PDG::pi_plus); + if(m_vtx2Daug2MassHypo < 0.) m_vtx2Daug2MassHypo = getParticleMass(PDG::pi_plus); + + return StatusCode::SUCCESS; + } + + + StatusCode JpsiPlusDs1Cascade::addBranches() const + { + std::vector<Trk::VxCascadeInfo*> cascadeinfoContainer; + constexpr int topoN = 3; + std::array<xAOD::VertexContainer*, topoN> Vtxwritehandles; + std::array<xAOD::VertexAuxContainer*, topoN> Vtxwritehandlesaux; + if(m_cascadeOutputsKeys.size() !=topoN) { ATH_MSG_FATAL("Incorrect number of VtxContainers"); return StatusCode::FAILURE; } + + for(int i =0; i<topoN;i++){ + Vtxwritehandles[i] = new xAOD::VertexContainer(); + Vtxwritehandlesaux[i] = new xAOD::VertexAuxContainer(); + Vtxwritehandles[i]->setStore(Vtxwritehandlesaux[i]); + ATH_CHECK(evtStore()->record(Vtxwritehandles[i] , m_cascadeOutputsKeys[i] )); + ATH_CHECK(evtStore()->record(Vtxwritehandlesaux[i], m_cascadeOutputsKeys[i] + "Aux.")); + } + + //---------------------------------------------------- + // retrieve primary vertices + //---------------------------------------------------- + const xAOD::Vertex * primaryVertex(nullptr); + const xAOD::VertexContainer *pvContainer(nullptr); + ATH_CHECK(evtStore()->retrieve(pvContainer, m_VxPrimaryCandidateName)); + ATH_MSG_DEBUG("Found " << m_VxPrimaryCandidateName << " in StoreGate!"); + + if (pvContainer->size()==0){ + ATH_MSG_WARNING("You have no primary vertices: " << pvContainer->size()); + return StatusCode::RECOVERABLE; + } else { + primaryVertex = (*pvContainer)[0]; + } + + //---------------------------------------------------- + // Try to retrieve refitted primary vertices + //---------------------------------------------------- + xAOD::VertexContainer* refPvContainer = nullptr; + xAOD::VertexAuxContainer* refPvAuxContainer = nullptr; + if (m_refitPV) { + if (evtStore()->contains<xAOD::VertexContainer>(m_refPVContainerName)) { + // refitted PV container exists. Get it from the store gate + ATH_CHECK(evtStore()->retrieve(refPvContainer , m_refPVContainerName )); + ATH_CHECK(evtStore()->retrieve(refPvAuxContainer, m_refPVContainerName + "Aux.")); + } else { + // refitted PV container does not exist. Create a new one. + refPvContainer = new xAOD::VertexContainer; + refPvAuxContainer = new xAOD::VertexAuxContainer; + refPvContainer->setStore(refPvAuxContainer); + ATH_CHECK(evtStore()->record(refPvContainer , m_refPVContainerName)); + ATH_CHECK(evtStore()->record(refPvAuxContainer, m_refPVContainerName+"Aux.")); + } + } + + ATH_CHECK(performSearch(&cascadeinfoContainer)); + + SG::ReadCondHandle<InDet::BeamSpotData> beamSpotHandle { m_beamSpotKey }; + if(not beamSpotHandle.isValid()) ATH_MSG_ERROR("Cannot Retrieve " << m_beamSpotKey.key() ); + BPhysPVCascadeTools helper(&(*m_CascadeTools), beamSpotHandle.cptr()); + helper.SetMinNTracksInPV(m_PV_minNTracks); + + // Decorators for the main vertex: chi2, ndf, pt and pt error, plus the D0, K0 vertex variables + SG::AuxElement::Decorator<VertexLinkVector> CascadeV1LinksDecor("CascadeVertex1Links"); + SG::AuxElement::Decorator<VertexLinkVector> CascadeV2LinksDecor("CascadeVertex2Links"); + SG::AuxElement::Decorator<VertexLinkVector> JpsipiLinksDecor("JpsipiVertexLinks"); + SG::AuxElement::Decorator<VertexLinkVector> D0LinksDecor("D0VertexLinks"); + SG::AuxElement::Decorator<VertexLinkVector> K0LinksDecor("K0VertexLinks"); + SG::AuxElement::Decorator<float> chi2_decor("ChiSquared"); + SG::AuxElement::Decorator<float> ndof_decor("NumberDoF"); + SG::AuxElement::Decorator<float> Pt_decor("Pt"); + SG::AuxElement::Decorator<float> PtErr_decor("PtErr"); + SG::AuxElement::Decorator<float> Mass_svdecor("D0_mass"); + SG::AuxElement::Decorator<float> MassErr_svdecor("D0_massErr"); + SG::AuxElement::Decorator<float> Pt_svdecor("D0_Pt"); + SG::AuxElement::Decorator<float> PtErr_svdecor("D0_PtErr"); + SG::AuxElement::Decorator<float> Lxy_svdecor("D0_Lxy"); + SG::AuxElement::Decorator<float> LxyErr_svdecor("D0_LxyErr"); + SG::AuxElement::Decorator<float> Tau_svdecor("D0_Tau"); + SG::AuxElement::Decorator<float> TauErr_svdecor("D0_TauErr"); + + SG::AuxElement::Decorator<float> Mass_sv2decor("K0_mass"); + SG::AuxElement::Decorator<float> MassErr_sv2decor("K0_massErr"); + SG::AuxElement::Decorator<float> Pt_sv2decor("K0_Pt"); + SG::AuxElement::Decorator<float> PtErr_sv2decor("K0_PtErr"); + SG::AuxElement::Decorator<float> Lxy_sv2decor("K0_Lxy"); + SG::AuxElement::Decorator<float> LxyErr_sv2decor("K0_LxyErr"); + SG::AuxElement::Decorator<float> Tau_sv2decor("K0_Tau"); + SG::AuxElement::Decorator<float> TauErr_sv2decor("K0_TauErr"); + + SG::AuxElement::Decorator<float> MassJpsi_decor("Jpsi_mass"); + SG::AuxElement::Decorator<float> MassPiD0_decor("PiD0_mass"); + SG::AuxElement::Decorator<float> MassPiD0K0_decor("PiD0K0_mass"); + + SG::AuxElement::Decorator<float> MassMumu_decor("Mumu_mass"); + SG::AuxElement::Decorator<float> MassKpi_svdecor("Kpi_mass"); + SG::AuxElement::Decorator<float> MassPipi_sv2decor("Pipi_mass"); + + ATH_MSG_DEBUG("cascadeinfoContainer size " << cascadeinfoContainer.size()); + + // Get Jpsi+pi container and identify the input Jpsi+pi + const xAOD::VertexContainer *jpsipiContainer(nullptr); + ATH_CHECK(evtStore()->retrieve(jpsipiContainer , m_vertexContainerKey )); + // Get D0 container and identify the input D0 + const xAOD::VertexContainer *d0Container(nullptr); + ATH_CHECK(evtStore()->retrieve(d0Container , m_vertexD0ContainerKey )); + // Get K0 container and identify the input K0 + const xAOD::VertexContainer *k0Container(nullptr); + ATH_CHECK(evtStore()->retrieve(k0Container , m_vertexK0ContainerKey )); + + for (Trk::VxCascadeInfo* x : cascadeinfoContainer) { + if(x==nullptr) ATH_MSG_ERROR("cascadeinfoContainer is null"); + + // the cascade fitter returns: + // std::vector<xAOD::Vertex*>, each xAOD::Vertex contains the refitted track parameters (perigee at the vertex position) + // vertices[iv] the links to the original TPs and a covariance of size 3+5*NTRK; the chi2 of the total fit + // is split between the cascade vertices as per track contribution + // std::vector< std::vector<TLorentzVector> >, each std::vector<TLorentzVector> contains the refitted momenta (TLorentzVector) + // momenta[iv][...] of all tracks in the corresponding vertex, including any pseudotracks (from cascade vertices) + // originating in this vertex; the masses are as assigned in the cascade fit + // std::vector<Amg::MatrixX>, the corresponding covariance matrices in momentum space + // covariance[iv] + // int nDoF, double Chi2 + // + // the invariant mass, pt, lifetime etc. errors should be calculated using the covariance matrices in momentum space as these + // take into account the full track-track and track-vertex correlations + // + // in the case of Jpsi+V0: vertices[0] is the V0 vertex, vertices[1] is the B/Lambda_b(bar) vertex, containing the 2 Jpsi tracks. + // The covariance terms between the two vertices are not stored. In momentum space momenta[0] contains the 2 V0 tracks, + // their momenta add up to the momentum of the 3rd track in momenta[1], the first two being the Jpsi tracks + + const std::vector<xAOD::Vertex*> &cascadeVertices = x->vertices(); + if(cascadeVertices.size()!=topoN) + ATH_MSG_ERROR("Incorrect number of vertices"); + if(cascadeVertices[0] == nullptr || cascadeVertices[1] == nullptr || cascadeVertices[2] == nullptr) ATH_MSG_ERROR("Error null vertex"); + // Keep vertices (bear in mind that they come in reverse order!) + for(int i =0;i<topoN;i++) Vtxwritehandles[i]->push_back(cascadeVertices[i]); + + x->setSVOwnership(false); // Prevent Container from deleting vertices + const auto mainVertex = cascadeVertices[2]; // this is the B_c+/- vertex + const std::vector< std::vector<TLorentzVector> > &moms = x->getParticleMoms(); + + // Set links to cascade vertices + std::vector<const xAOD::Vertex*> verticestoLink; + verticestoLink.push_back(cascadeVertices[0]); + //if(Vtxwritehandles[1] == nullptr) ATH_MSG_ERROR("Vtxwritehandles[1] is null"); + if(Vtxwritehandles[2] == nullptr) ATH_MSG_ERROR("Vtxwritehandles[2] is null"); + if(!BPhysPVCascadeTools::LinkVertices(CascadeV1LinksDecor, verticestoLink, Vtxwritehandles[0], cascadeVertices[2])) + ATH_MSG_ERROR("Error decorating with cascade vertices"); + + verticestoLink.clear(); + verticestoLink.push_back(cascadeVertices[1]); + if(!BPhysPVCascadeTools::LinkVertices(CascadeV2LinksDecor, verticestoLink, Vtxwritehandles[1], cascadeVertices[2])) + ATH_MSG_ERROR("Error decorating with cascade vertices"); + + // Identify the input Jpsi+pi + const xAOD::Vertex* jpsipiVertex = BPhysPVCascadeTools::FindVertex<3>(jpsipiContainer, cascadeVertices[2]); + ATH_MSG_DEBUG("1 pt Jpsi+pi tracks " << cascadeVertices[2]->trackParticle(0)->pt() << ", " << cascadeVertices[2]->trackParticle(1)->pt() << ", " << cascadeVertices[2]->trackParticle(2)->pt()); + if (jpsipiVertex) ATH_MSG_DEBUG("2 pt Jpsi+pi tracks " << jpsipiVertex->trackParticle(0)->pt() << ", " << jpsipiVertex->trackParticle(1)->pt() << ", " << jpsipiVertex->trackParticle(2)->pt()); + + // Identify the input D0 + const xAOD::Vertex* d0Vertex = BPhysPVCascadeTools::FindVertex<2>(d0Container, cascadeVertices[1]);; + ATH_MSG_DEBUG("1 pt D0 tracks " << cascadeVertices[1]->trackParticle(0)->pt() << ", " << cascadeVertices[1]->trackParticle(1)->pt()); + if (d0Vertex) ATH_MSG_DEBUG("2 pt D0 tracks " << d0Vertex->trackParticle(0)->pt() << ", " << d0Vertex->trackParticle(1)->pt()); + + // Identify the input K_S0 + const xAOD::Vertex* k0Vertex = BPhysPVCascadeTools::FindVertex<2>(k0Container, cascadeVertices[0]);; + ATH_MSG_DEBUG("1 pt K_S0 tracks " << cascadeVertices[0]->trackParticle(0)->pt() << ", " << cascadeVertices[0]->trackParticle(1)->pt()); + if (k0Vertex) ATH_MSG_DEBUG("2 pt K_S0 tracks " << k0Vertex->trackParticle(0)->pt() << ", " << k0Vertex->trackParticle(1)->pt()); + + // Set links to input vertices + std::vector<const xAOD::Vertex*> jpsipiVerticestoLink; + if (jpsipiVertex) jpsipiVerticestoLink.push_back(jpsipiVertex); + else ATH_MSG_WARNING("Could not find linking Jpsi+pi"); + if(!BPhysPVCascadeTools::LinkVertices(JpsipiLinksDecor, jpsipiVerticestoLink, jpsipiContainer, cascadeVertices[2])) + ATH_MSG_ERROR("Error decorating with Jpsi+pi vertices"); + + std::vector<const xAOD::Vertex*> d0VerticestoLink; + if (d0Vertex) d0VerticestoLink.push_back(d0Vertex); + else ATH_MSG_WARNING("Could not find linking D0"); + if(!BPhysPVCascadeTools::LinkVertices(D0LinksDecor, d0VerticestoLink, d0Container, cascadeVertices[2])) + ATH_MSG_ERROR("Error decorating with D0 vertices"); + + std::vector<const xAOD::Vertex*> k0VerticestoLink; + if (k0Vertex) k0VerticestoLink.push_back(k0Vertex); + else ATH_MSG_WARNING("Could not find linking K_S0"); + if(!BPhysPVCascadeTools::LinkVertices(K0LinksDecor, k0VerticestoLink, k0Container, cascadeVertices[2])) + ATH_MSG_ERROR("Error decorating with K_S0 vertices"); + + bool tagD0(true); + if (jpsipiVertex){ + if(abs(m_Dx_pid)==421 && (jpsipiVertex->trackParticle(2)->charge()==-1)) tagD0 = false; + } + + double mass_b = m_vtx0MassHypo; + double mass_d0 = m_vtx1MassHypo; + double mass_k0 = m_vtx2MassHypo; + std::vector<double> massesJpsipi; + massesJpsipi.push_back(m_vtx0Daug1MassHypo); + massesJpsipi.push_back(m_vtx0Daug2MassHypo); + massesJpsipi.push_back(m_vtx0Daug3MassHypo); + std::vector<double> massesD0; + if(tagD0){ + massesD0.push_back(m_vtx1Daug1MassHypo); + massesD0.push_back(m_vtx1Daug2MassHypo); + }else{ // Change the oreder of masses for D*-->D0bar pi-, D0bar->K+pi- + massesD0.push_back(m_vtx1Daug2MassHypo); + massesD0.push_back(m_vtx1Daug1MassHypo); + } + std::vector<double> massesK0; + massesK0.push_back(m_vtx2Daug1MassHypo); + massesK0.push_back(m_vtx2Daug2MassHypo); + std::vector<double> Masses; + Masses.push_back(m_vtx0Daug1MassHypo); + Masses.push_back(m_vtx0Daug2MassHypo); + Masses.push_back(m_vtx0Daug3MassHypo); + Masses.push_back(m_vtx1MassHypo); + Masses.push_back(m_vtx2MassHypo); + + // loop over candidates -- Don't apply PV_minNTracks requirement here + // because it may result in exclusion of the high-pt PV. + // get good PVs + + xAOD::BPhysHypoHelper vtx(m_hypoName, mainVertex); + + BPhysPVCascadeTools::SetVectorInfo(vtx, x); + + // Decorate main vertex + // + // 1.a) mass, mass error + BPHYS_CHECK( vtx.setMass(m_CascadeTools->invariantMass(moms[2])) ); + BPHYS_CHECK( vtx.setMassErr(m_CascadeTools->invariantMassError(moms[2],x->getCovariance()[2])) ); + // 1.b) pt and pT error (the default pt of mainVertex is != the pt of the full cascade fit!) + Pt_decor(*mainVertex) = m_CascadeTools->pT(moms[2]); + PtErr_decor(*mainVertex) = m_CascadeTools->pTError(moms[2],x->getCovariance()[2]); + // 1.c) chi2 and ndof (the default chi2 of mainVertex is != the chi2 of the full cascade fit!) + chi2_decor(*mainVertex) = x->fitChi2(); + ndof_decor(*mainVertex) = x->nDoF(); + + float massMumu = 0.; + if (jpsipiVertex) { + TLorentzVector p4_mu1, p4_mu2; + p4_mu1.SetPtEtaPhiM(jpsipiVertex->trackParticle(0)->pt(), + jpsipiVertex->trackParticle(0)->eta(), + jpsipiVertex->trackParticle(0)->phi(), m_vtx0Daug1MassHypo); + p4_mu2.SetPtEtaPhiM(jpsipiVertex->trackParticle(1)->pt(), + jpsipiVertex->trackParticle(1)->eta(), + jpsipiVertex->trackParticle(1)->phi(), m_vtx0Daug2MassHypo); + massMumu = (p4_mu1 + p4_mu2).M(); + } + MassMumu_decor(*mainVertex) = massMumu; + + float massKpi = 0.; + if (d0Vertex) { + TLorentzVector p4_ka, p4_pi; + if(tagD0){ + p4_pi.SetPtEtaPhiM(d0Vertex->trackParticle(0)->pt(), + d0Vertex->trackParticle(0)->eta(), + d0Vertex->trackParticle(0)->phi(), m_vtx1Daug1MassHypo); + p4_ka.SetPtEtaPhiM(d0Vertex->trackParticle(1)->pt(), + d0Vertex->trackParticle(1)->eta(), + d0Vertex->trackParticle(1)->phi(), m_vtx1Daug2MassHypo); + }else{ // Change the oreder of masses for D*-->D0bar pi-, D0bar->K+pi- + p4_pi.SetPtEtaPhiM(d0Vertex->trackParticle(1)->pt(), + d0Vertex->trackParticle(1)->eta(), + d0Vertex->trackParticle(1)->phi(), m_vtx1Daug1MassHypo); + p4_ka.SetPtEtaPhiM(d0Vertex->trackParticle(0)->pt(), + d0Vertex->trackParticle(0)->eta(), + d0Vertex->trackParticle(0)->phi(), m_vtx1Daug2MassHypo); + } + massKpi = (p4_ka + p4_pi).M(); + } + MassKpi_svdecor(*mainVertex) = massKpi; + + float massPipi = 0.; + if (k0Vertex) { + TLorentzVector p4_pip, p4_pim; + p4_pip.SetPtEtaPhiM(k0Vertex->trackParticle(0)->pt(), + k0Vertex->trackParticle(0)->eta(), + k0Vertex->trackParticle(0)->phi(), m_vtx2Daug1MassHypo); + p4_pim.SetPtEtaPhiM(k0Vertex->trackParticle(1)->pt(), + k0Vertex->trackParticle(1)->eta(), + k0Vertex->trackParticle(1)->phi(), m_vtx2Daug2MassHypo); + massPipi = (p4_pip + p4_pim).M(); + } + MassPipi_sv2decor(*mainVertex) = massPipi; + + MassJpsi_decor(*mainVertex) = (moms[2][0] + moms[2][1]).M(); + MassPiD0_decor(*mainVertex) = (moms[2][2] + moms[2][4]).M(); + MassPiD0K0_decor(*mainVertex) = (moms[2][2] + moms[2][4] + moms[2][3]).M(); + + ATH_CHECK(helper.FillCandwithRefittedVertices(m_refitPV, pvContainer, + refPvContainer, &(*m_pvRefitter), m_PV_max, m_DoVertexType, x, 2, mass_b, vtx)); + + // 4) decorate the main vertex with D0 vertex mass, pt, lifetime and lxy values (plus errors) + // D0 points to the main vertex, so lifetime and lxy are w.r.t the main vertex + Mass_svdecor(*mainVertex) = m_CascadeTools->invariantMass(moms[1]); + MassErr_svdecor(*mainVertex) = m_CascadeTools->invariantMassError(moms[1],x->getCovariance()[1]); + Pt_svdecor(*mainVertex) = m_CascadeTools->pT(moms[1]); + PtErr_svdecor(*mainVertex) = m_CascadeTools->pTError(moms[1],x->getCovariance()[1]); + Lxy_svdecor(*mainVertex) = m_CascadeTools->lxy(moms[1],cascadeVertices[1],cascadeVertices[2]); + LxyErr_svdecor(*mainVertex) = m_CascadeTools->lxyError(moms[1],x->getCovariance()[1],cascadeVertices[1],cascadeVertices[2]); + Tau_svdecor(*mainVertex) = m_CascadeTools->tau(moms[1],cascadeVertices[1],cascadeVertices[2]); + TauErr_svdecor(*mainVertex) = m_CascadeTools->tauError(moms[1],x->getCovariance()[1],cascadeVertices[1],cascadeVertices[2]); + + // 5) decorate the main vertex with K_S0 vertex mass, pt, lifetime and lxy values (plus errors) + // K_S0 points to the main vertex, so lifetime and lxy are w.r.t the main vertex + Mass_sv2decor(*mainVertex) = m_CascadeTools->invariantMass(moms[0]); + MassErr_sv2decor(*mainVertex) = m_CascadeTools->invariantMassError(moms[0],x->getCovariance()[0]); + Pt_sv2decor(*mainVertex) = m_CascadeTools->pT(moms[0]); + PtErr_sv2decor(*mainVertex) = m_CascadeTools->pTError(moms[0],x->getCovariance()[0]); + Lxy_sv2decor(*mainVertex) = m_CascadeTools->lxy(moms[0],cascadeVertices[0],cascadeVertices[2]); + LxyErr_sv2decor(*mainVertex) = m_CascadeTools->lxyError(moms[0],x->getCovariance()[0],cascadeVertices[0],cascadeVertices[2]); + Tau_sv2decor(*mainVertex) = m_CascadeTools->tau(moms[0],cascadeVertices[0],cascadeVertices[2]); + TauErr_sv2decor(*mainVertex) = m_CascadeTools->tauError(moms[0],x->getCovariance()[0],cascadeVertices[0],cascadeVertices[2]); + + // Some checks in DEBUG mode + ATH_MSG_DEBUG("chi2 " << x->fitChi2() + << " chi2_1 " << m_V0Tools->chisq(cascadeVertices[0]) + << " chi2_2 " << m_V0Tools->chisq(cascadeVertices[1]) + << " chi2_3 " << m_V0Tools->chisq(cascadeVertices[2]) + << " vprob " << m_CascadeTools->vertexProbability(x->nDoF(),x->fitChi2())); + ATH_MSG_DEBUG("ndf " << x->nDoF() << " ndf_1 " << m_V0Tools->ndof(cascadeVertices[0]) << " ndf_2 " << m_V0Tools->ndof(cascadeVertices[1]) << " ndf_3 " << m_V0Tools->ndof(cascadeVertices[2])); + ATH_MSG_DEBUG("V0Tools mass_k0 " << m_V0Tools->invariantMass(cascadeVertices[0],massesK0) + << " error " << m_V0Tools->invariantMassError(cascadeVertices[0],massesK0) + << " mass_d0 " << m_V0Tools->invariantMass(cascadeVertices[1],massesD0) + << " error " << m_V0Tools->invariantMassError(cascadeVertices[1],massesD0) + << " mass_J " << m_V0Tools->invariantMass(cascadeVertices[2],massesJpsipi) + << " error " << m_V0Tools->invariantMassError(cascadeVertices[2],massesJpsipi)); + // masses and errors, using track masses assigned in the fit + double Mass_B = m_CascadeTools->invariantMass(moms[2]); + double Mass_D0 = m_CascadeTools->invariantMass(moms[1]); + double Mass_K0 = m_CascadeTools->invariantMass(moms[0]); + double Mass_B_err = m_CascadeTools->invariantMassError(moms[2],x->getCovariance()[2]); + double Mass_D0_err = m_CascadeTools->invariantMassError(moms[1],x->getCovariance()[1]); + double Mass_K0_err = m_CascadeTools->invariantMassError(moms[0],x->getCovariance()[0]); + ATH_MSG_DEBUG("Mass_B " << Mass_B << " Mass_D0 " << Mass_D0 << " Mass_K0 " << Mass_K0); + ATH_MSG_DEBUG("Mass_B_err " << Mass_B_err << " Mass_D0_err " << Mass_D0_err << " Mass_K0_err " << Mass_K0_err); + double mprob_B = m_CascadeTools->massProbability(mass_b,Mass_B,Mass_B_err); + double mprob_D0 = m_CascadeTools->massProbability(mass_d0,Mass_D0,Mass_D0_err); + double mprob_K0 = m_CascadeTools->massProbability(mass_k0,Mass_K0,Mass_K0_err); + ATH_MSG_DEBUG("mprob_B " << mprob_B << " mprob_D0 " << mprob_D0 << " mprob_K0 " << mprob_K0); + // masses and errors, assigning user defined track masses + ATH_MSG_DEBUG("Mass_b " << m_CascadeTools->invariantMass(moms[2],Masses) + << " Mass_d0 " << m_CascadeTools->invariantMass(moms[1],massesD0) + << " Mass_k0 " << m_CascadeTools->invariantMass(moms[0],massesD0)); + ATH_MSG_DEBUG("Mass_b_err " << m_CascadeTools->invariantMassError(moms[2],x->getCovariance()[2],Masses) + << " Mass_d0_err " << m_CascadeTools->invariantMassError(moms[1],x->getCovariance()[1],massesD0) + << " Mass_k0_err " << m_CascadeTools->invariantMassError(moms[0],x->getCovariance()[0],massesK0)); + ATH_MSG_DEBUG("pt_b " << m_CascadeTools->pT(moms[2]) + << " pt_d " << m_CascadeTools->pT(moms[1]) + << " pt_d0 " << m_V0Tools->pT(cascadeVertices[1]) + << " pt_k " << m_CascadeTools->pT(moms[0]) + << " pt_k0 " << m_V0Tools->pT(cascadeVertices[0])); + ATH_MSG_DEBUG("ptErr_b " << m_CascadeTools->pTError(moms[2],x->getCovariance()[2]) + << " ptErr_d " << m_CascadeTools->pTError(moms[1],x->getCovariance()[1]) + << " ptErr_d0 " << m_V0Tools->pTError(cascadeVertices[1]) + << " ptErr_k " << m_CascadeTools->pTError(moms[0],x->getCovariance()[0]) + << " ptErr_k0 " << m_V0Tools->pTError(cascadeVertices[0])); + ATH_MSG_DEBUG("lxy_B " << m_V0Tools->lxy(cascadeVertices[2],primaryVertex) << " lxy_D " << m_V0Tools->lxy(cascadeVertices[1],cascadeVertices[2]) << " lxy_K " << m_V0Tools->lxy(cascadeVertices[0],cascadeVertices[2])); + ATH_MSG_DEBUG("lxy_b " << m_CascadeTools->lxy(moms[2],cascadeVertices[2],primaryVertex) << " lxy_d " << m_CascadeTools->lxy(moms[1],cascadeVertices[1],cascadeVertices[2]) << " lxy_k " << m_CascadeTools->lxy(moms[0],cascadeVertices[0],cascadeVertices[2])); + ATH_MSG_DEBUG("lxyErr_b " << m_CascadeTools->lxyError(moms[2],x->getCovariance()[2],cascadeVertices[2],primaryVertex) + << " lxyErr_d " << m_CascadeTools->lxyError(moms[1],x->getCovariance()[1],cascadeVertices[1],cascadeVertices[2]) + << " lxyErr_d0 " << m_V0Tools->lxyError(cascadeVertices[1],cascadeVertices[2]) + << " lxyErr_k " << m_CascadeTools->lxyError(moms[0],x->getCovariance()[0],cascadeVertices[0],cascadeVertices[2]) + << " lxyErr_k0 " << m_V0Tools->lxyError(cascadeVertices[0],cascadeVertices[2])); + ATH_MSG_DEBUG("tau_B " << m_CascadeTools->tau(moms[2],cascadeVertices[2],primaryVertex,mass_b) + << " tau_d0 " << m_V0Tools->tau(cascadeVertices[1],cascadeVertices[2],massesD0) + << " tau_k0 " << m_V0Tools->tau(cascadeVertices[0],cascadeVertices[2],massesK0)); + ATH_MSG_DEBUG("tau_b " << m_CascadeTools->tau(moms[2],cascadeVertices[2],primaryVertex) + << " tau_d " << m_CascadeTools->tau(moms[1],cascadeVertices[1],cascadeVertices[2]) + << " tau_D " << m_CascadeTools->tau(moms[1],cascadeVertices[1],cascadeVertices[2],mass_d0) + << " tau_k " << m_CascadeTools->tau(moms[0],cascadeVertices[0],cascadeVertices[2]) + << " tau_K " << m_CascadeTools->tau(moms[0],cascadeVertices[0],cascadeVertices[2],mass_k0)); + ATH_MSG_DEBUG("tauErr_b " << m_CascadeTools->tauError(moms[2],x->getCovariance()[2],cascadeVertices[2],primaryVertex) + << " tauErr_d " << m_CascadeTools->tauError(moms[1],x->getCovariance()[1],cascadeVertices[1],cascadeVertices[2]) + << " tauErr_d0 " << m_V0Tools->tauError(cascadeVertices[1],cascadeVertices[2],massesD0) + << " tauErr_k " << m_CascadeTools->tauError(moms[0],x->getCovariance()[0],cascadeVertices[0],cascadeVertices[2]) + << " tauErr_k0 " << m_V0Tools->tauError(cascadeVertices[0],cascadeVertices[2],massesK0)); + ATH_MSG_DEBUG("TauErr_b " << m_CascadeTools->tauError(moms[2],x->getCovariance()[2],cascadeVertices[2],primaryVertex,mass_b) + << " TauErr_d " << m_CascadeTools->tauError(moms[1],x->getCovariance()[1],cascadeVertices[1],cascadeVertices[2],mass_d0) + << " TauErr_d0 " << m_V0Tools->tauError(cascadeVertices[1],cascadeVertices[2],massesD0,mass_d0) + << " TauErr_k " << m_CascadeTools->tauError(moms[0],x->getCovariance()[0],cascadeVertices[0],cascadeVertices[2],mass_k0) + << " TauErr_k0 " << m_V0Tools->tauError(cascadeVertices[0],cascadeVertices[2],massesD0,mass_k0)); + + ATH_MSG_DEBUG("CascadeTools main vert wrt PV " << " CascadeTools SV " << " V0Tools SV"); + ATH_MSG_DEBUG("a0z " << m_CascadeTools->a0z(moms[2],cascadeVertices[2],primaryVertex) + << ", " << m_CascadeTools->a0z(moms[1],cascadeVertices[1],cascadeVertices[2]) + << ", " << m_CascadeTools->a0z(moms[0],cascadeVertices[0],cascadeVertices[2]) + << ", " << m_V0Tools->a0z(cascadeVertices[1],cascadeVertices[2]) + << ", " << m_V0Tools->a0z(cascadeVertices[0],cascadeVertices[2])); + ATH_MSG_DEBUG("a0zErr " << m_CascadeTools->a0zError(moms[2],x->getCovariance()[2],cascadeVertices[2],primaryVertex) + << ", " << m_CascadeTools->a0zError(moms[1],x->getCovariance()[1],cascadeVertices[1],cascadeVertices[2]) + << ", " << m_CascadeTools->a0zError(moms[0],x->getCovariance()[0],cascadeVertices[0],cascadeVertices[2]) + << ", " << m_V0Tools->a0zError(cascadeVertices[1],cascadeVertices[2]) + << ", " << m_V0Tools->a0zError(cascadeVertices[0],cascadeVertices[2])); + ATH_MSG_DEBUG("a0xy " << m_CascadeTools->a0xy(moms[2],cascadeVertices[2],primaryVertex) + << ", " << m_CascadeTools->a0xy(moms[1],cascadeVertices[1],cascadeVertices[2]) + << ", " << m_CascadeTools->a0xy(moms[0],cascadeVertices[0],cascadeVertices[2]) + << ", " << m_V0Tools->a0xy(cascadeVertices[1],cascadeVertices[2]) + << ", " << m_V0Tools->a0xy(cascadeVertices[0],cascadeVertices[2])); + ATH_MSG_DEBUG("a0xyErr " << m_CascadeTools->a0xyError(moms[2],x->getCovariance()[2],cascadeVertices[2],primaryVertex) + << ", " << m_CascadeTools->a0xyError(moms[1],x->getCovariance()[1],cascadeVertices[1],cascadeVertices[2]) + << ", " << m_CascadeTools->a0xyError(moms[0],x->getCovariance()[0],cascadeVertices[0],cascadeVertices[2]) + << ", " << m_V0Tools->a0xyError(cascadeVertices[1],cascadeVertices[2]) + << ", " << m_V0Tools->a0xyError(cascadeVertices[0],cascadeVertices[2])); + ATH_MSG_DEBUG("a0 " << m_CascadeTools->a0(moms[2],cascadeVertices[2],primaryVertex) + << ", " << m_CascadeTools->a0(moms[1],cascadeVertices[1],cascadeVertices[2]) + << ", " << m_CascadeTools->a0(moms[0],cascadeVertices[0],cascadeVertices[2]) + << ", " << m_V0Tools->a0(cascadeVertices[1],cascadeVertices[2]) + << ", " << m_V0Tools->a0(cascadeVertices[0],cascadeVertices[2])); + ATH_MSG_DEBUG("a0Err " << m_CascadeTools->a0Error(moms[2],x->getCovariance()[2],cascadeVertices[2],primaryVertex) + << ", " << m_CascadeTools->a0Error(moms[1],x->getCovariance()[1],cascadeVertices[1],cascadeVertices[2]) + << ", " << m_CascadeTools->a0Error(moms[0],x->getCovariance()[0],cascadeVertices[0],cascadeVertices[2]) + << ", " << m_V0Tools->a0Error(cascadeVertices[1],cascadeVertices[2]) + << ", " << m_V0Tools->a0Error(cascadeVertices[0],cascadeVertices[2])); + ATH_MSG_DEBUG("x0 " << m_V0Tools->vtx(cascadeVertices[0]).x() << " y0 " << m_V0Tools->vtx(cascadeVertices[0]).y() << " z0 " << m_V0Tools->vtx(cascadeVertices[0]).z()); + ATH_MSG_DEBUG("x1 " << m_V0Tools->vtx(cascadeVertices[1]).x() << " y1 " << m_V0Tools->vtx(cascadeVertices[1]).y() << " z1 " << m_V0Tools->vtx(cascadeVertices[1]).z()); + ATH_MSG_DEBUG("x2 " << m_V0Tools->vtx(cascadeVertices[2]).x() << " y2 " << m_V0Tools->vtx(cascadeVertices[2]).y() << " z2 " << m_V0Tools->vtx(cascadeVertices[2]).z()); + ATH_MSG_DEBUG("X0 " << primaryVertex->x() << " Y0 " << primaryVertex->y() << " Z0 " << primaryVertex->z()); + ATH_MSG_DEBUG("rxy0 " << m_V0Tools->rxy(cascadeVertices[0]) << " rxyErr0 " << m_V0Tools->rxyError(cascadeVertices[0])); + ATH_MSG_DEBUG("rxy1 " << m_V0Tools->rxy(cascadeVertices[1]) << " rxyErr1 " << m_V0Tools->rxyError(cascadeVertices[1])); + ATH_MSG_DEBUG("rxy2 " << m_V0Tools->rxy(cascadeVertices[2]) << " rxyErr2 " << m_V0Tools->rxyError(cascadeVertices[2])); + ATH_MSG_DEBUG("Rxy0 wrt PV " << m_V0Tools->rxy(cascadeVertices[0],primaryVertex) << " RxyErr0 wrt PV " << m_V0Tools->rxyError(cascadeVertices[0],primaryVertex)); + ATH_MSG_DEBUG("Rxy1 wrt PV " << m_V0Tools->rxy(cascadeVertices[1],primaryVertex) << " RxyErr1 wrt PV " << m_V0Tools->rxyError(cascadeVertices[1],primaryVertex)); + ATH_MSG_DEBUG("Rxy2 wrt PV " << m_V0Tools->rxy(cascadeVertices[2],primaryVertex) << " RxyErr2 wrt PV " << m_V0Tools->rxyError(cascadeVertices[2],primaryVertex)); + ATH_MSG_DEBUG("number of covariance matrices " << (x->getCovariance()).size()); + } // loop over cascadeinfoContainer + + // Deleting cascadeinfo since this won't be stored. + // Vertices have been kept in m_cascadeOutputs and should be owned by their container + for (auto x : cascadeinfoContainer) delete x; + + return StatusCode::SUCCESS; + } + + + JpsiPlusDs1Cascade::JpsiPlusDs1Cascade(const std::string& t, const std::string& n, const IInterface* p) : AthAlgTool(t,n,p), + m_vertexContainerKey(""), + m_vertexD0ContainerKey(""), + m_vertexK0ContainerKey(""), + m_cascadeOutputsKeys{ "JpsiPlusDs1CascadeVtx1", "JpsiPlusDs1CascadeVtx2", "JpsiPlusDs1CascadeVtx3" }, + m_VxPrimaryCandidateName("PrimaryVertices"), + m_jpsiMassLower(0.0), + m_jpsiMassUpper(10000.0), + m_jpsipiMassLower(0.0), + m_jpsipiMassUpper(10000.0), + m_D0MassLower(0.0), + m_D0MassUpper(10000.0), + m_K0MassLower(0.0), + m_K0MassUpper(10000.0), + m_DstMassLower(0.0), + m_DstMassUpper(10000.0), + m_MassLower(0.0), + m_MassUpper(20000.0), + m_vtx0MassHypo(-1), + m_vtx1MassHypo(-1), + m_vtx2MassHypo(-1), + m_vtx0Daug1MassHypo(-1), + m_vtx0Daug2MassHypo(-1), + m_vtx0Daug3MassHypo(-1), + m_vtx1Daug1MassHypo(-1), + m_vtx1Daug2MassHypo(-1), + m_vtx2Daug1MassHypo(-1), + m_vtx2Daug2MassHypo(-1), + m_particleDataTable(nullptr), + m_mass_jpsi(-1), + m_Dx_pid(421), + m_constrD0(true), + m_constrK0(true), + m_constrJpsi(true), + m_chi2cut(-1.0), + m_iVertexFitter("Trk::TrkVKalVrtFitter"), + m_pvRefitter("Analysis::PrimaryVertexRefitter"), + m_V0Tools("Trk::V0Tools"), + m_CascadeTools("DerivationFramework::CascadeTools") + { + declareProperty("JpsipiVertices", m_vertexContainerKey); + declareProperty("D0Vertices", m_vertexD0ContainerKey); + declareProperty("K0Vertices", m_vertexK0ContainerKey); + declareProperty("VxPrimaryCandidateName", m_VxPrimaryCandidateName); + declareProperty("RefPVContainerName", m_refPVContainerName = "RefittedPrimaryVertices"); + declareProperty("JpsiMassLowerCut", m_jpsiMassLower); + declareProperty("JpsiMassUpperCut", m_jpsiMassUpper); + declareProperty("JpsipiMassLowerCut", m_jpsipiMassLower); + declareProperty("JpsipiMassUpperCut", m_jpsipiMassUpper); + declareProperty("D0MassLowerCut", m_D0MassLower); + declareProperty("D0MassUpperCut", m_D0MassUpper); + declareProperty("K0MassLowerCut", m_K0MassLower); + declareProperty("K0MassUpperCut", m_K0MassUpper); + declareProperty("DstMassLowerCut", m_DstMassLower); + declareProperty("DstMassUpperCut", m_DstMassUpper); + declareProperty("MassLowerCut", m_MassLower); + declareProperty("MassUpperCut", m_MassUpper); + declareProperty("HypothesisName", m_hypoName = "Bc"); + declareProperty("Vtx0MassHypo", m_vtx0MassHypo); + declareProperty("Vtx1MassHypo", m_vtx1MassHypo); + declareProperty("Vtx2MassHypo", m_vtx2MassHypo); + declareProperty("Vtx0Daug1MassHypo", m_vtx0Daug1MassHypo); + declareProperty("Vtx0Daug2MassHypo", m_vtx0Daug2MassHypo); + declareProperty("Vtx0Daug3MassHypo", m_vtx0Daug3MassHypo); + declareProperty("Vtx1Daug1MassHypo", m_vtx1Daug1MassHypo); + declareProperty("Vtx1Daug2MassHypo", m_vtx1Daug2MassHypo); + declareProperty("Vtx2Daug1MassHypo", m_vtx2Daug1MassHypo); + declareProperty("Vtx2Daug2MassHypo", m_vtx2Daug2MassHypo); + declareProperty("JpsiMass", m_mass_jpsi); + declareProperty("DxHypothesis", m_Dx_pid); + declareProperty("ApplyD0MassConstraint", m_constrD0); + declareProperty("ApplyK0MassConstraint", m_constrK0); + declareProperty("ApplyJpsiMassConstraint", m_constrJpsi); + declareProperty("Chi2Cut", m_chi2cut); + declareProperty("RefitPV", m_refitPV = true); + declareProperty("MaxnPV", m_PV_max = 999); + declareProperty("MinNTracksInPV", m_PV_minNTracks = 0); + declareProperty("DoVertexType", m_DoVertexType = 7); + declareProperty("TrkVertexFitterTool", m_iVertexFitter); + declareProperty("PVRefitter", m_pvRefitter); + declareProperty("V0Tools", m_V0Tools); + declareProperty("CascadeTools", m_CascadeTools); + declareProperty("CascadeVertexCollections", m_cascadeOutputsKeys); + } + + JpsiPlusDs1Cascade::~JpsiPlusDs1Cascade(){ } + + StatusCode JpsiPlusDs1Cascade::performSearch(std::vector<Trk::VxCascadeInfo*> *cascadeinfoContainer) const + { + ATH_MSG_DEBUG( "JpsiPlusDs1Cascade::performSearch" ); + assert(cascadeinfoContainer!=nullptr); + + // Get TrackParticle container (for setting links to the original tracks) + const xAOD::TrackParticleContainer *trackContainer(nullptr); + ATH_CHECK(evtStore()->retrieve(trackContainer , "InDetTrackParticles" )); + + // Get Jpsi+pi container + const xAOD::VertexContainer *jpsipiContainer(nullptr); + ATH_CHECK(evtStore()->retrieve(jpsipiContainer , m_vertexContainerKey )); + + // Get D0 container + const xAOD::VertexContainer *d0Container(nullptr); + ATH_CHECK(evtStore()->retrieve(d0Container , m_vertexD0ContainerKey )); + + // Get K_S0 container + const xAOD::VertexContainer *k0Container(nullptr); + ATH_CHECK(evtStore()->retrieve(k0Container , m_vertexK0ContainerKey )); + + double mass_d0 = m_vtx1MassHypo; + double mass_k0 = m_vtx2MassHypo; + std::vector<const xAOD::TrackParticle*> tracksJpsipi; + std::vector<const xAOD::TrackParticle*> tracksJpsi; + std::vector<const xAOD::TrackParticle*> tracksD0; + std::vector<const xAOD::TrackParticle*> tracksK0; + std::vector<const xAOD::TrackParticle*> tracksBc; + std::vector<double> massesJpsipi; + massesJpsipi.push_back(m_vtx0Daug1MassHypo); + massesJpsipi.push_back(m_vtx0Daug2MassHypo); + massesJpsipi.push_back(m_vtx0Daug3MassHypo); + std::vector<double> massesD0; + massesD0.push_back(m_vtx1Daug1MassHypo); + massesD0.push_back(m_vtx1Daug2MassHypo); + std::vector<double> massesD0b; // Change the oreder of masses for D*-->D0bar pi-, D0bar->K+pi- + massesD0b.push_back(m_vtx1Daug2MassHypo); + massesD0b.push_back(m_vtx1Daug1MassHypo); + std::vector<double> massesK0; + massesK0.push_back(m_vtx2Daug1MassHypo); + massesK0.push_back(m_vtx2Daug2MassHypo); + std::vector<double> Masses; + Masses.push_back(m_vtx0Daug1MassHypo); + Masses.push_back(m_vtx0Daug2MassHypo); + Masses.push_back(m_vtx0Daug3MassHypo); + Masses.push_back(m_vtx1MassHypo); + Masses.push_back(m_vtx2MassHypo); + + // Select J/psi pi+ candidates before calling cascade fit + std::vector<const xAOD::Vertex*> selectedJpsipiCandidates; + for(auto vxcItr=jpsipiContainer->cbegin(); vxcItr!=jpsipiContainer->cend(); ++vxcItr) { + + // Check the passed flag first + const xAOD::Vertex* vtx = *vxcItr; + SG::AuxElement::Accessor<Char_t> flagAcc1("passed_Jpsipi"); + if(flagAcc1.isAvailable(*vtx)){ + if(!flagAcc1(*vtx)) continue; + } + + // Check J/psi candidate invariant mass and skip if need be + TLorentzVector p4Mup_in, p4Mum_in; + p4Mup_in.SetPtEtaPhiM((*vxcItr)->trackParticle(0)->pt(), + (*vxcItr)->trackParticle(0)->eta(), + (*vxcItr)->trackParticle(0)->phi(), m_vtx0Daug1MassHypo); + p4Mum_in.SetPtEtaPhiM((*vxcItr)->trackParticle(1)->pt(), + (*vxcItr)->trackParticle(1)->eta(), + (*vxcItr)->trackParticle(1)->phi(), m_vtx0Daug2MassHypo); + double mass_Jpsi = (p4Mup_in + p4Mum_in).M(); + ATH_MSG_DEBUG("Jpsi mass " << mass_Jpsi); + if (mass_Jpsi < m_jpsiMassLower || mass_Jpsi > m_jpsiMassUpper) { + ATH_MSG_DEBUG(" Original Jpsi candidate rejected by the mass cut: mass = " + << mass_Jpsi << " != (" << m_jpsiMassLower << ", " << m_jpsiMassUpper << ")" ); + continue; + } + + // Check J/psi pi+ candidate invariant mass and skip if need be + double mass_Jpsipi = m_V0Tools->invariantMass(*vxcItr, massesJpsipi); + ATH_MSG_DEBUG("Jpsipi mass " << mass_Jpsipi); + if (mass_Jpsipi < m_jpsipiMassLower || mass_Jpsipi > m_jpsipiMassUpper) { + ATH_MSG_DEBUG(" Original Jpsipi candidate rejected by the mass cut: mass = " + << mass_Jpsipi << " != (" << m_jpsipiMassLower << ", " << m_jpsipiMassUpper << ")" ); + continue; + } + + selectedJpsipiCandidates.push_back(*vxcItr); + } + if(selectedJpsipiCandidates.size()<1) return StatusCode::SUCCESS; + + // Select the D0/D0b candidates before calling cascade fit + std::vector<const xAOD::Vertex*> selectedD0Candidates; + for(auto vxcItr=d0Container->cbegin(); vxcItr!=d0Container->cend(); ++vxcItr) { + + // Check the passed flag first + const xAOD::Vertex* vtx = *vxcItr; + SG::AuxElement::Accessor<Char_t> flagAcc1("passed_D0"); + SG::AuxElement::Accessor<Char_t> flagAcc2("passed_D0b"); + bool isD0(true); + bool isD0b(true); + if(flagAcc1.isAvailable(*vtx)){ + if(!flagAcc1(*vtx)) isD0 = false; + } + if(flagAcc2.isAvailable(*vtx)){ + if(!flagAcc2(*vtx)) isD0b = false; + } + if(!(isD0||isD0b)) continue; + + // Ensure the total charge is correct + if ((*vxcItr)->trackParticle(0)->charge() != 1 || (*vxcItr)->trackParticle(1)->charge() != -1) { + ATH_MSG_DEBUG(" Original D0/D0-bar candidate rejected by the charge requirement: " + << (*vxcItr)->trackParticle(0)->charge() << ", " << (*vxcItr)->trackParticle(1)->charge() ); + continue; + } + + // Check D0/D0bar candidate invariant mass and skip if need be + double mass_D0 = m_V0Tools->invariantMass(*vxcItr,massesD0); + double mass_D0b = m_V0Tools->invariantMass(*vxcItr,massesD0b); + ATH_MSG_DEBUG("D0 mass " << mass_D0 << ", D0b mass "<<mass_D0b); + if ((mass_D0 < m_D0MassLower || mass_D0 > m_D0MassUpper) && (mass_D0b < m_D0MassLower || mass_D0b > m_D0MassUpper)) { + ATH_MSG_DEBUG(" Original D0 candidate rejected by the mass cut: mass = " + << mass_D0 << " != (" << m_D0MassLower << ", " << m_D0MassUpper << ") " + << mass_D0b << " != (" << m_D0MassLower << ", " << m_D0MassUpper << ") " ); + continue; + } + + selectedD0Candidates.push_back(*vxcItr); + } + if(selectedD0Candidates.size()<1) return StatusCode::SUCCESS; + + // Select the D0/D0b candidates before calling cascade fit + std::vector<const xAOD::Vertex*> selectedK0Candidates; + for(auto vxcItr=k0Container->cbegin(); vxcItr!=k0Container->cend(); ++vxcItr) { + + // Check the passed flag first + const xAOD::Vertex* vtx = *vxcItr; + SG::AuxElement::Accessor<Char_t> flagAcc1("passed_K0"); + if(flagAcc1.isAvailable(*vtx)){ + if(!flagAcc1(*vtx)) continue; + } + + // Check K_S0 candidate invariant mass and skip if need be + double mass_K0 = m_V0Tools->invariantMass(*vxcItr, massesK0); + ATH_MSG_DEBUG("K_S0 mass " << mass_K0); + if (mass_K0 < m_K0MassLower || mass_K0 > m_K0MassUpper) { + ATH_MSG_DEBUG(" Original K_S0 candidate rejected by the mass cut: mass = " + << mass_K0 << " != (" << m_K0MassLower << ", " << m_K0MassUpper << ")" ); + continue; + } + + selectedK0Candidates.push_back(*vxcItr); + } + if(selectedK0Candidates.size()<1) return StatusCode::SUCCESS; + + // Select J/psi D*+ candidates + // Iterate over Jpsi+pi vertices + for(auto jpsipiItr=selectedJpsipiCandidates.cbegin(); jpsipiItr!=selectedJpsipiCandidates.cend(); ++jpsipiItr) { + + size_t jpsipiTrkNum = (*jpsipiItr)->nTrackParticles(); + tracksJpsipi.clear(); + tracksJpsi.clear(); + for( unsigned int it=0; it<jpsipiTrkNum; it++) tracksJpsipi.push_back((*jpsipiItr)->trackParticle(it)); + for( unsigned int it=0; it<jpsipiTrkNum-1; it++) tracksJpsi.push_back((*jpsipiItr)->trackParticle(it)); + + if (tracksJpsipi.size() != 3 || massesJpsipi.size() != 3 ) { + ATH_MSG_INFO("problems with Jpsi+pi input"); + } + + bool tagD0(true); + if(abs(m_Dx_pid)==421 && (*jpsipiItr)->trackParticle(2)->charge()==-1) tagD0 = false; + + TLorentzVector p4_pi1; // Momentum of soft pion + p4_pi1.SetPtEtaPhiM((*jpsipiItr)->trackParticle(2)->pt(), + (*jpsipiItr)->trackParticle(2)->eta(), + (*jpsipiItr)->trackParticle(2)->phi(), m_vtx0Daug3MassHypo); + + // Iterate over D0/D0bar vertices + for(auto d0Itr=selectedD0Candidates.cbegin(); d0Itr!=selectedD0Candidates.cend(); ++d0Itr) { + + // Check identical tracks in input + if(std::find(tracksJpsipi.cbegin(), tracksJpsipi.cend(), (*d0Itr)->trackParticle(0)) != tracksJpsipi.cend()) continue; + if(std::find(tracksJpsipi.cbegin(), tracksJpsipi.cend(), (*d0Itr)->trackParticle(1)) != tracksJpsipi.cend()) continue; + + TLorentzVector p4_ka, p4_pi2; + if(tagD0){ // for D*+ + p4_pi2.SetPtEtaPhiM((*d0Itr)->trackParticle(0)->pt(), + (*d0Itr)->trackParticle(0)->eta(), + (*d0Itr)->trackParticle(0)->phi(), m_vtx1Daug1MassHypo); + p4_ka.SetPtEtaPhiM( (*d0Itr)->trackParticle(1)->pt(), + (*d0Itr)->trackParticle(1)->eta(), + (*d0Itr)->trackParticle(1)->phi(), m_vtx1Daug2MassHypo); + }else{ // change the order in the case of D*- + p4_pi2.SetPtEtaPhiM((*d0Itr)->trackParticle(1)->pt(), + (*d0Itr)->trackParticle(1)->eta(), + (*d0Itr)->trackParticle(1)->phi(), m_vtx1Daug1MassHypo); + p4_ka.SetPtEtaPhiM( (*d0Itr)->trackParticle(0)->pt(), + (*d0Itr)->trackParticle(0)->eta(), + (*d0Itr)->trackParticle(0)->phi(), m_vtx1Daug2MassHypo); + } + // Check D*+/- candidate invariant mass and skip if need be + double mass_Dst= (p4_pi1 + p4_ka + p4_pi2).M(); + ATH_MSG_DEBUG("D*+/- mass " << mass_Dst); + if (mass_Dst < m_DstMassLower || mass_Dst > m_DstMassUpper) { + ATH_MSG_DEBUG(" Original D*+/- candidate rejected by the mass cut: mass = " + << mass_Dst << " != (" << m_DstMassLower << ", " << m_DstMassUpper << ")" ); + continue; + } + + size_t d0TrkNum = (*d0Itr)->nTrackParticles(); + tracksD0.clear(); + for( unsigned int it=0; it<d0TrkNum; it++) tracksD0.push_back((*d0Itr)->trackParticle(it)); + if (tracksD0.size() != 2 || massesD0.size() != 2 ) { + ATH_MSG_INFO("problems with D0 input"); + } + + // Iterate over K0 vertices + for(auto k0Itr=selectedK0Candidates.cbegin(); k0Itr!=selectedK0Candidates.cend(); ++k0Itr) { + + // Check identical tracks in input + if(std::find(tracksJpsipi.cbegin(), tracksJpsipi.cend(), (*k0Itr)->trackParticle(0)) != tracksJpsipi.cend()) continue; + if(std::find(tracksJpsipi.cbegin(), tracksJpsipi.cend(), (*k0Itr)->trackParticle(1)) != tracksJpsipi.cend()) continue; + if(std::find(tracksD0.cbegin(), tracksD0.cend(), (*k0Itr)->trackParticle(0)) != tracksD0.cend()) continue; + if(std::find(tracksD0.cbegin(), tracksD0.cend(), (*k0Itr)->trackParticle(1)) != tracksD0.cend()) continue; + + size_t k0TrkNum = (*k0Itr)->nTrackParticles(); + tracksK0.clear(); + for( unsigned int it=0; it<k0TrkNum; it++) tracksK0.push_back((*k0Itr)->trackParticle(it)); + if (tracksK0.size() != 2 || massesK0.size() != 2 ) { + ATH_MSG_INFO("problems with K0 input"); + } + + ATH_MSG_DEBUG("using tracks" << tracksJpsipi[0] << ", " << tracksJpsipi[1] << ", " << tracksJpsipi[2] << ", " << tracksD0[0] << ", " << tracksD0[1] << ", " << tracksK0[0] << ", " << tracksK0[1]); + + tracksBc.clear(); + for( unsigned int it=0; it<jpsipiTrkNum; it++) tracksBc.push_back((*jpsipiItr)->trackParticle(it)); + for( unsigned int it=0; it<d0TrkNum; it++) tracksBc.push_back((*d0Itr)->trackParticle(it)); + for( unsigned int it=0; it<k0TrkNum; it++) tracksBc.push_back((*k0Itr)->trackParticle(it)); + + + // Apply the user's settings to the fitter + // Reset + std::unique_ptr<Trk::IVKalState> state (m_iVertexFitter->makeState()); + // Robustness + int robustness = 0; + m_iVertexFitter->setRobustness(robustness, *state); + // Build up the topology + // Vertex list + std::vector<Trk::VertexID> vrtList; + // K_S0 vertex + Trk::VertexID vK0ID; + if (m_constrK0) { + vK0ID = m_iVertexFitter->startVertex(tracksK0,massesK0, *state, mass_k0); + } else { + vK0ID = m_iVertexFitter->startVertex(tracksK0,massesK0, *state); + } + vrtList.push_back(vK0ID); + // D0 vertex + Trk::VertexID vD0ID; + if (m_constrD0) { + if(tagD0) vD0ID = m_iVertexFitter->nextVertex(tracksD0,massesD0, *state, mass_d0); + else vD0ID = m_iVertexFitter->nextVertex(tracksD0,massesD0b, *state, mass_d0); + } else { + if(tagD0) vD0ID = m_iVertexFitter->nextVertex(tracksD0,massesD0, *state); + else vD0ID = m_iVertexFitter->nextVertex(tracksD0,massesD0b, *state); + } + vrtList.push_back(vD0ID); + // B vertex including Jpsi+pi + Trk::VertexID vBcID = m_iVertexFitter->nextVertex(tracksJpsipi,massesJpsipi,vrtList, *state); + if (m_constrJpsi) { + std::vector<Trk::VertexID> cnstV; + cnstV.clear(); + if ( !m_iVertexFitter->addMassConstraint(vBcID,tracksJpsi,cnstV, *state, m_mass_jpsi).isSuccess() ) { + ATH_MSG_WARNING("addMassConstraint failed"); + //return StatusCode::FAILURE; + } + } + // Do the work + std::unique_ptr<Trk::VxCascadeInfo> result(m_iVertexFitter->fitCascade(*state)); + + if (result != nullptr) { + + // reset links to original tracks + BPhysPVCascadeTools::PrepareVertexLinks(result.get(), trackContainer); + ATH_MSG_DEBUG("storing tracks " << ((result->vertices())[0])->trackParticle(0) << ", " + << ((result->vertices())[0])->trackParticle(1) << ", " + << ((result->vertices())[1])->trackParticle(0) << ", " + << ((result->vertices())[1])->trackParticle(1) << ", " + << ((result->vertices())[2])->trackParticle(0) << ", " + << ((result->vertices())[2])->trackParticle(1) << ", " + << ((result->vertices())[2])->trackParticle(2)); + // necessary to prevent memory leak + result->setSVOwnership(true); + + // Chi2/DOF cut + double bChi2DOF = result->fitChi2()/result->nDoF(); + ATH_MSG_DEBUG("Candidate chi2/DOF is " << bChi2DOF); + bool chi2CutPassed = (m_chi2cut <= 0.0 || bChi2DOF < m_chi2cut); + + const std::vector< std::vector<TLorentzVector> > &moms = result->getParticleMoms(); + double mass = m_CascadeTools->invariantMass(moms[2]); + if(chi2CutPassed) { + if (mass >= m_MassLower && mass <= m_MassUpper) { + cascadeinfoContainer->push_back(result.release()); + } else { + ATH_MSG_DEBUG("Candidate rejected by the mass cut: mass = " + << mass << " != (" << m_MassLower << ", " << m_MassUpper << ")" ); + } + } + } + + } //Iterate over K0 vertices + + } //Iterate over D0 vertices + + } //Iterate over Jpsi+pi vertices + + ATH_MSG_DEBUG("cascadeinfoContainer size " << cascadeinfoContainer->size()); + + return StatusCode::SUCCESS; + } + +} diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/JpsiPlusDsCascade.cxx b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/JpsiPlusDsCascade.cxx new file mode 100644 index 00000000000..f7d485b75f7 --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/JpsiPlusDsCascade.cxx @@ -0,0 +1,702 @@ +/* + Copyright (C) 2002-2018 CERN for the benefit of the ATLAS collaboration +*/ +///////////////////////////////////////////////////////////////// +// JpsiPlusDsCascade.cxx, (c) ATLAS Detector software +///////////////////////////////////////////////////////////////// +#include "DerivationFrameworkBPhys/JpsiPlusDsCascade.h" +#include "TrkVertexFitterInterfaces/IVertexFitter.h" +#include "TrkVKalVrtFitter/TrkVKalVrtFitter.h" +#include "TrkVertexAnalysisUtils/V0Tools.h" +#include "GaudiKernel/IPartPropSvc.h" +#include "DerivationFrameworkBPhys/CascadeTools.h" +#include "DerivationFrameworkBPhys/BPhysPVCascadeTools.h" +#include "xAODTracking/VertexAuxContainer.h" +#include "BeamSpotConditionsData/BeamSpotData.h" +#include "xAODBPhys/BPhysHypoHelper.h" +#include <algorithm> +#include "xAODTracking/VertexContainer.h" +#include "DerivationFrameworkBPhys/LocalVector.h" +#include "HepPDT/ParticleDataTable.hh" + +namespace DerivationFramework { + typedef ElementLink<xAOD::VertexContainer> VertexLink; + typedef std::vector<VertexLink> VertexLinkVector; + typedef std::vector<const xAOD::TrackParticle*> TrackBag; + + StatusCode JpsiPlusDsCascade::initialize() { + + // retrieving vertex Fitter + ATH_CHECK( m_iVertexFitter.retrieve()); + + // retrieving the V0 tools + ATH_CHECK( m_V0Tools.retrieve()); + + // retrieving the Cascade tools + ATH_CHECK( m_CascadeTools.retrieve()); + + // Get the beam spot service + ATH_CHECK( m_beamSpotKey.initialize() ); + + IPartPropSvc* partPropSvc = nullptr; + ATH_CHECK( service("PartPropSvc", partPropSvc, true) ); + auto pdt = partPropSvc->PDT(); + + // retrieve particle masses + if(m_mass_jpsi < 0. ) m_mass_jpsi = BPhysPVCascadeTools::getParticleMass(pdt, PDG::J_psi); + if(m_vtx0MassHypo < 0.) + m_vtx0MassHypo = BPhysPVCascadeTools::getParticleMass(pdt, PDG::B_c_plus); + if(m_vtx1MassHypo < 0.) { + if(abs(m_Dx_pid) == 411) m_vtx1MassHypo = BPhysPVCascadeTools::getParticleMass(pdt, PDG::D_plus); + else m_vtx1MassHypo = BPhysPVCascadeTools::getParticleMass(pdt, PDG::D_s_plus); + } + + if(m_vtx0Daug1MassHypo < 0.) m_vtx0Daug1MassHypo = BPhysPVCascadeTools::getParticleMass(pdt, PDG::mu_minus); + if(m_vtx0Daug2MassHypo < 0.) m_vtx0Daug2MassHypo = BPhysPVCascadeTools::getParticleMass(pdt, PDG::mu_minus); + if(m_vtx1Daug1MassHypo < 0.) { + if(abs(m_Dx_pid) == 411) m_vtx1Daug1MassHypo = BPhysPVCascadeTools::getParticleMass(pdt, PDG::pi_plus); + else m_vtx1Daug1MassHypo = BPhysPVCascadeTools::getParticleMass(pdt, PDG::K_plus); + } + if(m_vtx1Daug2MassHypo < 0.) m_vtx1Daug2MassHypo = BPhysPVCascadeTools::getParticleMass(pdt, PDG::K_plus); + if(m_vtx1Daug3MassHypo < 0.) m_vtx1Daug3MassHypo = BPhysPVCascadeTools::getParticleMass(pdt, PDG::pi_plus); + + return StatusCode::SUCCESS; + } + + + StatusCode JpsiPlusDsCascade::addBranches() const + { + std::vector<Trk::VxCascadeInfo*> cascadeinfoContainer; + constexpr int topoN = 2; + std::array<xAOD::VertexContainer*, topoN> Vtxwritehandles; + std::array<xAOD::VertexAuxContainer*, topoN> Vtxwritehandlesaux; + if(m_cascadeOutputsKeys.size() !=topoN) { ATH_MSG_FATAL("Incorrect number of VtxContainers"); return StatusCode::FAILURE; } + + for(int i =0; i<topoN;i++){ + Vtxwritehandles[i] = new xAOD::VertexContainer(); + Vtxwritehandlesaux[i] = new xAOD::VertexAuxContainer(); + Vtxwritehandles[i]->setStore(Vtxwritehandlesaux[i]); + ATH_CHECK(evtStore()->record(Vtxwritehandles[i] , m_cascadeOutputsKeys[i] )); + ATH_CHECK(evtStore()->record(Vtxwritehandlesaux[i], m_cascadeOutputsKeys[i] + "Aux.")); + } + + //---------------------------------------------------- + // retrieve primary vertices + //---------------------------------------------------- + const xAOD::Vertex * primaryVertex(nullptr); + const xAOD::VertexContainer *pvContainer(nullptr); + ATH_CHECK(evtStore()->retrieve(pvContainer, m_VxPrimaryCandidateName)); + ATH_MSG_DEBUG("Found " << m_VxPrimaryCandidateName << " in StoreGate!"); + + if (pvContainer->size()==0){ + ATH_MSG_WARNING("You have no primary vertices: " << pvContainer->size()); + return StatusCode::RECOVERABLE; + } else { + primaryVertex = (*pvContainer)[0]; + } + + //---------------------------------------------------- + // Try to retrieve refitted primary vertices + //---------------------------------------------------- + xAOD::VertexContainer* refPvContainer = nullptr; + xAOD::VertexAuxContainer* refPvAuxContainer = nullptr; + if (m_refitPV) { + if (evtStore()->contains<xAOD::VertexContainer>(m_refPVContainerName)) { + // refitted PV container exists. Get it from the store gate + ATH_CHECK(evtStore()->retrieve(refPvContainer , m_refPVContainerName )); + ATH_CHECK(evtStore()->retrieve(refPvAuxContainer, m_refPVContainerName + "Aux.")); + } else { + // refitted PV container does not exist. Create a new one. + refPvContainer = new xAOD::VertexContainer; + refPvAuxContainer = new xAOD::VertexAuxContainer; + refPvContainer->setStore(refPvAuxContainer); + ATH_CHECK(evtStore()->record(refPvContainer , m_refPVContainerName)); + ATH_CHECK(evtStore()->record(refPvAuxContainer, m_refPVContainerName+"Aux.")); + } + } + + ATH_CHECK(performSearch(&cascadeinfoContainer)); + + SG::ReadCondHandle<InDet::BeamSpotData> beamSpotHandle { m_beamSpotKey }; + if(not beamSpotHandle.isValid()) ATH_MSG_ERROR("Cannot Retrieve " << m_beamSpotKey.key() ); + BPhysPVCascadeTools helper(&(*m_CascadeTools), beamSpotHandle.cptr()); + helper.SetMinNTracksInPV(m_PV_minNTracks); + + // Decorators for the main vertex: chi2, ndf, pt and pt error, plus the V0 vertex variables + SG::AuxElement::Decorator<VertexLinkVector> CascadeLinksDecor("CascadeVertexLinks"); + SG::AuxElement::Decorator<VertexLinkVector> JpsiLinksDecor("JpsiVertexLinks"); + SG::AuxElement::Decorator<VertexLinkVector> DxLinksDecor("DxVertexLinks"); + SG::AuxElement::Decorator<float> chi2_decor("ChiSquared"); + SG::AuxElement::Decorator<float> ndof_decor("NumberDoF"); + SG::AuxElement::Decorator<float> Pt_decor("Pt"); + SG::AuxElement::Decorator<float> PtErr_decor("PtErr"); + SG::AuxElement::Decorator<float> Mass_svdecor("Dx_mass"); + SG::AuxElement::Decorator<float> MassErr_svdecor("Dx_massErr"); + SG::AuxElement::Decorator<float> Pt_svdecor("Dx_Pt"); + SG::AuxElement::Decorator<float> PtErr_svdecor("Dx_PtErr"); + SG::AuxElement::Decorator<float> Lxy_svdecor("Dx_Lxy"); + SG::AuxElement::Decorator<float> LxyErr_svdecor("Dx_LxyErr"); + SG::AuxElement::Decorator<float> Tau_svdecor("Dx_Tau"); + SG::AuxElement::Decorator<float> TauErr_svdecor("Dx_TauErr"); + + SG::AuxElement::Decorator<float> MassMumu_decor("Mumu_mass"); + SG::AuxElement::Decorator<float> MassKX_svdecor("KX_mass"); + SG::AuxElement::Decorator<float> MassKXpi_svdecor("KXpi_mass"); + + ATH_MSG_DEBUG("cascadeinfoContainer size " << cascadeinfoContainer.size()); + + // Get Jpsi container and identify the input Jpsi + const xAOD::VertexContainer *jpsiContainer(nullptr); + ATH_CHECK(evtStore()->retrieve(jpsiContainer , m_vertexContainerKey )); + const xAOD::VertexContainer *dxContainer(nullptr); + ATH_CHECK(evtStore()->retrieve(dxContainer , m_vertexDxContainerKey )); + + for (Trk::VxCascadeInfo* x : cascadeinfoContainer) { + if(x==nullptr) ATH_MSG_ERROR("cascadeinfoContainer is null"); + + // the cascade fitter returns: + // std::vector<xAOD::Vertex*>, each xAOD::Vertex contains the refitted track parameters (perigee at the vertex position) + // vertices[iv] the links to the original TPs and a covariance of size 3+5*NTRK; the chi2 of the total fit + // is split between the cascade vertices as per track contribution + // std::vector< std::vector<TLorentzVector> >, each std::vector<TLorentzVector> contains the refitted momenta (TLorentzVector) + // momenta[iv][...] of all tracks in the corresponding vertex, including any pseudotracks (from cascade vertices) + // originating in this vertex; the masses are as assigned in the cascade fit + // std::vector<Amg::MatrixX>, the corresponding covariance matrices in momentum space + // covariance[iv] + // int nDoF, double Chi2 + // + // the invariant mass, pt, lifetime etc. errors should be calculated using the covariance matrices in momentum space as these + // take into account the full track-track and track-vertex correlations + // + // in the case of Jpsi+V0: vertices[0] is the V0 vertex, vertices[1] is the B/Lambda_b(bar) vertex, containing the 2 Jpsi tracks. + // The covariance terms between the two vertices are not stored. In momentum space momenta[0] contains the 2 V0 tracks, + // their momenta add up to the momentum of the 3rd track in momenta[1], the first two being the Jpsi tracks + + const std::vector<xAOD::Vertex*> &cascadeVertices = x->vertices(); + if(cascadeVertices.size()!=topoN) + ATH_MSG_ERROR("Incorrect number of vertices"); + if(cascadeVertices[0] == nullptr || cascadeVertices[1] == nullptr) ATH_MSG_ERROR("Error null vertex"); + // Keep vertices (bear in mind that they come in reverse order!) + for(int i =0;i<topoN;i++) Vtxwritehandles[i]->push_back(cascadeVertices[i]); + + x->setSVOwnership(false); // Prevent Container from deleting vertices + const auto mainVertex = cascadeVertices[1]; // this is the B_c+/- vertex + const std::vector< std::vector<TLorentzVector> > &moms = x->getParticleMoms(); + + // Set links to cascade vertices + std::vector<const xAOD::Vertex*> verticestoLink; + verticestoLink.push_back(cascadeVertices[0]); + if(Vtxwritehandles[1] == nullptr) ATH_MSG_ERROR("Vtxwritehandles[1] is null"); + if(!BPhysPVCascadeTools::LinkVertices(CascadeLinksDecor, verticestoLink, Vtxwritehandles[0], cascadeVertices[1])) + ATH_MSG_ERROR("Error decorating with cascade vertices"); + + // Identify the input Jpsi + const xAOD::Vertex* jpsiVertex = BPhysPVCascadeTools::FindVertex<2>(jpsiContainer, cascadeVertices[1]); + ATH_MSG_DEBUG("1 pt Jpsi tracks " << cascadeVertices[1]->trackParticle(0)->pt() << ", " << cascadeVertices[1]->trackParticle(1)->pt()); + if (jpsiVertex) ATH_MSG_DEBUG("2 pt Jpsi tracks " << jpsiVertex->trackParticle(0)->pt() << ", " << jpsiVertex->trackParticle(1)->pt()); + + // Identify the input D_(s)+ + const xAOD::Vertex* dxVertex = BPhysPVCascadeTools::FindVertex<3>(dxContainer, cascadeVertices[0]);; + ATH_MSG_DEBUG("1 pt D_(s)+ tracks " << cascadeVertices[0]->trackParticle(0)->pt() << ", " << cascadeVertices[0]->trackParticle(1)->pt() << ", " << cascadeVertices[0]->trackParticle(2)->pt()); + if (dxVertex) ATH_MSG_DEBUG("2 pt D_(s)+ tracks " << dxVertex->trackParticle(0)->pt() << ", " << dxVertex->trackParticle(1)->pt() << ", " << dxVertex->trackParticle(2)->pt()); + + // Set links to input vertices + std::vector<const xAOD::Vertex*> jpsiVerticestoLink; + if (jpsiVertex) jpsiVerticestoLink.push_back(jpsiVertex); + else ATH_MSG_WARNING("Could not find linking Jpsi"); + if(!BPhysPVCascadeTools::LinkVertices(JpsiLinksDecor, jpsiVerticestoLink, jpsiContainer, cascadeVertices[1])) + ATH_MSG_ERROR("Error decorating with Jpsi vertices"); + + std::vector<const xAOD::Vertex*> dxVerticestoLink; + if (dxVertex) dxVerticestoLink.push_back(dxVertex); + else ATH_MSG_WARNING("Could not find linking D_(s)+"); + if(!BPhysPVCascadeTools::LinkVertices(DxLinksDecor, dxVerticestoLink, dxContainer, cascadeVertices[1])) + ATH_MSG_ERROR("Error decorating with D_(s)+ vertices"); + + bool tagDp(true); + if (dxVertex) { + if(abs(m_Dx_pid)==411 && (dxVertex->trackParticle(2)->charge()==-1)) tagDp = false; + } + + double mass_b = m_vtx0MassHypo; + double mass_d = m_vtx1MassHypo; + std::vector<double> massesJpsi; + massesJpsi.push_back(m_vtx0Daug1MassHypo); + massesJpsi.push_back(m_vtx0Daug2MassHypo); + std::vector<double> massesDx; + if(tagDp){ + massesDx.push_back(m_vtx1Daug1MassHypo); + massesDx.push_back(m_vtx1Daug2MassHypo); + }else{ // Change the order for D- + massesDx.push_back(m_vtx1Daug2MassHypo); + massesDx.push_back(m_vtx1Daug1MassHypo); + } + massesDx.push_back(m_vtx1Daug3MassHypo); + std::vector<double> Masses; + Masses.push_back(m_vtx0Daug1MassHypo); + Masses.push_back(m_vtx0Daug2MassHypo); + Masses.push_back(m_vtx1MassHypo); + + // loop over candidates -- Don't apply PV_minNTracks requirement here + // because it may result in exclusion of the high-pt PV. + // get good PVs + + + xAOD::BPhysHypoHelper vtx(m_hypoName, mainVertex); + + // Get refitted track momenta from all vertices, charged tracks only + BPhysPVCascadeTools::SetVectorInfo(vtx, x); + + // Decorate main vertex + // + // 1.a) mass, mass error + BPHYS_CHECK( vtx.setMass(m_CascadeTools->invariantMass(moms[1])) ); + BPHYS_CHECK( vtx.setMassErr(m_CascadeTools->invariantMassError(moms[1],x->getCovariance()[1])) ); + // 1.b) pt and pT error (the default pt of mainVertex is != the pt of the full cascade fit!) + Pt_decor(*mainVertex) = m_CascadeTools->pT(moms[1]); + PtErr_decor(*mainVertex) = m_CascadeTools->pTError(moms[1],x->getCovariance()[1]); + // 1.c) chi2 and ndof (the default chi2 of mainVertex is != the chi2 of the full cascade fit!) + chi2_decor(*mainVertex) = x->fitChi2(); + ndof_decor(*mainVertex) = x->nDoF(); + + float massMumu = 0.; + if (jpsiVertex) { + TLorentzVector p4_mu1, p4_mu2; + p4_mu1.SetPtEtaPhiM(jpsiVertex->trackParticle(0)->pt(), + jpsiVertex->trackParticle(0)->eta(), + jpsiVertex->trackParticle(0)->phi(), m_vtx0Daug1MassHypo); + p4_mu2.SetPtEtaPhiM(jpsiVertex->trackParticle(1)->pt(), + jpsiVertex->trackParticle(1)->eta(), + jpsiVertex->trackParticle(1)->phi(), m_vtx0Daug2MassHypo); + massMumu = (p4_mu1 + p4_mu2).M(); + } + MassMumu_decor(*mainVertex) = massMumu; + + float massKX = 0., massKXpi = 0.; + if (dxVertex) { + TLorentzVector p4_h1, p4_h2, p4_h3; + if(tagDp){ + p4_h1.SetPtEtaPhiM(dxVertex->trackParticle(0)->pt(), + dxVertex->trackParticle(0)->eta(), + dxVertex->trackParticle(0)->phi(), m_vtx1Daug1MassHypo); + p4_h2.SetPtEtaPhiM(dxVertex->trackParticle(1)->pt(), + dxVertex->trackParticle(1)->eta(), + dxVertex->trackParticle(1)->phi(), m_vtx1Daug2MassHypo); + }else{ // Change the order for D- + p4_h1.SetPtEtaPhiM(dxVertex->trackParticle(0)->pt(), + dxVertex->trackParticle(0)->eta(), + dxVertex->trackParticle(0)->phi(), m_vtx1Daug2MassHypo); + p4_h2.SetPtEtaPhiM(dxVertex->trackParticle(1)->pt(), + dxVertex->trackParticle(1)->eta(), + dxVertex->trackParticle(1)->phi(), m_vtx1Daug1MassHypo); + } + p4_h3.SetPtEtaPhiM(dxVertex->trackParticle(2)->pt(), + dxVertex->trackParticle(2)->eta(), + dxVertex->trackParticle(2)->phi(), m_vtx1Daug3MassHypo); + massKX = (p4_h1 + p4_h2).M(); + massKXpi = (p4_h1 + p4_h2 + p4_h3).M(); + } + MassKX_svdecor(*mainVertex) = massKX; + MassKXpi_svdecor(*mainVertex) = massKXpi; + + ATH_CHECK(helper.FillCandwithRefittedVertices(m_refitPV, pvContainer, + refPvContainer, &(*m_pvRefitter), m_PV_max, m_DoVertexType, x, 1, mass_b, vtx)); + + + // 4) decorate the main vertex with V0 vertex mass, pt, lifetime and lxy values (plus errors) + // V0 points to the main vertex, so lifetime and lxy are w.r.t the main vertex + Mass_svdecor(*mainVertex) = m_CascadeTools->invariantMass(moms[0]); + MassErr_svdecor(*mainVertex) = m_CascadeTools->invariantMassError(moms[0],x->getCovariance()[0]); + Pt_svdecor(*mainVertex) = m_CascadeTools->pT(moms[0]); + PtErr_svdecor(*mainVertex) = m_CascadeTools->pTError(moms[0],x->getCovariance()[0]); + Lxy_svdecor(*mainVertex) = m_CascadeTools->lxy(moms[0],cascadeVertices[0],cascadeVertices[1]); + LxyErr_svdecor(*mainVertex) = m_CascadeTools->lxyError(moms[0],x->getCovariance()[0],cascadeVertices[0],cascadeVertices[1]); + Tau_svdecor(*mainVertex) = m_CascadeTools->tau(moms[0],cascadeVertices[0],cascadeVertices[1]); + TauErr_svdecor(*mainVertex) = m_CascadeTools->tauError(moms[0],x->getCovariance()[0],cascadeVertices[0],cascadeVertices[1]); + + // Some checks in DEBUG mode + ATH_MSG_DEBUG("chi2 " << x->fitChi2() + << " chi2_1 " << m_V0Tools->chisq(cascadeVertices[0]) + << " chi2_2 " << m_V0Tools->chisq(cascadeVertices[1]) + << " vprob " << m_CascadeTools->vertexProbability(x->nDoF(),x->fitChi2())); + ATH_MSG_DEBUG("ndf " << x->nDoF() << " ndf_1 " << m_V0Tools->ndof(cascadeVertices[0]) << " ndf_2 " << m_V0Tools->ndof(cascadeVertices[1])); + ATH_MSG_DEBUG("V0Tools mass_d " << m_V0Tools->invariantMass(cascadeVertices[0],massesDx) + << " error " << m_V0Tools->invariantMassError(cascadeVertices[0],massesDx) + << " mass_J " << m_V0Tools->invariantMass(cascadeVertices[1],massesJpsi) + << " error " << m_V0Tools->invariantMassError(cascadeVertices[1],massesJpsi)); + // masses and errors, using track masses assigned in the fit + double Mass_B = m_CascadeTools->invariantMass(moms[1]); + double Mass_D = m_CascadeTools->invariantMass(moms[0]); + double Mass_B_err = m_CascadeTools->invariantMassError(moms[1],x->getCovariance()[1]); + double Mass_D_err = m_CascadeTools->invariantMassError(moms[0],x->getCovariance()[0]); + ATH_MSG_DEBUG("Mass_B " << Mass_B << " Mass_D " << Mass_D); + ATH_MSG_DEBUG("Mass_B_err " << Mass_B_err << " Mass_D_err " << Mass_D_err); + double mprob_B = m_CascadeTools->massProbability(mass_b,Mass_B,Mass_B_err); + double mprob_D = m_CascadeTools->massProbability(mass_d,Mass_D,Mass_D_err); + ATH_MSG_DEBUG("mprob_B " << mprob_B << " mprob_D " << mprob_D); + // masses and errors, assigning user defined track masses + ATH_MSG_DEBUG("Mass_b " << m_CascadeTools->invariantMass(moms[1],Masses) + << " Mass_d " << m_CascadeTools->invariantMass(moms[0],massesDx)); + ATH_MSG_DEBUG("Mass_b_err " << m_CascadeTools->invariantMassError(moms[1],x->getCovariance()[1],Masses) + << " Mass_d_err " << m_CascadeTools->invariantMassError(moms[0],x->getCovariance()[0],massesDx)); + ATH_MSG_DEBUG("pt_b " << m_CascadeTools->pT(moms[1]) + << " pt_d " << m_CascadeTools->pT(moms[0]) + << " pt_dp " << m_V0Tools->pT(cascadeVertices[0])); + ATH_MSG_DEBUG("ptErr_b " << m_CascadeTools->pTError(moms[1],x->getCovariance()[1]) + << " ptErr_d " << m_CascadeTools->pTError(moms[0],x->getCovariance()[0]) + << " ptErr_dp " << m_V0Tools->pTError(cascadeVertices[0])); + ATH_MSG_DEBUG("lxy_B " << m_V0Tools->lxy(cascadeVertices[1],primaryVertex) << " lxy_D " << m_V0Tools->lxy(cascadeVertices[0],cascadeVertices[1])); + ATH_MSG_DEBUG("lxy_b " << m_CascadeTools->lxy(moms[1],cascadeVertices[1],primaryVertex) << " lxy_d " << m_CascadeTools->lxy(moms[0],cascadeVertices[0],cascadeVertices[1])); + ATH_MSG_DEBUG("lxyErr_b " << m_CascadeTools->lxyError(moms[1],x->getCovariance()[1],cascadeVertices[1],primaryVertex) + << " lxyErr_d " << m_CascadeTools->lxyError(moms[0],x->getCovariance()[0],cascadeVertices[0],cascadeVertices[1]) + << " lxyErr_dp " << m_V0Tools->lxyError(cascadeVertices[0],cascadeVertices[1])); + ATH_MSG_DEBUG("tau_B " << m_CascadeTools->tau(moms[1],cascadeVertices[1],primaryVertex,mass_b) + << " tau_dp " << m_V0Tools->tau(cascadeVertices[0],cascadeVertices[1],massesDx)); + ATH_MSG_DEBUG("tau_b " << m_CascadeTools->tau(moms[1],cascadeVertices[1],primaryVertex) + << " tau_d " << m_CascadeTools->tau(moms[0],cascadeVertices[0],cascadeVertices[1]) + << " tau_D " << m_CascadeTools->tau(moms[0],cascadeVertices[0],cascadeVertices[1],mass_d)); + ATH_MSG_DEBUG("tauErr_b " << m_CascadeTools->tauError(moms[1],x->getCovariance()[1],cascadeVertices[1],primaryVertex) + << " tauErr_d " << m_CascadeTools->tauError(moms[0],x->getCovariance()[0],cascadeVertices[0],cascadeVertices[1]) + << " tauErr_dp " << m_V0Tools->tauError(cascadeVertices[0],cascadeVertices[1],massesDx)); + ATH_MSG_DEBUG("TauErr_b " << m_CascadeTools->tauError(moms[1],x->getCovariance()[1],cascadeVertices[1],primaryVertex,mass_b) + << " TauErr_d " << m_CascadeTools->tauError(moms[0],x->getCovariance()[0],cascadeVertices[0],cascadeVertices[1],mass_d) + << " TauErr_dp " << m_V0Tools->tauError(cascadeVertices[0],cascadeVertices[1],massesDx,mass_d)); + + ATH_MSG_DEBUG("CascadeTools main vert wrt PV " << " CascadeTools SV " << " V0Tools SV"); + ATH_MSG_DEBUG("a0z " << m_CascadeTools->a0z(moms[1],cascadeVertices[1],primaryVertex) + << ", " << m_CascadeTools->a0z(moms[0],cascadeVertices[0],cascadeVertices[1]) + << ", " << m_V0Tools->a0z(cascadeVertices[0],cascadeVertices[1])); + ATH_MSG_DEBUG("a0zErr " << m_CascadeTools->a0zError(moms[1],x->getCovariance()[1],cascadeVertices[1],primaryVertex) + << ", " << m_CascadeTools->a0zError(moms[0],x->getCovariance()[0],cascadeVertices[0],cascadeVertices[1]) + << ", " << m_V0Tools->a0zError(cascadeVertices[0],cascadeVertices[1])); + ATH_MSG_DEBUG("a0xy " << m_CascadeTools->a0xy(moms[1],cascadeVertices[1],primaryVertex) + << ", " << m_CascadeTools->a0xy(moms[0],cascadeVertices[0],cascadeVertices[1]) + << ", " << m_V0Tools->a0xy(cascadeVertices[0],cascadeVertices[1])); + ATH_MSG_DEBUG("a0xyErr " << m_CascadeTools->a0xyError(moms[1],x->getCovariance()[1],cascadeVertices[1],primaryVertex) + << ", " << m_CascadeTools->a0xyError(moms[0],x->getCovariance()[0],cascadeVertices[0],cascadeVertices[1]) + << ", " << m_V0Tools->a0xyError(cascadeVertices[0],cascadeVertices[1])); + ATH_MSG_DEBUG("a0 " << m_CascadeTools->a0(moms[1],cascadeVertices[1],primaryVertex) + << ", " << m_CascadeTools->a0(moms[0],cascadeVertices[0],cascadeVertices[1]) + << ", " << m_V0Tools->a0(cascadeVertices[0],cascadeVertices[1])); + ATH_MSG_DEBUG("a0Err " << m_CascadeTools->a0Error(moms[1],x->getCovariance()[1],cascadeVertices[1],primaryVertex) + << ", " << m_CascadeTools->a0Error(moms[0],x->getCovariance()[0],cascadeVertices[0],cascadeVertices[1]) + << ", " << m_V0Tools->a0Error(cascadeVertices[0],cascadeVertices[1])); + ATH_MSG_DEBUG("x0 " << m_V0Tools->vtx(cascadeVertices[0]).x() << " y0 " << m_V0Tools->vtx(cascadeVertices[0]).y() << " z0 " << m_V0Tools->vtx(cascadeVertices[0]).z()); + ATH_MSG_DEBUG("x1 " << m_V0Tools->vtx(cascadeVertices[1]).x() << " y1 " << m_V0Tools->vtx(cascadeVertices[1]).y() << " z1 " << m_V0Tools->vtx(cascadeVertices[1]).z()); + ATH_MSG_DEBUG("X0 " << primaryVertex->x() << " Y0 " << primaryVertex->y() << " Z0 " << primaryVertex->z()); + ATH_MSG_DEBUG("rxy0 " << m_V0Tools->rxy(cascadeVertices[0]) << " rxyErr0 " << m_V0Tools->rxyError(cascadeVertices[0])); + ATH_MSG_DEBUG("rxy1 " << m_V0Tools->rxy(cascadeVertices[1]) << " rxyErr1 " << m_V0Tools->rxyError(cascadeVertices[1])); + ATH_MSG_DEBUG("Rxy0 wrt PV " << m_V0Tools->rxy(cascadeVertices[0],primaryVertex) << " RxyErr0 wrt PV " << m_V0Tools->rxyError(cascadeVertices[0],primaryVertex)); + ATH_MSG_DEBUG("Rxy1 wrt PV " << m_V0Tools->rxy(cascadeVertices[1],primaryVertex) << " RxyErr1 wrt PV " << m_V0Tools->rxyError(cascadeVertices[1],primaryVertex)); + ATH_MSG_DEBUG("number of covariance matrices " << (x->getCovariance()).size()); + } // loop over cascadeinfoContainer + + // Deleting cascadeinfo since this won't be stored. + // Vertices have been kept in m_cascadeOutputs and should be owned by their container + for (auto x : cascadeinfoContainer) delete x; + + return StatusCode::SUCCESS; + } + + + JpsiPlusDsCascade::JpsiPlusDsCascade(const std::string& t, const std::string& n, const IInterface* p) : AthAlgTool(t,n,p), + m_vertexContainerKey(""), + m_vertexDxContainerKey(""), + m_cascadeOutputsKeys{ "JpsiPlusDsCascadeVtx1", "JpsiPlusDsCascadeVtx2" }, + m_VxPrimaryCandidateName("PrimaryVertices"), + m_jpsiMassLower(0.0), + m_jpsiMassUpper(10000.0), + m_DxMassLower(0.0), + m_DxMassUpper(10000.0), + m_MassLower(0.0), + m_MassUpper(20000.0), + m_vtx0MassHypo(-1), + m_vtx1MassHypo(-1), + m_vtx0Daug1MassHypo(-1), + m_vtx0Daug2MassHypo(-1), + m_vtx1Daug1MassHypo(-1), + m_vtx1Daug2MassHypo(-1), + m_vtx1Daug3MassHypo(-1), + m_mass_jpsi(-1), + m_Dx_pid(431), + m_constrDx(true), + m_constrJpsi(true), + m_chi2cut(-1.0), + m_iVertexFitter("Trk::TrkVKalVrtFitter"), + m_pvRefitter("Analysis::PrimaryVertexRefitter"), + m_V0Tools("Trk::V0Tools"), + m_CascadeTools("DerivationFramework::CascadeTools") + { + declareProperty("JpsiVertices", m_vertexContainerKey); + declareProperty("DxVertices", m_vertexDxContainerKey); + declareProperty("VxPrimaryCandidateName", m_VxPrimaryCandidateName); + declareProperty("RefPVContainerName", m_refPVContainerName = "RefittedPrimaryVertices"); + declareProperty("JpsiMassLowerCut", m_jpsiMassLower); + declareProperty("JpsiMassUpperCut", m_jpsiMassUpper); + declareProperty("DxMassLowerCut", m_DxMassLower); + declareProperty("DxMassUpperCut", m_DxMassUpper); + declareProperty("MassLowerCut", m_MassLower); + declareProperty("MassUpperCut", m_MassUpper); + declareProperty("HypothesisName", m_hypoName = "Bc"); + declareProperty("Vtx0MassHypo", m_vtx0MassHypo); + declareProperty("Vtx1MassHypo", m_vtx1MassHypo); + declareProperty("Vtx0Daug1MassHypo", m_vtx0Daug1MassHypo); + declareProperty("Vtx0Daug2MassHypo", m_vtx0Daug2MassHypo); + declareProperty("Vtx1Daug1MassHypo", m_vtx1Daug1MassHypo); + declareProperty("Vtx1Daug2MassHypo", m_vtx1Daug2MassHypo); + declareProperty("Vtx1Daug3MassHypo", m_vtx1Daug3MassHypo); + declareProperty("JpsiMass", m_mass_jpsi); + declareProperty("DxHypothesis", m_Dx_pid); + declareProperty("ApplyDxMassConstraint", m_constrDx); + declareProperty("ApplyJpsiMassConstraint", m_constrJpsi); + declareProperty("Chi2Cut", m_chi2cut); + declareProperty("RefitPV", m_refitPV = true); + declareProperty("MaxnPV", m_PV_max = 999); + declareProperty("MinNTracksInPV", m_PV_minNTracks = 0); + declareProperty("DoVertexType", m_DoVertexType = 7); + declareProperty("TrkVertexFitterTool", m_iVertexFitter); + declareProperty("PVRefitter", m_pvRefitter); + declareProperty("V0Tools", m_V0Tools); + declareProperty("CascadeTools", m_CascadeTools); + declareProperty("CascadeVertexCollections", m_cascadeOutputsKeys); + } + + JpsiPlusDsCascade::~JpsiPlusDsCascade(){ } + + StatusCode JpsiPlusDsCascade::performSearch(std::vector<Trk::VxCascadeInfo*> *cascadeinfoContainer) const + { + ATH_MSG_DEBUG( "JpsiPlusDsCascade::performSearch" ); + assert(cascadeinfoContainer!=nullptr); + + // Get TrackParticle container (for setting links to the original tracks) + const xAOD::TrackParticleContainer *trackContainer(nullptr); + ATH_CHECK(evtStore()->retrieve(trackContainer , "InDetTrackParticles" )); + + // Get Jpsi container + const xAOD::VertexContainer *jpsiContainer(nullptr); + ATH_CHECK(evtStore()->retrieve(jpsiContainer , m_vertexContainerKey )); + + // Get V0 container + const xAOD::VertexContainer *dxContainer(nullptr); + ATH_CHECK(evtStore()->retrieve(dxContainer , m_vertexDxContainerKey )); + + + + double mass_d = m_vtx1MassHypo; + std::vector<const xAOD::TrackParticle*> tracksJpsi; + std::vector<const xAOD::TrackParticle*> tracksDx; + std::vector<const xAOD::TrackParticle*> tracksBc; + std::vector<double> massesJpsi; + massesJpsi.push_back(m_vtx0Daug1MassHypo); + massesJpsi.push_back(m_vtx0Daug2MassHypo); + std::vector<double> massesDx; + massesDx.push_back(m_vtx1Daug1MassHypo); + massesDx.push_back(m_vtx1Daug2MassHypo); + massesDx.push_back(m_vtx1Daug3MassHypo); + std::vector<double> massesDm; // Alter the order of masses for D- + massesDm.push_back(m_vtx1Daug2MassHypo); + massesDm.push_back(m_vtx1Daug1MassHypo); + massesDm.push_back(m_vtx1Daug3MassHypo); + std::vector<double> Masses; + Masses.push_back(m_vtx0Daug1MassHypo); + Masses.push_back(m_vtx0Daug2MassHypo); + Masses.push_back(m_vtx1MassHypo); + + // Select the J/psi candidates before calling cascade fit + std::vector<const xAOD::Vertex*> selectedJpsiCandidates; + for(auto vxcItr=jpsiContainer->cbegin(); vxcItr!=jpsiContainer->cend(); ++vxcItr) { + + // Check the passed flag first + const xAOD::Vertex* vtx = *vxcItr; + SG::AuxElement::Accessor<Char_t> flagAcc1("passed_Jpsi"); + if(flagAcc1.isAvailable(*vtx)){ + if(!flagAcc1(*vtx)) continue; + } + + // Check J/psi candidate invariant mass and skip if need be + double mass_Jpsi = m_V0Tools->invariantMass(*vxcItr, massesJpsi); + if (mass_Jpsi < m_jpsiMassLower || mass_Jpsi > m_jpsiMassUpper) { + ATH_MSG_DEBUG(" Original Jpsi candidate rejected by the mass cut: mass = " + << mass_Jpsi << " != (" << m_jpsiMassLower << ", " << m_jpsiMassUpper << ")" ); + continue; + } + selectedJpsiCandidates.push_back(*vxcItr); + } + if(selectedJpsiCandidates.size()<1) return StatusCode::SUCCESS; + + // Select the D_s+/D+ candidates before calling cascade fit + std::vector<const xAOD::Vertex*> selectedDxCandidates; + for(auto vxcItr=dxContainer->cbegin(); vxcItr!=dxContainer->cend(); ++vxcItr) { + + // Check the passed flag first + const xAOD::Vertex* vtx = *vxcItr; + if(abs(m_Dx_pid)==431) { // D_s+/- + SG::AuxElement::Accessor<Char_t> flagAcc1("passed_Ds"); + if(flagAcc1.isAvailable(*vtx)){ + if(!flagAcc1(*vtx)) continue; + } + } + + if(abs(m_Dx_pid==411)) { // D+/- + SG::AuxElement::Accessor<Char_t> flagAcc1("passed_Dp"); + SG::AuxElement::Accessor<Char_t> flagAcc2("passed_Dm"); + bool isDp(true); + bool isDm(true); + if(flagAcc1.isAvailable(*vtx)){ + if(!flagAcc1(*vtx)) isDp = false; + } + if(flagAcc2.isAvailable(*vtx)){ + if(!flagAcc2(*vtx)) isDm = false; + } + if(!(isDp||isDm)) continue; + } + + + // Ensure the total charge is correct + if(abs((*vxcItr)->trackParticle(0)->charge()+(*vxcItr)->trackParticle(1)->charge()+(*vxcItr)->trackParticle(2)->charge()) != 1){ + ATH_MSG_DEBUG(" Original D+ candidate rejected by the charge requirement: " + << (*vxcItr)->trackParticle(0)->charge() << ", " << (*vxcItr)->trackParticle(1)->charge() << ", " << (*vxcItr)->trackParticle(2)->charge() ); + continue; + } + + // Check D_(s)+/- candidate invariant mass and skip if need be + double mass_D; + if(abs(m_Dx_pid)==411 && (*vxcItr)->trackParticle(2)->charge()<0) // D- + mass_D = m_V0Tools->invariantMass(*vxcItr,massesDm); + else // D+, D_s+/- + mass_D = m_V0Tools->invariantMass(*vxcItr,massesDx); + ATH_MSG_DEBUG("D_(s) mass " << mass_D); + if(mass_D < m_DxMassLower || mass_D > m_DxMassUpper) { + ATH_MSG_DEBUG(" Original D_(s) candidate rejected by the mass cut: mass = " + << mass_D << " != (" << m_DxMassLower << ", " << m_DxMassUpper << ")" ); + continue; + } + + // Add loose cut on K+k- mass for D_s->phi pi + if(m_Dx_pid==431){ + TLorentzVector p4Kp_in, p4Km_in; + p4Kp_in.SetPtEtaPhiM( (*vxcItr)->trackParticle(0)->pt(), + (*vxcItr)->trackParticle(0)->eta(), + (*vxcItr)->trackParticle(0)->phi(), m_vtx1Daug1MassHypo); + p4Km_in.SetPtEtaPhiM( (*vxcItr)->trackParticle(1)->pt(), + (*vxcItr)->trackParticle(1)->eta(), + (*vxcItr)->trackParticle(1)->phi(), m_vtx1Daug2MassHypo); + double mass_phi = (p4Kp_in + p4Km_in).M(); + ATH_MSG_DEBUG("phi mass " << mass_phi); + if(mass_phi > 1200) { + ATH_MSG_DEBUG(" Original phi candidate rejected by the mass cut: mass = " << mass_phi ); + continue; + } + } + selectedDxCandidates.push_back(*vxcItr); + } + if(selectedDxCandidates.size()<1) return StatusCode::SUCCESS; + + // Select J/psi D_(s)+ candidates + // Iterate over Jpsi vertices + for(auto jpsiItr=selectedJpsiCandidates.cbegin(); jpsiItr!=selectedJpsiCandidates.cend(); ++jpsiItr) { + + size_t jpsiTrkNum = (*jpsiItr)->nTrackParticles(); + tracksJpsi.clear(); + for( unsigned int it=0; it<jpsiTrkNum; it++) tracksJpsi.push_back((*jpsiItr)->trackParticle(it)); + + if (tracksJpsi.size() != 2 || massesJpsi.size() != 2 ) { + ATH_MSG_INFO("problems with Jpsi input"); + } + + // Iterate over D_(s)+/- vertices + for(auto dxItr=selectedDxCandidates.cbegin(); dxItr!=selectedDxCandidates.cend(); ++dxItr) { + + // Check identical tracks in input + if(std::find(tracksJpsi.cbegin(), tracksJpsi.cend(), (*dxItr)->trackParticle(0)) != tracksJpsi.cend()) continue; + if(std::find(tracksJpsi.cbegin(), tracksJpsi.cend(), (*dxItr)->trackParticle(1)) != tracksJpsi.cend()) continue; + if(std::find(tracksJpsi.cbegin(), tracksJpsi.cend(), (*dxItr)->trackParticle(2)) != tracksJpsi.cend()) continue; + + size_t dxTrkNum = (*dxItr)->nTrackParticles(); + tracksDx.clear(); + for( unsigned int it=0; it<dxTrkNum; it++) tracksDx.push_back((*dxItr)->trackParticle(it)); + if (tracksDx.size() != 3 || massesDx.size() != 3 ) { + ATH_MSG_INFO("problems with D_(s) input"); + } + + ATH_MSG_DEBUG("using tracks" << tracksJpsi[0] << ", " << tracksJpsi[1] << ", " << tracksDx[0] << ", " << tracksDx[1] << ", " << tracksDx[2]); + tracksBc.clear(); + for( unsigned int it=0; it<jpsiTrkNum; it++) tracksBc.push_back((*jpsiItr)->trackParticle(it)); + for( unsigned int it=0; it<dxTrkNum; it++) tracksBc.push_back((*dxItr)->trackParticle(it)); + + // Apply the user's settings to the fitter + // Reset + std::unique_ptr<Trk::IVKalState> state (m_iVertexFitter->makeState()); + // Robustness + int robustness = 0; + m_iVertexFitter->setRobustness(robustness, *state); + // Build up the topology + // Vertex list + std::vector<Trk::VertexID> vrtList; + // D_(s)+/- vertex + Trk::VertexID vID; + if (m_constrDx) { + if(abs(m_Dx_pid)==411 && (*dxItr)->trackParticle(2)->charge()<0) // D- + vID = m_iVertexFitter->startVertex(tracksDx,massesDm,*state,mass_d); + else // D+, D_s+/- + vID = m_iVertexFitter->startVertex(tracksDx,massesDx,*state,mass_d); + } else { + if(abs(m_Dx_pid)==411 && (*dxItr)->trackParticle(2)->charge()<0) // D- + vID = m_iVertexFitter->startVertex(tracksDx,massesDm,*state); + else // D+, D_s+/- + vID = m_iVertexFitter->startVertex(tracksDx,massesDx,*state); + } + vrtList.push_back(vID); + // B vertex including Jpsi + Trk::VertexID vID2 = m_iVertexFitter->nextVertex(tracksJpsi,massesJpsi,vrtList,*state); + if (m_constrJpsi) { + std::vector<Trk::VertexID> cnstV; + cnstV.clear(); + if ( !m_iVertexFitter->addMassConstraint(vID2,tracksJpsi,cnstV,*state,m_mass_jpsi).isSuccess() ) { + ATH_MSG_WARNING("addMassConstraint failed"); + //return StatusCode::FAILURE; + } + } + // Do the work + std::unique_ptr<Trk::VxCascadeInfo> result(m_iVertexFitter->fitCascade(*state)); + + if (result != nullptr) { + // reset links to original tracks + BPhysPVCascadeTools::PrepareVertexLinks(result.get(), trackContainer); + ATH_MSG_DEBUG("storing tracks " << ((result->vertices())[0])->trackParticle(0) << ", " + << ((result->vertices())[0])->trackParticle(1) << ", " + << ((result->vertices())[0])->trackParticle(2) << ", " + << ((result->vertices())[1])->trackParticle(0) << ", " + << ((result->vertices())[1])->trackParticle(1)); + // necessary to prevent memory leak + result->setSVOwnership(true); + + // Chi2/DOF cut + double bChi2DOF = result->fitChi2()/result->nDoF(); + ATH_MSG_DEBUG("Candidate chi2/DOF is " << bChi2DOF); + bool chi2CutPassed = (m_chi2cut <= 0.0 || bChi2DOF < m_chi2cut); + + const std::vector< std::vector<TLorentzVector> > &moms = result->getParticleMoms(); + double mass = m_CascadeTools->invariantMass(moms[1]); + if(chi2CutPassed) { + if (mass >= m_MassLower && mass <= m_MassUpper) { + cascadeinfoContainer->push_back(result.release()); + } else { + ATH_MSG_DEBUG("Candidate rejected by the mass cut: mass = " + << mass << " != (" << m_MassLower << ", " << m_MassUpper << ")" ); + } + } + } + + } //Iterate over D_(s)+ vertices + + } //Iterate over Jpsi vertices + + ATH_MSG_DEBUG("cascadeinfoContainer size " << cascadeinfoContainer->size()); + + return StatusCode::SUCCESS; + } + +} + + diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/JpsiPlusV0Cascade.cxx b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/JpsiPlusV0Cascade.cxx new file mode 100644 index 00000000000..6f66478553a --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/JpsiPlusV0Cascade.cxx @@ -0,0 +1,576 @@ +/* + Copyright (C) 2002-2017 CERN for the benefit of the ATLAS collaboration +*/ +///////////////////////////////////////////////////////////////// +// JpsiPlusV0Cascade.cxx, (c) ATLAS Detector software +///////////////////////////////////////////////////////////////// +#include "DerivationFrameworkBPhys/JpsiPlusV0Cascade.h" +#include "TrkVertexFitterInterfaces/IVertexFitter.h" +#include "TrkVKalVrtFitter/TrkVKalVrtFitter.h" +#include "TrkVertexAnalysisUtils/V0Tools.h" +#include "GaudiKernel/IPartPropSvc.h" +#include "DerivationFrameworkBPhys/CascadeTools.h" +#include "DerivationFrameworkBPhys/BPhysPVCascadeTools.h" +#include "xAODTracking/VertexAuxContainer.h" +#include "BeamSpotConditionsData/BeamSpotData.h" +#include "xAODBPhys/BPhysHypoHelper.h" +#include <algorithm> +#include "xAODTracking/VertexContainer.h" +#include "DerivationFrameworkBPhys/LocalVector.h" +#include "HepPDT/ParticleDataTable.hh" + +namespace DerivationFramework { + typedef ElementLink<xAOD::VertexContainer> VertexLink; + typedef std::vector<VertexLink> VertexLinkVector; + typedef std::vector<const xAOD::TrackParticle*> TrackBag; + + + StatusCode JpsiPlusV0Cascade::initialize() { + + // retrieving vertex Fitter + if ( m_iVertexFitter.retrieve().isFailure() ) { + ATH_MSG_FATAL("Failed to retrieve tool " << m_iVertexFitter); + return StatusCode::FAILURE; + } else { + ATH_MSG_DEBUG("Retrieved tool " << m_iVertexFitter); + } + + // retrieving the V0 tools + if ( m_V0Tools.retrieve().isFailure() ) { + ATH_MSG_FATAL("Failed to retrieve tool " << m_V0Tools); + return StatusCode::FAILURE; + } else { + ATH_MSG_INFO("Retrieved tool " << m_V0Tools); + } + + // retrieving the Cascade tools + if ( m_CascadeTools.retrieve().isFailure() ) { + ATH_MSG_FATAL("Failed to retrieve tool " << m_CascadeTools); + return StatusCode::FAILURE; + } else { + ATH_MSG_INFO("Retrieved tool " << m_CascadeTools); + } + + ATH_CHECK(m_beamSpotKey.initialize()); + + IPartPropSvc* partPropSvc = 0; + StatusCode sc = service("PartPropSvc", partPropSvc, true); + if (sc.isFailure()) { + msg(MSG::ERROR) << "Could not initialize Particle Properties Service" << endmsg; + return StatusCode::FAILURE; + } + const HepPDT::ParticleDataTable* pdt = partPropSvc->PDT(); + + // retrieve particle masses + m_mass_muon = BPhysPVCascadeTools::getParticleMass(pdt, PDG::mu_minus); + m_mass_pion = BPhysPVCascadeTools::getParticleMass(pdt, PDG::pi_plus); + m_mass_proton = BPhysPVCascadeTools::getParticleMass(pdt, PDG::p_plus); + m_mass_lambda = BPhysPVCascadeTools::getParticleMass(pdt, PDG::Lambda0); + m_mass_ks = BPhysPVCascadeTools::getParticleMass(pdt, PDG::K_S0); + m_mass_jpsi = BPhysPVCascadeTools::getParticleMass(pdt, PDG::J_psi); + m_mass_b0 = BPhysPVCascadeTools::getParticleMass(pdt, PDG::B0); + m_mass_lambdaB = BPhysPVCascadeTools::getParticleMass(pdt, PDG::Lambda_b0); + + return StatusCode::SUCCESS; + } + + + StatusCode JpsiPlusV0Cascade::addBranches() const + { + std::vector<Trk::VxCascadeInfo*> cascadeinfoContainer; + constexpr int topoN = 2; + std::array<xAOD::VertexContainer*, topoN> Vtxwritehandles; + std::array<xAOD::VertexAuxContainer*, topoN> Vtxwritehandlesaux; + if(m_cascadeOutputsKeys.size() !=topoN) { ATH_MSG_FATAL("Incorrect number of VtxContainers"); return StatusCode::FAILURE; } + + for(int i =0; i<topoN;i++){ + Vtxwritehandles[i] = new xAOD::VertexContainer(); + Vtxwritehandlesaux[i] = new xAOD::VertexAuxContainer(); + Vtxwritehandles[i]->setStore(Vtxwritehandlesaux[i]); + CHECK(evtStore()->record(Vtxwritehandles[i] , m_cascadeOutputsKeys[i] )); + CHECK(evtStore()->record(Vtxwritehandlesaux[i], m_cascadeOutputsKeys[i] + "Aux.")); + } + + //---------------------------------------------------- + // retrieve primary vertices + //---------------------------------------------------- + const xAOD::Vertex * primaryVertex(nullptr); + const xAOD::VertexContainer *pvContainer(nullptr); + CHECK(evtStore()->retrieve(pvContainer, m_VxPrimaryCandidateName)); + ATH_MSG_DEBUG("Found " << m_VxPrimaryCandidateName << " in StoreGate!"); + + if (pvContainer->size()==0){ + ATH_MSG_WARNING("You have no primary vertices: " << pvContainer->size()); + return StatusCode::RECOVERABLE; + } else { + primaryVertex = (*pvContainer)[0]; + } + + //---------------------------------------------------- + // Try to retrieve refitted primary vertices + //---------------------------------------------------- + xAOD::VertexContainer* refPvContainer = NULL; + xAOD::VertexAuxContainer* refPvAuxContainer = NULL; + if (m_refitPV) { + if (evtStore()->contains<xAOD::VertexContainer>(m_refPVContainerName)) { + // refitted PV container exists. Get it from the store gate + CHECK(evtStore()->retrieve(refPvContainer , m_refPVContainerName )); + CHECK(evtStore()->retrieve(refPvAuxContainer, m_refPVContainerName + "Aux.")); + } else { + // refitted PV container does not exist. Create a new one. + refPvContainer = new xAOD::VertexContainer; + refPvAuxContainer = new xAOD::VertexAuxContainer; + refPvContainer->setStore(refPvAuxContainer); + CHECK(evtStore()->record(refPvContainer , m_refPVContainerName)); + CHECK(evtStore()->record(refPvAuxContainer, m_refPVContainerName+"Aux.")); + } + } + + ATH_CHECK(performSearch(&cascadeinfoContainer)); + + SG::ReadCondHandle<InDet::BeamSpotData> beamSpotHandle { m_beamSpotKey }; + if(not beamSpotHandle.isValid()) ATH_MSG_ERROR("Cannot Retrieve " << m_beamSpotKey.key() ); + BPhysPVCascadeTools helper(&(*m_CascadeTools), beamSpotHandle.cptr()); + helper.SetMinNTracksInPV(m_PV_minNTracks); + + // Decorators for the main vertex: chi2, ndf, pt and pt error, plus the V0 vertex variables + SG::AuxElement::Decorator<VertexLinkVector> CascadeLinksDecor("CascadeVertexLinks"); + SG::AuxElement::Decorator<VertexLinkVector> JpsiLinksDecor("JpsiVertexLinks"); + SG::AuxElement::Decorator<VertexLinkVector> V0LinksDecor("V0VertexLinks"); + SG::AuxElement::Decorator<float> chi2_decor("ChiSquared"); + SG::AuxElement::Decorator<float> ndof_decor("NumberDoF"); + SG::AuxElement::Decorator<float> Pt_decor("Pt"); + SG::AuxElement::Decorator<float> PtErr_decor("PtErr"); + SG::AuxElement::Decorator<float> Mass_svdecor("V0_mass"); + SG::AuxElement::Decorator<float> MassErr_svdecor("V0_massErr"); + SG::AuxElement::Decorator<float> Pt_svdecor("V0_Pt"); + SG::AuxElement::Decorator<float> PtErr_svdecor("V0_PtErr"); + SG::AuxElement::Decorator<float> Lxy_svdecor("V0_Lxy"); + SG::AuxElement::Decorator<float> LxyErr_svdecor("V0_LxyErr"); + SG::AuxElement::Decorator<float> Tau_svdecor("V0_Tau"); + SG::AuxElement::Decorator<float> TauErr_svdecor("V0_TauErr"); + + ATH_MSG_DEBUG("cascadeinfoContainer size " << cascadeinfoContainer.size()); + + // Get Jpsi container and identify the input Jpsi + const xAOD::VertexContainer *jpsiContainer(nullptr); + CHECK(evtStore()->retrieve(jpsiContainer , m_vertexContainerKey )); + const xAOD::VertexContainer *v0Container(nullptr); + CHECK(evtStore()->retrieve(v0Container , m_vertexV0ContainerKey )); + + for (Trk::VxCascadeInfo* x : cascadeinfoContainer) { + if(x==nullptr) ATH_MSG_ERROR("cascadeinfoContainer is null"); + + // the cascade fitter returns: + // std::vector<xAOD::Vertex*>, each xAOD::Vertex contains the refitted track parameters (perigee at the vertex position) + // vertices[iv] the links to the original TPs and a covariance of size 3+5*NTRK; the chi2 of the total fit + // is split between the cascade vertices as per track contribution + // std::vector< std::vector<TLorentzVector> >, each std::vector<TLorentzVector> contains the refitted momenta (TLorentzVector) + // momenta[iv][...] of all tracks in the corresponding vertex, including any pseudotracks (from cascade vertices) + // originating in this vertex; the masses are as assigned in the cascade fit + // std::vector<Amg::MatrixX>, the corresponding covariance matrices in momentum space + // covariance[iv] + // int nDoF, double Chi2 + // + // the invariant mass, pt, lifetime etc. errors should be calculated using the covariance matrices in momentum space as these + // take into account the full track-track and track-vertex correlations + // + // in the case of Jpsi+V0: vertices[0] is the V0 vertex, vertices[1] is the B/Lambda_b(bar) vertex, containing the 2 Jpsi tracks. + // The covariance terms between the two vertices are not stored. In momentum space momenta[0] contains the 2 V0 tracks, + // their momenta add up to the momentum of the 3rd track in momenta[1], the first two being the Jpsi tracks + + const std::vector<xAOD::Vertex*> &cascadeVertices = x->vertices(); + if(cascadeVertices.size()!=topoN) + ATH_MSG_ERROR("Incorrect number of vertices"); + if(cascadeVertices[0] == nullptr || cascadeVertices[1] == nullptr) ATH_MSG_ERROR("Error null vertex"); + // Keep vertices (bear in mind that they come in reverse order!) + for(int i =0;i<topoN;i++) Vtxwritehandles[i]->push_back(cascadeVertices[i]); + + x->setSVOwnership(false); // Prevent Container from deleting vertices + const auto mainVertex = cascadeVertices[1]; // this is the Bd (Bd, Lambda_b, Lambda_bbar) vertex + //const auto v0Vertex = cascadeVertices[0]; // this is the V0 (Kshort, Lambda, Lambdabar) vertex + const std::vector< std::vector<TLorentzVector> > &moms = x->getParticleMoms(); + + // Set links to cascade vertices + std::vector<const xAOD::Vertex*> verticestoLink; + verticestoLink.push_back(cascadeVertices[0]); + if(Vtxwritehandles[1] == nullptr) ATH_MSG_ERROR("Vtxwritehandles[1] is null"); + if(!BPhysPVCascadeTools::LinkVertices(CascadeLinksDecor, verticestoLink, Vtxwritehandles[0], cascadeVertices[1])) + ATH_MSG_ERROR("Error decorating with cascade vertices"); + + // Identify the input Jpsi + const xAOD::Vertex* jpsiVertex = BPhysPVCascadeTools::FindVertex<2>(jpsiContainer, cascadeVertices[1]); + ATH_MSG_DEBUG("1 pt Jpsi tracks " << cascadeVertices[1]->trackParticle(0)->pt() << ", " << cascadeVertices[1]->trackParticle(1)->pt()); + if (jpsiVertex) ATH_MSG_DEBUG("2 pt Jpsi tracks " << jpsiVertex->trackParticle(0)->pt() << ", " << jpsiVertex->trackParticle(1)->pt()); + + // Identify the input V0 + const xAOD::Vertex* v0Vertex = BPhysPVCascadeTools::FindVertex<2>(v0Container, cascadeVertices[0]);; + ATH_MSG_DEBUG("1 pt V0 tracks " << cascadeVertices[0]->trackParticle(0)->pt() << ", " << cascadeVertices[0]->trackParticle(1)->pt()); + if (v0Vertex) ATH_MSG_DEBUG("2 pt V0 tracks " << v0Vertex->trackParticle(0)->pt() << ", " << v0Vertex->trackParticle(1)->pt()); + + // Set links to input vertices + std::vector<const xAOD::Vertex*> jpsiVerticestoLink; + if (jpsiVertex) jpsiVerticestoLink.push_back(jpsiVertex); + else ATH_MSG_WARNING("Could not find linking Jpsi"); + if(!BPhysPVCascadeTools::LinkVertices(JpsiLinksDecor, jpsiVerticestoLink, jpsiContainer, cascadeVertices[1])) + ATH_MSG_ERROR("Error decorating with Jpsi vertices"); + + std::vector<const xAOD::Vertex*> v0VerticestoLink; + if (v0Vertex) v0VerticestoLink.push_back(v0Vertex); + else ATH_MSG_WARNING("Could not find linking V0"); + if(!BPhysPVCascadeTools::LinkVertices(V0LinksDecor, v0VerticestoLink, v0Container, cascadeVertices[1])) + ATH_MSG_ERROR("Error decorating with V0 vertices"); + + double mass_v0 = m_mass_ks; + double mass_b = m_mass_b0; + std::vector<double> massesJpsi(2, m_mass_muon); + std::vector<double> massesV0; + std::vector<double> Masses(2, m_mass_muon); + if (m_v0_pid == 310) { + massesV0.push_back(m_mass_pion); + massesV0.push_back(m_mass_pion); + Masses.push_back(m_mass_ks); + } else if (m_v0_pid == 3122) { + massesV0.push_back(m_mass_proton); + massesV0.push_back(m_mass_pion); + Masses.push_back(m_mass_lambda); + mass_v0 = m_mass_lambda; + mass_b = m_mass_lambdaB; + } else if (m_v0_pid == -3122) { + massesV0.push_back(m_mass_pion); + massesV0.push_back(m_mass_proton); + Masses.push_back(m_mass_lambda); + mass_v0 = m_mass_lambda; + mass_b = m_mass_lambdaB; + } + + // loop over candidates -- Don't apply PV_minNTracks requirement here + // because it may result in exclusion of the high-pt PV. + // get good PVs + + xAOD::BPhysHypoHelper vtx(m_hypoName, mainVertex); + + BPhysPVCascadeTools::SetVectorInfo(vtx, x); + + + // Decorate main vertex + // + // 1.a) mass, mass error + BPHYS_CHECK( vtx.setMass(m_CascadeTools->invariantMass(moms[1])) ); + BPHYS_CHECK( vtx.setMassErr(m_CascadeTools->invariantMassError(moms[1],x->getCovariance()[1])) ); + // 1.b) pt and pT error (the default pt of mainVertex is != the pt of the full cascade fit!) + Pt_decor(*mainVertex) = m_CascadeTools->pT(moms[1]); + PtErr_decor(*mainVertex) = m_CascadeTools->pTError(moms[1],x->getCovariance()[1]); + // 1.c) chi2 and ndof (the default chi2 of mainVertex is != the chi2 of the full cascade fit!) + chi2_decor(*mainVertex) = x->fitChi2(); + ndof_decor(*mainVertex) = x->nDoF(); + + ATH_CHECK(helper.FillCandwithRefittedVertices(m_refitPV, pvContainer, + refPvContainer, &(*m_pvRefitter), m_PV_max, m_DoVertexType, x, 1, mass_b, vtx)); + + // 4) decorate the main vertex with V0 vertex mass, pt, lifetime and lxy values (plus errors) + // V0 points to the main vertex, so lifetime and lxy are w.r.t the main vertex + Mass_svdecor(*mainVertex) = m_CascadeTools->invariantMass(moms[0]); + MassErr_svdecor(*mainVertex) = m_CascadeTools->invariantMassError(moms[0],x->getCovariance()[0]); + Pt_svdecor(*mainVertex) = m_CascadeTools->pT(moms[0]); + PtErr_svdecor(*mainVertex) = m_CascadeTools->pTError(moms[0],x->getCovariance()[0]); + Lxy_svdecor(*mainVertex) = m_CascadeTools->lxy(moms[0],cascadeVertices[0],cascadeVertices[1]); + LxyErr_svdecor(*mainVertex) = m_CascadeTools->lxyError(moms[0],x->getCovariance()[0],cascadeVertices[0],cascadeVertices[1]); + Tau_svdecor(*mainVertex) = m_CascadeTools->tau(moms[0],cascadeVertices[0],cascadeVertices[1]); + TauErr_svdecor(*mainVertex) = m_CascadeTools->tauError(moms[0],x->getCovariance()[0],cascadeVertices[0],cascadeVertices[1]); + + // Some checks in DEBUG mode + ATH_MSG_DEBUG("chi2 " << x->fitChi2() + << " chi2_1 " << m_V0Tools->chisq(cascadeVertices[0]) + << " chi2_2 " << m_V0Tools->chisq(cascadeVertices[1]) + << " vprob " << m_CascadeTools->vertexProbability(x->nDoF(),x->fitChi2())); + ATH_MSG_DEBUG("ndf " << x->nDoF() << " ndf_1 " << m_V0Tools->ndof(cascadeVertices[0]) << " ndf_2 " << m_V0Tools->ndof(cascadeVertices[1])); + ATH_MSG_DEBUG("V0Tools mass_v0 " << m_V0Tools->invariantMass(cascadeVertices[0],massesV0) + << " error " << m_V0Tools->invariantMassError(cascadeVertices[0],massesV0) + << " mass_J " << m_V0Tools->invariantMass(cascadeVertices[1],massesJpsi) + << " error " << m_V0Tools->invariantMassError(cascadeVertices[1],massesJpsi)); + // masses and errors, using track masses assigned in the fit + double Mass_B = m_CascadeTools->invariantMass(moms[1]); + double Mass_V0 = m_CascadeTools->invariantMass(moms[0]); + double Mass_B_err = m_CascadeTools->invariantMassError(moms[1],x->getCovariance()[1]); + double Mass_V0_err = m_CascadeTools->invariantMassError(moms[0],x->getCovariance()[0]); + ATH_MSG_DEBUG("Mass_B " << Mass_B << " Mass_V0 " << Mass_V0); + ATH_MSG_DEBUG("Mass_B_err " << Mass_B_err << " Mass_V0_err " << Mass_V0_err); + double mprob_B = m_CascadeTools->massProbability(mass_b,Mass_B,Mass_B_err); + double mprob_V0 = m_CascadeTools->massProbability(mass_v0,Mass_V0,Mass_V0_err); + ATH_MSG_DEBUG("mprob_B " << mprob_B << " mprob_V0 " << mprob_V0); + // masses and errors, assigning user defined track masses + ATH_MSG_DEBUG("Mass_b " << m_CascadeTools->invariantMass(moms[1],Masses) + << " Mass_v0 " << m_CascadeTools->invariantMass(moms[0],massesV0)); + ATH_MSG_DEBUG("Mass_b_err " << m_CascadeTools->invariantMassError(moms[1],x->getCovariance()[1],Masses) + << " Mass_v0_err " << m_CascadeTools->invariantMassError(moms[0],x->getCovariance()[0],massesV0)); + ATH_MSG_DEBUG("pt_b " << m_CascadeTools->pT(moms[1]) + << " pt_v " << m_CascadeTools->pT(moms[0]) + << " pt_v0 " << m_V0Tools->pT(cascadeVertices[0])); + ATH_MSG_DEBUG("ptErr_b " << m_CascadeTools->pTError(moms[1],x->getCovariance()[1]) + << " ptErr_v " << m_CascadeTools->pTError(moms[0],x->getCovariance()[0]) + << " ptErr_v0 " << m_V0Tools->pTError(cascadeVertices[0])); + ATH_MSG_DEBUG("lxy_B " << m_V0Tools->lxy(cascadeVertices[1],primaryVertex) << " lxy_V " << m_V0Tools->lxy(cascadeVertices[0],cascadeVertices[1])); + ATH_MSG_DEBUG("lxy_b " << m_CascadeTools->lxy(moms[1],cascadeVertices[1],primaryVertex) << " lxy_v " << m_CascadeTools->lxy(moms[0],cascadeVertices[0],cascadeVertices[1])); + ATH_MSG_DEBUG("lxyErr_b " << m_CascadeTools->lxyError(moms[1],x->getCovariance()[1],cascadeVertices[1],primaryVertex) + << " lxyErr_v " << m_CascadeTools->lxyError(moms[0],x->getCovariance()[0],cascadeVertices[0],cascadeVertices[1]) + << " lxyErr_v0 " << m_V0Tools->lxyError(cascadeVertices[0],cascadeVertices[1])); + ATH_MSG_DEBUG("tau_B " << m_CascadeTools->tau(moms[1],cascadeVertices[1],primaryVertex,mass_b) + << " tau_v0 " << m_V0Tools->tau(cascadeVertices[0],cascadeVertices[1],massesV0)); + ATH_MSG_DEBUG("tau_b " << m_CascadeTools->tau(moms[1],cascadeVertices[1],primaryVertex) + << " tau_v " << m_CascadeTools->tau(moms[0],cascadeVertices[0],cascadeVertices[1]) + << " tau_V " << m_CascadeTools->tau(moms[0],cascadeVertices[0],cascadeVertices[1],mass_v0)); + ATH_MSG_DEBUG("tauErr_b " << m_CascadeTools->tauError(moms[1],x->getCovariance()[1],cascadeVertices[1],primaryVertex) + << " tauErr_v " << m_CascadeTools->tauError(moms[0],x->getCovariance()[0],cascadeVertices[0],cascadeVertices[1]) + << " tauErr_v0 " << m_V0Tools->tauError(cascadeVertices[0],cascadeVertices[1],massesV0)); + ATH_MSG_DEBUG("TauErr_b " << m_CascadeTools->tauError(moms[1],x->getCovariance()[1],cascadeVertices[1],primaryVertex,mass_b) + << " TauErr_v " << m_CascadeTools->tauError(moms[0],x->getCovariance()[0],cascadeVertices[0],cascadeVertices[1],mass_v0) + << " TauErr_v0 " << m_V0Tools->tauError(cascadeVertices[0],cascadeVertices[1],massesV0,mass_v0)); + + ATH_MSG_DEBUG("CascadeTools main vert wrt PV " << " CascadeTools SV " << " V0Tools SV"); + ATH_MSG_DEBUG("a0z " << m_CascadeTools->a0z(moms[1],cascadeVertices[1],primaryVertex) + << ", " << m_CascadeTools->a0z(moms[0],cascadeVertices[0],cascadeVertices[1]) + << ", " << m_V0Tools->a0z(cascadeVertices[0],cascadeVertices[1])); + ATH_MSG_DEBUG("a0zErr " << m_CascadeTools->a0zError(moms[1],x->getCovariance()[1],cascadeVertices[1],primaryVertex) + << ", " << m_CascadeTools->a0zError(moms[0],x->getCovariance()[0],cascadeVertices[0],cascadeVertices[1]) + << ", " << m_V0Tools->a0zError(cascadeVertices[0],cascadeVertices[1])); + ATH_MSG_DEBUG("a0xy " << m_CascadeTools->a0xy(moms[1],cascadeVertices[1],primaryVertex) + << ", " << m_CascadeTools->a0xy(moms[0],cascadeVertices[0],cascadeVertices[1]) + << ", " << m_V0Tools->a0xy(cascadeVertices[0],cascadeVertices[1])); + ATH_MSG_DEBUG("a0xyErr " << m_CascadeTools->a0xyError(moms[1],x->getCovariance()[1],cascadeVertices[1],primaryVertex) + << ", " << m_CascadeTools->a0xyError(moms[0],x->getCovariance()[0],cascadeVertices[0],cascadeVertices[1]) + << ", " << m_V0Tools->a0xyError(cascadeVertices[0],cascadeVertices[1])); + ATH_MSG_DEBUG("a0 " << m_CascadeTools->a0(moms[1],cascadeVertices[1],primaryVertex) + << ", " << m_CascadeTools->a0(moms[0],cascadeVertices[0],cascadeVertices[1]) + << ", " << m_V0Tools->a0(cascadeVertices[0],cascadeVertices[1])); + ATH_MSG_DEBUG("a0Err " << m_CascadeTools->a0Error(moms[1],x->getCovariance()[1],cascadeVertices[1],primaryVertex) + << ", " << m_CascadeTools->a0Error(moms[0],x->getCovariance()[0],cascadeVertices[0],cascadeVertices[1]) + << ", " << m_V0Tools->a0Error(cascadeVertices[0],cascadeVertices[1])); + ATH_MSG_DEBUG("x0 " << m_V0Tools->vtx(cascadeVertices[0]).x() << " y0 " << m_V0Tools->vtx(cascadeVertices[0]).y() << " z0 " << m_V0Tools->vtx(cascadeVertices[0]).z()); + ATH_MSG_DEBUG("x1 " << m_V0Tools->vtx(cascadeVertices[1]).x() << " y1 " << m_V0Tools->vtx(cascadeVertices[1]).y() << " z1 " << m_V0Tools->vtx(cascadeVertices[1]).z()); + ATH_MSG_DEBUG("X0 " << primaryVertex->x() << " Y0 " << primaryVertex->y() << " Z0 " << primaryVertex->z()); + ATH_MSG_DEBUG("rxy0 " << m_V0Tools->rxy(cascadeVertices[0]) << " rxyErr0 " << m_V0Tools->rxyError(cascadeVertices[0])); + ATH_MSG_DEBUG("rxy1 " << m_V0Tools->rxy(cascadeVertices[1]) << " rxyErr1 " << m_V0Tools->rxyError(cascadeVertices[1])); + ATH_MSG_DEBUG("Rxy0 wrt PV " << m_V0Tools->rxy(cascadeVertices[0],primaryVertex) << " RxyErr0 wrt PV " << m_V0Tools->rxyError(cascadeVertices[0],primaryVertex)); + ATH_MSG_DEBUG("Rxy1 wrt PV " << m_V0Tools->rxy(cascadeVertices[1],primaryVertex) << " RxyErr1 wrt PV " << m_V0Tools->rxyError(cascadeVertices[1],primaryVertex)); + ATH_MSG_DEBUG("number of covariance matrices " << (x->getCovariance()).size()); + //const Amg::MatrixX& cov30 = (cascadeVertices[0])->covariancePosition(); + //const Amg::MatrixX& cov31 = (cascadeVertices[1])->covariancePosition(); + //ATH_MSG_DEBUG("cov30 " << cov30); + //ATH_MSG_DEBUG("cov31 " << cov31); + + + } // loop over cascadeinfoContainer + + // Deleting cascadeinfo since this won't be stored. + // Vertices have been kept in m_cascadeOutputs and should be owned by their container + for (auto x : cascadeinfoContainer) delete x; + + return StatusCode::SUCCESS; + } + + + JpsiPlusV0Cascade::JpsiPlusV0Cascade(const std::string& t, const std::string& n, const IInterface* p) : AthAlgTool(t,n,p), + m_vertexContainerKey(""), + m_vertexV0ContainerKey(""), + m_cascadeOutputsKeys{ "JpsiPlusV0CascadeVtx1", "JpsiPlusV0CascadeVtx2" }, + m_VxPrimaryCandidateName("PrimaryVertices"), + m_jpsiMassLower(0.0), + m_jpsiMassUpper(10000.0), + m_V0MassLower(0.0), + m_V0MassUpper(10000.0), + m_MassLower(0.0), + m_MassUpper(20000.0), + m_mass_muon ( 0 ), + m_mass_pion ( 0 ), + m_mass_proton ( 0 ), + m_mass_lambda ( 0 ), + m_mass_ks ( 0 ), + m_mass_jpsi ( 0 ), + m_mass_b0 ( 0 ), + m_mass_lambdaB( 0 ), + m_v0_pid(310), + m_constrV0(true), + m_constrJpsi(true), + m_iVertexFitter("Trk::TrkVKalVrtFitter"), + m_pvRefitter("Analysis::PrimaryVertexRefitter"), + m_V0Tools("Trk::V0Tools"), + m_CascadeTools("DerivationFramework::CascadeTools") + { + declareProperty("JpsiVertices", m_vertexContainerKey); + declareProperty("V0Vertices", m_vertexV0ContainerKey); + declareProperty("VxPrimaryCandidateName", m_VxPrimaryCandidateName); + declareProperty("RefPVContainerName", m_refPVContainerName = "RefittedPrimaryVertices"); + declareProperty("JpsiMassLowerCut", m_jpsiMassLower); + declareProperty("JpsiMassUpperCut", m_jpsiMassUpper); + declareProperty("V0MassLowerCut", m_V0MassLower); + declareProperty("V0MassUpperCut", m_V0MassUpper); + declareProperty("MassLowerCut", m_MassLower); + declareProperty("MassUpperCut", m_MassUpper); + declareProperty("HypothesisName", m_hypoName = "Bd"); + declareProperty("V0Hypothesis", m_v0_pid); + declareProperty("ApplyV0MassConstraint", m_constrV0); + declareProperty("ApplyJpsiMassConstraint", m_constrJpsi); + declareProperty("RefitPV", m_refitPV = true); + declareProperty("MaxnPV", m_PV_max = 999); + declareProperty("MinNTracksInPV", m_PV_minNTracks = 0); + declareProperty("DoVertexType", m_DoVertexType = 7); + declareProperty("TrkVertexFitterTool", m_iVertexFitter); + declareProperty("PVRefitter", m_pvRefitter); + declareProperty("V0Tools", m_V0Tools); + declareProperty("CascadeTools", m_CascadeTools); + declareProperty("CascadeVertexCollections", m_cascadeOutputsKeys); + } + + JpsiPlusV0Cascade::~JpsiPlusV0Cascade(){ } + + StatusCode JpsiPlusV0Cascade::performSearch(std::vector<Trk::VxCascadeInfo*> *cascadeinfoContainer) const + { + ATH_MSG_DEBUG( "JpsiPlusV0Cascade::performSearch" ); + assert(cascadeinfoContainer!=nullptr); + + // Get TrackParticle container (for setting links to the original tracks) + const xAOD::TrackParticleContainer *trackContainer(nullptr); + CHECK(evtStore()->retrieve(trackContainer , "InDetTrackParticles" )); + + // Get Jpsi container + const xAOD::VertexContainer *jpsiContainer(nullptr); + CHECK(evtStore()->retrieve(jpsiContainer , m_vertexContainerKey )); + + // Get V0 container + const xAOD::VertexContainer *v0Container(nullptr); + CHECK(evtStore()->retrieve(v0Container , m_vertexV0ContainerKey )); + + double mass_v0 = m_mass_ks; + std::vector<const xAOD::TrackParticle*> tracksJpsi; + std::vector<const xAOD::TrackParticle*> tracksV0; + std::vector<double> massesJpsi(2, m_mass_muon); + std::vector<double> massesV0; + std::vector<double> Masses(2, m_mass_muon); + if (m_v0_pid == 310) { + massesV0.push_back(m_mass_pion); + massesV0.push_back(m_mass_pion); + Masses.push_back(m_mass_ks); + } else if (m_v0_pid == 3122) { + massesV0.push_back(m_mass_proton); + massesV0.push_back(m_mass_pion); + mass_v0 = m_mass_lambda; + Masses.push_back(m_mass_lambda); + } else if (m_v0_pid == -3122) { + massesV0.push_back(m_mass_pion); + massesV0.push_back(m_mass_proton); + mass_v0 = m_mass_lambda; + Masses.push_back(m_mass_lambda); + } + + for(auto jpsi : *jpsiContainer) { //Iterate over Jpsi vertices + + size_t jpsiTrkNum = jpsi->nTrackParticles(); + tracksJpsi.clear(); + for( unsigned int it=0; it<jpsiTrkNum; it++) tracksJpsi.push_back(jpsi->trackParticle(it)); + + if (tracksJpsi.size() != 2 || massesJpsi.size() != 2 ) { + ATH_MSG_INFO("problems with Jpsi input"); + } + double mass_Jpsi = m_V0Tools->invariantMass(jpsi,massesJpsi); + ATH_MSG_DEBUG("Jpsi mass " << mass_Jpsi); + if (mass_Jpsi < m_jpsiMassLower || mass_Jpsi > m_jpsiMassUpper) { + ATH_MSG_DEBUG(" Original Jpsi candidate rejected by the mass cut: mass = " + << mass_Jpsi << " != (" << m_jpsiMassLower << ", " << m_jpsiMassUpper << ")" ); + continue; + } + + for(auto v0 : *v0Container) { //Iterate over V0 vertices + + size_t v0TrkNum = v0->nTrackParticles(); + tracksV0.clear(); + for( unsigned int it=0; it<v0TrkNum; it++) tracksV0.push_back(v0->trackParticle(it)); + if (tracksV0.size() != 2 || massesV0.size() != 2 ) { + ATH_MSG_INFO("problems with V0 input"); + } + double mass_V0 = m_V0Tools->invariantMass(v0,massesV0); + ATH_MSG_DEBUG("V0 mass " << mass_V0); + if (mass_V0 < m_V0MassLower || mass_V0 > m_V0MassUpper) { + ATH_MSG_DEBUG(" Original V0 candidate rejected by the mass cut: mass = " + << mass_V0 << " != (" << m_V0MassLower << ", " << m_V0MassUpper << ")" ); + continue; + } + ATH_MSG_DEBUG("using tracks" << tracksJpsi[0] << ", " << tracksJpsi[1] << ", " << tracksV0[0] << ", " << tracksV0[1]); + if(!BPhysPVCascadeTools::uniqueCollection(tracksJpsi, tracksV0)) continue; + + + //if (std::find(trackContainer->begin(), trackContainer->end(), tracksJpsi[0]) == trackContainer->end()) { + // ATH_MSG_ERROR("Track is not in standard container"); + //} else { + // ATH_MSG_DEBUG("Track " << tracksJpsi[0] << " is at position " << std::distance(trackContainer->begin(), std::find(trackContainer->begin(), trackContainer->end(), tracksJpsi[0])) ); + //} + //ATH_MSG_DEBUG("using tracks " << tracksJpsi[0] << ", " << tracksJpsi[1] << ", " << tracksV0[0] << ", " << tracksV0[1]); + + // Apply the user's settings to the fitter + // Reset + std::unique_ptr<Trk::IVKalState> state = m_iVertexFitter->makeState(); + // Robustness + int robustness = 0; + m_iVertexFitter->setRobustness(robustness, *state); + // Build up the topology + // Vertex list + std::vector<Trk::VertexID> vrtList; + // V0 vertex + Trk::VertexID vID; + if (m_constrV0) { + vID = m_iVertexFitter->startVertex(tracksV0,massesV0,*state, mass_v0); + } else { + vID = m_iVertexFitter->startVertex(tracksV0,massesV0, *state); + } + vrtList.push_back(vID); + // B vertex including Jpsi + Trk::VertexID vID2 = m_iVertexFitter->nextVertex(tracksJpsi,massesJpsi,vrtList, *state); + if (m_constrJpsi) { + std::vector<Trk::VertexID> cnstV; + cnstV.clear(); + if ( !m_iVertexFitter->addMassConstraint(vID2,tracksJpsi,cnstV,*state, m_mass_jpsi).isSuccess() ) { + ATH_MSG_WARNING("addMassConstraint failed"); + //return StatusCode::FAILURE; + } + } + // Do the work + std::unique_ptr<Trk::VxCascadeInfo> result(m_iVertexFitter->fitCascade(*state)); + + if (result != NULL) { + // reset links to original tracks + BPhysPVCascadeTools::PrepareVertexLinks(result.get(), trackContainer); + + ATH_MSG_DEBUG("storing tracks " << ((result->vertices())[0])->trackParticle(0) << ", " + << ((result->vertices())[0])->trackParticle(1) << ", " + << ((result->vertices())[1])->trackParticle(0) << ", " + << ((result->vertices())[1])->trackParticle(1)); + + // necessary to prevent memory leak + result->setSVOwnership(true); + const std::vector< std::vector<TLorentzVector> > &moms = result->getParticleMoms(); + if(moms.size() < 2){ + ATH_MSG_FATAL("Incorrect size " << __FILE__ << __LINE__ ); + return StatusCode::FAILURE; + } + double mass = m_CascadeTools->invariantMass(moms[1]); + if (mass >= m_MassLower && mass <= m_MassUpper) { + + cascadeinfoContainer->push_back(result.release()); + } else { + ATH_MSG_DEBUG("Candidate rejected by the mass cut: mass = " + << mass << " != (" << m_MassLower << ", " << m_MassUpper << ")" ); + } + } + + } //Iterate over V0 vertices + + } //Iterate over Jpsi vertices + + ATH_MSG_DEBUG("cascadeinfoContainer size " << cascadeinfoContainer->size()); + + return StatusCode::SUCCESS; + } + +} + + diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/MuonExtrapolationTool.cxx b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/MuonExtrapolationTool.cxx new file mode 100644 index 00000000000..4a55beb9601 --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/MuonExtrapolationTool.cxx @@ -0,0 +1,182 @@ +/* + Copyright (C) 2002-2017 CERN for the benefit of the ATLAS collaboration +*/ +// MuonExtrapolationTool.cxx +#include "DerivationFrameworkBPhys/MuonExtrapolationTool.h" +#include "xAODTruth/TruthParticleContainer.h" +#include "xAODEventInfo/EventInfo.h" +#include "TrkSurfaces/DiscSurface.h" +#include "TrkSurfaces/CylinderSurface.h" +#include "TrkExInterfaces/IExtrapolator.h" +#include "TVector2.h" +#include "xAODMuon/MuonContainer.h" +//********************************************************************** + +namespace DerivationFramework { + +MuonExtrapolationTool::MuonExtrapolationTool(const std::string &t, const std::string& n, const IInterface* p) + : + AthAlgTool(t, n, p), + m_extrapolator("Trk::Extrapolator/AtlasExtrapolator") +{ + declareInterface<DerivationFramework::IAugmentationTool>(this); + declareProperty("EndcapPivotPlaneZ", m_endcapPivotPlaneZ = 15525.);// z position of pivot plane in endcap region + declareProperty("EndcapPivotPlaneMinimumRadius", m_endcapPivotPlaneMinimumRadius = 0.);// minimum radius of pivot plane in endcap region + declareProperty("EndcapPivotPlaneMaximumRadius", m_endcapPivotPlaneMaximumRadius = 11977.); // maximum radius of pivot plane in endcap region + declareProperty("BarrelPivotPlaneRadius", m_barrelPivotPlaneRadius = 8000.);// radius of pivot plane in barrel region + declareProperty("BarrelPivotPlaneHalfLength", m_barrelPivotPlaneHalfLength = 9700.);// half length of pivot plane in barrel region + declareProperty("Extrapolator", m_extrapolator); + declareProperty("MuonCollection", m_muonContainerName = "Muons"); +} + +//********************************************************************** + + +StatusCode MuonExtrapolationTool::initialize() +{ + ATH_CHECK(m_extrapolator.retrieve()); + return StatusCode::SUCCESS; +} + + +//********************************************************************** + +bool MuonExtrapolationTool::extrapolateAndDecorateTrackParticle(const xAOD::TrackParticle* particle, float & eta, float & phi) const +{ + + // decorators used to access or store the information + static SG::AuxElement::Decorator< char > Decorated ("DecoratedPivotEtaPhi"); + static SG::AuxElement::Decorator< float > Eta ("EtaTriggerPivot"); + static SG::AuxElement::Decorator< float > Phi ("PhiTriggerPivot"); + + if (! Decorated.isAvailable(*particle) || !Decorated(*particle)){ + // in the athena release, we can run the extrapolation if needed + const Trk::TrackParameters* pTag = extrapolateToTriggerPivotPlane(*particle); + if(!pTag) { + Decorated(*particle) = false; + return false; + } + Eta(*particle) = pTag->position().eta(); + Phi(*particle) = pTag->position().phi(); + Decorated(*particle) = true; + delete pTag; + } + // if we get here, the decoration was either already present or just added by us + // so we can finally read the values + eta = Eta(*particle); + phi = Phi(*particle); + return true; +} + +//********************************************************************** + +const xAOD::TrackParticle* MuonExtrapolationTool::getPreferredTrackParticle (const xAOD::IParticle* muon) const +{ + if (dynamic_cast<const xAOD::TruthParticle*>(muon)){ + ATH_MSG_WARNING("Pivot plane extrapolation not supported for Truth muons!"); + return 0; + } + const xAOD::TrackParticle* muonTrack = dynamic_cast<const xAOD::TrackParticle*>(muon); + if(!muonTrack && dynamic_cast<const xAOD::Muon*>(muon)) { + const xAOD::Muon* theMuon = dynamic_cast<const xAOD::Muon*>(muon); + muonTrack = theMuon->trackParticle( xAOD::Muon::MuonSpectrometerTrackParticle ); + if(!muonTrack) { + muonTrack = theMuon->primaryTrackParticle(); + if(!muonTrack) { + muonTrack = theMuon->trackParticle( xAOD::Muon::InnerDetectorTrackParticle ); + } + } + } + if(!muonTrack){ + ATH_MSG_WARNING("no valid track found for extrapolating the muon to the pivot plane!"); + } + return muonTrack; + +} + +StatusCode MuonExtrapolationTool::addBranches() const +{ + const xAOD::MuonContainer* muons = NULL; + CHECK(evtStore()->retrieve(muons, m_muonContainerName)); + for(auto muon : *muons){ + const xAOD::TrackParticle* track = getPreferredTrackParticle(muon); + float eta, phi = 0; + if( !extrapolateAndDecorateTrackParticle( track, eta, phi )){ + if( muon->pt() > 3500.){ + //only complain if the muon has sufficient pT to actually reach the pivot plane + //extrapolation will often fail for muons with pT < 3500 MeV + ATH_MSG_WARNING("Failed to extrapolate+decorate muon with pivot plane coords - Muon params: pt "<<muon->pt()<<", eta "<< muon->eta()<<", phi "<< muon->phi()); + } + } + } + return StatusCode::SUCCESS; +} + +const Trk::TrackParameters* MuonExtrapolationTool::extrapolateToTriggerPivotPlane(const xAOD::TrackParticle& track) const +{ + // BARREL + const Trk::Perigee& perigee = track.perigeeParameters(); + + // create the barrel as a cylinder surface centered at 0,0,0 + Amg::Vector3D barrelCentre(0., 0., 0.); + Amg::Transform3D* matrix = new Amg::Transform3D(Amg::RotationMatrix3D::Identity(), barrelCentre); + + Trk::CylinderSurface* cylinder = + new Trk::CylinderSurface(matrix, + m_barrelPivotPlaneRadius, + m_barrelPivotPlaneHalfLength); + if (!cylinder) { + ATH_MSG_WARNING("extrapolateToTriggerPivotPlane :: new Trk::CylinderSurface failed."); + delete matrix; + matrix = 0; + return 0; + } + // and then attempt to extrapolate our track to this surface, checking for the boundaries of the barrel + bool boundaryCheck = true; + const Trk::Surface* surface = cylinder; + const Trk::TrackParameters* p = m_extrapolator->extrapolate(perigee, + *surface, + Trk::alongMomentum, + boundaryCheck, + Trk::muon); + delete cylinder; + // if the extrapolation worked out (so we are in the barrel) we are done and can return the + // track parameters at this surface. + if (p) return p; + + // if we get here, the muon did not cross the barrel surface + // so we assume it is going into the endcap. + // ENDCAP + + // After 2 years of using this code, we realised that ATLAS actually has endcaps on both sides ;-) + // So better make sure we place our endcap at the correct side of the detector! + // Hopefully no-one will ever read this comment... + float SignOfEta = track.eta() > 0 ? 1. : -1.; + + Amg::Vector3D endcapCentre(0., 0., m_endcapPivotPlaneZ); + // much better! + matrix = new Amg::Transform3D(Amg::RotationMatrix3D::Identity(), SignOfEta * endcapCentre); + + Trk::DiscSurface* disc = + new Trk::DiscSurface(matrix, + m_endcapPivotPlaneMinimumRadius, + m_endcapPivotPlaneMaximumRadius); + if (!disc) { + ATH_MSG_WARNING("extrapolateToTriggerPivotPlane :: new Trk::DiscSurface failed."); + delete matrix; + matrix = 0; + return 0; + } + + // for the endcap, we turn off the boundary check, extending the EC infinitely to catch stuff heading for the transition region + boundaryCheck = false; + surface = disc; + p = m_extrapolator->extrapolate(perigee, + *surface, + Trk::alongMomentum, + boundaryCheck, + Trk::muon); + delete disc; + return p; +} +} diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/ReVertex.cxx b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/ReVertex.cxx new file mode 100644 index 00000000000..9ca25240412 --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/ReVertex.cxx @@ -0,0 +1,284 @@ +/* + Copyright (C) 2002-2018 CERN for the benefit of the ATLAS collaboration +*/ +// **************************************************************************** +// ---------------------------------------------------------------------------- +// ReVertex +// +// Konstantin Beloborodov <Konstantin.Beloborodov@cern.ch> +// +// ---------------------------------------------------------------------------- +// **************************************************************************** +#include "DerivationFrameworkBPhys/ReVertex.h" +#include "xAODTracking/VertexContainer.h" +#include "xAODTracking/VertexAuxContainer.h" +#include "JpsiUpsilonTools/PrimaryVertexRefitter.h" +#include "JpsiUpsilonTools/JpsiUpsilonCommon.h" + +#include "TrkVertexAnalysisUtils/V0Tools.h" +#include "BeamSpotConditionsData/BeamSpotData.h" +#include "DerivationFrameworkBPhys/BPhysPVTools.h" +#include "TrkVertexFitterInterfaces/IVertexFitter.h" +#include "TrkVKalVrtFitter/TrkVKalVrtFitter.h" +#include "InDetConversionFinderTools/VertexPointEstimator.h" +#include "xAODBPhys/BPhysHypoHelper.h" + +using namespace DerivationFramework; + +ReVertex::ReVertex(const std::string& t, + const std::string& n, + const IInterface* p) : + AthAlgTool(t,n,p), m_vertexEstimator("InDet::VertexPointEstimator"), m_iVertexFitter("Trk::TrkVKalVrtFitter"), + m_massConst(0.), + m_totalMassConst(0.), + m_v0Tools("Trk::V0Tools"), + m_pvRefitter("Analysis::PrimaryVertexRefitter"), + m_doMassConst(false), + m_vertexFittingWithPV(false), + m_chi2cut(-1.0), + m_trkDeltaZ(-1.0), + m_useAdditionalTrack(false) +{ + + declareInterface<DerivationFramework::IAugmentationTool>(this); + declareProperty("TrackIndices", m_TrackIndices); + declareProperty("TrkVertexFitterTool", m_iVertexFitter); + declareProperty("VertexPointEstimator",m_vertexEstimator); + + declareProperty("OutputVtxContainerName", m_OutputContainerName); + declareProperty("InputVtxContainerName", m_inputContainerName); + declareProperty("TrackContainerName", m_trackContainer = "InDetTrackParticles"); + declareProperty("UseVertexFittingWithPV", m_vertexFittingWithPV); + + declareProperty("HypothesisNames",m_hypoNames); + + declareProperty("V0Tools" , m_v0Tools); + declareProperty("PVRefitter" , m_pvRefitter); + declareProperty("PVContainerName" , m_pvContainerName = "PrimaryVertices"); + declareProperty("RefPVContainerName" , m_refPVContainerName = "RefittedPrimaryVertices"); + + declareProperty("UseMassConstraint", m_doMassConst); + declareProperty("VertexMass", m_totalMassConst); + declareProperty("SubVertexMass", m_massConst); + declareProperty("MassInputParticles", m_trkMasses); + declareProperty("SubVertexTrackIndices", m_indices); + + declareProperty("UseAdditionalTrack", m_useAdditionalTrack); + + declareProperty("RefitPV" , m_refitPV = false); + //This parameter will allow us to optimize the number of PVs under consideration as the probability + //of a useful primary vertex drops significantly the higher you go + declareProperty("MaxPVrefit" , m_PV_max = 1000); + declareProperty("DoVertexType" , m_DoVertexType = 7); + // minimum number of tracks for PV to be considered for PV association + declareProperty("MinNTracksInPV" , m_PV_minNTracks = 0); + declareProperty("Do3d" , m_do3d = false); + declareProperty("AddPVData" , m_AddPVData = true); + declareProperty("StartingPoint0" , m_startingpoint0 = false); + declareProperty("BMassUpper",m_BMassUpper = std::numeric_limits<double>::max() ); + declareProperty("BMassLower",m_BMassLower = std::numeric_limits<double>::min() ); + declareProperty("Chi2Cut",m_chi2cut = std::numeric_limits<double>::max() ); + declareProperty("TrkDeltaZ",m_trkDeltaZ); + + +} + +StatusCode ReVertex::initialize() { + ATH_MSG_DEBUG("in initialize()"); + if(m_TrackIndices.empty()) { + ATH_MSG_FATAL("No Indices provided"); + return StatusCode::FAILURE; + } + m_VKVFitter = dynamic_cast<Trk::TrkVKalVrtFitter*>(&(*m_iVertexFitter)); + if(m_VKVFitter==nullptr) return StatusCode::FAILURE; + ATH_CHECK(m_OutputContainerName.initialize()); + ATH_CHECK(m_inputContainerName.initialize()); + ATH_CHECK(m_trackContainer.initialize()); + ATH_CHECK(m_pvContainerName.initialize()); + ATH_CHECK(m_refPVContainerName.initialize()); + + return StatusCode::SUCCESS; +} + + +StatusCode ReVertex::addBranches() const { + + SG::WriteHandle<xAOD::VertexContainer> vtxContainer(m_OutputContainerName); + ATH_CHECK(vtxContainer.record(std::make_unique<xAOD::VertexContainer>(), std::make_unique<xAOD::VertexAuxContainer>())); + + const size_t Ntracks = m_TrackIndices.size(); + + SG::ReadHandle<xAOD::VertexContainer> InVtxContainer(m_inputContainerName); + SG::ReadHandle<xAOD::TrackParticleContainer> importedTrackCollection(m_trackContainer); + ATH_CHECK(InVtxContainer.isValid()); + ATH_CHECK(importedTrackCollection.isValid()); + //---------------------------------------------------- + // retrieve primary vertices + //---------------------------------------------------- + SG::ReadHandle<xAOD::VertexContainer> pvContainer(m_pvContainerName); + ATH_CHECK(pvContainer.isValid()); + + std::vector<const xAOD::TrackParticle*> fitpair(Ntracks + m_useAdditionalTrack); + for(const xAOD::Vertex* v : *InVtxContainer) + { + + bool passed = false; + for(size_t i=0;i<m_hypoNames.size();i++) { + xAOD::BPhysHypoHelper onia(m_hypoNames.at(i), v); + passed |= onia.pass(); + } + if (!passed && m_hypoNames.size()) continue; + + for(size_t i =0; i<Ntracks; i++) + { + size_t trackN = m_TrackIndices[i]; + if(trackN >= v->nTrackParticles()) + { + ATH_MSG_FATAL("Indices exceeds limit in particle"); + return StatusCode::FAILURE; + } + fitpair[i] = v->trackParticle(trackN); + } + + if (m_useAdditionalTrack) + { + // Loop over ID tracks, call vertexing + for (auto trkItr=importedTrackCollection->cbegin(); trkItr!=importedTrackCollection->cend(); ++trkItr) { + const xAOD::TrackParticle* tp (*trkItr); + fitpair.back() = nullptr; + if (Analysis::JpsiUpsilonCommon::isContainedIn(tp,fitpair)) continue; // remove tracks which were used to build J/psi+2Tracks + fitpair.back() = tp; + + // Daniel Scheirich: remove track too far from the Jpsi+2Tracks vertex (DeltaZ cut) + if(m_trkDeltaZ>0 && + std::abs((tp)->z0() + (tp)->vz() - v->z()) > m_trkDeltaZ ) + continue; + + fitAndStore(vtxContainer.ptr(),v,InVtxContainer.cptr(),fitpair,importedTrackCollection.cptr(),pvContainer.cptr()); + } + } + else + { + fitAndStore(vtxContainer.ptr(),v,InVtxContainer.cptr(),fitpair,importedTrackCollection.cptr(),pvContainer.cptr()); + } + } + + if(m_AddPVData){ + // Give the helper class the ptr to v0tools and beamSpotsSvc to use + SG::ReadCondHandle<InDet::BeamSpotData> beamSpotHandle { m_beamSpotKey }; + if(not beamSpotHandle.isValid()) ATH_MSG_ERROR("Cannot Retrieve " << m_beamSpotKey.key() ); + BPhysPVTools helper(&(*m_v0Tools), beamSpotHandle.cptr()); + helper.SetMinNTracksInPV(m_PV_minNTracks); + helper.SetSave3d(m_do3d); + + if(m_refitPV) { + //---------------------------------------------------- + // Try to retrieve refitted primary vertices + //---------------------------------------------------- + SG::WriteHandle<xAOD::VertexContainer> refPvContainer(m_refPVContainerName); + ATH_CHECK(refPvContainer.record(std::make_unique<xAOD::VertexContainer>(), std::make_unique<xAOD::VertexAuxContainer>())); + + if(vtxContainer->size() >0){ + ATH_CHECK(helper.FillCandwithRefittedVertices(vtxContainer.ptr(), pvContainer.cptr(), refPvContainer.ptr(), &(*m_pvRefitter) , m_PV_max, m_DoVertexType)); + } + }else{ + if(vtxContainer->size() >0) ATH_CHECK(helper.FillCandExistingVertices(vtxContainer.ptr(), pvContainer.cptr(), m_DoVertexType)); + } + } + + return StatusCode::SUCCESS; +} + +void ReVertex::fitAndStore(xAOD::VertexContainer* vtxContainer, + const xAOD::Vertex* v, + const xAOD::VertexContainer *InVtxContainer, + const std::vector<const xAOD::TrackParticle*> &inputTracks, + const xAOD::TrackParticleContainer* importedTrackCollection, + const xAOD::VertexContainer* pvContainer) const +{ + std::unique_ptr<xAOD::Vertex> ptr(fit(inputTracks, nullptr)); + if(!ptr)return; + + double chi2DOF = ptr->chiSquared()/ptr->numberDoF(); + ATH_MSG_DEBUG("Candidate chi2/DOF is " << chi2DOF); + bool chi2CutPassed = (m_chi2cut <= 0.0 || chi2DOF < m_chi2cut); + if(!chi2CutPassed) { ATH_MSG_DEBUG("Chi Cut failed!"); return; } + xAOD::BPhysHelper bHelper(ptr.get());//"get" does not "release" still automatically deleted + bHelper.setRefTrks(); + if (m_trkMasses.size()==inputTracks.size()) { + TLorentzVector bMomentum = bHelper.totalP(m_trkMasses); + double bMass = bMomentum.M(); + bool passesCuts = (m_BMassUpper > bMass && bMass > m_BMassLower); + if(!passesCuts)return; + } + + DerivationFramework::BPhysPVTools::PrepareVertexLinks( ptr.get(), importedTrackCollection ); + std::vector<const xAOD::Vertex*> thePreceding; + thePreceding.push_back(v); + if(m_vertexFittingWithPV){ + const xAOD::Vertex* closestPV = Analysis::JpsiUpsilonCommon::ClosestPV(bHelper, pvContainer); + if (!closestPV) return; + std::unique_ptr<xAOD::Vertex> ptrPV(fit(inputTracks, closestPV)); + if(!ptrPV) return; + + double chi2DOFPV = ptrPV->chiSquared()/ptrPV->numberDoF(); + ATH_MSG_DEBUG("CandidatePV chi2/DOF is " << chi2DOFPV); + bool chi2CutPassed = (m_chi2cut <= 0.0 || chi2DOFPV < m_chi2cut); + if(!chi2CutPassed) { ATH_MSG_DEBUG("Chi Cut failed!"); return; } + xAOD::BPhysHelper bHelperPV(ptrPV.get());//"get" does not "release" still automatically deleted + bHelperPV.setRefTrks(); + if (m_trkMasses.size()==inputTracks.size()) { + TLorentzVector bMomentumPV = bHelperPV.totalP(m_trkMasses); + double bMass = bMomentumPV.M(); + bool passesCuts = (m_BMassUpper > bMass && bMass > m_BMassLower); + if(!passesCuts)return; + } + + bHelperPV.setPrecedingVertices(thePreceding, InVtxContainer); + vtxContainer->push_back(ptrPV.release()); + return; //Don't store other vertex + } + bHelper.setPrecedingVertices(thePreceding, InVtxContainer); + vtxContainer->push_back(ptr.release()); +} + + // ********************************************************************************* + + // --------------------------------------------------------------------------------- + // fit - does the fit + // --------------------------------------------------------------------------------- + +xAOD::Vertex* ReVertex::fit(const std::vector<const xAOD::TrackParticle*> &inputTracks, + const xAOD::Vertex* pv) const +{ + std::unique_ptr<Trk::IVKalState> state = m_VKVFitter->makeState(); + if (m_doMassConst && (m_trkMasses.size()==inputTracks.size())) { + m_VKVFitter->setMassInputParticles(m_trkMasses, *state); + if (m_totalMassConst) m_VKVFitter->setMassForConstraint(m_totalMassConst, *state); + if (m_massConst) m_VKVFitter->setMassForConstraint(m_massConst, m_indices, *state); + } + if (pv) { + m_VKVFitter->setCnstType(8, *state); + m_VKVFitter->setVertexForConstraint(pv->position().x(), + pv->position().y(), + pv->position().z(), *state); + m_VKVFitter->setCovVrtForConstraint(pv->covariancePosition()(Trk::x,Trk::x), + pv->covariancePosition()(Trk::y,Trk::x), + pv->covariancePosition()(Trk::y,Trk::y), + pv->covariancePosition()(Trk::z,Trk::x), + pv->covariancePosition()(Trk::z,Trk::y), + pv->covariancePosition()(Trk::z,Trk::z), *state ); + } + + // Do the fit itself....... + // Starting point (use the J/psi position) + const Trk::Perigee& aPerigee1 = inputTracks[0]->perigeeParameters(); + const Trk::Perigee& aPerigee2 = inputTracks[1]->perigeeParameters(); + int sflag = 0; + int errorcode = 0; + Amg::Vector3D startingPoint = m_vertexEstimator->getCirclesIntersectionPoint(&aPerigee1,&aPerigee2,sflag,errorcode); + if (errorcode != 0) {startingPoint(0) = 0.0; startingPoint(1) = 0.0; startingPoint(2) = 0.0;} + xAOD::Vertex* theResult = m_VKVFitter->fit(inputTracks, startingPoint, *state); + + return theResult; +} diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/Reco_4mu.cxx b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/Reco_4mu.cxx new file mode 100644 index 00000000000..4d4a7c80258 --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/Reco_4mu.cxx @@ -0,0 +1,269 @@ +/* + Copyright (C) 2002-2017 CERN for the benefit of the ATLAS collaboration +*/ + +///////////////////////////////////////////////////////////////// +// Reco_4mu.cxx +/////////////////////////////////////////////////////////////////// +// Author: James Catmore <james.catmore@cern.ch> + +#include "DerivationFrameworkBPhys/Reco_4mu.h" +#include "xAODTracking/VertexContainer.h" +#include "xAODTracking/VertexAuxContainer.h" +#include "TrkVertexAnalysisUtils/V0Tools.h" +#include "DerivationFrameworkBPhys/BPhysPVTools.h" +#include "xAODBPhys/BPhysHypoHelper.h" + +namespace DerivationFramework { + + Reco_4mu::Reco_4mu(const std::string& t, + const std::string& n, + const IInterface* p) : + AthAlgTool(t,n,p), + m_v0Tools("Trk::V0Tools"), + m_fourMuonTool("DerivationFramework::FourMuonTool"), + m_pvRefitter("Analysis::PrimaryVertexRefitter") + { + declareInterface<DerivationFramework::ISkimmingTool>(this); + + // Declare tools + declareProperty("V0Tools" , m_v0Tools); + declareProperty("FourMuonTool", m_fourMuonTool); + declareProperty("PVRefitter", m_pvRefitter); + + // Declare user-defined properties + declareProperty("PairContainerName" , m_pairName = "Pairs"); + declareProperty("QuadrupletContainerName", m_quadName = "Quadruplets"); + declareProperty("PVContainerName" , m_pvContainerName = "PrimaryVertices"); + declareProperty("RefPVContainerName" , m_refPVContainerName = "RefittedPrimaryVertices"); + declareProperty("RefitPV" , m_refitPV = false); + declareProperty("MaxPVrefit" , m_PV_max = 1); + declareProperty("DoVertexType" , m_DoVertexType = 1); + } + + // * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * + + StatusCode Reco_4mu::initialize() + { + + ATH_MSG_DEBUG("in initialize()"); + + // retrieve V0 tools + CHECK( m_v0Tools.retrieve() ); + + // get the JpsiFinder tool + CHECK( m_fourMuonTool.retrieve() ); + + // get the PrimaryVertexRefitter tool + CHECK( m_pvRefitter.retrieve() ); + + return StatusCode::SUCCESS; + + } + + // * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * + + StatusCode Reco_4mu::finalize() + { + // everything all right + return StatusCode::SUCCESS; + } + + // * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * + + bool Reco_4mu::eventPassesFilter() const + { + // Output containers and its auxilliary store + xAOD::VertexContainer* pairContainer = NULL; + xAOD::VertexAuxContainer* pairAuxContainer = NULL; + xAOD::VertexContainer* quadContainer = NULL; + xAOD::VertexAuxContainer* quadAuxContainer = NULL; + bool acceptEvent(false); + //---------------------------------------------------- + // call finder + //---------------------------------------------------- + if( !m_fourMuonTool->performSearch(pairContainer, pairAuxContainer, quadContainer, quadAuxContainer, acceptEvent).isSuccess() ) { + ATH_MSG_FATAL("4mu tool (" << m_fourMuonTool << ") failed."); + return(false); + } + + //---------------------------------------------------- + // event selection + //---------------------------------------------------- + /*if (quadContainer->size()==0) { + if (pairContainer!=NULL) delete pairContainer; + if (pairAuxContainer!=NULL) delete pairAuxContainer; + if (quadContainer!=NULL) delete quadContainer; + if (quadAuxContainer!=NULL) delete quadAuxContainer; + return(acceptEvent); + // acceptEvent based on muon selection only, not quads + }*/ + + //---------------------------------------------------- + // retrieve primary vertices + //---------------------------------------------------- + const xAOD::VertexContainer* pvContainer = NULL; + auto sc = evtStore()->retrieve(pvContainer, m_pvContainerName); + if(sc.isFailure()){ + ATH_MSG_FATAL("Cannot find PV Container"); + return false; + } + //---------------------------------------------------- + // Refit primary vertices + //---------------------------------------------------- + xAOD::VertexContainer* refPvContainer = NULL; + xAOD::VertexAuxContainer* refPvAuxContainer = NULL; + + if(m_refitPV) { + refPvContainer = new xAOD::VertexContainer; + refPvAuxContainer = new xAOD::VertexAuxContainer; + refPvContainer->setStore(refPvAuxContainer); + } + + BPhysPVTools helper(&(*m_v0Tools));//Give the helper class the ptr to v0tools to use + + if(m_refitPV){ + if(quadContainer->size() >0){ + StatusCode SC = helper.FillCandwithRefittedVertices(quadContainer, pvContainer, refPvContainer, &(*m_pvRefitter) , m_PV_max, m_DoVertexType); + if(SC.isFailure()){ + ATH_MSG_FATAL("refitting failed - check the vertices you passed"); + return false; + } + } + if(pairContainer->size()>0) { + StatusCode SC = helper.FillCandwithRefittedVertices(pairContainer, pvContainer, refPvContainer, &(*m_pvRefitter) , m_PV_max, m_DoVertexType); + if(SC.isFailure()){ + ATH_MSG_FATAL("refitting failed - check the vertices you passed"); + return false; + } + } + }else{ + if(quadContainer->size() >0){ + auto sc = helper.FillCandExistingVertices(quadContainer, pvContainer, m_DoVertexType); + sc.ignore(); + } + if(pairContainer->size() >0) + { + auto sc = helper.FillCandExistingVertices(pairContainer, pvContainer, m_DoVertexType); + sc.ignore(); + } + } + + //---------------------------------------------------- + // Mass-hypothesis dependent quantities + //---------------------------------------------------- + + std::vector<double> muonPairMasses = std::vector<double>(2, 105.658); + std::vector<double> muonQuadMasses = std::vector<double>(4, 105.658); + + bool doPt = (m_DoVertexType & 1) != 0; + bool doA0 = (m_DoVertexType & 2) != 0; + bool doZ0 = (m_DoVertexType & 4) != 0; + bool doZ0BA = (m_DoVertexType & 8) != 0; + + // loop over pairs + xAOD::VertexContainer::iterator pairItr = pairContainer->begin(); + ATH_MSG_DEBUG("Indices/masses of pairs follows...."); + for(; pairItr!=pairContainer->end(); ++pairItr) { + // create BPhysHypoHelper + xAOD::BPhysHypoHelper pairHelper("PAIR", *pairItr); + + //---------------------------------------------------- + // decorate the vertex + //---------------------------------------------------- + // a) invariant mass and error + if( !pairHelper.setMass(muonPairMasses) ) ATH_MSG_WARNING("Decoration pair.setMass failed"); + + double massErr = m_v0Tools->invariantMassError(pairHelper.vtx(), muonPairMasses); + if( !pairHelper.setMassErr(massErr) ) ATH_MSG_WARNING("Decoration pair.setMassErr failed"); + + // b) proper decay time and error: + // retrieve the refitted PV (or the original one, if the PV refitting was turned off) + if(doPt) ProcessVertex(pairHelper, xAOD::BPhysHelper::PV_MAX_SUM_PT2, muonPairMasses); + if(doA0) ProcessVertex(pairHelper, xAOD::BPhysHelper::PV_MIN_A0, muonPairMasses); + if(doZ0) ProcessVertex(pairHelper, xAOD::BPhysHelper::PV_MIN_Z0, muonPairMasses); + if(doZ0BA) ProcessVertex(pairHelper, xAOD::BPhysHelper::PV_MIN_Z0_BA, muonPairMasses); + ATH_MSG_DEBUG((*pairItr)->auxdata<std::string>("CombinationCode") << " : " << pairHelper.mass() << " +/- " << pairHelper.massErr()); + } + + // loop over quadruplets + xAOD::VertexContainer::iterator quadItr = quadContainer->begin(); + ATH_MSG_DEBUG("Indices/masses of quadruplets follows...."); + for(; quadItr!=quadContainer->end(); ++quadItr) { + // create BPhysHypoHelper + xAOD::BPhysHypoHelper quadHelper("QUAD", *quadItr); + + //---------------------------------------------------- + // decorate the vertex + //---------------------------------------------------- + // a) invariant mass and error + if( !quadHelper.setMass(muonQuadMasses) ) ATH_MSG_WARNING("Decoration quad.setMass failed"); + + double massErr = m_v0Tools->invariantMassError(quadHelper.vtx(), muonQuadMasses); + if( !quadHelper.setMassErr(massErr) ) ATH_MSG_WARNING("Decoration quad.setMassErr failed"); + + // b) proper decay time and error: + // retrieve the refitted PV (or the original one, if the PV refitting was turned off) + if(doPt) ProcessVertex(quadHelper, xAOD::BPhysHelper::PV_MAX_SUM_PT2, muonQuadMasses); + if(doA0) ProcessVertex(quadHelper, xAOD::BPhysHelper::PV_MIN_A0, muonQuadMasses); + if(doZ0) ProcessVertex(quadHelper, xAOD::BPhysHelper::PV_MIN_Z0, muonQuadMasses); + if(doZ0BA) ProcessVertex(quadHelper, xAOD::BPhysHelper::PV_MIN_Z0_BA, muonQuadMasses); + ATH_MSG_DEBUG((*quadItr)->auxdata<std::string>("CombinationCode") << " : " << quadHelper.mass() << " +/- " << quadHelper.massErr()); + } + + //---------------------------------------------------- + // save in the StoreGate + //---------------------------------------------------- + // Pairs + if (!evtStore()->contains<xAOD::VertexContainer>(m_pairName)) + evtStore()->record(pairContainer, m_pairName).ignore(); + if (!evtStore()->contains<xAOD::VertexAuxContainer>(m_pairName+"Aux.")) + evtStore()->record(pairAuxContainer, m_pairName+"Aux.").ignore(); + + // Quads + if (!evtStore()->contains<xAOD::VertexContainer>(m_quadName)) + evtStore()->record(quadContainer, m_quadName).ignore(); + if (!evtStore()->contains<xAOD::VertexAuxContainer>(m_quadName+"Aux.")) + evtStore()->record(quadAuxContainer, m_quadName+"Aux.").ignore(); + + // Refitted PVs + if(m_refitPV) { + evtStore()->record(refPvContainer , m_refPVContainerName).ignore(); + evtStore()->record(refPvAuxContainer, m_refPVContainerName+"Aux.").ignore(); + } + + return(acceptEvent); + } + + + void Reco_4mu::ProcessVertex(xAOD::BPhysHypoHelper &hypoHelper, xAOD::BPhysHelper::pv_type pv_t, std::vector<double> trackMasses) const{ + + const xAOD::Vertex* pv = hypoHelper.pv(pv_t); + if(pv) { + // decorate the vertex. + + BPHYS_CHECK( hypoHelper.setTau( m_v0Tools->tau(hypoHelper.vtx(), pv, trackMasses), + pv_t, + xAOD::BPhysHypoHelper::TAU_INV_MASS) ); + + BPHYS_CHECK( hypoHelper.setTauErr( m_v0Tools->tauError(hypoHelper.vtx(), pv, trackMasses), + pv_t, + xAOD::BPhysHypoHelper::TAU_INV_MASS) ); + + //enum pv_type {PV_MAX_SUM_PT2, PV_MIN_A0, PV_MIN_Z0, PV_MIN_Z0BA}; + }else{ + + const float errConst = -9999999; + BPHYS_CHECK( hypoHelper.setTau( errConst, + pv_t, + xAOD::BPhysHypoHelper::TAU_INV_MASS) ); + + BPHYS_CHECK( hypoHelper.setTauErr( errConst, + pv_t, + xAOD::BPhysHypoHelper::TAU_INV_MASS) ); + } + + return; + } + +} diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/Reco_V0Finder.cxx b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/Reco_V0Finder.cxx new file mode 100644 index 00000000000..0b19d017b13 --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/Reco_V0Finder.cxx @@ -0,0 +1,307 @@ +/* + Copyright (C) 2002-2017 CERN for the benefit of the ATLAS collaboration +*/ +///////////////////////////////////////////////////////////////// +// Reco_V0Finder.cxx, (c) ATLAS Detector software +/////////////////////////////////////////////////////////////////// +// Author: Adam Barton +#include "DerivationFrameworkBPhys/Reco_V0Finder.h" +#include "xAODTracking/VertexContainer.h" +#include "xAODTracking/VertexAuxContainer.h" +#include "TrkVertexAnalysisUtils/V0Tools.h" +#include "GaudiKernel/IPartPropSvc.h" +#include "EventKernel/PdtPdg.h" + +namespace DerivationFramework { + + Reco_V0Finder::Reco_V0Finder(const std::string& t, + const std::string& n, + const IInterface* p) : + AthAlgTool(t,n,p), + m_v0FinderTool("InDet::V0FinderTool"), + m_V0Tools("Trk::V0Tools"), + m_particleDataTable(nullptr), + m_masses(1), + m_masspi(139.57), + m_massp(938.272), + m_masse(0.510999), + m_massK0S(497.672), + m_massLambda(1115.68), + m_VxPrimaryCandidateName("PrimaryVertices"), + m_v0ContainerName("RecoV0Candidates"), + m_ksContainerName("RecoKshortCandidates"), + m_laContainerName("RecoLambdaCandidates"), + m_lbContainerName("RecoLambdabarCandidates") + { + declareInterface<DerivationFramework::IAugmentationTool>(this); + + // Declare user-defined properties + declareProperty("CheckVertexContainers", m_CollectionsToCheck); + declareProperty("V0FinderTool", m_v0FinderTool); + declareProperty("V0Tools", m_V0Tools); + declareProperty("masses", m_masses); + declareProperty("masspi", m_masspi); + declareProperty("massp", m_massp); + declareProperty("masse", m_masse); + declareProperty("massK0S", m_massK0S); + declareProperty("massLambda", m_massLambda); + declareProperty("VxPrimaryCandidateName", m_VxPrimaryCandidateName); + declareProperty("V0ContainerName", m_v0ContainerName); + declareProperty("KshortContainerName", m_ksContainerName); + declareProperty("LambdaContainerName", m_laContainerName); + declareProperty("LambdabarContainerName", m_lbContainerName); + } + + // * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * + + StatusCode Reco_V0Finder::initialize() + { + + ATH_MSG_DEBUG("in initialize()"); + + // get the V0Finder tool + ATH_CHECK( m_v0FinderTool.retrieve()); + + // uploading the V0 tools + ATH_CHECK( m_V0Tools.retrieve()); + + // get the Particle Properties Service + IPartPropSvc* partPropSvc = nullptr; + StatusCode sc = service("PartPropSvc", partPropSvc, true); + if (sc.isFailure()) { + msg(MSG::ERROR) << "Could not initialize Particle Properties Service" << endmsg; + return StatusCode::FAILURE; + } + m_particleDataTable = partPropSvc->PDT(); + + const HepPDT::ParticleData* pd_pi = m_particleDataTable->particle(PDG::pi_plus); + const HepPDT::ParticleData* pd_p = m_particleDataTable->particle(PDG::p_plus); + const HepPDT::ParticleData* pd_e = m_particleDataTable->particle(PDG::e_minus); + const HepPDT::ParticleData* pd_K = m_particleDataTable->particle(PDG::K_S0); + const HepPDT::ParticleData* pd_L = m_particleDataTable->particle(PDG::Lambda0); + if (m_masses == 1) { + m_masspi = pd_pi->mass(); + m_massp = pd_p->mass(); + m_masse = pd_e->mass(); + m_massK0S = pd_K->mass(); + m_massLambda = pd_L->mass(); + } + + + return StatusCode::SUCCESS; + + } + + // * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * + + StatusCode Reco_V0Finder::finalize() + { + // everything all right + return StatusCode::SUCCESS; + } + + // * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * + + StatusCode Reco_V0Finder::addBranches() const + { + bool callV0Finder = false; + // Jpsi container and its auxilliary store + for(const auto &str : m_CollectionsToCheck){ + const xAOD::VertexContainer* vertContainer = nullptr; + ATH_CHECK( evtStore()->retrieve(vertContainer, str) ); + if(vertContainer->size() == 0) { + ATH_MSG_DEBUG("Container VertexContainer (" << str << ") is empty"); + }else{ + callV0Finder = true; + ATH_MSG_DEBUG("Container VertexContainer (" << str << ") has events N= " << vertContainer->size()); + break;//No point checking other containers + } + } + + // Call V0Finder + + +// InDetV0 container and its auxilliary store + xAOD::VertexContainer* v0Container(nullptr); + xAOD::VertexAuxContainer* v0AuxContainer(nullptr); + xAOD::VertexContainer* ksContainer(nullptr); + xAOD::VertexAuxContainer* ksAuxContainer(nullptr); + xAOD::VertexContainer* laContainer(nullptr); + xAOD::VertexAuxContainer* laAuxContainer(nullptr); + xAOD::VertexContainer* lbContainer(nullptr); + xAOD::VertexAuxContainer* lbAuxContainer(nullptr); + + if (callV0Finder) { + + // Get primary vertex from StoreGate + const xAOD::Vertex * primaryVertex(0); + const xAOD::VertexContainer * importedVxContainer(0); + ATH_CHECK( evtStore()->retrieve(importedVxContainer, m_VxPrimaryCandidateName) ); + ATH_MSG_DEBUG("Found " << m_VxPrimaryCandidateName << " in StoreGate!"); + if (importedVxContainer->size()==0){ + ATH_MSG_WARNING("You have no primary vertices: " << importedVxContainer->size()); + } else { + primaryVertex = (*importedVxContainer)[0]; + } + + ATH_CHECK(m_v0FinderTool->performSearch(v0Container, v0AuxContainer, ksContainer, ksAuxContainer, laContainer, laAuxContainer, lbContainer, lbAuxContainer, primaryVertex, importedVxContainer)); + + ATH_MSG_DEBUG("Reco_V0Finder v0Container->size() " << v0Container->size()); + ATH_MSG_DEBUG("Reco_V0Finder ksContainer->size() " << ksContainer->size()); + ATH_MSG_DEBUG("Reco_V0Finder laContainer->size() " << laContainer->size()); + ATH_MSG_DEBUG("Reco_V0Finder lbContainer->size() " << lbContainer->size()); + + SG::AuxElement::Decorator<float> mDecor_Ksmass("Kshort_mass"); + SG::AuxElement::Decorator<float> mDecor_Ksmasse("Kshort_massError"); + SG::AuxElement::Decorator<float> mDecor_Lamass("Lambda_mass"); + SG::AuxElement::Decorator<float> mDecor_Lamasse("Lambda_massError"); + SG::AuxElement::Decorator<float> mDecor_Lbmass("Lambdabar_mass"); + SG::AuxElement::Decorator<float> mDecor_Lbmasse("Lambdabar_massError"); + SG::AuxElement::Decorator<float> mDecor_mass("mass"); + SG::AuxElement::Decorator<float> mDecor_massError("massError"); + SG::AuxElement::Decorator<float> mDecor_pt("pT"); + SG::AuxElement::Decorator<float> mDecor_ptError("pTError"); + SG::AuxElement::Decorator<float> mDecor_rxy("Rxy"); + SG::AuxElement::Decorator<float> mDecor_rxyError("RxyError"); + SG::AuxElement::Decorator<float> mDecor_px("px"); + SG::AuxElement::Decorator<float> mDecor_py("py"); + SG::AuxElement::Decorator<float> mDecor_pz("pz"); + + xAOD::VertexContainer::const_iterator v0Itr = v0Container->begin(); + for ( v0Itr=v0Container->begin(); v0Itr!=v0Container->end(); ++v0Itr ) + { + const xAOD::Vertex * unconstrV0 = (*v0Itr); + double mass_ks = m_V0Tools->invariantMass(unconstrV0,m_masspi,m_masspi); + double mass_error_ks = m_V0Tools->invariantMassError(unconstrV0,m_masspi,m_masspi); + double mass_la = m_V0Tools->invariantMass(unconstrV0,m_massp,m_masspi); + double mass_error_la = m_V0Tools->invariantMassError(unconstrV0,m_massp,m_masspi); + double mass_lb = m_V0Tools->invariantMass(unconstrV0,m_masspi,m_massp); + double mass_error_lb = m_V0Tools->invariantMassError(unconstrV0,m_masspi,m_massp); + double pt = m_V0Tools->pT(unconstrV0); + double ptError = m_V0Tools->pTError(unconstrV0); + double rxy = m_V0Tools->rxy(unconstrV0); + double rxyError = m_V0Tools->rxyError(unconstrV0); + Amg::Vector3D momentum = m_V0Tools->V0Momentum(unconstrV0); + mDecor_Ksmass( *unconstrV0 ) = mass_ks; + mDecor_Ksmasse( *unconstrV0 ) = mass_error_ks; + mDecor_Lamass( *unconstrV0 ) = mass_la; + mDecor_Lamasse( *unconstrV0 ) = mass_error_la; + mDecor_Lbmass( *unconstrV0 ) = mass_lb; + mDecor_Lbmasse( *unconstrV0 ) = mass_error_lb; + mDecor_pt( *unconstrV0 ) = pt; + mDecor_ptError( *unconstrV0 ) = ptError; + mDecor_rxy( *unconstrV0 ) = rxy; + mDecor_rxyError( *unconstrV0 ) = rxyError; + mDecor_px( *unconstrV0 ) = momentum.x(); + mDecor_py( *unconstrV0 ) = momentum.y(); + mDecor_pz( *unconstrV0 ) = momentum.z(); + ATH_MSG_DEBUG("Reco_V0Finder mass_ks " << mass_ks << " mass_la " << mass_la << " mass_lb " << mass_lb); + } + xAOD::VertexContainer::const_iterator ksItr = ksContainer->begin(); + for ( ksItr=ksContainer->begin(); ksItr!=ksContainer->end(); ++ksItr ) + { + const xAOD::Vertex * ksV0 = (*ksItr); + double mass_ks = m_V0Tools->invariantMass(ksV0,m_masspi,m_masspi); + double mass_error_ks = m_V0Tools->invariantMassError(ksV0,m_masspi,m_masspi); + double pt = m_V0Tools->pT(ksV0); + double ptError = m_V0Tools->pTError(ksV0); + double rxy = m_V0Tools->rxy(ksV0); + double rxyError = m_V0Tools->rxyError(ksV0); + Amg::Vector3D momentum = m_V0Tools->V0Momentum(ksV0); + mDecor_mass( *ksV0 ) = mass_ks; + mDecor_massError( *ksV0 ) = mass_error_ks; + mDecor_pt( *ksV0 ) = pt; + mDecor_ptError( *ksV0 ) = ptError; + mDecor_rxy( *ksV0 ) = rxy; + mDecor_rxyError( *ksV0 ) = rxyError; + mDecor_px( *ksV0 ) = momentum.x(); + mDecor_py( *ksV0 ) = momentum.y(); + mDecor_pz( *ksV0 ) = momentum.z(); + ATH_MSG_DEBUG("Reco_V0Finder mass_ks " << mass_ks << " mass_error_ks " << mass_error_ks << " pt " << pt << " rxy " << rxy); + } + xAOD::VertexContainer::const_iterator laItr = laContainer->begin(); + for ( laItr=laContainer->begin(); laItr!=laContainer->end(); ++laItr ) + { + const xAOD::Vertex * laV0 = (*laItr); + double mass_la = m_V0Tools->invariantMass(laV0,m_massp,m_masspi); + double mass_error_la = m_V0Tools->invariantMassError(laV0,m_massp,m_masspi); + double pt = m_V0Tools->pT(laV0); + double ptError = m_V0Tools->pTError(laV0); + double rxy = m_V0Tools->rxy(laV0); + double rxyError = m_V0Tools->rxyError(laV0); + Amg::Vector3D momentum = m_V0Tools->V0Momentum(laV0); + mDecor_mass( *laV0 ) = mass_la; + mDecor_massError( *laV0 ) = mass_error_la; + mDecor_pt( *laV0 ) = pt; + mDecor_ptError( *laV0 ) = ptError; + mDecor_rxy( *laV0 ) = rxy; + mDecor_rxyError( *laV0 ) = rxyError; + mDecor_px( *laV0 ) = momentum.x(); + mDecor_py( *laV0 ) = momentum.y(); + mDecor_pz( *laV0 ) = momentum.z(); + ATH_MSG_DEBUG("Reco_V0Finder mass_la " << mass_la << " mass_error_la " << mass_error_la << " pt " << pt << " rxy " << rxy); + } + xAOD::VertexContainer::const_iterator lbItr = lbContainer->begin(); + for ( lbItr=lbContainer->begin(); lbItr!=lbContainer->end(); ++lbItr ) + { + const xAOD::Vertex * lbV0 = (*lbItr); + double mass_lb = m_V0Tools->invariantMass(lbV0,m_masspi,m_massp); + double mass_error_lb = m_V0Tools->invariantMassError(lbV0,m_masspi,m_massp); + double pt = m_V0Tools->pT(lbV0); + double ptError = m_V0Tools->pTError(lbV0); + double rxy = m_V0Tools->rxy(lbV0); + double rxyError = m_V0Tools->rxyError(lbV0); + Amg::Vector3D momentum = m_V0Tools->V0Momentum(lbV0); + mDecor_mass( *lbV0 ) = mass_lb; + mDecor_massError( *lbV0 ) = mass_error_lb; + mDecor_pt( *lbV0 ) = pt; + mDecor_ptError( *lbV0 ) = ptError; + mDecor_rxy( *lbV0 ) = rxy; + mDecor_rxyError( *lbV0 ) = rxyError; + mDecor_px( *lbV0 ) = momentum.x(); + mDecor_py( *lbV0 ) = momentum.y(); + mDecor_pz( *lbV0 ) = momentum.z(); + ATH_MSG_DEBUG("Reco_V0Finder mass_lb " << mass_lb << " mass_error_lb " << mass_error_lb << " pt " << pt << " rxy " << rxy); + } + } + + + if(!callV0Finder){ //Fill with empty containers + v0Container = new xAOD::VertexContainer; + v0AuxContainer = new xAOD::VertexAuxContainer; + v0Container->setStore(v0AuxContainer); + ksContainer = new xAOD::VertexContainer; + ksAuxContainer = new xAOD::VertexAuxContainer; + ksContainer->setStore(ksAuxContainer); + laContainer = new xAOD::VertexContainer; + laAuxContainer = new xAOD::VertexAuxContainer; + laContainer->setStore(laAuxContainer); + lbContainer = new xAOD::VertexContainer; + lbAuxContainer = new xAOD::VertexAuxContainer; + lbContainer->setStore(lbAuxContainer); + } + + //---- Recording section: write the results to StoreGate ---// + CHECK(evtStore()->record(v0Container, m_v0ContainerName)); + + CHECK(evtStore()->record(v0AuxContainer, m_v0ContainerName+"Aux.")); + + CHECK(evtStore()->record(ksContainer, m_ksContainerName)); + + CHECK(evtStore()->record(ksAuxContainer, m_ksContainerName+"Aux.")); + + CHECK(evtStore()->record(laContainer, m_laContainerName)); + + CHECK(evtStore()->record(laAuxContainer, m_laContainerName+"Aux.")); + + CHECK(evtStore()->record(lbContainer, m_lbContainerName)); + + CHECK(evtStore()->record(lbAuxContainer, m_lbContainerName+"Aux.")); + + return StatusCode::SUCCESS; + } +} + + + + diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/Reco_Vertex.cxx b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/Reco_Vertex.cxx new file mode 100644 index 00000000000..9d2b236c841 --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/Reco_Vertex.cxx @@ -0,0 +1,163 @@ +/* + Copyright (C) 2002-2018 CERN for the benefit of the ATLAS collaboration +*/ + +///////////////////////////////////////////////////////////////// +// Reco_Vertex.cxx +/////////////////////////////////////////////////////////////////// +// Author: Daniel Scheirich <daniel.scheirich@cern.ch> +// Based on the Integrated Simulation Framework +// +// Basic Jpsi->mu mu derivation example + +#include "DerivationFrameworkBPhys/Reco_Vertex.h" +#include "DerivationFrameworkBPhys/BPhysPVTools.h" +#include "xAODTracking/VertexContainer.h" +#include "xAODTracking/VertexAuxContainer.h" + +namespace DerivationFramework { + + Reco_Vertex::Reco_Vertex(const std::string& t, + const std::string& n, + const IInterface* p) : + AthAlgTool(t,n,p), + m_v0Tools("Trk::V0Tools"), + m_SearchTool(), + m_pvRefitter("Analysis::PrimaryVertexRefitter") + { + declareInterface<DerivationFramework::IAugmentationTool>(this); + + // Declare tools + declareProperty("V0Tools" , m_v0Tools); + declareProperty("VertexSearchTool", m_SearchTool); + declareProperty("PVRefitter", m_pvRefitter); + + // Declare user-defined properties + declareProperty("OutputVtxContainerName", m_outputVtxContainerName = "OniaCandidates"); + declareProperty("PVContainerName" , m_pvContainerName = "PrimaryVertices"); + declareProperty("RefPVContainerName" , m_refPVContainerName = "RefittedPrimaryVertices"); + declareProperty("RefitPV" , m_refitPV = false); + declareProperty("MaxPVrefit" , m_PV_max = 1000); + declareProperty("DoVertexType" , m_DoVertexType = 7); + // minimum number of tracks for PV to be considered for PV association + declareProperty("MinNTracksInPV" , m_PV_minNTracks = 0); + declareProperty("Do3d" , m_do3d = false); + declareProperty("CheckCollections" , m_checkCollections = false); + declareProperty("CheckVertexContainers" , m_CollectionsToCheck); + } + + // * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * + + StatusCode Reco_Vertex::initialize() + { + + ATH_MSG_DEBUG("in initialize()"); + + // retrieve V0 tools + CHECK( m_v0Tools.retrieve() ); + + // get the Search tool + CHECK( m_SearchTool.retrieve() ); + + // get the PrimaryVertexRefitter tool + CHECK( m_pvRefitter.retrieve() ); + + // Get the beam spot service + ATH_CHECK(m_beamSpotKey.initialize()); + + + ATH_CHECK(m_outputVtxContainerName.initialize()); + ATH_CHECK(m_pvContainerName.initialize()); + ATH_CHECK(m_refPVContainerName.initialize()); + if(m_checkCollections) ATH_CHECK(m_CollectionsToCheck.initialize()); + return StatusCode::SUCCESS; + + } + + // * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * + + StatusCode Reco_Vertex::addBranches() const + { + bool callTool = true; + if(m_checkCollections) { + for(const auto &str : m_CollectionsToCheck){ + SG::ReadHandle<xAOD::VertexContainer> handle(str); + ATH_CHECK(handle.isValid()); + if(handle->size() == 0) { + callTool = false; + ATH_MSG_DEBUG("Container VertexContainer (" << str << ") is empty"); + break;//No point checking other containers + } + } + } + + // Vertex container and its auxilliary store + xAOD::VertexContainer* vtxContainer = nullptr; + xAOD::VertexAuxContainer* vtxAuxContainer = nullptr; + + if(callTool) { + //---------------------------------------------------- + // call Tool + //---------------------------------------------------- + if( !m_SearchTool->performSearch(vtxContainer, vtxAuxContainer).isSuccess() ) { + ATH_MSG_FATAL("Tool (" << m_SearchTool << ") failed."); + return StatusCode::FAILURE; + } + + //---------------------------------------------------- + // retrieve primary vertices + //---------------------------------------------------- + SG::ReadHandle<xAOD::VertexContainer> pvContainer(m_pvContainerName); + + //---------------------------------------------------- + // Try to retrieve refitted primary vertices + //---------------------------------------------------- + xAOD::VertexContainer* refPvContainer = nullptr; + xAOD::VertexAuxContainer* refPvAuxContainer = nullptr; + if(m_refitPV) { + // refitted PV container does not exist. Create a new one. + refPvContainer = new xAOD::VertexContainer; + refPvAuxContainer = new xAOD::VertexAuxContainer; + refPvContainer->setStore(refPvAuxContainer); + } + + // Give the helper class the ptr to v0tools and beamSpotsSvc to use + SG::ReadCondHandle<InDet::BeamSpotData> beamSpotHandle { m_beamSpotKey }; + if(not beamSpotHandle.isValid()) ATH_MSG_ERROR("Cannot Retrieve " << m_beamSpotKey.key() ); + BPhysPVTools helper(&(*m_v0Tools), beamSpotHandle.cptr()); + helper.SetMinNTracksInPV(m_PV_minNTracks); + helper.SetSave3d(m_do3d); + + if(m_refitPV){ + if(vtxContainer->size() >0){ + StatusCode SC = helper.FillCandwithRefittedVertices(vtxContainer, pvContainer.cptr(), refPvContainer, &(*m_pvRefitter) , m_PV_max, m_DoVertexType); + if(SC.isFailure()){ + ATH_MSG_FATAL("refitting failed - check the vertices you passed"); + return SC; + } + } + }else{ + if(vtxContainer->size() >0)CHECK(helper.FillCandExistingVertices(vtxContainer, pvContainer.cptr(), m_DoVertexType)); + } + + //---------------------------------------------------- + // save in the StoreGate + //---------------------------------------------------- + SG::WriteHandle<xAOD::VertexContainer> handle(m_outputVtxContainerName); + ATH_CHECK(handle.record(std::unique_ptr<xAOD::VertexContainer>(vtxContainer ), std::unique_ptr<xAOD::VertexAuxContainer>(vtxAuxContainer ))); + + if(m_refitPV) { + SG::WriteHandle<xAOD::VertexContainer> handle(m_refPVContainerName); + ATH_CHECK(handle.record(std::unique_ptr<xAOD::VertexContainer>(refPvContainer ), std::unique_ptr<xAOD::VertexAuxContainer>(refPvAuxContainer ))); + } + } + + if (!callTool) { //Fill with empty containers + SG::WriteHandle<xAOD::VertexContainer> handle(m_outputVtxContainerName); + ATH_CHECK(handle.record(std::unique_ptr<xAOD::VertexContainer>(new xAOD::VertexContainer ), + std::unique_ptr<xAOD::VertexAuxContainer>(new xAOD::VertexAuxContainer ))); + } + + return StatusCode::SUCCESS; + } +} diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/Select_Bmumu.cxx b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/Select_Bmumu.cxx new file mode 100644 index 00000000000..d1027b9b2c3 --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/Select_Bmumu.cxx @@ -0,0 +1,563 @@ +/* + Copyright (C) 2002-2018 CERN for the benefit of the ATLAS collaboration +*/ + +//============================================================================ +// Select_Bmumu.cxx +//============================================================================ +// +// Author : Wolfgang Walkowiak <Wolfgang.Walkowiak@cern.ch.> +// Changes: +// +// Based on Select_onia2mumu.h. +// Original author: Daniel Scheirich <daniel.scheirich@cern.ch> +// +// Select B candidates for the B(s)mumu analysis including for +// the reference channels used. +// +// For an example see BPHY8.py . +// +// Job options provided by this class: +// - V0Tools -- ToolHandle for V0Tools (default: Trk::V0Tools) +// - HypothesisName -- name given to the hypothesis (passed flag) +// - InputVtxContainerName -- name of the input vertex container +// - TrkMasses" -- list of masses to be assigned to the tracks +// used for lifetime calculation +// (Make sure to give them in correct order!) +// - VtxMassHypo -- mass used in the calculation of lifetime +// - MassMin -- minimum of mass range +// - MassMax -- maximum of mass range +// - Chi2Max -- maximum chi2 cut +// - DoVertexType -- bits defining vertex association types +// to be used +// - Do3d -- add 3d proper time +// - BlindMassMin -- minimum of blinded mass range +// - BlindMassMax -- maximum blinded mass range +// - DoBlinding -- switch to enable blinding (default: false) +// - DoCutBlinded -- cut blinded vertices (default: false) +// - BlindOnlyAllMuonsTight -- only blind candidates with all tight muons +// - UseMuCalcMass -- use MUCALC mass in mass cuts (default: false) +// - SubDecVtxContNames -- names of containers with sub-decay candidates +// (in order of sub decays) +// - SubDecVtxHypoCondNames -- names of hypothesis required to be passed +// by sub-decay candidates +// - SubDecVtxHypoFlagNames -- names of hypothesis passed flags set by +// this algorithm on sub-decay candidates +// (taken as +// SupDecVtxHypoCondName+'_'+HypthesisName +// if not explicitely given) +// +//============================================================================ +// +#include "DerivationFrameworkBPhys/Select_Bmumu.h" + +#include <vector> +#include <string> +#include "TVector3.h" + +#include "TrkVertexAnalysisUtils/V0Tools.h" +#include "xAODTracking/VertexContainer.h" +#include "xAODTracking/VertexAuxContainer.h" +#include "xAODBPhys/BPhysHypoHelper.h" +#include "AthContainers/AuxElement.h" + +/* + * Some useful typedefs + */ +typedef ElementLink<xAOD::VertexContainer> VertexLink; +typedef std::vector<VertexLink> VertexLinkVector; + +namespace DerivationFramework { + + Select_Bmumu::Select_Bmumu(const std::string& t, + const std::string& n, + const IInterface* p) : + CfAthAlgTool(t,n,p), + m_v0Tools("Trk::V0Tools"), + m_muSelectionTool("CP::MuonSelectionTool/MuonSelectionTool") { + + declareInterface<DerivationFramework::IAugmentationTool>(this); + + // Declare tools + declareProperty("V0Tools", m_v0Tools); + declareProperty("MuonSelectionTool", m_muSelectionTool); + + // Declare user-defined properties + + declareProperty("HypothesisName" , m_hypoName = "A"); + declareProperty("InputVtxContainerName" , m_inputVtxContainerName = "JpsiCandidates"); + declareProperty("TrkMasses" , m_trkMasses = std::vector<double>(2, 105.658) ); + declareProperty("VtxMassHypo" , m_massHypo = 3096.916 ); + declareProperty("MassMax" , m_massMax = 6000); + declareProperty("MassMin" , m_massMin = 2000); + declareProperty("Chi2Max" , m_chi2Max = 200); + declareProperty("DoVertexType" , m_DoVertexType = 7); + declareProperty("Do3d" , m_do3d = false); + declareProperty("BlindMassMin" , m_blindMassMin = 0.); + declareProperty("BlindMassMax" , m_blindMassMax = 0.); + declareProperty("DoBlinding" , m_doBlinding = false); + declareProperty("DoCutBlinded" , m_doCutBlinded = false); + declareProperty("BlindOnlyAllMuonsTight", m_blindOnlyAllMuonsTight = false); + declareProperty("UseMuCalcMass" , m_useMuCalcMass = false); + declareProperty("SubDecVtxContNames" , m_subDecVtxContNames = {}); + declareProperty("SubDecVtxHypoCondNames", m_subDecVtxHypoCondNames = {}); + declareProperty("SubDecVtxHypoFlagNames", m_subDecVtxHypoFlagNames = {}); + } + //---------------------------------------------------------------------------- + StatusCode Select_Bmumu::initialize() { + + ATH_MSG_DEBUG("in initialize()"); + + // retrieve V0 tools + CHECK( m_v0Tools.retrieve() ); + + // retrieve MuonSelectionTool + if ( m_blindOnlyAllMuonsTight ) { + CHECK( m_muSelectionTool.retrieve() ); + } + + // check length of sub-decay vertex container and required hypo name + // vectors + if ( m_subDecVtxContNames.size() != m_subDecVtxHypoCondNames.size() ) { + ATH_MSG_ERROR("initialize(): number of elements for options " + << "SubDecVtxContNames and SubDecVtxHypoCondNames does not " + << "match : " << m_subDecVtxContNames.size() + << " != " << m_subDecVtxHypoCondNames << " !!"); + } + // check the length of condition and flag hypo name vectors and append + // to the later if necessary + if ( m_subDecVtxHypoCondNames.size() > m_subDecVtxHypoFlagNames.size() ) { + ATH_MSG_INFO("initialize(): SubDecVtxHypoFlagNames (" + << m_subDecVtxHypoFlagNames.size() + << ") < SubDecVtxHypoCondNames (" + << m_subDecVtxHypoCondNames.size() + << ") ... appending to the first."); + for ( unsigned int i = m_subDecVtxHypoFlagNames.size(); + i < m_subDecVtxHypoCondNames.size(); ++i) { + std::string flagname = m_hypoName+"_"+m_subDecVtxHypoCondNames[i]; + ATH_MSG_INFO("initialize(): SubDecVtxHypoFlagNames[" << i << "] = " + << flagname); + m_subDecVtxHypoFlagNames.push_back(flagname); + } + } else if ( m_subDecVtxHypoCondNames.size() + < m_subDecVtxHypoFlagNames.size() ) { + ATH_MSG_ERROR("initialize(): SubDecVtxHypoFlagNames (" + << m_subDecVtxHypoFlagNames.size() + << ") > SubDecVtxHypoCondNames (" + << m_subDecVtxHypoCondNames.size() + << ") ! Configuration error!"); + } + return StatusCode::SUCCESS; + } + //---------------------------------------------------------------------------- + StatusCode Select_Bmumu::finalize() { + + // everything all right + return StatusCode::SUCCESS; + } + //--------------------------------------------------------------------------- + void Select_Bmumu::ProcessVertex(xAOD::BPhysHypoHelper &bcand, + xAOD::BPhysHelper::pv_type pv_t) const { + + constexpr float errConst = -9999999; + const xAOD::Vertex* pv = bcand.pv(pv_t); + if (pv) { + // decorate the vertex. + // Proper decay time assuming constant mass hypothesis m_massHypo + BPHYS_CHECK( bcand.setTau(m_v0Tools->tau(bcand.vtx(), pv, m_massHypo), + pv_t, + xAOD::BPhysHypoHelper::TAU_CONST_MASS) ); + // Proper decay time assuming error constant mass hypothesis m_massHypo + BPHYS_CHECK( bcand.setTauErr( m_v0Tools->tauError(bcand.vtx(), pv, + m_massHypo), + pv_t, + xAOD::BPhysHypoHelper::TAU_CONST_MASS) ); + + BPHYS_CHECK( bcand.setTau(m_v0Tools->tau(bcand.vtx(),pv, m_trkMasses), + pv_t, + xAOD::BPhysHypoHelper::TAU_INV_MASS) ); + + BPHYS_CHECK( bcand.setTauErr(m_v0Tools->tauError(bcand.vtx(), pv, + m_trkMasses), + pv_t, + xAOD::BPhysHypoHelper::TAU_INV_MASS) ); + //enum pv_type {PV_MAX_SUM_PT2, PV_MIN_A0, PV_MIN_Z0, PV_MIN_Z0_BA}; + } else { + + BPHYS_CHECK( bcand.setTau(errConst, pv_t, + xAOD::BPhysHypoHelper::TAU_CONST_MASS) ); + // Proper decay time assuming error constant mass hypothesis m_massHypo + BPHYS_CHECK( bcand.setTauErr( errConst, + pv_t, + xAOD::BPhysHypoHelper::TAU_CONST_MASS) ); + + BPHYS_CHECK( bcand.setTau( errConst, + pv_t, + xAOD::BPhysHypoHelper::TAU_INV_MASS) ); + + BPHYS_CHECK( bcand.setTauErr( errConst, + pv_t, + xAOD::BPhysHypoHelper::TAU_INV_MASS) ); + } + + if(m_do3d){ + + BPHYS_CHECK( bcand.setTau3d( pv ? + m_v0Tools->tau3D(bcand.vtx(), pv, + m_massHypo) + : errConst, pv_t, + xAOD::BPhysHypoHelper::TAU_CONST_MASS) ); + // Proper decay time assuming error constant mass hypothesis m_massHypo + BPHYS_CHECK( bcand.setTau3dErr( pv ? + m_v0Tools->tau3DError(bcand.vtx(), pv, + m_massHypo) + : errConst, pv_t, + xAOD::BPhysHypoHelper::TAU_CONST_MASS) + ); + + BPHYS_CHECK( bcand.setTau3d( pv ? + m_v0Tools->tau3D(bcand.vtx(), pv, + m_trkMasses) + : errConst, pv_t, + xAOD::BPhysHypoHelper::TAU_INV_MASS) ); + + BPHYS_CHECK( bcand.setTau3dErr( pv ? + m_v0Tools->tau3DError(bcand.vtx(), pv, + m_trkMasses) + : errConst, pv_t, + xAOD::BPhysHypoHelper::TAU_INV_MASS) + ); + } + + } + //--------------------------------------------------------------------------- + StatusCode Select_Bmumu::addBranches() const { + + // Jpsi container and its auxilliary store + xAOD::VertexContainer* bcandContainer = NULL; + xAOD::VertexAuxContainer* bcandAuxContainer = NULL; + + // retrieve from the StoreGate + CHECK(evtStore()->retrieve(bcandContainer, m_inputVtxContainerName)); + CHECK(evtStore()->retrieve(bcandAuxContainer, + m_inputVtxContainerName+"Aux.")); + + // for sub-decays + std::vector<xAOD::VertexContainer*> subCandConts; + std::vector<xAOD::VertexAuxContainer*> subCandAuxConts; + + // retrieve from StoreGate + for (auto cname : m_subDecVtxContNames) { + xAOD::VertexContainer* subCandCont = NULL; + xAOD::VertexAuxContainer* subCandAuxCont = NULL; + CHECK(evtStore()->retrieve(subCandCont , cname)); + CHECK(evtStore()->retrieve(subCandAuxCont, cname+"Aux.")); + subCandConts.push_back(subCandCont); + subCandAuxConts.push_back(subCandAuxCont); + } + + // preset pass flag to false for subdecays + for (unsigned int isub=0; isub < subCandConts.size(); ++isub) { + xAOD::VertexContainer* subCandCont = subCandConts[isub]; + if ( subCandCont != NULL ) { + for (xAOD::VertexContainer::iterator it = subCandCont->begin(); + it != subCandCont->end(); ++it) { + if ( *it != NULL ) { + // only set subdecay passed flag to false if not yet set at all + setPassIfNotAvailable(**it, m_subDecVtxHypoFlagNames[isub], false); + // set subdecay blinding flag to true if not yet set at all + // and blinding is requested + if ( m_doBlinding ) { + setPassIfNotAvailable(**it, + m_subDecVtxHypoFlagNames[isub]+"_blinded", + true); + } + } else { + ATH_MSG_WARNING("addBranches(): NULL pointer elements in " + "xAOD::VertexContainer !!"); + } + } // for subCandCont + } // if subCandCont != NULL + } // for subCandConts + + bool doPt = (m_DoVertexType & 1) != 0; + bool doA0 = (m_DoVertexType & 2) != 0; + bool doZ0 = (m_DoVertexType & 4) != 0; + bool doZ0BA = (m_DoVertexType & 8) != 0; + + // loop over B candidates and perform selection and augmentation + // counters + int nPassMassCuts = 0; + int nPassChi2Cut = 0; + int nPassPrecVtxCut = 0; + int nInBlindedRegion = 0; + int nInBlindedRegionAllMuonsTight = 0; + xAOD::VertexContainer::iterator bcandItr = bcandContainer->begin(); + for (; bcandItr!=bcandContainer->end(); ++bcandItr) { + // create BPhysHypoHelper + xAOD::BPhysHypoHelper bcand(m_hypoName, *bcandItr); + + //---------------------------------------------------- + // decorate the vertex - part 1 + //---------------------------------------------------- + // a) invariant mass and error + if ( !bcand.setMass(m_trkMasses) ) + ATH_MSG_WARNING("Decoration bcand.setMass failed"); + + double massErr = m_v0Tools->invariantMassError(bcand.vtx(), m_trkMasses); + if ( !bcand.setMassErr(massErr) ) + ATH_MSG_WARNING("Decoration bcand.setMassErr failed"); + + // b) proper decay time and error: + // retrieve the refitted PV (or the original one, + // if the PV refitting was turned off) + // -- deferred to after the selection -- + /* + if (doPt) ProcessVertex(bcand, xAOD::BPhysHelper::PV_MAX_SUM_PT2); + if (doA0) ProcessVertex(bcand, xAOD::BPhysHelper::PV_MIN_A0); + if (doZ0) ProcessVertex(bcand, xAOD::BPhysHelper::PV_MIN_Z0); + if (doZ0BA) ProcessVertex(bcand, xAOD::BPhysHelper::PV_MIN_Z0_BA); + */ + + //---------------------------------------------------- + // perform the selection (i.e. flag the vertex) + //---------------------------------------------------- + // flag the vertex indicating that it is selected by this selector + bcand.setPass(true); + if ( m_doBlinding ) { + setPass(*bcand.vtx(), + m_hypoName+"_blinded", false); + } + + // now we check other cuts. if one of them didn't pass, set the flag to 0 + // and continue to the next candidate: + + // 1) invariant mass cuts + bool passedMuCalcMassCut(m_useMuCalcMass); + bool blindedMuCalcMass(true); + if ( m_useMuCalcMass ) { + std::string bname = m_hypoName+"_MUCALC_mass"; + static SG::AuxElement::Accessor<float> mucalcAcc(bname); + if ( mucalcAcc.isAvailable(**bcandItr) ) { + passedMuCalcMassCut = massCuts(mucalcAcc(**bcandItr)); + blindedMuCalcMass = massInBlindedRegion(mucalcAcc(**bcandItr)); + } else { + passedMuCalcMassCut = false; + blindedMuCalcMass = false; + ATH_MSG_INFO("MUCALC mass not available: " << bname << " !"); + } + } + bool passedMassCut = massCuts(bcand.mass()); + bool blindedMass = massInBlindedRegion(bcand.mass()); + + // 1a) muon quality cuts + bool allMuonsTight = + !m_blindOnlyAllMuonsTight || checkAllMuonsTight(bcand.muons()); + + // 1b) mark candidates in blinded region + if ( blindedMass && blindedMuCalcMass ) { + if ( m_doBlinding ) { + nInBlindedRegion++; + if ( allMuonsTight ) { + nInBlindedRegionAllMuonsTight++; + setPass(*bcand.vtx(), + m_hypoName+"_blinded", true); + } + } + } + + // 1c) cut on the mass range + if ( !(passedMassCut || passedMuCalcMassCut) ) { + bcand.setPass(false); // flag as failed + continue; + } + nPassMassCuts++; + + // 2) chi2 cut + if ( bcand.vtx()->chiSquared() > m_chi2Max) { + bcand.setPass(false);; // flag as failed + continue; + } + nPassChi2Cut++; + + // 3) preceeding vertices: within their mass ranges? + int npVtx = bcand.nPrecedingVertices(); + if ( npVtx > (int)m_subDecVtxContNames.size() ) { + ATH_MSG_WARNING("addBranches(): npVtx > m_subDecVtxContNames.size() !" + " (" << npVtx << " > " << m_subDecVtxContNames.size() + << ")"); + } + npVtx = std::min(npVtx, (int)m_subDecVtxContNames.size()); + // check preceeding vertices + bool pVtxOk = true; + for (int ipv=0; ipv<npVtx; ++ipv) { + const xAOD::Vertex* pVtx = bcand.precedingVertex(ipv); + if ( !pass(*pVtx, m_subDecVtxHypoCondNames[ipv]) ) { + pVtxOk = false; + continue; + } + } + if ( !pVtxOk ) { + bcand.setPass(false);; // flag as failed + continue; + } + // mark preceeding vertices + for (int ipv=0; ipv<npVtx; ++ipv) { + setPass(*bcand.precedingVertex(ipv), + m_subDecVtxHypoFlagNames[ipv], true); + if ( m_doBlinding && !(blindedMass && blindedMuCalcMass + && allMuonsTight) ) { + setPass(*bcand.precedingVertex(ipv), + m_subDecVtxHypoFlagNames[ipv]+"_blinded", false); + } + } + nPassPrecVtxCut++; + + //---------------------------------------------------- + // decorate the vertex - part 2 + //---------------------------------------------------- + // b) proper decay time and error: + // retrieve the refitted PV (or the original one, + // if the PV refitting was turned off) + if (doPt) ProcessVertex(bcand, xAOD::BPhysHelper::PV_MAX_SUM_PT2); + if (doA0) ProcessVertex(bcand, xAOD::BPhysHelper::PV_MIN_A0); + if (doZ0) ProcessVertex(bcand, xAOD::BPhysHelper::PV_MIN_Z0); + if (doZ0BA) ProcessVertex(bcand, xAOD::BPhysHelper::PV_MIN_Z0_BA); + + } // end of loop over bcand candidates + + // counters + // event level + addEvent("allEvents"); + if ( bcandContainer->size() > 0 ) addEvent("eventsWithCands"); + if ( nPassMassCuts > 0 ) addEvent("massCutEvents"); + if ( nPassChi2Cut > 0 ) addEvent("chi2CutEvents"); + if ( nPassPrecVtxCut > 0 ) addEvent("precVtxCutEvents"); + if ( m_doBlinding && nInBlindedRegion > 0 ) addEvent("blindedRegionEvents"); + // candidate level + addToCounter("allCandidates" , bcandContainer->size()); + addToCounter("massCutCandidates" , nPassMassCuts); + addToCounter("chi2CutCandidates" , nPassChi2Cut); + addToCounter("precVtxCutCandidates", nPassPrecVtxCut); + if ( m_doBlinding ) { + addToCounter("blindedRegionCandidates", nInBlindedRegion); + if ( m_blindOnlyAllMuonsTight ) { + addToCounter("blindedRegionCandidatesWithAllMuonsTight", + nInBlindedRegionAllMuonsTight); + } + } + + // all OK + return StatusCode::SUCCESS; + } + //--------------------------------------------------------------------------- + // Check whether mass cuts (including a possibly blinding region cut) + // are passed. + //--------------------------------------------------------------------------- + bool Select_Bmumu::massCuts(float mass) const { + + return (mass > m_massMin && mass < m_massMax) + && !(m_doBlinding && m_doCutBlinded && massInBlindedRegion(mass) ); + } + //--------------------------------------------------------------------------- + // Check whether mass cuts (including a possibly blinding region cut) + // are passed. + //--------------------------------------------------------------------------- + bool Select_Bmumu::massInBlindedRegion(float mass) const { + return ( mass > m_blindMassMin && mass < m_blindMassMax ); + } + //-------------------------------------------------------------------------- + // Check whether all muons are of quality tight. + //-------------------------------------------------------------------------- + bool Select_Bmumu::checkAllMuonsTight(const std::vector<const xAOD::Muon*>& + muons, int maxMuonsToCheck) const { + + bool allTight(true); + int ncheckMax = muons.size(); + if ( maxMuonsToCheck > -1 ) { + ncheckMax = std::min((int)muons.size(), maxMuonsToCheck); + } + for (int imu=0; imu < ncheckMax; ++imu) { + xAOD::Muon::Quality muQuality = + m_muSelectionTool->getQuality(*muons[imu]); + if ( !(muQuality <= xAOD::Muon::Tight) ) { + allTight = false; + break; + } + } + return allTight; + } + //--------------------------------------------------------------------------- + // Helper to check whether an element is marked as passing a specific + // hypothesis. + //--------------------------------------------------------------------------- + bool Select_Bmumu::pass(const SG::AuxElement& em, std::string hypo) const { + + SG::AuxElement::Accessor<Char_t> flagAcc("passed_"+hypo); + return flagAcc.isAvailable(em) && flagAcc(em) != 0; + } + //--------------------------------------------------------------------------- + // Helper to set an element marked as passing a specific hypothesis. + //--------------------------------------------------------------------------- + bool Select_Bmumu::setPass(const SG::AuxElement& em, std::string hypo, + bool passVal) const { + + SG::AuxElement::Decorator<Char_t> flagDec("passed_"+hypo); + flagDec(em) = passVal; + return true; + } + //--------------------------------------------------------------------------- + // Helper to set an element marked as passing a specific hypothesis + // if the element doesn't have the specific flag yet. + // Returns true if action had to be taken. + //--------------------------------------------------------------------------- + bool Select_Bmumu::setPassIfNotAvailable(SG::AuxElement& em, std::string hypo, + bool passVal) const { + + SG::AuxElement::Accessor<Char_t> flagAcc("passed_"+hypo); + bool exists = flagAcc.isAvailable(em); + if ( !exists ) { + setPass(em, hypo, passVal); + } + return !exists; + } + //--------------------------------------------------------------------------- + // Fetch a vector of preceeding vertices for a specific vertex + //--------------------------------------------------------------------------- + /* + std::vector<xAOD::Vertex*> + Select_Bmumu::getPrecedingVertices(const xAOD::Vertex* vtx) { + + // new vector of vertices + std::vector<xAOD::Vertex*> vtxList; + + // Create auxiliary branches accessors + static SG::AuxElement::Accessor<VertexLinkVector> + precedingVertexLinksAcc("PrecedingVertexLinks"); + + // check if branch exists + if( precedingVertexLinksAcc.isAvailable(*vtx) ) { + + // retrieve the precedingVertex links... + const VertexLinkVector& precedingVertexLinks = + precedingVertexLinksAcc(*vtx); + + // ... and check if they are all valid + for ( VertexLinkVector::const_iterator precedingVertexLinksItr = + precedingVertexLinks.begin(); + precedingVertexLinksItr!=precedingVertexLinks.end(); + ++precedingVertexLinksItr) { + // check if links are valid + if( (*precedingVertexLinksItr).isValid() ) { + // xAOD::Vertex* vtx2 = *precedingVertexLinkItr; + // vtxList.push_back(*(*precedingVertexLinksItr)); + } + } // for + } // if available + + return vtxList; + } + */ + //--------------------------------------------------------------------------- + +} // namespace DerivationFramework diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/Select_onia2mumu.cxx b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/Select_onia2mumu.cxx new file mode 100644 index 00000000000..33913419bf7 --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/Select_onia2mumu.cxx @@ -0,0 +1,197 @@ +/* + Copyright (C) 2002-2017 CERN for the benefit of the ATLAS collaboration +*/ + +///////////////////////////////////////////////////////////////// +// Select_onia2mumu.cxx +/////////////////////////////////////////////////////////////////// +// Author: Daniel Scheirich <daniel.scheirich@cern.ch> +// Based on the Integrated Simulation Framework +// +// Basic Jpsi->mu mu derivation example + +#include "DerivationFrameworkBPhys/Select_onia2mumu.h" + +#include "TrkVertexAnalysisUtils/V0Tools.h" +#include "xAODBPhys/BPhysHypoHelper.h" + +#include <vector> +#include <string> + +namespace DerivationFramework { + + Select_onia2mumu::Select_onia2mumu(const std::string& t, + const std::string& n, + const IInterface* p) : + AthAlgTool(t,n,p), + m_v0Tools("Trk::V0Tools") + { + declareInterface<DerivationFramework::IAugmentationTool>(this); + + // Declare tools + declareProperty("V0Tools", m_v0Tools); + + // Declare user-defined properties + + declareProperty("HypothesisName" , m_hypoName = "A"); + declareProperty("InputVtxContainerName", m_inputVtxContainerName = "JpsiCandidates"); + declareProperty("TrkMasses" , m_trkMasses = std::vector<double>(2, 105.658) ); + declareProperty("VtxMassHypo" , m_massHypo = 3096.916 ); + declareProperty("MassMax" , m_massMax = 6000); + declareProperty("MassMin" , m_massMin = 2000); + declareProperty("Chi2Max" , m_chi2Max = 200); + declareProperty("DoVertexType" , m_DoVertexType = 7); + declareProperty("LxyMin" , m_lxyMin = std::numeric_limits<double>::lowest()); + declareProperty("Do3d" , m_do3d = false); + + } + + // * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * + + StatusCode Select_onia2mumu::initialize() + { + + ATH_MSG_DEBUG("in initialize()"); + + // retrieve V0 tools + CHECK( m_v0Tools.retrieve() ); + ATH_CHECK(m_inputVtxContainerName.initialize()); + + return StatusCode::SUCCESS; + + } + + // * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * + + void Select_onia2mumu::ProcessVertex(xAOD::BPhysHypoHelper &onia, xAOD::BPhysHelper::pv_type pv_t) const{ + constexpr float errConst = -9999999; + const xAOD::Vertex* pv = onia.pv(pv_t); + if(pv) { + // decorate the vertex. + // Proper decay time assuming constant mass hypothesis m_massHypo + BPHYS_CHECK( onia.setTau( m_v0Tools->tau(onia.vtx(), pv, m_massHypo), + pv_t, + xAOD::BPhysHypoHelper::TAU_CONST_MASS) ); + // Proper decay time assuming error constant mass hypothesis m_massHypo + BPHYS_CHECK( onia.setTauErr( m_v0Tools->tauError(onia.vtx(), pv, m_massHypo), + pv_t, + xAOD::BPhysHypoHelper::TAU_CONST_MASS) ); + + BPHYS_CHECK( onia.setTau( m_v0Tools->tau(onia.vtx(), pv, m_trkMasses), + pv_t, + xAOD::BPhysHypoHelper::TAU_INV_MASS) ); + + BPHYS_CHECK( onia.setTauErr( m_v0Tools->tauError(onia.vtx(), pv, m_trkMasses), + pv_t, + xAOD::BPhysHypoHelper::TAU_INV_MASS) ); + + //enum pv_type {PV_MAX_SUM_PT2, PV_MIN_A0, PV_MIN_Z0, PV_MIN_Z0_BA}; + }else{ + + + BPHYS_CHECK( onia.setTau(errConst, pv_t, + xAOD::BPhysHypoHelper::TAU_CONST_MASS) ); + // Proper decay time assuming error constant mass hypothesis m_massHypo + BPHYS_CHECK( onia.setTauErr( errConst, + pv_t, + xAOD::BPhysHypoHelper::TAU_CONST_MASS) ); + + BPHYS_CHECK( onia.setTau( errConst, + pv_t, + xAOD::BPhysHypoHelper::TAU_INV_MASS) ); + + BPHYS_CHECK( onia.setTauErr( errConst, + pv_t, + xAOD::BPhysHypoHelper::TAU_INV_MASS) ); + } + + if(m_do3d){ + BPHYS_CHECK( onia.setTau3d( pv ? m_v0Tools->tau3D(onia.vtx(), pv, m_massHypo) : errConst, + pv_t, + xAOD::BPhysHypoHelper::TAU_CONST_MASS) ); + // Proper decay time assuming error constant mass hypothesis m_massHypo + BPHYS_CHECK( onia.setTau3dErr( pv ? m_v0Tools->tau3DError(onia.vtx(), pv, m_massHypo) : errConst, + pv_t, + xAOD::BPhysHypoHelper::TAU_CONST_MASS) ); + + BPHYS_CHECK( onia.setTau3d( pv ? m_v0Tools->tau3D(onia.vtx(), pv, m_trkMasses) : errConst, + pv_t, + xAOD::BPhysHypoHelper::TAU_INV_MASS) ); + + BPHYS_CHECK( onia.setTau3dErr( pv ? m_v0Tools->tau3DError(onia.vtx(), pv, m_trkMasses) : errConst, + pv_t, + xAOD::BPhysHypoHelper::TAU_INV_MASS) ); + + } + + } + + + StatusCode Select_onia2mumu::addBranches() const + { + + SG::ReadHandle<xAOD::VertexContainer> oniaContainer(m_inputVtxContainerName); + + bool doPt = (m_DoVertexType & 1) != 0; + bool doA0 = (m_DoVertexType & 2) != 0; + bool doZ0 = (m_DoVertexType & 4) != 0; + bool doZ0BA = (m_DoVertexType & 8) != 0; + // loop over onia candidates and perform selection and augmentation + xAOD::VertexContainer::const_iterator oniaItr = oniaContainer->begin(); + for(; oniaItr!=oniaContainer->end(); ++oniaItr) { + // create BPhysHypoHelper + xAOD::BPhysHypoHelper onia(m_hypoName, *oniaItr); + if((*oniaItr)->nTrackParticles() != m_trkMasses.size()) + ATH_MSG_WARNING("Vertex has " << (*oniaItr)->nTrackParticles() << " while provided masses " << m_trkMasses.size()); + //---------------------------------------------------- + // decorate the vertex + //---------------------------------------------------- + // a) invariant mass and error + if( !onia.setMass(m_trkMasses) ) ATH_MSG_WARNING("Decoration onia.setMass failed"); + + double massErr = m_v0Tools->invariantMassError(onia.vtx(), m_trkMasses); + if( !onia.setMassErr(massErr) ) ATH_MSG_WARNING("Decoration onia.setMassErr failed"); + + // b) proper decay time and error: + // retrieve the refitted PV (or the original one, if the PV refitting was turned off) + if(doPt) ProcessVertex(onia, xAOD::BPhysHelper::PV_MAX_SUM_PT2); + if(doA0) ProcessVertex(onia, xAOD::BPhysHelper::PV_MIN_A0); + if(doZ0) ProcessVertex(onia, xAOD::BPhysHelper::PV_MIN_Z0); + if(doZ0BA) ProcessVertex(onia, xAOD::BPhysHelper::PV_MIN_Z0_BA); + + //---------------------------------------------------- + // perform the selection (i.e. flag the vertex) + //---------------------------------------------------- + // flag the vertex indicating that it is selected by this selector + onia.setPass(true); + + // now we check othe cuts. if one of them didn't pass, set the flag to 0 + // and continue to the next candidate: + + // 1) invariant mass cut + if( onia.mass() < m_massMin || onia.mass() > m_massMax) { + onia.setPass(false); // flag as failed + continue; + } + + // 2) chi2 cut + if( onia.vtx()->chiSquared() > m_chi2Max) { + onia.setPass(false);; // flag as failed + continue; + } + // 3) lxy cut + if( onia.lxy(xAOD::BPhysHelper::PV_MAX_SUM_PT2) < m_lxyMin) { + onia.setPass(false);; // flag as failed + continue; + } + + } // end of loop over onia candidates + + // all OK + return StatusCode::SUCCESS; + } + + // * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * + + +} diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/Thin_vtxDuplicates.cxx b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/Thin_vtxDuplicates.cxx new file mode 100644 index 00000000000..abb34bcd436 --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/Thin_vtxDuplicates.cxx @@ -0,0 +1,176 @@ +/* + Copyright (C) 2002-2017 CERN for the benefit of the ATLAS collaboration +*/ + +///////////////////////////////////////////////////////////////// +// Thin_vtxDuplicates.cxx +/////////////////////////////////////////////////////////////////// +// Matteo Bedognetti (matteo.bedognetti@cern.ch) +//Based on Thin_vtxTrk.cxx, by + + + +#include "DerivationFrameworkBPhys/Thin_vtxDuplicates.h" +#include "xAODTracking/TrackParticleContainer.h" +#include "xAODBPhys/BPhysHypoHelper.h" +#include <vector> +#include <string> +#include <algorithm> // for the sort function +#include <iomanip> +#include "StoreGate/ThinningHandle.h" +// Constructor +DerivationFramework::Thin_vtxDuplicates::Thin_vtxDuplicates(const std::string& t, const std::string& n, const IInterface* p ) : + AthAlgTool(t,n,p), + // m_acceptanceR(-1.), + m_noFlags(false), + m_nVtxTot(0), + m_nVtxPass(0) +{ + declareInterface<DerivationFramework::IThinningTool>(this); + + declareProperty("VertexContainerName" , m_vertexContainerNames); + declareProperty("PassFlags" , m_passFlags); + //declareProperty("AcceptanceRadius" , m_acceptanceR); + declareProperty("ApplyAnd" , m_and = true); //This will be applied depending on the order in which the thinning tools are added to the kernel + declareProperty("IgnoreFlags" , m_noFlags); + //declareProperty("ApplyAndForTracks" , m_trackAnd = false); + //declareProperty("ThinTracks" , m_thinTracks = true); +} + +// Destructor +DerivationFramework::Thin_vtxDuplicates::~Thin_vtxDuplicates() = default; + +// Athena initialize and finalize +StatusCode DerivationFramework::Thin_vtxDuplicates::initialize() +{ + // Decide which collections need to be checked for ID TrackParticles + ATH_MSG_VERBOSE("initialize() ..."); + ATH_CHECK(m_vertexContainerNames.initialize(m_streamName)); + + + if (m_passFlags.empty()) { + ATH_MSG_FATAL("No pass flags provided for thinning."); + return StatusCode::FAILURE; + } else { + for(auto itr = m_passFlags.cbegin(); itr!=m_passFlags.cend(); ++itr) { + ATH_MSG_INFO("Vertices must pass the \"" << itr->key() << "\" selection"); + } + } + + for(auto &key : m_passFlags){ + key = m_vertexContainerNames.key() + '.' + key.key(); + } + ATH_CHECK(m_passFlags.initialize()); + return StatusCode::SUCCESS; +} + +StatusCode DerivationFramework::Thin_vtxDuplicates::finalize() +{ + ATH_MSG_VERBOSE("finalize() ..."); + ATH_MSG_INFO("Processed "<< m_nVtxTot <<" vertices, "<< m_nVtxPass<< " were retained "); + + return StatusCode::SUCCESS; +} + +// The thinning itself +StatusCode DerivationFramework::Thin_vtxDuplicates::doThinning() const +{ + // retieve vertex + SG::ThinningHandle< xAOD::VertexContainer > vertexContainer(m_vertexContainerNames); + std::vector<bool> vtxMask(vertexContainer->size(), true); // default: keep all vertices + int vtxTot = 0; + int nVtxPass = 0; + // loop over vertices + int k = 0; + std::vector<SG::ReadDecorHandle<xAOD::VertexContainer, Char_t>> handles; + handles.reserve(m_passFlags.size()); + for(const auto &key : m_passFlags){ + handles.emplace_back(key); + if(!handles.back().isPresent()) return StatusCode::FAILURE; + } + for(auto vtxItr = vertexContainer->cbegin(); vtxItr!=vertexContainer->cend(); ++vtxItr, ++k) { + const xAOD::Vertex* vtx = *vtxItr; + // check if the vertex passed the required selections criteria (is run when the vertex is already excluded, because the counter needs the info) + bool passed = false; + if(m_noFlags){passed = true; vtxTot++; } + else{ + for(auto &flagAcc : handles) { + if(flagAcc(*vtx) != 0) { + passed = true; + vtxTot++;//Have to count the ones which are accepted to start with + break; + } + } // end of loop over flags + } + + // Skip if it has already been identified as duplicate + if(vtxMask[k] == false)continue; //After the flag-check to have the total-passed work correctly + + if(!passed)vtxMask[k]= false; + + if(passed) { + // vertex passed the selection + nVtxPass++; + + // determine the sum of the tracks at vertex as centre for the cone + std::vector<const xAOD::TrackParticle*> presentVertex, compareVertex; + + //Fill in the present vertex, for later comparison against other vertices + presentVertex.clear(); + for(uint j=0; j<vtx->nTrackParticles(); ++j) { + presentVertex.push_back(vtx->trackParticle(j)); + } + sort( presentVertex.begin(), presentVertex.end() ); //Sort the trackparticles BY POINTER ADDRESS + + //Loop over the remaining vertices and remove them if needed + int loop_k = k+1; + for(auto vtxLoopItr = vtxItr+1; vtxLoopItr!=vertexContainer->cend(); vtxLoopItr++, loop_k++){ + + const xAOD::Vertex* loop_vtx = *vtxLoopItr; + + //Vertices are distinct if have different size + if(vtx->nTrackParticles() != loop_vtx->nTrackParticles())continue; + + //If the vertex is still active load and compare + if(vtxMask[loop_k]){ + + compareVertex.clear(); + for(uint j=0; j<loop_vtx->nTrackParticles(); ++j) { + compareVertex.push_back(loop_vtx->trackParticle(j)); + } + + std::sort( compareVertex.begin(), compareVertex.end()); + + vtxMask[loop_k] = false; + + ATH_MSG_DEBUG("Compared tracks: "); + ATH_MSG_DEBUG(std::setw(14)<<compareVertex[0]<<std::setw(14) << compareVertex[1]<<std::setw(14)<<compareVertex[2]); + ATH_MSG_DEBUG(std::setw(14)<<presentVertex[0]<<std::setw(14) << presentVertex[1]<<std::setw(14)<<presentVertex[2]); + + for(uint j=0; j<loop_vtx->nTrackParticles(); ++j) { + if( compareVertex[j] != presentVertex[j] ){vtxMask[loop_k] = true; break;} + } + ATH_MSG_DEBUG("Verdict:"<<(vtxMask[loop_k]? "keep": "erase") ); + } + + } // Endo of extra loop over remaining vertices + + } // if( passed ) + } // end of loop over vertices + + // Execute the thinning service based on the vtxMask. + if (m_and) { + vertexContainer.keep(vtxMask, SG::ThinningHandleBase::Op::And); + } + if (!m_and) { + vertexContainer.keep(vtxMask, SG::ThinningHandleBase::Op::Or); + } + + m_nVtxTot.fetch_add( vtxTot, std::memory_order_relaxed); + m_nVtxPass.fetch_add( nVtxPass, std::memory_order_relaxed); + + + + return StatusCode::SUCCESS; +} + diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/Thin_vtxTrk.cxx b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/Thin_vtxTrk.cxx new file mode 100644 index 00000000000..14628f870cd --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/Thin_vtxTrk.cxx @@ -0,0 +1,200 @@ +/* + Copyright (C) 2002-2017 CERN for the benefit of the ATLAS collaboration +*/ + +///////////////////////////////////////////////////////////////// +// Thin_vtxTrk.cxx +/////////////////////////////////////////////////////////////////// +// Author: James Catmore (James.Catmore@cern.ch) +// This is a trivial example of an implementation of a thinning tool +// which removes all ID tracks which do not pass a user-defined cut + +#include "DerivationFrameworkBPhys/Thin_vtxTrk.h" + +#include "xAODBPhys/BPhysHypoHelper.h" +#include "StoreGate/ThinningHandle.h" +#include <vector> +#include <string> +// Constructor +DerivationFramework::Thin_vtxTrk::Thin_vtxTrk(const std::string& t, const std::string& n, const IInterface* p ) : + AthAlgTool(t,n,p), + m_ntot(0), + m_npass(0), + m_acceptanceR(-1.), // Do not add tracks within a cone from the vertex by default + m_nVtxTot(0), + m_nVtxPass(0), + m_noFlags(false) +{ + declareInterface<DerivationFramework::IThinningTool>(this); + + declareProperty("TrackParticleContainerName", m_trackParticleContainerName = "InDetTrackParticles"); + declareProperty("VertexContainerNames" , m_vertexContainerName); + declareProperty("PassFlags" , m_passFlags); + declareProperty("AcceptanceRadius" , m_acceptanceR); + declareProperty("IgnoreFlags" , m_noFlags); + declareProperty("ApplyAnd" , m_and = false); + declareProperty("ApplyAndForTracks" , m_trackAnd = false); + declareProperty("ThinTracks" , m_thinTracks = true); +} + +// Destructor +DerivationFramework::Thin_vtxTrk::~Thin_vtxTrk() = default; + +// Athena initialize and finalize +StatusCode DerivationFramework::Thin_vtxTrk::initialize() +{ + // Decide which collections need to be checked for ID TrackParticles + ATH_MSG_VERBOSE("initialize() ..."); + ATH_CHECK(m_trackParticleContainerName.initialize(m_streamName)); + + + if( m_noFlags){ + ATH_MSG_INFO("IgnoreFlags is set, all vertices in the container will be kept"); + } + + if( ! m_noFlags){ + if (m_passFlags.empty()) { + ATH_MSG_FATAL("No pass flags provided for thinning."); + return StatusCode::FAILURE; + } else { + for(auto itr = m_passFlags.begin(); itr!=m_passFlags.end(); ++itr) { + ATH_MSG_INFO("Vertices must pass the \"" << *itr << "\" selection"); + } + } + } + + if (m_acceptanceR > 0.) { + ATH_MSG_INFO("Extra tracks must be within cone of "<<m_acceptanceR<<" from vertex candidate."); + } + + for(auto &handle : m_vertexContainerName){ + ATH_CHECK(handle.initialize(m_streamName)); + } + for(const auto &tracknames : m_vertexContainerName){ + for(const auto &str : m_passFlags){ + m_passArray.emplace_back(tracknames.key() + '.' + str); + } + } + ATH_CHECK(m_passArray.initialize()); + return StatusCode::SUCCESS; +} + +StatusCode DerivationFramework::Thin_vtxTrk::finalize() +{ + ATH_MSG_VERBOSE("finalize() ..."); + ATH_MSG_INFO("Processed "<< m_ntot <<" tracks, "<< m_npass<< " were retained "); + ATH_MSG_INFO("Processed "<< m_nVtxTot <<" vertices, "<< m_nVtxPass<< " were retained "); + + return StatusCode::SUCCESS; +} + +// The thinning itself +StatusCode DerivationFramework::Thin_vtxTrk::doThinning() const +{ + // Retrieve main TrackParticle collection + SG::ThinningHandle<xAOD::TrackParticleContainer> importedTrackParticles(m_trackParticleContainerName); + + // Check the event contains tracks + unsigned int nTracks = importedTrackParticles->size(); + if (nTracks==0) return StatusCode::SUCCESS; + + // Set up a trackMask with the same entries as the full TrackParticle collection + std::vector<bool> trackMask(nTracks,false); // default: don't keep any tracks + m_ntot += nTracks; + int nVtxTot =0; + int nVtxPass=0; + + std::unordered_map<std::string, SG::ReadDecorHandle<xAOD::VertexContainer, Char_t>> handles; + handles.reserve(m_passArray.size()); + for(const auto &key : m_passArray){ + auto it = handles.emplace(key.key(), key); + if(!(*it.first).second.isPresent()) return StatusCode::FAILURE; + } + + // retieve vertex + for(const auto& name : m_vertexContainerName){ + SG::ThinningHandle<xAOD::VertexContainer> vertexContainer(name); + std::vector<bool> vtxMask(vertexContainer->size(), false); // default: don't keep any vertices + + // loop over vertices + int k = 0; + for(auto vtxItr = vertexContainer->begin(); vtxItr!=vertexContainer->end(); ++vtxItr, ++k) { + const xAOD::Vertex* vtx = *vtxItr; + nVtxTot++; + + // check if the vertex passed the required selections criteria + bool passed = false; + for(std::vector<std::string>::const_iterator flagItr = m_passFlags.begin(); flagItr!=m_passFlags.end(); ++flagItr) { + std::string lookupstr = name.key() + '.' + (*flagItr); + const auto& handle = handles.at(lookupstr); + if(handle(*vtx) != 0) { + passed = true; + break; + } + } // end of loop over flags + + if(passed || m_noFlags) { + // vertex passed the selection + vtxMask[k] = true; + nVtxPass++; + + // Add tracks according to DR selection + if(m_acceptanceR > 0.){ + + // determine the sum of the tracks at vertex as centre for the cone + TLorentzVector centreCandidate; + for(uint j=0; j<vtx->nTrackParticles(); ++j) { + centreCandidate += vtx->trackParticle(j)->p4(); + } + + for(uint i=0; i<nTracks; ++i) { + if(!trackMask[i]) { // do this only for tracks that haven't been selected, yet + const xAOD::TrackParticle* track = (*importedTrackParticles)[i]; + if(centreCandidate.DeltaR(track->p4()) < m_acceptanceR) trackMask[i]= true; + } + } + }// end adding tracks according to DR selection + + if(m_thinTracks) { + // loop over all tracks + for(uint i=0; i<nTracks; ++i) { + if(!trackMask[i]) { // do this only for tracks that haven't been selected, yet + const xAOD::TrackParticle* track = (*importedTrackParticles)[i]; + // loop over tracks at vertex + for(uint j=0; j<vtx->nTrackParticles(); ++j) { + if(vtx->trackParticle(j) == track) { + trackMask[i] = true; // accept track + } + } // end of loop over tracks at vertex + } + } // end of loop over all tracks + } + } + } // end of loop over vertices + + // Execute the thinning service based on the vtxMask. + if (m_and) { + vertexContainer.keep(vtxMask, SG::ThinningHandleBase::Op::And); + } + if (!m_and) { + vertexContainer.keep(vtxMask, SG::ThinningHandleBase::Op::Or); + } + } + + // Count up the trackMask contents + m_npass += std::accumulate(trackMask.begin(), trackMask.end(), 0); + m_nVtxTot += nVtxTot; + m_nVtxPass+= nVtxPass; + if(m_thinTracks || m_acceptanceR > 0.) { + // Execute the thinning service based on the trackMask. Finish. + if (m_trackAnd) { + importedTrackParticles.keep(trackMask, SG::ThinningHandleBase::Op::And); + } + if (!m_trackAnd) { + importedTrackParticles.keep(trackMask, SG::ThinningHandleBase::Op::Or); + } + } + + return StatusCode::SUCCESS; +} + diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/TriggerCountToMetadata.cxx b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/TriggerCountToMetadata.cxx new file mode 100644 index 00000000000..58af23444ec --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/TriggerCountToMetadata.cxx @@ -0,0 +1,62 @@ +/* +Copyright (C) 2002-2017 CERN for the benefit of the ATLAS collaboration +*/ +//============================================================================ +// +// Author : Matteo Bedognetti <matteo.bedognetti@cern.ch.> +// Changes: +// +// Store trigger counts for specific chains in the DAOD's MetaData. +// This allows it to store information about triggers upon which events are NOT selected during the derivation +// +// Job options: +// - TriggerList -- a vector containing all triggers to store as strings +// - FolderName -- Is supposed to be the derivation name (some convention I guess) +// - TrigDecisionTool -- if one wants to pass this a specific TrigDecisionTool +// +//============================================================================ +// + +#include "DerivationFrameworkBPhys/TriggerCountToMetadata.h" +#include "AthenaPoolUtilities/CondAttrListCollection.h" + +#include <memory> + +namespace DerivationFramework { + + //-------------------------------------------------------------------------- + TriggerCountToMetadata::TriggerCountToMetadata(const std::string& t, + const std::string& n, + const IInterface* p) + : CfAthAlgTool(t,n,p), m_trigDecisionTool( "Trig::TrigDecisionTool/TrigDecisionTool" ) + { + declareInterface<DerivationFramework::IAugmentationTool>(this); + + declareProperty("TrigDecisionTool", m_trigDecisionTool ); + declareProperty("FolderName", m_folderName = "DerivationLevel"); + declareProperty("TriggerList", m_triggerList); + + } + //-------------------------------------------------------------------------- + StatusCode TriggerCountToMetadata::initialize() { + ATH_CHECK(m_trigDecisionTool.retrieve()); + + return StatusCode::SUCCESS; + } + + //-------------------------------------------------------------------------- + StatusCode TriggerCountToMetadata::addBranches() const { + + ATH_MSG_DEBUG("Inside TriggerCountToMetadata::addBranches()"); + + // W.w. method + addEvent("AllEvents"); + + for( unsigned int i=0; i<m_triggerList.size(); i++){ + addEvent(m_triggerList[i] , m_trigDecisionTool->isPassed(m_triggerList[i]) ); + } + + return StatusCode::SUCCESS; + } + +} // End of namespace DerivationFramework diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/VertexCaloIsolation.cxx b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/VertexCaloIsolation.cxx new file mode 100644 index 00000000000..c47d71389e3 --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/VertexCaloIsolation.cxx @@ -0,0 +1,583 @@ +/* + Copyright (C) 2002-2017 CERN for the benefit of the ATLAS collaboration +*/ + +// VertexCaloIsolation.cxx by Matteo Bedognetti +// +// This code is based on CaloIsolationTool of IsolationTools package +// +// Etcone is determined as a topoCluster-isolation value minus Energy Density (ED) correction and minus the energy depositions of the muons +// Muon's energy deposition is already stored in side the xAOD::Muon objects, but the muon-clusters are used to correct for the fact that they muons may have overlapping clusters +// The muon-clusters are stored as well in connection with the muons themselves +// +// The idea of comparing topoClusters with muon-clusters to decide what part of the muon's deposition is of +// importance had to be abandoned because topCluster cells are not present in xAOD +// +// It enforces the fact that for muons no core-surface is removed for the energy-density correction (thus the corrections are independent from each other) +// +// "isReliable" flag reports of each isolation value if all particles crossing the cone have been correctly corrected for. +// In the case of 2mu+ 1 track it mirrors the fact that the track does not extrapolate into the cone (as tracks have no muon-cluster from which to determine the core-correction) +// + + + + +#include "DerivationFrameworkBPhys/VertexCaloIsolation.h" + +#include <vector> +#include <string> +#include "TVector3.h" + +#include "xAODTracking/VertexContainer.h" +#include "xAODTracking/VertexAuxContainer.h" +#include "xAODBPhys/BPhysHelper.h" +#include "xAODBPhys/BPhysHypoHelper.h" +#include "TrkVertexAnalysisUtils/V0Tools.h" + +//#include "IsolationTool/CaloIsolationTool.h" +//#include "IsolationTool/CaloIsolationTool.h" + +#include "RecoToolInterfaces/ICaloTopoClusterIsolationTool.h" + +//#include "IsolationTool/IsolationHelper.h" +//#include "InDetTrackSelectionTool/InDetTrackSelectionTool.h" +#include "CaloEvent/CaloCell.h" //Is used (though shown as auto) +//#include "TrkParameters/TrackParameters.h" +#include "CaloInterface/ICaloNoiseTool.h" +#include "TrkCaloExtension/CaloExtension.h" +//#include "CaloUtils/CaloClusterStoreHelper.h" +//#include "CaloUtils/CaloCellList.h" +//#include "CaloEvent/CaloCellContainer.h" +#include "xAODTracking/TrackingPrimitives.h" +#include "xAODPrimitives/IsolationHelpers.h" +#include "TrackToCalo/CaloCellCollector.h" +#include <set> + +//#include "Identifier/Identifier32.h" +using namespace std; +namespace DerivationFramework { + + VertexCaloIsolation::VertexCaloIsolation(const std::string& t, + const std::string& n, + const IInterface* p) : + AthAlgTool(t,n,p), + m_caloIsoTool("xAOD::CaloIsolationTool/CaloIsolationTool"), + m_trackContainerName("InDetTrackParticles"), + m_vertexContainerName("NONE"), + m_caloClusterContainerName("CaloCalTopoClusters"), + m_muonContainerName("Muons"), + m_caloExtTool("Trk::ParticleCaloExtensionTool/ParticleCaloExtensionTool"), + //m_caloNoiseTool(""), + m_cones(), + m_sigmaCaloNoiseCut(3.4), + m_vertexType(7) + + + // m_cellCollector("") + + +// m_caloExtTool +// m_caloNoiseTool, m_applyCaloNoiseCut, m_sigmaCaloNoiseCut +// m_cellCollector + + { + ATH_MSG_DEBUG("in constructor"); + declareInterface<DerivationFramework::IAugmentationTool>(this); + + // Declare tools + declareProperty("CaloIsoTool" , m_caloIsoTool); + declareProperty("TrackContainer" , m_trackContainerName); + declareProperty("InputVertexContainer" , m_vertexContainerName); + declareProperty("CaloClusterContainer" , m_caloClusterContainerName); + declareProperty("ParticleCaloExtensionTool", m_caloExtTool); + declareProperty("MuonContainer", m_muonContainerName); + declareProperty("PassFlags" , m_passFlags); + declareProperty("IsolationTypes" , m_cones); + declareProperty("DoVertexTypes" , m_vertexType); + + + + } + + // * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * + + StatusCode VertexCaloIsolation::initialize() + { + + ATH_MSG_DEBUG("in initialize()"); + + // retrieve CaloIsolationTool + CHECK( m_caloIsoTool.retrieve() ); + + // retrieve CaloIsolationTool + CHECK( m_caloExtTool.retrieve() ); + + //Check that flags were given to tag the correct vertices + if(m_passFlags.empty()){ + ATH_MSG_WARNING("As no pass-flags are given, no vertices will be decorated"); + } + + // Control the IsolationType sequence + if(m_cones.empty()){ + m_cones.push_back(xAOD::Iso::etcone40); + m_cones.push_back(xAOD::Iso::etcone30); + m_cones.push_back(xAOD::Iso::etcone20); + } + + //if(m_applyCaloNoiseCut){ + //ATH_MSG_ERROR("No handle to a caloNoiseTool is kept in this tool, "); + //return StatusCode::FAILURE; + //} + + return StatusCode::SUCCESS; + + } + + // * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * + + StatusCode VertexCaloIsolation::finalize() + { + // everything all right + return StatusCode::SUCCESS; + } + + // * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * + + StatusCode VertexCaloIsolation::addBranches() const { + + + // There is also the "MuonClusterCollection" which may already contain all the muon's clusters + + const xAOD::TrackParticleContainer* idTrackParticleContainer = NULL; + const xAOD::VertexContainer* vertexContainer = NULL; + + Rec::CaloCellCollector cellCollector; //To keep private dependence for this package it is used here + + + const xAOD::MuonContainer* muons = NULL; + + + //Load InDetTrackParticles + if(evtStore()->contains<xAOD::TrackParticleContainer>(m_trackContainerName)) { + CHECK( evtStore()->retrieve(idTrackParticleContainer, m_trackContainerName) ); + } + else{ATH_MSG_ERROR("Failed loading IdTrackparticleContainer container"); + return StatusCode::FAILURE; + } + + // load vertices + if(evtStore()->contains<xAOD::VertexContainer>(m_vertexContainerName)) { + CHECK( evtStore()->retrieve(vertexContainer, m_vertexContainerName) ); + } + else{ATH_MSG_ERROR("Failed loading vertex container"); + return StatusCode::FAILURE; + } + + + const xAOD::CaloClusterContainer* caloClusterContainer = NULL; + // load CaloCalTopoClusters + if(evtStore()->contains<xAOD::CaloClusterContainer>(m_caloClusterContainerName)) { + CHECK( evtStore()->retrieve(caloClusterContainer, m_caloClusterContainerName) ); + } + else{ATH_MSG_ERROR("Failed loading vertex container"); + return StatusCode::FAILURE; + } + + + //Retrieve muon container + if(evtStore()->contains<xAOD::MuonContainer>(m_muonContainerName)) { + CHECK( evtStore()->retrieve(muons, m_muonContainerName) ); + } + else{ATH_MSG_ERROR("Failed loading muon contianer"); + return StatusCode::FAILURE; + } + + +//------------------------------------------------- + + std::vector<xAOD::Iso::IsolationType> cones; cones.resize(m_cones.size()); + +// for(unsigned int cone : m_cones) +// cones.push_back(xAOD::Iso::IsolationType(cone)); + + for (unsigned int i =0; i< m_cones.size(); i++) + cones[i] = xAOD::Iso::IsolationType(m_cones[i]); + + + + //Loop over vertices + for(auto vertex : *vertexContainer){ + + bool passed = false; + for(std::vector<std::string>::const_iterator flagItr = m_passFlags.begin(); flagItr!=m_passFlags.end(); ++flagItr) { + SG::AuxElement::Accessor<Char_t> flagAcc(*flagItr); + if(flagAcc.isAvailable(*vertex) && flagAcc(*vertex) != 0) { + passed = true; + break; + } + } // end of loop over flags + if(passed){ + ATH_MSG_DEBUG("Entered loop over vertices"); + if(vertex->trackParticleLinks().size() != 3)ATH_MSG_WARNING("Vertex without 3 tracks, it has "<< vertex->trackParticleLinks().size() <<" instead"); + + TLorentzVector candidate; + + std::set<const xAOD::TrackParticle*> exclusionset; + + for(auto part : vertex->trackParticleLinks()){ //Loop over tracks linked to vertex + candidate += (*part)->p4(); + exclusionset.insert( *part ); //If it crashes use the direct TP from the vertex + } + + //List of corrections: only the pileup correction is applied within the tool + xAOD::CaloCorrection corrlist; + corrlist.calobitset.set(static_cast<unsigned int>(xAOD::Iso::pileupCorrection)); + + + + std::vector<const xAOD::Muon*> vtxMuons; + std::vector<TLorentzVector> extrVtxMuons ; + std::vector<const xAOD::CaloCluster*> vtxMuonCluster; + + std::vector<const xAOD::TrackParticle*> usedVtxTracks; + //The information whether we are missing some core-corrections in the final isolation value + map<xAOD::Iso::IsolationType, bool> is_reliable; + + + TLorentzVector muonref; //Place holder for the extrapolated position + //Load the caloclusters of the various muons (which you need to load from here) + const xAOD::MuonContainer* muons = 0; + CHECK( evtStore()->retrieve( muons, "Muons" ) ); + for ( auto muon : *muons ) { + //I ask for all information to be fine before filling in an entry (so all containers will have the same -matching- objects) + if(muon->inDetTrackParticleLink().isValid() && exclusionset.find(*muon->inDetTrackParticleLink() ) != exclusionset.end() ){ + const xAOD::CaloCluster* clus = muon->cluster(); + if(clus && extrapolateMuon(muonref, clus)){ + // have a muon, an extrapolation and a cluster (hurray) + vtxMuonCluster.push_back(clus); + vtxMuons.push_back(muon); + usedVtxTracks.push_back( *muon->inDetTrackParticleLink() ); + extrVtxMuons.push_back(muonref); + }else{ + ATH_MSG_DEBUG("Cannot find clusters. Would need a consistent set of Trk::Tracks to run extrapolation."); + + // //If working with the cluster failed, try extrapolating the track + // if(extrapolateTrack(muonref, *muon)){ //This does not use the muonic cluster, but uses both its tracks to determine a precise position + // vtxMuonCluster.push_back(clus); //Note clus can also be NULL (for if it's not in the cone there is no point to fret) + // vtxMuons.push_back(muon); + // usedVtxTracks.push_back( *muon->inDetTrackParticleLink() ); + // extrVtxMuons.push_back(muonref); + //} + } + } + } + + + //What if there was a track and not a muon?? + //Should be treated like the muon-without-cluster case + + if(vtxMuonCluster.size() !=3){ //remember that some of the ctxMuonCluster elements may be NULL + ATH_MSG_DEBUG( "Attempt at extrapolating the IDtrack" ); + + //Attempt extrapolating the IDtrack for the missing cases + for(const xAOD::TrackParticle* missingTrk : exclusionset){ + if(std::find(usedVtxTracks.begin(), usedVtxTracks.end(), missingTrk) == usedVtxTracks.end()){ + if(extrapolateTrack(muonref, *missingTrk)){ + vtxMuonCluster.push_back(NULL); //Null, for we didn't start from a muon + usedVtxTracks.push_back( missingTrk ); + extrVtxMuons.push_back(muonref); + } + } + } + + //If there are still missing ones values cannot be guaranteed to be reliable + if(vtxMuonCluster.size() !=3){ + ATH_MSG_DEBUG( "For this vertex there were less than 3 muons found (or extrapolated)" ); + for(xAOD::Iso::IsolationType isoCone : cones) is_reliable[isoCone] = false; + } + } + else{ + for(xAOD::Iso::IsolationType isoCone : cones) + is_reliable[isoCone] = true; + } + + + // Adapt this loop! + + for(unsigned int vertex_type = 0 ; vertex_type<= xAOD::BPhysHelper::PV_MIN_Z0 ; vertex_type++ ){ + + if((m_vertexType & (1 << vertex_type ) ) == 0)continue; //Stop if the type of vertex is not required + + //This can be in an inside loop + + xAOD::BPhysHelper::pv_type this_type = static_cast<xAOD::BPhysHelper::pv_type>( vertex_type ); + + xAOD::TrackParticle candidate_slyTrack; + makeSlyTrack(candidate_slyTrack, candidate, vertex, this_type); + + + xAOD::CaloIsolation result; + + ATH_MSG_DEBUG("Check if the caloclus container has to be given or not... see line from 755 on of CaloIsolationTool"); + + bool successful = m_caloIsoTool->caloTopoClusterIsolation(result, candidate_slyTrack, cones, corrlist, caloClusterContainer); + if( !successful ) { + ATH_MSG_DEBUG("Calculation of caloTopoClusterIsolation failed"); + return StatusCode::FAILURE; + } + + // Make the extension to the calorimeter, as it is done inside the other tools... + TLorentzVector extr_candidate; + if( !extrapolateTrack(extr_candidate, candidate_slyTrack) ){ + ATH_MSG_WARNING("Failure extrapolating the slyTrack "<<"pt="<<candidate_slyTrack.pt()<<" eta="<<candidate_slyTrack.eta()<<" phi="<<candidate_slyTrack.phi()); + ATH_MSG_WARNING("Taking the original coordinates"); + } + + + std::map<xAOD::Iso::IsolationType,float> coreCorrections; + + //See if this is inside the cone, to determine the correct correction ^^ + for(xAOD::Iso::IsolationType isoType : cones){ + + double conesize = xAOD::Iso::coneSize(isoType); + //check what is inside the cone + std::vector<xAOD::CaloCluster> clustersInCone; + + for(unsigned int j=0; j < vtxMuonCluster.size(); j++){ + auto mucluster = vtxMuonCluster[j]; + // I should use the propagated values here, though the variation is very small, coming from the vertex position + float dr=extrVtxMuons[j].DeltaR(extr_candidate); + + + + ATH_MSG_DEBUG("Cone size: "<<conesize<<" dr="<<dr); + ATH_MSG_DEBUG(extrVtxMuons[j].Eta() <<" - "<<extr_candidate.Eta()<<" and "<<extrVtxMuons[j].Phi() <<" - "<<extr_candidate.Phi()); + + if( dr < conesize ){ //This makes a copy, such that I can remove some cells if needed + + + //here do the check for the cluster, if it should go in, then prevent and set the bad for this cone + if(mucluster != NULL) clustersInCone.push_back( xAOD::CaloCluster(*mucluster) ); + else is_reliable[isoType] = false; + + + + } + } + + // ATH_MSG_DEBUG("Muon clusters in cone "<<xAOD::Iso::toString(isoType)<<" "<< clustersInCone.size()); + // if( msgLvl(MSG::DEBUG) ){ + // for(auto muon : vtxMuons) + // if(muon->isAvailable<float>("ET_Core")) ATH_MSG_DEBUG("ET_core stored inside: "<< muon->auxdataConst<float>("ET_Core") ); + // } + //remove eventually doubles in cells + if(clustersInCone.size() == 2){ + for(auto cell : clustersInCone[0]){ + clustersInCone[1].removeCell(cell); + } + } + if(clustersInCone.size() == 3){ + for(auto cell : clustersInCone[0]){ + clustersInCone[1].removeCell(cell); + clustersInCone[2].removeCell(cell); + } + for(auto cell : clustersInCone[1]){ + clustersInCone[2].removeCell(cell); + } + } + + //Calculate the core-correction + std::vector<float> etcore(4, 0.); + float coreCorr=0.; + for(auto cl : clustersInCone){ + if(cl.size() != 0){ //Maybe two muons have a full cluster overlap?? + ATH_MSG_DEBUG("Cells in this cluster: "<< cl.size()); + + cellCollector.collectEtCore( cl, etcore, nullptr, m_sigmaCaloNoiseCut ); //Note an empty handle to ICaloNoiseTool is passed + coreCorr += etcore[Rec::CaloCellCollector::ET_Core]; + ATH_MSG_DEBUG("Their core-energy: "<< etcore[Rec::CaloCellCollector::ET_Core]); + + } + } + + //Store the core-correction + coreCorrections[isoType] = coreCorr; + + } + + //For a pion I have no such cone energy, do I? But then I should also see what the original vertex was + //If something is not a muon there is no way the calocluster was stored, I think + //Would need further study + + + //Collect all the required information + string ED("_EDcorr"); + string core("_COREcorr"); + string reliable("_isReliable"); + + + string vtx_type[3] = {"SumPt", "A0", "Z0"}; + + string vtx = vtx_type[ vertex_type ]; + + ATH_MSG_DEBUG("Detailed: "); + for(unsigned int i=0; i< cones.size(); i++){ + xAOD::Iso::IsolationType isoType = cones[i]; + result.etcones[i] -= coreCorrections[isoType]; //Finish correcting the energy + + + // if(fabs(result.etcones[i]) < 0.1){ + // + // ATH_MSG_INFO("Isolation: "<<xAOD::Iso::toString(isoType) ); // The name of the isolation + // ATH_MSG_ERROR(result.etcones[i]<<" + "<<(result.noncoreCorrections[xAOD::Iso::pileupCorrection])[i]<<" + "<<coreCorrections[isoType] ); + // } + + //Here do the decoration (store all, and as well if three muons are found) + + + string variableName = xAOD::Iso::toString(isoType) + vtx; //I corrected for the closest vertex in A0 + SG::AuxElement::Decorator<float> isolation(variableName); + isolation(*vertex) = result.etcones[i]; + + isolation = SG::AuxElement::Decorator<float>(variableName + ED); + isolation(*vertex) = (result.noncoreCorrections[xAOD::Iso::pileupCorrection])[i]; + + isolation = SG::AuxElement::Decorator<float>(variableName + core); + isolation(*vertex) = coreCorrections[isoType]; + + //This variable contains the info whether 3 caloclusters have been found in the muons + //Future would be to see if their extrapolations are of interest anyhow (if not missing them is no issue) + //Fore some reason these seem to become chars (instead of bools) in the output + SG::AuxElement::Decorator<bool> reliability(variableName + reliable); + reliability(*vertex) = is_reliable[isoType]; + + } + } //Loop over primaryVertex choice + + //Decorate the candidate with the new information + + +// return StatusCode::SUCCESS; +// +// /////////////////////////////////// + + } + +//END OF NEW PART + }//End of loop over vertices + return StatusCode::SUCCESS; + } + + //Note that the full version had a different method for muons!!!! Maybe I should use that one instead! + + //This is almost a perfect copy of CaloIsolationTool::GetExtrapEtaPhi, but only for the part relative to tracks + bool VertexCaloIsolation::extrapolateTrack(TLorentzVector& extr_tp, const xAOD::IParticle& tp) const{ + extr_tp = tp.p4(); //Pre-set the output TLorentzVector to the input's 4-momentum + ATH_MSG_ERROR("VertexCaloIsolation::extrapolateTrack needs to be rewritten because of changes to the caloExtension"); + throw std::runtime_error("VertexCaloIsolation::extrapolateTrack needs to be rewritten because of changes to the caloExtension"); +/* + + + const Trk::CaloExtension* caloExtension = 0; + if(!m_caloExtTool->caloExtension(tp,caloExtension)){ + ATH_MSG_WARNING("Can not get caloExtension."); + return false; + } + + const std::vector<const Trk::CurvilinearParameters*>& intersections = caloExtension->caloLayerIntersections(); + if (intersections.size()>0) { + Amg::Vector3D avePoint(0,0,0); + for (unsigned int i = 0; i < intersections.size(); ++i){ + const Amg::Vector3D& point = intersections[i]->position(); + avePoint += point; + } + avePoint = (1./intersections.size())*avePoint; + + + extr_tp.SetPtEtaPhiE(1., avePoint.eta(), avePoint.phi(), 10.); //Using the three-vector constructor + //eta = avePoint.eta(); + //phi = avePoint.phi(); + ATH_MSG_DEBUG("Successfully extrapolated candidate eta/phi : "<<tp.eta()<<"/"<<tp.phi()<<" --> "<< extr_tp.Eta()<<"/"<<extr_tp.Phi()); + + } + else{ //This is very unlikely, it happens if a few cases in MC + ATH_MSG_WARNING("Candidate extrapolation failed. Keeping track's eta/phi values"); + return false; + + } + + return true; +*/ + + } + + //Version for the muons + bool VertexCaloIsolation::extrapolateMuon(TLorentzVector& extr_tp, const xAOD::CaloCluster* cluster) const + { + //auto cluster = mu->cluster(); //done outside + if(cluster){ + float etaT = 0, phiT = 0; + int nSample = 0; + for(unsigned int i=0; i<CaloSampling::Unknown; i++) // dangerous? + { + auto s = static_cast<CaloSampling::CaloSample>(i); + if(!cluster->hasSampling(s)) continue; + //ATH_MSG_DEBUG("Sampling: " << i << "eta-phi (" << cluster->etaSample(s) << ", " << cluster->phiSample(s) << ")"); + etaT += cluster->etaSample(s); + phiT += cluster->phiSample(s); + nSample++; + } + if(nSample>0){ + + extr_tp.SetPtEtaPhiE(1., etaT/nSample, phiT/nSample, 10.); //Using the three-vector constructor + return true ; + + }else{ + ATH_MSG_WARNING("Muon calo cluster is empty????"); + return false; + } + }else{ + ATH_MSG_WARNING("Muon calo cluster not found. Calo extension can not be obtained!!!"); + return false; + } + } + + //Make a sly track to be fed to the CaloIsolationTool + xAOD::TrackParticle& VertexCaloIsolation::makeSlyTrack(xAOD::TrackParticle& candidate_slyTrack, const TLorentzVector& candidate, const xAOD::Vertex* vertex, xAOD::BPhysHelper::pv_type vertexType) const { + + candidate_slyTrack.makePrivateStore(); + candidate_slyTrack.setDefiningParameters(0, 0., candidate.Phi(), candidate.Theta(), 0. ); // avoided q/p = 1./candidate.P() + + //I should set the correct d0 and z0, while setting momentum to enormous, to obtain a straight line + //I fear that q/p == 0 might cause some divide by 0, though. + + //Somewhere this information will be checked, so I need to provide it + SG::AuxElement::Decorator<uint8_t> hypothesis("particleHypothesis"); + hypothesis(candidate_slyTrack) = xAOD::undefined; //Value 99 as none of the common types (muon, pion, kaon, etc.) + SG::AuxElement::Decorator<std::vector<float> > covmat( "definingParametersCovMatrix" ); + covmat(candidate_slyTrack) = std::vector<float>(25, 0.); // I am saying that there are no errors on my parameters + //The precision goes down a bit, but it's a matter of 10e-7 with our values of interest + + xAOD::BPhysHelper vertex_h(vertex); //Use the BPhysHelper to access vertex quantities + + SG::AuxElement::Decorator<float> vx( "vx" ); + vx(candidate_slyTrack) = vertex_h.pv(vertexType)->x(); + + SG::AuxElement::Decorator<float> vy( "vy" ); + vy(candidate_slyTrack) = vertex_h.pv(vertexType)->y(); + + SG::AuxElement::Decorator<float> vz( "vz" ); + vz(candidate_slyTrack) = vertex_h.pv(vertexType)->z(); + //The precision goes down a bit, but it's a matter of 10e-7 with our values of interest + + return candidate_slyTrack; + + + } + + + + +}//End of namespace DerivationFramework + + diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/VertexPlus1TrackCascade.cxx b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/VertexPlus1TrackCascade.cxx new file mode 100644 index 00000000000..b3c4c221dd4 --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/VertexPlus1TrackCascade.cxx @@ -0,0 +1,215 @@ +/* + Copyright (C) 2002-2017 CERN for the benefit of the ATLAS collaboration +*/ + + +#include "DerivationFrameworkBPhys/VertexPlus1TrackCascade.h" + +#include "TrkVertexFitterInterfaces/IVertexFitter.h" +#include "TrkVKalVrtFitter/TrkVKalVrtFitter.h" +#include "TrkToolInterfaces/ITrackSelectorTool.h" + +namespace DerivationFramework { + + + typedef std::vector<const xAOD::TrackParticle*> TrackBag; + + StatusCode VertexPlus1TrackCascade::initialize() { + + // retrieving vertex Fitter + if ( m_iVertexFitter.retrieve().isFailure() ) { + ATH_MSG_FATAL("Failed to retrieve tool " << m_iVertexFitter); + return StatusCode::FAILURE; + } else { + ATH_MSG_DEBUG("Retrieved tool " << m_iVertexFitter); + } + + // Get the track selector tool from ToolSvc + if ( m_trkSelector.retrieve().isFailure() ) { + ATH_MSG_FATAL("Failed to retrieve tool " << m_trkSelector); + return StatusCode::FAILURE; + } else { + ATH_MSG_DEBUG("Retrieved tool " << m_trkSelector); + } + if(!m_vertexContainerKey.key().empty()) ATH_CHECK(m_vertexContainerKey.initialize()); + if(!m_TrackPContainerKey.key().empty()) ATH_CHECK(m_TrackPContainerKey.initialize()); + if(!m_MuonsUsedInJpsiKey.key().empty()) ATH_CHECK(m_MuonsUsedInJpsiKey.initialize()); + + return StatusCode::SUCCESS; + } + + StatusCode VertexPlus1TrackCascade::finalize() { + + return StatusCode::SUCCESS; + + } + + VertexPlus1TrackCascade::VertexPlus1TrackCascade(const std::string& t, const std::string& n, const IInterface* p) : AthAlgTool(t,n,p), + m_vertexContainerKey(""), + m_TrackPContainerKey(""), + m_MuonsUsedInJpsiKey(""), + m_Vtx1MassConstraint(0.), + m_Vtx2MassConstraint(0.0), + m_trkThresholdPt(0.0), + m_trkMaxEta(102.5), +// m_BThresholdPt(0.0), +// m_BMassUpper(0.0), +// m_BMassLower(0.0), + m_roughMassLower(0.0), + m_roughMassUpper(0.0), + m_iVertexFitter("Trk::TrkVKalVrtFitter"), + m_trkSelector("InDet::TrackSelectorTool") + { + declareProperty("InitialVertices", m_vertexContainerKey); + declareProperty("TrackParticleCollection", m_TrackPContainerKey); + declareProperty("MuonCollection", m_MuonsUsedInJpsiKey); + declareProperty("MassHypthesis", m_massHypothesis); + declareProperty("MassContraintTracksVtx1", m_massConstraintTracksVtx1); + declareProperty("MassContraintTracksVtx2", m_massConstraintTracksVtx2); + + declareProperty("Vtx1MassConstraint", m_Vtx1MassConstraint); + declareProperty("Vtx2MassConstraint", m_Vtx2MassConstraint); + + declareProperty("trkThresholdPtCut", m_trkThresholdPt); + declareProperty("trkMassEtaCut", m_trkMaxEta); +// declareProperty("BThresholdPtCut", m_BThresholdPt); +// declareProperty("BMassUpperCut", m_BMassUpper); +// declareProperty("BMassLowerCut", m_BMassLower); + + declareProperty("RoughMassUpperCut", m_roughMassLower); + declareProperty("RoughMassLowerCut", m_roughMassUpper); + + } + + VertexPlus1TrackCascade::~VertexPlus1TrackCascade(){ } + + double VertexPlus1TrackCascade::getInvariantMass(const TrackBag &Tracks, const std::vector<double> &massHypotheses){ + + TLorentzVector total; + total.SetVectM(Tracks[0]->p4().Vect(), massHypotheses[0]); + TLorentzVector temp; + for(size_t i=1; i < Tracks.size(); i++){ + temp.SetVectM(Tracks[i]->p4().Vect(), massHypotheses[i]); + total += temp; + } + return total.M(); + } + + bool VertexPlus1TrackCascade::isContainedIn(const xAOD::TrackParticle* theTrack, const xAOD::MuonContainer* theColl) { + bool isContained(false); + for (auto muItr=theColl->cbegin(); muItr!=theColl->cend(); ++muItr) { + auto& link = ( *muItr )->inDetTrackParticleLink(); + if ( link.isValid() && ( *link == theTrack ) ) {isContained=true; break;} + } + return isContained; + } + + StatusCode VertexPlus1TrackCascade::performSearch(std::vector<Trk::VxCascadeInfo*> *cascadeinfoContainer) const + { + ATH_MSG_DEBUG( "VertexPlus1TrackCascade::performSearch" ); + assert(cascadeinfoContainer!=nullptr); + SG::ReadHandle<xAOD::VertexContainer> vertexContainer(m_vertexContainerKey); + if(!vertexContainer.isValid()){ + ATH_MSG_ERROR("No VertexContainer with key " << m_vertexContainerKey.key() << " found in StoreGate. BCandidates will be EMPTY!"); + return StatusCode::FAILURE; + } + + // Get tracks + SG::ReadHandle<xAOD::TrackParticleContainer> TrackPContainer(m_TrackPContainerKey); + if(!TrackPContainer.isValid()){ + ATH_MSG_ERROR("No track particle collection with name " << m_TrackPContainerKey.key() << " found in StoreGate!"); + return StatusCode::FAILURE; + } + + + // Get the muon collection used to build the J/psis + const xAOD::MuonContainer* importedMuonCollection = nullptr; + if (!m_MuonsUsedInJpsiKey.key().empty()) { + SG::ReadHandle<xAOD::MuonContainer> handle(m_MuonsUsedInJpsiKey); + if(handle.isValid()) importedMuonCollection = handle.cptr(); + else { + ATH_MSG_FATAL("problem retrieving MuonContainer " << m_MuonsUsedInJpsiKey.key()); + return StatusCode::FAILURE; + } + ATH_MSG_DEBUG("Muon container size "<< importedMuonCollection->size()); + } + + // Select the inner detector tracks + TrackBag theIDTracksAfterSelection; + for (auto tp : *TrackPContainer){ + if ( tp->pt()<m_trkThresholdPt ) continue; + if ( fabs(tp->eta())>m_trkMaxEta ) continue; + if (importedMuonCollection!=NULL) { + if (isContainedIn(tp, importedMuonCollection)) continue; + } + if ( m_trkSelector->decision(*tp, NULL) ) theIDTracksAfterSelection.push_back(tp); + } + + const std::vector<double> &fullMassHypoth = (m_massHypothesis); + const std::vector<double> initialVertexMassHypo(fullMassHypoth.begin(), fullMassHypoth.end()-1); + + TrackBag originalVertexTracks(initialVertexMassHypo.size()); + TrackBag secondVertexTracks(fullMassHypoth.size()); + + const std::vector< Trk::VertexID > emptyVtxList; + TrackBag ConstraintTracksVtx1(m_massConstraintTracksVtx1.size()); + TrackBag ConstraintTracksVtx2(m_massConstraintTracksVtx2.size()); + + assert(fullMassHypoth.size() == secondVertexTracks.size()); + + for(auto vertex : *vertexContainer){ //Iterate over previous vertices + + size_t OriginaltrackNum = vertex->nTrackParticles(); + if(initialVertexMassHypo.size() != OriginaltrackNum){ + ATH_MSG_FATAL("Mass hypothesis not correctly set"); + return StatusCode::FAILURE; + } + for(size_t i = 0;i<OriginaltrackNum;i++) + originalVertexTracks[i] = secondVertexTracks[i] = (vertex->trackParticle(i)); + + for(auto newtrack : theIDTracksAfterSelection){ + //Skip any track already used in vertex + if(std::find(originalVertexTracks.begin(), originalVertexTracks.end(), newtrack) != originalVertexTracks.end()) continue; + + secondVertexTracks.back() = newtrack; + + double roughmass = getInvariantMass(secondVertexTracks, fullMassHypoth); + + if(m_roughMassUpper > 0.0 && (roughmass < m_roughMassLower || roughmass > m_roughMassUpper)) continue; + + std::unique_ptr<Trk::IVKalState> state = m_iVertexFitter->makeState(); + m_iVertexFitter->setRobustness( 0, *state ); + + auto vID1 = m_iVertexFitter->startVertex( originalVertexTracks, initialVertexMassHypo, *state ); + auto vID2 = m_iVertexFitter->nextVertex( secondVertexTracks, fullMassHypoth, *state ); + + if(!m_massConstraintTracksVtx1.empty()){ + for(size_t i =0; i<m_massConstraintTracksVtx1.size(); i++) ConstraintTracksVtx1[i] = originalVertexTracks.at(m_massConstraintTracksVtx1[i]); + if( !m_iVertexFitter->addMassConstraint( vID1, ConstraintTracksVtx1, emptyVtxList, *state, m_Vtx1MassConstraint ).isSuccess() ) { + ATH_MSG_WARNING( "cascade fit: addMassConstraint failed" ); + } + } + + if(!m_massConstraintTracksVtx2.empty()){ + for(size_t i =0; i<m_massConstraintTracksVtx2.size(); i++) ConstraintTracksVtx2[i] = secondVertexTracks.at(m_massConstraintTracksVtx2[i]); + if( !m_iVertexFitter->addMassConstraint( vID2, ConstraintTracksVtx2, emptyVtxList,*state, m_Vtx2MassConstraint ).isSuccess() ) { + ATH_MSG_WARNING( "cascade fit: addMassConstraint failed" ); + } + } + + auto result = m_iVertexFitter->fitCascade(*state); + if(result ==nullptr ){ ATH_MSG_WARNING("Cascade Fit failed"); continue; } + assert(result->vertices().size()==2); + cascadeinfoContainer->push_back(result); + + } + + } + ATH_MSG_DEBUG("cascadeinfoContainer size " << cascadeinfoContainer->size()); + return StatusCode::SUCCESS; + } + + +} + + diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/VertexTrackIsolation.cxx b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/VertexTrackIsolation.cxx new file mode 100644 index 00000000000..7ce8df7972f --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/VertexTrackIsolation.cxx @@ -0,0 +1,274 @@ +/* + Copyright (C) 2002-2020 CERN for the benefit of the ATLAS collaboration +*/ + + +#include "DerivationFrameworkBPhys/VertexTrackIsolation.h" + +#include <vector> +#include <string> +#include "TVector3.h" + +#include "xAODTracking/VertexContainer.h" +#include "xAODTracking/VertexAuxContainer.h" +#include "xAODBPhys/BPhysHelper.h" +#include "xAODBPhys/BPhysHypoHelper.h" +#include "TrkVertexAnalysisUtils/V0Tools.h" + +//#include "IsolationTool/CaloIsolationTool.h" +//#include "IsolationTool/TrackIsolationTool.h" +#include "RecoToolInterfaces/ITrackIsolationTool.h" +#include "xAODPrimitives/IsolationHelpers.h" //For the definition of Iso::conesize + +//#include "IsolationTool/IsolationHelper.h" +//#include "InDetTrackSelectionTool/InDetTrackSelectionTool.h" + + +//#include "Identifier/Identifier32.h" +using namespace std; +namespace DerivationFramework { + + VertexTrackIsolation::VertexTrackIsolation(const std::string& t, + const std::string& n, + const IInterface* p) : + AthAlgTool(t,n,p), + m_trackIsoTool("xAOD::TrackIsolationTool"), + m_trackContainerName("InDetTrackParticles"), + m_vertexContainerName("NONE"), + m_cones(), + m_vertexType(7), + + m_doIsoPerTrk(false), + m_removeDuplicate(2) + { + ATH_MSG_DEBUG("in constructor"); + declareInterface<DerivationFramework::IAugmentationTool>(this); + + // Declare tools + declareProperty("TrackIsoTool" , m_trackIsoTool); + + declareProperty("TrackContainer" , m_trackContainerName); + declareProperty("InputVertexContainer" , m_vertexContainerName); + declareProperty("PassFlags" , m_passFlags); + declareProperty("IsolationTypes" , m_cones); + declareProperty("DoVertexTypes" , m_vertexType); + + declareProperty("DoIsoPerTrk" , m_doIsoPerTrk, "New property to deal with track isolation per track, the default option (m_doIsoPerTrk=false) preserves the old behavior"); + declareProperty("RemoveDuplicate" , m_removeDuplicate, "Used with DoIsoPerTrk"); + } + + // * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * + + StatusCode VertexTrackIsolation::initialize() + { + + ATH_MSG_DEBUG("in initialize()"); + + // retrieve TrackIsolationTool + CHECK( m_trackIsoTool.retrieve() ); + + //Check that flags were given to tag the correct vertices + if(m_passFlags.empty()){ + ATH_MSG_WARNING("As no pass-flags are given, no vertices will be decorated"); + } + + // Control the IsolationType sequence + if(m_cones.empty()){ + ATH_MSG_INFO("Setting ptcones to default"); + + if(m_doIsoPerTrk) m_cones.push_back(xAOD::Iso::ptcone50); + m_cones.push_back(xAOD::Iso::ptcone40); + m_cones.push_back(xAOD::Iso::ptcone30); + m_cones.push_back(xAOD::Iso::ptcone20); + + } + + return StatusCode::SUCCESS; + + } + + // * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * + + StatusCode VertexTrackIsolation::finalize() + { + // everything all right + return StatusCode::SUCCESS; + } + + // check if the two vertices are composed of the same set of tracks + bool VertexTrackIsolation::isSame(const xAOD::Vertex* theVtx1, const xAOD::Vertex* theVtx2) const { + if(!theVtx1 || !theVtx2) return false; + if(theVtx1==theVtx2) return true; + if(theVtx1->nTrackParticles() != theVtx2->nTrackParticles()) return false; + + if(m_removeDuplicate==2 && theVtx1->nTrackParticles()==4) { // a special case with sub-structure + bool firstTwoAreSame = std::set<const xAOD::TrackParticle*>( { theVtx1->trackParticle(0), theVtx1->trackParticle(1)} ) == std::set<const xAOD::TrackParticle*>( {theVtx2->trackParticle(0), theVtx2->trackParticle(1)} ); // the 1st pair of tracks + bool lastTwoAreSame = std::set<const xAOD::TrackParticle*>( { theVtx1->trackParticle(2), theVtx1->trackParticle(3)} ) == std::set<const xAOD::TrackParticle*>( {theVtx2->trackParticle(2), theVtx2->trackParticle(3)} ); // the 2nd pair of tracks + if(firstTwoAreSame && lastTwoAreSame) return true; + else return false; + } + else { // the general case + std::set<const xAOD::TrackParticle*> vtxset1; + std::set<const xAOD::TrackParticle*> vtxset2; + for(size_t i=0; i<theVtx1->nTrackParticles(); i++) vtxset1.insert(theVtx1->trackParticle(i)); + for(size_t i=0; i<theVtx2->nTrackParticles(); i++) vtxset2.insert(theVtx2->trackParticle(i)); + return vtxset1 == vtxset2; + } + } + + bool VertexTrackIsolation::isContainedIn(const xAOD::Vertex* theVtx, const std::vector<const xAOD::Vertex*> &theColl) const { + for ( const auto vtxPtr : theColl ) { + if ( isSame(vtxPtr, theVtx) ) return true; + } + return false; + } + + // * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * + + StatusCode VertexTrackIsolation::addBranches() const { + + const xAOD::TrackParticleContainer* idTrackParticleContainer = NULL; + const xAOD::VertexContainer* vertexContainer = NULL; + + if(evtStore()->contains<xAOD::TrackParticleContainer>(m_trackContainerName)) { + CHECK( evtStore()->retrieve(idTrackParticleContainer, m_trackContainerName) ); + } + else{ATH_MSG_ERROR("Failed loading IdTrackparticleContainer container"); + return StatusCode::FAILURE; + } + + // load vertices + if(evtStore()->contains<xAOD::VertexContainer>(m_vertexContainerName)) { + CHECK( evtStore()->retrieve(vertexContainer, m_vertexContainerName) ); + } + else{ATH_MSG_ERROR("Failed loading vertex container"); + return StatusCode::FAILURE; + } + + std::vector<const xAOD::Vertex*> outVtxContainer; + + //Convert m_cones (done per-event to avoid needing extra public dependency) + + std::vector<xAOD::Iso::IsolationType> cones; cones.resize(m_cones.size()); + + for (unsigned int i =0; i< m_cones.size(); i++) + cones[i] = xAOD::Iso::IsolationType(m_cones[i]); + //for(unsigned int cone : m_cones) + // cones.push_back(xAOD::Iso::IsolationType(cone)); + + ATH_MSG_DEBUG("The provided IsolationTypes are re-ordered internally"); + std::sort(cones.begin(),cones.end(),[](xAOD::Iso::IsolationType i, xAOD::Iso::IsolationType j) { return xAOD::Iso::coneSize(i) > xAOD::Iso::coneSize(j); } ); + + // loop over vertices + for(auto vertex : *vertexContainer){ + + bool passed = false; + for(std::vector<std::string>::const_iterator flagItr = m_passFlags.begin(); flagItr!=m_passFlags.end(); ++flagItr) { + SG::AuxElement::Accessor<Char_t> flagAcc(*flagItr); + if(flagAcc.isAvailable(*vertex) && flagAcc(*vertex) != 0) { + passed = true; + break; + } + } // end of loop over flags + if(passed){ + if(!m_doIsoPerTrk) { // for legacy + if(vertex->trackParticleLinks().size() != 3)ATH_MSG_WARNING("Vertex without 3 tracks, it has "<< vertex->trackParticleLinks().size() <<" instead"); + } + else { + if(m_removeDuplicate) { + if( isContainedIn(vertex, outVtxContainer) ) continue; + outVtxContainer.push_back(vertex); + } + } + + TLorentzVector candidate; + + std::set<const xAOD::TrackParticle*> exclusionset; + + for(auto part : vertex->trackParticleLinks()){ //Loop over tracks linked to vertex + candidate += (*part)->p4(); + exclusionset.insert( *part ); //If it crashes use the direct TP from the vertex + } + //No! the above candidate will fail acceptance of isolation() because it's neither a muon nor a TrackParticle + + //Make a dummy TrackParticle, otherwise TrackIsolationTool cannot deal with it + xAOD::TrackParticle candidate_slyTrack; + candidate_slyTrack.makePrivateStore(); + candidate_slyTrack.setDefiningParameters(0, 0., candidate.Phi(), candidate.Theta(), 0./*1./candidate.P()*/); + //The precision goes down a bit, but it's a matter of 10e-7 with our values of interest + + //Make a correctionlist such that the given exclusionset will be removed from the used tracks + //There is no danger that the input particle will be excluded, as it is not part of inDetTrackContainer + xAOD::TrackCorrection corrlist; + corrlist.trackbitset.set(static_cast<unsigned int>(xAOD::Iso::coreTrackPtr)); + + + string vtxType_name[3] = {"SumPt", "A0", "Z0"}; + + xAOD::BPhysHelper vertex_h(vertex); //Use the BPhysHelper to access vertex quantities + + + //Loop over refitted primary vertex choice + for(unsigned int vertex_type = 0 ; vertex_type<= xAOD::BPhysHelper::PV_MIN_Z0 ; vertex_type++ ){ + + if((m_vertexType & (1 << vertex_type ) ) == 0)continue; //Stop if the type of vertex is not required + + + //if(debug should go outside!!!) + + ATH_MSG_DEBUG("List of cone types" ); + + + for(unsigned int i =0; i < cones.size(); i++){ + + ATH_MSG_DEBUG("cone type = "<< xAOD::Iso::toString(xAOD::Iso::IsolationType(cones[i])) ); + // ATH_MSG_DEBUG("isolation value "<< vtxType_name[vertex_type] << " = "<< result.ptcones[i] ); + // ATH_MSG_DEBUG("isolation value "<<vtxType_name[vertex_type] <<" = "<< result.ptcones[i] ); + } + + + + const xAOD::Vertex* refVtx = vertex_h.pv( static_cast<xAOD::BPhysHelper::pv_type>(vertex_type) ); //Fix the cast + + xAOD::TrackIsolation result; + + if(!m_doIsoPerTrk) { + m_trackIsoTool->trackIsolation(result, candidate_slyTrack, cones, corrlist, refVtx, &exclusionset, idTrackParticleContainer); + + + //Decorate the vertex with all the isolation values + for(unsigned int i =0; i < cones.size(); i++){ + + string variableName; + + variableName = xAOD::Iso::toString(xAOD::Iso::IsolationType(cones[i])); + variableName += vtxType_name[vertex_type]; + + SG::AuxElement::Decorator<float> isolation(variableName); + isolation(*vertex) = result.ptcones[i]; + + } + } + else { + for(size_t i=0; i<vertex->nTrackParticles(); i++) { + m_trackIsoTool->trackIsolation(result, *vertex->trackParticle(i), cones, corrlist, refVtx, &exclusionset, idTrackParticleContainer); + + for(unsigned int j =0; j < cones.size(); j++) { + string variableName; + variableName = xAOD::Iso::toString(xAOD::Iso::IsolationType(cones[j])); + variableName += vtxType_name[vertex_type] + "_trk" + std::to_string(i+1); + SG::AuxElement::Decorator<float> isolation(variableName); + isolation(*vertex) = result.ptcones[j]; + } + } + } + } + + }// End of if passed + }// end of loop over vertices + + return StatusCode::SUCCESS; + } + +}//End of namespace DerivationFramework + diff --git a/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/components/DerivationFrameworkBPhys_entries.cxx b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/components/DerivationFrameworkBPhys_entries.cxx new file mode 100644 index 00000000000..9bd3917c40d --- /dev/null +++ b/PhysicsAnalysis/DerivationFramework/DerivationFrameworkBPhys/src/components/DerivationFrameworkBPhys_entries.cxx @@ -0,0 +1,74 @@ +#include "DerivationFrameworkBPhys/Reco_Vertex.h" +#include "DerivationFrameworkBPhys/Reco_4mu.h" +#include "DerivationFrameworkBPhys/Select_onia2mumu.h" +#include "DerivationFrameworkBPhys/Thin_vtxTrk.h" +#include "DerivationFrameworkBPhys/Thin_vtxDuplicates.h" +#include "DerivationFrameworkBPhys/AugOriginalCounts.h" +#include "DerivationFrameworkBPhys/BPhysPVThinningTool.h" +#include "DerivationFrameworkBPhys/VertexCaloIsolation.h" +#include "DerivationFrameworkBPhys/VertexTrackIsolation.h" +#include "DerivationFrameworkBPhys/BPhysMetadataBase.h" +#include "DerivationFrameworkBPhys/Bmumu_metadata.h" +//#include "DerivationFrameworkBPhys/CfAthAlgTool.h" +#include "DerivationFrameworkBPhys/Bmumu_reco_mumu.h" +#include "DerivationFrameworkBPhys/FourMuonTool.h" +//#include "DerivationFrameworkBPhys/BPhysAddMuonBasedInvMass.h" +//#include "DerivationFrameworkBPhys/BPhysVertexTrackBase.h" +//#include "DerivationFrameworkBPhys/BVertexTrackIsoTool.h" +//#include "DerivationFrameworkBPhys/BMuonTrackIsoTool.h" +//#include "DerivationFrameworkBPhys/BVertexClosestTrackTool.h" +#include "DerivationFrameworkBPhys/BTrackVertexMapLogger.h" +//#include "DerivationFrameworkBPhys/Select_Bmumu.h" +//#include "DerivationFrameworkBPhys/BPhysVarBlinder.h" +//#include "DerivationFrameworkBPhys/BmumuThinningTool.h" +#include "DerivationFrameworkBPhys/VertexPlus1TrackCascade.h" +#include "DerivationFrameworkBPhys/TriggerCountToMetadata.h" +#include "DerivationFrameworkBPhys/MuonExtrapolationTool.h" +#include "DerivationFrameworkBPhys/CascadeTools.h" +#include "DerivationFrameworkBPhys/Reco_V0Finder.h" +#include "DerivationFrameworkBPhys/JpsiPlusV0Cascade.h" +#include "DerivationFrameworkBPhys/JpsiPlusDsCascade.h" +#include "DerivationFrameworkBPhys/JpsiPlusDpstCascade.h" +#include "DerivationFrameworkBPhys/JpsiPlusDs1Cascade.h" +#include "DerivationFrameworkBPhys/ReVertex.h" +#include "DerivationFrameworkBPhys/BPhysConversionFinder.h" +#include "DerivationFrameworkBPhys/Cascade3Plus1.h" + +using namespace DerivationFramework; + +DECLARE_COMPONENT( Reco_4mu ) +DECLARE_COMPONENT( Reco_Vertex ) +DECLARE_COMPONENT( Select_onia2mumu ) +DECLARE_COMPONENT( Thin_vtxTrk ) +DECLARE_COMPONENT( Thin_vtxDuplicates ) +DECLARE_COMPONENT( AugOriginalCounts ) +DECLARE_COMPONENT( BPhysPVThinningTool ) +DECLARE_COMPONENT( VertexCaloIsolation ) +DECLARE_COMPONENT( VertexTrackIsolation ) +DECLARE_COMPONENT( BPhysMetadataBase ) +DECLARE_COMPONENT( Bmumu_metadata ) +//DECLARE_COMPONENT( CfAthAlgTool ) +//DECLARE_COMPONENT( Bmumu_reco_mumu ) +DECLARE_COMPONENT( FourMuonTool ) +//DECLARE_COMPONENT( BPhysAddMuonBasedInvMass ) +//DECLARE_COMPONENT( BPhysVertexTrackBase ) +//DECLARE_COMPONENT( BVertexTrackIsoTool ) +//DECLARE_COMPONENT( BMuonTrackIsoTool ) +//DECLARE_COMPONENT( BVertexClosestTrackTool ) +DECLARE_COMPONENT( BTrackVertexMapLogger ) +//DECLARE_COMPONENT( Select_Bmumu ) +//DECLARE_COMPONENT( BPhysVarBlinder ) +//DECLARE_COMPONENT( BmumuThinningTool ) +DECLARE_COMPONENT( VertexPlus1TrackCascade ) +DECLARE_COMPONENT( TriggerCountToMetadata ) +DECLARE_COMPONENT( MuonExtrapolationTool ) +DECLARE_COMPONENT( CascadeTools ) +DECLARE_COMPONENT( Reco_V0Finder ) +DECLARE_COMPONENT( JpsiPlusV0Cascade ) +DECLARE_COMPONENT( JpsiPlusDsCascade ) +DECLARE_COMPONENT( JpsiPlusDpstCascade ) +DECLARE_COMPONENT( JpsiPlusDs1Cascade ) +DECLARE_COMPONENT( ReVertex ) +DECLARE_COMPONENT( BPhysConversionFinder ) +DECLARE_COMPONENT( Cascade3Plus1 ) + -- GitLab