mirror of
https://sourceware.org/git/glibc.git
synced 2024-11-09 23:00:07 +00:00
sigwait: Do not fail with EINTR and return error code [BZ #22478]
Since
commit 8b0e795aaa
Author: Adhemerval Zanella <adhemerval.zanella@linaro.org>
Date: Wed Nov 1 11:49:05 2017 -0200
Simplify Linux sig{timed}wait{info} implementations
sigwait can fail with EINTR. Applications do not expect that, and the
error code is not documented in POSIX or the manual pages.
This commit restores the previous behavior by retrying the system call
on EINTR. It also returns the error code, not -1, on the remaing
errors.
Reviewed-by: Adhemerval Zanella <adhemerval.zanella@linaro.org>
This commit is contained in:
parent
59d2cbb1fe
commit
cccb6d4e87
@ -1,3 +1,11 @@
|
||||
2017-11-23 Florian Weimer <fweimer@redhat.com>
|
||||
|
||||
[BZ #22478]
|
||||
* sysdeps/unix/sysv/linux/sigwait.c (__sigwait): Retry on EINTR.
|
||||
Return error code, not -1.
|
||||
* signal/tst-sigwait-eintr.c: New file.
|
||||
* signal/Makefile (tests): Add tst-sigwait-eintr.
|
||||
|
||||
2017-11-23 Florian Weimer <fweimer@redhat.com>
|
||||
|
||||
Linux: Add memfd_create system call wrapper
|
||||
|
@ -45,8 +45,8 @@ routines := signal raise killpg \
|
||||
allocrtsig sigtimedwait sigwaitinfo sigqueue \
|
||||
sighold sigrelse sigignore sigset
|
||||
|
||||
tests := tst-signal tst-sigset tst-sigsimple tst-raise tst-sigset2
|
||||
|
||||
tests := tst-signal tst-sigset tst-sigsimple tst-raise tst-sigset2 \
|
||||
tst-sigwait-eintr \
|
||||
|
||||
include ../Rules
|
||||
|
||||
|
85
signal/tst-sigwait-eintr.c
Normal file
85
signal/tst-sigwait-eintr.c
Normal file
@ -0,0 +1,85 @@
|
||||
/* Check that sigwait does not fail with EINTR (bug 22478).
|
||||
Copyright (C) 2017 Free Software Foundation, Inc.
|
||||
This file is part of the GNU C Library.
|
||||
|
||||
The GNU C Library is free software; you can redistribute it and/or
|
||||
modify it under the terms of the GNU Lesser General Public
|
||||
License as published by the Free Software Foundation; either
|
||||
version 2.1 of the License, or (at your option) any later version.
|
||||
|
||||
The GNU C Library is distributed in the hope that it will be useful,
|
||||
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||
Lesser General Public License for more details.
|
||||
|
||||
You should have received a copy of the GNU Lesser General Public
|
||||
License along with the GNU C Library; if not, see
|
||||
<http://www.gnu.org/licenses/>. */
|
||||
|
||||
#include <errno.h>
|
||||
#include <signal.h>
|
||||
#include <stdio.h>
|
||||
#include <support/check.h>
|
||||
#include <support/xunistd.h>
|
||||
#include <time.h>
|
||||
#include <unistd.h>
|
||||
|
||||
/* Handler for SIGUSR1. */
|
||||
static void
|
||||
sigusr1_handler (int signo)
|
||||
{
|
||||
TEST_VERIFY (signo == SIGUSR1);
|
||||
}
|
||||
|
||||
/* Spawn a subprocess to send two signals: First SIGUSR1, then
|
||||
SIGUSR2. Return the PID of the process. */
|
||||
static pid_t
|
||||
signal_sender (void)
|
||||
{
|
||||
pid_t pid = xfork ();
|
||||
if (pid == 0)
|
||||
{
|
||||
static const struct timespec delay = { 1, };
|
||||
if (nanosleep (&delay, NULL) != 0)
|
||||
FAIL_EXIT1 ("nanosleep: %m");
|
||||
if (kill (getppid (), SIGUSR1) != 0)
|
||||
FAIL_EXIT1 ("kill (SIGUSR1): %m");
|
||||
if (nanosleep (&delay, NULL) != 0)
|
||||
FAIL_EXIT1 ("nanosleep: %m");
|
||||
if (kill (getppid (), SIGUSR2) != 0)
|
||||
FAIL_EXIT1 ("kill (SIGUSR2): %m");
|
||||
_exit (0);
|
||||
}
|
||||
return pid;
|
||||
}
|
||||
|
||||
static int
|
||||
do_test (void)
|
||||
{
|
||||
if (signal (SIGUSR1, sigusr1_handler) == SIG_ERR)
|
||||
FAIL_EXIT1 ("signal (SIGUSR1): %m\n");
|
||||
|
||||
sigset_t sigs;
|
||||
sigemptyset (&sigs);
|
||||
sigaddset (&sigs, SIGUSR2);
|
||||
if (sigprocmask (SIG_BLOCK, &sigs, NULL) != 0)
|
||||
FAIL_EXIT1 ("sigprocmask (SIGBLOCK, SIGUSR2): %m");
|
||||
pid_t pid = signal_sender ();
|
||||
int signo = 0;
|
||||
int ret = sigwait (&sigs, &signo);
|
||||
if (ret != 0)
|
||||
{
|
||||
support_record_failure ();
|
||||
errno = ret;
|
||||
printf ("error: sigwait failed: %m (%d)\n", ret);
|
||||
}
|
||||
TEST_VERIFY (signo == SIGUSR2);
|
||||
|
||||
int status;
|
||||
xwaitpid (pid, &status, 0);
|
||||
TEST_VERIFY (status == 0);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
#include <support/test-driver.c>
|
@ -17,13 +17,20 @@
|
||||
|
||||
#include <signal.h>
|
||||
#include <sysdep-cancel.h>
|
||||
#include <errno.h>
|
||||
|
||||
int
|
||||
__sigwait (const sigset_t *set, int *sig)
|
||||
{
|
||||
siginfo_t si;
|
||||
if (__sigtimedwait (set, &si, 0) < 0)
|
||||
return -1;
|
||||
int ret;
|
||||
do
|
||||
ret = __sigtimedwait (set, &si, 0);
|
||||
/* Applications do not expect sigwait to return with EINTR, and the
|
||||
error code is not specified by POSIX. */
|
||||
while (ret < 0 && errno == EINTR);
|
||||
if (ret < 0)
|
||||
return errno;
|
||||
*sig = si.si_signo;
|
||||
return 0;
|
||||
}
|
||||
|
Loading…
Reference in New Issue
Block a user