你的浏览器版本过低,可能导致网站不能正常访问!
为了你能正常使用网站功能,请使用这些浏览器。

【经验分享】STM32F7xx —— QSPI

[复制链接]
STMCU小助手 发布时间:2021-12-11 12:00
一、QSPI
        SPI 是 Queued SPI 的简写,是 Motorola公司推出的 SPI 接口的扩展,比 SPI 应用更加广泛。在 SPI 协议的基础上,Motorola 公司对其功能进行了增强,增加了队列传输机制,推出了队列串行外围接口协议(即 QSPI 协议),使用该接口,用户可以一次性传输包含多达16个8位或16位数据的传输队列。一旦传输启动,直到传输结束,都不需要CPU干预,极大的提高了传输效率。该协议在ColdFire系列MCU得到广泛应用。与SPI相比,QSPI的最大结构特点是以80字节的RAM代替了SPI的发送和接收寄存器。QSPI 是一种专用的通信接口,连接单、双或四(条数据线) SPI Flash 存储介质。

该接口可以在以下三种模式下工作:
① 间接模式:使用 QSPI 寄存器执行全部操作
② 状态轮询模式:周期性读取外部 Flash 状态寄存器,而且标志位置 1 时会产生中断(如擦除或烧写完成,会产生中断)
③ 内存映射模式:外部 Flash 映射到微控制器地址空间,从而系统将其视作内部存储器。

QSPI通过6根线与SPI芯片通信,下图是内部框图:

20190123173205949.png


QSPI每条命令,必须包含一个或多个阶段:指令、地址、交替字节、空指令和数据。

QSPI发送命令:等待QSPI空闲;设置命令参数。

QSPI读数据:设置数据传输长度;设置QSPI工作模式并设置地址;读取数据。

QSPI写数据:设置数据传输长度;设置QSPI工作模式并设置地址;写数据。


二、几个重要的函数
  1. HAL_StatusTypeDef     HAL_QSPI_Init     (QSPI_HandleTypeDef *hqspi); // 初始化

  2. HAL_StatusTypeDef HAL_QSPI_Command(QSPI_HandleTypeDef *hqspi, QSPI_CommandTypeDef *cmd, uint32_t Timeout); // 发送命令

  3. HAL_StatusTypeDef     HAL_QSPI_Transmit     (QSPI_HandleTypeDef *hqspi, uint8_t *pData, uint32_t Timeout); // 发送数据

  4. HAL_StatusTypeDef     HAL_QSPI_Receive      (QSPI_HandleTypeDef *hqspi, uint8_t *pData, uint32_t Timeout); // 接收数据
复制代码

三、几个重要的结构
  1. // QSPI操作句柄
  2. typedef struct
  3. {
  4.   QUADSPI_TypeDef            *Instance;        /* QSPI registers base address        */
  5.   QSPI_InitTypeDef           Init;             /* QSPI communication parameters      */
  6.   uint8_t                    *pTxBuffPtr;      /* Pointer to QSPI Tx transfer Buffer */
  7.   __IO uint16_t              TxXferSize;       /* QSPI Tx Transfer size              */
  8.   __IO uint16_t              TxXferCount;      /* QSPI Tx Transfer Counter           */
  9.   uint8_t                    *pRxBuffPtr;      /* Pointer to QSPI Rx transfer Buffer */
  10.   __IO uint16_t              RxXferSize;       /* QSPI Rx Transfer size              */
  11.   __IO uint16_t              RxXferCount;      /* QSPI Rx Transfer Counter           */
  12.   DMA_HandleTypeDef          *hdma;            /* QSPI Rx/Tx DMA Handle parameters   */
  13.   __IO HAL_LockTypeDef       Lock;             /* Locking object                     */
  14.   __IO HAL_QSPI_StateTypeDef State;            /* QSPI communication state           */
  15.   __IO uint32_t              ErrorCode;        /* QSPI Error code                    */
  16.   uint32_t                   Timeout;          /* Timeout for the QSPI memory access */
  17. }QSPI_HandleTypeDef;

  18. // Instance:QSPI基地址  ---  QUADSPI
  19. // Init:设置QSPI参数
  20. // pTxBuffPtr,TxXferSize,TxXferCount:发送缓存指针 发送数据量 剩余数据量
  21. // pRxBuffPtr,RxXferSize,RxXferCount:接收缓存指针 发送数据量 剩余数据量
  22. // hdma与DMA相关, 其他为过程变量不需要关心
复制代码
  1. // 参数配置 时钟分频系数  FIFO阈值  采样移位  FLASH大小  片选高电平时间  时钟模式  闪存ID  双闪存模式设置
  2. typedef struct
  3. {
  4.   uint32_t ClockPrescaler;     /* Specifies the prescaler factor for generating clock based on the AHB clock.
  5.                                   This parameter can be a number between 0 and 255 */

  6.   uint32_t FifoThreshold;      /* Specifies the threshold number of bytes in the FIFO (used only in indirect mode)
  7.                                   This parameter can be a value between 1 and 32 */

  8.   uint32_t SampleShifting;     /* Specifies the Sample Shift. The data is sampled 1/2 clock cycle delay later to
  9.                                   take in account external signal delays. (It should be QSPI_SAMPLE_SHIFTING_NONE in DDR mode)
  10.                                   This parameter can be a value of @ref QSPI_SampleShifting */

  11.   uint32_t FlashSize;          /* Specifies the Flash Size. FlashSize+1 is effectively the number of address bits
  12.                                   required to address the flash memory. The flash capacity can be up to 4GB
  13.                                   (addressed using 32 bits) in indirect mode, but the addressable space in
  14.                                   memory-mapped mode is limited to 256MB
  15.                                   This parameter can be a number between 0 and 31 */

  16.   uint32_t ChipSelectHighTime; /* Specifies the Chip Select High Time. ChipSelectHighTime+1 defines the minimum number
  17.                                   of clock cycles which the chip select must remain high between commands.
  18.                                   This parameter can be a value of @ref QSPI_ChipSelectHighTime */

  19.   uint32_t ClockMode;          /* Specifies the Clock Mode. It indicates the level that clock takes between commands.
  20.                                   This parameter can be a value of @ref QSPI_ClockMode */

  21.   uint32_t FlashID;            /* Specifies the Flash which will be used,
  22.                                   This parameter can be a value of @ref QSPI_Flash_Select */

  23.   uint32_t DualFlash;          /* Specifies the Dual Flash Mode State
  24.                                   This parameter can be a value of @ref QSPI_DualFlash_Mode */                                               
  25. }QSPI_InitTypeDef;
复制代码
  1. // 采样移位
  2. #define QSPI_SAMPLE_SHIFTING_NONE           ((uint32_t)0x00000000U)        /*!<No clock cycle shift to sample data*/
  3. #define QSPI_SAMPLE_SHIFTING_HALFCYCLE      ((uint32_t)QUADSPI_CR_SSHIFT) /*!<1/2 clock cycle shift to sample data*/
复制代码

四、QSPI接口设计(仅供参考)

  1. #define QSPI_CLK_ENABLE()         __HAL_RCC_QSPI_CLK_ENABLE();

  2. #define QSPI_BK1_NCS_PORT         GPIOB
  3. #define QSPI_BK1_NCS_PIN          GPIO_PIN_6
  4. #define QSPI_BK1_NCS_AF           GPIO_AF10_QUADSPI
  5. #define QSPI_BK1_NCS_CONFIG()     GPIOConfigExt(QSPI_BK1_NCS_PORT, QSPI_BK1_NCS_PIN, GPIO_MODE_AF_PP, GPIO_PULLUP, QSPI_BK1_NCS_AF)

  6. #define QSPI_BK1_CLK_PORT         GPIOB
  7. #define QSPI_BK1_CLK_PIN          GPIO_PIN_2
  8. #define QSPI_BK1_CLK_AF           GPIO_AF9_QUADSPI
  9. #define QSPI_BK1_CLK_CONFIG()     GPIOConfigExt(QSPI_BK1_CLK_PORT, QSPI_BK1_CLK_PIN, GPIO_MODE_AF_PP, GPIO_NOPULL, QSPI_BK1_CLK_AF)

  10. #define QSPI_BK1_IO0_PORT         GPIOF
  11. #define QSPI_BK1_IO0_PIN          GPIO_PIN_8
  12. #define QSPI_BK1_IO0_AF           GPIO_AF10_QUADSPI
  13. #define QSPI_BK1_IO0_CONFIG()     GPIOConfigExt(QSPI_BK1_IO0_PORT, QSPI_BK1_IO0_PIN, GPIO_MODE_AF_PP, GPIO_NOPULL, QSPI_BK1_IO0_AF)

  14. #define QSPI_BK1_IO1_PORT         GPIOF
  15. #define QSPI_BK1_IO1_PIN          GPIO_PIN_9
  16. #define QSPI_BK1_IO1_AF           GPIO_AF10_QUADSPI
  17. #define QSPI_BK1_IO1_CONFIG()     GPIOConfigExt(QSPI_BK1_IO1_PORT, QSPI_BK1_IO1_PIN, GPIO_MODE_AF_PP, GPIO_NOPULL, QSPI_BK1_IO1_AF)

  18. #define QSPI_BK1_IO2_PORT         GPIOF
  19. #define QSPI_BK1_IO2_PIN          GPIO_PIN_7
  20. #define QSPI_BK1_IO2_AF           GPIO_AF9_QUADSPI
  21. #define QSPI_BK1_IO2_CONFIG()     GPIOConfigExt(QSPI_BK1_IO2_PORT, QSPI_BK1_IO2_PIN, GPIO_MODE_AF_PP, GPIO_NOPULL, QSPI_BK1_IO2_AF)

  22. #define QSPI_BK1_IO3_PORT         GPIOF
  23. #define QSPI_BK1_IO3_PIN          GPIO_PIN_6
  24. #define QSPI_BK1_IO3_AF           GPIO_AF9_QUADSPI
  25. #define QSPI_BK1_IO3_CONFIG()     GPIOConfigExt(QSPI_BK1_IO3_PORT, QSPI_BK1_IO3_PIN, GPIO_MODE_AF_PP, GPIO_NOPULL, QSPI_BK1_IO3_AF)
复制代码
  1. // 封装几个必要的接口
  2. static QSPI_HandleTypeDef qspi_handle;

  3. static void qspi_gpio_init(void)
  4. {
  5.   QSPI_CLK_ENABLE();

  6.   QSPI_BK1_NCS_CONFIG();
  7.   QSPI_BK1_CLK_CONFIG();
  8.   QSPI_BK1_IO0_CONFIG();
  9.   QSPI_BK1_IO1_CONFIG();
  10.   QSPI_BK1_IO2_CONFIG();
  11.   QSPI_BK1_IO3_CONFIG();
  12. }

  13. static void qspi_mode_init(void)
  14. {
  15.   qspi_handle.Instance = QUADSPI;                        // QSPI
  16.   qspi_handle.Init.ClockPrescaler = 2;                   // QPSI分频比,W25Q256最大频率为104M 所以此处应该为2,QSPI频率就为216/(2+1)=72MHZ
  17.   qspi_handle.Init.FifoThreshold = 4;                    // FIFO阈值为4个字节
  18.   qspi_handle.Init.SampleShifting = QSPI_SAMPLE_SHIFTING_HALFCYCLE; // 采样移位半个周期(DDR模式下,必须设置为0)
  19.   qspi_handle.Init.FlashSize = POSITION_VAL(0X2000000) - 1; // SPI FLASH大小,W25Q256大小为32M字节
  20.   qspi_handle.Init.ChipSelectHighTime = QSPI_CS_HIGH_TIME_4_CYCLE; // 片选高电平时间为4个时钟(13.8*4=55.2ns),即手册里面的tSHSL参数
  21.   qspi_handle.Init.ClockMode = QSPI_CLOCK_MODE_0;        // 模式0
  22.   qspi_handle.Init.FlashID = QSPI_FLASH_ID_1;            // 第一片flash
  23.   qspi_handle.Init.DualFlash = QSPI_DUALFLASH_DISABLE;   // 禁止双闪存模式
  24.   HAL_QSPI_Init(&qspi_handle);      //QSPI初始化
  25. }

  26. void QSPIInit(void)
  27. {
  28.   qspi_gpio_init();
  29.   qspi_mode_init();
  30. }

  31. // QSPI发送命令
  32. // instruction:要发送的指令
  33. // address:发送到的目的地址
  34. // dummyCycles:空指令周期数
  35. // instructionMode:指令模式;QSPI_INSTRUCTION_NONE,QSPI_INSTRUCTION_1_LINE,QSPI_INSTRUCTION_2_LINE,QSPI_INSTRUCTION_4_LINE
  36. // addressMode:地址模式; QSPI_ADDRESS_NONE,QSPI_ADDRESS_1_LINE,QSPI_ADDRESS_2_LINE,QSPI_ADDRESS_4_LINE
  37. // addressSize:地址长度;QSPI_ADDRESS_8_BITS,QSPI_ADDRESS_16_BITS,QSPI_ADDRESS_24_BITS,QSPI_ADDRESS_32_BITS
  38. // dataMode:数据模式; QSPI_DATA_NONE,QSPI_DATA_1_LINE,QSPI_DATA_2_LINE,QSPI_DATA_4_LINE
  39. void QSPISendCMD(uint32_t instruction, uint32_t address, uint32_t dummyCycles, uint32_t instructionMode, uint32_t addressMode, uint32_t addressSize, uint32_t dataMode)
  40. {
  41.   QSPI_CommandTypeDef Cmdhandler;

  42.   Cmdhandler.Instruction = instruction;           // 指令
  43.   Cmdhandler.Address = address;                   // 地址
  44.   Cmdhandler.DummyCycles = dummyCycles;           // 设置空指令周期数
  45.   Cmdhandler.InstructionMode = instructionMode;   // 指令模式
  46.   Cmdhandler.AddressMode = addressMode;           // 地址模式
  47.   Cmdhandler.AddressSize = addressSize;           // 地址长度
  48.   Cmdhandler.DataMode = dataMode;                 // 数据模式
  49.   Cmdhandler.SIOOMode = QSPI_SIOO_INST_EVERY_CMD; // 每次都发送指令
  50.   Cmdhandler.AlternateByteMode = QSPI_ALTERNATE_BYTES_NONE; // 无交替字节
  51.   Cmdhandler.DdrMode = QSPI_DDR_MODE_DISABLE;           // 关闭DDR模式
  52.   Cmdhandler.DdrHoldHalfCycle = QSPI_DDR_HHC_ANALOG_DELAY;

  53.   HAL_QSPI_Command(&qspi_handle, &Cmdhandler, 5000);
  54. }

  55. // 接收指定长度数据
  56. uint8_t QSPIReceive(uint8_t *buffer, uint32_t length)
  57. {
  58.   qspi_handle.Instance->DLR = length - 1;
  59.         
  60.   if(HAL_QSPI_Receive(&qspi_handle, buffer, 5000) == HAL_OK)
  61.   {
  62.     return 0;  //接收数据
  63.   }
  64.   else
  65.   {
  66.     return 1;
  67.   }
  68. }

  69. // 发送指定长度数据
  70. uint8_t QSPITransmit(uint8_t *buffer, uint32_t length)
  71. {
  72.   qspi_handle.Instance->DLR = length - 1;
  73.   if(HAL_QSPI_Transmit(&qspi_handle, buffer, 5000) == HAL_OK)
  74.   {
  75.     return 0;  //发送数据
  76.   }
  77.   else
  78.   {
  79.     return 1;
  80.   }
  81. }
复制代码

五、QSPI驱动W25Q256
W25Q256:32M  512块,每块16个扇区,每个扇区4K。具体指令看手册。封装了如下接口(初始化读写擦除等接口是以后移植文件系统的基础):

  1. #define W25QXX_WRITE_ENABLE      0x06
  2. #define W25QXX_WRITE_DISABLE     0x04
  3. #define W25QXX_READ_STATUS_REG1  0x05
  4. #define W25QXX_READ_STATUS_REG2  0x35
  5. #define W25QXX_READ_STATUS_REG3  0x15
  6. #define W25QXX_WRITE_STATUS_REG1 0x01
  7. #define W25QXX_WRITE_STATUS_REG2 0x31
  8. #define W25QXX_WRITE_STATUS_REG3 0x11
  9. #define W25QXX_READ_DATA         0x03
  10. #define W25QXX_FAST_READ_DATA    0x0B
  11. #define W25QXX_FAST_READ_DUAL    0x3B
  12. #define W25QXX_PAGE_PROGRAM      0x02
  13. #define W25QXX_BLOCK_ERASE       0xD8
  14. #define W25QXX_SECTOR_ERASE      0x20
  15. #define W25QXX_CHIP_ERASE        0xC7
  16. #define W25QXX_DEVICEID          0xAB
  17. #define W25QXX_MANUFACT_DEVICEID 0x90
  18. #define W25QXX_JEDEC_DEVICEID    0x9F
  19. #define W25QXX_EABLE_4BYTE_ADDR  0xB7
  20. #define W25QXX_EXIT_4BYTE_ADDR   0xE9
  21. #define W25QXX_SET_READ_PARAM    0xC0
  22. #define W25QXX_ENTER_QPIMODE     0x38
  23. #define W25QXX_EXIT_QPIMODE      0xFF

  24. uint8_t w25qxx_qpi_mode = 0; // QSPI模式标志:0,SPI模式;1,QPI模式.

  25. // 读取W25QXX的状态寄存器,W25QXX一共有3个状态寄存器
  26. // 状态寄存器1:
  27. // BIT7  6   5   4   3   2   1   0
  28. // SPR   RV  TB BP2 BP1 BP0 WEL BUSY
  29. // SPR:默认0,状态寄存器保护位,配合WP使用
  30. // TB,BP2,BP1,BP0:FLASH区域写保护设置
  31. // WEL:写使能锁定
  32. // BUSY:忙标记位(1,忙;0,空闲)
  33. // 默认:0x00
  34. // 状态寄存器2:
  35. // BIT7  6   5   4   3   2   1   0
  36. // SUS   CMP LB3 LB2 LB1 (R) QE  SRP1
  37. // 状态寄存器3:
  38. // BIT7      6    5    4   3   2   1   0
  39. // HOLD/RST  DRV1 DRV0 (R) (R) WPS ADP ADS
  40. // reg:状态寄存器号,范:1~3
  41. // 返回值:状态寄存器值
  42. static uint8_t w25qxx_read_status(uint8_t reg)
  43. {
  44.   uint8_t value = 0, command = 0;

  45.   switch(reg)
  46.   {
  47.   case 1:
  48.     command = W25QXX_READ_STATUS_REG1;  // 读状态寄存器1指令
  49.     break;

  50.   case 2:
  51.     command = W25QXX_READ_STATUS_REG2;  // 读状态寄存器2指令
  52.     break;

  53.   case 3:
  54.     command = W25QXX_READ_STATUS_REG3;  // 读状态寄存器3指令
  55.     break;

  56.   default:
  57.     command = W25QXX_READ_STATUS_REG1;
  58.     break;
  59.   }

  60.   if(w25qxx_qpi_mode)
  61.   {
  62.     QSPISendCMD(command, 0, 0, QSPI_INSTRUCTION_4_LINES, QSPI_ADDRESS_NONE, QSPI_ADDRESS_8_BITS, QSPI_DATA_4_LINES);  //QPI,写command指令,地址为0,4线传数据_8位地址_无地址_4线传输指令,无空周期,1个字节数据
  63.   }
  64.   else
  65.   {
  66.     QSPISendCMD(command, 0, 0, QSPI_INSTRUCTION_1_LINE, QSPI_ADDRESS_NONE, QSPI_ADDRESS_8_BITS, QSPI_DATA_1_LINE);  //SPI,写command指令,地址为0,单线传数据_8位地址_无地址_单线传输指令,无空周期,1个字节数据
  67.   }

  68.   QSPIReceive(&value, 1);

  69.   return value;
  70. }

  71. // 写状态寄存器
  72. static void w25qxx_write_status(uint8_t reg, uint8_t status)
  73. {
  74.   uint8_t command = 0;

  75.   switch(reg)
  76.   {
  77.   case 1:
  78.     command = W25QXX_WRITE_STATUS_REG1;  //写状态寄存器1指令
  79.     break;
  80.   case 2:
  81.     command = W25QXX_WRITE_STATUS_REG2;  //写状态寄存器2指令
  82.     break;
  83.   case 3:
  84.     command = W25QXX_WRITE_STATUS_REG3;  //写状态寄存器3指令
  85.     break;
  86.   default:
  87.     command = W25QXX_WRITE_STATUS_REG1;
  88.     break;
  89.   }
  90.   if(w25qxx_qpi_mode)
  91.   {
  92.     QSPISendCMD(command, 0, 0, QSPI_INSTRUCTION_4_LINES, QSPI_ADDRESS_NONE, QSPI_ADDRESS_8_BITS, QSPI_DATA_4_LINES); // QPI,写command指令,地址为0,4线传数据_8位地址_无地址_4线传输指令,无空周期,1个字节数据
  93.   }
  94.   else
  95.   {
  96.     QSPISendCMD(command, 0, 0, QSPI_INSTRUCTION_1_LINE, QSPI_ADDRESS_NONE, QSPI_ADDRESS_8_BITS, QSPI_DATA_1_LINE); // SPI,写command指令,地址为0,单线传数据_8位地址_无地址_单线传输指令,无空周期,1个字节数据
  97.   }
  98.   QSPITransmit(&status, sizeof(status));
  99. }

  100. // 写使能  将S1寄存器的WEL置位
  101. static void w25qxx_write_enable(void)
  102. {
  103.   if(w25qxx_qpi_mode)
  104.   {
  105.     QSPISendCMD(W25QXX_WRITE_ENABLE, 0, 0, QSPI_INSTRUCTION_4_LINES, QSPI_ADDRESS_NONE, QSPI_ADDRESS_8_BITS, QSPI_DATA_NONE); // QPI,写使能指令,地址为0,无数据_8位地址_无地址_4线传输指令,无空周期,0个字节数据
  106.   }
  107.   else
  108.   {
  109.     QSPISendCMD(W25QXX_WRITE_ENABLE, 0, 0, QSPI_INSTRUCTION_1_LINE, QSPI_ADDRESS_NONE, QSPI_ADDRESS_8_BITS, QSPI_DATA_NONE); // SPI,写使能指令,地址为0,无数据_8位地址_无地址_单线传输指令,无空周期,0个字节数据
  110.   }
  111. }

  112. // 写失能  将WEL清零
  113. void W25QXX_Write_Disable(void)
  114. {
  115.   if(w25qxx_qpi_mode)
  116.   {
  117.     QSPISendCMD(W25QXX_WRITE_DISABLE, 0, 0, QSPI_INSTRUCTION_4_LINES, QSPI_ADDRESS_NONE, QSPI_ADDRESS_8_BITS, QSPI_DATA_NONE); // QPI,写禁止指令,地址为0,无数据_8位地址_无地址_4线传输指令,无空周期,0个字节数据
  118.   }
  119.   else
  120.   {
  121.     QSPISendCMD(W25QXX_WRITE_DISABLE, 0, 0, QSPI_INSTRUCTION_1_LINE, QSPI_ADDRESS_NONE, QSPI_ADDRESS_8_BITS, QSPI_DATA_NONE); // SPI,写禁止指令,地址为0,无数据_8位地址_无地址_单线传输指令,无空周期,0个字节数据
  122.   }
  123. }

  124. // 等待空闲
  125. void w25qxx_wait_busy(void)
  126. {
  127.   while((w25qxx_read_status(1) & 0x01) == 0x01); // 等待BUSY位清空
  128. }

  129. // W25QXX进入QSPI模式
  130. static void w25qxx_qspi_init(void)
  131. {
  132.   uint8_t reg2;
  133.   reg2 = w25qxx_read_status(2); // 先读出状态寄存器2的原始值
  134.   if((reg2 & 0X02) == 0)  // QE位未使能
  135.   {
  136.     w25qxx_write_enable(); // 写使能
  137.     reg2 |= 1 << 1; // 使能QE位
  138.     w25qxx_write_status(2, reg2); // 写状态寄存器2
  139.   }

  140.   QSPISendCMD(W25QXX_ENTER_QPIMODE, 0, 0, QSPI_INSTRUCTION_1_LINE, QSPI_ADDRESS_NONE, QSPI_ADDRESS_8_BITS, QSPI_DATA_NONE); // 写command指令,地址为0,无数据_8位地址_无地址_单线传输指令,无空周期,0个字节数据

  141.   w25qxx_qpi_mode = 1; // 标记QSPI模式
  142. }

  143. // 0XEF13,表示芯片型号为W25Q80
  144. // 0XEF14,表示芯片型号为W25Q16
  145. // 0XEF15,表示芯片型号为W25Q32
  146. // 0XEF16,表示芯片型号为W25Q64
  147. // 0XEF17,表示芯片型号为W25Q128
  148. // 0XEF18,表示芯片型号为W25Q256
  149. static void w25qxx_id_get(void)
  150. {
  151.   uint8_t temp[2];
  152.   uint16_t device_id;

  153.   if(w25qxx_qpi_mode)
  154.   {
  155.     QSPISendCMD(W25QXX_MANUFACT_DEVICEID, 0, 0, QSPI_INSTRUCTION_4_LINES, QSPI_ADDRESS_4_LINES, QSPI_ADDRESS_24_BITS, QSPI_DATA_4_LINES); // QPI,读id,地址为0,4线传输数据_24位地址_4线传输地址_4线传输指令,无空周期,2个字节数据
  156.   }
  157.   else
  158.   {
  159.     QSPISendCMD(W25QXX_MANUFACT_DEVICEID, 0, 0, QSPI_INSTRUCTION_1_LINE, QSPI_ADDRESS_1_LINE, QSPI_ADDRESS_24_BITS, QSPI_DATA_1_LINE); // SPI,读id,地址为0,单线传输数据_24位地址_单线传输地址_单线传输指令,无空周期,2个字节数据
  160.   }

  161.   QSPIReceive(temp, 2);

  162.   device_id = (temp[0] << 8) | temp[1];

  163.   printf("QSPI Flash Device ID: %X\r\n", device_id);
  164. }

  165. void W25QXXInit(void)
  166. {
  167.   uint8_t temp;

  168.   QSPIInit();
  169.   w25qxx_qspi_init(); // 使能QSPI模式
  170.   w25qxx_id_get();    // 读取FLASH ID.

  171.   temp = w25qxx_read_status(3); // 读取状态寄存器3,判断地址模式
  172.   if((temp & 0X01) == 0) // 如果不是4字节地址模式,则进入4字节地址模式
  173.   {
  174.     w25qxx_write_enable(); // 写使能
  175.     QSPISendCMD(W25QXX_EABLE_4BYTE_ADDR, 0, 0, QSPI_INSTRUCTION_4_LINES, QSPI_ADDRESS_NONE, QSPI_ADDRESS_8_BITS, QSPI_DATA_NONE); // QPI,使能4字节地址指令,地址为0,无数据_8位地址_无地址_4线传输指令,无空周期,0个字节数据
  176.   }
  177.   w25qxx_write_enable(); // 写使能
  178.   QSPISendCMD(W25QXX_SET_READ_PARAM, 0, 0, QSPI_INSTRUCTION_4_LINES, QSPI_ADDRESS_NONE, QSPI_ADDRESS_8_BITS, QSPI_DATA_4_LINES); // QPI,设置读参数指令,地址为0,4线传数据_8位地址_无地址_4线传输指令,无空周期,1个字节数据
  179.   temp = 3 << 4;      // 设置P4&P5=11,8个dummy clocks,104M
  180.   QSPITransmit(&temp, 1);
  181. }

  182. // 擦除一块4096字节 最少需要150ms
  183. void W25QXXSectorErase(uint32_t addr)
  184. {
  185.   //addr /= 4096;
  186.   addr *= 4096;

  187.   w25qxx_write_enable();
  188.   w25qxx_wait_busy();
  189.   QSPISendCMD(W25QXX_SECTOR_ERASE, addr, 0, QSPI_INSTRUCTION_4_LINES, QSPI_ADDRESS_4_LINES, QSPI_ADDRESS_32_BITS, QSPI_DATA_NONE); // QPI,写扇区擦除指令,地址为0,无数据_32位地址_4线传输地址_4线传输指令,无空周期,0个字节数据
  190.   w25qxx_wait_busy();
  191. }


  192. // 擦除整个芯片
  193. void W25QXXChipErase(void)
  194. {
  195.   w25qxx_write_enable();          //SET WEL
  196.   w25qxx_wait_busy();
  197.   QSPISendCMD(W25QXX_CHIP_ERASE, 0, 0, QSPI_INSTRUCTION_4_LINES, QSPI_ADDRESS_NONE, QSPI_ADDRESS_8_BITS, QSPI_DATA_NONE); // QPI,写全片擦除指令,地址为0,无数据_8位地址_无地址_4线传输指令,无空周期,0个字节数据
  198.   w25qxx_wait_busy();           //等待芯片擦除结束
  199. }

  200. // 写最多256字节
  201. void W25QXXWritePage(uint32_t addr, uint8_t *buffer, uint16_t length)
  202. {
  203.   w25qxx_write_enable();          //写使能
  204.   QSPISendCMD(W25QXX_PAGE_PROGRAM, addr, 0, QSPI_INSTRUCTION_4_LINES, QSPI_ADDRESS_4_LINES, QSPI_ADDRESS_32_BITS, QSPI_DATA_4_LINES); // QPI,页写指令,地址为WriteAddr,4线传输数据_32位地址_4线传输地址_4线传输指令,无空周期,NumByteToWrite个数据
  205.   QSPITransmit(buffer, length);
  206.   w25qxx_wait_busy();            //等待写入结束
  207. }

  208. // 写最多65536字节 无擦除动作
  209. void W25QXXWriteExt(uint32_t addr, uint8_t *buffer, uint16_t length)
  210. {
  211.   uint16_t pageremain = 256 - addr % 256; // 单页剩余的字节数

  212.   if(addr + length - 1 > (32 * 1024 * 1024))
  213.   {
  214.     return;
  215.   }

  216.   if(length <= pageremain)
  217.   {
  218.     pageremain = length;  // 不大于256个字节
  219.   }
  220.   while(1)
  221.   {
  222.     // 分页写入
  223.     W25QXXWritePage(addr, buffer, pageremain);

  224.     if(length == pageremain)
  225.     {
  226.       break;  //写入结束了
  227.     }
  228.     else
  229.     {
  230.       buffer += pageremain;
  231.       addr += pageremain;

  232.       length -= pageremain; // 减去已经写入了的字节数
  233.       if(length > 256)
  234.       {
  235.         pageremain = 256;  // 一次可以写入256个字节
  236.       }
  237.       else
  238.       {
  239.         pageremain = length;  // 不够256个字节了
  240.       }
  241.     }
  242.   }
  243. }

  244. // 写最多65536字节 带擦除动作
  245. static uint8_t W25QXX_BUFFER[4096];
  246. void W25QXXWrite(uint32_t addr, uint8_t* buffer, uint16_t length)
  247. {
  248.   uint32_t secpos;
  249.   uint16_t secoff, secremain, i;
  250.   uint8_t * w25q_buf;

  251.   w25q_buf = W25QXX_BUFFER;

  252.   secpos = addr / 4096; // 扇区地址
  253.   secoff = addr % 4096; // 在扇区内的偏移
  254.   secremain = 4096 - secoff; // 扇区剩余空间大小

  255.   if(length <= secremain)
  256.   {
  257.     secremain = length;  // 不大于4096个字节
  258.   }
  259.   while(1)
  260.   {
  261.           WatchdogFeed();
  262.     W25QXXRead(secpos * 4096, w25q_buf, 4096); // 读出整个扇区的内容
  263.     for(i = 0; i < secremain; ++i)
  264.     {
  265.       if(w25q_buf[secoff + i] != 0XFF)
  266.       {
  267.         break;  // 需要擦除
  268.       }
  269.     }
  270.     if(i < secremain) // 需要擦除
  271.     {
  272.       W25QXXSectorErase(secpos); // 擦除这个扇区
  273.       for(i = 0; i < secremain; ++i)
  274.       {
  275.         w25q_buf[i + secoff] = buffer<i>;</i>
  276.       }
  277.       W25QXXWriteExt(secpos * 4096, w25q_buf, 4096); // 写入整个扇区

  278.     }
  279.     else
  280.     {
  281.       W25QXXWriteExt(addr, buffer, secremain);  // 写已经擦除了的,直接写入扇区剩余区间.
  282.     }
  283.     if(length == secremain)
  284.     {
  285.       break;
  286.     }
  287.     else
  288.     {
  289.       secpos++;
  290.       secoff = 0;

  291.       buffer += secremain;
  292.       addr += secremain;
  293.       length -= secremain;
  294.       if(length > 4096)
  295.       {
  296.         secremain = 4096;
  297.       }
  298.       else
  299.       {
  300.         secremain = length;
  301.       }
  302.     }
  303.   };
  304. }


  305. // 读取SPI FLASH,仅支持QPI模式  在指定地址开始读取指定长度的数据 65536
  306. void W25QXXRead(uint32_t addr, uint8_t *buffer, uint16_t length)
  307. {
  308.   QSPISendCMD(W25QXX_FAST_READ_DATA, addr, 8, QSPI_INSTRUCTION_4_LINES, QSPI_ADDRESS_4_LINES, QSPI_ADDRESS_32_BITS, QSPI_DATA_4_LINES); // QPI,快速读数据,地址为ReadAddr,4线传输数据_32位地址_4线传输地址_4线传输指令,8空周期,length个数据
  309.   QSPIReceive(buffer, length);
  310. }
复制代码

也可以使用SPI操作W25QXX。



收藏 评论0 发布时间:2021-12-11 12:00

举报

0个回答
关于
我们是谁
投资者关系
意法半导体可持续发展举措
创新与技术
意法半导体官网
联系我们
联系ST分支机构
寻找销售人员和分销渠道
社区
媒体中心
活动与培训
隐私策略
隐私策略
Cookies管理
行使您的权利
官方最新发布
STM32N6 AI生态系统
STM32MCU,MPU高性能GUI
ST ACEPACK电源模块
意法半导体生物传感器
STM32Cube扩展软件包
关注我们
st-img 微信公众号
st-img 手机版