// // connect.cpp // ~~~~~~~~~~~~~ // // Copyright (c) 2003-2023 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 "../unit_test.hpp" #if !defined(BOOST_ASIO_NO_DEPRECATED) namespace exec = boost::asio::execution; static int call_count = 0; struct operation_state { void start() BOOST_ASIO_NOEXCEPT { } }; namespace boost { namespace asio { namespace traits { #if !defined(BOOST_ASIO_HAS_DEDUCED_START_MEMBER_TRAIT) template <> struct start_member { BOOST_ASIO_STATIC_CONSTEXPR(bool, is_valid = true); BOOST_ASIO_STATIC_CONSTEXPR(bool, is_noexcept = true); typedef void result_type; }; #endif // !defined(BOOST_ASIO_HAS_DEDUCED_START_MEMBER_TRAIT) } // namespace traits } // namespace asio } // namespace boost struct no_connect_1 { }; struct no_connect_2 : exec::sender_base { }; struct no_connect_3 { template operation_state connect(BOOST_ASIO_MOVE_ARG(R) r) { (void)r; return operation_state(); } }; #if !defined(BOOST_ASIO_HAS_DEDUCED_CONNECT_MEMBER_TRAIT) namespace boost { namespace asio { namespace traits { template struct connect_member { BOOST_ASIO_STATIC_CONSTEXPR(bool, is_valid = true); BOOST_ASIO_STATIC_CONSTEXPR(bool, is_noexcept = false); typedef operation_state result_type; }; } // namespace traits } // namespace asio } // namespace boost #endif // !defined(BOOST_ASIO_HAS_DEDUCED_CONNECT_MEMBER_TRAIT) struct const_member_connect : exec::sender_base { const_member_connect() { } template operation_state connect(BOOST_ASIO_MOVE_ARG(R) r) const { (void)r; ++call_count; return operation_state(); } }; #if !defined(BOOST_ASIO_HAS_DEDUCED_CONNECT_MEMBER_TRAIT) namespace boost { namespace asio { namespace traits { template struct connect_member { BOOST_ASIO_STATIC_CONSTEXPR(bool, is_valid = true); BOOST_ASIO_STATIC_CONSTEXPR(bool, is_noexcept = false); typedef operation_state result_type; }; } // namespace traits } // namespace asio } // namespace boost #endif // !defined(BOOST_ASIO_HAS_DEDUCED_CONNECT_MEMBER_TRAIT) struct free_connect_const_receiver : exec::sender_base { free_connect_const_receiver() { } template friend operation_state connect( const free_connect_const_receiver&, BOOST_ASIO_MOVE_ARG(R) r) { (void)r; ++call_count; return operation_state(); } }; #if !defined(BOOST_ASIO_HAS_DEDUCED_CONNECT_FREE_TRAIT) namespace boost { namespace asio { namespace traits { template struct connect_free { BOOST_ASIO_STATIC_CONSTEXPR(bool, is_valid = true); BOOST_ASIO_STATIC_CONSTEXPR(bool, is_noexcept = false); typedef operation_state result_type; }; } // namespace traits } // namespace asio } // namespace boost #endif // !defined(BOOST_ASIO_HAS_DEDUCED_CONNECT_FREE_TRAIT) struct non_const_member_connect : exec::sender_base { template operation_state connect(BOOST_ASIO_MOVE_ARG(R) r) { (void)r; ++call_count; return operation_state(); } }; #if !defined(BOOST_ASIO_HAS_DEDUCED_CONNECT_MEMBER_TRAIT) namespace boost { namespace asio { namespace traits { template struct connect_member { BOOST_ASIO_STATIC_CONSTEXPR(bool, is_valid = true); BOOST_ASIO_STATIC_CONSTEXPR(bool, is_noexcept = false); typedef operation_state result_type; }; } // namespace traits } // namespace asio } // namespace boost #endif // !defined(BOOST_ASIO_HAS_DEDUCED_CONNECT_MEMBER_TRAIT) struct free_connect_non_const_receiver : exec::sender_base { free_connect_non_const_receiver() { } template friend operation_state connect( free_connect_non_const_receiver&, BOOST_ASIO_MOVE_ARG(R) r) { (void)r; ++call_count; return operation_state(); } }; #if !defined(BOOST_ASIO_HAS_DEDUCED_CONNECT_FREE_TRAIT) namespace boost { namespace asio { namespace traits { template struct connect_free { BOOST_ASIO_STATIC_CONSTEXPR(bool, is_valid = true); BOOST_ASIO_STATIC_CONSTEXPR(bool, is_noexcept = false); typedef operation_state result_type; }; } // namespace traits } // namespace asio } // namespace boost #endif // !defined(BOOST_ASIO_HAS_DEDUCED_CONNECT_FREE_TRAIT) struct receiver { receiver() { } receiver(const receiver&) { } #if defined(BOOST_ASIO_HAS_MOVE) receiver(receiver&&) BOOST_ASIO_NOEXCEPT { } #endif // defined(BOOST_ASIO_HAS_MOVE) template void set_error(BOOST_ASIO_MOVE_ARG(E) e) BOOST_ASIO_NOEXCEPT { (void)e; } void set_done() BOOST_ASIO_NOEXCEPT { } }; namespace boost { namespace asio { namespace traits { #if !defined(BOOST_ASIO_HAS_DEDUCED_SET_ERROR_MEMBER_TRAIT) template struct set_error_member { BOOST_ASIO_STATIC_CONSTEXPR(bool, is_valid = true); BOOST_ASIO_STATIC_CONSTEXPR(bool, is_noexcept = true); typedef void result_type; }; #endif // !defined(BOOST_ASIO_HAS_DEDUCED_SET_ERROR_MEMBER_TRAIT) #if !defined(BOOST_ASIO_HAS_DEDUCED_SET_DONE_MEMBER_TRAIT) template <> struct set_done_member { BOOST_ASIO_STATIC_CONSTEXPR(bool, is_valid = true); BOOST_ASIO_STATIC_CONSTEXPR(bool, is_noexcept = true); typedef void result_type; }; #endif // !defined(BOOST_ASIO_HAS_DEDUCED_SET_DONE_MEMBER_TRAIT) } // namespace traits } // namespace asio } // namespace boost struct executor { executor() { } executor(const executor&) BOOST_ASIO_NOEXCEPT { } #if defined(BOOST_ASIO_HAS_MOVE) executor(executor&&) BOOST_ASIO_NOEXCEPT { } #endif // defined(BOOST_ASIO_HAS_MOVE) template void execute(BOOST_ASIO_MOVE_ARG(F) f) const BOOST_ASIO_NOEXCEPT { (void)f; } bool operator==(const executor&) const BOOST_ASIO_NOEXCEPT { return true; } bool operator!=(const executor&) const BOOST_ASIO_NOEXCEPT { return false; } }; namespace boost { namespace asio { namespace traits { #if !defined(BOOST_ASIO_HAS_DEDUCED_EXECUTE_MEMBER_TRAIT) template struct execute_member { BOOST_ASIO_STATIC_CONSTEXPR(bool, is_valid = true); BOOST_ASIO_STATIC_CONSTEXPR(bool, is_noexcept = true); typedef void result_type; }; #endif // !defined(BOOST_ASIO_HAS_DEDUCED_SET_ERROR_MEMBER_TRAIT) #if !defined(BOOST_ASIO_HAS_DEDUCED_EQUALITY_COMPARABLE_TRAIT) template <> struct equality_comparable { BOOST_ASIO_STATIC_CONSTEXPR(bool, is_valid = true); BOOST_ASIO_STATIC_CONSTEXPR(bool, is_noexcept = true); }; #endif // !defined(BOOST_ASIO_HAS_DEDUCED_EQUALITY_COMPARABLE_TRAIT) } // namespace traits } // namespace asio } // namespace boost void test_can_connect() { BOOST_ASIO_CONSTEXPR bool b1 = exec::can_connect< no_connect_1&, receiver>::value; BOOST_ASIO_CHECK(b1 == false); BOOST_ASIO_CONSTEXPR bool b2 = exec::can_connect< const no_connect_1&, receiver>::value; BOOST_ASIO_CHECK(b2 == false); BOOST_ASIO_CONSTEXPR bool b3 = exec::can_connect< no_connect_2&, receiver>::value; BOOST_ASIO_CHECK(b3 == false); BOOST_ASIO_CONSTEXPR bool b4 = exec::can_connect< const no_connect_2&, receiver>::value; BOOST_ASIO_CHECK(b4 == false); BOOST_ASIO_CONSTEXPR bool b5 = exec::can_connect< no_connect_3&, receiver>::value; BOOST_ASIO_CHECK(b5 == false); BOOST_ASIO_CONSTEXPR bool b6 = exec::can_connect< const no_connect_3&, receiver>::value; BOOST_ASIO_CHECK(b6 == false); BOOST_ASIO_CONSTEXPR bool b7 = exec::can_connect< const_member_connect&, receiver>::value; BOOST_ASIO_CHECK(b7 == true); BOOST_ASIO_CONSTEXPR bool b8 = exec::can_connect< const const_member_connect&, receiver>::value; BOOST_ASIO_CHECK(b8 == true); BOOST_ASIO_CONSTEXPR bool b9 = exec::can_connect< free_connect_const_receiver&, receiver>::value; BOOST_ASIO_CHECK(b9 == true); BOOST_ASIO_CONSTEXPR bool b10 = exec::can_connect< const free_connect_const_receiver&, receiver>::value; BOOST_ASIO_CHECK(b10 == true); BOOST_ASIO_CONSTEXPR bool b11 = exec::can_connect< non_const_member_connect&, receiver>::value; BOOST_ASIO_CHECK(b11 == true); BOOST_ASIO_CONSTEXPR bool b12 = exec::can_connect< const non_const_member_connect&, receiver>::value; BOOST_ASIO_CHECK(b12 == false); BOOST_ASIO_CONSTEXPR bool b13 = exec::can_connect< free_connect_non_const_receiver&, receiver>::value; BOOST_ASIO_CHECK(b13 == true); BOOST_ASIO_CONSTEXPR bool b14 = exec::can_connect< const free_connect_non_const_receiver&, receiver>::value; BOOST_ASIO_CHECK(b14 == false); BOOST_ASIO_CONSTEXPR bool b15 = exec::can_connect< executor&, receiver>::value; BOOST_ASIO_CHECK(b15 == true); BOOST_ASIO_CONSTEXPR bool b16 = exec::can_connect< const executor&, receiver>::value; BOOST_ASIO_CHECK(b16 == true); } void increment(int* count) { ++(*count); } void test_connect() { receiver r; call_count = 0; const_member_connect s1; operation_state o1 = exec::connect(s1, r); BOOST_ASIO_CHECK(call_count == 1); (void)o1; call_count = 0; const const_member_connect s2; operation_state o2 = exec::connect(s2, r); BOOST_ASIO_CHECK(call_count == 1); (void)o2; call_count = 0; operation_state o3 = exec::connect(const_member_connect(), r); BOOST_ASIO_CHECK(call_count == 1); (void)o3; call_count = 0; free_connect_const_receiver s3; operation_state o4 = exec::connect(s3, r); BOOST_ASIO_CHECK(call_count == 1); (void)o4; call_count = 0; const free_connect_const_receiver s4; operation_state o5 = exec::connect(s4, r); BOOST_ASIO_CHECK(call_count == 1); (void)o5; call_count = 0; operation_state o6 = exec::connect(free_connect_const_receiver(), r); BOOST_ASIO_CHECK(call_count == 1); (void)o6; call_count = 0; non_const_member_connect s5; operation_state o7 = exec::connect(s5, r); BOOST_ASIO_CHECK(call_count == 1); (void)o7; call_count = 0; free_connect_non_const_receiver s6; operation_state o8 = exec::connect(s6, r); BOOST_ASIO_CHECK(call_count == 1); (void)o8; executor s7; exec::connect_result::type o9 = exec::connect(s7, r); BOOST_ASIO_CHECK(( exec::is_operation_state< exec::connect_result::type >::value)); (void)o9; const executor s8; exec::connect_result::type o10 = exec::connect(s8, r); (void)exec::connect(s8, r); BOOST_ASIO_CHECK(( exec::is_operation_state< exec::connect_result::type >::value)); (void)o10; } BOOST_ASIO_TEST_SUITE ( "connect", BOOST_ASIO_TEST_CASE(test_can_connect) BOOST_ASIO_TEST_CASE(test_connect) ) #else // !defined(BOOST_ASIO_NO_DEPRECATED) BOOST_ASIO_TEST_SUITE ( "connect", BOOST_ASIO_TEST_CASE(null_test) ) #endif // !defined(BOOST_ASIO_NO_DEPRECATED)