rust/src/librustc/middle/freevars.rs

152 lines
4.8 KiB
Rust
Raw Normal View History

// Copyright 2012 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.
// A pass that annotates for each loops and functions with the free
// variables that they contain.
2011-07-18 19:26:37 -05:00
use middle::resolve;
use middle::ty;
use std::hashmap::HashMap;
use syntax::codemap::Span;
use syntax::{ast, ast_util};
use syntax::visit;
use syntax::visit::Visitor;
use syntax::ast::{item};
2011-07-18 19:26:37 -05:00
// A vector of defs representing the free variables referred to in a function.
// (The def_upvar will already have been stripped).
#[deriving(Encodable, Decodable)]
pub struct freevar_entry {
def: ast::Def, //< The variable being accessed free.
span: Span //< First span where it is accessed (there can be multiple)
}
pub type freevar_info = @~[@freevar_entry];
pub type freevar_map = @mut HashMap<ast::NodeId, freevar_info>;
2011-07-18 19:26:37 -05:00
struct CollectFreevarsVisitor {
seen: HashMap<ast::NodeId, ()>,
refs: ~[@freevar_entry],
def_map: resolve::DefMap,
}
impl Visitor<int> for CollectFreevarsVisitor {
2011-07-18 19:26:37 -05:00
fn visit_item(&mut self, _:@item, _:int) {
// ignore_item
}
fn visit_expr(&mut self, expr:@ast::Expr, depth:int) {
2012-08-06 14:34:08 -05:00
match expr.node {
2013-11-28 14:22:53 -06:00
ast::ExprFnBlock(..) | ast::ExprProc(..) => {
visit::walk_expr(self, expr, depth + 1)
}
2013-11-28 14:22:53 -06:00
ast::ExprPath(..) | ast::ExprSelf => {
let mut i = 0;
match self.def_map.find(&expr.id) {
None => fail!("path not found"),
2013-03-22 21:26:41 -05:00
Some(&df) => {
let mut def = df;
while i < depth {
match def {
ast::DefUpvar(_, inner, _, _) => { def = *inner; }
2012-08-03 21:59:04 -05:00
_ => break
}
i += 1;
}
if i == depth { // Made it to end of loop
let dnum = ast_util::def_id_of_def(def).node;
if !self.seen.contains_key(&dnum) {
self.refs.push(@freevar_entry {
def: def,
span: expr.span,
});
self.seen.insert(dnum, ());
}
}
2011-09-02 17:34:58 -05:00
}
}
2011-09-02 17:34:58 -05:00
}
_ => visit::walk_expr(self, expr, depth)
2011-07-27 07:19:39 -05:00
}
}
}
// Searches through part of the AST for all references to locals or
// upvars in this frame and returns the list of definition IDs thus found.
// Since we want to be able to collect upvars in some arbitrary piece
// of the AST, we take a walker function that we invoke with a visitor
// in order to start the search.
fn collect_freevars(def_map: resolve::DefMap, blk: ast::P<ast::Block>)
-> freevar_info {
let seen = HashMap::new();
let refs = ~[];
let mut v = CollectFreevarsVisitor {
seen: seen,
refs: refs,
def_map: def_map,
};
v.visit_block(blk, 1);
let CollectFreevarsVisitor {
refs,
..
} = v;
return @refs;
2011-07-18 19:26:37 -05:00
}
struct AnnotateFreevarsVisitor {
def_map: resolve::DefMap,
freevars: freevar_map,
}
impl Visitor<()> for AnnotateFreevarsVisitor {
fn visit_fn(&mut self, fk:&visit::fn_kind, fd:&ast::fn_decl,
blk:ast::P<ast::Block>, s:Span, nid:ast::NodeId, _:()) {
let vars = collect_freevars(self.def_map, blk);
self.freevars.insert(nid, vars);
visit::walk_fn(self, fk, fd, blk, s, nid, ());
}
}
// Build a map from every function and for-each body to a set of the
// freevars contained in it. The implementation is not particularly
// efficient as it fully recomputes the free variables at every
// node of interest rather than building up the free variables in
// one pass. This could be improved upon if it turns out to matter.
pub fn annotate_freevars(def_map: resolve::DefMap, crate: &ast::Crate) ->
2011-09-02 17:34:58 -05:00
freevar_map {
let freevars = @mut HashMap::new();
let mut visitor = AnnotateFreevarsVisitor {
def_map: def_map,
freevars: freevars,
};
visit::walk_crate(&mut visitor, crate, ());
2012-08-01 19:30:05 -05:00
return freevars;
}
pub fn get_freevars(tcx: ty::ctxt, fid: ast::NodeId) -> freevar_info {
match tcx.freevars.find(&fid) {
None => fail!("get_freevars: {} has no freevars", fid),
2013-03-22 21:26:41 -05:00
Some(&d) => return d
2011-07-21 16:49:58 -05:00
}
}
pub fn has_freevars(tcx: ty::ctxt, fid: ast::NodeId) -> bool {
!get_freevars(tcx, fid).is_empty()
2011-07-21 16:49:58 -05:00
}