test_shared_mutex_timed_locks.cpp 9.1 KB

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