aboutsummaryrefslogtreecommitdiffstats
path: root/host/include/uhd/utils/atomic.hpp
diff options
context:
space:
mode:
Diffstat (limited to 'host/include/uhd/utils/atomic.hpp')
-rw-r--r--host/include/uhd/utils/atomic.hpp161
1 files changed, 161 insertions, 0 deletions
diff --git a/host/include/uhd/utils/atomic.hpp b/host/include/uhd/utils/atomic.hpp
new file mode 100644
index 000000000..7a81d8d5e
--- /dev/null
+++ b/host/include/uhd/utils/atomic.hpp
@@ -0,0 +1,161 @@
+//
+// Copyright 2012 Ettus Research LLC
+//
+// This program is free software: you can redistribute it and/or modify
+// it under the terms of the GNU General Public License as published by
+// the Free Software Foundation, either version 3 of the License, or
+// (at your option) any later version.
+//
+// This program is distributed in the hope that it will be useful,
+// but WITHOUT ANY WARRANTY; without even the implied warranty of
+// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+// GNU General Public License for more details.
+//
+// You should have received a copy of the GNU General Public License
+// along with this program. If not, see <http://www.gnu.org/licenses/>.
+//
+
+#ifndef INCLUDED_UHD_UTILS_ATOMIC_HPP
+#define INCLUDED_UHD_UTILS_ATOMIC_HPP
+
+#include <uhd/config.hpp>
+#include <uhd/types/time_spec.hpp>
+#include <boost/thread/thread.hpp>
+#include <boost/interprocess/detail/atomic.hpp>
+
+#include <boost/version.hpp>
+#if BOOST_VERSION >= 104800
+# define BOOST_IPC_DETAIL boost::interprocess::ipcdetail
+#else
+# define BOOST_IPC_DETAIL boost::interprocess::detail
+#endif
+
+namespace uhd{
+
+ //! A 32-bit integer that can be atomically accessed
+ class UHD_API atomic_uint32_t{
+ public:
+
+ //! Create a new atomic 32-bit integer, initialized to zero
+ UHD_INLINE atomic_uint32_t(void){
+ this->write(0);
+ }
+
+ //! Compare with cmp, swap with newval if same, return old value
+ UHD_INLINE boost::uint32_t cas(boost::uint32_t newval, boost::uint32_t cmp){
+ return BOOST_IPC_DETAIL::atomic_cas32(&_num, newval, cmp);
+ }
+
+ //! Sets the atomic integer to a new value
+ UHD_INLINE void write(const boost::uint32_t newval){
+ BOOST_IPC_DETAIL::atomic_write32(&_num, newval);
+ }
+
+ //! Gets the current value of the atomic integer
+ UHD_INLINE boost::uint32_t read(void){
+ return BOOST_IPC_DETAIL::atomic_read32(&_num);
+ }
+
+ //! Increment by 1 and return the old value
+ UHD_INLINE boost::uint32_t inc(void){
+ return BOOST_IPC_DETAIL::atomic_inc32(&_num);
+ }
+
+ //! Decrement by 1 and return the old value
+ UHD_INLINE boost::uint32_t dec(void){
+ return BOOST_IPC_DETAIL::atomic_dec32(&_num);
+ }
+
+ private: volatile boost::uint32_t _num;
+ };
+
+ /*!
+ * A reusable barrier to sync multiple threads.
+ * All threads spin on wait() until count is reset.
+ */
+ class UHD_API reusable_barrier{
+ public:
+
+ //! Resize the barrier for N threads
+ void resize(const size_t size){
+ _size = size;
+ _count.write(size);
+ }
+
+ /*!
+ * Force the barrier wait to throw a boost::thread_interrupted
+ * The threads were not getting the interruption_point on windows.
+ */
+ void interrupt(void)
+ {
+ _count.write(boost::uint32_t(~0));
+ }
+
+ //! Wait on the barrier condition
+ UHD_INLINE void wait(void){
+ _count.dec();
+ _count.cas(_size, 0);
+ while (_count.read() != _size){
+ boost::this_thread::interruption_point();
+ if (_count.read() == boost::uint32_t(~0))
+ throw boost::thread_interrupted();
+ boost::this_thread::yield();
+ }
+ }
+
+ private:
+ size_t _size;
+ atomic_uint32_t _count;
+ };
+
+ /*!
+ * 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
+ */
+ UHD_INLINE bool spin_wait_with_timeout(
+ atomic_uint32_t &cond,
+ boost::uint32_t value,
+ const double timeout
+ ){
+ if (cond.read() == value) return true;
+ const time_spec_t exit_time = time_spec_t::get_system_time() + time_spec_t(timeout);
+ while (cond.read() != value){
+ if (time_spec_t::get_system_time() > exit_time) return false;
+ boost::this_thread::interruption_point();
+ boost::this_thread::yield();
+ }
+ return true;
+ }
+
+ /*!
+ * Claimer class to provide synchronization for multi-thread access.
+ * Claiming enables buffer classes to be used with a buffer queue.
+ */
+ class simple_claimer{
+ public:
+ simple_claimer(void){
+ this->release();
+ }
+
+ UHD_INLINE void release(void){
+ _locked.write(0);
+ }
+
+ UHD_INLINE bool claim_with_wait(const double timeout){
+ if (spin_wait_with_timeout(_locked, 0, timeout)){
+ _locked.write(1);
+ return true;
+ }
+ return false;
+ }
+
+ private:
+ atomic_uint32_t _locked;
+ };
+
+} //namespace uhd
+
+#endif /* INCLUDED_UHD_UTILS_ATOMIC_HPP */