mirror of
https://github.com/rust-lang/rust-analyzer
synced 2025-01-28 04:45:05 +00:00
Auto merge of #17268 - Veykril:signatures, r=Veykril
feat: More callable info With this PR we retain more info about callables other than functions, allowing for closure parameter type inlay hints to be linkable as well as better signature help around closures and `Fn*` implementors.
This commit is contained in:
commit
daf66ad8eb
17 changed files with 387 additions and 225 deletions
|
@ -12,9 +12,6 @@ use std::sync::OnceLock;
|
||||||
|
|
||||||
use rustc_hash::FxHashMap;
|
use rustc_hash::FxHashMap;
|
||||||
|
|
||||||
/// Ignored attribute namespaces used by tools.
|
|
||||||
pub const TOOL_MODULES: &[&str] = &["rustfmt", "clippy"];
|
|
||||||
|
|
||||||
pub struct BuiltinAttribute {
|
pub struct BuiltinAttribute {
|
||||||
pub name: &'static str,
|
pub name: &'static str,
|
||||||
pub template: AttributeTemplate,
|
pub template: AttributeTemplate,
|
||||||
|
|
|
@ -84,6 +84,14 @@ use crate::{
|
||||||
LocalModuleId, Lookup, MacroExpander, MacroId, ModuleId, ProcMacroId, UseId,
|
LocalModuleId, Lookup, MacroExpander, MacroId, ModuleId, ProcMacroId, UseId,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const PREDEFINED_TOOLS: &[SmolStr] = &[
|
||||||
|
SmolStr::new_static("clippy"),
|
||||||
|
SmolStr::new_static("rustfmt"),
|
||||||
|
SmolStr::new_static("diagnostic"),
|
||||||
|
SmolStr::new_static("miri"),
|
||||||
|
SmolStr::new_static("rust_analyzer"),
|
||||||
|
];
|
||||||
|
|
||||||
/// Contains the results of (early) name resolution.
|
/// Contains the results of (early) name resolution.
|
||||||
///
|
///
|
||||||
/// A `DefMap` stores the module tree and the definitions that are in scope in every module after
|
/// A `DefMap` stores the module tree and the definitions that are in scope in every module after
|
||||||
|
@ -160,7 +168,7 @@ impl DefMapCrateData {
|
||||||
fn_proc_macro_mapping: FxHashMap::default(),
|
fn_proc_macro_mapping: FxHashMap::default(),
|
||||||
proc_macro_loading_error: None,
|
proc_macro_loading_error: None,
|
||||||
registered_attrs: Vec::new(),
|
registered_attrs: Vec::new(),
|
||||||
registered_tools: Vec::new(),
|
registered_tools: PREDEFINED_TOOLS.into(),
|
||||||
unstable_features: FxHashSet::default(),
|
unstable_features: FxHashSet::default(),
|
||||||
rustc_coherence_is_core: false,
|
rustc_coherence_is_core: false,
|
||||||
no_core: false,
|
no_core: false,
|
||||||
|
|
|
@ -10,7 +10,7 @@ use syntax::{ast, SmolStr};
|
||||||
use triomphe::Arc;
|
use triomphe::Arc;
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
attr::builtin::{find_builtin_attr_idx, TOOL_MODULES},
|
attr::builtin::find_builtin_attr_idx,
|
||||||
db::DefDatabase,
|
db::DefDatabase,
|
||||||
item_scope::BuiltinShadowMode,
|
item_scope::BuiltinShadowMode,
|
||||||
nameres::path_resolution::ResolveMode,
|
nameres::path_resolution::ResolveMode,
|
||||||
|
@ -82,8 +82,7 @@ impl DefMap {
|
||||||
let name = name.to_smol_str();
|
let name = name.to_smol_str();
|
||||||
let pred = |n: &_| *n == name;
|
let pred = |n: &_| *n == name;
|
||||||
|
|
||||||
let registered = self.data.registered_tools.iter().map(SmolStr::as_str);
|
let is_tool = self.data.registered_tools.iter().map(SmolStr::as_str).any(pred);
|
||||||
let is_tool = TOOL_MODULES.iter().copied().chain(registered).any(pred);
|
|
||||||
// FIXME: tool modules can be shadowed by actual modules
|
// FIXME: tool modules can be shadowed by actual modules
|
||||||
if is_tool {
|
if is_tool {
|
||||||
return true;
|
return true;
|
||||||
|
|
|
@ -797,19 +797,22 @@ impl<'a> InferenceTable<'a> {
|
||||||
})
|
})
|
||||||
.build();
|
.build();
|
||||||
|
|
||||||
let projection = {
|
let b = TyBuilder::trait_ref(self.db, fn_once_trait);
|
||||||
let b = TyBuilder::subst_for_def(self.db, fn_once_trait, None);
|
|
||||||
if b.remaining() != 2 {
|
if b.remaining() != 2 {
|
||||||
return None;
|
return None;
|
||||||
}
|
}
|
||||||
let fn_once_subst = b.push(ty.clone()).push(arg_ty).build();
|
let mut trait_ref = b.push(ty.clone()).push(arg_ty).build();
|
||||||
|
|
||||||
TyBuilder::assoc_type_projection(self.db, output_assoc_type, Some(fn_once_subst))
|
let projection = {
|
||||||
|
TyBuilder::assoc_type_projection(
|
||||||
|
self.db,
|
||||||
|
output_assoc_type,
|
||||||
|
Some(trait_ref.substitution.clone()),
|
||||||
|
)
|
||||||
.build()
|
.build()
|
||||||
};
|
};
|
||||||
|
|
||||||
let trait_env = self.trait_env.env.clone();
|
let trait_env = self.trait_env.env.clone();
|
||||||
let mut trait_ref = projection.trait_ref(self.db);
|
|
||||||
let obligation = InEnvironment {
|
let obligation = InEnvironment {
|
||||||
goal: trait_ref.clone().cast(Interner),
|
goal: trait_ref.clone().cast(Interner),
|
||||||
environment: trait_env.clone(),
|
environment: trait_env.clone(),
|
||||||
|
|
|
@ -570,6 +570,10 @@ impl CallableSig {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn abi(&self) -> FnAbi {
|
||||||
|
self.abi
|
||||||
|
}
|
||||||
|
|
||||||
pub fn params(&self) -> &[Ty] {
|
pub fn params(&self) -> &[Ty] {
|
||||||
&self.params_and_return[0..self.params_and_return.len() - 1]
|
&self.params_and_return[0..self.params_and_return.len() - 1]
|
||||||
}
|
}
|
||||||
|
@ -892,20 +896,16 @@ where
|
||||||
Canonical { value, binders: chalk_ir::CanonicalVarKinds::from_iter(Interner, kinds) }
|
Canonical { value, binders: chalk_ir::CanonicalVarKinds::from_iter(Interner, kinds) }
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn callable_sig_from_fnonce(
|
pub fn callable_sig_from_fn_trait(
|
||||||
mut self_ty: &Ty,
|
self_ty: &Ty,
|
||||||
env: Arc<TraitEnvironment>,
|
trait_env: Arc<TraitEnvironment>,
|
||||||
db: &dyn HirDatabase,
|
db: &dyn HirDatabase,
|
||||||
) -> Option<CallableSig> {
|
) -> Option<(FnTrait, CallableSig)> {
|
||||||
if let Some((ty, _, _)) = self_ty.as_reference() {
|
let krate = trait_env.krate;
|
||||||
// This will happen when it implements fn or fn mut, since we add a autoborrow adjustment
|
|
||||||
self_ty = ty;
|
|
||||||
}
|
|
||||||
let krate = env.krate;
|
|
||||||
let fn_once_trait = FnTrait::FnOnce.get_id(db, krate)?;
|
let fn_once_trait = FnTrait::FnOnce.get_id(db, krate)?;
|
||||||
let output_assoc_type = db.trait_data(fn_once_trait).associated_type_by_name(&name![Output])?;
|
let output_assoc_type = db.trait_data(fn_once_trait).associated_type_by_name(&name![Output])?;
|
||||||
|
|
||||||
let mut table = InferenceTable::new(db, env);
|
let mut table = InferenceTable::new(db, trait_env.clone());
|
||||||
let b = TyBuilder::trait_ref(db, fn_once_trait);
|
let b = TyBuilder::trait_ref(db, fn_once_trait);
|
||||||
if b.remaining() != 2 {
|
if b.remaining() != 2 {
|
||||||
return None;
|
return None;
|
||||||
|
@ -915,23 +915,56 @@ pub fn callable_sig_from_fnonce(
|
||||||
// - Self: FnOnce<?args_ty>
|
// - Self: FnOnce<?args_ty>
|
||||||
// - <Self as FnOnce<?args_ty>>::Output == ?ret_ty
|
// - <Self as FnOnce<?args_ty>>::Output == ?ret_ty
|
||||||
let args_ty = table.new_type_var();
|
let args_ty = table.new_type_var();
|
||||||
let trait_ref = b.push(self_ty.clone()).push(args_ty.clone()).build();
|
let mut trait_ref = b.push(self_ty.clone()).push(args_ty.clone()).build();
|
||||||
let projection = TyBuilder::assoc_type_projection(
|
let projection = TyBuilder::assoc_type_projection(
|
||||||
db,
|
db,
|
||||||
output_assoc_type,
|
output_assoc_type,
|
||||||
Some(trait_ref.substitution.clone()),
|
Some(trait_ref.substitution.clone()),
|
||||||
)
|
)
|
||||||
.build();
|
.build();
|
||||||
table.register_obligation(trait_ref.cast(Interner));
|
|
||||||
let ret_ty = table.normalize_projection_ty(projection);
|
|
||||||
|
|
||||||
let ret_ty = table.resolve_completely(ret_ty);
|
let block = trait_env.block;
|
||||||
|
let trait_env = trait_env.env.clone();
|
||||||
|
let obligation =
|
||||||
|
InEnvironment { goal: trait_ref.clone().cast(Interner), environment: trait_env.clone() };
|
||||||
|
let canonical = table.canonicalize(obligation.clone());
|
||||||
|
if db.trait_solve(krate, block, canonical.cast(Interner)).is_some() {
|
||||||
|
table.register_obligation(obligation.goal);
|
||||||
|
let return_ty = table.normalize_projection_ty(projection);
|
||||||
|
for fn_x in [FnTrait::Fn, FnTrait::FnMut, FnTrait::FnOnce] {
|
||||||
|
let fn_x_trait = fn_x.get_id(db, krate)?;
|
||||||
|
trait_ref.trait_id = to_chalk_trait_id(fn_x_trait);
|
||||||
|
let obligation: chalk_ir::InEnvironment<chalk_ir::Goal<Interner>> = InEnvironment {
|
||||||
|
goal: trait_ref.clone().cast(Interner),
|
||||||
|
environment: trait_env.clone(),
|
||||||
|
};
|
||||||
|
let canonical = table.canonicalize(obligation.clone());
|
||||||
|
if db.trait_solve(krate, block, canonical.cast(Interner)).is_some() {
|
||||||
|
let ret_ty = table.resolve_completely(return_ty);
|
||||||
let args_ty = table.resolve_completely(args_ty);
|
let args_ty = table.resolve_completely(args_ty);
|
||||||
|
let params = args_ty
|
||||||
|
.as_tuple()?
|
||||||
|
.iter(Interner)
|
||||||
|
.map(|it| it.assert_ty_ref(Interner))
|
||||||
|
.cloned()
|
||||||
|
.collect();
|
||||||
|
|
||||||
let params =
|
return Some((
|
||||||
args_ty.as_tuple()?.iter(Interner).map(|it| it.assert_ty_ref(Interner)).cloned().collect();
|
fn_x,
|
||||||
|
CallableSig::from_params_and_return(
|
||||||
Some(CallableSig::from_params_and_return(params, ret_ty, false, Safety::Safe, FnAbi::RustCall))
|
params,
|
||||||
|
ret_ty,
|
||||||
|
false,
|
||||||
|
Safety::Safe,
|
||||||
|
FnAbi::RustCall,
|
||||||
|
),
|
||||||
|
));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
unreachable!("It should at least implement FnOnce at this point");
|
||||||
|
} else {
|
||||||
|
None
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
struct PlaceholderCollector<'db> {
|
struct PlaceholderCollector<'db> {
|
||||||
|
|
|
@ -1,5 +1,6 @@
|
||||||
//! Trait solving using Chalk.
|
//! Trait solving using Chalk.
|
||||||
|
|
||||||
|
use core::fmt;
|
||||||
use std::env::var;
|
use std::env::var;
|
||||||
|
|
||||||
use chalk_ir::{fold::TypeFoldable, DebruijnIndex, GoalData};
|
use chalk_ir::{fold::TypeFoldable, DebruijnIndex, GoalData};
|
||||||
|
@ -209,7 +210,25 @@ pub enum FnTrait {
|
||||||
Fn,
|
Fn,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl fmt::Display for FnTrait {
|
||||||
|
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
|
||||||
|
match self {
|
||||||
|
FnTrait::FnOnce => write!(f, "FnOnce"),
|
||||||
|
FnTrait::FnMut => write!(f, "FnMut"),
|
||||||
|
FnTrait::Fn => write!(f, "Fn"),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
impl FnTrait {
|
impl FnTrait {
|
||||||
|
pub const fn function_name(&self) -> &'static str {
|
||||||
|
match self {
|
||||||
|
FnTrait::FnOnce => "call_once",
|
||||||
|
FnTrait::FnMut => "call_mut",
|
||||||
|
FnTrait::Fn => "call",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
const fn lang_item(self) -> LangItem {
|
const fn lang_item(self) -> LangItem {
|
||||||
match self {
|
match self {
|
||||||
FnTrait::FnOnce => LangItem::FnOnce,
|
FnTrait::FnOnce => LangItem::FnOnce,
|
||||||
|
|
|
@ -35,7 +35,7 @@ pub mod term_search;
|
||||||
|
|
||||||
mod display;
|
mod display;
|
||||||
|
|
||||||
use std::{iter, mem::discriminant, ops::ControlFlow};
|
use std::{mem::discriminant, ops::ControlFlow};
|
||||||
|
|
||||||
use arrayvec::ArrayVec;
|
use arrayvec::ArrayVec;
|
||||||
use base_db::{CrateDisplayName, CrateId, CrateOrigin, FileId};
|
use base_db::{CrateDisplayName, CrateId, CrateOrigin, FileId};
|
||||||
|
@ -52,7 +52,6 @@ use hir_def::{
|
||||||
path::ImportAlias,
|
path::ImportAlias,
|
||||||
per_ns::PerNs,
|
per_ns::PerNs,
|
||||||
resolver::{HasResolver, Resolver},
|
resolver::{HasResolver, Resolver},
|
||||||
src::HasSource as _,
|
|
||||||
AssocItemId, AssocItemLoc, AttrDefId, ConstId, ConstParamId, CrateRootModuleId, DefWithBodyId,
|
AssocItemId, AssocItemLoc, AttrDefId, ConstId, ConstParamId, CrateRootModuleId, DefWithBodyId,
|
||||||
EnumId, EnumVariantId, ExternCrateId, FunctionId, GenericDefId, GenericParamId, HasModule,
|
EnumId, EnumVariantId, ExternCrateId, FunctionId, GenericDefId, GenericParamId, HasModule,
|
||||||
ImplId, InTypeConstId, ItemContainerId, LifetimeParamId, LocalFieldId, Lookup, MacroExpander,
|
ImplId, InTypeConstId, ItemContainerId, LifetimeParamId, LocalFieldId, Lookup, MacroExpander,
|
||||||
|
@ -141,7 +140,7 @@ pub use {
|
||||||
display::{ClosureStyle, HirDisplay, HirDisplayError, HirWrite},
|
display::{ClosureStyle, HirDisplay, HirDisplayError, HirWrite},
|
||||||
layout::LayoutError,
|
layout::LayoutError,
|
||||||
mir::{MirEvalError, MirLowerError},
|
mir::{MirEvalError, MirLowerError},
|
||||||
PointerCast, Safety,
|
FnAbi, PointerCast, Safety,
|
||||||
},
|
},
|
||||||
// FIXME: Properly encapsulate mir
|
// FIXME: Properly encapsulate mir
|
||||||
hir_ty::{mir, Interner as ChalkTyInterner},
|
hir_ty::{mir, Interner as ChalkTyInterner},
|
||||||
|
@ -1965,7 +1964,7 @@ impl Function {
|
||||||
.enumerate()
|
.enumerate()
|
||||||
.map(|(idx, ty)| {
|
.map(|(idx, ty)| {
|
||||||
let ty = Type { env: environment.clone(), ty: ty.clone() };
|
let ty = Type { env: environment.clone(), ty: ty.clone() };
|
||||||
Param { func: self, ty, idx }
|
Param { func: Callee::Def(CallableDefId::FunctionId(self.id)), ty, idx }
|
||||||
})
|
})
|
||||||
.collect()
|
.collect()
|
||||||
}
|
}
|
||||||
|
@ -1991,7 +1990,7 @@ impl Function {
|
||||||
.skip(skip)
|
.skip(skip)
|
||||||
.map(|(idx, ty)| {
|
.map(|(idx, ty)| {
|
||||||
let ty = Type { env: environment.clone(), ty: ty.clone() };
|
let ty = Type { env: environment.clone(), ty: ty.clone() };
|
||||||
Param { func: self, ty, idx }
|
Param { func: Callee::Def(CallableDefId::FunctionId(self.id)), ty, idx }
|
||||||
})
|
})
|
||||||
.collect()
|
.collect()
|
||||||
}
|
}
|
||||||
|
@ -2037,7 +2036,7 @@ impl Function {
|
||||||
.skip(skip)
|
.skip(skip)
|
||||||
.map(|(idx, ty)| {
|
.map(|(idx, ty)| {
|
||||||
let ty = Type { env: environment.clone(), ty: ty.clone() };
|
let ty = Type { env: environment.clone(), ty: ty.clone() };
|
||||||
Param { func: self, ty, idx }
|
Param { func: Callee::Def(CallableDefId::FunctionId(self.id)), ty, idx }
|
||||||
})
|
})
|
||||||
.collect()
|
.collect()
|
||||||
}
|
}
|
||||||
|
@ -2167,16 +2166,23 @@ impl From<hir_ty::Mutability> for Access {
|
||||||
|
|
||||||
#[derive(Clone, PartialEq, Eq, Hash, Debug)]
|
#[derive(Clone, PartialEq, Eq, Hash, Debug)]
|
||||||
pub struct Param {
|
pub struct Param {
|
||||||
func: Function,
|
func: Callee,
|
||||||
/// The index in parameter list, including self parameter.
|
/// The index in parameter list, including self parameter.
|
||||||
idx: usize,
|
idx: usize,
|
||||||
ty: Type,
|
ty: Type,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Param {
|
impl Param {
|
||||||
pub fn parent_fn(&self) -> Function {
|
pub fn parent_fn(&self) -> Option<Function> {
|
||||||
self.func
|
match self.func {
|
||||||
|
Callee::Def(CallableDefId::FunctionId(f)) => Some(f.into()),
|
||||||
|
_ => None,
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// pub fn parent_closure(&self) -> Option<Closure> {
|
||||||
|
// self.func.as_ref().right().cloned()
|
||||||
|
// }
|
||||||
|
|
||||||
pub fn index(&self) -> usize {
|
pub fn index(&self) -> usize {
|
||||||
self.idx
|
self.idx
|
||||||
|
@ -2191,7 +2197,11 @@ impl Param {
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn as_local(&self, db: &dyn HirDatabase) -> Option<Local> {
|
pub fn as_local(&self, db: &dyn HirDatabase) -> Option<Local> {
|
||||||
let parent = DefWithBodyId::FunctionId(self.func.into());
|
let parent = match self.func {
|
||||||
|
Callee::Def(CallableDefId::FunctionId(it)) => DefWithBodyId::FunctionId(it),
|
||||||
|
Callee::Closure(closure, _) => db.lookup_intern_closure(closure.into()).0,
|
||||||
|
_ => return None,
|
||||||
|
};
|
||||||
let body = db.body(parent);
|
let body = db.body(parent);
|
||||||
if let Some(self_param) = body.self_param.filter(|_| self.idx == 0) {
|
if let Some(self_param) = body.self_param.filter(|_| self.idx == 0) {
|
||||||
Some(Local { parent, binding_id: self_param })
|
Some(Local { parent, binding_id: self_param })
|
||||||
|
@ -2205,19 +2215,46 @@ impl Param {
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn pattern_source(&self, db: &dyn HirDatabase) -> Option<ast::Pat> {
|
pub fn pattern_source(&self, db: &dyn HirDatabase) -> Option<ast::Pat> {
|
||||||
self.source(db).and_then(|p| p.value.pat())
|
self.source(db).and_then(|p| p.value.right()?.pat())
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn source(&self, db: &dyn HirDatabase) -> Option<InFile<ast::Param>> {
|
pub fn source(
|
||||||
let InFile { file_id, value } = self.func.source(db)?;
|
&self,
|
||||||
|
db: &dyn HirDatabase,
|
||||||
|
) -> Option<InFile<Either<ast::SelfParam, ast::Param>>> {
|
||||||
|
match self.func {
|
||||||
|
Callee::Def(CallableDefId::FunctionId(func)) => {
|
||||||
|
let InFile { file_id, value } = Function { id: func }.source(db)?;
|
||||||
let params = value.param_list()?;
|
let params = value.param_list()?;
|
||||||
if params.self_param().is_some() {
|
if let Some(self_param) = params.self_param() {
|
||||||
params.params().nth(self.idx.checked_sub(params.self_param().is_some() as usize)?)
|
if let Some(idx) = self.idx.checked_sub(1) {
|
||||||
|
params.params().nth(idx).map(Either::Right)
|
||||||
} else {
|
} else {
|
||||||
params.params().nth(self.idx)
|
Some(Either::Left(self_param))
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
params.params().nth(self.idx).map(Either::Right)
|
||||||
}
|
}
|
||||||
.map(|value| InFile { file_id, value })
|
.map(|value| InFile { file_id, value })
|
||||||
}
|
}
|
||||||
|
Callee::Closure(closure, _) => {
|
||||||
|
let InternedClosure(owner, expr_id) = db.lookup_intern_closure(closure.into());
|
||||||
|
let (_, source_map) = db.body_with_source_map(owner);
|
||||||
|
let ast @ InFile { file_id, value } = source_map.expr_syntax(expr_id).ok()?;
|
||||||
|
let root = db.parse_or_expand(file_id);
|
||||||
|
match value.to_node(&root) {
|
||||||
|
ast::Expr::ClosureExpr(it) => it
|
||||||
|
.param_list()?
|
||||||
|
.params()
|
||||||
|
.nth(self.idx)
|
||||||
|
.map(Either::Right)
|
||||||
|
.map(|value| InFile { file_id: ast.file_id, value }),
|
||||||
|
_ => None,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
_ => None,
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
|
#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
|
||||||
|
@ -3372,34 +3409,21 @@ impl BuiltinAttr {
|
||||||
|
|
||||||
#[derive(Clone, Copy, Debug, PartialEq, Eq, Hash)]
|
#[derive(Clone, Copy, Debug, PartialEq, Eq, Hash)]
|
||||||
pub struct ToolModule {
|
pub struct ToolModule {
|
||||||
krate: Option<CrateId>,
|
krate: CrateId,
|
||||||
idx: u32,
|
idx: u32,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl ToolModule {
|
impl ToolModule {
|
||||||
// FIXME: consider crates\hir_def\src\nameres\attr_resolution.rs?
|
|
||||||
pub(crate) fn by_name(db: &dyn HirDatabase, krate: Crate, name: &str) -> Option<Self> {
|
pub(crate) fn by_name(db: &dyn HirDatabase, krate: Crate, name: &str) -> Option<Self> {
|
||||||
if let builtin @ Some(_) = Self::builtin(name) {
|
let krate = krate.id;
|
||||||
return builtin;
|
|
||||||
}
|
|
||||||
let idx =
|
let idx =
|
||||||
db.crate_def_map(krate.id).registered_tools().iter().position(|it| it == name)? as u32;
|
db.crate_def_map(krate).registered_tools().iter().position(|it| it == name)? as u32;
|
||||||
Some(ToolModule { krate: Some(krate.id), idx })
|
Some(ToolModule { krate, idx })
|
||||||
}
|
|
||||||
|
|
||||||
fn builtin(name: &str) -> Option<Self> {
|
|
||||||
hir_def::attr::builtin::TOOL_MODULES
|
|
||||||
.iter()
|
|
||||||
.position(|&tool| tool == name)
|
|
||||||
.map(|idx| ToolModule { krate: None, idx: idx as u32 })
|
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn name(&self, db: &dyn HirDatabase) -> SmolStr {
|
pub fn name(&self, db: &dyn HirDatabase) -> SmolStr {
|
||||||
// FIXME: Return a `Name` here
|
// FIXME: Return a `Name` here
|
||||||
match self.krate {
|
db.crate_def_map(self.krate).registered_tools()[self.idx as usize].clone()
|
||||||
Some(krate) => db.crate_def_map(krate).registered_tools()[self.idx as usize].clone(),
|
|
||||||
None => SmolStr::new(hir_def::attr::builtin::TOOL_MODULES[self.idx as usize]),
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -4292,27 +4316,37 @@ impl Type {
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn as_callable(&self, db: &dyn HirDatabase) -> Option<Callable> {
|
pub fn as_callable(&self, db: &dyn HirDatabase) -> Option<Callable> {
|
||||||
let mut the_ty = &self.ty;
|
|
||||||
let callee = match self.ty.kind(Interner) {
|
let callee = match self.ty.kind(Interner) {
|
||||||
TyKind::Ref(_, _, ty) if ty.as_closure().is_some() => {
|
TyKind::Closure(id, subst) => Callee::Closure(*id, subst.clone()),
|
||||||
the_ty = ty;
|
|
||||||
Callee::Closure(ty.as_closure().unwrap())
|
|
||||||
}
|
|
||||||
TyKind::Closure(id, _) => Callee::Closure(*id),
|
|
||||||
TyKind::Function(_) => Callee::FnPtr,
|
TyKind::Function(_) => Callee::FnPtr,
|
||||||
TyKind::FnDef(..) => Callee::Def(self.ty.callable_def(db)?),
|
TyKind::FnDef(..) => Callee::Def(self.ty.callable_def(db)?),
|
||||||
_ => {
|
kind => {
|
||||||
let sig = hir_ty::callable_sig_from_fnonce(&self.ty, self.env.clone(), db)?;
|
// This will happen when it implements fn or fn mut, since we add an autoborrow adjustment
|
||||||
|
let (ty, kind) = if let TyKind::Ref(_, _, ty) = kind {
|
||||||
|
(ty, ty.kind(Interner))
|
||||||
|
} else {
|
||||||
|
(&self.ty, kind)
|
||||||
|
};
|
||||||
|
if let TyKind::Closure(closure, subst) = kind {
|
||||||
|
let sig = ty.callable_sig(db)?;
|
||||||
return Some(Callable {
|
return Some(Callable {
|
||||||
ty: self.clone(),
|
ty: self.clone(),
|
||||||
sig,
|
sig,
|
||||||
callee: Callee::Other,
|
callee: Callee::Closure(*closure, subst.clone()),
|
||||||
|
is_bound_method: false,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
let (fn_trait, sig) = hir_ty::callable_sig_from_fn_trait(ty, self.env.clone(), db)?;
|
||||||
|
return Some(Callable {
|
||||||
|
ty: self.clone(),
|
||||||
|
sig,
|
||||||
|
callee: Callee::FnImpl(fn_trait),
|
||||||
is_bound_method: false,
|
is_bound_method: false,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
let sig = the_ty.callable_sig(db)?;
|
let sig = self.ty.callable_sig(db)?;
|
||||||
Some(Callable { ty: self.clone(), sig, callee, is_bound_method: false })
|
Some(Callable { ty: self.clone(), sig, callee, is_bound_method: false })
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -4929,37 +4963,39 @@ pub struct Callable {
|
||||||
sig: CallableSig,
|
sig: CallableSig,
|
||||||
callee: Callee,
|
callee: Callee,
|
||||||
/// Whether this is a method that was called with method call syntax.
|
/// Whether this is a method that was called with method call syntax.
|
||||||
pub(crate) is_bound_method: bool,
|
is_bound_method: bool,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug)]
|
#[derive(Clone, PartialEq, Eq, Hash, Debug)]
|
||||||
enum Callee {
|
enum Callee {
|
||||||
Def(CallableDefId),
|
Def(CallableDefId),
|
||||||
Closure(ClosureId),
|
Closure(ClosureId, Substitution),
|
||||||
FnPtr,
|
FnPtr,
|
||||||
Other,
|
FnImpl(FnTrait),
|
||||||
}
|
}
|
||||||
|
|
||||||
pub enum CallableKind {
|
pub enum CallableKind {
|
||||||
Function(Function),
|
Function(Function),
|
||||||
TupleStruct(Struct),
|
TupleStruct(Struct),
|
||||||
TupleEnumVariant(Variant),
|
TupleEnumVariant(Variant),
|
||||||
Closure,
|
Closure(Closure),
|
||||||
FnPtr,
|
FnPtr,
|
||||||
/// Some other type that implements `FnOnce`.
|
FnImpl(FnTrait),
|
||||||
Other,
|
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Callable {
|
impl Callable {
|
||||||
pub fn kind(&self) -> CallableKind {
|
pub fn kind(&self) -> CallableKind {
|
||||||
use Callee::*;
|
|
||||||
match self.callee {
|
match self.callee {
|
||||||
Def(CallableDefId::FunctionId(it)) => CallableKind::Function(it.into()),
|
Callee::Def(CallableDefId::FunctionId(it)) => CallableKind::Function(it.into()),
|
||||||
Def(CallableDefId::StructId(it)) => CallableKind::TupleStruct(it.into()),
|
Callee::Def(CallableDefId::StructId(it)) => CallableKind::TupleStruct(it.into()),
|
||||||
Def(CallableDefId::EnumVariantId(it)) => CallableKind::TupleEnumVariant(it.into()),
|
Callee::Def(CallableDefId::EnumVariantId(it)) => {
|
||||||
Closure(_) => CallableKind::Closure,
|
CallableKind::TupleEnumVariant(it.into())
|
||||||
FnPtr => CallableKind::FnPtr,
|
}
|
||||||
Other => CallableKind::Other,
|
Callee::Closure(id, ref subst) => {
|
||||||
|
CallableKind::Closure(Closure { id, subst: subst.clone() })
|
||||||
|
}
|
||||||
|
Callee::FnPtr => CallableKind::FnPtr,
|
||||||
|
Callee::FnImpl(fn_) => CallableKind::FnImpl(fn_),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
pub fn receiver_param(&self, db: &dyn HirDatabase) -> Option<(SelfParam, Type)> {
|
pub fn receiver_param(&self, db: &dyn HirDatabase) -> Option<(SelfParam, Type)> {
|
||||||
|
@ -4973,43 +5009,15 @@ impl Callable {
|
||||||
pub fn n_params(&self) -> usize {
|
pub fn n_params(&self) -> usize {
|
||||||
self.sig.params().len() - if self.is_bound_method { 1 } else { 0 }
|
self.sig.params().len() - if self.is_bound_method { 1 } else { 0 }
|
||||||
}
|
}
|
||||||
pub fn params(
|
pub fn params(&self) -> Vec<Param> {
|
||||||
&self,
|
self.sig
|
||||||
db: &dyn HirDatabase,
|
|
||||||
) -> Vec<(Option<Either<ast::SelfParam, ast::Pat>>, Type)> {
|
|
||||||
let types = self
|
|
||||||
.sig
|
|
||||||
.params()
|
.params()
|
||||||
.iter()
|
.iter()
|
||||||
|
.enumerate()
|
||||||
.skip(if self.is_bound_method { 1 } else { 0 })
|
.skip(if self.is_bound_method { 1 } else { 0 })
|
||||||
.map(|ty| self.ty.derived(ty.clone()));
|
.map(|(idx, ty)| (idx, self.ty.derived(ty.clone())))
|
||||||
let map_param = |it: ast::Param| it.pat().map(Either::Right);
|
.map(|(idx, ty)| Param { func: self.callee.clone(), idx, ty })
|
||||||
let patterns = match self.callee {
|
.collect()
|
||||||
Callee::Def(CallableDefId::FunctionId(func)) => {
|
|
||||||
let src = func.lookup(db.upcast()).source(db.upcast());
|
|
||||||
src.value.param_list().map(|param_list| {
|
|
||||||
param_list
|
|
||||||
.self_param()
|
|
||||||
.map(|it| Some(Either::Left(it)))
|
|
||||||
.filter(|_| !self.is_bound_method)
|
|
||||||
.into_iter()
|
|
||||||
.chain(param_list.params().map(map_param))
|
|
||||||
})
|
|
||||||
}
|
|
||||||
Callee::Closure(closure_id) => match closure_source(db, closure_id) {
|
|
||||||
Some(src) => src.param_list().map(|param_list| {
|
|
||||||
param_list
|
|
||||||
.self_param()
|
|
||||||
.map(|it| Some(Either::Left(it)))
|
|
||||||
.filter(|_| !self.is_bound_method)
|
|
||||||
.into_iter()
|
|
||||||
.chain(param_list.params().map(map_param))
|
|
||||||
}),
|
|
||||||
None => None,
|
|
||||||
},
|
|
||||||
_ => None,
|
|
||||||
};
|
|
||||||
patterns.into_iter().flatten().chain(iter::repeat(None)).zip(types).collect()
|
|
||||||
}
|
}
|
||||||
pub fn return_type(&self) -> Type {
|
pub fn return_type(&self) -> Type {
|
||||||
self.ty.derived(self.sig.ret().clone())
|
self.ty.derived(self.sig.ret().clone())
|
||||||
|
@ -5017,17 +5025,9 @@ impl Callable {
|
||||||
pub fn sig(&self) -> &CallableSig {
|
pub fn sig(&self) -> &CallableSig {
|
||||||
&self.sig
|
&self.sig
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
fn closure_source(db: &dyn HirDatabase, closure: ClosureId) -> Option<ast::ClosureExpr> {
|
pub fn ty(&self) -> &Type {
|
||||||
let InternedClosure(owner, expr_id) = db.lookup_intern_closure(closure.into());
|
&self.ty
|
||||||
let (_, source_map) = db.body_with_source_map(owner);
|
|
||||||
let ast = source_map.expr_syntax(expr_id).ok()?;
|
|
||||||
let root = ast.file_syntax(db.upcast());
|
|
||||||
let expr = ast.value.to_node(&root);
|
|
||||||
match expr {
|
|
||||||
ast::Expr::ClosureExpr(it) => Some(it),
|
|
||||||
_ => None,
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -307,7 +307,8 @@ impl SourceAnalyzer {
|
||||||
db: &dyn HirDatabase,
|
db: &dyn HirDatabase,
|
||||||
call: &ast::Expr,
|
call: &ast::Expr,
|
||||||
) -> Option<Callable> {
|
) -> Option<Callable> {
|
||||||
self.type_of_expr(db, &call.clone())?.0.as_callable(db)
|
let (orig, adjusted) = self.type_of_expr(db, &call.clone())?;
|
||||||
|
adjusted.unwrap_or(orig).as_callable(db)
|
||||||
}
|
}
|
||||||
|
|
||||||
pub(crate) fn resolve_field(
|
pub(crate) fn resolve_field(
|
||||||
|
|
|
@ -114,10 +114,10 @@ pub(crate) fn replace_with_eager_method(acc: &mut Assists, ctx: &AssistContext<'
|
||||||
let callable = ctx.sema.resolve_method_call_as_callable(&call)?;
|
let callable = ctx.sema.resolve_method_call_as_callable(&call)?;
|
||||||
let (_, receiver_ty) = callable.receiver_param(ctx.sema.db)?;
|
let (_, receiver_ty) = callable.receiver_param(ctx.sema.db)?;
|
||||||
let n_params = callable.n_params() + 1;
|
let n_params = callable.n_params() + 1;
|
||||||
let params = callable.params(ctx.sema.db);
|
let params = callable.params();
|
||||||
|
|
||||||
// FIXME: Check that the arg is of the form `() -> T`
|
// FIXME: Check that the arg is of the form `() -> T`
|
||||||
if !params.first()?.1.impls_fnonce(ctx.sema.db) {
|
if !params.first()?.ty().impls_fnonce(ctx.sema.db) {
|
||||||
return None;
|
return None;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -253,11 +253,8 @@ fn from_param(expr: &ast::Expr, sema: &Semantics<'_, RootDatabase>) -> Option<St
|
||||||
};
|
};
|
||||||
|
|
||||||
let (idx, _) = arg_list.args().find_position(|it| it == expr).unwrap();
|
let (idx, _) = arg_list.args().find_position(|it| it == expr).unwrap();
|
||||||
let (pat, _) = func.params(sema.db).into_iter().nth(idx)?;
|
let param = func.params().into_iter().nth(idx)?;
|
||||||
let pat = match pat? {
|
let pat = param.source(sema.db)?.value.right()?.pat()?;
|
||||||
either::Either::Right(pat) => pat,
|
|
||||||
_ => return None,
|
|
||||||
};
|
|
||||||
let name = var_name_from_pat(&pat)?;
|
let name = var_name_from_pat(&pat)?;
|
||||||
normalize(&name.to_string())
|
normalize(&name.to_string())
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,7 +1,7 @@
|
||||||
//! This module provides functionality for querying callable information about a token.
|
//! This module provides functionality for querying callable information about a token.
|
||||||
|
|
||||||
use either::Either;
|
use either::Either;
|
||||||
use hir::{Semantics, Type};
|
use hir::{InFile, Semantics, Type};
|
||||||
use parser::T;
|
use parser::T;
|
||||||
use syntax::{
|
use syntax::{
|
||||||
ast::{self, HasArgList, HasName},
|
ast::{self, HasArgList, HasName},
|
||||||
|
@ -13,7 +13,7 @@ use crate::RootDatabase;
|
||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
pub struct ActiveParameter {
|
pub struct ActiveParameter {
|
||||||
pub ty: Type,
|
pub ty: Type,
|
||||||
pub pat: Option<Either<ast::SelfParam, ast::Pat>>,
|
pub src: Option<InFile<Either<ast::SelfParam, ast::Param>>>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl ActiveParameter {
|
impl ActiveParameter {
|
||||||
|
@ -22,18 +22,18 @@ impl ActiveParameter {
|
||||||
let (signature, active_parameter) = callable_for_token(sema, token)?;
|
let (signature, active_parameter) = callable_for_token(sema, token)?;
|
||||||
|
|
||||||
let idx = active_parameter?;
|
let idx = active_parameter?;
|
||||||
let mut params = signature.params(sema.db);
|
let mut params = signature.params();
|
||||||
if idx >= params.len() {
|
if idx >= params.len() {
|
||||||
cov_mark::hit!(too_many_arguments);
|
cov_mark::hit!(too_many_arguments);
|
||||||
return None;
|
return None;
|
||||||
}
|
}
|
||||||
let (pat, ty) = params.swap_remove(idx);
|
let param = params.swap_remove(idx);
|
||||||
Some(ActiveParameter { ty, pat })
|
Some(ActiveParameter { ty: param.ty().clone(), src: param.source(sema.db) })
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn ident(&self) -> Option<ast::Name> {
|
pub fn ident(&self) -> Option<ast::Name> {
|
||||||
self.pat.as_ref().and_then(|param| match param {
|
self.src.as_ref().and_then(|param| match param.value.as_ref().right()?.pat()? {
|
||||||
Either::Right(ast::Pat::IdentPat(ident)) => ident.name(),
|
ast::Pat::IdentPat(ident) => ident.name(),
|
||||||
_ => None,
|
_ => None,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
@ -60,10 +60,7 @@ pub fn callable_for_node(
|
||||||
token: &SyntaxToken,
|
token: &SyntaxToken,
|
||||||
) -> Option<(hir::Callable, Option<usize>)> {
|
) -> Option<(hir::Callable, Option<usize>)> {
|
||||||
let callable = match calling_node {
|
let callable = match calling_node {
|
||||||
ast::CallableExpr::Call(call) => {
|
ast::CallableExpr::Call(call) => sema.resolve_expr_as_callable(&call.expr()?),
|
||||||
let expr = call.expr()?;
|
|
||||||
sema.type_of_expr(&expr)?.adjusted().as_callable(sema.db)
|
|
||||||
}
|
|
||||||
ast::CallableExpr::MethodCall(call) => sema.resolve_method_call_as_callable(call),
|
ast::CallableExpr::MethodCall(call) => sema.resolve_method_call_as_callable(call),
|
||||||
}?;
|
}?;
|
||||||
let active_param = calling_node.arg_list().map(|arg_list| {
|
let active_param = calling_node.arg_list().map(|arg_list| {
|
||||||
|
|
|
@ -41,6 +41,7 @@ pub enum FormatSpecifier {
|
||||||
Escape,
|
Escape,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// FIXME: Remove this, we can use rustc_format_parse instead
|
||||||
pub fn lex_format_specifiers(
|
pub fn lex_format_specifiers(
|
||||||
string: &ast::String,
|
string: &ast::String,
|
||||||
mut callback: &mut dyn FnMut(TextRange, FormatSpecifier),
|
mut callback: &mut dyn FnMut(TextRange, FormatSpecifier),
|
||||||
|
|
|
@ -109,12 +109,12 @@ pub(crate) fn outgoing_calls(
|
||||||
let expr = call.expr()?;
|
let expr = call.expr()?;
|
||||||
let callable = sema.type_of_expr(&expr)?.original.as_callable(db)?;
|
let callable = sema.type_of_expr(&expr)?.original.as_callable(db)?;
|
||||||
match callable.kind() {
|
match callable.kind() {
|
||||||
hir::CallableKind::Function(it) => {
|
hir::CallableKind::Function(it) => it.try_to_nav(db),
|
||||||
let range = expr.syntax().text_range();
|
hir::CallableKind::TupleEnumVariant(it) => it.try_to_nav(db),
|
||||||
it.try_to_nav(db).zip(Some(range))
|
hir::CallableKind::TupleStruct(it) => it.try_to_nav(db),
|
||||||
}
|
|
||||||
_ => None,
|
_ => None,
|
||||||
}
|
}
|
||||||
|
.zip(Some(expr.syntax().text_range()))
|
||||||
}
|
}
|
||||||
ast::CallableExpr::MethodCall(expr) => {
|
ast::CallableExpr::MethodCall(expr) => {
|
||||||
let range = expr.name_ref()?.syntax().text_range();
|
let range = expr.name_ref()?.syntax().text_range();
|
||||||
|
|
|
@ -24,34 +24,29 @@ pub(super) fn hints(
|
||||||
|
|
||||||
let (callable, arg_list) = get_callable(sema, &expr)?;
|
let (callable, arg_list) = get_callable(sema, &expr)?;
|
||||||
let hints = callable
|
let hints = callable
|
||||||
.params(sema.db)
|
.params()
|
||||||
.into_iter()
|
.into_iter()
|
||||||
.zip(arg_list.args())
|
.zip(arg_list.args())
|
||||||
.filter_map(|((param, _ty), arg)| {
|
.filter_map(|(p, arg)| {
|
||||||
// Only annotate hints for expressions that exist in the original file
|
// Only annotate hints for expressions that exist in the original file
|
||||||
let range = sema.original_range_opt(arg.syntax())?;
|
let range = sema.original_range_opt(arg.syntax())?;
|
||||||
let (param_name, name_syntax) = match param.as_ref()? {
|
let source = p.source(sema.db)?;
|
||||||
|
let (param_name, name_syntax) = match source.value.as_ref() {
|
||||||
Either::Left(pat) => (pat.name()?, pat.name()),
|
Either::Left(pat) => (pat.name()?, pat.name()),
|
||||||
Either::Right(pat) => match pat {
|
Either::Right(param) => match param.pat()? {
|
||||||
ast::Pat::IdentPat(it) => (it.name()?, it.name()),
|
ast::Pat::IdentPat(it) => (it.name()?, it.name()),
|
||||||
_ => return None,
|
_ => return None,
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
// make sure the file is cached so we can map out of macros
|
||||||
|
sema.parse_or_expand(source.file_id);
|
||||||
Some((name_syntax, param_name, arg, range))
|
Some((name_syntax, param_name, arg, range))
|
||||||
})
|
})
|
||||||
.filter(|(_, param_name, arg, _)| {
|
.filter(|(_, param_name, arg, _)| {
|
||||||
!should_hide_param_name_hint(sema, &callable, ¶m_name.text(), arg)
|
!should_hide_param_name_hint(sema, &callable, ¶m_name.text(), arg)
|
||||||
})
|
})
|
||||||
.map(|(param, param_name, _, FileRange { range, .. })| {
|
.map(|(param, param_name, _, FileRange { range, .. })| {
|
||||||
let mut linked_location = None;
|
let linked_location = param.and_then(|name| sema.original_range_opt(name.syntax()));
|
||||||
if let Some(name) = param {
|
|
||||||
if let hir::CallableKind::Function(f) = callable.kind() {
|
|
||||||
// assert the file is cached so we can map out of macros
|
|
||||||
if sema.source(f).is_some() {
|
|
||||||
linked_location = sema.original_range_opt(name.syntax());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
let colon = if config.render_colons { ":" } else { "" };
|
let colon = if config.render_colons { ":" } else { "" };
|
||||||
let label =
|
let label =
|
||||||
|
|
|
@ -201,7 +201,21 @@ fn signature_help_for_call(
|
||||||
variant.name(db).display(db)
|
variant.name(db).display(db)
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
hir::CallableKind::Closure | hir::CallableKind::FnPtr | hir::CallableKind::Other => (),
|
hir::CallableKind::Closure(closure) => {
|
||||||
|
let fn_trait = closure.fn_trait(db);
|
||||||
|
format_to!(res.signature, "impl {fn_trait}")
|
||||||
|
}
|
||||||
|
hir::CallableKind::FnPtr => format_to!(res.signature, "fn"),
|
||||||
|
hir::CallableKind::FnImpl(fn_trait) => match callable.ty().as_adt() {
|
||||||
|
// FIXME: Render docs of the concrete trait impl function
|
||||||
|
Some(adt) => format_to!(
|
||||||
|
res.signature,
|
||||||
|
"<{} as {fn_trait}>::{}",
|
||||||
|
adt.name(db).display(db),
|
||||||
|
fn_trait.function_name()
|
||||||
|
),
|
||||||
|
None => format_to!(res.signature, "impl {fn_trait}"),
|
||||||
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
res.signature.push('(');
|
res.signature.push('(');
|
||||||
|
@ -210,12 +224,15 @@ fn signature_help_for_call(
|
||||||
format_to!(res.signature, "{}", self_param.display(db))
|
format_to!(res.signature, "{}", self_param.display(db))
|
||||||
}
|
}
|
||||||
let mut buf = String::new();
|
let mut buf = String::new();
|
||||||
for (idx, (pat, ty)) in callable.params(db).into_iter().enumerate() {
|
for (idx, p) in callable.params().into_iter().enumerate() {
|
||||||
buf.clear();
|
buf.clear();
|
||||||
if let Some(pat) = pat {
|
if let Some(param) = p.source(sema.db) {
|
||||||
match pat {
|
match param.value {
|
||||||
Either::Left(_self) => format_to!(buf, "self: "),
|
Either::Right(param) => match param.pat() {
|
||||||
Either::Right(pat) => format_to!(buf, "{}: ", pat),
|
Some(pat) => format_to!(buf, "{}: ", pat),
|
||||||
|
None => format_to!(buf, "?: "),
|
||||||
|
},
|
||||||
|
Either::Left(_) => format_to!(buf, "self: "),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
// APITs (argument position `impl Trait`s) are inferred as {unknown} as the user is
|
// APITs (argument position `impl Trait`s) are inferred as {unknown} as the user is
|
||||||
|
@ -223,9 +240,9 @@ fn signature_help_for_call(
|
||||||
// In that case, fall back to render definitions of the respective parameters.
|
// In that case, fall back to render definitions of the respective parameters.
|
||||||
// This is overly conservative: we do not substitute known type vars
|
// This is overly conservative: we do not substitute known type vars
|
||||||
// (see FIXME in tests::impl_trait) and falling back on any unknowns.
|
// (see FIXME in tests::impl_trait) and falling back on any unknowns.
|
||||||
match (ty.contains_unknown(), fn_params.as_deref()) {
|
match (p.ty().contains_unknown(), fn_params.as_deref()) {
|
||||||
(true, Some(fn_params)) => format_to!(buf, "{}", fn_params[idx].ty().display(db)),
|
(true, Some(fn_params)) => format_to!(buf, "{}", fn_params[idx].ty().display(db)),
|
||||||
_ => format_to!(buf, "{}", ty.display(db)),
|
_ => format_to!(buf, "{}", p.ty().display(db)),
|
||||||
}
|
}
|
||||||
res.push_call_param(&buf);
|
res.push_call_param(&buf);
|
||||||
}
|
}
|
||||||
|
@ -242,9 +259,9 @@ fn signature_help_for_call(
|
||||||
render(func.ret_type(db))
|
render(func.ret_type(db))
|
||||||
}
|
}
|
||||||
hir::CallableKind::Function(_)
|
hir::CallableKind::Function(_)
|
||||||
| hir::CallableKind::Closure
|
| hir::CallableKind::Closure(_)
|
||||||
| hir::CallableKind::FnPtr
|
| hir::CallableKind::FnPtr
|
||||||
| hir::CallableKind::Other => render(callable.return_type()),
|
| hir::CallableKind::FnImpl(_) => render(callable.return_type()),
|
||||||
hir::CallableKind::TupleStruct(_) | hir::CallableKind::TupleEnumVariant(_) => {}
|
hir::CallableKind::TupleStruct(_) | hir::CallableKind::TupleEnumVariant(_) => {}
|
||||||
}
|
}
|
||||||
Some(res)
|
Some(res)
|
||||||
|
@ -1345,15 +1362,43 @@ fn test() { S.foo($0); }
|
||||||
r#"
|
r#"
|
||||||
struct S;
|
struct S;
|
||||||
fn foo(s: S) -> i32 { 92 }
|
fn foo(s: S) -> i32 { 92 }
|
||||||
|
fn main() {
|
||||||
|
let _move = S;
|
||||||
|
(|s| {{_move}; foo(s)})($0)
|
||||||
|
}
|
||||||
|
"#,
|
||||||
|
expect![[r#"
|
||||||
|
impl FnOnce(s: S) -> i32
|
||||||
|
^^^^
|
||||||
|
"#]],
|
||||||
|
);
|
||||||
|
check(
|
||||||
|
r#"
|
||||||
|
struct S;
|
||||||
|
fn foo(s: S) -> i32 { 92 }
|
||||||
fn main() {
|
fn main() {
|
||||||
(|s| foo(s))($0)
|
(|s| foo(s))($0)
|
||||||
}
|
}
|
||||||
"#,
|
"#,
|
||||||
expect![[r#"
|
expect![[r#"
|
||||||
(s: S) -> i32
|
impl Fn(s: S) -> i32
|
||||||
^^^^
|
^^^^
|
||||||
"#]],
|
"#]],
|
||||||
)
|
);
|
||||||
|
check(
|
||||||
|
r#"
|
||||||
|
struct S;
|
||||||
|
fn foo(s: S) -> i32 { 92 }
|
||||||
|
fn main() {
|
||||||
|
let mut mutate = 0;
|
||||||
|
(|s| { mutate = 1; foo(s) })($0)
|
||||||
|
}
|
||||||
|
"#,
|
||||||
|
expect![[r#"
|
||||||
|
impl FnMut(s: S) -> i32
|
||||||
|
^^^^
|
||||||
|
"#]],
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
|
@ -1383,12 +1428,81 @@ fn main(f: fn(i32, f64) -> char) {
|
||||||
}
|
}
|
||||||
"#,
|
"#,
|
||||||
expect![[r#"
|
expect![[r#"
|
||||||
(i32, f64) -> char
|
fn(i32, f64) -> char
|
||||||
--- ^^^
|
--- ^^^
|
||||||
"#]],
|
"#]],
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn call_info_for_fn_impl() {
|
||||||
|
check(
|
||||||
|
r#"
|
||||||
|
struct S;
|
||||||
|
impl core::ops::FnOnce<(i32, f64)> for S {
|
||||||
|
type Output = char;
|
||||||
|
}
|
||||||
|
impl core::ops::FnMut<(i32, f64)> for S {}
|
||||||
|
impl core::ops::Fn<(i32, f64)> for S {}
|
||||||
|
fn main() {
|
||||||
|
S($0);
|
||||||
|
}
|
||||||
|
"#,
|
||||||
|
expect![[r#"
|
||||||
|
<S as Fn>::call(i32, f64) -> char
|
||||||
|
^^^ ---
|
||||||
|
"#]],
|
||||||
|
);
|
||||||
|
check(
|
||||||
|
r#"
|
||||||
|
struct S;
|
||||||
|
impl core::ops::FnOnce<(i32, f64)> for S {
|
||||||
|
type Output = char;
|
||||||
|
}
|
||||||
|
impl core::ops::FnMut<(i32, f64)> for S {}
|
||||||
|
impl core::ops::Fn<(i32, f64)> for S {}
|
||||||
|
fn main() {
|
||||||
|
S(1, $0);
|
||||||
|
}
|
||||||
|
"#,
|
||||||
|
expect![[r#"
|
||||||
|
<S as Fn>::call(i32, f64) -> char
|
||||||
|
--- ^^^
|
||||||
|
"#]],
|
||||||
|
);
|
||||||
|
check(
|
||||||
|
r#"
|
||||||
|
struct S;
|
||||||
|
impl core::ops::FnOnce<(i32, f64)> for S {
|
||||||
|
type Output = char;
|
||||||
|
}
|
||||||
|
impl core::ops::FnOnce<(char, char)> for S {
|
||||||
|
type Output = f64;
|
||||||
|
}
|
||||||
|
fn main() {
|
||||||
|
S($0);
|
||||||
|
}
|
||||||
|
"#,
|
||||||
|
expect![""],
|
||||||
|
);
|
||||||
|
check(
|
||||||
|
r#"
|
||||||
|
struct S;
|
||||||
|
impl core::ops::FnOnce<(i32, f64)> for S {
|
||||||
|
type Output = char;
|
||||||
|
}
|
||||||
|
impl core::ops::FnOnce<(char, char)> for S {
|
||||||
|
type Output = f64;
|
||||||
|
}
|
||||||
|
fn main() {
|
||||||
|
// FIXME: The ide layer loses the calling info here so we get an ambiguous trait solve result
|
||||||
|
S(0i32, $0);
|
||||||
|
}
|
||||||
|
"#,
|
||||||
|
expect![""],
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn call_info_for_unclosed_call() {
|
fn call_info_for_unclosed_call() {
|
||||||
check(
|
check(
|
||||||
|
@ -1794,18 +1908,18 @@ fn f<F: FnOnce(u8, u16) -> i32>(f: F) {
|
||||||
}
|
}
|
||||||
"#,
|
"#,
|
||||||
expect![[r#"
|
expect![[r#"
|
||||||
(u8, u16) -> i32
|
impl FnOnce(u8, u16) -> i32
|
||||||
^^ ---
|
^^ ---
|
||||||
"#]],
|
"#]],
|
||||||
);
|
);
|
||||||
check(
|
check(
|
||||||
r#"
|
r#"
|
||||||
fn f<T, F: FnOnce(&T, u16) -> &T>(f: F) {
|
fn f<T, F: FnMut(&T, u16) -> &T>(f: F) {
|
||||||
f($0)
|
f($0)
|
||||||
}
|
}
|
||||||
"#,
|
"#,
|
||||||
expect![[r#"
|
expect![[r#"
|
||||||
(&T, u16) -> &T
|
impl FnMut(&T, u16) -> &T
|
||||||
^^ ---
|
^^ ---
|
||||||
"#]],
|
"#]],
|
||||||
);
|
);
|
||||||
|
@ -1826,7 +1940,7 @@ fn take<C, Error>(
|
||||||
}
|
}
|
||||||
"#,
|
"#,
|
||||||
expect![[r#"
|
expect![[r#"
|
||||||
() -> i32
|
impl Fn() -> i32
|
||||||
"#]],
|
"#]],
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
|
@ -9,8 +9,9 @@ pub(super) fn highlight_escape_string<T: IsString>(
|
||||||
string: &T,
|
string: &T,
|
||||||
start: TextSize,
|
start: TextSize,
|
||||||
) {
|
) {
|
||||||
|
let text = string.text();
|
||||||
string.escaped_char_ranges(&mut |piece_range, char| {
|
string.escaped_char_ranges(&mut |piece_range, char| {
|
||||||
if string.text()[piece_range.start().into()..].starts_with('\\') {
|
if text[piece_range.start().into()..].starts_with('\\') {
|
||||||
let highlight = match char {
|
let highlight = match char {
|
||||||
Ok(_) => HlTag::EscapeSequence,
|
Ok(_) => HlTag::EscapeSequence,
|
||||||
Err(_) => HlTag::InvalidEscapeSequence,
|
Err(_) => HlTag::InvalidEscapeSequence,
|
||||||
|
@ -33,17 +34,15 @@ pub(super) fn highlight_escape_char(stack: &mut Highlights, char: &Char, start:
|
||||||
}
|
}
|
||||||
|
|
||||||
let text = char.text();
|
let text = char.text();
|
||||||
if !text.starts_with('\'') || !text.ends_with('\'') {
|
let Some(text) = text
|
||||||
|
.strip_prefix('\'')
|
||||||
|
.and_then(|it| it.strip_suffix('\''))
|
||||||
|
.filter(|it| it.starts_with('\\'))
|
||||||
|
else {
|
||||||
return;
|
return;
|
||||||
}
|
};
|
||||||
|
|
||||||
let text = &text[1..text.len() - 1];
|
let range = TextRange::at(start + TextSize::from(1), TextSize::from(text.len() as u32));
|
||||||
if !text.starts_with('\\') {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
let range =
|
|
||||||
TextRange::new(start + TextSize::from(1), start + TextSize::from(text.len() as u32 + 1));
|
|
||||||
stack.add(HlRange { range, highlight: HlTag::EscapeSequence.into(), binding_hash: None })
|
stack.add(HlRange { range, highlight: HlTag::EscapeSequence.into(), binding_hash: None })
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -54,16 +53,14 @@ pub(super) fn highlight_escape_byte(stack: &mut Highlights, byte: &Byte, start:
|
||||||
}
|
}
|
||||||
|
|
||||||
let text = byte.text();
|
let text = byte.text();
|
||||||
if !text.starts_with("b'") || !text.ends_with('\'') {
|
let Some(text) = text
|
||||||
|
.strip_prefix("b'")
|
||||||
|
.and_then(|it| it.strip_suffix('\''))
|
||||||
|
.filter(|it| it.starts_with('\\'))
|
||||||
|
else {
|
||||||
return;
|
return;
|
||||||
}
|
};
|
||||||
|
|
||||||
let text = &text[2..text.len() - 1];
|
let range = TextRange::at(start + TextSize::from(2), TextSize::from(text.len() as u32));
|
||||||
if !text.starts_with('\\') {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
let range =
|
|
||||||
TextRange::new(start + TextSize::from(2), start + TextSize::from(text.len() as u32 + 2));
|
|
||||||
stack.add(HlRange { range, highlight: HlTag::EscapeSequence.into(), binding_hash: None })
|
stack.add(HlRange { range, highlight: HlTag::EscapeSequence.into(), binding_hash: None })
|
||||||
}
|
}
|
||||||
|
|
|
@ -8,6 +8,7 @@ use std::{
|
||||||
use rustc_lexer::unescape::{
|
use rustc_lexer::unescape::{
|
||||||
unescape_byte, unescape_char, unescape_mixed, unescape_unicode, EscapeError, MixedUnit, Mode,
|
unescape_byte, unescape_char, unescape_mixed, unescape_unicode, EscapeError, MixedUnit, Mode,
|
||||||
};
|
};
|
||||||
|
use stdx::always;
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
ast::{self, AstToken},
|
ast::{self, AstToken},
|
||||||
|
@ -181,25 +182,25 @@ pub trait IsString: AstToken {
|
||||||
self.quote_offsets().map(|it| it.quotes.1)
|
self.quote_offsets().map(|it| it.quotes.1)
|
||||||
}
|
}
|
||||||
fn escaped_char_ranges(&self, cb: &mut dyn FnMut(TextRange, Result<char, EscapeError>)) {
|
fn escaped_char_ranges(&self, cb: &mut dyn FnMut(TextRange, Result<char, EscapeError>)) {
|
||||||
let text_range_no_quotes = match self.text_range_between_quotes() {
|
let Some(text_range_no_quotes) = self.text_range_between_quotes() else { return };
|
||||||
Some(it) => it,
|
|
||||||
None => return,
|
|
||||||
};
|
|
||||||
|
|
||||||
let start = self.syntax().text_range().start();
|
let start = self.syntax().text_range().start();
|
||||||
let text = &self.text()[text_range_no_quotes - start];
|
let text = &self.text()[text_range_no_quotes - start];
|
||||||
let offset = text_range_no_quotes.start() - start;
|
let offset = text_range_no_quotes.start() - start;
|
||||||
|
|
||||||
unescape_unicode(text, Self::MODE, &mut |range, unescaped_char| {
|
unescape_unicode(text, Self::MODE, &mut |range, unescaped_char| {
|
||||||
let text_range =
|
if let Some((s, e)) = range.start.try_into().ok().zip(range.end.try_into().ok()) {
|
||||||
TextRange::new(range.start.try_into().unwrap(), range.end.try_into().unwrap());
|
cb(TextRange::new(s, e) + offset, unescaped_char);
|
||||||
cb(text_range + offset, unescaped_char);
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
fn map_range_up(&self, range: TextRange) -> Option<TextRange> {
|
fn map_range_up(&self, range: TextRange) -> Option<TextRange> {
|
||||||
let contents_range = self.text_range_between_quotes()?;
|
let contents_range = self.text_range_between_quotes()?;
|
||||||
assert!(TextRange::up_to(contents_range.len()).contains_range(range));
|
if always!(TextRange::up_to(contents_range.len()).contains_range(range)) {
|
||||||
Some(range + contents_range.start())
|
Some(range + contents_range.start())
|
||||||
|
} else {
|
||||||
|
None
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue