make_ready_at_thread_exit_pass.cpp 4.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189
  1. //===----------------------------------------------------------------------===//
  2. //
  3. // The LLVM Compiler Infrastructure
  4. //
  5. // This file is dual licensed under the MIT and the University of Illinois Open
  6. // Source Licenses. See LICENSE.TXT for details.
  7. //
  8. //===----------------------------------------------------------------------===//
  9. // Copyright (C) 2011 Vicente J. Botet Escriba
  10. //
  11. // Distributed under the Boost Software License, Version 1.0. (See accompanying
  12. // file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
  13. // <boost/thread/future.hpp>
  14. // class packaged_task<R>
  15. // packaged_task(packaged_task&& other);
  16. #define BOOST_THREAD_VERSION 4
  17. #include <boost/thread/future.hpp>
  18. #include <boost/detail/lightweight_test.hpp>
  19. #if defined BOOST_THREAD_USES_CHRONO && \
  20. defined BOOST_THREAD_PROVIDES_SIGNATURE_PACKAGED_TASK && \
  21. defined(BOOST_THREAD_PROVIDES_VARIADIC_THREAD)
  22. class E : public std::exception
  23. {
  24. public:
  25. long data;
  26. explicit E(long i) :
  27. data(i)
  28. {
  29. }
  30. const char* what() const throw() { return ""; }
  31. ~E() throw() {}
  32. };
  33. class A
  34. {
  35. long data_;
  36. public:
  37. explicit A(long i) :
  38. data_(i)
  39. {
  40. }
  41. long operator()(long i, long j) const
  42. {
  43. if (j == 'z') BOOST_THROW_EXCEPTION( E(6) );
  44. return data_ + i + j;
  45. }
  46. };
  47. void func0_mv(BOOST_THREAD_RV_REF(boost::packaged_task<double(int, char)>) p)
  48. //void func0(boost::packaged_task<double(int, char)> p)
  49. {
  50. boost::this_thread::sleep_for(boost::chrono::milliseconds(500));
  51. p.make_ready_at_thread_exit(3, 'a');
  52. }
  53. void func0(boost::packaged_task<double(int, char)> *p)
  54. {
  55. boost::this_thread::sleep_for(boost::chrono::milliseconds(500));
  56. p->make_ready_at_thread_exit(3, 'a');
  57. }
  58. void func1(boost::packaged_task<double(int, char)> *p)
  59. {
  60. boost::this_thread::sleep_for(boost::chrono::milliseconds(500));
  61. p->make_ready_at_thread_exit(3, 'z');
  62. }
  63. void func2(boost::packaged_task<double(int, char)> *p)
  64. {
  65. p->make_ready_at_thread_exit(3, 'a');
  66. try
  67. {
  68. p->make_ready_at_thread_exit(3, 'c');
  69. }
  70. catch (const boost::future_error& e)
  71. {
  72. BOOST_TEST(e.code() == boost::system::make_error_code(boost::future_errc::promise_already_satisfied));
  73. }
  74. }
  75. void func3(boost::packaged_task<double(int, char)> *p)
  76. {
  77. try
  78. {
  79. p->make_ready_at_thread_exit(3, 'a');
  80. }
  81. catch (const boost::future_error& e)
  82. {
  83. BOOST_TEST(e.code() == boost::system::make_error_code(boost::future_errc::no_state));
  84. }
  85. }
  86. int main()
  87. {
  88. {
  89. boost::packaged_task<double(int, char)> p(A(5));
  90. boost::future<double> f = p.get_future();
  91. #if defined BOOST_THREAD_PROVIDES_VARIADIC_THREAD
  92. boost::thread(func0_mv, boost::move(p)).detach();
  93. #else
  94. boost::thread(func0, &p).detach();
  95. #endif
  96. BOOST_TEST(f.get() == 105.0);
  97. }
  98. {
  99. boost::packaged_task<double(int, char)> p2(A(5));
  100. boost::future<double> f = p2.get_future();
  101. boost::packaged_task<double(int, char)> p = boost::move(p2);
  102. boost::thread(func0, &p).detach();
  103. BOOST_TEST(f.get() == 105.0);
  104. }
  105. {
  106. boost::packaged_task<double(int, char)> p(A(5));
  107. boost::future<double> f = p.get_future();
  108. //boost::thread(func1, boost::move(p)).detach();
  109. boost::thread(func1, &p).detach();
  110. try
  111. {
  112. f.get();
  113. BOOST_TEST(false);
  114. }
  115. catch (const E& e)
  116. {
  117. BOOST_TEST(e.data == 6);
  118. }
  119. }
  120. {
  121. boost::packaged_task<double(int, char)> p2(A(5));
  122. boost::future<double> f = p2.get_future();
  123. boost::packaged_task<double(int, char)> p = boost::move(p2);
  124. boost::thread(func1, &p).detach();
  125. try
  126. {
  127. f.get();
  128. BOOST_TEST(false);
  129. }
  130. catch (const E& e)
  131. {
  132. BOOST_TEST(e.data == 6);
  133. }
  134. }
  135. {
  136. boost::packaged_task<double(int, char)> p(A(5));
  137. boost::future<double> f = p.get_future();
  138. //boost::thread(func2, boost::move(p)).detach();
  139. boost::thread(func2, &p).detach();
  140. BOOST_TEST(f.get() == 105.0);
  141. }
  142. {
  143. boost::packaged_task<double(int, char)> p2(A(5));
  144. boost::future<double> f = p2.get_future();
  145. boost::packaged_task<double(int, char)> p = boost::move(p2);
  146. boost::thread(func2, &p).detach();
  147. BOOST_TEST(f.get() == 105.0);
  148. }
  149. {
  150. boost::packaged_task<double(int, char)> p(A(5));
  151. //boost::thread t(func3, boost::move(p));
  152. boost::thread t(func3, &p);
  153. t.join();
  154. }
  155. {
  156. boost::packaged_task<double(int, char)> p2(A(5));
  157. boost::packaged_task<double(int, char)> p = boost::move(p2);
  158. boost::thread t(func3, &p);
  159. t.join();
  160. }
  161. return boost::report_errors();
  162. }
  163. #else
  164. int main()
  165. {
  166. return boost::report_errors();
  167. }
  168. //#error "Test not applicable: BOOST_THREAD_USES_CHRONO not defined for this platform as not supported"
  169. #endif