diff options
author | Simon Hausmann <simon.hausmann@nokia.com> | 2012-01-06 14:44:00 +0100 |
---|---|---|
committer | Simon Hausmann <simon.hausmann@nokia.com> | 2012-01-06 14:44:00 +0100 |
commit | 40736c5763bf61337c8c14e16d8587db021a87d4 (patch) | |
tree | b17a9c00042ad89cb1308e2484491799aa14e9f8 /Source/JavaScriptCore/heap/HandleHeap.cpp | |
download | qtwebkit-40736c5763bf61337c8c14e16d8587db021a87d4.tar.gz |
Imported WebKit commit 2ea9d364d0f6efa8fa64acf19f451504c59be0e4 (http://svn.webkit.org/repository/webkit/trunk@104285)
Diffstat (limited to 'Source/JavaScriptCore/heap/HandleHeap.cpp')
-rw-r--r-- | Source/JavaScriptCore/heap/HandleHeap.cpp | 211 |
1 files changed, 211 insertions, 0 deletions
diff --git a/Source/JavaScriptCore/heap/HandleHeap.cpp b/Source/JavaScriptCore/heap/HandleHeap.cpp new file mode 100644 index 000000000..2402f7efb --- /dev/null +++ b/Source/JavaScriptCore/heap/HandleHeap.cpp @@ -0,0 +1,211 @@ +/* + * Copyright (C) 2011 Apple Inc. All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * 1. Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * 2. Redistributions in binary form must reproduce the above copyright + * notice, this list of conditions and the following disclaimer in the + * documentation and/or other materials provided with the distribution. + * + * THIS SOFTWARE IS PROVIDED BY APPLE INC. AND ITS CONTRIBUTORS ``AS IS'' + * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, + * THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR + * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR ITS CONTRIBUTORS + * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR + * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF + * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS + * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN + * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF + * THE POSSIBILITY OF SUCH DAMAGE. + */ + +#include "config.h" +#include "HandleHeap.h" + +#include "HeapRootVisitor.h" +#include "JSObject.h" + +namespace JSC { + +WeakHandleOwner::~WeakHandleOwner() +{ +} + +bool WeakHandleOwner::isReachableFromOpaqueRoots(Handle<Unknown>, void*, SlotVisitor&) +{ + return false; +} + +void WeakHandleOwner::finalize(Handle<Unknown>, void*) +{ +} + +HandleHeap::HandleHeap(JSGlobalData* globalData) + : m_globalData(globalData) + , m_nextToFinalize(0) +{ + grow(); +} + +void HandleHeap::grow() +{ + Node* block = m_blockStack.grow(); + for (int i = m_blockStack.blockLength - 1; i >= 0; --i) { + Node* node = &block[i]; + new (NotNull, node) Node(this); + m_freeList.push(node); + } +} + +void HandleHeap::visitStrongHandles(HeapRootVisitor& heapRootVisitor) +{ + Node* end = m_strongList.end(); + for (Node* node = m_strongList.begin(); node != end; node = node->next()) { +#if ENABLE(GC_VALIDATION) + if (!isLiveNode(node)) + CRASH(); +#endif + heapRootVisitor.visit(node->slot()); + } +} + +void HandleHeap::visitWeakHandles(HeapRootVisitor& heapRootVisitor) +{ + SlotVisitor& visitor = heapRootVisitor.visitor(); + + Node* end = m_weakList.end(); + for (Node* node = m_weakList.begin(); node != end; node = node->next()) { +#if ENABLE(GC_VALIDATION) + if (!isValidWeakNode(node)) + CRASH(); +#endif + JSCell* cell = node->slot()->asCell(); + if (Heap::isMarked(cell)) + continue; + + WeakHandleOwner* weakOwner = node->weakOwner(); + if (!weakOwner) + continue; + + if (!weakOwner->isReachableFromOpaqueRoots(Handle<Unknown>::wrapSlot(node->slot()), node->weakOwnerContext(), visitor)) + continue; + + heapRootVisitor.visit(node->slot()); + } +} + +void HandleHeap::finalizeWeakHandles() +{ + Node* end = m_weakList.end(); + for (Node* node = m_weakList.begin(); node != end; node = m_nextToFinalize) { + m_nextToFinalize = node->next(); +#if ENABLE(GC_VALIDATION) + if (!isValidWeakNode(node)) + CRASH(); +#endif + + JSCell* cell = node->slot()->asCell(); + if (Heap::isMarked(cell)) + continue; + + if (WeakHandleOwner* weakOwner = node->weakOwner()) { + weakOwner->finalize(Handle<Unknown>::wrapSlot(node->slot()), node->weakOwnerContext()); + if (m_nextToFinalize != node->next()) // Owner deallocated node. + continue; + } +#if ENABLE(GC_VALIDATION) + if (!isLiveNode(node)) + CRASH(); +#endif + *node->slot() = JSValue(); + SentinelLinkedList<Node>::remove(node); + m_immediateList.push(node); + } + + m_nextToFinalize = 0; +} + +void HandleHeap::writeBarrier(HandleSlot slot, const JSValue& value) +{ + // Forbid assignment to handles during the finalization phase, since it would violate many GC invariants. + // File a bug with stack trace if you hit this. + if (m_nextToFinalize) + CRASH(); + + if (!value == !*slot && slot->isCell() == value.isCell()) + return; + + Node* node = toNode(slot); +#if ENABLE(GC_VALIDATION) + if (!isLiveNode(node)) + CRASH(); +#endif + SentinelLinkedList<Node>::remove(node); + if (!value || !value.isCell()) { + m_immediateList.push(node); + return; + } + + if (node->isWeak()) { + m_weakList.push(node); +#if ENABLE(GC_VALIDATION) + if (!isLiveNode(node)) + CRASH(); +#endif + return; + } + + m_strongList.push(node); +#if ENABLE(GC_VALIDATION) + if (!isLiveNode(node)) + CRASH(); +#endif +} + +unsigned HandleHeap::protectedGlobalObjectCount() +{ + unsigned count = 0; + Node* end = m_strongList.end(); + for (Node* node = m_strongList.begin(); node != end; node = node->next()) { + JSValue value = *node->slot(); + if (value.isObject() && asObject(value.asCell())->isGlobalObject()) + count++; + } + return count; +} + +#if ENABLE(GC_VALIDATION) || !ASSERT_DISABLED +bool HandleHeap::isLiveNode(Node* node) +{ + if (node->prev()->next() != node) + return false; + if (node->next()->prev() != node) + return false; + + return true; +} + +bool HandleHeap::isValidWeakNode(Node* node) +{ + if (!isLiveNode(node)) + return false; + if (!node->isWeak()) + return false; + + JSValue value = *node->slot(); + if (!value || !value.isCell()) + return false; + + JSCell* cell = value.asCell(); + if (!cell || !cell->structure()) + return false; + + return true; +} +#endif + +} // namespace JSC |