blob: b510631339bf0f559a5fb86bb3bf1d9740599181 [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"
#include "AccessCase.h"
#include "CacheableIdentifierInlines.h"
#include "CodeBlock.h"
#include "DFGCommonData.h"
#include "Heap.h"
#include "VM.h"
#include "JITStubRoutineSet.h"
#include "JSCellInlines.h"
#include "SharedJITStubSet.h"
#include <wtf/RefPtr.h>
namespace JSC {
GCAwareJITStubRoutine::GCAwareJITStubRoutine(Type type, const MacroAssemblerCodeRef<JITStubRoutinePtrTag>& code, JSCell* owner, bool isCodeImmutable)
: JITStubRoutine(type, code)
, m_owner(owner)
, m_isCodeImmutable(isCodeImmutable)
{
}
void GCAwareJITStubRoutine::makeGCAware(VM& vm)
{
vm.heap.m_jitStubRoutines->add(this);
m_isGCAware = true;
}
void GCAwareJITStubRoutine::observeZeroRefCountImpl()
{
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.
IGNORE_GCC_WARNINGS_BEGIN("sequence-point")
delete this;
IGNORE_GCC_WARNINGS_END
return;
}
RELEASE_ASSERT(!m_refCount);
m_isJettisoned = true;
}
void GCAwareJITStubRoutine::deleteFromGC()
{
ASSERT(m_isJettisoned);
ASSERT(!m_refCount);
ASSERT(!m_mayBeExecuting);
IGNORE_GCC_WARNINGS_BEGIN("sequence-point")
delete this;
IGNORE_GCC_WARNINGS_END
}
bool GCAwareJITStubRoutine::removeDeadOwners(VM& vm)
{
ASSERT(vm.heap.isInPhase(CollectorPhase::End));
if (m_owner)
return !vm.heap.isMarked(m_owner);
#if ENABLE(JIT)
if (m_isInSharedJITStubSet) {
auto& owners = static_cast<PolymorphicAccessJITStubRoutine*>(this)->m_owners;
owners.removeAllIf([&](auto pair) {
return !vm.heap.isMarked(pair.key);
});
if (owners.isEmpty()) {
// All owners are dead. Unregistering itself from m_vm.m_sharedJITStubs since it is no longer valid.
vm.m_sharedJITStubs->remove(static_cast<PolymorphicAccessJITStubRoutine*>(this));
return true;
}
return false;
}
#endif
return false;
}
#if ENABLE(JIT)
PolymorphicAccessJITStubRoutine::PolymorphicAccessJITStubRoutine(Type type, const MacroAssemblerCodeRef<JITStubRoutinePtrTag>& code, VM& vm, FixedVector<Ref<AccessCase>>&& cases, FixedVector<StructureID>&& weakStructures, JSCell* owner, bool isCodeImmutable)
: GCAwareJITStubRoutine(type, code, owner, isCodeImmutable)
, m_vm(vm)
, m_cases(WTFMove(cases))
, m_weakStructures(WTFMove(weakStructures))
, m_watchpointSet(WatchpointSet::create(IsWatched))
{
}
PolymorphicAccessJITStubRoutine::~PolymorphicAccessJITStubRoutine() = default;
void PolymorphicAccessJITStubRoutine::observeZeroRefCountImpl()
{
if (m_isInSharedJITStubSet) {
ASSERT(m_vm.m_sharedJITStubs);
m_vm.m_sharedJITStubs->remove(this);
}
// Now PolymorphicAccessJITStubRoutine is no longer referenced. So Watchpoints inside WatchpointSet do not matter. Let's eagerly clear them
m_watchpointSet = nullptr;
m_watchpoints.clear();
Base::observeZeroRefCountImpl();
}
void PolymorphicAccessJITStubRoutine::invalidate()
{
if (RefPtr watchpointSet = WTFMove(m_watchpointSet)) {
StringFireDetail detail("PolymorphicAccessJITStubRoutine has been invalidated");
VM& vm = m_vm;
watchpointSet->fireAll(vm, detail);
}
}
unsigned PolymorphicAccessJITStubRoutine::computeHash(std::span<const Ref<AccessCase>> cases)
{
if (cases.size() == 1)
return cases.front()->hash();
Hasher hasher;
for (auto& key : cases)
WTF::add(hasher, key->hash());
return hasher.hash();
}
void PolymorphicAccessJITStubRoutine::addGCAwareWatchpoint()
{
if (!m_isGCAware)
makeGCAware(m_vm);
}
void PolymorphicAccessJITStubRoutine::addedToSharedJITStubSet()
{
m_isInSharedJITStubSet = true;
}
bool PolymorphicAccessJITStubRoutine::visitWeakImpl(VM& vm)
{
bool isValid = true;
for (StructureID weakReference : m_weakStructures)
isValid &= vm.heap.isMarked(weakReference.decode());
isValid &= Base::visitWeakImpl(vm);
return isValid;
}
MarkingGCAwareJITStubRoutine::MarkingGCAwareJITStubRoutine(
Type type, const MacroAssemblerCodeRef<JITStubRoutinePtrTag>& code, VM& vm, FixedVector<Ref<AccessCase>>&& cases, FixedVector<StructureID>&& weakStructures, JSCell* owner,
const Vector<JSCell*>& cells, Vector<std::unique_ptr<OptimizingCallLinkInfo>, 16>&& callLinkInfos, bool isCodeImmutable)
: PolymorphicAccessJITStubRoutine(type, code, vm, WTFMove(cases), WTFMove(weakStructures), owner, isCodeImmutable)
, m_cells(cells.size())
, m_callLinkInfos(WTFMove(callLinkInfos))
{
for (unsigned i = cells.size(); i--;)
m_cells[i].set(vm, owner, cells[i]);
}
template<typename Visitor>
ALWAYS_INLINE void MarkingGCAwareJITStubRoutine::markRequiredObjectsInternalImpl(Visitor& visitor)
{
for (auto& entry : m_cells)
visitor.append(entry);
}
void MarkingGCAwareJITStubRoutine::markRequiredObjectsImpl(AbstractSlotVisitor& visitor)
{
markRequiredObjectsInternalImpl(visitor);
}
void MarkingGCAwareJITStubRoutine::markRequiredObjectsImpl(SlotVisitor& visitor)
{
markRequiredObjectsInternalImpl(visitor);
}
bool MarkingGCAwareJITStubRoutine::visitWeakImpl(VM& vm)
{
for (auto& callLinkInfo : m_callLinkInfos) {
if (callLinkInfo)
callLinkInfo->visitWeak(vm);
}
return Base::visitWeakImpl(vm);
}
CallLinkInfo* MarkingGCAwareJITStubRoutine::callLinkInfoAtImpl(const ConcurrentJSLocker&, unsigned index)
{
if (index < m_callLinkInfos.size())
return m_callLinkInfos[index].get();
return nullptr;
}
GCAwareJITStubRoutineWithExceptionHandler::GCAwareJITStubRoutineWithExceptionHandler(const MacroAssemblerCodeRef<JITStubRoutinePtrTag>& code, VM& vm, FixedVector<Ref<AccessCase>>&& cases, FixedVector<StructureID>&& weakStructures, JSCell* owner, const Vector<JSCell*>& cells, Vector<std::unique_ptr<OptimizingCallLinkInfo>, 16>&& callLinkInfos,
CodeBlock* codeBlockForExceptionHandlers, DisposableCallSiteIndex exceptionHandlerCallSiteIndex, bool isCodeImmutable)
: MarkingGCAwareJITStubRoutine(JITStubRoutine::Type::GCAwareJITStubRoutineWithExceptionHandlerType, code, vm, WTFMove(cases), WTFMove(weakStructures), owner, cells, WTFMove(callLinkInfos), isCodeImmutable)
, 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()) || vm().heap.isInPhase(CollectorPhase::End));
if (m_codeOriginPool)
m_codeOriginPool->removeDisposableCallSiteIndex(m_exceptionHandlerCallSiteIndex);
#endif
}
void GCAwareJITStubRoutineWithExceptionHandler::observeZeroRefCountImpl()
{
#if ENABLE(DFG_JIT)
if (m_codeBlockWithExceptionHandler) {
m_codeBlockWithExceptionHandler->removeExceptionHandlerForCallSite(m_exceptionHandlerCallSiteIndex);
m_codeBlockWithExceptionHandler = nullptr;
}
#endif
Base::observeZeroRefCountImpl();
}
Ref<PolymorphicAccessJITStubRoutine> createICJITStubRoutine(
const MacroAssemblerCodeRef<JITStubRoutinePtrTag>& code,
FixedVector<Ref<AccessCase>>&& cases,
FixedVector<StructureID>&& weakStructures,
VM& vm,
JSCell* owner,
bool makesCalls,
const Vector<JSCell*>& cells,
Vector<std::unique_ptr<OptimizingCallLinkInfo>, 16>&& callLinkInfos,
CodeBlock* codeBlockForExceptionHandlers,
DisposableCallSiteIndex exceptionHandlerCallSiteIndex)
{
if (!makesCalls) {
// Allocating CallLinkInfos means we should have calls.
#if ASSERT_ENABLED
for (auto& callLinkInfo : callLinkInfos)
ASSERT(!callLinkInfo);
#endif
constexpr bool isCodeImmutable = false;
auto stub = adoptRef(*new PolymorphicAccessJITStubRoutine(JITStubRoutine::Type::PolymorphicAccessJITStubRoutineType, code, vm, WTFMove(cases), WTFMove(weakStructures), owner, isCodeImmutable));
stub->makeGCAware(vm);
return stub;
}
if (codeBlockForExceptionHandlers) {
RELEASE_ASSERT(JSC::JITCode::isOptimizingJIT(codeBlockForExceptionHandlers->jitType()));
constexpr bool isCodeImmutable = false;
auto stub = adoptRef(*new GCAwareJITStubRoutineWithExceptionHandler(code, vm, WTFMove(cases), WTFMove(weakStructures), owner, cells, WTFMove(callLinkInfos), codeBlockForExceptionHandlers, exceptionHandlerCallSiteIndex, isCodeImmutable));
stub->makeGCAware(vm);
return stub;
}
bool hasCallLinkInfo = false;
for (auto& callLinkInfo : callLinkInfos) {
if (callLinkInfo) {
hasCallLinkInfo = true;
break;
}
}
if (cells.isEmpty() && !hasCallLinkInfo) {
constexpr bool isCodeImmutable = false;
auto stub = adoptRef(*new PolymorphicAccessJITStubRoutine(JITStubRoutine::Type::PolymorphicAccessJITStubRoutineType, code, vm, WTFMove(cases), WTFMove(weakStructures), owner, isCodeImmutable));
stub->makeGCAware(vm);
return stub;
}
constexpr bool isCodeImmutable = false;
auto stub = adoptRef(*new MarkingGCAwareJITStubRoutine(JITStubRoutine::Type::MarkingGCAwareJITStubRoutineType, code, vm, WTFMove(cases), WTFMove(weakStructures), owner, cells, WTFMove(callLinkInfos), isCodeImmutable));
stub->makeGCAware(vm);
return stub;
}
Ref<PolymorphicAccessJITStubRoutine> createPreCompiledICJITStubRoutine(const MacroAssemblerCodeRef<JITStubRoutinePtrTag>& code, VM& vm, JSCell* owner)
{
auto stub = adoptRef(*new PolymorphicAccessJITStubRoutine(JITStubRoutine::Type::PolymorphicAccessJITStubRoutineType, code, vm, { }, { }, owner, true));
return stub;
}
#endif // ENABLE(JIT)
} // namespace JSC