基于Linux 0.11 写一个自己的命令提示行
功能
写一个mysh.c文件,实现一个简单的Linux命令提示行,包括内部命令cd、sync、exit等;外部命令cat、ls 等命令。
前置知识
命令解释器程序的一般结构是怎样的
图片来源
如何获取当前目录的路径名
可以使用库函数 getcwd 。
如何实现内部命令 cd 和 sync ?
可以调用库函数 chdir 和 sync 。
如何执行一个可执行文件?
可以使用库函数 execvp 或者 execve ,前者更简单些。
在 bochs 虚拟机中如何编辑文件
可以使用命令 uemacs 编辑文件,如:
uemacs /etc/rc
保存的快捷键是 Ctrl-x,s ,退出的快捷键是 Ctrl-x, Ctrl-c 。
实现代码
#include <unistd.h>
#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include <sys/types.h>
#include <sys/wait.h>
void main(int argc, char *argv[]){
int pid;
int status;
char dirname[80];
char cmdstr[255];
int argc_cmd = 0;
char *argv_cmd[10];
char *p;
while (1){
/* prompt */
getcwd(dirname, 80);
printf("%s$ ", dirname);
fflush(stdout);
/* get command */
memset(cmdstr, 0, 255);
fgets(cmdstr, 255, stdin);
/* parse command */
argc_cmd = 0;
p = strtok(cmdstr, " \n\t");
printf("Hello,I'm hjl!\n");
while (p && argc_cmd < 9){
argv_cmd[argc_cmd] = p;
p = strtok(NULL, " \n\t");
argc_cmd++;
}
argv_cmd[argc_cmd] = NULL;
/* interpret internal commands */
if (argc_cmd == 0)
continue;
if (strcmp(argv_cmd[0], "exit") == 0)
break;
if (strcmp(argv_cmd[0], "sync") == 0){
sync();
continue;
}
if (strcmp(argv_cmd[0], "cd") == 0){
if ((argc_cmd == 1) || (strcmp(argv_cmd[1], "~") == 0))
argv_cmd[1] = "/";
if (chdir(argv_cmd[1]) < 0)
printf("cd error\n");
continue;
}
/* execute external commands */
if (!(pid = fork())){
if (execvp(argv_cmd[0], argv_cmd) == -1)
printf("execvp error\n");
exit(1);
}
while (1)
if (pid == wait(&status))
break;
}
/* exit */
exit(0);
}
如何将文件放入 bochs 虚拟机
首先将该文件放入目录 linux-0.11-lab/b/ 下:
然后用版本 0 内核启动虚拟机,并在虚拟机中用 mcopy 命令将该文件拷入虚拟机硬盘: