// // co_spawn.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 BOOST_ASIO_CO_SPAWN_HPP #define BOOST_ASIO_CO_SPAWN_HPP #if defined(_MSC_VER) && (_MSC_VER >= 1200) # pragma once #endif // defined(_MSC_VER) && (_MSC_VER >= 1200) #include #if defined(BOOST_ASIO_HAS_CO_AWAIT) || defined(GENERATING_DOCUMENTATION) #include #include #include #include #include namespace boost { namespace asio { namespace detail { template struct awaitable_signature; template struct awaitable_signature> { typedef void type(std::exception_ptr, T); }; template struct awaitable_signature> { typedef void type(std::exception_ptr); }; } // namespace detail /// Spawn a new coroutined-based thread of execution. /** * @param ex The executor that will be used to schedule the new thread of * execution. * * @param a The boost::asio::awaitable object that is the result of calling the * coroutine's entry point function. * * @param token The @ref completion_token that will handle the notification that * the thread of execution has completed. The function signature of the * completion handler must be: * @code void handler(std::exception_ptr, T); @endcode * * @par Completion Signature * @code void(std::exception_ptr, T) @endcode * * @par Example * @code * boost::asio::awaitable echo(tcp::socket socket) * { * std::size_t bytes_transferred = 0; * * try * { * char data[1024]; * for (;;) * { * std::size_t n = co_await socket.async_read_some( * boost::asio::buffer(data), boost::asio::use_awaitable); * * co_await boost::asio::async_write(socket, * boost::asio::buffer(data, n), boost::asio::use_awaitable); * * bytes_transferred += n; * } * } * catch (const std::exception&) * { * } * * co_return bytes_transferred; * } * * // ... * * boost::asio::co_spawn(my_executor, * echo(std::move(my_tcp_socket)), * [](std::exception_ptr e, std::size_t n) * { * std::cout << "transferred " << n << "\n"; * }); * @endcode * * @par Per-Operation Cancellation * The new thread of execution is created with a cancellation state that * supports @c cancellation_type::terminal values only. To change the * cancellation state, call boost::asio::this_coro::reset_cancellation_state. */ template inline BOOST_ASIO_INITFN_AUTO_RESULT_TYPE( CompletionToken, void(std::exception_ptr, T)) co_spawn(const Executor& ex, awaitable a, CompletionToken&& token BOOST_ASIO_DEFAULT_COMPLETION_TOKEN(Executor), constraint_t< (is_executor::value || execution::is_executor::value) && is_convertible::value > = 0); /// Spawn a new coroutined-based thread of execution. /** * @param ex The executor that will be used to schedule the new thread of * execution. * * @param a The boost::asio::awaitable object that is the result of calling the * coroutine's entry point function. * * @param token The @ref completion_token that will handle the notification that * the thread of execution has completed. The function signature of the * completion handler must be: * @code void handler(std::exception_ptr); @endcode * * @par Completion Signature * @code void(std::exception_ptr) @endcode * * @par Example * @code * boost::asio::awaitable echo(tcp::socket socket) * { * try * { * char data[1024]; * for (;;) * { * std::size_t n = co_await socket.async_read_some( * boost::asio::buffer(data), boost::asio::use_awaitable); * * co_await boost::asio::async_write(socket, * boost::asio::buffer(data, n), boost::asio::use_awaitable); * } * } * catch (const std::exception& e) * { * std::cerr << "Exception: " << e.what() << "\n"; * } * } * * // ... * * boost::asio::co_spawn(my_executor, * echo(std::move(my_tcp_socket)), * boost::asio::detached); * @endcode * * @par Per-Operation Cancellation * The new thread of execution is created with a cancellation state that * supports @c cancellation_type::terminal values only. To change the * cancellation state, call boost::asio::this_coro::reset_cancellation_state. */ template inline BOOST_ASIO_INITFN_AUTO_RESULT_TYPE( CompletionToken, void(std::exception_ptr)) co_spawn(const Executor& ex, awaitable a, CompletionToken&& token BOOST_ASIO_DEFAULT_COMPLETION_TOKEN(Executor), constraint_t< (is_executor::value || execution::is_executor::value) && is_convertible::value > = 0); /// Spawn a new coroutined-based thread of execution. /** * @param ctx An execution context that will provide the executor to be used to * schedule the new thread of execution. * * @param a The boost::asio::awaitable object that is the result of calling the * coroutine's entry point function. * * @param token The @ref completion_token that will handle the notification that * the thread of execution has completed. The function signature of the * completion handler must be: * @code void handler(std::exception_ptr); @endcode * * @par Completion Signature * @code void(std::exception_ptr, T) @endcode * * @par Example * @code * boost::asio::awaitable echo(tcp::socket socket) * { * std::size_t bytes_transferred = 0; * * try * { * char data[1024]; * for (;;) * { * std::size_t n = co_await socket.async_read_some( * boost::asio::buffer(data), boost::asio::use_awaitable); * * co_await boost::asio::async_write(socket, * boost::asio::buffer(data, n), boost::asio::use_awaitable); * * bytes_transferred += n; * } * } * catch (const std::exception&) * { * } * * co_return bytes_transferred; * } * * // ... * * boost::asio::co_spawn(my_io_context, * echo(std::move(my_tcp_socket)), * [](std::exception_ptr e, std::size_t n) * { * std::cout << "transferred " << n << "\n"; * }); * @endcode * * @par Per-Operation Cancellation * The new thread of execution is created with a cancellation state that * supports @c cancellation_type::terminal values only. To change the * cancellation state, call boost::asio::this_coro::reset_cancellation_state. */ template inline BOOST_ASIO_INITFN_AUTO_RESULT_TYPE( CompletionToken, void(std::exception_ptr, T)) co_spawn(ExecutionContext& ctx, awaitable a, CompletionToken&& token BOOST_ASIO_DEFAULT_COMPLETION_TOKEN( typename ExecutionContext::executor_type), constraint_t< is_convertible::value && is_convertible::value > = 0); /// Spawn a new coroutined-based thread of execution. /** * @param ctx An execution context that will provide the executor to be used to * schedule the new thread of execution. * * @param a The boost::asio::awaitable object that is the result of calling the * coroutine's entry point function. * * @param token The @ref completion_token that will handle the notification that * the thread of execution has completed. The function signature of the * completion handler must be: * @code void handler(std::exception_ptr); @endcode * * @par Completion Signature * @code void(std::exception_ptr) @endcode * * @par Example * @code * boost::asio::awaitable echo(tcp::socket socket) * { * try * { * char data[1024]; * for (;;) * { * std::size_t n = co_await socket.async_read_some( * boost::asio::buffer(data), boost::asio::use_awaitable); * * co_await boost::asio::async_write(socket, * boost::asio::buffer(data, n), boost::asio::use_awaitable); * } * } * catch (const std::exception& e) * { * std::cerr << "Exception: " << e.what() << "\n"; * } * } * * // ... * * boost::asio::co_spawn(my_io_context, * echo(std::move(my_tcp_socket)), * boost::asio::detached); * @endcode * * @par Per-Operation Cancellation * The new thread of execution is created with a cancellation state that * supports @c cancellation_type::terminal values only. To change the * cancellation state, call boost::asio::this_coro::reset_cancellation_state. */ template inline BOOST_ASIO_INITFN_AUTO_RESULT_TYPE( CompletionToken, void(std::exception_ptr)) co_spawn(ExecutionContext& ctx, awaitable a, CompletionToken&& token BOOST_ASIO_DEFAULT_COMPLETION_TOKEN( typename ExecutionContext::executor_type), constraint_t< is_convertible::value && is_convertible::value > = 0); /// Spawn a new coroutined-based thread of execution. /** * @param ex The executor that will be used to schedule the new thread of * execution. * * @param f A nullary function object with a return type of the form * @c boost::asio::awaitable that will be used as the coroutine's entry * point. * * @param token The @ref completion_token that will handle the notification * that the thread of execution has completed. If @c R is @c void, the function * signature of the completion handler must be: * * @code void handler(std::exception_ptr); @endcode * Otherwise, the function signature of the completion handler must be: * @code void handler(std::exception_ptr, R); @endcode * * @par Completion Signature * @code void(std::exception_ptr, R) @endcode * where @c R is the first template argument to the @c awaitable returned by the * supplied function object @c F: * @code boost::asio::awaitable F() @endcode * * @par Example * @code * boost::asio::awaitable echo(tcp::socket socket) * { * std::size_t bytes_transferred = 0; * * try * { * char data[1024]; * for (;;) * { * std::size_t n = co_await socket.async_read_some( * boost::asio::buffer(data), boost::asio::use_awaitable); * * co_await boost::asio::async_write(socket, * boost::asio::buffer(data, n), boost::asio::use_awaitable); * * bytes_transferred += n; * } * } * catch (const std::exception&) * { * } * * co_return bytes_transferred; * } * * // ... * * boost::asio::co_spawn(my_executor, * [socket = std::move(my_tcp_socket)]() mutable * -> boost::asio::awaitable * { * try * { * char data[1024]; * for (;;) * { * std::size_t n = co_await socket.async_read_some( * boost::asio::buffer(data), boost::asio::use_awaitable); * * co_await boost::asio::async_write(socket, * boost::asio::buffer(data, n), boost::asio::use_awaitable); * } * } * catch (const std::exception& e) * { * std::cerr << "Exception: " << e.what() << "\n"; * } * }, boost::asio::detached); * @endcode * * @par Per-Operation Cancellation * The new thread of execution is created with a cancellation state that * supports @c cancellation_type::terminal values only. To change the * cancellation state, call boost::asio::this_coro::reset_cancellation_state. */ template >::type) CompletionToken BOOST_ASIO_DEFAULT_COMPLETION_TOKEN_TYPE(Executor)> BOOST_ASIO_INITFN_AUTO_RESULT_TYPE(CompletionToken, typename detail::awaitable_signature>::type) co_spawn(const Executor& ex, F&& f, CompletionToken&& token BOOST_ASIO_DEFAULT_COMPLETION_TOKEN(Executor), constraint_t< is_executor::value || execution::is_executor::value > = 0); /// Spawn a new coroutined-based thread of execution. /** * @param ctx An execution context that will provide the executor to be used to * schedule the new thread of execution. * * @param f A nullary function object with a return type of the form * @c boost::asio::awaitable that will be used as the coroutine's entry * point. * * @param token The @ref completion_token that will handle the notification * that the thread of execution has completed. If @c R is @c void, the function * signature of the completion handler must be: * * @code void handler(std::exception_ptr); @endcode * Otherwise, the function signature of the completion handler must be: * @code void handler(std::exception_ptr, R); @endcode * * @par Completion Signature * @code void(std::exception_ptr, R) @endcode * where @c R is the first template argument to the @c awaitable returned by the * supplied function object @c F: * @code boost::asio::awaitable F() @endcode * * @par Example * @code * boost::asio::awaitable echo(tcp::socket socket) * { * std::size_t bytes_transferred = 0; * * try * { * char data[1024]; * for (;;) * { * std::size_t n = co_await socket.async_read_some( * boost::asio::buffer(data), boost::asio::use_awaitable); * * co_await boost::asio::async_write(socket, * boost::asio::buffer(data, n), boost::asio::use_awaitable); * * bytes_transferred += n; * } * } * catch (const std::exception&) * { * } * * co_return bytes_transferred; * } * * // ... * * boost::asio::co_spawn(my_io_context, * [socket = std::move(my_tcp_socket)]() mutable * -> boost::asio::awaitable * { * try * { * char data[1024]; * for (;;) * { * std::size_t n = co_await socket.async_read_some( * boost::asio::buffer(data), boost::asio::use_awaitable); * * co_await boost::asio::async_write(socket, * boost::asio::buffer(data, n), boost::asio::use_awaitable); * } * } * catch (const std::exception& e) * { * std::cerr << "Exception: " << e.what() << "\n"; * } * }, boost::asio::detached); * @endcode * * @par Per-Operation Cancellation * The new thread of execution is created with a cancellation state that * supports @c cancellation_type::terminal values only. To change the * cancellation state, call boost::asio::this_coro::reset_cancellation_state. */ template >::type) CompletionToken BOOST_ASIO_DEFAULT_COMPLETION_TOKEN_TYPE( typename ExecutionContext::executor_type)> BOOST_ASIO_INITFN_AUTO_RESULT_TYPE(CompletionToken, typename detail::awaitable_signature>::type) co_spawn(ExecutionContext& ctx, F&& f, CompletionToken&& token BOOST_ASIO_DEFAULT_COMPLETION_TOKEN( typename ExecutionContext::executor_type), constraint_t< is_convertible::value > = 0); } // namespace asio } // namespace boost #include #include #endif // defined(BOOST_ASIO_HAS_CO_AWAIT) || defined(GENERATING_DOCUMENTATION) #endif // BOOST_ASIO_CO_SPAWN_HPP