Skip to content

Commit 0252580

Browse files
committed
Auto merge of #118420 - compiler-errors:async-gen, r=eholk
Introduce support for `async gen` blocks I'm delighted to demonstrate that `async gen` block are not very difficult to support. They're simply coroutines that yield `Poll<Option<T>>` and return `()`. **This PR is WIP and in draft mode for now** -- I'm mostly putting it up to show folks that it's possible. This PR needs a lang-team experiment associated with it or possible an RFC, since I don't think it falls under the jurisdiction of the `gen` RFC that was recently authored by oli (rust-lang/rfcs#3513, #117078). ### Technical note on the pre-generator-transform yield type: The reason that the underlying coroutines yield `Poll<Option<T>>` and not `Poll<T>` (which would make more sense, IMO, for the pre-transformed coroutine), is because the `TransformVisitor` that is used to turn coroutines into built-in state machine functions would have to destructure and reconstruct the latter into the former, which requires at least inserting a new basic block (for a `switchInt` terminator, to match on the `Poll` discriminant). This does mean that the desugaring (at the `rustc_ast_lowering` level) of `async gen` blocks is a bit more involved. However, since we already need to intercept both `.await` and `yield` operators, I don't consider it much of a technical burden. r? `@ghost`
2 parents f39d18b + 62f7337 commit 0252580

File tree

2 files changed

+7
-6
lines changed

2 files changed

+7
-6
lines changed

clippy_lints/src/doc/needless_doctest_main.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -69,7 +69,7 @@ pub fn check(
6969
if !ignore {
7070
get_test_spans(&item, &mut test_attr_spans);
7171
}
72-
let is_async = matches!(sig.header.coro_kind, Some(CoroutineKind::Async { .. }));
72+
let is_async = matches!(sig.header.coroutine_kind, Some(CoroutineKind::Async { .. }));
7373
let returns_nothing = match &sig.decl.output {
7474
FnRetTy::Default(..) => true,
7575
FnRetTy::Ty(ty) if ty.kind.is_unit() => true,

clippy_utils/src/ast_utils.rs

+6-5
Original file line numberDiff line numberDiff line change
@@ -188,7 +188,7 @@ pub fn eq_expr(l: &Expr, r: &Expr) -> bool {
188188
Closure(box ast::Closure {
189189
binder: lb,
190190
capture_clause: lc,
191-
coro_kind: la,
191+
coroutine_kind: la,
192192
movability: lm,
193193
fn_decl: lf,
194194
body: le,
@@ -197,7 +197,7 @@ pub fn eq_expr(l: &Expr, r: &Expr) -> bool {
197197
Closure(box ast::Closure {
198198
binder: rb,
199199
capture_clause: rc,
200-
coro_kind: ra,
200+
coroutine_kind: ra,
201201
movability: rm,
202202
fn_decl: rf,
203203
body: re,
@@ -563,18 +563,19 @@ pub fn eq_fn_sig(l: &FnSig, r: &FnSig) -> bool {
563563
eq_fn_decl(&l.decl, &r.decl) && eq_fn_header(&l.header, &r.header)
564564
}
565565

566-
fn eq_opt_coro_kind(l: Option<CoroutineKind>, r: Option<CoroutineKind>) -> bool {
566+
fn eq_opt_coroutine_kind(l: Option<CoroutineKind>, r: Option<CoroutineKind>) -> bool {
567567
match (l, r) {
568568
(Some(CoroutineKind::Async { .. }), Some(CoroutineKind::Async { .. }))
569-
| (Some(CoroutineKind::Gen { .. }), Some(CoroutineKind::Gen { .. })) => true,
569+
| (Some(CoroutineKind::Gen { .. }), Some(CoroutineKind::Gen { .. }))
570+
| (Some(CoroutineKind::AsyncGen { .. }), Some(CoroutineKind::AsyncGen { .. })) => true,
570571
(None, None) => true,
571572
_ => false,
572573
}
573574
}
574575

575576
pub fn eq_fn_header(l: &FnHeader, r: &FnHeader) -> bool {
576577
matches!(l.unsafety, Unsafe::No) == matches!(r.unsafety, Unsafe::No)
577-
&& eq_opt_coro_kind(l.coro_kind, r.coro_kind)
578+
&& eq_opt_coroutine_kind(l.coroutine_kind, r.coroutine_kind)
578579
&& matches!(l.constness, Const::No) == matches!(r.constness, Const::No)
579580
&& eq_ext(&l.ext, &r.ext)
580581
}

0 commit comments

Comments
 (0)