#ifndef wasm_support_small_set_h
#define wasm_support_small_set_h
#include <algorithm>
#include <array>
#include <cassert>
#include <set>
#include <unordered_set>
#include "utilities.h"
namespace wasm {
template<typename T, size_t N> struct FixedStorageBase {
size_t used = 0;
std::array<T, N> storage;
enum InsertResult {
NoError,
CouldNotInsert
};
};
template<typename T, size_t N>
struct UnorderedFixedStorage : public FixedStorageBase<T, N> {
using InsertResult = typename FixedStorageBase<T, N>::InsertResult;
InsertResult insert(const T& x) {
for (size_t i = 0; i < this->used; i++) {
if (this->storage[i] == x) {
return InsertResult::NoError;
}
}
assert(this->used <= N);
if (this->used == N) {
return InsertResult::CouldNotInsert;
}
this->storage[this->used++] = x;
return InsertResult::NoError;
}
void erase(const T& x) {
for (size_t i = 0; i < this->used; i++) {
if (this->storage[i] == x) {
this->used--;
this->storage[i] = this->storage[this->used];
return;
}
}
}
};
template<typename T, size_t N>
struct OrderedFixedStorage : public FixedStorageBase<T, N> {
using InsertResult = typename FixedStorageBase<T, N>::InsertResult;
InsertResult insert(const T& x) {
size_t i = 0;
while (i < this->used && this->storage[i] < x) {
i++;
}
if (i < this->used && this->storage[i] == x) {
return InsertResult::NoError;
}
assert(this->used <= N);
if (this->used == N) {
return InsertResult::CouldNotInsert;
}
if (i != this->used) {
for (size_t j = this->used; j >= i + 1; j--) {
this->storage[j] = this->storage[j - 1];
}
}
this->storage[i] = x;
this->used++;
return InsertResult::NoError;
}
void erase(const T& x) {
for (size_t i = 0; i < this->used; i++) {
if (this->storage[i] == x) {
for (size_t j = i + 1; j < this->used; j++) {
this->storage[j - 1] = this->storage[j];
}
this->used--;
return;
}
}
}
};
template<typename T, size_t N, typename FixedStorage, typename FlexibleSet>
class SmallSetBase {
FixedStorage fixed;
FlexibleSet flexible;
bool usingFixed() const {
return flexible.empty();
}
public:
using value_type = T;
using key_type = T;
using reference = T&;
using const_reference = const T&;
using set_type = FlexibleSet;
using size_type = size_t;
SmallSetBase() {}
SmallSetBase(std::initializer_list<T> init) {
for (T item : init) {
insert(item);
}
}
void insert(const T& x) {
if (usingFixed()) {
if (fixed.insert(x) == FixedStorage::InsertResult::CouldNotInsert) {
assert(fixed.used == N);
assert(flexible.empty());
flexible.insert(fixed.storage.begin(),
fixed.storage.begin() + fixed.used);
flexible.insert(x);
assert(!usingFixed());
fixed.used = 0;
}
} else {
flexible.insert(x);
}
}
void erase(const T& x) {
if (usingFixed()) {
fixed.erase(x);
} else {
flexible.erase(x);
}
}
size_t count(const T& x) const {
if (usingFixed()) {
for (size_t i = 0; i < fixed.used; i++) {
if (fixed.storage[i] == x) {
return 1;
}
}
return 0;
} else {
return flexible.count(x);
}
}
size_t size() const {
if (usingFixed()) {
return fixed.used;
} else {
return flexible.size();
}
}
bool empty() const { return size() == 0; }
void clear() {
fixed.used = 0;
flexible.clear();
}
bool
operator==(const SmallSetBase<T, N, FixedStorage, FlexibleSet>& other) const {
if (size() != other.size()) {
return false;
}
if (usingFixed()) {
return std::all_of(fixed.storage.begin(),
fixed.storage.begin() + fixed.used,
[&other](const T& x) { return other.count(x); });
} else if (other.usingFixed()) {
return std::all_of(other.fixed.storage.begin(),
other.fixed.storage.begin() + other.fixed.used,
[this](const T& x) { return count(x); });
} else {
return flexible == other.flexible;
}
}
bool
operator!=(const SmallSetBase<T, N, FixedStorage, FlexibleSet>& other) const {
return !(*this == other);
}
template<typename Parent, typename FlexibleIterator> struct IteratorBase {
using iterator_category = std::forward_iterator_tag;
using difference_type = long;
using value_type = T;
using pointer = const value_type*;
using reference = const value_type&;
const Parent* parent;
using Iterator = IteratorBase<Parent, FlexibleIterator>;
bool usingFixed;
size_t fixedIndex;
FlexibleIterator flexibleIterator;
IteratorBase(const Parent* parent)
: parent(parent), usingFixed(parent->usingFixed()) {}
void setBegin() {
if (usingFixed) {
fixedIndex = 0;
} else {
flexibleIterator = parent->flexible.begin();
}
}
void setEnd() {
if (usingFixed) {
fixedIndex = parent->size();
} else {
flexibleIterator = parent->flexible.end();
}
}
bool operator==(const Iterator& other) const {
if (parent != other.parent) {
return false;
}
if (usingFixed != other.usingFixed) {
Fatal() << "SmallSet does not support changes while iterating";
}
if (usingFixed) {
return fixedIndex == other.fixedIndex;
} else {
return flexibleIterator == other.flexibleIterator;
}
}
bool operator!=(const Iterator& other) const { return !(*this == other); }
Iterator& operator++() {
if (usingFixed) {
fixedIndex++;
} else {
flexibleIterator++;
}
return *this;
}
const value_type& operator*() const {
if (this->usingFixed) {
return this->parent->fixed.storage[this->fixedIndex];
} else {
return *this->flexibleIterator;
}
}
};
using Iterator = IteratorBase<SmallSetBase<T, N, FixedStorage, FlexibleSet>,
typename FlexibleSet::const_iterator>;
Iterator begin() {
auto ret = Iterator(this);
ret.setBegin();
return ret;
}
Iterator end() {
auto ret = Iterator(this);
ret.setEnd();
return ret;
}
Iterator begin() const {
auto ret = Iterator(this);
ret.setBegin();
return ret;
}
Iterator end() const {
auto ret = Iterator(this);
ret.setEnd();
return ret;
}
using iterator = Iterator;
using const_iterator = Iterator;
bool TEST_ONLY_NEVER_USE_usingFixed() { return usingFixed(); }
};
template<typename T, size_t N>
class SmallSet
: public SmallSetBase<T, N, OrderedFixedStorage<T, N>, std::set<T>> {};
template<typename T, size_t N>
class SmallUnorderedSet : public SmallSetBase<T,
N,
UnorderedFixedStorage<T, N>,
std::unordered_set<T>> {};
}
#endif