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

完善資料讓更多小伙伴認(rèn)識(shí)你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

基于linux內(nèi)核啟動(dòng)程序

嵌入式小生 ? 來(lái)源:嵌入式小生 ? 作者:嵌入式小生 ? 2022-10-14 09:11 ? 次閱讀

linux內(nèi)核啟動(dòng)過(guò)程的后期,在kernel_init()函數(shù)代表的init線程中,會(huì)嘗試執(zhí)行用戶空間的init進(jìn)程:

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

從上述代碼可見(jiàn),會(huì)嘗試執(zhí)行/sbin/、/etc、/bin三個(gè)目錄中的init。從《busybox源碼分析筆記(一)》一文可以知道,在busybox編譯構(gòu)建完成并安裝后,會(huì)生成對(duì)應(yīng)的目錄(注:/etc目錄不存在)。在/sbin目錄中,則會(huì)存在一個(gè)init鏈接:

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

查看其屬性,其本質(zhì)則是鏈接到了../bin/busybox:

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

綜上所述,證明linux內(nèi)核啟動(dòng)后期,運(yùn)行的第一個(gè)用戶空間程序是init,在busybox源碼中,init程序則由位于/init目錄中的init.c編譯構(gòu)建而成,程序入口是:init_main(),小生在該函數(shù)中添加一行標(biāo)識(shí)代碼:

a9913efe-4b57-11ed-a3b6-dac502259ad0.pnglinux內(nèi)核運(yùn)行后期的結(jié)果如下:

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

可見(jiàn),linux內(nèi)核后期加載的就是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)*/
}

跳過(guò)條件宏定義下的編譯分支,主要分析其通用的代碼部分:

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

(2)然后找出系統(tǒng)默認(rèn)的控制臺(tái),并將路徑切換到/,接著重新創(chuàng)建一個(gè)新的會(huì)話:

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

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

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

(4)如果是單用戶模式,則會(huì)調(diào)用new_init_action(RESPAWN, bb_default_login_shell, "");在控制臺(tái)啟動(dòng)一個(gè)shell;否則,則會(huì)調(diào)用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,則會(huì)使用/etc/inittab文件中的action;如果CONFIG_FEATURE_USE_INITTAB沒(méi)有定義,parse_inittab()則會(huì)簡(jiǎn)單添加一些默認(rèn)actions(例如,運(yùn)行INIT_SCRIPT,然后啟動(dòng)一個(gè)“askfirst”shell)。如果ENABLE_FEATURE_USE_INITTAB已定義,但是/etc/inittab文件缺失也會(huì)使用相同的默認(rèn)行為集合。

(5)設(shè)置STOP信號(hào)處理程序:

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)接下來(lái)運(yùn)行想要運(yùn)行的命令,依次運(yùn)行: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)在最后,則是一個(gè)while(1)的死循環(huán),用于處理我們?cè)诿钚邢螺斎氲拿睿?/p>

while(1){
/*重新運(yùn)行respawn/askfisrt*/
run_actions(RESPAWN|ASKFIRST);

/*等待信號(hào)*/
check_delayed_sigs(NULL);/*NULLtimespecmakesitwait*/

/*等待所有的子進(jìn)程退出*/
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時(shí)間*/
sleep1();
}/*while(1)*/

補(bǔ)充

BusyBox的init不支持運(yùn)行級(jí)別。runlevels字段在BusyBox init中將會(huì)被完全忽略。

所以如果想要使用運(yùn)行級(jí)別的系統(tǒng),需使用sysvinit作為啟動(dòng)進(jìn)程。

一、7個(gè)運(yùn)行級(jí)別

(1)運(yùn)行級(jí)別0:系統(tǒng)停機(jī)狀態(tài),系統(tǒng)默認(rèn)運(yùn)行級(jí)別不能設(shè)為0,否則不能正常啟動(dòng)。其實(shí)就是關(guān)機(jī)。

(2)運(yùn)行級(jí)別1:?jiǎn)斡脩艄ぷ鳡顟B(tài),root權(quán)限,用于系統(tǒng)維護(hù),禁止遠(yuǎn)程登陸。在忘記root密碼時(shí)一般用這個(gè)運(yùn)行級(jí)別,進(jìn)去修改root密碼。

(3)運(yùn)行級(jí)別2:多用戶狀態(tài)(沒(méi)有NFS),沒(méi)有網(wǎng)絡(luò)連接。

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

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

(6)運(yùn)行級(jí)別5:X11控制臺(tái),登陸后進(jìn)入圖形GUI模式。就是圖形模式。

(7)運(yùn)行級(jí)別6:系統(tǒng)正常關(guān)閉并重啟,默認(rèn)運(yùn)行級(jí)別不能設(shè)為6,否則不能正常啟動(dòng)。

二、查看運(yùn)行級(jí)別

1、runlevel命令:打印系統(tǒng)的上一個(gè)和當(dāng)前運(yùn)行級(jí)別:

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

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

審核編輯:彭靜
聲明:本文內(nèi)容及配圖由入駐作者撰寫(xiě)或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點(diǎn)僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場(chǎng)。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問(wèn)題,請(qǐng)聯(lián)系本站處理。 舉報(bào)投訴
  • 函數(shù)
    +關(guān)注

    關(guān)注

    3

    文章

    4264

    瀏覽量

    62252
  • 代碼
    +關(guān)注

    關(guān)注

    30

    文章

    4701

    瀏覽量

    68122
  • LINUX內(nèi)核
    +關(guān)注

    關(guān)注

    1

    文章

    316

    瀏覽量

    21588
收藏 人收藏

    評(píng)論

    相關(guān)推薦

    Linux編譯驅(qū)動(dòng)、內(nèi)核及應(yīng)用程序分析

    作為一名嵌入式Linux新手,在學(xué)習(xí)的過(guò)程中會(huì)遇到很多問(wèn)題。寫(xiě)了一個(gè)驅(qū)動(dòng)程序怎么編譯?怎么加載進(jìn)內(nèi)核?
    的頭像 發(fā)表于 01-17 13:46 ?6564次閱讀
    <b class='flag-5'>Linux</b>編譯驅(qū)動(dòng)、<b class='flag-5'>內(nèi)核</b>及應(yīng)用<b class='flag-5'>程序</b>分析

    Linux內(nèi)核啟動(dòng)過(guò)程和Bootloader(總述)

    精簡(jiǎn)講述linux內(nèi)核啟動(dòng)過(guò)程。[轉(zhuǎn)]1.Linux內(nèi)核啟動(dòng)過(guò)程概述 一個(gè)嵌入式
    發(fā)表于 08-18 17:35

    【OK210試用體驗(yàn)】bootloader啟動(dòng)linux內(nèi)核

    的性能和穩(wěn)定性。用戶空間的文件系統(tǒng)用來(lái)提供管理系統(tǒng)的各種配置,提供相應(yīng)的應(yīng)用程序、服務(wù)、數(shù)據(jù)交換等。文件系統(tǒng)作為一種載體,它是用來(lái)實(shí)現(xiàn)用戶與操作系統(tǒng)內(nèi)核的交互。因此,一個(gè)可啟動(dòng)linux
    發(fā)表于 01-10 15:57

    Linux內(nèi)核編譯和啟動(dòng)的相關(guān)資料分享

    Linux環(huán)境下,我們想運(yùn)行一個(gè)應(yīng)用程序,在shell交互環(huán)境下直接敲命令就可以了,操作系統(tǒng)給程序提供了運(yùn)行環(huán)境和進(jìn)程管理。那Linux操作系統(tǒng)本身是如何運(yùn)行和
    發(fā)表于 12-20 06:28

    Android的Linux內(nèi)核與驅(qū)動(dòng)程序開(kāi)發(fā)教程

    Android內(nèi)核是基于Linux 2.6內(nèi)核的,它是一個(gè)增強(qiáng)內(nèi)核版本,除了修改部分Bug外,它提供了用于支持Android平臺(tái)的設(shè)備驅(qū)動(dòng),這里介紹了Android
    發(fā)表于 09-05 14:24 ?335次下載
    Android的<b class='flag-5'>Linux</b><b class='flag-5'>內(nèi)核</b>與驅(qū)動(dòng)<b class='flag-5'>程序</b>開(kāi)發(fā)教程

    linux內(nèi)核啟動(dòng)內(nèi)核解壓過(guò)程分析

    linux啟動(dòng)時(shí)內(nèi)核解壓過(guò)程分析,一份不錯(cuò)的文檔,深入了解內(nèi)核必備
    發(fā)表于 03-09 13:39 ?1次下載

    Linux內(nèi)核文檔:ARM-啟動(dòng)

    Linux內(nèi)核文檔:ARM-啟動(dòng)
    發(fā)表于 10-30 10:15 ?6次下載
    <b class='flag-5'>Linux</b><b class='flag-5'>內(nèi)核</b>文檔:ARM-<b class='flag-5'>啟動(dòng)</b>

    linux內(nèi)核啟動(dòng)流程

    Linux啟動(dòng)代碼真的挺大,從匯編到C,從Makefile到LDS文件,需要理解的東西很多。畢竟Linux內(nèi)核是由很多人,花費(fèi)了巨大的時(shí)間和精力寫(xiě)出來(lái)的。而且直到現(xiàn)在,這個(gè)世界上仍然
    發(fā)表于 11-14 16:19 ?4318次閱讀
    <b class='flag-5'>linux</b><b class='flag-5'>內(nèi)核</b><b class='flag-5'>啟動(dòng)</b>流程

    linux內(nèi)核無(wú)法啟動(dòng)

     Linux啟動(dòng)過(guò)程中會(huì)出現(xiàn)一些故障,導(dǎo)致系統(tǒng)無(wú)法正常啟動(dòng),本文列舉了幾個(gè)應(yīng)用單用戶模式、GRUB命令操作、Linux救援模式的典型故障修復(fù)案例幫助讀者了解此類問(wèn)題的解決。
    發(fā)表于 11-14 17:26 ?2896次閱讀
    <b class='flag-5'>linux</b><b class='flag-5'>內(nèi)核</b>無(wú)法<b class='flag-5'>啟動(dòng)</b>

    嵌入式Linux內(nèi)核的驅(qū)動(dòng)程序開(kāi)發(fā)是怎樣的

    設(shè)備驅(qū)動(dòng)程序linux內(nèi)核的一部分,是操作系統(tǒng)內(nèi)核和機(jī)器硬件之間的接口,它由一組函數(shù)和一些私有數(shù)據(jù)組成,是連接應(yīng)用程序與具體硬件的橋梁。
    發(fā)表于 11-06 11:33 ?1455次閱讀
    嵌入式<b class='flag-5'>Linux</b><b class='flag-5'>內(nèi)核</b>的驅(qū)動(dòng)<b class='flag-5'>程序</b>開(kāi)發(fā)是怎樣的

    最硬核的Linux內(nèi)核文章

    內(nèi)核。 擁有超過(guò)1300萬(wàn)行的代碼,Linux內(nèi)核是世界上最大的開(kāi)源項(xiàng)目之一,但是內(nèi)核是什么,它用于什么? 02 什么是內(nèi)核
    的頭像 發(fā)表于 10-19 17:46 ?2073次閱讀
    最硬核的<b class='flag-5'>Linux</b><b class='flag-5'>內(nèi)核</b>文章

    快速理解什么是Linux內(nèi)核以及Linux內(nèi)核的內(nèi)容

    01 前言 本文主要講解什么是Linux內(nèi)核,以及通過(guò)多張圖片展示Linux內(nèi)核的作用與功能,以便于讀者能快速理解什么是Linux
    的頭像 發(fā)表于 10-21 12:02 ?4221次閱讀
    快速理解什么是<b class='flag-5'>Linux</b><b class='flag-5'>內(nèi)核</b>以及<b class='flag-5'>Linux</b><b class='flag-5'>內(nèi)核</b>的內(nèi)容

    如何使用Linux內(nèi)核實(shí)現(xiàn)USB驅(qū)動(dòng)程序框架

    Linux內(nèi)核提供了完整的USB驅(qū)動(dòng)程序框架。USB總線采用樹(shù)形結(jié)構(gòu),在一條總線上只能有唯一的主機(jī)設(shè)備。 Linux內(nèi)核從主機(jī)和設(shè)備兩個(gè)角度
    發(fā)表于 11-06 17:59 ?19次下載
    如何使用<b class='flag-5'>Linux</b><b class='flag-5'>內(nèi)核</b>實(shí)現(xiàn)USB驅(qū)動(dòng)<b class='flag-5'>程序</b>框架

    Linux內(nèi)核啟動(dòng)流程(下)

    本篇是通用內(nèi)核啟動(dòng)階段,一般是C語(yǔ)言實(shí)現(xiàn)。
    發(fā)表于 06-23 14:08 ?492次閱讀
    <b class='flag-5'>Linux</b><b class='flag-5'>內(nèi)核</b><b class='flag-5'>啟動(dòng)</b>流程(下)

    linux驅(qū)動(dòng)程序如何加載進(jìn)內(nèi)核

    ,需要了解Linux內(nèi)核的基本概念和API。以下是一些關(guān)鍵概念: 1.1 內(nèi)核模塊:Linux內(nèi)核模塊是一種動(dòng)態(tài)加載和卸載的代碼,可以在不重
    的頭像 發(fā)表于 08-30 15:02 ?298次閱讀