/*
* Copyright (c) 2018-2020 Atmosphère-NX
*
* This program is free software; you can redistribute it and/or modify it
* under the terms and conditions of the GNU General Public License,
* version 2, as published by the Free Software Foundation.
*
* This program is distributed in the hope it will be useful, but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
* more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see .
*/
#pragma once
#include
#include
#include
#include
namespace ams::util {
namespace impl {
template
constexpr ALWAYS_INLINE void NegateImpl(Storage arr[]) {
for (size_t i = 0; i < Count; i++) {
arr[i] = ~arr[i];
}
}
template
constexpr ALWAYS_INLINE void AndImpl(Storage dst[], const Storage src[]) {
for (size_t i = 0; i < Count; i++) {
dst[i] &= src[i];
}
}
template
constexpr ALWAYS_INLINE void OrImpl(Storage dst[], const Storage src[]) {
for (size_t i = 0; i < Count; i++) {
dst[i] |= src[i];
}
}
template
constexpr ALWAYS_INLINE void XorImpl(Storage dst[], const Storage src[]) {
for (size_t i = 0; i < Count; i++) {
dst[i] ^= src[i];
}
}
template
constexpr ALWAYS_INLINE bool IsEqual(const Storage lhs[], const Storage rhs[]) {
for (size_t i = 0; i < Count; i++) {
if (lhs[i] != rhs[i]) {
return false;
}
}
return true;
}
template
constexpr ALWAYS_INLINE bool IsAnySet(const Storage arr[]) {
for (size_t i = 0; i < Count; i++) {
if (arr[i]) {
return true;
}
}
return false;
}
template
constexpr ALWAYS_INLINE int PopCount(const Storage arr[]) {
int count = 0;
for (size_t i = 0; i < Count; i++) {
count += ::ams::util::PopCount(arr[i]);
}
return count;
}
}
template
struct BitFlagSet {
static_assert(N > 0);
using Storage = typename std::conditional::type;
static constexpr size_t StorageBitCount = BITSIZEOF(Storage);
static constexpr size_t StorageCount = util::AlignUp(N, StorageBitCount) / StorageBitCount;
Storage _storage[StorageCount];
private:
constexpr ALWAYS_INLINE BitFlagSet &SetImpl(s32 idx, Storage mask, bool en) {
if (en) {
this->_storage[idx] |= mask;
} else {
this->_storage[idx] &= ~mask;
}
return *this;
}
constexpr ALWAYS_INLINE bool TestImpl(s32 idx, Storage mask) const { return (this->_storage[idx] & mask) != 0; }
constexpr ALWAYS_INLINE void Truncate() { TruncateIf(std::integral_constant{}); }
constexpr ALWAYS_INLINE void TruncateIf(std::true_type) { this->_storage[StorageCount - 1] &= MakeStorageMask(N) - 1; }
constexpr ALWAYS_INLINE void TruncateIf(std::false_type) { /* ... */ }
static constexpr ALWAYS_INLINE s32 GetStorageIndex(s32 idx) { return idx / StorageBitCount; }
static constexpr ALWAYS_INLINE Storage MakeStorageMask(s32 idx) { return static_cast(1) << (idx % StorageBitCount); }
public:
class Reference {
friend struct BitFlagSet;
private:
BitFlagSet *set;
s32 idx;
private:
constexpr ALWAYS_INLINE Reference() : set(nullptr), idx(0) { /* ... */ }
constexpr ALWAYS_INLINE Reference(BitFlagSet &s, s32 i) : set(std::addressof(s)), idx(i) { /* ... */ }
public:
constexpr ALWAYS_INLINE Reference &operator=(bool en) { this->set->Set(this->idx, en); return *this; }
constexpr ALWAYS_INLINE Reference &operator=(const Reference &r) { this->set->Set(this->idx, r); return *this; }
constexpr ALWAYS_INLINE Reference &Negate() { this->set->Negate(this->idx); return *this; }
constexpr ALWAYS_INLINE operator bool() const { return this->set->Test(this->idx); }
constexpr ALWAYS_INLINE bool operator~() const { return !this->set->Test(this->idx); }
};
template
struct Flag {
static_assert(_Index < static_cast(N));
friend struct BitFlagSet;
static constexpr s32 Index = _Index;
static const BitFlagSet Mask;
private:
static constexpr s32 StorageIndex = Index / StorageBitCount;
static constexpr Storage StorageMask = static_cast(1) << (Index % StorageBitCount);
template
struct SingleStorageTrait {
static_assert(StorageCount == 1);
using Type = Storage;
};
};
template
constexpr ALWAYS_INLINE bool Test() const { return this->TestImpl(FlagType::StorageIndex, FlagType::StorageMask); }
constexpr ALWAYS_INLINE bool Test(s32 idx) const { return this->TestImpl(GetStorageIndex(idx), MakeStorageMask(idx)); }
template
constexpr ALWAYS_INLINE BitFlagSet &Set(bool en = true) { return this->SetImpl(FlagType::StorageIndex, FlagType::StorageMask, en); }
constexpr ALWAYS_INLINE BitFlagSet &Set(s32 idx, bool en = true) { return this->SetImpl(GetStorageIndex(idx), MakeStorageMask(idx), en); }
constexpr ALWAYS_INLINE BitFlagSet &Set() { std::memset(this->_storage, ~0, sizeof(this->_storage)); this->Truncate(); return *this; }
template
constexpr ALWAYS_INLINE BitFlagSet &Reset() { return this->Set(false); }
constexpr ALWAYS_INLINE BitFlagSet &Reset(s32 idx) { return this->Set(idx, false); }
constexpr ALWAYS_INLINE BitFlagSet &Reset() { std::memset(this->_storage, 0, sizeof(this->_storage)); this->Truncate(); return *this; }
template
constexpr ALWAYS_INLINE BitFlagSet &Negate() { return this->Set(!this->Test()); }
constexpr ALWAYS_INLINE BitFlagSet &Negate(s32 idx) { return this->Set(idx, !this->Test(idx)); }
constexpr ALWAYS_INLINE BitFlagSet &Negate() { ams::util::impl::NegateImpl(this->_storage); this->Truncate(); return *this; }
consteval static int GetCount() { return static_cast(N); }
constexpr ALWAYS_INLINE bool IsAnySet() const { return ams::util::impl::IsAnySet(this->_storage); }
constexpr ALWAYS_INLINE int PopCount() const { return ams::util::impl::PopCount(this->_storage); }
constexpr ALWAYS_INLINE bool IsAllSet() const { return this->PopCount() == this->GetCount(); }
constexpr ALWAYS_INLINE bool IsAllOff() const { return !this->IsAnySet(); }
constexpr ALWAYS_INLINE bool operator[](s32 idx) const { return this->Test(idx); }
constexpr ALWAYS_INLINE Reference operator[](s32 idx) { return Reference(*this, idx); }
constexpr ALWAYS_INLINE bool operator==(const BitFlagSet &rhs) const { return ams::util::impl::IsEqual(this->_storage, rhs._storage); }
constexpr ALWAYS_INLINE bool operator!=(const BitFlagSet &rhs) const { return !(*this == rhs); }
constexpr ALWAYS_INLINE BitFlagSet operator~() const { BitFlagSet tmp = *this; return tmp.Negate(); }
constexpr ALWAYS_INLINE BitFlagSet operator&(const BitFlagSet &rhs) const { BitFlagSet v = *this; v &= rhs; return v; }
constexpr ALWAYS_INLINE BitFlagSet operator^(const BitFlagSet &rhs) const { BitFlagSet v = *this; v ^= rhs; return v; }
constexpr ALWAYS_INLINE BitFlagSet operator|(const BitFlagSet &rhs) const { BitFlagSet v = *this; v |= rhs; return v; }
constexpr ALWAYS_INLINE BitFlagSet &operator&=(const BitFlagSet &rhs) { ams::util::impl::AndImpl(this->_storage, rhs._storage); return *this; }
constexpr ALWAYS_INLINE BitFlagSet &operator^=(const BitFlagSet &rhs) { ams::util::impl::XorImpl(this->_storage, rhs._storage); return *this; }
constexpr ALWAYS_INLINE BitFlagSet &operator|=(const BitFlagSet &rhs) { ams::util::impl::OrImpl(this->_storage, rhs._storage); return *this; }
};
template
template
constexpr inline const BitFlagSet BitFlagSet::Flag::Mask = { { static_cast::StorageCount>::Type>(1) << Index } };
template
constexpr BitFlagSet MakeBitFlagSet() { return BitFlagSet{}; }
template
constexpr BitFlagSet MakeBitFlagSet() { return MakeBitFlagSet(); }
}