// // detail/work_dispatcher.hpp // ~~~~~~~~~~~~~~~~~~~~~~~~~~ // // Copyright (c) 2003-2024 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) // #ifndef ASIO_DETAIL_WORK_DISPATCHER_HPP #define ASIO_DETAIL_WORK_DISPATCHER_HPP #if defined(_MSC_VER) && (_MSC_VER >= 1200) # pragma once #endif // defined(_MSC_VER) && (_MSC_VER >= 1200) #include "asio/detail/config.hpp" #include "asio/detail/bind_handler.hpp" #include "asio/detail/type_traits.hpp" #include "asio/associated_executor.hpp" #include "asio/associated_allocator.hpp" #include "asio/executor_work_guard.hpp" #include "asio/execution/executor.hpp" #include "asio/execution/allocator.hpp" #include "asio/execution/blocking.hpp" #include "asio/execution/outstanding_work.hpp" #include "asio/prefer.hpp" #include "asio/detail/push_options.hpp" namespace asio { namespace detail { template <typename Handler, typename Executor, typename = void> struct is_work_dispatcher_required : true_type { }; template <typename Handler, typename Executor> struct is_work_dispatcher_required<Handler, Executor, enable_if_t< is_same< typename associated_executor<Handler, Executor>::asio_associated_executor_is_unspecialised, void >::value >> : false_type { }; template <typename Handler, typename Executor, typename = void> class work_dispatcher { public: template <typename CompletionHandler> work_dispatcher(CompletionHandler&& handler, const Executor& handler_ex) : handler_(static_cast<CompletionHandler&&>(handler)), executor_(asio::prefer(handler_ex, execution::outstanding_work.tracked)) { } work_dispatcher(const work_dispatcher& other) : handler_(other.handler_), executor_(other.executor_) { } work_dispatcher(work_dispatcher&& other) : handler_(static_cast<Handler&&>(other.handler_)), executor_(static_cast<work_executor_type&&>(other.executor_)) { } void operator()() { associated_allocator_t<Handler> alloc((get_associated_allocator)(handler_)); asio::prefer(executor_, execution::allocator(alloc)).execute( asio::detail::bind_handler( static_cast<Handler&&>(handler_))); } private: typedef decay_t< prefer_result_t<const Executor&, execution::outstanding_work_t::tracked_t > > work_executor_type; Handler handler_; work_executor_type executor_; }; #if !defined(ASIO_NO_TS_EXECUTORS) template <typename Handler, typename Executor> class work_dispatcher<Handler, Executor, enable_if_t<!execution::is_executor<Executor>::value>> { public: template <typename CompletionHandler> work_dispatcher(CompletionHandler&& handler, const Executor& handler_ex) : work_(handler_ex), handler_(static_cast<CompletionHandler&&>(handler)) { } work_dispatcher(const work_dispatcher& other) : work_(other.work_), handler_(other.handler_) { } work_dispatcher(work_dispatcher&& other) : work_(static_cast<executor_work_guard<Executor>&&>(other.work_)), handler_(static_cast<Handler&&>(other.handler_)) { } void operator()() { associated_allocator_t<Handler> alloc((get_associated_allocator)(handler_)); work_.get_executor().dispatch( asio::detail::bind_handler( static_cast<Handler&&>(handler_)), alloc); work_.reset(); } private: executor_work_guard<Executor> work_; Handler handler_; }; #endif // !defined(ASIO_NO_TS_EXECUTORS) } // namespace detail } // namespace asio #include "asio/detail/pop_options.hpp" #endif // ASIO_DETAIL_WORK_DISPATCHER_HPP