Skip to content

Commit b845985

Browse files
committed
Add new lint no_mangle_with_rust_abi
1 parent e1da002 commit b845985

9 files changed

+162
-3
lines changed

CHANGELOG.md

+1
Original file line numberDiff line numberDiff line change
@@ -4621,6 +4621,7 @@ Released 2018-09-13
46214621
[`no_effect`]: https://rust-lang.github.io/rust-clippy/master/index.html#no_effect
46224622
[`no_effect_replace`]: https://rust-lang.github.io/rust-clippy/master/index.html#no_effect_replace
46234623
[`no_effect_underscore_binding`]: https://rust-lang.github.io/rust-clippy/master/index.html#no_effect_underscore_binding
4624+
[`no_mangle_with_rust_abi`]: https://rust-lang.github.io/rust-clippy/master/index.html#no_mangle_with_rust_abi
46244625
[`non_ascii_literal`]: https://rust-lang.github.io/rust-clippy/master/index.html#non_ascii_literal
46254626
[`non_octal_unix_permissions`]: https://rust-lang.github.io/rust-clippy/master/index.html#non_octal_unix_permissions
46264627
[`non_send_fields_in_send_ty`]: https://rust-lang.github.io/rust-clippy/master/index.html#non_send_fields_in_send_ty

clippy_lints/src/declared_lints.rs

+1
Original file line numberDiff line numberDiff line change
@@ -449,6 +449,7 @@ pub(crate) static LINTS: &[&crate::LintInfo] = &[
449449
crate::no_effect::NO_EFFECT_INFO,
450450
crate::no_effect::NO_EFFECT_UNDERSCORE_BINDING_INFO,
451451
crate::no_effect::UNNECESSARY_OPERATION_INFO,
452+
crate::no_mangle_with_rust_abi::NO_MANGLE_WITH_RUST_ABI_INFO,
452453
crate::non_copy_const::BORROW_INTERIOR_MUTABLE_CONST_INFO,
453454
crate::non_copy_const::DECLARE_INTERIOR_MUTABLE_CONST_INFO,
454455
crate::non_expressive_names::JUST_UNDERSCORES_AND_DIGITS_INFO,

clippy_lints/src/lib.rs

+2
Original file line numberDiff line numberDiff line change
@@ -219,6 +219,7 @@ mod neg_cmp_op_on_partial_ord;
219219
mod neg_multiply;
220220
mod new_without_default;
221221
mod no_effect;
222+
mod no_mangle_with_rust_abi;
222223
mod non_copy_const;
223224
mod non_expressive_names;
224225
mod non_octal_unix_permissions;
@@ -916,6 +917,7 @@ pub fn register_plugins(store: &mut rustc_lint::LintStore, sess: &Session, conf:
916917
store.register_late_pass(|_| Box::new(size_of_ref::SizeOfRef));
917918
store.register_late_pass(|_| Box::new(multiple_unsafe_ops_per_block::MultipleUnsafeOpsPerBlock));
918919
store.register_late_pass(|_| Box::new(extra_unused_type_parameters::ExtraUnusedTypeParameters));
920+
store.register_late_pass(|_| Box::new(no_mangle_with_rust_abi::NoMangleWithRustAbi));
919921
// add lints here, do not remove this comment, it's used in `new_lint`
920922
}
921923

Original file line numberDiff line numberDiff line change
@@ -0,0 +1,64 @@
1+
use clippy_utils::diagnostics::span_lint_and_sugg;
2+
use clippy_utils::source::snippet_with_applicability;
3+
use rustc_errors::Applicability;
4+
use rustc_hir::{Item, ItemKind};
5+
use rustc_lint::{LateContext, LateLintPass};
6+
use rustc_session::{declare_lint_pass, declare_tool_lint};
7+
use rustc_target::spec::abi::Abi;
8+
9+
declare_clippy_lint! {
10+
/// ### What it does
11+
/// Checks for Rust ABI functions with the `#[no_mangle]` attribute.
12+
///
13+
/// ### Why is this bad?
14+
/// The Rust ABI is not stable, but in many simple cases matches
15+
/// enough with the C ABI that it is possible to forget to add
16+
/// `extern "C"` to a function called from C. Changes to the
17+
/// Rust ABI can break this at any point.
18+
///
19+
/// ### Example
20+
/// ```rust
21+
/// #[no_mangle]
22+
/// fn example(arg_one: u32, arg_two: usize) {}
23+
/// ```
24+
///
25+
/// Use instead:
26+
/// ```rust
27+
/// #[no_mangle]
28+
/// extern "C" fn example(arg_one: u32, arg_two: usize) {}
29+
/// ```
30+
#[clippy::version = "1.69.0"]
31+
pub NO_MANGLE_WITH_RUST_ABI,
32+
suspicious,
33+
"convert Rust ABI functions to C ABI"
34+
}
35+
declare_lint_pass!(NoMangleWithRustAbi => [NO_MANGLE_WITH_RUST_ABI]);
36+
37+
impl<'tcx> LateLintPass<'tcx> for NoMangleWithRustAbi {
38+
fn check_item(&mut self, cx: &LateContext<'tcx>, item: &'tcx Item<'tcx>) {
39+
if let ItemKind::Fn(fn_sig, _, _) = &item.kind {
40+
let attrs = cx.tcx.hir().attrs(item.hir_id());
41+
for attr in attrs {
42+
if let Some(ident) = attr.ident()
43+
&& ident.name.as_str() == "no_mangle"
44+
&& fn_sig.header.abi == Abi::Rust {
45+
46+
let mut applicability = Applicability::MachineApplicable;
47+
let snippet = snippet_with_applicability(cx, fn_sig.span, "..", &mut applicability);
48+
let suggestion = snippet.split_once("fn")
49+
.map(|(first, second)| format!(r#"{first}extern "C" fn{second}"#))
50+
.unwrap_or("".to_string());
51+
span_lint_and_sugg(
52+
cx,
53+
NO_MANGLE_WITH_RUST_ABI,
54+
fn_sig.span,
55+
"attribute #[no_mangle] set on a Rust ABI function",
56+
"try",
57+
suggestion,
58+
applicability
59+
);
60+
}
61+
}
62+
}
63+
}
64+
}

tests/ui/must_use_candidates.fixed

+1-1
Original file line numberDiff line numberDiff line change
@@ -84,7 +84,7 @@ pub unsafe fn mutates_static() -> usize {
8484
}
8585

8686
#[no_mangle]
87-
pub fn unmangled(i: bool) -> bool {
87+
extern "C" pub fn unmangled(i: bool) -> bool {
8888
!i
8989
}
9090

tests/ui/must_use_candidates.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -84,7 +84,7 @@ pub unsafe fn mutates_static() -> usize {
8484
}
8585

8686
#[no_mangle]
87-
pub fn unmangled(i: bool) -> bool {
87+
pub extern "C" fn unmangled(i: bool) -> bool {
8888
!i
8989
}
9090

tests/ui/must_use_candidates.stderr

+9-1
Original file line numberDiff line numberDiff line change
@@ -30,5 +30,13 @@ error: this function could have a `#[must_use]` attribute
3030
LL | pub fn arcd(_x: Arc<u32>) -> bool {
3131
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ help: add the attribute: `#[must_use] pub fn arcd(_x: Arc<u32>) -> bool`
3232

33-
error: aborting due to 5 previous errors
33+
error: attribute #[no_mangle] set on a Rust ABI function
34+
--> $DIR/must_use_candidates.rs:87:1
35+
|
36+
LL | pub fn unmangled(i: bool) -> bool {
37+
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ help: try: `extern "C" pub fn unmangled(i: bool) -> bool`
38+
|
39+
= note: `-D clippy::no-mangle-with-rust-abi` implied by `-D warnings`
40+
41+
error: aborting due to 6 previous errors
3442

tests/ui/no_mangle_with_rust_abi.rs

+38
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,38 @@
1+
#![allow(unused)]
2+
#![warn(clippy::no_mangle_with_rust_abi)]
3+
4+
#[no_mangle]
5+
fn rust_abi_fn_one(arg_one: u32, arg_two: usize) {}
6+
7+
#[no_mangle]
8+
pub fn rust_abi_fn_two(arg_one: u32, arg_two: usize) {}
9+
10+
11+
/// # Safety
12+
/// This function shouldn't be called unless the horsemen are ready
13+
#[no_mangle]
14+
pub unsafe fn rust_abi_fn_three(arg_one: u32, arg_two: usize) {}
15+
16+
/// # Safety
17+
/// This function shouldn't be called unless the horsemen are ready
18+
#[no_mangle]
19+
unsafe fn rust_abi_fn_four(arg_one: u32, arg_two: usize) {}
20+
21+
#[no_mangle]
22+
fn rust_abi_multiline_function_really_long_name_to_overflow_args_to_multiple_lines(
23+
arg_one: u32,
24+
arg_two: usize,
25+
) -> u32 {
26+
0
27+
}
28+
29+
fn rust_abi_fn_again(arg_one: u32, arg_two: usize) {}
30+
31+
#[no_mangle]
32+
extern "C" fn c_abi_fn(arg_one: u32, arg_two: usize) {}
33+
34+
extern "C" fn c_abi_fn_again(arg_one: u32, arg_two: usize) {}
35+
36+
fn main() {
37+
// test code goes here
38+
}
+45
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,45 @@
1+
error: attribute #[no_mangle] set on a Rust ABI function
2+
--> $DIR/no_mangle_with_rust_abi.rs:5:1
3+
|
4+
LL | fn rust_abi_fn_one(arg_one: u32, arg_two: usize) {}
5+
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ help: try: `extern "C" fn rust_abi_fn_one(arg_one: u32, arg_two: usize)`
6+
|
7+
= note: `-D clippy::no-mangle-with-rust-abi` implied by `-D warnings`
8+
9+
error: attribute #[no_mangle] set on a Rust ABI function
10+
--> $DIR/no_mangle_with_rust_abi.rs:8:1
11+
|
12+
LL | pub fn rust_abi_fn_two(arg_one: u32, arg_two: usize) {}
13+
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ help: try: `pub extern "C" fn rust_abi_fn_two(arg_one: u32, arg_two: usize)`
14+
15+
error: attribute #[no_mangle] set on a Rust ABI function
16+
--> $DIR/no_mangle_with_rust_abi.rs:14:1
17+
|
18+
LL | pub unsafe fn rust_abi_fn_three(arg_one: u32, arg_two: usize) {}
19+
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ help: try: `pub unsafe extern "C" fn rust_abi_fn_three(arg_one: u32, arg_two: usize)`
20+
21+
error: attribute #[no_mangle] set on a Rust ABI function
22+
--> $DIR/no_mangle_with_rust_abi.rs:19:1
23+
|
24+
LL | unsafe fn rust_abi_fn_four(arg_one: u32, arg_two: usize) {}
25+
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ help: try: `unsafe extern "C" fn rust_abi_fn_four(arg_one: u32, arg_two: usize)`
26+
27+
error: attribute #[no_mangle] set on a Rust ABI function
28+
--> $DIR/no_mangle_with_rust_abi.rs:22:1
29+
|
30+
LL | / fn rust_abi_multiline_function_really_long_name_to_overflow_args_to_multiple_lines(
31+
LL | | arg_one: u32,
32+
LL | | arg_two: usize,
33+
LL | | ) -> u32 {
34+
| |________^
35+
|
36+
help: try
37+
|
38+
LL + extern "C" fn rust_abi_multiline_function_really_long_name_to_overflow_args_to_multiple_lines(
39+
LL + arg_one: u32,
40+
LL + arg_two: usize,
41+
LL ~ ) -> u32 {
42+
|
43+
44+
error: aborting due to 5 previous errors
45+

0 commit comments

Comments
 (0)