新聞中心

EEPW首頁 > 嵌入式系統(tǒng) > 設(shè)計(jì)應(yīng)用 > 軟件復(fù)位問題(跳轉(zhuǎn)PC)

軟件復(fù)位問題(跳轉(zhuǎn)PC)

作者: 時間:2016-11-09 來源:網(wǎng)絡(luò) 收藏
#pragma asm
ljmp 0
#pragma endasm //C插入?yún)R編實(shí)現(xiàn)
深入--單片機(jī)軟復(fù)位(PC跳轉(zhuǎn))

對函數(shù)指針熟悉嗎?熟悉一切都容易了!
好書推薦《C陷阱與缺陷》《C專家編程》看看就上個檔次就不是菜鳥了,呵呵

(*(void(*)())0) ();//就是它了!
暈了嗎?沒暈,不錯不錯,大哥,你扎這厲害呢!

((void(*)())0) ();//如果這樣寫呢!

(**(void(*)())0) ();//如果這樣寫呢!

有區(qū)別嗎?看下面例子

#include

void fun()
{
}

int main()
{
printf("%p%p%p/n", &fun, fun, *fun);
return 0;
}
看看結(jié)果是否大吃一驚// 沒有,大哥你扎這厲害呀

假設(shè)fp是個float指針,聲明如下
float * fp;
把0強(qiáng)制轉(zhuǎn)換成一個float指針(把變量fp去掉就可以了)
(float *)0;

類似:
假設(shè)fp是函數(shù)指針為void類型的函數(shù)的指針),聲明如下:
void (*fp)();
把0強(qiáng)制轉(zhuǎn)換成該函數(shù)指針(變量fp去掉就可以了)
(void(*)())0

最后用(void(*)())0代替fp,從而得到調(diào)用的用法

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


單片機(jī)我是這樣寫的
((void(code *)(void))0x0000)();//簡單,證實(shí)可以
(*(void(code *)(void))0x0000)();//這樣行嗎

是不是看的很麻煩,typedef來幫忙呀(為復(fù)雜的聲明定義一個新的簡單的別名),這不這個大哥來了

typedef void (code *pfunction)(void);
//相當(dāng)于pfunction變成了一個函數(shù)指針的類型(和float含義一樣,只不過flaot表示的是一個指向浮點(diǎn)的指針,而pfunction表示一個指向函數(shù)的指針)

((function)0xE800)();//這樣行嗎
(*(function)0xE800)(); //這樣行嗎

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

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

}

跳轉(zhuǎn)PC,都是用來在線升級時候,什么什么你沒用過,大哥你扎這幸運(yùn)呢,想當(dāng)初我?guī)е鵁龑懫髯龌疖?,一個電信機(jī)房一個電信機(jī)房的跑,痛苦呀!

注意:跳轉(zhuǎn)復(fù)位PC危險?。。?!它只是PC從000開始,內(nèi)部寄存器并未回到復(fù)位值,所以程序初始化一定要徹底,一切系統(tǒng)資源都要初始化,哪怕未用!??!建議最好關(guān)狗,等RESET復(fù)位較安全(什么什么我用PC跳轉(zhuǎn)沒遇到問題???大哥你扎這幸運(yùn)呢!^_^)





上次搞一個51的IAP時,就是軟件復(fù)位寄存器搞得不夠徹底,導(dǎo)致程序運(yùn)行一段時間后莫名其妙死掉...后來干脆不用軟件去復(fù)位寄存器了,直接設(shè)置一個寄存器讓系統(tǒng)復(fù)位(或者打開內(nèi)狗),然后再判斷該執(zhí)行用戶程序還是IAP程序,不再倒塌,問題解決...
當(dāng)程序飛后,內(nèi)部模塊可能都亂了~~~
假定你根本未使用T2,那么你就不會考慮T2的初始化問題。
假定軟件復(fù)位沒進(jìn)行ET2=0,TR2=0,且未未在T2Isr處加reti
則EA=1后,若跑飛時ET2=1,TR2=1,后果可知~~~
所以忠告大家:
初始化一定要徹底,一切系統(tǒng)資源都要初始化,哪怕未用?。?!
硬件看門狗時,最好while(1)自毀~~~
這樣可對自己的有用模塊再初始化,切記:所有中斷向量表(程序)無用的都應(yīng)該用reti.


評論


技術(shù)專區(qū)

關(guān)閉