/* * Copyright (c) 2020-2021, Andreas Kling * * SPDX-License-Identifier: BSD-2-Clause */ #pragma once #include #include #include #include #include #include namespace JS { class DeclarativeEnvironment : public Environment { JS_ENVIRONMENT(DeclarativeEnvironment, Environment); struct Binding { DeprecatedFlyString name; Value value; bool strict { false }; bool mutable_ { false }; bool can_be_deleted { false }; bool initialized { false }; }; public: static DeclarativeEnvironment* create_for_per_iteration_bindings(Badge, DeclarativeEnvironment& other, size_t bindings_size); virtual ~DeclarativeEnvironment() override = default; virtual ThrowCompletionOr has_binding(DeprecatedFlyString const& name, Optional* = nullptr) const override final; virtual ThrowCompletionOr create_mutable_binding(VM&, DeprecatedFlyString const& name, bool can_be_deleted) override final; virtual ThrowCompletionOr create_immutable_binding(VM&, DeprecatedFlyString const& name, bool strict) override final; virtual ThrowCompletionOr initialize_binding(VM&, DeprecatedFlyString const& name, Value, InitializeBindingHint) override final; virtual ThrowCompletionOr set_mutable_binding(VM&, DeprecatedFlyString const& name, Value, bool strict) override final; virtual ThrowCompletionOr get_binding_value(VM&, DeprecatedFlyString const& name, bool strict) override; virtual ThrowCompletionOr delete_binding(VM&, DeprecatedFlyString const& name) override; void initialize_or_set_mutable_binding(Badge, VM&, DeprecatedFlyString const& name, Value value); ThrowCompletionOr initialize_or_set_mutable_binding(VM&, DeprecatedFlyString const& name, Value value); // This is not a method defined in the spec! Do not use this in any LibJS (or other spec related) code. [[nodiscard]] Vector bindings() const { Vector names; names.ensure_capacity(m_bindings.size()); for (auto const& binding : m_bindings) names.unchecked_append(binding.name); return names; } ThrowCompletionOr set_mutable_binding_direct(VM&, size_t index, Value, bool strict); ThrowCompletionOr get_binding_value_direct(VM&, size_t index, bool strict); void shrink_to_fit(); [[nodiscard]] u64 environment_serial_number() const { return m_environment_serial_number; } private: ThrowCompletionOr get_binding_value_direct(VM&, Binding&, bool strict); ThrowCompletionOr set_mutable_binding_direct(VM&, Binding&, Value, bool strict); friend Completion dispose_resources(VM&, GCPtr, Completion); Vector const& disposable_resource_stack() const { return m_disposable_resource_stack; } protected: DeclarativeEnvironment(); explicit DeclarativeEnvironment(Environment* parent_environment); DeclarativeEnvironment(Environment* parent_environment, ReadonlySpan bindings); virtual void visit_edges(Visitor&) override; class BindingAndIndex { public: Binding& binding() { if (m_referenced_binding) return *m_referenced_binding; return m_temporary_binding; } BindingAndIndex(Binding* binding, Optional index) : m_referenced_binding(binding) , m_index(move(index)) { } explicit BindingAndIndex(Binding temporary_binding) : m_temporary_binding(move(temporary_binding)) { } Optional const& index() const { return m_index; } private: Binding* m_referenced_binding { nullptr }; Binding m_temporary_binding {}; Optional m_index; }; friend class ModuleEnvironment; virtual Optional find_binding_and_index(DeprecatedFlyString const& name) const { auto it = m_bindings.find_if([&](auto const& binding) { return binding.name == name; }); if (it == m_bindings.end()) return {}; return BindingAndIndex { const_cast(&(*it)), it.index() }; } private: virtual bool is_declarative_environment() const override { return true; } Vector m_bindings; Vector m_disposable_resource_stack; u64 m_environment_serial_number { 0 }; }; template<> inline bool Environment::fast_is() const { return is_declarative_environment(); } }