#pragma once #include #include template struct FixedHashMapCell { using Mapped = TMapped; using State = TState; using value_type = PairNoInit; using mapped_type = TMapped; bool full; Mapped mapped; FixedHashMapCell() {} /// NOLINT FixedHashMapCell(const Key &, const State &) : full(true) {} FixedHashMapCell(const value_type & value_, const State &) : full(true), mapped(value_.second) {} const VoidKey getKey() const { return {}; } /// NOLINT Mapped & getMapped() { return mapped; } const Mapped & getMapped() const { return mapped; } bool isZero(const State &) const { return !full; } void setZero() { full = false; } /// Similar to FixedHashSetCell except that we need to contain a pointer to the Mapped field. /// Note that we have to assemble a continuous layout for the value_type on each call of getValue(). struct CellExt { CellExt() {} /// NOLINT CellExt(Key && key_, const FixedHashMapCell * ptr_) : key(key_), ptr(const_cast(ptr_)) {} void update(Key && key_, const FixedHashMapCell * ptr_) { key = key_; ptr = const_cast(ptr_); } Key key; FixedHashMapCell * ptr; const Key & getKey() const { return key; } Mapped & getMapped() { return ptr->mapped; } const Mapped & getMapped() const { return ptr->mapped; } const value_type getValue() const { return {key, ptr->mapped}; } /// NOLINT }; }; /// In case when we can encode empty cells with zero mapped values. template struct FixedHashMapImplicitZeroCell { using Mapped = TMapped; using State = TState; using value_type = PairNoInit; using mapped_type = TMapped; Mapped mapped; FixedHashMapImplicitZeroCell() {} /// NOLINT FixedHashMapImplicitZeroCell(const Key &, const State &) {} FixedHashMapImplicitZeroCell(const value_type & value_, const State &) : mapped(value_.second) {} const VoidKey getKey() const { return {}; } /// NOLINT Mapped & getMapped() { return mapped; } const Mapped & getMapped() const { return mapped; } bool isZero(const State &) const { return !mapped; } void setZero() { mapped = {}; } /// Similar to FixedHashSetCell except that we need to contain a pointer to the Mapped field. /// Note that we have to assemble a continuous layout for the value_type on each call of getValue(). struct CellExt { CellExt() {} /// NOLINT CellExt(Key && key_, const FixedHashMapImplicitZeroCell * ptr_) : key(key_), ptr(const_cast(ptr_)) {} void update(Key && key_, const FixedHashMapImplicitZeroCell * ptr_) { key = key_; ptr = const_cast(ptr_); } Key key; FixedHashMapImplicitZeroCell * ptr; const Key & getKey() const { return key; } Mapped & getMapped() { return ptr->mapped; } const Mapped & getMapped() const { return ptr->mapped; } const value_type getValue() const { return {key, ptr->mapped}; } /// NOLINT }; }; template < typename Key, typename Mapped, typename Cell = FixedHashMapCell, typename Size = FixedHashTableStoredSize, typename Allocator = HashTableAllocator> class FixedHashMap : public FixedHashTable { public: using Base = FixedHashTable; using Self = FixedHashMap; using LookupResult = typename Base::LookupResult; using Base::Base; FixedHashMap() = default; FixedHashMap(size_t ) {} /// NOLINT template void ALWAYS_INLINE mergeToViaEmplace(Self & that, Func && func) { for (auto it = this->begin(), end = this->end(); it != end; ++it) { typename Self::LookupResult res_it; bool inserted; that.emplace(it->getKey(), res_it, inserted, it.getHash()); func(res_it->getMapped(), it->getMapped(), inserted); } } template void ALWAYS_INLINE mergeToViaFind(Self & that, Func && func) { for (auto it = this->begin(), end = this->end(); it != end; ++it) { auto res_it = that.find(it->getKey(), it.getHash()); if (!res_it) func(it->getMapped(), it->getMapped(), false); else func(res_it->getMapped(), it->getMapped(), true); } } template void forEachValue(Func && func) { for (auto & v : *this) func(v.getKey(), v.getMapped()); } template void forEachMapped(Func && func) { for (auto & v : *this) func(v.getMapped()); } Mapped & ALWAYS_INLINE operator[](const Key & x) { LookupResult it; bool inserted; this->emplace(x, it, inserted); if (inserted) new (&it->getMapped()) Mapped(); return it->getMapped(); } }; template using FixedImplicitZeroHashMap = FixedHashMap< Key, Mapped, FixedHashMapImplicitZeroCell, FixedHashTableStoredSize>, Allocator>; template using FixedImplicitZeroHashMapWithCalculatedSize = FixedHashMap< Key, Mapped, FixedHashMapImplicitZeroCell, FixedHashTableCalculatedSize>, Allocator>;