Skip to content

Commit 65eb1ec

Browse files
committed
remove the semicolon for builtin macro call statements in statement_outside_block
The expansion of `asm!()` and `line!()` is not marked as from an expansion, in which case `SourceMap::stmt_span` returns the input span unchanged. So instead of using `stmt_span`, use `mac_call_stmt_semi_span` directly
1 parent 4de65a1 commit 65eb1ec

4 files changed

+60
-13
lines changed

clippy_lints/src/semicolon_block.rs

+15-12
Original file line numberDiff line numberDiff line change
@@ -101,17 +101,21 @@ impl SemicolonBlock {
101101
);
102102
}
103103

104-
fn semicolon_outside_block(
105-
&self,
106-
cx: &LateContext<'_>,
107-
block: &Block<'_>,
108-
tail_stmt_expr: &Expr<'_>,
109-
semi_span: Span,
110-
) {
104+
fn semicolon_outside_block(&self, cx: &LateContext<'_>, block: &Block<'_>, tail_stmt_expr: &Expr<'_>) {
111105
let insert_span = block.span.with_lo(block.span.hi());
112-
// account for macro calls
113-
let semi_span = cx.sess().source_map().stmt_span(semi_span, block.span);
114-
let remove_span = semi_span.with_lo(tail_stmt_expr.span.source_callsite().hi());
106+
107+
// For macro call semicolon statements (`mac!();`), the statement's span does not actually
108+
// include the semicolon itself, so use `mac_call_stmt_semi_span`, which finds the semicolon
109+
// based on a source snippet.
110+
// (Does not use `stmt_span` as that requires `.from_expansion()` to return true,
111+
// which is not the case for e.g. `line!();` and `asm!();`)
112+
let Some(remove_span) = cx
113+
.sess()
114+
.source_map()
115+
.mac_call_stmt_semi_span(tail_stmt_expr.span.source_callsite())
116+
else {
117+
return;
118+
};
115119

116120
if self.semicolon_outside_block_ignore_multiline && get_line(cx, remove_span) != get_line(cx, insert_span) {
117121
return;
@@ -150,13 +154,12 @@ impl LateLintPass<'_> for SemicolonBlock {
150154
};
151155
let &Stmt {
152156
kind: StmtKind::Semi(expr),
153-
span,
154157
..
155158
} = stmt
156159
else {
157160
return;
158161
};
159-
self.semicolon_outside_block(cx, block, expr, span);
162+
self.semicolon_outside_block(cx, block, expr);
160163
},
161164
StmtKind::Semi(Expr {
162165
kind: ExprKind::Block(block @ Block { expr: Some(tail), .. }, _),

tests/ui/semicolon_outside_block.fixed

+8
Original file line numberDiff line numberDiff line change
@@ -80,5 +80,13 @@ fn main() {
8080

8181
{ unit_fn_block(); };
8282

83+
unsafe {
84+
std::arch::asm!("")
85+
};
86+
87+
{
88+
line!()
89+
};
90+
8391
unit_fn_block()
8492
}

tests/ui/semicolon_outside_block.rs

+8
Original file line numberDiff line numberDiff line change
@@ -80,5 +80,13 @@ fn main() {
8080

8181
{ unit_fn_block(); };
8282

83+
unsafe {
84+
std::arch::asm!("");
85+
}
86+
87+
{
88+
line!();
89+
}
90+
8391
unit_fn_block()
8492
}

tests/ui/semicolon_outside_block.stderr

+29-1
Original file line numberDiff line numberDiff line change
@@ -51,5 +51,33 @@ LL - { m!(()); }
5151
LL + { m!(()) };
5252
|
5353

54-
error: aborting due to 4 previous errors
54+
error: consider moving the `;` outside the block for consistent formatting
55+
--> tests/ui/semicolon_outside_block.rs:83:5
56+
|
57+
LL | / unsafe {
58+
LL | | std::arch::asm!("");
59+
LL | | }
60+
| |_____^
61+
|
62+
help: put the `;` here
63+
|
64+
LL ~ std::arch::asm!("")
65+
LL ~ };
66+
|
67+
68+
error: consider moving the `;` outside the block for consistent formatting
69+
--> tests/ui/semicolon_outside_block.rs:87:5
70+
|
71+
LL | / {
72+
LL | | line!();
73+
LL | | }
74+
| |_____^
75+
|
76+
help: put the `;` here
77+
|
78+
LL ~ line!()
79+
LL ~ };
80+
|
81+
82+
error: aborting due to 6 previous errors
5583

0 commit comments

Comments
 (0)