2020-09-20 19:24:44 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								/*
  
						 
					
						
							
								
									
										
										
										
											2024-10-04 13:19:50 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								 *  Copyright  ( c )  2020 - 2023 ,  Andreas  Kling  < andreas @ ladybird . org > 
							 
						 
					
						
							
								
									
										
										
										
											2023-02-11 16:14:41 +00:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								 *  Copyright  ( c )  2020 - 2023 ,  Linus  Groh  < linusg @ serenityos . org > 
							 
						 
					
						
							
								
									
										
										
										
											2022-01-19 01:22:58 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								 *  Copyright  ( c )  2021 - 2022 ,  David  Tuin  < davidot @ serenityos . org > 
							 
						 
					
						
							
								
									
										
										
										
											2020-09-20 19:24:44 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								 * 
							 
						 
					
						
							
								
									
										
										
										
											2021-04-22 01:24:48 -07:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								 *  SPDX - License - Identifier :  BSD - 2 - Clause 
							 
						 
					
						
							
								
									
										
										
										
											2020-09-20 19:24:44 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								 */ 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2023-01-21 12:36:47 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								# include  <AK/Array.h> 
  
						 
					
						
							
								
									
										
											 
										
											
												LibJS: Add initial support for Promises
Almost a year after first working on this, it's finally done: an
implementation of Promises for LibJS! :^)
The core functionality is working and closely following the spec [1].
I mostly took the pseudo code and transformed it into C++ - if you read
and understand it, you will know how the spec implements Promises; and
if you read the spec first, the code will look very familiar.
Implemented functions are:
- Promise() constructor
- Promise.prototype.then()
- Promise.prototype.catch()
- Promise.prototype.finally()
- Promise.resolve()
- Promise.reject()
For the tests I added a new function to test-js's global object,
runQueuedPromiseJobs(), which calls vm.run_queued_promise_jobs().
By design, queued jobs normally only run after the script was fully
executed, making it improssible to test handlers in individual test()
calls by default [2].
Subsequent commits include integrations into LibWeb and js(1) -
pretty-printing, running queued promise jobs when necessary.
This has an unusual amount of dbgln() statements, all hidden behind the
PROMISE_DEBUG flag - I'm leaving them in for now as they've been very
useful while debugging this, things can get quite complex with so many
asynchronously executed functions.
I've not extensively explored use of these APIs for promise-based
functionality in LibWeb (fetch(), Notification.requestPermission()
etc.), but we'll get there in due time.
[1]: https://tc39.es/ecma262/#sec-promise-objects
[2]: https://tc39.es/ecma262/#sec-jobs-and-job-queues
											 
										 
										
											2021-04-01 22:13:29 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								# include  <AK/Debug.h> 
  
						 
					
						
							
								
									
										
										
										
											2022-01-19 01:22:58 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								# include  <AK/LexicalPath.h> 
  
						 
					
						
							
								
									
										
										
										
											2020-09-27 15:18:55 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								# include  <AK/ScopeGuard.h> 
  
						 
					
						
							
								
									
										
										
										
											2023-02-11 16:14:41 +00:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								# include  <AK/String.h> 
  
						 
					
						
							
								
									
										
										
										
											2020-09-27 17:24:14 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								# include  <AK/StringBuilder.h> 
  
						 
					
						
							
								
									
										
										
										
											2024-11-24 17:23:31 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								# include  <AK/Time.h> 
  
						 
					
						
							
								
									
										
										
										
											2023-03-22 02:35:30 +11:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								# include  <LibFileSystem/FileSystem.h> 
  
						 
					
						
							
								
									
										
										
										
											2022-11-23 13:41:50 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								# include  <LibJS/AST.h> 
  
						 
					
						
							
								
									
										
										
										
											2023-06-22 15:59:18 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								# include  <LibJS/Bytecode/Interpreter.h> 
  
						 
					
						
							
								
									
										
										
										
											2021-06-22 13:30:48 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								# include  <LibJS/Runtime/AbstractOperations.h> 
  
						 
					
						
							
								
									
										
										
										
											2020-12-05 16:38:29 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								# include  <LibJS/Runtime/Array.h> 
  
						 
					
						
							
								
									
										
										
										
											2023-10-14 19:10:55 -04:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								# include  <LibJS/Runtime/ArrayBuffer.h> 
  
						 
					
						
							
								
									
										
										
										
											2021-08-09 15:21:15 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								# include  <LibJS/Runtime/BoundFunction.h> 
  
						 
					
						
							
								
									
										
											 
										
											
												LibJS: Implement [[Call]] and [[Construct]] internal slots properly
This patch implements:
- Spec compliant [[Call]] and [[Construct]] internal slots, as virtual
  FunctionObject::internal_{call,construct}(). These effectively replace
  the old virtual FunctionObject::{call,construct}(), but with several
  advantages:
  - Clear and consistent naming, following the object internal methods
  - Use of completions
  - internal_construct() returns an Object, and not Value! This has been
    a source of confusion for a long time, since in the spec there's
    always an Object returned but the Value return type in LibJS meant
    that this could not be fully trusted and something could screw you
    over.
  - Arguments are passed explicitly in form of a MarkedValueList,
    allowing manipulation (BoundFunction). We still put them on the
    execution context as a lot of code depends on it (VM::arguments()),
    but not from the Call() / Construct() AOs anymore, which now allows
    for bypassing them and invoking [[Call]] / [[Construct]] directly.
    Nothing but Call() / Construct() themselves do that at the moment,
    but future additions to ECMA262 or already existing web specs might.
- Spec compliant, standalone Call() and Construct() AOs: currently the
  closest we have is VM::{call,construct}(), but those try to cater to
  all the different function object subclasses at once, resulting in a
  horrible mess and calling AOs with functions they should never be
  called with; most prominently PrepareForOrdinaryCall and
  OrdinaryCallBindThis, which are only for ECMAScriptFunctionObject.
As a result this also contains an implicit optimization: we no longer
need to create a new function environment for NativeFunctions - which,
worth mentioning, is what started this whole crusade in the first place
:^)
											 
										 
										
											2021-10-08 20:37:21 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								# include  <LibJS/Runtime/Completion.h> 
  
						 
					
						
							
								
									
										
										
										
											2021-09-24 22:40:38 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								# include  <LibJS/Runtime/ECMAScriptFunctionObject.h> 
  
						 
					
						
							
								
									
										
										
										
											2020-09-27 15:18:55 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								# include  <LibJS/Runtime/Error.h> 
  
						 
					
						
							
								
									
										
										
										
											2021-06-15 22:16:17 +03:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								# include  <LibJS/Runtime/FinalizationRegistry.h> 
  
						 
					
						
							
								
									
										
										
										
											2021-07-01 12:24:46 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								# include  <LibJS/Runtime/FunctionEnvironment.h> 
  
						 
					
						
							
								
									
										
										
										
											2023-06-24 09:30:46 -04:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								# include  <LibJS/Runtime/Iterator.h> 
  
						 
					
						
							
								
									
										
											 
										
											
												LibJS: Add initial support for Promises
Almost a year after first working on this, it's finally done: an
implementation of Promises for LibJS! :^)
The core functionality is working and closely following the spec [1].
I mostly took the pseudo code and transformed it into C++ - if you read
and understand it, you will know how the spec implements Promises; and
if you read the spec first, the code will look very familiar.
Implemented functions are:
- Promise() constructor
- Promise.prototype.then()
- Promise.prototype.catch()
- Promise.prototype.finally()
- Promise.resolve()
- Promise.reject()
For the tests I added a new function to test-js's global object,
runQueuedPromiseJobs(), which calls vm.run_queued_promise_jobs().
By design, queued jobs normally only run after the script was fully
executed, making it improssible to test handlers in individual test()
calls by default [2].
Subsequent commits include integrations into LibWeb and js(1) -
pretty-printing, running queued promise jobs when necessary.
This has an unusual amount of dbgln() statements, all hidden behind the
PROMISE_DEBUG flag - I'm leaving them in for now as they've been very
useful while debugging this, things can get quite complex with so many
asynchronously executed functions.
I've not extensively explored use of these APIs for promise-based
functionality in LibWeb (fetch(), Notification.requestPermission()
etc.), but we'll get there in due time.
[1]: https://tc39.es/ecma262/#sec-promise-objects
[2]: https://tc39.es/ecma262/#sec-jobs-and-job-queues
											 
										 
										
											2021-04-01 22:13:29 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								# include  <LibJS/Runtime/NativeFunction.h> 
  
						 
					
						
							
								
									
										
										
										
											2022-10-02 10:59:22 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								# include  <LibJS/Runtime/PromiseCapability.h> 
  
						 
					
						
							
								
									
										
										
										
											2020-09-27 15:18:55 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								# include  <LibJS/Runtime/Reference.h> 
  
						 
					
						
							
								
									
										
										
										
											2020-09-22 16:18:51 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								# include  <LibJS/Runtime/Symbol.h> 
  
						 
					
						
							
								
									
										
										
										
											2024-11-24 17:23:31 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								# include  <LibJS/Runtime/Temporal/Instant.h> 
  
						 
					
						
							
								
									
										
										
										
											2020-09-20 19:24:44 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								# include  <LibJS/Runtime/VM.h> 
  
						 
					
						
							
								
									
										
										
										
											2022-01-19 01:22:58 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								# include  <LibJS/SourceTextModule.h> 
  
						 
					
						
							
								
									
										
										
										
											2022-01-27 14:51:21 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								# include  <LibJS/SyntheticModule.h> 
  
						 
					
						
							
								
									
										
										
										
											2020-09-20 19:24:44 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								namespace  JS  {  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2025-04-24 15:58:00 +12:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								NonnullRefPtr < VM >  VM : : create ( )  
						 
					
						
							
								
									
										
										
										
											2020-09-20 19:24:44 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								{  
						 
					
						
							
								
									
										
										
										
											2023-03-17 10:21:51 -04:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    ErrorMessages  error_messages  { } ; 
							 
						 
					
						
							
								
									
										
										
										
											2025-03-18 18:08:02 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    error_messages [ to_underlying ( ErrorMessage : : OutOfMemory ) ]  =  ErrorType : : OutOfMemory . message ( ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2023-03-17 10:21:51 -04:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2025-04-24 15:58:00 +12:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    auto  vm  =  adopt_ref ( * new  VM ( move ( error_messages ) ) ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2023-03-17 10:21:51 -04:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    WellKnownSymbols  well_known_symbols  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								# define __JS_ENUMERATE(SymbolName, snake_name) \ 
  
						 
					
						
							
								
									
										
										
										
											2025-08-02 19:27:29 -04:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    Symbol : : create ( * vm ,  " Symbol. "  # SymbolName # # _utf16 ,  false ) , 
							 
						 
					
						
							
								
									
										
										
										
											2023-03-17 10:21:51 -04:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        JS_ENUMERATE_WELL_KNOWN_SYMBOLS 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								# undef __JS_ENUMERATE 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    } ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    vm - > set_well_known_symbols ( move ( well_known_symbols ) ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    return  vm ; 
							 
						 
					
						
							
								
									
										
										
										
											2020-09-20 19:24:44 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								}  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2024-02-08 18:52:03 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								template < size_t . . .  code_points >  
						 
					
						
							
								
									
										
										
										
											2023-01-21 12:36:47 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								static  constexpr  auto  make_single_ascii_character_strings ( IndexSequence < code_points . . . > )  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								{  
						 
					
						
							
								
									
										
										
										
											2024-02-08 18:52:03 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    return  AK : : Array  {  ( String : : from_code_point ( static_cast < u32 > ( code_points ) ) ) . . .  } ; 
							 
						 
					
						
							
								
									
										
										
										
											2023-01-21 12:36:47 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								}  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								static  constexpr  auto  single_ascii_character_strings  =  make_single_ascii_character_strings ( MakeIndexSequence < 128 > ( ) ) ;  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2025-04-24 15:58:00 +12:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								VM : : VM ( ErrorMessages  error_messages )  
						 
					
						
							
								
									
										
										
										
											2024-11-15 04:01:23 +13:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    :  m_heap ( this ,  [ this ] ( HashMap < GC : : Cell * ,  GC : : HeapRoot > &  roots )  { 
							 
						 
					
						
							
								
									
										
										
										
											2024-11-14 20:17:33 +13:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        gather_roots ( roots ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    } ) 
							 
						 
					
						
							
								
									
										
										
										
											2023-03-17 10:21:51 -04:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    ,  m_error_messages ( move ( error_messages ) ) 
							 
						 
					
						
							
								
									
										
										
										
											2020-09-20 19:24:44 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								{  
						 
					
						
							
								
									
										
										
										
											2023-06-22 15:59:18 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    m_bytecode_interpreter  =  make < Bytecode : : Interpreter > ( * this ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2024-11-14 06:13:46 +13:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    m_empty_string  =  m_heap . allocate < PrimitiveString > ( String  { } ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2023-01-21 12:36:47 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2025-04-14 23:42:31 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    cached_strings  =  { 
							 
						 
					
						
							
								
									
										
										
										
											2025-03-28 13:55:39 +00:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        . number  =  m_heap . allocate < PrimitiveString > ( " number " _string ) , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        . undefined  =  m_heap . allocate < PrimitiveString > ( " undefined " _string ) , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        . object  =  m_heap . allocate < PrimitiveString > ( " object " _string ) , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        . string  =  m_heap . allocate < PrimitiveString > ( " string " _string ) , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        . symbol  =  m_heap . allocate < PrimitiveString > ( " symbol " _string ) , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        . boolean  =  m_heap . allocate < PrimitiveString > ( " boolean " _string ) , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        . bigint  =  m_heap . allocate < PrimitiveString > ( " bigint " _string ) , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        . function  =  m_heap . allocate < PrimitiveString > ( " function " _string ) , 
							 
						 
					
						
							
								
									
										
										
										
											2025-04-14 23:42:31 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        . object_Object  =  m_heap . allocate < PrimitiveString > ( " [object Object] " _string ) , 
							 
						 
					
						
							
								
									
										
										
										
											2024-07-23 09:42:00 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    } ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2023-01-21 12:36:47 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    for  ( size_t  i  =  0 ;  i  <  single_ascii_character_strings . size ( ) ;  + + i ) 
							 
						 
					
						
							
								
									
										
										
										
											2024-11-14 06:13:46 +13:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        m_single_ascii_character_strings [ i ]  =  m_heap . allocate < PrimitiveString > ( single_ascii_character_strings [ i ] ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2020-10-22 17:43:48 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2022-02-06 03:46:45 +00:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    // Default hook implementations. These can be overridden by the host, for example, LibWeb overrides the default hooks to place promise jobs on the microtask queue.
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    host_promise_rejection_tracker  =  [ this ] ( Promise &  promise ,  Promise : : RejectionOperation  operation )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        promise_rejection_tracker ( promise ,  operation ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    } ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2023-11-27 12:56:20 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    host_call_job_callback  =  [ this ] ( JobCallback &  job_callback ,  Value  this_value ,  ReadonlySpan < Value >  arguments )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return  call_job_callback ( * this ,  job_callback ,  this_value ,  arguments ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2022-02-06 03:46:45 +00:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    } ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    host_enqueue_finalization_registry_cleanup_job  =  [ this ] ( FinalizationRegistry &  finalization_registry )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        enqueue_finalization_registry_cleanup_job ( finalization_registry ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    } ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2024-11-15 04:01:23 +13:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    host_enqueue_promise_job  =  [ this ] ( GC : : Ref < GC : : Function < ThrowCompletionOr < Value > ( ) > >  job ,  Realm *  realm )  { 
							 
						 
					
						
							
								
									
										
										
										
											2024-03-25 14:18:22 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        enqueue_promise_job ( job ,  realm ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2022-02-06 03:46:45 +00:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    } ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    host_make_job_callback  =  [ ] ( FunctionObject &  function_object )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return  make_job_callback ( function_object ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    } ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2024-11-15 04:01:23 +13:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    host_load_imported_module  =  [ this ] ( ImportedModuleReferrer  referrer ,  ModuleRequest  const &  module_request ,  GC : : Ptr < GraphLoadingState : : HostDefined >  load_state ,  ImportedModulePayload  payload )  - >  void  { 
							 
						 
					
						
							
								
									
										
										
										
											2023-12-02 22:56:47 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        return  load_imported_module ( referrer ,  module_request ,  load_state ,  move ( payload ) ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2022-01-18 19:39:36 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    } ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2022-01-18 19:40:43 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    host_get_import_meta_properties  =  [ & ] ( SourceTextModule  const & )  - >  HashMap < PropertyKey ,  Value >  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return  { } ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    } ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    host_finalize_import_meta  =  [ & ] ( Object * ,  SourceTextModule  const & )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    } ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2023-12-02 16:20:01 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    host_get_supported_import_attributes  =  [ & ]  { 
							 
						 
					
						
							
								
									
										
										
										
											2025-08-06 11:12:58 -04:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        return  Vector < Utf16String >  {  " type " _utf16  } ; 
							 
						 
					
						
							
								
									
										
										
										
											2022-01-27 02:44:03 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    } ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2024-12-03 10:56:21 +00:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    // 1 HostGetCodeForEval ( argument ), https://tc39.es/proposal-dynamic-code-brand-checks/#sec-hostgetcodeforeval
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    host_get_code_for_eval  =  [ ] ( Object  const & )  - >  GC : : Ptr < PrimitiveString >  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        // The host-defined abstract operation HostGetCodeForEval takes argument argument (an Object) and returns a
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        // String or NO-CODE. It allows host environments to return a String of code from argument to be used by eval,
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        // rather than eval returning argument.
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        //
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        // argument represents the Object to be checked for code.
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        //
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        // The default implementation of HostGetCodeForEval is to return NO-CODE.
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return  { } ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    } ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    // 2 HostEnsureCanCompileStrings ( calleeRealm, parameterStrings, bodyString, codeString, compilationType, parameterArgs, bodyArg ), https://tc39.es/proposal-dynamic-code-brand-checks/#sec-hostensurecancompilestrings
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    host_ensure_can_compile_strings  =  [ ] ( Realm & ,  ReadonlySpan < String > ,  StringView ,  StringView ,  CompilationType ,  ReadonlySpan < Value > ,  Value )  - >  ThrowCompletionOr < void >  { 
							 
						 
					
						
							
								
									
										
										
										
											2024-11-03 20:18:56 +13:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        // The host-defined abstract operation HostEnsureCanCompileStrings takes arguments calleeRealm (a Realm Record),
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        // parameterStrings (a List of Strings), bodyString (a String), and direct (a Boolean) and returns either a normal
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        // completion containing unused or a throw completion.
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        //
 
							 
						 
					
						
							
								
									
										
										
										
											2022-04-10 00:55:45 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        // It allows host environments to block certain ECMAScript functions which allow developers to compile strings into ECMAScript code.
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        // An implementation of HostEnsureCanCompileStrings must conform to the following requirements:
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        //   - If the returned Completion Record is a normal completion, it must be a normal completion containing unused.
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        // The default implementation of HostEnsureCanCompileStrings is to return NormalCompletion(unused).
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return  { } ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    } ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2022-08-14 00:51:32 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    host_ensure_can_add_private_element  =  [ ] ( Object & )  - >  ThrowCompletionOr < void >  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        // The host-defined abstract operation HostEnsureCanAddPrivateElement takes argument O (an Object)
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        // and returns either a normal completion containing unused or a throw completion.
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        // It allows host environments to prevent the addition of private elements to particular host-defined exotic objects.
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        // An implementation of HostEnsureCanAddPrivateElement must conform to the following requirements:
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        // - If O is not a host-defined exotic object, this abstract operation must return NormalCompletion(unused) and perform no other steps.
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        // - Any two calls of this abstract operation with the same argument must return the same kind of Completion Record.
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        // The default implementation of HostEnsureCanAddPrivateElement is to return NormalCompletion(unused).
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return  { } ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        // This abstract operation is only invoked by ECMAScript hosts that are web browsers.
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        // NOTE: Since LibJS has no way of knowing whether the current environment is a browser we always
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        //       call HostEnsureCanAddPrivateElement when needed.
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    } ; 
							 
						 
					
						
							
								
									
										
										
										
											2023-10-14 19:10:55 -04:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2024-08-20 10:29:45 -04:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    // 25.1.3.8 HostResizeArrayBuffer ( buffer, newByteLength ), https://tc39.es/ecma262/#sec-hostresizearraybuffer
 
							 
						 
					
						
							
								
									
										
										
										
											2023-10-14 19:10:55 -04:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    host_resize_array_buffer  =  [ this ] ( ArrayBuffer &  buffer ,  size_t  new_byte_length )  - >  ThrowCompletionOr < HandledByHost >  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        // The host-defined abstract operation HostResizeArrayBuffer takes arguments buffer (an ArrayBuffer) and
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        // newByteLength (a non-negative integer) and returns either a normal completion containing either handled or
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        // unhandled, or a throw completion. It gives the host an opportunity to perform implementation-defined resizing
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        // of buffer. If the host chooses not to handle resizing of buffer, it may return unhandled for the default behaviour.
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        // The implementation of HostResizeArrayBuffer must conform to the following requirements:
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        // - The abstract operation does not detach buffer.
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        // - If the abstract operation completes normally with handled, buffer.[[ArrayBufferByteLength]] is newByteLength.
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        // The default implementation of HostResizeArrayBuffer is to return NormalCompletion(unhandled).
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2023-12-27 08:41:22 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        if  ( auto  result  =  buffer . buffer ( ) . try_resize ( new_byte_length ,  ByteBuffer : : ZeroFillNewElements : : Yes ) ;  result . is_error ( ) ) 
							 
						 
					
						
							
								
									
										
										
										
											2023-10-14 19:10:55 -04:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								            return  throw_completion < RangeError > ( ErrorType : : NotEnoughMemoryToAllocate ,  new_byte_length ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return  HandledByHost : : Handled ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    } ; 
							 
						 
					
						
							
								
									
										
										
										
											2024-09-08 11:15:07 -04:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2024-11-17 05:55:20 +13:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    // 3.6.1 HostInitializeShadowRealm ( realm, context, O ), https://tc39.es/proposal-shadowrealm/#sec-hostinitializeshadowrealm
 
							 
						 
					
						
							
								
									
										
										
										
											2024-11-03 21:29:30 +13:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    host_initialize_shadow_realm  =  [ ] ( Realm & ,  NonnullOwnPtr < ExecutionContext > ,  ShadowRealm & )  - >  ThrowCompletionOr < void >  { 
							 
						 
					
						
							
								
									
										
										
										
											2024-11-17 05:55:20 +13:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        // The host-defined abstract operation HostInitializeShadowRealm takes arguments realm (a Realm Record),
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        // context (an execution context), and O (a ShadowRealm object) and returns either a normal completion
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        // containing unused or a throw completion. It is used to inform the host of any newly created realms
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        // from the ShadowRealm constructor. The idea of this hook is to initialize host data structures related
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        // to the ShadowRealm, e.g., for module loading.
 
							 
						 
					
						
							
								
									
										
										
										
											2024-11-03 21:29:30 +13:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        //
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        // The host may use this hook to add properties to the ShadowRealm's global object. Those properties must be configurable.
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return  { } ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    } ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2024-11-24 17:23:31 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    // 2.3.1 HostSystemUTCEpochNanoseconds ( global ), https://tc39.es/proposal-temporal/#sec-hostsystemutcepochnanoseconds
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    host_system_utc_epoch_nanoseconds  =  [ ] ( Object  const & )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        // 1. Let ns be the approximate current UTC date and time, in nanoseconds since the epoch.
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        Crypto : : SignedBigInteger  nanoseconds  {  AK : : UnixDateTime : : now ( ) . nanoseconds_since_epoch ( )  } ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        // 2. Return the result of clamping ns between nsMinInstant and nsMaxInstant.
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        if  ( nanoseconds  <  Temporal : : NANOSECONDS_MIN_INSTANT ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            nanoseconds  =  Temporal : : NANOSECONDS_MIN_INSTANT ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        if  ( nanoseconds  >  Temporal : : NANOSECONDS_MAX_INSTANT ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            nanoseconds  =  Temporal : : NANOSECONDS_MAX_INSTANT ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return  nanoseconds ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    } ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2024-09-08 11:15:07 -04:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    // AD-HOC: Inform the host that we received a date string we were unable to parse.
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    host_unrecognized_date_string  =  [ ] ( StringView )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    } ; 
							 
						 
					
						
							
								
									
										
										
										
											2023-02-16 12:55:22 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								}  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2023-06-22 15:59:18 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								VM : : ~ VM ( )  =  default ;  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2025-08-07 19:31:52 -04:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								Utf16String  const &  VM : : error_message ( ErrorMessage  type )  const  
						 
					
						
							
								
									
										
										
										
											2023-02-16 12:55:22 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								{  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    VERIFY ( type  <  ErrorMessage : : __Count ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    auto  const &  message  =  m_error_messages [ to_underlying ( type ) ] ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    VERIFY ( ! message . is_empty ( ) ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    return  message ; 
							 
						 
					
						
							
								
									
										
										
										
											2020-09-20 19:24:44 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								}  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2023-12-13 10:24:30 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								struct  ExecutionContextRootsCollector  :  public  Cell : : Visitor  {  
						 
					
						
							
								
									
										
										
										
											2024-11-15 04:01:23 +13:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    virtual  void  visit_impl ( GC : : Cell &  cell )  override 
							 
						 
					
						
							
								
									
										
										
										
											2023-12-13 10:24:30 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        roots . set ( & cell ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    } 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    virtual  void  visit_possible_values ( ReadonlyBytes )  override 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        VERIFY_NOT_REACHED ( ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    } 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2024-11-15 04:01:23 +13:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    HashTable < GC : : Ptr < GC : : Cell > >  roots ; 
							 
						 
					
						
							
								
									
										
										
										
											2023-12-13 10:24:30 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								} ;  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2024-11-15 04:01:23 +13:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								void  VM : : gather_roots ( HashMap < GC : : Cell * ,  GC : : HeapRoot > &  roots )  
						 
					
						
							
								
									
										
										
										
											2020-09-21 13:47:33 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								{  
						 
					
						
							
								
									
										
										
										
											2024-11-15 04:01:23 +13:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    roots . set ( m_empty_string ,  GC : : HeapRoot  {  . type  =  GC : : HeapRoot : : Type : : VM  } ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2023-02-26 16:09:02 -07:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    for  ( auto  string  :  m_single_ascii_character_strings ) 
							 
						 
					
						
							
								
									
										
										
										
											2024-11-15 04:01:23 +13:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        roots . set ( string ,  GC : : HeapRoot  {  . type  =  GC : : HeapRoot : : Type : : VM  } ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2020-10-22 17:43:48 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2025-04-14 23:42:31 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    roots . set ( cached_strings . number ,  GC : : HeapRoot  {  . type  =  GC : : HeapRoot : : Type : : VM  } ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    roots . set ( cached_strings . undefined ,  GC : : HeapRoot  {  . type  =  GC : : HeapRoot : : Type : : VM  } ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    roots . set ( cached_strings . object ,  GC : : HeapRoot  {  . type  =  GC : : HeapRoot : : Type : : VM  } ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    roots . set ( cached_strings . string ,  GC : : HeapRoot  {  . type  =  GC : : HeapRoot : : Type : : VM  } ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    roots . set ( cached_strings . symbol ,  GC : : HeapRoot  {  . type  =  GC : : HeapRoot : : Type : : VM  } ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    roots . set ( cached_strings . boolean ,  GC : : HeapRoot  {  . type  =  GC : : HeapRoot : : Type : : VM  } ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    roots . set ( cached_strings . bigint ,  GC : : HeapRoot  {  . type  =  GC : : HeapRoot : : Type : : VM  } ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    roots . set ( cached_strings . function ,  GC : : HeapRoot  {  . type  =  GC : : HeapRoot : : Type : : VM  } ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    roots . set ( cached_strings . object_Object ,  GC : : HeapRoot  {  . type  =  GC : : HeapRoot : : Type : : VM  } ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2024-07-23 09:42:00 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2020-09-22 16:18:51 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								# define __JS_ENUMERATE(SymbolName, snake_name) \ 
  
						 
					
						
							
								
									
										
										
										
											2024-11-15 04:01:23 +13:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    roots . set ( m_well_known_symbols . snake_name ,  GC : : HeapRoot  {  . type  =  GC : : HeapRoot : : Type : : VM  } ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2020-09-22 16:18:51 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    JS_ENUMERATE_WELL_KNOWN_SYMBOLS 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								# undef __JS_ENUMERATE 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2022-12-06 20:58:15 +00:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    for  ( auto &  symbol  :  m_global_symbol_registry ) 
							 
						 
					
						
							
								
									
										
										
										
											2024-11-15 04:01:23 +13:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        roots . set ( symbol . value ,  GC : : HeapRoot  {  . type  =  GC : : HeapRoot : : Type : : VM  } ) ; 
							 
						 
					
						
							
								
									
										
											 
										
											
												LibJS: Add initial support for Promises
Almost a year after first working on this, it's finally done: an
implementation of Promises for LibJS! :^)
The core functionality is working and closely following the spec [1].
I mostly took the pseudo code and transformed it into C++ - if you read
and understand it, you will know how the spec implements Promises; and
if you read the spec first, the code will look very familiar.
Implemented functions are:
- Promise() constructor
- Promise.prototype.then()
- Promise.prototype.catch()
- Promise.prototype.finally()
- Promise.resolve()
- Promise.reject()
For the tests I added a new function to test-js's global object,
runQueuedPromiseJobs(), which calls vm.run_queued_promise_jobs().
By design, queued jobs normally only run after the script was fully
executed, making it improssible to test handlers in individual test()
calls by default [2].
Subsequent commits include integrations into LibWeb and js(1) -
pretty-printing, running queued promise jobs when necessary.
This has an unusual amount of dbgln() statements, all hidden behind the
PROMISE_DEBUG flag - I'm leaving them in for now as they've been very
useful while debugging this, things can get quite complex with so many
asynchronously executed functions.
I've not extensively explored use of these APIs for promise-based
functionality in LibWeb (fetch(), Notification.requestPermission()
etc.), but we'll get there in due time.
[1]: https://tc39.es/ecma262/#sec-promise-objects
[2]: https://tc39.es/ecma262/#sec-jobs-and-job-queues
											 
										 
										
											2021-04-01 22:13:29 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2023-02-26 16:09:02 -07:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    for  ( auto  finalization_registry  :  m_finalization_registry_cleanup_jobs ) 
							 
						 
					
						
							
								
									
										
										
										
											2024-11-15 04:01:23 +13:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        roots . set ( finalization_registry ,  GC : : HeapRoot  {  . type  =  GC : : HeapRoot : : Type : : VM  } ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2023-12-13 10:24:30 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    auto  gather_roots_from_execution_context_stack  =  [ & roots ] ( Vector < ExecutionContext * >  const &  stack )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        for  ( auto  const &  execution_context  :  stack )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            ExecutionContextRootsCollector  visitor ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            execution_context - > visit_edges ( visitor ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2024-04-05 13:47:41 -07:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								            for  ( auto  cell  :  visitor . roots ) 
							 
						 
					
						
							
								
									
										
										
										
											2024-11-15 04:01:23 +13:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								                roots . set ( cell ,  GC : : HeapRoot  {  . type  =  GC : : HeapRoot : : Type : : VM  } ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2023-12-13 10:24:30 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        } 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    } ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    gather_roots_from_execution_context_stack ( m_execution_context_stack ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    for  ( auto &  saved_stack  :  m_saved_execution_context_stacks ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        gather_roots_from_execution_context_stack ( saved_stack ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2024-03-25 14:18:22 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    for  ( auto &  job  :  m_promise_jobs ) 
							 
						 
					
						
							
								
									
										
										
										
											2024-11-15 04:01:23 +13:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        roots . set ( job ,  GC : : HeapRoot  {  . type  =  GC : : HeapRoot : : Type : : VM  } ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2020-09-22 16:18:51 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								}  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-07-02 21:54:56 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								// 9.1.2.1 GetIdentifierReference ( env, name, strict ), https://tc39.es/ecma262/#sec-getidentifierreference
  
						 
					
						
							
								
									
										
										
										
											2025-08-02 19:27:29 -04:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								ThrowCompletionOr < Reference >  VM : : get_identifier_reference ( Environment *  environment ,  Utf16FlyString  name ,  bool  strict ,  size_t  hops )  
						 
					
						
							
								
									
										
										
										
											2020-09-27 15:18:55 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								{  
						 
					
						
							
								
									
										
										
										
											2021-07-02 21:54:56 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    // 1. If env is the value null, then
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    if  ( ! environment )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        // a. Return the Reference Record { [[Base]]: unresolvable, [[ReferencedName]]: name, [[Strict]]: strict, [[ThisValue]]: empty }.
 
							 
						 
					
						
							
								
									
										
										
										
											2021-09-11 19:03:38 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        return  Reference  {  Reference : : BaseType : : Unresolvable ,  move ( name ) ,  strict  } ; 
							 
						 
					
						
							
								
									
										
										
										
											2021-07-02 21:54:56 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    } 
							 
						 
					
						
							
								
									
										
										
										
											2021-10-07 00:16:37 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-12-30 13:19:05 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    // 2. Let exists be ? env.HasBinding(name).
 
							 
						 
					
						
							
								
									
										
										
										
											2021-10-07 00:16:37 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    Optional < size_t >  index ; 
							 
						 
					
						
							
								
									
										
										
										
											2021-12-30 13:19:05 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    auto  exists  =  TRY ( environment - > has_binding ( name ,  & index ) ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2021-07-02 21:54:56 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-12-30 13:19:05 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    // Note: This is an optimization for looking up the same reference.
 
							 
						 
					
						
							
								
									
										
										
										
											2021-10-07 01:06:21 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    Optional < EnvironmentCoordinate >  environment_coordinate ; 
							 
						 
					
						
							
								
									
										
										
										
											2022-11-21 18:01:22 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    if  ( index . has_value ( ) )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        VERIFY ( hops  < =  NumericLimits < u32 > : : max ( ) ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        VERIFY ( index . value ( )  < =  NumericLimits < u32 > : : max ( ) ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        environment_coordinate  =  EnvironmentCoordinate  {  . hops  =  static_cast < u32 > ( hops ) ,  . index  =  static_cast < u32 > ( index . value ( ) )  } ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    } 
							 
						 
					
						
							
								
									
										
										
										
											2021-10-07 01:06:21 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-12-30 13:19:05 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    // 3. If exists is true, then
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    if  ( exists )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        // a. Return the Reference Record { [[Base]]: env, [[ReferencedName]]: name, [[Strict]]: strict, [[ThisValue]]: empty }.
 
							 
						 
					
						
							
								
									
										
										
										
											2021-10-07 01:06:21 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        return  Reference  {  * environment ,  move ( name ) ,  strict ,  environment_coordinate  } ; 
							 
						 
					
						
							
								
									
										
										
										
											2021-12-30 13:19:05 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    } 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    // 4. Else,
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    else  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        // a. Let outer be env.[[OuterEnv]].
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        // b. Return ? GetIdentifierReference(outer, name, strict).
 
							 
						 
					
						
							
								
									
										
										
										
											2021-10-07 01:06:21 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        return  get_identifier_reference ( environment - > outer_environment ( ) ,  move ( name ) ,  strict ,  hops  +  1 ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2021-12-30 13:19:05 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    } 
							 
						 
					
						
							
								
									
										
										
										
											2021-07-02 21:54:56 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								}  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								// 9.4.2 ResolveBinding ( name [ , env ] ), https://tc39.es/ecma262/#sec-resolvebinding
  
						 
					
						
							
								
									
										
										
										
											2025-08-02 19:27:29 -04:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								ThrowCompletionOr < Reference >  VM : : resolve_binding ( Utf16FlyString  const &  name ,  Environment *  environment )  
						 
					
						
							
								
									
										
										
										
											2021-07-02 21:54:56 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								{  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    // 1. If env is not present or if env is undefined, then
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    if  ( ! environment )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        // a. Set env to the running execution context's LexicalEnvironment.
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        environment  =  running_execution_context ( ) . lexical_environment ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    } 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    // 2. Assert: env is an Environment Record.
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    VERIFY ( environment ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2022-04-30 22:22:52 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    // 3. If the source text matched by the syntactic production that is being evaluated is contained in strict mode code, let strict be true; else let strict be false.
 
							 
						 
					
						
							
								
									
										
										
										
											2021-07-02 21:54:56 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    bool  strict  =  in_strict_mode ( ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    // 4. Return ? GetIdentifierReference(env, name, strict).
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    return  get_identifier_reference ( environment ,  name ,  strict ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2021-12-30 14:13:20 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    // NOTE: The spec says:
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    //       Note: The result of ResolveBinding is always a Reference Record whose [[ReferencedName]] field is name.
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    //       But this is not actually correct as GetIdentifierReference (or really the methods it calls) can throw.
 
							 
						 
					
						
							
								
									
										
										
										
											2020-09-27 15:18:55 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								}  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-06-25 11:40:03 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								// 9.4.4 ResolveThisBinding ( ), https://tc39.es/ecma262/#sec-resolvethisbinding
  
						 
					
						
							
								
									
										
										
										
											2022-08-21 15:12:43 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								ThrowCompletionOr < Value >  VM : : resolve_this_binding ( )  
						 
					
						
							
								
									
										
										
										
											2021-06-25 11:40:03 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								{  
						 
					
						
							
								
									
										
										
										
											2022-08-21 15:12:43 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    auto &  vm  =  * this ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-12-30 23:00:37 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    // 1. Let envRec be GetThisEnvironment().
 
							 
						 
					
						
							
								
									
										
										
										
											2022-12-15 20:07:13 +00:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    auto  environment  =  get_this_environment ( vm ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2022-08-21 15:12:43 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-12-30 23:00:37 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    // 2. Return ? envRec.GetThisBinding().
 
							 
						 
					
						
							
								
									
										
										
										
											2022-12-15 20:07:13 +00:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    return  TRY ( environment - > get_this_binding ( vm ) ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2021-06-25 11:40:03 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								}  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-12-28 23:50:23 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								// 9.4.5 GetNewTarget ( ), https://tc39.es/ecma262/#sec-getnewtarget
  
						 
					
						
							
								
									
										
										
										
											2021-06-22 13:30:48 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								Value  VM : : get_new_target ( )  
						 
					
						
							
								
									
										
										
										
											2020-09-27 15:18:55 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								{  
						 
					
						
							
								
									
										
										
										
											2021-12-28 23:50:23 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    // 1. Let envRec be GetThisEnvironment().
 
							 
						 
					
						
							
								
									
										
										
										
											2022-12-15 20:07:13 +00:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    auto  env  =  get_this_environment ( * this ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2021-12-28 23:50:23 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    // 2. Assert: envRec has a [[NewTarget]] field.
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    // 3. Return envRec.[[NewTarget]].
 
							 
						 
					
						
							
								
									
										
										
										
											2025-01-21 09:12:05 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    return  as < FunctionEnvironment > ( * env ) . new_target ( ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2020-09-27 15:18:55 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								}  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2023-07-11 23:07:12 -04:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								// 13.3.12.1 Runtime Semantics: Evaluation, https://tc39.es/ecma262/#sec-meta-properties-runtime-semantics-evaluation
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								// ImportMeta branch only
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								Object *  VM : : get_import_meta ( )  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								{  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    // 1. Let module be GetActiveScriptOrModule().
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    auto  script_or_module  =  get_active_script_or_module ( ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    // 2. Assert: module is a Source Text Module Record.
 
							 
						 
					
						
							
								
									
										
										
										
											2025-01-21 09:12:05 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    auto &  module  =  as < SourceTextModule > ( * script_or_module . get < GC : : Ref < Module > > ( ) ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2023-07-11 23:07:12 -04:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    // 3. Let importMeta be module.[[ImportMeta]].
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    auto *  import_meta  =  module . import_meta ( ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    // 4. If importMeta is empty, then
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    if  ( import_meta  = =  nullptr )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        // a. Set importMeta to OrdinaryObjectCreate(null).
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        import_meta  =  Object : : create ( * current_realm ( ) ,  nullptr ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        // b. Let importMetaValues be HostGetImportMetaProperties(module).
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        auto  import_meta_values  =  host_get_import_meta_properties ( module ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        // c. For each Record { [[Key]], [[Value]] } p of importMetaValues, do
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        for  ( auto &  entry  :  import_meta_values )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            // i. Perform ! CreateDataPropertyOrThrow(importMeta, p.[[Key]], p.[[Value]]).
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            MUST ( import_meta - > create_data_property_or_throw ( entry . key ,  entry . value ) ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        } 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        // d. Perform HostFinalizeImportMeta(importMeta, module).
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        host_finalize_import_meta ( import_meta ,  module ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        // e. Set module.[[ImportMeta]] to importMeta.
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        module . set_import_meta ( { } ,  import_meta ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        // f. Return importMeta.
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return  import_meta ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    } 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    // 5. Else,
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    else  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        // a. Assert: Type(importMeta) is Object.
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        // Note: This is always true by the type.
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        // b. Return importMeta.
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return  import_meta ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    } 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								}  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2022-08-21 15:39:13 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								// 9.4.5 GetGlobalObject ( ), https://tc39.es/ecma262/#sec-getglobalobject
  
						 
					
						
							
								
									
										
										
										
											2022-08-28 15:03:45 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								Object &  VM : : get_global_object ( )  
						 
					
						
							
								
									
										
										
										
											2022-08-21 15:39:13 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								{  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    // 1. Let currentRealm be the current Realm Record.
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    auto &  current_realm  =  * this - > current_realm ( ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    // 2. Return currentRealm.[[GlobalObject]].
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    return  current_realm . global_object ( ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								}  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2025-04-28 11:53:15 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								void  VM : : run_queued_promise_jobs_impl ( )  
						 
					
						
							
								
									
										
											 
										
											
												LibJS: Add initial support for Promises
Almost a year after first working on this, it's finally done: an
implementation of Promises for LibJS! :^)
The core functionality is working and closely following the spec [1].
I mostly took the pseudo code and transformed it into C++ - if you read
and understand it, you will know how the spec implements Promises; and
if you read the spec first, the code will look very familiar.
Implemented functions are:
- Promise() constructor
- Promise.prototype.then()
- Promise.prototype.catch()
- Promise.prototype.finally()
- Promise.resolve()
- Promise.reject()
For the tests I added a new function to test-js's global object,
runQueuedPromiseJobs(), which calls vm.run_queued_promise_jobs().
By design, queued jobs normally only run after the script was fully
executed, making it improssible to test handlers in individual test()
calls by default [2].
Subsequent commits include integrations into LibWeb and js(1) -
pretty-printing, running queued promise jobs when necessary.
This has an unusual amount of dbgln() statements, all hidden behind the
PROMISE_DEBUG flag - I'm leaving them in for now as they've been very
useful while debugging this, things can get quite complex with so many
asynchronously executed functions.
I've not extensively explored use of these APIs for promise-based
functionality in LibWeb (fetch(), Notification.requestPermission()
etc.), but we'll get there in due time.
[1]: https://tc39.es/ecma262/#sec-promise-objects
[2]: https://tc39.es/ecma262/#sec-jobs-and-job-queues
											 
										 
										
											2021-04-01 22:13:29 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								{  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    dbgln_if ( PROMISE_DEBUG ,  " Running queued promise jobs " ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2022-02-07 15:12:41 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
											 
										
											
												LibJS: Add initial support for Promises
Almost a year after first working on this, it's finally done: an
implementation of Promises for LibJS! :^)
The core functionality is working and closely following the spec [1].
I mostly took the pseudo code and transformed it into C++ - if you read
and understand it, you will know how the spec implements Promises; and
if you read the spec first, the code will look very familiar.
Implemented functions are:
- Promise() constructor
- Promise.prototype.then()
- Promise.prototype.catch()
- Promise.prototype.finally()
- Promise.resolve()
- Promise.reject()
For the tests I added a new function to test-js's global object,
runQueuedPromiseJobs(), which calls vm.run_queued_promise_jobs().
By design, queued jobs normally only run after the script was fully
executed, making it improssible to test handlers in individual test()
calls by default [2].
Subsequent commits include integrations into LibWeb and js(1) -
pretty-printing, running queued promise jobs when necessary.
This has an unusual amount of dbgln() statements, all hidden behind the
PROMISE_DEBUG flag - I'm leaving them in for now as they've been very
useful while debugging this, things can get quite complex with so many
asynchronously executed functions.
I've not extensively explored use of these APIs for promise-based
functionality in LibWeb (fetch(), Notification.requestPermission()
etc.), but we'll get there in due time.
[1]: https://tc39.es/ecma262/#sec-promise-objects
[2]: https://tc39.es/ecma262/#sec-jobs-and-job-queues
											 
										 
										
											2021-04-01 22:13:29 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    while  ( ! m_promise_jobs . is_empty ( ) )  { 
							 
						 
					
						
							
								
									
										
										
										
											2022-02-06 03:46:45 +00:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        auto  job  =  m_promise_jobs . take_first ( ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        dbgln_if ( PROMISE_DEBUG ,  " Calling promise job function " ) ; 
							 
						 
					
						
							
								
									
										
											 
										
											
												LibJS: Clear exception after running each queued Promise job
It's not what the spec tells us to do. In fact, the spec tells us the
exact opposite:
    9.5 Jobs and Host Operations to Enqueue Jobs
    https://tc39.es/ecma262/#sec-jobs
    A Job is an Abstract Closure with no parameters that initiates an
    ECMAScript computation when no other ECMAScript computation is
    currently in progress.
    ...
    Their implementations must conform to the following requirements:
    - ...
    - The Abstract Closure must return a normal completion, implementing
      its own handling of errors.
However, this turned out to not be true in all cases. More specifically,
the NewPromiseReactionJob AO returns the completion result of calling a
user-provided function (PromiseCapability's [[Resolve]] / [[Reject]]),
which may be an abrupt completion:
    27.2.2.1 NewPromiseReactionJob ( reaction, argument )
    https://tc39.es/ecma262/#sec-newpromisereactionjob
    1. Let job be a new Job Abstract Closure with no parameters that
       captures reaction and argument and performs the following steps
       when called:
       ...
       h. If handlerResult is an abrupt completion, then
          i. Let status be Call(promiseCapability.[[Reject]],
             undefined, « handlerResult.[[Value]] »).
       i. Else,
          i. Let status be Call(promiseCapability.[[Resolve]],
             undefined, « handlerResult.[[Value]] »).
       j. Return Completion(status).
Interestingly, this case is explicitly handled in the HTML spec's
implementation of jobs as microtasks:
    8.1.5.3.3 HostEnqueuePromiseJob(job, realm)
    https://html.spec.whatwg.org/webappapis.html#hostenqueuepromisejob
    2. Queue a microtask on the surrounding agent's event loop to
       perform the following steps:
       ...
       5. If result is an abrupt completion, then report the exception
          given by result.[[Value]].
This is precisely what all the major engines do - but not only in
browsers; the provided code snippet in the test added in this commit
works just fine in Node.js, for example.
SpiderMonkey:
https://searchfox.org/mozilla-central/rev/25997ce8267ec9e3ea4b727e0973bd9ef02bba79/js/src/builtin/Promise.cpp#6292
https://searchfox.org/mozilla-central/rev/25997ce8267ec9e3ea4b727e0973bd9ef02bba79/js/src/builtin/Promise.cpp#1277
https://searchfox.org/mozilla-central/rev/25997ce8267ec9e3ea4b727e0973bd9ef02bba79/js/src/vm/JSContext.cpp#845
JavaScriptCore:
https://trac.webkit.org/browser/webkit/trunk/Source/JavaScriptCore/builtins/PromiseOperations.js?rev=273718#L562
https://trac.webkit.org/browser/webkit/trunk/Source/JavaScriptCore/runtime/JSMicrotask.cpp?rev=273718#L94
V8:
https://source.chromium.org/chromium/chromium/src/+/main:v8/src/builtins/promise-abstract-operations.tq;l=481;drc=a760f03a6e99bf4863d8d21c5f7896a74a0a39ea
https://source.chromium.org/chromium/chromium/src/+/main:v8/src/builtins/builtins-microtask-queue-gen.cc;l=331;drc=65c9257f1777731d6d0669598f6fe6fe65fa61d3
This should probably be fixed in the ECMAScript spec to relax the rule
that Jobs may not return an abrupt completion, just like in the HTML
spec. The important bit is that those are not surfaced to user code in
any way.
											 
										 
										
											2021-11-11 23:13:58 +00:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2024-03-25 14:18:22 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        [[maybe_unused]]  auto  result  =  job - > function ( ) ( ) ; 
							 
						 
					
						
							
								
									
										
											 
										
											
												LibJS: Add initial support for Promises
Almost a year after first working on this, it's finally done: an
implementation of Promises for LibJS! :^)
The core functionality is working and closely following the spec [1].
I mostly took the pseudo code and transformed it into C++ - if you read
and understand it, you will know how the spec implements Promises; and
if you read the spec first, the code will look very familiar.
Implemented functions are:
- Promise() constructor
- Promise.prototype.then()
- Promise.prototype.catch()
- Promise.prototype.finally()
- Promise.resolve()
- Promise.reject()
For the tests I added a new function to test-js's global object,
runQueuedPromiseJobs(), which calls vm.run_queued_promise_jobs().
By design, queued jobs normally only run after the script was fully
executed, making it improssible to test handlers in individual test()
calls by default [2].
Subsequent commits include integrations into LibWeb and js(1) -
pretty-printing, running queued promise jobs when necessary.
This has an unusual amount of dbgln() statements, all hidden behind the
PROMISE_DEBUG flag - I'm leaving them in for now as they've been very
useful while debugging this, things can get quite complex with so many
asynchronously executed functions.
I've not extensively explored use of these APIs for promise-based
functionality in LibWeb (fetch(), Notification.requestPermission()
etc.), but we'll get there in due time.
[1]: https://tc39.es/ecma262/#sec-promise-objects
[2]: https://tc39.es/ecma262/#sec-jobs-and-job-queues
											 
										 
										
											2021-04-01 22:13:29 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    } 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								}  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-06-13 00:22:35 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								// 9.5.4 HostEnqueuePromiseJob ( job, realm ), https://tc39.es/ecma262/#sec-hostenqueuepromisejob
  
						 
					
						
							
								
									
										
										
										
											2024-11-15 04:01:23 +13:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								void  VM : : enqueue_promise_job ( GC : : Ref < GC : : Function < ThrowCompletionOr < Value > ( ) > >  job ,  Realm * )  
						 
					
						
							
								
									
										
											 
										
											
												LibJS: Add initial support for Promises
Almost a year after first working on this, it's finally done: an
implementation of Promises for LibJS! :^)
The core functionality is working and closely following the spec [1].
I mostly took the pseudo code and transformed it into C++ - if you read
and understand it, you will know how the spec implements Promises; and
if you read the spec first, the code will look very familiar.
Implemented functions are:
- Promise() constructor
- Promise.prototype.then()
- Promise.prototype.catch()
- Promise.prototype.finally()
- Promise.resolve()
- Promise.reject()
For the tests I added a new function to test-js's global object,
runQueuedPromiseJobs(), which calls vm.run_queued_promise_jobs().
By design, queued jobs normally only run after the script was fully
executed, making it improssible to test handlers in individual test()
calls by default [2].
Subsequent commits include integrations into LibWeb and js(1) -
pretty-printing, running queued promise jobs when necessary.
This has an unusual amount of dbgln() statements, all hidden behind the
PROMISE_DEBUG flag - I'm leaving them in for now as they've been very
useful while debugging this, things can get quite complex with so many
asynchronously executed functions.
I've not extensively explored use of these APIs for promise-based
functionality in LibWeb (fetch(), Notification.requestPermission()
etc.), but we'll get there in due time.
[1]: https://tc39.es/ecma262/#sec-promise-objects
[2]: https://tc39.es/ecma262/#sec-jobs-and-job-queues
											 
										 
										
											2021-04-01 22:13:29 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								{  
						 
					
						
							
								
									
										
										
										
											2022-02-06 03:46:45 +00:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    // An implementation of HostEnqueuePromiseJob must conform to the requirements in 9.5 as well as the following:
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    // - FIXME: If realm is not null, each time job is invoked the implementation must perform implementation-defined steps such that execution is prepared to evaluate ECMAScript code at the time of job's invocation.
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    // - FIXME: Let scriptOrModule be GetActiveScriptOrModule() at the time HostEnqueuePromiseJob is invoked. If realm is not null, each time job is invoked the implementation must perform implementation-defined steps
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    //          such that scriptOrModule is the active script or module at the time of job's invocation.
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    // - Jobs must run in the same order as the HostEnqueuePromiseJob invocations that scheduled them.
 
							 
						 
					
						
							
								
									
										
										
										
											2024-03-25 14:18:22 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    m_promise_jobs . append ( job ) ; 
							 
						 
					
						
							
								
									
										
											 
										
											
												LibJS: Add initial support for Promises
Almost a year after first working on this, it's finally done: an
implementation of Promises for LibJS! :^)
The core functionality is working and closely following the spec [1].
I mostly took the pseudo code and transformed it into C++ - if you read
and understand it, you will know how the spec implements Promises; and
if you read the spec first, the code will look very familiar.
Implemented functions are:
- Promise() constructor
- Promise.prototype.then()
- Promise.prototype.catch()
- Promise.prototype.finally()
- Promise.resolve()
- Promise.reject()
For the tests I added a new function to test-js's global object,
runQueuedPromiseJobs(), which calls vm.run_queued_promise_jobs().
By design, queued jobs normally only run after the script was fully
executed, making it improssible to test handlers in individual test()
calls by default [2].
Subsequent commits include integrations into LibWeb and js(1) -
pretty-printing, running queued promise jobs when necessary.
This has an unusual amount of dbgln() statements, all hidden behind the
PROMISE_DEBUG flag - I'm leaving them in for now as they've been very
useful while debugging this, things can get quite complex with so many
asynchronously executed functions.
I've not extensively explored use of these APIs for promise-based
functionality in LibWeb (fetch(), Notification.requestPermission()
etc.), but we'll get there in due time.
[1]: https://tc39.es/ecma262/#sec-promise-objects
[2]: https://tc39.es/ecma262/#sec-jobs-and-job-queues
											 
										 
										
											2021-04-01 22:13:29 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								}  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-06-15 22:16:17 +03:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								void  VM : : run_queued_finalization_registry_cleanup_jobs ( )  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								{  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    while  ( ! m_finalization_registry_cleanup_jobs . is_empty ( ) )  { 
							 
						 
					
						
							
								
									
										
										
										
											2023-02-26 16:09:02 -07:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        auto  registry  =  m_finalization_registry_cleanup_jobs . take_first ( ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2022-02-07 15:12:41 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        // FIXME: Handle any uncatched exceptions here.
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        ( void ) registry - > cleanup ( ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2021-06-15 22:16:17 +03:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    } 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								}  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								// 9.10.4.1 HostEnqueueFinalizationRegistryCleanupJob ( finalizationRegistry ), https://tc39.es/ecma262/#sec-host-cleanup-finalization-registry
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								void  VM : : enqueue_finalization_registry_cleanup_job ( FinalizationRegistry &  registry )  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								{  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    m_finalization_registry_cleanup_jobs . append ( & registry ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								}  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-06-13 00:22:35 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								// 27.2.1.9 HostPromiseRejectionTracker ( promise, operation ), https://tc39.es/ecma262/#sec-host-promise-rejection-tracker
  
						 
					
						
							
								
									
										
										
										
											2022-02-06 03:46:45 +00:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								void  VM : : promise_rejection_tracker ( Promise &  promise ,  Promise : : RejectionOperation  operation )  const  
						 
					
						
							
								
									
										
											 
										
											
												LibJS: Add initial support for Promises
Almost a year after first working on this, it's finally done: an
implementation of Promises for LibJS! :^)
The core functionality is working and closely following the spec [1].
I mostly took the pseudo code and transformed it into C++ - if you read
and understand it, you will know how the spec implements Promises; and
if you read the spec first, the code will look very familiar.
Implemented functions are:
- Promise() constructor
- Promise.prototype.then()
- Promise.prototype.catch()
- Promise.prototype.finally()
- Promise.resolve()
- Promise.reject()
For the tests I added a new function to test-js's global object,
runQueuedPromiseJobs(), which calls vm.run_queued_promise_jobs().
By design, queued jobs normally only run after the script was fully
executed, making it improssible to test handlers in individual test()
calls by default [2].
Subsequent commits include integrations into LibWeb and js(1) -
pretty-printing, running queued promise jobs when necessary.
This has an unusual amount of dbgln() statements, all hidden behind the
PROMISE_DEBUG flag - I'm leaving them in for now as they've been very
useful while debugging this, things can get quite complex with so many
asynchronously executed functions.
I've not extensively explored use of these APIs for promise-based
functionality in LibWeb (fetch(), Notification.requestPermission()
etc.), but we'll get there in due time.
[1]: https://tc39.es/ecma262/#sec-promise-objects
[2]: https://tc39.es/ecma262/#sec-jobs-and-job-queues
											 
										 
										
											2021-04-01 22:13:29 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								{  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    switch  ( operation )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    case  Promise : : RejectionOperation : : Reject : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        // A promise was rejected without any handlers
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        if  ( on_promise_unhandled_rejection ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            on_promise_unhandled_rejection ( promise ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        break ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    case  Promise : : RejectionOperation : : Handle : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        // A handler was added to an already rejected promise
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        if  ( on_promise_rejection_handled ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            on_promise_rejection_handled ( promise ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        break ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    default : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        VERIFY_NOT_REACHED ( ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    } 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								}  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-06-06 22:53:44 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								void  VM : : dump_backtrace ( )  const  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								{  
						 
					
						
							
								
									
										
										
										
											2021-09-11 17:01:19 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    for  ( ssize_t  i  =  m_execution_context_stack . size ( )  -  1 ;  i  > =  0 ;  - - i )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        auto &  frame  =  m_execution_context_stack [ i ] ; 
							 
						 
					
						
							
								
									
										
										
										
											2025-04-28 19:02:56 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        if  ( frame - > executable )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            auto  source_range  =  frame - > executable - > source_range_at ( frame - > program_counter ) . realize ( ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2023-11-27 13:38:19 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								            dbgln ( " -> {} @ {}:{},{} " ,  frame - > function_name  ?  frame - > function_name - > utf8_string ( )  :  " " _string ,  source_range . filename ( ) ,  source_range . start . line ,  source_range . start . column ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2021-09-11 17:01:19 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        }  else  { 
							 
						 
					
						
							
								
									
										
										
										
											2023-11-27 13:38:19 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								            dbgln ( " -> {} " ,  frame - > function_name  ?  frame - > function_name - > utf8_string ( )  :  " " _string ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2021-09-11 17:01:19 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        } 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    } 
							 
						 
					
						
							
								
									
										
										
										
											2021-06-06 22:53:44 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								}  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-10-03 14:52:53 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								void  VM : : save_execution_context_stack ( )  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								{  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    m_saved_execution_context_stacks . append ( move ( m_execution_context_stack ) ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								}  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2024-01-18 12:49:35 -07:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								void  VM : : clear_execution_context_stack ( )  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								{  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    m_execution_context_stack . clear_with_capacity ( ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								}  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-10-03 14:52:53 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								void  VM : : restore_execution_context_stack ( )  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								{  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    m_execution_context_stack  =  m_saved_execution_context_stacks . take_last ( ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								}  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2022-01-17 14:48:22 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								// 9.4.1 GetActiveScriptOrModule ( ), https://tc39.es/ecma262/#sec-getactivescriptormodule
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								ScriptOrModule  VM : : get_active_script_or_module ( )  const  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								{  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    // 1. If the execution context stack is empty, return null.
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    if  ( m_execution_context_stack . is_empty ( ) ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return  Empty  { } ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    // 2. Let ec be the topmost execution context on the execution context stack whose ScriptOrModule component is not null.
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    for  ( auto  i  =  m_execution_context_stack . size ( )  -  1 ;  i  >  0 ;  i - - )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        if  ( ! m_execution_context_stack [ i ] - > script_or_module . has < Empty > ( ) ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            return  m_execution_context_stack [ i ] - > script_or_module ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    } 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    // 3. If no such execution context exists, return null. Otherwise, return ec's ScriptOrModule.
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    // Note: Since it is not empty we have 0 and since we got here all the
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    //       above contexts don't have a non-null ScriptOrModule
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    return  m_execution_context_stack [ 0 ] - > script_or_module ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								}  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2025-08-06 11:12:58 -04:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								VM : : StoredModule *  VM : : get_stored_module ( ImportedModuleReferrer  const & ,  ByteString  const &  filename ,  Utf16String  const & )  
						 
					
						
							
								
									
										
										
										
											2022-01-19 01:22:58 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								{  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    // Note the spec says:
 
							 
						 
					
						
							
								
									
										
										
										
											2023-12-03 14:22:05 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    // If this operation is called multiple times with the same (referrer, specifier) pair and it performs
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    // FinishLoadingImportedModule(referrer, specifier, payload, result) where result is a normal completion,
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    // then it must perform FinishLoadingImportedModule(referrer, specifier, payload, result) with the same result each time.
 
							 
						 
					
						
							
								
									
										
										
										
											2022-01-27 14:51:21 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2025-04-28 16:22:48 -04:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    // Editor's Note from https://tc39.es/ecma262/#sec-hostresolveimportedmodule
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    // The above text requires that hosts support JSON modules when imported with type: "json" (and HostLoadImportedModule
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    // completes normally), but it does not prohibit hosts from supporting JSON modules when imported without type: "json".
 
							 
						 
					
						
							
								
									
										
										
										
											2022-01-27 14:51:21 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2023-12-03 14:22:05 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    // FIXME: This should probably check referrer as well.
 
							 
						 
					
						
							
								
									
										
										
										
											2022-01-19 01:22:58 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    auto  end_or_module  =  m_loaded_modules . find_if ( [ & ] ( StoredModule  const &  stored_module )  { 
							 
						 
					
						
							
								
									
										
										
										
											2022-06-26 20:58:32 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        return  stored_module . filename  = =  filename ; 
							 
						 
					
						
							
								
									
										
										
										
											2022-01-19 01:22:58 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    } ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    if  ( end_or_module . is_end ( ) ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return  nullptr ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    return  & ( * end_or_module ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								}  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2023-06-15 12:36:57 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								ThrowCompletionOr < void >  VM : : link_and_eval_module ( Badge < Bytecode : : Interpreter > ,  SourceTextModule &  module )  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								{  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    return  link_and_eval_module ( module ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								}  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2023-12-06 11:14:41 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								ThrowCompletionOr < void >  VM : : link_and_eval_module ( CyclicModule &  module )  
						 
					
						
							
								
									
										
										
										
											2022-01-19 01:22:58 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								{  
						 
					
						
							
								
									
										
										
										
											2022-06-26 20:58:32 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    auto  filename  =  module . filename ( ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2025-01-20 11:10:07 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    auto &  promise_capability  =  module . load_requested_modules ( nullptr ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2025-01-21 09:12:05 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    if  ( auto  const &  promise  =  as < Promise > ( * promise_capability . promise ( ) ) ;  promise . state ( )  = =  Promise : : State : : Rejected ) 
							 
						 
					
						
							
								
									
										
										
										
											2025-01-20 11:10:07 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        return  JS : : throw_completion ( promise . result ( ) ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2022-01-19 01:22:58 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2022-06-26 20:58:32 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    dbgln_if ( JS_MODULE_DEBUG ,  " [JS MODULE] Linking module {} " ,  filename ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2022-01-19 01:22:58 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    auto  linked_or_error  =  module . link ( * this ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    if  ( linked_or_error . is_error ( ) ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return  linked_or_error . throw_completion ( ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2022-06-26 20:58:32 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    dbgln_if ( JS_MODULE_DEBUG ,  " [JS MODULE] Linking passed, now evaluating module {} " ,  filename ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2022-01-19 01:22:58 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    auto  evaluated_or_error  =  module . evaluate ( * this ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    if  ( evaluated_or_error . is_error ( ) ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return  evaluated_or_error . throw_completion ( ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2025-04-28 15:03:56 -04:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    auto  evaluated_value  =  evaluated_or_error . value ( ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2022-01-19 01:22:58 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    run_queued_promise_jobs ( ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    VERIFY ( m_promise_jobs . is_empty ( ) ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    // FIXME: This will break if we start doing promises actually asynchronously.
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    VERIFY ( evaluated_value - > state ( )  ! =  Promise : : State : : Pending ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2022-02-07 15:12:41 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    if  ( evaluated_value - > state ( )  = =  Promise : : State : : Rejected ) 
							 
						 
					
						
							
								
									
										
										
										
											2022-01-19 01:22:58 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        return  JS : : throw_completion ( evaluated_value - > result ( ) ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    dbgln_if ( JS_MODULE_DEBUG ,  " [JS MODULE] Evaluating passed for module {} " ,  module . filename ( ) ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    return  { } ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								}  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2025-08-06 11:12:58 -04:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								static  ByteString  resolve_module_filename ( StringView  filename ,  Utf16View  const &  module_type )  
						 
					
						
							
								
									
										
										
										
											2022-06-26 21:00:55 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								{  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    auto  extensions  =  Vector < StringView ,  2 >  {  " js " sv ,  " mjs " sv  } ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    if  ( module_type  = =  " json " sv ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        extensions  =  {  " json " sv  } ; 
							 
						 
					
						
							
								
									
										
										
										
											2023-03-22 02:35:30 +11:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    if  ( ! FileSystem : : exists ( filename ) )  { 
							 
						 
					
						
							
								
									
										
										
										
											2022-06-26 21:00:55 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        for  ( auto  extension  :  extensions )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            // import "./foo" -> import "./foo.ext"
 
							 
						 
					
						
							
								
									
										
										
										
											2023-12-16 17:49:34 +03:30 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								            auto  resolved_filepath  =  ByteString : : formatted ( " {}.{} " ,  filename ,  extension ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2023-03-22 02:35:30 +11:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								            if  ( FileSystem : : exists ( resolved_filepath ) ) 
							 
						 
					
						
							
								
									
										
										
										
											2022-06-26 21:00:55 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								                return  resolved_filepath ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        } 
							 
						 
					
						
							
								
									
										
										
										
											2023-03-22 02:35:30 +11:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    }  else  if  ( FileSystem : : is_directory ( filename ) )  { 
							 
						 
					
						
							
								
									
										
										
										
											2022-06-26 21:00:55 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        for  ( auto  extension  :  extensions )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            // import "./foo" -> import "./foo/index.ext"
 
							 
						 
					
						
							
								
									
										
										
										
											2023-12-16 17:49:34 +03:30 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								            auto  resolved_filepath  =  LexicalPath : : join ( filename ,  ByteString : : formatted ( " index.{} " ,  extension ) ) . string ( ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2023-03-22 02:35:30 +11:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								            if  ( FileSystem : : exists ( resolved_filepath ) ) 
							 
						 
					
						
							
								
									
										
										
										
											2022-06-26 21:00:55 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								                return  resolved_filepath ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        } 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    } 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    return  filename ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								}  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2023-12-02 22:56:47 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								// 16.2.1.8 HostLoadImportedModule ( referrer, specifier, hostDefined, payload ), https://tc39.es/ecma262/#sec-HostLoadImportedModule
  
						 
					
						
							
								
									
										
										
										
											2024-11-15 04:01:23 +13:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								void  VM : : load_imported_module ( ImportedModuleReferrer  referrer ,  ModuleRequest  const &  module_request ,  GC : : Ptr < GraphLoadingState : : HostDefined > ,  ImportedModulePayload  payload )  
						 
					
						
							
								
									
										
										
										
											2023-12-02 22:56:47 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								{  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    // An implementation of HostLoadImportedModule must conform to the following requirements:
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    //
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    // - The host environment must perform FinishLoadingImportedModule(referrer, specifier, payload, result),
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    //   where result is either a normal completion containing the loaded Module Record or a throw completion,
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    //   either synchronously or asynchronously.
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    // - If this operation is called multiple times with the same (referrer, specifier) pair and it performs
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    //   FinishLoadingImportedModule(referrer, specifier, payload, result) where result is a normal completion,
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    //   then it must perform FinishLoadingImportedModule(referrer, specifier, payload, result) with the same result each time.
 
							 
						 
					
						
							
								
									
										
										
										
											2025-04-28 16:22:48 -04:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    // - If moduleRequest.[[Attributes]] has an entry entry such that entry.[[Key]] is "type" and entry.[[Value]] is "json",
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    //   when the host environment performs FinishLoadingImportedModule(referrer, moduleRequest, payload, result), result
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    //   must either be the Completion Record returned by an invocation of ParseJSONModule or a throw completion.
 
							 
						 
					
						
							
								
									
										
										
										
											2023-12-02 22:56:47 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    // - The operation must treat payload as an opaque value to be passed through to FinishLoadingImportedModule.
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    //
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    // The actual process performed is host-defined, but typically consists of performing whatever I/O operations are necessary to
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    // load the appropriate Module Record. Multiple different (referrer, specifier) pairs may map to the same Module Record instance.
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    // The actual mapping semantics is host-defined but typically a normalization process is applied to specifier as part of the
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    // mapping process. A typical normalization process would include actions such as expansion of relative and abbreviated path specifiers.
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    // Here we check, against the spec, if payload is a promise capability, meaning that this was called for a dynamic import
 
							 
						 
					
						
							
								
									
										
										
										
											2024-11-15 04:01:23 +13:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    if  ( payload . has < GC : : Ref < PromiseCapability > > ( )  & &  ! m_dynamic_imports_allowed )  { 
							 
						 
					
						
							
								
									
										
										
										
											2023-12-02 22:56:47 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        // If you are here because you want to enable dynamic module importing make sure it won't be a security problem
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        // by checking the default implementation of HostImportModuleDynamically and creating your own hook or calling
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        // vm.allow_dynamic_imports().
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        finish_loading_imported_module ( referrer ,  module_request ,  payload ,  throw_completion < InternalError > ( ErrorType : : DynamicImportNotAllowed ,  module_request . module_specifier ) ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    } 
							 
						 
					
						
							
								
									
										
										
										
											2022-01-19 01:22:58 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2025-08-06 11:12:58 -04:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    Utf16String  module_type ; 
							 
						 
					
						
							
								
									
										
										
										
											2023-12-02 16:20:01 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    for  ( auto &  attribute  :  module_request . attributes )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        if  ( attribute . key  = =  " type " sv )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            module_type  =  attribute . value ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            break ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        } 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    } 
							 
						 
					
						
							
								
									
										
										
										
											2022-06-26 21:00:55 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2023-10-10 15:00:58 +03:30 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    dbgln_if ( JS_MODULE_DEBUG ,  " [JS MODULE] module at {} has type {} " ,  module_request . module_specifier ,  module_type ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2022-06-26 21:00:55 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2023-12-02 16:20:01 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    StringView  const  base_filename  =  referrer . visit ( 
							 
						 
					
						
							
								
									
										
										
										
											2024-11-15 04:01:23 +13:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        [ & ] ( GC : : Ref < Realm >  const & )  { 
							 
						 
					
						
							
								
									
										
										
										
											2023-12-02 22:56:47 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								            // Generally within ECMA262 we always get a referencing_script_or_module. However, ShadowRealm gives an explicit null.
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            // To get around this is we attempt to get the active script_or_module otherwise we might start loading "random" files from the working directory.
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            return  get_active_script_or_module ( ) . visit ( 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                [ ] ( Empty )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                    return  " . " sv ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                } , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                [ ] ( auto  const &  script_or_module )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                    return  script_or_module - > filename ( ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                } ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2022-01-19 01:22:58 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        } , 
							 
						 
					
						
							
								
									
										
										
										
											2023-12-02 22:56:47 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        [ & ] ( auto  const &  script_or_module )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            return  script_or_module - > filename ( ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2022-01-19 01:22:58 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        } ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    LexicalPath  base_path  {  base_filename  } ; 
							 
						 
					
						
							
								
									
										
										
										
											2025-08-06 11:12:58 -04:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    auto  filename  =  LexicalPath : : absolute_path ( base_path . dirname ( ) ,  MUST ( module_request . module_specifier . view ( ) . to_byte_string ( ) ) ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2022-01-19 01:22:58 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2022-06-26 21:00:55 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    dbgln_if ( JS_MODULE_DEBUG ,  " [JS MODULE] base path: '{}' " ,  base_path ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    dbgln_if ( JS_MODULE_DEBUG ,  " [JS MODULE] initial filename: '{}' " ,  filename ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    filename  =  resolve_module_filename ( filename ,  module_type ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    dbgln_if ( JS_MODULE_DEBUG ,  " [JS MODULE] resolved filename: '{}' " ,  filename ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2022-01-19 01:22:58 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								# if JS_MODULE_DEBUG 
  
						 
					
						
							
								
									
										
										
										
											2023-12-16 17:49:34 +03:30 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    ByteString  referencing_module_string  =  referrer . visit ( 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        [ & ] ( Empty )  - >  ByteString  { 
							 
						 
					
						
							
								
									
										
										
										
											2022-01-19 01:22:58 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								            return  " . " ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        } , 
							 
						 
					
						
							
								
									
										
										
										
											2022-02-07 19:58:50 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        [ & ] ( auto &  script_or_module )  { 
							 
						 
					
						
							
								
									
										
										
										
											2022-01-19 01:22:58 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								            if  constexpr  ( IsSame < Script * ,  decltype ( script_or_module ) > )  { 
							 
						 
					
						
							
								
									
										
										
										
											2023-12-16 17:49:34 +03:30 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								                return  ByteString : : formatted ( " Script @ {} " ,  script_or_module . ptr ( ) ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2022-01-19 01:22:58 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								            } 
							 
						 
					
						
							
								
									
										
										
										
											2023-12-16 17:49:34 +03:30 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								            return  ByteString : : formatted ( " Module @ {} " ,  script_or_module . ptr ( ) ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2022-01-19 01:22:58 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        } ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2023-12-02 22:56:47 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    dbgln_if ( JS_MODULE_DEBUG ,  " [JS MODULE] load_imported_module({}, {}) " ,  referencing_module_string ,  filename ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2022-06-26 20:58:32 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    dbgln_if ( JS_MODULE_DEBUG ,  " [JS MODULE]     resolved {} + {} -> {} " ,  base_path ,  module_request . module_specifier ,  filename ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2022-01-19 01:22:58 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								# endif 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2025-04-28 16:22:48 -04:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    if  ( auto *  loaded_module  =  get_stored_module ( referrer ,  filename ,  module_type ) )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        dbgln_if ( JS_MODULE_DEBUG ,  " [JS MODULE] load_imported_module({}) already loaded at {} " ,  filename ,  loaded_module - > module . ptr ( ) ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        finish_loading_imported_module ( referrer ,  module_request ,  payload ,  * loaded_module - > module ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2023-12-02 22:56:47 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        return ; 
							 
						 
					
						
							
								
									
										
										
										
											2022-01-19 01:22:58 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    } 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2022-06-26 20:58:32 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    dbgln_if ( JS_MODULE_DEBUG ,  " [JS MODULE] reading and parsing module {} " ,  filename ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2022-01-19 01:22:58 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2023-04-26 20:44:50 -04:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    auto  file_or_error  =  Core : : File : : open ( filename ,  Core : : File : : OpenMode : : Read ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2022-01-19 01:22:58 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    if  ( file_or_error . is_error ( ) )  { 
							 
						 
					
						
							
								
									
										
										
										
											2023-12-02 22:56:47 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        finish_loading_imported_module ( referrer ,  module_request ,  payload ,  throw_completion < SyntaxError > ( ErrorType : : ModuleNotFound ,  module_request . module_specifier ) ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return ; 
							 
						 
					
						
							
								
									
										
										
										
											2022-01-19 01:22:58 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    } 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    // FIXME: Don't read the file in one go.
 
							 
						 
					
						
							
								
									
										
										
										
											2023-04-26 20:44:50 -04:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    auto  file_content_or_error  =  file_or_error . value ( ) - > read_until_eof ( ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    if  ( file_content_or_error . is_error ( ) )  { 
							 
						 
					
						
							
								
									
										
										
										
											2023-12-02 22:56:47 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        if  ( file_content_or_error . error ( ) . code ( )  = =  ENOMEM )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            finish_loading_imported_module ( referrer ,  module_request ,  payload ,  throw_completion < JS : : InternalError > ( error_message ( : : JS : : VM : : ErrorMessage : : OutOfMemory ) ) ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            return ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        } 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        finish_loading_imported_module ( referrer ,  module_request ,  payload ,  throw_completion < SyntaxError > ( ErrorType : : ModuleNotFound ,  module_request . module_specifier ) ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return ; 
							 
						 
					
						
							
								
									
										
										
										
											2023-04-26 20:44:50 -04:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    } 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    StringView  const  content_view  {  file_content_or_error . value ( ) . bytes ( )  } ; 
							 
						 
					
						
							
								
									
										
										
										
											2022-01-19 01:22:58 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2025-04-28 16:22:48 -04:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    auto  module  =  [ & ,  content  =  file_content_or_error . release_value ( ) ] ( )  - >  ThrowCompletionOr < GC : : Ref < Module > >  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        // If moduleRequest.[[Attributes]] has an entry entry such that entry.[[Key]] is "type" and entry.[[Value]] is "json",
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        // when the host environment performs FinishLoadingImportedModule(referrer, moduleRequest, payload, result), result
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        // must either be the Completion Record returned by an invocation of ParseJSONModule or a throw completion.
 
							 
						 
					
						
							
								
									
										
										
										
											2022-01-27 14:51:21 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        if  ( module_type  = =  " json " sv )  { 
							 
						 
					
						
							
								
									
										
										
										
											2022-06-26 20:58:32 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								            dbgln_if ( JS_MODULE_DEBUG ,  " [JS MODULE] reading and parsing JSON module {} " ,  filename ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2025-04-28 16:22:48 -04:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								            return  parse_json_module ( * current_realm ( ) ,  content_view ,  filename ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2022-01-27 14:51:21 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        } 
							 
						 
					
						
							
								
									
										
										
										
											2022-01-19 01:22:58 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2022-06-26 20:58:32 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        dbgln_if ( JS_MODULE_DEBUG ,  " [JS MODULE] reading and parsing as SourceTextModule module {} " ,  filename ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2022-01-27 14:51:21 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        // Note: We treat all files as module, so if a script does not have exports it just runs it.
 
							 
						 
					
						
							
								
									
										
										
										
											2022-06-26 20:58:32 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        auto  module_or_errors  =  SourceTextModule : : parse ( content_view ,  * current_realm ( ) ,  filename ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2022-01-27 14:51:21 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        if  ( module_or_errors . is_error ( ) )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            VERIFY ( module_or_errors . error ( ) . size ( )  >  0 ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2023-12-16 17:49:34 +03:30 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								            return  throw_completion < SyntaxError > ( module_or_errors . error ( ) . first ( ) . to_byte_string ( ) ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2022-01-27 14:51:21 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        } 
							 
						 
					
						
							
								
									
										
										
										
											2022-01-18 19:39:36 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2025-04-28 16:22:48 -04:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        return  module_or_errors . release_value ( ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    } ( ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    if  ( ! module . is_error ( ) )  { 
							 
						 
					
						
							
								
									
										
										
										
											2023-12-03 14:22:05 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        m_loaded_modules . empend ( 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            referrer , 
							 
						 
					
						
							
								
									
										
										
										
											2025-04-28 16:22:48 -04:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								            module . value ( ) - > filename ( ) , 
							 
						 
					
						
							
								
									
										
										
										
											2025-03-18 18:08:02 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								            String  { } ,  // Null type
 
							 
						 
					
						
							
								
									
										
										
										
											2025-04-28 16:22:48 -04:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								            make_root ( module . value ( ) ) , 
							 
						 
					
						
							
								
									
										
										
										
											2023-12-03 14:22:05 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								            true ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2025-04-28 16:22:48 -04:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    } 
							 
						 
					
						
							
								
									
										
										
										
											2022-05-02 20:54:39 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2023-12-02 22:56:47 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    finish_loading_imported_module ( referrer ,  module_request ,  payload ,  module ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2022-01-18 19:39:36 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								}  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2024-10-14 07:49:12 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								static  RefPtr < CachedSourceRange >  get_source_range ( ExecutionContext  const *  context )  
						 
					
						
							
								
									
										
										
										
											2023-11-01 00:39:28 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								{  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    // native function
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    if  ( ! context - > executable ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return  { } ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2024-10-14 07:49:12 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    if  ( ! context - > cached_source_range 
							 
						 
					
						
							
								
									
										
										
										
											2025-04-28 19:02:56 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        | |  context - > cached_source_range - > program_counter  ! =  context - > program_counter )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        auto  unrealized_source_range  =  context - > executable - > source_range_at ( context - > program_counter ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2024-10-14 07:49:12 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        context - > cached_source_range  =  adopt_ref ( * new  CachedSourceRange ( 
							 
						 
					
						
							
								
									
										
										
										
											2025-04-28 19:02:56 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								            context - > program_counter , 
							 
						 
					
						
							
								
									
										
										
										
											2024-10-14 07:49:12 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								            move ( unrealized_source_range ) ) ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    } 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    return  context - > cached_source_range ; 
							 
						 
					
						
							
								
									
										
										
										
											2023-11-01 00:39:28 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								}  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								Vector < StackTraceElement >  VM : : stack_trace ( )  const  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								{  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    Vector < StackTraceElement >  stack_trace ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    for  ( ssize_t  i  =  m_execution_context_stack . size ( )  -  1 ;  i  > =  0 ;  i - - )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        auto *  context  =  m_execution_context_stack [ i ] ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        stack_trace . append ( { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            . execution_context  =  context , 
							 
						 
					
						
							
								
									
										
										
										
											2024-10-14 07:49:12 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								            . source_range  =  get_source_range ( context ) , 
							 
						 
					
						
							
								
									
										
										
										
											2023-11-01 00:39:28 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        } ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    } 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    return  stack_trace ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								}  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2020-09-20 19:24:44 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								}