FunctionEnvironment.cpp 3.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101
  1. /*
  2. * Copyright (c) 2021, Andreas Kling <kling@serenityos.org>
  3. *
  4. * SPDX-License-Identifier: BSD-2-Clause
  5. */
  6. #include <LibJS/Interpreter.h>
  7. #include <LibJS/Runtime/Completion.h>
  8. #include <LibJS/Runtime/FunctionEnvironment.h>
  9. #include <LibJS/Runtime/GlobalObject.h>
  10. namespace JS {
  11. FunctionEnvironment::FunctionEnvironment(Environment* parent_scope)
  12. : DeclarativeEnvironment(parent_scope)
  13. {
  14. }
  15. FunctionEnvironment::~FunctionEnvironment()
  16. {
  17. }
  18. void FunctionEnvironment::visit_edges(Visitor& visitor)
  19. {
  20. Base::visit_edges(visitor);
  21. visitor.visit(m_this_value);
  22. visitor.visit(m_new_target);
  23. visitor.visit(m_function_object);
  24. }
  25. // 9.1.1.3.5 GetSuperBase ( ), https://tc39.es/ecma262/#sec-getsuperbase
  26. ThrowCompletionOr<Value> FunctionEnvironment::get_super_base() const
  27. {
  28. VERIFY(m_function_object);
  29. // 1. Let home be envRec.[[FunctionObject]].[[HomeObject]].
  30. auto home_object = m_function_object->home_object();
  31. // 2. If home has the value undefined, return undefined.
  32. if (!home_object)
  33. return js_undefined();
  34. // 3. Assert: Type(home) is Object.
  35. // 4. Return ? home.[[GetPrototypeOf]]().
  36. return TRY(home_object->internal_get_prototype_of());
  37. }
  38. // 9.1.1.3.2 HasThisBinding ( ), https://tc39.es/ecma262/#sec-function-environment-records-hasthisbinding
  39. bool FunctionEnvironment::has_this_binding() const
  40. {
  41. if (this_binding_status() == ThisBindingStatus::Lexical)
  42. return false;
  43. return true;
  44. }
  45. // 9.1.1.3.3 HasSuperBinding ( ), https://tc39.es/ecma262/#sec-function-environment-records-hassuperbinding
  46. bool FunctionEnvironment::has_super_binding() const
  47. {
  48. if (this_binding_status() == ThisBindingStatus::Lexical)
  49. return false;
  50. if (!function_object().home_object())
  51. return false;
  52. return true;
  53. }
  54. // 9.1.1.3.4 GetThisBinding ( ), https://tc39.es/ecma262/#sec-function-environment-records-getthisbinding
  55. ThrowCompletionOr<Value> FunctionEnvironment::get_this_binding(GlobalObject& global_object) const
  56. {
  57. // 1. Assert: envRec.[[ThisBindingStatus]] is not lexical.
  58. VERIFY(m_this_binding_status != ThisBindingStatus::Lexical);
  59. // 2. If envRec.[[ThisBindingStatus]] is uninitialized, throw a ReferenceError exception.
  60. if (m_this_binding_status == ThisBindingStatus::Uninitialized)
  61. return vm().throw_completion<ReferenceError>(global_object, ErrorType::ThisHasNotBeenInitialized);
  62. // 3. Return envRec.[[ThisValue]].
  63. return m_this_value;
  64. }
  65. // 9.1.1.3.1 BindThisValue ( V ), https://tc39.es/ecma262/#sec-bindthisvalue
  66. ThrowCompletionOr<Value> FunctionEnvironment::bind_this_value(GlobalObject& global_object, Value this_value)
  67. {
  68. // 1. Assert: envRec.[[ThisBindingStatus]] is not lexical.
  69. VERIFY(m_this_binding_status != ThisBindingStatus::Lexical);
  70. // 2. If envRec.[[ThisBindingStatus]] is initialized, throw a ReferenceError exception.
  71. if (m_this_binding_status == ThisBindingStatus::Initialized)
  72. return vm().throw_completion<ReferenceError>(global_object, ErrorType::ThisIsAlreadyInitialized);
  73. // 3. Set envRec.[[ThisValue]] to V.
  74. m_this_value = this_value;
  75. // 4. Set envRec.[[ThisBindingStatus]] to initialized.
  76. m_this_binding_status = ThisBindingStatus::Initialized;
  77. // 5. Return V.
  78. return this_value;
  79. }
  80. }