diff options
| author | Andrew Kelley <andrew@ziglang.org> | 2023-08-02 16:36:58 -0700 |
|---|---|---|
| committer | Andrew Kelley <andrew@ziglang.org> | 2023-08-03 09:52:15 -0700 |
| commit | c012f5d55dc59f80a3cd4fce22443a39fcbb67e9 (patch) | |
| tree | a231342c69ad42700a726c70eef653fe739e944d /lib | |
| parent | e582a3642bb001a7a19ecb72b128f45ab7fc08aa (diff) | |
| download | zig-c012f5d55dc59f80a3cd4fce22443a39fcbb67e9.tar.gz zig-c012f5d55dc59f80a3cd4fce22443a39fcbb67e9.zip | |
std.zig.system.darwin.isSdkInstalled: fix implementation
* don't assert that the child process doesn't crash
* don't give a false negative on warnings printed to stderr
Also fix getSdk from the same file in the same way
Diffstat (limited to 'lib')
| -rw-r--r-- | lib/std/zig/system/darwin.zig | 44 |
1 files changed, 24 insertions, 20 deletions
diff --git a/lib/std/zig/system/darwin.zig b/lib/std/zig/system/darwin.zig index 3857d4f46c..992c0d814d 100644 --- a/lib/std/zig/system/darwin.zig +++ b/lib/std/zig/system/darwin.zig @@ -8,27 +8,33 @@ pub const macos = @import("darwin/macos.zig"); /// Check if SDK is installed on Darwin without triggering CLT installation popup window. /// Note: simply invoking `xcrun` will inevitably trigger the CLT installation popup. -/// Therefore, we resort to the same tool used by Homebrew, namely, invoking `xcode-select --print-path` -/// and checking if the status is nonzero or the returned string in nonempty. -/// https://github.com/Homebrew/brew/blob/e119bdc571dcb000305411bc1e26678b132afb98/Library/Homebrew/brew.sh#L630 +/// Therefore, we resort to invoking `xcode-select --print-path` and checking +/// if the status is nonzero. +/// stderr from xcode-select is ignored. +/// If error.OutOfMemory occurs in Allocator, this function returns null. pub fn isSdkInstalled(allocator: Allocator) bool { - const argv = &[_][]const u8{ "/usr/bin/xcode-select", "--print-path" }; - const result = std.ChildProcess.exec(.{ .allocator = allocator, .argv = argv }) catch return false; + const result = std.process.Child.exec(.{ + .allocator = allocator, + .argv = &.{ "/usr/bin/xcode-select", "--print-path" }, + }) catch return false; + defer { allocator.free(result.stderr); allocator.free(result.stdout); } - if (result.stderr.len != 0 or result.term.Exited != 0) { - // We don't actually care if there were errors as this is best-effort check anyhow. - return false; - } - return result.stdout.len > 0; + + return switch (result.term) { + .Exited => |code| if (code == 0) result.stdout.len > 0 else false, + else => false, + }; } /// Detect SDK on Darwin. /// Calls `xcrun --sdk <target_sdk> --show-sdk-path` which fetches the path to the SDK sysroot (if any). /// Subsequently calls `xcrun --sdk <target_sdk> --show-sdk-version` which fetches version of the SDK. /// The caller needs to deinit the resulting struct. +/// stderr from xcrun is ignored. +/// If error.OutOfMemory occurs in Allocator, this function returns null. pub fn getSdk(allocator: Allocator, target: Target) ?Sdk { const is_simulator_abi = target.abi == .simulator; const sdk = switch (target.os.tag) { @@ -40,30 +46,28 @@ pub fn getSdk(allocator: Allocator, target: Target) ?Sdk { }; const path = path: { const argv = &[_][]const u8{ "/usr/bin/xcrun", "--sdk", sdk, "--show-sdk-path" }; - const result = std.ChildProcess.exec(.{ .allocator = allocator, .argv = argv }) catch return null; + const result = std.process.Child.exec(.{ .allocator = allocator, .argv = argv }) catch return null; defer { allocator.free(result.stderr); allocator.free(result.stdout); } - if (result.stderr.len != 0 or result.term.Exited != 0) { - // We don't actually care if there were errors as this is best-effort check anyhow - // and in the worst case the user can specify the sysroot manually. - return null; + switch (result.term) { + .Exited => |code| if (code != 0) return null, + else => return null, } const path = allocator.dupe(u8, mem.trimRight(u8, result.stdout, "\r\n")) catch return null; break :path path; }; const version = version: { const argv = &[_][]const u8{ "/usr/bin/xcrun", "--sdk", sdk, "--show-sdk-version" }; - const result = std.ChildProcess.exec(.{ .allocator = allocator, .argv = argv }) catch return null; + const result = std.process.Child.exec(.{ .allocator = allocator, .argv = argv }) catch return null; defer { allocator.free(result.stderr); allocator.free(result.stdout); } - if (result.stderr.len != 0 or result.term.Exited != 0) { - // We don't actually care if there were errors as this is best-effort check anyhow - // and in the worst case the user can specify the sysroot manually. - return null; + switch (result.term) { + .Exited => |code| if (code != 0) return null, + else => return null, } const raw_version = mem.trimRight(u8, result.stdout, "\r\n"); const version = parseSdkVersion(raw_version) orelse Version{ |
