本書是一本介紹Linux內(nèi)核實踐的入門書,基于Linux 4.0內(nèi)核,重點講解Linux內(nèi)核的理論和實驗。本書分為12章,包括Linux系統(tǒng)入門、Linux內(nèi)核基礎(chǔ)知識、內(nèi)核編譯和調(diào)試、內(nèi)核模塊、簡單的字符設(shè)備驅(qū)動、系統(tǒng)調(diào)用、內(nèi)存管理、進(jìn)程管理、同步管理、中斷管理、調(diào)試和性能優(yōu)化,以及如何參與開源社區(qū)等內(nèi)容。此外,本書還介紹了Linux內(nèi)核社區(qū)常用的開發(fā)工具和理論,如Vim 8和git工具等。書中包括70多個實驗,幫助讀者深入理解Linux內(nèi)核。
1.原理 基礎(chǔ),從調(diào)試Linux內(nèi)核到動手編寫一個簡單的內(nèi)核模塊,真正從零開始學(xué)習(xí)Linux內(nèi)核。
2.實驗 案例,70余個創(chuàng)新實驗貫穿全書,與你分享業(yè)內(nèi)一線項目經(jīng)驗。
3.課件 視頻,教學(xué)資源豐富,視頻講解透徹,與你共享精品在線配套資源(可在異步社區(qū)下載)。
4.涵蓋當(dāng)前Linux社區(qū)中新的開發(fā)工具和社區(qū)運作方式,反映Linux內(nèi)核社區(qū)新發(fā)展。
張?zhí)祜w,筆名笨叔叔,Linux內(nèi)核愛好者,從事Linux內(nèi)核和驅(qū)動開發(fā)十余年,曾在多家芯片公司從事過手機芯片底層軟件開發(fā)和客戶支持工作。著有《奔跑吧 Linux內(nèi)核》一書。
第 1章 Linux系統(tǒng)入門1
1.1 Linux的發(fā)展歷史 1
1.2 Linux發(fā)行版 2
1.2.1 Red Hat Linux 2
1.2.2 Debian Linux 3
1.2.3 SuSE Linux 4
1.2.4 優(yōu)麒麟Linux 4
1.3 Linux內(nèi)核 5
1.3.1 宏內(nèi)核和微內(nèi)核 5
1.3.2 Linux內(nèi)核概貌 6
1.4 如何學(xué)習(xí)Linux內(nèi)核 9
1.5 Linux內(nèi)核實驗入門 10
1.5.1 實驗1:在虛擬機中安裝優(yōu)麒麟Linux 18.04系統(tǒng) 10
1.5.2 實驗2:給優(yōu)麒麟Linux系統(tǒng)更換心臟 14
1.5.3 實驗3:使用定制的內(nèi)核runninglinuxkernel 15
1.5.4 實驗4:如何編譯和運行一個ARM Linux內(nèi)核 19
第 2章 Linux內(nèi)核基礎(chǔ)知識 22
2.1 Linux常用的編譯工具 22
2.1.1 GCC工具 22
2.1.2 ARM GCC 23
2.1.3 GCC編譯 24
2.2 Linux內(nèi)核中常用的C語言技巧 25
2.3 Linux內(nèi)核中常用的數(shù)據(jù)結(jié)構(gòu)和算法 31
2.3.1 鏈表 31
2.3.2 紅黑樹 34
2.3.3 無鎖環(huán)形緩沖區(qū) 36
2.4 Vim工具的使用 38
2.4.1 Vim 8介紹 38
2.4.2 Vim的基本模式 38
2.4.3 Vim中3種模式的切換 39
2.4.4 Vim光標(biāo)的移動 40
2.4.5 刪除、復(fù)制和粘貼 41
2.4.6 查找和替換 41
2.4.7 文件相關(guān) 41
2.5 git工具的使用 42
2.5.1 安裝git 43
2.5.2 git基本操作 43
2.5.3 分支管理 46
2.6 實驗 48
2.6.1 實驗1:GCC編譯 48
2.6.2 實驗2:內(nèi)核鏈表 51
2.6.3 實驗3:紅黑樹 52
2.6.4 實驗4:使用Vim工具 52
2.6.5 實驗5:把Vim打造成一個強大的IDE編輯工具 52
2.6.6 實驗6:建立一個git本地倉庫 60
2.6.7 實驗7:解決合并分支沖突 62
2.6.8 實驗8:利用git來管理Linux內(nèi)核開發(fā) 65
2.6.9 實驗9:利用git來管理項目代碼 67
第3章 內(nèi)核編譯和調(diào)試 73
3.1 內(nèi)核配置 73
3.1.1 內(nèi)核配置工具 73
3.1.2 .config文件 74
3.2 實驗1:通過QEMU調(diào)試ARM Linux內(nèi)核 76
3.3 實驗2:通過QEMU調(diào)試ARMv8的Linux內(nèi)核 78
3.4 實驗3:通過Eclipse QEMU單步調(diào)試內(nèi)核 81
3.5 實驗4:在QEMU中添加文件系統(tǒng)的支持 85
第4章 內(nèi)核模塊 86
4.1 從一個內(nèi)核模塊開始 86
4.2 模塊參數(shù) 90
4.3 符號共享 92
4.4 實驗 93
4.4.1 實驗1:編寫一個簡單的內(nèi)核模塊 93
4.4.2 實驗2:向內(nèi)核模塊傳遞參數(shù) 95
4.4.3 實驗3:在模塊之間導(dǎo)出符號 95
第5章 簡單的字符設(shè)備驅(qū)動 96
5.1 實驗1:從一個簡單的字符設(shè)備開始 97
5.2 字符設(shè)備驅(qū)動詳解 102
5.2.1 字符設(shè)備驅(qū)動的抽象 102
5.2.2 設(shè)備號的管理 104
5.2.3 設(shè)備節(jié)點 104
5.2.4 字符設(shè)備操作方法集 105
5.3 實驗2:使用misc機制來創(chuàng)建設(shè)備 107
5.4 一個簡單的虛擬設(shè)備 109
5.4.1 實驗3:為虛擬設(shè)備編寫驅(qū)動 109
5.4.2 實驗4:使用KFIFO改進(jìn)設(shè)備驅(qū)動 112
5.5 阻塞I O和非阻塞I O 115
5.5.1 實驗5:把虛擬設(shè)備驅(qū)動改成非阻塞模式 115
5.5.2 實驗6:把虛擬設(shè)備驅(qū)動改成阻塞模式 118
5.6 I O多路復(fù)用 122
5.6.1 Linux的I O多路復(fù)用 122
5.6.2 實驗7:向虛擬設(shè)備中添加I O多路復(fù)用支持 123
5.7 實驗8:為什么不能喚醒讀寫進(jìn)程 128
5.8 實驗9:向虛擬設(shè)備中添加異步通知 129
5.9 本章小結(jié) 133
第6章 系統(tǒng)調(diào)用 134
6.1 系統(tǒng)調(diào)用概念 134
6.1.1 系統(tǒng)調(diào)用和POSIX標(biāo)準(zhǔn) 135
6.1.2 系統(tǒng)調(diào)用表 135
6.1.3 用程序訪問系統(tǒng)調(diào)用 136
6.1.4 新增系統(tǒng)調(diào)用 137
6.2 實驗 137
6.2.1 實驗1:在ARM32機器上新增一個系統(tǒng)調(diào)用 137
6.2.2 實驗2:在優(yōu)麒麟Linux機器上新增一個系統(tǒng)調(diào)用 138
第7章 內(nèi)存管理 139
7.1 從硬件角度看內(nèi)存管理 139
7.1.1 內(nèi)存管理的遠(yuǎn)古時代 139
7.1.2 分段機制 141
7.1.3 分頁機制 142
7.1.4 虛擬地址到物理地址的轉(zhuǎn)換 143
7.2 從軟件角度看內(nèi)存管理 144
7.2.1 free命令 144
7.2.2 從應(yīng)用編程角度看內(nèi)存管理 145
7.2.3 從內(nèi)存布局圖角度看內(nèi)存管理 146
7.2.4 從進(jìn)程角度看內(nèi)存管理 150
7.3 物理內(nèi)存管理 154
7.3.1 物理頁面 155
7.3.2 內(nèi)存管理區(qū) 159
7.3.3 分配和釋放頁面 162
7.3.4 分配小塊內(nèi)存 170
7.4 虛擬內(nèi)存管理 177
7.4.1 進(jìn)程地址空間 177
7.4.2 內(nèi)存描述符mm_struct 178
7.4.3 VMA管理 180
7.4.4 malloc分配函數(shù) 183
7.4.5 mmap 185
7.5 缺頁異常 188
7.5.1 do_page_fault函數(shù) 189
7.5.2 匿名頁面缺頁異常 190
7.5.3 文件映射缺頁中斷 190
7.5.4 寫時復(fù)制缺頁異常 191
7.5.5 缺頁異常小結(jié) 192
7.6 內(nèi)存短缺 193
7.6.1 頁面回收算法 193
7.6.2 OOM Killer機制 194
7.7 內(nèi)存管理實驗 195
7.7.1 實驗1:查看系統(tǒng)內(nèi)存信息 195
7.7.2 實驗2:獲取系統(tǒng)的物理內(nèi)存信息 197
7.7.3 實驗3:分配內(nèi)存 199
7.7.4 實驗4:slab 200
7.7.5 實驗5:VMA 201
7.7.6 實驗6:mmap 203
7.7.7 實驗7:映射用戶內(nèi)存 203
7.7.8 實驗8:OOM 204
第8章 進(jìn)程管理 205
8.1 進(jìn)程 205
8.1.1 進(jìn)程的來由 205
8.1.2 進(jìn)程描述符 207
8.1.3 進(jìn)程的生命周期 209
8.1.4 進(jìn)程標(biāo)識 212
8.1.5 進(jìn)程間的家族關(guān)系 212
8.1.6 獲取當(dāng)前進(jìn)程 214
8.2 進(jìn)程的創(chuàng)建和終止 216
8.2.1 寫時復(fù)制技術(shù) 216
8.2.2 fork()函數(shù) 217
8.2.3 vfork()函數(shù) 218
8.2.4 clone()函數(shù) 218
8.2.5 內(nèi)核線程 219
8.2.6 do_fork()函數(shù) 219
8.2.7 終止進(jìn)程 221
8.2.8 僵尸進(jìn)程和托孤進(jìn)程 222
8.2.9 進(jìn)程0和進(jìn)程1 222
8.3 進(jìn)程調(diào)度 223
8.3.1 進(jìn)程分類 224
8.3.2 進(jìn)程優(yōu)先級 224
8.3.3 時間片 225
8.3.4 經(jīng)典調(diào)度算法 225
8.3.5 Linux O(n)調(diào)度算法 228
8.3.6 Linux O(1)調(diào)度算法 228
8.3.7 Linux CFS調(diào)度算法 228
8.3.8 進(jìn)程切換 233
8.3.9 與調(diào)度相關(guān)的數(shù)據(jù)結(jié)構(gòu) 239
8.4 多核調(diào)度 241
8.4.1 調(diào)度域和調(diào)度組 241
8.4.2 負(fù)載計算 244
8.4.3 負(fù)載均衡算法 245
8.5 實驗 246
8.5.1 實驗1:fork和clone 246
8.5.2 實驗2:內(nèi)核線程 247
8.5.3 實驗3:后臺守護進(jìn)程 247
8.5.4 實驗4:進(jìn)程權(quán)限 247
8.5.5 實驗5:設(shè)置優(yōu)先級 247
8.5.6 實驗6:per-cpu變量 248
第9章 同步管理 250
9.1 原子操作與內(nèi)存屏障 251
9.1.1 原子操作 251
9.1.2 內(nèi)存屏障 253
9.2 自旋鎖機制 254
9.2.1 自旋鎖定義 254
9.2.2 自旋鎖變種 256
9.2.3 自旋鎖和raw_spin_lock 257
9.2.4 自旋鎖的改進(jìn) 257
9.3 信號量 258
9.4 互斥體 259
9.5 讀寫鎖 261
9.5.1 讀寫鎖定義 261
9.5.2 讀寫信號量 262
9.6 RCU 264
9.7 等待隊列 267
9.7.1 等待隊列頭 267
9.7.2 等待隊列節(jié)點 268
9.8 實驗 269
9.8.1 實驗1:自旋鎖 269
9.8.2 實驗2:互斥鎖 269
9.8.3 實驗3:RCU 269
第 10章 中斷管理 270
10.1 Linux中斷管理機制 270
10.1.1 ARM中斷控制器 271
10.1.2 硬件中斷號和Linux中斷號的映射 275
10.1.3 注冊中斷 276
10.2 軟中斷和tasklet 278
10.2.1 SoftIRQ軟中斷 279
10.2.2 tasklet 280
10.2.3 local_bh_disable local_bh_enable 281
10.2.4 本節(jié)小結(jié) 282
10.3 工作隊列機制 282
10.3.1 工作隊列類型 283
10.3.2 使用工作隊列 285
10.3.3 本節(jié)小結(jié) 285
10.4 實驗 286
10.4.1 實驗1:tasklet 286
10.4.2 實驗2:工作隊列 286
10.4.3 實驗3:定時器和內(nèi)核線程 287
第 11章 調(diào)試和性能優(yōu)化 288
11.1 printk和動態(tài)輸出 289
11.1.1 printk輸出函數(shù) 289
11.1.2 動態(tài)輸出 290
11.1.3 實驗1:printk 292
11.1.4 實驗2:動態(tài)輸出 292
11.2 proc和debugfs 293
11.2.1 proc文件系統(tǒng) 293
11.2.2 sys文件系統(tǒng) 295
11.2.3 debugfs 296
11.2.4 實驗3:procfs 297
11.2.5 實驗4:sysfs 298
11.2.6 實驗5:debugfs 300
11.3 ftrace 301
11.3.1 irqs跟蹤器 302
11.3.2 preemptoff跟蹤器 304
11.3.3 preemptirqsoff跟蹤器 305
11.3.4 function跟蹤器 306
11.3.5 動態(tài)ftrace 307
11.3.6 事件跟蹤 308
11.3.7 實驗6:使用frace 310
11.3.8 實驗7:添加一個新的跟蹤點 311
11.3.9 實驗8:使用示蹤標(biāo)志 314
11.3.10 實驗9:使用kernelshark來分析數(shù)據(jù) 317
11.4 實驗10:分析oops錯誤 319
11.5 perf性能分析工具 323
11.5.1 實驗11:使用perf工具來進(jìn)行性能分析 328
11.5.2 實驗12:采集perf數(shù)據(jù)生成火焰圖 329
11.6 內(nèi)存檢測 329
11.6.1 實驗13:使用slub_debug檢查內(nèi)存泄漏 330
11.6.2 實驗14:使用kmemleak檢查內(nèi)存泄漏 335
11.6.3 實驗15:使用kasan檢查內(nèi)存泄漏 337
11.6.4 實驗16:使用valgrind檢查內(nèi)存泄漏 340
11.7 實驗17:kdump 342
11.8 性能和測試 348
11.8.1 性能測試概述 348
11.8.2 實驗18:使用lkp-tests工具進(jìn)行性能測試 349
第 12章 開源社區(qū) 350
12.1 什么是開源社區(qū) 350
12.1.1 開源軟件的發(fā)展歷史 350
12.1.2 Linux基金會 351
12.1.3 開源協(xié)議 351
12.1.4 Linux內(nèi)核社區(qū) 353
12.1.5 國內(nèi)開源社區(qū) 354
12.2 參與開源社區(qū) 354
12.2.1 參與開源項目的好處 354
12.2.2 如何參與開源項目 355
12.3 實驗1:使用cppcheck檢查代碼 356
12.4 實驗2:提交第 一個Linux內(nèi)核補丁 357
12.5 實驗3:管理和提交多個補丁組成的補丁集 359
12.6 實驗4:在Gitee中創(chuàng)建一個開源項目 363
參考文獻(xiàn) 366