5

我正在寻找一种编程方式来获取驱动程序的版本号。我想要设备管理器在设备的驱动程序属性中显示的相同编号。

背景:我有一个与一些自定义硬件对话的应用程序。自定义硬件的设备驱动程序在某个版本号之前存在已知错误。我希望应用程序检查驱动程序版本并警告用户是否需要更新它。该应用程序在 Windows XP 和 7 上运行,并使用 C++ 编写。

我之前使用的一个技巧是直接从 system32/drivers 读取 .sys 文件并直接搜索“FileVersion”。这很糟糕,原因有很多。特别是它似乎需要 Windows 7 上的管理员权限。

我知道类 GUID 和硬件 ID(即“USB\VID_1234&PID_5678”)。

应用程序当前使用 SetupDiGetClassDevs、SetupDiEnumDeviceInterfaces 和 SetupDiGetDeviceInterfaceDetail 来获取“DevicePath”。然后它使用该路径调用 CreateFile 以与驱动程序对话。

看起来我需要从某个地方获取一个 SP_DRVINFO_DATA 结构。我尝试了 setupapi.h 中的各种函数,例如 SetupDiGetDeviceInterfaceDetail。这是我尝试过的一些失败的代码:

int main(void)
{
    HDEVINFO DeviceInfoSet = SetupDiGetClassDevs((LPGUID)&GUID_DEVINTERFACE_USBSPI, NULL, NULL,
        DIGCF_PRESENT | DIGCF_DEVICEINTERFACE);

    SP_INTERFACE_DEVICE_DATA InterfaceDeviceData;
    InterfaceDeviceData.cbSize = sizeof(SP_INTERFACE_DEVICE_DATA);

    // Cycle through all devices.
    for (int i = 0; i < 32; i++)
    {
        if (!SetupDiEnumDeviceInterfaces(DeviceInfoSet, 0, (LPGUID)&GUID_DEVINTERFACE_USBSPI, i, &InterfaceDeviceData))
            break;

        PSP_DEVICE_INTERFACE_DETAIL_DATA DeviceInterfaceDetailData;
        DWORD RequiredSize;

        SetupDiGetDeviceInterfaceDetail(DeviceInfoSet, &InterfaceDeviceData, NULL, 0, &RequiredSize, NULL);
        DeviceInterfaceDetailData = (PSP_DEVICE_INTERFACE_DETAIL_DATA)HeapAlloc(GetProcessHeap(), HEAP_GENERATE_EXCEPTIONS | HEAP_ZERO_MEMORY, RequiredSize); 
        try
        {
            DeviceInterfaceDetailData->cbSize = sizeof(SP_DEVICE_INTERFACE_DETAIL_DATA);
            SetupDiGetDeviceInterfaceDetail(DeviceInfoSet, &InterfaceDeviceData, DeviceInterfaceDetailData, RequiredSize, NULL, NULL);

            // Try to get the driver info. This part always fails with code 
            // 259 (ERROR_NO_MORE_ITEMS).
            SP_DRVINFO_DATA drvInfo;
            drvInfo.cbSize = sizeof(SP_DRVINFO_DATA);
            if (!SetupDiEnumDriverInfo(DeviceInfoSet, NULL, SPDIT_CLASSDRIVER, i, &drvInfo))
                printf("error = %d\n", GetLastError());

            printf("Driver version is %08x %08x\n", drvInfo.DriverVersion >> 32, drvInfo.DriverVersion & 0xffffffff);
        }
        catch(...)
        {
            HeapFree(GetProcessHeap(), 0, DeviceInterfaceDetailData);
            throw;
        }
        HeapFree(GetProcessHeap(), 0, DeviceInterfaceDetailData);
    }

    SetupDiDestroyDeviceInfoList(DeviceInfoSet);

    return 0;
}

编辑- 我更新的代码现在看起来像这样:

HDEVINFO devInfoSet = SetupDiGetClassDevs(&GUID_DEVINTERFACE_USBSPI, NULL, NULL,
                                          DIGCF_PRESENT | DIGCF_DEVICEINTERFACE);

// Cycle through all devices.
for (int i = 0; ; i++)
{
    // Get the device info for this device
    SP_DEVINFO_DATA devInfo;
    devInfo.cbSize = sizeof(SP_DEVINFO_DATA);
    if (!SetupDiEnumDeviceInfo(devInfoSet, i, &devInfo))
        break;

    // Get the first info item for this driver
    SP_DRVINFO_DATA drvInfo;
    drvInfo.cbSize = sizeof(SP_DRVINFO_DATA);
    if (!SetupDiEnumDriverInfo(devInfoSet, &devInfo, SPDIT_COMPATDRIVER, 0, &drvInfo))
        printf("err - %d\n", GetLastError()); // Still fails with "no more items"
}

SetupDiDestroyDeviceInfoList(devInfoSet);
4

2 回答 2

3

您错误地iSetupDiEnumDriverInfo. 这应该是每个驱动程序的每个驱动程序信息元素的内部循环。因此,您无法检索设备 #1 的驱动程序信息 #0。

尽管如此,这并不能解释为什么设备#0 的信息#0 失败。为此,您必须查看SetupDiEnumDriverInfo. 这是SP_DEVINFO_DATA您设备的结构,但您将其设置为NULL. 这将为您提供与设备类关联的驱动程序列表,而不是设备。即适用于具有类驱动程序的鼠标和 USB 记忆棒。您的设备可能具有特定于供应商的驱动程序,因此您需要该特定设备的驱动程序。

于 2013-04-16T11:49:51.570 回答
0

当您问了一个几乎相同的问题时,我只在此处发布了我的答案的链接:

为什么 SetupDiEnumDriverInfo 为我的驱动程序提供两个版本号

于 2020-05-07T21:17:18.870 回答