* configure.in: Remove NEWVFORK default.
* configure: Regenerate. * dcrt0.cc: Conditionalize vfork stuff throughout. * dtable.cc: Ditto. * perthread.h: Ditto. * pipe.cc (fhandler_pipe::close): Ditto. * spawn.cc (spawnve): Ditto. * syscalls.cc (setsid): Ditto. * exceptions.cc (sigpacket::process): Use macro to refer to vfork pid. * debug.cc (verify_handle): Define new function. * debug.h (VerifyHandle): Define new macro. (verify_handle): Declare new function * fhandler.cc (fhandler_base::dup): Verify that dup'ed handle is not supposed to be in use. (fhandler_base::set_inheritance): Ditto. (fhandler_base::fork_fixup): Ditto. * fhandler_socket.cc (fhandler_socket::dup): Ditto. * fhandler_tty.cc (fhandler_tty_slave::open): Ditto. * net.cc (set_socket_inheritance): Ditto. * pinfo.cc (pinfo_fixup_after_exec): Ditto. * sigproc.cc (proc_subproc): Ditto. (sig_send): Ditto. * spawn.cc (spawn_guts): Ditto. * thread.cc (pthread::init_mainthread): Ditto. * pipe.cc (fhandler_pipe::close): Close read_state with ForceCloseHandle since it was protected. (fhandler_pipe::fixup_after_exec): Protect read_state handle. (fhandler_pipe::dup): Correctly close open handles on error condition. Verify that dup'ed handle is not supposed to be in use. (fhandler_pipe::create): Protect read_state.
This commit is contained in:
@@ -1,3 +1,37 @@
|
|||||||
|
2004-01-23 Christopher Faylor <cgf@redhat.com>
|
||||||
|
|
||||||
|
* configure.in: Remove NEWVFORK default.
|
||||||
|
* configure: Regenerate.
|
||||||
|
* dcrt0.cc: Conditionalize vfork stuff throughout.
|
||||||
|
* dtable.cc: Ditto.
|
||||||
|
* perthread.h: Ditto.
|
||||||
|
* pipe.cc (fhandler_pipe::close): Ditto.
|
||||||
|
* spawn.cc (spawnve): Ditto.
|
||||||
|
* syscalls.cc (setsid): Ditto.
|
||||||
|
* exceptions.cc (sigpacket::process): Use macro to refer to vfork pid.
|
||||||
|
|
||||||
|
* debug.cc (verify_handle): Define new function.
|
||||||
|
* debug.h (VerifyHandle): Define new macro.
|
||||||
|
(verify_handle): Declare new function
|
||||||
|
* fhandler.cc (fhandler_base::dup): Verify that dup'ed handle is not
|
||||||
|
supposed to be in use.
|
||||||
|
(fhandler_base::set_inheritance): Ditto.
|
||||||
|
(fhandler_base::fork_fixup): Ditto.
|
||||||
|
* fhandler_socket.cc (fhandler_socket::dup): Ditto.
|
||||||
|
* fhandler_tty.cc (fhandler_tty_slave::open): Ditto.
|
||||||
|
* net.cc (set_socket_inheritance): Ditto.
|
||||||
|
* pinfo.cc (pinfo_fixup_after_exec): Ditto.
|
||||||
|
* sigproc.cc (proc_subproc): Ditto.
|
||||||
|
(sig_send): Ditto.
|
||||||
|
* spawn.cc (spawn_guts): Ditto.
|
||||||
|
* thread.cc (pthread::init_mainthread): Ditto.
|
||||||
|
* pipe.cc (fhandler_pipe::close): Close read_state with
|
||||||
|
ForceCloseHandle since it was protected.
|
||||||
|
(fhandler_pipe::fixup_after_exec): Protect read_state handle.
|
||||||
|
(fhandler_pipe::dup): Correctly close open handles on error condition.
|
||||||
|
Verify that dup'ed handle is not supposed to be in use.
|
||||||
|
(fhandler_pipe::create): Protect read_state.
|
||||||
|
|
||||||
2004-01-23 Christopher Faylor <cgf@redhat.com>
|
2004-01-23 Christopher Faylor <cgf@redhat.com>
|
||||||
|
|
||||||
* exceptions.cc (sig_handle_tty_stop): Fix boneheaded mistake by using
|
* exceptions.cc (sig_handle_tty_stop): Fix boneheaded mistake by using
|
||||||
|
2
winsup/cygwin/configure
vendored
2
winsup/cygwin/configure
vendored
@@ -1924,7 +1924,7 @@ fi
|
|||||||
|
|
||||||
case "$vfork" in
|
case "$vfork" in
|
||||||
no) ;;
|
no) ;;
|
||||||
yes|*) cat >> confdefs.h <<EOF
|
yes) cat >> confdefs.h <<EOF
|
||||||
#define NEWVFORK 1
|
#define NEWVFORK 1
|
||||||
EOF
|
EOF
|
||||||
;;
|
;;
|
||||||
|
@@ -151,7 +151,7 @@ vfork="${enableval}"
|
|||||||
|
|
||||||
case "$vfork" in
|
case "$vfork" in
|
||||||
no) ;;
|
no) ;;
|
||||||
yes|*) AC_DEFINE_UNQUOTED(NEWVFORK) ;;
|
yes) AC_DEFINE_UNQUOTED(NEWVFORK) ;;
|
||||||
esac
|
esac
|
||||||
|
|
||||||
dnl The only time we might want to transform the install names
|
dnl The only time we might want to transform the install names
|
||||||
|
@@ -178,6 +178,9 @@ init_cygheap::close_ctty ()
|
|||||||
debug_printf ("closing cygheap->ctty %p", cygheap->ctty);
|
debug_printf ("closing cygheap->ctty %p", cygheap->ctty);
|
||||||
int usecount = cygheap->ctty->usecount;
|
int usecount = cygheap->ctty->usecount;
|
||||||
cygheap->ctty->close ();
|
cygheap->ctty->close ();
|
||||||
|
#ifndef NEWVFORK
|
||||||
|
cygheap->ctty = NULL;
|
||||||
|
#else // FIXME: This code ain't right
|
||||||
if (cygheap->ctty_on_hold == cygheap->ctty)
|
if (cygheap->ctty_on_hold == cygheap->ctty)
|
||||||
cygheap->ctty_on_hold = NULL;
|
cygheap->ctty_on_hold = NULL;
|
||||||
if (usecount == 1)
|
if (usecount == 1)
|
||||||
@@ -185,6 +188,7 @@ init_cygheap::close_ctty ()
|
|||||||
cygheap->ctty = NULL;
|
cygheap->ctty = NULL;
|
||||||
debug_printf ("setting cygheap->ctty to NULL");
|
debug_printf ("setting cygheap->ctty to NULL");
|
||||||
}
|
}
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
#define pagetrunc(x) ((void *) (((DWORD) (x)) & ~(4096 - 1)))
|
#define pagetrunc(x) ((void *) (((DWORD) (x)) & ~(4096 - 1)))
|
||||||
|
@@ -263,7 +263,9 @@ struct init_cygheap
|
|||||||
struct sigaction *sigs;
|
struct sigaction *sigs;
|
||||||
|
|
||||||
fhandler_tty_slave *ctty; /* Current tty */
|
fhandler_tty_slave *ctty; /* Current tty */
|
||||||
|
#ifdef NEWVFORK
|
||||||
fhandler_tty_slave *ctty_on_hold;
|
fhandler_tty_slave *ctty_on_hold;
|
||||||
|
#endif
|
||||||
struct _threadinfo **threadlist;
|
struct _threadinfo **threadlist;
|
||||||
size_t sthreads;
|
size_t sthreads;
|
||||||
int open_fhs;
|
int open_fhs;
|
||||||
|
@@ -45,10 +45,14 @@ HANDLE NO_COPY hMainProc = (HANDLE) -1;
|
|||||||
HANDLE NO_COPY hMainThread;
|
HANDLE NO_COPY hMainThread;
|
||||||
|
|
||||||
per_thread_waitq NO_COPY waitq_storage;
|
per_thread_waitq NO_COPY waitq_storage;
|
||||||
|
#ifdef NEWVFORK
|
||||||
per_thread_vfork NO_COPY vfork_storage;
|
per_thread_vfork NO_COPY vfork_storage;
|
||||||
|
#endif
|
||||||
|
|
||||||
per_thread NO_COPY *threadstuff[] = {&waitq_storage,
|
per_thread NO_COPY *threadstuff[] = {&waitq_storage,
|
||||||
|
#ifdef NEWVFORK
|
||||||
&vfork_storage,
|
&vfork_storage,
|
||||||
|
#endif
|
||||||
NULL};
|
NULL};
|
||||||
|
|
||||||
bool display_title;
|
bool display_title;
|
||||||
@@ -59,7 +63,9 @@ codepage_type current_codepage = ansi_cp;
|
|||||||
int __argc_safe;
|
int __argc_safe;
|
||||||
int _declspec(dllexport) __argc;
|
int _declspec(dllexport) __argc;
|
||||||
char _declspec(dllexport) **__argv;
|
char _declspec(dllexport) **__argv;
|
||||||
|
#ifdef NEWVFORK
|
||||||
vfork_save NO_COPY *main_vfork;
|
vfork_save NO_COPY *main_vfork;
|
||||||
|
#endif
|
||||||
|
|
||||||
static int NO_COPY envc;
|
static int NO_COPY envc;
|
||||||
char NO_COPY **envp;
|
char NO_COPY **envp;
|
||||||
@@ -764,9 +770,10 @@ dll_crt0_1 (char *)
|
|||||||
Need to do this before any helper threads start. */
|
Need to do this before any helper threads start. */
|
||||||
debug_init ();
|
debug_init ();
|
||||||
|
|
||||||
|
#ifdef NEWVFORK
|
||||||
cygheap->fdtab.vfork_child_fixup ();
|
cygheap->fdtab.vfork_child_fixup ();
|
||||||
|
|
||||||
main_vfork = vfork_storage.create ();
|
main_vfork = vfork_storage.create ();
|
||||||
|
#endif
|
||||||
|
|
||||||
cygbench ("pre-forkee");
|
cygbench ("pre-forkee");
|
||||||
if (user_data->forkee)
|
if (user_data->forkee)
|
||||||
@@ -993,12 +1000,14 @@ do_exit (int status)
|
|||||||
{
|
{
|
||||||
syscall_printf ("do_exit (%d), exit_state %d", status, exit_state);
|
syscall_printf ("do_exit (%d), exit_state %d", status, exit_state);
|
||||||
|
|
||||||
|
#ifdef NEWVFORK
|
||||||
vfork_save *vf = vfork_storage.val ();
|
vfork_save *vf = vfork_storage.val ();
|
||||||
if (vf != NULL && vf->pid < 0)
|
if (vf != NULL && vf->pid < 0)
|
||||||
{
|
{
|
||||||
exit_state = ES_NOT_EXITING;
|
exit_state = ES_NOT_EXITING;
|
||||||
vf->restore_exit (status);
|
vf->restore_exit (status);
|
||||||
}
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
EnterCriticalSection (&exit_lock);
|
EnterCriticalSection (&exit_lock);
|
||||||
muto::set_exiting_thread ();
|
muto::set_exiting_thread ();
|
||||||
|
@@ -78,7 +78,18 @@ out:
|
|||||||
return hl;
|
return hl;
|
||||||
}
|
}
|
||||||
|
|
||||||
#ifdef DEBUGGING_AND_FDS_PROTECTED
|
void
|
||||||
|
verify_handle (const char *func, int ln, HANDLE h)
|
||||||
|
{
|
||||||
|
handle_list *hl = find_handle (h);
|
||||||
|
if (!hl)
|
||||||
|
return;
|
||||||
|
system_printf ("%s:%d - multiple attempts to add handle %p", func, ln, h);
|
||||||
|
|
||||||
|
system_printf (" previously allocated by %s:%d(%s<%p>) winpid %d",
|
||||||
|
hl->func, hl->ln, hl->name, hl->h, hl->pid);
|
||||||
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
setclexec (HANDLE oh, HANDLE nh, bool not_inheriting)
|
setclexec (HANDLE oh, HANDLE nh, bool not_inheriting)
|
||||||
{
|
{
|
||||||
@@ -90,7 +101,6 @@ setclexec (HANDLE oh, HANDLE nh, bool not_inheriting)
|
|||||||
hl->h = nh;
|
hl->h = nh;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
#endif
|
|
||||||
|
|
||||||
/* Create a new handle record */
|
/* Create a new handle record */
|
||||||
static handle_list * __stdcall
|
static handle_list * __stdcall
|
||||||
|
@@ -37,6 +37,7 @@ details. */
|
|||||||
# define debug_init() do {} while (0)
|
# define debug_init() do {} while (0)
|
||||||
# define setclexec(h, nh, b) do {} while (0)
|
# define setclexec(h, nh, b) do {} while (0)
|
||||||
# define debug_fixup_after_fork_exec() do {} while (0)
|
# define debug_fixup_after_fork_exec() do {} while (0)
|
||||||
|
# define VerifyHandle(h) do {} while (0)
|
||||||
|
|
||||||
#else
|
#else
|
||||||
|
|
||||||
@@ -59,10 +60,13 @@ details. */
|
|||||||
# define ProtectHandleINH(h) add_handle (__PRETTY_FUNCTION__, __LINE__, (h), #h, 1)
|
# define ProtectHandleINH(h) add_handle (__PRETTY_FUNCTION__, __LINE__, (h), #h, 1)
|
||||||
# define ProtectHandle1INH(h, n) add_handle (__PRETTY_FUNCTION__, __LINE__, (h), #n, 1)
|
# define ProtectHandle1INH(h, n) add_handle (__PRETTY_FUNCTION__, __LINE__, (h), #n, 1)
|
||||||
# define ProtectHandle2INH(h, n) add_handle (__PRETTY_FUNCTION__, __LINE__, (h), n, 1)
|
# define ProtectHandle2INH(h, n) add_handle (__PRETTY_FUNCTION__, __LINE__, (h), n, 1)
|
||||||
|
# define VerifyHandle(h) verify_handle (__PRETTY_FUNCTION__, __LINE__, (h))
|
||||||
|
|
||||||
void debug_init ();
|
void debug_init ();
|
||||||
void __stdcall add_handle (const char *, int, HANDLE, const char *, bool = false)
|
void __stdcall add_handle (const char *, int, HANDLE, const char *, bool = false)
|
||||||
__attribute__ ((regparm (3)));
|
__attribute__ ((regparm (3)));
|
||||||
|
void __stdcall verify_handle (const char *, int, HANDLE)
|
||||||
|
__attribute__ ((regparm (3)));
|
||||||
bool __stdcall close_handle (const char *, int, HANDLE, const char *, bool)
|
bool __stdcall close_handle (const char *, int, HANDLE, const char *, bool)
|
||||||
__attribute__ ((regparm (3)));
|
__attribute__ ((regparm (3)));
|
||||||
void __stdcall cygbench (const char *s) __attribute__ ((regparm (1)));
|
void __stdcall cygbench (const char *s) __attribute__ ((regparm (1)));
|
||||||
|
@@ -705,6 +705,7 @@ dtable::fixup_after_fork (HANDLE parent)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#ifdef NEWVFORK
|
||||||
int
|
int
|
||||||
dtable::vfork_child_dup ()
|
dtable::vfork_child_dup ()
|
||||||
{
|
{
|
||||||
@@ -803,6 +804,7 @@ dtable::vfork_child_fixup ()
|
|||||||
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
#endif /*NEWVFORK*/
|
||||||
|
|
||||||
#define DEVICE_PREFIX "\\device\\"
|
#define DEVICE_PREFIX "\\device\\"
|
||||||
#define DEVICE_PREFIX_LEN sizeof (DEVICE_PREFIX) - 1
|
#define DEVICE_PREFIX_LEN sizeof (DEVICE_PREFIX) - 1
|
||||||
|
@@ -22,7 +22,9 @@ class dtable
|
|||||||
{
|
{
|
||||||
muto *lock_cs;
|
muto *lock_cs;
|
||||||
fhandler_base **fds;
|
fhandler_base **fds;
|
||||||
|
#ifdef NEWVFORK
|
||||||
fhandler_base **fds_on_hold;
|
fhandler_base **fds_on_hold;
|
||||||
|
#endif
|
||||||
fhandler_base **archetypes;
|
fhandler_base **archetypes;
|
||||||
unsigned narchetypes;
|
unsigned narchetypes;
|
||||||
unsigned farchetype;
|
unsigned farchetype;
|
||||||
@@ -74,7 +76,9 @@ public:
|
|||||||
void stdio_init ();
|
void stdio_init ();
|
||||||
void get_debugger_info ();
|
void get_debugger_info ();
|
||||||
void set_file_pointers_for_exec ();
|
void set_file_pointers_for_exec ();
|
||||||
|
#ifdef NEWVFORK
|
||||||
bool in_vfork_cleanup () {return fds_on_hold == fds;}
|
bool in_vfork_cleanup () {return fds_on_hold == fds;}
|
||||||
|
#endif
|
||||||
fhandler_fifo *find_fifo (ATOM);
|
fhandler_fifo *find_fifo (ATOM);
|
||||||
fhandler_base *find_archetype (device& dev);
|
fhandler_base *find_archetype (device& dev);
|
||||||
fhandler_base **add_archetype ();
|
fhandler_base **add_archetype ();
|
||||||
|
@@ -957,7 +957,7 @@ sigpacket::process ()
|
|||||||
bool special_case;
|
bool special_case;
|
||||||
bool insigwait_mask;
|
bool insigwait_mask;
|
||||||
insigwait_mask = masked = false;
|
insigwait_mask = masked = false;
|
||||||
if (special_case = (main_vfork->pid || ISSTATE (myself, PID_STOPPED)))
|
if (special_case = (VFORKPID || ISSTATE (myself, PID_STOPPED)))
|
||||||
/* nothing to do */;
|
/* nothing to do */;
|
||||||
else if (tls && sigismember (&tls->sigwait_mask, si.si_signo))
|
else if (tls && sigismember (&tls->sigwait_mask, si.si_signo))
|
||||||
insigwait_mask = true;
|
insigwait_mask = true;
|
||||||
|
@@ -1044,6 +1044,7 @@ fhandler_base::dup (fhandler_base *child)
|
|||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
VerifyHandle (nh);
|
||||||
child->set_io_handle (nh);
|
child->set_io_handle (nh);
|
||||||
}
|
}
|
||||||
return 0;
|
return 0;
|
||||||
@@ -1209,17 +1210,17 @@ fhandler_dev_null::dump (void)
|
|||||||
void
|
void
|
||||||
fhandler_base::set_inheritance (HANDLE &h, int not_inheriting)
|
fhandler_base::set_inheritance (HANDLE &h, int not_inheriting)
|
||||||
{
|
{
|
||||||
#ifdef DEBUGGING_AND_FDS_PROTECTED
|
|
||||||
HANDLE oh = h;
|
HANDLE oh = h;
|
||||||
#endif
|
|
||||||
/* Note that we could use SetHandleInformation here but it is not available
|
/* Note that we could use SetHandleInformation here but it is not available
|
||||||
on all platforms. Test cases seem to indicate that using DuplicateHandle
|
on all platforms. Test cases seem to indicate that using DuplicateHandle
|
||||||
in this fashion does not actually close the original handle, which is
|
in this fashion does not actually close the original handle, which is
|
||||||
what we want. If this changes in the future, we may be forced to use
|
what we want. If this changes in the future, we may be forced to use
|
||||||
SetHandleInformation on newer OS's */
|
SetHandleInformation on newer OS's */
|
||||||
if (!DuplicateHandle (hMainProc, h, hMainProc, &h, 0, !not_inheriting,
|
if (!DuplicateHandle (hMainProc, oh, hMainProc, &h, 0, !not_inheriting,
|
||||||
DUPLICATE_SAME_ACCESS | DUPLICATE_CLOSE_SOURCE))
|
DUPLICATE_SAME_ACCESS | DUPLICATE_CLOSE_SOURCE))
|
||||||
debug_printf ("DuplicateHandle failed, %E");
|
debug_printf ("DuplicateHandle failed, %E");
|
||||||
|
if (oh != h)
|
||||||
|
VerifyHandle (h);
|
||||||
#ifdef DEBUGGING_AND_FDS_PROTECTED
|
#ifdef DEBUGGING_AND_FDS_PROTECTED
|
||||||
if (h)
|
if (h)
|
||||||
setclexec (oh, h, not_inheriting);
|
setclexec (oh, h, not_inheriting);
|
||||||
@@ -1229,17 +1230,14 @@ fhandler_base::set_inheritance (HANDLE &h, int not_inheriting)
|
|||||||
void
|
void
|
||||||
fhandler_base::fork_fixup (HANDLE parent, HANDLE &h, const char *name)
|
fhandler_base::fork_fixup (HANDLE parent, HANDLE &h, const char *name)
|
||||||
{
|
{
|
||||||
|
HANDLE oh = h;
|
||||||
if (/* !is_socket () && */ !get_close_on_exec ())
|
if (/* !is_socket () && */ !get_close_on_exec ())
|
||||||
debug_printf ("handle %p already opened", h);
|
debug_printf ("handle %p already opened", h);
|
||||||
else if (!DuplicateHandle (parent, h, hMainProc, &h, 0, !get_close_on_exec (),
|
else if (!DuplicateHandle (parent, h, hMainProc, &h, 0, !get_close_on_exec (),
|
||||||
DUPLICATE_SAME_ACCESS))
|
DUPLICATE_SAME_ACCESS))
|
||||||
system_printf ("%s - %E, handle %s<%p>", get_name (), name, h);
|
system_printf ("%s - %E, handle %s<%p>", get_name (), name, h);
|
||||||
#ifdef DEBUGGING_AND_FDS_PROTECTED
|
else if (oh != h)
|
||||||
else if (get_close_on_exec ())
|
VerifyHandle (h);
|
||||||
ProtectHandle (h); /* would have to be fancier than this */
|
|
||||||
else
|
|
||||||
/* ProtectHandleINH (h) */; /* Should already be protected */
|
|
||||||
#endif
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
|
@@ -363,6 +363,7 @@ fhandler_socket::dup (fhandler_base *child)
|
|||||||
__seterrno ();
|
__seterrno ();
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
VerifyHandle (nh);
|
||||||
fhs->set_io_handle (nh);
|
fhs->set_io_handle (nh);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
@@ -553,6 +553,7 @@ fhandler_tty_slave::open (int flags, mode_t)
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
VerifyHandle (from_master_local);
|
||||||
if (!DuplicateHandle (tty_owner, get_ttyp ()->to_master,
|
if (!DuplicateHandle (tty_owner, get_ttyp ()->to_master,
|
||||||
hMainProc, &to_master_local, 0, TRUE,
|
hMainProc, &to_master_local, 0, TRUE,
|
||||||
DUPLICATE_SAME_ACCESS))
|
DUPLICATE_SAME_ACCESS))
|
||||||
@@ -561,6 +562,7 @@ fhandler_tty_slave::open (int flags, mode_t)
|
|||||||
__seterrno ();
|
__seterrno ();
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
VerifyHandle (to_master_local);
|
||||||
CloseHandle (tty_owner);
|
CloseHandle (tty_owner);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@@ -130,6 +130,7 @@ set_socket_inheritance (SOCKET sock)
|
|||||||
system_printf ("DuplicateHandle failed %E");
|
system_printf ("DuplicateHandle failed %E");
|
||||||
else
|
else
|
||||||
debug_printf ("DuplicateHandle succeeded osock %p, sock %p", osock, sock);
|
debug_printf ("DuplicateHandle succeeded osock %p, sock %p", osock, sock);
|
||||||
|
VerifyHandle ((HANDLE) sock);
|
||||||
return sock;
|
return sock;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@@ -43,8 +43,14 @@ public:
|
|||||||
size_t size () {return sizeof (waitq);}
|
size_t size () {return sizeof (waitq);}
|
||||||
};
|
};
|
||||||
|
|
||||||
#if defined (NEED_VFORK)
|
#ifdef NEED_VFORK
|
||||||
#include "cygtls.h"
|
#include "cygtls.h"
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifndef NEWVFORK
|
||||||
|
#define VFORKPID 0
|
||||||
|
#else
|
||||||
|
#if defined (NEED_VFORK)
|
||||||
class vfork_save
|
class vfork_save
|
||||||
{
|
{
|
||||||
jmp_buf j;
|
jmp_buf j;
|
||||||
@@ -82,7 +88,9 @@ public:
|
|||||||
};
|
};
|
||||||
extern per_thread_vfork vfork_storage;
|
extern per_thread_vfork vfork_storage;
|
||||||
extern vfork_save *main_vfork;
|
extern vfork_save *main_vfork;
|
||||||
|
#define VFORKPID main_vfork->pid
|
||||||
#endif
|
#endif
|
||||||
|
#endif /*NEWVFORK*/
|
||||||
|
|
||||||
extern per_thread_waitq waitq_storage;
|
extern per_thread_waitq waitq_storage;
|
||||||
|
|
||||||
|
@@ -49,6 +49,7 @@ pinfo_fixup_after_fork ()
|
|||||||
system_printf ("couldn't save current process handle %p, %E", hMainProc);
|
system_printf ("couldn't save current process handle %p, %E", hMainProc);
|
||||||
hexec_proc = NULL;
|
hexec_proc = NULL;
|
||||||
}
|
}
|
||||||
|
VerifyHandle (hexec_proc);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Initialize the process table.
|
/* Initialize the process table.
|
||||||
|
@@ -89,11 +89,15 @@ fhandler_pipe::close ()
|
|||||||
CloseHandle (guard);
|
CloseHandle (guard);
|
||||||
if (writepipe_exists)
|
if (writepipe_exists)
|
||||||
CloseHandle (writepipe_exists);
|
CloseHandle (writepipe_exists);
|
||||||
|
#ifndef NEWVFORK
|
||||||
|
if (read_state)
|
||||||
|
#else
|
||||||
// FIXME is this vfork_cleanup test right? Is it responsible for some of
|
// FIXME is this vfork_cleanup test right? Is it responsible for some of
|
||||||
// the strange pipe behavior that has been reported in the cygwin mailing
|
// the strange pipe behavior that has been reported in the cygwin mailing
|
||||||
// list?
|
// list?
|
||||||
if (read_state && !cygheap->fdtab.in_vfork_cleanup ())
|
if (read_state && !cygheap->fdtab.in_vfork_cleanup ())
|
||||||
CloseHandle (read_state);
|
#endif
|
||||||
|
ForceCloseHandle (read_state);
|
||||||
if (get_handle ())
|
if (get_handle ())
|
||||||
{
|
{
|
||||||
CloseHandle (get_handle ());
|
CloseHandle (get_handle ());
|
||||||
@@ -122,7 +126,10 @@ void
|
|||||||
fhandler_pipe::fixup_after_exec (HANDLE parent)
|
fhandler_pipe::fixup_after_exec (HANDLE parent)
|
||||||
{
|
{
|
||||||
if (read_state)
|
if (read_state)
|
||||||
|
{
|
||||||
read_state = CreateEvent (&sec_none_nih, FALSE, FALSE, NULL);
|
read_state = CreateEvent (&sec_none_nih, FALSE, FALSE, NULL);
|
||||||
|
ProtectHandle (read_state);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
@@ -139,15 +146,17 @@ fhandler_pipe::fixup_after_fork (HANDLE parent)
|
|||||||
int
|
int
|
||||||
fhandler_pipe::dup (fhandler_base *child)
|
fhandler_pipe::dup (fhandler_base *child)
|
||||||
{
|
{
|
||||||
|
int res = -1;
|
||||||
|
fhandler_pipe *ftp = (fhandler_pipe *) child;
|
||||||
|
ftp->guard = ftp->writepipe_exists = ftp->read_state = NULL;
|
||||||
|
|
||||||
if (get_handle ())
|
if (get_handle ())
|
||||||
{
|
{
|
||||||
int res = fhandler_base::dup (child);
|
res = fhandler_base::dup (child);
|
||||||
if (res)
|
if (res)
|
||||||
return res;
|
goto err;
|
||||||
}
|
}
|
||||||
|
|
||||||
fhandler_pipe *ftp = (fhandler_pipe *) child;
|
|
||||||
|
|
||||||
/* FIXME: This leaks handles in the failing condition */
|
/* FIXME: This leaks handles in the failing condition */
|
||||||
if (guard == NULL)
|
if (guard == NULL)
|
||||||
ftp->guard = NULL;
|
ftp->guard = NULL;
|
||||||
@@ -155,7 +164,7 @@ fhandler_pipe::dup (fhandler_base *child)
|
|||||||
DUPLICATE_SAME_ACCESS))
|
DUPLICATE_SAME_ACCESS))
|
||||||
{
|
{
|
||||||
debug_printf ("couldn't duplicate guard %p, %E", guard);
|
debug_printf ("couldn't duplicate guard %p, %E", guard);
|
||||||
return -1;
|
goto err;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (writepipe_exists == NULL)
|
if (writepipe_exists == NULL)
|
||||||
@@ -165,7 +174,7 @@ fhandler_pipe::dup (fhandler_base *child)
|
|||||||
DUPLICATE_SAME_ACCESS))
|
DUPLICATE_SAME_ACCESS))
|
||||||
{
|
{
|
||||||
debug_printf ("couldn't duplicate writepipe_exists %p, %E", writepipe_exists);
|
debug_printf ("couldn't duplicate writepipe_exists %p, %E", writepipe_exists);
|
||||||
return -1;
|
goto err;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (read_state == NULL)
|
if (read_state == NULL)
|
||||||
@@ -175,12 +184,31 @@ fhandler_pipe::dup (fhandler_base *child)
|
|||||||
DUPLICATE_SAME_ACCESS))
|
DUPLICATE_SAME_ACCESS))
|
||||||
{
|
{
|
||||||
debug_printf ("couldn't duplicate read_state %p, %E", writepipe_exists);
|
debug_printf ("couldn't duplicate read_state %p, %E", writepipe_exists);
|
||||||
return -1;
|
goto err;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
res = 0;
|
||||||
|
goto out;
|
||||||
|
|
||||||
|
err:
|
||||||
|
if (!ftp->guard)
|
||||||
|
CloseHandle (ftp->guard);
|
||||||
|
if (!ftp->writepipe_exists)
|
||||||
|
CloseHandle (ftp->writepipe_exists);
|
||||||
|
if (!ftp->read_state)
|
||||||
|
CloseHandle (ftp->read_state);
|
||||||
|
goto leave;
|
||||||
|
|
||||||
|
out:
|
||||||
ftp->id = id;
|
ftp->id = id;
|
||||||
ftp->orig_pid = orig_pid;
|
ftp->orig_pid = orig_pid;
|
||||||
return 0;
|
VerifyHandle (ftp->guard);
|
||||||
|
VerifyHandle (ftp->writepipe_exists);
|
||||||
|
VerifyHandle (ftp->read_state);
|
||||||
|
|
||||||
|
leave:
|
||||||
|
debug_printf ("res %d", res);
|
||||||
|
return res;
|
||||||
}
|
}
|
||||||
|
|
||||||
int
|
int
|
||||||
@@ -208,6 +236,7 @@ fhandler_pipe::create (fhandler_pipe *fhs[2], unsigned psize, int mode, bool fif
|
|||||||
|
|
||||||
fhs[0]->read_state = CreateEvent (&sec_none_nih, FALSE, FALSE, NULL);
|
fhs[0]->read_state = CreateEvent (&sec_none_nih, FALSE, FALSE, NULL);
|
||||||
fhs[0]->set_need_fork_fixup ();
|
fhs[0]->set_need_fork_fixup ();
|
||||||
|
ProtectHandle1 (fhs[0]->read_state, read_state);
|
||||||
|
|
||||||
res = 0;
|
res = 0;
|
||||||
fhs[0]->create_guard (sa);
|
fhs[0]->create_guard (sa);
|
||||||
|
@@ -320,6 +320,7 @@ proc_subproc (DWORD what, DWORD val)
|
|||||||
if (!DuplicateHandle (hMainProc, hMainProc, vchild->hProcess, &vchild->ppid_handle,
|
if (!DuplicateHandle (hMainProc, hMainProc, vchild->hProcess, &vchild->ppid_handle,
|
||||||
SYNCHRONIZE | PROCESS_DUP_HANDLE, TRUE, 0))
|
SYNCHRONIZE | PROCESS_DUP_HANDLE, TRUE, 0))
|
||||||
system_printf ("Couldn't duplicate my handle<%p> for pid %d, %E", hMainProc, vchild->pid);
|
system_printf ("Couldn't duplicate my handle<%p> for pid %d, %E", hMainProc, vchild->pid);
|
||||||
|
VerifyHandle (vchild->ppid_handle);
|
||||||
vchild->ppid = myself->pid;
|
vchild->ppid = myself->pid;
|
||||||
vchild->uid = myself->uid;
|
vchild->uid = myself->uid;
|
||||||
vchild->gid = myself->gid;
|
vchild->gid = myself->gid;
|
||||||
@@ -716,17 +717,19 @@ sig_send (_pinfo *p, siginfo_t& si, _threadinfo *tls)
|
|||||||
__seterrno ();
|
__seterrno ();
|
||||||
goto out;
|
goto out;
|
||||||
}
|
}
|
||||||
|
VerifyHandle (hp);
|
||||||
for (int i = 0; !p->sendsig && i < 10000; i++)
|
for (int i = 0; !p->sendsig && i < 10000; i++)
|
||||||
low_priority_sleep (0);
|
low_priority_sleep (0);
|
||||||
if (!DuplicateHandle (hp, p->sendsig, hMainProc, &sendsig, false, 0,
|
if (!DuplicateHandle (hp, p->sendsig, hMainProc, &sendsig, false, 0,
|
||||||
DUPLICATE_SAME_ACCESS) || !sendsig)
|
DUPLICATE_SAME_ACCESS) || !sendsig)
|
||||||
{
|
{
|
||||||
|
CloseHandle (hp);
|
||||||
sigproc_printf ("DuplicateHandle failed, %E");
|
sigproc_printf ("DuplicateHandle failed, %E");
|
||||||
__seterrno ();
|
__seterrno ();
|
||||||
goto out;
|
goto out;
|
||||||
}
|
}
|
||||||
CloseHandle (hp);
|
CloseHandle (hp);
|
||||||
pack.wakeup = NULL;
|
VerifyHandle (sendsig);
|
||||||
}
|
}
|
||||||
|
|
||||||
sigproc_printf ("sendsig %p, pid %d, signal %d, its_me %d", sendsig, p->pid, si.si_signo, its_me);
|
sigproc_printf ("sendsig %p, pid %d, signal %d, its_me %d", sendsig, p->pid, si.si_signo, its_me);
|
||||||
|
@@ -404,6 +404,7 @@ spawn_guts (const char * prog_arg, const char *const *argv,
|
|||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
VerifyHandle (ciresrv.parent);
|
||||||
ciresrv.moreinfo = (cygheap_exec_info *) ccalloc (HEAP_1_EXEC, 1, sizeof (cygheap_exec_info));
|
ciresrv.moreinfo = (cygheap_exec_info *) ccalloc (HEAP_1_EXEC, 1, sizeof (cygheap_exec_info));
|
||||||
ciresrv.moreinfo->old_title = NULL;
|
ciresrv.moreinfo->old_title = NULL;
|
||||||
|
|
||||||
@@ -612,6 +613,8 @@ spawn_guts (const char * prog_arg, const char *const *argv,
|
|||||||
&ciresrv.moreinfo->myself_pinfo, 0,
|
&ciresrv.moreinfo->myself_pinfo, 0,
|
||||||
TRUE, DUPLICATE_SAME_ACCESS))
|
TRUE, DUPLICATE_SAME_ACCESS))
|
||||||
ciresrv.moreinfo->myself_pinfo = NULL;
|
ciresrv.moreinfo->myself_pinfo = NULL;
|
||||||
|
else
|
||||||
|
VerifyHandle (ciresrv.moreinfo->myself_pinfo);
|
||||||
|
|
||||||
skip_arg_parsing:
|
skip_arg_parsing:
|
||||||
PROCESS_INFORMATION pi = {NULL, 0, 0, 0};
|
PROCESS_INFORMATION pi = {NULL, 0, 0, 0};
|
||||||
@@ -901,6 +904,7 @@ spawn_guts (const char * prog_arg, const char *const *argv,
|
|||||||
0, FALSE, DUPLICATE_SAME_ACCESS);
|
0, FALSE, DUPLICATE_SAME_ACCESS);
|
||||||
sigproc_printf ("%d = DuplicateHandle, oldh %p, newh %p",
|
sigproc_printf ("%d = DuplicateHandle, oldh %p, newh %p",
|
||||||
rc, oldh, myself->hProcess);
|
rc, oldh, myself->hProcess);
|
||||||
|
VerifyHandle (myself->hProcess);
|
||||||
if (!rc && my_parent_is_alive ())
|
if (!rc && my_parent_is_alive ())
|
||||||
{
|
{
|
||||||
system_printf ("Reparent failed, parent handle %p, %E", h);
|
system_printf ("Reparent failed, parent handle %p, %E", h);
|
||||||
@@ -958,12 +962,14 @@ spawnve (int mode, const char *path, const char *const *argv,
|
|||||||
const char *const *envp)
|
const char *const *envp)
|
||||||
{
|
{
|
||||||
int ret;
|
int ret;
|
||||||
|
#ifdef NEWVFORK
|
||||||
vfork_save *vf = vfork_storage.val ();
|
vfork_save *vf = vfork_storage.val ();
|
||||||
|
|
||||||
if (vf != NULL && (vf->pid < 0) && mode == _P_OVERLAY)
|
if (vf != NULL && (vf->pid < 0) && mode == _P_OVERLAY)
|
||||||
mode = _P_NOWAIT;
|
mode = _P_NOWAIT;
|
||||||
else
|
else
|
||||||
vf = NULL;
|
vf = NULL;
|
||||||
|
#endif
|
||||||
|
|
||||||
syscall_printf ("spawnve (%s, %s, %x)", path, argv[0], envp);
|
syscall_printf ("spawnve (%s, %s, %x)", path, argv[0], envp);
|
||||||
|
|
||||||
@@ -984,6 +990,7 @@ spawnve (int mode, const char *path, const char *const *argv,
|
|||||||
case _P_SYSTEM:
|
case _P_SYSTEM:
|
||||||
subproc_init ();
|
subproc_init ();
|
||||||
ret = spawn_guts (path, argv, envp, mode);
|
ret = spawn_guts (path, argv, envp, mode);
|
||||||
|
#ifdef NEWVFORK
|
||||||
if (vf)
|
if (vf)
|
||||||
{
|
{
|
||||||
if (ret > 0)
|
if (ret > 0)
|
||||||
@@ -992,6 +999,7 @@ spawnve (int mode, const char *path, const char *const *argv,
|
|||||||
vf->restore_pid (ret);
|
vf->restore_pid (ret);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
#endif
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
set_errno (EINVAL);
|
set_errno (EINVAL);
|
||||||
|
@@ -300,6 +300,7 @@ getppid ()
|
|||||||
extern "C" pid_t
|
extern "C" pid_t
|
||||||
setsid (void)
|
setsid (void)
|
||||||
{
|
{
|
||||||
|
#ifdef NEWVFORK
|
||||||
vfork_save *vf = vfork_storage.val ();
|
vfork_save *vf = vfork_storage.val ();
|
||||||
/* This is a horrible, horrible kludge */
|
/* This is a horrible, horrible kludge */
|
||||||
if (vf && vf->pid < 0)
|
if (vf && vf->pid < 0)
|
||||||
@@ -312,6 +313,7 @@ setsid (void)
|
|||||||
}
|
}
|
||||||
/* assuming that fork was successful */
|
/* assuming that fork was successful */
|
||||||
}
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
if (myself->pgid == myself->pid)
|
if (myself->pgid == myself->pid)
|
||||||
syscall_printf ("hmm. pgid %d pid %d", myself->pgid, myself->pid);
|
syscall_printf ("hmm. pgid %d pid %d", myself->pgid, myself->pid);
|
||||||
|
@@ -144,6 +144,7 @@ pthread::init_mainthread ()
|
|||||||
api_fatal ("failed to create mainthread handle");
|
api_fatal ("failed to create mainthread handle");
|
||||||
if (!thread->create_cancel_event ())
|
if (!thread->create_cancel_event ())
|
||||||
api_fatal ("couldn't create cancel event for main thread");
|
api_fatal ("couldn't create cancel event for main thread");
|
||||||
|
VerifyHandle (thread->win32_obj_id);
|
||||||
thread->postcreate ();
|
thread->postcreate ();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Reference in New Issue
Block a user