/* * Copyright (c) 2021, Ali Mohammad Pur * Copyright (c) 2023, Tim Flynn * * SPDX-License-Identifier: BSD-2-Clause */ #include #include #include #include #include #include #include #include #include #include #include #include namespace Web::WebAssembly { GC_DEFINE_ALLOCATOR(Instance); WebIDL::ExceptionOr> Instance::construct_impl(JS::Realm& realm, Module& module, Optional>& import_object_handle) { GC::Ptr import_object = import_object_handle.has_value() ? import_object_handle.value().ptr() : nullptr; auto& vm = realm.vm(); auto module_instance = TRY(Detail::instantiate_module(vm, module.compiled_module()->module, import_object)); return realm.create(realm, move(module_instance)); } Instance::Instance(JS::Realm& realm, NonnullOwnPtr module_instance) : Bindings::PlatformObject(realm) , m_exports(Object::create(realm, nullptr)) , m_module_instance(move(module_instance)) { } void Instance::initialize(JS::Realm& realm) { auto& vm = this->vm(); Base::initialize(realm); WEB_SET_PROTOTYPE_FOR_INTERFACE_WITH_CUSTOM_NAME(Instance, WebAssembly.Instance); auto& cache = Detail::get_cache(realm); // https://webassembly.github.io/spec/js-api/#create-an-exports-object for (auto& export_ : m_module_instance->exports()) { export_.value().visit( [&](Wasm::FunctionAddress const& address) { Optional> object = m_function_instances.get(address); if (!object.has_value()) { object = Detail::create_native_function(vm, address, export_.name(), this); m_function_instances.set(address, *object); } m_exports->define_direct_property(export_.name(), *object, JS::default_attributes); }, [&](Wasm::GlobalAddress const& address) { Optional> object = cache.get_global_instance(address); if (!object.has_value()) { object = realm.create(realm, address); } m_exports->define_direct_property(export_.name(), *object, JS::default_attributes); }, [&](Wasm::MemoryAddress const& address) { Optional> object = m_memory_instances.get(address); if (!object.has_value()) { object = realm.create(realm, address, Memory::Shared::No); m_memory_instances.set(address, *object); } m_exports->define_direct_property(export_.name(), *object, JS::default_attributes); }, [&](Wasm::TableAddress const& address) { Optional> object = m_table_instances.get(address); if (!object.has_value()) { object = realm.create(realm, address); m_table_instances.set(address, *object); } m_exports->define_direct_property(export_.name(), *object, JS::default_attributes); }); } MUST(m_exports->set_integrity_level(IntegrityLevel::Frozen)); } void Instance::visit_edges(Visitor& visitor) { Base::visit_edges(visitor); visitor.visit(m_exports); visitor.visit(m_function_instances); visitor.visit(m_memory_instances); visitor.visit(m_table_instances); } }