2014-02-05 16:33:10 -06:00
|
|
|
// Copyright 2014 The Rust Project Developers. See the COPYRIGHT
|
|
|
|
// file at the top-level directory of this distribution and at
|
|
|
|
// http://rust-lang.org/COPYRIGHT.
|
|
|
|
//
|
|
|
|
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
|
|
|
|
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
|
|
|
|
// option. This file may not be copied, modified, or distributed
|
|
|
|
// except according to those terms.
|
|
|
|
|
2013-05-22 05:54:35 -05:00
|
|
|
// Test that we do not permit moves from &[] matched by a vec pattern.
|
|
|
|
|
2015-03-26 20:34:27 -05:00
|
|
|
#![feature(slice_patterns)]
|
|
|
|
|
2015-01-28 07:34:18 -06:00
|
|
|
#[derive(Clone, Debug)]
|
2013-05-22 05:54:35 -05:00
|
|
|
struct Foo {
|
2014-05-22 18:57:53 -05:00
|
|
|
string: String
|
2013-05-22 05:54:35 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn main() {
|
2014-03-05 16:02:44 -06:00
|
|
|
let x = vec!(
|
2014-05-25 05:17:19 -05:00
|
|
|
Foo { string: "foo".to_string() },
|
|
|
|
Foo { string: "bar".to_string() },
|
|
|
|
Foo { string: "baz".to_string() }
|
2014-03-05 16:02:44 -06:00
|
|
|
);
|
2015-02-01 20:53:25 -06:00
|
|
|
let x: &[Foo] = &x;
|
2013-05-22 05:54:35 -05:00
|
|
|
match x {
|
2014-09-06 17:23:55 -05:00
|
|
|
[_, tail..] => {
|
2013-05-22 05:54:35 -05:00
|
|
|
match tail {
|
2015-01-08 08:12:06 -06:00
|
|
|
[Foo { string: a }, //~ ERROR cannot move out of borrowed content
|
Collect move errors before reporting
This commit changes the way move errors are reported when some value is
captured by a PatIdent. First, we collect all of the "cannot move out
of" errors before reporting them, and those errors with the same "move
source" are reported together. If the move is caused by a PatIdent (that
binds by value), we add a note indicating where it is and suggest the
user to put `ref` if they don't want the value to move. This makes the
"cannot move out of" error in match expression nicer (though the extra
note may not feel that helpful in other places :P). For example, with
the following code snippet,
```rust
enum Foo {
Foo1(~u32, ~u32),
Foo2(~u32),
Foo3,
}
fn main() {
let f = &Foo1(~1u32, ~2u32);
match *f {
Foo1(num1, num2) => (),
Foo2(num) => (),
Foo3 => ()
}
}
```
Errors before the change:
```rust
test.rs:10:9: 10:25 error: cannot move out of dereference of `&`-pointer
test.rs:10 Foo1(num1, num2) => (),
^~~~~~~~~~~~~~~~
test.rs:10:9: 10:25 error: cannot move out of dereference of `&`-pointer
test.rs:10 Foo1(num1, num2) => (),
^~~~~~~~~~~~~~~~
test.rs:11:9: 11:18 error: cannot move out of dereference of `&`-pointer
test.rs:11 Foo2(num) => (),
^~~~~~~~~
```
After:
```rust
test.rs:9:11: 9:13 error: cannot move out of dereference of `&`-pointer
test.rs:9 match *f {
^~
test.rs:10:14: 10:18 note: attempting to move value to here (to prevent the move, you can use `ref num1` to capture value by reference)
test.rs:10 Foo1(num1, num2) => (),
^~~~
test.rs:10:20: 10:24 note: and here (use `ref num2`)
test.rs:10 Foo1(num1, num2) => (),
^~~~
test.rs:11:14: 11:17 note: and here (use `ref num`)
test.rs:11 Foo2(num) => (),
^~~
```
Close #8064
2014-04-08 20:14:14 -05:00
|
|
|
Foo { string: b }] => {
|
|
|
|
//~^^ NOTE attempting to move value to here
|
|
|
|
//~^^ NOTE and here
|
2013-05-22 05:54:35 -05:00
|
|
|
}
|
|
|
|
_ => {
|
2013-09-19 00:04:03 -05:00
|
|
|
unreachable!();
|
2013-05-22 05:54:35 -05:00
|
|
|
}
|
|
|
|
}
|
2013-07-02 14:47:32 -05:00
|
|
|
let z = tail[0].clone();
|
2014-12-20 02:09:35 -06:00
|
|
|
println!("{:?}", z);
|
2013-05-22 05:54:35 -05:00
|
|
|
}
|
|
|
|
_ => {
|
2013-09-19 00:04:03 -05:00
|
|
|
unreachable!();
|
2013-05-22 05:54:35 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|