Skip to content

Commit dc8c3d6

Browse files
authored
Unrolled build for rust-lang#124936
Rollup merge of rust-lang#124936 - lcnr:cool-beans, r=compiler-errors analyse visitor: build proof tree in probe see inline comments fixes rust-lang#124791 fixes rust-lang#124702 r? ```@compiler-errors```
2 parents 98dabb6 + feff752 commit dc8c3d6

File tree

8 files changed

+113
-34
lines changed

8 files changed

+113
-34
lines changed

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

+1
Original file line numberDiff line numberDiff line change
@@ -409,6 +409,7 @@ pub(in crate::solve) fn make_canonical_state<'tcx, T: TypeFoldable<TyCtxt<'tcx>>
409409
/// This currently assumes that unifying the var values trivially succeeds.
410410
/// Adding any inference constraints which weren't present when originally
411411
/// computing the canonical query can result in bugs.
412+
#[instrument(level = "debug", skip(infcx, span, param_env))]
412413
pub(in crate::solve) fn instantiate_canonical_state<'tcx, T: TypeFoldable<TyCtxt<'tcx>>>(
413414
infcx: &InferCtxt<'tcx>,
414415
span: Span,

compiler/rustc_trait_selection/src/solve/inspect/analyse.rs

+33-11
Original file line numberDiff line numberDiff line change
@@ -146,6 +146,11 @@ impl<'a, 'tcx> InspectCandidate<'a, 'tcx> {
146146
/// inference constraints, and optionally the args of an impl if this candidate
147147
/// came from a `CandidateSource::Impl`. This function modifies the state of the
148148
/// `infcx`.
149+
#[instrument(
150+
level = "debug",
151+
skip_all,
152+
fields(goal = ?self.goal.goal, nested_goals = ?self.nested_goals)
153+
)]
149154
pub fn instantiate_nested_goals_and_opt_impl_args(
150155
&self,
151156
span: Span,
@@ -213,10 +218,23 @@ impl<'a, 'tcx> InspectCandidate<'a, 'tcx> {
213218
};
214219
let goal =
215220
goal.with(infcx.tcx, ty::NormalizesTo { alias, term: unconstrained_term });
216-
let proof_tree = EvalCtxt::enter_root(infcx, GenerateProofTree::Yes, |ecx| {
217-
ecx.evaluate_goal_raw(GoalEvaluationKind::Root, GoalSource::Misc, goal)
218-
})
219-
.1;
221+
// We have to use a `probe` here as evaluating a `NormalizesTo` can constrain the
222+
// expected term. This means that candidates which only fail due to nested goals
223+
// and which normalize to a different term then the final result could ICE: when
224+
// building their proof tree, the expected term was unconstrained, but when
225+
// instantiating the candidate it is already constrained to the result of another
226+
// candidate.
227+
let proof_tree = infcx
228+
.probe(|_| {
229+
EvalCtxt::enter_root(infcx, GenerateProofTree::Yes, |ecx| {
230+
ecx.evaluate_goal_raw(
231+
GoalEvaluationKind::Root,
232+
GoalSource::Misc,
233+
goal,
234+
)
235+
})
236+
})
237+
.1;
220238
InspectGoal::new(
221239
infcx,
222240
self.goal.depth + 1,
@@ -225,13 +243,17 @@ impl<'a, 'tcx> InspectCandidate<'a, 'tcx> {
225243
source,
226244
)
227245
}
228-
_ => InspectGoal::new(
229-
infcx,
230-
self.goal.depth + 1,
231-
infcx.evaluate_root_goal(goal, GenerateProofTree::Yes).1.unwrap(),
232-
None,
233-
source,
234-
),
246+
_ => {
247+
// We're using a probe here as evaluating a goal could constrain
248+
// inference variables by choosing one candidate. If we then recurse
249+
// into another candidate who ends up with different inference
250+
// constraints, we get an ICE if we already applied the constraints
251+
// from the chosen candidate.
252+
let proof_tree = infcx
253+
.probe(|_| infcx.evaluate_root_goal(goal, GenerateProofTree::Yes).1)
254+
.unwrap();
255+
InspectGoal::new(infcx, self.goal.depth + 1, proof_tree, None, source)
256+
}
235257
})
236258
.collect();
237259

tests/crashes/124702.rs

-14
This file was deleted.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,43 @@
1+
//@ compile-flags: -Znext-solver=coherence
2+
//@ check-pass
3+
4+
// A regression test for #124791. Computing ambiguity causes
5+
// for the overlap of the `ToString` impls caused an ICE.
6+
#![crate_type = "lib"]
7+
#![feature(min_specialization)]
8+
trait Display {}
9+
10+
trait ToOwned {
11+
type Owned;
12+
}
13+
14+
impl<T> ToOwned for T {
15+
type Owned = T;
16+
}
17+
18+
struct Cow<B: ?Sized>(B);
19+
20+
impl<B: ?Sized> Display for Cow<B>
21+
where
22+
B: ToOwned,
23+
B::Owned: Display,
24+
{
25+
}
26+
27+
impl Display for () {}
28+
29+
trait ToString {
30+
fn to_string();
31+
}
32+
33+
impl<T: Display + ?Sized> ToString for T {
34+
default fn to_string() {}
35+
}
36+
37+
impl ToString for Cow<str> {
38+
fn to_string() {}
39+
}
40+
41+
impl ToOwned for str {
42+
type Owned = ();
43+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,19 @@
1+
//@ compile-flags: -Znext-solver=coherence
2+
3+
// A regression test for #124791. Computing ambiguity causes
4+
// for the overlap of the `ToString` impls caused an ICE.
5+
#![crate_type = "lib"]
6+
trait ToOwned {
7+
type Owned;
8+
}
9+
impl<T> ToOwned for T {
10+
type Owned = u8;
11+
}
12+
impl ToOwned for str {
13+
type Owned = i8;
14+
}
15+
16+
trait Overlap {}
17+
impl<T: ToOwned<Owned = i8> + ?Sized> Overlap for T {}
18+
impl Overlap for str {}
19+
//~^ ERROR conflicting implementations of trait `Overlap`
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,11 @@
1+
error[E0119]: conflicting implementations of trait `Overlap` for type `str`
2+
--> $DIR/ambiguity-causes-canonical-state-ice-2.rs:18:1
3+
|
4+
LL | impl<T: ToOwned<Owned = i8> + ?Sized> Overlap for T {}
5+
| --------------------------------------------------- first implementation here
6+
LL | impl Overlap for str {}
7+
| ^^^^^^^^^^^^^^^^^^^^ conflicting implementation for `str`
8+
9+
error: aborting due to 1 previous error
10+
11+
For more information about this error, try `rustc --explain E0119`.

tests/ui/traits/next-solver/cycles/coinduction/incompleteness-unstable-result.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -61,7 +61,7 @@ where
6161
// entering the cycle from `A` fails, but would work if we were to use the cache
6262
// result of `B<X>`.
6363
impls_trait::<A<X>, _, _, _>();
64-
//~^ ERROR the trait bound `A<X>: Trait<i32, u8, u8>` is not satisfied
64+
//~^ ERROR the trait bound `A<X>: Trait<_, _, _>` is not satisfied
6565
}
6666

6767
fn main() {

tests/ui/traits/next-solver/cycles/coinduction/incompleteness-unstable-result.stderr

+5-8
Original file line numberDiff line numberDiff line change
@@ -1,10 +1,11 @@
1-
error[E0277]: the trait bound `A<X>: Trait<i32, u8, u8>` is not satisfied
1+
error[E0277]: the trait bound `A<X>: Trait<_, _, _>` is not satisfied
22
--> $DIR/incompleteness-unstable-result.rs:63:19
33
|
44
LL | impls_trait::<A<X>, _, _, _>();
5-
| ^^^^ the trait `Trait<i32, u8, u8>` is not implemented for `A<X>`, which is required by `A<X>: Trait<_, _, _>`
5+
| ^^^^ the trait `Trait<_, _, _>` is not implemented for `A<X>`, which is required by `A<X>: Trait<_, _, _>`
66
|
7-
note: required for `A<X>` to implement `Trait<i32, u8, u8>`
7+
= help: the trait `Trait<U, V, D>` is implemented for `A<T>`
8+
note: required for `A<X>` to implement `Trait<_, _, _>`
89
--> $DIR/incompleteness-unstable-result.rs:32:50
910
|
1011
LL | impl<T: ?Sized, U: ?Sized, V: ?Sized, D: ?Sized> Trait<U, V, D> for A<T>
@@ -13,16 +14,12 @@ LL | impl<T: ?Sized, U: ?Sized, V: ?Sized, D: ?Sized> Trait<U, V, D> for A<T>
1314
LL | A<T>: Trait<U, D, V>,
1415
| -------------- unsatisfied trait bound introduced here
1516
= note: 8 redundant requirements hidden
16-
= note: required for `A<X>` to implement `Trait<i32, u8, u8>`
17+
= note: required for `A<X>` to implement `Trait<_, _, _>`
1718
note: required by a bound in `impls_trait`
1819
--> $DIR/incompleteness-unstable-result.rs:51:28
1920
|
2021
LL | fn impls_trait<T: ?Sized + Trait<U, V, D>, U: ?Sized, V: ?Sized, D: ?Sized>() {}
2122
| ^^^^^^^^^^^^^^ required by this bound in `impls_trait`
22-
help: consider extending the `where` clause, but there might be an alternative better way to express this requirement
23-
|
24-
LL | X: IncompleteGuidance<u32, i16>, A<X>: Trait<i32, u8, u8>
25-
| ~~~~~~~~~~~~~~~~~~~~~~~~~~
2623

2724
error: aborting due to 1 previous error
2825

0 commit comments

Comments
 (0)