//===----------------------------------------------------------------------===// // // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. // See https://llvm.org/LICENSE.txt for license information. // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception // //===----------------------------------------------------------------------===// // __clamp_to_integral(RealT) // Test the conversion function that truncates floating point types to the // closest representable value for the specified integer type, or // numeric_limits::max()/min() if the value isn't representable. #include <__cxx03/__random/clamp_to_integral.h> #include #include #include template void test() { typedef std::numeric_limits Lim; const bool MaxIsRepresentable = sizeof(IntT) < 8; const bool IsSigned = std::is_signed::value; struct TestCase { double Input; IntT Expect; bool IsRepresentable; } TestCases[] = { {0, 0, true}, {1, 1, true}, {IsSigned ? static_cast(-1) : 0, IsSigned ? static_cast(-1) : 0, true}, {Lim::lowest(), Lim::lowest(), true}, {static_cast(Lim::max()), Lim::max(), MaxIsRepresentable}, {static_cast(Lim::max()) + 1, Lim::max(), false}, {static_cast(Lim::max()) + 1024, Lim::max(), false}, {nextafter(static_cast(Lim::max()), INFINITY), Lim::max(), false}, }; for (TestCase TC : TestCases) { auto res = std::__clamp_to_integral(TC.Input); assert(res == TC.Expect); if (TC.IsRepresentable) { auto other = static_cast(std::trunc(TC.Input)); assert(res == other); } else assert(res == Lim::min() || res == Lim::max()); } } template void test_float() { typedef std::numeric_limits Lim; const bool MaxIsRepresentable = sizeof(IntT) < 4; ((void)MaxIsRepresentable); const bool IsSigned = std::is_signed::value; struct TestCase { float Input; IntT Expect; bool IsRepresentable; } TestCases[] = { {0, 0, true}, {1, 1, true}, {IsSigned ? static_cast(-1) : 0, IsSigned ? static_cast(-1) : 0, true}, {Lim::lowest(), Lim::lowest(), true}, {static_cast(Lim::max()), Lim::max(), MaxIsRepresentable }, {nextafter(static_cast(Lim::max()), INFINITY), Lim::max(), false}, }; for (TestCase TC : TestCases) { auto res = std::__clamp_to_integral(TC.Input); assert(res == TC.Expect); if (TC.IsRepresentable) { auto other = static_cast(std::trunc(TC.Input)); assert(res == other); } else assert(res == Lim::min() || res == Lim::max()); } } int main(int, char**) { test(); test(); test(); test(); test(); test(); test_float(); test_float(); test_float(); return 0; }