[应用相关] 关于STM32 IAP

[复制链接]
 楼主| sensitivity 发表于 2020-7-24 20:43 | 显示全部楼层
if(rbCanRead(&pRb)>1)

因为一次性要往Flash里面写16位数据,所以才会判断数据个数大于一个的时候再往里面写.

虽然现在程序没有问题,但是我还在想如果程序的个数是奇数个就完啦!但是好像程序的个数总是偶数个....其实可以在判断接收完成的里面

做一下判断,如果数据还残留一个,那就写进去....
 楼主| sensitivity 发表于 2020-7-24 20:44 | 显示全部楼层
好,那就看一下判断接收完程序

if(Usart1Flage ==1)//数据接收完成{

addr2= FLASH_APP2_ADDR;//存储数据的地址Usart1Flage =0;//清零if(((*(vu32*)(FLASH_APP2_ADDR+4))&0xFF000000)==0x08000000)//判断是否为0X08XXXXXX.{

printf("准备执行APP代码!!\r\n");

UserDataAddr= FLASH_DATA_ADDR;//存储其余的数据地址ReadDat16 =0x55;//写入标志告诉IAP程序有可更新的用户程序STMFLASH_Write(UserDataAddr,&ReadDat16,1);

UserDataAddr+=2;

printf("写入0x55标志!!\r\n");

ReadDat16= (u16)((Usart1RecCntCopy>>16)&0xffff);//存储接收到多少数据高位STMFLASH_Write(UserDataAddr,&ReadDat16,1);

UserDataAddr+=2;

ReadDat16= (u16)(Usart1RecCntCopy&0xffff);//存储接收到多少数据低位STMFLASH_Write(UserDataAddr,&ReadDat16,1);

UserDataAddr+=2;

Usart1RecCntCopy=0;

printf("开始复位重启!!\r\n");

Reset_MCU();

}else{

printf("非FLASH应用程序,无法执行!\r\n");

}
//printf("Cnt=%d\r\n",Usart1RecCntCopy);
//for(i=0;i>8)<=15)
//{
//printf("0%x ",ReadDat16>>8);
//}
//else
//{
//printf("%x ",ReadDat16>>8);
//}
//}
//addr1 = FLASH_APP1_ADDR;
//for(i=0;i<40;i++)
//{
//STMFLASH_Erase(addr1,1024);
//擦除FLASH_APP1_ADDR地址以及以上40页
//addr1 +=2048;
//}
//addr1 = FLASH_APP1_ADDR;
}
 楼主| sensitivity 发表于 2020-7-24 20:45 | 显示全部楼层
后边屏蔽的是测试的时候,看一下写入的数据,然后和源数据对比一下,看一下写入的对不对

if(((*(vu32*)(FLASH_APP2_ADDR+4))&0xFF000000)==0x08000000)

这句话
 楼主| sensitivity 发表于 2020-7-24 20:46 | 显示全部楼层
先问一个问题,怎么知道接收过来的是用户程序呢????要是别的数据怎么办???,,必须有一个判断依据才行对吧!!

就必须找到用户程序中永恒不变的变量....
 楼主| sensitivity 发表于 2020-7-24 20:46 | 显示全部楼层
然后呢,我是看别人的程序说,数据的第一个4个字节为栈顶地址,数据的第二个4字节为复位中断向量的入口地址
371345f1ad81f3eb55.png
 楼主| sensitivity 发表于 2020-7-24 20:47 | 显示全部楼层
FLASH_APP2_ADDR+4指针就移动到了IAP升级程序的E9或者说电压电流采集程序的D5上

(*(vu32*)(FLASH_APP2_ADDR+4))然后强制型的转成32位的,然后取出来,就是IAP升级程序的E9 20 00 08

或者说电压电流采集程序的D5 7E 00 08
 楼主| sensitivity 发表于 2020-7-24 20:47 | 显示全部楼层
还有一件事就是STM32是小端模式,,,,所谓小端模式就是低位在低地址,高位在高地址

举个例子

把60000存到STM32的Flash的,60000转换成16进制是EA60  EA是高8位,60是低八位,,存到Flash里面就是60EA这样存的

60存到了低地址,EA存到了高地址,,,,,当然有小端模式就有大端模式,,,大端模式就是低位在高地址,高位在低地址,电脑就是这样...
 楼主| sensitivity 发表于 2020-7-24 20:48 | 显示全部楼层
说到这里就要说一下共用体

typedef union Resolver_I{longData;charData_Table[4];

}Resolver_iData;

typedef union Resolver_f{floatData;charData_Table[4];}Resolver_fData;

Resolver_iData Resolver_7758;  //解析7758数据

Resolver_fData Resolver_Usart; //解析串口数据
 楼主| sensitivity 发表于 2020-7-24 20:49 | 显示全部楼层
一个整形数据快速的转换成16进制存到数组里面

Resolver_7758.Data = 60000;

那么Resolver_7758.Data_Table[0] = 0x60;

Resolver_7758.Data_Table[1] = 0xEA;

Resolver_7758.Data_Table[2] = 0x00;

Resolver_7758.Data_Table[3] = 0x00;
 楼主| sensitivity 发表于 2020-7-24 20:49 | 显示全部楼层
一个浮点型的数据转换成16进制存到数组里面--其实也是按照IEEE754规约来计算的

Resolver_Usart.Data = 220.5;

那么Resolver_Usart.Data_Table[0] = 0x00;

Resolver_Usart.Data_Table[1] = 0x80;

Resolver_Usart.Data_Table[2] = 0x5C;

Resolver_Usart.Data_Table[3] = 0x43;
 楼主| sensitivity 发表于 2020-7-24 20:50 | 显示全部楼层
所以说上面的数据取出来就是08 00 20 E9然后&0xFF000000 肯定就等于 0x08000000啦

其实这样还有一个原因是因为32的地址是从08000000开始的,复位中断向量的地址的最高两位肯定是08啦!!!!
susceptibility 发表于 2020-7-24 20:52 | 显示全部楼层
然后还有个if(((*(vu32*)appxaddr)&0x2FFE0000)==0x20000000) //检查栈顶地址是否合法.

这个......难道栈顶地址的最高位就是20......以后等看到相关的资料再说吧
susceptibility 发表于 2020-7-24 20:53 | 显示全部楼层
有人这样介绍的--仔细看,细细品味....

http://blog.csdn.net/yx_l128125/article/details/12992773
susceptibility 发表于 2020-7-24 20:53 | 显示全部楼层
对了说一下中断向量表

就从我的MSP430的**中摘抄过来
15155f1ad9cc73155.png
susceptibility 发表于 2020-7-24 20:54 | 显示全部楼层
susceptibility 发表于 2020-7-24 20:54 | 显示全部楼层
原文地址

http://www.cnblogs.com/yangfengwu/p/6064129.html
susceptibility 发表于 2020-7-24 20:57 | 显示全部楼层
突然想起来一句话,知识是相通的....

32也有中断向量表,就像上面的430的似的,

只不过呢!32的中断函数的入口地址是可以改变的!!!(F0好像固定,但是看过一篇**好像也能通过某种方式改改)
susceptibility 发表于 2020-7-24 20:58 | 显示全部楼层
我们写入IAP程序后单片机内部就有了一个中断向量表,这个呢是其内部一开始的固定的,

如果跳转到用户程序,用户程序肯定会有自己的中断函数吧!如串口,,定时器,,等等,,,,如果不改变中断向量表的话!!!产生的中断

岂不是跑到了IAP那边去了,IAP那边有自己的中断函数,,,乱了,彻底乱了,,,,,所以必须得让中断向量表改变改变,好让自己产生的

susceptibility 发表于 2020-7-24 20:58 | 显示全部楼层
中断,执行自己的中断函数......

那就加一句话,或者修改一个地方
101065f1adae9a6d39.png
825395f1adaf895239.png
susceptibility 发表于 2020-7-24 20:59 | 显示全部楼层
再看我的用户程序的一个地方,,感觉自己罗嗦了

  1. if(AppFlage ==1)//接收到更新程序{if(rbCanRead(&pRb)>1)

  2. {

  3. rbRead(&pRb, &ReadDat,2);//读取两个数据ReadDat16 = (u16)ReadDat[1]<<8;

  4. ReadDat16= ReadDat16|ReadDat[0];

  5. STMFLASH_Write(addr2,&ReadDat16,1);

  6. addr2+=2;

  7. }

  8. }elseif(AppFlage ==0)

  9. {if(rbCanRead(&pRb)==8)

  10. {

  11. rbRead(&pRb, &TestData,8);//读取数据if(TestData[3] ==0x20&& TestData[7] ==0x08)//判断是否是更新程序{

  12. AppFlage=1;//要更新程序for(i=0;i<4;i++)//先写入这八位数据{

  13. ReadDat16= (u16)TestData[(i<<1)+1]<<8;

  14. ReadDat16= ReadDat16|TestData[i<<1];

  15. STMFLASH_Write(addr2,&ReadDat16,1);

  16. addr2+=2;

  17. }

  18. }

  19. }

  20. }
您需要登录后才可以回帖 登录 | 注册

本版积分规则

快速回复 在线客服 返回列表 返回顶部
快速回复 在线客服 返回列表 返回顶部