将命令输出重定向到文件时错误的文件描述符

时间:2014-01-21 14:04:17

标签: c redirect pipe invoke

我正在尝试编写调用函数以在我的伪壳中调用,但是当简单(一个命令)和管道情况工作时,当我尝试重定向时,它给了我一个写错误,说文件描述符是错误的。 argv []包含第一个命令的名称和参数,而argv []包含第二个命令的相同内容;

int invoke(char *argv1[], char *argv2[], type_c tipo)
{
  int pid, pfd[2], fid;

  switch(tipo)
  {
    case(_SIMPLE):
      pid=fork();
      switch(pid) {
        case 0:
          execvp(argv1[0],argv1);
      }
      break;
    case(_PIPE):
      pipe(pfd);
      pid=fork();
      switch(pid) {
        case 0:
          close(1);
          dup(pfd[1]);
          close(pfd[0]); 
          close(pfd[1]);
          execvp(argv1[0],argv1);
        }
      pid=fork();
      switch(pid) {
        case 0:
          close(0);
          dup(pfd[0]);
          close(pfd[0]); 
          close(pfd[1]);
          execvp(argv2[0],argv2);
        }
        close(pfd[0]);
        close(pfd[1]); 
        break;
    case(_REDIRECT):
      pid=fork();
      switch(pid) {
        case 0:
          close(fileno(stdout));
          fid=open(argv2[0],O_CREAT|O_WRONLY);
          dup(fid);
          close(fid);
          execvp(argv1[0],argv1); 
        }
        break;          
      } 
      return pid; 
  }

测试它,在main()函数中,我编写了这段代码:

char *argv[2][3];
argv[0][0]="ls";
argv[0][1]="-l";
argv[0][2]=NULL;
argv[1][0]="test";
argv[1][1]=NULL;  
pid=invoke(argv[0],argv[1],_REDIRECT); 
那么,我该怎么办? :/ thanx提前全部

1 个答案:

答案 0 :(得分:0)

如果是_REDIRECT

  • 您首先关闭标准输出。这意味着有一个免费的文件描述符1
  • 现在,您正在打开要向其发送命令输出的file。这将直接打开描述符号1
  • 然后,你dup这个新的文件描述符,它将在描述符3上打开相同的内容(2用于stderror)。
  • 现在,您正在关闭之前fid上的1ls程序然后尝试打印到fd 1,在这种情况下已经关闭,从而使其成为bad-file descriptor.

有两种方法可以解决这个问题。

不需要额外的dup,您的代码也能完美运行。

close(fileno(stdout));
int  fid=open(argv2[0],O_CREAT|O_WRONLY);
execvp(argv1[0],argv1);

为了让我们的生活更轻松,还有另一个功能dup2,它会隐式关闭target fd,并将其作为fd的副本。

int  fid=open(argv2[0],O_CREAT|O_WRONLY);
dup2(fid,1);
close(fid);
execvp(argv1[0],argv1);