Skip to content

Commit 6fb0358

Browse files
committed
Auto merge of rust-lang#135707 - jyn514:linker-messages-2, r=bjorn3
Shorten linker output even more when `--verbose` is not present - Don't show environment variables. Seeing PATH is almost never useful, and it can be extremely long. - For .rlibs in the sysroot, replace crate hashes with a `"-*"` string. This will expand to the full crate name when pasted into the shell. - Move `.rlib` to outside the glob. - Abbreviate the sysroot path to `<sysroot>` wherever it appears in the arguments. This also adds an example of the linker output as a run-make test. Currently it only runs on x86_64-unknown-linux-gnu, because each platform has its own linker arguments. So that it's stable across machines, pass BUILD_ROOT as an argument through compiletest through to run-make tests. r? `@bjorn3` try-job: aarch64-apple
2 parents f7cc13a + c1b4ab0 commit 6fb0358

File tree

11 files changed

+118
-28
lines changed

11 files changed

+118
-28
lines changed

Cargo.lock

+1
Original file line numberDiff line numberDiff line change
@@ -3537,6 +3537,7 @@ dependencies = [
35373537
"ar_archive_writer",
35383538
"arrayvec",
35393539
"bitflags",
3540+
"bstr",
35403541
"cc",
35413542
"either",
35423543
"itertools",

compiler/rustc_codegen_ssa/Cargo.toml

+1
Original file line numberDiff line numberDiff line change
@@ -8,6 +8,7 @@ edition = "2021"
88
ar_archive_writer = "0.4.2"
99
arrayvec = { version = "0.7", default-features = false }
1010
bitflags = "2.4.1"
11+
bstr = "1.11.3"
1112
# Pinned so `cargo update` bumps don't cause breakage. Please also update the
1213
# `cc` in `rustc_llvm` if you update the `cc` here.
1314
cc = "=1.2.7"

compiler/rustc_codegen_ssa/src/back/command.rs

+16-1
Original file line numberDiff line numberDiff line change
@@ -13,6 +13,7 @@ pub(crate) struct Command {
1313
args: Vec<OsString>,
1414
env: Vec<(OsString, OsString)>,
1515
env_remove: Vec<OsString>,
16+
env_clear: bool,
1617
}
1718

1819
#[derive(Clone)]
@@ -36,7 +37,13 @@ impl Command {
3637
}
3738

3839
fn _new(program: Program) -> Command {
39-
Command { program, args: Vec::new(), env: Vec::new(), env_remove: Vec::new() }
40+
Command {
41+
program,
42+
args: Vec::new(),
43+
env: Vec::new(),
44+
env_remove: Vec::new(),
45+
env_clear: false,
46+
}
4047
}
4148

4249
pub(crate) fn arg<P: AsRef<OsStr>>(&mut self, arg: P) -> &mut Command {
@@ -79,6 +86,11 @@ impl Command {
7986
self
8087
}
8188

89+
pub(crate) fn env_clear(&mut self) -> &mut Command {
90+
self.env_clear = true;
91+
self
92+
}
93+
8294
fn _env_remove(&mut self, key: &OsStr) {
8395
self.env_remove.push(key.to_owned());
8496
}
@@ -106,6 +118,9 @@ impl Command {
106118
for k in &self.env_remove {
107119
ret.env_remove(k);
108120
}
121+
if self.env_clear {
122+
ret.env_clear();
123+
}
109124
ret
110125
}
111126

compiler/rustc_codegen_ssa/src/back/link.rs

+1
Original file line numberDiff line numberDiff line change
@@ -1007,6 +1007,7 @@ fn link_natively(
10071007
command: cmd,
10081008
escaped_output,
10091009
verbose: sess.opts.verbose,
1010+
sysroot_dir: sess.sysroot.clone(),
10101011
};
10111012
sess.dcx().emit_err(err);
10121013
// If MSVC's `link.exe` was expected but the return code

compiler/rustc_codegen_ssa/src/errors.rs

+47-15
Original file line numberDiff line numberDiff line change
@@ -351,6 +351,7 @@ pub(crate) struct LinkingFailed<'a> {
351351
pub command: Command,
352352
pub escaped_output: String,
353353
pub verbose: bool,
354+
pub sysroot_dir: PathBuf,
354355
}
355356

356357
impl<G: EmissionGuarantee> Diagnostic<'_, G> for LinkingFailed<'_> {
@@ -364,6 +365,8 @@ impl<G: EmissionGuarantee> Diagnostic<'_, G> for LinkingFailed<'_> {
364365
if self.verbose {
365366
diag.note(format!("{:?}", self.command));
366367
} else {
368+
self.command.env_clear();
369+
367370
enum ArgGroup {
368371
Regular(OsString),
369372
Objects(usize),
@@ -398,26 +401,55 @@ impl<G: EmissionGuarantee> Diagnostic<'_, G> for LinkingFailed<'_> {
398401
args.push(ArgGroup::Regular(arg));
399402
}
400403
}
401-
self.command.args(args.into_iter().map(|arg_group| match arg_group {
402-
ArgGroup::Regular(arg) => arg,
403-
ArgGroup::Objects(n) => OsString::from(format!("<{n} object files omitted>")),
404-
ArgGroup::Rlibs(dir, rlibs) => {
405-
let mut arg = dir.into_os_string();
406-
arg.push("/{");
407-
let mut first = true;
408-
for rlib in rlibs {
409-
if !first {
410-
arg.push(",");
404+
let crate_hash = regex::bytes::Regex::new(r"-[0-9a-f]+\.rlib$").unwrap();
405+
self.command.args(args.into_iter().map(|arg_group| {
406+
match arg_group {
407+
// SAFETY: we are only matching on ASCII, not any surrogate pairs, so any replacements we do will still be valid.
408+
ArgGroup::Regular(arg) => unsafe {
409+
use bstr::ByteSlice;
410+
OsString::from_encoded_bytes_unchecked(
411+
arg.as_encoded_bytes().replace(
412+
self.sysroot_dir.as_os_str().as_encoded_bytes(),
413+
b"<sysroot>",
414+
),
415+
)
416+
},
417+
ArgGroup::Objects(n) => OsString::from(format!("<{n} object files omitted>")),
418+
ArgGroup::Rlibs(mut dir, rlibs) => {
419+
let is_sysroot_dir = match dir.strip_prefix(&self.sysroot_dir) {
420+
Ok(short) => {
421+
dir = Path::new("<sysroot>").join(short);
422+
true
423+
}
424+
Err(_) => false,
425+
};
426+
let mut arg = dir.into_os_string();
427+
arg.push("/{");
428+
let mut first = true;
429+
for mut rlib in rlibs {
430+
if !first {
431+
arg.push(",");
432+
}
433+
first = false;
434+
if is_sysroot_dir {
435+
// SAFETY: Regex works one byte at a type, and our regex will not match surrogate pairs (because it only matches ascii).
436+
rlib = unsafe {
437+
OsString::from_encoded_bytes_unchecked(
438+
crate_hash
439+
.replace(rlib.as_encoded_bytes(), b"-*")
440+
.into_owned(),
441+
)
442+
};
443+
}
444+
arg.push(rlib);
411445
}
412-
first = false;
413-
arg.push(rlib);
446+
arg.push("}.rlib");
447+
arg
414448
}
415-
arg.push("}");
416-
arg
417449
}
418450
}));
419451

420-
diag.note(format!("{:?}", self.command));
452+
diag.note(format!("{:?}", self.command).trim_start_matches("env -i").to_owned());
421453
diag.note("some arguments are omitted. use `--verbose` to show all linker arguments");
422454
}
423455

src/tools/compiletest/src/runtest/run_make.rs

+2
Original file line numberDiff line numberDiff line change
@@ -414,6 +414,8 @@ impl TestCx<'_> {
414414
// Provide path to checkout root. This is the top-level directory containing
415415
// rust-lang/rust checkout.
416416
.env("SOURCE_ROOT", &source_root)
417+
// Path to the build directory. This is usually the same as `source_root.join("build").join("host")`.
418+
.env("BUILD_ROOT", &build_root)
417419
// Provide path to stage-corresponding rustc.
418420
.env("RUSTC", &self.config.rustc_path)
419421
// Provide the directory to libraries that are needed to run the *compiler*. This is not

src/tools/run-make-support/src/lib.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -90,7 +90,7 @@ pub use artifact_names::{
9090
/// Path-related helpers.
9191
pub use path_helpers::{
9292
cwd, filename_contains, filename_not_in_denylist, has_extension, has_prefix, has_suffix,
93-
not_contains, path, shallow_find_files, source_root,
93+
not_contains, path, shallow_find_files, build_root, source_root,
9494
};
9595

9696
/// Helpers for scoped test execution where certain properties are attempted to be maintained.

src/tools/run-make-support/src/path_helpers.rs

+6
Original file line numberDiff line numberDiff line change
@@ -34,6 +34,12 @@ pub fn source_root() -> PathBuf {
3434
env_var("SOURCE_ROOT").into()
3535
}
3636

37+
/// Path to the build directory root.
38+
#[must_use]
39+
pub fn build_root() -> PathBuf {
40+
env_var("BUILD_ROOT").into()
41+
}
42+
3743
/// Browse the directory `path` non-recursively and return all files which respect the parameters
3844
/// outlined by `closure`.
3945
#[track_caller]

src/tools/tidy/src/deps.rs

+1
Original file line numberDiff line numberDiff line change
@@ -253,6 +253,7 @@ const PERMITTED_RUSTC_DEPENDENCIES: &[&str] = &[
253253
"bitflags",
254254
"blake3",
255255
"block-buffer",
256+
"bstr",
256257
"byteorder", // via ruzstd in object in thorin-dwp
257258
"cc",
258259
"cfg-if",

tests/run-make/linker-warning/rmake.rs

+33-11
Original file line numberDiff line numberDiff line change
@@ -1,8 +1,19 @@
1-
use run_make_support::{Rustc, rustc};
1+
use run_make_support::{Rustc, diff, regex, rustc};
22

33
fn run_rustc() -> Rustc {
44
let mut rustc = rustc();
5-
rustc.arg("main.rs").output("main").linker("./fake-linker");
5+
rustc
6+
.arg("main.rs")
7+
// NOTE: `link-self-contained` can vary depending on config.toml.
8+
// Make sure we use a consistent value.
9+
.arg("-Clink-self-contained=-linker")
10+
.arg("-Zunstable-options")
11+
.output("main")
12+
.linker("./fake-linker");
13+
if run_make_support::target() == "x86_64-unknown-linux-gnu" {
14+
// The value of `rust.lld` is different between CI and locally. Override it explicitly.
15+
rustc.arg("-Clinker-flavor=gnu-cc");
16+
}
617
rustc
718
}
819

@@ -25,21 +36,32 @@ fn main() {
2536
run_rustc().link_arg("run_make_error").run_fail().assert_stderr_contains("note: error: baz");
2637

2738
// Make sure we don't show the linker args unless `--verbose` is passed
28-
run_rustc()
29-
.link_arg("run_make_error")
30-
.verbose()
31-
.run_fail()
32-
.assert_stderr_contains_regex("fake-linker.*run_make_error")
39+
let out = run_rustc().link_arg("run_make_error").verbose().run_fail();
40+
out.assert_stderr_contains_regex("fake-linker.*run_make_error")
3341
.assert_stderr_not_contains("object files omitted")
42+
.assert_stderr_contains(r".rcgu.o")
3443
.assert_stderr_contains_regex(r"lib(/|\\\\)libstd");
35-
run_rustc()
36-
.link_arg("run_make_error")
37-
.run_fail()
38-
.assert_stderr_contains("fake-linker")
44+
45+
let out = run_rustc().link_arg("run_make_error").run_fail();
46+
out.assert_stderr_contains("fake-linker")
3947
.assert_stderr_contains("object files omitted")
4048
.assert_stderr_contains_regex(r"\{")
49+
.assert_stderr_not_contains(r".rcgu.o")
4150
.assert_stderr_not_contains_regex(r"lib(/|\\\\)libstd");
4251

52+
// FIXME: we should have a version of this for mac and windows
53+
if run_make_support::target() == "x86_64-unknown-linux-gnu" {
54+
diff()
55+
.expected_file("short-error.txt")
56+
.actual_text("(linker error)", out.stderr())
57+
.normalize(r#"/rustc[^/]*/"#, "/rustc/")
58+
.normalize(
59+
regex::escape(run_make_support::build_root().to_str().unwrap()),
60+
"/build-root",
61+
)
62+
.run();
63+
}
64+
4365
// Make sure we show linker warnings even across `-Z no-link`
4466
rustc()
4567
.arg("-Zno-link")
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,9 @@
1+
error: linking with `./fake-linker` failed: exit status: 1
2+
|
3+
= note: "./fake-linker" "-m64" "/tmp/rustc/symbols.o" "<2 object files omitted>" "-Wl,--as-needed" "-Wl,-Bstatic" "<sysroot>/lib/rustlib/x86_64-unknown-linux-gnu/lib/{libstd-*,libpanic_unwind-*,libobject-*,libmemchr-*,libaddr2line-*,libgimli-*,librustc_demangle-*,libstd_detect-*,libhashbrown-*,librustc_std_workspace_alloc-*,libminiz_oxide-*,libadler2-*,libunwind-*,libcfg_if-*,liblibc-*,liballoc-*,librustc_std_workspace_core-*,libcore-*,libcompiler_builtins-*}.rlib" "-Wl,-Bdynamic" "-lgcc_s" "-lutil" "-lrt" "-lpthread" "-lm" "-ldl" "-lc" "-Wl,--eh-frame-hdr" "-Wl,-z,noexecstack" "-L" "/build-root/test/run-make/linker-warning/rmake_out" "-L" "<sysroot>/lib/rustlib/x86_64-unknown-linux-gnu/lib" "-o" "main" "-Wl,--gc-sections" "-pie" "-Wl,-z,relro,-z,now" "-nodefaultlibs" "run_make_error"
4+
= note: some arguments are omitted. use `--verbose` to show all linker arguments
5+
= note: error: baz
6+
7+
8+
error: aborting due to 1 previous error
9+

0 commit comments

Comments
 (0)