blob: 125177276b42b50a987bfcbddff518a8be752233 [file] [log] [blame]
// Copyright 2018 The Fuchsia Authors
//
// Use of this source code is governed by a MIT-style
// license that can be found in the LICENSE file or at
// https://opensource.org/licenses/MIT
#pragma once
// TODO(https://fxbug.dev/42106615): The libc++ <limits> defines specializations for
// floating-point types, which in GCC is incompatible with the command-line
// switches used for the kernel. So this header fakes out the other libc++
// headers with a std::numeric_limits that is close enough.
#include <limits.h>
namespace std {
template <class T>
class numeric_limits {};
#define SPECIALIZE_INT(type, sign, modulo, digits_, min_, max_) \
template <> \
class numeric_limits<type> { \
public: \
static constexpr bool is_specialized = true; \
static constexpr bool is_signed = (sign); \
static constexpr bool is_integer = true; \
static constexpr bool is_exact = true; \
static constexpr bool has_infinity = false; \
static constexpr bool has_quiet_NaN = false; \
static constexpr bool has_signaling_NaN = false; \
static constexpr bool has_denorm = false; \
static constexpr bool has_denorm_loss = false; \
static constexpr bool round_style = false; \
static constexpr bool is_iec559 = false; \
static constexpr bool is_bounded = true; \
static constexpr bool is_modulo = (modulo); \
static constexpr int digits = (digits_); \
static constexpr int digits10 = (digits_) * 3 / 10; \
static constexpr int max_digits10 = 0; \
static constexpr int radix = 2; \
static constexpr int min_exponent = 0; \
static constexpr int min_exponent10 = 0; \
static constexpr int max_exponent = 0; \
static constexpr int max_exponent10 = 0; \
static constexpr bool traps = false; \
static constexpr bool tinyness_before = false; \
static constexpr type min() { \
return (min_); \
} \
static constexpr type lowest() { \
return (min_); \
} \
static constexpr type max() { \
return (max_); \
} \
static constexpr type epsilon() { \
return 0; \
} \
static constexpr type round_error() { \
return 0; \
} \
static constexpr type infinity() { \
return 0; \
} \
static constexpr type quiet_NaN() { \
return 0; \
} \
static constexpr type signaling_NaN() { \
return 0; \
} \
static constexpr type denorm_min() { \
return 0; \
} \
}
#define SPECIALIZE_SIGNED(type, min, max) \
SPECIALIZE_INT(type, true, false, CHAR_BIT * sizeof(type) - 1, min, max)
#define SPECIALIZE_UNSIGNED(type, min, max) \
SPECIALIZE_INT(type, false, true, CHAR_BIT * sizeof(type), min, max)
SPECIALIZE_INT(bool, false, true, 1, false, true);
#if CHAR_MIN == 0
SPECIALIZE_SIGNED(char, CHAR_MIN, CHAR_MAX);
#else
SPECIALIZE_UNSIGNED(char, CHAR_MIN, CHAR_MAX);
#endif
SPECIALIZE_SIGNED(signed char, SCHAR_MIN, SCHAR_MAX);
SPECIALIZE_UNSIGNED(unsigned char, 0, UCHAR_MAX);
SPECIALIZE_SIGNED(short, SHRT_MIN, SHRT_MAX);
SPECIALIZE_UNSIGNED(unsigned short, 0, USHRT_MAX);
SPECIALIZE_SIGNED(int, INT_MIN, INT_MAX);
SPECIALIZE_UNSIGNED(unsigned int, 0, UINT_MAX);
SPECIALIZE_SIGNED(long, LONG_MIN, LONG_MAX);
SPECIALIZE_UNSIGNED(unsigned long, 0, ULONG_MAX);
SPECIALIZE_SIGNED(long long, (-__LONG_LONG_MAX__ - 1LL), __LONG_LONG_MAX__);
SPECIALIZE_UNSIGNED(unsigned long long, 0, (__LONG_LONG_MAX__ * 2ULL + 1ULL));
#ifdef __SIZEOF_INT128__
SPECIALIZE_SIGNED(__int128_t, (__int128_t{1} << 127), (~__uint128_t{0} ^ (__int128_t{1} << 127)));
SPECIALIZE_UNSIGNED(__uint128_t, 0, (~__uint128_t{0}));
#endif
#undef SPECIALIZE_SIGNED
#undef SPECIALIZE_UNSIGNED
#undef SPECIALIZE_INT
} // namespace std