/* * 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 (like std::enable_if_t) which will * become available with C++14 in the type_traits header (in the skstd * namespace). This header also provides several Skia specific additions such * as SK_WHEN and the sknonstd namespace. */ #ifndef SkTLogic_DEFINED #define SkTLogic_DEFINED #include #include #include #include #include namespace skstd { template using bool_constant = std::integral_constant; template using conditional_t = typename std::conditional::type; template using enable_if_t = typename std::enable_if::type; template using remove_const_t = typename std::remove_const::type; template using remove_volatile_t = typename std::remove_volatile::type; template using remove_cv_t = typename std::remove_cv::type; template using remove_pointer_t = typename std::remove_pointer::type; template using remove_reference_t = typename std::remove_reference::type; template using remove_extent_t = typename std::remove_extent::type; template using add_const_t = typename std::add_const::type; template using add_volatile_t = typename std::add_volatile::type; template using add_cv_t = typename std::add_cv::type; template using add_pointer_t = typename std::add_pointer::type; template using add_lvalue_reference_t = typename std::add_lvalue_reference::type; template using result_of_t = typename std::result_of::type; template using common_type_t = typename std::common_type::type; // Chromium currently requires gcc 4.8.2 or a recent clang compiler, but uses libstdc++4.6.4. // Note that Precise actually uses libstdc++4.6.3. // Unfortunately, libstdc++ STL before libstdc++4.7 do not define std::underlying_type. // Newer gcc and clang compilers have __underlying_type which does not depend on runtime support. // See https://gcc.gnu.org/onlinedocs/libstdc++/manual/abi.html for __GLIBCXX__ values. // Unfortunately __GLIBCXX__ is a date, but no updates to versions before 4.7 are now anticipated. #define SK_GLIBCXX_4_7_0 20120322 // Updates to versions before 4.7 but released after 4.7 was released. #define SK_GLIBCXX_4_5_4 20120702 #define SK_GLIBCXX_4_6_4 20121127 #if defined(__GLIBCXX__) && (__GLIBCXX__ < SK_GLIBCXX_4_7_0 || \ __GLIBCXX__ == SK_GLIBCXX_4_5_4 || \ __GLIBCXX__ == SK_GLIBCXX_4_6_4) template struct underlying_type { using type = __underlying_type(T); }; #else template using underlying_type = std::underlying_type; #endif template using underlying_type_t = typename skstd::underlying_type::type; template struct index_sequence { using type = index_sequence; using value_type = size_t; static constexpr std::size_t size() noexcept { return sizeof...(Ints); } }; template struct make_index_sequence_combine; template struct make_index_sequence_combine, skstd::index_sequence> : skstd::index_sequence { }; template struct make_index_sequence : make_index_sequence_combine::type, typename skstd::make_index_sequence::type>{}; template<> struct make_index_sequence<0> : skstd::index_sequence< >{}; template<> struct make_index_sequence<1> : skstd::index_sequence<0>{}; } // namespace skstd // The sknonstd namespace contains things we would like to be proposed and feel std-ish. namespace sknonstd { // The name 'copy' here is fraught with peril. In this case it means 'append', not 'overwrite'. // Alternate proposed names are 'propagate', 'augment', or 'append' (and 'add', but already taken). // std::experimental::propagate_const already exists for other purposes in TSv2. // These also follow the pattern used by boost. template struct copy_const { using type = skstd::conditional_t::value, skstd::add_const_t, D>; }; template using copy_const_t = typename copy_const::type; template struct copy_volatile { using type = skstd::conditional_t::value, skstd::add_volatile_t, D>; }; template using copy_volatile_t = typename copy_volatile::type; template struct copy_cv { using type = copy_volatile_t, S>; }; template using copy_cv_t = typename copy_cv::type; // The name 'same' here means 'overwrite'. // Alternate proposed names are 'replace', 'transfer', or 'qualify_from'. // same_xxx can be written as copy_xxx, S> template using same_const = copy_const, S>; template using same_const_t = typename same_const::type; template using same_volatile =copy_volatile,S>; template using same_volatile_t = typename same_volatile::type; template using same_cv = copy_cv, S>; template using same_cv_t = typename same_cv::type; } // namespace sknonstd // Just a pithier wrapper for enable_if_t. #define SK_WHEN(condition, T) skstd::enable_if_t #endif