Skip to content

Commit

Permalink
[APSInt] Fix bug in APSInt mentioned in llvm/llvm-project#59515
Browse files Browse the repository at this point in the history
Also provide a `tryExtValue()` API like APInt did in D139683

Reviewed By: RKSimon

Differential Revision: https://reviews.llvm.org/D140059
  • Loading branch information
DataCorrupted authored and veselypeta committed Jun 9, 2024
2 parents ac6485b + 7859d6e commit 37858a4
Show file tree
Hide file tree
Showing 2 changed files with 32 additions and 1 deletion.
16 changes: 15 additions & 1 deletion llvm/include/llvm/ADT/APSInt.h
Original file line number Diff line number Diff line change
Expand Up @@ -85,12 +85,26 @@ class [[nodiscard]] APSInt : public APInt {
}
using APInt::toString;

/// If this int is representable using an int64_t.
bool isRepresentableByInt64() const {
// For unsigned values with 64 active bits, they technically fit into a
// int64_t, but the user may get negative numbers and has to manually cast
// them to unsigned. Let's not bet the user has the sanity to do that and
// not give them a vague value at the first place.
return isSigned() ? isSignedIntN(64) : isIntN(63);
}

/// Get the correctly-extended \c int64_t value.
int64_t getExtValue() const {
assert(getMinSignedBits() <= 64 && "Too many bits for int64_t");
assert(isRepresentableByInt64() && "Too many bits for int64_t");
return isSigned() ? getSExtValue() : getZExtValue();
}

std::optional<int64_t> tryExtValue() const {
return isRepresentableByInt64() ? std::optional<int64_t>(getExtValue())
: std::nullopt;
}

APSInt trunc(uint32_t width) const {
return APSInt(APInt::trunc(width), IsUnsigned);
}
Expand Down
17 changes: 17 additions & 0 deletions llvm/unittests/ADT/APSIntTest.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -62,6 +62,13 @@ TEST(APSIntTest, getUnsigned) {
EXPECT_EQ(UINT64_C(0) - 7, APSInt::getUnsigned(-7).getZExtValue());
}

TEST(APSIntTest, isRepresentableByInt64) {
ASSERT_TRUE(APSInt(APInt(3, 7), true).isRepresentableByInt64());
ASSERT_TRUE(APSInt(APInt(128, 7), true).isRepresentableByInt64());
ASSERT_TRUE(APSInt(APInt(128, 7), false).isRepresentableByInt64());
ASSERT_TRUE(APSInt(APInt(64, -1), false).isRepresentableByInt64());
ASSERT_FALSE(APSInt(APInt(64, (uint64_t)-1), true).isRepresentableByInt64());
}
TEST(APSIntTest, getExtValue) {
EXPECT_TRUE(APSInt(APInt(3, 7), true).isUnsigned());
EXPECT_TRUE(APSInt(APInt(3, 7), false).isSigned());
Expand All @@ -76,6 +83,16 @@ TEST(APSIntTest, getExtValue) {
EXPECT_EQ(9, APSInt(APInt(4, -7), true).getExtValue());
EXPECT_EQ(-7, APSInt(APInt(4, -7), false).getExtValue());
}
TEST(APSIntTest, tryExtValue) {
ASSERT_EQ(-7, APSInt(APInt(64, -7), false).tryExtValue().value_or(42));
ASSERT_EQ(42, APSInt(APInt(128, -7), false).tryExtValue().value_or(42));
ASSERT_EQ(-1,
APSInt(APInt::getAllOnes(128), false).tryExtValue().value_or(42));
ASSERT_EQ(42, APSInt(APInt(64, -7), true).tryExtValue().value_or(42));
ASSERT_EQ(1, APSInt(APInt(128, 1), true).tryExtValue().value_or(42));
ASSERT_EQ(42,
APSInt(APInt::getAllOnes(128), true).tryExtValue().value_or(42));
}

TEST(APSIntTest, compareValues) {
auto U = [](uint64_t V) { return APSInt::getUnsigned(V); };
Expand Down

0 comments on commit 37858a4

Please sign in to comment.