| $$ -*- mode: c++; -*- |
| $var n = 10 $$ Maximum number of tuple fields we want to support. |
| $$ This meta comment fixes auto-indentation in Emacs. }} |
| // Copyright 2009 Google Inc. |
| // All Rights Reserved. |
| // |
| // Redistribution and use in source and binary forms, with or without |
| // modification, are permitted provided that the following conditions are |
| // met: |
| // |
| // * Redistributions of source code must retain the above copyright |
| // notice, this list of conditions and the following disclaimer. |
| // * Redistributions in binary form must reproduce the above |
| // copyright notice, this list of conditions and the following disclaimer |
| // in the documentation and/or other materials provided with the |
| // distribution. |
| // * Neither the name of Google Inc. nor the names of its |
| // contributors may be used to endorse or promote products derived from |
| // this software without specific prior written permission. |
| // |
| // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS |
| // "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT |
| // LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR |
| // A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT |
| // OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, |
| // SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT |
| // LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, |
| // DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY |
| // THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT |
| // (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE |
| // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
| // |
| // Author: wan@google.com (Zhanyong Wan) |
| |
| // Implements a subset of TR1 tuple needed by Google Test and Google Mock. |
| |
| #ifndef GTEST_INCLUDE_GTEST_INTERNAL_GTEST_TUPLE_H_ |
| #define GTEST_INCLUDE_GTEST_INTERNAL_GTEST_TUPLE_H_ |
| |
| #include <utility> // For ::std::pair. |
| |
| // The compiler used in Symbian has a bug that prevents us from declaring the |
| // tuple template as a friend (it complains that tuple is redefined). This |
| // hack bypasses the bug by declaring the members that should otherwise be |
| // private as public. |
| // Sun Studio versions < 12 also have the above bug. |
| #if defined(__SYMBIAN32__) || (defined(__SUNPRO_CC) && __SUNPRO_CC < 0x590) |
| # define GTEST_DECLARE_TUPLE_AS_FRIEND_ public: |
| #else |
| # define GTEST_DECLARE_TUPLE_AS_FRIEND_ \ |
| template <GTEST_$(n)_TYPENAMES_(U)> friend class tuple; \ |
| private: |
| #endif |
| |
| // Visual Studio 2010, 2012, and 2013 define symbols in std::tr1 that conflict |
| // with our own definitions. Therefore using our own tuple does not work on |
| // those compilers. |
| #if defined(_MSC_VER) && _MSC_VER >= 1600 /* 1600 is Visual Studio 2010 */ |
| # error "gtest's tuple doesn't compile on Visual Studio 2010 or later. \ |
| GTEST_USE_OWN_TR1_TUPLE must be set to 0 on those compilers." |
| #endif |
| |
| |
| $range i 0..n-1 |
| $range j 0..n |
| $range k 1..n |
| // GTEST_n_TUPLE_(T) is the type of an n-tuple. |
| #define GTEST_0_TUPLE_(T) tuple<> |
| |
| $for k [[ |
| $range m 0..k-1 |
| $range m2 k..n-1 |
| #define GTEST_$(k)_TUPLE_(T) tuple<$for m, [[T##$m]]$for m2 [[, void]]> |
| |
| ]] |
| |
| // GTEST_n_TYPENAMES_(T) declares a list of n typenames. |
| |
| $for j [[ |
| $range m 0..j-1 |
| #define GTEST_$(j)_TYPENAMES_(T) $for m, [[typename T##$m]] |
| |
| |
| ]] |
| |
| // In theory, defining stuff in the ::std namespace is undefined |
| // behavior. We can do this as we are playing the role of a standard |
| // library vendor. |
| namespace std { |
| namespace tr1 { |
| |
| template <$for i, [[typename T$i = void]]> |
| class tuple; |
| |
| // Anything in namespace gtest_internal is Google Test's INTERNAL |
| // IMPLEMENTATION DETAIL and MUST NOT BE USED DIRECTLY in user code. |
| namespace gtest_internal { |
| |
| // ByRef<T>::type is T if T is a reference; otherwise it's const T&. |
| template <typename T> |
| struct ByRef { typedef const T& type; }; // NOLINT |
| template <typename T> |
| struct ByRef<T&> { typedef T& type; }; // NOLINT |
| |
| // A handy wrapper for ByRef. |
| #define GTEST_BY_REF_(T) typename ::std::tr1::gtest_internal::ByRef<T>::type |
| |
| // AddRef<T>::type is T if T is a reference; otherwise it's T&. This |
| // is the same as tr1::add_reference<T>::type. |
| template <typename T> |
| struct AddRef { typedef T& type; }; // NOLINT |
| template <typename T> |
| struct AddRef<T&> { typedef T& type; }; // NOLINT |
| |
| // A handy wrapper for AddRef. |
| #define GTEST_ADD_REF_(T) typename ::std::tr1::gtest_internal::AddRef<T>::type |
| |
| // A helper for implementing get<k>(). |
| template <int k> class Get; |
| |
| // A helper for implementing tuple_element<k, T>. kIndexValid is true |
| // iff k < the number of fields in tuple type T. |
| template <bool kIndexValid, int kIndex, class Tuple> |
| struct TupleElement; |
| |
| |
| $for i [[ |
| template <GTEST_$(n)_TYPENAMES_(T)> |
| struct TupleElement<true, $i, GTEST_$(n)_TUPLE_(T) > { |
| typedef T$i type; |
| }; |
| |
| |
| ]] |
| } // namespace gtest_internal |
| |
| template <> |
| class tuple<> { |
| public: |
| tuple() {} |
| tuple(const tuple& /* t */) {} |
| tuple& operator=(const tuple& /* t */) { return *this; } |
| }; |
| |
| |
| $for k [[ |
| $range m 0..k-1 |
| template <GTEST_$(k)_TYPENAMES_(T)> |
| class $if k < n [[GTEST_$(k)_TUPLE_(T)]] $else [[tuple]] { |
| public: |
| template <int k> friend class gtest_internal::Get; |
| |
| tuple() : $for m, [[f$(m)_()]] {} |
| |
| explicit tuple($for m, [[GTEST_BY_REF_(T$m) f$m]]) : [[]] |
| $for m, [[f$(m)_(f$m)]] {} |
| |
| tuple(const tuple& t) : $for m, [[f$(m)_(t.f$(m)_)]] {} |
| |
| template <GTEST_$(k)_TYPENAMES_(U)> |
| tuple(const GTEST_$(k)_TUPLE_(U)& t) : $for m, [[f$(m)_(t.f$(m)_)]] {} |
| |
| $if k == 2 [[ |
| template <typename U0, typename U1> |
| tuple(const ::std::pair<U0, U1>& p) : f0_(p.first), f1_(p.second) {} |
| |
| ]] |
| |
| tuple& operator=(const tuple& t) { return CopyFrom(t); } |
| |
| template <GTEST_$(k)_TYPENAMES_(U)> |
| tuple& operator=(const GTEST_$(k)_TUPLE_(U)& t) { |
| return CopyFrom(t); |
| } |
| |
| $if k == 2 [[ |
| template <typename U0, typename U1> |
| tuple& operator=(const ::std::pair<U0, U1>& p) { |
| f0_ = p.first; |
| f1_ = p.second; |
| return *this; |
| } |
| |
| ]] |
| |
| GTEST_DECLARE_TUPLE_AS_FRIEND_ |
| |
| template <GTEST_$(k)_TYPENAMES_(U)> |
| tuple& CopyFrom(const GTEST_$(k)_TUPLE_(U)& t) { |
| |
| $for m [[ |
| f$(m)_ = t.f$(m)_; |
| |
| ]] |
| return *this; |
| } |
| |
| |
| $for m [[ |
| T$m f$(m)_; |
| |
| ]] |
| }; |
| |
| |
| ]] |
| // 6.1.3.2 Tuple creation functions. |
| |
| // Known limitations: we don't support passing an |
| // std::tr1::reference_wrapper<T> to make_tuple(). And we don't |
| // implement tie(). |
| |
| inline tuple<> make_tuple() { return tuple<>(); } |
| |
| $for k [[ |
| $range m 0..k-1 |
| |
| template <GTEST_$(k)_TYPENAMES_(T)> |
| inline GTEST_$(k)_TUPLE_(T) make_tuple($for m, [[const T$m& f$m]]) { |
| return GTEST_$(k)_TUPLE_(T)($for m, [[f$m]]); |
| } |
| |
| ]] |
| |
| // 6.1.3.3 Tuple helper classes. |
| |
| template <typename Tuple> struct tuple_size; |
| |
| |
| $for j [[ |
| template <GTEST_$(j)_TYPENAMES_(T)> |
| struct tuple_size<GTEST_$(j)_TUPLE_(T) > { |
| static const int value = $j; |
| }; |
| |
| |
| ]] |
| template <int k, class Tuple> |
| struct tuple_element { |
| typedef typename gtest_internal::TupleElement< |
| k < (tuple_size<Tuple>::value), k, Tuple>::type type; |
| }; |
| |
| #define GTEST_TUPLE_ELEMENT_(k, Tuple) typename tuple_element<k, Tuple >::type |
| |
| // 6.1.3.4 Element access. |
| |
| namespace gtest_internal { |
| |
| |
| $for i [[ |
| template <> |
| class Get<$i> { |
| public: |
| template <class Tuple> |
| static GTEST_ADD_REF_(GTEST_TUPLE_ELEMENT_($i, Tuple)) |
| Field(Tuple& t) { return t.f$(i)_; } // NOLINT |
| |
| template <class Tuple> |
| static GTEST_BY_REF_(GTEST_TUPLE_ELEMENT_($i, Tuple)) |
| ConstField(const Tuple& t) { return t.f$(i)_; } |
| }; |
| |
| |
| ]] |
| } // namespace gtest_internal |
| |
| template <int k, GTEST_$(n)_TYPENAMES_(T)> |
| GTEST_ADD_REF_(GTEST_TUPLE_ELEMENT_(k, GTEST_$(n)_TUPLE_(T))) |
| get(GTEST_$(n)_TUPLE_(T)& t) { |
| return gtest_internal::Get<k>::Field(t); |
| } |
| |
| template <int k, GTEST_$(n)_TYPENAMES_(T)> |
| GTEST_BY_REF_(GTEST_TUPLE_ELEMENT_(k, GTEST_$(n)_TUPLE_(T))) |
| get(const GTEST_$(n)_TUPLE_(T)& t) { |
| return gtest_internal::Get<k>::ConstField(t); |
| } |
| |
| // 6.1.3.5 Relational operators |
| |
| // We only implement == and !=, as we don't have a need for the rest yet. |
| |
| namespace gtest_internal { |
| |
| // SameSizeTuplePrefixComparator<k, k>::Eq(t1, t2) returns true if the |
| // first k fields of t1 equals the first k fields of t2. |
| // SameSizeTuplePrefixComparator(k1, k2) would be a compiler error if |
| // k1 != k2. |
| template <int kSize1, int kSize2> |
| struct SameSizeTuplePrefixComparator; |
| |
| template <> |
| struct SameSizeTuplePrefixComparator<0, 0> { |
| template <class Tuple1, class Tuple2> |
| static bool Eq(const Tuple1& /* t1 */, const Tuple2& /* t2 */) { |
| return true; |
| } |
| }; |
| |
| template <int k> |
| struct SameSizeTuplePrefixComparator<k, k> { |
| template <class Tuple1, class Tuple2> |
| static bool Eq(const Tuple1& t1, const Tuple2& t2) { |
| return SameSizeTuplePrefixComparator<k - 1, k - 1>::Eq(t1, t2) && |
| ::std::tr1::get<k - 1>(t1) == ::std::tr1::get<k - 1>(t2); |
| } |
| }; |
| |
| } // namespace gtest_internal |
| |
| template <GTEST_$(n)_TYPENAMES_(T), GTEST_$(n)_TYPENAMES_(U)> |
| inline bool operator==(const GTEST_$(n)_TUPLE_(T)& t, |
| const GTEST_$(n)_TUPLE_(U)& u) { |
| return gtest_internal::SameSizeTuplePrefixComparator< |
| tuple_size<GTEST_$(n)_TUPLE_(T) >::value, |
| tuple_size<GTEST_$(n)_TUPLE_(U) >::value>::Eq(t, u); |
| } |
| |
| template <GTEST_$(n)_TYPENAMES_(T), GTEST_$(n)_TYPENAMES_(U)> |
| inline bool operator!=(const GTEST_$(n)_TUPLE_(T)& t, |
| const GTEST_$(n)_TUPLE_(U)& u) { return !(t == u); } |
| |
| // 6.1.4 Pairs. |
| // Unimplemented. |
| |
| } // namespace tr1 |
| } // namespace std |
| |
| |
| $for j [[ |
| #undef GTEST_$(j)_TUPLE_ |
| |
| ]] |
| |
| |
| $for j [[ |
| #undef GTEST_$(j)_TYPENAMES_ |
| |
| ]] |
| |
| #undef GTEST_DECLARE_TUPLE_AS_FRIEND_ |
| #undef GTEST_BY_REF_ |
| #undef GTEST_ADD_REF_ |
| #undef GTEST_TUPLE_ELEMENT_ |
| |
| #endif // GTEST_INCLUDE_GTEST_INTERNAL_GTEST_TUPLE_H_ |