| 
									
										
										
										
											2020-04-27 12:10:16 +02:00
										 |  |  | /*
 | 
					
						
							| 
									
										
										
										
											2021-06-25 16:27:59 +02:00
										 |  |  |  * Copyright (c) 2020-2021, Andreas Kling <kling@serenityos.org> | 
					
						
							| 
									
										
										
										
											2020-04-27 12:10:16 +02:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2021-04-22 01:24:48 -07:00
										 |  |  |  * SPDX-License-Identifier: BSD-2-Clause | 
					
						
							| 
									
										
										
										
											2020-04-27 12:10:16 +02:00
										 |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #pragma once
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <AK/String.h>
 | 
					
						
							| 
									
										
										
										
											2021-07-01 12:24:46 +02:00
										 |  |  | #include <LibJS/Runtime/Environment.h>
 | 
					
						
							| 
									
										
										
										
											2021-10-07 01:06:21 +02:00
										 |  |  | #include <LibJS/Runtime/EnvironmentCoordinate.h>
 | 
					
						
							| 
									
										
										
										
											2021-10-07 20:13:22 +03:30
										 |  |  | #include <LibJS/Runtime/ExecutionContext.h>
 | 
					
						
							| 
									
										
										
										
											2021-10-24 16:01:24 +02:00
										 |  |  | #include <LibJS/Runtime/PropertyKey.h>
 | 
					
						
							| 
									
										
										
										
											2020-04-27 12:10:16 +02:00
										 |  |  | #include <LibJS/Runtime/Value.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace JS { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-12 21:48:50 +02:00
										 |  |  | Reference make_private_reference(VM&, Value base_value, FlyString const& private_identifier); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-27 12:10:16 +02:00
										 |  |  | class Reference { | 
					
						
							|  |  |  | public: | 
					
						
							| 
									
										
										
										
											2021-06-25 16:27:59 +02:00
										 |  |  |     enum class BaseType : u8 { | 
					
						
							|  |  |  |         Unresolvable, | 
					
						
							|  |  |  |         Value, | 
					
						
							| 
									
										
										
										
											2021-07-01 12:24:46 +02:00
										 |  |  |         Environment, | 
					
						
							| 
									
										
										
										
											2021-06-25 16:27:59 +02:00
										 |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-20 16:47:31 +02:00
										 |  |  |     Reference() { } | 
					
						
							| 
									
										
										
										
											2021-10-24 16:01:24 +02:00
										 |  |  |     Reference(BaseType type, PropertyKey name, bool strict) | 
					
						
							| 
									
										
										
										
											2021-06-25 16:27:59 +02:00
										 |  |  |         : m_base_type(type) | 
					
						
							| 
									
										
										
										
											2021-09-11 19:03:38 +02:00
										 |  |  |         , m_name(move(name)) | 
					
						
							| 
									
										
										
										
											2020-04-27 12:10:16 +02:00
										 |  |  |         , m_strict(strict) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-24 16:01:24 +02:00
										 |  |  |     Reference(Value base, PropertyKey name, Value this_value, bool strict = false) | 
					
						
							| 
									
										
										
										
											2021-06-25 16:27:59 +02:00
										 |  |  |         : m_base_type(BaseType::Value) | 
					
						
							|  |  |  |         , m_base_value(base) | 
					
						
							| 
									
										
										
										
											2021-09-11 19:03:38 +02:00
										 |  |  |         , m_name(move(name)) | 
					
						
							| 
									
										
										
										
											2021-06-25 16:27:59 +02:00
										 |  |  |         , m_this_value(this_value) | 
					
						
							| 
									
										
										
										
											2020-04-27 12:37:27 +02:00
										 |  |  |         , m_strict(strict) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2021-06-25 16:27:59 +02:00
										 |  |  |         if (base.is_nullish()) { | 
					
						
							|  |  |  |             m_base_type = BaseType::Unresolvable; | 
					
						
							|  |  |  |             m_base_value = {}; | 
					
						
							|  |  |  |             m_this_value = {}; | 
					
						
							|  |  |  |             m_name = {}; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-04-27 12:37:27 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-07 01:06:21 +02:00
										 |  |  |     Reference(Environment& base, FlyString referenced_name, bool strict = false, Optional<EnvironmentCoordinate> environment_coordinate = {}) | 
					
						
							| 
									
										
										
										
											2021-07-01 12:24:46 +02:00
										 |  |  |         : m_base_type(BaseType::Environment) | 
					
						
							|  |  |  |         , m_base_environment(&base) | 
					
						
							| 
									
										
										
										
											2021-09-11 19:03:38 +02:00
										 |  |  |         , m_name(move(referenced_name)) | 
					
						
							| 
									
										
										
										
											2020-04-28 14:44:48 +02:00
										 |  |  |         , m_strict(strict) | 
					
						
							| 
									
										
										
										
											2021-10-07 01:06:21 +02:00
										 |  |  |         , m_environment_coordinate(move(environment_coordinate)) | 
					
						
							| 
									
										
										
										
											2020-04-28 14:44:48 +02:00
										 |  |  |     { | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-12 21:48:50 +02:00
										 |  |  |     Reference(Value base, PrivateName name) | 
					
						
							|  |  |  |         : m_base_type(BaseType::Value) | 
					
						
							|  |  |  |         , m_base_value(base) | 
					
						
							|  |  |  |         , m_this_value(Value {}) | 
					
						
							|  |  |  |         , m_strict(true) | 
					
						
							|  |  |  |         , m_is_private(true) | 
					
						
							|  |  |  |         , m_private_name(move(name)) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-25 16:27:59 +02:00
										 |  |  |     Value base() const | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         VERIFY(m_base_type == BaseType::Value); | 
					
						
							|  |  |  |         return m_base_value; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-04-27 12:10:16 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-01 12:24:46 +02:00
										 |  |  |     Environment& base_environment() const | 
					
						
							| 
									
										
										
										
											2020-04-27 12:10:16 +02:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2021-07-01 12:24:46 +02:00
										 |  |  |         VERIFY(m_base_type == BaseType::Environment); | 
					
						
							|  |  |  |         return *m_base_environment; | 
					
						
							| 
									
										
										
										
											2020-04-27 12:10:16 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-24 16:01:24 +02:00
										 |  |  |     PropertyKey const& name() const { return m_name; } | 
					
						
							| 
									
										
										
										
											2021-06-25 16:27:59 +02:00
										 |  |  |     bool is_strict() const { return m_strict; } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // 6.2.4.2 IsUnresolvableReference ( V ), https://tc39.es/ecma262/#sec-isunresolvablereference
 | 
					
						
							|  |  |  |     bool is_unresolvable() const { return m_base_type == BaseType::Unresolvable; } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // 6.2.4.1 IsPropertyReference ( V ), https://tc39.es/ecma262/#sec-ispropertyreference
 | 
					
						
							|  |  |  |     bool is_property_reference() const | 
					
						
							| 
									
										
										
										
											2020-04-27 12:10:16 +02:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2021-06-25 16:27:59 +02:00
										 |  |  |         if (is_unresolvable()) | 
					
						
							|  |  |  |             return false; | 
					
						
							| 
									
										
										
										
											2021-07-01 12:24:46 +02:00
										 |  |  |         if (m_base_type == BaseType::Environment) | 
					
						
							| 
									
										
										
										
											2021-06-25 16:27:59 +02:00
										 |  |  |             return false; | 
					
						
							| 
									
										
										
										
											2021-10-12 21:48:50 +02:00
										 |  |  |         return true; | 
					
						
							| 
									
										
										
										
											2020-04-27 12:10:16 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-25 16:27:59 +02:00
										 |  |  |     // 6.2.4.7 GetThisValue ( V ), https://tc39.es/ecma262/#sec-getthisvalue
 | 
					
						
							|  |  |  |     Value get_this_value() const | 
					
						
							| 
									
										
										
										
											2020-04-27 12:37:27 +02:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2021-06-25 16:27:59 +02:00
										 |  |  |         VERIFY(is_property_reference()); | 
					
						
							|  |  |  |         if (is_super_reference()) | 
					
						
							|  |  |  |             return m_this_value; | 
					
						
							|  |  |  |         return m_base_value; | 
					
						
							| 
									
										
										
										
											2020-04-27 12:37:27 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-25 16:27:59 +02:00
										 |  |  |     // 6.2.4.3 IsSuperReference ( V ), https://tc39.es/ecma262/#sec-issuperreference
 | 
					
						
							|  |  |  |     bool is_super_reference() const | 
					
						
							| 
									
										
										
										
											2020-04-28 14:44:48 +02:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2021-06-25 16:27:59 +02:00
										 |  |  |         return !m_this_value.is_empty(); | 
					
						
							| 
									
										
										
										
											2020-04-28 14:44:48 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-12 21:48:50 +02:00
										 |  |  |     // 6.2.4.4 IsPrivateReference ( V ), https://tc39.es/ecma262/#sec-isprivatereference
 | 
					
						
							|  |  |  |     bool is_private_reference() const | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         return m_is_private; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-14 06:56:31 +04:30
										 |  |  |     // Note: Non-standard helper.
 | 
					
						
							|  |  |  |     bool is_environment_reference() const | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         return m_base_type == BaseType::Environment; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-22 12:44:56 +02:00
										 |  |  |     void initialize_referenced_binding(GlobalObject& global_object, Value value) const | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         VERIFY(!is_unresolvable()); | 
					
						
							|  |  |  |         VERIFY(m_base_type == BaseType::Environment); | 
					
						
							| 
									
										
										
										
											2021-10-09 19:16:24 +01:00
										 |  |  |         (void)m_base_environment->initialize_binding(global_object, m_name.as_string(), value); | 
					
						
							| 
									
										
										
										
											2021-09-22 12:44:56 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-25 17:19:01 +02:00
										 |  |  |     void put_value(GlobalObject&, Value); | 
					
						
							| 
									
										
										
										
											2021-09-22 12:44:56 +02:00
										 |  |  |     Value get_value(GlobalObject&) const; | 
					
						
							| 
									
										
										
										
											2021-06-08 04:00:53 +03:00
										 |  |  |     bool delete_(GlobalObject&); | 
					
						
							| 
									
										
										
										
											2020-04-27 12:56:09 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-25 16:27:59 +02:00
										 |  |  |     String to_string() const; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-12 21:48:50 +02:00
										 |  |  |     bool is_valid_reference() const { return m_name.is_valid() || m_is_private; } | 
					
						
							| 
									
										
										
										
											2021-09-27 22:28:56 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-07 01:06:21 +02:00
										 |  |  |     Optional<EnvironmentCoordinate> environment_coordinate() const { return m_environment_coordinate; } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-27 12:10:16 +02:00
										 |  |  | private: | 
					
						
							| 
									
										
										
										
											2021-09-14 16:53:08 +04:30
										 |  |  |     void throw_reference_error(GlobalObject&) const; | 
					
						
							| 
									
										
										
										
											2020-04-28 14:44:48 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-25 16:27:59 +02:00
										 |  |  |     BaseType m_base_type { BaseType::Unresolvable }; | 
					
						
							|  |  |  |     union { | 
					
						
							| 
									
										
										
										
											2021-06-25 17:25:55 +02:00
										 |  |  |         Value m_base_value {}; | 
					
						
							| 
									
										
										
										
											2021-10-07 20:13:22 +03:30
										 |  |  |         mutable Environment* m_base_environment; | 
					
						
							| 
									
										
										
										
											2021-06-25 16:27:59 +02:00
										 |  |  |     }; | 
					
						
							| 
									
										
										
										
											2021-10-24 16:01:24 +02:00
										 |  |  |     PropertyKey m_name; | 
					
						
							| 
									
										
										
										
											2021-06-25 16:27:59 +02:00
										 |  |  |     Value m_this_value; | 
					
						
							| 
									
										
										
										
											2020-04-27 12:10:16 +02:00
										 |  |  |     bool m_strict { false }; | 
					
						
							| 
									
										
										
										
											2021-10-12 21:48:50 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     bool m_is_private { false }; | 
					
						
							|  |  |  |     // FIXME: This can (probably) be an union with m_name.
 | 
					
						
							| 
									
										
										
										
											2021-10-25 13:28:51 +02:00
										 |  |  |     PrivateName m_private_name; | 
					
						
							| 
									
										
										
										
											2021-10-12 21:48:50 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-07 01:06:21 +02:00
										 |  |  |     Optional<EnvironmentCoordinate> m_environment_coordinate; | 
					
						
							| 
									
										
										
										
											2020-04-27 12:10:16 +02:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } |