介紹一個關(guān)于協(xié)作安裝程序的應用實例
如何讓我們的USB設備插上后,系統(tǒng)也能自動為它安裝驅(qū)動,而不需要煩勞用戶手動安裝呢?解決此問題的核心技術(shù)在于編寫一個類協(xié)作安裝程序。
首先,有一個問題大家要清楚,USB設備第一次插到機器上的一個USB口上時,系統(tǒng)要為它裝一次驅(qū)動程序。我們以USB設備為例,來了解一下支持熱插拔的PnP設備的安裝過程:
(1)設備插入系統(tǒng),USB總線驅(qū)動向內(nèi)核PnP管理器報告有新設備接入系統(tǒng);
(2)內(nèi)核PnP管理器向USB總線驅(qū)動詢問設備的具體信息,比如PID和VID等;
(3)內(nèi)核PnP管理器將設備的信息報告給用戶層的PnP管理器,并要求它為新設備安裝驅(qū)動;
(4)用戶層PnP管理器調(diào)用系統(tǒng)的Setup組件來為設備安裝驅(qū)動;
(5)Setup使用設備VID和PID到%Windir%inf下尋找適合它的inf文件,并獲得一個可用于設備的驅(qū)動程序列表;
(6)Setup在生成驅(qū)動程序列表的時候,會檢查inf文件是否經(jīng)過數(shù)字簽名,如果沒有經(jīng)過數(shù)字簽名,Setup會將此inf文件負責安裝的驅(qū)動程序設置成“不可信任的”驅(qū)動程序;
(7)Setup對驅(qū)動程序列表中的各驅(qū)動程序信息進行分析,選擇最匹配設備的驅(qū)動程序進行安裝;
這里,有必要提一下“不可信任的驅(qū)動程序”這個概念。這個概念在xp之后才有的,2k和98沒有。在Setup 生成的驅(qū)動程序列表中,每個驅(qū)動程序的信息結(jié)構(gòu)中都有一個Rank字段。在xp中,0x0 Rank = 0x3FFF的驅(qū)動程序被認為是“可信任的”;0x8000 = Rank = 0xFFFF的驅(qū)動被認為是“不可信任的”。如果我們的驅(qū)動程序沒有經(jīng)過數(shù)字簽名,那么它的Rank值肯定落在0x8000到0xFFFF之間。
再回到前面的安裝過程,如果驅(qū)動程序中有適合設備的“可信任”驅(qū)動程序,那么系統(tǒng)自動對它進行安裝;如果驅(qū)動程序列表中的所有驅(qū)動程序都是“不可信任的”,那么系統(tǒng)就會彈出“發(fā)現(xiàn)新硬件”向?qū)?,要你提供更好的?qū)動程序,或者要你確認安裝“不可信任的”驅(qū)動程序。這就是為什么在xp系統(tǒng)下,即便你在一個USB口上已經(jīng)安裝了設備的驅(qū)動程序,你再換個口插上設備,系統(tǒng)又會提示你安裝驅(qū)動程序的原因。
說了半天,我想現(xiàn)在各位肯定都明白過來了:影響設備驅(qū)動程序自動安裝的主要原因,是因為我們的驅(qū)動程序被系統(tǒng)認為是“不可信任的”。而系統(tǒng)判斷一個驅(qū)動程序是否“可信任”,是通過驅(qū)動程序信息結(jié)構(gòu)中的Rank字段的值來判斷的。那么,如果我們能把我們的驅(qū)動程序信息中的Rank值修改到“可信任”空間,那么系統(tǒng)是否就會信任我們的驅(qū)動程序,而自動對它進行安裝呢?答案是,有可能。我不敢說肯定可以,原因后面會提到。但是,如何修改驅(qū)動程序信息的Rank值呢?這就要用到“協(xié)作安裝程序”。
我們知道,在設備的安裝過程中,Setup要向設備類安裝程序、類協(xié)作安裝程序和設備協(xié)作安裝程序發(fā)送“設備安裝功能碼”(如果有這些安裝程序的話)。ddk文檔中又說,類安裝程序和類協(xié)作安裝程序可以對DIF_SELECTBESTCOMPATDRV請求進行處理(設備協(xié)作安裝程序不可以)。在對DIF_SELECTBESTCOMPATDRV進行處理的時候,類安裝程序和類協(xié)作安裝程序可以修改驅(qū)動程序列表中各驅(qū)動程序的信息。答案越來越清晰了,我們只要寫一個類協(xié)作安裝程序,對DIF_SELECTBESTCOMPATDRV進行處理,修改我們想要安裝的驅(qū)動程序的Rank值,那么就可能騙過系統(tǒng),使系統(tǒng)相信我們的驅(qū)動程序,并完成自動安裝。關(guān)于編寫協(xié)作安裝程序的具體要求和方法,可以參考ddk文檔中的“Writing a Co-installer(編寫協(xié)作安裝程序)”和ddksrcgeneraltoastercoinstaller。
接下來,我們來了解一下,在類協(xié)作安裝程序處理DIF_SELECTBESTCOMPATDRV時,應該做哪些事情。
(1)首先,調(diào)用SetupDiEnumDriverInfo遍歷驅(qū)動程序列表,獲得每個驅(qū)動程序的信息——一個SP_DRVINFO_DATA結(jié)構(gòu)。
(2)接著,用(1)中獲得的SP_DRVINFO_DATA作為輸入?yún)?shù),調(diào)用SetupDiGetDriverInstallParams,獲得驅(qū)動程序安裝參數(shù)——一個SP_DRVINSTALL_PARAMS結(jié)構(gòu),其中我們想要修改的Rank赫然在列。你可以按照自己的需要修改Rank的值,在這兒我們肯定是要把它改為0了(0表示驅(qū)動程序與設備最匹配)。
(3)最后,把修改后的SP_DRVINSTALL_PARAMS結(jié)構(gòu)作為輸入,調(diào)用SetupDiSetDriverInstallParams將我們修改的值設置生效。
在類協(xié)作安裝程序中只需作如此處理,便可以使Setup此后信任我們的驅(qū)動程序,從而達到我們想瞞天過海的目的。
再稍微提一下類協(xié)作安裝程序的注冊。協(xié)作安裝程序做好了,如何使它參與到設備安裝的過程中來呢?我們必須注冊它。ddk文檔對此講得非常清楚了,參看“Registering a Class Co-installer”,我就不在這兒把它翻譯出來了。
最后,要提醒一點:必須為我們的usb設備定義一個新的設備setup類,然后將我們的協(xié)作安裝程序注冊為這個setup類的類協(xié)作安裝程序。如果我們讓設備仍然屬于usb setup類,并將我們的類協(xié)作安裝程序注冊為usbsetup類的一個協(xié)作安裝程序,那么在安裝過程中,Setup仍然彈出一些窗體影響我們設備的自動安裝,似乎我們的小聰明并沒能瞞過它。這就是前面我說修改Rank值為“可信任”只是有可能瞞過系統(tǒng)而不是肯定能夠瞞過系統(tǒng)的原因。為什么會出現(xiàn)這種情況呢?從現(xiàn)象看,我感覺是usb setup類的類安裝程序仍然發(fā)現(xiàn)我們的驅(qū)動程序是不可信任的。但是,Setup是以類協(xié)作安裝程序、設備協(xié)作安裝程序和類安裝程序的順序調(diào)用它們的,在Setup調(diào)用usb setup類安裝程序之前,我們已經(jīng)修改了驅(qū)動程序的Rank值。按理說,它應該不會發(fā)現(xiàn)驅(qū)動程序是不可信任的。這是一個問題,具體原因我還沒有弄明白,希望有知其所以然者,能給點提示!不管如何,通過實驗我發(fā)現(xiàn),只要我們定義了新的setup類,那么我們就可以騙過系統(tǒng)Setup組件,使其自動為我們的設備安裝驅(qū)動程序。
就寫這些,有興趣的朋友可以試一下!我不想把具體的實現(xiàn)過程一步一步地寫出來,更不愿提供具體的實現(xiàn)代碼。因為我認為只要把原理和方法講清楚了(但愿我講得還算清楚),每個人都可以在此基礎上做自己的事情。
不管各位朋友看完之后感覺如何,能夸就夸夸,該罵就罵罵,都頂一下
評論