在C51中如何實(shí)現(xiàn)軟復(fù)位?
可以定義一個(gè)指向復(fù)位向量(0x0000)的函數(shù)指針,然后在C程序中需要軟復(fù)位的地方調(diào)用該函數(shù):
((void (code *) (void)) 0x0000) ();
例如,以下程序不斷地復(fù)位:
void reset (void)
{
((void (code *) (void)) 0x0000) ();
}
void main (void)
{
reset ();
}
也許你會(huì)注意到前面所提到的復(fù)位程序并不恢復(fù)8051的中斷系統(tǒng),它也不去復(fù)位任何的8051外設(shè)。若上述代碼在一個(gè)中斷例程中執(zhí)行,則8051
會(huì)阻止同級(jí)中斷的產(chǎn)生。因此,在中斷服務(wù)例程中不能使用這個(gè)復(fù)位程序。
以下的一小段匯編過(guò)程同樣實(shí)現(xiàn)軟復(fù)位,而且它可以在中斷或者主程序中調(diào)用。它通過(guò)將返回地址(0x0000)壓入堆棧并執(zhí)行一條RETI指令(從
中斷中返回)來(lái)實(shí)現(xiàn)復(fù)位目的。這個(gè)匯編過(guò)程會(huì)清掉中斷狀態(tài)然后從0000H開(kāi)始執(zhí)行程序:
?PR?RESET SEGMENT CODE
RSEG ?PR?RESET
; C prototype: void reset (void);
PUBLIC reset
reset: POP ACC ; pop return address
POP ACC
CLR A ; push 0 as new
PUSH ACC ; return address to stack
PUSH ACC
RETI ; execute return of interrupt
END
若調(diào)用該復(fù)位過(guò)程時(shí)選擇的不是寄存器組0,程序可能達(dá)不到期望的結(jié)果。這時(shí)應(yīng)該在上面的匯編過(guò)程中或者在啟動(dòng)代碼中添加一條選擇寄存器組0的指令:
MOV PSW, #0