Skip to content

Commit d284cd8

Browse files
authored
Rollup merge of rust-lang#92248 - compiler-errors:normalize-type-for-pointee, r=jackh726
Normalize struct tail type when checking Pointee trait Let's go ahead and implement the FIXMEs by properly normalizing the struct-tail type when satisfying a Pointee obligation. This should fix the ICE when we try to calculate a layout depending on `<Ty as Pointee>::Metadata` later. Fixes rust-lang#92128 Fixes rust-lang#92577 Additionally, mark the obligation as ambiguous if there are any infer types in that struct-tail type. This has the effect of causing `<_ as Pointee>::Metadata` to be properly replaced with an infer variable ([here](https://github.com/rust-lang/rust/blob/master/compiler/rustc_trait_selection/src/traits/project.rs#L813)) and registered as an obligation... this turns out to be very important in unifying function parameters with formals that are assoc types. Fixes rust-lang#91446
2 parents d9b023c + 5a1c460 commit d284cd8

File tree

4 files changed

+61
-10
lines changed

4 files changed

+61
-10
lines changed

compiler/rustc_middle/src/ty/sty.rs

+6-3
Original file line numberDiff line numberDiff line change
@@ -2143,9 +2143,12 @@ impl<'tcx> TyS<'tcx> {
21432143
}
21442144

21452145
/// Returns the type of metadata for (potentially fat) pointers to this type.
2146-
pub fn ptr_metadata_ty(&'tcx self, tcx: TyCtxt<'tcx>) -> Ty<'tcx> {
2147-
// FIXME: should this normalize?
2148-
let tail = tcx.struct_tail_without_normalization(self);
2146+
pub fn ptr_metadata_ty(
2147+
&'tcx self,
2148+
tcx: TyCtxt<'tcx>,
2149+
normalize: impl FnMut(Ty<'tcx>) -> Ty<'tcx>,
2150+
) -> Ty<'tcx> {
2151+
let tail = tcx.struct_tail_with_normalize(self, normalize);
21492152
match tail.kind() {
21502153
// Sized types
21512154
ty::Infer(ty::IntVar(_) | ty::FloatVar(_))

compiler/rustc_middle/src/ty/util.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -192,7 +192,7 @@ impl<'tcx> TyCtxt<'tcx> {
192192
pub fn struct_tail_with_normalize(
193193
self,
194194
mut ty: Ty<'tcx>,
195-
normalize: impl Fn(Ty<'tcx>) -> Ty<'tcx>,
195+
mut normalize: impl FnMut(Ty<'tcx>) -> Ty<'tcx>,
196196
) -> Ty<'tcx> {
197197
let recursion_limit = self.recursion_limit();
198198
for iteration in 0.. {

compiler/rustc_trait_selection/src/traits/project.rs

+32-6
Original file line numberDiff line numberDiff line change
@@ -1400,8 +1400,17 @@ fn assemble_candidates_from_impls<'cx, 'tcx>(
14001400
// Any type with multiple potential metadata types is therefore not eligible.
14011401
let self_ty = selcx.infcx().shallow_resolve(obligation.predicate.self_ty());
14021402

1403-
// FIXME: should this normalize?
1404-
let tail = selcx.tcx().struct_tail_without_normalization(self_ty);
1403+
let tail = selcx.tcx().struct_tail_with_normalize(self_ty, |ty| {
1404+
normalize_with_depth(
1405+
selcx,
1406+
obligation.param_env,
1407+
obligation.cause.clone(),
1408+
obligation.recursion_depth + 1,
1409+
ty,
1410+
)
1411+
.value
1412+
});
1413+
14051414
match tail.kind() {
14061415
ty::Bool
14071416
| ty::Char
@@ -1435,7 +1444,12 @@ fn assemble_candidates_from_impls<'cx, 'tcx>(
14351444
| ty::Bound(..)
14361445
| ty::Placeholder(..)
14371446
| ty::Infer(..)
1438-
| ty::Error(_) => false,
1447+
| ty::Error(_) => {
1448+
if tail.has_infer_types() {
1449+
candidate_set.mark_ambiguous();
1450+
}
1451+
false
1452+
},
14391453
}
14401454
}
14411455
super::ImplSource::Param(..) => {
@@ -1640,18 +1654,30 @@ fn confirm_pointee_candidate<'cx, 'tcx>(
16401654
_: ImplSourcePointeeData,
16411655
) -> Progress<'tcx> {
16421656
let tcx = selcx.tcx();
1643-
16441657
let self_ty = selcx.infcx().shallow_resolve(obligation.predicate.self_ty());
1645-
let substs = tcx.mk_substs([self_ty.into()].iter());
16461658

1659+
let mut obligations = vec![];
1660+
let metadata_ty = self_ty.ptr_metadata_ty(tcx, |ty| {
1661+
normalize_with_depth_to(
1662+
selcx,
1663+
obligation.param_env,
1664+
obligation.cause.clone(),
1665+
obligation.recursion_depth + 1,
1666+
ty,
1667+
&mut obligations,
1668+
)
1669+
});
1670+
1671+
let substs = tcx.mk_substs([self_ty.into()].iter());
16471672
let metadata_def_id = tcx.require_lang_item(LangItem::Metadata, None);
16481673

16491674
let predicate = ty::ProjectionPredicate {
16501675
projection_ty: ty::ProjectionTy { substs, item_def_id: metadata_def_id },
1651-
ty: self_ty.ptr_metadata_ty(tcx),
1676+
ty: metadata_ty,
16521677
};
16531678

16541679
confirm_param_env_candidate(selcx, obligation, ty::Binder::dummy(predicate), false)
1680+
.with_addl_obligations(obligations)
16551681
}
16561682

16571683
fn confirm_fn_pointer_candidate<'cx, 'tcx>(
+22
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,22 @@
1+
// run-pass
2+
3+
#![feature(ptr_metadata)]
4+
5+
use std::alloc::Layout;
6+
use std::ptr::Pointee;
7+
8+
trait Foo {
9+
type Bar;
10+
}
11+
12+
impl Foo for () {
13+
type Bar = ();
14+
}
15+
16+
struct Wrapper1<T: Foo>(<T as Foo>::Bar);
17+
struct Wrapper2<T: Foo>(<Wrapper1<T> as Pointee>::Metadata);
18+
19+
fn main() {
20+
let _: Wrapper2<()> = Wrapper2(());
21+
let _ = Layout::new::<Wrapper2<()>>();
22+
}

0 commit comments

Comments
 (0)