mirror of
https://github.com/ilyakooo0/nixpkgs.git
synced 2024-12-28 22:32:58 +03:00
lib: Better use the module type system in platform parsing
I need some module system types here so I can next fix meta-checks for derivations. I'd like to use a "proper" record type here, but submodule types seem overkill so holding off with ad-hoc stuff for now. In practice, all I need for the next step are the `.check` functions so this is good, especially as the submodule check function is shallow, saving full inductive type-checking for a later step.
This commit is contained in:
parent
dc52fc6dda
commit
ad78ba1efa
@ -4,6 +4,16 @@
|
|||||||
# http://llvm.org/docs/doxygen/html/Triple_8cpp_source.html especially
|
# http://llvm.org/docs/doxygen/html/Triple_8cpp_source.html especially
|
||||||
# Triple::normalize. Parsing should essentially act as a more conservative
|
# Triple::normalize. Parsing should essentially act as a more conservative
|
||||||
# version of that last function.
|
# version of that last function.
|
||||||
|
#
|
||||||
|
# Most of the types below come in "open" and "closed" pairs. The open ones
|
||||||
|
# specify what information we need to know about systems in general, and the
|
||||||
|
# closed ones are sub-types representing the whitelist of systems we support in
|
||||||
|
# practice.
|
||||||
|
#
|
||||||
|
# Code in the remainder of nixpkgs shouldn't rely on the closed ones in
|
||||||
|
# e.g. exhaustive cases. Its more a sanity check to make sure nobody defines
|
||||||
|
# systems that overlap with existing ones and won't notice something amiss.
|
||||||
|
#
|
||||||
{ lib }:
|
{ lib }:
|
||||||
with lib.lists;
|
with lib.lists;
|
||||||
with lib.types;
|
with lib.types;
|
||||||
@ -11,29 +21,52 @@ with lib.attrsets;
|
|||||||
with (import ./inspect.nix { inherit lib; }).predicates;
|
with (import ./inspect.nix { inherit lib; }).predicates;
|
||||||
|
|
||||||
let
|
let
|
||||||
setTypesAssert = type: pred:
|
inherit (lib.options) mergeOneOption;
|
||||||
|
|
||||||
|
setTypes = type:
|
||||||
mapAttrs (name: value:
|
mapAttrs (name: value:
|
||||||
assert pred value;
|
assert type.check value;
|
||||||
setType type ({ inherit name; } // value));
|
setType type.name ({ inherit name; } // value));
|
||||||
setTypes = type: setTypesAssert type (_: true);
|
|
||||||
|
|
||||||
in
|
in
|
||||||
|
|
||||||
rec {
|
rec {
|
||||||
|
|
||||||
isSignificantByte = isType "significant-byte";
|
################################################################################
|
||||||
significantBytes = setTypes "significant-byte" {
|
|
||||||
|
types.openSignifiantByte = mkOptionType {
|
||||||
|
name = "significant-byte";
|
||||||
|
description = "Endianness";
|
||||||
|
merge = mergeOneOption;
|
||||||
|
};
|
||||||
|
|
||||||
|
types.significantByte = enum (attrValues significantBytes);
|
||||||
|
|
||||||
|
significantBytes = setTypes types.openSignifiantByte {
|
||||||
bigEndian = {};
|
bigEndian = {};
|
||||||
littleEndian = {};
|
littleEndian = {};
|
||||||
};
|
};
|
||||||
|
|
||||||
isCpuType = isType "cpu-type";
|
################################################################################
|
||||||
cpuTypes = with significantBytes; setTypesAssert "cpu-type"
|
|
||||||
(x: elem x.bits [8 16 32 64 128]
|
# Reasonable power of 2
|
||||||
&& (if 8 < x.bits
|
types.bitWidth = enum [ 8 16 32 64 128 ];
|
||||||
then isSignificantByte x.significantByte
|
|
||||||
else !(x ? significantByte)))
|
################################################################################
|
||||||
{
|
|
||||||
|
types.openCpuType = mkOptionType {
|
||||||
|
name = "cpu-type";
|
||||||
|
description = "instruction set architecture name and information";
|
||||||
|
merge = mergeOneOption;
|
||||||
|
check = x: types.bitWidth.check x.bits
|
||||||
|
&& (if 8 < x.bits
|
||||||
|
then types.significantByte.check x.significantByte
|
||||||
|
else !(x ? significantByte));
|
||||||
|
};
|
||||||
|
|
||||||
|
types.cpuType = enum (attrValues cpuTypes);
|
||||||
|
|
||||||
|
cpuTypes = with significantBytes; setTypes types.openCpuType {
|
||||||
arm = { bits = 32; significantByte = littleEndian; family = "arm"; };
|
arm = { bits = 32; significantByte = littleEndian; family = "arm"; };
|
||||||
armv5tel = { bits = 32; significantByte = littleEndian; family = "arm"; };
|
armv5tel = { bits = 32; significantByte = littleEndian; family = "arm"; };
|
||||||
armv6l = { bits = 32; significantByte = littleEndian; family = "arm"; };
|
armv6l = { bits = 32; significantByte = littleEndian; family = "arm"; };
|
||||||
@ -50,16 +83,34 @@ rec {
|
|||||||
wasm64 = { bits = 64; significantByte = littleEndian; family = "wasm"; };
|
wasm64 = { bits = 64; significantByte = littleEndian; family = "wasm"; };
|
||||||
};
|
};
|
||||||
|
|
||||||
isVendor = isType "vendor";
|
################################################################################
|
||||||
vendors = setTypes "vendor" {
|
|
||||||
|
types.openVendor = mkOptionType {
|
||||||
|
name = "vendor";
|
||||||
|
description = "vendor for the platform";
|
||||||
|
merge = mergeOneOption;
|
||||||
|
};
|
||||||
|
|
||||||
|
types.vendor = enum (attrValues vendors);
|
||||||
|
|
||||||
|
vendors = setTypes types.openVendor {
|
||||||
apple = {};
|
apple = {};
|
||||||
pc = {};
|
pc = {};
|
||||||
|
|
||||||
unknown = {};
|
unknown = {};
|
||||||
};
|
};
|
||||||
|
|
||||||
isExecFormat = isType "exec-format";
|
################################################################################
|
||||||
execFormats = setTypes "exec-format" {
|
|
||||||
|
types.openExecFormat = mkOptionType {
|
||||||
|
name = "exec-format";
|
||||||
|
description = "executable container used by the kernel";
|
||||||
|
merge = mergeOneOption;
|
||||||
|
};
|
||||||
|
|
||||||
|
types.execFormat = enum (attrValues execFormats);
|
||||||
|
|
||||||
|
execFormats = setTypes types.openExecFormat {
|
||||||
aout = {}; # a.out
|
aout = {}; # a.out
|
||||||
elf = {};
|
elf = {};
|
||||||
macho = {};
|
macho = {};
|
||||||
@ -68,15 +119,33 @@ rec {
|
|||||||
unknown = {};
|
unknown = {};
|
||||||
};
|
};
|
||||||
|
|
||||||
isKernelFamily = isType "kernel-family";
|
################################################################################
|
||||||
kernelFamilies = setTypes "kernel-family" {
|
|
||||||
|
types.openKernelFamily = mkOptionType {
|
||||||
|
name = "exec-format";
|
||||||
|
description = "executable container used by the kernel";
|
||||||
|
merge = mergeOneOption;
|
||||||
|
};
|
||||||
|
|
||||||
|
types.kernelFamily = enum (attrValues kernelFamilies);
|
||||||
|
|
||||||
|
kernelFamilies = setTypes types.openKernelFamily {
|
||||||
bsd = {};
|
bsd = {};
|
||||||
};
|
};
|
||||||
|
|
||||||
isKernel = x: isType "kernel" x;
|
################################################################################
|
||||||
kernels = with execFormats; with kernelFamilies; setTypesAssert "kernel"
|
|
||||||
(x: isExecFormat x.execFormat && all isKernelFamily (attrValues x.families))
|
types.openKernel = mkOptionType {
|
||||||
{
|
name = "kernel";
|
||||||
|
description = "kernel name and information";
|
||||||
|
merge = mergeOneOption;
|
||||||
|
check = x: types.execFormat.check x.execFormat
|
||||||
|
&& all types.kernelFamily.check (attrValues x.families);
|
||||||
|
};
|
||||||
|
|
||||||
|
types.kernel = enum (attrValues kernels);
|
||||||
|
|
||||||
|
kernels = with execFormats; with kernelFamilies; setTypes types.openKernel {
|
||||||
darwin = { execFormat = macho; families = { }; };
|
darwin = { execFormat = macho; families = { }; };
|
||||||
freebsd = { execFormat = elf; families = { inherit bsd; }; };
|
freebsd = { execFormat = elf; families = { inherit bsd; }; };
|
||||||
hurd = { execFormat = elf; families = { }; };
|
hurd = { execFormat = elf; families = { }; };
|
||||||
@ -93,8 +162,17 @@ rec {
|
|||||||
win32 = kernels.windows;
|
win32 = kernels.windows;
|
||||||
};
|
};
|
||||||
|
|
||||||
isAbi = isType "abi";
|
################################################################################
|
||||||
abis = setTypes "abi" {
|
|
||||||
|
types.openAbi = mkOptionType {
|
||||||
|
name = "abi";
|
||||||
|
description = "binary interface for compiled code and syscalls";
|
||||||
|
merge = mergeOneOption;
|
||||||
|
};
|
||||||
|
|
||||||
|
types.abi = enum (attrValues abis);
|
||||||
|
|
||||||
|
abis = setTypes types.openAbi {
|
||||||
cygnus = {};
|
cygnus = {};
|
||||||
gnu = {};
|
gnu = {};
|
||||||
msvc = {};
|
msvc = {};
|
||||||
@ -106,12 +184,24 @@ rec {
|
|||||||
unknown = {};
|
unknown = {};
|
||||||
};
|
};
|
||||||
|
|
||||||
|
################################################################################
|
||||||
|
|
||||||
|
types.system = mkOptionType {
|
||||||
|
name = "system";
|
||||||
|
description = "fully parsed representation of llvm- or nix-style platform tuple";
|
||||||
|
merge = mergeOneOption;
|
||||||
|
check = { cpu, vendor, kernel, abi }:
|
||||||
|
types.cpuType.check cpu
|
||||||
|
&& types.vendor.check vendor
|
||||||
|
&& types.kernel.check kernel
|
||||||
|
&& types.abi.check abi;
|
||||||
|
};
|
||||||
|
|
||||||
isSystem = isType "system";
|
isSystem = isType "system";
|
||||||
mkSystem = { cpu, vendor, kernel, abi }:
|
|
||||||
assert isCpuType cpu && isVendor vendor && isKernel kernel && isAbi abi;
|
mkSystem = components:
|
||||||
setType "system" {
|
assert types.system.check components;
|
||||||
inherit cpu vendor kernel abi;
|
setType "system" components;
|
||||||
};
|
|
||||||
|
|
||||||
mkSkeletonFromList = l: {
|
mkSkeletonFromList = l: {
|
||||||
"2" = # We only do 2-part hacks for things Nix already supports
|
"2" = # We only do 2-part hacks for things Nix already supports
|
||||||
@ -174,4 +264,6 @@ rec {
|
|||||||
optAbi = lib.optionalString (abi != abis.unknown) "-${abi.name}";
|
optAbi = lib.optionalString (abi != abis.unknown) "-${abi.name}";
|
||||||
in "${cpu.name}-${vendor.name}-${kernel.name}${optAbi}";
|
in "${cpu.name}-${vendor.name}-${kernel.name}${optAbi}";
|
||||||
|
|
||||||
|
################################################################################
|
||||||
|
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user