diff options
Diffstat (limited to 'unittests/ADT/APIntTest.cpp')
-rw-r--r-- | unittests/ADT/APIntTest.cpp | 137 |
1 files changed, 137 insertions, 0 deletions
diff --git a/unittests/ADT/APIntTest.cpp b/unittests/ADT/APIntTest.cpp index 3b7ac5b..498f50c 100644 --- a/unittests/ADT/APIntTest.cpp +++ b/unittests/ADT/APIntTest.cpp @@ -209,6 +209,103 @@ TEST(APIntTest, i1) { } } + +// Tests different div/rem varaints using scheme (a * b + c) / a +void testDiv(APInt a, APInt b, APInt c) { + ASSERT_TRUE(a.uge(b)); // Must: a >= b + ASSERT_TRUE(a.ugt(c)); // Must: a > c + + auto p = a * b + c; + + auto q = p.udiv(a); + auto r = p.urem(a); + EXPECT_EQ(b, q); + EXPECT_EQ(c, r); + APInt::udivrem(p, a, q, r); + EXPECT_EQ(b, q); + EXPECT_EQ(c, r); + q = p.sdiv(a); + r = p.srem(a); + EXPECT_EQ(b, q); + EXPECT_EQ(c, r); + APInt::sdivrem(p, a, q, r); + EXPECT_EQ(b, q); + EXPECT_EQ(c, r); + + if (b.ugt(c)) { // Test also symmetric case + q = p.udiv(b); + r = p.urem(b); + EXPECT_EQ(a, q); + EXPECT_EQ(c, r); + APInt::udivrem(p, b, q, r); + EXPECT_EQ(a, q); + EXPECT_EQ(c, r); + q = p.sdiv(b); + r = p.srem(b); + EXPECT_EQ(a, q); + EXPECT_EQ(c, r); + APInt::sdivrem(p, b, q, r); + EXPECT_EQ(a, q); + EXPECT_EQ(c, r); + } +} + +TEST(APIntTest, divrem_big1) { + // Tests KnuthDiv rare step D6 + testDiv({256, "1ffffffffffffffff", 16}, + {256, "1ffffffffffffffff", 16}, + {256, 0}); +} + +TEST(APIntTest, divrem_big2) { + // Tests KnuthDiv rare step D6 + testDiv({1024, "112233ceff" + "cecece000000ffffffffffffffffffff" + "ffffffffffffffffffffffffffffffff" + "ffffffffffffffffffffffffffffffff" + "ffffffffffffffffffffffffffffff33", 16}, + {1024, "111111ffffffffffffffff" + "ffffffffffffffffffffffffffffffff" + "fffffffffffffffffffffffffffffccf" + "ffffffffffffffffffffffffffffff00", 16}, + {1024, 7919}); +} + +TEST(APIntTest, divrem_big3) { + // Tests KnuthDiv case without shift + testDiv({256, "80000001ffffffffffffffff", 16}, + {256, "ffffffffffffff0000000", 16}, + {256, 4219}); +} + +TEST(APIntTest, divrem_big4) { + // Tests heap allocation in divide() enfoced by huge numbers + testDiv(APInt{4096, 5}.shl(2001), + APInt{4096, 1}.shl(2000), + APInt{4096, 4219*13}); +} + +TEST(APIntTest, divrem_big5) { + // Tests one word divisor case of divide() + testDiv(APInt{1024, 19}.shl(811), + APInt{1024, 4356013}, // one word + APInt{1024, 1}); +} + +TEST(APIntTest, divrem_big6) { + // Tests some rare "borrow" cases in D4 step + testDiv(APInt{512, "ffffffffffffffff00000000000000000000000001", 16}, + APInt{512, "10000000000000001000000000000001", 16}, + APInt{512, "10000000000000000000000000000000", 16}); +} + +TEST(APIntTest, divrem_big7) { + // Yet another test for KnuthDiv rare step D6. + testDiv({224, "800000008000000200000005", 16}, + {224, "fffffffd", 16}, + {224, "80000000800000010000000f", 16}); +} + TEST(APIntTest, fromString) { EXPECT_EQ(APInt(32, 0), APInt(32, "0", 2)); EXPECT_EQ(APInt(32, 1), APInt(32, "1", 2)); @@ -678,6 +775,46 @@ TEST(APIntTest, nearestLogBase2) { EXPECT_EQ(A9.nearestLogBase2(), UINT32_MAX); } +TEST(APIntTest, IsSplat) { + APInt A(32, 0x01010101); + EXPECT_FALSE(A.isSplat(1)); + EXPECT_FALSE(A.isSplat(2)); + EXPECT_FALSE(A.isSplat(4)); + EXPECT_TRUE(A.isSplat(8)); + EXPECT_TRUE(A.isSplat(16)); + EXPECT_TRUE(A.isSplat(32)); + + APInt B(24, 0xAAAAAA); + EXPECT_FALSE(B.isSplat(1)); + EXPECT_TRUE(B.isSplat(2)); + EXPECT_TRUE(B.isSplat(4)); + EXPECT_TRUE(B.isSplat(8)); + EXPECT_TRUE(B.isSplat(24)); + + APInt C(24, 0xABAAAB); + EXPECT_FALSE(C.isSplat(1)); + EXPECT_FALSE(C.isSplat(2)); + EXPECT_FALSE(C.isSplat(4)); + EXPECT_FALSE(C.isSplat(8)); + EXPECT_TRUE(C.isSplat(24)); + + APInt D(32, 0xABBAABBA); + EXPECT_FALSE(D.isSplat(1)); + EXPECT_FALSE(D.isSplat(2)); + EXPECT_FALSE(D.isSplat(4)); + EXPECT_FALSE(D.isSplat(8)); + EXPECT_TRUE(D.isSplat(16)); + EXPECT_TRUE(D.isSplat(32)); + + APInt E(32, 0); + EXPECT_TRUE(E.isSplat(1)); + EXPECT_TRUE(E.isSplat(2)); + EXPECT_TRUE(E.isSplat(4)); + EXPECT_TRUE(E.isSplat(8)); + EXPECT_TRUE(E.isSplat(16)); + EXPECT_TRUE(E.isSplat(32)); +} + #if defined(__clang__) // Disable the pragma warning from versions of Clang without -Wself-move #pragma clang diagnostic push |