UHD  003.005.004-0-ga219f16c
atomic.hpp
Go to the documentation of this file.
1 //
2 // Copyright 2012-2013 Ettus Research LLC
3 //
4 // This program is free software: you can redistribute it and/or modify
5 // it under the terms of the GNU General Public License as published by
6 // the Free Software Foundation, either version 3 of the License, or
7 // (at your option) any later version.
8 //
9 // This program is distributed in the hope that it will be useful,
10 // but WITHOUT ANY WARRANTY; without even the implied warranty of
11 // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 // GNU General Public License for more details.
13 //
14 // You should have received a copy of the GNU General Public License
15 // along with this program. If not, see <http://www.gnu.org/licenses/>.
16 //
17 
18 #ifndef INCLUDED_UHD_UTILS_ATOMIC_HPP
19 #define INCLUDED_UHD_UTILS_ATOMIC_HPP
20 
21 #include <uhd/config.hpp>
22 #include <uhd/types/time_spec.hpp>
23 #include <boost/thread/thread.hpp>
24 #include <boost/thread/mutex.hpp>
25 #include <boost/thread/condition_variable.hpp>
26 #include <boost/interprocess/detail/atomic.hpp>
27 
28 #include <boost/version.hpp>
29 #if BOOST_VERSION >= 104800
30 # define BOOST_IPC_DETAIL boost::interprocess::ipcdetail
31 #else
32 # define BOOST_IPC_DETAIL boost::interprocess::detail
33 #endif
34 
35 namespace uhd{
36 
39  public:
40 
43  this->write(0);
44  }
45 
47  UHD_INLINE boost::uint32_t cas(boost::uint32_t newval, boost::uint32_t cmp){
48  return BOOST_IPC_DETAIL::atomic_cas32(&_num, newval, cmp);
49  }
50 
52  UHD_INLINE void write(const boost::uint32_t newval){
53  BOOST_IPC_DETAIL::atomic_write32(&_num, newval);
54  }
55 
57  UHD_INLINE boost::uint32_t read(void){
58  return BOOST_IPC_DETAIL::atomic_read32(&_num);
59  }
60 
62  UHD_INLINE boost::uint32_t inc(void){
63  return BOOST_IPC_DETAIL::atomic_inc32(&_num);
64  }
65 
67  UHD_INLINE boost::uint32_t dec(void){
68  return BOOST_IPC_DETAIL::atomic_dec32(&_num);
69  }
70 
71  private: volatile boost::uint32_t _num;
72  };
73 
79  public:
80 
82  void resize(const size_t size){
83  _size = size;
84  }
85 
90  void interrupt(void)
91  {
92  _done.inc();
93  }
94 
96  UHD_INLINE void wait(void)
97  {
98  if (_size == 1) return;
99 
100  //entry barrier with condition variable
101  _entry_counter.inc();
102  _entry_counter.cas(0, _size);
103  boost::mutex::scoped_lock lock(_mutex);
104  while (_entry_counter.read() != 0)
105  {
106  this->check_interrupt();
107  _cond.timed_wait(lock, boost::posix_time::milliseconds(1));
108  }
109  lock.unlock(); //unlock before notify
110  _cond.notify_one();
111 
112  //exit barrier to ensure known condition of entry count
113  _exit_counter.inc();
114  _exit_counter.cas(0, _size);
115  while (_exit_counter.read() != 0) this->check_interrupt();
116  }
117 
120  {
121  while (_entry_counter.read() != (_size-1)) this->check_interrupt();
122  }
123 
124  private:
125  size_t _size;
126  atomic_uint32_t _entry_counter;
127  atomic_uint32_t _exit_counter;
128  atomic_uint32_t _done;
129  boost::mutex _mutex;
130  boost::condition_variable _cond;
131 
132  UHD_INLINE void check_interrupt(void)
133  {
134  if (_done.read() != 0) throw boost::thread_interrupted();
135  boost::this_thread::interruption_point();
136  boost::this_thread::yield();
137  }
138  };
139 
148  atomic_uint32_t &cond,
149  boost::uint32_t value,
150  const double timeout
151  ){
152  if (cond.read() == value) return true;
153  const time_spec_t exit_time = time_spec_t::get_system_time() + time_spec_t(timeout);
154  while (cond.read() != value){
155  if (time_spec_t::get_system_time() > exit_time) return false;
156  boost::this_thread::interruption_point();
157  boost::this_thread::yield();
158  }
159  return true;
160  }
161 
167  public:
169  this->release();
170  }
171 
172  UHD_INLINE void release(void){
173  _locked.write(0);
174  }
175 
176  UHD_INLINE bool claim_with_wait(const double timeout){
177  if (spin_wait_with_timeout(_locked, 0, timeout)){
178  _locked.write(1);
179  return true;
180  }
181  return false;
182  }
183 
184  private:
185  atomic_uint32_t _locked;
186  };
187 
188 } //namespace uhd
189 
190 #endif /* INCLUDED_UHD_UTILS_ATOMIC_HPP */
UHD_INLINE boost::uint32_t read(void)
Gets the current value of the atomic integer.
Definition: atomic.hpp:57
UHD_INLINE void wait(void)
Wait on the barrier condition.
Definition: atomic.hpp:96
UHD_INLINE void release(void)
Definition: atomic.hpp:172
Definition: atomic.hpp:78
Definition: time_spec.hpp:39
A 32-bit integer that can be atomically accessed.
Definition: atomic.hpp:38
UHD_INLINE void wait_others(void)
Wait on the barrier condition.
Definition: atomic.hpp:119
UHD_INLINE bool spin_wait_with_timeout(atomic_uint32_t &cond, boost::uint32_t value, const double timeout)
Definition: atomic.hpp:147
static time_spec_t get_system_time(void)
#define UHD_API
Definition: config.hpp:76
Definition: convert.hpp:28
UHD_INLINE boost::uint32_t cas(boost::uint32_t newval, boost::uint32_t cmp)
Compare with cmp, swap with newval if same, return old value.
Definition: atomic.hpp:47
UHD_INLINE bool claim_with_wait(const double timeout)
Definition: atomic.hpp:176
#define UHD_INLINE
Definition: config.hpp:67
UHD_INLINE atomic_uint32_t(void)
Create a new atomic 32-bit integer, initialized to zero.
Definition: atomic.hpp:42
UHD_INLINE void write(const boost::uint32_t newval)
Sets the atomic integer to a new value.
Definition: atomic.hpp:52
UHD_INLINE boost::uint32_t inc(void)
Increment by 1 and return the old value.
Definition: atomic.hpp:62
simple_claimer(void)
Definition: atomic.hpp:168
UHD_INLINE boost::uint32_t dec(void)
Decrement by 1 and return the old value.
Definition: atomic.hpp:67
void resize(const size_t size)
Resize the barrier for N threads.
Definition: atomic.hpp:82
Definition: atomic.hpp:166
void interrupt(void)
Definition: atomic.hpp:90