...completion in child process because the cygheap should not be
changed to avoid mismatch between child_info::cygheap_max and
::cygheap_max. Otherwise, child_copy() might copy cygheap being
modified by other process. However, do not lock cygheap if the
child process is non-cygwin process, because child_copy() will
not be called in it. Not only it is unnecessary, it can also fall
into deadlock in close_all_files() while cygheap is already locked.
Fixes: 977ad5434cc0 ("* spawn.cc (spawn_guts): Call refresh_cygheap before
creating a new process to ensure that cygheap_max is up-to-date.")
Reviewed-by: Corinna Vinschen <[email protected]>
Signed-off-by: Takashi Yano <[email protected]>
---
winsup/cygwin/spawn.cc | 12 +++++++++++-
1 file changed, 11 insertions(+), 1 deletion(-)
diff --git a/winsup/cygwin/spawn.cc b/winsup/cygwin/spawn.cc
index cb58b6eed..cf344d382 100644
--- a/winsup/cygwin/spawn.cc
+++ b/winsup/cygwin/spawn.cc
@@ -542,7 +542,6 @@ child_info_spawn::worker (const char *prog_arg, const char
*const *argv,
::cygheap->ctty ? ::cygheap->ctty->tc_getpgid () : 0;
if (!iscygwin () && ctty_pgid && ctty_pgid != myself->pgid)
c_flags |= CREATE_NEW_PROCESS_GROUP;
- refresh_cygheap ();
if (mode == _P_DETACH)
/* all set */;
@@ -611,6 +610,9 @@ child_info_spawn::worker (const char *prog_arg, const char
*const *argv,
cygpid = (mode != _P_OVERLAY) ? create_cygwin_pid () : myself->pid;
+ if (iscygwin ())
+ cygheap->lock ();
+ refresh_cygheap ();
wchar_t wcmd[(size_t) cmd];
if (!::cygheap->user.issetuid ()
|| (::cygheap->user.saved_uid == ::cygheap->user.real_uid
@@ -728,6 +730,8 @@ child_info_spawn::worker (const char *prog_arg, const char
*const *argv,
::cygheap->user.reimpersonate ();
res = -1;
+ if (iscygwin ())
+ cygheap->unlock ();
__leave;
}
@@ -781,6 +785,8 @@ child_info_spawn::worker (const char *prog_arg, const char
*const *argv,
if (get_errno () != ENOMEM)
set_errno (EAGAIN);
res = -1;
+ if (iscygwin ())
+ cygheap->unlock ();
__leave;
}
child->dwProcessId = pi.dwProcessId;
@@ -816,6 +822,8 @@ child_info_spawn::worker (const char *prog_arg, const char
*const *argv,
CloseHandle (pi.hProcess);
ForceCloseHandle (pi.hThread);
res = -1;
+ if (iscygwin ())
+ cygheap->unlock ();
__leave;
}
}
@@ -844,6 +852,8 @@ child_info_spawn::worker (const char *prog_arg, const char
*const *argv,
/* Just mark a non-cygwin process as 'synced'. We will still eventually
wait for it to exit in maybe_set_exit_code_from_windows(). */
synced = iscygwin () ? sync (pi.dwProcessId, pi.hProcess, INFINITE) :
true;
+ if (iscygwin ())
+ cygheap->unlock ();
switch (mode)
{
--
2.45.1