2

我正在尝试在类 unix xv6 OS 的 shell 程序中实现基本命令执行。我正在编辑的部分 shell 代码是 runcmd 函数,我在其中使用 execvp 命令执行终端中使用的命令。该程序在我编译时编译没有错误,但是当我尝试在命令行上键入命令时没有任何反应。我已经阅读了exec 命令的手册页,但我仍然不明白这些参数需要在 exec() 命令中传递的正确方式,或者我什么时候使用哪个版本的 exec()对操作系统编程来说还是很新的。

我在这里没有实现哪些需要添加才能执行命令?我有下面的 runcmd 函数的代码:

编辑:

我只是为每个命令添加了更多带有二进制文件路径的 exec 语句;但是,只有第一个 exec 命令有效(在本例中为 cd)。当我使用任何其他命令时,命令行会像执行 CD 一样执行它。如何让它适用于多个命令?

struct cmd {
  int type;          //  ' ' (exec), | (pipe), '<' or '>' for redirection
};
struct
 execcmd {
  int type;              // ' '
  char *argv[MAXARGS];   // arguments to the command to be exec-ed
};

// Execute cmd.  Never returns.
void
runcmd(struct cmd *cmd)
{
  int p[2], r;
  struct execcmd *ecmd;
  struct pipecmd *pcmd;
  struct redircmd *rcmd;

  if(cmd == 0)
    exit(0);
  
  switch(cmd->type){
  default:
    fprintf(stderr, "unknown runcmd\n");
    exit(-1);

  case ' ':
    ecmd = (struct execcmd*)cmd;
    
    if(ecmd->argv[0] == 0)
      exit(0);
    //fprintf(stderr, "exec not implemented\n");
     execvp("/bin/cd" , ecmd->argv );
     execvp("/bin/grep" , ecmd->argv );
     execvp("/bin/echo" , ecmd->argv );
     execvp("/bin/cat" , ecmd->argv );
     execvp("/bin/ls" , ecmd->argv );
    break;

  case '>':
  case '<':
    rcmd = (struct redircmd*)cmd;
    //fprintf(stderr, "redir not implemented\n");
    execvp("/bin" , ecmd->argv );
    runcmd(rcmd->cmd);
    break;

  case '|':
    pcmd = (struct pipecmd*)cmd;
    fprintf(stderr, "pipe not implemented\n");
    int execl(const char *path, const char *arg, ...);
    break;
  }    
  exit(0);
}
4

2 回答 2

3

看起来您正在尝试执行“/bin”目录。

您应该将 exec 调用的第一个参数设为用户想要运行的二进制文件。

当命令失败时,使用 perror 函数还会为您提供有用的输出。

于 2015-09-13T21:53:48.207 回答
2

认为这是你真正需要的:

case ' ':
    ecmd = (struct execcmd*)cmd;
    if (ecmd->argv[0] == 0)
        _exit(0);
    execvp(ecmd->argv[0], ecmd->argv);
    /* if control reaches this point, execvp failed */
    perror(ecmd->argv[0]);
    _exit(127);

您可能想知道为什么execvp将可执行文件作为参数向量中的单独参数加载,而您只是将它交给它argv[0]。这是遗留功能;如果我今天从头开始设计这个 API,我认为我不会包含它。然而,这个想法是一个程序可能会根据它是什么而表现不同argv[0]。例如,在过去,ex并且vi相同的可执行文件(指向同一个 inode 的两个硬链接——符号链接尚未发明),它根据它的argv[0]. 而且,自从 Unix 诞生以来,它就login(1)调用了第一个字符argv[0]设置为'-';的 shell。没有/bin/-sh,所以它实际上需要能够与参数向量分开指定要调用的程序。

我目前无法想到外壳使用除.argv[0]execvp

关于您的代码的其他更改的注意事项:

  • 永远不要叫exit孩子身边的一个fork,只有_exit
  • 当系统调用失败时,总是打印出问题文件的名称和strerror(errno), to stderr, not stdoutperror是此操作的方便简写。
  • 为了perror安全调用,您必须在 .之前立即fflush(0)调用父级,并且您的 C 库必须正确实现 stderr 的行缓冲。我提到这一点是因为 XV6 似乎是一个教学操作系统,你必须自己实现它的一些部分,而且我不知道 stdio 是否是你的责任。fork
于 2015-09-14T00:27:23.710 回答