STM32 Nor Flash DFU

2024-06-04 18:38
文章标签 stm32 flash dfu

本文主要是介绍STM32 Nor Flash DFU,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

这次要讲讲怎么实现Nor Flash的升级。
Nor Flash的DFU工程还是基于之前的flash DFU的工程上修改而来。工程的目录如下:
STM32 Nor Flash DFU - ziye334 - ziye334的博客
 
我使用的Nor Flash芯片是M29W128F, 该芯片共有128Mb的空间,通过FSMC挂接在BANK0。 正好在UBS的官方程序里也有使用芯片的例子,所以也就是说管方的Nor Flash的驱动代码是使用 M29W128F这款芯片的。所以我们需要从拷贝fsmc_nor.c和fsmc_nor,h这两个文件添加到我们的USB_User这个组中。还要讲我们之前的flash_if文件修改为nor_if名。这样工程的文件就算齐了,下面就讲讲怎么修改个文件。
首先hw_config、usb_istr、usb_prop、usb_pwr这些个文件不需要修改。最新需要修该的文件是usb_desc.c这个文件的接口描述符,上面说过了,我们使用的Nor Flash芯片是M29W128,总空间为128Mb,即16MB。我设置这些空间都可读可写可擦除,并以64K为单位,所以接口描述符如下:

/*接口字符串描述符*/ uint8_t DFU_StringInterface0[DFU_SIZ_STRING_INTERFACE0] = { DFU_SIZ_STRING_INTERFACE0, 0x03, //Interface 0: "@ NOR Flash: M29W128F /0x64000000/256*064Kg" '@', 0, 'N', 0, 'O', 0, 'R', 0, ' ', 0, 'F', 0, 'l', 0, 'a', 0, 's', 0, /*18*/ 'h', 0, ' ', 0,':', 0, ' ',0,'M',0,'2',0,'9',0,'W',0,'1',0,'2',0,'8',0,'F',0, /*24*/ '/',0,'0',0,'x',0,'6',0,'4',0,'0',0,'0',0,'0',0,'0',0,'0',0,'0',0, /*22*/ '/', 0, '0', 0, '2', 0, '5', 0, '6', 0, '*', 0, '6', 0, '4', 0, 'K', 0, 'g', 0 /*20*/ };

接下去需要修改的是nor_if.c这个文件,这个文件是介于驱动和MAL层之间的桥梁。这个文件主要是进一步封装一下nor flash的底层驱动程序。这款需要定义NOR_If_Init()、NOR_If_Erase()、NOR_If_Write()、NOR_If_Read()这4个函数,都是调用驱动代码的相关函数:

/******************************************************************************* * Function Name : NOR_If_Init * Description : Initializes the Media on the STM32 * Input : None * Output : None * Return : None *******************************************************************************/ uint16_t NOR_If_Init(void) { FSMC_NOR_Init(); return MAL_OK; }/******************************************************************************* * Function Name : NOR_If_Erase * Description : Erase sector * Input : None * Output : None * Return : None *******************************************************************************/ uint16_t NOR_If_Erase(uint32_t Address) { printf("正在擦除Nor Flash\r\n"); /* Erase the destination memory */ FSMC_NOR_EraseBlock(Address & 0x00FFFFFF); printf("擦除成功\r\n"); return MAL_OK; }/******************************************************************************* * Function Name : NOR_If_Write * Description : Write sectors * Input : None * Output : None * Return : None *******************************************************************************/ uint16_t NOR_If_Write(uint32_t Address, uint32_t DataLength) { if ((DataLength & 1) == 1) /* Not an aligned data */ { DataLength += 1; MAL_Buffer[DataLength-1] = 0xFF; } printf("向0x%x地址处写入%d个字节\r\n",Address,DataLength); FSMC_NOR_WriteBuffer((uint16_t *)MAL_Buffer, (Address&0x00FFFFFF), DataLength >> 1); return MAL_OK; }/******************************************************************************* * Function Name : NOR_If_Read * Description : Read sectors * Input : None * Output : None * Return : buffer address pointer *******************************************************************************/ uint8_t *NOR_If_Read(uint32_t Address, uint32_t DataLength) { printf("在0x%x地址处读出%d个字节数据\r\n",Address,DataLength); return (uint8_t*)(Address); }

再接下去就是dfu_mal.c媒体接入层的函数了。这个文件也只定义了5个函数:MAL_Init()、MAL_Erase()、MAL_Write()、MAL_Read()、MAL_GetStatus(),这些函数基本上是调用nor_if.c中定义的相关函数:

uint16_t (*pMAL_Init) (void); uint16_t (*pMAL_Erase) (uint32_t SectorAddress); uint16_t (*pMAL_Write) (uint32_t SectorAddress, uint32_t DataLength); uint8_t *(*pMAL_Read) (uint32_t SectorAddress, uint32_t DataLength); uint8_t MAL_Buffer[wTransferSize]; /* RAM Buffer for Downloaded Data */NOR_IDTypeDef NOR_ID;extern ONE_DESCRIPTOR DFU_String_Descriptor[7];static const uint16_t TimingTable[3][2] = { /* 扇区擦写时间, 扇区编程时间*/ { SPI_FLASH_SECTOR_ERASE_TIME, SPI_FLASH_SECTOR_WRITE_TIME }, /* SPI Flash */ { M29W128F_SECTOR_ERASE_TIME, M29W128F_SECTOR_WRITE_TIME }, /* NOR Flash M29W128F */ { INTERN_FLASH_SECTOR_ERASE_TIME, INTERN_FLASH_SECTOR_WRITE_TIME }, /* Internal Flash */ }; /******************************************************************************* * Function Name : MAL_Init * Description : STM32初始化的媒体初始化 * Input : None * Output : None * Return : None *******************************************************************************/ uint16_t MAL_Init(void) { FSMC_NOR_Init(); NOR_If_Init(); FSMC_NOR_ReadID(&NOR_ID); printf(" Nor Flash ID:0x%x 0x%x\r\n",NOR_ID.Manufacturer_Code,NOR_ID.Device_Code1); FSMC_NOR_ReturnToReadMode(); return MAL_OK; }/******************************************************************************* * Function Name : MAL_Erase * Description : 擦除扇区 * Input : None * Output : None * Return : None *******************************************************************************/ uint16_t MAL_Erase(uint32_t SectorAddress) { switch (SectorAddress & MAL_MASK) //参看地址 { case NOR_FLASH_BASE: pMAL_Erase = NOR_If_Erase; break; default: return MAL_FAIL; } return pMAL_Erase(SectorAddress); //指向擦除函数 }/******************************************************************************* * Function Name : MAL_Write * Description : 写扇区 * Input : None * Output : None * Return : None *******************************************************************************/ uint16_t MAL_Write (uint32_t SectorAddress, uint32_t DataLength) { switch (SectorAddress & MAL_MASK) //查看地址 { case NOR_FLASH_BASE: pMAL_Write = NOR_If_Write; break; default: return MAL_FAIL; } return pMAL_Write(SectorAddress, DataLength);//调用写扇区函数 }/******************************************************************************* * Function Name : MAL_Read * Description : 度扇区 * Input : None * Output : None * Return : Buffer pointer *******************************************************************************/ uint8_t *MAL_Read (uint32_t SectorAddress, uint32_t DataLength) { switch (SectorAddress & MAL_MASK) //查看地址 { case NOR_FLASH_BASE: pMAL_Read = NOR_If_Read; break; default: return 0; } return pMAL_Read (SectorAddress, DataLength);//调用如扇区函数 }/******************************************************************************* * Function Name : MAL_GetStatus * Description : 获取状态 * Input : None * Output : None * Return : MAL_OK *******************************************************************************/ uint16_t MAL_GetStatus(uint32_t SectorAddress , uint8_t Cmd, uint8_t *buffer) { //更具地址查找定时表的对应的选项 uint8_t x = (SectorAddress >> 26) & 0x03 ; /* 0x000000000 --> 0 SPI Flash*/ /* 0x640000000 --> 1 Nor Flash*/ /* 0x080000000 --> 2 Internal Flash*/ uint8_t y = Cmd & 0x01; SET_POLLING_TIMING(TimingTable[x][y]); /* x: 擦除/写 定时 */ /* y: Media */ return MAL_OK; }

最后,我们在main中定义一些测试Nor Flash的一些代码, 按键1按下擦写0x64000000地址开始的那个扇区数据; 按键2按下表示向spi flash的0地址写入一组数据; 按键3按下表示表示向spi flash的0地址写入另一组数据; 按键4按下表示读取0地址开始的数据:

u16 TxBuffer0[8]={0x0000,0x0002,0x0004,0x00080,0x0010,0x0020,0x0040,0x0080}; u16 TxBuffer1[8]={0x0100,0x0200,0x0400,0x08000,0x1000,0x2000,0x4000,0x8000}; u16 RxBuffer[8];

int main(void) { u8 i=8; BSP_Init(); printf(" |===============================================|\r\n"); printf(" STM32 DFU 程序开始 \r\n"); printf("|===============================================|\r\n"); /* Enter DFU mode */ DeviceState = STATE_dfuERROR; //程序指向到这句话,说明DFU跳转不成功 DeviceStatus[0] = STATUS_ERRFIRMWARE; DeviceStatus[4] = DeviceState; USB_Configuration(); //初始化USB while(1) { if(!KEY1_STATE()) { while(!KEY1_STATE()); printf("正在擦除Nor Flash!\r\n"); FSMC_NOR_EraseBlock(0); printf("擦除成功!\r\n"); } if(!KEY2_STATE()) { while(!KEY2_STATE()); i=8; printf("正在擦除,请稍等...\r\n"); FSMC_NOR_EraseBlock(0); printf("擦除完毕,正在写入!"); FSMC_NOR_WriteBuffer(TxBuffer0,0,8); printf("向Nor Flash写入的数据为:\r\n"); while(i--) printf("0x%x ",TxBuffer0[i]); printf("\r\n数据写入完毕!\r\n"); } if(!KEY3_STATE()) { while(!KEY3_STATE()); i=8; printf("正在擦除,请稍等...\r\n"); FSMC_NOR_EraseBlock(0); printf("擦除完毕,正在写入!"); FSMC_NOR_WriteBuffer(TxBuffer1,0,8); printf("向Nor Flash写入的数据为:\r\n"); while(i--) printf("0x%x ",TxBuffer1[i]); printf("\r\n数据写入完毕!\r\n"); } if(!KEY4_STATE()) { while(!KEY4_STATE()); i=8; printf("正在读取数据...\r\n"); FSMC_NOR_ReadBuffer(RxBuffer,0,8); printf("读出来的数据为:\r\n"); while(i--) printf("0x%x ",RxBuffer[i]); printf("\r\n数据读取完毕!\r\n"); } } }

这篇关于STM32 Nor Flash DFU的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1030805

相关文章

【STM32】SPI通信-软件与硬件读写SPI

SPI通信-软件与硬件读写SPI 软件SPI一、SPI通信协议1、SPI通信2、硬件电路3、移位示意图4、SPI时序基本单元(1)开始通信和结束通信(2)模式0---用的最多(3)模式1(4)模式2(5)模式3 5、SPI时序(1)写使能(2)指定地址写(3)指定地址读 二、W25Q64模块介绍1、W25Q64简介2、硬件电路3、W25Q64框图4、Flash操作注意事项软件SPI读写W2

什么是 Flash Attention

Flash Attention 是 由 Tri Dao 和 Dan Fu 等人在2022年的论文 FlashAttention: Fast and Memory-Efficient Exact Attention with IO-Awareness 中 提出的, 论文可以从 https://arxiv.org/abs/2205.14135 页面下载,点击 View PDF 就可以下载。 下面我

STM32(十一):ADC数模转换器实验

AD单通道: 1.RCC开启GPIO和ADC时钟。配置ADCCLK分频器。 2.配置GPIO,把GPIO配置成模拟输入的模式。 3.配置多路开关,把左面通道接入到右面规则组列表里。 4.配置ADC转换器, 包括AD转换器和AD数据寄存器。单次转换,连续转换;扫描、非扫描;有几个通道,触发源是什么,数据对齐是左对齐还是右对齐。 5.ADC_CMD 开启ADC。 void RCC_AD

STM32内部闪存FLASH(内部ROM)、IAP

1 FLASH简介  1 利用程序存储器的剩余空间来保存掉电不丢失的用户数据 2 通过在程序中编程(IAP)实现程序的自我更新 (OTA) 3在线编程(ICP把整个程序都更新掉) 1 系统的Bootloader写死了,只能用串口下载到指定的位置,启动方式也不方便需要配置BOOT引脚触发启动  4 IAP(自己写的Bootloader,实现程序升级) 1 比如蓝牙转串口,

FreeRTOS-基本介绍和移植STM32

FreeRTOS-基本介绍和STM32移植 一、裸机开发和操作系统开发介绍二、任务调度和任务状态介绍2.1 任务调度2.1.1 抢占式调度2.1.2 时间片调度 2.2 任务状态 三、FreeRTOS源码和移植STM323.1 FreeRTOS源码3.2 FreeRTOS移植STM323.2.1 代码移植3.2.2 时钟中断配置 一、裸机开发和操作系统开发介绍 裸机:前后台系

寻迹模块TCRT5000的应用原理和功能实现(基于STM32)

目录 概述 1 认识TCRT5000 1.1 模块介绍 1.2 电气特性 2 系统应用 2.1 系统架构 2.2 STM32Cube创建工程 3 功能实现 3.1 代码实现 3.2 源代码文件 4 功能测试 4.1 检测黑线状态 4.2 未检测黑线状态 概述 本文主要介绍TCRT5000模块的使用原理,包括该模块的硬件实现方式,电路实现原理,还使用STM32类

STM32 ADC+DMA导致写FLASH失败

最近用STM32G070系列的ADC+DMA采样时,遇到了一些小坑记录一下; 一、ADC+DMA采样时进入死循环; 解决方法:ADC-dma死循环问题_stm32 adc dma死机-CSDN博客 将ADC的DMA中断调整为最高,且增大ADCHAL_ADC_Start_DMA(&hadc1, (uint32_t*)adc_buffer, ADC_Buffer_Size); 的ADC_Bu

基于stm32的河流检测系统-单片机毕业设计

文章目录 前言资料获取设计介绍功能介绍具体实现截图参考文献设计获取 前言 💗博主介绍:✌全网粉丝10W+,CSDN特邀作者、博客专家、CSDN新星计划导师,一名热衷于单片机技术探索与分享的博主、专注于 精通51/STM32/MSP430/AVR等单片机设计 主要对象是咱们电子相关专业的大学生,希望您们都共创辉煌!✌💗 👇🏻 精彩专栏 推荐订阅👇🏻 单片机设计精品

STM32的使用方法一

注:我采用的是STM32F103RC芯片、相应的电路图和STM32CubeIDE软件这是在STM32CubeIDE软件定义芯片后,所给的必要的代码逻辑,加上了注释 #include "main.h"/* Private variables ---------------------------------------------------------*//* Private function

STM32 HAL CAN通讯 实操

1、简介 相比于串口通讯,对于刚接触CAN通讯的小白来说,CAN通讯相对复杂,看各种视频、帖子理论,总是一知半解。本次通过傻瓜式操作,先实现CAN通讯的交互,以提高小白的信心,也便于自己复习观看。本次以STM32CubeMX进行初始化配置,通过Keil 5软件进行软件设计,通过CAN盒实现进行数据的交互。该流程实际以STM32F0、F1、F3、F4、F7实测好用(理论上都适用),这三种型号单片机