2012-12-03 16:48:01 -08:00
|
|
|
// 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.
|
|
|
|
|
2012-03-06 12:52:13 +01:00
|
|
|
// Finds items that are externally reachable, to determine which items
|
|
|
|
// need to have their metadata (and possibly their AST) serialized.
|
|
|
|
// All items that can be referred to through an exported name are
|
|
|
|
// reachable, and when a reachable thing is inline or generic, it
|
|
|
|
// makes all other generics or inline functions that it references
|
|
|
|
// reachable as well.
|
|
|
|
|
2013-01-07 14:16:52 -08:00
|
|
|
|
2012-12-23 17:41:37 -05:00
|
|
|
use middle::resolve;
|
|
|
|
use middle::ty;
|
|
|
|
use middle::typeck;
|
|
|
|
|
2013-02-25 14:11:21 -05:00
|
|
|
use core::prelude::*;
|
2013-04-03 09:28:36 -04:00
|
|
|
use core::hashmap::HashSet;
|
2013-02-25 14:11:21 -05:00
|
|
|
use syntax::ast;
|
2012-09-04 11:54:36 -07:00
|
|
|
use syntax::ast::*;
|
|
|
|
use syntax::ast_util::def_id_of_def;
|
|
|
|
use syntax::attr;
|
2013-02-25 14:11:21 -05:00
|
|
|
use syntax::codemap;
|
2012-09-04 11:54:36 -07:00
|
|
|
use syntax::print::pprust::expr_to_str;
|
2013-03-26 16:38:07 -04:00
|
|
|
use syntax::{visit, ast_map};
|
2012-03-06 12:52:13 +01:00
|
|
|
|
2013-04-03 09:28:36 -04:00
|
|
|
pub type map = @HashSet<node_id>;
|
2012-03-06 12:52:13 +01:00
|
|
|
|
2013-03-26 20:52:16 -07:00
|
|
|
struct ctx<'self> {
|
2013-01-06 11:16:14 -08:00
|
|
|
exp_map2: resolve::ExportMap2,
|
|
|
|
tcx: ty::ctxt,
|
|
|
|
method_map: typeck::method_map,
|
2013-04-03 09:28:36 -04:00
|
|
|
rmap: &'self mut HashSet<node_id>,
|
2013-01-06 11:16:14 -08:00
|
|
|
}
|
2012-03-06 12:52:13 +01:00
|
|
|
|
2013-02-18 21:25:44 -08:00
|
|
|
pub fn find_reachable(crate_mod: &_mod, exp_map2: resolve::ExportMap2,
|
2013-01-30 11:46:19 -08:00
|
|
|
tcx: ty::ctxt, method_map: typeck::method_map) -> map {
|
2013-04-03 09:28:36 -04:00
|
|
|
let mut rmap = HashSet::new();
|
2013-03-22 22:26:41 -04:00
|
|
|
{
|
|
|
|
let cx = ctx {
|
|
|
|
exp_map2: exp_map2,
|
|
|
|
tcx: tcx,
|
|
|
|
method_map: method_map,
|
|
|
|
rmap: &mut rmap
|
|
|
|
};
|
|
|
|
traverse_public_mod(cx, ast::crate_node_id, crate_mod);
|
|
|
|
traverse_all_resources_and_impls(cx, crate_mod);
|
|
|
|
}
|
|
|
|
return @rmap;
|
2012-03-06 12:52:13 +01:00
|
|
|
}
|
|
|
|
|
2012-09-20 13:08:45 -07:00
|
|
|
fn traverse_exports(cx: ctx, mod_id: node_id) -> bool {
|
2012-03-15 09:47:03 -04:00
|
|
|
let mut found_export = false;
|
2013-02-05 19:41:45 -08:00
|
|
|
match cx.exp_map2.find(&mod_id) {
|
2012-12-04 10:50:00 -08:00
|
|
|
Some(ref exp2s) => {
|
|
|
|
for (*exp2s).each |e2| {
|
2012-03-06 12:52:13 +01:00
|
|
|
found_export = true;
|
2012-09-20 13:08:45 -07:00
|
|
|
traverse_def_id(cx, e2.def_id)
|
|
|
|
};
|
|
|
|
}
|
|
|
|
None => ()
|
2012-03-06 12:52:13 +01:00
|
|
|
}
|
2012-09-20 13:08:45 -07:00
|
|
|
return found_export;
|
2012-03-06 12:52:13 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
fn traverse_def_id(cx: ctx, did: def_id) {
|
2012-08-01 17:30:05 -07:00
|
|
|
if did.crate != local_crate { return; }
|
2013-03-23 21:45:27 -04:00
|
|
|
match cx.tcx.items.find(&did.node) {
|
|
|
|
None => (), // This can happen for self, for example
|
|
|
|
Some(&ast_map::node_item(item, _)) => traverse_public_item(cx, item),
|
|
|
|
Some(&ast_map::node_method(_, impl_id, _)) => traverse_def_id(cx, impl_id),
|
|
|
|
Some(&ast_map::node_foreign_item(item, _, _, _)) => {
|
|
|
|
cx.rmap.insert(item.id);
|
|
|
|
}
|
|
|
|
Some(&ast_map::node_variant(ref v, _, _)) => {
|
|
|
|
cx.rmap.insert(v.node.id);
|
|
|
|
}
|
|
|
|
_ => ()
|
2012-03-06 12:52:13 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-02-18 21:25:44 -08:00
|
|
|
fn traverse_public_mod(cx: ctx, mod_id: node_id, m: &_mod) {
|
2012-09-20 13:08:45 -07:00
|
|
|
if !traverse_exports(cx, mod_id) {
|
2012-03-06 12:52:13 +01:00
|
|
|
// No exports, so every local item is exported
|
2013-02-18 21:25:44 -08:00
|
|
|
for m.items.each |item| {
|
2012-09-18 21:41:37 -07:00
|
|
|
traverse_public_item(cx, *item);
|
|
|
|
}
|
2012-03-06 12:52:13 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn traverse_public_item(cx: ctx, item: @item) {
|
2013-03-22 22:26:41 -04:00
|
|
|
// XXX: it shouldn't be necessary to do this
|
2013-04-03 09:28:36 -04:00
|
|
|
let rmap: &mut HashSet<node_id> = cx.rmap;
|
2013-03-22 22:26:41 -04:00
|
|
|
if rmap.contains(&item.id) { return; }
|
|
|
|
rmap.insert(item.id);
|
2013-03-20 01:17:42 -04:00
|
|
|
match item.node {
|
2013-02-18 21:25:44 -08:00
|
|
|
item_mod(ref m) => traverse_public_mod(cx, item.id, m),
|
2013-02-14 21:50:03 -08:00
|
|
|
item_foreign_mod(ref nm) => {
|
2012-09-20 13:08:45 -07:00
|
|
|
if !traverse_exports(cx, item.id) {
|
2013-02-18 21:25:44 -08:00
|
|
|
for nm.items.each |item| {
|
2013-03-22 22:26:41 -04:00
|
|
|
cx.rmap.insert(item.id);
|
2012-09-18 21:41:37 -07:00
|
|
|
}
|
2012-03-06 12:52:13 +01:00
|
|
|
}
|
|
|
|
}
|
2013-03-13 22:25:28 -04:00
|
|
|
item_fn(_, _, _, ref generics, ref blk) => {
|
2013-02-14 21:50:03 -08:00
|
|
|
if generics.ty_params.len() > 0u ||
|
2012-03-06 12:52:13 +01:00
|
|
|
attr::find_inline_attr(item.attrs) != attr::ia_none {
|
2013-02-17 22:20:36 -08:00
|
|
|
traverse_inline_body(cx, blk);
|
2012-03-06 12:52:13 +01:00
|
|
|
}
|
|
|
|
}
|
2013-02-14 21:50:03 -08:00
|
|
|
item_impl(ref generics, _, _, ref ms) => {
|
|
|
|
for ms.each |m| {
|
|
|
|
if generics.ty_params.len() > 0u ||
|
|
|
|
m.generics.ty_params.len() > 0u ||
|
|
|
|
attr::find_inline_attr(m.attrs) != attr::ia_none
|
|
|
|
{
|
2013-03-22 22:26:41 -04:00
|
|
|
cx.rmap.insert(m.id);
|
2013-02-17 22:20:36 -08:00
|
|
|
traverse_inline_body(cx, &m.body);
|
2012-03-06 12:52:13 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2013-02-14 21:50:03 -08:00
|
|
|
item_struct(ref struct_def, ref generics) => {
|
2013-02-11 16:28:39 -08:00
|
|
|
for struct_def.ctor_id.each |&ctor_id| {
|
2013-03-22 22:26:41 -04:00
|
|
|
cx.rmap.insert(ctor_id);
|
2013-02-11 16:28:39 -08:00
|
|
|
}
|
2013-03-03 07:33:39 -05:00
|
|
|
for struct_def.dtor.each |dtor| {
|
2013-03-22 22:26:41 -04:00
|
|
|
cx.rmap.insert(dtor.node.id);
|
2013-02-14 21:50:03 -08:00
|
|
|
if generics.ty_params.len() > 0u ||
|
|
|
|
attr::find_inline_attr(dtor.node.attrs) != attr::ia_none
|
|
|
|
{
|
2013-02-17 22:20:36 -08:00
|
|
|
traverse_inline_body(cx, &dtor.node.body);
|
2012-06-12 16:25:09 -07:00
|
|
|
}
|
2012-05-14 14:13:32 -07:00
|
|
|
}
|
2012-03-20 13:19:33 +01:00
|
|
|
}
|
2012-08-03 19:59:04 -07:00
|
|
|
item_ty(t, _) => {
|
2012-06-13 11:20:21 -07:00
|
|
|
traverse_ty(t, cx, mk_ty_visitor());
|
|
|
|
}
|
|
|
|
item_const(*) |
|
2012-08-03 19:59:04 -07:00
|
|
|
item_enum(*) | item_trait(*) => (),
|
2013-02-11 19:26:38 -08:00
|
|
|
item_mac(*) => fail!(~"item macros unimplemented")
|
2012-03-06 12:52:13 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-06-13 11:20:21 -07:00
|
|
|
fn mk_ty_visitor() -> visit::vt<ctx> {
|
2013-01-08 14:00:45 -08:00
|
|
|
visit::mk_vt(@visit::Visitor {visit_ty: traverse_ty,
|
|
|
|
..*visit::default_visitor()})
|
2012-06-13 11:20:21 -07:00
|
|
|
}
|
|
|
|
|
2013-03-26 20:52:16 -07:00
|
|
|
fn traverse_ty<'a>(ty: @Ty, cx: ctx<'a>, v: visit::vt<ctx<'a>>) {
|
2013-03-22 22:26:41 -04:00
|
|
|
// XXX: it shouldn't be necessary to do this
|
2013-04-03 09:28:36 -04:00
|
|
|
let rmap: &mut HashSet<node_id> = cx.rmap;
|
2013-03-22 22:26:41 -04:00
|
|
|
if rmap.contains(&ty.id) { return; }
|
|
|
|
rmap.insert(ty.id);
|
2012-06-13 11:20:21 -07:00
|
|
|
|
2012-08-06 12:34:08 -07:00
|
|
|
match ty.node {
|
2012-08-03 19:59:04 -07:00
|
|
|
ty_path(p, p_id) => {
|
2013-02-05 19:41:45 -08:00
|
|
|
match cx.tcx.def_map.find(&p_id) {
|
2012-06-13 11:20:21 -07:00
|
|
|
// Kind of a hack to check this here, but I'm not sure what else
|
|
|
|
// to do
|
2013-03-22 22:26:41 -04:00
|
|
|
Some(&def_prim_ty(_)) => { /* do nothing */ }
|
|
|
|
Some(&d) => traverse_def_id(cx, def_id_of_def(d)),
|
2012-08-20 12:23:37 -07:00
|
|
|
None => { /* do nothing -- but should we fail here? */ }
|
2012-06-13 11:20:21 -07:00
|
|
|
}
|
2012-09-19 16:55:01 -07:00
|
|
|
for p.types.each |t| {
|
2012-11-29 17:51:16 -08:00
|
|
|
(v.visit_ty)(*t, cx, v);
|
2012-09-19 16:55:01 -07:00
|
|
|
}
|
2012-06-13 11:20:21 -07:00
|
|
|
}
|
2012-08-03 19:59:04 -07:00
|
|
|
_ => visit::visit_ty(ty, cx, v)
|
2012-06-13 11:20:21 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-02-17 22:20:36 -08:00
|
|
|
fn traverse_inline_body(cx: ctx, body: &blk) {
|
2013-03-26 20:52:16 -07:00
|
|
|
fn traverse_expr<'a>(e: @expr, cx: ctx<'a>, v: visit::vt<ctx<'a>>) {
|
2012-08-06 12:34:08 -07:00
|
|
|
match e.node {
|
2012-08-03 19:59:04 -07:00
|
|
|
expr_path(_) => {
|
2013-02-05 19:41:45 -08:00
|
|
|
match cx.tcx.def_map.find(&e.id) {
|
2013-03-22 22:26:41 -04:00
|
|
|
Some(&d) => {
|
2012-06-12 16:25:09 -07:00
|
|
|
traverse_def_id(cx, def_id_of_def(d));
|
|
|
|
}
|
2012-08-20 12:23:37 -07:00
|
|
|
None => cx.tcx.sess.span_bug(e.span, fmt!("Unbound node \
|
2012-07-18 16:18:02 -07:00
|
|
|
id %? while traversing %s", e.id,
|
2012-08-22 17:24:52 -07:00
|
|
|
expr_to_str(e, cx.tcx.sess.intr())))
|
2012-06-12 16:25:09 -07:00
|
|
|
}
|
2012-03-06 12:52:13 +01:00
|
|
|
}
|
2012-08-03 19:59:04 -07:00
|
|
|
expr_field(_, _, _) => {
|
2013-02-05 19:41:45 -08:00
|
|
|
match cx.method_map.find(&e.id) {
|
2013-03-23 18:55:58 -04:00
|
|
|
Some(&typeck::method_map_entry {
|
2013-01-16 19:24:10 -08:00
|
|
|
origin: typeck::method_static(did),
|
|
|
|
_
|
|
|
|
}) => {
|
2012-06-07 10:51:21 -07:00
|
|
|
traverse_def_id(cx, did);
|
|
|
|
}
|
2012-08-03 19:59:04 -07:00
|
|
|
_ => ()
|
2012-03-06 12:52:13 +01:00
|
|
|
}
|
|
|
|
}
|
2012-11-30 11:18:25 -08:00
|
|
|
expr_method_call(*) => {
|
2013-02-05 19:41:45 -08:00
|
|
|
match cx.method_map.find(&e.id) {
|
2013-03-23 18:55:58 -04:00
|
|
|
Some(&typeck::method_map_entry {
|
2013-01-16 19:24:10 -08:00
|
|
|
origin: typeck::method_static(did),
|
|
|
|
_
|
|
|
|
}) => {
|
2012-11-30 11:18:25 -08:00
|
|
|
traverse_def_id(cx, did);
|
|
|
|
}
|
|
|
|
Some(_) => {}
|
|
|
|
None => {
|
|
|
|
cx.tcx.sess.span_bug(e.span, ~"expr_method_call not in \
|
|
|
|
method map");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2012-08-03 19:59:04 -07:00
|
|
|
_ => ()
|
2012-03-06 12:52:13 +01:00
|
|
|
}
|
|
|
|
visit::visit_expr(e, cx, v);
|
|
|
|
}
|
2012-05-29 14:39:22 -07:00
|
|
|
// Don't ignore nested items: for example if a generic fn contains a
|
|
|
|
// generic impl (as in deque::create), we need to monomorphize the
|
|
|
|
// impl as well
|
|
|
|
fn traverse_item(i: @item, cx: ctx, _v: visit::vt<ctx>) {
|
|
|
|
traverse_public_item(cx, i);
|
|
|
|
}
|
2013-02-17 22:20:36 -08:00
|
|
|
visit::visit_block(body, cx, visit::mk_vt(@visit::Visitor {
|
2012-03-06 12:52:13 +01:00
|
|
|
visit_expr: traverse_expr,
|
2012-09-04 13:29:32 -07:00
|
|
|
visit_item: traverse_item,
|
2012-08-28 15:54:45 -07:00
|
|
|
..*visit::default_visitor()
|
2012-03-06 12:52:13 +01:00
|
|
|
}));
|
|
|
|
}
|
2012-03-08 21:16:04 +01:00
|
|
|
|
2013-02-18 21:25:44 -08:00
|
|
|
fn traverse_all_resources_and_impls(cx: ctx, crate_mod: &_mod) {
|
2013-01-08 14:00:45 -08:00
|
|
|
visit::visit_mod(
|
2013-02-18 21:25:44 -08:00
|
|
|
crate_mod,
|
|
|
|
codemap::dummy_sp(),
|
|
|
|
0,
|
|
|
|
cx,
|
2013-01-08 14:00:45 -08:00
|
|
|
visit::mk_vt(@visit::Visitor {
|
|
|
|
visit_expr: |_e, _cx, _v| { },
|
|
|
|
visit_item: |i, cx, v| {
|
|
|
|
visit::visit_item(i, cx, v);
|
|
|
|
match i.node {
|
|
|
|
item_struct(sdef, _) if sdef.dtor.is_some() => {
|
|
|
|
traverse_public_item(cx, i);
|
|
|
|
}
|
|
|
|
item_impl(*) => {
|
|
|
|
traverse_public_item(cx, i);
|
|
|
|
}
|
|
|
|
_ => ()
|
|
|
|
}
|
|
|
|
},
|
|
|
|
..*visit::default_visitor()
|
|
|
|
}));
|
2012-03-08 21:16:04 +01:00
|
|
|
}
|
2012-07-11 18:38:12 -07:00
|
|
|
|