kernel的啟動主要分為兩個(gè)階段。
1、階段一
從入口跳轉(zhuǎn)到start_kernel之前的階段。
對應(yīng)代碼arch/arm/kernel/head.S中stext的實(shí)現(xiàn):
ENTRY(stext)
- 這個(gè)階段主要由匯編語言實(shí)現(xiàn)。
- 這個(gè)階段主要負(fù)責(zé)MMU打開之前的一些操作,以及打開MMU的操作。
- 由于這個(gè)階段MMU還沒有打開,并且kernel加載地址和連接地址并一致,所以需要使用位置無關(guān)設(shè)計(jì)。在運(yùn)行過程中運(yùn)行地址和加載地址一致(如果不明白的話建議先參考一下《[kernel 啟動流程] 前篇——vmlinux.lds分析》)。
(上一篇從uboot到kernel的地方,講了kernel啟動后的幾個(gè)階段,停在start_kernel部分)
2、階段二
start_kernel開始的階段。
正題-kernel-uboot
Android生在linux內(nèi)核基礎(chǔ)上,linux內(nèi)核啟動的最后一步,一定是啟動的android的進(jìn)程。
然后我們也知道了內(nèi)核啟動分為三個(gè)階段,
- 第一二是運(yùn)行head.S文件和head-common.S,
- 第三個(gè)階段是允許第二是運(yùn)行main.c文件。
對于ARM的處理器,內(nèi)核第一個(gè)啟動的文件是arc/arm/kernel下面的head.S文件。、
當(dāng)然arc/arm/boot/compress下面 也有這個(gè)文件,這個(gè)文件和上面的文件略有不同,當(dāng)要生成壓縮的內(nèi)核時(shí)zImage時(shí), 啟動的是后者 ,后者與前者不同的是:它前面的代碼是做自解壓的,后面的代碼都相同。
我們這里這分析arc/arm/kernel下面的head.S文件。當(dāng)head.S所作的工作完成后它會跳到init/目錄下跌的 main.c的start_kernel函數(shù)開始執(zhí)行。
因?yàn)槲覀円芯康氖沁^渡階段,而不是整個(gè)啟動流程。(后面會研究的。)這里直接看第三個(gè)--start_kernel階段。
asmlinkage void __init start_kernel(void)
{
…………………….
……………………..
printk(KERN_NOTICE);
printk(linux_banner);
setup_arch(&command_line);
setup_command_line(command_line);
parse_early_param();
parse_args("Booting kernel",static_command_line, __start___param,
__stop___param - __start___param,
&unknown_bootoption);
……………………
…………………………
init_IRQ();
pidhash_init();
init_timers();
hrtimers_init();
softirq_init();
timekeeping_init();
time_init();
profile_init();
…………………………
……………………………
console_init();
………………………………
………………………………
rest_init();
}
從上面可以看出start_kernel首先是打印內(nèi)核信息,然后對bootloader傳進(jìn)來的一些參數(shù)進(jìn)行處理,再接著執(zhí)行各種各樣的初始化,在這其中會初始化控制臺。最后會調(diào)用rest_init();
我們再來看 rest_init ()函數(shù)
static void noinline __init_refok rest_init(void)
__releases(kernel_lock)
{
int pid;
kernel_thread(kernel_init, NULL, CLONE_FS | CLONE_SIGHAND);
............
}
他啟動了kernel_init這個(gè)函數(shù),再來看kerne_init函數(shù)
static int __init kernel_init(void * unused)
{
..............................
if (!ramdisk_execute_command)
ramdisk_execute_command = "/init";
if (sys_access((const char __user *) ramdisk_execute_command, 0) != 0) {
ramdisk_execute_command = NULL;
prepare_namespace();
}
/*
* Ok, we have completed the initial bootup, and
* we're essentially up and running. Get rid of the
* initmem segments and start the user-mode stuff..
*/
init_post();
return 0;
}
kernel_init先調(diào)用了 prepare_namespace() ;然后調(diào)用了init_post函數(shù)
void __init prepare_namespace(void)
{
..........................
mount_root();
.....................
}
可以看出prepare_namespace調(diào)用了mount_root掛接根文件系統(tǒng)。接著kernel_init再執(zhí)行init_post
static int noinline init_post(void)
{
.......................................
/*打開dev/console控制臺,并設(shè)置為標(biāo)準(zhǔn)輸入、輸出*/
if (sys_open((const char __user *) "/dev/console", O_RDWR, 0) < 0)
printk(KERN_WARNING "Warning: unable to open an initial console.n");
(void) sys_dup(0);
(void) sys_dup(0);
if (ramdisk_execute_command) {
run_init_process(ramdisk_execute_command);
printk(KERN_WARNING "Failed to execute %sn",
ramdisk_execute_command);
}
/*
* We try each of these until one succeeds.
*
* The Bourne shell can be used instead of init if we are
* trying to recover a really broken machine.
*/
//如果bootloader指定了init參數(shù),則啟動init參數(shù)指定的進(jìn)程
if (execute_command) {
run_init_process(execute_command);
printk(KERN_WARNING "Failed to execute %s. Attempting "
"defaults...n", execute_command);
}
//如果沒有指定init參數(shù),則分別帶sbin、etc、bin目錄下啟動init進(jìn)程
run_init_process("/sbin/init");
run_init_process("/etc/init");
run_init_process("/bin/init");
run_init_process("/bin/sh");
panic("No init found. Try passing init= option to kernel.");
}
注意上面的run_init_process的會等待init進(jìn)程返回才往后面執(zhí)行,所有它一旦找到一個(gè)init可執(zhí)行的文件它將一去不復(fù)返。
綜上,內(nèi)核啟動的過程大致為以下幾步:
- 1.檢查CPU和機(jī)器類型
- 2.進(jìn)行堆棧、MMU等其他程序運(yùn)行關(guān)鍵的東西進(jìn)行初始化
- 3.打印內(nèi)核信息
- 4.執(zhí)行各種模塊的初始化
- 5.掛接根文件系統(tǒng)
- 6.啟動第一個(gè)init進(jìn)程
- 7.android啟動
-
Android
+關(guān)注
關(guān)注
12文章
3958瀏覽量
129076 -
Linux
+關(guān)注
關(guān)注
87文章
11403瀏覽量
212074 -
MMU
+關(guān)注
關(guān)注
0文章
92瀏覽量
18541 -
Kernel
+關(guān)注
關(guān)注
0文章
48瀏覽量
11473
發(fā)布評論請先 登錄
相關(guān)推薦
求助,以下兩個(gè)ADC轉(zhuǎn)換的通道,分別是CC2541上的哪兩個(gè)引腳啊?
BootLoader啟動過程分為哪幾個(gè)階段呢
技術(shù)分享|TQ3568如何更換啟動logo(包括uboot和kernel)
倫敦區(qū)塊鏈銀行金融初創(chuàng)公司BABB啟動ICO 欲分兩個(gè)階段穩(wěn)扎穩(wěn)打
兩個(gè)接觸器如何實(shí)現(xiàn)順序啟動
從兩個(gè)大階段來看新能源電動汽車的發(fā)展前景
兩個(gè)變頻器不能啟動的故障處理案例分享

兩個(gè)LED和兩個(gè)按鈕的使用

kernel到android核心啟動過程

評論