mirror of
git://sourceware.org/git/newlib-cygwin.git
synced 2025-01-15 02:09:19 +08:00
7ac6173643
* devices.gperf: New file. * devices.shilka: New file. * cygwin-gperf: New file. * cygwin-shilka: New file. * fhandler_fifo.cc: New file. * fhandler_nodevice.cc : New file. Reorganize headers so that path.h precedes fhandler.h throughout. Remove device argument and unit arguments from fhandler constructors throughout. Remove pc arguments to fhandler functions and use internal pc element instead, throughout. Use dev element in pc throughout. Use major/minor elements rather than units and device numbers previously in fhandler class. Use correct methods for fhandler file names rather than directly accessing file name variables, throughout. * Makefile.in (DLL_OFILES): Add devices.o, fhandler_fifo.o * dcrt0.cc (dll_crt0_1): Call device::init. * devices.h: Renumber devices based on more Linux-like major/minor numbers. Add more devices. Declare standard device storage. (device): Declare struct. * dir.cc (opendir): Use new 'build_fh_name' to construct a fhandler_* type. * dtable.cc (dtable::get_debugger_info): Ditto. (cygwin_attach_handle_to_fd): Ditto. (dtable::release): Remove special FH_SOCKET case in favor of generic "need_fixup_before" test. (dtable::init_std_file_from_handle): Use either build_fh_dev or build_fh_name to build standard fhandler. (dtable::build_fh_name): Renamed from dtable::build_fhandler_from_name. Move out of dtable class. Don't accept a path_conv argument. Just build it here and pass it to: (build_fh_pc): Renamed from dtable::build_fhandler. Move out of dtable class. Use intrinsic device type in path_conv to create new fhandler. (build_fh_dev): Renamed from dtable::build_fhandler. Move out of dtable class. Simplify arguments to just take new 'device' type and a name. Just return pointer to fhandler rather than trying to insert into dtable. (dtable::dup_worker): Accommodate above build_fh name changes. (dtable::find_fifo): New (currently broken) function. (handle_to_fn): Use strechr for efficiency. * dtable.h: Reflect above build_fh name changes and argument differences. (fhandler_base *&operator []): Return self rather than copy of self. * fhandler.cc (fhandler_base::operator =): Use pc element to set normalized path. (fhandler_base::set_name): Ditto. (fhandler_base::raw_read): Use method to access name. (fhandler_base::write): Correctly use get_output_handle rather than get_handle. (handler_base::device_access_denied): New function. (fhandler_base::open): Eliminate pc argument and use pc element of fhandler_base throughout. (fhandler_base::fstat): Detect if device is based in filesystem and use fstat_fs to calculate stat, if so. (fhandler_base::fhandler_base): Eliminate handling of file names and, instead, just free appropriate component from pc. (fhandler_base::opendir): Remove path_conv parameter. * fhandler.h: Remove all device flags. (fhandler_base::pc): New element. (fhandler_base::set_name): Change argument to path_conv. (fhandler_base::error): New function. (fhandler_base::exists): New function. (fhandler_base::pc_binmode): New function. (fhandler_base::dev): New function. (fhandler_base::open_fs): New function. (fhandler_base::fstat_fs): New function. (fhandler_base::fstat_by_name): New function. (fhandler_base::fstat_by_handle): New function. (fhandler_base::isfifo): New function. (fhandler_base::is_slow): New function. (fhandler_base::is_auto_device): New function. (fhandler_base::is_fs_special): New function. (fhandler_base::device_access_denied): New function. (fhandler_base::operator DWORD&): New operator. (fhandler_base::get_name): Return normalized path from pc. (fhandler_base::get_win32_name): Return windows path from pc. (fhandler_base::isdevice): Renamed from is_device. (fhandler_base::get_native_name): Return device format. (fhandler_fifo): New class. (fhandler_nodevice): New class. (select_stuff::device_specific): Remove array. (select_stuff::device_specific_pipe): New class element. (select_stuff::device_specific_socket): New class element. (select_stuff::device_specific_serial): New class element. (select_stuff::select_stuff): Initialize new elements. * fhandler_disk_file.cc (fhandler_base::fstat_by_handle): Move to base class from fhandler_disk_file. (fhandler_base::fstat_by_name): Ditto. (fhandler_base::fstat_by_name): Ditto. (fhandler_disk_file::open): Move most functionality into fhandler_base::open_fs. (fhandler_base::open_fs): New function. (fhandler_disk_file::close): Move most functionality into fhandler_base::close_fs. (fhandler_base::close_fs): New function. * fhandler_mem.cc (fhandler_dev_mem::open): Use device name in debugging output. * fhandler_socket.cc (fhandler_socket::set_connect_secret): Copy standard urandom device into appropriate place. (fhandler_socket::accept): Reflect change in fdsock return value. * fhandler_tty.cc: See "throughouts" above. * net.cc: Accommodate fdsock change throughout. (fdsock): Return success or failure, accept fd argument and device argument. * path.cc (symlink_info::major): New element. (symlink_info::minor): New element. (symlink_info::parse_device): Declare new function. (fs_info::update): Accommodate changes in path_conv class. (path_conv::fillin): Ditto. (path_conv::return_and_clear_normalized_path): Eliminate. (path_conv::set_normalized_path): New function. (path_conv::path_conv): Set info in dev element. Use path_conv methods Check for FH_FS rather than FH_BAD to indicate when to fill in filesystem stuff. where appropriate rather than direct access. Use set_normalized_path to set normalized path. (windows_device_names): Eliminate. (get_dev): Ditto. (get_raw_device_number): Ditto. (get_device_number): Ditto. (win32_device_name): Call new device name parser to do most of the heavy lifting. (mount_info::conv_to_win32_path): Fill in dev field as appropriate. (symlink_worker): Handle new device files. (symlink_info::check): Ditto. (symlink_info::parse_device): Define new function. * path.h (executable_states): Move here from fhandler.h. (fs_info): Rename variables to *_storage and create methods for accessing same. (path_conv): Add dev element, remove devn and unit and adjust inline methods to accommodate. (set_normalized_path): Declare new function. * pinfo.cc (_pinfo::commune_recv): Add broken support for handling fifos. (_pinfo::commune_send): Ditto. * pipe.cc (fhandler_pipe::close): check for existence of handle before closing it. (handler_pipe::create): Rename from make_pipe. Change arguments to accept fhandler_pipe array. Accommodate fifos. (pipe): Rework to deal with fhandler_pipe::create changes. (_pipe): Ditto. * select.cc: Use individual device_specific types throughout rather than indexing with obsolete device number. (set_bits): Use is_socket call rather than checking device number. * shared_info.h (CURR_MOUNT_MAGIC): Update. (conv_to_win32_path): Reflect addition of device argument. * syscalls.cc (mknod_worker): New function. (open): Use build_fh_name to build fhandler. (chown_worker): Detect if this is an 'auto' device rather than an on-filesystem device and handle appropriately. (chmod_device): New function. (chmod): Detect if this is an 'auto' device rather than an on-filesystem device and handle appropriately. Use chmod_device to set mode of in-filesystem devices. (stat_worker): Eliminate path_conv argument. Call build_fh_name to construct fhandler. Use fh->error() rather than pc->error to detect errors in fhandler construction. (access_worker): New function pulled from access. Accommodate in-filesystem devices. (access): Use access_worker. (fpathconf): Detect if this is an 'auto' device rather than an on-filesystem device and handle appropriately. (mknod_worker): New function. (mknod32): New function. (chroot): Free normalized path -- assuming it was actually cmalloced. * tty.cc (create_tty_master): Tweak for new device class. (tty::common_init): Ditto. * winsup.h (stat_worker): Remove. (symlink_worker): Declare. * exceptions.cc (set_process_mask): Just call sig_dispatch_pending and don't worry about pending_signals since sig_dispatch_pending should always do the right thing now. (sig_handle): Reorganize SIGCONT handling to more closely conform to SUSv3. * pinfo.h: Move __SIG enum to sigproc.h. (PICOM_FIFO): New enum element. (_pinfo): Remove 'thread2signal' stuff throughout class. (_pinfo::commune_send): Make varargs. (_pinfo::sigtodo): Eliminate. (_pinfo::thread2signal): Ditto. * signal.cc (kill_worker): Eliminate call to setthread2signal. * sigproc.cc (local_sigtodo): Eliminate. (getlocal_sigtodo): Ditto. (sigelem): New class. (pending_signals): New class. (sigqueue): New variable, start of sigqueue linked list. (sigcatch_nonmain): Eliminate. (sigcatch_main): Eliminate. (sigcatch_nosync): Eliminate. (sigcomplete_nonmain): Eliminate. (pending_signals): Eliminate. (sig_clear): Call signal thread to clear pending signals, unless already in signal thread. (sigpending): Call signal thread to get pending signals. (sig_dispatch_pending): Eliminate use of pending_signals and just check sigqueue. (sigproc_terminate): Eliminate all of the obsolete semaphore stuff. Close signal pipe handle. (sig_send): Eliminate all of the obsolete semaphore stuff and use pipe to send signals. (getevent): Eliminate. (pending_signals::add): New function. (pending_signals::del): New function. (pending_signals::next): New function. (wait_sig): Eliminate all of the obsolete semaphore stuff. Use pipe to communicate and maintain a linked list of signals. * sigproc.h: Move __SIG defines here. Add __SIGPENDING. (sig_dispatch_pending): Remove "C" specifier. (sig_handle): Accept a mask argument. * thread.cc: Remove signal handling considerations throughout.
1259 lines
34 KiB
C++
1259 lines
34 KiB
C++
/* sigproc.cc: inter/intra signal and sub process handler
|
|
|
|
Copyright 1997, 1998, 1999, 2000, 2001, 2002 Red Hat, Inc.
|
|
|
|
Written by Christopher Faylor <cgf@cygnus.com>
|
|
|
|
This file is part of Cygwin.
|
|
|
|
This software is a copyrighted work licensed under the terms of the
|
|
Cygwin license. Please consult the file "CYGWIN_LICENSE" for
|
|
details. */
|
|
|
|
#include "winsup.h"
|
|
#include <stdlib.h>
|
|
#include <time.h>
|
|
#include <sys/wait.h>
|
|
#include <stdlib.h>
|
|
#include <sys/cygwin.h>
|
|
#include <assert.h>
|
|
#include <sys/signal.h>
|
|
#include "cygerrno.h"
|
|
#include "sync.h"
|
|
#include "sigproc.h"
|
|
#include "pinfo.h"
|
|
#include "security.h"
|
|
#include "path.h"
|
|
#include "fhandler.h"
|
|
#include "dtable.h"
|
|
#include "cygheap.h"
|
|
#include "child_info_magic.h"
|
|
#include "perthread.h"
|
|
#include "shared_info.h"
|
|
#include "cygthread.h"
|
|
|
|
/*
|
|
* Convenience defines
|
|
*/
|
|
#define WSSC 60000 // Wait for signal completion
|
|
#define WPSP 40000 // Wait for proc_subproc mutex
|
|
#define WSPX 20000 // Wait for wait_sig to terminate
|
|
#define WWSP 20000 // Wait for wait_subproc to terminate
|
|
|
|
#define TOTSIGS (NSIG + __SIGOFFSET)
|
|
|
|
#define wake_wait_subproc() SetEvent (events[0])
|
|
|
|
#define no_signals_available() (!hwait_sig || !sig_loop_wait)
|
|
|
|
#define NZOMBIES 256
|
|
|
|
class sigelem
|
|
{
|
|
int sig;
|
|
class sigelem *next;
|
|
friend class pending_signals;
|
|
friend int __stdcall sig_dispatch_pending ();
|
|
};
|
|
|
|
class pending_signals
|
|
{
|
|
sigelem sigs[NSIG + 1];
|
|
sigelem start;
|
|
sigelem *end;
|
|
sigelem *prev;
|
|
sigelem *curr;
|
|
int empty;
|
|
public:
|
|
void reset () {curr = &start; prev = &start;}
|
|
void add (int sig);
|
|
void del ();
|
|
int next ();
|
|
friend int __stdcall sig_dispatch_pending ();
|
|
};
|
|
|
|
static pending_signals sigqueue;
|
|
|
|
struct sigaction *global_sigs;
|
|
|
|
void __stdcall
|
|
sigalloc ()
|
|
{
|
|
cygheap->sigs = global_sigs =
|
|
(struct sigaction *) ccalloc (HEAP_SIGS, NSIG, sizeof (struct sigaction));
|
|
}
|
|
|
|
void __stdcall
|
|
signal_fixup_after_exec ()
|
|
{
|
|
global_sigs = cygheap->sigs;
|
|
/* Set up child's signal handlers */
|
|
for (int i = 0; i < NSIG; i++)
|
|
{
|
|
myself->getsig (i).sa_mask = 0;
|
|
if (myself->getsig (i).sa_handler != SIG_IGN)
|
|
myself->getsig (i).sa_handler = SIG_DFL;
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Global variables
|
|
*/
|
|
const char *__sp_fn ;
|
|
int __sp_ln;
|
|
|
|
char NO_COPY myself_nowait_dummy[1] = {'0'};// Flag to sig_send that signal goes to
|
|
// current process but no wait is required
|
|
char NO_COPY myself_nowait_nonmain_dummy[1] = {'1'};// Flag to sig_send that signal goes to
|
|
// current process but no wait is required
|
|
// if this is the main thread.
|
|
|
|
HANDLE NO_COPY signal_arrived; // Event signaled when a signal has
|
|
// resulted in a user-specified
|
|
// function call
|
|
/*
|
|
* Common variables
|
|
*/
|
|
|
|
|
|
/* How long to wait for message/signals. Normally this is infinite.
|
|
* On termination, however, these are set to zero as a flag to exit.
|
|
*/
|
|
|
|
#define Static static NO_COPY
|
|
|
|
Static DWORD proc_loop_wait = 1000; // Wait for subprocesses to exit
|
|
Static DWORD sig_loop_wait = INFINITE; // Wait for signals to arrive
|
|
|
|
Static HANDLE sigcomplete_main; // Event signaled when a signal has
|
|
// finished processing for the main
|
|
// thread
|
|
HANDLE NO_COPY sigCONT; // Used to "STOP" a process
|
|
Static cygthread *hwait_sig; // Handle of wait_sig thread
|
|
Static cygthread *hwait_subproc; // Handle of sig_subproc thread
|
|
|
|
Static HANDLE wait_sig_inited; // Control synchronization of
|
|
// message queue startup
|
|
|
|
/* Used by WaitForMultipleObjects. These are handles to child processes.
|
|
*/
|
|
Static HANDLE events[PSIZE + 1]; // All my children's handles++
|
|
#define hchildren (events + 1) // Where the children handles begin
|
|
Static char cpchildren[PSIZE * sizeof (pinfo)]; // All my children info
|
|
Static int nchildren; // Number of active children
|
|
Static char czombies[(NZOMBIES + 1) * sizeof (pinfo)]; // All my deceased children info
|
|
Static int nzombies; // Number of deceased children
|
|
|
|
#define pchildren ((pinfo *) cpchildren)
|
|
#define zombies ((pinfo *) czombies)
|
|
|
|
Static waitq waitq_head = {0, 0, 0, 0, 0, 0, 0};// Start of queue for wait'ing threads
|
|
Static waitq waitq_main; // Storage for main thread
|
|
|
|
muto NO_COPY *sync_proc_subproc = NULL; // Control access to subproc stuff
|
|
|
|
DWORD NO_COPY sigtid = 0; // ID of the signal thread
|
|
|
|
/* Functions
|
|
*/
|
|
static int __stdcall checkstate (waitq *) __attribute__ ((regparm (1)));
|
|
static __inline__ BOOL get_proc_lock (DWORD, DWORD);
|
|
static void __stdcall remove_zombie (int);
|
|
static DWORD WINAPI wait_sig (VOID *arg);
|
|
static int __stdcall stopped_or_terminated (waitq *, _pinfo *);
|
|
static DWORD WINAPI wait_subproc (VOID *);
|
|
|
|
/* Determine if the parent process is alive.
|
|
*/
|
|
|
|
BOOL __stdcall
|
|
my_parent_is_alive ()
|
|
{
|
|
DWORD res;
|
|
if (!myself->ppid_handle)
|
|
{
|
|
debug_printf ("No myself->ppid_handle");
|
|
res = FALSE;
|
|
}
|
|
else
|
|
for (int i = 0; i < 2; i++)
|
|
switch (res = WaitForSingleObject (myself->ppid_handle, 0))
|
|
{
|
|
case WAIT_OBJECT_0:
|
|
debug_printf ("parent dead.");
|
|
res = FALSE;
|
|
goto out;
|
|
case WAIT_TIMEOUT:
|
|
debug_printf ("parent still alive");
|
|
res = TRUE;
|
|
goto out;
|
|
case WAIT_FAILED:
|
|
DWORD werr = GetLastError ();
|
|
if (werr == ERROR_INVALID_HANDLE && i == 0)
|
|
continue;
|
|
system_printf ("WFSO for myself->ppid_handle(%p) failed, error %d",
|
|
myself->ppid_handle, werr);
|
|
res = FALSE;
|
|
goto out;
|
|
}
|
|
out:
|
|
return res;
|
|
}
|
|
|
|
void __stdcall
|
|
wait_for_sigthread ()
|
|
{
|
|
sigproc_printf ("wait_sig_inited %p", wait_sig_inited);
|
|
HANDLE hsig_inited = wait_sig_inited;
|
|
(void) WaitForSingleObject (hsig_inited, INFINITE);
|
|
wait_sig_inited = NULL;
|
|
(void) ForceCloseHandle1 (hsig_inited, wait_sig_inited);
|
|
}
|
|
|
|
/* Get the sync_proc_subproc muto to control access to
|
|
* children, zombie arrays.
|
|
* Attempt to handle case where process is exiting as we try to grab
|
|
* the mutex.
|
|
*/
|
|
static BOOL
|
|
get_proc_lock (DWORD what, DWORD val)
|
|
{
|
|
Static int lastwhat = -1;
|
|
if (!sync_proc_subproc)
|
|
return FALSE;
|
|
if (sync_proc_subproc->acquire (WPSP))
|
|
{
|
|
lastwhat = what;
|
|
return TRUE;
|
|
}
|
|
if (!sync_proc_subproc)
|
|
return FALSE;
|
|
system_printf ("Couldn't aquire sync_proc_subproc for(%d,%d), %E, last %d",
|
|
what, val, lastwhat);
|
|
return TRUE;
|
|
}
|
|
|
|
static BOOL __stdcall
|
|
proc_can_be_signalled (_pinfo *p)
|
|
{
|
|
if (p == myself_nowait || p == myself_nowait_nonmain || p == myself)
|
|
{
|
|
assert (!wait_sig_inited);
|
|
return 1;
|
|
}
|
|
|
|
return ISSTATE (p, PID_INITIALIZING) ||
|
|
(((p)->process_state & (PID_ACTIVE | PID_IN_USE)) ==
|
|
(PID_ACTIVE | PID_IN_USE));
|
|
}
|
|
|
|
BOOL __stdcall
|
|
pid_exists (pid_t pid)
|
|
{
|
|
pinfo p (pid);
|
|
return proc_exists (p);
|
|
}
|
|
|
|
/* Test to determine if a process really exists and is processing signals.
|
|
*/
|
|
BOOL __stdcall
|
|
proc_exists (_pinfo *p)
|
|
{
|
|
return p && !(p->process_state & PID_EXITED);
|
|
}
|
|
|
|
/* Return 1 if this is one of our children, zero otherwise.
|
|
FIXME: This really should be integrated with the rest of the proc_subproc
|
|
testing. Scanning these lists twice is inefficient. */
|
|
int __stdcall
|
|
mychild (int pid)
|
|
{
|
|
for (int i = 0; i < nchildren; i++)
|
|
if (pchildren[i]->pid == pid)
|
|
return 1;
|
|
for (int i = 0; i < nzombies; i++)
|
|
if (zombies[i]->pid == pid)
|
|
return 1;
|
|
return 0;
|
|
}
|
|
|
|
/* Handle all subprocess requests
|
|
*/
|
|
#define vchild (*((pinfo *) val))
|
|
int __stdcall
|
|
proc_subproc (DWORD what, DWORD val)
|
|
{
|
|
int rc = 1;
|
|
int potential_match;
|
|
_pinfo *child;
|
|
int clearing;
|
|
waitq *w;
|
|
|
|
#define wval ((waitq *) val)
|
|
|
|
sigproc_printf ("args: %x, %d", what, val);
|
|
|
|
if (!get_proc_lock (what, val)) // Serialize access to this function
|
|
{
|
|
system_printf ("couldn't get proc lock. Something is wrong.");
|
|
goto out1;
|
|
}
|
|
|
|
switch (what)
|
|
{
|
|
/* Add a new subprocess to the children arrays.
|
|
* (usually called from the main thread)
|
|
*/
|
|
case PROC_ADDCHILD:
|
|
if (nchildren >= PSIZE - 1)
|
|
{
|
|
rc = 0;
|
|
break;
|
|
}
|
|
pchildren[nchildren] = vchild;
|
|
hchildren[nchildren] = vchild->hProcess;
|
|
if (!DuplicateHandle (hMainProc, vchild->hProcess, hMainProc, &vchild->pid_handle,
|
|
0, 0, DUPLICATE_SAME_ACCESS))
|
|
system_printf ("Couldn't duplicate child handle for pid %d, %E", vchild->pid);
|
|
ProtectHandle1 (vchild->pid_handle, pid_handle);
|
|
|
|
if (!DuplicateHandle (hMainProc, hMainProc, vchild->hProcess, &vchild->ppid_handle,
|
|
0, TRUE, DUPLICATE_SAME_ACCESS))
|
|
system_printf ("Couldn't duplicate my handle<%p> for pid %d, %E", hMainProc, vchild->pid);
|
|
vchild->ppid = myself->pid;
|
|
vchild->uid = myself->uid;
|
|
vchild->gid = myself->gid;
|
|
vchild->pgid = myself->pgid;
|
|
vchild->sid = myself->sid;
|
|
vchild->ctty = myself->ctty;
|
|
vchild->process_state |= PID_INITIALIZING | (myself->process_state & PID_USETTY);
|
|
|
|
sigproc_printf ("added pid %d to wait list, slot %d, winpid %p, handle %p",
|
|
vchild->pid, nchildren, vchild->dwProcessId,
|
|
vchild->hProcess);
|
|
nchildren++;
|
|
|
|
wake_wait_subproc ();
|
|
break;
|
|
|
|
/* A child process had terminated.
|
|
Possibly this is just due to an exec(). Cygwin implements an exec()
|
|
as a "handoff" from one windows process to another. If child->hProcess
|
|
is different from what is recorded in hchildren, then this is an exec().
|
|
Otherwise this is a normal child termination event.
|
|
(called from wait_subproc thread) */
|
|
case PROC_CHILDTERMINATED:
|
|
if (hchildren[val] != pchildren[val]->hProcess)
|
|
{
|
|
sigproc_printf ("pid %d[%d], reparented old hProcess %p, new %p",
|
|
pchildren[val]->pid, val, hchildren[val], pchildren[val]->hProcess);
|
|
HANDLE h = hchildren[val];
|
|
hchildren[val] = pchildren[val]->hProcess; /* Filled out by child */
|
|
sync_proc_subproc->release (); // Release the lock ASAP
|
|
ForceCloseHandle1 (h, childhProc);
|
|
ProtectHandle1 (pchildren[val]->hProcess, childhProc);
|
|
rc = 0;
|
|
goto out; // This was an exec()
|
|
}
|
|
|
|
sigproc_printf ("pid %d[%d] terminated, handle %p, nchildren %d, nzombies %d",
|
|
pchildren[val]->pid, val, hchildren[val], nchildren, nzombies);
|
|
|
|
int thiszombie;
|
|
thiszombie = nzombies;
|
|
zombies[nzombies] = pchildren[val]; // Add to zombie array
|
|
zombies[nzombies++]->process_state = PID_ZOMBIE;// Walking dead
|
|
|
|
sigproc_printf ("zombifying [%d], pid %d, handle %p, nchildren %d",
|
|
val, pchildren[val]->pid, hchildren[val], nchildren);
|
|
if ((int) val < --nchildren)
|
|
{
|
|
hchildren[val] = hchildren[nchildren];
|
|
pchildren[val] = pchildren[nchildren];
|
|
}
|
|
|
|
/* See if we should care about the this terminated process. If we've
|
|
filled up our table or if we're ignoring SIGCHLD, then we immediately
|
|
remove the process and move on. Otherwise, this process becomes a zombie
|
|
which must be reaped by a wait() call. */
|
|
if (nzombies >= NZOMBIES
|
|
|| myself->getsig (SIGCHLD).sa_handler == (void *) SIG_IGN)
|
|
{
|
|
sigproc_printf ("automatically removing zombie %d", thiszombie);
|
|
remove_zombie (thiszombie);
|
|
}
|
|
|
|
/* Don't scan the wait queue yet. Caller will send SIGCHLD to this process.
|
|
This will cause an eventual scan of waiters. */
|
|
break;
|
|
|
|
/* Handle a wait4() operation. Allocates an event for the calling
|
|
* thread which is signaled when the appropriate pid exits or stops.
|
|
* (usually called from the main thread)
|
|
*/
|
|
case PROC_WAIT:
|
|
wval->ev = NULL; // Don't know event flag yet
|
|
|
|
if (wval->pid <= 0)
|
|
child = NULL; // Not looking for a specific pid
|
|
else if (!mychild (wval->pid))
|
|
goto out; // invalid pid. flag no such child
|
|
|
|
wval->status = 0; // Don't know status yet
|
|
sigproc_printf ("wval->pid %d, wval->options %d", wval->pid, wval->options);
|
|
|
|
/* If the first time for this thread, create a new event, otherwise
|
|
* reset the event.
|
|
*/
|
|
if ((wval->ev = wval->thread_ev) == NULL)
|
|
{
|
|
wval->ev = wval->thread_ev = CreateEvent (&sec_none_nih, TRUE,
|
|
FALSE, NULL);
|
|
ProtectHandle (wval->ev);
|
|
}
|
|
|
|
ResetEvent (wval->ev);
|
|
w = waitq_head.next;
|
|
waitq_head.next = wval; /* Add at the beginning. */
|
|
wval->next = w; /* Link in rest of the list. */
|
|
clearing = 0;
|
|
goto scan_wait;
|
|
|
|
/* Clear all waiting threads. Called from exceptions.cc prior to
|
|
* the main thread's dispatch to a signal handler function.
|
|
* (called from wait_sig thread)
|
|
*/
|
|
case PROC_CLEARWAIT:
|
|
/* Clear all "wait"ing threads. */
|
|
if (val)
|
|
sigproc_printf ("clear waiting threads");
|
|
else
|
|
sigproc_printf ("looking for processes to reap");
|
|
clearing = val;
|
|
|
|
scan_wait:
|
|
/* Scan the linked list of wait()ing threads. If a wait's parameters
|
|
* match this pid, then activate it.
|
|
*/
|
|
for (w = &waitq_head; w->next != NULL; w = w->next)
|
|
{
|
|
if ((potential_match = checkstate (w)) > 0)
|
|
sigproc_printf ("released waiting thread");
|
|
else if (!clearing && !(w->next->options & WNOHANG) && potential_match < 0)
|
|
sigproc_printf ("only found non-terminated children");
|
|
else if (potential_match <= 0) // nothing matched
|
|
{
|
|
sigproc_printf ("waiting thread found no children");
|
|
HANDLE oldw = w->next->ev;
|
|
w->next->pid = 0;
|
|
if (clearing)
|
|
w->next->status = -1; /* flag that a signal was received */
|
|
else if (!potential_match || !(w->next->options & WNOHANG))
|
|
w->next->ev = NULL;
|
|
if (!SetEvent (oldw))
|
|
system_printf ("couldn't wake up wait event %p, %E", oldw);
|
|
w->next = w->next->next;
|
|
}
|
|
if (w->next == NULL)
|
|
break;
|
|
}
|
|
|
|
if (!clearing)
|
|
sigproc_printf ("finished processing terminated/stopped child");
|
|
else
|
|
{
|
|
waitq_head.next = NULL;
|
|
sigproc_printf ("finished clearing");
|
|
}
|
|
break;
|
|
}
|
|
|
|
out:
|
|
sync_proc_subproc->release (); // Release the lock
|
|
out1:
|
|
sigproc_printf ("returning %d", rc);
|
|
return rc;
|
|
}
|
|
|
|
/* Terminate the wait_subproc thread.
|
|
* Called on process exit.
|
|
* Also called by spawn_guts to disassociate any subprocesses from this
|
|
* process. Subprocesses will then know to clean up after themselves and
|
|
* will not become zombies.
|
|
*/
|
|
void __stdcall
|
|
proc_terminate (void)
|
|
{
|
|
sigproc_printf ("nchildren %d, nzombies %d", nchildren, nzombies);
|
|
/* Signal processing is assumed to be blocked in this routine. */
|
|
if (hwait_subproc)
|
|
{
|
|
proc_loop_wait = 0; // Tell wait_subproc thread to exit
|
|
sync_proc_subproc->acquire (WPSP);
|
|
wake_wait_subproc (); // Wake wait_subproc loop
|
|
hwait_subproc = NULL;
|
|
|
|
(void) proc_subproc (PROC_CLEARWAIT, 1);
|
|
|
|
/* Clean out zombie processes from the pid list. */
|
|
int i;
|
|
for (i = 0; i < nzombies; i++)
|
|
{
|
|
if (zombies[i]->hProcess)
|
|
{
|
|
ForceCloseHandle1 (zombies[i]->hProcess, childhProc);
|
|
ForceCloseHandle1 (zombies[i]->pid_handle, pid_handle);
|
|
}
|
|
zombies[i]->ppid = 1;
|
|
zombies[i]->process_state = PID_EXITED; /* CGF FIXME - still needed? */
|
|
zombies[i].release (); // FIXME: this breaks older gccs for some reason
|
|
}
|
|
|
|
/* Disassociate my subprocesses */
|
|
for (i = 0; i < nchildren; i++)
|
|
{
|
|
if (!pchildren[i]->hProcess)
|
|
sigproc_printf ("%d(%d) hProcess cleared already?", pchildren[i]->pid,
|
|
pchildren[i]->dwProcessId);
|
|
else
|
|
{
|
|
ForceCloseHandle1 (pchildren[i]->hProcess, childhProc);
|
|
sigproc_printf ("%d(%d) closed child handle", pchildren[i]->pid,
|
|
pchildren[i]->dwProcessId);
|
|
pchildren[i]->ppid = 1;
|
|
if (pchildren[i]->pgid == myself->pid)
|
|
pchildren[i]->process_state |= PID_ORPHANED;
|
|
}
|
|
pchildren[i].release ();
|
|
}
|
|
nchildren = nzombies = 0;
|
|
/* Just zero sync_proc_subproc as the delete below seems to cause
|
|
problems for older gccs. */
|
|
sync_proc_subproc = NULL;
|
|
}
|
|
sigproc_printf ("leaving");
|
|
}
|
|
|
|
/* Clear pending signal */
|
|
void __stdcall
|
|
sig_clear (int target_sig)
|
|
{
|
|
if (GetCurrentThreadId () != sigtid)
|
|
sig_send (myself, -target_sig);
|
|
else
|
|
{
|
|
int sig;
|
|
sigqueue.reset ();
|
|
while ((sig = sigqueue.next ()))
|
|
if (sig == target_sig)
|
|
{
|
|
sigqueue.del ();
|
|
break;
|
|
}
|
|
}
|
|
return;
|
|
}
|
|
|
|
extern "C" int
|
|
sigpending (sigset_t *mask)
|
|
{
|
|
sigset_t outset = (sigset_t) sig_send (myself, __SIGPENDING);
|
|
if (outset == SIG_BAD_MASK)
|
|
return -1;
|
|
*mask = outset;
|
|
return 0;
|
|
}
|
|
|
|
/* Force the wait_sig thread to wake up and scan for pending signals */
|
|
int __stdcall
|
|
sig_dispatch_pending ()
|
|
{
|
|
if (exit_state || GetCurrentThreadId () == sigtid || !sigqueue.start.next)
|
|
return 0;
|
|
|
|
sigframe thisframe (mainthread);
|
|
(void) sig_send (myself, __SIGFLUSH);
|
|
return thisframe.call_signal_handler ();
|
|
}
|
|
|
|
/* Message initialization. Called from dll_crt0_1
|
|
*
|
|
* This routine starts the signal handling thread. The wait_sig_inited
|
|
* event is used to signal that the thread is ready to handle signals.
|
|
* We don't wait for this during initialization but instead detect it
|
|
* in sig_send to gain a little concurrency.
|
|
*/
|
|
void __stdcall
|
|
sigproc_init ()
|
|
{
|
|
wait_sig_inited = CreateEvent (&sec_none_nih, TRUE, FALSE, NULL);
|
|
ProtectHandle (wait_sig_inited);
|
|
|
|
/* sync_proc_subproc is used by proc_subproc. It serialises
|
|
* access to the children and zombie arrays.
|
|
*/
|
|
new_muto (sync_proc_subproc);
|
|
|
|
/* local event signaled when main thread has been dispatched
|
|
to a signal handler function. */
|
|
signal_arrived = CreateEvent (&sec_none_nih, TRUE, FALSE, NULL);
|
|
ProtectHandle (signal_arrived);
|
|
|
|
hwait_sig = new cygthread (wait_sig, cygself, "sig");
|
|
hwait_sig->zap_h ();
|
|
|
|
/* Initialize waitq structure for main thread. A waitq structure is
|
|
* allocated for each thread that executes a wait to allow multiple threads
|
|
* to perform waits. Pre-allocate a waitq structure for the main thread.
|
|
*/
|
|
waitq *w;
|
|
if ((w = (waitq *)waitq_storage.get ()) == NULL)
|
|
{
|
|
w = &waitq_main;
|
|
waitq_storage.set (w);
|
|
}
|
|
memset (w, 0, sizeof *w); // Just to be safe
|
|
|
|
myself->getsig (SIGSTOP).sa_flags = SA_RESTART | SA_NODEFER;
|
|
sigproc_printf ("process/signal handling enabled(%x)", myself->process_state);
|
|
return;
|
|
}
|
|
|
|
/* Called on process termination to terminate signal and process threads.
|
|
*/
|
|
void __stdcall
|
|
sigproc_terminate (void)
|
|
{
|
|
hwait_sig = NULL;
|
|
|
|
if (!sig_loop_wait)
|
|
sigproc_printf ("sigproc handling not active");
|
|
else
|
|
{
|
|
sigproc_printf ("entering");
|
|
sig_loop_wait = 0; // Tell wait_sig to exit when it is
|
|
// finished with anything it is doing
|
|
ForceCloseHandle (sigcomplete_main);
|
|
CloseHandle (myself->sendsig);
|
|
}
|
|
proc_terminate (); // Terminate process handling thread
|
|
|
|
return;
|
|
}
|
|
|
|
/* Send a signal to another process by raising its signal semaphore.
|
|
* If pinfo *p == NULL, send to the current process.
|
|
* If sending to this process, wait for notification that a signal has
|
|
* completed before returning.
|
|
*/
|
|
int __stdcall
|
|
sig_send (_pinfo *p, int sig, DWORD ebp, bool exception)
|
|
{
|
|
int rc = 1;
|
|
DWORD tid = GetCurrentThreadId ();
|
|
BOOL its_me;
|
|
HANDLE thiscomplete;
|
|
HANDLE sendsig;
|
|
bool wait_for_completion;
|
|
sigframe thisframe;
|
|
|
|
if (p == myself_nowait_nonmain)
|
|
p = (tid == mainthread.id) ? (_pinfo *) myself : myself_nowait;
|
|
if (!(its_me = (p == NULL || p == myself || p == myself_nowait)))
|
|
wait_for_completion = false;
|
|
else
|
|
{
|
|
if (no_signals_available ())
|
|
goto out; // Either exiting or not yet initializing
|
|
if (wait_sig_inited)
|
|
wait_for_sigthread ();
|
|
wait_for_completion = p != myself_nowait;
|
|
p = myself;
|
|
}
|
|
|
|
/* It is possible that the process is not yet ready to receive messages
|
|
* or that it has exited. Detect this.
|
|
*/
|
|
if (!proc_can_be_signalled (p)) /* Is the process accepting messages? */
|
|
{
|
|
sigproc_printf ("invalid pid %d(%x), signal %d",
|
|
p->pid, p->process_state, sig);
|
|
set_errno (ESRCH);
|
|
goto out;
|
|
}
|
|
|
|
sigproc_printf ("pid %d, signal %d, its_me %d", p->pid, sig, its_me);
|
|
|
|
if (its_me)
|
|
{
|
|
sendsig = myself->sendsig;
|
|
if (wait_for_completion)
|
|
{
|
|
if (tid == mainthread.id)
|
|
thisframe.init (mainthread, ebp, exception);
|
|
thiscomplete = sigcomplete_main;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
HANDLE hp = OpenProcess (PROCESS_DUP_HANDLE, false, p->dwProcessId);
|
|
if (!hp)
|
|
{
|
|
__seterrno ();
|
|
goto out;
|
|
}
|
|
if (!DuplicateHandle (hp, p->sendsig, hMainProc, &sendsig, false, 0,
|
|
DUPLICATE_SAME_ACCESS) || !sendsig)
|
|
{
|
|
__seterrno ();
|
|
goto out;
|
|
}
|
|
thiscomplete = NULL;
|
|
}
|
|
|
|
DWORD nb;
|
|
if (!WriteFile (sendsig, &sig, sizeof (sig), &nb, NULL) || nb != sizeof (sig))
|
|
{
|
|
/* Couldn't signal the semaphore. This probably means that the
|
|
* process is exiting.
|
|
*/
|
|
if (!its_me)
|
|
{
|
|
__seterrno ();
|
|
ForceCloseHandle (sendsig);
|
|
}
|
|
else
|
|
{
|
|
if (no_signals_available ())
|
|
sigproc_printf ("I'm going away now");
|
|
else
|
|
system_printf ("error sending signal %d to pid %d, pipe handle %p, %E",
|
|
sig, p->pid, sendsig);
|
|
}
|
|
goto out;
|
|
}
|
|
|
|
/* Write completion handle or NULL */
|
|
if (!WriteFile (sendsig, &thiscomplete, sizeof (thiscomplete), &nb, NULL)
|
|
|| nb != sizeof (thiscomplete))
|
|
{
|
|
__seterrno ();
|
|
goto out;
|
|
}
|
|
|
|
sigset_t pending;
|
|
sigset_t *mask;
|
|
if (sig == __SIGPENDING)
|
|
mask = &pending;
|
|
else if (sig == __SIGFLUSH || sig > 0)
|
|
mask = &myself->getsigmask ();
|
|
else
|
|
mask = NULL;
|
|
if (mask && !WriteFile (sendsig, &mask, sizeof (mask), &nb, NULL)
|
|
|| nb != sizeof (pending))
|
|
{
|
|
__seterrno ();
|
|
goto out;
|
|
}
|
|
|
|
/* No need to wait for signal completion unless this was a signal to
|
|
* this process.
|
|
*
|
|
* If it was a signal to this process, wait for a dispatched signal.
|
|
* Otherwise just wait for the wait_sig to signal that it has finished
|
|
* processing the signal.
|
|
*/
|
|
if (!wait_for_completion)
|
|
{
|
|
rc = WAIT_OBJECT_0;
|
|
sigproc_printf ("Not waiting for sigcomplete. its_me %d signal %d", its_me, sig);
|
|
if (!its_me)
|
|
ForceCloseHandle (sendsig);
|
|
}
|
|
else
|
|
{
|
|
sigproc_printf ("Waiting for thiscomplete %p", thiscomplete);
|
|
rc = WaitForSingleObject (thiscomplete, WSSC);
|
|
}
|
|
|
|
if (rc == WAIT_OBJECT_0)
|
|
rc = 0; // Successful exit
|
|
else
|
|
{
|
|
/* It's an error unless sig_loop_wait == 0 (the process is exiting). */
|
|
if (!no_signals_available ())
|
|
system_printf ("wait for sig_complete event failed, signal %d, rc %d, %E",
|
|
sig, rc);
|
|
set_errno (ENOSYS);
|
|
rc = -1;
|
|
}
|
|
|
|
out:
|
|
if (sig != __SIGPENDING)
|
|
/* nothing */;
|
|
else if (!rc)
|
|
rc = (int) pending;
|
|
else
|
|
rc = SIG_BAD_MASK;
|
|
sigproc_printf ("returning %p from sending signal %d", rc, sig);
|
|
return rc;
|
|
}
|
|
|
|
/* Initialize the wait_subproc thread.
|
|
* Called from fork() or spawn() to initialize the handling of subprocesses.
|
|
*/
|
|
void __stdcall
|
|
subproc_init (void)
|
|
{
|
|
if (hwait_subproc)
|
|
return;
|
|
|
|
/* A "wakeup" handle which can be toggled to make wait_subproc reexamine
|
|
* the hchildren array.
|
|
*/
|
|
events[0] = CreateEvent (&sec_none_nih, FALSE, FALSE, NULL);
|
|
hwait_subproc = new cygthread (wait_subproc, NULL, "proc");
|
|
hwait_subproc->zap_h ();
|
|
ProtectHandle (events[0]);
|
|
sigproc_printf ("started wait_subproc thread");
|
|
}
|
|
|
|
/* Initialize some of the memory block passed to child processes
|
|
by fork/spawn/exec. */
|
|
|
|
void __stdcall
|
|
init_child_info (DWORD chtype, child_info *ch, pid_t pid, HANDLE subproc_ready)
|
|
{
|
|
memset (ch, 0, sizeof *ch);
|
|
ch->cb = chtype == PROC_FORK ? sizeof (child_info_fork) : sizeof (child_info);
|
|
ch->intro = PROC_MAGIC_GENERIC;
|
|
ch->magic = CHILD_INFO_MAGIC;
|
|
ch->type = chtype;
|
|
ch->cygpid = pid;
|
|
ch->subproc_ready = subproc_ready;
|
|
ch->pppid_handle = myself->ppid_handle;
|
|
ch->fhandler_union_cb = sizeof (fhandler_union);
|
|
ch->mount_h = cygwin_mount_h;
|
|
}
|
|
|
|
/* Check the state of all of our children to see if any are stopped or
|
|
* terminated.
|
|
*/
|
|
static int __stdcall
|
|
checkstate (waitq *parent_w)
|
|
{
|
|
int potential_match = 0;
|
|
|
|
sigproc_printf ("nchildren %d, nzombies %d", nchildren, nzombies);
|
|
|
|
/* Check already dead processes first to see if they match the criteria
|
|
* given in w->next.
|
|
*/
|
|
for (int i = 0; i < nzombies; i++)
|
|
switch (stopped_or_terminated (parent_w, zombies[i]))
|
|
{
|
|
case -1:
|
|
potential_match = -1;
|
|
break;
|
|
case 1:
|
|
remove_zombie (i);
|
|
potential_match = 1;
|
|
goto out;
|
|
}
|
|
|
|
sigproc_printf ("checking alive children");
|
|
|
|
/* No dead terminated children matched. Check for stopped children. */
|
|
for (int i = 0; i < nchildren; i++)
|
|
switch (stopped_or_terminated (parent_w, pchildren[i]))
|
|
{
|
|
case -1:
|
|
potential_match = -1;
|
|
break;
|
|
case 1:
|
|
potential_match = 1;
|
|
goto out;
|
|
}
|
|
|
|
out:
|
|
sigproc_printf ("returning %d", potential_match);
|
|
return potential_match;
|
|
}
|
|
|
|
/* Remove a zombie from zombies by swapping it with the last child in the list.
|
|
*/
|
|
static void __stdcall
|
|
remove_zombie (int ci)
|
|
{
|
|
sigproc_printf ("removing %d, pid %d, nzombies %d", ci, zombies[ci]->pid,
|
|
nzombies);
|
|
|
|
if (zombies[ci])
|
|
{
|
|
ForceCloseHandle1 (zombies[ci]->hProcess, childhProc);
|
|
ForceCloseHandle1 (zombies[ci]->pid_handle, pid_handle);
|
|
zombies[ci].release ();
|
|
}
|
|
|
|
if (ci < --nzombies)
|
|
zombies[ci] = zombies[nzombies];
|
|
|
|
return;
|
|
}
|
|
|
|
/* Check status of child process vs. waitq member.
|
|
*
|
|
* parent_w is the pointer to the parent of the waitq member in question.
|
|
* child is the subprocess being considered.
|
|
*
|
|
* Returns
|
|
* 1 if stopped or terminated child matches parent_w->next criteria
|
|
* -1 if a non-stopped/terminated child matches parent_w->next criteria
|
|
* 0 if child does not match parent_w->next criteria
|
|
*/
|
|
static int __stdcall
|
|
stopped_or_terminated (waitq *parent_w, _pinfo *child)
|
|
{
|
|
int potential_match;
|
|
waitq *w = parent_w->next;
|
|
|
|
sigproc_printf ("considering pid %d", child->pid);
|
|
if (w->pid == -1)
|
|
potential_match = 1;
|
|
else if (w->pid == 0)
|
|
potential_match = child->pgid == myself->pgid;
|
|
else if (w->pid < 0)
|
|
potential_match = child->pgid == -w->pid;
|
|
else
|
|
potential_match = (w->pid == child->pid);
|
|
|
|
if (!potential_match)
|
|
return 0;
|
|
|
|
BOOL terminated;
|
|
|
|
if ((terminated = child->process_state == PID_ZOMBIE) ||
|
|
((w->options & WUNTRACED) && child->stopsig))
|
|
{
|
|
parent_w->next = w->next; /* successful wait. remove from wait queue */
|
|
w->pid = child->pid;
|
|
|
|
if (!terminated)
|
|
{
|
|
sigproc_printf ("stopped child");
|
|
w->status = (child->stopsig << 8) | 0x7f;
|
|
child->stopsig = 0;
|
|
}
|
|
else /* Should only get here when child has been moved to the zombies array */
|
|
{
|
|
DWORD status;
|
|
if (!GetExitCodeProcess (child->hProcess, &status))
|
|
status = 0xffff;
|
|
if (status & EXIT_SIGNAL)
|
|
w->status = (status >> 8) & 0xff; /* exited due to signal */
|
|
else
|
|
w->status = (status & 0xff) << 8; /* exited via "exit ()" */
|
|
|
|
add_rusage (&myself->rusage_children, &child->rusage_children);
|
|
add_rusage (&myself->rusage_children, &child->rusage_self);
|
|
|
|
if (w->rusage)
|
|
{
|
|
add_rusage ((struct rusage *) w->rusage, &child->rusage_children);
|
|
add_rusage ((struct rusage *) w->rusage, &child->rusage_self);
|
|
}
|
|
}
|
|
|
|
if (!SetEvent (w->ev)) /* wake up wait4 () immediately */
|
|
system_printf ("couldn't wake up wait event %p, %E", w->ev);
|
|
return 1;
|
|
}
|
|
|
|
return -potential_match;
|
|
}
|
|
|
|
static void
|
|
talktome ()
|
|
{
|
|
winpids pids ((DWORD) PID_MAP_RW);
|
|
for (unsigned i = 0; i < pids.npids; i++)
|
|
if (pids[i]->hello_pid == myself->pid)
|
|
if (!IsBadWritePtr (pids[i], sizeof (_pinfo)))
|
|
pids[i]->commune_recv ();
|
|
}
|
|
|
|
/* Process signals by waiting for a semaphore to become signaled.
|
|
Then scan an in-memory array representing queued signals.
|
|
Executes in a separate thread.
|
|
|
|
Signals sent from this process are sent a completion signal so
|
|
that returns from kill/raise do not occur until the signal has
|
|
has been handled, as per POSIX. */
|
|
|
|
void
|
|
pending_signals::add (int sig)
|
|
{
|
|
sigelem *se;
|
|
for (se = start.next; se; se = se->next)
|
|
if (se->sig == sig)
|
|
return;
|
|
while (sigs[empty].sig)
|
|
if (++empty == NSIG)
|
|
empty = 0;
|
|
se = sigs + empty;
|
|
se->sig = sig;
|
|
se->next = NULL;
|
|
if (end)
|
|
end->next = se;
|
|
end = se;
|
|
if (!start.next)
|
|
start.next = se;
|
|
empty++;
|
|
}
|
|
|
|
void
|
|
pending_signals::del ()
|
|
{
|
|
sigelem *next = curr->next;
|
|
prev->next = next;
|
|
curr->sig = 0;
|
|
#ifdef DEBUGGING
|
|
curr->next = NULL;
|
|
#endif
|
|
if (end == curr)
|
|
end = prev;
|
|
empty = curr - sigs;
|
|
curr = next;
|
|
}
|
|
|
|
int
|
|
pending_signals::next ()
|
|
{
|
|
int sig;
|
|
prev = curr;
|
|
if (!curr || !(curr = curr->next))
|
|
sig = 0;
|
|
else
|
|
sig = curr->sig;
|
|
return sig;
|
|
}
|
|
|
|
/* Process signals by waiting for signal data to arrive in a pipe.
|
|
Set a completion event if one was specified. */
|
|
static DWORD WINAPI
|
|
wait_sig (VOID *self)
|
|
{
|
|
HANDLE readsig;
|
|
char sa_buf[1024];
|
|
|
|
/* Initialization */
|
|
(void) SetThreadPriority (GetCurrentThread (), WAIT_SIG_PRIORITY);
|
|
|
|
/* sigcomplete_main - event used to signal main thread on signal
|
|
completion */
|
|
if (!CreatePipe (&readsig, &myself->sendsig, sec_user_nih (sa_buf), 0))
|
|
api_fatal ("couldn't create signal pipe, %E");
|
|
sigcomplete_main = CreateEvent (&sec_none_nih, FALSE, FALSE, NULL);
|
|
sigproc_printf ("sigcomplete_main %p", sigcomplete_main);
|
|
sigCONT = CreateEvent (&sec_none_nih, FALSE, FALSE, NULL);
|
|
|
|
/* Setting dwProcessId flags that this process is now capable of receiving
|
|
signals. Prior to this, dwProcessId was set to the windows pid of
|
|
of the original windows process which spawned us unless this was a
|
|
"toplevel" process. */
|
|
myself->dwProcessId = GetCurrentProcessId ();
|
|
myself->process_state |= PID_ACTIVE;
|
|
myself->process_state &= ~PID_INITIALIZING;
|
|
|
|
ProtectHandle (sigcomplete_main);
|
|
|
|
/* If we've been execed, then there is still a stub left in the previous
|
|
windows process waiting to see if it's started a cygwin process or not.
|
|
Signalling subproc_ready indicates that we are a cygwin process. */
|
|
if (child_proc_info && child_proc_info->type == PROC_EXEC)
|
|
{
|
|
debug_printf ("subproc_ready %p", child_proc_info->subproc_ready);
|
|
if (!SetEvent (child_proc_info->subproc_ready))
|
|
system_printf ("SetEvent (subproc_ready) failed, %E");
|
|
ForceCloseHandle1 (child_proc_info->subproc_ready, subproc_ready);
|
|
/* Initialize an "indirect" pid block so that if someone looks up this
|
|
process via its Windows PID it will be redirected to the appropriate
|
|
Cygwin PID shared memory block. */
|
|
static pinfo NO_COPY myself_identity;
|
|
myself_identity.init (cygwin_pid (myself->dwProcessId), PID_EXECED);
|
|
}
|
|
|
|
SetEvent (wait_sig_inited);
|
|
sigtid = GetCurrentThreadId ();
|
|
|
|
for (;;)
|
|
{
|
|
int sig;
|
|
DWORD nb;
|
|
if (!ReadFile (readsig, &sig, sizeof (sig), &nb, NULL))
|
|
break;
|
|
|
|
HANDLE wakeup;
|
|
if (!ReadFile (readsig, &wakeup, sizeof (wakeup), &nb, NULL)
|
|
|| nb != sizeof (wakeup))
|
|
{
|
|
system_printf ("signal notification handle read failure, %E");
|
|
continue;
|
|
}
|
|
|
|
if (!sig)
|
|
continue; /* Just checking to see if we exist */
|
|
|
|
sigset_t *mask;
|
|
if ((sig == __SIGFLUSH || sig == __SIGPENDING || sig > 0)
|
|
&& (!ReadFile (readsig, &mask, sizeof (mask), &nb, NULL)
|
|
|| nb != sizeof (mask)))
|
|
{
|
|
system_printf ("signal mask handle read failure, %E");
|
|
continue;
|
|
}
|
|
|
|
switch (sig)
|
|
{
|
|
case __SIGCOMMUNE:
|
|
talktome ();
|
|
continue;
|
|
case __SIGSTRACE:
|
|
strace.hello ();
|
|
continue;
|
|
case __SIGPENDING:
|
|
*mask = 0;
|
|
unsigned bit;
|
|
sigqueue.reset ();
|
|
while ((sig = sigqueue.next ()))
|
|
if (myself->getsigmask () & (bit = SIGTOMASK (sig)))
|
|
*mask |= bit;
|
|
break;
|
|
default:
|
|
if (sig < 0)
|
|
sig_clear (-sig);
|
|
else
|
|
{
|
|
int sh;
|
|
for (int i = 0; !(sh = sig_handle (sig, *mask)) && i < 100 ; i++)
|
|
low_priority_sleep (0); // hopefully a temporary condition
|
|
if (sh <= 0)
|
|
sigqueue.add (sig); // FIXME: Shouldn't add this in !sh condition
|
|
if (sig == SIGCHLD)
|
|
proc_subproc (PROC_CLEARWAIT, 0);
|
|
}
|
|
case __SIGFLUSH:
|
|
sigqueue.reset ();
|
|
while ((sig = sigqueue.next ()))
|
|
if (sig_handle (sig, *mask) > 0)
|
|
sigqueue.del ();
|
|
break;
|
|
}
|
|
if (wakeup)
|
|
SetEvent (wakeup);
|
|
}
|
|
|
|
sigproc_printf ("done");
|
|
ExitThread (0);
|
|
}
|
|
|
|
/* Wait for subprocesses to terminate. Executes in a separate thread. */
|
|
static DWORD WINAPI
|
|
wait_subproc (VOID *)
|
|
{
|
|
sigproc_printf ("starting");
|
|
int errloop = 0;
|
|
|
|
for (;;)
|
|
{
|
|
DWORD rc = WaitForMultipleObjects (nchildren + 1, events, FALSE,
|
|
proc_loop_wait);
|
|
if (rc == WAIT_TIMEOUT)
|
|
if (!proc_loop_wait)
|
|
break; // Exiting
|
|
else
|
|
continue;
|
|
|
|
if (rc == WAIT_FAILED)
|
|
{
|
|
if (!proc_loop_wait)
|
|
break;
|
|
|
|
/* It's ok to get an ERROR_INVALID_HANDLE since another thread may have
|
|
closed a handle in the children[] array. So, we try looping a couple
|
|
of times to stabilize. FIXME - this is not foolproof. Probably, this
|
|
thread should be responsible for closing the children. */
|
|
if (!errloop++)
|
|
proc_subproc (PROC_NOTHING, 0); // Just synchronize and continue
|
|
if (errloop < 10)
|
|
continue;
|
|
|
|
system_printf ("wait failed. nchildren %d, wait %d, %E",
|
|
nchildren, proc_loop_wait);
|
|
|
|
for (int i = 0; i <= nchildren; i++)
|
|
if ((rc = WaitForSingleObject (events[i], 0)) == WAIT_OBJECT_0 ||
|
|
rc == WAIT_TIMEOUT)
|
|
continue;
|
|
else if (i == 0)
|
|
system_printf ("nchildren %d, event[%d] %p, %E", nchildren, i, events[i]);
|
|
else
|
|
{
|
|
system_printf ("nchildren %d, event[%d] %p, pchildren[%d] %p, events[0] %p, %E",
|
|
nchildren, i, events[i], i - 1, (_pinfo *) pchildren[i - 1], events[0]);
|
|
system_printf ("pid %d, dwProcessId %u, hProcess %p, progname '%s'",
|
|
pchildren[i - 1]->pid, pchildren[i - 1]->dwProcessId,
|
|
pchildren[i - 1]->hProcess, pchildren[i - 1]->progname);
|
|
}
|
|
break;
|
|
}
|
|
|
|
errloop = 0;
|
|
rc -= WAIT_OBJECT_0;
|
|
if (rc-- != 0)
|
|
{
|
|
rc = proc_subproc (PROC_CHILDTERMINATED, rc);
|
|
if (!proc_loop_wait) // Don't bother if wait_subproc is
|
|
break; // exiting
|
|
|
|
/* Send a SIGCHLD to myself. We do this here, rather than in proc_subproc
|
|
to avoid the proc_subproc lock since the signal thread will eventually
|
|
be calling proc_subproc and could unnecessarily block. */
|
|
if (rc)
|
|
sig_send (myself_nowait, SIGCHLD);
|
|
}
|
|
sigproc_printf ("looping");
|
|
}
|
|
|
|
ForceCloseHandle (events[0]);
|
|
events[0] = NULL;
|
|
sigproc_printf ("done");
|
|
ExitThread (0);
|
|
}
|
|
|
|
extern "C" {
|
|
/* Provide a stack frame when calling WaitFor* functions */
|
|
|
|
#undef WaitForSingleObject
|
|
|
|
DWORD __stdcall
|
|
WFSO (HANDLE hHandle, DWORD dwMilliseconds)
|
|
{
|
|
DWORD ret;
|
|
sigframe thisframe (mainthread);
|
|
ret = WaitForSingleObject (hHandle, dwMilliseconds);
|
|
return ret;
|
|
}
|
|
|
|
#undef WaitForMultipleObjects
|
|
|
|
DWORD __stdcall
|
|
WFMO (DWORD nCount, CONST HANDLE *lpHandles, BOOL fWaitAll, DWORD dwMilliseconds)
|
|
{
|
|
DWORD ret;
|
|
sigframe thisframe (mainthread);
|
|
ret = WaitForMultipleObjects (nCount, lpHandles, fWaitAll, dwMilliseconds);
|
|
return ret;
|
|
}
|
|
}
|