-
Sergei Zharko authoredSergei Zharko authored
CbmCaMCModule.h 19.50 KiB
/* Copyright (C) 2022-2023 GSI Helmholtzzentrum fuer Schwerionenforschung, Darmstadt
SPDX-License-Identifier: GPL-3.0-only
Authors: Sergei Zharko [committer] */
/// @file CbmCaMCModule.h
/// @brief CA Tracking performance interface for CBM (header)
/// @since 23.09.2022
/// @author S.Zharko <s.zharko@gsi.de>
#ifndef CbmCaMCModule_h
#define CbmCaMCModule_h 1
#include "CbmL1DetectorID.h"
#include "CbmL1Track.h"
#include "CbmLink.h"
#include "CbmMCDataArray.h"
#include "CbmMCEventList.h"
#include "CbmMCTrack.h"
#include "CbmMatch.h"
#include "CbmMuchPoint.h"
#include "CbmMvdPoint.h"
#include "CbmStsPoint.h"
#include "CbmTimeSlice.h"
#include "CbmTofPoint.h"
#include "CbmTrdPoint.h"
#include "TClonesArray.h"
#include "TDatabasePDG.h"
#include <numeric>
#include <string>
#include <string_view>
#include <type_traits>
#include "CaToolsMCData.h"
#include "CaToolsMCPoint.h"
#include "CaToolsQa.h"
#include "L1Constants.h"
#include "L1Parameters.h"
#include "L1Undef.h"
class CbmEvent;
class CbmMCDataObject;
class CbmL1HitDebugInfo;
class CbmL1Track;
enum class L1DetectorID;
/// Class CbmCaPerformance is an interface to communicate between
///
class CbmCaMCModule {
public:
/// @brief Constructor
/// @param verbosity Verbosity level
CbmCaMCModule(int verb = 0, int perfMode = 3) : fVerbose(verb), fPerformanceMode(perfMode) {}
/// @brief Destructor
~CbmCaMCModule() = default;
/// @brief Copy constructor
CbmCaMCModule(const CbmCaMCModule&) = delete;
/// @brief Move constructor
CbmCaMCModule(CbmCaMCModule&&) = delete;
/// @brief Copy assignment operator
CbmCaMCModule& operator=(const CbmCaMCModule&) = delete;
/// @brief Move assignment operator
CbmCaMCModule& operator=(CbmCaMCModule&&) = delete;
/// @brief Defines performance action in the end of the run
void Finish();
/// @brief Gets the first point index for a given detector subsystem
int GetFirstPointIndex(L1DetectorID detID) const
{
return std::accumulate(fvNofPointsUsed.cbegin(), fvNofPointsUsed.cbegin() + int(detID), 0);
}
/// @brief Gets the first point index for a given detector subsystem
int GetLastPointIndex(L1DetectorID detID) const
{
return std::accumulate(fvNofPointsUsed.cbegin(), fvNofPointsUsed.cbegin() + int(detID) + 1, 0) - 1;
}
/// @brief Gets MC data object
/// @return Constant reference to MC data object
const ca::tools::MCData& GetMCData() const { return fMCData; }
/// @brief Gets mutable pointer to MC data object
/// @return Mutable reference to MC data object
ca::tools::MCData* GetMCData() { return &fMCData; }
/// @brief Defines performance action in the beginning of each event or time slice
/// @note This function should be called before hits initialization
/// @param pEvent Pointer to a current CbmEvent
void InitEvent(CbmEvent* pEvent);
/// @brief Defines action on the module in the beginning of the run
/// @return Success flag
bool InitRun();
/// @brief Initializes MC track
/// @param vHits Vector of hit objects
///
/// Initializes information about arrangement of points and hits of MC tracks within stations and the status
/// of track reconstructability, calculates max number of points and hits on a station, number of consecutive
/// stations containing a hit or point and number of stations and points with hits. Provides the determination
/// of track reconstructability status.
void InitTrackInfo(const L1Vector<CbmL1HitDebugInfo>& vHits);
/// @brief Processes event
///
/// Fills histograms and tables, should be called after the tracking done
void ProcessEvent(CbmEvent* pEvent);
/// @brief Matches hit with MC point
/// @tparam DetId Detector ID
/// @param iHit External index of hit
/// @return MC-point index in fvMCPoints array
///
/// This method finds a match for a given hit or matches for hits clusters (in case of STS), finds the best
/// link in the match and returns the corresponding global ID of the MC points
template<L1DetectorID DetId>
int MatchHitWithMc(int iHit) const;
/// @brief Assigns MC point indexes to each hit and hit indexes to each MC point
/// @param vInputExtHits Vector of external input hits
void MatchPointsWithHits(L1Vector<CbmL1HitDebugInfo>& vInputExtHits);
/// @brief Matches reconstructed tracks with MC tracks
/// @param vRecoTracks Vector of reconstructed tracks
/// @param vInputExtHits Vector of external input hits
///
/// In the procedure, the maps of associated MC track indexes to corresponding number of hits are filled out and the
/// max purity is calculated for each reconstructed track in the TS/event. If the value of purity for a given MC track
/// is larger then a threshold, the corresponding track index is saved to the reconstructed track object, in the same
/// time the index of the reconstructed track object is saved to this MC track object. If purity for the MC track does
/// not pass the threshold, its index will not be accounted as a matched track, and the index of reconstructed track
/// will be added as an index of "touched" track.
void MatchRecoAndMCTracks(L1Vector<CbmL1Track>& vRecoTracks, L1Vector<CbmL1HitDebugInfo>& vInputExtHits);
/// @brief Sets used detector subsystems
/// @param detID Id of detector
/// @param flag Flag: true - detector is used
/// @note Should be called before this->Init()
void SetDetector(L1DetectorID detID, bool flag);
/// @brief Sets pointer to the vector of hits from reference
void SetHitContainer(const L1Vector<CbmL1HitDebugInfo>& vHits) { fpvHits = &vHits; }
/// @brief Sets legacy event mode:
/// @param flag Flag:
/// - true: runs on events base,
/// - false: runs on time-slice base
void SetLegacyEventMode(bool flag) { fbLegacyEventMode = flag; }
/// @brief Registers pointer to the tracking parameters object
void SetParameters(const L1Parameters* pParameters) { fpParameters = pParameters; }
/// @brief Sets pointer to the vector of reconstructed tracks from reference
void SetRecoTrackContainer(const L1Vector<CbmL1Track>& vRecoTracks) { fpvRecoTracks = &vRecoTracks; }
private:
/// @brief Calculates global index of MC point
/// @param iPointLocal Local index of MC point
/// @param detID Detector ID
///
/// Calculates global index of MC point as a sum of a given local index and total provided number of points in previous
/// detector subsystem
int CalcGlobMCPointIndex(int iPointLocal, L1DetectorID detID) const
{
return iPointLocal + std::accumulate(fvNofPointsOrig.cbegin(), fvNofPointsOrig.cbegin() + int(detID), 0);
}
/// @brief Check class initialization
/// @note The function throws std::logic_error, if initialization is incomplete
void CheckInit() const;
/// @brief Reads MC tracks from external trees and saves them to MCDataObject
void ReadMCTracks();
/// @brief Reads MC points from external trees and saves them to MCDataObject
void ReadMCPoints();
/// @brief Reads MC points in particular detector
template<L1DetectorID DetID>
void ReadMCPointsForDetector(CbmMCDataArray* pPoints);
/// @brief Fills a single detector-specific MC point
/// @tparam DetID Detector subsystem ID
/// @param[in] iExtId Index of point in the external points container
/// @param[in] iEvent EventID of point in the external points container
/// @param[in] iFile FileID of point int the external points container
/// @param[out] intMCPoint Reference to the internal tracking MC point object (ca::tools::MCData)
/// @return true Point is correct and is to be saved to the MCData object
/// @return false Point is incorrect and will be ignored
template<L1DetectorID DetID>
bool FillMCPoint(int iExtId, int iEvent, int iFile, ca::tools::MCPoint& point);
// *******************************
// ** Utility variables **
// *******************************
const L1Parameters* fpParameters = nullptr; ///< Pointer to tracking parameters object
// ** Flags **
bool fbUseMvd = false; ///< is MVD used
bool fbUseSts = false; ///< is STS used
bool fbUseMuch = false; ///< is MuCh used
bool fbUseTrd = false; ///< is TRD used
bool fbUseTof = false; ///< is TOF used
bool fbLegacyEventMode = false; ///< if tracking uses events instead of time-slices (back compatibility)
int fVerbose = 0; ///< Verbosity level
int fPerformanceMode = -1; ///< Mode of performance
// *********************************
// ** Input data branches **
// *********************************
const CbmTimeSlice* fpTimeSlice = nullptr; ///< Current time slice
// ------ Mc-event
CbmMCEventList* fpMCEventList = nullptr; ///< MC event list
CbmMCDataObject* fpMCEventHeader = nullptr; ///< MC event header
CbmMCDataArray* fpMCTracks = nullptr; ///< MC tracks input
// ------ Mc-points
CbmMCDataArray* fpMvdPoints = nullptr; ///< MVD MC-points input container
CbmMCDataArray* fpStsPoints = nullptr; ///< STS MC-points input container
CbmMCDataArray* fpMuchPoints = nullptr; ///< MuCh MC-points input container
CbmMCDataArray* fpTrdPoints = nullptr; ///< TRD MC-points input container
CbmMCDataArray* fpTofPoints = nullptr; ///< TOF MC-points input container
std::array<int, L1Constants::size::kMaxNdetectors> fvNofPointsOrig = {}; ///< Number of points by detector provided
std::array<int, L1Constants::size::kMaxNdetectors> fvNofPointsUsed = {}; ///< Number of points used in performance
// ------ Matches
TClonesArray* fpMvdHitMatches = nullptr; ///< Array of MVD hit matches
TClonesArray* fpStsHitMatches = nullptr; ///< Array of STS hit matches
TClonesArray* fpMuchHitMatches = nullptr; ///< Array of MuCh hit matches
TClonesArray* fpTrdHitMatches = nullptr; ///< Array of TRD hit matches
TClonesArray* fpTofHitMatches = nullptr; ///< Array of TOF hit matches
TClonesArray* fpStsHits = nullptr; ///< Array of STS hits (currently needed for matching)
TClonesArray* fpStsClusterMatches = nullptr; ///< Array of STS cluster matches
// ------ Matching information
std::set<std::pair<int, int>> fFileEventIDs; ///< Set of file and event indexes: first - iFile, second - iEvent
// ************************************
// ** Current MC data object **
// ************************************
ca::tools::MCData fMCData {}; ///< MC-information in CA tracking internal format (tracks and points)
std::unique_ptr<ca::tools::Qa> fpQaModule = nullptr; ///< Pointer to CA QA module
// ********************************************
// ** Pointers to reconstructed data **
// ********************************************
const L1Vector<CbmL1Track>* fpvRecoTracks = nullptr; ///< Pointer to reconstructed track container
const L1Vector<CbmL1HitDebugInfo>* fpvHits = nullptr; ///< Pointer to hits container
};
// **********************************************
// ** Template function implementation **
// **********************************************
// ---------------------------------------------------------------------------------------------------------------------
//
template<L1DetectorID DetID>
bool CbmCaMCModule::FillMCPoint(int iExtId, int iEvent, int iFile, ca::tools::MCPoint& point)
{
// ----- Get positions, momenta, time and track ID
TVector3 posIn; // Position at entrance to station [cm]
TVector3 posOut; // Position at exist of station [cm]
TVector3 momIn; // 3-momentum at entrance to station [GeV/c]
TVector3 momOut; // 3-momentum at exit of station [GeV/c]
double time = undef::kD; // Time of MC point (after beginning of the event) [ns]
int iTmcExt = undef::kI32; // Track ID in the external container
// MVD
if constexpr (L1DetectorID::kMvd == DetID) {
auto* pExtPoint = L1_DYNAMIC_CAST<CbmMvdPoint*>(fpMvdPoints->Get(iFile, iEvent, iExtId));
if (!pExtPoint) {
LOG(warn) << "CbmCaMCModule: MVD MC point with iExtId = " << iExtId << ", iEvent = " << iEvent
<< ", iFile = " << iFile << " does not exist";
return false;
}
pExtPoint->Position(posIn);
pExtPoint->PositionOut(posOut);
pExtPoint->Momentum(momIn);
pExtPoint->MomentumOut(momOut);
time = pExtPoint->GetTime();
iTmcExt = pExtPoint->GetTrackID();
}
// STS
else if constexpr (L1DetectorID::kSts == DetID) {
auto* pExtPoint = L1_DYNAMIC_CAST<CbmStsPoint*>(fpStsPoints->Get(iFile, iEvent, iExtId));
if (!pExtPoint) {
LOG(warn) << "CbmCaMCModule: STS MC point with iExtId = " << iExtId << ", iEvent = " << iEvent
<< ", iFile = " << iFile << " does not exist";
return false;
}
pExtPoint->Position(posIn);
pExtPoint->PositionOut(posOut);
pExtPoint->Momentum(momIn);
pExtPoint->MomentumOut(momOut);
time = pExtPoint->GetTime();
iTmcExt = pExtPoint->GetTrackID();
}
// MuCh
else if constexpr (L1DetectorID::kMuch == DetID) {
auto* pExtPoint = L1_DYNAMIC_CAST<CbmMuchPoint*>(fpMuchPoints->Get(iFile, iEvent, iExtId));
if (!pExtPoint) {
LOG(warn) << "CbmCaMCModule: MuCh MC point with iExtId = " << iExtId << ", iEvent = " << iEvent
<< ", iFile = " << iFile << " does not exist";
return false;
}
pExtPoint->Position(posIn);
pExtPoint->PositionOut(posOut);
pExtPoint->Momentum(momIn);
pExtPoint->Momentum(momOut);
time = pExtPoint->GetTime();
iTmcExt = pExtPoint->GetTrackID();
}
// TRD
else if constexpr (L1DetectorID::kTrd == DetID) {
auto* pExtPoint = L1_DYNAMIC_CAST<CbmTrdPoint*>(fpTrdPoints->Get(iFile, iEvent, iExtId));
if (!pExtPoint) {
LOG(warn) << "CbmCaMCModule: TRD MC point with iExtId = " << iExtId << ", iEvent = " << iEvent
<< ", iFile = " << iFile << " does not exist";
return false;
}
pExtPoint->Position(posIn);
pExtPoint->PositionOut(posOut);
pExtPoint->Momentum(momIn);
pExtPoint->MomentumOut(momOut);
time = pExtPoint->GetTime();
iTmcExt = pExtPoint->GetTrackID();
}
// TOF
else if constexpr (L1DetectorID::kTof == DetID) {
auto* pExtPoint = L1_DYNAMIC_CAST<CbmTofPoint*>(fpTofPoints->Get(iFile, iEvent, iExtId));
if (!pExtPoint) {
LOG(warn) << "CbmCaMCModule: TOF MC point with iExtId = " << iExtId << ", iEvent = " << iEvent
<< ", iFile = " << iFile << " does not exist";
return false;
}
pExtPoint->Position(posIn);
pExtPoint->Position(posOut);
pExtPoint->Momentum(momIn);
pExtPoint->Momentum(momOut);
time = pExtPoint->GetTime();
iTmcExt = pExtPoint->GetTrackID();
}
if (iTmcExt < 0) {
LOG(warn) << "CbmCaMCModule: For MC point with iExtId = " << iExtId << ", iEvent = " << iEvent
<< ", iFile = " << iFile << " MC track is undefined (has ID = " << iTmcExt << ')';
return false;
}
TVector3 posMid = 0.5 * (posIn + posOut);
TVector3 momMid = 0.5 * (momIn + momOut);
// // ----- Get station index
int stationID = undef::kI32; // Index of station in the array of active tracking stations
int nStationsGeo = fpParameters->GetNstationsGeometry(DetID);
// MVD, STS TODO: Try to extend for MuCh and TRD
if constexpr (L1DetectorID::kMvd == DetID || L1DetectorID::kSts == DetID) {
// Determine active station global index
double bestDist = 1.e+20;
for (int iStLocGeo = 0; iStLocGeo < nStationsGeo; ++iStLocGeo) {
int iStActive = fpParameters->GetStationIndexActive(iStLocGeo, DetID);
if (iStActive < 0) { continue; }
const L1Station& station = fpParameters->GetStation(iStActive);
double dist = posIn.Z() - station.fZ[0];
if (fabs(dist) < fabs(bestDist)) {
bestDist = dist;
stationID = iStActive;
}
}
}
// MuCh, TRD
else if constexpr (L1DetectorID::kMuch == DetID || L1DetectorID::kTrd == DetID) {
// Offset for MuCh - 2.5 cm, for TRD - 20. cm. TODO: Where did these values came from?
double offset = (L1DetectorID::kMuch == DetID) ? 2.5 : 20.;
for (int iStLocGeo = 0; iStLocGeo < nStationsGeo; ++iStLocGeo) {
int iStActive = fpParameters->GetStationIndexActive(iStLocGeo, DetID);
if (iStActive < 0) { continue; }
const L1Station& station = fpParameters->GetStation(iStActive);
if (posMid.Z() > station.fZ[0] - offset) { stationID = iStActive; }
}
}
// ----- Reject MC points falling out of the time slice
// STS, MuCh, TRD, TOF
if constexpr (DetID != L1DetectorID::kMvd) {
if (!fbLegacyEventMode) { // fpTimeSlice != nullptr
double startT = fpTimeSlice->GetStartTime();
double endT = fpTimeSlice->GetEndTime();
double mcGlobT = time + fpMCEventList->GetEventTime(iEvent, iFile);
if ((startT > 0. && mcGlobT < startT) || (endT > 0. && mcGlobT > endT)) {
LOG(warn) << "CbmCaMCModule: MC point with iExtId = " << iExtId << ", iEvent = " << iEvent
<< ", iFile = " << iFile << " and det id " << int(DetID) << " fell out of the TS duration [" << startT
<< ", " << endT << "] with measured time = " << mcGlobT << " [ns]";
return false;
}
}
}
// ----- Fill MC point
point.SetEventId(iEvent);
point.SetFileId(iFile);
point.SetTime(time);
point.SetX(posMid.X());
point.SetY(posMid.Y());
point.SetZ(posMid.Z());
point.SetXIn(posIn.X());
point.SetYIn(posIn.Y());
point.SetZIn(posIn.Z());
point.SetXOut(posOut.X());
point.SetYOut(posOut.Y());
point.SetZOut(posOut.Z());
point.SetPx(momMid.X());
point.SetPy(momMid.Y());
point.SetPz(momMid.Z());
point.SetPxIn(momIn.X());
point.SetPyIn(momIn.Y());
point.SetPzIn(momIn.Z());
point.SetPxOut(momOut.X());
point.SetPyOut(momOut.Y());
point.SetPzOut(momOut.Z());
int iTmcInt = fMCData.FindInternalTrackIndex(iTmcExt, iEvent, iFile);
point.SetId(fMCData.GetNofPoints()); // select current number of points as a local id of points
point.SetTrackId(iTmcInt);
point.SetStationId(stationID);
point.SetDetectorId(DetID);
auto* pExtTrk = L1_DYNAMIC_CAST<CbmMCTrack*>(fpMCTracks->Get(iFile, iEvent, iTmcExt));
if (!pExtTrk) {
LOG(warn) << "CbmCaMCModule: MC track with iTmcExt = " << iTmcExt << ", iEvent = " << iEvent
<< ", iFile = " << iFile << " MC track is undefined (nullptr)";
return false;
}
point.SetPdgCode(pExtTrk->GetPdgCode());
point.SetMotherId(pExtTrk->GetMotherId());
auto* pPdgDB = TDatabasePDG::Instance()->GetParticle(point.GetPdgCode());
point.SetMass(pPdgDB ? pPdgDB->Mass() : 0.); /// TODO: Set from track
point.SetCharge(pPdgDB ? pPdgDB->Charge() / 3. : 0.);
return true;
}
// ---------------------------------------------------------------------------------------------------------------------
// NOTE: template is used, because another template function FillMCPoint is used inside
template<L1DetectorID DetID>
void CbmCaMCModule::ReadMCPointsForDetector(CbmMCDataArray* pPoints)
{
if constexpr (L1DetectorID::kTof != DetID) {
fvNofPointsUsed[int(DetID)] = 0;
for (const auto& key : fFileEventIDs) {
int iFile = key.first;
int iEvent = key.second;
int nPointsEvent = pPoints->Size(iFile, iEvent);
for (int iP = 0; iP < nPointsEvent; ++iP) {
ca::tools::MCPoint aPoint;
if (FillMCPoint<DetID>(iP, iEvent, iFile, aPoint)) {
aPoint.SetExternalId(CalcGlobMCPointIndex(iP, DetID));
int iPInt = fMCData.GetNofPoints(); // assign an index of point in internal container
if (aPoint.GetTrackId() > -1) { fMCData.GetTrack(aPoint.GetTrackId()).AddPointIndex(iPInt); }
fMCData.AddPoint(aPoint);
++fvNofPointsUsed[int(DetID)];
}
} // iP: end
} // key: end
}
}
#endif // CbmCaMCModule_h