一、真容初现
终于走进了Linux内核代码了。Linux作为操作系统,源码肯定十分十分庞大。要了解全部的实现代码,这也不现实。弱水三千,只取一瓢饮。了解一些小的片段对我们来说亦是大有裨益。
二、实验与分析
实验目的:使用gdb跟踪调试内核从start_kernel到init进程启动
实验步骤:
1.跟踪调试内核 -S 冻结CPU,-s 打开远程调试端口(默认使用1234端口,若不想使用1234端口,则可以使用-gdb tcp:xxxx来取代-s选项)
cd LinuxKernel/
qemu -kernel linux-3.18.6/arch/x86/boot/bzImage -initrd rootfs.img -S -s
2.使用水平分割打开另一个shell,跟踪运行。
gdb
file linux-3.18.6/vmlinux # 在gdb界面中targe remote之前加载符号表
target remote:1234 # 建立gdb和gdbserver之间的连接,按c 让qemu上的Linux继续运行
break start_kernel # 断点的设置可以在target remote之前,也可以在之后
结果分析:
start_kernel是linux内核初始化的入口,在 start_kernel()中 Linux 将完成整个系统的内核初始化。
PCB初始化
中断与内存管理模块初始化
调度模块初始化
等等以及一些其它模块的初始化
在Linux 内核初始化的末尾,是rest_init。
创建了 Linux 系统的第一个用户态进程,并一直存在。
创建的kthreadd2号进程,它是所有内核态线程的祖先。
三、总结
start_kernel是Linux内核初始化的入口, 将完成整个系统的内核初始化。内核初始化的最后一步—rest_init将启动 所有进程的祖先(用户进程与内核进程)。引用课程附件的一句话就是:道生一(start_kernel….cpu_idle),一生二(kernel_init和kthreadd),二生三(即前面0、1和2三个进程),三生万物(1号进程是所有用户态进程的祖先,2号进程是所有内核线程的祖先)。
writen by
江明星