2019-01-08 23:47:12 +00:00
|
|
|
//! HIR (previously known as descriptors) provides a high-level object oriented
|
|
|
|
//! access to Rust code.
|
2018-11-28 00:42:26 +00:00
|
|
|
//!
|
|
|
|
//! The principal difference between HIR and syntax trees is that HIR is bound
|
|
|
|
//! to a particular crate instance. That is, it has cfg flags and features
|
2019-01-08 23:47:12 +00:00
|
|
|
//! applied. So, the relation between syntax and HIR is many-to-one.
|
2020-03-23 12:00:51 +00:00
|
|
|
//!
|
|
|
|
//! HIR is the public API of the all of the compiler logic above syntax trees.
|
|
|
|
//! It is written in "OO" style. Each type is self contained (as in, it knows it's
|
|
|
|
//! parents and full context). It should be "clean code".
|
|
|
|
//!
|
2020-08-13 14:36:55 +00:00
|
|
|
//! `hir_*` crates are the implementation of the compiler logic.
|
2020-03-23 12:00:51 +00:00
|
|
|
//! They are written in "ECS" style, with relatively little abstractions.
|
2020-03-23 12:04:50 +00:00
|
|
|
//! Many types are not self-contained, and explicitly use local indexes, arenas, etc.
|
2020-03-23 12:00:51 +00:00
|
|
|
//!
|
2020-08-13 14:36:55 +00:00
|
|
|
//! `hir` is what insulates the "we don't know how to actually write an incremental compiler"
|
2020-03-23 12:00:51 +00:00
|
|
|
//! from the ide with completions, hovers, etc. It is a (soft, internal) boundary:
|
|
|
|
//! https://www.tedinski.com/2018/02/06/system-boundaries.html.
|
2018-11-28 00:42:26 +00:00
|
|
|
|
2019-09-30 08:58:53 +00:00
|
|
|
#![recursion_limit = "512"]
|
|
|
|
|
2020-02-18 17:35:10 +00:00
|
|
|
mod semantics;
|
2020-02-28 16:27:49 +00:00
|
|
|
mod source_analyzer;
|
2018-11-28 00:42:26 +00:00
|
|
|
|
2019-10-31 15:45:10 +00:00
|
|
|
mod from_id;
|
2020-08-25 10:56:01 +00:00
|
|
|
mod attrs;
|
2019-12-08 11:57:13 +00:00
|
|
|
mod has_source;
|
2019-09-16 10:48:54 +00:00
|
|
|
|
2021-03-08 19:08:30 +00:00
|
|
|
pub mod diagnostics;
|
|
|
|
pub mod db;
|
|
|
|
|
2021-03-14 09:36:04 +00:00
|
|
|
mod display;
|
|
|
|
|
2021-03-08 19:08:30 +00:00
|
|
|
use std::{iter, sync::Arc};
|
|
|
|
|
|
|
|
use arrayvec::ArrayVec;
|
|
|
|
use base_db::{CrateDisplayName, CrateId, Edition, FileId};
|
internal: move diagnostics to hir
The idea here is to eventually get rid of `dyn Diagnostic` and
`DiagnosticSink` infrastructure altogether, and just have a `enum
hir::Diagnostic` instead.
The problem with `dyn Diagnostic` is that it is defined in the lowest
level of the stack (hir_expand), but is used by the highest level (ide).
As a first step, we free hir_expand and hir_def from `dyn Diagnostic`
and kick the can up to `hir_ty`, as an intermediate state. The plan is
then to move DiagnosticSink similarly to the hir crate, and, as final
third step, remove its usage from the ide.
One currently unsolved problem is testing. You can notice that the test
which checks precise diagnostic ranges, unresolved_import_in_use_tree,
was moved to the ide layer. Logically, only IDE should have the infra to
render a specific range.
At the same time, the range is determined with the data produced in
hir_def and hir crates, so this layering is rather unfortunate. Working
on hir_def shouldn't require compiling `ide` for testing.
2021-05-23 20:31:59 +00:00
|
|
|
use diagnostics::{
|
2021-05-30 02:19:47 +00:00
|
|
|
InactiveCode, MacroError, UnimplementedBuiltinMacro, UnresolvedExternCrate, UnresolvedImport,
|
|
|
|
UnresolvedMacroCall, UnresolvedModule, UnresolvedProcMacro,
|
internal: move diagnostics to hir
The idea here is to eventually get rid of `dyn Diagnostic` and
`DiagnosticSink` infrastructure altogether, and just have a `enum
hir::Diagnostic` instead.
The problem with `dyn Diagnostic` is that it is defined in the lowest
level of the stack (hir_expand), but is used by the highest level (ide).
As a first step, we free hir_expand and hir_def from `dyn Diagnostic`
and kick the can up to `hir_ty`, as an intermediate state. The plan is
then to move DiagnosticSink similarly to the hir crate, and, as final
third step, remove its usage from the ide.
One currently unsolved problem is testing. You can notice that the test
which checks precise diagnostic ranges, unresolved_import_in_use_tree,
was moved to the ide layer. Logically, only IDE should have the infra to
render a specific range.
At the same time, the range is determined with the data produced in
hir_def and hir crates, so this layering is rather unfortunate. Working
on hir_def shouldn't require compiling `ide` for testing.
2021-05-23 20:31:59 +00:00
|
|
|
};
|
2021-03-08 19:08:30 +00:00
|
|
|
use either::Either;
|
|
|
|
use hir_def::{
|
|
|
|
adt::{ReprKind, VariantData},
|
internal: move diagnostics to hir
The idea here is to eventually get rid of `dyn Diagnostic` and
`DiagnosticSink` infrastructure altogether, and just have a `enum
hir::Diagnostic` instead.
The problem with `dyn Diagnostic` is that it is defined in the lowest
level of the stack (hir_expand), but is used by the highest level (ide).
As a first step, we free hir_expand and hir_def from `dyn Diagnostic`
and kick the can up to `hir_ty`, as an intermediate state. The plan is
then to move DiagnosticSink similarly to the hir crate, and, as final
third step, remove its usage from the ide.
One currently unsolved problem is testing. You can notice that the test
which checks precise diagnostic ranges, unresolved_import_in_use_tree,
was moved to the ide layer. Logically, only IDE should have the infra to
render a specific range.
At the same time, the range is determined with the data produced in
hir_def and hir crates, so this layering is rather unfortunate. Working
on hir_def shouldn't require compiling `ide` for testing.
2021-05-23 20:31:59 +00:00
|
|
|
body::BodyDiagnostic,
|
2021-03-08 19:08:30 +00:00
|
|
|
expr::{BindingAnnotation, LabelId, Pat, PatId},
|
|
|
|
item_tree::ItemTreeNode,
|
|
|
|
lang_item::LangItemTarget,
|
internal: move diagnostics to hir
The idea here is to eventually get rid of `dyn Diagnostic` and
`DiagnosticSink` infrastructure altogether, and just have a `enum
hir::Diagnostic` instead.
The problem with `dyn Diagnostic` is that it is defined in the lowest
level of the stack (hir_expand), but is used by the highest level (ide).
As a first step, we free hir_expand and hir_def from `dyn Diagnostic`
and kick the can up to `hir_ty`, as an intermediate state. The plan is
then to move DiagnosticSink similarly to the hir crate, and, as final
third step, remove its usage from the ide.
One currently unsolved problem is testing. You can notice that the test
which checks precise diagnostic ranges, unresolved_import_in_use_tree,
was moved to the ide layer. Logically, only IDE should have the infra to
render a specific range.
At the same time, the range is determined with the data produced in
hir_def and hir crates, so this layering is rather unfortunate. Working
on hir_def shouldn't require compiling `ide` for testing.
2021-05-23 20:31:59 +00:00
|
|
|
nameres,
|
2021-03-08 19:08:30 +00:00
|
|
|
per_ns::PerNs,
|
|
|
|
resolver::{HasResolver, Resolver},
|
|
|
|
src::HasSource as _,
|
|
|
|
AdtId, AssocContainerId, AssocItemId, AssocItemLoc, AttrDefId, ConstId, ConstParamId,
|
|
|
|
DefWithBodyId, EnumId, FunctionId, GenericDefId, HasModule, ImplId, LifetimeParamId,
|
|
|
|
LocalEnumVariantId, LocalFieldId, Lookup, ModuleId, StaticId, StructId, TraitId, TypeAliasId,
|
|
|
|
TypeParamId, UnionId,
|
|
|
|
};
|
internal: move diagnostics to hir
The idea here is to eventually get rid of `dyn Diagnostic` and
`DiagnosticSink` infrastructure altogether, and just have a `enum
hir::Diagnostic` instead.
The problem with `dyn Diagnostic` is that it is defined in the lowest
level of the stack (hir_expand), but is used by the highest level (ide).
As a first step, we free hir_expand and hir_def from `dyn Diagnostic`
and kick the can up to `hir_ty`, as an intermediate state. The plan is
then to move DiagnosticSink similarly to the hir crate, and, as final
third step, remove its usage from the ide.
One currently unsolved problem is testing. You can notice that the test
which checks precise diagnostic ranges, unresolved_import_in_use_tree,
was moved to the ide layer. Logically, only IDE should have the infra to
render a specific range.
At the same time, the range is determined with the data produced in
hir_def and hir crates, so this layering is rather unfortunate. Working
on hir_def shouldn't require compiling `ide` for testing.
2021-05-23 20:31:59 +00:00
|
|
|
use hir_expand::{name::name, MacroCallKind, MacroDefKind};
|
2021-03-08 19:08:30 +00:00
|
|
|
use hir_ty::{
|
2021-05-14 07:59:30 +00:00
|
|
|
autoderef,
|
2021-05-16 01:51:18 +00:00
|
|
|
consteval::ConstExt,
|
2021-05-14 07:59:30 +00:00
|
|
|
could_unify,
|
internal: move diagnostics to hir
The idea here is to eventually get rid of `dyn Diagnostic` and
`DiagnosticSink` infrastructure altogether, and just have a `enum
hir::Diagnostic` instead.
The problem with `dyn Diagnostic` is that it is defined in the lowest
level of the stack (hir_expand), but is used by the highest level (ide).
As a first step, we free hir_expand and hir_def from `dyn Diagnostic`
and kick the can up to `hir_ty`, as an intermediate state. The plan is
then to move DiagnosticSink similarly to the hir crate, and, as final
third step, remove its usage from the ide.
One currently unsolved problem is testing. You can notice that the test
which checks precise diagnostic ranges, unresolved_import_in_use_tree,
was moved to the ide layer. Logically, only IDE should have the infra to
render a specific range.
At the same time, the range is determined with the data produced in
hir_def and hir crates, so this layering is rather unfortunate. Working
on hir_def shouldn't require compiling `ide` for testing.
2021-05-23 20:31:59 +00:00
|
|
|
diagnostics_sink::DiagnosticSink,
|
2021-04-07 11:09:31 +00:00
|
|
|
method_resolution::{self, def_crates, TyFingerprint},
|
2021-03-15 20:28:21 +00:00
|
|
|
primitive::UintTy,
|
2021-04-05 19:56:40 +00:00
|
|
|
subst_prefix,
|
2021-04-04 18:27:40 +00:00
|
|
|
traits::FnTrait,
|
2021-03-21 19:05:38 +00:00
|
|
|
AliasEq, AliasTy, BoundVar, CallableDefId, CallableSig, Canonical, CanonicalVarKinds, Cast,
|
2021-04-06 21:46:32 +00:00
|
|
|
DebruijnIndex, InEnvironment, Interner, QuantifiedWhereClause, Scalar, Solution, Substitution,
|
2021-04-07 18:40:01 +00:00
|
|
|
TraitEnvironment, TraitRefExt, Ty, TyBuilder, TyDefId, TyExt, TyKind, TyVariableKind,
|
|
|
|
WhereClause,
|
2021-03-08 19:08:30 +00:00
|
|
|
};
|
2021-03-15 16:43:46 +00:00
|
|
|
use itertools::Itertools;
|
internal: move diagnostics to hir
The idea here is to eventually get rid of `dyn Diagnostic` and
`DiagnosticSink` infrastructure altogether, and just have a `enum
hir::Diagnostic` instead.
The problem with `dyn Diagnostic` is that it is defined in the lowest
level of the stack (hir_expand), but is used by the highest level (ide).
As a first step, we free hir_expand and hir_def from `dyn Diagnostic`
and kick the can up to `hir_ty`, as an intermediate state. The plan is
then to move DiagnosticSink similarly to the hir crate, and, as final
third step, remove its usage from the ide.
One currently unsolved problem is testing. You can notice that the test
which checks precise diagnostic ranges, unresolved_import_in_use_tree,
was moved to the ide layer. Logically, only IDE should have the infra to
render a specific range.
At the same time, the range is determined with the data produced in
hir_def and hir crates, so this layering is rather unfortunate. Working
on hir_def shouldn't require compiling `ide` for testing.
2021-05-23 20:31:59 +00:00
|
|
|
use nameres::diagnostics::DefDiagnosticKind;
|
2021-03-08 19:08:30 +00:00
|
|
|
use rustc_hash::FxHashSet;
|
|
|
|
use stdx::{format_to, impl_from};
|
|
|
|
use syntax::{
|
|
|
|
ast::{self, AttrsOwner, NameOwner},
|
internal: move diagnostics to hir
The idea here is to eventually get rid of `dyn Diagnostic` and
`DiagnosticSink` infrastructure altogether, and just have a `enum
hir::Diagnostic` instead.
The problem with `dyn Diagnostic` is that it is defined in the lowest
level of the stack (hir_expand), but is used by the highest level (ide).
As a first step, we free hir_expand and hir_def from `dyn Diagnostic`
and kick the can up to `hir_ty`, as an intermediate state. The plan is
then to move DiagnosticSink similarly to the hir crate, and, as final
third step, remove its usage from the ide.
One currently unsolved problem is testing. You can notice that the test
which checks precise diagnostic ranges, unresolved_import_in_use_tree,
was moved to the ide layer. Logically, only IDE should have the infra to
render a specific range.
At the same time, the range is determined with the data produced in
hir_def and hir crates, so this layering is rather unfortunate. Working
on hir_def shouldn't require compiling `ide` for testing.
2021-05-23 20:31:59 +00:00
|
|
|
AstNode, AstPtr, SmolStr, SyntaxKind, SyntaxNodePtr,
|
2021-03-08 19:08:30 +00:00
|
|
|
};
|
|
|
|
use tt::{Ident, Leaf, Literal, TokenTree};
|
|
|
|
|
|
|
|
use crate::db::{DefDatabase, HirDatabase};
|
|
|
|
|
2019-10-29 12:53:25 +00:00
|
|
|
pub use crate::{
|
2020-08-26 16:56:41 +00:00
|
|
|
attrs::{HasAttrs, Namespace},
|
2019-12-08 11:57:13 +00:00
|
|
|
has_source::HasSource,
|
2020-12-08 18:01:27 +00:00
|
|
|
semantics::{PathResolution, Semantics, SemanticsScope},
|
2018-11-28 00:42:26 +00:00
|
|
|
};
|
|
|
|
|
2021-03-08 19:08:30 +00:00
|
|
|
// Be careful with these re-exports.
|
|
|
|
//
|
|
|
|
// `hir` is the boundary between the compiler and the IDE. It should try hard to
|
|
|
|
// isolate the compiler from the ide, to allow the two to be refactored
|
|
|
|
// independently. Re-exporting something from the compiler is the sure way to
|
|
|
|
// breach the boundary.
|
|
|
|
//
|
|
|
|
// Generally, a refactoring which *removes* a name from this list is a good
|
|
|
|
// idea!
|
|
|
|
pub use {
|
2021-05-07 20:35:43 +00:00
|
|
|
cfg::{CfgAtom, CfgExpr, CfgOptions},
|
2021-03-08 19:08:30 +00:00
|
|
|
hir_def::{
|
|
|
|
adt::StructKind,
|
2021-03-19 20:23:57 +00:00
|
|
|
attr::{Attr, Attrs, AttrsWithOwner, Documentation},
|
2021-03-08 19:08:30 +00:00
|
|
|
body::scope::ExprScopes,
|
|
|
|
find_path::PrefixKind,
|
|
|
|
import_map,
|
|
|
|
item_scope::ItemInNs,
|
|
|
|
nameres::ModuleSource,
|
|
|
|
path::{ModPath, PathKind},
|
|
|
|
type_ref::{Mutability, TypeRef},
|
|
|
|
visibility::Visibility,
|
|
|
|
},
|
|
|
|
hir_expand::{
|
2021-03-08 19:14:52 +00:00
|
|
|
name::{known, Name},
|
2021-03-08 19:08:30 +00:00
|
|
|
ExpandResult, HirFileId, InFile, MacroCallId, MacroCallLoc, /* FIXME */ MacroDefId,
|
|
|
|
MacroFile, Origin,
|
|
|
|
},
|
|
|
|
hir_ty::display::HirDisplay,
|
2019-11-24 11:02:08 +00:00
|
|
|
};
|
2020-08-13 21:52:14 +00:00
|
|
|
|
|
|
|
// These are negative re-exports: pub using these names is forbidden, they
|
|
|
|
// should remain private to hir internals.
|
|
|
|
#[allow(unused)]
|
2021-03-08 19:14:52 +00:00
|
|
|
use {
|
|
|
|
hir_def::path::Path,
|
|
|
|
hir_expand::{hygiene::Hygiene, name::AsName},
|
|
|
|
};
|
2021-03-08 19:08:30 +00:00
|
|
|
|
|
|
|
/// hir::Crate describes a single crate. It's the main interface with which
|
|
|
|
/// a crate's dependencies interact. Mostly, it should be just a proxy for the
|
|
|
|
/// root module.
|
|
|
|
#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
|
|
|
|
pub struct Crate {
|
|
|
|
pub(crate) id: CrateId,
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Debug)]
|
|
|
|
pub struct CrateDependency {
|
|
|
|
pub krate: Crate,
|
|
|
|
pub name: Name,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Crate {
|
|
|
|
pub fn dependencies(self, db: &dyn HirDatabase) -> Vec<CrateDependency> {
|
|
|
|
db.crate_graph()[self.id]
|
|
|
|
.dependencies
|
|
|
|
.iter()
|
|
|
|
.map(|dep| {
|
|
|
|
let krate = Crate { id: dep.crate_id };
|
|
|
|
let name = dep.as_name();
|
|
|
|
CrateDependency { krate, name }
|
|
|
|
})
|
|
|
|
.collect()
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn reverse_dependencies(self, db: &dyn HirDatabase) -> Vec<Crate> {
|
|
|
|
let crate_graph = db.crate_graph();
|
|
|
|
crate_graph
|
|
|
|
.iter()
|
|
|
|
.filter(|&krate| {
|
|
|
|
crate_graph[krate].dependencies.iter().any(|it| it.crate_id == self.id)
|
|
|
|
})
|
|
|
|
.map(|id| Crate { id })
|
|
|
|
.collect()
|
|
|
|
}
|
|
|
|
|
2021-03-15 16:43:46 +00:00
|
|
|
pub fn transitive_reverse_dependencies(self, db: &dyn HirDatabase) -> Vec<Crate> {
|
2021-03-23 09:59:51 +00:00
|
|
|
db.crate_graph().transitive_rev_deps(self.id).into_iter().map(|id| Crate { id }).collect()
|
2021-03-15 16:43:46 +00:00
|
|
|
}
|
|
|
|
|
2021-03-08 19:08:30 +00:00
|
|
|
pub fn root_module(self, db: &dyn HirDatabase) -> Module {
|
|
|
|
let def_map = db.crate_def_map(self.id);
|
|
|
|
Module { id: def_map.module_id(def_map.root()) }
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn root_file(self, db: &dyn HirDatabase) -> FileId {
|
|
|
|
db.crate_graph()[self.id].root_file_id
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn edition(self, db: &dyn HirDatabase) -> Edition {
|
|
|
|
db.crate_graph()[self.id].edition
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn display_name(self, db: &dyn HirDatabase) -> Option<CrateDisplayName> {
|
|
|
|
db.crate_graph()[self.id].display_name.clone()
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn query_external_importables(
|
|
|
|
self,
|
|
|
|
db: &dyn DefDatabase,
|
|
|
|
query: import_map::Query,
|
|
|
|
) -> impl Iterator<Item = Either<ModuleDef, MacroDef>> {
|
|
|
|
import_map::search_dependencies(db, self.into(), query).into_iter().map(|item| match item {
|
|
|
|
ItemInNs::Types(mod_id) | ItemInNs::Values(mod_id) => Either::Left(mod_id.into()),
|
|
|
|
ItemInNs::Macros(mac_id) => Either::Right(mac_id.into()),
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn all(db: &dyn HirDatabase) -> Vec<Crate> {
|
|
|
|
db.crate_graph().iter().map(|id| Crate { id }).collect()
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Try to get the root URL of the documentation of a crate.
|
|
|
|
pub fn get_html_root_url(self: &Crate, db: &dyn HirDatabase) -> Option<String> {
|
|
|
|
// Look for #![doc(html_root_url = "...")]
|
|
|
|
let attrs = db.attrs(AttrDefId::ModuleId(self.root_module(db).into()));
|
|
|
|
let doc_attr_q = attrs.by_key("doc");
|
|
|
|
|
|
|
|
if !doc_attr_q.exists() {
|
|
|
|
return None;
|
|
|
|
}
|
|
|
|
|
|
|
|
let doc_url = doc_attr_q.tt_values().map(|tt| {
|
|
|
|
let name = tt.token_trees.iter()
|
|
|
|
.skip_while(|tt| !matches!(tt, TokenTree::Leaf(Leaf::Ident(Ident{text: ref ident, ..})) if ident == "html_root_url"))
|
|
|
|
.skip(2)
|
|
|
|
.next();
|
|
|
|
|
|
|
|
match name {
|
|
|
|
Some(TokenTree::Leaf(Leaf::Literal(Literal{ref text, ..}))) => Some(text),
|
|
|
|
_ => None
|
|
|
|
}
|
2021-03-21 12:13:34 +00:00
|
|
|
}).flatten().next();
|
2021-03-08 19:08:30 +00:00
|
|
|
|
|
|
|
doc_url.map(|s| s.trim_matches('"').trim_end_matches('/').to_owned() + "/")
|
|
|
|
}
|
2021-05-07 20:35:43 +00:00
|
|
|
|
|
|
|
pub fn cfg(&self, db: &dyn HirDatabase) -> CfgOptions {
|
|
|
|
db.crate_graph()[self.id].cfg_options.clone()
|
|
|
|
}
|
2021-03-08 19:08:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
|
|
|
|
pub struct Module {
|
|
|
|
pub(crate) id: ModuleId,
|
|
|
|
}
|
|
|
|
|
|
|
|
/// The defs which can be visible in the module.
|
|
|
|
#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
|
|
|
|
pub enum ModuleDef {
|
|
|
|
Module(Module),
|
|
|
|
Function(Function),
|
|
|
|
Adt(Adt),
|
|
|
|
// Can't be directly declared, but can be imported.
|
|
|
|
Variant(Variant),
|
|
|
|
Const(Const),
|
|
|
|
Static(Static),
|
|
|
|
Trait(Trait),
|
|
|
|
TypeAlias(TypeAlias),
|
|
|
|
BuiltinType(BuiltinType),
|
|
|
|
}
|
|
|
|
impl_from!(
|
|
|
|
Module,
|
|
|
|
Function,
|
|
|
|
Adt(Struct, Enum, Union),
|
|
|
|
Variant,
|
|
|
|
Const,
|
|
|
|
Static,
|
|
|
|
Trait,
|
|
|
|
TypeAlias,
|
|
|
|
BuiltinType
|
|
|
|
for ModuleDef
|
|
|
|
);
|
|
|
|
|
|
|
|
impl From<VariantDef> for ModuleDef {
|
|
|
|
fn from(var: VariantDef) -> Self {
|
|
|
|
match var {
|
|
|
|
VariantDef::Struct(t) => Adt::from(t).into(),
|
|
|
|
VariantDef::Union(t) => Adt::from(t).into(),
|
|
|
|
VariantDef::Variant(t) => t.into(),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl ModuleDef {
|
|
|
|
pub fn module(self, db: &dyn HirDatabase) -> Option<Module> {
|
|
|
|
match self {
|
|
|
|
ModuleDef::Module(it) => it.parent(db),
|
|
|
|
ModuleDef::Function(it) => Some(it.module(db)),
|
|
|
|
ModuleDef::Adt(it) => Some(it.module(db)),
|
|
|
|
ModuleDef::Variant(it) => Some(it.module(db)),
|
|
|
|
ModuleDef::Const(it) => Some(it.module(db)),
|
|
|
|
ModuleDef::Static(it) => Some(it.module(db)),
|
|
|
|
ModuleDef::Trait(it) => Some(it.module(db)),
|
|
|
|
ModuleDef::TypeAlias(it) => Some(it.module(db)),
|
|
|
|
ModuleDef::BuiltinType(_) => None,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn canonical_path(&self, db: &dyn HirDatabase) -> Option<String> {
|
2021-03-15 09:15:08 +00:00
|
|
|
let mut segments = vec![self.name(db)?.to_string()];
|
2021-03-08 19:08:30 +00:00
|
|
|
for m in self.module(db)?.path_to_root(db) {
|
|
|
|
segments.extend(m.name(db).map(|it| it.to_string()))
|
|
|
|
}
|
|
|
|
segments.reverse();
|
|
|
|
Some(segments.join("::"))
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn definition_visibility(&self, db: &dyn HirDatabase) -> Option<Visibility> {
|
|
|
|
let module = match self {
|
|
|
|
ModuleDef::Module(it) => it.parent(db)?,
|
|
|
|
ModuleDef::Function(it) => return Some(it.visibility(db)),
|
|
|
|
ModuleDef::Adt(it) => it.module(db),
|
|
|
|
ModuleDef::Variant(it) => {
|
|
|
|
let parent = it.parent_enum(db);
|
|
|
|
let module = it.module(db);
|
|
|
|
return module.visibility_of(db, &ModuleDef::Adt(Adt::Enum(parent)));
|
|
|
|
}
|
|
|
|
ModuleDef::Const(it) => return Some(it.visibility(db)),
|
|
|
|
ModuleDef::Static(it) => it.module(db),
|
|
|
|
ModuleDef::Trait(it) => it.module(db),
|
|
|
|
ModuleDef::TypeAlias(it) => return Some(it.visibility(db)),
|
|
|
|
ModuleDef::BuiltinType(_) => return None,
|
|
|
|
};
|
|
|
|
|
|
|
|
module.visibility_of(db, self)
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn name(self, db: &dyn HirDatabase) -> Option<Name> {
|
|
|
|
match self {
|
|
|
|
ModuleDef::Adt(it) => Some(it.name(db)),
|
|
|
|
ModuleDef::Trait(it) => Some(it.name(db)),
|
|
|
|
ModuleDef::Function(it) => Some(it.name(db)),
|
|
|
|
ModuleDef::Variant(it) => Some(it.name(db)),
|
|
|
|
ModuleDef::TypeAlias(it) => Some(it.name(db)),
|
|
|
|
ModuleDef::Module(it) => it.name(db),
|
|
|
|
ModuleDef::Const(it) => it.name(db),
|
|
|
|
ModuleDef::Static(it) => it.name(db),
|
|
|
|
ModuleDef::BuiltinType(it) => Some(it.name()),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn diagnostics(self, db: &dyn HirDatabase, sink: &mut DiagnosticSink) {
|
|
|
|
let id = match self {
|
|
|
|
ModuleDef::Adt(it) => match it {
|
|
|
|
Adt::Struct(it) => it.id.into(),
|
|
|
|
Adt::Enum(it) => it.id.into(),
|
|
|
|
Adt::Union(it) => it.id.into(),
|
|
|
|
},
|
|
|
|
ModuleDef::Trait(it) => it.id.into(),
|
|
|
|
ModuleDef::Function(it) => it.id.into(),
|
|
|
|
ModuleDef::TypeAlias(it) => it.id.into(),
|
|
|
|
ModuleDef::Module(it) => it.id.into(),
|
|
|
|
ModuleDef::Const(it) => it.id.into(),
|
|
|
|
ModuleDef::Static(it) => it.id.into(),
|
|
|
|
_ => return,
|
|
|
|
};
|
|
|
|
|
|
|
|
let module = match self.module(db) {
|
|
|
|
Some(it) => it,
|
|
|
|
None => return,
|
|
|
|
};
|
|
|
|
|
|
|
|
hir_ty::diagnostics::validate_module_item(db, module.id.krate(), id, sink)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Module {
|
|
|
|
/// Name of this module.
|
|
|
|
pub fn name(self, db: &dyn HirDatabase) -> Option<Name> {
|
|
|
|
let def_map = self.id.def_map(db.upcast());
|
|
|
|
let parent = def_map[self.id.local_id].parent?;
|
|
|
|
def_map[parent].children.iter().find_map(|(name, module_id)| {
|
|
|
|
if *module_id == self.id.local_id {
|
|
|
|
Some(name.clone())
|
|
|
|
} else {
|
|
|
|
None
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns the crate this module is part of.
|
|
|
|
pub fn krate(self) -> Crate {
|
|
|
|
Crate { id: self.id.krate() }
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Topmost parent of this module. Every module has a `crate_root`, but some
|
|
|
|
/// might be missing `krate`. This can happen if a module's file is not included
|
|
|
|
/// in the module tree of any target in `Cargo.toml`.
|
|
|
|
pub fn crate_root(self, db: &dyn HirDatabase) -> Module {
|
|
|
|
let def_map = db.crate_def_map(self.id.krate());
|
|
|
|
Module { id: def_map.module_id(def_map.root()) }
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Iterates over all child modules.
|
|
|
|
pub fn children(self, db: &dyn HirDatabase) -> impl Iterator<Item = Module> {
|
|
|
|
let def_map = self.id.def_map(db.upcast());
|
|
|
|
let children = def_map[self.id.local_id]
|
|
|
|
.children
|
|
|
|
.iter()
|
|
|
|
.map(|(_, module_id)| Module { id: def_map.module_id(*module_id) })
|
|
|
|
.collect::<Vec<_>>();
|
|
|
|
children.into_iter()
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Finds a parent module.
|
|
|
|
pub fn parent(self, db: &dyn HirDatabase) -> Option<Module> {
|
|
|
|
// FIXME: handle block expressions as modules (their parent is in a different DefMap)
|
|
|
|
let def_map = self.id.def_map(db.upcast());
|
|
|
|
let parent_id = def_map[self.id.local_id].parent?;
|
|
|
|
Some(Module { id: def_map.module_id(parent_id) })
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn path_to_root(self, db: &dyn HirDatabase) -> Vec<Module> {
|
|
|
|
let mut res = vec![self];
|
|
|
|
let mut curr = self;
|
|
|
|
while let Some(next) = curr.parent(db) {
|
|
|
|
res.push(next);
|
|
|
|
curr = next
|
|
|
|
}
|
|
|
|
res
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns a `ModuleScope`: a set of items, visible in this module.
|
|
|
|
pub fn scope(
|
|
|
|
self,
|
|
|
|
db: &dyn HirDatabase,
|
|
|
|
visible_from: Option<Module>,
|
|
|
|
) -> Vec<(Name, ScopeDef)> {
|
|
|
|
self.id.def_map(db.upcast())[self.id.local_id]
|
|
|
|
.scope
|
|
|
|
.entries()
|
|
|
|
.filter_map(|(name, def)| {
|
|
|
|
if let Some(m) = visible_from {
|
|
|
|
let filtered =
|
|
|
|
def.filter_visibility(|vis| vis.is_visible_from(db.upcast(), m.id));
|
|
|
|
if filtered.is_none() && !def.is_none() {
|
|
|
|
None
|
|
|
|
} else {
|
|
|
|
Some((name, filtered))
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
Some((name, def))
|
|
|
|
}
|
|
|
|
})
|
|
|
|
.flat_map(|(name, def)| {
|
|
|
|
ScopeDef::all_items(def).into_iter().map(move |item| (name.clone(), item))
|
|
|
|
})
|
|
|
|
.collect()
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn visibility_of(self, db: &dyn HirDatabase, def: &ModuleDef) -> Option<Visibility> {
|
|
|
|
self.id.def_map(db.upcast())[self.id.local_id].scope.visibility_of(def.clone().into())
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn diagnostics(self, db: &dyn HirDatabase, sink: &mut DiagnosticSink) {
|
|
|
|
let _p = profile::span("Module::diagnostics").detail(|| {
|
|
|
|
format!("{:?}", self.name(db).map_or("<unknown>".into(), |name| name.to_string()))
|
|
|
|
});
|
|
|
|
let def_map = self.id.def_map(db.upcast());
|
internal: move diagnostics to hir
The idea here is to eventually get rid of `dyn Diagnostic` and
`DiagnosticSink` infrastructure altogether, and just have a `enum
hir::Diagnostic` instead.
The problem with `dyn Diagnostic` is that it is defined in the lowest
level of the stack (hir_expand), but is used by the highest level (ide).
As a first step, we free hir_expand and hir_def from `dyn Diagnostic`
and kick the can up to `hir_ty`, as an intermediate state. The plan is
then to move DiagnosticSink similarly to the hir crate, and, as final
third step, remove its usage from the ide.
One currently unsolved problem is testing. You can notice that the test
which checks precise diagnostic ranges, unresolved_import_in_use_tree,
was moved to the ide layer. Logically, only IDE should have the infra to
render a specific range.
At the same time, the range is determined with the data produced in
hir_def and hir crates, so this layering is rather unfortunate. Working
on hir_def shouldn't require compiling `ide` for testing.
2021-05-23 20:31:59 +00:00
|
|
|
for diag in def_map.diagnostics() {
|
|
|
|
if diag.in_module != self.id.local_id {
|
|
|
|
// FIXME: This is accidentally quadratic.
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
match &diag.kind {
|
|
|
|
DefDiagnosticKind::UnresolvedModule { ast: declaration, candidate } => {
|
|
|
|
let decl = declaration.to_node(db.upcast());
|
|
|
|
sink.push(UnresolvedModule {
|
|
|
|
file: declaration.file_id,
|
|
|
|
decl: AstPtr::new(&decl),
|
|
|
|
candidate: candidate.clone(),
|
|
|
|
})
|
|
|
|
}
|
|
|
|
DefDiagnosticKind::UnresolvedExternCrate { ast } => {
|
|
|
|
let item = ast.to_node(db.upcast());
|
|
|
|
sink.push(UnresolvedExternCrate {
|
|
|
|
file: ast.file_id,
|
|
|
|
item: AstPtr::new(&item),
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2021-05-25 23:01:58 +00:00
|
|
|
DefDiagnosticKind::UnresolvedImport { id, index } => {
|
|
|
|
let file_id = id.file_id();
|
|
|
|
let item_tree = id.item_tree(db.upcast());
|
|
|
|
let import = &item_tree[id.value];
|
|
|
|
|
|
|
|
let use_tree = import.use_tree_to_ast(db.upcast(), file_id, *index);
|
|
|
|
sink.push(UnresolvedImport { file: file_id, node: AstPtr::new(&use_tree) });
|
internal: move diagnostics to hir
The idea here is to eventually get rid of `dyn Diagnostic` and
`DiagnosticSink` infrastructure altogether, and just have a `enum
hir::Diagnostic` instead.
The problem with `dyn Diagnostic` is that it is defined in the lowest
level of the stack (hir_expand), but is used by the highest level (ide).
As a first step, we free hir_expand and hir_def from `dyn Diagnostic`
and kick the can up to `hir_ty`, as an intermediate state. The plan is
then to move DiagnosticSink similarly to the hir crate, and, as final
third step, remove its usage from the ide.
One currently unsolved problem is testing. You can notice that the test
which checks precise diagnostic ranges, unresolved_import_in_use_tree,
was moved to the ide layer. Logically, only IDE should have the infra to
render a specific range.
At the same time, the range is determined with the data produced in
hir_def and hir crates, so this layering is rather unfortunate. Working
on hir_def shouldn't require compiling `ide` for testing.
2021-05-23 20:31:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
DefDiagnosticKind::UnconfiguredCode { ast, cfg, opts } => {
|
|
|
|
let item = ast.to_node(db.upcast());
|
|
|
|
sink.push(InactiveCode {
|
|
|
|
file: ast.file_id,
|
|
|
|
node: AstPtr::new(&item).into(),
|
|
|
|
cfg: cfg.clone(),
|
|
|
|
opts: opts.clone(),
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
DefDiagnosticKind::UnresolvedProcMacro { ast } => {
|
|
|
|
let mut precise_location = None;
|
|
|
|
let (file, ast, name) = match ast {
|
|
|
|
MacroCallKind::FnLike { ast_id, .. } => {
|
|
|
|
let node = ast_id.to_node(db.upcast());
|
|
|
|
(ast_id.file_id, SyntaxNodePtr::from(AstPtr::new(&node)), None)
|
|
|
|
}
|
|
|
|
MacroCallKind::Derive { ast_id, derive_name, .. } => {
|
|
|
|
let node = ast_id.to_node(db.upcast());
|
|
|
|
|
|
|
|
// Compute the precise location of the macro name's token in the derive
|
|
|
|
// list.
|
|
|
|
// FIXME: This does not handle paths to the macro, but neither does the
|
|
|
|
// rest of r-a.
|
|
|
|
let derive_attrs =
|
|
|
|
node.attrs().filter_map(|attr| match attr.as_simple_call() {
|
|
|
|
Some((name, args)) if name == "derive" => Some(args),
|
|
|
|
_ => None,
|
|
|
|
});
|
|
|
|
'outer: for attr in derive_attrs {
|
|
|
|
let tokens =
|
|
|
|
attr.syntax().children_with_tokens().filter_map(|elem| {
|
|
|
|
match elem {
|
|
|
|
syntax::NodeOrToken::Node(_) => None,
|
|
|
|
syntax::NodeOrToken::Token(tok) => Some(tok),
|
|
|
|
}
|
|
|
|
});
|
|
|
|
for token in tokens {
|
|
|
|
if token.kind() == SyntaxKind::IDENT
|
|
|
|
&& token.text() == derive_name.as_str()
|
|
|
|
{
|
|
|
|
precise_location = Some(token.text_range());
|
|
|
|
break 'outer;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
(
|
|
|
|
ast_id.file_id,
|
|
|
|
SyntaxNodePtr::from(AstPtr::new(&node)),
|
|
|
|
Some(derive_name.clone()),
|
|
|
|
)
|
|
|
|
}
|
2021-05-31 11:37:11 +00:00
|
|
|
MacroCallKind::Attr { ast_id, invoc_attr_index, attr_name, .. } => {
|
|
|
|
let node = ast_id.to_node(db.upcast());
|
|
|
|
let attr =
|
|
|
|
node.attrs().nth((*invoc_attr_index) as usize).unwrap_or_else(
|
|
|
|
|| panic!("cannot find attribute #{}", invoc_attr_index),
|
|
|
|
);
|
|
|
|
(
|
|
|
|
ast_id.file_id,
|
|
|
|
SyntaxNodePtr::from(AstPtr::new(&attr)),
|
|
|
|
Some(attr_name.clone()),
|
|
|
|
)
|
|
|
|
}
|
internal: move diagnostics to hir
The idea here is to eventually get rid of `dyn Diagnostic` and
`DiagnosticSink` infrastructure altogether, and just have a `enum
hir::Diagnostic` instead.
The problem with `dyn Diagnostic` is that it is defined in the lowest
level of the stack (hir_expand), but is used by the highest level (ide).
As a first step, we free hir_expand and hir_def from `dyn Diagnostic`
and kick the can up to `hir_ty`, as an intermediate state. The plan is
then to move DiagnosticSink similarly to the hir crate, and, as final
third step, remove its usage from the ide.
One currently unsolved problem is testing. You can notice that the test
which checks precise diagnostic ranges, unresolved_import_in_use_tree,
was moved to the ide layer. Logically, only IDE should have the infra to
render a specific range.
At the same time, the range is determined with the data produced in
hir_def and hir crates, so this layering is rather unfortunate. Working
on hir_def shouldn't require compiling `ide` for testing.
2021-05-23 20:31:59 +00:00
|
|
|
};
|
|
|
|
sink.push(UnresolvedProcMacro {
|
|
|
|
file,
|
|
|
|
node: ast,
|
|
|
|
precise_location,
|
|
|
|
macro_name: name,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
DefDiagnosticKind::UnresolvedMacroCall { ast, path } => {
|
|
|
|
let node = ast.to_node(db.upcast());
|
|
|
|
sink.push(UnresolvedMacroCall {
|
|
|
|
file: ast.file_id,
|
|
|
|
node: AstPtr::new(&node),
|
|
|
|
path: path.clone(),
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
DefDiagnosticKind::MacroError { ast, message } => {
|
|
|
|
let (file, ast) = match ast {
|
|
|
|
MacroCallKind::FnLike { ast_id, .. } => {
|
|
|
|
let node = ast_id.to_node(db.upcast());
|
|
|
|
(ast_id.file_id, SyntaxNodePtr::from(AstPtr::new(&node)))
|
|
|
|
}
|
2021-05-31 11:37:11 +00:00
|
|
|
MacroCallKind::Derive { ast_id, .. }
|
|
|
|
| MacroCallKind::Attr { ast_id, .. } => {
|
|
|
|
// FIXME: point to the attribute instead, this creates very large diagnostics
|
internal: move diagnostics to hir
The idea here is to eventually get rid of `dyn Diagnostic` and
`DiagnosticSink` infrastructure altogether, and just have a `enum
hir::Diagnostic` instead.
The problem with `dyn Diagnostic` is that it is defined in the lowest
level of the stack (hir_expand), but is used by the highest level (ide).
As a first step, we free hir_expand and hir_def from `dyn Diagnostic`
and kick the can up to `hir_ty`, as an intermediate state. The plan is
then to move DiagnosticSink similarly to the hir crate, and, as final
third step, remove its usage from the ide.
One currently unsolved problem is testing. You can notice that the test
which checks precise diagnostic ranges, unresolved_import_in_use_tree,
was moved to the ide layer. Logically, only IDE should have the infra to
render a specific range.
At the same time, the range is determined with the data produced in
hir_def and hir crates, so this layering is rather unfortunate. Working
on hir_def shouldn't require compiling `ide` for testing.
2021-05-23 20:31:59 +00:00
|
|
|
let node = ast_id.to_node(db.upcast());
|
|
|
|
(ast_id.file_id, SyntaxNodePtr::from(AstPtr::new(&node)))
|
|
|
|
}
|
|
|
|
};
|
|
|
|
sink.push(MacroError { file, node: ast, message: message.clone() });
|
|
|
|
}
|
2021-05-30 02:19:47 +00:00
|
|
|
|
|
|
|
DefDiagnosticKind::UnimplementedBuiltinMacro { ast } => {
|
|
|
|
let node = ast.to_node(db.upcast());
|
|
|
|
// Must have a name, otherwise we wouldn't emit it.
|
|
|
|
let name = node.name().expect("unimplemented builtin macro with no name");
|
|
|
|
let ptr = SyntaxNodePtr::from(AstPtr::new(&name));
|
|
|
|
sink.push(UnimplementedBuiltinMacro { file: ast.file_id, node: ptr });
|
|
|
|
}
|
internal: move diagnostics to hir
The idea here is to eventually get rid of `dyn Diagnostic` and
`DiagnosticSink` infrastructure altogether, and just have a `enum
hir::Diagnostic` instead.
The problem with `dyn Diagnostic` is that it is defined in the lowest
level of the stack (hir_expand), but is used by the highest level (ide).
As a first step, we free hir_expand and hir_def from `dyn Diagnostic`
and kick the can up to `hir_ty`, as an intermediate state. The plan is
then to move DiagnosticSink similarly to the hir crate, and, as final
third step, remove its usage from the ide.
One currently unsolved problem is testing. You can notice that the test
which checks precise diagnostic ranges, unresolved_import_in_use_tree,
was moved to the ide layer. Logically, only IDE should have the infra to
render a specific range.
At the same time, the range is determined with the data produced in
hir_def and hir crates, so this layering is rather unfortunate. Working
on hir_def shouldn't require compiling `ide` for testing.
2021-05-23 20:31:59 +00:00
|
|
|
}
|
|
|
|
}
|
2021-03-08 19:08:30 +00:00
|
|
|
for decl in self.declarations(db) {
|
|
|
|
match decl {
|
|
|
|
crate::ModuleDef::Function(f) => f.diagnostics(db, sink),
|
|
|
|
crate::ModuleDef::Module(m) => {
|
|
|
|
// Only add diagnostics from inline modules
|
|
|
|
if def_map[m.id.local_id].origin.is_inline() {
|
|
|
|
m.diagnostics(db, sink)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
_ => {
|
|
|
|
decl.diagnostics(db, sink);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for impl_def in self.impl_defs(db) {
|
|
|
|
for item in impl_def.items(db) {
|
|
|
|
if let AssocItem::Function(f) = item {
|
|
|
|
f.diagnostics(db, sink);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn declarations(self, db: &dyn HirDatabase) -> Vec<ModuleDef> {
|
|
|
|
let def_map = self.id.def_map(db.upcast());
|
|
|
|
def_map[self.id.local_id].scope.declarations().map(ModuleDef::from).collect()
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn impl_defs(self, db: &dyn HirDatabase) -> Vec<Impl> {
|
|
|
|
let def_map = self.id.def_map(db.upcast());
|
|
|
|
def_map[self.id.local_id].scope.impls().map(Impl::from).collect()
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Finds a path that can be used to refer to the given item from within
|
|
|
|
/// this module, if possible.
|
|
|
|
pub fn find_use_path(self, db: &dyn DefDatabase, item: impl Into<ItemInNs>) -> Option<ModPath> {
|
|
|
|
hir_def::find_path::find_path(db, item.into(), self.into())
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Finds a path that can be used to refer to the given item from within
|
|
|
|
/// this module, if possible. This is used for returning import paths for use-statements.
|
|
|
|
pub fn find_use_path_prefixed(
|
|
|
|
self,
|
|
|
|
db: &dyn DefDatabase,
|
|
|
|
item: impl Into<ItemInNs>,
|
|
|
|
prefix_kind: PrefixKind,
|
|
|
|
) -> Option<ModPath> {
|
|
|
|
hir_def::find_path::find_path_prefixed(db, item.into(), self.into(), prefix_kind)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
|
|
|
|
pub struct Field {
|
|
|
|
pub(crate) parent: VariantDef,
|
|
|
|
pub(crate) id: LocalFieldId,
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Debug, PartialEq, Eq)]
|
|
|
|
pub enum FieldSource {
|
|
|
|
Named(ast::RecordField),
|
|
|
|
Pos(ast::TupleField),
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Field {
|
|
|
|
pub fn name(&self, db: &dyn HirDatabase) -> Name {
|
|
|
|
self.parent.variant_data(db).fields()[self.id].name.clone()
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns the type as in the signature of the struct (i.e., with
|
2021-05-23 16:10:40 +00:00
|
|
|
/// placeholder types for type parameters). Only use this in the context of
|
2021-05-23 21:54:35 +00:00
|
|
|
/// the field definition.
|
2021-04-30 08:52:31 +00:00
|
|
|
pub fn ty(&self, db: &dyn HirDatabase) -> Type {
|
2021-03-08 19:08:30 +00:00
|
|
|
let var_id = self.parent.into();
|
|
|
|
let generic_def_id: GenericDefId = match self.parent {
|
|
|
|
VariantDef::Struct(it) => it.id.into(),
|
|
|
|
VariantDef::Union(it) => it.id.into(),
|
|
|
|
VariantDef::Variant(it) => it.parent.id.into(),
|
|
|
|
};
|
2021-04-04 11:16:16 +00:00
|
|
|
let substs = TyBuilder::type_params_subst(db, generic_def_id);
|
2021-04-05 16:49:26 +00:00
|
|
|
let ty = db.field_types(var_id)[self.id].clone().substitute(&Interner, &substs);
|
2021-03-08 19:08:30 +00:00
|
|
|
Type::new(db, self.parent.module(db).id.krate(), var_id, ty)
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn parent_def(&self, _db: &dyn HirDatabase) -> VariantDef {
|
|
|
|
self.parent
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl HasVisibility for Field {
|
|
|
|
fn visibility(&self, db: &dyn HirDatabase) -> Visibility {
|
|
|
|
let variant_data = self.parent.variant_data(db);
|
|
|
|
let visibility = &variant_data.fields()[self.id].visibility;
|
|
|
|
let parent_id: hir_def::VariantId = self.parent.into();
|
|
|
|
visibility.resolve(db.upcast(), &parent_id.resolver(db.upcast()))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
|
|
|
|
pub struct Struct {
|
|
|
|
pub(crate) id: StructId,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Struct {
|
|
|
|
pub fn module(self, db: &dyn HirDatabase) -> Module {
|
2021-03-09 18:09:02 +00:00
|
|
|
Module { id: self.id.lookup(db.upcast()).container }
|
2021-03-08 19:08:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn name(self, db: &dyn HirDatabase) -> Name {
|
|
|
|
db.struct_data(self.id).name.clone()
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn fields(self, db: &dyn HirDatabase) -> Vec<Field> {
|
|
|
|
db.struct_data(self.id)
|
|
|
|
.variant_data
|
|
|
|
.fields()
|
|
|
|
.iter()
|
|
|
|
.map(|(id, _)| Field { parent: self.into(), id })
|
|
|
|
.collect()
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn ty(self, db: &dyn HirDatabase) -> Type {
|
2021-03-09 18:09:02 +00:00
|
|
|
Type::from_def(db, self.id.lookup(db.upcast()).container.krate(), self.id)
|
2021-03-08 19:08:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn repr(self, db: &dyn HirDatabase) -> Option<ReprKind> {
|
|
|
|
db.struct_data(self.id).repr.clone()
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn kind(self, db: &dyn HirDatabase) -> StructKind {
|
|
|
|
self.variant_data(db).kind()
|
|
|
|
}
|
|
|
|
|
|
|
|
fn variant_data(self, db: &dyn HirDatabase) -> Arc<VariantData> {
|
|
|
|
db.struct_data(self.id).variant_data.clone()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-03-15 16:05:03 +00:00
|
|
|
impl HasVisibility for Struct {
|
|
|
|
fn visibility(&self, db: &dyn HirDatabase) -> Visibility {
|
|
|
|
db.struct_data(self.id).visibility.resolve(db.upcast(), &self.id.resolver(db.upcast()))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-03-08 19:08:30 +00:00
|
|
|
#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
|
|
|
|
pub struct Union {
|
|
|
|
pub(crate) id: UnionId,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Union {
|
|
|
|
pub fn name(self, db: &dyn HirDatabase) -> Name {
|
|
|
|
db.union_data(self.id).name.clone()
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn module(self, db: &dyn HirDatabase) -> Module {
|
2021-03-09 18:09:02 +00:00
|
|
|
Module { id: self.id.lookup(db.upcast()).container }
|
2021-03-08 19:08:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn ty(self, db: &dyn HirDatabase) -> Type {
|
2021-03-09 18:09:02 +00:00
|
|
|
Type::from_def(db, self.id.lookup(db.upcast()).container.krate(), self.id)
|
2021-03-08 19:08:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn fields(self, db: &dyn HirDatabase) -> Vec<Field> {
|
|
|
|
db.union_data(self.id)
|
|
|
|
.variant_data
|
|
|
|
.fields()
|
|
|
|
.iter()
|
|
|
|
.map(|(id, _)| Field { parent: self.into(), id })
|
|
|
|
.collect()
|
|
|
|
}
|
|
|
|
|
|
|
|
fn variant_data(self, db: &dyn HirDatabase) -> Arc<VariantData> {
|
|
|
|
db.union_data(self.id).variant_data.clone()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-03-15 16:05:03 +00:00
|
|
|
impl HasVisibility for Union {
|
|
|
|
fn visibility(&self, db: &dyn HirDatabase) -> Visibility {
|
|
|
|
db.union_data(self.id).visibility.resolve(db.upcast(), &self.id.resolver(db.upcast()))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-03-08 19:08:30 +00:00
|
|
|
#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
|
|
|
|
pub struct Enum {
|
|
|
|
pub(crate) id: EnumId,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Enum {
|
|
|
|
pub fn module(self, db: &dyn HirDatabase) -> Module {
|
2021-03-09 18:09:02 +00:00
|
|
|
Module { id: self.id.lookup(db.upcast()).container }
|
2021-03-08 19:08:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn name(self, db: &dyn HirDatabase) -> Name {
|
|
|
|
db.enum_data(self.id).name.clone()
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn variants(self, db: &dyn HirDatabase) -> Vec<Variant> {
|
|
|
|
db.enum_data(self.id).variants.iter().map(|(id, _)| Variant { parent: self, id }).collect()
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn ty(self, db: &dyn HirDatabase) -> Type {
|
2021-03-09 18:09:02 +00:00
|
|
|
Type::from_def(db, self.id.lookup(db.upcast()).container.krate(), self.id)
|
2021-03-08 19:08:30 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-03-15 16:05:03 +00:00
|
|
|
impl HasVisibility for Enum {
|
|
|
|
fn visibility(&self, db: &dyn HirDatabase) -> Visibility {
|
|
|
|
db.enum_data(self.id).visibility.resolve(db.upcast(), &self.id.resolver(db.upcast()))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-03-08 19:08:30 +00:00
|
|
|
#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
|
|
|
|
pub struct Variant {
|
|
|
|
pub(crate) parent: Enum,
|
|
|
|
pub(crate) id: LocalEnumVariantId,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Variant {
|
|
|
|
pub fn module(self, db: &dyn HirDatabase) -> Module {
|
|
|
|
self.parent.module(db)
|
|
|
|
}
|
2021-05-23 13:45:26 +00:00
|
|
|
|
2021-03-08 19:08:30 +00:00
|
|
|
pub fn parent_enum(self, _db: &dyn HirDatabase) -> Enum {
|
|
|
|
self.parent
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn name(self, db: &dyn HirDatabase) -> Name {
|
|
|
|
db.enum_data(self.parent.id).variants[self.id].name.clone()
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn fields(self, db: &dyn HirDatabase) -> Vec<Field> {
|
|
|
|
self.variant_data(db)
|
|
|
|
.fields()
|
|
|
|
.iter()
|
|
|
|
.map(|(id, _)| Field { parent: self.into(), id })
|
|
|
|
.collect()
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn kind(self, db: &dyn HirDatabase) -> StructKind {
|
|
|
|
self.variant_data(db).kind()
|
|
|
|
}
|
|
|
|
|
|
|
|
pub(crate) fn variant_data(self, db: &dyn HirDatabase) -> Arc<VariantData> {
|
|
|
|
db.enum_data(self.parent.id).variants[self.id].variant_data.clone()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// A Data Type
|
|
|
|
#[derive(Clone, Copy, Debug, PartialEq, Eq, Hash)]
|
|
|
|
pub enum Adt {
|
|
|
|
Struct(Struct),
|
|
|
|
Union(Union),
|
|
|
|
Enum(Enum),
|
|
|
|
}
|
|
|
|
impl_from!(Struct, Union, Enum for Adt);
|
|
|
|
|
|
|
|
impl Adt {
|
|
|
|
pub fn has_non_default_type_params(self, db: &dyn HirDatabase) -> bool {
|
|
|
|
let subst = db.generic_defaults(self.into());
|
2021-04-05 15:13:50 +00:00
|
|
|
subst.iter().any(|ty| ty.skip_binders().is_unknown())
|
2021-03-08 19:08:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Turns this ADT into a type. Any type parameters of the ADT will be
|
|
|
|
/// turned into unknown types, which is good for e.g. finding the most
|
|
|
|
/// general set of completions, but will not look very nice when printed.
|
|
|
|
pub fn ty(self, db: &dyn HirDatabase) -> Type {
|
|
|
|
let id = AdtId::from(self);
|
|
|
|
Type::from_def(db, id.module(db.upcast()).krate(), id)
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn module(self, db: &dyn HirDatabase) -> Module {
|
|
|
|
match self {
|
|
|
|
Adt::Struct(s) => s.module(db),
|
|
|
|
Adt::Union(s) => s.module(db),
|
|
|
|
Adt::Enum(e) => e.module(db),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn name(self, db: &dyn HirDatabase) -> Name {
|
|
|
|
match self {
|
|
|
|
Adt::Struct(s) => s.name(db),
|
|
|
|
Adt::Union(u) => u.name(db),
|
|
|
|
Adt::Enum(e) => e.name(db),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Clone, Copy, Debug, PartialEq, Eq, Hash)]
|
|
|
|
pub enum VariantDef {
|
|
|
|
Struct(Struct),
|
|
|
|
Union(Union),
|
|
|
|
Variant(Variant),
|
|
|
|
}
|
|
|
|
impl_from!(Struct, Union, Variant for VariantDef);
|
|
|
|
|
|
|
|
impl VariantDef {
|
|
|
|
pub fn fields(self, db: &dyn HirDatabase) -> Vec<Field> {
|
|
|
|
match self {
|
|
|
|
VariantDef::Struct(it) => it.fields(db),
|
|
|
|
VariantDef::Union(it) => it.fields(db),
|
|
|
|
VariantDef::Variant(it) => it.fields(db),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn module(self, db: &dyn HirDatabase) -> Module {
|
|
|
|
match self {
|
|
|
|
VariantDef::Struct(it) => it.module(db),
|
|
|
|
VariantDef::Union(it) => it.module(db),
|
|
|
|
VariantDef::Variant(it) => it.module(db),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn name(&self, db: &dyn HirDatabase) -> Name {
|
|
|
|
match self {
|
|
|
|
VariantDef::Struct(s) => s.name(db),
|
|
|
|
VariantDef::Union(u) => u.name(db),
|
|
|
|
VariantDef::Variant(e) => e.name(db),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub(crate) fn variant_data(self, db: &dyn HirDatabase) -> Arc<VariantData> {
|
|
|
|
match self {
|
|
|
|
VariantDef::Struct(it) => it.variant_data(db),
|
|
|
|
VariantDef::Union(it) => it.variant_data(db),
|
|
|
|
VariantDef::Variant(it) => it.variant_data(db),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// The defs which have a body.
|
|
|
|
#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
|
|
|
|
pub enum DefWithBody {
|
|
|
|
Function(Function),
|
|
|
|
Static(Static),
|
|
|
|
Const(Const),
|
|
|
|
}
|
|
|
|
impl_from!(Function, Const, Static for DefWithBody);
|
|
|
|
|
|
|
|
impl DefWithBody {
|
|
|
|
pub fn module(self, db: &dyn HirDatabase) -> Module {
|
|
|
|
match self {
|
|
|
|
DefWithBody::Const(c) => c.module(db),
|
|
|
|
DefWithBody::Function(f) => f.module(db),
|
|
|
|
DefWithBody::Static(s) => s.module(db),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn name(self, db: &dyn HirDatabase) -> Option<Name> {
|
|
|
|
match self {
|
|
|
|
DefWithBody::Function(f) => Some(f.name(db)),
|
|
|
|
DefWithBody::Static(s) => s.name(db),
|
|
|
|
DefWithBody::Const(c) => c.name(db),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
|
|
|
|
pub struct Function {
|
|
|
|
pub(crate) id: FunctionId,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Function {
|
|
|
|
pub fn module(self, db: &dyn HirDatabase) -> Module {
|
|
|
|
self.id.lookup(db.upcast()).module(db.upcast()).into()
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn name(self, db: &dyn HirDatabase) -> Name {
|
|
|
|
db.function_data(self.id).name.clone()
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Get this function's return type
|
|
|
|
pub fn ret_type(self, db: &dyn HirDatabase) -> Type {
|
|
|
|
let resolver = self.id.resolver(db.upcast());
|
2021-03-09 11:31:16 +00:00
|
|
|
let krate = self.id.lookup(db.upcast()).container.module(db.upcast()).krate();
|
2021-03-08 19:08:30 +00:00
|
|
|
let ret_type = &db.function_data(self.id).ret_type;
|
|
|
|
let ctx = hir_ty::TyLoweringContext::new(db, &resolver);
|
2021-03-13 21:44:36 +00:00
|
|
|
let ty = ctx.lower_ty(ret_type);
|
2021-03-09 11:31:16 +00:00
|
|
|
Type::new_with_resolver_inner(db, krate, &resolver, ty)
|
2021-03-08 19:08:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn self_param(self, db: &dyn HirDatabase) -> Option<SelfParam> {
|
2021-04-03 18:58:42 +00:00
|
|
|
if !db.function_data(self.id).has_self_param() {
|
2021-03-08 19:08:30 +00:00
|
|
|
return None;
|
|
|
|
}
|
|
|
|
Some(SelfParam { func: self.id })
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn assoc_fn_params(self, db: &dyn HirDatabase) -> Vec<Param> {
|
|
|
|
let resolver = self.id.resolver(db.upcast());
|
2021-03-09 11:31:16 +00:00
|
|
|
let krate = self.id.lookup(db.upcast()).container.module(db.upcast()).krate();
|
2021-03-08 19:08:30 +00:00
|
|
|
let ctx = hir_ty::TyLoweringContext::new(db, &resolver);
|
2021-03-13 19:38:11 +00:00
|
|
|
let environment = db.trait_environment(self.id.into());
|
2021-03-08 19:08:30 +00:00
|
|
|
db.function_data(self.id)
|
|
|
|
.params
|
|
|
|
.iter()
|
2021-03-14 12:03:39 +00:00
|
|
|
.enumerate()
|
|
|
|
.map(|(idx, type_ref)| {
|
2021-03-21 19:19:07 +00:00
|
|
|
let ty = Type { krate, env: environment.clone(), ty: ctx.lower_ty(type_ref) };
|
2021-03-14 12:03:39 +00:00
|
|
|
Param { func: self, ty, idx }
|
2021-03-08 19:08:30 +00:00
|
|
|
})
|
|
|
|
.collect()
|
|
|
|
}
|
2021-03-23 15:50:36 +00:00
|
|
|
|
2021-03-08 19:08:30 +00:00
|
|
|
pub fn method_params(self, db: &dyn HirDatabase) -> Option<Vec<Param>> {
|
|
|
|
if self.self_param(db).is_none() {
|
|
|
|
return None;
|
|
|
|
}
|
|
|
|
let mut res = self.assoc_fn_params(db);
|
|
|
|
res.remove(0);
|
|
|
|
Some(res)
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn is_unsafe(self, db: &dyn HirDatabase) -> bool {
|
2021-04-03 18:58:42 +00:00
|
|
|
db.function_data(self.id).is_unsafe()
|
2021-03-08 19:08:30 +00:00
|
|
|
}
|
|
|
|
|
2021-05-14 02:36:12 +00:00
|
|
|
pub fn is_async(self, db: &dyn HirDatabase) -> bool {
|
|
|
|
db.function_data(self.id).is_async()
|
|
|
|
}
|
|
|
|
|
2021-03-08 19:08:30 +00:00
|
|
|
pub fn diagnostics(self, db: &dyn HirDatabase, sink: &mut DiagnosticSink) {
|
|
|
|
let krate = self.module(db).id.krate();
|
internal: move diagnostics to hir
The idea here is to eventually get rid of `dyn Diagnostic` and
`DiagnosticSink` infrastructure altogether, and just have a `enum
hir::Diagnostic` instead.
The problem with `dyn Diagnostic` is that it is defined in the lowest
level of the stack (hir_expand), but is used by the highest level (ide).
As a first step, we free hir_expand and hir_def from `dyn Diagnostic`
and kick the can up to `hir_ty`, as an intermediate state. The plan is
then to move DiagnosticSink similarly to the hir crate, and, as final
third step, remove its usage from the ide.
One currently unsolved problem is testing. You can notice that the test
which checks precise diagnostic ranges, unresolved_import_in_use_tree,
was moved to the ide layer. Logically, only IDE should have the infra to
render a specific range.
At the same time, the range is determined with the data produced in
hir_def and hir crates, so this layering is rather unfortunate. Working
on hir_def shouldn't require compiling `ide` for testing.
2021-05-23 20:31:59 +00:00
|
|
|
|
|
|
|
let source_map = db.body_with_source_map(self.id.into()).1;
|
|
|
|
for diag in source_map.diagnostics() {
|
|
|
|
match diag {
|
|
|
|
BodyDiagnostic::InactiveCode { node, cfg, opts } => sink.push(InactiveCode {
|
|
|
|
file: node.file_id,
|
|
|
|
node: node.value.clone(),
|
|
|
|
cfg: cfg.clone(),
|
|
|
|
opts: opts.clone(),
|
|
|
|
}),
|
|
|
|
BodyDiagnostic::MacroError { node, message } => sink.push(MacroError {
|
|
|
|
file: node.file_id,
|
|
|
|
node: node.value.clone().into(),
|
|
|
|
message: message.to_string(),
|
|
|
|
}),
|
|
|
|
BodyDiagnostic::UnresolvedProcMacro { node } => sink.push(UnresolvedProcMacro {
|
|
|
|
file: node.file_id,
|
|
|
|
node: node.value.clone().into(),
|
|
|
|
precise_location: None,
|
|
|
|
macro_name: None,
|
|
|
|
}),
|
|
|
|
BodyDiagnostic::UnresolvedMacroCall { node, path } => {
|
|
|
|
sink.push(UnresolvedMacroCall {
|
|
|
|
file: node.file_id,
|
|
|
|
node: node.value.clone(),
|
|
|
|
path: path.clone(),
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-03-08 19:08:30 +00:00
|
|
|
hir_ty::diagnostics::validate_module_item(db, krate, self.id.into(), sink);
|
|
|
|
hir_ty::diagnostics::validate_body(db, self.id.into(), sink);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Whether this function declaration has a definition.
|
|
|
|
///
|
|
|
|
/// This is false in the case of required (not provided) trait methods.
|
|
|
|
pub fn has_body(self, db: &dyn HirDatabase) -> bool {
|
2021-04-03 18:58:42 +00:00
|
|
|
db.function_data(self.id).has_body()
|
2021-03-08 19:08:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/// A textual representation of the HIR of this function for debugging purposes.
|
|
|
|
pub fn debug_hir(self, db: &dyn HirDatabase) -> String {
|
|
|
|
let body = db.body(self.id.into());
|
|
|
|
|
|
|
|
let mut result = String::new();
|
|
|
|
format_to!(result, "HIR expressions in the body of `{}`:\n", self.name(db));
|
|
|
|
for (id, expr) in body.exprs.iter() {
|
|
|
|
format_to!(result, "{:?}: {:?}\n", id, expr);
|
|
|
|
}
|
|
|
|
|
|
|
|
result
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Note: logically, this belongs to `hir_ty`, but we are not using it there yet.
|
|
|
|
pub enum Access {
|
|
|
|
Shared,
|
|
|
|
Exclusive,
|
|
|
|
Owned,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl From<hir_ty::Mutability> for Access {
|
|
|
|
fn from(mutability: hir_ty::Mutability) -> Access {
|
|
|
|
match mutability {
|
|
|
|
hir_ty::Mutability::Not => Access::Shared,
|
|
|
|
hir_ty::Mutability::Mut => Access::Exclusive,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-03-23 15:50:36 +00:00
|
|
|
#[derive(Clone, Debug)]
|
2021-03-08 19:08:30 +00:00
|
|
|
pub struct Param {
|
2021-03-14 12:03:39 +00:00
|
|
|
func: Function,
|
|
|
|
/// The index in parameter list, including self parameter.
|
|
|
|
idx: usize,
|
2021-03-08 19:08:30 +00:00
|
|
|
ty: Type,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Param {
|
|
|
|
pub fn ty(&self) -> &Type {
|
|
|
|
&self.ty
|
|
|
|
}
|
2021-03-14 12:03:39 +00:00
|
|
|
|
2021-03-23 15:50:36 +00:00
|
|
|
pub fn as_local(&self, db: &dyn HirDatabase) -> Local {
|
|
|
|
let parent = DefWithBodyId::FunctionId(self.func.into());
|
|
|
|
let body = db.body(parent);
|
|
|
|
Local { parent, pat_id: body.params[self.idx] }
|
|
|
|
}
|
|
|
|
|
2021-03-14 12:03:39 +00:00
|
|
|
pub fn pattern_source(&self, db: &dyn HirDatabase) -> Option<ast::Pat> {
|
2021-03-23 15:50:36 +00:00
|
|
|
self.source(db).and_then(|p| p.value.pat())
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn source(&self, db: &dyn HirDatabase) -> Option<InFile<ast::Param>> {
|
|
|
|
let InFile { file_id, value } = self.func.source(db)?;
|
|
|
|
let params = value.param_list()?;
|
2021-03-14 12:03:39 +00:00
|
|
|
if params.self_param().is_some() {
|
2021-03-23 15:50:36 +00:00
|
|
|
params.params().nth(self.idx.checked_sub(1)?)
|
2021-03-14 12:03:39 +00:00
|
|
|
} else {
|
2021-03-23 15:50:36 +00:00
|
|
|
params.params().nth(self.idx)
|
2021-03-14 12:03:39 +00:00
|
|
|
}
|
2021-03-23 15:50:36 +00:00
|
|
|
.map(|value| InFile { file_id, value })
|
2021-03-14 12:03:39 +00:00
|
|
|
}
|
2021-03-08 19:08:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
|
|
|
|
pub struct SelfParam {
|
|
|
|
func: FunctionId,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl SelfParam {
|
|
|
|
pub fn access(self, db: &dyn HirDatabase) -> Access {
|
|
|
|
let func_data = db.function_data(self.func);
|
|
|
|
func_data
|
|
|
|
.params
|
|
|
|
.first()
|
2021-04-01 17:46:43 +00:00
|
|
|
.map(|param| match &**param {
|
2021-03-08 19:08:30 +00:00
|
|
|
TypeRef::Reference(.., mutability) => match mutability {
|
|
|
|
hir_def::type_ref::Mutability::Shared => Access::Shared,
|
|
|
|
hir_def::type_ref::Mutability::Mut => Access::Exclusive,
|
|
|
|
},
|
|
|
|
_ => Access::Owned,
|
|
|
|
})
|
|
|
|
.unwrap_or(Access::Owned)
|
|
|
|
}
|
2021-03-07 00:56:07 +00:00
|
|
|
|
|
|
|
pub fn display(self, db: &dyn HirDatabase) -> &'static str {
|
|
|
|
match self.access(db) {
|
|
|
|
Access::Shared => "&self",
|
|
|
|
Access::Exclusive => "&mut self",
|
|
|
|
Access::Owned => "self",
|
|
|
|
}
|
|
|
|
}
|
2021-03-23 16:39:43 +00:00
|
|
|
|
|
|
|
pub fn source(&self, db: &dyn HirDatabase) -> Option<InFile<ast::SelfParam>> {
|
|
|
|
let InFile { file_id, value } = Function::from(self.func).source(db)?;
|
|
|
|
value
|
|
|
|
.param_list()
|
|
|
|
.and_then(|params| params.self_param())
|
|
|
|
.map(|value| InFile { file_id, value })
|
|
|
|
}
|
2021-03-08 19:08:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
impl HasVisibility for Function {
|
|
|
|
fn visibility(&self, db: &dyn HirDatabase) -> Visibility {
|
|
|
|
let function_data = db.function_data(self.id);
|
|
|
|
let visibility = &function_data.visibility;
|
|
|
|
visibility.resolve(db.upcast(), &self.id.resolver(db.upcast()))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
|
|
|
|
pub struct Const {
|
|
|
|
pub(crate) id: ConstId,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Const {
|
|
|
|
pub fn module(self, db: &dyn HirDatabase) -> Module {
|
|
|
|
Module { id: self.id.lookup(db.upcast()).module(db.upcast()) }
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn name(self, db: &dyn HirDatabase) -> Option<Name> {
|
|
|
|
db.const_data(self.id).name.clone()
|
|
|
|
}
|
2021-03-15 16:05:03 +00:00
|
|
|
|
|
|
|
pub fn type_ref(self, db: &dyn HirDatabase) -> TypeRef {
|
2021-04-01 17:46:43 +00:00
|
|
|
db.const_data(self.id).type_ref.as_ref().clone()
|
2021-03-15 16:05:03 +00:00
|
|
|
}
|
2021-03-08 19:08:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
impl HasVisibility for Const {
|
|
|
|
fn visibility(&self, db: &dyn HirDatabase) -> Visibility {
|
|
|
|
let function_data = db.const_data(self.id);
|
|
|
|
let visibility = &function_data.visibility;
|
|
|
|
visibility.resolve(db.upcast(), &self.id.resolver(db.upcast()))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
|
|
|
|
pub struct Static {
|
|
|
|
pub(crate) id: StaticId,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Static {
|
|
|
|
pub fn module(self, db: &dyn HirDatabase) -> Module {
|
|
|
|
Module { id: self.id.lookup(db.upcast()).module(db.upcast()) }
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn name(self, db: &dyn HirDatabase) -> Option<Name> {
|
|
|
|
db.static_data(self.id).name.clone()
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn is_mut(self, db: &dyn HirDatabase) -> bool {
|
|
|
|
db.static_data(self.id).mutable
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-03-15 16:05:03 +00:00
|
|
|
impl HasVisibility for Static {
|
|
|
|
fn visibility(&self, db: &dyn HirDatabase) -> Visibility {
|
|
|
|
db.static_data(self.id).visibility.resolve(db.upcast(), &self.id.resolver(db.upcast()))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-03-08 19:08:30 +00:00
|
|
|
#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
|
|
|
|
pub struct Trait {
|
|
|
|
pub(crate) id: TraitId,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Trait {
|
|
|
|
pub fn module(self, db: &dyn HirDatabase) -> Module {
|
2021-03-09 18:09:02 +00:00
|
|
|
Module { id: self.id.lookup(db.upcast()).container }
|
2021-03-08 19:08:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn name(self, db: &dyn HirDatabase) -> Name {
|
|
|
|
db.trait_data(self.id).name.clone()
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn items(self, db: &dyn HirDatabase) -> Vec<AssocItem> {
|
|
|
|
db.trait_data(self.id).items.iter().map(|(_name, it)| (*it).into()).collect()
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn is_auto(self, db: &dyn HirDatabase) -> bool {
|
2021-03-15 16:05:03 +00:00
|
|
|
db.trait_data(self.id).is_auto
|
|
|
|
}
|
2021-05-23 11:43:23 +00:00
|
|
|
|
|
|
|
pub fn is_unsafe(&self, db: &dyn HirDatabase) -> bool {
|
|
|
|
db.trait_data(self.id).is_unsafe
|
|
|
|
}
|
2021-03-15 16:05:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
impl HasVisibility for Trait {
|
|
|
|
fn visibility(&self, db: &dyn HirDatabase) -> Visibility {
|
|
|
|
db.trait_data(self.id).visibility.resolve(db.upcast(), &self.id.resolver(db.upcast()))
|
2021-03-08 19:08:30 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
|
|
|
|
pub struct TypeAlias {
|
|
|
|
pub(crate) id: TypeAliasId,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl TypeAlias {
|
|
|
|
pub fn has_non_default_type_params(self, db: &dyn HirDatabase) -> bool {
|
|
|
|
let subst = db.generic_defaults(self.id.into());
|
2021-04-05 15:13:50 +00:00
|
|
|
subst.iter().any(|ty| ty.skip_binders().is_unknown())
|
2021-03-08 19:08:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn module(self, db: &dyn HirDatabase) -> Module {
|
|
|
|
Module { id: self.id.lookup(db.upcast()).module(db.upcast()) }
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn type_ref(self, db: &dyn HirDatabase) -> Option<TypeRef> {
|
2021-04-01 17:46:43 +00:00
|
|
|
db.type_alias_data(self.id).type_ref.as_deref().cloned()
|
2021-03-08 19:08:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn ty(self, db: &dyn HirDatabase) -> Type {
|
|
|
|
Type::from_def(db, self.id.lookup(db.upcast()).module(db.upcast()).krate(), self.id)
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn name(self, db: &dyn HirDatabase) -> Name {
|
|
|
|
db.type_alias_data(self.id).name.clone()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl HasVisibility for TypeAlias {
|
|
|
|
fn visibility(&self, db: &dyn HirDatabase) -> Visibility {
|
|
|
|
let function_data = db.type_alias_data(self.id);
|
|
|
|
let visibility = &function_data.visibility;
|
|
|
|
visibility.resolve(db.upcast(), &self.id.resolver(db.upcast()))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
|
|
|
|
pub struct BuiltinType {
|
|
|
|
pub(crate) inner: hir_def::builtin_type::BuiltinType,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl BuiltinType {
|
|
|
|
pub fn ty(self, db: &dyn HirDatabase, module: Module) -> Type {
|
|
|
|
let resolver = module.id.resolver(db.upcast());
|
2021-04-03 19:32:22 +00:00
|
|
|
Type::new_with_resolver(db, &resolver, TyBuilder::builtin(self.inner))
|
2021-03-08 19:08:30 +00:00
|
|
|
.expect("crate not present in resolver")
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn name(self) -> Name {
|
|
|
|
self.inner.as_name()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-03-22 14:56:59 +00:00
|
|
|
#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
|
|
|
|
pub enum MacroKind {
|
2021-05-29 18:32:57 +00:00
|
|
|
/// `macro_rules!` or Macros 2.0 macro.
|
2021-03-22 14:56:59 +00:00
|
|
|
Declarative,
|
2021-05-29 18:32:57 +00:00
|
|
|
/// A built-in or custom derive.
|
2021-03-22 14:56:59 +00:00
|
|
|
Derive,
|
2021-05-29 18:32:57 +00:00
|
|
|
/// A built-in function-like macro.
|
2021-03-22 14:56:59 +00:00
|
|
|
BuiltIn,
|
2021-05-29 18:32:57 +00:00
|
|
|
/// A procedural attribute macro.
|
|
|
|
Attr,
|
|
|
|
/// A function-like procedural macro.
|
|
|
|
ProcMacro,
|
2021-03-22 14:56:59 +00:00
|
|
|
}
|
|
|
|
|
2021-03-08 19:08:30 +00:00
|
|
|
#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
|
|
|
|
pub struct MacroDef {
|
|
|
|
pub(crate) id: MacroDefId,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl MacroDef {
|
|
|
|
/// FIXME: right now, this just returns the root module of the crate that
|
|
|
|
/// defines this macro. The reasons for this is that macros are expanded
|
|
|
|
/// early, in `hir_expand`, where modules simply do not exist yet.
|
|
|
|
pub fn module(self, db: &dyn HirDatabase) -> Option<Module> {
|
|
|
|
let krate = self.id.krate;
|
|
|
|
let def_map = db.crate_def_map(krate);
|
|
|
|
let module_id = def_map.root();
|
|
|
|
Some(Module { id: def_map.module_id(module_id) })
|
|
|
|
}
|
|
|
|
|
|
|
|
/// XXX: this parses the file
|
|
|
|
pub fn name(self, db: &dyn HirDatabase) -> Option<Name> {
|
2021-03-18 15:11:18 +00:00
|
|
|
match self.source(db)?.value {
|
|
|
|
Either::Left(it) => it.name().map(|it| it.as_name()),
|
|
|
|
Either::Right(it) => it.name().map(|it| it.as_name()),
|
|
|
|
}
|
2021-03-08 19:08:30 +00:00
|
|
|
}
|
|
|
|
|
2021-03-22 14:56:59 +00:00
|
|
|
pub fn kind(&self) -> MacroKind {
|
|
|
|
match self.id.kind {
|
|
|
|
MacroDefKind::Declarative(_) => MacroKind::Declarative,
|
2021-05-29 18:32:57 +00:00
|
|
|
MacroDefKind::BuiltIn(_, _) | MacroDefKind::BuiltInEager(_, _) => MacroKind::BuiltIn,
|
2021-03-22 14:56:59 +00:00
|
|
|
MacroDefKind::BuiltInDerive(_, _) => MacroKind::Derive,
|
2021-05-29 18:32:57 +00:00
|
|
|
MacroDefKind::ProcMacro(_, base_db::ProcMacroKind::CustomDerive, _) => {
|
|
|
|
MacroKind::Derive
|
|
|
|
}
|
|
|
|
MacroDefKind::ProcMacro(_, base_db::ProcMacroKind::Attr, _) => MacroKind::Attr,
|
|
|
|
MacroDefKind::ProcMacro(_, base_db::ProcMacroKind::FuncLike, _) => MacroKind::ProcMacro,
|
2021-03-22 14:56:59 +00:00
|
|
|
}
|
2021-03-21 19:08:08 +00:00
|
|
|
}
|
2021-06-08 15:31:47 +00:00
|
|
|
|
|
|
|
pub fn is_fn_like(&self) -> bool {
|
|
|
|
match self.kind() {
|
|
|
|
MacroKind::Declarative | MacroKind::BuiltIn | MacroKind::ProcMacro => true,
|
|
|
|
MacroKind::Attr | MacroKind::Derive => false,
|
|
|
|
}
|
|
|
|
}
|
2021-03-08 19:08:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Invariant: `inner.as_assoc_item(db).is_some()`
|
|
|
|
/// We do not actively enforce this invariant.
|
|
|
|
#[derive(Debug, Copy, Clone, PartialEq, Eq, Hash)]
|
|
|
|
pub enum AssocItem {
|
|
|
|
Function(Function),
|
|
|
|
Const(Const),
|
|
|
|
TypeAlias(TypeAlias),
|
|
|
|
}
|
2021-02-28 22:05:22 +00:00
|
|
|
#[derive(Debug)]
|
2021-03-08 19:08:30 +00:00
|
|
|
pub enum AssocItemContainer {
|
|
|
|
Trait(Trait),
|
|
|
|
Impl(Impl),
|
|
|
|
}
|
|
|
|
pub trait AsAssocItem {
|
|
|
|
fn as_assoc_item(self, db: &dyn HirDatabase) -> Option<AssocItem>;
|
|
|
|
}
|
|
|
|
|
|
|
|
impl AsAssocItem for Function {
|
|
|
|
fn as_assoc_item(self, db: &dyn HirDatabase) -> Option<AssocItem> {
|
|
|
|
as_assoc_item(db, AssocItem::Function, self.id)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
impl AsAssocItem for Const {
|
|
|
|
fn as_assoc_item(self, db: &dyn HirDatabase) -> Option<AssocItem> {
|
|
|
|
as_assoc_item(db, AssocItem::Const, self.id)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
impl AsAssocItem for TypeAlias {
|
|
|
|
fn as_assoc_item(self, db: &dyn HirDatabase) -> Option<AssocItem> {
|
|
|
|
as_assoc_item(db, AssocItem::TypeAlias, self.id)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
impl AsAssocItem for ModuleDef {
|
|
|
|
fn as_assoc_item(self, db: &dyn HirDatabase) -> Option<AssocItem> {
|
|
|
|
match self {
|
|
|
|
ModuleDef::Function(it) => it.as_assoc_item(db),
|
|
|
|
ModuleDef::Const(it) => it.as_assoc_item(db),
|
|
|
|
ModuleDef::TypeAlias(it) => it.as_assoc_item(db),
|
|
|
|
_ => None,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
fn as_assoc_item<ID, DEF, CTOR, AST>(db: &dyn HirDatabase, ctor: CTOR, id: ID) -> Option<AssocItem>
|
|
|
|
where
|
|
|
|
ID: Lookup<Data = AssocItemLoc<AST>>,
|
|
|
|
DEF: From<ID>,
|
|
|
|
CTOR: FnOnce(DEF) -> AssocItem,
|
|
|
|
AST: ItemTreeNode,
|
|
|
|
{
|
|
|
|
match id.lookup(db.upcast()).container {
|
|
|
|
AssocContainerId::TraitId(_) | AssocContainerId::ImplId(_) => Some(ctor(DEF::from(id))),
|
2021-03-09 17:27:16 +00:00
|
|
|
AssocContainerId::ModuleId(_) => None,
|
2021-03-08 19:08:30 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl AssocItem {
|
|
|
|
pub fn name(self, db: &dyn HirDatabase) -> Option<Name> {
|
|
|
|
match self {
|
|
|
|
AssocItem::Function(it) => Some(it.name(db)),
|
|
|
|
AssocItem::Const(it) => it.name(db),
|
|
|
|
AssocItem::TypeAlias(it) => Some(it.name(db)),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
pub fn module(self, db: &dyn HirDatabase) -> Module {
|
|
|
|
match self {
|
|
|
|
AssocItem::Function(f) => f.module(db),
|
|
|
|
AssocItem::Const(c) => c.module(db),
|
|
|
|
AssocItem::TypeAlias(t) => t.module(db),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
pub fn container(self, db: &dyn HirDatabase) -> AssocItemContainer {
|
|
|
|
let container = match self {
|
|
|
|
AssocItem::Function(it) => it.id.lookup(db.upcast()).container,
|
|
|
|
AssocItem::Const(it) => it.id.lookup(db.upcast()).container,
|
|
|
|
AssocItem::TypeAlias(it) => it.id.lookup(db.upcast()).container,
|
|
|
|
};
|
|
|
|
match container {
|
|
|
|
AssocContainerId::TraitId(id) => AssocItemContainer::Trait(id.into()),
|
|
|
|
AssocContainerId::ImplId(id) => AssocItemContainer::Impl(id.into()),
|
2021-03-09 17:27:16 +00:00
|
|
|
AssocContainerId::ModuleId(_) => panic!("invalid AssocItem"),
|
2021-03-08 19:08:30 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn containing_trait(self, db: &dyn HirDatabase) -> Option<Trait> {
|
|
|
|
match self.container(db) {
|
|
|
|
AssocItemContainer::Trait(t) => Some(t),
|
|
|
|
_ => None,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl HasVisibility for AssocItem {
|
|
|
|
fn visibility(&self, db: &dyn HirDatabase) -> Visibility {
|
|
|
|
match self {
|
|
|
|
AssocItem::Function(f) => f.visibility(db),
|
|
|
|
AssocItem::Const(c) => c.visibility(db),
|
|
|
|
AssocItem::TypeAlias(t) => t.visibility(db),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Clone, Copy, PartialEq, Eq, Debug, Hash)]
|
|
|
|
pub enum GenericDef {
|
|
|
|
Function(Function),
|
|
|
|
Adt(Adt),
|
|
|
|
Trait(Trait),
|
|
|
|
TypeAlias(TypeAlias),
|
|
|
|
Impl(Impl),
|
|
|
|
// enum variants cannot have generics themselves, but their parent enums
|
|
|
|
// can, and this makes some code easier to write
|
|
|
|
Variant(Variant),
|
|
|
|
// consts can have type parameters from their parents (i.e. associated consts of traits)
|
|
|
|
Const(Const),
|
|
|
|
}
|
|
|
|
impl_from!(
|
|
|
|
Function,
|
|
|
|
Adt(Struct, Enum, Union),
|
|
|
|
Trait,
|
|
|
|
TypeAlias,
|
|
|
|
Impl,
|
|
|
|
Variant,
|
|
|
|
Const
|
|
|
|
for GenericDef
|
|
|
|
);
|
|
|
|
|
|
|
|
impl GenericDef {
|
|
|
|
pub fn params(self, db: &dyn HirDatabase) -> Vec<GenericParam> {
|
|
|
|
let generics = db.generic_params(self.into());
|
|
|
|
let ty_params = generics
|
|
|
|
.types
|
|
|
|
.iter()
|
|
|
|
.map(|(local_id, _)| TypeParam { id: TypeParamId { parent: self.into(), local_id } })
|
|
|
|
.map(GenericParam::TypeParam);
|
|
|
|
let lt_params = generics
|
|
|
|
.lifetimes
|
|
|
|
.iter()
|
|
|
|
.map(|(local_id, _)| LifetimeParam {
|
|
|
|
id: LifetimeParamId { parent: self.into(), local_id },
|
|
|
|
})
|
|
|
|
.map(GenericParam::LifetimeParam);
|
|
|
|
let const_params = generics
|
|
|
|
.consts
|
|
|
|
.iter()
|
|
|
|
.map(|(local_id, _)| ConstParam { id: ConstParamId { parent: self.into(), local_id } })
|
|
|
|
.map(GenericParam::ConstParam);
|
|
|
|
ty_params.chain(lt_params).chain(const_params).collect()
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn type_params(self, db: &dyn HirDatabase) -> Vec<TypeParam> {
|
|
|
|
let generics = db.generic_params(self.into());
|
|
|
|
generics
|
|
|
|
.types
|
|
|
|
.iter()
|
|
|
|
.map(|(local_id, _)| TypeParam { id: TypeParamId { parent: self.into(), local_id } })
|
|
|
|
.collect()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Clone, Copy, Debug, PartialEq, Eq, Hash)]
|
|
|
|
pub struct Local {
|
|
|
|
pub(crate) parent: DefWithBodyId,
|
|
|
|
pub(crate) pat_id: PatId,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Local {
|
|
|
|
pub fn is_param(self, db: &dyn HirDatabase) -> bool {
|
|
|
|
let src = self.source(db);
|
|
|
|
match src.value {
|
|
|
|
Either::Left(bind_pat) => {
|
|
|
|
bind_pat.syntax().ancestors().any(|it| ast::Param::can_cast(it.kind()))
|
|
|
|
}
|
|
|
|
Either::Right(_self_param) => true,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-03-23 16:39:43 +00:00
|
|
|
pub fn as_self_param(self, db: &dyn HirDatabase) -> Option<SelfParam> {
|
|
|
|
match self.parent {
|
|
|
|
DefWithBodyId::FunctionId(func) if self.is_self(db) => Some(SelfParam { func }),
|
|
|
|
_ => None,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-03-08 19:08:30 +00:00
|
|
|
// FIXME: why is this an option? It shouldn't be?
|
|
|
|
pub fn name(self, db: &dyn HirDatabase) -> Option<Name> {
|
2021-03-17 00:27:56 +00:00
|
|
|
let body = db.body(self.parent);
|
2021-03-08 19:08:30 +00:00
|
|
|
match &body[self.pat_id] {
|
|
|
|
Pat::Bind { name, .. } => Some(name.clone()),
|
|
|
|
_ => None,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn is_self(self, db: &dyn HirDatabase) -> bool {
|
|
|
|
self.name(db) == Some(name![self])
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn is_mut(self, db: &dyn HirDatabase) -> bool {
|
2021-03-17 00:27:56 +00:00
|
|
|
let body = db.body(self.parent);
|
2021-03-09 15:33:41 +00:00
|
|
|
matches!(&body[self.pat_id], Pat::Bind { mode: BindingAnnotation::Mutable, .. })
|
2021-03-08 19:08:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn parent(self, _db: &dyn HirDatabase) -> DefWithBody {
|
|
|
|
self.parent.into()
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn module(self, db: &dyn HirDatabase) -> Module {
|
|
|
|
self.parent(db).module(db)
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn ty(self, db: &dyn HirDatabase) -> Type {
|
2021-03-17 00:27:56 +00:00
|
|
|
let def = self.parent;
|
2021-03-08 19:08:30 +00:00
|
|
|
let infer = db.infer(def);
|
|
|
|
let ty = infer[self.pat_id].clone();
|
|
|
|
let krate = def.module(db.upcast()).krate();
|
|
|
|
Type::new(db, krate, def, ty)
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn source(self, db: &dyn HirDatabase) -> InFile<Either<ast::IdentPat, ast::SelfParam>> {
|
2021-03-17 00:27:56 +00:00
|
|
|
let (_body, source_map) = db.body_with_source_map(self.parent);
|
2021-03-08 19:08:30 +00:00
|
|
|
let src = source_map.pat_syntax(self.pat_id).unwrap(); // Hmm...
|
|
|
|
let root = src.file_syntax(db.upcast());
|
|
|
|
src.map(|ast| {
|
|
|
|
ast.map_left(|it| it.cast().unwrap().to_node(&root)).map_right(|it| it.to_node(&root))
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Clone, Copy, Debug, PartialEq, Eq, Hash)]
|
|
|
|
pub struct Label {
|
|
|
|
pub(crate) parent: DefWithBodyId,
|
|
|
|
pub(crate) label_id: LabelId,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Label {
|
|
|
|
pub fn module(self, db: &dyn HirDatabase) -> Module {
|
|
|
|
self.parent(db).module(db)
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn parent(self, _db: &dyn HirDatabase) -> DefWithBody {
|
|
|
|
self.parent.into()
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn name(self, db: &dyn HirDatabase) -> Name {
|
2021-03-17 00:27:56 +00:00
|
|
|
let body = db.body(self.parent);
|
2021-03-08 19:08:30 +00:00
|
|
|
body[self.label_id].name.clone()
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn source(self, db: &dyn HirDatabase) -> InFile<ast::Label> {
|
2021-03-17 00:27:56 +00:00
|
|
|
let (_body, source_map) = db.body_with_source_map(self.parent);
|
2021-03-08 19:08:30 +00:00
|
|
|
let src = source_map.label_syntax(self.label_id);
|
|
|
|
let root = src.file_syntax(db.upcast());
|
|
|
|
src.map(|ast| ast.to_node(&root))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Clone, Copy, Debug, PartialEq, Eq, Hash)]
|
|
|
|
pub enum GenericParam {
|
|
|
|
TypeParam(TypeParam),
|
|
|
|
LifetimeParam(LifetimeParam),
|
|
|
|
ConstParam(ConstParam),
|
|
|
|
}
|
|
|
|
impl_from!(TypeParam, LifetimeParam, ConstParam for GenericParam);
|
|
|
|
|
|
|
|
impl GenericParam {
|
|
|
|
pub fn module(self, db: &dyn HirDatabase) -> Module {
|
|
|
|
match self {
|
|
|
|
GenericParam::TypeParam(it) => it.module(db),
|
|
|
|
GenericParam::LifetimeParam(it) => it.module(db),
|
|
|
|
GenericParam::ConstParam(it) => it.module(db),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn name(self, db: &dyn HirDatabase) -> Name {
|
|
|
|
match self {
|
|
|
|
GenericParam::TypeParam(it) => it.name(db),
|
|
|
|
GenericParam::LifetimeParam(it) => it.name(db),
|
|
|
|
GenericParam::ConstParam(it) => it.name(db),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Clone, Copy, Debug, PartialEq, Eq, Hash)]
|
|
|
|
pub struct TypeParam {
|
|
|
|
pub(crate) id: TypeParamId,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl TypeParam {
|
|
|
|
pub fn name(self, db: &dyn HirDatabase) -> Name {
|
|
|
|
let params = db.generic_params(self.id.parent);
|
|
|
|
params.types[self.id.local_id].name.clone().unwrap_or_else(Name::missing)
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn module(self, db: &dyn HirDatabase) -> Module {
|
|
|
|
self.id.parent.module(db.upcast()).into()
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn ty(self, db: &dyn HirDatabase) -> Type {
|
|
|
|
let resolver = self.id.parent.resolver(db.upcast());
|
2021-03-09 11:31:16 +00:00
|
|
|
let krate = self.id.parent.module(db.upcast()).krate();
|
2021-03-13 18:47:34 +00:00
|
|
|
let ty = TyKind::Placeholder(hir_ty::to_placeholder_idx(db, self.id)).intern(&Interner);
|
2021-03-09 11:31:16 +00:00
|
|
|
Type::new_with_resolver_inner(db, krate, &resolver, ty)
|
2021-03-08 19:08:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn trait_bounds(self, db: &dyn HirDatabase) -> Vec<Trait> {
|
|
|
|
db.generic_predicates_for_param(self.id)
|
|
|
|
.into_iter()
|
2021-03-21 16:40:14 +00:00
|
|
|
.filter_map(|pred| match &pred.skip_binders().skip_binders() {
|
2021-03-20 09:46:36 +00:00
|
|
|
hir_ty::WhereClause::Implemented(trait_ref) => {
|
2021-03-18 20:53:19 +00:00
|
|
|
Some(Trait::from(trait_ref.hir_trait_id()))
|
2021-03-08 19:08:30 +00:00
|
|
|
}
|
|
|
|
_ => None,
|
|
|
|
})
|
|
|
|
.collect()
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn default(self, db: &dyn HirDatabase) -> Option<Type> {
|
|
|
|
let params = db.generic_defaults(self.id.parent);
|
|
|
|
let local_idx = hir_ty::param_idx(db, self.id)?;
|
|
|
|
let resolver = self.id.parent.resolver(db.upcast());
|
2021-03-09 11:31:16 +00:00
|
|
|
let krate = self.id.parent.module(db.upcast()).krate();
|
2021-03-08 19:08:30 +00:00
|
|
|
let ty = params.get(local_idx)?.clone();
|
2021-04-04 11:16:16 +00:00
|
|
|
let subst = TyBuilder::type_params_subst(db, self.id.parent);
|
2021-04-05 19:56:40 +00:00
|
|
|
let ty = ty.substitute(&Interner, &subst_prefix(&subst, local_idx));
|
2021-03-09 11:31:16 +00:00
|
|
|
Some(Type::new_with_resolver_inner(db, krate, &resolver, ty))
|
2021-03-08 19:08:30 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Clone, Copy, Debug, PartialEq, Eq, Hash)]
|
|
|
|
pub struct LifetimeParam {
|
|
|
|
pub(crate) id: LifetimeParamId,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl LifetimeParam {
|
|
|
|
pub fn name(self, db: &dyn HirDatabase) -> Name {
|
|
|
|
let params = db.generic_params(self.id.parent);
|
|
|
|
params.lifetimes[self.id.local_id].name.clone()
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn module(self, db: &dyn HirDatabase) -> Module {
|
|
|
|
self.id.parent.module(db.upcast()).into()
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn parent(self, _db: &dyn HirDatabase) -> GenericDef {
|
|
|
|
self.id.parent.into()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Clone, Copy, Debug, PartialEq, Eq, Hash)]
|
|
|
|
pub struct ConstParam {
|
|
|
|
pub(crate) id: ConstParamId,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl ConstParam {
|
|
|
|
pub fn name(self, db: &dyn HirDatabase) -> Name {
|
|
|
|
let params = db.generic_params(self.id.parent);
|
|
|
|
params.consts[self.id.local_id].name.clone()
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn module(self, db: &dyn HirDatabase) -> Module {
|
|
|
|
self.id.parent.module(db.upcast()).into()
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn parent(self, _db: &dyn HirDatabase) -> GenericDef {
|
|
|
|
self.id.parent.into()
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn ty(self, db: &dyn HirDatabase) -> Type {
|
|
|
|
let def = self.id.parent;
|
|
|
|
let krate = def.module(db.upcast()).krate();
|
|
|
|
Type::new(db, krate, def, db.const_param_ty(self.id))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
|
|
|
|
pub struct Impl {
|
|
|
|
pub(crate) id: ImplId,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Impl {
|
|
|
|
pub fn all_in_crate(db: &dyn HirDatabase, krate: Crate) -> Vec<Impl> {
|
|
|
|
let inherent = db.inherent_impls_in_crate(krate.id);
|
|
|
|
let trait_ = db.trait_impls_in_crate(krate.id);
|
|
|
|
|
|
|
|
inherent.all_impls().chain(trait_.all_impls()).map(Self::from).collect()
|
|
|
|
}
|
2021-03-15 09:11:48 +00:00
|
|
|
|
2021-03-21 19:19:07 +00:00
|
|
|
pub fn all_for_type(db: &dyn HirDatabase, Type { krate, ty, .. }: Type) -> Vec<Impl> {
|
2021-04-07 11:09:31 +00:00
|
|
|
let def_crates = match def_crates(db, &ty, krate) {
|
2021-03-15 09:11:48 +00:00
|
|
|
Some(def_crates) => def_crates,
|
2021-03-15 13:31:55 +00:00
|
|
|
None => return Vec::new(),
|
2021-03-15 09:11:48 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
let filter = |impl_def: &Impl| {
|
2021-03-29 15:46:33 +00:00
|
|
|
let self_ty = impl_def.self_ty(db);
|
|
|
|
let rref = self_ty.remove_ref();
|
|
|
|
ty.equals_ctor(rref.as_ref().map_or(&self_ty.ty, |it| &it.ty))
|
2021-03-15 09:11:48 +00:00
|
|
|
};
|
|
|
|
|
2021-04-07 17:35:24 +00:00
|
|
|
let fp = TyFingerprint::for_inherent_impl(&ty);
|
|
|
|
let fp = if let Some(fp) = fp {
|
|
|
|
fp
|
|
|
|
} else {
|
|
|
|
return Vec::new();
|
|
|
|
};
|
|
|
|
|
2021-03-15 09:11:48 +00:00
|
|
|
let mut all = Vec::new();
|
2021-03-15 16:43:46 +00:00
|
|
|
def_crates.iter().for_each(|&id| {
|
2021-04-07 17:35:24 +00:00
|
|
|
all.extend(
|
|
|
|
db.inherent_impls_in_crate(id)
|
|
|
|
.for_self_ty(&ty)
|
|
|
|
.into_iter()
|
|
|
|
.cloned()
|
|
|
|
.map(Self::from)
|
|
|
|
.filter(filter),
|
|
|
|
)
|
2021-03-15 09:11:48 +00:00
|
|
|
});
|
2021-03-15 16:43:46 +00:00
|
|
|
for id in def_crates
|
|
|
|
.iter()
|
|
|
|
.flat_map(|&id| Crate { id }.transitive_reverse_dependencies(db))
|
|
|
|
.map(|Crate { id }| id)
|
|
|
|
.chain(def_crates.iter().copied())
|
|
|
|
.unique()
|
|
|
|
{
|
2021-04-07 17:35:24 +00:00
|
|
|
all.extend(
|
|
|
|
db.trait_impls_in_crate(id)
|
|
|
|
.for_self_ty_without_blanket_impls(fp)
|
|
|
|
.map(Self::from)
|
|
|
|
.filter(filter),
|
|
|
|
);
|
2021-03-15 09:11:48 +00:00
|
|
|
}
|
|
|
|
all
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn all_for_trait(db: &dyn HirDatabase, trait_: Trait) -> Vec<Impl> {
|
|
|
|
let krate = trait_.module(db).krate();
|
|
|
|
let mut all = Vec::new();
|
2021-03-23 09:49:55 +00:00
|
|
|
for Crate { id } in krate.transitive_reverse_dependencies(db).into_iter() {
|
2021-03-15 09:11:48 +00:00
|
|
|
let impls = db.trait_impls_in_crate(id);
|
|
|
|
all.extend(impls.for_trait(trait_.id).map(Self::from))
|
|
|
|
}
|
|
|
|
all
|
2021-03-08 19:08:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// FIXME: the return type is wrong. This should be a hir version of
|
2021-06-01 23:16:59 +00:00
|
|
|
// `TraitRef` (to account for parameters and qualifiers)
|
|
|
|
pub fn trait_(self, db: &dyn HirDatabase) -> Option<Trait> {
|
|
|
|
let trait_ref = db.impl_trait(self.id)?.skip_binders().clone();
|
|
|
|
let id = hir_ty::from_chalk_trait_id(trait_ref.trait_id);
|
|
|
|
Some(Trait { id })
|
2021-03-08 19:08:30 +00:00
|
|
|
}
|
|
|
|
|
2021-03-29 15:46:33 +00:00
|
|
|
pub fn self_ty(self, db: &dyn HirDatabase) -> Type {
|
2021-03-08 19:08:30 +00:00
|
|
|
let impl_data = db.impl_data(self.id);
|
|
|
|
let resolver = self.id.resolver(db.upcast());
|
2021-03-09 18:09:02 +00:00
|
|
|
let krate = self.id.lookup(db.upcast()).container.krate();
|
2021-03-08 19:08:30 +00:00
|
|
|
let ctx = hir_ty::TyLoweringContext::new(db, &resolver);
|
2021-03-29 15:46:33 +00:00
|
|
|
let ty = ctx.lower_ty(&impl_data.self_ty);
|
2021-03-09 11:31:16 +00:00
|
|
|
Type::new_with_resolver_inner(db, krate, &resolver, ty)
|
2021-03-08 19:08:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn items(self, db: &dyn HirDatabase) -> Vec<AssocItem> {
|
|
|
|
db.impl_data(self.id).items.iter().map(|it| (*it).into()).collect()
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn is_negative(self, db: &dyn HirDatabase) -> bool {
|
|
|
|
db.impl_data(self.id).is_negative
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn module(self, db: &dyn HirDatabase) -> Module {
|
2021-03-09 18:09:02 +00:00
|
|
|
self.id.lookup(db.upcast()).container.into()
|
2021-03-08 19:08:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn is_builtin_derive(self, db: &dyn HirDatabase) -> Option<InFile<ast::Attr>> {
|
|
|
|
let src = self.source(db)?;
|
|
|
|
let item = src.file_id.is_builtin_derive(db.upcast())?;
|
|
|
|
let hygenic = hir_expand::hygiene::Hygiene::new(db.upcast(), item.file_id);
|
|
|
|
|
|
|
|
// FIXME: handle `cfg_attr`
|
|
|
|
let attr = item
|
|
|
|
.value
|
|
|
|
.attrs()
|
|
|
|
.filter_map(|it| {
|
2021-05-06 17:59:54 +00:00
|
|
|
let path = ModPath::from_src(db.upcast(), it.path()?, &hygenic)?;
|
2021-03-08 19:08:30 +00:00
|
|
|
if path.as_ident()?.to_string() == "derive" {
|
|
|
|
Some(it)
|
|
|
|
} else {
|
|
|
|
None
|
|
|
|
}
|
|
|
|
})
|
|
|
|
.last()?;
|
|
|
|
|
|
|
|
Some(item.with_value(attr))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Clone, PartialEq, Eq, Debug)]
|
|
|
|
pub struct Type {
|
|
|
|
krate: CrateId,
|
2021-03-21 19:19:07 +00:00
|
|
|
env: Arc<TraitEnvironment>,
|
|
|
|
ty: Ty,
|
2021-03-08 19:08:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
impl Type {
|
|
|
|
pub(crate) fn new_with_resolver(
|
|
|
|
db: &dyn HirDatabase,
|
|
|
|
resolver: &Resolver,
|
|
|
|
ty: Ty,
|
|
|
|
) -> Option<Type> {
|
|
|
|
let krate = resolver.krate()?;
|
|
|
|
Some(Type::new_with_resolver_inner(db, krate, resolver, ty))
|
|
|
|
}
|
|
|
|
pub(crate) fn new_with_resolver_inner(
|
|
|
|
db: &dyn HirDatabase,
|
|
|
|
krate: CrateId,
|
|
|
|
resolver: &Resolver,
|
|
|
|
ty: Ty,
|
|
|
|
) -> Type {
|
2021-05-16 13:50:28 +00:00
|
|
|
let environment = resolver
|
|
|
|
.generic_def()
|
|
|
|
.map_or_else(|| Arc::new(TraitEnvironment::empty(krate)), |d| db.trait_environment(d));
|
2021-03-21 19:19:07 +00:00
|
|
|
Type { krate, env: environment, ty }
|
2021-03-08 19:08:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
fn new(db: &dyn HirDatabase, krate: CrateId, lexical_env: impl HasResolver, ty: Ty) -> Type {
|
|
|
|
let resolver = lexical_env.resolver(db.upcast());
|
2021-05-16 13:50:28 +00:00
|
|
|
let environment = resolver
|
|
|
|
.generic_def()
|
|
|
|
.map_or_else(|| Arc::new(TraitEnvironment::empty(krate)), |d| db.trait_environment(d));
|
2021-03-21 19:19:07 +00:00
|
|
|
Type { krate, env: environment, ty }
|
2021-03-08 19:08:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
fn from_def(
|
|
|
|
db: &dyn HirDatabase,
|
|
|
|
krate: CrateId,
|
2021-04-03 20:18:02 +00:00
|
|
|
def: impl HasResolver + Into<TyDefId>,
|
2021-03-08 19:08:30 +00:00
|
|
|
) -> Type {
|
2021-04-03 20:18:02 +00:00
|
|
|
let ty = TyBuilder::def_ty(db, def.into()).fill_with_unknown().build();
|
2021-03-08 19:08:30 +00:00
|
|
|
Type::new(db, krate, def, ty)
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn is_unit(&self) -> bool {
|
2021-04-03 11:08:29 +00:00
|
|
|
matches!(self.ty.kind(&Interner), TyKind::Tuple(0, ..))
|
2021-03-08 19:08:30 +00:00
|
|
|
}
|
|
|
|
pub fn is_bool(&self) -> bool {
|
2021-04-03 11:08:29 +00:00
|
|
|
matches!(self.ty.kind(&Interner), TyKind::Scalar(Scalar::Bool))
|
2021-03-08 19:08:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn is_mutable_reference(&self) -> bool {
|
2021-04-03 11:08:29 +00:00
|
|
|
matches!(self.ty.kind(&Interner), TyKind::Ref(hir_ty::Mutability::Mut, ..))
|
2021-03-08 19:08:30 +00:00
|
|
|
}
|
|
|
|
|
2021-03-15 19:46:59 +00:00
|
|
|
pub fn is_usize(&self) -> bool {
|
2021-04-03 11:08:29 +00:00
|
|
|
matches!(self.ty.kind(&Interner), TyKind::Scalar(Scalar::Uint(UintTy::Usize)))
|
2021-03-15 19:46:59 +00:00
|
|
|
}
|
|
|
|
|
2021-03-08 19:08:30 +00:00
|
|
|
pub fn remove_ref(&self) -> Option<Type> {
|
2021-04-03 11:08:29 +00:00
|
|
|
match &self.ty.kind(&Interner) {
|
2021-03-14 16:40:55 +00:00
|
|
|
TyKind::Ref(.., ty) => Some(self.derived(ty.clone())),
|
2021-03-09 15:06:08 +00:00
|
|
|
_ => None,
|
2021-03-08 19:08:30 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-05-05 20:55:12 +00:00
|
|
|
pub fn strip_references(&self) -> Type {
|
|
|
|
self.derived(self.ty.strip_references().clone())
|
|
|
|
}
|
|
|
|
|
2021-03-08 19:08:30 +00:00
|
|
|
pub fn is_unknown(&self) -> bool {
|
2021-03-21 19:19:07 +00:00
|
|
|
self.ty.is_unknown()
|
2021-03-08 19:08:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Checks that particular type `ty` implements `std::future::Future`.
|
|
|
|
/// This function is used in `.await` syntax completion.
|
|
|
|
pub fn impls_future(&self, db: &dyn HirDatabase) -> bool {
|
|
|
|
// No special case for the type of async block, since Chalk can figure it out.
|
|
|
|
|
|
|
|
let krate = self.krate;
|
|
|
|
|
|
|
|
let std_future_trait =
|
|
|
|
db.lang_item(krate, "future_trait".into()).and_then(|it| it.as_trait());
|
|
|
|
let std_future_trait = match std_future_trait {
|
|
|
|
Some(it) => it,
|
|
|
|
None => return false,
|
|
|
|
};
|
|
|
|
|
2021-03-21 19:19:07 +00:00
|
|
|
let canonical_ty =
|
|
|
|
Canonical { value: self.ty.clone(), binders: CanonicalVarKinds::empty(&Interner) };
|
2021-03-08 19:08:30 +00:00
|
|
|
method_resolution::implements_trait(
|
|
|
|
&canonical_ty,
|
|
|
|
db,
|
2021-03-21 19:19:07 +00:00
|
|
|
self.env.clone(),
|
2021-03-08 19:08:30 +00:00
|
|
|
krate,
|
|
|
|
std_future_trait,
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Checks that particular type `ty` implements `std::ops::FnOnce`.
|
|
|
|
///
|
|
|
|
/// This function can be used to check if a particular type is callable, since FnOnce is a
|
|
|
|
/// supertrait of Fn and FnMut, so all callable types implements at least FnOnce.
|
|
|
|
pub fn impls_fnonce(&self, db: &dyn HirDatabase) -> bool {
|
|
|
|
let krate = self.krate;
|
|
|
|
|
|
|
|
let fnonce_trait = match FnTrait::FnOnce.get_id(db, krate) {
|
|
|
|
Some(it) => it,
|
|
|
|
None => return false,
|
|
|
|
};
|
|
|
|
|
2021-03-21 19:19:07 +00:00
|
|
|
let canonical_ty =
|
|
|
|
Canonical { value: self.ty.clone(), binders: CanonicalVarKinds::empty(&Interner) };
|
2021-03-08 19:08:30 +00:00
|
|
|
method_resolution::implements_trait_unique(
|
|
|
|
&canonical_ty,
|
|
|
|
db,
|
2021-03-21 19:19:07 +00:00
|
|
|
self.env.clone(),
|
2021-03-08 19:08:30 +00:00
|
|
|
krate,
|
|
|
|
fnonce_trait,
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn impls_trait(&self, db: &dyn HirDatabase, trait_: Trait, args: &[Type]) -> bool {
|
2021-04-03 19:50:52 +00:00
|
|
|
let trait_ref = TyBuilder::trait_ref(db, trait_.id)
|
|
|
|
.push(self.ty.clone())
|
|
|
|
.fill(args.iter().map(|t| t.ty.clone()))
|
|
|
|
.build();
|
2021-03-08 19:08:30 +00:00
|
|
|
|
|
|
|
let goal = Canonical {
|
2021-04-07 18:48:58 +00:00
|
|
|
value: hir_ty::InEnvironment::new(&self.env.env, trait_ref.cast(&Interner)),
|
2021-03-21 19:05:38 +00:00
|
|
|
binders: CanonicalVarKinds::empty(&Interner),
|
2021-03-08 19:08:30 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
db.trait_solve(self.krate, goal).is_some()
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn normalize_trait_assoc_type(
|
|
|
|
&self,
|
|
|
|
db: &dyn HirDatabase,
|
|
|
|
args: &[Type],
|
|
|
|
alias: TypeAlias,
|
|
|
|
) -> Option<Type> {
|
2021-04-03 19:50:52 +00:00
|
|
|
let projection = TyBuilder::assoc_type_projection(db, alias.id)
|
2021-03-21 19:19:07 +00:00
|
|
|
.push(self.ty.clone())
|
|
|
|
.fill(args.iter().map(|t| t.ty.clone()))
|
2021-03-08 19:08:30 +00:00
|
|
|
.build();
|
2021-04-07 18:47:04 +00:00
|
|
|
let goal = hir_ty::make_canonical(
|
2021-03-21 19:05:38 +00:00
|
|
|
InEnvironment::new(
|
2021-04-07 18:48:58 +00:00
|
|
|
&self.env.env,
|
2021-03-20 10:23:59 +00:00
|
|
|
AliasEq {
|
2021-04-03 19:50:52 +00:00
|
|
|
alias: AliasTy::Projection(projection),
|
2021-03-19 01:07:15 +00:00
|
|
|
ty: TyKind::BoundVar(BoundVar::new(DebruijnIndex::INNERMOST, 0))
|
|
|
|
.intern(&Interner),
|
2021-03-20 10:23:59 +00:00
|
|
|
}
|
|
|
|
.cast(&Interner),
|
2021-03-08 19:08:30 +00:00
|
|
|
),
|
2021-03-21 19:05:38 +00:00
|
|
|
[TyVariableKind::General].iter().copied(),
|
|
|
|
);
|
2021-03-08 19:08:30 +00:00
|
|
|
|
|
|
|
match db.trait_solve(self.krate, goal)? {
|
2021-04-06 21:46:32 +00:00
|
|
|
Solution::Unique(s) => s
|
2021-04-01 19:04:02 +00:00
|
|
|
.value
|
2021-04-06 21:46:32 +00:00
|
|
|
.subst
|
2021-04-08 16:25:18 +00:00
|
|
|
.as_slice(&Interner)
|
2021-04-01 19:04:02 +00:00
|
|
|
.first()
|
|
|
|
.map(|ty| self.derived(ty.assert_ty_ref(&Interner).clone())),
|
2021-03-08 19:08:30 +00:00
|
|
|
Solution::Ambig(_) => None,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn is_copy(&self, db: &dyn HirDatabase) -> bool {
|
|
|
|
let lang_item = db.lang_item(self.krate, SmolStr::new("copy"));
|
|
|
|
let copy_trait = match lang_item {
|
|
|
|
Some(LangItemTarget::TraitId(it)) => it,
|
|
|
|
_ => return false,
|
|
|
|
};
|
|
|
|
self.impls_trait(db, copy_trait.into(), &[])
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn as_callable(&self, db: &dyn HirDatabase) -> Option<Callable> {
|
2021-03-21 19:19:07 +00:00
|
|
|
let def = self.ty.callable_def(db);
|
2021-03-08 19:08:30 +00:00
|
|
|
|
2021-03-21 19:19:07 +00:00
|
|
|
let sig = self.ty.callable_sig(db)?;
|
2021-03-08 19:08:30 +00:00
|
|
|
Some(Callable { ty: self.clone(), sig, def, is_bound_method: false })
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn is_closure(&self) -> bool {
|
2021-04-03 11:08:29 +00:00
|
|
|
matches!(&self.ty.kind(&Interner), TyKind::Closure { .. })
|
2021-03-08 19:08:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn is_fn(&self) -> bool {
|
2021-04-03 11:08:29 +00:00
|
|
|
matches!(&self.ty.kind(&Interner), TyKind::FnDef(..) | TyKind::Function { .. })
|
2021-03-08 19:08:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn is_packed(&self, db: &dyn HirDatabase) -> bool {
|
2021-04-03 11:08:29 +00:00
|
|
|
let adt_id = match self.ty.kind(&Interner) {
|
2021-03-13 13:44:51 +00:00
|
|
|
&TyKind::Adt(hir_ty::AdtId(adt_id), ..) => adt_id,
|
2021-03-08 19:08:30 +00:00
|
|
|
_ => return false,
|
|
|
|
};
|
|
|
|
|
|
|
|
let adt = adt_id.into();
|
|
|
|
match adt {
|
|
|
|
Adt::Struct(s) => matches!(s.repr(db), Some(ReprKind::Packed)),
|
|
|
|
_ => false,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn is_raw_ptr(&self) -> bool {
|
2021-04-03 11:08:29 +00:00
|
|
|
matches!(&self.ty.kind(&Interner), TyKind::Raw(..))
|
2021-03-08 19:08:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn contains_unknown(&self) -> bool {
|
2021-03-21 19:19:07 +00:00
|
|
|
return go(&self.ty);
|
2021-03-08 19:08:30 +00:00
|
|
|
|
|
|
|
fn go(ty: &Ty) -> bool {
|
2021-04-03 11:08:29 +00:00
|
|
|
match ty.kind(&Interner) {
|
2021-04-05 12:37:11 +00:00
|
|
|
TyKind::Error => true,
|
2021-03-14 16:40:55 +00:00
|
|
|
|
|
|
|
TyKind::Adt(_, substs)
|
|
|
|
| TyKind::AssociatedType(_, substs)
|
|
|
|
| TyKind::Tuple(_, substs)
|
|
|
|
| TyKind::OpaqueType(_, substs)
|
|
|
|
| TyKind::FnDef(_, substs)
|
2021-04-01 19:04:02 +00:00
|
|
|
| TyKind::Closure(_, substs) => {
|
|
|
|
substs.iter(&Interner).filter_map(|a| a.ty(&Interner)).any(go)
|
|
|
|
}
|
2021-03-14 16:40:55 +00:00
|
|
|
|
2021-05-14 07:59:30 +00:00
|
|
|
TyKind::Array(_ty, len) if len.is_unknown() => true,
|
2021-04-06 09:45:41 +00:00
|
|
|
TyKind::Array(ty, _)
|
2021-04-05 20:08:16 +00:00
|
|
|
| TyKind::Slice(ty)
|
|
|
|
| TyKind::Raw(_, ty)
|
|
|
|
| TyKind::Ref(_, _, ty) => go(ty),
|
2021-03-14 16:40:55 +00:00
|
|
|
|
|
|
|
TyKind::Scalar(_)
|
|
|
|
| TyKind::Str
|
|
|
|
| TyKind::Never
|
|
|
|
| TyKind::Placeholder(_)
|
|
|
|
| TyKind::BoundVar(_)
|
|
|
|
| TyKind::InferenceVar(_, _)
|
|
|
|
| TyKind::Dyn(_)
|
|
|
|
| TyKind::Function(_)
|
|
|
|
| TyKind::Alias(_)
|
2021-04-08 11:51:04 +00:00
|
|
|
| TyKind::Foreign(_)
|
|
|
|
| TyKind::Generator(..)
|
|
|
|
| TyKind::GeneratorWitness(..) => false,
|
2021-03-08 19:08:30 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn fields(&self, db: &dyn HirDatabase) -> Vec<(Field, Type)> {
|
2021-04-03 11:08:29 +00:00
|
|
|
let (variant_id, substs) = match self.ty.kind(&Interner) {
|
2021-03-13 13:44:51 +00:00
|
|
|
&TyKind::Adt(hir_ty::AdtId(AdtId::StructId(s)), ref substs) => (s.into(), substs),
|
|
|
|
&TyKind::Adt(hir_ty::AdtId(AdtId::UnionId(u)), ref substs) => (u.into(), substs),
|
2021-03-08 19:08:30 +00:00
|
|
|
_ => return Vec::new(),
|
|
|
|
};
|
|
|
|
|
|
|
|
db.field_types(variant_id)
|
|
|
|
.iter()
|
|
|
|
.map(|(local_id, ty)| {
|
|
|
|
let def = Field { parent: variant_id.into(), id: local_id };
|
2021-04-05 16:49:26 +00:00
|
|
|
let ty = ty.clone().substitute(&Interner, substs);
|
2021-03-08 19:08:30 +00:00
|
|
|
(def, self.derived(ty))
|
|
|
|
})
|
|
|
|
.collect()
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn tuple_fields(&self, _db: &dyn HirDatabase) -> Vec<Type> {
|
2021-04-03 11:08:29 +00:00
|
|
|
if let TyKind::Tuple(_, substs) = &self.ty.kind(&Interner) {
|
2021-04-01 19:04:02 +00:00
|
|
|
substs
|
|
|
|
.iter(&Interner)
|
|
|
|
.map(|ty| self.derived(ty.assert_ty_ref(&Interner).clone()))
|
|
|
|
.collect()
|
2021-03-08 19:08:30 +00:00
|
|
|
} else {
|
|
|
|
Vec::new()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn autoderef<'a>(&'a self, db: &'a dyn HirDatabase) -> impl Iterator<Item = Type> + 'a {
|
|
|
|
// There should be no inference vars in types passed here
|
|
|
|
// FIXME check that?
|
2021-03-21 19:19:07 +00:00
|
|
|
let canonical =
|
|
|
|
Canonical { value: self.ty.clone(), binders: CanonicalVarKinds::empty(&Interner) };
|
|
|
|
let environment = self.env.env.clone();
|
|
|
|
let ty = InEnvironment { goal: canonical, environment };
|
2021-03-08 19:08:30 +00:00
|
|
|
autoderef(db, Some(self.krate), ty)
|
|
|
|
.map(|canonical| canonical.value)
|
|
|
|
.map(move |ty| self.derived(ty))
|
|
|
|
}
|
|
|
|
|
|
|
|
// This would be nicer if it just returned an iterator, but that runs into
|
|
|
|
// lifetime problems, because we need to borrow temp `CrateImplDefs`.
|
|
|
|
pub fn iterate_assoc_items<T>(
|
|
|
|
self,
|
|
|
|
db: &dyn HirDatabase,
|
|
|
|
krate: Crate,
|
|
|
|
mut callback: impl FnMut(AssocItem) -> Option<T>,
|
|
|
|
) -> Option<T> {
|
2021-04-07 11:09:31 +00:00
|
|
|
for krate in def_crates(db, &self.ty, krate.id)? {
|
2021-03-08 19:08:30 +00:00
|
|
|
let impls = db.inherent_impls_in_crate(krate);
|
|
|
|
|
2021-03-21 19:19:07 +00:00
|
|
|
for impl_def in impls.for_self_ty(&self.ty) {
|
2021-03-08 19:08:30 +00:00
|
|
|
for &item in db.impl_data(*impl_def).items.iter() {
|
|
|
|
if let Some(result) = callback(item.into()) {
|
|
|
|
return Some(result);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
None
|
|
|
|
}
|
|
|
|
|
2021-04-30 08:55:59 +00:00
|
|
|
pub fn type_arguments(&self) -> impl Iterator<Item = Type> + '_ {
|
2021-03-08 19:08:30 +00:00
|
|
|
self.ty
|
|
|
|
.strip_references()
|
2021-04-07 15:26:01 +00:00
|
|
|
.as_adt()
|
2021-03-08 19:08:30 +00:00
|
|
|
.into_iter()
|
2021-04-07 15:26:01 +00:00
|
|
|
.flat_map(|(_, substs)| substs.iter(&Interner))
|
2021-04-01 19:04:02 +00:00
|
|
|
.filter_map(|arg| arg.ty(&Interner).cloned())
|
|
|
|
.map(move |ty| self.derived(ty))
|
2021-03-08 19:08:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn iterate_method_candidates<T>(
|
|
|
|
&self,
|
|
|
|
db: &dyn HirDatabase,
|
|
|
|
krate: Crate,
|
|
|
|
traits_in_scope: &FxHashSet<TraitId>,
|
|
|
|
name: Option<&Name>,
|
|
|
|
mut callback: impl FnMut(&Ty, Function) -> Option<T>,
|
|
|
|
) -> Option<T> {
|
|
|
|
// There should be no inference vars in types passed here
|
|
|
|
// FIXME check that?
|
|
|
|
// FIXME replace Unknown by bound vars here
|
2021-03-21 19:19:07 +00:00
|
|
|
let canonical =
|
|
|
|
Canonical { value: self.ty.clone(), binders: CanonicalVarKinds::empty(&Interner) };
|
2021-03-08 19:08:30 +00:00
|
|
|
|
2021-03-21 19:19:07 +00:00
|
|
|
let env = self.env.clone();
|
2021-03-08 19:08:30 +00:00
|
|
|
let krate = krate.id;
|
|
|
|
|
|
|
|
method_resolution::iterate_method_candidates(
|
|
|
|
&canonical,
|
|
|
|
db,
|
|
|
|
env,
|
|
|
|
krate,
|
|
|
|
traits_in_scope,
|
2021-03-24 22:09:22 +00:00
|
|
|
None,
|
2021-03-08 19:08:30 +00:00
|
|
|
name,
|
|
|
|
method_resolution::LookupMode::MethodCall,
|
|
|
|
|ty, it| match it {
|
|
|
|
AssocItemId::FunctionId(f) => callback(ty, f.into()),
|
|
|
|
_ => None,
|
|
|
|
},
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn iterate_path_candidates<T>(
|
|
|
|
&self,
|
|
|
|
db: &dyn HirDatabase,
|
|
|
|
krate: Crate,
|
|
|
|
traits_in_scope: &FxHashSet<TraitId>,
|
|
|
|
name: Option<&Name>,
|
|
|
|
mut callback: impl FnMut(&Ty, AssocItem) -> Option<T>,
|
|
|
|
) -> Option<T> {
|
2021-05-23 10:52:41 +00:00
|
|
|
let canonical = hir_ty::replace_errors_with_variables(&self.ty);
|
2021-03-08 19:08:30 +00:00
|
|
|
|
2021-03-21 19:19:07 +00:00
|
|
|
let env = self.env.clone();
|
2021-03-08 19:08:30 +00:00
|
|
|
let krate = krate.id;
|
|
|
|
|
|
|
|
method_resolution::iterate_method_candidates(
|
|
|
|
&canonical,
|
|
|
|
db,
|
|
|
|
env,
|
|
|
|
krate,
|
|
|
|
traits_in_scope,
|
2021-03-20 18:28:26 +00:00
|
|
|
None,
|
2021-03-08 19:08:30 +00:00
|
|
|
name,
|
|
|
|
method_resolution::LookupMode::Path,
|
|
|
|
|ty, it| callback(ty, it.into()),
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn as_adt(&self) -> Option<Adt> {
|
2021-03-21 19:19:07 +00:00
|
|
|
let (adt, _subst) = self.ty.as_adt()?;
|
2021-03-08 19:08:30 +00:00
|
|
|
Some(adt.into())
|
|
|
|
}
|
|
|
|
|
2021-05-08 20:34:55 +00:00
|
|
|
pub fn as_builtin(&self) -> Option<BuiltinType> {
|
|
|
|
self.ty.as_builtin().map(|inner| BuiltinType { inner })
|
|
|
|
}
|
|
|
|
|
2021-03-08 19:08:30 +00:00
|
|
|
pub fn as_dyn_trait(&self) -> Option<Trait> {
|
2021-03-21 19:19:07 +00:00
|
|
|
self.ty.dyn_trait().map(Into::into)
|
2021-03-08 19:08:30 +00:00
|
|
|
}
|
|
|
|
|
2021-04-15 20:31:42 +00:00
|
|
|
/// If a type can be represented as `dyn Trait`, returns all traits accessible via this type,
|
|
|
|
/// or an empty iterator otherwise.
|
|
|
|
pub fn applicable_inherent_traits<'a>(
|
|
|
|
&'a self,
|
|
|
|
db: &'a dyn HirDatabase,
|
|
|
|
) -> impl Iterator<Item = Trait> + 'a {
|
|
|
|
self.autoderef(db)
|
|
|
|
.filter_map(|derefed_type| derefed_type.ty.dyn_trait())
|
|
|
|
.flat_map(move |dyn_trait_id| hir_ty::all_super_traits(db.upcast(), dyn_trait_id))
|
|
|
|
.map(Trait::from)
|
|
|
|
}
|
|
|
|
|
2021-03-08 19:08:30 +00:00
|
|
|
pub fn as_impl_traits(&self, db: &dyn HirDatabase) -> Option<Vec<Trait>> {
|
2021-03-21 19:19:07 +00:00
|
|
|
self.ty.impl_trait_bounds(db).map(|it| {
|
2021-03-08 19:08:30 +00:00
|
|
|
it.into_iter()
|
2021-03-21 12:22:22 +00:00
|
|
|
.filter_map(|pred| match pred.skip_binders() {
|
2021-03-20 09:46:36 +00:00
|
|
|
hir_ty::WhereClause::Implemented(trait_ref) => {
|
2021-03-18 20:53:19 +00:00
|
|
|
Some(Trait::from(trait_ref.hir_trait_id()))
|
2021-03-08 19:08:30 +00:00
|
|
|
}
|
|
|
|
_ => None,
|
|
|
|
})
|
|
|
|
.collect()
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn as_associated_type_parent_trait(&self, db: &dyn HirDatabase) -> Option<Trait> {
|
2021-03-21 19:19:07 +00:00
|
|
|
self.ty.associated_type_parent_trait(db).map(Into::into)
|
2021-03-08 19:08:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
fn derived(&self, ty: Ty) -> Type {
|
2021-03-21 19:19:07 +00:00
|
|
|
Type { krate: self.krate, env: self.env.clone(), ty }
|
2021-03-08 19:08:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn walk(&self, db: &dyn HirDatabase, mut cb: impl FnMut(Type)) {
|
|
|
|
// TypeWalk::walk for a Ty at first visits parameters and only after that the Ty itself.
|
|
|
|
// We need a different order here.
|
|
|
|
|
|
|
|
fn walk_substs(
|
|
|
|
db: &dyn HirDatabase,
|
|
|
|
type_: &Type,
|
2021-03-15 20:02:34 +00:00
|
|
|
substs: &Substitution,
|
2021-03-08 19:08:30 +00:00
|
|
|
cb: &mut impl FnMut(Type),
|
|
|
|
) {
|
2021-04-01 19:04:02 +00:00
|
|
|
for ty in substs.iter(&Interner).filter_map(|a| a.ty(&Interner)) {
|
2021-03-08 19:08:30 +00:00
|
|
|
walk_type(db, &type_.derived(ty.clone()), cb);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn walk_bounds(
|
|
|
|
db: &dyn HirDatabase,
|
|
|
|
type_: &Type,
|
2021-03-21 12:22:22 +00:00
|
|
|
bounds: &[QuantifiedWhereClause],
|
2021-03-08 19:08:30 +00:00
|
|
|
cb: &mut impl FnMut(Type),
|
|
|
|
) {
|
|
|
|
for pred in bounds {
|
2021-03-21 12:22:22 +00:00
|
|
|
match pred.skip_binders() {
|
2021-03-20 09:46:36 +00:00
|
|
|
WhereClause::Implemented(trait_ref) => {
|
2021-03-08 19:08:30 +00:00
|
|
|
cb(type_.clone());
|
2021-03-20 19:07:36 +00:00
|
|
|
// skip the self type. it's likely the type we just got the bounds from
|
2021-04-01 19:04:02 +00:00
|
|
|
for ty in trait_ref
|
|
|
|
.substitution
|
|
|
|
.iter(&Interner)
|
|
|
|
.skip(1)
|
|
|
|
.filter_map(|a| a.ty(&Interner))
|
|
|
|
{
|
2021-03-20 19:07:36 +00:00
|
|
|
walk_type(db, &type_.derived(ty.clone()), cb);
|
|
|
|
}
|
2021-03-08 19:08:30 +00:00
|
|
|
}
|
|
|
|
_ => (),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn walk_type(db: &dyn HirDatabase, type_: &Type, cb: &mut impl FnMut(Type)) {
|
2021-03-21 19:19:07 +00:00
|
|
|
let ty = type_.ty.strip_references();
|
2021-04-03 11:08:29 +00:00
|
|
|
match ty.kind(&Interner) {
|
2021-04-07 15:26:01 +00:00
|
|
|
TyKind::Adt(_, substs) => {
|
2021-03-08 19:08:30 +00:00
|
|
|
cb(type_.derived(ty.clone()));
|
2021-04-07 15:26:01 +00:00
|
|
|
walk_substs(db, type_, &substs, cb);
|
2021-03-08 19:08:30 +00:00
|
|
|
}
|
2021-04-07 15:26:01 +00:00
|
|
|
TyKind::AssociatedType(_, substs) => {
|
2021-03-08 19:08:30 +00:00
|
|
|
if let Some(_) = ty.associated_type_parent_trait(db) {
|
|
|
|
cb(type_.derived(ty.clone()));
|
|
|
|
}
|
2021-04-07 15:26:01 +00:00
|
|
|
walk_substs(db, type_, &substs, cb);
|
2021-03-08 19:08:30 +00:00
|
|
|
}
|
2021-04-07 15:26:01 +00:00
|
|
|
TyKind::OpaqueType(_, subst) => {
|
2021-03-08 19:08:30 +00:00
|
|
|
if let Some(bounds) = ty.impl_trait_bounds(db) {
|
|
|
|
walk_bounds(db, &type_.derived(ty.clone()), &bounds, cb);
|
|
|
|
}
|
2021-04-07 15:26:01 +00:00
|
|
|
|
|
|
|
walk_substs(db, type_, subst, cb);
|
2021-03-08 19:08:30 +00:00
|
|
|
}
|
2021-03-13 13:44:51 +00:00
|
|
|
TyKind::Alias(AliasTy::Opaque(opaque_ty)) => {
|
2021-03-08 19:08:30 +00:00
|
|
|
if let Some(bounds) = ty.impl_trait_bounds(db) {
|
|
|
|
walk_bounds(db, &type_.derived(ty.clone()), &bounds, cb);
|
|
|
|
}
|
|
|
|
|
2021-03-14 15:33:27 +00:00
|
|
|
walk_substs(db, type_, &opaque_ty.substitution, cb);
|
2021-03-08 19:08:30 +00:00
|
|
|
}
|
2021-03-13 13:44:51 +00:00
|
|
|
TyKind::Placeholder(_) => {
|
2021-03-08 19:08:30 +00:00
|
|
|
if let Some(bounds) = ty.impl_trait_bounds(db) {
|
|
|
|
walk_bounds(db, &type_.derived(ty.clone()), &bounds, cb);
|
|
|
|
}
|
|
|
|
}
|
2021-03-13 13:44:51 +00:00
|
|
|
TyKind::Dyn(bounds) => {
|
2021-03-21 12:22:22 +00:00
|
|
|
walk_bounds(
|
|
|
|
db,
|
|
|
|
&type_.derived(ty.clone()),
|
|
|
|
bounds.bounds.skip_binders().interned(),
|
|
|
|
cb,
|
|
|
|
);
|
2021-03-08 19:08:30 +00:00
|
|
|
}
|
|
|
|
|
2021-04-05 20:08:16 +00:00
|
|
|
TyKind::Ref(_, _, ty)
|
|
|
|
| TyKind::Raw(_, ty)
|
2021-04-06 09:45:41 +00:00
|
|
|
| TyKind::Array(ty, _)
|
2021-04-05 20:08:16 +00:00
|
|
|
| TyKind::Slice(ty) => {
|
2021-03-14 16:40:55 +00:00
|
|
|
walk_type(db, &type_.derived(ty.clone()), cb);
|
|
|
|
}
|
|
|
|
|
2021-04-07 15:26:01 +00:00
|
|
|
TyKind::FnDef(_, substs)
|
|
|
|
| TyKind::Tuple(_, substs)
|
|
|
|
| TyKind::Closure(.., substs) => {
|
|
|
|
walk_substs(db, type_, &substs, cb);
|
|
|
|
}
|
|
|
|
TyKind::Function(hir_ty::FnPointer { substitution, .. }) => {
|
|
|
|
walk_substs(db, type_, &substitution.0, cb);
|
|
|
|
}
|
|
|
|
|
2021-03-08 19:08:30 +00:00
|
|
|
_ => {}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
walk_type(db, self, &mut cb);
|
|
|
|
}
|
2021-03-16 15:45:46 +00:00
|
|
|
|
2021-05-15 18:18:54 +00:00
|
|
|
pub fn could_unify_with(&self, db: &dyn HirDatabase, other: &Type) -> bool {
|
2021-05-23 10:52:41 +00:00
|
|
|
let tys = hir_ty::replace_errors_with_variables(&(self.ty.clone(), other.ty.clone()));
|
2021-05-15 18:28:07 +00:00
|
|
|
could_unify(db, self.env.clone(), &tys)
|
2021-03-16 15:45:46 +00:00
|
|
|
}
|
2021-03-08 19:08:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// FIXME: closures
|
|
|
|
#[derive(Debug)]
|
|
|
|
pub struct Callable {
|
|
|
|
ty: Type,
|
|
|
|
sig: CallableSig,
|
|
|
|
def: Option<CallableDefId>,
|
|
|
|
pub(crate) is_bound_method: bool,
|
|
|
|
}
|
|
|
|
|
|
|
|
pub enum CallableKind {
|
|
|
|
Function(Function),
|
|
|
|
TupleStruct(Struct),
|
|
|
|
TupleEnumVariant(Variant),
|
|
|
|
Closure,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Callable {
|
|
|
|
pub fn kind(&self) -> CallableKind {
|
|
|
|
match self.def {
|
|
|
|
Some(CallableDefId::FunctionId(it)) => CallableKind::Function(it.into()),
|
|
|
|
Some(CallableDefId::StructId(it)) => CallableKind::TupleStruct(it.into()),
|
|
|
|
Some(CallableDefId::EnumVariantId(it)) => CallableKind::TupleEnumVariant(it.into()),
|
|
|
|
None => CallableKind::Closure,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
pub fn receiver_param(&self, db: &dyn HirDatabase) -> Option<ast::SelfParam> {
|
|
|
|
let func = match self.def {
|
|
|
|
Some(CallableDefId::FunctionId(it)) if self.is_bound_method => it,
|
|
|
|
_ => return None,
|
|
|
|
};
|
|
|
|
let src = func.lookup(db.upcast()).source(db.upcast());
|
|
|
|
let param_list = src.value.param_list()?;
|
|
|
|
param_list.self_param()
|
|
|
|
}
|
|
|
|
pub fn n_params(&self) -> usize {
|
|
|
|
self.sig.params().len() - if self.is_bound_method { 1 } else { 0 }
|
|
|
|
}
|
|
|
|
pub fn params(
|
|
|
|
&self,
|
|
|
|
db: &dyn HirDatabase,
|
|
|
|
) -> Vec<(Option<Either<ast::SelfParam, ast::Pat>>, Type)> {
|
|
|
|
let types = self
|
|
|
|
.sig
|
|
|
|
.params()
|
|
|
|
.iter()
|
|
|
|
.skip(if self.is_bound_method { 1 } else { 0 })
|
|
|
|
.map(|ty| self.ty.derived(ty.clone()));
|
|
|
|
let patterns = match self.def {
|
|
|
|
Some(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(|it| it.pat().map(Either::Right)))
|
|
|
|
})
|
|
|
|
}
|
|
|
|
_ => None,
|
|
|
|
};
|
|
|
|
patterns.into_iter().flatten().chain(iter::repeat(None)).zip(types).collect()
|
|
|
|
}
|
|
|
|
pub fn return_type(&self) -> Type {
|
|
|
|
self.ty.derived(self.sig.ret().clone())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// For IDE only
|
|
|
|
#[derive(Debug, PartialEq, Eq, Hash)]
|
|
|
|
pub enum ScopeDef {
|
|
|
|
ModuleDef(ModuleDef),
|
|
|
|
MacroDef(MacroDef),
|
|
|
|
GenericParam(GenericParam),
|
|
|
|
ImplSelfType(Impl),
|
|
|
|
AdtSelfType(Adt),
|
|
|
|
Local(Local),
|
2021-03-20 23:59:45 +00:00
|
|
|
Label(Label),
|
2021-03-08 19:08:30 +00:00
|
|
|
Unknown,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl ScopeDef {
|
2021-03-25 19:03:20 +00:00
|
|
|
pub fn all_items(def: PerNs) -> ArrayVec<Self, 3> {
|
2021-03-08 19:08:30 +00:00
|
|
|
let mut items = ArrayVec::new();
|
|
|
|
|
|
|
|
match (def.take_types(), def.take_values()) {
|
|
|
|
(Some(m1), None) => items.push(ScopeDef::ModuleDef(m1.into())),
|
|
|
|
(None, Some(m2)) => items.push(ScopeDef::ModuleDef(m2.into())),
|
|
|
|
(Some(m1), Some(m2)) => {
|
|
|
|
// Some items, like unit structs and enum variants, are
|
|
|
|
// returned as both a type and a value. Here we want
|
|
|
|
// to de-duplicate them.
|
|
|
|
if m1 != m2 {
|
|
|
|
items.push(ScopeDef::ModuleDef(m1.into()));
|
|
|
|
items.push(ScopeDef::ModuleDef(m2.into()));
|
|
|
|
} else {
|
|
|
|
items.push(ScopeDef::ModuleDef(m1.into()));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
(None, None) => {}
|
|
|
|
};
|
|
|
|
|
|
|
|
if let Some(macro_def_id) = def.take_macros() {
|
|
|
|
items.push(ScopeDef::MacroDef(macro_def_id.into()));
|
|
|
|
}
|
|
|
|
|
|
|
|
if items.is_empty() {
|
|
|
|
items.push(ScopeDef::Unknown);
|
|
|
|
}
|
|
|
|
|
|
|
|
items
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-02-28 22:05:22 +00:00
|
|
|
impl From<ItemInNs> for ScopeDef {
|
|
|
|
fn from(item: ItemInNs) -> Self {
|
|
|
|
match item {
|
|
|
|
ItemInNs::Types(id) => ScopeDef::ModuleDef(id.into()),
|
|
|
|
ItemInNs::Values(id) => ScopeDef::ModuleDef(id.into()),
|
|
|
|
ItemInNs::Macros(id) => ScopeDef::MacroDef(id.into()),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-03-08 19:08:30 +00:00
|
|
|
pub trait HasVisibility {
|
|
|
|
fn visibility(&self, db: &dyn HirDatabase) -> Visibility;
|
|
|
|
fn is_visible_from(&self, db: &dyn HirDatabase, module: Module) -> bool {
|
|
|
|
let vis = self.visibility(db);
|
|
|
|
vis.is_visible_from(db.upcast(), module.id)
|
|
|
|
}
|
|
|
|
}
|