DeclarativeEnvironment.h 5.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151
  1. /*
  2. * Copyright (c) 2020-2021, Andreas Kling <andreas@ladybird.org>
  3. *
  4. * SPDX-License-Identifier: BSD-2-Clause
  5. */
  6. #pragma once
  7. #include <AK/DeprecatedFlyString.h>
  8. #include <AK/HashMap.h>
  9. #include <LibJS/Runtime/AbstractOperations.h>
  10. #include <LibJS/Runtime/Completion.h>
  11. #include <LibJS/Runtime/Environment.h>
  12. #include <LibJS/Runtime/Value.h>
  13. namespace JS {
  14. class DeclarativeEnvironment : public Environment {
  15. JS_ENVIRONMENT(DeclarativeEnvironment, Environment);
  16. GC_DECLARE_ALLOCATOR(DeclarativeEnvironment);
  17. struct Binding {
  18. DeprecatedFlyString name;
  19. Value value;
  20. bool strict { false };
  21. bool mutable_ { false };
  22. bool can_be_deleted { false };
  23. bool initialized { false };
  24. };
  25. public:
  26. static DeclarativeEnvironment* create_for_per_iteration_bindings(Badge<ForStatement>, DeclarativeEnvironment& other, size_t bindings_size);
  27. virtual ~DeclarativeEnvironment() override = default;
  28. virtual ThrowCompletionOr<bool> has_binding(DeprecatedFlyString const& name, Optional<size_t>* = nullptr) const override final;
  29. virtual ThrowCompletionOr<void> create_mutable_binding(VM&, DeprecatedFlyString const& name, bool can_be_deleted) override final;
  30. virtual ThrowCompletionOr<void> create_immutable_binding(VM&, DeprecatedFlyString const& name, bool strict) override final;
  31. virtual ThrowCompletionOr<void> initialize_binding(VM&, DeprecatedFlyString const& name, Value, InitializeBindingHint) override final;
  32. virtual ThrowCompletionOr<void> set_mutable_binding(VM&, DeprecatedFlyString const& name, Value, bool strict) override final;
  33. virtual ThrowCompletionOr<Value> get_binding_value(VM&, DeprecatedFlyString const& name, bool strict) override;
  34. virtual ThrowCompletionOr<bool> delete_binding(VM&, DeprecatedFlyString const& name) override;
  35. void initialize_or_set_mutable_binding(Badge<ScopeNode>, VM&, DeprecatedFlyString const& name, Value value);
  36. ThrowCompletionOr<void> initialize_or_set_mutable_binding(VM&, DeprecatedFlyString const& name, Value value);
  37. // This is not a method defined in the spec! Do not use this in any LibJS (or other spec related) code.
  38. [[nodiscard]] Vector<DeprecatedFlyString> bindings() const
  39. {
  40. Vector<DeprecatedFlyString> names;
  41. names.ensure_capacity(m_bindings.size());
  42. for (auto const& binding : m_bindings)
  43. names.unchecked_append(binding.name);
  44. return names;
  45. }
  46. ThrowCompletionOr<void> initialize_binding_direct(VM&, size_t index, Value, InitializeBindingHint);
  47. ThrowCompletionOr<void> set_mutable_binding_direct(VM&, size_t index, Value, bool strict);
  48. ThrowCompletionOr<Value> get_binding_value_direct(VM&, size_t index) const;
  49. void shrink_to_fit();
  50. void ensure_capacity(size_t needed_capacity)
  51. {
  52. m_bindings.ensure_capacity(needed_capacity);
  53. }
  54. [[nodiscard]] u64 environment_serial_number() const { return m_environment_serial_number; }
  55. private:
  56. ThrowCompletionOr<Value> get_binding_value_direct(VM&, Binding const&) const;
  57. ThrowCompletionOr<void> set_mutable_binding_direct(VM&, Binding&, Value, bool strict);
  58. friend Completion dispose_resources(VM&, GC::Ptr<DeclarativeEnvironment>, Completion);
  59. Vector<DisposableResource> const& disposable_resource_stack() const { return m_disposable_resource_stack; }
  60. protected:
  61. DeclarativeEnvironment();
  62. explicit DeclarativeEnvironment(Environment* parent_environment);
  63. DeclarativeEnvironment(Environment* parent_environment, ReadonlySpan<Binding> bindings);
  64. virtual void visit_edges(Visitor&) override;
  65. class BindingAndIndex {
  66. public:
  67. Binding& binding()
  68. {
  69. if (m_referenced_binding)
  70. return *m_referenced_binding;
  71. return m_temporary_binding;
  72. }
  73. BindingAndIndex(Binding* binding, Optional<size_t> index)
  74. : m_referenced_binding(binding)
  75. , m_index(move(index))
  76. {
  77. }
  78. explicit BindingAndIndex(Binding temporary_binding)
  79. : m_temporary_binding(move(temporary_binding))
  80. {
  81. }
  82. Optional<size_t> const& index() const { return m_index; }
  83. private:
  84. Binding* m_referenced_binding { nullptr };
  85. Binding m_temporary_binding {};
  86. Optional<size_t> m_index;
  87. };
  88. friend class ModuleEnvironment;
  89. virtual Optional<BindingAndIndex> find_binding_and_index(DeprecatedFlyString const& name) const
  90. {
  91. if (auto it = m_bindings_assoc.find(name); it != m_bindings_assoc.end()) {
  92. return BindingAndIndex { const_cast<Binding*>(&m_bindings.at(it->value)), it->value };
  93. }
  94. return {};
  95. }
  96. private:
  97. Vector<Binding> m_bindings;
  98. HashMap<DeprecatedFlyString, size_t> m_bindings_assoc;
  99. Vector<DisposableResource> m_disposable_resource_stack;
  100. u64 m_environment_serial_number { 0 };
  101. };
  102. inline ThrowCompletionOr<Value> DeclarativeEnvironment::get_binding_value_direct(VM& vm, size_t index) const
  103. {
  104. return get_binding_value_direct(vm, m_bindings[index]);
  105. }
  106. inline ThrowCompletionOr<Value> DeclarativeEnvironment::get_binding_value_direct(VM&, Binding const& binding) const
  107. {
  108. // 2. If the binding for N in envRec is an uninitialized binding, throw a ReferenceError exception.
  109. if (!binding.initialized)
  110. return vm().throw_completion<ReferenceError>(ErrorType::BindingNotInitialized, binding.name);
  111. // 3. Return the value currently bound to N in envRec.
  112. return binding.value;
  113. }
  114. template<>
  115. inline bool Environment::fast_is<DeclarativeEnvironment>() const { return is_declarative_environment(); }
  116. }