2013-10-31 13:26:06 +01:00
|
|
|
# Systemd services for libvirtd.
|
2011-02-25 16:07:52 +01:00
|
|
|
|
2014-04-14 16:26:48 +02:00
|
|
|
{ config, lib, pkgs, ... }:
|
2011-02-25 16:07:52 +01:00
|
|
|
|
2014-04-14 16:26:48 +02:00
|
|
|
with lib;
|
2011-02-25 16:07:52 +01:00
|
|
|
|
2011-09-14 20:20:50 +02:00
|
|
|
let
|
2011-02-25 16:07:52 +01:00
|
|
|
|
2011-09-14 20:20:50 +02:00
|
|
|
cfg = config.virtualisation.libvirtd;
|
2014-06-12 15:16:38 +02:00
|
|
|
vswitch = config.virtualisation.vswitch;
|
2013-08-15 21:50:16 +02:00
|
|
|
configFile = pkgs.writeText "libvirtd.conf" ''
|
|
|
|
unix_sock_group = "libvirtd"
|
|
|
|
unix_sock_rw_perms = "0770"
|
|
|
|
auth_unix_ro = "none"
|
|
|
|
auth_unix_rw = "none"
|
|
|
|
${cfg.extraConfig}
|
|
|
|
'';
|
2011-02-25 16:07:52 +01:00
|
|
|
|
|
|
|
in
|
|
|
|
|
|
|
|
{
|
|
|
|
###### interface
|
|
|
|
|
|
|
|
options = {
|
|
|
|
|
2011-09-14 20:20:50 +02:00
|
|
|
virtualisation.libvirtd.enable =
|
2011-02-25 16:07:52 +01:00
|
|
|
mkOption {
|
2014-04-07 21:31:29 +02:00
|
|
|
type = types.bool;
|
2011-02-25 16:07:52 +01:00
|
|
|
default = false;
|
|
|
|
description =
|
|
|
|
''
|
|
|
|
This option enables libvirtd, a daemon that manages
|
2013-08-16 21:25:00 +02:00
|
|
|
virtual machines. Users in the "libvirtd" group can interact with
|
|
|
|
the daemon (e.g. to start or stop VMs) using the
|
2011-02-25 16:07:52 +01:00
|
|
|
<command>virsh</command> command line tool, among others.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
2011-09-14 20:20:50 +02:00
|
|
|
virtualisation.libvirtd.enableKVM =
|
2011-02-25 16:07:52 +01:00
|
|
|
mkOption {
|
2014-04-07 21:31:29 +02:00
|
|
|
type = types.bool;
|
2011-02-25 16:07:52 +01:00
|
|
|
default = true;
|
|
|
|
description =
|
|
|
|
''
|
|
|
|
This option enables support for QEMU/KVM in libvirtd.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
2013-08-15 21:50:16 +02:00
|
|
|
virtualisation.libvirtd.extraConfig =
|
|
|
|
mkOption {
|
2014-04-07 21:31:29 +02:00
|
|
|
type = types.lines;
|
2013-08-15 21:50:16 +02:00
|
|
|
default = "";
|
|
|
|
description =
|
|
|
|
''
|
|
|
|
Extra contents appended to the libvirtd configuration file,
|
|
|
|
libvirtd.conf.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
2015-05-20 09:51:42 +02:00
|
|
|
virtualisation.libvirtd.extraOptions =
|
|
|
|
mkOption {
|
|
|
|
type = types.listOf types.str;
|
2015-05-20 10:06:21 +02:00
|
|
|
default = [ ];
|
2015-05-20 09:51:42 +02:00
|
|
|
example = [ "--verbose" ];
|
|
|
|
description =
|
|
|
|
''
|
|
|
|
Extra command line arguments passed to libvirtd on startup.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
2014-06-16 16:45:08 +02:00
|
|
|
virtualisation.libvirtd.onShutdown =
|
|
|
|
mkOption {
|
|
|
|
type = types.enum ["shutdown" "suspend" ];
|
|
|
|
default = "suspend";
|
|
|
|
description =
|
|
|
|
''
|
|
|
|
When shutting down / restarting the host what method should
|
|
|
|
be used to gracefully halt the guests. Setting to "shutdown"
|
|
|
|
will cause an ACPI shutdown of each guest. "suspend" will
|
|
|
|
attempt to save the state of the guests ready to restore on boot.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2011-02-25 16:07:52 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
###### implementation
|
|
|
|
|
|
|
|
config = mkIf cfg.enable {
|
|
|
|
|
2011-09-14 20:20:50 +02:00
|
|
|
environment.systemPackages =
|
2013-10-17 14:17:29 +02:00
|
|
|
[ pkgs.libvirt pkgs.netcat-openbsd ]
|
2011-03-16 13:31:06 +01:00
|
|
|
++ optional cfg.enableKVM pkgs.qemu_kvm;
|
2011-02-25 16:07:52 +01:00
|
|
|
|
2011-03-15 11:52:44 +01:00
|
|
|
boot.kernelModules = [ "tun" ];
|
|
|
|
|
2013-01-16 12:33:18 +01:00
|
|
|
systemd.services.libvirtd =
|
2012-10-26 16:21:35 +02:00
|
|
|
{ description = "Libvirt Virtual Machine Management Daemon";
|
2011-02-25 16:07:52 +01:00
|
|
|
|
2012-10-26 16:21:35 +02:00
|
|
|
wantedBy = [ "multi-user.target" ];
|
2014-06-16 16:45:08 +02:00
|
|
|
after = [ "systemd-udev-settle.service" ]
|
|
|
|
++ optional vswitch.enable "vswitchd.service";
|
2011-02-25 16:07:52 +01:00
|
|
|
|
2014-06-12 15:16:38 +02:00
|
|
|
path = [
|
2015-02-20 21:43:09 +01:00
|
|
|
pkgs.bridge-utils
|
2014-06-12 15:16:38 +02:00
|
|
|
pkgs.dmidecode
|
|
|
|
pkgs.dnsmasq
|
2011-03-31 17:24:13 +02:00
|
|
|
pkgs.ebtables
|
2014-06-12 15:16:38 +02:00
|
|
|
]
|
|
|
|
++ optional cfg.enableKVM pkgs.qemu_kvm
|
|
|
|
++ optional vswitch.enable vswitch.package;
|
2011-02-25 16:07:52 +01:00
|
|
|
|
2011-09-14 20:20:50 +02:00
|
|
|
preStart =
|
2011-03-15 10:42:49 +01:00
|
|
|
''
|
|
|
|
mkdir -p /var/log/libvirt/qemu -m 755
|
2011-03-16 14:52:52 +01:00
|
|
|
rm -f /var/run/libvirtd.pid
|
2011-04-01 00:10:26 +02:00
|
|
|
|
2013-11-09 16:31:10 +01:00
|
|
|
mkdir -p /var/lib/libvirt
|
|
|
|
mkdir -p /var/lib/libvirt/dnsmasq
|
|
|
|
|
|
|
|
chmod 755 /var/lib/libvirt
|
|
|
|
chmod 755 /var/lib/libvirt/dnsmasq
|
2011-04-01 00:10:26 +02:00
|
|
|
|
|
|
|
# Libvirt unfortunately writes mutable state (such as
|
|
|
|
# runtime changes to VM, network or filter configurations)
|
|
|
|
# to /etc. So we can't use environment.etc to make the
|
|
|
|
# default network and filter definitions available, since
|
|
|
|
# libvirt will then modify the originals in the Nix store.
|
|
|
|
# So here we copy them instead. Ugly.
|
|
|
|
for i in $(cd ${pkgs.libvirt}/etc && echo \
|
|
|
|
libvirt/qemu/networks/*.xml libvirt/qemu/networks/autostart/*.xml \
|
|
|
|
libvirt/nwfilter/*.xml );
|
|
|
|
do
|
|
|
|
mkdir -p /etc/$(dirname $i) -m 755
|
|
|
|
cp -fpd ${pkgs.libvirt}/etc/$i /etc/$i
|
|
|
|
done
|
2013-11-27 21:31:15 +01:00
|
|
|
|
|
|
|
# libvirtd puts the full path of the emulator binary in the machine
|
|
|
|
# config file. But this path can unfortunately be garbage collected
|
|
|
|
# while still being used by the virtual machine. So update the
|
|
|
|
# emulator path on each startup to something valid (re-scan $PATH).
|
2014-09-15 23:03:20 +02:00
|
|
|
for file in /etc/libvirt/qemu/*.xml /etc/libvirt/lxc/*.xml; do
|
2013-12-09 19:41:44 +01:00
|
|
|
test -f "$file" || continue
|
2013-11-27 21:31:15 +01:00
|
|
|
# get (old) emulator path from config file
|
|
|
|
emulator=$(grep "^[[:space:]]*<emulator>" "$file" | sed 's,^[[:space:]]*<emulator>\(.*\)</emulator>.*,\1,')
|
|
|
|
# get a (definitely) working emulator path by re-scanning $PATH
|
2014-09-15 23:03:20 +02:00
|
|
|
new_emulator=$(PATH=${pkgs.libvirt}/libexec:$PATH command -v $(basename "$emulator"))
|
2013-11-27 21:31:15 +01:00
|
|
|
# write back
|
|
|
|
sed -i "s,^[[:space:]]*<emulator>.*, <emulator>$new_emulator</emulator> <!-- WARNING: emulator dirname is auto-updated by the nixos libvirtd module -->," "$file"
|
|
|
|
done
|
2011-04-01 00:10:26 +02:00
|
|
|
''; # */
|
2011-03-15 10:42:49 +01:00
|
|
|
|
2015-05-20 09:51:42 +02:00
|
|
|
serviceConfig.ExecStart = ''@${pkgs.libvirt}/sbin/libvirtd libvirtd --config "${configFile}" --daemon ${concatStringsSep " " cfg.extraOptions}'';
|
2012-10-26 16:21:35 +02:00
|
|
|
serviceConfig.Type = "forking";
|
|
|
|
serviceConfig.KillMode = "process"; # when stopping, leave the VMs alone
|
2011-02-25 16:07:52 +01:00
|
|
|
|
2011-04-01 20:08:53 +02:00
|
|
|
# Wait until libvirtd is ready to accept requests.
|
|
|
|
postStart =
|
|
|
|
''
|
|
|
|
for ((i = 0; i < 60; i++)); do
|
|
|
|
if ${pkgs.libvirt}/bin/virsh list > /dev/null; then exit 0; fi
|
|
|
|
sleep 1
|
|
|
|
done
|
|
|
|
exit 1 # !!! seems to be ignored
|
|
|
|
'';
|
2011-02-25 16:07:52 +01:00
|
|
|
};
|
|
|
|
|
2016-01-06 07:50:18 +01:00
|
|
|
systemd.services."libvirt-guests" = {
|
|
|
|
description = "Libvirt Virtual Machines";
|
2011-02-25 16:56:49 +01:00
|
|
|
|
2016-01-06 07:50:18 +01:00
|
|
|
wantedBy = [ "multi-user.target" ];
|
|
|
|
wants = [ "libvirtd.service" ];
|
|
|
|
after = [ "libvirtd.service" ];
|
2011-02-25 16:56:49 +01:00
|
|
|
|
2016-01-06 07:50:18 +01:00
|
|
|
restartIfChanged = false;
|
2011-02-25 16:56:49 +01:00
|
|
|
|
2016-01-06 07:50:18 +01:00
|
|
|
path = with pkgs; [ gettext libvirt gawk ];
|
2011-02-25 16:56:49 +01:00
|
|
|
|
2016-01-06 07:50:18 +01:00
|
|
|
preStart = ''
|
|
|
|
mkdir -p /var/lock/subsys -m 755
|
|
|
|
${pkgs.libvirt}/etc/rc.d/init.d/libvirt-guests start || true
|
|
|
|
'';
|
2011-02-25 16:56:49 +01:00
|
|
|
|
2016-01-06 07:50:18 +01:00
|
|
|
postStop = ''
|
|
|
|
export PATH=${pkgs.gettext}/bin:$PATH
|
|
|
|
export ON_SHUTDOWN=${cfg.onShutdown}
|
|
|
|
${pkgs.libvirt}/etc/rc.d/init.d/libvirt-guests stop
|
|
|
|
'';
|
2011-04-01 20:08:53 +02:00
|
|
|
|
2016-01-06 07:50:18 +01:00
|
|
|
serviceConfig = {
|
|
|
|
Type = "oneshot";
|
|
|
|
RemainAfterExit = true;
|
2012-04-19 17:12:55 +02:00
|
|
|
};
|
2016-01-06 07:50:18 +01:00
|
|
|
};
|
2012-04-19 17:12:55 +02:00
|
|
|
|
2013-08-16 00:47:21 +02:00
|
|
|
users.extraGroups.libvirtd.gid = config.ids.gids.libvirtd;
|
2013-08-15 21:50:16 +02:00
|
|
|
|
2011-02-25 16:07:52 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
}
|