2020-01-09 00:52:01 -06:00
|
|
|
use crate::{EarlyContext, EarlyLintPass, LintContext};
|
2020-04-27 12:56:11 -05:00
|
|
|
use rustc_ast::{Block, StmtKind};
|
2020-01-09 04:18:47 -06:00
|
|
|
use rustc_errors::Applicability;
|
2020-02-26 21:10:42 -06:00
|
|
|
use rustc_span::Span;
|
2019-07-30 12:48:39 -05:00
|
|
|
|
|
|
|
declare_lint! {
|
2020-09-08 17:09:57 -05:00
|
|
|
/// The `redundant_semicolons` lint detects unnecessary trailing
|
|
|
|
/// semicolons.
|
|
|
|
///
|
|
|
|
/// ### Example
|
|
|
|
///
|
|
|
|
/// ```rust
|
|
|
|
/// let _ = 123;;
|
|
|
|
/// ```
|
|
|
|
///
|
|
|
|
/// {{produces}}
|
|
|
|
///
|
|
|
|
/// ### Explanation
|
|
|
|
///
|
|
|
|
/// Extra semicolons are not needed, and may be removed to avoid confusion
|
|
|
|
/// and visual clutter.
|
2020-02-26 21:10:42 -06:00
|
|
|
pub REDUNDANT_SEMICOLONS,
|
2019-07-30 12:48:39 -05:00
|
|
|
Warn,
|
|
|
|
"detects unnecessary trailing semicolons"
|
|
|
|
}
|
|
|
|
|
2020-02-26 21:10:42 -06:00
|
|
|
declare_lint_pass!(RedundantSemicolons => [REDUNDANT_SEMICOLONS]);
|
2019-07-30 12:48:39 -05:00
|
|
|
|
2020-02-26 21:10:42 -06:00
|
|
|
impl EarlyLintPass for RedundantSemicolons {
|
|
|
|
fn check_block(&mut self, cx: &EarlyContext<'_>, block: &Block) {
|
2020-11-27 08:36:59 -06:00
|
|
|
let mut after_item_stmt = false;
|
2020-02-26 21:10:42 -06:00
|
|
|
let mut seq = None;
|
|
|
|
for stmt in block.stmts.iter() {
|
|
|
|
match (&stmt.kind, &mut seq) {
|
|
|
|
(StmtKind::Empty, None) => seq = Some((stmt.span, false)),
|
|
|
|
(StmtKind::Empty, Some(seq)) => *seq = (seq.0.to(stmt.span), true),
|
2020-11-27 08:36:59 -06:00
|
|
|
(_, seq) => {
|
|
|
|
maybe_lint_redundant_semis(cx, seq, after_item_stmt);
|
|
|
|
after_item_stmt = matches!(stmt.kind, StmtKind::Item(_));
|
|
|
|
}
|
2019-07-30 12:48:39 -05:00
|
|
|
}
|
|
|
|
}
|
2020-11-27 08:36:59 -06:00
|
|
|
maybe_lint_redundant_semis(cx, &mut seq, after_item_stmt);
|
2020-02-26 21:10:42 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-11-27 08:36:59 -06:00
|
|
|
fn maybe_lint_redundant_semis(
|
|
|
|
cx: &EarlyContext<'_>,
|
|
|
|
seq: &mut Option<(Span, bool)>,
|
|
|
|
after_item_stmt: bool,
|
|
|
|
) {
|
2020-02-26 21:10:42 -06:00
|
|
|
if let Some((span, multiple)) = seq.take() {
|
2020-10-25 16:14:19 -05:00
|
|
|
// FIXME: Find a better way of ignoring the trailing
|
|
|
|
// semicolon from macro expansion
|
|
|
|
if span == rustc_span::DUMMY_SP {
|
|
|
|
return;
|
|
|
|
}
|
2020-11-27 08:36:59 -06:00
|
|
|
|
|
|
|
// FIXME: Lint on semicolons after item statements
|
|
|
|
// once doing so doesn't break bootstrapping
|
|
|
|
if after_item_stmt {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-02-26 21:10:42 -06:00
|
|
|
cx.struct_span_lint(REDUNDANT_SEMICOLONS, span, |lint| {
|
|
|
|
let (msg, rem) = if multiple {
|
|
|
|
("unnecessary trailing semicolons", "remove these semicolons")
|
|
|
|
} else {
|
|
|
|
("unnecessary trailing semicolon", "remove this semicolon")
|
|
|
|
};
|
|
|
|
lint.build(msg)
|
|
|
|
.span_suggestion(span, rem, String::new(), Applicability::MaybeIncorrect)
|
|
|
|
.emit();
|
|
|
|
});
|
2019-07-30 12:48:39 -05:00
|
|
|
}
|
|
|
|
}
|