3 // Copyright (C) 2007-2018 Free Software Foundation, Inc.
     5 // This file is part of the GNU ISO C++ Library.  This library is free
     6 // software; you can redistribute it and/or modify it under the
     7 // terms of the GNU General Public License as published by the
     8 // Free Software Foundation; either version 3, or (at your option)
    11 // This library is distributed in the hope that it will be useful,
    12 // but WITHOUT ANY WARRANTY; without even the implied warranty of
    13 // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
    14 // GNU General Public License for more details.
    16 // Under Section 7 of GPL version 3, you are granted additional
    17 // permissions described in the GCC Runtime Library Exception, version
    18 // 3.1, as published by the Free Software Foundation.
    20 // You should have received a copy of the GNU General Public License and
    21 // a copy of the GCC Runtime Library Exception along with this program;
    22 // see the files COPYING3 and COPYING.RUNTIME respectively.  If not, see
    23 // <http://www.gnu.org/licenses/>.
    25 /** @file include/tuple
    26  *  This is a Standard C++ Library header.
    29 #ifndef _GLIBCXX_TUPLE
    30 #define _GLIBCXX_TUPLE 1
    32 #pragma GCC system_header
    34 #if __cplusplus < 201103L
    35 # include <bits/c++0x_warning.h>
    40 #include <bits/uses_allocator.h>
    41 #include <bits/invoke.h>
    43 namespace std _GLIBCXX_VISIBILITY(default)
    45 _GLIBCXX_BEGIN_NAMESPACE_VERSION
    48    *  @addtogroup utilities
    52   template<typename... _Elements>
    55   template<typename _Tp>
    56     struct __is_empty_non_tuple : is_empty<_Tp> { };
    58   // Using EBO for elements that are tuples causes ambiguous base errors.
    59   template<typename _El0, typename... _El>
    60     struct __is_empty_non_tuple<tuple<_El0, _El...>> : false_type { };
    62   // Use the Empty Base-class Optimization for empty, non-final types.
    63   template<typename _Tp>
    64     using __empty_not_final
    65     = typename conditional<__is_final(_Tp), false_type,
    66                       __is_empty_non_tuple<_Tp>>::type;
    68   template<std::size_t _Idx, typename _Head,
    69       bool = __empty_not_final<_Head>::value>
    72   template<std::size_t _Idx, typename _Head>
    73     struct _Head_base<_Idx, _Head, true>
    76       constexpr _Head_base()
    79       constexpr _Head_base(const _Head& __h)
    82       constexpr _Head_base(const _Head_base&) = default;
    83       constexpr _Head_base(_Head_base&&) = default;
    85       template<typename _UHead>
    86         constexpr _Head_base(_UHead&& __h)
    87    : _Head(std::forward<_UHead>(__h)) { }
    89       _Head_base(allocator_arg_t, __uses_alloc0)
    92       template<typename _Alloc>
    93    _Head_base(allocator_arg_t, __uses_alloc1<_Alloc> __a)
    94    : _Head(allocator_arg, *__a._M_a) { }
    96       template<typename _Alloc>
    97    _Head_base(allocator_arg_t, __uses_alloc2<_Alloc> __a)
    98    : _Head(*__a._M_a) { }
   100       template<typename _UHead>
   101    _Head_base(__uses_alloc0, _UHead&& __uhead)
   102    : _Head(std::forward<_UHead>(__uhead)) { }
   104       template<typename _Alloc, typename _UHead>
   105    _Head_base(__uses_alloc1<_Alloc> __a, _UHead&& __uhead)
   106    : _Head(allocator_arg, *__a._M_a, std::forward<_UHead>(__uhead)) { }
   108       template<typename _Alloc, typename _UHead>
   109    _Head_base(__uses_alloc2<_Alloc> __a, _UHead&& __uhead)
   110    : _Head(std::forward<_UHead>(__uhead), *__a._M_a) { }
   112       static constexpr _Head&
   113       _M_head(_Head_base& __b) noexcept { return __b; }
   115       static constexpr const _Head&
   116       _M_head(const _Head_base& __b) noexcept { return __b; }
   119   template<std::size_t _Idx, typename _Head>
   120     struct _Head_base<_Idx, _Head, false>
   122       constexpr _Head_base()
   125       constexpr _Head_base(const _Head& __h)
   126       : _M_head_impl(__h) { }
   128       constexpr _Head_base(const _Head_base&) = default;
   129       constexpr _Head_base(_Head_base&&) = default;
   131       template<typename _UHead>
   132         constexpr _Head_base(_UHead&& __h)
   133    : _M_head_impl(std::forward<_UHead>(__h)) { }
   135       _Head_base(allocator_arg_t, __uses_alloc0)
   138       template<typename _Alloc>
   139    _Head_base(allocator_arg_t, __uses_alloc1<_Alloc> __a)
   140    : _M_head_impl(allocator_arg, *__a._M_a) { }
   142       template<typename _Alloc>
   143    _Head_base(allocator_arg_t, __uses_alloc2<_Alloc> __a)
   144    : _M_head_impl(*__a._M_a) { }
   146       template<typename _UHead>
   147    _Head_base(__uses_alloc0, _UHead&& __uhead)
   148    : _M_head_impl(std::forward<_UHead>(__uhead)) { }
   150       template<typename _Alloc, typename _UHead>
   151    _Head_base(__uses_alloc1<_Alloc> __a, _UHead&& __uhead)
   152    : _M_head_impl(allocator_arg, *__a._M_a, std::forward<_UHead>(__uhead))
   155       template<typename _Alloc, typename _UHead>
   156    _Head_base(__uses_alloc2<_Alloc> __a, _UHead&& __uhead)
   157    : _M_head_impl(std::forward<_UHead>(__uhead), *__a._M_a) { }
   159       static constexpr _Head&
   160       _M_head(_Head_base& __b) noexcept { return __b._M_head_impl; }
   162       static constexpr const _Head&
   163       _M_head(const _Head_base& __b) noexcept { return __b._M_head_impl; }
   169    * Contains the actual implementation of the @c tuple template, stored
   170    * as a recursive inheritance hierarchy from the first element (most
   171    * derived class) to the last (least derived class). The @c Idx
   172    * parameter gives the 0-based index of the element stored at this
   173    * point in the hierarchy; we use it to implement a constant-time
   176   template<std::size_t _Idx, typename... _Elements>
   180    * Recursive tuple implementation. Here we store the @c Head element
   181    * and derive from a @c Tuple_impl containing the remaining elements
   182    * (which contains the @c Tail).
   184   template<std::size_t _Idx, typename _Head, typename... _Tail>
   185     struct _Tuple_impl<_Idx, _Head, _Tail...>
   186     : public _Tuple_impl<_Idx + 1, _Tail...>,
   187       private _Head_base<_Idx, _Head>
   189       template<std::size_t, typename...> friend class _Tuple_impl;
   191       typedef _Tuple_impl<_Idx + 1, _Tail...> _Inherited;
   192       typedef _Head_base<_Idx, _Head> _Base;
   194       static constexpr _Head&
   195       _M_head(_Tuple_impl& __t) noexcept { return _Base::_M_head(__t); }
   197       static constexpr const _Head&
   198       _M_head(const _Tuple_impl& __t) noexcept { return _Base::_M_head(__t); }
   200       static constexpr _Inherited&
   201       _M_tail(_Tuple_impl& __t) noexcept { return __t; }
   203       static constexpr const _Inherited&
   204       _M_tail(const _Tuple_impl& __t) noexcept { return __t; }
   206       constexpr _Tuple_impl()
   207       : _Inherited(), _Base() { }
   210       constexpr _Tuple_impl(const _Head& __head, const _Tail&... __tail)
   211       : _Inherited(__tail...), _Base(__head) { }
   213       template<typename _UHead, typename... _UTail, typename = typename
   214                enable_if<sizeof...(_Tail) == sizeof...(_UTail)>::type>
   216         constexpr _Tuple_impl(_UHead&& __head, _UTail&&... __tail)
   217    : _Inherited(std::forward<_UTail>(__tail)...),
   218      _Base(std::forward<_UHead>(__head)) { }
   220       constexpr _Tuple_impl(const _Tuple_impl&) = default;
   223       _Tuple_impl(_Tuple_impl&& __in)
   224       noexcept(__and_<is_nothrow_move_constructible<_Head>,
   225                  is_nothrow_move_constructible<_Inherited>>::value)
   226       : _Inherited(std::move(_M_tail(__in))),
   227    _Base(std::forward<_Head>(_M_head(__in))) { }
   229       template<typename... _UElements>
   230         constexpr _Tuple_impl(const _Tuple_impl<_Idx, _UElements...>& __in)
   231    : _Inherited(_Tuple_impl<_Idx, _UElements...>::_M_tail(__in)),
   232      _Base(_Tuple_impl<_Idx, _UElements...>::_M_head(__in)) { }
   234       template<typename _UHead, typename... _UTails>
   235         constexpr _Tuple_impl(_Tuple_impl<_Idx, _UHead, _UTails...>&& __in)
   236    : _Inherited(std::move
   237                 (_Tuple_impl<_Idx, _UHead, _UTails...>::_M_tail(__in))),
   238      _Base(std::forward<_UHead>
   239            (_Tuple_impl<_Idx, _UHead, _UTails...>::_M_head(__in))) { }
   241       template<typename _Alloc>
   242    _Tuple_impl(allocator_arg_t __tag, const _Alloc& __a)
   243    : _Inherited(__tag, __a),
   244           _Base(__tag, __use_alloc<_Head>(__a)) { }
   246       template<typename _Alloc>
   247    _Tuple_impl(allocator_arg_t __tag, const _Alloc& __a,
   248                const _Head& __head, const _Tail&... __tail)
   249    : _Inherited(__tag, __a, __tail...),
   250           _Base(__use_alloc<_Head, _Alloc, _Head>(__a), __head) { }
   252       template<typename _Alloc, typename _UHead, typename... _UTail,
   253                typename = typename enable_if<sizeof...(_Tail)
   254                                         == sizeof...(_UTail)>::type>
   255    _Tuple_impl(allocator_arg_t __tag, const _Alloc& __a,
   256                _UHead&& __head, _UTail&&... __tail)
   257    : _Inherited(__tag, __a, std::forward<_UTail>(__tail)...),
   258           _Base(__use_alloc<_Head, _Alloc, _UHead>(__a),
   259            std::forward<_UHead>(__head)) { }
   261       template<typename _Alloc>
   262         _Tuple_impl(allocator_arg_t __tag, const _Alloc& __a,
   263                const _Tuple_impl& __in)
   264    : _Inherited(__tag, __a, _M_tail(__in)),
   265           _Base(__use_alloc<_Head, _Alloc, _Head>(__a), _M_head(__in)) { }
   267       template<typename _Alloc>
   268    _Tuple_impl(allocator_arg_t __tag, const _Alloc& __a,
   270    : _Inherited(__tag, __a, std::move(_M_tail(__in))),
   271      _Base(__use_alloc<_Head, _Alloc, _Head>(__a),
   272            std::forward<_Head>(_M_head(__in))) { }
   274       template<typename _Alloc, typename... _UElements>
   275    _Tuple_impl(allocator_arg_t __tag, const _Alloc& __a,
   276                const _Tuple_impl<_Idx, _UElements...>& __in)
   277    : _Inherited(__tag, __a,
   278                 _Tuple_impl<_Idx, _UElements...>::_M_tail(__in)),
   279      _Base(__use_alloc<_Head, _Alloc, _Head>(__a),
   280            _Tuple_impl<_Idx, _UElements...>::_M_head(__in)) { }
   282       template<typename _Alloc, typename _UHead, typename... _UTails>
   283    _Tuple_impl(allocator_arg_t __tag, const _Alloc& __a,
   284                _Tuple_impl<_Idx, _UHead, _UTails...>&& __in)
   285    : _Inherited(__tag, __a, std::move
   286                 (_Tuple_impl<_Idx, _UHead, _UTails...>::_M_tail(__in))),
   287      _Base(__use_alloc<_Head, _Alloc, _UHead>(__a),
   289            (_Tuple_impl<_Idx, _UHead, _UTails...>::_M_head(__in))) { }
   292       operator=(const _Tuple_impl& __in)
   294    _M_head(*this) = _M_head(__in);
   295    _M_tail(*this) = _M_tail(__in);
   300       operator=(_Tuple_impl&& __in)
   301       noexcept(__and_<is_nothrow_move_assignable<_Head>,
   302                  is_nothrow_move_assignable<_Inherited>>::value)
   304    _M_head(*this) = std::forward<_Head>(_M_head(__in));
   305    _M_tail(*this) = std::move(_M_tail(__in));
   309       template<typename... _UElements>
   311         operator=(const _Tuple_impl<_Idx, _UElements...>& __in)
   313      _M_head(*this) = _Tuple_impl<_Idx, _UElements...>::_M_head(__in);
   314      _M_tail(*this) = _Tuple_impl<_Idx, _UElements...>::_M_tail(__in);
   318       template<typename _UHead, typename... _UTails>
   320         operator=(_Tuple_impl<_Idx, _UHead, _UTails...>&& __in)
   322      _M_head(*this) = std::forward<_UHead>
   323        (_Tuple_impl<_Idx, _UHead, _UTails...>::_M_head(__in));
   324      _M_tail(*this) = std::move
   325        (_Tuple_impl<_Idx, _UHead, _UTails...>::_M_tail(__in));
   331       _M_swap(_Tuple_impl& __in)
   332       noexcept(__is_nothrow_swappable<_Head>::value
   333                && noexcept(_M_tail(__in)._M_swap(_M_tail(__in))))
   336    swap(_M_head(*this), _M_head(__in));
   337    _Inherited::_M_swap(_M_tail(__in));
   341   // Basis case of inheritance recursion.
   342   template<std::size_t _Idx, typename _Head>
   343     struct _Tuple_impl<_Idx, _Head>
   344     : private _Head_base<_Idx, _Head>
   346       template<std::size_t, typename...> friend class _Tuple_impl;
   348       typedef _Head_base<_Idx, _Head> _Base;
   350       static constexpr _Head&
   351       _M_head(_Tuple_impl& __t) noexcept { return _Base::_M_head(__t); }
   353       static constexpr const _Head&
   354       _M_head(const _Tuple_impl& __t) noexcept { return _Base::_M_head(__t); }
   356       constexpr _Tuple_impl()
   360       constexpr _Tuple_impl(const _Head& __head)
   363       template<typename _UHead>
   365         constexpr _Tuple_impl(_UHead&& __head)
   366    : _Base(std::forward<_UHead>(__head)) { }
   368       constexpr _Tuple_impl(const _Tuple_impl&) = default;
   371       _Tuple_impl(_Tuple_impl&& __in)
   372       noexcept(is_nothrow_move_constructible<_Head>::value)
   373       : _Base(std::forward<_Head>(_M_head(__in))) { }
   375       template<typename _UHead>
   376         constexpr _Tuple_impl(const _Tuple_impl<_Idx, _UHead>& __in)
   377    : _Base(_Tuple_impl<_Idx, _UHead>::_M_head(__in)) { }
   379       template<typename _UHead>
   380         constexpr _Tuple_impl(_Tuple_impl<_Idx, _UHead>&& __in)
   381    : _Base(std::forward<_UHead>(_Tuple_impl<_Idx, _UHead>::_M_head(__in)))
   384       template<typename _Alloc>
   385    _Tuple_impl(allocator_arg_t __tag, const _Alloc& __a)
   386    : _Base(__tag, __use_alloc<_Head>(__a)) { }
   388       template<typename _Alloc>
   389    _Tuple_impl(allocator_arg_t __tag, const _Alloc& __a,
   391    : _Base(__use_alloc<_Head, _Alloc, _Head>(__a), __head) { }
   393       template<typename _Alloc, typename _UHead>
   394    _Tuple_impl(allocator_arg_t __tag, const _Alloc& __a,
   396    : _Base(__use_alloc<_Head, _Alloc, _UHead>(__a),
   397            std::forward<_UHead>(__head)) { }
   399       template<typename _Alloc>
   400         _Tuple_impl(allocator_arg_t __tag, const _Alloc& __a,
   401                const _Tuple_impl& __in)
   402    : _Base(__use_alloc<_Head, _Alloc, _Head>(__a), _M_head(__in)) { }
   404       template<typename _Alloc>
   405    _Tuple_impl(allocator_arg_t __tag, const _Alloc& __a,
   407    : _Base(__use_alloc<_Head, _Alloc, _Head>(__a),
   408            std::forward<_Head>(_M_head(__in))) { }
   410       template<typename _Alloc, typename _UHead>
   411    _Tuple_impl(allocator_arg_t __tag, const _Alloc& __a,
   412                const _Tuple_impl<_Idx, _UHead>& __in)
   413    : _Base(__use_alloc<_Head, _Alloc, _Head>(__a),
   414            _Tuple_impl<_Idx, _UHead>::_M_head(__in)) { }
   416       template<typename _Alloc, typename _UHead>
   417    _Tuple_impl(allocator_arg_t __tag, const _Alloc& __a,
   418                _Tuple_impl<_Idx, _UHead>&& __in)
   419    : _Base(__use_alloc<_Head, _Alloc, _UHead>(__a),
   420                 std::forward<_UHead>(_Tuple_impl<_Idx, _UHead>::_M_head(__in)))
   424       operator=(const _Tuple_impl& __in)
   426    _M_head(*this) = _M_head(__in);
   431       operator=(_Tuple_impl&& __in)
   432       noexcept(is_nothrow_move_assignable<_Head>::value)
   434    _M_head(*this) = std::forward<_Head>(_M_head(__in));
   438       template<typename _UHead>
   440         operator=(const _Tuple_impl<_Idx, _UHead>& __in)
   442      _M_head(*this) = _Tuple_impl<_Idx, _UHead>::_M_head(__in);
   446       template<typename _UHead>
   448         operator=(_Tuple_impl<_Idx, _UHead>&& __in)
   451        = std::forward<_UHead>(_Tuple_impl<_Idx, _UHead>::_M_head(__in));
   457       _M_swap(_Tuple_impl& __in)
   458       noexcept(__is_nothrow_swappable<_Head>::value)
   461    swap(_M_head(*this), _M_head(__in));
   465   // Concept utility functions, reused in conditionally-explicit
   467   template<bool, typename... _Elements>
   470     template<typename... _UElements>
   471     static constexpr bool _ConstructibleTuple()
   473       return __and_<is_constructible<_Elements, const _UElements&>...>::value;
   476     template<typename... _UElements>
   477     static constexpr bool _ImplicitlyConvertibleTuple()
   479       return __and_<is_convertible<const _UElements&, _Elements>...>::value;
   482     template<typename... _UElements>
   483     static constexpr bool _MoveConstructibleTuple()
   485       return __and_<is_constructible<_Elements, _UElements&&>...>::value;
   488     template<typename... _UElements>
   489     static constexpr bool _ImplicitlyMoveConvertibleTuple()
   491       return __and_<is_convertible<_UElements&&, _Elements>...>::value;
   494     template<typename _SrcTuple>
   495     static constexpr bool _NonNestedTuple()
   497       return  __and_<__not_<is_same<tuple<_Elements...>,
   499                                      typename remove_reference<_SrcTuple>::type
   501                      __not_<is_convertible<_SrcTuple, _Elements...>>,
   502                      __not_<is_constructible<_Elements..., _SrcTuple>>
   505     template<typename... _UElements>
   506     static constexpr bool _NotSameTuple()
   508       return  __not_<is_same<tuple<_Elements...>,
   509                         typename remove_const<
   510                           typename remove_reference<_UElements...>::type
   515   template<typename... _Elements>
   516   struct _TC<false, _Elements...>
   518     template<typename... _UElements>
   519     static constexpr bool _ConstructibleTuple()
   524     template<typename... _UElements>
   525     static constexpr bool _ImplicitlyConvertibleTuple()
   530     template<typename... _UElements>
   531     static constexpr bool _MoveConstructibleTuple()
   536     template<typename... _UElements>
   537     static constexpr bool _ImplicitlyMoveConvertibleTuple()
   542     template<typename... _UElements>
   543     static constexpr bool _NonNestedTuple()
   547     template<typename... _UElements>
   548     static constexpr bool _NotSameTuple()
   554   /// Primary class template, tuple
   555   template<typename... _Elements>
   556     class tuple : public _Tuple_impl<0, _Elements...>
   558       typedef _Tuple_impl<0, _Elements...> _Inherited;
   560       // Used for constraining the default constructor so
   561       // that it becomes dependent on the constraints.
   562       template<typename _Dummy>
   565         static constexpr bool _DefaultConstructibleTuple()
   567           return __and_<is_default_constructible<_Elements>...>::value;
   569         static constexpr bool _ImplicitlyDefaultConstructibleTuple()
   571           return __and_<__is_implicitly_default_constructible<_Elements>...>
   577       template<typename _Dummy = void,
   578                typename enable_if<_TC2<_Dummy>::
   579                                     _ImplicitlyDefaultConstructibleTuple(),
   584       template<typename _Dummy = void,
   585                typename enable_if<_TC2<_Dummy>::
   586                                     _DefaultConstructibleTuple()
   589                                     _ImplicitlyDefaultConstructibleTuple(),
   591       explicit constexpr tuple()
   594       // Shortcut for the cases where constructors taking _Elements...
   595       // need to be constrained.
   596       template<typename _Dummy> using _TCC =
   597         _TC<is_same<_Dummy, void>::value,
   600       template<typename _Dummy = void,
   602                  _TCC<_Dummy>::template
   603                    _ConstructibleTuple<_Elements...>()
   604                  && _TCC<_Dummy>::template
   605                    _ImplicitlyConvertibleTuple<_Elements...>()
   606                  && (sizeof...(_Elements) >= 1),
   608         constexpr tuple(const _Elements&... __elements)
   609       : _Inherited(__elements...) { }
   611       template<typename _Dummy = void,
   613                  _TCC<_Dummy>::template
   614                    _ConstructibleTuple<_Elements...>()
   615                  && !_TCC<_Dummy>::template
   616                    _ImplicitlyConvertibleTuple<_Elements...>()
   617                  && (sizeof...(_Elements) >= 1),
   619       explicit constexpr tuple(const _Elements&... __elements)
   620       : _Inherited(__elements...) { }
   622       // Shortcut for the cases where constructors taking _UElements...
   623       // need to be constrained.
   624       template<typename... _UElements> using _TMC =
   625                   _TC<(sizeof...(_Elements) == sizeof...(_UElements))
   626                  && (_TC<(sizeof...(_UElements)==1), _Elements...>::
   627                      template _NotSameTuple<_UElements...>()),
   630       // Shortcut for the cases where constructors taking tuple<_UElements...>
   631       // need to be constrained.
   632       template<typename... _UElements> using _TMCT =
   633                   _TC<(sizeof...(_Elements) == sizeof...(_UElements))
   634                  && !is_same<tuple<_Elements...>,
   635                              tuple<_UElements...>>::value,
   638       template<typename... _UElements, typename
   640              _TMC<_UElements...>::template
   641                     _MoveConstructibleTuple<_UElements...>()
   642                   && _TMC<_UElements...>::template
   643                     _ImplicitlyMoveConvertibleTuple<_UElements...>()
   644                   && (sizeof...(_Elements) >= 1),
   646         constexpr tuple(_UElements&&... __elements)
   647         : _Inherited(std::forward<_UElements>(__elements)...) { }
   649       template<typename... _UElements, typename
   651              _TMC<_UElements...>::template
   652                     _MoveConstructibleTuple<_UElements...>()
   653                   && !_TMC<_UElements...>::template
   654                     _ImplicitlyMoveConvertibleTuple<_UElements...>()
   655                   && (sizeof...(_Elements) >= 1),
   657         explicit constexpr tuple(_UElements&&... __elements)
   658    : _Inherited(std::forward<_UElements>(__elements)...) { }
   660       constexpr tuple(const tuple&) = default;
   662       constexpr tuple(tuple&&) = default;
   664       // Shortcut for the cases where constructors taking tuples
   665       // must avoid creating temporaries.
   666       template<typename _Dummy> using _TNTC =
   667         _TC<is_same<_Dummy, void>::value && sizeof...(_Elements) == 1,
   670       template<typename... _UElements, typename _Dummy = void, typename
   671         enable_if<_TMCT<_UElements...>::template
   672                     _ConstructibleTuple<_UElements...>()
   673                   && _TMCT<_UElements...>::template
   674                     _ImplicitlyConvertibleTuple<_UElements...>()
   675                   && _TNTC<_Dummy>::template
   676                     _NonNestedTuple<const tuple<_UElements...>&>(),
   678         constexpr tuple(const tuple<_UElements...>& __in)
   679         : _Inherited(static_cast<const _Tuple_impl<0, _UElements...>&>(__in))
   682       template<typename... _UElements, typename _Dummy = void, typename
   683         enable_if<_TMCT<_UElements...>::template
   684                     _ConstructibleTuple<_UElements...>()
   685                   && !_TMCT<_UElements...>::template
   686                     _ImplicitlyConvertibleTuple<_UElements...>()
   687                   && _TNTC<_Dummy>::template
   688                     _NonNestedTuple<const tuple<_UElements...>&>(),
   690         explicit constexpr tuple(const tuple<_UElements...>& __in)
   691         : _Inherited(static_cast<const _Tuple_impl<0, _UElements...>&>(__in))
   694       template<typename... _UElements, typename _Dummy = void, typename
   695         enable_if<_TMCT<_UElements...>::template
   696                     _MoveConstructibleTuple<_UElements...>()
   697                   && _TMCT<_UElements...>::template
   698                     _ImplicitlyMoveConvertibleTuple<_UElements...>()
   699                   && _TNTC<_Dummy>::template
   700                     _NonNestedTuple<tuple<_UElements...>&&>(),
   702         constexpr tuple(tuple<_UElements...>&& __in)
   703         : _Inherited(static_cast<_Tuple_impl<0, _UElements...>&&>(__in)) { }
   705       template<typename... _UElements, typename _Dummy = void, typename
   706         enable_if<_TMCT<_UElements...>::template
   707                     _MoveConstructibleTuple<_UElements...>()
   708                   && !_TMCT<_UElements...>::template
   709                     _ImplicitlyMoveConvertibleTuple<_UElements...>()
   710                   && _TNTC<_Dummy>::template
   711                     _NonNestedTuple<tuple<_UElements...>&&>(),
   713         explicit constexpr tuple(tuple<_UElements...>&& __in)
   714         : _Inherited(static_cast<_Tuple_impl<0, _UElements...>&&>(__in)) { }
   716       // Allocator-extended constructors.
   718       template<typename _Alloc>
   719    tuple(allocator_arg_t __tag, const _Alloc& __a)
   720    : _Inherited(__tag, __a) { }
   722       template<typename _Alloc, typename _Dummy = void,
   724                  _TCC<_Dummy>::template
   725                    _ConstructibleTuple<_Elements...>()
   726                  && _TCC<_Dummy>::template
   727                    _ImplicitlyConvertibleTuple<_Elements...>(),
   729    tuple(allocator_arg_t __tag, const _Alloc& __a,
   730          const _Elements&... __elements)
   731    : _Inherited(__tag, __a, __elements...) { }
   733       template<typename _Alloc, typename _Dummy = void,
   735                  _TCC<_Dummy>::template
   736                    _ConstructibleTuple<_Elements...>()
   737                  && !_TCC<_Dummy>::template
   738                    _ImplicitlyConvertibleTuple<_Elements...>(),
   740    explicit tuple(allocator_arg_t __tag, const _Alloc& __a,
   741                        const _Elements&... __elements)
   742    : _Inherited(__tag, __a, __elements...) { }
   744       template<typename _Alloc, typename... _UElements, typename
   745         enable_if<_TMC<_UElements...>::template
   746                     _MoveConstructibleTuple<_UElements...>()
   747                   && _TMC<_UElements...>::template
   748                     _ImplicitlyMoveConvertibleTuple<_UElements...>(),
   750    tuple(allocator_arg_t __tag, const _Alloc& __a,
   751          _UElements&&... __elements)
   752    : _Inherited(__tag, __a, std::forward<_UElements>(__elements)...)
   755       template<typename _Alloc, typename... _UElements, typename
   756         enable_if<_TMC<_UElements...>::template
   757                     _MoveConstructibleTuple<_UElements...>()
   758                   && !_TMC<_UElements...>::template
   759                     _ImplicitlyMoveConvertibleTuple<_UElements...>(),
   761    explicit tuple(allocator_arg_t __tag, const _Alloc& __a,
   762          _UElements&&... __elements)
   763    : _Inherited(__tag, __a, std::forward<_UElements>(__elements)...)
   766       template<typename _Alloc>
   767    tuple(allocator_arg_t __tag, const _Alloc& __a, const tuple& __in)
   768    : _Inherited(__tag, __a, static_cast<const _Inherited&>(__in)) { }
   770       template<typename _Alloc>
   771    tuple(allocator_arg_t __tag, const _Alloc& __a, tuple&& __in)
   772    : _Inherited(__tag, __a, static_cast<_Inherited&&>(__in)) { }
   774       template<typename _Alloc, typename _Dummy = void,
   775           typename... _UElements, typename
   776         enable_if<_TMCT<_UElements...>::template
   777                     _ConstructibleTuple<_UElements...>()
   778                   && _TMCT<_UElements...>::template
   779                     _ImplicitlyConvertibleTuple<_UElements...>()
   780                   && _TNTC<_Dummy>::template
   781                     _NonNestedTuple<tuple<_UElements...>&&>(),
   783    tuple(allocator_arg_t __tag, const _Alloc& __a,
   784          const tuple<_UElements...>& __in)
   785    : _Inherited(__tag, __a,
   786                 static_cast<const _Tuple_impl<0, _UElements...>&>(__in))
   789       template<typename _Alloc, typename _Dummy = void,
   790           typename... _UElements, typename
   791         enable_if<_TMCT<_UElements...>::template
   792                     _ConstructibleTuple<_UElements...>()
   793                   && !_TMCT<_UElements...>::template
   794                     _ImplicitlyConvertibleTuple<_UElements...>()
   795                   && _TNTC<_Dummy>::template
   796                     _NonNestedTuple<tuple<_UElements...>&&>(),
   798    explicit tuple(allocator_arg_t __tag, const _Alloc& __a,
   799          const tuple<_UElements...>& __in)
   800    : _Inherited(__tag, __a,
   801                 static_cast<const _Tuple_impl<0, _UElements...>&>(__in))
   804       template<typename _Alloc, typename _Dummy = void,
   805           typename... _UElements, typename
   806         enable_if<_TMCT<_UElements...>::template
   807                     _MoveConstructibleTuple<_UElements...>()
   808                   && _TMCT<_UElements...>::template
   809                     _ImplicitlyMoveConvertibleTuple<_UElements...>()
   810                   && _TNTC<_Dummy>::template
   811                     _NonNestedTuple<tuple<_UElements...>&&>(),
   813    tuple(allocator_arg_t __tag, const _Alloc& __a,
   814          tuple<_UElements...>&& __in)
   815    : _Inherited(__tag, __a,
   816                 static_cast<_Tuple_impl<0, _UElements...>&&>(__in))
   819       template<typename _Alloc, typename _Dummy = void,
   820           typename... _UElements, typename
   821         enable_if<_TMCT<_UElements...>::template
   822                     _MoveConstructibleTuple<_UElements...>()
   823                   && !_TMCT<_UElements...>::template
   824                     _ImplicitlyMoveConvertibleTuple<_UElements...>()
   825                   && _TNTC<_Dummy>::template
   826                     _NonNestedTuple<tuple<_UElements...>&&>(),
   828    explicit tuple(allocator_arg_t __tag, const _Alloc& __a,
   829          tuple<_UElements...>&& __in)
   830    : _Inherited(__tag, __a,
   831                 static_cast<_Tuple_impl<0, _UElements...>&&>(__in))
   835       operator=(const tuple& __in)
   837    static_cast<_Inherited&>(*this) = __in;
   842       operator=(tuple&& __in)
   843       noexcept(is_nothrow_move_assignable<_Inherited>::value)
   845    static_cast<_Inherited&>(*this) = std::move(__in);
   849       template<typename... _UElements>
   851           enable_if<sizeof...(_UElements)
   852                     == sizeof...(_Elements), tuple&>::type
   853         operator=(const tuple<_UElements...>& __in)
   855      static_cast<_Inherited&>(*this) = __in;
   859       template<typename... _UElements>
   861           enable_if<sizeof...(_UElements)
   862                     == sizeof...(_Elements), tuple&>::type
   863         operator=(tuple<_UElements...>&& __in)
   865      static_cast<_Inherited&>(*this) = std::move(__in);
   871       noexcept(noexcept(__in._M_swap(__in)))
   872       { _Inherited::_M_swap(__in); }
   875 #if __cpp_deduction_guides >= 201606
   876   template<typename... _UTypes>
   877     tuple(_UTypes...) -> tuple<_UTypes...>;
   878   template<typename _T1, typename _T2>
   879     tuple(pair<_T1, _T2>) -> tuple<_T1, _T2>;
   880   template<typename _Alloc, typename... _UTypes>
   881     tuple(allocator_arg_t, _Alloc, _UTypes...) -> tuple<_UTypes...>;
   882   template<typename _Alloc, typename _T1, typename _T2>
   883     tuple(allocator_arg_t, _Alloc, pair<_T1, _T2>) -> tuple<_T1, _T2>;
   884   template<typename _Alloc, typename... _UTypes>
   885     tuple(allocator_arg_t, _Alloc, tuple<_UTypes...>) -> tuple<_UTypes...>;
   888   // Explicit specialization, zero-element tuple.
   893       void swap(tuple&) noexcept { /* no-op */ }
   894       // We need the default since we're going to define no-op
   895       // allocator constructors.
   897       // No-op allocator constructors.
   898       template<typename _Alloc>
   899    tuple(allocator_arg_t, const _Alloc&) { }
   900       template<typename _Alloc>
   901    tuple(allocator_arg_t, const _Alloc&, const tuple&) { }
   904   /// Partial specialization, 2-element tuple.
   905   /// Includes construction and assignment from a pair.
   906   template<typename _T1, typename _T2>
   907     class tuple<_T1, _T2> : public _Tuple_impl<0, _T1, _T2>
   909       typedef _Tuple_impl<0, _T1, _T2> _Inherited;
   912       template <typename _U1 = _T1,
   914                 typename enable_if<__and_<
   915                                      __is_implicitly_default_constructible<_U1>,
   916                                      __is_implicitly_default_constructible<_U2>>
   917                                    ::value, bool>::type = true>
   922       template <typename _U1 = _T1,
   926                     is_default_constructible<_U1>,
   927                     is_default_constructible<_U2>,
   929                       __and_<__is_implicitly_default_constructible<_U1>,
   930                              __is_implicitly_default_constructible<_U2>>>>
   931                   ::value, bool>::type = false>
   933       explicit constexpr tuple()
   936       // Shortcut for the cases where constructors taking _T1, _T2
   937       // need to be constrained.
   938       template<typename _Dummy> using _TCC =
   939         _TC<is_same<_Dummy, void>::value, _T1, _T2>;
   941       template<typename _Dummy = void, typename
   942                enable_if<_TCC<_Dummy>::template
   943                            _ConstructibleTuple<_T1, _T2>()
   944                          && _TCC<_Dummy>::template
   945                            _ImplicitlyConvertibleTuple<_T1, _T2>(),
   947         constexpr tuple(const _T1& __a1, const _T2& __a2)
   948         : _Inherited(__a1, __a2) { }
   950       template<typename _Dummy = void, typename
   951                enable_if<_TCC<_Dummy>::template
   952                            _ConstructibleTuple<_T1, _T2>()
   953                          && !_TCC<_Dummy>::template
   954                            _ImplicitlyConvertibleTuple<_T1, _T2>(),
   956         explicit constexpr tuple(const _T1& __a1, const _T2& __a2)
   957         : _Inherited(__a1, __a2) { }
   959       // Shortcut for the cases where constructors taking _U1, _U2
   960       // need to be constrained.
   961       using _TMC = _TC<true, _T1, _T2>;
   963       template<typename _U1, typename _U2, typename
   964         enable_if<_TMC::template
   965                     _MoveConstructibleTuple<_U1, _U2>()
   967                     _ImplicitlyMoveConvertibleTuple<_U1, _U2>()
   968              && !is_same<typename decay<_U1>::type,
   969                          allocator_arg_t>::value,
   971         constexpr tuple(_U1&& __a1, _U2&& __a2)
   972    : _Inherited(std::forward<_U1>(__a1), std::forward<_U2>(__a2)) { }
   974       template<typename _U1, typename _U2, typename
   975         enable_if<_TMC::template
   976                     _MoveConstructibleTuple<_U1, _U2>()
   978                     _ImplicitlyMoveConvertibleTuple<_U1, _U2>()
   979              && !is_same<typename decay<_U1>::type,
   980                          allocator_arg_t>::value,
   982         explicit constexpr tuple(_U1&& __a1, _U2&& __a2)
   983    : _Inherited(std::forward<_U1>(__a1), std::forward<_U2>(__a2)) { }
   985       constexpr tuple(const tuple&) = default;
   987       constexpr tuple(tuple&&) = default;
   989       template<typename _U1, typename _U2, typename
   990         enable_if<_TMC::template
   991                     _ConstructibleTuple<_U1, _U2>()
   993                     _ImplicitlyConvertibleTuple<_U1, _U2>(),
   995         constexpr tuple(const tuple<_U1, _U2>& __in)
   996    : _Inherited(static_cast<const _Tuple_impl<0, _U1, _U2>&>(__in)) { }
   998       template<typename _U1, typename _U2, typename
   999         enable_if<_TMC::template
  1000                     _ConstructibleTuple<_U1, _U2>()
  1002                     _ImplicitlyConvertibleTuple<_U1, _U2>(),
  1003    bool>::type = false>
  1004         explicit constexpr tuple(const tuple<_U1, _U2>& __in)
  1005    : _Inherited(static_cast<const _Tuple_impl<0, _U1, _U2>&>(__in)) { }
  1007       template<typename _U1, typename _U2, typename
  1008         enable_if<_TMC::template
  1009                     _MoveConstructibleTuple<_U1, _U2>()
  1011                     _ImplicitlyMoveConvertibleTuple<_U1, _U2>(),
  1013         constexpr tuple(tuple<_U1, _U2>&& __in)
  1014    : _Inherited(static_cast<_Tuple_impl<0, _U1, _U2>&&>(__in)) { }
  1016       template<typename _U1, typename _U2, typename
  1017         enable_if<_TMC::template
  1018                     _MoveConstructibleTuple<_U1, _U2>()
  1020                     _ImplicitlyMoveConvertibleTuple<_U1, _U2>(),
  1021    bool>::type = false>
  1022         explicit constexpr tuple(tuple<_U1, _U2>&& __in)
  1023    : _Inherited(static_cast<_Tuple_impl<0, _U1, _U2>&&>(__in)) { }
  1025       template<typename _U1, typename _U2, typename
  1026         enable_if<_TMC::template
  1027                     _ConstructibleTuple<_U1, _U2>()
  1029                     _ImplicitlyConvertibleTuple<_U1, _U2>(),
  1031         constexpr tuple(const pair<_U1, _U2>& __in)
  1032    : _Inherited(__in.first, __in.second) { }
  1034       template<typename _U1, typename _U2, typename
  1035         enable_if<_TMC::template
  1036                     _ConstructibleTuple<_U1, _U2>()
  1038                     _ImplicitlyConvertibleTuple<_U1, _U2>(),
  1039    bool>::type = false>
  1040         explicit constexpr tuple(const pair<_U1, _U2>& __in)
  1041    : _Inherited(__in.first, __in.second) { }
  1043       template<typename _U1, typename _U2, typename
  1044         enable_if<_TMC::template
  1045                     _MoveConstructibleTuple<_U1, _U2>()
  1047                     _ImplicitlyMoveConvertibleTuple<_U1, _U2>(),
  1049         constexpr tuple(pair<_U1, _U2>&& __in)
  1050    : _Inherited(std::forward<_U1>(__in.first),
  1051                 std::forward<_U2>(__in.second)) { }
  1053       template<typename _U1, typename _U2, typename
  1054         enable_if<_TMC::template
  1055                     _MoveConstructibleTuple<_U1, _U2>()
  1057                     _ImplicitlyMoveConvertibleTuple<_U1, _U2>(),
  1058    bool>::type = false>
  1059         explicit constexpr tuple(pair<_U1, _U2>&& __in)
  1060    : _Inherited(std::forward<_U1>(__in.first),
  1061                 std::forward<_U2>(__in.second)) { }
  1063       // Allocator-extended constructors.
  1065       template<typename _Alloc>
  1066    tuple(allocator_arg_t __tag, const _Alloc& __a)
  1067    : _Inherited(__tag, __a) { }
  1069       template<typename _Alloc, typename _Dummy = void,
  1071                  _TCC<_Dummy>::template
  1072                    _ConstructibleTuple<_T1, _T2>()
  1073                  && _TCC<_Dummy>::template
  1074                    _ImplicitlyConvertibleTuple<_T1, _T2>(),
  1077    tuple(allocator_arg_t __tag, const _Alloc& __a,
  1078          const _T1& __a1, const _T2& __a2)
  1079    : _Inherited(__tag, __a, __a1, __a2) { }
  1081       template<typename _Alloc, typename _Dummy = void,
  1083                  _TCC<_Dummy>::template
  1084                    _ConstructibleTuple<_T1, _T2>()
  1085                  && !_TCC<_Dummy>::template
  1086                    _ImplicitlyConvertibleTuple<_T1, _T2>(),
  1089    explicit tuple(allocator_arg_t __tag, const _Alloc& __a,
  1090          const _T1& __a1, const _T2& __a2)
  1091    : _Inherited(__tag, __a, __a1, __a2) { }
  1093       template<typename _Alloc, typename _U1, typename _U2, typename
  1094         enable_if<_TMC::template
  1095                     _MoveConstructibleTuple<_U1, _U2>()
  1097                     _ImplicitlyMoveConvertibleTuple<_U1, _U2>(),
  1099    tuple(allocator_arg_t __tag, const _Alloc& __a, _U1&& __a1, _U2&& __a2)
  1100    : _Inherited(__tag, __a, std::forward<_U1>(__a1),
  1101                 std::forward<_U2>(__a2)) { }
  1103       template<typename _Alloc, typename _U1, typename _U2, typename
  1104         enable_if<_TMC::template
  1105                     _MoveConstructibleTuple<_U1, _U2>()
  1107                     _ImplicitlyMoveConvertibleTuple<_U1, _U2>(),
  1108    bool>::type = false>
  1109    explicit tuple(allocator_arg_t __tag, const _Alloc& __a,
  1110                        _U1&& __a1, _U2&& __a2)
  1111    : _Inherited(__tag, __a, std::forward<_U1>(__a1),
  1112                 std::forward<_U2>(__a2)) { }
  1114       template<typename _Alloc>
  1115    tuple(allocator_arg_t __tag, const _Alloc& __a, const tuple& __in)
  1116    : _Inherited(__tag, __a, static_cast<const _Inherited&>(__in)) { }
  1118       template<typename _Alloc>
  1119    tuple(allocator_arg_t __tag, const _Alloc& __a, tuple&& __in)
  1120    : _Inherited(__tag, __a, static_cast<_Inherited&&>(__in)) { }
  1122       template<typename _Alloc, typename _U1, typename _U2, typename
  1123         enable_if<_TMC::template
  1124                     _ConstructibleTuple<_U1, _U2>()
  1126                     _ImplicitlyConvertibleTuple<_U1, _U2>(),
  1128    tuple(allocator_arg_t __tag, const _Alloc& __a,
  1129          const tuple<_U1, _U2>& __in)
  1130    : _Inherited(__tag, __a,
  1131                 static_cast<const _Tuple_impl<0, _U1, _U2>&>(__in))
  1134       template<typename _Alloc, typename _U1, typename _U2, typename
  1135         enable_if<_TMC::template
  1136                     _ConstructibleTuple<_U1, _U2>()
  1138                     _ImplicitlyConvertibleTuple<_U1, _U2>(),
  1139    bool>::type = false>
  1140    explicit tuple(allocator_arg_t __tag, const _Alloc& __a,
  1141          const tuple<_U1, _U2>& __in)
  1142    : _Inherited(__tag, __a,
  1143                 static_cast<const _Tuple_impl<0, _U1, _U2>&>(__in))
  1146       template<typename _Alloc, typename _U1, typename _U2, typename
  1147         enable_if<_TMC::template
  1148                     _MoveConstructibleTuple<_U1, _U2>()
  1150                     _ImplicitlyMoveConvertibleTuple<_U1, _U2>(),
  1152    tuple(allocator_arg_t __tag, const _Alloc& __a, tuple<_U1, _U2>&& __in)
  1153    : _Inherited(__tag, __a, static_cast<_Tuple_impl<0, _U1, _U2>&&>(__in))
  1156       template<typename _Alloc, typename _U1, typename _U2, typename
  1157         enable_if<_TMC::template
  1158                     _MoveConstructibleTuple<_U1, _U2>()
  1160                     _ImplicitlyMoveConvertibleTuple<_U1, _U2>(),
  1161    bool>::type = false>
  1162    explicit tuple(allocator_arg_t __tag, const _Alloc& __a,
  1163                        tuple<_U1, _U2>&& __in)
  1164    : _Inherited(__tag, __a, static_cast<_Tuple_impl<0, _U1, _U2>&&>(__in))
  1167       template<typename _Alloc, typename _U1, typename _U2, typename
  1168         enable_if<_TMC::template
  1169                     _ConstructibleTuple<_U1, _U2>()
  1171                     _ImplicitlyConvertibleTuple<_U1, _U2>(),
  1173         tuple(allocator_arg_t __tag, const _Alloc& __a,
  1174          const pair<_U1, _U2>& __in)
  1175    : _Inherited(__tag, __a, __in.first, __in.second) { }
  1177       template<typename _Alloc, typename _U1, typename _U2, typename
  1178         enable_if<_TMC::template
  1179                     _ConstructibleTuple<_U1, _U2>()
  1181                     _ImplicitlyConvertibleTuple<_U1, _U2>(),
  1182    bool>::type = false>
  1183         explicit tuple(allocator_arg_t __tag, const _Alloc& __a,
  1184          const pair<_U1, _U2>& __in)
  1185    : _Inherited(__tag, __a, __in.first, __in.second) { }
  1187       template<typename _Alloc, typename _U1, typename _U2, typename
  1188         enable_if<_TMC::template
  1189                     _MoveConstructibleTuple<_U1, _U2>()
  1191                     _ImplicitlyMoveConvertibleTuple<_U1, _U2>(),
  1193         tuple(allocator_arg_t __tag, const _Alloc& __a, pair<_U1, _U2>&& __in)
  1194    : _Inherited(__tag, __a, std::forward<_U1>(__in.first),
  1195                 std::forward<_U2>(__in.second)) { }
  1197       template<typename _Alloc, typename _U1, typename _U2, typename
  1198         enable_if<_TMC::template
  1199                     _MoveConstructibleTuple<_U1, _U2>()
  1201                     _ImplicitlyMoveConvertibleTuple<_U1, _U2>(),
  1202    bool>::type = false>
  1203         explicit tuple(allocator_arg_t __tag, const _Alloc& __a,
  1204                        pair<_U1, _U2>&& __in)
  1205    : _Inherited(__tag, __a, std::forward<_U1>(__in.first),
  1206                 std::forward<_U2>(__in.second)) { }
  1209       operator=(const tuple& __in)
  1211    static_cast<_Inherited&>(*this) = __in;
  1216       operator=(tuple&& __in)
  1217       noexcept(is_nothrow_move_assignable<_Inherited>::value)
  1219    static_cast<_Inherited&>(*this) = std::move(__in);
  1223       template<typename _U1, typename _U2>
  1225         operator=(const tuple<_U1, _U2>& __in)
  1227      static_cast<_Inherited&>(*this) = __in;
  1231       template<typename _U1, typename _U2>
  1233         operator=(tuple<_U1, _U2>&& __in)
  1235      static_cast<_Inherited&>(*this) = std::move(__in);
  1239       template<typename _U1, typename _U2>
  1241         operator=(const pair<_U1, _U2>& __in)
  1243      this->_M_head(*this) = __in.first;
  1244      this->_M_tail(*this)._M_head(*this) = __in.second;
  1248       template<typename _U1, typename _U2>
  1250         operator=(pair<_U1, _U2>&& __in)
  1252      this->_M_head(*this) = std::forward<_U1>(__in.first);
  1253      this->_M_tail(*this)._M_head(*this) = std::forward<_U2>(__in.second);
  1259       noexcept(noexcept(__in._M_swap(__in)))
  1260       { _Inherited::_M_swap(__in); }
  1264   /// class tuple_size
  1265   template<typename... _Elements>
  1266     struct tuple_size<tuple<_Elements...>>
  1267     : public integral_constant<std::size_t, sizeof...(_Elements)> { };
  1269 #if __cplusplus > 201402L
  1270   template <typename _Tp>
  1271     inline constexpr size_t tuple_size_v = tuple_size<_Tp>::value;
  1275    * Recursive case for tuple_element: strip off the first element in
  1276    * the tuple and retrieve the (i-1)th element of the remaining tuple.
  1278   template<std::size_t __i, typename _Head, typename... _Tail>
  1279     struct tuple_element<__i, tuple<_Head, _Tail...> >
  1280     : tuple_element<__i - 1, tuple<_Tail...> > { };
  1283    * Basis case for tuple_element: The first element is the one we're seeking.
  1285   template<typename _Head, typename... _Tail>
  1286     struct tuple_element<0, tuple<_Head, _Tail...> >
  1292    * Error case for tuple_element: invalid index.
  1294   template<size_t __i>
  1295     struct tuple_element<__i, tuple<>>
  1297       static_assert(__i < tuple_size<tuple<>>::value,
  1298      "tuple index is in range");
  1301   template<std::size_t __i, typename _Head, typename... _Tail>
  1303     __get_helper(_Tuple_impl<__i, _Head, _Tail...>& __t) noexcept
  1304     { return _Tuple_impl<__i, _Head, _Tail...>::_M_head(__t); }
  1306   template<std::size_t __i, typename _Head, typename... _Tail>
  1307     constexpr const _Head&
  1308     __get_helper(const _Tuple_impl<__i, _Head, _Tail...>& __t) noexcept
  1309     { return _Tuple_impl<__i, _Head, _Tail...>::_M_head(__t); }
  1311   /// Return a reference to the ith element of a tuple.
  1312   template<std::size_t __i, typename... _Elements>
  1313     constexpr __tuple_element_t<__i, tuple<_Elements...>>&
  1314     get(tuple<_Elements...>& __t) noexcept
  1315     { return std::__get_helper<__i>(__t); }
  1317   /// Return a const reference to the ith element of a const tuple.
  1318   template<std::size_t __i, typename... _Elements>
  1319     constexpr const __tuple_element_t<__i, tuple<_Elements...>>&
  1320     get(const tuple<_Elements...>& __t) noexcept
  1321     { return std::__get_helper<__i>(__t); }
  1323   /// Return an rvalue reference to the ith element of a tuple rvalue.
  1324   template<std::size_t __i, typename... _Elements>
  1325     constexpr __tuple_element_t<__i, tuple<_Elements...>>&&
  1326     get(tuple<_Elements...>&& __t) noexcept
  1328       typedef __tuple_element_t<__i, tuple<_Elements...>> __element_type;
  1329       return std::forward<__element_type&&>(std::get<__i>(__t));
  1332   /// Return a const rvalue reference to the ith element of a const tuple rvalue.
  1333   template<std::size_t __i, typename... _Elements>
  1334     constexpr const __tuple_element_t<__i, tuple<_Elements...>>&&
  1335     get(const tuple<_Elements...>&& __t) noexcept
  1337       typedef __tuple_element_t<__i, tuple<_Elements...>> __element_type;
  1338       return std::forward<const __element_type&&>(std::get<__i>(__t));
  1341 #if __cplusplus > 201103L
  1343 #define __cpp_lib_tuples_by_type 201304
  1345   template<typename _Head, size_t __i, typename... _Tail>
  1347     __get_helper2(_Tuple_impl<__i, _Head, _Tail...>& __t) noexcept
  1348     { return _Tuple_impl<__i, _Head, _Tail...>::_M_head(__t); }
  1350   template<typename _Head, size_t __i, typename... _Tail>
  1351     constexpr const _Head&
  1352     __get_helper2(const _Tuple_impl<__i, _Head, _Tail...>& __t) noexcept
  1353     { return _Tuple_impl<__i, _Head, _Tail...>::_M_head(__t); }
  1355   /// Return a reference to the unique element of type _Tp of a tuple.
  1356   template <typename _Tp, typename... _Types>
  1358     get(tuple<_Types...>& __t) noexcept
  1359     { return std::__get_helper2<_Tp>(__t); }
  1361   /// Return a reference to the unique element of type _Tp of a tuple rvalue.
  1362   template <typename _Tp, typename... _Types>
  1364     get(tuple<_Types...>&& __t) noexcept
  1365     { return std::forward<_Tp&&>(std::__get_helper2<_Tp>(__t)); }
  1367   /// Return a const reference to the unique element of type _Tp of a tuple.
  1368   template <typename _Tp, typename... _Types>
  1369     constexpr const _Tp&
  1370     get(const tuple<_Types...>& __t) noexcept
  1371     { return std::__get_helper2<_Tp>(__t); }
  1373   /// Return a const reference to the unique element of type _Tp of
  1374   /// a const tuple rvalue.
  1375   template <typename _Tp, typename... _Types>
  1376     constexpr const _Tp&&
  1377     get(const tuple<_Types...>&& __t) noexcept
  1378     { return std::forward<const _Tp&&>(std::__get_helper2<_Tp>(__t)); }
  1381   // This class performs the comparison operations on tuples
  1382   template<typename _Tp, typename _Up, size_t __i, size_t __size>
  1383     struct __tuple_compare
  1385       static constexpr bool
  1386       __eq(const _Tp& __t, const _Up& __u)
  1388    return bool(std::get<__i>(__t) == std::get<__i>(__u))
  1389      && __tuple_compare<_Tp, _Up, __i + 1, __size>::__eq(__t, __u);
  1392       static constexpr bool
  1393       __less(const _Tp& __t, const _Up& __u)
  1395    return bool(std::get<__i>(__t) < std::get<__i>(__u))
  1396      || (!bool(std::get<__i>(__u) < std::get<__i>(__t))
  1397          && __tuple_compare<_Tp, _Up, __i + 1, __size>::__less(__t, __u));
  1401   template<typename _Tp, typename _Up, size_t __size>
  1402     struct __tuple_compare<_Tp, _Up, __size, __size>
  1404       static constexpr bool
  1405       __eq(const _Tp&, const _Up&) { return true; }
  1407       static constexpr bool
  1408       __less(const _Tp&, const _Up&) { return false; }
  1411   template<typename... _TElements, typename... _UElements>
  1413     operator==(const tuple<_TElements...>& __t,
  1414           const tuple<_UElements...>& __u)
  1416       static_assert(sizeof...(_TElements) == sizeof...(_UElements),
  1417      "tuple objects can only be compared if they have equal sizes.");
  1418       using __compare = __tuple_compare<tuple<_TElements...>,
  1419                                    tuple<_UElements...>,
  1420                                    0, sizeof...(_TElements)>;
  1421       return __compare::__eq(__t, __u);
  1424   template<typename... _TElements, typename... _UElements>
  1426     operator<(const tuple<_TElements...>& __t,
  1427          const tuple<_UElements...>& __u)
  1429       static_assert(sizeof...(_TElements) == sizeof...(_UElements),
  1430      "tuple objects can only be compared if they have equal sizes.");
  1431       using __compare = __tuple_compare<tuple<_TElements...>,
  1432                                    tuple<_UElements...>,
  1433                                    0, sizeof...(_TElements)>;
  1434       return __compare::__less(__t, __u);
  1437   template<typename... _TElements, typename... _UElements>
  1439     operator!=(const tuple<_TElements...>& __t,
  1440           const tuple<_UElements...>& __u)
  1441     { return !(__t == __u); }
  1443   template<typename... _TElements, typename... _UElements>
  1445     operator>(const tuple<_TElements...>& __t,
  1446          const tuple<_UElements...>& __u)
  1447     { return __u < __t; }
  1449   template<typename... _TElements, typename... _UElements>
  1451     operator<=(const tuple<_TElements...>& __t,
  1452           const tuple<_UElements...>& __u)
  1453     { return !(__u < __t); }
  1455   template<typename... _TElements, typename... _UElements>
  1457     operator>=(const tuple<_TElements...>& __t,
  1458           const tuple<_UElements...>& __u)
  1459     { return !(__t < __u); }
  1462   template<typename... _Elements>
  1463     constexpr tuple<typename __decay_and_strip<_Elements>::__type...>
  1464     make_tuple(_Elements&&... __args)
  1466       typedef tuple<typename __decay_and_strip<_Elements>::__type...>
  1468       return __result_type(std::forward<_Elements>(__args)...);
  1471   // _GLIBCXX_RESOLVE_LIB_DEFECTS
  1472   // 2275. Why is forward_as_tuple not constexpr?
  1473   template<typename... _Elements>
  1474     constexpr tuple<_Elements&&...>
  1475     forward_as_tuple(_Elements&&... __args) noexcept
  1476     { return tuple<_Elements&&...>(std::forward<_Elements>(__args)...); }
  1478   template<size_t, typename, typename, size_t>
  1479     struct __make_tuple_impl;
  1481   template<size_t _Idx, typename _Tuple, typename... _Tp, size_t _Nm>
  1482     struct __make_tuple_impl<_Idx, tuple<_Tp...>, _Tuple, _Nm>
  1483     : __make_tuple_impl<_Idx + 1,
  1484                    tuple<_Tp..., __tuple_element_t<_Idx, _Tuple>>,
  1488   template<std::size_t _Nm, typename _Tuple, typename... _Tp>
  1489     struct __make_tuple_impl<_Nm, tuple<_Tp...>, _Tuple, _Nm>
  1491       typedef tuple<_Tp...> __type;
  1494   template<typename _Tuple>
  1495     struct __do_make_tuple
  1496     : __make_tuple_impl<0, tuple<>, _Tuple, std::tuple_size<_Tuple>::value>
  1499   // Returns the std::tuple equivalent of a tuple-like type.
  1500   template<typename _Tuple>
  1502     : public __do_make_tuple<typename std::remove_cv
  1503             <typename std::remove_reference<_Tuple>::type>::type>
  1506   // Combines several std::tuple's into a single one.
  1507   template<typename...>
  1508     struct __combine_tuples;
  1511     struct __combine_tuples<>
  1513       typedef tuple<> __type;
  1516   template<typename... _Ts>
  1517     struct __combine_tuples<tuple<_Ts...>>
  1519       typedef tuple<_Ts...> __type;
  1522   template<typename... _T1s, typename... _T2s, typename... _Rem>
  1523     struct __combine_tuples<tuple<_T1s...>, tuple<_T2s...>, _Rem...>
  1525       typedef typename __combine_tuples<tuple<_T1s..., _T2s...>,
  1526                                    _Rem...>::__type __type;
  1529   // Computes the result type of tuple_cat given a set of tuple-like types.
  1530   template<typename... _Tpls>
  1531     struct __tuple_cat_result
  1533       typedef typename __combine_tuples
  1534         <typename __make_tuple<_Tpls>::__type...>::__type __type;
  1537   // Helper to determine the index set for the first tuple-like
  1538   // type of a given set.
  1539   template<typename...>
  1540     struct __make_1st_indices;
  1543     struct __make_1st_indices<>
  1545       typedef std::_Index_tuple<> __type;
  1548   template<typename _Tp, typename... _Tpls>
  1549     struct __make_1st_indices<_Tp, _Tpls...>
  1551       typedef typename std::_Build_index_tuple<std::tuple_size<
  1552    typename std::remove_reference<_Tp>::type>::value>::__type __type;
  1555   // Performs the actual concatenation by step-wise expanding tuple-like
  1556   // objects into the elements,  which are finally forwarded into the
  1558   template<typename _Ret, typename _Indices, typename... _Tpls>
  1559     struct __tuple_concater;
  1561   template<typename _Ret, std::size_t... _Is, typename _Tp, typename... _Tpls>
  1562     struct __tuple_concater<_Ret, std::_Index_tuple<_Is...>, _Tp, _Tpls...>
  1564       template<typename... _Us>
  1565         static constexpr _Ret
  1566         _S_do(_Tp&& __tp, _Tpls&&... __tps, _Us&&... __us)
  1568      typedef typename __make_1st_indices<_Tpls...>::__type __idx;
  1569      typedef __tuple_concater<_Ret, __idx, _Tpls...>      __next;
  1570      return __next::_S_do(std::forward<_Tpls>(__tps)...,
  1571                           std::forward<_Us>(__us)...,
  1572                           std::get<_Is>(std::forward<_Tp>(__tp))...);
  1576   template<typename _Ret>
  1577     struct __tuple_concater<_Ret, std::_Index_tuple<>>
  1579       template<typename... _Us>
  1580    static constexpr _Ret
  1581    _S_do(_Us&&... __us)
  1583      return _Ret(std::forward<_Us>(__us)...);
  1588   template<typename... _Tpls, typename = typename
  1589            enable_if<__and_<__is_tuple_like<_Tpls>...>::value>::type>
  1591     tuple_cat(_Tpls&&... __tpls)
  1592     -> typename __tuple_cat_result<_Tpls...>::__type
  1594       typedef typename __tuple_cat_result<_Tpls...>::__type __ret;
  1595       typedef typename __make_1st_indices<_Tpls...>::__type __idx;
  1596       typedef __tuple_concater<__ret, __idx, _Tpls...> __concater;
  1597       return __concater::_S_do(std::forward<_Tpls>(__tpls)...);
  1600   // _GLIBCXX_RESOLVE_LIB_DEFECTS
  1601   // 2301. Why is tie not constexpr?
  1603   template<typename... _Elements>
  1604     constexpr tuple<_Elements&...>
  1605     tie(_Elements&... __args) noexcept
  1606     { return tuple<_Elements&...>(__args...); }
  1609   template<typename... _Elements>
  1611 #if __cplusplus > 201402L || !defined(__STRICT_ANSI__) // c++1z or gnu++11
  1612     // Constrained free swap overload, see p0185r1
  1613     typename enable_if<__and_<__is_swappable<_Elements>...>::value
  1618     swap(tuple<_Elements...>& __x, tuple<_Elements...>& __y)
  1619     noexcept(noexcept(__x.swap(__y)))
  1622 #if __cplusplus > 201402L || !defined(__STRICT_ANSI__) // c++1z or gnu++11
  1623   template<typename... _Elements>
  1624     typename enable_if<!__and_<__is_swappable<_Elements>...>::value>::type
  1625     swap(tuple<_Elements...>&, tuple<_Elements...>&) = delete;
  1628   // A class (and instance) which can be used in 'tie' when an element
  1629   // of a tuple is not required.
  1630   // _GLIBCXX14_CONSTEXPR
  1631   // 2933. PR for LWG 2773 could be clearer
  1632   struct _Swallow_assign
  1635       _GLIBCXX14_CONSTEXPR const _Swallow_assign&
  1636       operator=(const _Tp&) const
  1640   // _GLIBCXX_RESOLVE_LIB_DEFECTS
  1641   // 2773. Making std::ignore constexpr
  1642   _GLIBCXX17_INLINE constexpr _Swallow_assign ignore{};
  1644   /// Partial specialization for tuples
  1645   template<typename... _Types, typename _Alloc>
  1646     struct uses_allocator<tuple<_Types...>, _Alloc> : true_type { };
  1648   // See stl_pair.h...
  1649   template<class _T1, class _T2>
  1650     template<typename... _Args1, typename... _Args2>
  1653       pair(piecewise_construct_t,
  1654       tuple<_Args1...> __first, tuple<_Args2...> __second)
  1655       : pair(__first, __second,
  1656         typename _Build_index_tuple<sizeof...(_Args1)>::__type(),
  1657         typename _Build_index_tuple<sizeof...(_Args2)>::__type())
  1660   template<class _T1, class _T2>
  1661     template<typename... _Args1, std::size_t... _Indexes1,
  1662              typename... _Args2, std::size_t... _Indexes2>
  1665       pair(tuple<_Args1...>& __tuple1, tuple<_Args2...>& __tuple2,
  1666       _Index_tuple<_Indexes1...>, _Index_tuple<_Indexes2...>)
  1667       : first(std::forward<_Args1>(std::get<_Indexes1>(__tuple1))...),
  1668         second(std::forward<_Args2>(std::get<_Indexes2>(__tuple2))...)
  1671 #if __cplusplus > 201402L
  1672 # define __cpp_lib_apply 201603
  1674   template <typename _Fn, typename _Tuple, size_t... _Idx>
  1675     constexpr decltype(auto)
  1676     __apply_impl(_Fn&& __f, _Tuple&& __t, index_sequence<_Idx...>)
  1678       return std::__invoke(std::forward<_Fn>(__f),
  1679                       std::get<_Idx>(std::forward<_Tuple>(__t))...);
  1682   template <typename _Fn, typename _Tuple>
  1683     constexpr decltype(auto)
  1684     apply(_Fn&& __f, _Tuple&& __t)
  1686       using _Indices = make_index_sequence<tuple_size_v<decay_t<_Tuple>>>;
  1687       return std::__apply_impl(std::forward<_Fn>(__f),
  1688                           std::forward<_Tuple>(__t),
  1692 #define __cpp_lib_make_from_tuple  201606
  1694   template <typename _Tp, typename _Tuple, size_t... _Idx>
  1696     __make_from_tuple_impl(_Tuple&& __t, index_sequence<_Idx...>)
  1697     { return _Tp(std::get<_Idx>(std::forward<_Tuple>(__t))...); }
  1699   template <typename _Tp, typename _Tuple>
  1701     make_from_tuple(_Tuple&& __t)
  1703       return __make_from_tuple_impl<_Tp>(
  1704         std::forward<_Tuple>(__t),
  1705    make_index_sequence<tuple_size_v<decay_t<_Tuple>>>{});
  1711 _GLIBCXX_END_NAMESPACE_VERSION
  1716 #endif // _GLIBCXX_TUPLE