Extend the solution to encompass HRTB
This commit is contained in:
parent
be8d9bb98a
commit
8d9bb17204
@ -22,6 +22,47 @@ use util::ppaux::Repr;
|
|||||||
use super::{Obligation, ObligationCause, PredicateObligation,
|
use super::{Obligation, ObligationCause, PredicateObligation,
|
||||||
VtableImpl, VtableParam, VtableImplData};
|
VtableImpl, VtableParam, VtableImplData};
|
||||||
|
|
||||||
|
struct PredicateSet<'a,'tcx:'a> {
|
||||||
|
tcx: &'a ty::ctxt<'tcx>,
|
||||||
|
set: FnvHashSet<ty::Predicate<'tcx>>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a,'tcx> PredicateSet<'a,'tcx> {
|
||||||
|
fn new(tcx: &'a ty::ctxt<'tcx>) -> PredicateSet<'a,'tcx> {
|
||||||
|
PredicateSet { tcx: tcx, set: FnvHashSet() }
|
||||||
|
}
|
||||||
|
|
||||||
|
fn insert(&mut self, pred: &ty::Predicate<'tcx>) -> bool {
|
||||||
|
// We have to be careful here because we want
|
||||||
|
//
|
||||||
|
// for<'a> Foo<&'a int>
|
||||||
|
//
|
||||||
|
// and
|
||||||
|
//
|
||||||
|
// for<'b> Foo<&'b int>
|
||||||
|
//
|
||||||
|
// to be considered equivalent. So normalize all late-bound
|
||||||
|
// regions before we throw things into the underlying set.
|
||||||
|
let normalized_pred = match *pred {
|
||||||
|
ty::Predicate::Trait(ref data) =>
|
||||||
|
ty::Predicate::Trait(ty::anonymize_late_bound_regions(self.tcx, data)),
|
||||||
|
|
||||||
|
ty::Predicate::Equate(ref data) =>
|
||||||
|
ty::Predicate::Equate(ty::anonymize_late_bound_regions(self.tcx, data)),
|
||||||
|
|
||||||
|
ty::Predicate::RegionOutlives(ref data) =>
|
||||||
|
ty::Predicate::RegionOutlives(ty::anonymize_late_bound_regions(self.tcx, data)),
|
||||||
|
|
||||||
|
ty::Predicate::TypeOutlives(ref data) =>
|
||||||
|
ty::Predicate::TypeOutlives(ty::anonymize_late_bound_regions(self.tcx, data)),
|
||||||
|
|
||||||
|
ty::Predicate::Projection(ref data) =>
|
||||||
|
ty::Predicate::Projection(ty::anonymize_late_bound_regions(self.tcx, data)),
|
||||||
|
};
|
||||||
|
self.set.insert(normalized_pred)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
///////////////////////////////////////////////////////////////////////////
|
///////////////////////////////////////////////////////////////////////////
|
||||||
// `Elaboration` iterator
|
// `Elaboration` iterator
|
||||||
///////////////////////////////////////////////////////////////////////////
|
///////////////////////////////////////////////////////////////////////////
|
||||||
@ -36,7 +77,7 @@ use super::{Obligation, ObligationCause, PredicateObligation,
|
|||||||
pub struct Elaborator<'cx, 'tcx:'cx> {
|
pub struct Elaborator<'cx, 'tcx:'cx> {
|
||||||
tcx: &'cx ty::ctxt<'tcx>,
|
tcx: &'cx ty::ctxt<'tcx>,
|
||||||
stack: Vec<StackEntry<'tcx>>,
|
stack: Vec<StackEntry<'tcx>>,
|
||||||
visited: FnvHashSet<ty::Predicate<'tcx>>,
|
visited: PredicateSet<'cx,'tcx>,
|
||||||
}
|
}
|
||||||
|
|
||||||
struct StackEntry<'tcx> {
|
struct StackEntry<'tcx> {
|
||||||
@ -68,8 +109,8 @@ pub fn elaborate_predicates<'cx, 'tcx>(
|
|||||||
mut predicates: Vec<ty::Predicate<'tcx>>)
|
mut predicates: Vec<ty::Predicate<'tcx>>)
|
||||||
-> Elaborator<'cx, 'tcx>
|
-> Elaborator<'cx, 'tcx>
|
||||||
{
|
{
|
||||||
let mut visited = FnvHashSet();
|
let mut visited = PredicateSet::new(tcx);
|
||||||
predicates.retain(|pred| visited.insert(pred.clone()));
|
predicates.retain(|pred| visited.insert(pred));
|
||||||
let entry = StackEntry { position: 0, predicates: predicates };
|
let entry = StackEntry { position: 0, predicates: predicates };
|
||||||
Elaborator { tcx: tcx, stack: vec![entry], visited: visited }
|
Elaborator { tcx: tcx, stack: vec![entry], visited: visited }
|
||||||
}
|
}
|
||||||
@ -91,7 +132,7 @@ impl<'cx, 'tcx> Elaborator<'cx, 'tcx> {
|
|||||||
// recursion in some cases. One common case is when
|
// recursion in some cases. One common case is when
|
||||||
// people define `trait Sized: Sized { }` rather than `trait
|
// people define `trait Sized: Sized { }` rather than `trait
|
||||||
// Sized { }`.
|
// Sized { }`.
|
||||||
predicates.retain(|r| self.visited.insert(r.clone()));
|
predicates.retain(|r| self.visited.insert(r));
|
||||||
|
|
||||||
self.stack.push(StackEntry { position: 0,
|
self.stack.push(StackEntry { position: 0,
|
||||||
predicates: predicates });
|
predicates: predicates });
|
||||||
|
@ -0,0 +1,9 @@
|
|||||||
|
fn foo<T>(t: T) -> i32
|
||||||
|
where T : for<'a> Fn(&'a u8) -> i32,
|
||||||
|
T : for<'b> Fn(&'b u8) -> i32,
|
||||||
|
{
|
||||||
|
t(&3)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn main() {
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user