mirror of
https://gitlab.com/libvirt/libvirt.git
synced 2025-01-12 07:42:56 +00:00
qemu: Add ability to abort existing console while creating new one
This patch fixes console corruption, that happens if two concurrent sessions are opened for a single console on a domain. Result of this corruption was that each of the console streams recieved just a part of the data written to the pipe so every console rendered unusable. New helper function for safe console handling is used to establish the console stream connection. This function ensures that no other libvirt client is using the console (with the ability to disconnect consoles of libvirt clients) and that no UUCP style lockfile is placed on the PTY device. * src/qemu/qemu_domain.h - add data structure to domain's private data dealing with console connections * src/qemu/qemu_domain.c: - allocate/free domain's console data structure * src/qemu/qemu_driver.c - use the new helper function for console handling
This commit is contained in:
parent
3f4238d771
commit
4716138229
@ -192,6 +192,9 @@ static void *qemuDomainObjPrivateAlloc(void)
|
|||||||
if (qemuDomainObjInitJob(priv) < 0)
|
if (qemuDomainObjInitJob(priv) < 0)
|
||||||
goto error;
|
goto error;
|
||||||
|
|
||||||
|
if (!(priv->cons = virConsoleAlloc()))
|
||||||
|
goto error;
|
||||||
|
|
||||||
priv->migMaxBandwidth = QEMU_DOMAIN_DEFAULT_MIG_BANDWIDTH_MAX;
|
priv->migMaxBandwidth = QEMU_DOMAIN_DEFAULT_MIG_BANDWIDTH_MAX;
|
||||||
|
|
||||||
return priv;
|
return priv;
|
||||||
@ -214,6 +217,8 @@ static void qemuDomainObjPrivateFree(void *data)
|
|||||||
VIR_FREE(priv->lockState);
|
VIR_FREE(priv->lockState);
|
||||||
VIR_FREE(priv->origname);
|
VIR_FREE(priv->origname);
|
||||||
|
|
||||||
|
virConsoleFree(priv->cons);
|
||||||
|
|
||||||
/* This should never be non-NULL if we get here, but just in case... */
|
/* This should never be non-NULL if we get here, but just in case... */
|
||||||
if (priv->mon) {
|
if (priv->mon) {
|
||||||
VIR_ERROR(_("Unexpected QEMU monitor still active during domain deletion"));
|
VIR_ERROR(_("Unexpected QEMU monitor still active during domain deletion"));
|
||||||
|
@ -30,6 +30,7 @@
|
|||||||
# include "qemu_agent.h"
|
# include "qemu_agent.h"
|
||||||
# include "qemu_conf.h"
|
# include "qemu_conf.h"
|
||||||
# include "bitmap.h"
|
# include "bitmap.h"
|
||||||
|
# include "virconsole.h"
|
||||||
|
|
||||||
# define QEMU_EXPECTED_VIRT_TYPES \
|
# define QEMU_EXPECTED_VIRT_TYPES \
|
||||||
((1 << VIR_DOMAIN_VIRT_QEMU) | \
|
((1 << VIR_DOMAIN_VIRT_QEMU) | \
|
||||||
@ -127,6 +128,8 @@ struct _qemuDomainObjPrivate {
|
|||||||
|
|
||||||
unsigned long migMaxBandwidth;
|
unsigned long migMaxBandwidth;
|
||||||
char *origname;
|
char *origname;
|
||||||
|
|
||||||
|
virConsolesPtr cons;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct qemuDomainWatchdogEvent
|
struct qemuDomainWatchdogEvent
|
||||||
|
@ -11244,8 +11244,10 @@ qemuDomainOpenConsole(virDomainPtr dom,
|
|||||||
int ret = -1;
|
int ret = -1;
|
||||||
int i;
|
int i;
|
||||||
virDomainChrDefPtr chr = NULL;
|
virDomainChrDefPtr chr = NULL;
|
||||||
|
qemuDomainObjPrivatePtr priv;
|
||||||
|
|
||||||
virCheckFlags(0, -1);
|
virCheckFlags(VIR_DOMAIN_CONSOLE_SAFE |
|
||||||
|
VIR_DOMAIN_CONSOLE_FORCE, -1);
|
||||||
|
|
||||||
qemuDriverLock(driver);
|
qemuDriverLock(driver);
|
||||||
virUUIDFormat(dom->uuid, uuidstr);
|
virUUIDFormat(dom->uuid, uuidstr);
|
||||||
@ -11262,6 +11264,8 @@ qemuDomainOpenConsole(virDomainPtr dom,
|
|||||||
goto cleanup;
|
goto cleanup;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
priv = vm->privateData;
|
||||||
|
|
||||||
if (dev_name) {
|
if (dev_name) {
|
||||||
for (i = 0 ; !chr && i < vm->def->nconsoles ; i++) {
|
for (i = 0 ; !chr && i < vm->def->nconsoles ; i++) {
|
||||||
if (vm->def->consoles[i]->info.alias &&
|
if (vm->def->consoles[i]->info.alias &&
|
||||||
@ -11297,11 +11301,18 @@ qemuDomainOpenConsole(virDomainPtr dom,
|
|||||||
goto cleanup;
|
goto cleanup;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (virFDStreamOpenFile(st, chr->source.data.file.path,
|
/* handle mutually exclusive access to console devices */
|
||||||
0, 0, O_RDWR) < 0)
|
ret = virConsoleOpen(priv->cons,
|
||||||
goto cleanup;
|
chr->source.data.file.path,
|
||||||
|
st,
|
||||||
|
(flags & VIR_DOMAIN_CONSOLE_FORCE) != 0);
|
||||||
|
|
||||||
|
if (ret == 1) {
|
||||||
|
qemuReportError(VIR_ERR_OPERATION_FAILED,
|
||||||
|
_("Active console session exists for this domain"));
|
||||||
|
ret = -1;
|
||||||
|
}
|
||||||
|
|
||||||
ret = 0;
|
|
||||||
cleanup:
|
cleanup:
|
||||||
if (vm)
|
if (vm)
|
||||||
virDomainObjUnlock(vm);
|
virDomainObjUnlock(vm);
|
||||||
|
Loading…
x
Reference in New Issue
Block a user