mirror of
https://github.com/rust-lang/rust-analyzer
synced 2025-01-12 13:18:47 +00:00
Merge from rustc
This commit is contained in:
commit
a2336ab277
72 changed files with 8775 additions and 2159 deletions
2
.github/workflows/ci.yaml
vendored
2
.github/workflows/ci.yaml
vendored
|
@ -220,7 +220,7 @@ jobs:
|
|||
timeout-minutes: 10
|
||||
env:
|
||||
FORCE_COLOR: 1
|
||||
TYPOS_VERSION: v1.18.0
|
||||
TYPOS_VERSION: v1.28.3
|
||||
steps:
|
||||
- name: download typos
|
||||
run: curl -LsSf https://github.com/crate-ci/typos/releases/download/$TYPOS_VERSION/typos-$TYPOS_VERSION-x86_64-unknown-linux-musl.tar.gz | tar zxf - -C ${CARGO_HOME:-~/.cargo}/bin
|
||||
|
|
|
@ -16,7 +16,8 @@ extend-ignore-re = [
|
|||
"raison d'être",
|
||||
"inout",
|
||||
"INOUT",
|
||||
"optin"
|
||||
"optin",
|
||||
"=Pn",
|
||||
]
|
||||
|
||||
[default.extend-words]
|
||||
|
@ -26,8 +27,12 @@ fo = "fo"
|
|||
ket = "ket"
|
||||
makro = "makro"
|
||||
trivias = "trivias"
|
||||
thir = "thir"
|
||||
jod = "jod"
|
||||
|
||||
[default.extend-identifiers]
|
||||
anc = "anc"
|
||||
datas = "datas"
|
||||
impl_froms = "impl_froms"
|
||||
selfs = "selfs"
|
||||
taits = "taits"
|
||||
|
|
6
Cargo.lock
generated
6
Cargo.lock
generated
|
@ -389,6 +389,10 @@ version = "0.1.5"
|
|||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "9bda8e21c04aca2ae33ffc2fd8c23134f3cac46db123ba97bd9d3f3b8a4a85e1"
|
||||
|
||||
[[package]]
|
||||
name = "edition"
|
||||
version = "0.0.0"
|
||||
|
||||
[[package]]
|
||||
name = "either"
|
||||
version = "1.13.0"
|
||||
|
@ -1266,6 +1270,7 @@ name = "parser"
|
|||
version = "0.0.0"
|
||||
dependencies = [
|
||||
"drop_bomb",
|
||||
"edition",
|
||||
"expect-test",
|
||||
"limit",
|
||||
"ra-ap-rustc_lexer",
|
||||
|
@ -2662,6 +2667,7 @@ version = "0.1.0"
|
|||
dependencies = [
|
||||
"anyhow",
|
||||
"directories",
|
||||
"edition",
|
||||
"either",
|
||||
"flate2",
|
||||
"itertools",
|
||||
|
|
|
@ -83,6 +83,7 @@ toolchain = { path = "./crates/toolchain", version = "0.0.0" }
|
|||
tt = { path = "./crates/tt", version = "0.0.0" }
|
||||
vfs-notify = { path = "./crates/vfs-notify", version = "0.0.0" }
|
||||
vfs = { path = "./crates/vfs", version = "0.0.0" }
|
||||
edition = { path = "./crates/edition", version = "0.0.0" }
|
||||
|
||||
ra-ap-rustc_lexer = { version = "0.85", default-features = false }
|
||||
ra-ap-rustc_parse_format = { version = "0.85", default-features = false }
|
||||
|
|
13
crates/edition/Cargo.toml
Normal file
13
crates/edition/Cargo.toml
Normal file
|
@ -0,0 +1,13 @@
|
|||
[package]
|
||||
name = "edition"
|
||||
version = "0.0.0"
|
||||
rust-version.workspace = true
|
||||
edition.workspace = true
|
||||
license.workspace = true
|
||||
authors.workspace = true
|
||||
repository.workspace = true
|
||||
|
||||
[dependencies]
|
||||
|
||||
[lints]
|
||||
workspace = true
|
|
@ -1,6 +1,5 @@
|
|||
//! The edition of the Rust language used in a crate.
|
||||
// Ideally this would be defined in the span crate, but the dependency chain is all over the place
|
||||
// wrt to span, parser and syntax.
|
||||
// This should live in a separate crate because we use it in both actual code and codegen.
|
||||
use std::fmt;
|
||||
|
||||
#[derive(Debug, Clone, Copy, PartialEq, Eq, PartialOrd, Ord, Hash)]
|
|
@ -74,7 +74,7 @@ impl BuiltinFnLikeExpander {
|
|||
}
|
||||
|
||||
pub fn is_asm(&self) -> bool {
|
||||
matches!(self, Self::Asm | Self::GlobalAsm)
|
||||
matches!(self, Self::Asm | Self::GlobalAsm | Self::NakedAsm)
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -122,6 +122,7 @@ register_builtin! {
|
|||
(stringify, Stringify) => stringify_expand,
|
||||
(asm, Asm) => asm_expand,
|
||||
(global_asm, GlobalAsm) => asm_expand,
|
||||
(naked_asm, NakedAsm) => asm_expand,
|
||||
(cfg, Cfg) => cfg_expand,
|
||||
(core_panic, CorePanic) => panic_expand,
|
||||
(std_panic, StdPanic) => panic_expand,
|
||||
|
|
|
@ -237,7 +237,7 @@ pub const INERT_ATTRIBUTES: &[BuiltinAttribute] = &[
|
|||
template!(List: "address, kcfi, memory, thread"), DuplicatesOk,
|
||||
experimental!(no_sanitize)
|
||||
),
|
||||
gated!(coverage, Normal, template!(Word, List: "on|off"), WarnFollowing, coverage_attribute, experimental!(coverage)),
|
||||
ungated!(coverage, Normal, template!(Word, List: "on|off"), WarnFollowing),
|
||||
|
||||
ungated!(
|
||||
doc, Normal, template!(List: "hidden|inline|...", NameValueStr: "string"), DuplicatesOk
|
||||
|
|
|
@ -28,6 +28,7 @@ use rustc_hash::FxHashMap;
|
|||
use stdx::TupleExt;
|
||||
use triomphe::Arc;
|
||||
|
||||
use core::fmt;
|
||||
use std::hash::Hash;
|
||||
|
||||
use base_db::{ra_salsa::InternValueTrivial, CrateId};
|
||||
|
@ -147,6 +148,10 @@ impl ExpandError {
|
|||
pub fn span(&self) -> Span {
|
||||
self.inner.1
|
||||
}
|
||||
|
||||
pub fn render_to_string(&self, db: &dyn ExpandDatabase) -> RenderedExpandError {
|
||||
self.inner.0.render_to_string(db)
|
||||
}
|
||||
}
|
||||
|
||||
#[derive(Debug, PartialEq, Eq, Clone, Hash)]
|
||||
|
@ -164,18 +169,18 @@ pub enum ExpandErrorKind {
|
|||
ProcMacroPanic(Box<str>),
|
||||
}
|
||||
|
||||
impl ExpandError {
|
||||
pub fn render_to_string(&self, db: &dyn ExpandDatabase) -> RenderedExpandError {
|
||||
self.inner.0.render_to_string(db)
|
||||
}
|
||||
}
|
||||
|
||||
pub struct RenderedExpandError {
|
||||
pub message: String,
|
||||
pub error: bool,
|
||||
pub kind: &'static str,
|
||||
}
|
||||
|
||||
impl fmt::Display for RenderedExpandError {
|
||||
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
||||
write!(f, "{}", self.message)
|
||||
}
|
||||
}
|
||||
|
||||
impl RenderedExpandError {
|
||||
const GENERAL_KIND: &str = "macro-error";
|
||||
}
|
||||
|
|
|
@ -1053,8 +1053,21 @@ impl HirDisplay for Ty {
|
|||
generic_args_sans_defaults(f, Some(generic_def_id), parameters);
|
||||
assert!(params_len >= parameters.len());
|
||||
let defaults = params_len - parameters.len();
|
||||
let without_impl =
|
||||
self_param as usize + type_ + const_ + lifetime - defaults;
|
||||
|
||||
// Normally, functions cannot have default parameters, but they can,
|
||||
// for function-like things such as struct names or enum variants.
|
||||
// The former cannot have defaults but parents, and the later cannot have
|
||||
// parents but defaults.
|
||||
// So, if `parent_len` > 0, it have a parent and thus it doesn't have any
|
||||
// default. Therefore, we shouldn't subtract defaults because those defaults
|
||||
// are from their parents.
|
||||
// And if `parent_len` == 0, either parents don't exists or they don't have
|
||||
// any defaults. Thus, we can - and should - subtract defaults.
|
||||
let without_impl = if parent_len > 0 {
|
||||
params_len - parent_len - impl_
|
||||
} else {
|
||||
params_len - parent_len - impl_ - defaults
|
||||
};
|
||||
// parent's params (those from enclosing impl or trait, if any).
|
||||
let (fn_params, parent_params) = parameters.split_at(without_impl + impl_);
|
||||
|
||||
|
|
|
@ -55,7 +55,7 @@ impl chalk_ir::interner::Interner for Interner {
|
|||
type InternedConst = Interned<InternedWrapper<ConstData>>;
|
||||
type InternedConcreteConst = ConstScalar;
|
||||
type InternedGenericArg = GenericArgData;
|
||||
// We could do the following, but that saves "only" 20mb on self while increasing inferecene
|
||||
// We could do the following, but that saves "only" 20mb on self while increasing inference
|
||||
// time by ~2.5%
|
||||
// type InternedGoal = Interned<InternedWrapper<GoalData>>;
|
||||
type InternedGoal = Arc<GoalData>;
|
||||
|
|
|
@ -4380,7 +4380,7 @@ fn test() {
|
|||
fn associated_type_in_struct_expr_path() {
|
||||
// FIXME: All annotation should be resolvable.
|
||||
// For lines marked as unstable, see rust-lang/rust#86935.
|
||||
// FIXME: Remove the comments once stablized.
|
||||
// FIXME: Remove the comments once stabilized.
|
||||
check_types(
|
||||
r#"
|
||||
trait Trait {
|
||||
|
@ -4416,7 +4416,7 @@ impl Trait for () {
|
|||
fn associated_type_in_struct_expr_path_enum() {
|
||||
// FIXME: All annotation should be resolvable.
|
||||
// For lines marked as unstable, see rust-lang/rust#86935.
|
||||
// FIXME: Remove the comments once stablized.
|
||||
// FIXME: Remove the comments once stabilized.
|
||||
check_types(
|
||||
r#"
|
||||
trait Trait {
|
||||
|
|
|
@ -2649,24 +2649,31 @@ impl Const {
|
|||
Type::from_value_def(db, self.id)
|
||||
}
|
||||
|
||||
/// Evaluate the constant and return the result as a string.
|
||||
///
|
||||
/// This function is intended for IDE assistance, different from [`Const::render_eval`].
|
||||
pub fn eval(self, db: &dyn HirDatabase) -> Result<String, ConstEvalError> {
|
||||
let c = db.const_eval(self.id.into(), Substitution::empty(Interner), None)?;
|
||||
Ok(format!("{}", c.display(db, self.krate(db).edition(db))))
|
||||
/// Evaluate the constant.
|
||||
pub fn eval(self, db: &dyn HirDatabase) -> Result<EvaluatedConst, ConstEvalError> {
|
||||
db.const_eval(self.id.into(), Substitution::empty(Interner), None)
|
||||
.map(|it| EvaluatedConst { const_: it, def: self.id.into() })
|
||||
}
|
||||
}
|
||||
|
||||
impl HasVisibility for Const {
|
||||
fn visibility(&self, db: &dyn HirDatabase) -> Visibility {
|
||||
db.const_visibility(self.id)
|
||||
}
|
||||
}
|
||||
|
||||
pub struct EvaluatedConst {
|
||||
def: DefWithBodyId,
|
||||
const_: hir_ty::Const,
|
||||
}
|
||||
|
||||
impl EvaluatedConst {
|
||||
pub fn render(&self, db: &dyn HirDatabase, edition: Edition) -> String {
|
||||
format!("{}", self.const_.display(db, edition))
|
||||
}
|
||||
|
||||
/// Evaluate the constant and return the result as a string, with more detailed information.
|
||||
///
|
||||
/// This function is intended for user-facing display.
|
||||
pub fn render_eval(
|
||||
self,
|
||||
db: &dyn HirDatabase,
|
||||
edition: Edition,
|
||||
) -> Result<String, ConstEvalError> {
|
||||
let c = db.const_eval(self.id.into(), Substitution::empty(Interner), None)?;
|
||||
let data = &c.data(Interner);
|
||||
pub fn render_debug(&self, db: &dyn HirDatabase) -> Result<String, MirEvalError> {
|
||||
let data = self.const_.data(Interner);
|
||||
if let TyKind::Scalar(s) = data.ty.kind(Interner) {
|
||||
if matches!(s, Scalar::Int(_) | Scalar::Uint(_)) {
|
||||
if let hir_ty::ConstValue::Concrete(c) = &data.value {
|
||||
|
@ -2689,17 +2696,7 @@ impl Const {
|
|||
}
|
||||
}
|
||||
}
|
||||
if let Ok(s) = mir::render_const_using_debug_impl(db, self.id.into(), &c) {
|
||||
Ok(s)
|
||||
} else {
|
||||
Ok(format!("{}", c.display(db, edition)))
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
impl HasVisibility for Const {
|
||||
fn visibility(&self, db: &dyn HirDatabase) -> Visibility {
|
||||
db.const_visibility(self.id)
|
||||
mir::render_const_using_debug_impl(db, self.def, &self.const_)
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -2729,51 +2726,10 @@ impl Static {
|
|||
Type::from_value_def(db, self.id)
|
||||
}
|
||||
|
||||
/// Evaluate the static and return the result as a string.
|
||||
///
|
||||
/// This function is intended for IDE assistance, different from [`Static::render_eval`].
|
||||
pub fn eval(self, db: &dyn HirDatabase) -> Result<String, ConstEvalError> {
|
||||
let c = db.const_eval(self.id.into(), Substitution::empty(Interner), None)?;
|
||||
Ok(format!("{}", c.display(db, self.krate(db).edition(db))))
|
||||
}
|
||||
|
||||
/// Evaluate the static and return the result as a string, with more detailed information.
|
||||
///
|
||||
/// This function is intended for user-facing display.
|
||||
pub fn render_eval(
|
||||
self,
|
||||
db: &dyn HirDatabase,
|
||||
edition: Edition,
|
||||
) -> Result<String, ConstEvalError> {
|
||||
let c = db.const_eval(self.id.into(), Substitution::empty(Interner), None)?;
|
||||
let data = &c.data(Interner);
|
||||
if let TyKind::Scalar(s) = data.ty.kind(Interner) {
|
||||
if matches!(s, Scalar::Int(_) | Scalar::Uint(_)) {
|
||||
if let hir_ty::ConstValue::Concrete(c) = &data.value {
|
||||
if let hir_ty::ConstScalar::Bytes(b, _) = &c.interned {
|
||||
let value = u128::from_le_bytes(mir::pad16(b, false));
|
||||
let value_signed =
|
||||
i128::from_le_bytes(mir::pad16(b, matches!(s, Scalar::Int(_))));
|
||||
let mut result = if let Scalar::Int(_) = s {
|
||||
value_signed.to_string()
|
||||
} else {
|
||||
value.to_string()
|
||||
};
|
||||
if value >= 10 {
|
||||
format_to!(result, " ({value:#X})");
|
||||
return Ok(result);
|
||||
} else {
|
||||
return Ok(result);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
if let Ok(s) = mir::render_const_using_debug_impl(db, self.id.into(), &c) {
|
||||
Ok(s)
|
||||
} else {
|
||||
Ok(format!("{}", c.display(db, edition)))
|
||||
}
|
||||
/// Evaluate the static initializer.
|
||||
pub fn eval(self, db: &dyn HirDatabase) -> Result<EvaluatedConst, ConstEvalError> {
|
||||
db.const_eval(self.id.into(), Substitution::empty(Interner), None)
|
||||
.map(|it| EvaluatedConst { const_: it, def: self.id.into() })
|
||||
}
|
||||
}
|
||||
|
||||
|
|
|
@ -28,7 +28,7 @@ use hir_expand::{
|
|||
hygiene::SyntaxContextExt as _,
|
||||
inert_attr_macro::find_builtin_attr_idx,
|
||||
name::AsName,
|
||||
FileRange, InMacroFile, MacroCallId, MacroFileId, MacroFileIdExt,
|
||||
ExpandResult, FileRange, InMacroFile, MacroCallId, MacroFileId, MacroFileIdExt,
|
||||
};
|
||||
use intern::Symbol;
|
||||
use itertools::Itertools;
|
||||
|
@ -381,7 +381,13 @@ impl<'db> SemanticsImpl<'db> {
|
|||
node
|
||||
}
|
||||
|
||||
pub fn expand(&self, macro_call: &ast::MacroCall) -> Option<SyntaxNode> {
|
||||
pub fn expand(&self, file_id: MacroFileId) -> ExpandResult<SyntaxNode> {
|
||||
let res = self.db.parse_macro_expansion(file_id).map(|it| it.0.syntax_node());
|
||||
self.cache(res.value.clone(), file_id.into());
|
||||
res
|
||||
}
|
||||
|
||||
pub fn expand_macro_call(&self, macro_call: &ast::MacroCall) -> Option<SyntaxNode> {
|
||||
let sa = self.analyze_no_infer(macro_call.syntax())?;
|
||||
|
||||
let macro_call = InFile::new(sa.file_id, macro_call);
|
||||
|
@ -412,7 +418,10 @@ impl<'db> SemanticsImpl<'db> {
|
|||
|
||||
/// Expands the macro if it isn't one of the built-in ones that expand to custom syntax or dummy
|
||||
/// expansions.
|
||||
pub fn expand_allowed_builtins(&self, macro_call: &ast::MacroCall) -> Option<SyntaxNode> {
|
||||
pub fn expand_allowed_builtins(
|
||||
&self,
|
||||
macro_call: &ast::MacroCall,
|
||||
) -> Option<ExpandResult<SyntaxNode>> {
|
||||
let sa = self.analyze_no_infer(macro_call.syntax())?;
|
||||
|
||||
let macro_call = InFile::new(sa.file_id, macro_call);
|
||||
|
@ -434,6 +443,7 @@ impl<'db> SemanticsImpl<'db> {
|
|||
| BuiltinFnLikeExpander::ModulePath
|
||||
| BuiltinFnLikeExpander::Asm
|
||||
| BuiltinFnLikeExpander::GlobalAsm
|
||||
| BuiltinFnLikeExpander::NakedAsm
|
||||
| BuiltinFnLikeExpander::LogSyntax
|
||||
| BuiltinFnLikeExpander::TraceMacros
|
||||
| BuiltinFnLikeExpander::FormatArgs
|
||||
|
@ -447,15 +457,15 @@ impl<'db> SemanticsImpl<'db> {
|
|||
return None;
|
||||
}
|
||||
|
||||
let node = self.parse_or_expand(file_id.into());
|
||||
let node = self.expand(file_id);
|
||||
Some(node)
|
||||
}
|
||||
|
||||
/// If `item` has an attribute macro attached to it, expands it.
|
||||
pub fn expand_attr_macro(&self, item: &ast::Item) -> Option<SyntaxNode> {
|
||||
pub fn expand_attr_macro(&self, item: &ast::Item) -> Option<ExpandResult<SyntaxNode>> {
|
||||
let src = self.wrap_node_infile(item.clone());
|
||||
let macro_call_id = self.with_ctx(|ctx| ctx.item_to_macro_call(src.as_ref()))?;
|
||||
Some(self.parse_or_expand(macro_call_id.as_file()))
|
||||
Some(self.expand(macro_call_id.as_macro_file()))
|
||||
}
|
||||
|
||||
pub fn expand_derive_as_pseudo_attr_macro(&self, attr: &ast::Attr) -> Option<SyntaxNode> {
|
||||
|
@ -479,15 +489,16 @@ impl<'db> SemanticsImpl<'db> {
|
|||
})
|
||||
}
|
||||
|
||||
pub fn expand_derive_macro(&self, attr: &ast::Attr) -> Option<Vec<SyntaxNode>> {
|
||||
pub fn expand_derive_macro(&self, attr: &ast::Attr) -> Option<Vec<ExpandResult<SyntaxNode>>> {
|
||||
let res: Vec<_> = self
|
||||
.derive_macro_calls(attr)?
|
||||
.into_iter()
|
||||
.flat_map(|call| {
|
||||
let file_id = call?.as_file();
|
||||
let node = self.db.parse_or_expand(file_id);
|
||||
self.cache(node.clone(), file_id);
|
||||
Some(node)
|
||||
let file_id = call?.as_macro_file();
|
||||
let ExpandResult { value, err } = self.db.parse_macro_expansion(file_id);
|
||||
let root_node = value.0.syntax_node();
|
||||
self.cache(root_node.clone(), file_id.into());
|
||||
Some(ExpandResult { value: root_node, err })
|
||||
})
|
||||
.collect();
|
||||
Some(res)
|
||||
|
@ -555,7 +566,7 @@ impl<'db> SemanticsImpl<'db> {
|
|||
|
||||
/// Expand the macro call with a different token tree, mapping the `token_to_map` down into the
|
||||
/// expansion. `token_to_map` should be a token from the `speculative args` node.
|
||||
pub fn speculative_expand(
|
||||
pub fn speculative_expand_macro_call(
|
||||
&self,
|
||||
actual_macro_call: &ast::MacroCall,
|
||||
speculative_args: &ast::TokenTree,
|
||||
|
|
|
@ -7,6 +7,7 @@ use hir::{
|
|||
TypeInfo, TypeParam,
|
||||
};
|
||||
use ide_db::{
|
||||
assists::GroupLabel,
|
||||
defs::{Definition, NameRefClass},
|
||||
famous_defs::FamousDefs,
|
||||
helpers::mod_path_to_ast,
|
||||
|
@ -104,7 +105,8 @@ pub(crate) fn extract_function(acc: &mut Assists, ctx: &AssistContext<'_>) -> Op
|
|||
|
||||
let scope = ImportScope::find_insert_use_container(&node, &ctx.sema)?;
|
||||
|
||||
acc.add(
|
||||
acc.add_group(
|
||||
&GroupLabel("Extract into...".to_owned()),
|
||||
AssistId("extract_function", crate::AssistKind::RefactorExtract),
|
||||
"Extract into function",
|
||||
target_range,
|
||||
|
@ -5009,7 +5011,7 @@ fn $0fun_name(bar: &str) {
|
|||
}
|
||||
|
||||
#[test]
|
||||
fn unresolveable_types_default_to_placeholder() {
|
||||
fn unresolvable_types_default_to_placeholder() {
|
||||
check_assist(
|
||||
extract_function,
|
||||
r#"
|
||||
|
|
File diff suppressed because it is too large
Load diff
|
@ -2055,7 +2055,7 @@ fn bar(closure: impl Fn(i64) -> i64) {
|
|||
}
|
||||
|
||||
#[test]
|
||||
fn unresolveable_types_default_to_placeholder() {
|
||||
fn unresolvable_types_default_to_placeholder() {
|
||||
check_assist(
|
||||
generate_function,
|
||||
r"
|
||||
|
|
|
@ -1,3 +1,4 @@
|
|||
use hir::HasCrate;
|
||||
use syntax::{ast, AstNode};
|
||||
|
||||
use crate::{AssistContext, AssistId, AssistKind, Assists};
|
||||
|
@ -51,7 +52,10 @@ pub(crate) fn inline_const_as_literal(acc: &mut Assists, ctx: &AssistContext<'_>
|
|||
| ast::Expr::MatchExpr(_)
|
||||
| ast::Expr::MacroExpr(_)
|
||||
| ast::Expr::BinExpr(_)
|
||||
| ast::Expr::CallExpr(_) => konst.eval(ctx.sema.db).ok()?,
|
||||
| ast::Expr::CallExpr(_) => konst
|
||||
.eval(ctx.sema.db)
|
||||
.ok()?
|
||||
.render(ctx.sema.db, konst.krate(ctx.sema.db).edition(ctx.sema.db)),
|
||||
_ => return None,
|
||||
};
|
||||
|
||||
|
|
|
@ -36,7 +36,7 @@ pub(crate) fn move_guard_to_arm_body(acc: &mut Assists, ctx: &AssistContext<'_>)
|
|||
let match_arm = ctx.find_node_at_offset::<MatchArm>()?;
|
||||
let guard = match_arm.guard()?;
|
||||
if ctx.offset() > guard.syntax().text_range().end() {
|
||||
cov_mark::hit!(move_guard_unapplicable_in_arm_body);
|
||||
cov_mark::hit!(move_guard_inapplicable_in_arm_body);
|
||||
return None;
|
||||
}
|
||||
let space_before_guard = guard.syntax().prev_sibling_or_token();
|
||||
|
@ -219,7 +219,7 @@ mod tests {
|
|||
|
||||
#[test]
|
||||
fn move_guard_to_arm_body_range() {
|
||||
cov_mark::check!(move_guard_unapplicable_in_arm_body);
|
||||
cov_mark::check!(move_guard_inapplicable_in_arm_body);
|
||||
check_assist_not_applicable(
|
||||
move_guard_to_arm_body,
|
||||
r#"
|
||||
|
|
|
@ -1,19 +1,17 @@
|
|||
use hir::{HirDisplay, ModuleDef, PathResolution, Semantics};
|
||||
use hir::HirDisplay;
|
||||
use ide_db::{
|
||||
assists::{AssistId, AssistKind},
|
||||
defs::Definition,
|
||||
syntax_helpers::node_ext::preorder_expr,
|
||||
RootDatabase,
|
||||
};
|
||||
use stdx::to_upper_snake_case;
|
||||
use syntax::{
|
||||
ast::{self, make, HasName},
|
||||
ted, AstNode, WalkEvent,
|
||||
ted, AstNode,
|
||||
};
|
||||
|
||||
use crate::{
|
||||
assist_context::{AssistContext, Assists},
|
||||
utils,
|
||||
utils::{self},
|
||||
};
|
||||
|
||||
// Assist: promote_local_to_const
|
||||
|
@ -63,7 +61,7 @@ pub(crate) fn promote_local_to_const(acc: &mut Assists, ctx: &AssistContext<'_>)
|
|||
};
|
||||
|
||||
let initializer = let_stmt.initializer()?;
|
||||
if !is_body_const(&ctx.sema, &initializer) {
|
||||
if !utils::is_body_const(&ctx.sema, &initializer) {
|
||||
cov_mark::hit!(promote_local_non_const);
|
||||
return None;
|
||||
}
|
||||
|
@ -103,40 +101,6 @@ pub(crate) fn promote_local_to_const(acc: &mut Assists, ctx: &AssistContext<'_>)
|
|||
)
|
||||
}
|
||||
|
||||
fn is_body_const(sema: &Semantics<'_, RootDatabase>, expr: &ast::Expr) -> bool {
|
||||
let mut is_const = true;
|
||||
preorder_expr(expr, &mut |ev| {
|
||||
let expr = match ev {
|
||||
WalkEvent::Enter(_) if !is_const => return true,
|
||||
WalkEvent::Enter(expr) => expr,
|
||||
WalkEvent::Leave(_) => return false,
|
||||
};
|
||||
match expr {
|
||||
ast::Expr::CallExpr(call) => {
|
||||
if let Some(ast::Expr::PathExpr(path_expr)) = call.expr() {
|
||||
if let Some(PathResolution::Def(ModuleDef::Function(func))) =
|
||||
path_expr.path().and_then(|path| sema.resolve_path(&path))
|
||||
{
|
||||
is_const &= func.is_const(sema.db);
|
||||
}
|
||||
}
|
||||
}
|
||||
ast::Expr::MethodCallExpr(call) => {
|
||||
is_const &=
|
||||
sema.resolve_method_call(&call).map(|it| it.is_const(sema.db)).unwrap_or(true)
|
||||
}
|
||||
ast::Expr::ForExpr(_)
|
||||
| ast::Expr::ReturnExpr(_)
|
||||
| ast::Expr::TryExpr(_)
|
||||
| ast::Expr::YieldExpr(_)
|
||||
| ast::Expr::AwaitExpr(_) => is_const = false,
|
||||
_ => (),
|
||||
}
|
||||
!is_const
|
||||
});
|
||||
is_const
|
||||
}
|
||||
|
||||
#[cfg(test)]
|
||||
mod tests {
|
||||
use crate::tests::{check_assist, check_assist_not_applicable};
|
||||
|
|
|
@ -139,7 +139,7 @@ fn add_assist(
|
|||
let trait_path = make::ty_path(replace_trait_path.clone());
|
||||
|
||||
match (ctx.config.snippet_cap, impl_def_with_items) {
|
||||
(None, _) => {
|
||||
(None, None) => {
|
||||
let impl_def = generate_trait_impl(adt, trait_path);
|
||||
|
||||
ted::insert_all(
|
||||
|
@ -147,6 +147,12 @@ fn add_assist(
|
|||
vec![make::tokens::blank_line().into(), impl_def.syntax().clone().into()],
|
||||
);
|
||||
}
|
||||
(None, Some((impl_def, _))) => {
|
||||
ted::insert_all(
|
||||
insert_after,
|
||||
vec![make::tokens::blank_line().into(), impl_def.syntax().clone().into()],
|
||||
);
|
||||
}
|
||||
(Some(cap), None) => {
|
||||
let impl_def = generate_trait_impl(adt, trait_path);
|
||||
|
||||
|
@ -272,7 +278,7 @@ fn update_attribute(
|
|||
|
||||
#[cfg(test)]
|
||||
mod tests {
|
||||
use crate::tests::{check_assist, check_assist_not_applicable};
|
||||
use crate::tests::{check_assist, check_assist_no_snippet_cap, check_assist_not_applicable};
|
||||
|
||||
use super::*;
|
||||
|
||||
|
@ -297,6 +303,30 @@ impl core::fmt::Debug for Foo {
|
|||
f.debug_struct("Foo").field("bar", &self.bar).finish()
|
||||
}
|
||||
}
|
||||
"#,
|
||||
)
|
||||
}
|
||||
#[test]
|
||||
fn add_custom_impl_without_snippet() {
|
||||
check_assist_no_snippet_cap(
|
||||
replace_derive_with_manual_impl,
|
||||
r#"
|
||||
//- minicore: fmt, derive
|
||||
#[derive(Debu$0g)]
|
||||
struct Foo {
|
||||
bar: String,
|
||||
}
|
||||
"#,
|
||||
r#"
|
||||
struct Foo {
|
||||
bar: String,
|
||||
}
|
||||
|
||||
impl core::fmt::Debug for Foo {
|
||||
fn fmt(&self, f: &mut core::fmt::Formatter<'_>) -> core::fmt::Result {
|
||||
f.debug_struct("Foo").field("bar", &self.bar).finish()
|
||||
}
|
||||
}
|
||||
"#,
|
||||
)
|
||||
}
|
||||
|
|
|
@ -360,7 +360,7 @@ mod tests {
|
|||
use crate::tests::{check_assist, check_assist_not_applicable, check_assist_target};
|
||||
|
||||
#[test]
|
||||
fn test_if_let_with_match_unapplicable_for_simple_ifs() {
|
||||
fn test_if_let_with_match_inapplicable_for_simple_ifs() {
|
||||
check_assist_not_applicable(
|
||||
replace_if_let_with_match,
|
||||
r#"
|
||||
|
|
|
@ -127,7 +127,7 @@ pub(crate) fn desugar_async_into_impl_future(
|
|||
|
||||
let rparen = function.param_list()?.r_paren_token()?;
|
||||
let return_type = match function.ret_type() {
|
||||
// unable to get a `ty` makes the action unapplicable
|
||||
// unable to get a `ty` makes the action inapplicable
|
||||
Some(ret_type) => Some(ret_type.ty()?),
|
||||
// No type means `-> ()`
|
||||
None => None,
|
||||
|
|
|
@ -362,8 +362,7 @@ pub fn test_some_range(a: int) -> bool {
|
|||
|
||||
expect![[r#"
|
||||
Convert integer base
|
||||
Extract into variable
|
||||
Extract into function
|
||||
Extract into...
|
||||
Replace if let with match
|
||||
"#]]
|
||||
.assert_eq(&expected);
|
||||
|
@ -391,8 +390,7 @@ pub fn test_some_range(a: int) -> bool {
|
|||
|
||||
expect![[r#"
|
||||
Convert integer base
|
||||
Extract into variable
|
||||
Extract into function
|
||||
Extract into...
|
||||
Replace if let with match
|
||||
"#]]
|
||||
.assert_eq(&expected);
|
||||
|
@ -405,8 +403,7 @@ pub fn test_some_range(a: int) -> bool {
|
|||
let expected = labels(&assists);
|
||||
|
||||
expect![[r#"
|
||||
Extract into variable
|
||||
Extract into function
|
||||
Extract into...
|
||||
"#]]
|
||||
.assert_eq(&expected);
|
||||
}
|
||||
|
@ -440,7 +437,7 @@ pub fn test_some_range(a: int) -> bool {
|
|||
|
||||
{
|
||||
let assists = assists(&db, &cfg, AssistResolveStrategy::None, frange.into());
|
||||
assert_eq!(2, assists.len());
|
||||
assert_eq!(4, assists.len());
|
||||
let mut assists = assists.into_iter();
|
||||
|
||||
let extract_into_variable_assist = assists.next().unwrap();
|
||||
|
@ -451,7 +448,11 @@ pub fn test_some_range(a: int) -> bool {
|
|||
RefactorExtract,
|
||||
),
|
||||
label: "Extract into variable",
|
||||
group: None,
|
||||
group: Some(
|
||||
GroupLabel(
|
||||
"Extract into...",
|
||||
),
|
||||
),
|
||||
target: 59..60,
|
||||
source_change: None,
|
||||
command: None,
|
||||
|
@ -459,6 +460,46 @@ pub fn test_some_range(a: int) -> bool {
|
|||
"#]]
|
||||
.assert_debug_eq(&extract_into_variable_assist);
|
||||
|
||||
let extract_into_constant_assist = assists.next().unwrap();
|
||||
expect![[r#"
|
||||
Assist {
|
||||
id: AssistId(
|
||||
"extract_constant",
|
||||
RefactorExtract,
|
||||
),
|
||||
label: "Extract into constant",
|
||||
group: Some(
|
||||
GroupLabel(
|
||||
"Extract into...",
|
||||
),
|
||||
),
|
||||
target: 59..60,
|
||||
source_change: None,
|
||||
command: None,
|
||||
}
|
||||
"#]]
|
||||
.assert_debug_eq(&extract_into_constant_assist);
|
||||
|
||||
let extract_into_static_assist = assists.next().unwrap();
|
||||
expect![[r#"
|
||||
Assist {
|
||||
id: AssistId(
|
||||
"extract_static",
|
||||
RefactorExtract,
|
||||
),
|
||||
label: "Extract into static",
|
||||
group: Some(
|
||||
GroupLabel(
|
||||
"Extract into...",
|
||||
),
|
||||
),
|
||||
target: 59..60,
|
||||
source_change: None,
|
||||
command: None,
|
||||
}
|
||||
"#]]
|
||||
.assert_debug_eq(&extract_into_static_assist);
|
||||
|
||||
let extract_into_function_assist = assists.next().unwrap();
|
||||
expect![[r#"
|
||||
Assist {
|
||||
|
@ -467,7 +508,11 @@ pub fn test_some_range(a: int) -> bool {
|
|||
RefactorExtract,
|
||||
),
|
||||
label: "Extract into function",
|
||||
group: None,
|
||||
group: Some(
|
||||
GroupLabel(
|
||||
"Extract into...",
|
||||
),
|
||||
),
|
||||
target: 59..60,
|
||||
source_change: None,
|
||||
command: None,
|
||||
|
@ -486,7 +531,7 @@ pub fn test_some_range(a: int) -> bool {
|
|||
}),
|
||||
frange.into(),
|
||||
);
|
||||
assert_eq!(2, assists.len());
|
||||
assert_eq!(4, assists.len());
|
||||
let mut assists = assists.into_iter();
|
||||
|
||||
let extract_into_variable_assist = assists.next().unwrap();
|
||||
|
@ -497,7 +542,11 @@ pub fn test_some_range(a: int) -> bool {
|
|||
RefactorExtract,
|
||||
),
|
||||
label: "Extract into variable",
|
||||
group: None,
|
||||
group: Some(
|
||||
GroupLabel(
|
||||
"Extract into...",
|
||||
),
|
||||
),
|
||||
target: 59..60,
|
||||
source_change: None,
|
||||
command: None,
|
||||
|
@ -505,6 +554,46 @@ pub fn test_some_range(a: int) -> bool {
|
|||
"#]]
|
||||
.assert_debug_eq(&extract_into_variable_assist);
|
||||
|
||||
let extract_into_constant_assist = assists.next().unwrap();
|
||||
expect![[r#"
|
||||
Assist {
|
||||
id: AssistId(
|
||||
"extract_constant",
|
||||
RefactorExtract,
|
||||
),
|
||||
label: "Extract into constant",
|
||||
group: Some(
|
||||
GroupLabel(
|
||||
"Extract into...",
|
||||
),
|
||||
),
|
||||
target: 59..60,
|
||||
source_change: None,
|
||||
command: None,
|
||||
}
|
||||
"#]]
|
||||
.assert_debug_eq(&extract_into_constant_assist);
|
||||
|
||||
let extract_into_static_assist = assists.next().unwrap();
|
||||
expect![[r#"
|
||||
Assist {
|
||||
id: AssistId(
|
||||
"extract_static",
|
||||
RefactorExtract,
|
||||
),
|
||||
label: "Extract into static",
|
||||
group: Some(
|
||||
GroupLabel(
|
||||
"Extract into...",
|
||||
),
|
||||
),
|
||||
target: 59..60,
|
||||
source_change: None,
|
||||
command: None,
|
||||
}
|
||||
"#]]
|
||||
.assert_debug_eq(&extract_into_static_assist);
|
||||
|
||||
let extract_into_function_assist = assists.next().unwrap();
|
||||
expect![[r#"
|
||||
Assist {
|
||||
|
@ -513,7 +602,11 @@ pub fn test_some_range(a: int) -> bool {
|
|||
RefactorExtract,
|
||||
),
|
||||
label: "Extract into function",
|
||||
group: None,
|
||||
group: Some(
|
||||
GroupLabel(
|
||||
"Extract into...",
|
||||
),
|
||||
),
|
||||
target: 59..60,
|
||||
source_change: None,
|
||||
command: None,
|
||||
|
@ -532,7 +625,7 @@ pub fn test_some_range(a: int) -> bool {
|
|||
}),
|
||||
frange.into(),
|
||||
);
|
||||
assert_eq!(2, assists.len());
|
||||
assert_eq!(4, assists.len());
|
||||
let mut assists = assists.into_iter();
|
||||
|
||||
let extract_into_variable_assist = assists.next().unwrap();
|
||||
|
@ -543,7 +636,11 @@ pub fn test_some_range(a: int) -> bool {
|
|||
RefactorExtract,
|
||||
),
|
||||
label: "Extract into variable",
|
||||
group: None,
|
||||
group: Some(
|
||||
GroupLabel(
|
||||
"Extract into...",
|
||||
),
|
||||
),
|
||||
target: 59..60,
|
||||
source_change: Some(
|
||||
SourceChange {
|
||||
|
@ -594,6 +691,46 @@ pub fn test_some_range(a: int) -> bool {
|
|||
"#]]
|
||||
.assert_debug_eq(&extract_into_variable_assist);
|
||||
|
||||
let extract_into_constant_assist = assists.next().unwrap();
|
||||
expect![[r#"
|
||||
Assist {
|
||||
id: AssistId(
|
||||
"extract_constant",
|
||||
RefactorExtract,
|
||||
),
|
||||
label: "Extract into constant",
|
||||
group: Some(
|
||||
GroupLabel(
|
||||
"Extract into...",
|
||||
),
|
||||
),
|
||||
target: 59..60,
|
||||
source_change: None,
|
||||
command: None,
|
||||
}
|
||||
"#]]
|
||||
.assert_debug_eq(&extract_into_constant_assist);
|
||||
|
||||
let extract_into_static_assist = assists.next().unwrap();
|
||||
expect![[r#"
|
||||
Assist {
|
||||
id: AssistId(
|
||||
"extract_static",
|
||||
RefactorExtract,
|
||||
),
|
||||
label: "Extract into static",
|
||||
group: Some(
|
||||
GroupLabel(
|
||||
"Extract into...",
|
||||
),
|
||||
),
|
||||
target: 59..60,
|
||||
source_change: None,
|
||||
command: None,
|
||||
}
|
||||
"#]]
|
||||
.assert_debug_eq(&extract_into_static_assist);
|
||||
|
||||
let extract_into_function_assist = assists.next().unwrap();
|
||||
expect![[r#"
|
||||
Assist {
|
||||
|
@ -602,7 +739,11 @@ pub fn test_some_range(a: int) -> bool {
|
|||
RefactorExtract,
|
||||
),
|
||||
label: "Extract into function",
|
||||
group: None,
|
||||
group: Some(
|
||||
GroupLabel(
|
||||
"Extract into...",
|
||||
),
|
||||
),
|
||||
target: 59..60,
|
||||
source_change: None,
|
||||
command: None,
|
||||
|
@ -613,7 +754,7 @@ pub fn test_some_range(a: int) -> bool {
|
|||
|
||||
{
|
||||
let assists = assists(&db, &cfg, AssistResolveStrategy::All, frange.into());
|
||||
assert_eq!(2, assists.len());
|
||||
assert_eq!(4, assists.len());
|
||||
let mut assists = assists.into_iter();
|
||||
|
||||
let extract_into_variable_assist = assists.next().unwrap();
|
||||
|
@ -624,7 +765,11 @@ pub fn test_some_range(a: int) -> bool {
|
|||
RefactorExtract,
|
||||
),
|
||||
label: "Extract into variable",
|
||||
group: None,
|
||||
group: Some(
|
||||
GroupLabel(
|
||||
"Extract into...",
|
||||
),
|
||||
),
|
||||
target: 59..60,
|
||||
source_change: Some(
|
||||
SourceChange {
|
||||
|
@ -675,6 +820,140 @@ pub fn test_some_range(a: int) -> bool {
|
|||
"#]]
|
||||
.assert_debug_eq(&extract_into_variable_assist);
|
||||
|
||||
let extract_into_constant_assist = assists.next().unwrap();
|
||||
expect![[r#"
|
||||
Assist {
|
||||
id: AssistId(
|
||||
"extract_constant",
|
||||
RefactorExtract,
|
||||
),
|
||||
label: "Extract into constant",
|
||||
group: Some(
|
||||
GroupLabel(
|
||||
"Extract into...",
|
||||
),
|
||||
),
|
||||
target: 59..60,
|
||||
source_change: Some(
|
||||
SourceChange {
|
||||
source_file_edits: {
|
||||
FileId(
|
||||
0,
|
||||
): (
|
||||
TextEdit {
|
||||
indels: [
|
||||
Indel {
|
||||
insert: "const",
|
||||
delete: 45..47,
|
||||
},
|
||||
Indel {
|
||||
insert: "VAR_NAME:",
|
||||
delete: 48..60,
|
||||
},
|
||||
Indel {
|
||||
insert: "i32",
|
||||
delete: 61..81,
|
||||
},
|
||||
Indel {
|
||||
insert: "=",
|
||||
delete: 82..86,
|
||||
},
|
||||
Indel {
|
||||
insert: "5;\n if let 2..6 = VAR_NAME {\n true\n } else {\n false\n }",
|
||||
delete: 87..108,
|
||||
},
|
||||
],
|
||||
},
|
||||
Some(
|
||||
SnippetEdit(
|
||||
[
|
||||
(
|
||||
0,
|
||||
51..51,
|
||||
),
|
||||
],
|
||||
),
|
||||
),
|
||||
),
|
||||
},
|
||||
file_system_edits: [],
|
||||
is_snippet: true,
|
||||
},
|
||||
),
|
||||
command: Some(
|
||||
Rename,
|
||||
),
|
||||
}
|
||||
"#]]
|
||||
.assert_debug_eq(&extract_into_constant_assist);
|
||||
|
||||
let extract_into_static_assist = assists.next().unwrap();
|
||||
expect![[r#"
|
||||
Assist {
|
||||
id: AssistId(
|
||||
"extract_static",
|
||||
RefactorExtract,
|
||||
),
|
||||
label: "Extract into static",
|
||||
group: Some(
|
||||
GroupLabel(
|
||||
"Extract into...",
|
||||
),
|
||||
),
|
||||
target: 59..60,
|
||||
source_change: Some(
|
||||
SourceChange {
|
||||
source_file_edits: {
|
||||
FileId(
|
||||
0,
|
||||
): (
|
||||
TextEdit {
|
||||
indels: [
|
||||
Indel {
|
||||
insert: "static",
|
||||
delete: 45..47,
|
||||
},
|
||||
Indel {
|
||||
insert: "VAR_NAME:",
|
||||
delete: 48..60,
|
||||
},
|
||||
Indel {
|
||||
insert: "i32",
|
||||
delete: 61..81,
|
||||
},
|
||||
Indel {
|
||||
insert: "=",
|
||||
delete: 82..86,
|
||||
},
|
||||
Indel {
|
||||
insert: "5;\n if let 2..6 = VAR_NAME {\n true\n } else {\n false\n }",
|
||||
delete: 87..108,
|
||||
},
|
||||
],
|
||||
},
|
||||
Some(
|
||||
SnippetEdit(
|
||||
[
|
||||
(
|
||||
0,
|
||||
52..52,
|
||||
),
|
||||
],
|
||||
),
|
||||
),
|
||||
),
|
||||
},
|
||||
file_system_edits: [],
|
||||
is_snippet: true,
|
||||
},
|
||||
),
|
||||
command: Some(
|
||||
Rename,
|
||||
),
|
||||
}
|
||||
"#]]
|
||||
.assert_debug_eq(&extract_into_static_assist);
|
||||
|
||||
let extract_into_function_assist = assists.next().unwrap();
|
||||
expect![[r#"
|
||||
Assist {
|
||||
|
@ -683,7 +962,11 @@ pub fn test_some_range(a: int) -> bool {
|
|||
RefactorExtract,
|
||||
),
|
||||
label: "Extract into function",
|
||||
group: None,
|
||||
group: Some(
|
||||
GroupLabel(
|
||||
"Extract into...",
|
||||
),
|
||||
),
|
||||
target: 59..60,
|
||||
source_change: Some(
|
||||
SourceChange {
|
||||
|
|
|
@ -932,6 +932,24 @@ enum TheEnum {
|
|||
)
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn doctest_extract_constant() {
|
||||
check_doc_test(
|
||||
"extract_constant",
|
||||
r#####"
|
||||
fn main() {
|
||||
$0(1 + 2)$0 * 4;
|
||||
}
|
||||
"#####,
|
||||
r#####"
|
||||
fn main() {
|
||||
const $0VAR_NAME: i32 = 1 + 2;
|
||||
VAR_NAME * 4;
|
||||
}
|
||||
"#####,
|
||||
)
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn doctest_extract_expressions_from_format_string() {
|
||||
check_doc_test(
|
||||
|
@ -1006,6 +1024,24 @@ fn bar(name: i32) -> i32 {
|
|||
)
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn doctest_extract_static() {
|
||||
check_doc_test(
|
||||
"extract_static",
|
||||
r#####"
|
||||
fn main() {
|
||||
$0(1 + 2)$0 * 4;
|
||||
}
|
||||
"#####,
|
||||
r#####"
|
||||
fn main() {
|
||||
static $0VAR_NAME: i32 = 1 + 2;
|
||||
VAR_NAME * 4;
|
||||
}
|
||||
"#####,
|
||||
)
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn doctest_extract_struct_from_enum_variant() {
|
||||
check_doc_test(
|
||||
|
|
|
@ -3,11 +3,13 @@
|
|||
pub(crate) use gen_trait_fn_body::gen_trait_fn_body;
|
||||
use hir::{
|
||||
db::{ExpandDatabase, HirDatabase},
|
||||
HasAttrs as HirHasAttrs, HirDisplay, InFile, Semantics,
|
||||
HasAttrs as HirHasAttrs, HirDisplay, InFile, ModuleDef, PathResolution, Semantics,
|
||||
};
|
||||
use ide_db::{
|
||||
famous_defs::FamousDefs, path_transform::PathTransform,
|
||||
syntax_helpers::prettify_macro_expansion, RootDatabase,
|
||||
famous_defs::FamousDefs,
|
||||
path_transform::PathTransform,
|
||||
syntax_helpers::{node_ext::preorder_expr, prettify_macro_expansion},
|
||||
RootDatabase,
|
||||
};
|
||||
use stdx::format_to;
|
||||
use syntax::{
|
||||
|
@ -19,7 +21,7 @@ use syntax::{
|
|||
},
|
||||
ted, AstNode, AstToken, Direction, Edition, NodeOrToken, SourceFile,
|
||||
SyntaxKind::*,
|
||||
SyntaxNode, SyntaxToken, TextRange, TextSize, T,
|
||||
SyntaxNode, SyntaxToken, TextRange, TextSize, WalkEvent, T,
|
||||
};
|
||||
|
||||
use crate::assist_context::{AssistContext, SourceChangeBuilder};
|
||||
|
@ -966,3 +968,37 @@ pub(crate) fn tt_from_syntax(node: SyntaxNode) -> Vec<NodeOrToken<ast::TokenTree
|
|||
|
||||
tt_stack.pop().expect("parent token tree was closed before it was completed").1
|
||||
}
|
||||
|
||||
pub fn is_body_const(sema: &Semantics<'_, RootDatabase>, expr: &ast::Expr) -> bool {
|
||||
let mut is_const = true;
|
||||
preorder_expr(expr, &mut |ev| {
|
||||
let expr = match ev {
|
||||
WalkEvent::Enter(_) if !is_const => return true,
|
||||
WalkEvent::Enter(expr) => expr,
|
||||
WalkEvent::Leave(_) => return false,
|
||||
};
|
||||
match expr {
|
||||
ast::Expr::CallExpr(call) => {
|
||||
if let Some(ast::Expr::PathExpr(path_expr)) = call.expr() {
|
||||
if let Some(PathResolution::Def(ModuleDef::Function(func))) =
|
||||
path_expr.path().and_then(|path| sema.resolve_path(&path))
|
||||
{
|
||||
is_const &= func.is_const(sema.db);
|
||||
}
|
||||
}
|
||||
}
|
||||
ast::Expr::MethodCallExpr(call) => {
|
||||
is_const &=
|
||||
sema.resolve_method_call(&call).map(|it| it.is_const(sema.db)).unwrap_or(true)
|
||||
}
|
||||
ast::Expr::ForExpr(_)
|
||||
| ast::Expr::ReturnExpr(_)
|
||||
| ast::Expr::TryExpr(_)
|
||||
| ast::Expr::YieldExpr(_)
|
||||
| ast::Expr::AwaitExpr(_) => is_const = false,
|
||||
_ => (),
|
||||
}
|
||||
!is_const
|
||||
});
|
||||
is_const
|
||||
}
|
||||
|
|
|
@ -11,7 +11,7 @@ pub(super) fn complete_lint(
|
|||
existing_lints: &[ast::Path],
|
||||
lints_completions: &[Lint],
|
||||
) {
|
||||
for &Lint { label, description } in lints_completions {
|
||||
for &Lint { label, description, .. } in lints_completions {
|
||||
let (qual, name) = {
|
||||
// FIXME: change `Lint`'s label to not store a path in it but split the prefix off instead?
|
||||
let mut parts = label.split("::");
|
||||
|
|
|
@ -1,7 +1,7 @@
|
|||
//! Module responsible for analyzing the code surrounding the cursor for completion.
|
||||
use std::iter;
|
||||
|
||||
use hir::{Semantics, Type, TypeInfo, Variant};
|
||||
use hir::{ExpandResult, Semantics, Type, TypeInfo, Variant};
|
||||
use ide_db::{active_parameter::ActiveParameter, RootDatabase};
|
||||
use itertools::Either;
|
||||
use syntax::{
|
||||
|
@ -104,7 +104,10 @@ fn expand(
|
|||
// maybe parent items have attributes, so continue walking the ancestors
|
||||
(None, None) => continue 'ancestors,
|
||||
// successful expansions
|
||||
(Some(actual_expansion), Some((fake_expansion, fake_mapped_token))) => {
|
||||
(
|
||||
Some(ExpandResult { value: actual_expansion, err: _ }),
|
||||
Some((fake_expansion, fake_mapped_token)),
|
||||
) => {
|
||||
let new_offset = fake_mapped_token.text_range().start();
|
||||
if new_offset + relative_offset > actual_expansion.text_range().end() {
|
||||
// offset outside of bounds from the original expansion,
|
||||
|
@ -239,8 +242,8 @@ fn expand(
|
|||
};
|
||||
|
||||
match (
|
||||
sema.expand(&actual_macro_call),
|
||||
sema.speculative_expand(
|
||||
sema.expand_macro_call(&actual_macro_call),
|
||||
sema.speculative_expand_macro_call(
|
||||
&actual_macro_call,
|
||||
&speculative_args,
|
||||
fake_ident_token.clone(),
|
||||
|
|
File diff suppressed because it is too large
Load diff
|
@ -327,3 +327,11 @@ impl<'a> Ranker<'a> {
|
|||
| ((no_tt_parent as usize) << 3)
|
||||
}
|
||||
}
|
||||
|
||||
#[derive(Debug, Copy, Clone, PartialEq, Eq, Hash)]
|
||||
pub enum Severity {
|
||||
Error,
|
||||
Warning,
|
||||
WeakWarning,
|
||||
Allow,
|
||||
}
|
||||
|
|
|
@ -184,7 +184,7 @@ impl<'a> PathTransform<'a> {
|
|||
if let Some(expr) = v.expr() {
|
||||
// FIXME: expressions in curly brackets can cause ambiguity after insertion
|
||||
// (e.g. `N * 2` -> `{1 + 1} * 2`; it's unclear whether `{1 + 1}`
|
||||
// is a standalone statement or a part of another expresson)
|
||||
// is a standalone statement or a part of another expression)
|
||||
// and sometimes require slight modifications; see
|
||||
// https://doc.rust-lang.org/reference/statements.html#expression-statements
|
||||
// (default values in curly brackets can cause the same problem)
|
||||
|
|
|
@ -59,7 +59,7 @@ pub(crate) fn invalid_cast(ctx: &DiagnosticsContext<'_>, d: &hir::InvalidCast) -
|
|||
DiagnosticCode::RustcHardError("E0606"),
|
||||
format_ty!(
|
||||
ctx,
|
||||
"casting `{}` as `{}` is invalid: needs defererence or removal of unneeded borrow",
|
||||
"casting `{}` as `{}` is invalid: needs dereference or removal of unneeded borrow",
|
||||
d.expr_ty,
|
||||
d.cast_ty
|
||||
),
|
||||
|
|
|
@ -586,14 +586,47 @@ fn main() {
|
|||
}
|
||||
|
||||
#[test]
|
||||
fn unsafe_op_in_unsafe_fn_allowed_by_default() {
|
||||
fn unsafe_op_in_unsafe_fn_allowed_by_default_in_edition_2021() {
|
||||
check_diagnostics(
|
||||
r#"
|
||||
//- /lib.rs crate:foo edition:2021
|
||||
unsafe fn foo(p: *mut i32) {
|
||||
*p = 123;
|
||||
}
|
||||
"#,
|
||||
)
|
||||
);
|
||||
check_diagnostics(
|
||||
r#"
|
||||
//- /lib.rs crate:foo edition:2021
|
||||
#![deny(warnings)]
|
||||
unsafe fn foo(p: *mut i32) {
|
||||
*p = 123;
|
||||
}
|
||||
"#,
|
||||
);
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn unsafe_op_in_unsafe_fn_warn_by_default_in_edition_2024() {
|
||||
check_diagnostics(
|
||||
r#"
|
||||
//- /lib.rs crate:foo edition:2024
|
||||
unsafe fn foo(p: *mut i32) {
|
||||
*p = 123;
|
||||
//^^💡 warn: dereference of raw pointer is unsafe and requires an unsafe function or block
|
||||
}
|
||||
"#,
|
||||
);
|
||||
check_diagnostics(
|
||||
r#"
|
||||
//- /lib.rs crate:foo edition:2024
|
||||
#![deny(warnings)]
|
||||
unsafe fn foo(p: *mut i32) {
|
||||
*p = 123;
|
||||
//^^💡 error: dereference of raw pointer is unsafe and requires an unsafe function or block
|
||||
}
|
||||
"#,
|
||||
);
|
||||
}
|
||||
|
||||
#[test]
|
||||
|
|
|
@ -84,12 +84,12 @@ use hir::{db::ExpandDatabase, diagnostics::AnyDiagnostic, Crate, HirFileId, InFi
|
|||
use ide_db::{
|
||||
assists::{Assist, AssistId, AssistKind, AssistResolveStrategy},
|
||||
base_db::SourceDatabase,
|
||||
generated::lints::{LintGroup, CLIPPY_LINT_GROUPS, DEFAULT_LINT_GROUPS},
|
||||
generated::lints::{Lint, LintGroup, CLIPPY_LINT_GROUPS, DEFAULT_LINTS, DEFAULT_LINT_GROUPS},
|
||||
imports::insert_use::InsertUseConfig,
|
||||
label::Label,
|
||||
source_change::SourceChange,
|
||||
syntax_helpers::node_ext::parse_tt_as_comma_sep_paths,
|
||||
EditionedFileId, FileId, FileRange, FxHashMap, FxHashSet, RootDatabase, SnippetCap,
|
||||
EditionedFileId, FileId, FileRange, FxHashMap, FxHashSet, RootDatabase, Severity, SnippetCap,
|
||||
};
|
||||
use itertools::Itertools;
|
||||
use syntax::{
|
||||
|
@ -210,14 +210,6 @@ impl Diagnostic {
|
|||
}
|
||||
}
|
||||
|
||||
#[derive(Debug, Copy, Clone, PartialEq, Eq, Hash)]
|
||||
pub enum Severity {
|
||||
Error,
|
||||
Warning,
|
||||
WeakWarning,
|
||||
Allow,
|
||||
}
|
||||
|
||||
#[derive(Clone, Debug, PartialEq, Eq)]
|
||||
pub enum ExprFillDefaultMode {
|
||||
Todo,
|
||||
|
@ -568,26 +560,35 @@ fn handle_diag_from_macros(
|
|||
|
||||
// `__RA_EVERY_LINT` is a fake lint group to allow every lint in proc macros
|
||||
|
||||
static RUSTC_LINT_GROUPS_DICT: LazyLock<FxHashMap<&str, Vec<&str>>> =
|
||||
LazyLock::new(|| build_group_dict(DEFAULT_LINT_GROUPS, &["warnings", "__RA_EVERY_LINT"], ""));
|
||||
struct BuiltLint {
|
||||
lint: &'static Lint,
|
||||
groups: Vec<&'static str>,
|
||||
}
|
||||
|
||||
static CLIPPY_LINT_GROUPS_DICT: LazyLock<FxHashMap<&str, Vec<&str>>> =
|
||||
LazyLock::new(|| build_group_dict(CLIPPY_LINT_GROUPS, &["__RA_EVERY_LINT"], "clippy::"));
|
||||
static RUSTC_LINTS: LazyLock<FxHashMap<&str, BuiltLint>> =
|
||||
LazyLock::new(|| build_lints_map(DEFAULT_LINTS, DEFAULT_LINT_GROUPS, ""));
|
||||
|
||||
static CLIPPY_LINTS: LazyLock<FxHashMap<&str, BuiltLint>> = LazyLock::new(|| {
|
||||
build_lints_map(ide_db::generated::lints::CLIPPY_LINTS, CLIPPY_LINT_GROUPS, "clippy::")
|
||||
});
|
||||
|
||||
// FIXME: Autogenerate this instead of enumerating by hand.
|
||||
static LINTS_TO_REPORT_IN_EXTERNAL_MACROS: LazyLock<FxHashSet<&str>> =
|
||||
LazyLock::new(|| FxHashSet::from_iter([]));
|
||||
|
||||
fn build_group_dict(
|
||||
fn build_lints_map(
|
||||
lints: &'static [Lint],
|
||||
lint_group: &'static [LintGroup],
|
||||
all_groups: &'static [&'static str],
|
||||
prefix: &'static str,
|
||||
) -> FxHashMap<&'static str, Vec<&'static str>> {
|
||||
let mut map_with_prefixes: FxHashMap<&str, Vec<&str>> = FxHashMap::default();
|
||||
) -> FxHashMap<&'static str, BuiltLint> {
|
||||
let mut map_with_prefixes: FxHashMap<_, _> = lints
|
||||
.iter()
|
||||
.map(|lint| (lint.label, BuiltLint { lint, groups: vec![lint.label, "__RA_EVERY_LINT"] }))
|
||||
.collect();
|
||||
for g in lint_group {
|
||||
let mut add_children = |label: &'static str| {
|
||||
for child in g.children {
|
||||
map_with_prefixes.entry(child).or_default().push(label);
|
||||
map_with_prefixes.get_mut(child).unwrap().groups.push(label);
|
||||
}
|
||||
};
|
||||
add_children(g.lint.label);
|
||||
|
@ -597,18 +598,9 @@ fn build_group_dict(
|
|||
add_children("bad_style");
|
||||
}
|
||||
}
|
||||
for (lint, groups) in map_with_prefixes.iter_mut() {
|
||||
groups.push(lint);
|
||||
groups.extend_from_slice(all_groups);
|
||||
}
|
||||
map_with_prefixes.into_iter().map(|(k, v)| (k.strip_prefix(prefix).unwrap(), v)).collect()
|
||||
}
|
||||
|
||||
/// Thd default severity for lints that are not warn by default.
|
||||
// FIXME: Autogenerate this instead of write manually.
|
||||
static LINTS_DEFAULT_SEVERITY: LazyLock<FxHashMap<&str, Severity>> =
|
||||
LazyLock::new(|| FxHashMap::from_iter([("unsafe_op_in_unsafe_fn", Severity::Allow)]));
|
||||
|
||||
fn handle_lints(
|
||||
sema: &Semantics<'_, RootDatabase>,
|
||||
cache: &mut FxHashMap<HirFileId, FxHashMap<SmolStr, SeverityAttr>>,
|
||||
|
@ -618,10 +610,12 @@ fn handle_lints(
|
|||
) {
|
||||
for (node, diag) in diagnostics {
|
||||
let lint = match diag.code {
|
||||
DiagnosticCode::RustcLint(lint) | DiagnosticCode::Clippy(lint) => lint,
|
||||
DiagnosticCode::RustcLint(lint) => RUSTC_LINTS[lint].lint,
|
||||
DiagnosticCode::Clippy(lint) => CLIPPY_LINTS[lint].lint,
|
||||
_ => panic!("non-lint passed to `handle_lints()`"),
|
||||
};
|
||||
if let Some(&default_severity) = LINTS_DEFAULT_SEVERITY.get(lint) {
|
||||
let default_severity = default_lint_severity(lint, edition);
|
||||
if !(default_severity == Severity::Allow && diag.severity == Severity::WeakWarning) {
|
||||
diag.severity = default_severity;
|
||||
}
|
||||
|
||||
|
@ -639,6 +633,16 @@ fn handle_lints(
|
|||
}
|
||||
}
|
||||
|
||||
fn default_lint_severity(lint: &Lint, edition: Edition) -> Severity {
|
||||
if lint.deny_since.is_some_and(|e| edition >= e) {
|
||||
Severity::Error
|
||||
} else if lint.warn_since.is_some_and(|e| edition >= e) {
|
||||
Severity::Warning
|
||||
} else {
|
||||
lint.default_severity
|
||||
}
|
||||
}
|
||||
|
||||
fn find_outline_mod_lint_severity(
|
||||
sema: &Semantics<'_, RootDatabase>,
|
||||
node: &InFile<SyntaxNode>,
|
||||
|
@ -654,14 +658,14 @@ fn find_outline_mod_lint_severity(
|
|||
let mod_def = sema.to_module_def(&mod_node)?;
|
||||
let module_source_file = sema.module_definition_node(mod_def);
|
||||
let mut result = None;
|
||||
let lint_groups = lint_groups(&diag.code);
|
||||
let lint_groups = lint_groups(&diag.code, edition);
|
||||
lint_attrs(
|
||||
sema,
|
||||
ast::AnyHasAttrs::cast(module_source_file.value).expect("SourceFile always has attrs"),
|
||||
edition,
|
||||
)
|
||||
.for_each(|(lint, severity)| {
|
||||
if lint_groups.contains(&&*lint) {
|
||||
if lint_groups.contains(&lint) {
|
||||
result = Some(severity);
|
||||
}
|
||||
});
|
||||
|
@ -737,9 +741,9 @@ fn fill_lint_attrs(
|
|||
}
|
||||
});
|
||||
|
||||
let all_matching_groups = lint_groups(&diag.code)
|
||||
let all_matching_groups = lint_groups(&diag.code, edition)
|
||||
.iter()
|
||||
.filter_map(|lint_group| cached.get(&**lint_group));
|
||||
.filter_map(|lint_group| cached.get(lint_group));
|
||||
let cached_severity =
|
||||
all_matching_groups.min_by_key(|it| it.depth).map(|it| it.severity);
|
||||
|
||||
|
@ -751,7 +755,7 @@ fn fill_lint_attrs(
|
|||
// Insert this node's descendants' attributes into any outline descendant, but not including this node.
|
||||
// This must come before inserting this node's own attributes to preserve order.
|
||||
collected_lint_attrs.drain().for_each(|(lint, severity)| {
|
||||
if diag_severity.is_none() && lint_groups(&diag.code).contains(&&*lint) {
|
||||
if diag_severity.is_none() && lint_groups(&diag.code, edition).contains(&lint) {
|
||||
diag_severity = Some(severity.severity);
|
||||
}
|
||||
|
||||
|
@ -774,7 +778,7 @@ fn fill_lint_attrs(
|
|||
if let Some(ancestor) = ast::AnyHasAttrs::cast(ancestor) {
|
||||
// Insert this node's attributes into any outline descendant, including this node.
|
||||
lint_attrs(sema, ancestor, edition).for_each(|(lint, severity)| {
|
||||
if diag_severity.is_none() && lint_groups(&diag.code).contains(&&*lint) {
|
||||
if diag_severity.is_none() && lint_groups(&diag.code, edition).contains(&lint) {
|
||||
diag_severity = Some(severity);
|
||||
}
|
||||
|
||||
|
@ -804,7 +808,7 @@ fn fill_lint_attrs(
|
|||
return diag_severity;
|
||||
} else if let Some(ancestor) = ast::AnyHasAttrs::cast(ancestor) {
|
||||
lint_attrs(sema, ancestor, edition).for_each(|(lint, severity)| {
|
||||
if diag_severity.is_none() && lint_groups(&diag.code).contains(&&*lint) {
|
||||
if diag_severity.is_none() && lint_groups(&diag.code, edition).contains(&lint) {
|
||||
diag_severity = Some(severity);
|
||||
}
|
||||
|
||||
|
@ -905,16 +909,37 @@ fn cfg_attr_lint_attrs(
|
|||
}
|
||||
}
|
||||
|
||||
fn lint_groups(lint: &DiagnosticCode) -> &'static [&'static str] {
|
||||
match lint {
|
||||
#[derive(Debug)]
|
||||
struct LintGroups {
|
||||
groups: &'static [&'static str],
|
||||
inside_warnings: bool,
|
||||
}
|
||||
|
||||
impl LintGroups {
|
||||
fn contains(&self, group: &str) -> bool {
|
||||
self.groups.contains(&group) || (self.inside_warnings && group == "warnings")
|
||||
}
|
||||
|
||||
fn iter(&self) -> impl Iterator<Item = &'static str> {
|
||||
self.groups.iter().copied().chain(self.inside_warnings.then_some("warnings"))
|
||||
}
|
||||
}
|
||||
|
||||
fn lint_groups(lint: &DiagnosticCode, edition: Edition) -> LintGroups {
|
||||
let (groups, inside_warnings) = match lint {
|
||||
DiagnosticCode::RustcLint(name) => {
|
||||
RUSTC_LINT_GROUPS_DICT.get(name).map(|it| &**it).unwrap_or_default()
|
||||
let lint = &RUSTC_LINTS[name];
|
||||
let inside_warnings = default_lint_severity(lint.lint, edition) == Severity::Warning;
|
||||
(&lint.groups, inside_warnings)
|
||||
}
|
||||
DiagnosticCode::Clippy(name) => {
|
||||
CLIPPY_LINT_GROUPS_DICT.get(name).map(|it| &**it).unwrap_or_default()
|
||||
let lint = &CLIPPY_LINTS[name];
|
||||
let inside_warnings = default_lint_severity(lint.lint, edition) == Severity::Warning;
|
||||
(&lint.groups, inside_warnings)
|
||||
}
|
||||
_ => &[],
|
||||
}
|
||||
_ => panic!("non-lint passed to `handle_lints()`"),
|
||||
};
|
||||
LintGroups { groups, inside_warnings }
|
||||
}
|
||||
|
||||
fn fix(id: &'static str, label: &str, source_change: SourceChange, target: TextRange) -> Assist {
|
||||
|
|
|
@ -286,7 +286,7 @@ impl<'db> MatchFinder<'db> {
|
|||
});
|
||||
}
|
||||
} else if let Some(macro_call) = ast::MacroCall::cast(node.clone()) {
|
||||
if let Some(expanded) = self.sema.expand(¯o_call) {
|
||||
if let Some(expanded) = self.sema.expand_macro_call(¯o_call) {
|
||||
if let Some(tt) = macro_call.token_tree() {
|
||||
self.output_debug_for_nodes_at_range(
|
||||
&expanded,
|
||||
|
|
|
@ -189,7 +189,7 @@ impl MatchFinder<'_> {
|
|||
// If we've got a macro call, we already tried matching it pre-expansion, which is the only
|
||||
// way to match the whole macro, now try expanding it and matching the expansion.
|
||||
if let Some(macro_call) = ast::MacroCall::cast(code.clone()) {
|
||||
if let Some(expanded) = self.sema.expand(¯o_call) {
|
||||
if let Some(expanded) = self.sema.expand_macro_call(¯o_call) {
|
||||
if let Some(tt) = macro_call.token_tree() {
|
||||
// When matching within a macro expansion, we only want to allow matches of
|
||||
// nodes that originated entirely from within the token tree of the macro call.
|
||||
|
|
|
@ -1,10 +1,11 @@
|
|||
use hir::db::ExpandDatabase;
|
||||
use hir::{InFile, MacroFileIdExt, Semantics};
|
||||
use hir::{ExpandResult, InFile, MacroFileIdExt, Semantics};
|
||||
use ide_db::base_db::CrateId;
|
||||
use ide_db::{
|
||||
helpers::pick_best_token, syntax_helpers::prettify_macro_expansion, FileId, RootDatabase,
|
||||
};
|
||||
use span::{Edition, SpanMap, SyntaxContextId, TextRange, TextSize};
|
||||
use stdx::format_to;
|
||||
use syntax::{ast, ted, AstNode, NodeOrToken, SyntaxKind, SyntaxNode, T};
|
||||
|
||||
use crate::FilePosition;
|
||||
|
@ -63,10 +64,10 @@ pub(crate) fn expand_macro(db: &RootDatabase, position: FilePosition) -> Option<
|
|||
.take_while(|it| it != &token)
|
||||
.filter(|it| it.kind() == T![,])
|
||||
.count();
|
||||
let expansion = expansions.get(idx)?.clone();
|
||||
let ExpandResult { err, value: expansion } = expansions.get(idx)?.clone();
|
||||
let expansion_file_id = sema.hir_file_for(&expansion).macro_file()?;
|
||||
let expansion_span_map = db.expansion_span_map(expansion_file_id);
|
||||
let expansion = format(
|
||||
let mut expansion = format(
|
||||
db,
|
||||
SyntaxKind::MACRO_ITEMS,
|
||||
position.file_id,
|
||||
|
@ -74,6 +75,12 @@ pub(crate) fn expand_macro(db: &RootDatabase, position: FilePosition) -> Option<
|
|||
&expansion_span_map,
|
||||
krate,
|
||||
);
|
||||
if let Some(err) = err {
|
||||
expansion.insert_str(
|
||||
0,
|
||||
&format!("Expansion had errors: {}\n\n", err.render_to_string(sema.db)),
|
||||
);
|
||||
}
|
||||
Some(ExpandedMacro { name, expansion })
|
||||
});
|
||||
|
||||
|
@ -83,6 +90,7 @@ pub(crate) fn expand_macro(db: &RootDatabase, position: FilePosition) -> Option<
|
|||
|
||||
let mut anc = tok.parent_ancestors();
|
||||
let mut span_map = SpanMap::empty();
|
||||
let mut error = String::new();
|
||||
let (name, expanded, kind) = loop {
|
||||
let node = anc.next()?;
|
||||
|
||||
|
@ -97,7 +105,7 @@ pub(crate) fn expand_macro(db: &RootDatabase, position: FilePosition) -> Option<
|
|||
.unwrap_or(Edition::CURRENT),
|
||||
)
|
||||
.to_string(),
|
||||
expand_macro_recur(&sema, &item, &mut span_map, TextSize::new(0))?,
|
||||
expand_macro_recur(&sema, &item, &mut error, &mut span_map, TextSize::new(0))?,
|
||||
SyntaxKind::MACRO_ITEMS,
|
||||
);
|
||||
}
|
||||
|
@ -112,6 +120,7 @@ pub(crate) fn expand_macro(db: &RootDatabase, position: FilePosition) -> Option<
|
|||
expand_macro_recur(
|
||||
&sema,
|
||||
&ast::Item::MacroCall(mac),
|
||||
&mut error,
|
||||
&mut span_map,
|
||||
TextSize::new(0),
|
||||
)?,
|
||||
|
@ -123,24 +132,31 @@ pub(crate) fn expand_macro(db: &RootDatabase, position: FilePosition) -> Option<
|
|||
// FIXME:
|
||||
// macro expansion may lose all white space information
|
||||
// But we hope someday we can use ra_fmt for that
|
||||
let expansion = format(db, kind, position.file_id, expanded, &span_map, krate);
|
||||
let mut expansion = format(db, kind, position.file_id, expanded, &span_map, krate);
|
||||
|
||||
if !error.is_empty() {
|
||||
expansion.insert_str(0, &format!("Expansion had errors:{error}\n\n"));
|
||||
}
|
||||
Some(ExpandedMacro { name, expansion })
|
||||
}
|
||||
|
||||
fn expand_macro_recur(
|
||||
sema: &Semantics<'_, RootDatabase>,
|
||||
macro_call: &ast::Item,
|
||||
error: &mut String,
|
||||
result_span_map: &mut SpanMap<SyntaxContextId>,
|
||||
offset_in_original_node: TextSize,
|
||||
) -> Option<SyntaxNode> {
|
||||
let expanded = match macro_call {
|
||||
item @ ast::Item::MacroCall(macro_call) => sema
|
||||
.expand_attr_macro(item)
|
||||
.or_else(|| sema.expand_allowed_builtins(macro_call))?
|
||||
.clone_for_update(),
|
||||
item => sema.expand_attr_macro(item)?.clone_for_update(),
|
||||
let ExpandResult { value: expanded, err } = match macro_call {
|
||||
item @ ast::Item::MacroCall(macro_call) => {
|
||||
sema.expand_attr_macro(item).or_else(|| sema.expand_allowed_builtins(macro_call))?
|
||||
}
|
||||
item => sema.expand_attr_macro(item)?,
|
||||
};
|
||||
let expanded = expanded.clone_for_update();
|
||||
if let Some(err) = err {
|
||||
format_to!(error, "\n{}", err.render_to_string(sema.db));
|
||||
}
|
||||
let file_id =
|
||||
sema.hir_file_for(&expanded).macro_file().expect("expansion must produce a macro file");
|
||||
let expansion_span_map = sema.db.expansion_span_map(file_id);
|
||||
|
@ -149,12 +165,13 @@ fn expand_macro_recur(
|
|||
expanded.text_range().len(),
|
||||
&expansion_span_map,
|
||||
);
|
||||
Some(expand(sema, expanded, result_span_map, u32::from(offset_in_original_node) as i32))
|
||||
Some(expand(sema, expanded, error, result_span_map, u32::from(offset_in_original_node) as i32))
|
||||
}
|
||||
|
||||
fn expand(
|
||||
sema: &Semantics<'_, RootDatabase>,
|
||||
expanded: SyntaxNode,
|
||||
error: &mut String,
|
||||
result_span_map: &mut SpanMap<SyntaxContextId>,
|
||||
mut offset_in_original_node: i32,
|
||||
) -> SyntaxNode {
|
||||
|
@ -165,6 +182,7 @@ fn expand(
|
|||
if let Some(new_node) = expand_macro_recur(
|
||||
sema,
|
||||
&child,
|
||||
error,
|
||||
result_span_map,
|
||||
TextSize::new(
|
||||
(offset_in_original_node + (u32::from(child.syntax().text_range().start()) as i32))
|
||||
|
@ -495,6 +513,9 @@ fn main() {
|
|||
"#,
|
||||
expect![[r#"
|
||||
foo!
|
||||
Expansion had errors:
|
||||
expected ident: `BAD`
|
||||
|
||||
"#]],
|
||||
);
|
||||
}
|
||||
|
|
|
@ -608,7 +608,7 @@ impl<'a> WalkExpandedExprCtx<'a> {
|
|||
|
||||
if let ast::Expr::MacroExpr(expr) = expr {
|
||||
if let Some(expanded) =
|
||||
expr.macro_call().and_then(|call| self.sema.expand(&call))
|
||||
expr.macro_call().and_then(|call| self.sema.expand_macro_call(&call))
|
||||
{
|
||||
match_ast! {
|
||||
match expanded {
|
||||
|
|
|
@ -1,5 +1,5 @@
|
|||
//! Logic for rendering the different hover messages
|
||||
use std::{mem, ops::Not};
|
||||
use std::{env, mem, ops::Not};
|
||||
|
||||
use either::Either;
|
||||
use hir::{
|
||||
|
@ -28,6 +28,7 @@ use syntax::{algo, ast, match_ast, AstNode, AstToken, Direction, SyntaxToken, T}
|
|||
use crate::{
|
||||
doc_links::{remove_links, rewrite_links},
|
||||
hover::{notable_traits, walk_and_push_ty},
|
||||
interpret::render_const_eval_error,
|
||||
HoverAction, HoverConfig, HoverResult, Markup, MemoryLayoutHoverConfig,
|
||||
MemoryLayoutHoverRenderKind,
|
||||
};
|
||||
|
@ -464,41 +465,77 @@ pub(super) fn definition(
|
|||
Ok(it) => {
|
||||
Some(if it >= 10 { format!("{it} ({it:#X})") } else { format!("{it}") })
|
||||
}
|
||||
Err(_) => it.value(db).map(|it| format!("{it:?}")),
|
||||
Err(err) => {
|
||||
let res = it.value(db).map(|it| format!("{it:?}"));
|
||||
if env::var_os("RA_DEV").is_some() {
|
||||
let res = res.as_deref().unwrap_or("");
|
||||
Some(format!("{res} ({})", render_const_eval_error(db, err, edition)))
|
||||
} else {
|
||||
res
|
||||
}
|
||||
}
|
||||
}
|
||||
} else {
|
||||
None
|
||||
}
|
||||
}
|
||||
Definition::Const(it) => {
|
||||
let body = it.render_eval(db, edition);
|
||||
match body {
|
||||
Ok(it) => Some(it),
|
||||
Err(_) => {
|
||||
let body = it.eval(db);
|
||||
Some(match body {
|
||||
Ok(it) => match it.render_debug(db) {
|
||||
Ok(it) => it,
|
||||
Err(err) => {
|
||||
let it = it.render(db, edition);
|
||||
if env::var_os("RA_DEV").is_some() {
|
||||
format!("{it}\n{}", render_const_eval_error(db, err.into(), edition))
|
||||
} else {
|
||||
it
|
||||
}
|
||||
}
|
||||
},
|
||||
Err(err) => {
|
||||
let source = it.source(db)?;
|
||||
let mut body = source.value.body()?.syntax().clone();
|
||||
if let Some(macro_file) = source.file_id.macro_file() {
|
||||
let span_map = db.expansion_span_map(macro_file);
|
||||
body = prettify_macro_expansion(db, body, &span_map, it.krate(db).into());
|
||||
}
|
||||
Some(body.to_string())
|
||||
if env::var_os("RA_DEV").is_some() {
|
||||
format!("{body}\n{}", render_const_eval_error(db, err, edition))
|
||||
} else {
|
||||
body.to_string()
|
||||
}
|
||||
}
|
||||
}
|
||||
})
|
||||
}
|
||||
Definition::Static(it) => {
|
||||
let body = it.render_eval(db, edition);
|
||||
match body {
|
||||
Ok(it) => Some(it),
|
||||
Err(_) => {
|
||||
let body = it.eval(db);
|
||||
Some(match body {
|
||||
Ok(it) => match it.render_debug(db) {
|
||||
Ok(it) => it,
|
||||
Err(err) => {
|
||||
let it = it.render(db, edition);
|
||||
if env::var_os("RA_DEV").is_some() {
|
||||
format!("{it}\n{}", render_const_eval_error(db, err.into(), edition))
|
||||
} else {
|
||||
it
|
||||
}
|
||||
}
|
||||
},
|
||||
Err(err) => {
|
||||
let source = it.source(db)?;
|
||||
let mut body = source.value.body()?.syntax().clone();
|
||||
if let Some(macro_file) = source.file_id.macro_file() {
|
||||
let span_map = db.expansion_span_map(macro_file);
|
||||
body = prettify_macro_expansion(db, body, &span_map, it.krate(db).into());
|
||||
}
|
||||
Some(body.to_string())
|
||||
if env::var_os("RA_DEV").is_some() {
|
||||
format!("{body}\n{}", render_const_eval_error(db, err, edition))
|
||||
} else {
|
||||
body.to_string()
|
||||
}
|
||||
}
|
||||
}
|
||||
})
|
||||
}
|
||||
_ => None,
|
||||
};
|
||||
|
|
|
@ -6413,7 +6413,7 @@ fn hover_feature() {
|
|||
by the codegen backend, but not the MIR inliner.
|
||||
|
||||
```rust
|
||||
#![feature(rustc_attrs)]
|
||||
#![feature(intrinsics)]
|
||||
#![allow(internal_features)]
|
||||
|
||||
#[rustc_intrinsic]
|
||||
|
@ -6423,7 +6423,7 @@ fn hover_feature() {
|
|||
Since these are just regular functions, it is perfectly ok to create the intrinsic twice:
|
||||
|
||||
```rust
|
||||
#![feature(rustc_attrs)]
|
||||
#![feature(intrinsics)]
|
||||
#![allow(internal_features)]
|
||||
|
||||
#[rustc_intrinsic]
|
||||
|
@ -9465,4 +9465,39 @@ fn main() {
|
|||
size = 0, align = 1
|
||||
"#]],
|
||||
);
|
||||
|
||||
check(
|
||||
r#"
|
||||
//- minicore: eq
|
||||
pub struct RandomState;
|
||||
pub struct HashMap<K, V, S = RandomState>(K, V, S);
|
||||
|
||||
impl<K, V> HashMap<K, V, RandomState> {
|
||||
pub fn new() -> HashMap<K, V, RandomState> {
|
||||
loop {}
|
||||
}
|
||||
}
|
||||
|
||||
impl<K, V, S> PartialEq for HashMap<K, V, S> {
|
||||
fn eq(&self, other: &HashMap<K, V, S>) -> bool {
|
||||
false
|
||||
}
|
||||
}
|
||||
|
||||
fn main() {
|
||||
let s$0 = HashMap::<_, u64>::ne;
|
||||
}
|
||||
"#,
|
||||
expect![[r#"
|
||||
*s*
|
||||
|
||||
```rust
|
||||
let s: fn ne<HashMap<{unknown}, u64>>(&HashMap<{unknown}, u64>, &HashMap<{unknown}, u64>) -> bool
|
||||
```
|
||||
|
||||
---
|
||||
|
||||
size = 0, align = 1
|
||||
"#]],
|
||||
);
|
||||
}
|
||||
|
|
|
@ -1,5 +1,6 @@
|
|||
use hir::{DefWithBody, Semantics};
|
||||
use hir::{ConstEvalError, DefWithBody, Semantics};
|
||||
use ide_db::{base_db::SourceRootDatabase, FilePosition, LineIndexDatabase, RootDatabase};
|
||||
use span::Edition;
|
||||
use std::time::{Duration, Instant};
|
||||
use stdx::format_to;
|
||||
use syntax::{algo::ancestors_at_offset, ast, AstNode, TextRange};
|
||||
|
@ -47,18 +48,36 @@ fn find_and_interpret(db: &RootDatabase, position: FilePosition) -> Option<(Dura
|
|||
None => format!("file://{path} range {text_range:?}"),
|
||||
}
|
||||
};
|
||||
let edition = def.module(db).krate().edition(db);
|
||||
let start_time = Instant::now();
|
||||
let res = match def {
|
||||
DefWithBody::Function(it) => it.eval(db, span_formatter),
|
||||
DefWithBody::Static(it) => it.eval(db),
|
||||
DefWithBody::Const(it) => it.eval(db),
|
||||
DefWithBody::Static(it) => it.eval(db).map(|it| it.render(db, edition)),
|
||||
DefWithBody::Const(it) => it.eval(db).map(|it| it.render(db, edition)),
|
||||
_ => unreachable!(),
|
||||
};
|
||||
let res = res.unwrap_or_else(|e| {
|
||||
let mut r = String::new();
|
||||
_ = e.pretty_print(&mut r, db, span_formatter, def.module(db).krate().edition(db));
|
||||
r
|
||||
});
|
||||
let res = res.unwrap_or_else(|e| render_const_eval_error(db, e, edition));
|
||||
let duration = Instant::now() - start_time;
|
||||
Some((duration, res))
|
||||
}
|
||||
|
||||
pub(crate) fn render_const_eval_error(
|
||||
db: &RootDatabase,
|
||||
e: ConstEvalError,
|
||||
edition: Edition,
|
||||
) -> String {
|
||||
let span_formatter = |file_id, text_range: TextRange| {
|
||||
let path = &db
|
||||
.source_root(db.file_source_root(file_id))
|
||||
.path_for_file(&file_id)
|
||||
.map(|x| x.to_string());
|
||||
let path = path.as_deref().unwrap_or("<unknown file>");
|
||||
match db.line_index(file_id).try_line_col(text_range.start()) {
|
||||
Some(line_col) => format!("file://{path}:{}:{}", line_col.line + 1, line_col.col),
|
||||
None => format!("file://{path} range {text_range:?}"),
|
||||
}
|
||||
};
|
||||
let mut r = String::new();
|
||||
_ = e.pretty_print(&mut r, db, span_formatter, edition);
|
||||
r
|
||||
}
|
||||
|
|
|
@ -132,11 +132,9 @@ pub use ide_db::{
|
|||
search::{ReferenceCategory, SearchScope},
|
||||
source_change::{FileSystemEdit, SnippetEdit, SourceChange},
|
||||
symbol_index::Query,
|
||||
FileId, FilePosition, FileRange, RootDatabase, SymbolKind,
|
||||
};
|
||||
pub use ide_diagnostics::{
|
||||
Diagnostic, DiagnosticCode, DiagnosticsConfig, ExprFillDefaultMode, Severity,
|
||||
FileId, FilePosition, FileRange, RootDatabase, Severity, SymbolKind,
|
||||
};
|
||||
pub use ide_diagnostics::{Diagnostic, DiagnosticCode, DiagnosticsConfig, ExprFillDefaultMode};
|
||||
pub use ide_ssr::SsrError;
|
||||
pub use span::Edition;
|
||||
pub use syntax::{TextRange, TextSize};
|
||||
|
@ -301,7 +299,7 @@ impl Analysis {
|
|||
|
||||
/// Gets the syntax tree of the file.
|
||||
pub fn parse(&self, file_id: FileId) -> Cancellable<SourceFile> {
|
||||
// FIXME editiojn
|
||||
// FIXME edition
|
||||
self.with_db(|db| db.parse(EditionedFileId::current_edition(file_id)).tree())
|
||||
}
|
||||
|
||||
|
@ -542,7 +540,7 @@ impl Analysis {
|
|||
/// Returns URL(s) for the documentation of the symbol under the cursor.
|
||||
/// # Arguments
|
||||
/// * `position` - Position in the file.
|
||||
/// * `target_dir` - Directory where the build output is storeda.
|
||||
/// * `target_dir` - Directory where the build output is stored.
|
||||
pub fn external_docs(
|
||||
&self,
|
||||
position: FilePosition,
|
||||
|
|
|
@ -307,6 +307,7 @@ define_symbols! {
|
|||
module_path,
|
||||
mul_assign,
|
||||
mul,
|
||||
naked_asm,
|
||||
ne,
|
||||
neg,
|
||||
Neg,
|
||||
|
|
|
@ -15,9 +15,7 @@ use ide_db::{
|
|||
};
|
||||
use itertools::Itertools;
|
||||
use proc_macro_api::{MacroDylib, ProcMacroServer};
|
||||
use project_model::{
|
||||
CargoConfig, PackageRoot, ProjectManifest, ProjectWorkspace, ProjectWorkspaceKind,
|
||||
};
|
||||
use project_model::{CargoConfig, PackageRoot, ProjectManifest, ProjectWorkspace};
|
||||
use span::Span;
|
||||
use vfs::{
|
||||
file_set::FileSetConfig,
|
||||
|
@ -244,6 +242,9 @@ impl ProjectFolders {
|
|||
}
|
||||
}
|
||||
|
||||
if dirs.include.is_empty() {
|
||||
continue;
|
||||
}
|
||||
vfs::loader::Entry::Directories(dirs)
|
||||
};
|
||||
|
||||
|
@ -258,43 +259,6 @@ impl ProjectFolders {
|
|||
fsc.add_file_set(file_set_roots)
|
||||
}
|
||||
|
||||
// register the workspace manifest as well, note that this currently causes duplicates for
|
||||
// non-virtual cargo workspaces! We ought to fix that
|
||||
for ws in workspaces.iter() {
|
||||
let mut file_set_roots: Vec<VfsPath> = vec![];
|
||||
let mut entries = vec![];
|
||||
|
||||
if let Some(manifest) = ws.manifest().map(|it| it.to_path_buf()) {
|
||||
file_set_roots.push(VfsPath::from(manifest.to_owned()));
|
||||
entries.push(manifest.to_owned());
|
||||
}
|
||||
|
||||
for buildfile in ws.buildfiles() {
|
||||
file_set_roots.push(VfsPath::from(buildfile.to_owned()));
|
||||
entries.push(buildfile.to_owned());
|
||||
}
|
||||
|
||||
// In case of detached files we do **not** look for a rust-analyzer.toml.
|
||||
if !matches!(ws.kind, ProjectWorkspaceKind::DetachedFile { .. }) {
|
||||
let ws_root = ws.workspace_root();
|
||||
let ratoml_path = {
|
||||
let mut p = ws_root.to_path_buf();
|
||||
p.push("rust-analyzer.toml");
|
||||
p
|
||||
};
|
||||
file_set_roots.push(VfsPath::from(ratoml_path.to_owned()));
|
||||
entries.push(ratoml_path.to_owned());
|
||||
}
|
||||
|
||||
if !file_set_roots.is_empty() {
|
||||
let entry = vfs::loader::Entry::Files(entries);
|
||||
res.watch.push(res.load.len());
|
||||
res.load.push(entry);
|
||||
local_filesets.push(fsc.len() as u64);
|
||||
fsc.add_file_set(file_set_roots)
|
||||
}
|
||||
}
|
||||
|
||||
if let Some(user_config_path) = user_config_dir_path {
|
||||
let ratoml_path = {
|
||||
let mut p = user_config_path.to_path_buf();
|
||||
|
@ -303,7 +267,7 @@ impl ProjectFolders {
|
|||
};
|
||||
|
||||
let file_set_roots = vec![VfsPath::from(ratoml_path.to_owned())];
|
||||
let entry = vfs::loader::Entry::Files(vec![ratoml_path.to_owned()]);
|
||||
let entry = vfs::loader::Entry::Files(vec![ratoml_path]);
|
||||
|
||||
res.watch.push(res.load.len());
|
||||
res.load.push(entry);
|
||||
|
|
|
@ -18,6 +18,8 @@ ra-ap-rustc_lexer.workspace = true
|
|||
limit.workspace = true
|
||||
tracing = { workspace = true, optional = true }
|
||||
|
||||
edition.workspace = true
|
||||
|
||||
[dev-dependencies]
|
||||
expect-test = "1.4.0"
|
||||
|
||||
|
|
|
@ -25,7 +25,6 @@ extern crate ra_ap_rustc_lexer as rustc_lexer;
|
|||
#[cfg(feature = "in-rust-tree")]
|
||||
extern crate rustc_lexer;
|
||||
|
||||
mod edition;
|
||||
mod event;
|
||||
mod grammar;
|
||||
mod input;
|
||||
|
@ -41,8 +40,9 @@ mod tests;
|
|||
|
||||
pub(crate) use token_set::TokenSet;
|
||||
|
||||
pub use edition::Edition;
|
||||
|
||||
pub use crate::{
|
||||
edition::Edition,
|
||||
input::Input,
|
||||
lexed_str::LexedStr,
|
||||
output::{Output, Step},
|
||||
|
|
|
@ -42,7 +42,7 @@ use rustc_hash::FxHashMap;
|
|||
use serde::{Deserialize, Serialize};
|
||||
use span::{EditionedFileId, ErasedFileAstId, Span, SpanAnchor, SyntaxContextId, TextRange};
|
||||
|
||||
use crate::msg::{ENCODE_CLOSE_SPAN_VERSION, EXTENDED_LEAF_DATA};
|
||||
use crate::msg::EXTENDED_LEAF_DATA;
|
||||
|
||||
pub type SpanDataIndexMap =
|
||||
indexmap::IndexSet<Span, std::hash::BuildHasherDefault<rustc_hash::FxHasher>>;
|
||||
|
@ -145,11 +145,7 @@ impl FlatTree {
|
|||
w.write(subtree);
|
||||
|
||||
FlatTree {
|
||||
subtree: if version >= ENCODE_CLOSE_SPAN_VERSION {
|
||||
write_vec(w.subtree, SubtreeRepr::write_with_close_span)
|
||||
} else {
|
||||
write_vec(w.subtree, SubtreeRepr::write)
|
||||
},
|
||||
subtree: write_vec(w.subtree, SubtreeRepr::write),
|
||||
literal: if version >= EXTENDED_LEAF_DATA {
|
||||
write_vec(w.literal, LiteralRepr::write_with_kind)
|
||||
} else {
|
||||
|
@ -183,11 +179,7 @@ impl FlatTree {
|
|||
w.write(subtree);
|
||||
|
||||
FlatTree {
|
||||
subtree: if version >= ENCODE_CLOSE_SPAN_VERSION {
|
||||
write_vec(w.subtree, SubtreeRepr::write_with_close_span)
|
||||
} else {
|
||||
write_vec(w.subtree, SubtreeRepr::write)
|
||||
},
|
||||
subtree: write_vec(w.subtree, SubtreeRepr::write),
|
||||
literal: if version >= EXTENDED_LEAF_DATA {
|
||||
write_vec(w.literal, LiteralRepr::write_with_kind)
|
||||
} else {
|
||||
|
@ -210,11 +202,7 @@ impl FlatTree {
|
|||
span_data_table: &SpanDataIndexMap,
|
||||
) -> tt::Subtree<Span> {
|
||||
Reader {
|
||||
subtree: if version >= ENCODE_CLOSE_SPAN_VERSION {
|
||||
read_vec(self.subtree, SubtreeRepr::read_with_close_span)
|
||||
} else {
|
||||
read_vec(self.subtree, SubtreeRepr::read)
|
||||
},
|
||||
subtree: read_vec(self.subtree, SubtreeRepr::read),
|
||||
literal: if version >= EXTENDED_LEAF_DATA {
|
||||
read_vec(self.literal, LiteralRepr::read_with_kind)
|
||||
} else {
|
||||
|
@ -236,11 +224,7 @@ impl FlatTree {
|
|||
|
||||
pub fn to_subtree_unresolved(self, version: u32) -> tt::Subtree<TokenId> {
|
||||
Reader {
|
||||
subtree: if version >= ENCODE_CLOSE_SPAN_VERSION {
|
||||
read_vec(self.subtree, SubtreeRepr::read_with_close_span)
|
||||
} else {
|
||||
read_vec(self.subtree, SubtreeRepr::read)
|
||||
},
|
||||
subtree: read_vec(self.subtree, SubtreeRepr::read),
|
||||
literal: if version >= EXTENDED_LEAF_DATA {
|
||||
read_vec(self.literal, LiteralRepr::read_with_kind)
|
||||
} else {
|
||||
|
@ -273,26 +257,7 @@ fn write_vec<T, F: Fn(T) -> [u32; N], const N: usize>(xs: Vec<T>, f: F) -> Vec<u
|
|||
}
|
||||
|
||||
impl SubtreeRepr {
|
||||
fn write(self) -> [u32; 4] {
|
||||
let kind = match self.kind {
|
||||
tt::DelimiterKind::Invisible => 0,
|
||||
tt::DelimiterKind::Parenthesis => 1,
|
||||
tt::DelimiterKind::Brace => 2,
|
||||
tt::DelimiterKind::Bracket => 3,
|
||||
};
|
||||
[self.open.0, kind, self.tt[0], self.tt[1]]
|
||||
}
|
||||
fn read([open, kind, lo, len]: [u32; 4]) -> SubtreeRepr {
|
||||
let kind = match kind {
|
||||
0 => tt::DelimiterKind::Invisible,
|
||||
1 => tt::DelimiterKind::Parenthesis,
|
||||
2 => tt::DelimiterKind::Brace,
|
||||
3 => tt::DelimiterKind::Bracket,
|
||||
other => panic!("bad kind {other}"),
|
||||
};
|
||||
SubtreeRepr { open: TokenId(open), close: TokenId(!0), kind, tt: [lo, len] }
|
||||
}
|
||||
fn write_with_close_span(self) -> [u32; 5] {
|
||||
fn write(self) -> [u32; 5] {
|
||||
let kind = match self.kind {
|
||||
tt::DelimiterKind::Invisible => 0,
|
||||
tt::DelimiterKind::Parenthesis => 1,
|
||||
|
@ -301,7 +266,7 @@ impl SubtreeRepr {
|
|||
};
|
||||
[self.open.0, self.close.0, kind, self.tt[0], self.tt[1]]
|
||||
}
|
||||
fn read_with_close_span([open, close, kind, lo, len]: [u32; 5]) -> SubtreeRepr {
|
||||
fn read([open, close, kind, lo, len]: [u32; 5]) -> SubtreeRepr {
|
||||
let kind = match kind {
|
||||
0 => tt::DelimiterKind::Invisible,
|
||||
1 => tt::DelimiterKind::Parenthesis,
|
||||
|
|
|
@ -56,8 +56,25 @@ impl ProcMacroProcessSrv {
|
|||
match srv.version_check() {
|
||||
Ok(v) if v > CURRENT_API_VERSION => Err(io::Error::new(
|
||||
io::ErrorKind::Other,
|
||||
format!( "The version of the proc-macro server ({v}) in your Rust toolchain is newer than the version supported by your rust-analyzer ({CURRENT_API_VERSION}).
|
||||
This will prevent proc-macro expansion from working. Please consider updating your rust-analyzer to ensure compatibility with your current toolchain."
|
||||
format!(
|
||||
"The version of the proc-macro server ({v}) in your Rust toolchain \
|
||||
is newer than the version supported by your rust-analyzer ({CURRENT_API_VERSION}).
|
||||
\
|
||||
This will prevent proc-macro expansion from working. \
|
||||
Please consider updating your rust-analyzer to ensure compatibility with your \
|
||||
current toolchain."
|
||||
),
|
||||
)),
|
||||
Ok(v) if v < RUST_ANALYZER_SPAN_SUPPORT => Err(io::Error::new(
|
||||
io::ErrorKind::Other,
|
||||
format!(
|
||||
"The version of the proc-macro server ({v}) in your Rust toolchain \
|
||||
is too old and no longer supported by your rust-analyzer which requires\
|
||||
version {RUST_ANALYZER_SPAN_SUPPORT} or higher.
|
||||
\
|
||||
This will prevent proc-macro expansion from working. \
|
||||
Please consider updating your toolchain or downgrading your rust-analyzer \
|
||||
to ensure compatibility with your current toolchain."
|
||||
),
|
||||
)),
|
||||
Ok(v) => {
|
||||
|
@ -72,10 +89,10 @@ impl ProcMacroProcessSrv {
|
|||
tracing::info!("Proc-macro server span mode: {:?}", srv.mode);
|
||||
Ok(srv)
|
||||
}
|
||||
Err(e) => {
|
||||
tracing::info!(%e, "proc-macro version check failed, restarting and assuming version 0");
|
||||
create_srv(false)
|
||||
}
|
||||
Err(e) => Err(io::Error::new(
|
||||
io::ErrorKind::Other,
|
||||
format!("Failed to fetch proc-macro server version: {e}"),
|
||||
)),
|
||||
}
|
||||
}
|
||||
|
||||
|
|
|
@ -6,21 +6,16 @@
|
|||
#[cfg(feature = "in-rust-tree")]
|
||||
extern crate rustc_driver as _;
|
||||
|
||||
use proc_macro_api::json::{read_json, write_json};
|
||||
|
||||
use std::io;
|
||||
|
||||
fn main() -> std::io::Result<()> {
|
||||
let v = std::env::var("RUST_ANALYZER_INTERNALS_DO_NOT_USE");
|
||||
match v.as_deref() {
|
||||
Ok("this is unstable") => {
|
||||
// very well, if you must
|
||||
}
|
||||
_ => {
|
||||
eprintln!("If you're rust-analyzer, you can use this tool by exporting RUST_ANALYZER_INTERNALS_DO_NOT_USE='this is unstable'.");
|
||||
eprintln!("If not, you probably shouldn't use this tool. But do what you want: I'm an error message, not a cop.");
|
||||
std::process::exit(122);
|
||||
}
|
||||
if v.is_err() {
|
||||
eprintln!("This is an IDE implementation detail, you can use this tool by exporting RUST_ANALYZER_INTERNALS_DO_NOT_USE.");
|
||||
eprintln!(
|
||||
"Note that this tool's API is highly unstable and may break without prior notice"
|
||||
);
|
||||
std::process::exit(122);
|
||||
}
|
||||
|
||||
run()
|
||||
|
@ -28,40 +23,19 @@ fn main() -> std::io::Result<()> {
|
|||
|
||||
#[cfg(not(any(feature = "sysroot-abi", rust_analyzer)))]
|
||||
fn run() -> io::Result<()> {
|
||||
let err = "proc-macro-srv-cli needs to be compiled with the `sysroot-abi` feature to function";
|
||||
eprintln!("{err}");
|
||||
use proc_macro_api::msg::{self, Message};
|
||||
|
||||
let read_request =
|
||||
|buf: &mut String| msg::Request::read(read_json, &mut io::stdin().lock(), buf);
|
||||
|
||||
let write_response = |msg: msg::Response| msg.write(write_json, &mut io::stdout().lock());
|
||||
|
||||
let mut buf = String::new();
|
||||
|
||||
while let Some(req) = read_request(&mut buf)? {
|
||||
let res = match req {
|
||||
msg::Request::ListMacros { .. } => msg::Response::ListMacros(Err(err.to_owned())),
|
||||
msg::Request::ExpandMacro(_) => {
|
||||
msg::Response::ExpandMacro(Err(msg::PanicMessage(err.to_owned())))
|
||||
}
|
||||
msg::Request::ApiVersionCheck {} => {
|
||||
msg::Response::ApiVersionCheck(proc_macro_api::msg::CURRENT_API_VERSION)
|
||||
}
|
||||
msg::Request::SetConfig(_) => {
|
||||
msg::Response::SetConfig(proc_macro_api::msg::ServerConfig {
|
||||
span_mode: msg::SpanMode::Id,
|
||||
})
|
||||
}
|
||||
};
|
||||
write_response(res)?
|
||||
}
|
||||
Ok(())
|
||||
Err(io::Error::new(
|
||||
io::ErrorKind::Unsupported,
|
||||
"proc-macro-srv-cli needs to be compiled with the `sysroot-abi` feature to function"
|
||||
.to_owned(),
|
||||
))
|
||||
}
|
||||
|
||||
#[cfg(any(feature = "sysroot-abi", rust_analyzer))]
|
||||
fn run() -> io::Result<()> {
|
||||
use proc_macro_api::msg::{self, Message};
|
||||
use proc_macro_api::{
|
||||
json::{read_json, write_json},
|
||||
msg::{self, Message},
|
||||
};
|
||||
use proc_macro_srv::EnvSnapshot;
|
||||
|
||||
let read_request =
|
||||
|
@ -69,7 +43,7 @@ fn run() -> io::Result<()> {
|
|||
|
||||
let write_response = |msg: msg::Response| msg.write(write_json, &mut io::stdout().lock());
|
||||
|
||||
let env = EnvSnapshot::new();
|
||||
let env = EnvSnapshot::default();
|
||||
let mut srv = proc_macro_srv::ProcMacroSrv::new(&env);
|
||||
let mut buf = String::new();
|
||||
|
||||
|
|
|
@ -3,12 +3,11 @@
|
|||
mod version;
|
||||
|
||||
use proc_macro::bridge;
|
||||
use std::{fmt, fs::File, io};
|
||||
use std::{fmt, fs, io, time::SystemTime};
|
||||
|
||||
use libloading::Library;
|
||||
use memmap2::Mmap;
|
||||
use object::Object;
|
||||
use paths::{AbsPath, Utf8Path, Utf8PathBuf};
|
||||
use paths::{Utf8Path, Utf8PathBuf};
|
||||
use proc_macro_api::ProcMacroKind;
|
||||
|
||||
use crate::ProcMacroSrvSpan;
|
||||
|
@ -23,14 +22,9 @@ fn is_derive_registrar_symbol(symbol: &str) -> bool {
|
|||
symbol.contains(NEW_REGISTRAR_SYMBOL)
|
||||
}
|
||||
|
||||
fn find_registrar_symbol(file: &Utf8Path) -> io::Result<Option<String>> {
|
||||
let file = File::open(file)?;
|
||||
let buffer = unsafe { Mmap::map(&file)? };
|
||||
|
||||
Ok(object::File::parse(&*buffer)
|
||||
.map_err(invalid_data_err)?
|
||||
.exports()
|
||||
.map_err(invalid_data_err)?
|
||||
fn find_registrar_symbol(obj: &object::File<'_>) -> object::Result<Option<String>> {
|
||||
Ok(obj
|
||||
.exports()?
|
||||
.into_iter()
|
||||
.map(|export| export.name())
|
||||
.filter_map(|sym| String::from_utf8(sym.into()).ok())
|
||||
|
@ -113,17 +107,18 @@ struct ProcMacroLibraryLibloading {
|
|||
}
|
||||
|
||||
impl ProcMacroLibraryLibloading {
|
||||
fn open(file: &Utf8Path) -> Result<Self, LoadProcMacroDylibError> {
|
||||
let symbol_name = find_registrar_symbol(file)?.ok_or_else(|| {
|
||||
invalid_data_err(format!("Cannot find registrar symbol in file {file}"))
|
||||
})?;
|
||||
fn open(path: &Utf8Path) -> Result<Self, LoadProcMacroDylibError> {
|
||||
let file = fs::File::open(path)?;
|
||||
let file = unsafe { memmap2::Mmap::map(&file) }?;
|
||||
let obj = object::File::parse(&*file)
|
||||
.map_err(|e| io::Error::new(io::ErrorKind::InvalidData, e))?;
|
||||
let version_info = version::read_dylib_info(&obj)?;
|
||||
let symbol_name =
|
||||
find_registrar_symbol(&obj).map_err(invalid_data_err)?.ok_or_else(|| {
|
||||
invalid_data_err(format!("Cannot find registrar symbol in file {path}"))
|
||||
})?;
|
||||
|
||||
let abs_file: &AbsPath = file
|
||||
.try_into()
|
||||
.map_err(|_| invalid_data_err(format!("expected an absolute path, got {file}")))?;
|
||||
let version_info = version::read_dylib_info(abs_file)?;
|
||||
|
||||
let lib = load_library(file).map_err(invalid_data_err)?;
|
||||
let lib = load_library(path).map_err(invalid_data_err)?;
|
||||
let proc_macros = crate::proc_macros::ProcMacros::from_lib(
|
||||
&lib,
|
||||
symbol_name,
|
||||
|
@ -133,30 +128,33 @@ impl ProcMacroLibraryLibloading {
|
|||
}
|
||||
}
|
||||
|
||||
pub(crate) struct Expander {
|
||||
inner: ProcMacroLibraryLibloading,
|
||||
path: Utf8PathBuf,
|
||||
}
|
||||
|
||||
impl Drop for Expander {
|
||||
struct RemoveFileOnDrop(Utf8PathBuf);
|
||||
impl Drop for RemoveFileOnDrop {
|
||||
fn drop(&mut self) {
|
||||
#[cfg(windows)]
|
||||
std::fs::remove_file(&self.path).ok();
|
||||
_ = self.path;
|
||||
std::fs::remove_file(&self.0).unwrap();
|
||||
_ = self.0;
|
||||
}
|
||||
}
|
||||
|
||||
// Drop order matters as we can't remove the dylib before the library is unloaded
|
||||
pub(crate) struct Expander {
|
||||
inner: ProcMacroLibraryLibloading,
|
||||
_remove_on_drop: RemoveFileOnDrop,
|
||||
modified_time: SystemTime,
|
||||
}
|
||||
|
||||
impl Expander {
|
||||
pub(crate) fn new(lib: &Utf8Path) -> Result<Expander, LoadProcMacroDylibError> {
|
||||
// Some libraries for dynamic loading require canonicalized path even when it is
|
||||
// already absolute
|
||||
let lib = lib.canonicalize_utf8()?;
|
||||
let modified_time = fs::metadata(&lib).and_then(|it| it.modified())?;
|
||||
|
||||
let path = ensure_file_with_lock_free_access(&lib)?;
|
||||
|
||||
let library = ProcMacroLibraryLibloading::open(path.as_ref())?;
|
||||
|
||||
Ok(Expander { inner: library, path })
|
||||
Ok(Expander { inner: library, _remove_on_drop: RemoveFileOnDrop(path), modified_time })
|
||||
}
|
||||
|
||||
pub(crate) fn expand<S: ProcMacroSrvSpan>(
|
||||
|
@ -181,6 +179,10 @@ impl Expander {
|
|||
pub(crate) fn list_macros(&self) -> Vec<(String, ProcMacroKind)> {
|
||||
self.inner.proc_macros.list_macros()
|
||||
}
|
||||
|
||||
pub(crate) fn modified_time(&self) -> SystemTime {
|
||||
self.modified_time
|
||||
}
|
||||
}
|
||||
|
||||
/// Copy the dylib to temp directory to prevent locking in Windows
|
||||
|
@ -194,20 +196,20 @@ fn ensure_file_with_lock_free_access(path: &Utf8Path) -> io::Result<Utf8PathBuf>
|
|||
}
|
||||
|
||||
let mut to = Utf8PathBuf::from_path_buf(std::env::temp_dir()).unwrap();
|
||||
to.push("rust-analyzer-proc-macros");
|
||||
_ = fs::create_dir(&to);
|
||||
|
||||
let file_name = path.file_name().ok_or_else(|| {
|
||||
let file_name = path.file_stem().ok_or_else(|| {
|
||||
io::Error::new(io::ErrorKind::InvalidInput, format!("File path is invalid: {path}"))
|
||||
})?;
|
||||
|
||||
// Generate a unique number by abusing `HashMap`'s hasher.
|
||||
// Maybe this will also "inspire" a libs team member to finally put `rand` in libstd.
|
||||
let t = RandomState::new().build_hasher().finish();
|
||||
|
||||
let mut unique_name = t.to_string();
|
||||
unique_name.push_str(file_name);
|
||||
|
||||
to.push(unique_name);
|
||||
std::fs::copy(path, &to)?;
|
||||
to.push({
|
||||
// Generate a unique number by abusing `HashMap`'s hasher.
|
||||
// Maybe this will also "inspire" a libs team member to finally put `rand` in libstd.
|
||||
let unique_name = RandomState::new().build_hasher().finish();
|
||||
format!("{file_name}-{unique_name}.dll")
|
||||
});
|
||||
fs::copy(path, &to)?;
|
||||
Ok(to)
|
||||
}
|
||||
|
||||
|
|
|
@ -1,13 +1,8 @@
|
|||
//! Reading proc-macro rustc version information from binary data
|
||||
|
||||
use std::{
|
||||
fs::File,
|
||||
io::{self, Read},
|
||||
};
|
||||
use std::io::{self, Read};
|
||||
|
||||
use memmap2::Mmap;
|
||||
use object::read::{File as BinaryFile, Object, ObjectSection};
|
||||
use paths::AbsPath;
|
||||
use object::read::{Object, ObjectSection};
|
||||
|
||||
#[derive(Debug)]
|
||||
#[allow(dead_code)]
|
||||
|
@ -21,14 +16,14 @@ pub struct RustCInfo {
|
|||
}
|
||||
|
||||
/// Read rustc dylib information
|
||||
pub fn read_dylib_info(dylib_path: &AbsPath) -> io::Result<RustCInfo> {
|
||||
pub fn read_dylib_info(obj: &object::File<'_>) -> io::Result<RustCInfo> {
|
||||
macro_rules! err {
|
||||
($e:literal) => {
|
||||
io::Error::new(io::ErrorKind::InvalidData, $e)
|
||||
};
|
||||
}
|
||||
|
||||
let ver_str = read_version(dylib_path)?;
|
||||
let ver_str = read_version(obj)?;
|
||||
let mut items = ver_str.split_whitespace();
|
||||
let tag = items.next().ok_or_else(|| err!("version format error"))?;
|
||||
if tag != "rustc" {
|
||||
|
@ -75,10 +70,8 @@ pub fn read_dylib_info(dylib_path: &AbsPath) -> io::Result<RustCInfo> {
|
|||
|
||||
/// This is used inside read_version() to locate the ".rustc" section
|
||||
/// from a proc macro crate's binary file.
|
||||
fn read_section<'a>(dylib_binary: &'a [u8], section_name: &str) -> io::Result<&'a [u8]> {
|
||||
BinaryFile::parse(dylib_binary)
|
||||
.map_err(|e| io::Error::new(io::ErrorKind::InvalidData, e))?
|
||||
.section_by_name(section_name)
|
||||
fn read_section<'a>(obj: &object::File<'a>, section_name: &str) -> io::Result<&'a [u8]> {
|
||||
obj.section_by_name(section_name)
|
||||
.ok_or_else(|| io::Error::new(io::ErrorKind::InvalidData, "section read error"))?
|
||||
.data()
|
||||
.map_err(|e| io::Error::new(io::ErrorKind::InvalidData, e))
|
||||
|
@ -106,11 +99,8 @@ fn read_section<'a>(dylib_binary: &'a [u8], section_name: &str) -> io::Result<&'
|
|||
///
|
||||
/// Check this issue for more about the bytes layout:
|
||||
/// <https://github.com/rust-lang/rust-analyzer/issues/6174>
|
||||
pub fn read_version(dylib_path: &AbsPath) -> io::Result<String> {
|
||||
let dylib_file = File::open(dylib_path)?;
|
||||
let dylib_mmapped = unsafe { Mmap::map(&dylib_file) }?;
|
||||
|
||||
let dot_rustc = read_section(&dylib_mmapped, ".rustc")?;
|
||||
pub fn read_version(obj: &object::File<'_>) -> io::Result<String> {
|
||||
let dot_rustc = read_section(obj, ".rustc")?;
|
||||
|
||||
// check if magic is valid
|
||||
if &dot_rustc[0..4] != b"rust" {
|
||||
|
@ -159,8 +149,12 @@ pub fn read_version(dylib_path: &AbsPath) -> io::Result<String> {
|
|||
|
||||
#[test]
|
||||
fn test_version_check() {
|
||||
let path = paths::AbsPathBuf::assert(crate::proc_macro_test_dylib_path());
|
||||
let info = read_dylib_info(&path).unwrap();
|
||||
let info = read_dylib_info(
|
||||
&object::File::parse(&*std::fs::read(crate::proc_macro_test_dylib_path()).unwrap())
|
||||
.unwrap(),
|
||||
)
|
||||
.unwrap();
|
||||
|
||||
assert_eq!(
|
||||
info.version_string,
|
||||
crate::RUSTC_VERSION_STRING,
|
||||
|
|
|
@ -13,7 +13,7 @@
|
|||
#![cfg(any(feature = "sysroot-abi", rust_analyzer))]
|
||||
#![cfg_attr(feature = "in-rust-tree", feature(rustc_private))]
|
||||
#![feature(proc_macro_internals, proc_macro_diagnostic, proc_macro_span)]
|
||||
#![allow(unreachable_pub, internal_features)]
|
||||
#![allow(unreachable_pub, internal_features, clippy::disallowed_types, clippy::print_stderr)]
|
||||
|
||||
extern crate proc_macro;
|
||||
#[cfg(feature = "in-rust-tree")]
|
||||
|
@ -35,7 +35,6 @@ use std::{
|
|||
fs,
|
||||
path::{Path, PathBuf},
|
||||
thread,
|
||||
time::SystemTime,
|
||||
};
|
||||
|
||||
use paths::{Utf8Path, Utf8PathBuf};
|
||||
|
@ -53,7 +52,7 @@ use crate::server_impl::TokenStream;
|
|||
pub const RUSTC_VERSION_STRING: &str = env!("RUSTC_VERSION");
|
||||
|
||||
pub struct ProcMacroSrv<'env> {
|
||||
expanders: HashMap<(Utf8PathBuf, SystemTime), dylib::Expander>,
|
||||
expanders: HashMap<Utf8PathBuf, dylib::Expander>,
|
||||
span_mode: SpanMode,
|
||||
env: &'env EnvSnapshot,
|
||||
}
|
||||
|
@ -66,7 +65,7 @@ impl<'env> ProcMacroSrv<'env> {
|
|||
|
||||
const EXPANDER_STACK_SIZE: usize = 8 * 1024 * 1024;
|
||||
|
||||
impl<'env> ProcMacroSrv<'env> {
|
||||
impl ProcMacroSrv<'_> {
|
||||
pub fn set_span_mode(&mut self, span_mode: SpanMode) {
|
||||
self.span_mode = span_mode;
|
||||
}
|
||||
|
@ -81,10 +80,9 @@ impl<'env> ProcMacroSrv<'env> {
|
|||
) -> Result<(msg::FlatTree, Vec<u32>), msg::PanicMessage> {
|
||||
let span_mode = self.span_mode;
|
||||
let snapped_env = self.env;
|
||||
let expander = self.expander(lib.as_ref()).map_err(|err| {
|
||||
debug_assert!(false, "should list macros before asking to expand");
|
||||
msg::PanicMessage(format!("failed to load macro: {err}"))
|
||||
})?;
|
||||
let expander = self
|
||||
.expander(lib.as_ref())
|
||||
.map_err(|err| msg::PanicMessage(format!("failed to load macro: {err}")))?;
|
||||
|
||||
let prev_env = EnvChange::apply(snapped_env, env, current_dir.as_ref().map(<_>::as_ref));
|
||||
|
||||
|
@ -107,16 +105,20 @@ impl<'env> ProcMacroSrv<'env> {
|
|||
}
|
||||
|
||||
fn expander(&mut self, path: &Utf8Path) -> Result<&dylib::Expander, String> {
|
||||
let time = fs::metadata(path)
|
||||
.and_then(|it| it.modified())
|
||||
.map_err(|err| format!("Failed to get file metadata for {path}: {err}",))?;
|
||||
let expander = || {
|
||||
dylib::Expander::new(path)
|
||||
.map_err(|err| format!("Cannot create expander for {path}: {err}",))
|
||||
};
|
||||
|
||||
Ok(match self.expanders.entry((path.to_path_buf(), time)) {
|
||||
Entry::Vacant(v) => v.insert(
|
||||
dylib::Expander::new(path)
|
||||
.map_err(|err| format!("Cannot create expander for {path}: {err}",))?,
|
||||
),
|
||||
Entry::Occupied(e) => e.into_mut(),
|
||||
Ok(match self.expanders.entry(path.to_path_buf()) {
|
||||
Entry::Vacant(v) => v.insert(expander()?),
|
||||
Entry::Occupied(mut e) => {
|
||||
let time = fs::metadata(path).and_then(|it| it.modified()).ok();
|
||||
if Some(e.get().modified_time()) != time {
|
||||
e.insert(expander()?);
|
||||
}
|
||||
e.into_mut()
|
||||
}
|
||||
})
|
||||
}
|
||||
}
|
||||
|
@ -246,8 +248,8 @@ pub struct EnvSnapshot {
|
|||
vars: HashMap<OsString, OsString>,
|
||||
}
|
||||
|
||||
impl EnvSnapshot {
|
||||
pub fn new() -> EnvSnapshot {
|
||||
impl Default for EnvSnapshot {
|
||||
fn default() -> EnvSnapshot {
|
||||
EnvSnapshot { vars: env::vars_os().collect() }
|
||||
}
|
||||
}
|
||||
|
@ -303,7 +305,7 @@ impl Drop for EnvChange<'_> {
|
|||
}
|
||||
|
||||
if let Some(dir) = &self.prev_working_dir {
|
||||
if let Err(err) = std::env::set_current_dir(&dir) {
|
||||
if let Err(err) = std::env::set_current_dir(dir) {
|
||||
eprintln!(
|
||||
"Failed to set the current working dir to {}. Error: {:?}",
|
||||
dir.display(),
|
||||
|
|
|
@ -13,7 +13,7 @@ pub(crate) struct ProcMacros {
|
|||
|
||||
impl From<bridge::PanicMessage> for crate::PanicMessage {
|
||||
fn from(p: bridge::PanicMessage) -> Self {
|
||||
Self { message: p.as_str().map(|s| s.to_string()) }
|
||||
Self { message: p.as_str().map(|s| s.to_owned()) }
|
||||
}
|
||||
}
|
||||
|
||||
|
|
|
@ -498,7 +498,7 @@ mod tests {
|
|||
})),
|
||||
tt::TokenTree::Leaf(tt::Leaf::Ident(tt::Ident {
|
||||
sym: Symbol::intern("T"),
|
||||
span: span,
|
||||
span,
|
||||
is_raw: tt::IdentIsRaw::No,
|
||||
})),
|
||||
tt::TokenTree::Subtree(tt::Subtree {
|
||||
|
|
|
@ -99,7 +99,7 @@ pub(super) struct TokenStreamBuilder<S> {
|
|||
}
|
||||
|
||||
/// pub(super)lic implementation details for the `TokenStream` type, such as iterators.
|
||||
pub(super) mod token_stream {
|
||||
pub(super) mod token_stream_impls {
|
||||
|
||||
use core::fmt;
|
||||
|
||||
|
@ -137,6 +137,7 @@ pub(super) mod token_stream {
|
|||
}
|
||||
}
|
||||
|
||||
#[allow(clippy::to_string_trait_impl)]
|
||||
impl<S> ToString for TokenStream<S> {
|
||||
fn to_string(&self) -> String {
|
||||
::tt::pretty(&self.token_trees)
|
||||
|
@ -150,7 +151,7 @@ impl<S> TokenStreamBuilder<S> {
|
|||
}
|
||||
|
||||
pub(super) fn push(&mut self, stream: TokenStream<S>) {
|
||||
self.acc.extend(stream.into_iter())
|
||||
self.acc.extend(stream)
|
||||
}
|
||||
|
||||
pub(super) fn build(self) -> TokenStream<S> {
|
||||
|
|
|
@ -97,7 +97,7 @@ fn assert_expand_impl(
|
|||
|
||||
pub(crate) fn list() -> Vec<String> {
|
||||
let dylib_path = proc_macro_test_dylib_path();
|
||||
let env = EnvSnapshot::new();
|
||||
let env = EnvSnapshot::default();
|
||||
let mut srv = ProcMacroSrv::new(&env);
|
||||
let res = srv.list_macros(&dylib_path).unwrap();
|
||||
res.into_iter().map(|(name, kind)| format!("{name} [{kind:?}]")).collect()
|
||||
|
|
|
@ -33,6 +33,7 @@ pub struct CargoWorkspace {
|
|||
workspace_root: AbsPathBuf,
|
||||
target_directory: AbsPathBuf,
|
||||
manifest_path: ManifestPath,
|
||||
is_virtual_workspace: bool,
|
||||
}
|
||||
|
||||
impl ops::Index<Package> for CargoWorkspace {
|
||||
|
@ -384,13 +385,20 @@ impl CargoWorkspace {
|
|||
.with_context(|| format!("Failed to run `{:?}`", meta.cargo_command()))
|
||||
}
|
||||
|
||||
pub fn new(mut meta: cargo_metadata::Metadata, manifest_path: ManifestPath) -> CargoWorkspace {
|
||||
pub fn new(
|
||||
mut meta: cargo_metadata::Metadata,
|
||||
ws_manifest_path: ManifestPath,
|
||||
) -> CargoWorkspace {
|
||||
let mut pkg_by_id = FxHashMap::default();
|
||||
let mut packages = Arena::default();
|
||||
let mut targets = Arena::default();
|
||||
|
||||
let ws_members = &meta.workspace_members;
|
||||
|
||||
let workspace_root = AbsPathBuf::assert(meta.workspace_root);
|
||||
let target_directory = AbsPathBuf::assert(meta.target_directory);
|
||||
let mut is_virtual_workspace = true;
|
||||
|
||||
meta.packages.sort_by(|a, b| a.id.cmp(&b.id));
|
||||
for meta_pkg in meta.packages {
|
||||
let cargo_metadata::Package {
|
||||
|
@ -429,12 +437,13 @@ impl CargoWorkspace {
|
|||
let is_local = source.is_none();
|
||||
let is_member = ws_members.contains(&id);
|
||||
|
||||
let manifest = AbsPathBuf::assert(manifest_path);
|
||||
let manifest = ManifestPath::try_from(AbsPathBuf::assert(manifest_path)).unwrap();
|
||||
is_virtual_workspace &= manifest != ws_manifest_path;
|
||||
let pkg = packages.alloc(PackageData {
|
||||
id: id.repr.clone(),
|
||||
name,
|
||||
version,
|
||||
manifest: manifest.clone().try_into().unwrap(),
|
||||
manifest: manifest.clone(),
|
||||
targets: Vec::new(),
|
||||
is_local,
|
||||
is_member,
|
||||
|
@ -468,7 +477,7 @@ impl CargoWorkspace {
|
|||
// modified manifest file into a special target dir which is then used as
|
||||
// the source path. We don't want that, we want the original here so map it
|
||||
// back
|
||||
manifest.clone()
|
||||
manifest.clone().into()
|
||||
} else {
|
||||
AbsPathBuf::assert(src_path)
|
||||
},
|
||||
|
@ -493,11 +502,14 @@ impl CargoWorkspace {
|
|||
packages[source].active_features.extend(node.features);
|
||||
}
|
||||
|
||||
let workspace_root = AbsPathBuf::assert(meta.workspace_root);
|
||||
|
||||
let target_directory = AbsPathBuf::assert(meta.target_directory);
|
||||
|
||||
CargoWorkspace { packages, targets, workspace_root, target_directory, manifest_path }
|
||||
CargoWorkspace {
|
||||
packages,
|
||||
targets,
|
||||
workspace_root,
|
||||
target_directory,
|
||||
manifest_path: ws_manifest_path,
|
||||
is_virtual_workspace,
|
||||
}
|
||||
}
|
||||
|
||||
pub fn packages(&self) -> impl ExactSizeIterator<Item = Package> + '_ {
|
||||
|
@ -579,6 +591,10 @@ impl CargoWorkspace {
|
|||
fn is_unique(&self, name: &str) -> bool {
|
||||
self.packages.iter().filter(|(_, v)| v.name == name).count() == 1
|
||||
}
|
||||
|
||||
pub fn is_virtual_workspace(&self) -> bool {
|
||||
self.is_virtual_workspace
|
||||
}
|
||||
}
|
||||
|
||||
fn find_list_of_build_targets(
|
||||
|
|
|
@ -29,6 +29,12 @@ impl TryFrom<AbsPathBuf> for ManifestPath {
|
|||
}
|
||||
}
|
||||
|
||||
impl From<ManifestPath> for AbsPathBuf {
|
||||
fn from(it: ManifestPath) -> Self {
|
||||
it.file
|
||||
}
|
||||
}
|
||||
|
||||
impl ManifestPath {
|
||||
// Shadow `parent` from `Deref`.
|
||||
pub fn parent(&self) -> &AbsPath {
|
||||
|
|
|
@ -11,8 +11,9 @@ use base_db::{
|
|||
};
|
||||
use cfg::{CfgAtom, CfgDiff, CfgOptions};
|
||||
use intern::{sym, Symbol};
|
||||
use itertools::Itertools;
|
||||
use paths::{AbsPath, AbsPathBuf};
|
||||
use rustc_hash::{FxHashMap, FxHashSet};
|
||||
use rustc_hash::FxHashMap;
|
||||
use semver::Version;
|
||||
use span::{Edition, FileId};
|
||||
use toolchain::Tool;
|
||||
|
@ -41,7 +42,9 @@ pub type FileLoader<'a> = &'a mut dyn for<'b> FnMut(&'b AbsPath) -> Option<FileI
|
|||
pub struct PackageRoot {
|
||||
/// Is from the local filesystem and may be edited
|
||||
pub is_local: bool,
|
||||
/// Directories to include
|
||||
pub include: Vec<AbsPathBuf>,
|
||||
/// Directories to exclude
|
||||
pub exclude: Vec<AbsPathBuf>,
|
||||
}
|
||||
|
||||
|
@ -553,17 +556,6 @@ impl ProjectWorkspace {
|
|||
}
|
||||
}
|
||||
|
||||
pub fn buildfiles(&self) -> Vec<AbsPathBuf> {
|
||||
match &self.kind {
|
||||
ProjectWorkspaceKind::Json(project) => project
|
||||
.crates()
|
||||
.filter_map(|(_, krate)| krate.build.as_ref().map(|build| build.build_file.clone()))
|
||||
.map(|build_file| self.workspace_root().join(build_file))
|
||||
.collect(),
|
||||
_ => vec![],
|
||||
}
|
||||
}
|
||||
|
||||
pub fn find_sysroot_proc_macro_srv(&self) -> anyhow::Result<AbsPathBuf> {
|
||||
self.sysroot.discover_proc_macro_srv()
|
||||
}
|
||||
|
@ -608,15 +600,25 @@ impl ProjectWorkspace {
|
|||
match &self.kind {
|
||||
ProjectWorkspaceKind::Json(project) => project
|
||||
.crates()
|
||||
.map(|(_, krate)| PackageRoot {
|
||||
is_local: krate.is_workspace_member,
|
||||
include: krate.include.clone(),
|
||||
exclude: krate.exclude.clone(),
|
||||
.map(|(_, krate)| {
|
||||
let build_files = project
|
||||
.crates()
|
||||
.filter_map(|(_, krate)| {
|
||||
krate.build.as_ref().map(|build| build.build_file.clone())
|
||||
})
|
||||
// FIXME: PackageRoots dont allow specifying files, only directories
|
||||
.filter_map(|build_file| {
|
||||
self.workspace_root().join(build_file).parent().map(ToOwned::to_owned)
|
||||
});
|
||||
PackageRoot {
|
||||
is_local: krate.is_workspace_member,
|
||||
include: krate.include.iter().cloned().chain(build_files).collect(),
|
||||
exclude: krate.exclude.clone(),
|
||||
}
|
||||
})
|
||||
.collect::<FxHashSet<_>>()
|
||||
.into_iter()
|
||||
.chain(mk_sysroot())
|
||||
.collect::<Vec<_>>(),
|
||||
.unique()
|
||||
.collect(),
|
||||
ProjectWorkspaceKind::Cargo {
|
||||
cargo,
|
||||
rustc,
|
||||
|
@ -671,6 +673,11 @@ impl ProjectWorkspace {
|
|||
exclude: Vec::new(),
|
||||
})
|
||||
}))
|
||||
.chain(cargo.is_virtual_workspace().then(|| PackageRoot {
|
||||
is_local: true,
|
||||
include: vec![cargo.workspace_root().to_path_buf()],
|
||||
exclude: Vec::new(),
|
||||
}))
|
||||
.collect()
|
||||
}
|
||||
ProjectWorkspaceKind::DetachedFile { file, cargo: cargo_script, .. } => {
|
||||
|
|
|
@ -11,7 +11,7 @@
|
|||
/// case), and we know that the query only used inputs of medium
|
||||
/// durability or higher, then we can skip that enumeration.
|
||||
///
|
||||
/// Typically, one assigns low durabilites to inputs that the user is
|
||||
/// Typically, one assigns low durabilities to inputs that the user is
|
||||
/// frequently editing. Medium or high durabilities are used for
|
||||
/// configuration, the source from library crates, or other things
|
||||
/// that are unlikely to be edited.
|
||||
|
|
|
@ -291,7 +291,7 @@ pub trait ParallelDatabase: Database + Send {
|
|||
/// # Panics
|
||||
///
|
||||
/// It is not permitted to create a snapshot from inside of a
|
||||
/// query. Attepting to do so will panic.
|
||||
/// query. Attempting to do so will panic.
|
||||
///
|
||||
/// # Deadlock warning
|
||||
///
|
||||
|
|
|
@ -255,7 +255,7 @@ fn cycle_revalidate_unchanged_twice() {
|
|||
db.set_b_invokes(CycleQuery::A);
|
||||
|
||||
assert!(db.cycle_a().is_err());
|
||||
db.set_c_invokes(CycleQuery::A); // force new revisi5on
|
||||
db.set_c_invokes(CycleQuery::A); // force new revision
|
||||
|
||||
// on this run
|
||||
expect![[r#"
|
||||
|
|
|
@ -51,7 +51,9 @@ fn actual_main() -> anyhow::Result<ExitCode> {
|
|||
}
|
||||
}
|
||||
|
||||
setup_logging(flags.log_file.clone())?;
|
||||
if let Err(e) = setup_logging(flags.log_file.clone()) {
|
||||
eprintln!("Failed to setup logging: {e:#}");
|
||||
}
|
||||
|
||||
let verbosity = flags.verbosity();
|
||||
|
||||
|
|
|
@ -331,8 +331,8 @@ impl flags::AnalysisStats {
|
|||
let mut fail = 0;
|
||||
for &b in bodies {
|
||||
let res = match b {
|
||||
DefWithBody::Const(c) => c.render_eval(db, Edition::LATEST),
|
||||
DefWithBody::Static(s) => s.render_eval(db, Edition::LATEST),
|
||||
DefWithBody::Const(c) => c.eval(db),
|
||||
DefWithBody::Static(s) => s.eval(db),
|
||||
_ => continue,
|
||||
};
|
||||
all += 1;
|
||||
|
|
|
@ -827,6 +827,7 @@ impl Config {
|
|||
let mut should_update = false;
|
||||
|
||||
if let Some(change) = change.user_config_change {
|
||||
tracing::info!("updating config from user config toml: {:#}", change);
|
||||
if let Ok(table) = toml::from_str(&change) {
|
||||
let mut toml_errors = vec![];
|
||||
validate_toml_table(
|
||||
|
@ -919,7 +920,7 @@ impl Config {
|
|||
RatomlFileKind::Crate => {
|
||||
if let Some(text) = text {
|
||||
let mut toml_errors = vec![];
|
||||
tracing::info!("updating ra-toml config: {:#}", text);
|
||||
tracing::info!("updating ra-toml crate config: {:#}", text);
|
||||
match toml::from_str(&text) {
|
||||
Ok(table) => {
|
||||
validate_toml_table(
|
||||
|
@ -961,6 +962,7 @@ impl Config {
|
|||
}
|
||||
RatomlFileKind::Workspace => {
|
||||
if let Some(text) = text {
|
||||
tracing::info!("updating ra-toml workspace config: {:#}", text);
|
||||
let mut toml_errors = vec![];
|
||||
match toml::from_str(&text) {
|
||||
Ok(table) => {
|
||||
|
|
|
@ -417,8 +417,10 @@ impl GlobalState {
|
|||
})
|
||||
.collect_vec();
|
||||
|
||||
for (file_id, (_change_kind, vfs_path)) in modified_ratoml_files {
|
||||
for (file_id, (change_kind, vfs_path)) in modified_ratoml_files {
|
||||
tracing::info!(%vfs_path, ?change_kind, "Processing rust-analyzer.toml changes");
|
||||
if vfs_path.as_path() == user_config_abs_path {
|
||||
tracing::info!(%vfs_path, ?change_kind, "Use config rust-analyzer.toml changes");
|
||||
change.change_user_config(Some(db.file_text(file_id)));
|
||||
continue;
|
||||
}
|
||||
|
@ -430,12 +432,14 @@ impl GlobalState {
|
|||
|
||||
if !sr.is_library {
|
||||
let entry = if workspace_ratoml_paths.contains(&vfs_path) {
|
||||
tracing::info!(%vfs_path, ?sr_id, "workspace rust-analyzer.toml changes");
|
||||
change.change_workspace_ratoml(
|
||||
sr_id,
|
||||
vfs_path.clone(),
|
||||
Some(db.file_text(file_id)),
|
||||
)
|
||||
} else {
|
||||
tracing::info!(%vfs_path, ?sr_id, "crate rust-analyzer.toml changes");
|
||||
change.change_ratoml(
|
||||
sr_id,
|
||||
vfs_path.clone(),
|
||||
|
@ -446,7 +450,7 @@ impl GlobalState {
|
|||
if let Some((kind, old_path, old_text)) = entry {
|
||||
// SourceRoot has more than 1 RATOML files. In this case lexicographically smaller wins.
|
||||
if old_path < vfs_path {
|
||||
span!(Level::ERROR, "Two `rust-analyzer.toml` files were found inside the same crate. {vfs_path} has no effect.");
|
||||
tracing::error!("Two `rust-analyzer.toml` files were found inside the same crate. {vfs_path} has no effect.");
|
||||
// Put the old one back in.
|
||||
match kind {
|
||||
RatomlFileKind::Crate => {
|
||||
|
@ -459,8 +463,7 @@ impl GlobalState {
|
|||
}
|
||||
}
|
||||
} else {
|
||||
// Mapping to a SourceRoot should always end up in `Ok`
|
||||
span!(Level::ERROR, "Mapping to SourceRootId failed.");
|
||||
tracing::info!(%vfs_path, "Ignoring library rust-analyzer.toml");
|
||||
}
|
||||
}
|
||||
change.change_source_root_parent_map(self.local_roots_parent_map.clone());
|
||||
|
|
|
@ -895,7 +895,29 @@ pub fn item_const(
|
|||
None => String::new(),
|
||||
Some(it) => format!("{it} "),
|
||||
};
|
||||
ast_from_text(&format!("{visibility} const {name}: {ty} = {expr};"))
|
||||
ast_from_text(&format!("{visibility}const {name}: {ty} = {expr};"))
|
||||
}
|
||||
|
||||
pub fn item_static(
|
||||
visibility: Option<ast::Visibility>,
|
||||
is_unsafe: bool,
|
||||
is_mut: bool,
|
||||
name: ast::Name,
|
||||
ty: ast::Type,
|
||||
expr: Option<ast::Expr>,
|
||||
) -> ast::Static {
|
||||
let visibility = match visibility {
|
||||
None => String::new(),
|
||||
Some(it) => format!("{it} "),
|
||||
};
|
||||
let is_unsafe = if is_unsafe { "unsafe " } else { "" };
|
||||
let is_mut = if is_mut { "mut " } else { "" };
|
||||
let expr = match expr {
|
||||
Some(it) => &format!(" = {it}"),
|
||||
None => "",
|
||||
};
|
||||
|
||||
ast_from_text(&format!("{visibility}{is_unsafe}static {is_mut}{name}: {ty}{expr};"))
|
||||
}
|
||||
|
||||
pub fn unnamed_param(ty: ast::Type) -> ast::Param {
|
||||
|
|
|
@ -188,6 +188,73 @@ impl SyntaxFactory {
|
|||
ast
|
||||
}
|
||||
|
||||
pub fn item_const(
|
||||
&self,
|
||||
visibility: Option<ast::Visibility>,
|
||||
name: ast::Name,
|
||||
ty: ast::Type,
|
||||
expr: ast::Expr,
|
||||
) -> ast::Const {
|
||||
let ast = make::item_const(visibility.clone(), name.clone(), ty.clone(), expr.clone())
|
||||
.clone_for_update();
|
||||
|
||||
if let Some(mut mapping) = self.mappings() {
|
||||
let mut builder = SyntaxMappingBuilder::new(ast.syntax().clone());
|
||||
if let Some(visibility) = visibility {
|
||||
builder.map_node(
|
||||
visibility.syntax().clone(),
|
||||
ast.visibility().unwrap().syntax().clone(),
|
||||
);
|
||||
}
|
||||
builder.map_node(name.syntax().clone(), ast.name().unwrap().syntax().clone());
|
||||
builder.map_node(ty.syntax().clone(), ast.ty().unwrap().syntax().clone());
|
||||
builder.map_node(expr.syntax().clone(), ast.body().unwrap().syntax().clone());
|
||||
builder.finish(&mut mapping);
|
||||
}
|
||||
|
||||
ast
|
||||
}
|
||||
|
||||
pub fn item_static(
|
||||
&self,
|
||||
visibility: Option<ast::Visibility>,
|
||||
is_unsafe: bool,
|
||||
is_mut: bool,
|
||||
name: ast::Name,
|
||||
ty: ast::Type,
|
||||
expr: Option<ast::Expr>,
|
||||
) -> ast::Static {
|
||||
let ast = make::item_static(
|
||||
visibility.clone(),
|
||||
is_unsafe,
|
||||
is_mut,
|
||||
name.clone(),
|
||||
ty.clone(),
|
||||
expr.clone(),
|
||||
)
|
||||
.clone_for_update();
|
||||
|
||||
if let Some(mut mapping) = self.mappings() {
|
||||
let mut builder = SyntaxMappingBuilder::new(ast.syntax().clone());
|
||||
if let Some(visibility) = visibility {
|
||||
builder.map_node(
|
||||
visibility.syntax().clone(),
|
||||
ast.visibility().unwrap().syntax().clone(),
|
||||
);
|
||||
}
|
||||
|
||||
builder.map_node(name.syntax().clone(), ast.name().unwrap().syntax().clone());
|
||||
builder.map_node(ty.syntax().clone(), ast.ty().unwrap().syntax().clone());
|
||||
|
||||
if let Some(expr) = expr {
|
||||
builder.map_node(expr.syntax().clone(), ast.body().unwrap().syntax().clone());
|
||||
}
|
||||
builder.finish(&mut mapping);
|
||||
}
|
||||
|
||||
ast
|
||||
}
|
||||
|
||||
pub fn turbofish_generic_arg_list(
|
||||
&self,
|
||||
args: impl IntoIterator<Item = ast::GenericArg> + Clone,
|
||||
|
|
|
@ -580,7 +580,7 @@ Unfortunately, it downloads an old version of `rust-analyzer`, but you can set t
|
|||
|
||||
There is a package named `ra_ap_rust_analyzer` available on https://crates.io/crates/ra_ap_rust-analyzer[crates.io], for someone who wants to use it programmatically.
|
||||
|
||||
For more details, see https://github.com/rust-lang/rust-analyzer/blob/master/.github/workflows/publish.yml[the publish workflow].
|
||||
For more details, see https://github.com/rust-lang/rust-analyzer/blob/master/.github/workflows/autopublish.yaml[the publish workflow].
|
||||
|
||||
=== Zed
|
||||
|
||||
|
|
|
@ -324,7 +324,7 @@ class ExperimentalFeatures implements lc.StaticFeature {
|
|||
}
|
||||
fillClientCapabilities(capabilities: lc.ClientCapabilities): void {
|
||||
capabilities.experimental = {
|
||||
snippetTextEdit: false,
|
||||
snippetTextEdit: true,
|
||||
codeActionGroup: true,
|
||||
hoverActions: true,
|
||||
serverStatusNotification: true,
|
||||
|
|
|
@ -21,6 +21,7 @@ quote = "1.0.20"
|
|||
ungrammar = "1.16.1"
|
||||
either.workspace = true
|
||||
itertools.workspace = true
|
||||
edition.workspace = true
|
||||
# Avoid adding more dependencies to this crate
|
||||
|
||||
[lints]
|
||||
|
|
|
@ -1,7 +1,15 @@
|
|||
//! Generates descriptor structures for unstable features from the unstable book
|
||||
//! and lints from rustc, rustdoc, and clippy.
|
||||
use std::{borrow::Cow, fs, path::Path};
|
||||
#![allow(clippy::disallowed_types)]
|
||||
|
||||
use std::{
|
||||
collections::{hash_map, HashMap},
|
||||
fs,
|
||||
path::Path,
|
||||
str::FromStr,
|
||||
};
|
||||
|
||||
use edition::Edition;
|
||||
use stdx::format_to;
|
||||
use xshell::{cmd, Shell};
|
||||
|
||||
|
@ -36,10 +44,17 @@ pub(crate) fn generate(check: bool) {
|
|||
|
||||
let mut contents = String::from(
|
||||
r"
|
||||
use span::Edition;
|
||||
|
||||
use crate::Severity;
|
||||
|
||||
#[derive(Clone)]
|
||||
pub struct Lint {
|
||||
pub label: &'static str,
|
||||
pub description: &'static str,
|
||||
pub default_severity: Severity,
|
||||
pub warn_since: Option<Edition>,
|
||||
pub deny_since: Option<Edition>,
|
||||
}
|
||||
|
||||
pub struct LintGroup {
|
||||
|
@ -68,7 +83,7 @@ pub struct LintGroup {
|
|||
let lints_json = project_root().join("./target/clippy_lints.json");
|
||||
cmd!(
|
||||
sh,
|
||||
"curl https://rust-lang.github.io/rust-clippy/master/lints.json --output {lints_json}"
|
||||
"curl https://rust-lang.github.io/rust-clippy/stable/lints.json --output {lints_json}"
|
||||
)
|
||||
.run()
|
||||
.unwrap();
|
||||
|
@ -85,6 +100,48 @@ pub struct LintGroup {
|
|||
);
|
||||
}
|
||||
|
||||
#[derive(Debug, PartialEq, Eq, PartialOrd, Ord)]
|
||||
enum Severity {
|
||||
Allow,
|
||||
Warn,
|
||||
Deny,
|
||||
}
|
||||
|
||||
impl std::fmt::Display for Severity {
|
||||
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
||||
write!(
|
||||
f,
|
||||
"Severity::{}",
|
||||
match self {
|
||||
Severity::Allow => "Allow",
|
||||
Severity::Warn => "Warning",
|
||||
Severity::Deny => "Error",
|
||||
}
|
||||
)
|
||||
}
|
||||
}
|
||||
|
||||
impl FromStr for Severity {
|
||||
type Err = &'static str;
|
||||
|
||||
fn from_str(s: &str) -> Result<Self, Self::Err> {
|
||||
match s {
|
||||
"allow" => Ok(Self::Allow),
|
||||
"warn" => Ok(Self::Warn),
|
||||
"deny" => Ok(Self::Deny),
|
||||
_ => Err("invalid severity"),
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
#[derive(Debug)]
|
||||
struct Lint {
|
||||
description: String,
|
||||
default_severity: Severity,
|
||||
warn_since: Option<Edition>,
|
||||
deny_since: Option<Edition>,
|
||||
}
|
||||
|
||||
/// Parses the output of `rustdoc -Whelp` and prints `Lint` and `LintGroup` constants into `buf`.
|
||||
///
|
||||
/// As of writing, the output of `rustc -Whelp` (not rustdoc) has the following format:
|
||||
|
@ -108,52 +165,203 @@ pub struct LintGroup {
|
|||
/// `rustdoc -Whelp` (and any other custom `rustc` driver) adds another two
|
||||
/// tables after the `rustc` ones, with a different title but the same format.
|
||||
fn generate_lint_descriptor(sh: &Shell, buf: &mut String) {
|
||||
let stdout = cmd!(sh, "rustdoc -Whelp").read().unwrap();
|
||||
let lints_pat = "---- ------- -------\n";
|
||||
let lint_groups_pat = "---- ---------\n";
|
||||
let lints = find_and_slice(&stdout, lints_pat);
|
||||
let lint_groups = find_and_slice(lints, lint_groups_pat);
|
||||
let lints_rustdoc = find_and_slice(lint_groups, lints_pat);
|
||||
let lint_groups_rustdoc = find_and_slice(lints_rustdoc, lint_groups_pat);
|
||||
fn get_lints_as_text(
|
||||
stdout: &str,
|
||||
) -> (
|
||||
impl Iterator<Item = (String, &str, Severity)> + '_,
|
||||
impl Iterator<Item = (String, Lint, impl Iterator<Item = String> + '_)> + '_,
|
||||
impl Iterator<Item = (String, &str, Severity)> + '_,
|
||||
impl Iterator<Item = (String, Lint, impl Iterator<Item = String> + '_)> + '_,
|
||||
) {
|
||||
let lints_pat = "---- ------- -------\n";
|
||||
let lint_groups_pat = "---- ---------\n";
|
||||
let lints = find_and_slice(stdout, lints_pat);
|
||||
let lint_groups = find_and_slice(lints, lint_groups_pat);
|
||||
let lints_rustdoc = find_and_slice(lint_groups, lints_pat);
|
||||
let lint_groups_rustdoc = find_and_slice(lints_rustdoc, lint_groups_pat);
|
||||
|
||||
let lints = lints.lines().take_while(|l| !l.is_empty()).map(|line| {
|
||||
let (name, rest) = line.trim().split_once(char::is_whitespace).unwrap();
|
||||
let (severity, description) = rest.trim().split_once(char::is_whitespace).unwrap();
|
||||
(name.trim().replace('-', "_"), description.trim(), severity.parse().unwrap())
|
||||
});
|
||||
let lint_groups = lint_groups.lines().take_while(|l| !l.is_empty()).map(|line| {
|
||||
let (name, lints) = line.trim().split_once(char::is_whitespace).unwrap();
|
||||
let label = name.trim().replace('-', "_");
|
||||
let lint = Lint {
|
||||
description: format!("lint group for: {}", lints.trim()),
|
||||
default_severity: Severity::Allow,
|
||||
warn_since: None,
|
||||
deny_since: None,
|
||||
};
|
||||
let children = lints
|
||||
.split_ascii_whitespace()
|
||||
.map(|s| s.trim().trim_matches(',').replace('-', "_"));
|
||||
(label, lint, children)
|
||||
});
|
||||
|
||||
let lints_rustdoc = lints_rustdoc.lines().take_while(|l| !l.is_empty()).map(|line| {
|
||||
let (name, rest) = line.trim().split_once(char::is_whitespace).unwrap();
|
||||
let (severity, description) = rest.trim().split_once(char::is_whitespace).unwrap();
|
||||
(name.trim().replace('-', "_"), description.trim(), severity.parse().unwrap())
|
||||
});
|
||||
let lint_groups_rustdoc =
|
||||
lint_groups_rustdoc.lines().take_while(|l| !l.is_empty()).map(|line| {
|
||||
let (name, lints) = line.trim().split_once(char::is_whitespace).unwrap();
|
||||
let label = name.trim().replace('-', "_");
|
||||
let lint = Lint {
|
||||
description: format!("lint group for: {}", lints.trim()),
|
||||
default_severity: Severity::Allow,
|
||||
warn_since: None,
|
||||
deny_since: None,
|
||||
};
|
||||
let children = lints
|
||||
.split_ascii_whitespace()
|
||||
.map(|s| s.trim().trim_matches(',').replace('-', "_"));
|
||||
(label, lint, children)
|
||||
});
|
||||
|
||||
(lints, lint_groups, lints_rustdoc, lint_groups_rustdoc)
|
||||
}
|
||||
|
||||
fn insert_lints<'a>(
|
||||
edition: Edition,
|
||||
lints_map: &mut HashMap<String, Lint>,
|
||||
lint_groups_map: &mut HashMap<String, (Lint, Vec<String>)>,
|
||||
lints: impl Iterator<Item = (String, &'a str, Severity)>,
|
||||
lint_groups: impl Iterator<Item = (String, Lint, impl Iterator<Item = String>)>,
|
||||
) {
|
||||
for (lint_name, lint_description, lint_severity) in lints {
|
||||
let lint = lints_map.entry(lint_name).or_insert_with(|| Lint {
|
||||
description: lint_description.to_owned(),
|
||||
default_severity: Severity::Allow,
|
||||
warn_since: None,
|
||||
deny_since: None,
|
||||
});
|
||||
if lint_severity == Severity::Warn
|
||||
&& lint.warn_since.is_none()
|
||||
&& lint.default_severity < Severity::Warn
|
||||
{
|
||||
lint.warn_since = Some(edition);
|
||||
}
|
||||
if lint_severity == Severity::Deny
|
||||
&& lint.deny_since.is_none()
|
||||
&& lint.default_severity < Severity::Deny
|
||||
{
|
||||
lint.deny_since = Some(edition);
|
||||
}
|
||||
}
|
||||
|
||||
for (group_name, lint, children) in lint_groups {
|
||||
match lint_groups_map.entry(group_name) {
|
||||
hash_map::Entry::Vacant(entry) => {
|
||||
entry.insert((lint, Vec::from_iter(children)));
|
||||
}
|
||||
hash_map::Entry::Occupied(mut entry) => {
|
||||
// Overwrite, because some groups (such as edition incompatibility) are changed.
|
||||
*entry.get_mut() = (lint, Vec::from_iter(children));
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
fn get_lints(
|
||||
sh: &Shell,
|
||||
edition: Edition,
|
||||
lints_map: &mut HashMap<String, Lint>,
|
||||
lint_groups_map: &mut HashMap<String, (Lint, Vec<String>)>,
|
||||
lints_rustdoc_map: &mut HashMap<String, Lint>,
|
||||
lint_groups_rustdoc_map: &mut HashMap<String, (Lint, Vec<String>)>,
|
||||
) {
|
||||
let edition_str = edition.to_string();
|
||||
let stdout = cmd!(sh, "rustdoc +nightly -Whelp -Zunstable-options --edition={edition_str}")
|
||||
.read()
|
||||
.unwrap();
|
||||
let (lints, lint_groups, lints_rustdoc, lint_groups_rustdoc) = get_lints_as_text(&stdout);
|
||||
|
||||
insert_lints(edition, lints_map, lint_groups_map, lints, lint_groups);
|
||||
insert_lints(
|
||||
edition,
|
||||
lints_rustdoc_map,
|
||||
lint_groups_rustdoc_map,
|
||||
lints_rustdoc,
|
||||
lint_groups_rustdoc,
|
||||
);
|
||||
}
|
||||
|
||||
let basic_lints = cmd!(sh, "rustdoc +nightly -Whelp --edition=2015").read().unwrap();
|
||||
let (lints, lint_groups, lints_rustdoc, lint_groups_rustdoc) = get_lints_as_text(&basic_lints);
|
||||
|
||||
let mut lints = lints
|
||||
.map(|(label, description, severity)| {
|
||||
(
|
||||
label,
|
||||
Lint {
|
||||
description: description.to_owned(),
|
||||
default_severity: severity,
|
||||
warn_since: None,
|
||||
deny_since: None,
|
||||
},
|
||||
)
|
||||
})
|
||||
.collect::<HashMap<_, _>>();
|
||||
let mut lint_groups = lint_groups
|
||||
.map(|(label, lint, children)| (label, (lint, Vec::from_iter(children))))
|
||||
.collect::<HashMap<_, _>>();
|
||||
let mut lints_rustdoc = lints_rustdoc
|
||||
.map(|(label, description, severity)| {
|
||||
(
|
||||
label,
|
||||
Lint {
|
||||
description: description.to_owned(),
|
||||
default_severity: severity,
|
||||
warn_since: None,
|
||||
deny_since: None,
|
||||
},
|
||||
)
|
||||
})
|
||||
.collect::<HashMap<_, _>>();
|
||||
let mut lint_groups_rustdoc = lint_groups_rustdoc
|
||||
.map(|(label, lint, children)| (label, (lint, Vec::from_iter(children))))
|
||||
.collect::<HashMap<_, _>>();
|
||||
|
||||
for edition in Edition::iter().skip(1) {
|
||||
get_lints(
|
||||
sh,
|
||||
edition,
|
||||
&mut lints,
|
||||
&mut lint_groups,
|
||||
&mut lints_rustdoc,
|
||||
&mut lint_groups_rustdoc,
|
||||
);
|
||||
}
|
||||
|
||||
let mut lints = Vec::from_iter(lints);
|
||||
lints.sort_unstable_by(|a, b| a.0.cmp(&b.0));
|
||||
let mut lint_groups = Vec::from_iter(lint_groups);
|
||||
lint_groups.sort_unstable_by(|a, b| a.0.cmp(&b.0));
|
||||
let mut lints_rustdoc = Vec::from_iter(lints_rustdoc);
|
||||
lints_rustdoc.sort_unstable_by(|a, b| a.0.cmp(&b.0));
|
||||
let mut lint_groups_rustdoc = Vec::from_iter(lint_groups_rustdoc);
|
||||
lint_groups_rustdoc.sort_unstable_by(|a, b| a.0.cmp(&b.0));
|
||||
|
||||
buf.push_str(r#"pub const DEFAULT_LINTS: &[Lint] = &["#);
|
||||
buf.push('\n');
|
||||
|
||||
let lints = lints.lines().take_while(|l| !l.is_empty()).map(|line| {
|
||||
let (name, rest) = line.trim().split_once(char::is_whitespace).unwrap();
|
||||
let (_default_level, description) = rest.trim().split_once(char::is_whitespace).unwrap();
|
||||
(name.trim(), Cow::Borrowed(description.trim()), vec![])
|
||||
});
|
||||
let lint_groups = lint_groups.lines().take_while(|l| !l.is_empty()).map(|line| {
|
||||
let (name, lints) = line.trim().split_once(char::is_whitespace).unwrap();
|
||||
(
|
||||
name.trim(),
|
||||
format!("lint group for: {}", lints.trim()).into(),
|
||||
lints
|
||||
.split_ascii_whitespace()
|
||||
.map(|s| s.trim().trim_matches(',').replace('-', "_"))
|
||||
.collect(),
|
||||
)
|
||||
});
|
||||
|
||||
let mut lints = lints.chain(lint_groups).collect::<Vec<_>>();
|
||||
lints.sort_by(|(ident, ..), (ident2, ..)| ident.cmp(ident2));
|
||||
|
||||
for (name, description, ..) in &lints {
|
||||
push_lint_completion(buf, &name.replace('-', "_"), description);
|
||||
for (name, lint) in &lints {
|
||||
push_lint_completion(buf, name, lint);
|
||||
}
|
||||
for (name, (group, _)) in &lint_groups {
|
||||
push_lint_completion(buf, name, group);
|
||||
}
|
||||
buf.push_str("];\n\n");
|
||||
|
||||
buf.push_str(r#"pub const DEFAULT_LINT_GROUPS: &[LintGroup] = &["#);
|
||||
for (name, description, children) in &lints {
|
||||
if !children.is_empty() {
|
||||
// HACK: warnings is emitted with a general description, not with its members
|
||||
if name == &"warnings" {
|
||||
push_lint_group(buf, name, description, &Vec::new());
|
||||
continue;
|
||||
}
|
||||
push_lint_group(buf, &name.replace('-', "_"), description, children);
|
||||
for (name, (lint, children)) in &lint_groups {
|
||||
if name == "warnings" {
|
||||
continue;
|
||||
}
|
||||
push_lint_group(buf, name, lint, children);
|
||||
}
|
||||
buf.push('\n');
|
||||
buf.push_str("];\n");
|
||||
|
@ -164,37 +372,17 @@ fn generate_lint_descriptor(sh: &Shell, buf: &mut String) {
|
|||
buf.push_str(r#"pub const RUSTDOC_LINTS: &[Lint] = &["#);
|
||||
buf.push('\n');
|
||||
|
||||
let lints_rustdoc = lints_rustdoc.lines().take_while(|l| !l.is_empty()).map(|line| {
|
||||
let (name, rest) = line.trim().split_once(char::is_whitespace).unwrap();
|
||||
let (_default_level, description) = rest.trim().split_once(char::is_whitespace).unwrap();
|
||||
(name.trim(), Cow::Borrowed(description.trim()), vec![])
|
||||
});
|
||||
let lint_groups_rustdoc =
|
||||
lint_groups_rustdoc.lines().take_while(|l| !l.is_empty()).map(|line| {
|
||||
let (name, lints) = line.trim().split_once(char::is_whitespace).unwrap();
|
||||
(
|
||||
name.trim(),
|
||||
format!("lint group for: {}", lints.trim()).into(),
|
||||
lints
|
||||
.split_ascii_whitespace()
|
||||
.map(|s| s.trim().trim_matches(',').replace('-', "_"))
|
||||
.collect(),
|
||||
)
|
||||
});
|
||||
|
||||
let mut lints_rustdoc = lints_rustdoc.chain(lint_groups_rustdoc).collect::<Vec<_>>();
|
||||
lints_rustdoc.sort_by(|(ident, ..), (ident2, ..)| ident.cmp(ident2));
|
||||
|
||||
for (name, description, ..) in &lints_rustdoc {
|
||||
push_lint_completion(buf, &name.replace('-', "_"), description)
|
||||
for (name, lint) in &lints_rustdoc {
|
||||
push_lint_completion(buf, name, lint);
|
||||
}
|
||||
for (name, (group, _)) in &lint_groups_rustdoc {
|
||||
push_lint_completion(buf, name, group);
|
||||
}
|
||||
buf.push_str("];\n\n");
|
||||
|
||||
buf.push_str(r#"pub const RUSTDOC_LINT_GROUPS: &[LintGroup] = &["#);
|
||||
for (name, description, children) in &lints_rustdoc {
|
||||
if !children.is_empty() {
|
||||
push_lint_group(buf, &name.replace('-', "_"), description, children);
|
||||
}
|
||||
for (name, (lint, children)) in &lint_groups_rustdoc {
|
||||
push_lint_group(buf, name, lint, children);
|
||||
}
|
||||
buf.push('\n');
|
||||
buf.push_str("];\n");
|
||||
|
@ -228,13 +416,19 @@ fn generate_feature_descriptor(buf: &mut String, src_dir: &Path) {
|
|||
|
||||
buf.push_str(r#"pub const FEATURES: &[Lint] = &["#);
|
||||
for (feature_ident, doc) in features.into_iter() {
|
||||
push_lint_completion(buf, &feature_ident, &doc)
|
||||
let lint = Lint {
|
||||
description: doc,
|
||||
default_severity: Severity::Allow,
|
||||
warn_since: None,
|
||||
deny_since: None,
|
||||
};
|
||||
push_lint_completion(buf, &feature_ident, &lint);
|
||||
}
|
||||
buf.push('\n');
|
||||
buf.push_str("];\n");
|
||||
}
|
||||
|
||||
#[derive(Default)]
|
||||
#[derive(Debug, Default)]
|
||||
struct ClippyLint {
|
||||
help: String,
|
||||
id: String,
|
||||
|
@ -295,8 +489,14 @@ fn generate_descriptor_clippy(buf: &mut String, path: &Path) {
|
|||
buf.push('\n');
|
||||
for clippy_lint in clippy_lints.into_iter() {
|
||||
let lint_ident = format!("clippy::{}", clippy_lint.id);
|
||||
let doc = clippy_lint.help;
|
||||
push_lint_completion(buf, &lint_ident, &doc);
|
||||
let lint = Lint {
|
||||
description: clippy_lint.help,
|
||||
// Allow clippy lints by default, not all users want them.
|
||||
default_severity: Severity::Allow,
|
||||
warn_since: None,
|
||||
deny_since: None,
|
||||
};
|
||||
push_lint_completion(buf, &lint_ident, &lint);
|
||||
}
|
||||
buf.push_str("];\n");
|
||||
|
||||
|
@ -306,33 +506,59 @@ fn generate_descriptor_clippy(buf: &mut String, path: &Path) {
|
|||
if !children.is_empty() {
|
||||
let lint_ident = format!("clippy::{id}");
|
||||
let description = format!("lint group for: {}", children.join(", "));
|
||||
push_lint_group(buf, &lint_ident, &description, &children);
|
||||
let lint = Lint {
|
||||
description,
|
||||
default_severity: Severity::Allow,
|
||||
warn_since: None,
|
||||
deny_since: None,
|
||||
};
|
||||
push_lint_group(buf, &lint_ident, &lint, &children);
|
||||
}
|
||||
}
|
||||
buf.push('\n');
|
||||
buf.push_str("];\n");
|
||||
}
|
||||
|
||||
fn push_lint_completion(buf: &mut String, label: &str, description: &str) {
|
||||
fn push_lint_completion(buf: &mut String, name: &str, lint: &Lint) {
|
||||
format_to!(
|
||||
buf,
|
||||
r###" Lint {{
|
||||
label: "{}",
|
||||
description: r##"{}"##,
|
||||
}},"###,
|
||||
label,
|
||||
description,
|
||||
default_severity: {},
|
||||
warn_since: "###,
|
||||
name,
|
||||
lint.description,
|
||||
lint.default_severity,
|
||||
);
|
||||
match lint.warn_since {
|
||||
Some(edition) => format_to!(buf, "Some(Edition::Edition{edition})"),
|
||||
None => buf.push_str("None"),
|
||||
}
|
||||
format_to!(
|
||||
buf,
|
||||
r###",
|
||||
deny_since: "###
|
||||
);
|
||||
match lint.deny_since {
|
||||
Some(edition) => format_to!(buf, "Some(Edition::Edition{edition})"),
|
||||
None => buf.push_str("None"),
|
||||
}
|
||||
format_to!(
|
||||
buf,
|
||||
r###",
|
||||
}},"###
|
||||
);
|
||||
}
|
||||
|
||||
fn push_lint_group(buf: &mut String, label: &str, description: &str, children: &[String]) {
|
||||
fn push_lint_group(buf: &mut String, name: &str, lint: &Lint, children: &[String]) {
|
||||
buf.push_str(
|
||||
r###" LintGroup {
|
||||
lint:
|
||||
"###,
|
||||
);
|
||||
|
||||
push_lint_completion(buf, label, description);
|
||||
push_lint_completion(buf, name, lint);
|
||||
|
||||
let children = format!(
|
||||
"&[{}]",
|
||||
|
|
Loading…
Reference in a new issue