我一直在完成编写一个简单外壳的任务,我已经设法获得了一些非常基本的功能,但是对于其中一个阶段,它要求我获取和设置环境。这是任务的详细信息。
第 4 阶段:获取和设置路径 – 10 分
保留原路径
这是必要的原因是因为我们希望将路径恢复到最初退出 shell 时的路径。这一点很重要,因为您对 shell 环境(即上面的 setenv() 函数)所做的任何更改,不仅会影响 shell 本身,还会影响之后将从同一终端执行的任何其他程序。出于这个原因,最好将事情恢复到退出时的状态。
单个字符串足以保留原始路径。
保存路径应该是你的 shell 在启动时做的第一件事。
打印和更改路径 - 内置命令
在 C 程序中,我们可以使用 getenv() 函数访问环境,也可以使用 setenv() 函数更改环境。如果您查看 setenv() 的手册页,您会发现它是如何工作的(即需要的参数和返回值)以及使用它需要包含的内容。
getpath – 打印系统路径 & setpath – 设置系统路径
这两个命令是关于环境参数 PATH 的。第一个只是获取并打印它的值,而第二个将路径(冒号分隔的目录列表的字符串)作为参数并使其成为 PATH 的值。为此,您可以分别 getenv() 和 setenv()。
恢复路径
您只需将 PATH 环境参数更改为其原始值(即您在 shell 启动时保存的那个)。
恢复路径应该是您的 shell 在退出之前所做的最后一件事。
第 4 阶段:测试
首先,确保您为第 3 阶段执行的所有测试仍然有效。不过要小心,因为我们现在正在更改路径,这将影响外部程序的执行。
要检查附加功能,您应该首先检查路径的保存和恢复是否有效。这里的一个好主意是在您在 shell 执行开始时保存路径时打印路径,然后在最后退出时再次打印。在这两种情况下,打印的路径应该完全相同!
之后,您应该检查调用 getpath 时是否打印当前路径,该路径应该与原始路径相同。
然后你应该专注于测试 setpath。首先,将路径设置为新值并测试 getpath 打印它,然后尝试查看更改路径如何真正影响外部命令的执行(例如,将路径设置为仅 '.' 并尝试 'ls' 或尝试壳本身等)。
这是我的代码:
/* S.NO:201148541 Simple Shell Example
Completed upto Stage 3 */
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>
#include <errno.h>
#include <sys/types.h>
#define BUFFER_SIZE 1<<16
#define ARR_SIZE 1<<16
void parse_args(char *buffer, char** args,
size_t args_size, size_t *nargs)
{
char *buf_args[args_size];
char **cp;
char *wbuf;
size_t i, j;
wbuf=buffer;
buf_args[0]=buffer;
args[0] =buffer;
for(cp=buf_args; (*cp=strsep(&wbuf, " \n\t")) != NULL ;){
if ((*cp != '\0') && (++cp >= &buf_args[args_size]))
break;
}
for (j=i=0; buf_args[i]!=NULL; i++){
if(strlen(buf_args[i])>0)
args[j++]=buf_args[i];
}
*nargs=j;
args[j]=NULL;
}
int main(int argc, char *argv[], char *envp[]){
char buffer[BUFFER_SIZE];
char *args[ARR_SIZE];
int *ret_status;
size_t nargs;
pid_t pid;
char curDir[100];
while(1){
getcwd(curDir, 100);
printf("%s->", curDir);
fgets(buffer, BUFFER_SIZE, stdin);
parse_args(buffer, args, ARR_SIZE, &nargs);
if (nargs==0) continue;
if(strcmp(args[0], "cd") == 0){
chdir(args[1]);
}
else if (!strcmp(args[0], "exit" )) exit(0);
pid = fork();
if (pid){
pid = wait(ret_status);
printf("finished\n", pid);
}
else {
if( execvp(args[0], args)) {
puts(strerror(errno));
exit(127);
}
}
}
}
return 0;
}
我真的很茫然,任何指导都会有所帮助。