/////////////////////////////////////////////////////////////////////////////// // Copyright (c) Electronic Arts Inc. All rights reserved. /////////////////////////////////////////////////////////////////////////////// #ifndef EASTL_TUPLE_H #define EASTL_TUPLE_H #include #include #include #include #include EA_DISABLE_VC_WARNING(4623) // warning C4623: default constructor was implicitly defined as deleted EA_DISABLE_VC_WARNING(4625) // warning C4625: copy constructor was implicitly defined as deleted EA_DISABLE_VC_WARNING(4510) // warning C4510: default constructor could not be generated #if EASTL_TUPLE_ENABLED namespace eastl { // non-recursive tuple implementation based on libc++ tuple implementation and description at // http://mitchnull.blogspot.ca/2012/06/c11-tuple-implementation-details-part-1.html // TupleTypes helper template struct TupleTypes {}; // tuple_size helper template class tuple_size {}; template class tuple_size : public tuple_size {}; template class tuple_size : public tuple_size {}; template class tuple_size : public tuple_size {}; template class tuple_size> : public integral_constant {}; template class tuple_size> : public integral_constant {}; #if EASTL_VARIABLE_TEMPLATES_ENABLED template EA_CONSTEXPR size_t tuple_size_v = tuple_size::value; #endif namespace Internal { template struct TupleImpl; } // namespace Internal template class tuple_size> : public integral_constant { }; // tuple_element helper to be able to isolate a type given an index template class tuple_element { }; template class tuple_element> { public: static_assert(I != I, "tuple_element index out of range"); }; template class tuple_element<0, TupleTypes> { public: typedef H type; }; template class tuple_element> { public: typedef tuple_element_t> type; }; // specialization for tuple template class tuple_element> { public: typedef tuple_element_t> type; }; template class tuple_element> { public: typedef typename add_const>>::type type; }; template class tuple_element> { public: typedef typename add_volatile>>::type type; }; template class tuple_element> { public: typedef typename add_cv>>::type type; }; // specialization for TupleImpl template class tuple_element> : public tuple_element> { }; template class tuple_element> : public tuple_element> { }; template class tuple_element> : public tuple_element> { }; template class tuple_element> : public tuple_element< I, const volatile tuple> { }; // attempt to isolate index given a type template struct tuple_index { }; template struct tuple_index> { typedef void DuplicateTypeCheck; tuple_index() = delete; // tuple_index should only be used for compile-time assistance, and never be instantiated static const size_t index = 0; }; template struct tuple_index> { typedef int DuplicateTypeCheck; // after finding type T in the list of types, try to find type T in TsRest. // If we stumble back into this version of tuple_index, i.e. type T appears twice in the list of types, then DuplicateTypeCheck will be of type int, and the static_assert will fail. // If we don't, then we'll go through the version of tuple_index above, where all of the types have been exhausted, and DuplicateTypeCheck will be void. static_assert(is_void>::DuplicateTypeCheck>::value, "duplicate type T in tuple_vector::get(); unique types must be provided in declaration, or only use get()"); static const size_t index = 0; }; template struct tuple_index> { typedef typename tuple_index>::DuplicateTypeCheck DuplicateTypeCheck; static const size_t index = tuple_index>::index + 1; }; template struct tuple_index> : public tuple_index> { }; namespace Internal { // swallow // // Provides a vessel to expand variadic packs. // template void swallow(Ts&&...) {} // TupleLeaf // template > class TupleLeaf; template inline void swap(TupleLeaf& a, TupleLeaf& b) { eastl::swap(a.getInternal(), b.getInternal()); } template class TupleLeaf { public: TupleLeaf() : mValue() {} TupleLeaf(const TupleLeaf&) = default; TupleLeaf& operator=(const TupleLeaf&) = delete; // We shouldn't need this explicit constructor as it should be handled by the template below but OSX clang // is_constructible type trait incorrectly gives false for is_constructible::value explicit TupleLeaf(ValueType&& v) : mValue(eastl::move(v)) {} template ::value>::type> explicit TupleLeaf(T&& t) : mValue(eastl::forward(t)) { } template explicit TupleLeaf(const TupleLeaf& t) : mValue(t.getInternal()) { } template TupleLeaf& operator=(T&& t) { mValue = eastl::forward(t); return *this; } int swap(TupleLeaf& t) { eastl::Internal::swap(*this, t); return 0; } ValueType& getInternal() { return mValue; } const ValueType& getInternal() const { return mValue; } private: ValueType mValue; }; // TupleLeaf: Specialize for when ValueType is a reference template class TupleLeaf { public: TupleLeaf(const TupleLeaf&) = default; TupleLeaf& operator=(const TupleLeaf&) = delete; template ::value>::type> explicit TupleLeaf(T&& t) : mValue(eastl::forward(t)) { } explicit TupleLeaf(ValueType& t) : mValue(t) { } template explicit TupleLeaf(const TupleLeaf& t) : mValue(t.getInternal()) { } template TupleLeaf& operator=(T&& t) { mValue = eastl::forward(t); return *this; } int swap(TupleLeaf& t) { eastl::Internal::swap(*this, t); return 0; } ValueType& getInternal() { return mValue; } const ValueType& getInternal() const { return mValue; } private: ValueType& mValue; }; // TupleLeaf: partial specialization for when we can use the Empty Base Class Optimization template class TupleLeaf : private ValueType { public: // true_type / false_type constructors for case where ValueType is default constructible and should be value // initialized and case where it is not TupleLeaf(const TupleLeaf&) = default; template ::value>::type> explicit TupleLeaf(T&& t) : ValueType(eastl::forward(t)) { } template explicit TupleLeaf(const TupleLeaf& t) : ValueType(t.getInternal()) { } template TupleLeaf& operator=(T&& t) { ValueType::operator=(eastl::forward(t)); return *this; } int swap(TupleLeaf& t) { eastl::Internal::swap(*this, t); return 0; } ValueType& getInternal() { return static_cast(*this); } const ValueType& getInternal() const { return static_cast(*this); } private: TupleLeaf& operator=(const TupleLeaf&) = delete; }; // MakeTupleTypes // // template struct MakeTupleTypesImpl; template struct MakeTupleTypesImpl, Tuple, Start, End> { typedef typename remove_reference::type TupleType; typedef typename MakeTupleTypesImpl< TupleTypes::value, // append ref if Tuple is ref tuple_element_t&, // append non-ref otherwise tuple_element_t>::type>, Tuple, Start + 1, End>::type type; }; template struct MakeTupleTypesImpl, Tuple, End, End> { typedef TupleTypes type; }; template using MakeTupleTypes_t = typename MakeTupleTypesImpl, Tuple, 0, tuple_size::type>::value>::type; // TupleImpl // // template tuple_element_t>& get(TupleImpl& t); template const_tuple_element_t>& get(const TupleImpl& t); template tuple_element_t>&& get(TupleImpl&& t); template T& get(TupleImpl& t); template const T& get(const TupleImpl& t); template T&& get(TupleImpl&& t); template struct TupleImpl, Ts...> : public TupleLeaf... { EA_CONSTEXPR TupleImpl() = default; // index_sequence changed to integer_sequence due to issues described below in VS2015 CTP 6. // https://connect.microsoft.com/VisualStudio/feedback/details/1126958/error-in-template-parameter-pack-expansion-of-std-index-sequence // template explicit TupleImpl(integer_sequence, TupleTypes, ValueTypes&&... values) : TupleLeaf(eastl::forward(values))... { } template TupleImpl(OtherTuple&& t) : TupleLeaf(eastl::forward>>(get(t)))... { } template TupleImpl& operator=(OtherTuple&& t) { swallow(TupleLeaf::operator=( eastl::forward>>(get(t)))...); return *this; } TupleImpl& operator=(const TupleImpl& t) { swallow(TupleLeaf::operator=(static_cast&>(t).getInternal())...); return *this; } void swap(TupleImpl& t) { swallow(TupleLeaf::swap(static_cast&>(t))...); } }; template inline tuple_element_t>& get(TupleImpl& t) { typedef tuple_element_t> Type; return static_cast&>(t).getInternal(); } template inline const_tuple_element_t>& get(const TupleImpl& t) { typedef tuple_element_t> Type; return static_cast&>(t).getInternal(); } template inline tuple_element_t>&& get(TupleImpl&& t) { typedef tuple_element_t> Type; return static_cast(static_cast&>(t).getInternal()); } template inline T& get(TupleImpl& t) { typedef tuple_index> Index; return static_cast&>(t).getInternal(); } template inline const T& get(const TupleImpl& t) { typedef tuple_index> Index; return static_cast&>(t).getInternal(); } template inline T&& get(TupleImpl&& t) { typedef tuple_index> Index; return static_cast(static_cast&>(t).getInternal()); } // TupleLike // // type-trait that determines if a type is an eastl::tuple or an eastl::pair. // template struct TupleLike : public false_type {}; template struct TupleLike : public TupleLike {}; template struct TupleLike : public TupleLike {}; template struct TupleLike : public TupleLike {}; template struct TupleLike> : public true_type {}; template struct TupleLike> : public true_type {}; // TupleConvertible // // // template struct TupleConvertibleImpl : public false_type { }; template struct TupleConvertibleImpl, TupleTypes> : public integral_constant...>::value> { }; template ::type>::value, bool = TupleLike::type>::value> struct TupleConvertible : public false_type { }; template struct TupleConvertible : public TupleConvertibleImpl::type>::value == tuple_size::type>::value, MakeTupleTypes_t, MakeTupleTypes_t> { }; // TupleAssignable // // // template struct TupleAssignableImpl : public false_type { }; template struct TupleAssignableImpl, TupleTypes> : public bool_constant...>::value> { }; template ::type>::value, bool = TupleLike::type>::value> struct TupleAssignable : public false_type { }; template struct TupleAssignable : public TupleAssignableImpl< tuple_size::type>::value == tuple_size::type>::value, MakeTupleTypes_t, MakeTupleTypes_t> { }; // TupleImplicitlyConvertible and TupleExplicitlyConvertible // // helpers for constraining conditionally-explicit ctors // template struct TupleImplicitlyConvertibleImpl : public false_type { }; template struct TupleImplicitlyConvertibleImpl, FromTypes...> : public conjunction< is_constructible..., is_convertible...> { }; template struct TupleImplicitlyConvertible : public TupleImplicitlyConvertibleImpl< tuple_size::value == sizeof...(FromTypes), MakeTupleTypes_t, FromTypes...>::type { }; template using TupleImplicitlyConvertible_t = enable_if_t::value, bool>; template struct TupleExplicitlyConvertibleImpl : public false_type { }; template struct TupleExplicitlyConvertibleImpl, FromTypes...> : public conjunction< is_constructible..., negation...>>> { }; template struct TupleExplicitlyConvertible : public TupleExplicitlyConvertibleImpl< tuple_size::value == sizeof...(FromTypes), MakeTupleTypes_t, FromTypes...>::type { }; template using TupleExplicitlyConvertible_t = enable_if_t::value, bool>; // TupleEqual // // // template struct TupleEqual { template bool operator()(const Tuple1& t1, const Tuple2& t2) { static_assert(tuple_size::value == tuple_size::value, "comparing tuples of different sizes."); return TupleEqual()(t1, t2) && get(t1) == get(t2); } }; template <> struct TupleEqual<0> { template bool operator()(const Tuple1&, const Tuple2&) { return true; } }; // TupleLess // // // template struct TupleLess { template bool operator()(const Tuple1& t1, const Tuple2& t2) { static_assert(tuple_size::value == tuple_size::value, "comparing tuples of different sizes."); return TupleLess()(t1, t2) || (!TupleLess()(t2, t1) && get(t1) < get(t2)); } }; template <> struct TupleLess<0> { template bool operator()(const Tuple1&, const Tuple2&) { return false; } }; // MakeTupleReturnImpl // // // template struct MakeTupleReturnImpl { typedef T type; }; template struct MakeTupleReturnImpl> { typedef T& type; }; template using MakeTupleReturn_t = typename MakeTupleReturnImpl>::type; // tuple_cat helpers // // // // TupleCat2Impl template struct TupleCat2Impl; template struct TupleCat2Impl, index_sequence, tuple, index_sequence> { using ResultType = tuple; template static inline ResultType DoCat2(Tuple1&& t1, Tuple2&& t2) { return ResultType(get(eastl::forward(t1))..., get(eastl::forward(t2))...); } }; // TupleCat2 template struct TupleCat2; template struct TupleCat2, tuple> { using Is1 = make_index_sequence; using Is2 = make_index_sequence; using TCI = TupleCat2Impl, Is1, tuple, Is2>; using ResultType = typename TCI::ResultType; template static inline ResultType DoCat2(Tuple1&& t1, Tuple2&& t2) { return TCI::DoCat2(eastl::forward(t1), eastl::forward(t2)); } }; // TupleCat template struct TupleCat; template struct TupleCat { using FirstResultType = typename TupleCat2::ResultType; using ResultType = typename TupleCat::ResultType; template static inline ResultType DoCat(TupleArg1&& t1, TupleArg2&& t2, TupleArgsRest&&... ts) { return TupleCat::DoCat( TupleCat2::DoCat2(eastl::forward(t1), eastl::forward(t2)), eastl::forward(ts)...); } }; template struct TupleCat { using TC2 = TupleCat2>; using ResultType = typename TC2::ResultType; template static inline ResultType DoCat(TupleArg1&& t1, TupleArg2&& t2) { return TC2::DoCat2(eastl::forward(t1), eastl::forward(t2)); } }; } // namespace Internal // tuple // // eastl::tuple is a fixed-size container of heterogeneous values. It is a // generalization of eastl::pair which hold only two heterogeneous values. // // https://en.cppreference.com/w/cpp/utility/tuple // template class tuple; template class tuple { public: EA_CONSTEXPR tuple() = default; template = 0> EA_CONSTEXPR tuple(const T& t, const Ts&... ts) : mImpl(make_index_sequence{}, Internal::MakeTupleTypes_t{}, t, ts...) { } template = 0> explicit EA_CONSTEXPR tuple(const T& t, const Ts&... ts) : mImpl(make_index_sequence{}, Internal::MakeTupleTypes_t{}, t, ts...) { } template = 0> EA_CONSTEXPR tuple(U&& u, Us&&... us) : mImpl(make_index_sequence{}, Internal::MakeTupleTypes_t{}, eastl::forward(u), eastl::forward(us)...) { } template = 0> explicit EA_CONSTEXPR tuple(U&& u, Us&&... us) : mImpl(make_index_sequence{}, Internal::MakeTupleTypes_t{}, eastl::forward(u), eastl::forward(us)...) { } template ::value, bool>::type = false> tuple(OtherTuple&& t) : mImpl(eastl::forward(t)) { } template ::value, bool>::type = false> tuple& operator=(OtherTuple&& t) { mImpl.operator=(eastl::forward(t)); return *this; } void swap(tuple& t) { mImpl.swap(t.mImpl); } private: typedef Internal::TupleImpl, T, Ts...> Impl; Impl mImpl; template friend tuple_element_t>& get(tuple& t); template friend const_tuple_element_t>& get(const tuple& t); template friend tuple_element_t>&& get(tuple&& t); template friend T_& get(tuple& t); template friend const T_& get(const tuple& t); template friend T_&& get(tuple&& t); }; // template specialization for an empty tuple template <> class tuple<> { public: void swap(tuple&) {} }; template inline tuple_element_t>& get(tuple& t) { return get(t.mImpl); } template inline const_tuple_element_t>& get(const tuple& t) { return get(t.mImpl); } template inline tuple_element_t>&& get(tuple&& t) { return get(eastl::move(t.mImpl)); } template inline T& get(tuple& t) { return get(t.mImpl); } template inline const T& get(const tuple& t) { return get(t.mImpl); } template inline T&& get(tuple&& t) { return get(eastl::move(t.mImpl)); } template inline void swap(tuple& a, tuple& b) { a.swap(b); } // tuple operators // // template inline bool operator==(const tuple& t1, const tuple& t2) { return Internal::TupleEqual()(t1, t2); } template inline bool operator<(const tuple& t1, const tuple& t2) { return Internal::TupleLess()(t1, t2); } template inline bool operator!=(const tuple& t1, const tuple& t2) { return !(t1 == t2); } template inline bool operator> (const tuple& t1, const tuple& t2) { return t2 < t1; } template inline bool operator<=(const tuple& t1, const tuple& t2) { return !(t2 < t1); } template inline bool operator>=(const tuple& t1, const tuple& t2) { return !(t1 < t2); } // tuple_cat // // template inline typename Internal::TupleCat::ResultType tuple_cat(Tuples&&... ts) { return Internal::TupleCat::DoCat(eastl::forward(ts)...); } // make_tuple // // template inline EA_CONSTEXPR tuple...> make_tuple(Ts&&... values) { return tuple...>(eastl::forward(values)...); } // forward_as_tuple // // template inline EA_CONSTEXPR tuple forward_as_tuple(Ts&&... ts) EA_NOEXCEPT { return tuple(eastl::forward(ts)...); } // ignore // // An object of unspecified type such that any value can be assigned to it with no effect. // // https://en.cppreference.com/w/cpp/utility/tuple/ignore // namespace Internal { struct ignore_t { ignore_t() = default; template const ignore_t& operator=(const T&) const { return *this; } }; }// namespace Internal static const Internal::ignore_t ignore; // tie // // Creates a tuple of lvalue references to its arguments or instances of eastl::ignore. // // https://en.cppreference.com/w/cpp/utility/tuple/tie // template inline EA_CONSTEXPR tuple tie(Ts&... ts) EA_NOEXCEPT { return tuple(ts...); } // apply // // Invoke a callable object using a tuple to supply the arguments. // // http://en.cppreference.com/w/cpp/utility/apply // namespace detail { template EA_CONSTEXPR decltype(auto) apply_impl(F&& f, Tuple&& t, index_sequence) { return invoke(eastl::forward(f), get(eastl::forward(t))...); } } // namespace detail template EA_CONSTEXPR decltype(auto) apply(F&& f, Tuple&& t) { return detail::apply_impl(eastl::forward(f), eastl::forward(t), make_index_sequence>>{}); } } // namespace eastl /////////////////////////////////////////////////////////////// // C++17 structured bindings support for eastl::tuple // #ifndef EA_COMPILER_NO_STRUCTURED_BINDING #include namespace std { // NOTE(rparolin): Some platform implementations didn't check the standard specification and implemented the // "tuple_size" and "tuple_element" primary template with as a struct. The standard specifies they are // implemented with the class keyword so we provide the template specializations as a class and disable the // generated warning. EA_DISABLE_CLANG_WARNING(-Wmismatched-tags) template class tuple_size<::eastl::tuple> : public ::eastl::integral_constant { }; template class tuple_element> : public ::eastl::tuple_element> { }; EA_RESTORE_CLANG_WARNING() } #endif #endif // EASTL_TUPLE_ENABLED EA_RESTORE_VC_WARNING() EA_RESTORE_VC_WARNING() EA_RESTORE_VC_WARNING() #endif // EASTL_TUPLE_H