2014-02-10 15:36:31 +01:00
|
|
|
// Copyright 2012-2014 The Rust Project Developers. See the COPYRIGHT
|
2012-12-03 16:48:01 -08:00
|
|
|
// 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.
|
|
|
|
|
2011-07-18 19:14:01 -07:00
|
|
|
// A pass that annotates for each loops and functions with the free
|
|
|
|
// variables that they contain.
|
2011-07-18 17:26:37 -07:00
|
|
|
|
2014-02-10 15:36:31 +01:00
|
|
|
#[allow(non_camel_case_types)];
|
2013-05-17 15:28:44 -07:00
|
|
|
|
2012-12-23 17:41:37 -05:00
|
|
|
use middle::resolve;
|
|
|
|
use middle::ty;
|
2014-02-28 14:34:26 -08:00
|
|
|
use util::nodemap::{NodeMap, NodeSet};
|
2012-12-23 17:41:37 -05:00
|
|
|
|
2013-08-31 18:13:04 +02:00
|
|
|
use syntax::codemap::Span;
|
2013-08-13 13:37:54 +02:00
|
|
|
use syntax::{ast, ast_util};
|
|
|
|
use syntax::visit;
|
|
|
|
use syntax::visit::Visitor;
|
2011-07-18 17:26:37 -07:00
|
|
|
|
2011-09-01 14:35:00 +02:00
|
|
|
// A vector of defs representing the free variables referred to in a function.
|
|
|
|
// (The def_upvar will already have been stripped).
|
2013-05-16 08:55:57 +10:00
|
|
|
#[deriving(Encodable, Decodable)]
|
2013-01-30 13:44:24 -08:00
|
|
|
pub struct freevar_entry {
|
2013-09-02 03:45:37 +02:00
|
|
|
def: ast::Def, //< The variable being accessed free.
|
2013-08-31 18:13:04 +02:00
|
|
|
span: Span //< First span where it is accessed (there can be multiple)
|
2013-01-16 19:24:10 -08:00
|
|
|
}
|
2014-03-04 10:02:49 -08:00
|
|
|
pub type freevar_info = @Vec<@freevar_entry> ;
|
2014-02-28 14:34:26 -08:00
|
|
|
pub type freevar_map = NodeMap<freevar_info>;
|
2011-07-18 17:26:37 -07:00
|
|
|
|
2013-08-13 13:37:54 +02:00
|
|
|
struct CollectFreevarsVisitor {
|
2014-02-28 14:34:26 -08:00
|
|
|
seen: NodeSet,
|
2014-03-04 10:02:49 -08:00
|
|
|
refs: Vec<@freevar_entry> ,
|
2013-08-13 13:37:54 +02:00
|
|
|
def_map: resolve::DefMap,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Visitor<int> for CollectFreevarsVisitor {
|
2011-07-18 17:26:37 -07:00
|
|
|
|
2014-01-09 15:05:33 +02:00
|
|
|
fn visit_item(&mut self, _: &ast::Item, _: int) {
|
2013-08-13 13:37:54 +02:00
|
|
|
// ignore_item
|
|
|
|
}
|
|
|
|
|
2014-01-06 14:00:46 +02:00
|
|
|
fn visit_expr(&mut self, expr: &ast::Expr, depth: int) {
|
|
|
|
match expr.node {
|
|
|
|
ast::ExprFnBlock(..) | ast::ExprProc(..) => {
|
2013-08-13 13:37:54 +02:00
|
|
|
visit::walk_expr(self, expr, depth + 1)
|
2014-01-06 14:00:46 +02:00
|
|
|
}
|
2014-01-27 14:18:36 +02:00
|
|
|
ast::ExprPath(..) => {
|
2014-01-06 14:00:46 +02:00
|
|
|
let mut i = 0;
|
|
|
|
let def_map = self.def_map.borrow();
|
|
|
|
match def_map.get().find(&expr.id) {
|
2013-10-21 13:08:31 -07:00
|
|
|
None => fail!("path not found"),
|
2013-03-22 22:26:41 -04:00
|
|
|
Some(&df) => {
|
2014-01-06 14:00:46 +02:00
|
|
|
let mut def = df;
|
|
|
|
while i < depth {
|
|
|
|
match def {
|
|
|
|
ast::DefUpvar(_, inner, _, _) => { def = *inner; }
|
|
|
|
_ => break
|
|
|
|
}
|
|
|
|
i += 1;
|
2012-01-05 11:33:22 -08:00
|
|
|
}
|
2014-01-06 14:00:46 +02:00
|
|
|
if i == depth { // Made it to end of loop
|
|
|
|
let dnum = ast_util::def_id_of_def(def).node;
|
2014-02-28 14:34:26 -08:00
|
|
|
if !self.seen.contains(&dnum) {
|
2014-01-06 14:00:46 +02:00
|
|
|
self.refs.push(@freevar_entry {
|
|
|
|
def: def,
|
|
|
|
span: expr.span,
|
|
|
|
});
|
2014-02-28 14:34:26 -08:00
|
|
|
self.seen.insert(dnum);
|
2014-01-06 14:00:46 +02:00
|
|
|
}
|
2012-01-05 11:33:22 -08:00
|
|
|
}
|
2011-09-02 15:34:58 -07:00
|
|
|
}
|
2014-01-06 14:00:46 +02:00
|
|
|
}
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
2014-01-06 14:00:46 +02:00
|
|
|
_ => visit::walk_expr(self, expr, depth)
|
|
|
|
}
|
2013-08-13 13:37:54 +02:00
|
|
|
}
|
2011-09-01 14:35:00 +02:00
|
|
|
|
2013-08-13 13:37:54 +02: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.
|
2014-01-06 14:00:46 +02:00
|
|
|
fn collect_freevars(def_map: resolve::DefMap, blk: &ast::Block) -> freevar_info {
|
2014-02-28 14:34:26 -08:00
|
|
|
let seen = NodeSet::new();
|
2014-03-04 10:02:49 -08:00
|
|
|
let refs = Vec::new();
|
2013-08-13 13:37:54 +02:00
|
|
|
|
|
|
|
let mut v = CollectFreevarsVisitor {
|
|
|
|
seen: seen,
|
|
|
|
refs: refs,
|
|
|
|
def_map: def_map,
|
|
|
|
};
|
|
|
|
|
|
|
|
v.visit_block(blk, 1);
|
2013-12-20 20:24:18 -08:00
|
|
|
let CollectFreevarsVisitor {
|
|
|
|
refs,
|
|
|
|
..
|
|
|
|
} = v;
|
|
|
|
return @refs;
|
2011-07-18 17:26:37 -07:00
|
|
|
}
|
|
|
|
|
2013-08-13 13:37:54 +02:00
|
|
|
struct AnnotateFreevarsVisitor {
|
|
|
|
def_map: resolve::DefMap,
|
|
|
|
freevars: freevar_map,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Visitor<()> for AnnotateFreevarsVisitor {
|
2014-01-09 15:05:33 +02:00
|
|
|
fn visit_fn(&mut self, fk: &visit::FnKind, fd: &ast::FnDecl,
|
2014-01-06 14:00:46 +02:00
|
|
|
blk: &ast::Block, s: Span, nid: ast::NodeId, _: ()) {
|
2013-08-13 13:37:54 +02:00
|
|
|
let vars = collect_freevars(self.def_map, blk);
|
|
|
|
self.freevars.insert(nid, vars);
|
|
|
|
visit::walk_fn(self, fk, fd, blk, s, nid, ());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-07-18 19:14:01 -07:00
|
|
|
// 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.
|
2014-02-05 22:15:24 +01:00
|
|
|
pub fn annotate_freevars(def_map: resolve::DefMap, krate: &ast::Crate) ->
|
2011-09-02 15:34:58 -07:00
|
|
|
freevar_map {
|
2013-08-13 13:37:54 +02:00
|
|
|
let mut visitor = AnnotateFreevarsVisitor {
|
|
|
|
def_map: def_map,
|
2014-02-28 14:34:26 -08:00
|
|
|
freevars: NodeMap::new(),
|
2011-12-20 11:03:21 -08:00
|
|
|
};
|
2014-02-05 22:15:24 +01:00
|
|
|
visit::walk_crate(&mut visitor, krate, ());
|
2011-07-18 19:14:01 -07:00
|
|
|
|
2013-12-20 20:29:03 -08:00
|
|
|
let AnnotateFreevarsVisitor {
|
|
|
|
freevars,
|
|
|
|
..
|
|
|
|
} = visitor;
|
|
|
|
freevars
|
2011-07-18 19:14:01 -07:00
|
|
|
}
|
|
|
|
|
2013-07-27 10:25:59 +02:00
|
|
|
pub fn get_freevars(tcx: ty::ctxt, fid: ast::NodeId) -> freevar_info {
|
2013-12-20 20:29:03 -08:00
|
|
|
let freevars = tcx.freevars.borrow();
|
|
|
|
match freevars.get().find(&fid) {
|
|
|
|
None => fail!("get_freevars: {} has no freevars", fid),
|
|
|
|
Some(&d) => return d
|
2011-07-21 14:49:58 -07:00
|
|
|
}
|
|
|
|
}
|
2013-01-10 10:59:58 -08:00
|
|
|
|
2013-07-27 10:25:59 +02:00
|
|
|
pub fn has_freevars(tcx: ty::ctxt, fid: ast::NodeId) -> bool {
|
2013-06-08 21:38:47 -04:00
|
|
|
!get_freevars(tcx, fid).is_empty()
|
2011-07-21 14:49:58 -07:00
|
|
|
}
|