第三周Linux学习报告
Linux内核源代码简介:
arch/x86中内容重点关注
init目录重要,内核启动相关的代码基本上都在init目录下。如main.c等。Start_kernel函数相当于普通C程序的main函数。
ipc进程间通信的一些代码
Linux内核的核心代码在Kernel目录中。
lib公用的库文件
mm内存管理。
实验
使用实验楼的虚拟机打开shell
cd LinuxKernel/
qemu -kernel linux-3.18.6/arch/x86/boot/bzImage -initrd rootfs.img
(1)linux-3.18.6.tar为内核源代码,rootfs里面含有init可执行文件,生成了rootfs.img。
(2)该系统支持三条命令:help、version和quit
2.使用gdb跟踪调试内核
输入以下命令:
qemu -kernel linux-3.18.6/arch/x86/boot/bzImage -initrd rootfs.img -S -s
-S:在CPU初始化之前冻结
-s:在1234端口上创建gdb,若不想使用1234端口,则可以使用-gdb tcp:xxxx来取代-s选项
-
(gdb)file linux-3.18.6/vmlinux # 在gdb界面中targe remote之前加载符号表
-
(gdb)target remote:1234 # 建立gdb和gdbserver之间的连接,按c 让qemu上的Linux继续运行
-
(gdb)break start_kernel # 断点的设置可以在target remote之前,也可以在之后
输入list可查看函数代码
基本是通过调用init函数来启动的
其中一些模块:
-
trap_init();初始化一些中断向量
-
mm_init() 内存管理模块
-
sched_init() 调度模块
kernel_init里面的run_init_process为1号进程,第一个用户态进程,默认根目录下的init程序
总结:
从rest_init开始,Linux开始产生进程,因为init_task是静态制造出来的,pid=0,它试图将从最早的汇编代码一直到 start_kernel的执行都纳入到init_task进程上下文中。在rest_init函数中,内核将通过下面的代码产生第一个真正的进程 (pid=1)。然后init_task变为一个idle task,init_idle函数的第一个参数current就是&init_task,在init_idle中将会把init_task加入到 cpu的运行队列中,这样当运行队列中没有别的就绪进程时,init_task(也就是idle task)将会被调用,它的核心是一个while(1)循环,在循环中它将会调用schedule函数以便在运行队列中有新进程加入时切换到该新进程上。