From 568c02c0e75e6d6e3cfb10a14ed321816719695a Mon Sep 17 00:00:00 2001 From: Martin Braun Date: Mon, 20 Dec 2021 15:17:28 +0100 Subject: lib: Make simple_claimer atomic This fixes multiple issues: - The simple_claimer was not truly atomic, it tested and set the locked-flag on separate lines - It used a Boost yield statement, although we're not running Boost threads - It used the deprecated UHD_INLINE macro We also remove spin_wait_with_timeout(), which was only used in claim_with_wait() because it's not worth putting into its own function. This is no API change on simple_claimer, but it may result in different performance of this spinlock. --- host/lib/include/uhdlib/utils/atomic.hpp | 57 +++++++++++--------------------- 1 file changed, 19 insertions(+), 38 deletions(-) (limited to 'host/lib/include/uhdlib/utils/atomic.hpp') diff --git a/host/lib/include/uhdlib/utils/atomic.hpp b/host/lib/include/uhdlib/utils/atomic.hpp index 33bc78fed..0db828ee9 100644 --- a/host/lib/include/uhdlib/utils/atomic.hpp +++ b/host/lib/include/uhdlib/utils/atomic.hpp @@ -9,67 +9,48 @@ #include #include -#include #include #include namespace uhd { -/*! DEPRECATED -- Will be removed in coming versions of UHD. - * - * Spin-wait on a condition with a timeout. - * \param cond an atomic variable to compare - * \param value compare to atomic for true/false - * \param timeout the timeout in seconds - * \return true for cond == value, false for timeout - */ -template -UHD_INLINE bool spin_wait_with_timeout( - std::atomic& cond, const T value, const double timeout) -{ - if (cond == value) - return true; - const auto exit_time = std::chrono::high_resolution_clock::now() - + std::chrono::microseconds(int64_t(timeout * 1e6)); - while (cond != value) { - if (std::chrono::high_resolution_clock::now() > exit_time) { - return false; - } - boost::this_thread::interruption_point(); - boost::this_thread::yield(); - } - return true; -} - -/*! DEPRECATED -- Will be removed in coming versions of UHD. +/*! DEPRECATED -- May be removed in coming versions of UHD. * * Claimer class to provide synchronization for multi-thread access. - * Claiming enables buffer classes to be used with a buffer queue. + * + * Implements a spinlock mutex with a timeout. */ class simple_claimer { public: simple_claimer(void) { - this->release(); + _locked.clear(); } - UHD_INLINE void release(void) + inline void release(void) { - _locked = false; + _locked.clear(std::memory_order_release); } - UHD_INLINE bool claim_with_wait(const double timeout) + //! + // \param timeout Timeout in seconds + inline bool claim_with_wait(const double timeout) { - if (spin_wait_with_timeout(_locked, false, timeout)) { - _locked = true; - return true; + const auto exit_time = std::chrono::high_resolution_clock::now() + + std::chrono::microseconds(int64_t(timeout * 1e6)); + // Try acquiring lock until successful or timeout + while (_locked.test_and_set(std::memory_order_acquire)) { + if (std::chrono::high_resolution_clock::now() > exit_time) { + return false; + } + std::this_thread::yield(); } - return false; + return true; } private: - std::atomic _locked; + std::atomic_flag _locked; }; } // namespace uhd -- cgit v1.2.3