externally_locked.hpp 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351
  1. // (C) Copyright 2012 Vicente J. Botet Escriba
  2. // Distributed under the Boost Software License, Version 1.0. (See
  3. // accompanying file LICENSE_1_0.txt or copy at
  4. // http://www.boost.org/LICENSE_1_0.txt)
  5. #ifndef BOOST_THREAD_EXTERNALLY_LOCKED_HPP
  6. #define BOOST_THREAD_EXTERNALLY_LOCKED_HPP
  7. #include <boost/thread/detail/config.hpp>
  8. #include <boost/thread/exceptions.hpp>
  9. #include <boost/thread/lock_concepts.hpp>
  10. #include <boost/thread/lock_traits.hpp>
  11. #include <boost/thread/lockable_concepts.hpp>
  12. #include <boost/thread/strict_lock.hpp>
  13. #include <boost/static_assert.hpp>
  14. #include <boost/type_traits/is_same.hpp>
  15. #include <boost/throw_exception.hpp>
  16. #include <boost/core/swap.hpp>
  17. #include <boost/config/abi_prefix.hpp>
  18. namespace boost
  19. {
  20. /**
  21. * externally_locked cloaks an object of type T, and actually provides full
  22. * access to that object through the get and set member functions, provided you
  23. * pass a reference to a strict lock object
  24. */
  25. //[externally_locked
  26. template <typename T, typename MutexType = boost::mutex>
  27. class externally_locked;
  28. template <typename T, typename MutexType>
  29. class externally_locked
  30. {
  31. //BOOST_CONCEPT_ASSERT(( CopyConstructible<T> ));
  32. BOOST_CONCEPT_ASSERT(( BasicLockable<MutexType> ));
  33. public:
  34. typedef MutexType mutex_type;
  35. BOOST_THREAD_COPYABLE_AND_MOVABLE( externally_locked )
  36. /**
  37. * Requires: T is a model of CopyConstructible.
  38. * Effects: Constructs an externally locked object copying the cloaked type.
  39. */
  40. externally_locked(mutex_type& mtx, const T& obj) :
  41. obj_(obj), mtx_(&mtx)
  42. {
  43. }
  44. /**
  45. * Requires: T is a model of Movable.
  46. * Effects: Constructs an externally locked object by moving the cloaked type.
  47. */
  48. externally_locked(mutex_type& mtx, BOOST_THREAD_RV_REF(T) obj) :
  49. obj_(move(obj)), mtx_(&mtx)
  50. {
  51. }
  52. /**
  53. * Requires: T is a model of DefaultConstructible.
  54. * Effects: Constructs an externally locked object initializing the cloaked type with the default constructor.
  55. */
  56. externally_locked(mutex_type& mtx) // BOOST_NOEXCEPT_IF(BOOST_NOEXCEPT_EXPR(T()))
  57. : obj_(), mtx_(&mtx)
  58. {
  59. }
  60. /**
  61. * Copy constructor
  62. */
  63. externally_locked(externally_locked const& rhs) //BOOST_NOEXCEPT
  64. : obj_(rhs.obj_), mtx_(rhs.mtx_)
  65. {
  66. }
  67. /**
  68. * Move constructor
  69. */
  70. externally_locked(BOOST_THREAD_RV_REF(externally_locked) rhs) //BOOST_NOEXCEPT
  71. : obj_(move(rhs.obj_)), mtx_(rhs.mtx_)
  72. {
  73. }
  74. /// assignment
  75. externally_locked& operator=(externally_locked const& rhs) //BOOST_NOEXCEPT
  76. {
  77. obj_=rhs.obj_;
  78. mtx_=rhs.mtx_;
  79. return *this;
  80. }
  81. /// move assignment
  82. externally_locked& operator=(BOOST_THREAD_RV_REF(externally_locked) rhs) // BOOST_NOEXCEPT
  83. {
  84. obj_=move(BOOST_THREAD_RV(rhs).obj_);
  85. mtx_=rhs.mtx_;
  86. return *this;
  87. }
  88. void swap(externally_locked& rhs) //BOOST_NOEXCEPT_IF(BOOST_NOEXCEPT_EXPR)
  89. {
  90. swap(obj_, rhs.obj_);
  91. swap(mtx_, rhs.mtx_);
  92. }
  93. /**
  94. * Requires: The lk parameter must be locking the associated mtx.
  95. *
  96. * Returns: The address of the cloaked object..
  97. *
  98. * Throws: lock_error if BOOST_THREAD_THROW_IF_PRECONDITION_NOT_SATISFIED is not defined and the lk parameter doesn't satisfy the preconditions
  99. */
  100. T& get(strict_lock<mutex_type>& lk)
  101. {
  102. BOOST_THREAD_ASSERT_PRECONDITION( lk.owns_lock(mtx_), lock_error() ); /*< run time check throw if not locks the same >*/
  103. return obj_;
  104. }
  105. const T& get(strict_lock<mutex_type>& lk) const
  106. {
  107. BOOST_THREAD_ASSERT_PRECONDITION( lk.owns_lock(mtx_), lock_error() ); /*< run time check throw if not locks the same >*/
  108. return obj_;
  109. }
  110. template <class Lock>
  111. T& get(nested_strict_lock<Lock>& lk)
  112. {
  113. BOOST_STATIC_ASSERT( (is_same<mutex_type, typename Lock::mutex_type>::value)); /*< that locks the same type >*/
  114. BOOST_THREAD_ASSERT_PRECONDITION( lk.owns_lock(mtx_), lock_error() ); /*< run time check throw if not locks the same >*/
  115. return obj_;
  116. }
  117. template <class Lock>
  118. const T& get(nested_strict_lock<Lock>& lk) const
  119. {
  120. BOOST_STATIC_ASSERT( (is_same<mutex_type, typename Lock::mutex_type>::value)); /*< that locks the same type >*/
  121. BOOST_THREAD_ASSERT_PRECONDITION( lk.owns_lock(mtx_), lock_error() ); /*< run time check throw if not locks the same >*/
  122. return obj_;
  123. }
  124. /**
  125. * Requires: The lk parameter must be locking the associated mtx.
  126. * Returns: The address of the cloaked object..
  127. *
  128. * Throws: lock_error if BOOST_THREAD_THROW_IF_PRECONDITION_NOT_SATISFIED is not defined and the lk parameter doesn't satisfy the preconditions
  129. */
  130. template <class Lock>
  131. T& get(Lock& lk)
  132. {
  133. BOOST_CONCEPT_ASSERT(( StrictLock<Lock> ));
  134. BOOST_STATIC_ASSERT( (is_strict_lock<Lock>::value)); /*< lk is a strict lock "sur parolle" >*/
  135. BOOST_STATIC_ASSERT( (is_same<mutex_type, typename Lock::mutex_type>::value)); /*< that locks the same type >*/
  136. BOOST_THREAD_ASSERT_PRECONDITION( lk.owns_lock(mtx_), lock_error() ); /*< run time check throw if not locks the same >*/
  137. return obj_;
  138. }
  139. mutex_type* mutex() const BOOST_NOEXCEPT
  140. {
  141. return mtx_;
  142. }
  143. // modifiers
  144. void lock()
  145. {
  146. mtx_->lock();
  147. }
  148. void unlock()
  149. {
  150. mtx_->unlock();
  151. }
  152. bool try_lock()
  153. {
  154. return mtx_->try_lock();
  155. }
  156. // todo add time related functions
  157. private:
  158. T obj_;
  159. mutex_type* mtx_;
  160. };
  161. //]
  162. /**
  163. * externally_locked<T&,M> specialization for T& that cloaks an reference to an object of type T, and actually
  164. * provides full access to that object through the get and set member functions, provided you
  165. * pass a reference to a strict lock object.
  166. */
  167. //[externally_locked_ref
  168. template <typename T, typename MutexType>
  169. class externally_locked<T&, MutexType>
  170. {
  171. //BOOST_CONCEPT_ASSERT(( CopyConstructible<T> ));
  172. BOOST_CONCEPT_ASSERT(( BasicLockable<MutexType> ));
  173. public:
  174. typedef MutexType mutex_type;
  175. BOOST_THREAD_COPYABLE_AND_MOVABLE( externally_locked )
  176. /**
  177. * Effects: Constructs an externally locked object storing the cloaked reference object.
  178. */
  179. externally_locked(T& obj, mutex_type& mtx) BOOST_NOEXCEPT :
  180. obj_(&obj), mtx_(&mtx)
  181. {
  182. }
  183. /// copy constructor
  184. externally_locked(externally_locked const& rhs) BOOST_NOEXCEPT :
  185. obj_(rhs.obj_), mtx_(rhs.mtx_)
  186. {
  187. }
  188. /// move constructor
  189. externally_locked(BOOST_THREAD_RV_REF(externally_locked) rhs) BOOST_NOEXCEPT :
  190. obj_(rhs.obj_), mtx_(rhs.mtx_)
  191. {
  192. }
  193. /// assignment
  194. externally_locked& operator=(externally_locked const& rhs) BOOST_NOEXCEPT
  195. {
  196. obj_=rhs.obj_;
  197. mtx_=rhs.mtx_;
  198. return *this;
  199. }
  200. /// move assignment
  201. externally_locked& operator=(BOOST_THREAD_RV_REF(externally_locked) rhs) BOOST_NOEXCEPT
  202. {
  203. obj_=rhs.obj_;
  204. mtx_=rhs.mtx_;
  205. return *this;
  206. }
  207. void swap(externally_locked& rhs) BOOST_NOEXCEPT
  208. {
  209. swap(obj_, rhs.obj_);
  210. swap(mtx_, rhs.mtx_);
  211. }
  212. /**
  213. * Requires: The lk parameter must be locking the associated mtx.
  214. *
  215. * Returns: The address of the cloaked object..
  216. *
  217. * Throws: lock_error if BOOST_THREAD_THROW_IF_PRECONDITION_NOT_SATISFIED is not defined and the lk parameter doesn't satisfy the preconditions
  218. */
  219. T& get(strict_lock<mutex_type> const& lk)
  220. {
  221. BOOST_THREAD_ASSERT_PRECONDITION( lk.owns_lock(mtx_), lock_error() ); /*< run time check throw if not locks the same >*/
  222. return *obj_;
  223. }
  224. const T& get(strict_lock<mutex_type> const& lk) const
  225. {
  226. BOOST_THREAD_ASSERT_PRECONDITION( lk.owns_lock(mtx_), lock_error() ); /*< run time check throw if not locks the same >*/
  227. return *obj_;
  228. }
  229. template <class Lock>
  230. T& get(nested_strict_lock<Lock> const& lk)
  231. {
  232. BOOST_STATIC_ASSERT( (is_same<mutex_type, typename Lock::mutex_type>::value)); /*< that locks the same type >*/
  233. BOOST_THREAD_ASSERT_PRECONDITION( lk.owns_lock(mtx_), lock_error() ); /*< run time check throw if not locks the same >*/
  234. return *obj_;
  235. }
  236. template <class Lock>
  237. const T& get(nested_strict_lock<Lock> const& lk) const
  238. {
  239. BOOST_STATIC_ASSERT( (is_same<mutex_type, typename Lock::mutex_type>::value)); /*< that locks the same type >*/
  240. BOOST_THREAD_ASSERT_PRECONDITION( lk.owns_lock(mtx_), lock_error() ); /*< run time check throw if not locks the same >*/
  241. return *obj_;
  242. }
  243. /**
  244. * Requires: The lk parameter must be locking the associated mtx.
  245. * Returns: The address of the cloaked object..
  246. *
  247. * Throws: lock_error if BOOST_THREAD_THROW_IF_PRECONDITION_NOT_SATISFIED is not defined and the lk parameter doesn't satisfy the preconditions
  248. */
  249. template <class Lock>
  250. T& get(Lock const& lk)
  251. {
  252. BOOST_CONCEPT_ASSERT(( StrictLock<Lock> ));
  253. BOOST_STATIC_ASSERT( (is_strict_lock<Lock>::value)); /*< lk is a strict lock "sur parolle" >*/
  254. BOOST_STATIC_ASSERT( (is_same<mutex_type, typename Lock::mutex_type>::value)); /*< that locks the same type >*/
  255. BOOST_THREAD_ASSERT_PRECONDITION( lk.owns_lock(mtx_), lock_error() ); /*< run time check throw if not locks the same >*/
  256. return *obj_;
  257. }
  258. /**
  259. * Requires: The lk parameter must be locking the associated mtx.
  260. * Returns: The address of the cloaked object..
  261. *
  262. * Throws: lock_error if BOOST_THREAD_THROW_IF_PRECONDITION_NOT_SATISFIED is not defined and the lk parameter doesn't satisfy the preconditions
  263. */
  264. template <class Lock>
  265. T const& get(Lock const& lk) const
  266. {
  267. BOOST_CONCEPT_ASSERT(( StrictLock<Lock> ));
  268. BOOST_STATIC_ASSERT( (is_strict_lock<Lock>::value)); /*< lk is a strict lock "sur parolle" >*/
  269. BOOST_STATIC_ASSERT( (is_same<mutex_type, typename Lock::mutex_type>::value)); /*< that locks the same type >*/
  270. BOOST_THREAD_ASSERT_PRECONDITION( lk.owns_lock(mtx_), lock_error() ); /*< run time check throw if not locks the same >*/
  271. return *obj_;
  272. }
  273. mutex_type* mutex() const BOOST_NOEXCEPT
  274. {
  275. return mtx_;
  276. }
  277. void lock()
  278. {
  279. mtx_->lock();
  280. }
  281. void unlock()
  282. {
  283. mtx_->unlock();
  284. }
  285. bool try_lock()
  286. {
  287. return mtx_->try_lock();
  288. }
  289. // todo add time related functions
  290. protected:
  291. T* obj_;
  292. mutex_type* mtx_;
  293. };
  294. //]
  295. template <typename T, typename MutexType>
  296. void swap(externally_locked<T, MutexType> & lhs, externally_locked<T, MutexType> & rhs) // BOOST_NOEXCEPT
  297. {
  298. lhs.swap(rhs);
  299. }
  300. }
  301. #include <boost/config/abi_suffix.hpp>
  302. #endif // header