rust/src/rustc/metadata/cstore.rs

190 lines
6.0 KiB
Rust
Raw Normal View History

2011-07-07 20:39:44 -05:00
// The crate store - a central repo for information collected about external
// crates and libraries
import std::map;
import std::map::hashmap;
import syntax::{ast, attr};
2012-05-23 02:38:39 -05:00
import syntax::ast_util::new_def_hash;
2012-07-18 18:18:02 -05:00
import syntax::parse::token::ident_interner;
export cstore;
export cnum_map;
export crate_metadata;
export mk_cstore;
export get_crate_data;
export set_crate_data;
export get_crate_hash;
export get_crate_vers;
export have_crate_data;
export iter_crate_data;
export add_used_crate_file;
export get_used_crate_files;
export add_used_library;
export get_used_libraries;
export add_used_link_args;
export get_used_link_args;
export add_use_stmt_cnum;
export find_use_stmt_cnum;
export get_dep_hashes;
export get_path;
// A map from external crate numbers (as decoded from some crate file) to
// local crate numbers (as generated during this session). Each external
// crate may refer to types in other external crates, and each has their
// own crate numbers.
type cnum_map = map::hashmap<ast::crate_num, ast::crate_num>;
// Multiple items may have the same def_id in crate metadata. They may be
// renamed imports or reexports. This map keeps the "real" module path
// and def_id.
type mod_path_map = map::hashmap<ast::def_id, @~str>;
type crate_metadata = @{name: ~str,
data: @~[u8],
cnum_map: cnum_map,
cnum: ast::crate_num};
// This is a bit of an experiment at encapsulating the data in cstore. By
2012-01-19 16:24:03 -06:00
// keeping all the data in a non-exported enum variant, it's impossible for
// other modules to access the cstore's private data. This could also be
// achieved with an obj, but at the expense of a vtable. Not sure if this is a
// good pattern or not.
2012-01-19 19:56:05 -06:00
enum cstore { private(cstore_private), }
2011-07-27 07:19:39 -05:00
type cstore_private =
@{metas: map::hashmap<ast::crate_num, crate_metadata>,
2011-07-27 07:19:39 -05:00
use_crate_map: use_crate_map,
mod_path_map: mod_path_map,
mut used_crate_files: ~[Path],
mut used_libraries: ~[~str],
2012-07-18 18:18:02 -05:00
mut used_link_args: ~[~str],
intr: ident_interner};
2011-07-07 23:37:56 -05:00
// Map from node_id's of local use statements to crate numbers
type use_crate_map = map::hashmap<ast::node_id, ast::crate_num>;
2011-07-07 23:37:56 -05:00
// Internal method to retrieve the data from the cstore
2012-06-04 10:03:14 -05:00
pure fn p(cstore: cstore) -> cstore_private {
2012-08-06 14:34:08 -05:00
match cstore { private(p) => p }
2012-06-04 10:03:14 -05:00
}
2012-07-18 18:18:02 -05:00
fn mk_cstore(intr: ident_interner) -> cstore {
let meta_cache = map::int_hash::<crate_metadata>();
let crate_map = map::int_hash::<ast::crate_num>();
let mod_path_map = new_def_hash();
2012-08-01 19:30:05 -05:00
return private(@{metas: meta_cache,
2012-07-18 18:18:02 -05:00
use_crate_map: crate_map,
mod_path_map: mod_path_map,
mut used_crate_files: ~[],
mut used_libraries: ~[],
mut used_link_args: ~[],
intr: intr});
}
fn get_crate_data(cstore: cstore, cnum: ast::crate_num) -> crate_metadata {
2012-08-01 19:30:05 -05:00
return p(cstore).metas.get(cnum);
}
2012-07-18 18:18:02 -05:00
fn get_crate_hash(cstore: cstore, cnum: ast::crate_num) -> ~str {
let cdata = get_crate_data(cstore, cnum);
2012-08-01 19:30:05 -05:00
return decoder::get_crate_hash(cdata.data);
}
2012-07-18 18:18:02 -05:00
fn get_crate_vers(cstore: cstore, cnum: ast::crate_num) -> ~str {
let cdata = get_crate_data(cstore, cnum);
2012-08-01 19:30:05 -05:00
return decoder::get_crate_vers(cdata.data);
}
fn set_crate_data(cstore: cstore, cnum: ast::crate_num,
data: crate_metadata) {
p(cstore).metas.insert(cnum, data);
2012-07-18 18:18:02 -05:00
do vec::iter(decoder::get_crate_module_paths(cstore.intr, data)) |dp| {
let (did, path) = dp;
let d = {crate: cnum, node: did.node};
2012-06-10 02:49:59 -05:00
p(cstore).mod_path_map.insert(d, @path);
}
}
fn have_crate_data(cstore: cstore, cnum: ast::crate_num) -> bool {
2012-08-01 19:30:05 -05:00
return p(cstore).metas.contains_key(cnum);
}
2012-01-23 16:59:00 -06:00
fn iter_crate_data(cstore: cstore, i: fn(ast::crate_num, crate_metadata)) {
2012-06-30 18:19:07 -05:00
for p(cstore).metas.each |k,v| { i(k, v);};
}
fn add_used_crate_file(cstore: cstore, lib: &Path) {
if !vec::contains(p(cstore).used_crate_files, copy *lib) {
vec::push(p(cstore).used_crate_files, copy *lib);
}
}
fn get_used_crate_files(cstore: cstore) -> ~[Path] {
2012-08-01 19:30:05 -05:00
return p(cstore).used_crate_files;
}
fn add_used_library(cstore: cstore, lib: ~str) -> bool {
assert lib != ~"";
2012-08-01 19:30:05 -05:00
if vec::contains(p(cstore).used_libraries, lib) { return false; }
vec::push(p(cstore).used_libraries, lib);
2012-08-01 19:30:05 -05:00
return true;
}
fn get_used_libraries(cstore: cstore) -> ~[~str] {
2012-08-01 19:30:05 -05:00
return p(cstore).used_libraries;
}
fn add_used_link_args(cstore: cstore, args: ~str) {
vec::push_all(p(cstore).used_link_args, str::split_char(args, ' '));
}
fn get_used_link_args(cstore: cstore) -> ~[~str] {
2012-08-01 19:30:05 -05:00
return p(cstore).used_link_args;
}
fn add_use_stmt_cnum(cstore: cstore, use_id: ast::node_id,
2011-07-27 07:19:39 -05:00
cnum: ast::crate_num) {
p(cstore).use_crate_map.insert(use_id, cnum);
}
fn find_use_stmt_cnum(cstore: cstore,
use_id: ast::node_id) -> option<ast::crate_num> {
p(cstore).use_crate_map.find(use_id)
2011-07-07 23:37:56 -05:00
}
// returns hashes of crates directly used by this crate. Hashes are
// sorted by crate name.
2012-07-18 18:18:02 -05:00
fn get_dep_hashes(cstore: cstore) -> ~[~str] {
type crate_hash = {name: ~str, hash: ~str};
let mut result = ~[];
2012-06-30 18:19:07 -05:00
for p(cstore).use_crate_map.each_value |cnum| {
let cdata = cstore::get_crate_data(cstore, cnum);
let hash = decoder::get_crate_hash(cdata.data);
2012-08-22 19:24:52 -05:00
debug!("Add hash[%s]: %s", cdata.name, hash);
2012-07-18 18:18:02 -05:00
vec::push(result, {name: cdata.name, hash: hash});
};
2012-07-18 18:18:02 -05:00
pure fn lteq(a: &crate_hash, b: &crate_hash) -> bool {a.name <= b.name}
let sorted = std::sort::merge_sort(lteq, result);
2012-08-22 19:24:52 -05:00
debug!("sorted:");
2012-06-30 18:19:07 -05:00
for sorted.each |x| {
2012-08-22 19:24:52 -05:00
debug!(" hash[%s]: %s", x.name, x.hash);
}
2012-07-18 18:18:02 -05:00
fn mapper(ch: crate_hash) -> ~str { return ch.hash; }
2012-08-01 19:30:05 -05:00
return vec::map(sorted, mapper);
}
2012-07-18 18:18:02 -05:00
fn get_path(cstore: cstore, d: ast::def_id) -> ~[~str] {
option::map_default(p(cstore).mod_path_map.find(d), ~[],
2012-07-18 18:18:02 -05:00
|ds| str::split_str(*ds, ~"::"))
}
// Local Variables:
// mode: rust
// fill-column: 78;
// indent-tabs-mode: nil
// c-basic-offset: 4
// buffer-file-coding-system: utf-8-unix
// End: