summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorJP Abgrall <jpa@google.com>2011-09-17 16:33:55 -0700
committerAndroid Git Automerger <android-git-automerger@android.com>2011-09-17 16:33:55 -0700
commit26bee12899f8cc55e5195482b71024bee5b8eda8 (patch)
tree6280b32ab308ebd595a92e255b8fca670bb34906
parent370bbc5dea6d73a98ddea6a1c693354c086249ad (diff)
parent771600415f41b9075b83dcf1e5395d2292ea05b2 (diff)
downloadbionic-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.c25
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 */
}