micro-test-plus 3.2.2
The µTest++ Testing Framework
Loading...
Searching...
No Matches
type-traits.h
Go to the documentation of this file.
1/*
2 * This file is part of the µOS++ project (https://micro-os-plus.github.io/).
3 * Copyright (c) 2021 Liviu Ionescu. All rights reserved.
4 *
5 * Permission to use, copy, modify, and/or distribute this software
6 * for any purpose is hereby granted, under the terms of the MIT license.
7 *
8 * If a copy of the license was not distributed with this file, it can
9 * be obtained from https://opensource.org/licenses/mit.
10 *
11 * Major parts of the code are inspired from v1.1.8 of the Boost UT project,
12 * released under the terms of the Boost Version 1.0 Software License,
13 * which can be obtained from https://www.boost.org/LICENSE_1_0.txt.
14 */
15
16#ifndef MICRO_TEST_PLUS_TYPE_TRAITS_H_
17#define MICRO_TEST_PLUS_TYPE_TRAITS_H_
18
19// ----------------------------------------------------------------------------
20
21#ifdef __cplusplus
22
23// ----------------------------------------------------------------------------
24
25#include "math.h"
26
27// ----------------------------------------------------------------------------
28
29#if defined(__GNUC__)
30#pragma GCC diagnostic push
31#if defined(__clang__)
32#pragma clang diagnostic ignored "-Wc++98-compat"
33#pragma clang diagnostic ignored "-Wc++98-compat-pedantic"
34#endif
35#endif
36
38{
39 // --------------------------------------------------------------------------
40
45 namespace type_traits
46 {
50 template <class...>
51 struct list
52 {
53 };
54
58 template <class T, class...>
59 struct identity
60 {
61 using type = T;
62 };
63
64#if defined(__DOXYGEN__)
65 // error: Detected potential recursive class relation between class
66 // micro_os_plus::micro_test_plus::type_traits::function_traits and base
67 // class micro_os_plus::micro_test_plus::type_traits::function_traits<
68 // decltype(&T::operator())>!
69 // https://github.com/doxygen/doxygen/issues/9915
70#else
71 template <class T>
72 struct function_traits : function_traits<decltype (&T::operator())>
73 {
74 };
75#endif
76
80 template <class R, class... Args_T>
81 struct function_traits<R (*) (Args_T...)>
82 {
83 using result_type = R;
84 using args = list<Args_T...>;
85 };
86
90 template <class R, class... Args_T>
91 struct function_traits<R (Args_T...)>
92 {
93 using result_type = R;
94 using args = list<Args_T...>;
95 };
96
100 template <class R, class T, class... Args_T>
101 struct function_traits<R (T::*) (Args_T...)>
102 {
103 using result_type = R;
104 using args = list<Args_T...>;
105 };
106
110 template <class R, class T, class... Args_T>
111 struct function_traits<R (T::*) (Args_T...) const>
112 {
113 using result_type = R;
114 using args = list<Args_T...>;
115 };
116
117 template <class T>
118 T&&
120 template <class... Ts, class Expr_T>
121 constexpr auto
122 is_valid (Expr_T expr) -> decltype (expr (declval<Ts...> ()), bool ())
123 {
124 return true;
125 }
126
127 template <class...>
128 constexpr auto
129 is_valid (...) -> bool
130 {
131 return false;
132 }
133
134 template <class T>
135 static constexpr auto is_container_v = is_valid<T> (
136 [] (auto t) -> decltype (t.begin (), t.end (), void ()) {});
137
138 template <class T>
139 static constexpr auto has_npos_v
140 = is_valid<T> ([] (auto t) -> decltype (void (t.npos)) {});
141
142 template <class T>
143 static constexpr auto has_value_v
144 = is_valid<T> ([] (auto t) -> decltype (void (t.value)) {});
145
146 template <class T>
147 static constexpr auto has_epsilon_v
148 = is_valid<T> ([] (auto t) -> decltype (void (t.epsilon)) {});
149
150 template <class T>
151 inline constexpr auto is_floating_point_v = false;
152 template <>
153 inline constexpr auto is_floating_point_v<float> = true;
154 template <>
155 inline constexpr auto is_floating_point_v<double> = true;
156 template <>
157 inline constexpr auto is_floating_point_v<long double> = true;
158
159#if defined(__clang__) or defined(_MSC_VER)
160 template <class From, class To>
161 static constexpr auto is_convertible_v = __is_convertible_to (From, To);
162#else
163 template <class From, class To>
164 constexpr auto
165 is_convertible (int) -> decltype (bool (To (declval<From> ())))
166 {
167 return true;
168 }
169 template <class...>
170 constexpr auto
172 {
173 return false;
174 }
175 template <class From, class To>
177#endif
178
182 template <bool>
184 {
185 };
186
190 template <>
192 {
193 using type = int;
194 };
195
196 template <bool Cond>
198
203 struct op
204 {
205 };
206
212 template <auto N>
214 {
215 using value_type = decltype (N);
216 static constexpr auto value = N;
217
218 [[nodiscard]] constexpr auto
219 operator- () const
220 {
221 return integral_constant<-N>{};
222 }
223
224 [[nodiscard]] constexpr explicit
225 operator value_type () const
226 {
227 return N;
228 }
229
230 [[nodiscard]] constexpr auto
231 get () const
232 {
233 return N;
234 }
235 };
236
243 template <class T, auto N, auto D, auto Size, auto P = 1>
245 {
246 using value_type = T;
247
248 static constexpr auto epsilon = T (1) / math::pow (T (10), Size - 1);
249 static constexpr auto value
250 = T (P) * (T (N) + (T (D) / math::pow (T (10), Size)));
251
252 [[nodiscard]] constexpr auto
254 {
255 return floating_point_constant<T, N, D, Size, -1>{};
256 }
257
258 [[nodiscard]] constexpr explicit
259 operator value_type () const
260 {
261 return value;
262 }
263
264 [[nodiscard]] constexpr auto
265 get () const
266 {
267 return value;
268 }
269 };
270
274 template <class T>
276 {
277 using value_type = T;
278
279 constexpr genuine_integral_value (const T& _value) : value_{ _value }
280 {
281 }
282
283 [[nodiscard]] constexpr explicit
284 operator T () const
285 {
286 return value_;
287 }
288
289 [[nodiscard]] constexpr decltype (auto)
290 get () const
291 {
292 return value_;
293 }
294
296 };
297
298 template <class T>
299 inline constexpr auto is_op_v = __is_base_of (type_traits::op, T);
300
305 template <class T, class = int>
307 {
308 using value_type = T;
309
310 constexpr value (const T& _value) : value_{ _value }
311 {
312 }
313
314 [[nodiscard]] constexpr explicit
315 operator T () const
316 {
317 return value_;
318 }
319
320 [[nodiscard]] constexpr decltype (auto)
321 get () const
322 {
323 return value_;
324 }
325
327 };
328
336 template <class T>
337 struct value<T,
338 type_traits::requires_t<type_traits::is_floating_point_v<T>>>
340 {
341 using value_type = T;
342 static inline auto epsilon = T{}; // Why static?
343
344 constexpr value (const T& _value, const T precision) : value_{ _value }
345 {
346 epsilon = precision;
347 }
348
349 constexpr /*explicit(false)*/ value (const T& val)
350 : value{ val,
351 T (1)
352 / math::pow (T (10),
353 math::den_size<unsigned long long> (val)) }
354 {
355 }
356
357 [[nodiscard]] constexpr explicit
358 operator T () const
359 {
360 return value_;
361 }
362
363 [[nodiscard]] constexpr decltype (auto)
364 get () const
365 {
366 return value_;
367 }
368
370 };
371
372 } // namespace type_traits
373
374 // --------------------------------------------------------------------------
375} // namespace micro_os_plus::micro_test_plus
376
377#if defined(__GNUC__)
378#pragma GCC diagnostic pop
379#endif
380
381// ----------------------------------------------------------------------------
382
383#endif // __cplusplus
384
385// ----------------------------------------------------------------------------
386
387#endif // MICRO_TEST_PLUS_TYPE_TRAITS_H_
388
389// ----------------------------------------------------------------------------
Local mathematical functions.
Definition math.h:49
constexpr auto pow(const T base, const Exp_T exp) -> T
Generic 'power of', to raise base to exponent (base ^ exp).
Definition math.h:75
Local type traits. Some may have standard equivalents, but better keep them locally.
Definition type-traits.h:46
typename requires_< Cond >::type requires_t
constexpr auto is_valid(Expr_T expr) -> decltype(expr(declval< Ts... >()), bool())
constexpr auto is_convertible(int) -> decltype(bool(To(declval< From >())))
A generic floating point constant, with custom size and precision. It has a getter and a '-' operator...
A generic integral constant. It has a getter and a '-' operator to return the negative value.
Empty base class of all operators.