關(guān)于單片機(jī)的軟復(fù)位
main()
{
unsignedcharcoderst[]={0xe4,0xc0,0xe0,0xc0,0xe0,0x32};//復(fù)位代碼
(*((void(*)())(rst)))();//執(zhí)行上一行代碼,將rst數(shù)組當(dāng)函數(shù)調(diào)用
}
本來我告訴他嵌入如下代碼:
clra
pushacc
pushacc
reti
結(jié)果他卻玩了前面哪一段,而數(shù)組rst[]中的內(nèi)容恰恰是上面的匯編機(jī)器碼,他的做法是將
rst數(shù)組的數(shù)據(jù)當(dāng)作代碼保存,然后采用絕對地址方式指向該數(shù)組,將該數(shù)組中的代碼當(dāng)作
函數(shù)來運(yùn)行。居然通過了!
l單片機(jī)復(fù)位的更好方法
帖子中匯編語言解釋如下:
clra//清除ACC=0
pushacc//壓0到堆棧——8位
pushacc//再壓0到堆棧——再8位
reti//返回到0地址,從而執(zhí)行。
本句的分析方法同上,但更加精煉,沒有多余的匯編語句。
軟件復(fù)位跟真正上電復(fù)位有很大差別:上電復(fù)位時(shí)大部分寄存器都有確定的復(fù)位值;軟件
復(fù)位則只相當(dāng)于從0地址開始執(zhí)行而已,寄存器不會(huì)變?yōu)榇_定的復(fù)位值。
例:修改出錯(cuò)處理如下:
ERR: CLR EA ;正確的軟件復(fù)位入口
MOV 66H,#0AAH ;重建上電標(biāo)志
MOV 67H,#55H
MOV DPTR,#ERR1 ;準(zhǔn)備第一次返回地址
PUSH DPL
PUSH DPH
RETI ;清除高級中斷激活標(biāo)志
ERR1: CLR A
PUSH ACC
PUSH ACC
RETI ;清除低級中斷激活標(biāo)志
這時(shí),必須執(zhí)行兩次RETI,才能到達(dá)0000H,以保證清除全部中斷激活標(biāo)志,達(dá)到和硬件復(fù)位相同的效果。同樣,軟件陷井也必須由下列三條指令
NOP
NOP
LJMP STAT
改成:
NOP
NOP
LJMP ERR
才能達(dá)到目的。
當(dāng)主程序受到干擾被軟件陷阱捕獲時(shí),中斷標(biāo)志并未置位,執(zhí)行ERR過程中,RETI指令等效于RET指令,同樣可以達(dá)到軟件復(fù)位的目的。有興趣的讀者可以將軟件陷阱代替死循環(huán),分別用LJMP STAT和LJMP ERR1來替代LJMP ERR,再將干擾檢測分別設(shè)在低級中斷和主程序中,實(shí)驗(yàn)結(jié)果必然證明同:只有LJMP ERR才能萬無一失地實(shí)現(xiàn)軟件復(fù)位,使系統(tǒng)擺脫干擾同,恢復(fù)正常。在MCS-51單片機(jī)的軟件復(fù)位過程中,必須連續(xù)執(zhí)行兩次中斷返回指令RETI才能確保系統(tǒng)恢復(fù)正常。
2、C語言復(fù)位
void reset (void){((void (code *) (void)) 0x0000) ();
復(fù)位程序并不能清除8051的中斷系統(tǒng)和某些8051的外圍設(shè)備,當(dāng)您在中斷程序中調(diào)用上面的軟件復(fù)位程序后,中斷將再不能觸發(fā)。因此,以上的軟復(fù)位程序不能在中斷子程序中調(diào)用。
可以使用下面的程序跳到0000H實(shí)現(xiàn)軟復(fù)位,下面的程序?qū)嶋H上是一個(gè)函數(shù)指針,指針指向了0000H地址。((void (code *) (void)) 0x0000) ();
下面的例子將實(shí)現(xiàn)軟件自復(fù)位
void reset (void)
{
((void (code *) (void)) 0x0000) ();
}
void main (void)
{
reset ();
}
你可能注意到以上的軟復(fù)位程序并不能清除8051的中斷系統(tǒng)和某些8051的外圍設(shè)備,當(dāng)您在中斷程序中調(diào)用上面的軟件復(fù)位程序后,中斷將再不能觸發(fā)。因此,以上的軟復(fù)位程序不能在中斷子程序中調(diào)用。
下面的小段匯編函數(shù)可以在中斷程序或主程序中調(diào)用,該函數(shù)將0x0000壓棧,然后通過“RETI”出棧,這將清除中斷環(huán)境并讓程序從0000H重新開始運(yùn)行。
?PR?RESET SEGMENT CODERSEG ?PR?RESET
; C prototype: void reset (void);PUBLIC resetreset: POP ACC ; pop return addressPOP ACCCLR A ; push 0 as newPUSH ACC ; return address to stackPUSH ACCRETI ; execute return of interruptEND以上程序在選擇bank 0寄存器組時(shí)工作良好,假如選擇的不是bank0寄存器組,那么可能無法獲得預(yù)料的結(jié)果。你應(yīng)該在以上的程序或啟動(dòng)代碼中加上“MOV PSW, #0”來選擇bank 0寄存器組。
我的軟復(fù)位,用了多年沒問題
clrEA
callreset
callreset;用了多少中斷優(yōu)先級,就調(diào)用幾次reset
jmpstart;跳轉(zhuǎn)至0000h
評論