mirror of
https://gitlab.com/libvirt/libvirt.git
synced 2025-01-23 21:15:20 +00:00
Fix busy wait loop in LXC container I/O handling
If the host side of an LXC container console disconnected and the guest side continued to write data, until the PTY buffer filled up, the LXC controller would busy wait. It would repeatedly see POLLHUP from poll() and not disable the watch. This was due to some bogus logic detecting blocking conditions. Upon seeing a POLLHUP we must disable all reading & writing from the PTY, and setup the epoll to wake us up again when the connection comes back. Signed-off-by: Daniel P. Berrange <berrange@redhat.com> (cherry picked from commit 5087a5a0092853702eb5e0c0297937a7859bcab3)
This commit is contained in:
parent
5ee95caaeb
commit
3d9a32163f
@ -76,13 +76,11 @@ struct _virLXCControllerConsole {
|
|||||||
int hostFd; /* PTY FD in the host OS */
|
int hostFd; /* PTY FD in the host OS */
|
||||||
bool hostClosed;
|
bool hostClosed;
|
||||||
int hostEpoll;
|
int hostEpoll;
|
||||||
bool hostBlocking;
|
|
||||||
|
|
||||||
int contWatch;
|
int contWatch;
|
||||||
int contFd; /* PTY FD in the container */
|
int contFd; /* PTY FD in the container */
|
||||||
bool contClosed;
|
bool contClosed;
|
||||||
int contEpoll;
|
int contEpoll;
|
||||||
bool contBlocking;
|
|
||||||
|
|
||||||
int epollWatch;
|
int epollWatch;
|
||||||
int epollFd; /* epoll FD for dealing with EOF */
|
int epollFd; /* epoll FD for dealing with EOF */
|
||||||
@ -804,12 +802,15 @@ static void virLXCControllerSignalChildIO(virNetServerPtr server,
|
|||||||
int status;
|
int status;
|
||||||
|
|
||||||
ret = waitpid(-1, &status, WNOHANG);
|
ret = waitpid(-1, &status, WNOHANG);
|
||||||
|
VIR_DEBUG("Got sig child %d vs %lld", ret, (unsigned long long)ctrl->initpid);
|
||||||
if (ret == ctrl->initpid) {
|
if (ret == ctrl->initpid) {
|
||||||
virNetServerQuit(server);
|
virNetServerQuit(server);
|
||||||
virMutexLock(&lock);
|
virMutexLock(&lock);
|
||||||
if (WIFSIGNALED(status) &&
|
if (WIFSIGNALED(status) &&
|
||||||
WTERMSIG(status) == SIGHUP)
|
WTERMSIG(status) == SIGHUP) {
|
||||||
|
VIR_DEBUG("Status indicates reboot");
|
||||||
wantReboot = true;
|
wantReboot = true;
|
||||||
|
}
|
||||||
virMutexUnlock(&lock);
|
virMutexUnlock(&lock);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -820,28 +821,32 @@ static void virLXCControllerConsoleUpdateWatch(virLXCControllerConsolePtr consol
|
|||||||
int hostEvents = 0;
|
int hostEvents = 0;
|
||||||
int contEvents = 0;
|
int contEvents = 0;
|
||||||
|
|
||||||
if (!console->hostClosed || (!console->hostBlocking && console->fromContLen)) {
|
/* If host console is open, then we can look to read/write */
|
||||||
|
if (!console->hostClosed) {
|
||||||
if (console->fromHostLen < sizeof(console->fromHostBuf))
|
if (console->fromHostLen < sizeof(console->fromHostBuf))
|
||||||
hostEvents |= VIR_EVENT_HANDLE_READABLE;
|
hostEvents |= VIR_EVENT_HANDLE_READABLE;
|
||||||
if (console->fromContLen)
|
if (console->fromContLen)
|
||||||
hostEvents |= VIR_EVENT_HANDLE_WRITABLE;
|
hostEvents |= VIR_EVENT_HANDLE_WRITABLE;
|
||||||
}
|
}
|
||||||
if (!console->contClosed || (!console->contBlocking && console->fromHostLen)) {
|
|
||||||
|
/* If cont console is open, then we can look to read/write */
|
||||||
|
if (!console->contClosed) {
|
||||||
if (console->fromContLen < sizeof(console->fromContBuf))
|
if (console->fromContLen < sizeof(console->fromContBuf))
|
||||||
contEvents |= VIR_EVENT_HANDLE_READABLE;
|
contEvents |= VIR_EVENT_HANDLE_READABLE;
|
||||||
if (console->fromHostLen)
|
if (console->fromHostLen)
|
||||||
contEvents |= VIR_EVENT_HANDLE_WRITABLE;
|
contEvents |= VIR_EVENT_HANDLE_WRITABLE;
|
||||||
}
|
}
|
||||||
|
|
||||||
VIR_DEBUG("Container watch %d=%d host watch %d=%d",
|
VIR_DEBUG("Container watch=%d, events=%d closed=%d; host watch=%d events=%d closed=%d",
|
||||||
console->contWatch, contEvents,
|
console->contWatch, contEvents, console->contClosed,
|
||||||
console->hostWatch, hostEvents);
|
console->hostWatch, hostEvents, console->hostClosed);
|
||||||
virEventUpdateHandle(console->contWatch, contEvents);
|
virEventUpdateHandle(console->contWatch, contEvents);
|
||||||
virEventUpdateHandle(console->hostWatch, hostEvents);
|
virEventUpdateHandle(console->hostWatch, hostEvents);
|
||||||
|
|
||||||
if (console->hostClosed) {
|
if (console->hostClosed) {
|
||||||
|
/* Must setup an epoll to detect when host becomes accessible again */
|
||||||
int events = EPOLLIN | EPOLLET;
|
int events = EPOLLIN | EPOLLET;
|
||||||
if (console->hostBlocking)
|
if (console->fromContLen)
|
||||||
events |= EPOLLOUT;
|
events |= EPOLLOUT;
|
||||||
|
|
||||||
if (events != console->hostEpoll) {
|
if (events != console->hostEpoll) {
|
||||||
@ -877,8 +882,9 @@ static void virLXCControllerConsoleUpdateWatch(virLXCControllerConsolePtr consol
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (console->contClosed) {
|
if (console->contClosed) {
|
||||||
|
/* Must setup an epoll to detect when guest becomes accessible again */
|
||||||
int events = EPOLLIN | EPOLLET;
|
int events = EPOLLIN | EPOLLET;
|
||||||
if (console->contBlocking)
|
if (console->fromHostLen)
|
||||||
events |= EPOLLOUT;
|
events |= EPOLLOUT;
|
||||||
|
|
||||||
if (events != console->contEpoll) {
|
if (events != console->contEpoll) {
|
||||||
@ -949,7 +955,7 @@ static void virLXCControllerConsoleEPoll(int watch, int fd, int events, void *op
|
|||||||
|
|
||||||
/* If we get HUP+dead PID, we just re-enable the main loop
|
/* If we get HUP+dead PID, we just re-enable the main loop
|
||||||
* which will see the PID has died and exit */
|
* which will see the PID has died and exit */
|
||||||
if ((event.events & EPOLLIN)) {
|
if ((event.events & (EPOLLIN|EPOLLOUT))) {
|
||||||
if (event.data.fd == console->hostFd) {
|
if (event.data.fd == console->hostFd) {
|
||||||
console->hostClosed = false;
|
console->hostClosed = false;
|
||||||
} else {
|
} else {
|
||||||
@ -1029,10 +1035,6 @@ static void virLXCControllerConsoleIO(int watch, int fd, int events, void *opaqu
|
|||||||
*len -= done;
|
*len -= done;
|
||||||
} else {
|
} else {
|
||||||
VIR_DEBUG("Write fd %d done %d errno %d", fd, (int)done, errno);
|
VIR_DEBUG("Write fd %d done %d errno %d", fd, (int)done, errno);
|
||||||
if (watch == console->hostWatch)
|
|
||||||
console->hostBlocking = true;
|
|
||||||
else
|
|
||||||
console->contBlocking = true;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user