shanyuxiang 发表于 2024-7-1 21:19

玩转APM32的DMA-用SPI的DMA实现NorFlash的读写

本帖最后由 shanyuxiang 于 2024-7-2 20:57 编辑

#申请原创# @21小跑堂
玩转APM32的DMA-用SPI的DMA实现NorFlash的读写
一、前言
1.1 关于NorFlash
NorFlash是一种常见的非易失性存储介质,经常用来保存程序固件或一些数据,其中以SOP-8封装的25Qxx系列为代表。常规的方式是一个字节一个字节的读写,对于SPI这种接口来说,这种方法不太划算,当需要多个字节读写的操作时可以用DMA去接管,从而省去大量CPU的时间。这里就以W25Q64为例,介绍一下如何用SPI的DMA实现flash的读写,对于其他容量或其他品牌的NOR FLASH芯片也是类似的方法。
完整的SPI要用到四个信号,这里连接到SPI2,CS使用任意一个IO口即可,Flash与MCU的连接关系如下:
CS--   PB12
CLK --PB13(SPI2_SCK)
DO--PB14(SPI2_MISO)
DI   --PB15(SPI2_MOSI)





Nor Flash的指令挺多,常用的其实只有几条,这里要用DMA实现的指令就只有"读数据"和"页编程",其他指令还是按照传统的方式实现。
Read Data      03h
Page Program 02h



1.2 关于SPI和DMA
通过查阅用户手册“APM32E103xCxE用户手册”的DMA章节可以知道SPI2的对应通道,SPI2接收对应DMA1_Channel4,SPI2发送对应DMA1_Channel5。
Flash的读数据可以用DMA1_Channel4来传输,Flash的页编程可以用DMA1_Channel5来传输。


二、SPI的DMA读写
2.1 SPI的初始化
这里参考SDK把SPI初始化为主机、全双工模式即可,比较简单,代码如下:
//SPI2 初始化
void spi_flash_init(void)
{
    GPIO_Config_T gpioConfig;
    SPI_Config_T spiConfig;

    /* Enable related Clock */
    RCM_EnableAPB1PeriphClock(RCM_APB1_PERIPH_SPI2);
    RCM_EnableAPB2PeriphClock(RCM_APB2_PERIPH_GPIOB);

    /* config MISO*/
    gpioConfig.pin =GPIO_PIN_14 ;
    gpioConfig.mode = GPIO_MODE_AF_PP;
    gpioConfig.speed = GPIO_SPEED_50MHz;
    GPIO_Config(GPIOB, &gpioConfig);

    /** config SCK,MOSI*/
    gpioConfig.pin = GPIO_PIN_13 | GPIO_PIN_15;
    gpioConfig.mode = GPIO_MODE_AF_PP;
    gpioConfig.speed = GPIO_SPEED_50MHz;
    GPIO_Config(GPIOB, &gpioConfig);

    /** config CS */
    gpioConfig.pin = GPIO_PIN_12;
    gpioConfig.mode = GPIO_MODE_OUT_PP;
    gpioConfig.speed = GPIO_SPEED_50MHz;
    GPIO_Config(GPIOB, &gpioConfig);

    GPIO_SetBit(GPIOB, GPIO_PIN_12);
    GPIO_SetBit(GPIOB, GPIO_PIN_13);
    GPIO_SetBit(GPIOB, GPIO_PIN_14);
    GPIO_SetBit(GPIOB, GPIO_PIN_15);

    SPI_ConfigStructInit(&spiConfig);

    spiConfig.length = SPI_DATA_LENGTH_8B;

    spiConfig.baudrateDiv = SPI_BAUDRATE_DIV_64;
    /*2 line full duplex*/
    spiConfig.direction = SPI_DIRECTION_2LINES_FULLDUPLEX;
    /*LSB first*/
    spiConfig.firstBit = SPI_FIRSTBIT_MSB;
    /*Slave mode*/
    spiConfig.mode = SPI_MODE_MASTER;
    /*Polarity is low*/
    spiConfig.polarity = SPI_CLKPOL_HIGH;
    /*Software select slave enable*/
    spiConfig.nss = SPI_NSS_SOFT;
    /*Phase is 1 edge*/
    spiConfig.phase = SPI_CLKPHA_2EDGE;

    spiConfig.crcPolynomial = 7;

    /*SPI config*/
    SPI_Config(SPI2, &spiConfig);

    SPI_ConfigDataSize(SPI2, SPI_DATA_LENGTH_8B);

    SPI_Enable(SPI2);
}

2.2 DMA的初始化
要实现读写需要配置两个DMA通道,两个DMA通道的配置有些不同,主要是注意传输方向。
DMA1_Channel5 作为发送,方向是从内存到外设;
DMA1_Channel4作为接受,方向是从外设到内存。
外设的地址是SPI的数据寄存地址,通过查看手册可知是 SPI2_BASE + 0x0C。
内存地址和传输长度可以先不配,等后面真正要传输时再配置,两个DMA通道都先不要使能。
DMA_Config_T    DMA_ConfigStruct;
void spi_fullduplex_dma_init(void)
{
    RCM_EnableAHBPeriphClock(RCM_AHB_PERIPH_DMA1);

    DMA_ConfigStruct.peripheralBaseAddr = (uint32_t)(SPI2_BASE + 0x0C) ;
    DMA_ConfigStruct.peripheralInc      = DMA_PERIPHERAL_INC_DISABLE;
    DMA_ConfigStruct.memoryInc          = DMA_MEMORY_INC_ENABLE;
    DMA_ConfigStruct.peripheralDataSize = DMA_PERIPHERAL_DATA_SIZE_BYTE;
    DMA_ConfigStruct.memoryDataSize   = DMA_MEMORY_DATA_SIZE_BYTE;
    DMA_ConfigStruct.loopMode = DMA_MODE_NORMAL;
    DMA_ConfigStruct.priority = DMA_PRIORITY_HIGH;
    DMA_ConfigStruct.M2M      = DMA_M2MEN_DISABLE;

    //SPI2_TX DMA_CH5
    DMA_Reset(DMA1_Channel5);

    DMA_ConfigStruct.memoryBaseAddr = (uint32_t) AddressBuffer;
    DMA_ConfigStruct.dir                   = DMA_DIR_PERIPHERAL_DST;
    DMA_ConfigStruct.bufferSize         = 0;
    DMA_Config(DMA1_Channel5, &DMA_ConfigStruct);
    SPI_I2S_EnableDMA(SPI2, SPI_I2S_DMA_REQ_TX);

    //SPI2_RX DMA_CH4
    DMA_Reset(DMA1_Channel4);

    DMA_ConfigStruct.memoryBaseAddr = (uint32_t) AddressBuffer;
    DMA_ConfigStruct.dir                   = DMA_DIR_PERIPHERAL_SRC;
    DMA_ConfigStruct.bufferSize         = 0;
    DMA_Config(DMA1_Channel4, &DMA_ConfigStruct);
    SPI_I2S_EnableDMA(SPI2, SPI_I2S_DMA_REQ_RX);

    //DMA_Enable(DMA1_Channel4);
    //DMA_Enable(DMA1_Channel5;
}
2.3 SPI的DMA发送
要发送多个数据时,设置一下DMA1_Channel5内存地址、发送长度、开启的内存地址自增,然后使能DMA的传输即可;DMA1_Channel4类似,只是地址不需要自增,最后等待传输完成就实现了SPI的DMA发送。
如果发送的数据较多的话,等待的时间可能较长,在等待过程中CPU可以去做别的事。
void spi_dam_transmit_buffer(unsigned char *buffer, unsigned int length)
{

    DMA1_Channel4->CHCFG_B.MIMODE = DISABLE;
    DMA1_Channel4->CHMADDR = (uint32_t)AddressBuffer;
    DMA1_Channel4->CHNDATA = length;

    DMA1_Channel5->CHCFG_B.MIMODE = ENABLE ;
    DMA1_Channel5->CHMADDR = (uint32_t)buffer;
    DMA1_Channel5->CHNDATA = length;

    DMA1_Channel4->CHCFG_B.CHEN = ENABLE;
    DMA1_Channel5->CHCFG_B.CHEN = ENABLE;

    while (DMA_ReadStatusFlag(DMA1_FLAG_TC5) == RESET);
    DMA_ClearStatusFlag(DMA1_FLAG_TC5);

    while (DMA_ReadStatusFlag(DMA1_FLAG_TC4) == RESET);
    DMA_ClearStatusFlag(DMA1_FLAG_TC4);

    DMA1_Channel5->CHCFG_B.CHEN = DISABLE;
    DMA1_Channel4->CHCFG_B.CHEN = DISABLE;
}

2.4 SPI的DMA接收
要读取多个数据时,除了要设置DMA1_Channel4,也要设置DMA1_Channel5,因为MCU作为主机要给flash芯片提供时钟,所以读取的同时也要开启发送。
这里的发送也只要发送任意数据就可以,因此发送通道的内存地址不需要递增,地址给一个自定义的小数组即可。
关于接收部分,和前面类似,设置一下内存地址、发送长度、开启内存地址自增,使能DMA1_Channel4的传输,等待两个通道都传输完成就完成了读取。
void spi_dam_receive_buffer(unsigned char *buffer, unsigned int length)
{

    DMA1_Channel5->CHCFG_B.MIMODE = DISABLE;
    DMA1_Channel5->CHMADDR = (uint32_t)AddressBuffer;
    DMA1_Channel5->CHNDATA = length;

    DMA1_Channel4->CHCFG_B.MIMODE = ENABLE ;
    DMA1_Channel4->CHMADDR = (uint32_t)buffer;
    DMA1_Channel4->CHNDATA = length;

    DMA1_Channel4->CHCFG_B.CHEN = ENABLE;
    DMA1_Channel5->CHCFG_B.CHEN = ENABLE;

    while (DMA_ReadStatusFlag(DMA1_FLAG_TC5) == RESET);
    DMA_ClearStatusFlag(DMA1_FLAG_TC5);

    while (DMA_ReadStatusFlag(DMA1_FLAG_TC4) == RESET);
    DMA_ClearStatusFlag(DMA1_FLAG_TC4);

    DMA1_Channel5->CHCFG_B.CHEN = DISABLE;
    DMA1_Channel4->CHCFG_B.CHEN = DISABLE;
}

三、NorFlash的驱动
NorFlash的驱动代码大部分还是和常规SPI的方式一样,只是要修改关于这两条指令的代码:
#define W25XXX_ReadData         0x03
#define W25XXX_PageProgram    0x02

3.1 NorFlash的数据读取
读取多个数据的流程与不用DMA的方式类似,只是原来用for循环读数据的部分用spi_dam_receive_buffer()替换。。
//SPI FLASH的数据读取
void w25qxxx_read(unsigned char *buf, unsigned int addr, unsigned short num)
{
    unsigned short i;
         
    w25qxxx_cs_set(0);                              //使能器件

    w25qxxx_wr_byte(W25XXX_ReadData);               //发送读取命令
    w25qxxx_wr_byte((unsigned char)((addr) >> 16)); //发送24bit地址
    w25qxxx_wr_byte((unsigned char)((addr) >> 8));
    w25qxxx_wr_byte((unsigned char)addr);

    spi_dam_receive_buffer(buf,num);

    w25qxxx_cs_set(1);
}


3.2 NorFlash的页编程
写入多个数据的流程与以往类似,只是原来用for循环写数据的部分用spi_dam_transmit_buffer()替换。//SPI FLASH的页编程
void w25qxxx_write_page(unsigned char *buf, unsigned int addr, unsigned short num)
{
    unsigned short i;
    w25qxxx_write_enable();                      // SET WEL
    w25qxxx_cs_set(0);                              //使能器件

    w25qxxx_wr_byte(W25XXX_PageProgram);            //发送写页命令
    w25qxxx_wr_byte((unsigned char)((addr) >> 16)); //发送24bit地址
    w25qxxx_wr_byte((unsigned char)((addr) >> 8));
    w25qxxx_wr_byte((unsigned char)addr);

    spi_dam_transmit_buffer(buf, num);

    w25qxxx_cs_set(1);         //取消片选
    w25qxxx_wait_busy();      //等待写入结束
}


四、测试验证
最后我们验证上面的代码是否可行,先初始化SPI和对应的DMA。//初始化
void w25qxxx_init()
{
spi_flash_init();
spi_fullduplex_dma_init();
}
然后读一下芯片的ID,这步主要是确认一下硬件是否正常。FlashChipID = w25qxxx_read_id();
接下来就往任意一个扇区写入从0x00-0xFF的数据,擦除扇区后,写入该扇区,再读出该扇区的数据,把读出的数据和写入的原始数据作对比。//flash测试
unsigned short FlashChipID;
unsigned char flash_test_write_buffer;
unsigned char flash_test_read_buffer;
unsigned int AddrSector=0;
void w25qxxx_test()
{
unsigned int i;
      for(i=0;i<4096;i++)
      {
          flash_test_write_buffer=i%256;
      }
      
      w25qxxx_erase_sector(AddrSector);

      w25qxxx_write_sector(flash_test_write_buffer,AddrSector);
      
      w25qxxx_read(flash_test_read_buffer,AddrSector*4096,4096);
      
      if(memcmp(flash_test_read_buffer,flash_test_write_buffer,4096)==0)
      {
         printf("flash write success.");
      }
      else
      {
         printf("flash write fail!!!");
      }
}
下载程序并执行,查看输出的日志,可以看到芯片的DeviceID=0xEF16,说明硬件正常;写入数据和读出数据完全一致,读写正常。


**** Hidden Message *****

caigang13 发表于 2024-7-2 08:01

用DMA能够大大提升CPU的利用效率。

Clearhu 发表于 2024-7-7 20:50

感谢分享

Bobbyxzh 发表于 2024-9-10 16:15

学习一下

nnqtdf 发表于 2024-9-22 09:37

感谢分享

xiaogu666 发表于 2024-10-15 13:16

看看正好用到

295433181 发表于 2025-8-6 17:48

{:smile:}{:smile:}{:biggrin:}
页: [1]
查看完整版本: 玩转APM32的DMA-用SPI的DMA实现NorFlash的读写