2019-12-13 05:12:36 -06:00
|
|
|
//! Transforms syntax into `Path` objects, ideally with accounting for hygiene
|
|
|
|
|
|
|
|
mod lower_use;
|
|
|
|
|
2021-05-06 16:23:50 -05:00
|
|
|
use crate::intern::Interned;
|
2019-12-13 05:12:36 -06:00
|
|
|
|
|
|
|
use either::Either;
|
2021-04-10 10:49:12 -05:00
|
|
|
use hir_expand::name::{name, AsName};
|
2021-09-27 05:54:24 -05:00
|
|
|
use syntax::ast::{self, AstNode, HasTypeBounds};
|
2019-12-13 05:12:36 -06:00
|
|
|
|
2020-04-10 15:05:46 -05:00
|
|
|
use super::AssociatedTypeBinding;
|
2019-12-13 05:12:36 -06:00
|
|
|
use crate::{
|
2020-04-30 05:20:13 -05:00
|
|
|
body::LowerCtx,
|
2019-12-13 05:12:36 -06:00
|
|
|
path::{GenericArg, GenericArgs, ModPath, Path, PathKind},
|
2020-12-11 06:49:32 -06:00
|
|
|
type_ref::{LifetimeRef, TypeBound, TypeRef},
|
2019-12-13 05:12:36 -06:00
|
|
|
};
|
|
|
|
|
2021-05-25 18:01:58 -05:00
|
|
|
pub(super) use lower_use::convert_path;
|
2019-12-13 05:12:36 -06:00
|
|
|
|
|
|
|
/// Converts an `ast::Path` to `Path`. Works with use trees.
|
|
|
|
/// It correctly handles `$crate` based path from macro call.
|
2021-05-06 16:23:50 -05:00
|
|
|
pub(super) fn lower_path(mut path: ast::Path, ctx: &LowerCtx) -> Option<Path> {
|
2019-12-13 05:12:36 -06:00
|
|
|
let mut kind = PathKind::Plain;
|
2019-12-18 10:41:33 -06:00
|
|
|
let mut type_anchor = None;
|
2019-12-13 05:12:36 -06:00
|
|
|
let mut segments = Vec::new();
|
|
|
|
let mut generic_args = Vec::new();
|
2021-04-10 10:49:12 -05:00
|
|
|
let hygiene = ctx.hygiene();
|
2019-12-13 05:12:36 -06:00
|
|
|
loop {
|
|
|
|
let segment = path.segment()?;
|
|
|
|
|
2020-04-09 11:25:36 -05:00
|
|
|
if segment.coloncolon_token().is_some() {
|
2019-12-13 05:12:36 -06:00
|
|
|
kind = PathKind::Abs;
|
|
|
|
}
|
|
|
|
|
|
|
|
match segment.kind()? {
|
|
|
|
ast::PathSegmentKind::Name(name_ref) => {
|
|
|
|
// FIXME: this should just return name
|
2021-05-06 16:23:50 -05:00
|
|
|
match hygiene.name_ref_to_name(ctx.db.upcast(), name_ref) {
|
2019-12-13 05:12:36 -06:00
|
|
|
Either::Left(name) => {
|
|
|
|
let args = segment
|
2020-07-31 11:29:29 -05:00
|
|
|
.generic_arg_list()
|
2021-04-10 10:49:12 -05:00
|
|
|
.and_then(|it| lower_generic_args(ctx, it))
|
2019-12-13 05:12:36 -06:00
|
|
|
.or_else(|| {
|
|
|
|
lower_generic_args_from_fn_path(
|
2021-04-10 10:49:12 -05:00
|
|
|
ctx,
|
2019-12-13 05:12:36 -06:00
|
|
|
segment.param_list(),
|
|
|
|
segment.ret_type(),
|
|
|
|
)
|
|
|
|
})
|
2021-05-24 08:35:46 -05:00
|
|
|
.map(Interned::new);
|
2019-12-13 05:12:36 -06:00
|
|
|
segments.push(name);
|
|
|
|
generic_args.push(args)
|
|
|
|
}
|
|
|
|
Either::Right(crate_id) => {
|
|
|
|
kind = PathKind::DollarCrate(crate_id);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
ast::PathSegmentKind::Type { type_ref, trait_ref } => {
|
|
|
|
assert!(path.qualifier().is_none()); // this can only occur at the first segment
|
|
|
|
|
2021-04-10 10:49:12 -05:00
|
|
|
let self_type = TypeRef::from_ast(ctx, type_ref?);
|
2019-12-13 05:12:36 -06:00
|
|
|
|
|
|
|
match trait_ref {
|
|
|
|
// <T>::foo
|
|
|
|
None => {
|
2021-04-04 19:03:37 -05:00
|
|
|
type_anchor = Some(Interned::new(self_type));
|
2019-12-18 10:41:33 -06:00
|
|
|
kind = PathKind::Plain;
|
2019-12-13 05:12:36 -06:00
|
|
|
}
|
|
|
|
// <T as Trait<A>>::Foo desugars to Trait<Self=T, A>::Foo
|
|
|
|
Some(trait_ref) => {
|
2021-11-20 09:37:41 -06:00
|
|
|
let Path { mod_path, generic_args: path_generic_args, .. } =
|
|
|
|
Path::from_src(trait_ref.path()?, ctx)?;
|
|
|
|
let num_segments = mod_path.segments.len();
|
2021-04-01 13:35:21 -05:00
|
|
|
kind = mod_path.kind;
|
2019-12-13 05:12:36 -06:00
|
|
|
|
2021-11-20 09:37:41 -06:00
|
|
|
segments.extend(mod_path.segments.iter().cloned().rev());
|
|
|
|
generic_args.extend(path_generic_args.iter().cloned().rev());
|
2019-12-13 05:12:36 -06:00
|
|
|
|
|
|
|
// Insert the type reference (T in the above example) as Self parameter for the trait
|
2021-03-30 14:43:23 -05:00
|
|
|
let last_segment =
|
|
|
|
generic_args.iter_mut().rev().nth(num_segments.saturating_sub(1))?;
|
2021-05-24 08:35:46 -05:00
|
|
|
let mut args_inner = match last_segment {
|
|
|
|
Some(it) => it.as_ref().clone(),
|
|
|
|
None => GenericArgs::empty(),
|
2019-12-13 05:12:36 -06:00
|
|
|
};
|
|
|
|
args_inner.has_self_type = true;
|
|
|
|
args_inner.args.insert(0, GenericArg::Type(self_type));
|
2021-05-24 08:35:46 -05:00
|
|
|
*last_segment = Some(Interned::new(args_inner));
|
2019-12-13 05:12:36 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
ast::PathSegmentKind::CrateKw => {
|
|
|
|
kind = PathKind::Crate;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
ast::PathSegmentKind::SelfKw => {
|
2021-08-22 10:35:52 -05:00
|
|
|
// don't break out if `self` is the last segment of a path, this mean we got a
|
2021-02-24 16:37:08 -06:00
|
|
|
// use tree like `foo::{self}` which we want to resolve as `foo`
|
|
|
|
if !segments.is_empty() {
|
|
|
|
kind = PathKind::Super(0);
|
|
|
|
break;
|
|
|
|
}
|
2019-12-13 05:12:36 -06:00
|
|
|
}
|
|
|
|
ast::PathSegmentKind::SuperKw => {
|
2020-03-02 07:28:34 -06:00
|
|
|
let nested_super_count = if let PathKind::Super(n) = kind { n } else { 0 };
|
2020-02-29 22:53:01 -06:00
|
|
|
kind = PathKind::Super(nested_super_count + 1);
|
2019-12-13 05:12:36 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
path = match qualifier(&path) {
|
|
|
|
Some(it) => it,
|
|
|
|
None => break,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
segments.reverse();
|
|
|
|
generic_args.reverse();
|
2020-04-30 22:23:03 -05:00
|
|
|
|
2021-02-24 16:37:08 -06:00
|
|
|
if segments.is_empty() && kind == PathKind::Plain && type_anchor.is_none() {
|
|
|
|
// plain empty paths don't exist, this means we got a single `self` segment as our path
|
|
|
|
kind = PathKind::Super(0);
|
|
|
|
}
|
|
|
|
|
2020-04-30 22:23:03 -05:00
|
|
|
// handle local_inner_macros :
|
|
|
|
// Basically, even in rustc it is quite hacky:
|
|
|
|
// https://github.com/rust-lang/rust/blob/614f273e9388ddd7804d5cbc80b8865068a3744e/src/librustc_resolve/macros.rs#L456
|
|
|
|
// We follow what it did anyway :)
|
|
|
|
if segments.len() == 1 && kind == PathKind::Plain {
|
2020-12-15 08:37:37 -06:00
|
|
|
if let Some(_macro_call) = path.syntax().parent().and_then(ast::MacroCall::cast) {
|
2021-05-06 16:23:50 -05:00
|
|
|
if let Some(crate_id) = hygiene.local_inner_macros(ctx.db.upcast(), path) {
|
2020-12-15 08:37:37 -06:00
|
|
|
kind = PathKind::DollarCrate(crate_id);
|
2020-04-30 22:23:03 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-04-01 13:35:21 -05:00
|
|
|
let mod_path = Interned::new(ModPath::from_segments(kind, segments));
|
2021-11-20 09:37:41 -06:00
|
|
|
return Some(Path { type_anchor, mod_path, generic_args: generic_args.into() });
|
2019-12-13 05:12:36 -06:00
|
|
|
|
|
|
|
fn qualifier(path: &ast::Path) -> Option<ast::Path> {
|
|
|
|
if let Some(q) = path.qualifier() {
|
|
|
|
return Some(q);
|
|
|
|
}
|
|
|
|
// FIXME: this bottom up traversal is not too precise.
|
|
|
|
// Should we handle do a top-down analysis, recording results?
|
|
|
|
let use_tree_list = path.syntax().ancestors().find_map(ast::UseTreeList::cast)?;
|
|
|
|
let use_tree = use_tree_list.parent_use_tree();
|
|
|
|
use_tree.path()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-04-30 05:20:13 -05:00
|
|
|
pub(super) fn lower_generic_args(
|
|
|
|
lower_ctx: &LowerCtx,
|
2020-07-31 11:29:29 -05:00
|
|
|
node: ast::GenericArgList,
|
2020-04-30 05:20:13 -05:00
|
|
|
) -> Option<GenericArgs> {
|
2019-12-13 05:12:36 -06:00
|
|
|
let mut args = Vec::new();
|
|
|
|
let mut bindings = Vec::new();
|
2020-07-31 11:41:37 -05:00
|
|
|
for generic_arg in node.generic_args() {
|
|
|
|
match generic_arg {
|
|
|
|
ast::GenericArg::TypeArg(type_arg) => {
|
|
|
|
let type_ref = TypeRef::from_ast_opt(lower_ctx, type_arg.ty());
|
|
|
|
args.push(GenericArg::Type(type_ref));
|
|
|
|
}
|
|
|
|
ast::GenericArg::AssocTypeArg(assoc_type_arg) => {
|
|
|
|
if let Some(name_ref) = assoc_type_arg.name_ref() {
|
|
|
|
let name = name_ref.as_name();
|
|
|
|
let type_ref = assoc_type_arg.ty().map(|it| TypeRef::from_ast(lower_ctx, it));
|
|
|
|
let bounds = if let Some(l) = assoc_type_arg.type_bound_list() {
|
2021-05-24 08:13:23 -05:00
|
|
|
l.bounds()
|
|
|
|
.map(|it| Interned::new(TypeBound::from_ast(lower_ctx, it)))
|
|
|
|
.collect()
|
2020-07-31 11:41:37 -05:00
|
|
|
} else {
|
|
|
|
Vec::new()
|
|
|
|
};
|
|
|
|
bindings.push(AssociatedTypeBinding { name, type_ref, bounds });
|
|
|
|
}
|
|
|
|
}
|
2020-12-11 06:49:32 -06:00
|
|
|
ast::GenericArg::LifetimeArg(lifetime_arg) => {
|
2020-12-15 12:23:51 -06:00
|
|
|
if let Some(lifetime) = lifetime_arg.lifetime() {
|
|
|
|
let lifetime_ref = LifetimeRef::new(&lifetime);
|
2020-12-11 06:49:32 -06:00
|
|
|
args.push(GenericArg::Lifetime(lifetime_ref))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// constants are ignored for now.
|
|
|
|
ast::GenericArg::ConstArg(_) => (),
|
2019-12-13 05:12:36 -06:00
|
|
|
}
|
|
|
|
}
|
2020-07-31 11:41:37 -05:00
|
|
|
|
2019-12-13 05:12:36 -06:00
|
|
|
if args.is_empty() && bindings.is_empty() {
|
2020-07-31 11:41:37 -05:00
|
|
|
return None;
|
2019-12-13 05:12:36 -06:00
|
|
|
}
|
2021-11-19 12:58:00 -06:00
|
|
|
Some(GenericArgs { args, has_self_type: false, bindings, desugared_from_fn: false })
|
2019-12-13 05:12:36 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Collect `GenericArgs` from the parts of a fn-like path, i.e. `Fn(X, Y)
|
|
|
|
/// -> Z` (which desugars to `Fn<(X, Y), Output=Z>`).
|
|
|
|
fn lower_generic_args_from_fn_path(
|
2020-04-30 05:20:13 -05:00
|
|
|
ctx: &LowerCtx,
|
2019-12-13 05:12:36 -06:00
|
|
|
params: Option<ast::ParamList>,
|
|
|
|
ret_type: Option<ast::RetType>,
|
|
|
|
) -> Option<GenericArgs> {
|
|
|
|
let mut args = Vec::new();
|
|
|
|
let mut bindings = Vec::new();
|
2021-05-25 07:24:08 -05:00
|
|
|
let params = params?;
|
|
|
|
let mut param_types = Vec::new();
|
|
|
|
for param in params.params() {
|
2021-06-12 22:54:16 -05:00
|
|
|
let type_ref = TypeRef::from_ast_opt(ctx, param.ty());
|
2021-05-25 07:24:08 -05:00
|
|
|
param_types.push(type_ref);
|
2019-12-13 05:12:36 -06:00
|
|
|
}
|
2021-05-25 07:24:08 -05:00
|
|
|
let arg = GenericArg::Type(TypeRef::Tuple(param_types));
|
|
|
|
args.push(arg);
|
2019-12-13 05:12:36 -06:00
|
|
|
if let Some(ret_type) = ret_type {
|
2021-06-12 22:54:16 -05:00
|
|
|
let type_ref = TypeRef::from_ast_opt(ctx, ret_type.ty());
|
2020-04-10 15:05:46 -05:00
|
|
|
bindings.push(AssociatedTypeBinding {
|
|
|
|
name: name![Output],
|
|
|
|
type_ref: Some(type_ref),
|
|
|
|
bounds: Vec::new(),
|
|
|
|
});
|
2019-12-13 05:12:36 -06:00
|
|
|
} else {
|
2021-05-25 07:24:08 -05:00
|
|
|
// -> ()
|
|
|
|
let type_ref = TypeRef::Tuple(Vec::new());
|
|
|
|
bindings.push(AssociatedTypeBinding {
|
|
|
|
name: name![Output],
|
|
|
|
type_ref: Some(type_ref),
|
|
|
|
bounds: Vec::new(),
|
|
|
|
});
|
2019-12-13 05:12:36 -06:00
|
|
|
}
|
2021-11-19 12:58:00 -06:00
|
|
|
Some(GenericArgs { args, has_self_type: false, bindings, desugared_from_fn: true })
|
2019-12-13 05:12:36 -06:00
|
|
|
}
|