linux 内核进程初始化,浅谈linux内核从start_kernel到init进程启动

首先是实验指导:

使用实验楼的虚拟机打开shell

cd LinuxKernel/

qemu -kernel linux-3.18.6/arch/x86/boot/bzImage -initrd

rootfs.img

内核启动完成后进入menu程序(《软件工程C编码实践篇》的课程项目),支持三个命令help、version和quit,您也可

以添加更多的命令,对选修过《软件工程C编码实践篇》的童鞋应该是a piece of cake.

使用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)

# -s shorthand for -gdb tcp::1234 若不想使用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之前,也可以在之后

实验过程截图:​

a4c26d1e5885305701be709a3d33442f.png

a4c26d1e5885305701be709a3d33442f.pngshell分层

上层是qemu该系统,下层是使用gdb进行断点调试

内核代码分析:

​gdb在start_kernel处设置断点,从此处内核进入了c语言部分,主要完成全部的内核主要功能的初始化。

对于start_kernel函数:

我们来看看在start_kernel函数中到rest_init函数前内核都做了什么。​

smp_setup_processor_id() 单CPU位为空。

local_irq_disable() 关闭中断

​boot_cpu_init()

确定有多少个CPU可用。

page_address_init()

初始化高端内存。使用高端内存(128M)来进行映射处理。​

setup_arch(&command_line)在arch/arm/kernel/setup.c中找到对应系统的启动参数。(每种体系结构都有自己的setup_arch()函数,具体编译哪个体系结构的setup_arch()函数,由源码树顶层目录下的Makefile中的ARCH变量决定)

setup_per_cpu_areas()

每个cpu分配pre-cpu结构内存

smp_prepare_boot_cpu()复制.data.precpu段内数据

pidhash_init() hash线程id初始化

vfs_caches_init_early()

虚拟文件系统的初始化

trap_init()函数完成对系统保留中断向量(异常、非屏蔽中断以及系统调用)的初始化

​​mm_init() 内存管理初始化

sched_init() 进程调度器初始化

rcu_init()初始化直接读拷贝更新的锁机制

early_irq_init()

init_IRQ()中断初始化

init_timers() ... time_init()

timer初始化,高精度time初始化

profile_init()​对内核的profile(一个内核性能调式工具)功能进行初始化

接下来是rest_init()​

​kernel_thread(kernel_init,

NULL, CLONE_FS)

启动内核线程kernel_init,1号进程,它是内核线程之父,管理调度其他的内核线程,内核线程列表由kthread_create_list全局链表管理。

​之后的几个函数,增加idle进程的need_resched​标志,并且调用schedule释放cpu

​再谈kernel_init函数

在执行free_initmem之前必须结束所有的初始化代码

接下来就是 启动init进程。

init进程是第一个用户态进程,叫做”1号进程“。通过start_kernel函数->

rest_init函数 -> kernel_init函数 ->

run_init_process生成,找根目录下的程序来作为“1号进程”。

当系统没有进程需要执行时就调度到idle进程。这就是“0号进程”。从系统启动之后就一直存在。它创建了1号进程“kernel_init”和其他进程。这样系统就启动起来了。

作者:王俊博

《Linux内核分析》MOOC课程http://mooc.study.163.com/course/USTC-1000029000

依照学术诚信条款,我保证此回答为本人原创,所有回答中引用的外部材料已经做了出处标记。原创作品转载请注明出处 。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值