2012-08-16 16:41:06 +01:00
|
|
|
/*
|
|
|
|
* device_conf.c: device XML handling
|
|
|
|
*
|
conf: allow type='pci' addresses with no address attributes specified
Prior to this, <address type='pci'/> wasn't allowed when parsing
(domain+bus+slot+function needed to be a "valid" PCI address, meaning
that at least one of domain/bus/slot had to be non-0), the RNG
required bus to be specified, and if type was set to PCI when
formatting, domain+bus+slot+function would always be output.
This makes all the address attributes optional during parse and RNG
validation, and suppresses domain+bus+slot+function if domain+bus+slot
are all 0 (NB: if d+b+s are all 0, any value for function is
nonsensical as that will never happen in the real world, and after
the next patch we will always assign a real working address to any
empty PCI address before it is ever output to anywhere).
Note that explicitly setting all attributes to 0 is equivalent to
setting none of them, which is okay, since 0000:00:00 is reserved in
any PCI bus setup, and can't be used anyway.
2016-05-17 14:03:00 -04:00
|
|
|
* Copyright (C) 2006-2016 Red Hat, Inc.
|
2012-08-16 16:41:06 +01:00
|
|
|
*
|
|
|
|
* This library is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
|
|
* License as published by the Free Software Foundation; either
|
|
|
|
* version 2.1 of the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This library is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
* Lesser General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU Lesser General Public
|
2012-09-20 16:30:55 -06:00
|
|
|
* License along with this library. If not, see
|
2012-08-16 16:41:06 +01:00
|
|
|
* <http://www.gnu.org/licenses/>.
|
|
|
|
*
|
|
|
|
* Author: Daniel P. Berrange <berrange@redhat.com>
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <config.h>
|
2012-12-13 18:21:53 +00:00
|
|
|
#include "virerror.h"
|
2012-08-16 16:41:06 +01:00
|
|
|
#include "datatypes.h"
|
2012-12-12 18:06:53 +00:00
|
|
|
#include "viralloc.h"
|
2012-12-13 18:13:21 +00:00
|
|
|
#include "virxml.h"
|
2012-12-13 18:01:25 +00:00
|
|
|
#include "viruuid.h"
|
2012-12-04 12:04:07 +00:00
|
|
|
#include "virbuffer.h"
|
2012-08-16 16:41:06 +01:00
|
|
|
#include "device_conf.h"
|
2013-04-03 12:36:23 +02:00
|
|
|
#include "virstring.h"
|
2012-08-16 16:41:06 +01:00
|
|
|
|
|
|
|
#define VIR_FROM_THIS VIR_FROM_DEVICE
|
|
|
|
|
2017-06-27 08:30:58 +02:00
|
|
|
int
|
|
|
|
virDomainDeviceInfoCopy(virDomainDeviceInfoPtr dst,
|
|
|
|
virDomainDeviceInfoPtr src)
|
|
|
|
{
|
|
|
|
/* Assume that dst is already cleared */
|
|
|
|
|
|
|
|
/* first a shallow copy of *everything* */
|
|
|
|
*dst = *src;
|
|
|
|
|
|
|
|
/* then copy whatever's left */
|
|
|
|
dst->alias = NULL;
|
|
|
|
dst->romfile = NULL;
|
|
|
|
dst->loadparm = NULL;
|
|
|
|
|
|
|
|
if (VIR_STRDUP(dst->alias, src->alias) < 0 ||
|
|
|
|
VIR_STRDUP(dst->romfile, src->romfile) < 0 ||
|
|
|
|
VIR_STRDUP(dst->loadparm, src->loadparm) < 0)
|
|
|
|
return -1;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
virDomainDeviceInfoClear(virDomainDeviceInfoPtr info)
|
|
|
|
{
|
|
|
|
VIR_FREE(info->alias);
|
|
|
|
memset(&info->addr, 0, sizeof(info->addr));
|
|
|
|
info->type = VIR_DOMAIN_DEVICE_ADDRESS_TYPE_NONE;
|
|
|
|
VIR_FREE(info->romfile);
|
|
|
|
VIR_FREE(info->loadparm);
|
|
|
|
info->isolationGroup = 0;
|
|
|
|
info->isolationGroupLocked = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
virDomainDeviceInfoFree(virDomainDeviceInfoPtr info)
|
|
|
|
{
|
|
|
|
if (info) {
|
|
|
|
virDomainDeviceInfoClear(info);
|
|
|
|
VIR_FREE(info);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool
|
|
|
|
virDomainDeviceInfoAddressIsEqual(const virDomainDeviceInfo *a,
|
|
|
|
const virDomainDeviceInfo *b)
|
|
|
|
{
|
|
|
|
if (a->type != b->type)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
switch ((virDomainDeviceAddressType) a->type) {
|
|
|
|
case VIR_DOMAIN_DEVICE_ADDRESS_TYPE_NONE:
|
|
|
|
case VIR_DOMAIN_DEVICE_ADDRESS_TYPE_LAST:
|
|
|
|
/* address types below don't have any specific data */
|
|
|
|
case VIR_DOMAIN_DEVICE_ADDRESS_TYPE_VIRTIO_MMIO:
|
|
|
|
case VIR_DOMAIN_DEVICE_ADDRESS_TYPE_VIRTIO_S390:
|
|
|
|
break;
|
|
|
|
|
|
|
|
case VIR_DOMAIN_DEVICE_ADDRESS_TYPE_PCI:
|
|
|
|
/* the 'multi' field shouldn't be checked */
|
|
|
|
if (a->addr.pci.domain != b->addr.pci.domain ||
|
|
|
|
a->addr.pci.bus != b->addr.pci.bus ||
|
|
|
|
a->addr.pci.slot != b->addr.pci.slot ||
|
|
|
|
a->addr.pci.function != b->addr.pci.function)
|
|
|
|
return false;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case VIR_DOMAIN_DEVICE_ADDRESS_TYPE_DRIVE:
|
|
|
|
if (memcmp(&a->addr.drive, &b->addr.drive, sizeof(a->addr.drive)))
|
|
|
|
return false;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case VIR_DOMAIN_DEVICE_ADDRESS_TYPE_VIRTIO_SERIAL:
|
|
|
|
if (memcmp(&a->addr.vioserial, &b->addr.vioserial, sizeof(a->addr.vioserial)))
|
|
|
|
return false;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case VIR_DOMAIN_DEVICE_ADDRESS_TYPE_CCID:
|
|
|
|
if (memcmp(&a->addr.ccid, &b->addr.ccid, sizeof(a->addr.ccid)))
|
|
|
|
return false;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case VIR_DOMAIN_DEVICE_ADDRESS_TYPE_USB:
|
|
|
|
if (memcmp(&a->addr.usb, &b->addr.usb, sizeof(a->addr.usb)))
|
|
|
|
return false;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case VIR_DOMAIN_DEVICE_ADDRESS_TYPE_SPAPRVIO:
|
|
|
|
if (memcmp(&a->addr.spaprvio, &b->addr.spaprvio, sizeof(a->addr.spaprvio)))
|
|
|
|
return false;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case VIR_DOMAIN_DEVICE_ADDRESS_TYPE_CCW:
|
|
|
|
/* the 'assigned' field denotes that the address was generated */
|
|
|
|
if (a->addr.ccw.cssid != b->addr.ccw.cssid ||
|
|
|
|
a->addr.ccw.ssid != b->addr.ccw.ssid ||
|
|
|
|
a->addr.ccw.devno != b->addr.ccw.devno)
|
|
|
|
return false;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case VIR_DOMAIN_DEVICE_ADDRESS_TYPE_ISA:
|
|
|
|
if (memcmp(&a->addr.isa, &b->addr.isa, sizeof(a->addr.isa)))
|
|
|
|
return false;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case VIR_DOMAIN_DEVICE_ADDRESS_TYPE_DIMM:
|
|
|
|
if (memcmp(&a->addr.dimm, &b->addr.dimm, sizeof(a->addr.dimm)))
|
|
|
|
return false;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2016-04-03 20:16:51 +02:00
|
|
|
int virPCIDeviceAddressIsValid(virPCIDeviceAddressPtr addr,
|
conf: more useful error message when pci function is out of range
If a pci address had a function number out of range, the error message
would be:
Insufficient specification for PCI address
which is logged by virDevicePCIAddressParseXML() after
virDevicePCIAddressIsValid returns a failure.
This patch enhances virDevicePCIAddressIsValid() to optionally report
the error itself (since it is the place that decides which part of the
address is "invalid"), and uses that feature when calling from
virDevicePCIAddressParseXML(), so that the error will be more useful,
e.g.:
Invalid PCI address function=0x8, must be <= 7
Previously, virDevicePCIAddressIsValid didn't check for the
theoretical limits of domain or bus, only for slot or function. While
adding log messages, we also correct that ommission. (The RNG for PCI
addresses already enforces this limit, which by the way means that we
can't add any negative tests for this - as far as I know our
domainschematest has no provisions for passing XML that is supposed to
fail).
Note that virDevicePCIAddressIsValid() can only check against the
absolute maximum attribute values for *any* possible PCI controller,
not for the actual maximums of the specific controller that this
device is attaching to; fortunately there is later more specific
validation for guest-side PCI addresses when building the set of
assigned PCI addresses. For host-side PCI addresses (e.g. for
<hostdev> and for network device pools), we rely on the error that
will be logged when it is found that the device doesn't actually
exist.
This resolves:
https://bugzilla.redhat.com/show_bug.cgi?id=1004596
2015-07-22 11:59:00 -04:00
|
|
|
bool report)
|
2012-08-16 16:41:06 +01:00
|
|
|
{
|
conf: more useful error message when pci function is out of range
If a pci address had a function number out of range, the error message
would be:
Insufficient specification for PCI address
which is logged by virDevicePCIAddressParseXML() after
virDevicePCIAddressIsValid returns a failure.
This patch enhances virDevicePCIAddressIsValid() to optionally report
the error itself (since it is the place that decides which part of the
address is "invalid"), and uses that feature when calling from
virDevicePCIAddressParseXML(), so that the error will be more useful,
e.g.:
Invalid PCI address function=0x8, must be <= 7
Previously, virDevicePCIAddressIsValid didn't check for the
theoretical limits of domain or bus, only for slot or function. While
adding log messages, we also correct that ommission. (The RNG for PCI
addresses already enforces this limit, which by the way means that we
can't add any negative tests for this - as far as I know our
domainschematest has no provisions for passing XML that is supposed to
fail).
Note that virDevicePCIAddressIsValid() can only check against the
absolute maximum attribute values for *any* possible PCI controller,
not for the actual maximums of the specific controller that this
device is attaching to; fortunately there is later more specific
validation for guest-side PCI addresses when building the set of
assigned PCI addresses. For host-side PCI addresses (e.g. for
<hostdev> and for network device pools), we rely on the error that
will be logged when it is found that the device doesn't actually
exist.
This resolves:
https://bugzilla.redhat.com/show_bug.cgi?id=1004596
2015-07-22 11:59:00 -04:00
|
|
|
if (addr->domain > 0xFFFF) {
|
|
|
|
if (report)
|
|
|
|
virReportError(VIR_ERR_XML_ERROR,
|
|
|
|
_("Invalid PCI address domain='0x%x', "
|
|
|
|
"must be <= 0xFFFF"),
|
|
|
|
addr->domain);
|
2012-08-16 16:41:06 +01:00
|
|
|
return 0;
|
conf: more useful error message when pci function is out of range
If a pci address had a function number out of range, the error message
would be:
Insufficient specification for PCI address
which is logged by virDevicePCIAddressParseXML() after
virDevicePCIAddressIsValid returns a failure.
This patch enhances virDevicePCIAddressIsValid() to optionally report
the error itself (since it is the place that decides which part of the
address is "invalid"), and uses that feature when calling from
virDevicePCIAddressParseXML(), so that the error will be more useful,
e.g.:
Invalid PCI address function=0x8, must be <= 7
Previously, virDevicePCIAddressIsValid didn't check for the
theoretical limits of domain or bus, only for slot or function. While
adding log messages, we also correct that ommission. (The RNG for PCI
addresses already enforces this limit, which by the way means that we
can't add any negative tests for this - as far as I know our
domainschematest has no provisions for passing XML that is supposed to
fail).
Note that virDevicePCIAddressIsValid() can only check against the
absolute maximum attribute values for *any* possible PCI controller,
not for the actual maximums of the specific controller that this
device is attaching to; fortunately there is later more specific
validation for guest-side PCI addresses when building the set of
assigned PCI addresses. For host-side PCI addresses (e.g. for
<hostdev> and for network device pools), we rely on the error that
will be logged when it is found that the device doesn't actually
exist.
This resolves:
https://bugzilla.redhat.com/show_bug.cgi?id=1004596
2015-07-22 11:59:00 -04:00
|
|
|
}
|
|
|
|
if (addr->bus > 0xFF) {
|
|
|
|
if (report)
|
|
|
|
virReportError(VIR_ERR_XML_ERROR,
|
|
|
|
_("Invalid PCI address bus='0x%x', "
|
|
|
|
"must be <= 0xFF"),
|
|
|
|
addr->bus);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
if (addr->slot > 0x1F) {
|
|
|
|
if (report)
|
|
|
|
virReportError(VIR_ERR_XML_ERROR,
|
|
|
|
_("Invalid PCI address slot='0x%x', "
|
|
|
|
"must be <= 0x1F"),
|
|
|
|
addr->slot);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
if (addr->function > 7) {
|
|
|
|
if (report)
|
|
|
|
virReportError(VIR_ERR_XML_ERROR,
|
|
|
|
_("Invalid PCI address function=0x%x, "
|
|
|
|
"must be <= 7"),
|
|
|
|
addr->function);
|
|
|
|
return 0;
|
|
|
|
}
|
conf: allow type='pci' addresses with no address attributes specified
Prior to this, <address type='pci'/> wasn't allowed when parsing
(domain+bus+slot+function needed to be a "valid" PCI address, meaning
that at least one of domain/bus/slot had to be non-0), the RNG
required bus to be specified, and if type was set to PCI when
formatting, domain+bus+slot+function would always be output.
This makes all the address attributes optional during parse and RNG
validation, and suppresses domain+bus+slot+function if domain+bus+slot
are all 0 (NB: if d+b+s are all 0, any value for function is
nonsensical as that will never happen in the real world, and after
the next patch we will always assign a real working address to any
empty PCI address before it is ever output to anywhere).
Note that explicitly setting all attributes to 0 is equivalent to
setting none of them, which is okay, since 0000:00:00 is reserved in
any PCI bus setup, and can't be used anyway.
2016-05-17 14:03:00 -04:00
|
|
|
if (virPCIDeviceAddressIsEmpty(addr)) {
|
conf: more useful error message when pci function is out of range
If a pci address had a function number out of range, the error message
would be:
Insufficient specification for PCI address
which is logged by virDevicePCIAddressParseXML() after
virDevicePCIAddressIsValid returns a failure.
This patch enhances virDevicePCIAddressIsValid() to optionally report
the error itself (since it is the place that decides which part of the
address is "invalid"), and uses that feature when calling from
virDevicePCIAddressParseXML(), so that the error will be more useful,
e.g.:
Invalid PCI address function=0x8, must be <= 7
Previously, virDevicePCIAddressIsValid didn't check for the
theoretical limits of domain or bus, only for slot or function. While
adding log messages, we also correct that ommission. (The RNG for PCI
addresses already enforces this limit, which by the way means that we
can't add any negative tests for this - as far as I know our
domainschematest has no provisions for passing XML that is supposed to
fail).
Note that virDevicePCIAddressIsValid() can only check against the
absolute maximum attribute values for *any* possible PCI controller,
not for the actual maximums of the specific controller that this
device is attaching to; fortunately there is later more specific
validation for guest-side PCI addresses when building the set of
assigned PCI addresses. For host-side PCI addresses (e.g. for
<hostdev> and for network device pools), we rely on the error that
will be logged when it is found that the device doesn't actually
exist.
This resolves:
https://bugzilla.redhat.com/show_bug.cgi?id=1004596
2015-07-22 11:59:00 -04:00
|
|
|
if (report)
|
|
|
|
virReportError(VIR_ERR_XML_ERROR, "%s",
|
|
|
|
_("Invalid PCI address 0000:00:00, at least "
|
|
|
|
"one of domain, bus, or slot must be > 0"));
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
return 1;
|
2012-08-16 16:41:06 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int
|
2016-04-03 20:16:51 +02:00
|
|
|
virPCIDeviceAddressParseXML(xmlNodePtr node,
|
|
|
|
virPCIDeviceAddressPtr addr)
|
2012-08-16 16:41:06 +01:00
|
|
|
{
|
|
|
|
char *domain, *slot, *bus, *function, *multi;
|
|
|
|
int ret = -1;
|
|
|
|
|
|
|
|
memset(addr, 0, sizeof(*addr));
|
|
|
|
|
|
|
|
domain = virXMLPropString(node, "domain");
|
|
|
|
bus = virXMLPropString(node, "bus");
|
|
|
|
slot = virXMLPropString(node, "slot");
|
|
|
|
function = virXMLPropString(node, "function");
|
|
|
|
multi = virXMLPropString(node, "multifunction");
|
|
|
|
|
|
|
|
if (domain &&
|
2014-12-08 16:27:26 +08:00
|
|
|
virStrToLong_uip(domain, NULL, 0, &addr->domain) < 0) {
|
2012-08-16 16:41:06 +01:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("Cannot parse <address> 'domain' attribute"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (bus &&
|
2014-12-08 16:27:26 +08:00
|
|
|
virStrToLong_uip(bus, NULL, 0, &addr->bus) < 0) {
|
2012-08-16 16:41:06 +01:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("Cannot parse <address> 'bus' attribute"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (slot &&
|
2014-12-08 16:27:26 +08:00
|
|
|
virStrToLong_uip(slot, NULL, 0, &addr->slot) < 0) {
|
2012-08-16 16:41:06 +01:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("Cannot parse <address> 'slot' attribute"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (function &&
|
2014-12-08 16:27:26 +08:00
|
|
|
virStrToLong_uip(function, NULL, 0, &addr->function) < 0) {
|
2012-08-16 16:41:06 +01:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("Cannot parse <address> 'function' attribute"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (multi &&
|
2014-06-27 17:18:53 +02:00
|
|
|
((addr->multi = virTristateSwitchTypeFromString(multi)) <= 0)) {
|
2012-08-16 16:41:06 +01:00
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED,
|
|
|
|
_("Unknown value '%s' for <address> 'multifunction' attribute"),
|
|
|
|
multi);
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
}
|
conf: allow type='pci' addresses with no address attributes specified
Prior to this, <address type='pci'/> wasn't allowed when parsing
(domain+bus+slot+function needed to be a "valid" PCI address, meaning
that at least one of domain/bus/slot had to be non-0), the RNG
required bus to be specified, and if type was set to PCI when
formatting, domain+bus+slot+function would always be output.
This makes all the address attributes optional during parse and RNG
validation, and suppresses domain+bus+slot+function if domain+bus+slot
are all 0 (NB: if d+b+s are all 0, any value for function is
nonsensical as that will never happen in the real world, and after
the next patch we will always assign a real working address to any
empty PCI address before it is ever output to anywhere).
Note that explicitly setting all attributes to 0 is equivalent to
setting none of them, which is okay, since 0000:00:00 is reserved in
any PCI bus setup, and can't be used anyway.
2016-05-17 14:03:00 -04:00
|
|
|
if (!virPCIDeviceAddressIsEmpty(addr) && !virPCIDeviceAddressIsValid(addr, true))
|
2012-08-16 16:41:06 +01:00
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
ret = 0;
|
|
|
|
|
2014-03-25 07:48:31 +01:00
|
|
|
cleanup:
|
2012-08-16 16:41:06 +01:00
|
|
|
VIR_FREE(domain);
|
|
|
|
VIR_FREE(bus);
|
|
|
|
VIR_FREE(slot);
|
|
|
|
VIR_FREE(function);
|
|
|
|
VIR_FREE(multi);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
2016-04-03 20:16:51 +02:00
|
|
|
virPCIDeviceAddressFormat(virBufferPtr buf,
|
|
|
|
virPCIDeviceAddress addr,
|
2012-08-16 16:41:06 +01:00
|
|
|
bool includeTypeInAddr)
|
|
|
|
{
|
|
|
|
virBufferAsprintf(buf, "<address %sdomain='0x%.4x' bus='0x%.2x' "
|
|
|
|
"slot='0x%.2x' function='0x%.1x'/>\n",
|
|
|
|
includeTypeInAddr ? "type='pci' " : "",
|
|
|
|
addr.domain,
|
|
|
|
addr.bus,
|
|
|
|
addr.slot,
|
|
|
|
addr.function);
|
|
|
|
return 0;
|
|
|
|
}
|
2012-08-16 16:42:14 +01:00
|
|
|
|
2012-10-11 12:34:14 -04:00
|
|
|
bool
|
2016-04-03 20:16:51 +02:00
|
|
|
virPCIDeviceAddressEqual(virPCIDeviceAddress *addr1,
|
|
|
|
virPCIDeviceAddress *addr2)
|
2012-08-16 16:42:14 +01:00
|
|
|
{
|
2012-10-11 12:34:14 -04:00
|
|
|
if (addr1->domain == addr2->domain &&
|
|
|
|
addr1->bus == addr2->bus &&
|
|
|
|
addr1->slot == addr2->slot &&
|
|
|
|
addr1->function == addr2->function) {
|
|
|
|
return true;
|
2012-08-16 16:42:14 +01:00
|
|
|
}
|
2012-10-11 12:34:14 -04:00
|
|
|
return false;
|
2012-08-16 16:42:14 +01:00
|
|
|
}
|
2014-05-07 14:21:35 +02:00
|
|
|
|
|
|
|
int
|
|
|
|
virInterfaceLinkParseXML(xmlNodePtr node,
|
2016-06-13 13:06:15 -04:00
|
|
|
virNetDevIfLinkPtr lnk)
|
2014-05-07 14:21:35 +02:00
|
|
|
{
|
|
|
|
int ret = -1;
|
|
|
|
char *stateStr, *speedStr;
|
|
|
|
int state;
|
|
|
|
|
|
|
|
stateStr = virXMLPropString(node, "state");
|
|
|
|
speedStr = virXMLPropString(node, "speed");
|
|
|
|
|
|
|
|
if (stateStr) {
|
2016-06-13 13:06:15 -04:00
|
|
|
if ((state = virNetDevIfStateTypeFromString(stateStr)) < 0) {
|
2014-05-07 14:21:35 +02:00
|
|
|
virReportError(VIR_ERR_XML_ERROR,
|
|
|
|
_("unknown link state: %s"),
|
|
|
|
stateStr);
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
lnk->state = state;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (speedStr &&
|
|
|
|
virStrToLong_ui(speedStr, NULL, 10, &lnk->speed) < 0) {
|
|
|
|
virReportError(VIR_ERR_XML_ERROR,
|
|
|
|
_("Unable to parse link speed: %s"),
|
|
|
|
speedStr);
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = 0;
|
|
|
|
cleanup:
|
|
|
|
VIR_FREE(stateStr);
|
|
|
|
VIR_FREE(speedStr);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
virInterfaceLinkFormat(virBufferPtr buf,
|
2016-06-13 13:06:15 -04:00
|
|
|
const virNetDevIfLink *lnk)
|
2014-05-07 14:21:35 +02:00
|
|
|
{
|
|
|
|
if (!lnk->speed && !lnk->state) {
|
|
|
|
/* If there's nothing to format, return early. */
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
virBufferAddLit(buf, "<link");
|
|
|
|
if (lnk->speed)
|
|
|
|
virBufferAsprintf(buf, " speed='%u'", lnk->speed);
|
|
|
|
if (lnk->state)
|
|
|
|
virBufferAsprintf(buf, " state='%s'",
|
2016-06-13 13:06:15 -04:00
|
|
|
virNetDevIfStateTypeToString(lnk->state));
|
2014-05-07 14:21:35 +02:00
|
|
|
virBufferAddLit(buf, "/>\n");
|
|
|
|
return 0;
|
|
|
|
}
|