DeviceIoControl获取max transfer长度

在调试USB Massstorage设备工具时遇到DeviceIoControl函数读取数据长度限制问题。通过IOCTL_STORAGE_QUERY_PROPERTY命令获取STORAGE_ADAPTER_DESCRIPTOR结构,其中的MaximumTransferLength字段指示了最大传输长度,从而实现数据分段连续接收。
摘要由CSDN通过智能技术生成

   最近在调试一个PC操作USB Massstorage设备的工具,通过DeviceIoControl()函数读取约1-2M的数据,但是传入数据长度以后函数返回错误,原因是长度超过了规定的transfer len。从网上找了半天,在国外论坛发现一个类似的问题解决,即:用IOCTL_STORAGE_QUERY_PROPERTY 命令得到一个Max transfer length,然后根据实际情况解决,没有具体说明。我自己定的方向是,将数据分段连续接收,所以需要动态地知道Max transfer length,话不多说,直接上方法:

通过IOCTL_STORAGE_QUERY_PROPERTY 读取一个STORAGE_ADAPTER_DESCRIPTOR的结构,这个结构中的MaximumTransferLength就是要Transfer length的限值。

<p>
</p><p>    int  MaxTransferLen = 0;<
可以使用DeviceIOControl函数获取物理磁盘的个数。具体实现方法如下: 1. 打开一个物理磁盘设备句柄,调用CreateFile函数,设置参数为: 文件名:物理磁盘设备路径,例如"\\\\.\\PhysicalDrive0",数字0表示第一个物理磁盘。 访问模式:GENERIC_READ | GENERIC_WRITE 共享模式:FILE_SHARE_READ | FILE_SHARE_WRITE 安全性描述符:NULL 2. 调用DeviceIOControl函数,设置参数为: 设备句柄:上一步中返回的句柄 控制码:IOCTL_STORAGE_QUERY_PROPERTY 输入缓冲区:STORAGE_PROPERTY_QUERY结构体,其中的PropertyId字段设置为StorageDeviceProperty,QueryType字段设置为PropertyStandardQuery 输入缓冲区大小:sizeof(STORAGE_PROPERTY_QUERY) 输出缓冲区:存放设备属性信息的缓冲区,可以先设置为一定大小,如果返回的信息超过缓冲区大小,则继续调用DeviceIOControl函数,并增大缓冲区大小,直到返回的信息能够全部存放在缓冲区中 输出缓冲区大小:缓冲区大小 返回值:如果函数调用成功,则返回非零值,否则返回0 3. 解析返回的设备属性信息,判断是否为物理磁盘,并统计物理磁盘的个数。 4. 关闭设备句柄,调用CloseHandle函数。 参考代码如下: ```c++ #include <windows.h> #include <winioctl.h> #include <stdio.h> int main() { HANDLE hDevice = CreateFile("\\\\.\\PhysicalDrive0", GENERIC_READ | GENERIC_WRITE, FILE_SHARE_READ | FILE_SHARE_WRITE, NULL, OPEN_EXISTING, 0, NULL); if (hDevice == INVALID_HANDLE_VALUE) { printf("CreateFile failed, error code: %d\n", GetLastError()); return 0; } STORAGE_PROPERTY_QUERY query; query.PropertyId = StorageDeviceProperty; query.QueryType = PropertyStandardQuery; STORAGE_DESCRIPTOR_HEADER header; DWORD dwBytesReturned = 0; BOOL bRet = DeviceIoControl(hDevice, IOCTL_STORAGE_QUERY_PROPERTY, &query, sizeof(query), &header, sizeof(header), &dwBytesReturned, NULL); if (!bRet) { printf("DeviceIoControl failed, error code: %d\n", GetLastError()); CloseHandle(hDevice); return 0; } DWORD dwBufferSize = sizeof(STORAGE_DEVICE_DESCRIPTOR) + 512; PSTORAGE_DEVICE_DESCRIPTOR pDesc = (PSTORAGE_DEVICE_DESCRIPTOR)malloc(dwBufferSize); ZeroMemory(pDesc, dwBufferSize); pDesc->Size = dwBufferSize; bRet = DeviceIoControl(hDevice, IOCTL_STORAGE_QUERY_PROPERTY, &query, sizeof(query), pDesc, dwBufferSize, &dwBytesReturned, NULL); if (!bRet) { printf("DeviceIoControl failed, error code: %d\n", GetLastError()); CloseHandle(hDevice); free(pDesc); return 0; } if ((pDesc->DeviceType == DIRECT_ACCESS_DEVICE) && (pDesc->BusType == BusTypeSata)) { printf("This is a physical disk.\n"); } else { printf("This is not a physical disk.\n"); } CloseHandle(hDevice); free(pDesc); return 0; } ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值