Skip to content

Commit c73e232

Browse files
authored
Rollup merge of #114200 - compiler-errors:detect-tail-unsize-then-upcast, r=lcnr
Detect trait upcasting through struct tail unsizing in new solver select Oops, we were able to hide trait upcasting behind a parent unsize goal that evaluated to `Certainty::Yes`. Let's do rematching for `Certainty::Yes` unsize goals with `BuiltinImplSource::Misc` sources (corresponding to all of the other unsize rules) to make sure we end up selecting any nested goals which may be satisfied via `BuiltinImplSource::TraitUpcasting` or `::TupleUnsizing`. r? ``@lcnr``
2 parents 57c57a5 + 752e6e1 commit c73e232

File tree

4 files changed

+60
-5
lines changed

4 files changed

+60
-5
lines changed

compiler/rustc_trait_selection/src/solve/eval_ctxt/select.rs

+20-5
Original file line numberDiff line numberDiff line change
@@ -100,10 +100,18 @@ impl<'tcx> InferCtxtSelectExt<'tcx> for InferCtxt<'tcx> {
100100
rematch_impl(self, goal, def_id, nested_obligations)
101101
}
102102

103+
// If an unsize goal is ambiguous, then we can manually rematch it to make
104+
// selection progress for coercion during HIR typeck. If it is *not* ambiguous,
105+
// but is `BuiltinImplSource::Misc`, it may have nested `Unsize` goals,
106+
// and we need to rematch those to detect tuple unsizing and trait upcasting.
107+
// FIXME: This will be wrong if we have param-env or where-clause bounds
108+
// with the unsize goal -- we may need to mark those with different impl
109+
// sources.
103110
(Certainty::Maybe(_), CandidateSource::BuiltinImpl(src))
111+
| (Certainty::Yes, CandidateSource::BuiltinImpl(src @ BuiltinImplSource::Misc))
104112
if self.tcx.lang_items().unsize_trait() == Some(goal.predicate.def_id()) =>
105113
{
106-
rematch_unsize(self, goal, nested_obligations, src)
114+
rematch_unsize(self, goal, nested_obligations, src, certainty)
107115
}
108116

109117
// Technically some builtin impls have nested obligations, but if
@@ -217,6 +225,7 @@ fn rematch_unsize<'tcx>(
217225
goal: Goal<'tcx, ty::TraitPredicate<'tcx>>,
218226
mut nested: Vec<PredicateObligation<'tcx>>,
219227
source: BuiltinImplSource,
228+
certainty: Certainty,
220229
) -> SelectionResult<'tcx, Selection<'tcx>> {
221230
let tcx = infcx.tcx;
222231
let a_ty = structurally_normalize(goal.predicate.self_ty(), infcx, goal.param_env, &mut nested);
@@ -227,6 +236,12 @@ fn rematch_unsize<'tcx>(
227236
&mut nested,
228237
);
229238
match (a_ty.kind(), b_ty.kind()) {
239+
// Stall any ambiguous upcasting goals, since we can't rematch those
240+
(ty::Dynamic(_, _, ty::Dyn), ty::Dynamic(_, _, ty::Dyn)) => match certainty {
241+
Certainty::Yes => Ok(Some(ImplSource::Builtin(source, nested))),
242+
_ => Ok(None),
243+
},
244+
// `T` -> `dyn Trait` upcasting
230245
(_, &ty::Dynamic(data, region, ty::Dyn)) => {
231246
// Check that the type implements all of the predicates of the def-id.
232247
// (i.e. the principal, all of the associated types match, and any auto traits)
@@ -354,10 +369,10 @@ fn rematch_unsize<'tcx>(
354369
);
355370
Ok(Some(ImplSource::Builtin(source, nested)))
356371
}
357-
// FIXME: We *could* ICE here if either:
358-
// 1. the certainty is `Certainty::Yes`,
359-
// 2. we're in codegen (which should mean `Certainty::Yes`).
360-
_ => Ok(None),
372+
_ => {
373+
assert_ne!(certainty, Certainty::Yes);
374+
Ok(None)
375+
}
361376
}
362377
}
363378

Original file line numberDiff line numberDiff line change
@@ -0,0 +1,13 @@
1+
error[E0658]: cannot cast `dyn A` to `dyn B`, trait upcasting coercion is experimental
2+
--> $DIR/upcast-through-struct-tail.rs:10:5
3+
|
4+
LL | x
5+
| ^
6+
|
7+
= note: see issue #65991 <https://github.com/rust-lang/rust/issues/65991> for more information
8+
= help: add `#![feature(trait_upcasting)]` to the crate attributes to enable
9+
= note: required when coercing `Box<Wrapper<(dyn A + 'a)>>` into `Box<Wrapper<(dyn B + 'a)>>`
10+
11+
error: aborting due to previous error
12+
13+
For more information about this error, try `rustc --explain E0658`.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,13 @@
1+
error[E0658]: cannot cast `dyn A` to `dyn B`, trait upcasting coercion is experimental
2+
--> $DIR/upcast-through-struct-tail.rs:10:5
3+
|
4+
LL | x
5+
| ^
6+
|
7+
= note: see issue #65991 <https://github.com/rust-lang/rust/issues/65991> for more information
8+
= help: add `#![feature(trait_upcasting)]` to the crate attributes to enable
9+
= note: required when coercing `Box<Wrapper<(dyn A + 'a)>>` into `Box<Wrapper<(dyn B + 'a)>>`
10+
11+
error: aborting due to previous error
12+
13+
For more information about this error, try `rustc --explain E0658`.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,14 @@
1+
// revisions: current next
2+
//[next] compile-flags: -Ztrait-solver=next
3+
4+
struct Wrapper<T: ?Sized>(T);
5+
6+
trait A: B {}
7+
trait B {}
8+
9+
fn test<'a>(x: Box<Wrapper<dyn A + 'a>>) -> Box<Wrapper<dyn B + 'a>> {
10+
x
11+
//~^ ERROR cannot cast `dyn A` to `dyn B`, trait upcasting coercion is experimental
12+
}
13+
14+
fn main() {}

0 commit comments

Comments
 (0)