软件复位问题(跳转PC)

#pragma asm  
ljmp 0
#pragma endasm      //C插入汇编实现

深入--单片机软复位(PC跳转)

对函数指针熟悉吗?熟悉一切都容易了!
好书推荐《C陷阱与缺陷》《C专家编程》看看就上个档次就不是菜鸟了,呵呵

(*(void(*)())0) ();   //就是它了!
晕了吗?没晕,不错不错,大哥,你扎这厉害呢!

((void(*)())0) ();   //如果这样写呢!

(**(void(*)())0) ();   //如果这样写呢!

有区别吗?看下面例子

#include

void fun()
{
}

int main()
{
        printf("%p      %p      %p/n", &fun, fun, *fun);
        return 0;
}
看看结果是否大吃一惊  // 没有,大哥你扎这厉害呀

假设fp是个float指针,声明如下
float * fp;
把0强制转换成一个float指针(把变量fp去掉就可以了)
(float *)0;

类似:
假设fp是函数指针为void类型的函数的指针),声明如下:
void (*fp)();      
把0强制转换成该函数指针(变量fp去掉就可以了)
(void(*)())0

最后用(void(*)())0代替fp,从而得到调用的用法

(*(void(*)())0) ();


单片机我是这样写的
((void(code *)(void))0x0000)();   //简单,证实可以
(*(void(code *)(void))0x0000)();  //这样行吗

是不是看的很麻烦,typedef来帮忙呀(为复杂的声明定义一个新的简单的别名),这不这个大哥来了

typedef void (code *pfunction)(void);
//相当于pfunction变成了一个函数指针的类型(和float含义一样,只不过flaot表示的是一个指向浮点的指针,而pfunction表示一个指向函数的指针)

((function)0xE800)();   //这样行吗
(*(function)0xE800)(); //这样行吗

下面是我在ARM下用过的
typedef void (*pfunction)(void);

void FMI_Jump(void)
{   
  pfunction jump;
  jump=(pfunction)(0x80000);
  jump();
  
}

跳转PC,都是用来在线升级时候,什么什么你没用过,大哥你扎这幸运呢,想当初我带着烧写器做火车,一个电信机房一个电信机房的跑,痛苦呀!

注意:跳转复位PC危险!!!!它只是PC从000开始,内部寄存器并未回到复位值,所以程序初始化一定要彻底,一切系统资源都要初始化,哪怕未用!!!建议最好关狗,等RESET复位较安全(什么什么我用PC跳转没遇到问题啊?大哥你扎这幸运呢!^_^)

 

上次搞一个51IAP时,就是软件复位寄存器搞得不够彻底,导致程序运行一段时间后莫名其妙死掉...后来干脆不用软件去复位寄存器了,直接设置一个寄存器让系统复位(或者打开内狗),然后再判断该执行用户程序还是IAP程序,不再倒塌,问题解决...

当程序飞后,内部模块可能都乱了~~~
假定你根本未使用T2,那么你就不会考虑T2的初始化问题。

假定软件复位没进行ET2=0,TR2=0,且未未在T2Isr处加reti
EA=1后,若跑飞时ET2=1,TR2=1,后果可知
~~~
所以忠告大家:

初始化一定要彻底,一切系统资源都要初始化,哪怕未用!!!
有硬件看门狗时,最好while(1)自毁~~~
这样可对自己的有用模块再初始化,切记:所有中断向量表(程序)无用的都应该用reti.

你可能感兴趣的:(单片机技术)