| 
									
										
										
										
											2021-09-14 19:35:23 +02:00
										 |  |  | /*
 | 
					
						
							| 
									
										
										
										
											2025-03-16 17:41:24 -05:00
										 |  |  |  * Copyright (c) 2021-2025, Andreas Kling <andreas@ladybird.org> | 
					
						
							| 
									
										
										
										
											2022-01-18 19:29:17 +01:00
										 |  |  |  * Copyright (c) 2022, David Tuin <davidot@serenityos.org> | 
					
						
							| 
									
										
										
										
											2021-09-14 19:35:23 +02:00
										 |  |  |  * | 
					
						
							|  |  |  |  * SPDX-License-Identifier: BSD-2-Clause | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #pragma once
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-02 19:27:29 -04:00
										 |  |  | #include <AK/Utf16FlyString.h>
 | 
					
						
							| 
									
										
										
										
											2024-11-15 04:01:23 +13:00
										 |  |  | #include <LibGC/Ptr.h>
 | 
					
						
							| 
									
										
										
										
											2025-07-19 13:49:30 -07:00
										 |  |  | #include <LibJS/Export.h>
 | 
					
						
							| 
									
										
										
										
											2023-12-02 22:56:47 +01:00
										 |  |  | #include <LibJS/ModuleLoading.h>
 | 
					
						
							| 
									
										
										
										
											2021-10-14 16:12:53 +01:00
										 |  |  | #include <LibJS/Runtime/Environment.h>
 | 
					
						
							| 
									
										
										
										
											2021-09-14 19:35:23 +02:00
										 |  |  | #include <LibJS/Runtime/Realm.h>
 | 
					
						
							| 
									
										
										
										
											2022-10-02 21:18:33 +02:00
										 |  |  | #include <LibJS/Script.h>
 | 
					
						
							| 
									
										
										
										
											2021-09-14 19:35:23 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | namespace JS { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-18 19:29:17 +01:00
										 |  |  | struct ResolvedBinding { | 
					
						
							|  |  |  |     enum Type { | 
					
						
							|  |  |  |         BindingName, | 
					
						
							|  |  |  |         Namespace, | 
					
						
							|  |  |  |         Ambiguous, | 
					
						
							|  |  |  |         Null, | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     static ResolvedBinding null() | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         return {}; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     static ResolvedBinding ambiguous() | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         ResolvedBinding binding; | 
					
						
							|  |  |  |         binding.type = Ambiguous; | 
					
						
							|  |  |  |         return binding; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Type type { Null }; | 
					
						
							| 
									
										
										
										
											2024-11-15 04:01:23 +13:00
										 |  |  |     GC::Ptr<Module> module; | 
					
						
							| 
									
										
										
										
											2025-08-02 19:27:29 -04:00
										 |  |  |     Utf16FlyString export_name; | 
					
						
							| 
									
										
										
										
											2022-01-18 19:29:17 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     bool is_valid() const | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         return type == BindingName || type == Namespace; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     bool is_namespace() const | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         return type == Namespace; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     bool is_ambiguous() const | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         return type == Ambiguous; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-12-03 11:25:12 +01:00
										 |  |  | // https://tc39.es/ecma262/#graphloadingstate-record
 | 
					
						
							| 
									
										
										
										
											2025-07-19 10:41:08 -07:00
										 |  |  | struct GraphLoadingState : public Cell { | 
					
						
							| 
									
										
										
										
											2024-11-15 04:01:23 +13:00
										 |  |  |     GC_CELL(GraphLoadingState, Cell); | 
					
						
							|  |  |  |     GC_DECLARE_ALLOCATOR(GraphLoadingState); | 
					
						
							| 
									
										
										
										
											2023-12-03 11:25:12 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  |     struct HostDefined : Cell { | 
					
						
							| 
									
										
										
										
											2024-11-15 04:01:23 +13:00
										 |  |  |         GC_CELL(HostDefined, Cell); | 
					
						
							| 
									
										
										
										
											2023-12-03 11:25:12 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     public: | 
					
						
							|  |  |  |         virtual ~HostDefined() = default; | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-15 04:01:23 +13:00
										 |  |  |     GC::Ptr<PromiseCapability> promise_capability; // [[PromiseCapability]]
 | 
					
						
							|  |  |  |     bool is_loading { false };                     // [[IsLoading]]
 | 
					
						
							|  |  |  |     size_t pending_module_count { 0 };             // [[PendingModulesCount]]
 | 
					
						
							|  |  |  |     HashTable<GC::Ptr<CyclicModule>> visited;      // [[Visited]]
 | 
					
						
							|  |  |  |     GC::Ptr<HostDefined> host_defined;             // [[HostDefined]]
 | 
					
						
							| 
									
										
										
										
											2023-12-03 11:25:12 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | private: | 
					
						
							| 
									
										
										
										
											2024-11-15 04:01:23 +13:00
										 |  |  |     GraphLoadingState(GC::Ptr<PromiseCapability> promise_capability, bool is_loading, size_t pending_module_count, HashTable<GC::Ptr<CyclicModule>> visited, GC::Ptr<HostDefined> host_defined) | 
					
						
							| 
									
										
										
										
											2023-12-03 11:25:12 +01:00
										 |  |  |         : promise_capability(move(promise_capability)) | 
					
						
							|  |  |  |         , is_loading(is_loading) | 
					
						
							|  |  |  |         , pending_module_count(pending_module_count) | 
					
						
							|  |  |  |         , visited(move(visited)) | 
					
						
							|  |  |  |         , host_defined(move(host_defined)) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     virtual void visit_edges(Cell::Visitor&) override; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-14 19:35:23 +02:00
										 |  |  | // 16.2.1.4 Abstract Module Records, https://tc39.es/ecma262/#sec-abstract-module-records
 | 
					
						
							| 
									
										
										
										
											2025-07-19 10:41:08 -07:00
										 |  |  | class Module : public Cell { | 
					
						
							| 
									
										
										
										
											2024-11-15 04:01:23 +13:00
										 |  |  |     GC_CELL(Module, Cell); | 
					
						
							|  |  |  |     GC_DECLARE_ALLOCATOR(Module); | 
					
						
							| 
									
										
										
										
											2022-09-05 14:31:25 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-14 19:35:23 +02:00
										 |  |  | public: | 
					
						
							| 
									
										
										
										
											2022-09-05 14:31:25 +02:00
										 |  |  |     virtual ~Module() override; | 
					
						
							| 
									
										
										
										
											2021-09-14 19:35:23 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-05 14:31:25 +02:00
										 |  |  |     Realm& realm() { return *m_realm; } | 
					
						
							|  |  |  |     Realm const& realm() const { return *m_realm; } | 
					
						
							| 
									
										
										
										
											2021-09-14 19:35:23 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-18 19:21:42 +01:00
										 |  |  |     StringView filename() const { return m_filename; } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-03-16 17:41:24 -05:00
										 |  |  |     GC::Ptr<ModuleEnvironment> environment() { return m_environment; } | 
					
						
							| 
									
										
										
										
											2022-01-18 19:29:17 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-02 21:18:33 +02:00
										 |  |  |     Script::HostDefined* host_defined() const { return m_host_defined; } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-04-28 15:03:56 -04:00
										 |  |  |     GC::Ref<Object> get_module_namespace(VM& vm); | 
					
						
							| 
									
										
										
										
											2022-01-18 19:29:17 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     virtual ThrowCompletionOr<void> link(VM& vm) = 0; | 
					
						
							| 
									
										
										
										
											2025-04-28 15:03:56 -04:00
										 |  |  |     virtual ThrowCompletionOr<GC::Ref<Promise>> evaluate(VM& vm) = 0; | 
					
						
							| 
									
										
										
										
											2022-01-18 19:29:17 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-02 19:27:29 -04:00
										 |  |  |     Vector<Utf16FlyString> get_exported_names(VM& vm); | 
					
						
							|  |  |  |     virtual Vector<Utf16FlyString> get_exported_names(VM& vm, HashTable<Module const*>& export_star_set) = 0; | 
					
						
							| 
									
										
										
										
											2025-04-28 15:03:56 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-02 19:27:29 -04:00
										 |  |  |     virtual ResolvedBinding resolve_export(VM& vm, Utf16FlyString const& export_name, Vector<ResolvedBinding> resolve_set = {}) = 0; | 
					
						
							| 
									
										
										
										
											2022-01-18 19:29:17 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     virtual ThrowCompletionOr<u32> inner_module_linking(VM& vm, Vector<Module*>& stack, u32 index); | 
					
						
							|  |  |  |     virtual ThrowCompletionOr<u32> inner_module_evaluation(VM& vm, Vector<Module*>& stack, u32 index); | 
					
						
							| 
									
										
										
										
											2021-09-14 19:35:23 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-15 04:01:23 +13:00
										 |  |  |     virtual PromiseCapability& load_requested_modules(GC::Ptr<GraphLoadingState::HostDefined>) = 0; | 
					
						
							| 
									
										
										
										
											2023-12-03 11:25:12 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-14 19:35:23 +02:00
										 |  |  | protected: | 
					
						
							| 
									
										
										
										
											2023-12-16 17:49:34 +03:30
										 |  |  |     Module(Realm&, ByteString filename, Script::HostDefined* host_defined = nullptr); | 
					
						
							| 
									
										
										
										
											2022-01-18 19:29:17 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-05 14:31:25 +02:00
										 |  |  |     virtual void visit_edges(Cell::Visitor&) override; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-03-16 17:41:24 -05:00
										 |  |  |     void set_environment(GC::Ref<ModuleEnvironment> environment) | 
					
						
							| 
									
										
										
										
											2022-01-18 19:29:17 +01:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2022-09-05 14:31:25 +02:00
										 |  |  |         m_environment = environment; | 
					
						
							| 
									
										
										
										
											2022-01-18 19:29:17 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-09-14 19:35:23 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | private: | 
					
						
							| 
									
										
										
										
											2025-08-02 19:27:29 -04:00
										 |  |  |     GC::Ref<Object> module_namespace_create(Vector<Utf16FlyString> unambiguous_names); | 
					
						
							| 
									
										
										
										
											2025-05-04 10:38:13 -04:00
										 |  |  |     ThrowCompletionOr<void> evaluate_module_sync(VM&); | 
					
						
							| 
									
										
										
										
											2021-09-14 19:35:23 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-18 19:29:17 +01:00
										 |  |  |     // These handles are only safe as long as the VM they live in is valid.
 | 
					
						
							|  |  |  |     // But evaluated modules SHOULD be stored in the VM so unless you intentionally
 | 
					
						
							|  |  |  |     // destroy the VM but keep the modules this should not happen. Because VM
 | 
					
						
							|  |  |  |     // stores modules with a RefPtr we cannot just store the VM as that leads to
 | 
					
						
							|  |  |  |     // cycles.
 | 
					
						
							| 
									
										
										
										
											2024-11-15 04:01:23 +13:00
										 |  |  |     GC::Ptr<Realm> m_realm;                          // [[Realm]]
 | 
					
						
							| 
									
										
										
										
											2025-03-16 17:41:24 -05:00
										 |  |  |     GC::Ptr<ModuleEnvironment> m_environment;        // [[Environment]]
 | 
					
						
							| 
									
										
										
										
											2024-11-15 04:01:23 +13:00
										 |  |  |     GC::Ptr<Object> m_namespace;                     // [[Namespace]]
 | 
					
						
							| 
									
										
										
										
											2022-10-02 21:18:33 +02:00
										 |  |  |     Script::HostDefined* m_host_defined { nullptr }; // [[HostDefined]]
 | 
					
						
							| 
									
										
										
										
											2022-01-18 19:21:42 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // Needed for potential lookups of modules.
 | 
					
						
							| 
									
										
										
										
											2023-12-16 17:49:34 +03:30
										 |  |  |     ByteString m_filename; | 
					
						
							| 
									
										
										
										
											2021-09-14 19:35:23 +02:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-10-29 02:53:53 +01:00
										 |  |  | class CyclicModule; | 
					
						
							|  |  |  | struct GraphLoadingState; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-07-19 13:49:30 -07:00
										 |  |  | JS_API void finish_loading_imported_module(ImportedModuleReferrer, ModuleRequest const&, ImportedModulePayload, ThrowCompletionOr<GC::Ref<Module>> const&); | 
					
						
							| 
									
										
										
										
											2023-10-29 02:53:53 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-14 19:35:23 +02:00
										 |  |  | } |