Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Make -Z gen-crate-map usable for I/O #10437

Closed
Closed
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
21 changes: 11 additions & 10 deletions src/librustc/middle/trans/base.rs
Original file line number Diff line number Diff line change
Expand Up @@ -77,7 +77,7 @@ use extra::time;
use extra::sort;
use syntax::ast::Name;
use syntax::ast_map::{path, path_elt_to_str, path_name, path_pretty_name};
use syntax::ast_util::{local_def};
use syntax::ast_util::{local_def, is_local};
use syntax::attr;
use syntax::attr::AttrMetaMethods;
use syntax::codemap::Span;
Expand Down Expand Up @@ -2979,7 +2979,7 @@ pub fn decl_crate_map(sess: session::Session, mapmeta: LinkMeta,
return map;
}

pub fn fill_crate_map(ccx: &mut CrateContext, map: ValueRef) {
pub fn fill_crate_map(ccx: @mut CrateContext, map: ValueRef) {
let mut subcrates: ~[ValueRef] = ~[];
let mut i = 1;
let cstore = ccx.sess.cstore;
Expand All @@ -2997,19 +2997,20 @@ pub fn fill_crate_map(ccx: &mut CrateContext, map: ValueRef) {
subcrates.push(p2i(ccx, cr));
i += 1;
}
let event_loop_factory = if !*ccx.sess.building_library {
match ccx.tcx.lang_items.event_loop_factory() {
Some(did) => unsafe {
let event_loop_factory = match ccx.tcx.lang_items.event_loop_factory() {
Some(did) => unsafe {
if is_local(did) {
llvm::LLVMConstPointerCast(get_item_val(ccx, did.node),
ccx.int_type.ptr_to().to_ref())
} else {
let name = csearch::get_symbol(ccx.sess.cstore, did);
let global = do name.with_c_str |buf| {
llvm::LLVMAddGlobal(ccx.llmod, ccx.int_type.to_ref(), buf)
};
global
},
None => C_null(ccx.int_type.ptr_to())
}
} else {
C_null(ccx.int_type.ptr_to())
}
},
None => C_null(ccx.int_type.ptr_to())
};
unsafe {
let maptype = Type::array(&ccx.int_type, subcrates.len() as u64);
Expand Down
88 changes: 88 additions & 0 deletions src/test/run-pass-fulldeps/bootstrap-from-c.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,88 @@
// 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.

// This test exercises bootstrapping the runtime from C to make sure that this
// continues to work. The runtime is expected to have all local I/O services
// that one would normally expect.

#[feature(managed_boxes)];

extern mod extra;
extern mod rustc;
extern mod syntax;

use extra::glob;
use extra::tempfile::TempDir;
use rustc::back::link;
use rustc::driver::driver;
use rustc::driver::session;
use std::os;
use std::rt::io::fs;
use std::run;
use std::str;
use syntax::diagnostic;

fn main() {
// Sketchily figure out where our sysroot is
let mut sysroot = Path::new(os::self_exe_path().unwrap());
sysroot.pop();
sysroot.pop();
sysroot.push("stage2");
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This is going from $arch/test/run-pass-fulldeps to $arch/stage2, right?

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Correct, I'm not entirely certain that this is correct in cross-arch builds, I figured I'd let bors sort that out.


// Give ourselves a little workspace
let d = TempDir::new("mytest").unwrap();
let d = d.path();

// Figure out where we are
let mut me = Path::new(file!());
me.pop();
let srcfile = me.join("bootstrap-from-c/lib.rs");
let cfile = me.join("bootstrap-from-c/main.c");

// Compile the rust crate
let options = @session::options {
maybe_sysroot: Some(@sysroot),
debugging_opts: session::gen_crate_map,
..(*session::basic_options()).clone()
};
let diagnostic = @diagnostic::DefaultEmitter as @diagnostic::Emitter;
let session = driver::build_session(options, diagnostic);
driver::compile_input(session, ~[], &driver::file_input(srcfile),
&Some(d.clone()), &None);

// Copy the C source into place
let cdst = d.join("main.c");
let exe = d.join("out" + os::consts::EXE_SUFFIX);
fs::copy(&cfile, &cdst);

// Figure out where we put the dynamic library
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This is unfortunate... no way to force the filename even when compiling entirely by hand. :(

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Sadly no, it's that darn crate hash messing up our pretty filenames.

let dll = os::dll_filename("boot-*");
let dll = glob::glob(d.as_str().unwrap() + "/" + dll).next().unwrap();

// Compile the c program with all the appropriate arguments. We're compiling
// the cfile and the library together, and may have to do some linker rpath
// magic to make sure that the dll can get found when the executable is
// running.
let cc = link::get_cc_prog(session);
let mut cc_args = session.targ_cfg.target_strs.cc_args.clone();
cc_args.push_all([~"-o", exe.as_str().unwrap().to_owned()]);
cc_args.push(cdst.as_str().unwrap().to_owned());
cc_args.push(dll.as_str().unwrap().to_owned());
if cfg!(target_os = "macos") {
cc_args.push("-Wl,-rpath," + d.as_str().unwrap().to_owned());
}
let status = run::process_status(cc, cc_args);
assert!(status.success());

// Finally, run the program and make sure that it tells us hello.
let res = run::process_output(exe.as_str().unwrap().to_owned(), []);
assert!(res.status.success());
assert_eq!(str::from_utf8(res.output), ~"hello\n");
}
25 changes: 25 additions & 0 deletions src/test/run-pass-fulldeps/bootstrap-from-c/lib.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,25 @@
// 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.

#[link(package_id = "boot", name = "boot", vers = "0.1")];

extern mod rustuv; // pull in uvio

use std::rt;
use std::ptr;

#[no_mangle] // this needs to get called from C
pub extern "C" fn foo(argc: int, argv: **u8) -> int {
do rt::start(argc, argv) {
do spawn {
println!("hello");
}
}
}
16 changes: 16 additions & 0 deletions src/test/run-pass-fulldeps/bootstrap-from-c/main.c
Original file line number Diff line number Diff line change
@@ -0,0 +1,16 @@
// 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.

// this is the rust entry point that we're going to call.
int foo(int argc, char *argv[]);

int main(int argc, char *argv[]) {
return foo(argc, argv);
}