Skip to content

Commit

Permalink
Always set the deployment target when building std
Browse files Browse the repository at this point in the history
  • Loading branch information
madsmtm committed Nov 16, 2024
1 parent 8729ba3 commit 2294b05
Show file tree
Hide file tree
Showing 3 changed files with 73 additions and 3 deletions.
51 changes: 49 additions & 2 deletions src/bootstrap/src/core/build_steps/compile.rs
Original file line number Diff line number Diff line change
Expand Up @@ -462,8 +462,55 @@ fn compiler_rt_for_profiler(builder: &Builder<'_>) -> PathBuf {
/// Configure cargo to compile the standard library, adding appropriate env vars
/// and such.
pub fn std_cargo(builder: &Builder<'_>, target: TargetSelection, stage: u32, cargo: &mut Cargo) {
if let Some(target) = env::var_os("MACOSX_STD_DEPLOYMENT_TARGET") {
cargo.env("MACOSX_DEPLOYMENT_TARGET", target);
// rustc already ensures that it builds with the minimum deployment
// target, so ideally we shouldn't need to do anything here.
//
// However, `cc` currently defaults to a higher version for backwards
// compatibility, which means that compiler-rt, which is built via
// compiler-builtins' build script, gets built with a higher deployment
// target. This in turn causes warnings while linking, and is generally
// a compatibility hazard.
//
// So, at least until https://github.com/rust-lang/cc-rs/issues/1171, or
// perhaps https://github.com/rust-lang/cargo/issues/13115 is resolved, we
// explicitly set the deployment target environment variables to avoid
// this issue.
//
// This place also serves as an extension point if we ever wanted to raise
// rustc's default deployment target while keeping the prebuilt `std` at
// a lower version, so it's kinda nice to have in any case.
if target.contains("apple") && !builder.config.dry_run() {
// Query rustc for the deployment target.
let mut cmd = command(builder.rustc(cargo.compiler()));
cmd.arg("--target").arg(target.rustc_target_arg());
cmd.arg("--print=deployment-target");
let output = cmd.run_capture_stdout(builder).stdout();

let value = output.split('=').last().unwrap().trim();

// FIXME: Simplify after https://github.com/rust-lang/rust/pull/133041
let env_var = if target.contains("apple-darwin") {
"MACOSX_DEPLOYMENT_TARGET"
} else if target.contains("apple-ios") {
"IPHONEOS_DEPLOYMENT_TARGET"
} else if target.contains("apple-tvos") {
"TVOS_DEPLOYMENT_TARGET"
} else if target.contains("apple-watchos") {
"WATCHOS_DEPLOYMENT_TARGET"
} else if target.contains("apple-visionos") {
"XROS_DEPLOYMENT_TARGET"
} else {
panic!("unknown target OS for apple target");
};

// Unconditionally set the env var (if it was set in the environment
// already, rustc should've picked that up).
cargo.env(env_var, value);

// Allow CI to override the deployment target for `std`.
if let Some(target) = env::var_os("MACOSX_STD_DEPLOYMENT_TARGET") {
cargo.env("MACOSX_DEPLOYMENT_TARGET", target);
}
}

// Paths needed by `library/profiler_builtins/build.rs`.
Expand Down
4 changes: 4 additions & 0 deletions src/bootstrap/src/core/builder/cargo.rs
Original file line number Diff line number Diff line change
Expand Up @@ -122,6 +122,10 @@ impl Cargo {
cargo
}

pub fn compiler(&self) -> Compiler {
self.compiler
}

pub fn into_cmd(self) -> BootstrapCommand {
self.into()
}
Expand Down
21 changes: 20 additions & 1 deletion tests/run-make/apple-deployment-target/rmake.rs
Original file line number Diff line number Diff line change
Expand Up @@ -7,7 +7,11 @@
//@ only-apple

use run_make_support::{apple_os, cmd, run_in_tmpdir, rustc, target};
use std::collections::HashSet;

use run_make_support::{
apple_os, cmd, has_extension, path, regex, run_in_tmpdir, rustc, shallow_find_files, target,
};

/// Run vtool to check the `minos` field in LC_BUILD_VERSION.
///
Expand Down Expand Up @@ -156,4 +160,19 @@ fn main() {
rustc().env_remove(env_var).run();
minos("foo.o", default_version);
});

// Test that all binaries in rlibs produced by `rustc` have the same version.
let sysroot = rustc().print("sysroot").run().stdout_utf8();
let target_sysroot = path(sysroot.trim()).join("lib/rustlib").join(target()).join("lib");
let rlibs = shallow_find_files(&target_sysroot, |path| has_extension(path, "rlib"));

let output = cmd("otool").arg("-l").args(rlibs).run().stdout_utf8();
let re = regex::Regex::new(r"(minos|version) ([0-9.]*)").unwrap();
let mut versions = HashSet::new();
for (_, [_, version]) in re.captures_iter(&output).map(|c| c.extract()) {
versions.insert(version);
}
if versions.len() != 1 {
panic!("std rlibs contained multiple different deployment target versions: {versions:?}");
}
}

0 comments on commit 2294b05

Please sign in to comment.