blob: 0308850b5cc0f754ab6b30fca414d6ab6cd3d0be [file] [log] [blame] [edit]
// Copyright 2019 the V8 project authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#ifndef V8_WASM_MODULE_INSTANTIATE_H_
#define V8_WASM_MODULE_INSTANTIATE_H_
#if !V8_ENABLE_WEBASSEMBLY
#error This header should only be included if WebAssembly is enabled.
#endif // !V8_ENABLE_WEBASSEMBLY
#include <stdint.h>
#include <optional>
#include "src/common/message-template.h"
#include "src/objects/code-kind.h"
#include "src/wasm/wasm-value.h"
#include "src/wasm/well-known-imports.h"
namespace v8 {
namespace internal {
class FixedArray;
class JSArrayBuffer;
class WasmFunctionData;
class WasmModuleObject;
class WasmInstanceObject;
class WasmTrustedInstanceData;
class Zone;
namespace wasm {
class ErrorThrower;
enum Suspend : int { kSuspend, kNoSuspend };
// kStressSwitch: switch to a secondary stack, but without the JSPI semantics:
// do not handle async imports and do not return a Promise. For testing only.
enum Promise : int { kPromise, kNoPromise, kStressSwitch };
struct WasmModule;
// Calls to Wasm imports are handled in several different ways, depending on the
// type of the target function/callable and whether the signature matches the
// argument arity.
enum class ImportCallKind : uint8_t {
kLinkError, // static Wasm->Wasm type error
kRuntimeTypeError, // runtime Wasm->JS type error
kWasmToCapi, // fast Wasm->C-API call
kWasmToJSFastApi, // fast Wasm->JS Fast API C call
kWasmToWasm, // fast Wasm->Wasm call
kJSFunction, // fast Wasm->JS call
// For everything else, there's the call builtin.
kUseCallBuiltin
};
enum PrecreateExternal : bool {
kOnlyInternalFunction = false,
kPrecreateExternal = true,
};
constexpr ImportCallKind kDefaultImportCallKind = ImportCallKind::kJSFunction;
// Resolves which import call wrapper is required for the given JS callable.
// Provides the kind of wrapper needed, the ultimate target callable, and the
// suspender object if applicable. Note that some callables (e.g. a
// {WasmExportedFunction} or {WasmJSFunction}) just wrap another target, which
// is why the ultimate target is provided as well.
class ResolvedWasmImport {
public:
V8_EXPORT_PRIVATE ResolvedWasmImport(
DirectHandle<WasmTrustedInstanceData> trusted_instance_data,
int func_index, DirectHandle<JSReceiver> callable,
const wasm::CanonicalSig* sig, WellKnownImport preknown_import);
ImportCallKind kind() const { return kind_; }
WellKnownImport well_known_status() const { return well_known_status_; }
Suspend suspend() const { return suspend_; }
DirectHandle<JSReceiver> callable() const { return callable_; }
// Avoid reading function data from the result of `callable()`, because it
// might have been corrupted in the meantime (in a compromised sandbox).
// Instead, use this cached copy.
DirectHandle<WasmFunctionData> trusted_function_data() const {
return trusted_function_data_;
}
private:
void SetCallable(Isolate* isolate, Tagged<JSReceiver> callable);
void SetCallable(Isolate* isolate, DirectHandle<JSReceiver> callable);
ImportCallKind ComputeKind(
DirectHandle<WasmTrustedInstanceData> trusted_instance_data,
int func_index, const wasm::CanonicalSig* expected_sig,
WellKnownImport preknown_import);
ImportCallKind kind_;
WellKnownImport well_known_status_{WellKnownImport::kGeneric};
Suspend suspend_{kNoSuspend};
DirectHandle<JSReceiver> callable_;
DirectHandle<WasmFunctionData> trusted_function_data_;
};
MaybeDirectHandle<WasmInstanceObject> InstantiateToInstanceObject(
Isolate* isolate, ErrorThrower* thrower,
DirectHandle<WasmModuleObject> module_object,
MaybeDirectHandle<JSReceiver> imports,
MaybeDirectHandle<JSArrayBuffer> memory);
// Initializes a segment at index {segment_index} of the segment array of
// {instance}. If successful, returns the empty {Optional}, otherwise an
// {Optional} that contains the error message. Exits early if the segment is
// already initialized.
// {precreate_external_functions} is a non-binding hint that it would be
// beneficial for performance to create the corresponding WasmExportedFunctions
// along with any internal funcrefs.
std::optional<MessageTemplate> InitializeElementSegment(
Zone* zone, Isolate* isolate,
DirectHandle<WasmTrustedInstanceData> trusted_instance_data,
DirectHandle<WasmTrustedInstanceData> shared_trusted_instance_data,
uint32_t segment_index,
PrecreateExternal precreate_external_functions = kOnlyInternalFunction);
V8_EXPORT_PRIVATE void CreateMapForType(
Isolate* isolate, const WasmModule* module, ModuleTypeIndex type_index,
DirectHandle<FixedArray> maybe_shared_maps);
// Wrapper information required for graph building.
struct WrapperCompilationInfo {
CodeKind code_kind;
// For wasm-js wrappers only:
wasm::ImportCallKind import_kind = kDefaultImportCallKind;
int expected_arity = 0;
wasm::Suspend suspend = kNoSuspend;
// For js-wasm wrappers:
bool receiver_is_first_param = false;
};
} // namespace wasm
} // namespace internal
} // namespace v8
#endif // V8_WASM_MODULE_INSTANTIATE_H_