我有以下函数通过fork和execvp执行命令。我在fork中启动的脚本正在侦听输入数据。我如何将数据发送到myscript?
int external_command()
{
int pfds[2];
if (pipe(pfds) < 0)
return -1;
if ((uproc.pid = fork()) == -1)
return -1;
if (uproc.pid == 0) {
/* child */
const char *argv[4];
int i = 0;
argv[i++] = "/bin/sh";
argv[i++] = "myscript.sh";
argv[i++] = NULL;
close(pfds[0]);
dup2(pfds[1], 1);
close(pfds[1]);
execvp(argv[0], (char **) argv);
exit(ESRCH);
} else if (uproc.pid < 0)
return -1;
/* parent */
close(pfds[1]);
int status;
while (wait(&status) != uproc.pid) {
DD("waiting for child to exit");
}
char buffer[64];
ssize_t rxed;
char *c;
int t;
//read from fork pipe
*value = NULL;
while ((rxed = read(pfds[0], buffer, sizeof(buffer))) > 0) {
if (*value)
t = asprintf(&c, "%s%.*s", *value, (int) rxed, buffer);
else
t = asprintf(&c, "%.*s", (int) rxed, buffer);
if (t == -1) return -1;
free(*value);
*value = strdup(c);
free(c);
}
// how to write to the pipe fork?
}
答案 0 :(得分:2)
管道是单向的。调用pipe
两次,得到一对文件描述符,用于从孩子那里读取数据(正如你已经拥有的那样),另一对用于向孩子发送数据(这将是新的)。你需要做你正在做的同样的dup2
和close
魔法,以便将新管道设置为孩子的stdin。
答案 1 :(得分:2)
我猜你的意思是你想从父进程写入子进程stdin
?为此,您需要创建两个管道。在孩子中使用stdout
的一个,就像你现在一样,另一个必须用于stdin
,这与你现在的方式大致相同(但当然,索引相反)
当然,在你写这个孩子之前你不能wait
,因为如果孩子需要输入继续,你可能会陷入僵局。
答案 2 :(得分:0)
管道的基础
请研究这段代码,以便了解自己在做什么
int fd[2]; /*write(fd[1],buffer,strlen)
/ read(fd[0],buffer2,SIZE)*/
pid_t cpid;
if(pipe(fd)==-1){
perror("pipe");
exit(EXIT_FAILURE);
}
if((cpid=fork())<0){/* FORK INIT */
printf("\n\tFORK ERROR\n");
exit(1);
}
if(cpid==0){ /*SON*/
close(fd[0]);
/********CODE******/
if((write(fd[1],final2,strlen(final2)))<0){
perror("\n\tWRITE ERROR");
}
close(fd[1]);
}else{ /*FATHER*/
close(fd[1]);
if((read(fd[0],aler,NN))<0){
perror("\n\tREAD ERROR");
}
wait(NULL);
/********CODE******/
close(fd[0]);
}
在这种情况下父进程(父)读取来自新(子)进程的信息
如果你想要双向 read()write()创建2个管道(fd1 [2],fd2 [2])
使用与上面相同的逻辑,父读,关闭写端fd1 [1],子写,关闭读端fd1 [0]
反之亦然
父亲写道,关闭读取结束fd2 [0],儿子读取,关闭写入结束fd2 [1]