Packaging/Binaries: Difference between revisions
imported>Makefu link to archive.org |
Phanirithvij (talk | contribs) m update url to wayback machine url |
||
(34 intermediate revisions by 15 users not shown) | |||
Line 1: | Line 1: | ||
Downloading a binary on NixOS | Downloading and attempting to run a binary on NixOS will almost never work. This is due to hard-coded paths in the executable. Unfortunately, almost all unfree and proprietary software comes in binary form - the main reason to include binaries is because no source code is available. | ||
This tutorial will guide you through packaging a binary executable. | This tutorial will guide you through packaging a binary executable. | ||
If a downloaded binary is not packaged for NixOS then running the binary may fail like the following: | |||
<syntaxHighlight lang=bash> | |||
$ ~/.local/share/nvim/mason/packages/rust-analyzer/rust-analyzer-x86_64-unknown-linux-gnu --help | |||
bash: /home/gdforj/.local/share/nvim/mason/packages/rust-analyzer/rust-analyzer-x86_64-unknown-linux-gnu: cannot execute: required file not found | |||
</syntaxHighlight> | |||
Which indicates the interpreter, [[#The Dynamic Loader|the dynamic loader]] in this case, could not be located. | |||
== Using AutoPatchelfHook == | |||
<code>autoPatchelfHook</code> is the current (as of 19.09) preferred way to package binaries. | |||
<syntaxHighlight lang=nix> | |||
{ stdenv, lib | |||
, fetchurl | |||
, alsaLib | |||
, openssl | |||
, zlib | |||
, pulseaudio | |||
, autoPatchelfHook | |||
}: | |||
stdenv.mkDerivation rec { | |||
pname = "studio-link"; | |||
version = "21.07.0"; | |||
src = fetchurl { | |||
url = "https://download.studio.link/releases/v${version}-stable/linux/studio-link-standalone-v${version}.tar.gz"; | |||
hash = "sha256-4CkijAlenhht8tyk3nBULaBPE0GBf6DVII699/RmmWI="; | |||
}; | |||
nativeBuildInputs = [ | |||
autoPatchelfHook | |||
]; | |||
buildInputs = [ | |||
alsaLib | |||
openssl | |||
zlib | |||
pulseaudio | |||
]; | |||
sourceRoot = "."; | |||
installPhase = '' | |||
runHook preInstall | |||
install -m755 -D studio-link-standalone-v${version} $out/bin/studio-link | |||
runHook postInstall | |||
''; | |||
meta = with lib; { | |||
homepage = "https://studio-link.com"; | |||
description = "Voip transfer"; | |||
platforms = platforms.linux; | |||
}; | |||
} | |||
</syntaxHighlight> | |||
See [https://github.com/NixOS/nixpkgs/commit/ea5787ad5291ee1c131326cb9c9fec03d359edff this commit], or the example in method 5 of [https://unix.stackexchange.com/a/522823 this answer], for more details. | |||
=== Additional Library Paths === | |||
Some packages may put their library files somewhere other than <code>$out/lib/</code> in the install phase, while <code>autoPatchelfHook</code> by default only looks in that one directory for every package in <code>buildInputs</code>. You can tell <code>autoPatchelfHook</code> to look in additional directories by passing it to the command <code>addAutoPatchelfSearchPath</code> in any phase before <code>autoPatchelfHook</code> is run. | |||
For example, | |||
<pre> | |||
preBuild = '' | |||
addAutoPatchelfSearchPath ${jre8}/lib/openjdk/jre/lib/ | |||
''; | |||
</pre> | |||
== Manual Method == | |||
This sections describes the manual method of packaging a binary. It serves as a reference on the packaging issue with binaries and how these issues can be solved in nix derivations. | |||
This tutorial is about how to patch the executable with <code>patchelf</code> which is sufficient for most cases. If no source is available for a program patchelf is the preferred way in nixpkgs to add support for the package. | This tutorial is about how to patch the executable with <code>patchelf</code> which is sufficient for most cases. If no source is available for a program patchelf is the preferred way in nixpkgs to add support for the package. | ||
However sometimes this is not enough, more hardcoded paths may be scattered all over the place. For this you may need to set up an FHSUserEnv, a Linux Standard Base style directory structure with a final <code>chroot</code> call to fixate all paths. For a tutorial on how to use this technique, check out [https://web.archive.org/web/ | However sometimes this is not enough, more hardcoded paths may be scattered all over the place. For this you may need to set up an FHSUserEnv, a Linux Standard Base style directory structure with a final <code>chroot</code> call to fixate all paths. For a tutorial on how to use this technique, check out [https://web.archive.org/web/20240125095859/http://reflexivereflection.com/posts/2015-02-28-deb-installation-nixos.html Anders Papittos blog post on installing debian packages in nixos]. | ||
== Starting Point == | === Starting Point === | ||
We want to package a tool called "MasterPDFEditor", the package for debian can be found at [http://get.code-industry.net/public/master-pdf-editor-4.3.10_qt5.amd64.deb] ([https://web.archive.org/web/20170914112947/http://get.code-industry.net/public/master-pdf-editor-4.3.10_qt5.amd64.deb archive.org mirror]). | We want to package a tool called "MasterPDFEditor", the package for debian can be found at [http://get.code-industry.net/public/master-pdf-editor-4.3.10_qt5.amd64.deb] ([https://web.archive.org/web/20170914112947/http://get.code-industry.net/public/master-pdf-editor-4.3.10_qt5.amd64.deb archive.org mirror]). | ||
This tutorial assumes you run nixos-17.09 or later (for <code>nix-index</code> to work). | This tutorial assumes you run nixos-17.09 or later (for <code>nix-index</code> to work). | ||
Line 14: | Line 86: | ||
stdenv.cc # stdenv.cc is required for setting $NIX_CC env | stdenv.cc # stdenv.cc is required for setting $NIX_CC env | ||
$ wget https://web.archive.org/web/20170914112947/http://get.code-industry.net/public/master-pdf-editor-4.3.10_qt5.amd64.deb | $ wget https://web.archive.org/web/20170914112947/http://get.code-industry.net/public/master-pdf-editor-4.3.10_qt5.amd64.deb | ||
# we extract data.tar.xz from the deb package and untar it | $ # we extract data.tar.xz from the deb package and untar it | ||
$ | $ dpkg-deb -x master-pdf-editor-4.3.10_qt5.amd64.deb . | ||
ls opt/master-pdf-editor-4/ | $ ls opt/master-pdf-editor-4/ | ||
fonts license.txt masterpdfeditor4.png templates | fonts license.txt masterpdfeditor4.png templates | ||
lang masterpdfeditor4 stamps | lang masterpdfeditor4 stamps | ||
# running the executable does not `just work` | $ # running the executable does not `just work` | ||
$ opt/master-pdf-editor-4/masterpdfeditor4 | $ opt/master-pdf-editor-4/masterpdfeditor4 | ||
bash: opt/master-pdf-editor-4/masterpdfeditor4: No such file or directory | bash: opt/master-pdf-editor-4/masterpdfeditor4: No such file or directory | ||
</syntaxHighlight> | </syntaxHighlight> | ||
== The Dynamic Loader == | == The Dynamic Loader == | ||
Line 68: | Line 141: | ||
libgcc_s.so.1 | libgcc_s.so.1 | ||
libc.so.6 | libc.so.6 | ||
# now that the interpreter path is patched we can use ldd to find the libraries which are currently not found (check '=> not found'): | $ # now that the interpreter path is patched we can use ldd to find the libraries which are currently not found (check '=> not found'): | ||
$ ldd masterpdfeditor4 | grep 'not found' | $ ldd masterpdfeditor4 | grep 'not found' | ||
libsane.so.1 => not found | libsane.so.1 => not found | ||
Line 82: | Line 155: | ||
All the libraries which are not yet found are the ones we need to add to the runtime search path of the executable (RPATH). Again we can use <code>patchelf</code> to do this. We will be using <code>nix-index</code> for finding the files we are looking for: | All the libraries which are not yet found are the ones we need to add to the runtime search path of the executable (RPATH). Again we can use <code>patchelf</code> to do this. We will be using <code>nix-index</code> for finding the files we are looking for: | ||
<syntaxHighlight lang="console"> | <syntaxHighlight lang="console"> | ||
# we generate the database index of all files in our channel first | $ # we generate the database index of all files in our channel first | ||
$ nix-index | $ nix-index | ||
+ querying available packages | + querying available packages | ||
Line 114: | Line 187: | ||
The next step is to create a library path for all these packages. We use nix-repl to resolve the paths: | The next step is to create a library path for all these packages. We use nix-repl to resolve the paths: | ||
<syntaxHighlight lang="console"> | <syntaxHighlight lang="console"> | ||
$ nix | $ nix repl '<nixpkgs>' | ||
# .out can be omitted because this is the default output for all packages | # .out can be omitted because this is the default output for all packages | ||
# makeLibraryPath outputs the correct path for each package to use as rpath | # makeLibraryPath outputs the correct path for each package to use as rpath | ||
Line 132: | Line 203: | ||
== Creating the Derivation for upstream Packaging == | == Creating the Derivation for upstream Packaging == | ||
Packaging is straight forward. We just have to add all the steps we did into a simple derivation file. We call it <code> | Packaging is straight forward. We just have to add all the steps we did into a simple derivation file. We call it <code>package.nix</code> and store it in the checked out nixpkgs repository at <code>pkgs/by-name/ma/master-pdf-editor</code> | ||
The content looks like this: | The content looks like this: | ||
<syntaxHighlight lang="nix"> | <syntaxHighlight lang="nix"> | ||
Line 141: | Line 212: | ||
src = fetchurl { | src = fetchurl { | ||
url = "http://get.code-industry.net/public/${ | url = "http://get.code-industry.net/public/master-pdf-editor-${version}_qt5.amd64.deb"; | ||
hash = "sha256-1z26qjhbiyz33rm7mp8ycgl5ka0v3v5lv5i5v0b5mx35arvx2zzy"; | |||
}; | }; | ||
sourceRoot = "."; | sourceRoot = "."; | ||
unpackCmd = | unpackCmd = "dpkg-deb -x master-pdf-editor-${version}_qt5.amd64.deb ."; | ||
dontConfigure = true; | |||
dontBuild = true; | |||
installPhase = '' | installPhase = '' | ||
runHook preInstall | |||
mkdir -p $out/bin | mkdir -p $out/bin | ||
cp -R usr/share opt $out/ | cp -R usr/share opt $out/ | ||
Line 160: | Line 232: | ||
# symlink the binary to bin/ | # symlink the binary to bin/ | ||
ln -s $out/opt/master-pdf-editor-4/masterpdfeditor4 $out/bin/masterpdfeditor4 | ln -s $out/opt/master-pdf-editor-4/masterpdfeditor4 $out/bin/masterpdfeditor4 | ||
runHook postInstall | |||
''; | ''; | ||
preFixup = let | preFixup = let | ||
Line 176: | Line 250: | ||
''; | ''; | ||
meta = with | meta = with lib; { | ||
homepage = https://code-industry.net/masterpdfeditor/; | homepage = https://code-industry.net/masterpdfeditor/; | ||
description = "a multifunctional PDF Editor"; | description = "a multifunctional PDF Editor"; | ||
license = licenses. | license = licenses.unfree; | ||
platforms = platforms.linux; | platforms = platforms.linux; | ||
maintainers = [ your_name ]; | maintainers = [ your_name ]; | ||
Line 187: | Line 261: | ||
Because we created a derivation which is meant to be called by callPackage we can build the package now only via: | Because we created a derivation which is meant to be called by callPackage we can build the package now only via: | ||
<code>nix-build -E '((import <nixpkgs> {}).callPackage (import ./ | <code>nix-build -E '((import <nixpkgs> {}).callPackage (import ./package.nix) { })' --keep-failed --no-out-link</code> | ||
== Creating a Pull Request == | |||
With this new package you can [[Pull request|create a pull request for nixpkgs]]. Be aware that binary distributions are frowned upon if the source is available. | |||
== autoPatchelfHook == | |||
<code>autoPatchelfHook</code> can make the manual written patchelf invocations unnecessary. [[Category:Tutorial]] | |||
[[Category:nixpkgs]] | |||
== Wrong file paths == | |||
Some programs will try to access hard-coded FHS file paths like <code>/usr/lib</code> or <code>/opt</code>. Mostly though, this will produce silent <code>No such file</code> errors, which can break the program. | |||
To make these errors visible, we can use <code>strace</code> | |||
== | <pre> | ||
strace --trace=file,process --follow-forks --string-limit=200 \ | |||
< | ./result/bin/some-program 2>strace.log | ||
cat strace.log | grep -e 'No such file' -e 'execve("' \ | |||
| grep -v -E -e '(open|stat|access)(at)?\(.*"/nix/store/' \ | |||
-e resumed -e '/etc/ld-nix.so.preload' | |||
</pre> | |||
=== Example === | |||
<pre> | |||
[pid 357679] openat(AT_FDCWD, "/opt/brother/Printers/hll3210cw/inf/lut/0600-k_cache17.bin", O_RDONLY) = -1 ENOENT (No such file or directory) | |||
[pid 357679] openat(AT_FDCWD, "0600-k_cache17.bin", O_RDONLY) = -1 ENOENT (No such file or directory) | |||
</pre> | |||
This means: process <code>357679</code> is trying to open file <code>0600-k_cache17.bin</code> either from the hard-coded path in <code>/opt/brother/Printers/hll3210cw/inf/lut</code> or from the current workdir. | |||
So, as a quick fix, we could change the working directory of the wrapped binary with [https://github.com/NixOS/nixpkgs/blob/master/pkgs/build-support/setup-hooks/make-wrapper.sh makeWrapper]: | |||
<pre> | |||
makeWrapper <executable> <out_path> --run 'cd /nix/store/path/to/workdir' | |||
</pre> | |||
=== Finding the failing process === | |||
How can we find which process is throwing the <code>No such file</code> error? Let's search for the process ID and the <code>exec</code> syscall: | |||
<pre> | |||
cat strace.log | grep 357679 | grep exec | |||
[pid 357679] execve("/nix/store/ybl1pacslmhci9zy5qv95hshdgz6ihjl-brother-hll3210cw-1.0.2-0/opt/brother/Printers/hll3210cw/lpd/brhll3210cwfilter", ["/nix/store/ybl1pacslmhci9zy5qv95"..., "-pi", "/nix/store/ybl1pacslmhci9zy5qv95"..., "-rc", "/nix/store/ybl1pacslmhci9zy5qv95"...], 0x8ba010 /* 132 vars */ <unfinished ...> | |||
</pre> | |||
== See also == | |||
[[ | * [[Steam#steam-run|steam-run]] as a quick way to run binaries | ||