## 4.1.?內核中的調試支持
在第 2 章, 我們建議你建立并安裝你自己的內核, 而不是運行來自你的發布商的現成的內核. 運行你自己的內核的最充分的理由之一是內核開發者已經在內核自身中構建了多個調試特性. 這些特性能產生額外的輸出并降低性能, 因此發布商的產品內核中往往不會使能它們. 作為一個內核開發者, 但是, 你有不同的優先權并且會樂于接收這些格外的內核調試支持帶來的開銷.
這里, 我們列出用來開發的內核應當激活的配置選項. 除了另外指出的, 所有的這些選項都在 "kernel hacking" 菜單, 不管什么樣的你喜歡的內核配置工具. 注意有些選項不是所有體系都支持.
CONFIG_DEBUG_KERNEL
這個選項只是使其他調試選項可用; 它應當打開, 但是它自己不激活任何的特性.
CONFIG_DEBUG_SLAB
這個重要的選項打開了內核內存分配函數的幾類檢查; 激活這些檢查, 就可能探測到一些內存覆蓋和遺漏初始化的錯誤. 被分配的每一個字節在遞交給調用者之前都設成 0xa5, 隨后在釋放時被設成 0x6b. 你在任何時候如果見到任一個這種"壞"模式重復出現在你的驅動輸出(或者常常在一個 oops 的列表), 你會確切知道去找什么類型的錯誤. 當激活調試, 內核還會在每個分配的內存對象的前后放置特別的守護值; 如果這些值曾被改動, 內核知道有人已覆蓋了一個內存分配區, 它大聲抱怨. 各種的對更模糊的問題的檢查也給激活了.
CONFIG_DEBUG_PAGEALLOC
滿的頁在釋放時被從內核地址空間去除. 這個選項會顯著拖慢系統, 但是它也能快速指出某些類型的內存損壞錯誤.
CONFIG_DEBUG_SPINLOCK
激活這個選項, 內核捕捉對未初始化的自旋鎖的操作, 以及各種其他的錯誤( 例如 2 次解鎖同一個鎖 ).
CONFIG_DEBUG_SPINLOCK_SLEEP
這個選項激活對持有自旋鎖時進入睡眠的檢查. 實際上, 如果你調用一個可能會睡眠的函數, 它就抱怨, 即便這個有疑問的調用沒有睡眠.
CONFIG_INIT_DEBUG
用__init (或者 __initdata) 標志的項在系統初始化或者模塊加載后都被丟棄. 這個選項激活了對代碼的檢查, 這些代碼試圖在初始化完成后存取初始化時內存.
CONFIG_DEBUG_INFO
這個選項使得內核在建立時包含完整的調試信息. 如果你想使用 gdb 調試內核, 你將需要這些信息. 如果你打算使用 gdb, 你還要激活 CONFIG_FRAME_POINTER.
CONFIG_MAGIC_SYSRQ
激活"魔術 SysRq"鍵. 我們在本章后面的"系統掛起"一節查看這個鍵.
CONFIG_DEBUG_STACKOVERFLOWCONFIG_DEBUG_STACK_USAGE
這些選項能幫助跟蹤內核堆棧溢出. 堆棧溢出的確證是一個 oops 輸出, 但是沒有任何形式的合理的回溯. 第一個選項給內核增加了明確的溢出檢查; 第 2 個使得內核監測堆棧使用并作一些統計, 這些統計可以用魔術 SysRq 鍵得到.
CONFIG_KALLSYMS
這個選項(在"Generl setup/Standard features"下)使得內核符號信息建在內核中; 缺省是激活的. 符號選項用在調試上下文中; 沒有它, 一個 oops 列表只能以 16 進制格式給你一個內核回溯, 這不是很有用.
CONFIG_IKCONFIGCONFIG_IKCONFIG_PROC
這些選項(在"Generl setup"菜單)使得完整的內核配置狀態被建立到內核中, 可以通過 /proc 來使其可用. 大部分內核開發者知道他們使用的哪個配置, 并不需要這些選項(會使得內核更大). 但是如果你試著調試由其他人建立的內核中的問題, 它們可能有用.
CONFIG_ACPI_DEBUG
在"Power management/ACPI"下. 這個選項打開詳細的 ACPI (Advanced Configuration and Power Interface) 調試信息, 它可能有用如果你懷疑一個問題和 ACPI 相關.
CONFIG_DEBUG_DRIVER
在"Device drivers"下. 打開了驅動核心的調試信息, 可用以追蹤低層支持代碼的問題. 我們在第 14 章查看驅動核心.
CONFIG_SCSI_CONSTANTS
這個選項, 在"Device drivers/SCSI device support"下, 建立詳細的 SCSI 錯誤消息的信息. 如果你在使用 SCSI 驅動, 你可能需要這個選項.
CONFIG_INPUT_EVBUG
這個選項(在"Device drivers/Input device support"下)打開輸入事件的詳細日志. 如果你使用一個輸入設備的驅動, 這個選項可能會有用. 然而要小心這個選項的安全性的隱含意義: 它記錄了你鍵入的任何東西, 包括你的密碼.
CONFIG_PROFILING
這個選項位于"Profiling support"之下. 剖析通常用在系統性能調整, 但是在追蹤一些內核掛起和相關問題上也有用.
我們會再次遇到一些上面的選項, 當我們查看各種方法來追蹤內核問題時. 但是首先, 我們要看一下經典的調試技術: print 語句.
- Linux設備驅動第三版
- 第 1 章 設備驅動簡介
- 1.1. 驅動程序的角色
- 1.2. 劃分內核
- 1.3. 設備和模塊的分類
- 1.4. 安全問題
- 1.5. 版本編號
- 1.6. 版權條款
- 1.7. 加入內核開發社團
- 1.8. 本書的內容
- 第 2 章 建立和運行模塊
- 2.1. 設置你的測試系統
- 2.2. Hello World 模塊
- 2.3. 內核模塊相比于應用程序
- 2.4. 編譯和加載
- 2.5. 內核符號表
- 2.6. 預備知識
- 2.7. 初始化和關停
- 2.8. 模塊參數
- 2.9. 在用戶空間做
- 2.10. 快速參考
- 第 3 章 字符驅動
- 3.1. scull 的設計
- 3.2. 主次編號
- 3.3. 一些重要數據結構
- 3.4. 字符設備注冊
- 3.5. open 和 release
- 3.6. scull 的內存使用
- 3.7. 讀和寫
- 3.8. 使用新設備
- 3.9. 快速參考
- 第 4 章 調試技術
- 4.1. 內核中的調試支持
- 4.2. 用打印調試
- 4.3. 用查詢來調試
- 4.4. 使用觀察來調試
- 4.5. 調試系統故障
- 4.6. 調試器和相關工具
- 第 5 章 并發和競爭情況
- 5.1. scull 中的缺陷
- 5.2. 并發和它的管理
- 5.3. 旗標和互斥體
- 5.4. Completions 機制
- 5.5. 自旋鎖
- 5.6. 鎖陷阱
- 5.7. 加鎖的各種選擇
- 5.8. 快速參考
- 第 6 章 高級字符驅動操作
- 6.1. ioctl 接口
- 6.2. 阻塞 I/O
- 6.3. poll 和 select
- 6.4. 異步通知
- 6.5. 移位一個設備
- 6.6. 在一個設備文件上的存取控制
- 6.7. 快速參考
- 第 7 章 時間, 延時, 和延后工作
- 7.1. 測量時間流失
- 7.2. 獲知當前時間
- 7.3. 延后執行
- 7.4. 內核定時器
- 7.5. Tasklets 機制
- 7.6. 工作隊列
- 7.7. 快速參考
- 第 8 章 分配內存
- 8.1. kmalloc 的真實故事
- 8.2. 后備緩存
- 8.3. get_free_page 和其友
- 8.4. 每-CPU 的變量
- 8.5. 獲得大量緩沖
- 8.6. 快速參考
- 第 9 章 與硬件通訊
- 9.1. I/O 端口和 I/O 內存
- 9.2. 使用 I/O 端口
- 9.3. 一個 I/O 端口例子
- 9.4. 使用 I/O 內存
- 9.5. 快速參考
- 第 10 章 中斷處理
- 10.1. 準備并口
- 10.2. 安裝一個中斷處理
- 10.3. 前和后半部
- 10.4. 中斷共享
- 10.5. 中斷驅動 I/O
- 10.6. 快速參考
- 第 11 章 內核中的數據類型
- 11.1. 標準 C 類型的使用
- 11.2. 安排一個明確大小給數據項
- 11.3. 接口特定的類型
- 11.4. 其他移植性問題
- 11.5. 鏈表
- 11.6. 快速參考
- 第 12 章 PCI 驅動
- 12.1. PCI 接口
- 12.2. 回顧: ISA
- 12.3. PC/104 和 PC/104+
- 12.4. 其他的 PC 總線
- 12.5. SBus
- 12.6. NuBus 總線
- 12.7. 外部總線
- 12.8. 快速參考
- 第 13 章 USB 驅動
- 13.1. USB 設備基礎知識
- 13.2. USB 和 sysfs
- 13.3. USB 的 Urbs
- 13.4. 編寫一個 USB 驅動
- 13.5. 無 urb 的 USB 傳送
- 13.6. 快速參考
- 第 14 章 Linux 設備模型
- 14.1. Kobjects, Ksets 和 Subsystems
- 14.2. 低級 sysfs 操作
- 14.3. 熱插拔事件產生
- 14.4. 總線, 設備, 和驅動
- 14.5. 類
- 14.6. 集成起來
- 14.7. 熱插拔
- 14.8. 處理固件
- 14.9. 快速參考
- 第 15 章 內存映射和 DMA
- 15.1. Linux 中的內存管理
- 15.2. mmap 設備操作
- 15.3. 進行直接 I/O
- 15.4. 直接內存存取
- 15.5. 快速參考
- 第 16 章 塊驅動
- 16.1. 注冊
- 16.2. 塊設備操作
- 16.3. 請求處理
- 16.4. 一些其他的細節
- 16.5. 快速參考
- 第 17 章 網絡驅動
- 17.1. snull 是如何設計的
- 17.2. 連接到內核
- 17.3. net_device 結構的詳情
- 17.4. 打開與關閉
- 17.5. 報文傳送
- 17.6. 報文接收
- 17.7. 中斷處理
- 17.8. 接收中斷緩解
- 17.9. 連接狀態的改變
- 17.10. Socket 緩存
- 17.11. MAC 地址解析
- 17.12. 定制 ioctl 命令
- 17.13. 統計信息
- 17.14. 多播
- 17.15. 幾個其他細節
- 17.16. 快速參考
- 第 18 章 TTY 驅動
- 18.1. 一個小 TTY 驅動
- 18.2. tty_driver 函數指針
- 18.3. TTY 線路設置
- 18.4. ioctls 函數
- 18.5. TTY 設備的 proc 和 sysfs 處理
- 18.6. tty_driver 結構的細節
- 18.7. tty_operaions 結構的細節
- 18.8. tty_struct 結構的細節
- 18.9. 快速參考