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:
bors[bot] 2020-12-01 12:27:52 +00:00 committed by GitHub
commit eb7f969510
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
4 changed files with 89 additions and 29 deletions

View file

@ -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, &macro_call); let macro_call = InFile::new(self.current_file_id, &macro_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) {

View file

@ -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());
}
} }
} }
} }

View file

@ -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

View file

@ -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 =