191 lines
6.5 KiB
Rust
191 lines
6.5 KiB
Rust
// Copyright 2012-2014 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.
|
|
|
|
#![allow(non_camel_case_types)]
|
|
|
|
use middle::def;
|
|
use middle::mem_categorization::Typer;
|
|
use middle::resolve;
|
|
use middle::ty;
|
|
use util::nodemap::{NodeMap, NodeSet};
|
|
|
|
use syntax::ast;
|
|
use syntax::codemap::Span;
|
|
use syntax::visit::Visitor;
|
|
use syntax::visit;
|
|
|
|
#[deriving(Clone, Decodable, Encodable, Show)]
|
|
pub enum CaptureMode {
|
|
/// Copy/move the value from this llvm ValueRef into the environment.
|
|
CaptureByValue,
|
|
|
|
/// Access by reference (used for stack closures).
|
|
CaptureByRef
|
|
}
|
|
|
|
// 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 {
|
|
pub def: def::Def, //< The variable being accessed free.
|
|
pub span: Span //< First span where it is accessed (there can be multiple)
|
|
}
|
|
|
|
pub type freevar_map = NodeMap<Vec<freevar_entry>>;
|
|
|
|
pub type CaptureModeMap = NodeMap<CaptureMode>;
|
|
|
|
struct CollectFreevarsVisitor<'a> {
|
|
seen: NodeSet,
|
|
refs: Vec<freevar_entry>,
|
|
def_map: &'a resolve::DefMap,
|
|
capture_mode_map: &'a mut CaptureModeMap,
|
|
depth: uint
|
|
}
|
|
|
|
impl<'a, 'v> Visitor<'v> for CollectFreevarsVisitor<'a> {
|
|
fn visit_item(&mut self, _: &ast::Item) {
|
|
// ignore_item
|
|
}
|
|
|
|
fn visit_expr(&mut self, expr: &ast::Expr) {
|
|
match expr.node {
|
|
ast::ExprProc(..) => {
|
|
self.capture_mode_map.insert(expr.id, CaptureByValue);
|
|
self.depth += 1;
|
|
visit::walk_expr(self, expr);
|
|
self.depth -= 1;
|
|
}
|
|
ast::ExprFnBlock(_, _, _) => {
|
|
// NOTE(stage0): After snapshot, change to:
|
|
//
|
|
//let capture_mode = match capture_clause {
|
|
// ast::CaptureByValue => CaptureByValue,
|
|
// ast::CaptureByRef => CaptureByRef,
|
|
//};
|
|
let capture_mode = CaptureByRef;
|
|
self.capture_mode_map.insert(expr.id, capture_mode);
|
|
self.depth += 1;
|
|
visit::walk_expr(self, expr);
|
|
self.depth -= 1;
|
|
}
|
|
ast::ExprUnboxedFn(capture_clause, _, _, _) => {
|
|
let capture_mode = match capture_clause {
|
|
ast::CaptureByValue => CaptureByValue,
|
|
ast::CaptureByRef => CaptureByRef,
|
|
};
|
|
self.capture_mode_map.insert(expr.id, capture_mode);
|
|
self.depth += 1;
|
|
visit::walk_expr(self, expr);
|
|
self.depth -= 1;
|
|
}
|
|
ast::ExprPath(..) => {
|
|
let mut def = *self.def_map.borrow().find(&expr.id)
|
|
.expect("path not found");
|
|
let mut i = 0;
|
|
while i < self.depth {
|
|
match def {
|
|
def::DefUpvar(_, inner, _, _) => { def = *inner; }
|
|
_ => break
|
|
}
|
|
i += 1;
|
|
}
|
|
if i == self.depth { // Made it to end of loop
|
|
let dnum = def.def_id().node;
|
|
if !self.seen.contains(&dnum) {
|
|
self.refs.push(freevar_entry {
|
|
def: def,
|
|
span: expr.span,
|
|
});
|
|
self.seen.insert(dnum);
|
|
}
|
|
}
|
|
}
|
|
_ => visit::walk_expr(self, expr)
|
|
}
|
|
}
|
|
}
|
|
|
|
// 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::Block,
|
|
capture_mode_map: &mut CaptureModeMap)
|
|
-> Vec<freevar_entry> {
|
|
let mut v = CollectFreevarsVisitor {
|
|
seen: NodeSet::new(),
|
|
refs: Vec::new(),
|
|
def_map: def_map,
|
|
capture_mode_map: &mut *capture_mode_map,
|
|
depth: 1
|
|
};
|
|
|
|
v.visit_block(blk);
|
|
|
|
v.refs
|
|
}
|
|
|
|
struct AnnotateFreevarsVisitor<'a> {
|
|
def_map: &'a resolve::DefMap,
|
|
freevars: freevar_map,
|
|
capture_mode_map: CaptureModeMap,
|
|
}
|
|
|
|
impl<'a, 'v> Visitor<'v> for AnnotateFreevarsVisitor<'a> {
|
|
fn visit_fn(&mut self, fk: visit::FnKind<'v>, fd: &'v ast::FnDecl,
|
|
blk: &'v ast::Block, s: Span, nid: ast::NodeId) {
|
|
let vars = collect_freevars(self.def_map,
|
|
blk,
|
|
&mut self.capture_mode_map);
|
|
self.freevars.insert(nid, vars);
|
|
visit::walk_fn(self, fk, fd, blk, s);
|
|
}
|
|
}
|
|
|
|
// 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, krate: &ast::Crate)
|
|
-> (freevar_map, CaptureModeMap) {
|
|
let mut visitor = AnnotateFreevarsVisitor {
|
|
def_map: def_map,
|
|
freevars: NodeMap::new(),
|
|
capture_mode_map: NodeMap::new(),
|
|
};
|
|
visit::walk_crate(&mut visitor, krate);
|
|
|
|
let AnnotateFreevarsVisitor {
|
|
freevars,
|
|
capture_mode_map,
|
|
..
|
|
} = visitor;
|
|
(freevars, capture_mode_map)
|
|
}
|
|
|
|
pub fn with_freevars<T>(tcx: &ty::ctxt, fid: ast::NodeId, f: |&[freevar_entry]| -> T) -> T {
|
|
match tcx.freevars.borrow().find(&fid) {
|
|
None => fail!("with_freevars: {} has no freevars", fid),
|
|
Some(d) => f(d.as_slice())
|
|
}
|
|
}
|
|
|
|
pub fn get_capture_mode<'tcx, T:Typer<'tcx>>(tcx: &T, closure_expr_id: ast::NodeId)
|
|
-> CaptureMode {
|
|
tcx.capture_mode(closure_expr_id)
|
|
}
|