0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
会员中心
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

基于linux内核启动程序

嵌入式小生 来源:嵌入式小生 作者:嵌入式小生 2022-10-14 09:11 次阅读

linux内核启动过程的后期,在kernel_init()函数代表的init线程中,会尝试执行用户空间的init进程:

a9272532-4b57-11ed-a3b6-dac502259ad0.png

从上述代码可见,会尝试执行/sbin/、/etc、/bin三个目录中的init。从《busybox源码分析笔记(一)》一文可以知道,在busybox编译构建完成并安装后,会生成对应的目录(注:/etc目录不存在)。在/sbin目录中,则会存在一个init链接:

a955e37c-4b57-11ed-a3b6-dac502259ad0.png

查看其属性,其本质则是链接到了../bin/busybox:

a975872c-4b57-11ed-a3b6-dac502259ad0.png

综上所述,证明linux内核启动后期,运行的第一个用户空间程序是init,在busybox源码中,init程序则由位于/init目录中的init.c编译构建而成,程序入口是:init_main(),小生在该函数中添加一行标识代码:

a9913efe-4b57-11ed-a3b6-dac502259ad0.pnglinux内核运行后期的结果如下:

a9b277a4-4b57-11ed-a3b6-dac502259ad0.gif

可见,linux内核后期加载的就是busybox下的init程序。

init_main分析

贴上该函数的完整代码,下文将分段描述:

intinit_main(intargcUNUSED_PARAM,char**argv)
{
structsigactionsa;

INIT_G();

/*SomeuserssendpoweroffsignalstoinitVERYearly.
*Tohandlethis,masksignalsearly.
*/
/*sigemptyset(&G.delayed_sigset);-donebyINIT_G()*/
sigaddset(&G.delayed_sigset,SIGINT);/*Ctrl-Alt-Del*/
sigaddset(&G.delayed_sigset,SIGQUIT);/*re-execanotherinit*/
#ifdefSIGPWR
sigaddset(&G.delayed_sigset,SIGPWR);/*halt*/
#endif
sigaddset(&G.delayed_sigset,SIGUSR1);/*halt*/
sigaddset(&G.delayed_sigset,SIGTERM);/*reboot*/
sigaddset(&G.delayed_sigset,SIGUSR2);/*poweroff*/
#ifENABLE_FEATURE_USE_INITTAB
sigaddset(&G.delayed_sigset,SIGHUP);/*reread/etc/inittab*/
#endif
sigaddset(&G.delayed_sigset,SIGCHLD);/*makesigtimedwait()exitonSIGCHLD*/
sigprocmask(SIG_BLOCK,&G.delayed_sigset,NULL);

#ifDEBUG_SEGV_HANDLER
memset(&sa,0,sizeof(sa));
sa.sa_sigaction=handle_sigsegv;
sa.sa_flags=SA_SIGINFO;
sigaction_set(SIGSEGV,&sa);
sigaction_set(SIGILL,&sa);
sigaction_set(SIGFPE,&sa);
sigaction_set(SIGBUS,&sa);
#endif

if(argv[1]&&strcmp(argv[1],"-q")==0){
returnkill(1,SIGHUP);
}

#if!DEBUG_INIT
/*ExpecttobeinvokedasinitwithPID=1orbeinvokedaslinuxrc*/
if(getpid()!=1
&&(!ENABLE_LINUXRC||applet_name[0]!='l')/*notlinuxrc?*/
){
bb_simple_error_msg_and_die("mustberunasPID1");
}

#ifdefRB_DISABLE_CAD
/*TurnoffrebootingviaCTL-ALT-DEL-wegeta
*SIGINTonCADsowecanshutthingsdowngracefully...*/
reboot(RB_DISABLE_CAD);/*misnomer*/
#endif
#endif

/*If,say,xmallocwouldeverdie,wedon'twanttooopskernel
*byexiting.
*NB:wesetdie_func*after*PID1checkandbb_show_usage.
*Otherwise,forexample,"initu"("pleaserexecyourself"
*commandforsysvinit)willshowhelptext(whichisn'ttoobad),
**andsleepforever*(whichisbad!)
*/
die_func=sleep_much;

/*Figureoutwherethedefaultconsoleshouldbe*/
console_init();
set_sane_term();
xchdir("/");
setsid();

/*Makesureenvironsissettosomethingsane*/
putenv((char*)"HOME=/");
putenv((char*)bb_PATH_root_path);
putenv((char*)"SHELL=/bin/sh");
putenv((char*)"USER=root");/*needed?why?*/

if(argv[1])
xsetenv("RUNLEVEL",argv[1]);

#if!ENABLE_FEATURE_INIT_QUIET
/*Helloworld*/
message(L_CONSOLE|L_LOG,"initstarted:%s",bb_banner);
#endif

/*Checkifwearesupposedtobeinsingleusermode*/
if(argv[1]
&&(strcmp(argv[1],"single")==0||strcmp(argv[1],"-s")==0||LONE_CHAR(argv[1],'1'))
){
/*???shouldn'twesetRUNLEVEL="b"here?*/
/*Startashellonconsole*/
new_init_action(RESPAWN,bb_default_login_shell,"");
}else{
/*Notinsingleusermode-seewhatinittabsays*/

/*NOTEthatifCONFIG_FEATURE_USE_INITTABisNOTdefined,
*thenparse_inittab()simplyaddsinsomedefault
*actions(i.e.,INIT_SCRIPTandapair
*of"askfirst"shells)*/
parse_inittab();
}

#ifENABLE_SELINUX
if(getenv("SELINUX_INIT")==NULL){
intenforce=0;

putenv((char*)"SELINUX_INIT=YES");
if(selinux_init_load_policy(&enforce)==0){
BB_EXECVP(argv[0],argv);
}elseif(enforce>0){
/*SELinuxinenforcingmodebutload_policyfailed*/
message(L_CONSOLE,"can'tloadSELinuxPolicy."
"Machineisinenforcingmode.Haltingnow.");
returnEXIT_FAILURE;
}
}
#endif

#ifENABLE_FEATURE_INIT_MODIFY_CMDLINE
/*Makethecommandlinejustsay"init"-that'sall,nothingelse*/
strncpy(argv[0],"init",strlen(argv[0]));
/*Wipeargv[1]-argv[N]sotheydon'tclutterthepslisting*/
while(*++argv)
nuke_str(*argv);
#endif

/*SetupSTOPsignalhandlers*/
/*StophandlermustallowonlySIGCONTinsideitself*/
memset(&sa,0,sizeof(sa));
sigfillset(&sa.sa_mask);
sigdelset(&sa.sa_mask,SIGCONT);
sa.sa_handler=stop_handler;
sa.sa_flags=SA_RESTART;
sigaction_set(SIGTSTP,&sa);/*pause*/
/*Doesnotworkasintended,atleastin2.6.20.
*SIGSTOPissimplyignoredbyinit
*(NB:behaviormightdifferunderstrace):
*/
sigaction_set(SIGSTOP,&sa);/*pause*/

/*Nowruneverythingthatneedstoberun*/
/*Firstrunthesysinitcommand*/
run_actions(SYSINIT);
check_delayed_sigs(&G.zero_ts);
/*Nextrunanythingthatwantstoblock*/
run_actions(WAIT);
check_delayed_sigs(&G.zero_ts);
/*Nextrunanythingtoberunonlyonce*/
run_actions(ONCE);

/*Nowruntheloopingstufffortherestofforever*/
while(1){
/*(Re)runtherespawn/askfirststuff*/
run_actions(RESPAWN|ASKFIRST);

/*Waitforanysignal(typicallyit'sSIGCHLD)*/
check_delayed_sigs(NULL);/*NULLtimespecmakesitwait*/

/*Waitforanychildprocess(es)toexit*/
while(1){
pid_twpid;
structinit_action*a;

wpid=waitpid(-1,NULL,WNOHANG);
if(wpid<= 0)
    break;

   a = mark_terminated(wpid);
   if (a) {
    message(L_LOG, "process '%s' (pid %u) exited. "
      "Scheduling for restart.",
      a->command,(unsigned)wpid);
}
}

/*Don'tconsumeallCPUtime-sleepabit*/
sleep1();
}/*while(1)*/
}

跳过条件宏定义下的编译分支,主要分析其通用的代码部分:

(1)首先使用sigaddset()将信号添加到信号集合,添加的信号有:Ctrl-Alt-Del、SIGQUIT、SIGPWR、SIGUSR1、SIGTERM、SIGUSR2、SIGUSR2。

(2)然后找出系统默认的控制台,并将路径切换到/,接着重新创建一个新的会话:

console_init();
set_sane_term();
xchdir("/");
setsid();

(3)设置默认的环境变量:

putenv((char*)"HOME=/");
putenv((char*)bb_PATH_root_path);
putenv((char*)"SHELL=/bin/sh");
putenv((char*)"USER=root");/*needed?why?*/

(4)如果是单用户模式,则会调用new_init_action(RESPAWN, bb_default_login_shell, "");在控制台启动一个shell;否则,则会调用parse_inittab()函数。

parse_inittab()函数定义如下:

staticvoidparse_inittab(void)
{
#ifENABLE_FEATURE_USE_INITTAB
char*token[4];
parser_t*parser=config_open2("/etc/inittab",fopen_for_read);

if(parser==NULL)
#endif
{
/*Noinittabfile-setupsomedefaultbehavior*/
/*Sysinit*/
new_init_action(SYSINIT,INIT_SCRIPT,"");
/*Askfirstshellontty1-4*/
new_init_action(ASKFIRST,bb_default_login_shell,"");
//TODO:VC_1insteadof""?""isconsole->cttyproblems->angryusers
new_init_action(ASKFIRST,bb_default_login_shell,VC_2);
new_init_action(ASKFIRST,bb_default_login_shell,VC_3);
new_init_action(ASKFIRST,bb_default_login_shell,VC_4);
/*RebootonCtrl-Alt-Del*/
new_init_action(CTRLALTDEL,"reboot","");
/*Umountallfilesystemsonhalt/reboot*/
new_init_action(SHUTDOWN,"umount-a-r","");
/*Swapoffonhalt/reboot*/
new_init_action(SHUTDOWN,"swapoff-a","");
/*RestartinitwhenaQUITisreceived*/
new_init_action(RESTART,"init","");
return;
}

#ifENABLE_FEATURE_USE_INITTAB
/*optional_ttyaction:command
*Delimsarenottobecollapsedandneedexactly4tokens
*/
while(config_read(parser,token,4,0,"#:",
PARSE_NORMAL&~(PARSE_TRIM|PARSE_COLLAPSE))){
/*ordermustcorrespondtoSYSINIT..RESTARTconstants*/
staticconstcharactions[]ALIGN1=
"sysinit�""wait�""once�""respawn�""askfirst�"
"ctrlaltdel�""shutdown�""restart�";
intaction;
char*tty=token[0];

if(!token[3])/*lessthan4tokens*/
gotobad_entry;
action=index_in_strings(actions,token[2]);
if(action< 0 || !token[3][0]) /* token[3]: command */
   goto bad_entry;
  /* turn .*TTY ->/dev/TTY*/
if(tty[0]){
tty=concat_path_file("/dev/",skip_dev_pfx(tty));
}
new_init_action(1<< action, token[3], tty);
  if (tty[0])
   free(tty);
  continue;
 bad_entry:
  message(L_LOG | L_CONSOLE, "Bad inittab entry at line %d",
    parser->lineno);
}
config_close(parser);
#endif
}

如果定义了CONFIG_FEATURE_USE_INITTAB,则会使用/etc/inittab文件中的action;如果CONFIG_FEATURE_USE_INITTAB没有定义,parse_inittab()则会简单添加一些默认actions(例如,运行INIT_SCRIPT,然后启动一个“askfirst”shell)。如果ENABLE_FEATURE_USE_INITTAB已定义,但是/etc/inittab文件缺失也会使用相同的默认行为集合。

(5)设置STOP信号处理程序:

memset(&sa,0,sizeof(sa));
sigfillset(&sa.sa_mask);
sigdelset(&sa.sa_mask,SIGCONT);
sa.sa_handler=stop_handler;
sa.sa_flags=SA_RESTART;
sigaction_set(SIGTSTP,&sa);/*pause*/

sigaction_set(SIGSTOP,&sa);/*pause*/

(6)接下来运行想要运行的命令,依次运行:SYSINIT、WAIT、ONCE:

run_actions(SYSINIT);
check_delayed_sigs(&G.zero_ts);

run_actions(WAIT);
check_delayed_sigs(&G.zero_ts);

run_actions(ONCE);

(7)在最后,则是一个while(1)的死循环,用于处理我们在命令行下输入的命令:

while(1){
/*重新运行respawn/askfisrt*/
run_actions(RESPAWN|ASKFIRST);

/*等待信号*/
check_delayed_sigs(NULL);/*NULLtimespecmakesitwait*/

/*等待所有的子进程退出*/
while(1){
pid_twpid;
structinit_action*a;

wpid=waitpid(-1,NULL,WNOHANG);
if(wpid<= 0)
    break;

   a = mark_terminated(wpid);
   if (a) {
    message(L_LOG, "process '%s' (pid %u) exited. "
      "Scheduling for restart.",
      a->command,(unsigned)wpid);
}
}

/*短暂让出CPU,不要消耗所有的CPU时间*/
sleep1();
}/*while(1)*/

补充

BusyBox的init不支持运行级别。runlevels字段在BusyBox init中将会被完全忽略。

所以如果想要使用运行级别的系统,需使用sysvinit作为启动进程。

一、7个运行级别

(1)运行级别0:系统停机状态,系统默认运行级别不能设为0,否则不能正常启动。其实就是关机。

(2)运行级别1:单用户工作状态,root权限,用于系统维护,禁止远程登陆。在忘记root密码时一般用这个运行级别,进去修改root密码。

(3)运行级别2:多用户状态(没有NFS),没有网络连接。

(4)运行级别3:完全的多用户状态(有NFS),登陆后进入控制台命令行模式。linux很常见的运行级别

(5)运行级别4:系统未使用,保留。

(6)运行级别5:X11控制台,登陆后进入图形GUI模式。就是图形模式。

(7)运行级别6:系统正常关闭并重启,默认运行级别不能设为6,否则不能正常启动。

二、查看运行级别

1、runlevel命令:打印系统的上一个和当前运行级别:

aa1518dc-4b57-11ed-a3b6-dac502259ad0.png

N:“N”表示自系统启动后运行级别尚未更改。从上图可见,小生的Ubuntu系统的运行级别为5。

审核编辑:彭静
声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • 函数
    +关注

    关注

    3

    文章

    4305

    浏览量

    62430
  • 代码
    +关注

    关注

    30

    文章

    4748

    浏览量

    68349
  • LINUX内核
    +关注

    关注

    1

    文章

    316

    浏览量

    21618
收藏 人收藏

    评论

    相关推荐

    Linux编译驱动、内核及应用程序分析

    作为一名嵌入式Linux新手,在学习的过程中会遇到很多问题。写了一个驱动程序怎么编译?怎么加载进内核
    的头像 发表于 01-17 13:46 6598次阅读
    <b class='flag-5'>Linux</b>编译驱动、<b class='flag-5'>内核</b>及应用<b class='flag-5'>程序</b>分析

    Linux内核启动过程和Bootloader(总述)

    精简讲述linux内核启动过程。[转]1.Linux内核启动过程概述 一个嵌入式
    发表于 08-18 17:35

    【OK210试用体验】bootloader启动linux内核

    的性能和稳定性。用户空间的文件系统用来提供管理系统的各种配置,提供相应的应用程序、服务、数据交换等。文件系统作为一种载体,它是用来实现用户与操作系统内核的交互。因此,一个可启动linux
    发表于 01-10 15:57

    Linux内核编译和启动的相关资料分享

    Linux环境下,我们想运行一个应用程序,在shell交互环境下直接敲命令就可以了,操作系统给程序提供了运行环境和进程管理。那Linux操作系统本身是如何运行和
    发表于 12-20 06:28

    Android的Linux内核与驱动程序开发教程

    Android内核是基于Linux 2.6内核的,它是一个增强内核版本,除了修改部分Bug外,它提供了用于支持Android平台的设备驱动,这里介绍了Android
    发表于 09-05 14:24 335次下载
    Android的<b class='flag-5'>Linux</b><b class='flag-5'>内核</b>与驱动<b class='flag-5'>程序</b>开发教程

    linux内核启动内核解压过程分析

    linux启动内核解压过程分析,一份不错的文档,深入了解内核必备
    发表于 03-09 13:39 1次下载

    Linux内核文档:ARM-启动

    Linux内核文档:ARM-启动
    发表于 10-30 10:15 6次下载
    <b class='flag-5'>Linux</b><b class='flag-5'>内核</b>文档:ARM-<b class='flag-5'>启动</b>

    linux内核启动流程

    Linux启动代码真的挺大,从汇编到C,从Makefile到LDS文件,需要理解的东西很多。毕竟Linux内核是由很多人,花费了巨大的时间和精力写出来的。而且直到现在,这个世界上仍然
    发表于 11-14 16:19 4339次阅读
    <b class='flag-5'>linux</b><b class='flag-5'>内核</b><b class='flag-5'>启动</b>流程

    linux内核无法启动

     Linux启动过程中会出现一些故障,导致系统无法正常启动,本文列举了几个应用单用户模式、GRUB命令操作、Linux救援模式的典型故障修复案例帮助读者了解此类问题的解决。
    发表于 11-14 17:26 2975次阅读
    <b class='flag-5'>linux</b><b class='flag-5'>内核</b>无法<b class='flag-5'>启动</b>

    嵌入式Linux内核的驱动程序开发是怎样的

    设备驱动程序linux内核的一部分,是操作系统内核和机器硬件之间的接口,它由一组函数和一些私有数据组成,是连接应用程序与具体硬件的桥梁。
    发表于 11-06 11:33 1483次阅读
    嵌入式<b class='flag-5'>Linux</b><b class='flag-5'>内核</b>的驱动<b class='flag-5'>程序</b>开发是怎样的

    最硬核的Linux内核文章

    内核。 拥有超过1300万行的代码,Linux内核是世界上最大的开源项目之一,但是内核是什么,它用于什么? 02 什么是内核
    的头像 发表于 10-19 17:46 2096次阅读
    最硬核的<b class='flag-5'>Linux</b><b class='flag-5'>内核</b>文章

    快速理解什么是Linux内核以及Linux内核的内容

    01 前言 本文主要讲解什么是Linux内核,以及通过多张图片展示Linux内核的作用与功能,以便于读者能快速理解什么是Linux
    的头像 发表于 10-21 12:02 4258次阅读
    快速理解什么是<b class='flag-5'>Linux</b><b class='flag-5'>内核</b>以及<b class='flag-5'>Linux</b><b class='flag-5'>内核</b>的内容

    如何使用Linux内核实现USB驱动程序框架

    Linux内核提供了完整的USB驱动程序框架。USB总线采用树形结构,在一条总线上只能有唯一的主机设备。 Linux内核从主机和设备两个角度
    发表于 11-06 17:59 20次下载
    如何使用<b class='flag-5'>Linux</b><b class='flag-5'>内核</b>实现USB驱动<b class='flag-5'>程序</b>框架

    Linux内核启动流程(下)

    本篇是通用内核启动阶段,一般是C语言实现。
    发表于 06-23 14:08 519次阅读
    <b class='flag-5'>Linux</b><b class='flag-5'>内核</b><b class='flag-5'>启动</b>流程(下)

    linux驱动程序如何加载进内核

    ,需要了解Linux内核的基本概念和API。以下是一些关键概念: 1.1 内核模块:Linux内核模块是一种动态加载和卸载的代码,可以在不重
    的头像 发表于 08-30 15:02 380次阅读