xcbuild: refactor

This reworks some of xcbuild logic to make it more compatible with
Apple’s SDK.

- Add a fake version of xcrun & xcode-select
- Cleanup platform generation. Clang does not like having 20 char
  hashes in sysroot so it is much easier to just build the parent
  directory for each runCommand. This is a little awkward but I have
  renamed everything with an added ‘s’ to make the distinction more clear.
- Cleaned up wrapper.nix in some different ways
- Reuse some versioning logic so that we don’t end up with two
  different versions of Xcode or SDK reported.
This commit is contained in:
Matthew Bauer 2018-07-06 18:49:38 -04:00
parent fec309aee4
commit 35b2c0edf2
5 changed files with 199 additions and 142 deletions

View File

@ -1,13 +1,13 @@
{ runCommand, lib, sdk, platformName, writeText }: { runCommand, lib, sdks, xcodePlatform, writeText }:
let let
inherit (lib.generators) toPlist; inherit (lib.generators) toPlist;
Info = { Info = {
CFBundleIdentifier = platformName; CFBundleIdentifier = "com.apple.platform.${lib.toLower xcodePlatform}";
Type = "Platform"; Type = "Platform";
Name = "macosx"; Name = lib.toLower xcodePlatform;
}; };
Version = { Version = {
@ -285,14 +285,18 @@ let
in in
runCommand "MacOSX.platform" {} '' runCommand "Platforms" {} ''
install -D ${writeText "Info.plist" (toPlist {} Info)} $out/Info.plist platform=$out/${xcodePlatform}.platform
install -D ${writeText "version.plist" (toPlist {} Version)} $out/version.plist
install -D ${writeText "Architectures.xcspec" (toPlist {} Architectures)} $out/Developer/Library/Xcode/Specifications/Architectures.xcspec
install -D ${writeText "PackageTypes.xcspec" (toPlist {} PackageTypes)} $out/Developer/Library/Xcode/Specifications/PackageTypes.xcspec
install -D ${writeText "ProductTypes.xcspec" (toPlist {} ProductTypes)} $out/Developer/Library/Xcode/Specifications/ProductTypes.xcspec
mkdir -p $out/Developer/SDKs/ install -D ${writeText "Info.plist" (toPlist {} Info)} $platform/Info.plist
cd $out/Developer/SDKs/ install -D ${writeText "version.plist" (toPlist {} Version)} $platform/version.plist
cp -r ${sdk} ${sdk.name} install -D ${writeText "Architectures.xcspec" (toPlist {} Architectures)} $platform/Developer/Library/Xcode/Specifications/Architectures.xcspec
install -D ${writeText "PackageTypes.xcspec" (toPlist {} PackageTypes)} $platform/Developer/Library/Xcode/Specifications/PackageTypes.xcspec
install -D ${writeText "ProductTypes.xcspec" (toPlist {} ProductTypes)} $platform/Developer/Library/Xcode/Specifications/ProductTypes.xcspec
# per-platform bins go here
mkdir -p $platform/usr/bin
mkdir -p $platform/Developer
ln -s ${sdks} $platform/Developer/SDKs
'' ''

View File

@ -1,11 +1,8 @@
{ runCommand, lib, toolchainName, sdkName, writeText }: { runCommand, lib, toolchainName, sdkName, writeText, version, xcodePlatform }:
let let
inherit (lib.generators) toPlist; inherit (lib.generators) toPlist;
# TODO: expose MACOSX_DEPLOYMENT_TARGET in nix so we can use it here.
version = "10.10";
SDKSettings = { SDKSettings = {
CanonicalName = sdkName; CanonicalName = sdkName;
DisplayName = sdkName; DisplayName = sdkName;
@ -21,9 +18,11 @@ let
}; };
in in
runCommand "MacOSX${version}.sdk" { runCommand "SDKs" {
inherit version; inherit version;
} '' } ''
install -D ${writeText "SDKSettings.plist" (toPlist {} SDKSettings)} $out/SDKSettings.plist sdk=$out/${sdkName}.sdk
install -D ${writeText "SystemVersion.plist" (toPlist {} SystemVersion)} $out/System/Library/CoreServices/SystemVersion.plist install -D ${writeText "SDKSettings.plist" (toPlist {} SDKSettings)} $sdk/SDKSettings.plist
install -D ${writeText "SystemVersion.plist" (toPlist {} SystemVersion)} $sdk/System/Library/CoreServices/SystemVersion.plist
ln -s $sdk $out/${xcodePlatform}.sdk
'' ''

View File

@ -1,69 +0,0 @@
{ runCommand, toolchainName, fetchurl, makeWrapper, stdenv
, buildPackages, lib, writeText }:
let
inherit (lib) getBin optionalString;
inherit (lib.generators) toPlist;
ToolchainInfo = {
Identifier = toolchainName;
};
# We could pull this out of developer_cmds but it adds an annoying
# loop if we want to bootstrap and this is just a tiny script so I'm
# not going to bother.
mkdep-darwin-src = fetchurl {
url = "https://opensource.apple.com/source/developer_cmds/developer_cmds-63/mkdep/mkdep.sh";
sha256 = "0n4wpqfslfjs5zbys5yri8pfi2awyhlmknsf6laa5jzqbzq9x541";
executable = true;
};
in
runCommand "nixpkgs.xctoolchain" {
nativeBuildInputs = [ makeWrapper ];
} (''
mkdir -p $out
install -D ${writeText "ToolchainInfo.plist" (toPlist {} ToolchainInfo)} $out/ToolchainInfo.plist
mkdir -p $out/usr/include
mkdir -p $out/usr/lib
mkdir -p $out/usr/libexec
mkdir -p $out/usr/share
mkdir -p $out/usr/bin
for bin in ${getBin stdenv.cc}/bin/*; do
ln -s $bin $out/usr/bin
done
for bin in ${getBin stdenv.cc.bintools.bintools}/bin/*; do
if ! [ -e "$out/usr/bin/$(basename $bin)" ]; then
ln -s $bin $out/usr/bin
fi
done
ln -s ${buildPackages.yacc}/bin/yacc $out/usr/bin/yacc
ln -s ${buildPackages.yacc}/bin/bison $out/usr/bin/bison
ln -s ${buildPackages.flex}/bin/flex $out/usr/bin/flex
ln -s ${buildPackages.flex}/bin/flex++ $out/usr/bin/flex++
ln -s $out/bin/flex $out/usr/bin/lex
ln -s ${buildPackages.m4}/bin/m4 $out/usr/bin/m4
ln -s $out/usr/bin/m4 $out/usr/bin/gm4
ln -s ${buildPackages.unifdef}/bin/unifdef $out/usr/bin/unifdef
ln -s ${buildPackages.unifdef}/bin/unifdefall $out/usr/bin/unifdefall
ln -s ${buildPackages.gperf}/bin/gperf $out/usr/bin/gperf
ln -s ${buildPackages.indent}/bin/indent $out/usr/bin/indent
ln -s ${buildPackages.ctags}/bin/ctags $out/usr/bin/ctags
'' + optionalString stdenv.isDarwin ''
for bin in ${getBin buildPackages.darwin.cctools}/bin/*; do
if ! [ -e "$out/usr/bin/$(basename $bin)" ]; then
ln -s $bin $out/usr/bin
fi
done
ln -s ${buildPackages.darwin.bootstrap_cmds}/bin/mig $out/usr/bin
ln -s ${mkdep-darwin-src} $out/usr/bin/mkdep
'')

View File

@ -0,0 +1,71 @@
{ runCommand, toolchainName, fetchurl, makeWrapper, stdenv
, buildPackages, lib, writeText }:
let
inherit (lib) getBin optionalString;
inherit (lib.generators) toPlist;
ToolchainInfo = {
Identifier = toolchainName;
};
# We could pull this out of developer_cmds but it adds an annoying
# loop if we want to bootstrap and this is just a tiny script so I'm
# not going to bother.
mkdep-darwin-src = fetchurl {
url = "https://opensource.apple.com/source/developer_cmds/developer_cmds-63/mkdep/mkdep.sh";
sha256 = "0n4wpqfslfjs5zbys5yri8pfi2awyhlmknsf6laa5jzqbzq9x541";
executable = true;
};
in
runCommand "Toolchains" {
nativeBuildInputs = [ makeWrapper ];
} (''
toolchain=$out/XcodeDefault.xctoolchain
mkdir -p $toolchain
install -D ${writeText "ToolchainInfo.plist" (toPlist {} ToolchainInfo)} $toolchain/ToolchainInfo.plist
mkdir -p $toolchain/usr/include
mkdir -p $toolchain/usr/lib
mkdir -p $toolchain/usr/libexec
mkdir -p $toolchain/usr/share
mkdir -p $toolchain/usr/bin
for bin in ${getBin stdenv.cc}/bin/*; do
ln -s $bin $toolchain/usr/bin
done
for bin in ${getBin stdenv.cc.bintools.bintools}/bin/*; do
if ! [ -e "$toolchain/usr/bin/$(basename $bin)" ]; then
ln -s $bin $toolchain/usr/bin
fi
done
ln -s ${buildPackages.yacc}/bin/yacc $toolchain/usr/bin/yacc
ln -s ${buildPackages.yacc}/bin/bison $toolchain/usr/bin/bison
ln -s ${buildPackages.flex}/bin/flex $toolchain/usr/bin/flex
ln -s ${buildPackages.flex}/bin/flex++ $toolchain/usr/bin/flex++
ln -s $toolchain/bin/flex $toolchain/usr/bin/lex
ln -s ${buildPackages.m4}/bin/m4 $toolchain/usr/bin/m4
ln -s $toolchain/usr/bin/m4 $toolchain/usr/bin/gm4
ln -s ${buildPackages.unifdef}/bin/unifdef $toolchain/usr/bin/unifdef
ln -s ${buildPackages.unifdef}/bin/unifdefall $toolchain/usr/bin/unifdefall
ln -s ${buildPackages.gperf}/bin/gperf $toolchain/usr/bin/gperf
ln -s ${buildPackages.indent}/bin/indent $toolchain/usr/bin/indent
ln -s ${buildPackages.ctags}/bin/ctags $toolchain/usr/bin/ctags
'' + optionalString stdenv.isDarwin ''
for bin in ${getBin buildPackages.darwin.cctools}/bin/*; do
if ! [ -e "$toolchain/usr/bin/$(basename $bin)" ]; then
ln -s $bin $toolchain/usr/bin
fi
done
ln -s ${buildPackages.darwin.bootstrap_cmds}/bin/mig $toolchain/usr/bin
ln -s ${mkdep-darwin-src} $toolchain/usr/bin/mkdep
'')

View File

@ -1,80 +1,132 @@
{ stdenv, buildPackages, makeWrapper, writeText, runCommand { stdenv, lib, buildPackages, makeWrapper, writeText, runCommand
, CoreServices, ImageIO, CoreGraphics }: , CoreServices, ImageIO, CoreGraphics
, targetPlatform
, xcodePlatform ? targetPlatform.xcodePlatform or "MacOSX"
, xcodeVer ? targetPlatform.xcodeVer or "9.4.1"
, sdkVer ? targetPlatform.sdkVer or "10.10" }:
let let
inherit (lib) toLower;
toolchainName = "com.apple.dt.toolchain.XcodeDefault"; toolchainName = "com.apple.dt.toolchain.XcodeDefault";
platformName = "com.apple.platform.macosx"; sdkName = "${xcodePlatform}${sdkVer}";
sdkName = "macosx10.10";
# TODO: expose MACOSX_DEPLOYMENT_TARGET in nix so we can use it here.
sdkBuildVersion = "17E189";
xcodeSelectVersion = "2349";
xcbuild = buildPackages.callPackage ./default.nix { xcbuild = buildPackages.callPackage ./default.nix {
inherit CoreServices ImageIO CoreGraphics; inherit CoreServices ImageIO CoreGraphics;
}; };
toolchain = buildPackages.callPackage ./toolchain.nix { toolchains = buildPackages.callPackage ./toolchains.nix {
inherit toolchainName; inherit toolchainName;
}; };
sdk = buildPackages.callPackage ./sdk.nix { sdks = buildPackages.callPackage ./sdks.nix {
inherit toolchainName sdkName; inherit toolchainName sdkName xcodePlatform;
version = sdkVer;
}; };
platform = buildPackages.callPackage ./platform.nix { platforms = buildPackages.callPackage ./platforms.nix {
inherit sdk platformName; inherit sdks xcodePlatform;
}; };
xcconfig = writeText "nix.xcconfig" '' xcconfig = writeText "nix.xcconfig" ''
SDKROOT=${sdkName} SDKROOT=${sdkName}
''; '';
xcode-select = writeText "xcode-select" ''
#!/usr/bin/env sh
while [ $# -gt 0 ]; do
case "$1" in
-h | --help) ;; # noop
-s | --switch) shift;; # noop
-r | --reset) ;; # noop
-v | --version) echo xcode-select version ${xcodeSelectVersion} ;;
-p | --print-path) echo @DEVELOPER_DIR@ ;;
--install) ;; # noop
esac
shift
done
'';
xcrun = writeText "xcrun" ''
#!/usr/bin/env sh
while [ $# -gt 0 ]; do
case "$1" in
--sdk | -sdk) shift ;;
--find | -find)
shift
command -v $1 ;;
--log | -log) ;; # noop
--verbose | -verbose) ;; # noop
--no-cache | -no-cache) ;; # noop
--kill-cache | -kill-cache) ;; # noop
--show-sdk-path | -show-sdk-path)
echo ${sdks}/${sdkName}.sdk ;;
--show-sdk-platform-path | -show-sdk-platform-path)
echo ${platforms}/${xcodePlatform}.platform ;;
--show-sdk-version | -show-sdk-version)
echo ${sdkVer} ;;
--show-sdk-build-version | -show-sdk-build-version)
echo ${sdkBuildVersion} ;;
*) break ;;
esac
shift
done
if ! [[ -z "$@" ]]; then
exec "$@"
fi
'';
in in
stdenv.mkDerivation { runCommand "xcodebuild-${xcbuild.version}" {
name = "xcbuild-wrapper-${xcbuild.version}";
nativeBuildInputs = [ makeWrapper ]; nativeBuildInputs = [ makeWrapper ];
inherit (xcbuild) meta;
setupHook = ./setup-hook.sh; # ensure that the toolchain goes in PATH
propagatedBuildInputs = [ "${toolchains}/XcodeDefault.xctoolchain/usr" ];
phases = [ "installPhase" "fixupPhase" ]; passthru = { inherit xcbuild; };
installPhase = '' preferLocalBuild = true;
} ''
mkdir -p $out/bin mkdir -p $out/bin
for file in ${xcbuild}/bin/*; do
ln -s $file $out/bin
done
mkdir -p $out/usr mkdir -p $out/usr
ln -s $out/bin $out/usr/bin ln -s $out/bin $out/usr/bin
mkdir -p $out/Library/Xcode mkdir -p $out/Library/Xcode
ln -s ${xcbuild}/Library/Xcode/Specifications $out/Library/Xcode/Specifications ln -s ${xcbuild}/Library/Xcode/Specifications $out/Library/Xcode/Specifications
mkdir -p $out/Platforms ln -s ${platforms} $out/Platforms
ln -s ${platform} $out/Platforms/nixpkgs.platform ln -s ${toolchains} $out/Toolchains
mkdir -p $out/Toolchains makeWrapper ${xcbuild}/bin/xcodebuild $out/bin/xcodebuild \
ln -s ${toolchain} $out/Toolchains/nixpkgs.xctoolchain
wrapProgram $out/bin/xcodebuild \
--add-flags "-xcconfig ${xcconfig}" \ --add-flags "-xcconfig ${xcconfig}" \
--add-flags "DERIVED_DATA_DIR=." \ --add-flags "DERIVED_DATA_DIR=." \
--set DEVELOPER_DIR "$out" \ --set DEVELOPER_DIR "$out" \
--set SDKROOT ${sdkName} --set SDKROOT ${sdkName} \
wrapProgram $out/bin/xcrun \ --run '[ "$1" = "-version" ] && (echo Xcode ${xcodeVer}; echo Build version ${sdkBuildVersion}) && exit 0'
--set DEVELOPER_DIR "$out" \
--set SDKROOT ${sdkName}
wrapProgram $out/bin/xcode-select \
--set DEVELOPER_DIR "$out" \
--set SDKROOT ${sdkName}
'';
inherit (xcbuild) meta; substitute ${xcode-select} $out/bin/xcode-select \
--subst-var-by DEVELOPER_DIR $out
chmod +x $out/bin/xcode-select
passthru = { substitute ${xcrun} $out/bin/xcrun
raw = xcbuild; chmod +x $out/bin/xcrun
};
preferLocalBuild = true; for bin in PlistBuddy actool builtin-copy builtin-copyPlist \
} builtin-copyStrings builtin-copyTiff \
builtin-embeddedBinaryValidationUtility \
builtin-infoPlistUtility builtin-lsRegisterURL \
builtin-productPackagingUtility builtin-validationUtility \
lsbom plutil; do
ln -s ${xcbuild}/bin/$bin $out/bin/$bin
done
fixupPhase
''