Use variances for defining use diag.
This commit is contained in:
parent
b676fc4d8e
commit
47de11f1ed
@ -1345,61 +1345,13 @@ pub fn remap_generic_params_to_declaration_params(
|
|||||||
// type Foo<'a, 'b, 'c> = impl Trait<'a> + 'b;
|
// type Foo<'a, 'b, 'c> = impl Trait<'a> + 'b;
|
||||||
// ```
|
// ```
|
||||||
// we may not use `'c` in the hidden type.
|
// we may not use `'c` in the hidden type.
|
||||||
struct OpaqueTypeLifetimeCollector<'tcx> {
|
let variances = tcx.variances_of(def_id);
|
||||||
lifetimes: FxHashSet<ty::Region<'tcx>>,
|
debug!(?variances);
|
||||||
}
|
|
||||||
|
|
||||||
impl<'tcx> ty::TypeVisitor<'tcx> for OpaqueTypeLifetimeCollector<'tcx> {
|
|
||||||
fn visit_region(&mut self, r: ty::Region<'tcx>) -> ControlFlow<Self::BreakTy> {
|
|
||||||
self.lifetimes.insert(r);
|
|
||||||
r.super_visit_with(self)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
let mut collector = OpaqueTypeLifetimeCollector { lifetimes: Default::default() };
|
|
||||||
|
|
||||||
for pred in tcx.bound_explicit_item_bounds(def_id.to_def_id()).transpose_iter() {
|
|
||||||
let pred = pred.map_bound(|(pred, _)| *pred).subst(tcx, id_substs);
|
|
||||||
|
|
||||||
trace!(pred=?pred.kind());
|
|
||||||
|
|
||||||
// We only ignore opaque type substs if the opaque type is the outermost type.
|
|
||||||
// The opaque type may be nested within itself via recursion in e.g.
|
|
||||||
// type Foo<'a> = impl PartialEq<Foo<'a>>;
|
|
||||||
// which thus mentions `'a` and should thus accept hidden types that borrow 'a
|
|
||||||
// instead of requiring an additional `+ 'a`.
|
|
||||||
match pred.kind().skip_binder() {
|
|
||||||
ty::PredicateKind::Trait(TraitPredicate {
|
|
||||||
trait_ref: ty::TraitRef { def_id: _, substs },
|
|
||||||
constness: _,
|
|
||||||
polarity: _,
|
|
||||||
}) => {
|
|
||||||
trace!(?substs);
|
|
||||||
for subst in &substs[1..] {
|
|
||||||
subst.visit_with(&mut collector);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
ty::PredicateKind::Projection(ty::ProjectionPredicate {
|
|
||||||
projection_ty: ty::ProjectionTy { substs, item_def_id: _ },
|
|
||||||
term,
|
|
||||||
}) => {
|
|
||||||
for subst in &substs[1..] {
|
|
||||||
subst.visit_with(&mut collector);
|
|
||||||
}
|
|
||||||
term.visit_with(&mut collector);
|
|
||||||
}
|
|
||||||
_ => {
|
|
||||||
pred.visit_with(&mut collector);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
let lifetimes = collector.lifetimes;
|
|
||||||
trace!(?lifetimes);
|
|
||||||
map.filter(|(_, v)| {
|
map.filter(|(_, v)| {
|
||||||
let ty::GenericArgKind::Lifetime(lt) = v.unpack() else {
|
let ty::GenericArgKind::Lifetime(lt) = v.unpack() else { return true };
|
||||||
return true;
|
let ty::ReEarlyBound(ebr) = lt.kind() else { bug!() };
|
||||||
};
|
variances[ebr.index as usize] == ty::Variance::Invariant
|
||||||
lifetimes.contains(<)
|
|
||||||
})
|
})
|
||||||
.collect()
|
.collect()
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user