1
1

Merge pull request #6784 from abouteiller/export/event-infloop

Address a race condition in libevent select.
Этот коммит содержится в:
Austen Lauria 2020-06-30 10:26:57 -04:00 коммит произвёл GitHub
родитель 4358e75a75 c39fb5758a
Коммит b6b300d25d
Не найден ключ, соответствующий данной подписи
Идентификатор ключа GPG: 4AEE18F83AFDEB23

Просмотреть файл

@ -42,6 +42,7 @@
#include <string.h>
#include <unistd.h>
#include <errno.h>
#include <fcntl.h>
#include "event-internal.h"
#include "evsignal-internal.h"
@ -166,12 +167,30 @@ select_dispatch(struct event_base *base, struct timeval *tv)
check_selectop(sop);
if (res == -1) {
if (errno != EINTR) {
event_warn("select");
return (-1);
if (errno == EINTR) {
return (0);
}
return (0);
/* There seems to be a very subtle race condition between the
* event_del and the select, where the fd is still active on the
* event_readset_in but no libevent structure make reference
* to it so it. Thus, any call to progress will no nothing more
* than print a warning and do nothing, leading to deadlocks.
* If we force remove the problematic fd, we get the warning only
* once, and things work as expected.
*/
event_warn("select");
for (j = 0; j < nfds; ++j) {
if (FD_ISSET(j, sop->event_readset_in) ||
FD_ISSET(j, sop->event_writeset_in)) {
res = fcntl(j, F_GETFL);
if( res == -1 ) {
event_warn("bad file descriptor %d/%d\n", j, nfds);
FD_CLR(j, sop->event_readset_in);
FD_CLR(j, sop->event_writeset_in);
}
}
}
return (-1);
}
event_debug(("%s: select reports %d", __func__, res));