1 簡述 $!yzk e|hiD 所有的中檔系列PIC單片機,PORTB端口最高的4個引腳(RB7~RB4)在設為輸入模式時,當輸入電平由高到低或由低到高發(fā)生變化時,可以讓單片機產(chǎn)生中斷。這就是通常所說的引腳狀態(tài)變化中斷。 J mES X&-: *]R.{,j-I 在設計引腳中斷程序時,有三個需要特別注意的地方。一是,在清除PORTB中斷標志位RBIF之前,必須安排一條必不可少的,以PORTB端口數(shù)據(jù)寄存器PORTB為源寄存器的讀操作指令。放置這一指令的目的有時并不只是為了讀取有用的數(shù)據(jù),而是為了取消狀態(tài)變化的硬件信號,以便順利清除RBIF標志位,為下一次中斷做好準備。二是,由于端口PORTB是引腳電平變化中斷,即無論引腳出現(xiàn)上升沿還是下降沿都會產(chǎn)生中斷請求,所以必須處理好不需要的虛假中斷。三是,一般都利用PIC單片機的引腳功能來檢測按鍵,所以必須處理好按鍵消抖的問題。 v/-lKBrB' >#59=H 2 引腳中斷程序設計 [iq W:X'I >#_Gz"|{ 在主程序里先設置有關的寄存器。 e}4xd.G Z=3)j}jrP ◇ 設置TRISB寄存器,使RB7~RB4相關的引腳處于輸入狀態(tài); wwvD ◇ 如果需要弱上拉,通過OPTION_REG的第7位設置; GC?Uy2C8 ◇ RBIF=0; ecop< ◇ RBIE=1; 1N/R*G ◇ GIF=1。 ,$_1,e (y&JMZD3 響應狀態(tài)變化后的中斷服務程序。 7\0GAm% qX=&v}x $ ◇ 檢查RBIF是否為1,為1則是引腳變化引起的中斷; 5Xwt~>U~[ ◇ 調(diào)用延時程序,延時20~30 ms,目的是為了按鍵去抖; K2m= Al%9# ◇ 判斷是引腳出現(xiàn)上升沿還是下降沿引起的中斷; BPScZ)|! ◇ 調(diào)用按鍵處理程序; $Wr=Q8P ◇ 讀PORTB口的值,取消狀態(tài)變化的硬件信號; @>T (iUD) ◇ 清除RBIF標志。 b(-oh$ @|E1AJz0b 筆者認為上面程序設計最大的問題是在中斷程序里調(diào)用延時程序。大家知道,中檔PIC單片機只有8層深度的硬件堆棧,在中斷里調(diào)用子程序出現(xiàn)極易堆棧溢出的情況。另外,PIC單片機中斷程序入口只有一個,在響應中斷的請求時,PIC單片機就會自動把全局中斷的使能位(INTCON的第7位GIF)清除,這樣其他中斷就暫時不能被響應(此時,如果別的中斷發(fā)出的中斷請求,標志位將一直保留著),直到這個中斷程序退出后才會得到響應。這就要求我們設計中斷程序的時候必須盡量短,避免調(diào)用子程序,更不要在中斷里進行復雜的運算。 tD,ARy aM/RJP 下面給出筆者設計程序時的思路。 gQ6UtM4 mVwB7mG 當引腳狀態(tài)變化引起中斷時,在中斷子程序里首先判斷引起中斷的原因是不是我們需要的變化引起的中斷。如果是,不要在這里延時,而是設置一個標志位,接著清除中斷標志,退出中斷。中斷程序如下: ()l"5Xt ; a^F else if ( (RBIE & RBIF) == 1) {//如果引腳變化引起中斷 C:PWUV if (RB4==0) {//RB4上的按鈕接地 z~c5f key=1;//按鍵標志位置位 gd%]* X(9 } R9c+o|~ RBIF=0;//清除引腳中斷標志位 [6U3 9Yd } }.|k,ZI(g Wpcs/_Oy 其中,if (RB4==0)語句相當于讀取了PORTB端口數(shù)據(jù)寄存器,取消了狀態(tài)變化的硬件信號。 #a`IvX' zc do+" 下面詳細介紹怎么樣進行按鍵去抖。 IQMv]on!X 56"#0UX 首先,在定時器中斷里設置一個1 ms的時間基準標志位“SYS1ms”,每到1ms,“SYS1ms”便置位。程序如下: ,/\`U" o}.^jMWU! unsigned char count; cljUo`M if ( (T0IE & T0IF) == 1) {//定時器中斷 K@}6%5^ TMR0 += 0x09; //每250 μs中斷一次 [U_}\ijm if(count==4) { v(YYc$N<3 count=0; p:R1ke*y SYS1ms=1;//系統(tǒng)時間標志 8~'khzD } 5:J8SO5 count++; pK]UZBK{% T0IF = 0;//清除時鐘中斷標志位 ohV]f } b'=EX!:Qe "; <%sun 有了這個時間基準,便可以在主程序里進行按鍵去抖處理了。為了更好地利用這個時間基準,定義一個消息標志SYSTime,筆者把它稱作時間消息。為了讓這個消息有自我發(fā)布和自我消失的功能,定義了如下一個宏: fgY188 Cf CJ_~J6 bit SYSTime; Ao:B)3NyD #define TimeEnable()SYSTime=0; if(SYS1ms){ SYSTime=1; SYS1ms=0; } (@\|Wp9 yZJ+yCW 可以把TimeEnable()放到主程序死循環(huán)的任何地方,每當程序執(zhí)行這個宏,SYSTime就會清零,這就是標志位的自我消失。如果在定時器時間基準標志位SYS1ms已經(jīng)置位的話,SYSTime就會置1,這樣別的程序就可以利用這個時間消息了,這就是消息的自我發(fā)布。下面就是利用這個時間消息來進行按鍵延時去抖的,首先看一下按鍵掃描子程序: }rq=Wz$. ;zG? X:{A void scankey() { K)I/5bfoy: unsigned char KeyTime,KeyTask; //定義任務時間參數(shù)、 e:X+I{E} //任務參數(shù) /sgd9%~G switch(KeyTask) { $(D!G'* case 0: if(key) { wc3[kER KeyTime=30; //準備延時30 ms _V9 )0aVD KeyTask++; //準備好下一個任務 f9x:F key=0; '?1g$T[3 } CQ?qbTa break; 8 +X/K7f case 1:KeyTime--;//延時30 ms wvyJrW_ if(KeyTime==0) KeyTask++; Bj2Z6 break; Pq VG case 2:if(RB4==0) { j&;%8 [c] //調(diào)按鍵處理程序 d!$)Wm KeyTask=0; bD|5B0A' } wiQEY else KeyTask=0; //退出任務 o2iie_3 break; gZ&g)~'' } y1T<iB^kx } <[d@{[z\ Xy"bu6W 在主程序的死循環(huán)中這樣用: F6&QuAt}; Y"<\~AN while(1) { Kcv]q -/4 TimeEnable(); !I/L@kD If(SYSTime==1) {scankey();} `uK+g29%p //在此可以添加其他程序 Mh~JWD|) } vFhESSP8m mvFT:@7 只有有時間消息的時候才執(zhí)行按鍵掃描程序?梢钥吹,進入掃描程序執(zhí)行第一次的時候,程序首先判斷按鍵標志位有沒有置位,置位的話(也就是有按鍵按下的話),任務時間參數(shù)(KeyTime)賦值為30,這是延時30 ms去抖,當然你也可以設置為其他的時間值;同時任務參數(shù)(KeyTask)加1。1 ms后,再進入掃描程序,這個時候掃描程序執(zhí)行case 1的語句,這樣30次后(延時了30 ms),任務參數(shù)(KeyTask)加1,值為2。1 ms后,再進入掃描程序,將執(zhí)行case 2的語句,首先在這里再次判斷是不是按鍵還在按下,如果是就調(diào)按鍵的處理程序,如果不是,就退出按鍵掃描程序。在這里,還可以加入按鍵是否抬起的判斷程序。 Dm\-a)Tus &S- aT/X 這樣設計的引腳變化程序,CPU開銷小,效率高,不會出現(xiàn)堆淺溢出的問題,提高了系統(tǒng)的實時性。
|
|