From ee66656aa56302469ec2109aada5f4eef85b987c Mon Sep 17 00:00:00 2001 From: Matti Kortelainen Date: Mon, 12 Jun 2023 18:46:41 +0200 Subject: [PATCH] Protect assignments of StripDetset() StMeasurementDetSet::getDetSet() The assigments are a critical section and must be protected. --- .../MeasurementDet/src/TkMeasurementDetSet.h | 41 +++++++++++-------- 1 file changed, 25 insertions(+), 16 deletions(-) diff --git a/RecoTracker/MeasurementDet/src/TkMeasurementDetSet.h b/RecoTracker/MeasurementDet/src/TkMeasurementDetSet.h index f9dabef9b6c82..cf6882dc09b18 100644 --- a/RecoTracker/MeasurementDet/src/TkMeasurementDetSet.h +++ b/RecoTracker/MeasurementDet/src/TkMeasurementDetSet.h @@ -165,7 +165,7 @@ class StMeasurementDetSet { void update(int i, int j) { assert(j >= 0); assert(detSet_[i].empty_); - assert(detSet_[i].ready_); + assert(detSet_[i].ready_ == ReadyState::kUnset); detIndex_[i] = j; detSet_[i].empty_ = false; incReady(); @@ -176,7 +176,7 @@ class StMeasurementDetSet { unsigned int id(int i) const { return conditions().id(i); } int find(unsigned int jd, int i = 0) const { return conditions().find(jd, i); } - bool empty(int i) const { return detSet_[i].empty_; } + bool empty(int i) const { return detSet_[i].empty_.load(std::memory_order_relaxed); } bool isActive(int i) const { return activeThisEvent_[i] && conditions().isActiveThisPeriod(i); } void setEmpty(int i) { @@ -189,7 +189,7 @@ class StMeasurementDetSet { printStat(); for (auto& d : detSet_) { d.empty_ = true; - d.ready_ = true; + d.ready_ = ReadyState::kUnset; } std::fill(detIndex_.begin(), detIndex_.end(), -1); std::fill(activeThisEvent_.begin(), activeThisEvent_.end(), true); @@ -208,7 +208,7 @@ class StMeasurementDetSet { const edm::Handle>& handle() const { return handle_; } // StripDetset & detSet(int i) { return detSet_[i]; } const StripDetset& detSet(int i) const { - if (detSet_[i].ready_) + if (detSet_[i].ready_.load(std::memory_order_acquire) != ReadyState::kSet) getDetSet(i); return detSet_[i].detSet_; } @@ -232,16 +232,24 @@ class StMeasurementDetSet { private: void getDetSet(int i) const { const auto& det = detSet_[i]; - if (detIndex_[i] >= 0) { - det.detSet_ = StripDetset(*handle_, handle_->item(detIndex_[i]), true); - det.empty_ = false; // better be false already - incAct(); - } else { // we should not be here - det.detSet_ = StripDetset(); - det.empty_ = true; + + ReadyState expected = ReadyState::kUnset; + if (det.ready_.compare_exchange_strong(expected, ReadyState::kSetting, std::memory_order_acq_rel)) { + if (detIndex_[i] >= 0) { + det.detSet_ = StripDetset(*handle_, handle_->item(detIndex_[i]), true); + det.empty_.store(false, std::memory_order_relaxed); // better be false already + incAct(); + } else { // we should not be here + det.detSet_ = StripDetset(); + det.empty_.store(true, std::memory_order_relaxed); + } + det.ready_.store(ReadyState::kSet, std::memory_order_release); + incSet(); + } else { + // need to wait + while (ReadyState::kSet != det.ready_.load(std::memory_order_acquire)) { + } } - det.ready_ = false; - incSet(); } friend class MeasurementTrackerImpl; @@ -251,14 +259,15 @@ class StMeasurementDetSet { // Globals, per-event edm::Handle> handle_; + enum class ReadyState : char { kUnset, kSetting, kSet }; + // Helper struct to define only the vector elements as mutable and // to have a vector of atomics without an explicit loop over // elements to set their values struct DetSetHelper { mutable std::atomic empty_ = true; - mutable std::atomic ready_ = true; // to be cleaned - // only thread-safe non-const member functions are called from a const function - CMS_THREAD_SAFE mutable StripDetset detSet_; + mutable std::atomic ready_ = ReadyState::kUnset; // to be cleaned + CMS_THREAD_GUARD(ready_) mutable StripDetset detSet_; }; std::vector activeThisEvent_;