为什么我的管道只在我发送SIGINT时结束

时间:2017-01-26 11:51:47

标签: c shell pipe zsh waitpid

Basicaly,我在C中写一个shell .. 我试图实现管道功能,这几乎完成了:

 > ls | cat -e | wc | wc -l
 > 1

但是当我尝试管道更慢/更长的执行时我遇到了问题。

事实上,当我试图管道'cat / dev / urandom'的结果时,它基本上等待......

> cat /dev/urandom | cat
>

但是,当我发送一个SIGINT(用ctrl + c)来阻止它时,它会在STDOUT上输出结果缓冲区..

when i ctrl + c , cat /dev/urandom

所以我的问题是:我应该在哪里尝试修复此问题?

我的管道执行部分:

    int         exec_apipe(t_me *me, t_node *curs, int is_last)
{
    int     pfd[2];
    int     pid;

    if (pipe(pfd) == -1 || !curs)
        return (0);
    if (!(curs->pid = fork()))
    {
        dup2(me->fd_in, 0);
        me->fd_in > 0 ? close(me->fd_in) : 0;
        if (!is_last)
            dup2(pfd[1], 1);
        else if (curs->parent->parent && curs->parent->parent->fd_out >= 0)
            dup2(curs->parent->parent->fd_out, 1);
        close(pfd[0]);
        exec_mpipe(me, curs);
        exit(-1);
    }
    else if (curs->pid > 0)
    {
        waitpid(curs->pid, &(curs->ret), WUNTRACED);
        handle_pid(me, curs->ret);
        close(pfd[1]);
        me->fd_in = pfd[0];
    }
    return (pid >= 0 ? 1 : -1);
}

我希望有人会明白我的意思,也许可以帮助.. 感谢

1 个答案:

答案 0 :(得分:2)

在等待任何完成

之前,您需要在管道中启动所有程序。

您的shell实现正在等待每个管道元素在启动下一个管道元素之前完成。但/dev/urandom是一个层出不穷的流; cat /dev/urandom将一直运行直到被杀。所以第二个cat永远不会开始,直到你控制-C第一个。