PCI驅動(dòng)程序開(kāi)發(fā)實(shí)例(2)
I/O請求包(IRP)是驅動(dòng)程序操作的中心,是一個(gè)預先定義的數據結構,帶有一組對它進(jìn)行操作的I/O管理器例程。一個(gè)IRP有固定的首部和可變數目的 IRP棧單元。IRP的固定部分含有IRP的固定屬性,每個(gè)棧單元含有大多數有關(guān)的IRP參數。當IRP由多個(gè)驅動(dòng)程序處理時(shí),使用多個(gè)IRP棧單元。每個(gè)驅動(dòng)程序從當前IRP棧單元得到它的IRP參數。如果把IRP沿當前設備的驅動(dòng)程序棧向下傳遞,必須在當前驅動(dòng)程序中使用正確的參數設置下一個(gè)棧單元,然后在此驅動(dòng)程序中利用函數IoCalldriver()調用更低層的驅動(dòng)程序。驅動(dòng)程序不必處理所有的IRP,但至少需要處理“創(chuàng )建”和“關(guān)閉”這兩個(gè) IRP。I/O管理器接收I/O請求,然后在把它傳遞到合適的驅動(dòng)程序棧中的最高驅動(dòng)程序之前,分配并初始化IRP。驅動(dòng)程序處理IRP的過(guò)程如圖2所示。

IRP首先到達最高層的驅動(dòng)程序1,驅動(dòng)程序1使用函數IoGetCurrentIrpStackLocation()獲得指向當前棧單元的指針。
然后驅動(dòng)程序1使用IoCallDriver()函數調用下一個(gè)驅動(dòng)程序。I/O管理器現在改變“當前IRF’棧單元”指針,所以驅動(dòng)程序2看到向下的第二個(gè)IRP棧單元(驅動(dòng)程序1為它設置的棧單元)。這個(gè)過(guò)程繼續,直到最底層的的驅動(dòng)程序4收到這個(gè)IRP。
驅動(dòng)程序4現在處理這個(gè)IRP。當它完成IRP的處理時(shí),驅動(dòng)程序4調用IoCompleteRequest()函數。指示它已經(jīng)完成IRP的處理。IRP再沿設備棧向上傳遞,直到它最終彈出棧頂,回到用戶(hù)。
2.3 IRP的完成
當一個(gè)驅動(dòng)程序完成對IRP的處理時(shí),它必須告訴I/O管理器,這稱(chēng)為IRP完成。如下面代碼所示,必須設置IRP IoStatus域結構中的幾個(gè)域。IoStatus,Status設置為一個(gè)NTSTATUS狀態(tài)碼,IoStatus.In-formation通常存儲傳輸的字節數。如:
Irp一>loStatus.Status=S T ATUS_SUCCESS
Irp一>IoStatus.Information=info;
IoCompleteRequest(Irp,IO_NO_INCREMENT);
調用IoCompleteRequest()表明低層驅動(dòng)程序已經(jīng)完成了IRP的請求,并將這個(gè)IRP返回給I/O管理器。IO_No_INCREMENT是個(gè)系統定義的常量,指定啟動(dòng)該IRP的優(yōu)先級,需要驅動(dòng)程序快速處理。
3 驅動(dòng)程序功能實(shí)現
當把板卡第一次插到計算機的PCI插槽以后,計算機的系統總線(xiàn)會(huì )檢測到有個(gè)新設備沒(méi)有安裝驅動(dòng)程序,并提示安裝驅動(dòng)程序。正確地安裝驅動(dòng)程序以后,用戶(hù)就可以在應用程序中與驅動(dòng)程序進(jìn)行通信。
3.1 打開(kāi)設備

評論