]> git.saurik.com Git - apple/javascriptcore.git/blobdiff - runtime/WeakGCMap.h
JavaScriptCore-7600.1.4.16.1.tar.gz
[apple/javascriptcore.git] / runtime / WeakGCMap.h
index 39a91c5fa59adb605ae37351eaec1723395b1987..cc0fb8f578be572d39f8c86b917846fbfdd471f5 100644 (file)
 #ifndef WeakGCMap_h
 #define WeakGCMap_h
 
-#include "Collector.h"
+#include "Weak.h"
+#include "WeakInlines.h"
 #include <wtf/HashMap.h>
 
 namespace JSC {
 
-class JSCell;
+// A HashMap with Weak<JSCell> values, which automatically removes values once they're garbage collected.
 
-// A HashMap whose get() function returns emptyValue() for cells awaiting destruction.
-template<typename KeyType, typename MappedType>
-class WeakGCMap : public FastAllocBase {
-    /*
-    Invariants:
-        * A value enters the WeakGCMap marked. (Guaranteed by set().)
-        * A value that becomes unmarked leaves the WeakGCMap before being recycled. (Guaranteed by the value's destructor removing it from the WeakGCMap.)
-        * A value that becomes unmarked leaves the WeakGCMap before becoming marked again. (Guaranteed by all destructors running before the mark phase begins.)
-        * During the mark phase, all values in the WeakGCMap are valid. (Guaranteed by all destructors running before the mark phase begins.)
-    */
+template<typename KeyArg, typename ValueArg, typename HashArg = typename DefaultHash<KeyArg>::Hash,
+    typename KeyTraitsArg = HashTraits<KeyArg>>
+class WeakGCMap {
+    WTF_MAKE_FAST_ALLOCATED;
+    typedef Weak<ValueArg> ValueType;
+    typedef HashMap<KeyArg, ValueType, HashArg, KeyTraitsArg> HashMapType;
 
 public:
-    typedef typename HashMap<KeyType, MappedType>::iterator iterator;
-    typedef typename HashMap<KeyType, MappedType>::const_iterator const_iterator;
-    
-    bool isEmpty() { return m_map.isEmpty(); }
+    typedef typename HashMapType::KeyType KeyType;
+    typedef typename HashMapType::AddResult AddResult;
+    typedef typename HashMapType::iterator iterator;
+    typedef typename HashMapType::const_iterator const_iterator;
+
+    WeakGCMap()
+        : m_gcThreshold(minGCThreshold)
+    {
+    }
 
-    MappedType get(const KeyType& key) const;
-    pair<iterator, bool> set(const KeyType&, const MappedType&); 
-    MappedType take(const KeyType& key);
+    ValueArg* get(const KeyType& key) const
+    {
+        return m_map.get(key);
+    }
 
-    // These unchecked functions provide access to a value even if the value's
-    // mark bit is not set. This is used, among other things, to retrieve values
-    // during the GC mark phase, which begins by clearing all mark bits.
+    AddResult set(const KeyType& key, ValueType value)
+    {
+        gcMapIfNeeded();
+        return m_map.set(key, WTF::move(value));
+    }
 
-    MappedType uncheckedGet(const KeyType& key) const { return m_map.get(key); }
-    bool uncheckedRemove(const KeyType&, const MappedType&);
+    ALWAYS_INLINE AddResult add(const KeyType& key, ValueType value)
+    {
+        gcMapIfNeeded();
+        AddResult addResult = m_map.fastAdd(key, nullptr);
+        if (!addResult.iterator->value) { // New value or found a zombie value.
+            addResult.isNewEntry = true;
+            addResult.iterator->value = WTF::move(value);
+        }
+        return addResult;
+    }
 
-    iterator uncheckedBegin() { return m_map.begin(); }
-    iterator uncheckedEnd() { return m_map.end(); }
+    bool remove(const KeyType& key)
+    {
+        return m_map.remove(key);
+    }
 
-    const_iterator uncheckedBegin() const { return m_map.begin(); }
-    const_iterator uncheckedEnd() const { return m_map.end(); }
+    void clear()
+    {
+        m_map.clear();
+    }
+
+    iterator find(const KeyType& key)
+    {
+        iterator it = m_map.find(key);
+        iterator end = m_map.end();
+        if (it != end && !it->value) // Found a zombie value.
+            return end;
+        return it;
+    }
+
+    const_iterator find(const KeyType& key) const
+    {
+        return const_cast<WeakGCMap*>(this)->find(key);
+    }
+
+    bool contains(const KeyType& key) const
+    {
+        return find(key) != m_map.end();
+    }
 
 private:
-    HashMap<KeyType, MappedType> m_map;
-};
+    static const int minGCThreshold = 3;
+
+    NEVER_INLINE void gcMap()
+    {
+        Vector<KeyType, 4> zombies;
+
+        for (iterator it = m_map.begin(), end = m_map.end(); it != end; ++it) {
+            if (!it->value)
+                zombies.append(it->key);
+        }
+
+        for (size_t i = 0; i < zombies.size(); ++i)
+            m_map.remove(zombies[i]);
+    }
+
+    void gcMapIfNeeded()
+    {
+        if (m_map.size() < m_gcThreshold)
+            return;
 
-template<typename KeyType, typename MappedType>
-inline MappedType WeakGCMap<KeyType, MappedType>::get(const KeyType& key) const
-{
-    MappedType result = m_map.get(key);
-    if (result == HashTraits<MappedType>::emptyValue())
-        return result;
-    if (!Heap::isCellMarked(result))
-        return HashTraits<MappedType>::emptyValue();
-    return result;
-}
-
-template<typename KeyType, typename MappedType>
-MappedType WeakGCMap<KeyType, MappedType>::take(const KeyType& key)
-{
-    MappedType result = m_map.take(key);
-    if (result == HashTraits<MappedType>::emptyValue())
-        return result;
-    if (!Heap::isCellMarked(result))
-        return HashTraits<MappedType>::emptyValue();
-    return result;
-}
-
-template<typename KeyType, typename MappedType>
-pair<typename HashMap<KeyType, MappedType>::iterator, bool> WeakGCMap<KeyType, MappedType>::set(const KeyType& key, const MappedType& value)
-{
-    Heap::markCell(value); // If value is newly allocated, it's not marked, so mark it now.
-    pair<iterator, bool> result = m_map.add(key, value);
-    if (!result.second) { // pre-existing entry
-        result.second = !Heap::isCellMarked(result.first->second);
-        result.first->second = value;
+        gcMap();
+        m_gcThreshold = std::max(minGCThreshold, m_map.size() * 2 - 1);
     }
-    return result;
-}
-
-template<typename KeyType, typename MappedType>
-bool WeakGCMap<KeyType, MappedType>::uncheckedRemove(const KeyType& key, const MappedType& value)
-{
-    iterator it = m_map.find(key);
-    if (it == m_map.end())
-        return false;
-    if (it->second != value)
-        return false;
-    m_map.remove(it);
-    return true;
-}
+
+    HashMapType m_map;
+    int m_gcThreshold;
+};
+
+template<typename KeyArg, typename RawMappedArg, typename HashArg, typename KeyTraitsArg>
+const int WeakGCMap<KeyArg, RawMappedArg, HashArg, KeyTraitsArg>::minGCThreshold;
 
 } // namespace JSC