原理圖
什么是中斷?
為微控制器編寫的簡單程序通常都可以在主函數內部完成,并且幾乎不需要使用外設。但是,大多數其他微控制器程序更復雜,需要大量代碼。當發生這種情況時,中斷會變得非常有用,但究竟什么是中斷?
想象一下,我們的微控制器需要同時做兩件事:準確跟蹤時間并使LED閃爍。我們的程序可以通過重置計時器,遞增計數器,然后等待計時器溢出來開始。完成后,我們的代碼可以使LED閃爍。雖然這有點完成工作,但是有兩個問題。 CPU花費大部分時間坐在延遲循環中,這浪費了CPU時間,并且LED的執行時間很難計算。
那么,我們如何解決這個問題呢?我們可以在計時器上使用中斷!因此,我們不是在主代碼中遞增計數器,而是將代碼轉換為處理時序的中斷服務程序。
通常,微控制器將運行LED閃爍代碼,但是一旦定時器生成中斷請求,微控制器停止LED閃爍代碼,執行定時器中斷服務程序,然后返回到LED閃爍代碼。這樣,LED閃爍代碼不會干擾我們的定時器代碼,它可以更準確(并且更容易)跟蹤時間。
AVR Core上的中斷
AVR有一個向量表,每個中斷源都跳轉到一個唯一的地址。這是非常有利的,因為我們不再需要執行比較來查看觸發了哪個中斷,這可能需要一些時間。
下表顯示了Atmega168上可用的不同中斷以及它們跳轉到的地址。程序記憶。但是,在我們使用它們之前必須配置幾個中斷選項。
從ATmega168數據表中提取
表位置
Atmega168具有允許的引導加載程序區域它可以動態地重寫自己的程序存儲器,這對固件更新很有用。因此,ISR向量表將位于內存中很重要。如果表位于引導加載程序區域中,則在啟用引導加載程序時永遠不會更新(不推薦)。
因此,如果沒有引導加載程序,則應將向量表放在內存的底部(接近地址0x0000),但如果使用引導加載程序,則應將向量表移動到引導加載程序上方。這可以通過改變MCUCR寄存器中的幾個位來輕松完成。
如果IVSEL = 0,則ISR位于向量表的起始,否則ISR駐留在引導加載程序中。現在,將其保留為0,因為我們沒有使用引導加載程序
如果IVCE = 1,則執行ISR切換。暫時保留為0
中斷啟用位
每個中斷源(I/O引腳,外設等)都有關聯中斷使能位。與PIC類似,STATUS寄存器中有一個全局中斷使能位,需要將其設置為允許中斷工作。要找出這些中斷標志所在的位置,需要參考數據手冊中的特定外設章節。
例如,我們將在定時器0上使用溢出中斷,所以如果我們看一下定時器0在章節中,我們發現中斷使能位位于TIMSK0寄存器(第89頁)中,稱為TOIE0。需要將此位設置為1才能觸發定時器溢出。該寄存器還有另外兩個中斷源,A匹配溢出和B匹配溢出,這對PWM功能很有用(將來會介紹)。
注意,設置我在SREG中的位不是使用SREG本身,而是使用函數sei();設置I位和cei();清除I位。
在WinAVR中編寫ISR
所以我們現在明白需要啟用中斷才能啟動,但我們如何使用C和WINAVR編譯器編寫?答案很簡單:我們使用特殊保留字ISR并傳遞中斷名稱參數來告訴編譯器哪個中斷函數處理。注意我們需要包含中斷頭文件,否則中斷函數將不起作用!
#include
ISR(TIMER0_OVF_vect)
{
// Interestingly, the AVR automatically clears interrupt flags.。。.unlike the PIC
// Put your code here
}
簡單閃爍示例
在這個例子中,ATmega168會使連接到PD0的LED頻繁閃爍,其中閃爍的速率受到控制通過定時器0但是,您可能會注意到主功能為空,并且LED在定時器溢出中斷服務程序(ISR)內閃爍。這意味著我們可以在while循環中放入我們想要的任何代碼,并且該代碼不會阻止中斷運行。
/*
* AVR Interrupt.c
*
* Created: 09/01/2018
* Author : RobinLaptop
*/
// These are really useful macros that help to get rid of unreadable bit masking code
#define setBit(reg, bit) (reg = reg | (1 《《 bit))
#define clearBit(reg, bit) (reg = reg & ~(1 《《 bit))
#define toggleBit(reg, bit) (reg = reg ^ (1 《《 bit))
#define clearFlag(reg, bit) (reg = reg | (1 《《 bit))
#include
#include
ISR(TIMER0_OVF_vect)
{
// Interestingly, the AVR automatically clears interrupt flags =) 。。..unlike the PIC =(
// Toggle the LED (PD0 , Pin 2)
toggleBit(PORTD, PD0);
}
int main(void)
{
// Initialize Registers
clearBit(TCCR0A, WGM00); // Configure WGM to be 0x00 for normal mode
clearBit(TCCR0A, WGM01);
clearBit(TCCR0B, WGM02);
setBit(TCCR0B, CS00); // Configure clock source to be clock io at 1024 pre-scale
clearBit(TCCR0B, CS01);
setBit(TCCR0B, CS02);
DDRD = 0xFF; // Make PORT D and output
sei(); // Enable interrupts
setBit(TIMSK0, TOIE0); // Enable the timer interrupt
while (1)
{
// Put any code you want here
// It should not affect the interrupt service routine!
}
}
結論
本教程僅涵蓋單個中斷,即定時器0溢出中斷,但它清楚地表明中斷是非常強大。如果使用得當,您可以擁有一個系統,它可以在信號到達時立即響應并暫停主代碼。這可以用來做很多事情,包括多任務處理,不同外圍設備的多重處理,以及創建實時代碼!
-
中斷
+關注
關注
5文章
900瀏覽量
41645
發布評論請先 登錄
相關推薦
評論