completion_latch.hpp 6.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226
  1. // Distributed under the Boost Software License, Version 1.0. (See
  2. // accompanying file LICENSE_1_0.txt or copy at
  3. // http://www.boost.org/LICENSE_1_0.txt)
  4. // (C) Copyright 2013 Vicente J. Botet Escriba
  5. #ifndef BOOST_THREAD_COMPLETION_LATCH_HPP
  6. #define BOOST_THREAD_COMPLETION_LATCH_HPP
  7. #include <boost/thread/detail/config.hpp>
  8. #include <boost/thread/detail/delete.hpp>
  9. #include <boost/thread/detail/counter.hpp>
  10. #include <boost/thread/mutex.hpp>
  11. #include <boost/thread/lock_types.hpp>
  12. #include <boost/thread/condition_variable.hpp>
  13. #include <boost/chrono/duration.hpp>
  14. #include <boost/chrono/time_point.hpp>
  15. #include <boost/assert.hpp>
  16. //#include <boost/thread/detail/nullary_function.hpp>
  17. #include <boost/thread/csbl/functional.hpp>
  18. #include <boost/config/abi_prefix.hpp>
  19. namespace boost
  20. {
  21. namespace thread_detail
  22. {
  23. void noop()
  24. {
  25. }
  26. }
  27. class completion_latch
  28. {
  29. public:
  30. /// the implementation defined completion function type
  31. //typedef detail::nullary_function<void()> completion_function;
  32. typedef csbl::function<void()> completion_function;
  33. /// noop completion function factory
  34. static completion_function noop()
  35. {
  36. return completion_function(&thread_detail::noop);
  37. }
  38. private:
  39. struct around_wait;
  40. friend struct around_wait;
  41. struct around_wait
  42. {
  43. completion_latch &that_;
  44. boost::unique_lock<boost::mutex> &lk_;
  45. around_wait(completion_latch &that, boost::unique_lock<boost::mutex> &lk)
  46. : that_(that), lk_(lk)
  47. {
  48. that_.leavers_.cond_.wait(lk, detail::counter_is_zero(that_.leavers_));
  49. that_.waiters_.inc_and_notify_all();
  50. that_.leavers_.cond_.wait(lk, detail::counter_is_not_zero(that_.leavers_));
  51. }
  52. ~around_wait()
  53. {
  54. that_.waiters_.dec_and_notify_all();
  55. }
  56. };
  57. bool count_down(unique_lock<mutex> &lk)
  58. {
  59. BOOST_ASSERT(count_ > 0);
  60. if (--count_ == 0)
  61. {
  62. waiters_.cond_.wait(lk, detail::counter_is_not_zero(waiters_));
  63. leavers_.assign_and_notify_all(waiters_);
  64. count_.cond_.notify_all();
  65. waiters_.cond_.wait(lk, detail::counter_is_zero(waiters_));
  66. leavers_.assign_and_notify_all(0);
  67. lk.unlock();
  68. funct_();
  69. return true;
  70. }
  71. return false;
  72. }
  73. public:
  74. BOOST_THREAD_NO_COPYABLE( completion_latch )
  75. /// Constructs a latch with a given count.
  76. completion_latch(std::size_t count) :
  77. count_(count), funct_(noop()), waiters_(0), leavers_(0)
  78. {
  79. }
  80. /// Constructs a latch with a given count and a completion function.
  81. template <typename F>
  82. completion_latch(std::size_t count, BOOST_THREAD_RV_REF(F) funct) :
  83. count_(count),
  84. funct_(boost::move(funct)),
  85. waiters_(0),
  86. leavers_(0)
  87. {
  88. }
  89. template <typename F>
  90. completion_latch(std::size_t count, void(*funct)()) :
  91. count_(count), funct_(funct), waiters_(0), leavers_(0)
  92. {
  93. }
  94. ///
  95. ~completion_latch()
  96. {
  97. }
  98. /// Blocks until the latch has counted down to zero.
  99. void wait()
  100. {
  101. boost::unique_lock<boost::mutex> lk(mutex_);
  102. around_wait aw(*this, lk);
  103. count_.cond_.wait(lk, detail::counter_is_zero(count_));
  104. }
  105. /// @return true if the internal counter is already 0, false otherwise
  106. bool try_wait()
  107. {
  108. boost::unique_lock<boost::mutex> lk(mutex_);
  109. around_wait aw(*this, lk);
  110. return (count_ == 0);
  111. }
  112. /// try to wait for a specified amount of time
  113. /// @return whether there is a timeout or not.
  114. template <class Rep, class Period>
  115. cv_status wait_for(const chrono::duration<Rep, Period>& rel_time)
  116. {
  117. boost::unique_lock<boost::mutex> lk(mutex_);
  118. around_wait aw(*this, lk);
  119. return count_.cond_.wait_for(lk, rel_time, detail::counter_is_zero(count_))
  120. ? cv_status::no_timeout
  121. : cv_status::timeout;
  122. }
  123. /// try to wait until the specified time_point is reached
  124. /// @return whether there is a timeout or not.
  125. template <class Clock, class Duration>
  126. cv_status wait_until(const chrono::time_point<Clock, Duration>& abs_time)
  127. {
  128. boost::unique_lock<boost::mutex> lk(mutex_);
  129. around_wait aw(*this, lk);
  130. return count_.cond_.wait_until(lk, abs_time, detail::counter_is_zero(count_))
  131. ? cv_status::no_timeout
  132. : cv_status::timeout;
  133. }
  134. /// Decrement the count and notify anyone waiting if we reach zero.
  135. /// @Requires count must be greater than 0
  136. void count_down()
  137. {
  138. unique_lock<mutex> lk(mutex_);
  139. count_down(lk);
  140. }
  141. void signal()
  142. {
  143. count_down();
  144. }
  145. /// Decrement the count and notify anyone waiting if we reach zero.
  146. /// Blocks until the latch has counted down to zero.
  147. /// @Requires count must be greater than 0
  148. void count_down_and_wait()
  149. {
  150. boost::unique_lock<boost::mutex> lk(mutex_);
  151. if (count_down(lk))
  152. {
  153. return;
  154. }
  155. around_wait aw(*this, lk);
  156. count_.cond_.wait(lk, detail::counter_is_zero(count_));
  157. }
  158. void sync()
  159. {
  160. count_down_and_wait();
  161. }
  162. /// Reset the counter
  163. /// #Requires This method may only be invoked when there are no other threads currently inside the count_down_and_wait() method.
  164. void reset(std::size_t count)
  165. {
  166. boost::lock_guard<boost::mutex> lk(mutex_);
  167. //BOOST_ASSERT(count_ == 0);
  168. count_ = count;
  169. }
  170. /// Resets the latch with the new completion function.
  171. /// The next time the internal count reaches 0, this function will be invoked.
  172. /// This completion function may only be invoked when there are no other threads
  173. /// currently inside the count_down and wait related functions.
  174. /// It may also be invoked from within the registered completion function.
  175. /// @Returns the old completion function if any or noop if
  176. #ifdef BOOST_NO_CXX11_HDR_FUNCTIONAL
  177. template <typename F>
  178. completion_function then(BOOST_THREAD_RV_REF(F) funct)
  179. {
  180. boost::lock_guard<boost::mutex> lk(mutex_);
  181. completion_function tmp(funct_);
  182. funct_ = boost::move(funct);
  183. return tmp;
  184. }
  185. #endif
  186. completion_function then(void(*funct)())
  187. {
  188. boost::lock_guard<boost::mutex> lk(mutex_);
  189. completion_function tmp(funct_);
  190. funct_ = completion_function(funct);
  191. return tmp;
  192. }
  193. private:
  194. mutex mutex_;
  195. detail::counter count_;
  196. completion_function funct_;
  197. detail::counter waiters_;
  198. detail::counter leavers_;
  199. };
  200. } // namespace boost
  201. #include <boost/config/abi_suffix.hpp>
  202. #endif