134 lines
3.9 KiB
Nix
134 lines
3.9 KiB
Nix
# Module for VirtualBox guests.
|
|
{
|
|
config,
|
|
lib,
|
|
pkgs,
|
|
...
|
|
}:
|
|
let
|
|
cfg = config.virtualisation.virtualbox.guest;
|
|
kernel = config.boot.kernelPackages;
|
|
|
|
mkVirtualBoxUserService = serviceArgs: {
|
|
description = "VirtualBox Guest User Services ${serviceArgs}";
|
|
|
|
wantedBy = [ "graphical-session.target" ];
|
|
partOf = [ "graphical-session.target" ];
|
|
|
|
# The graphical session may not be ready when starting the service
|
|
# Hence, check if the DISPLAY env var is set, otherwise fail, wait and retry again
|
|
startLimitBurst = 20;
|
|
|
|
unitConfig.ConditionVirtualization = "oracle";
|
|
|
|
# Check if the display environment is ready, otherwise fail
|
|
preStart = "${pkgs.bash}/bin/bash -c \"if [ -z $DISPLAY ]; then exit 1; fi\"";
|
|
serviceConfig = {
|
|
ExecStart = "@${kernel.virtualboxGuestAdditions}/bin/VBoxClient --foreground ${serviceArgs}";
|
|
# Wait after a failure, hoping that the display environment is ready after waiting
|
|
RestartSec = 2;
|
|
Restart = "always";
|
|
};
|
|
};
|
|
in
|
|
{
|
|
imports = [
|
|
(lib.mkRenamedOptionModule
|
|
[
|
|
"virtualisation"
|
|
"virtualbox"
|
|
"guest"
|
|
"draganddrop"
|
|
]
|
|
[
|
|
"virtualisation"
|
|
"virtualbox"
|
|
"guest"
|
|
"dragAndDrop"
|
|
]
|
|
)
|
|
];
|
|
|
|
options.virtualisation.virtualbox.guest = {
|
|
enable = lib.mkOption {
|
|
default = false;
|
|
type = lib.types.bool;
|
|
description = "Whether to enable the VirtualBox service and other guest additions.";
|
|
};
|
|
|
|
clipboard = lib.mkOption {
|
|
default = true;
|
|
type = lib.types.bool;
|
|
description = "Whether to enable clipboard support.";
|
|
};
|
|
|
|
seamless = lib.mkOption {
|
|
default = true;
|
|
type = lib.types.bool;
|
|
description = "Whether to enable seamless mode. When activated windows from the guest appear next to the windows of the host.";
|
|
};
|
|
|
|
dragAndDrop = lib.mkOption {
|
|
default = true;
|
|
type = lib.types.bool;
|
|
description = "Whether to enable drag and drop support.";
|
|
};
|
|
};
|
|
|
|
###### implementation
|
|
|
|
config = lib.mkIf cfg.enable (
|
|
lib.mkMerge [
|
|
{
|
|
assertions = [
|
|
{
|
|
assertion = pkgs.stdenv.hostPlatform.isx86;
|
|
message = "Virtualbox not currently supported on ${pkgs.stdenv.hostPlatform.system}";
|
|
}
|
|
];
|
|
|
|
environment.systemPackages = [ kernel.virtualboxGuestAdditions ];
|
|
|
|
boot.extraModulePackages = [ kernel.virtualboxGuestAdditions ];
|
|
|
|
boot.supportedFilesystems = [ "vboxsf" ];
|
|
boot.initrd.supportedFilesystems = [ "vboxsf" ];
|
|
|
|
users.groups.vboxsf.gid = config.ids.gids.vboxsf;
|
|
|
|
systemd.services.virtualbox = {
|
|
description = "VirtualBox Guest Services";
|
|
|
|
wantedBy = [ "multi-user.target" ];
|
|
requires = [ "dev-vboxguest.device" ];
|
|
after = [ "dev-vboxguest.device" ];
|
|
|
|
unitConfig.ConditionVirtualization = "oracle";
|
|
|
|
serviceConfig.ExecStart = "@${kernel.virtualboxGuestAdditions}/bin/VBoxService VBoxService --foreground";
|
|
};
|
|
|
|
services.udev.extraRules = ''
|
|
# /dev/vboxuser is necessary for VBoxClient to work. Maybe we
|
|
# should restrict this to logged-in users.
|
|
KERNEL=="vboxuser", OWNER="root", GROUP="root", MODE="0666"
|
|
|
|
# Allow systemd dependencies on vboxguest.
|
|
SUBSYSTEM=="misc", KERNEL=="vboxguest", TAG+="systemd"
|
|
'';
|
|
|
|
systemd.user.services.virtualboxClientVmsvga = mkVirtualBoxUserService "--vmsvga-session";
|
|
}
|
|
(lib.mkIf cfg.clipboard {
|
|
systemd.user.services.virtualboxClientClipboard = mkVirtualBoxUserService "--clipboard";
|
|
})
|
|
(lib.mkIf cfg.seamless {
|
|
systemd.user.services.virtualboxClientSeamless = mkVirtualBoxUserService "--seamless";
|
|
})
|
|
(lib.mkIf cfg.dragAndDrop {
|
|
systemd.user.services.virtualboxClientDragAndDrop = mkVirtualBoxUserService "--draganddrop";
|
|
})
|
|
]
|
|
);
|
|
}
|