STM32快速复制MX25L1606E系列Flash

news2025/1/13 19:35:39

去年做了一个使用RS485对PIC18F45K80系列单片机进行在线升级的程序,如果是小批量的出厂烧录程序和升级验证(出厂前肯定要测试单片机是否能正常读写Flash)是可以的,但是后来产品订单量很大,生产线的烧录及升级验证就很缓慢,主要是发送升级包这一步,主要原因如下:

1、升级数据包有20多K字节,而PIC18F45K80系列的内存只有3K多;

2、数据包先保存到PIC18F45K80的缓存中,然后再写入MX25L1606E;

3、根据客户软件协议要求,RS485通信需要将一个字节拆分成两个字节进行传输,例如:如果要发送0xAB,则会拆分成0x3A和0x3B两个字节,接收处理程序再将这两个字节合并成0xAB,这无疑使数据的传输时间加倍;

4、RS485通信的波特率被限制为9600,说是为了数据传输的稳定性;

生产部门表达了不满,我本想给他们解释原因,但后来想与其和一些不懂技术的人瞎争论、浪费时间,还不如想点实际的办法。简单分析:整个IAP中最费时间的就是将升级包通过RS485写入到MX25L1606E,那如果有办法直接PIN对PIN通过SPI通信将升级包复制到MX25L1606E不就解决问题了吗?那速度是极快的。软件倒是好做,但问题是:MX25L1606E这个芯片太小了,不好设计工装或者治具,而且必须要先贴片,那说明彻底不能使用治具了。后来负责该项目的硬件工程师在网上找到了个IC夹子,针对各种PIN针的都有,于是用STM32F103VET6开发板写了个测试程序并在原来的PCB板上测试,只要IC夹子和PCB板上的MX25L1606E脚位对齐还真是可以复制!太开心了,赶紧拿来上电测试,果然可以正确执行升级程序!!!

以下是STM32的主程序,内容比较简单,主要就是使用了一个按键触发执行复制操作,最后使用显示屏简单提示操作结果。最主要是它内存资源充足,那20多KB的升级包轻松放入内存,当然也可使用内存资源少的单片机,只是多次读取升级文件,速度也不会慢多少。整个工程可以免费到以下链接下载:【免费】STM32快速复制MX25L1606E资源-CSDN文库,有需要的朋友可以试一下。

主程序如下,升级文件内容是16进制格式的,也就是firm_ware这个数组,内容我就省略了很多,因为太占篇幅了,大概有两万多个字节,这个不是重点:

/******************** (C) COPYRIGHT 2023 DS **************************
 * @File name  :main.c
 * @Description    :Fast flash clone
 * @Hardware platform:STM32F103VET6
 * @Library version :ST3.5.0 *
 * @Developer  :Power
**********************************************************************************/


/* Includes ------------------------------------------------------------------*/
#include "stm32f10x.h"
#include "spi.h"
#include "led.h"
#include "delay.h"
#include "sys_temp.h"
#include "tim.h"
#include "wwdg.h"
#include "usart1.h"
#include "key.h"
#include "bsp_ili9341_lcd.h"

#define     FLASH_PAGE_SIZE                     256

struct ALARM {
    unsigned    KEY_ERASE_PRESSED               : 1;
    unsigned    CLONE_DONE                      : 1;
    unsigned                                    : 1;
    unsigned                                    : 1;
    unsigned                                    : 1;
    unsigned                                    : 1;
    unsigned                                    : 1;
    unsigned                                    : 1;

    unsigned                                    : 1;
    unsigned                                    : 1;
    unsigned                                    : 1;
    unsigned                                    : 1;
    unsigned                                    : 1;
    unsigned                                    : 1;
    unsigned                                    : 1;
    unsigned                                    : 1;
};

union ALARM_STATUS_UNION {
    struct ALARM    alarm;
    uint16_t    allbits;
};

typedef union ALARM_STATUS_UNION ALARM_STATUS;

uint16_t    led_cnt = 0;//LED test
volatile uint8_t Flag_1ms;      //1 milli-second overflow flag
volatile uint8_t cDelay_80ms;   //80 milli-seconds counter
ALARM_STATUS AlarmStatus;
uint32_t chk_sum = 0;
uint32_t chk_sum_flash = 0;
uint16_t i = 0;
uint16_t data_len = 0;
uint8_t version[8] = {0x00};
uint8_t buffer[FLASH_PAGE_SIZE] = {0x00}; //used to save the data read from MX25L1606E
uint8_t page_num = 0;
uint8_t single_data_num = 0;
uint32_t flash_addr = 0;
uint8_t key = 0xFF;

/* the new firmware hex file */
uint8_t firm_ware[] =
{
	0x12,
	0x40,
	0x18,
	0x00,
	0x00,
	0x0F,
	0xD0,
	0x9E,
	0x90,
	0xFA,
	0x0E,
	0xCF,
	0x6E,
	0x9A,
};

/**
*@name: Process_1MS
*@description: 1 milli-second overflow process
*@params: none
*@return: none
*/
static void Process_1MS(void)
{
    if (Flag_1ms == 0)
    {
        return;
    }

    Flag_1ms = 0;

    if (led_cnt % 500 == 0) //LED Toggle Test
    {
        GPIO_WriteBit(GPIOB, GPIO_Pin_0, (BitAction)(1 - GPIO_ReadOutputDataBit(GPIOB, GPIO_Pin_0)));
    }
    led_cnt++;
    if (led_cnt >= 60000)
    {
        led_cnt = 0;
    }
    //if(cDelay10ms != 0)   cDelay10ms--;
    //if(++cDelay10ms1 >= 100)  cDelay10ms1 = 100;
    //if(++cDelay10ms2 >= 100)  cDelay10ms1 = 100;
    //usTmr_ms++;
}

/**
*@name: Copy_Data_Process
*@description: erase the designated area of the target MX25L1606E series flash
               and write it with the firmware
*@params: none
*@return: none
*/
static void Copy_Data_Process(void)
{
    if (AlarmStatus.alarm.KEY_ERASE_PRESSED == 1 && AlarmStatus.alarm.CLONE_DONE == 0)
    {
        uint32_t id = 0;
        LCD9341_Clear(0xFFFF);


        /* STEP1: read the manufacture ID from the target */
        // Manufacture ID: 0x000000EF
        // Device ID: 0x00000016
        id = SPI_FLASH_ReadDeviceID();
        if (id == 0x0000) //invalid device ID
        {
			//LCD_Display_String("Reading device ID failed!", 0);
			//AlarmStatus.alarm.KEY_ERASE_PRESSED = 0;
			//AlarmStatus.alarm.CLONE_DONE = 1;
			//return;
        }
        

        /* STEP2: erase the target */
        SPI_FLASH_SectorErase(FLASH_Sector4);//used to store the version and check sum information
        Delay_ms(100);
        SPI_FLASH_BlockErase(FLASH_Sector3);//one block is enough to store the firmware data
        Delay_ms(100);


        /* STEP3: copy the version data as well as calculate the check sum */
        chk_sum = 0;
        chk_sum_flash = 0;
        data_len = sizeof(firm_ware) / sizeof(uint8_t) -1;//minus 1 here means that the first byte is version and should be excluded from the total data length
        for (i = 0; i < data_len; i++)
        {
            chk_sum += firm_ware[i + 1];
        }
        version[0] = 0x56; //ASCII code standing for letter 'V'
        version[1] = firm_ware[0]; //version, for example, 0x13 means V1.3
        version[2] = data_len >> 8;
        version[3] = data_len & 0xFF;
        version[4] = (chk_sum & 0xFF000000) >> 24;
        version[5] = (chk_sum & 0x00FF0000) >> 16;
        version[6] = (chk_sum & 0x0000FF00) >> 8;
        version[7] = chk_sum & 0x000000FF;
        for (i = 0; i < 8; i++)//the version information check sum
        {
            chk_sum += version[i];
        }
        SPI_FLASH_BufferWrite(version, FLASH_Sector4, sizeof(version) / sizeof(uint8_t));
        Delay_ms(100);


        /* STEP4: copy the firmware data */
        SPI_FLASH_BufferWrite(&firm_ware[1], FLASH_Sector3, data_len);
        Delay_ms(100);
        /* STEP5: check the data */
        for (i = 0; i < 8; i++)
        {
            buffer[i] = 0x00;
        }
        SPI_FLASH_BufferRead(buffer, FLASH_Sector4, 8);//read the version information
        for (i = 0; i < 8; i++)
        {
            chk_sum_flash += buffer[i];
        }
        page_num = data_len / FLASH_PAGE_SIZE; //page number
        single_data_num = data_len % FLASH_PAGE_SIZE; //single data number
        flash_addr = FLASH_Sector3;
        while (page_num--)
        {
            SPI_FLASH_BufferRead(buffer, flash_addr, FLASH_PAGE_SIZE);
            for (i = 0; i < FLASH_PAGE_SIZE; i++)
            {
                chk_sum_flash += buffer[i];
            }
            flash_addr += FLASH_PAGE_SIZE;
        }
        if (single_data_num > 0)
        {
            SPI_FLASH_BufferRead(buffer, flash_addr, single_data_num);
            for (i = 0; i < single_data_num; i++)
            {
                chk_sum_flash += buffer[i];
            }
        }
        if (chk_sum == chk_sum_flash && chk_sum != 0) //the data written to the flash is correct
        {
            LCD_Display_String("Clone Succeeded! Please change another MX25L1606E", 1);
        }
        else
        {
            LCD_Display_String("The Clone operation failed!", 0);
        }
        AlarmStatus.alarm.KEY_ERASE_PRESSED = 0;
        AlarmStatus.alarm.CLONE_DONE = 1;
    }
}

/**
*@name: Process_80MS
*@description: 80 milli-seconds timeout process
*@params: none
*@return: none
*/
static void Process_80MS(void)
{
    if (cDelay_80ms >= 240)
    {
        cDelay_80ms = 0;
    }
}

static void Key_Process(void)
{
    key = KEY_Scan(1);
    if (key == KEY_CLONE && AlarmStatus.alarm.KEY_ERASE_PRESSED == 0)
    {
        AlarmStatus.alarm.KEY_ERASE_PRESSED = 1;
        AlarmStatus.alarm.CLONE_DONE = 0;
        Copy_Data_Process();
    }
}

/*
 main function
 */
int main(void)
{
    /**************************************************************************************************/
    SysTick_Init();   //has already been called in the startup file, so here we don't need to call it explicitly.
    USART1_Config();//print the debug info
    Source_MX25L1605E_Init();//source and target flash initialization
    LED_Init();
    KEY_Init();
    TIM_Configuration();
    bsp_InitLCD();
    LCD9341_Clear(0xFFFF);
    //WWDG_Init(0x7F, 0x5F, WWDG_Prescaler_2);
    while (1)
    {
        /* STM32 software reset */
        //SystemReset();

        //wr=WWDG->CFR&0X7F; // 重新设置看门狗窗口值
        //tr=WWDG->CR&0X7F;  // 重新设置看门狗递减计数器值
        /* 注意tr和wr的比较,确定喂狗时间 */
        //if(tr<wr)//计数器值tr必须小于窗口值wr时才能喂狗,在之前喂狗则太早,会产生看门狗复位
        //{
        //WWDG_SetCounter(WWDG_CNT);//更新计数器
        //printf("Feeding dog now......\r\n");
        //}
        Process_1MS();
        Process_80MS();
        Key_Process();
    }
}

复制MX25L1606E的步骤如下:

1、先读取它的ID相关信息,如果读取正确则证明SPI通信正常;

2、擦除MX25L1606E上的相关块和扇区,比如:本程序一个扇区用于保存事先计算好的校验和及总的升级包长度,一个块用于存储实际的新的升级固件,一个块用于保存旧的在ROM中运行的程序。在执行IAP时会对比事先计算好的校验和及长度是否和实际读取的校验和及长度相同,如果相同证明数据完整无误,否则数据无效不能升级;

3、计算新固件的校验和及长度并写入到第2步擦除的扇区中,同时写入新固件到第2步指定的块;

4、验证MX25L1606E中写入的数据是否和新的固件完全一致并通过显示屏提示操作结果。

使用这个IC夹子后生产效率极大提升,之前光发送升级固件这一个步骤就需要约两分种,现在只要熟悉这个IC夹子和MX25L1606E对应PIN针的位置,基本不到10秒就可完成一个!

附IC夹子示意图,我是使用野火的STM32开发板通过SPI接口连接MX25L1606E

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1385652.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

QT上位机开发(多线程处理)

【 声明&#xff1a;版权所有&#xff0c;欢迎转载&#xff0c;请勿用于商业用途。 联系信箱&#xff1a;feixiaoxing 163.com】 目前大部分cpu都是多核cpu&#xff0c;所以如果不用多线程进行数据处理的话&#xff0c;实在是太浪费资源了。另外&#xff0c;很多的操作都是阻塞…

数据中心温湿度监测,这个细节你真的看懂了吗?

在当今数字化和智能化的时代&#xff0c;温湿度监控系统成为了各行业中不可或缺的关键技术&#xff0c;其在维护设备、保障产品质量和确保生产环境稳定性方面发挥着至关重要的作用。 随着科技的不断进步&#xff0c;企业和机构越来越意识到对环境参数的实时监测对于提高效率、降…

采样次数与频率的关系

采样次数&#xff08;Sampling Points&#xff09; 在给定时间内记录信号值的次数。 假设在1秒内对一个连续信号采样10次&#xff0c;这意味着每0.1秒记录一次信号值。 假设在1秒内对一个连续信号采样100次&#xff0c;这意味着每0.01秒记录一次信号值。 频率&#xff08;Fre…

GitLab CI 实现项目A更新代码自动触发项目B更新错误码文档

一、CI/CD简介 CI/CD 是持续集成&#xff08;Continuous Integration&#xff09;和持续交付/持续部署&#xff08;Continuous Delivery/Continuous Deployment&#xff09;的缩写&#xff0c;是一种软件开发和交付的最佳实践。这两个概念通常一起使用&#xff0c;但有些时候它…

XCTF:Hidden-Message[WriteUP]

使用Wireshark打开文件 分析能分析的流&#xff0c;这里直接选择UDP流 分别有两段流&#xff0c;内容都是关于物理的 和flag没啥关系&#xff0c;只能从别的方面下手 分析&#xff1a;整个数据包&#xff0c;全部由UDP协议组成 其中发送IP和接收IP固定不变&#xff0c;数据长…

[python]pyside6安装和在pycharm配置

安装命令&#xff1a; pip install PySide6 -i https://mirror.baidu.com/pypi/simple Pycharm配置Pyside6 打开Pycharm点击File -> Settings -> Tools -> External Tools&#xff0c;点击&#xff0b;。需要添加 Pyside6-Designer 、 Pyside6-UIC 和 Pyside6-rcc三…

第9章 通信动力与环境

文章目录 9.1.1 动力与环境的组成9.1.2 动力与环境的特点9.1.3 动力与环境的地位与作用9.1.4 动力与环境的基本要求9.2.1 通信电源的组成和结构9.2.2 交流供电系统1、交流供电系统的组成2、市电交流供电的质量指标3、常用高压电电器4、电力变压器5、常用低压电器6、油机发电机组…

Mac电脑采样器软Native Instruments Kontakt 7

Native Instruments Kontakt 7是一款功能强大的采样器软件&#xff0c;专为音乐制作和声音设计而设计。它提供了一个直观的界面和丰富的功能&#xff0c;使用户能够轻松地加载、编辑和演奏采样样本。Kontakt 7支持多种采样格式&#xff0c;包括WAV、AIFF、 Kontakt .nki等&…

CTFhub-phpinfo

CTFhub-Web-信息泄露-“phpinfo” 题目信息 解题过程 ctrlF搜索关键字…

JMeter定时器之同步定时器

JMeter定时器之同步定时器 1. 背景2. 目的3. 介绍4. 例子4.1单个请求4.2多个请求 1. 背景 在实际生活中大家肯定遇到过一种场景&#xff0c;就是在某一时间或某一时刻&#xff0c;某件商品进行抢购&#xff0c;相当于秒杀&#xff1b;但是用JMeter进行测试的时候&#xff0c;如…

WEB前端人机交互导论实验-实训4 DIV+CSS综合运用

1.项目1 设计《中国教育网络》杂志简介 A题目要求&#xff1a; 综合运用DIVCSS设计《中国教育网络》杂志简介页面&#xff0c;效果如下图所示: B.思路 整体思路是将《中国教育网络》杂志的简介信息以有序的方式呈现出来&#xff0c;同时通过样式和布局设置&#xff0c;使页…

erlang/OTP 平台(学习笔记)(四)

Erlang语言精要 Erlang shell 相较于日常惯用的系统&#xff0c;Erlang系统是一套更富交互性的环境。使用大部分编程语言时&#xff0c;要么把程序编译成OS可执行文件后运行&#xff0c;要么用解释器来执行一堆脚本文件或编译后的字节码文件。无论哪种情况&#xff0c;都是让…

【极光系列】SpringBoot集成Mybatis

【极光系列】SpringBoot集成Mybatis 一.gitee地址 浅夏的猫 shawsongyue 直接下载可用 https://gitee.com/shawsongyue/aurora.git 二.mysql安装教程 详细参考我的另外一遍博客&#xff1a; https://blog.csdn.net/weixin_40736233/article/details/135582926?spm1001.201…

Hologres + Flink 流式湖仓建设

Hologres Flink 流式湖仓建设 1 Flink Hologres2 实时维表 Lookup 1 Flink Hologres holo在实时数仓领域非常受欢迎&#xff0c;一般搭配flinkhologres来做实时数仓&#xff0c;中间分层用holo&#xff0c;上下游一般依赖于holo的binlog来下发数据 2 实时维表 Lookup Holo…

计算机毕业设计-----SSH高校科研管理系统平台

项目介绍 本项目包含超级管理员、管理员、教师三种角色&#xff1b; 超级管理员角色包含以下功能&#xff1a; 登录,教师管理,管理员管理等功能。 管理员角色包含以下功能&#xff1a; 登录,专业参赛奖项管理,科技论文发表管理,出版专业著作管理,科研项目立项管理,科研项目结…

USB-C一线通桌面显示器你有见过么?

新型的TYPE-C接口桌面显示器&#xff0c;宛如一位多才多艺的艺术家&#xff0c;它不仅精于视频传输&#xff0c;更在充电领域展现出无与伦比的才华。不同于传统的显示器&#xff0c;它化平凡为神奇&#xff0c;将显示器的DC电源巧妙地转换成PD协议&#xff0c;为各种设备提供稳…

APP加固原理与作用

目录 引言 正文 1. 什么是APP加固&#xff0c;加固原理是什么&#xff1f; 2. 加固的作用是什么&#xff1f;加固后是否完全不会被破解&#xff1f; 3. 如何提前规避加固新版本可能带来的风险&#xff1f; 4. 付费加固与免费加固的区别是什么&#xff1f; 白嫖的混淆加密…

docker部署Jira+配置MySQL8数据库

写在前面&#xff1a;如果你通过docker安装Jira且启动过&#xff0c;然后你现在又想使用mysql数据库&#xff0c;需要注意 你除了停掉原有容器&#xff0c;还需要删除&#xff1a;/var/lib/docker/volumes/jiraVolume/_data下的文件&#xff0c;否则启动后会无法正常使用。注意…

【EI会议征稿通知】第四届图像处理与智能控制国际学术会议(IPIC 2024)

第四届图像处理与智能控制国际学术会议&#xff08;IPIC 2024&#xff09; 2024 4th International Conference on Image Processing and Intelligent Control 2024年第四届图像处理与智能控制国际学术会议&#xff08;IPIC 2024&#xff09;将于2024年5月3日-5日在吉隆坡举…

ECharts 多季度连续显示到一个图中。

效果图 二.相关option 以下option可以复制到 echarts的编辑器 进行查看修改 const site test1; const site2 test2;const qtrlyOption function (data: any, titleText: string): any {//获取最大值 。最大最小值的目的是&#xff1a;使左右里边的所有bar使用同一个指标let …