rust/src/libsyntax/ast_map.rs

406 lines
12 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.
use abi::AbiSet;
2012-09-04 13:37:29 -05:00
use ast::*;
use ast;
2013-03-26 15:38:07 -05:00
use ast_util::{inlined_item_utils, stmt_id};
use ast_util;
use codemap;
2012-09-04 13:37:29 -05:00
use diagnostic::span_handler;
use parse::token::ident_interner;
use print::pprust;
use visit;
2013-03-15 14:24:24 -05:00
use syntax::parse::token::special_idents;
use core::hashmap::HashMap;
pub enum path_elt {
2012-08-27 18:26:35 -05:00
path_mod(ident),
path_name(ident)
}
impl cmp::Eq for path_elt {
fn eq(&self, other: &path_elt) -> bool {
match (*self) {
path_mod(e0a) => {
match (*other) {
path_mod(e0b) => e0a == e0b,
_ => false
}
}
path_name(e0a) => {
match (*other) {
path_name(e0b) => e0a == e0b,
_ => false
}
}
}
}
fn ne(&self, other: &path_elt) -> bool { !(*self).eq(other) }
}
2012-08-27 18:26:35 -05:00
pub type path = ~[path_elt];
pub fn path_to_str_with_sep(p: &[path_elt], sep: ~str, itr: @ident_interner)
-> ~str {
let strs = do p.map |e| {
match *e {
2013-02-24 20:32:02 -06:00
path_mod(s) => copy *itr.get(s),
path_name(s) => copy *itr.get(s)
}
};
str::connect(strs, sep)
}
pub fn path_ident_to_str(p: path, i: ident, itr: @ident_interner) -> ~str {
if vec::is_empty(p) {
2012-07-18 18:18:02 -05:00
//FIXME /* FIXME (#2543) */ copy *i
2013-02-24 20:32:02 -06:00
copy *itr.get(i)
} else {
2012-08-22 19:24:52 -05:00
fmt!("%s::%s", path_to_str(p, itr), *itr.get(i))
}
}
pub fn path_to_str(p: &[path_elt], itr: @ident_interner) -> ~str {
2012-07-18 18:18:02 -05:00
path_to_str_with_sep(p, ~"::", itr)
}
pub fn path_elt_to_str(pe: path_elt, itr: @ident_interner) -> ~str {
match pe {
2013-02-24 20:32:02 -06:00
path_mod(s) => copy *itr.get(s),
path_name(s) => copy *itr.get(s)
}
}
pub enum ast_node {
node_item(@item, @path),
node_foreign_item(@foreign_item, AbiSet, visibility, @path),
node_trait_method(@trait_method, def_id /* trait did */,
@path /* path to the trait */),
node_method(@method, def_id /* impl did */, @path /* path to the impl */),
node_variant(variant, @item, @path),
2012-01-19 19:56:05 -06:00
node_expr(@expr),
node_stmt(@stmt),
2013-03-15 14:24:24 -05:00
node_arg,
node_local(ident),
2012-03-15 20:46:18 -05:00
node_block(blk),
node_struct_ctor(@struct_def, @item, @path),
2013-03-15 14:24:24 -05:00
node_callee_scope(@expr)
}
pub type map = @mut HashMap<node_id, ast_node>;
pub struct Ctx {
2013-03-23 20:45:27 -05:00
map: map,
path: path,
diag: @span_handler,
}
pub type vt = visit::vt<@mut Ctx>;
pub fn extend(cx: @mut Ctx, elt: ident) -> @path {
@(vec::append(copy cx.path, ~[path_name(elt)]))
}
pub fn mk_ast_map_visitor() -> vt {
return visit::mk_vt(@visit::Visitor {
visit_item: map_item,
visit_expr: map_expr,
visit_stmt: map_stmt,
visit_fn: map_fn,
2012-09-04 15:29:32 -05:00
visit_block: map_block,
2013-03-15 14:24:24 -05:00
visit_pat: map_pat,
2012-09-04 15:29:32 -05:00
.. *visit::default_visitor()
});
}
pub fn map_crate(diag: @span_handler, c: @crate) -> map {
let cx = @mut Ctx {
map: @mut HashMap::new(),
path: ~[],
diag: diag,
};
visit::visit_crate(c, cx, mk_ast_map_visitor());
2013-03-23 20:45:27 -05:00
cx.map
}
// Used for items loaded from external crate that are being inlined into this
// crate. The `path` should be the path to the item but should not include
// the item itself.
pub fn map_decoded_item(diag: @span_handler,
map: map,
path: path,
ii: &inlined_item) {
// I believe it is ok for the local IDs of inlined items from other crates
// to overlap with the local ids from this crate, so just generate the ids
// starting from 0. (In particular, I think these ids are only used in
// alias analysis, which we will not be running on the inlined items, and
// even if we did I think it only needs an ordering between local
// variables that are simultaneously in scope).
let cx = @mut Ctx {
2013-03-23 20:45:27 -05:00
map: map,
path: copy path,
diag: diag,
};
let v = mk_ast_map_visitor();
// methods get added to the AST map when their impl is visited. Since we
// don't decode and instantiate the impl, but just the method, we have to
// add it to the table now:
match *ii {
ii_item(*) => { /* fallthrough */ }
2012-08-03 21:59:04 -05:00
ii_foreign(i) => {
cx.map.insert(i.id, node_foreign_item(i,
AbiSet::Intrinsic(),
i.vis, // Wrong but OK
@path));
}
2012-08-03 21:59:04 -05:00
ii_method(impl_did, m) => {
map_method(impl_did, @path, m, cx);
}
}
// visit the item / method contents and add those to the map:
ii.accept(cx, v);
}
pub fn map_fn(
fk: &visit::fn_kind,
decl: &fn_decl,
body: &blk,
sp: codemap::span,
id: node_id,
cx: @mut Ctx,
v: visit::vt<@mut Ctx>
) {
2012-06-30 18:19:07 -05:00
for decl.inputs.each |a| {
2013-03-15 14:24:24 -05:00
cx.map.insert(a.id, node_arg);
}
visit::visit_fn(fk, decl, body, sp, id, cx, v);
}
pub fn map_block(b: &blk, cx: @mut Ctx, v: visit::vt<@mut Ctx>) {
cx.map.insert(b.node.id, node_block(/* FIXME (#2543) */ copy *b));
2012-03-15 20:46:18 -05:00
visit::visit_block(b, cx, v);
}
2013-03-15 14:24:24 -05:00
pub fn map_pat(pat: @pat, cx: @mut Ctx, v: visit::vt<@mut Ctx>) {
match pat.node {
pat_ident(_, path, _) => {
// Note: this is at least *potentially* a pattern...
cx.map.insert(pat.id, node_local(ast_util::path_to_ident(path)));
}
2013-03-15 14:24:24 -05:00
_ => ()
}
2013-03-15 14:24:24 -05:00
visit::visit_pat(pat, cx, v);
}
pub fn map_method(impl_did: def_id, impl_path: @path,
m: @method, cx: @mut Ctx) {
cx.map.insert(m.id, node_method(m, impl_did, impl_path));
2013-03-15 14:24:24 -05:00
cx.map.insert(m.self_id, node_local(special_idents::self_));
}
pub fn map_item(i: @item, cx: @mut Ctx, v: visit::vt<@mut Ctx>) {
let item_path = @/* FIXME (#2543) */ copy cx.path;
cx.map.insert(i.id, node_item(i, item_path));
2012-08-06 14:34:08 -05:00
match i.node {
item_impl(_, _, _, ref ms) => {
let impl_did = ast_util::local_def(i.id);
for ms.each |m| {
map_method(impl_did, extend(cx, i.ident), *m, cx);
}
}
item_enum(ref enum_definition, _) => {
for (*enum_definition).variants.each |v| {
cx.map.insert(v.node.id, node_variant(
/* FIXME (#2543) */ copy *v, i,
extend(cx, i.ident)));
}
}
item_foreign_mod(ref nm) => {
for nm.items.each |nitem| {
// Compute the visibility for this native item.
let visibility = match nitem.vis {
public => public,
private => private,
inherited => i.vis
};
cx.map.insert(nitem.id,
node_foreign_item(
*nitem,
nm.abis,
visibility,
// FIXME (#2543)
if nm.sort == ast::named {
extend(cx, i.ident)
} else {
// Anonymous extern mods go in the parent scope
@copy cx.path
}
)
);
}
}
item_struct(struct_def, _) => {
map_struct_def(
struct_def,
node_item(i, item_path),
i.ident,
cx,
v
);
}
item_trait(_, ref traits, ref methods) => {
for traits.each |p| {
cx.map.insert(p.ref_id, node_item(i, item_path));
}
for methods.each |tm| {
let id = ast_util::trait_method_to_ty_method(tm).id;
let d_id = ast_util::local_def(i.id);
2013-02-24 20:32:02 -06:00
cx.map.insert(
id,
node_trait_method(@copy *tm, d_id, item_path)
);
}
}
_ => ()
}
2013-03-15 14:24:24 -05:00
2012-08-06 14:34:08 -05:00
match i.node {
item_mod(_) | item_foreign_mod(_) => {
cx.path.push(path_mod(i.ident));
}
_ => cx.path.push(path_name(i.ident))
}
visit::visit_item(i, cx, v);
2012-09-28 00:20:47 -05:00
cx.path.pop();
}
pub fn map_struct_def(
struct_def: @ast::struct_def,
parent_node: ast_node,
ident: ast::ident,
cx: @mut Ctx,
_v: visit::vt<@mut Ctx>
) {
let p = extend(cx, ident);
// If this is a tuple-like struct, register the constructor.
match struct_def.ctor_id {
None => {}
Some(ctor_id) => {
match parent_node {
node_item(item, _) => {
cx.map.insert(ctor_id,
node_struct_ctor(struct_def, item, p));
}
_ => fail!("struct def parent wasn't an item")
}
}
}
}
pub fn map_expr(ex: @expr, cx: @mut Ctx, v: visit::vt<@mut Ctx>) {
cx.map.insert(ex.id, node_expr(ex));
2013-03-15 14:24:24 -05:00
match ex.node {
// Expressions which are or might be calls:
ast::expr_call(*) |
ast::expr_method_call(*) |
ast::expr_index(*) |
ast::expr_binary(*) |
ast::expr_assign_op(*) |
ast::expr_unary(*) => {
cx.map.insert(ex.callee_id, node_callee_scope(ex));
}
_ => {}
}
visit::visit_expr(ex, cx, v);
}
pub fn map_stmt(stmt: @stmt, cx: @mut Ctx, v: visit::vt<@mut Ctx>) {
cx.map.insert(stmt_id(stmt), node_stmt(stmt));
visit::visit_stmt(stmt, cx, v);
}
pub fn node_id_to_str(map: map, id: node_id, itr: @ident_interner) -> ~str {
match map.find(&id) {
2012-08-20 14:23:37 -05:00
None => {
2012-08-22 19:24:52 -05:00
fmt!("unknown node (id=%d)", id)
}
2013-03-23 20:45:27 -05:00
Some(&node_item(item, path)) => {
let path_str = path_ident_to_str(*path, item.ident, itr);
let item_str = match item.node {
item_const(*) => ~"const",
item_fn(*) => ~"fn",
item_mod(*) => ~"mod",
item_foreign_mod(*) => ~"foreign mod",
item_ty(*) => ~"ty",
item_enum(*) => ~"enum",
item_struct(*) => ~"struct",
item_trait(*) => ~"trait",
item_impl(*) => ~"impl",
item_mac(*) => ~"macro"
};
fmt!("%s %s (id=%?)", item_str, path_str, id)
}
2013-03-23 20:45:27 -05:00
Some(&node_foreign_item(item, abi, _, path)) => {
2012-08-22 19:24:52 -05:00
fmt!("foreign item %s with abi %? (id=%?)",
path_ident_to_str(*path, item.ident, itr), abi, id)
}
2013-03-23 20:45:27 -05:00
Some(&node_method(m, _, path)) => {
2012-08-22 19:24:52 -05:00
fmt!("method %s in %s (id=%?)",
*itr.get(m.ident), path_to_str(*path, itr), id)
}
2013-03-23 20:45:27 -05:00
Some(&node_trait_method(ref tm, _, path)) => {
let m = ast_util::trait_method_to_ty_method(&**tm);
2012-08-22 19:24:52 -05:00
fmt!("method %s in %s (id=%?)",
*itr.get(m.ident), path_to_str(*path, itr), id)
}
2013-03-23 20:45:27 -05:00
Some(&node_variant(ref variant, _, path)) => {
2012-08-22 19:24:52 -05:00
fmt!("variant %s in %s (id=%?)",
*itr.get(variant.node.name), path_to_str(*path, itr), id)
}
2013-03-23 20:45:27 -05:00
Some(&node_expr(expr)) => {
2012-08-22 19:24:52 -05:00
fmt!("expr %s (id=%?)", pprust::expr_to_str(expr, itr), id)
}
2013-03-15 14:24:24 -05:00
Some(&node_callee_scope(expr)) => {
fmt!("callee_scope %s (id=%?)", pprust::expr_to_str(expr, itr), id)
}
2013-03-23 20:45:27 -05:00
Some(&node_stmt(stmt)) => {
2012-08-22 19:24:52 -05:00
fmt!("stmt %s (id=%?)",
pprust::stmt_to_str(stmt, itr), id)
}
2013-03-15 14:24:24 -05:00
Some(&node_arg) => {
2012-08-22 19:24:52 -05:00
fmt!("arg (id=%?)", id)
}
2013-03-15 14:24:24 -05:00
Some(&node_local(ident)) => {
fmt!("local (id=%?, name=%s)", id, *itr.get(ident))
}
2013-03-23 20:45:27 -05:00
Some(&node_block(_)) => {
2012-08-22 19:24:52 -05:00
fmt!("block")
}
2013-03-23 20:45:27 -05:00
Some(&node_struct_ctor(*)) => {
fmt!("struct_ctor")
}
}
}
pub fn node_item_query<Result>(items: map, id: node_id,
query: &fn(@item) -> Result,
error_msg: ~str) -> Result {
match items.find(&id) {
2013-03-23 20:45:27 -05:00
Some(&node_item(it, _)) => query(it),
_ => fail!(error_msg)
}
}