OLD | NEW |
1 /* | 1 /* |
2 * Copyright 2017 The WebRTC Project Authors. All rights reserved. | 2 * Copyright 2017 The WebRTC Project Authors. All rights reserved. |
3 * | 3 * |
4 * Use of this source code is governed by a BSD-style license | 4 * Use of this source code is governed by a BSD-style license |
5 * that can be found in the LICENSE file in the root of the source | 5 * that can be found in the LICENSE file in the root of the source |
6 * tree. An additional intellectual property rights grant can be found | 6 * tree. An additional intellectual property rights grant can be found |
7 * in the file PATENTS. All contributing project authors may | 7 * in the file PATENTS. All contributing project authors may |
8 * be found in the AUTHORS file in the root of the source tree. | 8 * be found in the AUTHORS file in the root of the source tree. |
9 */ | 9 */ |
10 | 10 |
11 #include <algorithm> | 11 #include <algorithm> |
12 #include <limits> | 12 #include <limits> |
13 | 13 |
14 #include "webrtc/base/safe_minmax.h" | 14 #include "webrtc/base/safe_minmax.h" |
15 #include "webrtc/test/gtest.h" | 15 #include "webrtc/test/gtest.h" |
16 | 16 |
17 namespace rtc { | 17 namespace rtc { |
18 | 18 |
19 namespace { | 19 namespace { |
20 | 20 |
21 // Functions that check that SafeMin() and SafeMax() return the specified type. | 21 // Functions that check that SafeMin(), SafeMax(), and SafeClamp() return the |
22 // The functions that end in "R" use an explicitly given return type. | 22 // specified type. The functions that end in "R" use an explicitly given return |
| 23 // type. |
23 | 24 |
24 template <typename T1, typename T2, typename Tmin, typename Tmax> | 25 template <typename T1, typename T2, typename Tmin, typename Tmax> |
25 constexpr bool TypeCheckMinMax() { | 26 constexpr bool TypeCheckMinMax() { |
26 return std::is_same<decltype(SafeMin(std::declval<T1>(), std::declval<T2>())), | 27 return std::is_same<decltype(SafeMin(std::declval<T1>(), std::declval<T2>())), |
27 Tmin>::value && | 28 Tmin>::value && |
28 std::is_same<decltype(SafeMax(std::declval<T1>(), std::declval<T2>())), | 29 std::is_same<decltype(SafeMax(std::declval<T1>(), std::declval<T2>())), |
29 Tmax>::value; | 30 Tmax>::value; |
30 } | 31 } |
31 | 32 |
32 template <typename T1, typename T2, typename R> | 33 template <typename T1, typename T2, typename R> |
33 constexpr bool TypeCheckMinR() { | 34 constexpr bool TypeCheckMinR() { |
34 return std::is_same< | 35 return std::is_same< |
35 decltype(SafeMin<R>(std::declval<T1>(), std::declval<T2>())), R>::value; | 36 decltype(SafeMin<R>(std::declval<T1>(), std::declval<T2>())), R>::value; |
36 } | 37 } |
37 | 38 |
38 template <typename T1, typename T2, typename R> | 39 template <typename T1, typename T2, typename R> |
39 constexpr bool TypeCheckMaxR() { | 40 constexpr bool TypeCheckMaxR() { |
40 return std::is_same< | 41 return std::is_same< |
41 decltype(SafeMax<R>(std::declval<T1>(), std::declval<T2>())), R>::value; | 42 decltype(SafeMax<R>(std::declval<T1>(), std::declval<T2>())), R>::value; |
42 } | 43 } |
43 | 44 |
| 45 template <typename L, typename H, typename T, typename R> |
| 46 constexpr bool TypeCheckClamp() { |
| 47 return std::is_same<decltype(SafeClamp(std::declval<L>(), std::declval<H>(), |
| 48 std::declval<T>())), |
| 49 R>::value; |
| 50 } |
| 51 |
| 52 template <typename L, typename H, typename T, typename R> |
| 53 constexpr bool TypeCheckClampR() { |
| 54 return std::is_same<decltype(SafeClamp<R>(std::declval<L>(), |
| 55 std::declval<H>(), |
| 56 std::declval<T>())), |
| 57 R>::value; |
| 58 } |
| 59 |
44 // clang-format off | 60 // clang-format off |
45 | 61 |
46 // SafeMin/SafeMax: Check that all combinations of signed/unsigned 8/64 bits | 62 // SafeMin/SafeMax: Check that all combinations of signed/unsigned 8/64 bits |
47 // give the correct default result type. | 63 // give the correct default result type. |
48 static_assert(TypeCheckMinMax< int8_t, int8_t, int8_t, int8_t>(), ""); | 64 static_assert(TypeCheckMinMax< int8_t, int8_t, int8_t, int8_t>(), ""); |
49 static_assert(TypeCheckMinMax< int8_t, uint8_t, int8_t, uint8_t>(), ""); | 65 static_assert(TypeCheckMinMax< int8_t, uint8_t, int8_t, uint8_t>(), ""); |
50 static_assert(TypeCheckMinMax< int8_t, int64_t, int64_t, int64_t>(), ""); | 66 static_assert(TypeCheckMinMax< int8_t, int64_t, int64_t, int64_t>(), ""); |
51 static_assert(TypeCheckMinMax< int8_t, uint64_t, int8_t, uint64_t>(), ""); | 67 static_assert(TypeCheckMinMax< int8_t, uint64_t, int8_t, uint64_t>(), ""); |
52 static_assert(TypeCheckMinMax< uint8_t, int8_t, int8_t, uint8_t>(), ""); | 68 static_assert(TypeCheckMinMax< uint8_t, int8_t, int8_t, uint8_t>(), ""); |
53 static_assert(TypeCheckMinMax< uint8_t, uint8_t, uint8_t, uint8_t>(), ""); | 69 static_assert(TypeCheckMinMax< uint8_t, uint8_t, uint8_t, uint8_t>(), ""); |
54 static_assert(TypeCheckMinMax< uint8_t, int64_t, int64_t, int64_t>(), ""); | 70 static_assert(TypeCheckMinMax< uint8_t, int64_t, int64_t, int64_t>(), ""); |
55 static_assert(TypeCheckMinMax< uint8_t, uint64_t, uint8_t, uint64_t>(), ""); | 71 static_assert(TypeCheckMinMax< uint8_t, uint64_t, uint8_t, uint64_t>(), ""); |
56 static_assert(TypeCheckMinMax< int64_t, int8_t, int64_t, int64_t>(), ""); | 72 static_assert(TypeCheckMinMax< int64_t, int8_t, int64_t, int64_t>(), ""); |
57 static_assert(TypeCheckMinMax< int64_t, uint8_t, int64_t, int64_t>(), ""); | 73 static_assert(TypeCheckMinMax< int64_t, uint8_t, int64_t, int64_t>(), ""); |
58 static_assert(TypeCheckMinMax< int64_t, int64_t, int64_t, int64_t>(), ""); | 74 static_assert(TypeCheckMinMax< int64_t, int64_t, int64_t, int64_t>(), ""); |
59 static_assert(TypeCheckMinMax< int64_t, uint64_t, int64_t, uint64_t>(), ""); | 75 static_assert(TypeCheckMinMax< int64_t, uint64_t, int64_t, uint64_t>(), ""); |
60 static_assert(TypeCheckMinMax<uint64_t, int8_t, int8_t, uint64_t>(), ""); | 76 static_assert(TypeCheckMinMax<uint64_t, int8_t, int8_t, uint64_t>(), ""); |
61 static_assert(TypeCheckMinMax<uint64_t, uint8_t, uint8_t, uint64_t>(), ""); | 77 static_assert(TypeCheckMinMax<uint64_t, uint8_t, uint8_t, uint64_t>(), ""); |
62 static_assert(TypeCheckMinMax<uint64_t, int64_t, int64_t, uint64_t>(), ""); | 78 static_assert(TypeCheckMinMax<uint64_t, int64_t, int64_t, uint64_t>(), ""); |
63 static_assert(TypeCheckMinMax<uint64_t, uint64_t, uint64_t, uint64_t>(), ""); | 79 static_assert(TypeCheckMinMax<uint64_t, uint64_t, uint64_t, uint64_t>(), ""); |
64 | 80 |
| 81 // SafeClamp: Check that all combinations of signed/unsigned 8/64 bits give the |
| 82 // correct result type. |
| 83 static_assert(TypeCheckClamp< int8_t, int8_t, int8_t, int8_t>(), ""); |
| 84 static_assert(TypeCheckClamp< int8_t, int8_t, uint8_t, int8_t>(), ""); |
| 85 static_assert(TypeCheckClamp< int8_t, int8_t, int64_t, int8_t>(), ""); |
| 86 static_assert(TypeCheckClamp< int8_t, int8_t, uint64_t, int8_t>(), ""); |
| 87 static_assert(TypeCheckClamp< int8_t, uint8_t, int8_t, int8_t>(), ""); |
| 88 static_assert(TypeCheckClamp< int8_t, uint8_t, uint8_t, uint8_t>(), ""); |
| 89 static_assert(TypeCheckClamp< int8_t, uint8_t, int64_t, int16_t>(), ""); |
| 90 static_assert(TypeCheckClamp< int8_t, uint8_t, uint64_t, uint8_t>(), ""); |
| 91 static_assert(TypeCheckClamp< int8_t, int64_t, int8_t, int8_t>(), ""); |
| 92 static_assert(TypeCheckClamp< int8_t, int64_t, uint8_t, int16_t>(), ""); |
| 93 static_assert(TypeCheckClamp< int8_t, int64_t, int64_t, int64_t>(), ""); |
| 94 static_assert(TypeCheckClamp< int8_t, int64_t, uint64_t, int64_t>(), ""); |
| 95 static_assert(TypeCheckClamp< int8_t, uint64_t, int8_t, int8_t>(), ""); |
| 96 static_assert(TypeCheckClamp< int8_t, uint64_t, uint8_t, uint8_t>(), ""); |
| 97 static_assert(TypeCheckClamp< int8_t, uint64_t, int64_t, int64_t>(), ""); |
| 98 static_assert(TypeCheckClamp< int8_t, uint64_t, uint64_t, uint64_t>(), ""); |
| 99 static_assert(TypeCheckClamp< uint8_t, int8_t, int8_t, int8_t>(), ""); |
| 100 static_assert(TypeCheckClamp< uint8_t, int8_t, uint8_t, uint8_t>(), ""); |
| 101 static_assert(TypeCheckClamp< uint8_t, int8_t, int64_t, int8_t>(), ""); |
| 102 static_assert(TypeCheckClamp< uint8_t, int8_t, uint64_t, uint8_t>(), ""); |
| 103 static_assert(TypeCheckClamp< uint8_t, uint8_t, int8_t, uint8_t>(), ""); |
| 104 static_assert(TypeCheckClamp< uint8_t, uint8_t, uint8_t, uint8_t>(), ""); |
| 105 static_assert(TypeCheckClamp< uint8_t, uint8_t, int64_t, int16_t>(), ""); |
| 106 static_assert(TypeCheckClamp< uint8_t, uint8_t, uint64_t, uint8_t>(), ""); |
| 107 static_assert(TypeCheckClamp< uint8_t, int64_t, int8_t, int16_t>(), ""); |
| 108 static_assert(TypeCheckClamp< uint8_t, int64_t, uint8_t, uint8_t>(), ""); |
| 109 static_assert(TypeCheckClamp< uint8_t, int64_t, int64_t, int64_t>(), ""); |
| 110 static_assert(TypeCheckClamp< uint8_t, int64_t, uint64_t, uint64_t>(), ""); |
| 111 static_assert(TypeCheckClamp< uint8_t, uint64_t, int8_t, int16_t>(), ""); |
| 112 static_assert(TypeCheckClamp< uint8_t, uint64_t, uint8_t, uint8_t>(), ""); |
| 113 static_assert(TypeCheckClamp< uint8_t, uint64_t, int64_t, int64_t>(), ""); |
| 114 static_assert(TypeCheckClamp< uint8_t, uint64_t, uint64_t, uint64_t>(), ""); |
| 115 static_assert(TypeCheckClamp< int64_t, int8_t, int8_t, int8_t>(), ""); |
| 116 static_assert(TypeCheckClamp< int64_t, int8_t, uint8_t, int8_t>(), ""); |
| 117 static_assert(TypeCheckClamp< int64_t, int8_t, int64_t, int64_t>(), ""); |
| 118 static_assert(TypeCheckClamp< int64_t, int8_t, uint64_t, int8_t>(), ""); |
| 119 static_assert(TypeCheckClamp< int64_t, uint8_t, int8_t, int16_t>(), ""); |
| 120 static_assert(TypeCheckClamp< int64_t, uint8_t, uint8_t, uint8_t>(), ""); |
| 121 static_assert(TypeCheckClamp< int64_t, uint8_t, int64_t, int64_t>(), ""); |
| 122 static_assert(TypeCheckClamp< int64_t, uint8_t, uint64_t, uint8_t>(), ""); |
| 123 static_assert(TypeCheckClamp< int64_t, int64_t, int8_t, int64_t>(), ""); |
| 124 static_assert(TypeCheckClamp< int64_t, int64_t, uint8_t, int64_t>(), ""); |
| 125 static_assert(TypeCheckClamp< int64_t, int64_t, int64_t, int64_t>(), ""); |
| 126 static_assert(TypeCheckClamp< int64_t, int64_t, uint64_t, int64_t>(), ""); |
| 127 static_assert(TypeCheckClamp< int64_t, uint64_t, int8_t, int64_t>(), ""); |
| 128 static_assert(TypeCheckClamp< int64_t, uint64_t, uint8_t, uint64_t>(), ""); |
| 129 static_assert(TypeCheckClamp< int64_t, uint64_t, int64_t, int64_t>(), ""); |
| 130 static_assert(TypeCheckClamp< int64_t, uint64_t, uint64_t, uint64_t>(), ""); |
| 131 static_assert(TypeCheckClamp<uint64_t, int8_t, int8_t, int8_t>(), ""); |
| 132 static_assert(TypeCheckClamp<uint64_t, int8_t, uint8_t, uint8_t>(), ""); |
| 133 static_assert(TypeCheckClamp<uint64_t, int8_t, int64_t, int8_t>(), ""); |
| 134 static_assert(TypeCheckClamp<uint64_t, int8_t, uint64_t, uint8_t>(), ""); |
| 135 static_assert(TypeCheckClamp<uint64_t, uint8_t, int8_t, int16_t>(), ""); |
| 136 static_assert(TypeCheckClamp<uint64_t, uint8_t, uint8_t, uint8_t>(), ""); |
| 137 static_assert(TypeCheckClamp<uint64_t, uint8_t, int64_t, int16_t>(), ""); |
| 138 static_assert(TypeCheckClamp<uint64_t, uint8_t, uint64_t, uint8_t>(), ""); |
| 139 static_assert(TypeCheckClamp<uint64_t, int64_t, int8_t, int64_t>(), ""); |
| 140 static_assert(TypeCheckClamp<uint64_t, int64_t, uint8_t, uint64_t>(), ""); |
| 141 static_assert(TypeCheckClamp<uint64_t, int64_t, int64_t, int64_t>(), ""); |
| 142 static_assert(TypeCheckClamp<uint64_t, int64_t, uint64_t, uint64_t>(), ""); |
| 143 static_assert(TypeCheckClamp<uint64_t, uint64_t, int8_t, uint64_t>(), ""); |
| 144 static_assert(TypeCheckClamp<uint64_t, uint64_t, uint8_t, uint64_t>(), ""); |
| 145 static_assert(TypeCheckClamp<uint64_t, uint64_t, int64_t, uint64_t>(), ""); |
| 146 static_assert(TypeCheckClamp<uint64_t, uint64_t, uint64_t, uint64_t>(), ""); |
| 147 |
65 enum DefaultE { kFoo = -17 }; | 148 enum DefaultE { kFoo = -17 }; |
66 enum UInt8E : uint8_t { kBar = 17 }; | 149 enum UInt8E : uint8_t { kBar = 17 }; |
67 | 150 |
68 // SafeMin/SafeMax: Check that we can use enum types. | 151 // SafeMin/SafeMax: Check that we can use enum types. |
69 static_assert(TypeCheckMinMax<unsigned, unsigned, unsigned, unsigned>(), ""); | 152 static_assert(TypeCheckMinMax<unsigned, unsigned, unsigned, unsigned>(), ""); |
70 static_assert(TypeCheckMinMax<unsigned, DefaultE, int, unsigned>(), ""); | 153 static_assert(TypeCheckMinMax<unsigned, DefaultE, int, unsigned>(), ""); |
71 static_assert(TypeCheckMinMax<unsigned, UInt8E, uint8_t, unsigned>(), ""); | 154 static_assert(TypeCheckMinMax<unsigned, UInt8E, uint8_t, unsigned>(), ""); |
72 static_assert(TypeCheckMinMax<DefaultE, unsigned, int, unsigned>(), ""); | 155 static_assert(TypeCheckMinMax<DefaultE, unsigned, int, unsigned>(), ""); |
73 static_assert(TypeCheckMinMax<DefaultE, DefaultE, int, int>(), ""); | 156 static_assert(TypeCheckMinMax<DefaultE, DefaultE, int, int>(), ""); |
74 static_assert(TypeCheckMinMax<DefaultE, UInt8E, int, int>(), ""); | 157 static_assert(TypeCheckMinMax<DefaultE, UInt8E, int, int>(), ""); |
75 static_assert(TypeCheckMinMax< UInt8E, unsigned, uint8_t, unsigned>(), ""); | 158 static_assert(TypeCheckMinMax< UInt8E, unsigned, uint8_t, unsigned>(), ""); |
76 static_assert(TypeCheckMinMax< UInt8E, DefaultE, int, int>(), ""); | 159 static_assert(TypeCheckMinMax< UInt8E, DefaultE, int, int>(), ""); |
77 static_assert(TypeCheckMinMax< UInt8E, UInt8E, uint8_t, uint8_t>(), ""); | 160 static_assert(TypeCheckMinMax< UInt8E, UInt8E, uint8_t, uint8_t>(), ""); |
78 | 161 |
| 162 // SafeClamp: Check that we can use enum types. |
| 163 static_assert(TypeCheckClamp<unsigned, unsigned, unsigned, unsigned>(), ""); |
| 164 static_assert(TypeCheckClamp<unsigned, unsigned, DefaultE, unsigned>(), ""); |
| 165 static_assert(TypeCheckClamp<unsigned, unsigned, UInt8E, unsigned>(), ""); |
| 166 static_assert(TypeCheckClamp<unsigned, DefaultE, unsigned, unsigned>(), ""); |
| 167 static_assert(TypeCheckClamp<unsigned, DefaultE, DefaultE, int>(), ""); |
| 168 static_assert(TypeCheckClamp<unsigned, DefaultE, UInt8E, unsigned>(), ""); |
| 169 static_assert(TypeCheckClamp<unsigned, UInt8E, unsigned, uint8_t>(), ""); |
| 170 static_assert(TypeCheckClamp<unsigned, UInt8E, DefaultE, int16_t>(), ""); |
| 171 static_assert(TypeCheckClamp<unsigned, UInt8E, UInt8E, uint8_t>(), ""); |
| 172 static_assert(TypeCheckClamp<DefaultE, unsigned, unsigned, unsigned>(), ""); |
| 173 static_assert(TypeCheckClamp<DefaultE, unsigned, DefaultE, int>(), ""); |
| 174 static_assert(TypeCheckClamp<DefaultE, unsigned, UInt8E, unsigned>(), ""); |
| 175 static_assert(TypeCheckClamp<DefaultE, DefaultE, unsigned, int>(), ""); |
| 176 static_assert(TypeCheckClamp<DefaultE, DefaultE, DefaultE, int>(), ""); |
| 177 static_assert(TypeCheckClamp<DefaultE, DefaultE, UInt8E, int>(), ""); |
| 178 static_assert(TypeCheckClamp<DefaultE, UInt8E, unsigned, uint8_t>(), ""); |
| 179 static_assert(TypeCheckClamp<DefaultE, UInt8E, DefaultE, int>(), ""); |
| 180 static_assert(TypeCheckClamp<DefaultE, UInt8E, UInt8E, uint8_t>(), ""); |
| 181 static_assert(TypeCheckClamp< UInt8E, unsigned, unsigned, unsigned>(), ""); |
| 182 static_assert(TypeCheckClamp< UInt8E, unsigned, DefaultE, int>(), ""); |
| 183 static_assert(TypeCheckClamp< UInt8E, unsigned, UInt8E, uint8_t>(), ""); |
| 184 static_assert(TypeCheckClamp< UInt8E, DefaultE, unsigned, unsigned>(), ""); |
| 185 static_assert(TypeCheckClamp< UInt8E, DefaultE, DefaultE, int>(), ""); |
| 186 static_assert(TypeCheckClamp< UInt8E, DefaultE, UInt8E, uint8_t>(), ""); |
| 187 static_assert(TypeCheckClamp< UInt8E, UInt8E, unsigned, uint8_t>(), ""); |
| 188 static_assert(TypeCheckClamp< UInt8E, UInt8E, DefaultE, int16_t>(), ""); |
| 189 static_assert(TypeCheckClamp< UInt8E, UInt8E, UInt8E, uint8_t>(), ""); |
| 190 |
79 using ld = long double; | 191 using ld = long double; |
80 | 192 |
81 // SafeMin/SafeMax: Check that all floating-point combinations give the | 193 // SafeMin/SafeMax: Check that all floating-point combinations give the |
82 // correct result type. | 194 // correct result type. |
83 static_assert(TypeCheckMinMax< float, float, float, float>(), ""); | 195 static_assert(TypeCheckMinMax< float, float, float, float>(), ""); |
84 static_assert(TypeCheckMinMax< float, double, double, double>(), ""); | 196 static_assert(TypeCheckMinMax< float, double, double, double>(), ""); |
85 static_assert(TypeCheckMinMax< float, ld, ld, ld>(), ""); | 197 static_assert(TypeCheckMinMax< float, ld, ld, ld>(), ""); |
86 static_assert(TypeCheckMinMax<double, float, double, double>(), ""); | 198 static_assert(TypeCheckMinMax<double, float, double, double>(), ""); |
87 static_assert(TypeCheckMinMax<double, double, double, double>(), ""); | 199 static_assert(TypeCheckMinMax<double, double, double, double>(), ""); |
88 static_assert(TypeCheckMinMax<double, ld, ld, ld>(), ""); | 200 static_assert(TypeCheckMinMax<double, ld, ld, ld>(), ""); |
89 static_assert(TypeCheckMinMax< ld, float, ld, ld>(), ""); | 201 static_assert(TypeCheckMinMax< ld, float, ld, ld>(), ""); |
90 static_assert(TypeCheckMinMax< ld, double, ld, ld>(), ""); | 202 static_assert(TypeCheckMinMax< ld, double, ld, ld>(), ""); |
91 static_assert(TypeCheckMinMax< ld, ld, ld, ld>(), ""); | 203 static_assert(TypeCheckMinMax< ld, ld, ld, ld>(), ""); |
92 | 204 |
| 205 // SafeClamp: Check that all floating-point combinations give the correct |
| 206 // result type. |
| 207 static_assert(TypeCheckClamp< float, float, float, float>(), ""); |
| 208 static_assert(TypeCheckClamp< float, float, double, double>(), ""); |
| 209 static_assert(TypeCheckClamp< float, float, ld, ld>(), ""); |
| 210 static_assert(TypeCheckClamp< float, double, float, double>(), ""); |
| 211 static_assert(TypeCheckClamp< float, double, double, double>(), ""); |
| 212 static_assert(TypeCheckClamp< float, double, ld, ld>(), ""); |
| 213 static_assert(TypeCheckClamp< float, ld, float, ld>(), ""); |
| 214 static_assert(TypeCheckClamp< float, ld, double, ld>(), ""); |
| 215 static_assert(TypeCheckClamp< float, ld, ld, ld>(), ""); |
| 216 static_assert(TypeCheckClamp<double, float, float, double>(), ""); |
| 217 static_assert(TypeCheckClamp<double, float, double, double>(), ""); |
| 218 static_assert(TypeCheckClamp<double, float, ld, ld>(), ""); |
| 219 static_assert(TypeCheckClamp<double, double, float, double>(), ""); |
| 220 static_assert(TypeCheckClamp<double, double, double, double>(), ""); |
| 221 static_assert(TypeCheckClamp<double, double, ld, ld>(), ""); |
| 222 static_assert(TypeCheckClamp<double, ld, float, ld>(), ""); |
| 223 static_assert(TypeCheckClamp<double, ld, double, ld>(), ""); |
| 224 static_assert(TypeCheckClamp<double, ld, ld, ld>(), ""); |
| 225 static_assert(TypeCheckClamp< ld, float, float, ld>(), ""); |
| 226 static_assert(TypeCheckClamp< ld, float, double, ld>(), ""); |
| 227 static_assert(TypeCheckClamp< ld, float, ld, ld>(), ""); |
| 228 static_assert(TypeCheckClamp< ld, double, float, ld>(), ""); |
| 229 static_assert(TypeCheckClamp< ld, double, double, ld>(), ""); |
| 230 static_assert(TypeCheckClamp< ld, double, ld, ld>(), ""); |
| 231 static_assert(TypeCheckClamp< ld, ld, float, ld>(), ""); |
| 232 static_assert(TypeCheckClamp< ld, ld, double, ld>(), ""); |
| 233 static_assert(TypeCheckClamp< ld, ld, ld, ld>(), ""); |
| 234 |
93 // clang-format on | 235 // clang-format on |
94 | 236 |
95 // SafeMin/SafeMax: Check some cases of explicitly specified return type. The | 237 // SafeMin/SafeMax: Check some cases of explicitly specified return type. The |
96 // commented-out lines give compilation errors due to the requested return type | 238 // commented-out lines give compilation errors due to the requested return type |
97 // being too small or requiring an int<->float conversion. | 239 // being too small or requiring an int<->float conversion. |
98 static_assert(TypeCheckMinR<int8_t, int8_t, int16_t>(), ""); | 240 static_assert(TypeCheckMinR<int8_t, int8_t, int16_t>(), ""); |
99 // static_assert(TypeCheckMinR<int8_t, int8_t, float>(), ""); | 241 // static_assert(TypeCheckMinR<int8_t, int8_t, float>(), ""); |
100 static_assert(TypeCheckMinR<uint32_t, uint64_t, uint32_t>(), ""); | 242 static_assert(TypeCheckMinR<uint32_t, uint64_t, uint32_t>(), ""); |
101 // static_assert(TypeCheckMaxR<uint64_t, float, float>(), ""); | 243 // static_assert(TypeCheckMaxR<uint64_t, float, float>(), ""); |
102 // static_assert(TypeCheckMaxR<uint64_t, double, float>(), ""); | 244 // static_assert(TypeCheckMaxR<uint64_t, double, float>(), ""); |
103 static_assert(TypeCheckMaxR<uint32_t, int32_t, uint32_t>(), ""); | 245 static_assert(TypeCheckMaxR<uint32_t, int32_t, uint32_t>(), ""); |
104 // static_assert(TypeCheckMaxR<uint32_t, int32_t, int32_t>(), ""); | 246 // static_assert(TypeCheckMaxR<uint32_t, int32_t, int32_t>(), ""); |
105 | 247 |
| 248 // SafeClamp: Check some cases of explicitly specified return type. The |
| 249 // commented-out lines give compilation errors due to the requested return type |
| 250 // being too small. |
| 251 static_assert(TypeCheckClampR<int8_t, uint8_t, int16_t, int16_t>(), ""); |
| 252 static_assert(TypeCheckClampR<int8_t, uint8_t, int16_t, int32_t>(), ""); |
| 253 // static_assert(TypeCheckClampR<int8_t, uint8_t, int16_t, uint32_t>(), ""); |
| 254 |
106 template <typename T1, typename T2, typename Tmin, typename Tmax> | 255 template <typename T1, typename T2, typename Tmin, typename Tmax> |
107 constexpr bool CheckMinMax(T1 a, T2 b, Tmin min, Tmax max) { | 256 constexpr bool CheckMinMax(T1 a, T2 b, Tmin min, Tmax max) { |
108 return TypeCheckMinMax<T1, T2, Tmin, Tmax>() && SafeMin(a, b) == min && | 257 return TypeCheckMinMax<T1, T2, Tmin, Tmax>() && SafeMin(a, b) == min && |
109 SafeMax(a, b) == max; | 258 SafeMax(a, b) == max; |
110 } | 259 } |
111 | 260 |
| 261 template <typename L, typename H, typename T, typename R> |
| 262 bool CheckClamp(L min, H max, T x, R clamped) { |
| 263 return TypeCheckClamp<L, H, T, R>() && SafeClamp(min, max, x) == clamped; |
| 264 } |
| 265 |
112 // SafeMin/SafeMax: Check a few values. | 266 // SafeMin/SafeMax: Check a few values. |
113 static_assert(CheckMinMax(int8_t{1}, int8_t{-1}, int8_t{-1}, int8_t{1}), ""); | 267 static_assert(CheckMinMax(int8_t{1}, int8_t{-1}, int8_t{-1}, int8_t{1}), ""); |
114 static_assert(CheckMinMax(uint8_t{1}, int8_t{-1}, int8_t{-1}, uint8_t{1}), ""); | 268 static_assert(CheckMinMax(uint8_t{1}, int8_t{-1}, int8_t{-1}, uint8_t{1}), ""); |
115 static_assert(CheckMinMax(uint8_t{5}, uint64_t{2}, uint8_t{2}, uint64_t{5}), | 269 static_assert(CheckMinMax(uint8_t{5}, uint64_t{2}, uint8_t{2}, uint64_t{5}), |
116 ""); | 270 ""); |
117 static_assert(CheckMinMax(std::numeric_limits<int32_t>::min(), | 271 static_assert(CheckMinMax(std::numeric_limits<int32_t>::min(), |
118 std::numeric_limits<uint32_t>::max(), | 272 std::numeric_limits<uint32_t>::max(), |
119 std::numeric_limits<int32_t>::min(), | 273 std::numeric_limits<int32_t>::min(), |
120 std::numeric_limits<uint32_t>::max()), | 274 std::numeric_limits<uint32_t>::max()), |
121 ""); | 275 ""); |
122 static_assert(CheckMinMax(std::numeric_limits<int32_t>::min(), | 276 static_assert(CheckMinMax(std::numeric_limits<int32_t>::min(), |
123 std::numeric_limits<uint16_t>::max(), | 277 std::numeric_limits<uint16_t>::max(), |
124 std::numeric_limits<int32_t>::min(), | 278 std::numeric_limits<int32_t>::min(), |
125 int32_t{std::numeric_limits<uint16_t>::max()}), | 279 int32_t{std::numeric_limits<uint16_t>::max()}), |
126 ""); | 280 ""); |
127 // static_assert(CheckMinMax(1.f, 2, 1.f, 2.f), ""); | 281 // static_assert(CheckMinMax(1.f, 2, 1.f, 2.f), ""); |
128 static_assert(CheckMinMax(1.f, 0.0, 0.0, 1.0), ""); | 282 static_assert(CheckMinMax(1.f, 0.0, 0.0, 1.0), ""); |
129 | 283 |
| 284 // SafeClamp: Check a few values. |
| 285 TEST(SafeMinmaxTest, Clamp) { |
| 286 EXPECT_TRUE(CheckClamp(std::numeric_limits<int16_t>::min(), |
| 287 std::numeric_limits<int16_t>::max(), int32_t{-1000000}, |
| 288 std::numeric_limits<int16_t>::min())); |
| 289 EXPECT_TRUE(CheckClamp(std::numeric_limits<int16_t>::min(), |
| 290 std::numeric_limits<int16_t>::max(), uint32_t{1000000}, |
| 291 std::numeric_limits<int16_t>::max())); |
| 292 EXPECT_TRUE(CheckClamp(-1.0, 1.f, 3.f, 1.0)); |
| 293 EXPECT_TRUE(CheckClamp(-1.f, 1.f, 3.0, 1.0)); |
| 294 } |
| 295 |
130 } // namespace | 296 } // namespace |
131 | 297 |
132 // clang-format off | 298 // clang-format off |
133 | 299 |
134 // These functions aren't used in the tests, but it's useful to look at the | 300 // These functions aren't used in the tests, but it's useful to look at the |
135 // compiler output for them, and verify that (1) the same-signedness *Safe | 301 // compiler output for them, and verify that (1) the same-signedness *Safe |
136 // functions result in exactly the same code as their *Ref counterparts, and | 302 // functions result in exactly the same code as their *Ref counterparts, and |
137 // that (2) the mixed-signedness *Safe functions have just a few extra | 303 // that (2) the mixed-signedness *Safe functions have just a few extra |
138 // arithmetic and logic instructions (but no extra control flow instructions). | 304 // arithmetic and logic instructions (but no extra control flow instructions). |
139 int32_t TestMinRef( int32_t a, int32_t b) { return std::min(a, b); } | 305 int32_t TestMinRef( int32_t a, int32_t b) { return std::min(a, b); } |
140 uint32_t TestMinRef( uint32_t a, uint32_t b) { return std::min(a, b); } | 306 uint32_t TestMinRef( uint32_t a, uint32_t b) { return std::min(a, b); } |
141 int32_t TestMinSafe( int32_t a, int32_t b) { return SafeMin(a, b); } | 307 int32_t TestMinSafe( int32_t a, int32_t b) { return SafeMin(a, b); } |
142 int32_t TestMinSafe( int32_t a, uint32_t b) { return SafeMin(a, b); } | 308 int32_t TestMinSafe( int32_t a, uint32_t b) { return SafeMin(a, b); } |
143 int32_t TestMinSafe(uint32_t a, int32_t b) { return SafeMin(a, b); } | 309 int32_t TestMinSafe(uint32_t a, int32_t b) { return SafeMin(a, b); } |
144 uint32_t TestMinSafe(uint32_t a, uint32_t b) { return SafeMin(a, b); } | 310 uint32_t TestMinSafe(uint32_t a, uint32_t b) { return SafeMin(a, b); } |
145 | 311 |
146 // clang-format on | 312 // clang-format on |
147 | 313 |
| 314 int32_t TestClampRef(int32_t a, int32_t b, int32_t x) { |
| 315 return std::max(a, std::min(x, b)); |
| 316 } |
| 317 uint32_t TestClampRef(uint32_t a, uint32_t b, uint32_t x) { |
| 318 return std::max(a, std::min(x, b)); |
| 319 } |
| 320 int32_t TestClampSafe(int32_t a, int32_t b, int32_t x) { |
| 321 return SafeClamp(a, b, x); |
| 322 } |
| 323 int32_t TestClampSafe(int32_t a, int32_t b, uint32_t x) { |
| 324 return SafeClamp(a, b, x); |
| 325 } |
| 326 int32_t TestClampSafe(int32_t a, uint32_t b, int32_t x) { |
| 327 return SafeClamp(a, b, x); |
| 328 } |
| 329 uint32_t TestClampSafe(int32_t a, uint32_t b, uint32_t x) { |
| 330 return SafeClamp(a, b, x); |
| 331 } |
| 332 int32_t TestClampSafe(uint32_t a, int32_t b, int32_t x) { |
| 333 return SafeClamp(a, b, x); |
| 334 } |
| 335 uint32_t TestClampSafe(uint32_t a, int32_t b, uint32_t x) { |
| 336 return SafeClamp(a, b, x); |
| 337 } |
| 338 int32_t TestClampSafe(uint32_t a, uint32_t b, int32_t x) { |
| 339 return SafeClamp(a, b, x); |
| 340 } |
| 341 uint32_t TestClampSafe(uint32_t a, uint32_t b, uint32_t x) { |
| 342 return SafeClamp(a, b, x); |
| 343 } |
| 344 |
148 } // namespace rtc | 345 } // namespace rtc |
OLD | NEW |