1.教材第六章学习及与chatgpt的对话
https://chat.openai.com/share/88b29d62-23e7-43a3-be3a-a830a4783068
2.实验楼
系统调用内核函数(mkdir)
1.修改menu中test.c文件,将两种调用函数写入
2.在int main()中添加如下两行函数
int my_mkdir() {
const char *dirname = "20233808"; // 指定要创建的目录名
// 调用mkdir函数,创建名为"20233808"的目录,权限为0755
if (mkdir(dirname, 0755) == 0) {
printf("Directory created successfully.\n"); // 如果创建成功,打印成功消息
} else {
perror("mkdir"); // 如果创建失败,使用perror函数打印错误消息
exit(1); // 退出程序并返回1作为错误码
}
return 0; // 返回0表示程序正常执行完成
}
int mkdir() {
const char *dirname = "202338082"; // 指定要创建的目录名
int result; // 用于保存系统调用结果的变量
asm volatile (
"mov $39, %%eax\n" // 设置系统调用号 (39 是 "mkdir" 的系统调用号)
"mov %0, %%ebx\n" // 设置目录名参数 (dirname)
"mov $0755, %%ecx\n" // 设置权限参数 (0755 权限)
"int $0x80\n" // 调用 0x80 中断来触发系统调用
"mov %%eax, %1\n" // 保存返回值 (结果) 到变量result
: "=r" (result) // 输出操作数
: "r" (dirname) // 输入操作数
: "%eax", "%ebx", "%ecx" // 用于寄存器占用的约束
);
if (result == 0) {
printf("Directory created successfully.\n");
} else {
perror("mkdir"); // 如果创建失败,使用perror函数打印错误消息
exit(1); // 退出程序并返回1作为错误码
}
return 0; // 返回0表示程序正常执行完成
}
int main()
{
PrintMenuOS();
SetPrompt("MenuOS>>");
MenuConfig("version","XXX V1.0(Menu program v1.0 inside)",NULL);
MenuConfig("quit","Quit from XXX",Quit);
MenuConfig("my_mkdir","make a directory",my_mkdir);
MenuConfig("mkdirasm","make a directory",mkdirasm);
ExecuteMenu();
}
3.运行结果
-help后可以看到多了my_mkdir和mkdir两个输出
gdb跟踪分析
$ qemu -kernel linux-3.18.6/arch/x86/boot/bzImage -initrd rootfs.img -s -S
# 关于-s和-S选项的说明:
#1. -S
#-S 在启动时冻结CPU(使用'c'开始执行)
#2. -s
#-s 是'-gdb tcp::1234'的简写形式
# 若不想使用1234端口,则可以使用-gdb tcp:xxxx来取代-s选项
# 打开 GDB 调试器
$ gdb
# 在 GDB 中输入以下命令:
# 在gdb界面中targe remote之前加载符号表
(gdb)file linux-3.18.6/vmlinux
# 建立gdb和gdbserver之间的连接,按c 让qemu上的Linux继续运行
(gdb)target remote:1234
# 断点的设置可以在target remote之前,也可以在之后
(gdb)break start_kernel
#继续运行
(gdb) c
#查看startkernel这段代码
(gdb)list
#在要分析的这个系统调用处设置断点
(gdb)b sys_mkdir
#继续运行
(gdb) c
#继续运行
(gdb) c
跟踪结果
3.从 system_call 开始到 iret 结束之间的整个过程
4.实验中遇到的问题
test.c中写入API调用函数时不能命名为mkdir需要自己设定名称,因此我改为了 int my_mkdir.