mirror of
https://github.com/RGBCube/serenity
synced 2025-07-27 22:37:35 +00:00
LibJS: Handle abrupt closures from Iterator.prototype.flatMap
This is in preparation of implementing %IteratorHelperPrototype%.return. That will invoke GeneratorResumeAbrupt, which will execute the generator with an abrupt completion. At that time, we must take care to close the current inner iterator.
This commit is contained in:
parent
c04476f09d
commit
57e7112a20
3 changed files with 40 additions and 11 deletions
|
@ -11,15 +11,16 @@
|
|||
|
||||
namespace JS {
|
||||
|
||||
ThrowCompletionOr<NonnullGCPtr<IteratorHelper>> IteratorHelper::create(Realm& realm, IteratorRecord underlying_iterator, Closure closure)
|
||||
ThrowCompletionOr<NonnullGCPtr<IteratorHelper>> IteratorHelper::create(Realm& realm, IteratorRecord underlying_iterator, Closure closure, Optional<AbruptClosure> abrupt_closure)
|
||||
{
|
||||
return TRY(realm.heap().allocate<IteratorHelper>(realm, realm, realm.intrinsics().iterator_helper_prototype(), move(underlying_iterator), move(closure)));
|
||||
return TRY(realm.heap().allocate<IteratorHelper>(realm, realm, realm.intrinsics().iterator_helper_prototype(), move(underlying_iterator), move(closure), move(abrupt_closure)));
|
||||
}
|
||||
|
||||
IteratorHelper::IteratorHelper(Realm& realm, Object& prototype, IteratorRecord underlying_iterator, Closure closure)
|
||||
IteratorHelper::IteratorHelper(Realm& realm, Object& prototype, IteratorRecord underlying_iterator, Closure closure, Optional<AbruptClosure> abrupt_closure)
|
||||
: GeneratorObject(realm, prototype, realm.vm().running_execution_context().copy(), "Iterator Helper"sv)
|
||||
, m_underlying_iterator(move(underlying_iterator))
|
||||
, m_closure(move(closure))
|
||||
, m_abrupt_closure(move(abrupt_closure))
|
||||
{
|
||||
}
|
||||
|
||||
|
@ -41,9 +42,16 @@ ThrowCompletionOr<Value> IteratorHelper::close_result(VM& vm, Completion complet
|
|||
return *TRY(iterator_close(vm, underlying_iterator(), move(completion)));
|
||||
}
|
||||
|
||||
ThrowCompletionOr<Value> IteratorHelper::execute(VM& vm, JS::Completion const&)
|
||||
ThrowCompletionOr<Value> IteratorHelper::execute(VM& vm, JS::Completion const& completion)
|
||||
{
|
||||
ScopeGuard guard { [&] { vm.pop_execution_context(); } };
|
||||
|
||||
if (completion.is_abrupt()) {
|
||||
if (m_abrupt_closure.has_value())
|
||||
return (*m_abrupt_closure)(vm, *this, completion);
|
||||
return close_result(vm, completion);
|
||||
}
|
||||
|
||||
auto result_value = m_closure(vm, *this);
|
||||
|
||||
if (result_value.is_throw_completion()) {
|
||||
|
|
|
@ -19,8 +19,9 @@ class IteratorHelper final : public GeneratorObject {
|
|||
|
||||
public:
|
||||
using Closure = JS::SafeFunction<ThrowCompletionOr<Value>(VM&, IteratorHelper&)>;
|
||||
using AbruptClosure = JS::SafeFunction<ThrowCompletionOr<Value>(VM&, IteratorHelper&, Completion const&)>;
|
||||
|
||||
static ThrowCompletionOr<NonnullGCPtr<IteratorHelper>> create(Realm&, IteratorRecord, Closure);
|
||||
static ThrowCompletionOr<NonnullGCPtr<IteratorHelper>> create(Realm&, IteratorRecord, Closure, Optional<AbruptClosure> = {});
|
||||
|
||||
IteratorRecord const& underlying_iterator() const { return m_underlying_iterator; }
|
||||
|
||||
|
@ -31,13 +32,14 @@ public:
|
|||
ThrowCompletionOr<Value> close_result(VM&, Completion);
|
||||
|
||||
private:
|
||||
IteratorHelper(Realm&, Object& prototype, IteratorRecord, Closure);
|
||||
IteratorHelper(Realm&, Object& prototype, IteratorRecord, Closure, Optional<AbruptClosure>);
|
||||
|
||||
virtual void visit_edges(Visitor&) override;
|
||||
virtual ThrowCompletionOr<Value> execute(VM&, JS::Completion const& completion) override;
|
||||
|
||||
IteratorRecord m_underlying_iterator; // [[UnderlyingIterator]]
|
||||
Closure m_closure;
|
||||
Optional<AbruptClosure> m_abrupt_closure;
|
||||
|
||||
size_t m_counter { 0 };
|
||||
bool m_done { false };
|
||||
|
|
|
@ -321,6 +321,24 @@ public:
|
|||
return next_outer_iterator(vm, iterated, iterator, mapper);
|
||||
}
|
||||
|
||||
// NOTE: This implements step 5.b.ix.4.d of Iterator.prototype.flatMap.
|
||||
ThrowCompletionOr<Value> on_abrupt_completion(VM& vm, IteratorHelper& iterator, Completion const& completion)
|
||||
{
|
||||
VERIFY(m_inner_iterator.has_value());
|
||||
|
||||
// d. If completion is an abrupt completion, then
|
||||
|
||||
// i. Let backupCompletion be Completion(IteratorClose(innerIterator, completion)).
|
||||
auto backup_completion = iterator_close(vm, *m_inner_iterator, completion);
|
||||
|
||||
// ii. IfAbruptCloseIterator(backupCompletion, iterated).
|
||||
if (backup_completion.is_error())
|
||||
return iterator.close_result(vm, backup_completion.release_error());
|
||||
|
||||
// iii. Return ? IteratorClose(completion, iterated).
|
||||
return iterator.close_result(vm, completion);
|
||||
}
|
||||
|
||||
private:
|
||||
FlatMapIterator() = default;
|
||||
|
||||
|
@ -397,10 +415,7 @@ private:
|
|||
return iterator.close_result(vm, inner_value.release_error());
|
||||
|
||||
// c. Let completion be Completion(Yield(innerValue)).
|
||||
// d. If completion is an abrupt completion, then
|
||||
// i. Let backupCompletion be Completion(IteratorClose(innerIterator, completion)).
|
||||
// ii. IfAbruptCloseIterator(backupCompletion, iterated).
|
||||
// iii. Return ? IteratorClose(completion, iterated).
|
||||
// NOTE: Step d is implemented via on_abrupt_completion.
|
||||
return inner_value.release_value();
|
||||
}
|
||||
}
|
||||
|
@ -434,9 +449,13 @@ JS_DEFINE_NATIVE_FUNCTION(IteratorPrototype::flat_map)
|
|||
return flat_map_iterator->next(vm, iterated, iterator, *mapper);
|
||||
};
|
||||
|
||||
IteratorHelper::AbruptClosure abrupt_closure = [flat_map_iterator](auto& vm, auto& iterator, auto const& completion) -> ThrowCompletionOr<Value> {
|
||||
return flat_map_iterator->on_abrupt_completion(vm, iterator, completion);
|
||||
};
|
||||
|
||||
// 6. Let result be CreateIteratorFromClosure(closure, "Iterator Helper", %IteratorHelperPrototype%, « [[UnderlyingIterator]] »).
|
||||
// 7. Set result.[[UnderlyingIterator]] to iterated.
|
||||
auto result = TRY(IteratorHelper::create(realm, move(iterated), move(closure)));
|
||||
auto result = TRY(IteratorHelper::create(realm, move(iterated), move(closure), move(abrupt_closure)));
|
||||
|
||||
// 8. Return result.
|
||||
return result;
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue