1、目的
本篇學習筆記我們主要來了解ME32F030的中斷系統,首先通過對ME32F030終端系統和中斷控制存器進行學習,最后通過實際的GPIO端口中斷實例掌握中斷函數的編程實現來加深掌握ME32F030中斷系統的編程方法。
2、ME32F030中斷概述
中斷是單片機系統重要的組成部分,使單片機能夠快速的對事件請求做出響應。ME32F030靠內部的嵌套向量中斷控制器(NVIC)來進行中斷的調度,它是 Cortex?-M0 內核的一部分。它可以讓 CPU 以最短的時間對中斷作出反應。主要的特征有:
? 較短的中斷響應延遲.
? 處理系統異常和外設中斷.
? 支持 32 個中斷向量.
? 四種可編程的中斷響應優先級別.
? 產生軟件中斷.
? 可配置的不可屏蔽中斷源會有詳細的說明。
3、ME32F030嵌套向量中斷控制器(NVIC)
嵌套向量中斷控制器(NVIC)負責著整個MCU的中斷管理,除了管理我們常用的外設中斷源,還包括非屏蔽中斷源的管理。具體管理的中斷源及其中斷序號,其實在CMSDK_CM0.h中定義好了,通過程序定義對具體管理的中斷源一目了然,定義的內容如下圖所示。
中斷源
ME32F030的中斷系統是由一系列的NVIC寄存器組成的,這些寄存器可用于中斷 IRQ0~IRQ31, 包括中斷使能,等待和優先級等操作。如果中斷被允許,并且相應中斷掛起被設置,NVIC 將會根據中斷優先級觸發中斷。反之,中斷被禁止,中斷源只會改變中斷掛起狀態,而 NVIC 不會對中斷源信號采取任何動作,不論任何中斷優先級。具體的NVIC寄存器組如表格所示:
NVIC寄存器列表
2-1 中斷允許寄存器
中斷允許寄存器(ISER)用于使能中斷設置,同時可返回當前允許中斷設置。需要注意的是,對該寄存器寫0是無效的,是不能禁止中斷的。要禁止中斷,就需要下面介紹的中斷禁止寄存器。
2-2 中斷禁止寄存器
有使能中斷的設置,相應的就會有禁止中斷設置,這就是中斷禁止寄存器的作用。對寄存器進行寫1操作,就可以禁止中斷設置。。需要注意的是,對該寄存器寫0是無效的。對寄存器進行讀操作,同返回當前禁止中斷設置。
2-3 中斷掛起寄存器
當有中斷事件發生時,中斷掛起寄存器(ISPR)中對應的中斷位就會置位 。此時讀取寄存器就可以判斷具體的中斷源。同時我們也可以向寄存器的中斷位寫1,來強制中斷進入掛起狀態。
2-4 清除中斷掛起寄存器
當MCU響應了中斷請求,并且執行完對應的中斷子程序后,MCU便會返回斷點處繼續運行。但在返回前需要通過清除中斷掛起寄存器(ICPR),來清除對應的中斷掛起,這樣當次的中斷流程算是完整結束。
2-5 中斷優先級寄存器
如同我們做事情有輕重緩急之分,單片機對中斷的處理也有“輕重緩急”。具體就是靠8組中斷優先級寄存器IPR0~7來實現。每組寄存器對應4個中斷源的優先級。這樣剛好決定了中斷0~中斷31的優先級。
每一組的中斷優先級寄存器IPRn的每個字節最高兩位決定優先級,因此有0~3共4個優先級可以選擇,越低的值表示優先級越高,當優先級更高的中斷發生時,高優先級的中斷會打斷低優先級中斷。如果是同優先級中斷,則并不會打斷當前中斷,而是依次響應中斷。中斷優先級寄存器IPRn如圖所示:
這里介紹個快速的方法來計算中斷 M 的 IPR 寄存器號:
? 計算對應的 IPR 寄存器號, N, N = M / 4
? 計算 IPR 寄存器內的字節偏移量 M % 4, 其中:
– 字節偏移量 0 對應寄存器的位 7:0
– 字節偏移量 1 對應寄存器的位 15:8
– 字節偏移量 2 對應寄存器的位 23:16
– 字節偏移量 3 對應寄存器的位 31:24
4、ME32F030端口中斷例程
本篇中我們首先講解了ME32F030的GPIO中斷系統,然后又介紹了嵌套向量中斷控制器(NVIC)的原理。
最后,我們還是要通過具體的實例來把我們學到的理論知識應用到實際的例子中。我們將兩者結合起來做個小實驗,測試程序的代碼如下:
unsignedintuiCnt=0;//端口反轉次數 intmain(void) { PA->DIR_b.DIR0=1;//PA_0設置為輸出口 PA->IS_b.ISENSE0=0;//PA_0設置為沿觸發 PA->IBE_b.IBE0=1;//PA_0上升沿和下降沿都觸發中斷 PA->IC_b.CLR0=1;//PA_0中斷標志位清除 PA->IE_b.MASK0=1;//PA_0中斷使能 PB->DIR_b.DIR9=1;//PB_9設置為輸出口 NVIC_EnableIRQ(PA_IRQn);//使能PA_IRQ中斷 lcd_init();//LCD液晶初始化 while(1) { uiCnt++;//端口反轉次數加1 PA->NOT_b.NOT0=1;//PA_0輸出取反 SYS_DelaymS(500); if(uiCnt==20)//當反轉20次時 { PA->IE_b.MASK0=0;//PA_0中斷關閉 } } } //PA_IRQ中斷子程序 voidPA_IRQHandler(void) { PB->NOT_b.NOT9=1;//PB_9(LED燈)輸出取反 PA->IC_b.CLR0=1;//清除PA_0中斷位 //LCD顯示中斷發生的次數 LCD->MEMMAP1=lcd[uiCnt/10]|(lcd[uiCnt%10]<<16); }
測試程序是通過PA_0端口輸出反轉,來產生下降沿和上升沿。但同時它的端口中斷功能是被使能的,因此可以通過輸出電平來“觸發”自己的中斷。在中斷服務子程序中,LED小燈端口輸出取反來進行點亮和熄滅,同時加入了LCD段碼液晶來顯示中斷發生的次數。在程序全速運行的過程中,當端口輸出反轉20次之后,會關閉端口的中斷功能。接下來下載并仿真例程來進行說明。
程序下載并仿真后,先在程序這兩處打上斷點。然后用快捷鍵F5全速運行,程序首先會運行到第78行處的斷點,這時端口還沒有進行輸出反轉。接下來用快捷鍵F10單步運行觀察。
仿真1
F10單步運行后,發現程序已經跳轉到了PA_IRQ中斷服務子程序中,繼續F10單步運行并觀察執行每一步的現象,直到把中斷服務子程序走完。
仿真2
中斷服務子程序內的代碼全部運行后的效果如圖所示,首先LED小燈的端口輸出取反,把LED小燈給點亮后(下次再進中斷會輸出取反熄滅,依次往復)。LCD段碼液晶顯示01,這說明發生了1次中斷。
仿真結果
通過單步仿真我們清楚了中斷發生后的處理流程,接下來就可以把之前打的兩個斷點取消掉,然后在83行的位置打上一個斷點,隨后F5全速運行程序,等待程序停到斷點處。在等待的過程中,LED小燈保持閃爍,LCD段碼液晶上的數字一直在自加。當程序停到斷點處后,LCD段碼液晶顯示為20。繼續單步運行后,端口PA_0的中斷功能就被關閉了。
仿真3
關閉中斷后,再次全速運行程序。我們發現小燈不再閃爍了,段碼液晶顯示的數字也不再自加。這是因為我們已經把端口中斷關閉掉了,雖然uiCnt還在自加。但是已經進不了中斷子程序去更新顯示。因此依舊停留顯示在20。我們不妨把uiCnt添加到Watch窗口中來看一下,添加方法如圖所示。雙擊ucCnt變量名,選中后右鍵選擇“Add uiCnt to”,“Watch 1”,這樣就添加到Watch1窗口中了。
仿真4
通過Watch1窗口看到端口已經反轉37次了,但LCD液晶已經停留在20。這也說明中斷確實被關閉了,因此液晶一直沒能更新顯示。
仿真5
仿真結果2
來源:敏矽MCU
-
控制器
+關注
關注
112文章
16332瀏覽量
177812 -
端口
+關注
關注
4文章
963瀏覽量
32052 -
Cortex-M0
+關注
關注
4文章
124瀏覽量
38675 -
外部中斷
+關注
關注
1文章
131瀏覽量
15816
發布評論請先 登錄
相關推薦
評論