/* * Copyright 2013 Google Inc. * * Use of this source code is governed by a BSD-style license that can be * found in the LICENSE file. * * * This header provides some of the helpers (std::integral_constant) and * type transformations (std::conditional) which will become available with * C++11 in the type_traits header. * * Because we lack constexpr, we cannot mimic * std::integral_constant::'constexpr operator T()'. * As a result we introduce SkTBool and SkTIf similar to Boost in order to * minimize the visual noise of many uses of '::value'. */ #ifndef SkTLogic_DEFINED #define SkTLogic_DEFINED /** Represents a templated integer constant. * Pre-C++11 version of std::integral_constant. */ template struct SkTIntegralConstant { static const T value = v; typedef T value_type; typedef SkTIntegralConstant type; }; /** Convenience specialization of SkTIntegralConstant. */ template struct SkTBool : SkTIntegralConstant { }; /** Pre-C++11 version of std::is_empty. */ template class SkTIsEmpty { struct Derived : public T { char unused; }; public: static const bool value = sizeof(Derived) == sizeof(char); }; /** Pre-C++11 version of std::true_type. */ typedef SkTBool SkTrue; /** Pre-C++11 version of std::false_type. */ typedef SkTBool SkFalse; /** SkTIf_c::type = (condition) ? T : F; * Pre-C++11 version of std::conditional. */ template struct SkTIf_c { typedef F type; }; template struct SkTIf_c { typedef T type; }; /** SkTIf::type = (Condition::value) ? T : F; */ template struct SkTIf { typedef typename SkTIf_c(Condition::value), T, F>::type type; }; /** SkTMux::type = (a && b) ? Both : (a) ? A : (b) ? B : Neither; */ template struct SkTMux { typedef typename SkTIf::type, typename SkTIf::type>::type type; }; /** SkTEnableIf_c::type = (condition) ? T : [does not exist]; */ template struct SkTEnableIf_c { }; template struct SkTEnableIf_c { typedef T type; }; /** SkTEnableIf::type = (Condition::value) ? T : [does not exist]; */ template struct SkTEnableIf : public SkTEnableIf_c(Condition::value), T> { }; /** Use as a return type to enable a function only when cond_type::value is true, * like C++14's std::enable_if_t. E.g. (N.B. this is a dumb example.) * SK_WHEN(SkTrue, int) f(void* ptr) { return 1; } * SK_WHEN(!SkTrue, int) f(void* ptr) { return 2; } */ #define SK_WHEN(cond_prefix, T) typename SkTEnableIf_c::type #define SK_WHEN_C(cond, T) typename SkTEnableIf_c::type // See http://en.wikibooks.org/wiki/More_C++_Idioms/Member_Detector #define SK_CREATE_MEMBER_DETECTOR(member) \ template \ class HasMember_##member { \ struct Fallback { int member; }; \ struct Derived : T, Fallback {}; \ template struct Check; \ template static uint8_t func(Check*); \ template static uint16_t func(...); \ public: \ typedef HasMember_##member type; \ static const bool value = sizeof(func(NULL)) == sizeof(uint16_t); \ } // Same sort of thing as SK_CREATE_MEMBER_DETECTOR, but checks for the existence of a nested type. #define SK_CREATE_TYPE_DETECTOR(type) \ template \ class HasType_##type { \ template static uint8_t func(typename U::type*); \ template static uint16_t func(...); \ public: \ static const bool value = sizeof(func(NULL)) == sizeof(uint8_t); \ } #endif