diff options
| author | Patrick Walton <pcwalton@mimiga.net> | 2014-09-08 09:50:18 -0700 |
|---|---|---|
| committer | Patrick Walton <pcwalton@mimiga.net> | 2014-09-08 10:50:34 -0700 |
| commit | 3ca53d3a100a4466ed91df6d5a696d112db82b68 (patch) | |
| tree | 7d94650fab70c594c30e457cf76e976a63a7da83 | |
| parent | d8a26184dc103025e7ad457ffd8deb391019dbfe (diff) | |
| download | rust-3ca53d3a100a4466ed91df6d5a696d112db82b68.tar.gz rust-3ca53d3a100a4466ed91df6d5a696d112db82b68.zip | |
librustc: Make sure lifetimes in `for` loop heads outlive the `for` loop
itself.
This breaks code like:
for &x in my_vector.iter() {
my_vector[2] = "wibble";
...
}
Change this code to not invalidate iterators. For example:
for i in range(0, my_vector.len()) {
my_vector[2] = "wibble";
...
}
The `for-loop-does-not-borrow-iterators` test for #8372 was incorrect
and has been removed.
Closes #16820.
[breaking-change]
| -rw-r--r-- | src/librustc/middle/typeck/check/regionck.rs | 4 | ||||
| -rw-r--r-- | src/test/compile-fail/borrowck-for-loop-head-linkage.rs | 19 | ||||
| -rw-r--r-- | src/test/run-pass/for-loop-does-not-borrow-iterators.rs | 30 |
3 files changed, 23 insertions, 30 deletions
diff --git a/src/librustc/middle/typeck/check/regionck.rs b/src/librustc/middle/typeck/check/regionck.rs index 16ecaa9714e..951b42e5b80 100644 --- a/src/librustc/middle/typeck/check/regionck.rs +++ b/src/librustc/middle/typeck/check/regionck.rs @@ -769,6 +769,10 @@ fn visit_expr(rcx: &mut Rcx, expr: &ast::Expr) { } rcx.visit_expr(&**head, ()); + type_of_node_must_outlive(rcx, + infer::AddrOf(expr.span), + head.id, + ty::ReScope(expr.id)); let repeating_scope = rcx.set_repeating_scope(body.id); rcx.visit_block(&**body, ()); diff --git a/src/test/compile-fail/borrowck-for-loop-head-linkage.rs b/src/test/compile-fail/borrowck-for-loop-head-linkage.rs new file mode 100644 index 00000000000..600c0ac801b --- /dev/null +++ b/src/test/compile-fail/borrowck-for-loop-head-linkage.rs @@ -0,0 +1,19 @@ +// 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. + +fn main() { + let mut vector = vec![1u, 2]; + for &x in vector.iter() { + let cap = vector.capacity(); + vector.grow(cap, &0u); //~ ERROR cannot borrow + *vector.get_mut(1u) = 5u; //~ ERROR cannot borrow + } +} + diff --git a/src/test/run-pass/for-loop-does-not-borrow-iterators.rs b/src/test/run-pass/for-loop-does-not-borrow-iterators.rs deleted file mode 100644 index 206ab0dfeb2..00000000000 --- a/src/test/run-pass/for-loop-does-not-borrow-iterators.rs +++ /dev/null @@ -1,30 +0,0 @@ -// 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. - - -// The `for` loop use to keep a mutable borrow when executing its body, -// making it impossible to re-use the iterator as follows. -// https://github.com/rust-lang/rust/issues/8372 -// -// This was fixed in https://github.com/rust-lang/rust/pull/15809 - -pub fn main() { - let mut for_loop_values = Vec::new(); - let mut explicit_next_call_values = Vec::new(); - - let mut iter = range(1i, 10); - for i in iter { - for_loop_values.push(i); - explicit_next_call_values.push(iter.next()); - } - - assert_eq!(for_loop_values, vec![1, 3, 5, 7, 9]); - assert_eq!(explicit_next_call_values, vec![Some(2), Some(4), Some(6), Some(8), None]); -} |
