// // bind_immediate_executor.cpp // ~~~~~~~~~~~~~~~~~~~~~~~~~~~ // // Copyright (c) 2003-2025 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) // // Disable autolinking for unit tests. #if !defined(BOOST_ALL_NO_LIB) #define BOOST_ALL_NO_LIB 1 #endif // !defined(BOOST_ALL_NO_LIB) // Test that header file is self-contained. #include #include #include #include #include #include "unit_test.hpp" using namespace boost::asio; namespace bindns = std; void increment(int* count) { ++(*count); } void bind_immediate_executor_to_function_object_test() { io_context ioc1; io_context ioc2; int count = 0; async_immediate(ioc1, bind_immediate_executor( ioc2.get_executor(), bindns::bind(&increment, &count))); ioc1.run(); BOOST_ASIO_CHECK(count == 0); ioc2.run(); BOOST_ASIO_CHECK(count == 1); async_immediate(ioc1.get_executor(), bind_immediate_executor( ioc2.get_executor(), bind_immediate_executor( boost::asio::inline_executor(), bindns::bind(&increment, &count)))); ioc1.restart(); ioc1.run(); BOOST_ASIO_CHECK(count == 1); ioc2.restart(); ioc2.run(); BOOST_ASIO_CHECK(count == 2); } struct incrementer_token_v1 { explicit incrementer_token_v1(int* c) : count(c) {} int* count; }; struct incrementer_handler_v1 { explicit incrementer_handler_v1(incrementer_token_v1 t) : count(t.count) {} void operator()(){ increment(count); } int* count; }; namespace boost { namespace asio { template <> class async_result { public: typedef incrementer_handler_v1 completion_handler_type; typedef void return_type; explicit async_result(completion_handler_type&) {} return_type get() {} }; } // namespace asio } // namespace boost void bind_immediate_executor_to_completion_token_v1_test() { io_context ioc1; io_context ioc2; int count = 0; async_immediate(ioc1, bind_immediate_executor( ioc2.get_executor(), incrementer_token_v1(&count))); ioc1.run(); BOOST_ASIO_CHECK(count == 0); ioc2.run(); BOOST_ASIO_CHECK(count == 1); } struct incrementer_token_v2 { explicit incrementer_token_v2(int* c) : count(c) {} int* count; }; namespace boost { namespace asio { template <> class async_result { public: #if !defined(BOOST_ASIO_HAS_RETURN_TYPE_DEDUCTION) typedef void return_type; #endif // !defined(BOOST_ASIO_HAS_RETURN_TYPE_DEDUCTION) template static void initiate(Initiation initiation, incrementer_token_v2 token, Args&&... args) { initiation(bindns::bind(&increment, token.count), static_cast(args)...); } }; } // namespace asio } // namespace boost void bind_immediate_executor_to_completion_token_v2_test() { io_context ioc1; io_context ioc2; int count = 0; async_immediate(ioc1, bind_immediate_executor( ioc2.get_executor(), incrementer_token_v2(&count))); ioc1.run(); BOOST_ASIO_CHECK(count == 0); ioc2.run(); BOOST_ASIO_CHECK(count == 1); } void partial_bind_immediate_executor_test() { io_context ioc1; io_context ioc2; int count = 0; async_immediate(ioc1, bind_immediate_executor(ioc2.get_executor()))( bindns::bind(&increment, &count)); ioc1.run(); BOOST_ASIO_CHECK(count == 0); ioc2.run(); BOOST_ASIO_CHECK(count == 1); async_immediate(ioc1, bind_immediate_executor(ioc2.get_executor()))( incrementer_token_v2(&count)); ioc1.restart(); ioc1.run(); BOOST_ASIO_CHECK(count == 1); ioc2.restart(); ioc2.run(); BOOST_ASIO_CHECK(count == 2); } BOOST_ASIO_TEST_SUITE ( "bind_immediate_executor", BOOST_ASIO_TEST_CASE(bind_immediate_executor_to_function_object_test) BOOST_ASIO_TEST_CASE(bind_immediate_executor_to_completion_token_v1_test) BOOST_ASIO_TEST_CASE(bind_immediate_executor_to_completion_token_v2_test) BOOST_ASIO_TEST_CASE(partial_bind_immediate_executor_test) )