mirror of
https://github.com/NixOS/nixpkgs.git
synced 2024-12-18 03:34:58 +00:00
f2bb67c806
(cherry picked from commit 5f4bf240f5
)
132 lines
4.6 KiB
Nix
132 lines
4.6 KiB
Nix
{ lib, stdenv
|
|
, makeWrapper
|
|
, runCommand, wrapBintoolsWith, wrapCCWith
|
|
, buildAndroidndk, androidndk, targetAndroidndkPkgs
|
|
}:
|
|
|
|
let
|
|
# Mapping from a platform to information needed to unpack NDK stuff for that
|
|
# platform.
|
|
#
|
|
# N.B. The Android NDK uses slightly different LLVM-style platform triples
|
|
# than we do. We don't just use theirs because ours are less ambiguous and
|
|
# some builds need that clarity.
|
|
ndkInfoFun = { config, ... }: {
|
|
"x86_64-apple-darwin" = {
|
|
double = "darwin-x86_64";
|
|
};
|
|
"x86_64-unknown-linux-gnu" = {
|
|
double = "linux-x86_64";
|
|
};
|
|
"armv5tel-unknown-linux-androideabi" = {
|
|
arch = "arm";
|
|
triple = "arm-linux-androideabi";
|
|
gccVer = "4.8";
|
|
};
|
|
"armv7a-unknown-linux-androideabi" = {
|
|
arch = "arm";
|
|
triple = "arm-linux-androideabi";
|
|
gccVer = "4.9";
|
|
};
|
|
"aarch64-unknown-linux-android" = {
|
|
arch = "arm64";
|
|
triple = "aarch64-linux-android";
|
|
gccVer = "4.9";
|
|
};
|
|
}.${config} or
|
|
(throw "Android NDK doesn't support ${config}, as far as we know");
|
|
|
|
hostInfo = ndkInfoFun stdenv.hostPlatform;
|
|
targetInfo = ndkInfoFun stdenv.targetPlatform;
|
|
in
|
|
|
|
rec {
|
|
# Misc tools
|
|
binaries = let
|
|
ndkBinDir =
|
|
"${androidndk}/libexec/android-sdk/ndk-bundle/toolchains/${targetInfo.triple}-${targetInfo.gccVer}/prebuilt/${hostInfo.double}/bin";
|
|
ndkGCCLibDir =
|
|
"${androidndk}/libexec/android-sdk/ndk-bundle/toolchains/${targetInfo.triple}-${targetInfo.gccVer}/prebuilt/${hostInfo.double}/lib/gcc/${targetInfo.triple}/4.9.x";
|
|
|
|
in runCommand "ndk-gcc-binutils" {
|
|
isGNU = true; # for cc-wrapper
|
|
nativeBuildInputs = [ makeWrapper ];
|
|
propgatedBuildInputs = [ androidndk ];
|
|
} ''
|
|
mkdir -p $out/bin
|
|
for prog in ${ndkBinDir}/${targetInfo.triple}-*; do
|
|
prog_suffix=$(basename $prog | sed 's/${targetInfo.triple}-//')
|
|
cat > $out/bin/${stdenv.targetPlatform.config}-$prog_suffix <<EOF
|
|
#! ${stdenv.shell} -e
|
|
$prog "\$@"
|
|
EOF
|
|
chmod +x $out/bin/${stdenv.targetPlatform.config}-$prog_suffix
|
|
done
|
|
|
|
ln -s $out/bin/${stdenv.targetPlatform.config}-ld $out/bin/ld
|
|
ln -s ${ndkGCCLibDir} $out/lib
|
|
'';
|
|
|
|
binutils = wrapBintoolsWith {
|
|
bintools = binaries;
|
|
libc = targetAndroidndkPkgs.libraries;
|
|
extraBuildCommands = ''
|
|
echo "--build-id" >> $out/nix-support/libc-ldflags
|
|
'';
|
|
};
|
|
|
|
gcc = wrapCCWith {
|
|
cc = binaries;
|
|
bintools = binutils;
|
|
libc = targetAndroidndkPkgs.libraries;
|
|
extraBuildCommands = ''
|
|
echo "-D__ANDROID_API__=${stdenv.targetPlatform.sdkVer}" >> $out/nix-support/cc-cflags
|
|
''
|
|
+ lib.optionalString stdenv.targetPlatform.isAarch32 (let
|
|
p = stdenv.targetPlatform.platform.gcc or {}
|
|
// stdenv.targetPlatform.parsed.abi;
|
|
flags = lib.concatLists [
|
|
(lib.optional (p ? arch) "-march=${p.arch}")
|
|
(lib.optional (p ? cpu) "-mcpu=${p.cpu}")
|
|
(lib.optional (p ? abi) "-mabi=${p.abi}")
|
|
(lib.optional (p ? fpu) "-mfpu=${p.fpu}")
|
|
(lib.optional (p ? float) "-mfloat=${p.float}")
|
|
(lib.optional (p ? float-abi) "-mfloat-abi=${p.float-abi}")
|
|
(lib.optional (p ? mode) "-mmode=${p.mode}")
|
|
];
|
|
in ''
|
|
sed -E -i \
|
|
$out/bin/${stdenv.targetPlatform.config}-cc \
|
|
$out/bin/${stdenv.targetPlatform.config}-c++ \
|
|
$out/bin/${stdenv.targetPlatform.config}-gcc \
|
|
$out/bin/${stdenv.targetPlatform.config}-g++ \
|
|
-e 's|^(extraBefore=)\((.*)\)$|\1(\2 -Wl,--fix-cortex-a8 ${builtins.toString flags})|'
|
|
'')
|
|
# GCC 4.9 is the first relase with "-fstack-protector"
|
|
+ lib.optionalString (lib.versionOlder targetInfo.gccVer "4.9") ''
|
|
sed -E \
|
|
-i $out/nix-support/add-hardening.sh \
|
|
-e 's|(-fstack-protector)-strong|\1|g'
|
|
'';
|
|
};
|
|
|
|
# Bionic lib C and other libraries.
|
|
#
|
|
# We use androidndk from the previous stage, else we waste time or get cycles
|
|
# cross-compiling packages to wrap incorrectly wrap binaries we don't include
|
|
# anyways.
|
|
libraries =
|
|
let
|
|
includePath = "${buildAndroidndk}/libexec/android-sdk/ndk-bundle/sysroot/usr/include";
|
|
asmIncludePath = "${buildAndroidndk}/libexec/android-sdk/ndk-bundle/sysroot/usr/include/${targetInfo.triple}";
|
|
libPath = "${buildAndroidndk}/libexec/android-sdk/ndk-bundle/platforms/android-${stdenv.hostPlatform.sdkVer}/arch-${hostInfo.arch}/usr/lib/";
|
|
in
|
|
runCommand "bionic-prebuilt" {} ''
|
|
mkdir -p $out
|
|
cp -r ${includePath} $out/include
|
|
chmod +w $out/include
|
|
cp -r ${asmIncludePath}/* $out/include
|
|
ln -s ${libPath} $out/lib
|
|
'';
|
|
}
|