20135313吴子怡.北京电子科技学院
chapter1 知识点梳理
一、Linux内核源代码简介
(视频中对目录下的文件进行了简介,记录如下)
-
arch目录
占有相当庞大的空间 arch/x86目录下的代码是需要重点关注的。 arch下其他目录可以删掉。
-
init目录
内核启动相关的基本代码基本都在init目录下。 main.c 文件中有一个start_kernel函数,初始化Linux内核的起点,这个函数相当于普通c程序的main函数。
-
kernel目录
Linux内核的核心代码在kernel目录中
-
其他
Documentation 文档 drivers 驱动 fs-filesystem 文件系统 include ipc 进程间通信
-
README:
INSTALLING 怎样安装内核源代码——怎么解压怎么打补丁 make mrproper 把生成的中间代码清理干净 menuconfig
二、构造一个简单的Linux系统
-
使用实验楼的虚拟机打开shell,命令如下:
cd LinuxKernel/ qemu -kernel linux-3.18.6/arch/x86/boot/bzImage -initrd rootfs.img
内核启动完成后进入menu程序,支持三个命令help、version和quit。
-
通过gdb跟踪内核的启动
-
使用自己的Linux系统环境搭建MenuOS的过程
下载内核源代码编译内核
cd ~/LinuxKernel/ wget https://www.kernel.org/pub/linux/kernel/v3.x/linux-3.18.6.tar.xz xz -d linux-3.18.6.tar.xz tar -xvf linux-3.18.6.tar cd linux-3.18.6 make i386_defconfig make
制作根文件系统
cd ~/LinuxKernel/ mkdir rootfs git clone https://github.com/mengning/menu.git # 如果被墙,可以使用附件menu.zip cd menu gcc -o init linktable.c menu.c test.c -m32 -static –lpthread # 系统会默认启动init,即第一个用户态进程,1号进程 cd ../rootfs cp ../menu/init ./ find . | cpio -o -Hnewc |gzip -9 > ../rootfs.img # 镜像
启动MenuOS系统
cd ~/LinuxKernel/ qemu -kernel linux-3.18.6/arch/x86/boot/bzImage -initrd rootfs.img
- 重新配置编译Linux使之携带调试信息
-
在原来配置的基础上,make menuconfig选中如下选项重新配置Linux,使之携带调试信息(由于在实验楼中操作,因此省略个别步骤)
-
make重新编译
-
使用gdb跟踪调试内核
qemu -kernel linux-3.18.6/arch/x86/boot/bzImage -initrd rootfs.img -s -S
关于-s和-S选项的说明: -S freeze CPU at startup (use ’c’ to start execution) cpu初始化之前把它冻结起来 -s shorthand for -gdb tcp::1234 在1234端口上建立了一个gdb server 若不想使用1234端口,则可以使用-gdb tcp:xxxx来取代-s选项
打开另外一个shell
gdb (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之前,也可以在之后
三、跟踪调试Linux内核的启动过程
1.使用gdb跟踪调试内核的方法
除了前半部分提及的,以下部分为补充:
(gdb)c # 系统开始启动,启动到start_kernel (gdb)list # 可以看到start_kernel上下的代码 (gdb)break rest_init (gdb)c # 当前系统执行到rest_init (gdb)list # 可以看到rest_init是在start_kernel的尾部调用的。
chapter2 实践实验过程
①构建Menu系统的过程
②使用gdb进行调试
③将断点设置到start_kernal
④将断点设置到rest_init
chapter3 简单分析一下start_kernel(作业)
①在init目录下main.c里找到start_kernel函数
②全局变量init_task,即手工创建的PCB,0号进程即最终的idle进程。
不管分析内核的哪一部分都会涉及到start_kernel.
③一些简要提及的函数
trap_init() 初始化一些中断向量 mm_init() 内存管理模块 sched_init() 调度模块 rest_init()中有kernel_thread(kernel_init,NULL,CLONE_FS),kernel_init中有run_init_process,创建了一号进程,默认路径下的程序。
注意:init_process是一号进程。
④创建kthread(一个内核线程来管理系统的资源)。
启动后,进入了一个cpu_idle,cpu_idle_loop,即一个while(0)的无限循环,或者说是idle进程,它作为0号进程适中存在于系统中。
当系统没有进程需要执行时就需要调度idle进程。
chapter4 总结
阐明对“Linux系统启动过程”的理解。
sched_init()初始化函数内对0号进程,即idle进程进行初始化。然后rest_init()其他初始化函数,函数内将创建1号进程,即init进程。随后rest_init实际上就是start_kernel内核一启动的时候会一直存在,这个就叫0号进程;0号进程创建了1号进程kernel_init和其他服务线程。这就是内核的启动过程。
chapter5 附录
作者:吴子怡
学号:20135313
原创作品转载请注明出处
《Linux内核分析》MOOC课程http://mooc.study.163.com/course/USTC-1000029000