Skip to content

Commit 71a6c7c

Browse files
committed
Auto merge of #87381 - Aaron1011:note-semi-trailing-macro, r=petrochenkov
Display an extra note for trailing semicolon lint with trailing macro Currently, we parse macros at the end of a block (e.g. `fn foo() { my_macro!() }`) as expressions, rather than statements. This means that a macro invoked in this position cannot expand to items or semicolon-terminated expressions. In the future, we might want to start parsing these kinds of macros as statements. This would make expansion more 'token-based' (i.e. macro expansion behaves (almost) as if you just textually replaced the macro invocation with its output). However, this is a breaking change (see PR #78991), so it will require further discussion. Since the current behavior will not be changing any time soon, we need to address the interaction with the `SEMICOLON_IN_EXPRESSIONS_FROM_MACROS` lint. Since we are parsing the result of macro expansion as an expression, we will emit a lint if there's a trailing semicolon in the macro output. However, this results in a somewhat confusing message for users, since it visually looks like there should be no problem with having a semicolon at the end of a block (e.g. `fn foo() { my_macro!() }` => `fn foo() { produced_expr; }`) To help reduce confusion, this commit adds a note explaining that the macro is being interpreted as an expression. Additionally, we suggest adding a semicolon after the macro *invocation* - this will cause us to parse the macro call as a statement. We do *not* use a structured suggestion for this, since the user may actually want to remove the semicolon from the macro definition (allowing the block to evaluate to the expression produced by the macro).
2 parents f63ec77 + 0df5ac8 commit 71a6c7c

File tree

6 files changed

+41
-5
lines changed

6 files changed

+41
-5
lines changed

compiler/rustc_expand/src/base.rs

+2
Original file line numberDiff line numberDiff line change
@@ -929,6 +929,7 @@ pub struct ExpansionData {
929929
pub prior_type_ascription: Option<(Span, bool)>,
930930
/// Some parent node that is close to this macro call
931931
pub lint_node_id: NodeId,
932+
pub is_trailing_mac: bool,
932933
}
933934

934935
type OnExternModLoaded<'a> =
@@ -979,6 +980,7 @@ impl<'a> ExtCtxt<'a> {
979980
dir_ownership: DirOwnership::Owned { relative: None },
980981
prior_type_ascription: None,
981982
lint_node_id: ast::CRATE_NODE_ID,
983+
is_trailing_mac: false,
982984
},
983985
force_mode: false,
984986
expansions: FxHashMap::default(),

compiler/rustc_expand/src/expand.rs

+18-2
Original file line numberDiff line numberDiff line change
@@ -1328,14 +1328,30 @@ impl<'a, 'b> MutVisitor for InvocationCollector<'a, 'b> {
13281328
return placeholder;
13291329
}
13301330

1331+
// The only way that we can end up with a `MacCall` expression statement,
1332+
// (as opposed to a `StmtKind::MacCall`) is if we have a macro as the
1333+
// traiing expression in a block (e.g. `fn foo() { my_macro!() }`).
1334+
// Record this information, so that we can report a more specific
1335+
// `SEMICOLON_IN_EXPRESSIONS_FROM_MACROS` lint if needed.
1336+
// See #78991 for an investigation of treating macros in this position
1337+
// as statements, rather than expressions, during parsing.
1338+
if let StmtKind::Expr(expr) = &stmt.kind {
1339+
if matches!(**expr, ast::Expr { kind: ast::ExprKind::MacCall(..), .. }) {
1340+
self.cx.current_expansion.is_trailing_mac = true;
1341+
}
1342+
}
1343+
13311344
// The placeholder expander gives ids to statements, so we avoid folding the id here.
13321345
// We don't use `assign_id!` - it will be called when we visit statement's contents
13331346
// (e.g. an expression, item, or local)
13341347
let ast::Stmt { id, kind, span } = stmt;
1335-
noop_flat_map_stmt_kind(kind, self)
1348+
let res = noop_flat_map_stmt_kind(kind, self)
13361349
.into_iter()
13371350
.map(|kind| ast::Stmt { id, kind, span })
1338-
.collect()
1351+
.collect();
1352+
1353+
self.cx.current_expansion.is_trailing_mac = false;
1354+
res
13391355
}
13401356

13411357
fn visit_block(&mut self, block: &mut P<Block>) {

compiler/rustc_expand/src/mbe/macro_rules.rs

+12-3
Original file line numberDiff line numberDiff line change
@@ -43,6 +43,7 @@ crate struct ParserAnyMacro<'a> {
4343
/// The ident of the macro we're parsing
4444
macro_ident: Ident,
4545
lint_node_id: NodeId,
46+
is_trailing_mac: bool,
4647
arm_span: Span,
4748
}
4849

@@ -116,8 +117,14 @@ fn emit_frag_parse_err(
116117

117118
impl<'a> ParserAnyMacro<'a> {
118119
crate fn make(mut self: Box<ParserAnyMacro<'a>>, kind: AstFragmentKind) -> AstFragment {
119-
let ParserAnyMacro { site_span, macro_ident, ref mut parser, lint_node_id, arm_span } =
120-
*self;
120+
let ParserAnyMacro {
121+
site_span,
122+
macro_ident,
123+
ref mut parser,
124+
lint_node_id,
125+
arm_span,
126+
is_trailing_mac,
127+
} = *self;
121128
let snapshot = &mut parser.clone();
122129
let fragment = match parse_ast_fragment(parser, kind) {
123130
Ok(f) => f,
@@ -131,11 +138,12 @@ impl<'a> ParserAnyMacro<'a> {
131138
// `macro_rules! m { () => { panic!(); } }` isn't parsed by `.parse_expr()`,
132139
// but `m!()` is allowed in expression positions (cf. issue #34706).
133140
if kind == AstFragmentKind::Expr && parser.token == token::Semi {
134-
parser.sess.buffer_lint(
141+
parser.sess.buffer_lint_with_diagnostic(
135142
SEMICOLON_IN_EXPRESSIONS_FROM_MACROS,
136143
parser.token.span,
137144
lint_node_id,
138145
"trailing semicolon in macro used in expression position",
146+
BuiltinLintDiagnostics::TrailingMacro(is_trailing_mac, macro_ident),
139147
);
140148
parser.bump();
141149
}
@@ -301,6 +309,7 @@ fn generic_extension<'cx>(
301309
site_span: sp,
302310
macro_ident: name,
303311
lint_node_id: cx.current_expansion.lint_node_id,
312+
is_trailing_mac: cx.current_expansion.is_trailing_mac,
304313
arm_span,
305314
});
306315
}

compiler/rustc_lint/src/context.rs

+6
Original file line numberDiff line numberDiff line change
@@ -744,6 +744,12 @@ pub trait LintContext: Sized {
744744
&format!("the built-in attribute `{attr_name}` will be ignored, since it's applied to the macro invocation `{macro_name}`")
745745
);
746746
}
747+
BuiltinLintDiagnostics::TrailingMacro(is_trailing, name) => {
748+
if is_trailing {
749+
db.note("macro invocations at the end of a block are treated as expressions");
750+
db.note(&format!("to ignore the value produced by the macro, add a semicolon after the invocation of `{name}`"));
751+
}
752+
}
747753
}
748754
// Rewrap `db`, and pass control to the user.
749755
decorate(LintDiagnosticBuilder::new(db));

compiler/rustc_lint_defs/src/lib.rs

+1
Original file line numberDiff line numberDiff line change
@@ -303,6 +303,7 @@ pub enum BuiltinLintDiagnostics {
303303
ProcMacroBackCompat(String),
304304
OrPatternsBackCompat(Span, String),
305305
ReservedPrefix(Span),
306+
TrailingMacro(bool, Ident),
306307
}
307308

308309
/// Lints that are buffered up early on in the `Session` before the

src/test/ui/lint/semicolon-in-expressions-from-macros/semicolon-in-expressions-from-macros.stderr

+2
Original file line numberDiff line numberDiff line change
@@ -14,6 +14,8 @@ LL | #![warn(semicolon_in_expressions_from_macros)]
1414
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
1515
= warning: this was previously accepted by the compiler but is being phased out; it will become a hard error in a future release!
1616
= note: for more information, see issue #79813 <https://github.com/rust-lang/rust/issues/79813>
17+
= note: macro invocations at the end of a block are treated as expressions
18+
= note: to ignore the value produced by the macro, add a semicolon after the invocation of `foo`
1719
= note: this warning originates in the macro `foo` (in Nightly builds, run with -Z macro-backtrace for more info)
1820

1921
warning: trailing semicolon in macro used in expression position

0 commit comments

Comments
 (0)