Skip to content

Commit 3cd7581

Browse files
Normalize field before checking PhantomData in coerce/dispatch impl validation
1 parent 11bc805 commit 3cd7581

File tree

2 files changed

+45
-3
lines changed

2 files changed

+45
-3
lines changed

compiler/rustc_hir_analysis/src/coherence/builtin.rs

+20-3
Original file line numberDiff line numberDiff line change
@@ -260,13 +260,22 @@ fn visit_implementation_of_dispatch_from_dyn(checker: &Checker<'_>) -> Result<()
260260
.iter()
261261
.filter(|field| {
262262
// Ignore PhantomData fields
263-
if tcx.type_of(field.did).instantiate_identity().is_phantom_data() {
263+
let unnormalized_ty = tcx.type_of(field.did).instantiate_identity();
264+
if tcx
265+
.try_normalize_erasing_regions(
266+
ty::TypingEnv::non_body_analysis(tcx, def_a.did()),
267+
unnormalized_ty,
268+
)
269+
.unwrap_or(unnormalized_ty)
270+
.is_phantom_data()
271+
{
264272
return false;
265273
}
266274

267275
let ty_a = field.ty(tcx, args_a);
268276
let ty_b = field.ty(tcx, args_b);
269277

278+
// FIXME: We could do normalization here, but is it really worth it?
270279
if ty_a == ty_b {
271280
// Allow 1-ZSTs that don't mention type params.
272281
//
@@ -469,8 +478,16 @@ pub(crate) fn coerce_unsized_info<'tcx>(
469478
.filter_map(|(i, f)| {
470479
let (a, b) = (f.ty(tcx, args_a), f.ty(tcx, args_b));
471480

472-
if tcx.type_of(f.did).instantiate_identity().is_phantom_data() {
473-
// Ignore PhantomData fields
481+
// Ignore PhantomData fields
482+
let unnormalized_ty = tcx.type_of(f.did).instantiate_identity();
483+
if tcx
484+
.try_normalize_erasing_regions(
485+
ty::TypingEnv::non_body_analysis(tcx, def_a.did()),
486+
unnormalized_ty,
487+
)
488+
.unwrap_or(unnormalized_ty)
489+
.is_phantom_data()
490+
{
474491
return None;
475492
}
476493

Original file line numberDiff line numberDiff line change
@@ -0,0 +1,25 @@
1+
//@ check-pass
2+
3+
#![feature(coerce_unsized, dispatch_from_dyn, unsize)]
4+
5+
use std::marker::Unsize;
6+
use std::ops::{CoerceUnsized, DispatchFromDyn};
7+
use std::marker::PhantomData;
8+
9+
trait Mirror {
10+
type Assoc;
11+
}
12+
impl<T> Mirror for T {
13+
type Assoc = T;
14+
}
15+
16+
struct W<T: 'static> {
17+
t: &'static T,
18+
f: <PhantomData<T> as Mirror>::Assoc,
19+
}
20+
21+
impl<T, U> CoerceUnsized<W<U>> for W<T> where T: Unsize<U> {}
22+
23+
impl<T, U> DispatchFromDyn<W<U>> for W<T> where T: Unsize<U> {}
24+
25+
fn main() {}

0 commit comments

Comments
 (0)