mirror of
https://gitlab.gnome.org/GNOME/glib.git
synced 2025-06-15 09:03:49 +02:00
Merge branch 'th/child-watch-waitpid' into 'main'
[th/child-watch-waitpid] fix race with waitpid() and child watcher sources See merge request GNOME/glib!3353
This commit is contained in:
commit
eaaa2b5f66
@ -4003,8 +4003,6 @@ socket_source_prepare (GSource *source,
|
|||||||
{
|
{
|
||||||
GSocketSource *socket_source = (GSocketSource *)source;
|
GSocketSource *socket_source = (GSocketSource *)source;
|
||||||
|
|
||||||
*timeout = -1;
|
|
||||||
|
|
||||||
#ifdef G_OS_WIN32
|
#ifdef G_OS_WIN32
|
||||||
if ((socket_source->pollfd.revents & G_IO_NVAL) != 0)
|
if ((socket_source->pollfd.revents & G_IO_NVAL) != 0)
|
||||||
return TRUE;
|
return TRUE;
|
||||||
|
@ -135,8 +135,6 @@ g_io_unix_prepare (GSource *source,
|
|||||||
GIOUnixWatch *watch = (GIOUnixWatch *)source;
|
GIOUnixWatch *watch = (GIOUnixWatch *)source;
|
||||||
GIOCondition buffer_condition = g_io_channel_get_buffer_condition (watch->channel);
|
GIOCondition buffer_condition = g_io_channel_get_buffer_condition (watch->channel);
|
||||||
|
|
||||||
*timeout = -1;
|
|
||||||
|
|
||||||
/* Only return TRUE here if _all_ bits in watch->condition will be set
|
/* Only return TRUE here if _all_ bits in watch->condition will be set
|
||||||
*/
|
*/
|
||||||
return ((watch->condition & buffer_condition) == watch->condition);
|
return ((watch->condition & buffer_condition) == watch->condition);
|
||||||
|
@ -709,8 +709,6 @@ g_io_win32_prepare (GSource *source,
|
|||||||
GIOWin32Channel *channel = (GIOWin32Channel *)watch->channel;
|
GIOWin32Channel *channel = (GIOWin32Channel *)watch->channel;
|
||||||
int event_mask;
|
int event_mask;
|
||||||
|
|
||||||
*timeout = -1;
|
|
||||||
|
|
||||||
if (channel->debug)
|
if (channel->debug)
|
||||||
g_print ("g_io_win32_prepare: source=%p channel=%p", source, channel);
|
g_print ("g_io_win32_prepare: source=%p channel=%p", source, channel);
|
||||||
|
|
||||||
|
339
glib/gmain.c
339
glib/gmain.c
@ -374,14 +374,11 @@ struct _GChildWatchSource
|
|||||||
{
|
{
|
||||||
GSource source;
|
GSource source;
|
||||||
GPid pid;
|
GPid pid;
|
||||||
/* On Unix this is a wait status, which is the thing you pass to WEXITSTATUS()
|
/* @poll is always used on Windows.
|
||||||
* to get the status returned from the process’ main() or passed to exit(): */
|
* On Unix, poll.fd will be negative if PIDFD is unavailable. */
|
||||||
gint child_status;
|
|
||||||
/* @poll is always used on Windows, and used on Unix iff @using_pidfd is set: */
|
|
||||||
GPollFD poll;
|
GPollFD poll;
|
||||||
#ifndef G_OS_WIN32
|
#ifndef G_OS_WIN32
|
||||||
gboolean child_exited; /* (atomic); not used iff @using_pidfd is set */
|
gboolean child_maybe_exited; /* (atomic) */
|
||||||
gboolean using_pidfd;
|
|
||||||
#endif /* G_OS_WIN32 */
|
#endif /* G_OS_WIN32 */
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -488,6 +485,11 @@ static gboolean g_child_watch_dispatch (GSource *source,
|
|||||||
GSourceFunc callback,
|
GSourceFunc callback,
|
||||||
gpointer user_data);
|
gpointer user_data);
|
||||||
static void g_child_watch_finalize (GSource *source);
|
static void g_child_watch_finalize (GSource *source);
|
||||||
|
|
||||||
|
#ifndef G_OS_WIN32
|
||||||
|
static void unref_unix_signal_handler_unlocked (int signum);
|
||||||
|
#endif
|
||||||
|
|
||||||
#ifdef G_OS_UNIX
|
#ifdef G_OS_UNIX
|
||||||
static void g_unix_signal_handler (int signum);
|
static void g_unix_signal_handler (int signum);
|
||||||
static gboolean g_unix_signal_watch_prepare (GSource *source,
|
static gboolean g_unix_signal_watch_prepare (GSource *source,
|
||||||
@ -3850,10 +3852,7 @@ g_main_context_prepare_unlocked (GMainContext *context,
|
|||||||
context->in_check_or_prepare--;
|
context->in_check_or_prepare--;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
|
||||||
source_timeout = -1;
|
|
||||||
result = FALSE;
|
result = FALSE;
|
||||||
}
|
|
||||||
|
|
||||||
if (result == FALSE && source->priv->ready_time != -1)
|
if (result == FALSE && source->priv->ready_time != -1)
|
||||||
{
|
{
|
||||||
@ -5479,14 +5478,48 @@ g_timeout_add_seconds_once (guint interval,
|
|||||||
|
|
||||||
/* Child watch functions */
|
/* Child watch functions */
|
||||||
|
|
||||||
#ifdef G_OS_WIN32
|
#ifdef HAVE_PIDFD
|
||||||
|
static int
|
||||||
|
siginfo_t_to_wait_status (const siginfo_t *info)
|
||||||
|
{
|
||||||
|
/* Each of these returns is essentially the inverse of WIFEXITED(),
|
||||||
|
* WIFSIGNALED(), etc. */
|
||||||
|
switch (info->si_code)
|
||||||
|
{
|
||||||
|
case CLD_EXITED:
|
||||||
|
return W_EXITCODE (info->si_status, 0);
|
||||||
|
case CLD_KILLED:
|
||||||
|
return W_EXITCODE (0, info->si_status);
|
||||||
|
case CLD_DUMPED:
|
||||||
|
return W_EXITCODE (0, info->si_status | WCOREFLAG);
|
||||||
|
case CLD_CONTINUED:
|
||||||
|
return __W_CONTINUED;
|
||||||
|
case CLD_STOPPED:
|
||||||
|
case CLD_TRAPPED:
|
||||||
|
default:
|
||||||
|
return W_STOPCODE (info->si_status);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
#endif /* HAVE_PIDFD */
|
||||||
|
|
||||||
static gboolean
|
static gboolean
|
||||||
g_child_watch_prepare (GSource *source,
|
g_child_watch_prepare (GSource *source,
|
||||||
gint *timeout)
|
gint *timeout)
|
||||||
{
|
{
|
||||||
*timeout = -1;
|
#ifdef G_OS_WIN32
|
||||||
return FALSE;
|
return FALSE;
|
||||||
|
#else /* G_OS_WIN32 */
|
||||||
|
{
|
||||||
|
GChildWatchSource *child_watch_source;
|
||||||
|
|
||||||
|
child_watch_source = (GChildWatchSource *) source;
|
||||||
|
|
||||||
|
if (child_watch_source->poll.fd >= 0)
|
||||||
|
return FALSE;
|
||||||
|
|
||||||
|
return g_atomic_int_get (&child_watch_source->child_maybe_exited);
|
||||||
|
}
|
||||||
|
#endif /* G_OS_WIN32 */
|
||||||
}
|
}
|
||||||
|
|
||||||
static gboolean
|
static gboolean
|
||||||
@ -5497,29 +5530,18 @@ g_child_watch_check (GSource *source)
|
|||||||
|
|
||||||
child_watch_source = (GChildWatchSource *) source;
|
child_watch_source = (GChildWatchSource *) source;
|
||||||
|
|
||||||
child_exited = child_watch_source->poll.revents & G_IO_IN;
|
#ifdef G_OS_WIN32
|
||||||
|
child_exited = !!(child_watch_source->poll.revents & G_IO_IN);
|
||||||
if (child_exited)
|
#else /* G_OS_WIN32 */
|
||||||
|
#ifdef HAVE_PIDFD
|
||||||
|
if (child_watch_source->poll.fd >= 0)
|
||||||
{
|
{
|
||||||
DWORD child_status;
|
child_exited = !!(child_watch_source->poll.revents & G_IO_IN);
|
||||||
|
return child_exited;
|
||||||
/*
|
|
||||||
* Note: We do _not_ check for the special value of STILL_ACTIVE
|
|
||||||
* since we know that the process has exited and doing so runs into
|
|
||||||
* problems if the child process "happens to return STILL_ACTIVE(259)"
|
|
||||||
* as Microsoft's Platform SDK puts it.
|
|
||||||
*/
|
|
||||||
if (!GetExitCodeProcess (child_watch_source->pid, &child_status))
|
|
||||||
{
|
|
||||||
gchar *emsg = g_win32_error_message (GetLastError ());
|
|
||||||
g_warning (G_STRLOC ": GetExitCodeProcess() failed: %s", emsg);
|
|
||||||
g_free (emsg);
|
|
||||||
|
|
||||||
child_watch_source->child_status = -1;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
child_watch_source->child_status = child_status;
|
|
||||||
}
|
}
|
||||||
|
#endif /* HAVE_PIDFD */
|
||||||
|
child_exited = g_atomic_int_get (&child_watch_source->child_maybe_exited);
|
||||||
|
#endif /* G_OS_WIN32 */
|
||||||
|
|
||||||
return child_exited;
|
return child_exited;
|
||||||
}
|
}
|
||||||
@ -5527,9 +5549,23 @@ g_child_watch_check (GSource *source)
|
|||||||
static void
|
static void
|
||||||
g_child_watch_finalize (GSource *source)
|
g_child_watch_finalize (GSource *source)
|
||||||
{
|
{
|
||||||
|
#ifndef G_OS_WIN32
|
||||||
|
GChildWatchSource *child_watch_source = (GChildWatchSource *) source;
|
||||||
|
|
||||||
|
if (child_watch_source->poll.fd >= 0)
|
||||||
|
{
|
||||||
|
close (child_watch_source->poll.fd);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
G_LOCK (unix_signal_lock);
|
||||||
|
unix_child_watches = g_slist_remove (unix_child_watches, source);
|
||||||
|
unref_unix_signal_handler_unlocked (SIGCHLD);
|
||||||
|
G_UNLOCK (unix_signal_lock);
|
||||||
|
#endif /* G_OS_WIN32 */
|
||||||
}
|
}
|
||||||
|
|
||||||
#else /* G_OS_WIN32 */
|
#ifndef G_OS_WIN32
|
||||||
|
|
||||||
static void
|
static void
|
||||||
wake_source (GSource *source)
|
wake_source (GSource *source)
|
||||||
@ -5613,31 +5649,9 @@ dispatch_unix_signals_unlocked (void)
|
|||||||
{
|
{
|
||||||
GChildWatchSource *source = node->data;
|
GChildWatchSource *source = node->data;
|
||||||
|
|
||||||
if (!source->using_pidfd &&
|
if (g_atomic_int_compare_and_exchange (&source->child_maybe_exited, FALSE, TRUE))
|
||||||
!g_atomic_int_get (&source->child_exited))
|
|
||||||
{
|
|
||||||
pid_t pid;
|
|
||||||
do
|
|
||||||
{
|
|
||||||
g_assert (source->pid > 0);
|
|
||||||
|
|
||||||
pid = waitpid (source->pid, &source->child_status, WNOHANG);
|
|
||||||
if (pid > 0)
|
|
||||||
{
|
|
||||||
g_atomic_int_set (&source->child_exited, TRUE);
|
|
||||||
wake_source ((GSource *) source);
|
wake_source ((GSource *) source);
|
||||||
}
|
}
|
||||||
else if (pid == -1 && errno == ECHILD)
|
|
||||||
{
|
|
||||||
g_warning ("GChildWatchSource: Exit status of a child process was requested but ECHILD was received by waitpid(). See the documentation of g_child_watch_source_new() for possible causes.");
|
|
||||||
source->child_status = 0;
|
|
||||||
g_atomic_int_set (&source->child_exited, TRUE);
|
|
||||||
wake_source ((GSource *) source);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
while (pid == -1 && errno == EINTR);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* handle GUnixSignalWatchSource instances */
|
/* handle GUnixSignalWatchSource instances */
|
||||||
@ -5662,79 +5676,6 @@ dispatch_unix_signals (void)
|
|||||||
G_UNLOCK(unix_signal_lock);
|
G_UNLOCK(unix_signal_lock);
|
||||||
}
|
}
|
||||||
|
|
||||||
static gboolean
|
|
||||||
g_child_watch_prepare (GSource *source,
|
|
||||||
gint *timeout)
|
|
||||||
{
|
|
||||||
GChildWatchSource *child_watch_source;
|
|
||||||
|
|
||||||
child_watch_source = (GChildWatchSource *) source;
|
|
||||||
|
|
||||||
return g_atomic_int_get (&child_watch_source->child_exited);
|
|
||||||
}
|
|
||||||
|
|
||||||
#ifdef HAVE_PIDFD
|
|
||||||
static int
|
|
||||||
siginfo_t_to_wait_status (const siginfo_t *info)
|
|
||||||
{
|
|
||||||
/* Each of these returns is essentially the inverse of WIFEXITED(),
|
|
||||||
* WIFSIGNALED(), etc. */
|
|
||||||
switch (info->si_code)
|
|
||||||
{
|
|
||||||
case CLD_EXITED:
|
|
||||||
return W_EXITCODE (info->si_status, 0);
|
|
||||||
case CLD_KILLED:
|
|
||||||
return W_EXITCODE (0, info->si_status);
|
|
||||||
case CLD_DUMPED:
|
|
||||||
return W_EXITCODE (0, info->si_status | WCOREFLAG);
|
|
||||||
case CLD_CONTINUED:
|
|
||||||
return __W_CONTINUED;
|
|
||||||
case CLD_STOPPED:
|
|
||||||
case CLD_TRAPPED:
|
|
||||||
default:
|
|
||||||
return W_STOPCODE (info->si_status);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
#endif /* HAVE_PIDFD */
|
|
||||||
|
|
||||||
static gboolean
|
|
||||||
g_child_watch_check (GSource *source)
|
|
||||||
{
|
|
||||||
GChildWatchSource *child_watch_source;
|
|
||||||
|
|
||||||
child_watch_source = (GChildWatchSource *) source;
|
|
||||||
|
|
||||||
#ifdef HAVE_PIDFD
|
|
||||||
if (child_watch_source->using_pidfd)
|
|
||||||
{
|
|
||||||
gboolean child_exited = child_watch_source->poll.revents & G_IO_IN;
|
|
||||||
|
|
||||||
if (child_exited)
|
|
||||||
{
|
|
||||||
siginfo_t child_info = { 0, };
|
|
||||||
|
|
||||||
/* Get the exit status */
|
|
||||||
if (waitid (P_PIDFD, child_watch_source->poll.fd, &child_info, WEXITED | WNOHANG) >= 0 &&
|
|
||||||
child_info.si_pid != 0)
|
|
||||||
{
|
|
||||||
/* waitid() helpfully provides the wait status in a decomposed
|
|
||||||
* form which is quite useful. Unfortunately we have to report it
|
|
||||||
* to the #GChildWatchFunc as a waitpid()-style platform-specific
|
|
||||||
* wait status, so that the user code in #GChildWatchFunc can then
|
|
||||||
* call WIFEXITED() (etc.) on it. That means re-composing the
|
|
||||||
* status information. */
|
|
||||||
child_watch_source->child_status = siginfo_t_to_wait_status (&child_info);
|
|
||||||
child_watch_source->child_exited = TRUE;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return child_exited;
|
|
||||||
}
|
|
||||||
#endif /* HAVE_PIDFD */
|
|
||||||
|
|
||||||
return g_atomic_int_get (&child_watch_source->child_exited);
|
|
||||||
}
|
|
||||||
|
|
||||||
static gboolean
|
static gboolean
|
||||||
g_unix_signal_watch_prepare (GSource *source,
|
g_unix_signal_watch_prepare (GSource *source,
|
||||||
gint *timeout)
|
gint *timeout)
|
||||||
@ -5913,24 +5854,6 @@ g_unix_signal_watch_finalize (GSource *source)
|
|||||||
G_UNLOCK (unix_signal_lock);
|
G_UNLOCK (unix_signal_lock);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
|
||||||
g_child_watch_finalize (GSource *source)
|
|
||||||
{
|
|
||||||
GChildWatchSource *child_watch_source = (GChildWatchSource *) source;
|
|
||||||
|
|
||||||
if (child_watch_source->using_pidfd)
|
|
||||||
{
|
|
||||||
if (child_watch_source->poll.fd >= 0)
|
|
||||||
close (child_watch_source->poll.fd);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
G_LOCK (unix_signal_lock);
|
|
||||||
unix_child_watches = g_slist_remove (unix_child_watches, source);
|
|
||||||
unref_unix_signal_handler_unlocked (SIGCHLD);
|
|
||||||
G_UNLOCK (unix_signal_lock);
|
|
||||||
}
|
|
||||||
|
|
||||||
#endif /* G_OS_WIN32 */
|
#endif /* G_OS_WIN32 */
|
||||||
|
|
||||||
static gboolean
|
static gboolean
|
||||||
@ -5940,9 +5863,106 @@ g_child_watch_dispatch (GSource *source,
|
|||||||
{
|
{
|
||||||
GChildWatchSource *child_watch_source;
|
GChildWatchSource *child_watch_source;
|
||||||
GChildWatchFunc child_watch_callback = (GChildWatchFunc) callback;
|
GChildWatchFunc child_watch_callback = (GChildWatchFunc) callback;
|
||||||
|
int wait_status;
|
||||||
|
|
||||||
child_watch_source = (GChildWatchSource *) source;
|
child_watch_source = (GChildWatchSource *) source;
|
||||||
|
|
||||||
|
/* We only (try to) reap the child process right before dispatching the callback.
|
||||||
|
* That way, the caller can rely that the process is there until the callback
|
||||||
|
* is invoked; or, if the caller calls g_source_destroy() without the callback
|
||||||
|
* being dispatched, the process is still not reaped. */
|
||||||
|
|
||||||
|
#ifdef G_OS_WIN32
|
||||||
|
{
|
||||||
|
DWORD child_status;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Note: We do _not_ check for the special value of STILL_ACTIVE
|
||||||
|
* since we know that the process has exited and doing so runs into
|
||||||
|
* problems if the child process "happens to return STILL_ACTIVE(259)"
|
||||||
|
* as Microsoft's Platform SDK puts it.
|
||||||
|
*/
|
||||||
|
if (!GetExitCodeProcess (child_watch_source->pid, &child_status))
|
||||||
|
{
|
||||||
|
gchar *emsg = g_win32_error_message (GetLastError ());
|
||||||
|
g_warning (G_STRLOC ": GetExitCodeProcess() failed: %s", emsg);
|
||||||
|
g_free (emsg);
|
||||||
|
|
||||||
|
/* Unknown error. We got signaled that the process might be exited,
|
||||||
|
* but now we failed to reap it? Assume the process is gone and proceed. */
|
||||||
|
wait_status = -1;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
wait_status = child_status;
|
||||||
|
}
|
||||||
|
#else /* G_OS_WIN32 */
|
||||||
|
{
|
||||||
|
gboolean child_exited = FALSE;
|
||||||
|
|
||||||
|
wait_status = -1;
|
||||||
|
|
||||||
|
#ifdef HAVE_PIDFD
|
||||||
|
if (child_watch_source->poll.fd >= 0)
|
||||||
|
{
|
||||||
|
siginfo_t child_info = {
|
||||||
|
0,
|
||||||
|
};
|
||||||
|
|
||||||
|
/* Get the exit status */
|
||||||
|
if (waitid (P_PIDFD, child_watch_source->poll.fd, &child_info, WEXITED | WNOHANG) >= 0 &&
|
||||||
|
child_info.si_pid != 0)
|
||||||
|
{
|
||||||
|
/* waitid() helpfully provides the wait status in a decomposed
|
||||||
|
* form which is quite useful. Unfortunately we have to report it
|
||||||
|
* to the #GChildWatchFunc as a waitpid()-style platform-specific
|
||||||
|
* wait status, so that the user code in #GChildWatchFunc can then
|
||||||
|
* call WIFEXITED() (etc.) on it. That means re-composing the
|
||||||
|
* status information. */
|
||||||
|
wait_status = siginfo_t_to_wait_status (&child_info);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
/* Unknown error. We got signaled that the process might be exited,
|
||||||
|
* but now we failed to reap it? Assume the process is gone and proceed. */
|
||||||
|
g_warning (G_STRLOC ": pidfd signaled ready but failed");
|
||||||
|
}
|
||||||
|
child_exited = TRUE;
|
||||||
|
}
|
||||||
|
#endif /* HAVE_PIDFD*/
|
||||||
|
|
||||||
|
if (!child_exited)
|
||||||
|
{
|
||||||
|
pid_t pid;
|
||||||
|
int wstatus;
|
||||||
|
|
||||||
|
waitpid_again:
|
||||||
|
|
||||||
|
/* We must reset the flag before waitpid(). Otherwise, there would be a
|
||||||
|
* race. */
|
||||||
|
g_atomic_int_set (&child_watch_source->child_maybe_exited, FALSE);
|
||||||
|
|
||||||
|
pid = waitpid (child_watch_source->pid, &wstatus, WNOHANG);
|
||||||
|
|
||||||
|
if (pid == 0)
|
||||||
|
{
|
||||||
|
/* Not exited yet. Wait longer. */
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (pid > 0)
|
||||||
|
wait_status = wstatus;
|
||||||
|
else if (errno == ECHILD)
|
||||||
|
g_warning ("GChildWatchSource: Exit status of a child process was requested but ECHILD was received by waitpid(). See the documentation of g_child_watch_source_new() for possible causes.");
|
||||||
|
else if (errno == EINTR)
|
||||||
|
goto waitpid_again;
|
||||||
|
else
|
||||||
|
{
|
||||||
|
/* Unexpected error. Whatever happened, we are done waiting for this child. */
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
#endif /* G_OS_WIN32 */
|
||||||
|
|
||||||
if (!callback)
|
if (!callback)
|
||||||
{
|
{
|
||||||
g_warning ("Child watch source dispatched without callback. "
|
g_warning ("Child watch source dispatched without callback. "
|
||||||
@ -5950,7 +5970,7 @@ g_child_watch_dispatch (GSource *source,
|
|||||||
return FALSE;
|
return FALSE;
|
||||||
}
|
}
|
||||||
|
|
||||||
(child_watch_callback) (child_watch_source->pid, child_watch_source->child_status, user_data);
|
(child_watch_callback) (child_watch_source->pid, wait_status, user_data);
|
||||||
|
|
||||||
/* We never keep a child watch source around as the child is gone */
|
/* We never keep a child watch source around as the child is gone */
|
||||||
return FALSE;
|
return FALSE;
|
||||||
@ -6009,6 +6029,14 @@ g_unix_signal_handler (int signum)
|
|||||||
* mechanism, including `waitpid(pid, ...)` or a second child-watch
|
* mechanism, including `waitpid(pid, ...)` or a second child-watch
|
||||||
* source for the same @pid
|
* source for the same @pid
|
||||||
* * the application must not ignore `SIGCHLD`
|
* * the application must not ignore `SIGCHLD`
|
||||||
|
* * Before 2.78, the application could not send a signal (`kill()`) to the
|
||||||
|
* watched @pid in a race free manner. Since 2.78, you can do that while the
|
||||||
|
* associated #GMainContext is acquired.
|
||||||
|
* * Before 2.78, even after destroying the #GSource, you could not
|
||||||
|
* be sure that @pid wasn't already reaped. Hence, it was also not
|
||||||
|
* safe to `kill()` or `waitpid()` on the process ID after the child watch
|
||||||
|
* source was gone. Destroying the source before it fired made it
|
||||||
|
* impossible to reliably reap the process.
|
||||||
*
|
*
|
||||||
* If any of those conditions are not met, this and related APIs will
|
* If any of those conditions are not met, this and related APIs will
|
||||||
* not work correctly. This can often be diagnosed via a GLib warning
|
* not work correctly. This can often be diagnosed via a GLib warning
|
||||||
@ -6059,29 +6087,28 @@ g_child_watch_source_new (GPid pid)
|
|||||||
* better than SIGCHLD.
|
* better than SIGCHLD.
|
||||||
*/
|
*/
|
||||||
child_watch_source->poll.fd = (int) syscall (SYS_pidfd_open, pid, 0);
|
child_watch_source->poll.fd = (int) syscall (SYS_pidfd_open, pid, 0);
|
||||||
errsv = errno;
|
|
||||||
|
|
||||||
if (child_watch_source->poll.fd >= 0)
|
if (child_watch_source->poll.fd >= 0)
|
||||||
{
|
{
|
||||||
child_watch_source->using_pidfd = TRUE;
|
|
||||||
child_watch_source->poll.events = G_IO_IN;
|
child_watch_source->poll.events = G_IO_IN;
|
||||||
g_source_add_poll (source, &child_watch_source->poll);
|
g_source_add_poll (source, &child_watch_source->poll);
|
||||||
|
|
||||||
return source;
|
return source;
|
||||||
}
|
}
|
||||||
else
|
|
||||||
{
|
errsv = errno;
|
||||||
g_debug ("pidfd_open(%" G_PID_FORMAT ") failed with error: %s",
|
g_debug ("pidfd_open(%" G_PID_FORMAT ") failed with error: %s",
|
||||||
pid, g_strerror (errsv));
|
pid, g_strerror (errsv));
|
||||||
/* Fall through; likely the kernel isn’t new enough to support pidfd_open() */
|
/* Fall through; likely the kernel isn’t new enough to support pidfd_open() */
|
||||||
}
|
|
||||||
#endif /* HAVE_PIDFD */
|
#endif /* HAVE_PIDFD */
|
||||||
|
|
||||||
|
/* We can do that without atomic, as the source is not yet added in
|
||||||
|
* unix_child_watches (which we do next under a lock). */
|
||||||
|
child_watch_source->child_maybe_exited = TRUE;
|
||||||
|
child_watch_source->poll.fd = -1;
|
||||||
|
|
||||||
G_LOCK (unix_signal_lock);
|
G_LOCK (unix_signal_lock);
|
||||||
ref_unix_signal_handler_unlocked (SIGCHLD);
|
ref_unix_signal_handler_unlocked (SIGCHLD);
|
||||||
unix_child_watches = g_slist_prepend (unix_child_watches, child_watch_source);
|
unix_child_watches = g_slist_prepend (unix_child_watches, child_watch_source);
|
||||||
if (waitpid (pid, &child_watch_source->child_status, WNOHANG) > 0)
|
|
||||||
child_watch_source->child_exited = TRUE;
|
|
||||||
G_UNLOCK (unix_signal_lock);
|
G_UNLOCK (unix_signal_lock);
|
||||||
#endif /* !G_OS_WIN32 */
|
#endif /* !G_OS_WIN32 */
|
||||||
|
|
||||||
|
@ -37,6 +37,8 @@ cb (gpointer data)
|
|||||||
static gboolean
|
static gboolean
|
||||||
prepare (GSource *source, gint *time)
|
prepare (GSource *source, gint *time)
|
||||||
{
|
{
|
||||||
|
g_assert_nonnull (time);
|
||||||
|
g_assert_cmpint (*time, ==, -1);
|
||||||
return FALSE;
|
return FALSE;
|
||||||
}
|
}
|
||||||
static gboolean
|
static gboolean
|
||||||
|
@ -372,6 +372,99 @@ test_get_passwd_entry_nonexistent (void)
|
|||||||
g_clear_error (&local_error);
|
g_clear_error (&local_error);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
_child_wait_watch_cb (GPid pid,
|
||||||
|
gint wait_status,
|
||||||
|
gpointer user_data)
|
||||||
|
{
|
||||||
|
gboolean *p_got_callback = user_data;
|
||||||
|
|
||||||
|
g_assert_nonnull (p_got_callback);
|
||||||
|
g_assert_false (*p_got_callback);
|
||||||
|
*p_got_callback = TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
test_child_wait (void)
|
||||||
|
{
|
||||||
|
gboolean r;
|
||||||
|
GPid pid;
|
||||||
|
guint id;
|
||||||
|
pid_t pid2;
|
||||||
|
int wstatus;
|
||||||
|
gboolean got_callback = FALSE;
|
||||||
|
gboolean iterate_maincontext = g_test_rand_bit ();
|
||||||
|
char **argv;
|
||||||
|
int errsv;
|
||||||
|
|
||||||
|
/* - We spawn a trivial child process that exits after a short time.
|
||||||
|
* - We schedule a g_child_watch_add()
|
||||||
|
* - we may iterate the GMainContext a bit. Randomly we either get the
|
||||||
|
* child-watcher callback or not.
|
||||||
|
* - if we didn't get the callback, we g_source_remove() the child watcher.
|
||||||
|
*
|
||||||
|
* Afterwards, if the callback didn't fire, we check that we are able to waitpid()
|
||||||
|
* on the process ourselves. Of course, if the child watcher notified, the waitpid()
|
||||||
|
* will fail with ECHILD.
|
||||||
|
*/
|
||||||
|
|
||||||
|
argv = g_test_rand_bit () ? ((char *[]){ "/bin/sleep", "0.05", NULL }) : ((char *[]){ "/bin/true", NULL });
|
||||||
|
|
||||||
|
r = g_spawn_async (NULL,
|
||||||
|
argv,
|
||||||
|
NULL,
|
||||||
|
G_SPAWN_DO_NOT_REAP_CHILD,
|
||||||
|
NULL,
|
||||||
|
NULL,
|
||||||
|
&pid,
|
||||||
|
NULL);
|
||||||
|
if (!r)
|
||||||
|
{
|
||||||
|
/* Some odd system without /bin/sleep? Skip the test. */
|
||||||
|
g_test_skip ("failure to spawn test process in test_child_wait()");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
g_assert_cmpint (pid, >=, 1);
|
||||||
|
|
||||||
|
if (g_test_rand_bit ())
|
||||||
|
g_usleep (g_test_rand_int_range (0, (G_USEC_PER_SEC / 10)));
|
||||||
|
|
||||||
|
id = g_child_watch_add (pid, _child_wait_watch_cb, &got_callback);
|
||||||
|
|
||||||
|
if (g_test_rand_bit ())
|
||||||
|
g_usleep (g_test_rand_int_range (0, (G_USEC_PER_SEC / 10)));
|
||||||
|
|
||||||
|
if (iterate_maincontext)
|
||||||
|
{
|
||||||
|
gint64 start_usec = g_get_monotonic_time ();
|
||||||
|
gint64 end_usec = start_usec + g_test_rand_int_range (0, (G_USEC_PER_SEC / 10));
|
||||||
|
|
||||||
|
while (!got_callback && g_get_monotonic_time () < end_usec)
|
||||||
|
g_main_context_iteration (NULL, FALSE);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!got_callback)
|
||||||
|
g_source_remove (id);
|
||||||
|
|
||||||
|
errno = 0;
|
||||||
|
pid2 = waitpid (pid, &wstatus, 0);
|
||||||
|
errsv = errno;
|
||||||
|
if (got_callback)
|
||||||
|
{
|
||||||
|
g_assert_true (iterate_maincontext);
|
||||||
|
g_assert_cmpint (errsv, ==, ECHILD);
|
||||||
|
g_assert_cmpint (pid2, <, 0);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
g_assert_cmpint (errsv, ==, 0);
|
||||||
|
g_assert_cmpint (pid2, ==, pid);
|
||||||
|
g_assert_true (WIFEXITED (wstatus));
|
||||||
|
g_assert_cmpint (WEXITSTATUS (wstatus), ==, 0);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
int
|
int
|
||||||
main (int argc,
|
main (int argc,
|
||||||
char *argv[])
|
char *argv[])
|
||||||
@ -390,6 +483,7 @@ main (int argc,
|
|||||||
g_test_add_func ("/glib-unix/callback_after_signal", test_callback_after_signal);
|
g_test_add_func ("/glib-unix/callback_after_signal", test_callback_after_signal);
|
||||||
g_test_add_func ("/glib-unix/get-passwd-entry/root", test_get_passwd_entry_root);
|
g_test_add_func ("/glib-unix/get-passwd-entry/root", test_get_passwd_entry_root);
|
||||||
g_test_add_func ("/glib-unix/get-passwd-entry/nonexistent", test_get_passwd_entry_nonexistent);
|
g_test_add_func ("/glib-unix/get-passwd-entry/nonexistent", test_get_passwd_entry_nonexistent);
|
||||||
|
g_test_add_func ("/glib-unix/child-wait", test_child_wait);
|
||||||
|
|
||||||
return g_test_run();
|
return g_test_run();
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user