Skip to content

Commit 31fbc9f

Browse files
committed
Fix async generator in case of exception in implicit await in the 'return' statement
Ref: bellard/quickjs@57105c7
1 parent d2633dd commit 31fbc9f

File tree

2 files changed

+19
-10
lines changed

2 files changed

+19
-10
lines changed

quickjs.c

+19-8
Original file line numberDiff line numberDiff line change
@@ -17709,7 +17709,6 @@ static void js_async_generator_resume_next(JSContext *ctx,
1770917709
} else if (next->completion_type == GEN_MAGIC_RETURN) {
1771017710
s->state = JS_ASYNC_GENERATOR_STATE_AWAITING_RETURN;
1771117711
js_async_generator_completed_return(ctx, s, next->result);
17712-
goto done;
1771317712
} else {
1771417713
js_async_generator_reject(ctx, s, next->result);
1771517714
}
@@ -17740,7 +17739,7 @@ static void js_async_generator_resume_next(JSContext *ctx,
1774017739
js_async_generator_reject(ctx, s, value);
1774117740
JS_FreeValue(ctx, value);
1774217741
} else if (JS_VALUE_GET_TAG(func_ret) == JS_TAG_INT) {
17743-
int func_ret_code;
17742+
int func_ret_code, ret;
1774417743
value = s->func_state.frame.cur_sp[-1];
1774517744
s->func_state.frame.cur_sp[-1] = JS_UNDEFINED;
1774617745
func_ret_code = JS_VALUE_GET_INT(func_ret);
@@ -17755,8 +17754,13 @@ static void js_async_generator_resume_next(JSContext *ctx,
1775517754
JS_FreeValue(ctx, value);
1775617755
break;
1775717756
case FUNC_RET_AWAIT:
17758-
js_async_generator_await(ctx, s, value);
17757+
ret = js_async_generator_await(ctx, s, value);
1775917758
JS_FreeValue(ctx, value);
17759+
if (ret < 0) {
17760+
/* exception: throw it */
17761+
s->func_state.throw_flag = TRUE;
17762+
goto resume_exec;
17763+
}
1776017764
goto done;
1776117765
default:
1776217766
abort();
@@ -23850,6 +23854,18 @@ static void emit_return(JSParseState *s, BOOL hasval)
2385023854
BlockEnv *top;
2385123855
int drop_count;
2385223856

23857+
if (s->cur_func->func_kind != JS_FUNC_NORMAL) {
23858+
if (!hasval) {
23859+
/* no value: direct return in case of async generator */
23860+
emit_op(s, OP_undefined);
23861+
hasval = TRUE;
23862+
} else if (s->cur_func->func_kind == JS_FUNC_ASYNC_GENERATOR) {
23863+
/* the await must be done before handling the "finally" in
23864+
case it raises an exception */
23865+
emit_op(s, OP_await);
23866+
}
23867+
}
23868+
2385323869
drop_count = 0;
2385423870
top = s->cur_func->top_break;
2385523871
while (top != NULL) {
@@ -23929,11 +23945,6 @@ static void emit_return(JSParseState *s, BOOL hasval)
2392923945
emit_label(s, label_return);
2393023946
emit_op(s, OP_return);
2393123947
} else if (s->cur_func->func_kind != JS_FUNC_NORMAL) {
23932-
if (!hasval) {
23933-
emit_op(s, OP_undefined);
23934-
} else if (s->cur_func->func_kind == JS_FUNC_ASYNC_GENERATOR) {
23935-
emit_op(s, OP_await);
23936-
}
2393723948
emit_op(s, OP_return_async);
2393823949
} else {
2393923950
emit_op(s, hasval ? OP_return : OP_return_undef);

test262_errors.txt

-2
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,4 @@
11
test262/test/annexB/language/eval-code/direct/script-decl-lex-collision-in-sloppy-mode.js:13: Test262Error: Expected a SyntaxError to be thrown but no exception was thrown at all
2-
test262/test/built-ins/AsyncGeneratorPrototype/return/return-suspendedYield-broken-promise-try-catch.js:39: TypeError: $DONE() not called
3-
test262/test/built-ins/AsyncGeneratorPrototype/return/return-suspendedYield-broken-promise-try-catch.js:39: strict mode: TypeError: $DONE() not called
42
test262/test/built-ins/RegExp/lookahead-quantifier-match-groups.js:27: Test262Error: Expected [a, abc] and [a, undefined] to have the same contents. ? quantifier
53
test262/test/built-ins/RegExp/lookahead-quantifier-match-groups.js:27: strict mode: Test262Error: Expected [a, abc] and [a, undefined] to have the same contents. ? quantifier
64
test262/test/built-ins/TypedArrayConstructors/internals/DefineOwnProperty/BigInt/detached-buffer.js:46: Test262Error: (Testing with BigInt64Array.)

0 commit comments

Comments
 (0)