Packaging/32bit Applications: Difference between revisions
Added Rust and LTO Cross Compile Notes |
|||
| (One intermediate revision by the same user not shown) | |||
| Line 24: | Line 24: | ||
clang_multi | clang_multi | ||
]; | ]; | ||
</syntaxhighlight>An example that will let you build with <code>cargo test --target i686-unknown-linux-gnu</code>. | </syntaxhighlight>An example that will let you build with <code>cargo test --target i686-unknown-linux-gnu</code> from an x86_64 machine. | ||
You will want something similar to the following in your <code>flake.nix</code>, <code>devenv.sh</code> , or whatever you're using as your entry point. | You will want something similar to the following in your <code>flake.nix</code>, <code>devenv.sh</code> , or whatever you're using as your entry point. | ||
==== Rust LTO & cargo bench ==== | |||
If you are using LTO and are having trouble building benchmarks (<code>cargo bench</code>) you may need to disable it at the build step until someone documents a better workaround here.<syntaxhighlight lang="nix"> | |||
env = { | |||
# Disable LTO for C/C++ code compiled by the `cc` crate for i686. | |||
# Some crates (like `alloca`) add -flto when CC=clang, but binutils ld can't | |||
# process LLVM IR bitcode. Using -fno-lto ensures native object files. | |||
CFLAGS_i686_unknown_linux_gnu = "-fno-lto"; | |||
CXXFLAGS_i686_unknown_linux_gnu = "-fno-lto"; | |||
}; | |||
</syntaxhighlight> | |||
[[Category:Development]] | [[Category:Development]] | ||
Latest revision as of 18:01, 13 December 2025
Building software with 32bit gcc
Question: I'm invoking gcc with -m32 and it fails to find `gnu/stubs-32.h`
Answer (clever): you want to use pkgsi686Linux instead of pkgs, so things like pkgsi686Linux.stdenv.mkDerivation or pkgsi686Linux.callPackage then nix will give you 32bit everything[1]
Building software with both 32- and 64-bit executables
If a package wants to compile both 32/64-bit executables, you need a compiler with multilib support.
Nixpkgs provides multiStdenv.mkDerivation that should be used instead stdenv.mkDerivation.
This is equivalent to using gcc-multilib in debian derivatives.
Rust
env = {
# Use multilib clang as linker for i686 target.
# Required because Rust's bundled LLD doesn't know about Nix's multilib sysroot paths.
# clang_multi has the correct 32-bit library paths (glibc_multi) baked into its driver.
CARGO_TARGET_I686_UNKNOWN_LINUX_GNU_LINKER = "${pkgs.clang_multi}/bin/clang";
};
packages = with pkgs; [
# clang with multilib support (both 32-bit and 64-bit)
# This properly handles -m32 compilation with correct headers/libs
clang_multi
];
An example that will let you build with cargo test --target i686-unknown-linux-gnu from an x86_64 machine.
You will want something similar to the following in your flake.nix, devenv.sh , or whatever you're using as your entry point.
Rust LTO & cargo bench
If you are using LTO and are having trouble building benchmarks (cargo bench) you may need to disable it at the build step until someone documents a better workaround here.
env = {
# Disable LTO for C/C++ code compiled by the `cc` crate for i686.
# Some crates (like `alloca`) add -flto when CC=clang, but binutils ld can't
# process LLVM IR bitcode. Using -fno-lto ensures native object files.
CFLAGS_i686_unknown_linux_gnu = "-fno-lto";
CXXFLAGS_i686_unknown_linux_gnu = "-fno-lto";
};