qga-win: fsinfo: pci-info: allow partial info

The call to SetupDiGetDeviceRegistryProperty might fail because the
value doesn't exist in the registry, in this case we shouldn't exit from
the loop but instead continue to look for other available values in the
registry and set this value as unavailable (-1).

Signed-off-by: Sameeh Jubran <sjubran@redhat.com>
Signed-off-by: Michael Roth <mdroth@linux.vnet.ibm.com>
Signed-off-by: Tomáš Golembiovský <tgolembi@redhat.com>
*squash in fix for when get_pci_info() returns NULL pci_controller field
*fix handling for error_set() cases in get_pci_info(), not just NULL return
*force all -1 PCI addr fields if any single one of them isn't found
Signed-off-by: Michael Roth <mdroth@linux.vnet.ibm.com>
This commit is contained in:
Sameeh Jubran 2018-10-23 13:23:14 +02:00 committed by Michael Roth
parent 0d7f937e2a
commit 6880b94f8a

View File

@ -499,6 +499,7 @@ static GuestPCIAddress *get_pci_info(char *guid, Error **errp)
char *buffer = NULL; char *buffer = NULL;
GuestPCIAddress *pci = NULL; GuestPCIAddress *pci = NULL;
char *name = g_strdup(&guid[4]); char *name = g_strdup(&guid[4]);
bool partial_pci = false;
pci = g_malloc0(sizeof(*pci)); pci = g_malloc0(sizeof(*pci));
pci->domain = -1; pci->domain = -1;
pci->slot = -1; pci->slot = -1;
@ -519,7 +520,8 @@ static GuestPCIAddress *get_pci_info(char *guid, Error **errp)
dev_info_data.cbSize = sizeof(SP_DEVINFO_DATA); dev_info_data.cbSize = sizeof(SP_DEVINFO_DATA);
for (i = 0; SetupDiEnumDeviceInfo(dev_info, i, &dev_info_data); i++) { for (i = 0; SetupDiEnumDeviceInfo(dev_info, i, &dev_info_data); i++) {
DWORD addr, bus, slot, func, dev, data, size2; DWORD addr, bus, slot, data, size2;
int func, dev;
while (!SetupDiGetDeviceRegistryProperty(dev_info, &dev_info_data, while (!SetupDiGetDeviceRegistryProperty(dev_info, &dev_info_data,
SPDRP_PHYSICAL_DEVICE_OBJECT_NAME, SPDRP_PHYSICAL_DEVICE_OBJECT_NAME,
&data, (PBYTE)buffer, size, &data, (PBYTE)buffer, size,
@ -549,21 +551,24 @@ static GuestPCIAddress *get_pci_info(char *guid, Error **errp)
*/ */
if (!SetupDiGetDeviceRegistryProperty(dev_info, &dev_info_data, if (!SetupDiGetDeviceRegistryProperty(dev_info, &dev_info_data,
SPDRP_BUSNUMBER, &data, (PBYTE)&bus, size, NULL)) { SPDRP_BUSNUMBER, &data, (PBYTE)&bus, size, NULL)) {
break; bus = -1;
partial_pci = true;
} }
/* The function retrieves the device's address. This value will be /* The function retrieves the device's address. This value will be
* transformed into device function and number */ * transformed into device function and number */
if (!SetupDiGetDeviceRegistryProperty(dev_info, &dev_info_data, if (!SetupDiGetDeviceRegistryProperty(dev_info, &dev_info_data,
SPDRP_ADDRESS, &data, (PBYTE)&addr, size, NULL)) { SPDRP_ADDRESS, &data, (PBYTE)&addr, size, NULL)) {
break; addr = -1;
partial_pci = true;
} }
/* This call returns UINumber of DEVICE_CAPABILITIES structure. /* This call returns UINumber of DEVICE_CAPABILITIES structure.
* This number is typically a user-perceived slot number. */ * This number is typically a user-perceived slot number. */
if (!SetupDiGetDeviceRegistryProperty(dev_info, &dev_info_data, if (!SetupDiGetDeviceRegistryProperty(dev_info, &dev_info_data,
SPDRP_UI_NUMBER, &data, (PBYTE)&slot, size, NULL)) { SPDRP_UI_NUMBER, &data, (PBYTE)&slot, size, NULL)) {
break; slot = -1;
partial_pci = true;
} }
/* SetupApi gives us the same information as driver with /* SetupApi gives us the same information as driver with
@ -573,12 +578,19 @@ static GuestPCIAddress *get_pci_info(char *guid, Error **errp)
* DeviceNumber = (USHORT)(((propertyAddress) >> 16) & 0x0000FFFF); * DeviceNumber = (USHORT)(((propertyAddress) >> 16) & 0x0000FFFF);
* SPDRP_ADDRESS is propertyAddress, so we do the same.*/ * SPDRP_ADDRESS is propertyAddress, so we do the same.*/
func = addr & 0x0000FFFF; if (partial_pci) {
dev = (addr >> 16) & 0x0000FFFF; pci->domain = -1;
pci->domain = dev; pci->slot = -1;
pci->slot = slot; pci->function = -1;
pci->function = func; pci->bus = -1;
pci->bus = bus; } else {
func = ((int) addr == -1) ? -1 : addr & 0x0000FFFF;
dev = ((int) addr == -1) ? -1 : (addr >> 16) & 0x0000FFFF;
pci->domain = dev;
pci->slot = (int) slot;
pci->function = func;
pci->bus = (int) bus;
}
break; break;
} }
@ -622,6 +634,7 @@ static GuestDiskAddressList *build_guest_disk_info(char *guid, Error **errp)
DWORD len; DWORD len;
int bus; int bus;
HANDLE vol_h; HANDLE vol_h;
Error *local_err = NULL;
scsi_ad = &addr; scsi_ad = &addr;
char *name = g_strndup(guid, strlen(guid)-1); char *name = g_strndup(guid, strlen(guid)-1);
@ -640,6 +653,16 @@ static GuestDiskAddressList *build_guest_disk_info(char *guid, Error **errp)
disk = g_malloc0(sizeof(*disk)); disk = g_malloc0(sizeof(*disk));
disk->bus_type = find_bus_type(bus); disk->bus_type = find_bus_type(bus);
/* always set pci_controller as required by schema. get_pci_info() should
* report -1 values for non-PCI buses rather than fail. fail the command
* if that doesn't hold since that suggests some other unexpected
* breakage
*/
disk->pci_controller = get_pci_info(name, &local_err);
if (local_err) {
error_propagate(errp, local_err);
goto out_close;
}
if (bus == BusTypeScsi || bus == BusTypeAta || bus == BusTypeRAID if (bus == BusTypeScsi || bus == BusTypeAta || bus == BusTypeRAID
#if (_WIN32_WINNT >= 0x0600) #if (_WIN32_WINNT >= 0x0600)
/* This bus type is not supported before Windows Server 2003 SP1 */ /* This bus type is not supported before Windows Server 2003 SP1 */
@ -654,12 +677,9 @@ static GuestDiskAddressList *build_guest_disk_info(char *guid, Error **errp)
disk->unit = addr.Lun; disk->unit = addr.Lun;
disk->target = addr.TargetId; disk->target = addr.TargetId;
disk->bus = addr.PathId; disk->bus = addr.PathId;
disk->pci_controller = get_pci_info(name, errp);
} }
/* We do not set error in this case, because we still have enough /* We do not set error in this case, because we still have enough
* information about volume. */ * information about volume. */
} else {
disk->pci_controller = NULL;
} }
list = g_malloc0(sizeof(*list)); list = g_malloc0(sizeof(*list));