Skip to content

Commit 757a65b

Browse files
committed
Auto merge of #88250 - rusticstuff:macos-lld, r=nagisa
Make `-Z gcc-ld=lld` work for Apple targets `-Z gcc-ld=lld` was introduced in #85961. It does not work on Macos because lld needs be either named `ld64` or passed `-flavor darwin` as the first two arguments in order to select the Mach-O flavor. Rust invokes cc (=clang) on Macos for linking which calls `ld` as linker binary and not `ld64`, so just creating an `ld64` binary and modifying the search path with `-B` does not work. In order to solve this patch does: * Set the `lld_flavor` for all Apple-derived targets to `LldFlavor::Ld64`. As far as I can see this actually works towards fixing `-Xlinker=rust-lld` as all those targets use the Mach-O object format. * Copy/hardlink rust-lld to the gcc-ld subdirectory as ld64 next to ld. * If `-Z gcc-ld=lld` is used and the target lld flavor is Ld64 add `-fuse-ld=/path/to/ld64` to the linker invocation. Fixes #86945.
2 parents 3a21a5b + 0f7702e commit 757a65b

File tree

4 files changed

+41
-15
lines changed

4 files changed

+41
-15
lines changed

compiler/rustc_codegen_ssa/src/back/link.rs

+33-14
Original file line numberDiff line numberDiff line change
@@ -2482,20 +2482,39 @@ fn add_gcc_ld_path(cmd: &mut dyn Linker, sess: &Session, flavor: LinkerFlavor) {
24822482
if let LinkerFlavor::Gcc = flavor {
24832483
match ld_impl {
24842484
LdImpl::Lld => {
2485-
let tools_path =
2486-
sess.host_filesearch(PathKind::All).get_tools_search_paths(false);
2487-
let lld_path = tools_path
2488-
.into_iter()
2489-
.map(|p| p.join("gcc-ld"))
2490-
.find(|p| {
2491-
p.join(if sess.host.is_like_windows { "ld.exe" } else { "ld" }).exists()
2492-
})
2493-
.unwrap_or_else(|| sess.fatal("rust-lld (as ld) not found"));
2494-
cmd.cmd().arg({
2495-
let mut arg = OsString::from("-B");
2496-
arg.push(lld_path);
2497-
arg
2498-
});
2485+
if sess.target.lld_flavor == LldFlavor::Ld64 {
2486+
let tools_path =
2487+
sess.host_filesearch(PathKind::All).get_tools_search_paths(false);
2488+
let ld64_exe = tools_path
2489+
.into_iter()
2490+
.map(|p| p.join("gcc-ld"))
2491+
.map(|p| {
2492+
p.join(if sess.host.is_like_windows { "ld64.exe" } else { "ld64" })
2493+
})
2494+
.find(|p| p.exists())
2495+
.unwrap_or_else(|| sess.fatal("rust-lld (as ld64) not found"));
2496+
cmd.cmd().arg({
2497+
let mut arg = OsString::from("-fuse-ld=");
2498+
arg.push(ld64_exe);
2499+
arg
2500+
});
2501+
} else {
2502+
let tools_path =
2503+
sess.host_filesearch(PathKind::All).get_tools_search_paths(false);
2504+
let lld_path = tools_path
2505+
.into_iter()
2506+
.map(|p| p.join("gcc-ld"))
2507+
.find(|p| {
2508+
p.join(if sess.host.is_like_windows { "ld.exe" } else { "ld" })
2509+
.exists()
2510+
})
2511+
.unwrap_or_else(|| sess.fatal("rust-lld (as ld) not found"));
2512+
cmd.cmd().arg({
2513+
let mut arg = OsString::from("-B");
2514+
arg.push(lld_path);
2515+
arg
2516+
});
2517+
}
24992518
}
25002519
}
25012520
} else {

compiler/rustc_target/src/spec/apple_base.rs

+2-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
use std::env;
22

3-
use crate::spec::{FramePointer, SplitDebuginfo, TargetOptions};
3+
use crate::spec::{FramePointer, LldFlavor, SplitDebuginfo, TargetOptions};
44

55
pub fn opts(os: &str) -> TargetOptions {
66
// ELF TLS is only available in macOS 10.7+. If you try to compile for 10.6
@@ -35,6 +35,7 @@ pub fn opts(os: &str) -> TargetOptions {
3535
abi_return_struct_as_int: true,
3636
emit_debug_gdb_scripts: false,
3737
eh_frame_header: false,
38+
lld_flavor: LldFlavor::Ld64,
3839

3940
// The historical default for macOS targets is to run `dsymutil` which
4041
// generates a packed version of debuginfo split from the main file.

src/bootstrap/compile.rs

+4
Original file line numberDiff line numberDiff line change
@@ -1133,6 +1133,10 @@ impl Step for Assemble {
11331133
&lld_install.join("bin").join(&src_exe),
11341134
&gcc_ld_dir.join(exe("ld", target_compiler.host)),
11351135
);
1136+
builder.copy(
1137+
&lld_install.join("bin").join(&src_exe),
1138+
&gcc_ld_dir.join(exe("ld64", target_compiler.host)),
1139+
);
11361140
}
11371141

11381142
// Similarly, copy `llvm-dwp` into libdir for Split DWARF. Only copy it when the LLVM

src/bootstrap/dist.rs

+2
Original file line numberDiff line numberDiff line change
@@ -412,6 +412,8 @@ impl Step for Rustc {
412412
let gcc_lld_dir = dst_dir.join("gcc-ld");
413413
t!(fs::create_dir(&gcc_lld_dir));
414414
builder.copy(&src_dir.join(&rust_lld), &gcc_lld_dir.join(exe("ld", compiler.host)));
415+
builder
416+
.copy(&src_dir.join(&rust_lld), &gcc_lld_dir.join(exe("ld64", compiler.host)));
415417
}
416418

417419
// Copy over llvm-dwp if it's there

0 commit comments

Comments
 (0)