Implement small shell with fork and exec
小さなシェルを実装する. シェルとしてのループを実行中に,標準入力からコマンドが入ると fork() して子プロセスを exec() で塗り替える
// implementation for shell with fork and exec
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <sys/wait.h>
#include <sys/types.h>
#define MAX_ARGS 100
void myexec(char *input){
char *args[MAX_ARGS];
int arg_count = 0;
char *p = strtok(input, " \t\n");
while(p != NULL && arg_count < MAX_ARGS){
args[arg_count++] = p;
p = strtok(NULL, " \t\n");
}
args[arg_count] = NULL;
if(arg_count == 0){
return;
}
pid_t pid = fork();
if(pid < 0){
perror("fork");
exit(1);
}
if(pid == 0){ // child
int err;
err = execvp(args[0], args);
if(err < 0){
perror("execvp");
exit(1);
}
}else{ // parent
int status;
waitpid(pid, &status, 0);
}
}
int main(){
char buf[1024];
printf("---- my shell ----\n");
while(1){
printf("myshell> ");
fflush(stdout);
if(fgets(buf, sizeof(buf), stdin) == NULL){
break;
}
if(strncmp(buf, "exit", 4) == 0 && buf[4] == '\n'){
break;
}
myexec(buf);
}
printf("Good Bye\n");
exit(0);
}
実行するとこんな感じ
lubuntu@lubuntu-virtualbox:~/linux_tutorial$ ./myshell
---- my shell ----
myshell> ls
grep.c head mycat mycat.c mygrep myshell myshell.c t2.txt t.txt
myshell> hello
execvp: No such file or directory
myshell> pwd
/home/lubuntu/linux_tutorial
myshell> exit
Good Bye
execvp のエラーが変だけどいっか...
次にリダイレクトを実装する. dup2 で標準入力と標準出力のファイルディスクリプタを子プロセスに割り当ててあげる. 冗長でダサいが暫定的に下記.
例えば “>” にぶつかったら,その次に宣言されているファイルを探し,そのファイルディスクリプタを複製する.(dup2(int oldfd, int newfd);
)
これによって,その次のファイルディスクリプタを標準入力に割り当てる.
oldfd を複製して newfd に割り当てる.
つまり,newfd は oldfd と同じファイルを指す.
なので,標準出力はその次のファイルディスクリプタを(その fork() 内では)指す.
これによって,そのコマンドの実行結果はファイルディスクリプタに出力される.
if(pid == 0){ // child
for(int i = 0; i < arg_count; i++){
if(strncmp(args[i], ">", 1) == 0){
int fd = open(args[i + 1], O_WRONLY | O_CREAT | O_TRUNC, 0644);
if(fd < 0){
perror("open");
exit(1);
}
dup2(fd, STDOUT_FILENO);
close(fd);
args[i] = NULL;
break;
}
if(strncmp(args[i], ">>", 1) == 0){
int fd = open(args[i + 1], O_WRONLY | O_CREAT | O_APPEND, 0644);
if(fd < 0){
perror("open");
exit(1);
}
dup2(fd, STDOUT_FILENO);
close(fd);
args[i] = NULL;
break;
}
if(strncmp(args[i], "<", 1) == 0){
int fd = open(args[i + 1], O_RDONLY);
if(fd < 0){
perror("open");
exit(1);
}
dup2(fd, STDIN_FILENO);
close(fd);
args[i] = NULL;
break;
}
}
int err;
err = execvp(args[0], args);
if(err < 0){
perror("execvp");
exit(1);
}
}else{ // parent
int status;
waitpid(pid, &status, 0);
}
lubuntu@lubuntu-virtualbox:~/linux_tutorial$ ./myshell
---- my shell ----
myshell> ls
grep.c head mycat mycat.c mygrep myshell myshell.c t2.txt t.txt
myshell> cat mycat.c > tem.txt
myshell> cat tem.txt
#include <stdio.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h>
#include <stdlib.h>
...
次にパイプを実装する.
void mypipe(char *input){
char *args[MAX_ARGS];
int arg_count = 0;
char *p = strtok(input, "|");
while(p != NULL && arg_count < MAX_ARGS){
args[arg_count++] = p;
p = strtok(NULL, "|");
}
int pipefd[2];
int fd_in = 0;
for(int i = 0; i < arg_count; i++){
pipe(pipefd);
if(fork() == 0){ // child
dup2(fd_in, STDIN_FILENO);
if(i < arg_count - 1){
dup2(pipefd[1], STDOUT_FILENO);
}
close(pipefd[0]);
myexec(args[i]);
exit(0);
}
close(pipefd[1]);
fd_in = pipefd[0];
}
for (int i = 0; i < arg_count; i++){
wait(NULL);
}
}
int main(){
char buf[1024];
printf("---- my shell ----\n");
while(1){
printf("myshell> ");
fflush(stdout);
if(fgets(buf, sizeof(buf), stdin) == NULL){
break;
}
if(strncmp(buf, "exit", 4) == 0 && buf[4] == '\n'){
break;
}
if(strchr(buf, '|') != NULL){
mypipe(buf);
}
myexec(buf);
}
printf("Good Bye\n");
exit(0);
}
よさそう. パイプラインもリダイレクトも一つにしか対応できていないがまあモックとしよう... ちゃんとしたシェルはまた Rust で書こう
おしまい