Source code

Revision control

Copy as Markdown

Other Tools

/* -*- 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/. */
#ifndef ds_InlineTable_h
#define ds_InlineTable_h
#include "mozilla/Maybe.h"
#include "mozilla/Variant.h"
#include <utility>
#include "js/AllocPolicy.h"
#include "js/HashTable.h"
namespace js {
namespace detail {
template <typename InlineEntry, typename Entry, typename Table,
typename HashPolicy, typename AllocPolicy, size_t InlineEntries>
class InlineTable : private AllocPolicy {
private:
using TablePtr = typename Table::Ptr;
using TableAddPtr = typename Table::AddPtr;
using TableRange = typename Table::Range;
using Lookup = typename HashPolicy::Lookup;
struct InlineArray {
uint32_t count = 0;
InlineEntry inl[InlineEntries];
};
mozilla::Variant<InlineArray, Table> data_{InlineArray()};
#ifdef DEBUG
// Used to check that entries aren't added/removed while using Ptr/AddPtr or
// Range. Similar to HashTable::mMutationCount.
uint64_t mutationCount_ = 0;
#endif
#ifndef DEBUG
// If this assertion fails, you should probably increase InlineEntries because
// an extra inline entry could likely be added "for free".
static_assert(sizeof(InlineArray) + sizeof(InlineEntry) >= sizeof(Table),
"Space for additional inline elements in InlineTable?");
#endif
InlineArray& inlineArray() { return data_.template as<InlineArray>(); }
const InlineArray& inlineArray() const {
return data_.template as<InlineArray>();
}
Table& table() { return data_.template as<Table>(); }
const Table& table() const { return data_.template as<Table>(); }
InlineEntry* inlineStart() {
MOZ_ASSERT(!usingTable());
return inlineArray().inl;
}
const InlineEntry* inlineStart() const {
MOZ_ASSERT(!usingTable());
return inlineArray().inl;
}
InlineEntry* inlineEnd() {
MOZ_ASSERT(!usingTable());
return inlineArray().inl + inlineArray().count;
}
const InlineEntry* inlineEnd() const {
MOZ_ASSERT(!usingTable());
return inlineArray().inl + inlineArray().count;
}
bool usingTable() const { return data_.template is<Table>(); }
void bumpMutationCount() {
#ifdef DEBUG
mutationCount_++;
#endif
}
[[nodiscard]] bool switchToTable() {
MOZ_ASSERT(inlineArray().count == InlineEntries);
Table table(*static_cast<AllocPolicy*>(this));
// This is called before adding the next element, so reserve space for it
// too.
if (!table.reserve(InlineEntries + 1)) {
return false;
}
InlineEntry* end = inlineStart() + InlineEntries;
for (InlineEntry* it = inlineStart(); it != end; ++it) {
// Note: don't use putNewInfallible because hashing can be fallible too.
if (!it->moveTo(table)) {
return false;
}
}
MOZ_ASSERT(table.count() == InlineEntries);
data_.template emplace<Table>(std::move(table));
MOZ_ASSERT(usingTable());
bumpMutationCount();
return true;
}
public:
static const size_t SizeOfInlineEntries = sizeof(InlineEntry) * InlineEntries;
explicit InlineTable(AllocPolicy a = AllocPolicy())
: AllocPolicy(std::move(a)) {}
class Ptr {
friend class InlineTable;
MOZ_INIT_OUTSIDE_CTOR Entry entry_;
MOZ_INIT_OUTSIDE_CTOR TablePtr tablePtr_;
MOZ_INIT_OUTSIDE_CTOR InlineEntry* inlPtr_;
MOZ_INIT_OUTSIDE_CTOR bool isInlinePtr_;
#ifdef DEBUG
uint64_t mutationCount_ = 0xbadbad;
#endif
Ptr(const InlineTable& table, TablePtr p)
: entry_(p.found() ? &*p : nullptr), tablePtr_(p), isInlinePtr_(false) {
#ifdef DEBUG
mutationCount_ = table.mutationCount_;
#endif
}
Ptr(const InlineTable& table, InlineEntry* inlineEntry)
: entry_(inlineEntry), inlPtr_(inlineEntry), isInlinePtr_(true) {
#ifdef DEBUG
mutationCount_ = table.mutationCount_;
#endif
}
public:
// Leaves Ptr uninitialized.
Ptr() {
#ifdef DEBUG
inlPtr_ = (InlineEntry*)0xbad;
isInlinePtr_ = true;
#endif
}
// Default copy constructor works for this structure.
bool found() const {
return isInlinePtr_ ? bool(inlPtr_) : tablePtr_.found();
}
explicit operator bool() const { return found(); }
Entry& operator*() {
MOZ_ASSERT(found());
return entry_;
}
Entry* operator->() {
MOZ_ASSERT(found());
return &entry_;
}
};
class AddPtr {
friend class InlineTable;
MOZ_INIT_OUTSIDE_CTOR Entry entry_;
MOZ_INIT_OUTSIDE_CTOR TableAddPtr tableAddPtr_;
MOZ_INIT_OUTSIDE_CTOR InlineEntry* inlAddPtr_;
MOZ_INIT_OUTSIDE_CTOR bool isInlinePtr_;
// Indicates whether inlAddPtr is a found result or an add pointer.
MOZ_INIT_OUTSIDE_CTOR bool inlPtrFound_;
#ifdef DEBUG
uint64_t mutationCount_ = 0xbadbad;
#endif
AddPtr(const InlineTable& table, InlineEntry* ptr, bool found)
: entry_(ptr),
inlAddPtr_(ptr),
isInlinePtr_(true),
inlPtrFound_(found) {
#ifdef DEBUG
mutationCount_ = table.mutationCount_;
#endif
}
AddPtr(const InlineTable& table, const TableAddPtr& p)
: entry_(p.found() ? &*p : nullptr),
tableAddPtr_(p),
isInlinePtr_(false) {
#ifdef DEBUG
mutationCount_ = table.mutationCount_;
#endif
}
public:
AddPtr() = default;
bool found() const {
return isInlinePtr_ ? inlPtrFound_ : tableAddPtr_.found();
}
explicit operator bool() const { return found(); }
Entry& operator*() {
MOZ_ASSERT(found());
return entry_;
}
Entry* operator->() {
MOZ_ASSERT(found());
return &entry_;
}
};
size_t count() const {
return usingTable() ? table().count() : inlineArray().count;
}
bool empty() const {
return usingTable() ? table().empty() : !inlineArray().count;
}
void clear() {
data_.template emplace<InlineArray>();
bumpMutationCount();
}
MOZ_ALWAYS_INLINE
Ptr lookup(const Lookup& l) {
if (usingTable()) {
return Ptr(*this, table().lookup(l));
}
InlineEntry* end = inlineEnd();
for (InlineEntry* it = inlineStart(); it != end; ++it) {
if (HashPolicy::match(it->key, l)) {
return Ptr(*this, it);
}
}
return Ptr(*this, nullptr);
}
MOZ_ALWAYS_INLINE
AddPtr lookupForAdd(const Lookup& l) {
if (usingTable()) {
return AddPtr(*this, table().lookupForAdd(l));
}
InlineEntry* end = inlineEnd();
for (InlineEntry* it = inlineStart(); it != end; ++it) {
if (HashPolicy::match(it->key, l)) {
return AddPtr(*this, it, true);
}
}
// The add pointer that's returned here may indicate the limit entry of
// the linear space, in which case the |add| operation will initialize
// the table if necessary and add the entry there.
return AddPtr(*this, inlineEnd(), false);
}
template <typename KeyInput, typename... Args>
[[nodiscard]] MOZ_ALWAYS_INLINE bool add(AddPtr& p, KeyInput&& key,
Args&&... args) {
MOZ_ASSERT(!p);
MOZ_ASSERT(p.mutationCount_ == mutationCount_);
if (p.isInlinePtr_) {
InlineEntry* addPtr = p.inlAddPtr_;
MOZ_ASSERT(addPtr == inlineEnd());
// Switching to table mode before we add this pointer.
if (addPtr == inlineStart() + InlineEntries) {
if (!switchToTable()) {
return false;
}
if (!table().putNew(std::forward<KeyInput>(key),
std::forward<Args>(args)...)) {
return false;
}
bumpMutationCount();
return true;
}
MOZ_ASSERT(!p.found());
MOZ_ASSERT(uintptr_t(inlineEnd()) == uintptr_t(p.inlAddPtr_));
if (!this->checkSimulatedOOM()) {
return false;
}
addPtr->update(std::forward<KeyInput>(key), std::forward<Args>(args)...);
inlineArray().count++;
bumpMutationCount();
return true;
}
if (!table().add(p.tableAddPtr_, std::forward<KeyInput>(key),
std::forward<Args>(args)...)) {
return false;
}
bumpMutationCount();
return true;
}
void remove(Ptr& p) {
MOZ_ASSERT(p);
MOZ_ASSERT(p.mutationCount_ == mutationCount_);
if (p.isInlinePtr_) {
InlineArray& arr = inlineArray();
MOZ_ASSERT(arr.count > 0);
InlineEntry* last = &arr.inl[arr.count - 1];
MOZ_ASSERT(p.inlPtr_ <= last);
if (p.inlPtr_ != last) {
// Removing an entry that's not the last one. Move the last entry.
*p.inlPtr_ = std::move(*last);
}
arr.count--;
} else {
MOZ_ASSERT(usingTable());
table().remove(p.tablePtr_);
}
bumpMutationCount();
}
void remove(const Lookup& l) {
if (Ptr p = lookup(l)) {
remove(p);
}
}
class Range {
friend class InlineTable;
mozilla::Maybe<TableRange> tableRange_; // `Nothing` if `isInline_==true`
InlineEntry* cur_;
InlineEntry* end_;
bool isInline_;
#ifdef DEBUG
const InlineTable* table_ = nullptr;
uint64_t mutationCount_ = 0xbadbad;
#endif
Range(const InlineTable& table, TableRange r)
: tableRange_(mozilla::Some(r)),
cur_(nullptr),
end_(nullptr),
isInline_(false) {
MOZ_ASSERT(!isInlineRange());
#ifdef DEBUG
table_ = &table;
mutationCount_ = table.mutationCount_;
#endif
}
Range(const InlineTable& table, const InlineEntry* begin,
const InlineEntry* end)
: tableRange_(mozilla::Nothing()),
cur_(const_cast<InlineEntry*>(begin)),
end_(const_cast<InlineEntry*>(end)),
isInline_(true) {
MOZ_ASSERT(isInlineRange());
#ifdef DEBUG
table_ = &table;
mutationCount_ = table.mutationCount_;
#endif
}
bool assertInlineRangeInvariants() const {
MOZ_ASSERT(uintptr_t(cur_) <= uintptr_t(end_));
return true;
}
bool isInlineRange() const {
MOZ_ASSERT_IF(isInline_, assertInlineRangeInvariants());
return isInline_;
}
void bumpCurPtr() {
MOZ_ASSERT(isInlineRange());
cur_++;
}
public:
bool empty() const {
MOZ_ASSERT(table_->mutationCount_ == mutationCount_);
return isInlineRange() ? cur_ == end_ : tableRange_->empty();
}
Entry front() {
MOZ_ASSERT(!empty());
MOZ_ASSERT(table_->mutationCount_ == mutationCount_);
if (isInlineRange()) {
return Entry(cur_);
}
return Entry(&tableRange_->front());
}
void popFront() {
MOZ_ASSERT(!empty());
MOZ_ASSERT(table_->mutationCount_ == mutationCount_);
if (isInlineRange()) {
bumpCurPtr();
} else {
tableRange_->popFront();
}
}
};
Range all() const {
return usingTable() ? Range(*this, table().all())
: Range(*this, inlineStart(), inlineEnd());
}
};
} // namespace detail
// A map with InlineEntries number of entries kept inline in an array.
//
// The Value type must have a default constructor.
//
// The API is very much like HashMap's.
template <typename Key, typename Value, size_t InlineEntries,
typename HashPolicy = DefaultHasher<Key>,
typename AllocPolicy = TempAllocPolicy>
class InlineMap {
using Map = HashMap<Key, Value, HashPolicy, AllocPolicy>;
struct InlineEntry {
Key key;
Value value;
template <typename KeyInput, typename ValueInput>
void update(KeyInput&& key, ValueInput&& value) {
this->key = std::forward<KeyInput>(key);
this->value = std::forward<ValueInput>(value);
}
[[nodiscard]] bool moveTo(Map& map) {
return map.putNew(std::move(key), std::move(value));
}
};
class Entry {
using MapEntry = typename Map::Entry;
MapEntry* mapEntry_;
InlineEntry* inlineEntry_;
public:
Entry() = default;
explicit Entry(MapEntry* mapEntry)
: mapEntry_(mapEntry), inlineEntry_(nullptr) {}
explicit Entry(InlineEntry* inlineEntry)
: mapEntry_(nullptr), inlineEntry_(inlineEntry) {}
const Key& key() const {
MOZ_ASSERT(!!mapEntry_ != !!inlineEntry_);
if (mapEntry_) {
return mapEntry_->key();
}
return inlineEntry_->key;
}
Value& value() {
MOZ_ASSERT(!!mapEntry_ != !!inlineEntry_);
if (mapEntry_) {
return mapEntry_->value();
}
return inlineEntry_->value;
}
};
using Impl = detail::InlineTable<InlineEntry, Entry, Map, HashPolicy,
AllocPolicy, InlineEntries>;
Impl impl_;
public:
using Table = Map;
using Ptr = typename Impl::Ptr;
using AddPtr = typename Impl::AddPtr;
using Range = typename Impl::Range;
using Lookup = typename HashPolicy::Lookup;
static const size_t SizeOfInlineEntries = Impl::SizeOfInlineEntries;
explicit InlineMap(AllocPolicy a = AllocPolicy()) : impl_(std::move(a)) {}
size_t count() const { return impl_.count(); }
bool empty() const { return impl_.empty(); }
void clear() { impl_.clear(); }
Range all() const { return impl_.all(); }
MOZ_ALWAYS_INLINE
Ptr lookup(const Lookup& l) { return impl_.lookup(l); }
MOZ_ALWAYS_INLINE
bool has(const Lookup& l) const {
return const_cast<InlineMap*>(this)->lookup(l).found();
}
MOZ_ALWAYS_INLINE
AddPtr lookupForAdd(const Lookup& l) { return impl_.lookupForAdd(l); }
template <typename KeyInput, typename ValueInput>
[[nodiscard]] MOZ_ALWAYS_INLINE bool add(AddPtr& p, KeyInput&& key,
ValueInput&& value) {
return impl_.add(p, std::forward<KeyInput>(key),
std::forward<ValueInput>(value));
}
template <typename KeyInput, typename ValueInput>
[[nodiscard]] bool put(KeyInput&& key, ValueInput&& value) {
AddPtr p = lookupForAdd(key);
if (p) {
p->value() = std::forward<ValueInput>(value);
return true;
}
return add(p, std::forward<KeyInput>(key), std::forward<ValueInput>(value));
}
void remove(Ptr& p) { impl_.remove(p); }
void remove(const Lookup& l) { impl_.remove(l); }
};
// A set with InlineEntries number of entries kept inline in an array.
//
// The T type must have a default constructor.
//
// The API is very much like HashSet's.
template <typename T, size_t InlineEntries,
typename HashPolicy = DefaultHasher<T>,
typename AllocPolicy = TempAllocPolicy>
class InlineSet {
using Set = HashSet<T, HashPolicy, AllocPolicy>;
struct InlineEntry {
T key;
template <typename TInput>
void update(TInput&& key) {
this->key = std::forward<TInput>(key);
}
[[nodiscard]] bool moveTo(Set& set) { return set.putNew(std::move(key)); }
};
class Entry {
using SetEntry = typename Set::Entry;
SetEntry* setEntry_;
InlineEntry* inlineEntry_;
public:
Entry() = default;
explicit Entry(const SetEntry* setEntry)
: setEntry_(const_cast<SetEntry*>(setEntry)), inlineEntry_(nullptr) {}
explicit Entry(InlineEntry* inlineEntry)
: setEntry_(nullptr), inlineEntry_(inlineEntry) {}
operator T() const {
MOZ_ASSERT(!!setEntry_ != !!inlineEntry_);
if (setEntry_) {
return *setEntry_;
}
return inlineEntry_->key;
}
};
using Impl = detail::InlineTable<InlineEntry, Entry, Set, HashPolicy,
AllocPolicy, InlineEntries>;
Impl impl_;
public:
using Table = Set;
using Ptr = typename Impl::Ptr;
using AddPtr = typename Impl::AddPtr;
using Range = typename Impl::Range;
using Lookup = typename HashPolicy::Lookup;
static const size_t SizeOfInlineEntries = Impl::SizeOfInlineEntries;
explicit InlineSet(AllocPolicy a = AllocPolicy()) : impl_(std::move(a)) {}
size_t count() const { return impl_.count(); }
bool empty() const { return impl_.empty(); }
void clear() { impl_.clear(); }
Range all() const { return impl_.all(); }
MOZ_ALWAYS_INLINE
Ptr lookup(const Lookup& l) { return impl_.lookup(l); }
MOZ_ALWAYS_INLINE
bool has(const Lookup& l) const {
return const_cast<InlineSet*>(this)->lookup(l).found();
}
MOZ_ALWAYS_INLINE
AddPtr lookupForAdd(const Lookup& l) { return impl_.lookupForAdd(l); }
template <typename TInput>
[[nodiscard]] MOZ_ALWAYS_INLINE bool add(AddPtr& p, TInput&& key) {
return impl_.add(p, std::forward<TInput>(key));
}
template <typename TInput>
[[nodiscard]] bool put(TInput&& key) {
AddPtr p = lookupForAdd(key);
return p ? true : add(p, std::forward<TInput>(key));
}
void remove(Ptr& p) { impl_.remove(p); }
void remove(const Lookup& l) { impl_.remove(l); }
};
} // namespace js
#endif // ds_InlineTable_h