12#include "Identifier/Identifier.h"
22#include "CLHEP/Geometry/Point3D.h"
26#define AUXDATA(OBJ, TYP, NAME) \
27 static const SG::AuxElement::Accessor<TYP> acc_##NAME (#NAME); acc_##NAME(*(OBJ))
65 return StatusCode::SUCCESS;
76 std::map<Identifier, const SCT_RDORawData*> idToRAWDataMap;
81 for (
const auto collection: *rdoContainer) {
83 for (
const auto rdo : *collection) {
89 idToRAWDataMap.insert(std::pair<Identifier, const SCT_RDORawData*>{rdoId, rdo});
96 ATH_MSG_DEBUG(
"Size of RDO map is " << idToRAWDataMap.size());
102 if (prdmtCollHandle.
isValid()) {
103 prdmtColl = &*prdmtCollHandle;
107 if (truthParticleLinksHandle.
isValid()) {
108 truth_particle_links = truthParticleLinksHandle.
cptr();
116 if (sdoCollectionHandle.
isValid()) {
117 sdoCollection = &*sdoCollectionHandle;
121 std::vector<std::vector<const SiHit*>> siHits(
m_SCTHelper->wafer_hash_max());
124 if (sihitCollection.
isValid()) {
125 for (
const SiHit& siHit: *sihitCollection) {
127 if (not siHit.isSCT())
continue;
130 siHit.getLayerDisk(),
131 siHit.getPhiModule(),
132 siHit.getEtaModule(),
135 siHits[wafer_hash].push_back(&siHit);
142 if (not sctClusterContainer.
isValid()) {
144 return StatusCode::FAILURE;
149 ATH_CHECK(xaod.
record(std::make_unique<xAOD::TrackMeasurementValidationContainer>(),
150 std::make_unique<xAOD::TrackMeasurementValidationAuxContainer>()));
155 unsigned int have_truth_link=0u;
156 unsigned int missing_truth_particle=0u;
157 unsigned int missing_parent_particle=0u;
159 unsigned int counter{0};
160 for (
const auto clusterCollection: *sctClusterContainer) {
162 (*offsets)[clusterCollection->identifyHash()] = counter;
165 if (clusterCollection->empty())
continue;
167 xaod->resize(counter + clusterCollection->size());
177 xaod->at(counter) = xprd;
190 float locX{
static_cast<float>(locpos.x())};
191 if ((not std::isinf(locpos.y()) or std::isnan(locpos.y()))) {
192 if (locpos.y()>=1e-07) locY = locpos.y();
201 if (localCov.size() == 1) {
203 }
else if (localCov.size() == 4) {
210 std::vector<uint64_t> rdoIdentifierList;
211 rdoIdentifierList.reserve(prd->rdoList().size());
212 for (
const auto& hitIdentifier: prd->rdoList()) {
213 rdoIdentifierList.push_back(hitIdentifier.get_compact());
219 AUXDATA(xprd,
int, SiWidth) =
static_cast<int>(cw.
colRow()[0]);
220 AUXDATA(xprd,
int, hitsInThirdTimeBin) =
static_cast<int>(prd->hitsInThirdTimeBin());
230 uint64_t detElementId{0};
237 AUXDATA(xprd, uint64_t, detectorElementID) = detElementId;
247 auto range{prdmtColl->equal_range(clusterId)};
248 if (truth_particle_links) {
249 std::vector<unsigned int> tp_indices;
250 for (
auto i{range.first}; i!=range.second; ++i) {
252 if (a_truth_particle_link) {
254 if (truth_particle) {
256 tp_indices.push_back(
static_cast<int>(truth_particle->
index()));
259 ++missing_parent_particle;
263 tp_indices.push_back(std::numeric_limits<unsigned int>::max());
264 ++missing_truth_particle;
268 AUXDATA(xprd, std::vector<unsigned int>, truth_index) = tp_indices;
270 std::vector<int> uniqueIDs;
271 for (
auto& i{range.first}; i!=range.second; ++i) {
275 AUXDATA(xprd, std::vector<int>, truth_barcode) = uniqueIDs;
294 ATH_MSG_DEBUG(
" recorded SCT_PrepData objects: size " << xaod->size());
301 return StatusCode::SUCCESS;
308 std::vector<int> sdo_word;
309 std::vector<std::vector<int>> sdo_depositsUniqueID;
310 std::vector<std::vector<float>> sdo_depositsEnergy;
312 for (
const auto& hitIdentifier: prd->
rdoList()) {
313 auto pos{sdoCollection->find(hitIdentifier)};
314 if (pos == sdoCollection->end())
continue;
315 sdo_word.push_back(pos->second.word());
317 std::vector<float> sdoDepEnergy(pos->second.getdeposits().size());
318 unsigned int nDepos{0};
319 for (
auto& deposit: pos->second.getdeposits()) {
320 if (deposit.first) sdoDepUniqueID[nDepos] =
HepMC::uniqueID(deposit.first);
322 sdoDepEnergy[nDepos] = deposit.second;
325 sdo_depositsUniqueID.push_back(sdoDepUniqueID);
326 sdo_depositsEnergy.push_back(sdoDepEnergy);
328 AUXDATA(xprd, std::vector<int>, sdo_words) = sdo_word;
329 AUXDATA(xprd, std::vector<std::vector<int>>, sdo_depositsBarcode) = sdo_depositsUniqueID;
330 AUXDATA(xprd, std::vector<std::vector<float>>, sdo_depositsEnergy) = sdo_depositsEnergy;
336 const std::vector<const SiHit*>* siHits)
const
338 std::vector<SiHit> matchingHits;
341 long unsigned int numHits{matchingHits.size()};
343 std::vector<float> sihit_energyDeposit(numHits, 0.);
344 std::vector<float> sihit_meanTime(numHits, 0.);
347 std::vector<float> sihit_startPosX(numHits, 0.);
348 std::vector<float> sihit_startPosY(numHits, 0.);
349 std::vector<float> sihit_startPosZ(numHits, 0.);
351 std::vector<float> sihit_endPosX(numHits, 0);
352 std::vector<float> sihit_endPosY(numHits, 0);
353 std::vector<float> sihit_endPosZ(numHits, 0);
358 for (
const SiHit& sihit : matchingHits) {
359 sihit_energyDeposit[hitNumber] = sihit.energyLoss();
360 sihit_meanTime[hitNumber] = sihit.meanTime();
364 const HepGeom::Point3D<double>& startPos{sihit.localStartPosition()};
367 sihit_startPosX[hitNumber] = pos[0];
368 sihit_startPosY[hitNumber] = pos[1];
369 sihit_startPosZ[hitNumber] = startPos.x();
371 const HepGeom::Point3D<double>& endPos{sihit.localEndPosition()};
373 sihit_endPosX[hitNumber] = pos[0];
374 sihit_endPosY[hitNumber] = pos[1];
375 sihit_endPosZ[hitNumber] = endPos.x();
380 AUXDATA(xprd, std::vector<float>, sihit_energyDeposit) = sihit_energyDeposit;
381 AUXDATA(xprd, std::vector<float>, sihit_meanTime) = sihit_meanTime;
382 AUXDATA(xprd, std::vector<int>, sihit_barcode) = sihit_uniqueID;
384 AUXDATA(xprd, std::vector<float>, sihit_startPosX) = sihit_startPosX;
385 AUXDATA(xprd, std::vector<float>, sihit_startPosY) = sihit_startPosY;
386 AUXDATA(xprd, std::vector<float>, sihit_startPosZ) = sihit_startPosZ;
388 AUXDATA(xprd, std::vector<float>, sihit_endPosX) = sihit_endPosX;
389 AUXDATA(xprd, std::vector<float>, sihit_endPosY) = sihit_endPosY;
390 AUXDATA(xprd, std::vector<float>, sihit_endPosZ) = sihit_endPosZ;
394 const std::vector<const SiHit*>* siHits,
395 std::vector<SiHit>& matchingHits)
const
397 ATH_MSG_VERBOSE(
"Got " << siHits->size() <<
" SiHits to look through");
401 if (de==
nullptr)
return;
403 std::vector<const SiHit*> multiMatchingHits;
405 for (
const SiHit* siHit: *siHits) {
409 HepGeom::Point3D<double> averagePosition{siHit->localStartPosition() + siHit->localEndPosition()};
410 averagePosition *= 0.5;
414 for (
const auto& hitIdentifier: prd->
rdoList()) {
418 multiMatchingHits.push_back(siHit);
424 matchingHits.reserve(multiMatchingHits.size());
426 std::vector<const SiHit*>::iterator siHitIter{multiMatchingHits.begin()};
427 std::vector<const SiHit*>::iterator siHitIter2{multiMatchingHits.begin()};
428 ATH_MSG_DEBUG(
"Found " << multiMatchingHits.size() <<
" SiHit ");
429 for (; siHitIter != multiMatchingHits.end(); ++siHitIter) {
430 const SiHit* lowestXPos{*siHitIter};
431 const SiHit* highestXPos{*siHitIter};
434 std::vector<const SiHit*> ajoiningHits;
435 ajoiningHits.push_back(*siHitIter);
437 siHitIter2 = siHitIter+1;
439 while (siHitIter2 != multiMatchingHits.end()) {
446 constexpr double maxDiff = 0.00005;
448 if (std::abs((highestXPos->
localEndPosition().x()-(*siHitIter2)->localStartPosition().x()))<maxDiff and
449 std::abs((highestXPos->
localEndPosition().y()-(*siHitIter2)->localStartPosition().y()))<maxDiff and
450 std::abs((highestXPos->
localEndPosition().z()-(*siHitIter2)->localStartPosition().z()))<maxDiff) {
451 highestXPos = *siHitIter2;
452 ajoiningHits.push_back(*siHitIter2);
454 siHitIter2 = multiMatchingHits.erase(siHitIter2);
455 }
else if (std::abs((lowestXPos->
localStartPosition().x()-(*siHitIter2)->localEndPosition().x()))<maxDiff and
456 std::abs((lowestXPos->
localStartPosition().y()-(*siHitIter2)->localEndPosition().y()))<maxDiff and
457 std::abs((lowestXPos->
localStartPosition().z()-(*siHitIter2)->localEndPosition().z()))<maxDiff) {
458 lowestXPos = *siHitIter2;
459 ajoiningHits.push_back(*siHitIter2);
461 siHitIter2 = multiMatchingHits.erase(siHitIter2);
467 if (ajoiningHits.size()==0) {
470 }
else if (ajoiningHits.size()==1) {
472 matchingHits.emplace_back(*ajoiningHits[0]);
476 ATH_MSG_DEBUG(
"Merging " << ajoiningHits.size() <<
" SiHits together.");
479 for (
auto& siHit: ajoiningHits) {
480 energyDep += siHit->energyLoss();
481 time += siHit->meanTime();
483 time /=
static_cast<float>(ajoiningHits.size());
488 (*siHitIter)->particleLink(),
490 (*siHitIter)->getBarrelEndcap(),
491 (*siHitIter)->getLayerDisk(),
492 (*siHitIter)->getEtaModule(),
493 (*siHitIter)->getPhiModule(),
494 (*siHitIter)->getSide());
501 const std::map<Identifier, const SCT_RDORawData*>& idToRAWDataMap)
const {
503 std::vector<int> timebin(prd->
rdoList().size(), -1);
504 std::vector<int> groupsize(prd->
rdoList().size(), -1);
506 unsigned int nRDOs{0};
507 for (
const auto& hitIdentifier: prd->
rdoList()) {
508 auto result{idToRAWDataMap.find(hitIdentifier)};
509 if (
result != idToRAWDataMap.end()) {
522 AUXDATA(xprd, std::vector<int>, rdo_timebin) = timebin;
523 AUXDATA(xprd, std::vector<int>, rdo_groupsize) = groupsize;
537 return StatusCode::SUCCESS;
#define ATH_CHECK
Evaluate an expression and check for errors.
#define ATH_MSG_VERBOSE(x)
#define ATH_MSG_WARNING(x)
#define AUXDATA(OBJ, TYP, NAME)
This is an Identifier helper class for the SCT subdetector.
Handle class for reading from StoreGate.
Handle class for recording to StoreGate.
const ServiceHandle< StoreGateSvc > & detStore() const
An algorithm that can be simultaneously executed in multiple threads.
ElementLink implementation for ROOT usage.
This is a "hash" representation of an Identifier.
bool is_valid() const
Check if id is in a valid state.
value_type get_compact() const
Get the compact id.
Identifier for the strip or pixel cell.
int phiIndex() const
Get phi index. Equivalent to strip().
Class to hold geometrical description of a silicon detector element.
SiCellId cellIdOfPosition(const Amg::Vector2D &localPos) const
As in previous method but returns SiCellId.
virtual Identifier identify() const override final
identifier of this detector element (inline)
Amg::Vector2D hitLocalToLocal(double xEta, double xPhi) const
Simulation/Hit local frame to reconstruction local frame.
virtual Identifier identify() const override final
virtual const InDetDD::SiDetectorElement * detectorElement() const override final
return the detector element corresponding to this PRD The pointer will be zero if the det el is not d...
const Amg::Vector2D & colRow() const
A PRD is mapped onto all contributing particles.
virtual int getGroupSize() const override final
void addSiHitInformation(xAOD::TrackMeasurementValidation *xprd, const InDet::SCT_Cluster *prd, const std::vector< const SiHit * > *siHits) const
std::atomic< unsigned int > m_missingParentParticle
void addSDOInformation(xAOD::TrackMeasurementValidation *xprd, const InDet::SCT_Cluster *prd, const InDetSimDataCollection *sdoCollection) const
SG::ReadHandleKey< SiHitCollection > m_sihitContainer
SG::ReadHandleKey< PRD_MultiTruthCollection > m_multiTruth
SG::ReadHandleKey< SCT_RDO_Container > m_rdoContainer
BooleanProperty m_writeSDOs
SG::ReadHandleKey< InDetSimDataCollection > m_SDOcontainer
SG::ReadCondHandleKey< InDetDD::SiDetectorElementCollection > m_SCTDetEleCollKey
virtual StatusCode execute(const EventContext &ctx) const override
std::atomic_bool m_firstEventWarnings
BooleanProperty m_writeSiHits
SG::WriteHandleKey< xAOD::TrackMeasurementValidationContainer > m_xAodContainer
virtual StatusCode finalize() override
SG::WriteHandleKey< std::vector< unsigned int > > m_xAodOffset
SCT_PrepDataToxAOD(const std::string &name, ISvcLocator *pSvcLocator)
BooleanProperty m_writeRDOinformation
const SCT_ID * m_SCTHelper
SG::ReadHandleKey< xAODTruthParticleLinkVector > m_truthParticleLinks
std::atomic< unsigned int > m_missingTruthParticle
void findAllHitsCompatibleWithCluster(const InDet::SCT_Cluster *prd, const std::vector< const SiHit * > *siHits, std::vector< SiHit > &matchingHits) const
SG::ReadHandleKey< InDet::SCT_ClusterContainer > m_clustercontainer
void addRDOInformation(xAOD::TrackMeasurementValidation *, const InDet::SCT_Cluster *, const std::map< Identifier, const SCT_RDORawData * > &idToRAWDataMap) const
std::atomic< unsigned int > m_haveTruthLink
BooleanProperty m_useTruthInfo
virtual StatusCode initialize() override
size_t index() const
Return the index of this element within its container.
virtual bool isValid() override final
Can the handle be successfully dereferenced?
const_pointer_type cptr()
Dereference the pointer.
StatusCode record(std::unique_ptr< T > data)
Record a const object to the store.
HepGeom::Point3D< double > localStartPosition() const
HepGeom::Point3D< double > localEndPosition() const
const std::vector< Identifier > & rdoList() const
return the List of rdo identifiers (pointers)
ElementLink< xAOD::TruthParticleContainer > find(const HepMcParticleLink &hepMCLink) const
void setRdoIdentifierList(const std::vector< uint64_t > &rdoIdentifierList)
Sets the list of RDO identifiers.
void setLocalPositionError(float localXError, float localYError, float localXYCorrelation)
Sets the local position error.
void setLocalPosition(float localX, float localY)
Sets the local position.
void setIdentifier(uint64_t identifier)
Sets the identifier.
void setGlobalPosition(float globalX, float globalY, float globalZ)
Sets the global position.
Eigen::Matrix< double, Eigen::Dynamic, Eigen::Dynamic > MatrixX
Dynamic Matrix - dynamic allocation.
Eigen::Matrix< double, 2, 1 > Vector2D
Eigen::Matrix< double, 3, 1 > Vector3D
constexpr int INVALID_PARTICLE_ID
constexpr int UNDEFINED_ID
TrackMeasurementValidation_v1 TrackMeasurementValidation
Reference the current persistent version:
TruthParticle_v1 TruthParticle
Typedef to implementation.