mirror of
https://github.com/rust-lang/rust-analyzer
synced 2025-01-15 14:43:58 +00:00
Merge #6683
6683: Emit macro diagnostics when lowering bodies r=matklad a=jonas-schievink Changes `Expander::enter_expand` to return an `ExpandResult`, and adds any contained errors to the body diagnostic list. Co-authored-by: Jonas Schievink <jonasschievink@gmail.com>
This commit is contained in:
commit
eb7f969510
4 changed files with 89 additions and 29 deletions
|
@ -14,8 +14,8 @@ use cfg::CfgOptions;
|
||||||
use drop_bomb::DropBomb;
|
use drop_bomb::DropBomb;
|
||||||
use either::Either;
|
use either::Either;
|
||||||
use hir_expand::{
|
use hir_expand::{
|
||||||
ast_id_map::AstIdMap, diagnostics::DiagnosticSink, hygiene::Hygiene, AstId, HirFileId, InFile,
|
ast_id_map::AstIdMap, diagnostics::DiagnosticSink, hygiene::Hygiene, AstId, ExpandResult,
|
||||||
MacroDefId,
|
HirFileId, InFile, MacroDefId,
|
||||||
};
|
};
|
||||||
use rustc_hash::FxHashMap;
|
use rustc_hash::FxHashMap;
|
||||||
use syntax::{ast, AstNode, AstPtr};
|
use syntax::{ast, AstNode, AstPtr};
|
||||||
|
@ -102,11 +102,11 @@ impl Expander {
|
||||||
db: &dyn DefDatabase,
|
db: &dyn DefDatabase,
|
||||||
local_scope: Option<&ItemScope>,
|
local_scope: Option<&ItemScope>,
|
||||||
macro_call: ast::MacroCall,
|
macro_call: ast::MacroCall,
|
||||||
) -> Option<(Mark, T)> {
|
) -> ExpandResult<Option<(Mark, T)>> {
|
||||||
self.recursion_limit += 1;
|
self.recursion_limit += 1;
|
||||||
if self.recursion_limit > EXPANSION_RECURSION_LIMIT {
|
if self.recursion_limit > EXPANSION_RECURSION_LIMIT {
|
||||||
mark::hit!(your_stack_belongs_to_me);
|
mark::hit!(your_stack_belongs_to_me);
|
||||||
return None;
|
return ExpandResult::str_err("reached recursion limit during macro expansion".into());
|
||||||
}
|
}
|
||||||
|
|
||||||
let macro_call = InFile::new(self.current_file_id, ¯o_call);
|
let macro_call = InFile::new(self.current_file_id, ¯o_call);
|
||||||
|
@ -120,28 +120,55 @@ impl Expander {
|
||||||
self.resolve_path_as_macro(db, &path)
|
self.resolve_path_as_macro(db, &path)
|
||||||
};
|
};
|
||||||
|
|
||||||
if let Some(call_id) = macro_call.as_call_id(db, self.crate_def_map.krate, resolver) {
|
let call_id = match macro_call.as_call_id(db, self.crate_def_map.krate, resolver) {
|
||||||
let file_id = call_id.as_file();
|
Some(it) => it,
|
||||||
if let Some(node) = db.parse_or_expand(file_id) {
|
None => {
|
||||||
if let Some(expr) = T::cast(node) {
|
// FIXME: this can mean other things too, but `as_call_id` doesn't provide enough
|
||||||
log::debug!("macro expansion {:#?}", expr.syntax());
|
// info.
|
||||||
|
return ExpandResult::only_err(mbe::ExpandError::Other(
|
||||||
let mark = Mark {
|
"failed to parse or resolve macro invocation".into(),
|
||||||
file_id: self.current_file_id,
|
));
|
||||||
ast_id_map: mem::take(&mut self.ast_id_map),
|
|
||||||
bomb: DropBomb::new("expansion mark dropped"),
|
|
||||||
};
|
|
||||||
self.cfg_expander.hygiene = Hygiene::new(db.upcast(), file_id);
|
|
||||||
self.current_file_id = file_id;
|
|
||||||
self.ast_id_map = db.ast_id_map(file_id);
|
|
||||||
return Some((mark, expr));
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
};
|
||||||
|
|
||||||
// FIXME: Instead of just dropping the error from expansion
|
let err = db.macro_expand_error(call_id);
|
||||||
// report it
|
|
||||||
None
|
let file_id = call_id.as_file();
|
||||||
|
|
||||||
|
let raw_node = match db.parse_or_expand(file_id) {
|
||||||
|
Some(it) => it,
|
||||||
|
None => {
|
||||||
|
// Only `None` if the macro expansion produced no usable AST.
|
||||||
|
if err.is_none() {
|
||||||
|
log::warn!("no error despite `parse_or_expand` failing");
|
||||||
|
}
|
||||||
|
|
||||||
|
return ExpandResult::only_err(err.unwrap_or_else(|| {
|
||||||
|
mbe::ExpandError::Other("failed to parse macro invocation".into())
|
||||||
|
}));
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
let node = match T::cast(raw_node) {
|
||||||
|
Some(it) => it,
|
||||||
|
None => {
|
||||||
|
// This can happen without being an error, so only forward previous errors.
|
||||||
|
return ExpandResult { value: None, err };
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
log::debug!("macro expansion {:#?}", node.syntax());
|
||||||
|
|
||||||
|
let mark = Mark {
|
||||||
|
file_id: self.current_file_id,
|
||||||
|
ast_id_map: mem::take(&mut self.ast_id_map),
|
||||||
|
bomb: DropBomb::new("expansion mark dropped"),
|
||||||
|
};
|
||||||
|
self.cfg_expander.hygiene = Hygiene::new(db.upcast(), file_id);
|
||||||
|
self.current_file_id = file_id;
|
||||||
|
self.ast_id_map = db.ast_id_map(file_id);
|
||||||
|
|
||||||
|
ExpandResult { value: Some((mark, node)), err }
|
||||||
}
|
}
|
||||||
|
|
||||||
pub(crate) fn exit(&mut self, db: &dyn DefDatabase, mut mark: Mark) {
|
pub(crate) fn exit(&mut self, db: &dyn DefDatabase, mut mark: Mark) {
|
||||||
|
|
|
@ -2,11 +2,13 @@
|
||||||
|
|
||||||
use hir_expand::diagnostics::DiagnosticSink;
|
use hir_expand::diagnostics::DiagnosticSink;
|
||||||
|
|
||||||
use crate::diagnostics::InactiveCode;
|
use crate::diagnostics::{InactiveCode, MacroError, UnresolvedProcMacro};
|
||||||
|
|
||||||
#[derive(Debug, Eq, PartialEq)]
|
#[derive(Debug, Eq, PartialEq)]
|
||||||
pub(crate) enum BodyDiagnostic {
|
pub(crate) enum BodyDiagnostic {
|
||||||
InactiveCode(InactiveCode),
|
InactiveCode(InactiveCode),
|
||||||
|
MacroError(MacroError),
|
||||||
|
UnresolvedProcMacro(UnresolvedProcMacro),
|
||||||
}
|
}
|
||||||
|
|
||||||
impl BodyDiagnostic {
|
impl BodyDiagnostic {
|
||||||
|
@ -15,6 +17,12 @@ impl BodyDiagnostic {
|
||||||
BodyDiagnostic::InactiveCode(diag) => {
|
BodyDiagnostic::InactiveCode(diag) => {
|
||||||
sink.push(diag.clone());
|
sink.push(diag.clone());
|
||||||
}
|
}
|
||||||
|
BodyDiagnostic::MacroError(diag) => {
|
||||||
|
sink.push(diag.clone());
|
||||||
|
}
|
||||||
|
BodyDiagnostic::UnresolvedProcMacro(diag) => {
|
||||||
|
sink.push(diag.clone());
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -8,7 +8,7 @@ use either::Either;
|
||||||
use hir_expand::{
|
use hir_expand::{
|
||||||
hygiene::Hygiene,
|
hygiene::Hygiene,
|
||||||
name::{name, AsName, Name},
|
name::{name, AsName, Name},
|
||||||
HirFileId, MacroDefId, MacroDefKind,
|
ExpandError, HirFileId, MacroDefId, MacroDefKind,
|
||||||
};
|
};
|
||||||
use rustc_hash::FxHashMap;
|
use rustc_hash::FxHashMap;
|
||||||
use syntax::{
|
use syntax::{
|
||||||
|
@ -25,7 +25,7 @@ use crate::{
|
||||||
body::{Body, BodySourceMap, Expander, PatPtr, SyntheticSyntax},
|
body::{Body, BodySourceMap, Expander, PatPtr, SyntheticSyntax},
|
||||||
builtin_type::{BuiltinFloat, BuiltinInt},
|
builtin_type::{BuiltinFloat, BuiltinInt},
|
||||||
db::DefDatabase,
|
db::DefDatabase,
|
||||||
diagnostics::InactiveCode,
|
diagnostics::{InactiveCode, MacroError, UnresolvedProcMacro},
|
||||||
expr::{
|
expr::{
|
||||||
dummy_expr_id, ArithOp, Array, BinaryOp, BindingAnnotation, CmpOp, Expr, ExprId, Literal,
|
dummy_expr_id, ArithOp, Array, BinaryOp, BindingAnnotation, CmpOp, Expr, ExprId, Literal,
|
||||||
LogicOp, MatchArm, Ordering, Pat, PatId, RecordFieldPat, RecordLitField, Statement,
|
LogicOp, MatchArm, Ordering, Pat, PatId, RecordFieldPat, RecordLitField, Statement,
|
||||||
|
@ -561,7 +561,32 @@ impl ExprCollector<'_> {
|
||||||
self.alloc_expr(Expr::Missing, syntax_ptr)
|
self.alloc_expr(Expr::Missing, syntax_ptr)
|
||||||
} else {
|
} else {
|
||||||
let macro_call = self.expander.to_source(AstPtr::new(&e));
|
let macro_call = self.expander.to_source(AstPtr::new(&e));
|
||||||
match self.expander.enter_expand(self.db, Some(&self.body.item_scope), e) {
|
let res = self.expander.enter_expand(self.db, Some(&self.body.item_scope), e);
|
||||||
|
|
||||||
|
match res.err {
|
||||||
|
Some(ExpandError::UnresolvedProcMacro) => {
|
||||||
|
self.source_map.diagnostics.push(BodyDiagnostic::UnresolvedProcMacro(
|
||||||
|
UnresolvedProcMacro {
|
||||||
|
file: self.expander.current_file_id,
|
||||||
|
node: syntax_ptr.clone().into(),
|
||||||
|
precise_location: None,
|
||||||
|
macro_name: None,
|
||||||
|
},
|
||||||
|
));
|
||||||
|
}
|
||||||
|
Some(err) => {
|
||||||
|
self.source_map.diagnostics.push(BodyDiagnostic::MacroError(
|
||||||
|
MacroError {
|
||||||
|
file: self.expander.current_file_id,
|
||||||
|
node: syntax_ptr.clone().into(),
|
||||||
|
message: err.to_string(),
|
||||||
|
},
|
||||||
|
));
|
||||||
|
}
|
||||||
|
None => {}
|
||||||
|
}
|
||||||
|
|
||||||
|
match res.value {
|
||||||
Some((mark, expansion)) => {
|
Some((mark, expansion)) => {
|
||||||
self.source_map
|
self.source_map
|
||||||
.expansions
|
.expansions
|
||||||
|
|
|
@ -257,7 +257,7 @@ fn collect_items(
|
||||||
let root = db.parse_or_expand(file_id).unwrap();
|
let root = db.parse_or_expand(file_id).unwrap();
|
||||||
let call = ast_id_map.get(call.ast_id).to_node(&root);
|
let call = ast_id_map.get(call.ast_id).to_node(&root);
|
||||||
|
|
||||||
if let Some((mark, mac)) = expander.enter_expand(db, None, call) {
|
if let Some((mark, mac)) = expander.enter_expand(db, None, call).value {
|
||||||
let src: InFile<ast::MacroItems> = expander.to_source(mac);
|
let src: InFile<ast::MacroItems> = expander.to_source(mac);
|
||||||
let item_tree = db.item_tree(src.file_id);
|
let item_tree = db.item_tree(src.file_id);
|
||||||
let iter =
|
let iter =
|
||||||
|
|
Loading…
Reference in a new issue