1
0
mirror of git://sourceware.org/git/lvm2.git synced 2025-01-03 05:18:29 +03:00

libdaemon: use pselect to avoid condition checking race

To avoid tiny race on checking arrival of signal and entering select
(that can latter remain stuck as signal was already delivered) switch
to use  pselect().

If it would needed, we can eventually add extra code for older systems
without pselect(), but there are probably no such ancient systems in
use.
This commit is contained in:
Zdenek Kabelac 2019-04-10 12:50:53 +02:00
parent 116bd314cb
commit 44cfa55843
4 changed files with 31 additions and 12 deletions

View File

@ -1,5 +1,6 @@
Version 2.03.02 - Version 2.03.02 -
=================================== ===================================
Fix signal delivery checking race in libdaemon (lvmetad).
Add missing Before=shutdown.target to LVM2 services to fix shutdown ordering. Add missing Before=shutdown.target to LVM2 services to fix shutdown ordering.
Skip autoactivation for a PV when PV size does not match device size. Skip autoactivation for a PV when PV size does not match device size.
Remove first-pvscan-initialization which should no longer be needed. Remove first-pvscan-initialization which should no longer be needed.

View File

@ -915,7 +915,7 @@ int main(int argc, char *argv[])
int option_index = 0; int option_index = 0;
int client = 0, server = 0; int client = 0, server = 0;
unsigned action = ACTION_MAX; unsigned action = ACTION_MAX;
struct timeval timeout; struct timespec timeout;
daemon_idle di = { .ptimeout = &timeout }; daemon_idle di = { .ptimeout = &timeout };
struct lvmpolld_state ls = { .log_config = "" }; struct lvmpolld_state ls = { .log_config = "" };
daemon_state s = { daemon_state s = {

View File

@ -85,7 +85,7 @@ static int _is_idle(daemon_state s)
return s.idle && s.idle->is_idle && !s.threads->next; return s.idle && s.idle->is_idle && !s.threads->next;
} }
static struct timeval *_get_timeout(daemon_state s) static struct timespec *_get_timeout(daemon_state s)
{ {
return s.idle ? s.idle->ptimeout : NULL; return s.idle ? s.idle->ptimeout : NULL;
} }
@ -94,7 +94,7 @@ static void _reset_timeout(daemon_state s)
{ {
if (s.idle) { if (s.idle) {
s.idle->ptimeout->tv_sec = 1; s.idle->ptimeout->tv_sec = 1;
s.idle->ptimeout->tv_usec = 0; s.idle->ptimeout->tv_nsec = 0;
} }
} }
@ -559,6 +559,8 @@ void daemon_start(daemon_state s)
thread_state _threads = { .next = NULL }; thread_state _threads = { .next = NULL };
unsigned timeout_count = 0; unsigned timeout_count = 0;
fd_set in; fd_set in;
sigset_t new_set, old_set;
int ret;
/* /*
* Switch to C locale to avoid reading large locale-archive file used by * Switch to C locale to avoid reading large locale-archive file used by
@ -626,8 +628,7 @@ void daemon_start(daemon_state s)
if (!s.foreground) if (!s.foreground)
kill(getppid(), SIGTERM); kill(getppid(), SIGTERM);
/* /* * Use daemon_main for daemon-specific init and polling, or
* Use daemon_main for daemon-specific init and polling, or
* use daemon_init for daemon-specific init and generic lib polling. * use daemon_init for daemon-specific init and generic lib polling.
*/ */
@ -641,22 +642,39 @@ void daemon_start(daemon_state s)
if (!s.daemon_init(&s)) if (!s.daemon_init(&s))
failed = 1; failed = 1;
if (s.socket_fd >= FD_SETSIZE)
failed = 1; /* FD out of available selectable set */
sigfillset(&new_set);
sigprocmask(SIG_SETMASK, NULL, &old_set);
while (!failed) { while (!failed) {
_reset_timeout(s); _reset_timeout(s);
FD_ZERO(&in); FD_ZERO(&in);
FD_SET(s.socket_fd, &in); FD_SET(s.socket_fd, &in);
if (select(FD_SETSIZE, &in, NULL, NULL, _get_timeout(s)) < 0 && errno != EINTR)
perror("select error"); sigprocmask(SIG_SETMASK, &new_set, NULL);
if (_shutdown_requested && !s.threads->next) {
sigprocmask(SIG_SETMASK, &old_set, NULL);
INFO(&s, "shutdown requested", s.name);
break;
}
ret = pselect(s.socket_fd + 1, &in, NULL, NULL, _get_timeout(s), &old_set);
sigprocmask(SIG_SETMASK, &old_set, NULL);
if (ret < 0) {
if (errno != EINTR && errno != EAGAIN &&
(EWOULDBLOCK == EAGAIN || errno != EWOULDBLOCK))
perror("select error");
continue;
}
if (FD_ISSET(s.socket_fd, &in)) { if (FD_ISSET(s.socket_fd, &in)) {
timeout_count = 0; timeout_count = 0;
_handle_connect(s); _handle_connect(s);
} }
_reap(s, 0); _reap(s, 0);
if (_shutdown_requested && !s.threads->next)
break;
/* s.idle == NULL equals no shutdown on timeout */ /* s.idle == NULL equals no shutdown on timeout */
if (_is_idle(s)) { if (_is_idle(s)) {
DEBUGLOG(&s, "timeout occured"); DEBUGLOG(&s, "timeout occured");

View File

@ -47,7 +47,7 @@ struct timeval;
typedef struct { typedef struct {
volatile unsigned is_idle; volatile unsigned is_idle;
unsigned max_timeouts; unsigned max_timeouts;
struct timeval *ptimeout; struct timespec *ptimeout;
} daemon_idle; } daemon_idle;
struct daemon_state; struct daemon_state;