單片機(jī)復(fù)位程序
今天在網(wǎng)上看到這么一個(gè)單片機(jī)復(fù)位的程序,這個(gè)程序據(jù)說是一個(gè)大三的學(xué)生寫出來的,不錯(cuò),寫的很有一定的道理,其C編程也達(dá)到了一定程度了【小盒子我還是很佩服這個(gè)人的】。下面我們來看看這個(gè)代碼:
本文引用地址:http://www.ex-cimer.com/article/201611/317165.htmvoid main(void)
{
unsignedcharcoderst[]={0xe4,0xc0,0xe0,0xc0,0xe0,0x32};//復(fù)位代碼
(*((void(*)())(rst)))();//執(zhí)行上一行代碼,將rst數(shù)組當(dāng)函數(shù)調(diào)用
}
第一句定義一個(gè)數(shù)組rst[],數(shù)組內(nèi)數(shù)據(jù)就是完成復(fù)位功能的匯編機(jī)器碼,具體對應(yīng)關(guān)系為:
clra==0xe4、pushacc==0xc0,0xe0、reti==0x32
可以看出其程序起到復(fù)位的作用,完全就是匯編機(jī)器碼的功勞。
而單片機(jī)復(fù)位的更好方法
clra//清除ACC=0
pushacc//壓0到堆棧——8位
pushacc//再壓0到堆棧——再8位
reti//返回到0地址,從新執(zhí)行。
這種復(fù)位方法比較麻煩,更加簡單的復(fù)位寫法是(摘自《C缺陷與陷阱》):
(*(void(*)())0)();
看過上面更簡單的復(fù)位方法,讓我們多加考慮一下,為什么要寫成0?別的不行嗎?換成別的后會是什么樣的效果呢?抱著這個(gè)想法,我親自經(jīng)過KEIL V2.4.0編譯后的匯編程序:
可以看出若將(*(void(*)())0)();
改成(*(void(*)())3)();
則程序會跳轉(zhuǎn)到main()函數(shù)開始,避開startup文件的初始化……
只所以我說的是會從main()開始,是因?yàn)槲铱催^編譯后的匯編文件,找到main的實(shí)際物理地址而已,否則我也不會寫成3了。呵呵……下面就是編譯后的匯編結(jié)果
C:0x0003 E4 CLR A
C:0x0004 F508 MOV 0x08,A
C:0x0006 F509 MOV 0x09,A
14: while(1) {
15: if(i == 10) {
16: //( *( ( void (*)( ) ) (rst) ) )(); // 執(zhí)行上一行代碼,將rst數(shù)組當(dāng)函數(shù)調(diào)用
C:0x0008 E509 MOV A,0x09
C:0x000A 640A XRL A,#0x0A
C:0x000C 4508 ORL A,0x08
C:0x000E 7005 JNZ C:0015
17: ( *( ( void (*)( ) ) (3) ) )(); // 執(zhí)行上一行代碼,將rst數(shù)組當(dāng)函數(shù)調(diào)用
C:0x0010 120003LCALL main(C:0003)
18: } else {
C:0x0013 80F3 SJMP C:0008
19: i++;
C:0x0015 0509 INC 0x09
C:0x0017 E509 MOV A,0x09
C:0x0019 70ED JNZ C:0008
C:0x001B 0508 INC 0x08
20: }
為了進(jìn)行給大家一個(gè)很好的比較,從視覺上得到一定的感覺,我又再次將3改回成0,大家看看編譯后的匯編結(jié)果是什么樣子的;
下面的代碼是函數(shù)(*(void(*)())0)(); 這個(gè)編譯后的結(jié)果
C:0x0003 E4 CLR A
C:0x0004 F508 MOV 0x08,A
C:0x0006 F509 MOV 0x09,A
14: while(1) {
15: if(i == 10) {
16: //( *( ( void (*)( ) ) (rst) ) )(); // 執(zhí)行上一行代碼,將rst數(shù)組當(dāng)函數(shù)調(diào)用
C:0x0008 E509 MOV A,0x09
C:0x000A 640A XRL A,#0x0A
C:0x000C 4508 ORL A,0x08
C:0x000E 7005 JNZ C:0015
17: ( *( ( void (*)( ) ) (0) ) )(); // 執(zhí)行上一行代碼,將rst數(shù)組當(dāng)函數(shù)調(diào)用
C:0x0010 120000LCALL C_STARTUP(C:0000)
18: } else {
C:0x0013 80F3 SJMP C:0008
19: i++;
C:0x0015 0509 INC 0x09
C:0x0017 E509 MOV A,0x09
C:0x0019 70ED JNZ C:0008
C:0x001B 0508 INC 0x08
20: }
請大家注意紅色的部分。
評論