2018-05-10 16:21:24 +00:00
|
|
|
/*
|
|
|
|
* virnwfilterbindingobj.c: network filter binding object processing
|
|
|
|
*
|
|
|
|
* Copyright (C) 2018 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
|
|
|
|
* <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <config.h>
|
|
|
|
|
|
|
|
#include "viralloc.h"
|
|
|
|
#include "virerror.h"
|
|
|
|
#include "virstring.h"
|
|
|
|
#include "nwfilter_params.h"
|
|
|
|
#include "virnwfilterbindingobj.h"
|
|
|
|
#include "viruuid.h"
|
|
|
|
#include "virfile.h"
|
|
|
|
|
|
|
|
|
|
|
|
#define VIR_FROM_THIS VIR_FROM_NWFILTER
|
|
|
|
|
|
|
|
struct _virNWFilterBindingObj {
|
|
|
|
virObjectLockable parent;
|
|
|
|
|
|
|
|
bool removing;
|
|
|
|
virNWFilterBindingDefPtr def;
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
static virClassPtr virNWFilterBindingObjClass;
|
|
|
|
static void virNWFilterBindingObjDispose(void *obj);
|
|
|
|
|
|
|
|
static int
|
|
|
|
virNWFilterBindingObjOnceInit(void)
|
|
|
|
{
|
|
|
|
if (!VIR_CLASS_NEW(virNWFilterBindingObj, virClassForObjectLockable()))
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2019-01-20 17:23:29 +00:00
|
|
|
VIR_ONCE_GLOBAL_INIT(virNWFilterBindingObj);
|
2018-05-10 16:21:24 +00:00
|
|
|
|
|
|
|
virNWFilterBindingObjPtr
|
|
|
|
virNWFilterBindingObjNew(void)
|
|
|
|
{
|
|
|
|
if (virNWFilterBindingObjInitialize() < 0)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
return virObjectNew(virNWFilterBindingObjClass);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
virNWFilterBindingObjDispose(void *obj)
|
|
|
|
{
|
|
|
|
virNWFilterBindingObjPtr bobj = obj;
|
|
|
|
|
|
|
|
virNWFilterBindingDefFree(bobj->def);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
virNWFilterBindingDefPtr
|
|
|
|
virNWFilterBindingObjGetDef(virNWFilterBindingObjPtr obj)
|
|
|
|
{
|
|
|
|
return obj->def;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
virNWFilterBindingObjSetDef(virNWFilterBindingObjPtr obj,
|
|
|
|
virNWFilterBindingDefPtr def)
|
|
|
|
{
|
|
|
|
virNWFilterBindingDefFree(obj->def);
|
|
|
|
obj->def = def;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-03-20 08:59:48 +00:00
|
|
|
virNWFilterBindingDefPtr
|
|
|
|
virNWFilterBindingObjStealDef(virNWFilterBindingObjPtr obj)
|
|
|
|
{
|
|
|
|
virNWFilterBindingDefPtr def;
|
|
|
|
|
|
|
|
VIR_STEAL_PTR(def, obj->def);
|
|
|
|
return def;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-05-10 16:21:24 +00:00
|
|
|
bool
|
|
|
|
virNWFilterBindingObjGetRemoving(virNWFilterBindingObjPtr obj)
|
|
|
|
{
|
|
|
|
return obj->removing;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
virNWFilterBindingObjSetRemoving(virNWFilterBindingObjPtr obj,
|
|
|
|
bool removing)
|
|
|
|
{
|
|
|
|
obj->removing = removing;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* virNWFilterBindingObjEndAPI:
|
|
|
|
* @obj: binding object
|
|
|
|
*
|
|
|
|
* Finish working with a binding object in an API. This function
|
|
|
|
* clears whatever was left of a domain that was gathered using
|
|
|
|
* virNWFilterBindingObjListFindByPortDev(). Currently that means
|
|
|
|
* only unlocking and decrementing the reference counter of that
|
|
|
|
* object. And in order to make sure the caller does not access
|
|
|
|
* the object, the pointer is cleared.
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
virNWFilterBindingObjEndAPI(virNWFilterBindingObjPtr *obj)
|
|
|
|
{
|
|
|
|
if (!*obj)
|
|
|
|
return;
|
|
|
|
|
|
|
|
virObjectUnlock(*obj);
|
|
|
|
virObjectUnref(*obj);
|
|
|
|
*obj = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
char *
|
|
|
|
virNWFilterBindingObjConfigFile(const char *dir,
|
|
|
|
const char *name)
|
|
|
|
{
|
|
|
|
char *ret;
|
|
|
|
|
|
|
|
ignore_value(virAsprintf(&ret, "%s/%s.xml", dir, name));
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
virNWFilterBindingObjSave(const virNWFilterBindingObj *obj,
|
|
|
|
const char *statusDir)
|
|
|
|
{
|
|
|
|
char *filename;
|
|
|
|
char *xml = NULL;
|
|
|
|
int ret = -1;
|
|
|
|
|
|
|
|
if (!(filename = virNWFilterBindingObjConfigFile(statusDir,
|
|
|
|
obj->def->portdevname)))
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (!(xml = virNWFilterBindingObjFormat(obj)))
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
if (virFileMakePath(statusDir) < 0) {
|
|
|
|
virReportSystemError(errno,
|
|
|
|
_("cannot create config directory '%s'"),
|
|
|
|
statusDir);
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = virXMLSaveFile(filename,
|
|
|
|
obj->def->portdevname, "nwfilter-binding-create",
|
|
|
|
xml);
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
VIR_FREE(xml);
|
|
|
|
VIR_FREE(filename);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
virNWFilterBindingObjDelete(const virNWFilterBindingObj *obj,
|
|
|
|
const char *statusDir)
|
|
|
|
{
|
|
|
|
char *filename;
|
|
|
|
int ret = -1;
|
|
|
|
|
|
|
|
if (!(filename = virNWFilterBindingObjConfigFile(statusDir,
|
|
|
|
obj->def->portdevname)))
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (unlink(filename) < 0 &&
|
|
|
|
errno != ENOENT) {
|
|
|
|
virReportSystemError(errno,
|
|
|
|
_("Unable to remove status '%s' for nwfilter binding %s'"),
|
|
|
|
filename, obj->def->portdevname);
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = 0;
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
VIR_FREE(filename);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static virNWFilterBindingObjPtr
|
|
|
|
virNWFilterBindingObjParseXML(xmlDocPtr doc,
|
|
|
|
xmlXPathContextPtr ctxt)
|
|
|
|
{
|
|
|
|
virNWFilterBindingObjPtr ret;
|
|
|
|
xmlNodePtr node;
|
|
|
|
|
|
|
|
if (!(ret = virNWFilterBindingObjNew()))
|
|
|
|
return NULL;
|
|
|
|
|
2018-05-10 14:29:46 +00:00
|
|
|
if (!(node = virXPathNode("./filterbinding", ctxt))) {
|
2018-05-10 16:21:24 +00:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("filter binding status missing content"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!(ret->def = virNWFilterBindingDefParseNode(doc, node)))
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
virObjectUnref(ret);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static virNWFilterBindingObjPtr
|
|
|
|
virNWFilterBindingObjParseNode(xmlDocPtr doc,
|
|
|
|
xmlNodePtr root)
|
|
|
|
{
|
|
|
|
xmlXPathContextPtr ctxt = NULL;
|
|
|
|
virNWFilterBindingObjPtr obj = NULL;
|
|
|
|
|
2018-05-10 14:29:46 +00:00
|
|
|
if (STRNEQ((const char *)root->name, "filterbindingstatus")) {
|
|
|
|
virReportError(VIR_ERR_XML_ERROR,
|
|
|
|
_("unknown root element '%s' for filter binding"),
|
|
|
|
root->name);
|
2018-05-10 16:21:24 +00:00
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
2019-09-09 06:33:58 +00:00
|
|
|
if (!(ctxt = virXMLXPathContextNew(doc)))
|
2018-05-10 16:21:24 +00:00
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
ctxt->node = root;
|
|
|
|
obj = virNWFilterBindingObjParseXML(doc, ctxt);
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
xmlXPathFreeContext(ctxt);
|
|
|
|
return obj;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static virNWFilterBindingObjPtr
|
|
|
|
virNWFilterBindingObjParse(const char *xmlStr,
|
|
|
|
const char *filename)
|
|
|
|
{
|
|
|
|
virNWFilterBindingObjPtr obj = NULL;
|
|
|
|
xmlDocPtr xml;
|
|
|
|
|
|
|
|
if ((xml = virXMLParse(filename, xmlStr, _("(nwfilterbinding_status)")))) {
|
|
|
|
obj = virNWFilterBindingObjParseNode(xml, xmlDocGetRootElement(xml));
|
|
|
|
xmlFreeDoc(xml);
|
|
|
|
}
|
|
|
|
|
|
|
|
return obj;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
virNWFilterBindingObjPtr
|
|
|
|
virNWFilterBindingObjParseFile(const char *filename)
|
|
|
|
{
|
|
|
|
return virNWFilterBindingObjParse(NULL, filename);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
char *
|
|
|
|
virNWFilterBindingObjFormat(const virNWFilterBindingObj *obj)
|
|
|
|
{
|
|
|
|
virBuffer buf = VIR_BUFFER_INITIALIZER;
|
|
|
|
|
|
|
|
virBufferAddLit(&buf, "<filterbindingstatus>\n");
|
|
|
|
|
|
|
|
virBufferAdjustIndent(&buf, 2);
|
|
|
|
|
|
|
|
if (virNWFilterBindingDefFormatBuf(&buf, obj->def) < 0) {
|
|
|
|
virBufferFreeAndReset(&buf);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
virBufferAdjustIndent(&buf, -2);
|
|
|
|
virBufferAddLit(&buf, "</filterbindingstatus>\n");
|
|
|
|
|
|
|
|
if (virBufferCheckError(&buf) < 0)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
return virBufferContentAndReset(&buf);
|
|
|
|
}
|