diff options
author | Christopher Faylor <me@cgf.cx> | 2003-08-20 02:31:26 +0000 |
---|---|---|
committer | Christopher Faylor <me@cgf.cx> | 2003-08-20 02:31:26 +0000 |
commit | 5f31e0f30557788f0f451fd69fbc70a89c271c18 (patch) | |
tree | 1b3763c8d947acb63beddda0bc7e4c610b1fed96 | |
parent | 6ac844b5e3ef1c50a6faeac37abd23f9b8b8fe6d (diff) | |
download | newlib-5f31e0f30557788f0f451fd69fbc70a89c271c18.zip newlib-5f31e0f30557788f0f451fd69fbc70a89c271c18.tar.gz newlib-5f31e0f30557788f0f451fd69fbc70a89c271c18.tar.bz2 |
* exceptions.cc (pending_signals): Remove unneeded declaration.
* sigproc.cc (pending_signals): Make static.
(wait_sig): Use defined values rather than integers for rc. Never scan both
todo arrays as this could cause hangs if signals arrive from two different
sources. Rename saw_pending_signals to saw_failed_interrupt. Exit loop when
signal found. Enter low-priority sleep, if necessary, after finished
signalling completion. Set pending_signals when blocked
(from Pierre Humblet).
-rw-r--r-- | winsup/cygwin/ChangeLog | 11 | ||||
-rw-r--r-- | winsup/cygwin/exceptions.cc | 1 | ||||
-rw-r--r-- | winsup/cygwin/sigproc.cc | 56 |
3 files changed, 37 insertions, 31 deletions
diff --git a/winsup/cygwin/ChangeLog b/winsup/cygwin/ChangeLog index 3b49b38..83a902e 100644 --- a/winsup/cygwin/ChangeLog +++ b/winsup/cygwin/ChangeLog @@ -1,5 +1,16 @@ 2003-08-19 Christopher Faylor <cgf@redhat.com> + * exceptions.cc (pending_signals): Remove unneeded declaration. + * sigproc.cc (pending_signals): Make static. + (wait_sig): Use defined values rather than integers for rc. Never scan + both todo arrays as this could cause hangs if signals arrive from two + different sources. Rename saw_pending_signals to saw_failed_interrupt. + Exit loop when signal found. Enter low-priority sleep, if necessary, + after finished signalling completion. Set pending_signals when blocked + (from Pierre Humblet). + +2003-08-19 Christopher Faylor <cgf@redhat.com> + * signal.cc (sigpending): Move. * sigproc.cc (sigpending): To here. (getlocal_sigtodo): Return process-local signal array. diff --git a/winsup/cygwin/exceptions.cc b/winsup/cygwin/exceptions.cc index dea7c8b..537f55e 100644 --- a/winsup/cygwin/exceptions.cc +++ b/winsup/cygwin/exceptions.cc @@ -591,7 +591,6 @@ handle_sigsuspend (sigset_t tempmask) } extern DWORD exec_exit; // Possible exit value for exec -extern int pending_signals; extern "C" { static void diff --git a/winsup/cygwin/sigproc.cc b/winsup/cygwin/sigproc.cc index 4ac1d8e..f9fed06 100644 --- a/winsup/cygwin/sigproc.cc +++ b/winsup/cygwin/sigproc.cc @@ -123,7 +123,7 @@ muto NO_COPY *sync_proc_subproc = NULL; // Control access to subproc stuff DWORD NO_COPY sigtid = 0; // ID of the signal thread -int NO_COPY pending_signals = 0; // TRUE if signals pending +static bool NO_COPY pending_signals = false; // true if signals pending /* Functions */ @@ -1042,6 +1042,9 @@ talktome () pids[i]->commune_recv (); } +#define RC_MAIN 0 +#define RC_NONMAIN 1 +#define RC_NOSYNC 2 /* Process signals by waiting for a semaphore to become signaled. * Then scan an in-memory array representing queued signals. * Executes in a separate thread. @@ -1141,25 +1144,24 @@ wait_sig (VOID *self) LONG **start_todo, **end_todo; switch (rc) { - case 0: + case RC_MAIN: + case RC_NONMAIN: start_todo = todos; end_todo = todos; break; - case 1: + case RC_NOSYNC: + default: // silence compiler warning start_todo = todos + 1; end_todo = todos + 1; - default: - start_todo = todos; - end_todo = todos + 1; + break; } /* A sigcatch semaphore has been signaled. Scan the sigtodo * array looking for any unprocessed signals. */ - pending_signals = -1; - bool saw_pending_signals = false; - bool saw_sigchld = false; - for (LONG **todo = start_todo; todo <= end_todo; todo++) + pending_signals = false; + bool saw_failed_interrupt = false; + for (LONG **todo = todos; todo <= end_todo; todo++) for (int sig = -__SIGOFFSET; sig < NSIG; sig++) { LONG x = InterlockedDecrement (*todo + sig); @@ -1168,10 +1170,7 @@ wait_sig (VOID *self) else if (x >= 0) { if (x > 0) - pending_signals = 1; - - if (sig == SIGCHLD) - saw_sigchld = true; + pending_signals = true; /* semaphore should already be armed */ if (sig > 0 && sig != SIGKILL && sig != SIGSTOP && (sigismember (&myself->getsigmask (), sig) || @@ -1180,6 +1179,7 @@ wait_sig (VOID *self) { sigproc_printf ("signal %d blocked", sig); InterlockedIncrement (*todo + sig); + pending_signals = true; // FIXME: This will cause unnecessary sig_dispatch_pending spins } else { @@ -1206,41 +1206,37 @@ wait_sig (VOID *self) if (!sig_handle (sig)) { sigproc_printf ("couldn't send signal %d", sig); - low_priority_sleep (SLEEP_0_STAY_LOW); /* Hopefully, other process will be waking up soon. */ - saw_pending_signals = true; + pending_signals = saw_failed_interrupt = true; ReleaseSemaphore (sigcatch_nosync, 1, NULL); - InterlockedIncrement (*todo + sig); + InterlockedIncrement (myself->getsigtodo (sig)); } - break; } } + + if (sig == SIGCHLD) + proc_subproc (PROC_CLEARWAIT, 0); + goto out; } } - if (saw_pending_signals) - pending_signals = 1; - else if (pending_signals < 0) - pending_signals = 0; - - if (saw_sigchld) - proc_subproc (PROC_CLEARWAIT, 0); - + out: /* Signal completion of signal handling depending on which semaphore - * woke up the WaitForMultipleObjects above. - */ + woke up the WaitForMultipleObjects above. */ switch (rc) { - case 0: + case RC_MAIN: SetEvent (sigcomplete_main); sigproc_printf ("set main thread completion event"); break; - case 1: + case RC_NONMAIN: ReleaseSemaphore (sigcomplete_nonmain, 1, NULL); break; default: /* Signal from another process. No need to synchronize. */ break; } + if (saw_failed_interrupt) + low_priority_sleep (SLEEP_0_STAY_LOW); /* Hopefully, other thread will be waking up soon. */ sigproc_printf ("looping"); } |