Home | History | Annotate | Download | only in include
      1 // <tuple> -*- C++ -*-
      2 
      3 // Copyright (C) 2007, 2008, 2009 Free Software Foundation, Inc.
      4 //
      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)
      9 // any later version.
     10 
     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.
     15 
     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.
     19 
     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/>.
     24 
     25 /** @file include/tuple
     26  *  This is a Standard C++ Library header.
     27  */
     28 
     29 #ifndef _GLIBCXX_TUPLE
     30 #define _GLIBCXX_TUPLE 1
     31 
     32 #pragma GCC system_header
     33 
     34 #ifndef __GXX_EXPERIMENTAL_CXX0X__
     35 # include <c++0x_warning.h>
     36 #else
     37 
     38 #include <utility>
     39 
     40 namespace std
     41 {
     42   // Adds a const reference to a non-reference type.
     43   template<typename _Tp>
     44     struct __add_c_ref
     45     { typedef const _Tp& type; };
     46 
     47   template<typename _Tp>
     48     struct __add_c_ref<_Tp&>
     49     { typedef _Tp& type; };
     50 
     51   // Adds a reference to a non-reference type.
     52   template<typename _Tp>
     53     struct __add_ref
     54     { typedef _Tp& type; };
     55 
     56   template<typename _Tp>
     57     struct __add_ref<_Tp&>
     58     { typedef _Tp& type; };
     59 
     60   template<std::size_t _Idx, typename _Head, bool _IsEmpty>
     61     struct _Head_base;
     62 
     63   template<std::size_t _Idx, typename _Head>
     64     struct _Head_base<_Idx, _Head, true>
     65     : public _Head
     66     {
     67       _Head_base()
     68       : _Head() { }
     69 
     70       _Head_base(const _Head& __h)
     71       : _Head(__h) { }
     72 
     73       template<typename _UHead>
     74         _Head_base(_UHead&& __h)
     75 	: _Head(std::forward<_UHead>(__h)) { }
     76 
     77       _Head&       _M_head()       { return *this; }
     78       const _Head& _M_head() const { return *this; }
     79     
     80       void _M_swap_impl(_Head&&) { /* no-op */ }
     81     };
     82 
     83   template<std::size_t _Idx, typename _Head>
     84     struct _Head_base<_Idx, _Head, false>
     85     {
     86       _Head_base()
     87       : _M_head_impl() { }
     88 
     89       _Head_base(const _Head& __h)
     90       : _M_head_impl(__h) { }
     91 
     92       template<typename _UHead>
     93         _Head_base(_UHead&& __h)
     94 	: _M_head_impl(std::forward<_UHead>(__h)) { }
     95 
     96       _Head&       _M_head()       { return _M_head_impl; }
     97       const _Head& _M_head() const { return _M_head_impl; }        
     98 
     99       void
    100       _M_swap_impl(_Head&& __h)
    101       { 
    102 	using std::swap;
    103 	swap(__h, _M_head_impl);
    104       }
    105 
    106       _Head _M_head_impl; 
    107     };
    108 
    109   /**
    110    * Contains the actual implementation of the @c tuple template, stored
    111    * as a recursive inheritance hierarchy from the first element (most
    112    * derived class) to the last (least derived class). The @c Idx
    113    * parameter gives the 0-based index of the element stored at this
    114    * point in the hierarchy; we use it to implement a constant-time
    115    * get() operation.
    116    */
    117   template<std::size_t _Idx, typename... _Elements>
    118     struct _Tuple_impl; 
    119 
    120   /**
    121    * Zero-element tuple implementation. This is the basis case for the 
    122    * inheritance recursion.
    123    */
    124   template<std::size_t _Idx>
    125     struct _Tuple_impl<_Idx>
    126     { 
    127     protected:
    128       void _M_swap_impl(_Tuple_impl&&) { /* no-op */ }
    129     };
    130 
    131   /**
    132    * Recursive tuple implementation. Here we store the @c Head element
    133    * and derive from a @c Tuple_impl containing the remaining elements
    134    * (which contains the @c Tail).
    135    */
    136   template<std::size_t _Idx, typename _Head, typename... _Tail>
    137     struct _Tuple_impl<_Idx, _Head, _Tail...>
    138     : public _Tuple_impl<_Idx + 1, _Tail...>,
    139       private _Head_base<_Idx, _Head, std::is_empty<_Head>::value>
    140     {
    141       typedef _Tuple_impl<_Idx + 1, _Tail...> _Inherited;
    142       typedef _Head_base<_Idx, _Head, std::is_empty<_Head>::value> _Base;
    143 
    144       _Head&            _M_head()       { return _Base::_M_head(); }
    145       const _Head&      _M_head() const { return _Base::_M_head(); }
    146 
    147       _Inherited&       _M_tail()       { return *this; }
    148       const _Inherited& _M_tail() const { return *this; }
    149 
    150       _Tuple_impl()
    151       : _Inherited(), _Base() { }
    152 
    153       explicit 
    154       _Tuple_impl(const _Head& __head, const _Tail&... __tail)
    155       : _Inherited(__tail...), _Base(__head) { }
    156 
    157       template<typename _UHead, typename... _UTail> 
    158         explicit
    159         _Tuple_impl(_UHead&& __head, _UTail&&... __tail)
    160 	: _Inherited(std::forward<_UTail>(__tail)...),
    161 	  _Base(std::forward<_UHead>(__head)) { }
    162 
    163       _Tuple_impl(const _Tuple_impl& __in)
    164       : _Inherited(__in._M_tail()), _Base(__in._M_head()) { }
    165 
    166       _Tuple_impl(_Tuple_impl&& __in)
    167       : _Inherited(std::move<_Inherited&&>(__in._M_tail())),
    168 	_Base(std::forward<_Head>(__in._M_head())) { }
    169 
    170       template<typename... _UElements>
    171         _Tuple_impl(const _Tuple_impl<_Idx, _UElements...>& __in)
    172 	: _Inherited(__in._M_tail()), _Base(__in._M_head()) { }
    173 
    174       template<typename... _UElements>
    175         _Tuple_impl(_Tuple_impl<_Idx, _UElements...>&& __in)
    176 	: _Inherited(std::move<typename _Tuple_impl<_Idx, _UElements...>::
    177 		     _Inherited&&>(__in._M_tail())),
    178 	  _Base(std::forward<typename _Tuple_impl<_Idx, _UElements...>::
    179 		_Base>(__in._M_head())) { }
    180 
    181       _Tuple_impl&
    182       operator=(const _Tuple_impl& __in)
    183       {
    184 	_M_head() = __in._M_head();
    185 	_M_tail() = __in._M_tail();
    186 	return *this;
    187       }
    188 
    189       _Tuple_impl&
    190       operator=(_Tuple_impl&& __in)
    191       {
    192 	_M_head() = std::move(__in._M_head());
    193 	_M_tail() = std::move(__in._M_tail());
    194 	return *this;
    195       }
    196 
    197       template<typename... _UElements>
    198         _Tuple_impl&
    199         operator=(const _Tuple_impl<_Idx, _UElements...>& __in)
    200         {
    201 	  _M_head() = __in._M_head();
    202 	  _M_tail() = __in._M_tail();
    203 	  return *this;
    204 	}
    205 
    206       template<typename... _UElements>
    207         _Tuple_impl&
    208         operator=(_Tuple_impl<_Idx, _UElements...>&& __in)
    209         {
    210 	  _M_head() = std::move(__in._M_head());
    211 	  _M_tail() = std::move(__in._M_tail());
    212 	  return *this;
    213 	}
    214 
    215     protected:
    216       void
    217       _M_swap_impl(_Tuple_impl&& __in)
    218       {
    219 	_Base::_M_swap_impl(__in._M_head());
    220 	_Inherited::_M_swap_impl(__in._M_tail());
    221       }
    222     };
    223 
    224   /// tuple
    225   template<typename... _Elements> 
    226     class tuple : public _Tuple_impl<0, _Elements...>
    227     {
    228       typedef _Tuple_impl<0, _Elements...> _Inherited;
    229 
    230     public:
    231       tuple()
    232       : _Inherited() { }
    233 
    234       explicit
    235       tuple(const _Elements&... __elements)
    236       : _Inherited(__elements...) { }
    237 
    238       template<typename... _UElements>
    239         explicit
    240         tuple(_UElements&&... __elements)
    241 	: _Inherited(std::forward<_UElements>(__elements)...) {	}
    242 
    243       tuple(const tuple& __in)
    244       : _Inherited(static_cast<const _Inherited&>(__in)) { }
    245 
    246       tuple(tuple&& __in)
    247       : _Inherited(std::move<_Inherited>(__in)) { }
    248 
    249       template<typename... _UElements>
    250         tuple(const tuple<_UElements...>& __in)
    251 	: _Inherited(static_cast<const _Tuple_impl<0, _UElements...>&>(__in))
    252 	{ }
    253 
    254       template<typename... _UElements>
    255         tuple(tuple<_UElements...>&& __in)
    256 	: _Inherited(std::move<_Tuple_impl<0, _UElements...> >(__in)) { }
    257 
    258       // XXX http://gcc.gnu.org/ml/libstdc++/2008-02/msg00047.html
    259       template<typename... _UElements>
    260         tuple(tuple<_UElements...>& __in)
    261 	: _Inherited(static_cast<const _Tuple_impl<0, _UElements...>&>(__in))
    262 	{ }
    263 
    264       tuple&
    265       operator=(const tuple& __in)
    266       {
    267 	static_cast<_Inherited&>(*this) = __in;
    268 	return *this;
    269       }
    270 
    271       tuple&
    272       operator=(tuple&& __in)
    273       {
    274 	static_cast<_Inherited&>(*this) = std::move(__in);
    275 	return *this;
    276       }
    277 
    278       template<typename... _UElements>
    279         tuple&
    280         operator=(const tuple<_UElements...>& __in)
    281         {
    282 	  static_cast<_Inherited&>(*this) = __in;
    283 	  return *this;
    284 	}
    285 
    286       template<typename... _UElements>
    287         tuple&
    288         operator=(tuple<_UElements...>&& __in)
    289         {
    290 	  static_cast<_Inherited&>(*this) = std::move(__in);
    291 	  return *this;
    292 	}
    293 
    294       void
    295       swap(tuple&& __in)
    296       { _Inherited::_M_swap_impl(__in); }
    297     };
    298 
    299 
    300   template<>  
    301     class tuple<>
    302     {
    303     public:
    304       void swap(tuple&&) { /* no-op */ }
    305     };
    306 
    307   /// tuple (2-element), with construction and assignment from a pair.
    308   template<typename _T1, typename _T2>
    309     class tuple<_T1, _T2> : public _Tuple_impl<0, _T1, _T2>
    310     {
    311       typedef _Tuple_impl<0, _T1, _T2> _Inherited;
    312 
    313     public:
    314       tuple()
    315       : _Inherited() { }
    316 
    317       explicit
    318       tuple(const _T1& __a1, const _T2& __a2)
    319       : _Inherited(__a1, __a2) { }
    320 
    321       template<typename _U1, typename _U2>
    322         explicit
    323         tuple(_U1&& __a1, _U2&& __a2)
    324 	: _Inherited(std::forward<_U1>(__a1), std::forward<_U2>(__a2)) { }
    325 
    326       tuple(const tuple& __in)
    327       : _Inherited(static_cast<const _Inherited&>(__in)) { }
    328 
    329       tuple(tuple&& __in)
    330       : _Inherited(std::move<_Inherited>(__in)) { }
    331 
    332       template<typename _U1, typename _U2>
    333         tuple(const tuple<_U1, _U2>& __in)
    334 	: _Inherited(static_cast<const _Tuple_impl<0, _U1, _U2>&>(__in)) { }
    335 
    336       template<typename _U1, typename _U2>
    337         tuple(tuple<_U1, _U2>&& __in)
    338 	: _Inherited(std::move<_Tuple_impl<0, _U1, _U2> >(__in)) { }
    339 
    340       template<typename _U1, typename _U2>
    341         tuple(const pair<_U1, _U2>& __in)
    342 	: _Inherited(__in.first, __in.second) { }
    343 
    344       template<typename _U1, typename _U2>
    345         tuple(pair<_U1, _U2>&& __in)
    346 	: _Inherited(std::move(__in.first), std::move(__in.second)) { }
    347 
    348       tuple&
    349       operator=(const tuple& __in)
    350       {
    351 	static_cast<_Inherited&>(*this) = __in;
    352 	return *this;
    353       }
    354 
    355       tuple&
    356       operator=(tuple&& __in)
    357       {
    358 	static_cast<_Inherited&>(*this) = std::move(__in);
    359 	return *this;
    360       }
    361 
    362       template<typename _U1, typename _U2>
    363         tuple&
    364         operator=(const tuple<_U1, _U2>& __in)
    365         {
    366 	  static_cast<_Inherited&>(*this) = __in;
    367 	  return *this;
    368 	}
    369 
    370       template<typename _U1, typename _U2>
    371         tuple&
    372         operator=(tuple<_U1, _U2>&& __in)
    373         {
    374 	  static_cast<_Inherited&>(*this) = std::move(__in);
    375 	  return *this;
    376 	}
    377 
    378       template<typename _U1, typename _U2>
    379         tuple&
    380         operator=(const pair<_U1, _U2>& __in)
    381         {
    382 	  this->_M_head() = __in.first;
    383 	  this->_M_tail()._M_head() = __in.second;
    384 	  return *this;
    385 	}
    386 
    387       template<typename _U1, typename _U2>
    388         tuple&
    389         operator=(pair<_U1, _U2>&& __in)
    390         {
    391 	  this->_M_head() = std::move(__in.first);
    392 	  this->_M_tail()._M_head() = std::move(__in.second);
    393 	  return *this;
    394 	}
    395 
    396       void
    397       swap(tuple&& __in)
    398       { 
    399 	using std::swap;
    400 	swap(this->_M_head(), __in._M_head());
    401 	swap(this->_M_tail()._M_head(), __in._M_tail()._M_head());	
    402       }
    403     };
    404 
    405 
    406   /// Gives the type of the ith element of a given tuple type.
    407   template<std::size_t __i, typename _Tp>
    408     struct tuple_element;
    409 
    410   /**
    411    * Recursive case for tuple_element: strip off the first element in
    412    * the tuple and retrieve the (i-1)th element of the remaining tuple.
    413    */
    414   template<std::size_t __i, typename _Head, typename... _Tail>
    415     struct tuple_element<__i, tuple<_Head, _Tail...> >
    416     : tuple_element<__i - 1, tuple<_Tail...> > { };
    417 
    418   /**
    419    * Basis case for tuple_element: The first element is the one we're seeking.
    420    */
    421   template<typename _Head, typename... _Tail>
    422     struct tuple_element<0, tuple<_Head, _Tail...> >
    423     {
    424       typedef _Head type;
    425     };
    426 
    427   /// Finds the size of a given tuple type.
    428   template<typename _Tp>
    429     struct tuple_size;
    430 
    431   /// class tuple_size
    432   template<typename... _Elements>
    433     struct tuple_size<tuple<_Elements...> >
    434     {
    435       static const std::size_t value = sizeof...(_Elements);
    436     };
    437 
    438   template<typename... _Elements>
    439     const std::size_t tuple_size<tuple<_Elements...> >::value;
    440 
    441   template<std::size_t __i, typename _Head, typename... _Tail>
    442     inline typename __add_ref<_Head>::type
    443     __get_helper(_Tuple_impl<__i, _Head, _Tail...>& __t)
    444     { return __t._M_head(); }
    445 
    446   template<std::size_t __i, typename _Head, typename... _Tail>
    447     inline typename __add_c_ref<_Head>::type
    448     __get_helper(const _Tuple_impl<__i, _Head, _Tail...>& __t)
    449     { return __t._M_head(); }
    450 
    451   // Return a reference (const reference) to the ith element of a tuple.
    452   // Any const or non-const ref elements are returned with their original type.
    453   template<std::size_t __i, typename... _Elements>
    454     inline typename __add_ref<
    455                       typename tuple_element<__i, tuple<_Elements...> >::type
    456                     >::type
    457     get(tuple<_Elements...>& __t)
    458     { return __get_helper<__i>(__t); }
    459 
    460   template<std::size_t __i, typename... _Elements>
    461     inline typename __add_c_ref<
    462                       typename tuple_element<__i, tuple<_Elements...> >::type
    463                     >::type
    464     get(const tuple<_Elements...>& __t)
    465     { return __get_helper<__i>(__t); }
    466 
    467   // This class helps construct the various comparison operations on tuples
    468   template<std::size_t __check_equal_size, std::size_t __i, std::size_t __j,
    469 	   typename _Tp, typename _Up>
    470     struct __tuple_compare;
    471 
    472   template<std::size_t __i, std::size_t __j, typename _Tp, typename _Up>
    473     struct __tuple_compare<0, __i, __j, _Tp, _Up>
    474     {
    475       static bool __eq(const _Tp& __t, const _Up& __u)
    476       {
    477 	return (get<__i>(__t) == get<__i>(__u) &&
    478 		__tuple_compare<0, __i + 1, __j, _Tp, _Up>::__eq(__t, __u));
    479       }
    480      
    481       static bool __less(const _Tp& __t, const _Up& __u)
    482       {
    483 	return ((get<__i>(__t) < get<__i>(__u))
    484 		|| !(get<__i>(__u) < get<__i>(__t)) &&
    485 		__tuple_compare<0, __i + 1, __j, _Tp, _Up>::__less(__t, __u));
    486       }
    487     };
    488 
    489   template<std::size_t __i, typename _Tp, typename _Up>
    490     struct __tuple_compare<0, __i, __i, _Tp, _Up>
    491     {
    492       static bool __eq(const _Tp&, const _Up&)
    493       { return true; }
    494      
    495       static bool __less(const _Tp&, const _Up&)
    496       { return false; }
    497     };
    498 
    499   template<typename... _TElements, typename... _UElements>
    500     bool
    501     operator==(const tuple<_TElements...>& __t,
    502 	       const tuple<_UElements...>& __u)
    503     {
    504       typedef tuple<_TElements...> _Tp;
    505       typedef tuple<_UElements...> _Up;
    506       return (__tuple_compare<tuple_size<_Tp>::value - tuple_size<_Up>::value,
    507 	      0, tuple_size<_Tp>::value, _Tp, _Up>::__eq(__t, __u));
    508     }
    509 
    510   template<typename... _TElements, typename... _UElements>
    511     bool
    512     operator<(const tuple<_TElements...>& __t,
    513 	      const tuple<_UElements...>& __u)
    514     {
    515       typedef tuple<_TElements...> _Tp;
    516       typedef tuple<_UElements...> _Up;
    517       return (__tuple_compare<tuple_size<_Tp>::value - tuple_size<_Up>::value,
    518 	      0, tuple_size<_Tp>::value, _Tp, _Up>::__less(__t, __u));
    519     }
    520 
    521   template<typename... _TElements, typename... _UElements>
    522     inline bool
    523     operator!=(const tuple<_TElements...>& __t,
    524 	       const tuple<_UElements...>& __u)
    525     { return !(__t == __u); }
    526 
    527   template<typename... _TElements, typename... _UElements>
    528     inline bool
    529     operator>(const tuple<_TElements...>& __t,
    530 	      const tuple<_UElements...>& __u)
    531     { return __u < __t; }
    532 
    533   template<typename... _TElements, typename... _UElements>
    534     inline bool
    535     operator<=(const tuple<_TElements...>& __t,
    536 	       const tuple<_UElements...>& __u)
    537     { return !(__u < __t); }
    538 
    539   template<typename... _TElements, typename... _UElements>
    540     inline bool
    541     operator>=(const tuple<_TElements...>& __t,
    542 	       const tuple<_UElements...>& __u)
    543     { return !(__t < __u); }
    544 
    545   // NB: DR 705.
    546   template<typename... _Elements>
    547     inline tuple<typename __decay_and_strip<_Elements>::__type...>
    548     make_tuple(_Elements&&... __args)
    549     {
    550       typedef tuple<typename __decay_and_strip<_Elements>::__type...>
    551 	__result_type;
    552       return __result_type(std::forward<_Elements>(__args)...);
    553     }
    554 
    555   template<std::size_t...> struct __index_holder { };    
    556 
    557   template<std::size_t __i, typename _IdxHolder, typename... _Elements>
    558     struct __index_holder_impl;
    559 
    560   template<std::size_t __i, std::size_t... _Indexes, typename _IdxHolder,
    561 	   typename... _Elements>
    562     struct __index_holder_impl<__i, __index_holder<_Indexes...>,
    563 			       _IdxHolder, _Elements...> 
    564     {
    565       typedef typename __index_holder_impl<__i + 1,
    566 					   __index_holder<_Indexes..., __i>,
    567 					   _Elements...>::type type;
    568     };
    569  
    570   template<std::size_t __i, std::size_t... _Indexes>
    571     struct __index_holder_impl<__i, __index_holder<_Indexes...> >
    572     { typedef __index_holder<_Indexes...> type; };
    573 
    574   template<typename... _Elements>
    575     struct __make_index_holder 
    576     : __index_holder_impl<0, __index_holder<>, _Elements...> { };
    577     
    578   template<typename... _TElements, std::size_t... _TIdx,
    579 	   typename... _UElements, std::size_t... _UIdx> 
    580     inline tuple<_TElements..., _UElements...> 
    581     __tuple_cat_helper(const tuple<_TElements...>& __t,
    582 		       const __index_holder<_TIdx...>&,
    583                        const tuple<_UElements...>& __u,
    584 		       const __index_holder<_UIdx...>&)
    585     { return tuple<_TElements..., _UElements...>(get<_TIdx>(__t)...,
    586 						 get<_UIdx>(__u)...); }
    587 
    588   template<typename... _TElements, std::size_t... _TIdx,
    589 	   typename... _UElements, std::size_t... _UIdx> 
    590     inline tuple<_TElements..., _UElements...> 
    591     __tuple_cat_helper(tuple<_TElements...>&& __t,
    592 		       const __index_holder<_TIdx...>&, 
    593 		       const tuple<_UElements...>& __u,
    594 		       const __index_holder<_UIdx...>&)
    595     { return tuple<_TElements..., _UElements...>
    596 	(std::move(get<_TIdx>(__t))..., get<_UIdx>(__u)...); }
    597 
    598   template<typename... _TElements, std::size_t... _TIdx,
    599 	   typename... _UElements, std::size_t... _UIdx>
    600     inline tuple<_TElements..., _UElements...> 
    601     __tuple_cat_helper(const tuple<_TElements...>& __t,
    602 		       const __index_holder<_TIdx...>&, 
    603 		       tuple<_UElements...>&& __u,
    604 		       const __index_holder<_UIdx...>&)
    605     { return tuple<_TElements..., _UElements...>
    606 	(get<_TIdx>(__t)..., std::move(get<_UIdx>(__u))...); }
    607 
    608   template<typename... _TElements, std::size_t... _TIdx,
    609 	   typename... _UElements, std::size_t... _UIdx> 
    610     inline tuple<_TElements..., _UElements...> 
    611     __tuple_cat_helper(tuple<_TElements...>&& __t,
    612 		       const __index_holder<_TIdx...>&, 
    613 		       tuple<_UElements...>&& __u,
    614 		       const __index_holder<_UIdx...>&)
    615     { return tuple<_TElements..., _UElements...>
    616 	(std::move(get<_TIdx>(__t))..., std::move(get<_UIdx>(__u))...); }
    617 
    618   template<typename... _TElements, typename... _UElements>
    619     inline tuple<_TElements..., _UElements...> 
    620     tuple_cat(const tuple<_TElements...>& __t, const tuple<_UElements...>& __u)
    621     {
    622       return __tuple_cat_helper(__t, typename
    623 				__make_index_holder<_TElements...>::type(),
    624 				__u, typename
    625 				__make_index_holder<_UElements...>::type());
    626     }
    627 
    628   template<typename... _TElements, typename... _UElements>
    629     inline tuple<_TElements..., _UElements...> 
    630     tuple_cat(tuple<_TElements...>&& __t, const tuple<_UElements...>& __u)
    631     {
    632       return __tuple_cat_helper(std::move(__t), typename
    633 				 __make_index_holder<_TElements...>::type(),
    634 				 __u, typename
    635 				 __make_index_holder<_UElements...>::type());
    636     }
    637 
    638   template<typename... _TElements, typename... _UElements>
    639     inline tuple<_TElements..., _UElements...> 
    640     tuple_cat(const tuple<_TElements...>& __t, tuple<_UElements...>&& __u)
    641     {
    642       return __tuple_cat_helper(__t, typename
    643 				__make_index_holder<_TElements...>::type(),
    644 				std::move(__u), typename
    645 				__make_index_holder<_UElements...>::type());
    646     }
    647 
    648   template<typename... _TElements, typename... _UElements>
    649     inline tuple<_TElements..., _UElements...>
    650     tuple_cat(tuple<_TElements...>&& __t, tuple<_UElements...>&& __u)
    651     {
    652       return __tuple_cat_helper(std::move(__t), typename
    653 				__make_index_holder<_TElements...>::type(),
    654 				std::move(__u), typename
    655 				__make_index_holder<_UElements...>::type());
    656     }
    657 
    658   template<typename... _Elements>
    659     inline tuple<_Elements&...>
    660     tie(_Elements&... __args)
    661     { return tuple<_Elements&...>(__args...); }
    662 
    663   template<typename... _Elements>
    664     inline void 
    665     swap(tuple<_Elements...>& __x, tuple<_Elements...>& __y)
    666     { __x.swap(__y); }
    667 
    668   template<typename... _Elements>
    669     inline void
    670     swap(tuple<_Elements...>&& __x, tuple<_Elements...>& __y)
    671     { __x.swap(__y); }
    672 
    673   template<typename... _Elements>
    674     inline void
    675     swap(tuple<_Elements...>& __x, tuple<_Elements...>&& __y)
    676     { __x.swap(__y); }
    677 
    678   // A class (and instance) which can be used in 'tie' when an element
    679   // of a tuple is not required
    680   struct _Swallow_assign
    681   {
    682     template<class _Tp>
    683       _Swallow_assign&
    684       operator=(const _Tp&)
    685       { return *this; }
    686   };
    687 
    688   // TODO: Put this in some kind of shared file.
    689   namespace
    690   {
    691     _Swallow_assign ignore;
    692   }; // anonymous namespace
    693 }
    694 
    695 #endif // __GXX_EXPERIMENTAL_CXX0X__
    696 
    697 #endif // _GLIBCXX_TUPLE
    698