| 
									
										
										
										
											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>
 | 
					
						
							| 
									
										
										
										
											2020-04-27 12:10:16 +02:00
										 |  |  | #include <LibJS/Runtime/PropertyName.h>
 | 
					
						
							|  |  |  | #include <LibJS/Runtime/Value.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace JS { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 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-06-25 16:27:59 +02:00
										 |  |  |     Reference(BaseType type, PropertyName const& name, bool strict) | 
					
						
							|  |  |  |         : m_base_type(type) | 
					
						
							| 
									
										
										
										
											2020-04-27 12:10:16 +02:00
										 |  |  |         , m_name(name) | 
					
						
							|  |  |  |         , m_strict(strict) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-25 16:27:59 +02:00
										 |  |  |     Reference(Value base, PropertyName const& name, Value this_value, bool strict = false) | 
					
						
							|  |  |  |         : m_base_type(BaseType::Value) | 
					
						
							|  |  |  |         , m_base_value(base) | 
					
						
							| 
									
										
										
										
											2020-04-27 12:37:27 +02:00
										 |  |  |         , m_name(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-07-01 12:24:46 +02:00
										 |  |  |     Reference(Environment& base, FlyString const& referenced_name, bool strict = false) | 
					
						
							|  |  |  |         : m_base_type(BaseType::Environment) | 
					
						
							|  |  |  |         , m_base_environment(&base) | 
					
						
							| 
									
										
										
										
											2021-06-25 16:27:59 +02:00
										 |  |  |         , m_name(referenced_name) | 
					
						
							| 
									
										
										
										
											2020-04-28 14:44:48 +02:00
										 |  |  |         , m_strict(strict) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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-06-25 16:27:59 +02:00
										 |  |  |     PropertyName const& name() const { return m_name; } | 
					
						
							|  |  |  |     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; | 
					
						
							|  |  |  |         if (m_base_value.is_boolean() || m_base_value.is_string() || m_base_value.is_symbol() || m_base_value.is_bigint() || m_base_value.is_number() || m_base_value.is_object()) | 
					
						
							|  |  |  |             return true; | 
					
						
							|  |  |  |         return false; | 
					
						
							| 
									
										
										
										
											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-06-25 17:19:01 +02:00
										 |  |  |     void put_value(GlobalObject&, Value); | 
					
						
							|  |  |  |     Value get_value(GlobalObject&, bool throw_if_undefined = true); | 
					
						
							| 
									
										
										
										
											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; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-27 12:10:16 +02:00
										 |  |  | private: | 
					
						
							| 
									
										
										
										
											2020-09-29 16:45:24 +02:00
										 |  |  |     void throw_reference_error(GlobalObject&); | 
					
						
							| 
									
										
										
										
											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-07-01 12:24:46 +02:00
										 |  |  |         Environment* m_base_environment; | 
					
						
							| 
									
										
										
										
											2021-06-25 16:27:59 +02:00
										 |  |  |     }; | 
					
						
							| 
									
										
										
										
											2020-04-27 12:10:16 +02:00
										 |  |  |     PropertyName 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 }; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } |