Skip to content

Commit

Permalink
[wip] prebuilt-tdlib: crosscompile to Linux arm64
Browse files Browse the repository at this point in the history
  • Loading branch information
eilvelia committed Jul 17, 2024
1 parent d3580a4 commit b2bf432
Show file tree
Hide file tree
Showing 7 changed files with 219 additions and 16 deletions.
30 changes: 29 additions & 1 deletion .github/workflows/prebuilt-tdlib.yml
Original file line number Diff line number Diff line change
Expand Up @@ -35,9 +35,31 @@ jobs:
with:
name: tdlib-linux-x86_64-glibc
path: ${{ env.TO_UPLOAD }}
build-linux-arm64:
name: 'Build TDLib / Linux arm64 glibc (cross)'
runs-on: ubuntu-22.04
steps:
- uses: actions/checkout@v4
- uses: cachix/install-nix-action@v27
with:
nix_path: nixpkgs=channel:nixos-unstable
- name: Build TDLib
run: |
cd packages/prebuilt-tdlib/ci
./build-linux.sh ${{ inputs.tdlib }} aarch64-linux-gnu.2.22
- name: Setup tmate session
uses: mxschmitt/action-tmate@v3
if: failure()
with:
limit-access-to-actor: true
- uses: actions/upload-artifact@v4
with:
name: tdlib-linux-arm64
path: ${{ env.TO_UPLOAD }}
build-macos-x86_64:
name: 'Build TDLib / macOS x86_64'
runs-on: macos-13
if: ${{ false }}
steps:
- uses: actions/checkout@v4
- uses: cachix/install-nix-action@v27
Expand All @@ -54,6 +76,7 @@ jobs:
build-macos-arm64:
name: 'Build TDLib / macOS arm64'
runs-on: macos-14
if: ${{ false }}
steps:
- uses: actions/checkout@v4
- uses: cachix/install-nix-action@v27
Expand Down Expand Up @@ -90,6 +113,7 @@ jobs:
build-windows-x86_64:
name: 'Build TDLib / Windows x86_64'
runs-on: windows-2019
if: ${{ false }}
steps:
- uses: actions/checkout@v4
with:
Expand Down Expand Up @@ -126,7 +150,11 @@ jobs:

test:
name: 'Test / ${{ matrix.v.bin }} / ${{ matrix.v.os }}'
needs: [build-linux-x86_64, build-macos, build-windows-x86_64]
needs:
- build-linux-x86_64
- build-linux-arm64
- build-macos
- build-windows-x86_64
runs-on: ${{ matrix.v.os }}
strategy:
fail-fast: false
Expand Down
4 changes: 3 additions & 1 deletion packages/prebuilt-tdlib/README.md
Original file line number Diff line number Diff line change
Expand Up @@ -12,7 +12,7 @@ The shared libraries are statically linked against OpenSSL and zlib, for one, to
prevent compatibility issues in Node.js.

Supported systems:
- Linux x86_64 (requires glibc >= 2.22)
- Linux x86_64, arm64 (requires glibc >= 2.22)
- macOS x86_64, arm64 (universal, requires macOS >= 10.12)
- Windows x86_64

Expand Down Expand Up @@ -142,6 +142,8 @@ The building process is significantly changed in this update.
crosscompiled anymore).
- On Linux, TDLib is now built using zig. The minimal glibc version is 2.22
instead of 2.17.
- Added a crosscompiled prebuild for Linux arm64. However, this prebuild is
not tested to work in CI.

### 2024-05-08

Expand Down
57 changes: 57 additions & 0 deletions packages/prebuilt-tdlib/ci/tdlib-linux-old.nix
Original file line number Diff line number Diff line change
@@ -0,0 +1,57 @@
{ pkgs ? import <nixpkgs> {}, target, rev }:
let
inherit (pkgs) lib stdenv;
zig-toolchain = import ./zig-toolchain.nix { inherit pkgs target; };
zlib = pkgs.callPackage ./zlib-zig.nix { inherit zig-toolchain; };
openssl = pkgs.callPackage ./openssl-zig.nix { inherit zig-toolchain; };
in
# Based on https://github.com/NixOS/nixpkgs/blob/af51e23ce535b1bfa8484021ff3913d876e09082/pkgs/development/libraries/tdlib/default.nix
stdenv.mkDerivation {
pname = "tdlib";
version = "0.0";

src = builtins.fetchTarball "https://github.com/tdlib/td/archive/${rev}.tar.gz";

buildInputs = [ openssl zlib ];
nativeBuildInputs = with pkgs; [ cmake gperf ];

preConfigure = ''
mkdir native-build && cd native-build
cmake ..
cmake --build . --target prepare_cross_compiling
cd ..
'' + zig-toolchain.env;

patches = [ ./tdlib-zig.patch ];

cmakeFlags = [
"-DCMAKE_BUILD_TYPE=Release"
"-DOPENSSL_USE_STATIC_LIBS=TRUE"
"-DZLIB_USE_STATIC_LIBS=TRUE"
"-DCMAKE_SYSTEM_NAME=Linux"
"-DCMAKE_CROSSCOMPILING=TRUE"
];

buildPhase = ''
cmake --build . --target tdjson -j $NIX_BUILD_CORES
'';

installPhase = ''
runHook preInstall
mkdir -p "$out"/lib
cp -L ./libtdjson.so "$out"/lib/
runHook postInstall
'';

# https://github.com/tdlib/td/issues/1974
postPatch = ''
substituteInPlace CMake/GeneratePkgConfig.cmake \
--replace 'function(generate_pkgconfig' \
'include(GNUInstallDirs)
function(generate_pkgconfig' \
--replace '\$'{prefix}/'$'{CMAKE_INSTALL_LIBDIR} '$'{CMAKE_INSTALL_FULL_LIBDIR} \
--replace '\$'{prefix}/'$'{CMAKE_INSTALL_INCLUDEDIR} '$'{CMAKE_INSTALL_FULL_INCLUDEDIR}
'' + lib.optionalString (stdenv.isDarwin && stdenv.isAarch64) ''
sed -i "/vptr/d" test/CMakeLists.txt
'';
}
17 changes: 12 additions & 5 deletions packages/prebuilt-tdlib/ci/tdlib-linux.nix
Original file line number Diff line number Diff line change
@@ -1,7 +1,10 @@
{ pkgs ? import <nixpkgs> {}, target, rev }:
{ target, rev }:
let
pkgs = if builtins.substring 0 7 target == "aarch64"
then import <nixpkgs> { crossSystem.config = "aarch64-unknown-linux-gnu"; }
else import <nixpkgs> {};
inherit (pkgs) lib stdenv;
zig-toolchain = import ./zig-toolchain.nix { inherit pkgs target; };
zig-toolchain = import ./zig-toolchain.nix { inherit target; pkgs = pkgs.buildPackages.buildPackages; };
zlib = pkgs.callPackage ./zlib-zig.nix { inherit zig-toolchain; };
openssl = pkgs.callPackage ./openssl-zig.nix { inherit zig-toolchain; };
in
Expand All @@ -13,12 +16,16 @@ stdenv.mkDerivation {
src = builtins.fetchTarball "https://github.com/tdlib/td/archive/${rev}.tar.gz";

buildInputs = [ openssl zlib ];
nativeBuildInputs = with pkgs; [ cmake gperf ];
nativeBuildInputs = with pkgs.buildPackages; [ cmake gperf ];

preConfigure = ''
export CC=${pkgs.buildPackages.stdenv.cc}/bin/cc
export CXX=${pkgs.buildPackages.stdenv.cc}/bin/c++
mkdir native-build && cd native-build
cmake ..
cmake --build . --target prepare_cross_compiling
cmake -DOPENSSL_ROOT_DIR=${pkgs.buildPackages.openssl.dev} \
-DZLIB_ROOT=${pkgs.buildPackages.zlib.dev} \
..
cmake --build . --target prepare_cross_compiling -j $NIX_BUILD_CORES
cd ..
'' + zig-toolchain.env;

Expand Down
107 changes: 107 additions & 0 deletions packages/prebuilt-tdlib/ci/zig-stdenv.nix
Original file line number Diff line number Diff line change
@@ -0,0 +1,107 @@
# https://gist.github.com/Cloudef/acb74ff9e36ab41709479240596ab501
{ pkgs ? import <nixpkgs> {}, lib ? pkgs.lib, stdenv ? pkgs.stdenv, zig ? pkgs.zig, static ? false, target, zig-target }:

with lib;
with builtins;

let
zig-stdenv = let
write-zig-wrapper = cmd: pkgs.writeShellScript "zig-${cmd}" ''${zig}/bin/zig ${cmd} "$@"'';
zig-cc = write-zig-wrapper "cc";
zig-cxx = write-zig-wrapper "c++";
zig-ar = write-zig-wrapper "ar";
zig-ranlib = write-zig-wrapper "ranlib";
zig-dlltool = write-zig-wrapper "dlltool";
zig-lib = write-zig-wrapper "lib";

# XXX: Consider LLVM binutils instead?
bintools = pkgs.bintoolsNoLibc;
symlinks = [
{ name = "bin/clang"; path = zig-cc; }
{ name = "bin/clang++"; path = zig-cxx; }
{ name = "bin/cc"; path = zig-cc; }
{ name = "bin/c++"; path = zig-cxx; }
{ name = "bin/ld"; path = zig-cc; }
{ name = "bin/gold"; path = zig-cc; }
{ name = "bin/ar"; path = zig-ar; }
{ name = "bin/ranlib"; path = zig-ranlib; }
{ name = "bin/dlltool"; path = zig-dlltool; }
{ name = "bin/lib"; path = zig-lib; }
] ++ map (x: { name = "bin/${x}"; path = "${bintools}/bin/${x}"; }) [
"addr2line" "c++filt" "elfedit" "gprof" "gprofng" "nlmconv" "nm" "as"
"objcopy" "objdump" "readelf" "size" "strings" "windmc" "windres" "strip"
];

zig-toolchain = pkgs.linkFarm "zig-toolchain" (symlinks
++ map (x: let y = last (splitString "/" x.name); in { name = "bin/${target}-${y}"; path = y; }) symlinks) // {
inherit (pkgs.llvmPackages.libclang) version;
pname = "zig-toolchain";
isClang = true;
libllvm.out = "";
};

# Bunch of overrides to make sure we don't ever start bootstrapping another cross-compiler
overrides = self: super: let
to-override-set = x: genAttrs x (x: zig-toolchain);
llvmPackages = to-override-set [ "clang" "libclang" "lld" "llvm" "libllvm" "compiler-rt" "libunwind" "libstdcxx" "libcxx" "libcxxapi" "openmp" ];
in (to-override-set [
"gcc" "libgcc" "glibc" "bintools" "bintoolsNoLibc" "binutils" "binutilsNoLibc"
"gccCrossStageStatic" "preLibcCrossHeaders" "glibcCross" "muslCross" "libcCross"
]) // {
inherit llvmPackages;
inherit (llvmPackages) clang libclang lld llvm libllvm libunwind libstdcxx libcxx libcxxapi openmp;
gccForLibs = null; # also disables --gcc-toolchain passed to compiler
};

cross0 = import pkgs.path {
crossSystem.config = target;
localSystem.config = pkgs.buildPlatform.config;
crossOverlays = [overrides];
};

static-stdenv-maybe = x: if static then cross0.makeStatic x else x;

zig-stdenv0 = static-stdenv-maybe (cross0.overrideCC cross0.stdenv (cross0.wrapCCWith rec {
inherit (pkgs) gnugrep coreutils;
cc = zig-toolchain; libc = cc; libcxx = cc;
bintools = cross0.wrapBintoolsWith { inherit libc gnugrep coreutils; bintools = cc; };
# XXX: -march is not compatible
# https://github.com/ziglang/zig/issues/4911
extraBuildCommands = ''
substituteInPlace $out/nix-support/add-local-cc-cflags-before.sh --replace "${target}" "${zig-target}"
sed -i 's/-march[^ ]* *//g' $out/nix-support/cc-cflags-before
'';
}));

# Aaand .. finally our final stdenv :)
in zig-stdenv0.override {
inherit overrides;
# XXX: Zig doesn't support response file. Nixpkgs wants to use this for clang
# while zig cc is basically clang, it's still not 100% compatible.
# Probably should report this as a bug to zig upstream though.
preHook = ''
${cross0.stdenv.preHook}
export NIX_CC_USE_RESPONSE_FILE=0
export ZIG_LOCAL_CACHE_DIR="$TMPDIR/zig-cache-${target}"
export ZIG_GLOBAL_CACHE_DIR="$ZIG_LOCAL_CACHE_DIR"
'';
};

cross = (import pkgs.path {
crossSystem.config = target;
localSystem.config = pkgs.buildPlatform.config;
crossOverlays = [(self: super: {
stdenv = zig-stdenv;
# XXX: libsepol issue on darwin, should be fixed upstrem instead
libsepol = super.libsepol.overrideAttrs (old: {
nativeBuildInputs = with pkgs; old.nativeBuildInputs ++ optionals (stdenv.isDarwin) [
(writeShellScriptBin "gln" ''${pkgs.coreutils}/bin/ln "$@"'')
];
});
})];
});
in {
inherit target;
stdenv = zig-stdenv;
pkgs = cross;
}
9 changes: 0 additions & 9 deletions packages/prebuilt-tdlib/ci/zlib-zig.nix
Original file line number Diff line number Diff line change
Expand Up @@ -2,13 +2,4 @@
(zlib.override { static = true; shared = false; }).overrideAttrs (final: prev: {
preConfigure = prev.preConfigure + zig-toolchain.env;
doCheck = false;
# preCheck = ''
# ls
# echo ---
# ldd ./minigzip
# ./minigzip
# echo START
# make check
# echo FINISH
# '';
})
11 changes: 11 additions & 0 deletions packages/prebuilt-tdlib/prebuild-list.js
Original file line number Diff line number Diff line change
Expand Up @@ -45,6 +45,17 @@ const prebuilds/*: PrebuildInfo[] */ = [
cpu: ['x64']
},
libc: 'glibc'
},
{
packageName: 'linux-arm64-glibc',
prebuildDir: 'tdlib-linux-arm64-glibc',
libfile: SHARED_LINUX,
descr: 'Linux arm64 (glibc)',
requirements: {
os: ['linux'],
cpu: ['arm64']
},
libc: 'glibc'
}
]

Expand Down

0 comments on commit b2bf432

Please sign in to comment.