Skip to content

Commit f6d1a8f

Browse files
committed
Add static_mut lint
Fixes #12896
1 parent 48686ad commit f6d1a8f

File tree

6 files changed

+79
-0
lines changed

6 files changed

+79
-0
lines changed

CHANGELOG.md

+1
Original file line numberDiff line numberDiff line change
@@ -5819,6 +5819,7 @@ Released 2018-09-13
58195819
[`skip_while_next`]: https://rust-lang.github.io/rust-clippy/master/index.html#skip_while_next
58205820
[`slow_vector_initialization`]: https://rust-lang.github.io/rust-clippy/master/index.html#slow_vector_initialization
58215821
[`stable_sort_primitive`]: https://rust-lang.github.io/rust-clippy/master/index.html#stable_sort_primitive
5822+
[`static_mut`]: https://rust-lang.github.io/rust-clippy/master/index.html#static_mut
58225823
[`std_instead_of_alloc`]: https://rust-lang.github.io/rust-clippy/master/index.html#std_instead_of_alloc
58235824
[`std_instead_of_core`]: https://rust-lang.github.io/rust-clippy/master/index.html#std_instead_of_core
58245825
[`str_split_at_newline`]: https://rust-lang.github.io/rust-clippy/master/index.html#str_split_at_newline

clippy_lints/src/declared_lints.rs

+1
Original file line numberDiff line numberDiff line change
@@ -653,6 +653,7 @@ pub(crate) static LINTS: &[&crate::LintInfo] = &[
653653
crate::size_of_in_element_count::SIZE_OF_IN_ELEMENT_COUNT_INFO,
654654
crate::size_of_ref::SIZE_OF_REF_INFO,
655655
crate::slow_vector_initialization::SLOW_VECTOR_INITIALIZATION_INFO,
656+
crate::static_mut::STATIC_MUT_INFO,
656657
crate::std_instead_of_core::ALLOC_INSTEAD_OF_CORE_INFO,
657658
crate::std_instead_of_core::STD_INSTEAD_OF_ALLOC_INFO,
658659
crate::std_instead_of_core::STD_INSTEAD_OF_CORE_INFO,

clippy_lints/src/lib.rs

+2
Original file line numberDiff line numberDiff line change
@@ -325,6 +325,7 @@ mod single_range_in_vec_init;
325325
mod size_of_in_element_count;
326326
mod size_of_ref;
327327
mod slow_vector_initialization;
328+
mod static_mut;
328329
mod std_instead_of_core;
329330
mod strings;
330331
mod strlen_on_c_strings;
@@ -1167,6 +1168,7 @@ pub fn register_lints(store: &mut rustc_lint::LintStore, conf: &'static Conf) {
11671168
..Default::default()
11681169
})
11691170
});
1171+
store.register_early_pass(|| Box::new(static_mut::StaticMut));
11701172
// add lints here, do not remove this comment, it's used in `new_lint`
11711173
}
11721174

clippy_lints/src/static_mut.rs

+60
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,60 @@
1+
use clippy_utils::diagnostics::span_lint_and_help;
2+
use rustc_ast::ast::*;
3+
use rustc_lint::{EarlyContext, EarlyLintPass, LintContext};
4+
use rustc_middle::lint::in_external_macro;
5+
use rustc_session::declare_lint_pass;
6+
7+
declare_clippy_lint! {
8+
/// ### What it does
9+
/// Produces warnings when a `static mut` is declared.
10+
///
11+
/// ### Why is this bad?
12+
/// `static mut` can [easily produce undefined behavior][1] and
13+
/// [may be removed in the future][2].
14+
///
15+
/// ### Example
16+
/// ```no_run
17+
/// static mut GLOBAL_INT: u8 = 0;
18+
/// ```
19+
/// Use instead:
20+
/// ```no_run
21+
/// use std::sync::RwLock;
22+
///
23+
/// static GLOBAL_INT: RwLock<u8> = RwLock::new(0);
24+
/// ```
25+
///
26+
/// [1]: https://doc.rust-lang.org/nightly/edition-guide/rust-2024/static-mut-reference.html
27+
/// [2]: https://github.com/rust-lang/rfcs/pull/3560
28+
#[clippy::version = "1.80.0"]
29+
pub STATIC_MUT,
30+
nursery,
31+
"detect mutable static definitions"
32+
}
33+
34+
declare_lint_pass!(StaticMut => [STATIC_MUT]);
35+
36+
impl EarlyLintPass for StaticMut {
37+
fn check_item(&mut self, cx: &EarlyContext<'_>, item: &Item) {
38+
if in_external_macro(cx.sess(), item.span) {
39+
return;
40+
};
41+
let ItemKind::Static(ref static_item_box) = item.kind else {
42+
return;
43+
};
44+
let StaticItem {
45+
mutability: Mutability::Mut,
46+
..
47+
} = static_item_box.as_ref()
48+
else {
49+
return;
50+
};
51+
span_lint_and_help(
52+
cx,
53+
STATIC_MUT,
54+
item.span,
55+
"declaration of static mut",
56+
None,
57+
"remove the `mut` and use a type with interior mutibability that implements `Sync`, such as `std::sync::Mutex`",
58+
)
59+
}
60+
}

tests/ui/static_mut.rs

+3
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
#![warn(clippy::static_mut)]
2+
3+
static mut NUMBER: u8 = 3;

tests/ui/static_mut.stderr

+12
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,12 @@
1+
error: declaration of static mut
2+
--> tests/ui/static_mut.rs:3:1
3+
|
4+
LL | static mut NUMBER: u8 = 3;
5+
| ^^^^^^^^^^^^^^^^^^^^^^^^^^
6+
|
7+
= help: remove the `mut` and use a type with interior mutibability that implements `Sync`, such as `std::sync::Mutex`
8+
= note: `-D clippy::static-mut` implied by `-D warnings`
9+
= help: to override `-D warnings` add `#[allow(clippy::static_mut)]`
10+
11+
error: aborting due to 1 previous error
12+

0 commit comments

Comments
 (0)