1 //===----------------------------------------------------------------------===//
3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4 // See https://llvm.org/LICENSE.txt for license information.
5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
7 //===----------------------------------------------------------------------===//
9 #ifndef _LIBCPP___RANDOM_CLAMP_TO_INTEGRAL_H
10 #define _LIBCPP___RANDOM_CLAMP_TO_INTEGRAL_H
12 #include <__cxx03/__config>
13 #include <__cxx03/cmath>
14 #include <__cxx03/limits>
16 #if !defined(_LIBCPP_HAS_NO_PRAGMA_SYSTEM_HEADER)
17 # pragma GCC system_header
21 #include <__cxx03/__undef_macros>
23 _LIBCPP_BEGIN_NAMESPACE_STD
25 template <class _IntT
,
27 bool _FloatBigger
= (numeric_limits
<_FloatT
>::digits
> numeric_limits
<_IntT
>::digits
),
28 int _Bits
= (numeric_limits
<_IntT
>::digits
- numeric_limits
<_FloatT
>::digits
)>
29 _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR _IntT
__max_representable_int_for_float() _NOEXCEPT
{
30 static_assert(is_floating_point
<_FloatT
>::value
, "must be a floating point type");
31 static_assert(is_integral
<_IntT
>::value
, "must be an integral type");
32 static_assert(numeric_limits
<_FloatT
>::radix
== 2, "FloatT has incorrect radix");
34 (_IsSame
<_FloatT
, float>::value
|| _IsSame
<_FloatT
, double>::value
|| _IsSame
<_FloatT
, long double>::value
),
35 "unsupported floating point type");
36 return _FloatBigger
? numeric_limits
<_IntT
>::max() : (numeric_limits
<_IntT
>::max() >> _Bits
<< _Bits
);
39 // Convert a floating point number to the specified integral type after
40 // clamping to the integral type's representable range.
42 // The behavior is undefined if `__r` is NaN.
43 template <class _IntT
, class _RealT
>
44 _LIBCPP_HIDE_FROM_ABI _IntT
__clamp_to_integral(_RealT __r
) _NOEXCEPT
{
45 using _Lim
= numeric_limits
<_IntT
>;
46 const _IntT __max_val
= __max_representable_int_for_float
<_IntT
, _RealT
>();
47 if (__r
>= ::nextafter(static_cast<_RealT
>(__max_val
), INFINITY
)) {
49 } else if (__r
<= _Lim::lowest()) {
52 return static_cast<_IntT
>(__r
);
55 _LIBCPP_END_NAMESPACE_STD
59 #endif // _LIBCPP___RANDOM_CLAMP_TO_INTEGRAL_H