近日在工作中需要根據裝置的HardwareID來擷取裝置的驅動程式資訊,比如驅動程式版本等。經過摸索,得到了兩種不同的解決辦法,兩種辦法各有千秋,寫出來給大家分享。
1 使用WMI中的Win32_PnPSignedDriver類
Win32_PnPSignedDriver的詳細資料:http://msdn2.microsoft.com/en-us/library/aa394354.aspx
使用WMI(Windows Management Instrumentation)是最為方便的方法。可以根據下面的程式片段來得到我們所需要的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";
}
這樣取得驅動程式的方式是非常簡潔的,但是有一個非常嚴重的問題就是效率問題。平均說來,每執行一次查詢,得到一個DriverVersion需要大約3秒的時間。對於我們的應用來說,這個時間是不可以接受的。也許你會說,為什麼不用更多的限定符號來進一步減少查詢的次數呢?
如果我們把連接字串改成:
string queryString = "SELECT HardwareID, DriverVersion FROM Win32_PnPSignedDriver WHERE HardwareID='somehardware'";
程式的效率並沒有明顯的改進。而且還發現一個問題,如果我們somehardware裡面含有一個'\'(也就是HardwareID='some\\hardware'),那麼一定會得到一個“Invalid Query”異常。但是在WMITOOLS裡面查詢又是正常的,希望達人出來指點一下。最後根據MSDN的描述,只有Windows Vista,Windows XP和Windows 2003支援這個類。由於我們的程式需要跑在2000下,因此這種方法是行不通的了。
2 使用PInvoke
由於無法使用WMI,因此就想到了使用PInvoke的方式調用Windows API。通過查詢MSDN,知道可以使用SetupDixxxx這種函數來實現我們的功能。基本的思路如下:
(1)利用SetupDiGetClassDevs這個函數得到一個含有所有裝置資訊的類。
(2)利用SetupDiEnumDeviceInfo得到某個具體裝置的資訊,儲存在一個名為SP_DEVINFO_DATA的結構中。
(3)利用SetupDiGetDeviceRegistryProperty得到裝置的HardwareID,和輸入的HardwareID比較
(4)如果兩個HardwareID是一樣的,那麼就利用SetupDiBuildDriverInfoList得到這個裝置的驅動程式資訊列表
(5)利用SetupDiEnumDriverInfo遍曆驅動程式資訊列表,得到所有需要的資訊,儲存在一個名為SP_DRVINFO_DATA的結構中
(6)從SP_DRVINFO_DATA中就可以得到驅動程式的版本。是一個DWORDLONG類型的數,需要轉換成x.x.x.x的結構
要值得注意的是上述函數都封裝在setupapi.dll中,要使用這些函數,需要安裝Windows DDK。
在C#中,我們利用pInvoke的方式來調用Windows API的時候,需要注意類型的對應和結構對齊。比如上面的SP_DEVINFO_DATA結構需要按照如下方式聲明
[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。如果不按照這樣聲明,無法調用成功。
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轉換成x.x.x.x的版本,可以按照下面的方式轉換。DWORDLONG是8位元組的不帶正負號的整數,x.x.x.x中的x是從0到65536的不帶正負號的整數,佔2個位元組。因此可以直接把8位元組的整數分成4個2位元組的整數,最後合起來就是版本號碼了。假設版本version = 1407379348914176,將version轉換成2進位數為:
101 00000000 00000001 00001010 00101000 00000000 00000000
--- --------------------- ---------------------- ---------------------
5 1 2600 0
因此,可以得到版本是5.1.2600.0。
可以用下面這個樣本函數來得到版本資訊
//version = 1407379348914176,轉換後的版本為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();
}
通過調用API這種方式,速度得到了很大的提高,1秒之內就可以完成一次查詢。而且適合於Win2000,Win XP,Win2003和Vista。