[开发工具] STM32CubeMX-18 | 使用硬件QSPI读写SPI Flash(W25Q64)

[复制链接]
4308|48
 楼主| programmable 发表于 2020-4-12 19:17 | 显示全部楼层 |阅读模式
本篇详细的记录了如何使用STM32CubeMX配置STM32L431RCT6的硬件QSPI外设与 SPI Flash 通信(W25Q64)。
 楼主| programmable 发表于 2020-4-12 19:18 | 显示全部楼层
1. 准备工作
硬件准备
开发板
首先需要准备一个开发板,这里我准备的是STM32L4的开发板(BearPi):
105935e92f8f0a22f5.png
 楼主| programmable 发表于 2020-4-12 19:19 | 显示全部楼层
SPI Flash
小熊派开发板板载一片SPI Flash,型号为 W25Q64,大小为 8 MB,最大支持 80 Mhz的操作频率。
 楼主| programmable 发表于 2020-4-12 19:19 | 显示全部楼层
软件准备
    需要安装好Keil - MDK及芯片对应的包,以便编译和下载生成的代码;
    准备一个串口调试助手,这里我使用的是Serial Port Utility;

Keil MDK和串口助手Serial Port Utility 的安装包都可以在文末关注公众号获取,回复关键字获取相应的安装包:
799785e92f94de2804.png
 楼主| programmable 发表于 2020-4-12 19:20 | 显示全部楼层
2.生成MDK工程
选择芯片型号
打开STM32CubeMX,打开MCU选择器:
329375e92f984a141e.png
 楼主| programmable 发表于 2020-4-12 19:21 | 显示全部楼层
搜索并选中芯片STM32L431RCT6:

988655e92f9994aac9.png
 楼主| programmable 发表于 2020-4-12 19:21 | 显示全部楼层
配置时钟源
    如果选择使用外部高速时钟(HSE),则需要在System Core中配置RCC;
    如果使用默认内部时钟(HSI),这一步可以略过;

这里我都使用外部时钟:
765345e92f9c2f120e.png
 楼主| programmable 发表于 2020-4-12 19:22 | 显示全部楼层
配置串口
小熊派开发板板载ST-Link并且虚拟了一个串口,原理图如下:
462705e92f9d980739.png
这里我将开关拨到AT-MCU模式,使PC的串口与USART1之间连接。
 楼主| programmable 发表于 2020-4-12 19:23 | 显示全部楼层
接下来开始配置USART1:

63935e92fa2eefc79.png
 楼主| programmable 发表于 2020-4-12 19:24 | 显示全部楼层
配置QSPI接口
首先查看小熊派开发板上 SPI Flash 的原理图:
552375e92fa486affa.png
 楼主| programmable 发表于 2020-4-12 19:24 | 显示全部楼层
其引脚连接情况如下:


SPI Flash连接引脚        对应引脚
QUADSPI_BK1_NCS        PB11
QUADSPI_BK1_CLK        PB10
QUADSPI_BK1_IO0        PB1
QUADSPI_BK1_IO1        PB0
 楼主| programmable 发表于 2020-4-12 19:25 | 显示全部楼层
接下来配置 QSPI 接口:

236095e92fa8c75cf5.png
 楼主| programmable 发表于 2020-4-12 19:25 | 显示全部楼层
配置时钟树
STM32L4的最高主频到80M,所以配置PLL,最后使HCLK = 80Mhz即可:

199795e92fabbdc0d9.png
phosphate 发表于 2020-4-12 19:28 | 显示全部楼层
生成工程设置

580355e92fb3f3a36b.png
phosphate 发表于 2020-4-12 19:28 | 显示全部楼层
代码生成设置
最后设置生成独立的初始化文件:
872535e92fb59dc742.png
phosphate 发表于 2020-4-12 19:29 | 显示全部楼层
生成代码
点击GENERATE CODE即可生成MDK-V5工程:

180955e92fb77cf76f.png
phosphate 发表于 2020-4-12 19:29 | 显示全部楼层
3. 在MDK中编写、编译、下载用户代码
  重定向printf( )函数
  参考:【STM32Cube_09】重定向printf函数到串口输出的多种方法。
      https://bbs.21ic.com/icview-2938342-1-1.html
phosphate 发表于 2020-4-12 19:30 | 显示全部楼层
4. 封装 SPI Flash(W25Q64)的命令和底层函数
MCU 通过向 SPI Flash 发送各种命令 来读写 SPI Flash内部的寄存器,所以这种裸机驱动,首先要先宏定义出需要使用的命令,然后利用 HAL 库提供的库函数,封装出三个底层函数,便于移植:
     向 SPI Flash 发送命令的函数
     向 SPI Flash 发送数据的函数
     从 SPI Flash 接收数据的函数

接下来开始编写代码~
phosphate 发表于 2020-4-12 19:31 | 显示全部楼层
宏定义操作命令

  1. #define ManufactDeviceID_CMD    0x90
  2. #define READ_STATU_REGISTER_1   0x05
  3. #define READ_STATU_REGISTER_2   0x35
  4. #define READ_DATA_CMD               0x03
  5. #define WRITE_ENABLE_CMD          0x06
  6. #define WRITE_DISABLE_CMD         0x04
  7. #define SECTOR_ERASE_CMD          0x20
  8. #define CHIP_ERASE_CMD              0xc7
  9. #define PAGE_PROGRAM_CMD        0x02
phosphate 发表于 2020-4-12 19:32 | 显示全部楼层
封装发送命令的函数(重点)

  1. /**
  2. * [url=home.php?mod=space&uid=247401]@brief[/url]        向SPI Flash发送指令
  3. * @param        instruction —— 要发送的指令
  4. * @param        address     —— 要发送的地址
  5. * @param        dummyCycles    —— 空指令周期数
  6. * @param        instructionMode —— 指令发送模式
  7. * @param        addressMode —— 地址发送模式
  8. * @param        addressSize    —— 地址大小
  9. * @param        dataMode    —— 数据发送模式
  10. * @retval        成功返回HAL_OK
  11. */
  12. HAL_StatusTypeDef QSPI_Send_Command(uint32_t instruction,
  13.                                     uint32_t address,
  14.                                     uint32_t dummyCycles,
  15.                                     uint32_t instructionMode,
  16.                                     uint32_t addressMode,
  17.                                     uint32_t addressSize,
  18.                                     uint32_t dataMode)
  19. {
  20.     QSPI_CommandTypeDef cmd;

  21.     cmd.Instruction = instruction;                     //指令
  22.     cmd.Address = address;                          //地址
  23.     cmd.DummyCycles = dummyCycles;                  //设置空指令周期数
  24.     cmd.InstructionMode = instructionMode;            //指令模式
  25.     cmd.AddressMode = addressMode;                   //地址模式
  26.     cmd.AddressSize = addressSize;                   //地址长度
  27.     cmd.DataMode = dataMode;                         //数据模式
  28.     cmd.SIOOMode = QSPI_SIOO_INST_EVERY_CMD;           //每次都发送指令
  29.     cmd.AlternateByteMode = QSPI_ALTERNATE_BYTES_NONE; //无交替字节
  30.     cmd.DdrMode = QSPI_DDR_MODE_DISABLE;               //关闭DDR模式
  31.     cmd.DdrHoldHalfCycle = QSPI_DDR_HHC_ANALOG_DELAY;

  32.     return HAL_QSPI_Command(&hqspi, &cmd, 5000);
  33. }
您需要登录后才可以回帖 登录 | 注册

本版积分规则

28

主题

394

帖子

0

粉丝
快速回复 在线客服 返回列表 返回顶部