原创 【连载】【ALIENTEK 战舰STM32开发板】STM32开发指南--第五十五章 USB读卡器实验

2013-4-22 21:13 3962 20 20 分类: MCU/ 嵌入式 文集: STM32学习

  

第五十五章 USB读卡器实验

    上一章我们向大家介绍了如何利用STM32的USB来做一个触控USB鼠标,本章我们将利用STM32的USB来做一个USB读卡器。本章分为如下几个部分:

55.1 USB读卡器简介

55.2 硬件设计

55.3 软件设计

55.4 下载验证


 

55.1 USB读卡器简介

ALIENTEK 战舰STM32开发板板载了一个SD卡插槽,可以用来接入SD卡,另外战舰STM32开发板板载了一个8M字节的SPI FLASH芯片,通过STM32的USB接口,我们可以实现一个简单的USB读卡器,来读写SD卡和SPI FLASH。

本章我们还是通过移植官方的USB Mass_Storage例程来实现,该例程在MDK的安装目录下可以找到(..\MDK\ARM\Examples\ST\STM32F10xUSBLib\Demos\Mass_Storage)。

USB Mass Storage类支持两个传输协议:

1)Bulk-Only 传输(BOT)

2)Control/Bulk/Interrupt传输(CBI)

Mass Storage类规范定义了两个类规定的请求:Get_Max_LUN和Mass Storage Reset,所有的Mass Storage类设备都必须支持这两个请求。

Get_Max_LUN(bmRequestType= 10100001b and bRequest= 11111110b)用来确认设备支持的逻辑单元数。Max LUN的值必须是0~15。注意:LUN是从0开始的。主机不能向不存在的LUN发送CBW,本章我们定义Max LUN的值为1,即代表2个逻辑单元。

Mass Storage Reset(bmRequestType=00100001b and bRequest= 11111111b)用来复位Mass Storage设备及其相关接口。

支持BOT传输的Mass Storage设备接口描述符要求如下:

接口类代码bInterfaceClass=08h,表示为Mass Storage设备;

接口类子代码bInterfaceSubClass=06h,表示设备支持SCSI Primary Command-2(SPC-2);

协议代码bInterfaceProtocol有3种:0x00、0x01、0x50,前两种需要使用中断传输,最后一种仅使用批量传输(BOT)。

支持BOT的设备必须支持最少3个endpoint:Control, Bulk-In和Bulk-Out。USB2.0的规范定义了控制端点0。Bulk-In端点用来从设备向主机传送数据(本章用端点1实现)。Bulk-Out端点用来从主机向设备传送数据(本章用端点2实现)。

ST官方的例程是通过USB来读写SD卡(SDIO方式)和NAND FALSH,支持2个逻辑单元,我们在官方例程的基础上,只需要修改SD驱动部分代码(改为SPI),并将对NAND FLASH的操作修改为对SPI FLASH的操作。只要这两步完成了,剩下的就比较简单了,对底层磁盘的读写,都是在mass_mal.c文件实现的,所以我们只需要修改该函数的MAL_Init、MAL_Write、MAL_Read和MAL_GetStatus等4个函数,使之与我们的SD卡和SPI FLASH对应起来即可。

本章我对SD卡和SPI FLASH的操作都是采用SPI方式,所以速度相对SDIO和FSMC控制的NAND FLASH来说,相对会慢一些。

55.2 硬件设计

本节实验功能简介:开机的时候先检测SD卡和SPI FLASH是否存在,如果存在则获取其容量,并显示在LCD上面(如果不存在,则报错)。之后开始USB配置,在配置成功之后就可以在电脑上发现两个可移动磁盘。我们用DS1来指示USB正在读写SD卡,并在液晶上显示出来,同样我们还是用DS0来指示程序正在运行。

所要用到的硬件资源如下:

1)  指示灯DS0 、DS1

2)  串口

3)  TFTLCD模块

4)  SD卡

5)  SPI FLASH

6)  USB接口

这几个部分,在之前的实例中都已经介绍过了,我们在此就不多说了。不过还是要注意一下P13的连接,要和上一章一样!

55.3 软件设计

本章,我们在第四十四章实验(实验39)的基础上修改,先打开实验39 的工程,在HARDWARE文件夹所在文件夹下新建一个USB的文件夹,然后在USB文件夹下面新建LIB和CONFIG文件夹,分别用来存放与USB核相关的代码以及配置部分代码。这两部分代码我们也不细说(详见光盘本例程源码),其中USB文件夹里面的代码同上一章的一模一样,而CONFIG文件夹里面的源码则来自MDK自带的Mass_Storage例程: X:\Keil3.80A\ARM\

Examples\ST\STM32F10xUSBLib\Demos\Mass_Storage下的source和include文件夹(X为你安装MDK的磁盘)。

然后,我们在工程文件里面新建USB和USBCFG组,分别加入USB\LIB下面的代码和USB\CONFIG下面的代码。然后把LIB和CONFIG文件夹加入头文件包含路径。

在test.c里面,我们修改main函数如下:

//设置USB 连接/断线

//enable:0,断开

//      1,允许连接        

void usb_port_set(u8 enable)

{

       RCC->APB2ENR|=1<<2;    //使能PORTA时钟           

       if(enable)_SetCNTR(_GetCNTR()&(~(1<<1)));//退出断电模式

       else

       {      

              _SetCNTR(_GetCNTR()|(1<<1));  // 断电模式

              GPIOA->CRH&=0XFFF00FFF;

              GPIOA->CRH|=0X00033000;

              PAout(12)=0;                      

       }

}

int main(void)

{                                         

       u8 offline_cnt=0;

       u8 tct=0;

       u8 USB_STA;

       u8 Divece_STA;

     Stm32_Clock_Init(9);    //系统时钟设置

       delay_init(72);                     //延时初始化

       uart_init(72,9600);       //串口1初始化      

       LCD_Init();                  //初始化液晶

       LED_Init();            //LED初始化

       KEY_Init();                  //按键初始化                

       usmart_dev.init(72);      //usmart初始化           

      POINT_COLOR=RED;  //设置字体为蓝色    

       LCD_ShowString(60,50,200,16,16,"WarShip STM32");   

       LCD_ShowString(60,70,200,16,16,"USB Card Reader TEST");

       LCD_ShowString(60,90,200,16,16,"ATOM@ALIENTEK");

       LCD_ShowString(60,110,200,16,16,"2012/9/25");    

      SPI_Flash_Init();

       if(SD_Initialize())LCD_ShowString(60,130,200,16,16,"SD Card Error!");     

//检测SD卡错误

       else //SD 卡正常

       {                                                                                                              

              LCD_ShowString(60,130,200,16,16,"SD Card Size:     MB");

          Mass_Memory_Size[0]=(long long)SD_GetSectorCount()*512;

//得到SD卡容量(字节),当容量超过4G的时候,需要用到两个u32来表示

           Mass_Block_Size[0] =512;

//因为在Init里面设置了SD卡的操作字节为512个,所以这里一定是512个字节.

           Mass_Block_Count[0]=Mass_Memory_Size[0]/Mass_Block_Size[0];

             LCD_ShowNum(164,130,Mass_Memory_Size[0]>>20,5,16);//显示SD卡容量

      }

       if(SPI_FLASH_TYPE!=W25Q64)LCD_ShowString(60,130,200,16,16,"W25Q64 Error!");       //检测SD卡错误

       else //SPI FLASH 正常

       {                                                                                                              

           Mass_Memory_Size[1]=1024*1024*6;//前6M字节

           Mass_Block_Size[1] =512;

//因为在Init里面设置了SD卡的操作字节为512个,所以这里一定是512个字节.

           Mass_Block_Count[1]=Mass_Memory_Size[1]/Mass_Block_Size[1];

              LCD_ShowString(60,150,200,16,16,"SPI FLASH Size:6144KB");  

       }

       delay_ms(1800); usb_port_set(0);        //USB先断开

       delay_ms(300); usb_port_set(1);          //USB再次连接

      LCD_ShowString(60,170,200,16,16,"USB Connecting...");//提示SD卡已经准备了     

     //USB配置

      USB_Interrupts_Config();   

      Set_USBClock();  

      USB_Init();             

       while(1)

       {    

              delay_ms(1);                         

              if(USB_STA!=USB_STATUS_REG)//状态改变了

              {                                                

                     LCD_Fill(60,190,240,190+16,WHITE);//清除显示                           

                     if(USB_STATUS_REG&0x01)//正在写               

                     {

                            LCD_ShowString(60,190,200,16,16,"USB Writing...");//USB正在写入数据                     }

                     if(USB_STATUS_REG&0x02)//正在读

                     {

                            LCD_ShowString(60,190,200,16,16,"USB Reading...");//USB正在读数据                        }                                                                           

                     if(USB_STATUS_REG&0x04)LCD_ShowString(60,210,200,16,16,"USB Write

Err ");//提示写入错误

                     else LCD_Fill(60,210,240,210+16,WHITE);//清除显示      

                     if(USB_STATUS_REG&0x08)LCD_ShowString(60,230,200,16,16,"USB Read

  Err ");//提示读出错误

                     else LCD_Fill(60,230,240,230+16,WHITE);//清除显示   

                     USB_STA=USB_STATUS_REG;//记录最后的状态

              }

              if(Divece_STA!=bDeviceState)

              {

                     if(bDeviceState==CONFIGURED)LCD_ShowString(60,170,200,16,16,"USB

Connected    ");//提示USB连接已经建立

                     else LCD_ShowString(60,170,200,16,16,"USB DisConnected ");//USB被拔出了

                     Divece_STA=bDeviceState;

              }

              tct++;

              if(tct==200)

              {

                     tct=0; LED0=!LED0;//提示系统在运行

                     if(USB_STATUS_REG&0x10)

                     {

                            offline_cnt=0;//USB连接了,则清除offline计数器

                            bDeviceState=CONFIGURED;

                     }else//没有得到轮询

                     {

                            offline_cnt++; 

                            if(offline_cnt>10)bDeviceState=UNCONNECTED;

//2s内没收到在线标记,代表USB被拔出了

                     }

                     USB_STATUS_REG=0;

              }

       };                                                                                              

此部分代码除了main函数,还有一个usb_port_set函数,usb_port_set函数我们在上一章已经介绍过了,这里就不多说。我们将SPI FLASH的最开始6M地址范围用作SPI FLASH Disk,也就是文件系统管理的范围大小,这个我们在之前的SPI FLASH也介绍过。

通过此部分代码就可以实现了我们之前在硬件设计部分描述的功能,这里我们用到了一个全局变量Usb_Status_Reg,用来标记USB的相关状态,这样我们就可以在液晶上显示当前USB的状态了。

软件设计部分就为大家介绍到这里。

55.4 下载验证

在代码编译成功之后,我们通过下载代码到战舰STM32开发板上,在USB配置成功后(假设已经插入SD卡,注意:USB数据线,要插在USB口!不是USB_232端口!),LCD显示效果如图55.4.1所示:

 


图55.4.1 USB连接成功

此时,电脑提示发现新硬件如图55.4.2所示:

 


图55.4.2 USB读卡器被电脑找到

    等USB配置成功后,DS1不亮,DS0闪烁,并且在电脑上可以看到我们的磁盘,如图55.4.3所示:

 


图55.4.3 电脑找到USB读卡器的两个盘符

我们打开设备管理器,在通用串行总线控制器里面可以发现多出了一个USB Mass Storage Device,同时看到磁盘驱动器里面多了2个磁盘,如图55.4.4所示:

 


图55.4.4 通过设备管理器查看磁盘驱动器

此时,我们就可以通过电脑读写SD卡或者SPI FLASH里面的内容了。在执行读写操作的时候,就可以看到DS1亮,并且会在液晶上显示当前的读写状态。

注意,在对SPI FLASH操作的时候,最好不要频繁的往里面写数据,否则很容易将SPI FLASH写爆!!

 

PARTNER CONTENT

文章评论0条评论)

登录后参与讨论
EE直播间
更多
我要评论
0
20
关闭 站长推荐上一条 /3 下一条