你的浏览器版本过低,可能导致网站不能正常访问!
为了你能正常使用网站功能,请使用这些浏览器。

STM32远程升级(基于串口本地升级与WiFi通信远程升级)  

[复制链接]
Y-D 发布时间:2017-6-9 10:45
/***************************************************
*注意要想二次升级必须要在用户APP中将 UPDATA_FLAG[0]='0'并保存在flash中
*设置完成后将单片机重启,重启完成将升级包通过串口发送给单片机即可升级完成
*同时注意用户APP中需要在程序的开头添加SCB->VTOR = FLASH_BASE | 0x10000;语句
*本实验程序为考虑后期利用8266升级所以设置波特率为 115200
*注意本程序代码的升级包限定最小20个字节才为升级包否则为控制命令
****************************************************/
//主函数部分
#include "led.h"
#include "delay.h"
#include "sys.h"
#include "usart.h"
#include "stmflash.h"
#include "iap.h"
#include "time.h"
u16  UPDATA_FLAG[1];
int main(void)
{               
        u16 oldcount=0;                                //老的串口接收数据值
        u16 applenth=0;                                //接收到的app代码长度
        static u8 START_UPDATA=0;
  NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2); //设置NVIC中断分组2:2位抢占优先级,2位响应优先级
        uart_init(256000);        //串口初始化为256000
        delay_init();                            //延时初始化
        LED_Init();
        TIM2_Init(9,7199);  //升级延时定时器
        STMFLASH_Read(0X08071000,(u16*)UPDATA_FLAG,1); //读取更新标志   如果此标志位为:'1'代表程序已经更新了 如果为 '0'代表程序在执行IAP等待接收升级包
        if(UPDATA_FLAG[0]=='0')
        {
                //等待发送更新包,说明程序从用户程序进行了重启
                //提示发送更新包
                printf("Please send update package....\r\n");
        }
        else if(UPDATA_FLAG[0]=='1')
        {
                printf("Loading...\r\n");
        }
        while(1)
        {
                LED1=!LED1;  //LED闪烁指示
                delay_ms(100);
                 if(USART_RX_CNT)
                {
                        if(oldcount==USART_RX_CNT)//新周期内,没有收到任何数据,认为本次数据接收完成.
                        {
                                applenth=USART_RX_CNT;
                                oldcount=0;
                                USART_RX_CNT=0;
                                if(applenth>20) //此处强行规定发过来的升级包必须大于20个字节否则不予升级
                                {
                                        printf("USER  CODE  RXC\r\n");  //提示升级包接收完成
                                        printf("CODE  SIZE:%dBytes\r\n",applenth);
                                        START_UPDATA=1; //此标志为1时可以进入升级
                                        UP_DATA_Time_3000ms=0;
                                }
                        }else oldcount=USART_RX_CNT;                       
                }
                if(START_UPDATA==1&&UP_DATA_Time_3000ms>=3000)
                {
                        UP_DATA_Time_3000ms=0;
                        START_UPDATA=2;
                }
                if(START_UPDATA==2)
                {
                        if(applenth)
                        {
                                printf("Start  updating...\r\n");       
                                if(((*(vu32*)(0X20001000+4))&0xFF000000)==0x08000000)//判断是否为0X08XXXXXX.
                                {         
                                        UPDATA_FLAG[0]='1';
                                        STMFLASH_Write(0X08071000,(u16*)UPDATA_FLAG,1);  //允许更新标志
                                        STMFLASH_Read(0X08071000,(u16*)UPDATA_FLAG,1); //读取更新标志
                                        iap_write_appbin(FLASH_APP1_ADDR,USART_RX_BUF,applenth);//更新FLASH代码   
                                        printf("Updated  firmware\r\n");       
                                        printf("Resart  \r\n");
                                        delay_ms(100);                //延时一小会儿等待上一帧数据发送完成
                                        //软件重启CPU
                                        __disable_fault_irq();
                                        NVIC_SystemReset();
                                }
                                else
                                {   
                                        printf("Illegal  Code!\r\n");
                                }
                        }
                        START_UPDATA=0;                                       
                }
/**********************************************************************
*有升级包进行升级
*此处的标志位需要在用户APP中进行清零后才可进行下一次的更新否则每次上电都会直接跳转到用户APP执行
***********************************************************************/       
                if(UPDATA_FLAG[0]=='1')
                {
                        if(((*(vu32*)(FLASH_APP1_ADDR+4))&0xFF000000)==0x08000000)//判断是否为0X08XXXXXX.  此处测试一下是否会多次跳入
                        {       
                                iap_load_app(FLASH_APP1_ADDR);//执行FLASH APP代码
                        }
                }
        }             
}

//API部分
#include "sys.h"
#include "delay.h"
#include "usart.h"
#include "stmflash.h"
#include "iap.h"
iapfun jump2app;
u16 iapbuf[1024];   
//appxaddr:应用程序的起始地址
//appbuf:应用程序CODE.
//appsize:应用程序大小(字节).
void iap_write_appbin(u32 appxaddr,u8 *appbuf,u32 appsize)
{
        u16 t;
        u16 i=0;
        u16 temp;
        u32 fwaddr=appxaddr;//当前写入的地址
        u8 *dfu=appbuf;
        for(t=0;t<appsize;t+=2)
        {                                                    
                temp=(u16)dfu[1]<<8;
                temp+=(u16)dfu[0];          
                dfu+=2;//偏移2个字节
                iapbuf[i++]=temp;            
                if(i==1024)
                {
                        i=0;
                        STMFLASH_Write(fwaddr,iapbuf,1024);       
                        fwaddr+=2048;//偏移2048  16=2*8.所以要乘以2.
                }
        }
        if(i)STMFLASH_Write(fwaddr,iapbuf,i);//将最后的一些内容字节写进去.  
}
//跳转到应用程序段
//appxaddr:用户代码起始地址.
void iap_load_app(u32 appxaddr)
{
        if(((*(vu32*)appxaddr)&0x2FFE0000)==0x20000000)        //检查栈顶地址是否合法.
        {
                jump2app=(iapfun)*(vu32*)(appxaddr+4);                //用户代码区第二个字为程序开始地址(复位地址)               
                MSR_MSP(*(vu32*)appxaddr);                                        //初始化APP堆栈指针(用户代码区的第一个字用于存放栈顶地址)
                jump2app();                                                                        //跳转到APP.
        }
}               

配套上位机.rar

下载

897.98 KB, 下载次数: 1125

本地升级和远程升级上位机

评分

参与人数 2 ST金币 +4 收起 理由
yyawl -1 很给力!
MrJiu + 5 很给力!

查看全部评分

1 收藏 11 评论36 发布时间:2017-6-9 10:45

举报

36个回答
kylongmu 回答时间:2018-5-21 13:20:13
可以把wifi改为蓝牙串口模块,用手机app扫蓝牙来升级,为了安全最好用RSA验证一下身份。升级文件传输可以用ymodem协议。
wkc20180808 回答时间:2019-11-5 15:34:53
通俗易懂,可以在此基础上加上xmoder 的协议  ,保证了传输的可靠性,安全性。
mojiaxia 回答时间:2018-5-21 09:58:13
楼主你好,能否把这个参考单片机例程发给我参考一下,我的油箱是103572498@qq.com
zhao.zhao 回答时间:2017-6-10 11:49:06
谢谢楼主
MrJiu 回答时间:2017-6-10 13:56:23
要是能更加详细以及有一套完整的帖子,那就更好啦!!!!!!
Y-D 回答时间:2017-6-15 10:43:09
后面有时间了在更新的详细些。
zero99 回答时间:2017-6-26 13:18:49
学习了
huaiqiao 回答时间:2017-6-30 11:47:17
多谢楼主分享。

问下楼主,wifi模块是用esp8266吗?
A.K. 回答时间:2017-7-3 16:36:13
谢谢楼主的分享
cpu-340177 回答时间:2017-7-18 09:53:29

谢谢楼主的分享
奔跑的蜗牛321 回答时间:2017-10-10 10:12:29
能否把单片机程序也上传一下
zhao.zhao 回答时间:2017-11-27 10:42:42
谢谢
a838899 回答时间:2017-12-11 11:10:15
这个厉害啊 不错啊
西点钟灵毓秀 回答时间:2018-2-5 15:58:42
不错,这样可以空中升级了
zkevin8879 回答时间:2018-3-12 10:13:20
谢谢楼主了,这几天正在想这事呢
琦子 回答时间:2018-3-12 10:25:26
好文章啊 支持一下
大陶 回答时间:2018-3-12 11:14:49
学习了 谢谢楼主
123下一页

所属标签

关于
我们是谁
投资者关系
意法半导体可持续发展举措
创新与技术
意法半导体官网
联系我们
联系ST分支机构
寻找销售人员和分销渠道
社区
媒体中心
活动与培训
隐私策略
隐私策略
Cookies管理
行使您的权利
官方最新发布
STM32N6 AI生态系统
STM32MCU,MPU高性能GUI
ST ACEPACK电源模块
意法半导体生物传感器
STM32Cube扩展软件包
关注我们
st-img 微信公众号
st-img 手机版