【问题标题】:How to get the physical device that a volume GUID path belongs to?如何获取卷 GUID 路径所属的物理设备?
【发布时间】:2019-09-06 16:24:30
【问题描述】:

我需要使用 winapi 将卷 GUID 路径转换为物理驱动器路径。

我使用FindFirstVolume/FindNextVolume 列出了我电脑上的卷,我得到以下输出:

First volume found: \\?\Volume{42f73c69-4b40-11e9-a0b2-806e6f6e6963}\
Found next volume:  \\?\Volume{8aef5fee-0000-0000-0000-100000000000}\
Found next volume:  \\?\Volume{8aef5fee-0000-0000-0000-90c30b000000}\
Found next volume:  \\?\Volume{8aef5fee-0000-0000-0000-501f1e000000}\
Found next volume:  \\?\Volume{ec716472-e587-11e8-a031-806e6f6e6963}\
Search finished: [18] There are no more files.

我需要知道每个卷所属的物理设备,例如\\.\PHYSICALDRIVE1

这是我的代码:

/* This program attempts to list the volumes and get the corresponding physical device for each one */

#include <stdio.h>
#include <windows.h>

HANDLE openDevice(const char *deviceID);
BOOL getAllVolumes();

/* Main function
   * Access the device
   * Get all volume GUID paths
   * Close handler
*/
int main()
{
    char *deviceID = "\\\\.\\PHYSICALDRIVE1";
    HANDLE hDevice = openDevice(deviceID);
    if (hDevice != INVALID_HANDLE_VALUE) { /* success */
        fprintf(stderr, "Device %s opened.\n", deviceID);
        /* find all volumes in the device and get the corresponding device to each one */
        BOOL result = getAllVolumes();
        CloseHandle(hDevice); /* close the handler to the device */
    }

    return 0;
}

HANDLE openDevice(const char *deviceID)
{
    /* Access the device */
    HANDLE hDevice = CreateFileA(
        deviceID, /* device id (get it with `$ wmic DISKDRIVE`) */
        FILE_READ_DATA | FILE_WRITE_DATA, /* read/write access */
        FILE_SHARE_READ | FILE_SHARE_WRITE, /* shared */
        NULL, /* default security attributes */
        OPEN_EXISTING, /* only open if the device exists */
        0, /* file attributes */
        NULL); /* do not copy file attributes */

    if (hDevice == INVALID_HANDLE_VALUE) { /* cannot open the physical drive */
        DWORD errorCode = GetLastError();
        fprintf(stderr, "Cannot open the device %s: [%lu]\n", deviceID, errorCode); /* print the error code and message */
    }

    return hDevice;
}

BOOL getAllVolumes()
{
    char volumePath[512] = {0};
    fprintf(stderr, "Attempt to find first volume...\n");
    HANDLE searchHandlerDevice = FindFirstVolume(volumePath, sizeof(volumePath));
    fprintf(stderr, "Done. Retrieving results...\n");
    if (searchHandlerDevice == INVALID_HANDLE_VALUE) {
        DWORD errorCode = GetLastError();
        fprintf(stderr, "Cannot find a volume: [%lu]\n", errorCode); /* print the error code and message */
        return FALSE;
    } else {
        fprintf(stdout, "First volume found: %s\n", volumePath);
    }

    /* find the other volumes */
    while (FindNextVolume(searchHandlerDevice, volumePath, sizeof(volumePath))) {
        fprintf(stdout, "Found volume: %s\n", volumePath);
    }

    /* searching failed */
    DWORD errorCode = GetLastError();
    fprintf(stderr, "Search failed: [%lu]\n", errorCode); /* print the error code and message */

    /* close the search */
    BOOL result = FindVolumeClose(searchHandlerDevice);
    if (!result) {
        fprintf(stderr, "Search failed: [%lu]\n", errorCode); /* print the error code */
    }

    return result;
}

【问题讨论】:

  • @RbMm, FindNextVolumeW 添加反斜杠以保持一致性,因为它在连接挂载点中是必需的,否则打开没有尾随反斜杠的连接(例如SetCurrentDirectoryW)将打开卷设备而不是文件系统根目录。我们还从GetVolumeNameForVolumeMountPointW 获取带有尾部反斜杠的卷名。 SetVolumeMountPointW 也需要它并将其传递到一个联结点(但不是 DOS 驱动器安装点,因为对象符号链接必须以卷设备为目标)。
  • @RbMm,也就是说,我认为他们在保持一致性方面走错了路。在需要时添加反斜杠应该是内部实现细节。
  • 另外,如果一个卷跨越多个磁盘,IOCTL_STORAGE_GET_DEVICE_NUMBER 将(或应该?)失败,在这种情况下,我们需要IOCTL_VOLUME_GET_VOLUME_DISK_EXTENTS 列出所有磁盘范围。然而,新的存储空间的实现方式不同。它将卷(未分配 GUID 名称)组合到一个新的磁盘设备中,可见卷使用这个单个磁盘。
  • @RbMm,我有空闲时间,所以我从 VM 中的两个磁盘创建了一个跨接卷。正如我所料,为卷获取IOCTL_STORAGE_GET_DEVICE_NUMBER 失败,ERROR_INVALID_FUNCTION(即STATUS_INVALID_DEVICE_REQUEST)。我必须从IOCTL_VOLUME_GET_VOLUME_DISK_EXTENTS 获取磁盘号。
  • @RbMm,这两个磁盘没有分区,即我们只有磁盘本身的“Partition0”符号链接,例如"\Device\Harddisk3\Partition0" -> "\Device\Harddisk3\DR3"。磁盘管理管理单元显示有关其他操作系统可访问性的警告。

标签: c windows winapi disk


【解决方案1】:

嗯,我在DiskDrive 课程中找到了答案。它代表运行 Windows 操作系统的计算机所看到的物理磁盘驱动器。

【讨论】:

    猜你喜欢
    • 2011-10-30
    • 1970-01-01
    • 1970-01-01
    • 2011-09-20
    • 2011-03-01
    • 1970-01-01
    • 2012-06-04
    • 1970-01-01
    • 2012-08-01
    相关资源
    最近更新 更多