diff options
author | JP Abgrall <jpa@google.com> | 2011-09-17 16:33:55 -0700 |
---|---|---|
committer | Android Git Automerger <android-git-automerger@android.com> | 2011-09-17 16:33:55 -0700 |
commit | 26bee12899f8cc55e5195482b71024bee5b8eda8 (patch) | |
tree | 6280b32ab308ebd595a92e255b8fca670bb34906 | |
parent | 370bbc5dea6d73a98ddea6a1c693354c086249ad (diff) | |
parent | 771600415f41b9075b83dcf1e5395d2292ea05b2 (diff) | |
download | bionic-26bee12899f8cc55e5195482b71024bee5b8eda8.zip bionic-26bee12899f8cc55e5195482b71024bee5b8eda8.tar.gz bionic-26bee12899f8cc55e5195482b71024bee5b8eda8.tar.bz2 |
am 77160041: am 3884bfe9: libc: popen: work around data corruption
* commit '771600415f41b9075b83dcf1e5395d2292ea05b2':
libc: popen: work around data corruption
-rw-r--r-- | libc/unistd/popen.c | 25 |
1 files changed, 11 insertions, 14 deletions
diff --git a/libc/unistd/popen.c b/libc/unistd/popen.c index 15f8325..c2355c1 100644 --- a/libc/unistd/popen.c +++ b/libc/unistd/popen.c @@ -48,6 +48,8 @@ static struct pid { pid_t pid; } *pidlist; +extern char **environ; + FILE * popen(const char *program, const char *type) { @@ -55,6 +57,7 @@ popen(const char *program, const char *type) FILE *iop; int pdes[2]; pid_t pid; + char *argp[] = {"sh", "-c", NULL, NULL}; if ((*type != 'r' && *type != 'w') || type[1] != '\0') { errno = EINVAL; @@ -69,7 +72,7 @@ popen(const char *program, const char *type) return (NULL); } - switch (pid = vfork()) { + switch (pid = fork()) { case -1: /* Error. */ (void)close(pdes[0]); (void)close(pdes[1]); @@ -80,24 +83,17 @@ popen(const char *program, const char *type) { struct pid *pcur; /* - * because vfork() instead of fork(), must leak FILE *, - * but luckily we are terminally headed for an execl() + * We fork()'d, we got our own copy of the list, no + * contention. */ for (pcur = pidlist; pcur; pcur = pcur->next) close(fileno(pcur->fp)); if (*type == 'r') { - int tpdes1 = pdes[1]; - (void) close(pdes[0]); - /* - * We must NOT modify pdes, due to the - * semantics of vfork. - */ - if (tpdes1 != STDOUT_FILENO) { - (void)dup2(tpdes1, STDOUT_FILENO); - (void)close(tpdes1); - tpdes1 = STDOUT_FILENO; + if (pdes[1] != STDOUT_FILENO) { + (void)dup2(pdes[1], STDOUT_FILENO); + (void)close(pdes[1]); } } else { (void)close(pdes[1]); @@ -106,7 +102,8 @@ popen(const char *program, const char *type) (void)close(pdes[0]); } } - execl(_PATH_BSHELL, "sh", "-c", program, (char *)NULL); + argp[2] = (char *)program; + execve(_PATH_BSHELL, argp, environ); _exit(127); /* NOTREACHED */ } |