2020-02-25 23:11:52 +08:00
|
|
|
#include "flang/Evaluate/integer.h"
|
2018-06-02 03:37:23 +08:00
|
|
|
#include "testing.h"
|
2018-05-31 07:08:42 +08:00
|
|
|
#include <cstdio>
|
2018-06-21 06:10:34 +08:00
|
|
|
#include <string>
|
2018-05-31 07:08:42 +08:00
|
|
|
|
2018-05-31 07:09:52 +08:00
|
|
|
using Fortran::evaluate::Ordering;
|
2018-06-13 04:33:08 +08:00
|
|
|
using Fortran::evaluate::value::Integer;
|
2018-05-31 07:08:42 +08:00
|
|
|
|
2020-03-29 12:00:16 +08:00
|
|
|
template <int BITS, typename INT = Integer<BITS>> void exhaustiveTesting() {
|
2018-05-31 07:08:42 +08:00
|
|
|
std::uint64_t maxUnsignedValue{(std::uint64_t{1} << BITS) - 1};
|
|
|
|
std::int64_t maxPositiveSignedValue{(std::int64_t{1} << (BITS - 1)) - 1};
|
2018-05-31 07:09:52 +08:00
|
|
|
std::int64_t mostNegativeSignedValue{-(std::int64_t{1} << (BITS - 1))};
|
2018-05-31 07:08:42 +08:00
|
|
|
char desc[64];
|
2018-06-02 02:48:31 +08:00
|
|
|
std::snprintf(desc, sizeof desc,
|
|
|
|
"BITS=%d, PARTBITS=%d, sizeof(Part)=%d, LE=%d", BITS, INT::partBits,
|
|
|
|
static_cast<int>(sizeof(typename INT::Part)), INT::littleEndian);
|
2018-06-01 07:32:36 +08:00
|
|
|
|
2018-06-02 01:05:15 +08:00
|
|
|
MATCH(BITS, INT::bits)(desc);
|
|
|
|
MATCH(maxPositiveSignedValue, INT::HUGE().ToUInt64())(desc);
|
|
|
|
INT zero;
|
2018-05-31 07:08:42 +08:00
|
|
|
TEST(zero.IsZero())(desc);
|
2018-06-01 07:32:36 +08:00
|
|
|
MATCH(0, zero.ToUInt64())(desc);
|
|
|
|
MATCH(0, zero.ToInt64())(desc);
|
|
|
|
|
2018-05-31 07:08:42 +08:00
|
|
|
for (std::uint64_t x{0}; x <= maxUnsignedValue; ++x) {
|
2018-06-02 03:37:23 +08:00
|
|
|
unsigned long long ullx = x;
|
2018-06-02 01:05:15 +08:00
|
|
|
INT a{x};
|
2018-06-01 01:45:41 +08:00
|
|
|
MATCH(x, a.ToUInt64())(desc);
|
2018-06-02 01:05:15 +08:00
|
|
|
INT copy{a};
|
2018-06-01 01:45:41 +08:00
|
|
|
MATCH(x, copy.ToUInt64())(desc);
|
2018-05-31 07:08:42 +08:00
|
|
|
copy = a;
|
2018-06-01 01:45:41 +08:00
|
|
|
MATCH(x, copy.ToUInt64())(desc);
|
|
|
|
MATCH(x == 0, a.IsZero())("%s, x=0x%llx", desc, x);
|
2018-06-02 03:37:23 +08:00
|
|
|
char buffer[64];
|
|
|
|
std::snprintf(buffer, sizeof buffer, " %llu", ullx);
|
|
|
|
const char *p{buffer};
|
2019-03-23 05:27:18 +08:00
|
|
|
auto readcheck{INT::Read(p)};
|
2018-06-02 03:37:23 +08:00
|
|
|
TEST(!readcheck.overflow)("%s, x=0x%llx", desc, x);
|
|
|
|
MATCH(x, readcheck.value.ToUInt64())("%s, x=0x%llx", desc, x);
|
|
|
|
TEST(!*p)("%s, x=0x%llx", desc, x);
|
|
|
|
std::snprintf(buffer, sizeof buffer, "%llx", ullx);
|
|
|
|
p = buffer;
|
2019-03-23 05:27:18 +08:00
|
|
|
readcheck = INT::Read(p, 16);
|
2018-06-02 03:37:23 +08:00
|
|
|
TEST(!readcheck.overflow)("%s, x=0x%llx", desc, x);
|
|
|
|
MATCH(x, readcheck.value.ToUInt64())("%s, x=0x%llx", desc, x);
|
|
|
|
TEST(!*p)("%s, x=0x%llx", desc, x);
|
2018-06-21 06:10:34 +08:00
|
|
|
std::string udec{a.UnsignedDecimal()};
|
|
|
|
p = udec.data();
|
2019-03-23 05:27:18 +08:00
|
|
|
readcheck = INT::Read(p);
|
2018-06-21 06:10:34 +08:00
|
|
|
TEST(!readcheck.overflow)("%s, x=0x%llx", desc, x);
|
|
|
|
MATCH(x, readcheck.value.ToUInt64())("%s, x=0x%llx", desc, x);
|
|
|
|
TEST(!*p)("%s, x=0x%llx", desc, x);
|
|
|
|
std::string hex{a.Hexadecimal()};
|
|
|
|
p = hex.data();
|
2019-03-23 05:27:18 +08:00
|
|
|
readcheck = INT::Read(p, 16);
|
2018-06-21 06:10:34 +08:00
|
|
|
TEST(!readcheck.overflow)("%s, x=0x%llx", desc, x);
|
|
|
|
MATCH(x, readcheck.value.ToUInt64())("%s, x=0x%llx", desc, x);
|
|
|
|
TEST(!*p)("%s, x=0x%llx", desc, x);
|
2018-06-02 01:05:15 +08:00
|
|
|
INT t{a.NOT()};
|
2018-06-01 01:45:41 +08:00
|
|
|
MATCH(x ^ maxUnsignedValue, t.ToUInt64())("%s, x=0x%llx", desc, x);
|
|
|
|
auto negated{a.Negate()};
|
|
|
|
MATCH(x == std::uint64_t{1} << (BITS - 1), negated.overflow)
|
2018-06-02 02:48:31 +08:00
|
|
|
("%s, x=0x%llx", desc, x);
|
2018-06-01 07:32:36 +08:00
|
|
|
MATCH(-x & maxUnsignedValue, negated.value.ToUInt64())
|
2018-06-02 02:48:31 +08:00
|
|
|
("%s, x=0x%llx", desc, x);
|
2018-06-01 07:32:36 +08:00
|
|
|
auto abs{a.ABS()};
|
|
|
|
MATCH(x == std::uint64_t{1} << (BITS - 1), abs.overflow)
|
2018-06-02 02:48:31 +08:00
|
|
|
("%s, x=0x%llx", desc, x);
|
|
|
|
MATCH(x >> (BITS - 1) ? -x & maxUnsignedValue : x, abs.value.ToUInt64())
|
|
|
|
("%s, x=0x%llx", desc, x);
|
2018-06-01 02:18:14 +08:00
|
|
|
int lzbc{a.LEADZ()};
|
2018-05-31 07:08:42 +08:00
|
|
|
COMPARE(lzbc, >=, 0)("%s, x=0x%llx", desc, x);
|
|
|
|
COMPARE(lzbc, <=, BITS)("%s, x=0x%llx", desc, x);
|
2018-06-01 01:45:41 +08:00
|
|
|
MATCH(x == 0, lzbc == BITS)("%s, x=0x%llx, lzbc=%d", desc, x, lzbc);
|
2018-05-31 07:08:42 +08:00
|
|
|
std::uint64_t lzcheck{std::uint64_t{1} << (BITS - lzbc)};
|
|
|
|
COMPARE(x, <, lzcheck)("%s, x=0x%llx, lzbc=%d", desc, x, lzbc);
|
|
|
|
COMPARE(x + x + !x, >=, lzcheck)("%s, x=0x%llx, lzbc=%d", desc, x, lzbc);
|
2018-06-01 07:15:18 +08:00
|
|
|
int popcheck{0};
|
|
|
|
for (int j{0}; j < BITS; ++j) {
|
|
|
|
popcheck += (x >> j) & 1;
|
|
|
|
}
|
|
|
|
MATCH(popcheck, a.POPCNT())("%s, x=0x%llx", desc, x);
|
|
|
|
MATCH(popcheck & 1, a.POPPAR())("%s, x=0x%llx", desc, x);
|
2018-06-01 07:32:36 +08:00
|
|
|
int trailcheck{0};
|
|
|
|
for (; trailcheck < BITS; ++trailcheck) {
|
|
|
|
if ((x >> trailcheck) & 1) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
MATCH(trailcheck, a.TRAILZ())("%s, x=0x%llx", desc, x);
|
2018-06-01 08:00:04 +08:00
|
|
|
for (int j{0}; j < BITS; ++j) {
|
|
|
|
MATCH((x >> j) & 1, a.BTEST(j))
|
2018-06-02 02:48:31 +08:00
|
|
|
("%s, x=0x%llx, bit %d", desc, x, j);
|
2018-06-01 08:00:04 +08:00
|
|
|
}
|
|
|
|
// TODO test DIM, MODULO, ISHFTC, DSHIFTL/R
|
2018-06-01 07:32:36 +08:00
|
|
|
// TODO test IBCLR, IBSET, IBITS, MAX, MIN, MERGE_BITS, RANGE, SIGN
|
|
|
|
|
2018-05-31 07:08:42 +08:00
|
|
|
Ordering ord{Ordering::Equal};
|
|
|
|
std::int64_t sx = x;
|
|
|
|
if (x + x > maxUnsignedValue) {
|
|
|
|
TEST(a.IsNegative())("%s, x=0x%llx", desc, x);
|
|
|
|
sx = x | (~std::uint64_t{0} << BITS);
|
|
|
|
TEST(sx < 0)("%s, x=0x%llx %lld", desc, x, sx);
|
|
|
|
ord = Ordering::Less;
|
|
|
|
} else {
|
|
|
|
TEST(!a.IsNegative())("%s, x=0x%llx", desc, x);
|
|
|
|
TEST(sx >= 0)("%s, x=0x%llx %lld", desc, x, sx);
|
|
|
|
if (sx > 0) {
|
|
|
|
ord = Ordering::Greater;
|
|
|
|
} else {
|
|
|
|
ord = Ordering::Equal;
|
|
|
|
}
|
|
|
|
}
|
2018-06-01 07:32:36 +08:00
|
|
|
|
2018-05-31 07:08:42 +08:00
|
|
|
TEST(sx == a.ToInt64())("%s, x=0x%llx %lld", desc, x, sx);
|
|
|
|
TEST(a.CompareToZeroSigned() == ord)("%s, x=0x%llx %lld", desc, x, sx);
|
|
|
|
for (int count{0}; count <= BITS + 1; ++count) {
|
2018-06-01 02:18:14 +08:00
|
|
|
t = a.SHIFTL(count);
|
|
|
|
MATCH((x << count) & maxUnsignedValue, t.ToUInt64())
|
2018-06-02 02:48:31 +08:00
|
|
|
("%s, x=0x%llx, count=%d", desc, x, count);
|
2018-06-01 02:18:14 +08:00
|
|
|
t = a.ISHFT(count);
|
|
|
|
MATCH((x << count) & maxUnsignedValue, t.ToUInt64())
|
2018-06-02 02:48:31 +08:00
|
|
|
("%s, x=0x%llx, count=%d", desc, x, count);
|
2018-06-01 02:18:14 +08:00
|
|
|
t = a.SHIFTR(count);
|
|
|
|
MATCH(x >> count, t.ToUInt64())
|
2018-06-02 02:48:31 +08:00
|
|
|
("%s, x=0x%llx, count=%d", desc, x, count);
|
2018-06-01 02:18:14 +08:00
|
|
|
t = a.ISHFT(-count);
|
|
|
|
MATCH(x >> count, t.ToUInt64())("%s, x=0x%llx, count=%d", desc, x, count);
|
|
|
|
t = a.SHIFTA(count);
|
2018-06-02 02:48:31 +08:00
|
|
|
std::uint64_t fill{-(x >> (BITS - 1))};
|
|
|
|
std::uint64_t sra{
|
|
|
|
count >= BITS ? fill : (x >> count) | (fill << (BITS - count))};
|
2018-06-01 02:18:14 +08:00
|
|
|
MATCH(sra, t.ToInt64())
|
2018-06-02 02:48:31 +08:00
|
|
|
("%s, x=0x%llx, count=%d", desc, x, count);
|
2018-05-31 07:08:42 +08:00
|
|
|
}
|
2018-06-01 07:32:36 +08:00
|
|
|
|
2018-05-31 07:08:42 +08:00
|
|
|
for (std::uint64_t y{0}; y <= maxUnsignedValue; ++y) {
|
|
|
|
std::int64_t sy = y;
|
|
|
|
if (y + y > maxUnsignedValue) {
|
|
|
|
sy = y | (~std::uint64_t{0} << BITS);
|
|
|
|
}
|
2018-06-02 01:05:15 +08:00
|
|
|
INT b{y};
|
2018-05-31 07:08:42 +08:00
|
|
|
if (x < y) {
|
|
|
|
ord = Ordering::Less;
|
|
|
|
} else if (x > y) {
|
|
|
|
ord = Ordering::Greater;
|
|
|
|
} else {
|
|
|
|
ord = Ordering::Equal;
|
|
|
|
}
|
|
|
|
TEST(a.CompareUnsigned(b) == ord)("%s, x=0x%llx, y=0x%llx", desc, x, y);
|
2018-06-01 07:32:36 +08:00
|
|
|
MATCH(x >= y, a.BGE(b))("%s, x=0x%llx, y=0x%llx", desc, x, y);
|
|
|
|
MATCH(x > y, a.BGT(b))("%s, x=0x%llx, y=0x%llx", desc, x, y);
|
|
|
|
MATCH(x <= y, a.BLE(b))("%s, x=0x%llx, y=0x%llx", desc, x, y);
|
|
|
|
MATCH(x < y, a.BLT(b))("%s, x=0x%llx, y=0x%llx", desc, x, y);
|
2018-05-31 07:08:42 +08:00
|
|
|
if (sx < sy) {
|
|
|
|
ord = Ordering::Less;
|
|
|
|
} else if (sx > sy) {
|
|
|
|
ord = Ordering::Greater;
|
|
|
|
} else {
|
|
|
|
ord = Ordering::Equal;
|
|
|
|
}
|
2018-05-31 07:09:52 +08:00
|
|
|
TEST(a.CompareSigned(b) == ord)
|
2018-06-02 02:48:31 +08:00
|
|
|
("%s, x=0x%llx %lld %d, y=0x%llx %lld %d", desc, x, sx, a.IsNegative(), y,
|
|
|
|
sy, b.IsNegative());
|
2018-06-01 07:32:36 +08:00
|
|
|
|
2018-06-01 02:58:50 +08:00
|
|
|
t = a.IAND(b);
|
|
|
|
MATCH(x & y, t.ToUInt64())("%s, x=0x%llx, y=0x%llx", desc, x, y);
|
|
|
|
t = a.IOR(b);
|
|
|
|
MATCH(x | y, t.ToUInt64())("%s, x=0x%llx, y=0x%llx", desc, x, y);
|
|
|
|
t = a.IEOR(b);
|
|
|
|
MATCH(x ^ y, t.ToUInt64())("%s, x=0x%llx, y=0x%llx", desc, x, y);
|
|
|
|
auto sum{a.AddUnsigned(b)};
|
2018-06-02 02:48:31 +08:00
|
|
|
COMPARE(
|
|
|
|
x + y, ==, sum.value.ToUInt64() + (std::uint64_t{sum.carry} << BITS))
|
|
|
|
("%s, x=0x%llx, y=0x%llx, carry=%d", desc, x, y, sum.carry);
|
2018-06-01 02:58:50 +08:00
|
|
|
auto ssum{a.AddSigned(b)};
|
|
|
|
MATCH((sx + sy) & maxUnsignedValue, ssum.value.ToUInt64())
|
2018-06-02 02:48:31 +08:00
|
|
|
("%s, x=0x%llx, y=0x%llx", desc, x, y);
|
|
|
|
MATCH(
|
|
|
|
sx + sy < mostNegativeSignedValue || sx + sy > maxPositiveSignedValue,
|
|
|
|
ssum.overflow)
|
|
|
|
("%s, x=0x%llx, y=0x%llx", desc, x, y);
|
2018-06-01 02:58:50 +08:00
|
|
|
auto diff{a.SubtractSigned(b)};
|
|
|
|
MATCH((sx - sy) & maxUnsignedValue, diff.value.ToUInt64())
|
2018-06-02 02:48:31 +08:00
|
|
|
("%s, x=0x%llx, y=0x%llx", desc, x, y);
|
|
|
|
MATCH(
|
|
|
|
sx - sy < mostNegativeSignedValue || sx - sy > maxPositiveSignedValue,
|
|
|
|
diff.overflow)
|
|
|
|
("%s, x=0x%llx, y=0x%llx", desc, x, y);
|
2018-06-01 02:58:50 +08:00
|
|
|
auto product{a.MultiplyUnsigned(b)};
|
2018-06-02 02:48:31 +08:00
|
|
|
MATCH(
|
|
|
|
x * y, (product.upper.ToUInt64() << BITS) ^ product.lower.ToUInt64())
|
|
|
|
("%s, x=0x%llx, y=0x%llx, lower=0x%llx, upper=0x%llx", desc, x, y,
|
2018-06-01 02:58:50 +08:00
|
|
|
product.lower.ToUInt64(), product.upper.ToUInt64());
|
|
|
|
product = a.MultiplySigned(b);
|
|
|
|
MATCH((sx * sy) & maxUnsignedValue, product.lower.ToUInt64())
|
2018-06-02 02:48:31 +08:00
|
|
|
("%s, x=0x%llx, y=0x%llx", desc, x, y);
|
2018-06-01 02:58:50 +08:00
|
|
|
MATCH(((sx * sy) >> BITS) & maxUnsignedValue, product.upper.ToUInt64())
|
2018-06-02 02:48:31 +08:00
|
|
|
("%s, x=0x%llx, y=0x%llx", desc, x, y);
|
2018-06-01 03:25:21 +08:00
|
|
|
auto quot{a.DivideUnsigned(b)};
|
|
|
|
MATCH(y == 0, quot.divisionByZero)("%s, x=0x%llx, y=0x%llx", desc, x, y);
|
2018-06-01 01:45:41 +08:00
|
|
|
if (y == 0) {
|
2018-06-01 03:25:21 +08:00
|
|
|
MATCH(maxUnsignedValue, quot.quotient.ToUInt64())
|
2018-06-02 02:48:31 +08:00
|
|
|
("%s, x=0x%llx, y=0x%llx", desc, x, y);
|
2018-06-01 03:25:21 +08:00
|
|
|
MATCH(0, quot.remainder.ToUInt64())
|
2018-06-02 02:48:31 +08:00
|
|
|
("%s, x=0x%llx, y=0x%llx", desc, x, y);
|
2018-05-31 07:08:42 +08:00
|
|
|
} else {
|
2018-06-01 03:25:21 +08:00
|
|
|
MATCH(x / y, quot.quotient.ToUInt64())
|
2018-06-02 02:48:31 +08:00
|
|
|
("%s, x=0x%llx, y=0x%llx", desc, x, y);
|
2018-06-01 03:25:21 +08:00
|
|
|
MATCH(x % y, quot.remainder.ToUInt64())
|
2018-06-02 02:48:31 +08:00
|
|
|
("%s, x=0x%llx, y=0x%llx", desc, x, y);
|
2018-05-31 07:08:42 +08:00
|
|
|
}
|
2018-06-01 03:25:21 +08:00
|
|
|
quot = a.DivideSigned(b);
|
2018-05-31 07:08:42 +08:00
|
|
|
bool badCase{sx == mostNegativeSignedValue &&
|
2018-05-31 07:09:52 +08:00
|
|
|
((sy == -1 && sx != sy) || (BITS == 1 && sx == sy))};
|
2018-06-01 03:25:21 +08:00
|
|
|
MATCH(y == 0, quot.divisionByZero)("%s, x=0x%llx, y=0x%llx", desc, x, y);
|
|
|
|
MATCH(badCase, quot.overflow)("%s, x=0x%llx, y=0x%llx", desc, x, y);
|
2018-05-31 07:08:42 +08:00
|
|
|
if (y == 0) {
|
|
|
|
if (sx >= 0) {
|
2018-06-01 03:25:21 +08:00
|
|
|
MATCH(maxPositiveSignedValue, quot.quotient.ToInt64())
|
2018-05-31 07:09:52 +08:00
|
|
|
("%s, x=0x%llx, y=0x%llx", desc, x, y);
|
2018-05-31 07:08:42 +08:00
|
|
|
} else {
|
2018-06-01 03:25:21 +08:00
|
|
|
MATCH(mostNegativeSignedValue, quot.quotient.ToInt64())
|
2018-05-31 07:09:52 +08:00
|
|
|
("%s, x=0x%llx, y=0x%llx", desc, x, y);
|
2018-05-31 07:08:42 +08:00
|
|
|
}
|
2018-06-01 03:25:21 +08:00
|
|
|
MATCH(0, quot.remainder.ToUInt64())
|
2018-06-02 02:48:31 +08:00
|
|
|
("%s, x=0x%llx, y=0x%llx", desc, x, y);
|
2018-05-31 07:08:42 +08:00
|
|
|
} else if (badCase) {
|
2018-06-02 02:48:31 +08:00
|
|
|
MATCH(x, quot.quotient.ToUInt64())
|
|
|
|
("%s, x=0x%llx, y=0x%llx", desc, x, y);
|
|
|
|
MATCH(0, quot.remainder.ToUInt64())
|
|
|
|
("%s, x=0x%llx, y=0x%llx", desc, x, y);
|
2018-05-31 07:08:42 +08:00
|
|
|
} else {
|
2018-06-01 03:25:21 +08:00
|
|
|
MATCH(sx / sy, quot.quotient.ToInt64())
|
2018-06-02 02:48:31 +08:00
|
|
|
("%s, x=0x%llx %lld, y=0x%llx %lld; unsigned 0x%llx", desc, x, sx, y,
|
2018-06-01 03:25:21 +08:00
|
|
|
sy, quot.quotient.ToUInt64());
|
|
|
|
MATCH(sx - sy * (sx / sy), quot.remainder.ToInt64())
|
2018-06-02 02:48:31 +08:00
|
|
|
("%s, x=0x%llx, y=0x%llx", desc, x, y);
|
2018-05-31 07:08:42 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
int main() {
|
|
|
|
TEST(Reverse(Ordering::Less) == Ordering::Greater);
|
|
|
|
TEST(Reverse(Ordering::Greater) == Ordering::Less);
|
|
|
|
TEST(Reverse(Ordering::Equal) == Ordering::Equal);
|
2018-06-21 06:10:34 +08:00
|
|
|
TEST(Integer<128>{123456789}.UnsignedDecimal() == "123456789");
|
|
|
|
TEST(Integer<128>{123456789}.SignedDecimal() == "123456789");
|
|
|
|
TEST(Integer<128>{-123456789}.SignedDecimal() == "-123456789");
|
2018-07-18 03:34:07 +08:00
|
|
|
std::uint64_t big{0x123456789abcdef};
|
|
|
|
TEST(Integer<128>{big}.Hexadecimal() == "123456789abcdef");
|
2018-05-31 07:08:42 +08:00
|
|
|
exhaustiveTesting<1>();
|
|
|
|
exhaustiveTesting<2>();
|
|
|
|
exhaustiveTesting<7>();
|
|
|
|
exhaustiveTesting<8>();
|
|
|
|
exhaustiveTesting<9>();
|
2018-06-06 04:55:56 +08:00
|
|
|
exhaustiveTesting<9, Integer<9, true, 1>>();
|
|
|
|
exhaustiveTesting<9, Integer<9, true, 1, std::uint8_t, std::uint16_t>>();
|
|
|
|
exhaustiveTesting<9, Integer<9, true, 2>>();
|
|
|
|
exhaustiveTesting<9, Integer<9, true, 2, std::uint8_t, std::uint16_t>>();
|
|
|
|
exhaustiveTesting<9, Integer<9, true, 8, std::uint8_t, std::uint16_t>>();
|
|
|
|
exhaustiveTesting<9, Integer<9, false, 8, std::uint8_t, std::uint16_t>>();
|
2018-05-31 07:08:42 +08:00
|
|
|
return testing::Complete();
|
|
|
|
}
|