1# Define the list of system with their properties.
2#
3# See https://clang.llvm.org/docs/CrossCompilation.html and
4# http://llvm.org/docs/doxygen/html/Triple_8cpp_source.html especially
5# Triple::normalize. Parsing should essentially act as a more conservative
6# version of that last function.
7#
8# Most of the types below come in "open" and "closed" pairs. The open ones
9# specify what information we need to know about systems in general, and the
10# closed ones are sub-types representing the whitelist of systems we support in
11# practice.
12#
13# Code in the remainder of nixpkgs shouldn't rely on the closed ones in
14# e.g. exhaustive cases. Its more a sanity check to make sure nobody defines
15# systems that overlap with existing ones and won't notice something amiss.
16#
17{ lib }:
18with lib.lists;
19with lib.types;
20with lib.attrsets;
21with lib.strings;
22with (import ./inspect.nix { inherit lib; }).predicates;
23
24let
25 inherit (lib.options) mergeOneOption;
26
27 setTypes = type:
28 mapAttrs (name: value:
29 assert type.check value;
30 setType type.name ({ inherit name; } // value));
31
32 # gnu-config will ignore the portion of a triple matching the
33 # regex `e?abi.*$` when determining the validity of a triple. In
34 # other words, `i386-linuxabichickenlips` is a valid triple.
35 removeAbiSuffix = x:
36 let match = builtins.match "(.*)e?abi.*" x;
37 in if match==null
38 then x
39 else lib.elemAt match 0;
40
41in
42
43rec {
44
45 ################################################################################
46
47 types.openSignificantByte = mkOptionType {
48 name = "significant-byte";
49 description = "Endianness";
50 merge = mergeOneOption;
51 };
52
53 types.significantByte = enum (attrValues significantBytes);
54
55 significantBytes = setTypes types.openSignificantByte {
56 bigEndian = {};
57 littleEndian = {};
58 };
59
60 ################################################################################
61
62 # Reasonable power of 2
63 types.bitWidth = enum [ 8 16 32 64 128 ];
64
65 ################################################################################
66
67 types.openCpuType = mkOptionType {
68 name = "cpu-type";
69 description = "instruction set architecture name and information";
70 merge = mergeOneOption;
71 check = x: types.bitWidth.check x.bits
72 && (if 8 < x.bits
73 then types.significantByte.check x.significantByte
74 else !(x ? significantByte));
75 };
76
77 types.cpuType = enum (attrValues cpuTypes);
78
79 cpuTypes = with significantBytes; setTypes types.openCpuType {
80 arm = { bits = 32; significantByte = littleEndian; family = "arm"; };
81 armv5tel = { bits = 32; significantByte = littleEndian; family = "arm"; version = "5"; arch = "armv5t"; };
82 armv6m = { bits = 32; significantByte = littleEndian; family = "arm"; version = "6"; arch = "armv6-m"; };
83 armv6l = { bits = 32; significantByte = littleEndian; family = "arm"; version = "6"; arch = "armv6"; };
84 armv7a = { bits = 32; significantByte = littleEndian; family = "arm"; version = "7"; arch = "armv7-a"; };
85 armv7r = { bits = 32; significantByte = littleEndian; family = "arm"; version = "7"; arch = "armv7-r"; };
86 armv7m = { bits = 32; significantByte = littleEndian; family = "arm"; version = "7"; arch = "armv7-m"; };
87 armv7l = { bits = 32; significantByte = littleEndian; family = "arm"; version = "7"; arch = "armv7"; };
88 armv8a = { bits = 32; significantByte = littleEndian; family = "arm"; version = "8"; arch = "armv8-a"; };
89 armv8r = { bits = 32; significantByte = littleEndian; family = "arm"; version = "8"; arch = "armv8-a"; };
90 armv8m = { bits = 32; significantByte = littleEndian; family = "arm"; version = "8"; arch = "armv8-m"; };
91 aarch64 = { bits = 64; significantByte = littleEndian; family = "arm"; version = "8"; arch = "armv8-a"; };
92 aarch64_be = { bits = 64; significantByte = bigEndian; family = "arm"; version = "8"; arch = "armv8-a"; };
93
94 i386 = { bits = 32; significantByte = littleEndian; family = "x86"; arch = "i386"; };
95 i486 = { bits = 32; significantByte = littleEndian; family = "x86"; arch = "i486"; };
96 i586 = { bits = 32; significantByte = littleEndian; family = "x86"; arch = "i586"; };
97 i686 = { bits = 32; significantByte = littleEndian; family = "x86"; arch = "i686"; };
98 x86_64 = { bits = 64; significantByte = littleEndian; family = "x86"; arch = "x86-64"; };
99
100 microblaze = { bits = 32; significantByte = bigEndian; family = "microblaze"; };
101 microblazeel = { bits = 32; significantByte = littleEndian; family = "microblaze"; };
102
103 mips = { bits = 32; significantByte = bigEndian; family = "mips"; };
104 mipsel = { bits = 32; significantByte = littleEndian; family = "mips"; };
105 mips64 = { bits = 64; significantByte = bigEndian; family = "mips"; };
106 mips64el = { bits = 64; significantByte = littleEndian; family = "mips"; };
107
108 mmix = { bits = 64; significantByte = bigEndian; family = "mmix"; };
109
110 m68k = { bits = 32; significantByte = bigEndian; family = "m68k"; };
111
112 powerpc = { bits = 32; significantByte = bigEndian; family = "power"; };
113 powerpc64 = { bits = 64; significantByte = bigEndian; family = "power"; };
114 powerpc64le = { bits = 64; significantByte = littleEndian; family = "power"; };
115 powerpcle = { bits = 32; significantByte = littleEndian; family = "power"; };
116
117 riscv32 = { bits = 32; significantByte = littleEndian; family = "riscv"; };
118 riscv64 = { bits = 64; significantByte = littleEndian; family = "riscv"; };
119
120 s390 = { bits = 32; significantByte = bigEndian; family = "s390"; };
121 s390x = { bits = 64; significantByte = bigEndian; family = "s390"; };
122
123 sparc = { bits = 32; significantByte = bigEndian; family = "sparc"; };
124 sparc64 = { bits = 64; significantByte = bigEndian; family = "sparc"; };
125
126 wasm32 = { bits = 32; significantByte = littleEndian; family = "wasm"; };
127 wasm64 = { bits = 64; significantByte = littleEndian; family = "wasm"; };
128
129 alpha = { bits = 64; significantByte = littleEndian; family = "alpha"; };
130
131 rx = { bits = 32; significantByte = littleEndian; family = "rx"; };
132 msp430 = { bits = 16; significantByte = littleEndian; family = "msp430"; };
133 avr = { bits = 8; family = "avr"; };
134
135 vc4 = { bits = 32; significantByte = littleEndian; family = "vc4"; };
136
137 or1k = { bits = 32; significantByte = bigEndian; family = "or1k"; };
138
139 loongarch64 = { bits = 64; significantByte = littleEndian; family = "loongarch"; };
140
141 javascript = { bits = 32; significantByte = littleEndian; family = "javascript"; };
142 };
143
144 # GNU build systems assume that older NetBSD architectures are using a.out.
145 gnuNetBSDDefaultExecFormat = cpu:
146 if (cpu.family == "arm" && cpu.bits == 32) ||
147 (cpu.family == "sparc" && cpu.bits == 32) ||
148 (cpu.family == "m68k" && cpu.bits == 32) ||
149 (cpu.family == "x86" && cpu.bits == 32)
150 then execFormats.aout
151 else execFormats.elf;
152
153 # Determine when two CPUs are compatible with each other. That is,
154 # can code built for system B run on system A? For that to happen,
155 # the programs that system B accepts must be a subset of the
156 # programs that system A accepts.
157 #
158 # We have the following properties of the compatibility relation,
159 # which must be preserved when adding compatibility information for
160 # additional CPUs.
161 # - (reflexivity)
162 # Every CPU is compatible with itself.
163 # - (transitivity)
164 # If A is compatible with B and B is compatible with C then A is compatible with C.
165 #
166 # Note: Since 22.11 the archs of a mode switching CPU are no longer considered
167 # pairwise compatible. Mode switching implies that binaries built for A
168 # and B respectively can't be executed at the same time.
169 isCompatible = a: b: with cpuTypes; lib.any lib.id [
170 # x86
171 (b == i386 && isCompatible a i486)
172 (b == i486 && isCompatible a i586)
173 (b == i586 && isCompatible a i686)
174
175 # XXX: Not true in some cases. Like in WSL mode.
176 (b == i686 && isCompatible a x86_64)
177
178 # ARMv4
179 (b == arm && isCompatible a armv5tel)
180
181 # ARMv5
182 (b == armv5tel && isCompatible a armv6l)
183
184 # ARMv6
185 (b == armv6l && isCompatible a armv6m)
186 (b == armv6m && isCompatible a armv7l)
187
188 # ARMv7
189 (b == armv7l && isCompatible a armv7a)
190 (b == armv7l && isCompatible a armv7r)
191 (b == armv7l && isCompatible a armv7m)
192
193 # ARMv8
194 (b == aarch64 && a == armv8a)
195 (b == armv8a && isCompatible a aarch64)
196 (b == armv8r && isCompatible a armv8a)
197 (b == armv8m && isCompatible a armv8a)
198
199 # PowerPC
200 (b == powerpc && isCompatible a powerpc64)
201 (b == powerpcle && isCompatible a powerpc64le)
202
203 # MIPS
204 (b == mips && isCompatible a mips64)
205 (b == mipsel && isCompatible a mips64el)
206
207 # RISCV
208 (b == riscv32 && isCompatible a riscv64)
209
210 # SPARC
211 (b == sparc && isCompatible a sparc64)
212
213 # WASM
214 (b == wasm32 && isCompatible a wasm64)
215
216 # identity
217 (b == a)
218 ];
219
220 ################################################################################
221
222 types.openVendor = mkOptionType {
223 name = "vendor";
224 description = "vendor for the platform";
225 merge = mergeOneOption;
226 };
227
228 types.vendor = enum (attrValues vendors);
229
230 vendors = setTypes types.openVendor {
231 apple = {};
232 pc = {};
233 knuth = {};
234
235 # Actually matters, unlocking some MinGW-w64-specific options in GCC. See
236 # bottom of https://sourceforge.net/p/mingw-w64/wiki2/Unicode%20apps/
237 w64 = {};
238
239 none = {};
240 unknown = {};
241 };
242
243 ################################################################################
244
245 types.openExecFormat = mkOptionType {
246 name = "exec-format";
247 description = "executable container used by the kernel";
248 merge = mergeOneOption;
249 };
250
251 types.execFormat = enum (attrValues execFormats);
252
253 execFormats = setTypes types.openExecFormat {
254 aout = {}; # a.out
255 elf = {};
256 macho = {};
257 pe = {};
258 wasm = {};
259
260 unknown = {};
261 };
262
263 ################################################################################
264
265 types.openKernelFamily = mkOptionType {
266 name = "exec-format";
267 description = "executable container used by the kernel";
268 merge = mergeOneOption;
269 };
270
271 types.kernelFamily = enum (attrValues kernelFamilies);
272
273 kernelFamilies = setTypes types.openKernelFamily {
274 bsd = {};
275 darwin = {};
276 };
277
278 ################################################################################
279
280 types.openKernel = mkOptionType {
281 name = "kernel";
282 description = "kernel name and information";
283 merge = mergeOneOption;
284 check = x: types.execFormat.check x.execFormat
285 && all types.kernelFamily.check (attrValues x.families);
286 };
287
288 types.kernel = enum (attrValues kernels);
289
290 kernels = with execFormats; with kernelFamilies; setTypes types.openKernel {
291 # TODO(@Ericson2314): Don't want to mass-rebuild yet to keeping 'darwin' as
292 # the normalized name for macOS.
293 macos = { execFormat = macho; families = { inherit darwin; }; name = "darwin"; };
294 ios = { execFormat = macho; families = { inherit darwin; }; };
295 # A tricky thing about FreeBSD is that there is no stable ABI across
296 # versions. That means that putting in the version as part of the
297 # config string is paramount.
298 freebsd12 = { execFormat = elf; families = { inherit bsd; }; name = "freebsd"; version = 12; };
299 freebsd13 = { execFormat = elf; families = { inherit bsd; }; name = "freebsd"; version = 13; };
300 linux = { execFormat = elf; families = { }; };
301 netbsd = { execFormat = elf; families = { inherit bsd; }; };
302 none = { execFormat = unknown; families = { }; };
303 openbsd = { execFormat = elf; families = { inherit bsd; }; };
304 solaris = { execFormat = elf; families = { }; };
305 wasi = { execFormat = wasm; families = { }; };
306 redox = { execFormat = elf; families = { }; };
307 windows = { execFormat = pe; families = { }; };
308 ghcjs = { execFormat = unknown; families = { }; };
309 genode = { execFormat = elf; families = { }; };
310 mmixware = { execFormat = unknown; families = { }; };
311 } // { # aliases
312 # 'darwin' is the kernel for all of them. We choose macOS by default.
313 darwin = kernels.macos;
314 watchos = kernels.ios;
315 tvos = kernels.ios;
316 win32 = kernels.windows;
317 };
318
319 ################################################################################
320
321 types.openAbi = mkOptionType {
322 name = "abi";
323 description = "binary interface for compiled code and syscalls";
324 merge = mergeOneOption;
325 };
326
327 types.abi = enum (attrValues abis);
328
329 abis = setTypes types.openAbi {
330 cygnus = {};
331 msvc = {};
332
333 # Note: eabi is specific to ARM and PowerPC.
334 # On PowerPC, this corresponds to PPCEABI.
335 # On ARM, this corresponds to ARMEABI.
336 eabi = { float = "soft"; };
337 eabihf = { float = "hard"; };
338
339 # Other architectures should use ELF in embedded situations.
340 elf = {};
341
342 androideabi = {};
343 android = {
344 assertions = [
345 { assertion = platform: !platform.isAarch32;
346 message = ''
347 The "android" ABI is not for 32-bit ARM. Use "androideabi" instead.
348 '';
349 }
350 ];
351 };
352
353 gnueabi = { float = "soft"; };
354 gnueabihf = { float = "hard"; };
355 gnu = {
356 assertions = [
357 { assertion = platform: !platform.isAarch32;
358 message = ''
359 The "gnu" ABI is ambiguous on 32-bit ARM. Use "gnueabi" or "gnueabihf" instead.
360 '';
361 }
362 { assertion = platform: with platform; !(isPower64 && isBigEndian);
363 message = ''
364 The "gnu" ABI is ambiguous on big-endian 64-bit PowerPC. Use "gnuabielfv2" or "gnuabielfv1" instead.
365 '';
366 }
367 ];
368 };
369 gnuabi64 = { abi = "64"; };
370 muslabi64 = { abi = "64"; };
371
372 # NOTE: abi=n32 requires a 64-bit MIPS chip! That is not a typo.
373 # It is basically the 64-bit abi with 32-bit pointers. Details:
374 # https://www.linux-mips.org/pub/linux/mips/doc/ABI/MIPS-N32-ABI-Handbook.pdf
375 gnuabin32 = { abi = "n32"; };
376 muslabin32 = { abi = "n32"; };
377
378 gnuabielfv2 = { abi = "elfv2"; };
379 gnuabielfv1 = { abi = "elfv1"; };
380
381 musleabi = { float = "soft"; };
382 musleabihf = { float = "hard"; };
383 musl = {};
384
385 uclibceabi = { float = "soft"; };
386 uclibceabihf = { float = "hard"; };
387 uclibc = {};
388
389 unknown = {};
390 };
391
392 ################################################################################
393
394 types.parsedPlatform = mkOptionType {
395 name = "system";
396 description = "fully parsed representation of llvm- or nix-style platform tuple";
397 merge = mergeOneOption;
398 check = { cpu, vendor, kernel, abi }:
399 types.cpuType.check cpu
400 && types.vendor.check vendor
401 && types.kernel.check kernel
402 && types.abi.check abi;
403 };
404
405 isSystem = isType "system";
406
407 mkSystem = components:
408 assert types.parsedPlatform.check components;
409 setType "system" components;
410
411 mkSkeletonFromList = l: {
412 "1" = if elemAt l 0 == "avr"
413 then { cpu = elemAt l 0; kernel = "none"; abi = "unknown"; }
414 else throw "Target specification with 1 components is ambiguous";
415 "2" = # We only do 2-part hacks for things Nix already supports
416 if elemAt l 1 == "cygwin"
417 then { cpu = elemAt l 0; kernel = "windows"; abi = "cygnus"; }
418 # MSVC ought to be the default ABI so this case isn't needed. But then it
419 # becomes difficult to handle the gnu* variants for Aarch32 correctly for
420 # minGW. So it's easier to make gnu* the default for the MinGW, but
421 # hack-in MSVC for the non-MinGW case right here.
422 else if elemAt l 1 == "windows"
423 then { cpu = elemAt l 0; kernel = "windows"; abi = "msvc"; }
424 else if (elemAt l 1) == "elf"
425 then { cpu = elemAt l 0; vendor = "unknown"; kernel = "none"; abi = elemAt l 1; }
426 else { cpu = elemAt l 0; kernel = elemAt l 1; };
427 "3" =
428 # cpu-kernel-environment
429 if elemAt l 1 == "linux" ||
430 elem (elemAt l 2) ["eabi" "eabihf" "elf" "gnu"]
431 then {
432 cpu = elemAt l 0;
433 kernel = elemAt l 1;
434 abi = elemAt l 2;
435 vendor = "unknown";
436 }
437 # cpu-vendor-os
438 else if elemAt l 1 == "apple" ||
439 elem (elemAt l 2) [ "wasi" "redox" "mmixware" "ghcjs" "mingw32" ] ||
440 hasPrefix "freebsd" (elemAt l 2) ||
441 hasPrefix "netbsd" (elemAt l 2) ||
442 hasPrefix "genode" (elemAt l 2)
443 then {
444 cpu = elemAt l 0;
445 vendor = elemAt l 1;
446 kernel = if elemAt l 2 == "mingw32"
447 then "windows" # autotools breaks on -gnu for window
448 else elemAt l 2;
449 }
450 else throw "Target specification with 3 components is ambiguous";
451 "4" = { cpu = elemAt l 0; vendor = elemAt l 1; kernel = elemAt l 2; abi = elemAt l 3; };
452 }.${toString (length l)}
453 or (throw "system string has invalid number of hyphen-separated components");
454
455 # This should revert the job done by config.guess from the gcc compiler.
456 mkSystemFromSkeleton = { cpu
457 , # Optional, but fallback too complex for here.
458 # Inferred below instead.
459 vendor ? assert false; null
460 , kernel
461 , # Also inferred below
462 abi ? assert false; null
463 } @ args: let
464 getCpu = name: cpuTypes.${name} or (throw "Unknown CPU type: ${name}");
465 getVendor = name: vendors.${name} or (throw "Unknown vendor: ${name}");
466 getKernel = name: kernels.${name} or (throw "Unknown kernel: ${name}");
467 getAbi = name: abis.${name} or (throw "Unknown ABI: ${name}");
468
469 parsed = {
470 cpu = getCpu args.cpu;
471 vendor =
472 /**/ if args ? vendor then getVendor args.vendor
473 else if isDarwin parsed then vendors.apple
474 else if isWindows parsed then vendors.pc
475 else vendors.unknown;
476 kernel = if hasPrefix "darwin" args.kernel then getKernel "darwin"
477 else if hasPrefix "netbsd" args.kernel then getKernel "netbsd"
478 else getKernel (removeAbiSuffix args.kernel);
479 abi =
480 /**/ if args ? abi then getAbi args.abi
481 else if isLinux parsed || isWindows parsed then
482 if isAarch32 parsed then
483 if lib.versionAtLeast (parsed.cpu.version or "0") "6"
484 then abis.gnueabihf
485 else abis.gnueabi
486 # Default ppc64 BE to ELFv2
487 else if isPower64 parsed && isBigEndian parsed then abis.gnuabielfv2
488 else abis.gnu
489 else abis.unknown;
490 };
491
492 in mkSystem parsed;
493
494 mkSystemFromString = s: mkSystemFromSkeleton (mkSkeletonFromList (lib.splitString "-" s));
495
496 kernelName = kernel:
497 kernel.name + toString (kernel.version or "");
498
499 doubleFromSystem = { cpu, kernel, abi, ... }:
500 /**/ if abi == abis.cygnus then "${cpu.name}-cygwin"
501 else if kernel.families ? darwin then "${cpu.name}-darwin"
502 else "${cpu.name}-${kernelName kernel}";
503
504 tripleFromSystem = { cpu, vendor, kernel, abi, ... } @ sys: assert isSystem sys; let
505 optExecFormat =
506 lib.optionalString (kernel.name == "netbsd" &&
507 gnuNetBSDDefaultExecFormat cpu != kernel.execFormat)
508 kernel.execFormat.name;
509 optAbi = lib.optionalString (abi != abis.unknown) "-${abi.name}";
510 in "${cpu.name}-${vendor.name}-${kernelName kernel}${optExecFormat}${optAbi}";
511
512 ################################################################################
513
514}