0
  • 聊天消息
  • 系統(tǒng)消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會員中心
創(chuàng)作中心

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

3天內不再提示

linux內核啟動過程會執(zhí)行用戶空間的init進程

嵌入式小生 ? 來源:嵌入式小生 ? 作者:iriczhao ? 2022-10-14 09:12 ? 次閱讀

linux內核啟動過程的后期,在kernel_init()函數(shù)代表的init線程中,會嘗試執(zhí)行用戶空間的init進程:

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

從上述代碼可見,會嘗試執(zhí)行/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(),小生在該函數(shù)中添加一行標識代碼:

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

linux內核運行后期的結果如下:

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

可見,linux內核后期加載的就是busybox下的init程序。

init_main分析

貼上該函數(shù)的完整代碼,下文將分段描述:

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)然后找出系統(tǒng)默認的控制臺,并將路徑切換到/,接著重新創(chuàng)建一個新的會話:

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

(3)設置默認的環(huán)境變量:

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()函數(shù)。

parse_inittab()函數(shù)定義如下:

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/dev/TTY*/
if(tty[0]){
tty=concat_path_file("/dev/",skip_dev_pfx(tty));
}
new_init_action(1<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)的死循環(huán),用于處理我們在命令行下輸入的命令:

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中將會被完全忽略。

所以如果想要使用運行級別的系統(tǒng),需使用sysvinit作為啟動進程。

一、7個運行級別

(1)運行級別0:系統(tǒng)停機狀態(tài),系統(tǒng)默認運行級別不能設為0,否則不能正常啟動。其實就是關機。

(2)運行級別1:單用戶工作狀態(tài),root權限,用于系統(tǒng)維護,禁止遠程登陸。在忘記root密碼時一般用這個運行級別,進去修改root密碼。

(3)運行級別2:多用戶狀態(tài)(沒有NFS),沒有網絡連接。

(4)運行級別3:完全的多用戶狀態(tài)(有NFS),登陸后進入控制臺命令行模式。linux很常見的運行級別

(5)運行級別4:系統(tǒng)未使用,保留。

(6)運行級別5:X11控制臺,登陸后進入圖形GUI模式。就是圖形模式。

(7)運行級別6:系統(tǒng)正常關閉并重啟,默認運行級別不能設為6,否則不能正常啟動。

二、查看運行級別

1、runlevel命令:打印系統(tǒng)的上一個和當前運行級別:

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

N:“N”表示自系統(tǒng)啟動后運行級別尚未更改。從上圖可見,小生的Ubuntu系統(tǒng)的運行級別為5。





審核編輯:劉清

聲明:本文內容及配圖由入駐作者撰寫或者入駐合作網站授權轉載。文章觀點僅代表作者本人,不代表電子發(fā)燒友網立場。文章及其配圖僅供工程師學習之用,如有內容侵權或者其他違規(guī)問題,請聯(lián)系本站處理。 舉報投訴
  • busybox
    +關注

    關注

    0

    文章

    21

    瀏覽量

    11243
  • GUI
    GUI
    +關注

    關注

    3

    文章

    631

    瀏覽量

    39287
  • LINUX內核
    +關注

    關注

    1

    文章

    315

    瀏覽量

    21556
  • NFS
    NFS
    +關注

    關注

    1

    文章

    52

    瀏覽量

    26050
  • Ubuntu系統(tǒng)

    關注

    0

    文章

    85

    瀏覽量

    3859

原文標題:busybox源碼分析筆記(二) | init程序

文章出處:【微信號:嵌入式小生,微信公眾號:嵌入式小生】歡迎添加關注!文章轉載請注明出處。

收藏 人收藏

    評論

    相關推薦

    Linux內核啟動過程和Bootloader(總述)

    函數(shù),它主要完成剩余的與硬件平臺相關的初始化工作,在進行一系列與內核相關的初始化后,調用第一個用戶進程init 進程并等待
    發(fā)表于 08-18 17:35

    android--系統(tǒng)啟動--init進程啟動過程如何

    android--系統(tǒng)啟動--init進程啟動過程
    發(fā)表于 05-29 10:35

    【HarmonyOS HiSpark IPC DIY Camera試用連載 】鴻蒙OS內核如何啟動第一個用戶進程init_lite

    ,當前包括:Hi3516DV300平臺和Hi3518EV300平臺。 負責處理從內核加載第一個用戶態(tài)進程開始,到第一個應用程序啟動之間的系統(tǒng)服務進程
    發(fā)表于 11-20 10:27

    [文章] 【HarmonyOS HiSpark IPC DIY Camera試用連載 】鴻蒙OS內核如何啟動第一個用戶進程init_lite

    :Hi3516DV300平臺和Hi3518EV300平臺。負責處理從內核加載第一個用戶態(tài)進程開始,到第一個應用程序啟動之間的系統(tǒng)服務進程
    發(fā)表于 11-20 16:44

    Linux內核啟動過程解析

    Uart驅動卻把串口設備名寫死了,如本例中linux2.6.37串口設備名為ttyO0,而不是常用的ttyS0。有了控制臺內核啟動過程中就可以通過串口輸出信息以便開發(fā)者或用戶了解系統(tǒng)
    發(fā)表于 10-26 17:20

    嵌入式uCLinux內核啟動過程分析

    分析uCLinux的啟動過程,可以加快系統(tǒng)啟動速度、正確建立應用環(huán)境。本文要研究的就是uCLinux操作系統(tǒng)內核啟動過程
    發(fā)表于 08-15 16:51 ?760次閱讀

    init函數(shù)和init進程的區(qū)別

    由于執(zhí)行init()函數(shù)的內核線程和init進程進程標識符都是1,它們又都叫
    發(fā)表于 01-05 10:49 ?1650次閱讀

    Linux基礎命令之Linux啟動過程詳解

    。 2.2.1 概述 用戶開機啟動Linux過程如下: (1)當用戶打開PC(intel CPU)的電源時,CPU將自動進入實模式,并從地址
    發(fā)表于 10-18 14:17 ?2次下載
    <b class='flag-5'>Linux</b>基礎命令之<b class='flag-5'>Linux</b><b class='flag-5'>啟動過程</b>詳解

    試述shell的啟動過程詳情

    進程結束后wait函數(shù)返回,從而shell收到通知并回收資源。本文主要說明shell如何啟動用戶進程Linux系統(tǒng)中可
    發(fā)表于 11-15 11:06 ?3679次閱讀
    試述shell的<b class='flag-5'>啟動過程</b>詳情

    詳解bootloader的執(zhí)行流程與ARM Linux啟動過程分析

    S3C2410 ARM處理器為例,詳細分析了系統(tǒng)上電后 bootloader的執(zhí)行流程及 ARM Linux啟動過程。
    的頭像 發(fā)表于 12-21 09:24 ?1w次閱讀
    詳解bootloader的<b class='flag-5'>執(zhí)行</b>流程與ARM <b class='flag-5'>Linux</b><b class='flag-5'>啟動過程</b>分析

    走進Linux之systemd啟動過程

    Linux系統(tǒng)的啟動方式有點復雜,而且總是有需要優(yōu)化的地方。傳統(tǒng)的Linux系統(tǒng)啟動過程主要由著名的init
    發(fā)表于 04-27 19:14 ?3105次閱讀

    Linux系統(tǒng)下init進程的前世今生

    kernel_thread產生的進程,其開始在內核態(tài)執(zhí)行,然后通過一個系統(tǒng)調用,開始執(zhí)行用戶空間
    發(fā)表于 04-28 17:23 ?923次閱讀

    解析基于ARM64的init用戶進程究竟如何啟動

    [導讀] 前面的文章有提到linux啟動的第一個進程init,那么該進程究竟是如何從內核
    發(fā)表于 01-26 17:05 ?2次下載
    解析基于ARM64的<b class='flag-5'>init</b><b class='flag-5'>用戶</b><b class='flag-5'>進程</b>究竟如何<b class='flag-5'>啟動</b>?

    kernel執(zhí)行第一個init應用程序的實現(xiàn)原理

    Linux系統(tǒng)啟動過程中通過`init_task`創(chuàng)建0號idle進程。然后通過`kernel_thread`創(chuàng)建1號init
    的頭像 發(fā)表于 06-05 14:53 ?710次閱讀
    kernel<b class='flag-5'>執(zhí)行</b>第一個<b class='flag-5'>init</b>應用程序的實現(xiàn)原理

    kernel到android核心啟動過程

    總結一個圖:kernel 到android核心啟動過程 kernel鏡像執(zhí)行跳轉到start_kernel開始執(zhí)行,在rest_init創(chuàng)
    的頭像 發(fā)表于 12-04 16:59 ?628次閱讀
    kernel到android核心<b class='flag-5'>啟動過程</b>