PCI驅(qū)動(dòng)程序開(kāi)發(fā)實(shí)例全解析
IRP首先到達(dá)最高層的驅(qū)動(dòng)程序1,驅(qū)動(dòng)程序1使用函數(shù)IoGetCurrentIrpStackLocation()獲得指向當(dāng)前棧單元的指針。
然后驅(qū)動(dòng)程序1使用IoCallDriver()函數(shù)調(diào)用下一個(gè)驅(qū)動(dòng)程序。I/O管理器現(xiàn)在改變“當(dāng)前IRF’棧單元”指針,所以驅(qū)動(dòng)程序2看到向下的第二個(gè)IRP棧單元(驅(qū)動(dòng)程序1為它設(shè)置的棧單元)。這個(gè)過(guò)程繼續(xù),直到最底層的的驅(qū)動(dòng)程序4收到這個(gè)IRP。
驅(qū)動(dòng)程序4現(xiàn)在處理這個(gè)IRP。當(dāng)它完成IRP的處理時(shí),驅(qū)動(dòng)程序4調(diào)用IoCompleteRequest()函數(shù)。指示它已經(jīng)完成IRP的處理。IRP再沿設(shè)備棧向上傳遞,直到它最終彈出棧頂,回到用戶。
2.3 IRP的完成
當(dāng)一個(gè)驅(qū)動(dòng)程序完成對(duì)IRP的處理時(shí),它必須告訴I/O管理器,這稱為IRP完成。如下面代碼所示,必須設(shè)置IRP IoStatus域結(jié)構(gòu)中的幾個(gè)域。IoStatus,Status設(shè)置為一個(gè)NTSTATUS狀態(tài)碼,IoStatus.In-formation通常存儲(chǔ)傳輸?shù)淖止?jié)數(shù)。如:
Irp一>loStatus.Status=S T ATUS_SUCCESS
Irp一>IoStatus.Information=info;
IoCompleteRequest(Irp,IO_NO_INCREMENT);
調(diào)用IoCompleteRequest()表明低層驅(qū)動(dòng)程序已經(jīng)完成了IRP的請(qǐng)求,并將這個(gè)IRP返回給I/O管理器。IO_No_INCREMENT是個(gè)系統(tǒng)定義的常量,指定啟動(dòng)該IRP的優(yōu)先級(jí),需要驅(qū)動(dòng)程序快速處理。
3 驅(qū)動(dòng)程序功能實(shí)現(xiàn)
當(dāng)把板卡第一次插到計(jì)算機(jī)的PCI插槽以后,計(jì)算機(jī)的系統(tǒng)總線會(huì)檢測(cè)到有個(gè)新設(shè)備沒(méi)有安裝驅(qū)動(dòng)程序,并提示安裝驅(qū)動(dòng)程序。正確地安裝驅(qū)動(dòng)程序以后,用戶就可以在應(yīng)用程序中與驅(qū)動(dòng)程序進(jìn)行通信。
3.1 打開(kāi)設(shè)備
在應(yīng)用程序中調(diào)用系統(tǒng)提供的函數(shù)CreateFile()。如果系統(tǒng)根據(jù)設(shè)備名確實(shí)檢測(cè)到設(shè)備并成功打開(kāi)了這個(gè)設(shè)備,則返回一個(gè)指向這個(gè)設(shè)備的有效句柄;如果調(diào)用失敗,則返回一個(gè)錯(cuò)誤信息。
3.2 讀/寫(xiě)設(shè)備
讀/寫(xiě)設(shè)備包括讀/寫(xiě)設(shè)備的配置空間、讀/寫(xiě)設(shè)備的非配置空間,其中,非配置空間包括I/O空間、存儲(chǔ)空間。
設(shè)備被打開(kāi)以后,應(yīng)用程序就調(diào)用DeviceloCon-trol()函數(shù)來(lái)達(dá)到訪問(wèn)設(shè)備的目的。DeviceIoControl()函數(shù)有8個(gè)參數(shù),其中第3個(gè)參數(shù)是應(yīng)用程序傳遞給驅(qū)動(dòng)程序的數(shù)據(jù)緩沖區(qū)地址,在這個(gè)數(shù)據(jù)緩沖區(qū)存放的是應(yīng)用程序要讀寫(xiě)的設(shè)備的空間、偏移量、長(zhǎng)度,這些都需要在應(yīng)用程序中配置好。
這個(gè)調(diào)用由I/O系統(tǒng)服務(wù)接收。I/O管理器從這個(gè)請(qǐng)求構(gòu)造一個(gè)合適的I/O請(qǐng)求包(IRP)。在最簡(jiǎn)單的情況下,I/O管理器只是把IRP傳遞給一個(gè)設(shè)備驅(qū)動(dòng)程序,這個(gè)驅(qū)動(dòng)程序調(diào)用硬件,并完成IRP的處理。I/O管理器把數(shù)據(jù)和結(jié)果返回給Win 32和用戶應(yīng)用程序?,F(xiàn)在一個(gè)分層的設(shè)備驅(qū)動(dòng)程序棧是很常見(jiàn)的。每個(gè)驅(qū)動(dòng)程序把該請(qǐng)求劃分為更簡(jiǎn)單的請(qǐng)求。高層次的驅(qū)動(dòng)程序調(diào)用低層次的驅(qū)動(dòng)程序,最后,最低層的驅(qū)動(dòng)程序與硬件直接打交道完成用戶的請(qǐng)求。I/O管理器把數(shù)據(jù)和結(jié)果返回給Win 32和用戶應(yīng)用程序。設(shè)備程序調(diào)用如圖3所示。
當(dāng)應(yīng)用程序讀/寫(xiě)設(shè)備時(shí),驅(qū)動(dòng)程序工作的流程圖如圖4所示。
首先,獲取當(dāng)前IRP棧單元的指針;然后再讀取I/O控制代碼,判斷應(yīng)用程序想達(dá)到什么樣的目的:是讀/寫(xiě)配置空間,還是讀/寫(xiě)非配置空間,然后再調(diào)用相應(yīng)的處理程序。
4 結(jié) 語(yǔ)
這里采用微軟的驅(qū)動(dòng)程序開(kāi)發(fā)包Device Driver Kit(DDK)是因?yàn)樗瞧渌麕追N工具的基礎(chǔ),它要求開(kāi)發(fā)人員深刻了解驅(qū)動(dòng)底層,雖然不易掌握,但開(kāi)發(fā)出來(lái)的驅(qū)動(dòng)程序通用性好,兼容性強(qiáng)。板卡與PC機(jī)的通信速度得到了很大的提高。
評(píng)論