Implement checking against assignments to immutable obj fields

This commit is contained in:
Marijn Haverbeke 2011-06-15 14:59:51 +02:00
parent b0d46efd07
commit c7ab88cc35
4 changed files with 61 additions and 38 deletions

View File

@ -30,35 +30,53 @@ type restrict = @rec(vec[def_num] root_vars,
vec[ty::t] tys, vec[ty::t] tys,
vec[uint] depends_on, vec[uint] depends_on,
mutable valid ok); mutable valid ok);
type scope = vec[restrict];
type scope = rec(vec[tup(def_num, ast::mode)] args, tag local_info {
vec[restrict] rs); arg(ast::mode);
fn scope(&scope sc, vec[restrict] add) -> scope { objfield(ast::mutability);
ret rec(args=sc.args, rs=sc.rs + add);
} }
type ctx = rec(@ty::ctxt tcx, type ctx = rec(@ty::ctxt tcx,
resolve::def_map dm); resolve::def_map dm,
std::map::hashmap[def_num,local_info] local_map);
fn check_crate(@ty::ctxt tcx, resolve::def_map dm, &@ast::crate crate) { fn check_crate(@ty::ctxt tcx, resolve::def_map dm, &@ast::crate crate) {
auto cx = @rec(tcx = tcx, dm = dm); auto cx = @rec(tcx = tcx,
auto v = @rec(visit_fn = visit_fn, dm = dm,
// Stores information about object fields and function
// arguments that's otherwise not easily available.
local_map = util::common::new_int_hash());
auto v = @rec(visit_fn = bind visit_fn(cx, _, _, _, _, _, _, _, _),
visit_item = bind visit_item(cx, _, _, _),
visit_expr = bind visit_expr(cx, _, _, _) visit_expr = bind visit_expr(cx, _, _, _)
with *visit::default_visitor[scope]()); with *visit::default_visitor[scope]());
visit::visit_crate(*crate, rec(args=[], rs=[]), visit::vtor(v)); visit::visit_crate(*crate, [], visit::vtor(v));
} }
fn visit_fn(&ast::_fn f, &vec[ast::ty_param] tp, &span sp, &ident name, fn visit_fn(@ctx cx, &ast::_fn f, &vec[ast::ty_param] tp, &span sp,
&ast::def_id d_id, &ast::ann a, &scope sc, &vt[scope] v) { &ident name, &ast::def_id d_id, &ast::ann a, &scope sc,
&vt[scope] v) {
visit::visit_fn_decl(f.decl, sc, v); visit::visit_fn_decl(f.decl, sc, v);
auto args = []; for (ast::arg arg_ in f.decl.inputs) {
for (ast::arg arg in f.decl.inputs) { cx.local_map.insert(arg_.id._1, arg(arg_.mode));
vec::push(args, tup(arg.id._1, arg.mode));
} }
vt(v).visit_block(f.body, rec(args=args, rs=[]), v); vt(v).visit_block(f.body, [], v);
} }
fn visit_expr(&@ctx cx, &@ast::expr ex, &scope sc, &vt[scope] v) { fn visit_item(@ctx cx, &@ast::item i, &scope sc, &vt[scope] v) {
alt (i.node) {
case (ast::item_obj(_, ?o, _, _, _)) {
for (ast::obj_field f in o.fields) {
cx.local_map.insert(f.id._1, objfield(f.mut));
}
}
case (_) {}
}
visit::visit_item(i, sc, v);
}
fn visit_expr(@ctx cx, &@ast::expr ex, &scope sc, &vt[scope] v) {
auto handled = false; auto handled = false;
alt (ex.node) { alt (ex.node) {
case (ast::expr_call(?f, ?args, _)) { case (ast::expr_call(?f, ?args, _)) {
@ -193,7 +211,7 @@ fn check_call(&ctx cx, &@ast::expr f, &vec[@ast::expr] args, &scope sc)
// Ensure we're not passing a root by mutable alias. // Ensure we're not passing a root by mutable alias.
for (tup(uint, def_num) root in mut_roots) { for (tup(uint, def_num) root in mut_roots) {
auto mut_alias_to_root = vec::count(root._1, roots) > 1u; auto mut_alias_to_root = vec::count(root._1, roots) > 1u;
for (restrict r in sc.rs) { for (restrict r in sc) {
if (vec::member(root._1, r.root_vars)) { if (vec::member(root._1, r.root_vars)) {
mut_alias_to_root = true; mut_alias_to_root = true;
} }
@ -225,12 +243,12 @@ fn check_alt(&ctx cx, &@ast::expr input, &vec[ast::arm] arms,
auto dnums = arm_defnums(a); auto dnums = arm_defnums(a);
auto new_sc = sc; auto new_sc = sc;
if (vec::len(dnums) > 0u) { if (vec::len(dnums) > 0u) {
new_sc = scope(sc, [@rec(root_vars=roots, new_sc = sc + [@rec(root_vars=roots,
block_defnum=dnums.(0), block_defnum=dnums.(0),
bindings=dnums, bindings=dnums,
tys=forbidden_tp, tys=forbidden_tp,
depends_on=deps(sc, roots), depends_on=deps(sc, roots),
mutable ok=valid)]); mutable ok=valid)];
} }
visit::visit_arm(a, new_sc, v); visit::visit_arm(a, new_sc, v);
} }
@ -269,7 +287,7 @@ fn check_for_each(&ctx cx, &@ast::local local, &@ast::expr call,
tys=data.unsafe_ts, tys=data.unsafe_ts,
depends_on=deps(sc, data.root_vars), depends_on=deps(sc, data.root_vars),
mutable ok=valid); mutable ok=valid);
visit::visit_block(block, scope(sc, [new_sc]), v); visit::visit_block(block, sc + [new_sc], v);
} }
} }
} }
@ -303,7 +321,7 @@ fn check_for(&ctx cx, &@ast::local local, &@ast::expr seq,
tys=unsafe, tys=unsafe,
depends_on=deps(sc, root_def), depends_on=deps(sc, root_def),
mutable ok=valid); mutable ok=valid);
visit::visit_block(block, scope(sc, [new_sc]), v); visit::visit_block(block, sc + [new_sc], v);
} }
fn check_var(&ctx cx, &@ast::expr ex, &ast::path p, ast::ann ann, bool assign, fn check_var(&ctx cx, &@ast::expr ex, &ast::path p, ast::ann ann, bool assign,
@ -312,7 +330,7 @@ fn check_var(&ctx cx, &@ast::expr ex, &ast::path p, ast::ann ann, bool assign,
if (!def_is_local(def)) { ret; } if (!def_is_local(def)) { ret; }
auto my_defnum = ast::def_id_of_def(def)._1; auto my_defnum = ast::def_id_of_def(def)._1;
auto var_t = ty::expr_ty(*cx.tcx, ex); auto var_t = ty::expr_ty(*cx.tcx, ex);
for (restrict r in sc.rs) { for (restrict r in sc) {
// excludes variables introduced since the alias was made // excludes variables introduced since the alias was made
if (my_defnum < r.block_defnum) { if (my_defnum < r.block_defnum) {
for (ty::t t in r.tys) { for (ty::t t in r.tys) {
@ -334,13 +352,16 @@ fn check_assign(&@ctx cx, &@ast::expr dest, &@ast::expr src,
alt (dest.node) { alt (dest.node) {
case (ast::expr_path(?p, ?ann)) { case (ast::expr_path(?p, ?ann)) {
auto dnum = ast::def_id_of_def(cx.dm.get(ann.id))._1; auto dnum = ast::def_id_of_def(cx.dm.get(ann.id))._1;
if (is_immutable_alias(sc, dnum)) { if (is_immutable_alias(cx, sc, dnum)) {
cx.tcx.sess.span_err cx.tcx.sess.span_err
(dest.span, "assigning to immutable alias"); (dest.span, "assigning to immutable alias");
} else if (is_immutable_objfield(cx, dnum)) {
cx.tcx.sess.span_err
(dest.span, "assigning to immutable obj field");
} }
auto var_t = ty::expr_ty(*cx.tcx, dest); auto var_t = ty::expr_ty(*cx.tcx, dest);
for (restrict r in sc.rs) { for (restrict r in sc) {
if (vec::member(dnum, r.root_vars)) { if (vec::member(dnum, r.root_vars)) {
r.ok = overwritten(dest.span, p); r.ok = overwritten(dest.span, p);
} }
@ -365,21 +386,25 @@ fn check_assign(&@ctx cx, &@ast::expr dest, &@ast::expr src,
} }
} }
fn is_immutable_alias(&scope sc, def_num dnum) -> bool { fn is_immutable_alias(&@ctx cx, &scope sc, def_num dnum) -> bool {
for (tup(def_num, ast::mode) arg in sc.args) { alt (cx.local_map.find(dnum)) {
if (arg._0 == dnum && arg._1 == ast::alias(false)) { ret true; } case (some(arg(ast::alias(false)))) { ret true; }
case (_) {}
} }
for (restrict r in sc.rs) { for (restrict r in sc) {
if (vec::member(dnum, r.bindings)) { ret true; } if (vec::member(dnum, r.bindings)) { ret true; }
} }
ret false; ret false;
} }
fn is_immutable_objfield(&@ctx cx, def_num dnum) -> bool {
ret cx.local_map.find(dnum) == some(objfield(ast::imm));
}
fn test_scope(&ctx cx, &scope sc, &restrict r, &ast::path p) { fn test_scope(&ctx cx, &scope sc, &restrict r, &ast::path p) {
auto prob = r.ok; auto prob = r.ok;
for (uint dep in r.depends_on) { for (uint dep in r.depends_on) {
if (prob != valid) { break; } if (prob != valid) { break; }
prob = sc.rs.(dep).ok; prob = sc.(dep).ok;
} }
if (prob != valid) { if (prob != valid) {
auto msg = alt (prob) { auto msg = alt (prob) {
@ -399,7 +424,7 @@ fn test_scope(&ctx cx, &scope sc, &restrict r, &ast::path p) {
fn deps(&scope sc, vec[def_num] roots) -> vec[uint] { fn deps(&scope sc, vec[def_num] roots) -> vec[uint] {
auto i = 0u; auto i = 0u;
auto result = []; auto result = [];
for (restrict r in sc.rs) { for (restrict r in sc) {
for (def_num dn in roots) { for (def_num dn in roots) {
if (vec::member(dn, r.bindings)) { if (vec::member(dn, r.bindings)) {
vec::push(result, i); vec::push(result, i);

View File

@ -24,7 +24,7 @@ type visitor[E] =
fn(&@view_item v, &E e, &vt[E] v) visit_view_item, fn(&@view_item v, &E e, &vt[E] v) visit_view_item,
fn(&@native_item i, &E e, &vt[E] v) visit_native_item, fn(&@native_item i, &E e, &vt[E] v) visit_native_item,
fn(&@item i, &E e, &vt[E] v) visit_item, fn(&@item i, &E e, &vt[E] v) visit_item,
fn(&@local l, &E e, &vt[E] v) visit_local, fn(&@local l, &E e, &vt[E] v) visit_local,
fn(&block b, &E e, &vt[E] v) visit_block, fn(&block b, &E e, &vt[E] v) visit_block,
fn(&@stmt s, &E e, &vt[E] v) visit_stmt, fn(&@stmt s, &E e, &vt[E] v) visit_stmt,
fn(&arm a, &E e, &vt[E] v) visit_arm, fn(&arm a, &E e, &vt[E] v) visit_arm,

View File

@ -1,7 +1,5 @@
// xfail-stage0 // xfail-stage0
// xfail-stage1 // error-pattern:assigning to immutable obj field
// xfail-stage2
// error-pattern: writing to immutable type
obj objy(int x) { obj objy(int x) {
fn foo() -> () { fn foo() -> () {
x = 5; x = 5;

View File

@ -30,7 +30,7 @@ fn slow_growth() {
fn slow_growth2_helper(str s) { // ref up: s fn slow_growth2_helper(str s) { // ref up: s
obj acc(vec[str] v) { obj acc(mutable vec[str] v) {
fn add(&str s) { v += [s]; } fn add(&str s) { v += [s]; }
} }