#include "config.h"
#include "WasmModule.h"
#if ENABLE(WEBASSEMBLY)
#include "WasmLLIntPlan.h"
#include "WasmModuleInformation.h"
#include "WasmWorklist.h"
namespace JSC { namespace Wasm {
Module::Module(LLIntPlan& plan)
: m_moduleInformation(plan.takeModuleInformation())
, m_llintCallees(LLIntCallees::create(plan.takeCallees()))
, m_llintEntryThunks(plan.takeEntryThunks())
{
}
Module::~Module() { }
Wasm::SignatureIndex Module::signatureIndexFromFunctionIndexSpace(unsigned functionIndexSpace) const
{
return m_moduleInformation->signatureIndexFromFunctionIndexSpace(functionIndexSpace);
}
static Module::ValidationResult makeValidationResult(LLIntPlan& plan)
{
ASSERT(!plan.hasWork());
if (plan.failed())
return Unexpected<String>(plan.errorMessage());
return Module::ValidationResult(Module::create(plan));
}
static Plan::CompletionTask makeValidationCallback(Module::AsyncValidationCallback&& callback)
{
return createSharedTask<Plan::CallbackType>([callback = WTFMove(callback)] (Plan& plan) {
ASSERT(!plan.hasWork());
callback->run(makeValidationResult(static_cast<LLIntPlan&>(plan)));
});
}
Module::ValidationResult Module::validateSync(Context* context, Vector<uint8_t>&& source)
{
Ref<LLIntPlan> plan = adoptRef(*new LLIntPlan(context, WTFMove(source), EntryPlan::Validation, Plan::dontFinalize()));
Wasm::ensureWorklist().enqueue(plan.get());
plan->waitForCompletion();
return makeValidationResult(plan.get());
}
void Module::validateAsync(Context* context, Vector<uint8_t>&& source, Module::AsyncValidationCallback&& callback)
{
Ref<Plan> plan = adoptRef(*new LLIntPlan(context, WTFMove(source), EntryPlan::Validation, makeValidationCallback(WTFMove(callback))));
Wasm::ensureWorklist().enqueue(WTFMove(plan));
}
Ref<CodeBlock> Module::getOrCreateCodeBlock(Context* context, MemoryMode mode)
{
RefPtr<CodeBlock> codeBlock;
auto locker = holdLock(m_lock);
codeBlock = m_codeBlocks[static_cast<uint8_t>(mode)];
if (!codeBlock || (codeBlock->compilationFinished() && !codeBlock->runnable())) {
RefPtr<LLIntCallees> llintCallees = nullptr;
if (Options::useWasmLLInt())
llintCallees = m_llintCallees;
codeBlock = CodeBlock::create(context, mode, const_cast<ModuleInformation&>(moduleInformation()), llintCallees);
m_codeBlocks[static_cast<uint8_t>(mode)] = codeBlock;
}
return codeBlock.releaseNonNull();
}
Ref<CodeBlock> Module::compileSync(Context* context, MemoryMode mode)
{
Ref<CodeBlock> codeBlock = getOrCreateCodeBlock(context, mode);
codeBlock->waitUntilFinished();
return codeBlock;
}
void Module::compileAsync(Context* context, MemoryMode mode, CodeBlock::AsyncCompilationCallback&& task)
{
Ref<CodeBlock> codeBlock = getOrCreateCodeBlock(context, mode);
codeBlock->compileAsync(context, WTFMove(task));
}
} }
#endif // ENABLE(WEBASSEMBLY)