mirror of
https://github.com/darlinghq/darling-JavaScriptCore.git
synced 2024-11-23 04:09:40 +00:00
175 lines
6.6 KiB
C++
175 lines
6.6 KiB
C++
/*
|
|
* Copyright (C) 2017-2018 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 "WasmCodeBlock.h"
|
|
|
|
#if ENABLE(WEBASSEMBLY)
|
|
|
|
#include "WasmBBQPlan.h"
|
|
#include "WasmCallee.h"
|
|
#include "WasmLLIntPlan.h"
|
|
#include "WasmWorklist.h"
|
|
|
|
namespace JSC { namespace Wasm {
|
|
|
|
Ref<CodeBlock> CodeBlock::create(Context* context, MemoryMode mode, ModuleInformation& moduleInformation, RefPtr<LLIntCallees> llintCallees)
|
|
{
|
|
auto* result = new (NotNull, fastMalloc(sizeof(CodeBlock))) CodeBlock(context, mode, moduleInformation, llintCallees);
|
|
return adoptRef(*result);
|
|
}
|
|
|
|
CodeBlock::CodeBlock(Context* context, MemoryMode mode, ModuleInformation& moduleInformation, RefPtr<LLIntCallees> llintCallees)
|
|
: m_calleeCount(moduleInformation.internalFunctionCount())
|
|
, m_mode(mode)
|
|
, m_llintCallees(llintCallees)
|
|
{
|
|
RefPtr<CodeBlock> protectedThis = this;
|
|
|
|
if (Options::useWasmLLInt()) {
|
|
m_plan = adoptRef(*new LLIntPlan(context, makeRef(moduleInformation), m_llintCallees->data(), createSharedTask<Plan::CallbackType>([this, protectedThis = WTFMove(protectedThis)] (Plan&) {
|
|
auto locker = holdLock(m_lock);
|
|
if (m_plan->failed()) {
|
|
m_errorMessage = m_plan->errorMessage();
|
|
setCompilationFinished();
|
|
return;
|
|
}
|
|
|
|
// FIXME: we should eventually collect the BBQ code.
|
|
m_bbqCallees.resize(m_calleeCount);
|
|
m_omgCallees.resize(m_calleeCount);
|
|
m_wasmIndirectCallEntryPoints.resize(m_calleeCount);
|
|
|
|
for (unsigned i = 0; i < m_calleeCount; ++i)
|
|
m_wasmIndirectCallEntryPoints[i] = m_llintCallees->at(i)->entrypoint();
|
|
|
|
m_wasmToWasmExitStubs = m_plan->takeWasmToWasmExitStubs();
|
|
m_wasmToWasmCallsites = m_plan->takeWasmToWasmCallsites();
|
|
m_embedderCallees = static_cast<LLIntPlan*>(m_plan.get())->takeEmbedderCallees();
|
|
|
|
setCompilationFinished();
|
|
})));
|
|
} else {
|
|
m_plan = adoptRef(*new BBQPlan(context, makeRef(moduleInformation), EntryPlan::FullCompile, createSharedTask<Plan::CallbackType>([this, protectedThis = WTFMove(protectedThis)] (Plan&) {
|
|
auto locker = holdLock(m_lock);
|
|
if (m_plan->failed()) {
|
|
m_errorMessage = m_plan->errorMessage();
|
|
setCompilationFinished();
|
|
return;
|
|
}
|
|
|
|
// FIXME: we should eventually collect the BBQ code.
|
|
m_bbqCallees.resize(m_calleeCount);
|
|
m_omgCallees.resize(m_calleeCount);
|
|
m_wasmIndirectCallEntryPoints.resize(m_calleeCount);
|
|
|
|
BBQPlan* bbqPlan = static_cast<BBQPlan*>(m_plan.get());
|
|
bbqPlan->initializeCallees([&] (unsigned calleeIndex, RefPtr<EmbedderEntrypointCallee>&& embedderEntrypointCallee, RefPtr<BBQCallee>&& wasmEntrypoint) {
|
|
if (embedderEntrypointCallee) {
|
|
auto result = m_embedderCallees.set(calleeIndex, WTFMove(embedderEntrypointCallee));
|
|
ASSERT_UNUSED(result, result.isNewEntry);
|
|
}
|
|
m_wasmIndirectCallEntryPoints[calleeIndex] = wasmEntrypoint->entrypoint();
|
|
m_bbqCallees[calleeIndex] = adoptRef(static_cast<BBQCallee*>(wasmEntrypoint.leakRef()));
|
|
});
|
|
|
|
m_wasmToWasmExitStubs = m_plan->takeWasmToWasmExitStubs();
|
|
m_wasmToWasmCallsites = m_plan->takeWasmToWasmCallsites();
|
|
|
|
setCompilationFinished();
|
|
})));
|
|
}
|
|
m_plan->setMode(mode);
|
|
|
|
auto& worklist = Wasm::ensureWorklist();
|
|
// Note, immediately after we enqueue the plan, there is a chance the above callback will be called.
|
|
worklist.enqueue(makeRef(*m_plan.get()));
|
|
}
|
|
|
|
CodeBlock::~CodeBlock() { }
|
|
|
|
void CodeBlock::waitUntilFinished()
|
|
{
|
|
RefPtr<Plan> plan;
|
|
{
|
|
auto locker = holdLock(m_lock);
|
|
plan = m_plan;
|
|
}
|
|
|
|
if (plan) {
|
|
auto& worklist = Wasm::ensureWorklist();
|
|
worklist.completePlanSynchronously(*plan.get());
|
|
}
|
|
// else, if we don't have a plan, we're already compiled.
|
|
}
|
|
|
|
void CodeBlock::compileAsync(Context* context, AsyncCompilationCallback&& task)
|
|
{
|
|
RefPtr<Plan> plan;
|
|
{
|
|
auto locker = holdLock(m_lock);
|
|
plan = m_plan;
|
|
}
|
|
|
|
if (plan) {
|
|
// We don't need to keep a RefPtr on the Plan because the worklist will keep
|
|
// a RefPtr on the Plan until the plan finishes notifying all of its callbacks.
|
|
RefPtr<CodeBlock> protectedThis = this;
|
|
plan->addCompletionTask(context, createSharedTask<Plan::CallbackType>([this, task = WTFMove(task), protectedThis = WTFMove(protectedThis)] (Plan&) {
|
|
task->run(makeRef(*this));
|
|
}));
|
|
} else
|
|
task->run(makeRef(*this));
|
|
}
|
|
|
|
bool CodeBlock::isSafeToRun(MemoryMode memoryMode)
|
|
{
|
|
if (!runnable())
|
|
return false;
|
|
|
|
switch (m_mode) {
|
|
case Wasm::MemoryMode::BoundsChecking:
|
|
return true;
|
|
case Wasm::MemoryMode::Signaling:
|
|
// Code being in Signaling mode means that it performs no bounds checks.
|
|
// Its memory, even if empty, absolutely must also be in Signaling mode
|
|
// because the page protection detects out-of-bounds accesses.
|
|
return memoryMode == Wasm::MemoryMode::Signaling;
|
|
}
|
|
RELEASE_ASSERT_NOT_REACHED();
|
|
return false;
|
|
}
|
|
|
|
|
|
void CodeBlock::setCompilationFinished()
|
|
{
|
|
m_plan = nullptr;
|
|
m_compilationFinished.store(true);
|
|
}
|
|
|
|
} } // namespace JSC::Wasm
|
|
|
|
#endif // ENABLE(WEBASSEMBLY)
|