...or "How I learned to shut my x270 laptop and not worry about my VMs."
This is an implementation of vmmci(4) for
Linux using a customized version of the virtio_pci
driver from the
mainline kernel. It currently supports the following:
-
Clean Shutdowns on Request When requested by
vmctl(8)
...you can safely usevmctl stop <you linux guest>
and it'll nicely stop services and sync disks! -
System Time Synchronization When the host
vmd(8)
emulation of the hardware clock detects a clock drift (most likely due to the host being suspended/resumed), it fires aSYNCRTC
message that the Linuxvmmci
driver responds to by synchronizing system time to the hardware clock time. -
Tracking Clock Drift At regular intervals (currently 20s),
vmmci
will measure current clock drift, recording the current drift amount in seconds and nanoseconds parts readable viasysctl vmmci
This Linux VMMCI currently comes in two parts:
virtio_pci_obsd.ko
-- handles the quirks of getting Linux's virtio pci framework to properly work with the VMM Control Interface device fromvmd(8)
virtio_vmmci.ko
-- virtio device driver that replicates the behavior of OpenBSD'svmmci(4)
driver
You need both modules installed!
A few things:
-
I test and develop using OpenBSD snapshots, so relatively in sync with -current. (The good news is when OpenBSD 6.5 drops, things should be in good working order.)
-
I lean heavily on the simplification that OpenBSD virtualization guests are single CPU currently.
-
This won't solve larger clock issues...like not getting your kernel to trust the
tsc
clocksource. (Make sure you boot your system withclocksource=tsc
and possiblytsc=reliable
.) -
I primarily try to support Linux v4.x with intentions of supporting v5.x once distros start to pick it up. If you're on an older distro that uses v3.x or older...you're on your own!
Lastly, if you're seeing regular clock drifts over 2-3 seconds you need to check your clocksource is tsc in:
/sys/devices/system/clocksource/clocksource0/current_clocksource
Drifts in the 0-2 second range aren't that abnormal from my experience and shouldn't cause concern, but a monotonic increase in drift getting past that range should make you investigate your guest config!
Assuming you've got a recent Linux distro running as a guest already under OpenBSD, it shouldn't be more than a few minutes to get things up and running.
However, keep in mind my testing has been mostly on Ubuntu 18.04 with it's stock 4.15.0 Linux kernel as well as my own 4.20.12 customized kernel. Chances are if the module won't compile on a 4.x kernel, it's a simple matter of navigating the moving target that is virtio in the 4.x Linux kernel versions. Feel free to either submit a PR or just open an issue letting me know your kernel version and distro your using.
Install the tools required to build kernel modules using your package manager or whatever you normally use to install stuff. For Ubuntu/Debian-like systems, you can try the following:
$ sudo apt install build-essential libelf-dev linux-headers-$(uname -r)
Basically you need your kernel headers (you do NOT need a full source tree) and some GCC tooling.
This should be easy and expose issues with your lack of prerequisites or an incompatability with your kernel version:
$ make
If that worked and you see no warnings, you should have both a
virtio_pci_obsd.ko
file and a virtio_vmmci.ko
file.
A common source of compiler warnings are from variations in kernel versions. Please share your kernel version and the compiler output if you have issues!
You can either use make insmod
or manually load the modules:
$ sudo insmod virtio_pci_obsd.ko
$ sudo insmod virtio_vmmci.ko
If you want to install the module permanently and have it auto-load at boot, for now you'll have to do that manually. Maybe check out this askubuntu post for guidance: https://askubuntu.com/a/307375
After you load virtio_pci_obsd.ko
you should see your system match
and enable the vmmci PCI device. Check dmesg(1)
and you should see
something like:
[ 825.819945] virtio_pci_obsd: loading out-of-tree module taints kernel.
[ 825.819945] virtio_pci_obsd: module verification failed: signature and/or required key missing - tainting kernel
[ 825.819945] virtio-pci-obsd 0000:00:05.0: runtime IRQ mapping not provided by arch
[ 825.819945] virtio_pci_obsd_match: matching 0x0777
[ 825.819945] virtio_pci_obsd_match: found OpenBSD device
[ 825.819945] virtio-pci-obsd 0000:00:05.0: enabling bus mastering
If you check with lspci(8)
in verbose mode (lspci -v
) you should
see the device and the fact it's using our virti_pci_obsd
driver:
00:05.0 Communication controller: Device 0b5d:0777
Subsystem: Device 0b5d:ffff
Flags: bus master, fast devsel, latency 0, IRQ 9
I/O ports at 5000 [size=4K]
Kernel driver in use: virtio-pci-obsd
When you load virtio_vmmci.ko
, you should see a confirmation the
module is loaded:
[ 256.030878] virtio_vmmci: started VMM Control Interface driver
You can enable debug mode either by passing a debug=1
argument when
loading the virtio_vmmci.ko
module or toggle it afterwards by
writing either a 0
(off) or 1
/any positive integer (on) to
/sys/modules/virtio_vmmci/parameters/debug
as the root user. When
debug mode is on, you'll get extra dmesg noise like:
[17769.012388] virtio_vmmci: [vmmci_validate] not implemented
[17769.012388] virtio_vmmci: [vmmci_probe] initializing vmmci device
[17769.012388] virtio_vmmci: [vmmci_probe] ...found feature TIMESYNC
[17769.012388] virtio_vmmci: [vmmci_probe] ...found feature ACK
[17769.012388] virtio_vmmci: [vmmci_probe] ...found feature SYNCRTC
[17769.012388] virtio_vmmci: started VMM Control Interface driver
[17769.034540] virtio_vmmci: [clock_work_func] starting clock synchronization
[17769.034864] virtio_vmmci: [clock_work_func] guest clock: 1550959642.629898000, host clock: 1550959642.638556712
[17769.034867] virtio_vmmci: [clock_work_func] current time delta: -1.991341288
[17769.034870] virtio_vmmci: [clock_work_func] clock synchronization routine finished
Lastly, check the sysctl tables. The driver registers 2 particular values that contain the seconds and nanoseconds portion of the last measured drift amount:
you@guest:~/virtio_vmmci$ sudo sysctl vmmci
vmmci.drift_nsec = 199647574
vmmci.drift_sec = 1
In the above example, the total drift is 1.199647574 seconds
.
In the future I may expose the last measured time as well
You can easily test the clock synchronization by suspending your
OpenBSD host by triggering zzz
manually or by something like closing
your laptop lid. Wait at least 10 seconds or so and resume your
OpenBSD system. In the Linux guest, your dmesg(1)
output will tell
you (in less than 30 seconds) that it's detected a clock drift and
it's sync'ing the clock:
[15670.027879] virtio_vmmci: [clock_work_func] current time delta: 91.482370612
[15670.027879] virtio_vmmci: detected drift greater than 5 seconds, synchronizing clock
[15670.027879] virtio_vmmci: [clock_work_func] clock synchronization routine finished
If you check date
or timedatectl
on the Linux guest you should see
the system time is very close to our host time.
How can we test a clean shutdown? It's not too hard, but it might not work the same between distros and versions. Here's what I've done on Ubuntu 18.04.
Assuming your vm is up and running:
- Use
tmux(1)
or another means of getting 2 terminal sessions going at once. - In one session,
vmctl console <vm name or id>
to connect to the VM over the serial console. (This obviously assumes your guest is configured to work that way.) - In another session, issue
vmctl stop <vm name or id>
. - Back in the serial console session, you should see your init
system...probably
systemd
...start running through the shutdown process.
There may be some variations. The Linux vmmci driver calls a kernel helper function that handles orchestrating the shutdown via userspace. (The question of how to shutdown a Linux system from kernelspace is quite fascinating to explore.)
Some questions people...mainly myself...have had...
This isn't using the userland settimeofday(2)
system call and
instead using a particular kernel function (do_settimeofday64
[1])
that appears to be pretty analagous to OpenBSD's kernel's
tc_setclock
function [2] in that it steps the system clock while
triggering any alarms or timeouts that would fire.
Looking at how VirtualBox handles this with their userland guest
additions services, they look for large clock drifts where "large"
is currently > 30 minutes. If it's large, it just uses
settimeofday(2)
. Otherwise, it tries to use something like
adjtimex(2)
to accelerate the clock up to the correct time. (This is
something I may consider for vmmci after some more usage/testing.)
See their source for VBoxServiceTimeSync.cpp
[3].
Maybe for small clock disturbances/drifts, but it's not ideal for major stepping and only solves the clock problem.
There are two reasons I'd consider using virtio_vmmci
either in
addition to or in place of relying on an NTP daemon:
-
Not every guest has network access. This precludes NTP as an option. Even if the guest has limited network access, it still needs access to an NTP server, ideally multiple. This isn't always the case.
-
Large clock drifts like when you suspend your laptop for an evening make most NTP daemons sad. I've never seen an NTP daemon that is cool with just jumping the system time ahead (i.e. stepping) like that. Some require special config to even do. Yes,
ntpd(8)
supports a-s
flag to do an actual set of the time and not just an adjustment, but even as the man page says it's for startup. (Useful for embedded, clock-less systems like a Raspberry Pi.)
A lot of modern Linux distros install and enable an NTP daemon by default these days. That's fine. But don't forget vmmci gives you clean shutdowns as well as properly stepping the clock after a long suspend/hibernation!
Few reasons, but for more background see my email to misc@openbsd.org: https://marc.info/?t=155102953000002
In short:
- OpenBSD purposely uses self-asigned PCI and Virtio device identifiers to "hide" the VMM Control Interface device
- Linux's virtio pci code is a LOT more complex and is trying to handle a variety of virtio devices...but can't handle a particular quirk with how the VMM Control Interface deals with config register i/o.
Write a bloody man page...
-
Thanks to the OpenBSD
vmm(4)
/vmd(8)
hackers...especially those that put together OpenBSD'svmmci(4)
driver which acted as my reference point. -
The bootlin cross-referencer because holy hell is that thing 10x more useful than poking around Torvald's mirror of the official Linux Git repo.
-
This page from "The kernel development community" was very helpful in figuring out how to schedule "deferred work" in the kernel: https://linux-kernel-labs.github.io/master/labs/deferred_work.html
-
The
virtio_balloon.c
driver in the Linux kernel tree is a relatively simple virtio example to understand Linux virtio drivers. -
Folks that have helped test on different distros with different kernel versions :-)
(GitHub might not render these...but believe me they're here :-) )
[1] https://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git/tree/kernel/time/timekeeping.c?h=v4.20#n1220 [2] https://github.com/openbsd/src/blob/e12a049bd4bbd1e8315c373a739e08972ed6dd1d/sys/kern/kern_tc.c#L382 [3] https://www.virtualbox.org/browser/vbox/trunk/src/VBox/Additions/common/VBoxService/VBoxServiceTimeSync.cpp?rev=76553#L683