1*6777b538SAndroid Build Coastguard Worker // Copyright 2011 The Chromium Authors 2*6777b538SAndroid Build Coastguard Worker // Use of this source code is governed by a BSD-style license that can be 3*6777b538SAndroid Build Coastguard Worker // found in the LICENSE file. 4*6777b538SAndroid Build Coastguard Worker 5*6777b538SAndroid Build Coastguard Worker #ifndef BASE_TEMPLATE_UTIL_H_ 6*6777b538SAndroid Build Coastguard Worker #define BASE_TEMPLATE_UTIL_H_ 7*6777b538SAndroid Build Coastguard Worker 8*6777b538SAndroid Build Coastguard Worker #include <stddef.h> 9*6777b538SAndroid Build Coastguard Worker 10*6777b538SAndroid Build Coastguard Worker #include <iosfwd> 11*6777b538SAndroid Build Coastguard Worker #include <iterator> 12*6777b538SAndroid Build Coastguard Worker #include <type_traits> 13*6777b538SAndroid Build Coastguard Worker #include <utility> 14*6777b538SAndroid Build Coastguard Worker 15*6777b538SAndroid Build Coastguard Worker #include "base/compiler_specific.h" 16*6777b538SAndroid Build Coastguard Worker 17*6777b538SAndroid Build Coastguard Worker namespace base { 18*6777b538SAndroid Build Coastguard Worker 19*6777b538SAndroid Build Coastguard Worker namespace internal { 20*6777b538SAndroid Build Coastguard Worker 21*6777b538SAndroid Build Coastguard Worker // Used to detect whether the given type is an iterator. This is normally used 22*6777b538SAndroid Build Coastguard Worker // with std::enable_if to provide disambiguation for functions that take 23*6777b538SAndroid Build Coastguard Worker // templatzed iterators as input. 24*6777b538SAndroid Build Coastguard Worker template <typename T, typename = void> 25*6777b538SAndroid Build Coastguard Worker struct is_iterator : std::false_type {}; 26*6777b538SAndroid Build Coastguard Worker 27*6777b538SAndroid Build Coastguard Worker template <typename T> 28*6777b538SAndroid Build Coastguard Worker struct is_iterator< 29*6777b538SAndroid Build Coastguard Worker T, 30*6777b538SAndroid Build Coastguard Worker std::void_t<typename std::iterator_traits<T>::iterator_category>> 31*6777b538SAndroid Build Coastguard Worker : std::true_type {}; 32*6777b538SAndroid Build Coastguard Worker 33*6777b538SAndroid Build Coastguard Worker // Helper to express preferences in an overload set. If more than one overload 34*6777b538SAndroid Build Coastguard Worker // are available for a given set of parameters the overload with the higher 35*6777b538SAndroid Build Coastguard Worker // priority will be chosen. 36*6777b538SAndroid Build Coastguard Worker template <size_t I> 37*6777b538SAndroid Build Coastguard Worker struct priority_tag : priority_tag<I - 1> {}; 38*6777b538SAndroid Build Coastguard Worker 39*6777b538SAndroid Build Coastguard Worker template <> 40*6777b538SAndroid Build Coastguard Worker struct priority_tag<0> {}; 41*6777b538SAndroid Build Coastguard Worker 42*6777b538SAndroid Build Coastguard Worker } // namespace internal 43*6777b538SAndroid Build Coastguard Worker 44*6777b538SAndroid Build Coastguard Worker namespace internal { 45*6777b538SAndroid Build Coastguard Worker 46*6777b538SAndroid Build Coastguard Worker // The indirection with std::is_enum<T> is required, because instantiating 47*6777b538SAndroid Build Coastguard Worker // std::underlying_type_t<T> when T is not an enum is UB prior to C++20. 48*6777b538SAndroid Build Coastguard Worker template <typename T, bool = std::is_enum_v<T>> 49*6777b538SAndroid Build Coastguard Worker struct IsScopedEnumImpl : std::false_type {}; 50*6777b538SAndroid Build Coastguard Worker 51*6777b538SAndroid Build Coastguard Worker template <typename T> 52*6777b538SAndroid Build Coastguard Worker struct IsScopedEnumImpl<T, /*std::is_enum_v<T>=*/true> 53*6777b538SAndroid Build Coastguard Worker : std::negation<std::is_convertible<T, std::underlying_type_t<T>>> {}; 54*6777b538SAndroid Build Coastguard Worker 55*6777b538SAndroid Build Coastguard Worker } // namespace internal 56*6777b538SAndroid Build Coastguard Worker 57*6777b538SAndroid Build Coastguard Worker // Implementation of C++23's std::is_scoped_enum 58*6777b538SAndroid Build Coastguard Worker // 59*6777b538SAndroid Build Coastguard Worker // Reference: https://en.cppreference.com/w/cpp/types/is_scoped_enum 60*6777b538SAndroid Build Coastguard Worker template <typename T> 61*6777b538SAndroid Build Coastguard Worker struct is_scoped_enum : internal::IsScopedEnumImpl<T> {}; 62*6777b538SAndroid Build Coastguard Worker 63*6777b538SAndroid Build Coastguard Worker } // namespace base 64*6777b538SAndroid Build Coastguard Worker 65*6777b538SAndroid Build Coastguard Worker #endif // BASE_TEMPLATE_UTIL_H_ 66