Skip to content

Commit 26edd5a

Browse files
committed
Auto merge of #11104 - Alexendoo:arc-with-non-send-sync, r=Centri3
`arc_with_non_send_sync`: reword and move to `suspicious` Fixes #11079 changelog: [`arc_with_non_send_sync`]: move to complexity
2 parents dd8e44c + 4939a71 commit 26edd5a

File tree

3 files changed

+73
-43
lines changed

3 files changed

+73
-43
lines changed
+37-34
Original file line numberDiff line numberDiff line change
@@ -1,12 +1,11 @@
1-
use clippy_utils::diagnostics::span_lint_and_help;
1+
use clippy_utils::diagnostics::span_lint_and_then;
22
use clippy_utils::last_path_segment;
33
use clippy_utils::ty::{implements_trait, is_type_diagnostic_item};
4-
use if_chain::if_chain;
5-
64
use rustc_hir::{Expr, ExprKind};
75
use rustc_lint::LateContext;
86
use rustc_lint::LateLintPass;
97
use rustc_middle::ty;
8+
use rustc_middle::ty::print::with_forced_trimmed_paths;
109
use rustc_middle::ty::GenericArgKind;
1110
use rustc_session::{declare_lint_pass, declare_tool_lint};
1211
use rustc_span::symbol::sym;
@@ -16,61 +15,65 @@ declare_clippy_lint! {
1615
/// This lint warns when you use `Arc` with a type that does not implement `Send` or `Sync`.
1716
///
1817
/// ### Why is this bad?
19-
/// Wrapping a type in Arc doesn't add thread safety to the underlying data, so data races
20-
/// could occur when touching the underlying data.
18+
/// `Arc<T>` is only `Send`/`Sync` when `T` is [both `Send` and `Sync`](https://doc.rust-lang.org/std/sync/struct.Arc.html#impl-Send-for-Arc%3CT%3E),
19+
/// either `T` should be made `Send + Sync` or an `Rc` should be used instead of an `Arc`
2120
///
2221
/// ### Example
2322
/// ```rust
2423
/// # use std::cell::RefCell;
2524
/// # use std::sync::Arc;
2625
///
2726
/// fn main() {
28-
/// // This is safe, as `i32` implements `Send` and `Sync`.
27+
/// // This is fine, as `i32` implements `Send` and `Sync`.
2928
/// let a = Arc::new(42);
3029
///
31-
/// // This is not safe, as `RefCell` does not implement `Sync`.
30+
/// // `RefCell` is `!Sync`, so either the `Arc` should be replaced with an `Rc`
31+
/// // or the `RefCell` replaced with something like a `RwLock`
3232
/// let b = Arc::new(RefCell::new(42));
3333
/// }
3434
/// ```
3535
#[clippy::version = "1.72.0"]
3636
pub ARC_WITH_NON_SEND_SYNC,
37-
correctness,
37+
suspicious,
3838
"using `Arc` with a type that does not implement `Send` or `Sync`"
3939
}
4040
declare_lint_pass!(ArcWithNonSendSync => [ARC_WITH_NON_SEND_SYNC]);
4141

4242
impl LateLintPass<'_> for ArcWithNonSendSync {
4343
fn check_expr(&mut self, cx: &LateContext<'_>, expr: &Expr<'_>) {
4444
let ty = cx.typeck_results().expr_ty(expr);
45-
if_chain! {
46-
if is_type_diagnostic_item(cx, ty, sym::Arc);
47-
if let ExprKind::Call(func, [arg]) = expr.kind;
48-
if let ExprKind::Path(func_path) = func.kind;
49-
if last_path_segment(&func_path).ident.name == sym::new;
50-
if let arg_ty = cx.typeck_results().expr_ty(arg);
45+
if is_type_diagnostic_item(cx, ty, sym::Arc)
46+
&& let ExprKind::Call(func, [arg]) = expr.kind
47+
&& let ExprKind::Path(func_path) = func.kind
48+
&& last_path_segment(&func_path).ident.name == sym::new
49+
&& let arg_ty = cx.typeck_results().expr_ty(arg)
5150
// make sure that the type is not and does not contain any type parameters
52-
if arg_ty.walk().all(|arg| {
51+
&& arg_ty.walk().all(|arg| {
5352
!matches!(arg.unpack(), GenericArgKind::Type(ty) if matches!(ty.kind(), ty::Param(_)))
54-
});
55-
if !cx.tcx
56-
.lang_items()
57-
.sync_trait()
58-
.map_or(false, |id| implements_trait(cx, arg_ty, id, &[])) ||
59-
!cx.tcx
60-
.get_diagnostic_item(sym::Send)
61-
.map_or(false, |id| implements_trait(cx, arg_ty, id, &[]));
53+
})
54+
&& let Some(send) = cx.tcx.get_diagnostic_item(sym::Send)
55+
&& let Some(sync) = cx.tcx.lang_items().sync_trait()
56+
&& let [is_send, is_sync] = [send, sync].map(|id| implements_trait(cx, arg_ty, id, &[]))
57+
&& !(is_send && is_sync)
58+
{
59+
span_lint_and_then(
60+
cx,
61+
ARC_WITH_NON_SEND_SYNC,
62+
expr.span,
63+
"usage of an `Arc` that is not `Send` or `Sync`",
64+
|diag| with_forced_trimmed_paths!({
65+
if !is_send {
66+
diag.note(format!("the trait `Send` is not implemented for `{arg_ty}`"));
67+
}
68+
if !is_sync {
69+
diag.note(format!("the trait `Sync` is not implemented for `{arg_ty}`"));
70+
}
71+
72+
diag.note(format!("required for `{ty}` to implement `Send` and `Sync`"));
6273

63-
then {
64-
span_lint_and_help(
65-
cx,
66-
ARC_WITH_NON_SEND_SYNC,
67-
expr.span,
68-
"usage of `Arc<T>` where `T` is not `Send` or `Sync`",
69-
None,
70-
"consider using `Rc<T>` instead or wrapping `T` in a std::sync type like \
71-
`Mutex<T>`",
72-
);
73-
}
74+
diag.help("consider using an `Rc` instead or wrapping the inner type with a `Mutex`");
75+
}
76+
));
7477
}
7578
}
7679
}

tests/ui/arc_with_non_send_sync.rs

+8-4
Original file line numberDiff line numberDiff line change
@@ -12,9 +12,13 @@ fn issue11076<T>() {
1212
}
1313

1414
fn main() {
15-
// This is safe, as `i32` implements `Send` and `Sync`.
16-
let a = Arc::new(42);
15+
let _ = Arc::new(42);
1716

18-
// This is not safe, as `RefCell` does not implement `Sync`.
19-
let b = Arc::new(RefCell::new(42));
17+
// !Sync
18+
let _ = Arc::new(RefCell::new(42));
19+
let mutex = Mutex::new(1);
20+
// !Send
21+
let _ = Arc::new(mutex.lock().unwrap());
22+
// !Send + !Sync
23+
let _ = Arc::new(&42 as *const i32);
2024
}
+28-5
Original file line numberDiff line numberDiff line change
@@ -1,11 +1,34 @@
1-
error: usage of `Arc<T>` where `T` is not `Send` or `Sync`
2-
--> $DIR/arc_with_non_send_sync.rs:19:13
1+
error: usage of an `Arc` that is not `Send` or `Sync`
2+
--> $DIR/arc_with_non_send_sync.rs:18:13
33
|
4-
LL | let b = Arc::new(RefCell::new(42));
4+
LL | let _ = Arc::new(RefCell::new(42));
55
| ^^^^^^^^^^^^^^^^^^^^^^^^^^
66
|
7-
= help: consider using `Rc<T>` instead or wrapping `T` in a std::sync type like `Mutex<T>`
7+
= note: the trait `Sync` is not implemented for `RefCell<i32>`
8+
= note: required for `Arc<RefCell<i32>>` to implement `Send` and `Sync`
9+
= help: consider using an `Rc` instead or wrapping the inner type with a `Mutex`
810
= note: `-D clippy::arc-with-non-send-sync` implied by `-D warnings`
911

10-
error: aborting due to previous error
12+
error: usage of an `Arc` that is not `Send` or `Sync`
13+
--> $DIR/arc_with_non_send_sync.rs:21:13
14+
|
15+
LL | let _ = Arc::new(mutex.lock().unwrap());
16+
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
17+
|
18+
= note: the trait `Send` is not implemented for `MutexGuard<'_, i32>`
19+
= note: required for `Arc<MutexGuard<'_, i32>>` to implement `Send` and `Sync`
20+
= help: consider using an `Rc` instead or wrapping the inner type with a `Mutex`
21+
22+
error: usage of an `Arc` that is not `Send` or `Sync`
23+
--> $DIR/arc_with_non_send_sync.rs:23:13
24+
|
25+
LL | let _ = Arc::new(&42 as *const i32);
26+
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^
27+
|
28+
= note: the trait `Send` is not implemented for `*const i32`
29+
= note: the trait `Sync` is not implemented for `*const i32`
30+
= note: required for `Arc<*const i32>` to implement `Send` and `Sync`
31+
= help: consider using an `Rc` instead or wrapping the inner type with a `Mutex`
32+
33+
error: aborting due to 3 previous errors
1134

0 commit comments

Comments
 (0)