2014-04-14 16:26:48 +02:00
|
|
|
|
{ config, lib, pkgs, ... }:
|
2009-01-02 17:07:34 +01:00
|
|
|
|
|
2014-04-14 16:26:48 +02:00
|
|
|
|
with lib;
|
2009-01-02 17:07:34 +01:00
|
|
|
|
|
|
|
|
|
let
|
|
|
|
|
|
2012-05-14 03:53:47 +02:00
|
|
|
|
cfg = config.boot.loader.grub;
|
|
|
|
|
|
2015-01-14 10:30:57 +01:00
|
|
|
|
efi = config.boot.loader.efi;
|
|
|
|
|
|
2014-08-31 18:18:13 +02:00
|
|
|
|
realGrub = if cfg.version == 1 then pkgs.grub
|
2015-02-13 23:40:41 +01:00
|
|
|
|
else if cfg.zfsSupport then pkgs.grub2.override { zfsSupport = true; }
|
2015-12-21 20:20:29 +01:00
|
|
|
|
else if cfg.trustedBoot.enable
|
|
|
|
|
then if cfg.trustedBoot.isHPLaptop
|
|
|
|
|
then pkgs.trustedGrub-for-HP
|
|
|
|
|
else pkgs.trustedGrub
|
|
|
|
|
else pkgs.grub2;
|
2013-06-04 14:05:07 +02:00
|
|
|
|
|
|
|
|
|
grub =
|
|
|
|
|
# Don't include GRUB if we're only generating a GRUB menu (e.g.,
|
|
|
|
|
# in EC2 instances).
|
|
|
|
|
if cfg.devices == ["nodev"]
|
|
|
|
|
then null
|
|
|
|
|
else realGrub;
|
2009-12-15 22:11:39 +01:00
|
|
|
|
|
2015-01-14 10:30:57 +01:00
|
|
|
|
grubEfi =
|
|
|
|
|
# EFI version of Grub v2
|
2015-04-29 20:18:47 +02:00
|
|
|
|
if cfg.efiSupport && (cfg.version == 2)
|
2015-02-13 23:40:41 +01:00
|
|
|
|
then realGrub.override { efiSupport = cfg.efiSupport; }
|
2015-01-14 10:30:57 +01:00
|
|
|
|
else null;
|
|
|
|
|
|
2012-07-25 15:27:51 +02:00
|
|
|
|
f = x: if x == null then "" else "" + x;
|
|
|
|
|
|
2015-06-11 00:47:08 +02:00
|
|
|
|
grubConfig = args:
|
|
|
|
|
let
|
|
|
|
|
efiSysMountPoint = if args.efiSysMountPoint == null then args.path else args.efiSysMountPoint;
|
|
|
|
|
efiSysMountPoint' = replaceChars [ "/" ] [ "-" ] efiSysMountPoint;
|
|
|
|
|
in
|
|
|
|
|
pkgs.writeText "grub-config.xml" (builtins.toXML
|
2015-06-10 20:50:21 +02:00
|
|
|
|
{ splashImage = f cfg.splashImage;
|
2012-07-25 15:27:51 +02:00
|
|
|
|
grub = f grub;
|
2015-02-09 04:31:14 +01:00
|
|
|
|
grubTarget = f (grub.grubTarget or "");
|
2018-03-01 20:38:53 +01:00
|
|
|
|
shell = "${pkgs.runtimeShell}";
|
2015-07-05 18:54:35 +02:00
|
|
|
|
fullName = (builtins.parseDrvName realGrub.name).name;
|
2013-06-04 14:05:07 +02:00
|
|
|
|
fullVersion = (builtins.parseDrvName realGrub.name).version;
|
2015-01-14 10:30:57 +01:00
|
|
|
|
grubEfi = f grubEfi;
|
2015-02-16 20:19:44 +01:00
|
|
|
|
grubTargetEfi = if cfg.efiSupport && (cfg.version == 2) then f (grubEfi.grubTarget or "") else "";
|
2015-05-25 23:57:20 +02:00
|
|
|
|
bootPath = args.path;
|
2015-06-13 15:00:43 +02:00
|
|
|
|
storePath = config.boot.loader.grub.storePath;
|
2015-06-11 00:47:08 +02:00
|
|
|
|
bootloaderId = if args.efiBootloaderId == null then "NixOS${efiSysMountPoint'}" else args.efiBootloaderId;
|
2016-05-25 10:34:54 +02:00
|
|
|
|
timeout = if config.boot.loader.timeout == null then -1 else config.boot.loader.timeout;
|
2015-06-11 00:47:08 +02:00
|
|
|
|
inherit efiSysMountPoint;
|
2015-05-25 23:57:20 +02:00
|
|
|
|
inherit (args) devices;
|
|
|
|
|
inherit (efi) canTouchEfiVariables;
|
2013-06-04 14:05:07 +02:00
|
|
|
|
inherit (cfg)
|
2017-02-13 14:53:15 +01:00
|
|
|
|
version extraConfig extraPerEntryConfig extraEntries forceInstall useOSProber
|
2017-02-18 23:30:24 +01:00
|
|
|
|
extraEntriesBeforeNixOS extraPrepareConfig extraInitrd configurationLimit copyKernels
|
2016-09-13 19:46:53 +02:00
|
|
|
|
default fsIdentifier efiSupport efiInstallAsRemovable gfxmodeEfi gfxmodeBios;
|
2016-04-13 14:53:51 +02:00
|
|
|
|
path = (makeBinPath ([
|
2016-01-03 19:21:27 +01:00
|
|
|
|
pkgs.coreutils pkgs.gnused pkgs.gnugrep pkgs.findutils pkgs.diffutils pkgs.btrfs-progs
|
2017-02-13 14:53:15 +01:00
|
|
|
|
pkgs.utillinux ]
|
|
|
|
|
++ (optional (cfg.efiSupport && (cfg.version == 2)) pkgs.efibootmgr)
|
|
|
|
|
++ (optionals cfg.useOSProber [pkgs.busybox pkgs.os-prober])
|
2016-04-14 17:14:28 +02:00
|
|
|
|
)) + ":" + (makeSearchPathOutput "bin" "sbin" [
|
2014-08-31 18:18:13 +02:00
|
|
|
|
pkgs.mdadm pkgs.utillinux
|
2013-07-08 01:44:48 +02:00
|
|
|
|
]);
|
2018-06-05 16:37:12 +02:00
|
|
|
|
font = if cfg.font == null then ""
|
|
|
|
|
else (if lib.last (lib.splitString "." cfg.font) == "pf2"
|
2017-06-10 15:53:24 +02:00
|
|
|
|
then cfg.font
|
2018-06-05 16:37:12 +02:00
|
|
|
|
else "${convertedFont}");
|
2012-07-25 01:16:27 +02:00
|
|
|
|
});
|
2011-09-14 20:20:50 +02:00
|
|
|
|
|
2015-05-25 23:57:20 +02:00
|
|
|
|
bootDeviceCounters = fold (device: attr: attr // { "${device}" = (attr."${device}" or 0) + 1; }) {}
|
|
|
|
|
(concatMap (args: args.devices) cfg.mirroredBoots);
|
|
|
|
|
|
2017-06-10 15:53:24 +02:00
|
|
|
|
convertedFont = (pkgs.runCommand "grub-font-converted.pf2" {}
|
|
|
|
|
(builtins.concatStringsSep " "
|
|
|
|
|
([ "${realGrub}/bin/grub-mkfont"
|
|
|
|
|
cfg.font
|
|
|
|
|
"--output" "$out"
|
|
|
|
|
] ++ (optional (cfg.fontSize!=null) "--size ${toString cfg.fontSize}")))
|
|
|
|
|
);
|
2009-01-02 17:07:34 +01:00
|
|
|
|
in
|
|
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
2009-09-29 11:50:38 +02:00
|
|
|
|
###### interface
|
|
|
|
|
|
|
|
|
|
options = {
|
2009-10-13 23:39:23 +02:00
|
|
|
|
|
2009-09-29 11:50:38 +02:00
|
|
|
|
boot.loader.grub = {
|
|
|
|
|
|
|
|
|
|
enable = mkOption {
|
2013-11-27 16:54:20 +01:00
|
|
|
|
default = !config.boot.isContainer;
|
2013-10-07 11:05:33 +02:00
|
|
|
|
type = types.bool;
|
2009-09-29 11:50:38 +02:00
|
|
|
|
description = ''
|
2009-10-13 23:39:23 +02:00
|
|
|
|
Whether to enable the GNU GRUB boot loader.
|
2009-09-29 11:50:38 +02:00
|
|
|
|
'';
|
|
|
|
|
};
|
|
|
|
|
|
2009-10-13 23:39:18 +02:00
|
|
|
|
version = mkOption {
|
2013-10-07 11:06:08 +02:00
|
|
|
|
default = 2;
|
|
|
|
|
example = 1;
|
2013-10-07 11:05:33 +02:00
|
|
|
|
type = types.int;
|
2009-10-13 23:39:18 +02:00
|
|
|
|
description = ''
|
2013-10-07 11:05:33 +02:00
|
|
|
|
The version of GRUB to use: <literal>1</literal> for GRUB
|
2013-10-07 11:06:08 +02:00
|
|
|
|
Legacy (versions 0.9x), or <literal>2</literal> (the
|
|
|
|
|
default) for GRUB 2.
|
2009-10-13 23:39:18 +02:00
|
|
|
|
'';
|
|
|
|
|
};
|
|
|
|
|
|
2009-09-29 11:50:38 +02:00
|
|
|
|
device = mkOption {
|
|
|
|
|
default = "";
|
2018-02-24 11:50:23 +01:00
|
|
|
|
example = "/dev/disk/by-id/wwn-0x500001234567890a";
|
2013-10-30 11:02:04 +01:00
|
|
|
|
type = types.str;
|
2009-10-13 23:39:23 +02:00
|
|
|
|
description = ''
|
2012-05-14 03:53:47 +02:00
|
|
|
|
The device on which the GRUB boot loader will be installed.
|
|
|
|
|
The special value <literal>nodev</literal> means that a GRUB
|
|
|
|
|
boot menu will be generated, but GRUB itself will not
|
|
|
|
|
actually be installed. To install GRUB on multiple devices,
|
|
|
|
|
use <literal>boot.loader.grub.devices</literal>.
|
2012-03-08 22:37:30 +01:00
|
|
|
|
'';
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
devices = mkOption {
|
|
|
|
|
default = [];
|
2018-02-24 11:50:23 +01:00
|
|
|
|
example = [ "/dev/disk/by-id/wwn-0x500001234567890a" ];
|
2013-10-30 17:37:45 +01:00
|
|
|
|
type = types.listOf types.str;
|
2012-03-08 22:37:30 +01:00
|
|
|
|
description = ''
|
|
|
|
|
The devices on which the boot loader, GRUB, will be
|
|
|
|
|
installed. Can be used instead of <literal>device</literal> to
|
2015-07-25 18:54:26 +02:00
|
|
|
|
install GRUB onto multiple devices.
|
2009-10-13 23:39:23 +02:00
|
|
|
|
'';
|
2009-09-29 11:50:38 +02:00
|
|
|
|
};
|
|
|
|
|
|
2015-05-25 23:57:20 +02:00
|
|
|
|
mirroredBoots = mkOption {
|
|
|
|
|
default = [ ];
|
|
|
|
|
example = [
|
2018-02-24 11:50:23 +01:00
|
|
|
|
{ path = "/boot1"; devices = [ "/dev/disk/by-id/wwn-0x500001234567890a" ]; }
|
|
|
|
|
{ path = "/boot2"; devices = [ "/dev/disk/by-id/wwn-0x500009876543210a" ]; }
|
2015-05-25 23:57:20 +02:00
|
|
|
|
];
|
|
|
|
|
description = ''
|
|
|
|
|
Mirror the boot configuration to multiple partitions and install grub
|
|
|
|
|
to the respective devices corresponding to those partitions.
|
|
|
|
|
'';
|
|
|
|
|
|
2016-09-11 11:51:48 +02:00
|
|
|
|
type = with types; listOf (submodule {
|
|
|
|
|
options = {
|
|
|
|
|
|
|
|
|
|
path = mkOption {
|
|
|
|
|
example = "/boot1";
|
|
|
|
|
type = types.str;
|
|
|
|
|
description = ''
|
|
|
|
|
The path to the boot directory where GRUB will be written. Generally
|
|
|
|
|
this boot path should double as an EFI path.
|
|
|
|
|
'';
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
efiSysMountPoint = mkOption {
|
|
|
|
|
default = null;
|
|
|
|
|
example = "/boot1/efi";
|
|
|
|
|
type = types.nullOr types.str;
|
|
|
|
|
description = ''
|
|
|
|
|
The path to the efi system mount point. Usually this is the same
|
|
|
|
|
partition as the above path and can be left as null.
|
|
|
|
|
'';
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
efiBootloaderId = mkOption {
|
|
|
|
|
default = null;
|
|
|
|
|
example = "NixOS-fsid";
|
|
|
|
|
type = types.nullOr types.str;
|
|
|
|
|
description = ''
|
|
|
|
|
The id of the bootloader to store in efi nvram.
|
|
|
|
|
The default is to name it NixOS and append the path or efiSysMountPoint.
|
|
|
|
|
This is only used if <literal>boot.loader.efi.canTouchEfiVariables</literal> is true.
|
|
|
|
|
'';
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
devices = mkOption {
|
|
|
|
|
default = [ ];
|
2018-02-24 11:50:23 +01:00
|
|
|
|
example = [ "/dev/disk/by-id/wwn-0x500001234567890a" "/dev/disk/by-id/wwn-0x500009876543210a" ];
|
2016-09-11 11:51:48 +02:00
|
|
|
|
type = types.listOf types.str;
|
|
|
|
|
description = ''
|
|
|
|
|
The path to the devices which will have the GRUB MBR written.
|
|
|
|
|
Note these are typically device paths and not paths to partitions.
|
|
|
|
|
'';
|
|
|
|
|
};
|
2015-05-25 23:57:20 +02:00
|
|
|
|
|
|
|
|
|
};
|
2016-09-11 11:51:48 +02:00
|
|
|
|
});
|
2015-05-25 23:57:20 +02:00
|
|
|
|
};
|
|
|
|
|
|
2014-08-27 09:26:40 +02:00
|
|
|
|
configurationName = mkOption {
|
|
|
|
|
default = "";
|
|
|
|
|
example = "Stable 2.6.21";
|
|
|
|
|
type = types.str;
|
|
|
|
|
description = ''
|
|
|
|
|
GRUB entry name instead of default.
|
|
|
|
|
'';
|
|
|
|
|
};
|
|
|
|
|
|
2015-06-13 15:00:43 +02:00
|
|
|
|
storePath = mkOption {
|
|
|
|
|
default = "/nix/store";
|
|
|
|
|
type = types.str;
|
|
|
|
|
description = ''
|
|
|
|
|
Path to the Nix store when looking for kernels at boot.
|
|
|
|
|
Only makes sense when copyKernels is false.
|
|
|
|
|
'';
|
|
|
|
|
};
|
|
|
|
|
|
2012-04-02 19:19:21 +02:00
|
|
|
|
extraPrepareConfig = mkOption {
|
|
|
|
|
default = "";
|
2013-10-07 11:05:33 +02:00
|
|
|
|
type = types.lines;
|
2012-04-02 19:19:21 +02:00
|
|
|
|
description = ''
|
|
|
|
|
Additional bash commands to be run at the script that
|
2015-07-25 18:54:26 +02:00
|
|
|
|
prepares the GRUB menu entries.
|
2012-04-02 19:19:21 +02:00
|
|
|
|
'';
|
|
|
|
|
};
|
|
|
|
|
|
2010-06-17 00:18:26 +02:00
|
|
|
|
extraConfig = mkOption {
|
|
|
|
|
default = "";
|
|
|
|
|
example = "serial; terminal_output.serial";
|
2013-10-07 11:05:33 +02:00
|
|
|
|
type = types.lines;
|
2010-06-17 00:18:26 +02:00
|
|
|
|
description = ''
|
|
|
|
|
Additional GRUB commands inserted in the configuration file
|
|
|
|
|
just before the menu entries.
|
|
|
|
|
'';
|
|
|
|
|
};
|
|
|
|
|
|
2010-07-22 16:40:29 +02:00
|
|
|
|
extraPerEntryConfig = mkOption {
|
|
|
|
|
default = "";
|
|
|
|
|
example = "root (hd0)";
|
2013-10-07 11:05:33 +02:00
|
|
|
|
type = types.lines;
|
2010-07-22 16:40:29 +02:00
|
|
|
|
description = ''
|
|
|
|
|
Additional GRUB commands inserted in the configuration file
|
|
|
|
|
at the start of each NixOS menu entry.
|
|
|
|
|
'';
|
|
|
|
|
};
|
|
|
|
|
|
2009-09-29 11:50:38 +02:00
|
|
|
|
extraEntries = mkOption {
|
|
|
|
|
default = "";
|
2013-10-07 11:05:33 +02:00
|
|
|
|
type = types.lines;
|
2009-10-13 23:39:23 +02:00
|
|
|
|
example = ''
|
2012-03-28 12:34:40 +02:00
|
|
|
|
# GRUB 1 example (not GRUB 2 compatible)
|
2009-09-29 11:50:38 +02:00
|
|
|
|
title Windows
|
|
|
|
|
chainloader (hd0,1)+1
|
2012-03-28 12:34:40 +02:00
|
|
|
|
|
|
|
|
|
# GRUB 2 example
|
2014-04-20 19:41:15 +02:00
|
|
|
|
menuentry "Windows 7" {
|
|
|
|
|
chainloader (hd0,4)+1
|
2012-03-28 12:34:40 +02:00
|
|
|
|
}
|
2017-04-25 07:48:54 +02:00
|
|
|
|
|
|
|
|
|
# GRUB 2 with UEFI example, chainloading another distro
|
|
|
|
|
menuentry "Fedora" {
|
|
|
|
|
set root=(hd1,1)
|
|
|
|
|
chainloader /efi/fedora/grubx64.efi
|
|
|
|
|
}
|
2009-10-13 23:39:23 +02:00
|
|
|
|
'';
|
|
|
|
|
description = ''
|
|
|
|
|
Any additional entries you want added to the GRUB boot menu.
|
|
|
|
|
'';
|
2009-09-29 11:50:38 +02:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
extraEntriesBeforeNixOS = mkOption {
|
|
|
|
|
default = false;
|
2013-10-07 11:05:33 +02:00
|
|
|
|
type = types.bool;
|
2009-10-13 23:39:23 +02:00
|
|
|
|
description = ''
|
2009-09-29 11:50:38 +02:00
|
|
|
|
Whether extraEntries are included before the default option.
|
2009-10-13 23:39:23 +02:00
|
|
|
|
'';
|
2009-09-29 11:50:38 +02:00
|
|
|
|
};
|
|
|
|
|
|
2013-10-02 12:29:07 +02:00
|
|
|
|
extraFiles = mkOption {
|
2016-01-17 19:34:55 +01:00
|
|
|
|
type = types.attrsOf types.path;
|
2013-10-02 12:29:07 +02:00
|
|
|
|
default = {};
|
|
|
|
|
example = literalExample ''
|
2013-10-30 16:19:07 +01:00
|
|
|
|
{ "memtest.bin" = "''${pkgs.memtest86plus}/memtest.bin"; }
|
2013-10-02 12:29:07 +02:00
|
|
|
|
'';
|
|
|
|
|
description = ''
|
|
|
|
|
A set of files to be copied to <filename>/boot</filename>.
|
|
|
|
|
Each attribute name denotes the destination file name in
|
|
|
|
|
<filename>/boot</filename>, while the corresponding
|
|
|
|
|
attribute value specifies the source file.
|
|
|
|
|
'';
|
|
|
|
|
};
|
|
|
|
|
|
2017-02-18 23:30:24 +01:00
|
|
|
|
extraInitrd = mkOption {
|
|
|
|
|
type = types.nullOr types.path;
|
|
|
|
|
default = null;
|
2017-06-10 15:53:24 +02:00
|
|
|
|
example = "/boot/extra_initramfs.gz";
|
2017-02-18 23:30:24 +01:00
|
|
|
|
description = ''
|
|
|
|
|
The path to a second initramfs to be supplied to the kernel.
|
|
|
|
|
This ramfs will not be copied to the store, so that it can
|
|
|
|
|
contain secrets such as LUKS keyfiles or ssh keys.
|
|
|
|
|
This implies that rolling back to a previous configuration
|
|
|
|
|
won't rollback the state of this file.
|
|
|
|
|
'';
|
|
|
|
|
};
|
|
|
|
|
|
2017-02-13 14:53:15 +01:00
|
|
|
|
useOSProber = mkOption {
|
|
|
|
|
default = false;
|
|
|
|
|
type = types.bool;
|
|
|
|
|
description = ''
|
|
|
|
|
If set to true, append entries for other OSs detected by os-prober.
|
|
|
|
|
'';
|
|
|
|
|
};
|
|
|
|
|
|
2009-09-29 11:50:38 +02:00
|
|
|
|
splashImage = mkOption {
|
2015-02-23 18:00:21 +01:00
|
|
|
|
type = types.nullOr types.path;
|
2013-10-23 20:06:39 +02:00
|
|
|
|
example = literalExample "./my-background.png";
|
2009-10-13 23:39:23 +02:00
|
|
|
|
description = ''
|
2018-05-13 22:59:51 +02:00
|
|
|
|
Background image used for GRUB.
|
|
|
|
|
Set to <literal>null</literal> to run GRUB in text mode.
|
|
|
|
|
|
|
|
|
|
<note><para>
|
|
|
|
|
For grub 1:
|
|
|
|
|
It must be a 640x480,
|
2009-09-29 11:50:38 +02:00
|
|
|
|
14-colour image in XPM format, optionally compressed with
|
2018-05-13 22:59:51 +02:00
|
|
|
|
<command>gzip</command> or <command>bzip2</command>.
|
|
|
|
|
</para></note>
|
|
|
|
|
|
|
|
|
|
<note><para>
|
|
|
|
|
For grub 2:
|
|
|
|
|
File must be one of .png, .tga, .jpg, or .jpeg. JPEG images must
|
|
|
|
|
not be progressive.
|
|
|
|
|
The image will be scaled if necessary to fit the screen.
|
|
|
|
|
</para></note>
|
2009-10-13 23:39:23 +02:00
|
|
|
|
'';
|
2009-09-29 11:50:38 +02:00
|
|
|
|
};
|
|
|
|
|
|
2017-06-10 15:53:24 +02:00
|
|
|
|
font = mkOption {
|
|
|
|
|
type = types.nullOr types.path;
|
|
|
|
|
default = "${realGrub}/share/grub/unicode.pf2";
|
|
|
|
|
description = ''
|
|
|
|
|
Path to a TrueType, OpenType, or pf2 font to be used by Grub.
|
|
|
|
|
'';
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
fontSize = mkOption {
|
|
|
|
|
type = types.nullOr types.int;
|
|
|
|
|
example = literalExample 16;
|
|
|
|
|
default = null;
|
|
|
|
|
description = ''
|
|
|
|
|
Font size for the grub menu. Ignored unless <literal>font</literal>
|
|
|
|
|
is set to a ttf or otf font.
|
|
|
|
|
'';
|
|
|
|
|
};
|
|
|
|
|
|
2015-06-10 20:50:21 +02:00
|
|
|
|
gfxmodeEfi = mkOption {
|
|
|
|
|
default = "auto";
|
|
|
|
|
example = "1024x768";
|
|
|
|
|
type = types.str;
|
|
|
|
|
description = ''
|
2015-07-25 18:54:26 +02:00
|
|
|
|
The gfxmode to pass to GRUB when loading a graphical boot interface under EFI.
|
2015-06-10 20:50:21 +02:00
|
|
|
|
'';
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
gfxmodeBios = mkOption {
|
|
|
|
|
default = "1024x768";
|
|
|
|
|
example = "auto";
|
|
|
|
|
type = types.str;
|
|
|
|
|
description = ''
|
2015-07-25 18:54:26 +02:00
|
|
|
|
The gfxmode to pass to GRUB when loading a graphical boot interface under BIOS.
|
2015-06-10 20:50:21 +02:00
|
|
|
|
'';
|
|
|
|
|
};
|
|
|
|
|
|
2009-09-29 11:50:38 +02:00
|
|
|
|
configurationLimit = mkOption {
|
|
|
|
|
default = 100;
|
|
|
|
|
example = 120;
|
2013-10-07 11:05:33 +02:00
|
|
|
|
type = types.int;
|
2009-10-13 23:39:23 +02:00
|
|
|
|
description = ''
|
2009-09-29 11:50:38 +02:00
|
|
|
|
Maximum of configurations in boot menu. GRUB has problems when
|
|
|
|
|
there are too many entries.
|
2009-10-13 23:39:23 +02:00
|
|
|
|
'';
|
2009-09-29 11:50:38 +02:00
|
|
|
|
};
|
2009-01-02 17:07:34 +01:00
|
|
|
|
|
2009-09-29 11:50:38 +02:00
|
|
|
|
copyKernels = mkOption {
|
|
|
|
|
default = false;
|
2013-10-07 11:05:33 +02:00
|
|
|
|
type = types.bool;
|
2009-10-13 23:39:23 +02:00
|
|
|
|
description = ''
|
|
|
|
|
Whether the GRUB menu builder should copy kernels and initial
|
2009-12-16 19:57:02 +01:00
|
|
|
|
ramdisks to /boot. This is done automatically if /boot is
|
|
|
|
|
on a different partition than /.
|
2009-10-13 23:39:23 +02:00
|
|
|
|
'';
|
2009-09-29 11:50:38 +02:00
|
|
|
|
};
|
2009-10-13 23:39:23 +02:00
|
|
|
|
|
2009-12-11 01:51:07 +01:00
|
|
|
|
default = mkOption {
|
2009-12-15 19:21:55 +01:00
|
|
|
|
default = 0;
|
2013-10-07 11:05:33 +02:00
|
|
|
|
type = types.int;
|
2009-12-11 01:51:07 +01:00
|
|
|
|
description = ''
|
2009-12-15 19:21:55 +01:00
|
|
|
|
Index of the default menu item to be booted.
|
2009-12-11 01:51:07 +01:00
|
|
|
|
'';
|
|
|
|
|
};
|
|
|
|
|
|
2014-08-31 18:18:13 +02:00
|
|
|
|
fsIdentifier = mkOption {
|
|
|
|
|
default = "uuid";
|
2016-11-04 05:05:13 +01:00
|
|
|
|
type = types.enum [ "uuid" "label" "provided" ];
|
2014-04-09 20:27:18 +02:00
|
|
|
|
description = ''
|
2015-07-25 18:54:26 +02:00
|
|
|
|
Determines how GRUB will identify devices when generating the
|
2014-08-31 18:18:13 +02:00
|
|
|
|
configuration file. A value of uuid / label signifies that grub
|
|
|
|
|
will always resolve the uuid or label of the device before using
|
2015-07-25 18:54:26 +02:00
|
|
|
|
it in the configuration. A value of provided means that GRUB will
|
2014-08-31 18:18:13 +02:00
|
|
|
|
use the device name as show in <command>df</command> or
|
|
|
|
|
<command>mount</command>. Note, zfs zpools / datasets are ignored
|
|
|
|
|
and will always be mounted using their labels.
|
|
|
|
|
'';
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
zfsSupport = mkOption {
|
|
|
|
|
default = false;
|
|
|
|
|
type = types.bool;
|
|
|
|
|
description = ''
|
2016-04-06 18:16:23 +02:00
|
|
|
|
Whether GRUB should be built against libzfs.
|
2015-01-14 10:30:57 +01:00
|
|
|
|
ZFS support is only available for GRUB v2.
|
|
|
|
|
This option is ignored for GRUB v1.
|
|
|
|
|
'';
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
efiSupport = mkOption {
|
|
|
|
|
default = false;
|
|
|
|
|
type = types.bool;
|
|
|
|
|
description = ''
|
2016-04-06 18:16:23 +02:00
|
|
|
|
Whether GRUB should be built with EFI support.
|
2015-01-14 10:30:57 +01:00
|
|
|
|
EFI support is only available for GRUB v2.
|
|
|
|
|
This option is ignored for GRUB v1.
|
2014-04-09 20:27:18 +02:00
|
|
|
|
'';
|
|
|
|
|
};
|
|
|
|
|
|
2016-09-13 19:46:53 +02:00
|
|
|
|
efiInstallAsRemovable = mkOption {
|
|
|
|
|
default = false;
|
|
|
|
|
type = types.bool;
|
|
|
|
|
description = ''
|
2017-04-18 14:21:48 +02:00
|
|
|
|
Whether to invoke <literal>grub-install</literal> with
|
2016-09-16 20:12:35 +02:00
|
|
|
|
<literal>--removable</literal>.</para>
|
2016-09-13 19:46:53 +02:00
|
|
|
|
|
2016-09-16 20:12:35 +02:00
|
|
|
|
<para>Unless you turn this on, GRUB will install itself somewhere in
|
2016-09-16 19:09:50 +02:00
|
|
|
|
<literal>boot.loader.efi.efiSysMountPoint</literal> (exactly where
|
|
|
|
|
depends on other config variables). If you've set
|
2016-09-13 19:46:53 +02:00
|
|
|
|
<literal>boot.loader.efi.canTouchEfiVariables</literal> *AND* you
|
|
|
|
|
are currently booted in UEFI mode, then GRUB will use
|
2016-09-16 19:09:50 +02:00
|
|
|
|
<literal>efibootmgr</literal> to modify the boot order in the
|
|
|
|
|
EFI variables of your firmware to include this location. If you are
|
|
|
|
|
*not* booted in UEFI mode at the time GRUB is being installed, the
|
2016-09-13 19:46:53 +02:00
|
|
|
|
NVRAM will not be modified, and your system will not find GRUB at
|
2016-09-16 19:09:50 +02:00
|
|
|
|
boot time. However, GRUB will still return success so you may miss
|
|
|
|
|
the warning that gets printed ("<literal>efibootmgr: EFI variables
|
2016-09-16 20:12:35 +02:00
|
|
|
|
are not supported on this system.</literal>").</para>
|
2016-09-13 19:46:53 +02:00
|
|
|
|
|
2016-09-16 20:12:35 +02:00
|
|
|
|
<para>If you turn this feature on, GRUB will install itself in a
|
|
|
|
|
special location within <literal>efiSysMountPoint</literal> (namely
|
2016-09-16 19:09:50 +02:00
|
|
|
|
<literal>EFI/boot/boot$arch.efi</literal>) which the firmwares
|
2016-09-16 20:12:35 +02:00
|
|
|
|
are hardcoded to try first, regardless of NVRAM EFI variables.</para>
|
2016-09-13 19:46:53 +02:00
|
|
|
|
|
2016-09-16 20:12:35 +02:00
|
|
|
|
<para>To summarize, turn this on if:
|
2016-09-13 19:46:53 +02:00
|
|
|
|
<itemizedlist>
|
2016-09-16 20:12:35 +02:00
|
|
|
|
<listitem><para>You are installing NixOS and want it to boot in UEFI mode,
|
|
|
|
|
but you are currently booted in legacy mode</para></listitem>
|
|
|
|
|
<listitem><para>You want to make a drive that will boot regardless of
|
|
|
|
|
the NVRAM state of the computer (like a USB "removable" drive)</para></listitem>
|
|
|
|
|
<listitem><para>You simply dislike the idea of depending on NVRAM
|
|
|
|
|
state to make your drive bootable</para></listitem>
|
2017-04-18 14:21:48 +02:00
|
|
|
|
</itemizedlist>
|
2016-09-13 19:46:53 +02:00
|
|
|
|
'';
|
|
|
|
|
};
|
|
|
|
|
|
2014-09-21 20:41:46 +02:00
|
|
|
|
enableCryptodisk = mkOption {
|
|
|
|
|
default = false;
|
|
|
|
|
type = types.bool;
|
|
|
|
|
description = ''
|
2015-07-25 18:54:26 +02:00
|
|
|
|
Enable support for encrypted partitions. GRUB should automatically
|
2014-09-21 20:41:46 +02:00
|
|
|
|
unlock the correct encrypted partition and look for filesystems.
|
|
|
|
|
'';
|
|
|
|
|
};
|
|
|
|
|
|
2016-10-09 05:59:42 +02:00
|
|
|
|
forceInstall = mkOption {
|
|
|
|
|
default = false;
|
|
|
|
|
type = types.bool;
|
|
|
|
|
description = ''
|
|
|
|
|
Whether to try and forcibly install GRUB even if problems are
|
|
|
|
|
detected. It is not recommended to enable this unless you know what
|
|
|
|
|
you are doing.
|
|
|
|
|
'';
|
|
|
|
|
};
|
|
|
|
|
|
2015-12-21 20:20:29 +01:00
|
|
|
|
trustedBoot = {
|
|
|
|
|
|
|
|
|
|
enable = mkOption {
|
|
|
|
|
default = false;
|
|
|
|
|
type = types.bool;
|
|
|
|
|
description = ''
|
|
|
|
|
Enable trusted boot. GRUB will measure all critical components during
|
|
|
|
|
the boot process to offer TCG (TPM) support.
|
|
|
|
|
'';
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
systemHasTPM = mkOption {
|
|
|
|
|
default = "";
|
|
|
|
|
example = "YES_TPM_is_activated";
|
|
|
|
|
type = types.string;
|
|
|
|
|
description = ''
|
|
|
|
|
Assertion that the target system has an activated TPM. It is a safety
|
|
|
|
|
check before allowing the activation of 'trustedBoot.enable'. TrustedBoot
|
|
|
|
|
WILL FAIL TO BOOT YOUR SYSTEM if no TPM is available.
|
|
|
|
|
'';
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
isHPLaptop = mkOption {
|
|
|
|
|
default = false;
|
|
|
|
|
type = types.bool;
|
|
|
|
|
description = ''
|
|
|
|
|
Use a special version of TrustedGRUB that is needed by some HP laptops
|
|
|
|
|
and works only for the HP laptops.
|
|
|
|
|
'';
|
|
|
|
|
};
|
2015-07-05 18:54:35 +02:00
|
|
|
|
|
2015-10-05 13:45:20 +02:00
|
|
|
|
};
|
|
|
|
|
|
2009-01-02 17:07:34 +01:00
|
|
|
|
};
|
Making modular my previous changes for armv5tel. I updated the way to use
grub. Its options are no more inside 'boot', but inside 'boot.loader.grub'.
I added a new bootloader configuration for nixos, generationsDir. It creates
/boot/default/{init,initrd,kernel,system} symlinks, and the same for the generations
in /boot/system-$gen/{init,initrd,kernel,system}.
I can program the u-boot loader to load /boot/default files always, and have
a minimal nixos boot loader installer functionality. Additionally, I can refer
to the other system generations easily, with a simple 'ls' in /boot.
svn path=/nixos/trunk/; revision=17460
2009-09-27 23:51:37 +02:00
|
|
|
|
|
2009-01-02 17:07:34 +01:00
|
|
|
|
};
|
2011-09-14 20:20:50 +02:00
|
|
|
|
|
2009-09-29 11:50:38 +02:00
|
|
|
|
|
|
|
|
|
###### implementation
|
2009-01-02 17:07:34 +01:00
|
|
|
|
|
2013-10-24 01:48:07 +02:00
|
|
|
|
config = mkMerge [
|
2011-09-14 20:20:50 +02:00
|
|
|
|
|
2013-10-24 01:48:07 +02:00
|
|
|
|
{ boot.loader.grub.splashImage = mkDefault (
|
|
|
|
|
if cfg.version == 1 then pkgs.fetchurl {
|
|
|
|
|
url = http://www.gnome-look.org/CONTENT/content-files/36909-soft-tux.xpm.gz;
|
|
|
|
|
sha256 = "14kqdx2lfqvh40h6fjjzqgff1mwk74dmbjvmqphi6azzra7z8d59";
|
|
|
|
|
}
|
|
|
|
|
# GRUB 1.97 doesn't support gzipped XPMs.
|
2017-06-07 23:00:58 +02:00
|
|
|
|
else "${pkgs.nixos-artwork.wallpapers.gnome-dark}/share/artwork/gnome/Gnome_Dark.png");
|
2013-10-24 01:48:07 +02:00
|
|
|
|
}
|
2012-05-14 03:53:47 +02:00
|
|
|
|
|
2013-10-24 01:48:07 +02:00
|
|
|
|
(mkIf cfg.enable {
|
2013-10-23 20:06:39 +02:00
|
|
|
|
|
2013-10-24 01:48:07 +02:00
|
|
|
|
boot.loader.grub.devices = optional (cfg.device != "") cfg.device;
|
2013-10-17 13:30:49 +02:00
|
|
|
|
|
2015-05-25 23:57:20 +02:00
|
|
|
|
boot.loader.grub.mirroredBoots = optionals (cfg.devices != [ ]) [
|
|
|
|
|
{ path = "/boot"; inherit (cfg) devices; inherit (efi) efiSysMountPoint; }
|
|
|
|
|
];
|
|
|
|
|
|
2016-09-01 10:36:38 +02:00
|
|
|
|
system.build.installBootLoader =
|
|
|
|
|
let
|
|
|
|
|
install-grub-pl = pkgs.substituteAll {
|
|
|
|
|
src = ./install-grub.pl;
|
|
|
|
|
inherit (pkgs) utillinux;
|
|
|
|
|
btrfsprogs = pkgs.btrfs-progs;
|
|
|
|
|
};
|
|
|
|
|
in pkgs.writeScript "install-grub.sh" (''
|
2018-03-01 20:38:53 +01:00
|
|
|
|
#!${pkgs.runtimeShell}
|
2015-05-25 23:57:20 +02:00
|
|
|
|
set -e
|
2018-03-15 05:49:34 +01:00
|
|
|
|
export PERL5LIB=${makePerlPath (with pkgs.perlPackages; [ FileSlurp XMLLibXML XMLSAX XMLSAXBase ListCompare ])}
|
2015-05-25 23:57:20 +02:00
|
|
|
|
${optionalString cfg.enableCryptodisk "export GRUB_ENABLE_CRYPTODISK=y"}
|
|
|
|
|
'' + flip concatMapStrings cfg.mirroredBoots (args: ''
|
2016-09-01 10:36:38 +02:00
|
|
|
|
${pkgs.perl}/bin/perl ${install-grub-pl} ${grubConfig args} $@
|
2015-05-25 23:57:20 +02:00
|
|
|
|
''));
|
Making modular my previous changes for armv5tel. I updated the way to use
grub. Its options are no more inside 'boot', but inside 'boot.loader.grub'.
I added a new bootloader configuration for nixos, generationsDir. It creates
/boot/default/{init,initrd,kernel,system} symlinks, and the same for the generations
in /boot/system-$gen/{init,initrd,kernel,system}.
I can program the u-boot loader to load /boot/default files always, and have
a minimal nixos boot loader installer functionality. Additionally, I can refer
to the other system generations easily, with a simple 'ls' in /boot.
svn path=/nixos/trunk/; revision=17460
2009-09-27 23:51:37 +02:00
|
|
|
|
|
2013-10-24 01:48:07 +02:00
|
|
|
|
system.build.grub = grub;
|
2009-10-13 23:39:18 +02:00
|
|
|
|
|
2013-10-24 01:48:07 +02:00
|
|
|
|
# Common attribute for boot loaders so only one of them can be
|
|
|
|
|
# set at once.
|
|
|
|
|
system.boot.loader.id = "grub";
|
2009-10-13 23:39:18 +02:00
|
|
|
|
|
2013-10-30 14:18:41 +01:00
|
|
|
|
environment.systemPackages = optional (grub != null) grub;
|
2013-10-02 12:29:07 +02:00
|
|
|
|
|
2013-10-24 01:48:07 +02:00
|
|
|
|
boot.loader.grub.extraPrepareConfig =
|
|
|
|
|
concatStrings (mapAttrsToList (n: v: ''
|
|
|
|
|
${pkgs.coreutils}/bin/cp -pf "${v}" "/boot/${n}"
|
|
|
|
|
'') config.boot.loader.grub.extraFiles);
|
|
|
|
|
|
2015-05-25 23:57:20 +02:00
|
|
|
|
assertions = [
|
|
|
|
|
{
|
|
|
|
|
assertion = !cfg.zfsSupport || cfg.version == 2;
|
2015-07-25 18:54:26 +02:00
|
|
|
|
message = "Only GRUB version 2 provides ZFS support";
|
2015-05-25 23:57:20 +02:00
|
|
|
|
}
|
|
|
|
|
{
|
|
|
|
|
assertion = cfg.mirroredBoots != [ ];
|
|
|
|
|
message = "You must set the option ‘boot.loader.grub.devices’ or "
|
|
|
|
|
+ "'boot.loader.grub.mirroredBoots' to make the system bootable.";
|
|
|
|
|
}
|
|
|
|
|
{
|
2016-09-25 07:37:18 +02:00
|
|
|
|
assertion = cfg.efiSupport || all (c: c < 2) (mapAttrsToList (_: c: c) bootDeviceCounters);
|
2015-05-25 23:57:20 +02:00
|
|
|
|
message = "You cannot have duplicated devices in mirroredBoots";
|
|
|
|
|
}
|
2015-07-05 18:54:35 +02:00
|
|
|
|
{
|
2015-12-21 20:20:29 +01:00
|
|
|
|
assertion = !cfg.trustedBoot.enable || cfg.version == 2;
|
2015-07-05 18:54:35 +02:00
|
|
|
|
message = "Trusted GRUB is only available for GRUB 2";
|
|
|
|
|
}
|
|
|
|
|
{
|
2015-12-21 20:20:29 +01:00
|
|
|
|
assertion = !cfg.efiSupport || !cfg.trustedBoot.enable;
|
2015-07-05 18:54:35 +02:00
|
|
|
|
message = "Trusted GRUB does not have EFI support";
|
|
|
|
|
}
|
|
|
|
|
{
|
2015-12-21 20:20:29 +01:00
|
|
|
|
assertion = !cfg.zfsSupport || !cfg.trustedBoot.enable;
|
2015-07-05 18:54:35 +02:00
|
|
|
|
message = "Trusted GRUB does not have ZFS support";
|
|
|
|
|
}
|
|
|
|
|
{
|
2015-12-21 20:20:29 +01:00
|
|
|
|
assertion = !cfg.trustedBoot.enable || cfg.trustedBoot.systemHasTPM == "YES_TPM_is_activated";
|
2015-10-05 13:45:20 +02:00
|
|
|
|
message = "Trusted GRUB can break the system! Confirm that the system has an activated TPM by setting 'systemHasTPM'.";
|
2015-07-05 18:54:35 +02:00
|
|
|
|
}
|
2016-09-13 19:46:53 +02:00
|
|
|
|
{
|
|
|
|
|
assertion = cfg.efiInstallAsRemovable -> cfg.efiSupport;
|
|
|
|
|
message = "If you wish to to use boot.loader.grub.efiInstallAsRemovable, then turn on boot.loader.grub.efiSupport";
|
|
|
|
|
}
|
|
|
|
|
{
|
|
|
|
|
assertion = cfg.efiInstallAsRemovable -> !config.boot.loader.efi.canTouchEfiVariables;
|
|
|
|
|
message = "If you wish to to use boot.loader.grub.efiInstallAsRemovable, then turn off boot.loader.efi.canTouchEfiVariables";
|
|
|
|
|
}
|
2015-05-25 23:57:20 +02:00
|
|
|
|
] ++ flip concatMap cfg.mirroredBoots (args: [
|
|
|
|
|
{
|
|
|
|
|
assertion = args.devices != [ ];
|
2015-12-10 19:52:08 +01:00
|
|
|
|
message = "A boot path cannot have an empty devices string in ${args.path}";
|
2015-05-25 23:57:20 +02:00
|
|
|
|
}
|
|
|
|
|
{
|
|
|
|
|
assertion = hasPrefix "/" args.path;
|
|
|
|
|
message = "Boot paths must be absolute, not ${args.path}";
|
|
|
|
|
}
|
|
|
|
|
{
|
2015-05-26 08:03:24 +02:00
|
|
|
|
assertion = if args.efiSysMountPoint == null then true else hasPrefix "/" args.efiSysMountPoint;
|
2016-05-31 16:52:40 +02:00
|
|
|
|
message = "EFI paths must be absolute, not ${args.efiSysMountPoint}";
|
2015-05-25 23:57:20 +02:00
|
|
|
|
}
|
|
|
|
|
] ++ flip map args.devices (device: {
|
|
|
|
|
assertion = device == "nodev" || hasPrefix "/" device;
|
2016-04-20 22:27:34 +02:00
|
|
|
|
message = "GRUB devices must be absolute paths, not ${device} in ${args.path}";
|
2015-05-25 23:57:20 +02:00
|
|
|
|
}));
|
2013-10-24 01:48:07 +02:00
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
];
|
2011-09-14 20:20:50 +02:00
|
|
|
|
|
2015-10-14 18:05:50 +02:00
|
|
|
|
|
|
|
|
|
imports =
|
2016-03-27 00:01:43 +01:00
|
|
|
|
[ (mkRemovedOptionModule [ "boot" "loader" "grub" "bootDevice" ] "")
|
2015-10-14 18:05:50 +02:00
|
|
|
|
(mkRenamedOptionModule [ "boot" "copyKernels" ] [ "boot" "loader" "grub" "copyKernels" ])
|
|
|
|
|
(mkRenamedOptionModule [ "boot" "extraGrubEntries" ] [ "boot" "loader" "grub" "extraEntries" ])
|
|
|
|
|
(mkRenamedOptionModule [ "boot" "extraGrubEntriesBeforeNixos" ] [ "boot" "loader" "grub" "extraEntriesBeforeNixOS" ])
|
|
|
|
|
(mkRenamedOptionModule [ "boot" "grubDevice" ] [ "boot" "loader" "grub" "device" ])
|
|
|
|
|
(mkRenamedOptionModule [ "boot" "bootMount" ] [ "boot" "loader" "grub" "bootDevice" ])
|
|
|
|
|
(mkRenamedOptionModule [ "boot" "grubSplashImage" ] [ "boot" "loader" "grub" "splashImage" ])
|
|
|
|
|
];
|
|
|
|
|
|
2009-01-02 17:07:34 +01:00
|
|
|
|
}
|