2014-11-27 18:24:57 +01:00
|
|
|
{ config, lib, pkgs, ... }:
|
|
|
|
|
|
|
|
with lib;
|
|
|
|
|
|
|
|
let
|
2015-08-13 12:17:32 +02:00
|
|
|
cfg = config.virtualisation.virtualbox.host;
|
2016-09-13 03:42:16 +02:00
|
|
|
|
2018-08-11 20:40:00 +02:00
|
|
|
virtualbox = cfg.package.override {
|
2020-11-02 07:54:00 +01:00
|
|
|
inherit (cfg) enableHardening headless enableWebService;
|
2018-08-10 19:48:29 +02:00
|
|
|
extensionPack = if cfg.enableExtensionPack then pkgs.virtualboxExtpack else null;
|
2014-12-15 07:08:56 +01:00
|
|
|
};
|
|
|
|
|
2016-09-13 03:42:16 +02:00
|
|
|
kernelModules = config.boot.kernelPackages.virtualbox.override {
|
|
|
|
inherit virtualbox;
|
|
|
|
};
|
|
|
|
|
2014-11-27 18:24:57 +01:00
|
|
|
in
|
|
|
|
|
|
|
|
{
|
2015-08-13 12:17:32 +02:00
|
|
|
options.virtualisation.virtualbox.host = {
|
2022-08-30 02:30:04 +02:00
|
|
|
enable = mkEnableOption (lib.mdDoc "VirtualBox") // {
|
|
|
|
description = lib.mdDoc ''
|
2015-08-12 14:11:03 +02:00
|
|
|
Whether to enable VirtualBox.
|
2014-12-15 07:33:56 +01:00
|
|
|
|
2022-08-30 02:30:04 +02:00
|
|
|
::: {.note}
|
|
|
|
In order to pass USB devices from the host to the guests, the user
|
|
|
|
needs to be in the `vboxusers` group.
|
|
|
|
:::
|
2014-12-15 07:33:56 +01:00
|
|
|
'';
|
|
|
|
};
|
2014-12-15 07:08:56 +01:00
|
|
|
|
2022-08-30 02:30:04 +02:00
|
|
|
enableExtensionPack = mkEnableOption (lib.mdDoc "VirtualBox extension pack") // {
|
|
|
|
description = lib.mdDoc ''
|
2018-08-10 19:48:29 +02:00
|
|
|
Whether to install the Oracle Extension Pack for VirtualBox.
|
|
|
|
|
2022-08-30 02:30:04 +02:00
|
|
|
::: {.important}
|
|
|
|
You must set `nixpkgs.config.allowUnfree = true` in
|
|
|
|
order to use this. This requires you accept the VirtualBox PUEL.
|
|
|
|
:::
|
2018-08-10 19:48:29 +02:00
|
|
|
'';
|
|
|
|
};
|
2018-05-01 22:45:31 +02:00
|
|
|
|
2018-08-11 20:40:00 +02:00
|
|
|
package = mkOption {
|
|
|
|
type = types.package;
|
|
|
|
default = pkgs.virtualbox;
|
2021-10-03 18:06:03 +02:00
|
|
|
defaultText = literalExpression "pkgs.virtualbox";
|
2022-07-28 23:19:15 +02:00
|
|
|
description = lib.mdDoc ''
|
2018-08-11 20:40:00 +02:00
|
|
|
Which VirtualBox package to use.
|
|
|
|
'';
|
|
|
|
};
|
2018-05-01 22:45:31 +02:00
|
|
|
|
2014-12-15 07:08:56 +01:00
|
|
|
addNetworkInterface = mkOption {
|
|
|
|
type = types.bool;
|
|
|
|
default = true;
|
2022-07-28 23:19:15 +02:00
|
|
|
description = lib.mdDoc ''
|
2014-12-15 07:08:56 +01:00
|
|
|
Automatically set up a vboxnet0 host-only network interface.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
enableHardening = mkOption {
|
2014-12-11 23:28:09 +01:00
|
|
|
type = types.bool;
|
2014-12-18 18:12:25 +01:00
|
|
|
default = true;
|
2022-08-30 02:30:04 +02:00
|
|
|
description = lib.mdDoc ''
|
2014-12-15 07:08:56 +01:00
|
|
|
Enable hardened VirtualBox, which ensures that only the binaries in the
|
|
|
|
system path get access to the devices exposed by the kernel modules
|
|
|
|
instead of all users in the vboxusers group.
|
|
|
|
|
2022-08-30 02:30:04 +02:00
|
|
|
::: {.important}
|
|
|
|
Disabling this can put your system's security at risk, as local users
|
|
|
|
in the vboxusers group can tamper with the VirtualBox device files.
|
|
|
|
:::
|
2014-12-15 07:08:56 +01:00
|
|
|
'';
|
2014-12-11 23:28:09 +01:00
|
|
|
};
|
2016-09-01 20:54:58 +02:00
|
|
|
|
|
|
|
headless = mkOption {
|
|
|
|
type = types.bool;
|
|
|
|
default = false;
|
2022-07-28 23:19:15 +02:00
|
|
|
description = lib.mdDoc ''
|
2016-09-01 20:54:58 +02:00
|
|
|
Use VirtualBox installation without GUI and Qt dependency. Useful to enable on servers
|
|
|
|
and when virtual machines are controlled only via SSH.
|
|
|
|
'';
|
|
|
|
};
|
2020-11-02 07:54:00 +01:00
|
|
|
|
|
|
|
enableWebService = mkOption {
|
|
|
|
type = types.bool;
|
|
|
|
default = false;
|
2022-07-28 23:19:15 +02:00
|
|
|
description = lib.mdDoc ''
|
2020-11-02 07:54:00 +01:00
|
|
|
Build VirtualBox web service tool (vboxwebsrv) to allow managing VMs via other webpage frontend tools. Useful for headless servers.
|
|
|
|
'';
|
|
|
|
};
|
2014-11-27 18:24:57 +01:00
|
|
|
};
|
|
|
|
|
2014-12-15 07:08:56 +01:00
|
|
|
config = mkIf cfg.enable (mkMerge [{
|
2019-02-23 18:50:37 +01:00
|
|
|
warnings = mkIf (config.nixpkgs.config.virtualbox.enableExtensionPack or false)
|
2019-03-09 11:39:22 +01:00
|
|
|
["'nixpkgs.virtualbox.enableExtensionPack' has no effect, please use 'virtualisation.virtualbox.host.enableExtensionPack'"];
|
2014-11-27 18:24:57 +01:00
|
|
|
boot.kernelModules = [ "vboxdrv" "vboxnetadp" "vboxnetflt" ];
|
2016-09-13 03:42:16 +02:00
|
|
|
boot.extraModulePackages = [ kernelModules ];
|
2014-11-27 18:24:57 +01:00
|
|
|
environment.systemPackages = [ virtualbox ];
|
|
|
|
|
2017-01-29 08:58:12 +01:00
|
|
|
security.wrappers = let
|
2017-02-14 22:56:37 +01:00
|
|
|
mkSuid = program: {
|
2014-12-18 11:37:32 +01:00
|
|
|
source = "${virtualbox}/libexec/virtualbox/${program}";
|
2014-11-29 08:09:50 +01:00
|
|
|
owner = "root";
|
|
|
|
group = "vboxusers";
|
|
|
|
setuid = true;
|
2017-02-14 22:56:37 +01:00
|
|
|
};
|
2022-10-08 23:41:17 +02:00
|
|
|
executables = [
|
|
|
|
"VBoxHeadless"
|
|
|
|
"VBoxNetAdpCtl"
|
|
|
|
"VBoxNetDHCP"
|
|
|
|
"VBoxNetNAT"
|
|
|
|
"VBoxVolInfo"
|
|
|
|
] ++ (lib.optionals (!cfg.headless) [
|
|
|
|
"VBoxSDL"
|
|
|
|
"VirtualBoxVM"
|
|
|
|
]);
|
2017-02-14 22:56:37 +01:00
|
|
|
in mkIf cfg.enableHardening
|
2022-10-08 23:41:17 +02:00
|
|
|
(builtins.listToAttrs (map (x: { name = x; value = mkSuid x; }) executables));
|
2014-11-29 08:09:50 +01:00
|
|
|
|
2018-06-30 01:58:35 +02:00
|
|
|
users.groups.vboxusers.gid = config.ids.gids.vboxusers;
|
2014-11-27 18:24:57 +01:00
|
|
|
|
|
|
|
services.udev.extraRules =
|
|
|
|
''
|
|
|
|
KERNEL=="vboxdrv", OWNER="root", GROUP="vboxusers", MODE="0660", TAG+="systemd"
|
|
|
|
KERNEL=="vboxdrvu", OWNER="root", GROUP="root", MODE="0666", TAG+="systemd"
|
|
|
|
KERNEL=="vboxnetctl", OWNER="root", GROUP="vboxusers", MODE="0660", TAG+="systemd"
|
|
|
|
SUBSYSTEM=="usb_device", ACTION=="add", RUN+="${virtualbox}/libexec/virtualbox/VBoxCreateUSBNode.sh $major $minor $attr{bDeviceClass}"
|
|
|
|
SUBSYSTEM=="usb", ACTION=="add", ENV{DEVTYPE}=="usb_device", RUN+="${virtualbox}/libexec/virtualbox/VBoxCreateUSBNode.sh $major $minor $attr{bDeviceClass}"
|
|
|
|
SUBSYSTEM=="usb_device", ACTION=="remove", RUN+="${virtualbox}/libexec/virtualbox/VBoxCreateUSBNode.sh --remove $major $minor"
|
|
|
|
SUBSYSTEM=="usb", ACTION=="remove", ENV{DEVTYPE}=="usb_device", RUN+="${virtualbox}/libexec/virtualbox/VBoxCreateUSBNode.sh --remove $major $minor"
|
|
|
|
'';
|
|
|
|
|
2017-01-29 05:48:03 +01:00
|
|
|
# Since we lack the right setuid/setcap binaries, set up a host-only network by default.
|
2014-12-15 07:08:56 +01:00
|
|
|
} (mkIf cfg.addNetworkInterface {
|
2019-08-13 23:52:01 +02:00
|
|
|
systemd.services.vboxnet0 =
|
2014-11-27 18:24:57 +01:00
|
|
|
{ description = "VirtualBox vboxnet0 Interface";
|
|
|
|
requires = [ "dev-vboxnetctl.device" ];
|
|
|
|
after = [ "dev-vboxnetctl.device" ];
|
|
|
|
wantedBy = [ "network.target" "sys-subsystem-net-devices-vboxnet0.device" ];
|
|
|
|
path = [ virtualbox ];
|
|
|
|
serviceConfig.RemainAfterExit = true;
|
|
|
|
serviceConfig.Type = "oneshot";
|
2014-12-15 19:12:58 +01:00
|
|
|
serviceConfig.PrivateTmp = true;
|
|
|
|
environment.VBOX_USER_HOME = "/tmp";
|
2014-11-27 18:24:57 +01:00
|
|
|
script =
|
|
|
|
''
|
|
|
|
if ! [ -e /sys/class/net/vboxnet0 ]; then
|
|
|
|
VBoxManage hostonlyif create
|
2014-12-15 19:12:58 +01:00
|
|
|
cat /tmp/VBoxSVC.log >&2
|
2014-11-27 18:24:57 +01:00
|
|
|
fi
|
|
|
|
'';
|
|
|
|
postStop =
|
|
|
|
''
|
|
|
|
VBoxManage hostonlyif remove vboxnet0
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
2018-02-19 17:48:15 +01:00
|
|
|
networking.interfaces.vboxnet0.ipv4.addresses = [{ address = "192.168.56.1"; prefixLength = 24; }];
|
2015-11-07 14:07:11 +01:00
|
|
|
# Make sure NetworkManager won't assume this interface being up
|
|
|
|
# means we have internet access.
|
|
|
|
networking.networkmanager.unmanaged = ["vboxnet0"];
|
2019-10-26 00:45:42 +02:00
|
|
|
}) (mkIf config.networking.useNetworkd {
|
|
|
|
systemd.network.networks."40-vboxnet0".extraConfig = ''
|
|
|
|
[Link]
|
|
|
|
RequiredForOnline=no
|
|
|
|
'';
|
|
|
|
})
|
|
|
|
|
|
|
|
]);
|
2014-11-27 18:24:57 +01:00
|
|
|
}
|