微波EDA网,见证研发工程师的成长!
首页 > 通信和网络 > 通信网络技术文库 > 基于USB2.0的高性能移动存储设备的设计

基于USB2.0的高性能移动存储设备的设计

时间:02-05 来源:21IC 点击:
3.1 响应USB总线标准请求

  当USB移动存储设备接入主机后,USB主机控制器对设备进行总线枚举过程,首先向设备发送标准USB请求GET_DESCRIPTOR来获得最大数据包长度;接着发出SET ADDRESS请求,为设备分配地址;使用新分配的地址再次发出GET_DESCRIPTOR请求,读取设备的配置信息[4],如设备描述符、配置描述符、接口描述符、端口描述符和字符串描述符等,并选择合适的驱动程序;最后,发出SET CONFIGURATION请求配置端口属性。
USB设备通过控制端口响应USB标准请求,实现主机和设备间的通信。除控制端口外,Bulk-Only传输方式还需支持Bulk-In和Bulk-Out端口。端口初始化代码如下:

  void Usb_Init(void)

  {

  UEPNUM=0x00; UEPCONX=0x80;

  //端口0,控制端口

  UEPNUM=0x01; UEPCONX=0x86;

  //端口1,Bulk-In端口

  UEPNUM=0x02; UEPCONX=0x82;

  //端口2,Bulk-Out端口

  UEPRST=0x07; UEPRST= 0x00;//端口复位

  UEPIEN=0x07; USBIEN|=EEOFINT;

  USBADDR=FEN; //功能使能位

  }

  当控制端口配置成功后,主机会发出Inquiry、Mode_Sense、Read_Capacity等请求,提供闪盘基本信息,如扇区大小、簇大小、闪盘容量等,当请求结果正确后,便会发出Read(10)命令,进入文件系统识别阶段。

  3.2 解析SCSI命令并处理

  UFI 命令规范基于 SFF-8070i 和 SCSI-2 的规范,总共定义了19个固定12字节长度的操作命令,用于 USB主机和 USB 移动存储设备之间进行命令字CBW (Command Block Wrapper)、普通数据、状态字CSW (Command Status Wrapper)的交换。

  USB移动存储设备接收到来自于USB主机Bulk-Out端口发给其的CBW命令字后,按照SCSI的命令格式进行解析,得到其中的命令信息,如:格式化设备、查询设备信息、读写设备等,对移动存储设备执行相应的命令后,向主机Bulk-In端口返回状态字 CSW,报告命令执行情况,主机根据反馈的状态字决定是否继续发送命令字或是数据。

  3.3 闪存的操作实现

  K9K2G08U0A芯片以页为单位来进行读写,以块为单位进行擦除。K9K2G08U0A支持的操作主要有几种:读操作(Read)、页编程(Page Program)、缓存编程(Cache Program)、块擦除(Block Erase)、块复制(Copy-Back Program)、随机数据输入(Random Data Input)、随机数据输出(Random Data Output)、复位(Reset)、读设备号(ReadID)、读状态(Read Status)等操作。在进行写操作之前,必须对写单元所在块进行擦除,因此事先需要对所擦除块内容进行保存。

  如果闪存存在坏块,则在进行读、页编程、块擦除、块复制等操作时会失败,因此对坏块要进行提前检测并进行标注。芯片在出厂时,在每块的第一页和第二页的2048列的首字节做出好坏标记,如果标志位不是FFh则为坏块,基于此建立坏块列表。

  基于篇幅的原因,这里以写操作过程为例描述闪存使用方法。一般闪存的使用顺序是:块擦除-编程-多次读取-块擦除…, 对K9K2G08U0A芯片进行数据写的步骤如下:(1)将写入数据扇区地址与坏块列表对照,检查错误扇区。若存在坏区,则继续检查下一扇区;(2)开辟缓冲区域,将写入数据扇区的原始数据利用Copy-Back Program方式保存到缓冲区;(3) 利用Block Erase擦除要写入的数据扇区;(4) 利用 Page Program操作将主机传输的数据写到闪存中;(5)利用Copy-Back Program操作将缓冲区的数据写入指定扇区。
K9K2G08U0A编程器件以页为单位编程,它允许在单页编程周期中对部分页或一个甚至连续的多达2112个连续字节编程。一个页编程周期由2个阶段组成[3]:(1)串行数据加载阶段:数据被加载到数据寄存器中,以输入命令80h为标志,紧接着是5个字节的地址输入和串行数据加载;(2)非易失性的编程阶段:命令10h标志着该编程阶段的开始,将已加载的数据写入实际的存储单元,编程典型时间为200μs。之后R/ 跳低,进入闪存内部编程,最后进入读状态确认操作,命令70h表示读状态命令,I/O0表示读到的状态。页编程实现代码如下:

  void PageWrite(void ) //页编程

  {

  CheckBlock(startpage); //检测坏块列表

  if(FirstP)

  // FirstP为1,开始对一页进行写操作,否则为0

  {//片选信号有效

  F_CE=0; F_RE=1; F_ALE=0;F_CLE=1;F_WE=0;

  OutputCommand(0x80); //输入命令0x80

  F_WE=1;

  F_ALE=1;//发送地址开始

  AddrOut (addr1); //得到页地址,五个周期

  AddrOut (addr2); AddrOut (addr3); AddrOut (addr4); AddrOut (addr5);

  F_ALE=0;

  FirstP =0; }

  WriteData(BuffBlock); //将数据写入缓冲区

  FlagWrite =0;

  Do{

  F_WE=0;

  WriteFlash(); //将数据写入flash

  FlagWrite++;

  F_WE=1;

  }while(FlagWrite<64); //写满一页数据

  OutputCommand (0x10); //输入命令0x10

  while(!F_RB); //等待读信号有效

  OutputCommand (0x70); //输入命令0x70

  Wait();

  F_CE=1; //片选结束

  startpage++; //继续写下一页

  FirstP =1;

  }

Copyright © 2017-2020 微波EDA网 版权所有

网站地图

Top