Skip to content

Commit 48c1607

Browse files
committed
Introduce LldMode and generalize parsing of use-lld
1 parent d9f9e67 commit 48c1607

File tree

5 files changed

+99
-12
lines changed

5 files changed

+99
-12
lines changed

src/bootstrap/src/core/build_steps/compile.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -870,7 +870,7 @@ impl Step for Rustc {
870870
// is already on by default in MSVC optimized builds, which is interpreted as --icf=all:
871871
// https://github.com/llvm/llvm-project/blob/3329cec2f79185bafd678f310fafadba2a8c76d2/lld/COFF/Driver.cpp#L1746
872872
// https://github.com/rust-lang/rust/blob/f22819bcce4abaff7d1246a56eec493418f9f4ee/compiler/rustc_codegen_ssa/src/back/linker.rs#L827
873-
if builder.config.use_lld && !compiler.host.is_msvc() {
873+
if builder.config.lld_mode.is_used() && !compiler.host.is_msvc() {
874874
cargo.rustflag("-Clink-args=-Wl,--icf=all");
875875
}
876876

src/bootstrap/src/core/config/config.rs

+74-3
Original file line numberDiff line numberDiff line change
@@ -105,6 +105,39 @@ impl Display for DebuginfoLevel {
105105
}
106106
}
107107

108+
/// LLD in bootstrap works like this:
109+
/// - Self-contained lld: use `rust-lld` from the compiler's sysroot
110+
/// - External: use an external `lld` binary
111+
///
112+
/// It is configured depending on the target:
113+
/// 1) Everything except MSVC
114+
/// - Self-contained: -Clinker-flavor=gnu-lld-cc -Clink-self-contained=+linker
115+
/// - External: -Clinker-flavor=gnu-lld-cc
116+
/// 2) MSVC
117+
/// - Self-contained: -Clinker=<path to rust-lld>
118+
/// - External: -Clinker=lld
119+
#[derive(Default, Clone)]
120+
pub enum LldMode {
121+
/// Do not use LLD
122+
#[default]
123+
Unused,
124+
/// Use `rust-lld` from the compiler's sysroot
125+
SelfContained,
126+
/// Use an externally provided `lld` binary.
127+
/// Note that the linker name cannot be overridden, the binary has to be named `lld` and it has
128+
/// to be in $PATH.
129+
External,
130+
}
131+
132+
impl LldMode {
133+
pub fn is_used(&self) -> bool {
134+
match self {
135+
LldMode::SelfContained | LldMode::External => true,
136+
LldMode::Unused => false,
137+
}
138+
}
139+
}
140+
108141
/// Global configuration for the entire build and/or bootstrap.
109142
///
110143
/// This structure is parsed from `config.toml`, and some of the fields are inferred from `git` or build-time parameters.
@@ -199,7 +232,7 @@ pub struct Config {
199232
pub llvm_from_ci: bool,
200233
pub llvm_build_config: HashMap<String, String>,
201234

202-
pub use_lld: bool,
235+
pub lld_mode: LldMode,
203236
pub lld_enabled: bool,
204237
pub llvm_tools_enabled: bool,
205238

@@ -981,6 +1014,44 @@ enum StringOrInt<'a> {
9811014
String(&'a str),
9821015
Int(i64),
9831016
}
1017+
1018+
impl<'de> Deserialize<'de> for LldMode {
1019+
fn deserialize<D>(deserializer: D) -> Result<Self, D::Error>
1020+
where
1021+
D: Deserializer<'de>,
1022+
{
1023+
struct LldModeVisitor;
1024+
1025+
impl<'de> serde::de::Visitor<'de> for LldModeVisitor {
1026+
type Value = LldMode;
1027+
1028+
fn expecting(&self, formatter: &mut fmt::Formatter<'_>) -> fmt::Result {
1029+
formatter.write_str("one of true, 'self-contained' or 'external'")
1030+
}
1031+
1032+
fn visit_bool<E>(self, v: bool) -> Result<Self::Value, E>
1033+
where
1034+
E: serde::de::Error,
1035+
{
1036+
Ok(if v { LldMode::External } else { LldMode::Unused })
1037+
}
1038+
1039+
fn visit_str<E>(self, v: &str) -> Result<Self::Value, E>
1040+
where
1041+
E: serde::de::Error,
1042+
{
1043+
match v {
1044+
"external" => Ok(LldMode::External),
1045+
"self-contained" => Ok(LldMode::SelfContained),
1046+
_ => Err(E::custom("unknown mode {v}")),
1047+
}
1048+
}
1049+
}
1050+
1051+
deserializer.deserialize_any(LldModeVisitor)
1052+
}
1053+
}
1054+
9841055
define_config! {
9851056
/// TOML representation of how the Rust build is configured.
9861057
struct Rust {
@@ -1018,7 +1089,7 @@ define_config! {
10181089
save_toolstates: Option<String> = "save-toolstates",
10191090
codegen_backends: Option<Vec<String>> = "codegen-backends",
10201091
lld: Option<bool> = "lld",
1021-
use_lld: Option<bool> = "use-lld",
1092+
lld_mode: Option<LldMode> = "use-lld",
10221093
llvm_tools: Option<bool> = "llvm-tools",
10231094
deny_warnings: Option<bool> = "deny-warnings",
10241095
backtrace_on_ice: Option<bool> = "backtrace-on-ice",
@@ -1446,7 +1517,7 @@ impl Config {
14461517
if let Some(true) = rust.incremental {
14471518
config.incremental = true;
14481519
}
1449-
set(&mut config.use_lld, rust.use_lld);
1520+
set(&mut config.lld_mode, rust.lld_mode);
14501521
set(&mut config.lld_enabled, rust.lld);
14511522
set(&mut config.llvm_tools_enabled, rust.llvm_tools);
14521523
config.rustc_parallel = rust

src/bootstrap/src/lib.rs

+13-6
Original file line numberDiff line numberDiff line change
@@ -37,7 +37,7 @@ use utils::channel::GitInfo;
3737

3838
use crate::core::builder;
3939
use crate::core::builder::Kind;
40-
use crate::core::config::flags;
40+
use crate::core::config::{flags, LldMode};
4141
use crate::core::config::{DryRun, Target};
4242
use crate::core::config::{LlvmLibunwind, TargetSelection};
4343
use crate::utils::cache::{Interned, INTERNER};
@@ -1258,17 +1258,24 @@ impl Build {
12581258
&& !target.is_msvc()
12591259
{
12601260
Some(self.cc(target))
1261-
} else if self.config.use_lld && !self.is_fuse_ld_lld(target) && self.build == target {
1262-
Some(self.initial_lld.clone())
1261+
} else if self.config.lld_mode.is_used()
1262+
&& self.is_lld_direct_linker(target)
1263+
&& self.build == target
1264+
{
1265+
match self.config.lld_mode {
1266+
LldMode::SelfContained => Some(self.initial_lld.clone()),
1267+
LldMode::External => Some("lld".into()),
1268+
LldMode::Unused => None,
1269+
}
12631270
} else {
12641271
None
12651272
}
12661273
}
12671274

1268-
// LLD is used through `-fuse-ld=lld` rather than directly.
1275+
// Is LLD configured directly through `-Clinker`?
12691276
// Only MSVC targets use LLD directly at the moment.
1270-
fn is_fuse_ld_lld(&self, target: TargetSelection) -> bool {
1271-
self.config.use_lld && !target.is_msvc()
1277+
fn is_lld_direct_linker(&self, target: TargetSelection) -> bool {
1278+
target.is_msvc()
12721279
}
12731280

12741281
/// Returns if this target should statically link the C runtime, if specified

src/bootstrap/src/tests/config.rs

+10-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
1-
use crate::core::config::TomlConfig;
21
use super::{Config, Flags};
2+
use crate::core::config::{LldMode, TomlConfig};
33

44
use clap::CommandFactory;
55
use serde::Deserialize;
@@ -217,3 +217,12 @@ fn verify_file_integrity() {
217217

218218
remove_file(tempfile).unwrap();
219219
}
220+
221+
#[test]
222+
fn rust_lld() {
223+
assert!(matches!(parse("").lld_mode, LldMode::Unused));
224+
assert!(matches!(parse("rust.use-lld = \"self-contained\"").lld_mode, LldMode::SelfContained));
225+
assert!(matches!(parse("rust.use-lld = \"external\"").lld_mode, LldMode::External));
226+
assert!(matches!(parse("rust.use-lld = true").lld_mode, LldMode::External));
227+
assert!(matches!(parse("rust.use-lld = false").lld_mode, LldMode::Unused));
228+
}

src/bootstrap/src/utils/helpers.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -499,7 +499,7 @@ pub fn linker_flags(
499499
lld_threads: LldThreads,
500500
) -> Vec<String> {
501501
let mut args = vec![];
502-
if builder.is_fuse_ld_lld(target) {
502+
if !builder.is_lld_direct_linker(target) && builder.config.lld_mode.is_used() {
503503
args.push(String::from("-Clink-arg=-fuse-ld=lld"));
504504

505505
if matches!(lld_threads, LldThreads::No) {

0 commit comments

Comments
 (0)