Home | History | Annotate | Download | only in tr1
      1 // class template tuple -*- C++ -*-
      2 
      3 // Copyright (C) 2004, 2005, 2006, 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 tr1/tuple
     26 *  This is a TR1 C++ Library header.
     27 */
     28 
     29 // Chris Jefferson <chris (a] bubblescope.net>
     30 // Variadic Templates support by Douglas Gregor <doug.gregor (a] gmail.com>
     31 
     32 #ifndef _GLIBCXX_TR1_TUPLE
     33 #define _GLIBCXX_TR1_TUPLE 1
     34 
     35 #pragma GCC system_header
     36 
     37 #include <utility>
     38 
     39 namespace std _GLIBCXX_VISIBILITY(default)
     40 {
     41 namespace tr1
     42 {
     43 _GLIBCXX_BEGIN_NAMESPACE_VERSION
     44 
     45   // Adds a const reference to a non-reference type.
     46   template<typename _Tp>
     47     struct __add_c_ref
     48     { typedef const _Tp& type; };
     49 
     50   template<typename _Tp>
     51     struct __add_c_ref<_Tp&>
     52     { typedef _Tp& type; };
     53 
     54   // Adds a reference to a non-reference type.
     55   template<typename _Tp>
     56     struct __add_ref
     57     { typedef _Tp& type; };
     58 
     59   template<typename _Tp>
     60     struct __add_ref<_Tp&>
     61     { typedef _Tp& type; };
     62 
     63   /**
     64    * Contains the actual implementation of the @c tuple template, stored
     65    * as a recursive inheritance hierarchy from the first element (most
     66    * derived class) to the last (least derived class). The @c Idx
     67    * parameter gives the 0-based index of the element stored at this
     68    * point in the hierarchy; we use it to implement a constant-time
     69    * get() operation.
     70    */
     71   template<int _Idx, typename... _Elements>
     72     struct _Tuple_impl; 
     73 
     74   /**
     75    * Zero-element tuple implementation. This is the basis case for the 
     76    * inheritance recursion.
     77    */
     78   template<int _Idx>
     79     struct _Tuple_impl<_Idx> { };
     80 
     81   /**
     82    * Recursive tuple implementation. Here we store the @c Head element
     83    * and derive from a @c Tuple_impl containing the remaining elements
     84    * (which contains the @c Tail).
     85    */
     86   template<int _Idx, typename _Head, typename... _Tail>
     87     struct _Tuple_impl<_Idx, _Head, _Tail...>
     88     : public _Tuple_impl<_Idx + 1, _Tail...>
     89     {
     90       typedef _Tuple_impl<_Idx + 1, _Tail...> _Inherited;
     91       
     92       _Head _M_head;
     93       
     94       _Inherited&       _M_tail()       { return *this; }
     95       const _Inherited& _M_tail() const { return *this; }
     96       
     97       _Tuple_impl() : _Inherited(), _M_head() { }
     98       
     99       explicit 
    100       _Tuple_impl(typename __add_c_ref<_Head>::type __head,
    101 		  typename __add_c_ref<_Tail>::type... __tail)
    102       : _Inherited(__tail...), _M_head(__head) { }
    103 
    104       template<typename... _UElements>
    105       _Tuple_impl(const _Tuple_impl<_Idx, _UElements...>& __in)
    106       : _Inherited(__in._M_tail()), _M_head(__in._M_head) { }
    107 
    108       _Tuple_impl(const _Tuple_impl& __in)
    109       : _Inherited(__in._M_tail()), _M_head(__in._M_head) { }
    110      
    111       template<typename... _UElements>
    112         _Tuple_impl&
    113         operator=(const _Tuple_impl<_Idx, _UElements...>& __in)
    114         {
    115 	  _M_head = __in._M_head;
    116 	  _M_tail() = __in._M_tail();
    117 	  return *this;
    118 	}
    119 
    120       _Tuple_impl&
    121       operator=(const _Tuple_impl& __in)
    122       {
    123 	_M_head = __in._M_head;
    124 	_M_tail() = __in._M_tail();
    125 	return *this;
    126       }
    127     };
    128 
    129   template<typename... _Elements> 
    130     class tuple : public _Tuple_impl<0, _Elements...>
    131     {
    132       typedef _Tuple_impl<0, _Elements...> _Inherited;
    133 
    134     public:
    135       tuple() : _Inherited() { }
    136 
    137       explicit
    138       tuple(typename __add_c_ref<_Elements>::type... __elements)
    139       : _Inherited(__elements...) { }
    140 
    141       template<typename... _UElements>
    142         tuple(const tuple<_UElements...>& __in)
    143 	: _Inherited(__in) { }
    144 
    145       tuple(const tuple& __in)
    146       : _Inherited(__in) { }
    147 
    148       template<typename... _UElements>
    149         tuple&
    150         operator=(const tuple<_UElements...>& __in)
    151         {
    152 	  static_cast<_Inherited&>(*this) = __in;
    153 	  return *this;
    154 	}
    155 
    156       tuple&
    157       operator=(const tuple& __in)
    158       {
    159 	static_cast<_Inherited&>(*this) = __in;
    160 	return *this;
    161       }
    162     };
    163 
    164   template<> class tuple<> { };
    165 
    166   // 2-element tuple, with construction and assignment from a pair.
    167   template<typename _T1, typename _T2>
    168     class tuple<_T1, _T2> : public _Tuple_impl<0, _T1, _T2>
    169     {
    170       typedef _Tuple_impl<0, _T1, _T2> _Inherited;
    171 
    172     public:
    173       tuple() : _Inherited() { }
    174 
    175       explicit
    176       tuple(typename __add_c_ref<_T1>::type __a1,
    177 	    typename __add_c_ref<_T2>::type __a2)
    178       : _Inherited(__a1, __a2) { }
    179 
    180       template<typename _U1, typename _U2>
    181         tuple(const tuple<_U1, _U2>& __in)
    182 	: _Inherited(__in) { }
    183 
    184       tuple(const tuple& __in)
    185       : _Inherited(__in) { }
    186 
    187       template<typename _U1, typename _U2>
    188         tuple(const pair<_U1, _U2>& __in)
    189 	: _Inherited(_Tuple_impl<0, 
    190 		     typename __add_c_ref<_U1>::type,
    191 		     typename __add_c_ref<_U2>::type>(__in.first, 
    192 						      __in.second))
    193         { }
    194   
    195       template<typename _U1, typename _U2>
    196         tuple&
    197         operator=(const tuple<_U1, _U2>& __in)
    198         {
    199 	  static_cast<_Inherited&>(*this) = __in;
    200 	  return *this;
    201 	}
    202 
    203       tuple&
    204       operator=(const tuple& __in)
    205       {
    206 	static_cast<_Inherited&>(*this) = __in;
    207 	return *this;
    208       }
    209 
    210       template<typename _U1, typename _U2>
    211         tuple&
    212         operator=(const pair<_U1, _U2>& __in)
    213         {
    214 	  this->_M_head = __in.first;
    215 	  this->_M_tail()._M_head = __in.second;
    216 	  return *this;
    217 	}
    218     };
    219 
    220   
    221   /// Gives the type of the ith element of a given tuple type.
    222   template<int __i, typename _Tp>
    223     struct tuple_element;
    224 
    225   /**
    226    * Recursive case for tuple_element: strip off the first element in
    227    * the tuple and retrieve the (i-1)th element of the remaining tuple.
    228    */
    229   template<int __i, typename _Head, typename... _Tail>
    230     struct tuple_element<__i, tuple<_Head, _Tail...> >
    231     : tuple_element<__i - 1, tuple<_Tail...> > { };
    232 
    233   /**
    234    * Basis case for tuple_element: The first element is the one we're seeking.
    235    */
    236   template<typename _Head, typename... _Tail>
    237     struct tuple_element<0, tuple<_Head, _Tail...> >
    238     {
    239       typedef _Head type;
    240     };
    241 
    242   /// Finds the size of a given tuple type.
    243   template<typename _Tp>
    244     struct tuple_size;
    245 
    246   /// class tuple_size
    247   template<typename... _Elements>
    248     struct tuple_size<tuple<_Elements...> >
    249     {
    250       static const int value = sizeof...(_Elements);
    251     };
    252 
    253   template<typename... _Elements>
    254     const int tuple_size<tuple<_Elements...> >::value;
    255 
    256   template<int __i, typename _Head, typename... _Tail>
    257     inline typename __add_ref<_Head>::type
    258     __get_helper(_Tuple_impl<__i, _Head, _Tail...>& __t)
    259     {
    260       return __t._M_head;
    261     }
    262 
    263   template<int __i, typename _Head, typename... _Tail>
    264     inline typename __add_c_ref<_Head>::type
    265     __get_helper(const _Tuple_impl<__i, _Head, _Tail...>& __t)
    266     {
    267       return __t._M_head;
    268     }
    269 
    270   // Return a reference (const reference) to the ith element of a tuple.
    271   // Any const or non-const ref elements are returned with their original type.
    272   template<int __i, typename... _Elements>
    273     inline typename __add_ref<
    274                       typename tuple_element<__i, tuple<_Elements...> >::type
    275                     >::type
    276     get(tuple<_Elements...>& __t)
    277     { 
    278       return __get_helper<__i>(__t); 
    279     }
    280 
    281   template<int __i, typename... _Elements>
    282     inline typename __add_c_ref<
    283                       typename tuple_element<__i, tuple<_Elements...> >::type
    284                     >::type
    285     get(const tuple<_Elements...>& __t)
    286     {
    287       return __get_helper<__i>(__t);
    288     }
    289 
    290   // This class helps construct the various comparison operations on tuples
    291   template<int __check_equal_size, int __i, int __j,
    292 	   typename _Tp, typename _Up>
    293     struct __tuple_compare;
    294 
    295   template<int __i, int __j, typename _Tp, typename _Up>
    296     struct __tuple_compare<0, __i, __j, _Tp, _Up>
    297     {
    298       static bool __eq(const _Tp& __t, const _Up& __u)
    299       {
    300 	return (get<__i>(__t) == get<__i>(__u) &&
    301 		__tuple_compare<0, __i+1, __j, _Tp, _Up>::__eq(__t, __u));
    302       }
    303      
    304       static bool __less(const _Tp& __t, const _Up& __u)
    305       {
    306 	return ((get<__i>(__t) < get<__i>(__u))
    307 		|| !(get<__i>(__u) < get<__i>(__t)) &&
    308 		__tuple_compare<0, __i+1, __j, _Tp, _Up>::__less(__t, __u));
    309       }
    310     };
    311 
    312   template<int __i, typename _Tp, typename _Up>
    313     struct __tuple_compare<0, __i, __i, _Tp, _Up>
    314     {
    315       static bool __eq(const _Tp&, const _Up&)
    316       { return true; }
    317      
    318       static bool __less(const _Tp&, const _Up&)
    319       { return false; }
    320     };
    321 
    322   template<typename... _TElements, typename... _UElements>
    323     bool
    324     operator==(const tuple<_TElements...>& __t,
    325 	       const tuple<_UElements...>& __u)
    326     {
    327       typedef tuple<_TElements...> _Tp;
    328       typedef tuple<_UElements...> _Up;
    329       return (__tuple_compare<tuple_size<_Tp>::value - tuple_size<_Up>::value,
    330 	      0, tuple_size<_Tp>::value, _Tp, _Up>::__eq(__t, __u));
    331     }
    332 
    333   template<typename... _TElements, typename... _UElements>
    334     bool
    335     operator<(const tuple<_TElements...>& __t,
    336 	      const tuple<_UElements...>& __u)
    337     {
    338       typedef tuple<_TElements...> _Tp;
    339       typedef tuple<_UElements...> _Up;
    340       return (__tuple_compare<tuple_size<_Tp>::value - tuple_size<_Up>::value,
    341 	      0, tuple_size<_Tp>::value, _Tp, _Up>::__less(__t, __u));
    342     }
    343 
    344   template<typename... _TElements, typename... _UElements>
    345     inline bool
    346     operator!=(const tuple<_TElements...>& __t,
    347 	       const tuple<_UElements...>& __u)
    348     { return !(__t == __u); }
    349 
    350   template<typename... _TElements, typename... _UElements>
    351     inline bool
    352     operator>(const tuple<_TElements...>& __t,
    353 	      const tuple<_UElements...>& __u)
    354     { return __u < __t; }
    355 
    356   template<typename... _TElements, typename... _UElements>
    357     inline bool
    358     operator<=(const tuple<_TElements...>& __t,
    359 	       const tuple<_UElements...>& __u)
    360     { return !(__u < __t); }
    361 
    362   template<typename... _TElements, typename... _UElements>
    363     inline bool
    364     operator>=(const tuple<_TElements...>& __t,
    365 	       const tuple<_UElements...>& __u)
    366     { return !(__t < __u); }
    367 
    368   template<typename _Tp>
    369     class reference_wrapper;
    370 
    371   // Helper which adds a reference to a type when given a reference_wrapper
    372   template<typename _Tp>
    373     struct __strip_reference_wrapper
    374     {
    375       typedef _Tp __type;
    376     };
    377 
    378   template<typename _Tp>
    379     struct __strip_reference_wrapper<reference_wrapper<_Tp> >
    380     {
    381       typedef _Tp& __type;
    382     };
    383 
    384   template<typename _Tp>
    385     struct __strip_reference_wrapper<const reference_wrapper<_Tp> >
    386     {
    387       typedef _Tp& __type;
    388     };
    389 
    390   template<typename... _Elements>
    391     inline tuple<typename __strip_reference_wrapper<_Elements>::__type...>
    392     make_tuple(_Elements... __args)
    393     {
    394       typedef tuple<typename __strip_reference_wrapper<_Elements>::__type...>
    395         __result_type;
    396       return __result_type(__args...);
    397     }
    398 
    399   template<typename... _Elements>
    400     inline tuple<_Elements&...>
    401     tie(_Elements&... __args)
    402     {
    403       return tuple<_Elements&...>(__args...);
    404     }
    405 
    406   // A class (and instance) which can be used in 'tie' when an element
    407   // of a tuple is not required
    408   struct _Swallow_assign
    409   {
    410     template<class _Tp>
    411       _Swallow_assign&
    412       operator=(const _Tp&)
    413       { return *this; }
    414   };
    415 
    416   // TODO: Put this in some kind of shared file.
    417   namespace
    418   {
    419     _Swallow_assign ignore;
    420   }; // anonymous namespace
    421 
    422 _GLIBCXX_END_NAMESPACE_VERSION
    423 }
    424 }
    425 
    426 #endif // _GLIBCXX_TR1_TUPLE
    427