2020-04-24 23:36:52 +00:00
|
|
|
|
/* This file contains various functions that take a stdenv and return
|
|
|
|
|
a new stdenv with different behaviour, e.g. using a different C
|
|
|
|
|
compiler. */
|
|
|
|
|
|
2021-08-22 07:53:02 +00:00
|
|
|
|
{ lib, pkgs, config }:
|
|
|
|
|
|
|
|
|
|
let
|
|
|
|
|
# N.B. Keep in sync with default arg for stdenv/generic.
|
|
|
|
|
defaultMkDerivationFromStdenv = import ./generic/make-derivation.nix { inherit lib config; };
|
|
|
|
|
|
|
|
|
|
# Low level function to help with overriding `mkDerivationFromStdenv`. One
|
|
|
|
|
# gives it the old stdenv arguments and a "continuation" function, and
|
|
|
|
|
# underneath the final stdenv argument it yields to the continuation to do
|
|
|
|
|
# whatever it wants with old `mkDerivation` (old `mkDerivationFromStdenv`
|
|
|
|
|
# applied to the *new, final* stdenv) provided for convenience.
|
|
|
|
|
withOldMkDerivation = stdenvSuperArgs: k: stdenvSelf: let
|
|
|
|
|
mkDerivationFromStdenv-super = stdenvSuperArgs.mkDerivationFromStdenv or defaultMkDerivationFromStdenv;
|
|
|
|
|
mkDerivationSuper = mkDerivationFromStdenv-super stdenvSelf;
|
|
|
|
|
in
|
|
|
|
|
k stdenvSelf mkDerivationSuper;
|
|
|
|
|
|
|
|
|
|
# Wrap the original `mkDerivation` providing extra args to it.
|
|
|
|
|
extendMkDerivationArgs = old: f: withOldMkDerivation old (_: mkDerivationSuper: args:
|
|
|
|
|
mkDerivationSuper (args // f args));
|
|
|
|
|
|
|
|
|
|
# Wrap the original `mkDerivation` transforming the result.
|
|
|
|
|
overrideMkDerivationResult = old: f: withOldMkDerivation old (_: mkDerivationSuper: args:
|
|
|
|
|
f (mkDerivationSuper args));
|
|
|
|
|
in
|
2020-04-24 23:36:52 +00:00
|
|
|
|
|
|
|
|
|
rec {
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# Override the compiler in stdenv for specific packages.
|
|
|
|
|
overrideCC = stdenv: cc: stdenv.override { allowedRequisites = null; cc = cc; };
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# Add some arbitrary packages to buildInputs for specific packages.
|
|
|
|
|
# Used to override packages in stdenv like Make. Should not be used
|
|
|
|
|
# for other dependencies.
|
|
|
|
|
overrideInStdenv = stdenv: pkgs:
|
2020-10-07 09:15:18 +00:00
|
|
|
|
stdenv.override (prev: { allowedRequisites = null; extraBuildInputs = (prev.extraBuildInputs or []) ++ pkgs; });
|
2020-04-24 23:36:52 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# Override the setup script of stdenv. Useful for testing new
|
|
|
|
|
# versions of the setup script without causing a rebuild of
|
|
|
|
|
# everything.
|
|
|
|
|
#
|
|
|
|
|
# Example:
|
|
|
|
|
# randomPkg = import ../bla { ...
|
|
|
|
|
# stdenv = overrideSetup stdenv ../stdenv/generic/setup-latest.sh;
|
|
|
|
|
# };
|
|
|
|
|
overrideSetup = stdenv: setupScript: stdenv.override { inherit setupScript; };
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# Return a modified stdenv that tries to build statically linked
|
|
|
|
|
# binaries.
|
2021-08-22 07:53:02 +00:00
|
|
|
|
makeStaticBinaries = stdenv0:
|
|
|
|
|
stdenv0.override (old: {
|
|
|
|
|
mkDerivationFromStdenv = withOldMkDerivation old (stdenv: mkDerivationSuper: args:
|
|
|
|
|
if stdenv.hostPlatform.isDarwin
|
2020-04-24 23:36:52 +00:00
|
|
|
|
then throw "Cannot build fully static binaries on Darwin/macOS"
|
2021-08-22 07:53:02 +00:00
|
|
|
|
else mkDerivationSuper (args // {
|
2020-04-24 23:36:52 +00:00
|
|
|
|
NIX_CFLAGS_LINK = toString (args.NIX_CFLAGS_LINK or "") + " -static";
|
2021-08-22 07:53:02 +00:00
|
|
|
|
} // lib.optionalAttrs (!(args.dontAddStaticConfigureFlags or false)) {
|
2020-04-24 23:36:52 +00:00
|
|
|
|
configureFlags = (args.configureFlags or []) ++ [
|
|
|
|
|
"--disable-shared" # brrr...
|
|
|
|
|
];
|
2021-08-22 07:53:02 +00:00
|
|
|
|
}));
|
|
|
|
|
} // lib.optionalAttrs (stdenv0.hostPlatform.libc == "libc") {
|
|
|
|
|
extraBuildInputs = (old.extraBuildInputs or []) ++ [
|
|
|
|
|
stdenv0.glibc.static
|
|
|
|
|
];
|
|
|
|
|
});
|
2020-04-24 23:36:52 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# Return a modified stdenv that builds static libraries instead of
|
|
|
|
|
# shared libraries.
|
2021-08-22 07:53:02 +00:00
|
|
|
|
makeStaticLibraries = stdenv:
|
|
|
|
|
stdenv.override (old: {
|
|
|
|
|
mkDerivationFromStdenv = extendMkDerivationArgs old (args: {
|
2020-04-24 23:36:52 +00:00
|
|
|
|
dontDisableStatic = true;
|
2021-08-22 07:53:02 +00:00
|
|
|
|
} // lib.optionalAttrs (!(args.dontAddStaticConfigureFlags or false)) {
|
2020-04-24 23:36:52 +00:00
|
|
|
|
configureFlags = (args.configureFlags or []) ++ [
|
|
|
|
|
"--enable-static"
|
|
|
|
|
"--disable-shared"
|
|
|
|
|
];
|
|
|
|
|
cmakeFlags = (args.cmakeFlags or []) ++ [ "-DBUILD_SHARED_LIBS:BOOL=OFF" ];
|
|
|
|
|
mesonFlags = (args.mesonFlags or []) ++ [ "-Ddefault_library=static" ];
|
|
|
|
|
});
|
2021-08-22 07:53:02 +00:00
|
|
|
|
});
|
2020-04-24 23:36:52 +00:00
|
|
|
|
|
2021-09-18 10:52:07 +00:00
|
|
|
|
# Best effort static binaries. Will still be linked to libSystem,
|
|
|
|
|
# but more portable than Nix store binaries.
|
|
|
|
|
makeStaticDarwin = stdenv: stdenv.override (old: {
|
|
|
|
|
# extraBuildInputs are dropped in cross.nix, but darwin still needs them
|
|
|
|
|
extraBuildInputs = [ pkgs.buildPackages.darwin.CF ];
|
|
|
|
|
mkDerivationFromStdenv = extendMkDerivationArgs old (args: {
|
|
|
|
|
NIX_CFLAGS_LINK = toString (args.NIX_CFLAGS_LINK or "")
|
|
|
|
|
+ lib.optionalString (stdenv.cc.isGNU or false) " -static-libgcc";
|
|
|
|
|
nativeBuildInputs = (args.nativeBuildInputs or []) ++ [
|
|
|
|
|
(pkgs.buildPackages.makeSetupHook {
|
|
|
|
|
substitutions = {
|
|
|
|
|
libsystem = "${stdenv.cc.libc}/lib/libSystem.B.dylib";
|
|
|
|
|
};
|
|
|
|
|
} ./darwin/portable-libsystem.sh)
|
|
|
|
|
];
|
|
|
|
|
});
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
# Puts all the other ones together
|
|
|
|
|
makeStatic = stdenv: lib.foldl (lib.flip lib.id) stdenv (
|
|
|
|
|
lib.optional stdenv.hostPlatform.isDarwin makeStaticDarwin
|
|
|
|
|
|
|
|
|
|
++ [ makeStaticLibraries propagateBuildInputs ]
|
|
|
|
|
|
|
|
|
|
# Apple does not provide a static version of libSystem or crt0.o
|
|
|
|
|
# So we can’t build static binaries without extensive hacks.
|
|
|
|
|
++ lib.optional (!stdenv.hostPlatform.isDarwin) makeStaticBinaries
|
|
|
|
|
|
|
|
|
|
# Glibc doesn’t come with static runtimes by default.
|
|
|
|
|
# ++ lib.optional (stdenv.hostPlatform.libc == "glibc") ((lib.flip overrideInStdenv) [ self.stdenv.glibc.static ])
|
|
|
|
|
);
|
|
|
|
|
|
2020-04-24 23:36:52 +00:00
|
|
|
|
|
|
|
|
|
/* Modify a stdenv so that all buildInputs are implicitly propagated to
|
|
|
|
|
consuming derivations
|
|
|
|
|
*/
|
2021-08-22 07:53:02 +00:00
|
|
|
|
propagateBuildInputs = stdenv:
|
|
|
|
|
stdenv.override (old: {
|
|
|
|
|
mkDerivationFromStdenv = extendMkDerivationArgs old (args: {
|
2020-04-24 23:36:52 +00:00
|
|
|
|
propagatedBuildInputs = (args.propagatedBuildInputs or []) ++ (args.buildInputs or []);
|
|
|
|
|
buildInputs = [];
|
|
|
|
|
});
|
2021-08-22 07:53:02 +00:00
|
|
|
|
});
|
2020-04-24 23:36:52 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Modify a stdenv so that the specified attributes are added to
|
|
|
|
|
every derivation returned by its mkDerivation function.
|
|
|
|
|
|
|
|
|
|
Example:
|
|
|
|
|
stdenvNoOptimise =
|
|
|
|
|
addAttrsToDerivation
|
|
|
|
|
{ NIX_CFLAGS_COMPILE = "-O0"; }
|
|
|
|
|
stdenv;
|
|
|
|
|
*/
|
2021-08-22 07:53:02 +00:00
|
|
|
|
addAttrsToDerivation = extraAttrs: stdenv: stdenv.override (old: {
|
|
|
|
|
mkDerivationFromStdenv = extendMkDerivationArgs old (_: extraAttrs);
|
|
|
|
|
});
|
2020-04-24 23:36:52 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Return a modified stdenv that builds packages with GCC's coverage
|
|
|
|
|
instrumentation. The coverage note files (*.gcno) are stored in
|
|
|
|
|
$out/.build, along with the source code of the package, to enable
|
|
|
|
|
programs like lcov to produce pretty-printed reports.
|
|
|
|
|
*/
|
|
|
|
|
addCoverageInstrumentation = stdenv:
|
|
|
|
|
overrideInStdenv stdenv [ pkgs.enableGCOVInstrumentation pkgs.keepBuildTree ];
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Replace the meta.maintainers field of a derivation. This is useful
|
|
|
|
|
when you want to fork to update some packages without disturbing other
|
|
|
|
|
developers.
|
|
|
|
|
|
|
|
|
|
e.g.: in all-packages.nix:
|
|
|
|
|
|
|
|
|
|
# remove all maintainers.
|
|
|
|
|
defaultStdenv = replaceMaintainersField allStdenvs.stdenv pkgs [];
|
|
|
|
|
*/
|
2021-08-22 07:53:02 +00:00
|
|
|
|
replaceMaintainersField = stdenv: pkgs: maintainers:
|
|
|
|
|
stdenv.override (old: {
|
|
|
|
|
mkDerivationFromStdenv = overrideMkDerivationResult (pkg:
|
|
|
|
|
lib.recursiveUpdate pkg { meta.maintainers = maintainers; });
|
|
|
|
|
});
|
2020-04-24 23:36:52 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Use the trace output to report all processed derivations with their
|
|
|
|
|
license name.
|
|
|
|
|
*/
|
2021-08-22 07:53:02 +00:00
|
|
|
|
traceDrvLicenses = stdenv:
|
|
|
|
|
stdenv.override (old: {
|
|
|
|
|
mkDerivationFromStdenv = overrideMkDerivationResult (pkg:
|
2020-04-24 23:36:52 +00:00
|
|
|
|
let
|
|
|
|
|
printDrvPath = val: let
|
|
|
|
|
drvPath = builtins.unsafeDiscardStringContext pkg.drvPath;
|
|
|
|
|
license = pkg.meta.license or null;
|
|
|
|
|
in
|
|
|
|
|
builtins.trace "@:drv:${toString drvPath}:${builtins.toString license}:@" val;
|
|
|
|
|
in pkg // {
|
|
|
|
|
outPath = printDrvPath pkg.outPath;
|
|
|
|
|
drvPath = printDrvPath pkg.drvPath;
|
2021-08-22 07:53:02 +00:00
|
|
|
|
});
|
|
|
|
|
});
|
2020-04-24 23:36:52 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Abort if the license predicate is not verified for a derivation
|
|
|
|
|
declared with mkDerivation.
|
|
|
|
|
|
|
|
|
|
One possible predicate to avoid all non-free packages can be achieved
|
|
|
|
|
with the following function:
|
|
|
|
|
|
|
|
|
|
isFree = license: with builtins;
|
|
|
|
|
if license == null then true
|
|
|
|
|
else if isList license then lib.all isFree license
|
|
|
|
|
else license != "non-free" && license != "unfree";
|
|
|
|
|
|
|
|
|
|
This adapter can be defined on the defaultStdenv definition. You can
|
|
|
|
|
use it by patching the all-packages.nix file or by using the override
|
|
|
|
|
feature of ~/.config/nixpkgs/config.nix .
|
|
|
|
|
*/
|
2021-08-22 07:53:02 +00:00
|
|
|
|
validateLicenses = licensePred: stdenv:
|
|
|
|
|
stdenv.override (old: {
|
|
|
|
|
mkDerivationFromStdenv = overrideMkDerivationResult (pkg:
|
2020-04-24 23:36:52 +00:00
|
|
|
|
let
|
|
|
|
|
drv = builtins.unsafeDiscardStringContext pkg.drvPath;
|
|
|
|
|
license =
|
|
|
|
|
pkg.meta.license or
|
|
|
|
|
# Fixed-output derivations such as source tarballs usually
|
|
|
|
|
# don't have licensing information, but that's OK.
|
|
|
|
|
(pkg.outputHash or
|
|
|
|
|
(builtins.trace
|
|
|
|
|
"warning: ${drv} lacks licensing information" null));
|
|
|
|
|
|
|
|
|
|
validate = arg:
|
|
|
|
|
if licensePred license then arg
|
|
|
|
|
else abort ''
|
|
|
|
|
while building ${drv}:
|
|
|
|
|
license `${builtins.toString license}' does not pass the predicate.
|
|
|
|
|
'';
|
|
|
|
|
|
|
|
|
|
in pkg // {
|
|
|
|
|
outPath = validate pkg.outPath;
|
|
|
|
|
drvPath = validate pkg.drvPath;
|
2021-08-22 07:53:02 +00:00
|
|
|
|
});
|
|
|
|
|
});
|
2020-04-24 23:36:52 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Modify a stdenv so that it produces debug builds; that is,
|
|
|
|
|
binaries have debug info, and compiler optimisations are
|
|
|
|
|
disabled. */
|
2021-08-22 07:53:02 +00:00
|
|
|
|
keepDebugInfo = stdenv:
|
|
|
|
|
stdenv.override (old: {
|
|
|
|
|
mkDerivationFromStdenv = extendMkDerivationArgs old (args: {
|
2020-04-24 23:36:52 +00:00
|
|
|
|
dontStrip = true;
|
|
|
|
|
NIX_CFLAGS_COMPILE = toString (args.NIX_CFLAGS_COMPILE or "") + " -ggdb -Og";
|
|
|
|
|
});
|
2021-08-22 07:53:02 +00:00
|
|
|
|
});
|
2020-04-24 23:36:52 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Modify a stdenv so that it uses the Gold linker. */
|
2021-08-22 07:53:02 +00:00
|
|
|
|
useGoldLinker = stdenv:
|
|
|
|
|
stdenv.override (old: {
|
|
|
|
|
mkDerivationFromStdenv = extendMkDerivationArgs old (args: {
|
2020-04-24 23:36:52 +00:00
|
|
|
|
NIX_CFLAGS_LINK = toString (args.NIX_CFLAGS_LINK or "") + " -fuse-ld=gold";
|
|
|
|
|
});
|
2021-08-22 07:53:02 +00:00
|
|
|
|
});
|
2020-04-24 23:36:52 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Modify a stdenv so that it builds binaries optimized specifically
|
|
|
|
|
for the machine they are built on.
|
|
|
|
|
|
|
|
|
|
WARNING: this breaks purity! */
|
2021-08-22 07:53:02 +00:00
|
|
|
|
impureUseNativeOptimizations = stdenv:
|
|
|
|
|
stdenv.override (old: {
|
|
|
|
|
mkDerivationFromStdenv = extendMkDerivationArgs old (args: {
|
2020-04-24 23:36:52 +00:00
|
|
|
|
NIX_CFLAGS_COMPILE = toString (args.NIX_CFLAGS_COMPILE or "") + " -march=native";
|
|
|
|
|
NIX_ENFORCE_NO_NATIVE = false;
|
|
|
|
|
|
|
|
|
|
preferLocalBuild = true;
|
|
|
|
|
allowSubstitutes = false;
|
|
|
|
|
});
|
2021-08-22 07:53:02 +00:00
|
|
|
|
});
|
2022-02-10 20:34:41 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Modify a stdenv so that it builds binaries with the specified list of
|
|
|
|
|
compilerFlags appended and passed to the compiler.
|
|
|
|
|
|
|
|
|
|
This example would recompile every derivation on the system with
|
|
|
|
|
-funroll-loops and -O3 passed to each gcc invocation.
|
|
|
|
|
|
|
|
|
|
Example:
|
|
|
|
|
nixpkgs.overlays = [
|
|
|
|
|
(self: super: {
|
|
|
|
|
stdenv = super.withCFlags [ "-funroll-loops" "-O3" ] super.stdenv;
|
|
|
|
|
})
|
|
|
|
|
];
|
|
|
|
|
*/
|
|
|
|
|
withCFlags = compilerFlags: stdenv:
|
|
|
|
|
stdenv.override (old: {
|
|
|
|
|
mkDerivationFromStdenv = extendMkDerivationArgs old (args: {
|
|
|
|
|
NIX_CFLAGS_COMPILE = toString (args.NIX_CFLAGS_COMPILE or "") + " ${toString compilerFlags}";
|
|
|
|
|
});
|
|
|
|
|
});
|
2020-04-24 23:36:52 +00:00
|
|
|
|
}
|