洞察探索如何通过一套代码实现跨平台小程序开发与高效管理,助力企业数字化转型
1527
2022-09-26
内核启动流程3--Busybox的init进程
Busybox是用来制作文件系统的一个工具集,可以用来替换GNU fileutils shellutils等工具集,它为各种小型的或者嵌入式系统提供了比较完全的工具集。
它提供的核心程序中包括了用户空间的init进程。用户空间的init进程是整个系统启动流程的最后一个阶段,经过该进程的初始化,整个系统进入服务状态,提供诸如系统调用、任务管理服务及设备管理等服务。
1)init进程启动流程
busybox中实现的init进程一般放在开发板的"/sbin"目录下。
busybox的init进程会根据配置文件决定启动哪些程序,如执行某些脚本、启动shell、运行用户指定的程序等。总之,该init进程将成为后续所有进程的发起者,如在init进程启动"/bin/sh"程序后,才能在控制台上输入各种命令。
busybox的init程序对应代码在busybox的init/init.c下,其初始化流程如下
其中与构建根文件系统关系密切的是控制台的初始化、对inittab文件的解释及执行。
2)添加初始化活动
init进程的初始化任务被分解为一系列初始化活动来完成,busybox定义了8种初始化活动,
Sysinit:为init进程提供初始化命令脚本的路径。
Wait:告诉init进程必须等到相应的进程执行完成之后,才能继续执行。
Once:仅执行相应的进程一次,而且不会等待它执行完成。
Respawn:当相应的进程终止执行时,重新启动该进程。
Askfirst:与respawn类似,不过init进程先输出“please press enter to active this console”,等用户按回车键之后,才启动子进程。
Shutdown:当系统关机时,执行相应的进程。
Restart:当init进程重新启动时,执行相应的进程,通常此处执行的进程就是init本身。
Ctrlatldel:当按下crtl+alt_delete组合键时,执行相应的进程
init进程中需要添加的活动,通常被写到inittab文件中,inittab文件通常位于根文件系统的"/etc"目录下,其中每一条配置信息定义一个初始化活动。格式如下:
inittab中的每个条目有4个字段,各字段间由冒号分开。
一个简单的inittab示例:
::sysinit:/etc/init.d/rcS::askfirst:-/bin/sh::ctrllaltdel:/sbin/reboot::shutdown:/bin/umount -a -r
在init_main()函数中调用parse_inittab()函数解析inittab,也正是在该函数中完成向init进程添加活动的任务。尽管没有inittab文件,parse_inittab()中也会添加默认的活动,如下
static void parse_inittab(void){ #if ENABLE_FEATURE_USE_INITTAB char *token[4]; parser_t *parser = config_open2("/etc/inittab",fopen_for_read); if(parser == NULL) #endif { /*No inittab file--set up some default behavior*/ /*Reboot on Ctrl-Alt-Del*/ new_init_action(CTRLALTDEL,"reboot",""); /*Umount all filesystems on halt/reboot*/ new_init_action(SHUTDOWN,"umount -a -r",""); new_init_action(RESTART,"init",""); /*sysinit*/ new_init_action(SYSINIT,INIT_SCRIPT,""); return; }}
new_init_action函数的三个参数分别表示活动类型、相关的命令和使用的控制台,若不指定最后一个参数,则表示使用与init进程相同的控制台。
最后一个新的初始化活动,即当SYSINIT活动发生时,由INIT_SCRIPT确定的脚本被执行。SYSINIT活动表示只在系统初始化阶段被init进程加载一次。INIT_SCRIPT宏的默认值定义如下,BUSYBOX init进程默认的初始化脚本是/etc/init.d/rcS
#define INITTAB "/etc/inittab"#ifndef INIT_SCRIPT#define INIT_SCRIPT "/etc/init.d/rcS"#endif
parse_inittab()函数的其余代码解析inittab文件的具体内容,最终根据Inittab中给定的配置,向init进程添加各项活动。
3)执行初始化活动
init进程解析inittab后开始执行各类初始化活动。它通过run_actions()函数执行指定类型的初始化活动,并为符合条件的活动,执行其 相关命令。init进程各类初始化活动的执行顺序如下:
/*First run the sysinit command*/run_actions(SYSINIT);/*Next run anything that wants to block*/run_actions(WAIT);/*Next run anything to be run ony once*/run_actions(ONCE);...../*Now run the looping stuff for the rest of forever*/while(1){ /*run the respawn/askfirst stuff*/ run_actions(RESPAWN|ASKFIRST); /*Don't consume all CPU time--sleep a bit*/ sleep(1); /*wait for any child process to exit*/ wpid = wait(NULL); while(wpid>0) { /*Find out who died and clean up their corpse*/ ..... } }
从上面的代码可以看出,活动的执行顺序如下:
1)执行系统初始化脚本,默认为/etc/init.d/rcS,活动类型为SYSINIT;
2)执行所有将会导致阻塞的初始化活动对应的命令,活动类型是WAIT;
3)执行所有一次执行的初始化活动的命令,活动类型是ONCE;
完成以上工作后,init循环执行以下任务:
1)执行所有中止后必须重启的活动命令,类型是RESPAWN。
2)执行所有中止后必须重启,但必须先询问的活动命令,类型是ASKFIRST。
3)等待子进程推出。
版权声明:本文内容由网络用户投稿,版权归原作者所有,本站不拥有其著作权,亦不承担相应法律责任。如果您发现本站中有涉嫌抄袭或描述失实的内容,请联系我们jiasou666@gmail.com 处理,核实后本网站将在24小时内删除侵权内容。
发表评论
暂时没有评论,来抢沙发吧~