2011-09-01 19:27:58 -05:00
|
|
|
import std::{vec, str, option};
|
2011-08-31 11:45:37 -05:00
|
|
|
import option::{some, none};
|
|
|
|
import syntax::ast::*;
|
|
|
|
import syntax::visit;
|
|
|
|
import syntax::ast_util;
|
|
|
|
|
|
|
|
tag deref_t { unbox; field; index; }
|
|
|
|
|
|
|
|
type deref = @{mut: bool, kind: deref_t, outer_t: ty::t};
|
|
|
|
|
|
|
|
// Finds the root (the thing that is dereferenced) for the given expr, and a
|
|
|
|
// vec of dereferences that were used on this root. Note that, in this vec,
|
|
|
|
// the inner derefs come in front, so foo.bar[1] becomes rec(ex=foo,
|
|
|
|
// ds=[index,field])
|
2011-09-12 04:27:30 -05:00
|
|
|
fn expr_root(tcx: ty::ctxt, ex: @expr, autoderef: bool) ->
|
2011-09-02 17:34:58 -05:00
|
|
|
{ex: @expr, ds: @[deref]} {
|
2011-09-12 04:27:30 -05:00
|
|
|
fn maybe_auto_unbox(tcx: ty::ctxt, t: ty::t) -> {t: ty::t, ds: [deref]} {
|
2011-08-31 11:45:37 -05:00
|
|
|
let ds = [];
|
|
|
|
while true {
|
|
|
|
alt ty::struct(tcx, t) {
|
|
|
|
ty::ty_box(mt) {
|
|
|
|
ds += [@{mut: mt.mut != imm, kind: unbox, outer_t: t}];
|
|
|
|
t = mt.ty;
|
|
|
|
}
|
|
|
|
ty::ty_uniq(mt) {
|
2011-09-21 20:54:54 -05:00
|
|
|
ds += [@{mut: mt.mut != imm, kind: unbox, outer_t: t}];
|
2011-08-31 11:45:37 -05:00
|
|
|
}
|
|
|
|
ty::ty_res(_, inner, tps) {
|
|
|
|
ds += [@{mut: false, kind: unbox, outer_t: t}];
|
|
|
|
t = ty::substitute_type_params(tcx, tps, inner);
|
|
|
|
}
|
|
|
|
ty::ty_tag(did, tps) {
|
|
|
|
let variants = ty::tag_variants(tcx, did);
|
|
|
|
if vec::len(variants) != 1u ||
|
2011-09-02 17:34:58 -05:00
|
|
|
vec::len(variants[0].args) != 1u {
|
2011-08-31 11:45:37 -05:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
ds += [@{mut: false, kind: unbox, outer_t: t}];
|
2011-09-02 17:34:58 -05:00
|
|
|
t = ty::substitute_type_params(tcx, tps, variants[0].args[0]);
|
2011-08-31 11:45:37 -05:00
|
|
|
}
|
|
|
|
_ { break; }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
ret {t: t, ds: ds};
|
|
|
|
}
|
|
|
|
let ds: [deref] = [];
|
|
|
|
while true {
|
2011-09-12 06:13:20 -05:00
|
|
|
alt copy ex.node {
|
2011-08-31 11:45:37 -05:00
|
|
|
expr_field(base, ident) {
|
|
|
|
let auto_unbox = maybe_auto_unbox(tcx, ty::expr_ty(tcx, base));
|
|
|
|
let mut = false;
|
|
|
|
alt ty::struct(tcx, auto_unbox.t) {
|
|
|
|
ty::ty_rec(fields) {
|
|
|
|
for fld: ty::field in fields {
|
2011-09-01 19:27:58 -05:00
|
|
|
if str::eq(ident, fld.ident) {
|
2011-08-31 11:45:37 -05:00
|
|
|
mut = fld.mt.mut != imm;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
ty::ty_obj(_) { }
|
|
|
|
}
|
|
|
|
ds += [@{mut: mut, kind: field, outer_t: auto_unbox.t}];
|
|
|
|
ds += auto_unbox.ds;
|
|
|
|
ex = base;
|
|
|
|
}
|
|
|
|
expr_index(base, _) {
|
|
|
|
let auto_unbox = maybe_auto_unbox(tcx, ty::expr_ty(tcx, base));
|
|
|
|
alt ty::struct(tcx, auto_unbox.t) {
|
|
|
|
ty::ty_vec(mt) {
|
2011-09-12 04:27:30 -05:00
|
|
|
ds +=
|
|
|
|
[@{mut: mt.mut != imm,
|
|
|
|
kind: index,
|
|
|
|
outer_t: auto_unbox.t}];
|
2011-09-07 08:13:19 -05:00
|
|
|
}
|
|
|
|
ty::ty_str. {
|
2011-09-12 04:27:30 -05:00
|
|
|
ds += [@{mut: false, kind: index, outer_t: auto_unbox.t}];
|
2011-08-31 11:45:37 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
ds += auto_unbox.ds;
|
|
|
|
ex = base;
|
|
|
|
}
|
|
|
|
expr_unary(op, base) {
|
|
|
|
if op == deref {
|
|
|
|
let base_t = ty::expr_ty(tcx, base);
|
|
|
|
let mut = false;
|
|
|
|
alt ty::struct(tcx, base_t) {
|
|
|
|
ty::ty_box(mt) { mut = mt.mut != imm; }
|
|
|
|
ty::ty_uniq(_) { }
|
|
|
|
ty::ty_res(_, _, _) { }
|
|
|
|
ty::ty_tag(_, _) { }
|
|
|
|
ty::ty_ptr(mt) { mut = mt.mut != imm; }
|
|
|
|
}
|
|
|
|
ds += [@{mut: mut, kind: unbox, outer_t: base_t}];
|
|
|
|
ex = base;
|
|
|
|
} else { break; }
|
|
|
|
}
|
|
|
|
_ { break; }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if autoderef {
|
|
|
|
let auto_unbox = maybe_auto_unbox(tcx, ty::expr_ty(tcx, ex));
|
|
|
|
ds += auto_unbox.ds;
|
|
|
|
}
|
|
|
|
ret {ex: ex, ds: @ds};
|
|
|
|
}
|
|
|
|
|
|
|
|
// Actual mut-checking pass
|
|
|
|
|
|
|
|
type mut_map = std::map::hashmap<node_id, ()>;
|
|
|
|
type ctx = {tcx: ty::ctxt, mut_map: mut_map};
|
|
|
|
|
2011-09-12 04:27:30 -05:00
|
|
|
fn check_crate(tcx: ty::ctxt, crate: @crate) -> mut_map {
|
2011-08-31 11:45:37 -05:00
|
|
|
let cx = @{tcx: tcx, mut_map: std::map::new_int_hash()};
|
2011-09-02 17:34:58 -05:00
|
|
|
let v =
|
|
|
|
@{visit_expr: bind visit_expr(cx, _, _, _),
|
|
|
|
visit_decl: bind visit_decl(cx, _, _, _)
|
|
|
|
with *visit::default_visitor::<()>()};
|
2011-08-31 11:45:37 -05:00
|
|
|
visit::visit_crate(*crate, (), visit::mk_vt(v));
|
|
|
|
ret cx.mut_map;
|
|
|
|
}
|
|
|
|
|
2011-09-12 07:24:12 -05:00
|
|
|
tag msg { msg_assign; msg_move_out; msg_mut_ref; }
|
2011-08-31 11:45:37 -05:00
|
|
|
|
2011-09-12 04:27:30 -05:00
|
|
|
fn mk_err(cx: @ctx, span: syntax::codemap::span, msg: msg, name: str) {
|
2011-09-02 17:34:58 -05:00
|
|
|
cx.tcx.sess.span_err(span,
|
|
|
|
alt msg {
|
|
|
|
msg_assign. { "assigning to " + name }
|
|
|
|
msg_move_out. { "moving out of " + name }
|
2011-09-12 07:24:12 -05:00
|
|
|
msg_mut_ref. {
|
|
|
|
"passing " + name + " by mutable reference"
|
2011-09-02 17:34:58 -05:00
|
|
|
}
|
|
|
|
});
|
2011-08-31 11:45:37 -05:00
|
|
|
}
|
|
|
|
|
2011-09-12 04:27:30 -05:00
|
|
|
fn visit_decl(cx: @ctx, d: @decl, e: (), v: visit::vt<()>) {
|
2011-08-31 11:45:37 -05:00
|
|
|
visit::visit_decl(d, e, v);
|
|
|
|
alt d.node {
|
|
|
|
decl_local(locs) {
|
2011-09-15 04:42:56 -05:00
|
|
|
for (_, loc) in locs {
|
2011-08-31 11:45:37 -05:00
|
|
|
alt loc.node.init {
|
|
|
|
some(init) {
|
2011-09-02 17:34:58 -05:00
|
|
|
if init.op == init_move { check_move_rhs(cx, init.expr); }
|
2011-08-31 11:45:37 -05:00
|
|
|
}
|
|
|
|
none. { }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
_ { }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-09-12 04:27:30 -05:00
|
|
|
fn visit_expr(cx: @ctx, ex: @expr, e: (), v: visit::vt<()>) {
|
2011-08-31 11:45:37 -05:00
|
|
|
alt ex.node {
|
2011-09-02 17:34:58 -05:00
|
|
|
expr_call(f, args) { check_call(cx, f, args); }
|
2011-08-31 11:45:37 -05:00
|
|
|
expr_swap(lhs, rhs) {
|
|
|
|
check_lval(cx, lhs, msg_assign);
|
|
|
|
check_lval(cx, rhs, msg_assign);
|
|
|
|
}
|
|
|
|
expr_move(dest, src) {
|
|
|
|
check_lval(cx, dest, msg_assign);
|
|
|
|
check_move_rhs(cx, src);
|
|
|
|
}
|
|
|
|
expr_assign(dest, src) | expr_assign_op(_, dest, src) {
|
|
|
|
check_lval(cx, dest, msg_assign);
|
|
|
|
}
|
2011-09-02 17:34:58 -05:00
|
|
|
_ { }
|
2011-08-31 11:45:37 -05:00
|
|
|
}
|
|
|
|
visit::visit_expr(ex, e, v);
|
|
|
|
}
|
|
|
|
|
2011-09-12 04:27:30 -05:00
|
|
|
fn check_lval(cx: @ctx, dest: @expr, msg: msg) {
|
2011-08-31 11:45:37 -05:00
|
|
|
alt dest.node {
|
|
|
|
expr_path(p) {
|
|
|
|
let def = cx.tcx.def_map.get(dest.id);
|
|
|
|
alt is_immutable_def(def) {
|
|
|
|
some(name) { mk_err(cx, dest.span, msg, name); }
|
2011-09-02 17:34:58 -05:00
|
|
|
_ { }
|
2011-08-31 11:45:37 -05:00
|
|
|
}
|
|
|
|
cx.mut_map.insert(ast_util::def_id_of_def(def).node, ());
|
|
|
|
}
|
|
|
|
_ {
|
|
|
|
let root = expr_root(cx.tcx, dest, false);
|
|
|
|
if vec::len(*root.ds) == 0u {
|
2011-09-12 04:27:30 -05:00
|
|
|
if msg == msg_assign { mk_err(cx, dest.span, msg, "non-lvalue"); }
|
2011-08-31 11:45:37 -05:00
|
|
|
} else if !root.ds[0].mut {
|
2011-09-02 17:34:58 -05:00
|
|
|
let name =
|
|
|
|
alt root.ds[0].kind {
|
|
|
|
mut::unbox. { "immutable box" }
|
|
|
|
mut::field. { "immutable field" }
|
|
|
|
mut::index. { "immutable vec content" }
|
|
|
|
};
|
2011-08-31 11:45:37 -05:00
|
|
|
mk_err(cx, dest.span, msg, name);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-09-12 04:27:30 -05:00
|
|
|
fn check_move_rhs(cx: @ctx, src: @expr) {
|
2011-08-31 11:45:37 -05:00
|
|
|
alt src.node {
|
|
|
|
expr_path(p) {
|
|
|
|
alt cx.tcx.def_map.get(src.id) {
|
|
|
|
def_obj_field(_, _) {
|
2011-09-02 17:34:58 -05:00
|
|
|
mk_err(cx, src.span, msg_move_out, "object field");
|
2011-08-31 11:45:37 -05:00
|
|
|
}
|
|
|
|
_ { }
|
|
|
|
}
|
|
|
|
check_lval(cx, src, msg_move_out);
|
|
|
|
}
|
|
|
|
_ {
|
|
|
|
let root = expr_root(cx.tcx, src, false);
|
2011-09-02 17:34:58 -05:00
|
|
|
|
2011-08-31 11:45:37 -05:00
|
|
|
// Not a path and no-derefs means this is a temporary.
|
|
|
|
if vec::len(*root.ds) != 0u {
|
2011-09-02 17:34:58 -05:00
|
|
|
cx.tcx.sess.span_err(src.span, "moving out of a data structure");
|
2011-08-31 11:45:37 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-09-12 04:27:30 -05:00
|
|
|
fn check_call(cx: @ctx, f: @expr, args: [@expr]) {
|
2011-09-16 07:35:39 -05:00
|
|
|
let arg_ts = ty::ty_fn_args(cx.tcx, ty::expr_ty(cx.tcx, f));
|
2011-08-31 11:45:37 -05:00
|
|
|
let i = 0u;
|
|
|
|
for arg_t: ty::arg in arg_ts {
|
2011-09-12 07:24:12 -05:00
|
|
|
if arg_t.mode != by_ref { check_lval(cx, args[i], msg_mut_ref); }
|
2011-08-31 11:45:37 -05:00
|
|
|
i += 1u;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-09-12 04:27:30 -05:00
|
|
|
fn is_immutable_def(def: def) -> option::t<str> {
|
2011-09-01 07:35:00 -05:00
|
|
|
alt def {
|
2011-08-31 11:45:37 -05:00
|
|
|
def_fn(_, _) | def_mod(_) | def_native_mod(_) | def_const(_) |
|
2011-09-02 17:34:58 -05:00
|
|
|
def_use(_) {
|
|
|
|
some("static item")
|
|
|
|
}
|
|
|
|
def_obj_field(_, imm.) { some("immutable object field") }
|
2011-09-01 07:35:00 -05:00
|
|
|
def_upvar(_, inner, mut) {
|
2011-09-02 17:34:58 -05:00
|
|
|
if !mut { some("upvar") } else { is_immutable_def(*inner) }
|
2011-09-01 07:35:00 -05:00
|
|
|
}
|
2011-09-02 17:34:58 -05:00
|
|
|
def_binding(_) { some("binding") }
|
2011-09-15 07:08:54 -05:00
|
|
|
def_local(_, let_ref.) { some("by-reference binding") }
|
2011-08-31 11:45:37 -05:00
|
|
|
_ { none }
|
2011-09-01 07:35:00 -05:00
|
|
|
}
|
2011-08-31 11:45:37 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
// Local Variables:
|
|
|
|
// mode: rust
|
|
|
|
// fill-column: 78;
|
|
|
|
// indent-tabs-mode: nil
|
|
|
|
// c-basic-offset: 4
|
|
|
|
// buffer-file-coding-system: utf-8-unix
|
|
|
|
// compile-command: "make -k -C $RBUILD 2>&1 | sed -e 's/\\/x\\//x:\\//g'";
|
|
|
|
// End:
|