mirror of
https://gitlab.com/libvirt/libvirt.git
synced 2025-01-09 22:45:21 +00:00
Support bond interfaces attached to bridges in interface xml.
This was accomplished in xml parsing by doing away with the stripped-down virInterfaceBareDef object, and just always using virInterfaceDef, but with restrictions in certain places (eg, the type of subordinate interface allowed in parsing depends on the parent interface). xml formatting was similarly adjusted. In addition, the formatting functions keep track of the level of interface nesting, and insert extra leading spaces on each line accordingly (using %*s). The only change in formatted xml from previous (aside frmo supporting new combinations of interface types) is that the subordinate ethernet interfaces take up 2 lines rather than one, eg: <interface type='ethernet' name='eth0'> </interface> instead of: <interface type='ethernet' name='eth0'/>
This commit is contained in:
parent
86304e35a3
commit
0022995555
@ -39,20 +39,17 @@ VIR_ENUM_IMPL(virInterface,
|
|||||||
VIR_INTERFACE_TYPE_LAST,
|
VIR_INTERFACE_TYPE_LAST,
|
||||||
"ethernet", "bridge", "bond", "vlan" )
|
"ethernet", "bridge", "bond", "vlan" )
|
||||||
|
|
||||||
|
static virInterfaceDefPtr
|
||||||
|
virInterfaceDefParseXML(virConnectPtr conn,
|
||||||
|
xmlXPathContextPtr ctxt, int parentIfType);
|
||||||
|
static int
|
||||||
|
virInterfaceDefDevFormat(virConnectPtr conn, virBufferPtr buf,
|
||||||
|
const virInterfaceDefPtr def, int level);
|
||||||
|
|
||||||
#define virInterfaceReportError(conn, code, fmt...) \
|
#define virInterfaceReportError(conn, code, fmt...) \
|
||||||
virReportErrorHelper(conn, VIR_FROM_INTERFACE, code, __FILE__, \
|
virReportErrorHelper(conn, VIR_FROM_INTERFACE, code, __FILE__, \
|
||||||
__FUNCTION__, __LINE__, fmt)
|
__FUNCTION__, __LINE__, fmt)
|
||||||
|
|
||||||
static
|
|
||||||
void virInterfaceBareDefFree(virInterfaceBareDefPtr def) {
|
|
||||||
if (def == NULL)
|
|
||||||
return;
|
|
||||||
VIR_FREE(def->name);
|
|
||||||
VIR_FREE(def->mac_or_tag);
|
|
||||||
VIR_FREE(def->devname);
|
|
||||||
VIR_FREE(def);
|
|
||||||
}
|
|
||||||
|
|
||||||
static
|
static
|
||||||
void virInterfaceIpDefFree(virInterfaceIpDefPtr def) {
|
void virInterfaceIpDefFree(virInterfaceIpDefPtr def) {
|
||||||
if (def == NULL)
|
if (def == NULL)
|
||||||
@ -90,7 +87,7 @@ void virInterfaceDefFree(virInterfaceDefPtr def)
|
|||||||
case VIR_INTERFACE_TYPE_BRIDGE:
|
case VIR_INTERFACE_TYPE_BRIDGE:
|
||||||
for (i = 0;i < def->data.bridge.nbItf;i++) {
|
for (i = 0;i < def->data.bridge.nbItf;i++) {
|
||||||
if (def->data.bridge.itf[i] != NULL)
|
if (def->data.bridge.itf[i] != NULL)
|
||||||
virInterfaceBareDefFree(def->data.bridge.itf[i]);
|
virInterfaceDefFree(def->data.bridge.itf[i]);
|
||||||
else
|
else
|
||||||
break; /* to cope with half parsed data on errors */
|
break; /* to cope with half parsed data on errors */
|
||||||
}
|
}
|
||||||
@ -100,7 +97,7 @@ void virInterfaceDefFree(virInterfaceDefPtr def)
|
|||||||
VIR_FREE(def->data.bond.target);
|
VIR_FREE(def->data.bond.target);
|
||||||
for (i = 0;i < def->data.bond.nbItf;i++) {
|
for (i = 0;i < def->data.bond.nbItf;i++) {
|
||||||
if (def->data.bond.itf[i] != NULL)
|
if (def->data.bond.itf[i] != NULL)
|
||||||
virInterfaceBareDefFree(def->data.bond.itf[i]);
|
virInterfaceDefFree(def->data.bond.itf[i]);
|
||||||
else
|
else
|
||||||
break; /* to cope with half parsed data on errors */
|
break; /* to cope with half parsed data on errors */
|
||||||
}
|
}
|
||||||
@ -121,11 +118,9 @@ void virInterfaceDefFree(virInterfaceDefPtr def)
|
|||||||
}
|
}
|
||||||
|
|
||||||
static int
|
static int
|
||||||
virInterfaceDefParseBasicAttrs(virConnectPtr conn, virInterfaceDefPtr def,
|
virInterfaceDefParseName(virConnectPtr conn, virInterfaceDefPtr def,
|
||||||
xmlXPathContextPtr ctxt) {
|
xmlXPathContextPtr ctxt) {
|
||||||
char *tmp;
|
char *tmp;
|
||||||
unsigned long mtu;
|
|
||||||
int ret;
|
|
||||||
|
|
||||||
tmp = virXPathString(conn, "string(./@name)", ctxt);
|
tmp = virXPathString(conn, "string(./@name)", ctxt);
|
||||||
if (tmp == NULL) {
|
if (tmp == NULL) {
|
||||||
@ -134,6 +129,14 @@ virInterfaceDefParseBasicAttrs(virConnectPtr conn, virInterfaceDefPtr def,
|
|||||||
return(-1);
|
return(-1);
|
||||||
}
|
}
|
||||||
def->name = tmp;
|
def->name = tmp;
|
||||||
|
return(0);
|
||||||
|
}
|
||||||
|
|
||||||
|
static int
|
||||||
|
virInterfaceDefParseMtu(virConnectPtr conn, virInterfaceDefPtr def,
|
||||||
|
xmlXPathContextPtr ctxt) {
|
||||||
|
unsigned long mtu;
|
||||||
|
int ret;
|
||||||
|
|
||||||
ret = virXPathULong(conn, "string(./mtu/@size)", ctxt, &mtu);
|
ret = virXPathULong(conn, "string(./mtu/@size)", ctxt, &mtu);
|
||||||
if ((ret == -2) || ((ret == 0) && (mtu > 100000))) {
|
if ((ret == -2) || ((ret == 0) && (mtu > 100000))) {
|
||||||
@ -478,80 +481,12 @@ error:
|
|||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static virInterfaceBareDefPtr
|
|
||||||
virInterfaceDefParseBareInterface(virConnectPtr conn, xmlXPathContextPtr ctxt,
|
|
||||||
int ethernet_only) {
|
|
||||||
int t;
|
|
||||||
char *name = NULL;
|
|
||||||
char *type = NULL;
|
|
||||||
char *mac_or_tag = NULL;
|
|
||||||
char *devname = NULL;
|
|
||||||
virInterfaceBareDefPtr ret = NULL;
|
|
||||||
|
|
||||||
type = virXPathString(conn, "string(./@type)", ctxt);
|
|
||||||
if (type == NULL) {
|
|
||||||
virInterfaceReportError(conn, VIR_ERR_XML_ERROR,
|
|
||||||
"%s", _("interface has no type"));
|
|
||||||
goto error;
|
|
||||||
}
|
|
||||||
if (STREQ(type, "ethernet")) {
|
|
||||||
t = VIR_INTERFACE_TYPE_ETHERNET;
|
|
||||||
name = virXPathString(conn, "string(./@name)", ctxt);
|
|
||||||
mac_or_tag = virXPathString(conn, "string(./mac/@address)", ctxt);
|
|
||||||
} else if ((STREQ(type, "vlan")) && (ethernet_only == 0)) {
|
|
||||||
t = VIR_INTERFACE_TYPE_VLAN;
|
|
||||||
name = virXPathString(conn, "string(./@name)", ctxt);
|
|
||||||
mac_or_tag = virXPathString(conn, "string(./vlan/@tag)", ctxt);
|
|
||||||
devname = virXPathString(conn, "string(./vlan/interface/@name)", ctxt);
|
|
||||||
} else {
|
|
||||||
virInterfaceReportError(conn, VIR_ERR_XML_ERROR,
|
|
||||||
_("interface has unsupported type '%s'"), type);
|
|
||||||
VIR_FREE(type);
|
|
||||||
goto error;
|
|
||||||
}
|
|
||||||
VIR_FREE(type);
|
|
||||||
if (name == NULL) {
|
|
||||||
virInterfaceReportError(conn, VIR_ERR_XML_ERROR,
|
|
||||||
"%s", _("interface has no name"));
|
|
||||||
goto error;
|
|
||||||
}
|
|
||||||
if (t == VIR_INTERFACE_TYPE_VLAN) {
|
|
||||||
if (mac_or_tag == NULL) {
|
|
||||||
virInterfaceReportError(conn, VIR_ERR_XML_ERROR,
|
|
||||||
_("vlan %s has no tag"), name);
|
|
||||||
goto error;
|
|
||||||
}
|
|
||||||
if (devname == NULL) {
|
|
||||||
virInterfaceReportError(conn, VIR_ERR_XML_ERROR,
|
|
||||||
_("vlan %s has interface name"), name);
|
|
||||||
goto error;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (VIR_ALLOC(ret) < 0) {
|
|
||||||
virReportOOMError(conn);
|
|
||||||
goto error;
|
|
||||||
}
|
|
||||||
ret->type = t;
|
|
||||||
ret->name = name;
|
|
||||||
ret->mac_or_tag = mac_or_tag;
|
|
||||||
ret->devname = devname;
|
|
||||||
return(ret);
|
|
||||||
|
|
||||||
error:
|
|
||||||
VIR_FREE(name);
|
|
||||||
VIR_FREE(type);
|
|
||||||
VIR_FREE(name);
|
|
||||||
VIR_FREE(name);
|
|
||||||
VIR_FREE(ret);
|
|
||||||
return(NULL);
|
|
||||||
}
|
|
||||||
|
|
||||||
static int
|
static int
|
||||||
virInterfaceDefParseBridge(virConnectPtr conn, virInterfaceDefPtr def,
|
virInterfaceDefParseBridge(virConnectPtr conn, virInterfaceDefPtr def,
|
||||||
xmlXPathContextPtr ctxt) {
|
xmlXPathContextPtr ctxt) {
|
||||||
xmlNodePtr *interfaces = NULL;
|
xmlNodePtr *interfaces = NULL;
|
||||||
xmlNodePtr bridge;
|
xmlNodePtr bridge;
|
||||||
virInterfaceBareDefPtr itf;
|
virInterfaceDefPtr itf;
|
||||||
int nbItf, i;
|
int nbItf, i;
|
||||||
int ret = 0;
|
int ret = 0;
|
||||||
|
|
||||||
@ -573,7 +508,7 @@ virInterfaceDefParseBridge(virConnectPtr conn, virInterfaceDefPtr def,
|
|||||||
|
|
||||||
for (i = 0; i < nbItf;i++) {
|
for (i = 0; i < nbItf;i++) {
|
||||||
ctxt->node = interfaces[i];
|
ctxt->node = interfaces[i];
|
||||||
itf = virInterfaceDefParseBareInterface(conn, ctxt, 0);
|
itf = virInterfaceDefParseXML(conn, ctxt, VIR_INTERFACE_TYPE_BRIDGE);
|
||||||
if (itf == NULL) {
|
if (itf == NULL) {
|
||||||
ret = -1;
|
ret = -1;
|
||||||
def->data.bridge.nbItf = i;
|
def->data.bridge.nbItf = i;
|
||||||
@ -594,7 +529,7 @@ virInterfaceDefParseBondItfs(virConnectPtr conn, virInterfaceDefPtr def,
|
|||||||
xmlXPathContextPtr ctxt) {
|
xmlXPathContextPtr ctxt) {
|
||||||
xmlNodePtr *interfaces = NULL;
|
xmlNodePtr *interfaces = NULL;
|
||||||
xmlNodePtr bond = ctxt->node;
|
xmlNodePtr bond = ctxt->node;
|
||||||
virInterfaceBareDefPtr itf;
|
virInterfaceDefPtr itf;
|
||||||
int nbItf, i;
|
int nbItf, i;
|
||||||
int ret = 0;
|
int ret = 0;
|
||||||
|
|
||||||
@ -614,7 +549,7 @@ virInterfaceDefParseBondItfs(virConnectPtr conn, virInterfaceDefPtr def,
|
|||||||
|
|
||||||
for (i = 0; i < nbItf;i++) {
|
for (i = 0; i < nbItf;i++) {
|
||||||
ctxt->node = interfaces[i];
|
ctxt->node = interfaces[i];
|
||||||
itf = virInterfaceDefParseBareInterface(conn, ctxt, 1);
|
itf = virInterfaceDefParseXML(conn, ctxt, VIR_INTERFACE_TYPE_BOND);
|
||||||
if (itf == NULL) {
|
if (itf == NULL) {
|
||||||
ret = -1;
|
ret = -1;
|
||||||
def->data.bond.nbItf = i;
|
def->data.bond.nbItf = i;
|
||||||
@ -732,7 +667,8 @@ virInterfaceDefParseVlan(virConnectPtr conn, virInterfaceDefPtr def,
|
|||||||
}
|
}
|
||||||
|
|
||||||
static virInterfaceDefPtr
|
static virInterfaceDefPtr
|
||||||
virInterfaceDefParseXML(virConnectPtr conn, xmlXPathContextPtr ctxt) {
|
virInterfaceDefParseXML(virConnectPtr conn,
|
||||||
|
xmlXPathContextPtr ctxt, int parentIfType) {
|
||||||
virInterfaceDefPtr def;
|
virInterfaceDefPtr def;
|
||||||
int type;
|
int type;
|
||||||
char *tmp;
|
char *tmp;
|
||||||
@ -758,25 +694,47 @@ virInterfaceDefParseXML(virConnectPtr conn, xmlXPathContextPtr ctxt) {
|
|||||||
virReportOOMError(conn);
|
virReportOOMError(conn);
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (((parentIfType == VIR_INTERFACE_TYPE_BOND)
|
||||||
|
&& (type != VIR_INTERFACE_TYPE_ETHERNET))
|
||||||
|
|| ((parentIfType == VIR_INTERFACE_TYPE_BRIDGE)
|
||||||
|
&& (type != VIR_INTERFACE_TYPE_ETHERNET)
|
||||||
|
&& (type != VIR_INTERFACE_TYPE_BOND)
|
||||||
|
&& (type != VIR_INTERFACE_TYPE_VLAN))
|
||||||
|
|| (parentIfType == VIR_INTERFACE_TYPE_ETHERNET)
|
||||||
|
|| (parentIfType == VIR_INTERFACE_TYPE_VLAN))
|
||||||
|
{
|
||||||
|
virInterfaceReportError(conn, VIR_ERR_XML_ERROR,
|
||||||
|
_("interface has unsupported type '%s'"),
|
||||||
|
virInterfaceTypeToString(type));
|
||||||
|
goto error;
|
||||||
|
}
|
||||||
def->type = type;
|
def->type = type;
|
||||||
switch (type) {
|
switch (type) {
|
||||||
case VIR_INTERFACE_TYPE_ETHERNET:
|
case VIR_INTERFACE_TYPE_ETHERNET:
|
||||||
if (virInterfaceDefParseBasicAttrs(conn, def, ctxt) < 0)
|
if (virInterfaceDefParseName(conn, def, ctxt) < 0)
|
||||||
goto error;
|
goto error;
|
||||||
tmp = virXPathString(conn, "string(./mac/@address)", ctxt);
|
tmp = virXPathString(conn, "string(./mac/@address)", ctxt);
|
||||||
if (tmp != NULL)
|
if (tmp != NULL)
|
||||||
def->mac = tmp;
|
def->mac = tmp;
|
||||||
if (virInterfaceDefParseStartMode(conn, def, ctxt) < 0)
|
if (parentIfType == VIR_INTERFACE_TYPE_LAST) {
|
||||||
goto error;
|
/* only recognize these in toplevel bond interfaces */
|
||||||
if (virInterfaceDefParseIfAdressing(conn, def, ctxt) < 0)
|
if (virInterfaceDefParseStartMode(conn, def, ctxt) < 0)
|
||||||
goto error;
|
goto error;
|
||||||
|
if (virInterfaceDefParseMtu(conn, def, ctxt) < 0)
|
||||||
|
goto error;
|
||||||
|
if (virInterfaceDefParseIfAdressing(conn, def, ctxt) < 0)
|
||||||
|
goto error;
|
||||||
|
}
|
||||||
break;
|
break;
|
||||||
case VIR_INTERFACE_TYPE_BRIDGE: {
|
case VIR_INTERFACE_TYPE_BRIDGE: {
|
||||||
xmlNodePtr bridge;
|
xmlNodePtr bridge;
|
||||||
|
|
||||||
|
if (virInterfaceDefParseName(conn, def, ctxt) < 0)
|
||||||
|
goto error;
|
||||||
if (virInterfaceDefParseStartMode(conn, def, ctxt) < 0)
|
if (virInterfaceDefParseStartMode(conn, def, ctxt) < 0)
|
||||||
goto error;
|
goto error;
|
||||||
if (virInterfaceDefParseBasicAttrs(conn, def, ctxt) < 0)
|
if (virInterfaceDefParseMtu(conn, def, ctxt) < 0)
|
||||||
goto error;
|
goto error;
|
||||||
if (virInterfaceDefParseIfAdressing(conn, def, ctxt) < 0)
|
if (virInterfaceDefParseIfAdressing(conn, def, ctxt) < 0)
|
||||||
goto error;
|
goto error;
|
||||||
@ -811,12 +769,18 @@ virInterfaceDefParseXML(virConnectPtr conn, xmlXPathContextPtr ctxt) {
|
|||||||
case VIR_INTERFACE_TYPE_BOND: {
|
case VIR_INTERFACE_TYPE_BOND: {
|
||||||
xmlNodePtr bond;
|
xmlNodePtr bond;
|
||||||
|
|
||||||
if (virInterfaceDefParseBasicAttrs(conn, def, ctxt) < 0)
|
if (virInterfaceDefParseName(conn, def, ctxt) < 0)
|
||||||
goto error;
|
|
||||||
if (virInterfaceDefParseStartMode(conn, def, ctxt) < 0)
|
|
||||||
goto error;
|
|
||||||
if (virInterfaceDefParseIfAdressing(conn, def, ctxt) < 0)
|
|
||||||
goto error;
|
goto error;
|
||||||
|
if (parentIfType == VIR_INTERFACE_TYPE_LAST) {
|
||||||
|
/* only recognize these in toplevel bond interfaces */
|
||||||
|
if (virInterfaceDefParseStartMode(conn, def, ctxt) < 0)
|
||||||
|
goto error;
|
||||||
|
if (virInterfaceDefParseMtu(conn, def, ctxt) < 0)
|
||||||
|
goto error;
|
||||||
|
if (virInterfaceDefParseIfAdressing(conn, def, ctxt) < 0)
|
||||||
|
goto error;
|
||||||
|
}
|
||||||
|
|
||||||
bond = virXPathNode(conn, "./bond[1]", ctxt);
|
bond = virXPathNode(conn, "./bond[1]", ctxt);
|
||||||
if (bond == NULL) {
|
if (bond == NULL) {
|
||||||
virInterfaceReportError(conn, VIR_ERR_XML_ERROR,
|
virInterfaceReportError(conn, VIR_ERR_XML_ERROR,
|
||||||
@ -881,7 +845,7 @@ virInterfaceDefPtr virInterfaceDefParseNode(virConnectPtr conn,
|
|||||||
}
|
}
|
||||||
|
|
||||||
ctxt->node = root;
|
ctxt->node = root;
|
||||||
def = virInterfaceDefParseXML(conn, ctxt);
|
def = virInterfaceDefParseXML(conn, ctxt, VIR_INTERFACE_TYPE_LAST);
|
||||||
|
|
||||||
cleanup:
|
cleanup:
|
||||||
xmlXPathFreeContext(ctxt);
|
xmlXPathFreeContext(ctxt);
|
||||||
@ -989,59 +953,13 @@ cleanup:
|
|||||||
return def;
|
return def;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int
|
|
||||||
virInterfaceBareDevDefFormat(virConnectPtr conn ATTRIBUTE_UNUSED,
|
|
||||||
virBufferPtr buf,
|
|
||||||
const virInterfaceBareDefPtr def) {
|
|
||||||
if (def->type == VIR_INTERFACE_TYPE_ETHERNET) {
|
|
||||||
if (def->name == NULL) {
|
|
||||||
virInterfaceReportError(conn, VIR_ERR_INTERNAL_ERROR,
|
|
||||||
"%s", _("bare ethernet has no name"));
|
|
||||||
return(-1);
|
|
||||||
}
|
|
||||||
virBufferVSprintf(buf, " <interface type='ethernet' name='%s'",
|
|
||||||
def->name);
|
|
||||||
if (def->mac_or_tag != NULL) {
|
|
||||||
virBufferVSprintf(buf, ">\n <mac address='%s'/>\n",
|
|
||||||
def->mac_or_tag);
|
|
||||||
virBufferAddLit(buf, " </interface>\n");
|
|
||||||
} else {
|
|
||||||
virBufferAddLit(buf, "/>\n");
|
|
||||||
}
|
|
||||||
} else if (def->type == VIR_INTERFACE_TYPE_VLAN) {
|
|
||||||
virBufferAddLit(buf, " <interface type='vlan'");
|
|
||||||
if (def->name != NULL)
|
|
||||||
virBufferVSprintf(buf, " name='%s'", def->name);
|
|
||||||
if (def->mac_or_tag != NULL) {
|
|
||||||
virBufferAddLit(buf, ">\n");
|
|
||||||
virBufferVSprintf(buf, " <vlan tag='%s'", def->mac_or_tag);
|
|
||||||
if (def->devname != NULL) {
|
|
||||||
virBufferAddLit(buf, ">\n");
|
|
||||||
virBufferVSprintf(buf, " <interface name='%s'/>\n",
|
|
||||||
def->devname);
|
|
||||||
virBufferAddLit(buf, " </vlan>\n");
|
|
||||||
} else
|
|
||||||
virBufferAddLit(buf, "/>\n");
|
|
||||||
virBufferAddLit(buf, " </interface>\n");
|
|
||||||
} else {
|
|
||||||
virBufferAddLit(buf, "/>\n");
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
virInterfaceReportError(conn, VIR_ERR_INTERNAL_ERROR,
|
|
||||||
_("bare interface type %d unknown"),
|
|
||||||
def->type);
|
|
||||||
return(-1);
|
|
||||||
}
|
|
||||||
return(0);
|
|
||||||
}
|
|
||||||
|
|
||||||
static int
|
static int
|
||||||
virInterfaceBridgeDefFormat(virConnectPtr conn, virBufferPtr buf,
|
virInterfaceBridgeDefFormat(virConnectPtr conn, virBufferPtr buf,
|
||||||
const virInterfaceDefPtr def) {
|
const virInterfaceDefPtr def, int level) {
|
||||||
int i;
|
int i;
|
||||||
int ret = 0;
|
int ret = 0;
|
||||||
|
|
||||||
virBufferAddLit(buf, " <bridge");
|
virBufferVSprintf(buf, "%*s <bridge", level*2, "");
|
||||||
if (def->data.bridge.stp == 1)
|
if (def->data.bridge.stp == 1)
|
||||||
virBufferAddLit(buf, " stp='on'");
|
virBufferAddLit(buf, " stp='on'");
|
||||||
else if (def->data.bridge.stp == 0)
|
else if (def->data.bridge.stp == 0)
|
||||||
@ -1051,22 +969,22 @@ virInterfaceBridgeDefFormat(virConnectPtr conn, virBufferPtr buf,
|
|||||||
virBufferAddLit(buf, ">\n");
|
virBufferAddLit(buf, ">\n");
|
||||||
|
|
||||||
for (i = 0;i < def->data.bridge.nbItf;i++) {
|
for (i = 0;i < def->data.bridge.nbItf;i++) {
|
||||||
if (virInterfaceBareDevDefFormat(conn, buf, def->data.bridge.itf[i])
|
if (virInterfaceDefDevFormat(conn, buf,
|
||||||
< 0)
|
def->data.bridge.itf[i], level+2) < 0)
|
||||||
ret = -1;
|
ret = -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
virBufferAddLit(buf, " </bridge>\n");
|
virBufferVSprintf(buf, "%*s </bridge>\n", level*2, "");
|
||||||
return(ret);
|
return(ret);
|
||||||
}
|
}
|
||||||
|
|
||||||
static int
|
static int
|
||||||
virInterfaceBondDefFormat(virConnectPtr conn, virBufferPtr buf,
|
virInterfaceBondDefFormat(virConnectPtr conn, virBufferPtr buf,
|
||||||
const virInterfaceDefPtr def) {
|
const virInterfaceDefPtr def, int level) {
|
||||||
int i;
|
int i;
|
||||||
int ret = 0;
|
int ret = 0;
|
||||||
|
|
||||||
virBufferAddLit(buf, " <bond");
|
virBufferVSprintf(buf, "%*s <bond", level*2, "");
|
||||||
if (def->data.bond.mode == VIR_INTERFACE_BOND_BALRR)
|
if (def->data.bond.mode == VIR_INTERFACE_BOND_BALRR)
|
||||||
virBufferAddLit(buf, " mode='balance-rr'");
|
virBufferAddLit(buf, " mode='balance-rr'");
|
||||||
else if (def->data.bond.mode == VIR_INTERFACE_BOND_ABACKUP)
|
else if (def->data.bond.mode == VIR_INTERFACE_BOND_ABACKUP)
|
||||||
@ -1084,8 +1002,8 @@ virInterfaceBondDefFormat(virConnectPtr conn, virBufferPtr buf,
|
|||||||
virBufferAddLit(buf, ">\n");
|
virBufferAddLit(buf, ">\n");
|
||||||
|
|
||||||
if (def->data.bond.monit == VIR_INTERFACE_BOND_MONIT_MII) {
|
if (def->data.bond.monit == VIR_INTERFACE_BOND_MONIT_MII) {
|
||||||
virBufferVSprintf(buf, " <miimon freq='%d'",
|
virBufferVSprintf(buf, "%*s <miimon freq='%d'",
|
||||||
def->data.bond.frequency);
|
level*2, "", def->data.bond.frequency);
|
||||||
if (def->data.bond.downdelay > 0)
|
if (def->data.bond.downdelay > 0)
|
||||||
virBufferVSprintf(buf, " downdelay='%d'", def->data.bond.downdelay);
|
virBufferVSprintf(buf, " downdelay='%d'", def->data.bond.downdelay);
|
||||||
if (def->data.bond.updelay > 0)
|
if (def->data.bond.updelay > 0)
|
||||||
@ -1101,7 +1019,8 @@ virInterfaceBondDefFormat(virConnectPtr conn, virBufferPtr buf,
|
|||||||
"%s", _("bond arp monitoring has no target"));
|
"%s", _("bond arp monitoring has no target"));
|
||||||
return(-1);
|
return(-1);
|
||||||
}
|
}
|
||||||
virBufferVSprintf(buf, " <arpmon interval='%d' target='%s'",
|
virBufferVSprintf(buf, "%*s <arpmon interval='%d' target='%s'",
|
||||||
|
level*2, "",
|
||||||
def->data.bond.interval, def->data.bond.target);
|
def->data.bond.interval, def->data.bond.target);
|
||||||
if (def->data.bond.validate == VIR_INTERFACE_BOND_ARP_ACTIVE)
|
if (def->data.bond.validate == VIR_INTERFACE_BOND_ARP_ACTIVE)
|
||||||
virBufferAddLit(buf, " validate='active'");
|
virBufferAddLit(buf, " validate='active'");
|
||||||
@ -1112,29 +1031,30 @@ virInterfaceBondDefFormat(virConnectPtr conn, virBufferPtr buf,
|
|||||||
virBufferAddLit(buf, "/>\n");
|
virBufferAddLit(buf, "/>\n");
|
||||||
}
|
}
|
||||||
for (i = 0;i < def->data.bond.nbItf;i++) {
|
for (i = 0;i < def->data.bond.nbItf;i++) {
|
||||||
if (virInterfaceBareDevDefFormat(conn, buf, def->data.bond.itf[i]) < 0)
|
if (virInterfaceDefDevFormat(conn, buf, def->data.bond.itf[i], level+2) < 0)
|
||||||
ret = -1;
|
ret = -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
virBufferAddLit(buf, " </bond>\n");
|
virBufferVSprintf(buf, "%*s </bond>\n", level*2, "");
|
||||||
return(ret);
|
return(ret);
|
||||||
}
|
}
|
||||||
|
|
||||||
static int
|
static int
|
||||||
virInterfaceVlanDefFormat(virConnectPtr conn, virBufferPtr buf,
|
virInterfaceVlanDefFormat(virConnectPtr conn, virBufferPtr buf,
|
||||||
const virInterfaceDefPtr def) {
|
const virInterfaceDefPtr def, int level) {
|
||||||
if (def->data.vlan.tag == NULL) {
|
if (def->data.vlan.tag == NULL) {
|
||||||
virInterfaceReportError(conn, VIR_ERR_INTERNAL_ERROR,
|
virInterfaceReportError(conn, VIR_ERR_INTERNAL_ERROR,
|
||||||
"%s", _("vlan misses the tag name"));
|
"%s", _("vlan misses the tag name"));
|
||||||
return(-1);
|
return(-1);
|
||||||
}
|
}
|
||||||
|
|
||||||
virBufferVSprintf(buf, " <vlan tag='%s'", def->data.vlan.tag);
|
virBufferVSprintf(buf, "%*s <vlan tag='%s'",
|
||||||
|
level*2, "", def->data.vlan.tag);
|
||||||
if (def->data.vlan.devname != NULL) {
|
if (def->data.vlan.devname != NULL) {
|
||||||
virBufferAddLit(buf, ">\n");
|
virBufferAddLit(buf, ">\n");
|
||||||
virBufferVSprintf(buf, " <interface name='%s'/>\n",
|
virBufferVSprintf(buf, "%*s <interface name='%s'/>\n",
|
||||||
def->data.vlan.devname);
|
level*2, "", def->data.vlan.devname);
|
||||||
virBufferAddLit(buf, " </vlan>\n");
|
virBufferVSprintf(buf, "%*s </vlan>\n", level*2, "");
|
||||||
} else
|
} else
|
||||||
virBufferAddLit(buf, "/>\n");
|
virBufferAddLit(buf, "/>\n");
|
||||||
return(0);
|
return(0);
|
||||||
@ -1142,30 +1062,34 @@ virInterfaceVlanDefFormat(virConnectPtr conn, virBufferPtr buf,
|
|||||||
|
|
||||||
static int
|
static int
|
||||||
virInterfaceProtocolDefFormat(virConnectPtr conn ATTRIBUTE_UNUSED,
|
virInterfaceProtocolDefFormat(virConnectPtr conn ATTRIBUTE_UNUSED,
|
||||||
virBufferPtr buf, const virInterfaceDefPtr def) {
|
virBufferPtr buf, const virInterfaceDefPtr def,
|
||||||
|
int level) {
|
||||||
int pp, ii;
|
int pp, ii;
|
||||||
|
|
||||||
for (pp = 0; pp < def->nprotos; pp++) {
|
for (pp = 0; pp < def->nprotos; pp++) {
|
||||||
|
|
||||||
virBufferVSprintf(buf, " <protocol family='%s'>\n", def->protos[pp]->family);
|
virBufferVSprintf(buf, "%*s <protocol family='%s'>\n",
|
||||||
|
level*2, "", def->protos[pp]->family);
|
||||||
|
|
||||||
if (def->protos[pp]->autoconf) {
|
if (def->protos[pp]->autoconf) {
|
||||||
virBufferAddLit(buf, " <autoconf/>\n");
|
virBufferVSprintf(buf, "%*s <autoconf/>\n", level*2, "");
|
||||||
}
|
}
|
||||||
|
|
||||||
if (def->protos[pp]->dhcp) {
|
if (def->protos[pp]->dhcp) {
|
||||||
if (def->protos[pp]->peerdns == 0)
|
if (def->protos[pp]->peerdns == 0)
|
||||||
virBufferAddLit(buf, " <dhcp peerdns='no'/>\n");
|
virBufferVSprintf(buf, "%*s <dhcp peerdns='no'/>\n",
|
||||||
|
level*2, "");
|
||||||
else if (def->protos[pp]->peerdns == 1)
|
else if (def->protos[pp]->peerdns == 1)
|
||||||
virBufferAddLit(buf, " <dhcp peerdns='yes'/>\n");
|
virBufferVSprintf(buf, "%*s <dhcp peerdns='yes'/>\n",
|
||||||
|
level*2, "");
|
||||||
else
|
else
|
||||||
virBufferAddLit(buf, " <dhcp/>\n");
|
virBufferVSprintf(buf, "%*s <dhcp/>\n", level*2, "");
|
||||||
}
|
}
|
||||||
|
|
||||||
for (ii = 0; ii < def->protos[pp]->nips; ii++) {
|
for (ii = 0; ii < def->protos[pp]->nips; ii++) {
|
||||||
if (def->protos[pp]->ips[ii]->address != NULL) {
|
if (def->protos[pp]->ips[ii]->address != NULL) {
|
||||||
|
|
||||||
virBufferVSprintf(buf, " <ip address='%s'",
|
virBufferVSprintf(buf, "%*s <ip address='%s'", level*2, "",
|
||||||
def->protos[pp]->ips[ii]->address);
|
def->protos[pp]->ips[ii]->address);
|
||||||
if (def->protos[pp]->ips[ii]->prefix != 0) {
|
if (def->protos[pp]->ips[ii]->prefix != 0) {
|
||||||
virBufferVSprintf(buf, " prefix='%d'",
|
virBufferVSprintf(buf, " prefix='%d'",
|
||||||
@ -1175,18 +1099,19 @@ virInterfaceProtocolDefFormat(virConnectPtr conn ATTRIBUTE_UNUSED,
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (def->protos[pp]->gateway != NULL) {
|
if (def->protos[pp]->gateway != NULL) {
|
||||||
virBufferVSprintf(buf, " <route gateway='%s'/>\n",
|
virBufferVSprintf(buf, "%*s <route gateway='%s'/>\n",
|
||||||
def->protos[pp]->gateway);
|
level*2, "", def->protos[pp]->gateway);
|
||||||
}
|
}
|
||||||
|
|
||||||
virBufferAddLit(buf, " </protocol>\n");
|
virBufferVSprintf(buf, "%*s </protocol>\n", level*2, "");
|
||||||
}
|
}
|
||||||
return(0);
|
return(0);
|
||||||
}
|
}
|
||||||
|
|
||||||
static int
|
static int
|
||||||
virInterfaceStartmodeDefFormat(virConnectPtr conn, virBufferPtr buf,
|
virInterfaceStartmodeDefFormat(virConnectPtr conn, virBufferPtr buf,
|
||||||
enum virInterfaceStartMode startmode) {
|
enum virInterfaceStartMode startmode,
|
||||||
|
int level) {
|
||||||
const char *mode;
|
const char *mode;
|
||||||
switch (startmode) {
|
switch (startmode) {
|
||||||
case VIR_INTERFACE_START_UNSPECIFIED:
|
case VIR_INTERFACE_START_UNSPECIFIED:
|
||||||
@ -1205,20 +1130,24 @@ virInterfaceStartmodeDefFormat(virConnectPtr conn, virBufferPtr buf,
|
|||||||
"%s", _("virInterfaceDefFormat unknown startmode"));
|
"%s", _("virInterfaceDefFormat unknown startmode"));
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
virBufferVSprintf(buf, " <start mode='%s'/>\n", mode);
|
virBufferVSprintf(buf, "%*s <start mode='%s'/>\n", level*2, "", mode);
|
||||||
return(0);
|
return(0);
|
||||||
}
|
}
|
||||||
|
|
||||||
char *virInterfaceDefFormat(virConnectPtr conn,
|
static int
|
||||||
const virInterfaceDefPtr def)
|
virInterfaceDefDevFormat(virConnectPtr conn, virBufferPtr buf,
|
||||||
{
|
const virInterfaceDefPtr def, int level) {
|
||||||
virBuffer buf = VIR_BUFFER_INITIALIZER;
|
|
||||||
const char *type = NULL;
|
const char *type = NULL;
|
||||||
|
|
||||||
if ((def == NULL) ||
|
if (def == NULL) {
|
||||||
((def->name == NULL) && (def->type != VIR_INTERFACE_TYPE_VLAN))) {
|
|
||||||
virInterfaceReportError(conn, VIR_ERR_INTERNAL_ERROR,
|
virInterfaceReportError(conn, VIR_ERR_INTERNAL_ERROR,
|
||||||
"%s", _("virInterfaceDefFormat argument problems"));
|
"%s", _("virInterfaceDefFormat NULL def"));
|
||||||
|
goto cleanup;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ((def->name == NULL) && (def->type != VIR_INTERFACE_TYPE_VLAN)) {
|
||||||
|
virInterfaceReportError(conn, VIR_ERR_INTERNAL_ERROR,
|
||||||
|
"%s", _("virInterfaceDefFormat missing interface name"));
|
||||||
goto cleanup;
|
goto cleanup;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1228,56 +1157,72 @@ char *virInterfaceDefFormat(virConnectPtr conn,
|
|||||||
goto cleanup;
|
goto cleanup;
|
||||||
}
|
}
|
||||||
|
|
||||||
virBufferVSprintf(&buf, "<interface type='%s' ", type);
|
virBufferVSprintf(buf, "%*s<interface type='%s' ", level*2, "", type);
|
||||||
if (def->name != NULL)
|
if (def->name != NULL)
|
||||||
virBufferEscapeString(&buf, "name='%s'", def->name);
|
virBufferEscapeString(buf, "name='%s'", def->name);
|
||||||
virBufferAddLit(&buf, ">\n");
|
virBufferAddLit(buf, ">\n");
|
||||||
|
|
||||||
switch (def->type) {
|
switch (def->type) {
|
||||||
case VIR_INTERFACE_TYPE_ETHERNET:
|
case VIR_INTERFACE_TYPE_ETHERNET:
|
||||||
virInterfaceStartmodeDefFormat(conn, &buf, def->startmode);
|
virInterfaceStartmodeDefFormat(conn, buf, def->startmode, level);
|
||||||
if (def->mac != NULL)
|
if (def->mac != NULL)
|
||||||
virBufferVSprintf(&buf, " <mac address='%s'/>\n", def->mac);
|
virBufferVSprintf(buf, "%*s <mac address='%s'/>\n",
|
||||||
|
level*2, "", def->mac);
|
||||||
if (def->mtu != 0)
|
if (def->mtu != 0)
|
||||||
virBufferVSprintf(&buf, " <mtu size='%d'/>\n", def->mtu);
|
virBufferVSprintf(buf, "%*s <mtu size='%d'/>\n",
|
||||||
virInterfaceProtocolDefFormat(conn, &buf, def);
|
level*2, "", def->mtu);
|
||||||
|
virInterfaceProtocolDefFormat(conn, buf, def, level);
|
||||||
break;
|
break;
|
||||||
case VIR_INTERFACE_TYPE_BRIDGE:
|
case VIR_INTERFACE_TYPE_BRIDGE:
|
||||||
virInterfaceStartmodeDefFormat(conn, &buf, def->startmode);
|
virInterfaceStartmodeDefFormat(conn, buf, def->startmode, level);
|
||||||
if (def->mtu != 0)
|
if (def->mtu != 0)
|
||||||
virBufferVSprintf(&buf, " <mtu size='%d'/>\n", def->mtu);
|
virBufferVSprintf(buf, "%*s <mtu size='%d'/>\n",
|
||||||
virInterfaceProtocolDefFormat(conn, &buf, def);
|
level*2, "", def->mtu);
|
||||||
virInterfaceBridgeDefFormat(conn, &buf, def);
|
virInterfaceProtocolDefFormat(conn, buf, def, level);
|
||||||
|
virInterfaceBridgeDefFormat(conn, buf, def, level);
|
||||||
break;
|
break;
|
||||||
case VIR_INTERFACE_TYPE_BOND:
|
case VIR_INTERFACE_TYPE_BOND:
|
||||||
virInterfaceStartmodeDefFormat(conn, &buf, def->startmode);
|
virInterfaceStartmodeDefFormat(conn, buf, def->startmode, level);
|
||||||
if (def->mtu != 0)
|
if (def->mtu != 0)
|
||||||
virBufferVSprintf(&buf, " <mtu size='%d'/>\n", def->mtu);
|
virBufferVSprintf(buf, "%*s <mtu size='%d'/>\n",
|
||||||
virInterfaceProtocolDefFormat(conn, &buf, def);
|
level*2, "", def->mtu);
|
||||||
virInterfaceBondDefFormat(conn, &buf, def);
|
virInterfaceProtocolDefFormat(conn, buf, def, level);
|
||||||
|
virInterfaceBondDefFormat(conn, buf, def, level);
|
||||||
break;
|
break;
|
||||||
case VIR_INTERFACE_TYPE_VLAN:
|
case VIR_INTERFACE_TYPE_VLAN:
|
||||||
virInterfaceStartmodeDefFormat(conn, &buf, def->startmode);
|
virInterfaceStartmodeDefFormat(conn, buf, def->startmode, level);
|
||||||
if (def->mac != NULL)
|
if (def->mac != NULL)
|
||||||
virBufferVSprintf(&buf, " <mac address='%s'/>\n", def->mac);
|
virBufferVSprintf(buf, "%*s <mac address='%s'/>\n",
|
||||||
|
level*2, "", def->mac);
|
||||||
if (def->mtu != 0)
|
if (def->mtu != 0)
|
||||||
virBufferVSprintf(&buf, " <mtu size='%d'/>\n", def->mtu);
|
virBufferVSprintf(buf, "%*s <mtu size='%d'/>\n",
|
||||||
virInterfaceProtocolDefFormat(conn, &buf, def);
|
level*2, "", def->mtu);
|
||||||
virInterfaceVlanDefFormat(conn, &buf, def);
|
virInterfaceProtocolDefFormat(conn, buf, def, level);
|
||||||
|
virInterfaceVlanDefFormat(conn, buf, def, level);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
virBufferAddLit(&buf, "</interface>\n");
|
virBufferVSprintf(buf, "%*s</interface>\n", level*2, "");
|
||||||
|
|
||||||
if (virBufferError(&buf))
|
if (virBufferError(buf))
|
||||||
goto no_memory;
|
goto no_memory;
|
||||||
return virBufferContentAndReset(&buf);
|
return 0;
|
||||||
|
|
||||||
no_memory:
|
no_memory:
|
||||||
virReportOOMError(conn);
|
virReportOOMError(conn);
|
||||||
cleanup:
|
cleanup:
|
||||||
virBufferFreeAndReset(&buf);
|
return -1;
|
||||||
return NULL;
|
}
|
||||||
|
|
||||||
|
char *virInterfaceDefFormat(virConnectPtr conn,
|
||||||
|
const virInterfaceDefPtr def)
|
||||||
|
{
|
||||||
|
virBuffer buf = VIR_BUFFER_INITIALIZER;
|
||||||
|
|
||||||
|
if (virInterfaceDefDevFormat(conn, &buf, def, 0) < 0) {
|
||||||
|
virBufferFreeAndReset(&buf);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
return virBufferContentAndReset(&buf);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* virInterfaceObj manipulation */
|
/* virInterfaceObj manipulation */
|
||||||
|
@ -84,14 +84,7 @@ enum virInterfaceBondArpValid {
|
|||||||
VIR_INTERFACE_BOND_ARP_ALL, /* validate all */
|
VIR_INTERFACE_BOND_ARP_ALL, /* validate all */
|
||||||
};
|
};
|
||||||
|
|
||||||
typedef struct _virInterfaceBareDef virInterfaceBareDef;
|
struct _virInterfaceDef; /* forward declaration required for bridge/bond */
|
||||||
typedef virInterfaceBareDef *virInterfaceBareDefPtr;
|
|
||||||
struct _virInterfaceBareDef {
|
|
||||||
int type; /* should be only ethernet or vlan */
|
|
||||||
char *name; /* the interface name */
|
|
||||||
char *mac_or_tag; /* MAC address for ethernet, TAG for vlan */
|
|
||||||
char *devname; /* device name for vlan */
|
|
||||||
};
|
|
||||||
|
|
||||||
typedef struct _virInterfaceBridgeDef virInterfaceBridgeDef;
|
typedef struct _virInterfaceBridgeDef virInterfaceBridgeDef;
|
||||||
typedef virInterfaceBridgeDef *virInterfaceBridgeDefPtr;
|
typedef virInterfaceBridgeDef *virInterfaceBridgeDefPtr;
|
||||||
@ -99,7 +92,7 @@ struct _virInterfaceBridgeDef {
|
|||||||
int stp; /* 0, 1 or -1 if undefined */
|
int stp; /* 0, 1 or -1 if undefined */
|
||||||
char *delay;
|
char *delay;
|
||||||
int nbItf; /* number of defined interfaces */
|
int nbItf; /* number of defined interfaces */
|
||||||
virInterfaceBareDefPtr *itf;/* interfaces */
|
struct _virInterfaceDef **itf;/* interfaces */
|
||||||
};
|
};
|
||||||
|
|
||||||
typedef struct _virInterfaceBondDef virInterfaceBondDef;
|
typedef struct _virInterfaceBondDef virInterfaceBondDef;
|
||||||
@ -115,7 +108,7 @@ struct _virInterfaceBondDef {
|
|||||||
char *target; /* arp monitoring target */
|
char *target; /* arp monitoring target */
|
||||||
int validate; /* virInterfaceBondArpmValid */
|
int validate; /* virInterfaceBondArpmValid */
|
||||||
int nbItf; /* number of defined interfaces */
|
int nbItf; /* number of defined interfaces */
|
||||||
virInterfaceBareDefPtr *itf; /* interfaces ethernet only */
|
struct _virInterfaceDef **itf; /* interfaces ethernet only */
|
||||||
};
|
};
|
||||||
|
|
||||||
typedef struct _virInterfaceVlanDef virInterfaceVlanDef;
|
typedef struct _virInterfaceVlanDef virInterfaceVlanDef;
|
||||||
|
Loading…
Reference in New Issue
Block a user