這節的 HelloWDM 是基於 WDM 的驅動程式, 和前面介紹的 HelloDDK 非常相似且多了對隨插即用功能的支持. NT 式驅動程式和 WDM 驅動程式的異同將在後面章節介紹.
HelloWDM 的標頭檔 :
HelloWDM 的標頭檔主要是為了匯入驅動程式開發所必需的 WDM.h 標頭檔, 此標頭檔裡包含了對DDK 所有匯出函式的宣告. NT 式的驅動要匯入的是 NTDDK.h, 而 WDM 驅動要匯入的標頭檔為 WDM.h. 另外此標頭檔中定義了幾個標籤, 分別在程式中指名函式和變數分配在分頁記憶體中或非分頁記憶體中, 最後該標頭檔列出了此驅動程式的函式宣告.
HelloWDM 的入口函式 :
和 NT 式驅動程式一樣, WDM 的入口函式位址同樣是 DriverEntry, 且在 C++ 編譯的同時需要用 extern "C" 修飾.
程式碼第 9 行, 將此函式放在 INIT 段中, 當驅動載入結束後, 此函式就可以從記憶體中卸掉.
程式碼第 10-11 行, 定義驅動入口函式, 用 extern "C" 修飾. 其中傳進兩個參數, 第一個參數 pDriverObject 為驅動元件, 第二個參數 pRegistryPath 為此驅動的登錄表路徑.
程式碼第 15 行, 設置 AddDevice 的 Callback 函式, 此函式出現在 WDM 驅動程式中, 而在 NT 式驅動中沒有此 Callback 函式. 此 Callback 函式的作用是新建裝置並由 PNP (隨插即用) 管理器呼叫.
程式碼第 16 行, 設置對 IRP_MJ_PNP 的 Callback 函式. 對 PNP 的 IRP 處理, 是 NT 式驅動和 WDM 驅動重大區別之一.
程式碼第 17-20 行, 設置常用的 IRP Callback 函式. 這裡只是簡單指向一個預設函式 HelloWDMDispatchRoutine.
程式碼第 21 行, 向系統登錄卸載常式. 在 WDM 程式中, 大部分卸載工作以不再此處理, 而是放在對 IRP_MN_REMOVE_DEVICE 的 IRP 的 處理函式中處理.
HelloWDM 的 AddDevice 常式 :
在 WDM 的驅動程式中, 新建裝置的任務不再由 DriverEntry 承擔, 而需要驅動程式向系統登錄一個叫做 AddDevice 的常式. 此常式由 PNP 管理器負責呼叫, 其函式主要職責就是新建裝置物件. HelloWDMAddDevice 常式有兩個參數, DriverObject 和 PhysicalDeviceObject. DriverObject 式由 PNP 管理器傳遞而來的驅動物件, 此物件是 DriverEntry 中的驅動物件. PhysicalDeviceObject 是 PNP 管理器傳遞進來的底層驅動裝置物件, 這個概念在 NT 式的驅動中是沒有的. 關於這個概念會在後續章節說明.
HelloWDM 處理 PNP 的 Callback 函式 :
WDM 式驅動程式主要區別在於對 IRP_MJ_PNP 的 IRP 處理. 其中 IRP_MJ_PNP 會細分為若干個子類. 例如 IRP_MN_START_DEVICE, IRP_MN_REMOVE_DEVICE, IRP_MN_STOP_DEVICE 等. 本例中除了對 IRP_MN_REMOVE_DEVICE 做特殊處理, 其他 IRP 則作相同處理.
HelloWDM 對 PNP 的預設處理 :
除了 IRP_MN_STOP_DEVICE 以外, HelloWDM 對其他 PNP 的 IRP 做同樣的處理, 即直接傳遞到底層驅動, 並將底層驅動結果返回.
HelloWDM 對 IRP_MN_REMOVE_DEVICE 的處理 :
對 IRP_MN_REMOVE_DEVICE 的處理類似於 NT 式驅動中的卸載常式, 而在 WDM 式的驅動中, 卸載常式幾乎不用做處理.
HelloWDM 對其他 IRP 的 Callback 函式 :
此處對新建, 關閉, 讀寫裝置的預設處理同 HelloDDK :
HelloWDM 卸載常式 :
由於 WDM 式驅動程式將主要的卸載任務放在了對 IRP_MN_REMOVE_DEVICE 的處理函式中, 在標準的卸載常式幾乎沒有什麼需要作的. 在這裡只是列印除錯訊息.
HelloWDK 的編譯與安裝 :
HelloWDK 的編譯與安裝與 HelloDDK 過程中有一些不同, 尤其是安裝過程. 這是因為 HelloWDM 是隨插即用的驅動程式, 並且需要藉助 INF 檔安裝.
- 編譯 HelloWDM
編譯 HelloWDM 需要兩個指令檔 makefile 和 Sources. makefile 同 HelloDDK 中一樣, 這裡不再列出, Sources 稍有不同 :
將兩個檔與原始檔案放在同一目錄下, 進入 "Windows XP Checked Build Environment" 編譯環境下. 用 cd 進入對應目錄後執行 build 會編譯出來對應的 HelloWDM.sys 檔案. 執行結果如下 :
- 安裝 HelloWDM
WDM 驅動程式和 NT 式驅動程式的安裝有很大出入, 首先為了安裝 HelloWDM 驅動程式, 需要為驅動程式編寫一個 inf 檔. inf 檔描述了 WDM 驅動程式的操作硬體裝置的資訊和驅動程式的一些資訊. 下面是這個 inf 檔的程式碼, 請將這個檔案和其他原始檔放在同一個目錄裡 :
HelloWDM 是個虛擬裝置, 安裝需要如下步驟 :
1. 首先在第一次安裝 HelloWDM 驅動程式時, 進入控制台, 選擇新增硬體. 系統會自動搜尋是否有新裝置插入, 並彈出對話框詢問是否將硬體連到電腦, 選擇是 如下圖 :
2. 在彈出下一個對話框中選擇 "新增硬體裝置", 並按一下 "下一步" 按鈕如下圖 :
3. 再彈出下一個對話框選擇 "安裝我從清單中手動選取的硬體(進接選項)", 並按下 "下一步" 按鈕 :
4. 再談出下一個對話框中選擇顯示所有裝置, 並按"下一步" 按鈕
5. 在彈出的對話框按下 "從磁片安裝" 紐並選擇之前目錄中的 inf 檔.
6. 最後系統提示安裝成功 :
和 HelloDDK 一樣, 你也可以在裝置管理員中找到這個虛擬裝置, 如下圖 :
這種安裝 WDM 驅動程式作法過於繁雜. 使用一個叫做 ExDriverInstaller 的工具, 可以進行快速安裝. 這個工具也是 DriverStudio 內附的一個工具軟體. 打開 EzDriverInstaller 選擇 "File" | "Open" 在彈出對話框中選擇需要安裝的 inf 檔, 按一下 "Add New Device" 按鈕就可以完成安裝. EzDriverInstaller 還提供刪除驅動程式功能, 開啟/關閉驅動功能, 遮罩驅動功能, 重啟驅動功能等.
沒有留言:
張貼留言