|**银杏科技有限公司旗下技术文档发布平台** ||||
|技术支持电话|**0379-69926675-801** |||
|技术支持邮件|Gingko@vip.163.com |||
^ 版本 ^ 日期 ^ 作者 ^ 修改内容 ^
| V1.0 | 2020-04-17 | gingko | 初次建立 |
\\
\\
\\
\\
\\
===== STM32CubeMX教程二十——读/写U盘(大容量存储器) =====
1. 新建工程:在主界面选择File-->New Project 或者直接点击ACCEE TO MCU SELECTOR
{{ :icore3:icore3_cube_20_1.png?direct | }}
2. 出现芯片型号选择,搜索自己芯片的型号,双击型号,或者点击Start Project进入配置
在搜索栏的下面,提供的各种查找方式,可以选择芯片内核、型号等等,可以帮助你查找芯片。本实验选取的芯片型号为:STM32F407IGTx。
{{ :icore3:icore3_cube_20_2.png?direct | }}
3. 配置RCC,使用外部时钟源
{{ :icore3:icore3_cube_20_3.png?direct | }}
4. 配置调试引脚
{{ :icore3:icore3_cube_20_4.png?direct | }}
5. 将LED对应的3个引脚(PI5,PI6,PI7)设置为GPIO_Output
{{ :icore3:icore3_cube_20_5.png?direct | }}
6. 引脚模式配置
{{ :icore3:icore3_cube_20_6.png?direct | }}
7. 配置UART4
{{ :icore3:icore3_cube_20_7.png?direct | }}
8. 配置USB_OTG_HS
{{ :icore3:icore3_cube_20_8.png?direct | }}
9. 配置USB_HOST
{{ :icore3:icore3_cube_20_9.png?direct | }}
10. 配置文件系统
{{ :icore3:icore3_cube_20_10.png?direct | }}
11. 时钟源设置,选择外部高速时钟源,配置为最大主频
{{ :icore3:icore3_cube_20_11.png?direct | }}
12. 工程文件的设置, 这里就是工程的各种配置 我们只用到有限几个,其他的默认即可 IDE我们使用的是 MDK5
{{ :icore3:icore3_cube_20_12.png?direct | }}
13. 点击Code Generator,进行进一步配置
{{ :icore3:icore3_cube_20_13.png?direct | }}
* **Copy all used libraries into the project folder**
* 将HAL库的所有.C和.H都复制到所建工程中
* 优点:这样如果后续需要新增其他外设又可能不再用STM32CubeMX的时候便会很方便
* 缺点:体积大,编译时间很长
* **Copy only the necessary library files**
* 只复制所需要的.C和.H(推荐)
* 优点:体积相对小,编译时间短,并且工程可复制拷贝
* 缺点:新增外设时需要重新用STM32CubeMX导入
* **Add necessary library files as reference in the toolchain project configuration file**
* 不复制文件,直接从软件包存放位置导入.C和.H
* 优点:体积小,比较节约硬盘空间
* 缺点:复制到其他电脑上或者软件包位置改变,就需要修改相对应的路径
自行选择方式即可
14. 然后点击GENERATE CODE 创建工程
{{ :icore3:icore3_cube_20_14.png?direct | }}
创建成功,打开工程。
\\
\\
\\
\\
===== 实验二十:USBH_MSC实验——读/写U盘(大容量存储器) =====
==== 一、 实验目的与意义 ====
- 了解STM32 USB HOST结构。
- 了解STM32 USB HOST特征。
- 掌握STM32 HAL库中USBH_MSC的配置方法。
- 掌握USBH_MSC 使用方法。
- 掌握Keil MDK集成开发环境使用方法。
==== 二、 实验设备及平台 ====
- iCore3 双核心板。[[https://item.taobao.com/item.htm?spm=a1z10.1-c.w4024-251734887.3.5923532fXD2RIN&id=524229438677&scene=taobao_shop|点击购买]]
- JLINK(或相同功能)仿真器。[[https://item.taobao.com/item.htm?spm=a1z10.5-c.w4002-251734908.13.20822b61MmPeNN&id=554869837940|点击购买]]
- Micro USB线缆。
- Keil MDK 开发平台。
- STM32CubeMX开发平台。
- 装有WIN XP(及更高版本)系统的计算机。
==== 三、 实验原理 ====
=== 1、USBH_MSC及USB大容量存储设备 ===
* MSC是一种计算机和移动设备之间的传输协议,它允许一个通用串行总线(USB)设备来访问主机的计算设备,使两者之间进行文件传输。
* USB大容量存储设备类(The USB mass storage device class)是一种计算机和移动设备之间的传输协议,它允许一个通用串行总线(USB)设备来访问主机的计算设备,使两者之间进行文件传输
=== 2、USB_MSC HOST ===
* USB MSC设备中的固件(firmware)或者硬件(hardware),必须要实现下面这些功能:
* 检测和响应通用的USB Request和USB总线上的事件。
* 检测和响应来自USB设备的关于信息或者动作的USB Mass Storage Request。
* 检测和响应,从USB Transfer中获得的SCSI Command。这些业界标准的命令,是用来获得状态信息,控制设备操作,向存储介质块中读取(read block)和写入(write block)数据的。
* 另外,设备如果想要向存储介质中,创建/读取/写入,文件/文件夹的话,那么就涉及到文件系统,还要实现对应的文件系统。嵌入式系统中常见的文件系统有FAT16或FAT32。
=== 3、USB MSC Device ==
* 我们所关注的U盘,就是所谓的MSC设备,大容量存储设备。
* U盘的功能,就是数据存储。而对应的数据传输,用的是USB中的Bulk Transfer。
=== 4、USB MSC相关的协议 ===
* USB MSC Bulk-Only (BBB) Transport
* 如上所述,Bulk-only是USB设备端,此处的U盘和USB Host端,即普通PC,之间信息交换的协议。Bulk Only Transport,也被简称为BOT。
* USB MSC USB Attached SCSI Protocol (UASP)
* “Attached”顾名思义,是附在某个上面的,此处即附在SCSI协议的上面的,即SCSI协议的补充部分。
* UASP规范,定义了关于如何在USB 2.0和USB 3.0中,UAS的传输标准是如何实现的,并且给出了一些范例和一些推荐的做法。
{{ :icore3:icore3_arm_hal_20_1.png?direct |}}
* 如上图,我们U盘实现的功能,主要就是数据的读写,而Device和Host之间的数据通信,主要有两种:
* CBI:主要用于Floppy设备,所以新的设备,都很少用此协议
* BOT:Bulk-Only Transport,也称BBB(Bulk/Bulk/Bulk),而对于BOT/BBB来说,对其提高USB总线利用率,提高了USB速度后,就是对应的UASP协议,故此处称UASP为BOT的增强版的协议。
=== 5、USBH_MSC实验介绍 ===
* 硬件框架图:
{{ :icore3:icore3_arm_hal_20_2.png?direct&600 |}}
* USBH_MSC实验是用STM32F407的USB接口实现iCore3作为主机对U盘(即USB大容量存储器)实现读/写操作并通过串口打印到电脑上并显示的实验。
* 实验内容:
* 通过cube MX库提供的代码来实现STM32对U盘或者读卡器等大容量USB存储设备的读写操作,本实验是向存储设备中新建一个名为test.txt的文件,并向文件中写入数据,待写入成功后,读出文件的内容,并通过终端显示出来。
==== 四、 实验程序 ====
=== 1. 主函数 ===
int main(void)
{
HAL_Init();
SystemClock_Config();
MX_GPIO_Init();
MX_UART4_Init();
MX_FATFS_Init();
MX_USB_HOST_Init();
uart4.printf("\x0c");
uart4.printf("\033[1;32;40m");
uart4.printf("\r\nHello, I am iCore3.\r\n");
while (1)
{
MX_USB_HOST_Process();
LED_RED_ON;
}
}
* 对于USB主机,需要调用一个重要函数MX_USB_HOST_Process,该函数用于实现USB主机通信的核心状态机处理,该函数必须在主函数里面,被循环调用,而且调用频率得比较快才行(越快越好),以便及时处理各种事务。注意,MX_USB_HOST_Process函数仅在U盘识别阶段,需要频繁反复调用,但是当U盘被识别后,剩下的操作(U盘读写),都可以由USB中断处理。
=== 2. 用户处理函数 ===
static void USBH_UserProcess (USBH_HandleTypeDef *phost, uint8_t id)
{
int i,j;
static FRESULT res;
unsigned char write_buffer[512];
unsigned char read_buffer[512];
unsigned int counter;
switch(id)
{
case HOST_USER_SELECT_CONFIGURATION:
break;
case HOST_USER_DISCONNECTION:
Appli_state = APPLICATION_DISCONNECT;
break;
case HOST_USER_CLASS_ACTIVE:
//f_mount
res = f_mount(&fatfs,"0:",1);
if(res != RES_OK){
USBH_UsrLog("\r\nf_mount error!");
return;
}else{
USBH_UsrLog("\r\nf_mount successful!");
}
//f_open
for(i = 0; i < 512 ; i ++)write_buffer[i] = i % 256;
res = f_open(&file,"0:/test.txt",FA_READ | FA_WRITE | FA_OPEN_ALWAYS);
if(res != RES_OK){
USBH_UsrLog("f_open error!");
return;
}else{
USBH_UsrLog("f_open successful!");
}
//f_lseek
res = f_lseek(&file,0);
if(res != RES_OK){
USBH_UsrLog("f_lseek error!");
return;
}else{
USBH_UsrLog("f_lseek successful!");
}
//f_write
res = f_write(&file,write_buffer,512,&counter);
if(res != RES_OK || counter != 512){
USBH_UsrLog("f_write error!");
return;
}else{
USBH_UsrLog("f_write successful!");
} //f_lseek
res = f_lseek(&file,0);
if(res != RES_OK){
USBH_UsrLog("f_lseek error!");
return;
}else{
USBH_UsrLog("f_lseek successful!");
}
//f_read
res = f_read(&file,read_buffer,512,&counter);
if(res != RES_OK || counter != 512){
return;
}else{
USBH_UsrLog("f_read successful!");
}
f_close(&file);
USBH_UsrLog("read data:");
for(i = 0;i < 32;i++){
for(j = 0; j < 16; j ++)
USBH_UsrLog("%02X ",read_buffer[i*16+j]);
}
memset(read_buffer,0,sizeof(read_buffer));
break;
case HOST_USER_CONNECTION:
Appli_state = APPLICATION_START;
break;
default:
break;
}
}
=== 3. 打印函数 ===
* USBH自带打印输出,但是需要向串口4打印输出,需要修改相应的打印输出参数。在打印调试输出之前需要打开USBH_DEBUG。打开方式参照iCore3_CubeMX教程二十_USBH_MSC或在usbh_conf.h中将宏定义USBH_DEBUG_LEVEL改为
#define USBH_DEBUG_LEVEL 1U
* 打开USBH_DEBUG后,需将打印信息输出到串口4,即将打印信息修改为
#if (USBH_DEBUG_LEVEL > 0U)
#define USBH_UsrLog(...) uart4.printf(__VA_ARGS__);\
uart4.printf("\r\n");
#else
#define USBH_UsrLog(...) do {} while (0)
#endif
#if (USBH_DEBUG_LEVEL > 1U)
#define USBH_ErrLog(...) do { \
uart4.printf("ERROR: ") ; \
uart4.printf(__VA_ARGS__); \
#define USBH_ErrLog(...) do {} while (0)
#endif
#if (USBH_DEBUG_LEVEL > 2U)
#define USBH_DbgLog(...) do { \
uart4.printf("DEBUG : ") ; \
uart4.printf(__VA_ARGS__); \
uart4.printf("\r\n"); \
} while (0)
#else
* 即可正常向串口4打印调试信息。
==== 五、 实验步骤 ====
- 把仿真器与iCore3的SWD调试口相连(直接相连或者通过转接器相连);
- 将跳线帽插在USB_UART;
- 把iCore3(USB_UART)通过Micro USB线与计算机相连,为iCore3供电;
- 把USB_OTG通过Micor USB线与U盘或者读卡器相连,向此存储设备写入文件;
- 打开Keil MDK 开发环境,并打开本实验工程;
- 打开PuTTY串口中断(注:PuTTY使用方法见附录);
- 烧写程序到iCore3上;
- 也可以进入Debug 模式,单步运行或设置断点验证程序逻辑。
==== 六、 实验现象 ====
* 打开串口终端可以显示操作过程,操作完成后可以通过电脑查看U盘是否操作成功,成功后U盘内新建一个test.txt文件。
{{ :icore3:icore3_arm_hal_20_3.png?direct |}}
**附录:**
**PuTTY使用方法:**
* 1、iCore3供电后,打开计算机——属性——设备管理器——端口,查看iCore3所占用的COM口;
{{ :icore3:icore3_arm_hal_20_4.png?direct |}}
* 2、打开PuTTY;
{{ :icore3:icore3_arm_hal_20_5.png?direct |}}
* 3、烧写程序验证