__tuple 24 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635
  1. // -*- C++ -*-
  2. //===----------------------------------------------------------------------===//
  3. //
  4. // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
  5. // See https://llvm.org/LICENSE.txt for license information.
  6. // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
  7. //
  8. //===----------------------------------------------------------------------===//
  9. #ifndef _LIBCPP___TUPLE
  10. #define _LIBCPP___TUPLE
  11. #include <__config>
  12. #include <cstddef>
  13. #include <type_traits>
  14. #if !defined(_LIBCPP_HAS_NO_PRAGMA_SYSTEM_HEADER)
  15. # pragma GCC system_header
  16. #endif
  17. _LIBCPP_BEGIN_NAMESPACE_STD
  18. template <class _Tp> class _LIBCPP_TEMPLATE_VIS tuple_size;
  19. #if !defined(_LIBCPP_CXX03_LANG)
  20. template <class _Tp, class...>
  21. using __enable_if_tuple_size_imp = _Tp;
  22. template <class _Tp>
  23. class _LIBCPP_TEMPLATE_VIS tuple_size<__enable_if_tuple_size_imp<
  24. const _Tp,
  25. typename enable_if<!is_volatile<_Tp>::value>::type,
  26. integral_constant<size_t, sizeof(tuple_size<_Tp>)>>>
  27. : public integral_constant<size_t, tuple_size<_Tp>::value> {};
  28. template <class _Tp>
  29. class _LIBCPP_TEMPLATE_VIS tuple_size<__enable_if_tuple_size_imp<
  30. volatile _Tp,
  31. typename enable_if<!is_const<_Tp>::value>::type,
  32. integral_constant<size_t, sizeof(tuple_size<_Tp>)>>>
  33. : public integral_constant<size_t, tuple_size<_Tp>::value> {};
  34. template <class _Tp>
  35. class _LIBCPP_TEMPLATE_VIS tuple_size<__enable_if_tuple_size_imp<
  36. const volatile _Tp,
  37. integral_constant<size_t, sizeof(tuple_size<_Tp>)>>>
  38. : public integral_constant<size_t, tuple_size<_Tp>::value> {};
  39. #else
  40. template <class _Tp> class _LIBCPP_TEMPLATE_VIS tuple_size<const _Tp> : public tuple_size<_Tp> {};
  41. template <class _Tp> class _LIBCPP_TEMPLATE_VIS tuple_size<volatile _Tp> : public tuple_size<_Tp> {};
  42. template <class _Tp> class _LIBCPP_TEMPLATE_VIS tuple_size<const volatile _Tp> : public tuple_size<_Tp> {};
  43. #endif
  44. template <size_t _Ip, class _Tp> struct _LIBCPP_TEMPLATE_VIS tuple_element;
  45. template <size_t _Ip, class _Tp>
  46. struct _LIBCPP_TEMPLATE_VIS tuple_element<_Ip, const _Tp>
  47. {
  48. typedef _LIBCPP_NODEBUG typename add_const<typename tuple_element<_Ip, _Tp>::type>::type type;
  49. };
  50. template <size_t _Ip, class _Tp>
  51. struct _LIBCPP_TEMPLATE_VIS tuple_element<_Ip, volatile _Tp>
  52. {
  53. typedef _LIBCPP_NODEBUG typename add_volatile<typename tuple_element<_Ip, _Tp>::type>::type type;
  54. };
  55. template <size_t _Ip, class _Tp>
  56. struct _LIBCPP_TEMPLATE_VIS tuple_element<_Ip, const volatile _Tp>
  57. {
  58. typedef _LIBCPP_NODEBUG typename add_cv<typename tuple_element<_Ip, _Tp>::type>::type type;
  59. };
  60. template <class _Tp> struct __tuple_like : false_type {};
  61. template <class _Tp> struct __tuple_like<const _Tp> : public __tuple_like<_Tp> {};
  62. template <class _Tp> struct __tuple_like<volatile _Tp> : public __tuple_like<_Tp> {};
  63. template <class _Tp> struct __tuple_like<const volatile _Tp> : public __tuple_like<_Tp> {};
  64. // tuple specializations
  65. #ifndef _LIBCPP_CXX03_LANG
  66. template <size_t...> struct __tuple_indices {};
  67. template <class _IdxType, _IdxType... _Values>
  68. struct __integer_sequence {
  69. template <size_t _Sp>
  70. using __to_tuple_indices = __tuple_indices<(_Values + _Sp)...>;
  71. };
  72. #if defined(__CUDACC__) || !__has_builtin(__make_integer_seq) || defined(_LIBCPP_TESTING_FALLBACK_MAKE_INTEGER_SEQUENCE)
  73. namespace __detail {
  74. template<typename _Tp, size_t ..._Extra> struct __repeat;
  75. template<typename _Tp, _Tp ..._Np, size_t ..._Extra> struct __repeat<__integer_sequence<_Tp, _Np...>, _Extra...> {
  76. typedef _LIBCPP_NODEBUG __integer_sequence<_Tp,
  77. _Np...,
  78. sizeof...(_Np) + _Np...,
  79. 2 * sizeof...(_Np) + _Np...,
  80. 3 * sizeof...(_Np) + _Np...,
  81. 4 * sizeof...(_Np) + _Np...,
  82. 5 * sizeof...(_Np) + _Np...,
  83. 6 * sizeof...(_Np) + _Np...,
  84. 7 * sizeof...(_Np) + _Np...,
  85. _Extra...> type;
  86. };
  87. template<size_t _Np> struct __parity;
  88. template<size_t _Np> struct __make : __parity<_Np % 8>::template __pmake<_Np> {};
  89. template<> struct __make<0> { typedef __integer_sequence<size_t> type; };
  90. template<> struct __make<1> { typedef __integer_sequence<size_t, 0> type; };
  91. template<> struct __make<2> { typedef __integer_sequence<size_t, 0, 1> type; };
  92. template<> struct __make<3> { typedef __integer_sequence<size_t, 0, 1, 2> type; };
  93. template<> struct __make<4> { typedef __integer_sequence<size_t, 0, 1, 2, 3> type; };
  94. template<> struct __make<5> { typedef __integer_sequence<size_t, 0, 1, 2, 3, 4> type; };
  95. template<> struct __make<6> { typedef __integer_sequence<size_t, 0, 1, 2, 3, 4, 5> type; };
  96. template<> struct __make<7> { typedef __integer_sequence<size_t, 0, 1, 2, 3, 4, 5, 6> type; };
  97. template<> struct __parity<0> { template<size_t _Np> struct __pmake : __repeat<typename __make<_Np / 8>::type> {}; };
  98. template<> struct __parity<1> { template<size_t _Np> struct __pmake : __repeat<typename __make<_Np / 8>::type, _Np - 1> {}; };
  99. template<> struct __parity<2> { template<size_t _Np> struct __pmake : __repeat<typename __make<_Np / 8>::type, _Np - 2, _Np - 1> {}; };
  100. template<> struct __parity<3> { template<size_t _Np> struct __pmake : __repeat<typename __make<_Np / 8>::type, _Np - 3, _Np - 2, _Np - 1> {}; };
  101. template<> struct __parity<4> { template<size_t _Np> struct __pmake : __repeat<typename __make<_Np / 8>::type, _Np - 4, _Np - 3, _Np - 2, _Np - 1> {}; };
  102. template<> struct __parity<5> { template<size_t _Np> struct __pmake : __repeat<typename __make<_Np / 8>::type, _Np - 5, _Np - 4, _Np - 3, _Np - 2, _Np - 1> {}; };
  103. template<> struct __parity<6> { template<size_t _Np> struct __pmake : __repeat<typename __make<_Np / 8>::type, _Np - 6, _Np - 5, _Np - 4, _Np - 3, _Np - 2, _Np - 1> {}; };
  104. template<> struct __parity<7> { template<size_t _Np> struct __pmake : __repeat<typename __make<_Np / 8>::type, _Np - 7, _Np - 6, _Np - 5, _Np - 4, _Np - 3, _Np - 2, _Np - 1> {}; };
  105. } // namespace detail
  106. #endif // !__has_builtin(__make_integer_seq) || defined(_LIBCPP_TESTING_FALLBACK_MAKE_INTEGER_SEQUENCE)
  107. #if __has_builtin(__make_integer_seq) && !defined(__CUDACC__)
  108. template <size_t _Ep, size_t _Sp>
  109. using __make_indices_imp =
  110. typename __make_integer_seq<__integer_sequence, size_t, _Ep - _Sp>::template
  111. __to_tuple_indices<_Sp>;
  112. #else
  113. template <size_t _Ep, size_t _Sp>
  114. using __make_indices_imp =
  115. typename __detail::__make<_Ep - _Sp>::type::template __to_tuple_indices<_Sp>;
  116. #endif
  117. template <size_t _Ep, size_t _Sp = 0>
  118. struct __make_tuple_indices
  119. {
  120. static_assert(_Sp <= _Ep, "__make_tuple_indices input error");
  121. typedef __make_indices_imp<_Ep, _Sp> type;
  122. };
  123. template <class ..._Tp> class _LIBCPP_TEMPLATE_VIS tuple;
  124. template <class... _Tp> struct __tuple_like<tuple<_Tp...> > : true_type {};
  125. template <class ..._Tp>
  126. class _LIBCPP_TEMPLATE_VIS tuple_size<tuple<_Tp...> >
  127. : public integral_constant<size_t, sizeof...(_Tp)>
  128. {
  129. };
  130. template <size_t _Ip, class ..._Tp>
  131. _LIBCPP_INLINE_VISIBILITY _LIBCPP_CONSTEXPR_AFTER_CXX11
  132. typename tuple_element<_Ip, tuple<_Tp...> >::type&
  133. get(tuple<_Tp...>&) _NOEXCEPT;
  134. template <size_t _Ip, class ..._Tp>
  135. _LIBCPP_INLINE_VISIBILITY _LIBCPP_CONSTEXPR_AFTER_CXX11
  136. const typename tuple_element<_Ip, tuple<_Tp...> >::type&
  137. get(const tuple<_Tp...>&) _NOEXCEPT;
  138. template <size_t _Ip, class ..._Tp>
  139. _LIBCPP_INLINE_VISIBILITY _LIBCPP_CONSTEXPR_AFTER_CXX11
  140. typename tuple_element<_Ip, tuple<_Tp...> >::type&&
  141. get(tuple<_Tp...>&&) _NOEXCEPT;
  142. template <size_t _Ip, class ..._Tp>
  143. _LIBCPP_INLINE_VISIBILITY _LIBCPP_CONSTEXPR_AFTER_CXX11
  144. const typename tuple_element<_Ip, tuple<_Tp...> >::type&&
  145. get(const tuple<_Tp...>&&) _NOEXCEPT;
  146. #endif // !defined(_LIBCPP_CXX03_LANG)
  147. // pair specializations
  148. template <class _T1, class _T2> struct __tuple_like<pair<_T1, _T2> > : true_type {};
  149. template <size_t _Ip, class _T1, class _T2>
  150. _LIBCPP_INLINE_VISIBILITY _LIBCPP_CONSTEXPR_AFTER_CXX11
  151. typename tuple_element<_Ip, pair<_T1, _T2> >::type&
  152. get(pair<_T1, _T2>&) _NOEXCEPT;
  153. template <size_t _Ip, class _T1, class _T2>
  154. _LIBCPP_INLINE_VISIBILITY _LIBCPP_CONSTEXPR_AFTER_CXX11
  155. const typename tuple_element<_Ip, pair<_T1, _T2> >::type&
  156. get(const pair<_T1, _T2>&) _NOEXCEPT;
  157. #ifndef _LIBCPP_CXX03_LANG
  158. template <size_t _Ip, class _T1, class _T2>
  159. _LIBCPP_INLINE_VISIBILITY _LIBCPP_CONSTEXPR_AFTER_CXX11
  160. typename tuple_element<_Ip, pair<_T1, _T2> >::type&&
  161. get(pair<_T1, _T2>&&) _NOEXCEPT;
  162. template <size_t _Ip, class _T1, class _T2>
  163. _LIBCPP_INLINE_VISIBILITY _LIBCPP_CONSTEXPR_AFTER_CXX11
  164. const typename tuple_element<_Ip, pair<_T1, _T2> >::type&&
  165. get(const pair<_T1, _T2>&&) _NOEXCEPT;
  166. #endif
  167. // array specializations
  168. template <class _Tp, size_t _Size> struct _LIBCPP_TEMPLATE_VIS array;
  169. template <class _Tp, size_t _Size> struct __tuple_like<array<_Tp, _Size> > : true_type {};
  170. template <size_t _Ip, class _Tp, size_t _Size>
  171. _LIBCPP_INLINE_VISIBILITY _LIBCPP_CONSTEXPR_AFTER_CXX11
  172. _Tp&
  173. get(array<_Tp, _Size>&) _NOEXCEPT;
  174. template <size_t _Ip, class _Tp, size_t _Size>
  175. _LIBCPP_INLINE_VISIBILITY _LIBCPP_CONSTEXPR_AFTER_CXX11
  176. const _Tp&
  177. get(const array<_Tp, _Size>&) _NOEXCEPT;
  178. #ifndef _LIBCPP_CXX03_LANG
  179. template <size_t _Ip, class _Tp, size_t _Size>
  180. _LIBCPP_INLINE_VISIBILITY _LIBCPP_CONSTEXPR_AFTER_CXX11
  181. _Tp&&
  182. get(array<_Tp, _Size>&&) _NOEXCEPT;
  183. template <size_t _Ip, class _Tp, size_t _Size>
  184. _LIBCPP_INLINE_VISIBILITY _LIBCPP_CONSTEXPR_AFTER_CXX11
  185. const _Tp&&
  186. get(const array<_Tp, _Size>&&) _NOEXCEPT;
  187. #endif
  188. #ifndef _LIBCPP_CXX03_LANG
  189. // __tuple_types
  190. template <class ..._Tp> struct __tuple_types {};
  191. #if !__has_builtin(__type_pack_element)
  192. namespace __indexer_detail {
  193. template <size_t _Idx, class _Tp>
  194. struct __indexed { using type _LIBCPP_NODEBUG = _Tp; };
  195. template <class _Types, class _Indexes> struct __indexer;
  196. template <class ..._Types, size_t ..._Idx>
  197. struct __indexer<__tuple_types<_Types...>, __tuple_indices<_Idx...>>
  198. : __indexed<_Idx, _Types>...
  199. {};
  200. template <size_t _Idx, class _Tp>
  201. __indexed<_Idx, _Tp> __at_index(__indexed<_Idx, _Tp> const&);
  202. } // namespace __indexer_detail
  203. #if !defined(__CUDACC__) || !defined(_MSC_VER)
  204. template <size_t _Idx, class ..._Types>
  205. using __type_pack_element _LIBCPP_NODEBUG = typename decltype(
  206. __indexer_detail::__at_index<_Idx>(
  207. __indexer_detail::__indexer<
  208. __tuple_types<_Types...>,
  209. typename __make_tuple_indices<sizeof...(_Types)>::type
  210. >{})
  211. )::type;
  212. #else // !defined(__CUDACC__) || !defined(_MSC_VER)
  213. template <size_t _Idx, class ..._Types>
  214. struct __y_type_pack_element {
  215. using __t1 = typename __make_tuple_indices<sizeof...(_Types)>::type;
  216. using __t2 = __indexer_detail::__indexer<__tuple_types<_Types...>, __t1>;
  217. using __t3 = decltype(__indexer_detail::__at_index<_Idx>(__t2{}));
  218. using __t4 = typename __t3::type;
  219. };
  220. template <size_t _Idx, class ..._Types>
  221. using __type_pack_element = typename __y_type_pack_element<_Idx, _Types...>::__t4;
  222. #endif // !defined(__CUDACC__) || !defined(_MSC_VER)
  223. #endif // __has_builtin(__type_pack_element)
  224. template <size_t _Ip, class ..._Types>
  225. struct _LIBCPP_TEMPLATE_VIS tuple_element<_Ip, __tuple_types<_Types...> >
  226. {
  227. static_assert(_Ip < sizeof...(_Types), "tuple_element index out of range");
  228. typedef _LIBCPP_NODEBUG __type_pack_element<_Ip, _Types...> type;
  229. };
  230. template <class ..._Tp>
  231. class _LIBCPP_TEMPLATE_VIS tuple_size<__tuple_types<_Tp...> >
  232. : public integral_constant<size_t, sizeof...(_Tp)>
  233. {
  234. };
  235. template <class... _Tp> struct __tuple_like<__tuple_types<_Tp...> > : true_type {};
  236. template <bool _ApplyLV, bool _ApplyConst, bool _ApplyVolatile>
  237. struct __apply_cv_mf;
  238. template <>
  239. struct __apply_cv_mf<false, false, false> {
  240. template <class _Tp> using __apply = _Tp;
  241. };
  242. template <>
  243. struct __apply_cv_mf<false, true, false> {
  244. template <class _Tp> using __apply _LIBCPP_NODEBUG = const _Tp;
  245. };
  246. template <>
  247. struct __apply_cv_mf<false, false, true> {
  248. template <class _Tp> using __apply _LIBCPP_NODEBUG = volatile _Tp;
  249. };
  250. template <>
  251. struct __apply_cv_mf<false, true, true> {
  252. template <class _Tp> using __apply _LIBCPP_NODEBUG = const volatile _Tp;
  253. };
  254. template <>
  255. struct __apply_cv_mf<true, false, false> {
  256. template <class _Tp> using __apply _LIBCPP_NODEBUG = _Tp&;
  257. };
  258. template <>
  259. struct __apply_cv_mf<true, true, false> {
  260. template <class _Tp> using __apply _LIBCPP_NODEBUG = const _Tp&;
  261. };
  262. template <>
  263. struct __apply_cv_mf<true, false, true> {
  264. template <class _Tp> using __apply _LIBCPP_NODEBUG = volatile _Tp&;
  265. };
  266. template <>
  267. struct __apply_cv_mf<true, true, true> {
  268. template <class _Tp> using __apply _LIBCPP_NODEBUG = const volatile _Tp&;
  269. };
  270. template <class _Tp, class _RawTp = typename remove_reference<_Tp>::type>
  271. using __apply_cv_t _LIBCPP_NODEBUG = __apply_cv_mf<
  272. is_lvalue_reference<_Tp>::value,
  273. is_const<_RawTp>::value,
  274. is_volatile<_RawTp>::value>;
  275. // __make_tuple_types
  276. // __make_tuple_types<_Tuple<_Types...>, _Ep, _Sp>::type is a
  277. // __tuple_types<_Types...> using only those _Types in the range [_Sp, _Ep).
  278. // _Sp defaults to 0 and _Ep defaults to tuple_size<_Tuple>. If _Tuple is a
  279. // lvalue_reference type, then __tuple_types<_Types&...> is the result.
  280. #ifdef _LIBCPP_COMPILER_MSVC
  281. template<class _TupleTypes, class _Tp>
  282. struct __make_tuple_types_append_front;
  283. template<class ..._Types, class _Tp>
  284. struct __make_tuple_types_append_front<__tuple_types<_Types...>, _Tp> {
  285. using type = __tuple_types<_Tp, _Types...>;
  286. };
  287. template<class _TupleTypes, class _Apply, size_t _Ep, size_t _Sp, class = void>
  288. struct __make_tuple_types_flat;
  289. template<class _Tuple, class _Apply>
  290. struct __make_tuple_types_flat<_Tuple, _Apply, 0, 0, void> {
  291. using type = __tuple_types<>;
  292. };
  293. template<template<class...> class _Tuple, class _Tp, class ..._Types, class _Apply, size_t _Ep>
  294. struct __make_tuple_types_flat<_Tuple<_Tp, _Types...>, _Apply, _Ep, 0, enable_if_t<(_Ep > 0)>> {
  295. using type = typename __make_tuple_types_append_front<
  296. typename __make_tuple_types_flat<__tuple_types<_Types...>, _Apply, _Ep - 1, 0>::type,
  297. typename _Apply::template __apply<_Tp>>::type;
  298. };
  299. template<template <class...> class _Tuple, class _Tp, class ..._Types, class _Apply, size_t _Ep, size_t _Sp>
  300. struct __make_tuple_types_flat<_Tuple<_Tp, _Types...>, _Apply, _Ep, _Sp, enable_if_t<(_Ep > 0 && _Sp > 0)>> {
  301. using type = typename __make_tuple_types_flat<__tuple_types<_Types...>, _Apply, _Ep - 1, _Sp - 1>::type;
  302. };
  303. template<class _Tp, size_t _Np, class _Apply, size_t _Ep>
  304. struct __make_tuple_types_flat<array<_Tp, _Np>, _Apply, _Ep, 0, enable_if_t<(_Ep > 0)>> {
  305. using type = typename __make_tuple_types_append_front<
  306. typename __make_tuple_types_flat<array<_Tp, _Np>, _Apply, _Ep - 1, 0>::type,
  307. typename _Apply::template __apply<_Tp>>::type;
  308. };
  309. template<class _Tp, size_t _Np, class _Apply, size_t _Ep, size_t _Sp>
  310. struct __make_tuple_types_flat<array<_Tp, _Np>, _Apply, _Ep, _Sp, enable_if_t<(_Sp > 0)>> {
  311. using type = typename __make_tuple_types_flat<array<_Tp, _Np>, _Apply, _Ep - _Sp, 0>::type;
  312. };
  313. template<class _Tp,
  314. size_t _Ep = tuple_size<remove_reference_t<_Tp>>::value,
  315. size_t _Sp = 0,
  316. bool _SameSize = (_Ep == tuple_size<typename remove_reference<_Tp>::type>::value)>
  317. struct __make_tuple_types {
  318. static_assert(_Sp <= _Ep, "__make_tuple_types input error");
  319. using _RawTp = typename remove_cv<typename remove_reference<_Tp>::type>::type;
  320. using type = typename __make_tuple_types_flat<_RawTp, __apply_cv_t<_Tp>, _Ep, _Sp>::type;
  321. };
  322. template <class ..._Types, size_t _Ep>
  323. struct __make_tuple_types<tuple<_Types...>, _Ep, 0, true> {
  324. typedef __tuple_types<_Types...> type;
  325. };
  326. template <class ..._Types, size_t _Ep>
  327. struct __make_tuple_types<__tuple_types<_Types...>, _Ep, 0, true> {
  328. typedef __tuple_types<_Types...> type;
  329. };
  330. #else
  331. template <class _TupleTypes, class _TupleIndices>
  332. struct __make_tuple_types_flat;
  333. template <template <class...> class _Tuple, class ..._Types, size_t ..._Idx>
  334. struct __make_tuple_types_flat<_Tuple<_Types...>, __tuple_indices<_Idx...>> {
  335. // Specialization for pair, tuple, and __tuple_types
  336. template <class _Tp, class _ApplyFn = __apply_cv_t<_Tp>>
  337. using __apply_quals _LIBCPP_NODEBUG = __tuple_types<
  338. typename _ApplyFn::template __apply<__type_pack_element<_Idx, _Types...>>...
  339. >;
  340. };
  341. template <class _Vt, size_t _Np, size_t ..._Idx>
  342. struct __make_tuple_types_flat<array<_Vt, _Np>, __tuple_indices<_Idx...>> {
  343. template <size_t>
  344. using __value_type = _Vt;
  345. template <class _Tp, class _ApplyFn = __apply_cv_t<_Tp>>
  346. using __apply_quals = __tuple_types<
  347. typename _ApplyFn::template __apply<__value_type<_Idx>>...
  348. >;
  349. };
  350. template <class _Tp, size_t _Ep = tuple_size<typename remove_reference<_Tp>::type>::value,
  351. size_t _Sp = 0,
  352. bool _SameSize = (_Ep == tuple_size<typename remove_reference<_Tp>::type>::value)>
  353. struct __make_tuple_types
  354. {
  355. static_assert(_Sp <= _Ep, "__make_tuple_types input error");
  356. using _RawTp = typename remove_cv<typename remove_reference<_Tp>::type>::type;
  357. using _Maker = __make_tuple_types_flat<_RawTp, typename __make_tuple_indices<_Ep, _Sp>::type>;
  358. using type = typename _Maker::template __apply_quals<_Tp>;
  359. };
  360. template <class ..._Types, size_t _Ep>
  361. struct __make_tuple_types<tuple<_Types...>, _Ep, 0, true> {
  362. typedef _LIBCPP_NODEBUG __tuple_types<_Types...> type;
  363. };
  364. template <class ..._Types, size_t _Ep>
  365. struct __make_tuple_types<__tuple_types<_Types...>, _Ep, 0, true> {
  366. typedef _LIBCPP_NODEBUG __tuple_types<_Types...> type;
  367. };
  368. #endif // _LIBCPP_COMPILER_MSVC
  369. template <bool ..._Preds>
  370. struct __all_dummy;
  371. template <bool ..._Pred>
  372. struct __all : _IsSame<__all_dummy<_Pred...>, __all_dummy<(_Pred | true)...>>{};
  373. struct __tuple_sfinae_base {
  374. template <template <class, class...> class _Trait,
  375. class ..._LArgs, class ..._RArgs>
  376. static auto __do_test(__tuple_types<_LArgs...>, __tuple_types<_RArgs...>)
  377. -> __all<_Trait<_LArgs, _RArgs>::value...>;
  378. template <template <class...> class>
  379. static auto __do_test(...) -> false_type;
  380. template <class _FromArgs, class _ToArgs>
  381. using __constructible = decltype(__do_test<is_constructible>(declval<_ToArgs>(), declval<_FromArgs>()));
  382. template <class _FromArgs, class _ToArgs>
  383. using __convertible = decltype(__do_test<is_convertible>(declval<_FromArgs>(), declval<_ToArgs>()));
  384. template <class _FromArgs, class _ToArgs>
  385. using __assignable = decltype(__do_test<is_assignable>(declval<_ToArgs>(), declval<_FromArgs>()));
  386. };
  387. // __tuple_convertible
  388. template <class _Tp, class _Up, bool = __tuple_like<typename remove_reference<_Tp>::type>::value,
  389. bool = __tuple_like<_Up>::value,
  390. class = void>
  391. struct __tuple_convertible
  392. : public false_type {};
  393. template <class _Tp, class _Up>
  394. struct __tuple_convertible<_Tp, _Up, true, true,
  395. typename enable_if<(tuple_size<typename remove_reference<_Tp>::type>::value == tuple_size<_Up>::value)>::type>
  396. : public __tuple_sfinae_base::__convertible<
  397. typename __make_tuple_types<_Tp>::type
  398. , typename __make_tuple_types<_Up>::type
  399. >
  400. {};
  401. // __tuple_constructible
  402. template <class _Tp, class _Up, bool = __tuple_like<typename remove_reference<_Tp>::type>::value,
  403. bool = __tuple_like<_Up>::value,
  404. class = void>
  405. struct __tuple_constructible
  406. : public false_type {};
  407. template <class _Tp, class _Up>
  408. struct __tuple_constructible<_Tp, _Up, true, true,
  409. typename enable_if<(tuple_size<typename remove_reference<_Tp>::type>::value == tuple_size<_Up>::value)>::type>
  410. : public __tuple_sfinae_base::__constructible<
  411. typename __make_tuple_types<_Tp>::type
  412. , typename __make_tuple_types<_Up>::type
  413. >
  414. {};
  415. // __tuple_assignable
  416. template <class _Tp, class _Up, bool = __tuple_like<typename remove_reference<_Tp>::type>::value,
  417. bool = __tuple_like<_Up>::value,
  418. class = void>
  419. struct __tuple_assignable
  420. : public false_type {};
  421. template <class _Tp, class _Up>
  422. struct __tuple_assignable<_Tp, _Up, true, true,
  423. typename enable_if<(tuple_size<typename remove_reference<_Tp>::type>::value == tuple_size<_Up>::value)>::type>
  424. : public __tuple_sfinae_base::__assignable<
  425. typename __make_tuple_types<_Tp>::type
  426. , typename __make_tuple_types<_Up&>::type
  427. >
  428. {};
  429. template <size_t _Ip, class ..._Tp>
  430. struct _LIBCPP_TEMPLATE_VIS tuple_element<_Ip, tuple<_Tp...> >
  431. {
  432. typedef _LIBCPP_NODEBUG typename tuple_element<_Ip, __tuple_types<_Tp...> >::type type;
  433. };
  434. #if _LIBCPP_STD_VER > 11
  435. template <size_t _Ip, class ..._Tp>
  436. using tuple_element_t _LIBCPP_NODEBUG = typename tuple_element <_Ip, _Tp...>::type;
  437. #endif
  438. template <bool _IsTuple, class _SizeTrait, size_t _Expected>
  439. struct __tuple_like_with_size_imp : false_type {};
  440. template <class _SizeTrait, size_t _Expected>
  441. struct __tuple_like_with_size_imp<true, _SizeTrait, _Expected>
  442. : integral_constant<bool, _SizeTrait::value == _Expected> {};
  443. template <class _Tuple, size_t _ExpectedSize, class _RawTuple = __uncvref_t<_Tuple> >
  444. using __tuple_like_with_size _LIBCPP_NODEBUG = __tuple_like_with_size_imp<
  445. __tuple_like<_RawTuple>::value,
  446. tuple_size<_RawTuple>, _ExpectedSize
  447. >;
  448. struct _LIBCPP_TYPE_VIS __check_tuple_constructor_fail {
  449. static constexpr bool __enable_explicit_default() { return false; }
  450. static constexpr bool __enable_implicit_default() { return false; }
  451. template <class ...>
  452. static constexpr bool __enable_explicit() { return false; }
  453. template <class ...>
  454. static constexpr bool __enable_implicit() { return false; }
  455. template <class ...>
  456. static constexpr bool __enable_assign() { return false; }
  457. };
  458. #endif // !defined(_LIBCPP_CXX03_LANG)
  459. #if _LIBCPP_STD_VER > 14
  460. template <bool _CanCopy, bool _CanMove>
  461. struct __sfinae_ctor_base {};
  462. template <>
  463. struct __sfinae_ctor_base<false, false> {
  464. __sfinae_ctor_base() = default;
  465. __sfinae_ctor_base(__sfinae_ctor_base const&) = delete;
  466. __sfinae_ctor_base(__sfinae_ctor_base &&) = delete;
  467. __sfinae_ctor_base& operator=(__sfinae_ctor_base const&) = default;
  468. __sfinae_ctor_base& operator=(__sfinae_ctor_base&&) = default;
  469. };
  470. template <>
  471. struct __sfinae_ctor_base<true, false> {
  472. __sfinae_ctor_base() = default;
  473. __sfinae_ctor_base(__sfinae_ctor_base const&) = default;
  474. __sfinae_ctor_base(__sfinae_ctor_base &&) = delete;
  475. __sfinae_ctor_base& operator=(__sfinae_ctor_base const&) = default;
  476. __sfinae_ctor_base& operator=(__sfinae_ctor_base&&) = default;
  477. };
  478. template <>
  479. struct __sfinae_ctor_base<false, true> {
  480. __sfinae_ctor_base() = default;
  481. __sfinae_ctor_base(__sfinae_ctor_base const&) = delete;
  482. __sfinae_ctor_base(__sfinae_ctor_base &&) = default;
  483. __sfinae_ctor_base& operator=(__sfinae_ctor_base const&) = default;
  484. __sfinae_ctor_base& operator=(__sfinae_ctor_base&&) = default;
  485. };
  486. template <bool _CanCopy, bool _CanMove>
  487. struct __sfinae_assign_base {};
  488. template <>
  489. struct __sfinae_assign_base<false, false> {
  490. __sfinae_assign_base() = default;
  491. __sfinae_assign_base(__sfinae_assign_base const&) = default;
  492. __sfinae_assign_base(__sfinae_assign_base &&) = default;
  493. __sfinae_assign_base& operator=(__sfinae_assign_base const&) = delete;
  494. __sfinae_assign_base& operator=(__sfinae_assign_base&&) = delete;
  495. };
  496. template <>
  497. struct __sfinae_assign_base<true, false> {
  498. __sfinae_assign_base() = default;
  499. __sfinae_assign_base(__sfinae_assign_base const&) = default;
  500. __sfinae_assign_base(__sfinae_assign_base &&) = default;
  501. __sfinae_assign_base& operator=(__sfinae_assign_base const&) = default;
  502. __sfinae_assign_base& operator=(__sfinae_assign_base&&) = delete;
  503. };
  504. template <>
  505. struct __sfinae_assign_base<false, true> {
  506. __sfinae_assign_base() = default;
  507. __sfinae_assign_base(__sfinae_assign_base const&) = default;
  508. __sfinae_assign_base(__sfinae_assign_base &&) = default;
  509. __sfinae_assign_base& operator=(__sfinae_assign_base const&) = delete;
  510. __sfinae_assign_base& operator=(__sfinae_assign_base&&) = default;
  511. };
  512. #endif // _LIBCPP_STD_VER > 14
  513. _LIBCPP_END_NAMESPACE_STD
  514. #endif // _LIBCPP___TUPLE