// // experimental/detail/channel_payload.hpp // ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ // // 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) // #ifndef ASIO_EXPERIMENTAL_DETAIL_CHANNEL_PAYLOAD_HPP #define ASIO_EXPERIMENTAL_DETAIL_CHANNEL_PAYLOAD_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/type_traits.hpp" #include "asio/error_code.hpp" #include "asio/experimental/detail/channel_message.hpp" #if defined(ASIO_HAS_STD_VARIANT) # include #else // defined(ASIO_HAS_STD_VARIANT) # include #endif // defined(ASIO_HAS_STD_VARIANT) #include "asio/detail/push_options.hpp" namespace asio { namespace experimental { namespace detail { template class channel_payload; template class channel_payload { public: explicit channel_payload(channel_message) { } template void receive(Handler& handler) { static_cast(handler)(); } }; template class channel_payload { public: channel_payload(channel_message&& m) : message_(static_cast&&>(m)) { } template void receive(Handler& handler) { message_.receive(handler); } private: channel_message message_; }; #if defined(ASIO_HAS_STD_VARIANT) template class channel_payload { public: template channel_payload(channel_message&& m) : message_(static_cast&&>(m)) { } template void receive(Handler& handler) { std::visit( [&](auto& message) { message.receive(handler); }, message_); } private: std::variant...> message_; }; #else // defined(ASIO_HAS_STD_VARIANT) template class channel_payload { public: typedef channel_message void_message_type; typedef channel_message error_message_type; channel_payload(void_message_type&&) : message_(0, asio::error_code()), empty_(true) { } channel_payload(error_message_type&& m) : message_(static_cast(m)), empty_(false) { } template void receive(Handler& handler) { if (empty_) channel_message(0).receive(handler); else message_.receive(handler); } private: error_message_type message_; 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(ASIO_HAS_STD_VARIANT) } // namespace detail } // namespace experimental } // namespace asio #include "asio/detail/pop_options.hpp" #endif // ASIO_EXPERIMENTAL_DETAIL_CHANNEL_PAYLOAD_HPP