Skip to content

Commit

Permalink
Bug 1733963 - Part 1: Make BitSet work as storage for EnumSet. r=glan…
Browse files Browse the repository at this point in the history
  • Loading branch information
farre committed Nov 9, 2021
1 parent 8f4fabf commit f31ef6f
Show file tree
Hide file tree
Showing 3 changed files with 113 additions and 23 deletions.
2 changes: 1 addition & 1 deletion mfbt/Array.h
Original file line number Diff line number Diff line change
Expand Up @@ -29,7 +29,7 @@ class Array {
using ElementType = T;
static constexpr size_t Length = _Length;

Array() = default;
constexpr Array() = default;

template <typename... Args>
MOZ_IMPLICIT constexpr Array(Args&&... aArgs)
Expand Down
72 changes: 67 additions & 5 deletions mfbt/BitSet.h
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,8 @@
#define mozilla_BitSet_h

#include "mozilla/Array.h"
#include "mozilla/ArrayUtils.h"
#include "mozilla/MathAlgorithms.h"
#include "mozilla/PodOperations.h"
#include "mozilla/Span.h"

Expand All @@ -28,10 +30,18 @@ class BitSet {
private:
static constexpr size_t kBitsPerWord = 8 * sizeof(Word);
static constexpr size_t kNumWords = (N + kBitsPerWord - 1) / kBitsPerWord;
static constexpr size_t kPaddingBits = (kNumWords * kBitsPerWord) - N;
static constexpr Word kPaddingMask = Word(-1) >> kPaddingBits;

// The zeroth bit in the bitset is the least significant bit of mStorage[0].
Array<Word, kNumWords> mStorage;

constexpr void ResetPaddingBits() {
if constexpr (kPaddingBits != 0) {
mStorage[kNumWords - 1] &= kPaddingMask;
}
}

public:
class Reference {
public:
Expand All @@ -52,7 +62,7 @@ class BitSet {
size_t mPos;
};

BitSet() { ResetAll(); }
constexpr BitSet() : mStorage() {}

BitSet(const BitSet& aOther) { *this = aOther; }

Expand All @@ -72,6 +82,17 @@ class BitSet {
return mStorage[aPos / kBitsPerWord] & (Word(1) << (aPos % kBitsPerWord));
}

constexpr bool IsEmpty() const {
for (const Word& word : mStorage) {
if (word) {
return false;
}
}
return true;
}

explicit constexpr operator bool() { return !IsEmpty(); }

constexpr bool operator[](size_t aPos) const { return Test(aPos); }

Reference operator[](size_t aPos) {
Expand All @@ -92,17 +113,58 @@ class BitSet {
return *this;
}

BitSet operator~() const {
BitSet result = *this;
result.Flip();
return result;
}

BitSet& operator&=(const BitSet<N, Word>& aOther) {
for (size_t i = 0; i < ArrayLength(mStorage); i++) {
mStorage[i] &= aOther.mStorage[i];
}
return *this;
}

BitSet operator&(const BitSet<N, Word>& aOther) const {
BitSet result = *this;
result &= aOther;
return result;
}

bool operator==(const BitSet<N, Word>& aOther) const {
return mStorage == aOther.mStorage;
}

size_t Count() const {
size_t count = 0;

for (const Word& word : mStorage) {
if constexpr (kBitsPerWord > 32) {
count += CountPopulation64(word);
} else {
count += CountPopulation32(word);
}
}

return count;
}

// Set all bits to false.
void ResetAll() { PodArrayZero(mStorage); }

// Set all bits to true.
void SetAll() {
memset(mStorage.begin(), 0xff, kNumWords * sizeof(Word));
constexpr size_t paddingBits = (kNumWords * kBitsPerWord) - N;
constexpr Word paddingMask = Word(-1) >> paddingBits;
if constexpr (paddingBits != 0) {
mStorage[kNumWords - 1] &= paddingMask;
ResetPaddingBits();
}

void Flip() {
for (Word& word : mStorage) {
word = ~word;
}

ResetPaddingBits();
}

Span<Word> Storage() { return mStorage; }
Expand Down
62 changes: 45 additions & 17 deletions mfbt/EnumSet.h
Original file line number Diff line number Diff line change
Expand Up @@ -11,6 +11,7 @@

#include "mozilla/Assertions.h"
#include "mozilla/Attributes.h"
#include "mozilla/MathAlgorithms.h"

#include <initializer_list>
#include <type_traits>
Expand All @@ -22,15 +23,16 @@ namespace mozilla {
/**
* EnumSet<T, U> is a set of values defined by an enumeration. It is implemented
* using a bit mask with the size of U for each value. It works both for enum
* and enum class types.
* and enum class types. EnumSet also works with U being a BitSet.
*/
template <typename T, typename Serialized = typename std::make_unsigned<
typename std::underlying_type<T>::type>::type>
class EnumSet {
public:
typedef T valueType;
using valueType = T;
using serializedType = Serialized;

constexpr EnumSet() : mBitField(0) {}
constexpr EnumSet() : mBitField() {}

constexpr MOZ_IMPLICIT EnumSet(T aEnum) : mBitField(bitFor(aEnum)) {}

Expand All @@ -44,7 +46,7 @@ class EnumSet {
: mBitField(bitFor(aEnum1) | bitFor(aEnum2) | bitFor(aEnum3) |
bitFor(aEnum4)) {}

constexpr MOZ_IMPLICIT EnumSet(std::initializer_list<T> list) : mBitField(0) {
constexpr MOZ_IMPLICIT EnumSet(std::initializer_list<T> list) : mBitField() {
for (auto value : list) {
(*this) += value;
}
Expand Down Expand Up @@ -133,7 +135,7 @@ class EnumSet {
*/
void clear() {
incVersion();
mBitField = 0;
mBitField = Serialized();
}

/**
Expand Down Expand Up @@ -180,7 +182,9 @@ class EnumSet {
/**
* Test is an element is contained in the set.
*/
bool contains(T aEnum) const { return mBitField & bitFor(aEnum); }
bool contains(T aEnum) const {
return static_cast<bool>(mBitField & bitFor(aEnum));
}

/**
* Test if a set is contained in the set.
Expand All @@ -192,17 +196,25 @@ class EnumSet {
/**
* Return the number of elements in the set.
*/
uint8_t size() const {
uint8_t count = 0;
for (Serialized bitField = mBitField; bitField; bitField >>= 1) {
if (bitField & 1) {
count++;
size_t size() const {
if constexpr (std::is_unsigned_v<Serialized>) {
if constexpr (kMaxBits > 32) {
return CountPopulation64(mBitField);
} else {
return CountPopulation32(mBitField);
}
} else {
return mBitField.Count();
}
return count;
}

bool isEmpty() const { return mBitField == 0; }
bool isEmpty() const {
if constexpr (std::is_unsigned_v<Serialized>) {
return mBitField == 0;
} else {
return mBitField.IsEmpty();
}
}

Serialized serialize() const { return mBitField; }

Expand Down Expand Up @@ -230,7 +242,9 @@ class EnumSet {
mVersion = mSet->mVersion;
#endif
MOZ_ASSERT(aPos <= kMaxBits);
if (aPos != kMaxBits && !mSet->contains(T(mPos))) ++*this;
if (aPos != kMaxBits && !mSet->contains(T(mPos))) {
++*this;
}
}

ConstIterator(const ConstIterator& aOther)
Expand Down Expand Up @@ -287,9 +301,15 @@ class EnumSet {

private:
constexpr static Serialized bitFor(T aEnum) {
auto bitNumber = static_cast<Serialized>(aEnum);
auto bitNumber = static_cast<size_t>(aEnum);
MOZ_DIAGNOSTIC_ASSERT(bitNumber < kMaxBits);
return static_cast<Serialized>(Serialized{1} << bitNumber);
if constexpr (std::is_unsigned_v<Serialized>) {
return static_cast<Serialized>(Serialized{1} << bitNumber);
} else {
Serialized bitField;
bitField[bitNumber] = true;
return bitField;
}
}

constexpr void incVersion() {
Expand All @@ -298,7 +318,15 @@ class EnumSet {
#endif
}

static const size_t kMaxBits = sizeof(Serialized) * 8;
constexpr size_t MaxBits() const {
if constexpr (std::is_unsigned_v<Serialized>) {
return sizeof(Serialized) * 8;
} else {
return mBitField.Size();
}
};

static constexpr size_t kMaxBits = EnumSet().MaxBits();

Serialized mBitField;

Expand Down

0 comments on commit f31ef6f

Please sign in to comment.