// // MessagePack for C++ static resolution routine // // Copyright (C) 2008-2013 FURUHASHI Sadayuki and KONDO Takatoshi // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. // #ifndef MSGPACK_TYPE_TUPLE_HPP #define MSGPACK_TYPE_TUPLE_HPP #include #include "msgpack/object.hpp" #include "msgpack/cpp_config.hpp" namespace msgpack { namespace type { // tuple using std::get; using std::tuple_size; using std::tuple_element; using std::uses_allocator; using std::ignore; template< class... Types > class tuple : public std::tuple { public: using std::tuple::tuple; template< std::size_t I> typename tuple_element >::type& get() { return std::get(*this); } template< std::size_t I> typename tuple_element >::type const& get() const { return std::get(*this); } template< std::size_t I> typename tuple_element >::type&& get() && { return std::get(*this); } }; template< class... Types > tuple make_tuple( Types&&... args ) { return tuple(std::forward(args)...); } template< class... Types > tuple make_tuple( std::tuple&& arg ) { return tuple(std::forward&&>(arg)); } template< class... Types > tuple tie( Types&... args ) { return std::tie(args...); } template< class... Types > tuple forward_as_tuple( Types&&... args ) { return std::forward_as_tuple(std::forward(args)...); } namespace detail { template < typename... Types > std::tuple&& get_std_tuple(tuple&& t) { return std::forward&&>(t); } template < typename... Types > std::tuple& get_std_tuple(tuple& t) { return t; } template < typename... Types > std::tuple const& get_std_tuple(tuple const& t) { return t; } template < typename T > T&& get_std_tuple(T&& t) { return t; } } template< class... Tuples > auto tuple_cat(Tuples&&... args) -> decltype( msgpack::type::make_tuple(std::tuple_cat(detail::get_std_tuple(std::forward(args))...)) ) { return std::tuple_cat(detail::get_std_tuple(std::forward(args))...); } template< class... Types > void swap( tuple& lhs, tuple& rhs ) { lhs.swap(rhs); } } // namespace type // --- Pack ( from tuple to packer stream --- template struct Packer { static void pack( packer& o, const Tuple& v) { Packer::pack(o, v); o.pack(type::get(v)); } }; template struct Packer { static void pack ( packer& o, const Tuple& v) { o.pack(type::get<0>(v)); } }; template const packer& operator<< ( packer& o, const type::tuple& v) { o.pack_array(sizeof...(Args)); Packer::pack(o, v); return o; } // --- Convert from tuple to object --- template struct Converter { static void convert( object const& o, Tuple& v) { Converter::convert(o, v); o.via.array.ptr[N-1].convert(v))>::type>(type::get(v)); } }; template struct Converter { static void convert ( object const& o, Tuple& v) { o.via.array.ptr[0].convert(v))>::type>(type::get<0>(v)); } }; template type::tuple& operator>> ( object const& o, type::tuple& v) { if(o.type != type::ARRAY) { throw type_error(); } if(o.via.array.size < sizeof...(Args)) { throw type_error(); } Converter::convert(o, v); return v; } // --- Convert from tuple to object with zone --- template struct TupleToObjectWithZone { static void convert( object::with_zone& o, const Tuple& v) { TupleToObjectWithZone::convert(o, v); o.via.array.ptr[N-1] = object(type::get(v), o.zone); } }; template struct TupleToObjectWithZone { static void convert ( object::with_zone& o, const Tuple& v) { o.via.array.ptr[0] = object(type::get<0>(v), o.zone); } }; template inline void operator<< ( object::with_zone& o, type::tuple& v) { o.type = type::ARRAY; o.via.array.ptr = static_cast(o.zone->malloc(sizeof(object)*sizeof...(Args))); o.via.array.size = sizeof...(Args); TupleToObjectWithZone::convert(o, v); } } // msgpack #endif /* msgpack/type/tuple.hpp */