2011-11-17 16:42:17 +01:00
|
|
|
import syntax::{visit, ast_util};
|
|
|
|
import syntax::ast::*;
|
|
|
|
import std::list::{list, nil, cons, tail};
|
|
|
|
import std::{vec, list, option};
|
|
|
|
|
2011-11-18 17:09:14 +01:00
|
|
|
// Last use analysis pass.
|
|
|
|
//
|
|
|
|
// Finds the last read of each value stored in a local variable or
|
|
|
|
// callee-owned argument (arguments with by-move or by-copy passing
|
|
|
|
// style). This is a limited form of liveness analysis, peformed
|
|
|
|
// (perhaps foolishly) directly on the AST.
|
|
|
|
//
|
|
|
|
// The algorithm walks the AST, keeping a set of (def, last_use)
|
|
|
|
// pairs. When the function is exited, or the local is overwritten,
|
|
|
|
// the current set of last uses is marked with 'true' in a table.
|
|
|
|
// Other branches may later overwrite them with 'false' again, since
|
|
|
|
// they may find a use coming after them. (Marking an expression as a
|
|
|
|
// last use is only done if it has not already been marked with
|
|
|
|
// 'false'.)
|
|
|
|
//
|
|
|
|
// Some complexity is added to deal with joining control flow branches
|
|
|
|
// (by `break` or conditionals), and for handling loops.
|
|
|
|
|
2011-11-17 16:42:17 +01:00
|
|
|
// Marks expr_paths that are last uses.
|
|
|
|
type last_uses = std::map::hashmap<node_id, ()>;
|
|
|
|
|
|
|
|
tag seen { unset; seen(node_id); }
|
|
|
|
tag block_type { func; loop; }
|
|
|
|
|
|
|
|
type set = [{def: node_id, exprs: list<node_id>}];
|
|
|
|
type bl = @{type: block_type, mutable second: bool, mutable exits: [set]};
|
|
|
|
|
|
|
|
type ctx = {last_uses: std::map::hashmap<node_id, bool>,
|
|
|
|
def_map: resolve::def_map,
|
2011-11-21 09:25:42 +01:00
|
|
|
ref_map: alias::ref_map,
|
2011-11-17 16:42:17 +01:00
|
|
|
tcx: ty::ctxt,
|
|
|
|
// The current set of local last uses
|
|
|
|
mutable current: set,
|
|
|
|
mutable blocks: list<bl>};
|
|
|
|
|
2011-11-21 09:25:42 +01:00
|
|
|
fn find_last_uses(c: @crate, def_map: resolve::def_map,
|
|
|
|
ref_map: alias::ref_map, tcx: ty::ctxt) -> last_uses {
|
2011-11-17 16:42:17 +01:00
|
|
|
let v = visit::mk_vt(@{visit_expr: visit_expr,
|
|
|
|
visit_fn: visit_fn
|
|
|
|
with *visit::default_visitor()});
|
|
|
|
let cx = {last_uses: std::map::new_int_hash(),
|
|
|
|
def_map: def_map,
|
2011-11-21 09:25:42 +01:00
|
|
|
ref_map: ref_map,
|
2011-11-17 16:42:17 +01:00
|
|
|
tcx: tcx,
|
|
|
|
mutable current: [],
|
|
|
|
mutable blocks: nil};
|
|
|
|
visit::visit_crate(*c, cx, v);
|
|
|
|
let mini_table = std::map::new_int_hash();
|
2011-11-18 15:10:14 +01:00
|
|
|
cx.last_uses.items {|key, val|
|
|
|
|
if val {
|
|
|
|
mini_table.insert(key, ());
|
|
|
|
let def_node = ast_util::def_id_of_def(def_map.get(key)).node;
|
|
|
|
mini_table.insert(def_node, ());
|
|
|
|
}
|
|
|
|
}
|
2011-11-17 16:42:17 +01:00
|
|
|
ret mini_table;
|
|
|
|
}
|
|
|
|
|
|
|
|
fn visit_expr(ex: @expr, cx: ctx, v: visit::vt<ctx>) {
|
|
|
|
alt ex.node {
|
|
|
|
expr_ret(oexpr) {
|
|
|
|
visit::visit_expr_opt(oexpr, cx, v);
|
|
|
|
if !add_block_exit(cx, func) { leave_fn(cx); }
|
|
|
|
}
|
|
|
|
expr_fail(oexpr) {
|
|
|
|
visit::visit_expr_opt(oexpr, cx, v);
|
|
|
|
leave_fn(cx);
|
|
|
|
}
|
|
|
|
expr_break. { add_block_exit(cx, loop); }
|
|
|
|
expr_while(_, _) | expr_do_while(_, _) {
|
|
|
|
visit_block(loop, cx) {|| visit::visit_expr(ex, cx, v);}
|
|
|
|
}
|
|
|
|
expr_for(_, coll, blk) {
|
|
|
|
v.visit_expr(coll, cx, v);
|
|
|
|
visit_block(loop, cx) {|| visit::visit_block(blk, cx, v);}
|
|
|
|
}
|
|
|
|
expr_ternary(_, _, _) {
|
|
|
|
v.visit_expr(ast_util::ternary_to_if(ex), cx, v);
|
|
|
|
}
|
|
|
|
expr_alt(input, arms) {
|
|
|
|
v.visit_expr(input, cx, v);
|
|
|
|
let before = cx.current, sets = [];
|
|
|
|
for arm in arms {
|
|
|
|
cx.current = before;
|
|
|
|
v.visit_arm(arm, cx, v);
|
|
|
|
sets += [cx.current];
|
|
|
|
}
|
|
|
|
cx.current = join_branches(sets);
|
|
|
|
}
|
|
|
|
expr_if(cond, then, els) {
|
|
|
|
v.visit_expr(cond, cx, v);
|
|
|
|
let cur = cx.current;
|
|
|
|
visit::visit_block(then, cx, v);
|
|
|
|
cx.current <-> cur;
|
|
|
|
visit::visit_expr_opt(els, cx, v);
|
|
|
|
cx.current = join_branches([cur, cx.current]);
|
|
|
|
}
|
|
|
|
expr_path(_) {
|
2011-11-21 09:25:42 +01:00
|
|
|
let my_def = ast_util::def_id_of_def(cx.def_map.get(ex.id)).node;
|
|
|
|
alt cx.ref_map.find(my_def) {
|
|
|
|
option::some(root_id) { clear_in_current(cx, root_id, false); }
|
|
|
|
_ {
|
|
|
|
alt clear_if_path(cx, ex, v, false) {
|
|
|
|
option::some(my_def) {
|
|
|
|
cx.current += [{def: my_def, exprs: cons(ex.id, @nil)}];
|
|
|
|
}
|
|
|
|
_ {}
|
|
|
|
}
|
2011-11-17 16:42:17 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
expr_swap(lhs, rhs) {
|
|
|
|
clear_if_path(cx, lhs, v, false);
|
|
|
|
clear_if_path(cx, rhs, v, false);
|
|
|
|
}
|
|
|
|
expr_move(dest, src) | expr_assign(dest, src) {
|
|
|
|
v.visit_expr(src, cx, v);
|
|
|
|
clear_if_path(cx, dest, v, true);
|
|
|
|
}
|
|
|
|
expr_assign_op(_, dest, src) {
|
|
|
|
v.visit_expr(src, cx, v);
|
|
|
|
v.visit_expr(dest, cx, v);
|
|
|
|
clear_if_path(cx, dest, v, true);
|
|
|
|
}
|
|
|
|
expr_call(f, args, _) {
|
|
|
|
v.visit_expr(f, cx, v);
|
2011-12-01 16:47:07 +01:00
|
|
|
let i = 0u, fns = [];
|
|
|
|
let arg_ts = ty::ty_fn_args(cx.tcx, ty::expr_ty(cx.tcx, f));
|
|
|
|
for arg in args {
|
|
|
|
alt arg.node {
|
|
|
|
expr_fn(_) { fns += [arg]; }
|
|
|
|
_ {
|
|
|
|
alt arg_ts[i].mode {
|
|
|
|
by_mut_ref. { clear_if_path(cx, arg, v, false); }
|
|
|
|
_ { v.visit_expr(arg, cx, v); }
|
|
|
|
}
|
|
|
|
}
|
2011-11-17 16:42:17 +01:00
|
|
|
}
|
|
|
|
i += 1u;
|
|
|
|
}
|
2011-12-01 16:47:07 +01:00
|
|
|
for f in fns { v.visit_expr(f, cx, v); }
|
2011-11-17 16:42:17 +01:00
|
|
|
}
|
|
|
|
_ { visit::visit_expr(ex, cx, v); }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn visit_fn(f: _fn, tps: [ty_param], sp: syntax::codemap::span,
|
|
|
|
ident: fn_ident, id: node_id, cx: ctx, v: visit::vt<ctx>) {
|
|
|
|
if f.proto == proto_block {
|
|
|
|
visit_block(func, cx, {||
|
|
|
|
visit::visit_fn(f, tps, sp, ident, id, cx, v);
|
|
|
|
});
|
|
|
|
} else {
|
|
|
|
let old = nil;
|
|
|
|
cx.blocks <-> old;
|
|
|
|
visit::visit_fn(f, tps, sp, ident, id, cx, v);
|
|
|
|
cx.blocks <-> old;
|
|
|
|
leave_fn(cx);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn visit_block(tp: block_type, cx: ctx, visit: block()) {
|
|
|
|
let local = @{type: tp, mutable second: false, mutable exits: []};
|
|
|
|
cx.blocks = cons(local, @cx.blocks);
|
|
|
|
visit();
|
|
|
|
local.second = true;
|
|
|
|
visit();
|
|
|
|
cx.blocks = tail(cx.blocks);
|
|
|
|
cx.current = join_branches(local.exits);
|
|
|
|
}
|
|
|
|
|
|
|
|
fn add_block_exit(cx: ctx, tp: block_type) -> bool {
|
|
|
|
let cur = cx.blocks;
|
|
|
|
while cur != nil {
|
|
|
|
alt cur {
|
|
|
|
cons(b, tail) {
|
|
|
|
if (b.type == tp) {
|
|
|
|
if !b.second { b.exits += [cx.current]; }
|
|
|
|
ret true;
|
|
|
|
}
|
|
|
|
cur = *tail;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
ret false;
|
|
|
|
}
|
|
|
|
|
|
|
|
fn join_branches(branches: [set]) -> set {
|
|
|
|
let found: set = [], i = 0u, l = vec::len(branches);
|
|
|
|
for set in branches {
|
|
|
|
i += 1u;
|
|
|
|
for {def, exprs} in set {
|
|
|
|
if !vec::any({|v| v.def == def}, found) {
|
|
|
|
let j = i, ne = exprs;
|
|
|
|
while j < l {
|
|
|
|
for {def: d2, exprs} in branches[j] {
|
|
|
|
if d2 == def {
|
|
|
|
list::iter(exprs) {|e|
|
|
|
|
if !list::has(ne, e) { ne = cons(e, @ne); }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
j += 1u;
|
|
|
|
}
|
|
|
|
found += [{def: def, exprs: ne}];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
ret found;
|
|
|
|
}
|
|
|
|
|
|
|
|
fn leave_fn(cx: ctx) {
|
|
|
|
for {def, exprs} in cx.current {
|
|
|
|
list::iter(exprs) {|ex_id|
|
|
|
|
if !cx.last_uses.contains_key(ex_id) {
|
|
|
|
cx.last_uses.insert(ex_id, true);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn clear_in_current(cx: ctx, my_def: node_id, to: bool) {
|
|
|
|
for {def, exprs} in cx.current {
|
|
|
|
if def == my_def {
|
|
|
|
list::iter(exprs) {|expr|
|
|
|
|
if !to || !cx.last_uses.contains_key(expr) {
|
|
|
|
cx.last_uses.insert(expr, to);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
cx.current = vec::filter({|x| x.def != my_def},
|
|
|
|
copy cx.current);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn clear_if_path(cx: ctx, ex: @expr, v: visit::vt<ctx>, to: bool)
|
|
|
|
-> option::t<node_id> {
|
|
|
|
alt ex.node {
|
|
|
|
expr_path(_) {
|
|
|
|
alt cx.def_map.get(ex.id) {
|
|
|
|
def_local(def_id, let_copy.) | def_arg(def_id, by_copy.) |
|
|
|
|
def_arg(def_id, by_move.) {
|
|
|
|
clear_in_current(cx, def_id.node, to);
|
|
|
|
ret option::some(def_id.node);
|
|
|
|
}
|
|
|
|
_ {}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
_ { v.visit_expr(ex, cx, v); }
|
|
|
|
}
|
|
|
|
ret option::none;
|
|
|
|
}
|