Skip to content

Commit 9ab0749

Browse files
committed
Auto merge of #112875 - compiler-errors:negative-coherence-rework, r=lcnr
Rework negative coherence to properly consider impls that only partly overlap This PR implements a modified negative coherence that handles impls that only have partial overlap. It does this by: 1. taking both impl trait refs, instantiating them with infer vars 2. equating both trait refs 3. taking the equated trait ref (which represents the two impls' intersection), and resolving any vars 4. plugging all remaining infer vars with placeholder types these placeholder-plugged trait refs can then be used normally with the new trait solver, since we no longer have to worry about the issue with infer vars in param-envs. We use the **new trait solver** to reason correctly about unnormalized trait refs (due to deferred projection equality), since this avoid having to normalize anything under param-envs with infer vars in them. This PR then additionally: * removes the `FnPtr` knowable hack by implementing proper negative `FnPtr` trait bounds for rigid types. --- An example: Consider these two partially overlapping impls: ``` impl<T, U> PartialEq<&U> for &T where T: PartialEq<U> {} impl<F> PartialEq<F> for F where F: FnPtr {} ``` Under the old algorithm, we would take one of these impls and replace it with infer vars, then try unifying it with the other impl under identity substitutions. This is not possible in either direction, since it either sets `T = U`, or tries to equate `F = &?0`. Under the new algorithm, we try to unify `?0: PartialEq<?0>` with `&?1: PartialEq<&?2>`. This gives us `?0 = &?1 = &?2` and thus `?1 = ?2`. The intersection of these two trait refs therefore looks like: `&?1: PartialEq<&?1>`. After plugging this with placeholders, we get a trait ref that looks like `&!0: PartialEq<&!0>`, with the first impl having substs `?T = ?U = !0` and the second having substs `?F = &!0`[^1]. Then we can take the param-env from the first impl, and try to prove the negated where clause of the second. We know that `&!0: !FnPtr` never holds, since it's a rigid type that is also not a fn ptr, we successfully detect that these impls may never overlap. [^1]: For the purposes of this example, I just ignored lifetimes, since it doesn't really matter.
2 parents 056f5b0 + 0626f2e commit 9ab0749

19 files changed

+314
-143
lines changed

compiler/rustc_middle/src/ty/mod.rs

+1
Original file line numberDiff line numberDiff line change
@@ -233,6 +233,7 @@ impl MainDefinition {
233233
#[derive(Clone, Debug, TypeFoldable, TypeVisitable)]
234234
pub struct ImplHeader<'tcx> {
235235
pub impl_def_id: DefId,
236+
pub impl_args: ty::GenericArgsRef<'tcx>,
236237
pub self_ty: Ty<'tcx>,
237238
pub trait_ref: Option<TraitRef<'tcx>>,
238239
pub predicates: Vec<Predicate<'tcx>>,

compiler/rustc_trait_selection/src/solve/assembly/mod.rs

+2
Original file line numberDiff line numberDiff line change
@@ -37,6 +37,8 @@ pub(super) trait GoalKind<'tcx>:
3737

3838
fn trait_ref(self, tcx: TyCtxt<'tcx>) -> ty::TraitRef<'tcx>;
3939

40+
fn polarity(self) -> ty::ImplPolarity;
41+
4042
fn with_self_ty(self, tcx: TyCtxt<'tcx>, self_ty: Ty<'tcx>) -> Self;
4143

4244
fn trait_def_id(self, tcx: TyCtxt<'tcx>) -> DefId;

compiler/rustc_trait_selection/src/solve/project_goals/mod.rs

+4
Original file line numberDiff line numberDiff line change
@@ -101,6 +101,10 @@ impl<'tcx> assembly::GoalKind<'tcx> for ProjectionPredicate<'tcx> {
101101
self.projection_ty.trait_ref(tcx)
102102
}
103103

104+
fn polarity(self) -> ty::ImplPolarity {
105+
ty::ImplPolarity::Positive
106+
}
107+
104108
fn with_self_ty(self, tcx: TyCtxt<'tcx>, self_ty: Ty<'tcx>) -> Self {
105109
self.with_self_ty(tcx, self_ty)
106110
}

compiler/rustc_trait_selection/src/solve/trait_goals.rs

+23-8
Original file line numberDiff line numberDiff line change
@@ -22,6 +22,10 @@ impl<'tcx> assembly::GoalKind<'tcx> for TraitPredicate<'tcx> {
2222
self.trait_ref
2323
}
2424

25+
fn polarity(self) -> ty::ImplPolarity {
26+
self.polarity
27+
}
28+
2529
fn with_self_ty(self, tcx: TyCtxt<'tcx>, self_ty: Ty<'tcx>) -> Self {
2630
self.with_self_ty(tcx, self_ty)
2731
}
@@ -238,14 +242,25 @@ impl<'tcx> assembly::GoalKind<'tcx> for TraitPredicate<'tcx> {
238242
ecx: &mut EvalCtxt<'_, 'tcx>,
239243
goal: Goal<'tcx, Self>,
240244
) -> QueryResult<'tcx> {
241-
if goal.predicate.polarity != ty::ImplPolarity::Positive {
242-
return Err(NoSolution);
243-
}
244-
245-
if let ty::FnPtr(..) = goal.predicate.self_ty().kind() {
246-
ecx.evaluate_added_goals_and_make_canonical_response(Certainty::Yes)
247-
} else {
248-
Err(NoSolution)
245+
let self_ty = goal.predicate.self_ty();
246+
match goal.predicate.polarity {
247+
ty::ImplPolarity::Positive => {
248+
if self_ty.is_fn_ptr() {
249+
ecx.evaluate_added_goals_and_make_canonical_response(Certainty::Yes)
250+
} else {
251+
Err(NoSolution)
252+
}
253+
}
254+
ty::ImplPolarity::Negative => {
255+
// If a type is rigid and not a fn ptr, then we know for certain
256+
// that it does *not* implement `FnPtr`.
257+
if !self_ty.is_fn_ptr() && self_ty.is_known_rigid() {
258+
ecx.evaluate_added_goals_and_make_canonical_response(Certainty::Yes)
259+
} else {
260+
Err(NoSolution)
261+
}
262+
}
263+
ty::ImplPolarity::Reservation => bug!(),
249264
}
250265
}
251266

0 commit comments

Comments
 (0)