Skip to content

Commit 863cdf1

Browse files
Don't build by-move body when async closure is tainted
1 parent d9a2cc4 commit 863cdf1

File tree

3 files changed

+36
-2
lines changed

3 files changed

+36
-2
lines changed

compiler/rustc_mir_transform/src/coroutine/by_move_body.rs

+9-2
Original file line numberDiff line numberDiff line change
@@ -88,6 +88,11 @@ pub fn coroutine_by_move_body_def_id<'tcx>(
8888
) -> DefId {
8989
let body = tcx.mir_built(coroutine_def_id).borrow();
9090

91+
// If the typeck results are tainted, no need to make a by-ref body.
92+
if tcx.typeck(coroutine_def_id).tainted_by_errors.is_some() {
93+
return coroutine_def_id.to_def_id();
94+
}
95+
9196
let Some(hir::CoroutineKind::Desugared(_, hir::CoroutineSource::Closure)) =
9297
tcx.coroutine_kind(coroutine_def_id)
9398
else {
@@ -98,7 +103,9 @@ pub fn coroutine_by_move_body_def_id<'tcx>(
98103
// the MIR body will be constructed well.
99104
let coroutine_ty = body.local_decls[ty::CAPTURE_STRUCT_LOCAL].ty;
100105

101-
let ty::Coroutine(_, args) = *coroutine_ty.kind() else { bug!("{body:#?}") };
106+
let ty::Coroutine(_, args) = *coroutine_ty.kind() else {
107+
bug!("tried to create by-move body of non-coroutine receiver");
108+
};
102109
let args = args.as_coroutine();
103110

104111
let coroutine_kind = args.kind_ty().to_opt_closure_kind().unwrap();
@@ -107,7 +114,7 @@ pub fn coroutine_by_move_body_def_id<'tcx>(
107114
let ty::CoroutineClosure(_, parent_args) =
108115
*tcx.type_of(parent_def_id).instantiate_identity().kind()
109116
else {
110-
bug!();
117+
bug!("coroutine's parent was not a coroutine-closure");
111118
};
112119
if parent_args.references_error() {
113120
return coroutine_def_id.to_def_id();
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,18 @@
1+
//@ edition: 2021
2+
3+
#![feature(async_closure)]
4+
5+
// Ensure that building a by-ref async closure body doesn't ICE when the parent
6+
// body is tainted.
7+
8+
fn main() {
9+
missing;
10+
//~^ ERROR cannot find value `missing` in this scope
11+
12+
// We don't do numerical inference fallback when the body is tainted.
13+
// This leads to writeback folding the type of the coroutine-closure
14+
// into an error type, since its signature contains that numerical
15+
// infer var.
16+
let c = async |_| {};
17+
c(1);
18+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,9 @@
1+
error[E0425]: cannot find value `missing` in this scope
2+
--> $DIR/tainted-body-2.rs:9:5
3+
|
4+
LL | missing;
5+
| ^^^^^^^ not found in this scope
6+
7+
error: aborting due to 1 previous error
8+
9+
For more information about this error, try `rustc --explain E0425`.

0 commit comments

Comments
 (0)