On Sat, Feb 8, 2020 at 7:41 PM Harald van Dijk <harald@xxxxxxxxxxx> wrote: > I changed gwsh to call sigclearmask() on shell startup, but plan to > check whether this loop is really necessary at some later time. It was > added to dash to fix a race condition, where that race condition was > apparently introduced by a fix for another race condition. sigsuspend() is needed to make "wait" builtin interruptible by signals. Attempts to use EINTR error return of waitpid() a-la: if (got_sigs) { handle signals } got_sigs = 0; pid = waitpid(...); /* without WNOHANG */ if (pid < 0 && errno == EINTR) { handle signals } are racy, since signals can be delivered not only while waitpid() syscall is in kernel, but also when we are only about to enter the kernel - and in this case, the shell's sighandler will set the flag variable, but then we enter the kernel and sleep. Masking signals doesn't help, since you need to unmask them just before waitpid() if you want to get EINTR on a signal, hence there is still a window for the race. > If NetBSD sh > manages to avoid this pattern, and assuming NetBSD sh is not still > susceptible to one of those race conditions Please let us know what you discovered.