2012-06-01 12:46:17 -05:00
|
|
|
// ----------------------------------------------------------------------
|
|
|
|
// Gathering loans
|
|
|
|
//
|
|
|
|
// The borrow check proceeds in two phases. In phase one, we gather the full
|
|
|
|
// set of loans that are required at any point. These are sorted according to
|
|
|
|
// their associated scopes. In phase two, checking loans, we will then make
|
|
|
|
// sure that all of these loans are honored.
|
|
|
|
|
2012-09-11 23:25:01 -05:00
|
|
|
use mem_categorization::{mem_categorization_ctxt, opt_deref_kind};
|
2012-09-04 13:54:36 -05:00
|
|
|
use preserve::{preserve_condition, pc_ok, pc_if_pure};
|
2012-09-11 23:25:01 -05:00
|
|
|
use ty::{ty_region};
|
2012-06-01 12:46:17 -05:00
|
|
|
|
|
|
|
export gather_loans;
|
|
|
|
|
2012-07-26 10:51:57 -05:00
|
|
|
/// Context used while gathering loans:
|
|
|
|
///
|
|
|
|
/// - `bccx`: the the borrow check context
|
|
|
|
/// - `req_maps`: the maps computed by `gather_loans()`, see def'n of the
|
|
|
|
/// type `req_maps` for more info
|
|
|
|
/// - `item_ub`: the id of the block for the enclosing fn/method item
|
|
|
|
/// - `root_ub`: the id of the outermost block for which we can root
|
|
|
|
/// an `@T`. This is the id of the innermost enclosing
|
|
|
|
/// loop or function body.
|
|
|
|
///
|
|
|
|
/// The role of `root_ub` is to prevent us from having to accumulate
|
|
|
|
/// vectors of rooted items at runtime. Consider this case:
|
|
|
|
///
|
|
|
|
/// fn foo(...) -> int {
|
|
|
|
/// let mut ptr: ∫
|
|
|
|
/// while some_cond {
|
|
|
|
/// let x: @int = ...;
|
|
|
|
/// ptr = &*x;
|
|
|
|
/// }
|
|
|
|
/// *ptr
|
|
|
|
/// }
|
|
|
|
///
|
|
|
|
/// If we are not careful here, we would infer the scope of the borrow `&*x`
|
|
|
|
/// to be the body of the function `foo()` as a whole. We would then
|
|
|
|
/// have root each `@int` that is produced, which is an unbounded number.
|
|
|
|
/// No good. Instead what will happen is that `root_ub` will be set to the
|
|
|
|
/// body of the while loop and we will refuse to root the pointer `&*x`
|
|
|
|
/// because it would have to be rooted for a region greater than `root_ub`.
|
|
|
|
enum gather_loan_ctxt = @{bccx: borrowck_ctxt,
|
|
|
|
req_maps: req_maps,
|
|
|
|
mut item_ub: ast::node_id,
|
|
|
|
mut root_ub: ast::node_id};
|
2012-06-01 12:46:17 -05:00
|
|
|
|
|
|
|
fn gather_loans(bccx: borrowck_ctxt, crate: @ast::crate) -> req_maps {
|
|
|
|
let glcx = gather_loan_ctxt(@{bccx: bccx,
|
2012-09-19 17:13:04 -05:00
|
|
|
req_maps: {req_loan_map: HashMap(),
|
|
|
|
pure_map: HashMap()},
|
2012-07-26 10:51:57 -05:00
|
|
|
mut item_ub: 0,
|
|
|
|
mut root_ub: 0});
|
|
|
|
let v = visit::mk_vt(@{visit_expr: req_loans_in_expr,
|
|
|
|
visit_fn: req_loans_in_fn,
|
2012-09-04 15:29:32 -05:00
|
|
|
.. *visit::default_visitor()});
|
2012-06-01 12:46:17 -05:00
|
|
|
visit::visit_crate(*crate, glcx, v);
|
2012-08-01 19:30:05 -05:00
|
|
|
return glcx.req_maps;
|
2012-06-01 12:46:17 -05:00
|
|
|
}
|
|
|
|
|
2012-07-26 10:51:57 -05:00
|
|
|
fn req_loans_in_fn(fk: visit::fn_kind,
|
|
|
|
decl: ast::fn_decl,
|
|
|
|
body: ast::blk,
|
|
|
|
sp: span,
|
|
|
|
id: ast::node_id,
|
|
|
|
&&self: gather_loan_ctxt,
|
|
|
|
v: visit::vt<gather_loan_ctxt>) {
|
|
|
|
// see explanation attached to the `root_ub` field:
|
|
|
|
let old_item_id = self.item_ub;
|
|
|
|
let old_root_ub = self.root_ub;
|
|
|
|
self.root_ub = body.node.id;
|
|
|
|
|
2012-08-06 14:34:08 -05:00
|
|
|
match fk {
|
2012-10-08 13:49:01 -05:00
|
|
|
visit::fk_anon(*) | visit::fk_fn_block(*) => {}
|
|
|
|
visit::fk_item_fn(*) | visit::fk_method(*) |
|
|
|
|
visit::fk_dtor(*) => {
|
|
|
|
self.item_ub = body.node.id;
|
|
|
|
}
|
2012-07-26 10:51:57 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
visit::visit_fn(fk, decl, body, sp, id, self, v);
|
|
|
|
self.root_ub = old_root_ub;
|
|
|
|
self.item_ub = old_item_id;
|
|
|
|
}
|
|
|
|
|
2012-06-01 12:46:17 -05:00
|
|
|
fn req_loans_in_expr(ex: @ast::expr,
|
|
|
|
&&self: gather_loan_ctxt,
|
|
|
|
vt: visit::vt<gather_loan_ctxt>) {
|
|
|
|
let bccx = self.bccx;
|
|
|
|
let tcx = bccx.tcx;
|
2012-07-26 10:51:57 -05:00
|
|
|
let old_root_ub = self.root_ub;
|
2012-06-01 12:46:17 -05:00
|
|
|
|
2012-08-28 17:54:45 -05:00
|
|
|
debug!("req_loans_in_expr(expr=%?/%s)",
|
|
|
|
ex.id, pprust::expr_to_str(ex, tcx.sess.intr()));
|
2012-06-01 17:46:32 -05:00
|
|
|
|
2012-06-01 12:46:17 -05:00
|
|
|
// If this expression is borrowed, have to ensure it remains valid:
|
2012-09-11 23:25:01 -05:00
|
|
|
for tcx.adjustments.find(ex.id).each |adjustments| {
|
2012-09-19 18:55:01 -05:00
|
|
|
self.guarantee_adjustments(ex, *adjustments);
|
2012-06-01 12:46:17 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
// Special checks for various kinds of expressions:
|
2012-08-06 14:34:08 -05:00
|
|
|
match ex.node {
|
2012-08-03 21:59:04 -05:00
|
|
|
ast::expr_addr_of(mutbl, base) => {
|
2012-06-01 12:46:17 -05:00
|
|
|
let base_cmt = self.bccx.cat_expr(base);
|
|
|
|
|
|
|
|
// make sure that the thing we are pointing out stays valid
|
|
|
|
// for the lifetime `scope_r` of the resulting ptr:
|
2012-08-06 20:14:46 -05:00
|
|
|
let scope_r = ty_region(tcx.ty(ex));
|
2012-06-01 12:46:17 -05:00
|
|
|
self.guarantee_valid(base_cmt, mutbl, scope_r);
|
2012-07-26 10:51:57 -05:00
|
|
|
visit::visit_expr(ex, self, vt);
|
2012-06-01 12:46:17 -05:00
|
|
|
}
|
|
|
|
|
2012-08-03 21:59:04 -05:00
|
|
|
ast::expr_call(f, args, _) => {
|
2012-06-01 12:46:17 -05:00
|
|
|
let arg_tys = ty::ty_fn_args(ty::expr_ty(self.tcx(), f));
|
|
|
|
let scope_r = ty::re_scope(ex.id);
|
2012-09-28 17:48:25 -05:00
|
|
|
for vec::each2(args, arg_tys) |arg, arg_ty| {
|
2012-08-06 14:34:08 -05:00
|
|
|
match ty::resolved_mode(self.tcx(), arg_ty.mode) {
|
2012-08-03 21:59:04 -05:00
|
|
|
ast::by_ref => {
|
2012-09-28 00:20:47 -05:00
|
|
|
let arg_cmt = self.bccx.cat_expr(*arg);
|
2012-06-01 12:46:17 -05:00
|
|
|
self.guarantee_valid(arg_cmt, m_imm, scope_r);
|
|
|
|
}
|
2012-10-11 18:11:47 -05:00
|
|
|
ast::by_val | ast::by_move | ast::by_copy => {}
|
2012-06-01 12:46:17 -05:00
|
|
|
}
|
|
|
|
}
|
2012-07-26 10:51:57 -05:00
|
|
|
visit::visit_expr(ex, self, vt);
|
2012-06-01 12:46:17 -05:00
|
|
|
}
|
|
|
|
|
2012-08-24 23:03:51 -05:00
|
|
|
ast::expr_match(ex_v, arms) => {
|
2012-06-01 12:46:17 -05:00
|
|
|
let cmt = self.bccx.cat_expr(ex_v);
|
2012-06-30 18:19:07 -05:00
|
|
|
for arms.each |arm| {
|
|
|
|
for arm.pats.each |pat| {
|
2012-09-19 18:55:01 -05:00
|
|
|
self.gather_pat(cmt, *pat, arm.body.node.id, ex.id);
|
2012-06-01 12:46:17 -05:00
|
|
|
}
|
|
|
|
}
|
2012-07-26 10:51:57 -05:00
|
|
|
visit::visit_expr(ex, self, vt);
|
2012-06-01 12:46:17 -05:00
|
|
|
}
|
|
|
|
|
2012-06-01 23:54:38 -05:00
|
|
|
ast::expr_index(rcvr, _) |
|
2012-06-01 17:46:32 -05:00
|
|
|
ast::expr_binary(_, rcvr, _) |
|
2012-08-03 21:59:04 -05:00
|
|
|
ast::expr_unary(_, rcvr)
|
|
|
|
if self.bccx.method_map.contains_key(ex.id) => {
|
2012-06-01 17:46:32 -05:00
|
|
|
// Receivers in method calls are always passed by ref.
|
|
|
|
//
|
2012-06-01 23:54:38 -05:00
|
|
|
// Here, in an overloaded operator, the call is this expression,
|
|
|
|
// and hence the scope of the borrow is this call.
|
|
|
|
//
|
2012-06-21 18:44:10 -05:00
|
|
|
// FIX? / NOT REALLY---technically we should check the other
|
2012-06-01 23:54:38 -05:00
|
|
|
// argument and consider the argument mode. But how annoying.
|
|
|
|
// And this problem when goes away when argument modes are
|
|
|
|
// phased out. So I elect to leave this undone.
|
|
|
|
let scope_r = ty::re_scope(ex.id);
|
|
|
|
let rcvr_cmt = self.bccx.cat_expr(rcvr);
|
|
|
|
self.guarantee_valid(rcvr_cmt, m_imm, scope_r);
|
2012-07-26 10:51:57 -05:00
|
|
|
visit::visit_expr(ex, self, vt);
|
2012-06-01 23:54:38 -05:00
|
|
|
}
|
|
|
|
|
2012-08-28 17:54:45 -05:00
|
|
|
// FIXME--#3387
|
|
|
|
// ast::expr_binary(_, lhs, rhs) => {
|
|
|
|
// // Universal comparison operators like ==, >=, etc
|
|
|
|
// // take their arguments by reference.
|
|
|
|
// let lhs_ty = ty::expr_ty(self.tcx(), lhs);
|
|
|
|
// if !ty::type_is_scalar(lhs_ty) {
|
|
|
|
// let scope_r = ty::re_scope(ex.id);
|
|
|
|
// let lhs_cmt = self.bccx.cat_expr(lhs);
|
|
|
|
// self.guarantee_valid(lhs_cmt, m_imm, scope_r);
|
|
|
|
// let rhs_cmt = self.bccx.cat_expr(rhs);
|
|
|
|
// self.guarantee_valid(rhs_cmt, m_imm, scope_r);
|
|
|
|
// }
|
|
|
|
// visit::visit_expr(ex, self, vt);
|
|
|
|
// }
|
|
|
|
|
2012-06-01 23:54:38 -05:00
|
|
|
ast::expr_field(rcvr, _, _)
|
2012-08-03 21:59:04 -05:00
|
|
|
if self.bccx.method_map.contains_key(ex.id) => {
|
2012-06-01 23:54:38 -05:00
|
|
|
// Receivers in method calls are always passed by ref.
|
|
|
|
//
|
|
|
|
// Here, the field a.b is in fact a closure. Eventually, this
|
|
|
|
// should be an fn&, but for now it's an fn@. In any case,
|
|
|
|
// the enclosing scope is either the call where it is a rcvr
|
|
|
|
// (if used like `a.b(...)`), the call where it's an argument
|
|
|
|
// (if used like `x(a.b)`), or the block (if used like `let x
|
|
|
|
// = a.b`).
|
2012-06-01 17:46:32 -05:00
|
|
|
let scope_r = ty::re_scope(self.tcx().region_map.get(ex.id));
|
|
|
|
let rcvr_cmt = self.bccx.cat_expr(rcvr);
|
|
|
|
self.guarantee_valid(rcvr_cmt, m_imm, scope_r);
|
2012-07-26 10:51:57 -05:00
|
|
|
visit::visit_expr(ex, self, vt);
|
2012-06-01 17:46:32 -05:00
|
|
|
}
|
|
|
|
|
2012-07-26 10:51:57 -05:00
|
|
|
// see explanation attached to the `root_ub` field:
|
2012-08-03 21:59:04 -05:00
|
|
|
ast::expr_while(cond, body) => {
|
2012-07-26 10:51:57 -05:00
|
|
|
// during the condition, can only root for the condition
|
|
|
|
self.root_ub = cond.id;
|
|
|
|
vt.visit_expr(cond, self, vt);
|
|
|
|
|
|
|
|
// during body, can only root for the body
|
|
|
|
self.root_ub = body.node.id;
|
|
|
|
vt.visit_block(body, self, vt);
|
|
|
|
}
|
|
|
|
|
|
|
|
// see explanation attached to the `root_ub` field:
|
2012-08-14 21:20:56 -05:00
|
|
|
ast::expr_loop(body, _) => {
|
2012-07-26 10:51:57 -05:00
|
|
|
self.root_ub = body.node.id;
|
|
|
|
visit::visit_expr(ex, self, vt);
|
|
|
|
}
|
|
|
|
|
|
|
|
_ => {
|
|
|
|
visit::visit_expr(ex, self, vt);
|
|
|
|
}
|
2012-06-01 12:46:17 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
// Check any contained expressions:
|
2012-07-26 10:51:57 -05:00
|
|
|
|
|
|
|
self.root_ub = old_root_ub;
|
2012-06-01 12:46:17 -05:00
|
|
|
}
|
|
|
|
|
2012-08-07 20:10:06 -05:00
|
|
|
impl gather_loan_ctxt {
|
2012-10-14 15:39:17 -05:00
|
|
|
fn tcx(&self) -> ty::ctxt { self.bccx.tcx }
|
2012-06-01 12:46:17 -05:00
|
|
|
|
2012-10-14 15:39:17 -05:00
|
|
|
fn guarantee_adjustments(&self,
|
|
|
|
expr: @ast::expr,
|
2012-09-11 23:25:01 -05:00
|
|
|
adjustment: &ty::AutoAdjustment) {
|
|
|
|
debug!("guarantee_adjustments(expr=%s, adjustment=%?)",
|
|
|
|
expr_repr(self.tcx(), expr), adjustment);
|
|
|
|
let _i = indenter();
|
|
|
|
|
|
|
|
match adjustment.autoref {
|
|
|
|
None => {
|
|
|
|
debug!("no autoref");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
Some(ref autoref) => {
|
|
|
|
let mcx = &mem_categorization_ctxt {
|
|
|
|
tcx: self.tcx(),
|
|
|
|
method_map: self.bccx.method_map};
|
|
|
|
let mut cmt = mcx.cat_expr_autoderefd(expr, adjustment);
|
|
|
|
debug!("after autoderef, cmt=%s", self.bccx.cmt_to_repr(cmt));
|
|
|
|
|
|
|
|
match autoref.kind {
|
|
|
|
ty::AutoPtr => {
|
|
|
|
self.guarantee_valid(cmt,
|
|
|
|
autoref.mutbl,
|
|
|
|
autoref.region)
|
|
|
|
}
|
|
|
|
ty::AutoSlice => {
|
|
|
|
let cmt_index = mcx.cat_index(expr, cmt);
|
|
|
|
self.guarantee_valid(cmt_index,
|
|
|
|
autoref.mutbl,
|
|
|
|
autoref.region)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-06-01 12:46:17 -05:00
|
|
|
// guarantees that addr_of(cmt) will be valid for the duration of
|
|
|
|
// `static_scope_r`, or reports an error. This may entail taking
|
|
|
|
// out loans, which will be added to the `req_loan_map`. This can
|
|
|
|
// also entail "rooting" GC'd pointers, which means ensuring
|
|
|
|
// dynamically that they are not freed.
|
2012-10-14 15:39:17 -05:00
|
|
|
fn guarantee_valid(&self,
|
|
|
|
cmt: cmt,
|
2012-06-01 12:46:17 -05:00
|
|
|
req_mutbl: ast::mutability,
|
2012-10-15 16:56:42 -05:00
|
|
|
scope_r: ty::Region) {
|
2012-06-01 12:46:17 -05:00
|
|
|
|
2012-07-05 14:59:03 -05:00
|
|
|
self.bccx.guaranteed_paths += 1;
|
|
|
|
|
2012-08-22 19:24:52 -05:00
|
|
|
debug!("guarantee_valid(cmt=%s, req_mutbl=%s, scope_r=%s)",
|
2012-06-01 12:46:17 -05:00
|
|
|
self.bccx.cmt_to_repr(cmt),
|
|
|
|
self.bccx.mut_to_str(req_mutbl),
|
2012-08-22 19:24:52 -05:00
|
|
|
region_to_str(self.tcx(), scope_r));
|
2012-06-01 12:46:17 -05:00
|
|
|
let _i = indenter();
|
|
|
|
|
2012-08-06 14:34:08 -05:00
|
|
|
match cmt.lp {
|
2012-07-05 14:59:03 -05:00
|
|
|
// If this expression is a loanable path, we MUST take out a
|
|
|
|
// loan. This is somewhat non-obvious. You might think,
|
|
|
|
// for example, that if we have an immutable local variable
|
|
|
|
// `x` whose value is being borrowed, we could rely on `x`
|
|
|
|
// not to change. This is not so, however, because even
|
|
|
|
// immutable locals can be moved. So we take out a loan on
|
|
|
|
// `x`, guaranteeing that it remains immutable for the
|
|
|
|
// duration of the reference: if there is an attempt to move
|
|
|
|
// it within that scope, the loan will be detected and an
|
|
|
|
// error will be reported.
|
2012-08-20 14:23:37 -05:00
|
|
|
Some(_) => {
|
2012-10-14 15:39:17 -05:00
|
|
|
match self.bccx.loan(cmt, scope_r, req_mutbl) {
|
|
|
|
Err(e) => { self.bccx.report(e); }
|
|
|
|
Ok(move loans) => {
|
|
|
|
self.add_loans(cmt, req_mutbl, scope_r, move loans);
|
2012-07-26 10:51:57 -05:00
|
|
|
}
|
2012-06-01 12:46:17 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-07-26 10:51:57 -05:00
|
|
|
// The path is not loanable: in that case, we must try and
|
|
|
|
// preserve it dynamically (or see that it is preserved by
|
|
|
|
// virtue of being rooted in some immutable path). We must
|
|
|
|
// also check that the mutability of the desired pointer
|
|
|
|
// matches with the actual mutability (but if an immutable
|
|
|
|
// pointer is desired, that is ok as long as we are pure)
|
2012-08-20 14:23:37 -05:00
|
|
|
None => {
|
2012-07-26 10:51:57 -05:00
|
|
|
let result: bckres<preserve_condition> = {
|
|
|
|
do self.check_mutbl(req_mutbl, cmt).chain |pc1| {
|
|
|
|
do self.bccx.preserve(cmt, scope_r,
|
|
|
|
self.item_ub,
|
|
|
|
self.root_ub).chain |pc2| {
|
2012-08-26 18:54:31 -05:00
|
|
|
Ok(pc1.combine(pc2))
|
2012-07-26 10:51:57 -05:00
|
|
|
}
|
2012-06-01 12:46:17 -05:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2012-08-06 14:34:08 -05:00
|
|
|
match result {
|
2012-09-14 23:58:04 -05:00
|
|
|
Ok(pc_ok) => {
|
|
|
|
debug!("result of preserve: pc_ok");
|
|
|
|
|
|
|
|
// we were able guarantee the validity of the ptr,
|
|
|
|
// perhaps by rooting or because it is immutably
|
|
|
|
// rooted. good.
|
|
|
|
self.bccx.stable_paths += 1;
|
|
|
|
}
|
|
|
|
Ok(pc_if_pure(e)) => {
|
|
|
|
debug!("result of preserve: %?", pc_if_pure(e));
|
|
|
|
|
|
|
|
// we are only able to guarantee the validity if
|
|
|
|
// the scope is pure
|
|
|
|
match scope_r {
|
|
|
|
ty::re_scope(pure_id) => {
|
|
|
|
// if the scope is some block/expr in the
|
|
|
|
// fn, then just require that this scope
|
|
|
|
// be pure
|
|
|
|
self.req_maps.pure_map.insert(pure_id, e);
|
|
|
|
self.bccx.req_pure_paths += 1;
|
|
|
|
|
|
|
|
debug!("requiring purity for scope %?",
|
|
|
|
scope_r);
|
|
|
|
|
|
|
|
if self.tcx().sess.borrowck_note_pure() {
|
|
|
|
self.bccx.span_note(
|
|
|
|
cmt.span,
|
|
|
|
fmt!("purity required"));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
_ => {
|
|
|
|
// otherwise, we can't enforce purity for
|
|
|
|
// that scope, so give up and report an
|
|
|
|
// error
|
|
|
|
self.bccx.report(e);
|
|
|
|
}
|
2012-07-06 17:03:18 -05:00
|
|
|
}
|
2012-09-14 23:58:04 -05:00
|
|
|
}
|
|
|
|
Err(e) => {
|
|
|
|
// we cannot guarantee the validity of this pointer
|
|
|
|
debug!("result of preserve: error");
|
2012-06-01 12:46:17 -05:00
|
|
|
self.bccx.report(e);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check that the pat `cmt` is compatible with the required
|
|
|
|
// mutability, presuming that it can be preserved to stay alive
|
|
|
|
// long enough.
|
|
|
|
//
|
|
|
|
// For example, if you have an expression like `&x.f` where `x`
|
|
|
|
// has type `@mut{f:int}`, this check might fail because `&x.f`
|
|
|
|
// reqires an immutable pointer, but `f` lives in (aliased)
|
|
|
|
// mutable memory.
|
2012-10-14 15:39:17 -05:00
|
|
|
fn check_mutbl(&self,
|
|
|
|
req_mutbl: ast::mutability,
|
2012-07-26 10:51:57 -05:00
|
|
|
cmt: cmt) -> bckres<preserve_condition> {
|
2012-09-14 23:58:04 -05:00
|
|
|
debug!("check_mutbl(req_mutbl=%?, cmt.mutbl=%?)",
|
|
|
|
req_mutbl, cmt.mutbl);
|
|
|
|
|
2012-09-11 23:25:01 -05:00
|
|
|
if req_mutbl == m_const || req_mutbl == cmt.mutbl {
|
2012-09-14 23:58:04 -05:00
|
|
|
debug!("required is const or they are the same");
|
2012-08-26 18:54:31 -05:00
|
|
|
Ok(pc_ok)
|
2012-09-11 23:25:01 -05:00
|
|
|
} else {
|
2012-07-26 10:51:57 -05:00
|
|
|
let e = {cmt: cmt,
|
2012-09-11 23:25:01 -05:00
|
|
|
code: err_mutbl(req_mutbl)};
|
2012-07-26 10:51:57 -05:00
|
|
|
if req_mutbl == m_imm {
|
|
|
|
// you can treat mutable things as imm if you are pure
|
2012-09-14 23:58:04 -05:00
|
|
|
debug!("imm required, must be pure");
|
|
|
|
|
2012-08-26 18:54:31 -05:00
|
|
|
Ok(pc_if_pure(e))
|
2012-07-26 10:51:57 -05:00
|
|
|
} else {
|
2012-08-26 18:54:31 -05:00
|
|
|
Err(e)
|
2012-07-26 10:51:57 -05:00
|
|
|
}
|
2012-06-01 12:46:17 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-10-14 15:39:17 -05:00
|
|
|
fn add_loans(&self,
|
|
|
|
cmt: cmt,
|
|
|
|
req_mutbl: ast::mutability,
|
2012-10-15 16:56:42 -05:00
|
|
|
scope_r: ty::Region,
|
2012-10-14 15:39:17 -05:00
|
|
|
+loans: ~[Loan]) {
|
|
|
|
if loans.len() == 0 {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
let scope_id = match scope_r {
|
|
|
|
ty::re_scope(scope_id) => scope_id,
|
|
|
|
_ => {
|
|
|
|
self.bccx.tcx.sess.span_bug(
|
|
|
|
cmt.span,
|
|
|
|
fmt!("loans required but scope is scope_region is %s",
|
|
|
|
region_to_str(self.tcx(), scope_r)));
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
self.add_loans_to_scope_id(scope_id, move loans);
|
|
|
|
|
|
|
|
if req_mutbl == m_imm && cmt.mutbl != m_imm {
|
|
|
|
self.bccx.loaned_paths_imm += 1;
|
|
|
|
|
|
|
|
if self.tcx().sess.borrowck_note_loan() {
|
|
|
|
self.bccx.span_note(
|
|
|
|
cmt.span,
|
|
|
|
fmt!("immutable loan required"));
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
self.bccx.loaned_paths_same += 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn add_loans_to_scope_id(&self, scope_id: ast::node_id, +loans: ~[Loan]) {
|
2012-08-28 17:54:45 -05:00
|
|
|
debug!("adding %u loans to scope_id %?", loans.len(), scope_id);
|
2012-08-06 14:34:08 -05:00
|
|
|
match self.req_maps.req_loan_map.find(scope_id) {
|
2012-10-14 15:39:17 -05:00
|
|
|
Some(req_loans) => {
|
|
|
|
req_loans.push_all(loans);
|
2012-08-28 17:54:45 -05:00
|
|
|
}
|
|
|
|
None => {
|
2012-10-14 15:39:17 -05:00
|
|
|
let dvec = @dvec::from_vec(move loans);
|
|
|
|
self.req_maps.req_loan_map.insert(scope_id, dvec);
|
2012-08-28 17:54:45 -05:00
|
|
|
}
|
2012-06-01 12:46:17 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-10-14 15:39:17 -05:00
|
|
|
fn gather_pat(&self,
|
|
|
|
discr_cmt: cmt,
|
|
|
|
root_pat: @ast::pat,
|
|
|
|
arm_id: ast::node_id,
|
|
|
|
alt_id: ast::node_id) {
|
2012-08-08 10:15:32 -05:00
|
|
|
do self.bccx.cat_pattern(discr_cmt, root_pat) |cmt, pat| {
|
|
|
|
match pat.node {
|
2012-08-26 14:12:05 -05:00
|
|
|
ast::pat_ident(bm, _, _) if !self.pat_is_variant(pat) => {
|
2012-08-08 10:15:32 -05:00
|
|
|
match bm {
|
2012-08-22 18:00:28 -05:00
|
|
|
ast::bind_by_value | ast::bind_by_move => {
|
2012-08-08 10:15:32 -05:00
|
|
|
// copying does not borrow anything, so no check
|
|
|
|
// is required
|
2012-08-22 18:00:28 -05:00
|
|
|
// as for move, check::alt ensures it's from an rvalue.
|
2012-08-08 10:15:32 -05:00
|
|
|
}
|
|
|
|
ast::bind_by_ref(mutbl) => {
|
|
|
|
// ref x or ref x @ p --- creates a ptr which must
|
|
|
|
// remain valid for the scope of the alt
|
|
|
|
|
|
|
|
// find the region of the resulting pointer (note that
|
|
|
|
// the type of such a pattern will *always* be a
|
|
|
|
// region pointer)
|
|
|
|
let scope_r = ty_region(self.tcx().ty(pat));
|
|
|
|
|
|
|
|
// if the scope of the region ptr turns out to be
|
|
|
|
// specific to this arm, wrap the categorization with
|
|
|
|
// a cat_discr() node. There is a detailed discussion
|
|
|
|
// of the function of this node in method preserve():
|
|
|
|
let arm_scope = ty::re_scope(arm_id);
|
|
|
|
if self.bccx.is_subregion_of(scope_r, arm_scope) {
|
|
|
|
let cmt_discr = self.bccx.cat_discr(cmt, alt_id);
|
|
|
|
self.guarantee_valid(cmt_discr, mutbl, scope_r);
|
|
|
|
} else {
|
|
|
|
self.guarantee_valid(cmt, mutbl, scope_r);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
ast::bind_by_implicit_ref => {
|
|
|
|
// Note: there is a discussion of the function of
|
|
|
|
// cat_discr in the method preserve():
|
|
|
|
let cmt1 = self.bccx.cat_discr(cmt, alt_id);
|
|
|
|
let arm_scope = ty::re_scope(arm_id);
|
|
|
|
|
|
|
|
// We used to remember the mutability of the location
|
|
|
|
// that this binding refers to and use it later when
|
|
|
|
// categorizing the binding. This hack is being
|
|
|
|
// removed in favor of ref mode bindings.
|
|
|
|
//
|
|
|
|
// self.bccx.binding_map.insert(pat.id, cmt1.mutbl);
|
|
|
|
|
|
|
|
self.guarantee_valid(cmt1, m_const, arm_scope);
|
|
|
|
}
|
2012-08-06 20:14:46 -05:00
|
|
|
}
|
|
|
|
}
|
2012-06-01 12:46:17 -05:00
|
|
|
|
2012-08-08 10:15:32 -05:00
|
|
|
_ => {}
|
2012-06-01 12:46:17 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-10-14 15:39:17 -05:00
|
|
|
fn pat_is_variant(&self, pat: @ast::pat) -> bool {
|
2012-06-01 12:46:17 -05:00
|
|
|
pat_util::pat_is_variant(self.bccx.tcx.def_map, pat)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|