make not blocking signals the default

This is the next step in determining whether we can disable blocking
signals without a good reason to do so. This makes not blocking signals
the default behavior. If someone finds a problem they can add this to
their ~/config/fish/config.fish file:

set FISH_NO_SIGNAL_BLOCK 0

Alternatively set that env var before starting fish. I won't be surprised
if people report problems. Till now we have relied on people opting in
to this behavior to tell us whether it causes problems. This makes the
experimental behavior the default that has to be opted out of. This will
give us a lot more confidence this change doesn't cause problems before
the next minor release.

Note that there are still a few places where we force blocking of
signals. Primarily to keep SIGTSTP from interfering with the shell in
response to manipulating the controlling tty. Bash is more selective
in the signals it blocks around the problematic syscalls (c.f., its
`git_terminal_to()` function). However, I don't see any value in that
refinement.
This commit is contained in:
Kurtis Rader 2017-02-26 21:46:15 -08:00
parent 6d02bec4c7
commit e0f62c178f
7 changed files with 44 additions and 33 deletions

View file

@ -109,7 +109,6 @@ demangled_backtrace(int max_frames, int skip_levels) {
void __attribute__((noinline)) void __attribute__((noinline))
show_stackframe(const wchar_t msg_level, int frame_count, int skip_levels) { show_stackframe(const wchar_t msg_level, int frame_count, int skip_levels) {
ASSERT_IS_NOT_FORKED_CHILD();
if (frame_count < 1) return; if (frame_count < 1) return;
// TODO: Decide if this is still needed. I'm commenting it out because it caused me some grief // TODO: Decide if this is still needed. I'm commenting it out because it caused me some grief

View file

@ -166,7 +166,6 @@ static void safe_launch_process(process_t *p, const char *actual_cmd, const char
char *const *argv = const_cast<char *const *>(cargv); char *const *argv = const_cast<char *const *>(cargv);
execve(actual_cmd, argv, envv); execve(actual_cmd, argv, envv);
err = errno; err = errno;
// Something went wrong with execve, check for a ":", and run /bin/sh if encountered. This is a // Something went wrong with execve, check for a ":", and run /bin/sh if encountered. This is a

View file

@ -368,7 +368,9 @@ int main(int argc, char **argv) {
// TODO: Remove this once we're confident that not blocking/unblocking every signal around // TODO: Remove this once we're confident that not blocking/unblocking every signal around
// some critical sections is no longer necessary. // some critical sections is no longer necessary.
env_var_t fish_no_signal_block = env_get_string(L"FISH_NO_SIGNAL_BLOCK"); env_var_t fish_no_signal_block = env_get_string(L"FISH_NO_SIGNAL_BLOCK");
if (!fish_no_signal_block.missing()) ignore_signal_block = true; if (!fish_no_signal_block.missing_or_empty() && !from_string<bool>(fish_no_signal_block)) {
ignore_signal_block = false;
}
// Stomp the exit status of any initialization commands (issue #635). // Stomp the exit status of any initialization commands (issue #635).
proc_set_last_status(STATUS_BUILTIN_OK); proc_set_last_status(STATUS_BUILTIN_OK);

View file

@ -108,7 +108,9 @@ bool set_child_group(job_t *j, process_t *p, int print_errors) {
int result = -1; int result = -1;
errno = EINTR; errno = EINTR;
while (result == -1 && errno == EINTR) { while (result == -1 && errno == EINTR) {
signal_block(true);
result = tcsetpgrp(STDIN_FILENO, j->pgid); result = tcsetpgrp(STDIN_FILENO, j->pgid);
signal_unblock(true);
} }
if (result == -1) { if (result == -1) {
if (errno == ENOTTY) redirect_tty_output(); if (errno == ENOTTY) redirect_tty_output();

View file

@ -783,6 +783,14 @@ static void read_try(job_t *j) {
/// \param cont If this variable is set, we are giving back control to a job that has previously /// \param cont If this variable is set, we are giving back control to a job that has previously
/// been stopped. In that case, we need to set the terminal attributes to those saved in the job. /// been stopped. In that case, we need to set the terminal attributes to those saved in the job.
static bool terminal_give_to_job(job_t *j, int cont) { static bool terminal_give_to_job(job_t *j, int cont) {
int s = 0;
errno = 0;
if (j->pgid == 0) {
debug(2, "terminal_give_to_job() returning early due to no process group");
return true;
}
signal_block(true);
int result = -1; int result = -1;
errno = EINTR; errno = EINTR;
while (result == -1 && errno == EINTR) { while (result == -1 && errno == EINTR) {
@ -792,11 +800,15 @@ static bool terminal_give_to_job(job_t *j, int cont) {
if (errno == ENOTTY) redirect_tty_output(); if (errno == ENOTTY) redirect_tty_output();
debug(1, _(L"Could not send job %d ('%ls') to foreground"), j->job_id, j->command_wcstr()); debug(1, _(L"Could not send job %d ('%ls') to foreground"), j->job_id, j->command_wcstr());
wperror(L"tcsetpgrp"); wperror(L"tcsetpgrp");
signal_unblock(true);
return false; return false;
} }
if (cont) { if (cont) {
int result = -1; int result = -1;
// TODO: Remove this EINTR loop since we have blocked all signals and thus cannot be
// interrupted. I'm leaving it in place because all of the logic involving controlling
// terminal management is more than a little opaque and smacks of voodoo programming.
errno = EINTR; errno = EINTR;
while (result == -1 && errno == EINTR) { while (result == -1 && errno == EINTR) {
result = tcsetattr(STDIN_FILENO, TCSADRAIN, &j->tmodes); result = tcsetattr(STDIN_FILENO, TCSADRAIN, &j->tmodes);
@ -806,21 +818,32 @@ static bool terminal_give_to_job(job_t *j, int cont) {
debug(1, _(L"terminal_give_to_job(): Could not send job %d ('%ls') to foreground"), debug(1, _(L"terminal_give_to_job(): Could not send job %d ('%ls') to foreground"),
j->job_id, j->command_wcstr()); j->job_id, j->command_wcstr());
wperror(L"tcsetattr"); wperror(L"tcsetattr");
signal_unblock(true);
return false; return false;
} }
} }
signal_unblock(true);
return true; return true;
} }
/// Returns control of the terminal to the shell, and saves the terminal attribute state to the job, /// Returns control of the terminal to the shell, and saves the terminal attribute state to the job,
/// so that we can restore the terminal ownership to the job at a later time. /// so that we can restore the terminal ownership to the job at a later time.
static int terminal_return_from_job(job_t *j) { static bool terminal_return_from_job(job_t *j) {
int s = 0;
errno = 0;
if (j->pgid == 0) {
debug(2, "terminal_return_from_job() returning early due to no process group");
return true;
}
signal_block(true);
if (tcsetpgrp(STDIN_FILENO, getpgrp()) == -1) { if (tcsetpgrp(STDIN_FILENO, getpgrp()) == -1) {
if (errno == ENOTTY) redirect_tty_output(); if (errno == ENOTTY) redirect_tty_output();
debug(1, _(L"Could not return shell to foreground")); debug(1, _(L"Could not return shell to foreground"));
wperror(L"tcsetpgrp"); wperror(L"tcsetpgrp");
return 0; signal_unblock(true);
return false;
} }
// Save jobs terminal modes. // Save jobs terminal modes.
@ -828,7 +851,8 @@ static int terminal_return_from_job(job_t *j) {
if (errno == EIO) redirect_tty_output(); if (errno == EIO) redirect_tty_output();
debug(1, _(L"Could not return shell to foreground")); debug(1, _(L"Could not return shell to foreground"));
wperror(L"tcgetattr"); wperror(L"tcgetattr");
return 0; signal_unblock(true);
return false;
} }
// Disabling this per // Disabling this per
@ -841,11 +865,12 @@ static int terminal_return_from_job(job_t *j) {
if (errno == EIO) redirect_tty_output(); if (errno == EIO) redirect_tty_output();
debug(1, _(L"Could not return shell to foreground")); debug(1, _(L"Could not return shell to foreground"));
wperror(L"tcsetattr"); wperror(L"tcsetattr");
return 0; return false;
} }
#endif #endif
return 1; signal_unblock(true);
return true;
} }
void job_continue(job_t *j, bool cont) { void job_continue(job_t *j, bool cont) {
@ -854,7 +879,6 @@ void job_continue(job_t *j, bool cont) {
j->set_flag(JOB_NOTIFIED, false); j->set_flag(JOB_NOTIFIED, false);
CHECK_BLOCK(); CHECK_BLOCK();
debug(4, L"Continue job %d, gid %d (%ls), %ls, %ls", j->job_id, j->pgid, j->command_wcstr(), debug(4, L"Continue job %d, gid %d (%ls), %ls, %ls", j->job_id, j->pgid, j->command_wcstr(),
job_is_completed(j) ? L"COMPLETED" : L"UNCOMPLETED", job_is_completed(j) ? L"COMPLETED" : L"UNCOMPLETED",
is_interactive ? L"INTERACTIVE" : L"NON-INTERACTIVE"); is_interactive ? L"INTERACTIVE" : L"NON-INTERACTIVE");
@ -864,14 +888,7 @@ void job_continue(job_t *j, bool cont) {
// Put the job into the foreground. Hack: ensure that stdin is marked as blocking first // Put the job into the foreground. Hack: ensure that stdin is marked as blocking first
// (issue #176). // (issue #176).
make_fd_blocking(STDIN_FILENO); make_fd_blocking(STDIN_FILENO);
if (!terminal_give_to_job(j, cont)) return;
signal_block();
bool ok = terminal_give_to_job(j, cont);
signal_unblock();
if (!ok) return;
} }
// Send the job a continue signal, if necessary. // Send the job a continue signal, if necessary.
@ -955,15 +972,7 @@ void job_continue(job_t *j, bool cont) {
// Put the shell back in the foreground. // Put the shell back in the foreground.
if (j->get_flag(JOB_TERMINAL) && j->get_flag(JOB_FOREGROUND)) { if (j->get_flag(JOB_TERMINAL) && j->get_flag(JOB_FOREGROUND)) {
int ok; terminal_return_from_job(j);
signal_block();
ok = terminal_return_from_job(j);
signal_unblock();
if (!ok) return;
} }
} }
} }

View file

@ -17,7 +17,7 @@
#include "wutil.h" // IWYU pragma: keep #include "wutil.h" // IWYU pragma: keep
// This is a temporary var while we explore whether signal_block() and friends is needed. // This is a temporary var while we explore whether signal_block() and friends is needed.
bool ignore_signal_block = false; bool ignore_signal_block = true;
/// Struct describing an entry for the lookup table used to convert between signal names and signal /// Struct describing an entry for the lookup table used to convert between signal names and signal
/// ids, etc. /// ids, etc.
@ -372,8 +372,8 @@ void get_signals_with_handlers(sigset_t *set) {
} }
} }
void signal_block() { void signal_block(bool force) {
if (ignore_signal_block) return; if (!force && ignore_signal_block) return;
ASSERT_IS_MAIN_THREAD(); ASSERT_IS_MAIN_THREAD();
sigset_t chldset; sigset_t chldset;
@ -387,8 +387,8 @@ void signal_block() {
// debug( 0, L"signal block level increased to %d", block_count ); // debug( 0, L"signal block level increased to %d", block_count );
} }
void signal_unblock() { void signal_unblock(bool force) {
if (ignore_signal_block) return; if (!force && ignore_signal_block) return;
ASSERT_IS_MAIN_THREAD(); ASSERT_IS_MAIN_THREAD();
sigset_t chldset; sigset_t chldset;

View file

@ -27,10 +27,10 @@ void signal_set_handlers();
void signal_handle(int sig, int do_handle); void signal_handle(int sig, int do_handle);
/// Block all signals. /// Block all signals.
void signal_block(); void signal_block(bool force = false);
/// Unblock all signals. /// Unblock all signals.
void signal_unblock(); void signal_unblock(bool force = false);
/// Returns true if signals are being blocked. /// Returns true if signals are being blocked.
bool signal_is_blocked(); bool signal_is_blocked();