資料介紹
針對好多Linux 愛好者對內核很有興趣卻無從下口,本文旨在介紹一種解讀linux內核源碼的入門方法,而不是解說linux復雜的內核機制;
一.核心源程序的文件組織:
1.Linux核心源程序通常都安裝在/usr/src/linux下,而且它有一個非常簡單的編號約定:任何偶數的核心(例如2.0.30)都是一個穩定地發行的核心,而任何奇數的核心(例如2.1.42)都是一個開發中的核心。
本文基于穩定的2.2.5源代碼,第二部分的實現平臺為 Redhat Linux 6.0。
2.核心源程序的文件按樹形結構進行組織,在源程序樹的最上層你會看到這樣一些目錄:
●Arch :arch子目錄包括了所有和體系結構相關的核心代碼。它的每一個子目錄都代表一種支持的體系結構,例如i386就是關于intel cpu及與之相兼容體系結構的子目錄。PC機一般都基于此目錄;
●Include: include子目錄包括編譯核心所需要的大部分頭文件。與平臺無關的頭文件在 include/linux子目錄下,與 intel cpu相關的頭文件在include/asm-i386子目錄下,而include/scsi目錄則是有關scsi設備的頭文件目錄;
●Init: 這個目錄包含核心的初始化代碼(注:不是系統的引導代碼),包含兩個文件main.c和Version.c,這是研究核心如何工作的一個非常好的起點。
●Mm :這個目錄包括所有獨立于 cpu 體系結構的內存管理代碼,如頁式存儲管理內存的分配和釋放等;而和體系結構相關的內存管理代碼則位于arch/*/mm/,例如arch/i386/mm/Fault.c
●Kernel:主要的核心代碼,此目錄下的文件實現了大多數linux系統的內核函數,其中最重要的文件當屬sched.c;同樣,和體系結構相關的代碼在arch/*/kernel中;
●Drivers: 放置系統所有的設備驅動程序;每種驅動程序又各占用一個子目錄:如,/block 下為塊設備驅動程序,比如ide(ide.c)。如果你希望查看所有可能包含文件系統的設備是如何初始化的,你可以看drivers/block/genhd.c中的device_setup()。它不僅初始化硬盤,也初始化網絡,因為安裝nfs文件系統的時候需要網絡其他: 如, Lib放置核心的庫代碼; Net,核心與網絡相關的代碼; Ipc,這個目錄包含核心的進程間通訊的代碼; Fs ,所有的文件系統代碼和各種類型的文件操作代碼,它的每一個子目錄支持一個文件系統,例如fat和ext2;
scripts, 此目錄包含用于配置核心的腳本文件等。
一般,在每個目錄下,都有一個 .depend 文件和一個 Makefile 文件,這兩個文件都是編譯時使用的輔助文件,仔細閱讀這兩個文件對弄清各個文件這間的聯系和依托關系很有幫助;而且,在有的目錄下還有Readme 文件,它是對該目錄下的文件的一些說明,同樣有利于我們對內核源碼的理解;
二.解讀實戰:為你的內核增加一個系統調用
雖然,Linux 的內核源碼用樹形結構組織得非常合理、科學,把功能相關聯的文件都放在同一個子目錄下,這樣使得程序更具可讀性。然而,Linux 的內核源碼實在是太大而且非常復雜,即便采用了很合理的文件組織方法,在不同目錄下的文件之間還是有很多的關聯,分析核心的一部分代碼通常會要查看其它的幾個相關的文件,而且可能這些文件還不在同一個子目錄下。
體系的龐大復雜和文件之間關聯的錯綜復雜,可能就是很多人對其望而生畏的主要原因。當然,這種令人生畏的勞動所帶來的回報也是非常令人著迷的:你不僅可以從中學到很多的計算機的底層的知識(如下面將講到的系統的引導),體會到整個操作系統體系結構的精妙和在解決某個具體細節問題時,算法的巧妙;而且更重要的是:在源碼的分析過程中,你就會被一點一點地、潛移默化地專業化;甚至,只要分析十分之一的代碼后,你就會深刻地體會到,什么樣的代碼才是一個專業的程序員寫的,什么樣的代碼是一個業余愛好者寫的。
為了使讀者能更好的體會到這一特點,下面舉了一個具體的內核分析實例,希望能通過這個實例,使讀者對 Linux的內核的組織有些具體的認識,從中讀者也可以學到一些對內核的分析方法。
以下即為分析實例:
【一】操作平臺:
硬件:cpu intel Pentium II
軟件:Redhat Linux 6.0; 內核版本2.2.5【二】相關內核源代碼分析:
1.系統的引導和初始化:Linux 系統的引導有好幾種方式:常見的有 Lilo, Loadin引導和Linux的自舉引導
(bootsect-loader),而后者所對應源程序為arch/i386/boot/bootsect.S,它為實模式的匯編程序,限于篇幅在此不做分析;無論是哪種引導方式,最后都要跳轉到 arch/i386/Kernel/setup.S, setup.S主要是進行時模式下的初始化,為系統進入保護模式做準備;此后,系統執行 arch/i386/kernel/head.S (對經壓縮后存放的內核要先執行 arch/i386/boot/compressed/head.S); head.S 中定義的一段匯編程序setup_idt ,它負責建立一張256項的 idt 表(Interrupt Descriptor Table),此表保存著所有自陷和中斷的入口地址;其中包括系統調用總控程序 system_call 的入口地址;當然,除此之外,head.S還要做一些其他的初始化工作;
2.系統初始化后運行的第一個內核程序asmlinkage void __init start_kernel(void) 定義在
/usr/src/linux/init/main.c中,它通過調用usr/src/linux/arch/i386/kernel/traps.c 中的一個函數
void __init trap_init(void) 把各自陷和中斷服務程序的入口地址設置到 idt 表中,其中系統調用總控程序system_cal就是中斷服務程序之一;void __init trap_init(void) 函數則通過調用一個宏set_system_gate(SYSCALL_VECTOR,&system_call); 把系統調用總控程序的入口掛在中斷0x80上; 其中SYSCALL_VECTOR是定義在 /usr/src/linux/arch/i386/kernel/irq.h中的一個常量0x80; 而 system_call 即為中斷總控程序的入口地址;中斷總控程序用匯編語言定義在/usr/src/linux/arch/i386/kernel/entry.S中;
3.中斷總控程序主要負責保存處理機執行系統調用前的狀態,檢驗當前調用是否合法, 并根據系統調用向量,使處理機跳轉到保存在 sys_call_table 表中的相應系統服務例程的入口; 從系統服務例程返回后恢復處理機狀態退回用戶程序;
而系統調用向量則定義在/usr/src/linux/include/asm-386/unistd.h 中;sys_call_table 表定義在/usr/src/linux/arch/i386/kernel/entry.S 中; 同時在 /usr/src/linux/include/asm-386/unistd.h 中也定義了系統調用的用戶編程接口;
一.核心源程序的文件組織:
1.Linux核心源程序通常都安裝在/usr/src/linux下,而且它有一個非常簡單的編號約定:任何偶數的核心(例如2.0.30)都是一個穩定地發行的核心,而任何奇數的核心(例如2.1.42)都是一個開發中的核心。
本文基于穩定的2.2.5源代碼,第二部分的實現平臺為 Redhat Linux 6.0。
2.核心源程序的文件按樹形結構進行組織,在源程序樹的最上層你會看到這樣一些目錄:
●Arch :arch子目錄包括了所有和體系結構相關的核心代碼。它的每一個子目錄都代表一種支持的體系結構,例如i386就是關于intel cpu及與之相兼容體系結構的子目錄。PC機一般都基于此目錄;
●Include: include子目錄包括編譯核心所需要的大部分頭文件。與平臺無關的頭文件在 include/linux子目錄下,與 intel cpu相關的頭文件在include/asm-i386子目錄下,而include/scsi目錄則是有關scsi設備的頭文件目錄;
●Init: 這個目錄包含核心的初始化代碼(注:不是系統的引導代碼),包含兩個文件main.c和Version.c,這是研究核心如何工作的一個非常好的起點。
●Mm :這個目錄包括所有獨立于 cpu 體系結構的內存管理代碼,如頁式存儲管理內存的分配和釋放等;而和體系結構相關的內存管理代碼則位于arch/*/mm/,例如arch/i386/mm/Fault.c
●Kernel:主要的核心代碼,此目錄下的文件實現了大多數linux系統的內核函數,其中最重要的文件當屬sched.c;同樣,和體系結構相關的代碼在arch/*/kernel中;
●Drivers: 放置系統所有的設備驅動程序;每種驅動程序又各占用一個子目錄:如,/block 下為塊設備驅動程序,比如ide(ide.c)。如果你希望查看所有可能包含文件系統的設備是如何初始化的,你可以看drivers/block/genhd.c中的device_setup()。它不僅初始化硬盤,也初始化網絡,因為安裝nfs文件系統的時候需要網絡其他: 如, Lib放置核心的庫代碼; Net,核心與網絡相關的代碼; Ipc,這個目錄包含核心的進程間通訊的代碼; Fs ,所有的文件系統代碼和各種類型的文件操作代碼,它的每一個子目錄支持一個文件系統,例如fat和ext2;
scripts, 此目錄包含用于配置核心的腳本文件等。
一般,在每個目錄下,都有一個 .depend 文件和一個 Makefile 文件,這兩個文件都是編譯時使用的輔助文件,仔細閱讀這兩個文件對弄清各個文件這間的聯系和依托關系很有幫助;而且,在有的目錄下還有Readme 文件,它是對該目錄下的文件的一些說明,同樣有利于我們對內核源碼的理解;
二.解讀實戰:為你的內核增加一個系統調用
雖然,Linux 的內核源碼用樹形結構組織得非常合理、科學,把功能相關聯的文件都放在同一個子目錄下,這樣使得程序更具可讀性。然而,Linux 的內核源碼實在是太大而且非常復雜,即便采用了很合理的文件組織方法,在不同目錄下的文件之間還是有很多的關聯,分析核心的一部分代碼通常會要查看其它的幾個相關的文件,而且可能這些文件還不在同一個子目錄下。
體系的龐大復雜和文件之間關聯的錯綜復雜,可能就是很多人對其望而生畏的主要原因。當然,這種令人生畏的勞動所帶來的回報也是非常令人著迷的:你不僅可以從中學到很多的計算機的底層的知識(如下面將講到的系統的引導),體會到整個操作系統體系結構的精妙和在解決某個具體細節問題時,算法的巧妙;而且更重要的是:在源碼的分析過程中,你就會被一點一點地、潛移默化地專業化;甚至,只要分析十分之一的代碼后,你就會深刻地體會到,什么樣的代碼才是一個專業的程序員寫的,什么樣的代碼是一個業余愛好者寫的。
為了使讀者能更好的體會到這一特點,下面舉了一個具體的內核分析實例,希望能通過這個實例,使讀者對 Linux的內核的組織有些具體的認識,從中讀者也可以學到一些對內核的分析方法。
以下即為分析實例:
【一】操作平臺:
硬件:cpu intel Pentium II
軟件:Redhat Linux 6.0; 內核版本2.2.5【二】相關內核源代碼分析:
1.系統的引導和初始化:Linux 系統的引導有好幾種方式:常見的有 Lilo, Loadin引導和Linux的自舉引導
(bootsect-loader),而后者所對應源程序為arch/i386/boot/bootsect.S,它為實模式的匯編程序,限于篇幅在此不做分析;無論是哪種引導方式,最后都要跳轉到 arch/i386/Kernel/setup.S, setup.S主要是進行時模式下的初始化,為系統進入保護模式做準備;此后,系統執行 arch/i386/kernel/head.S (對經壓縮后存放的內核要先執行 arch/i386/boot/compressed/head.S); head.S 中定義的一段匯編程序setup_idt ,它負責建立一張256項的 idt 表(Interrupt Descriptor Table),此表保存著所有自陷和中斷的入口地址;其中包括系統調用總控程序 system_call 的入口地址;當然,除此之外,head.S還要做一些其他的初始化工作;
2.系統初始化后運行的第一個內核程序asmlinkage void __init start_kernel(void) 定義在
/usr/src/linux/init/main.c中,它通過調用usr/src/linux/arch/i386/kernel/traps.c 中的一個函數
void __init trap_init(void) 把各自陷和中斷服務程序的入口地址設置到 idt 表中,其中系統調用總控程序system_cal就是中斷服務程序之一;void __init trap_init(void) 函數則通過調用一個宏set_system_gate(SYSCALL_VECTOR,&system_call); 把系統調用總控程序的入口掛在中斷0x80上; 其中SYSCALL_VECTOR是定義在 /usr/src/linux/arch/i386/kernel/irq.h中的一個常量0x80; 而 system_call 即為中斷總控程序的入口地址;中斷總控程序用匯編語言定義在/usr/src/linux/arch/i386/kernel/entry.S中;
3.中斷總控程序主要負責保存處理機執行系統調用前的狀態,檢驗當前調用是否合法, 并根據系統調用向量,使處理機跳轉到保存在 sys_call_table 表中的相應系統服務例程的入口; 從系統服務例程返回后恢復處理機狀態退回用戶程序;
而系統調用向量則定義在/usr/src/linux/include/asm-386/unistd.h 中;sys_call_table 表定義在/usr/src/linux/arch/i386/kernel/entry.S 中; 同時在 /usr/src/linux/include/asm-386/unistd.h 中也定義了系統調用的用戶編程接口;
下載該資料的人也在下載
下載該資料的人還在閱讀
更多 >
- Linux內核文件Cache機制
- 嵌入式LINUX系統內核和內核模塊調試
- Linux內核開發框架學習資料匯總 24次下載
- Linux環境編程:從應用到內核 18次下載
- Linux內核的編譯與運行 16次下載
- 如何使用Linux內核實現USB驅動程序框架 20次下載
- 如何才能編譯Linux的內核 8次下載
- 嵌入式Linux與物聯網軟件開發C語言內核深度解析書籍的介紹
- Linux入門教程之Linux的內核詳細資料概述 10次下載
- Linux入門教程之LINUX入門與安裝配置 4次下載
- linux內核入門教材之linux內核設計與實現第二版中文版免費下載 0次下載
- Linux操作系統下C語言編程入門.pdf 96次下載
- LINUX系統引導和初始化-LINUX內核解讀 53次下載
- Linux內核解讀入門
- Linux的內核教程 0次下載
- Linux內核中的頁面分配機制 323次閱讀
- 使用 PREEMPT_RT 在 Ubuntu 中構建實時 Linux 內核 2658次閱讀
- 獲取Linux內核源碼的方法 694次閱讀
- Linux內核自解壓過程分析 955次閱讀
- Linux系統內核與Linux發行套件的區別 1212次閱讀
- Linux內核到底是什么應該如何學習 1925次閱讀
- Linux:QEMU調試內核的步驟 3172次閱讀
- 深入linux內核架構 Linux內核架構分析解讀 3806次閱讀
- Linux內核中有哪些鎖 3467次閱讀
- Linux 5.4.1內核已經發布你期待使用嗎 3310次閱讀
- Linux內核與Android的關系 4606次閱讀
- macOS和Linux的內核之間的區別詳解 5690次閱讀
- Linux內核地址映射模型與Linux內核高端內存詳解 3474次閱讀
- 嵌入式未來還是Linux的天下,并通過內核學習來闡述kernel的機理 3770次閱讀
- Linux內核開發工具介紹 4720次閱讀
下載排行
本周
- 1AN158 GD32VW553 Wi-Fi開發指南
- 1.51MB | 2次下載 | 免費
- 2嵌入式軟件開發符合ISO 26262 功能安全標準
- 1.61 MB | 1次下載 | 免費
- 3AN148 GD32VW553射頻硬件開發指南
- 2.07MB | 1次下載 | 免費
- 4PZT驅動開關電路
- 0.09 MB | 1次下載 | 免費
- 5模擬電路仿真實現
- 2.94MB | 1次下載 | 免費
- 6PCB繪制基礎知識
- 2.44MB | 1次下載 | 免費
- 7EE-38:ADSP-2181 IDMA端口-周期竊取時序
- 75.0KB | 次下載 | 免費
- 8EE-82:使用ADSP-2181 DSP的IO空間對另一個ADSP-2181進行IDMA引導
- 86.86KB | 次下載 | 免費
本月
- 1ADI高性能電源管理解決方案
- 2.43 MB | 452次下載 | 免費
- 2免費開源CC3D飛控資料(電路圖&PCB源文件、BOM、
- 5.67 MB | 139次下載 | 1 積分
- 3基于STM32單片機智能手環心率計步器體溫顯示設計
- 0.10 MB | 132次下載 | 免費
- 4美的電磁爐維修手冊大全
- 1.56 MB | 24次下載 | 5 積分
- 5如何正確測試電源的紋波
- 0.36 MB | 19次下載 | 免費
- 6感應筆電路圖
- 0.06 MB | 10次下載 | 免費
- 7LZC3106G高性能諧振控制器中文手冊
- 1.29 MB | 9次下載 | 1 積分
- 8萬用表UT58A原理圖
- 0.09 MB | 9次下載 | 5 積分
總榜
- 1matlab軟件下載入口
- 未知 | 935121次下載 | 10 積分
- 2開源硬件-PMP21529.1-4 開關降壓/升壓雙向直流/直流轉換器 PCB layout 設計
- 1.48MB | 420062次下載 | 10 積分
- 3Altium DXP2002下載入口
- 未知 | 233088次下載 | 10 積分
- 4電路仿真軟件multisim 10.0免費下載
- 340992 | 191367次下載 | 10 積分
- 5十天學會AVR單片機與C語言視頻教程 下載
- 158M | 183335次下載 | 10 積分
- 6labview8.5下載
- 未知 | 81581次下載 | 10 積分
- 7Keil工具MDK-Arm免費下載
- 0.02 MB | 73810次下載 | 10 積分
- 8LabVIEW 8.6下載
- 未知 | 65988次下載 | 10 積分
評論