From e1d4890f0282ac897551d27298ebdc7a1898ac3b Mon Sep 17 00:00:00 2001 From: Christopher Kohlhoff Date: Wed, 1 Nov 2023 23:09:53 +1100 Subject: [PATCH] Add support for two channel payload signatures with c++11. --- .../experimental/detail/channel_payload.hpp | 83 ++++++++++++++++ test/experimental/channel.cpp | 94 +++++++++++++++++++ 2 files changed, 177 insertions(+) diff --git a/include/boost/asio/experimental/detail/channel_payload.hpp b/include/boost/asio/experimental/detail/channel_payload.hpp index 4ff4bf8e77..00a3fc10ed 100644 --- a/include/boost/asio/experimental/detail/channel_payload.hpp +++ b/include/boost/asio/experimental/detail/channel_payload.hpp @@ -22,6 +22,8 @@ #if defined(BOOST_ASIO_HAS_STD_VARIANT) # include +#else // defined(BOOST_ASIO_HAS_STD_VARIANT) +# include #endif // defined(BOOST_ASIO_HAS_STD_VARIANT) #include @@ -129,6 +131,87 @@ class channel_payload bool empty_; }; +template +class channel_payload +{ +public: + typedef channel_message message_1_type; + typedef channel_message message_2_type; + + channel_payload(message_1_type&& m) + : index_(1) + { + new (&storage_.message_1_) message_1_type(static_cast(m)); + } + + channel_payload(message_2_type&& m) + : index_(2) + { + new (&storage_.message_2_) message_2_type(static_cast(m)); + } + + channel_payload(channel_payload&& other) + : index_(other.index_) + { + switch (index_) + { + case 1: + new (&storage_.message_1_) message_1_type( + static_cast(other.storage_.message_1_)); + break; + case 2: + new (&storage_.message_2_) message_2_type( + static_cast(other.storage_.message_2_)); + break; + default: + break; + } + } + + ~channel_payload() + { + switch (index_) + { + case 1: + storage_.message_1_.~message_1_type(); + break; + case 2: + storage_.message_2_.~message_2_type(); + break; + default: + break; + } + } + + template + void receive(Handler& handler) + { + switch (index_) + { + case 1: + storage_.message_1_.receive(handler); + break; + case 2: + storage_.message_2_.receive(handler); + break; + default: + break; + } + } + +private: + union storage + { + storage() {} + ~storage() {} + + char dummy_; + message_1_type message_1_; + message_2_type message_2_; + } storage_; + unsigned char index_; +}; + #endif // defined(BOOST_ASIO_HAS_STD_VARIANT) } // namespace detail diff --git a/test/experimental/channel.cpp b/test/experimental/channel.cpp index eef18ecedd..31e063226b 100644 --- a/test/experimental/channel.cpp +++ b/test/experimental/channel.cpp @@ -733,6 +733,99 @@ void try_send_n_via_dispatch() BOOST_ASIO_CHECK(s3.empty()); } +struct multi_signature_handler +{ + std::string* s_; + boost::system::error_code* ec_; + + void operator()(std::string s) + { + *s_ = s; + } + + void operator()(boost::system::error_code ec) + { + *ec_ = ec; + } +}; + +void implicit_error_signature_channel_test() +{ + io_context ctx; + + channel ch1(ctx); + + BOOST_ASIO_CHECK(ch1.is_open()); + BOOST_ASIO_CHECK(!ch1.ready()); + + bool b1 = ch1.try_send("hello"); + + BOOST_ASIO_CHECK(!b1); + + std::string s1 = "abcdefghijklmnopqrstuvwxyz"; + bool b2 = ch1.try_send(std::move(s1)); + + BOOST_ASIO_CHECK(!b2); + BOOST_ASIO_CHECK(!s1.empty()); + + std::string s2; + boost::system::error_code ec1 = boost::asio::error::would_block; + multi_signature_handler h1 = {&s2, &ec1}; + ch1.async_receive(h1); + + bool b3 = ch1.try_send(std::move(s1)); + + BOOST_ASIO_CHECK(b3); + BOOST_ASIO_CHECK(s1.empty()); + + ctx.run(); + + BOOST_ASIO_CHECK(s2 == "abcdefghijklmnopqrstuvwxyz"); + BOOST_ASIO_CHECK(ec1 == boost::asio::error::would_block); + + std::string s3; + boost::system::error_code ec2; + multi_signature_handler h2 = {&s3, &ec2}; + bool b4 = ch1.try_receive(h2); + + BOOST_ASIO_CHECK(!b4); + + std::string s4 = "zyxwvutsrqponmlkjihgfedcba"; + boost::system::error_code ec3; + ch1.async_send(std::move(s4), + [&](boost::system::error_code ec) + { + ec3 = ec; + }); + + std::string s5; + boost::system::error_code ec4 = boost::asio::error::would_block; + multi_signature_handler h3 = {&s5, &ec4}; + bool b5 = ch1.try_receive(h3); + + BOOST_ASIO_CHECK(b5); + BOOST_ASIO_CHECK(ec4 == boost::asio::error::would_block); + BOOST_ASIO_CHECK(s5 == "zyxwvutsrqponmlkjihgfedcba"); + + ctx.restart(); + ctx.run(); + + BOOST_ASIO_CHECK(!ec3); + + std::string s6; + boost::system::error_code ec5 = boost::asio::error::would_block; + multi_signature_handler h4 = {&s6, &ec5}; + ch1.async_receive(h4); + + ch1.close(); + + ctx.restart(); + ctx.run(); + + BOOST_ASIO_CHECK(s6.empty()); + BOOST_ASIO_CHECK(ec5 == boost::asio::experimental::channel_errc::channel_closed); +} + BOOST_ASIO_TEST_SUITE ( "experimental/channel", @@ -753,4 +846,5 @@ BOOST_ASIO_TEST_SUITE BOOST_ASIO_TEST_CASE(buffered_executor_send) BOOST_ASIO_TEST_CASE(try_send_via_dispatch) BOOST_ASIO_TEST_CASE(try_send_n_via_dispatch) + BOOST_ASIO_TEST_CASE(implicit_error_signature_channel_test) )