mirror of
https://github.com/mozilla/gecko-dev.git
synced 2024-11-27 06:43:32 +00:00
875628fc84
This requires to flag some method from Enumset as constexpr but make it clear that those don't require a global constructor. Differential Revision: https://phabricator.services.mozilla.com/D222792
353 lines
7.7 KiB
C++
353 lines
7.7 KiB
C++
/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
|
/* vim: set ts=8 sts=2 et sw=2 tw=80: */
|
|
/* This Source Code Form is subject to the terms of the Mozilla Public
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
|
|
|
|
/* A set abstraction for enumeration values. */
|
|
|
|
#ifndef mozilla_EnumSet_h
|
|
#define mozilla_EnumSet_h
|
|
|
|
#include "mozilla/Assertions.h"
|
|
#include "mozilla/Attributes.h"
|
|
#include "mozilla/MathAlgorithms.h"
|
|
|
|
#include <initializer_list>
|
|
#include <type_traits>
|
|
|
|
#include <stdint.h>
|
|
|
|
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. 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:
|
|
using valueType = T;
|
|
using serializedType = Serialized;
|
|
|
|
constexpr EnumSet() : mBitField() {}
|
|
|
|
constexpr MOZ_IMPLICIT EnumSet(T aEnum) : mBitField(BitFor(aEnum)) {}
|
|
|
|
constexpr EnumSet(T aEnum1, T aEnum2)
|
|
: mBitField(BitFor(aEnum1) | BitFor(aEnum2)) {}
|
|
|
|
constexpr EnumSet(T aEnum1, T aEnum2, T aEnum3)
|
|
: mBitField(BitFor(aEnum1) | BitFor(aEnum2) | BitFor(aEnum3)) {}
|
|
|
|
constexpr EnumSet(T aEnum1, T aEnum2, T aEnum3, T aEnum4)
|
|
: mBitField(BitFor(aEnum1) | BitFor(aEnum2) | BitFor(aEnum3) |
|
|
BitFor(aEnum4)) {}
|
|
|
|
constexpr MOZ_IMPLICIT EnumSet(std::initializer_list<T> list) : mBitField() {
|
|
for (auto value : list) {
|
|
(*this) += value;
|
|
}
|
|
}
|
|
|
|
#ifdef DEBUG
|
|
constexpr EnumSet(const EnumSet& aEnumSet) : mBitField(aEnumSet.mBitField) {}
|
|
|
|
constexpr EnumSet& operator=(const EnumSet& aEnumSet) {
|
|
mBitField = aEnumSet.mBitField;
|
|
IncVersion();
|
|
return *this;
|
|
}
|
|
#endif
|
|
|
|
/**
|
|
* Add an element
|
|
*/
|
|
constexpr void operator+=(T aEnum) {
|
|
IncVersion();
|
|
mBitField |= BitFor(aEnum);
|
|
}
|
|
|
|
/**
|
|
* Add an element
|
|
*/
|
|
constexpr EnumSet operator+(T aEnum) const {
|
|
EnumSet result(*this);
|
|
result += aEnum;
|
|
return result;
|
|
}
|
|
|
|
/**
|
|
* Union
|
|
*/
|
|
constexpr void operator+=(const EnumSet& aEnumSet) {
|
|
IncVersion();
|
|
mBitField |= aEnumSet.mBitField;
|
|
}
|
|
|
|
/**
|
|
* Union
|
|
*/
|
|
constexpr EnumSet operator+(const EnumSet& aEnumSet) const {
|
|
EnumSet result(*this);
|
|
result += aEnumSet;
|
|
return result;
|
|
}
|
|
|
|
/**
|
|
* Remove an element
|
|
*/
|
|
constexpr void operator-=(T aEnum) {
|
|
IncVersion();
|
|
mBitField &= ~(BitFor(aEnum));
|
|
}
|
|
|
|
/**
|
|
* Remove an element
|
|
*/
|
|
constexpr EnumSet operator-(T aEnum) const {
|
|
EnumSet result(*this);
|
|
result -= aEnum;
|
|
return result;
|
|
}
|
|
|
|
/**
|
|
* Remove a set of elements
|
|
*/
|
|
constexpr void operator-=(const EnumSet& aEnumSet) {
|
|
IncVersion();
|
|
mBitField &= ~(aEnumSet.mBitField);
|
|
}
|
|
|
|
/**
|
|
* Remove a set of elements
|
|
*/
|
|
constexpr EnumSet operator-(const EnumSet& aEnumSet) const {
|
|
EnumSet result(*this);
|
|
result -= aEnumSet;
|
|
return result;
|
|
}
|
|
|
|
/**
|
|
* Clear
|
|
*/
|
|
constexpr void clear() {
|
|
IncVersion();
|
|
mBitField = Serialized();
|
|
}
|
|
|
|
/**
|
|
* Intersection
|
|
*/
|
|
constexpr void operator&=(const EnumSet& aEnumSet) {
|
|
IncVersion();
|
|
mBitField &= aEnumSet.mBitField;
|
|
}
|
|
|
|
/**
|
|
* Intersection
|
|
*/
|
|
constexpr EnumSet operator&(const EnumSet& aEnumSet) const {
|
|
EnumSet result(*this);
|
|
result &= aEnumSet;
|
|
return result;
|
|
}
|
|
|
|
/**
|
|
* Equality
|
|
*/
|
|
constexpr bool operator==(const EnumSet& aEnumSet) const {
|
|
return mBitField == aEnumSet.mBitField;
|
|
}
|
|
|
|
/**
|
|
* Equality
|
|
*/
|
|
constexpr bool operator==(T aEnum) const {
|
|
return mBitField == BitFor(aEnum);
|
|
}
|
|
|
|
/**
|
|
* Not equal
|
|
*/
|
|
constexpr bool operator!=(const EnumSet& aEnumSet) const {
|
|
return !operator==(aEnumSet);
|
|
}
|
|
|
|
/**
|
|
* Not equal
|
|
*/
|
|
constexpr bool operator!=(T aEnum) const { return !operator==(aEnum); }
|
|
|
|
/**
|
|
* Test is an element is contained in the set.
|
|
*/
|
|
constexpr bool contains(T aEnum) const { return HasBitFor(aEnum); }
|
|
|
|
/**
|
|
* Test if a set is contained in the set.
|
|
*/
|
|
constexpr bool contains(const EnumSet& aEnumSet) const {
|
|
return (mBitField & aEnumSet.mBitField) == aEnumSet.mBitField;
|
|
}
|
|
|
|
/**
|
|
* Return the number of elements in the set.
|
|
*/
|
|
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();
|
|
}
|
|
}
|
|
|
|
constexpr bool isEmpty() const {
|
|
if constexpr (std::is_unsigned_v<Serialized>) {
|
|
return mBitField == 0;
|
|
} else {
|
|
return mBitField.IsEmpty();
|
|
}
|
|
}
|
|
|
|
Serialized serialize() const { return mBitField; }
|
|
|
|
void deserialize(Serialized aValue) {
|
|
IncVersion();
|
|
mBitField = aValue;
|
|
}
|
|
|
|
class ConstIterator {
|
|
const EnumSet* mSet;
|
|
size_t mPos;
|
|
#ifdef DEBUG
|
|
uint64_t mVersion;
|
|
#endif
|
|
|
|
void checkVersion() const {
|
|
// Check that the set has not been modified while being iterated.
|
|
MOZ_ASSERT_IF(mSet, mSet->mVersion == mVersion);
|
|
}
|
|
|
|
public:
|
|
ConstIterator(const EnumSet& aSet, size_t aPos) : mSet(&aSet), mPos(aPos) {
|
|
#ifdef DEBUG
|
|
mVersion = mSet->mVersion;
|
|
#endif
|
|
MOZ_ASSERT(aPos <= kMaxBits);
|
|
if (aPos != kMaxBits && !mSet->HasBitAt(mPos)) {
|
|
++*this;
|
|
}
|
|
}
|
|
|
|
ConstIterator(const ConstIterator& aOther)
|
|
: mSet(aOther.mSet), mPos(aOther.mPos) {
|
|
#ifdef DEBUG
|
|
mVersion = aOther.mVersion;
|
|
checkVersion();
|
|
#endif
|
|
}
|
|
|
|
ConstIterator(ConstIterator&& aOther)
|
|
: mSet(aOther.mSet), mPos(aOther.mPos) {
|
|
#ifdef DEBUG
|
|
mVersion = aOther.mVersion;
|
|
checkVersion();
|
|
#endif
|
|
aOther.mSet = nullptr;
|
|
}
|
|
|
|
~ConstIterator() { checkVersion(); }
|
|
|
|
bool operator==(const ConstIterator& other) const {
|
|
MOZ_ASSERT(mSet == other.mSet);
|
|
checkVersion();
|
|
return mPos == other.mPos;
|
|
}
|
|
|
|
bool operator!=(const ConstIterator& other) const {
|
|
return !(*this == other);
|
|
}
|
|
|
|
T operator*() const {
|
|
MOZ_ASSERT(mSet);
|
|
MOZ_ASSERT(mPos < kMaxBits);
|
|
MOZ_ASSERT(mSet->HasBitAt(mPos));
|
|
checkVersion();
|
|
return T(mPos);
|
|
}
|
|
|
|
ConstIterator& operator++() {
|
|
MOZ_ASSERT(mSet);
|
|
MOZ_ASSERT(mPos < kMaxBits);
|
|
checkVersion();
|
|
do {
|
|
mPos++;
|
|
} while (mPos < kMaxBits && !mSet->HasBitAt(mPos));
|
|
return *this;
|
|
}
|
|
};
|
|
|
|
ConstIterator begin() const { return ConstIterator(*this, 0); }
|
|
|
|
ConstIterator end() const { return ConstIterator(*this, kMaxBits); }
|
|
|
|
private:
|
|
constexpr static Serialized BitFor(T aEnum) {
|
|
const auto pos = static_cast<size_t>(aEnum);
|
|
return BitAt(pos);
|
|
}
|
|
|
|
constexpr static Serialized BitAt(size_t aPos) {
|
|
MOZ_DIAGNOSTIC_ASSERT(aPos < kMaxBits);
|
|
if constexpr (std::is_unsigned_v<Serialized>) {
|
|
return static_cast<Serialized>(Serialized{1} << aPos);
|
|
} else {
|
|
Serialized bitField;
|
|
bitField[aPos] = true;
|
|
return bitField;
|
|
}
|
|
}
|
|
|
|
constexpr bool HasBitFor(T aEnum) const {
|
|
const auto pos = static_cast<size_t>(aEnum);
|
|
return HasBitAt(pos);
|
|
}
|
|
|
|
constexpr bool HasBitAt(size_t aPos) const {
|
|
return static_cast<bool>(mBitField & BitAt(aPos));
|
|
}
|
|
|
|
constexpr void IncVersion() {
|
|
#ifdef DEBUG
|
|
mVersion++;
|
|
#endif
|
|
}
|
|
|
|
static constexpr size_t MaxBits() {
|
|
if constexpr (std::is_unsigned_v<Serialized>) {
|
|
return sizeof(Serialized) * 8;
|
|
} else {
|
|
return Serialized::Size();
|
|
}
|
|
}
|
|
|
|
static constexpr size_t kMaxBits = MaxBits();
|
|
|
|
Serialized mBitField;
|
|
|
|
#ifdef DEBUG
|
|
uint64_t mVersion = 0;
|
|
#endif
|
|
};
|
|
|
|
} // namespace mozilla
|
|
|
|
#endif /* mozilla_EnumSet_h_*/
|