Skip to content

Commit 97e382d

Browse files
committed
Always attempt to get SDK root, and pass it to Clang via env var
The exact reasoning why we do not always pass the SDK root with `-isysroot` to `cc` when linking on macOS eludes me (the git history dead ends in rust-lang#100286), but I suspect it's because we want to support `cc`s which do not support this option. So instead, we pass the SDK root via the environment variable SDKROOT. This way, compiler drivers that support setting the SDK root (such as Clang and GCC) can use it, while compiler drivers that don't (presumably because they figure out the SDK root in some other way) can just ignore it. This fixes rust-lang#80817 (by always passing the SDK root, even when linking with cc on macOS).
1 parent f91a3b8 commit 97e382d

File tree

2 files changed

+42
-48
lines changed
  • compiler
    • rustc_codegen_ssa/src/back
    • rustc_target/src/spec/base/apple

2 files changed

+42
-48
lines changed

compiler/rustc_codegen_ssa/src/back/link.rs

+40-27
Original file line numberDiff line numberDiff line change
@@ -3124,47 +3124,60 @@ fn add_apple_link_args(cmd: &mut dyn Linker, sess: &Session, flavor: LinkerFlavo
31243124
}
31253125

31263126
fn add_apple_sdk(cmd: &mut dyn Linker, sess: &Session, flavor: LinkerFlavor) -> Option<PathBuf> {
3127-
let os = &sess.target.os;
3128-
if sess.target.vendor != "apple"
3129-
|| !matches!(os.as_ref(), "ios" | "tvos" | "watchos" | "visionos" | "macos")
3130-
|| !matches!(flavor, LinkerFlavor::Darwin(..))
3131-
{
3127+
if !sess.target.is_like_osx {
31323128
return None;
31333129
}
3134-
3135-
if os == "macos" && !matches!(flavor, LinkerFlavor::Darwin(Cc::No, _)) {
3130+
let LinkerFlavor::Darwin(cc, _) = flavor else {
31363131
return None;
3137-
}
3132+
};
31383133

31393134
let sdk_name = rustc_target::spec::apple_sdk_name(&sess.target);
31403135

3141-
let sdk_root = match get_apple_sdk_root(sdk_name) {
3136+
let sdkroot = match get_apple_sdk_root(sdk_name) {
31423137
Ok(s) => s,
31433138
Err(e) => {
3144-
sess.dcx().emit_err(e);
3139+
// If cross compiling from non-macOS, the user might be using something like `zig cc`.
3140+
//
3141+
// In that case, we shouldn't error when the SDK is missing, though we still warn.
3142+
if cfg!(target_os = "macos") {
3143+
sess.dcx().emit_err(e);
3144+
} else {
3145+
sess.dcx().emit_warn(e);
3146+
}
31453147
return None;
31463148
}
31473149
};
31483150

3149-
match flavor {
3150-
LinkerFlavor::Darwin(Cc::Yes, _) => {
3151-
// Use `-isysroot` instead of `--sysroot`, as only the former
3152-
// makes Clang treat it as a platform SDK.
3153-
//
3154-
// This is admittedly a bit strange, as on most targets
3155-
// `-isysroot` only applies to include header files, but on Apple
3156-
// targets this also applies to libraries and frameworks.
3157-
cmd.cc_arg("-isysroot");
3158-
cmd.cc_arg(&sdk_root);
3159-
}
3160-
LinkerFlavor::Darwin(Cc::No, _) => {
3161-
cmd.link_arg("-syslibroot");
3162-
cmd.link_arg(&sdk_root);
3163-
}
3164-
_ => unreachable!(),
3151+
if cc == Cc::Yes {
3152+
// To pass the SDK root to `cc`, we have a few options:
3153+
// 1. `--sysroot` flag.
3154+
// 2. `-isysroot` flag.
3155+
// 3. `SDKROOT` environment variable.
3156+
//
3157+
// `--sysroot` isn't actually enough to get Clang to treat it as a platform SDK, you need to
3158+
// specify `-isysroot` - this is admittedly a bit strange, as on most targets `-isysroot`
3159+
// only applies to include header files, but on Apple targets it also applies to libraries
3160+
// and frameworks.
3161+
//
3162+
// Now, while the `-isysroot` flag is pretty well supported (both Clang and GCC implements
3163+
// the desired behaviour), it may not be understood by any `cc`'s that the user might want
3164+
// to use.
3165+
//
3166+
// So to better support such use-cases, we pass the SDK root in the standard environment
3167+
// variable instead. This is also supported by GCC since 2019:
3168+
// <https://gcc.gnu.org/bugzilla/show_bug.cgi?id=87243>
3169+
//
3170+
// This also works better with the trampoline `/usr/bin/cc` which calls `xcrun cc`
3171+
// internally, since the presence of `SDKROOT` means it won't have to look up the SDK root
3172+
// itself.
3173+
cmd.cmd().env("SDKROOT", &sdkroot);
3174+
} else {
3175+
// For `ld64`, we use the `-syslibroot` parameter (and there are no other options).
3176+
cmd.link_arg("-syslibroot");
3177+
cmd.link_arg(&sdkroot);
31653178
}
31663179

3167-
Some(sdk_root)
3180+
Some(sdkroot)
31683181
}
31693182

31703183
fn get_apple_sdk_root(sdk_name: &'static str) -> Result<PathBuf, errors::AppleSdkError> {

compiler/rustc_target/src/spec/base/apple/mod.rs

+2-21
Original file line numberDiff line numberDiff line change
@@ -309,29 +309,10 @@ fn link_env_remove(os: &'static str) -> StaticCow<[StaticCow<str>]> {
309309
// that's only applicable to cross-OS compilation. Always leave anything for the
310310
// host OS alone though.
311311
if os == "macos" {
312-
let mut env_remove = Vec::with_capacity(2);
313-
// Remove the `SDKROOT` environment variable if it's clearly set for the wrong platform, which
314-
// may occur when we're linking a custom build script while targeting iOS for example.
315-
if let Ok(sdkroot) = env::var("SDKROOT") {
316-
if sdkroot.contains("iPhoneOS.platform")
317-
|| sdkroot.contains("iPhoneSimulator.platform")
318-
|| sdkroot.contains("AppleTVOS.platform")
319-
|| sdkroot.contains("AppleTVSimulator.platform")
320-
|| sdkroot.contains("WatchOS.platform")
321-
|| sdkroot.contains("WatchSimulator.platform")
322-
|| sdkroot.contains("XROS.platform")
323-
|| sdkroot.contains("XRSimulator.platform")
324-
{
325-
env_remove.push("SDKROOT".into())
326-
}
327-
}
328-
// Additionally, `IPHONEOS_DEPLOYMENT_TARGET` must not be set when using the Xcode linker at
312+
// `IPHONEOS_DEPLOYMENT_TARGET` must not be set when using the Xcode linker at
329313
// "/Applications/Xcode.app/Contents/Developer/Toolchains/XcodeDefault.xctoolchain/usr/bin/ld",
330314
// although this is apparently ignored when using the linker at "/usr/bin/ld".
331-
env_remove.push("IPHONEOS_DEPLOYMENT_TARGET".into());
332-
env_remove.push("TVOS_DEPLOYMENT_TARGET".into());
333-
env_remove.push("XROS_DEPLOYMENT_TARGET".into());
334-
env_remove.into()
315+
cvs!["IPHONEOS_DEPLOYMENT_TARGET", "TVOS_DEPLOYMENT_TARGET", "XROS_DEPLOYMENT_TARGET"]
335316
} else {
336317
// Otherwise if cross-compiling for a different OS/SDK (including Mac Catalyst), remove any part
337318
// of the linking environment that's wrong and reversed.

0 commit comments

Comments
 (0)