大家都知道51單片機有的寄存器R0-R7共有四組。最近1年來,我在與新手朋友交流時發(fā)現(xiàn),很多朋友對寄存器組的使用時經(jīng)常出現(xiàn)問題。雖然這并不是多難的問題,但如果出現(xiàn)錯誤,也會造成很嚴重的后果。
首先介紹一下51的寄存器組:
通過設(shè)置PSW寄存器的第3位和第4位可以任意切換寄存器組。在進入中斷前,切換寄存器組,可以方便的保護原寄存器組的數(shù)據(jù)不被中斷里的語句破壞,很方便。
RS1RS0字節(jié)地址
000組寄存器00H~07H
011組寄存器08H~0FH
102組寄存器10H~17H
113組寄存器18H~1FH
RS1=PSW.4RS0=PSW.3
常見錯誤有三種:
1、為中斷函數(shù)指定了第0組寄存器
C程序:voidint0()interrupt0using0
編譯后的匯編如下:
PUSHACC
PUSHB
PUSHDPH
PUSHDPL
PUSHPSW
MOVPSW,#0x00
。。。。。。
因為main()函數(shù)使用的就是第0組寄存器,中斷程序會改變寄存器組的數(shù)據(jù)。主程序運行時,隨時都有可能產(chǎn)生中斷,等中斷返回主程序時,寄存器R0-R7的值已經(jīng)被改變了。這是非常嚴重的錯誤。而且故障時有時無,錯誤也是莫明其妙。
2、中斷優(yōu)先級不同,寄存器組號相同
C程序:voidint0()interrupt0using1//低優(yōu)先級中斷
voidT0()interrupt1using1//高優(yōu)先級中斷
因為高優(yōu)先級的中斷可以打斷正在執(zhí)行的低級中斷,轉(zhuǎn)向持行高級中斷。這就是所謂的中斷的中斷。與第1種錯語一樣,高級中斷的程序,會改變低級中斷正在使用的寄存器。
3、不寫using。嚴格的說,這樣寫不能算是錯誤。但這是相當不好的習(xí)慣。
C程序:voidint0()interrupt0
編譯后的匯編如下:
PUSHACC
PUSHB
PUSHDPH
PUSHDPL
PUSHPSW
MOVPSW,#0x00
PUSH0x00
PUSH0x01
PUSH0x02
PUSH0x03
PUSH0x04
PUSH0x05
PUSH0x06
PUSH0x07
。。。。。。
沒用using指定寄存器組,編譯器就默認分配了第0組寄存器,然后又用8條語句把第0組的R0-R7保存到棧中,退出中斷時還需要8個彈棧。這樣“笨拙”的寫法,占用了程序空間32個字節(jié)、占有堆棧8個字節(jié)。也許高級版本的編譯器會改進吧,搞單片機的還是規(guī)矩些好。
經(jīng)驗總結(jié):
1、寫中斷程序一定要用using語句指定寄存器組。第1、2、3組都可以,不能是0.
2、51單片機的中斷有兩個優(yōu)先級。一個中斷不會打斷另一個相同優(yōu)先級的中斷。這樣相同級別中斷可以使用同一個組。比如:低優(yōu)先級的中斷函數(shù)都用using1,高優(yōu)先級的中斷都用using2。這樣不會沖突。
下面是一個正常的例子:
C程序:voidint0()interrupt0using1
編譯后的匯編如下:
PUSHACC
PUSHB
PUSHDPH
PUSHDPL
PUSHPSW
MOVPSW,#0x08
。。。。。。
就是這么簡單。雖然這點事兒對老鳥不算什么問題,但新手朋友犯此錯誤的可不少。我遇到的就不下15個了。今天我寫這個貼子,也是為所有新手提個醒。到處救火不如防患于未燃。