C shell中的多个管道

时间:2014-12-01 07:42:08

标签: c linux shell ipc

任何人都可以告诉我这段代码有什么问题。我能够成功写入管道,但当我使用该管道输出端作为另一个命令的输入时,无法获得输出。以下是代码

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>     //for pid_t fork() and other system calls
#include <signal.h>     //for signal()
#include <sys/types.h>


void command_EXECUTER(char *args[]);
void command_HANDLER(char *args[]);
int pipe_EXECUTER(char *args[], int in, int pos);

void shell_INIT(){

    int is_interactive = isatty(STDOUT_FILENO);

    if(is_interactive){



    }

}

int main(int argc, char *argv[], char **envp){

    char shell_INPUT[1024];
    char *tokens[256];
    int tok_counts = 0;

    while(1){

        memset(shell_INPUT,'\0',sizeof(shell_INPUT));

        printf("\n%s@%s: ",getenv("USER"),getenv("SESSION"));
        fgets(shell_INPUT,1024,stdin);

        if((tokens[0] = strtok(shell_INPUT," \n)\t")) == NULL) continue;

        tok_counts = 1;

        while((tokens[tok_counts] = strtok(NULL, " \n\t")) != NULL) tok_counts++;
        /*
        int i=0;
        while(tokens[i]){
            printf("%s ",tokens[i]);
            i++;
        }
        tokens[tok_counts] = NULL;*/
        command_HANDLER(tokens);

    }


    return 0;
}

int fileIO_EXECUTER(char *args[], int in, int pos){
    int fd;
    pid_t pid;

    if((pid=fork())==0){
        switch(pos){

            case 0: 
                    dup2(fd, STDOUT_FILENO);
                    break;


        }
    }
}

int pipe_EXECUTER(char *args[], int in, int pos){

    int fd[2];
    pid_t pid;
    pipe(fd);

    if((pid=fork())==0){
        printf("%s %s %s %d\n",args[0],args[1],args[2],pid);
        fflush(stdout);
        switch(pos){

            case 0: dup2(fd[1], STDOUT_FILENO);     //first command
                    //close(fd[1]);
                    break;

            case 1: dup2(in, STDIN_FILENO);             //in between command
                    dup2(fd[1], STDOUT_FILENO);
                    //close(fd[1]);
                    break;

            case 2: dup2(in, STDIN_FILENO);         //last command
                    //close(fd[1]);
                    break;

            default: printf("wrong input variable\nexiting");
                     exit(1);   

        }
        if(execvp(args[0],args)==-1) printf("%s: command not found",args[0]); fflush(stdout); kill(getpid(),SIGTERM);

    }else if(pid<0){
        printf("couldn't create child");
        exit(1);
    }else{
        wait(pid);
    }

    return fd[0];

}

void command_EXECUTER(char *args[]){

    pid_t pid;

    if((pid=fork())<0){
        printf("couldn't create the child");
        exit(1);
    }
    else if(pid==0){

        if((execvp(args[0],args))==-1) printf("%s: command not found",args[0]); fflush(stdout); kill(getpid(),SIGTERM);
    }
    else wait(pid);

}

void command_HANDLER(char *args[]){

    int i=0;
    int j=0;
    char spec_chars[100];

    while(args[i]){
        if((strcmp(args[i],">")==0) || (strcmp(args[i],"<")==0) || (strcmp(args[i],"|")==0) || (strcmp(args[i],"&")==0) || (strcmp(args[i],"$")==0)){
            spec_chars[j] = *args[i];
            j++;
        }
        i++;
    }

    if(j==0){
        if(strcmp(args[0],"exit")==0) exit(0);
        else if(strcmp(args[0],"clear")==0) system("clear");
        else command_EXECUTER(args);
    }
    else{

        int in = 0;
        int h = 0;
        int pos=0;
        char data[1024];
        int k=i;
        i = 0;
        while(args[i]){
            if(strcmp(args[i],"|")==0){
                args[i] = (char *)NULL;
                in = pipe_EXECUTER(args,in,pos);
                int n = read(in,data,sizeof(data));
                printf("%s %d\n",data,n);
                fflush(stdout);
                args = args+i+1;

                //printf("%s something %s",args[0],args[1]);
                pos = 1;
                i=0;
            }else if(strcmp(args[i],"<")==0){


            }else if(strcmp(args[i],">")==0){

            }else if(strcmp(args[i],"&")==0){

            }
            else{
                i++;
            }

        }
        in = pipe_EXECUTER(args,in,2);
    }

}

例如,如果我使用

  

ls |排序

&#39;在&#39;描述会在里面有ls但是 当通过&#39; sort&#39;访问相同的输出时它挂在那里......或者执行该命令的孩子永远不会死,并且由于这位父母一直在等待。

1 个答案:

答案 0 :(得分:0)

  

&#39;在&#39;描述符将在其中具有ls但是当相同的输出时   通过&#39; sort&#39;访问它挂在那里

这是由于未关闭父进程中管道的写入结束而导致的,因为sort自然会等到所有文件描述符引用关闭写入结束,确保不再有输入。因此,在pipe_EXECUTER()中,您必须在close(fd[1])之前调用return fd[0];,当然还需要调试代码

                int n = read(in,data,sizeof(data));
                printf("%s %d\n",data,n);
必须删除command_HANDLER()中的

,以便不使用sort的输入。