then_executor_pass.cpp 3.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152
  1. // Copyright (C) 2014 Vicente Botet
  2. //
  3. // Distributed under the Boost Software License, Version 1.0. (See accompanying
  4. // file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
  5. // <boost/thread/future.hpp>
  6. // class future<R>
  7. // template<typename F>
  8. // auto then(F&& func) -> future<decltype(func(*this))>;
  9. #define BOOST_THREAD_VERSION 5
  10. //#define BOOST_THREAD_USES_LOG
  11. #define BOOST_THREAD_USES_LOG_THREAD_ID
  12. #include <boost/thread/detail/log.hpp>
  13. #include <boost/thread/future.hpp>
  14. #include <boost/thread/executors/basic_thread_pool.hpp>
  15. #include <boost/thread/executor.hpp>
  16. #include <boost/detail/lightweight_test.hpp>
  17. #include <cassert>
  18. #if defined BOOST_THREAD_PROVIDES_FUTURE_CONTINUATION
  19. #ifdef BOOST_MSVC
  20. #pragma warning(disable: 4127) // conditional expression is constant
  21. #endif
  22. int p1()
  23. {
  24. BOOST_THREAD_LOG << "p1 < " << BOOST_THREAD_END_LOG;
  25. boost::this_thread::sleep_for(boost::chrono::milliseconds(500));
  26. BOOST_THREAD_LOG << "p1 >" << BOOST_THREAD_END_LOG;
  27. return 1;
  28. }
  29. int p2(boost::future<int> f)
  30. {
  31. assert(f.is_ready());
  32. BOOST_THREAD_LOG << "p2 <" << &f << BOOST_THREAD_END_LOG;
  33. BOOST_TEST(f.valid());
  34. int i = f.get();
  35. boost::this_thread::sleep_for(boost::chrono::milliseconds(500));
  36. BOOST_THREAD_LOG << "p2 <" << &f << BOOST_THREAD_END_LOG;
  37. return 2 * i;
  38. }
  39. void p3(boost::future<int> f)
  40. {
  41. assert(f.is_ready());
  42. BOOST_THREAD_LOG << "p3 <" << &f << BOOST_THREAD_END_LOG;
  43. BOOST_TEST(f.valid());
  44. int i = f.get();
  45. boost::this_thread::sleep_for(boost::chrono::milliseconds(500));
  46. BOOST_THREAD_LOG << "p3 <" << &f << " " <<i << BOOST_THREAD_END_LOG;
  47. return;
  48. }
  49. int main()
  50. {
  51. BOOST_THREAD_LOG << BOOST_THREAD_END_LOG;
  52. {
  53. boost::basic_thread_pool ex(1);
  54. boost::future<int> f1 = boost::async(boost::launch::async, &p1);
  55. BOOST_TEST(f1.valid());
  56. boost::future<int> f2 = f1.then(ex, &p2);
  57. BOOST_TEST(f2.valid());
  58. BOOST_TEST(! f1.valid());
  59. try
  60. {
  61. BOOST_TEST(f2.get()==2);
  62. }
  63. catch (std::exception& ex)
  64. {
  65. BOOST_THREAD_LOG << "ERRORRRRR "<<ex.what() << "" << BOOST_THREAD_END_LOG;
  66. BOOST_TEST(false);
  67. }
  68. catch (...)
  69. {
  70. BOOST_THREAD_LOG << " ERRORRRRR exception thrown" << BOOST_THREAD_END_LOG;
  71. BOOST_TEST(false);
  72. }
  73. }
  74. BOOST_THREAD_LOG << BOOST_THREAD_END_LOG;
  75. {
  76. boost::basic_thread_pool ex(1);
  77. boost::future<int> f1 = boost::async(boost::launch::async, &p1);
  78. BOOST_TEST(f1.valid());
  79. boost::future<void> f2 = f1.then(ex, &p3);
  80. BOOST_TEST(f2.valid());
  81. try
  82. {
  83. f2.wait();
  84. }
  85. catch (std::exception& ex)
  86. {
  87. BOOST_THREAD_LOG << "ERRORRRRR "<<ex.what() << "" << BOOST_THREAD_END_LOG;
  88. BOOST_TEST(false);
  89. }
  90. catch (...)
  91. {
  92. BOOST_THREAD_LOG << " ERRORRRRR exception thrown" << BOOST_THREAD_END_LOG;
  93. BOOST_TEST(false);
  94. }
  95. }
  96. BOOST_THREAD_LOG << BOOST_THREAD_END_LOG;
  97. {
  98. boost::basic_thread_pool ex(1);
  99. boost::future<int> f2 = boost::async(p1).then(ex, &p2);
  100. BOOST_TEST(f2.get()==2);
  101. }
  102. BOOST_THREAD_LOG << BOOST_THREAD_END_LOG;
  103. {
  104. boost::basic_thread_pool ex(1);
  105. boost::future<int> f1 = boost::async(p1);
  106. boost::future<int> f21 = f1.then(ex, &p2);
  107. boost::future<int> f2= f21.then(ex, &p2);
  108. BOOST_TEST(f2.get()==4);
  109. }
  110. BOOST_THREAD_LOG << BOOST_THREAD_END_LOG;
  111. {
  112. boost::basic_thread_pool ex(1);
  113. boost::future<int> f1 = boost::async(p1);
  114. boost::future<int> f21 = f1.then(ex, &p2);
  115. boost::future<int> f2= f21.then(&p2);
  116. BOOST_TEST(f2.get()==4);
  117. }
  118. BOOST_THREAD_LOG << BOOST_THREAD_END_LOG;
  119. {
  120. boost::basic_thread_pool ex(1);
  121. boost::future<int> f1 = boost::async(p1);
  122. boost::future<int> f2= f1.then(&p2).then(ex, &p2);
  123. BOOST_TEST(f2.get()==4);
  124. }
  125. BOOST_THREAD_LOG << BOOST_THREAD_END_LOG;
  126. {
  127. boost::basic_thread_pool ex(1);
  128. boost::future<int> f2 = boost::async(p1).then(ex, &p2).then(ex, &p2);
  129. BOOST_TEST(f2.get()==4);
  130. }
  131. return boost::report_errors();
  132. }
  133. #else
  134. int main()
  135. {
  136. return 0;
  137. }
  138. #endif