首先是对于端点的数据处理

#ifndef __USBEP2_H_

#define __USBEP2_H_

#include "usb.h"

#include "usbhw.h"

#include "msc.h"

#include "mscuser.h"

void usb_ep2_in_process(void);

void usb_ep2_out_process(void);

#endif

#include "usbep2.h"

//批量输入事件

void usb_ep2_in_process(void)

{

switch (BulkStage)

{

case MSC_BS_DATA_IN:

switch (CBW.CB[0])

{

case SCSI_READ10:

MSC_MemoryRead();//读取数据并等待下一次传输

break;

}

break;

case MSC_BS_DATA_IN_LAST:

MSC_SetCSW();//上一次传输并且传输已经完成

break;

case MSC_BS_DATA_IN_LAST_STALL:

USB_SetStallEP(MSC_EP_IN);

MSC_SetCSW();

break;

case MSC_BS_CSW:

BulkStage = MSC_BS_CBW;//简单的进行状态切换

break;

}

}

//批量输出事件

void usb_ep2_out_process(void)

{

BulkLen = USB_ReadEP(MSC_EP_OUT, BulkBuf);//读取缓存

switch (BulkStage)//根据阶段判定

{

case MSC_BS_CBW://最开始的阶段必然是命令阶段

MSC_GetCBW();//获取并处理cbw命令

break;

case MSC_BS_DATA_OUT://数据输出阶段

switch (CBW.CB[0]) //分辨写入指令

{

case SCSI_WRITE10:

MSC_MemoryWrite();

break;

case SCSI_VERIFY10:

MSC_MemoryVerify();

break;

}

break;

default://不支持的状态

USB_SetStallEP(MSC_EP_OUT);

CSW.bStatus = CSW_PHASE_ERROR;

MSC_SetCSW();

break;

}

}

然后再在子程序中对命令详细分析

#ifndef __MSCUSER_H_

#define __MSCUSER_H_

#include "msc.h"

#include "usbhw.h"

#include "memory.h"

//批量传输节点的最大包长度

#define MSC_MAX_PACKET  64

//批量传输节点地址

#define MSC_EP_IN       0x82

#define MSC_EP_OUT      0x02

extern uint8_t  BulkStage;               //传输状态,指明下一次如何传输

extern uint8_t  BulkLen;                 //输出接点,主机输出的数据长度

extern uint8_t  BulkBuf[MSC_MAX_PACKET]; //数据缓存中心

extern MSC_CBW CBW;                     //cbw块

extern MSC_CSW CSW;                     //csw块

extern uint32_t  MemOK;                   /* 测试mem是否完好 */

#define MSC_DEBUG   0

#if MSC_DEBUG

#define msc_debug_printf(format,args...)    printf(format,##args)       //变参宏定义

#else

#define  msc_debug_printf(x,...)  while(0);

#endif

//枚举过程中的重要节点

extern uint32_t MSC_Reset     (void);

extern uint32_t MSC_GetMaxLUN (void);

//msc设备的方法

extern void MSC_GetCBW (void);

extern void MSC_SetCSW (void);

void MSC_MemoryRead (void) ;

void MSC_MemoryWrite(void);

void MSC_MemoryVerify(void);

#endif

#include "mscuser.h"

uint32_t  MemOK = __TRUE;                   /* 测试mem是否完好 */

uint32_t Offset;                  /* 读取写入的定位 */

uint32_t Length;                  /* 读取写入的长度 */

uint8_t  BulkStage;               /* 批量传输的阶段, 数据阶段 命令阶段 状态阶段 */

uint8_t  BulkBuf[MSC_MAX_PACKET]; /* 读取批量端点传输来的数据 */

uint8_t  BulkLen;                 /* 传输长度 */

MSC_CBW CBW;                   /*CBW结构体 */

MSC_CSW CSW;                   /*CSW结构体 */

#if NORFLASH

u8 usb_mac_global_buffer[MSC_MAX_PACKET];

#endif

#if NANDFLASH

u8 usb_mac_global_buffer[MSC_MAX_PACKET];

#endif

//msc设备复位

uint32_t MSC_Reset (void)

{

BulkStage = MSC_BS_CBW;

return (__TRUE);

}

//获取标号

uint32_t MSC_GetMaxLUN (void)

{

EP0Buf[0] = 0;    //0为一个设备

return (__TRUE);

}

//设备读取数据,在in阶段发送出去

void MSC_MemoryRead (void)

{

uint32_t n;

if (Length > MSC_MAX_PACKET) //根据长度来计算,不能大于包长度

{

n = MSC_MAX_PACKET;

}

else

{

n = Length;

}

if ((Offset + n) > MSC_MemorySize) //缓冲区最大长度

{

n = MSC_MemorySize - Offset;

BulkStage = MSC_BS_DATA_IN_LAST_STALL;//传输失败的发送

}

#if NORFLASH

NOR_FLASH_Read_Buffer_Mal(Offset,usb_mac_global_buffer,n);//读取数据

USB_WriteEP(MSC_EP_IN, (u8*)usb_mac_global_buffer, n);

#endif

#if NANDFLASH

while(nandReady == 0);

nandReady = 0;

NAND_Read_Addr_Mal(Offset,usb_mac_global_buffer,n);

USB_WriteEP(MSC_EP_IN, (u8*)usb_mac_global_buffer, n);

nandReady = 1;

#endif

Offset += n;

Length -= n;//传输完成后对指针进行相应的处理

CSW.dDataResidue -= n;

if (Length == 0)

{

BulkStage = MSC_BS_DATA_IN_LAST;//数据传输成功

}

if (BulkStage != MSC_BS_DATA_IN) //上一次传输为0 ,命令通过

{

CSW.bStatus = CSW_CMD_PASSED;

}

}

u8 norflash_buffer[2048] = {0};

//写入数据到设备

void MSC_MemoryWrite (void)

{

if ((Offset + BulkLen) > MSC_MemorySize) //防止超界

{

BulkLen = MSC_MemorySize - Offset;

BulkStage = MSC_BS_CSW;

USB_SetStallEP(MSC_EP_OUT);

}

#if NORFLASH

NOR_FLASH_Write_Mal(Offset,BulkBuf,BulkLen);

msc_idle_time_count = 0;//发生写入时,空闲时间设置为0

#endif

#if NANDFLASH

while(nandReady == 0);

nandReady = 0;

NAND_Write_Addr_Mal(Offset,BulkBuf,BulkLen);

nandReady = 1;

msc_idle_time_count = 0;

#endif

Offset += BulkLen;//指针变换(写入和删除都有相应的执指针变化,自动切换位置)

Length -= BulkLen;

CSW.dDataResidue -= BulkLen;

if ((Length == 0) || (BulkStage == MSC_BS_CSW))

{

CSW.bStatus = CSW_CMD_PASSED;

MSC_SetCSW();//指令通过

}

}

//数据校验

void MSC_MemoryVerify (void)

{

uint32_t n;

if ((Offset + BulkLen) > MSC_MemorySize)  //防止超界

{

BulkLen = MSC_MemorySize - Offset;

BulkStage = MSC_BS_CSW;

USB_SetStallEP(MSC_EP_OUT);

}

#if NORFLASH

NOR_FLASH_Read_Buffer_Mal(Offset,usb_mac_global_buffer,BulkLen);

for(n = 0; n < BulkLen;n++)

{

if(usb_mac_global_buffer[n] != BulkBuf[n])

{

MemOK = __FALSE;

break;

}

}

#endif

#if NANDFLASH

while(nandReady == 0);

nandReady = 0;

NAND_Read_Addr_Mal(Offset,usb_mac_global_buffer,BulkLen);

nandReady = 1;

for(n = 0; n < BulkLen;n++)

{

if(usb_mac_global_buffer[n] != BulkBuf[n])

{

MemOK = __FALSE;

break;

}

}

#endif

Offset += BulkLen;

Length -= BulkLen;

CSW.dDataResidue -= BulkLen;

if ((Length == 0) || (BulkStage == MSC_BS_CSW))

{

CSW.bStatus = (MemOK) ? CSW_CMD_PASSED : CSW_CMD_FAILED;//根据校验的最终状态,选择返回命令失败或者成功

MSC_SetCSW();

}

}

//usb实际与数据相关的处理工作

uint32_t MSC_RWSetup (void)

{

uint32_t n;

//首先获得逻辑地址对应的物理地址,主机发送下来的block地址,我们要转换成响应的nand block和nand page

n = (CBW.CB[2] << 24) |(CBW.CB[3] << 16) |(CBW.CB[4] <<  8) | (CBW.CB[5] <<  0);

Offset = n * MSC_BlockSize;//定位逻辑位置

//需要传输的块数量

n = (CBW.CB[7] <<  8) |

(CBW.CB[8] <<  0);

//要传输的数据总长度

Length = n * MSC_BlockSize;

if (CBW.dDataLength != Length) //接受长度必须和发送的要读取长度计算一致,否则不能起作用

{

USB_SetStallEP(MSC_EP_IN);

USB_SetStallEP(MSC_EP_OUT);

CSW.bStatus = CSW_PHASE_ERROR;

MSC_SetCSW();

return (__FALSE);

}

return (__TRUE);//算好了,核心是offect 和length

}

//检测数据是否符合规则

uint32_t DataInFormat (void)

{

if (CBW.dDataLength == 0) //检测数据长度

{

CSW.bStatus = CSW_PHASE_ERROR;

MSC_SetCSW();

return (__FALSE);

}

if ((CBW.bmFlags & 0x80) == 0) //检测数据方向

{

USB_SetStallEP(MSC_EP_OUT);

CSW.bStatus = CSW_PHASE_ERROR;

MSC_SetCSW();

return (__FALSE);

}

return (__TRUE);//数据ok

}

void DataInTransfer (void)

{

if (BulkLen > CBW.dDataLength) {

BulkLen = CBW.dDataLength;

}

USB_WriteEP(MSC_EP_IN, BulkBuf, BulkLen);

BulkStage = MSC_BS_DATA_IN_LAST;

CSW.dDataResidue -= BulkLen;

CSW.bStatus = CSW_CMD_PASSED;

}

//scsi指明测试存储单元是否准备好

void MSC_TestUnitReady (void)

{

if (CBW.dDataLength != 0) //发送来的cbw不带数据就准备好了

{

if ((CBW.bmFlags & 0x80) != 0)

{

USB_SetStallEP(MSC_EP_IN);//设置断点暂停

} else

{

USB_SetStallEP(MSC_EP_OUT);

}

}

#if NANDFLASH

if(nandReady == 0)MemOK = __FALSE;

else MemOK = __TRUE;

#endif

if(MemOK == __TRUE)

{

CSW.bStatus = CSW_CMD_PASSED;//准备好存储空间

MSC_SetCSW();

}

else

{

CSW.bStatus = CSW_CMD_FAILED;//还没准备好存储空间

MSC_SetCSW();

}

#if NORFLASH

if(msc_idle_time_count < MSC_IDLE_TIME_LIMIT)msc_idle_time_count++;//设备空闲时间(没发生写入的时间)

else NOR_FLASH_Flush();

#endif

#if NANDFLASH

if(msc_idle_time_count < MSC_IDLE_TIME_LIMIT)msc_idle_time_count++;//设备空闲时间(没发生写入的时间)

else Nand_Flush();

#endif

}

//检测上一次传输失败的原因 返回失败原因代码

void MSC_RequestSense (void)

{

if (!DataInFormat()) return;

BulkBuf[ 0] = 0x70;          //错误代码,固定为0x70

BulkBuf[ 1] = 0x00;         //保留

BulkBuf[ 2] = 0x00;         //Sense Key为0x05,表示无效请求(ILLEGAL REQUEST)

BulkBuf[ 3] = 0x00;         //Information为0 四个字节

BulkBuf[ 4] = 0x00;

BulkBuf[ 5] = 0x00;

BulkBuf[ 6] = 0x00;

BulkBuf[ 7] = 0x0A;          //附加数据长度为10字节

BulkBuf[ 8] = 0x00;          //保留 四字节

BulkBuf[ 9] = 0x00;

BulkBuf[10] = 0x00;

BulkBuf[11] = 0x00;

BulkBuf[12] = 0x00;          //Additional Sense Code(ASC)为0x20,表示无效命令操作码(INVALID COMMAND OPERATION CODE)

BulkBuf[13] = 0x00;          //Additional Sense Code Qualifier(ASCQ)为0

BulkBuf[14] = 0x00;          //保留 四字节

BulkBuf[15] = 0x00;

BulkBuf[16] = 0x00;

BulkBuf[17] = 0x00;

if (CBW.CB[4] <= 18)

{

BulkLen = CBW.CB[4];

}

else

{

BulkLen = 18;

}

DataInTransfer();

}

void MSC_Inquiry (void)

{

if (!DataInFormat()) return;

BulkBuf[ 0] = 0x00;          //磁盘设备

BulkBuf[ 1] = 0x80;          //其中最高位D7为RMB。RMB=0,表示不可移除设备。如果RMB=1,则为可移除设备。

BulkBuf[ 2] = 0x00;          //各种版本号0

BulkBuf[ 3] = 0x01;          //数据响应格式

BulkBuf[ 4] = 36-4;          //附加数据长度,为32字节

BulkBuf[ 5] = 0x80;          /* SCCS = 1: Storage Controller Component */

BulkBuf[ 6] = 0x00;          //保留

BulkBuf[ 7] = 0x00;          //保留

BulkBuf[ 8] = 'D';           //厂商标识

BulkBuf[ 9] = 'I';

BulkBuf[10] = 'N';

BulkBuf[11] = 'K';

BulkBuf[12] = ' ';

BulkBuf[13] = ' ';

BulkBuf[14] = ' ';

BulkBuf[15] = ' ';

BulkBuf[16] = 'D';           //产品标识,

BulkBuf[17] = 'E';

BulkBuf[18] = 'N';

BulkBuf[19] = 'G';

BulkBuf[20] = 'X';

BulkBuf[21] = 'I';

BulkBuf[22] = 'A';

BulkBuf[23] = 'O';

BulkBuf[24] = 'J';

BulkBuf[25] = 'U';

BulkBuf[26] = 'N';

BulkBuf[27] = 'D';

BulkBuf[28] = 'I';

BulkBuf[29] = 'S';

BulkBuf[30] = 'K';

BulkBuf[31] = ' ';

BulkBuf[32] = '1';           //产品版本号

BulkBuf[33] = '.';

BulkBuf[34] = '0';

BulkBuf[35] = ' ';

BulkLen = 36;

DataInTransfer();

}

//scsi响应函数

void MSC_ModeSense6 (void)

{

if (!DataInFormat()) return;

BulkBuf[ 0] = 0x03;

BulkBuf[ 1] = 0x00;

BulkBuf[ 2] = 0x00;

BulkBuf[ 3] = 0x00;

BulkLen = 4;

DataInTransfer();

}

void MSC_ModeSense10 (void) {

if (!DataInFormat()) return;

BulkBuf[ 0] = 0x00;

BulkBuf[ 1] = 0x06;

BulkBuf[ 2] = 0x00;

BulkBuf[ 3] = 0x00;

BulkBuf[ 4] = 0x00;

BulkBuf[ 5] = 0x00;

BulkBuf[ 6] = 0x00;

BulkBuf[ 7] = 0x00;

BulkLen = 8;

DataInTransfer();

}

//读取设备的实际容量

void MSC_ReadCapacity (void)

{

if (!DataInFormat()) return;

//最大逻辑块大小 因为逻辑块从0开始,所以需要-1

BulkBuf[ 0] = ((MSC_BlockCount - 1) >> 24) & 0xFF;

BulkBuf[ 1] = ((MSC_BlockCount - 1) >> 16) & 0xFF;

BulkBuf[ 2] = ((MSC_BlockCount - 1) >>  8) & 0xFF;

BulkBuf[ 3] = ((MSC_BlockCount - 1) >>  0) & 0xFF;

//逻辑块长度

BulkBuf[ 4] = (MSC_BlockSize >> 24) & 0xFF;

BulkBuf[ 5] = (MSC_BlockSize >> 16) & 0xFF;

BulkBuf[ 6] = (MSC_BlockSize >>  8) & 0xFF;

BulkBuf[ 7] = (MSC_BlockSize >>  0) & 0xFF;

BulkLen = 8;

DataInTransfer();

}

//读格式化容量信息,返回最大能格式化的数据信息

void MSC_ReadFormatCapacity (void)

{

if (!DataInFormat()) return;

BulkBuf[ 0] = 0x00;

BulkBuf[ 1] = 0x00;

BulkBuf[ 2] = 0x00;

BulkBuf[ 3] = 0x08;          //容量列表描述符长度

//设备块数量 正对于不同的介质这里可以有不同的设置

BulkBuf[ 4] = (MSC_BlockCount >> 24) & 0xFF;

BulkBuf[ 5] = (MSC_BlockCount >> 16) & 0xFF;

BulkBuf[ 6] = (MSC_BlockCount >>  8) & 0xFF;

BulkBuf[ 7] = (MSC_BlockCount >>  0) & 0xFF;

//每一块的长度

BulkBuf[ 8] = 0x02;                     /* Descriptor Code: Formatted Media */

BulkBuf[ 9] = (MSC_BlockSize >> 16) & 0xFF;

BulkBuf[10] = (MSC_BlockSize >>  8) & 0xFF;

BulkBuf[11] = (MSC_BlockSize >>  0) & 0xFF;

BulkLen = 12;

DataInTransfer();

}

void MSC_GetCBW (void)

{

uint32_t n;//将buf数据拷贝入cbw结构体,便于下一次处理

for (n = 0; n < BulkLen; n++)

{

*((uint8_t *)&CBW + n) = BulkBuf[n];

}

if ((BulkLen == sizeof(CBW)) && (CBW.dSignature == MSC_CBW_Signature)) //检测命令长度以及命令头 usbc

{

//完整有效地cbw

CSW.dTag = CBW.dTag;//发送来的cbw需要csw返回,csw的tag必须对应发送来的cbw tag

CSW.dDataResidue = CBW.dDataLength;//需要传输的数据长度

if ((CBW.bLUN != 0) || (CBW.bCBLength < 1) || CBW.bCBLength > 16) //目标逻辑单元不对或者长度不对都会导致命令失败

{

fail: CSW.bStatus = CSW_CMD_FAILED;//命令失败,0x01

MSC_SetCSW();//返回失败csw

}

else

{

switch (CBW.CB[0]) //检测命令代码,进行散转处理

{

case SCSI_TEST_UNIT_READY://测试设备是否准备好

MSC_TestUnitReady();

msc_debug_printf("SCSI_TEST_UNIT_READY\r\n");

break;

case SCSI_REQUEST_SENSE://检测上一次传输失败的原因

MSC_RequestSense();

msc_debug_printf("SCSI_REQUEST_SENSE\r\n");

break;

case SCSI_FORMAT_UNIT:  //格式化单元,不支持

msc_debug_printf("SCSI_FORMAT_UNIT\r\n");

goto fail;

case SCSI_INQUIRY:

MSC_Inquiry();      //查询设备的基本信息

msc_debug_printf("SCSI_INQUIRY\r\n");

break;

case SCSI_START_STOP_UNIT://不支持

msc_debug_printf("SCSI_START_STOP_UNIT\r\n");

goto fail;

case SCSI_MEDIA_REMOVAL:

CSW.bStatus = CSW_CMD_PASSED;

MSC_SetCSW();//返回失败csw

msc_debug_printf("SCSI_MEDIA_REMOVAL\r\n");

//goto fail;

break;

case SCSI_MODE_SELECT6://不支持

msc_debug_printf("SCSI_MODE_SELECT6\r\n");

goto fail;

case SCSI_MODE_SENSE6://不知干嘛的

MSC_ModeSense6();

msc_debug_printf("SCSI_MODE_SENSE6\r\n");

break;

case SCSI_MODE_SELECT10:

msc_debug_printf("SCSI_MODE_SELECT10\r\n");

goto fail;

case SCSI_MODE_SENSE10:

MSC_ModeSense10();

msc_debug_printf("SCSI_MODE_SENSE10\r\n");

break;

case SCSI_READ_FORMAT_CAPACITIES://读格式化容量信息

MSC_ReadFormatCapacity();

msc_debug_printf("SCSI_READ_FORMAT_CAPACITIES\r\n");

break;

case SCSI_READ_CAPACITY://读容量信息

MSC_ReadCapacity();

msc_debug_printf("SCSI_READ_CAPACITY\r\n");

break;

case SCSI_READ10://读取实际的磁盘数据

if (MSC_RWSetup())

{

if ((CBW.bmFlags & 0x80) != 0) //方向没有错误

{

BulkStage = MSC_BS_DATA_IN;//数据输入状态,下次in事件来的时候就能传输正确数据

MSC_MemoryRead();

msc_debug_printf("SCSI_READ10 right\r\n");

}

else

{

USB_SetStallEP(MSC_EP_OUT);//方向错误,暂停端点

CSW.bStatus = CSW_CMD_FAILED;

MSC_SetCSW();

msc_debug_printf("SCSI_READ10 error\r\n");

}

}

break;

case SCSI_WRITE10://写入实际磁盘数据

if (MSC_RWSetup())

{

if ((CBW.bmFlags & 0x80) == 0) //检查方向

{

BulkStage = MSC_BS_DATA_OUT;//说明下一次out带来就是要写入数据,修改设备的状态

msc_debug_printf("SCSI_WRITE10 right\r\n");

}

else

{

USB_SetStallEP(MSC_EP_IN);//命令不支持(核心思想状态机)

CSW.bStatus = CSW_PHASE_ERROR;

MSC_SetCSW();

msc_debug_printf("SCSI_WRITE10 error\r\n");

}

}

msc_debug_printf("SCSI_WRITE10\r\n");

break;

case SCSI_VERIFY10://校验数据

if (MSC_RWSetup())

{

if ((CBW.bmFlags & 0x80) == 0)

{

BulkStage = MSC_BS_DATA_OUT;

MemOK = __TRUE;//直接返回数据校验ok

}

else

{

USB_SetStallEP(MSC_EP_IN);

CSW.bStatus = CSW_PHASE_ERROR;//方向不对,命令错误

MSC_SetCSW();

}

}

msc_debug_printf("SCSI_VERIFY10\r\n");

break;

default:

goto fail;

}

}

}

else

{//无效的cbw指令,直接禁用全部端点,等待usb重新设置端点 从错误中恢复

USB_SetStallEP(MSC_EP_IN);

USB_SetStallEP(MSC_EP_OUT);

BulkStage = MSC_BS_ERROR;

msc_debug_printf("无效的cbw指令\r\n");

}

}

//设备返回的状态封包

void MSC_SetCSW (void) {

CSW.dSignature = MSC_CSW_Signature;

USB_WriteEP(MSC_EP_IN, (uint8_t *)&CSW, sizeof(CSW));

BulkStage = MSC_BS_CSW;

}

Usb大容量存储器的宏如下

#ifndef __MSC_H__

#define __MSC_H__

#include "sys.h"

/* MSC Subclass Codes */

#define MSC_SUBCLASS_RBC                0x01

#define MSC_SUBCLASS_SFF8020I_MMC2      0x02

#define MSC_SUBCLASS_QIC157             0x03

#define MSC_SUBCLASS_UFI                0x04

#define MSC_SUBCLASS_SFF8070I           0x05

#define MSC_SUBCLASS_SCSI               0x06

/* MSC Protocol Codes */

#define MSC_PROTOCOL_CBI_INT            0x00

#define MSC_PROTOCOL_CBI_NOINT          0x01

#define MSC_PROTOCOL_BULK_ONLY          0x50

/* MSC Request Codes */

#define MSC_REQUEST_RESET               0xFF

#define MSC_REQUEST_GET_MAX_LUN         0xFE

/* MSC Bulk-only Stage */

#define MSC_BS_CBW                      0       /* Command Block Wrapper */

#define MSC_BS_DATA_OUT                 1       /* Data Out Phase */

#define MSC_BS_DATA_IN                  2       /* Data In Phase */

#define MSC_BS_DATA_IN_LAST             3       /* Data In Last Phase */

#define MSC_BS_DATA_IN_LAST_STALL       4       /* Data In Last Phase with Stall */

#define MSC_BS_CSW                      5       /* Command Status Wrapper */

#define MSC_BS_ERROR                    6       /* Error */

typedef __packed struct _MSC_CBW {

uint32_t dSignature;  //cbw标志  为USBC四个字符

uint32_t dTag;        //cbw标签,返回csw的时候需要添加

uint32_t dDataLength; //需要在数据阶段传送的字节数,低字节在前

uint8_t  bmFlags;     //cbw标志,最高位标识数据方向

uint8_t  bLUN;        //目标逻辑单元编号,仅仅使用低四位

uint8_t  bCBLength;   //cbw cb的长度,使用低五位

uint8_t  CB[16];      //选择执行的命令,由选择的子类决定使用的命令,(一般是scsi命令集)

} MSC_CBW;

typedef __packed struct _MSC_CSW {

uint32_t dSignature;  //csw标志,为字符 usbs

uint32_t dTag;    //命令状态标签,从cbw的标签中来

uint32_t dDataResidue;    //命令完成时的完成字节数

uint8_t  bStatus; //命令执行状态 00代表执行成功

} MSC_CSW;

#define MSC_CBW_Signature               0x43425355  //usbc

#define MSC_CSW_Signature               0x53425355  //usbs

/* CSW Status Definitions */

#define CSW_CMD_PASSED                  0x00

#define CSW_CMD_FAILED                  0x01

#define CSW_PHASE_ERROR                 0x02

/* SCSI Commands */

#define SCSI_TEST_UNIT_READY            0x00

#define SCSI_REQUEST_SENSE              0x03

#define SCSI_FORMAT_UNIT                0x04

#define SCSI_INQUIRY                    0x12

#define SCSI_MODE_SELECT6               0x15

#define SCSI_MODE_SENSE6                0x1A

#define SCSI_START_STOP_UNIT            0x1B

#define SCSI_MEDIA_REMOVAL              0x1E

#define SCSI_READ_FORMAT_CAPACITIES     0x23

#define SCSI_READ_CAPACITY              0x25

#define SCSI_READ10                     0x28

#define SCSI_WRITE10                    0x2A

#define SCSI_VERIFY10                   0x2F

#define SCSI_MODE_SELECT10              0x55

#define SCSI_MODE_SENSE10               0x5A

#endif

在之前的代码中会调用两个存储设备读写命令,如下

#ifndef __MEMORY_H_

#define __MEMORY_H_

#include "sst39vf32.h"

#include "k9f1g08.h"

#include "hy57v256.h"

#define NORFLASH    0

#define NANDFLASH   1

#if NORFLASH

#define MSC_MemorySize  NOR_FLASH_SIZE  //norflash大小

#define MSC_BlockSize   SECTOR_SIZE     //norflash扇区大小

#define MSC_BlockCount  SECTOR_COUNT    //norflash扇区总数

extern u8 msc_idle_time_count;

#define MSC_IDLE_TIME_LIMIT     90

#endif

#if NANDFLASH

#define MSC_MemorySize  NAND_USE_NUMOF_BLOCK*NAND_PAGE_NUM*NAND_SERECT_SIZE //nand flash大小

#define MSC_BlockSize   NAND_SERECT_SIZE            //nand flash扇区大小

#define MSC_BlockCount  NAND_USE_NUMOF_BLOCK*NAND_PAGE_NUM  //nand flash扇区数量

extern u8 msc_idle_time_count;

#define MSC_IDLE_TIME_LIMIT     90

#endif

#endif

#include "memory.h"

#if NORFLASH

u8 msc_idle_time_count; //岣дȫĿՏЊ䍊

#endif

#if NANDFLASH

u8 msc_idle_time_count;

#endif

LPC1788做U盘的时候对命令的响应的更多相关文章

  1. UltraISO做U盘启动盘教程

    用UltraISO做U盘启动盘教程 注意:制作前请先备份u盘内重要文件 vista.win7系统需要以管理员身份运行UltraISO,其他系统直接运行即可 1.打开UltraISO选择iso镜像文件 ...

  2. 【开发者portal在线开发插件系列五】命令的响应,即命令结果的上报(mid的使用)

    基础篇 [开发者portal在线开发插件系列一]profile和基本上下行消息 [开发者portal在线开发插件系列二]多条上下行消息(messageId的使用) 基础场景见上面两个帖子,这里单独介绍 ...

  3. 使用USBWriter做U盘启动盘后容量变小的解决办法

    环境windows10 , 以administrator登录: 1. 按下windows键盘上的Win+R键, 输入cmd 2. 输入DiskPart, 回车,  然后弹出新的命令行窗口 3. 在弹出 ...

  4. dell r710 服务器配置RAID5(3块硬盘做RAID5,另外再弄一块做数据冗余盘)

    本文完全转载于:http://www.jb51.net/article/53814.htm,只为做笔记使用 ①4块硬盘做成RAID5 ②3块硬盘做RAID5,一块硬盘做热备盘 这两种配置之间的区别.大 ...

  5. U盘开发之SCSI命令

    借助硬件USB协议分析仪,可以清楚的看到U盘启动时和上位机之间交互的USB协议流程,从get desciptor get congfiguration set configuration到scsi命令 ...

  6. 做u盘启动重装系统 进winPE 出现 cdboot:couldn't find ntldr 解决办法

    公司的QA本来用的ubuntu系统 觉得不是很好使 就找我重装win10系统  之前有重装过系统 就信心满满的答应了 我拿出U盘 把U盘格式化了下 去下载了个雨林木风的win10 系统(ISO文件) ...

  7. centos7下做内存盘的方法

    在找这个资料的时候,基本没几个能用的或者过时了的,或者是换了概念,做的不是需要的那种盘,只有少数文章有提到关键部分应该怎么去操作,现在还是自己总结一下 内存盘tmpfs和ramdisk的区别 这个在网 ...

  8. QT 做软件盘

    最近搞了一个组织细胞脱水机项目,当然,对于国内的项目都是仿来仿去的,我们也不例外,开启被仿机器后,第一个看到的界面就是用户登录界面,需要输入中文,作为一个程序员,我的第一反应就是我需要采用什么用的框架 ...

  9. (4)分布式下的爬虫Scrapy应该如何做-规则自动爬取及命令行下传参

    本次探讨的主题是规则爬取的实现及命令行下的自定义参数的传递,规则下的爬虫在我看来才是真正意义上的爬虫. 我们选从逻辑上来看,这种爬虫是如何工作的: 我们给定一个起点的url link ,进入页面之后提 ...

随机推荐

  1. strncasecmp与strcasecmp用法

    strcasecmp strcasecmp(忽略大小写比较字符串) 相关函数 bcmp,memcmp,strcmp,strcoll,strncmp 表头文件 #include<string.h& ...

  2. SuperSocket+unity 网络笔记

    学习SuperSocket 必须要注意的 代码是 static void Main(string[] args) { WebSocketServer appServer = new WebSocket ...

  3. UVALive 2521 Game Prediction 题解

    这个我上来把题目理解错了,我以为所有人的牌都是一样的,感觉这个题太麻烦了吧,而且题目样例过不去啊……后来发现理解错了,给出的数据是他一个人的数据,就是让我们求他一定能赢的轮数,所有的牌是固定的(1 - ...

  4. HDU 5739 Fantasia

    可以将这个图转换成森林来进行树形dp求解.看了这篇具体教学才会的:http://www.cnblogs.com/WABoss/p/5696926.html 大致思路:求解一下点双连通分量(Tarjan ...

  5. java项目中使用guava的工具包的心得

    原文:http://www.   zuidaima.com/blog/3182745969511424.htm 以前只是听过这个工具包,但是真正用的话还是在D75需求中.我要实现把前台传入的list集 ...

  6. SORT函数的使用方法(转载)

    sort函数的用法(转载出处:http://blog.sina.com.cn/s/blog_6439f26f01012xw3.html) 做ACM题的时候,排序是一种经常要用到的操作.如果每次都自己写 ...

  7. 尝试在数据库 5 中提取逻辑页 (1:1640) 失败。该逻辑页属于分配单元XXX ,而非 XXX。

    此信息表明数据库或表 已经部分损坏可以通过以下步骤尝试修复: 1. DBCC CHECKDB 重启服务器后,在没有进行任何操作的情况下,在SQL查询分析器中执行以下SQL进行数据库的修复,修复数据库存 ...

  8. zf-关于更改账号密码的问题

    一般项目的数据库里都会有一个 SYS_USER表 里面有账号密码 一般 202……70 的都是123加密后的字符串 如果碰到项目运行之后不知道登陆密码的时候 可以在数据库中 把USER_PASS 改成 ...

  9. Struts+Spring+Hibernate的Web应用执行过程

    struts1和spring有两种整合的方法  一种是action和spring bean映射:一种是将action交给spring初始化 第一种方式:访问.do的URL->tomcat接收到r ...

  10. j2ee常用包的作用

    1.antlr-2.7.7.jar 呵呵 一句话,没有此包,hibernate不会执行hql语句   2.aopalliance-1.0.jar 这个包是AOP联盟的API包,里面包含了针对面向切面的 ...