123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399 |
- /* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 4 -*-
- * 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 GCHashTable_h
- #define GCHashTable_h
- #include "js/GCPolicyAPI.h"
- #include "js/HashTable.h"
- #include "js/RootingAPI.h"
- #include "js/SweepingAPI.h"
- #include "js/TracingAPI.h"
- namespace JS {
- // Define a reasonable default GC policy for GC-aware Maps.
- template <typename Key, typename Value>
- struct DefaultMapSweepPolicy {
- static bool needsSweep(Key* key, Value* value) {
- return GCPolicy<Key>::needsSweep(key) || GCPolicy<Value>::needsSweep(value);
- }
- };
- // A GCHashMap is a GC-aware HashMap, meaning that it has additional trace and
- // sweep methods that know how to visit all keys and values in the table.
- // HashMaps that contain GC pointers will generally want to use this GCHashMap
- // specialization instead of HashMap, because this conveniently supports tracing
- // keys and values, and cleaning up weak entries.
- //
- // GCHashMap::trace applies GCPolicy<T>::trace to each entry's key and value.
- // Most types of GC pointers already have appropriate specializations of
- // GCPolicy, so they should just work as keys and values. Any struct type with a
- // default constructor and trace and sweep functions should work as well. If you
- // need to define your own GCPolicy specialization, generic helpers can be found
- // in js/public/TracingAPI.h.
- //
- // The MapSweepPolicy template parameter controls how the table drops entries
- // when swept. GCHashMap::sweep applies MapSweepPolicy::needsSweep to each table
- // entry; if it returns true, the entry is dropped. The default MapSweepPolicy
- // drops the entry if either the key or value is about to be finalized,
- // according to its GCPolicy<T>::needsSweep method. (This default is almost
- // always fine: it's hard to imagine keeping such an entry around anyway.)
- //
- // Note that this HashMap only knows *how* to trace and sweep, but it does not
- // itself cause tracing or sweeping to be invoked. For tracing, it must be used
- // with Rooted or PersistentRooted, or barriered and traced manually. For
- // sweeping, currently it requires an explicit call to <map>.sweep().
- template <typename Key,
- typename Value,
- typename HashPolicy = js::DefaultHasher<Key>,
- typename AllocPolicy = js::TempAllocPolicy,
- typename MapSweepPolicy = DefaultMapSweepPolicy<Key, Value>>
- class GCHashMap : public js::HashMap<Key, Value, HashPolicy, AllocPolicy>
- {
- using Base = js::HashMap<Key, Value, HashPolicy, AllocPolicy>;
- public:
- explicit GCHashMap(AllocPolicy a = AllocPolicy()) : Base(a) {}
- static void trace(GCHashMap* map, JSTracer* trc) { map->trace(trc); }
- void trace(JSTracer* trc) {
- if (!this->initialized())
- return;
- for (typename Base::Enum e(*this); !e.empty(); e.popFront()) {
- GCPolicy<Value>::trace(trc, &e.front().value(), "hashmap value");
- GCPolicy<Key>::trace(trc, &e.front().mutableKey(), "hashmap key");
- }
- }
- void sweep() {
- if (!this->initialized())
- return;
- for (typename Base::Enum e(*this); !e.empty(); e.popFront()) {
- if (MapSweepPolicy::needsSweep(&e.front().mutableKey(), &e.front().value()))
- e.removeFront();
- }
- }
- // GCHashMap is movable
- GCHashMap(GCHashMap&& rhs) : Base(mozilla::Move(rhs)) {}
- void operator=(GCHashMap&& rhs) {
- MOZ_ASSERT(this != &rhs, "self-move assignment is prohibited");
- Base::operator=(mozilla::Move(rhs));
- }
- private:
- // GCHashMap is not copyable or assignable
- GCHashMap(const GCHashMap& hm) = delete;
- GCHashMap& operator=(const GCHashMap& hm) = delete;
- };
- } // namespace JS
- namespace js {
- // HashMap that supports rekeying.
- //
- // If your keys are pointers to something like JSObject that can be tenured or
- // compacted, prefer to use GCHashMap with MovableCellHasher, which takes
- // advantage of the Zone's stable id table to make rekeying unnecessary.
- template <typename Key,
- typename Value,
- typename HashPolicy = DefaultHasher<Key>,
- typename AllocPolicy = TempAllocPolicy,
- typename MapSweepPolicy = JS::DefaultMapSweepPolicy<Key, Value>>
- class GCRekeyableHashMap : public JS::GCHashMap<Key, Value, HashPolicy, AllocPolicy, MapSweepPolicy>
- {
- using Base = JS::GCHashMap<Key, Value, HashPolicy, AllocPolicy>;
- public:
- explicit GCRekeyableHashMap(AllocPolicy a = AllocPolicy()) : Base(a) {}
- void sweep() {
- if (!this->initialized())
- return;
- for (typename Base::Enum e(*this); !e.empty(); e.popFront()) {
- Key key(e.front().key());
- if (MapSweepPolicy::needsSweep(&key, &e.front().value()))
- e.removeFront();
- else if (!HashPolicy::match(key, e.front().key()))
- e.rekeyFront(key);
- }
- }
- // GCRekeyableHashMap is movable
- GCRekeyableHashMap(GCRekeyableHashMap&& rhs) : Base(mozilla::Move(rhs)) {}
- void operator=(GCRekeyableHashMap&& rhs) {
- MOZ_ASSERT(this != &rhs, "self-move assignment is prohibited");
- Base::operator=(mozilla::Move(rhs));
- }
- };
- template <typename Outer, typename... Args>
- class GCHashMapOperations
- {
- using Map = JS::GCHashMap<Args...>;
- using Lookup = typename Map::Lookup;
- const Map& map() const { return static_cast<const Outer*>(this)->get(); }
- public:
- using AddPtr = typename Map::AddPtr;
- using Ptr = typename Map::Ptr;
- using Range = typename Map::Range;
- bool initialized() const { return map().initialized(); }
- Ptr lookup(const Lookup& l) const { return map().lookup(l); }
- AddPtr lookupForAdd(const Lookup& l) const { return map().lookupForAdd(l); }
- Range all() const { return map().all(); }
- bool empty() const { return map().empty(); }
- uint32_t count() const { return map().count(); }
- size_t capacity() const { return map().capacity(); }
- bool has(const Lookup& l) const { return map().lookup(l).found(); }
- size_t sizeOfExcludingThis(mozilla::MallocSizeOf mallocSizeOf) const {
- return map().sizeOfExcludingThis(mallocSizeOf);
- }
- size_t sizeOfIncludingThis(mozilla::MallocSizeOf mallocSizeOf) const {
- return mallocSizeOf(this) + map().sizeOfExcludingThis(mallocSizeOf);
- }
- };
- template <typename Outer, typename... Args>
- class MutableGCHashMapOperations
- : public GCHashMapOperations<Outer, Args...>
- {
- using Map = JS::GCHashMap<Args...>;
- using Lookup = typename Map::Lookup;
- Map& map() { return static_cast<Outer*>(this)->get(); }
- public:
- using AddPtr = typename Map::AddPtr;
- struct Enum : public Map::Enum { explicit Enum(Outer& o) : Map::Enum(o.map()) {} };
- using Ptr = typename Map::Ptr;
- using Range = typename Map::Range;
- bool init(uint32_t len = 16) { return map().init(len); }
- void clear() { map().clear(); }
- void finish() { map().finish(); }
- void remove(Ptr p) { map().remove(p); }
- template<typename KeyInput, typename ValueInput>
- bool add(AddPtr& p, KeyInput&& k, ValueInput&& v) {
- return map().add(p, mozilla::Forward<KeyInput>(k), mozilla::Forward<ValueInput>(v));
- }
- template<typename KeyInput>
- bool add(AddPtr& p, KeyInput&& k) {
- return map().add(p, mozilla::Forward<KeyInput>(k), Map::Value());
- }
- template<typename KeyInput, typename ValueInput>
- bool relookupOrAdd(AddPtr& p, KeyInput&& k, ValueInput&& v) {
- return map().relookupOrAdd(p, k,
- mozilla::Forward<KeyInput>(k),
- mozilla::Forward<ValueInput>(v));
- }
- template<typename KeyInput, typename ValueInput>
- bool put(KeyInput&& k, ValueInput&& v) {
- return map().put(mozilla::Forward<KeyInput>(k), mozilla::Forward<ValueInput>(v));
- }
- template<typename KeyInput, typename ValueInput>
- bool putNew(KeyInput&& k, ValueInput&& v) {
- return map().putNew(mozilla::Forward<KeyInput>(k), mozilla::Forward<ValueInput>(v));
- }
- };
- template <typename A, typename B, typename C, typename D, typename E>
- class RootedBase<JS::GCHashMap<A,B,C,D,E>>
- : public MutableGCHashMapOperations<JS::Rooted<JS::GCHashMap<A,B,C,D,E>>, A,B,C,D,E>
- {};
- template <typename A, typename B, typename C, typename D, typename E>
- class MutableHandleBase<JS::GCHashMap<A,B,C,D,E>>
- : public MutableGCHashMapOperations<JS::MutableHandle<JS::GCHashMap<A,B,C,D,E>>, A,B,C,D,E>
- {};
- template <typename A, typename B, typename C, typename D, typename E>
- class HandleBase<JS::GCHashMap<A,B,C,D,E>>
- : public GCHashMapOperations<JS::Handle<JS::GCHashMap<A,B,C,D,E>>, A,B,C,D,E>
- {};
- template <typename A, typename B, typename C, typename D, typename E>
- class WeakCacheBase<JS::GCHashMap<A,B,C,D,E>>
- : public MutableGCHashMapOperations<JS::WeakCache<JS::GCHashMap<A,B,C,D,E>>, A,B,C,D,E>
- {};
- } // namespace js
- namespace JS {
- // A GCHashSet is a HashSet with an additional trace method that knows
- // be traced to be kept alive will generally want to use this GCHashSet
- // specialization in lieu of HashSet.
- //
- // Most types of GC pointers can be traced with no extra infrastructure. For
- // structs and non-gc-pointer members, ensure that there is a specialization of
- // GCPolicy<T> with an appropriate trace method available to handle the custom
- // type. Generic helpers can be found in js/public/TracingAPI.h.
- //
- // Note that although this HashSet's trace will deal correctly with moved
- // elements, it does not itself know when to barrier or trace elements. To
- // function properly it must either be used with Rooted or barriered and traced
- // manually.
- template <typename T,
- typename HashPolicy = js::DefaultHasher<T>,
- typename AllocPolicy = js::TempAllocPolicy>
- class GCHashSet : public js::HashSet<T, HashPolicy, AllocPolicy>
- {
- using Base = js::HashSet<T, HashPolicy, AllocPolicy>;
- public:
- explicit GCHashSet(AllocPolicy a = AllocPolicy()) : Base(a) {}
- static void trace(GCHashSet* set, JSTracer* trc) { set->trace(trc); }
- void trace(JSTracer* trc) {
- if (!this->initialized())
- return;
- for (typename Base::Enum e(*this); !e.empty(); e.popFront())
- GCPolicy<T>::trace(trc, &e.mutableFront(), "hashset element");
- }
- void sweep() {
- if (!this->initialized())
- return;
- for (typename Base::Enum e(*this); !e.empty(); e.popFront()) {
- if (GCPolicy<T>::needsSweep(&e.mutableFront()))
- e.removeFront();
- }
- }
- // GCHashSet is movable
- GCHashSet(GCHashSet&& rhs) : Base(mozilla::Move(rhs)) {}
- void operator=(GCHashSet&& rhs) {
- MOZ_ASSERT(this != &rhs, "self-move assignment is prohibited");
- Base::operator=(mozilla::Move(rhs));
- }
- private:
- // GCHashSet is not copyable or assignable
- GCHashSet(const GCHashSet& hs) = delete;
- GCHashSet& operator=(const GCHashSet& hs) = delete;
- };
- } // namespace JS
- namespace js {
- template <typename Outer, typename... Args>
- class GCHashSetOperations
- {
- using Set = JS::GCHashSet<Args...>;
- using Lookup = typename Set::Lookup;
- const Set& set() const { return static_cast<const Outer*>(this)->get(); }
- public:
- using AddPtr = typename Set::AddPtr;
- using Entry = typename Set::Entry;
- using Ptr = typename Set::Ptr;
- using Range = typename Set::Range;
- bool initialized() const { return set().initialized(); }
- Ptr lookup(const Lookup& l) const { return set().lookup(l); }
- AddPtr lookupForAdd(const Lookup& l) const { return set().lookupForAdd(l); }
- Range all() const { return set().all(); }
- bool empty() const { return set().empty(); }
- uint32_t count() const { return set().count(); }
- size_t capacity() const { return set().capacity(); }
- bool has(const Lookup& l) const { return set().lookup(l).found(); }
- size_t sizeOfExcludingThis(mozilla::MallocSizeOf mallocSizeOf) const {
- return set().sizeOfExcludingThis(mallocSizeOf);
- }
- size_t sizeOfIncludingThis(mozilla::MallocSizeOf mallocSizeOf) const {
- return mallocSizeOf(this) + set().sizeOfExcludingThis(mallocSizeOf);
- }
- };
- template <typename Outer, typename... Args>
- class MutableGCHashSetOperations
- : public GCHashSetOperations<Outer, Args...>
- {
- using Set = JS::GCHashSet<Args...>;
- using Lookup = typename Set::Lookup;
- Set& set() { return static_cast<Outer*>(this)->get(); }
- public:
- using AddPtr = typename Set::AddPtr;
- using Entry = typename Set::Entry;
- struct Enum : public Set::Enum { explicit Enum(Outer& o) : Set::Enum(o.set()) {} };
- using Ptr = typename Set::Ptr;
- using Range = typename Set::Range;
- bool init(uint32_t len = 16) { return set().init(len); }
- void clear() { set().clear(); }
- void finish() { set().finish(); }
- void remove(Ptr p) { set().remove(p); }
- void remove(const Lookup& l) { set().remove(l); }
- template<typename TInput>
- bool add(AddPtr& p, TInput&& t) {
- return set().add(p, mozilla::Forward<TInput>(t));
- }
- template<typename TInput>
- bool relookupOrAdd(AddPtr& p, const Lookup& l, TInput&& t) {
- return set().relookupOrAdd(p, l, mozilla::Forward<TInput>(t));
- }
- template<typename TInput>
- bool put(TInput&& t) {
- return set().put(mozilla::Forward<TInput>(t));
- }
- template<typename TInput>
- bool putNew(TInput&& t) {
- return set().putNew(mozilla::Forward<TInput>(t));
- }
- template<typename TInput>
- bool putNew(const Lookup& l, TInput&& t) {
- return set().putNew(l, mozilla::Forward<TInput>(t));
- }
- };
- template <typename T, typename HP, typename AP>
- class RootedBase<JS::GCHashSet<T, HP, AP>>
- : public MutableGCHashSetOperations<JS::Rooted<JS::GCHashSet<T, HP, AP>>, T, HP, AP>
- {
- };
- template <typename T, typename HP, typename AP>
- class MutableHandleBase<JS::GCHashSet<T, HP, AP>>
- : public MutableGCHashSetOperations<JS::MutableHandle<JS::GCHashSet<T, HP, AP>>, T, HP, AP>
- {
- };
- template <typename T, typename HP, typename AP>
- class HandleBase<JS::GCHashSet<T, HP, AP>>
- : public GCHashSetOperations<JS::Handle<JS::GCHashSet<T, HP, AP>>, T, HP, AP>
- {
- };
- template <typename T, typename HP, typename AP>
- class WeakCacheBase<JS::GCHashSet<T, HP, AP>>
- : public MutableGCHashSetOperations<JS::WeakCache<JS::GCHashSet<T, HP, AP>>, T, HP, AP>
- {
- };
- } /* namespace js */
- #endif /* GCHashTable_h */
|