Is there a way to have same derivation hashes for the same application built for the same host system which are known to build exactly the same application but built on different systems if the derivation is not a FOD?
Why is the target system a whole nixpkgs level thing and not just an option you (or stdenv) override for gcc, clang, rustc, etc.?
Example: You are on x86_64-linux and you cross compile hello from aarch64-linux Nixpkgs.
If I’m understanding this correctly and if pkgs.gcc supported it, pkgs.stdenv could just use pkgs.gcc.override { targetSystem = "aarch64-linux"; } instead of pkgs.gcc + global variables.
This has something to do with how Nix itself works, and the Nixpkgs stdenv is only a thin abstraction in that regard: as you probably already know, derivation hashes that aren’t fixed-output are determined by their inputs, and the system string is one of them. That parameter, as you will find in the linked reference documentation, determines the system type on which Nix will build the derivation locally. Nix itself has no notion of cross compilation, this is only introduced in stdenv, at the layer of the builder executable which in this case is a Bash script cobbled together from a pile of things such as compiler flags, the entirety of which will again constitute an input to the resulting derivation. But even mkDerivation has to set system on the underlying call to builtins.derivation, since that build has to run in some execution environment.
Therefore, without FOD or content-addressed derivations, you will necessarily get a different hash if you change parameters.
Nixpkgs is now structured so that each deps<host><target> is automatically taken from pkgs<host><target>. (These pkgs<host><target>s are quite new, so there is no special case for nativeBuildInputs and buildInputs.)
I have couple of questions
Could somebody explain more about what is said in the paranthesis?
Does stdenv.mkDerivation do black magic behind the scenes and swap out inputs of packages or for the example, the overridable functions around derivations?
{ lib, # This is the "overridable function" I'm referring to
stdenv,
openssl,
dtc,
pkgsBuildHost
}:
stdenv.mkDerivation {
buildInputs = [ # or depsHostTarget for that matter
openssl # I'm guessing this comes from raw pkgs without any tricks
];
nativeBuildInputs = [ # or depsBuildHost
dtc # Is this the same as putting `dtc.override pkgsBuildHost` in buildInputs or as a store path in buildPhase?
];
preBuildPhase = ''
echo ${openssl} # Is this from raw pkgs aka pkgsHostTarget?
'';
}
Is the “default target system” the same as the host system of Nixpkgs? How and where in Nixpkgs code does the target system get overridden when cross-compiling?
A depends on B in nativeBuildInputs, let’s say as a compiler.
B’s host system has to be A’s build system and B’s target system has to be A’s host system.
When making the hash for B, how does Nix decide its build system?
Another example: If I build hello (and stdenv has no other dependencies than gcc) its hash depends on up to three systems?
The package inputs you receive are “spliced”; meaning that they contain the pkgsHostTarget and pkgsBuildHost version of themselves. mkDerivation automagically picks the correct one depending on which attribute they’re put into.
Overrides have nothing to do with this.
Correct. If you wanted to run this during cross, you’d need to take it from pkgsBuildHost/buildPackages or put it in nativeBuildInputs.
Yes, I believe so.
The target platform is only relevant when building code-generating components (i.e. a compiler). In order to build a cross-compiler, you need to be able to build an artifact that runs on some platform but generates code for another. The targetPlatform is the platform for which code is generated.
You must either specify it explicitly or it can be implicitly assumed to be the platform which Nix itself was built for in impure eval (builtins.currentSystem).
Correct. However note that, in the case of plain old pkgs.hello, gcc’s build system will be your host’s build system and that will also be the same as the host and target system.