| 
									
										
										
										
											2021-05-17 00:16:44 +04:30
										 |  |  | /*
 | 
					
						
							|  |  |  |  * Copyright (c) 2021, Ali Mohammad Pur <mpfard@serenityos.org> | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * SPDX-License-Identifier: BSD-2-Clause | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #pragma once
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <LibJS/Runtime/Object.h>
 | 
					
						
							|  |  |  | #include <LibWasm/AbstractMachine/AbstractMachine.h>
 | 
					
						
							|  |  |  | #include <LibWeb/Forward.h>
 | 
					
						
							| 
									
										
										
										
											2021-05-17 21:42:56 +04:30
										 |  |  | #include <LibWeb/WebAssembly/WebAssemblyObjectPrototype.h>
 | 
					
						
							| 
									
										
										
										
											2021-05-17 00:16:44 +04:30
										 |  |  | 
 | 
					
						
							|  |  |  | namespace Web::Bindings { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-21 04:12:15 +04:30
										 |  |  | class WebAssemblyMemoryObject; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-17 00:16:44 +04:30
										 |  |  | class WebAssemblyObject final : public JS::Object { | 
					
						
							|  |  |  |     JS_OBJECT(WebAssemblyObject, JS::Object); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  |     explicit WebAssemblyObject(JS::GlobalObject&); | 
					
						
							|  |  |  |     virtual void initialize(JS::GlobalObject&) override; | 
					
						
							|  |  |  |     virtual ~WebAssemblyObject() override = default; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-21 04:12:15 +04:30
										 |  |  |     virtual void visit_edges(Cell::Visitor&) override; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-17 00:16:44 +04:30
										 |  |  |     struct CompiledWebAssemblyModule { | 
					
						
							|  |  |  |         explicit CompiledWebAssemblyModule(Wasm::Module&& module) | 
					
						
							|  |  |  |             : module(move(module)) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         Wasm::Module module; | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // FIXME: These should just be members of the module (instance) object,
 | 
					
						
							|  |  |  |     //        but the module needs to stick around while its instance is alive
 | 
					
						
							|  |  |  |     //        so ideally this would be a refcounted object, shared between
 | 
					
						
							|  |  |  |     //        WebAssemblyModuleObject's and WebAssemblyInstantiatedModuleObject's.
 | 
					
						
							| 
									
										
										
										
											2021-06-21 04:12:15 +04:30
										 |  |  |     struct ModuleCache { | 
					
						
							|  |  |  |         HashMap<Wasm::FunctionAddress, JS::Function*> function_instances; | 
					
						
							|  |  |  |         HashMap<Wasm::MemoryAddress, WebAssemblyMemoryObject*> memory_instances; | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  |     struct GlobalModuleCache { | 
					
						
							|  |  |  |         HashMap<Wasm::FunctionAddress, JS::NativeFunction*> function_instances; | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-17 00:16:44 +04:30
										 |  |  |     static NonnullOwnPtrVector<CompiledWebAssemblyModule> s_compiled_modules; | 
					
						
							|  |  |  |     static NonnullOwnPtrVector<Wasm::ModuleInstance> s_instantiated_modules; | 
					
						
							| 
									
										
										
										
											2021-06-21 04:12:15 +04:30
										 |  |  |     static Vector<ModuleCache> s_module_caches; | 
					
						
							|  |  |  |     static GlobalModuleCache s_global_cache; | 
					
						
							| 
									
										
										
										
											2021-05-17 00:16:44 +04:30
										 |  |  | 
 | 
					
						
							|  |  |  |     static Wasm::AbstractMachine s_abstract_machine; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | private: | 
					
						
							|  |  |  |     JS_DECLARE_NATIVE_FUNCTION(validate); | 
					
						
							|  |  |  |     JS_DECLARE_NATIVE_FUNCTION(compile); | 
					
						
							|  |  |  |     JS_DECLARE_NATIVE_FUNCTION(instantiate); | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class WebAssemblyModuleObject final : public JS::Object { | 
					
						
							|  |  |  |     JS_OBJECT(WebAssemblyModuleObject, JS::Object); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  |     explicit WebAssemblyModuleObject(JS::GlobalObject&, size_t index); | 
					
						
							|  |  |  |     virtual ~WebAssemblyModuleObject() override = default; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     size_t index() const { return m_index; } | 
					
						
							|  |  |  |     const Wasm::Module& module() const { return WebAssemblyObject::s_compiled_modules.at(m_index).module; } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | private: | 
					
						
							|  |  |  |     size_t m_index { 0 }; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class WebAssemblyInstanceObject final : public JS::Object { | 
					
						
							|  |  |  |     JS_OBJECT(WebAssemblyInstanceObject, JS::Object); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  |     explicit WebAssemblyInstanceObject(JS::GlobalObject&, size_t index); | 
					
						
							|  |  |  |     virtual void initialize(JS::GlobalObject&) override; | 
					
						
							|  |  |  |     virtual ~WebAssemblyInstanceObject() override = default; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     size_t index() const { return m_index; } | 
					
						
							|  |  |  |     Wasm::ModuleInstance& instance() const { return WebAssemblyObject::s_instantiated_modules.at(m_index); } | 
					
						
							| 
									
										
										
										
											2021-06-21 04:12:15 +04:30
										 |  |  |     auto& cache() { return WebAssemblyObject::s_module_caches.at(m_index); } | 
					
						
							| 
									
										
										
										
											2021-05-17 00:16:44 +04:30
										 |  |  | 
 | 
					
						
							|  |  |  |     void visit_edges(Cell::Visitor&) override; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     friend class WebAssemblyInstancePrototype; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | private: | 
					
						
							|  |  |  |     size_t m_index { 0 }; | 
					
						
							|  |  |  |     JS::Object* m_exports_object { nullptr }; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-17 21:42:56 +04:30
										 |  |  | class WebAssemblyMemoryObject final : public JS::Object { | 
					
						
							| 
									
										
										
										
											2021-06-21 08:42:58 +04:30
										 |  |  |     JS_OBJECT(WebAssemblyMemoryObject, JS::Object); | 
					
						
							| 
									
										
										
										
											2021-05-17 21:42:56 +04:30
										 |  |  | 
 | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  |     explicit WebAssemblyMemoryObject(JS::GlobalObject&, Wasm::MemoryAddress); | 
					
						
							|  |  |  |     virtual ~WebAssemblyMemoryObject() override = default; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     auto address() const { return m_address; } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | private: | 
					
						
							|  |  |  |     Wasm::MemoryAddress m_address; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-17 00:16:44 +04:30
										 |  |  | } |