xref: /aosp_15_r20/external/cronet/base/template_util.h (revision 6777b5387eb2ff775bb5750e3f5d96f37fb7352b)
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