blob: 77f919313197fa0bb4c16a7a1f71a127f02c1cd9 [file] [log] [blame]
/*
* Copyright (C) 2016-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. ``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 "WasmCallee.h"
#if ENABLE(WEBASSEMBLY)
#include "LLIntExceptions.h"
#include "WasmCalleeRegistry.h"
#include "WasmCallingConvention.h"
namespace JSC { namespace Wasm {
Callee::Callee(Wasm::CompilationMode compilationMode)
: m_compilationMode(compilationMode)
{
}
Callee::Callee(Wasm::CompilationMode compilationMode, size_t index, std::pair<const Name*, RefPtr<NameSection>>&& name)
: m_compilationMode(compilationMode)
, m_indexOrName(index, WTFMove(name))
{
}
template<typename Func>
inline void Callee::runWithDowncast(const Func& func)
{
switch (m_compilationMode) {
case CompilationMode::LLIntMode:
func(static_cast<LLIntCallee*>(this));
break;
#if ENABLE(WEBASSEMBLY_B3JIT)
case CompilationMode::BBQMode:
func(static_cast<BBQCallee*>(this));
break;
case CompilationMode::BBQForOSREntryMode:
func(static_cast<OSREntryCallee*>(this));
break;
case CompilationMode::OMGMode:
func(static_cast<OMGCallee*>(this));
break;
case CompilationMode::OMGForOSREntryMode:
func(static_cast<OSREntryCallee*>(this));
break;
#else
case CompilationMode::BBQMode:
case CompilationMode::BBQForOSREntryMode:
case CompilationMode::OMGMode:
case CompilationMode::OMGForOSREntryMode:
break;
#endif
case CompilationMode::EmbedderEntrypointMode:
func(static_cast<EmbedderEntrypointCallee*>(this));
break;
case CompilationMode::JSToWasmICMode:
func(static_cast<JSToWasmICCallee*>(this));
break;
}
}
template<typename Func>
inline void Callee::runWithDowncast(const Func& func) const
{
const_cast<Callee*>(this)->runWithDowncast(func);
}
void Callee::dump(PrintStream& out) const
{
out.print(makeString(m_indexOrName));
}
CodePtr<WasmEntryPtrTag> Callee::entrypoint() const
{
CodePtr<WasmEntryPtrTag> codePtr;
runWithDowncast([&](auto* derived) {
codePtr = derived->entrypointImpl();
});
return codePtr;
}
std::tuple<void*, void*> Callee::range() const
{
std::tuple<void*, void*> result;
runWithDowncast([&](auto* derived) {
result = derived->rangeImpl();
});
return result;
}
RegisterAtOffsetList* Callee::calleeSaveRegisters()
{
RegisterAtOffsetList* result = nullptr;
runWithDowncast([&](auto* derived) {
result = derived->calleeSaveRegistersImpl();
});
return result;
}
void Callee::operator delete(Callee* callee, std::destroying_delete_t)
{
CalleeRegistry::singleton().unregisterCallee(callee);
callee->runWithDowncast([](auto* derived) {
std::destroy_at(derived);
std::decay_t<decltype(*derived)>::freeAfterDestruction(derived);
});
}
const HandlerInfo* Callee::handlerForIndex(Instance& instance, unsigned index, const Tag* tag)
{
ASSERT(hasExceptionHandlers());
return HandlerInfo::handlerForIndex(instance, m_exceptionHandlers, index, tag);
}
JITCallee::JITCallee(Wasm::CompilationMode compilationMode)
: Callee(compilationMode)
{
}
JITCallee::JITCallee(Wasm::CompilationMode compilationMode, size_t index, std::pair<const Name*, RefPtr<NameSection>>&& name)
: Callee(compilationMode, index, WTFMove(name))
{
}
void JITCallee::setEntrypoint(Wasm::Entrypoint&& entrypoint)
{
m_entrypoint = WTFMove(entrypoint);
CalleeRegistry::singleton().registerCallee(this);
}
ptrdiff_t JSToWasmICCallee::previousInstanceOffset(const RegisterAtOffsetList& calleeSaveRegisters)
{
ptrdiff_t result = calleeSaveRegisters.registerCount() * sizeof(CPURegister);
result = -result - sizeof(CPURegister);
#if ASSERT_ENABLED
ptrdiff_t minOffset = 1;
for (const RegisterAtOffset& regAtOffset : calleeSaveRegisters) {
ptrdiff_t offset = regAtOffset.offset();
ASSERT(offset < 0);
minOffset = std::min(offset, minOffset);
#if USE(JSVALUE32_64)
ASSERT(!regAtOffset.reg().isFPR()); // Because FPRs are wider than sizeof(CPURegister)
#endif
}
ASSERT(minOffset - static_cast<ptrdiff_t>(sizeof(CPURegister)) == result);
#endif
return result;
}
Wasm::Instance* JSToWasmICCallee::previousInstance(CallFrame* callFrame)
{
ASSERT(callFrame->callee().asWasmCallee() == this);
auto* result = *bitwise_cast<Wasm::Instance**>(bitwise_cast<char*>(callFrame) + previousInstanceOffset(m_entrypoint.calleeSaveRegisters));
return result;
}
LLIntCallee::LLIntCallee(FunctionCodeBlockGenerator& generator, size_t index, std::pair<const Name*, RefPtr<NameSection>>&& name)
: Callee(Wasm::CompilationMode::LLIntMode, index, WTFMove(name))
, m_functionIndex(generator.m_functionIndex)
, m_numVars(generator.m_numVars)
, m_numCalleeLocals(generator.m_numCalleeLocals)
, m_numArguments(generator.m_numArguments)
, m_constantTypes(WTFMove(generator.m_constantTypes))
, m_constants(WTFMove(generator.m_constants))
, m_instructions(WTFMove(generator.m_instructions))
, m_instructionsRawPointer(generator.m_instructionsRawPointer)
, m_jumpTargets(WTFMove(generator.m_jumpTargets))
, m_signatures(WTFMove(generator.m_signatures))
, m_outOfLineJumpTargets(WTFMove(generator.m_outOfLineJumpTargets))
, m_tierUpCounter(WTFMove(generator.m_tierUpCounter))
, m_jumpTables(WTFMove(generator.m_jumpTables))
{
if (size_t count = generator.numberOfExceptionHandlers()) {
m_exceptionHandlers = FixedVector<HandlerInfo>(count);
for (size_t i = 0; i < count; i++) {
const UnlinkedHandlerInfo& unlinkedHandler = generator.exceptionHandler(i);
HandlerInfo& handler = m_exceptionHandlers[i];
auto& instruction = *m_instructions->at(unlinkedHandler.m_target).ptr();
CodeLocationLabel<ExceptionHandlerPtrTag> target;
if (unlinkedHandler.m_type == HandlerType::Catch)
target = CodeLocationLabel<ExceptionHandlerPtrTag>(LLInt::handleWasmCatch(instruction.width()).code());
else
target = CodeLocationLabel<ExceptionHandlerPtrTag>(LLInt::handleWasmCatchAll(instruction.width()).code());
handler.initialize(unlinkedHandler, target);
}
}
}
void LLIntCallee::setEntrypoint(CodePtr<WasmEntryPtrTag> entrypoint)
{
ASSERT(!m_entrypoint);
m_entrypoint = entrypoint;
CalleeRegistry::singleton().registerCallee(this);
}
RegisterAtOffsetList* LLIntCallee::calleeSaveRegistersImpl()
{
static LazyNeverDestroyed<RegisterAtOffsetList> calleeSaveRegisters;
static std::once_flag initializeFlag;
std::call_once(initializeFlag, [] {
RegisterSet registers;
registers.add(GPRInfo::regCS0, IgnoreVectors); // Wasm::Instance
#if CPU(X86_64)
registers.add(GPRInfo::regCS2, IgnoreVectors); // PB
#elif CPU(ARM64) || CPU(RISCV64)
registers.add(GPRInfo::regCS7, IgnoreVectors); // PB
#elif CPU(ARM)
registers.add(GPRInfo::regCS1, IgnoreVectors); // PB
#else
#error Unsupported architecture.
#endif
ASSERT(registers.numberOfSetRegisters() == numberOfLLIntCalleeSaveRegisters);
calleeSaveRegisters.construct(WTFMove(registers));
});
return &calleeSaveRegisters.get();
}
WasmInstructionStream::Offset LLIntCallee::outOfLineJumpOffset(WasmInstructionStream::Offset bytecodeOffset)
{
ASSERT(m_outOfLineJumpTargets.contains(bytecodeOffset));
return m_outOfLineJumpTargets.get(bytecodeOffset);
}
const WasmInstruction* LLIntCallee::outOfLineJumpTarget(const WasmInstruction* pc)
{
int offset = bytecodeOffset(pc);
int target = outOfLineJumpOffset(offset);
return m_instructions->at(offset + target).ptr();
}
#if ENABLE(WEBASSEMBLY_B3JIT)
void OptimizingJITCallee::linkExceptionHandlers(Vector<UnlinkedHandlerInfo> unlinkedExceptionHandlers, Vector<CodeLocationLabel<ExceptionHandlerPtrTag>> exceptionHandlerLocations)
{
size_t count = unlinkedExceptionHandlers.size();
m_exceptionHandlers = FixedVector<HandlerInfo>(count);
for (size_t i = 0; i < count; i++) {
HandlerInfo& handler = m_exceptionHandlers[i];
const UnlinkedHandlerInfo& unlinkedHandler = unlinkedExceptionHandlers[i];
CodeLocationLabel<ExceptionHandlerPtrTag> location = exceptionHandlerLocations[i];
handler.initialize(unlinkedHandler, location);
}
}
const StackMap& OptimizingJITCallee::stackmap(CallSiteIndex callSiteIndex) const
{
auto iter = m_stackmaps.find(callSiteIndex);
if (iter == m_stackmaps.end()) {
for (auto pair : m_stackmaps) {
dataLog(pair.key.bits(), ": ");
for (auto value : pair.value)
dataLog(value, ", ");
dataLogLn("");
}
}
RELEASE_ASSERT(iter != m_stackmaps.end());
return iter->value;
}
#endif
} } // namespace JSC::Wasm
#endif // ENABLE(WEBASSEMBLY)