f native image, just build a jar

This commit is contained in:
Gregor Lohaus
2026-05-29 12:33:32 +02:00
parent 08ad025f76
commit b98a18b49f
12 changed files with 128 additions and 447 deletions

262
flake.nix
View File

@@ -11,6 +11,33 @@
let
supportedSystems = [ "x86_64-linux" ];
forAllSystems = nixpkgs.lib.genAttrs supportedSystems;
# Everything the JavaFX natives (and jlibghostty) dlopen at runtime, EXCEPT the
# system OpenGL/graphics drivers. libGL is intentionally left out: it is supplied
# by the host at runtime via the GL shim in the wrapper below, so the same closure
# works on NixOS and on a plain Debian box with vendor GPU drivers installed.
runtimeLibsFor = pkgs: ghosttyVt: [
pkgs.glib
pkgs.gtk3
pkgs.pango
pkgs.cairo
pkgs.gdk-pixbuf
pkgs.harfbuzz
pkgs.freetype
pkgs.fontconfig.lib
pkgs.libx11
pkgs.libxext
pkgs.libxrender
pkgs.libxtst
pkgs.libxi
pkgs.libxcursor
pkgs.libxrandr
pkgs.libxinerama
pkgs.libxcb
pkgs.libxxf86vm
pkgs.zlib
ghosttyVt
];
in {
packages = forAllSystems (system:
let
@@ -19,218 +46,73 @@
jlib = jlibghostty.packages.${system}.jlibghostty;
ghosttyVt = ghostty.packages.${system}.libghostty-vt;
javafxStaticSdkZip = pkgs.fetchurl {
url = "https://download2.gluonhq.com/substrate/javafxstaticsdk/openjfx-21-ea+11.3-linux-x86_64-static.zip";
hash = "sha256-ovoByMPwhvU54mtxGYyrgLxDLNn0tA3XUK3rtnGfAAM=";
};
runtimeLibs = runtimeLibsFor pkgs ghosttyVt;
patchedSubstrateJar =
let
substrateJar = pkgs.fetchurl {
url = "https://plugins.gradle.org/m2/com/gluonhq/substrate/0.0.68/substrate-0.0.68.jar";
hash = "sha256-ZOchzSN8IPSQVnItGAvo3T6OG93G0lqU+BmemdFa6lE=";
};
in
pkgs.runCommand "substrate-0.0.68-patched.jar"
{
nativeBuildInputs = [
pkgs.perl
pkgs.unzip
pkgs.zip
];
}
''
mkdir work
cd work
unzip -q ${substrateJar}
# Gluon Substrate hardcodes /usr/bin/pkg-config. Keep the
# replacement string the same byte length to avoid rewriting
# the Java class constant pool structure.
perl -0pi -e 's|/usr/bin/pkg-config|/tmp/nix/pkg-config|g' \
com/gluonhq/substrate/util/linux/LinuxLinkerFlags.class
if grep -a -q "/usr/bin/pkg-config" com/gluonhq/substrate/util/linux/LinuxLinkerFlags.class; then
echo "Failed to patch hardcoded pkg-config path in Substrate" >&2
exit 1
fi
zip -qr "$out" .
'';
gluonGraalvm = pkgs.stdenv.mkDerivation {
pname = "graalvm-java23-gluon";
version = "23+25.1-dev-2409082136";
src = pkgs.fetchurl {
url = "https://github.com/gluonhq/graal/releases/download/gluon-23%2B25.1-dev-2409082136/graalvm-java23-linux-amd64-gluon-23+25.1-dev.tar.gz";
hash = "sha256-/NyMutn3pT4ZKL2pkzPdBZghxg0ERK5VJ2bFQF0VBfU=";
};
nativeBuildInputs = [
pkgs.autoPatchelfHook
];
buildInputs = [
pkgs.stdenv.cc.cc.lib
pkgs.zlib
pkgs.freetype
pkgs.fontconfig
pkgs.alsa-lib
pkgs.glib
pkgs.gtk3
pkgs.pango
pkgs.libx11
pkgs.libxext
pkgs.libxrender
pkgs.libxtst
pkgs.libxi
pkgs.libxrandr
pkgs.libxinerama
pkgs.libxcb
];
installPhase = ''
runHook preInstall
mkdir -p "$out"
cp -R . "$out/"
# The GluonFX Gradle plugin expects these two static libraries
# directly under linux-amd64, but this tarball keeps them one
# level deeper under linux-amd64/glibc.
ln -sfn ./glibc/libjvm.a "$out/lib/svm/clibraries/linux-amd64/libjvm.a"
ln -sfn ./glibc/liblibchelper.a "$out/lib/svm/clibraries/linux-amd64/liblibchelper.a"
runHook postInstall
'';
};
runtimeLibs = [
pkgs.glib
pkgs.gtk3
pkgs.pango
pkgs.alsa-lib
pkgs.ffmpeg.dev
pkgs.ffmpeg.lib
pkgs.freetype
pkgs.fontconfig
pkgs.libx11
pkgs.libx11.dev
pkgs.libxext
pkgs.libxrender
pkgs.libxtst
pkgs.libxi
pkgs.libxcursor
pkgs.libxrandr
pkgs.libxinerama
pkgs.libxcb
pkgs.libxxf86vm
pkgs.zlib
pkgs.zlib.dev
ghosttyVt
];
jprototerm = pkgs.stdenv.mkDerivation (finalAttrs: {
pname = "jprototerm";
version = "0.1.0";
src = ./.;
nativeBuildInputs = [
gluonGraalvm
pkgs.jdk25
pkgs.gradle_9
pkgs.makeWrapper
pkgs.patchelf
pkgs.pkg-config
];
buildInputs = runtimeLibs;
baseMitmCache = pkgs.gradle_9.fetchDeps {
mitmCache = pkgs.gradle_9.fetchDeps {
pkg = finalAttrs.finalPackage;
data = ./deps.json;
silent = false;
useBwrap = false;
};
mitmCache = pkgs.runCommand "jprototerm-deps-patched"
{
passthru.updateScript = finalAttrs.baseMitmCache.updateScript;
}
''
mkdir -p "$out"
cp -a ${finalAttrs.baseMitmCache}/. "$out/"
chmod -R u+w "$out"
substratePath="$out/https/plugins.gradle.org/m2/com/gluonhq/substrate/0.0.68/substrate-0.0.68.jar"
if [ ! -e "$substratePath" ]; then
echo "Could not find substrate jar in Gradle MITM cache: $substratePath" >&2
find "$out" -path '*substrate*' -print >&2
exit 1
fi
rm "$substratePath"
ln -s ${patchedSubstrateJar} "$substratePath"
'';
gradleBuildTask = "nativeBuild";
gradleUpdateTask = "nixDownloadDeps";
# Builds build/install/jprototerm/{bin,lib} with every runtime jar, including
# the maven javafx-*-linux jars that carry the platform natives.
gradleBuildTask = "installDist";
gradleFlags = [
"--no-build-cache"
"--stacktrace"
"--info"
"-Dorg.gradle.java.home=${gluonGraalvm}"
"-Dorg.gradle.java.home=${pkgs.jdk25}"
];
GRAALVM_HOME = "${gluonGraalvm}";
JAVA_HOME = "${gluonGraalvm}";
JAVA_HOME = "${pkgs.jdk25}";
JLIBGHOSTTY_MAVEN_REPO = "${jlib}/maven";
preConfigure = ''
export HOME="$TMPDIR/home"
export GRADLE_OPTS="-Duser.home=$HOME ''${GRADLE_OPTS:-}"
mkdir -p /tmp/nix
ln -sfn ${pkgs.pkg-config}/bin/pkg-config /tmp/nix/pkg-config
for gluonHome in "$HOME/.gluon" /build/.gluon; do
if mkdir -p "$gluonHome/substrate" 2>/dev/null; then
cp -f ${javafxStaticSdkZip} "$gluonHome/substrate/openjfx-21-ea+11.3-linux-x86_64-static.zip"
chmod u+w "$gluonHome/substrate/openjfx-21-ea+11.3-linux-x86_64-static.zip"
fi
done
'';
preBuild = ''
export HOME="$TMPDIR/home"
export GRADLE_OPTS="-Duser.home=$HOME ''${GRADLE_OPTS:-}"
mkdir -p /tmp/nix
ln -sfn ${pkgs.pkg-config}/bin/pkg-config /tmp/nix/pkg-config
for gluonHome in "$HOME/.gluon" /build/.gluon; do
if mkdir -p "$gluonHome/substrate" 2>/dev/null; then
cp -f ${javafxStaticSdkZip} "$gluonHome/substrate/openjfx-21-ea+11.3-linux-x86_64-static.zip"
chmod u+w "$gluonHome/substrate/openjfx-21-ea+11.3-linux-x86_64-static.zip"
fi
done
export LD_LIBRARY_PATH="${pkgs.lib.makeLibraryPath runtimeLibs}:$LD_LIBRARY_PATH"
'';
preGradleUpdate = ''
export HOME="$TMPDIR/home"
export LD_LIBRARY_PATH="${pkgs.lib.makeLibraryPath runtimeLibs}:$LD_LIBRARY_PATH"
'';
installPhase = ''
runHook preInstall
mkdir -p "$out/bin"
binary="$(find build/gluonfx -path '*/gvm/*' -prune -o -type f -perm -0100 -print | head -n1)"
if [ -z "$binary" ]; then
echo "Could not find native executable under build/gluonfx" >&2
find build/gluonfx -type f -perm -0100 >&2 || true
exit 1
fi
mkdir -p "$out/share/jprototerm"
cp -a build/install/jprototerm/lib "$out/share/jprototerm/lib"
cp "$binary" "$out/bin/jprototerm"
# JavaFX is a set of proper modular jars: put them on the module path and
# keep the application + plain dependency jars on the classpath, so the two
# worlds do not collide.
mkdir -p "$out/share/jprototerm/javafx"
mv "$out/share/jprototerm/lib"/javafx-*.jar "$out/share/jprototerm/javafx/"
wrapProgram "$out/bin/jprototerm" \
# Build an explicit colon-separated classpath. A "lib/*" glob would be
# expanded by the wrapper's shell before java sees it, breaking -cp.
classpath=""
for jar in "$out"/share/jprototerm/lib/*.jar; do
classpath="$classpath''${classpath:+:}$jar"
done
makeWrapper "${pkgs.jdk25}/bin/java" "$out/bin/jprototerm" \
--add-flags "--enable-native-access=ALL-UNNAMED,javafx.graphics" \
--add-flags "--module-path $out/share/jprototerm/javafx" \
--add-flags "--add-modules javafx.controls,javafx.fxml" \
--add-flags "-cp $classpath" \
--add-flags "com.gregor.jprototerm.Main" \
--prefix LD_LIBRARY_PATH : "${pkgs.lib.makeLibraryPath runtimeLibs}" \
--run 'glShimDir="''${XDG_RUNTIME_DIR:-/tmp}/jprototerm-gl"; mkdir -p "$glShimDir"; for lib in /lib/x86_64-linux-gnu/libGL.so.1 /lib/x86_64-linux-gnu/libGLX.so.0 /lib/x86_64-linux-gnu/libGLdispatch.so.0 /usr/lib/x86_64-linux-gnu/libGLX_nvidia.so* /usr/lib/x86_64-linux-gnu/libEGL_nvidia.so* /usr/lib/x86_64-linux-gnu/libnvidia*.so* /usr/lib/x86_64-linux-gnu/nvidia/current/lib*.so*; do [ -e "$lib" ] && ln -sfn "$lib" "$glShimDir/$(basename "$lib")"; done; export LD_LIBRARY_PATH="$glShimDir''${LD_LIBRARY_PATH:+:$LD_LIBRARY_PATH}"; export __GLX_VENDOR_LIBRARY_NAME="''${__GLX_VENDOR_LIBRARY_NAME:-nvidia}"; if [ -e /usr/share/glvnd/egl_vendor.d/10_nvidia.json ]; then export __EGL_VENDOR_LIBRARY_FILENAMES="''${__EGL_VENDOR_LIBRARY_FILENAMES:-/usr/share/glvnd/egl_vendor.d/10_nvidia.json}"; fi' \
--set JLIBGHOSTTY_LIBRARY "${ghosttyVt}/lib/libghostty-vt.so" \
@@ -238,13 +120,6 @@
runHook postInstall
'';
postFixup = ''
binary="$out/bin/.jprototerm-wrapped"
currentRpath="$(patchelf --print-rpath "$binary" || true)"
filteredRpath="$(printf '%s' "$currentRpath" | tr ':' '\n' | grep -v 'libglvnd' | paste -sd: -)"
patchelf --set-rpath "$filteredRpath" "$binary"
'';
});
in {
default = jprototerm;
@@ -256,35 +131,12 @@
pkgs = import nixpkgs { inherit system; };
jlib = jlibghostty.packages.${system}.jlibghostty;
ghosttyVt = ghostty.packages.${system}.libghostty-vt;
runtimeLibs = [
pkgs.glib
pkgs.gtk3
pkgs.pango
pkgs.alsa-lib
pkgs.ffmpeg.dev
pkgs.ffmpeg.lib
pkgs.freetype
pkgs.fontconfig
pkgs.libx11
pkgs.libx11.dev
pkgs.libxext
pkgs.libxrender
pkgs.libxtst
pkgs.libxi
pkgs.libxcursor
pkgs.libxrandr
pkgs.libxinerama
pkgs.libxcb
pkgs.libxxf86vm
pkgs.zlib
pkgs.zlib.dev
ghosttyVt
];
runtimeLibs = runtimeLibsFor pkgs ghosttyVt;
in {
default = pkgs.mkShell {
packages = [
pkgs.gradle_9
pkgs.jdk23
pkgs.jdk25
pkgs.jdt-language-server
] ++ runtimeLibs;