所有的中檔系列PIC單片機,PORTB端口最高的4個引腳(RB7~RB4)在設(shè)為輸入模式時,當(dāng)輸入電平由高到低或由低到高發(fā)生變化時,可以讓單片機產(chǎn)生中斷。這就是通常所說的引腳狀態(tài)變化中斷。
在設(shè)計引腳中斷程序時,有三個需要特別注意的地方。一是,在清除P0RTB中斷標(biāo)志位RBIF之前,必須安排一條必不可少的,以PORTB端口數(shù)據(jù)寄存器PORTB為源寄存器的讀操作指令。放置這一指令的目的有時并不只是為了讀取有用的數(shù)據(jù),而是為了取消狀態(tài)變化的硬件信號,以便順利清除RBIF標(biāo)志位,為下一次中斷做好準備。二是,由于端口PORTB是引腳電子變化中斷,即無論引腳出現(xiàn)上升沿還是下降沿都會產(chǎn)生中斷請求,所以必須處理好不需要的虛假中斷。三是,一般都利用PIC單片機的引腳功能來檢測按鍵,所以必須處理好按鍵消抖的問題。
2 引腳中斷程序設(shè)計
在主程序里先設(shè)置有關(guān)的寄存器。
◇設(shè)置TRISB寄存器,使RB7~RB4相關(guān)的引腳處于輸入狀態(tài);
◇如果需要弱上拉,通過OPTION_REG的第7位設(shè)置;
◇RBIF=O;
◇RBIE=1;
◇GIF=1。
響應(yīng)狀態(tài)變化后的中斷服務(wù)程序。
◇檢查RBIF是否為l,為l則是引腳變化引起的中斷;
◇調(diào)用延時程序,延時20~30 ms,目的是為了按鍵去抖;
◇判斷是引腳出現(xiàn)上升沿還是下降沿引起的中斷;
◇調(diào)用按鍵處理程序;
◇讀PORTB口的值,取消狀態(tài)變化的硬件信號;
◇清除RBIF標(biāo)志。
筆者認為上面程序設(shè)計最大的問題是在中斷程序里調(diào)用延時程序。大家知道,中檔PIC單片機只有8層深度的硬件堆棧,在中斷里調(diào)用于程序出現(xiàn)極易堆棧溢出的情況。另外,PIC單片機中斷程序人口只有一個,在響應(yīng)中斷的請求時,PIC單片機就會自動把全局中斷的使能位(INTCON的第7位GIF)清除,這樣其他中斷就暫時不能被響應(yīng)(此時,如果別的中斷發(fā)出的中斷請求,標(biāo)志位將一直保留著),直到這個中斷程序退出后才會得到響應(yīng)。這就要求我們設(shè)計中斷程序的時候必須盡量短,避免調(diào)用子程序,更不要在中斷里進行復(fù)雜的運算。
下面給出筆者設(shè)計程序時的思路。
當(dāng)引腳狀態(tài)變化引起中斷時,在中斷子程序里首先判斷引起中斷的原因是不是我們需要的變化引起的中斷。如果是,不要在這里延時,而是設(shè)置一個標(biāo)志位,接著清除中斷標(biāo)志,退出中斷。中斷程序如下:
else if((RBIE&RBlF)==1){ //如果引腳變化引起中斷
if(RB4==0){ //RB4上的按鈕接地
key=1; //按鍵標(biāo)志位置位
}
RBIF=0; //清除引腳中斷標(biāo)志位
}
其中,if(RB4==0)語句相當(dāng)于讀取了PORTB端口數(shù)據(jù)寄存器,取消了狀態(tài)變化的硬件信號。
下面詳細介紹怎么樣進行按鍵去抖。
首先,在定時器中斷里設(shè)置一個lms的時間基準標(biāo)志位“SYSlms”,每到lms,“SYSlms”便置位。程序如下:
unsigned char count;
if((ToIE&TOIF)==1){ //定時器中斷
TMRO+=0x09; //每250μs中斷一次
if(count==4){
count=0;
SYSlms=l; //系統(tǒng)時間標(biāo)志
couot++;
}
T0IF=0; //清除時鐘中斷標(biāo)志位
}
有了這個時間基準,便可以在主程序里進行按鍵去抖處理了。為了更好地利用這個時間基準,定義一個消息標(biāo)志SYSTime,筆者把它稱作時間消息。為了讓這個消息有自我發(fā)布和自我消失的功能.定義了如下一個宏:
bit SYSTime;
#defincTimeEnahle()SYSTime=0,if(SYSlms){SYSTime=l;SYSlms=0;)
可以把TimeEnable()放到主程序死循環(huán)的任何地方,每當(dāng)程序執(zhí)行這個宏,SYSTime就會清零,這就是標(biāo)志位的自我消失.如果在定時器時間基準標(biāo)志位SYSlms已經(jīng)置位的話,SYSTime就會置1,這樣別的程序就可以利用這個時間消息了,這就是消息的自我發(fā)布。下面就是利用這個時間消息來進行按鍵延時去抖的,首先看一下按鍵掃描子程序;
void seaakey(){
unsigned char KeyTime,KeyTask;//定義任務(wù)時間參數(shù)、
//任務(wù)參數(shù)
switch(KeyTask){
case0:if(key){
KeyTime=30; //準備延時30 ms
KeyTask++; //準備好下一個任務(wù)
kcy=0;
}
break;
case I:KeyTime--; //延時30 ms
if(KeyTime==0)Key+ask++;
break;
case2;if(RB4==o){
//調(diào)按鍵處理程序
KeyTask=0;
}
else KeyTask=0;//退出任務(wù)
break;
}
}
在主程序的死循環(huán)中這樣用:
while(1){
TimeEnable();
If(SYSTime==1){scankey();)
//在此可以添加其他程序
只有有時問消息的時候才執(zhí)行按鍵掃描程序??梢钥吹?,進入掃描程序執(zhí)行第一次的時候,程序首先判斷按鍵標(biāo)志位有沒有置位,置位的話(也就是有按鍵按下的話),任務(wù)時間參數(shù)(KeyTime)賦值為30,這是延時30ms,去抖,當(dāng)然你也可以設(shè)置為其他的時間值;同時任務(wù)參數(shù) (KeyTask)加1。1ms后,再進入掃描程序,這個時候掃描程序執(zhí)行casel的語句,這樣30次后(延時了30ms),任務(wù)參數(shù)(KeyTask)加1,值為2。lms后,再進入掃描程序,將執(zhí)行case 2的語句,首先在這里再次判斷是不是按鍵還在按下,如果是就調(diào)按鍵的處理程序,如果不是。就退出按鍵掃描程序。在這里,還可以加入按鍵是否抬起的判斷程序。
這樣設(shè)計的引腳變化程序,CPU開銷小,效率高,不會出現(xiàn)堆淺溢出的問題,提高了系統(tǒng)的實時性。