/* * Copyright (C) 2009-2013 Red Hat, Inc. * * 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 * License along with this library. If not, see * . * * Authors: * Laine Stump * James Robson */ #include #include "netdev_vlan_conf.h" #include "virerror.h" #include "viralloc.h" #define VIR_FROM_THIS VIR_FROM_NONE VIR_ENUM_IMPL(virNativeVlanMode, VIR_NATIVE_VLAN_MODE_LAST, "default", "tagged", "untagged") int virNetDevVlanParse(xmlNodePtr node, xmlXPathContextPtr ctxt, virNetDevVlanPtr def) { int ret = -1; xmlNodePtr save = ctxt->node; char *trunk = NULL; char *nativeMode = NULL; xmlNodePtr *tagNodes = NULL; int nTags; size_t i; ctxt->node = node; nTags = virXPathNodeSet("./tag", ctxt, &tagNodes); if (nTags < 0) goto cleanup; if (nTags == 0) { virReportError(VIR_ERR_XML_ERROR, "%s", _("missing tag id - each must have " "at least one subelement")); goto cleanup; } if (VIR_ALLOC_N(def->tag, nTags) < 0) goto cleanup; def->nativeMode = 0; def->nativeTag = 0; for (i = 0; i < nTags; i++) { unsigned long id; ctxt->node = tagNodes[i]; if (virXPathULong("string(./@id)", ctxt, &id) < 0) { virReportError(VIR_ERR_XML_ERROR, "%s", _("missing or invalid vlan tag id attribute")); goto cleanup; } if (id > 4095) { virReportError(VIR_ERR_XML_ERROR, _("vlan tag id %lu too large (maximum 4095)"), id); goto cleanup; } if ((nativeMode = virXPathString("string(./@nativeMode)", ctxt))) { if (def->nativeMode != 0) { virReportError(VIR_ERR_XML_ERROR, "%s", _("duplicate native vlan setting")); goto cleanup; } if ((def->nativeMode = virNativeVlanModeTypeFromString(nativeMode)) <= 0) { virReportError(VIR_ERR_XML_ERROR, _("Invalid \"nativeMode='%s'\" " "in vlan element"), nativeMode); goto cleanup; } VIR_FREE(nativeMode); def->nativeTag = id; } def->tag[i] = id; } def->nTags = nTags; /* now that we know how many tags there are, look for an explicit * trunk setting. */ if (nTags > 1) def->trunk = true; ctxt->node = node; if ((trunk = virXPathString("string(./@trunk)", ctxt)) != NULL) { def->trunk = STRCASEEQ(trunk, "yes"); if (!def->trunk) { if (nTags > 1) { virReportError(VIR_ERR_XML_ERROR, _("invalid \"trunk='%s'\" in - trunk='yes' " "is required for more than one vlan tag"), trunk); goto cleanup; } if (def->nativeMode != 0) { virReportError(VIR_ERR_XML_ERROR, "%s", _("invalid configuration in - \"trunk='no'\" is " "not allowed with a native vlan id")); goto cleanup; } /* allow (but discard) "trunk='no' if there is a single tag */ if (STRCASENEQ(trunk, "no")) { virReportError(VIR_ERR_XML_ERROR, _("invalid \"trunk='%s'\" in " "- must be yes or no"), trunk); goto cleanup; } } } ret = 0; cleanup: ctxt->node = save; VIR_FREE(tagNodes); VIR_FREE(trunk); VIR_FREE(nativeMode); if (ret < 0) virNetDevVlanClear(def); return ret; } int virNetDevVlanFormat(const virNetDevVlan *def, virBufferPtr buf) { size_t i; if (def->nTags == 0) return 0; if (!def->tag) { virReportError(VIR_ERR_INTERNAL_ERROR, "%s", _("missing vlan tag data")); return -1; } virBufferAsprintf(buf, "\n", def->trunk ? " trunk='yes'" : ""); for (i = 0; i < def->nTags; i++) { if (def->nativeMode != VIR_NATIVE_VLAN_MODE_DEFAULT && def->nativeTag == def->tag[i]) { /* check the nativeMode in case we get */ const char *mode = virNativeVlanModeTypeToString(def->nativeMode); if (!mode) { virReportError(VIR_ERR_INTERNAL_ERROR, "%s", _("Bad value for nativeMode")); } virBufferAsprintf(buf, " \n", def->tag[i], mode); } else { virBufferAsprintf(buf, " \n", def->tag[i]); } } virBufferAddLit(buf, "\n"); return 0; }