IRP和IO_STACK_LOCATION
當(dāng)一個(gè)應(yīng)用程序調(diào)用函數(shù)去操作某個(gè)設(shè)備時(shí),比如調(diào)用createFile,deviceIOControl,等等時(shí),I/O管理器為此函數(shù)創(chuàng)建一個(gè)IRP數(shù)據(jù)結(jié)構(gòu)對象和一個(gè)IRP_STACK_LOCATION數(shù)據(jù)結(jié)構(gòu)對象數(shù)組。
(數(shù)組個(gè)數(shù)等于驅(qū)動(dòng)程序堆棧上驅(qū)動(dòng)的個(gè)數(shù))。IRP對象中的數(shù)據(jù)成員是已經(jīng)被填充好了的,其中有一個(gè)CurrentLocation是當(dāng)前IRP_STACK_LOCATION堆棧單元的索引,Tail.Overlay.CurrentStackLocation
是當(dāng)前IRP_STACK_LOCATION單元的指針。
一開始的時(shí)候,它當(dāng)然是指向IRP_STACK_LOCATION數(shù)組的第一個(gè)元素。
這調(diào)用某個(gè)驅(qū)動(dòng)程序的分發(fā)函數(shù)時(shí),IRP作為參數(shù)傳給了該分發(fā)函數(shù),這樣,該函數(shù)就可以訪問IRP中的Tail.Overlay.CurrentStackLocation, 這樣就可以訪問IRP_STACK_LOCATION的
成員了(當(dāng)然我們不需要直接這樣做,而是調(diào)用IoGetCurrentStackLocation來實(shí)現(xiàn))。因?yàn)镺S并不會(huì)為我們初始化IRP_STACK_LOCATION數(shù)組中的每一個(gè)對象,所以初始化下一層驅(qū)動(dòng)要使用的IRP_STACK_LOCATION就由它的上一層驅(qū)動(dòng)來完成。要初始化它們,首先要找到它們,方法是調(diào)用IoGetNextStackLocation(它內(nèi)部實(shí)現(xiàn)只是將CurrentStackLocation++),得到下一個(gè)IRP_STACK_LOCATION
對象指針。然后為其賦值。然后調(diào)用IOCALLDRIVER().
理解的關(guān)鍵點(diǎn)是:
一 OS為我們生成IRP,IPR_STACK_LOCATION數(shù)組,注意,是個(gè)數(shù)組。IRP中的CurrentStackLocation指向IRP_STACK_LOCATION中的某一個(gè)元素。
二 OS 并不會(huì)為我們填充好IRP_STACK_LOCATION數(shù)組,每一個(gè)元素是由上一層驅(qū)動(dòng)負(fù)責(zé)填充的。
三 IRP_STACK_LOCATION數(shù)組,它們的元素之間不需要指針聯(lián)系。
四 OS只負(fù)責(zé)把IRP包給最上層的驅(qū)動(dòng)程序,至于如何向下層,就是驅(qū)動(dòng)程序自己的事情了。所以下層對就的那些IRP_STACK_LOCATION,完全由上層驅(qū)動(dòng)函數(shù)負(fù)責(zé)填充,OS不管。
?
驅(qū)動(dòng)程序如何填充一層驅(qū)動(dòng)需要用的IRP_STACK_LOCATION呢?
可以通過調(diào)用IoGetNextIrpStackLocation調(diào)用得到。其實(shí)該函數(shù)內(nèi)部就是返回CurrentStackLocation加1而已。對數(shù)組值加1,當(dāng)然就是得到數(shù)組的下一個(gè)值了。
這樣就可以對它進(jìn)行賦值了。然后調(diào)用IoCallDriver(),IoCallDriver()會(huì)將irp包中的CurrentStackLocation值加1,然后調(diào)用那個(gè)DRIVER.
?
如果下一層的驅(qū)動(dòng)需要的IRP_STACK_LOCATION和本層驅(qū)動(dòng)的一樣,則可以直接調(diào)用IoSkipCurrentIrpStackLocation或IoCopyCurrentIrpStackLocationToNext.
IoSkipCurrentIrpStackLocation將CurrentStackLocation減1. 正好與IoCallDriver的加1抵消。所以相當(dāng)于下層驅(qū)動(dòng)和本層驅(qū)動(dòng)用的是同一個(gè)IRP_STACK_LOCATIN元素。
總結(jié)
以上是生活随笔為你收集整理的IRP和IO_STACK_LOCATION的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: ninja: error: 'LIBSO
- 下一篇: 线程同步之条件变量