forked from NixOS/nixpkgs
-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathadapters.nix
More file actions
429 lines (392 loc) · 13.2 KB
/
Copy pathadapters.nix
File metadata and controls
429 lines (392 loc) · 13.2 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
/*
This file contains various functions that take a stdenv and return
a new stdenv with different behaviour, e.g. using a different C
compiler.
*/
{
lib,
pkgs,
config,
}:
let
# N.B. Keep in sync with default arg for stdenv/generic.
defaultMkDerivationFromStdenv =
stdenv: (import ./generic/make-derivation.nix { inherit lib config; } stdenv).mkDerivation;
# 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).overrideAttrs f
);
# Wrap the original `mkDerivation` transforming the result.
overrideMkDerivationResult =
old: f:
withOldMkDerivation old (
_: mkDerivationSuper: args:
f (mkDerivationSuper args)
);
in
rec {
# Override the compiler in stdenv for specific packages.
overrideCC =
stdenv: cc:
stdenv.override {
allowedRequisites = null;
cc = cc;
hasCC = cc != null;
};
# 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:
stdenv.override (prev: {
allowedRequisites = null;
extraBuildInputs = (prev.extraBuildInputs or [ ]) ++ pkgs;
});
# 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.
makeStaticBinaries =
stdenv0:
stdenv0.override (
old:
{
mkDerivationFromStdenv = withOldMkDerivation old (
stdenv: mkDerivationSuper: args:
if stdenv.hostPlatform.isDarwin then
throw "Cannot build fully static binaries on Darwin/macOS"
else
(mkDerivationSuper args).overrideAttrs (
args:
(
if (args ? NIX_CFLAGS_LINK) then
lib.warn
(
"NIX_CFLAGS_LINK is an environment variable and should be defined inside `env`"
+ lib.optionalString (args ? pname) " for package ${args.pname}"
+ lib.optionalString (args ? version) "-${args.version}"
)
{
NIX_CFLAGS_LINK = toString (args.NIX_CFLAGS_LINK or "") + " -static";
}
else
{
env = (args.env or { }) // {
NIX_CFLAGS_LINK = toString (args.env.NIX_CFLAGS_LINK or "") + " -static";
};
}
)
// lib.optionalAttrs (!(args.dontAddStaticConfigureFlags or false)) {
configureFlags = (args.configureFlags or [ ]) ++ [
"--disable-shared" # brrr...
];
cmakeFlags = (args.cmakeFlags or [ ]) ++ [ "-DCMAKE_SKIP_INSTALL_RPATH=On" ];
}
)
);
}
// lib.optionalAttrs (stdenv0.hostPlatform.libc == "glibc") {
extraBuildInputs = (old.extraBuildInputs or [ ]) ++ [
pkgs.glibc.static
];
}
);
# Return a modified stdenv that builds static libraries instead of
# shared libraries.
makeStaticLibraries = overrideMkDerivationArgs (
args:
{
dontDisableStatic = true;
}
// lib.optionalAttrs (!(args.dontAddStaticConfigureFlags or false)) {
configureFlags = (args.configureFlags or [ ]) ++ [
"--enable-static"
"--disable-shared"
];
cmakeFlags = (args.cmakeFlags or [ ]) ++ [ "-DBUILD_SHARED_LIBS:BOOL=OFF" ];
mesonFlags = (args.mesonFlags or [ ]) ++ [
"-Ddefault_library=static"
"-Ddefault_both_libraries=static"
];
}
);
# Best effort static binaries. Will still be linked to libSystem,
# but more portable than Nix store binaries.
makeStaticDarwin =
stdenv:
stdenv.override (old: {
mkDerivationFromStdenv = withOldMkDerivation old (
stdenv: mkDerivationSuper: args:
(mkDerivationSuper args).overrideAttrs (
prevAttrs:
if prevAttrs ? env.NIX_CFLAGS_LINK then
{
env = prevAttrs.env // {
NIX_CFLAGS_LINK =
toString (args.env.NIX_CFLAGS_LINK or "")
+ lib.optionalString (stdenv.cc.isGNU or false) " -static-libgcc";
};
}
else
{
NIX_CFLAGS_LINK =
toString (prevAttrs.NIX_CFLAGS_LINK or "")
+ lib.optionalString (stdenv.cc.isGNU or false) " -static-libgcc";
}
)
);
});
# 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
);
/*
Modify a stdenv so that all buildInputs are implicitly propagated to
consuming derivations
*/
propagateBuildInputs = overrideMkDerivationArgs (args: {
propagatedBuildInputs = (args.propagatedBuildInputs or [ ]) ++ (args.buildInputs or [ ]);
buildInputs = [ ];
});
/*
Modify a stdenv so that the specified attributes are added to
every derivation returned by its mkDerivation function.
Example:
stdenvNoOptimise =
addAttrsToDerivation
{ env.NIX_CFLAGS_COMPILE = "-O0"; }
stdenv;
*/
addAttrsToDerivation = extraAttrs: overrideMkDerivationArgs (_: extraAttrs);
/*
Modify a stdenv so as to extend `mkDerivation`'s arguments.
A stronger version of `addAttrsToDerivation`.
Example:
requireCcache =
overrideMkDerivationArgs
(oldAttrs: {
requiredSystemFeatures = oldAttrs.requiredSystemFeatures or [ ] ++ [ "ccache" ];
});
*/
overrideMkDerivationArgs =
extension: stdenv:
stdenv.override (old: {
mkDerivationFromStdenv = extendMkDerivationArgs old extension;
});
/*
Use the trace output to report all processed derivations with their
license name.
*/
traceDrvLicenses =
stdenv:
stdenv.override (old: {
mkDerivationFromStdenv = overrideMkDerivationResult (
pkg:
let
printDrvPath =
val:
let
drvPath = builtins.unsafeDiscardStringContext pkg.drvPath;
license = pkg.meta.license or null;
in
builtins.trace "@:drv:${toString drvPath}:${toString license}:@" val;
in
pkg
// {
outPath = printDrvPath pkg.outPath;
drvPath = printDrvPath pkg.drvPath;
}
);
});
/*
Modify a stdenv so that it produces debug builds; that is,
binaries have debug info, and compiler optimisations are
disabled.
*/
keepDebugInfo = overrideMkDerivationArgs (args: {
dontStrip = true;
env = (args.env or { }) // {
NIX_CFLAGS_COMPILE = toString (args.env.NIX_CFLAGS_COMPILE or "") + " -ggdb -Og";
NIX_RUSTFLAGS = toString (args.env.NIX_RUSTFLAGS or "") + " -g -C opt-level=0 -C strip=none";
};
});
# Modify a stdenv so that it uses the Gold linker.
useGoldLinker = overrideMkDerivationArgs (args: {
env = (args.env or { }) // {
NIX_CFLAGS_LINK = toString (args.env.NIX_CFLAGS_LINK or "") + " -fuse-ld=gold";
};
});
/*
Copy the libstdc++ from the model stdenv to the target stdenv.
TODO(@connorbaker):
This interface provides behavior which should be revisited prior to the
release of 24.05. For a more detailed explanation and discussion, see
https://github.com/NixOS/nixpkgs/issues/283517.
*/
useLibsFrom =
modelStdenv: targetStdenv:
let
ccForLibs = modelStdenv.cc.cc;
/*
NOTE(@connorbaker):
This assumes targetStdenv.cc is a cc-wrapper.
*/
cc = targetStdenv.cc.override {
/*
NOTE(originally by rrbutani):
Normally the `useCcForLibs`/`gccForLibs` mechanism is used to get a
clang based `cc` to use `libstdc++` (from gcc).
Here we (ab)use it to use a `libstdc++` from a different `gcc` than our
`cc`.
Note that this does not inhibit our `cc`'s lib dir from being added to
cflags/ldflags (see `cc_solib` in `cc-wrapper`) but this is okay: our
`gccForLibs`'s paths should take precedence.
*/
useCcForLibs = true;
gccForLibs = ccForLibs;
};
in
overrideCC targetStdenv cc;
useMoldLinker =
stdenv:
if stdenv.targetPlatform.isDarwin then
throw "Mold can't be used to emit Mach-O (Darwin) binaries"
else
let
bintools = stdenv.cc.bintools.override {
extraBuildCommands = ''
pushd $out/bin
ln -s ${pkgs.buildPackages.mold}/bin/${stdenv.cc.bintools.targetPrefix}ld.mold ${stdenv.cc.bintools.targetPrefix}ld.mold
'' # Pre-generated configure scripts call the linker binary without the target prefix when cross compiling.
+ lib.optionalString (stdenv.cc.bintools.targetPrefix != "") ''
ln -s ${stdenv.cc.bintools.targetPrefix}ld.mold ld.mold
''
+ ''
popd
'';
};
in
stdenv.override (
old:
{
allowedRequisites = null;
cc = stdenv.cc.override { inherit bintools; };
# gcc >12.1.0 supports '-fuse-ld=mold'
# the wrap ld above in bintools supports gcc <12.1.0 and shouldn't harm >12.1.0
# https://github.com/rui314/mold#how-to-use
}
//
lib.optionalAttrs
(stdenv.cc.isClang || (stdenv.cc.isGNU && lib.versionAtLeast stdenv.cc.version "12"))
{
mkDerivationFromStdenv = extendMkDerivationArgs old (args: {
env = (args.env or { }) // {
NIX_CFLAGS_LINK = toString (args.env.NIX_CFLAGS_LINK or "") + " -fuse-ld=mold";
};
});
}
);
useWildLinker =
stdenv:
if !stdenv.targetPlatform.isLinux then
throw "Wild only supports building Linux ELF files from Linux hosts."
else
stdenv.override (prev: {
allowedRequisites = null;
cc = prev.cc.override {
bintools = prev.cc.bintools.override {
extraBuildCommands = ''
ln -fs ${pkgs.buildPackages.wild}/bin/* "$out/bin"
'';
};
};
});
/*
Modify a stdenv so that it builds binaries optimized specifically
for the machine they are built on.
WARNING: this breaks purity!
*/
impureUseNativeOptimizations = overrideMkDerivationArgs (args: {
env = (args.env or { }) // {
NIX_CFLAGS_COMPILE = toString (args.env.NIX_CFLAGS_COMPILE or "") + " -march=native";
};
NIX_ENFORCE_NO_NATIVE = false;
preferLocalBuild = true;
allowSubstitutes = false;
});
/*
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:
overrideMkDerivationArgs (args: {
env = (args.env or { }) // {
NIX_CFLAGS_COMPILE = toString (args.env.NIX_CFLAGS_COMPILE or "") + " ${toString compilerFlags}";
};
});
withDefaultHardeningFlags =
defaultHardeningFlags: stdenv:
let
bintools =
let
bintools' = stdenv.cc.bintools;
in
if bintools' ? override then
(bintools'.override {
inherit defaultHardeningFlags;
})
else
bintools';
in
stdenv.override (old: {
cc =
if stdenv.cc == null then
null
else
stdenv.cc.override {
inherit bintools;
};
allowedRequisites = lib.mapNullable (rs: rs ++ [ bintools ]) (stdenv.allowedRequisites or null);
});
}