mirror of
https://github.com/rust-lang/rust.git
synced 2025-10-27 19:16:36 +00:00
raw-dylib is a link kind that allows rustc to link against a library without having any library files present. This currently only exists on Windows. rustc will take all the symbols from raw-dylib link blocks and put them in an import library, where they can then be resolved by the linker. While import libraries don't exist on ELF, it would still be convenient to have this same functionality. Not having the libraries present at build-time can be convenient for several reasons, especially cross-compilation. With raw-dylib, code linking against a library can be cross-compiled without needing to have these libraries available on the build machine. If the libc crate makes use of this, it would allow cross-compilation without having any libc available on the build machine. This is not yet possible with this implementation, at least against libc's like glibc that use symbol versioning. The raw-dylib kind could be extended with support for symbol versioning in the future. This implementation is very experimental and I have not tested it very well. I have tested it for a toy example and the lz4-sys crate, where it was able to successfully link a binary despite not having a corresponding library at build-time.
97 lines
3.6 KiB
Rust
97 lines
3.6 KiB
Rust
use run_make_support::{Rustc, diff, regex, rustc};
|
|
|
|
fn run_rustc() -> Rustc {
|
|
let mut rustc = rustc();
|
|
rustc
|
|
.arg("main.rs")
|
|
// NOTE: `link-self-contained` can vary depending on config.toml.
|
|
// Make sure we use a consistent value.
|
|
.arg("-Clink-self-contained=-linker")
|
|
.arg("-Zunstable-options")
|
|
.arg("-Wlinker-messages")
|
|
.output("main")
|
|
.linker("./fake-linker");
|
|
if run_make_support::target() == "x86_64-unknown-linux-gnu" {
|
|
// The value of `rust.lld` is different between CI and locally. Override it explicitly.
|
|
rustc.arg("-Clinker-flavor=gnu-cc");
|
|
}
|
|
rustc
|
|
}
|
|
|
|
fn main() {
|
|
// first, compile our linker
|
|
rustc().arg("fake-linker.rs").output("fake-linker").run();
|
|
|
|
// Run rustc with our fake linker, and make sure it shows warnings
|
|
let warnings = run_rustc().link_arg("run_make_warn").run();
|
|
warnings.assert_stderr_contains("warning: linker stderr: bar");
|
|
|
|
// Make sure it shows stdout
|
|
run_rustc()
|
|
.link_arg("run_make_info")
|
|
.run()
|
|
.assert_stderr_contains("warning: linker stdout: foo");
|
|
|
|
// Make sure we short-circuit this new path if the linker exits with an error
|
|
// (so the diagnostic is less verbose)
|
|
run_rustc().link_arg("run_make_error").run_fail().assert_stderr_contains("note: error: baz");
|
|
|
|
// Make sure we don't show the linker args unless `--verbose` is passed
|
|
let out = run_rustc().link_arg("run_make_error").verbose().run_fail();
|
|
out.assert_stderr_contains_regex("fake-linker.*run_make_error")
|
|
.assert_stderr_not_contains("object files omitted")
|
|
.assert_stderr_contains(r".rcgu.o")
|
|
.assert_stderr_contains_regex(r"lib(/|\\\\)libstd");
|
|
|
|
let out = run_rustc().link_arg("run_make_error").run_fail();
|
|
out.assert_stderr_contains("fake-linker")
|
|
.assert_stderr_contains("object files omitted")
|
|
.assert_stderr_contains_regex(r"\{")
|
|
.assert_stderr_not_contains(r".rcgu.o")
|
|
.assert_stderr_not_contains_regex(r"lib(/|\\\\)libstd");
|
|
|
|
// FIXME: we should have a version of this for mac and windows
|
|
if run_make_support::target() == "x86_64-unknown-linux-gnu" {
|
|
diff()
|
|
.expected_file("short-error.txt")
|
|
.actual_text("(linker error)", out.stderr())
|
|
.normalize(r#"/rustc[^/]*/"#, "/rustc/")
|
|
.normalize(
|
|
regex::escape(run_make_support::build_root().to_str().unwrap()),
|
|
"/build-root",
|
|
)
|
|
.normalize(r#""[^"]*\/symbols.o""#, "\"/symbols.o\"")
|
|
.normalize(r#""[^"]*\/raw-dylibs""#, "\"/raw-dylibs\"")
|
|
.run();
|
|
}
|
|
|
|
// Make sure we show linker warnings even across `-Z no-link`
|
|
rustc()
|
|
.arg("-Zno-link")
|
|
.input("-")
|
|
.stdin_buf("#![deny(linker_messages)] \n fn main() {}")
|
|
.run()
|
|
.assert_stderr_equals("");
|
|
rustc()
|
|
.arg("-Zlink-only")
|
|
.arg("rust_out.rlink")
|
|
.linker("./fake-linker")
|
|
.link_arg("run_make_warn")
|
|
.run_fail()
|
|
// NOTE: the error message here is quite bad (we don't have a source
|
|
// span, but still try to print the lint source). But `-Z link-only` is
|
|
// unstable and this still shows the linker warning itself so this is
|
|
// probably good enough.
|
|
.assert_stderr_contains("linker stderr: bar");
|
|
|
|
// Same thing, but with json output.
|
|
rustc()
|
|
.error_format("json")
|
|
.arg("-Zlink-only")
|
|
.arg("rust_out.rlink")
|
|
.linker("./fake-linker")
|
|
.link_arg("run_make_warn")
|
|
.run_fail()
|
|
.assert_stderr_contains(r#""$message_type":"diagnostic""#);
|
|
}
|