Skip to content

Commit 03c88aa

Browse files
committed
Tweak .clone() suggestion to work in more cases
When going through auto-deref, the `<T as Clone>` impl sometimes needs to be specified for rustc to actually clone the value and not the reference. ``` error[E0507]: cannot move out of dereference of `S` --> $DIR/needs-clone-through-deref.rs:15:18 | LL | for _ in self.clone().into_iter() {} | ^^^^^^^^^^^^ ----------- value moved due to this method call | | | move occurs because value has type `Vec<usize>`, which does not implement the `Copy` trait | note: `into_iter` takes ownership of the receiver `self`, which moves value --> $SRC_DIR/core/src/iter/traits/collect.rs:LL:COL help: you can `clone` the value and consume it, but this might not be your desired behavior | LL | for _ in <Vec<usize> as Clone>::clone(&self.clone()).into_iter() {} | ++++++++++++++++++++++++++++++ + ``` CC rust-lang#109429.
1 parent 0e2dac8 commit 03c88aa

12 files changed

+86
-18
lines changed

compiler/rustc_borrowck/src/diagnostics/mod.rs

+17-3
Original file line numberDiff line numberDiff line change
@@ -1135,11 +1135,25 @@ impl<'cx, 'tcx> MirBorrowckCtxt<'cx, 'tcx> {
11351135
)
11361136
&& self.infcx.predicate_must_hold_modulo_regions(&o)
11371137
{
1138-
err.span_suggestion_verbose(
1139-
move_span.shrink_to_hi(),
1138+
let sugg = if moved_place
1139+
.iter_projections()
1140+
.any(|(_, elem)| matches!(elem, ProjectionElem::Deref))
1141+
{
1142+
vec![
1143+
// We use the fully-qualified path because `.clone()` can
1144+
// sometimes choose `<&T as Clone>` instead of `<T as Clone>`
1145+
// when going through auto-deref, so this ensures that doesn't
1146+
// happen, causing suggestions for `.clone().clone()`.
1147+
(move_span.shrink_to_lo(), format!("<{ty} as Clone>::clone(&")),
1148+
(move_span.shrink_to_hi(), ")".to_string()),
1149+
]
1150+
} else {
1151+
vec![(move_span.shrink_to_hi(), ".clone()".to_string())]
1152+
};
1153+
err.multipart_suggestion_verbose(
11401154
"you can `clone` the value and consume it, but this might not be \
11411155
your desired behavior",
1142-
".clone()".to_string(),
1156+
sugg,
11431157
Applicability::MaybeIncorrect,
11441158
);
11451159
}

tests/ui/borrowck/borrowck-move-out-of-overloaded-auto-deref.stderr

+2-2
Original file line numberDiff line numberDiff line change
@@ -10,8 +10,8 @@ note: `into_iter` takes ownership of the receiver `self`, which moves value
1010
--> $SRC_DIR/core/src/iter/traits/collect.rs:LL:COL
1111
help: you can `clone` the value and consume it, but this might not be your desired behavior
1212
|
13-
LL | let _x = Rc::new(vec![1, 2]).clone().into_iter();
14-
| ++++++++
13+
LL | let _x = <Vec<i32> as Clone>::clone(&Rc::new(vec![1, 2])).into_iter();
14+
| ++++++++++++++++++++++++++++ +
1515

1616
error: aborting due to 1 previous error
1717

tests/ui/borrowck/clone-span-on-try-operator.fixed

+1-1
Original file line numberDiff line numberDiff line change
@@ -7,5 +7,5 @@ impl Foo {
77
}
88
fn main() {
99
let foo = &Foo;
10-
(*foo).clone().foo(); //~ ERROR cannot move out
10+
<Foo as Clone>::clone(&(*foo)).foo(); //~ ERROR cannot move out
1111
}

tests/ui/borrowck/clone-span-on-try-operator.stderr

+2-2
Original file line numberDiff line numberDiff line change
@@ -13,8 +13,8 @@ LL | fn foo(self) {}
1313
| ^^^^
1414
help: you can `clone` the value and consume it, but this might not be your desired behavior
1515
|
16-
LL | (*foo).clone().foo();
17-
| ++++++++
16+
LL | <Foo as Clone>::clone(&(*foo)).foo();
17+
| +++++++++++++++++++++++ +
1818

1919
error: aborting due to 1 previous error
2020

tests/ui/borrowck/unboxed-closures-move-upvar-from-non-once-ref-closure.fixed

+1-1
Original file line numberDiff line numberDiff line change
@@ -9,7 +9,7 @@ fn call<F>(f: F) where F : Fn() {
99
fn main() {
1010
let y = vec![format!("World")];
1111
call(|| {
12-
y.clone().into_iter();
12+
<Vec<String> as Clone>::clone(&y).into_iter();
1313
//~^ ERROR cannot move out of `y`, a captured variable in an `Fn` closure
1414
});
1515
}

tests/ui/borrowck/unboxed-closures-move-upvar-from-non-once-ref-closure.stderr

+2-2
Original file line numberDiff line numberDiff line change
@@ -14,8 +14,8 @@ note: `into_iter` takes ownership of the receiver `self`, which moves `y`
1414
--> $SRC_DIR/core/src/iter/traits/collect.rs:LL:COL
1515
help: you can `clone` the value and consume it, but this might not be your desired behavior
1616
|
17-
LL | y.clone().into_iter();
18-
| ++++++++
17+
LL | <Vec<String> as Clone>::clone(&y).into_iter();
18+
| +++++++++++++++++++++++++++++++ +
1919

2020
error: aborting due to 1 previous error
2121

Original file line numberDiff line numberDiff line change
@@ -0,0 +1,18 @@
1+
// run-rustfix
2+
#![allow(dead_code, noop_method_call)]
3+
use std::ops::Deref;
4+
struct S(Vec<usize>);
5+
impl Deref for S {
6+
type Target = Vec<usize>;
7+
fn deref(&self) -> &Self::Target {
8+
&self.0
9+
}
10+
}
11+
12+
impl S {
13+
fn foo(&self) {
14+
// `self.clone()` returns `&S`, not `Vec`
15+
for _ in <Vec<usize> as Clone>::clone(&self.clone()).into_iter() {} //~ ERROR cannot move out of dereference of `S`
16+
}
17+
}
18+
fn main() {}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,18 @@
1+
// run-rustfix
2+
#![allow(dead_code, noop_method_call)]
3+
use std::ops::Deref;
4+
struct S(Vec<usize>);
5+
impl Deref for S {
6+
type Target = Vec<usize>;
7+
fn deref(&self) -> &Self::Target {
8+
&self.0
9+
}
10+
}
11+
12+
impl S {
13+
fn foo(&self) {
14+
// `self.clone()` returns `&S`, not `Vec`
15+
for _ in self.clone().into_iter() {} //~ ERROR cannot move out of dereference of `S`
16+
}
17+
}
18+
fn main() {}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,18 @@
1+
error[E0507]: cannot move out of dereference of `S`
2+
--> $DIR/needs-clone-through-deref.rs:15:18
3+
|
4+
LL | for _ in self.clone().into_iter() {}
5+
| ^^^^^^^^^^^^ ----------- value moved due to this method call
6+
| |
7+
| move occurs because value has type `Vec<usize>`, which does not implement the `Copy` trait
8+
|
9+
note: `into_iter` takes ownership of the receiver `self`, which moves value
10+
--> $SRC_DIR/core/src/iter/traits/collect.rs:LL:COL
11+
help: you can `clone` the value and consume it, but this might not be your desired behavior
12+
|
13+
LL | for _ in <Vec<usize> as Clone>::clone(&self.clone()).into_iter() {}
14+
| ++++++++++++++++++++++++++++++ +
15+
16+
error: aborting due to previous error
17+
18+
For more information about this error, try `rustc --explain E0507`.

tests/ui/moves/suggest-clone.fixed

+1-1
Original file line numberDiff line numberDiff line change
@@ -7,5 +7,5 @@ impl Foo {
77
}
88
fn main() {
99
let foo = &Foo;
10-
foo.clone().foo(); //~ ERROR cannot move out
10+
<Foo as Clone>::clone(&foo).foo(); //~ ERROR cannot move out
1111
}

tests/ui/moves/suggest-clone.stderr

+2-2
Original file line numberDiff line numberDiff line change
@@ -13,8 +13,8 @@ LL | fn foo(self) {}
1313
| ^^^^
1414
help: you can `clone` the value and consume it, but this might not be your desired behavior
1515
|
16-
LL | foo.clone().foo();
17-
| ++++++++
16+
LL | <Foo as Clone>::clone(&foo).foo();
17+
| +++++++++++++++++++++++ +
1818

1919
error: aborting due to 1 previous error
2020

tests/ui/suggestions/option-content-move.stderr

+4-4
Original file line numberDiff line numberDiff line change
@@ -11,8 +11,8 @@ note: `Option::<T>::unwrap` takes ownership of the receiver `self`, which moves
1111
--> $SRC_DIR/core/src/option.rs:LL:COL
1212
help: you can `clone` the value and consume it, but this might not be your desired behavior
1313
|
14-
LL | if selection.1.clone().unwrap().contains(selection.0) {
15-
| ++++++++
14+
LL | if <Option<String> as Clone>::clone(&selection.1).unwrap().contains(selection.0) {
15+
| ++++++++++++++++++++++++++++++++++ +
1616

1717
error[E0507]: cannot move out of `selection.1` which is behind a shared reference
1818
--> $DIR/option-content-move.rs:27:20
@@ -27,8 +27,8 @@ note: `Result::<T, E>::unwrap` takes ownership of the receiver `self`, which mov
2727
--> $SRC_DIR/core/src/result.rs:LL:COL
2828
help: you can `clone` the value and consume it, but this might not be your desired behavior
2929
|
30-
LL | if selection.1.clone().unwrap().contains(selection.0) {
31-
| ++++++++
30+
LL | if <Result<String, String> as Clone>::clone(&selection.1).unwrap().contains(selection.0) {
31+
| ++++++++++++++++++++++++++++++++++++++++++ +
3232

3333
error: aborting due to 2 previous errors
3434

0 commit comments

Comments
 (0)