使用管道和叉子在C中重定向stdout

时间:2013-04-23 23:23:50

标签: c pipe

我有关于ptrace和管道的练习。以下代码是整个程序的一部分。 管道在主要部分之前制作,this_trace.s_out为1.主要父亲创建孩子,这个孩子为stdout创建自己的孩子。当程序运行ls然后它在屏幕上打印并且不写入文件。有什么问题?

if(pid == 0)
{
char buf0[BUFFSIZE], buf1[BUFFSIZE], buf2[BUFFSIZE];
int length0, length1, length2;


if(this_trace.s_out == 1)   //For stdout redirection
{
    if((pid1=fork()) == -1)
    {
        perror("fork");
        exit(1); 
    }

    if(pid1 == 0) //child for stdout redirect
    {//sleep(2);
        if(fd1 = open(this_trace.out_name,O_WRONLY | O_CREAT | O_TRUNC, 0666) == -1)
        {
            perror("create stdout file");
            exit(1);
        }


        close(p_out[WRITE]);
        close(p_in[READ]);
        close(p_in[WRITE]);
        close(p_err[READ]);
        close(p_err[WRITE]);

        do{
            if((length1 = read(p_out[READ],buf1,BUFFSIZE)) == -1)
            {
                perror("Read for stdout redirection");
                exit(1);
            }
            write(fd1, buf1, length1);
        }while(length1 > 0);


        close(fd1);
        //close(p_out[READ]);
        return 0;
        //break;

    }
    else if(pid1 > 0)//child from main father
    {
        close(p_out[READ]);
        close(p_in[READ]);
        close(p_in[WRITE]);
        close(p_err[READ]);
        close(p_err[WRITE]);

        dup2(p_out[WRITE], 1);
    }

}




ptrace(PTRACE_TRACEME, 0, NULL, NULL);  


//execv(argv[1],NULL);
execl("/bin/ls","ls",NULL);


}

抱歉我的英语不好。

1 个答案:

答案 0 :(得分:1)

目前尚不清楚为什么有这么多流程。您遇到问题:

if (fd1 = open(this_trace.out_name,O_WRONLY | O_CREAT | O_TRUNC, 0666) == -1)

无论打开的文件描述符如何,都会将0或1分配给fd1。它应该是:

if ((fd1 = open(this_trace.out_name,O_WRONLY | O_CREAT | O_TRUNC, 0666)) == -1)

使用dup2()(或dup())将文件描述符重定向到标准通道后,应关闭原始文件描述符。因此,在dup2(p_out[WRITE], 1);之后,您需要close(p_out[WRITE]);

您应该从execl()检测失败并处理它;如果execl()返回,则失败。

您展示未使用的变量buf0buf2(以及相应的length0length2)。