Skip to content

Commit 6136381

Browse files
committedAug 1, 2014
auto merge of #16102 : zwarich/rust/borrowck-unboxed, r=pcwalton
This removes the ability of the borrow checker to determine that repeated dereferences of a Box<T> refer to the same memory object.
2 parents f2fa559 + 5d4d09d commit 6136381

File tree

7 files changed

+231
-17
lines changed

7 files changed

+231
-17
lines changed
 

‎src/libcollections/treemap.rs

+2-1
Original file line numberDiff line numberDiff line change
@@ -1564,7 +1564,8 @@ fn remove<K: Ord, V>(node: &mut Option<Box<TreeNode<K, V>>>,
15641564
save.level -= 1;
15651565

15661566
if right_level > save.level {
1567-
for x in save.right.mut_iter() { x.level = save.level }
1567+
let save_level = save.level;
1568+
for x in save.right.mut_iter() { x.level = save_level }
15681569
}
15691570

15701571
skew(save);

‎src/libcollections/trie.rs

+13-7
Original file line numberDiff line numberDiff line change
@@ -783,7 +783,7 @@ fn insert<T>(count: &mut uint, child: &mut Child<T>, key: uint, value: T,
783783
*child = External(key, value);
784784
return None;
785785
}
786-
Internal(ref mut x) => {
786+
Internal(box ref mut x) => {
787787
return insert(&mut x.count, &mut x.children[chunk(key, idx)], key, value, idx + 1);
788788
}
789789
External(stored_key, ref mut stored_value) if stored_key == key => {
@@ -799,11 +799,17 @@ fn insert<T>(count: &mut uint, child: &mut Child<T>, key: uint, value: T,
799799
match mem::replace(child, Nothing) {
800800
External(stored_key, stored_value) => {
801801
let mut new = box TrieNode::new();
802-
insert(&mut new.count,
803-
&mut new.children[chunk(stored_key, idx)],
804-
stored_key, stored_value, idx + 1);
805-
let ret = insert(&mut new.count, &mut new.children[chunk(key, idx)],
806-
key, value, idx + 1);
802+
803+
let ret = {
804+
let new_interior = &mut *new;
805+
insert(&mut new_interior.count,
806+
&mut new_interior.children[chunk(stored_key, idx)],
807+
stored_key, stored_value, idx + 1);
808+
insert(&mut new_interior.count,
809+
&mut new_interior.children[chunk(key, idx)],
810+
key, value, idx + 1)
811+
};
812+
807813
*child = Internal(new);
808814
return ret;
809815
}
@@ -821,7 +827,7 @@ fn remove<T>(count: &mut uint, child: &mut Child<T>, key: uint,
821827
}
822828
}
823829
External(..) => (None, false),
824-
Internal(ref mut x) => {
830+
Internal(box ref mut x) => {
825831
let ret = remove(&mut x.count, &mut x.children[chunk(key, idx)],
826832
key, idx + 1);
827833
(ret, x.count == 0)

‎src/librustc/middle/borrowck/check_loans.rs

+56-2
Original file line numberDiff line numberDiff line change
@@ -28,6 +28,57 @@ use util::ppaux::Repr;
2828

2929
use std::rc::Rc;
3030

31+
// FIXME (#16118): These functions are intended to allow the borrow checker to
32+
// be less precise in its handling of Box while still allowing moves out of a
33+
// Box. They should be removed when OwnedPtr is removed from LoanPath.
34+
35+
fn owned_ptr_base_path<'a>(loan_path: &'a LoanPath) -> &'a LoanPath {
36+
//! Returns the base of the leftmost dereference of an OwnedPtr in
37+
//! `loan_path`. If there is no dereference of an OwnedPtr in `loan_path`,
38+
//! then it just returns `loan_path` itself.
39+
40+
return match owned_ptr_base_path_helper(loan_path) {
41+
Some(new_loan_path) => new_loan_path,
42+
None => loan_path.clone()
43+
};
44+
45+
fn owned_ptr_base_path_helper<'a>(loan_path: &'a LoanPath) -> Option<&'a LoanPath> {
46+
match *loan_path {
47+
LpVar(_) | LpUpvar(_) => None,
48+
LpExtend(ref lp_base, _, LpDeref(mc::OwnedPtr)) => {
49+
match owned_ptr_base_path_helper(&**lp_base) {
50+
v @ Some(_) => v,
51+
None => Some(&**lp_base)
52+
}
53+
}
54+
LpExtend(ref lp_base, _, _) => owned_ptr_base_path_helper(&**lp_base)
55+
}
56+
}
57+
}
58+
59+
fn owned_ptr_base_path_rc(loan_path: &Rc<LoanPath>) -> Rc<LoanPath> {
60+
//! The equivalent of `owned_ptr_base_path` for an &Rc<LoanPath> rather than
61+
//! a &LoanPath.
62+
63+
return match owned_ptr_base_path_helper(loan_path) {
64+
Some(new_loan_path) => new_loan_path,
65+
None => loan_path.clone()
66+
};
67+
68+
fn owned_ptr_base_path_helper(loan_path: &Rc<LoanPath>) -> Option<Rc<LoanPath>> {
69+
match **loan_path {
70+
LpVar(_) | LpUpvar(_) => None,
71+
LpExtend(ref lp_base, _, LpDeref(mc::OwnedPtr)) => {
72+
match owned_ptr_base_path_helper(lp_base) {
73+
v @ Some(_) => v,
74+
None => Some(lp_base.clone())
75+
}
76+
}
77+
LpExtend(ref lp_base, _, _) => owned_ptr_base_path_helper(lp_base)
78+
}
79+
}
80+
}
81+
3182
struct CheckLoanCtxt<'a> {
3283
bccx: &'a BorrowckCtxt<'a>,
3384
dfcx_loans: &'a LoanDataFlow<'a>,
@@ -210,6 +261,7 @@ impl<'a> CheckLoanCtxt<'a> {
210261
// let x = &mut a.b.c; // Restricts a, a.b, and a.b.c
211262
// let y = a; // Conflicts with restriction
212263

264+
let loan_path = owned_ptr_base_path(loan_path);
213265
let cont = self.each_in_scope_loan(scope_id, |loan| {
214266
let mut ret = true;
215267
for restr_path in loan.restricted_paths.iter() {
@@ -344,8 +396,9 @@ impl<'a> CheckLoanCtxt<'a> {
344396
return true;
345397
}
346398

399+
let loan2_base_path = owned_ptr_base_path_rc(&loan2.loan_path);
347400
for restr_path in loan1.restricted_paths.iter() {
348-
if *restr_path != loan2.loan_path { continue; }
401+
if *restr_path != loan2_base_path { continue; }
349402

350403
let old_pronoun = if new_loan.loan_path == old_loan.loan_path {
351404
"it".to_string()
@@ -597,7 +650,8 @@ impl<'a> CheckLoanCtxt<'a> {
597650

598651
debug!("check_if_path_is_moved(id={:?}, use_kind={:?}, lp={})",
599652
id, use_kind, lp.repr(self.bccx.tcx));
600-
self.move_data.each_move_of(id, lp, |move, moved_lp| {
653+
let base_lp = owned_ptr_base_path_rc(lp);
654+
self.move_data.each_move_of(id, &base_lp, |move, moved_lp| {
601655
self.bccx.report_use_of_moved_value(
602656
span,
603657
use_kind,

‎src/libstd/collections/lru_cache.rs

+2-1
Original file line numberDiff line numberDiff line change
@@ -331,7 +331,8 @@ impl<K, V> Drop for LruCache<K, V> {
331331
unsafe {
332332
let node: Box<LruEntry<K, V>> = mem::transmute(self.head);
333333
// Prevent compiler from trying to drop the un-initialized field in the sigil node.
334-
let box LruEntry { key: k, value: v, .. } = node;
334+
let box internal_node = node;
335+
let LruEntry { next: _, prev: _, key: k, value: v } = internal_node;
335336
mem::forget(k);
336337
mem::forget(v);
337338
}

‎src/test/compile-fail/borrowck-borrow-from-owned-ptr.rs

+5-4
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
// Copyright 2012 The Rust Project Developers. See the COPYRIGHT
1+
// Copyright 2014 The Rust Project Developers. See the COPYRIGHT
22
// file at the top-level directory of this distribution and at
33
// http://rust-lang.org/COPYRIGHT.
44
//
@@ -52,14 +52,15 @@ fn borrow_same_field_twice_imm_imm() {
5252
fn borrow_both_fields_mut() {
5353
let mut foo = make_foo();
5454
let bar1 = &mut foo.bar1;
55-
let _bar2 = &mut foo.bar2;
55+
let _bar2 = &mut foo.bar2; //~ ERROR cannot borrow
5656
*bar1;
5757
}
5858

5959
fn borrow_both_mut_pattern() {
6060
let mut foo = make_foo();
6161
match *foo {
6262
Foo { bar1: ref mut _bar1, bar2: ref mut _bar2 } => {}
63+
//~^ ERROR cannot borrow
6364
}
6465
}
6566

@@ -120,7 +121,7 @@ fn borrow_imm_and_base_imm() {
120121
fn borrow_mut_and_imm() {
121122
let mut foo = make_foo();
122123
let bar1 = &mut foo.bar1;
123-
let _foo1 = &foo.bar2;
124+
let _foo1 = &foo.bar2; //~ ERROR cannot borrow
124125
*bar1;
125126
}
126127

@@ -133,7 +134,7 @@ fn borrow_mut_from_imm() {
133134
fn borrow_long_path_both_mut() {
134135
let mut foo = make_foo();
135136
let bar1 = &mut foo.bar1.int1;
136-
let foo1 = &mut foo.bar2.int2;
137+
let foo1 = &mut foo.bar2.int2; //~ ERROR cannot borrow
137138
*bar1;
138139
*foo1;
139140
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,150 @@
1+
// Copyright 2014 The Rust Project Developers. See the COPYRIGHT
2+
// file at the top-level directory of this distribution and at
3+
// http://rust-lang.org/COPYRIGHT.
4+
//
5+
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
6+
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
7+
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
8+
// option. This file may not be copied, modified, or distributed
9+
// except according to those terms.
10+
11+
struct A {
12+
x: Box<int>,
13+
y: int,
14+
}
15+
16+
struct B {
17+
x: Box<int>,
18+
y: Box<int>,
19+
}
20+
21+
struct C {
22+
x: Box<A>,
23+
y: int,
24+
}
25+
26+
struct D {
27+
x: Box<A>,
28+
y: Box<int>,
29+
}
30+
31+
fn copy_after_move() {
32+
let a = box A { x: box 0, y: 1 };
33+
let _x = a.x;
34+
let _y = a.y; //~ ERROR use of partially moved
35+
}
36+
37+
fn move_after_move() {
38+
let a = box B { x: box 0, y: box 1 };
39+
let _x = a.x;
40+
let _y = a.y; //~ ERROR use of partially moved
41+
}
42+
43+
fn borrow_after_move() {
44+
let a = box A { x: box 0, y: 1 };
45+
let _x = a.x;
46+
let _y = &a.y; //~ ERROR use of partially moved
47+
}
48+
49+
fn move_after_borrow() {
50+
let a = box B { x: box 0, y: box 1 };
51+
let _x = &a.x;
52+
let _y = a.y; //~ ERROR cannot move
53+
}
54+
55+
fn copy_after_mut_borrow() {
56+
let mut a = box A { x: box 0, y: 1 };
57+
let _x = &mut a.x;
58+
let _y = a.y; //~ ERROR cannot use
59+
}
60+
61+
fn move_after_mut_borrow() {
62+
let mut a = box B { x: box 0, y: box 1 };
63+
let _x = &mut a.x;
64+
let _y = a.y; //~ ERROR cannot move
65+
}
66+
67+
fn borrow_after_mut_borrow() {
68+
let mut a = box A { x: box 0, y: 1 };
69+
let _x = &mut a.x;
70+
let _y = &a.y; //~ ERROR cannot borrow
71+
}
72+
73+
fn mut_borrow_after_borrow() {
74+
let mut a = box A { x: box 0, y: 1 };
75+
let _x = &a.x;
76+
let _y = &mut a.y; //~ ERROR cannot borrow
77+
}
78+
79+
fn copy_after_move_nested() {
80+
let a = box C { x: box A { x: box 0, y: 1 }, y: 2 };
81+
let _x = a.x.x;
82+
let _y = a.y; //~ ERROR use of partially moved
83+
}
84+
85+
fn move_after_move_nested() {
86+
let a = box D { x: box A { x: box 0, y: 1 }, y: box 2 };
87+
let _x = a.x.x;
88+
let _y = a.y; //~ ERROR use of partially moved
89+
}
90+
91+
fn borrow_after_move_nested() {
92+
let a = box C { x: box A { x: box 0, y: 1 }, y: 2 };
93+
let _x = a.x.x;
94+
let _y = &a.y; //~ ERROR use of partially moved
95+
}
96+
97+
fn move_after_borrow_nested() {
98+
let a = box D { x: box A { x: box 0, y: 1 }, y: box 2 };
99+
let _x = &a.x.x;
100+
let _y = a.y; //~ ERROR cannot move
101+
}
102+
103+
fn copy_after_mut_borrow_nested() {
104+
let mut a = box C { x: box A { x: box 0, y: 1 }, y: 2 };
105+
let _x = &mut a.x.x;
106+
let _y = a.y; //~ ERROR cannot use
107+
}
108+
109+
fn move_after_mut_borrow_nested() {
110+
let mut a = box D { x: box A { x: box 0, y: 1 }, y: box 2 };
111+
let _x = &mut a.x.x;
112+
let _y = a.y; //~ ERROR cannot move
113+
}
114+
115+
fn borrow_after_mut_borrow_nested() {
116+
let mut a = box C { x: box A { x: box 0, y: 1 }, y: 2 };
117+
let _x = &mut a.x.x;
118+
let _y = &a.y; //~ ERROR cannot borrow
119+
}
120+
121+
fn mut_borrow_after_borrow_nested() {
122+
let mut a = box C { x: box A { x: box 0, y: 1 }, y: 2 };
123+
let _x = &a.x.x;
124+
let _y = &mut a.y; //~ ERROR cannot borrow
125+
}
126+
127+
fn main() {
128+
copy_after_move();
129+
move_after_move();
130+
borrow_after_move();
131+
132+
move_after_borrow();
133+
134+
copy_after_mut_borrow();
135+
move_after_mut_borrow();
136+
borrow_after_mut_borrow();
137+
mut_borrow_after_borrow();
138+
139+
copy_after_move_nested();
140+
move_after_move_nested();
141+
borrow_after_move_nested();
142+
143+
move_after_borrow_nested();
144+
145+
copy_after_mut_borrow_nested();
146+
move_after_mut_borrow_nested();
147+
borrow_after_mut_borrow_nested();
148+
mut_borrow_after_borrow_nested();
149+
}
150+

‎src/test/run-pass/match-implicit-copy-unique.rs

+3-2
Original file line numberDiff line numberDiff line change
@@ -13,8 +13,9 @@ struct Pair { a: Box<int>, b: Box<int> }
1313

1414
pub fn main() {
1515
let mut x = box Pair {a: box 10, b: box 20};
16-
match x {
17-
box Pair {a: ref mut a, b: ref mut _b} => {
16+
let x_internal = &mut *x;
17+
match *x_internal {
18+
Pair {a: ref mut a, b: ref mut _b} => {
1819
assert!(**a == 10); *a = box 30; assert!(**a == 30);
1920
}
2021
}

0 commit comments

Comments
 (0)
Please sign in to comment.