2012-07-13 10:16:23 +00:00
|
|
|
/*
|
2014-03-05 09:50:42 +00:00
|
|
|
* Copyright (C) 2010-2014 Red Hat, Inc.
|
2012-07-13 10:16:23 +00:00
|
|
|
* Copyright IBM Corp. 2008
|
|
|
|
*
|
|
|
|
* lxc_domain.h: LXC domain helpers
|
|
|
|
*
|
|
|
|
* 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 22:30:55 +00:00
|
|
|
* License along with this library. If not, see
|
2012-07-21 10:06:23 +00:00
|
|
|
* <http://www.gnu.org/licenses/>.
|
2012-07-13 10:16:23 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
#include <config.h>
|
|
|
|
|
|
|
|
#include "lxc_domain.h"
|
|
|
|
|
2012-12-12 18:06:53 +00:00
|
|
|
#include "viralloc.h"
|
2012-12-12 17:59:27 +00:00
|
|
|
#include "virlog.h"
|
2012-12-13 18:21:53 +00:00
|
|
|
#include "virerror.h"
|
2015-08-20 13:46:17 +00:00
|
|
|
#include <libxml/xpathInternals.h>
|
|
|
|
#include "virstring.h"
|
|
|
|
#include "virutil.h"
|
|
|
|
#include "virfile.h"
|
2016-05-16 21:36:47 +00:00
|
|
|
#include "virtime.h"
|
2017-07-21 13:51:03 +00:00
|
|
|
#include "virsystemd.h"
|
2019-01-25 11:42:54 +00:00
|
|
|
#include "virinitctl.h"
|
2012-11-22 17:15:54 +00:00
|
|
|
|
|
|
|
#define VIR_FROM_THIS VIR_FROM_LXC
|
2015-08-20 13:46:17 +00:00
|
|
|
#define LXC_NAMESPACE_HREF "http://libvirt.org/schemas/domain/lxc/1.0"
|
2012-07-13 10:16:23 +00:00
|
|
|
|
2019-03-16 18:20:32 +00:00
|
|
|
VIR_ENUM_IMPL(virLXCDomainJob,
|
|
|
|
LXC_JOB_LAST,
|
2016-05-16 21:36:47 +00:00
|
|
|
"none",
|
|
|
|
"query",
|
|
|
|
"destroy",
|
|
|
|
"modify",
|
|
|
|
);
|
|
|
|
|
2014-02-28 12:16:17 +00:00
|
|
|
VIR_LOG_INIT("lxc.lxc_domain");
|
|
|
|
|
2016-05-16 21:36:47 +00:00
|
|
|
static int
|
|
|
|
virLXCDomainObjInitJob(virLXCDomainObjPrivatePtr priv)
|
|
|
|
{
|
|
|
|
memset(&priv->job, 0, sizeof(priv->job));
|
|
|
|
|
|
|
|
if (virCondInit(&priv->job.cond) < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
virLXCDomainObjResetJob(virLXCDomainObjPrivatePtr priv)
|
|
|
|
{
|
|
|
|
struct virLXCDomainJobObj *job = &priv->job;
|
|
|
|
|
|
|
|
job->active = LXC_JOB_NONE;
|
|
|
|
job->owner = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
virLXCDomainObjFreeJob(virLXCDomainObjPrivatePtr priv)
|
|
|
|
{
|
|
|
|
ignore_value(virCondDestroy(&priv->job.cond));
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Give up waiting for mutex after 30 seconds */
|
|
|
|
#define LXC_JOB_WAIT_TIME (1000ull * 30)
|
|
|
|
|
|
|
|
/*
|
|
|
|
* obj must be locked before calling, virLXCDriverPtr must NOT be locked
|
|
|
|
*
|
|
|
|
* This must be called by anything that will change the VM state
|
|
|
|
* in any way
|
|
|
|
*
|
2016-05-20 15:17:01 +00:00
|
|
|
* Upon successful return, the object will have its ref count increased.
|
|
|
|
* Successful calls must be followed by EndJob eventually.
|
2016-05-16 21:36:47 +00:00
|
|
|
*/
|
|
|
|
int
|
|
|
|
virLXCDomainObjBeginJob(virLXCDriverPtr driver ATTRIBUTE_UNUSED,
|
|
|
|
virDomainObjPtr obj,
|
|
|
|
enum virLXCDomainJob job)
|
|
|
|
{
|
|
|
|
virLXCDomainObjPrivatePtr priv = obj->privateData;
|
|
|
|
unsigned long long now;
|
|
|
|
unsigned long long then;
|
|
|
|
|
|
|
|
if (virTimeMillisNow(&now) < 0)
|
|
|
|
return -1;
|
|
|
|
then = now + LXC_JOB_WAIT_TIME;
|
|
|
|
|
|
|
|
while (priv->job.active) {
|
|
|
|
VIR_DEBUG("Wait normal job condition for starting job: %s",
|
|
|
|
virLXCDomainJobTypeToString(job));
|
|
|
|
if (virCondWaitUntil(&priv->job.cond, &obj->parent.lock, then) < 0)
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
|
|
|
virLXCDomainObjResetJob(priv);
|
|
|
|
|
|
|
|
VIR_DEBUG("Starting job: %s", virLXCDomainJobTypeToString(job));
|
|
|
|
priv->job.active = job;
|
|
|
|
priv->job.owner = virThreadSelfID();
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
error:
|
|
|
|
VIR_WARN("Cannot start job (%s) for domain %s;"
|
|
|
|
" current job is (%s) owned by (%d)",
|
|
|
|
virLXCDomainJobTypeToString(job),
|
|
|
|
obj->def->name,
|
|
|
|
virLXCDomainJobTypeToString(priv->job.active),
|
|
|
|
priv->job.owner);
|
|
|
|
|
|
|
|
if (errno == ETIMEDOUT)
|
|
|
|
virReportError(VIR_ERR_OPERATION_TIMEOUT,
|
|
|
|
"%s", _("cannot acquire state change lock"));
|
|
|
|
else
|
|
|
|
virReportSystemError(errno,
|
|
|
|
"%s", _("cannot acquire job mutex"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
2016-05-20 15:17:01 +00:00
|
|
|
* obj must be locked and have a reference before calling
|
2016-05-16 21:36:47 +00:00
|
|
|
*
|
|
|
|
* To be called after completing the work associated with the
|
|
|
|
* earlier virLXCDomainBeginJob() call
|
|
|
|
*/
|
2016-05-20 15:17:01 +00:00
|
|
|
void
|
2016-05-16 21:36:47 +00:00
|
|
|
virLXCDomainObjEndJob(virLXCDriverPtr driver ATTRIBUTE_UNUSED,
|
|
|
|
virDomainObjPtr obj)
|
|
|
|
{
|
|
|
|
virLXCDomainObjPrivatePtr priv = obj->privateData;
|
|
|
|
enum virLXCDomainJob job = priv->job.active;
|
|
|
|
|
|
|
|
VIR_DEBUG("Stopping job: %s",
|
|
|
|
virLXCDomainJobTypeToString(job));
|
|
|
|
|
|
|
|
virLXCDomainObjResetJob(priv);
|
|
|
|
virCondSignal(&priv->job.cond);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static void *
|
2017-07-21 13:29:00 +00:00
|
|
|
virLXCDomainObjPrivateAlloc(void *opaque ATTRIBUTE_UNUSED)
|
2012-07-13 10:16:23 +00:00
|
|
|
{
|
2012-07-18 16:20:46 +00:00
|
|
|
virLXCDomainObjPrivatePtr priv;
|
2012-07-13 10:16:23 +00:00
|
|
|
|
|
|
|
if (VIR_ALLOC(priv) < 0)
|
|
|
|
return NULL;
|
|
|
|
|
2016-05-16 21:36:47 +00:00
|
|
|
if (virLXCDomainObjInitJob(priv) < 0) {
|
|
|
|
VIR_FREE(priv);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2012-07-13 10:16:23 +00:00
|
|
|
return priv;
|
|
|
|
}
|
|
|
|
|
2016-05-16 21:36:47 +00:00
|
|
|
|
|
|
|
static void
|
|
|
|
virLXCDomainObjPrivateFree(void *data)
|
|
|
|
{
|
|
|
|
virLXCDomainObjPrivatePtr priv = data;
|
|
|
|
|
2018-07-30 09:06:31 +00:00
|
|
|
virCgroupFree(&priv->cgroup);
|
2016-05-16 21:36:47 +00:00
|
|
|
virLXCDomainObjFreeJob(priv);
|
|
|
|
VIR_FREE(priv);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2015-08-20 13:46:17 +00:00
|
|
|
VIR_ENUM_IMPL(virLXCDomainNamespace,
|
|
|
|
VIR_LXC_DOMAIN_NAMESPACE_LAST,
|
|
|
|
"sharenet",
|
|
|
|
"shareipc",
|
2019-01-20 16:30:15 +00:00
|
|
|
"shareuts",
|
|
|
|
);
|
2015-08-20 13:46:17 +00:00
|
|
|
|
|
|
|
VIR_ENUM_IMPL(virLXCDomainNamespaceSource,
|
|
|
|
VIR_LXC_DOMAIN_NAMESPACE_SOURCE_LAST,
|
|
|
|
"none",
|
|
|
|
"name",
|
|
|
|
"pid",
|
2019-01-20 16:30:15 +00:00
|
|
|
"netns",
|
|
|
|
);
|
2015-08-20 13:46:17 +00:00
|
|
|
|
|
|
|
static void
|
|
|
|
lxcDomainDefNamespaceFree(void *nsdata)
|
|
|
|
{
|
|
|
|
size_t i;
|
|
|
|
lxcDomainDefPtr lxcDef = nsdata;
|
|
|
|
for (i = 0; i < VIR_LXC_DOMAIN_NAMESPACE_LAST; i++)
|
|
|
|
VIR_FREE(lxcDef->ns_val[i]);
|
|
|
|
VIR_FREE(nsdata);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
lxcDomainDefNamespaceParse(xmlDocPtr xml ATTRIBUTE_UNUSED,
|
|
|
|
xmlNodePtr root ATTRIBUTE_UNUSED,
|
|
|
|
xmlXPathContextPtr ctxt,
|
|
|
|
void **data)
|
|
|
|
{
|
|
|
|
lxcDomainDefPtr lxcDef = NULL;
|
|
|
|
xmlNodePtr *nodes = NULL;
|
|
|
|
bool uses_lxc_ns = false;
|
|
|
|
xmlNodePtr node;
|
|
|
|
int feature;
|
|
|
|
int n;
|
|
|
|
char *tmp = NULL;
|
|
|
|
size_t i;
|
|
|
|
|
|
|
|
if (xmlXPathRegisterNs(ctxt, BAD_CAST "lxc", BAD_CAST LXC_NAMESPACE_HREF) < 0) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("Failed to register xml namespace '%s'"),
|
|
|
|
LXC_NAMESPACE_HREF);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (VIR_ALLOC(lxcDef) < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
node = ctxt->node;
|
|
|
|
if ((n = virXPathNodeSet("./lxc:namespace/*", ctxt, &nodes)) < 0)
|
|
|
|
goto error;
|
|
|
|
uses_lxc_ns |= n > 0;
|
|
|
|
|
|
|
|
for (i = 0; i < n; i++) {
|
|
|
|
if ((feature = virLXCDomainNamespaceTypeFromString(
|
2018-04-25 12:42:34 +00:00
|
|
|
(const char *)nodes[i]->name)) < 0) {
|
2015-08-20 13:46:17 +00:00
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED,
|
|
|
|
_("unsupported Namespace feature: %s"),
|
|
|
|
nodes[i]->name);
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
|
|
|
ctxt->node = nodes[i];
|
|
|
|
|
|
|
|
if (!(tmp = virXMLPropString(nodes[i], "type"))) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
"%s", _("No lxc environment type specified"));
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
if ((lxcDef->ns_source[feature] =
|
|
|
|
virLXCDomainNamespaceSourceTypeFromString(tmp)) < 0) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("Unknown LXC namespace source '%s'"),
|
|
|
|
tmp);
|
|
|
|
VIR_FREE(tmp);
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
VIR_FREE(tmp);
|
|
|
|
|
|
|
|
if (!(lxcDef->ns_val[feature] =
|
|
|
|
virXMLPropString(nodes[i], "value"))) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
"%s", _("No lxc environment type specified"));
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
VIR_FREE(nodes);
|
|
|
|
ctxt->node = node;
|
|
|
|
if (uses_lxc_ns)
|
|
|
|
*data = lxcDef;
|
|
|
|
else
|
|
|
|
VIR_FREE(lxcDef);
|
|
|
|
return 0;
|
|
|
|
error:
|
|
|
|
VIR_FREE(nodes);
|
|
|
|
lxcDomainDefNamespaceFree(lxcDef);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static int
|
|
|
|
lxcDomainDefNamespaceFormatXML(virBufferPtr buf,
|
|
|
|
void *nsdata)
|
|
|
|
{
|
|
|
|
lxcDomainDefPtr lxcDef = nsdata;
|
|
|
|
size_t i;
|
|
|
|
|
|
|
|
if (!lxcDef)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
virBufferAddLit(buf, "<lxc:namespace>\n");
|
|
|
|
virBufferAdjustIndent(buf, 2);
|
|
|
|
|
|
|
|
for (i = 0; i < VIR_LXC_DOMAIN_NAMESPACE_LAST; i++) {
|
|
|
|
if (lxcDef->ns_source[i] == VIR_LXC_DOMAIN_NAMESPACE_SOURCE_NONE)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
virBufferAsprintf(buf, "<lxc:%s type='%s' value='%s'/>\n",
|
|
|
|
virLXCDomainNamespaceTypeToString(i),
|
|
|
|
virLXCDomainNamespaceSourceTypeToString(
|
|
|
|
lxcDef->ns_source[i]),
|
|
|
|
lxcDef->ns_val[i]);
|
|
|
|
}
|
|
|
|
|
|
|
|
virBufferAdjustIndent(buf, -2);
|
|
|
|
virBufferAddLit(buf, "</lxc:namespace>\n");
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static const char *
|
|
|
|
lxcDomainDefNamespaceHref(void)
|
|
|
|
{
|
|
|
|
return "xmlns:lxc='" LXC_NAMESPACE_HREF "'";
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
virDomainXMLNamespace virLXCDriverDomainXMLNamespace = {
|
|
|
|
.parse = lxcDomainDefNamespaceParse,
|
|
|
|
.free = lxcDomainDefNamespaceFree,
|
|
|
|
.format = lxcDomainDefNamespaceFormatXML,
|
|
|
|
.href = lxcDomainDefNamespaceHref,
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2015-05-19 08:14:19 +00:00
|
|
|
static int
|
|
|
|
virLXCDomainObjPrivateXMLFormat(virBufferPtr buf,
|
|
|
|
virDomainObjPtr vm)
|
2012-11-22 17:15:54 +00:00
|
|
|
{
|
2015-05-19 08:14:19 +00:00
|
|
|
virLXCDomainObjPrivatePtr priv = vm->privateData;
|
2012-11-22 17:15:54 +00:00
|
|
|
|
2016-10-06 14:54:41 +00:00
|
|
|
virBufferAsprintf(buf, "<init pid='%lld'/>\n",
|
2018-04-25 12:42:34 +00:00
|
|
|
(long long)priv->initpid);
|
2012-11-22 17:15:54 +00:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2015-05-19 08:14:19 +00:00
|
|
|
static int
|
|
|
|
virLXCDomainObjPrivateXMLParse(xmlXPathContextPtr ctxt,
|
2015-07-24 17:35:00 +00:00
|
|
|
virDomainObjPtr vm,
|
|
|
|
virDomainDefParserConfigPtr config ATTRIBUTE_UNUSED)
|
2012-11-22 17:15:54 +00:00
|
|
|
{
|
2015-05-19 08:14:19 +00:00
|
|
|
virLXCDomainObjPrivatePtr priv = vm->privateData;
|
2016-10-06 14:54:41 +00:00
|
|
|
long long thepid;
|
2012-11-22 17:15:54 +00:00
|
|
|
|
2016-10-06 14:54:41 +00:00
|
|
|
if (virXPathLongLong("string(./init[1]/@pid)", ctxt, &thepid) < 0) {
|
2016-05-19 19:10:19 +00:00
|
|
|
VIR_WARN("Failed to load init pid from state %s",
|
|
|
|
virGetLastErrorMessage());
|
2012-11-22 17:15:54 +00:00
|
|
|
priv->initpid = 0;
|
|
|
|
} else {
|
|
|
|
priv->initpid = thepid;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2013-03-05 15:17:24 +00:00
|
|
|
virDomainXMLPrivateDataCallbacks virLXCDriverPrivateDataCallbacks = {
|
|
|
|
.alloc = virLXCDomainObjPrivateAlloc,
|
|
|
|
.free = virLXCDomainObjPrivateFree,
|
|
|
|
.format = virLXCDomainObjPrivateXMLFormat,
|
|
|
|
.parse = virLXCDomainObjPrivateXMLParse,
|
|
|
|
};
|
2013-03-11 11:12:08 +00:00
|
|
|
|
|
|
|
static int
|
|
|
|
virLXCDomainDefPostParse(virDomainDefPtr def,
|
|
|
|
virCapsPtr caps,
|
2016-01-08 13:00:56 +00:00
|
|
|
unsigned int parseFlags ATTRIBUTE_UNUSED,
|
2016-09-22 14:41:33 +00:00
|
|
|
void *opaque ATTRIBUTE_UNUSED,
|
|
|
|
void *parseOpaque ATTRIBUTE_UNUSED)
|
2013-03-11 11:12:08 +00:00
|
|
|
{
|
|
|
|
/* check for emulator and create a default one if needed */
|
|
|
|
if (!def->emulator &&
|
|
|
|
!(def->emulator = virDomainDefGetDefaultEmulator(def, caps)))
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2013-03-13 20:39:34 +00:00
|
|
|
|
|
|
|
static int
|
|
|
|
virLXCDomainDeviceDefPostParse(virDomainDeviceDefPtr dev,
|
maint: avoid 'const fooPtr' in domain_conf
'const fooPtr' is the same as 'foo * const' (the pointer won't
change, but it's contents can). But in general, if an interface
is trying to be const-correct, it should be using 'const foo *'
(the pointer is to data that can't be changed).
Fix up offenders in src/conf/domain_conf, and their fallout.
Several things to note: virObjectLock() requires a non-const
argument; if this were C++, we could treat the locking field
as 'mutable' and allow locking an otherwise 'const' object, but
that is a more invasive change, so I instead dropped attempts
to be const-correct on domain lookup. virXMLPropString and
friends require a non-const xmlNodePtr - this is because libxml2
is not a const-correct library. We could make the src/util/virxml
wrappers cast away const, but I figured it was easier to not
try to mark xmlNodePtr as const. Finally, virDomainDeviceDefCopy
was a rather hard conversion - it calls virDomainDeviceDefPostParse,
which in turn in the xen driver was actually modifying the domain
outside of the current device being visited. We should not be
adding a device on the first per-device callback, but waiting until
after all per-device callbacks are complete.
* src/conf/domain_conf.h (virDomainObjListFindByID)
(virDomainObjListFindByUUID, virDomainObjListFindByName)
(virDomainObjAssignDef, virDomainObjListAdd): Drop attempt at
const.
(virDomainDeviceDefCopy): Use intended type.
(virDomainDeviceDefParse, virDomainDeviceDefPostParseCallback)
(virDomainVideoDefaultType, virDomainVideoDefaultRAM)
(virDomainChrGetDomainPtrs): Make const-correct.
* src/conf/domain_conf.c (virDomainObjListFindByID)
(virDomainObjListFindByUUID, virDomainObjListFindByName)
(virDomainDeviceDefCopy, virDomainObjListAdd)
(virDomainObjAssignDef, virDomainHostdevSubsysUsbDefParseXML)
(virDomainHostdevSubsysPciOrigStatesDefParseXML)
(virDomainHostdevSubsysPciDefParseXML)
(virDomainHostdevSubsysScsiDefParseXML)
(virDomainControllerModelTypeFromString)
(virDomainTPMDefParseXML, virDomainTimerDefParseXML)
(virDomainSoundCodecDefParseXML, virDomainSoundDefParseXML)
(virDomainWatchdogDefParseXML, virDomainRNGDefParseXML)
(virDomainMemballoonDefParseXML, virDomainNVRAMDefParseXML)
(virSysinfoParseXML, virDomainVideoAccelDefParseXML)
(virDomainVideoDefParseXML, virDomainHostdevDefParseXML)
(virDomainRedirdevDefParseXML)
(virDomainRedirFilterUsbDevDefParseXML)
(virDomainRedirFilterDefParseXML, virDomainIdMapEntrySort)
(virDomainIdmapDefParseXML, virDomainVcpuPinDefParseXML)
(virDiskNameToBusDeviceIndex, virDomainDeviceDefCopy)
(virDomainVideoDefaultType, virDomainHostdevAssignAddress)
(virDomainDeviceDefPostParseInternal, virDomainDeviceDefPostParse)
(virDomainChrGetDomainPtrs, virDomainControllerSCSINextUnit)
(virDomainSCSIDriveAddressIsUsed)
(virDomainDriveAddressIsUsedByDisk)
(virDomainDriveAddressIsUsedByHostdev): Fix fallout.
* src/openvz/openvz_driver.c (openvzDomainDeviceDefPostParse):
Likewise.
* src/libxl/libxl_domain.c (libxlDomainDeviceDefPostParse):
Likewise.
* src/qemu/qemu_domain.c (qemuDomainDeviceDefPostParse)
(qemuDomainDefaultNetModel): Likewise.
* src/lxc/lxc_domain.c (virLXCDomainDeviceDefPostParse):
Likewise.
* src/uml/uml_driver.c (umlDomainDeviceDefPostParse): Likewise.
* src/xen/xen_driver.c (xenDomainDeviceDefPostParse): Split...
(xenDomainDefPostParse): ...since per-device callback is not the
time to be adding a device.
Signed-off-by: Eric Blake <eblake@redhat.com>
2013-10-08 15:08:25 +00:00
|
|
|
const virDomainDef *def ATTRIBUTE_UNUSED,
|
2013-03-13 20:39:34 +00:00
|
|
|
virCapsPtr caps ATTRIBUTE_UNUSED,
|
2016-01-08 13:00:56 +00:00
|
|
|
unsigned int parseFlags ATTRIBUTE_UNUSED,
|
2016-09-23 08:53:38 +00:00
|
|
|
void *opaque ATTRIBUTE_UNUSED,
|
|
|
|
void *parseOpaque ATTRIBUTE_UNUSED)
|
2013-03-13 20:39:34 +00:00
|
|
|
{
|
|
|
|
if (dev->type == VIR_DOMAIN_DEVICE_CHR &&
|
|
|
|
dev->data.chr->deviceType == VIR_DOMAIN_CHR_DEVICE_TYPE_CONSOLE &&
|
|
|
|
dev->data.chr->targetType == VIR_DOMAIN_CHR_CONSOLE_TARGET_TYPE_NONE)
|
|
|
|
dev->data.chr->targetType = VIR_DOMAIN_CHR_CONSOLE_TARGET_TYPE_LXC;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-03-11 11:12:08 +00:00
|
|
|
virDomainDefParserConfig virLXCDriverDomainDefParserConfig = {
|
|
|
|
.domainPostParseCallback = virLXCDomainDefPostParse,
|
2013-03-13 20:39:34 +00:00
|
|
|
.devicesPostParseCallback = virLXCDomainDeviceDefPostParse,
|
2013-03-11 11:12:08 +00:00
|
|
|
};
|
2017-07-21 13:51:03 +00:00
|
|
|
|
|
|
|
|
|
|
|
char *
|
|
|
|
virLXCDomainGetMachineName(virDomainDefPtr def, pid_t pid)
|
|
|
|
{
|
|
|
|
char *ret = NULL;
|
|
|
|
|
|
|
|
if (pid) {
|
|
|
|
ret = virSystemdGetMachineNameByPID(pid);
|
|
|
|
if (!ret)
|
|
|
|
virResetLastError();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!ret)
|
|
|
|
ret = virDomainGenerateMachineName("lxc", def->id, def->name, true);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
2019-01-25 11:42:54 +00:00
|
|
|
|
|
|
|
|
|
|
|
typedef struct _lxcDomainInitctlCallbackData lxcDomainInitctlCallbackData;
|
|
|
|
struct _lxcDomainInitctlCallbackData {
|
|
|
|
int runlevel;
|
|
|
|
bool *st_valid;
|
|
|
|
struct stat *st;
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
static int
|
|
|
|
lxcDomainInitctlCallback(pid_t pid ATTRIBUTE_UNUSED,
|
|
|
|
void *opaque)
|
|
|
|
{
|
|
|
|
lxcDomainInitctlCallbackData *data = opaque;
|
|
|
|
size_t i;
|
|
|
|
|
|
|
|
for (i = 0; virInitctlFifos[i]; i++) {
|
|
|
|
const char *fifo = virInitctlFifos[i];
|
|
|
|
struct stat cont_sb;
|
|
|
|
|
|
|
|
if (stat(fifo, &cont_sb) < 0) {
|
|
|
|
if (errno == ENOENT)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
virReportSystemError(errno, _("Unable to stat %s"), fifo);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Check if the init fifo is not the very one that's on
|
|
|
|
* the host. We don't want to change the host's runlevel.
|
|
|
|
*/
|
|
|
|
if (data->st_valid[i] &&
|
|
|
|
data->st[i].st_dev == cont_sb.st_dev &&
|
|
|
|
data->st[i].st_ino == cont_sb.st_ino)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
return virInitctlSetRunLevel(fifo, data->runlevel);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* If no usable fifo was found then declare success. Caller
|
|
|
|
* will try killing the domain with signal. */
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
virLXCDomainSetRunlevel(virDomainObjPtr vm,
|
|
|
|
int runlevel)
|
|
|
|
{
|
|
|
|
virLXCDomainObjPrivatePtr priv = vm->privateData;
|
|
|
|
lxcDomainInitctlCallbackData data;
|
|
|
|
size_t nfifos = 0;
|
|
|
|
size_t i;
|
|
|
|
int ret = -1;
|
|
|
|
|
|
|
|
memset(&data, 0, sizeof(data));
|
|
|
|
|
|
|
|
data.runlevel = runlevel;
|
|
|
|
|
|
|
|
for (nfifos = 0; virInitctlFifos[nfifos]; nfifos++)
|
|
|
|
;
|
|
|
|
|
|
|
|
if (VIR_ALLOC_N(data.st, nfifos) < 0 ||
|
|
|
|
VIR_ALLOC_N(data.st_valid, nfifos) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
for (i = 0; virInitctlFifos[i]; i++) {
|
|
|
|
const char *fifo = virInitctlFifos[i];
|
|
|
|
|
|
|
|
if (stat(fifo, &(data.st[i])) < 0) {
|
|
|
|
if (errno == ENOENT)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
virReportSystemError(errno, _("Unable to stat %s"), fifo);
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
data.st_valid[i] = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = virProcessRunInMountNamespace(priv->initpid,
|
|
|
|
lxcDomainInitctlCallback,
|
|
|
|
&data);
|
|
|
|
cleanup:
|
|
|
|
VIR_FREE(data.st);
|
|
|
|
VIR_FREE(data.st_valid);
|
|
|
|
return ret;
|
|
|
|
}
|