test_shared_mutex_timed_locks_chrono.cpp 9.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259
  1. // (C) Copyright 2006-7 Anthony Williams
  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. #define BOOST_THREAD_VERSION 2
  6. #define BOOST_TEST_MODULE Boost.Threads: shared_mutex_locks_chrono test suite
  7. #include <boost/test/unit_test.hpp>
  8. #include <boost/thread/thread.hpp>
  9. #include <boost/thread/shared_mutex.hpp>
  10. #include "./util.inl"
  11. #include "./shared_mutex_locking_thread.hpp"
  12. #if defined BOOST_THREAD_USES_CHRONO
  13. #define CHECK_LOCKED_VALUE_EQUAL(mutex_name,value,expected_value) \
  14. { \
  15. boost::unique_lock<boost::mutex> lock(mutex_name); \
  16. BOOST_CHECK_EQUAL(value,expected_value); \
  17. }
  18. BOOST_AUTO_TEST_CASE(test_timed_lock_shared_times_out_if_write_lock_held)
  19. {
  20. boost::shared_mutex rw_mutex;
  21. boost::mutex finish_mutex;
  22. boost::mutex unblocked_mutex;
  23. unsigned unblocked_count=0;
  24. boost::unique_lock<boost::mutex> finish_lock(finish_mutex);
  25. boost::thread writer(simple_writing_thread(rw_mutex,finish_mutex,unblocked_mutex,unblocked_count));
  26. boost::this_thread::sleep_for(boost::chrono::seconds(1));
  27. CHECK_LOCKED_VALUE_EQUAL(unblocked_mutex,unblocked_count,1u);
  28. boost::chrono::steady_clock::time_point const start=boost::chrono::steady_clock::now();
  29. boost::chrono::steady_clock::time_point const timeout=start+boost::chrono::milliseconds(500);
  30. boost::chrono::milliseconds const timeout_resolution(50);
  31. bool timed_lock_succeeded=rw_mutex.try_lock_shared_until(timeout);
  32. BOOST_CHECK((timeout-timeout_resolution)<boost::chrono::steady_clock::now());
  33. BOOST_CHECK(!timed_lock_succeeded);
  34. if(timed_lock_succeeded)
  35. {
  36. rw_mutex.unlock_shared();
  37. }
  38. boost::chrono::milliseconds const wait_duration(500);
  39. boost::chrono::steady_clock::time_point const timeout2=boost::chrono::steady_clock::now()+wait_duration;
  40. timed_lock_succeeded=rw_mutex.try_lock_shared_for(wait_duration);
  41. BOOST_CHECK((timeout2-timeout_resolution)<boost::chrono::steady_clock::now());
  42. BOOST_CHECK(!timed_lock_succeeded);
  43. if(timed_lock_succeeded)
  44. {
  45. rw_mutex.unlock_shared();
  46. }
  47. finish_lock.unlock();
  48. writer.join();
  49. }
  50. BOOST_AUTO_TEST_CASE(test_timed_lock_shared_succeeds_if_no_lock_held)
  51. {
  52. boost::shared_mutex rw_mutex;
  53. boost::mutex finish_mutex;
  54. boost::mutex unblocked_mutex;
  55. boost::chrono::steady_clock::time_point const start=boost::chrono::steady_clock::now();
  56. boost::chrono::steady_clock::time_point const timeout=start+boost::chrono::milliseconds(500);
  57. bool timed_lock_succeeded=rw_mutex.try_lock_shared_until(timeout);
  58. BOOST_CHECK(boost::chrono::steady_clock::now()<timeout);
  59. BOOST_CHECK(timed_lock_succeeded);
  60. if(timed_lock_succeeded)
  61. {
  62. rw_mutex.unlock_shared();
  63. }
  64. boost::chrono::milliseconds const wait_duration(500);
  65. boost::chrono::steady_clock::time_point const timeout2=boost::chrono::steady_clock::now()+wait_duration;
  66. timed_lock_succeeded=rw_mutex.try_lock_shared_for(wait_duration);
  67. BOOST_CHECK(boost::chrono::steady_clock::now()<timeout2);
  68. BOOST_CHECK(timed_lock_succeeded);
  69. if(timed_lock_succeeded)
  70. {
  71. rw_mutex.unlock_shared();
  72. }
  73. }
  74. BOOST_AUTO_TEST_CASE(test_timed_lock_shared_succeeds_if_read_lock_held)
  75. {
  76. boost::shared_mutex rw_mutex;
  77. boost::mutex finish_mutex;
  78. boost::mutex unblocked_mutex;
  79. unsigned unblocked_count=0;
  80. boost::unique_lock<boost::mutex> finish_lock(finish_mutex);
  81. boost::thread reader(simple_reading_thread(rw_mutex,finish_mutex,unblocked_mutex,unblocked_count));
  82. boost::this_thread::sleep_for(boost::chrono::seconds(1));
  83. CHECK_LOCKED_VALUE_EQUAL(unblocked_mutex,unblocked_count,1u);
  84. boost::chrono::steady_clock::time_point const start=boost::chrono::steady_clock::now();
  85. boost::chrono::steady_clock::time_point const timeout=start+boost::chrono::milliseconds(500);
  86. bool timed_lock_succeeded=rw_mutex.try_lock_shared_until(timeout);
  87. BOOST_CHECK(boost::chrono::steady_clock::now()<timeout);
  88. BOOST_CHECK(timed_lock_succeeded);
  89. if(timed_lock_succeeded)
  90. {
  91. rw_mutex.unlock_shared();
  92. }
  93. boost::chrono::milliseconds const wait_duration(500);
  94. boost::chrono::steady_clock::time_point const timeout2=boost::chrono::steady_clock::now()+wait_duration;
  95. timed_lock_succeeded=rw_mutex.try_lock_shared_for(wait_duration);
  96. BOOST_CHECK(boost::chrono::steady_clock::now()<timeout2);
  97. BOOST_CHECK(timed_lock_succeeded);
  98. if(timed_lock_succeeded)
  99. {
  100. rw_mutex.unlock_shared();
  101. }
  102. finish_lock.unlock();
  103. reader.join();
  104. }
  105. BOOST_AUTO_TEST_CASE(test_timed_lock_times_out_if_write_lock_held)
  106. {
  107. boost::shared_mutex rw_mutex;
  108. boost::mutex finish_mutex;
  109. boost::mutex unblocked_mutex;
  110. unsigned unblocked_count=0;
  111. boost::unique_lock<boost::mutex> finish_lock(finish_mutex);
  112. boost::thread writer(simple_writing_thread(rw_mutex,finish_mutex,unblocked_mutex,unblocked_count));
  113. boost::this_thread::sleep_for(boost::chrono::seconds(1));
  114. CHECK_LOCKED_VALUE_EQUAL(unblocked_mutex,unblocked_count,1u);
  115. boost::chrono::steady_clock::time_point const start=boost::chrono::steady_clock::now();
  116. boost::chrono::steady_clock::time_point const timeout=start+boost::chrono::milliseconds(500);
  117. boost::chrono::milliseconds const timeout_resolution(50);
  118. bool timed_lock_succeeded=rw_mutex.try_lock_until(timeout);
  119. BOOST_CHECK((timeout-timeout_resolution)<boost::chrono::steady_clock::now());
  120. BOOST_CHECK(!timed_lock_succeeded);
  121. if(timed_lock_succeeded)
  122. {
  123. rw_mutex.unlock();
  124. }
  125. boost::chrono::milliseconds const wait_duration(500);
  126. boost::chrono::steady_clock::time_point const timeout2=boost::chrono::steady_clock::now()+wait_duration;
  127. timed_lock_succeeded=rw_mutex.try_lock_for(wait_duration);
  128. BOOST_CHECK((timeout2-timeout_resolution)<boost::chrono::steady_clock::now());
  129. BOOST_CHECK(!timed_lock_succeeded);
  130. if(timed_lock_succeeded)
  131. {
  132. rw_mutex.unlock();
  133. }
  134. finish_lock.unlock();
  135. writer.join();
  136. }
  137. BOOST_AUTO_TEST_CASE(test_timed_lock_succeeds_if_no_lock_held)
  138. {
  139. boost::shared_mutex rw_mutex;
  140. boost::mutex finish_mutex;
  141. boost::mutex unblocked_mutex;
  142. boost::chrono::steady_clock::time_point const start=boost::chrono::steady_clock::now();
  143. boost::chrono::steady_clock::time_point const timeout=start+boost::chrono::milliseconds(500);
  144. bool timed_lock_succeeded=rw_mutex.try_lock_until(timeout);
  145. BOOST_CHECK(boost::chrono::steady_clock::now()<timeout);
  146. BOOST_CHECK(timed_lock_succeeded);
  147. if(timed_lock_succeeded)
  148. {
  149. rw_mutex.unlock();
  150. }
  151. boost::chrono::milliseconds const wait_duration(500);
  152. boost::chrono::steady_clock::time_point const timeout2=boost::chrono::steady_clock::now()+wait_duration;
  153. timed_lock_succeeded=rw_mutex.try_lock_for(wait_duration);
  154. BOOST_CHECK(boost::chrono::steady_clock::now()<timeout2);
  155. BOOST_CHECK(timed_lock_succeeded);
  156. if(timed_lock_succeeded)
  157. {
  158. rw_mutex.unlock();
  159. }
  160. }
  161. BOOST_AUTO_TEST_CASE(test_timed_lock_times_out_if_read_lock_held)
  162. {
  163. boost::shared_mutex rw_mutex;
  164. boost::mutex finish_mutex;
  165. boost::mutex unblocked_mutex;
  166. unsigned unblocked_count=0;
  167. boost::unique_lock<boost::mutex> finish_lock(finish_mutex);
  168. boost::thread reader(simple_reading_thread(rw_mutex,finish_mutex,unblocked_mutex,unblocked_count));
  169. boost::this_thread::sleep_for(boost::chrono::seconds(1));
  170. CHECK_LOCKED_VALUE_EQUAL(unblocked_mutex,unblocked_count,1u);
  171. boost::chrono::steady_clock::time_point const start=boost::chrono::steady_clock::now();
  172. boost::chrono::steady_clock::time_point const timeout=start+boost::chrono::milliseconds(500);
  173. boost::chrono::milliseconds const timeout_resolution(50);
  174. bool timed_lock_succeeded=rw_mutex.try_lock_until(timeout);
  175. BOOST_CHECK((timeout-timeout_resolution)<boost::chrono::steady_clock::now());
  176. BOOST_CHECK(!timed_lock_succeeded);
  177. if(timed_lock_succeeded)
  178. {
  179. rw_mutex.unlock();
  180. }
  181. boost::chrono::milliseconds const wait_duration(500);
  182. boost::chrono::steady_clock::time_point const timeout2=boost::chrono::steady_clock::now()+wait_duration;
  183. timed_lock_succeeded=rw_mutex.try_lock_for(wait_duration);
  184. BOOST_CHECK((timeout2-timeout_resolution)<boost::chrono::steady_clock::now());
  185. BOOST_CHECK(!timed_lock_succeeded);
  186. if(timed_lock_succeeded)
  187. {
  188. rw_mutex.unlock();
  189. }
  190. finish_lock.unlock();
  191. reader.join();
  192. }
  193. BOOST_AUTO_TEST_CASE(test_timed_lock_times_out_but_read_lock_succeeds_if_read_lock_held)
  194. {
  195. boost::shared_mutex rw_mutex;
  196. boost::mutex finish_mutex;
  197. boost::mutex unblocked_mutex;
  198. unsigned unblocked_count=0;
  199. boost::unique_lock<boost::mutex> finish_lock(finish_mutex);
  200. boost::thread reader(simple_reading_thread(rw_mutex,finish_mutex,unblocked_mutex,unblocked_count));
  201. boost::this_thread::sleep_for(boost::chrono::seconds(1));
  202. CHECK_LOCKED_VALUE_EQUAL(unblocked_mutex,unblocked_count,1u);
  203. boost::chrono::steady_clock::time_point const start=boost::chrono::steady_clock::now();
  204. boost::chrono::steady_clock::time_point const timeout=start+boost::chrono::milliseconds(500);
  205. bool timed_lock_succeeded=rw_mutex.try_lock_until(timeout);
  206. BOOST_CHECK(!timed_lock_succeeded);
  207. if(timed_lock_succeeded)
  208. {
  209. rw_mutex.unlock();
  210. }
  211. boost::chrono::milliseconds const wait_duration(500);
  212. timed_lock_succeeded=rw_mutex.try_lock_shared_for(wait_duration);
  213. BOOST_CHECK(timed_lock_succeeded);
  214. if(timed_lock_succeeded)
  215. {
  216. rw_mutex.unlock_shared();
  217. }
  218. finish_lock.unlock();
  219. reader.join();
  220. }
  221. #else
  222. #error "Test not applicable: BOOST_THREAD_USES_CHRONO not defined for this platform as not supported"
  223. #endif