Skip to content

Commit c40919b

Browse files
committed
Auto merge of #106938 - GuillaumeGomez:normalize-projection-field-ty, r=oli-obk
Add missing normalization for union fields types Overshadows #106808. From the experiment #103985. In short, it allows to use projections as a type for union's fields. cc `@compiler-errors` r? `@oli-obk`
2 parents e7acd07 + 6c63b94 commit c40919b

6 files changed

+81
-1
lines changed

compiler/rustc_hir_analysis/src/check/check.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -121,7 +121,7 @@ fn check_union_fields(tcx: TyCtxt<'_>, span: Span, item_def_id: LocalDefId) -> b
121121

122122
let param_env = tcx.param_env(item_def_id);
123123
for field in &def.non_enum_variant().fields {
124-
let field_ty = field.ty(tcx, substs);
124+
let field_ty = tcx.normalize_erasing_regions(param_env, field.ty(tcx, substs));
125125

126126
if !allowed_union_field(field_ty, tcx, param_env) {
127127
let (field_span, ty_span) = match tcx.hir().get_if_local(field.did) {
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,20 @@
1+
// Test to ensure that there is no ICE when normalizing a projection
2+
// which is invalid (from <https://github.com/rust-lang/rust/pull/106938>).
3+
4+
#![crate_type = "lib"]
5+
6+
trait Identity {
7+
type Identity;
8+
}
9+
trait NotImplemented {}
10+
11+
impl<T: NotImplemented> Identity for T {
12+
type Identity = Self;
13+
}
14+
15+
type Foo = u8;
16+
17+
union Bar {
18+
a: <Foo as Identity>::Identity, //~ ERROR
19+
b: u8,
20+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,17 @@
1+
error[E0277]: the trait bound `u8: NotImplemented` is not satisfied
2+
--> $DIR/projection-as-union-type-error-2.rs:18:8
3+
|
4+
LL | a: <Foo as Identity>::Identity,
5+
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^ the trait `NotImplemented` is not implemented for `u8`
6+
|
7+
note: required for `u8` to implement `Identity`
8+
--> $DIR/projection-as-union-type-error-2.rs:11:25
9+
|
10+
LL | impl<T: NotImplemented> Identity for T {
11+
| -------------- ^^^^^^^^ ^
12+
| |
13+
| unsatisfied trait bound introduced here
14+
15+
error: aborting due to previous error
16+
17+
For more information about this error, try `rustc --explain E0277`.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,15 @@
1+
// Test to ensure that there is no ICE when normalizing a projection
2+
// which is invalid (from <https://github.com/rust-lang/rust/pull/106938>).
3+
4+
#![crate_type = "lib"]
5+
6+
pub trait Identity {
7+
type Identity;
8+
}
9+
10+
pub type Foo = u8;
11+
12+
pub union Bar {
13+
a: <Foo as Identity>::Identity, //~ ERROR
14+
b: u8,
15+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,9 @@
1+
error[E0277]: the trait bound `u8: Identity` is not satisfied
2+
--> $DIR/projection-as-union-type-error.rs:13:9
3+
|
4+
LL | a: <Foo as Identity>::Identity,
5+
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^ the trait `Identity` is not implemented for `u8`
6+
7+
error: aborting due to previous error
8+
9+
For more information about this error, try `rustc --explain E0277`.
+19
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,19 @@
1+
// Ensures that we can use projections as union field's type.
2+
// check-pass
3+
4+
#![crate_type = "lib"]
5+
6+
pub trait Identity {
7+
type Identity;
8+
}
9+
10+
impl<T> Identity for T {
11+
type Identity = Self;
12+
}
13+
14+
pub type Foo = u8;
15+
16+
pub union Bar {
17+
pub a: <Foo as Identity>::Identity,
18+
pub b: u8,
19+
}

0 commit comments

Comments
 (0)