init进程启动过程梳理

init进程作为Linux内核启动后的1号进程,其将会随着系统运行一直保持运行状态直到系统关机,本文梳理了init进程从内核到根文件系统过程中init进程执行的一个流程。
内核在启动结束时,其会调用kernel_thread(kernel_init, NULL, CLONE_FS | CLONE_SIGHAND);函数挂起kernel_init进程,虽说是进程,其实质还是一个函数,在kernel_init函数中会继续完成一些内核要完成的事,比如挂载根文件系统等等。最后在kernel_init函数结尾会init_post函数中调用run_init_process(execute_command);函数,这个函数对于init进程有着很重要的意义,那就是init进程要从内核态转换为用户态,通俗的来说就是由kernel中的init函数跳转到根文件系统中的init函数。在run_init_process函数中有一个参数execute_command,execute_command实际上是一个全局变量,其值在早期内核启动时由init_setup函数赋值了,赋值的内容是BootLoader的bootargs中为init变量赋的值,例如我的BootLoader中bootargs的init赋值为/linuxrc,最后execute_command的值就是/linuxrc。run_init_process函数内部调用kernel_execve函数,该函数会将执行busybox中的lbb_main函数,同时将/linuxrc与其他环境变量一并传参给它。
这里谈到的busybox是一个用于制作小型根文件系统的开源项目,该根文件系统入口便是lbb_main函数,这个函数负责解析传参内容,执行对应的应用程序。上面提到内核最后会将/linuxrc与其他环境变量一并传参过来,所以接下来就要执行linuxrc程序,但在applets.h文件中有这么一行定义IF_FEATURE_INITRD(APPLET_ODDNAME(linuxrc, init, BB_DIR_ROOT, BB_SUID_DROP, linuxrc)) ,这行定义意思基本上若执行程序linuxrc,最后转为init程序。init程序放在/sbin/目录下,所以如果之前BootLoader中bootargs的init赋值为/sbin/init会是一样的效果。分析到现在,我们可以直到内核转换到根文件系统后执行的就是/sbin/init程序,改程序定义为init.c文件中的init_main函数,该函数实质上就是用户态下的init进程。
在init_main中,其会解析inittab文件完成系统启动配置,设置信号量,最后进入死循环。至此,init进程启动完成。

你可能感兴趣的:(嵌入式/kernel)