pq_single_thread_pass.cpp 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429
  1. // Copyright (C) 2014 Ian Forbed
  2. // Copyright (C) 2014,2015 Vicente J. Botet Escriba
  3. //
  4. // Distributed under the Boost Software License, Version 1.0. (See accompanying
  5. // file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
  6. //
  7. #include <boost/config.hpp>
  8. #if ! defined BOOST_NO_CXX11_DECLTYPE
  9. #define BOOST_RESULT_OF_USE_DECLTYPE
  10. #endif
  11. #define BOOST_THREAD_VERSION 4
  12. #define BOOST_THREAD_PROVIDES_EXECUTORS
  13. #include <iostream>
  14. #include <boost/thread.hpp>
  15. #include <boost/chrono.hpp>
  16. #include <boost/thread/concurrent_queues/sync_priority_queue.hpp>
  17. #include <boost/detail/lightweight_test.hpp>
  18. #include "../../../timming.hpp"
  19. using namespace boost::chrono;
  20. typedef boost::chrono::milliseconds ms;
  21. typedef boost::chrono::nanoseconds ns;
  22. typedef boost::concurrent::sync_priority_queue<int> sync_pq;
  23. class non_copyable
  24. {
  25. BOOST_THREAD_MOVABLE_ONLY(non_copyable)
  26. int val;
  27. public:
  28. non_copyable(int v) : val(v){}
  29. non_copyable(BOOST_RV_REF(non_copyable) x): val(x.val) {}
  30. non_copyable& operator=(BOOST_RV_REF(non_copyable) x) { val=x.val; return *this; }
  31. bool operator==(non_copyable const& x) const {return val==x.val;}
  32. template <typename OSTREAM>
  33. friend OSTREAM& operator <<(OSTREAM& os, non_copyable const&x )
  34. {
  35. os << x.val;
  36. return os;
  37. }
  38. bool operator <(const non_copyable& other) const
  39. {
  40. return val < other.val;
  41. }
  42. };
  43. const ms max_diff(BOOST_THREAD_TEST_TIME_MS);
  44. void test_pull_for()
  45. {
  46. sync_pq pq;
  47. steady_clock::time_point start = steady_clock::now();
  48. int val;
  49. boost::queue_op_status st = pq.pull_for(milliseconds(500), val);
  50. ns d = steady_clock::now() - start - milliseconds(500);
  51. BOOST_THREAD_TEST_IT(d, ns(max_diff));
  52. BOOST_TEST(boost::queue_op_status::timeout == st);
  53. }
  54. void test_pull_until()
  55. {
  56. sync_pq pq;
  57. steady_clock::time_point start = steady_clock::now();
  58. int val;
  59. boost::queue_op_status st = pq.pull_until(start + milliseconds(500), val);
  60. ns d = steady_clock::now() - start - milliseconds(500);
  61. BOOST_THREAD_TEST_IT(d, ns(max_diff));
  62. BOOST_TEST(boost::queue_op_status::timeout == st);
  63. }
  64. void test_nonblocking_pull()
  65. {
  66. sync_pq pq;
  67. steady_clock::time_point start = steady_clock::now();
  68. int val;
  69. boost::queue_op_status st = pq.nonblocking_pull(val);
  70. ns d = steady_clock::now() - start;
  71. BOOST_THREAD_TEST_IT(d, ns(max_diff));
  72. BOOST_TEST(boost::queue_op_status::empty == st);
  73. }
  74. void test_pull_for_when_not_empty()
  75. {
  76. sync_pq pq;
  77. pq.push(1);
  78. steady_clock::time_point start = steady_clock::now();
  79. int val;
  80. boost::queue_op_status st = pq.pull_for(milliseconds(500), val);
  81. ns d = steady_clock::now() - start;
  82. BOOST_THREAD_TEST_IT(d, ns(max_diff));
  83. BOOST_TEST(boost::queue_op_status::success == st);
  84. BOOST_TEST(1 == val);
  85. }
  86. void test_pull_until_when_not_empty()
  87. {
  88. sync_pq pq;
  89. pq.push(1);
  90. steady_clock::time_point start = steady_clock::now();
  91. int val;
  92. boost::queue_op_status st = pq.pull_until(start + milliseconds(500), val);
  93. ns d = steady_clock::now() - start;
  94. BOOST_THREAD_TEST_IT(d, ns(max_diff));
  95. BOOST_TEST(boost::queue_op_status::success == st);
  96. BOOST_TEST(1 == val);
  97. }
  98. int main()
  99. {
  100. sync_pq pq;
  101. BOOST_TEST(pq.empty());
  102. BOOST_TEST(!pq.closed());
  103. BOOST_TEST_EQ(pq.size(), std::size_t(0));
  104. for(int i = 1; i <= 5; i++){
  105. pq.push(i);
  106. BOOST_TEST(!pq.empty());
  107. BOOST_TEST_EQ(pq.size(), std::size_t(i));
  108. }
  109. for(int i = 6; i <= 10; i++){
  110. boost::queue_op_status succ = pq.try_push(i);
  111. BOOST_TEST(succ == boost::queue_op_status::success );
  112. BOOST_TEST(!pq.empty());
  113. BOOST_TEST_EQ(pq.size(), std::size_t(i));
  114. }
  115. for(int i = 10; i > 5; i--){
  116. int val = pq.pull();
  117. BOOST_TEST_EQ(val, i);
  118. }
  119. // for(int i = 5; i > 0; i--){
  120. // boost::optional<int> val = pq.try_pull();
  121. // BOOST_TEST(val);
  122. // BOOST_TEST_EQ(*val, i);
  123. // }
  124. // BOOST_TEST(pq.empty());
  125. pq.close();
  126. BOOST_TEST(pq.closed());
  127. test_pull_for();
  128. test_pull_until();
  129. test_nonblocking_pull();
  130. test_pull_for_when_not_empty();
  131. //test_pull_until_when_not_empty();
  132. #if ! defined BOOST_NO_CXX11_RVALUE_REFERENCES
  133. {
  134. // empty queue try_push rvalue/non-copyable succeeds
  135. boost::concurrent::sync_priority_queue<non_copyable> q;
  136. BOOST_TEST(boost::queue_op_status::success ==q.try_push(non_copyable(1)));
  137. BOOST_TEST(! q.empty());
  138. BOOST_TEST(! q.full());
  139. BOOST_TEST_EQ(q.size(), 1u);
  140. BOOST_TEST(! q.closed());
  141. }
  142. {
  143. //fixme
  144. // empty queue try_push rvalue/non-copyable succeeds
  145. boost::concurrent::sync_priority_queue<non_copyable> q;
  146. non_copyable nc(1);
  147. BOOST_TEST(boost::queue_op_status::success == q.try_push(boost::move(nc)));
  148. BOOST_TEST(! q.empty());
  149. BOOST_TEST(! q.full());
  150. BOOST_TEST_EQ(q.size(), 1u);
  151. BOOST_TEST(! q.closed());
  152. }
  153. #endif
  154. {
  155. // empty queue try_push lvalue succeeds
  156. boost::concurrent::sync_priority_queue<int> q;
  157. int i=1;
  158. BOOST_TEST(boost::queue_op_status::success == q.try_push(i));
  159. BOOST_TEST(! q.empty());
  160. BOOST_TEST(! q.full());
  161. BOOST_TEST_EQ(q.size(), 1u);
  162. BOOST_TEST(! q.closed());
  163. }
  164. #if 0
  165. {
  166. // empty queue try_push rvalue succeeds
  167. boost::concurrent::sync_priority_queue<int> q;
  168. BOOST_TEST(boost::queue_op_status::success == q.nonblocking_push(1));
  169. BOOST_TEST(! q.empty());
  170. BOOST_TEST(! q.full());
  171. BOOST_TEST_EQ(q.size(), 1u);
  172. BOOST_TEST(! q.closed());
  173. }
  174. {
  175. // empty queue nonblocking_push rvalue/non-copyable succeeds
  176. boost::concurrent::sync_priority_queue<non_copyable> q;
  177. BOOST_TEST(boost::queue_op_status::success == q.nonblocking_push(non_copyable(1)));
  178. BOOST_TEST(! q.empty());
  179. BOOST_TEST(! q.full());
  180. BOOST_TEST_EQ(q.size(), 1u);
  181. BOOST_TEST(! q.closed());
  182. }
  183. {
  184. // empty queue nonblocking_push rvalue/non-copyable succeeds
  185. boost::concurrent::sync_priority_queue<non_copyable> q;
  186. non_copyable nc(1);
  187. BOOST_TEST(boost::queue_op_status::success == q.nonblocking_push(boost::move(nc)));
  188. BOOST_TEST(! q.empty());
  189. BOOST_TEST(! q.full());
  190. BOOST_TEST_EQ(q.size(), 1u);
  191. BOOST_TEST(! q.closed());
  192. }
  193. #endif
  194. {
  195. // 1-element queue pull succeed
  196. boost::concurrent::sync_priority_queue<int> q;
  197. q.push(1);
  198. int i;
  199. i=q.pull();
  200. BOOST_TEST_EQ(i, 1);
  201. BOOST_TEST(q.empty());
  202. BOOST_TEST(! q.full());
  203. BOOST_TEST_EQ(q.size(), 0u);
  204. BOOST_TEST(! q.closed());
  205. }
  206. #if ! defined BOOST_NO_CXX11_RVALUE_REFERENCES
  207. {
  208. // 1-element queue pull succeed
  209. boost::concurrent::sync_priority_queue<non_copyable> q;
  210. non_copyable nc1(1);
  211. q.push(boost::move(nc1));
  212. non_copyable nc2(2);
  213. nc2=q.pull();
  214. BOOST_TEST_EQ(nc1, nc2);
  215. BOOST_TEST(q.empty());
  216. BOOST_TEST(! q.full());
  217. BOOST_TEST_EQ(q.size(), 0u);
  218. BOOST_TEST(! q.closed());
  219. }
  220. #endif
  221. {
  222. // 1-element queue pull succeed
  223. boost::concurrent::sync_priority_queue<int> q;
  224. q.push(1);
  225. int i = q.pull();
  226. BOOST_TEST_EQ(i, 1);
  227. BOOST_TEST(q.empty());
  228. BOOST_TEST(! q.full());
  229. BOOST_TEST_EQ(q.size(), 0u);
  230. BOOST_TEST(! q.closed());
  231. }
  232. #if ! defined BOOST_NO_CXX11_RVALUE_REFERENCES
  233. {
  234. // 1-element queue pull succeed
  235. boost::concurrent::sync_priority_queue<non_copyable> q;
  236. non_copyable nc1(1);
  237. q.push(boost::move(nc1));
  238. non_copyable nc = q.pull();
  239. BOOST_TEST_EQ(nc, nc1);
  240. BOOST_TEST(q.empty());
  241. BOOST_TEST(! q.full());
  242. BOOST_TEST_EQ(q.size(), 0u);
  243. BOOST_TEST(! q.closed());
  244. }
  245. #endif
  246. {
  247. // 1-element queue try_pull succeed
  248. boost::concurrent::sync_priority_queue<int> q;
  249. q.push(1);
  250. int i;
  251. BOOST_TEST(boost::queue_op_status::success == q.try_pull(i));
  252. BOOST_TEST_EQ(i, 1);
  253. BOOST_TEST(q.empty());
  254. BOOST_TEST(! q.full());
  255. BOOST_TEST_EQ(q.size(), 0u);
  256. BOOST_TEST(! q.closed());
  257. }
  258. #if ! defined BOOST_NO_CXX11_RVALUE_REFERENCES
  259. {
  260. // 1-element queue try_pull succeed
  261. boost::concurrent::sync_priority_queue<non_copyable> q;
  262. non_copyable nc1(1);
  263. q.push(boost::move(nc1));
  264. non_copyable nc(2);
  265. BOOST_TEST(boost::queue_op_status::success == q.try_pull(nc));
  266. BOOST_TEST_EQ(nc, nc1);
  267. BOOST_TEST(q.empty());
  268. BOOST_TEST(! q.full());
  269. BOOST_TEST_EQ(q.size(), 0u);
  270. BOOST_TEST(! q.closed());
  271. }
  272. #endif
  273. {
  274. // 1-element queue nonblocking_pull succeed
  275. boost::concurrent::sync_priority_queue<int> q;
  276. q.push(1);
  277. int i;
  278. BOOST_TEST(boost::queue_op_status::success == q.nonblocking_pull(i));
  279. BOOST_TEST_EQ(i, 1);
  280. BOOST_TEST(q.empty());
  281. BOOST_TEST(! q.full());
  282. BOOST_TEST_EQ(q.size(), 0u);
  283. BOOST_TEST(! q.closed());
  284. }
  285. #if ! defined BOOST_NO_CXX11_RVALUE_REFERENCES
  286. {
  287. // 1-element queue nonblocking_pull succeed
  288. boost::concurrent::sync_priority_queue<non_copyable> q;
  289. non_copyable nc1(1);
  290. q.push(boost::move(nc1));
  291. non_copyable nc(2);
  292. BOOST_TEST(boost::queue_op_status::success == q.nonblocking_pull(nc));
  293. BOOST_TEST_EQ(nc, nc1);
  294. BOOST_TEST(q.empty());
  295. BOOST_TEST(! q.full());
  296. BOOST_TEST_EQ(q.size(), 0u);
  297. BOOST_TEST(! q.closed());
  298. }
  299. {
  300. // 1-element queue wait_pull succeed
  301. boost::concurrent::sync_priority_queue<non_copyable> q;
  302. non_copyable nc1(1);
  303. q.push(boost::move(nc1));
  304. non_copyable nc(2);
  305. BOOST_TEST(boost::queue_op_status::success == q.wait_pull(nc));
  306. BOOST_TEST_EQ(nc, nc1);
  307. BOOST_TEST(q.empty());
  308. BOOST_TEST(! q.full());
  309. BOOST_TEST_EQ(q.size(), 0u);
  310. BOOST_TEST(! q.closed());
  311. }
  312. #endif
  313. {
  314. // 1-element queue wait_pull succeed
  315. boost::concurrent::sync_priority_queue<int> q;
  316. q.push(1);
  317. int i;
  318. BOOST_TEST(boost::queue_op_status::success == q.wait_pull(i));
  319. BOOST_TEST_EQ(i, 1);
  320. BOOST_TEST(q.empty());
  321. BOOST_TEST(! q.full());
  322. BOOST_TEST_EQ(q.size(), 0u);
  323. BOOST_TEST(! q.closed());
  324. }
  325. #if ! defined BOOST_NO_CXX11_RVALUE_REFERENCES
  326. {
  327. // 1-element queue wait_pull succeed
  328. boost::concurrent::sync_priority_queue<non_copyable> q;
  329. non_copyable nc1(1);
  330. q.push(boost::move(nc1));
  331. non_copyable nc(2);
  332. BOOST_TEST(boost::queue_op_status::success == q.wait_pull(nc));
  333. BOOST_TEST_EQ(nc, nc1);
  334. BOOST_TEST(q.empty());
  335. BOOST_TEST(! q.full());
  336. BOOST_TEST_EQ(q.size(), 0u);
  337. BOOST_TEST(! q.closed());
  338. }
  339. #endif
  340. {
  341. // closed invariants
  342. boost::concurrent::sync_priority_queue<int> q;
  343. q.close();
  344. BOOST_TEST(q.empty());
  345. BOOST_TEST(! q.full());
  346. BOOST_TEST_EQ(q.size(), 0u);
  347. BOOST_TEST(q.closed());
  348. }
  349. {
  350. // closed queue push fails
  351. boost::concurrent::sync_priority_queue<int> q;
  352. q.close();
  353. try {
  354. q.push(1);
  355. BOOST_TEST(false); // fixme
  356. } catch (...) {
  357. BOOST_TEST(q.empty());
  358. BOOST_TEST(! q.full());
  359. BOOST_TEST_EQ(q.size(), 0u);
  360. BOOST_TEST(q.closed());
  361. }
  362. }
  363. {
  364. // 1-element closed queue pull succeed
  365. boost::concurrent::sync_priority_queue<int> q;
  366. q.push(1);
  367. q.close();
  368. int i;
  369. i=q.pull();
  370. BOOST_TEST_EQ(i, 1);
  371. BOOST_TEST(q.empty());
  372. BOOST_TEST(! q.full());
  373. BOOST_TEST_EQ(q.size(), 0u);
  374. BOOST_TEST(q.closed());
  375. }
  376. {
  377. // 1-element closed queue wait_pull succeed
  378. boost::concurrent::sync_priority_queue<int> q;
  379. q.push(1);
  380. q.close();
  381. int i;
  382. BOOST_TEST(boost::queue_op_status::success == q.wait_pull(i));
  383. BOOST_TEST_EQ(i, 1);
  384. BOOST_TEST(q.empty());
  385. BOOST_TEST(! q.full());
  386. BOOST_TEST_EQ(q.size(), 0u);
  387. BOOST_TEST(q.closed());
  388. }
  389. {
  390. // closed empty queue wait_pull fails
  391. boost::concurrent::sync_priority_queue<int> q;
  392. q.close();
  393. BOOST_TEST(q.empty());
  394. BOOST_TEST(q.closed());
  395. int i;
  396. BOOST_TEST(boost::queue_op_status::closed == q.wait_pull(i));
  397. BOOST_TEST(q.empty());
  398. BOOST_TEST(q.closed());
  399. }
  400. return boost::report_errors();
  401. }