/* * Copyright (c) 2021, Ali Mohammad Pur * * SPDX-License-Identifier: BSD-2-Clause */ #include #include #include #include #include #include #include namespace Web::Bindings { WebAssemblyTableConstructor::WebAssemblyTableConstructor(JS::Realm& realm) : NativeFunction(*realm.intrinsics().function_prototype()) { } WebAssemblyTableConstructor::~WebAssemblyTableConstructor() = default; JS::ThrowCompletionOr WebAssemblyTableConstructor::call() { return vm().throw_completion(JS::ErrorType::ConstructorWithoutNew, "WebAssembly.Table"); } JS::ThrowCompletionOr WebAssemblyTableConstructor::construct(FunctionObject&) { auto& vm = this->vm(); auto& realm = *vm.current_realm(); auto descriptor = TRY(vm.argument(0).to_object(vm)); auto element_value = TRY(descriptor->get("element")); if (!element_value.is_string()) return vm.throw_completion(JS::ErrorType::InvalidHint, element_value.to_string_without_side_effects()); auto& element = element_value.as_string().string(); Optional reference_type; if (element == "anyfunc"sv) reference_type = Wasm::ValueType(Wasm::ValueType::FunctionReference); else if (element == "externref"sv) reference_type = Wasm::ValueType(Wasm::ValueType::ExternReference); if (!reference_type.has_value()) return vm.throw_completion(JS::ErrorType::InvalidHint, element); auto initial_value = TRY(descriptor->get("initial")); auto maximum_value = TRY(descriptor->get("maximum")); auto initial = TRY(initial_value.to_u32(vm)); Optional maximum; if (!maximum_value.is_undefined()) maximum = TRY(maximum_value.to_u32(vm)); if (maximum.has_value() && maximum.value() < initial) return vm.throw_completion("maximum should be larger than or equal to initial"); auto value_value = TRY(descriptor->get("value")); auto reference_value = TRY([&]() -> JS::ThrowCompletionOr { if (value_value.is_undefined()) return Wasm::Value(*reference_type, 0ull); return to_webassembly_value(vm, value_value, *reference_type); }()); auto& reference = reference_value.value().get(); auto address = WebAssemblyObject::s_abstract_machine.store().allocate(Wasm::TableType { *reference_type, Wasm::Limits { initial, maximum } }); if (!address.has_value()) return vm.throw_completion("Wasm Table allocation failed"); auto& table = *WebAssemblyObject::s_abstract_machine.store().get(*address); for (auto& element : table.elements()) element = reference; return vm.heap().allocate(realm, realm, *address); } void WebAssemblyTableConstructor::initialize(JS::Realm& realm) { auto& vm = this->vm(); auto& window = verify_cast(realm.global_object()); NativeFunction::initialize(realm); define_direct_property(vm.names.prototype, &window.ensure_web_prototype("WebAssemblyTablePrototype"), 0); define_direct_property(vm.names.length, JS::Value(1), JS::Attribute::Configurable); } }