PCI驅(qū)動程序開發(fā)實(shí)例(2)
I/O請求包(IRP)是驅(qū)動程序操作的中心,是一個預(yù)先定義的數(shù)據(jù)結(jié)構(gòu),帶有一組對它進(jìn)行操作的I/O管理器例程。一個IRP有固定的首部和可變數(shù)目的 IRP棧單元。IRP的固定部分含有IRP的固定屬性,每個棧單元含有大多數(shù)有關(guān)的IRP參數(shù)。當(dāng)IRP由多個驅(qū)動程序處理時,使用多個IRP棧單元。每個驅(qū)動程序從當(dāng)前IRP棧單元得到它的IRP參數(shù)。如果把IRP沿當(dāng)前設(shè)備的驅(qū)動程序棧向下傳遞,必須在當(dāng)前驅(qū)動程序中使用正確的參數(shù)設(shè)置下一個棧單元,然后在此驅(qū)動程序中利用函數(shù)IoCalldriver()調(diào)用更低層的驅(qū)動程序。驅(qū)動程序不必處理所有的IRP,但至少需要處理“創(chuàng)建”和“關(guān)閉”這兩個 IRP。I/O管理器接收I/O請求,然后在把它傳遞到合適的驅(qū)動程序棧中的最高驅(qū)動程序之前,分配并初始化IRP。驅(qū)動程序處理IRP的過程如圖2所示。
IRP首先到達(dá)最高層的驅(qū)動程序1,驅(qū)動程序1使用函數(shù)IoGetCurrentIrpStackLocation()獲得指向當(dāng)前棧單元的指針。
然后驅(qū)動程序1使用IoCallDriver()函數(shù)調(diào)用下一個驅(qū)動程序。I/O管理器現(xiàn)在改變“當(dāng)前IRF’棧單元”指針,所以驅(qū)動程序2看到向下的第二個IRP棧單元(驅(qū)動程序1為它設(shè)置的棧單元)。這個過程繼續(xù),直到最底層的的驅(qū)動程序4收到這個IRP。
驅(qū)動程序4現(xiàn)在處理這個IRP。當(dāng)它完成IRP的處理時,驅(qū)動程序4調(diào)用IoCompleteRequest()函數(shù)。指示它已經(jīng)完成IRP的處理。IRP再沿設(shè)備棧向上傳遞,直到它最終彈出棧頂,回到用戶。
2.3 IRP的完成
當(dāng)一個驅(qū)動程序完成對IRP的處理時,它必須告訴I/O管理器,這稱為IRP完成。如下面代碼所示,必須設(shè)置IRP IoStatus域結(jié)構(gòu)中的幾個域。IoStatus,Status設(shè)置為一個NTSTATUS狀態(tài)碼,IoStatus.In-formation通常存儲傳輸?shù)淖止?jié)數(shù)。如:
Irp一>loStatus.Status=S T ATUS_SUCCESS
Irp一>IoStatus.Information=info;
IoCompleteRequest(Irp,IO_NO_INCREMENT);
調(diào)用IoCompleteRequest()表明低層驅(qū)動程序已經(jīng)完成了IRP的請求,并將這個IRP返回給I/O管理器。IO_No_INCREMENT是個系統(tǒng)定義的常量,指定啟動該IRP的優(yōu)先級,需要驅(qū)動程序快速處理。
3 驅(qū)動程序功能實(shí)現(xiàn)
當(dāng)把板卡第一次插到計算機(jī)的PCI插槽以后,計算機(jī)的系統(tǒng)總線會檢測到有個新設(shè)備沒有安裝驅(qū)動程序,并提示安裝驅(qū)動程序。正確地安裝驅(qū)動程序以后,用戶就可以在應(yīng)用程序中與驅(qū)動程序進(jìn)行通信。
3.1 打開設(shè)備
評論