1
1
use clippy_utils:: diagnostics:: span_lint_and_help;
2
- use clippy_utils:: source :: { indent_of , snippet , snippet_block , snippet_with_context } ;
3
- use clippy_utils:: { higher , is_from_proc_macro } ;
2
+ use clippy_utils:: higher ;
3
+ use clippy_utils:: source :: { indent_of , snippet_block , snippet_with_context } ;
4
4
use rustc_ast:: Label ;
5
5
use rustc_errors:: Applicability ;
6
6
use rustc_hir:: { Block , Expr , ExprKind , LoopSource , StmtKind } ;
7
7
use rustc_lint:: { LateContext , LateLintPass , LintContext } ;
8
8
use rustc_session:: declare_lint_pass;
9
- use rustc_span:: Span ;
9
+ use rustc_span:: { ExpnKind , Span } ;
10
10
11
11
declare_clippy_lint ! {
12
12
/// ### What it does
@@ -132,7 +132,9 @@ declare_lint_pass!(NeedlessContinue => [NEEDLESS_CONTINUE]);
132
132
133
133
impl < ' tcx > LateLintPass < ' tcx > for NeedlessContinue {
134
134
fn check_expr ( & mut self , cx : & LateContext < ' tcx > , expr : & ' tcx Expr < ' tcx > ) {
135
- if !expr. span . in_external_macro ( cx. sess ( ) . source_map ( ) ) && !is_from_proc_macro ( cx, expr) {
135
+ // We cannot use `from_expansion` because for loops, while loops and while let loops are desugared
136
+ // into `loop` expressions.
137
+ if !matches ! ( expr. span. ctxt( ) . outer_expn_data( ) . kind, ExpnKind :: Macro ( ..) ) {
136
138
check_and_warn ( cx, expr) ;
137
139
}
138
140
}
@@ -193,16 +195,16 @@ impl<'tcx> LateLintPass<'tcx> for NeedlessContinue {
193
195
/// - The expression node is a block with the first statement being a `continue`.
194
196
fn needless_continue_in_else ( else_expr : & Expr < ' _ > , label : Option < & Label > ) -> bool {
195
197
match else_expr. kind {
196
- ExprKind :: Block ( ref else_block, _) => is_first_block_stmt_continue ( else_block, label) ,
198
+ ExprKind :: Block ( else_block, _) => is_first_block_stmt_continue ( else_block, label) ,
197
199
ExprKind :: Continue ( l) => compare_labels ( label, l. label . as_ref ( ) ) ,
198
200
_ => false ,
199
201
}
200
202
}
201
203
202
204
fn is_first_block_stmt_continue ( block : & Block < ' _ > , label : Option < & Label > ) -> bool {
203
205
block. stmts . first ( ) . is_some_and ( |stmt| match stmt. kind {
204
- StmtKind :: Semi ( ref e) | StmtKind :: Expr ( ref e) => {
205
- if let ExprKind :: Continue ( ref l) = e. kind {
206
+ StmtKind :: Semi ( e) | StmtKind :: Expr ( e) => {
207
+ if let ExprKind :: Continue ( l) = e. kind {
206
208
compare_labels ( label, l. label . as_ref ( ) )
207
209
} else {
208
210
false
@@ -225,10 +227,10 @@ fn compare_labels(loop_label: Option<&Label>, continue_label: Option<&Label>) ->
225
227
}
226
228
227
229
/// If `expr` is a loop expression (while/while let/for/loop), calls `func` with
228
- /// the AST object representing the loop block of `expr`.
229
- fn with_loop_block < ' tcx , F > ( expr : & Expr < ' tcx > , mut func : F )
230
+ /// the HIR object representing the loop block of `expr`.
231
+ fn with_loop_block < F > ( expr : & Expr < ' _ > , mut func : F )
230
232
where
231
- F : FnMut ( & Block < ' tcx > , Option < & Label > ) ,
233
+ F : FnMut ( & Block < ' _ > , Option < & Label > ) ,
232
234
{
233
235
if let Some ( higher:: ForLoop { body, label, .. } ) = higher:: ForLoop :: hir ( expr)
234
236
&& let ExprKind :: Block ( block, _) = & body. kind
@@ -264,9 +266,9 @@ where
264
266
/// - The `if` condition expression,
265
267
/// - The `then` block, and
266
268
/// - The `else` expression.
267
- fn with_if_expr < ' tcx , F > ( expr : & Expr < ' tcx > , mut func : F )
269
+ fn with_if_expr < F > ( expr : & Expr < ' _ > , mut func : F )
268
270
where
269
- F : FnMut ( & Expr < ' tcx > , & Expr < ' tcx > , & Block < ' tcx > , & Expr < ' tcx > ) ,
271
+ F : FnMut ( & Expr < ' _ > , & Expr < ' _ > , & Block < ' _ > , & Expr < ' _ > ) ,
270
272
{
271
273
if let Some ( higher:: If {
272
274
cond,
@@ -288,20 +290,20 @@ enum LintType {
288
290
289
291
/// Data we pass around for construction of help messages.
290
292
#[ derive( Debug ) ]
291
- struct LintData < ' tcx > {
293
+ struct LintData < ' hir > {
292
294
/// The `if` expression encountered in the above loop.
293
- if_expr : & ' tcx Expr < ' tcx > ,
295
+ if_expr : & ' hir Expr < ' hir > ,
294
296
/// The condition expression for the above `if`.
295
- if_cond : & ' tcx Expr < ' tcx > ,
297
+ if_cond : & ' hir Expr < ' hir > ,
296
298
/// The `then` block of the `if` statement.
297
- if_block : & ' tcx Block < ' tcx > ,
299
+ if_block : & ' hir Block < ' hir > ,
298
300
/// The `else` block of the `if` statement.
299
301
/// Note that we only work with `if` exprs that have an `else` branch.
300
- else_expr : & ' tcx Expr < ' tcx > ,
302
+ else_expr : & ' hir Expr < ' hir > ,
301
303
/// The 0-based index of the `if` statement in the containing loop block.
302
304
stmt_idx : Option < usize > ,
303
305
/// The statements of the loop block.
304
- loop_block : & ' tcx Block < ' tcx > ,
306
+ loop_block : & ' hir Block < ' hir > ,
305
307
}
306
308
307
309
const MSG_REDUNDANT_CONTINUE_EXPRESSION : & str = "this `continue` expression is redundant" ;
@@ -366,7 +368,14 @@ fn suggestion_snippet_for_continue_inside_if(cx: &LateContext<'_>, data: &LintDa
366
368
}
367
369
368
370
fn suggestion_snippet_for_continue_inside_else ( cx : & LateContext < ' _ > , data : & LintData < ' _ > ) -> String {
369
- let cond_code = snippet ( cx, data. if_cond . span , ".." ) ;
371
+ let mut applicability = Applicability :: MachineApplicable ;
372
+ let ( cond_code, _) = snippet_with_context (
373
+ cx,
374
+ data. if_cond . span ,
375
+ data. if_expr . span . ctxt ( ) ,
376
+ ".." ,
377
+ & mut applicability,
378
+ ) ;
370
379
371
380
// Region B
372
381
let block_code = erode_from_back ( & snippet_block ( cx, data. if_block . span , ".." , Some ( data. if_expr . span ) ) ) ;
@@ -402,7 +411,7 @@ fn suggestion_snippet_for_continue_inside_else(cx: &LateContext<'_>, data: &Lint
402
411
}
403
412
lines. join ( "\n " )
404
413
} else {
405
- "" . to_string ( )
414
+ String :: new ( )
406
415
} ;
407
416
408
417
let indent_if = indent_of ( cx, data. if_expr . span ) . unwrap_or ( 0 ) ;
@@ -417,7 +426,7 @@ fn check_last_stmt_in_expr<F>(cx: &LateContext<'_>, inner_expr: &Expr<'_>, func:
417
426
where
418
427
F : Fn ( Option < & Label > , Span ) ,
419
428
{
420
- match & inner_expr. kind {
429
+ match inner_expr. kind {
421
430
ExprKind :: Continue ( continue_label) => {
422
431
func ( continue_label. label . as_ref ( ) , inner_expr. span ) ;
423
432
} ,
@@ -432,7 +441,7 @@ where
432
441
if !match_ty. is_unit ( ) && !match_ty. is_never ( ) {
433
442
return ;
434
443
}
435
- for arm in arms. iter ( ) {
444
+ for arm in arms {
436
445
check_last_stmt_in_expr ( cx, arm. body , func) ;
437
446
}
438
447
} ,
0 commit comments