mirror of
https://github.com/darlinghq/darling-JavaScriptCore.git
synced 2024-11-23 04:09:40 +00:00
178 lines
5.6 KiB
C++
178 lines
5.6 KiB
C++
/*
|
|
* Copyright (C) 2012-2017 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 "WeakBlock.h"
|
|
|
|
#include "CellContainerInlines.h"
|
|
#include "Heap.h"
|
|
#include "HeapAnalyzer.h"
|
|
#include "JSCInlines.h"
|
|
#include "WeakHandleOwner.h"
|
|
|
|
namespace JSC {
|
|
|
|
DEFINE_ALLOCATOR_WITH_HEAP_IDENTIFIER(WeakBlock);
|
|
|
|
WeakBlock* WeakBlock::create(Heap& heap, CellContainer container)
|
|
{
|
|
heap.didAllocateBlock(WeakBlock::blockSize);
|
|
return new (NotNull, WeakBlockMalloc::malloc(blockSize)) WeakBlock(container);
|
|
|
|
}
|
|
|
|
void WeakBlock::destroy(Heap& heap, WeakBlock* block)
|
|
{
|
|
block->~WeakBlock();
|
|
WeakBlockMalloc::free(block);
|
|
heap.didFreeBlock(WeakBlock::blockSize);
|
|
}
|
|
|
|
WeakBlock::WeakBlock(CellContainer container)
|
|
: DoublyLinkedListNode<WeakBlock>()
|
|
, m_container(container)
|
|
{
|
|
for (size_t i = 0; i < weakImplCount(); ++i) {
|
|
WeakImpl* weakImpl = &weakImpls()[i];
|
|
new (NotNull, weakImpl) WeakImpl;
|
|
addToFreeList(&m_sweepResult.freeList, weakImpl);
|
|
}
|
|
|
|
ASSERT(isEmpty());
|
|
}
|
|
|
|
void WeakBlock::lastChanceToFinalize()
|
|
{
|
|
for (size_t i = 0; i < weakImplCount(); ++i) {
|
|
WeakImpl* weakImpl = &weakImpls()[i];
|
|
if (weakImpl->state() >= WeakImpl::Finalized)
|
|
continue;
|
|
weakImpl->setState(WeakImpl::Dead);
|
|
finalize(weakImpl);
|
|
}
|
|
}
|
|
|
|
void WeakBlock::sweep()
|
|
{
|
|
// If a block is completely empty, a sweep won't have any effect.
|
|
if (isEmpty())
|
|
return;
|
|
|
|
SweepResult sweepResult;
|
|
for (size_t i = 0; i < weakImplCount(); ++i) {
|
|
WeakImpl* weakImpl = &weakImpls()[i];
|
|
if (weakImpl->state() == WeakImpl::Dead)
|
|
finalize(weakImpl);
|
|
if (weakImpl->state() == WeakImpl::Deallocated)
|
|
addToFreeList(&sweepResult.freeList, weakImpl);
|
|
else {
|
|
sweepResult.blockIsFree = false;
|
|
if (weakImpl->state() == WeakImpl::Live)
|
|
sweepResult.blockIsLogicallyEmpty = false;
|
|
}
|
|
}
|
|
|
|
m_sweepResult = sweepResult;
|
|
ASSERT(!m_sweepResult.isNull());
|
|
}
|
|
|
|
template<typename ContainerType>
|
|
void WeakBlock::specializedVisit(ContainerType& container, SlotVisitor& visitor)
|
|
{
|
|
HeapVersion markingVersion = visitor.markingVersion();
|
|
|
|
size_t count = weakImplCount();
|
|
for (size_t i = 0; i < count; ++i) {
|
|
WeakImpl* weakImpl = &weakImpls()[i];
|
|
if (weakImpl->state() != WeakImpl::Live)
|
|
continue;
|
|
|
|
WeakHandleOwner* weakHandleOwner = weakImpl->weakHandleOwner();
|
|
if (!weakHandleOwner)
|
|
continue;
|
|
|
|
JSValue jsValue = weakImpl->jsValue();
|
|
if (container.isMarked(markingVersion, jsValue.asCell()))
|
|
continue;
|
|
|
|
const char* reason = "";
|
|
const char** reasonPtr = nullptr;
|
|
if (UNLIKELY(visitor.isAnalyzingHeap()))
|
|
reasonPtr = &reason;
|
|
|
|
if (!weakHandleOwner->isReachableFromOpaqueRoots(Handle<Unknown>::wrapSlot(&const_cast<JSValue&>(jsValue)), weakImpl->context(), visitor, reasonPtr))
|
|
continue;
|
|
|
|
visitor.appendUnbarriered(jsValue);
|
|
|
|
if (UNLIKELY(visitor.isAnalyzingHeap())) {
|
|
if (jsValue.isCell())
|
|
visitor.heapAnalyzer()->setOpaqueRootReachabilityReasonForCell(jsValue.asCell(), *reasonPtr);
|
|
}
|
|
}
|
|
}
|
|
|
|
void WeakBlock::visit(SlotVisitor& visitor)
|
|
{
|
|
// If a block is completely empty, a visit won't have any effect.
|
|
if (isEmpty())
|
|
return;
|
|
|
|
// If this WeakBlock doesn't belong to a CellContainer, we won't even be here.
|
|
ASSERT(m_container);
|
|
|
|
if (m_container.isPreciseAllocation())
|
|
specializedVisit(m_container.preciseAllocation(), visitor);
|
|
else
|
|
specializedVisit(m_container.markedBlock(), visitor);
|
|
}
|
|
|
|
void WeakBlock::reap()
|
|
{
|
|
// If a block is completely empty, a reaping won't have any effect.
|
|
if (isEmpty())
|
|
return;
|
|
|
|
// If this WeakBlock doesn't belong to a CellContainer, we won't even be here.
|
|
ASSERT(m_container);
|
|
|
|
HeapVersion markingVersion = m_container.heap()->objectSpace().markingVersion();
|
|
|
|
for (size_t i = 0; i < weakImplCount(); ++i) {
|
|
WeakImpl* weakImpl = &weakImpls()[i];
|
|
if (weakImpl->state() > WeakImpl::Dead)
|
|
continue;
|
|
|
|
if (m_container.isMarked(markingVersion, weakImpl->jsValue().asCell())) {
|
|
ASSERT(weakImpl->state() == WeakImpl::Live);
|
|
continue;
|
|
}
|
|
|
|
weakImpl->setState(WeakImpl::Dead);
|
|
}
|
|
}
|
|
|
|
} // namespace JSC
|