STM32 SPI Flash DFU

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

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

这次讲的是将程序、图片或其他文件下载到SPI Flash中。我使用的是W25X16的SPI Flash,他共有2MB空间,2个Block,512ge Sector,8096个Page。由于SPI Flash不能直接跑程序,我们从接口就知道了。
接下去我们就来讲讲怎么编写SPI flash的升级功能。这次的工程是基于之前的Internal Flash修改而来的。修改的部分主要在USB_User组里:
STM32 SPI Flash DFU - ziye334 - ziye334的博客我只将改改的部分。
hw_config.c、usb_istr.c、usb_prop.c、usb_pwr.c这介个文件没有什么还修改的。usb_desc.c文件需要修改下接口字符串描述符,由于我们的SPI Flash空间2M,所以我们将SPI Flash的2M空间全部设置成可读可写可擦除。

/*接口字符串描述符*/ uint8_t DFU_StringInterface0[DFU_SIZ_STRING_INTERFACE0] = { DFU_SIZ_STRING_INTERFACE0, 0x03, //Interface 1: "@ SPI Flash: W25X16 /0x00000000/1*2048kg '@', 0, 'S', 0, 'P', 0, 'I', 0, ' ', 0, 'F', 0, 'l', 0, 'a', 0, 's', 0, /*18*/ 'h', 0, ' ', 0, ':', 0, ' ', 0, 'W', 0, '2', 0, '5', 0, 'X', 0, '1', 0, '6', 0, /*20*/ '/', 0, '0', 0, 'x', 0, '0', 0, '0', 0, '0', 0, '0', 0, '0', 0, '0', 0, '0', 0, '0', 0,/*22*/ '/', 0, '1', 0, '*', 0, '2', 0, '0', 0, '4', 0, '8', 0, 'K', 0, 'g', 0 /*18*/ };

接下去,添加我们的W25X16 SPI Flash的驱动代码spi_flash.c,这个代码可以网上下载。接下去将上个工程里的flash—_if.c文件修改成spi_if.c,并修改里面的函数:

/******************************************************************************* * Function Name : SPI_If_Init * Description : Initializes the Media on the STM32 * Input : None * Output : None * Return : None *******************************************************************************/ uint16_t SPI_If_Init(void) { SPI_Flash_Init(); return MAL_OK; }/******************************************************************************* * Function Name : SPI_If_Erase * Description : Erase sector * Input : None * Output : None * Return : None *******************************************************************************/ uint16_t SPI_If_Erase(uint32_t SectorAddress) { printf("正在擦除SPI Flash...\r\n"); SPI_Flash_Erase_Chip(); // SPI_Flash_Erase_Sector(SectorAddress); printf("擦除成功!\r\n"); return MAL_OK; }/******************************************************************************* * Function Name : SPI_If_Write * Description : Write sectors * Input : None * Output : None * Return : None *******************************************************************************/ uint16_t SPI_If_Write(uint32_t SectorAddress, uint32_t DataLength) { uint32_t idx, pages; printf("SPI_IFWrite写入数据长度为%d\r\n",DataLength); pages = (((DataLength & 0xFF00)) >> 8); if (DataLength & 0xFF) /* Not a 256 aligned data */ { for ( idx = DataLength; idx < ((DataLength & 0xFF00) + 0x100) ; idx++) // idx = DataLength; idx < ((DataLength & 0xFF00) + 0x100) ; idx++ { MAL_Buffer[idx] = 0xFF; } pages = (((DataLength & 0xFF00)) >> 8 ) + 1; } for (idx = 0; idx < pages; idx++) { printf("正在向0x%x地址写数据\r\n",SectorAddress); SPI_Flash_Write(&MAL_Buffer[idx*256], SectorAddress, 256); SectorAddress += 0x100; } return MAL_OK; }/******************************************************************************* * Function Name : SPI_If_Read * Description : Read sectors * Input : None * Output : None * Return : buffer address pointer *******************************************************************************/ uint8_t *SPI_If_Read(uint32_t SectorAddress, uint32_t DataLength) { printf("正在读取地址0x%x处开始的%d个数据\r\n",SectorAddress,DataLength); SPI_Flash_Read(MAL_Buffer, SectorAddress, (uint16_t)DataLength); return MAL_Buffer; }

然后要修改的是dfu_mal.c这个文件。修改成如下就可以了:

/******************************************************************************* * Function Name : MAL_Init * Description : STM32初始化的媒体初始化 * Input : None * Output : None * Return : None *******************************************************************************/ uint16_t MAL_Init(void) { SPI_If_Init(); /* SPI Flash */ 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 SPI_FLASH_BASE: pMAL_Erase = SPI_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 SPI_FLASH_BASE: pMAL_Write = SPI_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 SPI_FLASH_BASE: pMAL_Read = SPI_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 */ /* 0x640000000 --> 1 */ /* 0x080000000 --> 2 */ uint8_t y = Cmd & 0x01; SET_POLLING_TIMING(TimingTable[x][y]); /* x: 擦除/写 定时 */ /* y: Media */ return MAL_OK; }

最后的话,就是我们的main函数了,这里的main函数当然没有程序跳转了,我在这里用到了4个按键,
WAKEUP按键(PA0)按下表示向spi flash的0地址写入一组数据
TAMPER按键(PC13)按下表示读取0地址开始的数据
USER1按键(PA8)按下表示擦写0地址开始的那个扇区数据
USER2按键(PD3)按下表示向spi flash的0地址写入另一组数据
这样的话,就可以试试检测spi flash 读写是否正确了。

uint8_t DeviceState; uint8_t DeviceStatus[6];u8 WRITE_Buffer[]="神舟III号 SPI 读写访问程序"; //spi flash写入数据缓存 u8 WRITE_Buffer1[]="神舟I号 SPI 读写访问程序"; //spi flash写入数据缓存 u8 READ_Buffer[sizeof(WRITE_Buffer)]; //spi flash读出数据缓存/******************************************************** 函数:main() 描述:程序入口地址 参数:无 返回:无 ********************************************************/ int main(void) { BSP_Init(); printf(" |===============================================|\r\n"); printf(" STM32 DFU 程序开始 \r\n"); printf("|===============================================|\r\n"); SPI_Flash_Init(); { u32 i; i=SPI_Flash_ReadID(); //读取spi flash的芯片ID,一定要读,否则读写会出错 printf("ID:%x\r\n",i); } /* Enter DFU mode */ DeviceState = STATE_dfuERROR; //程序指向到这句话,说明DFU跳转不成功 DeviceStatus[0] = STATUS_ERRFIRMWARE; DeviceStatus[4] = DeviceState; USB_Configuration(); //初始化USB while(1) { if(KEY1_STATE()==0) //按键1按下,则向spi flash写入数据 { while(KEY1_STATE()==0); printf("开始写入W25X16 SPI FLASH芯片!\r\n"); SPI_Flash_Erase_Sector(0); SPI_Flash_Write(WRITE_Buffer,0,sizeof(WRITE_Buffer)); printf("写入完成!\r\n"); } if(KEY2_STATE()==0) //按键2按下,读出spi flash的数据 { while(KEY2_STATE()==0); printf("开始从W25X16 SPI FLASH芯片中读取数据!\r\n"); SPI_Flash_Read(READ_Buffer,0,sizeof(READ_Buffer)); printf("读取完成,读书的数据为:\r\n%s\r\n",READ_Buffer); } if(KEY3_STATE()==0) //按键3按下,擦除整块spi flash数据 { u8 i; while(KEY3_STATE()==0); printf("正在擦除W25X16 SPI FLASH芯片!\r\n"); SPI_Flash_Erase_Sector(0); // SPI_Flash_Erase_Chip(); printf("擦除完毕!\r\n"); for(i=0;i<sizeof(READ_Buffer);i++) { READ_Buffer[i]=0; } } if(KEY4_STATE()==0) //按键1按下,则向spi flash写入数据 { while(KEY4_STATE()==0); printf("开始写入W25X16 SPI FLASH芯片!\r\n"); SPI_Flash_Erase_Sector(0); SPI_Flash_Write(WRITE_Buffer1,0,sizeof(WRITE_Buffer1)); printf("写入完成!\r\n"); } } }


这个工程需要注意的是 *.dfu文件的制作,因为spi flash是外界的存储器,所以与单片机存储地址独立编址,即从0x00000000地址开始到0x001fffffff结束,所以在制作.dfu的时候,千万要设置地址在这个范围内。

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



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

相关文章

零基础STM32单片机编程入门(一)初识STM32单片机

文章目录 一.概要二.单片机型号命名规则三.STM32F103系统架构四.STM32F103C8T6单片机启动流程五.STM32F103C8T6单片机主要外设资源六.编程过程中芯片数据手册的作用1.单片机外设资源情况2.STM32单片机内部框图3.STM32单片机管脚图4.STM32单片机每个管脚可配功能5.单片机功耗数据6.FALSH编程时间,擦写次数7.I/O高低电平电压表格8.外设接口

STM32单片机PWR电源控制详解

文章目录 1. PWR概述 2. 电源结构框图 3. 上电复位和掉电复位 4. 可编程电压监测器 5. 低功耗模式 6. 模式选择 6.1 睡眠模式 6.2 停止模式 6.3 待机模式 7. 代码示例 1. PWR概述 PWR(Power Control)电源控制,负责管理STM32内部的电源供电部分,可以实现可编程电压监测器和低功耗模式的功能。 可编程电压监测器

google gemini1.5 flash视频图文理解能力初探(一)

市面能够对视频直接进行分析的大模型着实不多,而且很多支持多模态的大模型那效果着实也不好。 从这篇公众号不只是100万上下文,谷歌Gemini 1.5超强功能展示得知,Gemini 1.5可以一次性处理1小时的视频、11小时的音频或100,000行代码,并衍生出更多的数据分析玩法。能力覆盖: 跨模式理解和推理,当给出一部 44 分钟的巴斯特-基顿(Buster Keaton)无声电影时,该模型能准

STM32学习之一:什么是STM32

目录 1.什么是STM32 2.STM32命名规则 3.STM32外设资源 4. STM32的系统架构 5. 从0到1搭建一个STM32工程 学习stm32已经很久了,因为种种原因,也有很久一段时间没接触过stm32了。等我捡起来的时候,发现很多都已经忘记了,重新捡起来吧。 每次谈到stm32如何如何,那么该如何解释什么是stm32呢? 1.什么是STM32 stm32

Ubuntu安装火狐Flash Player插件

1、进入官网下载页面选择:.tar.gz,适用于其他Linux。此次文件名为install_flash_player_11_linux.x86_64.tar.gz 2、解压文件 tar -zxvf install_flash_player_11_linux.x86_64.tar.gz 3、利用whereis命令查找mozilla文件夹的路径 whereis mozill

STM32学习 修改系统主频

前面时钟树的学习说明单片机的主频是可以修改的,那么怎么更改系统的主频,这里做一个简单的介绍。首先要明白,单片机的程序是如何运行,这里简单说明一下。 对应的代码在startup_stm32....文件里面,这里是复位程序的汇编代码。 复位子程序是系统上电后第一个执行的程序,调用 SystemInit 函数初始化系统时钟,然后调用 C 库函数 _mian,最终调用 main 函数去到 C

使用J-Link Commander查找STM32死机问题

接口:PA13,PA14,请勿连接复位引脚。 输入usb命令 这里我已经连接过了STM32F407VET6了。 再输入connect命令 这里我已经默认选择了SWD接口,4000K速率。 可以输入speed 4000命令选择4000K速率: 写一段崩溃代码进行测试: void CashCode(void){*((volatile uint32_t*) 0x080FFFFF)

[技术笔记] 元器件采购之Flash的国内、外厂商Top5

国外Top5 1、Micron(镁光)半导体 2、Toshiba(东芝) 3、Hynix(海力士) 4、Samsung(三星) 5、Intel(因特尔) 6、SanDisk(闪迪) 7、Nanya(南亚科技) 8、SCSemicon(华芯科技) 中国企业在主流通用型存储器的市场全球市占率比较低,而在NOR Flash利基存储器领域成长了一批优秀企业,包括兆易创新、东芯股份

基于STM32的智能家居安防系统

目录 引言环境准备智能家居安防系统基础代码实现:实现智能家居安防系统 4.1 数据采集模块4.2 数据处理与分析4.3 控制系统实现4.4 用户界面与数据可视化应用场景:智能家居安防管理与优化问题解决方案与优化收尾与总结 1. 引言 智能家居安防系统通过使用STM32嵌入式系统,结合多种传感器和控制设备,实现对家庭环境的实时监测和安防管理。本文将详细介绍如何在STM32系统中实现一个智能家居

我在高职教STM32——LCD液晶显示(3)

大家好,我是老耿,高职青椒一枚,一直从事单片机、嵌入式、物联网等课程的教学。对于高职的学生层次,同行应该都懂的,老师在课堂上教学几乎是没什么成就感的。正因如此,才有了借助 CSDN 平台寻求认同感和成就感的想法。在这里,我准备陆续把自己花了很多心思的教学设计分享出来,主要面向广大师生朋友,单片机老鸟就略过吧。欢迎点赞+关注,各位的支持是本人持续输出的动力,多谢多谢!