歡迎來到Linux教程網
Linux教程網
Linux教程網
Linux教程網
Linux教程網 >> Linux編程 >> Linux編程 >> Linux Kernel系列一:開篇和Kernel啟動概要

Linux Kernel系列一:開篇和Kernel啟動概要

日期:2017/3/1 10:10:31   编辑:Linux編程

前言

最近幾個月將Linux Kernel的大概研究了一下,下面需要進行深入詳細的分析。主要將以S3C2440的一塊開發板為硬件實體。大概包括如下內容:

1 bootloader分析,以uboot為主,結合具體開發板的情況。我的目標是解釋清楚uboot的工作原理(說實話,分析過程中不太想被硬件綁架,但是需要以一個實際的例子

來做分析)

2 kernel部分,這就很多內容了。打算從kernel啟動的流程開始分析。

3 除kernel本身外,還有很多的知識,例如ld的輸入script分析等,這裡會一起介紹。

kernel啟動流程概要

一:內核Image的組成

1 ES(Embed System)啟動的時候,CPU加電,執行的第一條語句是Bootloader,這個非常類似PC機上的BIOS。BL將內核加載後,控制器移交給LK

2 LK執行的第一條語句是什麼?vmlinux是單體的內核表示。根據前面說的內核編譯連接知識,第一條語句是head.S中(歷史原因,MD,有很多文件都叫head.S)

我們需要重新分析一下內核(這裡就是zImage了)的組成,(方法很簡單,研究make的執行過程,通過make V=1 zImage可以得到幾乎全部信息)

vmlinux,這個是未壓縮、未strip的內核模塊,ELF結構

Image:二進制、未壓縮、但是strip後的內核

head.o:ARM相關的,由BL將控制權轉交給它。即前面提到的head.S生成

pigg.gz:Image文件的gzip壓縮

piggy.o:由piggy.S生成,這個S文件通過include Bin方式將Image包含進來。piggy的意思就是背負、肩扛。很形象不是?

misc.o:從上面看,涉及到一些解壓方面的內容,而misc提供一些輔助函數

vmlinux:悲催.....這個文件是head+pigg+misc構成的vmlinux。名字一樣不是?真的很混淆!

zImage:再由上面這個vmlinux壓縮而來

圖1很好得展示了這個過程。

圖1 內核的構成

3 piggy的故事

piggy.S很有意思,建立了一個section,並且有一個標志來指示piggy.gz的邊界。

piggy對應的是一個叫bootstrap的image,注意,Bootstrap和Bootloader不一樣,它是在BL之後的一段代碼,用來

解壓kernel,設置內存等作用。也可以叫second stage boot。

4 Bootloadre和BootstrapLoader

BL和BSL的區別是什麼?

BL只是初始化硬件,不依賴linux,不處理linux

BSL在BL後執行,依賴linux,因為要解壓linux。另外一個重要點就是BSL需要為LINUX的運行建立環境

BSL的工作包括:

head.O:初始化CPU等工作

misc.O:解壓,重定位(例如將kernel移動到另外一個位置上) decompress_kernel

其他工作

init/main.c:start_kernel

啟動調用圖見圖2.

圖2 啟動調用流程圖

下面來分析這個啟動流程

1 kernel中的head.o分析:盡量保持CPU系列的通用,例如arm的CPU等初始化都在做。但是具體板子(例如CPU+其他硬件)怎麼初始化?這就是由mach目錄中的初始化函數做到的。所以,kernel初始化分為:generic CPU初始化+具體板子的初始化。head.o初始化後,跳轉到main.o的start_kernel,繼續後面的流程

2 start_kernel:(init/main.c):start_kernel的轉移由head.O做的,不過代碼一般包含在更通用的head_common.S中

以後想做kernel的分析,就從main開始吧. start_kernel做了什麼事情呢?

剛才只是初始化了cpu相關的,而具體和板子相關的由start_arch執行

3 kernel 參數分析:kernel command line。注意,這個參數是由BL傳遞給kernel的,不過這個參數又是誰設置的呢?又存在什麼地方呢?這個line放在一個global的地方,

另外,kernel如何處理這些參數呢?有一個比較好的辦法,__set_up宏,將一些參數和對應的函數指針存在一個特殊的section中,然後循環調用這個section中的函數。(和驅動module中的很像)。定義在init.h中。關於一些特殊參數的取值,在arch/arm/kernel/vmlinux.lds.S中定義。(以後得去看看ld的manual了)__set_up這個宏還有一個flags比如early,表示處理階段是否在early-stage做。標志有__init的section最終占用的內存會被拋棄..

4 子系統初始化:包括中斷、等。?section嵌套section?

5 kernel_init進程:start_kernel最後會fork一個kernel_init進程,而原執行進程變成idle進程了..

6 用戶空間的init進程:由kernel_init進程最終通過execve init完成

Linux Kernel系列 相關閱讀:http://www.linuxidc.com/search.aspx?Where=Nkey&Keyword=Linux+Kernel%e7%b3%bb%e5%88%97

Copyright © Linux教程網 All Rights Reserved