子进程执行某些系统命令的结果无法通过管道发送到父进程

时间:2014-12-16 18:09:05

标签: c redirect pipe exec fork

也许这不是一个紧凑的标题,我很抱歉:)。我尝试使用管道将子进程的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;
}

1 个答案:

答案 0 :(得分:1)

有几点:

  1. 您需要执行非阻塞I / O.您正在从文件中读取一行,然后将其写入管道。但是不能保证tr能够方便地将该行反馈翻译。它可能会等待下一行进入。没有适当的线路规则。您需要做的是从您的文件中读取,写入tr(如果管道未满)并同时从tr读取(如果字节已准备好)。或者,更准确地说,根据fd(读取)上的数据的可用性或管道中的空间的可用性(写入)。否则你会遇到死锁问题。 tr不是写作,因为它宁愿先阅读更多,而且还没有EOF。您没有从tr读取,因为它尚未写入,因此您不再从该文件中读取任何内容。为此,您需要使用select()(或poll())。

  2. 如果execlp失败,exec将返回的唯一方式是;在这种情况下,您不希望exit(0),因为它一定是错误。