mirror of
				https://github.com/RGBCube/serenity
				synced 2025-10-26 11:22:32 +00:00 
			
		
		
		
	 2b19d1b5ab
			
		
	
	
		2b19d1b5ab
		
	
	
	
	
		
			
			These are not strictly unresolvable references. Treating them as such fails an assertion in the `delete UnaryExpression` semantic (which is Reference::delete_ in our implementation) - we enter the unresolvable, branch, which then asserts that the [[Strict]] slot of the reference is false.
		
			
				
	
	
		
			146 lines
		
	
	
	
		
			4.1 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			146 lines
		
	
	
	
		
			4.1 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*
 | |
|  * Copyright (c) 2020-2021, Andreas Kling <kling@serenityos.org>
 | |
|  *
 | |
|  * SPDX-License-Identifier: BSD-2-Clause
 | |
|  */
 | |
| 
 | |
| #pragma once
 | |
| 
 | |
| #include <LibJS/Runtime/Environment.h>
 | |
| #include <LibJS/Runtime/EnvironmentCoordinate.h>
 | |
| #include <LibJS/Runtime/PropertyKey.h>
 | |
| #include <LibJS/Runtime/Value.h>
 | |
| 
 | |
| namespace JS {
 | |
| 
 | |
| Reference make_private_reference(VM&, Value base_value, DeprecatedFlyString const& private_identifier);
 | |
| 
 | |
| class Reference {
 | |
| public:
 | |
|     enum class BaseType : u8 {
 | |
|         Unresolvable,
 | |
|         Value,
 | |
|         Environment,
 | |
|     };
 | |
| 
 | |
|     Reference() = default;
 | |
|     Reference(BaseType type, PropertyKey name, bool strict)
 | |
|         : m_base_type(type)
 | |
|         , m_name(move(name))
 | |
|         , m_strict(strict)
 | |
|     {
 | |
|     }
 | |
| 
 | |
|     Reference(Value base, PropertyKey name, Value this_value, bool strict = false)
 | |
|         : m_base_type(BaseType::Value)
 | |
|         , m_base_value(base)
 | |
|         , m_name(move(name))
 | |
|         , m_this_value(this_value)
 | |
|         , m_strict(strict)
 | |
|     {
 | |
|     }
 | |
| 
 | |
|     Reference(Environment& base, DeprecatedFlyString referenced_name, bool strict = false, Optional<EnvironmentCoordinate> environment_coordinate = {})
 | |
|         : m_base_type(BaseType::Environment)
 | |
|         , m_base_environment(&base)
 | |
|         , m_name(move(referenced_name))
 | |
|         , m_strict(strict)
 | |
|         , m_environment_coordinate(move(environment_coordinate))
 | |
|     {
 | |
|     }
 | |
| 
 | |
|     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))
 | |
|     {
 | |
|     }
 | |
| 
 | |
|     Value base() const
 | |
|     {
 | |
|         VERIFY(m_base_type == BaseType::Value);
 | |
|         return m_base_value;
 | |
|     }
 | |
| 
 | |
|     Environment& base_environment() const
 | |
|     {
 | |
|         VERIFY(m_base_type == BaseType::Environment);
 | |
|         return *m_base_environment;
 | |
|     }
 | |
| 
 | |
|     PropertyKey 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
 | |
|     {
 | |
|         if (is_unresolvable())
 | |
|             return false;
 | |
|         if (m_base_type == BaseType::Environment)
 | |
|             return false;
 | |
|         return true;
 | |
|     }
 | |
| 
 | |
|     // 6.2.4.7 GetThisValue ( V ), https://tc39.es/ecma262/#sec-getthisvalue
 | |
|     Value get_this_value() const
 | |
|     {
 | |
|         VERIFY(is_property_reference());
 | |
|         if (is_super_reference())
 | |
|             return m_this_value;
 | |
|         return m_base_value;
 | |
|     }
 | |
| 
 | |
|     // 6.2.4.3 IsSuperReference ( V ), https://tc39.es/ecma262/#sec-issuperreference
 | |
|     bool is_super_reference() const
 | |
|     {
 | |
|         return !m_this_value.is_empty();
 | |
|     }
 | |
| 
 | |
|     // 6.2.4.4 IsPrivateReference ( V ), https://tc39.es/ecma262/#sec-isprivatereference
 | |
|     bool is_private_reference() const
 | |
|     {
 | |
|         return m_is_private;
 | |
|     }
 | |
| 
 | |
|     // Note: Non-standard helper.
 | |
|     bool is_environment_reference() const
 | |
|     {
 | |
|         return m_base_type == BaseType::Environment;
 | |
|     }
 | |
| 
 | |
|     ThrowCompletionOr<void> initialize_referenced_binding(VM&, Value value, Environment::InitializeBindingHint hint = Environment::InitializeBindingHint::Normal) const;
 | |
| 
 | |
|     ThrowCompletionOr<void> put_value(VM&, Value);
 | |
|     ThrowCompletionOr<Value> get_value(VM&) const;
 | |
|     ThrowCompletionOr<bool> delete_(VM&);
 | |
| 
 | |
|     bool is_valid_reference() const { return m_name.is_valid() || m_is_private; }
 | |
| 
 | |
|     Optional<EnvironmentCoordinate> environment_coordinate() const { return m_environment_coordinate; }
 | |
| 
 | |
| private:
 | |
|     Completion throw_reference_error(VM&) const;
 | |
| 
 | |
|     BaseType m_base_type { BaseType::Unresolvable };
 | |
|     union {
 | |
|         Value m_base_value {};
 | |
|         mutable Environment* m_base_environment;
 | |
|     };
 | |
|     PropertyKey m_name;
 | |
|     Value m_this_value;
 | |
|     bool m_strict { false };
 | |
| 
 | |
|     bool m_is_private { false };
 | |
|     // FIXME: This can (probably) be an union with m_name.
 | |
|     PrivateName m_private_name;
 | |
| 
 | |
|     Optional<EnvironmentCoordinate> m_environment_coordinate;
 | |
| };
 | |
| 
 | |
| }
 |