提交 e6afacb0 编写于 作者: D Daniel P. Berrangé

qemu: start/stop an event loop thread for domains

The event loop thread will be responsible for handling
any per-domain I/O operations, most notably the QEMU
monitor and agent sockets.

We start this event loop when launching QEMU, but stopping
the event loop is a little more complicated. The obvious
idea is to stop it in qemuProcessStop(), but if we do that
we risk loosing the final events from the QEMU monitor, as
they might not have been read by the event thread at the
time we tell the thread to stop.

The solution is to delay shutdown of the event thread until
we have seen EOF from the QEMU monitor, and thus we know
there are no further events to process.

Note that this assumes that we don't have events to process
from the QEMU agent.
Reviewed-by: NMichal Privoznik <mprivozn@redhat.com>
Signed-off-by: NDaniel P. Berrangé <berrange@redhat.com>
上级 5c146f6e
...@@ -2150,6 +2150,33 @@ dbusVMStateHashFree(void *opaque) ...@@ -2150,6 +2150,33 @@ dbusVMStateHashFree(void *opaque)
} }
int
qemuDomainObjStartWorker(virDomainObjPtr dom)
{
qemuDomainObjPrivatePtr priv = dom->privateData;
if (!priv->eventThread) {
g_autofree char *threadName = g_strdup_printf("vm-%s", dom->def->name);
if (!(priv->eventThread = virEventThreadNew(threadName)))
return -1;
}
return 0;
}
void
qemuDomainObjStopWorker(virDomainObjPtr dom)
{
qemuDomainObjPrivatePtr priv = dom->privateData;
if (priv->eventThread) {
g_object_unref(priv->eventThread);
priv->eventThread = NULL;
}
}
static void * static void *
qemuDomainObjPrivateAlloc(void *opaque) qemuDomainObjPrivateAlloc(void *opaque)
{ {
...@@ -2289,6 +2316,12 @@ qemuDomainObjPrivateFree(void *data) ...@@ -2289,6 +2316,12 @@ qemuDomainObjPrivateFree(void *data)
virHashFree(priv->blockjobs); virHashFree(priv->blockjobs);
virHashFree(priv->dbusVMStates); virHashFree(priv->dbusVMStates);
/* This should never be non-NULL if we get here, but just in case... */
if (priv->eventThread) {
VIR_ERROR(_("Unexpected event thread still active during domain deletion"));
g_object_unref(priv->eventThread);
}
VIR_FREE(priv); VIR_FREE(priv);
} }
......
...@@ -40,6 +40,7 @@ ...@@ -40,6 +40,7 @@
#include "logging/log_manager.h" #include "logging/log_manager.h"
#include "virdomainmomentobjlist.h" #include "virdomainmomentobjlist.h"
#include "virenum.h" #include "virenum.h"
#include "vireventthread.h"
#define QEMU_DOMAIN_FORMAT_LIVE_FLAGS \ #define QEMU_DOMAIN_FORMAT_LIVE_FLAGS \
(VIR_DOMAIN_XML_SECURE) (VIR_DOMAIN_XML_SECURE)
...@@ -300,6 +301,8 @@ struct _qemuDomainObjPrivate { ...@@ -300,6 +301,8 @@ struct _qemuDomainObjPrivate {
virBitmapPtr namespaces; virBitmapPtr namespaces;
virEventThread *eventThread;
qemuMonitorPtr mon; qemuMonitorPtr mon;
virDomainChrSourceDefPtr monConfig; virDomainChrSourceDefPtr monConfig;
bool monError; bool monError;
...@@ -630,6 +633,9 @@ struct _qemuDomainXmlNsDef { ...@@ -630,6 +633,9 @@ struct _qemuDomainXmlNsDef {
char **capsdel; char **capsdel;
}; };
int qemuDomainObjStartWorker(virDomainObjPtr dom);
void qemuDomainObjStopWorker(virDomainObjPtr dom);
virDomainObjPtr qemuDomainObjFromDomain(virDomainPtr domain); virDomainObjPtr qemuDomainObjFromDomain(virDomainPtr domain);
qemuDomainSaveCookiePtr qemuDomainSaveCookieNew(virDomainObjPtr vm); qemuDomainSaveCookiePtr qemuDomainSaveCookieNew(virDomainObjPtr vm);
......
...@@ -320,6 +320,9 @@ qemuProcessHandleMonitorEOF(qemuMonitorPtr mon, ...@@ -320,6 +320,9 @@ qemuProcessHandleMonitorEOF(qemuMonitorPtr mon,
qemuDomainDestroyNamespace(driver, vm); qemuDomainDestroyNamespace(driver, vm);
cleanup: cleanup:
/* Now we got EOF we're not expecting more I/O, so we
* can finally kill the event thread */
qemuDomainObjStopWorker(vm);
virObjectUnlock(vm); virObjectUnlock(vm);
} }
...@@ -6908,6 +6911,9 @@ qemuProcessLaunch(virConnectPtr conn, ...@@ -6908,6 +6911,9 @@ qemuProcessLaunch(virConnectPtr conn,
if (rv == -1) /* The VM failed to start */ if (rv == -1) /* The VM failed to start */
goto cleanup; goto cleanup;
if (qemuDomainObjStartWorker(vm) < 0)
goto cleanup;
VIR_DEBUG("Waiting for monitor to show up"); VIR_DEBUG("Waiting for monitor to show up");
if (qemuProcessWaitForMonitor(driver, vm, asyncJob, logCtxt) < 0) if (qemuProcessWaitForMonitor(driver, vm, asyncJob, logCtxt) < 0)
goto cleanup; goto cleanup;
...@@ -7390,6 +7396,18 @@ void qemuProcessStop(virQEMUDriverPtr driver, ...@@ -7390,6 +7396,18 @@ void qemuProcessStop(virQEMUDriverPtr driver,
priv->monConfig = NULL; priv->monConfig = NULL;
} }
/*
* We cannot stop the event thread at this time. When
* we are in this code, we may not yet have processed the
* STOP event or EOF from the monitor. So the event loop
* may have pending input that we need to process still.
* The qemuProcessHandleMonitorEOF method will kill
* the event thread because at that point we don't
* expect any more I/O from the QEMU monitor. We are
* assuming we don't need to get any more events from the
* QEMU agent at that time.
*/
/* Remove the master key */ /* Remove the master key */
qemuDomainMasterKeyRemove(priv); qemuDomainMasterKeyRemove(priv);
...@@ -7981,6 +7999,9 @@ qemuProcessReconnect(void *opaque) ...@@ -7981,6 +7999,9 @@ qemuProcessReconnect(void *opaque)
virQEMUCapsGet(priv->qemuCaps, QEMU_CAPS_CHARDEV_FD_PASS)) virQEMUCapsGet(priv->qemuCaps, QEMU_CAPS_CHARDEV_FD_PASS))
retry = false; retry = false;
if (qemuDomainObjStartWorker(obj) < 0)
goto error;
VIR_DEBUG("Reconnect monitor to def=%p name='%s' retry=%d", VIR_DEBUG("Reconnect monitor to def=%p name='%s' retry=%d",
obj, obj->def->name, retry); obj, obj->def->name, retry);
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册