diff --git a/llvm/include/llvm/ADT/bit.h b/llvm/include/llvm/ADT/bit.h --- a/llvm/include/llvm/ADT/bit.h +++ b/llvm/include/llvm/ADT/bit.h @@ -217,6 +217,42 @@ return llvm::countr_zero(~Value); } +/// Returns the number of bits needed to represent Value if Value is nonzero. +/// Returns 0 otherwise. +/// +/// Ex. bit_width(5) == 3. +template int bit_width(T Value) { + static_assert(std::is_unsigned_v, + "Only unsigned integral types are allowed."); + return std::numeric_limits::digits - llvm::countl_zero(Value); +} + +/// Returns the largest integral power of two no greater than Value if Value is +/// nonzero. Returns 0 otherwise. +/// +/// Ex. bit_floor(5) == 4. +template T bit_floor(T Value) { + static_assert(std::is_unsigned_v, + "Only unsigned integral types are allowed."); + if (!Value) + return 0; + return T(1) << (llvm::bit_width(Value) - 1); +} + +/// Returns the smallest integral power of two no smaller than Value if Value is +/// nonzero. Returns 0 otherwise. +/// +/// Ex. bit_ceil(5) == 8. +/// +/// The return value is undefined if the input is larger than the largest power +/// of two representable in T. +template T bit_ceil(T Value) { + static_assert(std::is_unsigned_v, + "Only unsigned integral types are allowed."); + if (Value < 2) return 1; + return T(1) << llvm::bit_width(Value - 1u); +} + namespace detail { template struct PopulationCounter { static int count(T Value) { diff --git a/llvm/unittests/ADT/BitTest.cpp b/llvm/unittests/ADT/BitTest.cpp --- a/llvm/unittests/ADT/BitTest.cpp +++ b/llvm/unittests/ADT/BitTest.cpp @@ -48,6 +48,87 @@ EXPECT_TRUE(llvm::has_single_bit(static_cast(kValueS16))); } +TEST(BitTest, BitFloor) { + EXPECT_EQ(0u, llvm::bit_floor(uint8_t(0))); + EXPECT_EQ(0u, llvm::bit_floor(uint16_t(0))); + EXPECT_EQ(0u, llvm::bit_floor(uint32_t(0))); + EXPECT_EQ(0u, llvm::bit_floor(uint64_t(0))); + + EXPECT_EQ(1u, llvm::bit_floor(uint8_t(1))); + EXPECT_EQ(1u, llvm::bit_floor(uint16_t(1))); + EXPECT_EQ(1u, llvm::bit_floor(uint32_t(1))); + EXPECT_EQ(1u, llvm::bit_floor(uint64_t(1))); + + EXPECT_EQ(2u, llvm::bit_floor(uint8_t(2))); + EXPECT_EQ(2u, llvm::bit_floor(uint16_t(2))); + EXPECT_EQ(2u, llvm::bit_floor(uint32_t(2))); + EXPECT_EQ(2u, llvm::bit_floor(uint64_t(2))); + + EXPECT_EQ(2u, llvm::bit_floor(uint8_t(3))); + EXPECT_EQ(2u, llvm::bit_floor(uint16_t(3))); + EXPECT_EQ(2u, llvm::bit_floor(uint32_t(3))); + EXPECT_EQ(2u, llvm::bit_floor(uint64_t(3))); + + EXPECT_EQ(4u, llvm::bit_floor(uint8_t(4))); + EXPECT_EQ(4u, llvm::bit_floor(uint16_t(4))); + EXPECT_EQ(4u, llvm::bit_floor(uint32_t(4))); + EXPECT_EQ(4u, llvm::bit_floor(uint64_t(4))); +} + +TEST(BitTest, BitCeil) { + EXPECT_EQ(1u, llvm::bit_ceil(uint8_t(0))); + EXPECT_EQ(1u, llvm::bit_ceil(uint16_t(0))); + EXPECT_EQ(1u, llvm::bit_ceil(uint32_t(0))); + EXPECT_EQ(1u, llvm::bit_ceil(uint64_t(0))); + + EXPECT_EQ(1u, llvm::bit_ceil(uint8_t(1))); + EXPECT_EQ(1u, llvm::bit_ceil(uint16_t(1))); + EXPECT_EQ(1u, llvm::bit_ceil(uint32_t(1))); + EXPECT_EQ(1u, llvm::bit_ceil(uint64_t(1))); + + EXPECT_EQ(2u, llvm::bit_ceil(uint8_t(2))); + EXPECT_EQ(2u, llvm::bit_ceil(uint16_t(2))); + EXPECT_EQ(2u, llvm::bit_ceil(uint32_t(2))); + EXPECT_EQ(2u, llvm::bit_ceil(uint64_t(2))); + + EXPECT_EQ(4u, llvm::bit_ceil(uint8_t(3))); + EXPECT_EQ(4u, llvm::bit_ceil(uint16_t(3))); + EXPECT_EQ(4u, llvm::bit_ceil(uint32_t(3))); + EXPECT_EQ(4u, llvm::bit_ceil(uint64_t(3))); + + EXPECT_EQ(4u, llvm::bit_ceil(uint8_t(4))); + EXPECT_EQ(4u, llvm::bit_ceil(uint16_t(4))); + EXPECT_EQ(4u, llvm::bit_ceil(uint32_t(4))); + EXPECT_EQ(4u, llvm::bit_ceil(uint64_t(4))); +} + +TEST(BitTest, BitWidth) { + EXPECT_EQ(0, llvm::bit_width(uint8_t(0))); + EXPECT_EQ(0, llvm::bit_width(uint16_t(0))); + EXPECT_EQ(0, llvm::bit_width(uint32_t(0))); + EXPECT_EQ(0, llvm::bit_width(uint64_t(0))); + + EXPECT_EQ(1, llvm::bit_width(uint8_t(1))); + EXPECT_EQ(1, llvm::bit_width(uint16_t(1))); + EXPECT_EQ(1, llvm::bit_width(uint32_t(1))); + EXPECT_EQ(1, llvm::bit_width(uint64_t(1))); + + EXPECT_EQ(2, llvm::bit_width(uint8_t(2))); + EXPECT_EQ(2, llvm::bit_width(uint16_t(2))); + EXPECT_EQ(2, llvm::bit_width(uint32_t(2))); + EXPECT_EQ(2, llvm::bit_width(uint64_t(2))); + + EXPECT_EQ(2, llvm::bit_width(uint8_t(3))); + EXPECT_EQ(2, llvm::bit_width(uint16_t(3))); + EXPECT_EQ(2, llvm::bit_width(uint32_t(3))); + EXPECT_EQ(2, llvm::bit_width(uint64_t(3))); + + EXPECT_EQ(3, llvm::bit_width(uint8_t(4))); + EXPECT_EQ(3, llvm::bit_width(uint16_t(4))); + EXPECT_EQ(3, llvm::bit_width(uint32_t(4))); + EXPECT_EQ(3, llvm::bit_width(uint64_t(4))); +} + TEST(BitTest, CountlZero) { uint8_t Z8 = 0; uint16_t Z16 = 0;