Move NameRef classification logic out of reference_definition

This commit is contained in:
Laurențiu Nicola 2019-05-22 19:49:22 +03:00
parent a25e103e45
commit 444e52e519
5 changed files with 144 additions and 97 deletions

1
Cargo.lock generated
View file

@ -1089,6 +1089,7 @@ version = "0.1.0"
dependencies = [ dependencies = [
"parking_lot 0.7.1 (registry+https://github.com/rust-lang/crates.io-index)", "parking_lot 0.7.1 (registry+https://github.com/rust-lang/crates.io-index)",
"ra_arena 0.1.0", "ra_arena 0.1.0",
"ra_prof 0.1.0",
"ra_syntax 0.1.0", "ra_syntax 0.1.0",
"relative-path 0.4.0 (registry+https://github.com/rust-lang/crates.io-index)", "relative-path 0.4.0 (registry+https://github.com/rust-lang/crates.io-index)",
"rustc-hash 1.0.1 (registry+https://github.com/rust-lang/crates.io-index)", "rustc-hash 1.0.1 (registry+https://github.com/rust-lang/crates.io-index)",

View file

@ -5,7 +5,7 @@ use ra_syntax::{
ast::{self, NameOwner, VisibilityOwner, TypeAscriptionOwner}, ast::{self, NameOwner, VisibilityOwner, TypeAscriptionOwner},
algo::visit::{visitor, Visitor}, algo::visit::{visitor, Visitor},
}; };
use hir::{ModuleSource, FieldSource, ImplItem, Either}; use hir::{ModuleSource, FieldSource, ImplItem};
use crate::{FileSymbol, db::RootDatabase}; use crate::{FileSymbol, db::RootDatabase};
@ -77,17 +77,12 @@ impl NavigationTarget {
pub(crate) fn from_pat( pub(crate) fn from_pat(
db: &RootDatabase, db: &RootDatabase,
file_id: FileId, file_id: FileId,
pat: Either<AstPtr<ast::Pat>, AstPtr<ast::SelfParam>>, pat: AstPtr<ast::Pat>,
) -> NavigationTarget { ) -> NavigationTarget {
let file = db.parse(file_id); let file = db.parse(file_id);
let (name, full_range) = match pat { let (name, full_range) = match pat.to_node(file.syntax()).kind() {
Either::A(pat) => match pat.to_node(file.syntax()).kind() { ast::PatKind::BindPat(pat) => return NavigationTarget::from_bind_pat(file_id, &pat),
ast::PatKind::BindPat(pat) => {
return NavigationTarget::from_bind_pat(file_id, &pat)
}
_ => ("_".into(), pat.syntax_node_ptr().range()), _ => ("_".into(), pat.syntax_node_ptr().range()),
},
Either::B(slf) => ("self".into(), slf.syntax_node_ptr().range()),
}; };
NavigationTarget { NavigationTarget {
file_id, file_id,
@ -99,6 +94,21 @@ impl NavigationTarget {
} }
} }
pub(crate) fn from_self_param(
file_id: FileId,
par: AstPtr<ast::SelfParam>,
) -> NavigationTarget {
let (name, full_range) = ("self".into(), par.syntax_node_ptr().range());
NavigationTarget {
file_id,
name,
full_range,
focus_range: None,
kind: NAME,
container_name: None,
}
}
pub(crate) fn from_module(db: &RootDatabase, module: hir::Module) -> NavigationTarget { pub(crate) fn from_module(db: &RootDatabase, module: hir::Module) -> NavigationTarget {
let (file_id, source) = module.definition_source(db); let (file_id, source) = module.definition_source(db);
let file_id = file_id.as_original_file(); let file_id = file_id.as_original_file();

View file

@ -1,12 +1,19 @@
use ra_db::{FileId, SourceDatabase}; use ra_db::{FileId, SourceDatabase};
use ra_syntax::{ use ra_syntax::{
AstNode, ast, AstNode, ast,
algo::{find_node_at_offset, visit::{visitor, Visitor}}, algo::{
find_node_at_offset,
visit::{visitor, Visitor},
},
SyntaxNode, SyntaxNode,
}; };
use test_utils::tested_by;
use crate::{FilePosition, NavigationTarget, db::RootDatabase, RangeInfo}; use crate::{
FilePosition, NavigationTarget,
db::RootDatabase,
RangeInfo,
name_ref_kind::{NameRefKind::*, classify_name_ref},
};
pub(crate) fn goto_definition( pub(crate) fn goto_definition(
db: &RootDatabase, db: &RootDatabase,
@ -50,85 +57,24 @@ pub(crate) fn reference_definition(
let analyzer = hir::SourceAnalyzer::new(db, file_id, name_ref.syntax(), None); let analyzer = hir::SourceAnalyzer::new(db, file_id, name_ref.syntax(), None);
// Special cases: match classify_name_ref(db, &analyzer, name_ref) {
Some(Method(func)) => return Exact(NavigationTarget::from_function(db, func)),
// Check if it is a method Some(Macro(mac)) => return Exact(NavigationTarget::from_macro_def(db, mac)),
if let Some(method_call) = name_ref.syntax().parent().and_then(ast::MethodCallExpr::cast) { Some(FieldAccess(field)) => return Exact(NavigationTarget::from_field(db, field)),
tested_by!(goto_definition_works_for_methods); Some(AssocItem(assoc)) => return Exact(NavigationTarget::from_impl_item(db, assoc)),
if let Some(func) = analyzer.resolve_method_call(method_call) { Some(Def(def)) => return Exact(NavigationTarget::from_def(db, def)),
return Exact(NavigationTarget::from_function(db, func)); Some(SelfType(ty)) => {
}
}
//it could be a macro call
if let Some(macro_call) = name_ref
.syntax()
.parent()
.and_then(|node| node.parent())
.and_then(|node| node.parent())
.and_then(ast::MacroCall::cast)
{
tested_by!(goto_definition_works_for_macros);
if let Some(macro_call) = analyzer.resolve_macro_call(macro_call) {
return Exact(NavigationTarget::from_macro_def(db, macro_call));
}
}
// It could also be a field access
if let Some(field_expr) = name_ref.syntax().parent().and_then(ast::FieldExpr::cast) {
tested_by!(goto_definition_works_for_fields);
if let Some(field) = analyzer.resolve_field(field_expr) {
return Exact(NavigationTarget::from_field(db, field));
};
}
// It could also be a named field
if let Some(field_expr) = name_ref.syntax().parent().and_then(ast::NamedField::cast) {
tested_by!(goto_definition_works_for_named_fields);
let struct_lit = field_expr.syntax().ancestors().find_map(ast::StructLit::cast);
if let Some(ty) = struct_lit.and_then(|lit| analyzer.type_of(db, lit.into())) {
if let Some((hir::AdtDef::Struct(s), _)) = ty.as_adt() {
let hir_path = hir::Path::from_name_ref(name_ref);
let hir_name = hir_path.as_ident().unwrap();
if let Some(field) = s.field(db, hir_name) {
return Exact(NavigationTarget::from_field(db, field));
}
}
}
}
// General case, a path or a local:
if let Some(path) = name_ref.syntax().ancestors().find_map(ast::Path::cast) {
if let Some(resolved) = analyzer.resolve_path(db, path) {
match resolved {
hir::PathResolution::Def(def) => return Exact(NavigationTarget::from_def(db, def)),
hir::PathResolution::LocalBinding(pat) => {
let nav = NavigationTarget::from_pat(db, file_id, pat);
return Exact(nav);
}
hir::PathResolution::GenericParam(..) => {
// FIXME: go to the generic param def
}
hir::PathResolution::Macro(def) => {
let nav = NavigationTarget::from_macro_def(db, def);
return Exact(nav);
}
hir::PathResolution::SelfType(impl_block) => {
let ty = impl_block.target_ty(db);
if let Some((def_id, _)) = ty.as_adt() { if let Some((def_id, _)) = ty.as_adt() {
return Exact(NavigationTarget::from_adt_def(db, def_id)); return Exact(NavigationTarget::from_adt_def(db, def_id));
} }
} }
hir::PathResolution::AssocItem(assoc) => { Some(Pat(pat)) => return Exact(NavigationTarget::from_pat(db, file_id, pat)),
return Exact(NavigationTarget::from_impl_item(db, assoc)); Some(SelfParam(par)) => return Exact(NavigationTarget::from_self_param(file_id, par)),
} Some(GenericParam(_)) => {
} // FIXME: go to the generic param def
}
} }
None => {}
};
// Fallback index based approach: // Fallback index based approach:
let navs = crate::symbol_index::index_resolve(db, name_ref) let navs = crate::symbol_index::index_resolve(db, name_ref)

View file

@ -18,6 +18,7 @@ mod change;
mod status; mod status;
mod completion; mod completion;
mod runnables; mod runnables;
mod name_ref_kind;
mod goto_definition; mod goto_definition;
mod goto_type_definition; mod goto_type_definition;
mod extend_selection; mod extend_selection;
@ -53,10 +54,7 @@ use ra_db::{
}; };
use relative_path::RelativePathBuf; use relative_path::RelativePathBuf;
use crate::{ use crate::{symbol_index::FileSymbol, db::LineIndexDatabase};
symbol_index::FileSymbol,
db::LineIndexDatabase,
};
pub use crate::{ pub use crate::{
change::{AnalysisChange, LibraryData}, change::{AnalysisChange, LibraryData},
@ -73,10 +71,7 @@ pub use crate::{
display::{FunctionSignature, NavigationTarget, StructureNode, file_structure}, display::{FunctionSignature, NavigationTarget, StructureNode, file_structure},
}; };
pub use ra_db::{ pub use ra_db::{Canceled, CrateGraph, CrateId, FileId, FilePosition, FileRange, SourceRootId, Edition};
Canceled, CrateGraph, CrateId, FileId, FilePosition, FileRange, SourceRootId,
Edition
};
pub use hir::Documentation; pub use hir::Documentation;
// We use jemalloc mainly to get heap usage statistics, actual performance // We use jemalloc mainly to get heap usage statistics, actual performance

View file

@ -0,0 +1,95 @@
use ra_syntax::{AstNode, AstPtr, ast};
use hir::Either;
use crate::db::RootDatabase;
use test_utils::tested_by;
pub enum NameRefKind {
Method(hir::Function),
Macro(hir::MacroByExampleDef),
FieldAccess(hir::StructField),
AssocItem(hir::ImplItem),
Def(hir::ModuleDef),
SelfType(hir::Ty),
Pat(AstPtr<ast::Pat>),
SelfParam(AstPtr<ast::SelfParam>),
GenericParam(u32),
}
pub(crate) fn classify_name_ref(
db: &RootDatabase,
analyzer: &hir::SourceAnalyzer,
name_ref: &ast::NameRef,
) -> Option<NameRefKind> {
use NameRefKind::*;
// Check if it is a method
if let Some(method_call) = name_ref.syntax().parent().and_then(ast::MethodCallExpr::cast) {
tested_by!(goto_definition_works_for_methods);
if let Some(func) = analyzer.resolve_method_call(method_call) {
return Some(Method(func));
}
}
// It could be a macro call
if let Some(macro_call) = name_ref
.syntax()
.parent()
.and_then(|node| node.parent())
.and_then(|node| node.parent())
.and_then(ast::MacroCall::cast)
{
tested_by!(goto_definition_works_for_macros);
if let Some(mac) = analyzer.resolve_macro_call(macro_call) {
return Some(Macro(mac));
}
}
// It could also be a field access
if let Some(field_expr) = name_ref.syntax().parent().and_then(ast::FieldExpr::cast) {
tested_by!(goto_definition_works_for_fields);
if let Some(field) = analyzer.resolve_field(field_expr) {
return Some(FieldAccess(field));
};
}
// It could also be a named field
if let Some(field_expr) = name_ref.syntax().parent().and_then(ast::NamedField::cast) {
tested_by!(goto_definition_works_for_named_fields);
let struct_lit = field_expr.syntax().ancestors().find_map(ast::StructLit::cast);
if let Some(ty) = struct_lit.and_then(|lit| analyzer.type_of(db, lit.into())) {
if let Some((hir::AdtDef::Struct(s), _)) = ty.as_adt() {
let hir_path = hir::Path::from_name_ref(name_ref);
let hir_name = hir_path.as_ident().unwrap();
if let Some(field) = s.field(db, hir_name) {
return Some(FieldAccess(field));
}
}
}
}
// General case, a path or a local:
if let Some(path) = name_ref.syntax().ancestors().find_map(ast::Path::cast) {
if let Some(resolved) = analyzer.resolve_path(db, path) {
return match resolved {
hir::PathResolution::Def(def) => Some(Def(def)),
hir::PathResolution::LocalBinding(Either::A(pat)) => Some(Pat(pat)),
hir::PathResolution::LocalBinding(Either::B(par)) => Some(SelfParam(par)),
hir::PathResolution::GenericParam(par) => {
// FIXME: get generic param def
Some(GenericParam(par))
}
hir::PathResolution::Macro(def) => Some(Macro(def)),
hir::PathResolution::SelfType(impl_block) => {
let ty = impl_block.target_ty(db);
Some(SelfType(ty))
}
hir::PathResolution::AssocItem(assoc) => Some(AssocItem(assoc)),
};
}
}
None
}