1 /* Creation of subprocesses, communicating via pipes.
2 Copyright (C) 2001-2004, 2006-2020 Free Software Foundation, Inc.
3 Written by Bruno Haible <haible@clisp.cons.org>, 2001.
5 This program is free software: you can redistribute it and/or modify
6 it under the terms of the GNU General Public License as published by
7 the Free Software Foundation; either version 3 of the License, or
8 (at your option) any later version.
10 This program is distributed in the hope that it will be useful,
11 but WITHOUT ANY WARRANTY; without even the implied warranty of
12 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 GNU General Public License for more details.
15 You should have received a copy of the GNU General Public License
16 along with this program. If not, see <https://www.gnu.org/licenses/>. */
19 /* Tell clang not to warn about the 'child' variable, below. */
21 # pragma clang diagnostic ignored "-Wconditional-uninitialized"
27 #include "spawn-pipe.h"
36 #include "fatal-signal.h"
37 #include "unistd-safer.h"
38 #include "wait-process.h"
41 #define _(str) gettext (str)
43 #if (defined _WIN32 && ! defined __CYGWIN__) || defined __KLIBC__
45 /* Native Windows API. */
47 # include "w32spawn.h"
59 /* EINTR handling for close().
60 These functions can return -1/EINTR even though we don't have any
61 signal handlers set up, namely when we get interrupted via SIGSTOP. */
64 nonintr_close (int fd
)
70 while (retval
< 0 && errno
== EINTR
);
74 #undef close /* avoid warning related to gnulib module unistd */
75 #define close nonintr_close
77 #if defined _WIN32 && ! defined __CYGWIN__
79 nonintr_open (const char *pathname
, int oflag
, mode_t mode
)
84 retval
= open (pathname
, oflag
, mode
);
85 while (retval
< 0 && errno
== EINTR
);
89 # undef open /* avoid warning on VMS */
90 # define open nonintr_open
96 /* Open a pipe connected to a child process.
99 * parent -> fd[1] -> STDIN_FILENO -> child if pipe_stdin
100 * parent <- fd[0] <- STDOUT_FILENO <- child if pipe_stdout
103 * At least one of pipe_stdin, pipe_stdout must be true.
104 * pipe_stdin and prog_stdin together determine the child's standard input.
105 * pipe_stdout and prog_stdout together determine the child's standard output.
106 * If pipe_stdin is true, prog_stdin is ignored.
107 * If pipe_stdout is true, prog_stdout is ignored.
110 create_pipe (const char *progname
,
111 const char *prog_path
, char **prog_argv
,
112 bool pipe_stdin
, bool pipe_stdout
,
113 const char *prog_stdin
, const char *prog_stdout
,
115 bool slave_process
, bool exit_on_error
,
118 #if (defined _WIN32 && ! defined __CYGWIN__) || defined __KLIBC__
120 /* Native Windows API.
121 This uses _pipe(), dup2(), and _spawnv(). It could also be implemented
122 using the low-level functions CreatePipe(), DuplicateHandle(),
123 CreateProcess() and _open_osfhandle(); see the GNU make and GNU clisp
124 and cvs source code. */
136 /* FIXME: Need to free memory allocated by prepare_spawn. */
137 prog_argv
= prepare_spawn (prog_argv
);
140 if (pipe2_safer (ifd
, O_BINARY
| O_CLOEXEC
) < 0)
141 error (EXIT_FAILURE
, errno
, _("cannot create pipe"));
143 if (pipe2_safer (ofd
, O_BINARY
| O_CLOEXEC
) < 0)
144 error (EXIT_FAILURE
, errno
, _("cannot create pipe"));
145 /* Data flow diagram:
148 * parent -> ofd[1] -> ofd[0] -> child if pipe_stdin
149 * parent <- ifd[0] <- ifd[1] <- child if pipe_stdout
154 /* Save standard file handles of parent process. */
155 if (pipe_stdin
|| prog_stdin
!= NULL
)
156 orig_stdin
= dup_safer_noinherit (STDIN_FILENO
);
157 if (pipe_stdout
|| prog_stdout
!= NULL
)
158 orig_stdout
= dup_safer_noinherit (STDOUT_FILENO
);
160 orig_stderr
= dup_safer_noinherit (STDERR_FILENO
);
163 /* Create standard file handles of child process. */
167 if ((!pipe_stdin
|| dup2 (ofd
[0], STDIN_FILENO
) >= 0)
168 && (!pipe_stdout
|| dup2 (ifd
[1], STDOUT_FILENO
) >= 0)
170 || ((nulloutfd
= open ("NUL", O_RDWR
, 0)) >= 0
171 && (nulloutfd
== STDERR_FILENO
172 || (dup2 (nulloutfd
, STDERR_FILENO
) >= 0
173 && close (nulloutfd
) >= 0))))
175 || prog_stdin
== NULL
176 || ((stdinfd
= open (prog_stdin
, O_RDONLY
, 0)) >= 0
177 && (stdinfd
== STDIN_FILENO
178 || (dup2 (stdinfd
, STDIN_FILENO
) >= 0
179 && close (stdinfd
) >= 0))))
181 || prog_stdout
== NULL
182 || ((stdoutfd
= open (prog_stdout
, O_WRONLY
, 0)) >= 0
183 && (stdoutfd
== STDOUT_FILENO
184 || (dup2 (stdoutfd
, STDOUT_FILENO
) >= 0
185 && close (stdoutfd
) >= 0)))))
186 /* The child process doesn't inherit ifd[0], ifd[1], ofd[0], ofd[1],
187 but it inherits all open()ed or dup2()ed file handles (which is what
188 we want in the case of STD*_FILENO). */
189 /* Use _spawnvpe and pass the environment explicitly. This is needed if
190 the program has modified the environment using putenv() or [un]setenv().
191 On Windows, programs have two environments, one in the "environment
192 block" of the process and managed through SetEnvironmentVariable(), and
193 one inside the process, in the location retrieved by the 'environ'
194 macro. When using _spawnvp() without 'e', the child process inherits a
195 copy of the environment block - ignoring the effects of putenv() and
198 child
= _spawnvpe (P_NOWAIT
, prog_path
, (const char **) prog_argv
,
199 (const char **) environ
);
200 if (child
< 0 && errno
== ENOEXEC
)
202 /* prog is not a native executable. Try to execute it as a
203 shell script. Note that prepare_spawn() has already prepended
204 a hidden element "sh.exe" to prog_argv. */
206 child
= _spawnvpe (P_NOWAIT
, prog_argv
[0], (const char **) prog_argv
,
207 (const char **) environ
);
219 /* Restore standard file handles of parent process. */
221 undup_safer_noinherit (orig_stderr
, STDERR_FILENO
);
222 if (pipe_stdout
|| prog_stdout
!= NULL
)
223 undup_safer_noinherit (orig_stdout
, STDOUT_FILENO
);
224 if (pipe_stdin
|| prog_stdin
!= NULL
)
225 undup_safer_noinherit (orig_stdin
, STDIN_FILENO
);
233 if (exit_on_error
|| !null_stderr
)
234 error (exit_on_error
? EXIT_FAILURE
: 0, saved_errno
,
235 _("%s subprocess failed"), progname
);
255 sigset_t blocked_signals
;
256 posix_spawn_file_actions_t actions
;
257 bool actions_allocated
;
258 posix_spawnattr_t attrs
;
259 bool attrs_allocated
;
264 if (pipe_safer (ifd
) < 0)
265 error (EXIT_FAILURE
, errno
, _("cannot create pipe"));
267 if (pipe_safer (ofd
) < 0)
268 error (EXIT_FAILURE
, errno
, _("cannot create pipe"));
269 /* Data flow diagram:
272 * parent -> ofd[1] -> ofd[0] -> child if pipe_stdin
273 * parent <- ifd[0] <- ifd[1] <- child if pipe_stdout
280 sigprocmask (SIG_SETMASK
, NULL
, &blocked_signals
);
281 block_fatal_signals ();
283 actions_allocated
= false;
284 attrs_allocated
= false;
285 if ((err
= posix_spawn_file_actions_init (&actions
)) != 0
286 || (actions_allocated
= true,
288 && (err
= posix_spawn_file_actions_adddup2 (&actions
,
289 ofd
[0], STDIN_FILENO
))
292 && (err
= posix_spawn_file_actions_adddup2 (&actions
,
293 ifd
[1], STDOUT_FILENO
))
296 && (err
= posix_spawn_file_actions_addclose (&actions
, ofd
[0]))
299 && (err
= posix_spawn_file_actions_addclose (&actions
, ifd
[1]))
302 && (err
= posix_spawn_file_actions_addclose (&actions
, ofd
[1]))
305 && (err
= posix_spawn_file_actions_addclose (&actions
, ifd
[0]))
308 && (err
= posix_spawn_file_actions_addopen (&actions
,
314 && prog_stdin
!= NULL
315 && (err
= posix_spawn_file_actions_addopen (&actions
,
317 prog_stdin
, O_RDONLY
,
321 && prog_stdout
!= NULL
322 && (err
= posix_spawn_file_actions_addopen (&actions
,
324 prog_stdout
, O_WRONLY
,
328 && ((err
= posix_spawnattr_init (&attrs
)) != 0
329 || (attrs_allocated
= true,
330 (err
= posix_spawnattr_setsigmask (&attrs
,
333 || (err
= posix_spawnattr_setflags (&attrs
,
334 POSIX_SPAWN_SETSIGMASK
))
336 || (err
= posix_spawnp (&child
, prog_path
, &actions
,
337 attrs_allocated
? &attrs
: NULL
, prog_argv
,
341 if (actions_allocated
)
342 posix_spawn_file_actions_destroy (&actions
);
344 posix_spawnattr_destroy (&attrs
);
346 unblock_fatal_signals ();
347 if (exit_on_error
|| !null_stderr
)
348 error (exit_on_error
? EXIT_FAILURE
: 0, err
,
349 _("%s subprocess failed"), progname
);
363 posix_spawn_file_actions_destroy (&actions
);
365 posix_spawnattr_destroy (&attrs
);
368 register_slave_subprocess (child
);
369 unblock_fatal_signals ();
385 /* Open a bidirectional pipe.
388 * parent -> fd[1] -> STDIN_FILENO -> child
389 * parent <- fd[0] <- STDOUT_FILENO <- child
394 create_pipe_bidi (const char *progname
,
395 const char *prog_path
, char **prog_argv
,
397 bool slave_process
, bool exit_on_error
,
400 pid_t result
= create_pipe (progname
, prog_path
, prog_argv
,
401 true, true, NULL
, NULL
,
402 null_stderr
, slave_process
, exit_on_error
,
407 /* Open a pipe for input from a child process.
408 * The child's stdin comes from a file.
411 * parent <- fd[0] <- STDOUT_FILENO <- child
415 create_pipe_in (const char *progname
,
416 const char *prog_path
, char **prog_argv
,
417 const char *prog_stdin
, bool null_stderr
,
418 bool slave_process
, bool exit_on_error
,
422 pid_t result
= create_pipe (progname
, prog_path
, prog_argv
,
423 false, true, prog_stdin
, NULL
,
424 null_stderr
, slave_process
, exit_on_error
,
431 /* Open a pipe for output to a child process.
432 * The child's stdout goes to a file.
435 * parent -> fd[0] -> STDIN_FILENO -> child
439 create_pipe_out (const char *progname
,
440 const char *prog_path
, char **prog_argv
,
441 const char *prog_stdout
, bool null_stderr
,
442 bool slave_process
, bool exit_on_error
,
446 pid_t result
= create_pipe (progname
, prog_path
, prog_argv
,
447 true, false, NULL
, prog_stdout
,
448 null_stderr
, slave_process
, exit_on_error
,