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.
|
|
|
|
|
|
|
|
import syntax::ast::*;
|
2012-05-21 23:34:34 -07:00
|
|
|
import syntax::{visit, ast_util, ast_map};
|
2012-03-06 12:52:13 +01:00
|
|
|
import syntax::ast_util::def_id_of_def;
|
2012-03-22 18:16:22 -07:00
|
|
|
import syntax::attr;
|
2012-06-12 16:25:09 -07:00
|
|
|
import syntax::print::pprust::expr_to_str;
|
2012-03-07 16:48:57 -08:00
|
|
|
import std::map::hashmap;
|
2012-06-12 16:25:09 -07:00
|
|
|
import driver::session::*;
|
2012-03-06 12:52:13 +01:00
|
|
|
|
|
|
|
export map, find_reachable;
|
|
|
|
|
2012-03-07 16:48:57 -08:00
|
|
|
type map = std::map::hashmap<node_id, ()>;
|
2012-03-06 12:52:13 +01:00
|
|
|
|
2012-07-17 15:23:59 -07:00
|
|
|
type ctx = {exp_map: resolve3::ExportMap,
|
2012-03-20 12:28:46 +01:00
|
|
|
tcx: ty::ctxt,
|
|
|
|
method_map: typeck::method_map,
|
2012-03-06 12:52:13 +01:00
|
|
|
rmap: map};
|
|
|
|
|
2012-07-17 15:23:59 -07:00
|
|
|
fn find_reachable(crate_mod: _mod, exp_map: resolve3::ExportMap,
|
2012-03-20 12:28:46 +01:00
|
|
|
tcx: ty::ctxt, method_map: typeck::method_map) -> map {
|
2012-03-14 12:07:23 -07:00
|
|
|
let rmap = std::map::int_hash();
|
2012-03-20 12:28:46 +01:00
|
|
|
let cx = {exp_map: exp_map, tcx: tcx, method_map: method_map, rmap: rmap};
|
2012-03-08 21:16:04 +01:00
|
|
|
traverse_public_mod(cx, crate_mod);
|
2012-07-11 18:38:12 -07:00
|
|
|
traverse_all_resources_and_impls(cx, crate_mod);
|
2012-03-06 12:52:13 +01:00
|
|
|
rmap
|
|
|
|
}
|
|
|
|
|
2012-06-29 16:26:56 -07:00
|
|
|
fn traverse_exports(cx: ctx, vis: ~[@view_item]) -> bool {
|
2012-03-15 09:47:03 -04:00
|
|
|
let mut found_export = false;
|
2012-06-30 16:19:07 -07:00
|
|
|
for vec::each(vis) |vi| {
|
2012-08-06 12:34:08 -07:00
|
|
|
match vi.node {
|
2012-08-03 19:59:04 -07:00
|
|
|
view_item_export(vps) => {
|
2012-03-06 12:52:13 +01:00
|
|
|
found_export = true;
|
2012-06-30 16:19:07 -07:00
|
|
|
for vec::each(vps) |vp| {
|
2012-08-06 12:34:08 -07:00
|
|
|
match vp.node {
|
2012-03-06 12:52:13 +01:00
|
|
|
view_path_simple(_, _, id) | view_path_glob(_, id) |
|
2012-08-03 19:59:04 -07:00
|
|
|
view_path_list(_, _, id) => {
|
2012-03-06 12:52:13 +01:00
|
|
|
traverse_export(cx, id);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2012-08-03 19:59:04 -07:00
|
|
|
_ => ()
|
2012-03-06 12:52:13 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
found_export
|
|
|
|
}
|
|
|
|
|
|
|
|
fn traverse_export(cx: ctx, exp_id: node_id) {
|
2012-06-30 16:19:07 -07:00
|
|
|
do option::iter(cx.exp_map.find(exp_id)) |defs| {
|
|
|
|
for vec::each(defs) |def| { traverse_def_id(cx, def.id); }
|
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; }
|
2012-08-06 12:34:08 -07:00
|
|
|
let n = match cx.tcx.items.find(did.node) {
|
2012-08-03 19:59:04 -07:00
|
|
|
none => return, // This can happen for self, for example
|
|
|
|
some(n) => n
|
2012-06-12 16:25:09 -07:00
|
|
|
};
|
2012-08-06 12:34:08 -07:00
|
|
|
match n {
|
2012-08-03 19:59:04 -07:00
|
|
|
ast_map::node_item(item, _) => traverse_public_item(cx, item),
|
|
|
|
ast_map::node_method(_, impl_id, _) => traverse_def_id(cx, impl_id),
|
|
|
|
ast_map::node_foreign_item(item, _, _) => {
|
|
|
|
cx.rmap.insert(item.id, ());
|
|
|
|
}
|
|
|
|
ast_map::node_variant(v, _, _) => { cx.rmap.insert(v.node.id, ()); }
|
2012-06-24 15:09:57 -07:00
|
|
|
// If it's a ctor, consider the parent reachable
|
2012-08-03 19:59:04 -07:00
|
|
|
ast_map::node_ctor(_, _, _, parent_id, _) => {
|
2012-06-05 21:17:16 -07:00
|
|
|
traverse_def_id(cx, parent_id);
|
|
|
|
}
|
2012-08-03 19:59:04 -07:00
|
|
|
_ => ()
|
2012-03-06 12:52:13 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn traverse_public_mod(cx: ctx, m: _mod) {
|
|
|
|
if !traverse_exports(cx, m.view_items) {
|
|
|
|
// No exports, so every local item is exported
|
2012-06-30 16:19:07 -07:00
|
|
|
for vec::each(m.items) |item| { traverse_public_item(cx, item); }
|
2012-03-06 12:52:13 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn traverse_public_item(cx: ctx, item: @item) {
|
2012-08-01 17:30:05 -07:00
|
|
|
if cx.rmap.contains_key(item.id) { return; }
|
2012-03-06 12:52:13 +01:00
|
|
|
cx.rmap.insert(item.id, ());
|
2012-08-06 12:34:08 -07:00
|
|
|
match item.node {
|
2012-08-03 19:59:04 -07:00
|
|
|
item_mod(m) => traverse_public_mod(cx, m),
|
|
|
|
item_foreign_mod(nm) => {
|
2012-03-06 12:52:13 +01:00
|
|
|
if !traverse_exports(cx, nm.view_items) {
|
2012-06-30 16:19:07 -07:00
|
|
|
for vec::each(nm.items) |item| { cx.rmap.insert(item.id, ()); }
|
2012-03-06 12:52:13 +01:00
|
|
|
}
|
|
|
|
}
|
2012-08-23 18:17:16 -07:00
|
|
|
item_fn(_, _, tps, blk) => {
|
2012-03-06 12:52:13 +01:00
|
|
|
if tps.len() > 0u ||
|
|
|
|
attr::find_inline_attr(item.attrs) != attr::ia_none {
|
|
|
|
traverse_inline_body(cx, blk);
|
|
|
|
}
|
|
|
|
}
|
2012-08-03 19:59:04 -07:00
|
|
|
item_impl(tps, _, _, ms) => {
|
2012-06-30 16:19:07 -07:00
|
|
|
for vec::each(ms) |m| {
|
2012-03-06 12:52:13 +01:00
|
|
|
if tps.len() > 0u || m.tps.len() > 0u ||
|
|
|
|
attr::find_inline_attr(m.attrs) != attr::ia_none {
|
2012-05-29 14:39:22 -07:00
|
|
|
cx.rmap.insert(m.id, ());
|
2012-03-06 12:52:13 +01:00
|
|
|
traverse_inline_body(cx, m.body);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2012-08-07 15:34:07 -07:00
|
|
|
item_class(struct_def, tps) => {
|
|
|
|
do option::iter(struct_def.ctor) |ctor| {
|
2012-07-24 15:29:14 -07:00
|
|
|
cx.rmap.insert(ctor.node.id, ());
|
|
|
|
if tps.len() > 0u || attr::find_inline_attr(ctor.node.attrs)
|
|
|
|
!= attr::ia_none {
|
|
|
|
traverse_inline_body(cx, ctor.node.body);
|
|
|
|
}
|
2012-07-16 19:16:19 -07:00
|
|
|
}
|
2012-08-07 15:34:07 -07:00
|
|
|
do option::iter(struct_def.dtor) |dtor| {
|
2012-05-14 14:13:32 -07:00
|
|
|
cx.rmap.insert(dtor.node.id, ());
|
2012-07-16 19:16:19 -07:00
|
|
|
if tps.len() > 0u || attr::find_inline_attr(dtor.node.attrs)
|
|
|
|
!= attr::ia_none {
|
2012-06-12 16:25:09 -07:00
|
|
|
traverse_inline_body(cx, dtor.node.body);
|
|
|
|
}
|
2012-05-14 14:13:32 -07:00
|
|
|
}
|
2012-08-15 15:53:58 -07:00
|
|
|
for vec::each(struct_def.methods) |m| {
|
|
|
|
cx.rmap.insert(m.id, ());
|
|
|
|
if tps.len() > 0 ||
|
|
|
|
attr::find_inline_attr(m.attrs) != attr::ia_none {
|
|
|
|
traverse_inline_body(cx, m.body);
|
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(*) => (),
|
|
|
|
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> {
|
|
|
|
visit::mk_vt(@{visit_ty: traverse_ty with *visit::default_visitor()})
|
|
|
|
}
|
|
|
|
|
|
|
|
fn traverse_ty(ty: @ty, cx: ctx, v: visit::vt<ctx>) {
|
2012-08-01 17:30:05 -07:00
|
|
|
if cx.rmap.contains_key(ty.id) { return; }
|
2012-06-13 11:20:21 -07:00
|
|
|
cx.rmap.insert(ty.id, ());
|
|
|
|
|
2012-08-06 12:34:08 -07:00
|
|
|
match ty.node {
|
2012-08-03 19:59:04 -07:00
|
|
|
ty_path(p, p_id) => {
|
2012-08-06 12:34:08 -07: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
|
2012-08-03 19:59:04 -07:00
|
|
|
some(def_prim_ty(_)) => { /* do nothing */ }
|
|
|
|
some(d) => traverse_def_id(cx, def_id_of_def(d)),
|
|
|
|
none => { /* do nothing -- but should we fail here? */ }
|
2012-06-13 11:20:21 -07:00
|
|
|
}
|
2012-06-30 16:19:07 -07:00
|
|
|
for p.types.each |t| { v.visit_ty(t, cx, v); };
|
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
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-03-06 12:52:13 +01:00
|
|
|
fn traverse_inline_body(cx: ctx, body: blk) {
|
|
|
|
fn traverse_expr(e: @expr, cx: ctx, v: visit::vt<ctx>) {
|
2012-08-06 12:34:08 -07:00
|
|
|
match e.node {
|
2012-08-03 19:59:04 -07:00
|
|
|
expr_path(_) => {
|
2012-08-06 12:34:08 -07:00
|
|
|
match cx.tcx.def_map.find(e.id) {
|
2012-08-03 19:59:04 -07:00
|
|
|
some(d) => {
|
2012-06-12 16:25:09 -07:00
|
|
|
traverse_def_id(cx, def_id_of_def(d));
|
|
|
|
}
|
2012-08-22 17:24:52 -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(_, _, _) => {
|
2012-08-06 12:34:08 -07:00
|
|
|
match cx.method_map.find(e.id) {
|
2012-08-03 19:59:04 -07:00
|
|
|
some({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-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);
|
|
|
|
}
|
|
|
|
visit::visit_block(body, cx, visit::mk_vt(@{
|
2012-03-06 12:52:13 +01:00
|
|
|
visit_expr: traverse_expr,
|
|
|
|
visit_item: traverse_item
|
|
|
|
with *visit::default_visitor()
|
|
|
|
}));
|
|
|
|
}
|
2012-03-08 21:16:04 +01:00
|
|
|
|
2012-07-11 18:38:12 -07:00
|
|
|
fn traverse_all_resources_and_impls(cx: ctx, crate_mod: _mod) {
|
2012-03-08 21:16:04 +01:00
|
|
|
visit::visit_mod(crate_mod, ast_util::dummy_sp(), 0, cx, visit::mk_vt(@{
|
2012-06-30 16:19:07 -07:00
|
|
|
visit_expr: |_e, _cx, _v| { },
|
|
|
|
visit_item: |i, cx, v| {
|
2012-03-08 21:16:04 +01:00
|
|
|
visit::visit_item(i, cx, v);
|
2012-08-06 12:34:08 -07:00
|
|
|
match i.node {
|
2012-08-07 15:34:07 -07:00
|
|
|
item_class(struct_def, _) if struct_def.dtor.is_some() => {
|
2012-04-18 21:26:25 -07:00
|
|
|
traverse_public_item(cx, i);
|
|
|
|
}
|
2012-08-03 19:59:04 -07:00
|
|
|
item_impl(*) => {
|
2012-07-11 18:38:12 -07:00
|
|
|
traverse_public_item(cx, i);
|
|
|
|
}
|
2012-08-03 19:59:04 -07:00
|
|
|
_ => ()
|
2012-03-08 21:16:04 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
with *visit::default_visitor()
|
|
|
|
}));
|
|
|
|
}
|
2012-07-11 18:38:12 -07:00
|
|
|
|