2012-12-03 18:48:01 -06: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.
|
|
|
|
|
2013-01-07 16:16:52 -06:00
|
|
|
|
2011-07-07 20:39:44 -05:00
|
|
|
// The crate store - a central repo for information collected about external
|
|
|
|
// crates and libraries
|
|
|
|
|
2013-01-08 21:37:25 -06:00
|
|
|
use core::prelude::*;
|
|
|
|
|
2012-12-23 16:41:37 -06:00
|
|
|
use metadata::creader;
|
|
|
|
use metadata::cstore;
|
|
|
|
use metadata::decoder;
|
|
|
|
|
|
|
|
use core::option;
|
|
|
|
use core::str;
|
|
|
|
use core::vec;
|
2013-02-01 01:13:36 -06:00
|
|
|
use std::oldmap::HashMap;
|
|
|
|
use std::oldmap;
|
2012-12-23 16:41:37 -06:00
|
|
|
use std;
|
2012-09-04 13:54:36 -05:00
|
|
|
use syntax::{ast, attr};
|
|
|
|
use syntax::parse::token::ident_interner;
|
2011-07-07 20:00:16 -05:00
|
|
|
|
2011-07-08 14:08:43 -05:00
|
|
|
// 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.
|
2013-02-01 01:13:36 -06:00
|
|
|
pub type cnum_map = oldmap::HashMap<ast::crate_num, ast::crate_num>;
|
2011-07-08 14:08:43 -05:00
|
|
|
|
2013-01-29 18:51:16 -06:00
|
|
|
pub type crate_metadata = @{name: ~str,
|
|
|
|
data: @~[u8],
|
|
|
|
cnum_map: cnum_map,
|
|
|
|
cnum: ast::crate_num};
|
2011-07-07 20:00:16 -05:00
|
|
|
|
2011-07-10 00:56:12 -05:00
|
|
|
// 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
|
2011-07-10 00:56:12 -05:00
|
|
|
// 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.
|
2013-01-29 18:51:16 -06:00
|
|
|
pub enum CStore { private(cstore_private), }
|
2011-07-10 00:56:12 -05:00
|
|
|
|
2011-07-27 07:19:39 -05:00
|
|
|
type cstore_private =
|
2013-02-01 01:13:36 -06:00
|
|
|
@{metas: oldmap::HashMap<ast::crate_num, crate_metadata>,
|
2011-07-27 07:19:39 -05:00
|
|
|
use_crate_map: use_crate_map,
|
2012-08-24 17:28:43 -05:00
|
|
|
mut used_crate_files: ~[Path],
|
2012-07-14 00:57:48 -05:00
|
|
|
mut used_libraries: ~[~str],
|
2012-07-18 18:18:02 -05:00
|
|
|
mut used_link_args: ~[~str],
|
2012-09-19 20:50:24 -05:00
|
|
|
intr: @ident_interner};
|
2011-07-10 00:56:12 -05:00
|
|
|
|
2011-07-07 23:37:56 -05:00
|
|
|
// Map from node_id's of local use statements to crate numbers
|
2013-02-01 01:13:36 -06:00
|
|
|
type use_crate_map = oldmap::HashMap<ast::node_id, ast::crate_num>;
|
2011-07-07 23:37:56 -05:00
|
|
|
|
2011-07-10 00:56:12 -05:00
|
|
|
// Internal method to retrieve the data from the cstore
|
2012-10-15 16:56:42 -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
|
|
|
}
|
2011-07-07 20:00:16 -05:00
|
|
|
|
2013-01-29 18:51:16 -06:00
|
|
|
pub fn mk_cstore(intr: @ident_interner) -> CStore {
|
2013-02-01 01:13:36 -06:00
|
|
|
let meta_cache = oldmap::HashMap();
|
|
|
|
let crate_map = oldmap::HashMap();
|
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,
|
|
|
|
mut used_crate_files: ~[],
|
|
|
|
mut used_libraries: ~[],
|
|
|
|
mut used_link_args: ~[],
|
|
|
|
intr: intr});
|
2011-07-07 20:00:16 -05:00
|
|
|
}
|
|
|
|
|
2013-01-29 18:51:16 -06:00
|
|
|
pub fn get_crate_data(cstore: CStore, cnum: ast::crate_num)
|
|
|
|
-> crate_metadata {
|
2013-02-05 21:41:45 -06:00
|
|
|
return p(cstore).metas.get(&cnum);
|
2011-07-07 20:00:16 -05:00
|
|
|
}
|
|
|
|
|
2013-01-29 18:51:16 -06:00
|
|
|
pub fn get_crate_hash(cstore: CStore, cnum: ast::crate_num) -> ~str {
|
2012-04-06 05:45:49 -05:00
|
|
|
let cdata = get_crate_data(cstore, cnum);
|
2012-08-01 19:30:05 -05:00
|
|
|
return decoder::get_crate_hash(cdata.data);
|
2012-04-06 05:45:49 -05:00
|
|
|
}
|
|
|
|
|
2013-01-29 18:51:16 -06:00
|
|
|
pub fn get_crate_vers(cstore: CStore, cnum: ast::crate_num) -> ~str {
|
2012-04-06 05:45:49 -05:00
|
|
|
let cdata = get_crate_data(cstore, cnum);
|
2012-08-01 19:30:05 -05:00
|
|
|
return decoder::get_crate_vers(cdata.data);
|
2012-04-06 05:45:49 -05:00
|
|
|
}
|
|
|
|
|
2013-01-29 18:51:16 -06:00
|
|
|
pub fn set_crate_data(cstore: CStore,
|
|
|
|
cnum: ast::crate_num,
|
|
|
|
data: crate_metadata) {
|
2011-07-10 00:56:12 -05:00
|
|
|
p(cstore).metas.insert(cnum, data);
|
2011-07-07 20:00:16 -05:00
|
|
|
}
|
|
|
|
|
2013-01-29 18:51:16 -06:00
|
|
|
pub fn have_crate_data(cstore: CStore, cnum: ast::crate_num) -> bool {
|
2013-02-02 01:04:22 -06:00
|
|
|
return p(cstore).metas.contains_key_ref(&cnum);
|
2011-07-10 00:56:12 -05:00
|
|
|
}
|
|
|
|
|
2013-01-29 18:51:16 -06:00
|
|
|
pub fn iter_crate_data(cstore: CStore,
|
|
|
|
i: fn(ast::crate_num, crate_metadata)) {
|
2013-02-02 13:47:41 -06:00
|
|
|
for p(cstore).metas.each_ref |&k, &v| { i(k, v);};
|
2011-07-07 20:00:16 -05:00
|
|
|
}
|
|
|
|
|
2013-01-29 18:51:16 -06:00
|
|
|
pub fn add_used_crate_file(cstore: CStore, lib: &Path) {
|
2012-09-28 00:20:47 -05:00
|
|
|
if !vec::contains(p(cstore).used_crate_files, lib) {
|
2012-09-26 19:33:34 -05:00
|
|
|
p(cstore).used_crate_files.push(copy *lib);
|
2011-07-07 20:25:56 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-01-29 18:51:16 -06:00
|
|
|
pub fn get_used_crate_files(cstore: CStore) -> ~[Path] {
|
2013-01-07 16:16:52 -06:00
|
|
|
return /*bad*/copy p(cstore).used_crate_files;
|
2011-07-07 20:25:56 -05:00
|
|
|
}
|
|
|
|
|
2013-01-29 18:51:16 -06:00
|
|
|
pub fn add_used_library(cstore: CStore, +lib: ~str) -> bool {
|
2012-07-14 00:57:48 -05:00
|
|
|
assert lib != ~"";
|
2011-07-07 20:33:59 -05:00
|
|
|
|
2012-09-28 00:20:47 -05:00
|
|
|
if vec::contains(p(cstore).used_libraries, &lib) { return false; }
|
2012-09-26 19:33:34 -05:00
|
|
|
p(cstore).used_libraries.push(lib);
|
2012-08-01 19:30:05 -05:00
|
|
|
return true;
|
2011-07-07 20:33:59 -05:00
|
|
|
}
|
|
|
|
|
2013-01-29 18:51:16 -06:00
|
|
|
pub fn get_used_libraries(cstore: CStore) -> ~[~str] {
|
2013-01-07 16:16:52 -06:00
|
|
|
return /*bad*/copy p(cstore).used_libraries;
|
2011-07-07 20:33:59 -05:00
|
|
|
}
|
|
|
|
|
2013-01-29 18:51:16 -06:00
|
|
|
pub fn add_used_link_args(cstore: CStore, args: ~str) {
|
2012-09-26 19:33:34 -05:00
|
|
|
p(cstore).used_link_args.push_all(str::split_char(args, ' '));
|
2011-07-07 20:38:42 -05:00
|
|
|
}
|
|
|
|
|
2013-01-29 18:51:16 -06:00
|
|
|
pub fn get_used_link_args(cstore: CStore) -> ~[~str] {
|
2013-01-07 16:16:52 -06:00
|
|
|
return /*bad*/copy p(cstore).used_link_args;
|
2011-07-07 20:38:42 -05:00
|
|
|
}
|
|
|
|
|
2013-01-29 18:51:16 -06:00
|
|
|
pub fn add_use_stmt_cnum(cstore: CStore, use_id: ast::node_id,
|
|
|
|
cnum: ast::crate_num) {
|
2011-07-10 00:56:12 -05:00
|
|
|
p(cstore).use_crate_map.insert(use_id, cnum);
|
|
|
|
}
|
|
|
|
|
2013-01-29 18:51:16 -06:00
|
|
|
pub fn find_use_stmt_cnum(cstore: CStore,
|
|
|
|
use_id: ast::node_id) -> Option<ast::crate_num> {
|
2013-02-05 21:41:45 -06:00
|
|
|
p(cstore).use_crate_map.find(&use_id)
|
2011-07-07 23:37:56 -05:00
|
|
|
}
|
|
|
|
|
2011-12-11 09:42:32 -06:00
|
|
|
// returns hashes of crates directly used by this crate. Hashes are
|
|
|
|
// sorted by crate name.
|
2013-01-29 18:51:16 -06:00
|
|
|
pub fn get_dep_hashes(cstore: CStore) -> ~[~str] {
|
2012-07-18 18:18:02 -05:00
|
|
|
type crate_hash = {name: ~str, hash: ~str};
|
2012-06-29 18:26:56 -05:00
|
|
|
let mut result = ~[];
|
2011-12-11 09:42:32 -06:00
|
|
|
|
2013-02-02 12:19:40 -06:00
|
|
|
for p(cstore).use_crate_map.each_value_ref |&cnum| {
|
2011-12-11 09:42:32 -06:00
|
|
|
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);
|
2013-01-07 16:16:52 -06:00
|
|
|
result.push({name: /*bad*/copy cdata.name, hash: hash});
|
2011-12-11 09:42:32 -06:00
|
|
|
};
|
2012-07-18 18:18:02 -05:00
|
|
|
pure fn lteq(a: &crate_hash, b: &crate_hash) -> bool {a.name <= b.name}
|
2012-09-27 19:05:13 -05:00
|
|
|
let sorted = std::sort::merge_sort(result, lteq);
|
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);
|
2011-12-11 09:42:32 -06:00
|
|
|
}
|
2013-01-07 16:16:52 -06:00
|
|
|
fn mapper(ch: &crate_hash) -> ~str { return /*bad*/copy ch.hash; }
|
2012-08-01 19:30:05 -05:00
|
|
|
return vec::map(sorted, mapper);
|
2011-12-11 09:42:32 -06:00
|
|
|
}
|
2012-01-20 12:45:25 -06:00
|
|
|
|
2011-07-07 20:00:16 -05:00
|
|
|
// Local Variables:
|
|
|
|
// mode: rust
|
|
|
|
// fill-column: 78;
|
|
|
|
// indent-tabs-mode: nil
|
|
|
|
// c-basic-offset: 4
|
|
|
|
// buffer-file-coding-system: utf-8-unix
|
|
|
|
// End:
|