Jump to content

Packaging/32bit Applications

From Official NixOS Wiki

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";
};