blob: 2483628bccd53000a9d6ba0555746057a255908f [file] [log] [blame]
/*
* Copyright (C) 2019 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.
*/
#pragma once
#if ENABLE(WEBASSEMBLY)
#include "WasmModuleInformation.h"
#include "WasmPlan.h"
#include "WasmStreamingParser.h"
#include <wtf/Function.h>
#include <wtf/SharedTask.h>
#include <wtf/ThreadSafeRefCounted.h>
#include <wtf/Vector.h>
namespace JSC {
class CallLinkInfo;
namespace Wasm {
class EntryPlan : public Plan, public StreamingParserClient {
public:
using Base = Plan;
enum AsyncWork : uint8_t { FullCompile, Validation };
// Note: CompletionTask should not hold a reference to the Plan otherwise there will be a reference cycle.
EntryPlan(Context*, Ref<ModuleInformation>, AsyncWork, CompletionTask&&, CreateEmbedderWrapper&&, ThrowWasmException);
JS_EXPORT_PRIVATE EntryPlan(Context*, Vector<uint8_t>&&, AsyncWork, CompletionTask&&, CreateEmbedderWrapper&&, ThrowWasmException);
EntryPlan(Context*, Ref<ModuleInformation>, CompletionTask&&);
EntryPlan(Context*, AsyncWork, CompletionTask&&);
virtual ~EntryPlan() = default;
using CalleeInitializer = Function<void(uint32_t, RefPtr<Callee>&&, Ref<Callee>&&)>;
virtual void initializeCallees(const CalleeInitializer&) = 0;
bool parseAndValidateModule()
{
return parseAndValidateModule(m_source.data(), m_source.size());
}
bool parseAndValidateModule(const uint8_t*, size_t);
void compileFunctions(CompilationEffort);
Vector<Export>& exports() const
{
RELEASE_ASSERT(!failed() && !hasWork());
return m_moduleInformation->exports;
}
size_t internalFunctionCount() const
{
RELEASE_ASSERT(!failed() && !hasWork());
return m_moduleInformation->internalFunctionCount();
}
Ref<ModuleInformation>&& takeModuleInformation()
{
RELEASE_ASSERT(!failed() && !hasWork());
return WTFMove(m_moduleInformation);
}
Vector<MacroAssemblerCodeRef<WasmEntryPtrTag>>&& takeWasmToWasmExitStubs()
{
RELEASE_ASSERT(!failed() && !hasWork());
return WTFMove(m_wasmToWasmExitStubs);
}
Vector<Vector<UnlinkedWasmToWasmCall>> takeWasmToWasmCallsites()
{
RELEASE_ASSERT(!failed() && !hasWork());
return WTFMove(m_unlinkedWasmToWasmCalls);
}
enum class State : uint8_t {
Initial,
Validated,
Prepared,
Compiled,
Completed // We should only move to Completed if we are holding the lock.
};
bool hasWork() const override
{
if (m_asyncWork == AsyncWork::Validation)
return m_state < State::Validated;
return m_state < State::Compiled;
}
void work(CompilationEffort) override;
bool hasBeenPrepared() const { return m_state >= State::Prepared; }
bool multiThreaded() const override { return hasBeenPrepared(); }
bool didReceiveFunctionData(unsigned, const FunctionData&) override;
private:
class ThreadCountHolder;
friend class ThreadCountHolder;
void prepare();
bool isComplete() const override { return m_state == State::Completed; }
void complete(const AbstractLocker&) override;
const char* stateString(State);
protected:
// For some reason friendship doesn't extend to parent classes...
using Base::m_lock;
virtual bool prepareImpl() = 0;
virtual void compileFunction(uint32_t functionIndex) = 0;
virtual void didCompleteCompilation(const AbstractLocker&) = 0;
void moveToState(State);
template<typename T>
bool tryReserveCapacity(Vector<T>& vector, size_t size, const char* what)
{
if (UNLIKELY(!vector.tryReserveCapacity(size))) {
fail(holdLock(m_lock), WTF::makeString("Failed allocating enough space for ", size, what));
return false;
}
return true;
}
Vector<uint8_t> m_source;
Vector<MacroAssemblerCodeRef<WasmEntryPtrTag>> m_wasmToWasmExitStubs;
HashSet<uint32_t, typename DefaultHash<uint32_t>::Hash, WTF::UnsignedWithZeroKeyHashTraits<uint32_t>> m_exportedFunctionIndices;
Vector<Vector<UnlinkedWasmToWasmCall>> m_unlinkedWasmToWasmCalls;
StreamingParser m_streamingParser;
State m_state;
const AsyncWork m_asyncWork;
uint8_t m_numberOfActiveThreads { 0 };
uint32_t m_currentIndex { 0 };
};
} } // namespace JSC::Wasm
#endif // ENABLE(WEBASSEMBLY)