Windows 不挂载磁盘;它安装卷。但是,USBSTOR 类设备的卷并未列为设备树中的子节点。因此,您必须枚举所有卷,并进行一系列字符串操作和比较,以将 STORAGE\VOLUME 节点与 USBSTOR 节点匹配。
所有卷 GUID 值都使用FindFirstVolume函数集进行枚举。可以去除前导“\.\”和尾随“\”字符,然后将生成的字符串传递给QueryDosDevice。这提供了一个设备名称。
接下来,必须使用带有SetupDiGetClassDevs和朋友的 GUID_DEVINTERFACE_VOLUME 枚举所有卷。使用IOCTL_STORAGE_GET_DEVICE_NUMBER将每个卷的设备类型和数量与您要查找的 USBSTOR 设备进行比较。一旦这些匹配,您可以从卷中获取设备名称并将其与其他设备名称列表进行比较以查找卷 GUID。
最后,卷 GUID 可以成功地与SetVolumeMountPoint一起使用。
感谢Gabe在对我的问题的评论中提供的非常有帮助的帮助。
代码片段
从设备路径获取设备类型和编号:
STORAGE_DEVICE_NUMBER sdn;
HANDLE handle = CreateFile(devInterfaceDetail->DevicePath, 0, FILE_SHARE_READ | FILE_SHARE_WRITE, 0, OPEN_EXISTING, 0, NULL);
DWORD len = 0;
DeviceIoControl(h, IOCTL_STORAGE_GET_DEVICE_NUMBER, NULL, 0, &sdn, sizeof (sdn), &len, NULL);
通过遍历所有卷接口并比较上述代码段中的磁盘号,找到相应 USBSTOR 实例的设备名称:
std::string deviceName;
HDEVINFO devInfoSet = SetupDiGetClassDevs(&GUID_DEVINTERFACE_VOLUME, NULL, NULL, DIGCF_PRESENT | DIGCF_DEVICEINTERFACE);
SP_DEVICE_INTERFACE_DATA devInterface = { 0 };
devInterface.cbSize = sizeof (SP_DEVICE_INTERFACE_DATA);
for (int i = 0; SetupDiEnumDeviceInterfaces(devInfoSet, NULL, &GUID_DEVINTERFACE_VOLUME, i, &devInterface); ++i) {
SP_DEVINFO_DATA devInfoData = { 0 };
devInfoData.cbSize = sizeof (SP_DEVINFO_DATA);
DWORD len;
SetupDiGetDeviceInterfaceDetail(devInfoSet, &devInterface, NULL, 0, &len, &devInfoData);
std::vector<char> buf(len);
SP_DEVICE_INTERFACE_DETAIL_DATA *devInterfaceDetail = (SP_DEVICE_INTERFACE_DETAIL_DATA *) &buf[0];
devInterfaceDetail->cbSize = sizeof (SP_DEVICE_INTERFACE_DETAIL_DATA);
if (SetupDiGetDeviceInterfaceDetail(devInfoSet, &devInterface, devInterfaceDetail, len, NULL, &devInfoData)) {
if (DEVICE_NUMBER == this->getDeviceNumber(devInterfaceDetail->DevicePath)) {
std::vector<BYTE> buf(MAX_PATH + 1);
DWORD type, len;
if (SetupDiGetDeviceRegistryProperty(devInfoSet, &devInfoData, SPDRP_PHYSICAL_DEVICE_OBJECT_NAME, &type, &buf[0], buf.size(), &len)) {
deviceName.assign(buf.begin(), buf.begin() + len);
break;
}
}
}
}