加入收藏 | 设为首页 | 会员中心 | 我要投稿 | RSSRSS-巴斯仪表网
您当前的位置:首页 > 电子发烧 > 技术文章

在单片机上实现USB移动存储

时间:2012-11-05  来源:123485.com  作者:9stone
  

 总线枚举的所有命令都是通过SETUP包发送出去的,在SL811HS中就是通过启动DATA0把命令包发送出去。如果命令要求有数据传输,那么SETUP包后接着还有IN包或者OUT包可选数据发送,此时由于前面SETUP包已经启动了DATA0,这里就必须开启DATA1,如果数据大于端点的最大数据载荷,那么就用DATA1/DATA0的方式交替来发送。主机和设备在接收到USB包时,首先就要根据包标识域(PID)进行解码,进而区分出是什么包。USB控制传输主要是SETUP包,同时还有相应的IN包或OUT包;在控制传输完成之后接着是批量传输(BULK),这就是纯粹的IN包或OUT包的传输了。这些包在主机固件里具体如何区分、实现呢?根据表2.1可以知道检验PID的值可以区分出SETUP、IN和OUT包。那么IN和OUT包到底是控制传输中的呢?还是批量传输中的呢?此时要明白,控制传输是所有USB主机或者设备开发中的必要传输,并且只能由默认端点0来完成,所以可以在固件中判断当前信息交互的端点号就可以区分出IN和OUT包到底归属于哪一种传输。如果是批量传输中的IN包或者OUT包,那么可以直接启动DATA0/DATA1来发送数据。USB枚举程序底层数据包传输设计结构见图1.7。
数据包发送的乒乓设计 
    形象点讲USB主机枚举的过程就是:首先获取设备属性,设备会返回18个数值(值中对固件有用的是最大包端点长度),然后为设备分配一个操作地址,地址范围可以根据实际情况而定,并且配置设备,最后列举设备端点,获取设备的每一个端点号(地址)。枚举的实质目的就是想获取设备的端点地址,靠它来完成数据包的收发。获取设备属性、分配地址等枚举请求命令都有标准说明(见表1.3)。
 
    SL811HS 芯片必须初始化、复位后才能对设备进行枚举操作,实践表明对SL811HS寄存器初始化的顺序和延迟时间非常关键,会影响到整个系统的稳定性和速度。经测试比较好的初始代码如下:

void S811Init(void)
{  
    SL811Write(cDATASet,0xe0); //设定SOF计数器低8位
    SL811Write(cSOFcnt,0xae);  //设定主机工作模式
    SL811Write(CtrlReg,0x5);   //开SOF
    Delay(150);               //延时       
    SL811Write(EP0Counter,0);  
    SL811Write(IntEna,0x20);   // 写中断寄存器
    SL811Write(IntStatus,INT_CLEAR);   //清楚中断
}

1.3.2 USB批量传输和海量存储类协议的实现
   USB主机系统实现目的是使用USB移动存储,涉及到大量文件数据的传输,所以应该选择USB批量传输(BULK)。批量传输用BULK端点进行命令、数据和状态的传输[4],其流程结构见图1.8。CBW是命令块封装包,CSW是命令状态封状包,都是一系列包的集合。
 
(1)CBW
CBW的长度为31字节,包含了海量存储类协议的磁盘操作命令,其结构见图1.9。
CBW的结构 

dCBWSignature:是CBW的标志,固定值为0x43425355,所有CBW的值在USB总线上传输的时候都是按照LSB顺发送的,即最先发送低位,然后发送高位。

  1. dCBWTag:由主机产生的并发送给设备,设备会将此值填入CSW的dCSWTag,以此返回给主机。
  2. dCBWDataTransferLength:主机希望在批量端点上传输数据的大小。
  3. bmCBWFlag:一个字节的位图,D7=“0”时表示主机输出数据,反之主机接受数据,D6没有用到,D5-D0保留位。
  4. bCBWLUN:接受命令的设备逻辑单元号。
  5. bCBWCBLength:表示了CBWCB的长度,也就是磁盘操作命令的长度。
  6. CBWCB:填入磁盘操作命令。

CBW是以二进制位发送的,每个包必须是精确的31个字节,不满足的要补0。
(2) CSW
CSW的长度为13字节,其结构见图1.10。
 

  1. dCSWSignature:是CSW的标志,固定值为0x53425355,CSW的值也都是按LSB顺序发送。
  2. dCSWTag:命令状态标签,该值与CBW中的dCBWTag值相同。
  3. dCSWDataResidue:该字段表示dCBWDataTransferLength字段中主机希望的数据长度与实际发送的数据长度之间的差额。
  4. bCSWStatus:表示命令执行情况,见表1.4
     

(3)UFI
    UFI是Mass Storage 类的子类,支持海量存储类的USB主机应该实现这些子类命令。UFI子类命令是基于SFF-8070I和SCSI-2的,每个命令块的长度均为12字节,UFI的各种命令及操作码如表3.5。对USB移动存储的所有操作都是经过这些命令来完成的,UFI命令封装于CBW中的CBWCB块中,靠CBW传输出去,命令结果的状态保存于CSW中,通过读CSW相关结构中的数值,就可以了解命令的最终执行情况。UFI命令是标准的12字节,命令结构见图1.11。
 
 
    其中操作码与表1.5中每一种命令相对应。Lgical Unit Number(LUN),每一个设备上可能有很多个逻辑单元共享着该设备功能特性,设备上的逻辑单元都被连续从0X0-0XFF 进行编号;Logical Block Address (LBA), LBA的值从逻辑块0连续递增到最后一个逻辑块,系统中LBA代表的就是移动存储介质的绝对扇区。注意LUN和LBA的字节发送顺序都是MSB,在8位单片机里保持正常顺序即可,CBW结构定义如下:

typedef struct _COMMAND_BLOCK_WRAPPER{
       DWORD  dCBW_Signature;
       DWORD  dCBW_Tag;
       DWORD  dCBW_DataXferLen;
       BYTE   bCBW_Flag;
       BYTE   bCBW_LUN;BYTE   bCBW_CDBLen;
       CBWCB  CBWCommand;
       BYTE   Resverd[4];
    } CBW, *PCBW;

CBWCB是具体的磁盘操作命令,大小为12字节,用联合体定义如下:

typedef union _CBWCB{   
       READ                 Ufi_Read;
       READ_CAPACITY        Ufi_ReadCapacity;
       WRITE                Ufi_Write;
       INQUIRY              Ufi_Inquiry;
       REQUEST_SENSE        Ufi_RequestSense;
       TEST_UNIT            Ufi_TestUnit;
    } CBWCB, *PCBWCB;

CSW状态返回值填充在各个字段域里面,定义如下:

typedef struct _COMMAND_STATUS_WRAPPER{
       DWORD   dCSW_Signature;
       DWORD   dCSW_Tag;
       DWORD   dCSW_DataResidue;
       BYTE    bCSW_Status;
    } CSW, *PCSW;

为了节约存储空间和提高效率再把CBW和CSW定义在一个联合体里面,如下:

typedef union _Block{
       CBW    CbwBlock;
       CSW    CswBlock;
    } BLOCK,*pBlock;

在程序中定义BLOCK  idata BlockCommand,就可以访问CBW和CSW了。

分享到:
来顶一下
返回首页
返回首页
发表评论 共有条评论
用户名: 密码:
验证码: 匿名发表
栏目导航->技术文章
  • 电子应用基础
  • 电源技术
  • 无线传输技术
  • 信号处理
  • PCB设计
  • EDA技术
  • 单片机学习
  • 电子工具设备
  • 技术文章
  • 精彩拆解欣赏
  • 推荐资讯
    使用普通运放的仪表放大器
    使用普通运放的仪表放
    3V与5V混合系统中逻辑器接口问题
    3V与5V混合系统中逻辑
    数字PID控制及其改进算法的应用
    数字PID控制及其改进
    恶劣环境下的高性价比AD信号处理数据采集系统
    恶劣环境下的高性价比
    栏目更新
    栏目热门