跳转至

简介

在实验一中,我们使用了许多命令行操作来完成一系列工作,如编译 Linux 内核、打包 initrd 等。所有的命令行都由一个叫做 shell 的程序解释并执行。本实验我们将自己编写一个简单的 shell 并理解 Linux shell 程序的工作原理。你可以在本文末尾获得一段示例代码。

重要

本实验作为代码编程实验,会进行查重,一切抄袭行为后果自负!

shell 程序包含一些必须实现的功能点和「可选」功能。

必做部分得分上限 100%,可选部分得分上限 20%。你最多可以累计获得 120% 的分数。

如何提问?

提问的智慧

以下是一些要点:

  • 提问前,请先 STFW 和 RTFM,ChatGPT 也是一个可考虑的选择;
  • 尽量使你的问题是可复现的、明确的,并在此基础上裁剪定位到会导致出问题的核心代码;
  • 详细描述问题,并提供相关指令及相关问题的报错截图,如果涉及系统问题请提供系统版本。

优先在实验文档评论区、Issue 区 及课程群内讨论实验的相关问题以便解决共性问题。

实验要求

请按照以下目录结构组织你的 GitHub 仓库:

.                     // Git 仓库目录
├── lab2              // 实验二根目录
│   ├── shell.cpp     // 你的 shell 的源代码
│   ├── other.cpp
│   ├── Makefile      // 你提供的 Makefile
│   └── README.md     // 运行说明及简要的实验报告
├── .gitignore
└── README.md

如果选用 Rust 语言,lab2 目录下按照 Rust 工程组织(即包含 src 文件夹和 Cargo.lock, Cargo.toml 文件)即可。

本实验可以使用 libc, libstdc++, libm 以及 iostream, STL 等 C/C++ 语言标准和常用库。如果你愿意,你也可以使用 readline 和 ncurses 等 Linux 程序常用库。对 Rust 允许使用库的说明请参见实验零。使用此处没有列出的库前请询问助教。

Git 的使用应符合规范。当出现以下情况时,我们会酌情扣除一定分数:

  • 很大一部分的 commit 是由 GitHub 网页上传的文件;
  • commit 寥寥无几。最好的习惯是每实现一个新的功能及每一次代码重构都对应 commit;
  • 上传了大量与实验要求无关的文件。

实验报告

你在 README.md 中应该至少包括:

  • 你的 shell 实现可能与系统中的 bash(或助教期望的表现)有所不同,简要介绍这些潜在的区别,以免产生误会,导致不必要的扣分;
  • 介绍你完成的选做项目,方便助教进行更准确的评估。

本实验的主要内容为 shell 程序的编写,因此不必花费太多工夫在实验报告上。

关于自由选做

在必做部分以外,你可以参考标记为「可选」的几个功能来实现其他 shell 功能以得到选做分数(即不限本文档列出的功能)。每一项额外功能都会由助教参照其他可选功能讨论评估。

我们鼓励进行与操作系统相关的实验探究,因此过度脱离主题的项目可能不会获得加分,例如:

  • 过于简单的内置命令,如 : (colon), true, false, help
  • 严重偏离 shell 的基本功能的项目,例如你 模仿 Zsh 为你的 shell 内置了一个俄罗斯方块游戏

作为一个参考基准,GNU Bash 具有的功能大部分都会被认可。

关于编译

在本实验中,你需要提供 Makefile 或者 Cargo.toml 文件,我们将使用它来编译你提交的程序;在这些文件不能正常运行时,我们将尽可能去尝试编译 lab2/ 目录下所有的 *.c*.cpp 或者 *.rs 文件。

你可以在 README.md 中说明编译与运行相关的注意事项。如果在参考了你的说明后,我们依然无法正常编译你的代码,我们会尝试修复并酌情扣除一定分数。

示例程序

可以将文末程序命名为 shell.cpp,然后在相同目录下创建 Makefile

CC=g++
CFLAGS=-c -Wall
SOURCES=shell.cpp
OBJECTS=$(SOURCES:.cpp=.o)
EXECUTABLE=shell

all: $(SOURCES) $(EXECUTABLE)

$(EXECUTABLE): $(OBJECTS)
    $(CC) $(OBJECTS) -o $@

.cpp.o:
    $(CC) $(CFLAGS) $< -o $@

clean:
    rm -f *o $(EXECUTABLE)

继续输入 make 即可编译出一个可执行文件 shell。

你可以输入 ./shell 来运行它。这是一个非常简陋的 shell,它会提示你输入命令,你可以调用系统中有的其他命令来运行,例如 lscat 等,或输入 exit 退出。

输入 make clean 删除编译结果和中间产物。

gitignore

你可以通过将 .o 文件和可执行文件添加到 .gitignore 文件的方式,来确保不会误 commit 中间文件。具体方式请 STFW。

你可以改进框架(示例代码嵌套层数较多,且错误处理不算很完备)并在框架的基础上继续完成实验,也可以从头编写自己的 shell 或者采用其它语言编写。

思考

了解 exec 系统调用家族。execvp 是如何根据 ls 获取到 ls 的可执行文件位置并执行的?

// IO
#include <iostream>
// std::string
#include <string>
// std::vector
#include <vector>
// std::string 转 int
#include <sstream>
// PATH_MAX 等常量
#include <climits>
// POSIX API
#include <unistd.h>
// wait
#include <sys/wait.h>

std::vector<std::string> split(std::string s, const std::string &delimiter);

int main() {
  // 不同步 iostream 和 cstdio 的 buffer
  std::ios::sync_with_stdio(false);

  // 用来存储读入的一行命令
  std::string cmd;
  while (true) {
    // 打印提示符
    std::cout << "$ ";

    // 读入一行。std::getline 结果不包含换行符。
    std::getline(std::cin, cmd);

    // 按空格分割命令为单词
    std::vector<std::string> args = split(cmd, " ");

    // 没有可处理的命令
    if (args.empty()) {
      continue;
    }

    // 退出
    if (args[0] == "exit") {
      if (args.size() <= 1) {
        return 0;
      }

      // std::string 转 int
      std::stringstream code_stream(args[1]);
      int code = 0;
      code_stream >> code;

      // 转换失败
      if (!code_stream.eof() || code_stream.fail()) {
        std::cout << "Invalid exit code\n";
        continue;
      }

      return code;
    }

    if (args[0] == "pwd") {
      std::cout << "To be done!\n";
      continue;
    }

    if (args[0] == "cd") {
      std::cout << "To be done!\n";
      continue;
    }

    // 处理外部命令
    pid_t pid = fork();

    // std::vector<std::string> 转 char **
    char *arg_ptrs[args.size() + 1];
    for (auto i = 0; i < args.size(); i++) {
      arg_ptrs[i] = &args[i][0];
    }
    // exec p 系列的 argv 需要以 nullptr 结尾
    arg_ptrs[args.size()] = nullptr;

    if (pid == 0) {
      // 这里只有子进程才会进入
      // execvp 会完全更换子进程接下来的代码,所以正常情况下 execvp 之后这里的代码就没意义了
      // 如果 execvp 之后的代码被运行了,那就是 execvp 出问题了
      execvp(args[0].c_str(), arg_ptrs);

      // 所以这里直接报错
      exit(255);
    }

    // 这里只有父进程(原进程)才会进入
    int ret = wait(nullptr);
    if (ret < 0) {
      std::cout << "wait failed";
    }
  }
}

// 经典的 cpp string split 实现
// https://stackoverflow.com/a/14266139/11691878
std::vector<std::string> split(std::string s, const std::string &delimiter) {
  std::vector<std::string> res;
  size_t pos = 0;
  std::string token;
  while ((pos = s.find(delimiter)) != std::string::npos) {
    token = s.substr(0, pos);
    res.push_back(token);
    s = s.substr(pos + delimiter.length());
  }
  res.push_back(s);
  return res;
}