use_future.hpp 4.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174
  1. //
  2. // impl/use_future.hpp
  3. // ~~~~~~~~~~~~~~~~~~~
  4. //
  5. // Copyright (c) 2003-2015 Christopher M. Kohlhoff (chris at kohlhoff dot com)
  6. //
  7. // Distributed under the Boost Software License, Version 1.0. (See accompanying
  8. // file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
  9. //
  10. #ifndef BOOST_ASIO_IMPL_USE_FUTURE_HPP
  11. #define BOOST_ASIO_IMPL_USE_FUTURE_HPP
  12. #if defined(_MSC_VER) && (_MSC_VER >= 1200)
  13. # pragma once
  14. #endif // defined(_MSC_VER) && (_MSC_VER >= 1200)
  15. #include <boost/asio/detail/config.hpp>
  16. #include <future>
  17. #include <boost/asio/async_result.hpp>
  18. #include <boost/system/error_code.hpp>
  19. #include <boost/asio/handler_type.hpp>
  20. #include <boost/system/system_error.hpp>
  21. #include <boost/asio/detail/push_options.hpp>
  22. namespace boost {
  23. namespace asio {
  24. namespace detail {
  25. // Completion handler to adapt a promise as a completion handler.
  26. template <typename T>
  27. class promise_handler
  28. {
  29. public:
  30. // Construct from use_future special value.
  31. template <typename Allocator>
  32. promise_handler(use_future_t<Allocator> uf)
  33. : promise_(std::allocate_shared<std::promise<T> >(
  34. uf.get_allocator(), std::allocator_arg, uf.get_allocator()))
  35. {
  36. }
  37. void operator()(T t)
  38. {
  39. promise_->set_value(t);
  40. }
  41. void operator()(const boost::system::error_code& ec, T t)
  42. {
  43. if (ec)
  44. promise_->set_exception(
  45. std::make_exception_ptr(
  46. boost::system::system_error(ec)));
  47. else
  48. promise_->set_value(t);
  49. }
  50. //private:
  51. std::shared_ptr<std::promise<T> > promise_;
  52. };
  53. // Completion handler to adapt a void promise as a completion handler.
  54. template <>
  55. class promise_handler<void>
  56. {
  57. public:
  58. // Construct from use_future special value. Used during rebinding.
  59. template <typename Allocator>
  60. promise_handler(use_future_t<Allocator> uf)
  61. : promise_(std::allocate_shared<std::promise<void> >(
  62. uf.get_allocator(), std::allocator_arg, uf.get_allocator()))
  63. {
  64. }
  65. void operator()()
  66. {
  67. promise_->set_value();
  68. }
  69. void operator()(const boost::system::error_code& ec)
  70. {
  71. if (ec)
  72. promise_->set_exception(
  73. std::make_exception_ptr(
  74. boost::system::system_error(ec)));
  75. else
  76. promise_->set_value();
  77. }
  78. //private:
  79. std::shared_ptr<std::promise<void> > promise_;
  80. };
  81. // Ensure any exceptions thrown from the handler are propagated back to the
  82. // caller via the future.
  83. template <typename Function, typename T>
  84. void asio_handler_invoke(Function f, promise_handler<T>* h)
  85. {
  86. std::shared_ptr<std::promise<T> > p(h->promise_);
  87. try
  88. {
  89. f();
  90. }
  91. catch (...)
  92. {
  93. p->set_exception(std::current_exception());
  94. }
  95. }
  96. } // namespace detail
  97. #if !defined(GENERATING_DOCUMENTATION)
  98. // Handler traits specialisation for promise_handler.
  99. template <typename T>
  100. class async_result<detail::promise_handler<T> >
  101. {
  102. public:
  103. // The initiating function will return a future.
  104. typedef std::future<T> type;
  105. // Constructor creates a new promise for the async operation, and obtains the
  106. // corresponding future.
  107. explicit async_result(detail::promise_handler<T>& h)
  108. {
  109. value_ = h.promise_->get_future();
  110. }
  111. // Obtain the future to be returned from the initiating function.
  112. type get() { return std::move(value_); }
  113. private:
  114. type value_;
  115. };
  116. // Handler type specialisation for use_future.
  117. template <typename Allocator, typename ReturnType>
  118. struct handler_type<use_future_t<Allocator>, ReturnType()>
  119. {
  120. typedef detail::promise_handler<void> type;
  121. };
  122. // Handler type specialisation for use_future.
  123. template <typename Allocator, typename ReturnType, typename Arg1>
  124. struct handler_type<use_future_t<Allocator>, ReturnType(Arg1)>
  125. {
  126. typedef detail::promise_handler<Arg1> type;
  127. };
  128. // Handler type specialisation for use_future.
  129. template <typename Allocator, typename ReturnType>
  130. struct handler_type<use_future_t<Allocator>,
  131. ReturnType(boost::system::error_code)>
  132. {
  133. typedef detail::promise_handler<void> type;
  134. };
  135. // Handler type specialisation for use_future.
  136. template <typename Allocator, typename ReturnType, typename Arg2>
  137. struct handler_type<use_future_t<Allocator>,
  138. ReturnType(boost::system::error_code, Arg2)>
  139. {
  140. typedef detail::promise_handler<Arg2> type;
  141. };
  142. #endif // !defined(GENERATING_DOCUMENTATION)
  143. } // namespace asio
  144. } // namespace boost
  145. #include <boost/asio/detail/pop_options.hpp>
  146. #endif // BOOST_ASIO_IMPL_USE_FUTURE_HPP