borrowck: aliasability violation with closure captures can happen

It was considered to be impossible but actually it can
happen for nested closures. Also, because there must
be nested closures when this happens, we can use more
targeted help message.

Closes #21390
Closes #21600
This commit is contained in:
Edward Wang 2015-01-27 23:55:07 +08:00
parent a6a6fadbb9
commit 2c6440e2a2
2 changed files with 38 additions and 11 deletions

View File

@ -770,16 +770,7 @@ impl<'a, 'tcx> BorrowckCtxt<'a, 'tcx> {
MutabilityViolation => { MutabilityViolation => {
"cannot assign to data" "cannot assign to data"
} }
BorrowViolation(euv::ClosureCapture(_)) => { BorrowViolation(euv::ClosureCapture(_)) |
// I don't think we can get aliasability violations
// with closure captures, so no need to come up with a
// good error message. The reason this cannot happen
// is because we only capture local variables in
// closures, and those are never aliasable.
self.tcx.sess.span_bug(
span,
"aliasability violation with closure");
}
BorrowViolation(euv::OverloadedOperator) | BorrowViolation(euv::OverloadedOperator) |
BorrowViolation(euv::AddrOf) | BorrowViolation(euv::AddrOf) |
BorrowViolation(euv::AutoRef) | BorrowViolation(euv::AutoRef) |
@ -809,9 +800,18 @@ impl<'a, 'tcx> BorrowckCtxt<'a, 'tcx> {
self.tcx.sess.span_err(span, self.tcx.sess.span_err(span,
format!("{} in a captured outer \ format!("{} in a captured outer \
variable in an `Fn` closure", prefix).as_slice()); variable in an `Fn` closure", prefix).as_slice());
if let BorrowViolation(euv::ClosureCapture(_)) = kind {
// The aliasability violation with closure captures can
// happen for nested closures, so we know the enclosing
// closure incorrectly accepts an `Fn` while it needs to
// be `FnMut`.
span_help!(self.tcx.sess, self.tcx.map.span(id),
"consider changing this to accept closures that implement `FnMut`");
} else {
span_help!(self.tcx.sess, self.tcx.map.span(id), span_help!(self.tcx.sess, self.tcx.map.span(id),
"consider changing this closure to take self by mutable reference"); "consider changing this closure to take self by mutable reference");
} }
}
mc::AliasableStatic(..) | mc::AliasableStatic(..) |
mc::AliasableStaticMut(..) => { mc::AliasableStaticMut(..) => {
self.tcx.sess.span_err( self.tcx.sess.span_err(

View File

@ -0,0 +1,27 @@
// Copyright 2015 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 call_it<F>(f: F) where F: Fn() { f(); }
struct A;
impl A {
fn gen(&self) {}
fn gen_mut(&mut self) {}
}
fn main() {
let mut x = A;
call_it(|| { //~ HELP consider changing this to accept closures that implement `FnMut`
call_it(|| x.gen());
call_it(|| x.gen_mut()); //~ ERROR cannot borrow data mutably in a captured outer
//~^ ERROR cannot borrow data mutably in a captured outer
});
}