如何在子进程中使用dup2?

时间:2013-01-25 02:34:37

标签: c linux

在我的C程序中,我正在创建一个子进程并在子进程中运行execvp。但我正在尝试将错误消息更改为其他内容,对于execvp命令(如果有错误)。

我知道如果它返回,那么这是一个错误,然后我可以在下一行打印我自己的自定义错误消息。这是可能发生的一种错误,例如,如果我将命令“sdfsd”发送给execvp,就会发生这种情况。这部分对我有用。

但如果我输入“find sdfsd”,那么它就不会返回并打印“find:`sdfsd':没有这样的文件或目录”。

我想将此消息(实际上是来自exevcp的任何类型的错误消息)更改为我自己的自定义消息。

我相信我可以使用dup2来做到这一点,但我不确定如何......

在孩子的过程中,我尝试了

dup2(STDERR_FILENO, 1); 
fclose(stderr);

但这只会阻止子进程编写任何错误消息。在所有情况下我仍然无法打印自己的信息..

有谁知道怎么做? 感谢

1 个答案:

答案 0 :(得分:1)

由于execvp在成功启动新程序后永远不会返回,因此在execvp运行的程序失败后,您将无法在子进程中打印自己的错误消息。一种选择是将stderr传递给父进程,拦截那里的错误消息,然后打印自己的错误消息。

例如:

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>

int main(int argc, char **argv)
{
  int ff, p[2];
  FILE *f;
  char *vv[] = {"find", "garbage", (char *)NULL};
  char msg[100];

  if (pipe(p) != 0)
  {
    fprintf(stderr, "Pipe failed\n");
    return 1;
  }

  if ((ff = fork()) == -1 )
  {
    fprintf(stderr, "Fork failed\n");
    return 1;
  }

  if (ff == 0)
  {
    /* In the child process */
    close(2);
    close(p[0]);
    dup2(p[1], 2);
    execvp("find", vv);
    return 1;
  };

  /* In the parent process */
  close(p[1]);
  f = fdopen(p[0], "r");
  if (f == NULL)
  {
    fprintf(stderr, "Fdopen failed\n");
    return 1;
  }
  if (fgets(msg, sizeof(msg), f) == NULL)
  {
    fprintf(stderr, "Fgets failed\n");
    return 1;
  }
  printf("Error message was: %s", msg);
  /* and so on */
  return 0;
}