9c6c51
From 2010df9c6a8a4ff984e3f1b697398da648342953 Mon Sep 17 00:00:00 2001
9c6c51
Message-Id: <2010df9c6a8a4ff984e3f1b697398da648342953@dist-git>
9c6c51
From: Erik Skultety <eskultet@redhat.com>
9c6c51
Date: Thu, 19 Jul 2018 15:04:05 +0200
9c6c51
Subject: [PATCH] conf: Introduce new video type 'none'
9c6c51
MIME-Version: 1.0
9c6c51
Content-Type: text/plain; charset=UTF-8
9c6c51
Content-Transfer-Encoding: 8bit
9c6c51
9c6c51
Historically, we've always enabled an emulated video device every time we
9c6c51
see that graphics should be supported with a guest. With the appearance
9c6c51
of mediated devices which can support QEMU's vfio-display capability,
9c6c51
users might want to use such a device as the only video device.
9c6c51
Therefore introduce a new, effectively a 'disable', type for video
9c6c51
device.
9c6c51
9c6c51
Reviewed-by: Ján Tomko <jtomko@redhat.com>
9c6c51
Signed-off-by: Erik Skultety <eskultet@redhat.com>
9c6c51
(cherry picked from commit d48813e81af798e3027edcc2f41be2630111ba65)
9c6c51
9c6c51
https://bugzilla.redhat.com/show_bug.cgi?id=1475770
9c6c51
Signed-off-by: Erik Skultety <eskultet@redhat.com>
9c6c51
Reviewed-by: Ján Tomko <jtomko@redhat.com>
9c6c51
---
9c6c51
 docs/formatdomain.html.in                     | 13 +++-
9c6c51
 docs/schemas/domaincommon.rng                 |  1 +
9c6c51
 src/conf/domain_conf.c                        | 61 +++++++++++++------
9c6c51
 src/conf/domain_conf.h                        |  1 +
9c6c51
 src/qemu/qemu_command.c                       | 12 +++-
9c6c51
 src/qemu/qemu_domain.c                        |  2 +
9c6c51
 src/qemu/qemu_domain_address.c                |  7 ++-
9c6c51
 tests/domaincapsschemadata/full.xml           |  1 +
9c6c51
 .../video-invalid-multiple-devices.xml        | 33 ++++++++++
9c6c51
 tests/qemuxml2argvdata/video-none-device.args | 27 ++++++++
9c6c51
 tests/qemuxml2argvdata/video-none-device.xml  | 39 ++++++++++++
9c6c51
 tests/qemuxml2argvtest.c                      |  4 +-
9c6c51
 .../qemuxml2xmloutdata/video-none-device.xml  | 42 +++++++++++++
9c6c51
 tests/qemuxml2xmltest.c                       |  1 +
9c6c51
 14 files changed, 220 insertions(+), 24 deletions(-)
9c6c51
 create mode 100644 tests/qemuxml2argvdata/video-invalid-multiple-devices.xml
9c6c51
 create mode 100644 tests/qemuxml2argvdata/video-none-device.args
9c6c51
 create mode 100644 tests/qemuxml2argvdata/video-none-device.xml
9c6c51
 create mode 100644 tests/qemuxml2xmloutdata/video-none-device.xml
9c6c51
9c6c51
diff --git a/docs/formatdomain.html.in b/docs/formatdomain.html.in
9c6c51
index 0e8f0a125f..42acf7a828 100644
9c6c51
--- a/docs/formatdomain.html.in
9c6c51
+++ b/docs/formatdomain.html.in
9c6c51
@@ -6666,9 +6666,18 @@ qemu-kvm -net nic,model=? /dev/null
9c6c51
           The model element has a mandatory type
9c6c51
           attribute which takes the value "vga", "cirrus", "vmvga", "xen",
9c6c51
           "vbox", "qxl" (since 0.8.6),
9c6c51
-          "virtio" (since 1.3.0)
9c6c51
-          or "gop" (since 3.2.0)
9c6c51
+          "virtio" (since 1.3.0),
9c6c51
+          "gop" (since 3.2.0), or
9c6c51
+          "none" (since 4.6.0)
9c6c51
           depending on the hypervisor features available.
9c6c51
+          The purpose of the type none is to instruct libvirt not
9c6c51
+          to add a default video device in the guest (see the paragraph above).
9c6c51
+          This legacy behaviour can be inconvenient in cases where GPU mediated
9c6c51
+          devices are meant to be the only rendering device within a guest and
9c6c51
+          so specifying another video device along with type
9c6c51
+          none.
9c6c51
+          Refer to Host device assignment to see
9c6c51
+          how to add a mediated device into a guest.
9c6c51
         

9c6c51
         

9c6c51
           You can provide the amount of video memory in kibibytes (blocks of
9c6c51
diff --git a/docs/schemas/domaincommon.rng b/docs/schemas/domaincommon.rng
9c6c51
index be8430ab22..ac04af51a1 100644
9c6c51
--- a/docs/schemas/domaincommon.rng
9c6c51
+++ b/docs/schemas/domaincommon.rng
9c6c51
@@ -3454,6 +3454,7 @@
9c6c51
                 <value>vbox</value>
9c6c51
                 <value>virtio</value>
9c6c51
                 <value>gop</value>
9c6c51
+                <value>none</value>
9c6c51
               </choice>
9c6c51
             </attribute>
9c6c51
             <group>
9c6c51
diff --git a/src/conf/domain_conf.c b/src/conf/domain_conf.c
9c6c51
index 830c298158..23288aa01b 100644
9c6c51
--- a/src/conf/domain_conf.c
9c6c51
+++ b/src/conf/domain_conf.c
9c6c51
@@ -590,7 +590,8 @@ VIR_ENUM_IMPL(virDomainVideo, VIR_DOMAIN_VIDEO_TYPE_LAST,
9c6c51
               "qxl",
9c6c51
               "parallels",
9c6c51
               "virtio",
9c6c51
-              "gop")
9c6c51
+              "gop",
9c6c51
+              "none")
9c6c51
 
9c6c51
 VIR_ENUM_IMPL(virDomainVideoVGAConf, VIR_DOMAIN_VIDEO_VGACONF_LAST,
9c6c51
               "io",
9c6c51
@@ -5106,22 +5107,30 @@ virDomainDefPostParseVideo(virDomainDefPtr def,
9c6c51
     if (def->nvideos == 0)
9c6c51
         return 0;
9c6c51
 
9c6c51
-    virDomainDeviceDef device = {
9c6c51
-        .type = VIR_DOMAIN_DEVICE_VIDEO,
9c6c51
-        .data.video = def->videos[0],
9c6c51
-    };
9c6c51
+    if (def->videos[0]->type == VIR_DOMAIN_VIDEO_TYPE_NONE) {
9c6c51
+        /* we don't want to format any values we automatically fill in for
9c6c51
+         * videos into the XML, so clear them
9c6c51
+         */
9c6c51
+        virDomainVideoDefClear(def->videos[0]);
9c6c51
+        def->videos[0]->type = VIR_DOMAIN_VIDEO_TYPE_NONE;
9c6c51
+    } else {
9c6c51
+        virDomainDeviceDef device = {
9c6c51
+            .type = VIR_DOMAIN_DEVICE_VIDEO,
9c6c51
+            .data.video = def->videos[0],
9c6c51
+        };
9c6c51
 
9c6c51
-    /* Mark the first video as primary. If the user specified
9c6c51
-     * primary="yes", the parser already inserted the device at
9c6c51
-     * def->videos[0]
9c6c51
-     */
9c6c51
-    def->videos[0]->primary = true;
9c6c51
+        /* Mark the first video as primary. If the user specified
9c6c51
+         * primary="yes", the parser already inserted the device at
9c6c51
+         * def->videos[0]
9c6c51
+         */
9c6c51
+        def->videos[0]->primary = true;
9c6c51
 
9c6c51
-    /* videos[0] might have been added in AddImplicitDevices, after we've
9c6c51
-     * done the per-device post-parse */
9c6c51
-    if (virDomainDefPostParseDeviceIterator(def, &device,
9c6c51
-                                            NULL, opaque) < 0)
9c6c51
-        return -1;
9c6c51
+        /* videos[0] might have been added in AddImplicitDevices, after we've
9c6c51
+         * done the per-device post-parse */
9c6c51
+        if (virDomainDefPostParseDeviceIterator(def, &device,
9c6c51
+                                                NULL, opaque) < 0)
9c6c51
+            return -1;
9c6c51
+    }
9c6c51
 
9c6c51
     return 0;
9c6c51
 }
9c6c51
@@ -5670,13 +5679,30 @@ virDomainHostdevDefValidate(const virDomainHostdevDef *hostdev)
9c6c51
 
9c6c51
 
9c6c51
 static int
9c6c51
-virDomainVideoDefValidate(const virDomainVideoDef *video)
9c6c51
+virDomainVideoDefValidate(const virDomainVideoDef *video,
9c6c51
+                          const virDomainDef *def)
9c6c51
 {
9c6c51
+    size_t i;
9c6c51
+
9c6c51
     if (video->type == VIR_DOMAIN_VIDEO_TYPE_DEFAULT) {
9c6c51
         virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
9c6c51
                        _("missing video model and cannot determine default"));
9c6c51
         return -1;
9c6c51
     }
9c6c51
+
9c6c51
+    /* it doesn't make sense to pair video device type 'none' with any other
9c6c51
+     * types, there can be only a single video device in such case
9c6c51
+     */
9c6c51
+    for (i = 0; i < def->nvideos; i++) {
9c6c51
+        if (def->videos[i]->type == VIR_DOMAIN_VIDEO_TYPE_NONE &&
9c6c51
+            def->nvideos > 1) {
9c6c51
+            virReportError(VIR_ERR_CONFIG_UNSUPPORTED, "%s",
9c6c51
+                           _("a '%s' video type must be the only video device "
9c6c51
+                             "defined for the domain"));
9c6c51
+            return -1;
9c6c51
+        }
9c6c51
+    }
9c6c51
+
9c6c51
     return 0;
9c6c51
 }
9c6c51
 
9c6c51
@@ -5738,7 +5764,7 @@ virDomainDeviceDefValidateInternal(const virDomainDeviceDef *dev,
9c6c51
         return virDomainHostdevDefValidate(dev->data.hostdev);
9c6c51
 
9c6c51
     case VIR_DOMAIN_DEVICE_VIDEO:
9c6c51
-        return virDomainVideoDefValidate(dev->data.video);
9c6c51
+        return virDomainVideoDefValidate(dev->data.video, def);
9c6c51
 
9c6c51
     case VIR_DOMAIN_DEVICE_MEMORY:
9c6c51
         return virDomainMemoryDefValidate(dev->data.memory);
9c6c51
@@ -15048,6 +15074,7 @@ virDomainVideoDefaultRAM(const virDomainDef *def,
9c6c51
     case VIR_DOMAIN_VIDEO_TYPE_PARALLELS:
9c6c51
     case VIR_DOMAIN_VIDEO_TYPE_VIRTIO:
9c6c51
     case VIR_DOMAIN_VIDEO_TYPE_GOP:
9c6c51
+    case VIR_DOMAIN_VIDEO_TYPE_NONE:
9c6c51
     case VIR_DOMAIN_VIDEO_TYPE_LAST:
9c6c51
     default:
9c6c51
         return 0;
9c6c51
diff --git a/src/conf/domain_conf.h b/src/conf/domain_conf.h
9c6c51
index 8ca9558ceb..5e2f21dea3 100644
9c6c51
--- a/src/conf/domain_conf.h
9c6c51
+++ b/src/conf/domain_conf.h
9c6c51
@@ -1424,6 +1424,7 @@ typedef enum {
9c6c51
     VIR_DOMAIN_VIDEO_TYPE_PARALLELS, /* pseudo device for VNC in containers */
9c6c51
     VIR_DOMAIN_VIDEO_TYPE_VIRTIO,
9c6c51
     VIR_DOMAIN_VIDEO_TYPE_GOP,
9c6c51
+    VIR_DOMAIN_VIDEO_TYPE_NONE,
9c6c51
 
9c6c51
     VIR_DOMAIN_VIDEO_TYPE_LAST
9c6c51
 } virDomainVideoType;
9c6c51
diff --git a/src/qemu/qemu_command.c b/src/qemu/qemu_command.c
9c6c51
index 1fce45134f..954265feb0 100644
9c6c51
--- a/src/qemu/qemu_command.c
9c6c51
+++ b/src/qemu/qemu_command.c
9c6c51
@@ -105,7 +105,8 @@ VIR_ENUM_IMPL(qemuVideo, VIR_DOMAIN_VIDEO_TYPE_LAST,
9c6c51
               "qxl",
9c6c51
               "", /* don't support parallels */
9c6c51
               "", /* no need for virtio */
9c6c51
-              "" /* don't support gop */);
9c6c51
+              "" /* don't support gop */,
9c6c51
+              "" /* 'none' doesn't make sense here */);
9c6c51
 
9c6c51
 VIR_ENUM_DECL(qemuDeviceVideo)
9c6c51
 
9c6c51
@@ -119,7 +120,8 @@ VIR_ENUM_IMPL(qemuDeviceVideo, VIR_DOMAIN_VIDEO_TYPE_LAST,
9c6c51
               "qxl-vga",
9c6c51
               "", /* don't support parallels */
9c6c51
               "virtio-vga",
9c6c51
-              "" /* don't support gop */);
9c6c51
+              "" /* don't support gop */,
9c6c51
+              "" /* 'none' doesn't make sense here */);
9c6c51
 
9c6c51
 VIR_ENUM_DECL(qemuDeviceVideoSecondary)
9c6c51
 
9c6c51
@@ -133,7 +135,8 @@ VIR_ENUM_IMPL(qemuDeviceVideoSecondary, VIR_DOMAIN_VIDEO_TYPE_LAST,
9c6c51
               "qxl",
9c6c51
               "", /* don't support parallels */
9c6c51
               "virtio-gpu",
9c6c51
-              "" /* don't support gop */);
9c6c51
+              "" /* don't support gop */,
9c6c51
+              "" /* 'none' doesn't make sense here */);
9c6c51
 
9c6c51
 VIR_ENUM_DECL(qemuSoundCodec)
9c6c51
 
9c6c51
@@ -4421,6 +4424,9 @@ qemuBuildVideoCommandLine(virCommandPtr cmd,
9c6c51
         char *str = NULL;
9c6c51
         virDomainVideoDefPtr video = def->videos[i];
9c6c51
 
9c6c51
+        if (video->type == VIR_DOMAIN_VIDEO_TYPE_NONE)
9c6c51
+            continue;
9c6c51
+
9c6c51
         if (video->primary) {
9c6c51
             if (virQEMUCapsGet(qemuCaps, QEMU_CAPS_DEVICE_VIDEO_PRIMARY)) {
9c6c51
 
9c6c51
diff --git a/src/qemu/qemu_domain.c b/src/qemu/qemu_domain.c
9c6c51
index 5337f1ce55..508846116b 100644
9c6c51
--- a/src/qemu/qemu_domain.c
9c6c51
+++ b/src/qemu/qemu_domain.c
9c6c51
@@ -4528,6 +4528,8 @@ static int
9c6c51
 qemuDomainDeviceDefValidateVideo(const virDomainVideoDef *video)
9c6c51
 {
9c6c51
     switch ((virDomainVideoType) video->type) {
9c6c51
+    case VIR_DOMAIN_VIDEO_TYPE_NONE:
9c6c51
+        return 0;
9c6c51
     case VIR_DOMAIN_VIDEO_TYPE_XEN:
9c6c51
     case VIR_DOMAIN_VIDEO_TYPE_VBOX:
9c6c51
     case VIR_DOMAIN_VIDEO_TYPE_PARALLELS:
9c6c51
diff --git a/src/qemu/qemu_domain_address.c b/src/qemu/qemu_domain_address.c
9c6c51
index ab2ac022f1..e6996934b8 100644
9c6c51
--- a/src/qemu/qemu_domain_address.c
9c6c51
+++ b/src/qemu/qemu_domain_address.c
9c6c51
@@ -821,6 +821,7 @@ qemuDomainDeviceCalculatePCIConnectFlags(virDomainDeviceDefPtr dev,
9c6c51
 
9c6c51
         case VIR_DOMAIN_VIDEO_TYPE_DEFAULT:
9c6c51
         case VIR_DOMAIN_VIDEO_TYPE_GOP:
9c6c51
+        case VIR_DOMAIN_VIDEO_TYPE_NONE:
9c6c51
         case VIR_DOMAIN_VIDEO_TYPE_LAST:
9c6c51
             return 0;
9c6c51
         }
9c6c51
@@ -1532,7 +1533,8 @@ qemuDomainValidateDevicePCISlotsPIIX3(virDomainDefPtr def,
9c6c51
             goto cleanup;
9c6c51
     }
9c6c51
 
9c6c51
-    if (def->nvideos > 0) {
9c6c51
+    if (def->nvideos > 0 &&
9c6c51
+        def->videos[0]->type != VIR_DOMAIN_VIDEO_TYPE_NONE) {
9c6c51
         /* Because the PIIX3 integrated IDE/USB controllers are
9c6c51
          * already at slot 1, when qemu looks for the first free slot
9c6c51
          * to place the VGA controller (which is always the first
9c6c51
@@ -1540,6 +1542,7 @@ qemuDomainValidateDevicePCISlotsPIIX3(virDomainDefPtr def,
9c6c51
          * at slot 2.
9c6c51
          */
9c6c51
         virDomainVideoDefPtr primaryVideo = def->videos[0];
9c6c51
+
9c6c51
         if (virDeviceInfoPCIAddressWanted(&primaryVideo->info)) {
9c6c51
             memset(&tmp_addr, 0, sizeof(tmp_addr));
9c6c51
             tmp_addr.slot = 2;
9c6c51
@@ -2105,6 +2108,8 @@ qemuDomainAssignDevicePCISlots(virDomainDefPtr def,
9c6c51
 
9c6c51
     /* Video devices */
9c6c51
     for (i = 0; i < def->nvideos; i++) {
9c6c51
+        if (def->videos[i]->type == VIR_DOMAIN_VIDEO_TYPE_NONE)
9c6c51
+            continue;
9c6c51
 
9c6c51
         if (!virDeviceInfoPCIAddressWanted(&def->videos[i]->info))
9c6c51
             continue;
9c6c51
diff --git a/tests/domaincapsschemadata/full.xml b/tests/domaincapsschemadata/full.xml
9c6c51
index 154c4a6fe9..eafba1ae5b 100644
9c6c51
--- a/tests/domaincapsschemadata/full.xml
9c6c51
+++ b/tests/domaincapsschemadata/full.xml
9c6c51
@@ -74,6 +74,7 @@
9c6c51
         <value>parallels</value>
9c6c51
         <value>virtio</value>
9c6c51
         <value>gop</value>
9c6c51
+        <value>none</value>
9c6c51
       </enum>
9c6c51
     </video>
9c6c51
     <hostdev supported='yes'>
9c6c51
diff --git a/tests/qemuxml2argvdata/video-invalid-multiple-devices.xml b/tests/qemuxml2argvdata/video-invalid-multiple-devices.xml
9c6c51
new file mode 100644
9c6c51
index 0000000000..3f105efaae
9c6c51
--- /dev/null
9c6c51
+++ b/tests/qemuxml2argvdata/video-invalid-multiple-devices.xml
9c6c51
@@ -0,0 +1,33 @@
9c6c51
+<domain type='qemu'>
9c6c51
+  <name>QEMUGuest1</name>
9c6c51
+  <uuid>c7a5fdbd-edaf-9455-926a-d65c16db1809</uuid>
9c6c51
+  <memory unit='KiB'>219100</memory>
9c6c51
+  <currentMemory unit='KiB'>219100</currentMemory>
9c6c51
+  <vcpu placement='static' cpuset='1-4,8-20,525'>1</vcpu>
9c6c51
+  <os>
9c6c51
+    <type arch='i686' machine='pc'>hvm</type>
9c6c51
+    <boot dev='hd'/>
9c6c51
+  </os>
9c6c51
+  <clock offset='utc'/>
9c6c51
+  <on_poweroff>destroy</on_poweroff>
9c6c51
+  <on_reboot>restart</on_reboot>
9c6c51
+  <on_crash>destroy</on_crash>
9c6c51
+  <devices>
9c6c51
+    <emulator>/usr/bin/qemu-system-i686</emulator>
9c6c51
+    <disk type='block' device='disk'>
9c6c51
+      <source dev='/dev/HostVG/QEMUGuest1'/>
9c6c51
+      <target dev='hda' bus='ide'/>
9c6c51
+      <address type='drive' controller='0' bus='0' target='0' unit='0'/>
9c6c51
+    </disk>
9c6c51
+    <controller type='usb' index='0'/>
9c6c51
+    <controller type='ide' index='0'/>
9c6c51
+    <controller type='pci' index='0' model='pci-root'/>
9c6c51
+    <video>
9c6c51
+      <model type='qxl'/>
9c6c51
+    </video>
9c6c51
+    <video>
9c6c51
+      <model type='none'/>
9c6c51
+    </video>
9c6c51
+    <memballoon model='virtio'/>
9c6c51
+  </devices>
9c6c51
+</domain>
9c6c51
diff --git a/tests/qemuxml2argvdata/video-none-device.args b/tests/qemuxml2argvdata/video-none-device.args
9c6c51
new file mode 100644
9c6c51
index 0000000000..1b03c0cb97
9c6c51
--- /dev/null
9c6c51
+++ b/tests/qemuxml2argvdata/video-none-device.args
9c6c51
@@ -0,0 +1,27 @@
9c6c51
+LC_ALL=C \
9c6c51
+PATH=/bin \
9c6c51
+HOME=/home/test \
9c6c51
+USER=test \
9c6c51
+LOGNAME=test \
9c6c51
+QEMU_AUDIO_DRV=none \
9c6c51
+/usr/bin/qemu-system-i686 \
9c6c51
+-name QEMUGuest1 \
9c6c51
+-S \
9c6c51
+-machine pc,accel=tcg,usb=off,dump-guest-core=off \
9c6c51
+-m 214 \
9c6c51
+-smp 1,sockets=1,cores=1,threads=1 \
9c6c51
+-uuid c7a5fdbd-edaf-9455-926a-d65c16db1809 \
9c6c51
+-no-user-config \
9c6c51
+-nodefaults \
9c6c51
+-chardev socket,id=charmonitor,path=/tmp/lib/domain--1-QEMUGuest1/monitor.sock,\
9c6c51
+server,nowait \
9c6c51
+-mon chardev=charmonitor,id=monitor,mode=control \
9c6c51
+-rtc base=utc \
9c6c51
+-no-shutdown \
9c6c51
+-no-acpi \
9c6c51
+-boot c \
9c6c51
+-usb \
9c6c51
+-drive file=/dev/HostVG/QEMUGuest1,format=raw,if=none,id=drive-ide0-0-0 \
9c6c51
+-device ide-drive,bus=ide.0,unit=0,drive=drive-ide0-0-0,id=ide0-0-0 \
9c6c51
+-vnc 127.0.0.1:0 \
9c6c51
+-device virtio-balloon-pci,id=balloon0,bus=pci.0,addr=0x3
9c6c51
diff --git a/tests/qemuxml2argvdata/video-none-device.xml b/tests/qemuxml2argvdata/video-none-device.xml
9c6c51
new file mode 100644
9c6c51
index 0000000000..4b591562b7
9c6c51
--- /dev/null
9c6c51
+++ b/tests/qemuxml2argvdata/video-none-device.xml
9c6c51
@@ -0,0 +1,39 @@
9c6c51
+<domain type='qemu'>
9c6c51
+  <name>QEMUGuest1</name>
9c6c51
+  <uuid>c7a5fdbd-edaf-9455-926a-d65c16db1809</uuid>
9c6c51
+  <memory unit='KiB'>219136</memory>
9c6c51
+  <currentMemory unit='KiB'>219136</currentMemory>
9c6c51
+  <vcpu placement='static'>1</vcpu>
9c6c51
+  <os>
9c6c51
+    <type arch='i686' machine='pc'>hvm</type>
9c6c51
+    <boot dev='hd'/>
9c6c51
+  </os>
9c6c51
+  <clock offset='utc'/>
9c6c51
+  <on_poweroff>destroy</on_poweroff>
9c6c51
+  <on_reboot>restart</on_reboot>
9c6c51
+  <on_crash>destroy</on_crash>
9c6c51
+  <devices>
9c6c51
+    <emulator>/usr/bin/qemu-system-i686</emulator>
9c6c51
+    <disk type='block' device='disk'>
9c6c51
+      <source dev='/dev/HostVG/QEMUGuest1'/>
9c6c51
+      <target dev='hda' bus='ide'/>
9c6c51
+      <address type='drive' controller='0' bus='0' target='0' unit='0'/>
9c6c51
+    </disk>
9c6c51
+    <controller type='usb' index='0'>
9c6c51
+      <address type='pci' domain='0x0000' bus='0x00' slot='0x01' function='0x2'/>
9c6c51
+    </controller>
9c6c51
+    <controller type='ide' index='0'>
9c6c51
+      <address type='pci' domain='0x0000' bus='0x00' slot='0x01' function='0x1'/>
9c6c51
+    </controller>
9c6c51
+    <controller type='pci' index='0' model='pci-root'/>
9c6c51
+    <input type='mouse' bus='ps2'/>
9c6c51
+    <input type='keyboard' bus='ps2'/>
9c6c51
+    <graphics type='vnc'/>
9c6c51
+    <video>
9c6c51
+      <model type='none'/>
9c6c51
+    </video>
9c6c51
+    <memballoon model='virtio'>
9c6c51
+      <address type='pci' domain='0x0000' bus='0x00' slot='0x03' function='0x0'/>
9c6c51
+    </memballoon>
9c6c51
+  </devices>
9c6c51
+</domain>
9c6c51
diff --git a/tests/qemuxml2argvtest.c b/tests/qemuxml2argvtest.c
9c6c51
index 9237a4fb89..3cff4ffb5e 100644
9c6c51
--- a/tests/qemuxml2argvtest.c
9c6c51
+++ b/tests/qemuxml2argvtest.c
9c6c51
@@ -1998,7 +1998,9 @@ mymain(void)
9c6c51
             QEMU_CAPS_DEVICE_VIRTIO_VGA,
9c6c51
             QEMU_CAPS_DEVICE_VIDEO_PRIMARY,
9c6c51
             QEMU_CAPS_VIRTIO_GPU_MAX_OUTPUTS);
9c6c51
-    DO_TEST_PARSE_ERROR("video-invalid", NONE);
9c6c51
+    DO_TEST("video-none-device",
9c6c51
+            QEMU_CAPS_VNC);
9c6c51
+    DO_TEST_PARSE_ERROR("video-invalid-multiple-devices", NONE);
9c6c51
 
9c6c51
     DO_TEST("virtio-rng-default",
9c6c51
             QEMU_CAPS_DEVICE_VIRTIO_RNG,
9c6c51
diff --git a/tests/qemuxml2xmloutdata/video-none-device.xml b/tests/qemuxml2xmloutdata/video-none-device.xml
9c6c51
new file mode 100644
9c6c51
index 0000000000..6e76b394fe
9c6c51
--- /dev/null
9c6c51
+++ b/tests/qemuxml2xmloutdata/video-none-device.xml
9c6c51
@@ -0,0 +1,42 @@
9c6c51
+<domain type='qemu'>
9c6c51
+  <name>QEMUGuest1</name>
9c6c51
+  <uuid>c7a5fdbd-edaf-9455-926a-d65c16db1809</uuid>
9c6c51
+  <memory unit='KiB'>219136</memory>
9c6c51
+  <currentMemory unit='KiB'>219136</currentMemory>
9c6c51
+  <vcpu placement='static'>1</vcpu>
9c6c51
+  <os>
9c6c51
+    <type arch='i686' machine='pc'>hvm</type>
9c6c51
+    <boot dev='hd'/>
9c6c51
+  </os>
9c6c51
+  <clock offset='utc'/>
9c6c51
+  <on_poweroff>destroy</on_poweroff>
9c6c51
+  <on_reboot>restart</on_reboot>
9c6c51
+  <on_crash>destroy</on_crash>
9c6c51
+  <devices>
9c6c51
+    <emulator>/usr/bin/qemu-system-i686</emulator>
9c6c51
+    <disk type='block' device='disk'>
9c6c51
+      <driver name='qemu' type='raw'/>
9c6c51
+      <source dev='/dev/HostVG/QEMUGuest1'/>
9c6c51
+      <target dev='hda' bus='ide'/>
9c6c51
+      <address type='drive' controller='0' bus='0' target='0' unit='0'/>
9c6c51
+    </disk>
9c6c51
+    <controller type='usb' index='0'>
9c6c51
+      <address type='pci' domain='0x0000' bus='0x00' slot='0x01' function='0x2'/>
9c6c51
+    </controller>
9c6c51
+    <controller type='ide' index='0'>
9c6c51
+      <address type='pci' domain='0x0000' bus='0x00' slot='0x01' function='0x1'/>
9c6c51
+    </controller>
9c6c51
+    <controller type='pci' index='0' model='pci-root'/>
9c6c51
+    <input type='mouse' bus='ps2'/>
9c6c51
+    <input type='keyboard' bus='ps2'/>
9c6c51
+    <graphics type='vnc' port='-1' autoport='yes'>
9c6c51
+      <listen type='address'/>
9c6c51
+    </graphics>
9c6c51
+    <video>
9c6c51
+      <model type='none'/>
9c6c51
+    </video>
9c6c51
+    <memballoon model='virtio'>
9c6c51
+      <address type='pci' domain='0x0000' bus='0x00' slot='0x03' function='0x0'/>
9c6c51
+    </memballoon>
9c6c51
+  </devices>
9c6c51
+</domain>
9c6c51
diff --git a/tests/qemuxml2xmltest.c b/tests/qemuxml2xmltest.c
9c6c51
index e418e67f6c..e35644d479 100644
9c6c51
--- a/tests/qemuxml2xmltest.c
9c6c51
+++ b/tests/qemuxml2xmltest.c
9c6c51
@@ -1145,6 +1145,7 @@ mymain(void)
9c6c51
             QEMU_CAPS_VIRTIO_GPU_MAX_OUTPUTS,
9c6c51
             QEMU_CAPS_VNC,
9c6c51
             QEMU_CAPS_DEVICE_VIRTIO_GPU_CCW);
9c6c51
+    DO_TEST("video-none-device", NONE);
9c6c51
 
9c6c51
     DO_TEST("intel-iommu",
9c6c51
             QEMU_CAPS_DEVICE_INTEL_IOMMU);
9c6c51
-- 
9c6c51
2.18.0
9c6c51