2022-08-01 16:03:49 -03:00
|
|
|
use super::ResolverAstLoweringExt;
|
2022-07-29 22:38:07 -03:00
|
|
|
use rustc_ast::visit::{self, BoundKind, LifetimeCtxt, Visitor};
|
2022-08-11 11:05:26 +10:00
|
|
|
use rustc_ast::{FnRetTy, GenericBounds, Lifetime, NodeId, PathSegment, PolyTraitRef, Ty, TyKind};
|
2022-08-01 16:03:49 -03:00
|
|
|
use rustc_hir::def::LifetimeRes;
|
2022-08-04 12:40:00 -03:00
|
|
|
use rustc_middle::span_bug;
|
2022-08-01 16:03:49 -03:00
|
|
|
use rustc_middle::ty::ResolverAstLowering;
|
2022-08-02 11:03:29 -03:00
|
|
|
use rustc_span::symbol::{kw, Ident};
|
|
|
|
use rustc_span::Span;
|
2022-07-29 22:38:07 -03:00
|
|
|
|
2022-08-02 11:03:29 -03:00
|
|
|
struct LifetimeCollectVisitor<'ast> {
|
|
|
|
resolver: &'ast ResolverAstLowering,
|
2022-07-29 22:38:07 -03:00
|
|
|
current_binders: Vec<NodeId>,
|
2022-08-02 11:03:29 -03:00
|
|
|
collected_lifetimes: Vec<Lifetime>,
|
2022-07-29 22:38:07 -03:00
|
|
|
}
|
|
|
|
|
2022-08-02 11:03:29 -03:00
|
|
|
impl<'ast> LifetimeCollectVisitor<'ast> {
|
|
|
|
fn new(resolver: &'ast ResolverAstLowering) -> Self {
|
2022-08-01 16:03:49 -03:00
|
|
|
Self { resolver, current_binders: Vec::new(), collected_lifetimes: Vec::new() }
|
|
|
|
}
|
|
|
|
|
2022-08-02 11:03:29 -03:00
|
|
|
fn record_lifetime_use(&mut self, lifetime: Lifetime) {
|
2022-08-04 12:40:00 -03:00
|
|
|
match self.resolver.get_lifetime_res(lifetime.id).unwrap_or(LifetimeRes::Error) {
|
|
|
|
LifetimeRes::Param { binder, .. } | LifetimeRes::Fresh { binder, .. } => {
|
|
|
|
if !self.current_binders.contains(&binder) {
|
|
|
|
if !self.collected_lifetimes.contains(&lifetime) {
|
|
|
|
self.collected_lifetimes.push(lifetime);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
LifetimeRes::Static | LifetimeRes::Error => {
|
|
|
|
if !self.collected_lifetimes.contains(&lifetime) {
|
|
|
|
self.collected_lifetimes.push(lifetime);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
LifetimeRes::Infer => {}
|
|
|
|
res => {
|
|
|
|
let bug_msg = format!(
|
|
|
|
"Unexpected lifetime resolution {:?} for {:?} at {:?}",
|
|
|
|
res, lifetime.ident, lifetime.ident.span
|
|
|
|
);
|
|
|
|
span_bug!(lifetime.ident.span, "{}", bug_msg);
|
2022-08-01 16:03:49 -03:00
|
|
|
}
|
2022-07-29 22:38:07 -03:00
|
|
|
}
|
|
|
|
}
|
2022-08-04 10:10:04 -03:00
|
|
|
|
2022-08-04 10:12:56 -03:00
|
|
|
/// This collect lifetimes that are elided, for nodes like `Foo<T>` where there are no explicit
|
|
|
|
/// lifetime nodes. Is equivalent to having "pseudo" nodes introduced for each of the node ids
|
|
|
|
/// in the list start..end.
|
2022-08-04 10:10:04 -03:00
|
|
|
fn record_elided_anchor(&mut self, node_id: NodeId, span: Span) {
|
|
|
|
if let Some(LifetimeRes::ElidedAnchor { start, end }) =
|
|
|
|
self.resolver.get_lifetime_res(node_id)
|
|
|
|
{
|
|
|
|
for i in start..end {
|
|
|
|
let lifetime = Lifetime { id: i, ident: Ident::new(kw::UnderscoreLifetime, span) };
|
|
|
|
self.record_lifetime_use(lifetime);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2022-08-02 11:03:29 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
impl<'ast> Visitor<'ast> for LifetimeCollectVisitor<'ast> {
|
|
|
|
fn visit_lifetime(&mut self, lifetime: &'ast Lifetime, _: LifetimeCtxt) {
|
|
|
|
self.record_lifetime_use(*lifetime);
|
|
|
|
}
|
|
|
|
|
|
|
|
fn visit_path_segment(&mut self, path_span: Span, path_segment: &'ast PathSegment) {
|
2022-08-04 10:10:04 -03:00
|
|
|
self.record_elided_anchor(path_segment.id, path_span);
|
2022-08-02 11:03:29 -03:00
|
|
|
visit::walk_path_segment(self, path_span, path_segment);
|
|
|
|
}
|
2022-07-29 22:38:07 -03:00
|
|
|
|
2022-08-11 11:05:26 +10:00
|
|
|
fn visit_poly_trait_ref(&mut self, t: &'ast PolyTraitRef) {
|
2022-07-29 22:38:07 -03:00
|
|
|
self.current_binders.push(t.trait_ref.ref_id);
|
|
|
|
|
2022-08-11 11:05:26 +10:00
|
|
|
visit::walk_poly_trait_ref(self, t);
|
2022-07-29 22:38:07 -03:00
|
|
|
|
|
|
|
self.current_binders.pop();
|
|
|
|
}
|
|
|
|
|
|
|
|
fn visit_ty(&mut self, t: &'ast Ty) {
|
2022-08-02 11:00:34 -03:00
|
|
|
match t.kind {
|
|
|
|
TyKind::BareFn(_) => {
|
|
|
|
self.current_binders.push(t.id);
|
|
|
|
visit::walk_ty(self, t);
|
|
|
|
self.current_binders.pop();
|
|
|
|
}
|
2022-08-02 15:06:12 -03:00
|
|
|
TyKind::Rptr(None, _) => {
|
2022-08-04 10:10:04 -03:00
|
|
|
self.record_elided_anchor(t.id, t.span);
|
2022-08-02 15:06:12 -03:00
|
|
|
visit::walk_ty(self, t);
|
|
|
|
}
|
2022-08-02 11:00:34 -03:00
|
|
|
_ => {
|
|
|
|
visit::walk_ty(self, t);
|
|
|
|
}
|
2022-07-29 22:38:07 -03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-08-02 11:03:29 -03:00
|
|
|
pub fn lifetimes_in_ret_ty(resolver: &ResolverAstLowering, ret_ty: &FnRetTy) -> Vec<Lifetime> {
|
2022-08-01 16:03:49 -03:00
|
|
|
let mut visitor = LifetimeCollectVisitor::new(resolver);
|
2022-07-29 22:38:07 -03:00
|
|
|
visitor.visit_fn_ret_ty(ret_ty);
|
2022-08-01 16:03:49 -03:00
|
|
|
visitor.collected_lifetimes
|
2022-07-29 22:38:07 -03:00
|
|
|
}
|
|
|
|
|
2022-08-02 11:03:29 -03:00
|
|
|
pub fn lifetimes_in_bounds(
|
|
|
|
resolver: &ResolverAstLowering,
|
|
|
|
bounds: &GenericBounds,
|
|
|
|
) -> Vec<Lifetime> {
|
2022-08-01 16:03:49 -03:00
|
|
|
let mut visitor = LifetimeCollectVisitor::new(resolver);
|
2022-07-29 22:38:07 -03:00
|
|
|
for bound in bounds {
|
|
|
|
visitor.visit_param_bound(bound, BoundKind::Bound);
|
|
|
|
}
|
2022-08-01 16:03:49 -03:00
|
|
|
visitor.collected_lifetimes
|
2022-07-29 22:38:07 -03:00
|
|
|
}
|