From 09e49a8e6c11e57142a0f628745142d94fc87444 Mon Sep 17 00:00:00 2001 From: Alex Crichton Date: Wed, 17 Jul 2013 20:27:20 -0700 Subject: [PATCH] Allow linking against crates with #[no_std] Previously having optional lang_items caused an assertion failure at compile-time, and then once that was fixed there was a segfault at runtime of using a NULL crate-map (crates with no_std) --- src/librustc/metadata/encoder.rs | 3 +++ src/librustc/middle/lang_items.rs | 4 ++-- src/test/auxiliary/no_std_crate.rs | 3 +++ src/test/run-pass/no-std-xcrate.rs | 21 +++++++++++++++++ src/test/run-pass/no-std-xcrate2.rs | 35 +++++++++++++++++++++++++++++ 5 files changed, 64 insertions(+), 2 deletions(-) create mode 100644 src/test/auxiliary/no_std_crate.rs create mode 100644 src/test/run-pass/no-std-xcrate.rs create mode 100644 src/test/run-pass/no-std-xcrate2.rs diff --git a/src/librustc/metadata/encoder.rs b/src/librustc/metadata/encoder.rs index c216e8d2386..d6287d457c1 100644 --- a/src/librustc/metadata/encoder.rs +++ b/src/librustc/metadata/encoder.rs @@ -1455,6 +1455,9 @@ fn encode_lang_items(ecx: &EncodeContext, ebml_w: &mut writer::Encoder) { ebml_w.start_tag(tag_lang_items); for ecx.tcx.lang_items.each_item |def_id, i| { + let def_id = match def_id { + Some(id) => id, None => { loop } + }; if def_id.crate != local_crate { loop; } diff --git a/src/librustc/middle/lang_items.rs b/src/librustc/middle/lang_items.rs index 6bc4564bb13..54e7a2fd7e7 100644 --- a/src/librustc/middle/lang_items.rs +++ b/src/librustc/middle/lang_items.rs @@ -92,8 +92,8 @@ impl LanguageItems { } } - pub fn each_item(&self, f: &fn(def_id: def_id, i: uint) -> bool) -> bool { - self.items.iter().enumerate().advance(|(i, &item)| f(item.get(), i)) + pub fn each_item(&self, f: &fn(Option, uint) -> bool) -> bool { + self.items.iter().enumerate().advance(|(i, &item)| f(item, i)) } pub fn item_name(index: uint) -> &'static str { diff --git a/src/test/auxiliary/no_std_crate.rs b/src/test/auxiliary/no_std_crate.rs new file mode 100644 index 00000000000..70f1b76e246 --- /dev/null +++ b/src/test/auxiliary/no_std_crate.rs @@ -0,0 +1,3 @@ +#[no_std]; + +pub fn foo() {} diff --git a/src/test/run-pass/no-std-xcrate.rs b/src/test/run-pass/no-std-xcrate.rs new file mode 100644 index 00000000000..104e33b7488 --- /dev/null +++ b/src/test/run-pass/no-std-xcrate.rs @@ -0,0 +1,21 @@ +// 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 or the MIT license +// , at your +// option. This file may not be copied, modified, or distributed +// except according to those terms. + +// xfail-fast +// aux-build:no_std_crate.rs + +// This tests that crates which link to std can also be linked to crates with +// #[no_std] that have no lang items. + +extern mod no_std_crate; + +fn main() { + no_std_crate::foo(); +} diff --git a/src/test/run-pass/no-std-xcrate2.rs b/src/test/run-pass/no-std-xcrate2.rs new file mode 100644 index 00000000000..e393eb3a5c9 --- /dev/null +++ b/src/test/run-pass/no-std-xcrate2.rs @@ -0,0 +1,35 @@ +// 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 or the MIT license +// , at your +// option. This file may not be copied, modified, or distributed +// except according to those terms. + +// xfail-test: this has weird linking problems on linux, and it probably needs a +// solution along the lines of disabling segmented stacks and/or the +// stack checks. +// aux-build:no_std_crate.rs + +// This tests that libraries built with #[no_std] can be linked to crates with +// #[no_std] and actually run. + +#[no_std]; + +extern mod no_std_crate; + +// This is an unfortunate thing to have to do on linux :( +#[cfg(target_os = "linux")] +#[doc(hidden)] +pub mod linkhack { + #[link_args="-lrustrt -lrt"] + extern {} +} + +#[start] +fn main(_: int, _: **u8, _: *u8) -> int { + no_std_crate::foo(); + 0 +}