{ lib, config, }: let lib' = config.lib; types = config.lib.systems.types; setTypes = type: let assign = name: value: assert lib.errors.trace (type.check value) "${name} is not of type ${type.name}: ${lib.generators.pretty {} value}"; lib.types.set type.name ({inherit name;} // value); in builtins.mapAttrs assign; matchAnyAttrs = patterns: if builtins.isList patterns then value: builtins.any (pattern: lib.attrs.match pattern value) patterns else lib.attrs.match patterns; getDoubles = predicate: builtins.map lib'.systems.into.double (builtins.filter predicate lib'.systems.doubles.all); in { config = { lib.systems = { match = builtins.mapAttrs (lib.fp.const matchAnyAttrs) lib'.systems.patterns; platforms = { pc = { linux-kernel = { name = "pc"; baseConfig = "defconfig"; # Build whatever possible as a module, if not stated in the extra config. autoModules = true; target = "bzImage"; }; }; pc_simplekernel = lib.attrs.mergeRecursive lib'.systems.platforms.pc { linux-kernel.autoModules = false; }; powernv = { linux-kernel = { name = "PowerNV"; baseConfig = "powernv_defconfig"; target = "vmlinux"; autoModules = true; # avoid driver/FS trouble arising from unusual page size extraConfig = '' PPC_64K_PAGES n PPC_4K_PAGES y IPV6 y ATA_BMDMA y ATA_SFF y VIRTIO_MENU y ''; }; }; ## ## ARM ## pogoplug4 = { linux-kernel = { name = "pogoplug4"; baseConfig = "multi_v5_defconfig"; autoModules = false; extraConfig = '' # Ubi for the mtd MTD_UBI y UBIFS_FS y UBIFS_FS_XATTR y UBIFS_FS_ADVANCED_COMPR y UBIFS_FS_LZO y UBIFS_FS_ZLIB y UBIFS_FS_DEBUG n ''; makeFlags = ["LOADADDR=0x8000"]; target = "uImage"; # TODO reenable once manual-config's config actually builds a .dtb and this is checked to be working #DTB = true; }; gcc = { arch = "armv5te"; }; }; sheevaplug = { linux-kernel = { name = "sheevaplug"; baseConfig = "multi_v5_defconfig"; autoModules = false; extraConfig = '' BLK_DEV_RAM y BLK_DEV_INITRD y BLK_DEV_CRYPTOLOOP m BLK_DEV_DM m DM_CRYPT m MD y REISERFS_FS m BTRFS_FS m XFS_FS m JFS_FS m EXT4_FS m USB_STORAGE_CYPRESS_ATACB m # mv cesa requires this sw fallback, for mv-sha1 CRYPTO_SHA1 y # Fast crypto CRYPTO_TWOFISH y CRYPTO_TWOFISH_COMMON y CRYPTO_BLOWFISH y CRYPTO_BLOWFISH_COMMON y IP_PNP y IP_PNP_DHCP y NFS_FS y ROOT_NFS y TUN m NFS_V4 y NFS_V4_1 y NFS_FSCACHE y NFSD m NFSD_V2_ACL y NFSD_V3 y NFSD_V3_ACL y NFSD_V4 y NETFILTER y IP_NF_IPTABLES y IP_NF_FILTER y IP_NF_MATCH_ADDRTYPE y IP_NF_TARGET_LOG y IP_NF_MANGLE y IPV6 m VLAN_8021Q m CIFS y CIFS_XATTR y CIFS_POSIX y CIFS_FSCACHE y CIFS_ACL y WATCHDOG y WATCHDOG_CORE y ORION_WATCHDOG m ZRAM m NETCONSOLE m # Disable OABI to have seccomp_filter (required for systemd) # https://github.com/raspberrypi/firmware/issues/651 OABI_COMPAT n # Fail to build DRM n SCSI_ADVANSYS n USB_ISP1362_HCD n SND_SOC n SND_ALI5451 n FB_SAVAGE n SCSI_NSP32 n ATA_SFF n SUNGEM n IRDA n ATM_HE n SCSI_ACARD n BLK_DEV_CMD640_ENHANCED n FUSE_FS m # systemd uses cgroups CGROUPS y # Latencytop LATENCYTOP y # Ubi for the mtd MTD_UBI y UBIFS_FS y UBIFS_FS_XATTR y UBIFS_FS_ADVANCED_COMPR y UBIFS_FS_LZO y UBIFS_FS_ZLIB y UBIFS_FS_DEBUG n # Kdb, for kernel troubles KGDB y KGDB_SERIAL_CONSOLE y KGDB_KDB y ''; makeFlags = ["LOADADDR=0x0200000"]; target = "uImage"; DTB = true; # Beyond 3.10 }; gcc = { arch = "armv5te"; }; }; raspberrypi = { linux-kernel = { name = "raspberrypi"; baseConfig = "bcm2835_defconfig"; DTB = true; autoModules = true; preferBuiltin = true; extraConfig = '' # Disable OABI to have seccomp_filter (required for systemd) # https://github.com/raspberrypi/firmware/issues/651 OABI_COMPAT n ''; target = "zImage"; }; gcc = { arch = "armv6"; fpu = "vfp"; }; }; # Legacy attribute, for compatibility with existing configs only. raspberrypi2 = lib'.systems.platforms.armv7l-hf-multiplatform; # Nvidia Bluefield 2 (w. crypto support) bluefield2 = { gcc = { arch = "armv8-a+fp+simd+crc+crypto"; }; }; zero-gravitas = { linux-kernel = { name = "zero-gravitas"; baseConfig = "zero-gravitas_defconfig"; # Target verified by checking /boot on reMarkable 1 device target = "zImage"; autoModules = false; DTB = true; }; gcc = { fpu = "neon"; cpu = "cortex-a9"; }; }; zero-sugar = { linux-kernel = { name = "zero-sugar"; baseConfig = "zero-sugar_defconfig"; DTB = true; autoModules = false; preferBuiltin = true; target = "zImage"; }; gcc = { cpu = "cortex-a7"; fpu = "neon-vfpv4"; float-abi = "hard"; }; }; utilite = { linux-kernel = { name = "utilite"; maseConfig = "multi_v7_defconfig"; autoModules = false; extraConfig = '' # Ubi for the mtd MTD_UBI y UBIFS_FS y UBIFS_FS_XATTR y UBIFS_FS_ADVANCED_COMPR y UBIFS_FS_LZO y UBIFS_FS_ZLIB y UBIFS_FS_DEBUG n ''; makeFlags = ["LOADADDR=0x10800000"]; target = "uImage"; DTB = true; }; gcc = { cpu = "cortex-a9"; fpu = "neon"; }; }; guruplug = lib.recursiveUpdate lib'.systems.platforms.sheevaplug { # Define `CONFIG_MACH_GURUPLUG' (see # <http://kerneltrap.org/mailarchive/git-commits-head/2010/5/19/33618>) # and other GuruPlug-specific things. Requires the `guruplug-defconfig' # patch. linux-kernel.baseConfig = "guruplug_defconfig"; }; beaglebone = lib.recursiveUpdate lib'.systems.platforms.armv7l-hf-multiplatform { linux-kernel = { name = "beaglebone"; baseConfig = "bb.org_defconfig"; autoModules = false; extraConfig = ""; # TBD kernel config target = "zImage"; }; }; # https://developer.android.com/ndk/guides/abis#v7a armv7a-android = { linux-kernel.name = "armeabi-v7a"; gcc = { arch = "armv7-a"; float-abi = "softfp"; fpu = "vfpv3-d16"; }; }; armv7l-hf-multiplatform = { linux-kernel = { name = "armv7l-hf-multiplatform"; Major = "2.6"; # Using "2.6" enables 2.6 kernel syscalls in glibc. baseConfig = "multi_v7_defconfig"; DTB = true; autoModules = true; preferBuiltin = true; target = "zImage"; extraConfig = '' # Serial port for Raspberry Pi 3. Wasn't included in ARMv7 defconfig # until 4.17. SERIAL_8250_BCM2835AUX y SERIAL_8250_EXTENDED y SERIAL_8250_SHARE_IRQ y # Hangs ODROID-XU4 ARM_BIG_LITTLE_CPUIDLE n # Disable OABI to have seccomp_filter (required for systemd) # https://github.com/raspberrypi/firmware/issues/651 OABI_COMPAT n # >=5.12 fails with: # drivers/net/ethernet/micrel/ks8851_common.o: in function `ks8851_probe_common': # ks8851_common.c:(.text+0x179c): undefined reference to `__this_module' # See: https://lore.kernel.org/netdev/20210116164828.40545-1-marex@denx.de/T/ KS8851_MLL y ''; }; gcc = { # Some table about fpu flags: # http://community.arm.com/servlet/JiveServlet/showImage/38-1981-3827/blogentry-103749-004812900+1365712953_thumb.png # Cortex-A5: -mfpu=neon-fp16 # Cortex-A7 (rpi2): -mfpu=neon-vfpv4 # Cortex-A8 (beaglebone): -mfpu=neon # Cortex-A9: -mfpu=neon-fp16 # Cortex-A15: -mfpu=neon-vfpv4 # More about FPU: # https://wiki.debian.org/ArmHardFloatPort/VfpComparison # vfpv3-d16 is what Debian uses and seems to be the best compromise: NEON is not supported in e.g. Scaleway or Tegra 2, # and the above page suggests NEON is only an improvement with hand-written assembly. arch = "armv7-a"; fpu = "vfpv3-d16"; # For Raspberry Pi the 2 the best would be: # cpu = "cortex-a7"; # fpu = "neon-vfpv4"; }; }; aarch64-multiplatform = { linux-kernel = { name = "aarch64-multiplatform"; baseConfig = "defconfig"; DTB = true; autoModules = true; preferBuiltin = true; extraConfig = '' # Raspberry Pi 3 stuff. Not needed for s >= 4.10. ARCH_BCM2835 y BCM2835_MBOX y BCM2835_WDT y RASPBERRYPI_FIRMWARE y RASPBERRYPI_POWER y SERIAL_8250_BCM2835AUX y SERIAL_8250_EXTENDED y SERIAL_8250_SHARE_IRQ y # Cavium ThunderX stuff. PCI_HOST_THUNDER_ECAM y # Nvidia Tegra stuff. PCI_TEGRA y # The default (=y) forces us to have the XHCI firmware available in initrd, # which our initrd builder can't currently do easily. USB_XHCI_TEGRA m ''; target = "Image"; }; gcc = { arch = "armv8-a"; }; }; apple-m1 = { gcc = { arch = "armv8.3-a+crypto+sha2+aes+crc+fp16+lse+simd+ras+rdm+rcpc"; cpu = "apple-a13"; }; }; ## ## MIPS ## ben_nanonote = { linux-kernel = { name = "ben_nanonote"; }; gcc = { arch = "mips32"; float = "soft"; }; }; fuloong2f_n32 = { linux-kernel = { name = "fuloong2f_n32"; baseConfig = "lemote2f_defconfig"; autoModules = false; extraConfig = '' MIGRATION n COMPACTION n # nixos mounts some cgroup CGROUPS y BLK_DEV_RAM y BLK_DEV_INITRD y BLK_DEV_CRYPTOLOOP m BLK_DEV_DM m DM_CRYPT m MD y REISERFS_FS m EXT4_FS m USB_STORAGE_CYPRESS_ATACB m IP_PNP y IP_PNP_DHCP y IP_PNP_BOOTP y NFS_FS y ROOT_NFS y TUN m NFS_V4 y NFS_V4_1 y NFS_FSCACHE y NFSD m NFSD_V2_ACL y NFSD_V3 y NFSD_V3_ACL y NFSD_V4 y # Fail to build DRM n SCSI_ADVANSYS n USB_ISP1362_HCD n SND_SOC n SND_ALI5451 n FB_SAVAGE n SCSI_NSP32 n ATA_SFF n SUNGEM n IRDA n ATM_HE n SCSI_ACARD n BLK_DEV_CMD640_ENHANCED n FUSE_FS m # Needed for udev >= 150 SYSFS_DEPRECATED_V2 n VGA_CONSOLE n VT_HW_CONSOLE_BINDING y SERIAL_8250_CONSOLE y FRAMEBUFFER_CONSOLE y EXT2_FS y EXT3_FS y REISERFS_FS y MAGIC_SYSRQ y # The kernel doesn't boot at all, with FTRACE FTRACE n ''; target = "vmlinux"; }; gcc = { arch = "loongson2f"; float = "hard"; abi = "n32"; }; }; # can execute on 32bit chip gcc_mips32r2_o32 = { gcc = { arch = "mips32r2"; abi = "32"; }; }; gcc_mips32r6_o32 = { gcc = { arch = "mips32r6"; abi = "32"; }; }; gcc_mips64r2_n32 = { gcc = { arch = "mips64r2"; abi = "n32"; }; }; gcc_mips64r6_n32 = { gcc = { arch = "mips64r6"; abi = "n32"; }; }; gcc_mips64r2_64 = { gcc = { arch = "mips64r2"; abi = "64"; }; }; gcc_mips64r6_64 = { gcc = { arch = "mips64r6"; abi = "64"; }; }; # based on: # https://www.mail-archive.com/qemu-discuss@nongnu.org/msg05179.html # https://gmplib.org/~tege/qemu.html#mips64-debian mips64el-qemu-linux-gnuabi64 = { linux-kernel = { name = "mips64el"; baseConfig = "64r2el_defconfig"; target = "vmlinuz"; autoModules = false; DTB = true; # for qemu 9p passthrough filesystem extraConfig = '' MIPS_MALTA y PAGE_SIZE_4KB y CPU_LITTLE_ENDIAN y CPU_MIPS64_R2 y 64BIT y CPU_MIPS64_R2 y NET_9P y NET_9P_VIRTIO y 9P_FS y 9P_FS_POSIX_ACL y PCI y VIRTIO_PCI y ''; }; }; ## ## Other ## riscv-multiplatform = { linux-kernel = { name = "riscv-multiplatform"; target = "Image"; autoModules = true; baseConfig = "defconfig"; DTB = true; extraConfig = '' SERIAL_OF_PLATFORM y ''; }; }; mipsel-linux-gnu = { triple = "mipsel-unknown-linux-gnu"; } // lib'.systems.platforms.gcc_mips32r2_o32; # This function takes a minimally-valid "platform" and returns an # attrset containing zero or more additional attrs which should be # included in the platform in order to further elaborate it. select = platform: # x86 /**/ if platform.isx86 then lib'.systems.platforms.pc # ARM else if platform.isAarch32 then let version = platform.system.cpu.version or null; in if version == null then lib'.systems.platforms.pc else if lib.versions.gte "6" version then lib'.systems.platforms.sheevaplug else if lib.versions.gte "7" version then lib'.systems.platforms.raspberrypi else lib'.systems.platforms.armv7l-hf-multiplatform else if platform.isAarch64 then if platform.isDarwin then lib'.systems.platforms.apple-m1 else lib'.systems.platforms.aarch64-multiplatform else if platform.isRiscV then lib'.systems.platforms.riscv-multiplatform else if platform.system.cpu == types.cpus.mipsel then lib'.systems.platforms.mipsel-linux-gnu else if platform.system.cpu == types.cpus.powerpc64le then lib'.systems.platforms.powernv else {}; }; architectures = { features = { # x86_64 Generic # Spec: https://gitlab.com/x86-psABIs/x86-64-ABI/ default = []; x86-64 = []; x86-64-v2 = ["sse3" "ssse3" "sse4_1" "sse4_2"]; x86-64-v3 = ["sse3" "ssse3" "sse4_1" "sse4_2" "avx" "avx2" "fma"]; x86-64-v4 = ["sse3" "ssse3" "sse4_1" "sse4_2" "avx" "avx2" "avx512" "fma"]; # x86_64 Intel nehalem = ["sse3" "ssse3" "sse4_1" "sse4_2" "aes"]; westmere = ["sse3" "ssse3" "sse4_1" "sse4_2" "aes"]; sandybridge = ["sse3" "ssse3" "sse4_1" "sse4_2" "aes" "avx"]; ivybridge = ["sse3" "ssse3" "sse4_1" "sse4_2" "aes" "avx"]; haswell = ["sse3" "ssse3" "sse4_1" "sse4_2" "aes" "avx" "avx2" "fma"]; broadwell = ["sse3" "ssse3" "sse4_1" "sse4_2" "aes" "avx" "avx2" "fma"]; skylake = ["sse3" "ssse3" "sse4_1" "sse4_2" "aes" "avx" "avx2" "fma"]; skylake-avx512 = ["sse3" "ssse3" "sse4_1" "sse4_2" "aes" "avx" "avx2" "avx512" "fma"]; cannonlake = ["sse3" "ssse3" "sse4_1" "sse4_2" "aes" "avx" "avx2" "avx512" "fma"]; icelake-client = ["sse3" "ssse3" "sse4_1" "sse4_2" "aes" "avx" "avx2" "avx512" "fma"]; icelake-server = ["sse3" "ssse3" "sse4_1" "sse4_2" "aes" "avx" "avx2" "avx512" "fma"]; cascadelake = ["sse3" "ssse3" "sse4_1" "sse4_2" "aes" "avx" "avx2" "avx512" "fma"]; cooperlake = ["sse3" "ssse3" "sse4_1" "sse4_2" "aes" "avx" "avx2" "avx512" "fma"]; tigerlake = ["sse3" "ssse3" "sse4_1" "sse4_2" "aes" "avx" "avx2" "avx512" "fma"]; alderlake = ["sse3" "ssse3" "sse4_1" "sse4_2" "aes" "avx" "avx2" "fma"]; # x86_64 AMD btver1 = ["sse3" "ssse3" "sse4_1" "sse4_2"]; btver2 = ["sse3" "ssse3" "sse4_1" "sse4_2" "aes" "avx"]; bdver1 = ["sse3" "ssse3" "sse4_1" "sse4_2" "sse4a" "aes" "avx" "fma" "fma4"]; bdver2 = ["sse3" "ssse3" "sse4_1" "sse4_2" "sse4a" "aes" "avx" "fma" "fma4"]; bdver3 = ["sse3" "ssse3" "sse4_1" "sse4_2" "sse4a" "aes" "avx" "fma" "fma4"]; bdver4 = ["sse3" "ssse3" "sse4_1" "sse4_2" "sse4a" "aes" "avx" "avx2" "fma" "fma4"]; znver1 = ["sse3" "ssse3" "sse4_1" "sse4_2" "sse4a" "aes" "avx" "avx2" "fma"]; znver2 = ["sse3" "ssse3" "sse4_1" "sse4_2" "sse4a" "aes" "avx" "avx2" "fma"]; znver3 = ["sse3" "ssse3" "sse4_1" "sse4_2" "sse4a" "aes" "avx" "avx2" "fma"]; znver4 = ["sse3" "ssse3" "sse4_1" "sse4_2" "sse4a" "aes" "avx" "avx2" "avx512" "fma"]; # other armv5te = []; armv6 = []; armv7-a = []; armv8-a = []; mips32 = []; loongson2f = []; }; # a superior CPU has all the features of an inferior and is able to build and test code for it inferiors = { # x86_64 Generic default = []; x86-64 = []; x86-64-v2 = ["x86-64"]; x86-64-v3 = ["x86-64-v2"] ++ lib'.systems.architectures.inferiors.x86-64-v2; x86-64-v4 = ["x86-64-v3"] ++ lib'.systems.architectures.inferiors.x86-64-v3; # x86_64 Intel # https://gcc.gnu.org/onlinedocs/gcc/x86-Options.html nehalem = ["x86-64-v2"] ++ lib'.systems.architectures.inferiors.x86-64-v2; westmere = ["nehalem"] ++ lib'.systems.architectures.inferiors.nehalem; sandybridge = ["westmere"] ++ lib'.systems.architectures.inferiors.westmere; ivybridge = ["sandybridge"] ++ lib'.systems.architectures.inferiors.sandybridge; haswell = lib.unique (["ivybridge" "x86-64-v3"] ++ lib'.systems.architectures.inferiors.ivybridge ++ lib'.systems.architectures.inferiors.x86-64-v3); broadwell = ["haswell"] ++ lib'.systems.architectures.inferiors.haswell; skylake = ["broadwell"] ++ lib'.systems.architectures.inferiors.broadwell; skylake-avx512 = lib.unique (["skylake" "x86-64-v4"] ++ lib'.systems.architectures.inferiors.skylake ++ lib'.systems.architectures.inferiors.x86-64-v4); cannonlake = ["skylake-avx512"] ++ lib'.systems.architectures.inferiors.skylake-avx512; icelake-client = ["cannonlake"] ++ lib'.systems.architectures.inferiors.cannonlake; icelake-server = ["icelake-client"] ++ lib'.systems.architectures.inferiors.icelake-client; cascadelake = ["cannonlake"] ++ lib'.systems.architectures.inferiors.cannonlake; cooperlake = ["cascadelake"] ++ lib'.systems.architectures.inferiors.cascadelake; tigerlake = ["icelake-server"] ++ lib'.systems.architectures.inferiors.icelake-server; # CX16 does not exist on alderlake, while it does on nearly all other intel CPUs alderlake = []; # x86_64 AMD # TODO: fill this (need testing) btver1 = []; btver2 = []; bdver1 = []; bdver2 = []; bdver3 = []; bdver4 = []; # Regarding `skylake` as inferior of `znver1`, there are reports of # successful usage by Gentoo users and Phoronix benchmarking of different # `-march` targets. # # The GCC documentation on extensions used and wikichip documentation # regarding supperted extensions on znver1 and skylake was used to create # this partial order. # # Note: # # - The successors of `skylake` (`cannonlake`, `icelake`, etc) use `avx512` # which no current AMD Zen michroarch support. # - `znver1` uses `ABM`, `CLZERO`, `CX16`, `MWAITX`, and `SSE4A` which no # current Intel microarch support. # # https://www.phoronix.com/scan.php?page=article&item=amd-znver3-gcc11&num=1 # https://gcc.gnu.org/onlinedocs/gcc/x86-Options.html # https://en.wikichip.org/wiki/amd/microarchitectures/zen # https://en.wikichip.org/wiki/intel/microarchitectures/skylake znver1 = ["skylake"] ++ lib'.systems.architectures.inferiors.skylake; # Includes haswell and x86-64-v3 znver2 = ["znver1"] ++ lib'.systems.architectures.inferiors.znver1; znver3 = ["znver2"] ++ lib'.systems.architectures.inferiors.znver2; znver4 = lib.unique (["znver3" "x86-64-v4"] ++ lib'.systems.architectures.inferiors.znver3 ++ lib'.systems.architectures.inferiors.x86-64-v4); # other armv5te = []; armv6 = []; armv7-a = []; armv8-a = []; mips32 = []; loongson2f = []; }; }; validate = { architecture = let isSupported = feature: x: builtins.elem feature (lib'.systems.architectures.features.${x} or []); in { sse3Support = isSupported "sse3"; ssse3Support = isSupported "ssse3"; sse4_1Support = isSupported "sse4_1"; sse4_2Support = isSupported "sse4_2"; sse4_aSupport = isSupported "sse4a"; avxSupport = isSupported "avx"; avx2Support = isSupported "avx2"; avx512Support = isSupported "avx512"; aesSupport = isSupported "aes"; fmaSupport = isSupported "fma"; fma4Support = isSupported "fma4"; }; compatible = a: b: lib.any lib.id [ # x86 (b == lib'.systems.types.cpus.i386 && lib'.systems.validate.compatible a lib'.systems.types.cpus.i486) (b == lib'.systems.types.cpus.i486 && lib'.systems.validate.compatible a lib'.systems.types.cpus.i586) (b == lib'.systems.types.cpus.i586 && lib'.systems.validate.compatible a lib'.systems.types.cpus.i686) # XXX: Not true in some cases. Like in WSL mode. (b == lib'.systems.types.cpus.i686 && lib'.systems.validate.compatible a lib'.systems.types.cpus.x86_64) # ARMv4 (b == lib'.systems.types.cpus.arm && lib'.systems.validate.compatible a lib'.systems.types.cpus.armv5tel) # ARMv5 (b == lib'.systems.types.cpus.armv5tel && lib'.systems.validate.compatible a lib'.systems.types.cpus.armv6l) # ARMv6 (b == lib'.systems.types.cpus.armv6l && lib'.systems.validate.compatible a lib'.systems.types.cpus.armv6m) (b == lib'.systems.types.cpus.armv6m && lib'.systems.validate.compatible a lib'.systems.types.cpus.armv7l) # ARMv7 (b == lib'.systems.types.cpus.armv7l && lib'.systems.validate.compatible a lib'.systems.types.cpus.armv7a) (b == lib'.systems.types.cpus.armv7l && lib'.systems.validate.compatible a lib'.systems.types.cpus.armv7r) (b == lib'.systems.types.cpus.armv7l && lib'.systems.validate.compatible a lib'.systems.types.cpus.armv7m) # ARMv8 (b == lib'.systems.types.cpus.aarch64 && a == lib'.systems.types.cpus.armv8a) (b == lib'.systems.types.cpus.armv8a && lib'.systems.validate.compatible a lib'.systems.types.cpus.aarch64) (b == lib'.systems.types.cpus.armv8r && lib'.systems.validate.compatible a lib'.systems.types.cpus.armv8a) (b == lib'.systems.types.cpus.armv8m && lib'.systems.validate.compatible a lib'.systems.types.cpus.armv8a) # PowerPC (b == lib'.systems.types.cpus.powerpc && lib'.systems.validate.compatible a lib'.systems.types.cpus.powerpc64) (b == lib'.systems.types.cpus.powerpcle && lib'.systems.validate.compatible a lib'.systems.types.cpus.powerpc64le) # MIPS (b == lib'.systems.types.cpus.mips && lib'.systems.validate.compatible a lib'.systems.types.cpus.mips64) (b == lib'.systems.types.cpus.mipsel && lib'.systems.validate.compatible a lib'.systems.types.cpus.mips64el) # RISCV (b == lib'.systems.types.cpus.riscv32 && lib'.systems.validate.compatible a lib'.systems.types.cpus.riscv64) # SPARC (b == lib'.systems.types.cpus.sparc && lib'.systems.validate.compatible a lib'.systems.types.cpus.sparc64) # WASM (b == lib'.systems.types.cpus.wasm32 && lib'.systems.validate.compatible a lib'.systems.types.cpus.wasm64) # identity (b == a) ]; }; types = { generic = { endian = lib.types.create { name = "Endian"; description = "Endianness"; merge = lib.options.merge.one; }; cpu = lib.types.create { name = "Cpu"; description = "Instruction set architecture name and information"; merge = lib.options.merge.one; check = x: types.bits.check x.bits && ( if 8 < x.bits then types.endian.check x.endian else !(x ? endian) ); }; family = lib.types.create { name = "Family"; description = "The kernel family."; merge = lib.options.merge.one; }; exec = lib.types.create { name = "Exec"; description = "Executable container used by the kernel"; merge = lib.options.merge.one; }; kernel = lib.types.create { name = "Kernel"; description = "Kernel name and information"; merge = lib.options.merge.one; check = value: types.exec.check value.exec && builtins.all types.family.check (builtins.attrValues value.families); }; abi = lib.types.create { name = "Abi"; description = "Binary interface for compiled code and syscalls"; merge = lib.options.merge.one; }; vendor = lib.types.create { name = "Vendor"; description = "Vendor for the platform"; merge = lib.options.merge.one; }; }; platform = lib.types.create { name = "system"; description = "fully parsed representation of llvm- or nix-style platform tuple"; merge = lib.options.merge.one; check = { cpu, vendor, kernel, abi, }: types.cpu.check cpu && types.vendor.check vendor && types.kernel.check kernel && types.abi.check abi; }; endian = lib.types.enum (builtins.attrValues types.endians); endians = setTypes types.generic.endian { big = {}; little = {}; }; bits = lib.types.enum [8 16 32 64 128]; exec = lib.types.enum (builtins.attrValues types.execs); execs = setTypes types.generic.exec { aout = {}; # a.out elf = {}; macho = {}; pe = {}; wasm = {}; unknown = {}; }; vendor = lib.types.enum (builtins.attrValues types.vendors); vendors = setTypes types.generic.vendor { apple = {}; pc = {}; knuth = {}; # Actually matters, unlocking some MinGW-w64-specific options in GCC. See # bottom of https://sourceforge.net/p/mingw-w64/wiki2/Unicode%20apps/ w64 = {}; none = {}; unknown = {}; }; family = lib.types.enum (builtins.attrValues types.families); families = setTypes types.generic.family { bsd = {}; darwin = {}; }; kernel = lib.types.enum (builtins.attrValues types.kernels); kernels = setTypes types.generic.kernel { # TODO(@Ericson2314): Don't want to mass-rebuild yet to keeping 'darwin' as # the normalized name for macOS. macos = { exec = types.execs.macho; families = { darwin = types.families.darwin; }; name = "darwin"; }; ios = { exec = types.execs.macho; families = {darwin = types.families.darwin;}; }; # A tricky thing about FreeBSD is that there is no stable ABI across # versions. That means that putting in the version as part of the # config string is paramount. freebsd12 = { exec = types.execs.elf; families = {bsd = types.families.bsd;}; name = "freebsd"; version = 12; }; freebsd13 = { exec = types.execs.elf; families = {bsd = types.families.bsd;}; name = "freebsd"; version = 13; }; linux = { exec = types.execs.elf; families = {}; }; netbsd = { exec = types.execs.elf; families = {bsd = types.families.bsd;}; }; none = { exec = types.execs.unknown; families = {}; }; openbsd = { exec = types.execs.elf; families = {bsd = types.families.bsd;}; }; solaris = { exec = types.execs.elf; families = {}; }; wasi = { exec = types.execs.wasm; families = {}; }; redox = { exec = types.execs.elf; families = {}; }; windows = { exec = types.execs.pe; families = {}; }; ghcjs = { exec = types.execs.unknown; families = {}; }; genode = { exec = types.execs.elf; families = {}; }; mmixware = { exec = types.execs.unknown; families = {}; }; # aliases # 'darwin' is the kernel for all of them. We choose macOS by default. darwin = types.kernels.macos; watchos = types.kernels.ios; tvos = types.kernels.ios; win32 = types.kernels.windows; }; cpu = lib.types.enum (builtins.attrValues types.cpus); cpus = setTypes types.generic.cpu { arm = { bits = 32; endian = types.endians.little; family = "arm"; }; armv5tel = { bits = 32; endian = types.endians.little; family = "arm"; version = "5"; arch = "armv5t"; }; armv6m = { bits = 32; endian = types.endians.little; family = "arm"; version = "6"; arch = "armv6-m"; }; armv6l = { bits = 32; endian = types.endians.little; family = "arm"; version = "6"; arch = "armv6"; }; armv7a = { bits = 32; endian = types.endians.little; family = "arm"; version = "7"; arch = "armv7-a"; }; armv7r = { bits = 32; endian = types.endians.little; family = "arm"; version = "7"; arch = "armv7-r"; }; armv7m = { bits = 32; endian = types.endians.little; family = "arm"; version = "7"; arch = "armv7-m"; }; armv7l = { bits = 32; endian = types.endians.little; family = "arm"; version = "7"; arch = "armv7"; }; armv8a = { bits = 32; endian = types.endians.little; family = "arm"; version = "8"; arch = "armv8-a"; }; armv8r = { bits = 32; endian = types.endians.little; family = "arm"; version = "8"; arch = "armv8-a"; }; armv8m = { bits = 32; endian = types.endians.little; family = "arm"; version = "8"; arch = "armv8-m"; }; aarch64 = { bits = 64; endian = types.endians.little; family = "arm"; version = "8"; arch = "armv8-a"; }; aarch64_be = { bits = 64; endian = types.endians.big; family = "arm"; version = "8"; arch = "armv8-a"; }; i386 = { bits = 32; endian = types.endians.little; family = "x86"; arch = "i386"; }; i486 = { bits = 32; endian = types.endians.little; family = "x86"; arch = "i486"; }; i586 = { bits = 32; endian = types.endians.little; family = "x86"; arch = "i586"; }; i686 = { bits = 32; endian = types.endians.little; family = "x86"; arch = "i686"; }; x86_64 = { bits = 64; endian = types.endians.little; family = "x86"; arch = "x86-64"; }; microblaze = { bits = 32; endian = types.endians.big; family = "microblaze"; }; microblazeel = { bits = 32; endian = types.endians.little; family = "microblaze"; }; mips = { bits = 32; endian = types.endians.big; family = "mips"; }; mipsel = { bits = 32; endian = types.endians.little; family = "mips"; }; mips64 = { bits = 64; endian = types.endians.big; family = "mips"; }; mips64el = { bits = 64; endian = types.endians.little; family = "mips"; }; mmix = { bits = 64; endian = types.endians.big; family = "mmix"; }; m68k = { bits = 32; endian = types.endians.big; family = "m68k"; }; powerpc = { bits = 32; endian = types.endians.big; family = "power"; }; powerpc64 = { bits = 64; endian = types.endians.big; family = "power"; }; powerpc64le = { bits = 64; endian = types.endians.little; family = "power"; }; powerpcle = { bits = 32; endian = types.endians.little; family = "power"; }; riscv32 = { bits = 32; endian = types.endians.little; family = "riscv"; }; riscv64 = { bits = 64; endian = types.endians.little; family = "riscv"; }; s390 = { bits = 32; endian = types.endians.big; family = "s390"; }; s390x = { bits = 64; endian = types.endians.big; family = "s390"; }; sparc = { bits = 32; endian = types.endians.big; family = "sparc"; }; sparc64 = { bits = 64; endian = types.endians.big; family = "sparc"; }; wasm32 = { bits = 32; endian = types.endians.little; family = "wasm"; }; wasm64 = { bits = 64; endian = types.endians.little; family = "wasm"; }; alpha = { bits = 64; endian = types.endians.little; family = "alpha"; }; rx = { bits = 32; endian = types.endians.little; family = "rx"; }; msp430 = { bits = 16; endian = types.endians.little; family = "msp430"; }; avr = { bits = 8; family = "avr"; }; vc4 = { bits = 32; endian = types.endians.little; family = "vc4"; }; or1k = { bits = 32; endian = types.endians.big; family = "or1k"; }; loongarch64 = { bits = 64; endian = types.endians.little; family = "loongarch"; }; javascript = { bits = 32; endian = types.endians.little; family = "javascript"; }; }; abi = lib.types.enum (builtins.attrValues types.abis); abis = setTypes types.generic.abi { cygnus = {}; msvc = {}; # Note: eabi is specific to ARM and PowerPC. # On PowerPC, this corresponds to PPCEABI. # On ARM, this corresponds to ARMEABI. eabi = {float = "soft";}; eabihf = {float = "hard";}; # Other architectures should use ELF in embedded situations. elf = {}; androideabi = {}; android = { assertions = [ { assertion = platform: !platform.isAarch32; message = '' The "android" ABI is not for 32-bit ARM. Use "androideabi" instead. ''; } ]; }; gnueabi = {float = "soft";}; gnueabihf = {float = "hard";}; gnu = { assertions = [ { assertion = platform: !platform.isAarch32; message = '' The "gnu" ABI is ambiguous on 32-bit ARM. Use "gnueabi" or "gnueabihf" instead. ''; } { assertion = platform: with platform; !(isPower64 && isBigEndian); message = '' The "gnu" ABI is ambiguous on big-endian 64-bit PowerPC. Use "gnuabielfv2" or "gnuabielfv1" instead. ''; } ]; }; gnuabi64 = {abi = "64";}; muslabi64 = {abi = "64";}; # NOTE: abi=n32 requires a 64-bit MIPS chip! That is not a typo. # It is basically the 64-bit abi with 32-bit pointers. Details: # https://www.linux-mips.org/pub/linux/mips/doc/ABI/MIPS-N32-ABI-Handbook.pdf gnuabin32 = {abi = "n32";}; muslabin32 = {abi = "n32";}; gnuabielfv2 = {abi = "elfv2";}; gnuabielfv1 = {abi = "elfv1";}; musleabi = {float = "soft";}; musleabihf = {float = "hard";}; musl = {}; uclibceabi = {float = "soft";}; uclibceabihf = {float = "hard";}; uclibc = {}; unknown = {}; }; }; from = { string = value: let parts = lib.strings.split "-" value; skeleton = lib'.systems.skeleton parts; system = lib'.systems.create (lib'.systems.from.skeleton skeleton); in system; skeleton = spec @ { cpu, vendor ? assert false; null, kernel, abi ? assert false; null, }: let getCpu = name: types.cpus.${name} or (throw "Unknown CPU type: ${name}"); getVendor = name: types.vendors.${name} or (throw "Unknown vendor: ${name}"); getKernel = name: types.kernels.${name} or (throw "Unknown kernel: ${name}"); getAbi = name: types.abis.${name} or (throw "Unknown ABI: ${name}"); resolved = { cpu = getCpu spec.cpu; vendor = if spec ? vendor then getVendor spec.vendor else if lib'.systems.match.isDarwin resolved then types.vendors.apple else if lib'.systems.match.isWindows resolved then types.vendors.pc else types.vendors.unknown; kernel = if lib.strings.hasPrefix "darwin" spec.kernel then getKernel "darwin" else if lib.strings.hasPrefix "netbsd" spec.kernel then getKernel "netbsd" else getKernel spec.kernel; abi = if spec ? abi then getAbi spec.abi else if lib'.systems.match.isLinux resolved || lib'.systems.match.isWindows resolved then if lib'.systems.match.isAarch32 resolved then if lib.versions.gte "6" (resolved.cpu.version) then types.abis.gnueabihf else types.abis.gnueabi else if lib'.systems.match.isPower64 resolved && lib'.systems.match.isBigEndian resolved then types.abis.gnuabielfv2 else types.abis.gnu else types.abis.unknown; }; in resolved; }; into = { double = { cpu, kernel, abi, ... }: let kernelName = kernel.name + builtins.toString (kernel.version or ""); in if abi == types.abis.cygnus then "${cpu.name}-cygwin" else if kernel.families ? darwin then "${cpu.name}-darwin" else "${cpu.name}-${kernelName}"; triple = { cpu, vendor, kernel, abi, ... }: let kernelName = kernel.name + builtins.toString (kernel.version or ""); netbsdExec = if (cpu.family == "arm" && cpu.bits == 32) || (cpu.family == "sparc" && cpu.bits == 32) || (cpu.family == "m68k" && cpu.bits == 32) || (cpu.family == "x86" && cpu.bits == 32) then types.execs.aout else types.execs.elf; exec = lib.strings.when (kernel.name == "netbsd" && netbsdExec != kernel.exec) kernel.exec.name; abi = lib.strings.when (abi != types.abis.unknown) "-${abi.name}"; in "${cpu.name}-${vendor.name}-${kernelName}${exec}${abi}"; }; create = components: assert types.platform.check components; lib.types.set "system" components; skeleton = parts: let length = builtins.length parts; first = builtins.elemAt parts 0; second = builtins.elemAt parts 1; third = builtins.elemAt parts 2; fourth = builtins.elemAt parts 3; in if length == 1 then if first == "avr" then { cpu = first; kernel = "none"; abi = "unknown"; } else builtins.throw "Target specification with 1 component is ambiguous." else if length == 2 then if second == "cygwin" then { cpu = first; kernel = "windows"; abi = "cygnus"; } else if second == "windows" then { cpu = first; kernel = "windows"; abi = "msvc"; } else if second == "elf" then { cpu = first; vendor = "unkonwn"; kernel = "none"; abi = second; } else { cpu = first; kernel = second; } else if length == 3 then if second == "linux" || (builtins.elem third ["eabi" "eabihf" "elf" "gnu"]) then { cpu = first; vendor = "unknown"; kernel = second; abi = third; } else if (second == "apple") || lib.strings.hasPrefix "freebsd" third || lib.strings.hasPrefix "netbsd" third || lib.strings.hasPrefix "genode" third || (builtins.elem third ["wasi" "redox" "mmixware" "ghcjs" "mingw32"]) then { cpu = first; vendor = second; kernel = if third == "mingw32" then "windows" else third; } else builtins.throw "Target specification with 3 components is ambiguous." else if length == 4 then { cpu = first; vendor = second; kernel = third; abi = fourth; } else builtins.throw "Invalid component count for creating system skeleton. Expected 1-4, but got ${builtins.toString length}."; patterns = { isi686 = {cpu = types.cpus.i686;}; isx86_32 = { cpu = { family = "x86"; bits = 32; }; }; isx86_64 = { cpu = { family = "x86"; bits = 64; }; }; isPower = {cpu = {family = "power";};}; isPower64 = { cpu = { family = "power"; bits = 64; }; }; # This ABI is the default in NixOS PowerPC64 BE, but not on mainline GCC, # so it sometimes causes issues in certain packages that makes the wrong # assumption on the used ABI. isAbiElfv2 = [ {abi = {abi = "elfv2";};} { abi = {name = "musl";}; cpu = { family = "power"; bits = 64; }; } ]; isx86 = {cpu = {family = "x86";};}; isAarch32 = { cpu = { family = "arm"; bits = 32; }; }; isArmv7 = map ({arch, ...}: {cpu = {inherit arch;};}) (lib.filter (cpu: lib.hasPrefix "armv7" cpu.arch or "") (lib.attrValues types.cpus)); isAarch64 = { cpu = { family = "arm"; bits = 64; }; }; isAarch = {cpu = {family = "arm";};}; isMicroBlaze = {cpu = {family = "microblaze";};}; isMips = {cpu = {family = "mips";};}; isMips32 = { cpu = { family = "mips"; bits = 32; }; }; isMips64 = { cpu = { family = "mips"; bits = 64; }; }; isMips64n32 = { cpu = { family = "mips"; bits = 64; }; abi = {abi = "n32";}; }; isMips64n64 = { cpu = { family = "mips"; bits = 64; }; abi = {abi = "64";}; }; isMmix = {cpu = {family = "mmix";};}; isRiscV = {cpu = {family = "riscv";};}; isRiscV32 = { cpu = { family = "riscv"; bits = 32; }; }; isRiscV64 = { cpu = { family = "riscv"; bits = 64; }; }; isRx = {cpu = {family = "rx";};}; isSparc = {cpu = {family = "sparc";};}; isWasm = {cpu = {family = "wasm";};}; isMsp430 = {cpu = {family = "msp430";};}; isVc4 = {cpu = {family = "vc4";};}; isAvr = {cpu = {family = "avr";};}; isAlpha = {cpu = {family = "alpha";};}; isOr1k = {cpu = {family = "or1k";};}; isM68k = {cpu = {family = "m68k";};}; isS390 = {cpu = {family = "s390";};}; isS390x = { cpu = { family = "s390"; bits = 64; }; }; isLoongArch64 = { cpu = { family = "loongarch"; bits = 64; }; }; isJavaScript = {cpu = types.cpus.javascript;}; is32bit = {cpu = {bits = 32;};}; is64bit = {cpu = {bits = 64;};}; isILP32 = builtins.map (a: {abi = {abi = a;};}) ["n32" "ilp32" "x32"]; isBigEndian = {cpu = {endian = types.endians.big;};}; isLittleEndian = {cpu = {endian = types.endians.little;};}; isBSD = {kernel = {families = {bsd = types.families.bsd;};};}; isDarwin = {kernel = {families = {darwin = types.families.darwin;};};}; isUnix = [lib'.systems.match.isBSD lib'.systems.match.isDarwin lib'.systems.match.isLinux lib'.systems.match.isSunOS lib'.systems.match.isCygwin lib'.systems.match.isRedox]; isMacOS = {kernel = types.kernels.macos;}; isiOS = {kernel = types.kernels.ios;}; isLinux = {kernel = types.kernels.linux;}; isSunOS = {kernel = types.kernels.solaris;}; isFreeBSD = {kernel = {name = "freebsd";};}; isNetBSD = {kernel = types.kernels.netbsd;}; isOpenBSD = {kernel = types.kernels.openbsd;}; isWindows = {kernel = types.kernels.windows;}; isCygwin = { kernel = types.kernels.windows; abi = types.abis.cygnus; }; isMinGW = { kernel = types.kernels.windows; abi = types.abis.gnu; }; isWasi = {kernel = types.kernels.wasi;}; isRedox = {kernel = types.kernels.redox;}; isGhcjs = {kernel = types.kernels.ghcjs;}; isGenode = {kernel = types.kernels.genode;}; isNone = {kernel = types.kernels.none;}; isAndroid = [{abi = types.abis.android;} {abi = types.abis.androideabi;}]; isGnu = builtins.map (value: {abi = value;}) [types.abis.gnuabi64 types.abis.gnuabin32 types.abis.gnu types.abis.gnueabi types.abis.gnueabihf types.abis.gnuabielfv1 types.abis.gnuabielfv2]; isMusl = builtins.map (value: {abi = value;}) [types.abis.musl types.abis.musleabi types.abis.musleabihf types.abis.muslabin32 types.abis.muslabi64]; isUClibc = builtins.map (value: {abi = value;}) [types.abis.uclibc types.abis.uclibceabi types.abis.uclibceabihf]; isEfi = [ { cpu = { family = "arm"; version = "6"; }; } { cpu = { family = "arm"; version = "7"; }; } { cpu = { family = "arm"; version = "8"; }; } {cpu = {family = "riscv";};} {cpu = {family = "x86";};} ]; }; doubles = { resolved = builtins.map lib'.systems.from.string lib'.systems.doubles.all; all = [ # Cygwin "i686-cygwin" "x86_64-cygwin" # Darwin "x86_64-darwin" "i686-darwin" "aarch64-darwin" "armv7a-darwin" # FreeBSD "i686-freebsd13" "x86_64-freebsd13" # Genode "aarch64-genode" "i686-genode" "x86_64-genode" # illumos "x86_64-solaris" # JS "javascript-ghcjs" # Linux "aarch64-linux" "armv5tel-linux" "armv6l-linux" "armv7a-linux" "armv7l-linux" "i686-linux" "loongarch64-linux" "m68k-linux" "microblaze-linux" "microblazeel-linux" "mips-linux" "mips64-linux" "mips64el-linux" "mipsel-linux" "powerpc64-linux" "powerpc64le-linux" "riscv32-linux" "riscv64-linux" "s390-linux" "s390x-linux" "x86_64-linux" # MMIXware "mmix-mmixware" # NetBSD "aarch64-netbsd" "armv6l-netbsd" "armv7a-netbsd" "armv7l-netbsd" "i686-netbsd" "m68k-netbsd" "mipsel-netbsd" "powerpc-netbsd" "riscv32-netbsd" "riscv64-netbsd" "x86_64-netbsd" # none "aarch64_be-none" "aarch64-none" "arm-none" "armv6l-none" "avr-none" "i686-none" "microblaze-none" "microblazeel-none" "mips-none" "mips64-none" "msp430-none" "or1k-none" "m68k-none" "powerpc-none" "powerpcle-none" "riscv32-none" "riscv64-none" "rx-none" "s390-none" "s390x-none" "vc4-none" "x86_64-none" # OpenBSD "i686-openbsd" "x86_64-openbsd" # Redox "x86_64-redox" # WASI "wasm64-wasi" "wasm32-wasi" # Windows "x86_64-windows" "i686-windows" ]; arm = getDoubles lib'.systems.match.isAarch32; armv7 = getDoubles lib'.systems.match.isArmv7; aarch64 = getDoubles lib'.systems.match.isAarch64; x86 = getDoubles lib'.systems.match.isx86; i686 = getDoubles lib'.systems.match.isi686; x86_64 = getDoubles lib'.systems.match.isx86_64; microblaze = getDoubles lib'.systems.match.isMicroBlaze; mips = getDoubles lib'.systems.match.isMips; mmix = getDoubles lib'.systems.match.isMmix; power = getDoubles lib'.systems.match.isPower; riscv = getDoubles lib'.systems.match.isRiscV; riscv32 = getDoubles lib'.systems.match.isRiscV32; riscv64 = getDoubles lib'.systems.match.isRiscV64; rx = getDoubles lib'.systems.match.isRx; vc4 = getDoubles lib'.systems.match.isVc4; or1k = getDoubles lib'.systems.match.isOr1k; m68k = getDoubles lib'.systems.match.isM68k; s390 = getDoubles lib'.systems.match.isS390; s390x = getDoubles lib'.systems.match.isS390x; loongarch64 = getDoubles lib'.systems.match.isLoongArch64; js = getDoubles lib'.systems.match.isJavaScript; bigEndian = getDoubles lib'.systems.match.isBigEndian; littleEndian = getDoubles lib'.systems.match.isLittleEndian; cygwin = getDoubles lib'.systems.match.isCygwin; darwin = getDoubles lib'.systems.match.isDarwin; freebsd = getDoubles lib'.systems.match.isFreeBSD; # Should be better, but MinGW is unclear. gnu = getDoubles (lib.attrs.match { kernel = types.kernels.linux; abi = types.abis.gnu; }) ++ getDoubles (lib.attrs.match { kernel = types.kernels.linux; abi = types.abis.gnueabi; }) ++ getDoubles (lib.attrs.match { kernel = types.kernels.linux; abi = types.abis.gnueabihf; }) ++ getDoubles (lib.attrs.match { kernel = types.kernels.linux; abi = types.abis.gnuabin32; }) ++ getDoubles (lib.attrs.match { kernel = types.kernels.linux; abi = types.abis.gnuabi64; }) ++ getDoubles (lib.attrs.match { kernel = types.kernels.linux; abi = types.abis.gnuabielfv1; }) ++ getDoubles (lib.attrs.match { kernel = types.kernels.linux; abi = types.abis.gnuabielfv2; }); illumos = getDoubles lib'.systems.match.isSunOS; linux = getDoubles lib'.systems.match.isLinux; netbsd = getDoubles lib'.systems.match.isNetBSD; openbsd = getDoubles lib'.systems.match.isOpenBSD; unix = getDoubles lib'.systems.match.isUnix; wasi = getDoubles lib'.systems.match.isWasi; redox = getDoubles lib'.systems.match.isRedox; windows = getDoubles lib'.systems.match.isWindows; genode = getDoubles lib'.systems.match.isGenode; embedded = getDoubles lib'.systems.match.isNone; mesaPlatforms = ["i686-linux" "x86_64-linux" "x86_64-darwin" "armv5tel-linux" "armv6l-linux" "armv7l-linux" "armv7a-linux" "aarch64-linux" "powerpc64-linux" "powerpc64le-linux" "aarch64-darwin" "riscv64-linux"]; }; withBuildInfo = args: let settings = if builtins.isString args then {system = args;} else args; matchers = builtins.mapAttrs (name: match: match resolved.system) lib'.systems.match; validators = builtins.mapAttrs (name: validate: validate (resolved.gcc.arch or "default")) lib'.systems.match.architecture; platformInfo = { linux-kernel = settings.linux-kernel or {}; gcc = settings.gcc or {}; rustc = settings.rustc or {}; } // lib'.systems.platforms.select resolved; resolved = matchers // validators // platformInfo // { system = lib'.systems.from.string ( if settings ? triple then settings.triple else settings.system ); double = lib'.systems.into.double resolved.system; triple = lib'.systems.into.triple resolved.system; isExecutable = platform: (resolved.isAndroid == platform.isAndroid) && resolved.system.kernel == platform.system.kernel && lib'.systems.validate.compatible resolved.system.cpu platform.system.cpu; # The difference between `isStatic` and `hasSharedLibraries` is mainly the # addition of the `staticMarker` (see make-derivation.nix). Some # platforms, like embedded machines without a libc (e.g. arm-none-eabi) # don't support dynamic linking, but don't get the `staticMarker`. # `pkgsStatic` sets `isStatic=true`, so `pkgsStatic.hostPlatform` always # has the `staticMarker`. isStatic = resolved.isWasm || resolved.isRedox; # It is important that hasSharedLibraries==false when the platform has no # dynamic library loader. Various tools (including the gcc build system) # have knowledge of which platforms are incapable of dynamic linking, and # will still build on/for those platforms with --enable-shared, but simply # omit any `.so` build products such as libgcc_s.so. When that happens, # it causes hard-to-troubleshoot build failures. hasSharedLibraries = !resolved.isStatic && ( # Linux (allows multiple libcs) resolved.isAndroid || resolved.isGnu || resolved.isMusl # BSDs || resolved.isDarwin || resolved.isSunOS || resolved.isOpenBSD || resolved.isFreeBSD || resolved.isNetBSD # Windows || resolved.isCygwin || resolved.isMinGW ); libc = if resolved.isDarwin then "libSystem" else if resolved.isMinGW then "msvcrt" else if resolved.isWasi then "wasilibc" else if resolved.isRedox then "relibc" else if resolved.isMusl then "musl" else if resolved.isUClibc then "uclibc" else if resolved.isAndroid then "bionic" else if resolved.isLinux then "glibc" else if resolved.isFreeBSD then "fblibc" else if resolved.isNetBSD then "nblibc" else if resolved.isAvr then "avrlibc" else if resolved.isGhcjs then null else if resolved.isNone then "newlib" else "native/impure"; linker = if resolved.isDarwin then "cctools" else "bfd"; extensions = (lib.attrs.when resolved.hasSharedLibraries { shared = if resolved.isDarwin then ".dylib" else if resolved.isWindows then ".dll" else ".so"; }) // { static = if resolved.isWindows then ".lib" else ".a"; library = if resolved.isStatic then resolved.extensions.static else resolved.extensions.shared; executable = if resolved.isWindows then ".exe" else ""; }; uname = { system = if resolved.system.kernel.name == "linux" then "Linux" else if resolved.system.kernel.name == "windows" then "Windows" else if resolved.system.kernel.name == "darwin" then "Darwin" else if resolved.system.kernel.name == "netbsd" then "NetBSD" else if resolved.system.kernel.name == "freebsd" then "FreeBSD" else if resolved.system.kernel.name == "openbsd" then "OpenBSD" else if resolved.system.kernel.name == "wasi" then "Wasi" else if resolved.system.kernel.name == "redox" then "Redox" else if resolved.system.kernel.name == "redox" then "Genode" else null; processor = if resolved.isPower64 then "ppc64${lib.strings.when resolved.isLittleEndian "le"}" else if resolved.isPower then "ppc${lib.strings.when resolved.isLittleEndian "le"}" else if resolved.isMips64 then "mips64" else resolved.system.cpu.name; release = null; }; useAndroidPrebuilt = false; useiOSPrebuilt = false; linux.arch = if resolved.isAarch32 then "arm" else if resolved.isAarch64 then "arm64" else if resolved.isx86_32 then "i386" else if resolved.isx86_64 then "x86_64" # linux kernel does not distinguish microblaze/microblazeel else if resolved.isMicroBlaze then "microblaze" else if resolved.isMips32 then "mips" else if resolved.isMips64 then "mips" # linux kernel does not distinguish mips32/mips64 else if resolved.isPower then "powerpc" else if resolved.isRiscV then "riscv" else if resolved.isS390 then "s390" else if resolved.isLoongArch64 then "loongarch" else resolved.system.cpu.name; uboot.arch = if resolved.isx86_32 then "x86" # not i386 else if resolved.isMips64 then "mips64" # uboot *does* distinguish between mips32/mips64 else resolved.linuxArch; # other cases appear to agree with linuxArch qemu.arch = if resolved.isAarch32 then "arm" else if resolved.isS390 && !resolved.isS390x then null else if resolved.isx86_64 then "x86_64" else if resolved.isx86 then "i386" else if resolved.isMips64n32 then "mipsn32${lib.strings.when resolved.isLittleEndian "el"}" else if resolved.isMips64 then "mips64${lib.strings.when resolved.isLittleEndian "el"}" else resolved.uname.processor; efi.arch = if resolved.isx86_32 then "ia32" else if resolved.isx86_64 then "x64" else if resolved.isAarch32 then "arm" else if resolved.isAarch64 then "aa64" else resolved.system.cpu.name; darwin = { arch = if resolved.system.cpu.name == "armv7a" then "armv7" else if resolved.system.cpu.name == "aarch64" then "arm64" else resolved.system.cpu.name; platform = if resolved.isMacOS then "macos" else if resolved.isiOS then "ios" else null; sdk = { version = resolved.darwinSdkVersion or ( if resolved.isAarch64 then "11.0" else "10.12" ); min = resolved.darwin.sdk.version; variable = if resolved.isMacOS then "MACOSX_DEPLOYMENT_TARGET" else if resolved.isiOS then "IPHONEOS_DEPLOYMENT_TARGET" else null; }; }; } // settings; assertions = builtins.foldl' ( result: { assertion, message, }: if assertion resolved then result else builtins.throw message ) true (resolved.system.abi.assertions or []); in assert resolved.useAndroidPrebuild -> resolved.isAndroid; assert assertions; # And finally, return the generated system info. resolved; }; }; }