// // detail/completion_payload.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 ASIO_DETAIL_COMPLETION_PAYLOAD_HPP #define ASIO_DETAIL_COMPLETION_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/detail/completion_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 detail { template class completion_payload; template class completion_payload { public: explicit completion_payload(completion_message) { } template void receive(Handler& handler) { static_cast(handler)(); } }; template class completion_payload { public: completion_payload(completion_message&& m) : message_(static_cast&&>(m)) { } template void receive(Handler& handler) { message_.receive(handler); } private: completion_message message_; }; #if defined(ASIO_HAS_STD_VARIANT) template class completion_payload { public: template completion_payload(completion_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 completion_payload { public: typedef completion_message void_message_type; typedef completion_message error_message_type; completion_payload(void_message_type&&) : message_(0, asio::error_code()), empty_(true) { } completion_payload(error_message_type&& m) : message_(static_cast(m)), empty_(false) { } template void receive(Handler& handler) { if (empty_) completion_message(0).receive(handler); else message_.receive(handler); } private: error_message_type message_; bool empty_; }; template class completion_payload { public: typedef completion_message message_1_type; typedef completion_message message_2_type; completion_payload(message_1_type&& m) : index_(1) { new (&storage_.message_1_) message_1_type(static_cast(m)); } completion_payload(message_2_type&& m) : index_(2) { new (&storage_.message_2_) message_2_type(static_cast(m)); } completion_payload(completion_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; } } ~completion_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 asio #include "asio/detail/pop_options.hpp" #endif // ASIO_DETAIL_COMPLETION_PAYLOAD_HPP