diff --git a/PhysicsAnalysis/ElectronPhotonID/ElectronPhotonSelectorTools/ElectronPhotonSelectorTools/EGammaAmbiguityTool.h b/PhysicsAnalysis/ElectronPhotonID/ElectronPhotonSelectorTools/ElectronPhotonSelectorTools/EGammaAmbiguityTool.h index b0a4850a8185e69e9a9f06b993c629f3b230a4f7..124c8efa3f678e45b10513f82b02faf921256ca1 100644 --- a/PhysicsAnalysis/ElectronPhotonID/ElectronPhotonSelectorTools/ElectronPhotonSelectorTools/EGammaAmbiguityTool.h +++ b/PhysicsAnalysis/ElectronPhotonID/ElectronPhotonSelectorTools/ElectronPhotonSelectorTools/EGammaAmbiguityTool.h @@ -29,45 +29,39 @@ public: /** Standard destructor */ - virtual ~EGammaAmbiguityTool(); + virtual ~EGammaAmbiguityTool() = default; public: /** Gaudi Service Interface method implementations */ - virtual StatusCode initialize(); + virtual StatusCode initialize() override final; - /** Gaudi Service Interface method implementations */ - virtual StatusCode finalize(); - - /** Return value: AuthorElectron, AuthorPhoton, AuthorAmbiguous, AuthorUnknown */ + /** Return value: AuthorElectron, AuthorPhoton, AuthorAmbiguous, AuthorUnknown */ - virtual unsigned int ambiguityResolve(const xAOD::CaloCluster* cluster, - const xAOD::Vertex* vx, - const xAOD::TrackParticle* tp, xAOD::AmbiguityTool::AmbiguityType& type) const; + virtual unsigned int ambiguityResolve( + const xAOD::CaloCluster* cluster, + const xAOD::Vertex* vx, + const xAOD::TrackParticle* tp, + xAOD::AmbiguityTool::AmbiguityType& type) const override final; - /** Return value: AuthorElectron, AuthorPhoton, AuthorAmbiguous, AuthorUnknown - Needed because of existing client usage (i.e Trigger). Implementation calls method above + /** Return value: AuthorElectron, AuthorPhoton, AuthorAmbiguous, AuthorUnknown + Needed because of existing client usage (i.e Trigger). Implementation + calls method above */ - virtual unsigned int ambiguityResolve(const xAOD::CaloCluster* cluster, - const xAOD::Vertex* vx, - const xAOD::TrackParticle* tp) const; + virtual unsigned int ambiguityResolve( + const xAOD::CaloCluster* cluster, + const xAOD::Vertex* vx, + const xAOD::TrackParticle* tp) const override final; - /** Access the ambiguity resolution of central electrons and photons and return - * AuthorElectron, AuthorPhoton, AuthorAmbiguous, AuthorUnknown + /** Access the ambiguity resolution of central electrons and photons and + * return AuthorElectron, AuthorPhoton, AuthorAmbiguous, AuthorUnknown * or the author of the object if no overlapping object is found **/ - unsigned int ambiguityResolve(const xAOD::Egamma& egamma) const; + virtual unsigned int ambiguityResolve(const xAOD::Egamma& egamma) const override final; /** Accept or reject egamma object based on ambiguity resolution * (e.g. if object is a photon and ambiguity return value is electron -> reject) **/ - bool accept( const xAOD::Egamma& egamma) const; - - /** Accept or reject egamma object (passed via pointer) based on ambiguity resolution - * (e.g. if object is a photon and ambiguity return value is electron -> reject) - **/ - bool accept( const xAOD::Egamma* egamma) const{ - return accept(*egamma); - } + virtual bool accept( const xAOD::Egamma& egamma) const override final; /** Return true if track has innermost pixel hit * or next-to-innermost in case innermost is not expected diff --git a/PhysicsAnalysis/ElectronPhotonID/ElectronPhotonSelectorTools/Root/EGammaAmbiguityTool.cxx b/PhysicsAnalysis/ElectronPhotonID/ElectronPhotonSelectorTools/Root/EGammaAmbiguityTool.cxx index bf014a645175a7908fd1b90e87ecc8ea8ba54079..a2c62248bf6f0b3208facaf6b37c8abbf5b144fb 100644 --- a/PhysicsAnalysis/ElectronPhotonID/ElectronPhotonSelectorTools/Root/EGammaAmbiguityTool.cxx +++ b/PhysicsAnalysis/ElectronPhotonID/ElectronPhotonSelectorTools/Root/EGammaAmbiguityTool.cxx @@ -2,13 +2,10 @@ Copyright (C) 2002-2020 CERN for the benefit of the ATLAS collaboration */ -// Dear emacs, this is -*-c++-*- - /** @class EGammaAmbiguityTool @brief Electron / photon ambiguity resolution. Dual-use version - - @author Bruno Lenzi + @author Bruno Lenzi , Christos Anastopoulos, Ruggero Turra @date May 2015 */ @@ -29,36 +26,43 @@ #include "xAODEgamma/EgammaxAODHelpers.h" #include "FourMomUtils/xAODP4Helpers.h" -#define CHECK_HITS( EXP ) \ - if (!(EXP)) \ - { \ - ATH_MSG_WARNING("Failed \"" << #EXP << "\"" ); \ - return false; \ - } \ - +#define CHECK_HITS(EXP) \ + if (!(EXP)) { \ + ATH_MSG_WARNING("Failed \"" << #EXP << "\""); \ + return false; \ + } //============================================================================= // Standard constructor //============================================================================= -EGammaAmbiguityTool::EGammaAmbiguityTool(const std::string& myname) : - asg::AsgTool(myname) +EGammaAmbiguityTool::EGammaAmbiguityTool(const std::string& myname) + : asg::AsgTool(myname) { - declareProperty("minNoSiHits", m_MinNoSiHits = 4, "Minimum number of silicon hits to be an electron==>not photon for sure"); - declareProperty("minNoPixHits", m_MinNoPixHits = 2, "Minimum number of Pixel hits to be an electron==>not photon for sure"); - declareProperty("maxEoverPCut", m_maxEoverPCut = 10,"Maximum EoverP , more that this is ambiguous"); - declareProperty("minPCut", m_minPtCut = 2000 , "Minimum Pt, less than that is ambiguous"); - declareProperty("maxDeltaR_innermost", m_maxDeltaR_innermost = 40 , "Maximum value for Rconv - RfirstHit for Si+Si conversions where both tracks have innermost hits"); - declareProperty("noVertexNoInnermostAsAmb", m_noVertexNoInnermostAsAmb = true, - "If true classify as ambiguous when there is no vertex and no innermost Hits. " - "If false classify no vertex as for sure not photon"); - declareProperty("AcceptAmbiguous", m_acceptAmbiguous = true,"When used as a selector accept the ambiguous as default"); -} - - -//============================================================================= -// Standard destructor -//============================================================================= -EGammaAmbiguityTool::~EGammaAmbiguityTool(){ + declareProperty( + "minNoSiHits", + m_MinNoSiHits = 4, + "Minimum number of silicon hits to be an electron==>not photon for sure"); + declareProperty( + "minNoPixHits", + m_MinNoPixHits = 2, + "Minimum number of Pixel hits to be an electron==>not photon for sure"); + declareProperty("maxEoverPCut", + m_maxEoverPCut = 10, + "Maximum EoverP , more that this is ambiguous"); + declareProperty( + "minPCut", m_minPtCut = 2000, "Minimum Pt, less than that is ambiguous"); + declareProperty("maxDeltaR_innermost", + m_maxDeltaR_innermost = 40, + "Maximum value for Rconv - RfirstHit for Si+Si conversions " + "where both tracks have innermost hits"); + declareProperty("noVertexNoInnermostAsAmb", + m_noVertexNoInnermostAsAmb = true, + "If true classify as ambiguous when there is no vertex and " + "no innermost Hits. " + "If false classify no vertex as for sure not photon"); + declareProperty("AcceptAmbiguous", + m_acceptAmbiguous = true, + "When used as a selector accept the ambiguous as default"); } StatusCode EGammaAmbiguityTool::initialize(){ @@ -66,33 +70,24 @@ StatusCode EGammaAmbiguityTool::initialize(){ return StatusCode::SUCCESS; } - -//============================================================================= -// finalize method (now called by destructor) -//============================================================================= -StatusCode EGammaAmbiguityTool::finalize(){ - return StatusCode::SUCCESS; -} - // ==================================================================== // return value: AuthorElectron, AuthorPhoton, AuthorAmbiguous, AuthorUnknown - unsigned int EGammaAmbiguityTool::ambiguityResolve(const xAOD::CaloCluster* cluster, const xAOD::Vertex* vx, const xAOD::TrackParticle* tp, xAOD::AmbiguityTool::AmbiguityType& type) const{ type=xAOD::AmbiguityTool::unknown; - // Energy from 3x5 cluster, stored in altE after EMClusterTool // Result should stay the same before after calibration corrections - double cluster_e = (cluster->clusterSize() != xAOD::CaloCluster::SW_35ele && - cluster->clusterSize() != xAOD::CaloCluster::SW_35gam && - cluster->clusterSize() != xAOD::CaloCluster::SuperCluster) ? cluster->altE() : cluster->e(); + double cluster_e = (cluster->clusterSize() != xAOD::CaloCluster::SW_35ele && + cluster->clusterSize() != xAOD::CaloCluster::SW_35gam && + cluster->clusterSize() != xAOD::CaloCluster::SuperCluster) + ? cluster->altE() + : cluster->e(); - //Number of hits from the track uint8_t trkPixelHits(0); - uint8_t trkSiHits(0); + uint8_t trkSiHits(0); if (tp && !tp->summaryValue(trkPixelHits,xAOD::numberOfPixelHits)){ ATH_MSG_WARNING("Could not retrieve number of pixel hits from track"); } @@ -100,11 +95,11 @@ unsigned int EGammaAmbiguityTool::ambiguityResolve(const xAOD::CaloCluster* clus ATH_MSG_WARNING("Could not retrieve number of SCT hits from track"); } trkSiHits += trkPixelHits; - + //Check if the vertex is double silicon xAOD::EgammaParameters::ConversionType convType(xAOD::EgammaHelpers::conversionType(vx)); bool vxDoubleSi = (convType == xAOD::EgammaParameters::doubleSi); - + // Check if the track particle and the ones associated to the conversion vertex // have innermost pixel hits bool trkHasInnermostHit = tp && hasInnermostPixelHit(*tp); @@ -114,19 +109,19 @@ unsigned int EGammaAmbiguityTool::ambiguityResolve(const xAOD::CaloCluster* clus //if not we might have a trident. //if yes and the track is not good we definetely matched the conversion as electron. bool shareTrack = tp && trkSiHits>=m_MinNoSiHits && vxDoubleSi && isInVertex(*tp, *vx); - + //Debug messages - ATH_MSG_DEBUG("Vertex, SiSi, tracks with innermost pixel hits: " + ATH_MSG_DEBUG("Vertex, SiSi, tracks with innermost pixel hits: " << (vx != nullptr) << ", " << vxDoubleSi << ", " << nTrkVxWithInnermostHit); - ATH_MSG_DEBUG("Track, Si hits, pixel hits, has innermost pixel hit: " - << (tp != nullptr) << ", " << (int) trkSiHits << ", " << (int) trkPixelHits + ATH_MSG_DEBUG("Track, Si hits, pixel hits, has innermost pixel hit: " + << (tp != nullptr) << ", " << (int) trkSiHits << ", " << (int) trkPixelHits << " , " << (int) trkHasInnermostHit); ATH_MSG_DEBUG("Share track : " << shareTrack); //The Electron collection will NOT contain Photons //The Photon collection will NOT contain Electrons - //So for conversions vs electrons better to go to ambiguous + //So for conversions vs electrons better to go to ambiguous //except if we are sure. //Practically a photon here means definetely not Electron @@ -139,19 +134,19 @@ unsigned int EGammaAmbiguityTool::ambiguityResolve(const xAOD::CaloCluster* clus // - No pixel // - The electron track is part of the conversion // In this case we do not want this to be in Electrons - - if (!tp || - trkSiHits<m_MinNoSiHits || + if (!tp || + trkSiHits<m_MinNoSiHits || (vxDoubleSi && !trkPixelHits && shareTrack)){ ATH_MSG_DEBUG("Returning Photon"); type= xAOD::AmbiguityTool::photon; return xAOD::EgammaParameters::AuthorPhoton; } - - //Ambigous due to E/P, Min Pt, pixel hits - //We are not sure it is a Photon, but is not good enough either to be surely an Electron + + // Ambigous due to E/P, Min Pt, pixel hits + // We are not sure it is a Photon, but is not good enough either to be surely + // an Electron // - E/P >10 or track Pt < 2.0 GeV or no-pixel then Ambiguous - double track_ep= cluster_e * fabs(tp->qOverP()); + double track_ep = cluster_e * fabs(tp->qOverP()); if(tp->pt()<m_minPtCut) { ATH_MSG_DEBUG("Returning Ambiguous due min Pt"); @@ -172,58 +167,63 @@ unsigned int EGammaAmbiguityTool::ambiguityResolve(const xAOD::CaloCluster* clus return xAOD::EgammaParameters::AuthorAmbiguous; } - //Electron ==> Surely not Photon - // - No vertex Matched (and trk has "innermost pixel hits" when m_noVertexNoInnermostAsAmb is true, + // Electron ==> Surely not Photon + // - No vertex Matched (and trk has "innermost pixel hits" when + // m_noVertexNoInnermostAsAmb is true, // othewise ambiguous) - // - Track with at least the minimum Si and Pixel hits (previous selection for photons/ambiguous) + // - Track with at least the minimum Si and Pixel hits (previous selection for + // photons/ambiguous) // - And has E/P < 10 and Pt > 2.0 GeV (previous for ambiguous) // - Or if a vertex exists and: - // * is not Si+Si + // * is not Si+Si // * is Si+Si but only 1 trk has "innermost pixel hits" - // * is Si+Si and both tracks have "innermost pixel hits" but - // Rconv - RfirstHit > maxDeltaR_innermost + // * is Si+Si and both tracks have "innermost pixel hits" but + // Rconv - RfirstHit > maxDeltaR_innermost // In this case we do not want this to be in Photons if (!vx) { - if (trkHasInnermostHit || !m_noVertexNoInnermostAsAmb) { - ATH_MSG_DEBUG("Returning Electron"); - type=xAOD::AmbiguityTool::electron; - return xAOD::EgammaParameters::AuthorElectron; - } - else { - // the true photons falling here are classified as unconverted photons - // but it may happen that they are late single-track conversions - // very few true electrons are falling here, since there is no innermost hits - ATH_MSG_DEBUG("Returning Ambiguous due to no conv vertex and track with no innermost hits"); - type=xAOD::AmbiguityTool::ambiguousNoInnermost; // TODO: create new type. - // NOTE: now there is no distinction between TrackEOverPBetterTanVertexEoverP, ... - return xAOD::EgammaParameters::AuthorAmbiguous; - } + if (trkHasInnermostHit || !m_noVertexNoInnermostAsAmb) { + ATH_MSG_DEBUG("Returning Electron"); + type = xAOD::AmbiguityTool::electron; + return xAOD::EgammaParameters::AuthorElectron; + } else { + // the true photons falling here are classified as unconverted photons + // but it may happen that they are late single-track conversions + // very few true electrons are falling here, since there is no innermost + // hits + ATH_MSG_DEBUG("Returning Ambiguous due to no conv vertex and track with " + "no innermost hits"); + type = xAOD::AmbiguityTool::ambiguousNoInnermost; + return xAOD::EgammaParameters::AuthorAmbiguous; + } } // here we have a conv vertex - - if (trkHasInnermostHit && (!vxDoubleSi || nTrkVxWithInnermostHit == 1 || !passDeltaR_innermost(*vx))) { + if (trkHasInnermostHit && (!vxDoubleSi || nTrkVxWithInnermostHit == 1 || + !passDeltaR_innermost(*vx))) { ATH_MSG_DEBUG("Returning Electron"); - type=xAOD::AmbiguityTool::electron; + type = xAOD::AmbiguityTool::electron; return xAOD::EgammaParameters::AuthorElectron; } // Ambiguous all else, these will go to both electrons and photons - // A matched Si+Si vertex (where either both or none of the tracks have - // "innermost pixel hits") + // A matched Si+Si vertex (where either both or none of the tracks have + // "innermost pixel hits") // and a track with "innermost pixel hits" // A non Si+Si vertex matched and a track with no "innermost pixel hits" - //Here the types can become interesting - double track_p = fabs(1.0/tp->qOverP()); - double vertex_p=xAOD::EgammaHelpers::momentumAtVertex(*vx).mag() ; - - type = (fabs(track_p - cluster_e) < fabs(vertex_p- cluster_e)) ? - xAOD::AmbiguityTool::ambiguousTrackEoverPBetterThanVertexEoverP : + //If we have no double si + //or if the single track p matches the cluster E than the vertex p + //mark TrackEoverPBetterThanVertexEover. + //Otherwise maek as TrackEoverPBetterThanVertexEover + //as the double si vertex is a better match to the cluster E. + const double track_p = fabs(1.0/tp->qOverP()); + const double vertex_p=xAOD::EgammaHelpers::momentumAtVertex(*vx).mag() ; + type = !vxDoubleSi || (fabs(track_p - cluster_e) < fabs(vertex_p- cluster_e))? + xAOD::AmbiguityTool::ambiguousTrackEoverPBetterThanVertexEoverP : xAOD::AmbiguityTool::ambiguousVertexEoverPBetterThanTrackEoverP ; //end of types - + ATH_MSG_DEBUG("Returning Ambiguous"); return xAOD::EgammaParameters::AuthorAmbiguous; } @@ -243,41 +243,41 @@ unsigned int EGammaAmbiguityTool::ambiguityResolve(const xAOD::CaloCluster* clus /** Redo the ambiguity resolution of central electrons and photons and return - * AuthorElectron, AuthorPhoton, AuthorAmbiguous, AuthorUnknown + * AuthorElectron, AuthorPhoton, AuthorAmbiguous, AuthorUnknown * or the author of the object if no overlapping object is found **/ unsigned int EGammaAmbiguityTool::ambiguityResolve(const xAOD::Egamma& egamma) const{ - - //Fwd and Topo seeded not handled + + //Fwd and Topo seeded not handled if (egamma.author() == xAOD::EgammaParameters::AuthorFwdElectron || egamma.author() == xAOD::EgammaParameters::AuthorCaloTopo35){ ATH_MSG_DEBUG("Author Fwd of Topo seeded. Do not do overlap or ambiguity"); return egamma.author(); } - + //No overlap found so either photon or electron if (!egamma.ambiguousObject()){ //!ELink.isAvailable(*egamma)) { ATH_MSG_DEBUG("No overlaping object found"); return egamma.author(); } - - //Overlap found. define the electron and the photon - const xAOD::Electron *electron = - static_cast<const xAOD::Electron*>(egamma.type() == xAOD::Type::Electron ? &egamma : egamma.ambiguousObject()); - const xAOD::Photon *photon = - static_cast<const xAOD::Photon*>(egamma.type() == xAOD::Type::Photon ? &egamma : egamma.ambiguousObject()); - + + // Overlap found. define the electron and the photon + const xAOD::Electron* electron = static_cast<const xAOD::Electron*>( + egamma.type() == xAOD::Type::Electron ? &egamma : egamma.ambiguousObject()); + const xAOD::Photon* photon = static_cast<const xAOD::Photon*>( + egamma.type() == xAOD::Type::Photon ? &egamma : egamma.ambiguousObject()); + //Error if cannot define any of them if (!electron || !photon){ ATH_MSG_WARNING("Could not cast to electron and/or photon"); return egamma.author(); } - + //pass the proper input to the ambiguity resolve method xAOD::AmbiguityTool::AmbiguityType type; unsigned int result = ambiguityResolve(electron->caloCluster(), - photon->vertex(), + photon->vertex(), electron->trackParticle(), type); ATH_MSG_DEBUG("Performed ambiguity resolution, resulting type is: "<< type); @@ -297,11 +297,11 @@ bool EGammaAmbiguityTool::accept( const xAOD::Egamma& egamma) const{ return true; } - return (author == (egamma.type() == xAOD::Type::Electron ? + return (author == (egamma.type() == xAOD::Type::Electron ? xAOD::EgammaParameters::AuthorElectron : xAOD::EgammaParameters::AuthorPhoton) ); } -/** Return true if track has innermost pixel hit +/** Return true if track has innermost pixel hit * or next-to-innermost in case innermost is not expected * or at least m_MinNoPixHits pixel hits in case next-to-innermost is not expected **/ @@ -311,21 +311,21 @@ bool EGammaAmbiguityTool::hasInnermostPixelHit(const xAOD::TrackParticle& tp) co uint8_t trkNhits(0); CHECK_HITS( tp.summaryValue(trkNhits,xAOD::numberOfInnermostPixelLayerHits) ); if (trkNhits) {return true;} - + CHECK_HITS( tp.summaryValue(trkExpectHit,xAOD::expectInnermostPixelLayerHit) ); if (trkExpectHit) {return false;} - + CHECK_HITS( tp.summaryValue(trkNhits,xAOD::numberOfNextToInnermostPixelLayerHits) ); if (trkNhits) {return true;} - + CHECK_HITS( tp.summaryValue(trkExpectHit,xAOD::expectNextToInnermostPixelLayerHit) ); - if (trkExpectHit) {return false;} - + if (trkExpectHit) {return false;} + CHECK_HITS( tp.summaryValue(trkNhits,xAOD::numberOfPixelHits) ); return (trkNhits >= m_MinNoPixHits); } -/** Return the number of tracks with "innermost pixel hits" (see above) +/** Return the number of tracks with "innermost pixel hits" (see above) * in the given vertex **/ size_t EGammaAmbiguityTool::nTrkWithInnermostPixelHits(const xAOD::Vertex& vx) const{ size_t n = 0; @@ -354,5 +354,6 @@ bool EGammaAmbiguityTool::passDeltaR_innermost(const xAOD::Vertex& vx) const ATH_MSG_WARNING("minRfirstHit not available"); return false; } - return xAOD::EgammaHelpers::conversionRadius(&vx) - vx.auxdata<float>("minRfirstHit") < m_maxDeltaR_innermost; + static const SG::AuxElement::Accessor<float> acc("minRfirstHit"); + return xAOD::EgammaHelpers::conversionRadius(&vx) - acc(vx) < m_maxDeltaR_innermost; }