在C#中怎麼根據HardwareID擷取驅動程式資訊

來源:互聯網
上載者:User

近日在工作中需要根據裝置的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。

相關文章

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.