// // MessagePack for C++ static resolution routine // // Copyright (C) 2008-2016 FURUHASHI Sadayuki and KONDO Takatoshi // // 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 MSGPACK_V1_CPP11_MSGPACK_TUPLE_HPP #define MSGPACK_V1_CPP11_MSGPACK_TUPLE_HPP #include "msgpack/v1/adaptor/detail/cpp11_msgpack_tuple_decl.hpp" #include "msgpack/adaptor/adaptor_base.hpp" #include "msgpack/pack.hpp" namespace msgpack { /// @cond MSGPACK_API_VERSION_NAMESPACE(v1) { /// @endcond namespace type { template inline tuple make_tuple(Args&&... args) { return tuple(std::forward(args)...); } template inline tuple forward_as_tuple (Args&&... args) noexcept { return tuple(std::forward(args)...); } template inline auto tuple_cat(Tuples&&... args) -> decltype( std::tuple_cat(std::forward::type::base>(args)...) ) { return std::tuple_cat(std::forward::type::base>(args)...); } template inline tuple tie(Args&... args) { return tuple(args...); } } // namespace type // --- Pack from tuple to packer stream --- template struct MsgpackTuplePacker { static void pack( msgpack::packer& o, const Tuple& v) { MsgpackTuplePacker::pack(o, v); o.pack(type::get(v)); } }; template struct MsgpackTuplePacker { static void pack ( msgpack::packer& o, const Tuple& v) { o.pack(type::get<0>(v)); } }; template struct MsgpackTuplePacker { static void pack ( msgpack::packer&, const Tuple&) { } }; namespace adaptor { template struct pack> { template msgpack::packer& operator()( msgpack::packer& o, const msgpack::type::tuple& v) const { o.pack_array(sizeof...(Args)); MsgpackTuplePacker::pack(o, v); return o; } }; } // namespace adaptor // --- Convert from tuple to object --- template struct MsgpackTupleAsImpl { static msgpack::type::tuple as(msgpack::object const& o) { return msgpack::type::tuple_cat( msgpack::type::make_tuple(o.via.array.ptr[o.via.array.size - sizeof...(Args) - 1].as()), MsgpackTupleAs::as(o)); } }; template struct MsgpackTupleAs { static msgpack::type::tuple as(msgpack::object const& o) { return MsgpackTupleAsImpl::as(o); } }; template <> struct MsgpackTupleAs<> { static msgpack::type::tuple<> as (msgpack::object const&) { return msgpack::type::tuple<>(); } }; template struct MsgpackTupleConverter { static void convert( msgpack::object const& o, Tuple& v) { MsgpackTupleConverter::convert(o, v); if (o.via.array.size >= N) o.via.array.ptr[N-1].convert(v))>::type>(type::get(v)); } }; template struct MsgpackTupleConverter { static void convert ( msgpack::object const& o, Tuple& v) { o.via.array.ptr[0].convert(v))>::type>(type::get<0>(v)); } }; template struct MsgpackTupleConverter { static void convert ( msgpack::object const&, Tuple&) { } }; namespace adaptor { template struct as, typename std::enable_if::value>::type> { msgpack::type::tuple operator()( msgpack::object const& o) const { if (o.type != msgpack::type::ARRAY) { throw msgpack::type_error(); } return MsgpackTupleAs::as(o); } }; template struct convert> { msgpack::object const& operator()( msgpack::object const& o, msgpack::type::tuple& v) const { if(o.type != msgpack::type::ARRAY) { throw msgpack::type_error(); } MsgpackTupleConverter::convert(o, v); return o; } }; } // namespace adaptor // --- Convert from tuple to object with zone --- template struct MsgpackTupleToObjectWithZone { static void convert( msgpack::object::with_zone& o, const Tuple& v) { MsgpackTupleToObjectWithZone::convert(o, v); o.via.array.ptr[N-1] = msgpack::object(type::get(v), o.zone); } }; template struct MsgpackTupleToObjectWithZone { static void convert ( msgpack::object::with_zone& o, const Tuple& v) { o.via.array.ptr[0] = msgpack::object(type::get<0>(v), o.zone); } }; template struct MsgpackTupleToObjectWithZone { static void convert ( msgpack::object::with_zone&, const Tuple&) { } }; namespace adaptor { template struct object_with_zone> { void operator()( msgpack::object::with_zone& o, msgpack::type::tuple const& v) const { o.type = msgpack::type::ARRAY; o.via.array.ptr = static_cast(o.zone.allocate_align(sizeof(msgpack::object)*sizeof...(Args))); o.via.array.size = sizeof...(Args); MsgpackTupleToObjectWithZone::convert(o, v); } }; } // namespace adaptor /// @cond } // MSGPACK_API_VERSION_NAMESPACE(v1) ///@endcond } // namespace msgpack #endif // MSGPACK_CPP11_MSGPACK_TUPLE_HPP