From 8544db0faa9e0f7a70323ad5f3e75358bba6820d Mon Sep 17 00:00:00 2001 From: Esteban Küber Date: Tue, 23 Oct 2018 10:07:11 -0700 Subject: Add macro call span when lacking any other span in diagnostic --- src/libsyntax/ext/tt/macro_rules.rs | 7 ++++++- src/libsyntax_pos/lib.rs | 11 +++++++++++ src/test/ui/macros/macro-in-expression-context-2.rs | 7 +++++++ src/test/ui/macros/macro-in-expression-context-2.stderr | 8 ++++++++ 4 files changed, 32 insertions(+), 1 deletion(-) create mode 100644 src/test/ui/macros/macro-in-expression-context-2.rs create mode 100644 src/test/ui/macros/macro-in-expression-context-2.stderr (limited to 'src') diff --git a/src/libsyntax/ext/tt/macro_rules.rs b/src/libsyntax/ext/tt/macro_rules.rs index 87ade278c68..b70a7989275 100644 --- a/src/libsyntax/ext/tt/macro_rules.rs +++ b/src/libsyntax/ext/tt/macro_rules.rs @@ -50,7 +50,12 @@ pub struct ParserAnyMacro<'a> { impl<'a> ParserAnyMacro<'a> { pub fn make(mut self: Box>, kind: AstFragmentKind) -> AstFragment { let ParserAnyMacro { site_span, macro_ident, ref mut parser } = *self; - let fragment = panictry!(parser.parse_ast_fragment(kind, true)); + let fragment = panictry!(parser.parse_ast_fragment(kind, true).map_err(|mut e| { + if e.span.is_dummy() { // Get around lack of span in error (#30128) + e.set_span(site_span); + } + e + })); // We allow semicolons at the end of expressions -- e.g. the semicolon in // `macro_rules! m { () => { panic!(); } }` isn't parsed by `.parse_expr()`, diff --git a/src/libsyntax_pos/lib.rs b/src/libsyntax_pos/lib.rs index 45eaf1d3190..639155636ed 100644 --- a/src/libsyntax_pos/lib.rs +++ b/src/libsyntax_pos/lib.rs @@ -612,6 +612,17 @@ impl MultiSpan { &self.primary_spans } + /// Returns `true` if this contains only a dummy primary span with any hygienic context. + pub fn is_dummy(&self) -> bool { + let mut is_dummy = true; + for span in &self.primary_spans { + if !span.is_dummy() { + is_dummy = false; + } + } + is_dummy + } + /// Replaces all occurrences of one Span with another. Used to move Spans in areas that don't /// display well (like std macros). Returns true if replacements occurred. pub fn replace(&mut self, before: Span, after: Span) -> bool { diff --git a/src/test/ui/macros/macro-in-expression-context-2.rs b/src/test/ui/macros/macro-in-expression-context-2.rs new file mode 100644 index 00000000000..cf8572aefa2 --- /dev/null +++ b/src/test/ui/macros/macro-in-expression-context-2.rs @@ -0,0 +1,7 @@ +macro_rules! empty { () => () } + +fn main() { + match 42 { + _ => { empty!() } + }; +} diff --git a/src/test/ui/macros/macro-in-expression-context-2.stderr b/src/test/ui/macros/macro-in-expression-context-2.stderr new file mode 100644 index 00000000000..80d5dbd66cc --- /dev/null +++ b/src/test/ui/macros/macro-in-expression-context-2.stderr @@ -0,0 +1,8 @@ +error: expected expression, found `` + --> $DIR/macro-in-expression-context-2.rs:5:16 + | +LL | _ => { empty!() } + | ^^^^^^^^ + +error: aborting due to previous error + -- cgit 1.4.1-3-g733a5