Permit negative impls coherence to take advantage of implied bounds
This commit is contained in:
parent
5ff45dc89e
commit
ac0b6af37b
3 changed files with 29 additions and 22 deletions
|
@ -6,12 +6,13 @@
|
||||||
|
|
||||||
use crate::infer::outlives::env::OutlivesEnvironment;
|
use crate::infer::outlives::env::OutlivesEnvironment;
|
||||||
use crate::infer::{CombinedSnapshot, InferOk};
|
use crate::infer::{CombinedSnapshot, InferOk};
|
||||||
|
use crate::traits::outlives_bounds::InferCtxtExt as _;
|
||||||
use crate::traits::select::IntercrateAmbiguityCause;
|
use crate::traits::select::IntercrateAmbiguityCause;
|
||||||
use crate::traits::util::impl_subject_and_oblig;
|
use crate::traits::util::impl_subject_and_oblig;
|
||||||
use crate::traits::SkipLeakCheck;
|
use crate::traits::SkipLeakCheck;
|
||||||
use crate::traits::{
|
use crate::traits::{
|
||||||
self, Normalized, Obligation, ObligationCause, PredicateObligation, PredicateObligations,
|
self, Normalized, Obligation, ObligationCause, ObligationCtxt, PredicateObligation,
|
||||||
SelectionContext,
|
PredicateObligations, SelectionContext,
|
||||||
};
|
};
|
||||||
use rustc_data_structures::fx::FxIndexSet;
|
use rustc_data_structures::fx::FxIndexSet;
|
||||||
use rustc_errors::Diagnostic;
|
use rustc_errors::Diagnostic;
|
||||||
|
@ -322,7 +323,7 @@ fn negative_impl<'cx, 'tcx>(
|
||||||
let (subject2, obligations) =
|
let (subject2, obligations) =
|
||||||
impl_subject_and_oblig(selcx, impl_env, impl2_def_id, impl2_substs);
|
impl_subject_and_oblig(selcx, impl_env, impl2_def_id, impl2_substs);
|
||||||
|
|
||||||
!equate(&infcx, impl_env, subject1, subject2, obligations)
|
!equate(&infcx, impl_env, subject1, subject2, obligations, impl1_def_id)
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -332,6 +333,7 @@ fn equate<'cx, 'tcx>(
|
||||||
subject1: ImplSubject<'tcx>,
|
subject1: ImplSubject<'tcx>,
|
||||||
subject2: ImplSubject<'tcx>,
|
subject2: ImplSubject<'tcx>,
|
||||||
obligations: impl Iterator<Item = PredicateObligation<'tcx>>,
|
obligations: impl Iterator<Item = PredicateObligation<'tcx>>,
|
||||||
|
body_def_id: DefId,
|
||||||
) -> bool {
|
) -> bool {
|
||||||
// do the impls unify? If not, not disjoint.
|
// do the impls unify? If not, not disjoint.
|
||||||
let Ok(InferOk { obligations: more_obligations, .. }) =
|
let Ok(InferOk { obligations: more_obligations, .. }) =
|
||||||
|
@ -342,8 +344,10 @@ fn equate<'cx, 'tcx>(
|
||||||
};
|
};
|
||||||
|
|
||||||
let selcx = &mut SelectionContext::new(&infcx);
|
let selcx = &mut SelectionContext::new(&infcx);
|
||||||
let opt_failing_obligation =
|
let opt_failing_obligation = obligations
|
||||||
obligations.into_iter().chain(more_obligations).find(|o| negative_impl_exists(selcx, o));
|
.into_iter()
|
||||||
|
.chain(more_obligations)
|
||||||
|
.find(|o| negative_impl_exists(selcx, o, body_def_id));
|
||||||
|
|
||||||
if let Some(failing_obligation) = opt_failing_obligation {
|
if let Some(failing_obligation) = opt_failing_obligation {
|
||||||
debug!("overlap: obligation unsatisfiable {:?}", failing_obligation);
|
debug!("overlap: obligation unsatisfiable {:?}", failing_obligation);
|
||||||
|
@ -358,14 +362,15 @@ fn equate<'cx, 'tcx>(
|
||||||
fn negative_impl_exists<'cx, 'tcx>(
|
fn negative_impl_exists<'cx, 'tcx>(
|
||||||
selcx: &SelectionContext<'cx, 'tcx>,
|
selcx: &SelectionContext<'cx, 'tcx>,
|
||||||
o: &PredicateObligation<'tcx>,
|
o: &PredicateObligation<'tcx>,
|
||||||
|
body_def_id: DefId,
|
||||||
) -> bool {
|
) -> bool {
|
||||||
if resolve_negative_obligation(selcx.infcx().fork(), o) {
|
if resolve_negative_obligation(selcx.infcx().fork(), o, body_def_id) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Try to prove a negative obligation exists for super predicates
|
// Try to prove a negative obligation exists for super predicates
|
||||||
for o in util::elaborate_predicates(selcx.tcx(), iter::once(o.predicate)) {
|
for o in util::elaborate_predicates(selcx.tcx(), iter::once(o.predicate)) {
|
||||||
if resolve_negative_obligation(selcx.infcx().fork(), &o) {
|
if resolve_negative_obligation(selcx.infcx().fork(), &o, body_def_id) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -377,6 +382,7 @@ fn negative_impl_exists<'cx, 'tcx>(
|
||||||
fn resolve_negative_obligation<'cx, 'tcx>(
|
fn resolve_negative_obligation<'cx, 'tcx>(
|
||||||
infcx: InferCtxt<'cx, 'tcx>,
|
infcx: InferCtxt<'cx, 'tcx>,
|
||||||
o: &PredicateObligation<'tcx>,
|
o: &PredicateObligation<'tcx>,
|
||||||
|
body_def_id: DefId,
|
||||||
) -> bool {
|
) -> bool {
|
||||||
let tcx = infcx.tcx;
|
let tcx = infcx.tcx;
|
||||||
|
|
||||||
|
@ -390,7 +396,19 @@ fn resolve_negative_obligation<'cx, 'tcx>(
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
let outlives_env = OutlivesEnvironment::new(param_env);
|
let outlives_env = if let Some(body_def_id) = body_def_id.as_local() {
|
||||||
|
let body_id = tcx.hir().local_def_id_to_hir_id(body_def_id);
|
||||||
|
let ocx = ObligationCtxt::new(&infcx);
|
||||||
|
let wf_tys = ocx.assumed_wf_types(param_env, DUMMY_SP, body_def_id);
|
||||||
|
OutlivesEnvironment::with_bounds(
|
||||||
|
param_env,
|
||||||
|
Some(&infcx),
|
||||||
|
infcx.implied_bounds_tys(param_env, body_id, wf_tys),
|
||||||
|
)
|
||||||
|
} else {
|
||||||
|
OutlivesEnvironment::new(param_env)
|
||||||
|
};
|
||||||
|
|
||||||
infcx.process_registered_region_obligations(outlives_env.region_bound_pairs(), param_env);
|
infcx.process_registered_region_obligations(outlives_env.region_bound_pairs(), param_env);
|
||||||
|
|
||||||
infcx.resolve_regions(&outlives_env).is_empty()
|
infcx.resolve_regions(&outlives_env).is_empty()
|
||||||
|
|
|
@ -1,9 +1,9 @@
|
||||||
// revisions: stock with_negative_coherence
|
// revisions: stock with_negative_coherence
|
||||||
|
//[with_negative_coherence] check-pass
|
||||||
|
|
||||||
#![feature(negative_impls)]
|
#![feature(negative_impls)]
|
||||||
#![cfg_attr(with_negative_coherence, feature(with_negative_coherence))]
|
#![cfg_attr(with_negative_coherence, feature(with_negative_coherence))]
|
||||||
|
|
||||||
// FIXME: this should compile
|
|
||||||
|
|
||||||
trait MyPredicate<'a> {}
|
trait MyPredicate<'a> {}
|
||||||
|
|
||||||
impl<'a, T> !MyPredicate<'a> for &'a T where T: 'a {}
|
impl<'a, T> !MyPredicate<'a> for &'a T where T: 'a {}
|
||||||
|
@ -12,6 +12,6 @@ trait MyTrait<'a> {}
|
||||||
|
|
||||||
impl<'a, T: MyPredicate<'a>> MyTrait<'a> for T {}
|
impl<'a, T: MyPredicate<'a>> MyTrait<'a> for T {}
|
||||||
impl<'a, T> MyTrait<'a> for &'a T {}
|
impl<'a, T> MyTrait<'a> for &'a T {}
|
||||||
//~^ ERROR: conflicting implementations of trait `MyTrait<'_>` for type `&_`
|
//[stock]~^ ERROR: conflicting implementations of trait `MyTrait<'_>` for type `&_`
|
||||||
|
|
||||||
fn main() {}
|
fn main() {}
|
||||||
|
|
|
@ -1,11 +0,0 @@
|
||||||
error[E0119]: conflicting implementations of trait `MyTrait<'_>` for type `&_`
|
|
||||||
--> $DIR/coherence-negative-outlives-lifetimes.rs:14:1
|
|
||||||
|
|
|
||||||
LL | impl<'a, T: MyPredicate<'a>> MyTrait<'a> for T {}
|
|
||||||
| ---------------------------------------------- first implementation here
|
|
||||||
LL | impl<'a, T> MyTrait<'a> for &'a T {}
|
|
||||||
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ conflicting implementation for `&_`
|
|
||||||
|
|
||||||
error: aborting due to previous error
|
|
||||||
|
|
||||||
For more information about this error, try `rustc --explain E0119`.
|
|
Loading…
Add table
Reference in a new issue