123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138 |
- /*
- * Copyright (c) 2020-2021, Andreas Kling <kling@serenityos.org>
- *
- * SPDX-License-Identifier: BSD-2-Clause
- */
- #pragma once
- #include <AK/DeprecatedFlyString.h>
- #include <AK/HashMap.h>
- #include <LibJS/Runtime/AbstractOperations.h>
- #include <LibJS/Runtime/Completion.h>
- #include <LibJS/Runtime/Environment.h>
- #include <LibJS/Runtime/Value.h>
- 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<ForStatement>, DeclarativeEnvironment& other, size_t bindings_size);
- virtual ~DeclarativeEnvironment() override = default;
- virtual ThrowCompletionOr<bool> has_binding(DeprecatedFlyString const& name, Optional<size_t>* = nullptr) const override final;
- virtual ThrowCompletionOr<void> create_mutable_binding(VM&, DeprecatedFlyString const& name, bool can_be_deleted) override final;
- virtual ThrowCompletionOr<void> create_immutable_binding(VM&, DeprecatedFlyString const& name, bool strict) override final;
- virtual ThrowCompletionOr<void> initialize_binding(VM&, DeprecatedFlyString const& name, Value, InitializeBindingHint) override final;
- virtual ThrowCompletionOr<void> set_mutable_binding(VM&, DeprecatedFlyString const& name, Value, bool strict) override final;
- virtual ThrowCompletionOr<Value> get_binding_value(VM&, DeprecatedFlyString const& name, bool strict) override;
- virtual ThrowCompletionOr<bool> delete_binding(VM&, DeprecatedFlyString const& name) override;
- void initialize_or_set_mutable_binding(Badge<ScopeNode>, VM&, DeprecatedFlyString const& name, Value value);
- ThrowCompletionOr<void> 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<DeprecatedFlyString> bindings() const
- {
- Vector<DeprecatedFlyString> names;
- names.ensure_capacity(m_bindings.size());
- for (auto const& binding : m_bindings)
- names.unchecked_append(binding.name);
- return names;
- }
- ThrowCompletionOr<void> set_mutable_binding_direct(VM&, size_t index, Value, bool strict);
- ThrowCompletionOr<Value> get_binding_value_direct(VM&, size_t index, bool strict);
- void shrink_to_fit();
- void ensure_capacity(size_t needed_capacity)
- {
- m_bindings.ensure_capacity(needed_capacity);
- }
- [[nodiscard]] u64 environment_serial_number() const { return m_environment_serial_number; }
- private:
- ThrowCompletionOr<Value> get_binding_value_direct(VM&, Binding&, bool strict);
- ThrowCompletionOr<void> set_mutable_binding_direct(VM&, Binding&, Value, bool strict);
- friend Completion dispose_resources(VM&, GCPtr<DeclarativeEnvironment>, Completion);
- Vector<DisposableResource> const& disposable_resource_stack() const { return m_disposable_resource_stack; }
- protected:
- DeclarativeEnvironment();
- explicit DeclarativeEnvironment(Environment* parent_environment);
- DeclarativeEnvironment(Environment* parent_environment, ReadonlySpan<Binding> 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<size_t> index)
- : m_referenced_binding(binding)
- , m_index(move(index))
- {
- }
- explicit BindingAndIndex(Binding temporary_binding)
- : m_temporary_binding(move(temporary_binding))
- {
- }
- Optional<size_t> const& index() const { return m_index; }
- private:
- Binding* m_referenced_binding { nullptr };
- Binding m_temporary_binding {};
- Optional<size_t> m_index;
- };
- friend class ModuleEnvironment;
- virtual Optional<BindingAndIndex> 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<Binding*>(&(*it)), it.index() };
- }
- private:
- virtual bool is_declarative_environment() const override { return true; }
- Vector<Binding> m_bindings;
- Vector<DisposableResource> m_disposable_resource_stack;
- u64 m_environment_serial_number { 0 };
- };
- template<>
- inline bool Environment::fast_is<DeclarativeEnvironment>() const { return is_declarative_environment(); }
- }
|