2023-03-16 12:59:32 -04:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2021, Ali Mohammad Pur <mpfard@serenityos.org>
|
|
|
|
* Copyright (c) 2023, Tim Flynn <trflynn89@serenityos.org>
|
|
|
|
*
|
|
|
|
* SPDX-License-Identifier: BSD-2-Clause
|
|
|
|
*/
|
|
|
|
|
|
|
|
#pragma once
|
|
|
|
|
|
|
|
#include <AK/Optional.h>
|
2024-11-15 04:01:23 +13:00
|
|
|
#include <LibGC/Root.h>
|
2023-03-16 12:59:32 -04:00
|
|
|
#include <LibJS/Forward.h>
|
2023-03-16 14:11:21 -04:00
|
|
|
#include <LibJS/Runtime/Completion.h>
|
2024-12-21 18:09:36 +01:00
|
|
|
#include <LibJS/Runtime/NativeFunction.h>
|
2023-03-16 12:59:32 -04:00
|
|
|
#include <LibJS/Runtime/Value.h>
|
2023-03-16 14:11:21 -04:00
|
|
|
#include <LibWasm/AbstractMachine/AbstractMachine.h>
|
2023-03-16 12:59:32 -04:00
|
|
|
#include <LibWeb/Forward.h>
|
|
|
|
|
|
|
|
namespace Web::WebAssembly {
|
|
|
|
|
2024-04-25 19:09:34 +01:00
|
|
|
void visit_edges(JS::Object&, JS::Cell::Visitor&);
|
2024-04-25 19:09:34 +01:00
|
|
|
void finalize(JS::Object&);
|
2023-03-16 12:59:32 -04:00
|
|
|
|
2024-11-15 04:01:23 +13:00
|
|
|
bool validate(JS::VM&, GC::Root<WebIDL::BufferSource>& bytes);
|
|
|
|
WebIDL::ExceptionOr<GC::Ref<WebIDL::Promise>> compile(JS::VM&, GC::Root<WebIDL::BufferSource>& bytes);
|
|
|
|
WebIDL::ExceptionOr<GC::Ref<WebIDL::Promise>> compile_streaming(JS::VM&, GC::Root<WebIDL::Promise> source);
|
2023-03-16 12:59:32 -04:00
|
|
|
|
2024-11-15 04:01:23 +13:00
|
|
|
WebIDL::ExceptionOr<GC::Ref<WebIDL::Promise>> instantiate(JS::VM&, GC::Root<WebIDL::BufferSource>& bytes, Optional<GC::Root<JS::Object>>& import_object);
|
|
|
|
WebIDL::ExceptionOr<GC::Ref<WebIDL::Promise>> instantiate(JS::VM&, Module const& module_object, Optional<GC::Root<JS::Object>>& import_object);
|
|
|
|
WebIDL::ExceptionOr<GC::Ref<WebIDL::Promise>> instantiate_streaming(JS::VM&, GC::Root<WebIDL::Promise> source, Optional<GC::Root<JS::Object>>& import_object);
|
2023-03-16 12:59:32 -04:00
|
|
|
|
2023-03-16 14:11:21 -04:00
|
|
|
namespace Detail {
|
2024-04-25 19:09:34 +01:00
|
|
|
struct CompiledWebAssemblyModule : public RefCounted<CompiledWebAssemblyModule> {
|
2024-08-22 01:13:37 +02:00
|
|
|
explicit CompiledWebAssemblyModule(NonnullRefPtr<Wasm::Module> module)
|
2023-03-16 14:11:21 -04:00
|
|
|
: module(move(module))
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2024-08-22 01:13:37 +02:00
|
|
|
NonnullRefPtr<Wasm::Module> module;
|
2023-03-16 14:11:21 -04:00
|
|
|
};
|
|
|
|
|
2024-04-25 19:09:34 +01:00
|
|
|
class WebAssemblyCache {
|
|
|
|
public:
|
|
|
|
void add_compiled_module(NonnullRefPtr<CompiledWebAssemblyModule> module) { m_compiled_modules.append(module); }
|
2024-11-15 04:01:23 +13:00
|
|
|
void add_function_instance(Wasm::FunctionAddress address, GC::Ptr<JS::NativeFunction> function) { m_function_instances.set(address, function); }
|
|
|
|
void add_imported_object(GC::Ptr<JS::Object> object) { m_imported_objects.set(object); }
|
2024-08-17 15:40:21 -07:00
|
|
|
void add_extern_value(Wasm::ExternAddress address, JS::Value value) { m_extern_values.set(address, value); }
|
2024-12-23 14:47:25 -07:00
|
|
|
void add_global_instance(Wasm::GlobalAddress address, GC::Ptr<WebAssembly::Global> global) { m_global_instances.set(address, global); }
|
2024-04-25 19:09:34 +01:00
|
|
|
|
2024-11-15 04:01:23 +13:00
|
|
|
Optional<GC::Ptr<JS::NativeFunction>> get_function_instance(Wasm::FunctionAddress address) { return m_function_instances.get(address); }
|
2024-08-17 15:40:21 -07:00
|
|
|
Optional<JS::Value> get_extern_value(Wasm::ExternAddress address) { return m_extern_values.get(address); }
|
2024-12-23 14:47:25 -07:00
|
|
|
Optional<GC::Ptr<WebAssembly::Global>> get_global_instance(Wasm::GlobalAddress address) { return m_global_instances.get(address); }
|
2024-04-25 19:09:34 +01:00
|
|
|
|
2024-11-15 04:01:23 +13:00
|
|
|
HashMap<Wasm::FunctionAddress, GC::Ptr<JS::NativeFunction>> function_instances() const { return m_function_instances; }
|
2024-08-17 15:40:21 -07:00
|
|
|
HashMap<Wasm::ExternAddress, JS::Value> extern_values() const { return m_extern_values; }
|
2024-12-23 14:47:25 -07:00
|
|
|
HashMap<Wasm::GlobalAddress, GC::Ptr<WebAssembly::Global>> global_instances() const { return m_global_instances; }
|
2024-11-15 04:01:23 +13:00
|
|
|
HashTable<GC::Ptr<JS::Object>> imported_objects() const { return m_imported_objects; }
|
2024-04-25 19:09:34 +01:00
|
|
|
Wasm::AbstractMachine& abstract_machine() { return m_abstract_machine; }
|
|
|
|
|
|
|
|
private:
|
2024-11-15 04:01:23 +13:00
|
|
|
HashMap<Wasm::FunctionAddress, GC::Ptr<JS::NativeFunction>> m_function_instances;
|
2024-08-17 15:40:21 -07:00
|
|
|
HashMap<Wasm::ExternAddress, JS::Value> m_extern_values;
|
2024-12-23 14:47:25 -07:00
|
|
|
HashMap<Wasm::GlobalAddress, GC::Ptr<WebAssembly::Global>> m_global_instances;
|
2024-04-25 19:09:34 +01:00
|
|
|
Vector<NonnullRefPtr<CompiledWebAssemblyModule>> m_compiled_modules;
|
2024-11-15 04:01:23 +13:00
|
|
|
HashTable<GC::Ptr<JS::Object>> m_imported_objects;
|
2024-04-25 19:09:34 +01:00
|
|
|
Wasm::AbstractMachine m_abstract_machine;
|
2023-03-16 14:11:21 -04:00
|
|
|
};
|
|
|
|
|
2024-12-21 18:09:36 +01:00
|
|
|
class ExportedWasmFunction final : public JS::NativeFunction {
|
|
|
|
JS_OBJECT(ExportedWasmFunction, JS::NativeFunction);
|
|
|
|
GC_DECLARE_ALLOCATOR(ExportedWasmFunction);
|
|
|
|
|
|
|
|
public:
|
2025-03-18 18:08:02 -05:00
|
|
|
static GC::Ref<ExportedWasmFunction> create(JS::Realm&, FlyString const& name, ESCAPING Function<JS::ThrowCompletionOr<JS::Value>(JS::VM&)>, Wasm::FunctionAddress);
|
2024-12-21 18:09:36 +01:00
|
|
|
virtual ~ExportedWasmFunction() override = default;
|
|
|
|
|
|
|
|
Wasm::FunctionAddress exported_address() const { return m_exported_address; }
|
|
|
|
|
|
|
|
protected:
|
2025-04-20 13:08:56 +02:00
|
|
|
ExportedWasmFunction(FlyString name, AK::Function<JS::ThrowCompletionOr<JS::Value>(JS::VM&)>, Wasm::FunctionAddress, Object& prototype);
|
2024-12-21 18:09:36 +01:00
|
|
|
|
|
|
|
private:
|
|
|
|
Wasm::FunctionAddress m_exported_address;
|
|
|
|
};
|
|
|
|
|
2024-04-25 19:09:34 +01:00
|
|
|
WebAssemblyCache& get_cache(JS::Realm&);
|
|
|
|
|
2024-11-15 04:01:23 +13:00
|
|
|
JS::ThrowCompletionOr<NonnullOwnPtr<Wasm::ModuleInstance>> instantiate_module(JS::VM&, Wasm::Module const&, GC::Ptr<JS::Object> import_object);
|
2024-10-26 14:09:25 -06:00
|
|
|
JS::ThrowCompletionOr<NonnullRefPtr<CompiledWebAssemblyModule>> compile_a_webassembly_module(JS::VM&, ByteBuffer);
|
2025-03-18 18:08:02 -05:00
|
|
|
JS::NativeFunction* create_native_function(JS::VM&, Wasm::FunctionAddress address, String const& name, Instance* instance = nullptr);
|
2024-04-25 19:09:34 +01:00
|
|
|
JS::ThrowCompletionOr<Wasm::Value> to_webassembly_value(JS::VM&, JS::Value value, Wasm::ValueType const& type);
|
2024-08-17 17:02:59 -07:00
|
|
|
Wasm::Value default_webassembly_value(JS::VM&, Wasm::ValueType type);
|
2024-08-04 08:06:50 -07:00
|
|
|
JS::Value to_js_value(JS::VM&, Wasm::Value& wasm_value, Wasm::ValueType type);
|
2024-04-25 19:09:34 +01:00
|
|
|
|
2024-11-15 04:01:23 +13:00
|
|
|
extern HashMap<GC::Ptr<JS::Object>, WebAssemblyCache> s_caches;
|
2023-03-16 14:11:21 -04:00
|
|
|
|
|
|
|
}
|
|
|
|
|
2023-03-16 12:59:32 -04:00
|
|
|
}
|