提交 3801831c 编写于 作者: L Laine Stump

qemu: add "romfile" support to specify device boot ROM

This patch addresses: https://bugzilla.redhat.com/show_bug.cgi?id=781562

Along with the "rombar" option that controls whether or not a boot rom
is made visible to the guest, qemu also has a "romfile" option that
allows specifying a binary file to present as the ROM BIOS of any
emulated or passthrough PCI device. This patch adds support for
specifying romfile to both passthrough PCI devices, and emulated
network devices that attach to the guest's PCI bus (just about
everything other than ne2k_isa).

One example of the usefulness of this option is described in the
bugzilla report: 82576 sriov network adapters don't provide a ROM BIOS
for the cards virtual functions (VF), but an image of such a ROM is
available, and with this ROM visible to the guest, it can PXE boot.

In libvirt's xml, the new option is configured like this:

   <hostdev>
     ...
     <rom file='/etc/fake/boot.bin'/>
     ...
   </hostdev

(similarly for <interface>).
上级 3284ac04
...@@ -1741,7 +1741,7 @@ ...@@ -1741,7 +1741,7 @@
&lt;address bus='0x06' slot='0x02' function='0x0'/&gt; &lt;address bus='0x06' slot='0x02' function='0x0'/&gt;
&lt;/source&gt; &lt;/source&gt;
&lt;boot order='1'/&gt; &lt;boot order='1'/&gt;
&lt;rom bar='off'/&gt; &lt;rom bar='on' file='/etc/fake/boot.bin'/&gt;
&lt;/hostdev&gt; &lt;/hostdev&gt;
&lt;/devices&gt; &lt;/devices&gt;
...</pre> ...</pre>
...@@ -1783,7 +1783,7 @@ ...@@ -1783,7 +1783,7 @@
<span class="since">Since 0.8.8</span></dd> <span class="since">Since 0.8.8</span></dd>
<dt><code>rom</code></dt> <dt><code>rom</code></dt>
<dd>The <code>rom</code> element is used to change how a PCI <dd>The <code>rom</code> element is used to change how a PCI
device's ROM is presented to the guest. The <code>bar</code> device's ROM is presented to the guest. The optional <code>bar</code>
attribute can be set to "on" or "off", and determines whether attribute can be set to "on" or "off", and determines whether
or not the device's ROM will be visible in the guest's memory or not the device's ROM will be visible in the guest's memory
map. (In PCI documentation, the "rombar" setting controls the map. (In PCI documentation, the "rombar" setting controls the
...@@ -1791,7 +1791,13 @@ ...@@ -1791,7 +1791,13 @@
bar is specified, the qemu default will be used (older bar is specified, the qemu default will be used (older
versions of qemu used a default of "off", while newer qemus versions of qemu used a default of "off", while newer qemus
have a default of "on"). <span class="since">Since have a default of "on"). <span class="since">Since
0.9.7</span> 0.9.7 (QEMU and KVM only)</span>. The optional
<code>file</code> attribute is used to point to a binary file
to be presented to the guest as the device's ROM BIOS. This
can be useful, for example, to provide a PXE boot ROM for a
virtual function of an sr-iov capable ethernet device (which
has no boot ROMs for the VFs).
<span class="since">Since 0.9.10 (QEMU and KVM only)</span>.
</dd> </dd>
<dt><code>address</code></dt> <dt><code>address</code></dt>
<dd>The <code>address</code> element for USB devices has a <dd>The <code>address</code> element for USB devices has a
...@@ -2492,7 +2498,7 @@ qemu-kvm -net nic,model=? /dev/null ...@@ -2492,7 +2498,7 @@ qemu-kvm -net nic,model=? /dev/null
&lt;interface type='network'&gt; &lt;interface type='network'&gt;
&lt;source network='default'/&gt; &lt;source network='default'/&gt;
&lt;target dev='vnet1'/&gt; &lt;target dev='vnet1'/&gt;
<b>&lt;rom bar='off'/&gt;</b> <b>&lt;rom bar='on' file='/etc/fake/boot.bin'/&gt;</b>
&lt;/interface&gt; &lt;/interface&gt;
&lt;/devices&gt; &lt;/devices&gt;
...</pre> ...</pre>
...@@ -2506,8 +2512,12 @@ qemu-kvm -net nic,model=? /dev/null ...@@ -2506,8 +2512,12 @@ qemu-kvm -net nic,model=? /dev/null
presence of the Base Address Register for the ROM). If no rom presence of the Base Address Register for the ROM). If no rom
bar is specified, the qemu default will be used (older bar is specified, the qemu default will be used (older
versions of qemu used a default of "off", while newer qemus versions of qemu used a default of "off", while newer qemus
have a default of "on"). <span class="since">Since have a default of "on").
0.9.10 (QEMU and KVM only)</span> The optional <code>file</code> attribute is used to point to a
binary file to be presented to the guest as the device's ROM
BIOS. This can be useful to provide an alternative boot ROM for a
network device.
<span class="since">Since 0.9.10 (QEMU and KVM only)</span>.
</p> </p>
<h5><a name="elementQoS">Quality of service</a></h5> <h5><a name="elementQoS">Quality of service</a></h5>
......
...@@ -2818,12 +2818,19 @@ ...@@ -2818,12 +2818,19 @@
<define name="rom"> <define name="rom">
<element name="rom"> <element name="rom">
<attribute name="bar"> <optional>
<choice> <attribute name="bar">
<value>on</value> <choice>
<value>off</value> <value>on</value>
</choice> <value>off</value>
</attribute> </choice>
</attribute>
</optional>
<optional>
<attribute name="file">
<ref name="absFilePath"/>
</attribute>
</optional>
<empty/> <empty/>
</element> </element>
</define> </define>
......
...@@ -1812,6 +1812,7 @@ void virDomainDeviceInfoClear(virDomainDeviceInfoPtr info) ...@@ -1812,6 +1812,7 @@ void virDomainDeviceInfoClear(virDomainDeviceInfoPtr info)
} }
memset(&info->addr, 0, sizeof(info->addr)); memset(&info->addr, 0, sizeof(info->addr));
info->type = VIR_DOMAIN_DEVICE_ADDRESS_TYPE_NONE; info->type = VIR_DOMAIN_DEVICE_ADDRESS_TYPE_NONE;
VIR_FREE(info->romfile);
} }
...@@ -1924,16 +1925,25 @@ virDomainDeviceInfoFormat(virBufferPtr buf, ...@@ -1924,16 +1925,25 @@ virDomainDeviceInfoFormat(virBufferPtr buf,
info->master.usb.startport); info->master.usb.startport);
} }
if ((flags & VIR_DOMAIN_XML_INTERNAL_ALLOW_ROM) && info->rombar) { if ((flags & VIR_DOMAIN_XML_INTERNAL_ALLOW_ROM) &&
const char *rombar (info->rombar || info->romfile)) {
= virDomainPciRombarModeTypeToString(info->rombar);
if (!rombar) { virBufferAddLit(buf, " <rom");
virDomainReportError(VIR_ERR_INTERNAL_ERROR, if (info->rombar) {
_("unexpected rom bar value %d"),
info->rombar); const char *rombar = virDomainPciRombarModeTypeToString(info->rombar);
return -1;
if (!rombar) {
virDomainReportError(VIR_ERR_INTERNAL_ERROR,
_("unexpected rom bar value %d"),
info->rombar);
return -1;
}
virBufferAsprintf(buf, " bar='%s'", rombar);
} }
virBufferAsprintf(buf, " <rom bar='%s'/>\n", rombar); if (info->romfile)
virBufferAsprintf(buf, " file='%s'", info->romfile);
virBufferAddLit(buf, "/>\n");
} }
if (info->type == VIR_DOMAIN_DEVICE_ADDRESS_TYPE_NONE) if (info->type == VIR_DOMAIN_DEVICE_ADDRESS_TYPE_NONE)
...@@ -2390,18 +2400,15 @@ virDomainDeviceInfoParseXML(xmlNodePtr node, ...@@ -2390,18 +2400,15 @@ virDomainDeviceInfoParseXML(xmlNodePtr node,
if (rom) { if (rom) {
char *rombar = virXMLPropString(rom, "bar"); char *rombar = virXMLPropString(rom, "bar");
if (!rombar) { if (rombar &&
virDomainReportError(VIR_ERR_XML_ERROR, ((info->rombar = virDomainPciRombarModeTypeFromString(rombar)) <= 0)) {
"%s", _("missing rom bar attribute"));
goto cleanup;
}
if ((info->rombar = virDomainPciRombarModeTypeFromString(rombar)) <= 0) {
virDomainReportError(VIR_ERR_CONFIG_UNSUPPORTED, virDomainReportError(VIR_ERR_CONFIG_UNSUPPORTED,
_("unknown rom bar value '%s'"), rombar); _("unknown rom bar value '%s'"), rombar);
VIR_FREE(rombar); VIR_FREE(rombar);
goto cleanup; goto cleanup;
} }
VIR_FREE(rombar); VIR_FREE(rombar);
info->romfile = virXMLPropString(rom, "file");
} }
if (!address) if (!address)
......
...@@ -170,6 +170,7 @@ struct _virDomainDeviceInfo { ...@@ -170,6 +170,7 @@ struct _virDomainDeviceInfo {
/* rombar is only used for pci hostdev devices, and bootIndex only /* rombar is only used for pci hostdev devices, and bootIndex only
* for disk, network interface, and hostdev devices */ * for disk, network interface, and hostdev devices */
int rombar; /* enum virDomainPciRombarMode */ int rombar; /* enum virDomainPciRombarMode */
char *romfile;
int bootIndex; int bootIndex;
}; };
......
...@@ -1519,15 +1519,15 @@ qemuBuildRomStr(virBufferPtr buf, ...@@ -1519,15 +1519,15 @@ qemuBuildRomStr(virBufferPtr buf,
virDomainDeviceInfoPtr info, virDomainDeviceInfoPtr info,
virBitmapPtr qemuCaps) virBitmapPtr qemuCaps)
{ {
if (info->rombar) { if (info->rombar || info->romfile) {
if (info->type != VIR_DOMAIN_DEVICE_ADDRESS_TYPE_PCI) { if (info->type != VIR_DOMAIN_DEVICE_ADDRESS_TYPE_PCI) {
qemuReportError(VIR_ERR_CONFIG_UNSUPPORTED, qemuReportError(VIR_ERR_CONFIG_UNSUPPORTED,
"%s", _("rombar is supported only for PCI devices")); "%s", _("rombar and romfile are supported only for PCI devices"));
return -1; return -1;
} }
if (!qemuCapsGet(qemuCaps, QEMU_CAPS_PCI_ROMBAR)) { if (!qemuCapsGet(qemuCaps, QEMU_CAPS_PCI_ROMBAR)) {
qemuReportError(VIR_ERR_CONFIG_UNSUPPORTED, qemuReportError(VIR_ERR_CONFIG_UNSUPPORTED,
"%s", _("rombar not supported in this QEMU binary")); "%s", _("rombar and romfile not supported in this QEMU binary"));
return -1; return -1;
} }
...@@ -1541,6 +1541,8 @@ qemuBuildRomStr(virBufferPtr buf, ...@@ -1541,6 +1541,8 @@ qemuBuildRomStr(virBufferPtr buf,
default: default:
break; break;
} }
if (info->romfile)
virBufferAsprintf(buf, ",romfile=%s", info->romfile);
} }
return 0; return 0;
} }
......
...@@ -4,8 +4,9 @@ unix:/tmp/test-monitor,server,nowait -no-acpi -boot c -hda \ ...@@ -4,8 +4,9 @@ unix:/tmp/test-monitor,server,nowait -no-acpi -boot c -hda \
/dev/HostVG/QEMUGuest2 \ /dev/HostVG/QEMUGuest2 \
-device virtio-net-pci,vlan=0,id=net0,mac=52:54:00:24:a5:9f,bus=pci.0,addr=0x3,rombar=1 \ -device virtio-net-pci,vlan=0,id=net0,mac=52:54:00:24:a5:9f,bus=pci.0,addr=0x3,rombar=1 \
-net user,vlan=0,name=hostnet0 \ -net user,vlan=0,name=hostnet0 \
-device virtio-net-pci,vlan=1,id=net1,mac=52:54:00:24:a5:9e,bus=pci.0,addr=0x4 \ -device virtio-net-pci,vlan=1,id=net1,mac=52:54:00:24:a5:9e,bus=pci.0,addr=0x4,\
-net user,vlan=1,name=hostnet1 \ romfile=/etc/fake/bootrom.bin -net user,vlan=1,name=hostnet1 \
-usb -device pci-assign,host=06:12.5,id=hostdev0,bus=pci.0,addr=0x5,rombar=0 \ -usb -device pci-assign,host=06:12.5,id=hostdev0,bus=pci.0,addr=0x5,rombar=0 \
-device pci-assign,host=06:12.6,id=hostdev1,bus=pci.0,addr=0x6,rombar=1 \ -device pci-assign,host=06:12.6,id=hostdev1,bus=pci.0,addr=0x6,rombar=1,\
romfile=/etc/fake/bootrom.bin \
-device virtio-balloon-pci,id=balloon0,bus=pci.0,addr=0x7 -device virtio-balloon-pci,id=balloon0,bus=pci.0,addr=0x7
...@@ -28,6 +28,7 @@ ...@@ -28,6 +28,7 @@
<interface type='user'> <interface type='user'>
<mac address='52:54:00:24:a5:9e'/> <mac address='52:54:00:24:a5:9e'/>
<model type='virtio'/> <model type='virtio'/>
<rom file='/etc/fake/bootrom.bin'/>
</interface> </interface>
<hostdev mode='subsystem' type='pci' managed='yes'> <hostdev mode='subsystem' type='pci' managed='yes'>
<source> <source>
...@@ -39,7 +40,7 @@ ...@@ -39,7 +40,7 @@
<source> <source>
<address domain='0x0000' bus='0x06' slot='0x12' function='0x6'/> <address domain='0x0000' bus='0x06' slot='0x12' function='0x6'/>
</source> </source>
<rom bar='on'/> <rom bar='on' file='/etc/fake/bootrom.bin'/>
</hostdev> </hostdev>
<memballoon model='virtio'/> <memballoon model='virtio'/>
</devices> </devices>
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册