1 /* GIO - GLib Input, Output and Streaming Library
3 * Copyright © 2012 Red Hat, Inc.
4 * Copyright © 2012-2013 Canonical Limited
6 * This library is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU Lesser General Public
8 * License as published by the Free Software Foundation; either
9 * version 2.1 of the License, or (at your option) any later version.
11 * See the included COPYING file for more information.
13 * Authors: Colin Walters <walters@verbum.org>
14 * Ryan Lortie <desrt@desrt.ca>
18 * SECTION:gsubprocesslauncher
19 * @title: GSubprocess Launcher
20 * @short_description: Environment options for launching a child process
23 * This class contains a set of options for launching child processes,
24 * such as where its standard input and output will be directed, the
25 * argument list, the environment, and more.
27 * While the #GSubprocess class has high level functions covering
28 * popular cases, use of this class allows access to more advanced
29 * options. It can also be used to launch multiple subprocesses with
30 * a similar configuration.
35 #define ALL_STDIN_FLAGS (G_SUBPROCESS_FLAGS_STDIN_PIPE | \
36 G_SUBPROCESS_FLAGS_STDIN_INHERIT)
37 #define ALL_STDOUT_FLAGS (G_SUBPROCESS_FLAGS_STDOUT_PIPE | \
38 G_SUBPROCESS_FLAGS_STDOUT_SILENCE)
39 #define ALL_STDERR_FLAGS (G_SUBPROCESS_FLAGS_STDERR_PIPE | \
40 G_SUBPROCESS_FLAGS_STDERR_SILENCE | \
41 G_SUBPROCESS_FLAGS_STDERR_MERGE)
45 #include "gsubprocesslauncher-private.h"
46 #include "gioenumtypes.h"
47 #include "gsubprocess.h"
48 #include "ginitable.h"
55 typedef GObjectClass GSubprocessLauncherClass
;
57 G_DEFINE_TYPE (GSubprocessLauncher
, g_subprocess_launcher
, G_TYPE_OBJECT
)
60 verify_disposition (const gchar
*stream_name
,
61 GSubprocessFlags filtered_flags
,
63 const gchar
*filename
)
69 else if (((filtered_flags
- 1) & filtered_flags
) == 0)
72 n_bits
= 2; /* ...or more */
74 if (n_bits
+ (fd
>= 0) + (filename
!= NULL
) > 1)
78 err
= g_string_new (NULL
);
84 class = g_type_class_peek (G_TYPE_SUBPROCESS_FLAGS
);
86 for (i
= 0; i
< class->n_values
; i
++)
88 const GFlagsValue
*value
= &class->values
[i
];
90 if (filtered_flags
& value
->value
)
91 g_string_append_printf (err
, " %s", value
->value_name
);
94 g_type_class_unref (class);
98 g_string_append_printf (err
, " g_subprocess_launcher_take_%s_fd()", stream_name
);
101 g_string_append_printf (err
, " g_subprocess_launcher_set_%s_file_path()", stream_name
);
103 g_critical ("You may specify at most one disposition for the %s stream, but you specified:%s.",
104 stream_name
, err
->str
);
105 g_string_free (err
, TRUE
);
114 verify_flags (GSubprocessFlags flags
)
116 return verify_disposition ("stdin", flags
& ALL_STDIN_FLAGS
, -1, NULL
) &&
117 verify_disposition ("stdout", flags
& ALL_STDOUT_FLAGS
, -1, NULL
) &&
118 verify_disposition ("stderr", flags
& ALL_STDERR_FLAGS
, -1, NULL
);
122 g_subprocess_launcher_set_property (GObject
*object
, guint prop_id
,
123 const GValue
*value
, GParamSpec
*pspec
)
125 GSubprocessLauncher
*launcher
= G_SUBPROCESS_LAUNCHER (object
);
127 g_assert (prop_id
== 1);
129 if (verify_flags (g_value_get_flags (value
)))
130 launcher
->flags
= g_value_get_flags (value
);
134 g_subprocess_launcher_finalize (GObject
*object
)
136 GSubprocessLauncher
*self
= G_SUBPROCESS_LAUNCHER (object
);
141 g_free (self
->stdin_path
);
142 g_free (self
->stdout_path
);
143 g_free (self
->stderr_path
);
145 if (self
->stdin_fd
!= -1)
146 close (self
->stdin_fd
);
148 if (self
->stdout_fd
!= -1)
149 close (self
->stdout_fd
);
151 if (self
->stderr_fd
!= -1)
152 close (self
->stderr_fd
);
154 if (self
->basic_fd_assignments
)
156 for (i
= 0; i
< self
->basic_fd_assignments
->len
; i
++)
157 (void) close (g_array_index (self
->basic_fd_assignments
, int, i
));
158 g_array_unref (self
->basic_fd_assignments
);
160 if (self
->needdup_fd_assignments
)
162 for (i
= 0; i
< self
->needdup_fd_assignments
->len
; i
+= 2)
163 (void) close (g_array_index (self
->needdup_fd_assignments
, int, i
));
164 g_array_unref (self
->needdup_fd_assignments
);
167 if (self
->child_setup_destroy_notify
)
168 (* self
->child_setup_destroy_notify
) (self
->child_setup_user_data
);
171 g_strfreev (self
->envp
);
174 G_OBJECT_CLASS (g_subprocess_launcher_parent_class
)->finalize (object
);
178 g_subprocess_launcher_init (GSubprocessLauncher
*self
)
180 self
->envp
= g_get_environ ();
184 self
->stdout_fd
= -1;
185 self
->stderr_fd
= -1;
186 self
->basic_fd_assignments
= g_array_new (FALSE
, 0, sizeof (int));
187 self
->needdup_fd_assignments
= g_array_new (FALSE
, 0, sizeof (int));
192 g_subprocess_launcher_class_init (GSubprocessLauncherClass
*class)
194 GObjectClass
*gobject_class
= G_OBJECT_CLASS (class);
196 gobject_class
->set_property
= g_subprocess_launcher_set_property
;
197 gobject_class
->finalize
= g_subprocess_launcher_finalize
;
199 g_object_class_install_property (gobject_class
, 1,
200 g_param_spec_flags ("flags", "Flags", "GSubprocessFlags for launched processes",
201 G_TYPE_SUBPROCESS_FLAGS
, 0, G_PARAM_WRITABLE
|
202 G_PARAM_STATIC_STRINGS
| G_PARAM_CONSTRUCT_ONLY
));
206 * g_subprocess_launcher_new:
207 * @flags: #GSubprocessFlags
209 * Creates a new #GSubprocessLauncher.
211 * The launcher is created with the default options. A copy of the
212 * environment of the calling process is made at the time of this call
213 * and will be used as the environment that the process is launched in.
217 GSubprocessLauncher
*
218 g_subprocess_launcher_new (GSubprocessFlags flags
)
220 if (!verify_flags (flags
))
223 return g_object_new (G_TYPE_SUBPROCESS_LAUNCHER
,
229 * g_subprocess_launcher_set_environ:
230 * @self: a #GSubprocess
231 * @env: (array zero-terminated=1) (element-type filename) (transfer none):
232 * the replacement environment
234 * Replace the entire environment of processes launched from this
235 * launcher with the given 'environ' variable.
237 * Typically you will build this variable by using g_listenv() to copy
238 * the process 'environ' and using the functions g_environ_setenv(),
239 * g_environ_unsetenv(), etc.
241 * As an alternative, you can use g_subprocess_launcher_setenv(),
242 * g_subprocess_launcher_unsetenv(), etc.
244 * Pass an empty array to set an empty environment. Pass %NULL to inherit the
245 * parent process’ environment. As of GLib 2.54, the parent process’ environment
246 * will be copied when g_subprocess_launcher_set_environ() is called.
247 * Previously, it was copied when the subprocess was executed. This means the
248 * copied environment may now be modified (using g_subprocess_launcher_setenv(),
249 * etc.) before launching the subprocess.
251 * On UNIX, all strings in this array can be arbitrary byte strings.
252 * On Windows, they should be in UTF-8.
257 g_subprocess_launcher_set_environ (GSubprocessLauncher
*self
,
260 g_strfreev (self
->envp
);
261 self
->envp
= g_strdupv (env
);
263 if (self
->envp
== NULL
)
264 self
->envp
= g_get_environ ();
268 * g_subprocess_launcher_setenv:
269 * @self: a #GSubprocess
270 * @variable: (type filename): the environment variable to set,
271 * must not contain '='
272 * @value: (type filename): the new value for the variable
273 * @overwrite: whether to change the variable if it already exists
275 * Sets the environment variable @variable in the environment of
276 * processes launched from this launcher.
278 * On UNIX, both the variable's name and value can be arbitrary byte
279 * strings, except that the variable's name cannot contain '='.
280 * On Windows, they should be in UTF-8.
285 g_subprocess_launcher_setenv (GSubprocessLauncher
*self
,
286 const gchar
*variable
,
290 self
->envp
= g_environ_setenv (self
->envp
, variable
, value
, overwrite
);
294 * g_subprocess_launcher_unsetenv:
295 * @self: a #GSubprocess
296 * @variable: (type filename): the environment variable to unset,
297 * must not contain '='
299 * Removes the environment variable @variable from the environment of
300 * processes launched from this launcher.
302 * On UNIX, the variable's name can be an arbitrary byte string not
303 * containing '='. On Windows, it should be in UTF-8.
308 g_subprocess_launcher_unsetenv (GSubprocessLauncher
*self
,
309 const gchar
*variable
)
311 self
->envp
= g_environ_unsetenv (self
->envp
, variable
);
315 * g_subprocess_launcher_getenv:
316 * @self: a #GSubprocess
317 * @variable: (type filename): the environment variable to get
319 * Returns the value of the environment variable @variable in the
320 * environment of processes launched from this launcher.
322 * On UNIX, the returned string can be an arbitrary byte string.
323 * On Windows, it will be UTF-8.
325 * Returns: (type filename): the value of the environment variable,
331 g_subprocess_launcher_getenv (GSubprocessLauncher
*self
,
332 const gchar
*variable
)
334 return g_environ_getenv (self
->envp
, variable
);
338 * g_subprocess_launcher_set_cwd:
339 * @self: a #GSubprocess
340 * @cwd: (type filename): the cwd for launched processes
342 * Sets the current working directory that processes will be launched
345 * By default processes are launched with the current working directory
346 * of the launching process at the time of launch.
351 g_subprocess_launcher_set_cwd (GSubprocessLauncher
*self
,
355 self
->cwd
= g_strdup (cwd
);
359 * g_subprocess_launcher_set_flags:
360 * @self: a #GSubprocessLauncher
361 * @flags: #GSubprocessFlags
363 * Sets the flags on the launcher.
365 * The default flags are %G_SUBPROCESS_FLAGS_NONE.
367 * You may not set flags that specify conflicting options for how to
368 * handle a particular stdio stream (eg: specifying both
369 * %G_SUBPROCESS_FLAGS_STDIN_PIPE and
370 * %G_SUBPROCESS_FLAGS_STDIN_INHERIT).
372 * You may also not set a flag that conflicts with a previous call to a
373 * function like g_subprocess_launcher_set_stdin_file_path() or
374 * g_subprocess_launcher_take_stdout_fd().
379 g_subprocess_launcher_set_flags (GSubprocessLauncher
*self
,
380 GSubprocessFlags flags
)
382 const gchar
*stdin_path
= NULL
, *stdout_path
= NULL
, *stderr_path
= NULL
;
383 gint stdin_fd
= -1, stdout_fd
= -1, stderr_fd
= -1;
386 stdin_fd
= self
->stdin_fd
;
387 stdout_fd
= self
->stdout_fd
;
388 stderr_fd
= self
->stderr_fd
;
389 stdin_path
= self
->stdin_path
;
390 stdout_path
= self
->stdout_path
;
391 stderr_path
= self
->stderr_path
;
394 if (verify_disposition ("stdin", flags
& ALL_STDIN_FLAGS
, stdin_fd
, stdin_path
) &&
395 verify_disposition ("stdout", flags
& ALL_STDOUT_FLAGS
, stdout_fd
, stdout_path
) &&
396 verify_disposition ("stderr", flags
& ALL_STDERR_FLAGS
, stderr_fd
, stderr_path
))
402 assign_fd (gint
*fd_ptr
, gint fd
)
414 flags
= fcntl (fd
, F_GETFD
);
415 if (~flags
& FD_CLOEXEC
)
416 fcntl (fd
, F_SETFD
, flags
| FD_CLOEXEC
);
421 * g_subprocess_launcher_set_stdin_file_path:
422 * @self: a #GSubprocessLauncher
423 * @path: (type filename) (nullable: a filename or %NULL
425 * Sets the file path to use as the stdin for spawned processes.
427 * If @path is %NULL then any previously given path is unset.
429 * The file must exist or spawning the process will fail.
431 * You may not set a stdin file path if a stdin fd is already set or if
432 * the launcher flags contain any flags directing stdin elsewhere.
434 * This feature is only available on UNIX.
439 g_subprocess_launcher_set_stdin_file_path (GSubprocessLauncher
*self
,
442 if (verify_disposition ("stdin", self
->flags
& ALL_STDIN_FLAGS
, self
->stdin_fd
, path
))
444 g_free (self
->stdin_path
);
445 self
->stdin_path
= g_strdup (path
);
450 * g_subprocess_launcher_take_stdin_fd:
451 * @self: a #GSubprocessLauncher
452 * @fd: a file descriptor, or -1
454 * Sets the file descriptor to use as the stdin for spawned processes.
456 * If @fd is -1 then any previously given fd is unset.
458 * Note that if your intention is to have the stdin of the calling
459 * process inherited by the child then %G_SUBPROCESS_FLAGS_STDIN_INHERIT
460 * is a better way to go about doing that.
462 * The passed @fd is noted but will not be touched in the current
463 * process. It is therefore necessary that it be kept open by the
464 * caller until the subprocess is spawned. The file descriptor will
465 * also not be explicitly closed on the child side, so it must be marked
466 * O_CLOEXEC if that's what you want.
468 * You may not set a stdin fd if a stdin file path is already set or if
469 * the launcher flags contain any flags directing stdin elsewhere.
471 * This feature is only available on UNIX.
476 g_subprocess_launcher_take_stdin_fd (GSubprocessLauncher
*self
,
479 if (verify_disposition ("stdin", self
->flags
& ALL_STDIN_FLAGS
, fd
, self
->stdin_path
))
480 assign_fd (&self
->stdin_fd
, fd
);
484 * g_subprocess_launcher_set_stdout_file_path:
485 * @self: a #GSubprocessLauncher
486 * @path: (type filename) (nullable): a filename or %NULL
488 * Sets the file path to use as the stdout for spawned processes.
490 * If @path is %NULL then any previously given path is unset.
492 * The file will be created or truncated when the process is spawned, as
493 * would be the case if using '>' at the shell.
495 * You may not set a stdout file path if a stdout fd is already set or
496 * if the launcher flags contain any flags directing stdout elsewhere.
498 * This feature is only available on UNIX.
503 g_subprocess_launcher_set_stdout_file_path (GSubprocessLauncher
*self
,
506 if (verify_disposition ("stdout", self
->flags
& ALL_STDOUT_FLAGS
, self
->stdout_fd
, path
))
508 g_free (self
->stdout_path
);
509 self
->stdout_path
= g_strdup (path
);
514 * g_subprocess_launcher_take_stdout_fd:
515 * @self: a #GSubprocessLauncher
516 * @fd: a file descriptor, or -1
518 * Sets the file descriptor to use as the stdout for spawned processes.
520 * If @fd is -1 then any previously given fd is unset.
522 * Note that the default behaviour is to pass stdout through to the
523 * stdout of the parent process.
525 * The passed @fd is noted but will not be touched in the current
526 * process. It is therefore necessary that it be kept open by the
527 * caller until the subprocess is spawned. The file descriptor will
528 * also not be explicitly closed on the child side, so it must be marked
529 * O_CLOEXEC if that's what you want.
531 * You may not set a stdout fd if a stdout file path is already set or
532 * if the launcher flags contain any flags directing stdout elsewhere.
534 * This feature is only available on UNIX.
539 g_subprocess_launcher_take_stdout_fd (GSubprocessLauncher
*self
,
542 if (verify_disposition ("stdout", self
->flags
& ALL_STDOUT_FLAGS
, fd
, self
->stdout_path
))
543 assign_fd (&self
->stdout_fd
, fd
);
547 * g_subprocess_launcher_set_stderr_file_path:
548 * @self: a #GSubprocessLauncher
549 * @path: (type filename) (nullable): a filename or %NULL
551 * Sets the file path to use as the stderr for spawned processes.
553 * If @path is %NULL then any previously given path is unset.
555 * The file will be created or truncated when the process is spawned, as
556 * would be the case if using '2>' at the shell.
558 * If you want to send both stdout and stderr to the same file then use
559 * %G_SUBPROCESS_FLAGS_STDERR_MERGE.
561 * You may not set a stderr file path if a stderr fd is already set or
562 * if the launcher flags contain any flags directing stderr elsewhere.
564 * This feature is only available on UNIX.
569 g_subprocess_launcher_set_stderr_file_path (GSubprocessLauncher
*self
,
572 if (verify_disposition ("stderr", self
->flags
& ALL_STDERR_FLAGS
, self
->stderr_fd
, path
))
574 g_free (self
->stderr_path
);
575 self
->stderr_path
= g_strdup (path
);
580 * g_subprocess_launcher_take_stderr_fd:
581 * @self: a #GSubprocessLauncher
582 * @fd: a file descriptor, or -1
584 * Sets the file descriptor to use as the stderr for spawned processes.
586 * If @fd is -1 then any previously given fd is unset.
588 * Note that the default behaviour is to pass stderr through to the
589 * stderr of the parent process.
591 * The passed @fd belongs to the #GSubprocessLauncher. It will be
592 * automatically closed when the launcher is finalized. The file
593 * descriptor will also be closed on the child side when executing the
596 * You may not set a stderr fd if a stderr file path is already set or
597 * if the launcher flags contain any flags directing stderr elsewhere.
599 * This feature is only available on UNIX.
604 g_subprocess_launcher_take_stderr_fd (GSubprocessLauncher
*self
,
607 if (verify_disposition ("stderr", self
->flags
& ALL_STDERR_FLAGS
, fd
, self
->stderr_path
))
608 assign_fd (&self
->stderr_fd
, fd
);
612 * g_subprocess_launcher_take_fd:
613 * @self: a #GSubprocessLauncher
614 * @source_fd: File descriptor in parent process
615 * @target_fd: Target descriptor for child process
617 * Transfer an arbitrary file descriptor from parent process to the
618 * child. This function takes "ownership" of the fd; it will be closed
619 * in the parent when @self is freed.
621 * By default, all file descriptors from the parent will be closed.
622 * This function allows you to create (for example) a custom pipe() or
623 * socketpair() before launching the process, and choose the target
624 * descriptor in the child.
626 * An example use case is GNUPG, which has a command line argument
627 * --passphrase-fd providing a file descriptor number where it expects
628 * the passphrase to be written.
631 g_subprocess_launcher_take_fd (GSubprocessLauncher
*self
,
635 if (source_fd
== target_fd
)
637 g_array_append_val (self
->basic_fd_assignments
, source_fd
);
641 g_array_append_val (self
->needdup_fd_assignments
, source_fd
);
642 g_array_append_val (self
->needdup_fd_assignments
, target_fd
);
647 * g_subprocess_launcher_set_child_setup: (skip)
648 * @self: a #GSubprocessLauncher
649 * @child_setup: a #GSpawnChildSetupFunc to use as the child setup function
650 * @user_data: user data for @child_setup
651 * @destroy_notify: a #GDestroyNotify for @user_data
653 * Sets up a child setup function.
655 * The child setup function will be called after fork() but before
656 * exec() on the child's side.
658 * @destroy_notify will not be automatically called on the child's side
659 * of the fork(). It will only be called when the last reference on the
660 * #GSubprocessLauncher is dropped or when a new child setup function is
663 * %NULL can be given as @child_setup to disable the functionality.
665 * Child setup functions are only available on UNIX.
670 g_subprocess_launcher_set_child_setup (GSubprocessLauncher
*self
,
671 GSpawnChildSetupFunc child_setup
,
673 GDestroyNotify destroy_notify
)
675 if (self
->child_setup_destroy_notify
)
676 (* self
->child_setup_destroy_notify
) (self
->child_setup_user_data
);
678 self
->child_setup_func
= child_setup
;
679 self
->child_setup_user_data
= user_data
;
680 self
->child_setup_destroy_notify
= destroy_notify
;
685 * g_subprocess_launcher_spawn:
686 * @self: a #GSubprocessLauncher
688 * @argv0: Command line arguments
689 * @...: Continued arguments, %NULL terminated
691 * Creates a #GSubprocess given a provided varargs list of arguments.
694 * Returns: (transfer full): A new #GSubprocess, or %NULL on error (and @error will be set)
697 g_subprocess_launcher_spawn (GSubprocessLauncher
*launcher
,
707 g_return_val_if_fail (argv0
!= NULL
&& argv0
[0] != '\0', NULL
);
708 g_return_val_if_fail (error
== NULL
|| *error
== NULL
, NULL
);
710 args
= g_ptr_array_new ();
712 va_start (ap
, argv0
);
713 g_ptr_array_add (args
, (gchar
*) argv0
);
714 while ((arg
= va_arg (ap
, const gchar
*)))
715 g_ptr_array_add (args
, (gchar
*) arg
);
717 g_ptr_array_add (args
, NULL
);
720 result
= g_subprocess_launcher_spawnv (launcher
, (const gchar
* const *) args
->pdata
, error
);
722 g_ptr_array_free (args
, TRUE
);
729 * g_subprocess_launcher_spawnv:
730 * @self: a #GSubprocessLauncher
731 * @argv: (array zero-terminated=1) (element-type filename): Command line arguments
734 * Creates a #GSubprocess given a provided array of arguments.
737 * Returns: (transfer full): A new #GSubprocess, or %NULL on error (and @error will be set)
740 g_subprocess_launcher_spawnv (GSubprocessLauncher
*launcher
,
741 const gchar
* const *argv
,
744 GSubprocess
*subprocess
;
746 g_return_val_if_fail (argv
!= NULL
&& argv
[0] != NULL
&& argv
[0][0] != '\0', NULL
);
748 subprocess
= g_object_new (G_TYPE_SUBPROCESS
,
750 "flags", launcher
->flags
,
752 g_subprocess_set_launcher (subprocess
, launcher
);
754 if (!g_initable_init (G_INITABLE (subprocess
), NULL
, error
))
756 g_object_unref (subprocess
);