qga: Fix guest-get-fsinfo PCI address collection in Windows
The Windows QEMU guest agent erroneously tries to collect PCI information directly from the physical drive. However, windows stores SCSI/IDE information with the drive and PCI information with the underlying storage controller This changes get_pci_info to use the physical drive's underlying storage controller to get PCI information. * Additionally Fixes incorrect size being passed to DeviceIoControl when getting volume extents. Can occasionally crash the guest agent Signed-off-by: Matt Hines <mhines@scalecomputing.com> *fix up some checkpatch warnings *fix domain reporting and add some sanity checks for debug Signed-off-by: Michael Roth <mdroth@linux.vnet.ibm.com>
This commit is contained in:
parent
40cebc5811
commit
996b9cdc2f
2
configure
vendored
2
configure
vendored
@ -4934,7 +4934,7 @@ int main(void) {
|
|||||||
EOF
|
EOF
|
||||||
if compile_prog "" "" ; then
|
if compile_prog "" "" ; then
|
||||||
guest_agent_ntddscsi=yes
|
guest_agent_ntddscsi=yes
|
||||||
libs_qga="-lsetupapi $libs_qga"
|
libs_qga="-lsetupapi -lcfgmgr32 $libs_qga"
|
||||||
fi
|
fi
|
||||||
fi
|
fi
|
||||||
|
|
||||||
|
@ -22,6 +22,7 @@
|
|||||||
#include <winioctl.h>
|
#include <winioctl.h>
|
||||||
#include <ntddscsi.h>
|
#include <ntddscsi.h>
|
||||||
#include <setupapi.h>
|
#include <setupapi.h>
|
||||||
|
#include <cfgmgr32.h>
|
||||||
#include <initguid.h>
|
#include <initguid.h>
|
||||||
#endif
|
#endif
|
||||||
#include <lm.h>
|
#include <lm.h>
|
||||||
@ -485,56 +486,29 @@ static GuestDiskBusType find_bus_type(STORAGE_BUS_TYPE bus)
|
|||||||
return win2qemu[(int)bus];
|
return win2qemu[(int)bus];
|
||||||
}
|
}
|
||||||
|
|
||||||
/* XXX: The following function is BROKEN!
|
|
||||||
*
|
|
||||||
* It does not work and probably has never worked. When we query for list of
|
|
||||||
* disks we get cryptic names like "\Device\0000001d" instead of
|
|
||||||
* "\PhysicalDriveX" or "\HarddiskX". Whether the names can be translated one
|
|
||||||
* way or the other for comparison is an open question.
|
|
||||||
*
|
|
||||||
* When we query volume names (the original version) we are able to match those
|
|
||||||
* but then the property queries report error "Invalid function". (duh!)
|
|
||||||
*/
|
|
||||||
|
|
||||||
/*
|
|
||||||
DEFINE_GUID(GUID_DEVINTERFACE_VOLUME,
|
|
||||||
0x53f5630dL, 0xb6bf, 0x11d0, 0x94, 0xf2,
|
|
||||||
0x00, 0xa0, 0xc9, 0x1e, 0xfb, 0x8b);
|
|
||||||
*/
|
|
||||||
DEFINE_GUID(GUID_DEVINTERFACE_DISK,
|
DEFINE_GUID(GUID_DEVINTERFACE_DISK,
|
||||||
0x53f56307L, 0xb6bf, 0x11d0, 0x94, 0xf2,
|
0x53f56307L, 0xb6bf, 0x11d0, 0x94, 0xf2,
|
||||||
0x00, 0xa0, 0xc9, 0x1e, 0xfb, 0x8b);
|
0x00, 0xa0, 0xc9, 0x1e, 0xfb, 0x8b);
|
||||||
|
DEFINE_GUID(GUID_DEVINTERFACE_STORAGEPORT,
|
||||||
|
0x2accfe60L, 0xc130, 0x11d2, 0xb0, 0x82,
|
||||||
|
0x00, 0xa0, 0xc9, 0x1e, 0xfb, 0x8b);
|
||||||
|
|
||||||
|
static GuestPCIAddress *get_pci_info(int number, Error **errp)
|
||||||
static GuestPCIAddress *get_pci_info(char *guid, Error **errp)
|
|
||||||
{
|
{
|
||||||
HDEVINFO dev_info;
|
HDEVINFO dev_info;
|
||||||
SP_DEVINFO_DATA dev_info_data;
|
SP_DEVINFO_DATA dev_info_data;
|
||||||
DWORD size = 0;
|
SP_DEVICE_INTERFACE_DATA dev_iface_data;
|
||||||
|
HANDLE dev_file;
|
||||||
int i;
|
int i;
|
||||||
char dev_name[MAX_PATH];
|
|
||||||
char *buffer = NULL;
|
|
||||||
GuestPCIAddress *pci = NULL;
|
GuestPCIAddress *pci = NULL;
|
||||||
char *name = NULL;
|
|
||||||
bool partial_pci = false;
|
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;
|
||||||
pci->function = -1;
|
pci->function = -1;
|
||||||
pci->bus = -1;
|
pci->bus = -1;
|
||||||
|
|
||||||
if (g_str_has_prefix(guid, "\\\\.\\") ||
|
|
||||||
g_str_has_prefix(guid, "\\\\?\\")) {
|
|
||||||
name = g_strdup(guid + 4);
|
|
||||||
} else {
|
|
||||||
name = g_strdup(guid);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!QueryDosDevice(name, dev_name, ARRAY_SIZE(dev_name))) {
|
|
||||||
error_setg_win32(errp, GetLastError(), "failed to get dos device name");
|
|
||||||
goto out;
|
|
||||||
}
|
|
||||||
|
|
||||||
dev_info = SetupDiGetClassDevs(&GUID_DEVINTERFACE_DISK, 0, 0,
|
dev_info = SetupDiGetClassDevs(&GUID_DEVINTERFACE_DISK, 0, 0,
|
||||||
DIGCF_PRESENT | DIGCF_DEVICEINTERFACE);
|
DIGCF_PRESENT | DIGCF_DEVICEINTERFACE);
|
||||||
if (dev_info == INVALID_HANDLE_VALUE) {
|
if (dev_info == INVALID_HANDLE_VALUE) {
|
||||||
@ -544,90 +518,220 @@ static GuestPCIAddress *get_pci_info(char *guid, Error **errp)
|
|||||||
|
|
||||||
g_debug("enumerating devices");
|
g_debug("enumerating devices");
|
||||||
dev_info_data.cbSize = sizeof(SP_DEVINFO_DATA);
|
dev_info_data.cbSize = sizeof(SP_DEVINFO_DATA);
|
||||||
|
dev_iface_data.cbSize = sizeof(SP_DEVICE_INTERFACE_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, data, size2;
|
PSP_DEVICE_INTERFACE_DETAIL_DATA pdev_iface_detail_data = NULL;
|
||||||
int func, dev;
|
STORAGE_DEVICE_NUMBER sdn;
|
||||||
while (!SetupDiGetDeviceRegistryProperty(dev_info, &dev_info_data,
|
char *parent_dev_id = NULL;
|
||||||
SPDRP_PHYSICAL_DEVICE_OBJECT_NAME,
|
HDEVINFO parent_dev_info;
|
||||||
&data, (PBYTE)buffer, size,
|
SP_DEVINFO_DATA parent_dev_info_data;
|
||||||
&size2)) {
|
DWORD j;
|
||||||
size = MAX(size, size2);
|
DWORD size = 0;
|
||||||
if (GetLastError() == ERROR_INSUFFICIENT_BUFFER) {
|
|
||||||
g_free(buffer);
|
g_debug("getting device path");
|
||||||
/* Double the size to avoid problems on
|
if (SetupDiEnumDeviceInterfaces(dev_info, &dev_info_data,
|
||||||
* W2k MBCS systems per KB 888609.
|
&GUID_DEVINTERFACE_DISK, 0,
|
||||||
* https://support.microsoft.com/en-us/kb/259695 */
|
&dev_iface_data)) {
|
||||||
buffer = g_malloc(size * 2);
|
while (!SetupDiGetDeviceInterfaceDetail(dev_info, &dev_iface_data,
|
||||||
} else {
|
pdev_iface_detail_data,
|
||||||
|
size, &size,
|
||||||
|
&dev_info_data)) {
|
||||||
|
if (GetLastError() == ERROR_INSUFFICIENT_BUFFER) {
|
||||||
|
pdev_iface_detail_data = g_malloc(size);
|
||||||
|
pdev_iface_detail_data->cbSize =
|
||||||
|
sizeof(*pdev_iface_detail_data);
|
||||||
|
} else {
|
||||||
|
error_setg_win32(errp, GetLastError(),
|
||||||
|
"failed to get device interfaces");
|
||||||
|
goto free_dev_info;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
dev_file = CreateFile(pdev_iface_detail_data->DevicePath, 0,
|
||||||
|
FILE_SHARE_READ, NULL, OPEN_EXISTING, 0,
|
||||||
|
NULL);
|
||||||
|
g_free(pdev_iface_detail_data);
|
||||||
|
|
||||||
|
if (!DeviceIoControl(dev_file, IOCTL_STORAGE_GET_DEVICE_NUMBER,
|
||||||
|
NULL, 0, &sdn, sizeof(sdn), &size, NULL)) {
|
||||||
|
CloseHandle(dev_file);
|
||||||
error_setg_win32(errp, GetLastError(),
|
error_setg_win32(errp, GetLastError(),
|
||||||
"failed to get device name");
|
"failed to get device slot number");
|
||||||
goto free_dev_info;
|
goto free_dev_info;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
CloseHandle(dev_file);
|
||||||
|
if (sdn.DeviceNumber != number) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
error_setg_win32(errp, GetLastError(),
|
||||||
|
"failed to get device interfaces");
|
||||||
|
goto free_dev_info;
|
||||||
|
}
|
||||||
|
|
||||||
|
g_debug("found device slot %d. Getting storage controller", number);
|
||||||
|
{
|
||||||
|
CONFIGRET cr;
|
||||||
|
DEVINST dev_inst, parent_dev_inst;
|
||||||
|
ULONG dev_id_size = 0;
|
||||||
|
|
||||||
|
size = 0;
|
||||||
|
while (!SetupDiGetDeviceInstanceId(dev_info, &dev_info_data,
|
||||||
|
parent_dev_id, size, &size)) {
|
||||||
|
if (GetLastError() == ERROR_INSUFFICIENT_BUFFER) {
|
||||||
|
parent_dev_id = g_malloc(size);
|
||||||
|
} else {
|
||||||
|
error_setg_win32(errp, GetLastError(),
|
||||||
|
"failed to get device instance ID");
|
||||||
|
goto out;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* CM API used here as opposed to
|
||||||
|
* SetupDiGetDeviceProperty(..., DEVPKEY_Device_Parent, ...)
|
||||||
|
* which exports are only available in mingw-w64 6+
|
||||||
|
*/
|
||||||
|
cr = CM_Locate_DevInst(&dev_inst, parent_dev_id, 0);
|
||||||
|
if (cr != CR_SUCCESS) {
|
||||||
|
g_error("CM_Locate_DevInst failed with code %lx", cr);
|
||||||
|
error_setg_win32(errp, GetLastError(),
|
||||||
|
"failed to get device instance");
|
||||||
|
goto out;
|
||||||
|
}
|
||||||
|
cr = CM_Get_Parent(&parent_dev_inst, dev_inst, 0);
|
||||||
|
if (cr != CR_SUCCESS) {
|
||||||
|
g_error("CM_Get_Parent failed with code %lx", cr);
|
||||||
|
error_setg_win32(errp, GetLastError(),
|
||||||
|
"failed to get parent device instance");
|
||||||
|
goto out;
|
||||||
|
}
|
||||||
|
|
||||||
|
cr = CM_Get_Device_ID_Size(&dev_id_size, parent_dev_inst, 0);
|
||||||
|
if (cr != CR_SUCCESS) {
|
||||||
|
g_error("CM_Get_Device_ID_Size failed with code %lx", cr);
|
||||||
|
error_setg_win32(errp, GetLastError(),
|
||||||
|
"failed to get parent device ID length");
|
||||||
|
goto out;
|
||||||
|
}
|
||||||
|
|
||||||
|
++dev_id_size;
|
||||||
|
if (dev_id_size > size) {
|
||||||
|
g_free(parent_dev_id);
|
||||||
|
parent_dev_id = g_malloc(dev_id_size);
|
||||||
|
}
|
||||||
|
|
||||||
|
cr = CM_Get_Device_ID(parent_dev_inst, parent_dev_id, dev_id_size,
|
||||||
|
0);
|
||||||
|
if (cr != CR_SUCCESS) {
|
||||||
|
g_error("CM_Get_Device_ID failed with code %lx", cr);
|
||||||
|
error_setg_win32(errp, GetLastError(),
|
||||||
|
"failed to get parent device ID");
|
||||||
|
goto out;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (g_strcmp0(buffer, dev_name)) {
|
g_debug("querying storage controller %s for PCI information",
|
||||||
continue;
|
parent_dev_id);
|
||||||
}
|
parent_dev_info =
|
||||||
g_debug("found device %s", dev_name);
|
SetupDiGetClassDevs(&GUID_DEVINTERFACE_STORAGEPORT, parent_dev_id,
|
||||||
|
NULL, DIGCF_PRESENT | DIGCF_DEVICEINTERFACE);
|
||||||
|
g_free(parent_dev_id);
|
||||||
|
|
||||||
/* There is no need to allocate buffer in the next functions. The size
|
if (parent_dev_info == INVALID_HANDLE_VALUE) {
|
||||||
* is known and ULONG according to
|
error_setg_win32(errp, GetLastError(),
|
||||||
* https://support.microsoft.com/en-us/kb/253232
|
"failed to get parent device");
|
||||||
* https://msdn.microsoft.com/en-us/library/windows/hardware/ff543095(v=vs.85).aspx
|
goto out;
|
||||||
*/
|
|
||||||
if (!SetupDiGetDeviceRegistryProperty(dev_info, &dev_info_data,
|
|
||||||
SPDRP_BUSNUMBER, &data, (PBYTE)&bus, size, NULL)) {
|
|
||||||
debug_error("failed to get bus");
|
|
||||||
bus = -1;
|
|
||||||
partial_pci = true;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* The function retrieves the device's address. This value will be
|
parent_dev_info_data.cbSize = sizeof(SP_DEVINFO_DATA);
|
||||||
* transformed into device function and number */
|
if (!SetupDiEnumDeviceInfo(parent_dev_info, 0, &parent_dev_info_data)) {
|
||||||
if (!SetupDiGetDeviceRegistryProperty(dev_info, &dev_info_data,
|
error_setg_win32(errp, GetLastError(),
|
||||||
SPDRP_ADDRESS, &data, (PBYTE)&addr, size, NULL)) {
|
"failed to get parent device data");
|
||||||
debug_error("failed to get address");
|
goto out;
|
||||||
addr = -1;
|
|
||||||
partial_pci = true;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* This call returns UINumber of DEVICE_CAPABILITIES structure.
|
for (j = 0;
|
||||||
* This number is typically a user-perceived slot number. */
|
SetupDiEnumDeviceInfo(parent_dev_info, j, &parent_dev_info_data);
|
||||||
if (!SetupDiGetDeviceRegistryProperty(dev_info, &dev_info_data,
|
j++) {
|
||||||
SPDRP_UI_NUMBER, &data, (PBYTE)&slot, size, NULL)) {
|
DWORD addr, bus, ui_slot, type;
|
||||||
debug_error("failed to get slot");
|
int func, slot;
|
||||||
slot = -1;
|
|
||||||
partial_pci = true;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* SetupApi gives us the same information as driver with
|
/*
|
||||||
* IoGetDeviceProperty. According to Microsoft
|
* There is no need to allocate buffer in the next functions. The
|
||||||
* https://support.microsoft.com/en-us/kb/253232
|
* size is known and ULONG according to
|
||||||
* FunctionNumber = (USHORT)((propertyAddress) & 0x0000FFFF);
|
* https://msdn.microsoft.com/en-us/library/windows/hardware/ff543095(v=vs.85).aspx
|
||||||
* DeviceNumber = (USHORT)(((propertyAddress) >> 16) & 0x0000FFFF);
|
*/
|
||||||
* SPDRP_ADDRESS is propertyAddress, so we do the same.*/
|
if (!SetupDiGetDeviceRegistryProperty(
|
||||||
|
parent_dev_info, &parent_dev_info_data, SPDRP_BUSNUMBER,
|
||||||
|
&type, (PBYTE)&bus, size, NULL)) {
|
||||||
|
debug_error("failed to get PCI bus");
|
||||||
|
bus = -1;
|
||||||
|
partial_pci = true;
|
||||||
|
}
|
||||||
|
|
||||||
if (partial_pci) {
|
/*
|
||||||
pci->domain = -1;
|
* The function retrieves the device's address. This value will be
|
||||||
pci->slot = -1;
|
* transformed into device function and number
|
||||||
pci->function = -1;
|
*/
|
||||||
pci->bus = -1;
|
if (!SetupDiGetDeviceRegistryProperty(
|
||||||
} else {
|
parent_dev_info, &parent_dev_info_data, SPDRP_ADDRESS,
|
||||||
func = ((int) addr == -1) ? -1 : addr & 0x0000FFFF;
|
&type, (PBYTE)&addr, size, NULL)) {
|
||||||
dev = ((int) addr == -1) ? -1 : (addr >> 16) & 0x0000FFFF;
|
debug_error("failed to get PCI address");
|
||||||
pci->domain = dev;
|
addr = -1;
|
||||||
pci->slot = (int) slot;
|
partial_pci = true;
|
||||||
pci->function = func;
|
}
|
||||||
pci->bus = (int) bus;
|
|
||||||
|
/*
|
||||||
|
* This call returns UINumber of DEVICE_CAPABILITIES structure.
|
||||||
|
* This number is typically a user-perceived slot number.
|
||||||
|
*/
|
||||||
|
if (!SetupDiGetDeviceRegistryProperty(
|
||||||
|
parent_dev_info, &parent_dev_info_data, SPDRP_UI_NUMBER,
|
||||||
|
&type, (PBYTE)&ui_slot, size, NULL)) {
|
||||||
|
debug_error("failed to get PCI slot");
|
||||||
|
ui_slot = -1;
|
||||||
|
partial_pci = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* SetupApi gives us the same information as driver with
|
||||||
|
* IoGetDeviceProperty. According to Microsoft:
|
||||||
|
*
|
||||||
|
* FunctionNumber = (USHORT)((propertyAddress) & 0x0000FFFF)
|
||||||
|
* DeviceNumber = (USHORT)(((propertyAddress) >> 16) & 0x0000FFFF)
|
||||||
|
* SPDRP_ADDRESS is propertyAddress, so we do the same.
|
||||||
|
*
|
||||||
|
* https://docs.microsoft.com/en-us/windows/desktop/api/setupapi/nf-setupapi-setupdigetdeviceregistrypropertya
|
||||||
|
*/
|
||||||
|
if (partial_pci) {
|
||||||
|
pci->domain = -1;
|
||||||
|
pci->slot = -1;
|
||||||
|
pci->function = -1;
|
||||||
|
pci->bus = -1;
|
||||||
|
continue;
|
||||||
|
} else {
|
||||||
|
func = ((int)addr == -1) ? -1 : addr & 0x0000FFFF;
|
||||||
|
slot = ((int)addr == -1) ? -1 : (addr >> 16) & 0x0000FFFF;
|
||||||
|
if ((int)ui_slot != slot) {
|
||||||
|
g_debug("mismatch with reported slot values: %d vs %d",
|
||||||
|
(int)ui_slot, slot);
|
||||||
|
}
|
||||||
|
pci->domain = 0;
|
||||||
|
pci->slot = (int)ui_slot;
|
||||||
|
pci->function = func;
|
||||||
|
pci->bus = (int)bus;
|
||||||
|
break;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
SetupDiDestroyDeviceInfoList(parent_dev_info);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
free_dev_info:
|
free_dev_info:
|
||||||
SetupDiDestroyDeviceInfoList(dev_info);
|
SetupDiDestroyDeviceInfoList(dev_info);
|
||||||
out:
|
out:
|
||||||
g_free(buffer);
|
|
||||||
g_free(name);
|
|
||||||
return pci;
|
return pci;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -685,7 +789,8 @@ out_free:
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void get_single_disk_info(GuestDiskAddress *disk, Error **errp)
|
static void get_single_disk_info(int disk_number,
|
||||||
|
GuestDiskAddress *disk, Error **errp)
|
||||||
{
|
{
|
||||||
SCSI_ADDRESS addr, *scsi_ad;
|
SCSI_ADDRESS addr, *scsi_ad;
|
||||||
DWORD len;
|
DWORD len;
|
||||||
@ -714,7 +819,7 @@ static void get_single_disk_info(GuestDiskAddress *disk, Error **errp)
|
|||||||
* if that doesn't hold since that suggests some other unexpected
|
* if that doesn't hold since that suggests some other unexpected
|
||||||
* breakage
|
* breakage
|
||||||
*/
|
*/
|
||||||
disk->pci_controller = get_pci_info(disk->dev, &local_err);
|
disk->pci_controller = get_pci_info(disk_number, &local_err);
|
||||||
if (local_err) {
|
if (local_err) {
|
||||||
error_propagate(errp, local_err);
|
error_propagate(errp, local_err);
|
||||||
goto err_close;
|
goto err_close;
|
||||||
@ -728,7 +833,7 @@ static void get_single_disk_info(GuestDiskAddress *disk, Error **errp)
|
|||||||
/* We are able to use the same ioctls for different bus types
|
/* We are able to use the same ioctls for different bus types
|
||||||
* according to Microsoft docs
|
* according to Microsoft docs
|
||||||
* https://technet.microsoft.com/en-us/library/ee851589(v=ws.10).aspx */
|
* https://technet.microsoft.com/en-us/library/ee851589(v=ws.10).aspx */
|
||||||
g_debug("getting pci-controller info");
|
g_debug("getting SCSI info");
|
||||||
if (DeviceIoControl(disk_h, IOCTL_SCSI_GET_ADDRESS, NULL, 0, scsi_ad,
|
if (DeviceIoControl(disk_h, IOCTL_SCSI_GET_ADDRESS, NULL, 0, scsi_ad,
|
||||||
sizeof(SCSI_ADDRESS), &len, NULL)) {
|
sizeof(SCSI_ADDRESS), &len, NULL)) {
|
||||||
disk->unit = addr.Lun;
|
disk->unit = addr.Lun;
|
||||||
@ -776,12 +881,10 @@ static GuestDiskAddressList *build_guest_disk_info(char *guid, Error **errp)
|
|||||||
size = sizeof(VOLUME_DISK_EXTENTS);
|
size = sizeof(VOLUME_DISK_EXTENTS);
|
||||||
extents = g_malloc0(size);
|
extents = g_malloc0(size);
|
||||||
if (!DeviceIoControl(vol_h, IOCTL_VOLUME_GET_VOLUME_DISK_EXTENTS, NULL,
|
if (!DeviceIoControl(vol_h, IOCTL_VOLUME_GET_VOLUME_DISK_EXTENTS, NULL,
|
||||||
0, extents, size, NULL, NULL)) {
|
0, extents, size, &size, NULL)) {
|
||||||
DWORD last_err = GetLastError();
|
DWORD last_err = GetLastError();
|
||||||
if (last_err == ERROR_MORE_DATA) {
|
if (last_err == ERROR_MORE_DATA) {
|
||||||
/* Try once more with big enough buffer */
|
/* Try once more with big enough buffer */
|
||||||
size = sizeof(VOLUME_DISK_EXTENTS)
|
|
||||||
+ extents->NumberOfDiskExtents*sizeof(DISK_EXTENT);
|
|
||||||
g_free(extents);
|
g_free(extents);
|
||||||
extents = g_malloc0(size);
|
extents = g_malloc0(size);
|
||||||
if (!DeviceIoControl(
|
if (!DeviceIoControl(
|
||||||
@ -797,7 +900,7 @@ static GuestDiskAddressList *build_guest_disk_info(char *guid, Error **errp)
|
|||||||
disk = g_malloc0(sizeof(GuestDiskAddress));
|
disk = g_malloc0(sizeof(GuestDiskAddress));
|
||||||
disk->has_dev = true;
|
disk->has_dev = true;
|
||||||
disk->dev = g_strdup(name);
|
disk->dev = g_strdup(name);
|
||||||
get_single_disk_info(disk, &local_err);
|
get_single_disk_info(0xffffffff, disk, &local_err);
|
||||||
if (local_err) {
|
if (local_err) {
|
||||||
g_debug("failed to get disk info, ignoring error: %s",
|
g_debug("failed to get disk info, ignoring error: %s",
|
||||||
error_get_pretty(local_err));
|
error_get_pretty(local_err));
|
||||||
@ -831,9 +934,9 @@ static GuestDiskAddressList *build_guest_disk_info(char *guid, Error **errp)
|
|||||||
*/
|
*/
|
||||||
disk->has_dev = true;
|
disk->has_dev = true;
|
||||||
disk->dev = g_strdup_printf("\\\\.\\PhysicalDrive%lu",
|
disk->dev = g_strdup_printf("\\\\.\\PhysicalDrive%lu",
|
||||||
extents->Extents[i].DiskNumber);
|
extents->Extents[i].DiskNumber);
|
||||||
|
|
||||||
get_single_disk_info(disk, &local_err);
|
get_single_disk_info(extents->Extents[i].DiskNumber, disk, &local_err);
|
||||||
if (local_err) {
|
if (local_err) {
|
||||||
error_propagate(errp, local_err);
|
error_propagate(errp, local_err);
|
||||||
goto out;
|
goto out;
|
||||||
|
Loading…
Reference in New Issue
Block a user