在C#中根据HardwareID获取驱动程序信息的实现代码

2019-12-30 14:41:45王冬梅

近日在工作中需要根据设备的HardwareID来获取设备的驱动程序信息,比如驱动程序版本等。经过摸索,得到了两种不同的解决办法,两种办法各有千秋,写出来给大家分享。

1 使用WMI中的Win32_PnPSignedDriver类

Win32_PnPSignedDriver的详细信息:http://www.easck.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='somehardware'),那么一定会得到一个“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的结构中