aboutsummaryrefslogtreecommitdiffstats
path: root/host/lib/include/uhdlib/utils/atomic.hpp
diff options
context:
space:
mode:
authorMartin Braun <martin.braun@ettus.com>2021-12-20 15:17:28 +0100
committerAaron Rossetto <aaron.rossetto@ni.com>2022-01-14 14:45:01 -0600
commit568c02c0e75e6d6e3cfb10a14ed321816719695a (patch)
treea8615bdf1b97b973d8f1f59ed1ec49af390e26b3 /host/lib/include/uhdlib/utils/atomic.hpp
parent828f00012c9fcddc7ae422fdda29679bdde852ff (diff)
downloaduhd-568c02c0e75e6d6e3cfb10a14ed321816719695a.tar.gz
uhd-568c02c0e75e6d6e3cfb10a14ed321816719695a.tar.bz2
uhd-568c02c0e75e6d6e3cfb10a14ed321816719695a.zip
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.
Diffstat (limited to 'host/lib/include/uhdlib/utils/atomic.hpp')
-rw-r--r--host/lib/include/uhdlib/utils/atomic.hpp57
1 files changed, 19 insertions, 38 deletions
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 <uhd/config.hpp>
#include <uhd/types/time_spec.hpp>
-#include <boost/thread/thread.hpp>
#include <atomic>
#include <chrono>
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 <typename T>
-UHD_INLINE bool spin_wait_with_timeout(
- std::atomic<T>& 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<bool> _locked;
+ std::atomic_flag _locked;
};
} // namespace uhd