Skip to content

Commit f85f7af

Browse files
Adds expr_2024 migration lit
This is adding a migration lint for the current (in the 2021 edition and previous) to move expr to expr_2021 from expr Co-Developed-by: Eric Holk Signed-off-by: Vincenzo Palazzo <vincenzopalazzodev@gmail.com>
1 parent 21e6de7 commit f85f7af

File tree

4 files changed

+98
-0
lines changed

4 files changed

+98
-0
lines changed

compiler/rustc_lint/messages.ftl

+4
Original file line numberDiff line numberDiff line change
@@ -889,3 +889,7 @@ lint_variant_size_differences =
889889
890890
lint_wasm_c_abi =
891891
older versions of the `wasm-bindgen` crate will be incompatible with future versions of Rust; please update to `wasm-bindgen` v0.2.88
892+
893+
lint_macro_expr_fragment_specifier_2024_migration =
894+
the `expr` fragment specifier will accept more expressions in the 2024 edition.
895+
.suggestion = to keep the existing behavior, use the `expr_2021` fragment specifier.

compiler/rustc_lint/src/lib.rs

+3
Original file line numberDiff line numberDiff line change
@@ -77,6 +77,7 @@ mod traits;
7777
mod types;
7878
mod unit_bindings;
7979
mod unused;
80+
mod macro_expr_fragment_specifier_2024_migration;
8081

8182
pub use shadowed_into_iter::{ARRAY_INTO_ITER, BOXED_SLICE_INTO_ITER};
8283

@@ -113,6 +114,7 @@ use traits::*;
113114
use types::*;
114115
use unit_bindings::*;
115116
use unused::*;
117+
use macro_expr_fragment_specifier_2024_migration::*;
116118

117119
/// Useful for other parts of the compiler / Clippy.
118120
pub use builtin::{MissingDoc, SoftLints};
@@ -168,6 +170,7 @@ early_lint_methods!(
168170
IncompleteInternalFeatures: IncompleteInternalFeatures,
169171
RedundantSemicolons: RedundantSemicolons,
170172
UnusedDocComment: UnusedDocComment,
173+
Expr2024: Expr2024,
171174
]
172175
]
173176
);

compiler/rustc_lint/src/lints.rs

+7
Original file line numberDiff line numberDiff line change
@@ -329,6 +329,13 @@ pub struct BuiltinTypeAliasGenericBounds<'a, 'b> {
329329
pub sub: Option<SuggestChangingAssocTypes<'a, 'b>>,
330330
}
331331

332+
#[derive(LintDiagnostic)]
333+
#[diag(lint_macro_expr_fragment_specifier_2024_migration)]
334+
pub struct MacroExprFragment2023 {
335+
#[suggestion(code = "expr_2021", applicability = "machine-applicable")]
336+
pub suggestion: Span,
337+
}
338+
332339
pub struct BuiltinTypeAliasGenericBoundsSuggestion {
333340
pub suggestions: Vec<(Span, String)>,
334341
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,84 @@
1+
//! Migration code for the `expr_fragment_specifier_2024`
2+
//! rule.
3+
use tracing::debug;
4+
5+
use rustc_ast::token::Token;
6+
use rustc_ast::token::TokenKind;
7+
use rustc_ast::tokenstream::TokenStream;
8+
use rustc_ast::tokenstream::TokenTree;
9+
use rustc_session::declare_lint;
10+
use rustc_session::declare_lint_pass;
11+
use rustc_session::lint::FutureIncompatibilityReason;
12+
use rustc_span::edition::Edition;
13+
use rustc_span::sym;
14+
15+
use crate::lints::MacroExprFragment2023;
16+
use crate::EarlyLintPass;
17+
18+
declare_lint! {
19+
pub EDITION_2024_EXPR_FRAGMENT_SPECIFIER,
20+
Allow,
21+
"The `expr` fragment specifier will accept more expressions in the 2024 edition. \
22+
To keep the existing before, use the `expr_2021` fragment specifier.",
23+
@future_incompatible = FutureIncompatibleInfo {
24+
reason: FutureIncompatibilityReason::EditionSemanticsChange(Edition::Edition2024),
25+
reference: "issue #123742 <https://github.com/rust-lang/rust/issues/123742>",
26+
};
27+
}
28+
29+
declare_lint_pass!(Expr2024 => [EDITION_2024_EXPR_FRAGMENT_SPECIFIER,]);
30+
31+
impl Expr2024 {
32+
fn check_tokens(&mut self, cx: &crate::EarlyContext<'_>, tokens: &TokenStream) {
33+
// Check if the preceding token is `$`, because we want to allow `$async`, etc.
34+
let mut prev_dollar = false;
35+
for tt in tokens.trees() {
36+
match tt {
37+
// Only report non-raw idents.
38+
TokenTree::Token(token, _) => {
39+
if token.kind == TokenKind::Dollar {
40+
prev_dollar = true;
41+
continue;
42+
} else {
43+
if !prev_dollar {
44+
self.check_ident_token(cx, token);
45+
}
46+
}
47+
}
48+
TokenTree::Delimited(.., tts) => self.check_tokens(cx, tts),
49+
}
50+
prev_dollar = false;
51+
}
52+
}
53+
54+
fn check_ident_token(&mut self, cx: &crate::EarlyContext<'_>, token: &Token) {
55+
debug!("check_ident_token: {:?}", token);
56+
let (sym, edition) = match token.kind {
57+
TokenKind::Ident(sym, _) => (sym, Edition::Edition2024),
58+
_ => return,
59+
};
60+
61+
// Don't lint `r#foo`.
62+
debug!("token.span.edition(): {:?}", token.span.edition());
63+
if token.span.edition() >= edition {
64+
return;
65+
}
66+
67+
if sym != sym::expr {
68+
return;
69+
}
70+
71+
debug!("emitting lint");
72+
cx.builder.emit_span_lint(
73+
&EDITION_2024_EXPR_FRAGMENT_SPECIFIER,
74+
token.span.into(),
75+
MacroExprFragment2023 { suggestion: token.span },
76+
);
77+
}
78+
}
79+
80+
impl EarlyLintPass for Expr2024 {
81+
fn check_mac_def(&mut self, cx: &crate::EarlyContext<'_>, mc: &rustc_ast::MacroDef) {
82+
self.check_tokens(cx, &mc.body.tokens);
83+
}
84+
}

0 commit comments

Comments
 (0)