2013-09-09 17:04:29 -05:00
|
|
|
// Copyright 2013 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-10-01 16:53:56 -05:00
|
|
|
//#[cfg(not(stage0))] use cast::transmute;
|
2013-09-09 17:04:29 -05:00
|
|
|
use container::MutableSet;
|
2013-09-15 14:41:02 -05:00
|
|
|
use hashmap::HashSet;
|
2013-09-27 15:43:41 -05:00
|
|
|
use option::{Some, None};
|
|
|
|
use vec::ImmutableVector;
|
2013-09-09 17:04:29 -05:00
|
|
|
|
2013-10-01 16:53:56 -05:00
|
|
|
/// Imports for old crate map versions
|
|
|
|
use cast::transmute;
|
|
|
|
use libc::c_char;
|
|
|
|
use ptr;
|
|
|
|
use str::raw::from_c_str;
|
|
|
|
use vec;
|
|
|
|
|
2013-09-19 16:22:59 -05:00
|
|
|
// Need to tell the linker on OS X to not barf on undefined symbols
|
|
|
|
// and instead look them up at runtime, which we need to resolve
|
|
|
|
// the crate_map properly.
|
|
|
|
#[cfg(target_os = "macos")]
|
|
|
|
#[link_args = "-undefined dynamic_lookup"]
|
|
|
|
extern {}
|
|
|
|
|
2013-09-23 20:13:21 -05:00
|
|
|
#[cfg(not(windows))]
|
2013-09-18 03:58:52 -05:00
|
|
|
extern {
|
2013-09-18 16:44:04 -05:00
|
|
|
#[weak_linkage]
|
|
|
|
#[link_name = "_rust_crate_map_toplevel"]
|
2013-09-27 15:43:41 -05:00
|
|
|
static CRATE_MAP: CrateMap<'static>;
|
2013-09-18 03:58:52 -05:00
|
|
|
}
|
|
|
|
|
2013-10-01 16:53:56 -05:00
|
|
|
/// structs for old crate map versions
|
|
|
|
pub struct ModEntryV0 {
|
|
|
|
name: *c_char,
|
2013-09-09 17:04:29 -05:00
|
|
|
log_level: *mut u32
|
|
|
|
}
|
2013-10-01 16:53:56 -05:00
|
|
|
pub struct CrateMapV0 {
|
|
|
|
entries: *ModEntryV0,
|
|
|
|
children: [*CrateMapV0, ..1]
|
|
|
|
}
|
2013-09-15 14:41:02 -05:00
|
|
|
|
2013-10-01 16:53:56 -05:00
|
|
|
pub struct CrateMapV1 {
|
|
|
|
version: i32,
|
|
|
|
entries: *ModEntryV0,
|
|
|
|
/// a dynamically sized struct, where all pointers to children are listed adjacent
|
|
|
|
/// to the struct, terminated with NULL
|
|
|
|
children: [*CrateMapV1, ..1]
|
|
|
|
}
|
|
|
|
|
|
|
|
pub struct ModEntry<'self> {
|
|
|
|
name: &'self str,
|
|
|
|
log_level: *mut u32
|
2013-09-09 17:04:29 -05:00
|
|
|
}
|
|
|
|
|
2013-09-27 15:43:41 -05:00
|
|
|
pub struct CrateMap<'self> {
|
2013-09-24 12:35:42 -05:00
|
|
|
version: i32,
|
2013-09-29 07:23:58 -05:00
|
|
|
entries: &'self [ModEntry<'self>],
|
2013-09-24 12:35:42 -05:00
|
|
|
/// a dynamically sized struct, where all pointers to children are listed adjacent
|
|
|
|
/// to the struct, terminated with NULL
|
2013-09-27 15:43:41 -05:00
|
|
|
children: &'self [&'self CrateMap<'self>]
|
2013-09-24 12:35:42 -05:00
|
|
|
}
|
|
|
|
|
2013-10-01 16:53:56 -05:00
|
|
|
|
|
|
|
|
2013-09-23 20:13:21 -05:00
|
|
|
#[cfg(not(windows))]
|
2013-09-27 15:43:41 -05:00
|
|
|
pub fn get_crate_map() -> &'static CrateMap<'static> {
|
|
|
|
&'static CRATE_MAP
|
2013-09-18 03:58:52 -05:00
|
|
|
}
|
|
|
|
|
2013-09-23 20:13:21 -05:00
|
|
|
#[cfg(windows)]
|
2013-09-21 21:48:12 -05:00
|
|
|
#[fixed_stack_segment]
|
|
|
|
#[inline(never)]
|
2013-09-27 15:43:41 -05:00
|
|
|
pub fn get_crate_map() -> &'static CrateMap<'static> {
|
2013-09-21 21:48:12 -05:00
|
|
|
use c_str::ToCStr;
|
|
|
|
use unstable::dynamic_lib::dl;
|
|
|
|
|
|
|
|
let sym = unsafe {
|
|
|
|
let module = dl::open_internal();
|
|
|
|
let sym = do "__rust_crate_map_toplevel".with_c_str |buf| {
|
|
|
|
dl::symbol(module, buf)
|
|
|
|
};
|
|
|
|
dl::close(module);
|
|
|
|
sym
|
|
|
|
};
|
2013-09-27 15:43:41 -05:00
|
|
|
sym
|
2013-09-21 21:48:12 -05:00
|
|
|
}
|
|
|
|
|
2013-09-27 15:43:41 -05:00
|
|
|
fn version(crate_map: &CrateMap) -> i32 {
|
2013-09-15 14:41:02 -05:00
|
|
|
match crate_map.version {
|
2013-10-01 16:53:56 -05:00
|
|
|
2 => return 2,
|
2013-09-09 17:04:29 -05:00
|
|
|
1 => return 1,
|
|
|
|
_ => return 0
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-09-27 15:43:41 -05:00
|
|
|
fn iter_module_map(mod_entries: &[ModEntry], f: &fn(&ModEntry)) {
|
|
|
|
for entry in mod_entries.iter() {
|
|
|
|
f(entry);
|
2013-09-09 17:04:29 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-10-01 16:53:56 -05:00
|
|
|
unsafe fn iter_module_map_v0(entries: *ModEntryV0, f: &fn(&ModEntry)) {
|
|
|
|
let mut curr = entries;
|
|
|
|
while !(*curr).name.is_null() {
|
|
|
|
let mod_entry = ModEntry { name: from_c_str((*curr).name), log_level: (*curr).log_level };
|
|
|
|
f(&mod_entry);
|
|
|
|
curr = curr.offset(1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-09-27 15:43:41 -05:00
|
|
|
fn do_iter_crate_map<'a>(crate_map: &'a CrateMap<'a>, f: &fn(&ModEntry),
|
|
|
|
visited: &mut HashSet<*CrateMap<'a>>) {
|
2013-09-15 14:41:02 -05:00
|
|
|
if visited.insert(crate_map as *CrateMap) {
|
2013-10-01 16:53:56 -05:00
|
|
|
match version(crate_map) {
|
|
|
|
2 => {
|
|
|
|
let (entries, children) = (crate_map.entries, crate_map.children);
|
|
|
|
iter_module_map(entries, |x| f(x));
|
|
|
|
for child in children.iter() {
|
|
|
|
do_iter_crate_map(*child, |x| f(x), visited);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
/// code for old crate map versions
|
|
|
|
1 => unsafe {
|
|
|
|
let v1: *CrateMapV1 = transmute(crate_map);
|
|
|
|
iter_module_map_v0((*v1).entries, |x| f(x));
|
|
|
|
let children = vec::raw::to_ptr((*v1).children);
|
|
|
|
do ptr::array_each(children) |child| {
|
|
|
|
do_iter_crate_map(transmute(child), |x| f(x), visited);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
0 => unsafe {
|
|
|
|
let v0: *CrateMapV0 = transmute(crate_map);
|
|
|
|
iter_module_map_v0((*v0).entries, |x| f(x));
|
|
|
|
let children = vec::raw::to_ptr((*v0).children);
|
|
|
|
do ptr::array_each(children) |child| {
|
|
|
|
do_iter_crate_map(transmute(child), |x| f(x), visited);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
_ => fail2!("invalid crate map version")
|
2013-09-09 17:04:29 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-09-15 14:41:02 -05:00
|
|
|
/// Iterates recursively over `crate_map` and all child crate maps
|
2013-09-27 15:43:41 -05:00
|
|
|
pub fn iter_crate_map<'a>(crate_map: &'a CrateMap<'a>, f: &fn(&ModEntry)) {
|
2013-09-09 17:04:29 -05:00
|
|
|
// XXX: use random numbers as keys from the OS-level RNG when there is a nice
|
|
|
|
// way to do this
|
2013-09-27 15:43:41 -05:00
|
|
|
let mut v: HashSet<*CrateMap<'a>> = HashSet::with_capacity_and_keys(0, 0, 32);
|
|
|
|
do_iter_crate_map(crate_map, f, &mut v);
|
2013-09-09 17:04:29 -05:00
|
|
|
}
|
|
|
|
|
2013-09-15 14:41:02 -05:00
|
|
|
#[cfg(test)]
|
|
|
|
mod tests {
|
2013-09-27 15:43:41 -05:00
|
|
|
use rt::crate_map::{CrateMap, ModEntry, iter_crate_map};
|
2013-09-09 17:04:29 -05:00
|
|
|
|
2013-09-15 14:41:02 -05:00
|
|
|
#[test]
|
|
|
|
fn iter_crate_map_duplicates() {
|
2013-09-29 07:23:58 -05:00
|
|
|
let mut level3: u32 = 3;
|
|
|
|
|
|
|
|
let entries = [
|
|
|
|
ModEntry { name: "c::m1", log_level: &mut level3},
|
|
|
|
];
|
|
|
|
|
|
|
|
let child_crate = CrateMap {
|
2013-10-01 16:53:56 -05:00
|
|
|
version: 2,
|
2013-09-29 07:23:58 -05:00
|
|
|
entries: entries,
|
|
|
|
children: []
|
|
|
|
};
|
|
|
|
|
|
|
|
let root_crate = CrateMap {
|
2013-10-01 16:53:56 -05:00
|
|
|
version: 2,
|
2013-09-29 07:23:58 -05:00
|
|
|
entries: [],
|
|
|
|
children: [&child_crate, &child_crate]
|
|
|
|
};
|
|
|
|
|
|
|
|
let mut cnt = 0;
|
2013-09-15 14:41:02 -05:00
|
|
|
unsafe {
|
2013-09-29 07:23:58 -05:00
|
|
|
do iter_crate_map(&root_crate) |entry| {
|
2013-09-15 14:41:02 -05:00
|
|
|
assert!(*entry.log_level == 3);
|
|
|
|
cnt += 1;
|
|
|
|
}
|
|
|
|
assert!(cnt == 1);
|
|
|
|
}
|
|
|
|
}
|
2013-09-09 17:04:29 -05:00
|
|
|
|
2013-09-15 14:41:02 -05:00
|
|
|
#[test]
|
|
|
|
fn iter_crate_map_follow_children() {
|
2013-09-29 07:23:58 -05:00
|
|
|
let mut level2: u32 = 2;
|
|
|
|
let mut level3: u32 = 3;
|
|
|
|
let child_crate2 = CrateMap {
|
2013-10-01 16:53:56 -05:00
|
|
|
version: 2,
|
2013-09-29 07:23:58 -05:00
|
|
|
entries: [
|
|
|
|
ModEntry { name: "c::m1", log_level: &mut level2},
|
|
|
|
ModEntry { name: "c::m2", log_level: &mut level3},
|
|
|
|
],
|
|
|
|
children: []
|
|
|
|
};
|
|
|
|
|
|
|
|
let child_crate1 = CrateMap {
|
2013-10-01 16:53:56 -05:00
|
|
|
version: 2,
|
2013-09-29 07:23:58 -05:00
|
|
|
entries: [
|
|
|
|
ModEntry { name: "t::f1", log_level: &mut 1},
|
|
|
|
],
|
|
|
|
children: [&child_crate2]
|
|
|
|
};
|
|
|
|
|
|
|
|
let root_crate = CrateMap {
|
2013-10-01 16:53:56 -05:00
|
|
|
version: 2,
|
2013-09-29 07:23:58 -05:00
|
|
|
entries: [
|
|
|
|
ModEntry { name: "t::f2", log_level: &mut 0},
|
|
|
|
],
|
|
|
|
children: [&child_crate1]
|
|
|
|
};
|
|
|
|
|
|
|
|
let mut cnt = 0;
|
2013-09-15 14:41:02 -05:00
|
|
|
unsafe {
|
2013-09-29 07:23:58 -05:00
|
|
|
do iter_crate_map(&root_crate) |entry| {
|
2013-09-15 14:41:02 -05:00
|
|
|
assert!(*entry.log_level == cnt);
|
|
|
|
cnt += 1;
|
|
|
|
}
|
|
|
|
assert!(cnt == 4);
|
2013-09-09 17:04:29 -05:00
|
|
|
}
|
|
|
|
}
|
2013-10-01 16:53:56 -05:00
|
|
|
|
|
|
|
|
|
|
|
/// Tests for old crate map versions
|
|
|
|
#[test]
|
|
|
|
fn iter_crate_map_duplicates_v1() {
|
|
|
|
use c_str::ToCStr;
|
|
|
|
use cast::transmute;
|
|
|
|
use ptr;
|
|
|
|
use rt::crate_map::{CrateMapV1, ModEntryV0, iter_crate_map};
|
|
|
|
use vec;
|
|
|
|
|
|
|
|
struct CrateMapT3 {
|
|
|
|
version: i32,
|
|
|
|
entries: *ModEntryV0,
|
|
|
|
children: [*CrateMapV1, ..3]
|
|
|
|
}
|
|
|
|
|
|
|
|
unsafe {
|
|
|
|
let mod_name1 = "c::m1".to_c_str();
|
|
|
|
let mut level3: u32 = 3;
|
|
|
|
|
|
|
|
let entries: ~[ModEntryV0] = ~[
|
|
|
|
ModEntryV0 { name: mod_name1.with_ref(|buf| buf), log_level: &mut level3},
|
|
|
|
ModEntryV0 { name: ptr::null(), log_level: ptr::mut_null()}
|
|
|
|
];
|
|
|
|
let child_crate = CrateMapV1 {
|
|
|
|
version: 1,
|
|
|
|
entries: vec::raw::to_ptr(entries),
|
|
|
|
children: [ptr::null()]
|
|
|
|
};
|
|
|
|
|
|
|
|
let root_crate = CrateMapT3 {
|
|
|
|
version: 1,
|
|
|
|
entries: vec::raw::to_ptr([ModEntryV0 { name: ptr::null(), log_level: ptr::mut_null()}]),
|
|
|
|
children: [&child_crate as *CrateMapV1, &child_crate as *CrateMapV1, ptr::null()]
|
|
|
|
};
|
|
|
|
|
|
|
|
let mut cnt = 0;
|
|
|
|
do iter_crate_map(transmute(&root_crate)) |entry| {
|
|
|
|
assert!(*(*entry).log_level == 3);
|
|
|
|
cnt += 1;
|
|
|
|
}
|
|
|
|
assert!(cnt == 1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn iter_crate_map_follow_children_v1() {
|
|
|
|
use c_str::ToCStr;
|
|
|
|
use cast::transmute;
|
|
|
|
use ptr;
|
|
|
|
use rt::crate_map::{CrateMapV1, ModEntryV0, iter_crate_map};
|
|
|
|
use vec;
|
|
|
|
|
|
|
|
struct CrateMapT2 {
|
|
|
|
version: i32,
|
|
|
|
entries: *ModEntryV0,
|
|
|
|
children: [*CrateMapV1, ..2]
|
|
|
|
}
|
|
|
|
|
|
|
|
unsafe {
|
|
|
|
let mod_name1 = "c::m1".to_c_str();
|
|
|
|
let mod_name2 = "c::m2".to_c_str();
|
|
|
|
let mut level2: u32 = 2;
|
|
|
|
let mut level3: u32 = 3;
|
|
|
|
let child_crate2 = CrateMapV1 {
|
|
|
|
version: 1,
|
|
|
|
entries: vec::raw::to_ptr([
|
|
|
|
ModEntryV0 { name: mod_name1.with_ref(|buf| buf), log_level: &mut level2},
|
|
|
|
ModEntryV0 { name: mod_name2.with_ref(|buf| buf), log_level: &mut level3},
|
|
|
|
ModEntryV0 { name: ptr::null(), log_level: ptr::mut_null()}
|
|
|
|
]),
|
|
|
|
children: [ptr::null()]
|
|
|
|
};
|
|
|
|
|
|
|
|
let child_crate1 = CrateMapT2 {
|
|
|
|
version: 1,
|
|
|
|
entries: vec::raw::to_ptr([
|
|
|
|
ModEntryV0 { name: "t::f1".with_c_str(|buf| buf), log_level: &mut 1},
|
|
|
|
ModEntryV0 { name: ptr::null(), log_level: ptr::mut_null()}
|
|
|
|
]),
|
|
|
|
children: [&child_crate2 as *CrateMapV1, ptr::null()]
|
|
|
|
};
|
|
|
|
|
|
|
|
let child_crate1_ptr: *CrateMapV1 = transmute(&child_crate1);
|
|
|
|
let root_crate = CrateMapT2 {
|
|
|
|
version: 1,
|
|
|
|
entries: vec::raw::to_ptr([
|
|
|
|
ModEntryV0 { name: "t::f1".with_c_str(|buf| buf), log_level: &mut 0},
|
|
|
|
ModEntryV0 { name: ptr::null(), log_level: ptr::mut_null()}
|
|
|
|
]),
|
|
|
|
children: [child_crate1_ptr, ptr::null()]
|
|
|
|
};
|
|
|
|
|
|
|
|
let mut cnt = 0;
|
|
|
|
do iter_crate_map(transmute(&root_crate)) |entry| {
|
|
|
|
assert!(*(*entry).log_level == cnt);
|
|
|
|
cnt += 1;
|
|
|
|
}
|
|
|
|
assert!(cnt == 4);
|
|
|
|
}
|
|
|
|
}
|
2013-09-09 17:04:29 -05:00
|
|
|
}
|