1 /* Utilities to execute a program in a subprocess (possibly linked by pipes
2 with other subprocesses), and wait for it. Generic Unix version
3 (also used for UWIN and VMS).
4 Copyright (C) 1996-2025 Free Software Foundation, Inc.
6 This file is part of the libiberty library.
7 Libiberty is free software; you can redistribute it and/or
8 modify it under the terms of the GNU Library General Public
9 License as published by the Free Software Foundation; either
10 version 2 of the License, or (at your option) any later version.
12 Libiberty is distributed in the hope that it will be useful,
13 but WITHOUT ANY WARRANTY; without even the implied warranty of
14 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
15 Library General Public License for more details.
17 You should have received a copy of the GNU Library General Public
18 License along with libiberty; see the file COPYING.LIB. If not,
19 write to the Free Software Foundation, Inc., 51 Franklin Street - Fifth Floor,
20 Boston, MA 02110-1301, USA. */
23 #include "libiberty.h"
24 #include "pex-common.h"
30 #ifdef NEED_DECLARATION_ERRNO
43 #include <sys/types.h>
48 #ifdef HAVE_SYS_WAIT_H
53 #include <sys/resource.h>
55 #ifdef HAVE_SYS_STAT_H
65 #ifdef vfork /* Autoconf may define this to fork for us. */
66 # define VFORK_STRING "fork"
68 # define VFORK_STRING "vfork"
73 #if defined(VMS) && defined (__LONG_POINTERS)
75 typedef char * __char_ptr32
76 __attribute__ ((mode (SI
)));
79 typedef __char_ptr32
*__char_ptr_char_ptr32
80 __attribute__ ((mode (SI
)));
82 /* Return a 32 bit pointer to an array of 32 bit pointers
83 given a 64 bit pointer to an array of 64 bit pointers. */
85 static __char_ptr_char_ptr32
86 to_ptr32 (char **ptr64
)
89 __char_ptr_char_ptr32 short_argv
;
91 /* Count number of arguments. */
92 for (argc
= 0; ptr64
[argc
] != NULL
; argc
++)
95 /* Reallocate argv with 32 bit pointers. */
96 short_argv
= (__char_ptr_char_ptr32
) decc$malloc
97 (sizeof (__char_ptr32
) * (argc
+ 1));
99 for (argc
= 0; ptr64
[argc
] != NULL
; argc
++)
100 short_argv
[argc
] = (__char_ptr32
) decc$
strdup (ptr64
[argc
]);
102 short_argv
[argc
] = (__char_ptr32
) 0;
107 #define to_ptr32(argv) argv
110 /* File mode to use for private and world-readable files. */
112 #if defined (S_IRUSR) && defined (S_IWUSR) && defined (S_IRGRP) && defined (S_IWGRP) && defined (S_IROTH) && defined (S_IWOTH)
113 #define PUBLIC_MODE \
114 (S_IRUSR | S_IWUSR | S_IRGRP | S_IWGRP | S_IROTH | S_IWOTH)
116 #define PUBLIC_MODE 0666
119 /* Get the exit status of a particular process, and optionally get the
120 time that it took. This is simple if we have wait4, slightly
121 harder if we have waitpid, and is a pain if we only have wait. */
123 static pid_t
pex_wait (struct pex_obj
*, pid_t
, int *, struct pex_time
*);
128 pex_wait (struct pex_obj
*obj ATTRIBUTE_UNUSED
, pid_t pid
, int *status
,
129 struct pex_time
*time
)
136 return waitpid (pid
, status
, 0);
139 ret
= wait4 (pid
, status
, 0, &r
);
143 time
->user_seconds
= r
.ru_utime
.tv_sec
;
144 time
->user_microseconds
= r
.ru_utime
.tv_usec
;
145 time
->system_seconds
= r
.ru_stime
.tv_sec
;
146 time
->system_microseconds
= r
.ru_stime
.tv_usec
;
152 #else /* ! defined (HAVE_WAIT4) */
156 #ifndef HAVE_GETRUSAGE
159 pex_wait (struct pex_obj
*obj ATTRIBUTE_UNUSED
, pid_t pid
, int *status
,
160 struct pex_time
*time
)
163 memset (time
, 0, sizeof (struct pex_time
));
164 return waitpid (pid
, status
, 0);
167 #else /* defined (HAVE_GETRUSAGE) */
170 pex_wait (struct pex_obj
*obj ATTRIBUTE_UNUSED
, pid_t pid
, int *status
,
171 struct pex_time
*time
)
173 struct rusage r1
, r2
;
177 return waitpid (pid
, status
, 0);
179 getrusage (RUSAGE_CHILDREN
, &r1
);
181 ret
= waitpid (pid
, status
, 0);
185 getrusage (RUSAGE_CHILDREN
, &r2
);
187 time
->user_seconds
= r2
.ru_utime
.tv_sec
- r1
.ru_utime
.tv_sec
;
188 time
->user_microseconds
= r2
.ru_utime
.tv_usec
- r1
.ru_utime
.tv_usec
;
189 if (r2
.ru_utime
.tv_usec
< r1
.ru_utime
.tv_usec
)
191 --time
->user_seconds
;
192 time
->user_microseconds
+= 1000000;
195 time
->system_seconds
= r2
.ru_stime
.tv_sec
- r1
.ru_stime
.tv_sec
;
196 time
->system_microseconds
= r2
.ru_stime
.tv_usec
- r1
.ru_stime
.tv_usec
;
197 if (r2
.ru_stime
.tv_usec
< r1
.ru_stime
.tv_usec
)
199 --time
->system_seconds
;
200 time
->system_microseconds
+= 1000000;
206 #endif /* defined (HAVE_GETRUSAGE) */
208 #else /* ! defined (HAVE_WAITPID) */
212 struct status_list
*next
;
215 struct pex_time time
;
219 pex_wait (struct pex_obj
*obj
, pid_t pid
, int *status
, struct pex_time
*time
)
221 struct status_list
**pp
;
223 for (pp
= (struct status_list
**) &obj
->sysdep
;
227 if ((*pp
)->pid
== pid
)
229 struct status_list
*p
;
244 struct status_list
*psl
;
246 #ifdef HAVE_GETRUSAGE
247 struct rusage r1
, r2
;
252 #ifdef HAVE_GETRUSAGE
253 getrusage (RUSAGE_CHILDREN
, &r1
);
255 memset (&pt
, 0, sizeof (struct pex_time
));
259 cpid
= wait (status
);
261 #ifdef HAVE_GETRUSAGE
262 if (time
!= NULL
&& cpid
>= 0)
264 getrusage (RUSAGE_CHILDREN
, &r2
);
266 pt
.user_seconds
= r2
.ru_utime
.tv_sec
- r1
.ru_utime
.tv_sec
;
267 pt
.user_microseconds
= r2
.ru_utime
.tv_usec
- r1
.ru_utime
.tv_usec
;
268 if (pt
.user_microseconds
< 0)
271 pt
.user_microseconds
+= 1000000;
274 pt
.system_seconds
= r2
.ru_stime
.tv_sec
- r1
.ru_stime
.tv_sec
;
275 pt
.system_microseconds
= r2
.ru_stime
.tv_usec
- r1
.ru_stime
.tv_usec
;
276 if (pt
.system_microseconds
< 0)
279 pt
.system_microseconds
+= 1000000;
284 if (cpid
< 0 || cpid
== pid
)
291 psl
= XNEW (struct status_list
);
293 psl
->status
= *status
;
296 psl
->next
= (struct status_list
*) obj
->sysdep
;
297 obj
->sysdep
= (void *) psl
;
301 #endif /* ! defined (HAVE_WAITPID) */
302 #endif /* ! defined (HAVE_WAIT4) */
304 static int pex_unix_open_read (struct pex_obj
*, const char *, int);
305 static int pex_unix_open_write (struct pex_obj
*, const char *, int, int);
306 static pid_t
pex_unix_exec_child (struct pex_obj
*, int, const char *,
307 char * const *, char * const *,
309 const char **, int *);
310 static int pex_unix_close (struct pex_obj
*, int);
311 static pid_t
pex_unix_wait (struct pex_obj
*, pid_t
, int *, struct pex_time
*,
312 int, const char **, int *);
313 static int pex_unix_pipe (struct pex_obj
*, int *, int);
314 static FILE *pex_unix_fdopenr (struct pex_obj
*, int, int);
315 static FILE *pex_unix_fdopenw (struct pex_obj
*, int, int);
316 static void pex_unix_cleanup (struct pex_obj
*);
318 /* The list of functions we pass to the common routines. */
320 const struct pex_funcs funcs
=
333 /* Return a newly initialized pex_obj structure. */
336 pex_init (int flags
, const char *pname
, const char *tempbase
)
338 return pex_init_common (flags
, pname
, tempbase
, &funcs
);
341 /* Open a file for reading. */
344 pex_unix_open_read (struct pex_obj
*obj ATTRIBUTE_UNUSED
, const char *name
,
345 int binary ATTRIBUTE_UNUSED
)
347 return open (name
, O_RDONLY
);
350 /* Open a file for writing. */
353 pex_unix_open_write (struct pex_obj
*obj ATTRIBUTE_UNUSED
, const char *name
,
354 int binary ATTRIBUTE_UNUSED
, int append
)
356 /* Note that we can't use O_EXCL here because gcc may have already
357 created the temporary file via make_temp_file. */
358 return open (name
, O_WRONLY
| O_CREAT
359 | (append
? O_APPEND
: O_TRUNC
), PUBLIC_MODE
);
365 pex_unix_close (struct pex_obj
*obj ATTRIBUTE_UNUSED
, int fd
)
370 /* Execute a child. */
372 #if defined(HAVE_SPAWNVE) && defined(HAVE_SPAWNVPE)
373 /* Implementation of pex->exec_child using the Cygwin spawn operation. */
375 /* Subroutine of pex_unix_exec_child. Move OLD_FD to a new file descriptor
376 to be stored in *PNEW_FD, save the flags in *PFLAGS, and arrange for the
377 saved copy to be close-on-exec. Move CHILD_FD into OLD_FD. If CHILD_FD
378 is -1, OLD_FD is to be closed. Return -1 on error. */
381 save_and_install_fd(int *pnew_fd
, int *pflags
, int old_fd
, int child_fd
)
385 flags
= fcntl (old_fd
, F_GETFD
);
387 /* If we could not retrieve the flags, then OLD_FD was not open. */
390 new_fd
= -1, flags
= 0;
391 if (child_fd
>= 0 && dup2 (child_fd
, old_fd
) < 0)
394 /* If we wish to close OLD_FD, just mark it CLOEXEC. */
395 else if (child_fd
== -1)
398 if ((flags
& FD_CLOEXEC
) == 0 && fcntl (old_fd
, F_SETFD
, FD_CLOEXEC
) < 0)
401 /* Otherwise we need to save a copy of OLD_FD before installing CHILD_FD. */
404 #ifdef F_DUPFD_CLOEXEC
405 new_fd
= fcntl (old_fd
, F_DUPFD_CLOEXEC
, 3);
409 /* Prefer F_DUPFD over dup in order to avoid getting a new fd
410 in the range 0-2, right where a new stderr fd might get put. */
411 new_fd
= fcntl (old_fd
, F_DUPFD
, 3);
414 if (fcntl (new_fd
, F_SETFD
, FD_CLOEXEC
) < 0)
417 if (dup2 (child_fd
, old_fd
) < 0)
424 else if (new_fd
!= old_fd
)
430 /* Subroutine of pex_unix_exec_child. Move SAVE_FD back to OLD_FD
431 restoring FLAGS. If SAVE_FD < 0, OLD_FD is to be closed. */
434 restore_fd(int old_fd
, int save_fd
, int flags
)
436 /* For SAVE_FD < 0, all we have to do is restore the
437 "closed-ness" of the original. */
439 return close (old_fd
);
441 /* For SAVE_FD == OLD_FD, all we have to do is restore the
442 original setting of the CLOEXEC flag. */
443 if (save_fd
== old_fd
)
445 if (flags
& FD_CLOEXEC
)
447 return fcntl (old_fd
, F_SETFD
, flags
);
450 /* Otherwise we have to move the descriptor back, restore the flags,
451 and close the saved copy. */
453 if (flags
== FD_CLOEXEC
)
455 if (dup3 (save_fd
, old_fd
, O_CLOEXEC
) < 0)
461 if (dup2 (save_fd
, old_fd
) < 0)
463 if (flags
!= 0 && fcntl (old_fd
, F_SETFD
, flags
) < 0)
466 return close (save_fd
);
470 pex_unix_exec_child (struct pex_obj
*obj ATTRIBUTE_UNUSED
,
471 int flags
, const char *executable
,
472 char * const * argv
, char * const * env
,
473 int in
, int out
, int errdes
, int toclose
,
474 const char **errmsg
, int *err
)
476 int fl_in
= 0, fl_out
= 0, fl_err
= 0, fl_tc
= 0;
477 int save_in
= -1, save_out
= -1, save_err
= -1;
481 if (flags
& PEX_STDERR_TO_STDOUT
)
484 /* We need the three standard file descriptors to be set up as for
485 the child before we perform the spawn. The file descriptors for
486 the parent need to be moved and marked for close-on-exec. */
487 if (in
!= STDIN_FILE_NO
488 && save_and_install_fd (&save_in
, &fl_in
, STDIN_FILE_NO
, in
) < 0)
490 if (out
!= STDOUT_FILE_NO
491 && save_and_install_fd (&save_out
, &fl_out
, STDOUT_FILE_NO
, out
) < 0)
493 if (errdes
!= STDERR_FILE_NO
494 && save_and_install_fd (&save_err
, &fl_err
, STDERR_FILE_NO
, errdes
) < 0)
497 && save_and_install_fd (NULL
, &fl_tc
, toclose
, -1) < 0)
500 /* Now that we've moved the file descriptors for the child into place,
501 close the originals. Be careful not to close any of the standard
502 file descriptors that we just set up. */
505 max
= STDERR_FILE_NO
;
507 max
= STDOUT_FILE_NO
;
514 if (errdes
> max
&& errdes
!= out
)
517 /* If we were not given an environment, use the global environment. */
521 /* Launch the program. If we get EAGAIN (normally out of pid's), try
522 again a few times with increasing backoff times. */
526 typedef const char * const *cc_cp
;
528 if (flags
& PEX_SEARCH
)
529 pid
= spawnvpe (_P_NOWAITO
, executable
, (cc_cp
)argv
, (cc_cp
)env
);
531 pid
= spawnve (_P_NOWAITO
, executable
, (cc_cp
)argv
, (cc_cp
)env
);
538 if (errno
!= EAGAIN
|| ++retries
== 4)
540 sleep (1 << retries
);
543 /* Success. Restore the parent's file descriptors that we saved above. */
545 && restore_fd (toclose
, toclose
, fl_tc
) < 0)
547 if (in
!= STDIN_FILE_NO
548 && restore_fd (STDIN_FILE_NO
, save_in
, fl_in
) < 0)
550 if (out
!= STDOUT_FILE_NO
551 && restore_fd (STDOUT_FILE_NO
, save_out
, fl_out
) < 0)
553 if (errdes
!= STDERR_FILE_NO
554 && restore_fd (STDERR_FILE_NO
, save_err
, fl_err
) < 0)
565 #elif defined(HAVE_POSIX_SPAWN) && defined(HAVE_POSIX_SPAWNP)
566 /* Implementation of pex->exec_child using posix_spawn. */
569 pex_unix_exec_child (struct pex_obj
*obj ATTRIBUTE_UNUSED
,
570 int flags
, const char *executable
,
571 char * const * argv
, char * const * env
,
572 int in
, int out
, int errdes
,
573 int toclose
, const char **errmsg
, int *err
)
577 posix_spawnattr_t attr
;
578 posix_spawn_file_actions_t actions
;
579 int attr_initialized
= 0, actions_initialized
= 0;
583 ret
= posix_spawnattr_init (&attr
);
587 *errmsg
= "posix_spawnattr_init";
590 attr_initialized
= 1;
592 /* Use vfork() on glibc <=2.24. */
593 #ifdef POSIX_SPAWN_USEVFORK
594 ret
= posix_spawnattr_setflags (&attr
, POSIX_SPAWN_USEVFORK
);
598 *errmsg
= "posix_spawnattr_setflags";
603 ret
= posix_spawn_file_actions_init (&actions
);
607 *errmsg
= "posix_spawn_file_actions_init";
610 actions_initialized
= 1;
612 if (in
!= STDIN_FILE_NO
)
614 ret
= posix_spawn_file_actions_adddup2 (&actions
, in
, STDIN_FILE_NO
);
618 *errmsg
= "posix_spawn_file_actions_adddup2";
622 ret
= posix_spawn_file_actions_addclose (&actions
, in
);
626 *errmsg
= "posix_spawn_file_actions_addclose";
631 if (out
!= STDOUT_FILE_NO
)
633 ret
= posix_spawn_file_actions_adddup2 (&actions
, out
, STDOUT_FILE_NO
);
637 *errmsg
= "posix_spawn_file_actions_adddup2";
641 ret
= posix_spawn_file_actions_addclose (&actions
, out
);
645 *errmsg
= "posix_spawn_file_actions_addclose";
650 if (errdes
!= STDERR_FILE_NO
)
652 ret
= posix_spawn_file_actions_adddup2 (&actions
, errdes
, STDERR_FILE_NO
);
656 *errmsg
= "posix_spawn_file_actions_adddup2";
660 ret
= posix_spawn_file_actions_addclose (&actions
, errdes
);
664 *errmsg
= "posix_spawn_file_actions_addclose";
671 ret
= posix_spawn_file_actions_addclose (&actions
, toclose
);
675 *errmsg
= "posix_spawn_file_actions_addclose";
680 if ((flags
& PEX_STDERR_TO_STDOUT
) != 0)
682 ret
= posix_spawn_file_actions_adddup2 (&actions
, STDOUT_FILE_NO
, STDERR_FILE_NO
);
686 *errmsg
= "posix_spawn_file_actions_adddup2";
691 if ((flags
& PEX_SEARCH
) != 0)
693 ret
= posix_spawnp (&pid
, executable
, &actions
, &attr
, argv
, env
? env
: environ
);
697 *errmsg
= "posix_spawnp";
698 pid
= -1; /* The value of pid is unspecified on failure. */
704 ret
= posix_spawn (&pid
, executable
, &actions
, &attr
, argv
, env
? env
: environ
);
708 *errmsg
= "posix_spawn";
715 if (actions_initialized
)
716 posix_spawn_file_actions_destroy (&actions
);
717 if (attr_initialized
)
718 posix_spawnattr_destroy (&attr
);
720 if (!*err
&& in
!= STDIN_FILE_NO
)
722 *errmsg
= "close", *err
= errno
, pid
= -1;
723 if (!*err
&& out
!= STDOUT_FILE_NO
)
725 *errmsg
= "close", *err
= errno
, pid
= -1;
726 if (!*err
&& errdes
!= STDERR_FILE_NO
)
728 *errmsg
= "close", *err
= errno
, pid
= -1;
733 /* Implementation of pex->exec_child using standard vfork + exec. */
736 pex_unix_exec_child (struct pex_obj
*obj
, int flags
, const char *executable
,
737 char * const * argv
, char * const * env
,
738 int in
, int out
, int errdes
,
739 int toclose
, const char **errmsg
, int *err
)
742 /* Tuple to communicate error from child to parent. We can safely
743 transfer string literal pointers as both run with identical
750 volatile int do_pipe
= 0;
751 volatile int pipes
[2]; /* [0]:reader,[1]:writer. */
758 if (pipe2 ((int *)pipes
, O_CLOEXEC
))
761 if (pipe ((int *)pipes
))
765 if (fcntl (pipes
[1], F_SETFD
, FD_CLOEXEC
) == -1)
775 /* We declare these to be volatile to avoid warnings from gcc about
776 them being clobbered by vfork. */
777 volatile int sleep_interval
= 1;
778 volatile int retries
;
780 /* We vfork and then set environ in the child before calling execvp.
781 This clobbers the parent's environ so we need to restore it.
782 It would be nice to use one of the exec* functions that takes an
783 environment as a parameter, but that may have portability
784 issues. It is marked volatile so the child doesn't consider it a
785 dead variable and therefore clobber where ever it is stored. */
786 char **volatile save_environ
= environ
;
788 for (retries
= 0; retries
< 4; ++retries
)
793 sleep (sleep_interval
);
806 *errmsg
= VFORK_STRING
;
812 struct fn_err failed
;
817 if (!failed
.fn
&& in
!= STDIN_FILE_NO
)
819 if (dup2 (in
, STDIN_FILE_NO
) < 0)
820 failed
.fn
= "dup2", failed
.err
= errno
;
821 else if (close (in
) < 0)
822 failed
.fn
= "close", failed
.err
= errno
;
824 if (!failed
.fn
&& out
!= STDOUT_FILE_NO
)
826 if (dup2 (out
, STDOUT_FILE_NO
) < 0)
827 failed
.fn
= "dup2", failed
.err
= errno
;
828 else if (close (out
) < 0)
829 failed
.fn
= "close", failed
.err
= errno
;
831 if (!failed
.fn
&& errdes
!= STDERR_FILE_NO
)
833 if (dup2 (errdes
, STDERR_FILE_NO
) < 0)
834 failed
.fn
= "dup2", failed
.err
= errno
;
835 else if (close (errdes
) < 0)
836 failed
.fn
= "close", failed
.err
= errno
;
838 if (!failed
.fn
&& toclose
>= 0)
840 if (close (toclose
) < 0)
841 failed
.fn
= "close", failed
.err
= errno
;
843 if (!failed
.fn
&& (flags
& PEX_STDERR_TO_STDOUT
) != 0)
845 if (dup2 (STDOUT_FILE_NO
, STDERR_FILE_NO
) < 0)
846 failed
.fn
= "dup2", failed
.err
= errno
;
851 /* NOTE: In a standard vfork implementation this clobbers
852 the parent's copy of environ "too" (in reality there's
853 only one copy). This is ok as we restore it below. */
854 environ
= (char**) env
;
855 if ((flags
& PEX_SEARCH
) != 0)
857 execvp (executable
, to_ptr32 (argv
));
858 failed
.fn
= "execvp", failed
.err
= errno
;
862 execv (executable
, to_ptr32 (argv
));
863 failed
.fn
= "execv", failed
.err
= errno
;
867 /* Something failed, report an error. We don't use stdio
868 routines, because we might be here due to a vfork call. */
872 || write (pipes
[1], &failed
, sizeof (failed
)) != sizeof (failed
))
874 /* The parent will not see our scream above, so write to
876 #define writeerr(s) (retval |= write (STDERR_FILE_NO, s, strlen (s)))
877 writeerr (obj
->pname
);
878 writeerr (": error trying to exec '");
879 writeerr (executable
);
881 writeerr (failed
.fn
);
883 writeerr (xstrerror (failed
.err
));
888 /* Exit with -2 if the error output failed, too. */
889 _exit (retval
< 0 ? -2 : -1);
895 /* Parent process. */
897 /* Restore environ. Note that the parent either doesn't run
898 until the child execs/exits (standard vfork behaviour), or
899 if it does run then vfork is behaving more like fork. In
900 either case we needn't worry about clobbering the child's
902 environ
= save_environ
;
904 struct fn_err failed
;
909 ssize_t len
= read (pipes
[0], &failed
, sizeof (failed
));
915 if (!failed
.fn
&& in
!= STDIN_FILE_NO
)
917 failed
.fn
= "close", failed
.err
= errno
;
918 if (!failed
.fn
&& out
!= STDOUT_FILE_NO
)
920 failed
.fn
= "close", failed
.err
= errno
;
921 if (!failed
.fn
&& errdes
!= STDERR_FILE_NO
)
922 if (close (errdes
) < 0)
923 failed
.fn
= "close", failed
.err
= errno
;
937 /* Wait for a child process to complete. */
940 pex_unix_wait (struct pex_obj
*obj
, pid_t pid
, int *status
,
941 struct pex_time
*time
, int done
, const char **errmsg
,
944 /* If we are cleaning up when the caller didn't retrieve process
945 status for some reason, encourage the process to go away. */
949 if (pex_wait (obj
, pid
, status
, time
) < 0)
962 pex_unix_pipe (struct pex_obj
*obj ATTRIBUTE_UNUSED
, int *p
,
963 int binary ATTRIBUTE_UNUSED
)
968 /* Get a FILE pointer to read from a file descriptor. */
971 pex_unix_fdopenr (struct pex_obj
*obj ATTRIBUTE_UNUSED
, int fd
,
972 int binary ATTRIBUTE_UNUSED
)
974 return fdopen (fd
, "r");
978 pex_unix_fdopenw (struct pex_obj
*obj ATTRIBUTE_UNUSED
, int fd
,
979 int binary ATTRIBUTE_UNUSED
)
981 if (fcntl (fd
, F_SETFD
, FD_CLOEXEC
) < 0)
983 return fdopen (fd
, "w");
987 pex_unix_cleanup (struct pex_obj
*obj ATTRIBUTE_UNUSED
)
989 #if !defined (HAVE_WAIT4) && !defined (HAVE_WAITPID)
990 while (obj
->sysdep
!= NULL
)
992 struct status_list
*this;
993 struct status_list
*next
;
995 this = (struct status_list
*) obj
->sysdep
;
998 obj
->sysdep
= (void *) next
;