From bf22f8d4df48342fd62b7a5705d9f39259d242ec Mon Sep 17 00:00:00 2001 From: Joachim Schoeberl Date: Sat, 30 Apr 2022 10:39:26 +0200 Subject: [PATCH] fix shared loop: in rare cases elements have been called twice --- libsrc/core/taskmanager.hpp | 49 ++++++++++++++++++++++++++++--------- 1 file changed, 37 insertions(+), 12 deletions(-) diff --git a/libsrc/core/taskmanager.hpp b/libsrc/core/taskmanager.hpp index 8d2886b1..3f888fc0 100644 --- a/libsrc/core/taskmanager.hpp +++ b/libsrc/core/taskmanager.hpp @@ -464,23 +464,26 @@ public: bool PopFirst (size_t & first) { - first = begin++; - return first < end; - /* - // int oldbegin = begin; - size_t oldbegin = begin.load(std::memory_order_acquire); - if (oldbegin >= end) return false; - while (!begin.compare_exchange_weak (oldbegin, oldbegin+1, - std::memory_order_relaxed, std::memory_order_relaxed)) - if (oldbegin >= end) return false; + // first = begin++; + // return first < end; - first = oldbegin; - return true; - */ + first = begin; + + size_t nextfirst = first+1; + if (first >= end) nextfirst = std::numeric_limits::max()-1; + + while (!begin.compare_exchange_weak (first, nextfirst)) + { + first = begin; + nextfirst = first+1; + if (nextfirst >= end) nextfirst = std::numeric_limits::max()-1; + } + return first < end; } bool PopHalf (IntRange & r) { + /* // int oldbegin = begin; size_t oldbegin = begin.load(std::memory_order_acquire); size_t oldend = end.load(std::memory_order_acquire); @@ -496,6 +499,28 @@ public: r = IntRange(oldbegin, (oldbegin+oldend+1)/2); return true; + */ + + + size_t oldbegin = begin; // .load(std::memory_order_acquire); + size_t oldend = end; // .load(std::memory_order_acquire); + if (oldbegin >= oldend) return false; + + size_t nextbegin = (oldbegin+oldend+1)/2; + if (nextbegin >= oldend) nextbegin = std::numeric_limits::max()-1; + + while (!begin.compare_exchange_weak (oldbegin, nextbegin)) + // std::memory_order_relaxed, std::memory_order_relaxed)) + { + oldend = end; // .load(std::memory_order_acquire); + if (oldbegin >= oldend) return false; + + nextbegin = (oldbegin+oldend+1)/2; + if (nextbegin >= oldend) nextbegin = std::numeric_limits::max()-1; + } + + r = IntRange(oldbegin, (oldbegin+oldend+1)/2); + return true; } };