Skip to content

Commit 077fc26

Browse files
committed
Auto merge of #109732 - Urgau:uplift_drop_forget_ref_lints, r=davidtwco
Uplift `clippy::{drop,forget}_{ref,copy}` lints This PR aims at uplifting the `clippy::drop_ref`, `clippy::drop_copy`, `clippy::forget_ref` and `clippy::forget_copy` lints. Those lints are/were declared in the correctness category of clippy because they lint on useless and most probably is not what the developer wanted. ## `drop_ref` and `forget_ref` The `drop_ref` and `forget_ref` lint checks for calls to `std::mem::drop` or `std::mem::forget` with a reference instead of an owned value. ### Example ```rust let mut lock_guard = mutex.lock(); std::mem::drop(&lock_guard) // Should have been drop(lock_guard), mutex // still locked operation_that_requires_mutex_to_be_unlocked(); ``` ### Explanation Calling `drop` or `forget` on a reference will only drop the reference itself, which is a no-op. It will not call the `drop` or `forget` method on the underlying referenced value, which is likely what was intended. ## `drop_copy` and `forget_copy` The `drop_copy` and `forget_copy` lint checks for calls to `std::mem::forget` or `std::mem::drop` with a value that derives the Copy trait. ### Example ```rust let x: i32 = 42; // i32 implements Copy std::mem::forget(x) // A copy of x is passed to the function, leaving the // original unaffected ``` ### Explanation Calling `std::mem::forget` [does nothing for types that implement Copy](https://doc.rust-lang.org/std/mem/fn.drop.html) since the value will be copied and moved into the function on invocation. ----- Followed the instructions for uplift a clippy describe here: #99696 (review) cc `@m-ou-se` (as T-libs-api leader because the uplifting was discussed in a recent meeting)
2 parents 0b79504 + f5aede9 commit 077fc26

File tree

98 files changed

+1106
-772
lines changed

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

98 files changed

+1106
-772
lines changed

compiler/rustc_builtin_macros/src/format_foreign.rs

+1-3
Original file line numberDiff line numberDiff line change
@@ -562,15 +562,13 @@ pub(crate) mod printf {
562562
}
563563

564564
if let Type = state {
565-
drop(c);
566565
type_ = at.slice_between(next).unwrap();
567566

568567
// Don't use `move_to!` here, as we *can* be at the end of the input.
569568
at = next;
570569
}
571570

572-
drop(c);
573-
drop(next);
571+
let _ = c; // to avoid never used value
574572

575573
end = at;
576574
let position = InnerSpan::new(start.at, end.at);

compiler/rustc_infer/src/infer/nll_relate/mod.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -828,7 +828,7 @@ where
828828
} else {
829829
match variables.probe(vid) {
830830
TypeVariableValue::Known { value: u } => {
831-
drop(variables);
831+
drop(inner);
832832
self.relate(u, u)
833833
}
834834
TypeVariableValue::Unknown { universe: _universe } => {

compiler/rustc_lint/messages.ftl

+16
Original file line numberDiff line numberDiff line change
@@ -520,3 +520,19 @@ lint_opaque_hidden_inferred_bound = opaque type `{$ty}` does not satisfy its ass
520520
.specifically = this associated type bound is unsatisfied for `{$proj_ty}`
521521
522522
lint_opaque_hidden_inferred_bound_sugg = add this bound
523+
524+
lint_drop_ref = calls to `std::mem::drop` with a reference instead of an owned value does nothing
525+
.label = argument has type `{$arg_ty}`
526+
.note = use `let _ = ...` to ignore the expression or result
527+
528+
lint_drop_copy = calls to `std::mem::drop` with a value that implements `Copy` does nothing
529+
.label = argument has type `{$arg_ty}`
530+
.note = use `let _ = ...` to ignore the expression or result
531+
532+
lint_forget_ref = calls to `std::mem::forget` with a reference instead of an owned value does nothing
533+
.label = argument has type `{$arg_ty}`
534+
.note = use `let _ = ...` to ignore the expression or result
535+
536+
lint_forget_copy = calls to `std::mem::forget` with a value that implements `Copy` does nothing
537+
.label = argument has type `{$arg_ty}`
538+
.note = use `let _ = ...` to ignore the expression or result
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,164 @@
1+
use rustc_hir::{Arm, Expr, ExprKind, Node};
2+
use rustc_span::sym;
3+
4+
use crate::{
5+
lints::{DropCopyDiag, DropRefDiag, ForgetCopyDiag, ForgetRefDiag},
6+
LateContext, LateLintPass, LintContext,
7+
};
8+
9+
declare_lint! {
10+
/// The `drop_ref` lint checks for calls to `std::mem::drop` with a reference
11+
/// instead of an owned value.
12+
///
13+
/// ### Example
14+
///
15+
/// ```rust
16+
/// # fn operation_that_requires_mutex_to_be_unlocked() {} // just to make it compile
17+
/// # let mutex = std::sync::Mutex::new(1); // just to make it compile
18+
/// let mut lock_guard = mutex.lock();
19+
/// std::mem::drop(&lock_guard); // Should have been drop(lock_guard), mutex
20+
/// // still locked
21+
/// operation_that_requires_mutex_to_be_unlocked();
22+
/// ```
23+
///
24+
/// {{produces}}
25+
///
26+
/// ### Explanation
27+
///
28+
/// Calling `drop` on a reference will only drop the
29+
/// reference itself, which is a no-op. It will not call the `drop` method (from
30+
/// the `Drop` trait implementation) on the underlying referenced value, which
31+
/// is likely what was intended.
32+
pub DROP_REF,
33+
Warn,
34+
"calls to `std::mem::drop` with a reference instead of an owned value"
35+
}
36+
37+
declare_lint! {
38+
/// The `forget_ref` lint checks for calls to `std::mem::forget` with a reference
39+
/// instead of an owned value.
40+
///
41+
/// ### Example
42+
///
43+
/// ```rust
44+
/// let x = Box::new(1);
45+
/// std::mem::forget(&x); // Should have been forget(x), x will still be dropped
46+
/// ```
47+
///
48+
/// {{produces}}
49+
///
50+
/// ### Explanation
51+
///
52+
/// Calling `forget` on a reference will only forget the
53+
/// reference itself, which is a no-op. It will not forget the underlying
54+
/// referenced value, which is likely what was intended.
55+
pub FORGET_REF,
56+
Warn,
57+
"calls to `std::mem::forget` with a reference instead of an owned value"
58+
}
59+
60+
declare_lint! {
61+
/// The `drop_copy` lint checks for calls to `std::mem::drop` with a value
62+
/// that derives the Copy trait.
63+
///
64+
/// ### Example
65+
///
66+
/// ```rust
67+
/// let x: i32 = 42; // i32 implements Copy
68+
/// std::mem::drop(x); // A copy of x is passed to the function, leaving the
69+
/// // original unaffected
70+
/// ```
71+
///
72+
/// {{produces}}
73+
///
74+
/// ### Explanation
75+
///
76+
/// Calling `std::mem::drop` [does nothing for types that
77+
/// implement Copy](https://doc.rust-lang.org/std/mem/fn.drop.html), since the
78+
/// value will be copied and moved into the function on invocation.
79+
pub DROP_COPY,
80+
Warn,
81+
"calls to `std::mem::drop` with a value that implements Copy"
82+
}
83+
84+
declare_lint! {
85+
/// The `forget_copy` lint checks for calls to `std::mem::forget` with a value
86+
/// that derives the Copy trait.
87+
///
88+
/// ### Example
89+
///
90+
/// ```rust
91+
/// let x: i32 = 42; // i32 implements Copy
92+
/// std::mem::forget(x); // A copy of x is passed to the function, leaving the
93+
/// // original unaffected
94+
/// ```
95+
///
96+
/// {{produces}}
97+
///
98+
/// ### Explanation
99+
///
100+
/// Calling `std::mem::forget` [does nothing for types that
101+
/// implement Copy](https://doc.rust-lang.org/std/mem/fn.drop.html) since the
102+
/// value will be copied and moved into the function on invocation.
103+
///
104+
/// An alternative, but also valid, explanation is that Copy types do not
105+
/// implement the Drop trait, which means they have no destructors. Without a
106+
/// destructor, there is nothing for `std::mem::forget` to ignore.
107+
pub FORGET_COPY,
108+
Warn,
109+
"calls to `std::mem::forget` with a value that implements Copy"
110+
}
111+
112+
declare_lint_pass!(DropForgetUseless => [DROP_REF, FORGET_REF, DROP_COPY, FORGET_COPY]);
113+
114+
impl<'tcx> LateLintPass<'tcx> for DropForgetUseless {
115+
fn check_expr(&mut self, cx: &LateContext<'tcx>, expr: &'tcx Expr<'tcx>) {
116+
if let ExprKind::Call(path, [arg]) = expr.kind
117+
&& let ExprKind::Path(ref qpath) = path.kind
118+
&& let Some(def_id) = cx.qpath_res(qpath, path.hir_id).opt_def_id()
119+
&& let Some(fn_name) = cx.tcx.get_diagnostic_name(def_id)
120+
{
121+
let arg_ty = cx.typeck_results().expr_ty(arg);
122+
let is_copy = arg_ty.is_copy_modulo_regions(cx.tcx, cx.param_env);
123+
let drop_is_single_call_in_arm = is_single_call_in_arm(cx, arg, expr);
124+
match fn_name {
125+
sym::mem_drop if arg_ty.is_ref() && !drop_is_single_call_in_arm => {
126+
cx.emit_spanned_lint(DROP_REF, expr.span, DropRefDiag { arg_ty, label: arg.span });
127+
},
128+
sym::mem_forget if arg_ty.is_ref() => {
129+
cx.emit_spanned_lint(FORGET_REF, expr.span, ForgetRefDiag { arg_ty, label: arg.span });
130+
},
131+
sym::mem_drop if is_copy && !drop_is_single_call_in_arm => {
132+
cx.emit_spanned_lint(DROP_COPY, expr.span, DropCopyDiag { arg_ty, label: arg.span });
133+
}
134+
sym::mem_forget if is_copy => {
135+
cx.emit_spanned_lint(FORGET_COPY, expr.span, ForgetCopyDiag { arg_ty, label: arg.span });
136+
}
137+
_ => return,
138+
};
139+
}
140+
}
141+
}
142+
143+
// Dropping returned value of a function, as in the following snippet is considered idiomatic, see
144+
// rust-lang/rust-clippy#9482 for examples.
145+
//
146+
// ```
147+
// match <var> {
148+
// <pat> => drop(fn_with_side_effect_and_returning_some_value()),
149+
// ..
150+
// }
151+
// ```
152+
fn is_single_call_in_arm<'tcx>(
153+
cx: &LateContext<'tcx>,
154+
arg: &'tcx Expr<'_>,
155+
drop_expr: &'tcx Expr<'_>,
156+
) -> bool {
157+
if matches!(arg.kind, ExprKind::Call(..) | ExprKind::MethodCall(..)) {
158+
let parent_node = cx.tcx.hir().find_parent(drop_expr.hir_id);
159+
if let Some(Node::Arm(Arm { body, .. })) = &parent_node {
160+
return body.hir_id == drop_expr.hir_id;
161+
}
162+
}
163+
false
164+
}

compiler/rustc_lint/src/lib.rs

+3
Original file line numberDiff line numberDiff line change
@@ -52,6 +52,7 @@ mod array_into_iter;
5252
pub mod builtin;
5353
mod context;
5454
mod deref_into_dyn_supertrait;
55+
mod drop_forget_useless;
5556
mod early;
5657
mod enum_intrinsics_non_enums;
5758
mod errors;
@@ -96,6 +97,7 @@ use rustc_span::Span;
9697
use array_into_iter::ArrayIntoIter;
9798
use builtin::*;
9899
use deref_into_dyn_supertrait::*;
100+
use drop_forget_useless::*;
99101
use enum_intrinsics_non_enums::EnumIntrinsicsNonEnums;
100102
use for_loops_over_fallibles::*;
101103
use hidden_unicode_codepoints::*;
@@ -201,6 +203,7 @@ late_lint_methods!(
201203
[
202204
ForLoopsOverFallibles: ForLoopsOverFallibles,
203205
DerefIntoDynSupertrait: DerefIntoDynSupertrait,
206+
DropForgetUseless: DropForgetUseless,
204207
HardwiredLints: HardwiredLints,
205208
ImproperCTypesDeclarations: ImproperCTypesDeclarations,
206209
ImproperCTypesDefinitions: ImproperCTypesDefinitions,

compiler/rustc_lint/src/lints.rs

+37
Original file line numberDiff line numberDiff line change
@@ -662,6 +662,43 @@ pub struct ForLoopsOverFalliblesSuggestion<'a> {
662662
pub end_span: Span,
663663
}
664664

665+
// drop_ref.rs
666+
#[derive(LintDiagnostic)]
667+
#[diag(lint_drop_ref)]
668+
#[note]
669+
pub struct DropRefDiag<'a> {
670+
pub arg_ty: Ty<'a>,
671+
#[label]
672+
pub label: Span,
673+
}
674+
675+
#[derive(LintDiagnostic)]
676+
#[diag(lint_drop_copy)]
677+
#[note]
678+
pub struct DropCopyDiag<'a> {
679+
pub arg_ty: Ty<'a>,
680+
#[label]
681+
pub label: Span,
682+
}
683+
684+
#[derive(LintDiagnostic)]
685+
#[diag(lint_forget_ref)]
686+
#[note]
687+
pub struct ForgetRefDiag<'a> {
688+
pub arg_ty: Ty<'a>,
689+
#[label]
690+
pub label: Span,
691+
}
692+
693+
#[derive(LintDiagnostic)]
694+
#[diag(lint_forget_copy)]
695+
#[note]
696+
pub struct ForgetCopyDiag<'a> {
697+
pub arg_ty: Ty<'a>,
698+
#[label]
699+
pub label: Span,
700+
}
701+
665702
// hidden_unicode_codepoints.rs
666703
#[derive(LintDiagnostic)]
667704
#[diag(lint_hidden_unicode_codepoints)]

compiler/rustc_middle/src/mir/visit.rs

+1-2
Original file line numberDiff line numberDiff line change
@@ -880,12 +880,11 @@ macro_rules! make_mir_visitor {
880880
) {
881881
let Constant {
882882
span,
883-
user_ty,
883+
user_ty: _, // no visit method for this
884884
literal,
885885
} = constant;
886886

887887
self.visit_span($(& $mutability)? *span);
888-
drop(user_ty); // no visit method for this
889888
match literal {
890889
ConstantKind::Ty(ct) => self.visit_ty_const($(&$mutability)? *ct, location),
891890
ConstantKind::Val(_, ty) => self.visit_ty($(& $mutability)? *ty, TyContext::Location(location)),

library/core/src/mem/mod.rs

+1
Original file line numberDiff line numberDiff line change
@@ -968,6 +968,7 @@ pub const fn replace<T>(dest: &mut T, src: T) -> T {
968968
/// Integers and other types implementing [`Copy`] are unaffected by `drop`.
969969
///
970970
/// ```
971+
/// # #![cfg_attr(not(bootstrap), allow(drop_copy))]
971972
/// #[derive(Copy, Clone)]
972973
/// struct Foo(u8);
973974
///

library/core/src/task/poll.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -116,7 +116,7 @@ impl<T> Poll<T> {
116116
/// let fut = Pin::new(&mut fut);
117117
///
118118
/// let num = fut.poll(cx).ready()?;
119-
/// # drop(num);
119+
/// # let _ = num; // to silence unused warning
120120
/// // ... use num
121121
///
122122
/// Poll::Ready(())

library/core/src/task/ready.rs

+2-2
Original file line numberDiff line numberDiff line change
@@ -22,7 +22,7 @@ use core::task::Poll;
2222
/// let fut = Pin::new(&mut fut);
2323
///
2424
/// let num = ready!(fut.poll(cx));
25-
/// # drop(num);
25+
/// # let _ = num;
2626
/// // ... use num
2727
///
2828
/// Poll::Ready(())
@@ -44,7 +44,7 @@ use core::task::Poll;
4444
/// Poll::Ready(t) => t,
4545
/// Poll::Pending => return Poll::Pending,
4646
/// };
47-
/// # drop(num);
47+
/// # let _ = num; // to silence unused warning
4848
/// # // ... use num
4949
/// #
5050
/// # Poll::Ready(())

library/std/src/panicking.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -541,7 +541,7 @@ pub fn begin_panic_handler(info: &PanicInfo<'_>) -> ! {
541541
// Lazily, the first time this gets called, run the actual string formatting.
542542
self.string.get_or_insert_with(|| {
543543
let mut s = String::new();
544-
drop(s.write_fmt(*inner));
544+
let _err = s.write_fmt(*inner);
545545
s
546546
})
547547
}

library/std/src/sys/sgx/waitqueue/mod.rs

+13-3
Original file line numberDiff line numberDiff line change
@@ -202,12 +202,18 @@ impl WaitQueue {
202202
pub fn notify_one<T>(
203203
mut guard: SpinMutexGuard<'_, WaitVariable<T>>,
204204
) -> Result<WaitGuard<'_, T>, SpinMutexGuard<'_, WaitVariable<T>>> {
205+
// SAFETY: lifetime of the pop() return value is limited to the map
206+
// closure (The closure return value is 'static). The underlying
207+
// stack frame won't be freed until after the WaitGuard created below
208+
// is dropped.
205209
unsafe {
206-
if let Some(entry) = guard.queue.inner.pop() {
210+
let tcs = guard.queue.inner.pop().map(|entry| -> Tcs {
207211
let mut entry_guard = entry.lock();
208-
let tcs = entry_guard.tcs;
209212
entry_guard.wake = true;
210-
drop(entry);
213+
entry_guard.tcs
214+
});
215+
216+
if let Some(tcs) = tcs {
211217
Ok(WaitGuard { mutex_guard: Some(guard), notified_tcs: NotifiedTcs::Single(tcs) })
212218
} else {
213219
Err(guard)
@@ -223,13 +229,17 @@ impl WaitQueue {
223229
pub fn notify_all<T>(
224230
mut guard: SpinMutexGuard<'_, WaitVariable<T>>,
225231
) -> Result<WaitGuard<'_, T>, SpinMutexGuard<'_, WaitVariable<T>>> {
232+
// SAFETY: lifetime of the pop() return values are limited to the
233+
// while loop body. The underlying stack frames won't be freed until
234+
// after the WaitGuard created below is dropped.
226235
unsafe {
227236
let mut count = 0;
228237
while let Some(entry) = guard.queue.inner.pop() {
229238
count += 1;
230239
let mut entry_guard = entry.lock();
231240
entry_guard.wake = true;
232241
}
242+
233243
if let Some(count) = NonZeroUsize::new(count) {
234244
Ok(WaitGuard { mutex_guard: Some(guard), notified_tcs: NotifiedTcs::All { count } })
235245
} else {

library/std/src/sys/unix/fs.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -1210,7 +1210,7 @@ impl File {
12101210
// Redox doesn't appear to support `UTIME_OMIT`.
12111211
// ESP-IDF and HorizonOS do not support `futimens` at all and the behavior for those OS is therefore
12121212
// the same as for Redox.
1213-
drop(times);
1213+
let _ = times;
12141214
Err(io::const_io_error!(
12151215
io::ErrorKind::Unsupported,
12161216
"setting file times not supported",

library/std/src/thread/tests.rs

+3-1
Original file line numberDiff line numberDiff line change
@@ -375,7 +375,9 @@ fn test_scoped_threads_nll() {
375375
// this is mostly a *compilation test* for this exact function:
376376
fn foo(x: &u8) {
377377
thread::scope(|s| {
378-
s.spawn(|| drop(x));
378+
s.spawn(|| match x {
379+
_ => (),
380+
});
379381
});
380382
}
381383
// let's also run it for good measure

0 commit comments

Comments
 (0)