VFIO: How to enable Resizeable BAR (ReBAR) in your VFIO Virtual Machine

What is ReBAR and Why Do I Want It

Resizable BAR is a PCIe setting that allows the device (in this case GPU) to negotiate Base Address Register size (in this case access the whole frame buffer at once rather than in 256MB chunks). Some games make better use of this than others, and older GPUs do not support it. For games and hardware that make good use of ReBAR, the benefits are very nice to have.

Pre Requisites

You need ALL of these for this to work

  1. Your linux machine is booted using UEFI.
  2. Your GPU supports ReBar.
  3. Your Windows VM is installed and booted using UEFI.
  4. You are running linux Kernel 6.1 or later.
  5. You are running QEMU version 7.0.3 or later
  6. A CPU with 37 bits or more if you have nested pages enabled
    1. Check with egrep -w 'npt|ept' /proc/cpuinfo (npt is what AMD calls it, EPT is what Intel calls it)
    2. Check with the command grep 'bits physical' /proc/cpuinfo If you see numbers greater than or equal to 37 you are in the clear.

Host

Set the ReBAR size in command line. Find your GPU with lspci. Mine is listed as:

45:00.0 VGA compatible controller: Advanced Micro Devices, Inc. [AMD/ATI] Navi 21 [Radeon RX 6800/6800 XT / 6900 XT] (rev c0)

Next we need to get the bus address. Use the beginning part of that return, in my case 45:00.0, and run:

lspci -n | grep 45:00.0

With that output we now have all of the identifiers we need. In my case the returned result is:

45:00.0 0300: 1002:73bf (rev c0)

Now with these Identifiers, we can find out what our current settings are. Run the following command replacing $GPUAddress with your address to see what the ReBar is set to.

lspci -vvvs $GPUAddress | grep BAR

My output was

#lspci -vvvs 00:45:00.0 | grep BAR 130 ↵
Capabilities: [200 v1] Physical Resizable BAR
BAR 0: current size: 16GB, supported: 256MB 512MB 1GB 2GB 4GB 8GB 16GB
BAR 2: current size: 2MB, supported: 2MB 4MB 8MB 16MB 32MB 64MB 128MB 256MB

We can see that my AMD RX 6900 XT ReBar’s are 16GB for BAR0 (the main one) which is good as my video memory is 16GB on my RX 6900 XT. Bar 2 is set to 2MB, its smallest value. I’d like to change this to something higher. Window’s maximum for BAR2 is 8MB. NVIDIA cards have BAR0 set to a single value, and BAR1 is the Main BAR you want to edit. If BAR 2 on an AMD GPU size is greater than 8MB, Windows will not load the device, erroring out with Error Code 43.

 

Now this next part is specific to how you use your gaming virtual machine. In my case, I detach my my GPU from my host machine and apply the vfio-pci driver to it. It is detailed in the article on here titled: VFIO: Tuning your Windows gaming VM for optimal performance. I have a script that runs at the start of my virtual machine to detach the GPU from the amdgpu driver, apply the ReBar settings, and make it use the vfio-pci driver for safe pass through.

Here is a snippet from that script:

echo "unbind 6900xt gpu from amdgpu (1002:73bf)"
echo 0000:45:00.0 > /sys/bus/pci/drivers/amdgpu/unbind
sleep 1

echo "Setting rebar 0 size to 16GB" 
echo 14 > /sys/bus/pci/devices/0000:45:00.0/resource0_resize
sleep 1

echo "Setting the rebar 2 size to 8MB"
#Driver will error code 43 if above 8MB on BAR2 
echo 3 > /sys/bus/pci/devices/0000:45:00.0/resource2_resize

sleep 2
echo 1002 73bf > /sys/bus/pci/drivers/vfio-pci/new_id || echo -n "0000:45:00.0" > /sys/bus/pci/drivers/vfio-pci/bind
echo done
sleep 1

I am echoing the bit value of the ReBAR size I want. Values go as follows:

Bit Sizes
1 = 2MB
2 = 4MB
3 = 8MB
4 = 16MB
5 = 32MB
6 = 64MB
7 = 128MB
8 = 256MB
9 = 512MB
10 = 1GB
11 = 2GB
12 = 4GB
13 = 8GB
14 = 16GB
15 = 32GB

When setting the bar value, do not exceed the vram of your machine for BAR0 on amd or BAR1 on nvidia. There are ways to do this in kernel parameters and modprobe commands, but this is the method I use.

Virtual Machine

Here are the changes you need to make to your libvirt configuration

  1. Change the LibVirt Domain from <domain type="kvm/> to <domain xmlns:qemu="http://libvirt.org/schemas/domain/qemu/1.0" type="kvm">
  2. Append this to the bottom of your config:
    </devices>
    <qemu:commandline>
    <qemu:arg value="-fw_cfg"/>
    <qemu:arg value="opt/ovmf/X-PciMmio64Mb,string=65536"/>
    </qemu:commandline>
    </domain>

With those changes you can now boot your Virtual Machine. Download a program called GPU-Z to check if ReBar is working and enabled. GPU-Z will say Resizable Bar: Enabled

About: Ryan Parker

I'm a former captain of the Cyber Defense team, Current Infrastructure Security Specialist. I also have a side job helping small to medium business with anything technology doing everything imaginable. One of my hobbies is building out infrastructures for myself, friends, and clients. I current maintain a homelab with about 2TB of RAM, 180+ TB of storage, tons of CPU cores, and 100gbit networking backbone.


8 thoughts on “VFIO: How to enable Resizeable BAR (ReBAR) in your VFIO Virtual Machine”

  1. Hi!

    I’ve managed to enable rebar in BIOS and managed to boot into the guest OS and check that rebar is indeed enabled.

    My problem is, if I have rebar enabled, and launch the guest, the host system will go black/blank. And I can’t access it at all.

    1. Do you have multiple GPU’s? can you SSH to the machine from another computer? What are the host and vm specs? what is the config

      1. Yes, I use Ryzen 5700G’s vega iGPU for my primary GPU and then user DRI3 to render graphics with my RX6800XT.

        Something happened by enabling rebar and iommu got disabled in the BIOS. Re-enabled it. But then I still kept running to the black screen issue. Fixed it by deleting the xorg.conf in /etc/X11/ and set specific configs for my setup in /etc/X11/xorg.conf.d/.

        Now everything seems to be working correctly.

        I can provide more info if needed, if somebody runs into same problems.

  2. Back on this thread.

    I reduced performance after rebind to host? I seem to get about half the performance it should be.

    1. Really? Here are a few things i would verify:

      1. Make sure the GPU is on the same PCIe lanes as the active CPU with something like lstopo
      2. Make sure the cpu cache is getting correctly passed through
      3. If the card can enable msi or msi-x interrupts, make sure its priory is set to high
      4. If your lying to the host about being a virtual machine, this might mess with it in some ways. you will need to make sure you have cpu cache, pcie lanes, and other things 100% correct else it can cause some slowness.
      5. Check if windows event log has anything to say about it being weird. Check dmesg.

  3. With Kernel 6.12 Rebar no longer works for me, the AMD graphics driver of my Rx 6800 crashes after a short time with a driver timeout, while with Kernel 6.11 there are no problems. Do you also have such problems with kernel 6.12? And is there possibly a fix other than disabling Rebar?

    1. When new kernels come out there are sometimes bugs, I always have 2 kernels on my system (linux and linux-lts) because the latest sometimes has weird issues. Most of the time people smarter than me have already submitted a bug report for it. At the moment, im having issues with the latest kernel and im using the older LTS one because my nvidia drivers and displaylink drivers have some weird issues.

      1. I suspect that ReBar and the realtime function that has been introduced in the latest kernel are not compatible. It has probably never been properly tested because hardly any distribution uses the realtime kernel and ReBar has only been available since kernel 6.1. I hope that someone with knowledge of the matter notices and makes a bug report because for me this is too much.

Leave a Reply

Your email address will not be published. Required fields are marked *