2018-06-09 01:24:15 +08:00
|
|
|
#include "fp-testing.h"
|
2018-06-09 06:49:06 +08:00
|
|
|
#include "testing.h"
|
2020-02-25 23:11:52 +08:00
|
|
|
#include "flang/Evaluate/type.h"
|
2020-02-28 23:11:03 +08:00
|
|
|
#include "llvm/Support/raw_ostream.h"
|
2018-11-09 01:32:28 +08:00
|
|
|
#include <cmath>
|
2018-06-06 07:29:26 +08:00
|
|
|
#include <cstdio>
|
2018-06-13 06:45:52 +08:00
|
|
|
#include <cstdlib>
|
2018-11-14 03:29:54 +08:00
|
|
|
#include <type_traits>
|
2018-06-06 07:29:26 +08:00
|
|
|
|
2018-06-06 07:55:31 +08:00
|
|
|
using namespace Fortran::evaluate;
|
2018-08-02 00:45:59 +08:00
|
|
|
using namespace Fortran::common;
|
2018-06-13 06:14:18 +08:00
|
|
|
|
2018-08-14 08:05:15 +08:00
|
|
|
using Real2 = Scalar<Type<TypeCategory::Real, 2>>;
|
2018-12-01 06:03:05 +08:00
|
|
|
using Real3 = Scalar<Type<TypeCategory::Real, 3>>;
|
2018-08-14 08:05:15 +08:00
|
|
|
using Real4 = Scalar<Type<TypeCategory::Real, 4>>;
|
|
|
|
using Real8 = Scalar<Type<TypeCategory::Real, 8>>;
|
|
|
|
using Real10 = Scalar<Type<TypeCategory::Real, 10>>;
|
|
|
|
using Real16 = Scalar<Type<TypeCategory::Real, 16>>;
|
|
|
|
using Integer4 = Scalar<Type<TypeCategory::Integer, 4>>;
|
|
|
|
using Integer8 = Scalar<Type<TypeCategory::Integer, 8>>;
|
2018-06-06 07:29:26 +08:00
|
|
|
|
2018-06-21 06:10:34 +08:00
|
|
|
void dumpTest() {
|
|
|
|
struct {
|
|
|
|
std::uint64_t raw;
|
|
|
|
const char *expected;
|
2018-08-07 00:43:43 +08:00
|
|
|
} table[] = {
|
|
|
|
{0x7f876543, "NaN 0x7f876543"},
|
|
|
|
{0x7f800000, "Inf"},
|
|
|
|
{0xff800000, "-Inf"},
|
|
|
|
{0x00000000, "0.0"},
|
|
|
|
{0x80000000, "-0.0"},
|
|
|
|
{0x3f800000, "0x1.0p0"},
|
|
|
|
{0xbf800000, "-0x1.0p0"},
|
|
|
|
{0x40000000, "0x1.0p1"},
|
|
|
|
{0x3f000000, "0x1.0p-1"},
|
|
|
|
{0x7f7fffff, "0x1.fffffep127"},
|
|
|
|
{0x00800000, "0x1.0p-126"},
|
|
|
|
{0x00400000, "0x0.8p-127"},
|
|
|
|
{0x00000001, "0x0.000002p-127"},
|
|
|
|
{0, nullptr},
|
|
|
|
};
|
2018-06-21 06:10:34 +08:00
|
|
|
for (int j{0}; table[j].expected != nullptr; ++j) {
|
2018-08-02 00:45:59 +08:00
|
|
|
TEST(Real4{Integer4{table[j].raw}}.DumpHexadecimal() == table[j].expected)
|
|
|
|
("%d", j);
|
2018-06-21 06:10:34 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-03-29 12:00:16 +08:00
|
|
|
template <typename R> void basicTests(int rm, Rounding rounding) {
|
2018-11-29 08:20:16 +08:00
|
|
|
static constexpr int kind{R::bits / 8};
|
2018-06-06 07:29:26 +08:00
|
|
|
char desc[64];
|
|
|
|
using Word = typename R::Word;
|
2018-11-29 08:20:16 +08:00
|
|
|
std::snprintf(desc, sizeof desc, "bits=%d, le=%d, kind=%d", R::bits,
|
|
|
|
Word::littleEndian, kind);
|
2018-06-06 07:29:26 +08:00
|
|
|
R zero;
|
|
|
|
TEST(!zero.IsNegative())(desc);
|
|
|
|
TEST(!zero.IsNotANumber())(desc);
|
|
|
|
TEST(!zero.IsInfinite())(desc);
|
|
|
|
TEST(zero.IsZero())(desc);
|
|
|
|
MATCH(0, zero.Exponent())(desc);
|
|
|
|
TEST(zero.RawBits().IsZero())(desc);
|
|
|
|
MATCH(0, zero.RawBits().ToUInt64())(desc);
|
|
|
|
TEST(zero.ABS().RawBits().IsZero())(desc);
|
|
|
|
TEST(zero.Negate().RawBits().IEOR(Word::MASKL(1)).IsZero())(desc);
|
|
|
|
TEST(zero.Compare(zero) == Relation::Equal)(desc);
|
|
|
|
R minusZero{Word{std::uint64_t{1}}.SHIFTL(R::bits - 1)};
|
|
|
|
TEST(minusZero.IsNegative())(desc);
|
|
|
|
TEST(!minusZero.IsNotANumber())(desc);
|
|
|
|
TEST(!minusZero.IsInfinite())(desc);
|
|
|
|
TEST(minusZero.IsZero())(desc);
|
|
|
|
TEST(minusZero.ABS().RawBits().IsZero())(desc);
|
|
|
|
TEST(minusZero.Negate().RawBits().IsZero())(desc);
|
|
|
|
MATCH(0, minusZero.Exponent())(desc);
|
|
|
|
MATCH(0, minusZero.RawBits().LEADZ())(desc);
|
|
|
|
MATCH(1, minusZero.RawBits().POPCNT())(desc);
|
|
|
|
TEST(minusZero.Compare(minusZero) == Relation::Equal)(desc);
|
|
|
|
TEST(zero.Compare(minusZero) == Relation::Equal)(desc);
|
|
|
|
ValueWithRealFlags<R> vr;
|
|
|
|
MATCH(0, vr.value.RawBits().ToUInt64())(desc);
|
2018-06-07 02:11:35 +08:00
|
|
|
TEST(vr.flags.empty())(desc);
|
2018-06-09 06:49:06 +08:00
|
|
|
R nan{Word{std::uint64_t{1}}
|
|
|
|
.SHIFTL(R::bits)
|
|
|
|
.SubtractSigned(Word{std::uint64_t{1}})
|
|
|
|
.value};
|
2018-06-06 07:29:26 +08:00
|
|
|
MATCH(R::bits, nan.RawBits().POPCNT())(desc);
|
|
|
|
TEST(!nan.IsNegative())(desc);
|
|
|
|
TEST(nan.IsNotANumber())(desc);
|
|
|
|
TEST(!nan.IsInfinite())(desc);
|
|
|
|
TEST(!nan.IsZero())(desc);
|
|
|
|
TEST(zero.Compare(nan) == Relation::Unordered)(desc);
|
|
|
|
TEST(minusZero.Compare(nan) == Relation::Unordered)(desc);
|
|
|
|
TEST(nan.Compare(zero) == Relation::Unordered)(desc);
|
|
|
|
TEST(nan.Compare(minusZero) == Relation::Unordered)(desc);
|
|
|
|
TEST(nan.Compare(nan) == Relation::Unordered)(desc);
|
2020-02-05 08:55:45 +08:00
|
|
|
int significandBits{R::binaryPrecision - R::isImplicitMSB};
|
2018-06-06 07:29:26 +08:00
|
|
|
int exponentBits{R::bits - significandBits - 1};
|
|
|
|
std::uint64_t maxExponent{(std::uint64_t{1} << exponentBits) - 1};
|
|
|
|
MATCH(nan.Exponent(), maxExponent)(desc);
|
|
|
|
R inf{Word{maxExponent}.SHIFTL(significandBits)};
|
|
|
|
TEST(!inf.IsNegative())(desc);
|
|
|
|
TEST(!inf.IsNotANumber())(desc);
|
|
|
|
TEST(inf.IsInfinite())(desc);
|
|
|
|
TEST(!inf.IsZero())(desc);
|
2018-06-09 06:49:06 +08:00
|
|
|
TEST(inf.RawBits().CompareUnsigned(inf.ABS().RawBits()) == Ordering::Equal)
|
|
|
|
(desc);
|
2018-06-06 07:29:26 +08:00
|
|
|
TEST(zero.Compare(inf) == Relation::Less)(desc);
|
|
|
|
TEST(minusZero.Compare(inf) == Relation::Less)(desc);
|
|
|
|
TEST(nan.Compare(inf) == Relation::Unordered)(desc);
|
|
|
|
TEST(inf.Compare(inf) == Relation::Equal)(desc);
|
|
|
|
R negInf{Word{maxExponent}.SHIFTL(significandBits).IOR(Word::MASKL(1))};
|
|
|
|
TEST(negInf.IsNegative())(desc);
|
|
|
|
TEST(!negInf.IsNotANumber())(desc);
|
|
|
|
TEST(negInf.IsInfinite())(desc);
|
|
|
|
TEST(!negInf.IsZero())(desc);
|
2018-06-09 06:49:06 +08:00
|
|
|
TEST(inf.RawBits().CompareUnsigned(negInf.ABS().RawBits()) == Ordering::Equal)
|
|
|
|
(desc);
|
|
|
|
TEST(inf.RawBits().CompareUnsigned(negInf.Negate().RawBits()) ==
|
|
|
|
Ordering::Equal)
|
|
|
|
(desc);
|
|
|
|
TEST(inf.Negate().RawBits().CompareUnsigned(negInf.RawBits()) ==
|
|
|
|
Ordering::Equal)
|
|
|
|
(desc);
|
2018-06-06 07:29:26 +08:00
|
|
|
TEST(zero.Compare(negInf) == Relation::Greater)(desc);
|
|
|
|
TEST(minusZero.Compare(negInf) == Relation::Greater)(desc);
|
|
|
|
TEST(nan.Compare(negInf) == Relation::Unordered)(desc);
|
|
|
|
TEST(inf.Compare(negInf) == Relation::Greater)(desc);
|
|
|
|
TEST(negInf.Compare(negInf) == Relation::Equal)(desc);
|
|
|
|
for (std::uint64_t j{0}; j < 63; ++j) {
|
2018-06-12 02:35:53 +08:00
|
|
|
char ldesc[128];
|
2018-06-06 07:29:26 +08:00
|
|
|
std::uint64_t x{1};
|
|
|
|
x <<= j;
|
2019-08-17 02:12:18 +08:00
|
|
|
std::snprintf(ldesc, sizeof ldesc, "%s j=%d x=0x%jx rm=%d", desc,
|
|
|
|
static_cast<int>(j), static_cast<std::intmax_t>(x), rm);
|
2018-06-13 06:14:18 +08:00
|
|
|
Integer8 ix{x};
|
2018-06-12 02:35:53 +08:00
|
|
|
TEST(!ix.IsNegative())(ldesc);
|
|
|
|
MATCH(x, ix.ToUInt64())(ldesc);
|
2018-06-14 01:34:23 +08:00
|
|
|
vr = R::FromInteger(ix, rounding);
|
2018-06-12 02:35:53 +08:00
|
|
|
TEST(!vr.value.IsNegative())(ldesc);
|
|
|
|
TEST(!vr.value.IsNotANumber())(ldesc);
|
|
|
|
TEST(!vr.value.IsZero())(ldesc);
|
2018-06-13 06:14:18 +08:00
|
|
|
auto ivf = vr.value.template ToInteger<Integer8>();
|
2018-06-06 07:29:26 +08:00
|
|
|
if (j > (maxExponent / 2)) {
|
2018-06-12 02:35:53 +08:00
|
|
|
TEST(vr.flags.test(RealFlag::Overflow))(ldesc);
|
|
|
|
TEST(vr.value.IsInfinite())(ldesc);
|
|
|
|
TEST(ivf.flags.test(RealFlag::Overflow))(ldesc);
|
|
|
|
MATCH(0x7fffffffffffffff, ivf.value.ToUInt64())(ldesc);
|
2018-06-06 07:29:26 +08:00
|
|
|
} else {
|
2018-06-12 02:35:53 +08:00
|
|
|
TEST(vr.flags.empty())(ldesc);
|
|
|
|
TEST(!vr.value.IsInfinite())(ldesc);
|
|
|
|
TEST(ivf.flags.empty())(ldesc);
|
|
|
|
MATCH(x, ivf.value.ToUInt64())(ldesc);
|
2020-03-29 12:00:16 +08:00
|
|
|
if (rounding.mode == RoundingMode::TiesToEven) { // to match stold()
|
2020-02-28 23:11:03 +08:00
|
|
|
std::string buf;
|
|
|
|
llvm::raw_string_ostream ss{buf};
|
2019-08-24 02:14:48 +08:00
|
|
|
vr.value.AsFortran(ss, kind, false /*exact*/);
|
2019-08-17 02:12:18 +08:00
|
|
|
std::string decimal{ss.str()};
|
|
|
|
const char *p{decimal.data()};
|
|
|
|
MATCH(x, static_cast<std::uint64_t>(std::stold(decimal)))
|
|
|
|
("%s %s", ldesc, p);
|
|
|
|
auto check{R::Read(p, rounding)};
|
|
|
|
auto icheck{check.value.template ToInteger<Integer8>()};
|
|
|
|
MATCH(x, icheck.value.ToUInt64())(ldesc);
|
|
|
|
TEST(vr.value.Compare(check.value) == Relation::Equal)(ldesc);
|
|
|
|
}
|
2018-06-06 07:29:26 +08:00
|
|
|
}
|
2020-01-04 03:34:16 +08:00
|
|
|
TEST(vr.value.ToWholeNumber().value.Compare(vr.value) == Relation::Equal)
|
|
|
|
(ldesc);
|
2018-06-09 01:24:15 +08:00
|
|
|
ix = ix.Negate().value;
|
2018-06-12 02:35:53 +08:00
|
|
|
TEST(ix.IsNegative())(ldesc);
|
2018-06-09 01:24:15 +08:00
|
|
|
x = -x;
|
|
|
|
std::int64_t nx = x;
|
2018-06-12 02:35:53 +08:00
|
|
|
MATCH(x, ix.ToUInt64())(ldesc);
|
|
|
|
MATCH(nx, ix.ToInt64())(ldesc);
|
2018-06-14 01:34:23 +08:00
|
|
|
vr = R::FromInteger(ix);
|
2018-06-12 02:35:53 +08:00
|
|
|
TEST(vr.value.IsNegative())(ldesc);
|
|
|
|
TEST(!vr.value.IsNotANumber())(ldesc);
|
|
|
|
TEST(!vr.value.IsZero())(ldesc);
|
2018-06-13 06:14:18 +08:00
|
|
|
ivf = vr.value.template ToInteger<Integer8>();
|
2018-06-07 02:48:00 +08:00
|
|
|
if (j > (maxExponent / 2)) {
|
2018-06-12 02:35:53 +08:00
|
|
|
TEST(vr.flags.test(RealFlag::Overflow))(ldesc);
|
|
|
|
TEST(vr.value.IsInfinite())(ldesc);
|
|
|
|
TEST(ivf.flags.test(RealFlag::Overflow))(ldesc);
|
|
|
|
MATCH(0x8000000000000000, ivf.value.ToUInt64())(ldesc);
|
2018-06-07 02:48:00 +08:00
|
|
|
} else {
|
2018-06-12 02:35:53 +08:00
|
|
|
TEST(vr.flags.empty())(ldesc);
|
|
|
|
TEST(!vr.value.IsInfinite())(ldesc);
|
|
|
|
TEST(ivf.flags.empty())(ldesc);
|
|
|
|
MATCH(x, ivf.value.ToUInt64())(ldesc);
|
|
|
|
MATCH(nx, ivf.value.ToInt64())(ldesc);
|
2018-06-09 01:24:15 +08:00
|
|
|
}
|
2020-01-04 03:34:16 +08:00
|
|
|
TEST(vr.value.ToWholeNumber().value.Compare(vr.value) == Relation::Equal)
|
|
|
|
(ldesc);
|
2018-06-09 01:24:15 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-06-16 01:02:09 +08:00
|
|
|
// Takes an integer and distributes its bits across a floating
|
|
|
|
// point value. The LSB is used to complement the result.
|
2018-06-09 01:24:15 +08:00
|
|
|
std::uint32_t MakeReal(std::uint32_t n) {
|
2018-06-20 05:17:14 +08:00
|
|
|
int shifts[] = {-1, 31, 23, 30, 22, 0, 24, 29, 25, 28, 26, 1, 16, 21, 2, -1};
|
2018-06-16 01:02:09 +08:00
|
|
|
std::uint32_t x{0};
|
|
|
|
for (int j{1}; shifts[j] >= 0; ++j) {
|
|
|
|
x |= ((n >> j) & 1) << shifts[j];
|
|
|
|
}
|
|
|
|
x ^= -(n & 1);
|
|
|
|
return x;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::uint64_t MakeReal(std::uint64_t n) {
|
2018-06-20 05:17:14 +08:00
|
|
|
int shifts[] = {
|
|
|
|
-1, 63, 52, 62, 51, 0, 53, 61, 54, 60, 55, 59, 1, 16, 50, 2, -1};
|
2018-06-16 01:02:09 +08:00
|
|
|
std::uint64_t x{0};
|
|
|
|
for (int j{1}; shifts[j] >= 0; ++j) {
|
|
|
|
x |= ((n >> j) & 1) << shifts[j];
|
|
|
|
}
|
|
|
|
x ^= -(n & 1);
|
|
|
|
return x;
|
2018-06-09 01:24:15 +08:00
|
|
|
}
|
|
|
|
|
2018-11-09 01:32:28 +08:00
|
|
|
inline bool IsNaN(std::uint32_t x) {
|
|
|
|
return (x & 0x7f800000) == 0x7f800000 && (x & 0x007fffff) != 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline bool IsNaN(std::uint64_t x) {
|
|
|
|
return (x & 0x7ff0000000000000) == 0x7ff0000000000000 &&
|
|
|
|
(x & 0x000fffffffffffff) != 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline bool IsInfinite(std::uint32_t x) {
|
|
|
|
return (x & 0x7fffffff) == 0x7f800000;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline bool IsInfinite(std::uint64_t x) {
|
|
|
|
return (x & 0x7fffffffffffffff) == 0x7ff0000000000000;
|
|
|
|
}
|
|
|
|
|
2018-11-28 08:23:07 +08:00
|
|
|
inline bool IsNegative(std::uint32_t x) { return (x & 0x80000000) != 0; }
|
|
|
|
|
|
|
|
inline bool IsNegative(std::uint64_t x) {
|
|
|
|
return (x & 0x8000000000000000) != 0;
|
|
|
|
}
|
|
|
|
|
2018-11-09 01:32:28 +08:00
|
|
|
inline std::uint32_t NormalizeNaN(std::uint32_t x) {
|
|
|
|
if (IsNaN(x)) {
|
2018-06-09 01:24:15 +08:00
|
|
|
x = 0x7fe00000;
|
|
|
|
}
|
|
|
|
return x;
|
|
|
|
}
|
|
|
|
|
2018-11-09 01:32:28 +08:00
|
|
|
inline std::uint64_t NormalizeNaN(std::uint64_t x) {
|
|
|
|
if (IsNaN(x)) {
|
2018-06-16 01:02:09 +08:00
|
|
|
x = 0x7ffc000000000000;
|
|
|
|
}
|
|
|
|
return x;
|
|
|
|
}
|
|
|
|
|
2018-11-09 01:32:28 +08:00
|
|
|
enum FlagBits {
|
|
|
|
Overflow = 1,
|
|
|
|
DivideByZero = 2,
|
|
|
|
InvalidArgument = 4,
|
|
|
|
Underflow = 8,
|
|
|
|
Inexact = 16,
|
|
|
|
};
|
|
|
|
|
2019-08-16 05:25:05 +08:00
|
|
|
#ifdef __clang__
|
|
|
|
// clang support for fenv.h is broken, so tests of flag settings
|
|
|
|
// are disabled.
|
|
|
|
inline std::uint32_t FlagsToBits(const RealFlags &) { return 0; }
|
|
|
|
#else
|
|
|
|
inline std::uint32_t FlagsToBits(const RealFlags &flags) {
|
2018-06-09 06:49:06 +08:00
|
|
|
std::uint32_t bits{0};
|
|
|
|
if (flags.test(RealFlag::Overflow)) {
|
2018-11-09 01:32:28 +08:00
|
|
|
bits |= Overflow;
|
2018-06-09 06:49:06 +08:00
|
|
|
}
|
|
|
|
if (flags.test(RealFlag::DivideByZero)) {
|
2018-11-09 01:32:28 +08:00
|
|
|
bits |= DivideByZero;
|
2018-06-09 06:49:06 +08:00
|
|
|
}
|
|
|
|
if (flags.test(RealFlag::InvalidArgument)) {
|
2018-11-09 01:32:28 +08:00
|
|
|
bits |= InvalidArgument;
|
2018-06-09 06:49:06 +08:00
|
|
|
}
|
|
|
|
if (flags.test(RealFlag::Underflow)) {
|
2018-11-09 01:32:28 +08:00
|
|
|
bits |= Underflow;
|
2018-06-09 06:49:06 +08:00
|
|
|
}
|
|
|
|
if (flags.test(RealFlag::Inexact)) {
|
2018-11-09 01:32:28 +08:00
|
|
|
bits |= Inexact;
|
2018-06-09 06:49:06 +08:00
|
|
|
}
|
|
|
|
return bits;
|
|
|
|
}
|
2020-03-29 12:00:16 +08:00
|
|
|
#endif // __clang__
|
2018-06-09 06:49:06 +08:00
|
|
|
|
2020-03-29 12:00:16 +08:00
|
|
|
template <typename UINT = std::uint32_t, typename FLT = float, typename REAL>
|
2018-06-12 02:35:53 +08:00
|
|
|
void inttest(std::int64_t x, int pass, Rounding rounding) {
|
|
|
|
union {
|
2018-06-16 01:02:09 +08:00
|
|
|
UINT ui;
|
|
|
|
FLT f;
|
2018-06-12 02:35:53 +08:00
|
|
|
} u;
|
|
|
|
ScopedHostFloatingPointEnvironment fpenv;
|
2018-06-13 06:14:18 +08:00
|
|
|
Integer8 ix{x};
|
2018-06-16 01:02:09 +08:00
|
|
|
ValueWithRealFlags<REAL> real;
|
2018-06-14 01:34:23 +08:00
|
|
|
real = real.value.FromInteger(ix, rounding);
|
2020-03-29 12:00:16 +08:00
|
|
|
#ifndef __clang__ // broken and also slow
|
2018-06-12 02:35:53 +08:00
|
|
|
fpenv.ClearFlags();
|
2018-06-14 00:22:50 +08:00
|
|
|
#endif
|
2020-03-29 12:00:16 +08:00
|
|
|
FLT fcheck = x; // TODO unsigned too
|
2018-06-12 02:35:53 +08:00
|
|
|
auto actualFlags{FlagsToBits(fpenv.CurrentFlags())};
|
|
|
|
u.f = fcheck;
|
2018-06-16 01:02:09 +08:00
|
|
|
UINT rcheck{NormalizeNaN(u.ui)};
|
|
|
|
UINT check = real.value.RawBits().ToUInt64();
|
2018-06-12 02:35:53 +08:00
|
|
|
MATCH(rcheck, check)("%d 0x%llx", pass, x);
|
|
|
|
MATCH(actualFlags, FlagsToBits(real.flags))("%d 0x%llx", pass, x);
|
|
|
|
}
|
|
|
|
|
2020-03-29 12:00:16 +08:00
|
|
|
template <typename FLT = float> FLT ToIntPower(FLT x, int power) {
|
2018-11-14 03:29:54 +08:00
|
|
|
if (power == 0) {
|
|
|
|
return x / x;
|
|
|
|
}
|
|
|
|
bool negative{power < 0};
|
|
|
|
if (negative) {
|
|
|
|
power = -power;
|
|
|
|
}
|
|
|
|
FLT result{1};
|
|
|
|
while (power > 0) {
|
|
|
|
if (power & 1) {
|
|
|
|
result *= x;
|
|
|
|
}
|
|
|
|
x *= x;
|
|
|
|
power >>= 1;
|
|
|
|
}
|
|
|
|
if (negative) {
|
|
|
|
result = 1.0 / result;
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2020-03-29 12:00:16 +08:00
|
|
|
template <typename FLT, int decimalDigits>
|
2018-11-14 03:29:54 +08:00
|
|
|
FLT TimesIntPowerOfTen(FLT x, int power) {
|
|
|
|
if (power > decimalDigits || power < -decimalDigits) {
|
|
|
|
auto maxExactPowerOfTen{
|
|
|
|
TimesIntPowerOfTen<FLT, decimalDigits>(1, decimalDigits)};
|
|
|
|
auto big{ToIntPower<FLT>(maxExactPowerOfTen, power / decimalDigits)};
|
|
|
|
auto small{
|
|
|
|
TimesIntPowerOfTen<FLT, decimalDigits>(1, power % decimalDigits)};
|
|
|
|
return (x * big) * small;
|
|
|
|
}
|
|
|
|
return x * ToIntPower<FLT>(10.0, power);
|
|
|
|
}
|
|
|
|
|
2020-03-29 12:00:16 +08:00
|
|
|
template <typename UINT = std::uint32_t, typename FLT = float,
|
2018-06-20 05:17:14 +08:00
|
|
|
typename REAL = Real4>
|
2018-06-16 01:02:09 +08:00
|
|
|
void subsetTests(int pass, Rounding rounding, std::uint32_t opds) {
|
2018-06-12 02:35:53 +08:00
|
|
|
for (int j{0}; j < 63; ++j) {
|
|
|
|
std::int64_t x{1};
|
|
|
|
x <<= j;
|
2018-06-20 05:17:14 +08:00
|
|
|
inttest<UINT, FLT, REAL>(x, pass, rounding);
|
|
|
|
inttest<UINT, FLT, REAL>(-x, pass, rounding);
|
2018-06-12 02:35:53 +08:00
|
|
|
}
|
2018-06-20 05:17:14 +08:00
|
|
|
inttest<UINT, FLT, REAL>(0, pass, rounding);
|
|
|
|
inttest<UINT, FLT, REAL>(
|
|
|
|
static_cast<std::int64_t>(0x8000000000000000), pass, rounding);
|
2018-06-12 02:35:53 +08:00
|
|
|
|
2018-06-09 01:24:15 +08:00
|
|
|
union {
|
2018-06-16 01:02:09 +08:00
|
|
|
UINT ui;
|
|
|
|
FLT f;
|
2018-06-09 01:24:15 +08:00
|
|
|
} u;
|
2018-06-09 06:49:06 +08:00
|
|
|
ScopedHostFloatingPointEnvironment fpenv;
|
2018-06-12 02:35:53 +08:00
|
|
|
|
2018-06-16 01:02:09 +08:00
|
|
|
for (UINT j{0}; j < opds; ++j) {
|
2018-11-09 01:32:28 +08:00
|
|
|
|
2018-06-16 01:02:09 +08:00
|
|
|
UINT rj{MakeReal(j)};
|
|
|
|
u.ui = rj;
|
|
|
|
FLT fj{u.f};
|
|
|
|
REAL x{typename REAL::Word{std::uint64_t{rj}}};
|
2018-11-09 01:32:28 +08:00
|
|
|
|
|
|
|
// unary operations
|
|
|
|
{
|
2020-01-04 03:34:16 +08:00
|
|
|
ValueWithRealFlags<REAL> aint{x.ToWholeNumber()};
|
2020-03-29 12:00:16 +08:00
|
|
|
#ifndef __clang__ // broken and also slow
|
2018-11-09 01:32:28 +08:00
|
|
|
fpenv.ClearFlags();
|
|
|
|
#endif
|
|
|
|
FLT fcheck{std::trunc(fj)};
|
|
|
|
auto actualFlags{FlagsToBits(fpenv.CurrentFlags())};
|
2020-03-29 12:00:16 +08:00
|
|
|
actualFlags &= ~Inexact; // x86 std::trunc can set Inexact; AINT ain't
|
2018-11-09 01:32:28 +08:00
|
|
|
u.f = fcheck;
|
2018-11-14 08:02:52 +08:00
|
|
|
#ifndef __clang__
|
2018-11-09 01:32:28 +08:00
|
|
|
if (IsNaN(u.ui)) {
|
2020-03-29 12:00:16 +08:00
|
|
|
actualFlags |= InvalidArgument; // x86 std::trunc(NaN) workaround
|
2018-11-09 01:32:28 +08:00
|
|
|
}
|
2018-11-14 08:02:52 +08:00
|
|
|
#endif
|
2018-11-09 01:32:28 +08:00
|
|
|
UINT rcheck{NormalizeNaN(u.ui)};
|
|
|
|
UINT check = aint.value.RawBits().ToUInt64();
|
|
|
|
MATCH(rcheck, check)
|
2019-08-17 02:12:18 +08:00
|
|
|
("%d AINT(0x%jx)", pass, static_cast<std::intmax_t>(rj));
|
2018-11-09 01:32:28 +08:00
|
|
|
MATCH(actualFlags, FlagsToBits(aint.flags))
|
2019-08-17 02:12:18 +08:00
|
|
|
("%d AINT(0x%jx)", pass, static_cast<std::intmax_t>(rj));
|
2018-11-09 01:32:28 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
{
|
|
|
|
MATCH(IsNaN(rj), x.IsNotANumber())
|
2019-08-17 02:12:18 +08:00
|
|
|
("%d IsNaN(0x%jx)", pass, static_cast<std::intmax_t>(rj));
|
2018-11-09 01:32:28 +08:00
|
|
|
MATCH(IsInfinite(rj), x.IsInfinite())
|
2019-08-17 02:12:18 +08:00
|
|
|
("%d IsInfinite(0x%jx)", pass, static_cast<std::intmax_t>(rj));
|
2018-11-09 01:32:28 +08:00
|
|
|
|
2018-11-29 08:20:16 +08:00
|
|
|
static constexpr int kind{REAL::bits / 8};
|
2020-02-28 23:11:03 +08:00
|
|
|
std::string ssBuf, cssBuf;
|
|
|
|
llvm::raw_string_ostream ss{ssBuf};
|
|
|
|
llvm::raw_string_ostream css{cssBuf};
|
2019-08-24 02:14:48 +08:00
|
|
|
x.AsFortran(ss, kind, false /*exact*/);
|
2018-11-29 01:13:01 +08:00
|
|
|
std::string s{ss.str()};
|
|
|
|
if (IsNaN(rj)) {
|
|
|
|
css << "(0._" << kind << "/0.)";
|
|
|
|
MATCH(css.str(), s)
|
2019-08-17 02:12:18 +08:00
|
|
|
("%d invalid(0x%jx)", pass, static_cast<std::intmax_t>(rj));
|
2018-11-29 01:13:01 +08:00
|
|
|
} else if (IsInfinite(rj)) {
|
|
|
|
css << '(';
|
|
|
|
if (IsNegative(rj)) {
|
|
|
|
css << '-';
|
2018-11-14 03:29:54 +08:00
|
|
|
}
|
2018-11-29 01:13:01 +08:00
|
|
|
css << "1._" << kind << "/0.)";
|
|
|
|
MATCH(css.str(), s)
|
2019-08-17 02:12:18 +08:00
|
|
|
("%d overflow(0x%jx)", pass, static_cast<std::intmax_t>(rj));
|
2018-11-29 01:13:01 +08:00
|
|
|
} else {
|
|
|
|
const char *p = s.data();
|
|
|
|
if (*p == '(') {
|
|
|
|
++p;
|
|
|
|
}
|
|
|
|
auto readBack{REAL::Read(p, rounding)};
|
|
|
|
MATCH(rj, readBack.value.RawBits().ToUInt64())
|
2019-08-17 02:12:18 +08:00
|
|
|
("%d Read(AsFortran()) 0x%jx %s %g", pass,
|
|
|
|
static_cast<std::intmax_t>(rj), s.data(), static_cast<double>(fj));
|
2018-11-29 01:13:01 +08:00
|
|
|
MATCH('_', *p)
|
2019-08-17 02:12:18 +08:00
|
|
|
("%d Read(AsFortran()) 0x%jx %s %d", pass,
|
|
|
|
static_cast<std::intmax_t>(rj), s.data(),
|
|
|
|
static_cast<int>(p - s.data()));
|
2018-11-09 01:32:28 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// dyadic operations
|
2018-06-16 01:02:09 +08:00
|
|
|
for (UINT k{0}; k < opds; ++k) {
|
|
|
|
UINT rk{MakeReal(k)};
|
|
|
|
u.ui = rk;
|
|
|
|
FLT fk{u.f};
|
|
|
|
REAL y{typename REAL::Word{std::uint64_t{rk}}};
|
2018-06-09 06:49:06 +08:00
|
|
|
{
|
2018-06-16 01:02:09 +08:00
|
|
|
ValueWithRealFlags<REAL> sum{x.Add(y, rounding)};
|
2020-03-29 12:00:16 +08:00
|
|
|
#ifndef __clang__ // broken and also slow
|
2018-06-09 06:49:06 +08:00
|
|
|
fpenv.ClearFlags();
|
2018-06-14 00:22:50 +08:00
|
|
|
#endif
|
2018-06-16 01:02:09 +08:00
|
|
|
FLT fcheck{fj + fk};
|
2018-06-09 06:49:06 +08:00
|
|
|
auto actualFlags{FlagsToBits(fpenv.CurrentFlags())};
|
2018-06-09 01:24:15 +08:00
|
|
|
u.f = fcheck;
|
2018-06-16 01:02:09 +08:00
|
|
|
UINT rcheck{NormalizeNaN(u.ui)};
|
|
|
|
UINT check = sum.value.RawBits().ToUInt64();
|
|
|
|
MATCH(rcheck, check)
|
2019-08-17 02:12:18 +08:00
|
|
|
("%d 0x%jx + 0x%jx", pass, static_cast<std::intmax_t>(rj),
|
|
|
|
static_cast<std::intmax_t>(rk));
|
2018-06-12 02:35:53 +08:00
|
|
|
MATCH(actualFlags, FlagsToBits(sum.flags))
|
2019-08-17 02:12:18 +08:00
|
|
|
("%d 0x%jx + 0x%jx", pass, static_cast<std::intmax_t>(rj),
|
|
|
|
static_cast<std::intmax_t>(rk));
|
2018-06-09 01:24:15 +08:00
|
|
|
}
|
2018-06-09 06:49:06 +08:00
|
|
|
{
|
2018-06-16 01:02:09 +08:00
|
|
|
ValueWithRealFlags<REAL> diff{x.Subtract(y, rounding)};
|
2020-03-29 12:00:16 +08:00
|
|
|
#ifndef __clang__ // broken and also slow
|
2018-06-12 02:35:53 +08:00
|
|
|
fpenv.ClearFlags();
|
2018-06-14 00:22:50 +08:00
|
|
|
#endif
|
2018-06-16 01:02:09 +08:00
|
|
|
FLT fcheck{fj - fk};
|
2018-06-12 02:35:53 +08:00
|
|
|
auto actualFlags{FlagsToBits(fpenv.CurrentFlags())};
|
2018-06-09 01:24:15 +08:00
|
|
|
u.f = fcheck;
|
2018-06-16 01:02:09 +08:00
|
|
|
UINT rcheck{NormalizeNaN(u.ui)};
|
|
|
|
UINT check = diff.value.RawBits().ToUInt64();
|
|
|
|
MATCH(rcheck, check)
|
2019-08-17 02:12:18 +08:00
|
|
|
("%d 0x%jx - 0x%jx", pass, static_cast<std::intmax_t>(rj),
|
|
|
|
static_cast<std::intmax_t>(rk));
|
2018-06-12 02:35:53 +08:00
|
|
|
MATCH(actualFlags, FlagsToBits(diff.flags))
|
2019-08-17 02:12:18 +08:00
|
|
|
("%d 0x%jx - 0x%jx", pass, static_cast<std::intmax_t>(rj),
|
|
|
|
static_cast<std::intmax_t>(rk));
|
2018-06-09 01:24:15 +08:00
|
|
|
}
|
2018-06-09 06:49:06 +08:00
|
|
|
{
|
2018-06-16 01:02:09 +08:00
|
|
|
ValueWithRealFlags<REAL> prod{x.Multiply(y, rounding)};
|
2020-03-29 12:00:16 +08:00
|
|
|
#ifndef __clang__ // broken and also slow
|
2018-06-12 02:35:53 +08:00
|
|
|
fpenv.ClearFlags();
|
2018-06-14 00:22:50 +08:00
|
|
|
#endif
|
2018-06-16 01:02:09 +08:00
|
|
|
FLT fcheck{fj * fk};
|
2018-06-12 02:35:53 +08:00
|
|
|
auto actualFlags{FlagsToBits(fpenv.CurrentFlags())};
|
2018-06-09 01:24:15 +08:00
|
|
|
u.f = fcheck;
|
2018-06-16 01:02:09 +08:00
|
|
|
UINT rcheck{NormalizeNaN(u.ui)};
|
|
|
|
UINT check = prod.value.RawBits().ToUInt64();
|
|
|
|
MATCH(rcheck, check)
|
2019-08-17 02:12:18 +08:00
|
|
|
("%d 0x%jx * 0x%jx", pass, static_cast<std::intmax_t>(rj),
|
|
|
|
static_cast<std::intmax_t>(rk));
|
2018-06-12 02:35:53 +08:00
|
|
|
MATCH(actualFlags, FlagsToBits(prod.flags))
|
2019-08-17 02:12:18 +08:00
|
|
|
("%d 0x%jx * 0x%jx", pass, static_cast<std::intmax_t>(rj),
|
|
|
|
static_cast<std::intmax_t>(rk));
|
2018-06-09 01:24:15 +08:00
|
|
|
}
|
2018-06-12 07:01:54 +08:00
|
|
|
{
|
2018-06-16 01:02:09 +08:00
|
|
|
ValueWithRealFlags<REAL> quot{x.Divide(y, rounding)};
|
2020-03-29 12:00:16 +08:00
|
|
|
#ifndef __clang__ // broken and also slow
|
2018-06-12 02:35:53 +08:00
|
|
|
fpenv.ClearFlags();
|
2018-06-14 00:22:50 +08:00
|
|
|
#endif
|
2018-06-16 01:02:09 +08:00
|
|
|
FLT fcheck{fj / fk};
|
2018-06-12 02:35:53 +08:00
|
|
|
auto actualFlags{FlagsToBits(fpenv.CurrentFlags())};
|
2018-06-09 01:24:15 +08:00
|
|
|
u.f = fcheck;
|
2018-06-16 01:02:09 +08:00
|
|
|
UINT rcheck{NormalizeNaN(u.ui)};
|
|
|
|
UINT check = quot.value.RawBits().ToUInt64();
|
|
|
|
MATCH(rcheck, check)
|
2019-08-17 02:12:18 +08:00
|
|
|
("%d 0x%jx / 0x%jx", pass, static_cast<std::intmax_t>(rj),
|
|
|
|
static_cast<std::intmax_t>(rk));
|
2018-06-12 07:01:54 +08:00
|
|
|
MATCH(actualFlags, FlagsToBits(quot.flags))
|
2019-08-17 02:12:18 +08:00
|
|
|
("%d 0x%jx / 0x%jx", pass, static_cast<std::intmax_t>(rj),
|
|
|
|
static_cast<std::intmax_t>(rk));
|
2018-06-09 01:24:15 +08:00
|
|
|
}
|
2018-06-07 02:48:00 +08:00
|
|
|
}
|
|
|
|
}
|
2018-06-06 07:29:26 +08:00
|
|
|
}
|
|
|
|
|
2018-06-13 06:45:52 +08:00
|
|
|
void roundTest(int rm, Rounding rounding, std::uint32_t opds) {
|
2018-06-13 06:14:18 +08:00
|
|
|
basicTests<Real2>(rm, rounding);
|
2018-12-01 06:03:05 +08:00
|
|
|
basicTests<Real3>(rm, rounding);
|
2018-06-13 06:14:18 +08:00
|
|
|
basicTests<Real4>(rm, rounding);
|
|
|
|
basicTests<Real8>(rm, rounding);
|
|
|
|
basicTests<Real10>(rm, rounding);
|
|
|
|
basicTests<Real16>(rm, rounding);
|
2018-06-12 02:35:53 +08:00
|
|
|
ScopedHostFloatingPointEnvironment::SetRounding(rounding);
|
2018-06-16 01:02:09 +08:00
|
|
|
subsetTests<std::uint32_t, float, Real4>(rm, rounding, opds);
|
|
|
|
subsetTests<std::uint64_t, double, Real8>(rm, rounding, opds);
|
2018-06-12 02:35:53 +08:00
|
|
|
}
|
|
|
|
|
2018-06-06 07:29:26 +08:00
|
|
|
int main() {
|
2018-06-21 06:10:34 +08:00
|
|
|
dumpTest();
|
2020-03-29 12:00:16 +08:00
|
|
|
std::uint32_t opds{512}; // for quick testing by default
|
2018-06-13 06:45:52 +08:00
|
|
|
if (const char *p{std::getenv("REAL_TEST_OPERANDS")}) {
|
2018-06-16 01:02:09 +08:00
|
|
|
// Use 8192 or 16384 for more exhaustive testing.
|
2018-06-13 06:45:52 +08:00
|
|
|
opds = std::atol(p);
|
|
|
|
}
|
2019-01-23 03:13:44 +08:00
|
|
|
roundTest(0, Rounding{RoundingMode::TiesToEven}, opds);
|
|
|
|
roundTest(1, Rounding{RoundingMode::ToZero}, opds);
|
|
|
|
roundTest(2, Rounding{RoundingMode::Up}, opds);
|
|
|
|
roundTest(3, Rounding{RoundingMode::Down}, opds);
|
2018-06-12 02:35:53 +08:00
|
|
|
// TODO: how to test Rounding::TiesAwayFromZero on x86?
|
2018-06-20 05:16:01 +08:00
|
|
|
return testing::Complete();
|
2018-06-06 07:29:26 +08:00
|
|
|
}
|