IteratorHelper.cpp 2.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869
  1. /*
  2. * Copyright (c) 2023, Tim Flynn <trflynn89@serenityos.org>
  3. *
  4. * SPDX-License-Identifier: BSD-2-Clause
  5. */
  6. #include <LibJS/Runtime/GlobalObject.h>
  7. #include <LibJS/Runtime/Iterator.h>
  8. #include <LibJS/Runtime/IteratorHelper.h>
  9. #include <LibJS/Runtime/Realm.h>
  10. namespace JS {
  11. JS_DEFINE_ALLOCATOR(IteratorHelper);
  12. ThrowCompletionOr<NonnullGCPtr<IteratorHelper>> IteratorHelper::create(Realm& realm, NonnullGCPtr<IteratorRecord> underlying_iterator, NonnullGCPtr<Closure> closure, GCPtr<AbruptClosure> abrupt_closure)
  13. {
  14. return realm.create<IteratorHelper>(realm, realm.intrinsics().iterator_helper_prototype(), move(underlying_iterator), closure, abrupt_closure);
  15. }
  16. IteratorHelper::IteratorHelper(Realm& realm, Object& prototype, NonnullGCPtr<IteratorRecord> underlying_iterator, NonnullGCPtr<Closure> closure, GCPtr<AbruptClosure> abrupt_closure)
  17. : GeneratorObject(realm, prototype, realm.vm().running_execution_context().copy(), "Iterator Helper"sv)
  18. , m_underlying_iterator(move(underlying_iterator))
  19. , m_closure(closure)
  20. , m_abrupt_closure(abrupt_closure)
  21. {
  22. }
  23. void IteratorHelper::visit_edges(Visitor& visitor)
  24. {
  25. Base::visit_edges(visitor);
  26. visitor.visit(m_underlying_iterator);
  27. visitor.visit(m_closure);
  28. visitor.visit(m_abrupt_closure);
  29. }
  30. Value IteratorHelper::result(Value value)
  31. {
  32. set_generator_state(value.is_undefined() ? GeneratorState::Completed : GeneratorState::SuspendedYield);
  33. return value;
  34. }
  35. ThrowCompletionOr<Value> IteratorHelper::close_result(VM& vm, Completion completion)
  36. {
  37. set_generator_state(GeneratorState::Completed);
  38. return *TRY(iterator_close(vm, underlying_iterator(), move(completion)));
  39. }
  40. ThrowCompletionOr<Value> IteratorHelper::execute(VM& vm, JS::Completion const& completion)
  41. {
  42. ScopeGuard guard { [&] { vm.pop_execution_context(); } };
  43. if (completion.is_abrupt()) {
  44. if (m_abrupt_closure)
  45. return m_abrupt_closure->function()(vm, *this, completion);
  46. return close_result(vm, completion);
  47. }
  48. auto result_value = m_closure->function()(vm, *this);
  49. if (result_value.is_throw_completion()) {
  50. set_generator_state(GeneratorState::Completed);
  51. return result_value;
  52. }
  53. return create_iterator_result_object(vm, result(result_value.release_value()), generator_state() == GeneratorState::Completed);
  54. }
  55. }