Skip to content

Latest commit

 

History

History
971 lines (708 loc) · 54.5 KB

README.md

File metadata and controls

971 lines (708 loc) · 54.5 KB

pop!_OS-win10-KVM-setup

THIS IS NOW A COMPLETED GUIDE. I will be providing periodic changes, updates, and fixes to this repo as I see fit.

I would like to provide thanks to Noah, Isaiah, and Joey for taking the time out of their day to gloss over this document and aid me in the creation of this repo.

Nov-5-21 This still works on Pop 21.04.

LAST UPDATE: 7/17/23 - Grammar / word changes

I would also like to personally thank every person who has viewed and used this guide in any capacity. I never thought something I made would help this much. Remember to also look at the guides that my own guide was inspired from below!

Table of Contents:

PREFACE:

This is a repository that contains a tutorial and the necessary scripts to create a working Pop!_OS 20.10 x86_64 -> Windows 10 KVM.

You don't need strong terminal skills to do this, but it is highly recommended that you know basic terminal commands to get yourself through this. This is especially true if you are not using user-friendly desktop environments or plan to do this entirely through the Terminal. If you don't have good terminal skills I suggest this article.

I have moderate terminal knowledge, so this was a breeze for me, but I understand new users who want to get into the KVM environment after switching to Linux for the myriad of reasons that they may have(I don't blame you for moving. Welcome to Linux!).

If you are a seasoned UNIX/Linux/related user, this may not be the guide for you, as this will be a more verbose guide. If you are a more experienced user, I suggest going to the Arch KVM wiki or alternatively going to the guides listed below.

^If you want this information, install neofetch:

$ sudo apt install neofetch
$ neofetch

Introduction:

This KVM for Windows 10 will allow for PCIE and SATA devices to be passed through and used while having minimal performance loss that is directly comparable to bare metal performance.

Reasoning/Consideration:

My reasoning for creating this was that, at the time, I was a newer Linux user that was also a gamer, and I did not want my moving over to a better development environment to impact my ability to game. I also did not want to necessarily deal with the annoying dual booting that I had. It is just easier to pull my hair out for a few days and learn to set this up than dual booting.

I spent nearly a week sifting through various guides getting this environment to work (properly) on my desktop, and I want to share the tips and tricks I've learned along the way and present how to establish a KVM / Virt-Manager setup (for Windows 10) for inexperienced users in an easily digestible way. Some guides would get me halfway and then would not provide me with enough information on where to go next, others summed up the first but left delicate details out. All the guides I have read however are great and I want to give recognition to the individuals that helped me get this environment to work.

Guides:

Please consider checking out these guides to find out where I was able to get my information from - I highly recommend it, and the information is great.

This guide is designed to pull information from both sub-guides. All other guides will be referenced appropriately.

Bryan's Guide:

Bryan Steiner (bryansteiner)

Bryan's Guide

Aarons's Guide:

Aaron Anderson (aaronanderson)

Aaron's Guide

Below is a breakdown of my exact PC hardware setup. Please be aware that there are differences between AMD and Intel builds regarding BIOS options and selecting the correct options.

I am on an AMD/NVIDIA build but I will try to help Intel users as well, but your mileage may vary.

Hardware Setup:

Hardware Requirements:

  • Two graphics cards. (One for the host system, One for the guest system (passthrough))
  • Hardware that can support IOMMU (Please read before proceeding).
  • A monitor with more than one input or more than one monitor (Will be discussed later).

Tips/Tricks:

  1. If you are tired of having to enter a password for each sudo you do, do the following:
$ sudo -i

This will sign you into root.

  1. Remember tab completion! It will make your life a lot easier when going back and forth between directories. Simply type the beginning of your command and hit tab, the terminal will auto fill in the rest.

  2. It is a confusing experience but I am sure you can get through it with some perseverance. Keep at it.

  3. I suggest installing tree to see how directories are laid out in the terminal (useful later!):

$ sudo apt install tree

You can then run it simply by:

$ tree

It will produce output like this:

  1. Basic Terminal commands. clear, ls, cd, mkdir, cp, etc.

Tutorial

Prerequisites

Before anything, it is required to install these packages and download these files.

  1. Update your OS if it is out of date:
$ sudo apt-get update
$ sudo apt-get upgrade
$ sudo apt-get dist-upgrade
  1. (MANDATORY) Download virtIO ISO files Since this project is a KVM for Windows 10, you are required to download and use virtIO drivers. virtIO is a virtualization standard for network/disk device drivers. The addition of virtIO can be done by attaching the ISO to the windows VM in the application Virt-Manager (we will get this later). Get the virtIO drivers here

  2. (MANDATORY) Download Windows 10 ISO files Since we are going to be creating a Windows kvm, you need the ISO for it. Get the latest Windows 10 ISO here. Without the ISO you will not be able to boot the VM into Windows.

  3. (OPTIONAL): Preparing an AMD GPU for Host Machine

If you own an AMD graphics card as host:

You don't need the Vulkan Drivers but if you want the best performance you can get on the host, and if it uses an AMD GPU I highly suggest this. Ubuntu comes with mesa-vulkan-drivers which offer comparable or better performance but you can get the AMD Vulkan drivers here:

$ sudo wget -qO - http://repo.radeon.com/amdvlk/apt/debian/amdvlk.gpg.key | sudo apt-key add -
$ sudo sh -c 'echo deb [arch=amd64] http://repo.radeon.com/amdvlk/apt/debian/ bionic main > /etc/apt/sources.list.d/amdvlk.list'
$ sudo apt update
$ sudo apt-get install amdvlk
  1. (MANDATORY) Install the following packages
$ sudo apt install libvirt-daemon-system libvirt-clients qemu-kvm qemu-utils virt-manager ovmf

After this is completed, you are going to restart your PC and enter your BIOS. BIOS entry varies by manufacturer. In my case, I can access it with F2, F8, or DEL. Enable the feature called IOMMU. Once this is completed you will then need to enable CPU virtualization. For Intel processors, you will need to enable VT-d. For AMD, look for something called AMD-Vi or in the case of my motherboard SVM/SVM MODE and enable it. Save your changes to the BIOS and then go back into pop!

Once you are logged back in you are going to want to run this command:

AMD:

$ dmesg | grep AMD-Vi

Intel:

$ dmesg | grep VT-d

If you get this error:

dmesg: read kernel buffer failed: Operation not permitted

Run it as sudo.

$ sudo dmesg | grep AMD-Vi

If you get output that looks like this, you should be ready.

Once this is completed you will need to pass this hardware-enabled IOMMU functionality into the kernel. You can read more about kernel parameters here. Depending on your boot-loader you will have to figure out how to do this yourself. For me, I can use kernelstub. Other people use grub, GRUB2, or rEFInd.

AMD:

$ sudo kernelstub --add-options "amd_iommu=on"

Intel:

$ sudo kernelstub --add-options "intel_iommu=on"

If you use GRUB2, you can do it by going into /etc/default/grub with sudo permissions and include it in the kernel parameter below.

AMD:

GRUB_CMDLINE_LINUX_DEFAULT="quiet splash amd_iommu=on"

Intel:

GRUB_CMDLINE_LINUX_DEFAULT="quiet splash intel_iommu=on"

As mentioned in Bryan's guide, when planning the GPU passthrough setup, it was said to blacklist the NVIDIA/AMD drivers. {I don't think this has been mentioned yet in your own guide though} "The logic stems from the fact that since the native drivers can't attach to the GPU at boot-time, the GPU will be freed-up and available to bind to the vfio drivers instead." The tutorials will make you add a parameter called pci-stub with the PCI bus ID of the GPU you wish to use. I did not follow this approach and instead dynamically unbind the drivers and bind VFIO-PCI drivers to it. Alternatively, you can run this script to bind the VFIO-PCI drivers to the secondary card in your PC. But it is important to understand IOMMU groupings. The script provided by Bryan here is perfectly adequate for finding IOMMU groups, but I found this script to be better.

The reason we want to use either script is to find the devices we want to pass through (storage drivers, PCIe hardware, etc). IOMMU is a reference to the chipset device that maps virtual addresses to physical addresses on the input/output of the devices. At the end of this step, we want to make sure that we have appropriate IOMMU groupings. The reason for this is that you cannot separate the groupings.

Run the script:

./iommu2.sh

If you cannot run the script, with or without sudo, then you should run:

chmod +x ./iommu2.sh

chmod elevates permissions for files, and in this case, would allow you to run this without complications.

For AMD systems the output will look something like this:

Pulled from Bryan's guide, Intel output should look like.

...
IOMMU Group 1 00:01.0 PCI bridge [0604]: Intel Corporation Xeon E3-1200 v5/E3-1500 v5/6th Gen Core Processor PCIe Controller (x16) [8086:1901] (rev 07)
IOMMU Group 1 00:01.1 PCI bridge [0604]: Intel Corporation Xeon E3-1200 v5/E3-1500 v5/6th Gen Core Processor PCIe Controller (x8) [8086:1905] (rev 07)
IOMMU Group 1 01:00.0 VGA compatible controller [0300]: NVIDIA Corporation TU102 [GeForce RTX 2080 Ti Rev. A] [10de:1e07] (rev a1)
IOMMU Group 1 01:00.1 Audio device [0403]: NVIDIA Corporation TU102 High Definition Audio Controller [10de:10f7] (rev a1)
IOMMU Group 1 01:00.2 USB controller [0c03]: NVIDIA Corporation TU102 USB 3.1 Controller [10de:1ad6] (rev a1)
IOMMU Group 1 01:00.3 Serial bus controller [0c80]: NVIDIA Corporation TU102 UCSI Controller [10de:1ad7] (rev a1)
IOMMU Group 1 02:00.0 VGA compatible controller [0300]: Advanced Micro Devices, Inc. [AMD/ATI] Ellesmere [Radeon RX 470/480/570/570X/580/580X/590] [1002:67df] (rev e1)
IOMMU Group 1 02:00.1 Audio device [0403]: Advanced Micro Devices, Inc. [AMD/ATI] Ellesmere HDMI Audio [Radeon RX 470/480 / 570/580/590] [1002:aaf0]
...

If you have the problem presented in the Intel example, you have 2 options:

  1. You can try swapping which PCI slot the graphics cards are in. This may or may not provide the expected results.
  2. Alternatively you can conduct an ACS Override Patch. It's highly worth it to read this post from Alex Williamson. "Applying the ACS Override Patch may compromise system security. Check out this post to see why the ACS patch will probably never make its way upstream to the mainline kernel."

ACS Override Patch (Optional):

PLEASE go to Bryan's guide and read how to do it there and understand the complications and implications.

Since I did not need that part I will be skipping it. The next steps are applicable if you need the patch or not. Dynamic binding is not necessarily required. But it works in my case so I suggest looking into it. I will provide the instructions below.

OPTIONAL: VM Dynamic Binding

How: Libvirt has a hook (Libvirt Hooks) system that grants you access to running commands on startup or shutdown of the VM. The scripts that are located within the directory /etc/libvirt/hooks. If the directory cannot be found or does not exist, create it.

$ sudo mkdir /etc/libvirt/hooks

But lucky for us we have the hook helper which can be run as followed:

$ sudo wget 'https://raw.githubusercontent.com/PassthroughPOST/VFIO-Tools/master/libvirt_hooks/qemu' \-O /etc/libvirt/hooks/qemu
$ sudo chmod +x /etc/libvirt/hooks/qemu

Now restart the libvirt to use the hook helper:

$ sudo service libvirtd restart

If you want to dynamically bind the VFIO-PCI drivers before the VM starts and unbind after you end it, you can do as follows:

Recognize the most important hooks

# Before a VM is started, before resources are allocated:
/etc/libvirt/hooks/qemu.d/$vmname/prepare/begin

# Before a VM is started, after resources are allocated:
/etc/libvirt/hooks/qemu.d/$vmname/start/begin

# After a VM has started up:
/etc/libvirt/hooks/qemu.d/$vmname/started/begin

# After a VM has shut down, before releasing its resources:
/etc/libvirt/hooks/qemu.d/$vmname/stopped/end

# After a VM has shut down, after resources are released:
/etc/libvirt/hooks/qemu.d/$vmname/release/end

I have named my VM "pop" for this example. My directory structure is:

Now is when things get fun. Create a file named kvm.conf. My editor of choice is vim. If you want to avoid problems when creating these files:

$ sudo $editor_you_want_to_use kvm.conf

Once you have the file open add these entries:

## Virsh devices
VIRSH_GPU_VIDEO=pci_0000_0b_00_0
VIRSH_GPU_AUDIO=pci_0000_0b_00_1

These are how my groupings are made so you are *required to find your correct groupings and place them here. If you forget where to get them, use the iommu2.sh script by:

$ ./iommu2.sh

The output of the script will translate the address for each device. Ex: IOMMU group 27 0b:00.0 which is written as --> VIRSH_GPU_VIDEO=pci_0000_0b_00_0. You will need to figure this out on your own!

Once you got the current bus addresses you can then move on to create some scripts:

bind_vfio.sh:

#!/bin/bash

## Load the config file
source "/etc/libvirt/hooks/kvm.conf"

## Load vfio
modprobe vfio
modprobe vfio_iommu_type1
modprobe vfio_pci

## Unbind gpu from nvidia and bind to vfio
virsh nodedev-detach $VIRSH_GPU_VIDEO
virsh nodedev-detach $VIRSH_GPU_AUDIO

unbind_vfio.sh:

#!/bin/bash

## Load the config file
source "/etc/libvirt/hooks/kvm.conf"

## Unbind gpu from vfio and bind to nvidia
virsh nodedev-reattach $VIRSH_GPU_VIDEO
virsh nodedev-reattach $VIRSH_GPU_AUDIO

## Unload vfio
modprobe -r vfio_pci
modprobe -r vfio_iommu_type1
modprobe -r vfio

My script varies from Bryan's by removing groupings I do not have. It works for me. You may need to tweak it some.

Once the scripts are created, chmod them:

$ chmod +x bind_vfio.sh unbind_vfio.sh

If this doesn't work, just add sudo to the front.

It should look like the image above once completed. You should be all set in this case. Tweaks to the VM will be made later or you can skip the tweaks. I skipped some tweaks and my performance is just fine.

Take a breather! We are almost there!

Creating the VM in Virt-Manager

Once you are here we can begin the construction of our VM. If you are a new user I suggest the GUI approach that I will describe. Otherwise, you can take a read YuriAlek's series of GPU passthrough scripts. The scripting is obviously more involved and takes some skill to process. The GUI approach is easier. There is nothing wrong with this approach!

You can now start Virt-Manager, you will be presented with this screen:

Click on the screen with a yellow light icon or navigate to File > New Virtual Machine. You will be presented with this screen. Choose Local install media (ISO image or CDROM) and select Forward. You will then see:

Remember those ISOs we installed earlier? We are going to need them now. I store them on my Desktop/. Store them wherever you want and navigate to that installation location by selecting the browse button. Choose the Win10 ISO. The Choose the operating system you are installing: section should now autocomplete. Keep the button checked and continue Forward. You will be presented with:

You will now configure your Memory (RAM) and CPU settings. In my case, I will designate 16GB of ram for now (16384) and since I have a 12c/24t CPU, I will pass in all 12 cores. Proceed Forward and be met with:

In this case, you will be creating a custom storage for the Windows install. Select Enable storage for this virtual machine and then Select or create custom storage and then navigate to the Storage Volume menu and create a storage volume with any size above 50GB. In this case, you want to create a storage volume with any name you would like and with a format of qcow2 the rest doesn't matter. It can be stored wherever you like. Once created select it and proceed with Choose Volume and lastly go Forward.

NOTE: I had a problem when I initially created this because I named it win10. I highly suggest NOT naming it this. Instead do Windows10 if you really want. Lastly, select Customize configuration before install and then Finish.

A new window will now appear called $VM_NAME on QEMU/KVM. This will allow you to configure more advanced options. You can alter these options through GUI or libvirt XML settings. Please ensure that while on the Overview page under Firmware you select UEFI x86_64: /usr/share/OVMF_CODE_4M.fd and none of the other options.

Next up we will move to the CPUs tab. Here we are going to change under Configuration to make sure Copy host cpu configuration is NOT checked, change Model to host-passthrough, and select Enable available CPU security flaw mitigations. This is to prevent Spectre/Meltdown vulnerabilities. Don't bother with Topology yet.

Next up you can remove a few options from the side. Remove Tablet, Channel Spice, and Console.

Select Sata Disk 1 > Advanced options disk bus -> VirtIO

Navigate to NIC and change the Device model to virtio

Add Hardware, add Channel Device, keep the default name, and choose Device type - Unix Socket.

Add Hardware, Storage, Browse, Browse Local, choose virtio-win-0.1.185.iso.

Add Hardware, PCI host device, (in my case I added), PCI host device, 0b:00.0 NVIDIA Corporation Device, you will need to find your appropriate grouping with your VFIO-pci driver. This is my GPU visuals

Add Hardware, PCI host device, (in my case I added), PCI host device, 0b:00.1 NVIDIA Corporation Device, you will need to find your appropriate grouping with your VFIO-pci driver. This is my GPU audio.

Now if you want to pass in any USB Host Devices feel free to add whatever ones you want. I did this and changed it later to pass in the entire PCI device.

Now we are going to have to get our hands dirty with editing the XML file. Go to Virtual Machine Manager, select edit -> preferences -> General -> Enable XML editing and now you can navigate to the $VM_NAME on QEMU/KVM window and then select Overview -> XML.

If you are passing in an NVIDIA GPU to the VM you may run into Error 43. This is because NVIDIA doesn't enable virtualization on their GeForce cards. The workaround is to have the hypervisor hide its existence. From here navigate to the <hyperv> section and add this.

<hyperv>
<features>
    ...
    <hyperv>
        <relaxed state="on"/>
        <vapic state="on"/>
        <spinlocks state="on" retries="8191"/>
        <vendor_id state="on" value="kvm hyperv"/>
    </hyperv>
    ...
</features>

Next directly under the </hyperv> line add

<features>
    ...
    <hyperv>
        ...
    </hyperv>
    <kvm>
      <hidden state="on"/>
    </kvm>
    ...
</features>

If QEMU 4.0 is being used with a q35 chipset you will need to add this to the end of <features>.

<features>
    ...
    <ioapic driver="kvm"/>
</features>

Error 43 should no longer occur.

Installing Windows 10 in the VM

This section will detail how to install the Windows 10 VM and get ready to move forward with tweaks and updates to get your performance better. I will be pulling from Aaron's guide in this section.

START the VM. Click into the VM window and it should have your mouse and keyboard take over the window. Press enter when you are prompted to boot from the CDROM. If you are shown the UEFI shell then you weren't fast enough. If this is the case, type exit or go to boot manager -> UEFI QEMU CDROM QM03 (There may be more than 1 zero in yours).

Select 'I don't have a product key' I highly suggest not putting a Windows code in unless you are 100% completed with your installation and are fine with the performance. Reinstallation is not out of the ordinary.

On the installation screen for Windows, select Custom: Install Windows Only (advanced). This is because Windows does not have VirtIO drivers included. This will restrict your VM from seeing the drive. Click load driver, and browse for your drive. In my case, it was E: but for you, it may be different. It should say: E: virtio-win-0.1.173, once here select the folder amd64\w10.

Once this is done and the installation goes on, you can then try to shut down the VM before Windows auto-restarts. If you get stuck on a black screen due to an AMD GPU bug then force your VM off and put the host to sleep then wake up the host.

Adding a VBIOS to Your Guest (Windows 10) VM

Since we completed the Windows install we can do some more setup by sending the proper VBIOS for your guest. You don't have to do this but it's highly suggested to get the best performance. IF YOU NEED TO GET YOUR VBIOS navigate to -> https://www.techpowerup.com/vgabios/ and find your model and correct VBIOS. Once you have this you can download it to a location you will remember, and then:

$ sudo mkdir /etc/firmware
$ sudo cp EVGA.RTX3070.8192.201019.rom /etc/firmware/
$ sudo chown root:root /etc/firmware/EVGA.RTX3070.8192.201019.rom
$ sudo chmod 744 /etc/firmware/EVGA.RTX3070.8192.201019.rom

I use Vim so:

$ sudo vi /etc/apparmor.d/abstractions/libvirt-qemu

append this to the very end and the SPACES ARE IMPORTANT!!!

  /etc/firmware/* r,

Once written and you're ready to quit (on Vim) shift + : then type :wq!, this will write and quit. If you write :Wq! it will not work.

run:

$ sudo systemctl restart apparmor.service

Now navigate back to your virt-manager and find the PCI device that you added that your GPU is under. Click XML and edit the XML to include this:

<hostdev mode="subsystem" type="pci" managed="yes">
  <source>
    <address domain="0x0000" bus="0x0b" slot="0x00" function="0x0"/>
  </source>
  <rom bar="on" file="/etc/firmware/EVGA.RTX3070.8192.201019.rom"/>
  <address type="pci" domain="0x0000" bus="0x05" slot="0x00" function="0x0"/>
</hostdev>

The important bit is the <rom bar="on" file="/etc/firmware/EVGA.RTX3070.8192.201019.rom"/> This will be different depending on your IOMMU grouping, your graphics card, and your VBIOS so please keep an eye open and add the appropriate content.

IF YOUR GPU DOESN'T STICK WITH `vfio-pci` DRIVERS:

This is a problem that some may experience and I do not have the answer as to why it happens, but I have a remedy for it: Download this script and run it. This will take whatever the secondary GPU is and bind it with the required vfio-pci drivers. Once downloaded and run, reboot and check if you have your vfio-pci drivers by running the iommu2.sh file I provided. I found this script through the video: GPU passthrough guide for PopOS 20.04 by Pavol Elsig. >Pavol, if you are reading this, thank you for the great video. (This worked for me as of my current pop! version. YMMV).

Start the VM

Finish the Windows installation. Use the KVM and connect the second GPU to any monitor to ensure you're getting output. If you are that is good. Otherwise, you may have missed a step or did not configure correctly.

Open the Windows Search and type Device Manager. There will be missing drivers. I updated the missing drivers manually. Ignore the unknown device missing drivers. That is the QEMU bug.

Select the PCI Device VEN_1AF4&DEV_1045 (balloon), select update driver, browse to My Computer, and select E:\Balloon\w10\amd64. Next select the PCI Simple Communications Controller, update driver, E:\vioserial\w10. You may also be missing your ethernet connectivity. Go to the ethernet device and update the driver manually by selecting the E: drive, this will automatically find the correct driver. All should be set now. If your gpu is not appearing in the task manager do not freak out yet. We still have some more to do. If you are getting output that is good enough for now.

If you would like to backup your XML:

Check out Aaron's method, or alternatively, go to your XML through virt-manager and copy/paste it to an XML file though the text editor of your choice.

Ex:

$ cd Desktop/
$ vim whateverYouWantToNameIt.xml

copy paste and then:

:wq

to save it.

From this point, you can play with your new KVM:

I will provide my final XML here but make sure you are appropriately creating these entries relative to your hardware topology. I have a 12 core 24 thread CPU and some sections will not look the same for you.

If you are pulling my XML do NOT just copy my config and put it into your XML spot. It will 90% not work unless you are using the same hardware that I am using and even then there may be incompatibility with your topology versus mine. You have been warned.

Boot into your VM and double-check the drivers. Make sure everything is working correctly.

Now we can begin editing the XML for some changes that will boost performance!

Navigate to Overview -> XML we got some dirty work to do.

Change the very first line to be:

<domain xmlns:qemu="http://libvirt.org/schemas/domain/qemu/1.0" type="kvm">

Do not apply yet. Go all the way to the bottom of the XML and under the </devices> section add:

...
  </devices>
  <qemu:commandline>
    <qemu:arg value="-cpu"/>
    <qemu:arg value="host,hv_time,kvm=off,hv_vendor_id=null"/>
  </qemu:commandline>
</domain>

Now apply. These new insertions should stay. If you did it incorrectly they will disappear after applying. Double-check to make sure that it sticks. Proceed with the instructions in the next section.

CPU Pinning (ONLY if you are [multithreaded](https://en.wikipedia.org/wiki/Multithreading_(computer_architecture))):

Now we have to learn how to do some CPU pinning ONLY if you have a multithreaded capable CPU. VMs are incapable of distinguishing between physical and logical cores (threads). Virt-manager can see that 24 vCPUs exist and are available but the host has two cores mapped to a single physical core on the physical CPU die. If you want a terminal view of the cores run the command:

$ lscpu -e

This will provide output that looks like this (for me):

CPU NODE SOCKET CORE L1d:L1i:L2:L3 ONLINE    MAXMHZ    MINMHZ
  0    0      0    0 0:0:0:0          yes 3800.0000 2200.0000
  1    0      0    1 1:1:1:0          yes 3800.0000 2200.0000
  2    0      0    2 2:2:2:0          yes 3800.0000 2200.0000
  3    0      0    3 3:3:3:1          yes 3800.0000 2200.0000
  4    0      0    4 4:4:4:1          yes 3800.0000 2200.0000
  5    0      0    5 5:5:5:1          yes 3800.0000 2200.0000
  6    0      0    6 6:6:6:2          yes 3800.0000 2200.0000
  7    0      0    7 7:7:7:2          yes 3800.0000 2200.0000
  8    0      0    8 8:8:8:2          yes 3800.0000 2200.0000
  9    0      0    9 9:9:9:3          yes 3800.0000 2200.0000
 10    0      0   10 10:10:10:3       yes 3800.0000 2200.0000
 11    0      0   11 11:11:11:3       yes 3800.0000 2200.0000
 12    0      0    0 0:0:0:0          yes 3800.0000 2200.0000
 13    0      0    1 1:1:1:0          yes 3800.0000 2200.0000
 14    0      0    2 2:2:2:0          yes 3800.0000 2200.0000
 15    0      0    3 3:3:3:1          yes 3800.0000 2200.0000
 16    0      0    4 4:4:4:1          yes 3800.0000 2200.0000
 17    0      0    5 5:5:5:1          yes 3800.0000 2200.0000
 18    0      0    6 6:6:6:2          yes 3800.0000 2200.0000
 19    0      0    7 7:7:7:2          yes 3800.0000 2200.0000
 20    0      0    8 8:8:8:2          yes 3800.0000 2200.0000
 21    0      0    9 9:9:9:3          yes 3800.0000 2200.0000
 22    0      0   10 10:10:10:3       yes 3800.0000 2200.0000
 23    0      0   11 11:11:11:3       yes 3800.0000 2200.0000

As Bryan puts it, "A matching core id (I.e. "CORE" Column) means that the associated threads (i.e. "CPU" column) run on the same physical core.

If reading this information is a little scary from the terminal and you would like a GUI representation, please feel free to install hwloc:

$ sudo apt install hwloc

and then run it with:

$ lstopo

and you will get something that looks like:

We will now return to editing the XML configuration for the VM. Under the <currentMemory> section add the following:

...
<currentMemory unit="KiB">16777216</currentMemory>
<vcpu placement="static">12</vcpu>
<cputune>
    <vcpupin vcpu="0" cpuset="6"/>
    <vcpupin vcpu="1" cpuset="18"/>
    <vcpupin vcpu="2" cpuset="7"/>
    <vcpupin vcpu="3" cpuset="19"/>
    <vcpupin vcpu="4" cpuset="8"/>
    <vcpupin vcpu="5" cpuset="20"/>
    <vcpupin vcpu="6" cpuset="9"/>
    <vcpupin vcpu="7" cpuset="21"/>
    <vcpupin vcpu="8" cpuset="10"/>
    <vcpupin vcpu="9" cpuset="22"/>
    <vcpupin vcpu="10" cpuset="11"/>
    <vcpupin vcpu="11" cpuset="23"/>
    <emulatorpin cpuset="0-3"/>
    <iothreadpin iothread='1' cpuset='4-5,12-17'/>
</cputune>

REMEMBER THAT YOUR PINNING IS NOT GUARANTEED TO BE ANYTHING LIKE MINE! Please figure out your own pinning and apply the changes.

Now we will go down to the end of </features> and edit <cpu> by adding the following:

...
  </features>
  <cpu mode="host-passthrough" check="none" migratable="on">
    <topology sockets="1" dies="1" cores="6" threads="2"/>
    <cache mode="passthrough"/>
    <feature policy="require" name="topoext"/>
  </cpu>
...

This is based on the topology of your CPU and will vary. This is how I set it up for my AMD Ryzen 9 3900x as it will allocate 1 socket with 6 physical cores and 2 threads per core.

As Bryan states, "If you're wondering why I tuned my CPU configuration this way, I'll refer you to this section of the Libvirt domain XML format.16 More specifically, consider the cputune element and its underlying vcpupin, emulatorpin, and iothreadpin elements. The Arch Wiki recommends pinning the emulator and iothreads to host cores (if available) rather than the VCPUs assigned to the guest. In the example above, 12 out of my 24 threads are assigned as vCPUs to the guest, and from the remaining 12 threads on the host, 4 are assigned to the emulator and 8 are assigned to an iothread see below."

If you need disk tuning:

See how Bryan does it.

In my case, I did not need or use this method so I will not be going over it.

Benchmarking:

This is the end. You did it. Everything should be working. It is now up to you to find your way and tweak anything else that you may need to get your device set up and working exactly the way that you want it to.

The last thing that I did was add all of my PCI devices that controlled my USB I/O so I can plug and play USB ports for my VM and make use of my DAC/AMP for audio instead of using the GPU. It works for me but may not work for you. So have fun and enjoy the VM.

I'm hopeful that you will get great performance with your KVMs and hardware. If your CPU is not getting the performance you would like: look over the tweaks, topology and lastly if you can afford it, overclock your CPU. I have my CPU in a custom water-cooling loop so I'm pushing varying voltages and clock speeds.

Credits & Resources:

These are the sources I used to get my KVM running. There are a ton more and I suggest going over Bryan's Credits/Resources to find even more if you are interested in reading more or getting a deeper understanding on how or why this works.

Food For Thought & Things I've Found Out:

This will be a personal discussion section where I will discuss any personal problems or interesting things that I may encounter with the KVM.

So as of 3/22/21 I may have found a fix for a problem that I've been encountering with entirely random intervals and no way (that I know of) of figuring out why I'm getting hard crashes.

UPDATE: 4/1/2021 (NOT a cruel April Fools joke) I have found a fix for my specific problem. READ BELOW PLEASE!

Explanation: I will be using the KVM to play my games and maybe do some coding with C# and Microsoft-related applications and with entirely random intervals/chances my entire computer will turn off. This includes the host. It will act as a hard reset and take me to my BIOS where the PC reboots as normal with no problems, errors, or messages of any kind. My first thought is that there was or is something wrong with a PCIE device conflicting with the HOST and GUEST OS. I'm not sure if that is or isn't the problem but I tried removing and adding different devices with no fixes.

Possible fix?: I went back into the XML file for the project and removed pinning from CPU 0. I read on an obscure forum that removing the pinning from 0 fixed the problem. I am not entirely sure if this has fixed my problem but I have yet to encounter a hard crash. I ran a very demanding CPU stress test to see if that could cause a crash but doing so before or after the CPU pinning change did not result in any crashes. I will continue to monitor any complications that I get from this point forward.

Keeping this here for history's sake.

I found the actual answer to the variable crashing rates. So the problem is best summarized in this post on Reddit from u/yona_docova. The biggest "wow this is exactly the problem I'm having" moment was this mention.

"You play a game and randomly you will get a black screen crash for no apparent reason, and the PC restarts and you get back into the game, only not to have none, or another crash in 5 minutes or 4 hours later into the game.

It's totally random and impossible to predict."

This got my brain jogging. I had searched for hours/days about potential fixes that did not do anything for me. I reinstalled my VM, cloned different VMS tried different boot options, and different passthrough options, and none of those fixes worked except this (PLEASE NOTE: Although this worked for me IT MAY NOT WORK FOR YOU please consider trying everything before throwing in the towel. I understand how frustrating this can be as it happened to me. With patience and persistence you may find the answer that you are looking for.)

11/05/21 - I tested this configuration and reinstallation on the 21.04 update and did not run into any complications with the installation process. I did however notice that there was some performance degradation but I figured out it was due to being on an older BIOS version. Once I updated it was a better experience. Consistent performance, no crashing. I still use Scream to get audio through since I don't have enough space for a dedicated onboard DAC. It works well for what it's worth. Free, and slightly annoying to set up. But it works.

The Fix:

So here are the steps that I took to ensure that my KVM went back to its perfect working state:

  1. Back up your data just in case. You never know... I suggest Timeshift.
  2. Flash your BIOS to the newest stable version, if you are on the newest stable version and are experiencing this WITH the changes below, consider a beta branch or a downgrade. NOTE: Each BIOS is different. Your installation process may vary so PLEASE read your manual or go to your appropriate driver page to get the appropriate files and instructions. If you have an EZ flash functionality you can use that as well and you will be fine.
  3. Enter into your BIOS and change your Vcore voltage to normal or if that setting does not exist, set it to manual and change the offset to 0v. This may produce an offset with an incredibly small number which is fine as long as you input '0'.
  4. Suggestion - In the case of a BIOS reset being required, if your motherboard supports it, I would create a profile that can be saved to the BIOS that will save all of your configurations that will keep all of your options for overclocking and general settings.

This has fixed all of my problems up to this point and has removed the stress off of wondering if I can even play games with my friends or alone.

Having Audio complications?

Problem: I do not have a PCIE-based audio controller. Instead, I own a DAC and Amplifier for powering my collection of headphones. The problem that comes from this is that I either have to pass in the entire USB controller which causes mild conflicts OR pass the individual USB which has the possibility of producing the exact problem I had.

If you pass a USB audio device regardless of the sound quality disregarding the hz or bitrate you will get a crunchy, distorted, demonic, Decepticon level of audio. This is unbearable and destroys the quality of any media that is trying to be digested. I will detail below the fixes.

EASY FIX (Bandaid?)

I spent equally as long trying to fix the crashing problem as I was trying to fix my distorted, demonic-sounding audio. I'm going to cut to the chase with TWO fixes for you. One of the simplest fixes is to pass in the entire USB controller device but this causes problems if you want to be able to use your Linux host AND your Windows guest at the same time. Otherwise, you can do the next fix which I think is the coolest.

Better Fix For Me:

The best fix that I encountered for getting the best audio out of the VM is using SCREAM. SCREAM is "a virtual device driver for Windows that provides a discrete sound device. Audio played through this device is published on your local network as a PCM multicast stream." So essentially you're broadcasting your audio over the network and connecting to it through your HOST machine via the terminal or script. The installation is simple enough and running it is even easier. Just follow the directions presented in the git repo provided in the hyperlink above!

NOTE: If you are going to use the SCREAM method and you are deadset on getting the absolute fastest speed of audio without any type of delay at all for any professional level of gameplay then this is not the fix for you. You may be better off passing the entire controller. Since SCREAM uses your network to broadcast audio, you WILL have a slight delay. Think of Bluetooth. It's a very small amount of delay. I can play high-octane games just fine with barely any noticeable impact on gameplay. listening to audio or watching videos is not impacted in the slightest. I have noticed however during playing Rhythm games I just don't hit the notes perfectly as I usually do. That is easy for some games by creating a slight offset of a few milliseconds.

TL:DR if you do not want a delay. Do not use SCREAM. Find an alternative or pass the entire controller for audio which is the drawback of locking you out of your Linux host unless you have a proper workaround

SSD/HDD passthrough Disconnecting

So, one of the first real problems I have experienced other than the random crashing that I solved above is my PCI passthrough of an SSD or HDD disconnecting. I will start by saying the immediate fix that solved all my problems was upgrading my BIOS to the newest version. NOTE: This only occurred in the first place because I updated to the (at the time) more recent BIOS version which introduced the problem. But the version newer than that one fixed my problem.

If you are still having this problem some potential fixes are as follows:

  1. Downgrade your BIOS
  2. Turn off USB/PCI sleep in Windows power management under advanced settings
  3. Turn off Power Saving modes in power management settings

NOTE: I did find that turning off the USB power management settings actually helped my USB device pass-through from turning off and disconnecting from the VM. Worked great and I have not had a single problem since.

At this point (4/23/21) this works functionally as a proper near bare-metal-performance Windows 10 machine. It can play everything just fine considering I passed 16gb, 12 cores, and my RTX 3070.

I encounter some general Windows 10 problems here and there. If you encounter something extraordinarily strange it may be in your best interest to start breaking down the problem into some type of hierarchy. I will provide you with my breakdown of how I problem solved my harder complications.

General Troubleshooting

  1. Recognize the problem.

If you don't understand exactly what the problem is then that may cause a larger block of figuring out a solution. But honestly, sometimes the end-user just does not know what to do and that is okay. It happens to everyone and myself. So in the instance that you have a problem that you just do not know what to do then consider skipping this and moving onto the other steps.

  1. Check for a Windows update (CONTROVERSIAL!)

So a Windows update MIGHT fix the problem or it might cause even more. In this case be prepared to make a backup, snapshot, ISO, or whatever you want to preserve your data in the case of further complications.

  1. Check for drivers.

This is a no-brainer. Just check to see if there's a driver that may be causing wonky compatibility issues. This fixed some problems I had.

  1. For larger problems such as the Audio / SSD problems.

START by checking your XML and general KVM setup. Is there an incorrect setting? Is there a wrong topology? There can be plenty of problems here. If you have done some research by going to various websites or checking plenty of online boards then it may be time to consider moving to the next step.

  1. BIOS updates.

BE CAREFUL HERE - If you are going to update a BIOS it may be a good habit to write down your current problems. This will help in the instance of a BIOS update or BIOS setting causing more problems than solving. This was my problem with the SSDs dropping their connection. My STABLE BIOS update provided by the manufacturer caused my SSD to drop connection for an unknown reason. I did not change any settings aside from updating the BIOS to the newest BETA version and this fixed ALL of my disconnect problems. If you do not know how to flash a BIOS, watch a video or read your motherboard's manual.

  1. Check your hardware.

If you tried everything that you can think of and ensured that your XML is perfect, your drivers are fine, your updates are fine, etc, then it may be time to consider checking out your hardware. Test the basics like removing and placing back in your GPU, CPU, RAM, etc. It may also be a PSU not being able to handle multiple GPUs. There is a TON that can be at play once you reach this stage. A great way to test hardware though is to check in a different system if you have access to one. Best of luck.


DANGER ZONE: You need to be pretty desperate to attempt these changes as it could cause problems if you do not know what you're doing...

  1. Reinstall Pop!

Pretty straightforward and pretty annoying to deal with. Timeshift is your friend in cases like this though.

  1. Reinstall QEMU / Virt-Manager / your KVM

Honestly, this is the last thing anyone wants to do but I have had to do it and it sucks. Sometimes it is just the right thing to do to get a fresh install of something. Follow the instructions again and try your best! There is no shame in starting over.

  1. Trying a different Linux Distro

  2. Drop the KVM and install a dual boot to Windows.

This is an entirely valid fix as well. Do not feel like you are required to learn how to use and run a KVM just because you are on Linux. Never feel bad about using an OS. There is a learning curve to everything and some get it faster than others. You are not a failure. I have considered doing this when I couldn't find solutions to my problems. But it is so satisfying to find the fix and apply them. Keep at it, no one will judge you.