MenuOS的构造
1.Linux内核源代码简介
(一)计算机的“三大法宝”
- 存储程序计算机
- 函数调用堆栈
- 中断
(二)操作系统的“两把宝剑”
- 中断上下文
- 进程上下文
(三)Linux内核代码目录结构分析
- arch:arch目录是与体系结构相关的子目录列表,里面存放了许多CPU体系结构的相关代码,比如arm、x86、MIPS、PPC等。
- block:存放Linux存储体系中关于块设备管理的代码。
- crypto:存放常见的加密算法的C语言代码,譬如crc32、md5、sha1等。
- Documentation:存放一些文档。
- drivers:驱动目录,里面分门别类地存放了Linux内核支持的所有硬件设备的驱动代码。
- fs:文件系统,里面列出了Linux支持的各种文件系统的实现。
- init:init是初始化的意思,存放Linux内核启动时的初始化代码。
- 我们可以在main.c中找到start_kernel函数,start_kernel函数是初始化Linux内核启动的起点,start_kernel前的代码使用汇编语言来进行硬件初始化。
2.构造一个简单的Linux内核
(一)在实验楼平台上构建Linux系统MenuOS:
命令分析:
cd ~/LinuxKernel/
qemu -kernel linux-3.18.6/arch/x86/boot/bzImage -initrd rootfs.img
//qemu仿真kernel;bzImage是vmLinux经过gzip压缩后的文件,是压缩的内核映像,"b"代表的是"big"(bzImage适用于大内核,zImage适用于小内核)。
(二)gdb跟踪调试Linux内核
命令分析:
$ qemu -kernel linux-3.18.6/arch/x86/boot/bzImage -initrd rootfs.img -s -S
/* 关于-s和-S选项的说明:
1. -S
-S freeze CPU at startup (use ’c’ to start execution)
2. -s
-s shorthand for -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
// 查看断点处的start_kernel()函数
(gdb)list
3.总结
从 start_kernel 到 init 进程启动的过程分析如下:
1.start_kernel()相当于C语言中的main函数,它是一切的起点,在该函数被调用之前,内核代码主要是用汇编语言写的,用于完成该硬件系统的初始化工作,为c代码的运行设置环境
2.通过start_kernel()函数调用内部模块init_task()来创建init进程,之后调用cpu_idle()演变成了idle进程,执行一次调度后,init进程运行。1号内核线程负责执行内核的部分初始化工作及进行系统配置,最后调用do_execve加载init程序,演变成init进程(用户态1号进程),init进程是内核启动的第一个用户态进程。