fix popen not to leak pipes from one child to another
[musl] / src / stdio / popen.c
index ca3cdf9..3ec8339 100644 (file)
@@ -1,26 +1,24 @@
 #include <fcntl.h>
+#include <unistd.h>
+#include <errno.h>
+#include <string.h>
+#include <spawn.h>
 #include "stdio_impl.h"
-#include "pthread_impl.h"
 #include "syscall.h"
 
-static void dummy_0()
-{
-}
-weak_alias(dummy_0, __acquire_ptc);
-weak_alias(dummy_0, __release_ptc);
-
-pid_t __vfork(void);
+extern char **__environ;
 
 FILE *popen(const char *cmd, const char *mode)
 {
-       int p[2], op, i;
+       int p[2], op, e;
        pid_t pid;
        FILE *f;
-       sigset_t old;
-       const char *modes = "rw", *mi = strchr(modes, *mode);
+       posix_spawn_file_actions_t fa;
 
-       if (mi) {
-               op = mi-modes;
+       if (*mode == 'r') {
+               op = 0;
+       } else if (*mode == 'w') {
+               op = 1;
        } else {
                errno = EINVAL;
                return 0;
@@ -34,37 +32,30 @@ FILE *popen(const char *cmd, const char *mode)
                return NULL;
        }
 
-       sigprocmask(SIG_BLOCK, SIGALL_SET, &old);
-       
-       __acquire_ptc();
-       pid = __vfork();
-
-       if (pid) {
-               __release_ptc();
-               __syscall(SYS_close, p[1-op]);
-               sigprocmask(SIG_SETMASK, &old, 0);
-               if (pid < 0) {
-                       fclose(f);
-                       return 0;
+       e = ENOMEM;
+       if (!posix_spawn_file_actions_init(&fa)) {
+               for (FILE *l = *__ofl_lock(); l; l=l->next)
+                       if (l->pipe_pid && posix_spawn_file_actions_addclose(&fa, l->fd))
+                               goto fail;
+               if (!posix_spawn_file_actions_adddup2(&fa, p[1-op], 1-op)) {
+                       if (!(e = posix_spawn(&pid, "/bin/sh", &fa, 0,
+                           (char *[]){ "sh", "-c", (char *)cmd, 0 }, __environ))) {
+                               posix_spawn_file_actions_destroy(&fa);
+                               f->pipe_pid = pid;
+                               if (!strchr(mode, 'e'))
+                                       fcntl(p[op], F_SETFD, 0);
+                               __syscall(SYS_close, p[1-op]);
+                               __ofl_unlock();
+                               return f;
+                       }
                }
-               f->pipe_pid = pid;
-               return f;
+fail:
+               __ofl_unlock();
+               posix_spawn_file_actions_destroy(&fa);
        }
+       fclose(f);
+       __syscall(SYS_close, p[1-op]);
 
-       /* See notes in system.c for why this is needed. */
-       for (i=1; i<=8*__SYSCALL_SSLEN; i++) {
-               struct sigaction sa;
-               __libc_sigaction(i, 0, &sa);
-               if (sa.sa_handler!=SIG_IGN && sa.sa_handler!=SIG_DFL) {
-                       sa.sa_handler = SIG_DFL;
-                       __libc_sigaction(i, &sa, 0);
-               }
-       }
-       if (dup2(p[1-op], 1-op) < 0) _exit(127);
-       fcntl(1-op, F_SETFD, 0);
-       if (p[0] != 1-op) __syscall(SYS_close, p[0]);
-       if (p[1] != 1-op) __syscall(SYS_close, p[1]);
-       sigprocmask(SIG_SETMASK, &old, 0);
-       execl("/bin/sh", "sh", "-c", cmd, (char *)0);
-       _exit(127);
+       errno = e;
+       return 0;
 }