在C#中根據(jù)HardwareID獲取驅(qū)動程序信息的實現(xiàn)代碼
近日在工作中需要根據(jù)設備的HardwareID來獲取設備的驅(qū)動程序信息,比如驅(qū)動程序版本等。經(jīng)過摸索,得到了兩種不同的解決辦法,兩種辦法各有千秋,寫出來給大家分享。
1 使用WMI中的Win32_PnPSignedDriver類
Win32_PnPSignedDriver的詳細信息:http://msdn2.microsoft.com/en-us/library/aa394354.aspx
使用WMI(Windows Management Instrumentation)是最為方便的方法??梢愿鶕?jù)下面的程序片段來得到我們所需要的DriverVersion。
private string GetDriverVersion( string hardwareID ) { string queryString = "SELECT HardwareID, DriverVersion FROM Win32_PnPSignedDriver"; SelectQuery selectQuery = new SelectQuery( queryString ); ManagementObjectSearcher searcher = new ManagementObjectSearcher(selectQuery); foreach (ManagementObject mo in searcher.Get()) { object tempID = mo["HardwareID"]; if( tempID!=null && tempID.ToString().ToUpper() == hardwareID.Trim().ToUpper() ) { return mo["DriverVersion"].ToString(); } } return "UnknownVersion"; }
這樣取得驅(qū)動程序的方式是非常簡潔的,但是有一個非常嚴重的問題就是效率問題。平均說來,每執(zhí)行一次查詢,得到一個DriverVersion需要大約3秒的時間。對于我們的應用來說,這個時間是不可以接受的。也許你會說,為什么不用更多的限定符號來進一步減少查詢的次數(shù)呢?
如果我們把連接字符串改成:
string queryString = "SELECT HardwareID, DriverVersion FROM Win32_PnPSignedDriver WHERE HardwareID='somehardware'";
程序的效率并沒有明顯的改進。而且還發(fā)現(xiàn)一個問題,如果我們somehardware里面含有一個'\'(也就是HardwareID='some\\hardware'),那么一定會得到一個“Invalid Query”異常。但是在WMITOOLS里面查詢又是正常的,希望達人出來指點一下。最后根據(jù)MSDN的描述,只有Windows Vista,Windows XP和Windows 2003支持這個類。由于我們的程序需要跑在2000下,因此這種方法是行不通的了。
2 使用PInvoke
由于無法使用WMI,因此就想到了使用PInvoke的方式調(diào)用Windows API。通過查詢MSDN,知道可以使用SetupDixxxx這種函數(shù)來實現(xiàn)我們的功能?;镜乃悸啡缦拢?br />
(1)利用SetupDiGetClassDevs這個函數(shù)得到一個含有所有設備信息的類。
(2)利用SetupDiEnumDeviceInfo得到某個具體設備的信息,保存在一個名為SP_DEVINFO_DATA的結(jié)構(gòu)中。
(3)利用SetupDiGetDeviceRegistryProperty得到設備的HardwareID,和輸入的HardwareID比較
(4)如果兩個HardwareID是一樣的,那么就利用SetupDiBuildDriverInfoList得到這個設備的驅(qū)動程序信息列表
(5)利用SetupDiEnumDriverInfo遍歷驅(qū)動程序信息列表,得到所有需要的信息,保存在一個名為SP_DRVINFO_DATA的結(jié)構(gòu)中
(6)從SP_DRVINFO_DATA中就可以得到驅(qū)動程序的版本。是一個DWORDLONG類型的數(shù),需要轉(zhuǎn)換成x.x.x.x的結(jié)構(gòu)
要值得注意的是上述函數(shù)都封裝在setupapi.dll中,要使用這些函數(shù),需要安裝Windows DDK。
在C#中,我們利用pInvoke的方式來調(diào)用Windows API的時候,需要注意類型的對應和結(jié)構(gòu)對齊。比如上面的SP_DEVINFO_DATA結(jié)構(gòu)需要按照如下方式聲明
[StructLayout(LayoutKind.Sequential, Pack = 4, CharSet = CharSet.Auto)] public struct SP_DEVINFO_DATA { public int cbSize; public Guid ClassGuid; public IntPtr DevInst; public IntPtr Reserved; }
要注意的是LayoutKind.Sequential, Pack = 4 和 public IntPtr Reserved。如果不按照這樣聲明,無法調(diào)用成功。
SP_DRVINFO_DATA也可以按照一樣的方式進行聲明。
[StructLayout(LayoutKind.Sequential, Pack = 4, CharSet = CharSet.Auto)] public struct SP_DRVINFO_DATA { public int cbSize; public int DriverType; public IntPtr Reserved; [MarshalAs(UnmanagedType.ByValTStr, SizeConst = 256)] public string Description; [MarshalAs(UnmanagedType.ByValTStr, SizeConst = 256)] public string MfgName; [MarshalAs(UnmanagedType.ByValTStr, SizeConst = 256)] public string ProviderName; public FILETIME DriverDate; public ulong DriverVersion; }
對于最后的從DWORDLONG轉(zhuǎn)換成x.x.x.x的版本,可以按照下面的方式轉(zhuǎn)換。DWORDLONG是8字節(jié)的無符號整數(shù),x.x.x.x中的x是從0到65536的無符號整數(shù),占2個字節(jié)。因此可以直接把8字節(jié)的整數(shù)分成4個2字節(jié)的整數(shù),最后合起來就是版本號了。假設版本version = 1407379348914176,將version轉(zhuǎn)換成2進制數(shù)為:
101 00000000 00000001 00001010 00101000 00000000 00000000
--- --------------------- ---------------------- ---------------------
5 1 2600 0
因此,可以得到版本是5.1.2600.0。
可以用下面這個示例函數(shù)來得到版本信息
//version = 1407379348914176,轉(zhuǎn)換后的版本為5.1.2600.0 private string GetVersionFromLong( ulong version ) { ulong baseNumber = 0xFFFF; StringBuilder sb = new StringBuilder(); ulong temp = 0L; for( int offset = 48; offset >= 0; offset -= 16 ) { temp = (version >> offset) & baseNumber; sb.Append( temp.ToString() + "." ); } return sb.ToString(); }
通過調(diào)用API這種方式,速度得到了很大的提高,1秒之內(nèi)就可以完成一次查詢。而且適合于Win2000,Win XP,Win2003和Vista。
相關文章
C#多線程之Thread中Thread.Join()函數(shù)用法分析
這篇文章主要介紹了C#多線程之Thread中Thread.Join()函數(shù)用法,實例分析了Thread.Join()方法的原理與使用技巧,非常具有實用價值,需要的朋友可以參考下2015-04-04