]>
git.saurik.com Git - apple/javascriptcore.git/blob - heap/HandleHeap.h
41b7fb23d298e444ef61b1cf917a61287951a2cd
2 * Copyright (C) 2011 Apple Inc. All rights reserved.
4 * Redistribution and use in source and binary forms, with or without
5 * modification, are permitted provided that the following conditions
7 * 1. Redistributions of source code must retain the above copyright
8 * notice, this list of conditions and the following disclaimer.
9 * 2. Redistributions in binary form must reproduce the above copyright
10 * notice, this list of conditions and the following disclaimer in the
11 * documentation and/or other materials provided with the distribution.
13 * THIS SOFTWARE IS PROVIDED BY APPLE INC. AND ITS CONTRIBUTORS ``AS IS''
14 * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
15 * THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
16 * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR ITS CONTRIBUTORS
17 * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
18 * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
19 * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
20 * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
21 * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
22 * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF
23 * THE POSSIBILITY OF SUCH DAMAGE.
29 #include "BlockStack.h"
31 #include "SentinelLinkedList.h"
32 #include "SinglyLinkedList.h"
37 class HeapRootVisitor
;
42 typedef MarkStack SlotVisitor
;
44 class WeakHandleOwner
{
46 virtual ~WeakHandleOwner();
47 virtual bool isReachableFromOpaqueRoots(Handle
<Unknown
>, void* context
, SlotVisitor
&);
48 virtual void finalize(Handle
<Unknown
>, void* context
);
53 static HandleHeap
* heapFor(HandleSlot
);
55 HandleHeap(JSGlobalData
*);
57 JSGlobalData
* globalData();
59 HandleSlot
allocate();
60 void deallocate(HandleSlot
);
62 void makeWeak(HandleSlot
, WeakHandleOwner
* = 0, void* context
= 0);
63 HandleSlot
copyWeak(HandleSlot
);
65 void markStrongHandles(HeapRootVisitor
&);
66 void markWeakHandles(HeapRootVisitor
&);
67 void finalizeWeakHandles();
69 void writeBarrier(HandleSlot
, const JSValue
&);
72 bool hasWeakOwner(HandleSlot
, WeakHandleOwner
*);
73 bool hasFinalizer(HandleSlot
);
76 unsigned protectedGlobalObjectCount();
77 void protectedObjectTypeCounts(TypeCounter
&);
82 Node(WTF::SentinelTag
);
86 HandleHeap
* handleHeap();
88 void makeWeak(WeakHandleOwner
*, void* context
);
91 WeakHandleOwner
* weakOwner();
92 void* weakOwnerContext();
101 WeakHandleOwner
* emptyWeakOwner();
104 HandleHeap
* m_handleHeap
;
105 WeakHandleOwner
* m_weakOwner
;
106 void* m_weakOwnerContext
;
111 static HandleSlot
toHandle(Node
*);
112 static Node
* toNode(HandleSlot
);
117 bool isValidWeakNode(Node
*);
120 JSGlobalData
* m_globalData
;
121 BlockStack
<Node
> m_blockStack
;
123 SentinelLinkedList
<Node
> m_strongList
;
124 SentinelLinkedList
<Node
> m_weakList
;
125 SentinelLinkedList
<Node
> m_immediateList
;
126 SinglyLinkedList
<Node
> m_freeList
;
127 Node
* m_nextToFinalize
;
130 inline HandleHeap
* HandleHeap::heapFor(HandleSlot handle
)
132 return toNode(handle
)->handleHeap();
135 inline JSGlobalData
* HandleHeap::globalData()
140 inline HandleSlot
HandleHeap::toHandle(Node
* node
)
142 return reinterpret_cast<HandleSlot
>(node
);
145 inline HandleHeap::Node
* HandleHeap::toNode(HandleSlot handle
)
147 return reinterpret_cast<Node
*>(handle
);
150 inline HandleSlot
HandleHeap::allocate()
152 if (m_freeList
.isEmpty())
155 Node
* node
= m_freeList
.pop();
156 new (node
) Node(this);
157 m_immediateList
.push(node
);
158 return toHandle(node
);
161 inline void HandleHeap::deallocate(HandleSlot handle
)
163 Node
* node
= toNode(handle
);
164 if (node
== m_nextToFinalize
) {
165 m_nextToFinalize
= node
->next();
166 ASSERT(m_nextToFinalize
->next());
169 SentinelLinkedList
<Node
>::remove(node
);
170 m_freeList
.push(node
);
173 inline HandleSlot
HandleHeap::copyWeak(HandleSlot other
)
175 Node
* node
= toNode(allocate());
176 node
->makeWeak(toNode(other
)->weakOwner(), toNode(other
)->weakOwnerContext());
177 writeBarrier(node
->slot(), *other
);
178 *node
->slot() = *other
;
179 return toHandle(node
);
182 inline void HandleHeap::makeWeak(HandleSlot handle
, WeakHandleOwner
* weakOwner
, void* context
)
184 Node
* node
= toNode(handle
);
185 node
->makeWeak(weakOwner
, context
);
187 SentinelLinkedList
<Node
>::remove(node
);
188 if (!*handle
|| !handle
->isCell()) {
189 m_immediateList
.push(node
);
193 m_weakList
.push(node
);
197 inline bool HandleHeap::hasWeakOwner(HandleSlot handle
, WeakHandleOwner
* weakOwner
)
199 return toNode(handle
)->weakOwner() == weakOwner
;
202 inline bool HandleHeap::hasFinalizer(HandleSlot handle
)
204 return toNode(handle
)->weakOwner();
208 inline HandleHeap::Node::Node(HandleHeap
* handleHeap
)
209 : m_handleHeap(handleHeap
)
211 , m_weakOwnerContext(0)
215 inline HandleHeap::Node::Node(WTF::SentinelTag
)
218 , m_weakOwnerContext(0)
222 inline HandleSlot
HandleHeap::Node::slot()
227 inline HandleHeap
* HandleHeap::Node::handleHeap()
232 inline void HandleHeap::Node::makeWeak(WeakHandleOwner
* weakOwner
, void* context
)
234 m_weakOwner
= weakOwner
? weakOwner
: emptyWeakOwner();
235 m_weakOwnerContext
= context
;
238 inline bool HandleHeap::Node::isWeak()
240 return m_weakOwner
; // True for emptyWeakOwner().
243 inline WeakHandleOwner
* HandleHeap::Node::weakOwner()
245 return m_weakOwner
== emptyWeakOwner() ? 0 : m_weakOwner
; // 0 for emptyWeakOwner().
248 inline void* HandleHeap::Node::weakOwnerContext()
251 return m_weakOwnerContext
;
254 inline void HandleHeap::Node::setPrev(Node
* prev
)
259 inline HandleHeap::Node
* HandleHeap::Node::prev()
264 inline void HandleHeap::Node::setNext(Node
* next
)
269 inline HandleHeap::Node
* HandleHeap::Node::next()
274 // Sentinel to indicate that a node is weak, but its owner has no meaningful
275 // callbacks. This allows us to optimize by skipping such nodes.
276 inline WeakHandleOwner
* HandleHeap::Node::emptyWeakOwner()
278 return reinterpret_cast<WeakHandleOwner
*>(-1);