Skip to content

Commit

Permalink
Protect assignments of StripDetset() StMeasurementDetSet::getDetSet()
Browse files Browse the repository at this point in the history
The assigments are a critical section and must be protected.
  • Loading branch information
makortel committed Jun 21, 2023
1 parent 1b7e3bb commit 0bc879b
Showing 1 changed file with 25 additions and 16 deletions.
41 changes: 25 additions & 16 deletions RecoTracker/MeasurementDet/src/TkMeasurementDetSet.h
Original file line number Diff line number Diff line change
Expand Up @@ -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();
Expand All @@ -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) {
Expand All @@ -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);
Expand All @@ -208,7 +208,7 @@ class StMeasurementDetSet {
const edm::Handle<edmNew::DetSetVector<SiStripCluster>>& 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_;
}
Expand All @@ -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;
Expand All @@ -251,14 +259,15 @@ class StMeasurementDetSet {
// Globals, per-event
edm::Handle<edmNew::DetSetVector<SiStripCluster>> 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<bool> empty_ = true;
mutable std::atomic<bool> 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<ReadyState> ready_ = ReadyState::kUnset; // to be cleaned
CMS_THREAD_GUARD(ready_) mutable StripDetset detSet_;
};

std::vector<bool> activeThisEvent_;
Expand Down

0 comments on commit 0bc879b

Please sign in to comment.