Skip to content

Commit 83d0682

Browse files
committed
Auto merge of #11061 - Alexendoo:let-and-return-closures, r=llogiq
`let_and_return`: lint 'static lifetimes, don't lint borrows in closures Fixes #11056 Now also ignores functions returning `'static` lifetimes, since I noticed the `stdin.lock()` example was still being linted but doesn't need to be since rust-lang/rust#93965 changelog: none
2 parents 17a48c2 + e29a5ac commit 83d0682

File tree

3 files changed

+53
-36
lines changed

3 files changed

+53
-36
lines changed

clippy_lints/src/returns.rs

+3-3
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
use clippy_utils::diagnostics::{span_lint_and_then, span_lint_hir_and_then};
22
use clippy_utils::source::{snippet_opt, snippet_with_context};
3-
use clippy_utils::visitors::{for_each_expr, Descend};
3+
use clippy_utils::visitors::{for_each_expr_with_closures, Descend};
44
use clippy_utils::{fn_def_id, path_to_local_id, span_find_starting_semi};
55
use core::ops::ControlFlow;
66
use if_chain::if_chain;
@@ -328,7 +328,7 @@ fn emit_return_lint(cx: &LateContext<'_>, ret_span: Span, semi_spans: Vec<Span>,
328328
}
329329

330330
fn last_statement_borrows<'tcx>(cx: &LateContext<'tcx>, expr: &'tcx Expr<'tcx>) -> bool {
331-
for_each_expr(expr, |e| {
331+
for_each_expr_with_closures(cx, expr, |e| {
332332
if let Some(def_id) = fn_def_id(cx, e)
333333
&& cx
334334
.tcx
@@ -337,7 +337,7 @@ fn last_statement_borrows<'tcx>(cx: &LateContext<'tcx>, expr: &'tcx Expr<'tcx>)
337337
.skip_binder()
338338
.output()
339339
.walk()
340-
.any(|arg| matches!(arg.unpack(), GenericArgKind::Lifetime(_)))
340+
.any(|arg| matches!(arg.unpack(), GenericArgKind::Lifetime(re) if !re.is_static()))
341341
{
342342
ControlFlow::Break(())
343343
} else {

tests/ui/let_and_return.rs

+32-29
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,8 @@
11
#![allow(unused)]
22
#![warn(clippy::let_and_return)]
33

4+
use std::cell::RefCell;
5+
46
fn test() -> i32 {
57
let _y = 0; // no warning
68
let x = 5;
@@ -65,45 +67,46 @@ macro_rules! tuple_encode {
6567
);
6668
}
6769

70+
fn issue_3792() -> String {
71+
use std::io::{self, BufRead, Stdin};
72+
73+
let stdin = io::stdin();
74+
// `Stdin::lock` returns `StdinLock<'static>` so `line` doesn't borrow from `stdin`
75+
// https://github.com/rust-lang/rust/pull/93965
76+
let line = stdin.lock().lines().next().unwrap().unwrap();
77+
line
78+
}
79+
6880
tuple_encode!(T0, T1, T2, T3, T4, T5, T6, T7);
6981

7082
mod no_lint_if_stmt_borrows {
71-
mod issue_3792 {
72-
use std::io::{self, BufRead, Stdin};
83+
use std::cell::RefCell;
84+
use std::rc::{Rc, Weak};
85+
struct Bar;
7386

74-
fn read_line() -> String {
75-
let stdin = io::stdin();
76-
let line = stdin.lock().lines().next().unwrap().unwrap();
77-
line
87+
impl Bar {
88+
fn new() -> Self {
89+
Bar {}
7890
}
79-
}
80-
81-
mod issue_3324 {
82-
use std::cell::RefCell;
83-
use std::rc::{Rc, Weak};
84-
85-
fn test(value: Weak<RefCell<Bar>>) -> u32 {
86-
let value = value.upgrade().unwrap();
87-
let ret = value.borrow().baz();
88-
ret
91+
fn baz(&self) -> u32 {
92+
0
8993
}
94+
}
9095

91-
struct Bar;
96+
fn issue_3324(value: Weak<RefCell<Bar>>) -> u32 {
97+
let value = value.upgrade().unwrap();
98+
let ret = value.borrow().baz();
99+
ret
100+
}
92101

93-
impl Bar {
94-
fn new() -> Self {
95-
Bar {}
96-
}
97-
fn baz(&self) -> u32 {
98-
0
99-
}
102+
fn borrows_in_closure(value: Weak<RefCell<Bar>>) -> u32 {
103+
fn f(mut x: impl FnMut() -> u32) -> impl FnMut() -> u32 {
104+
x
100105
}
101106

102-
fn main() {
103-
let a = Rc::new(RefCell::new(Bar::new()));
104-
let b = Rc::downgrade(&a);
105-
test(b);
106-
}
107+
let value = value.upgrade().unwrap();
108+
let ret = f(|| value.borrow().baz())();
109+
ret
107110
}
108111

109112
mod free_function {

tests/ui/let_and_return.stderr

+18-4
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
error: returning the result of a `let` binding from a block
2-
--> $DIR/let_and_return.rs:7:5
2+
--> $DIR/let_and_return.rs:9:5
33
|
44
LL | let x = 5;
55
| ---------- unnecessary `let` binding
@@ -14,7 +14,7 @@ LL ~ 5
1414
|
1515

1616
error: returning the result of a `let` binding from a block
17-
--> $DIR/let_and_return.rs:13:9
17+
--> $DIR/let_and_return.rs:15:9
1818
|
1919
LL | let x = 5;
2020
| ---------- unnecessary `let` binding
@@ -28,7 +28,21 @@ LL ~ 5
2828
|
2929

3030
error: returning the result of a `let` binding from a block
31-
--> $DIR/let_and_return.rs:164:13
31+
--> $DIR/let_and_return.rs:77:5
32+
|
33+
LL | let line = stdin.lock().lines().next().unwrap().unwrap();
34+
| --------------------------------------------------------- unnecessary `let` binding
35+
LL | line
36+
| ^^^^
37+
|
38+
help: return the expression directly
39+
|
40+
LL ~
41+
LL ~ stdin.lock().lines().next().unwrap().unwrap()
42+
|
43+
44+
error: returning the result of a `let` binding from a block
45+
--> $DIR/let_and_return.rs:167:13
3246
|
3347
LL | let clone = Arc::clone(&self.foo);
3448
| ---------------------------------- unnecessary `let` binding
@@ -41,5 +55,5 @@ LL ~
4155
LL ~ Arc::clone(&self.foo) as _
4256
|
4357

44-
error: aborting due to 3 previous errors
58+
error: aborting due to 4 previous errors
4559

0 commit comments

Comments
 (0)