// This test is an artifact of the old policy that `Box` should not // be treated specially by the AST-borrowck. // // NLL goes back to treating `Box` specially (namely, knowing that // it uniquely owns the data it holds). See rust-lang/rfcs#130. // revisions: ast mir //[ast] compile-flags: -Z borrowck=ast //[mir] compile-flags: -Z borrowck=mir // ignore-compare-mode-nll #![feature(box_syntax, rustc_attrs)] struct A { x: Box, y: isize, } struct B { x: Box, y: Box, } struct C { x: Box, y: isize, } struct D { x: Box, y: Box, } fn copy_after_move() { let a: Box<_> = box A { x: box 0, y: 1 }; let _x = a.x; //[ast]~^ value moved here let _y = a.y; //[ast]~ ERROR use of moved //[ast]~^ move occurs because `a.x` has type `std::boxed::Box` //[ast]~| value used here after move } fn move_after_move() { let a: Box<_> = box B { x: box 0, y: box 1 }; let _x = a.x; //[ast]~^ value moved here let _y = a.y; //[ast]~ ERROR use of moved //[ast]~^ move occurs because `a.x` has type `std::boxed::Box` //[ast]~| value used here after move } fn borrow_after_move() { let a: Box<_> = box A { x: box 0, y: 1 }; let _x = a.x; //[ast]~^ value moved here let _y = &a.y; //[ast]~ ERROR use of moved //[ast]~^ move occurs because `a.x` has type `std::boxed::Box` //[ast]~| value used here after move } fn move_after_borrow() { let a: Box<_> = box B { x: box 0, y: box 1 }; let _x = &a.x; let _y = a.y; //[ast]~^ ERROR cannot move //[ast]~| move out of use_imm(_x); } fn copy_after_mut_borrow() { let mut a: Box<_> = box A { x: box 0, y: 1 }; let _x = &mut a.x; let _y = a.y; //[ast]~ ERROR cannot use use_mut(_x); } fn move_after_mut_borrow() { let mut a: Box<_> = box B { x: box 0, y: box 1 }; let _x = &mut a.x; let _y = a.y; //[ast]~^ ERROR cannot move //[ast]~| move out of use_mut(_x); } fn borrow_after_mut_borrow() { let mut a: Box<_> = box A { x: box 0, y: 1 }; let _x = &mut a.x; let _y = &a.y; //[ast]~ ERROR cannot borrow //[ast]~^ immutable borrow occurs here (via `a.y`) use_mut(_x); } fn mut_borrow_after_borrow() { let mut a: Box<_> = box A { x: box 0, y: 1 }; let _x = &a.x; let _y = &mut a.y; //[ast]~ ERROR cannot borrow //[ast]~^ mutable borrow occurs here (via `a.y`) use_imm(_x); } fn copy_after_move_nested() { let a: Box<_> = box C { x: box A { x: box 0, y: 1 }, y: 2 }; let _x = a.x.x; //[ast]~^ value moved here let _y = a.y; //[ast]~ ERROR use of collaterally moved //[ast]~| value used here after move } fn move_after_move_nested() { let a: Box<_> = box D { x: box A { x: box 0, y: 1 }, y: box 2 }; let _x = a.x.x; //[ast]~^ value moved here let _y = a.y; //[ast]~ ERROR use of collaterally moved //[ast]~| value used here after move } fn borrow_after_move_nested() { let a: Box<_> = box C { x: box A { x: box 0, y: 1 }, y: 2 }; let _x = a.x.x; //[ast]~^ value moved here let _y = &a.y; //[ast]~ ERROR use of collaterally moved //[ast]~| value used here after move } fn move_after_borrow_nested() { let a: Box<_> = box D { x: box A { x: box 0, y: 1 }, y: box 2 }; let _x = &a.x.x; //[ast]~^ borrow of `a.x.x` occurs here let _y = a.y; //[ast]~^ ERROR cannot move //[ast]~| move out of use_imm(_x); } fn copy_after_mut_borrow_nested() { let mut a: Box<_> = box C { x: box A { x: box 0, y: 1 }, y: 2 }; let _x = &mut a.x.x; let _y = a.y; //[ast]~ ERROR cannot use use_mut(_x); } fn move_after_mut_borrow_nested() { let mut a: Box<_> = box D { x: box A { x: box 0, y: 1 }, y: box 2 }; let _x = &mut a.x.x; let _y = a.y; //[ast]~^ ERROR cannot move //[ast]~| move out of use_mut(_x); } fn borrow_after_mut_borrow_nested() { let mut a: Box<_> = box C { x: box A { x: box 0, y: 1 }, y: 2 }; let _x = &mut a.x.x; //[ast]~^ mutable borrow occurs here let _y = &a.y; //[ast]~ ERROR cannot borrow //[ast]~^ immutable borrow occurs here use_mut(_x); } fn mut_borrow_after_borrow_nested() { let mut a: Box<_> = box C { x: box A { x: box 0, y: 1 }, y: 2 }; let _x = &a.x.x; //[ast]~^ immutable borrow occurs here let _y = &mut a.y; //[ast]~ ERROR cannot borrow //[ast]~^ mutable borrow occurs here use_imm(_x); } #[rustc_error] fn main() { //[mir]~ ERROR compilation successful copy_after_move(); move_after_move(); borrow_after_move(); move_after_borrow(); copy_after_mut_borrow(); move_after_mut_borrow(); borrow_after_mut_borrow(); mut_borrow_after_borrow(); copy_after_move_nested(); move_after_move_nested(); borrow_after_move_nested(); move_after_borrow_nested(); copy_after_mut_borrow_nested(); move_after_mut_borrow_nested(); borrow_after_mut_borrow_nested(); mut_borrow_after_borrow_nested(); } fn use_mut(_: &mut T) { } fn use_imm(_: &T) { }