mirror of
https://gitlab.gnome.org/GNOME/glib.git
synced 2025-01-28 06:56:16 +01:00
Merge branch '2332-subprocess-launcher-oops' into 'master'
gsubprocesslauncher: Don’t close target FDs in close() method Closes #2332 See merge request GNOME/glib!1958
This commit is contained in:
commit
7b0ac98afe
@ -42,8 +42,8 @@ struct _GSubprocessLauncher
|
|||||||
gint stderr_fd;
|
gint stderr_fd;
|
||||||
gchar *stderr_path;
|
gchar *stderr_path;
|
||||||
|
|
||||||
GArray *source_fds;
|
GArray *source_fds; /* GSubprocessLauncher has ownership of the FD elements */
|
||||||
GArray *target_fds; /* always the same length as source_fds */
|
GArray *target_fds; /* always the same length as source_fds; elements are just integers and not FDs in this process */
|
||||||
gboolean closed_fd;
|
gboolean closed_fd;
|
||||||
|
|
||||||
GSpawnChildSetupFunc child_setup_func;
|
GSpawnChildSetupFunc child_setup_func;
|
||||||
|
@ -596,16 +596,16 @@ g_subprocess_launcher_take_stderr_fd (GSubprocessLauncher *self,
|
|||||||
* @target_fd: Target descriptor for child process
|
* @target_fd: Target descriptor for child process
|
||||||
*
|
*
|
||||||
* Transfer an arbitrary file descriptor from parent process to the
|
* Transfer an arbitrary file descriptor from parent process to the
|
||||||
* child. This function takes "ownership" of the fd; it will be closed
|
* child. This function takes ownership of the @source_fd; it will be closed
|
||||||
* in the parent when @self is freed.
|
* in the parent when @self is freed.
|
||||||
*
|
*
|
||||||
* By default, all file descriptors from the parent will be closed.
|
* By default, all file descriptors from the parent will be closed.
|
||||||
* This function allows you to create (for example) a custom pipe() or
|
* This function allows you to create (for example) a custom `pipe()` or
|
||||||
* socketpair() before launching the process, and choose the target
|
* `socketpair()` before launching the process, and choose the target
|
||||||
* descriptor in the child.
|
* descriptor in the child.
|
||||||
*
|
*
|
||||||
* An example use case is GNUPG, which has a command line argument
|
* An example use case is GNUPG, which has a command line argument
|
||||||
* --passphrase-fd providing a file descriptor number where it expects
|
* `--passphrase-fd` providing a file descriptor number where it expects
|
||||||
* the passphrase to be written.
|
* the passphrase to be written.
|
||||||
*/
|
*/
|
||||||
void
|
void
|
||||||
@ -661,11 +661,11 @@ g_subprocess_launcher_close (GSubprocessLauncher *self)
|
|||||||
g_assert (self->target_fds != NULL);
|
g_assert (self->target_fds != NULL);
|
||||||
g_assert (self->source_fds->len == self->target_fds->len);
|
g_assert (self->source_fds->len == self->target_fds->len);
|
||||||
|
|
||||||
|
/* Note: Don’t close the target_fds, as they’re only valid FDs in the
|
||||||
|
* child process. This code never executes in the child process. */
|
||||||
for (i = 0; i < self->source_fds->len; i++)
|
for (i = 0; i < self->source_fds->len; i++)
|
||||||
{
|
|
||||||
(void) close (g_array_index (self->source_fds, int, i));
|
(void) close (g_array_index (self->source_fds, int, i));
|
||||||
(void) close (g_array_index (self->target_fds, int, i));
|
|
||||||
}
|
|
||||||
g_clear_pointer (&self->source_fds, g_array_unref);
|
g_clear_pointer (&self->source_fds, g_array_unref);
|
||||||
g_clear_pointer (&self->target_fds, g_array_unref);
|
g_clear_pointer (&self->target_fds, g_array_unref);
|
||||||
}
|
}
|
||||||
|
@ -1494,23 +1494,44 @@ test_subprocess_launcher_close (void)
|
|||||||
GSubprocessLauncher *launcher;
|
GSubprocessLauncher *launcher;
|
||||||
GSubprocess *proc;
|
GSubprocess *proc;
|
||||||
GPtrArray *args;
|
GPtrArray *args;
|
||||||
int fd;
|
int fd, fd2;
|
||||||
gboolean is_open;
|
gboolean is_open;
|
||||||
|
|
||||||
|
/* Open two arbitrary FDs. One of them, @fd, will be transferred to the
|
||||||
|
* launcher, and the other’s FD integer will be used as its target FD, giving
|
||||||
|
* the mapping `fd → fd2` if a child process were to be spawned.
|
||||||
|
*
|
||||||
|
* The launcher will then be closed, which should close @fd but *not* @fd2,
|
||||||
|
* as the value of @fd2 is only valid as an FD in a child process. (A child
|
||||||
|
* process is not actually spawned in this test.)
|
||||||
|
*/
|
||||||
fd = dup (0);
|
fd = dup (0);
|
||||||
|
fd2 = dup (0);
|
||||||
launcher = g_subprocess_launcher_new (G_SUBPROCESS_FLAGS_NONE);
|
launcher = g_subprocess_launcher_new (G_SUBPROCESS_FLAGS_NONE);
|
||||||
g_subprocess_launcher_take_fd (launcher, fd, fd);
|
g_subprocess_launcher_take_fd (launcher, fd, fd2);
|
||||||
|
|
||||||
is_open = fcntl (fd, F_GETFD) != -1;
|
is_open = fcntl (fd, F_GETFD) != -1;
|
||||||
g_assert_true (is_open);
|
g_assert_true (is_open);
|
||||||
|
is_open = fcntl (fd2, F_GETFD) != -1;
|
||||||
|
g_assert_true (is_open);
|
||||||
|
|
||||||
g_subprocess_launcher_close (launcher);
|
g_subprocess_launcher_close (launcher);
|
||||||
|
|
||||||
is_open = fcntl (fd, F_GETFD) != -1;
|
is_open = fcntl (fd, F_GETFD) != -1;
|
||||||
g_assert_false (is_open);
|
g_assert_false (is_open);
|
||||||
|
is_open = fcntl (fd2, F_GETFD) != -1;
|
||||||
|
g_assert_true (is_open);
|
||||||
|
|
||||||
|
/* Now test that actually trying to spawn the child gives %G_IO_ERROR_CLOSED,
|
||||||
|
* as g_subprocess_launcher_close() has been called. */
|
||||||
args = get_test_subprocess_args ("cat", NULL);
|
args = get_test_subprocess_args ("cat", NULL);
|
||||||
proc = g_subprocess_launcher_spawnv (launcher, (const gchar * const *) args->pdata, error);
|
proc = g_subprocess_launcher_spawnv (launcher, (const gchar * const *) args->pdata, error);
|
||||||
g_ptr_array_free (args, TRUE);
|
g_ptr_array_free (args, TRUE);
|
||||||
g_assert_null (proc);
|
g_assert_null (proc);
|
||||||
g_assert_error (local_error, G_IO_ERROR, G_IO_ERROR_CLOSED);
|
g_assert_error (local_error, G_IO_ERROR, G_IO_ERROR_CLOSED);
|
||||||
g_clear_error (error);
|
g_clear_error (error);
|
||||||
|
|
||||||
|
close (fd2);
|
||||||
g_object_unref (launcher);
|
g_object_unref (launcher);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user