[pve-devel] [PATCH 1/3] add pcie and x-vga passthrough

Alexandre Derumier aderumier at odiso.com
Mon Jun 23 17:41:53 CEST 2014


hostpci0:  .....,x-vga=on,pcie=1

x-vga require kernel 3.10 with vfio-vga support enable
if x-vga=on, we force vfio-pci device

pcie=1 choose the pciexpress bus (need q35 machine model)

Signed-off-by: Alexandre Derumier <aderumier at odiso.com>
---
 PVE/QemuServer.pm |   50 +++++++++++++++++++++++++++++++++++++++++++-------
 1 file changed, 43 insertions(+), 7 deletions(-)

diff --git a/PVE/QemuServer.pm b/PVE/QemuServer.pm
index c128258..597d013 100644
--- a/PVE/QemuServer.pm
+++ b/PVE/QemuServer.pm
@@ -565,7 +565,7 @@ PVE::JSONSchema::register_standard_option("pve-qm-usb", $usbdesc);
 my $hostpcidesc = {
         optional => 1,
         type => 'string', format => 'pve-qm-hostpci',
-        typetext => "[host=]HOSTPCIDEVICE [,driver=kvm|vfio] [,rombar=on|off]",
+        typetext => "[host=]HOSTPCIDEVICE [,driver=kvm|vfio] [,rombar=on|off] [,pcie=0|1] [,x-vga=on|off]",
         description => <<EODESCR,
 Map host pci devices. HOSTPCIDEVICE syntax is:
 
@@ -1274,6 +1274,10 @@ sub parse_hostpci {
 	    $res->{driver} = $1;
 	} elsif ($kv =~ m/^rombar=(on|off)$/) {
 	    $res->{rombar} = $1;
+	} elsif ($kv =~ m/^x-vga=(on|off)$/) {
+	    $res->{'x-vga'} = $1;
+	} elsif ($kv =~ m/^pcie=(\d+)$/) {
+	    $res->{pcie} = 1 if $1 == 1;
 	} else {
 	    warn "unknown hostpci setting '$kv'\n";
 	}
@@ -2411,12 +2415,24 @@ sub config_to_command {
 
     # host pci devices
     for (my $i = 0; $i < $MAX_HOSTPCI_DEVICES; $i++)  {
-          my $d = parse_hostpci($conf->{"hostpci$i"});
-          next if !$d;
-	  $pciaddr = print_pci_addr("hostpci$i", $bridges);
-	  my $rombar = $d->{rombar} && $d->{rombar} eq 'off' ? ",rombar=0" : "";
-	  my $driver = $d->{driver} && $d->{driver} eq 'vfio' ? "vfio-pci" : "pci-assign";
-	  push @$devices, '-device', "$driver,host=$d->{pciid},id=hostpci$i$pciaddr$rombar";
+	my $d = parse_hostpci($conf->{"hostpci$i"});
+	next if !$d;
+
+	my $pcie = $d->{pcie};
+	if($pcie){
+	    die "q35 machine model is not enabled" if !$q35;
+	    $pciaddr = print_pcie_addr("hostpci$i");
+	}else{
+	    $pciaddr = print_pci_addr("hostpci$i", $bridges);
+	}
+
+	my $rombar = $d->{rombar} && $d->{rombar} eq 'off' ? ",rombar=0" : "";
+	my $driver = $d->{driver} && $d->{driver} eq 'vfio' ? "vfio-pci" : "pci-assign";
+	my $xvga = $d->{'x-vga'} && $d->{'x-vga'} eq 'on' ? ",x-vga=on" : "";
+	$driver = "vfio-pci" if $xvga ne '';
+
+	push @$devices, '-device', "$driver,host=$d->{pciid},id=hostpci$i$pciaddr$rombar$xvga";
+
     }
 
     # usb devices
@@ -3808,6 +3824,26 @@ sub print_pci_addr {
 
 }
 
+sub print_pcie_addr {
+    my ($id) = @_;
+
+    my $res = '';
+    my $devices = {
+	hostpci0 => { bus => "ich9-pcie-port-1", addr => 0 },
+	hostpci1 => { bus => "ich9-pcie-port-2", addr => 0 },
+	hostpci2 => { bus => "ich9-pcie-port-3", addr => 0 },
+	hostpci3 => { bus => "ich9-pcie-port-4", addr => 0 },
+    };
+
+    if (defined($devices->{$id}->{bus}) && defined($devices->{$id}->{addr})) {
+	   my $addr = sprintf("0x%x", $devices->{$id}->{addr});
+	   my $bus = $devices->{$id}->{bus};
+	   $res = ",bus=$bus,addr=$addr";
+    }
+    return $res;
+
+}
+
 # vzdump restore implementaion
 
 sub tar_archive_read_firstfile {
-- 
1.7.10.4




More information about the pve-devel mailing list