| 
									
										
										
										
											2020-04-15 21:58:22 +02:00
										 |  |  | /*
 | 
					
						
							| 
									
										
										
										
											2021-06-21 23:17:24 +02:00
										 |  |  |  * Copyright (c) 2020-2021, Andreas Kling <kling@serenityos.org> | 
					
						
							| 
									
										
										
										
											2020-04-15 21:58:22 +02:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2021-04-22 01:24:48 -07:00
										 |  |  |  * SPDX-License-Identifier: BSD-2-Clause | 
					
						
							| 
									
										
										
										
											2020-04-15 21:58:22 +02:00
										 |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-08 13:31:21 -05:00
										 |  |  | #include <LibJS/Interpreter.h>
 | 
					
						
							| 
									
										
										
										
											2021-06-21 23:17:24 +02:00
										 |  |  | #include <LibJS/Runtime/DeclarativeEnvironmentRecord.h>
 | 
					
						
							| 
									
										
										
										
											2020-06-08 13:31:21 -05:00
										 |  |  | #include <LibJS/Runtime/Error.h>
 | 
					
						
							|  |  |  | #include <LibJS/Runtime/Function.h>
 | 
					
						
							| 
									
										
										
										
											2020-09-27 15:18:55 +02:00
										 |  |  | #include <LibJS/Runtime/GlobalObject.h>
 | 
					
						
							| 
									
										
										
										
											2020-06-08 13:31:21 -05:00
										 |  |  | #include <LibJS/Runtime/Value.h>
 | 
					
						
							| 
									
										
										
										
											2020-04-15 21:58:22 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | namespace JS { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-21 23:17:24 +02:00
										 |  |  | DeclarativeEnvironmentRecord::DeclarativeEnvironmentRecord() | 
					
						
							|  |  |  |     : EnvironmentRecord(nullptr) | 
					
						
							| 
									
										
										
										
											2020-04-15 21:58:22 +02:00
										 |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-22 00:56:14 +02:00
										 |  |  | DeclarativeEnvironmentRecord::DeclarativeEnvironmentRecord(EnvironmentRecord* parent_scope) | 
					
						
							|  |  |  |     : EnvironmentRecord(parent_scope) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-21 23:17:24 +02:00
										 |  |  | DeclarativeEnvironmentRecord::DeclarativeEnvironmentRecord(HashMap<FlyString, Variable> variables, EnvironmentRecord* parent_scope) | 
					
						
							|  |  |  |     : EnvironmentRecord(parent_scope) | 
					
						
							| 
									
										
										
										
											2020-04-15 21:58:22 +02:00
										 |  |  |     , m_variables(move(variables)) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-21 23:17:24 +02:00
										 |  |  | DeclarativeEnvironmentRecord::~DeclarativeEnvironmentRecord() | 
					
						
							| 
									
										
										
										
											2020-04-15 21:58:22 +02:00
										 |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-21 23:17:24 +02:00
										 |  |  | void DeclarativeEnvironmentRecord::visit_edges(Visitor& visitor) | 
					
						
							| 
									
										
										
										
											2020-04-15 21:58:22 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2021-01-28 10:13:47 +01:00
										 |  |  |     Base::visit_edges(visitor); | 
					
						
							| 
									
										
										
										
											2020-04-15 21:58:22 +02:00
										 |  |  |     for (auto& it : m_variables) | 
					
						
							|  |  |  |         visitor.visit(it.value.value); | 
					
						
							| 
									
										
										
										
											2021-06-23 12:26:37 +02:00
										 |  |  |     for (auto& it : m_bindings) | 
					
						
							|  |  |  |         visitor.visit(it.value.value); | 
					
						
							| 
									
										
										
										
											2020-04-15 21:58:22 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-21 23:41:38 +02:00
										 |  |  | Optional<Variable> DeclarativeEnvironmentRecord::get_from_environment_record(FlyString const& name) const | 
					
						
							| 
									
										
										
										
											2020-04-15 21:58:22 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     return m_variables.get(name); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-21 23:41:38 +02:00
										 |  |  | void DeclarativeEnvironmentRecord::put_into_environment_record(FlyString const& name, Variable variable) | 
					
						
							| 
									
										
										
										
											2020-04-15 21:58:22 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-11-28 16:02:27 +01:00
										 |  |  |     m_variables.set(name, variable); | 
					
						
							| 
									
										
										
										
											2020-04-15 21:58:22 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-21 23:41:38 +02:00
										 |  |  | bool DeclarativeEnvironmentRecord::delete_from_environment_record(FlyString const& name) | 
					
						
							| 
									
										
										
										
											2021-06-08 04:00:53 +03:00
										 |  |  | { | 
					
						
							|  |  |  |     return m_variables.remove(name); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-23 12:26:37 +02:00
										 |  |  | // 9.1.1.1.1 HasBinding ( N ), https://tc39.es/ecma262/#sec-declarative-environment-records-hasbinding-n
 | 
					
						
							|  |  |  | bool DeclarativeEnvironmentRecord::has_binding(FlyString const& name) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return m_bindings.contains(name); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-23 13:25:57 +02:00
										 |  |  | // 9.1.1.1.2 CreateMutableBinding ( N, D ), https://tc39.es/ecma262/#sec-declarative-environment-records-createmutablebinding-n-d
 | 
					
						
							| 
									
										
										
										
											2021-06-23 12:26:37 +02:00
										 |  |  | void DeclarativeEnvironmentRecord::create_mutable_binding(GlobalObject&, FlyString const& name, bool can_be_deleted) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     auto result = m_bindings.set(name, | 
					
						
							|  |  |  |         Binding { | 
					
						
							|  |  |  |             .value = {}, | 
					
						
							|  |  |  |             .strict = false, | 
					
						
							|  |  |  |             .mutable_ = true, | 
					
						
							|  |  |  |             .can_be_deleted = can_be_deleted, | 
					
						
							|  |  |  |             .initialized = false, | 
					
						
							|  |  |  |         }); | 
					
						
							|  |  |  |     VERIFY(result == AK::HashSetResult::InsertedNewEntry); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-23 13:25:57 +02:00
										 |  |  | // 9.1.1.1.3 CreateImmutableBinding ( N, S ), https://tc39.es/ecma262/#sec-declarative-environment-records-createimmutablebinding-n-s
 | 
					
						
							| 
									
										
										
										
											2021-06-23 12:26:37 +02:00
										 |  |  | void DeclarativeEnvironmentRecord::create_immutable_binding(GlobalObject&, FlyString const& name, bool strict) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     auto result = m_bindings.set(name, | 
					
						
							|  |  |  |         Binding { | 
					
						
							|  |  |  |             .value = {}, | 
					
						
							|  |  |  |             .strict = strict, | 
					
						
							|  |  |  |             .mutable_ = false, | 
					
						
							|  |  |  |             .can_be_deleted = false, | 
					
						
							|  |  |  |             .initialized = false, | 
					
						
							|  |  |  |         }); | 
					
						
							|  |  |  |     VERIFY(result == AK::HashSetResult::InsertedNewEntry); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-23 13:25:57 +02:00
										 |  |  | // 9.1.1.1.4 InitializeBinding ( N, V ), https://tc39.es/ecma262/#sec-declarative-environment-records-initializebinding-n-v
 | 
					
						
							| 
									
										
										
										
											2021-06-23 12:26:37 +02:00
										 |  |  | void DeclarativeEnvironmentRecord::initialize_binding(GlobalObject&, FlyString const& name, Value value) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     auto it = m_bindings.find(name); | 
					
						
							|  |  |  |     VERIFY(it != m_bindings.end()); | 
					
						
							|  |  |  |     VERIFY(it->value.initialized == false); | 
					
						
							|  |  |  |     it->value.value = value; | 
					
						
							|  |  |  |     it->value.initialized = true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-23 13:25:57 +02:00
										 |  |  | // 9.1.1.1.5 SetMutableBinding ( N, V, S ), https://tc39.es/ecma262/#sec-declarative-environment-records-setmutablebinding-n-v-s
 | 
					
						
							| 
									
										
										
										
											2021-06-23 12:26:37 +02:00
										 |  |  | void DeclarativeEnvironmentRecord::set_mutable_binding(GlobalObject& global_object, FlyString const& name, Value value, bool strict) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     auto it = m_bindings.find(name); | 
					
						
							|  |  |  |     if (it == m_bindings.end()) { | 
					
						
							|  |  |  |         if (strict) { | 
					
						
							|  |  |  |             global_object.vm().throw_exception<ReferenceError>(global_object, ErrorType::UnknownIdentifier, name); | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         create_mutable_binding(global_object, name, true); | 
					
						
							|  |  |  |         initialize_binding(global_object, name, value); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (it->value.strict) | 
					
						
							|  |  |  |         strict = true; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!it->value.initialized) { | 
					
						
							|  |  |  |         global_object.vm().throw_exception<ReferenceError>(global_object, ErrorType::BindingNotInitialized, name); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (it->value.mutable_) { | 
					
						
							|  |  |  |         it->value.value = value; | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         if (strict) { | 
					
						
							|  |  |  |             global_object.vm().throw_exception<TypeError>(global_object, ErrorType::InvalidAssignToConst); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-23 13:25:57 +02:00
										 |  |  | // 9.1.1.1.6 GetBindingValue ( N, S ), https://tc39.es/ecma262/#sec-declarative-environment-records-getbindingvalue-n-s
 | 
					
						
							| 
									
										
										
										
											2021-06-23 12:26:37 +02:00
										 |  |  | Value DeclarativeEnvironmentRecord::get_binding_value(GlobalObject& global_object, FlyString const& name, bool) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     auto it = m_bindings.find(name); | 
					
						
							|  |  |  |     VERIFY(it != m_bindings.end()); | 
					
						
							|  |  |  |     if (!it->value.initialized) { | 
					
						
							|  |  |  |         global_object.vm().throw_exception<ReferenceError>(global_object, ErrorType::BindingNotInitialized, name); | 
					
						
							|  |  |  |         return {}; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return it->value.value; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-23 13:25:57 +02:00
										 |  |  | // 9.1.1.1.7 DeleteBinding ( N ), https://tc39.es/ecma262/#sec-declarative-environment-records-deletebinding-n
 | 
					
						
							| 
									
										
										
										
											2021-06-23 12:26:37 +02:00
										 |  |  | bool DeclarativeEnvironmentRecord::delete_binding(GlobalObject&, FlyString const& name) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     auto it = m_bindings.find(name); | 
					
						
							|  |  |  |     VERIFY(it != m_bindings.end()); | 
					
						
							|  |  |  |     if (!it->value.can_be_deleted) | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     m_bindings.remove(it); | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-15 21:58:22 +02:00
										 |  |  | } |