qemu: Support ram bar size for qxl devices

Adds a "ram" attribute globally to the video.model element, that changes
the resulting qemu command line only if video.type == "qxl".

<video>
  <model type='qxl' ram='65536' vram='65536' heads='1'/>
</video>

That attribute gets a default value of 64*1024. The schema is unchanged
for other video element types.

The resulting qemu command line change is the addition of

-global qxl-vga.ram_size=<ram>*1024

or

-global qxl.ram_size=<ram>*1024

For the main and secondary qxl devices respectively.

The default for the qxl ram bar is 64*1024 kilobytes (the same as the
default qxl vram bar size).
This commit is contained in:
Alon Levy 2013-01-18 20:36:36 +02:00 committed by Eric Blake
parent c9a85af319
commit 55bfd020d8
12 changed files with 110 additions and 46 deletions

View File

@ -3565,7 +3565,11 @@ qemu-kvm -net nic,model=? /dev/null
video device in domain xml is the primary one, but the optional video device in domain xml is the primary one, but the optional
attribute <code>primary</code> (<span class="since">since 1.0.2</span>) attribute <code>primary</code> (<span class="since">since 1.0.2</span>)
with value 'yes' can be used to mark the primary in cases of mutiple with value 'yes' can be used to mark the primary in cases of mutiple
video device. The non-primary must be type of "qxl". video device. The non-primary must be type of "qxl". The optional
attribute <code>ram</code> (<span class="since">since
1.0.2</span>) is allowed for "qxl" type only and specifies
the size of the primary bar, while <code>vram</code> specifies the
secondary bar size. If "ram" is not supplied a default value is used.
</dd> </dd>
<dt><code>model</code></dt> <dt><code>model</code></dt>

View File

@ -2251,22 +2251,34 @@
</define> </define>
<!-- <!--
A video adapter description, allowing configuration of device A video adapter description, allowing configuration of device
model, number of virtual heads, and video ram size model, number of virtual heads, video ram size, and for qxl
both ram bar sizes.
--> -->
<define name="video"> <define name="video">
<element name="video"> <element name="video">
<optional> <optional>
<element name="model"> <element name="model">
<attribute name="type"> <choice>
<choice> <attribute name="type">
<value>vga</value> <choice>
<value>cirrus</value> <value>vga</value>
<value>vmvga</value> <value>cirrus</value>
<value>xen</value> <value>vmvga</value>
<value>vbox</value> <value>xen</value>
<value>qxl</value> <value>vbox</value>
</choice> </choice>
</attribute> </attribute>
<group>
<attribute name="type">
<value>qxl</value>
</attribute>
<optional>
<attribute name="ram">
<ref name="unsignedInt"/>
</attribute>
</optional>
</group>
</choice>
<optional> <optional>
<attribute name="vram"> <attribute name="vram">
<ref name="unsignedInt"/> <ref name="unsignedInt"/>

View File

@ -7391,6 +7391,7 @@ virDomainVideoDefParseXML(const xmlNodePtr node,
char *type = NULL; char *type = NULL;
char *heads = NULL; char *heads = NULL;
char *vram = NULL; char *vram = NULL;
char *ram = NULL;
char *primary = NULL; char *primary = NULL;
if (VIR_ALLOC(def) < 0) { if (VIR_ALLOC(def) < 0) {
@ -7401,9 +7402,10 @@ virDomainVideoDefParseXML(const xmlNodePtr node,
cur = node->children; cur = node->children;
while (cur != NULL) { while (cur != NULL) {
if (cur->type == XML_ELEMENT_NODE) { if (cur->type == XML_ELEMENT_NODE) {
if (!type && !vram && !heads && if (!type && !vram && !ram && !heads &&
xmlStrEqual(cur->name, BAD_CAST "model")) { xmlStrEqual(cur->name, BAD_CAST "model")) {
type = virXMLPropString(cur, "type"); type = virXMLPropString(cur, "type");
ram = virXMLPropString(cur, "ram");
vram = virXMLPropString(cur, "vram"); vram = virXMLPropString(cur, "vram");
heads = virXMLPropString(cur, "heads"); heads = virXMLPropString(cur, "heads");
@ -7431,9 +7433,24 @@ virDomainVideoDefParseXML(const xmlNodePtr node,
} }
} }
if (ram) {
if (def->type != VIR_DOMAIN_VIDEO_TYPE_QXL) {
virReportError(VIR_ERR_XML_ERROR, "%s",
_("ram attribute only supported for type of qxl"));
goto error;
}
if (virStrToLong_ui(ram, NULL, 10, &def->ram) < 0) {
virReportError(VIR_ERR_XML_ERROR,
_("cannot parse video ram '%s'"), ram);
goto error;
}
} else if (def->type == VIR_DOMAIN_VIDEO_TYPE_QXL) {
def->ram = virDomainVideoDefaultRAM(dom, def->type);
}
if (vram) { if (vram) {
if (virStrToLong_ui(vram, NULL, 10, &def->vram) < 0) { if (virStrToLong_ui(vram, NULL, 10, &def->vram) < 0) {
virReportError(VIR_ERR_INTERNAL_ERROR, virReportError(VIR_ERR_XML_ERROR,
_("cannot parse video ram '%s'"), vram); _("cannot parse video ram '%s'"), vram);
goto error; goto error;
} }
@ -7455,6 +7472,7 @@ virDomainVideoDefParseXML(const xmlNodePtr node,
goto error; goto error;
VIR_FREE(type); VIR_FREE(type);
VIR_FREE(ram);
VIR_FREE(vram); VIR_FREE(vram);
VIR_FREE(heads); VIR_FREE(heads);
@ -13384,6 +13402,8 @@ virDomainVideoDefFormat(virBufferPtr buf,
virBufferAddLit(buf, " <video>\n"); virBufferAddLit(buf, " <video>\n");
virBufferAsprintf(buf, " <model type='%s'", virBufferAsprintf(buf, " <model type='%s'",
model); model);
if (def->ram)
virBufferAsprintf(buf, " ram='%u'", def->ram);
if (def->vram) if (def->vram)
virBufferAsprintf(buf, " vram='%u'", def->vram); virBufferAsprintf(buf, " vram='%u'", def->vram);
if (def->heads) if (def->heads)

View File

@ -1,7 +1,7 @@
/* /*
* domain_conf.h: domain XML processing * domain_conf.h: domain XML processing
* *
* Copyright (C) 2006-2012 Red Hat, Inc. * Copyright (C) 2006-2013 Red Hat, Inc.
* Copyright (C) 2006-2008 Daniel P. Berrange * Copyright (C) 2006-2008 Daniel P. Berrange
* *
* This library is free software; you can redistribute it and/or * This library is free software; you can redistribute it and/or
@ -1157,7 +1157,8 @@ struct _virDomainVideoAccelDef {
struct _virDomainVideoDef { struct _virDomainVideoDef {
int type; int type;
unsigned int vram; unsigned int ram; /* kibibytes (multiples of 1024) */
unsigned int vram; /* kibibytes (multiples of 1024) */
unsigned int heads; unsigned int heads;
bool primary; bool primary;
virDomainVideoAccelDefPtr accel; virDomainVideoAccelDefPtr accel;

View File

@ -3558,11 +3558,20 @@ qemuBuildDeviceVideoStr(virDomainVideoDefPtr video,
if (video->type == VIR_DOMAIN_VIDEO_TYPE_QXL) { if (video->type == VIR_DOMAIN_VIDEO_TYPE_QXL) {
if (video->vram > (UINT_MAX / 1024)) { if (video->vram > (UINT_MAX / 1024)) {
virReportError(VIR_ERR_CONFIG_UNSUPPORTED, virReportError(VIR_ERR_OVERFLOW,
_("value for 'vram' must be less than '%u'"), _("value for 'vram' must be less than '%u'"),
UINT_MAX / 1024); UINT_MAX / 1024);
goto error; goto error;
} }
if (video->ram > (UINT_MAX / 1024)) {
virReportError(VIR_ERR_OVERFLOW,
_("value for 'ram' must be less than '%u'"),
UINT_MAX / 1024);
goto error;
}
/* QEMU accepts bytes for ram_size. */
virBufferAsprintf(&buf, ",ram_size=%u", video->ram * 1024);
/* QEMU accepts bytes for vram_size. */ /* QEMU accepts bytes for vram_size. */
virBufferAsprintf(&buf, ",vram_size=%u", video->vram * 1024); virBufferAsprintf(&buf, ",vram_size=%u", video->vram * 1024);
@ -6568,24 +6577,36 @@ qemuBuildCommandLine(virConnectPtr conn,
virCommandAddArgList(cmd, "-vga", vgastr, NULL); virCommandAddArgList(cmd, "-vga", vgastr, NULL);
if (def->videos[0]->type == VIR_DOMAIN_VIDEO_TYPE_QXL) { if (def->videos[0]->type == VIR_DOMAIN_VIDEO_TYPE_QXL &&
if (def->videos[0]->vram && (def->videos[0]->vram || def->videos[0]->ram) &&
qemuCapsGet(caps, QEMU_CAPS_DEVICE)) { qemuCapsGet(caps, QEMU_CAPS_DEVICE)) {
if (def->videos[0]->vram > (UINT_MAX / 1024)) { const char *dev = (qemuCapsGet(caps, QEMU_CAPS_DEVICE_QXL_VGA)
virReportError(VIR_ERR_CONFIG_UNSUPPORTED, ? "qxl-vga" : "qxl");
_("value for 'vram' must be less than '%u'"), int ram = def->videos[0]->ram;
UINT_MAX / 1024); int vram = def->videos[0]->vram;
goto error;
}
if (vram > (UINT_MAX / 1024)) {
virReportError(VIR_ERR_OVERFLOW,
_("value for 'vram' must be less than '%u'"),
UINT_MAX / 1024);
goto error;
}
if (ram > (UINT_MAX / 1024)) {
virReportError(VIR_ERR_OVERFLOW,
_("value for 'ram' must be less than '%u'"),
UINT_MAX / 1024);
goto error;
}
if (ram) {
virCommandAddArg(cmd, "-global"); virCommandAddArg(cmd, "-global");
virCommandAddArgFormat(cmd, "%s.ram_size=%u",
if (qemuCapsGet(caps, QEMU_CAPS_DEVICE_QXL_VGA)) dev, ram * 1024);
virCommandAddArgFormat(cmd, "qxl-vga.vram_size=%u", }
def->videos[0]->vram * 1024); if (vram) {
else virCommandAddArg(cmd, "-global");
virCommandAddArgFormat(cmd, "qxl.vram_size=%u", virCommandAddArgFormat(cmd, "%s.vram_size=%u",
def->videos[0]->vram * 1024); dev, vram * 1024);
} }
} }
} }
@ -9247,6 +9268,8 @@ virDomainDefPtr qemuParseCommandLine(virCapsPtr caps,
else else
vid->type = video; vid->type = video;
vid->vram = virDomainVideoDefaultRAM(def, vid->type); vid->vram = virDomainVideoDefaultRAM(def, vid->type);
vid->ram = vid->type == VIR_DOMAIN_VIDEO_TYPE_QXL ?
virDomainVideoDefaultRAM(def, vid->type) : 0;
vid->heads = 1; vid->heads = 1;
if (VIR_REALLOC_N(def->videos, def->nvideos+1) < 0) { if (VIR_REALLOC_N(def->videos, def->nvideos+1) < 0) {

View File

@ -3,7 +3,9 @@ LC_ALL=C PATH=/bin HOME=/home/test USER=test LOGNAME=test QEMU_AUDIO_DRV=spice \
unix:/tmp/test-monitor,server,nowait -no-acpi -boot c -usb -hda \ unix:/tmp/test-monitor,server,nowait -no-acpi -boot c -usb -hda \
/dev/HostVG/QEMUGuest1 -spice port=5903,tls-port=5904,addr=127.0.0.1,\ /dev/HostVG/QEMUGuest1 -spice port=5903,tls-port=5904,addr=127.0.0.1,\
x509-dir=/etc/pki/libvirt-spice,\ x509-dir=/etc/pki/libvirt-spice,\
image-compression=auto_glz,jpeg-wan-compression=auto,zlib-glz-wan-compression=auto,\ image-compression=auto_glz,jpeg-wan-compression=auto,\
zlib-glz-wan-compression=auto,\
playback-compression=on,streaming-video=filter -vga \ playback-compression=on,streaming-video=filter -vga \
qxl -global qxl.vram_size=18874368 -device qxl,id=video1,vram_size=33554432,bus=pci.0,addr=0x4 \ qxl -global qxl.ram_size=67108864 -global qxl.vram_size=18874368 \
-device qxl,id=video1,ram_size=67108864,vram_size=33554432,bus=pci.0,addr=0x4 \
-device virtio-balloon-pci,id=balloon0,bus=pci.0,addr=0x3 -device virtio-balloon-pci,id=balloon0,bus=pci.0,addr=0x3

View File

@ -31,10 +31,10 @@
<streaming mode='filter'/> <streaming mode='filter'/>
</graphics> </graphics>
<video> <video>
<model type='qxl' vram='18432' heads='1'/> <model type='qxl' ram='65536' vram='18432' heads='1'/>
</video> </video>
<video> <video>
<model type='qxl' vram='32768' heads='1'/> <model type='qxl' ram='65536' vram='32768' heads='1'/>
</video> </video>
<memballoon model='virtio'/> <memballoon model='virtio'/>
</devices> </devices>

View File

@ -3,5 +3,6 @@ LC_ALL=C PATH=/bin HOME=/home/test USER=test LOGNAME=test QEMU_AUDIO_DRV=spice \
unix:/tmp/test-monitor,server,nowait -no-acpi -boot c -usb -hda \ unix:/tmp/test-monitor,server,nowait -no-acpi -boot c -usb -hda \
/dev/HostVG/QEMUGuest1 -spice port=5903,tls-port=5904,addr=127.0.0.1,\ /dev/HostVG/QEMUGuest1 -spice port=5903,tls-port=5904,addr=127.0.0.1,\
x509-dir=/etc/pki/libvirt-spice,tls-channel=main,plaintext-channel=inputs -vga \ x509-dir=/etc/pki/libvirt-spice,tls-channel=main,plaintext-channel=inputs -vga \
qxl -global qxl-vga.vram_size=33554432 -device qxl,id=video1,vram_size=67108864,bus=pci.0,addr=0x4 \ qxl -global qxl-vga.ram_size=67108864 -global qxl-vga.vram_size=33554432 \
-device qxl,id=video1,ram_size=67108864,vram_size=67108864,bus=pci.0,addr=0x4 \
-device virtio-balloon-pci,id=balloon0,bus=pci.0,addr=0x3 -device virtio-balloon-pci,id=balloon0,bus=pci.0,addr=0x3

View File

@ -28,10 +28,10 @@
<channel name='inputs' mode='insecure'/> <channel name='inputs' mode='insecure'/>
</graphics> </graphics>
<video> <video>
<model type='qxl' vram='32768' heads='1'/> <model type='qxl' ram='65536' vram='32768' heads='1'/>
</video> </video>
<video> <video>
<model type='qxl' vram='65536' heads='1'/> <model type='qxl' ram='65536' vram='65536' heads='1'/>
</video> </video>
<memballoon model='virtio'/> <memballoon model='virtio'/>
</devices> </devices>

View File

@ -5,5 +5,6 @@ unix:/tmp/test-monitor,server,nowait -no-acpi -boot c -usb -hda \
x509-dir=/etc/pki/libvirt-spice,tls-channel=default,tls-channel=main,plaintext-channel=inputs,\ x509-dir=/etc/pki/libvirt-spice,tls-channel=default,tls-channel=main,plaintext-channel=inputs,\
image-compression=auto_glz,jpeg-wan-compression=auto,zlib-glz-wan-compression=auto,\ image-compression=auto_glz,jpeg-wan-compression=auto,zlib-glz-wan-compression=auto,\
playback-compression=on,streaming-video=filter,disable-copy-paste -vga \ playback-compression=on,streaming-video=filter,disable-copy-paste -vga \
qxl -global qxl.vram_size=18874368 -device qxl,id=video1,vram_size=33554432,bus=pci.0,addr=0x4 \ qxl -global qxl.ram_size=67108864 -global qxl.vram_size=18874368 \
-device qxl,id=video1,ram_size=67108864,vram_size=33554432,bus=pci.0,addr=0x4 \
-device virtio-balloon-pci,id=balloon0,bus=pci.0,addr=0x3 -device virtio-balloon-pci,id=balloon0,bus=pci.0,addr=0x3

View File

@ -34,10 +34,10 @@
<clipboard copypaste='no'/> <clipboard copypaste='no'/>
</graphics> </graphics>
<video> <video>
<model type='qxl' vram='18432' heads='1'/> <model type='qxl' ram='65536' vram='18432' heads='1'/>
</video> </video>
<video> <video>
<model type='qxl' vram='32768' heads='1'/> <model type='qxl' ram='65536' vram='32768' heads='1'/>
</video> </video>
<memballoon model='virtio'/> <memballoon model='virtio'/>
</devices> </devices>

View File

@ -3,7 +3,7 @@ LC_ALL=C PATH=/bin HOME=/home/test USER=test LOGNAME=test QEMU_AUDIO_DRV=none \
-monitor unix:/tmp/test-monitor,server,nowait -no-acpi -boot c \ -monitor unix:/tmp/test-monitor,server,nowait -no-acpi -boot c \
-usb \ -usb \
-hda /var/lib/libvirt/images/QEMUGuest1 -vnc 127.0.0.1:-5900 \ -hda /var/lib/libvirt/images/QEMUGuest1 -vnc 127.0.0.1:-5900 \
-device qxl-vga,id=video0,vram_size=67108864,bus=pci.0,addr=0x3 \ -device qxl-vga,id=video0,ram_size=67108864,vram_size=67108864,bus=pci.0,addr=0x3 \
-device qxl,id=video1,vram_size=67108864,bus=pci.0,addr=0x4 \ -device qxl,id=video1,ram_size=67108864,vram_size=67108864,bus=pci.0,addr=0x4 \
-device qxl,id=video2,vram_size=67108864,bus=pci.0,addr=0x5 \ -device qxl,id=video2,ram_size=67108864,vram_size=67108864,bus=pci.0,addr=0x5 \
-device virtio-balloon-pci,id=balloon0,bus=pci.0,addr=0x2 -device virtio-balloon-pci,id=balloon0,bus=pci.0,addr=0x2