IteratorHelper.cpp 2.2 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465
  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. ThrowCompletionOr<NonnullGCPtr<IteratorHelper>> IteratorHelper::create(Realm& realm, IteratorRecord underlying_iterator, Closure closure, Optional<AbruptClosure> abrupt_closure)
  12. {
  13. return realm.heap().allocate<IteratorHelper>(realm, realm, realm.intrinsics().iterator_helper_prototype(), move(underlying_iterator), move(closure), move(abrupt_closure));
  14. }
  15. IteratorHelper::IteratorHelper(Realm& realm, Object& prototype, IteratorRecord underlying_iterator, Closure closure, Optional<AbruptClosure> abrupt_closure)
  16. : GeneratorObject(realm, prototype, realm.vm().running_execution_context().copy(), "Iterator Helper"sv)
  17. , m_underlying_iterator(move(underlying_iterator))
  18. , m_closure(move(closure))
  19. , m_abrupt_closure(move(abrupt_closure))
  20. {
  21. }
  22. void IteratorHelper::visit_edges(Visitor& visitor)
  23. {
  24. Base::visit_edges(visitor);
  25. visitor.visit(m_underlying_iterator.iterator);
  26. }
  27. Value IteratorHelper::result(Value value)
  28. {
  29. set_generator_state(value.is_undefined() ? GeneratorState::Completed : GeneratorState::SuspendedYield);
  30. return value;
  31. }
  32. ThrowCompletionOr<Value> IteratorHelper::close_result(VM& vm, Completion completion)
  33. {
  34. set_generator_state(GeneratorState::Completed);
  35. return *TRY(iterator_close(vm, underlying_iterator(), move(completion)));
  36. }
  37. ThrowCompletionOr<Value> IteratorHelper::execute(VM& vm, JS::Completion const& completion)
  38. {
  39. ScopeGuard guard { [&] { vm.pop_execution_context(); } };
  40. if (completion.is_abrupt()) {
  41. if (m_abrupt_closure.has_value())
  42. return (*m_abrupt_closure)(vm, *this, completion);
  43. return close_result(vm, completion);
  44. }
  45. auto result_value = m_closure(vm, *this);
  46. if (result_value.is_throw_completion()) {
  47. set_generator_state(GeneratorState::Completed);
  48. return result_value;
  49. }
  50. return create_iterator_result_object(vm, result(result_value.release_value()), generator_state() == GeneratorState::Completed);
  51. }
  52. }