发新帖本帖赏金 50.00元(功能说明)我要提问
返回列表
[APM32E1]

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

[复制链接]
537|3
手机看帖
扫描二维码
随时随地手机跟帖
shanyuxiang|  楼主 | 2024-7-1 21:19 | 显示全部楼层 |阅读模式
本帖最后由 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)

1-1.png

2.png


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

Read Data      03h
Page Program 02h

2-5.png


1.2 关于SPI和DMA

通过查阅用户手册“APM32E103xCxE用户手册”的DMA章节可以知道SPI2的对应通道,SPI2接收对应DMA1_Channel4,SPI2发送对应DMA1_Channel5。

Flash的读数据可以用DMA1_Channel4来传输,Flash的页编程可以用DMA1_Channel5来传输。

3.png

二、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[4096];
unsigned char flash_test_read_buffer[4096];
unsigned int AddrSector=0;
void w25qxxx_test()
{
  unsigned int i;
        for(i=0;i<4096;i++)
        {
          flash_test_write_buffer[i]=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,说明硬件正常;写入数据和读出数据完全一致,读写正常。
4.png


游客,如果您要查看本帖隐藏内容请回复

使用特权

评论回复

打赏榜单

21小跑堂 打赏了 50.00 元 2024-07-18
理由:恭喜通过原创审核!期待您更多的原创作品~

评论
21小跑堂 2024-7-18 14:53 回复TA
在APM32上使用SPI+DMA的方式高效读写NorFlash,在DMA的加持下,MCU可以在大量数据的读写时留出更多的时间处理其他事件。文章思路清晰,结构完整,适合借鉴学习 
caigang13| | 2024-7-2 08:01 | 显示全部楼层
用DMA能够大大提升CPU的利用效率。

使用特权

评论回复
Clearhu| | 2024-7-7 20:50 | 显示全部楼层
感谢分享

使用特权

评论回复
发新帖 本帖赏金 50.00元(功能说明)我要提问
您需要登录后才可以回帖 登录 | 注册

本版积分规则

5

主题

23

帖子

1

粉丝