bungeman@google.com | f5cc5b1 | 2013-07-12 18:22:49 +0000 | [diff] [blame] | 1 | /* |
| 2 | * Copyright 2013 Google Inc. |
| 3 | * |
| 4 | * Use of this source code is governed by a BSD-style license that can be |
| 5 | * found in the LICENSE file. |
| 6 | * |
| 7 | * |
| 8 | * This header provides some of the helpers (std::integral_constant) and |
| 9 | * type transformations (std::conditional) which will become available with |
| 10 | * C++11 in the type_traits header. |
| 11 | * |
| 12 | * Because we lack constexpr, we cannot mimic |
| 13 | * std::integral_constant::'constexpr operator T()'. |
| 14 | * As a result we introduce SkTBool and SkTIf similar to Boost in order to |
| 15 | * minimize the visual noise of many uses of '::value'. |
| 16 | */ |
| 17 | |
| 18 | #ifndef SkTLogic_DEFINED |
| 19 | #define SkTLogic_DEFINED |
| 20 | |
| 21 | /** Represents a templated integer constant. |
| 22 | * Pre-C++11 version of std::integral_constant. |
| 23 | */ |
| 24 | template <typename T, T v> struct SkTIntegralConstant { |
| 25 | static const T value = v; |
| 26 | typedef T value_type; |
| 27 | typedef SkTIntegralConstant<T, v> type; |
| 28 | }; |
| 29 | |
| 30 | /** Convenience specialization of SkTIntegralConstant. */ |
| 31 | template <bool b> struct SkTBool : SkTIntegralConstant<bool, b> { }; |
| 32 | |
commit-bot@chromium.org | 08bf86c | 2014-05-07 18:01:57 +0000 | [diff] [blame] | 33 | /** Pre-C++11 version of std::is_empty<T>. */ |
| 34 | template <typename T> |
| 35 | class SkTIsEmpty { |
| 36 | struct Derived : public T { char unused; }; |
| 37 | public: |
| 38 | static const bool value = sizeof(Derived) == sizeof(char); |
| 39 | }; |
| 40 | |
bungeman@google.com | f5cc5b1 | 2013-07-12 18:22:49 +0000 | [diff] [blame] | 41 | /** Pre-C++11 version of std::true_type. */ |
| 42 | typedef SkTBool<true> SkTrue; |
| 43 | |
| 44 | /** Pre-C++11 version of std::false_type. */ |
| 45 | typedef SkTBool<false> SkFalse; |
| 46 | |
| 47 | /** SkTIf_c::type = (condition) ? T : F; |
| 48 | * Pre-C++11 version of std::conditional. |
| 49 | */ |
| 50 | template <bool condition, typename T, typename F> struct SkTIf_c { |
| 51 | typedef F type; |
| 52 | }; |
| 53 | template <typename T, typename F> struct SkTIf_c<true, T, F> { |
| 54 | typedef T type; |
| 55 | }; |
| 56 | |
| 57 | /** SkTIf::type = (Condition::value) ? T : F; */ |
| 58 | template <typename Condition, typename T, typename F> struct SkTIf { |
| 59 | typedef typename SkTIf_c<static_cast<bool>(Condition::value), T, F>::type type; |
| 60 | }; |
| 61 | |
| 62 | /** SkTMux::type = (a && b) ? Both : (a) ? A : (b) ? B : Neither; */ |
| 63 | template <typename a, typename b, typename Both, typename A, typename B, typename Neither> |
| 64 | struct SkTMux { |
| 65 | typedef typename SkTIf<a, typename SkTIf<b, Both, A>::type, |
| 66 | typename SkTIf<b, B, Neither>::type>::type type; |
| 67 | }; |
| 68 | |
commit-bot@chromium.org | 2e0c32a | 2014-04-28 16:19:45 +0000 | [diff] [blame] | 69 | /** SkTEnableIf_c::type = (condition) ? T : [does not exist]; */ |
| 70 | template <bool condition, class T = void> struct SkTEnableIf_c { }; |
| 71 | template <class T> struct SkTEnableIf_c<true, T> { |
| 72 | typedef T type; |
| 73 | }; |
| 74 | |
| 75 | /** SkTEnableIf::type = (Condition::value) ? T : [does not exist]; */ |
| 76 | template <class Condition, class T = void> struct SkTEnableIf |
| 77 | : public SkTEnableIf_c<static_cast<bool>(Condition::value), T> { }; |
| 78 | |
| 79 | /** Use as a return type to enable a function only when cond_type::value is true, |
| 80 | * like C++14's std::enable_if_t. E.g. (N.B. this is a dumb example.) |
| 81 | * SK_WHEN(SkTrue, int) f(void* ptr) { return 1; } |
| 82 | * SK_WHEN(!SkTrue, int) f(void* ptr) { return 2; } |
| 83 | */ |
| 84 | #define SK_WHEN(cond_prefix, T) typename SkTEnableIf_c<cond_prefix::value, T>::type |
mtklein | 97312d0 | 2015-04-01 08:11:16 -0700 | [diff] [blame] | 85 | #define SK_WHEN_C(cond, T) typename SkTEnableIf_c<cond, T>::type |
commit-bot@chromium.org | 2e0c32a | 2014-04-28 16:19:45 +0000 | [diff] [blame] | 86 | |
| 87 | // See http://en.wikibooks.org/wiki/More_C++_Idioms/Member_Detector |
| 88 | #define SK_CREATE_MEMBER_DETECTOR(member) \ |
| 89 | template <typename T> \ |
| 90 | class HasMember_##member { \ |
| 91 | struct Fallback { int member; }; \ |
| 92 | struct Derived : T, Fallback {}; \ |
| 93 | template <typename U, U> struct Check; \ |
| 94 | template <typename U> static uint8_t func(Check<int Fallback::*, &U::member>*); \ |
| 95 | template <typename U> static uint16_t func(...); \ |
| 96 | public: \ |
| 97 | typedef HasMember_##member type; \ |
| 98 | static const bool value = sizeof(func<Derived>(NULL)) == sizeof(uint16_t); \ |
| 99 | } |
| 100 | |
commit-bot@chromium.org | 73fffeb | 2014-05-05 21:59:52 +0000 | [diff] [blame] | 101 | // Same sort of thing as SK_CREATE_MEMBER_DETECTOR, but checks for the existence of a nested type. |
| 102 | #define SK_CREATE_TYPE_DETECTOR(type) \ |
| 103 | template <typename T> \ |
| 104 | class HasType_##type { \ |
| 105 | template <typename U> static uint8_t func(typename U::type*); \ |
| 106 | template <typename U> static uint16_t func(...); \ |
| 107 | public: \ |
| 108 | static const bool value = sizeof(func<T>(NULL)) == sizeof(uint8_t); \ |
| 109 | } |
| 110 | |
bungeman@google.com | 4b3ef5a | 2013-07-12 18:31:59 +0000 | [diff] [blame] | 111 | #endif |