取堆棧中的PC值一般采用2種方法:
1.利用PUSH/POP
;.........................
XXXXH: lcall _getPC;
YYYYH: ;................
等效為:
PUSH low YYYYH ;先壓入低8位地址,SP+1
PUSH high YYYYH ;后壓入高8位地址,SP+1
JMP _getPC;跳入_getPC后遇到RET返回YYYYH
_getPC: pop DPTH ;先彈出高8位地址,SP-1
pop DPTL ;后彈出低8位地址,SP-1
;...................................
; 假設此處立即響應某中斷
;...................................
ZZZZH:
INC SP ;恢復低8位地址low YYYYH
INC SP ;恢復高8位地址high YYYYH
;...................................
;...................................
RET ;返回YYYYH
但若在第1個INC SP前來中斷時,則由于2個POP和中斷的2個PUSH
將使原棧頂存入的返回地址YYYYH被中斷返回地址ZZZZH覆蓋!!!
以上程序等效為:
;.........................
XXXXH: lcall _getPC;
YYYYH: ;................
等效為:
PUSH low YYYYH ;先壓入低8位地址,SP+1
PUSH high YYYYH ;后壓入高8位地址,SP+1
JMP _getPC;跳入_getPC后遇到RET返回YYYYH
_getPC: POP DPTH ;先彈出高8位地址YYYYH,SP-1
POP DPTL ;后彈出低8位地址YYYYH,SP-1
;...................................
; 假設此處立即響應某中斷(INT0)
PUSH low ZZZZH;覆蓋YYYYH!!!,SP+1
PUSH high ZZZZH;覆蓋YYYYH!!!,SP+1
JMP 0x0003;跳入INT0_ISR;遇RETI返回ZZZZH,且SP-2
;...................................
ZZZZH:
INC SP ;恢復低8位地址low ZZZZH
INC SP ;恢復高8位地址high ZZZZH
;...................................
;...................................
RET ;返回ZZZZH;哈哈,等效為SP-2,JMP ZZZZH
結果很震驚---死循環!!!!!!!!!!!!!!!!!!!!!!!!!!!
所以,這種方法要特別注意!!!---也是一種中斷如何"傷人"的案例.
2.利用SP,@R0
;.........................
XXXXH: lcall _getPC;
YYYYH: ;................
_getPC: MOV R0,SP;取SP是安全的
MOV DPTH, @R0;先彈出高8位地址YYYYH
DEC R0
MOV DPTL, @R0;再彈出低8位地址YYYYH
..............................................
RET
這種方法不受中斷的影響,因為用戶沒有操作SP(沒改變其值)...
匯編程序要注意中斷搗鬼---進入死循環!
更新時間: 2006-03-09 14:19:55來源: 粵嵌教育瀏覽量:4100