c - 如何链接多个管道? (加上奇怪的 bug )

标签 c shell pipe fork pipeline

我正在尝试编写一个可以处理管道命令的简单 shell。我希望能够处理所有链接在一起的多个管道,但我很难弄清楚如何实现这样的事情。

这是我当前的尝试:

int status;
    int lastToken = 0;
    int pipe_pid;

    //create the pipes
    int pipefd[pipes][2];

    // Loop to run all commands in the vertical list.
    while(1){
        if (c->type == TOKEN_PIPE){

        // Here is where we deal with pipes
            for (int i = 0; i < pipes; i++){
                pipe(pipefd[i]);

                pipe_pid = fork();

                //this is a receiving pipe
                if (pipe_pid == 0){
                    // create the write end of the pipe
                    dup2(pipefd[i][WRITE_SIDE], STDOUT_FILENO);
                    close(pipefd[i][READ_SIDE]);
                    close(pipefd[i][WRITE_SIDE]);
                    execvp(c->argv[0], c->argv);
                    // printf("parent pipe\n");
                }
                //this is a writing pipe
                else{
                    close(pipefd[i][WRITE_SIDE]);
                    dup2(pipefd[i][READ_SIDE], STDIN_FILENO);
                    close(pipefd[i][READ_SIDE]);
                    // printf("child pipe\n");
                }
            }



        // This stuff happens for all commands
        lastToken = c->type;
        // If it's the last command, we're done
        if (c->next == NULL){
            break;
        }
        else{
            c = c->next;
        }
    }

命令在一个链表中链接在一起,c是我的命令指针

pipes 是我在解析字符串内时创建的变量,因此我知道有多少个“|”我在命令中看到了。这应该告诉我需要 fork 的子进程的数量。

我使用管道为管道描述符创建一个二维数组。

然后我想对每个管道和 fork 循环一次,并使用 dup2 映射输入和输出。

我遇到了无法弄清楚的不一致错误。首先,每次运行管道命令时,我的 shell 都会立即崩溃,没有段错误或其他打印错误。

其次,如果我运行 echo foo | 这样的命令wc -c 我有时会得到 4,有时会得到 0 作为输出。

我确信我只是做了一些愚蠢的事情,但我不确定是什么:/

最佳答案

我发现我做错了什么,在所有线程使用完毕之前我就关闭了管道。我通过取消千钧一发来解决了这个问题。

// writing side of the pipe
if (c->type == TOKEN_PIPE){
    close(c->pipefd[READ_SIDE]);
    dup2(c->pipefd[WRITE_SIDE], STDOUT_FILENO);

}
// receiving side of the pipe
if (commandPrev->type == TOKEN_PIPE){
    close(commandPrev->pipefd[WRITE_SIDE]);
    dup2(commandPrev->pipefd[READ_SIDE], STDIN_FILENO);

}

然后在父线程中,就在我重新获得僵尸之前,我检查已使用完毕的管道并关闭它们。

    // writing side of the pipe
    if (c->type == TOKEN_PIPE){
        close(c->pipefd[READ_SIDE]);
        dup2(c->pipefd[WRITE_SIDE], STDOUT_FILENO);
    }
    // receiving side of the pipe
    if (commandPrev->type == TOKEN_PIPE){
        close(commandPrev->pipefd[WRITE_SIDE]);
        dup2(commandPrev->pipefd[READ_SIDE], STDIN_FILENO);
    close(commandPrev->pipefd[READ_SIDE]);

我不确定这是否是最佳方法,但它对我来说没有错误。

关于c - 如何链接多个管道? (加上奇怪的 bug ),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/40696287/

相关文章:

javascript - Node - 在管道之后正确关闭流

c - 在递归函数中返回和使用多个树结构

Scientific Linux中ARM的交叉编译

linux - 如何将此 perl 脚本的输出重定向到文件?

java - 如何在不关闭前一个 shell 的情况下打开弹出 shell

linux - 带选项的 Bash Shell 命令

pipe - gnuplot : plot pipe output

C 线程行为不正确 - 简单代码

c - sigprocmask() 中的 Set 和 Oldset

python - MINGW64 在执行期间不打印