也许这不是一个紧凑的标题,我很抱歉:)。我尝试使用管道将子进程的stdin / stdout重定向到其父进程。子进程从父进程输入执行系统命令,并使用管道将exec结果返回给父进程。在这里,我实施了#34; cat -n"和" tr / a-z / / A-Z /",前者工作正常,但后来没有任何结果。是什么导致了这个?谢谢。
#include <sys/types.h>
#include <sys/stat.h>
#include <sys/ipc.h>
#include <sys/shm.h>
#include <unistd.h>
#include <fcntl.h>
#include <stdio.h>
#include <stdlib.h>
#include <errno.h>
#include <string.h>
#include <signal.h>
#include <assert.h>
#include <sys/sem.h>
#define ERR_EXIT(m) \
do { \
perror(m); \
exit(EXIT_FAILURE); \
} while( 0)
int main(int argc, char *argv[])
{
int chi_pipe[2], par_pipe[2];
if (pipe(chi_pipe) == -1 || pipe(par_pipe) == -1)
ERR_EXIT("pipe error");
/* Set O_NONBLOCK flag for the read end (pfd[0]) of the pipe. */
if (fcntl(chi_pipe[0], F_SETFL, O_NONBLOCK) == -1) {
fprintf(stderr, "Call to fcntl failed.\n"); exit(1);
}
/* Set O_NONBLOCK flag for the read end (pfd[0]) of the pipe. */
if (fcntl(chi_pipe[1], F_SETFL, O_NONBLOCK) == -1) {
fprintf(stderr, "Call to fcntl failed.\n"); exit(1);
}
pid_t pid;
pid = fork();
if (pid == -1)
ERR_EXIT("fork error");
if (pid == 0)
{
close(chi_pipe[0]); // I don't read in channel 1
close(par_pipe[1]); // I don't write in channel 2
dup2(chi_pipe[1], STDOUT_FILENO);
close(STDIN_FILENO);
dup2(par_pipe[0], STDIN_FILENO);
execlp("cat", "cat" , "-n", NULL);
//execlp("tr", "tr" , "/a-z/", "/A-Z/", NULL);
sleep(10);
close(chi_pipe[1]);
close(par_pipe[0]);
_exit(0);
}
close(par_pipe[0]);
close(chi_pipe[1]);
while(1) {
char input[1024];
memset(input, 0 , 1024);
fgets(input, 1024 ,stdin);
write(par_pipe[1], input, strlen(input));
char buf[3*1024];
int count = 0;
while (count <= 0)
count=read(chi_pipe[0], buf, 1024*3);
if (count >= 1)
{
printf("buf=%s", buf);
printf("\n");
}
}
close(par_pipe[1]);
close(chi_pipe[0]);
return 0;
}
答案 0 :(得分:1)
有几点:
您需要执行非阻塞I / O.您正在从文件中读取一行,然后将其写入管道。但是不能保证tr
能够方便地将该行反馈翻译。它可能会等待下一行进入。没有适当的线路规则。您需要做的是从您的文件中读取,写入tr
(如果管道未满)并同时从tr
读取(如果字节已准备好)。或者,更准确地说,根据fd(读取)上的数据的可用性或管道中的空间的可用性(写入)。否则你会遇到死锁问题。 tr
不是写作,因为它宁愿先阅读更多,而且还没有EOF
。您没有从tr
读取,因为它尚未写入,因此您不再从该文件中读取任何内容。为此,您需要使用select()
(或poll()
)。
如果execlp
失败,exec
将返回的唯一方式是;在这种情况下,您不希望exit(0)
,因为它一定是错误。