Skip to content

Commit

Permalink
[BitVector] Add find_last() and find_last_unset().
Browse files Browse the repository at this point in the history
Differential Revision: https://reviews.llvm.org/D32302

llvm-svn: 301014
  • Loading branch information
Zachary Turner committed Apr 21, 2017
1 parent 72fbd34 commit 492674e
Show file tree
Hide file tree
Showing 3 changed files with 70 additions and 4 deletions.
41 changes: 40 additions & 1 deletion llvm/include/llvm/ADT/BitVector.h
Expand Up @@ -15,7 +15,6 @@
#define LLVM_ADT_BITVECTOR_H

#include "llvm/ADT/ArrayRef.h"
#include "llvm/ADT/STLExtras.h"
#include "llvm/Support/MathExtras.h"
#include <algorithm>
#include <cassert>
Expand Down Expand Up @@ -163,6 +162,22 @@ class BitVector {
return -1;
}

/// find_last - Returns the index of the last set bit, -1 if none of the bits
/// are set.
int find_last() const {
if (Size == 0)
return -1;

unsigned N = NumBitWords(size());
assert(N > 0);

unsigned i = N - 1;
while (i > 0 && Bits[i] == BitWord(0))
--i;

return int((i + 1) * BITWORD_SIZE - countLeadingZeros(Bits[i])) - 1;
}

/// find_first_unset - Returns the index of the first unset bit, -1 if all
/// of the bits are set.
int find_first_unset() const {
Expand All @@ -174,6 +189,30 @@ class BitVector {
return -1;
}

/// find_last_unset - Returns the index of the last unset bit, -1 if all of
/// the bits are set.
int find_last_unset() const {
if (Size == 0)
return -1;

const unsigned N = NumBitWords(size());
assert(N > 0);

unsigned i = N - 1;
BitWord W = Bits[i];

// The last word in the BitVector has some unused bits, so we need to set
// them all to 1 first. Set them all to 1 so they don't get treated as
// valid unset bits.
unsigned UnusedCount = BITWORD_SIZE - size() % BITWORD_SIZE;
W |= maskLeadingOnes<BitWord>(UnusedCount);

while (W == ~BitWord(0) && --i > 0)
W = Bits[i];

return int((i + 1) * BITWORD_SIZE - countLeadingOnes(W)) - 1;
}

/// find_next - Returns the index of the next set bit following the
/// "Prev" bit. Returns -1 if the next set bit is not found.
int find_next(unsigned Prev) const {
Expand Down
25 changes: 22 additions & 3 deletions llvm/include/llvm/ADT/SmallBitVector.h
Expand Up @@ -117,9 +117,7 @@ class SmallBitVector {
}

// Return the size.
size_t getSmallSize() const {
return getSmallRawBits() >> SmallNumDataBits;
}
size_t getSmallSize() const { return getSmallRawBits() >> SmallNumDataBits; }

void setSmallSize(size_t Size) {
setSmallRawBits(getSmallBits() | (Size << SmallNumDataBits));
Expand Down Expand Up @@ -216,6 +214,16 @@ class SmallBitVector {
return getPointer()->find_first();
}

int find_last() const {
if (isSmall()) {
uintptr_t Bits = getSmallBits();
if (Bits == 0)
return -1;
return NumBaseBits - countLeadingZeros(Bits);
}
return getPointer()->find_last();
}

/// Returns the index of the first unset bit, -1 if all of the bits are set.
int find_first_unset() const {
if (isSmall()) {
Expand All @@ -228,6 +236,17 @@ class SmallBitVector {
return getPointer()->find_first_unset();
}

int find_last_unset() const {
if (isSmall()) {
if (count() == getSmallSize())
return -1;

uintptr_t Bits = getSmallBits();
return NumBaseBits - countLeadingOnes(Bits);
}
return getPointer()->find_last_unset();
}

/// Returns the index of the next set bit following the "Prev" bit.
/// Returns -1 if the next set bit is not found.
int find_next(unsigned Prev) const {
Expand Down
8 changes: 8 additions & 0 deletions llvm/unittests/ADT/BitVectorTest.cpp
Expand Up @@ -186,7 +186,9 @@ TYPED_TEST(BitVectorTest, FindOperations) {
// Test finding in an empty BitVector.
TypeParam A;
EXPECT_EQ(-1, A.find_first());
EXPECT_EQ(-1, A.find_last());
EXPECT_EQ(-1, A.find_first_unset());
EXPECT_EQ(-1, A.find_last_unset());
EXPECT_EQ(-1, A.find_next(0));
EXPECT_EQ(-1, A.find_next_unset(0));

Expand All @@ -196,12 +198,14 @@ TYPED_TEST(BitVectorTest, FindOperations) {
A.set(13);
A.set(75);

EXPECT_EQ(75, A.find_last());
EXPECT_EQ(12, A.find_first());
EXPECT_EQ(13, A.find_next(12));
EXPECT_EQ(75, A.find_next(13));
EXPECT_EQ(-1, A.find_next(75));

EXPECT_EQ(0, A.find_first_unset());
EXPECT_EQ(99, A.find_last_unset());
EXPECT_EQ(14, A.find_next_unset(11));
EXPECT_EQ(14, A.find_next_unset(12));
EXPECT_EQ(14, A.find_next_unset(13));
Expand All @@ -213,12 +217,16 @@ TYPED_TEST(BitVectorTest, FindOperations) {
A.set(0, 100);
EXPECT_EQ(100U, A.count());
EXPECT_EQ(0, A.find_first());
EXPECT_EQ(99, A.find_last());
EXPECT_EQ(-1, A.find_first_unset());
EXPECT_EQ(-1, A.find_last_unset());

A.reset(0, 100);
EXPECT_EQ(0U, A.count());
EXPECT_EQ(-1, A.find_first());
EXPECT_EQ(-1, A.find_last());
EXPECT_EQ(0, A.find_first_unset());
EXPECT_EQ(99, A.find_last_unset());
}

TYPED_TEST(BitVectorTest, CompoundAssignment) {
Expand Down

0 comments on commit 492674e

Please sign in to comment.