src/cxx_supportlib/vendor-modified/boost/asio/impl/post.hpp in passenger-6.0.7 vs src/cxx_supportlib/vendor-modified/boost/asio/impl/post.hpp in passenger-6.0.8

- old
+ new

@@ -1,10 +1,10 @@ // // impl/post.hpp // ~~~~~~~~~~~~~ // -// Copyright (c) 2003-2018 Christopher M. Kohlhoff (chris at kohlhoff dot com) +// Copyright (c) 2003-2020 Christopher M. Kohlhoff (chris at kohlhoff dot com) // // Distributed under the Boost Software License, Version 1.0. (See accompanying // file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt) // @@ -17,59 +17,230 @@ #include <boost/asio/detail/config.hpp> #include <boost/asio/associated_allocator.hpp> #include <boost/asio/associated_executor.hpp> #include <boost/asio/detail/work_dispatcher.hpp> +#include <boost/asio/execution/allocator.hpp> +#include <boost/asio/execution/blocking.hpp> +#include <boost/asio/execution/relationship.hpp> +#include <boost/asio/prefer.hpp> +#include <boost/asio/require.hpp> #include <boost/asio/detail/push_options.hpp> namespace boost { namespace asio { +namespace detail { -template <typename CompletionToken> -BOOST_ASIO_INITFN_RESULT_TYPE(CompletionToken, void()) post( - BOOST_ASIO_MOVE_ARG(CompletionToken) token) +class initiate_post { - typedef BOOST_ASIO_HANDLER_TYPE(CompletionToken, void()) handler; +public: + template <typename CompletionHandler> + void operator()(BOOST_ASIO_MOVE_ARG(CompletionHandler) handler, + typename enable_if< + execution::is_executor< + typename associated_executor< + typename decay<CompletionHandler>::type + >::type + >::value + >::type* = 0) const + { + typedef typename decay<CompletionHandler>::type handler_t; - async_completion<CompletionToken, void()> init(token); + typename associated_executor<handler_t>::type ex( + (get_associated_executor)(handler)); - typename associated_executor<handler>::type ex( - (get_associated_executor)(init.completion_handler)); + typename associated_allocator<handler_t>::type alloc( + (get_associated_allocator)(handler)); - typename associated_allocator<handler>::type alloc( - (get_associated_allocator)(init.completion_handler)); + execution::execute( + boost::asio::prefer( + boost::asio::require(ex, execution::blocking.never), + execution::relationship.fork, + execution::allocator(alloc)), + BOOST_ASIO_MOVE_CAST(CompletionHandler)(handler)); + } - ex.post(BOOST_ASIO_MOVE_CAST(handler)(init.completion_handler), alloc); + template <typename CompletionHandler> + void operator()(BOOST_ASIO_MOVE_ARG(CompletionHandler) handler, + typename enable_if< + !execution::is_executor< + typename associated_executor< + typename decay<CompletionHandler>::type + >::type + >::value + >::type* = 0) const + { + typedef typename decay<CompletionHandler>::type handler_t; - return init.result.get(); -} + typename associated_executor<handler_t>::type ex( + (get_associated_executor)(handler)); -template <typename Executor, typename CompletionToken> -BOOST_ASIO_INITFN_RESULT_TYPE(CompletionToken, void()) post( - const Executor& ex, BOOST_ASIO_MOVE_ARG(CompletionToken) token, - typename enable_if<is_executor<Executor>::value>::type*) + typename associated_allocator<handler_t>::type alloc( + (get_associated_allocator)(handler)); + + ex.post(BOOST_ASIO_MOVE_CAST(CompletionHandler)(handler), alloc); + } +}; + +template <typename Executor> +class initiate_post_with_executor { - typedef BOOST_ASIO_HANDLER_TYPE(CompletionToken, void()) handler; +public: + typedef Executor executor_type; - async_completion<CompletionToken, void()> init(token); + explicit initiate_post_with_executor(const Executor& ex) + : ex_(ex) + { + } - typename associated_allocator<handler>::type alloc( - (get_associated_allocator)(init.completion_handler)); + executor_type get_executor() const BOOST_ASIO_NOEXCEPT + { + return ex_; + } - ex.post(detail::work_dispatcher<handler>(init.completion_handler), alloc); + template <typename CompletionHandler> + void operator()(BOOST_ASIO_MOVE_ARG(CompletionHandler) handler, + typename enable_if< + execution::is_executor< + typename conditional<true, executor_type, CompletionHandler>::type + >::value + && + !detail::is_work_dispatcher_required< + typename decay<CompletionHandler>::type, + Executor + >::value + >::type* = 0) const + { + typedef typename decay<CompletionHandler>::type handler_t; - return init.result.get(); + typename associated_allocator<handler_t>::type alloc( + (get_associated_allocator)(handler)); + + execution::execute( + boost::asio::prefer( + boost::asio::require(ex_, execution::blocking.never), + execution::relationship.fork, + execution::allocator(alloc)), + BOOST_ASIO_MOVE_CAST(CompletionHandler)(handler)); + } + + template <typename CompletionHandler> + void operator()(BOOST_ASIO_MOVE_ARG(CompletionHandler) handler, + typename enable_if< + execution::is_executor< + typename conditional<true, executor_type, CompletionHandler>::type + >::value + && + detail::is_work_dispatcher_required< + typename decay<CompletionHandler>::type, + Executor + >::value + >::type* = 0) const + { + typedef typename decay<CompletionHandler>::type handler_t; + + typedef typename associated_executor< + handler_t, Executor>::type handler_ex_t; + handler_ex_t handler_ex((get_associated_executor)(handler, ex_)); + + typename associated_allocator<handler_t>::type alloc( + (get_associated_allocator)(handler)); + + execution::execute( + boost::asio::prefer( + boost::asio::require(ex_, execution::blocking.never), + execution::relationship.fork, + execution::allocator(alloc)), + detail::work_dispatcher<handler_t, handler_ex_t>( + BOOST_ASIO_MOVE_CAST(CompletionHandler)(handler), handler_ex)); + } + + template <typename CompletionHandler> + void operator()(BOOST_ASIO_MOVE_ARG(CompletionHandler) handler, + typename enable_if< + !execution::is_executor< + typename conditional<true, executor_type, CompletionHandler>::type + >::value + && + !detail::is_work_dispatcher_required< + typename decay<CompletionHandler>::type, + Executor + >::value + >::type* = 0) const + { + typedef typename decay<CompletionHandler>::type handler_t; + + typename associated_allocator<handler_t>::type alloc( + (get_associated_allocator)(handler)); + + ex_.post(BOOST_ASIO_MOVE_CAST(CompletionHandler)(handler), alloc); + } + + template <typename CompletionHandler> + void operator()(BOOST_ASIO_MOVE_ARG(CompletionHandler) handler, + typename enable_if< + !execution::is_executor< + typename conditional<true, executor_type, CompletionHandler>::type + >::value + && + detail::is_work_dispatcher_required< + typename decay<CompletionHandler>::type, + Executor + >::value + >::type* = 0) const + { + typedef typename decay<CompletionHandler>::type handler_t; + + typedef typename associated_executor< + handler_t, Executor>::type handler_ex_t; + handler_ex_t handler_ex((get_associated_executor)(handler, ex_)); + + typename associated_allocator<handler_t>::type alloc( + (get_associated_allocator)(handler)); + + ex_.post(detail::work_dispatcher<handler_t, handler_ex_t>( + BOOST_ASIO_MOVE_CAST(CompletionHandler)(handler), + handler_ex), alloc); + } + +private: + Executor ex_; +}; + +} // namespace detail + +template <BOOST_ASIO_COMPLETION_TOKEN_FOR(void()) CompletionToken> +BOOST_ASIO_INITFN_AUTO_RESULT_TYPE(CompletionToken, void()) post( + BOOST_ASIO_MOVE_ARG(CompletionToken) token) +{ + return async_initiate<CompletionToken, void()>( + detail::initiate_post(), token); } -template <typename ExecutionContext, typename CompletionToken> -inline BOOST_ASIO_INITFN_RESULT_TYPE(CompletionToken, void()) post( +template <typename Executor, + BOOST_ASIO_COMPLETION_TOKEN_FOR(void()) CompletionToken> +BOOST_ASIO_INITFN_AUTO_RESULT_TYPE(CompletionToken, void()) post( + const Executor& ex, BOOST_ASIO_MOVE_ARG(CompletionToken) token, + typename enable_if< + execution::is_executor<Executor>::value || is_executor<Executor>::value + >::type*) +{ + return async_initiate<CompletionToken, void()>( + detail::initiate_post_with_executor<Executor>(ex), token); +} + +template <typename ExecutionContext, + BOOST_ASIO_COMPLETION_TOKEN_FOR(void()) CompletionToken> +inline BOOST_ASIO_INITFN_AUTO_RESULT_TYPE(CompletionToken, void()) post( ExecutionContext& ctx, BOOST_ASIO_MOVE_ARG(CompletionToken) token, typename enable_if<is_convertible< ExecutionContext&, execution_context&>::value>::type*) { - return (post)(ctx.get_executor(), - BOOST_ASIO_MOVE_CAST(CompletionToken)(token)); + return async_initiate<CompletionToken, void()>( + detail::initiate_post_with_executor< + typename ExecutionContext::executor_type>( + ctx.get_executor()), token); } } // namespace asio } // namespace boost