blob: 865ae57d4155726715560a7b169f6dfbaa82b2b1 [file] [log] [blame]
/*
* Copyright (C) 2012-2021 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. ``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
* 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 "GCAwareJITStubRoutine.h"
#if ENABLE(JIT)
#include "AccessCase.h"
#include "CodeBlock.h"
#include "DFGCommonData.h"
#include "Heap.h"
#include "VM.h"
#include "JITStubRoutineSet.h"
#include "JSCellInlines.h"
#include <wtf/RefPtr.h>
namespace JSC {
GCAwareJITStubRoutine::GCAwareJITStubRoutine(const MacroAssemblerCodeRef<JITStubRoutinePtrTag>& code)
: JITStubRoutine(code)
{
}
GCAwareJITStubRoutine::~GCAwareJITStubRoutine() { }
void GCAwareJITStubRoutine::makeGCAware(VM& vm)
{
vm.heap.m_jitStubRoutines->add(this);
m_isGCAware = true;
}
void GCAwareJITStubRoutine::observeZeroRefCount()
{
if (m_isJettisoned || !m_isGCAware) {
// This case is needed for when the system shuts down. It may be that
// the JIT stub routine set gets deleted before we get around to deleting
// this guy. In that case the GC informs us that we're jettisoned already
// and that we should delete ourselves as soon as the ref count reaches
// zero.
delete this;
return;
}
RELEASE_ASSERT(!m_refCount);
m_isJettisoned = true;
}
void GCAwareJITStubRoutine::deleteFromGC()
{
ASSERT(m_isJettisoned);
ASSERT(!m_refCount);
ASSERT(!m_mayBeExecuting);
delete this;
}
PolymorphicAccessJITStubRoutine::PolymorphicAccessJITStubRoutine(const MacroAssemblerCodeRef<JITStubRoutinePtrTag>& code, VM& vm, FixedVector<RefPtr<AccessCase>>&& cases, FixedVector<StructureID>&& weakStructures)
: GCAwareJITStubRoutine(code)
, m_vm(vm)
, m_cases(WTFMove(cases))
, m_weakStructures(WTFMove(weakStructures))
{
}
void PolymorphicAccessJITStubRoutine::observeZeroRefCount()
{
if (m_vm.m_sharedJITStubs)
m_vm.m_sharedJITStubs->remove(this);
Base::observeZeroRefCount();
}
unsigned PolymorphicAccessJITStubRoutine::computeHash(const FixedVector<RefPtr<AccessCase>>& cases, const FixedVector<StructureID>& weakStructures)
{
Hasher hasher;
for (auto& key : cases)
WTF::add(hasher, key->hash());
for (auto& structureID : weakStructures)
WTF::add(hasher, structureID);
return hasher.hash();
}
MarkingGCAwareJITStubRoutine::MarkingGCAwareJITStubRoutine(
const MacroAssemblerCodeRef<JITStubRoutinePtrTag>& code, VM& vm, FixedVector<RefPtr<AccessCase>>&& cases, FixedVector<StructureID>&& weakStructures, const JSCell* owner,
const Vector<JSCell*>& cells, Bag<OptimizingCallLinkInfo>&& callLinkInfos)
: PolymorphicAccessJITStubRoutine(code, vm, WTFMove(cases), WTFMove(weakStructures))
, m_cells(cells.size())
, m_callLinkInfos(WTFMove(callLinkInfos))
{
for (unsigned i = cells.size(); i--;)
m_cells[i].set(vm, owner, cells[i]);
}
MarkingGCAwareJITStubRoutine::~MarkingGCAwareJITStubRoutine()
{
}
template<typename Visitor>
ALWAYS_INLINE void MarkingGCAwareJITStubRoutine::markRequiredObjectsInternalImpl(Visitor& visitor)
{
for (auto& entry : m_cells)
visitor.append(entry);
}
void MarkingGCAwareJITStubRoutine::markRequiredObjectsInternal(AbstractSlotVisitor& visitor)
{
markRequiredObjectsInternalImpl(visitor);
}
void MarkingGCAwareJITStubRoutine::markRequiredObjectsInternal(SlotVisitor& visitor)
{
markRequiredObjectsInternalImpl(visitor);
}
GCAwareJITStubRoutineWithExceptionHandler::GCAwareJITStubRoutineWithExceptionHandler(
const MacroAssemblerCodeRef<JITStubRoutinePtrTag>& code, VM& vm, FixedVector<RefPtr<AccessCase>>&& cases, FixedVector<StructureID>&& weakStructures, const JSCell* owner, const Vector<JSCell*>& cells, Bag<OptimizingCallLinkInfo>&& callLinkInfos,
CodeBlock* codeBlockForExceptionHandlers, DisposableCallSiteIndex exceptionHandlerCallSiteIndex)
: MarkingGCAwareJITStubRoutine(code, vm, WTFMove(cases), WTFMove(weakStructures), owner, cells, WTFMove(callLinkInfos))
, m_codeBlockWithExceptionHandler(codeBlockForExceptionHandlers)
#if ENABLE(DFG_JIT)
, m_codeOriginPool(&m_codeBlockWithExceptionHandler->codeOrigins())
#endif
, m_exceptionHandlerCallSiteIndex(exceptionHandlerCallSiteIndex)
{
RELEASE_ASSERT(m_codeBlockWithExceptionHandler);
ASSERT(!!m_codeBlockWithExceptionHandler->handlerForIndex(exceptionHandlerCallSiteIndex.bits()));
}
GCAwareJITStubRoutineWithExceptionHandler::~GCAwareJITStubRoutineWithExceptionHandler()
{
#if ENABLE(DFG_JIT)
// We delay deallocation of m_exceptionHandlerCallSiteIndex until GCAwareJITStubRoutineWithExceptionHandler gets destroyed.
// This means that CallSiteIndex can be reserved correctly so long as the code owned by GCAwareJITStubRoutineWithExceptionHandler is on the stack.
// This is important since CallSite can be queried so long as this code is on the stack: StackVisitor can retreive CallSiteIndex from the stack.
ASSERT(!isCompilationThread() && !Thread::mayBeGCThread());
if (m_codeOriginPool)
m_codeOriginPool->removeDisposableCallSiteIndex(m_exceptionHandlerCallSiteIndex);
#endif
}
void GCAwareJITStubRoutineWithExceptionHandler::aboutToDie()
{
m_codeBlockWithExceptionHandler = nullptr;
#if ENABLE(DFG_JIT)
m_codeOriginPool = nullptr;
#endif
}
void GCAwareJITStubRoutineWithExceptionHandler::observeZeroRefCount()
{
#if ENABLE(DFG_JIT)
if (m_codeBlockWithExceptionHandler) {
m_codeBlockWithExceptionHandler->removeExceptionHandlerForCallSite(m_exceptionHandlerCallSiteIndex);
m_codeBlockWithExceptionHandler = nullptr;
}
#endif
Base::observeZeroRefCount();
}
Ref<PolymorphicAccessJITStubRoutine> createICJITStubRoutine(
const MacroAssemblerCodeRef<JITStubRoutinePtrTag>& code,
FixedVector<RefPtr<AccessCase>>&& cases,
FixedVector<StructureID>&& weakStructures,
VM& vm,
const JSCell* owner,
bool makesCalls,
const Vector<JSCell*>& cells,
Bag<OptimizingCallLinkInfo>&& callLinkInfos,
CodeBlock* codeBlockForExceptionHandlers,
DisposableCallSiteIndex exceptionHandlerCallSiteIndex)
{
if (!makesCalls) {
// Allocating CallLinkInfos means we should have calls.
ASSERT(callLinkInfos.isEmpty());
return adoptRef(*new PolymorphicAccessJITStubRoutine(code, vm, WTFMove(cases), WTFMove(weakStructures)));
}
if (codeBlockForExceptionHandlers) {
RELEASE_ASSERT(JITCode::isOptimizingJIT(codeBlockForExceptionHandlers->jitType()));
auto stub = adoptRef(*new GCAwareJITStubRoutineWithExceptionHandler(code, vm, WTFMove(cases), WTFMove(weakStructures), owner, cells, WTFMove(callLinkInfos), codeBlockForExceptionHandlers, exceptionHandlerCallSiteIndex));
stub->makeGCAware(vm);
return stub;
}
if (cells.isEmpty() && callLinkInfos.isEmpty()) {
auto stub = adoptRef(*new PolymorphicAccessJITStubRoutine(code, vm, WTFMove(cases), WTFMove(weakStructures)));
stub->makeGCAware(vm);
return stub;
}
auto stub = adoptRef(*new MarkingGCAwareJITStubRoutine(code, vm, WTFMove(cases), WTFMove(weakStructures), owner, cells, WTFMove(callLinkInfos)));
stub->makeGCAware(vm);
return stub;
}
} // namespace JSC
#endif // ENABLE(JIT)