使用DeviceIoControl()执行USB write(10)命令返回超时是为什么啊?

麦子邪 2018-09-26 09:14:05
#include <stdio.h>
#include <stdlib.h>
#include <windows.h>
#define ULONG_PTR ULONG
#include <ddk/ntddscsi.h> //SDK里面的头文件
#include <ddk/ntifs.h>

int main()
{
HANDLE hUDisk;
SCSI_PASS_THROUGH_DIRECT sptdwb;
ULONG length = 0;
DWORD bytesReturn;
BYTE bufDataRead[64*1024+10];
int iRet;

//获得U盘句柄 ,假设U盘的盘符为I:盘
hUDisk = CreateFile("\\\\.\\F:",GENERIC_READ | GENERIC_WRITE,
FILE_SHARE_READ | FILE_SHARE_WRITE, NULL,OPEN_EXISTING,0,NULL
);
if (hUDisk ==INVALID_HANDLE_VALUE)
{
printf("Get Udisk hadnler error\n");
return 0;
}
ZeroMemory(&sptdwb, sizeof(SCSI_PASS_THROUGH_DIRECT));
sptdwb.Length = sizeof(SCSI_PASS_THROUGH_DIRECT);
sptdwb.PathId = 0;
sptdwb.TargetId = 1;
sptdwb.Lun = 0;
sptdwb.CdbLength = 10;
sptdwb.DataIn = SCSI_IOCTL_DATA_IN;
sptdwb.SenseInfoLength = 0x0;//24;
sptdwb.DataTransferLength = 8;
sptdwb.TimeOutValue = 2;
sptdwb.DataBuffer = bufDataRead; //存放数据的数组
sptdwb.SenseInfoOffset = 0x0;//offsetof(SCSI_PASS_THROUGH_DIRECT,ucSenseBuf);
sptdwb.Cdb[0] = 0x25 ; //获取容量命令
length = sizeof(SCSI_PASS_THROUGH_DIRECT);
iRet = DeviceIoControl(hUDisk,
IOCTL_SCSI_PASS_THROUGH_DIRECT,
&sptdwb,
length,
&sptdwb,
length,
&bytesReturn,
NULL);
if (0 == iRet)
{
iRet = GetLastError();
printf("Get\n");
return 0;
}

//扇区数
int sectors = bufDataRead[0]*(1<<24) + bufDataRead[1]*(1<<16)
+ bufDataRead[2]*(1<<8) + bufDataRead[3] + 1;
//每个扇区的字节数
int bytesPerSector = bufDataRead[4]*(1<<24) + bufDataRead[5]*(1<<16)
+ bufDataRead[6]*(1<<8) + bufDataRead[7];
printf("sectors=%d bytesPreSector=%d\n",sectors,bytesPerSector);


int posSector = 10000; //从第65个扇区开始读,扇区编号从0开始
int readSectors = 1 ; //读2个扇区
#if 1
//锁定卷
iRet = DeviceIoControl(hUDisk,
FSCTL_LOCK_VOLUME,
NULL,
0,
NULL,
0,
&bytesReturn,
NULL);
if (0 == iRet)
{
iRet = GetLastError();
printf("lock error\n");
return 0;
}
#endif
ZeroMemory(&sptdwb, sizeof(SCSI_PASS_THROUGH_DIRECT));
sptdwb.Length = sizeof(SCSI_PASS_THROUGH_DIRECT);
sptdwb.PathId = 0;
sptdwb.TargetId = 1;
sptdwb.Lun = 0;
sptdwb.CdbLength = 10; //SCSI命令长度
sptdwb.DataIn = SCSI_IOCTL_DATA_IN;
sptdwb.SenseInfoLength = 0x0;//24;
sptdwb.DataTransferLength = bytesPerSector * readSectors; //写数据量
sptdwb.TimeOutValue = 2000;
sptdwb.DataBuffer = bufDataRead;
sptdwb.SenseInfoOffset = 0x0;//offsetof(SCSI_PASS_THROUGH_DIRECT_WITH_BUFFER,ucSenseBuf);
sptdwb.Cdb[0] = 0x2A; //command write(10)
sptdwb.Cdb[2] = (posSector>>24)&0xff; //从第posSector开始读
sptdwb.Cdb[3] = (posSector>>16)&0xff;
sptdwb.Cdb[4] = (posSector>>8)&0xff;
sptdwb.Cdb[5] = posSector&0xff;
sptdwb.Cdb[7] = (readSectors>>8)&0xff;
sptdwb.Cdb[8] = readSectors&0xff; //write readSectors ,注意这个值一定要与DataTransferLength相对应
length = sizeof(SCSI_PASS_THROUGH_DIRECT);
iRet = DeviceIoControl(hUDisk,
IOCTL_SCSI_PASS_THROUGH_DIRECT,
&sptdwb,
length,
&sptdwb,
length,
&bytesReturn,
FALSE);
if (0 == iRet)
{
iRet = GetLastError(); //Get Last Error code is 121, timeout
printf("write udisk data error\n");
return 0;
}

//unlock volume
iRet = DeviceIoControl(hUDisk,
FSCTL_UNLOCK_VOLUME,
NULL,
0,
NULL,
0,
&bytesReturn,
NULL);
if (0 == iRet)
{
iRet = GetLastError();
printf(" unlock error\n");
return 0;
}

CloseHandle(hUDisk);
return 0;
}




不执行 FSCTL_LOCK_VOLUME 锁定卷的话,执行write(10)指令会报访问权限问题,但是添加锁定卷后一直报超时,这是为什么呢?
...全文
365 1 打赏 收藏 转发到动态 举报
写回复
用AI写文章
1 条回复
切换为时间正序
请发表友善的回复…
发表回复
麦子邪 2018-09-27
  • 打赏
  • 举报
回复
原来问题由第90行引起,

正确的应该是 sptdwb.DataIn = SCSI_IOCTL_DATA_OUT;

方向搞反了。

16,471

社区成员

发帖
与我相关
我的任务
社区描述
VC/MFC相关问题讨论
社区管理员
  • 基础类社区
  • Web++
  • encoderlee
加入社区
  • 近7日
  • 近30日
  • 至今
社区公告

        VC/MFC社区版块或许是CSDN最“古老”的版块了,记忆之中,与CSDN的年龄几乎差不多。随着时间的推移,MFC技术渐渐的偏离了开发主流,若干年之后的今天,当我们面对着微软的这个经典之笔,内心充满着敬意,那些曾经的记忆,可以说代表着二十年前曾经的辉煌……
        向经典致敬,或许是老一代程序员内心里面难以释怀的感受。互联网大行其道的今天,我们期待着MFC技术能够恢复其曾经的辉煌,或许这个期待会永远成为一种“梦想”,或许一切皆有可能……
        我们希望这个版块可以很好的适配Web时代,期待更好的互联网技术能够使得MFC技术框架得以重现活力,……

试试用AI创作助手写篇文章吧