Skip to content

Commit 3050938

Browse files
committed
Auto merge of #116081 - compiler-errors:closure-captures-sized, r=cjgillot
Check that closure/generator's interior/capture types are sized check that closure upvars and generator interiors are sized. this check is only necessary when `unsized_fn_params` or `unsized_locals` is enabled, so only check if those are active. Fixes #93622 Fixes #61335 Fixes #68543
2 parents 0237aa3 + 79d6853 commit 3050938

17 files changed

+296
-4
lines changed

compiler/rustc_hir_analysis/src/check/check.rs

+20-1
Original file line numberDiff line numberDiff line change
@@ -18,7 +18,7 @@ use rustc_infer::traits::{Obligation, TraitEngineExt as _};
1818
use rustc_lint_defs::builtin::REPR_TRANSPARENT_EXTERNAL_PRIVATE_FIELDS;
1919
use rustc_middle::hir::nested_filter;
2020
use rustc_middle::middle::stability::EvalResult;
21-
use rustc_middle::traits::DefiningAnchor;
21+
use rustc_middle::traits::{DefiningAnchor, ObligationCauseCode};
2222
use rustc_middle::ty::fold::BottomUpFolder;
2323
use rustc_middle::ty::layout::{LayoutError, MAX_SIMD_LANES};
2424
use rustc_middle::ty::util::{Discr, IntTypeExt};
@@ -1626,6 +1626,25 @@ pub(super) fn check_generator_obligations(tcx: TyCtxt<'_>, def_id: LocalDefId) {
16261626
let obligation = Obligation::new(tcx, cause.clone(), param_env, *predicate);
16271627
fulfillment_cx.register_predicate_obligation(&infcx, obligation);
16281628
}
1629+
1630+
if (tcx.features().unsized_locals || tcx.features().unsized_fn_params)
1631+
&& let Some(generator) = tcx.mir_generator_witnesses(def_id)
1632+
{
1633+
for field_ty in generator.field_tys.iter() {
1634+
fulfillment_cx.register_bound(
1635+
&infcx,
1636+
param_env,
1637+
field_ty.ty,
1638+
tcx.require_lang_item(hir::LangItem::Sized, Some(field_ty.source_info.span)),
1639+
ObligationCause::new(
1640+
field_ty.source_info.span,
1641+
def_id,
1642+
ObligationCauseCode::SizedGeneratorInterior(def_id),
1643+
),
1644+
);
1645+
}
1646+
}
1647+
16291648
let errors = fulfillment_cx.select_all_or_error(&infcx);
16301649
debug!(?errors);
16311650
if !errors.is_empty() {

compiler/rustc_hir_typeck/src/fn_ctxt/_impl.rs

+9-2
Original file line numberDiff line numberDiff line change
@@ -518,8 +518,15 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
518518
self.select_obligations_where_possible(|_| {});
519519

520520
let mut generators = self.deferred_generator_interiors.borrow_mut();
521-
for (_, body_id, interior, kind) in generators.drain(..) {
522-
crate::generator_interior::resolve_interior(self, def_id, body_id, interior, kind);
521+
for (generator_def_id, body_id, interior, kind) in generators.drain(..) {
522+
crate::generator_interior::resolve_interior(
523+
self,
524+
def_id,
525+
generator_def_id,
526+
body_id,
527+
interior,
528+
kind,
529+
);
523530
self.select_obligations_where_possible(|_| {});
524531
}
525532
}

compiler/rustc_hir_typeck/src/generator_interior/mod.rs

+13-1
Original file line numberDiff line numberDiff line change
@@ -9,12 +9,13 @@ use rustc_data_structures::fx::{FxHashSet, FxIndexSet};
99
use rustc_errors::{pluralize, DelayDm};
1010
use rustc_hir as hir;
1111
use rustc_hir::def::{CtorKind, DefKind, Res};
12-
use rustc_hir::def_id::DefId;
12+
use rustc_hir::def_id::{DefId, LocalDefId};
1313
use rustc_hir::hir_id::HirIdSet;
1414
use rustc_hir::intravisit::{self, Visitor};
1515
use rustc_hir::{Arm, Expr, ExprKind, Guard, HirId, Pat, PatKind};
1616
use rustc_infer::infer::{DefineOpaqueTypes, RegionVariableOrigin};
1717
use rustc_middle::middle::region::{self, Scope, ScopeData, YieldData};
18+
use rustc_middle::traits::ObligationCauseCode;
1819
use rustc_middle::ty::fold::FnMutDelegate;
1920
use rustc_middle::ty::{self, BoundVariableKind, RvalueScopes, Ty, TyCtxt, TypeVisitableExt};
2021
use rustc_span::symbol::sym;
@@ -188,6 +189,7 @@ impl<'a, 'tcx> InteriorVisitor<'a, 'tcx> {
188189
pub fn resolve_interior<'a, 'tcx>(
189190
fcx: &'a FnCtxt<'a, 'tcx>,
190191
def_id: DefId,
192+
generator_def_id: LocalDefId,
191193
body_id: hir::BodyId,
192194
interior: Ty<'tcx>,
193195
kind: hir::GeneratorKind,
@@ -214,6 +216,16 @@ pub fn resolve_interior<'a, 'tcx>(
214216
// The types are already kept in insertion order.
215217
let types = visitor.types;
216218

219+
if fcx.tcx.features().unsized_locals || fcx.tcx.features().unsized_fn_params {
220+
for interior_ty in &types {
221+
fcx.require_type_is_sized(
222+
interior_ty.ty,
223+
interior_ty.span,
224+
ObligationCauseCode::SizedGeneratorInterior(generator_def_id),
225+
);
226+
}
227+
}
228+
217229
// The types in the generator interior contain lifetimes local to the generator itself,
218230
// which should not be exposed outside of the generator. Therefore, we replace these
219231
// lifetimes with existentially-bound lifetimes, which reflect the exact value of the

compiler/rustc_hir_typeck/src/upvar.rs

+15
Original file line numberDiff line numberDiff line change
@@ -41,6 +41,7 @@ use rustc_hir::intravisit::{self, Visitor};
4141
use rustc_infer::infer::UpvarRegion;
4242
use rustc_middle::hir::place::{Place, PlaceBase, PlaceWithHirId, Projection, ProjectionKind};
4343
use rustc_middle::mir::FakeReadCause;
44+
use rustc_middle::traits::ObligationCauseCode;
4445
use rustc_middle::ty::{
4546
self, ClosureSizeProfileData, Ty, TyCtxt, TypeckResults, UpvarArgs, UpvarCapture,
4647
};
@@ -295,6 +296,20 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
295296
let final_upvar_tys = self.final_upvar_tys(closure_def_id);
296297
debug!(?closure_hir_id, ?args, ?final_upvar_tys);
297298

299+
if self.tcx.features().unsized_locals || self.tcx.features().unsized_fn_params {
300+
for capture in
301+
self.typeck_results.borrow().closure_min_captures_flattened(closure_def_id)
302+
{
303+
if let UpvarCapture::ByValue = capture.info.capture_kind {
304+
self.require_type_is_sized(
305+
capture.place.ty(),
306+
capture.get_path_span(self.tcx),
307+
ObligationCauseCode::SizedClosureCapture(closure_def_id),
308+
);
309+
}
310+
}
311+
}
312+
298313
// Build a tuple (U0..Un) of the final upvar types U0..Un
299314
// and unify the upvar tuple type in the closure with it:
300315
let final_tupled_upvars_type = Ty::new_tup(self.tcx, &final_upvar_tys);

compiler/rustc_middle/src/traits/mod.rs

+4
Original file line numberDiff line numberDiff line change
@@ -299,6 +299,10 @@ pub enum ObligationCauseCode<'tcx> {
299299
SizedYieldType,
300300
/// Inline asm operand type must be `Sized`.
301301
InlineAsmSized,
302+
/// Captured closure type must be `Sized`.
303+
SizedClosureCapture(LocalDefId),
304+
/// Types live across generator yields must be `Sized`.
305+
SizedGeneratorInterior(LocalDefId),
302306
/// `[expr; N]` requires `type_of(expr): Copy`.
303307
RepeatElementCopy {
304308
/// If element is a `const fn` we display a help message suggesting to move the

compiler/rustc_trait_selection/src/traits/error_reporting/suggestions.rs

+18
Original file line numberDiff line numberDiff line change
@@ -3007,6 +3007,24 @@ impl<'tcx> TypeErrCtxtExt<'tcx> for TypeErrCtxt<'_, 'tcx> {
30073007
ObligationCauseCode::InlineAsmSized => {
30083008
err.note("all inline asm arguments must have a statically known size");
30093009
}
3010+
ObligationCauseCode::SizedClosureCapture(closure_def_id) => {
3011+
err.note("all values captured by value by a closure must have a statically known size");
3012+
let hir::ExprKind::Closure(closure) = self.tcx.hir().get_by_def_id(closure_def_id).expect_expr().kind else {
3013+
bug!("expected closure in SizedClosureCapture obligation");
3014+
};
3015+
if let hir::CaptureBy::Value = closure.capture_clause
3016+
&& let Some(span) = closure.fn_arg_span
3017+
{
3018+
err.span_label(span, "this closure captures all values by move");
3019+
}
3020+
}
3021+
ObligationCauseCode::SizedGeneratorInterior(generator_def_id) => {
3022+
let what = match self.tcx.generator_kind(generator_def_id) {
3023+
None | Some(hir::GeneratorKind::Gen) => "yield",
3024+
Some(hir::GeneratorKind::Async(..)) => "await",
3025+
};
3026+
err.note(format!("all values live across `{what}` must have a statically known size"));
3027+
}
30103028
ObligationCauseCode::ConstPatternStructural => {
30113029
err.note("constants used for pattern-matching must derive `PartialEq` and `Eq`");
30123030
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,21 @@
1+
warning: the feature `unsized_locals` is incomplete and may not be safe to use and/or cause compiler crashes
2+
--> $DIR/awaiting-unsized-param.rs:5:31
3+
|
4+
LL | #![feature(unsized_fn_params, unsized_locals)]
5+
| ^^^^^^^^^^^^^^
6+
|
7+
= note: see issue #48055 <https://github.com/rust-lang/rust/issues/48055> for more information
8+
= note: `#[warn(incomplete_features)]` on by default
9+
10+
error[E0277]: the size for values of type `(dyn Future<Output = T> + Unpin + 'static)` cannot be known at compilation time
11+
--> $DIR/awaiting-unsized-param.rs:10:17
12+
|
13+
LL | async fn bug<T>(mut f: dyn Future<Output = T> + Unpin) -> T {
14+
| ^^^^^ doesn't have a size known at compile-time
15+
|
16+
= help: the trait `Sized` is not implemented for `(dyn Future<Output = T> + Unpin + 'static)`
17+
= note: all values captured by value by a closure must have a statically known size
18+
19+
error: aborting due to previous error; 1 warning emitted
20+
21+
For more information about this error, try `rustc --explain E0277`.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,21 @@
1+
warning: the feature `unsized_locals` is incomplete and may not be safe to use and/or cause compiler crashes
2+
--> $DIR/awaiting-unsized-param.rs:5:31
3+
|
4+
LL | #![feature(unsized_fn_params, unsized_locals)]
5+
| ^^^^^^^^^^^^^^
6+
|
7+
= note: see issue #48055 <https://github.com/rust-lang/rust/issues/48055> for more information
8+
= note: `#[warn(incomplete_features)]` on by default
9+
10+
error[E0277]: the size for values of type `(dyn Future<Output = T> + Unpin + 'static)` cannot be known at compilation time
11+
--> $DIR/awaiting-unsized-param.rs:10:17
12+
|
13+
LL | async fn bug<T>(mut f: dyn Future<Output = T> + Unpin) -> T {
14+
| ^^^^^ doesn't have a size known at compile-time
15+
|
16+
= help: the trait `Sized` is not implemented for `(dyn Future<Output = T> + Unpin + 'static)`
17+
= note: all values captured by value by a closure must have a statically known size
18+
19+
error: aborting due to previous error; 1 warning emitted
20+
21+
For more information about this error, try `rustc --explain E0277`.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,15 @@
1+
// edition: 2021
2+
// revisions: no_drop_tracking drop_tracking_mir
3+
// [drop_tracking_mir] compile-flags: -Zdrop-tracking-mir
4+
5+
#![feature(unsized_fn_params, unsized_locals)]
6+
//~^ WARN the feature `unsized_locals` is incomplete
7+
8+
use std::future::Future;
9+
10+
async fn bug<T>(mut f: dyn Future<Output = T> + Unpin) -> T {
11+
//~^ ERROR the size for values of type `(dyn Future<Output = T> + Unpin + 'static)` cannot be known at compilation time
12+
(&mut f).await
13+
}
14+
15+
fn main() {}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,21 @@
1+
warning: the feature `unsized_locals` is incomplete and may not be safe to use and/or cause compiler crashes
2+
--> $DIR/unsized-across-await.rs:5:12
3+
|
4+
LL | #![feature(unsized_locals)]
5+
| ^^^^^^^^^^^^^^
6+
|
7+
= note: see issue #48055 <https://github.com/rust-lang/rust/issues/48055> for more information
8+
= note: `#[warn(incomplete_features)]` on by default
9+
10+
error[E0277]: the size for values of type `dyn std::fmt::Display` cannot be known at compilation time
11+
--> $DIR/unsized-across-await.rs:11:9
12+
|
13+
LL | let _x = *x;
14+
| ^^ doesn't have a size known at compile-time
15+
|
16+
= help: the trait `Sized` is not implemented for `dyn std::fmt::Display`
17+
= note: all values live across `await` must have a statically known size
18+
19+
error: aborting due to previous error; 1 warning emitted
20+
21+
For more information about this error, try `rustc --explain E0277`.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,21 @@
1+
warning: the feature `unsized_locals` is incomplete and may not be safe to use and/or cause compiler crashes
2+
--> $DIR/unsized-across-await.rs:5:12
3+
|
4+
LL | #![feature(unsized_locals)]
5+
| ^^^^^^^^^^^^^^
6+
|
7+
= note: see issue #48055 <https://github.com/rust-lang/rust/issues/48055> for more information
8+
= note: `#[warn(incomplete_features)]` on by default
9+
10+
error[E0277]: the size for values of type `dyn std::fmt::Display` cannot be known at compilation time
11+
--> $DIR/unsized-across-await.rs:11:9
12+
|
13+
LL | let _x = *x;
14+
| ^^ doesn't have a size known at compile-time
15+
|
16+
= help: the trait `Sized` is not implemented for `dyn std::fmt::Display`
17+
= note: all values live across `await` must have a statically known size
18+
19+
error: aborting due to previous error; 1 warning emitted
20+
21+
For more information about this error, try `rustc --explain E0277`.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,18 @@
1+
// edition: 2021
2+
// revisions: no_drop_tracking drop_tracking_mir
3+
// [drop_tracking_mir] compile-flags: -Zdrop-tracking-mir
4+
5+
#![feature(unsized_locals)]
6+
//~^ WARN the feature `unsized_locals` is incomplete
7+
8+
async fn f() {}
9+
10+
async fn g(x: Box<dyn std::fmt::Display>) {
11+
let _x = *x;
12+
//~^ ERROR the size for values of type `dyn std::fmt::Display` cannot be known at compilation time
13+
f().await;
14+
}
15+
16+
fn main() {
17+
let _a = g(Box::new(5));
18+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,10 @@
1+
// compile-flags: --crate-type=lib
2+
3+
#![feature(unsized_fn_params)]
4+
5+
pub fn f(k: dyn std::fmt::Display) {
6+
let k2 = move || {
7+
k.to_string();
8+
//~^ ERROR the size for values of type `(dyn std::fmt::Display + 'static)` cannot be known at compilation time
9+
};
10+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,14 @@
1+
error[E0277]: the size for values of type `(dyn std::fmt::Display + 'static)` cannot be known at compilation time
2+
--> $DIR/capture-unsized-by-move.rs:7:9
3+
|
4+
LL | let k2 = move || {
5+
| -- this closure captures all values by move
6+
LL | k.to_string();
7+
| ^ doesn't have a size known at compile-time
8+
|
9+
= help: the trait `Sized` is not implemented for `(dyn std::fmt::Display + 'static)`
10+
= note: all values captured by value by a closure must have a statically known size
11+
12+
error: aborting due to previous error
13+
14+
For more information about this error, try `rustc --explain E0277`.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,10 @@
1+
// build-pass
2+
// compile-flags: --crate-type=lib
3+
4+
#![feature(unsized_fn_params)]
5+
6+
pub fn f(k: dyn std::fmt::Display) {
7+
let k2 = || {
8+
k.to_string();
9+
};
10+
}
+34
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,34 @@
1+
#![feature(generator_trait)]
2+
#![feature(generators)]
3+
#![feature(unsized_locals)]
4+
//~^ WARN the feature `unsized_locals` is incomplete and may not be safe to use and/or cause compiler crashes
5+
6+
use std::ops::Generator;
7+
8+
fn across() -> impl Generator {
9+
move || {
10+
let b: [u8] = *(Box::new([]) as Box<[u8]>);
11+
//~^ ERROR the size for values of type `[u8]` cannot be known at compilation time
12+
13+
yield;
14+
15+
for elem in b.iter() {}
16+
}
17+
}
18+
19+
fn capture() -> impl Generator {
20+
let b: [u8] = *(Box::new([]) as Box<[u8]>);
21+
move || {
22+
println!("{:?}", &b);
23+
//~^ ERROR the size for values of type `[u8]` cannot be known at compilation time
24+
25+
yield;
26+
27+
for elem in b.iter() {}
28+
}
29+
}
30+
31+
fn main() {
32+
across();
33+
capture();
34+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,32 @@
1+
warning: the feature `unsized_locals` is incomplete and may not be safe to use and/or cause compiler crashes
2+
--> $DIR/unsized-across-yield.rs:3:12
3+
|
4+
LL | #![feature(unsized_locals)]
5+
| ^^^^^^^^^^^^^^
6+
|
7+
= note: see issue #48055 <https://github.com/rust-lang/rust/issues/48055> for more information
8+
= note: `#[warn(incomplete_features)]` on by default
9+
10+
error[E0277]: the size for values of type `[u8]` cannot be known at compilation time
11+
--> $DIR/unsized-across-yield.rs:10:13
12+
|
13+
LL | let b: [u8] = *(Box::new([]) as Box<[u8]>);
14+
| ^ doesn't have a size known at compile-time
15+
|
16+
= help: the trait `Sized` is not implemented for `[u8]`
17+
= note: all values live across `yield` must have a statically known size
18+
19+
error[E0277]: the size for values of type `[u8]` cannot be known at compilation time
20+
--> $DIR/unsized-across-yield.rs:22:27
21+
|
22+
LL | move || {
23+
| -- this closure captures all values by move
24+
LL | println!("{:?}", &b);
25+
| ^ doesn't have a size known at compile-time
26+
|
27+
= help: the trait `Sized` is not implemented for `[u8]`
28+
= note: all values captured by value by a closure must have a statically known size
29+
30+
error: aborting due to 2 previous errors; 1 warning emitted
31+
32+
For more information about this error, try `rustc --explain E0277`.

0 commit comments

Comments
 (0)