我正在尝试编写一个基本的shell,它可以解释简单的命令,例如语言c中的date,ls。
我首先获取这样的PATH变量,然后将其传递给execv()函数。
const char *name = "PATH";
char *value;
value = getenv(name)
我打印出这个值,我得到了这个:
/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/sbin:/bin:/usr/games:/usr/local/games
请注意,我使用的是virutalbox来运行Ubuntu。这是我用来尝试简单的ls命令的代码。在下面的代码中,变量行是用户编写的实际命令,在我们的例子中它是" ls"
pid_t pid, wpid;
int status;
pid = fork();
if (pid == 0) {
// Child process
if (execv(value, line) == -1) {
perror("lsh");
}
exit(EXIT_FAILURE);
}
else if (pid < 0) {
// Error forking
perror("lsh");
}
else {
// Parent process
do {
wpid = waitpid(pid, &status, WUNTRACED);
}
while (!WIFEXITED(status) && !WIFSIGNALED(status));
}
我得到的结果是:
lsh: no such file or directory
任何想法?
答案 0 :(得分:5)
execv()
系统调用使用您在第一个参数中指定的名称作为可执行文件的文件名;它没有进行基于PATH的搜索。
这意味着如果指定"lsh"
作为第一个参数,则当前目录中必须存在可执行文件lsh
。
如果您想要基于PATH的搜索,请将execv()
替换为execvp()
。否则,在第一个参数中指定命令的路径名 - 绝对或相对,但绝对更正常。
请注意,如果任何exec*()
函数返回,则表示失败。没有必要测试返回值;它总是-1。
value
和line
的内容必须符合以下几行:
char *value = "ls";
char *line[] = { "ls", "-l", 0 };
execvp(value, line);
或更常规地:
execvp(line[0], line);
如果您自己分析PATH,则需要line[0]
指向您从PATH创建的完整文件名,然后使用execv()
而不是{{1 }}
答案 1 :(得分:0)
execv
的第一个参数是要运行的命令。这意味着您尝试将/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/sbin:/bin:/usr/games:/usr/local/games
作为命令运行。
假设line
数组中的第一个值是您要调用的程序,则应该执行此操作:
execv(line[0], line);
如果您想要执行基于路径的搜索,请改用execvp
(无需手动提取PATH变量):
execvp(line[0], line);
编辑:
例如,假设您想运行ls -l /usr/bin /var/log
,您的数组将如下所示:
char *line[] = { "ls", "-l", "/usr/bin", "/var/log", NULL};