QEMU/KVM setup with GPU passthrough for Windows on an Arch Linux host.
The script should make it comfortable and easy to launch a VM with QEMU/KVM with common and configurable parameters.
If you want to use your GPU inside the VM with PCI passthrough, it cares about driver loading/unloading with no reboot required, so if you are using the GPU on the host OS, you only have to save your work and stop the xserver and when you shutdown the VM, you can continue using the GPU on the host by restarting the xserver. If you have a secondary GPU, you can additionally start another xserver for the host OS while the VM is running.
If you take the cost of partitioning your hard disks, it has the advantage that you can use one single OS installation to natively boot into it (dual-boot) with full performance and use that installation also for the VM. For the VM, it additionally results in having much better IO performance compared with having large file system images on your host linux partition.
Mainboard | ASRock H87 Pro4 |
CPU | Intel(R) Core(TM) i5-4570 |
GPU | GeForce GTX 760 |
OS | Arch Linux |
In the BIOS settings of the mainboard, enable VT-d
and set the primary GPU to Onboard
.
Install QEMU and OVMF UEFI firmware files:
# pacman -S qemu ovmf
Add intel_iommu=on
to the linux command line. If your bootloader is GRUB, edit /etc/default/grub
and add it to GRUB_CMDLINE_LINUX
and regenerate grub.cfg
:
# grub-mkconfig -o /boot/grub/grub.cfg
The configuration uses the high performant virtio devices, that require to install the windows guest drivers from here.
The script loads some configuration parameters from the config
file.
The DRIVES
parameter contains the paths of file-systems that should be available inside the VM. It can contain file-system images or physical drives, e.g.:
DRIVES="/dev/sda \
/path/to/filesystem.img"
Mouse and keyboard are passed via evdev
, which allows you to easily switch between host and guest by pressing both control keys. The INPUTS
parameter contains the paths of input devices, e.g.:
INPUTS="/dev/input/by-id/usb-Logitech_Gaming_Mouse_G402_6D91317A5254-event-mouse \
/dev/input/by-id/usb-Logitech_G413_Carbon_Mechanical_Gaming_Keyboard_138736523537-event-kbd,grab_all=on,repeat=on"
A SMB share is accessible from within the VM via \\10.0.2.4\qemu
and allows access to the path defined in SHARE
on your host OS.
The default mode uses the virtual GPU qxl
on the guest. The VM is accessible via the QEMU window, can be maximized and scaled to fit the screen size and performs great for non intensive rendering tasks, as showing desktop and some UI applications.
Since it is the default, run the VM with:
$ sudo ./start-windows-vm.sh
The primary GPU is passed to the guest OS and gives native rendering performance. If the GPU is used by the host, e.g. by a running xserver, the session must be closed before using any of the following modes and can be restarted when the VM is shut down.
If you have a second GPU (including Intel Integrated Graphics), you can restart the xserver with an alternate xorg.conf.
To use Intel Integrated Graphics, it could look like this:
Section "Device"
Identifier "intel"
Driver "intel"
Option "TearFree" "true"
EndSection
Section "Screen"
Identifier "intel"
Device "intel"
EndSection
Save the config file as /etc/X11/xorg.intel.conf
and start a new xserver:
$ startx -- -config xorg.intel.conf
Then run the VM:
$ sudo VGA=passthrough ./start-windows-vm.sh
This option uses Looking-Glass, that allows to view the GPU output of the VM via a shared framebuffer with the looking-glass client inside a window on the host OS. It might be useful, if you have only a single monitor. You need to install the IVSHMEM driver, that is also part of the virtio drivers, and run the looking-glass server on the windows guest and afterwards the looking-glass-client on the host.
Run the VM:
$ sudo VGA=passthrough-lg ./start-windows-vm.sh
$ looking-glass-client -s
Here, the single GPU is passed to the guest VM which leaves the host OS with no display output. It requires to dump the VBIOS of your GPU and patch it according to this guide. Place it as VBIOS.rom
in the directory of this script.
Run the VM:
$ sudo VGA=passthrough-single ./start-windows-vm.sh