mirror of
https://git.tukaani.org/xz.git
synced 2024-04-04 12:36:23 +02:00
xz: Use the self-pipe trick to avoid a race condition with signals.
It is possible that a signal to set user_abort arrives right before a blocking system call is made. In this case the call may block until another signal arrives, while the wanted behavior is to make xz clean up and exit as soon as possible. After this commit, the race condition is avoided with the input side which already uses non-blocking I/O. The output side still uses blocking I/O and thus has the race condition.
This commit is contained in:
parent
3541bc79d0
commit
9dc319eabb
3 changed files with 57 additions and 12 deletions
|
@ -51,6 +51,10 @@ static bool restore_stdin_flags = false;
|
||||||
/// io_open_dest() and io_close_dest() to save and restore the flags.
|
/// io_open_dest() and io_close_dest() to save and restore the flags.
|
||||||
static int stdout_flags;
|
static int stdout_flags;
|
||||||
static bool restore_stdout_flags = false;
|
static bool restore_stdout_flags = false;
|
||||||
|
|
||||||
|
/// Self-pipe used together with the user_abort variable to avoid
|
||||||
|
/// race conditions with signal handling.
|
||||||
|
static int user_abort_pipe[2];
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
|
||||||
|
@ -70,6 +74,14 @@ io_init(void)
|
||||||
// If fchown() fails setting the owner, we warn about it only if
|
// If fchown() fails setting the owner, we warn about it only if
|
||||||
// we are root.
|
// we are root.
|
||||||
warn_fchown = geteuid() == 0;
|
warn_fchown = geteuid() == 0;
|
||||||
|
|
||||||
|
if (pipe(user_abort_pipe)
|
||||||
|
|| fcntl(user_abort_pipe[0], F_SETFL, O_NONBLOCK)
|
||||||
|
== -1
|
||||||
|
|| fcntl(user_abort_pipe[1], F_SETFL, O_NONBLOCK)
|
||||||
|
== -1)
|
||||||
|
message_fatal(_("Error creating a pipe: %s"),
|
||||||
|
strerror(errno));
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef __DJGPP__
|
#ifdef __DJGPP__
|
||||||
|
@ -82,6 +94,17 @@ io_init(void)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
#ifndef TUKLIB_DOSLIKE
|
||||||
|
extern void
|
||||||
|
io_write_to_user_abort_pipe(void)
|
||||||
|
{
|
||||||
|
uint8_t b = '\0';
|
||||||
|
(void)write(user_abort_pipe[1], &b, 1);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
|
||||||
extern void
|
extern void
|
||||||
io_no_sparse(void)
|
io_no_sparse(void)
|
||||||
{
|
{
|
||||||
|
@ -91,11 +114,20 @@ io_no_sparse(void)
|
||||||
|
|
||||||
|
|
||||||
#ifndef TUKLIB_DOSLIKE
|
#ifndef TUKLIB_DOSLIKE
|
||||||
/// \brief Waits for input or output to become available
|
/// \brief Waits for input or output to become available or for a signal
|
||||||
|
///
|
||||||
|
/// This uses the self-pipe trick to avoid a race condition that can occur
|
||||||
|
/// if a signal is caught after user_abort has been checked but before e.g.
|
||||||
|
/// read() has been called. In that situation read() could block unless
|
||||||
|
/// non-blocking I/O is used. With non-blocking I/O something like select()
|
||||||
|
/// or poll() is needed to avoid a busy-wait loop, and the same race condition
|
||||||
|
/// pops up again. There are pselect() (POSIX-1.2001) and ppoll() (not in
|
||||||
|
/// POSIX) but neither is portable enough in 2013. The self-pipe trick is
|
||||||
|
/// old and very portable.
|
||||||
static bool
|
static bool
|
||||||
io_wait(file_pair *pair, bool is_reading)
|
io_wait(file_pair *pair, bool is_reading)
|
||||||
{
|
{
|
||||||
struct pollfd pfd[1];
|
struct pollfd pfd[2];
|
||||||
|
|
||||||
if (is_reading) {
|
if (is_reading) {
|
||||||
pfd[0].fd = pair->src_fd;
|
pfd[0].fd = pair->src_fd;
|
||||||
|
@ -105,18 +137,17 @@ io_wait(file_pair *pair, bool is_reading)
|
||||||
pfd[0].events = POLLOUT;
|
pfd[0].events = POLLOUT;
|
||||||
}
|
}
|
||||||
|
|
||||||
while (true) {
|
pfd[1].fd = user_abort_pipe[0];
|
||||||
const int ret = poll(pfd, 1, -1);
|
pfd[1].events = POLLIN;
|
||||||
|
|
||||||
|
while (true) {
|
||||||
|
const int ret = poll(pfd, 2, -1);
|
||||||
|
|
||||||
if (ret == -1) {
|
|
||||||
if (errno == EINTR) {
|
|
||||||
if (user_abort)
|
if (user_abort)
|
||||||
return true;
|
return true;
|
||||||
|
|
||||||
continue;
|
if (ret == -1) {
|
||||||
}
|
if (errno == EINTR || errno == EAGAIN)
|
||||||
|
|
||||||
if (errno == EAGAIN)
|
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
message_error(_("%s: poll() failed: %s"),
|
message_error(_("%s: poll() failed: %s"),
|
||||||
|
@ -125,6 +156,7 @@ io_wait(file_pair *pair, bool is_reading)
|
||||||
strerror(errno));
|
strerror(errno));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (pfd[0].revents != 0)
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -68,6 +68,14 @@ typedef struct {
|
||||||
extern void io_init(void);
|
extern void io_init(void);
|
||||||
|
|
||||||
|
|
||||||
|
#ifndef TUKLIB_DOSLIKE
|
||||||
|
/// \brief Write a byte to user_abort_pipe[1]
|
||||||
|
///
|
||||||
|
/// This is called from a signal handler.
|
||||||
|
extern void io_write_to_user_abort_pipe(void);
|
||||||
|
#endif
|
||||||
|
|
||||||
|
|
||||||
/// \brief Disable creation of sparse files when decompressing
|
/// \brief Disable creation of sparse files when decompressing
|
||||||
extern void io_no_sparse(void);
|
extern void io_no_sparse(void);
|
||||||
|
|
||||||
|
|
|
@ -41,6 +41,11 @@ signal_handler(int sig)
|
||||||
{
|
{
|
||||||
exit_signal = sig;
|
exit_signal = sig;
|
||||||
user_abort = true;
|
user_abort = true;
|
||||||
|
|
||||||
|
#ifndef TUKLIB_DOSLIKE
|
||||||
|
io_write_to_user_abort_pipe();
|
||||||
|
#endif
|
||||||
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Add table
Reference in a new issue