mutex.hpp 9.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361
  1. #ifndef BOOST_THREAD_PTHREAD_MUTEX_HPP
  2. #define BOOST_THREAD_PTHREAD_MUTEX_HPP
  3. // (C) Copyright 2007-8 Anthony Williams
  4. // (C) Copyright 2011,2012,2015 Vicente J. Botet Escriba
  5. // Distributed under the Boost Software License, Version 1.0. (See
  6. // accompanying file LICENSE_1_0.txt or copy at
  7. // http://www.boost.org/LICENSE_1_0.txt)
  8. #include <boost/thread/detail/config.hpp>
  9. #include <boost/assert.hpp>
  10. #include <pthread.h>
  11. #include <boost/throw_exception.hpp>
  12. #include <boost/core/ignore_unused.hpp>
  13. #include <boost/thread/exceptions.hpp>
  14. #if defined BOOST_THREAD_PROVIDES_NESTED_LOCKS
  15. #include <boost/thread/lock_types.hpp>
  16. #endif
  17. #include <boost/thread/thread_time.hpp>
  18. #include <boost/thread/xtime.hpp>
  19. #include <boost/assert.hpp>
  20. #include <errno.h>
  21. #include <boost/thread/pthread/timespec.hpp>
  22. #include <boost/thread/pthread/pthread_mutex_scoped_lock.hpp>
  23. #ifdef BOOST_THREAD_USES_CHRONO
  24. #include <boost/chrono/system_clocks.hpp>
  25. #include <boost/chrono/ceil.hpp>
  26. #endif
  27. #include <boost/thread/detail/delete.hpp>
  28. #if (defined(_POSIX_TIMEOUTS) && (_POSIX_TIMEOUTS-0)>=200112L) \
  29. || (defined(__ANDROID__) && defined(__ANDROID_API__) && __ANDROID_API__ >= 21)
  30. #ifndef BOOST_PTHREAD_HAS_TIMEDLOCK
  31. #define BOOST_PTHREAD_HAS_TIMEDLOCK
  32. #endif
  33. #endif
  34. #include <boost/config/abi_prefix.hpp>
  35. #ifndef BOOST_THREAD_HAS_NO_EINTR_BUG
  36. #define BOOST_THREAD_HAS_EINTR_BUG
  37. #endif
  38. namespace boost
  39. {
  40. namespace posix {
  41. #ifdef BOOST_THREAD_HAS_EINTR_BUG
  42. BOOST_FORCEINLINE int pthread_mutex_destroy(pthread_mutex_t* m)
  43. {
  44. int ret;
  45. do
  46. {
  47. ret = ::pthread_mutex_destroy(m);
  48. } while (ret == EINTR);
  49. return ret;
  50. }
  51. BOOST_FORCEINLINE int pthread_mutex_lock(pthread_mutex_t* m)
  52. {
  53. int ret;
  54. do
  55. {
  56. ret = ::pthread_mutex_lock(m);
  57. } while (ret == EINTR);
  58. return ret;
  59. }
  60. BOOST_FORCEINLINE int pthread_mutex_unlock(pthread_mutex_t* m)
  61. {
  62. int ret;
  63. do
  64. {
  65. ret = ::pthread_mutex_unlock(m);
  66. } while (ret == EINTR);
  67. return ret;
  68. }
  69. #else
  70. BOOST_FORCEINLINE int pthread_mutex_destroy(pthread_mutex_t* m)
  71. {
  72. return ::pthread_mutex_destroy(m);
  73. }
  74. BOOST_FORCEINLINE int pthread_mutex_lock(pthread_mutex_t* m)
  75. {
  76. return ::pthread_mutex_lock(m);
  77. }
  78. BOOST_FORCEINLINE int pthread_mutex_unlock(pthread_mutex_t* m)
  79. {
  80. return ::pthread_mutex_unlock(m);
  81. }
  82. #endif
  83. }
  84. class mutex
  85. {
  86. private:
  87. pthread_mutex_t m;
  88. public:
  89. BOOST_THREAD_NO_COPYABLE(mutex)
  90. mutex()
  91. {
  92. int const res=pthread_mutex_init(&m,NULL);
  93. if(res)
  94. {
  95. boost::throw_exception(thread_resource_error(res, "boost:: mutex constructor failed in pthread_mutex_init"));
  96. }
  97. }
  98. ~mutex()
  99. {
  100. int const res = posix::pthread_mutex_destroy(&m);
  101. boost::ignore_unused(res);
  102. BOOST_ASSERT(!res);
  103. }
  104. void lock()
  105. {
  106. int res = posix::pthread_mutex_lock(&m);
  107. if (res)
  108. {
  109. boost::throw_exception(lock_error(res,"boost: mutex lock failed in pthread_mutex_lock"));
  110. }
  111. }
  112. void unlock()
  113. {
  114. int res = posix::pthread_mutex_unlock(&m);
  115. (void)res;
  116. BOOST_ASSERT(res == 0);
  117. // if (res)
  118. // {
  119. // boost::throw_exception(lock_error(res,"boost: mutex unlock failed in pthread_mutex_unlock"));
  120. // }
  121. }
  122. bool try_lock()
  123. {
  124. int res;
  125. do
  126. {
  127. res = pthread_mutex_trylock(&m);
  128. } while (res == EINTR);
  129. if (res==EBUSY)
  130. {
  131. return false;
  132. }
  133. return !res;
  134. }
  135. #define BOOST_THREAD_DEFINES_MUTEX_NATIVE_HANDLE
  136. typedef pthread_mutex_t* native_handle_type;
  137. native_handle_type native_handle()
  138. {
  139. return &m;
  140. }
  141. #if defined BOOST_THREAD_PROVIDES_NESTED_LOCKS
  142. typedef unique_lock<mutex> scoped_lock;
  143. typedef detail::try_lock_wrapper<mutex> scoped_try_lock;
  144. #endif
  145. };
  146. typedef mutex try_mutex;
  147. class timed_mutex
  148. {
  149. private:
  150. pthread_mutex_t m;
  151. #ifndef BOOST_PTHREAD_HAS_TIMEDLOCK
  152. pthread_cond_t cond;
  153. bool is_locked;
  154. #endif
  155. public:
  156. BOOST_THREAD_NO_COPYABLE(timed_mutex)
  157. timed_mutex()
  158. {
  159. int const res=pthread_mutex_init(&m,NULL);
  160. if(res)
  161. {
  162. boost::throw_exception(thread_resource_error(res, "boost:: timed_mutex constructor failed in pthread_mutex_init"));
  163. }
  164. #ifndef BOOST_PTHREAD_HAS_TIMEDLOCK
  165. int const res2=pthread_cond_init(&cond,NULL);
  166. if(res2)
  167. {
  168. BOOST_VERIFY(!posix::pthread_mutex_destroy(&m));
  169. //BOOST_VERIFY(!pthread_mutex_destroy(&m));
  170. boost::throw_exception(thread_resource_error(res2, "boost:: timed_mutex constructor failed in pthread_cond_init"));
  171. }
  172. is_locked=false;
  173. #endif
  174. }
  175. ~timed_mutex()
  176. {
  177. BOOST_VERIFY(!posix::pthread_mutex_destroy(&m));
  178. #ifndef BOOST_PTHREAD_HAS_TIMEDLOCK
  179. BOOST_VERIFY(!pthread_cond_destroy(&cond));
  180. #endif
  181. }
  182. #if defined BOOST_THREAD_USES_DATETIME
  183. template<typename TimeDuration>
  184. bool timed_lock(TimeDuration const & relative_time)
  185. {
  186. return timed_lock(get_system_time()+relative_time);
  187. }
  188. bool timed_lock(boost::xtime const & absolute_time)
  189. {
  190. return timed_lock(system_time(absolute_time));
  191. }
  192. #endif
  193. #ifdef BOOST_PTHREAD_HAS_TIMEDLOCK
  194. void lock()
  195. {
  196. int res = posix::pthread_mutex_lock(&m);
  197. if (res)
  198. {
  199. boost::throw_exception(lock_error(res,"boost: mutex lock failed in pthread_mutex_lock"));
  200. }
  201. }
  202. void unlock()
  203. {
  204. int res = posix::pthread_mutex_unlock(&m);
  205. (void)res;
  206. BOOST_ASSERT(res == 0);
  207. // if (res)
  208. // {
  209. // boost::throw_exception(lock_error(res,"boost: mutex unlock failed in pthread_mutex_unlock"));
  210. // }
  211. }
  212. bool try_lock()
  213. {
  214. int res;
  215. do
  216. {
  217. res = pthread_mutex_trylock(&m);
  218. } while (res == EINTR);
  219. if (res==EBUSY)
  220. {
  221. return false;
  222. }
  223. return !res;
  224. }
  225. private:
  226. bool do_try_lock_until(struct timespec const &timeout)
  227. {
  228. int const res=pthread_mutex_timedlock(&m,&timeout);
  229. BOOST_ASSERT(!res || res==ETIMEDOUT);
  230. return !res;
  231. }
  232. public:
  233. #else
  234. void lock()
  235. {
  236. boost::pthread::pthread_mutex_scoped_lock const local_lock(&m);
  237. while(is_locked)
  238. {
  239. BOOST_VERIFY(!pthread_cond_wait(&cond,&m));
  240. }
  241. is_locked=true;
  242. }
  243. void unlock()
  244. {
  245. boost::pthread::pthread_mutex_scoped_lock const local_lock(&m);
  246. is_locked=false;
  247. BOOST_VERIFY(!pthread_cond_signal(&cond));
  248. }
  249. bool try_lock()
  250. {
  251. boost::pthread::pthread_mutex_scoped_lock const local_lock(&m);
  252. if(is_locked)
  253. {
  254. return false;
  255. }
  256. is_locked=true;
  257. return true;
  258. }
  259. private:
  260. bool do_try_lock_until(struct timespec const &timeout)
  261. {
  262. boost::pthread::pthread_mutex_scoped_lock const local_lock(&m);
  263. while(is_locked)
  264. {
  265. int const cond_res=pthread_cond_timedwait(&cond,&m,&timeout);
  266. if(cond_res==ETIMEDOUT)
  267. {
  268. return false;
  269. }
  270. BOOST_ASSERT(!cond_res);
  271. }
  272. is_locked=true;
  273. return true;
  274. }
  275. public:
  276. #endif
  277. #if defined BOOST_THREAD_USES_DATETIME
  278. bool timed_lock(system_time const & abs_time)
  279. {
  280. struct timespec const ts=boost::detail::to_timespec(abs_time);
  281. return do_try_lock_until(ts);
  282. }
  283. #endif
  284. #ifdef BOOST_THREAD_USES_CHRONO
  285. template <class Rep, class Period>
  286. bool try_lock_for(const chrono::duration<Rep, Period>& rel_time)
  287. {
  288. return try_lock_until(chrono::steady_clock::now() + rel_time);
  289. }
  290. template <class Clock, class Duration>
  291. bool try_lock_until(const chrono::time_point<Clock, Duration>& t)
  292. {
  293. using namespace chrono;
  294. system_clock::time_point s_now = system_clock::now();
  295. typename Clock::time_point c_now = Clock::now();
  296. return try_lock_until(s_now + ceil<nanoseconds>(t - c_now));
  297. }
  298. template <class Duration>
  299. bool try_lock_until(const chrono::time_point<chrono::system_clock, Duration>& t)
  300. {
  301. using namespace chrono;
  302. typedef time_point<system_clock, nanoseconds> nano_sys_tmpt;
  303. return try_lock_until(nano_sys_tmpt(ceil<nanoseconds>(t.time_since_epoch())));
  304. }
  305. bool try_lock_until(const chrono::time_point<chrono::system_clock, chrono::nanoseconds>& tp)
  306. {
  307. //using namespace chrono;
  308. chrono::nanoseconds d = tp.time_since_epoch();
  309. timespec ts = boost::detail::to_timespec(d);
  310. return do_try_lock_until(ts);
  311. }
  312. #endif
  313. #define BOOST_THREAD_DEFINES_TIMED_MUTEX_NATIVE_HANDLE
  314. typedef pthread_mutex_t* native_handle_type;
  315. native_handle_type native_handle()
  316. {
  317. return &m;
  318. }
  319. #if defined BOOST_THREAD_PROVIDES_NESTED_LOCKS
  320. typedef unique_lock<timed_mutex> scoped_timed_lock;
  321. typedef detail::try_lock_wrapper<timed_mutex> scoped_try_lock;
  322. typedef scoped_timed_lock scoped_lock;
  323. #endif
  324. };
  325. }
  326. #include <boost/config/abi_suffix.hpp>
  327. #endif