Skip to content

Commit a0dbaf2

Browse files
Unconditionally check sizedness of body in typeck to taint typeck results
1 parent 5f081c2 commit a0dbaf2

File tree

50 files changed

+773
-153
lines changed

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

50 files changed

+773
-153
lines changed

compiler/rustc_hir_typeck/src/check.rs

+10-13
Original file line numberDiff line numberDiff line change
@@ -113,19 +113,16 @@ pub(super) fn check_fn<'a, 'tcx>(
113113

114114
fcx.typeck_results.borrow_mut().liberated_fn_sigs_mut().insert(fn_id, fn_sig);
115115

116-
// We checked the root's ret ty during wfcheck, but not the child.
117-
if fcx.tcx.is_typeck_child(fn_def_id.to_def_id()) {
118-
let return_or_body_span = match decl.output {
119-
hir::FnRetTy::DefaultReturn(_) => body.value.span,
120-
hir::FnRetTy::Return(ty) => ty.span,
121-
};
122-
123-
fcx.require_type_is_sized(
124-
declared_ret_ty,
125-
return_or_body_span,
126-
ObligationCauseCode::SizedReturnType,
127-
);
128-
}
116+
let return_or_body_span = match decl.output {
117+
hir::FnRetTy::DefaultReturn(_) => body.value.span,
118+
hir::FnRetTy::Return(ty) => ty.span,
119+
};
120+
121+
fcx.require_type_is_sized(
122+
declared_ret_ty,
123+
return_or_body_span,
124+
ObligationCauseCode::SizedReturnType,
125+
);
129126

130127
fcx.is_whole_body.set(true);
131128
fcx.check_return_or_body_tail(body.value, false);

compiler/rustc_hir_typeck/src/lib.rs

+5
Original file line numberDiff line numberDiff line change
@@ -185,6 +185,11 @@ fn typeck_with_inspect<'tcx>(
185185

186186
let wf_code = ObligationCauseCode::WellFormed(Some(WellFormedLoc::Ty(def_id)));
187187
fcx.register_wf_obligation(expected_type.into(), body.value.span, wf_code);
188+
fcx.require_type_is_sized(
189+
expected_type,
190+
node.ty().map_or(body.value.span, |ty| ty.span),
191+
ObligationCauseCode::SizedConstOrStatic,
192+
);
188193

189194
// Gather locals in statics (because of block expressions).
190195
GatherLocalsVisitor::new(&fcx).visit_body(body);

tests/ui/associated-consts/issue-58022.rs

+1
Original file line numberDiff line numberDiff line change
@@ -12,6 +12,7 @@ impl Bar<[u8]> {
1212

1313
fn new(slice: &[u8; Self::SIZE]) -> Self {
1414
//~^ ERROR: the size for values of type `[u8]` cannot be known at compilation time
15+
//~| ERROR: the size for values of type `[u8]` cannot be known at compilation time
1516
Foo(Box::new(*slice))
1617
//~^ ERROR: expected function, tuple struct or tuple variant, found trait `Foo`
1718
}

tests/ui/associated-consts/issue-58022.stderr

+17-2
Original file line numberDiff line numberDiff line change
@@ -21,13 +21,28 @@ LL |
2121
LL | fn new(slice: &[u8; Foo::SIZE]) -> Self;
2222
| ^^^^^^^^^ cannot refer to the associated constant of trait
2323

24+
error[E0277]: the size for values of type `[u8]` cannot be known at compilation time
25+
--> $DIR/issue-58022.rs:13:41
26+
|
27+
LL | fn new(slice: &[u8; Self::SIZE]) -> Self {
28+
| ^^^^ doesn't have a size known at compile-time
29+
|
30+
= help: within `Bar<[u8]>`, the trait `Sized` is not implemented for `[u8]`
31+
note: required because it appears within the type `Bar<[u8]>`
32+
--> $DIR/issue-58022.rs:8:12
33+
|
34+
LL | pub struct Bar<T: ?Sized>(T);
35+
| ^^^
36+
= note: the return type of a function must have a statically known size
37+
= note: duplicate diagnostic emitted due to `-Z deduplicate-diagnostics=no`
38+
2439
error[E0423]: expected function, tuple struct or tuple variant, found trait `Foo`
25-
--> $DIR/issue-58022.rs:15:9
40+
--> $DIR/issue-58022.rs:16:9
2641
|
2742
LL | Foo(Box::new(*slice))
2843
| ^^^ not a function, tuple struct or tuple variant
2944

30-
error: aborting due to 3 previous errors
45+
error: aborting due to 4 previous errors
3146

3247
Some errors have detailed explanations: E0277, E0423, E0790.
3348
For more information about an error, try `rustc --explain E0277`.

tests/ui/consts/const-slice-array-deref.rs

+1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,6 @@
11
const ONE: [u16] = [1];
22
//~^ ERROR the size for values of type `[u16]` cannot be known at compilation time
3+
//~| ERROR the size for values of type `[u16]` cannot be known at compilation time
34
//~| ERROR mismatched types
45

56
const TWO: &'static u16 = &ONE[0];

tests/ui/consts/const-slice-array-deref.stderr

+12-2
Original file line numberDiff line numberDiff line change
@@ -13,13 +13,23 @@ error[E0308]: mismatched types
1313
LL | const ONE: [u16] = [1];
1414
| ^^^ expected `[u16]`, found `[u16; 1]`
1515

16+
error[E0277]: the size for values of type `[u16]` cannot be known at compilation time
17+
--> $DIR/const-slice-array-deref.rs:1:12
18+
|
19+
LL | const ONE: [u16] = [1];
20+
| ^^^^^ doesn't have a size known at compile-time
21+
|
22+
= help: the trait `Sized` is not implemented for `[u16]`
23+
= note: statics and constants must have a statically known size
24+
= note: duplicate diagnostic emitted due to `-Z deduplicate-diagnostics=no`
25+
1626
error[E0161]: cannot move a value of type `[u16]`
17-
--> $DIR/const-slice-array-deref.rs:5:28
27+
--> $DIR/const-slice-array-deref.rs:6:28
1828
|
1929
LL | const TWO: &'static u16 = &ONE[0];
2030
| ^^^ the size of `[u16]` cannot be statically determined
2131

22-
error: aborting due to 3 previous errors
32+
error: aborting due to 4 previous errors
2333

2434
Some errors have detailed explanations: E0161, E0277, E0308.
2535
For more information about an error, try `rustc --explain E0161`.

tests/ui/consts/const-unsized.rs

+4-4
Original file line numberDiff line numberDiff line change
@@ -2,19 +2,19 @@ use std::fmt::Debug;
22

33
const CONST_0: dyn Debug + Sync = *(&0 as &(dyn Debug + Sync));
44
//~^ ERROR the size for values of type
5-
//~| ERROR cannot move out of a shared reference
5+
//~| ERROR the size for values of type
66

77
const CONST_FOO: str = *"foo";
88
//~^ ERROR the size for values of type
9-
//~| ERROR cannot move out of a shared reference
9+
//~| ERROR the size for values of type
1010

1111
static STATIC_1: dyn Debug + Sync = *(&1 as &(dyn Debug + Sync));
1212
//~^ ERROR the size for values of type
13-
//~| ERROR cannot move out of a shared reference
13+
//~| ERROR the size for values of type
1414

1515
static STATIC_BAR: str = *"bar";
1616
//~^ ERROR the size for values of type
17-
//~| ERROR cannot move out of a shared reference
17+
//~| ERROR the size for values of type
1818

1919
fn main() {
2020
println!("{:?} {:?} {:?} {:?}", &CONST_0, &CONST_FOO, &STATIC_1, &STATIC_BAR);

tests/ui/consts/const-unsized.stderr

+38-22
Original file line numberDiff line numberDiff line change
@@ -7,6 +7,16 @@ LL | const CONST_0: dyn Debug + Sync = *(&0 as &(dyn Debug + Sync));
77
= help: the trait `Sized` is not implemented for `(dyn Debug + Sync + 'static)`
88
= note: statics and constants must have a statically known size
99

10+
error[E0277]: the size for values of type `(dyn Debug + Sync + 'static)` cannot be known at compilation time
11+
--> $DIR/const-unsized.rs:3:16
12+
|
13+
LL | const CONST_0: dyn Debug + Sync = *(&0 as &(dyn Debug + Sync));
14+
| ^^^^^^^^^^^^^^^^ doesn't have a size known at compile-time
15+
|
16+
= help: the trait `Sized` is not implemented for `(dyn Debug + Sync + 'static)`
17+
= note: statics and constants must have a statically known size
18+
= note: duplicate diagnostic emitted due to `-Z deduplicate-diagnostics=no`
19+
1020
error[E0277]: the size for values of type `str` cannot be known at compilation time
1121
--> $DIR/const-unsized.rs:7:18
1222
|
@@ -16,6 +26,25 @@ LL | const CONST_FOO: str = *"foo";
1626
= help: the trait `Sized` is not implemented for `str`
1727
= note: statics and constants must have a statically known size
1828

29+
error[E0277]: the size for values of type `str` cannot be known at compilation time
30+
--> $DIR/const-unsized.rs:7:18
31+
|
32+
LL | const CONST_FOO: str = *"foo";
33+
| ^^^ doesn't have a size known at compile-time
34+
|
35+
= help: the trait `Sized` is not implemented for `str`
36+
= note: statics and constants must have a statically known size
37+
= note: duplicate diagnostic emitted due to `-Z deduplicate-diagnostics=no`
38+
39+
error[E0277]: the size for values of type `(dyn Debug + Sync + 'static)` cannot be known at compilation time
40+
--> $DIR/const-unsized.rs:11:18
41+
|
42+
LL | static STATIC_1: dyn Debug + Sync = *(&1 as &(dyn Debug + Sync));
43+
| ^^^^^^^^^^^^^^^^ doesn't have a size known at compile-time
44+
|
45+
= help: the trait `Sized` is not implemented for `(dyn Debug + Sync + 'static)`
46+
= note: statics and constants must have a statically known size
47+
1948
error[E0277]: the size for values of type `(dyn Debug + Sync + 'static)` cannot be known at compilation time
2049
--> $DIR/const-unsized.rs:11:18
2150
|
@@ -24,6 +53,7 @@ LL | static STATIC_1: dyn Debug + Sync = *(&1 as &(dyn Debug + Sync));
2453
|
2554
= help: the trait `Sized` is not implemented for `(dyn Debug + Sync + 'static)`
2655
= note: statics and constants must have a statically known size
56+
= note: duplicate diagnostic emitted due to `-Z deduplicate-diagnostics=no`
2757

2858
error[E0277]: the size for values of type `str` cannot be known at compilation time
2959
--> $DIR/const-unsized.rs:15:20
@@ -34,29 +64,15 @@ LL | static STATIC_BAR: str = *"bar";
3464
= help: the trait `Sized` is not implemented for `str`
3565
= note: statics and constants must have a statically known size
3666

37-
error[E0507]: cannot move out of a shared reference
38-
--> $DIR/const-unsized.rs:3:35
39-
|
40-
LL | const CONST_0: dyn Debug + Sync = *(&0 as &(dyn Debug + Sync));
41-
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^ move occurs because value has type `dyn Debug + Sync`, which does not implement the `Copy` trait
42-
43-
error[E0507]: cannot move out of a shared reference
44-
--> $DIR/const-unsized.rs:7:24
45-
|
46-
LL | const CONST_FOO: str = *"foo";
47-
| ^^^^^^ move occurs because value has type `str`, which does not implement the `Copy` trait
48-
49-
error[E0507]: cannot move out of a shared reference
50-
--> $DIR/const-unsized.rs:11:37
51-
|
52-
LL | static STATIC_1: dyn Debug + Sync = *(&1 as &(dyn Debug + Sync));
53-
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^ move occurs because value has type `dyn Debug + Sync`, which does not implement the `Copy` trait
54-
55-
error[E0507]: cannot move out of a shared reference
56-
--> $DIR/const-unsized.rs:15:26
67+
error[E0277]: the size for values of type `str` cannot be known at compilation time
68+
--> $DIR/const-unsized.rs:15:20
5769
|
5870
LL | static STATIC_BAR: str = *"bar";
59-
| ^^^^^^ move occurs because value has type `str`, which does not implement the `Copy` trait
71+
| ^^^ doesn't have a size known at compile-time
72+
|
73+
= help: the trait `Sized` is not implemented for `str`
74+
= note: statics and constants must have a statically known size
75+
= note: duplicate diagnostic emitted due to `-Z deduplicate-diagnostics=no`
6076

6177
error[E0161]: cannot move a value of type `str`
6278
--> $DIR/const-unsized.rs:20:48
@@ -72,5 +88,5 @@ LL | println!("{:?} {:?} {:?} {:?}", &CONST_0, &CONST_FOO, &STATIC_1, &STATI
7288

7389
error: aborting due to 10 previous errors
7490

75-
Some errors have detailed explanations: E0161, E0277, E0507.
91+
Some errors have detailed explanations: E0161, E0277.
7692
For more information about an error, try `rustc --explain E0161`.

tests/ui/consts/const_refs_to_static-ice-121413.rs

+2
Original file line numberDiff line numberDiff line change
@@ -9,9 +9,11 @@ const REF_INTERIOR_MUT: &usize = {
99
//~^ ERROR failed to resolve: use of undeclared type `AtomicUsize`
1010
//~| WARN trait objects without an explicit `dyn` are deprecated
1111
//~| ERROR the size for values of type `(dyn Sync + 'static)` cannot be known at compilation time
12+
//~| ERROR the size for values of type `(dyn Sync + 'static)` cannot be known at compilation time
1213
//~| WARN this is accepted in the current edition (Rust 2015) but is a hard error in Rust 2021!
1314
//~| HELP if this is a dyn-compatible trait, use `dyn`
1415
//~| HELP the trait `Sized` is not implemented for `(dyn Sync + 'static)`
16+
//~| HELP the trait `Sized` is not implemented for `(dyn Sync + 'static)`
1517
unsafe { &*(&FOO as *const _ as *const usize) }
1618
};
1719
pub fn main() {}

tests/ui/consts/const_refs_to_static-ice-121413.stderr

+11-1
Original file line numberDiff line numberDiff line change
@@ -32,7 +32,17 @@ LL | static FOO: Sync = AtomicUsize::new(0);
3232
= help: the trait `Sized` is not implemented for `(dyn Sync + 'static)`
3333
= note: statics and constants must have a statically known size
3434

35-
error: aborting due to 2 previous errors; 1 warning emitted
35+
error[E0277]: the size for values of type `(dyn Sync + 'static)` cannot be known at compilation time
36+
--> $DIR/const_refs_to_static-ice-121413.rs:8:17
37+
|
38+
LL | static FOO: Sync = AtomicUsize::new(0);
39+
| ^^^^ doesn't have a size known at compile-time
40+
|
41+
= help: the trait `Sized` is not implemented for `(dyn Sync + 'static)`
42+
= note: statics and constants must have a statically known size
43+
= note: duplicate diagnostic emitted due to `-Z deduplicate-diagnostics=no`
44+
45+
error: aborting due to 3 previous errors; 1 warning emitted
3646

3747
Some errors have detailed explanations: E0277, E0433.
3848
For more information about an error, try `rustc --explain E0277`.

tests/ui/consts/unsized.rs

+9
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,9 @@
1+
static S: str = todo!();
2+
//~^ ERROR the size for values of type `str` cannot be known at compilation time
3+
//~| ERROR the size for values of type `str` cannot be known at compilation time
4+
5+
const A: str = todo!();
6+
//~^ ERROR the size for values of type `str` cannot be known at compilation time
7+
//~| ERROR the size for values of type `str` cannot be known at compilation time
8+
9+
fn main() {}

tests/ui/consts/unsized.stderr

+41
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,41 @@
1+
error[E0277]: the size for values of type `str` cannot be known at compilation time
2+
--> $DIR/unsized.rs:1:11
3+
|
4+
LL | static S: str = todo!();
5+
| ^^^ doesn't have a size known at compile-time
6+
|
7+
= help: the trait `Sized` is not implemented for `str`
8+
= note: statics and constants must have a statically known size
9+
10+
error[E0277]: the size for values of type `str` cannot be known at compilation time
11+
--> $DIR/unsized.rs:1:11
12+
|
13+
LL | static S: str = todo!();
14+
| ^^^ doesn't have a size known at compile-time
15+
|
16+
= help: the trait `Sized` is not implemented for `str`
17+
= note: statics and constants must have a statically known size
18+
= note: duplicate diagnostic emitted due to `-Z deduplicate-diagnostics=no`
19+
20+
error[E0277]: the size for values of type `str` cannot be known at compilation time
21+
--> $DIR/unsized.rs:5:10
22+
|
23+
LL | const A: str = todo!();
24+
| ^^^ doesn't have a size known at compile-time
25+
|
26+
= help: the trait `Sized` is not implemented for `str`
27+
= note: statics and constants must have a statically known size
28+
29+
error[E0277]: the size for values of type `str` cannot be known at compilation time
30+
--> $DIR/unsized.rs:5:10
31+
|
32+
LL | const A: str = todo!();
33+
| ^^^ doesn't have a size known at compile-time
34+
|
35+
= help: the trait `Sized` is not implemented for `str`
36+
= note: statics and constants must have a statically known size
37+
= note: duplicate diagnostic emitted due to `-Z deduplicate-diagnostics=no`
38+
39+
error: aborting due to 4 previous errors
40+
41+
For more information about this error, try `rustc --explain E0277`.

tests/ui/error-codes/E0746.rs

+5-2
Original file line numberDiff line numberDiff line change
@@ -6,9 +6,12 @@ impl Trait for Struct {}
66
impl Trait for u32 {}
77

88
fn foo() -> dyn Trait { Struct }
9-
//~^ ERROR E0746
9+
//~^ ERROR return type cannot be a trait object without pointer indirection
10+
//~| ERROR return type cannot be a trait object without pointer indirection
1011

11-
fn bar() -> dyn Trait { //~ ERROR E0746
12+
fn bar() -> dyn Trait {
13+
//~^ ERROR return type cannot be a trait object without pointer indirection
14+
//~| ERROR return type cannot be a trait object without pointer indirection
1215
if true {
1316
return 0;
1417
}

tests/ui/error-codes/E0746.stderr

+44-2
Original file line numberDiff line numberDiff line change
@@ -15,7 +15,7 @@ LL | fn foo() -> Box<dyn Trait> { Box::new(Struct) }
1515
| ++++ + +++++++++ +
1616

1717
error[E0746]: return type cannot be a trait object without pointer indirection
18-
--> $DIR/E0746.rs:11:13
18+
--> $DIR/E0746.rs:12:13
1919
|
2020
LL | fn bar() -> dyn Trait {
2121
| ^^^^^^^^^ doesn't have a size known at compile-time
@@ -28,12 +28,54 @@ LL + fn bar() -> impl Trait {
2828
help: alternatively, box the return type, and wrap all of the returned values in `Box::new`
2929
|
3030
LL ~ fn bar() -> Box<dyn Trait> {
31+
LL |
32+
LL |
3133
LL | if true {
3234
LL ~ return Box::new(0);
3335
LL | }
3436
LL ~ Box::new(42)
3537
|
3638

37-
error: aborting due to 2 previous errors
39+
error[E0746]: return type cannot be a trait object without pointer indirection
40+
--> $DIR/E0746.rs:8:13
41+
|
42+
LL | fn foo() -> dyn Trait { Struct }
43+
| ^^^^^^^^^ doesn't have a size known at compile-time
44+
|
45+
= note: duplicate diagnostic emitted due to `-Z deduplicate-diagnostics=no`
46+
help: consider returning an `impl Trait` instead of a `dyn Trait`
47+
|
48+
LL - fn foo() -> dyn Trait { Struct }
49+
LL + fn foo() -> impl Trait { Struct }
50+
|
51+
help: alternatively, box the return type, and wrap all of the returned values in `Box::new`
52+
|
53+
LL | fn foo() -> Box<dyn Trait> { Box::new(Struct) }
54+
| ++++ + +++++++++ +
55+
56+
error[E0746]: return type cannot be a trait object without pointer indirection
57+
--> $DIR/E0746.rs:12:13
58+
|
59+
LL | fn bar() -> dyn Trait {
60+
| ^^^^^^^^^ doesn't have a size known at compile-time
61+
|
62+
= note: duplicate diagnostic emitted due to `-Z deduplicate-diagnostics=no`
63+
help: consider returning an `impl Trait` instead of a `dyn Trait`
64+
|
65+
LL - fn bar() -> dyn Trait {
66+
LL + fn bar() -> impl Trait {
67+
|
68+
help: alternatively, box the return type, and wrap all of the returned values in `Box::new`
69+
|
70+
LL ~ fn bar() -> Box<dyn Trait> {
71+
LL |
72+
LL |
73+
LL | if true {
74+
LL ~ return Box::new(0);
75+
LL | }
76+
LL ~ Box::new(42)
77+
|
78+
79+
error: aborting due to 4 previous errors
3880

3981
For more information about this error, try `rustc --explain E0746`.

0 commit comments

Comments
 (0)