welcom ! Handel home

2014年6月22日 星期日

GPIO 外部中斷範例

//STM32----4----EXTI
//配置的目標是:PA15、PA13、PA0為按鍵,PA8,PD2為LED,按鍵進入中斷,相應的LED亮滅。
// PA15&PA13  interrupt => EXTI15_10_IRQHandler , PA0 interrupt => EXTI0_IRQHandler()

void EXTI_Configuration() {
  EXTI_InitTypeDef EXTI_InitStructure;  
  NVIC_InitTypeDef NVIC_InitStructure;  

  //清空中斷標誌
  EXTI_ClearITPendingBit(EXTI_Line13);      
  EXTI_ClearITPendingBit(EXTI_Line15);      
  EXTI_ClearITPendingBit(EXTI_Line0);    

//(1)管腳配置    
  LED_Init();    
  KEY_Init();

  //(2)外部中斷Pin配置,選擇中斷線管腳PA13、PA15、PA0
  GPIO_EXTILineConfig(GPIO_PortSourceGPIOA, GPIO_PinSource13);      
  GPIO_EXTILineConfig(GPIO_PortSourceGPIOA, GPIO_PinSource15);      
  GPIO_EXTILineConfig(GPIO_PortSourceGPIOA, GPIO_PinSource0);  

  //(3)對中斷線進行中斷設置  
  EXTI_InitStructure.EXTI_Line = EXTI_Line13 | EXTI_Line15 ;//選擇中斷線路PA13/15      
  EXTI_InitStructure.EXTI_Mode = EXTI_Mode_Interrupt;       //設置為插斷要求,非事件請求      
  EXTI_InitStructure.EXTI_Trigger = EXTI_Trigger_Falling ;  //設置中斷觸發方式為上下降沿觸發      
  EXTI_InitStructure.EXTI_LineCmd = ENABLE;                 //外部中斷使能      
  EXTI_Init(&EXTI_InitStructure);                           //初始化中斷

  EXTI_InitStructure.EXTI_Line = EXTI_Line0;                //選擇中斷線路PA0
  EXTI_InitStructure.EXTI_Mode = EXTI_Mode_Interrupt;       //設置為插斷要求,非事件請求      
  EXTI_InitStructure.EXTI_Trigger = EXTI_Trigger_Rising ;   //設置中斷觸發方式為上下降沿觸發      
  EXTI_InitStructure.EXTI_LineCmd = ENABLE;                 //外部中斷線使能      
  EXTI_Init(&EXTI_InitStructure);
  //EXTI_GenerateSWInterrupt(EXTI_Line0| EXTI_Line13 | EXTI_Line15 );  //中斷線使能、中斷結構體初始化、以及設置軟中斷綜合起來才啟用了中斷    

  //(4)NVIC配置
  NVIC_PriorityGroupConfig(NVIC_PriorityGroup_0);           //選擇中斷分組2

  NVIC_InitStructure.NVIC_IRQChannel = EXTI15_10_IRQChannel; //選擇中斷通道2  
  NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0;          //搶佔式中斷優先順序設置為0        
  NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0;                       //回應式中斷優先順序設置為0        
  NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;                     //使能中斷        
  NVIC_Init(&NVIC_InitStructure);

  NVIC_InitStructure.NVIC_IRQChannel = EXTI0_IRQChannel;          //選擇中斷通道2  
  NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0;          //搶佔式中斷優先順序設置為0        
  NVIC_InitStructure.NVIC_IRQChannelSubPriority = 1;                       //回應式中斷優先順序設置為0        
  NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;                     //使能中斷        
  NVIC_Init(&NVIC_InitStructure);
}

void LED_Init( ) {
   GPIO_InitTypeDef GPIO_InitStructure;
   GPIO_InitStructure.GPIO_Pin = GPIO_Pin_8;                      //選擇PA8
   GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;//管腳頻率為50MHZ  
   GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP; //模式為推挽輸出  
   GPIO_Init(GPIOA, &GPIO_InitStructure);                    //初始化GPIOA寄存器  
 
   GPIO_InitStructure.GPIO_Pin = GPIO_Pin_2;                      //選擇PD2
   GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;//管腳頻率為50MHZ  
   GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP; //設置推挽輸出  
   GPIO_Init(GPIOD , &GPIO_InitStructure);                   //初始化GPIOD寄存器
 
   GPIO_WriteBit(GPIOA,GPIO_Pin_8,Bit_SET);         //PA8 = 1
   GPIO_WriteBit(GPIOD,GPIO_Pin_2,Bit_SET);         //PD2 = 1  
}  

void KEY_Init() {
/*PA13、PA15管口引腳的配置**下拉輸入******************************/
    GPIO_InitTypeDef GPIO_InitStructure;

    GPIO_InitStructure.GPIO_Pin=GPIO_Pin_13|GPIO_Pin_15;
    GPIO_InitStructure.GPIO_Mode=GPIO_Mode_IPU;  //pull low
    PIO_InitStructure.GPIO_Speed=GPIO_Speed_50MHz;
    GPIO_Init(GPIOA, &GPIO_InitStructure);
/*PA0管口引腳的配置**上拉輸入************************************/
    GPIO_InitStructure.GPIO_Pin=GPIO_Pin_0;
    GPIO_InitStructure.GPIO_Mode=GPIO_Mode_IPD;  //pull high
    GPIO_InitStructure.GPIO_Speed=GPIO_Speed_50MHz;
    GPIO_Init(GPIOA, &GPIO_InitStructure);
}

void EXTI15_10_IRQHandler(void) {
    //以EXTI_Line15 and 13 Flag status 判定中斷源
    if(EXTI_GetITStatus(EXTI_Line15)==RESET) {
        if((GPIO_ReadOutputDataBit(GPIOD,GPIO_Pin_2)==0) )
            LED1_Off();
        else
            LED1_On();
    }
    else if(EXTI_GetITStatus(EXTI_Line13)==RESET){
        if((GPIO_ReadOutputDataBit(GPIOA,GPIO_Pin_8)==0) )
            LED0_Off();    
        else
            LED0_On();
    }  
    // 清除中斷源 Flag
    EXTI_ClearITPendingBit(EXTI_Line13);
    EXTI_ClearITPendingBit(EXTI_Line15);  
}

void EXTI0_IRQHandler(void) {
    //delay_ms(10);
    if(EXTI_GetITStatus(EXTI_Line0)!=RESET) {
        if((GPIO_ReadOutputDataBit(GPIOA,GPIO_Pin_8)==0) )
            LED0_Off();
        else
            LED0_On();

        if((GPIO_ReadOutputDataBit(GPIOD,GPIO_Pin_2)==0) )
            LED1_Off();
        else
            LED1_On();          
    }
    EXTI_ClearITPendingBit(EXTI_Line0);  
}


/*===========================================================
STM32學習筆記:外部中斷的使用 
中斷對於開發嵌入式系統來講的地位絕對是毋庸置疑的,在C51單片機時代,一共只有5個中斷,其中2個外部中斷,2個定時/計數器中斷和一個串口中斷,
但是在STM32中,中斷數量大大增加,而且中斷的設置也更加複雜。今天就將來探討一下關於STM32中的中斷系統。 
1 基本概念 
ARM Coetex-M3內核共支援256個中斷,其中16個內部中斷,240個外部中斷和可程式設計的256級中斷優先順序的設置。
STM32目前支持的中斷共84個(16個內部+68個外部),還有16級可程式設計的中斷優先順序的設置,僅使用中斷優先順序設置8bit中的高4位。 STM32可支援68個中斷通道,已經固定分配給相應的外部設備,每個中斷通道都具備自己的中斷優先順序控制位元組PRI_n(8位元,但是STM32中只使用4位,高4位有效),每4個通道的8位中斷優先順序控制字構成一個32位的優先順序寄存器。( 8bit x 4 = 32bit )
68個通道的優先順序控制字至少構成17個32位的優先順序寄存器。 

4bit的中斷優先順序可以分成2組,從高位看,前面定義的是搶佔式優先順序,後面是響應優先順序

按照這種分組,4bit一共可以分成5組 
第0組:所有4bit用於指定回應優先順序; 
第1組:最高1位用於指定搶佔式優先順序,後面3位用於指定響應優先順序; 
第2組:最高2位用於指定搶佔式優先順序,後面2位用於指定響應優先順序; 
第3組:最高3位用於指定搶佔式優先順序,後面1位用於指定響應優先順序; 
第4組:所有4位用於指定搶佔式優先順序。 

所謂搶佔式優先順序和回應優先順序,他們之間的關係是:
具有高搶佔式優先順序的中斷可以在具有低搶佔式優先順序的中斷處理過程中被回應,即中斷嵌套。 
當兩個中斷源的搶佔式優先順序相同時,這兩個中斷將沒有嵌套關係,當一個中斷到來後,如果正在處理另一個中斷,這個後到來的中斷就要等到前一個中斷處理完之後才能被處理。
如果這兩個中斷同時到達,則中斷控制器根據他們的回應優先順序高低來決定先處理哪一個?
如果他們的搶佔式優先順序和回應優先順序都相等,則根據他們在中斷表中的排位元順序決定先處理哪一個。
每一個中斷源都必須定義2個優先順序。 
有幾點需要注意的是: 
1)如果指定的搶佔式優先順序別或回應優先順序別超出了選定的優先順序分組所限定的範圍,將可能得到意想不到的結果; 
2)搶佔式優先順序別相同的中斷源之間沒有嵌套關係; 
3)如果某個中斷源被指定為某個搶佔式優先順序別,又沒有其它中斷源處於同一個搶佔式優先順序別,則可以為這個中斷源指定任意有效的回應優先順序別。 

2 GPIO外部中斷 
STM32中,每一個GPIO都可以觸發一個外部中斷,但是,GPIO的中斷是以組位元一個單位的,
同組間的外部中斷同一時間只能使用一個。比如說,PA0,PB0,PC0,PD0,PE0,PF0,PG0這些為1組,
如果我們使用PA0作為外部中斷源,那麼別的就不能夠再使用了,在此情況下,我們智慧使用類似於PB1,
PC2這種末端序號不同的外部中斷源。
每一組使用一個中斷標誌EXTIx。

EXTI0 - EXTI4這5個外部中斷有著自己的單獨的中斷響應函數,
EXTI5-9共用一個中斷響應函數,EXTI10-15共用一個中斷回應函數。 

對於中斷的控制,STM32有一個專用的管理機構:NVIC。
中斷的使能,掛起,優先順序,活動等等部都是NVIC在管理的。
因為我學習STM32重點在於如何開發程式,所以內部的一些東西,在此我就不詳細說明了

3 程式開發 
其實上面那些基本概念和知識只是對STM32的中斷系統有一個大概的認識,用程式說話將會更能夠加深如何使用中斷。使用外部中斷的基本步驟如下: 
1. 設置好相應的時鐘; 
2. 設置相應的中斷; 
3. IO口初始化; 

4. 把相應的IO口設置為中斷線路(要在設置外部中斷之前)並初始化; 

5. 在選擇的中斷通道的回應函數中中斷函數。 
由於我用的奮鬥開發板沒有引出相應的晶片引腳,所以只能用按鍵來觸發相應的中斷。根據原理圖,K1/K2/K3連接的是PC5/PC2/PC3,因此我將用EXTI5/EXTI2/EXTI3三個外部中斷。
PB5/PD6/PD3分別連接了三個LED燈。中斷的效果是按下按鍵,相應的LED燈將會被點亮。 

1. 設置相應的時鐘 首先需要打開GPIOB、GPIOC和GPIOE(因為按鍵另外一端連接的是PE口)。
然後由於是要用於觸發中斷,所以還需要打開GPIO複用的時鐘。相應的函數在GPIO的學習筆記中有了詳細了解釋。詳細代碼如下: 
*/

void RCC_cfg() {  
    //打開PE PD PC PB埠時鐘,並且打開複用時鐘 
    RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOE | RCC_APB2Periph_GPIOC | 
    RCC_APB2Periph_GPIOD | RCC_APB2Periph_GPIOB | RCC_APB2Periph_AFIO, ENABLE); 

//設置相應的時鐘所需要的RCC函數在stm32f10x_rcc.c中,所以要在工程中添加此檔。 

//2. 設置好相應的中斷 
//設置相應的中斷實際上就是設置NVIC,在STM32的固件庫中有一個結構體NVIC_InitTypeDef,裡面有相應的標誌位元設置,然後再用NVIC_Init()函數進行初始化。詳細代碼如下: 

void NVIC_cfg() {   
    NVIC_InitTypeDef NVIC_InitStructure; 
    NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);   
    //選擇中斷分組2   
    NVIC_InitStructure.NVIC_IRQChannel = EXTI2_IRQChannel;      //選擇中斷通道2 
    NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0;   //搶佔式中斷優先順序設置為0   
    NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0;   //回應式中斷優先順序設置為0   
    NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;    

    //使能中斷 
    NVIC_Init(&NVIC_InitStructure);    
    NVIC_InitStructure.NVIC_IRQChannel = EXTI3_IRQChannel;   
    //選擇中斷通道3 
    NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 1; //搶佔式中斷優先順序設置為1   
    NVIC_InitStructure.NVIC_IRQChannelSubPriority = 1;  //回應式中斷優先順序設置為1   
    NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;    

    //使能中斷 
    NVIC_Init(&NVIC_InitStructure);
    NVIC_InitStructure.NVIC_IRQChannel = EXTI9_5_IRQChannel;  //選擇中斷通道5 
    NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 2; //搶佔式中斷優先順序設置為2   
    NVIC_InitStructure.NVIC_IRQChannelSubPriority = 2;  //回應式中斷優先順序設置為2   
    NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;  

    //使能中斷 
    NVIC_Init(&NVIC_InitStructure); 


/*由於有3個中斷,因此根據前文所述,需要有3個bit來指定搶佔優先順序,所以選擇第2組。
又由於EXTI5-9共用一個中斷響應函數,所以EXTI5選擇的中斷通道是EXTI9_5_IRQChannel,詳細資訊可以在標頭檔中查詢得到。
用到的NVIC相關的庫函數在stm32f10x_nivc.c中,需要將此檔複製並添加到工程中。具體位置可以查看關於GPIO的筆記。這段代碼編譯起來沒有任何問題,但是在連結的時候就會報錯,需要把STM32F10xR.LIB加入工程中,具體位置在…\Keil\ARM\RV31\LIB\ST\STM32F10xR.LIB。 
*/

//3. IO口初始化 
void IO_cfg() {   
    GPIO_InitTypeDef GPIO_InitStructure;   
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_2; 

 //選擇引腳2 
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;   //輸出頻率最大50MHz   
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;    //帶上拉電阻輸出 
    GPIO_Init(GPIOE,&GPIO_InitStructure);   
    GPIO_ResetBits(GPIOE,GPIO_Pin_2);  
  

 //將PE.2引腳設置為低電平輸出   
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_2 | GPIO_Pin_3 | GPIO_Pin_5; //選擇引腳2 3 5   
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING; //選擇輸入模式為浮空輸入   
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;   //輸出頻率最大50MHz  
    GPIO_Init(GPIOC,&GPIO_InitStructure);  

 //設置PC.2/PC.3/PC.5 
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_3 | GPIO_Pin_6;   
 //選擇引腳3 6 
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;   //輸出頻率最大50MHz   
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;   
 //帶上拉電阻輸出 
    GPIO_Init(GPIOD,&GPIO_InitStructure);     
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_5; 

 //選擇引腳5 
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;   //輸出頻率最大50MHz   
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;    //帶上拉電阻輸出 
    GPIO_Init(GPIOB,&GPIO_InitStructure); 


//其中連接外部中斷的引腳需要設置為輸入狀態,而連接LED的引腳需要設置為輸出狀態,初始化PE.2是為了使得按鍵的另外一端輸出低電平。GPIO中的函數在stm32f10x_gpio.c中。 

//4. 把相應的IO口設置為中斷線路 
//由於GPIO並不是專用的中斷引腳,因此在用GPIO來觸發外部中斷的時候需要設置將GPIO相應的引腳和中斷線連接起來,具體代碼如下: 


void EXTI_cfg() {   
    EXTI_InitTypeDef EXTI_InitStructure;   //清空中斷標誌 
    EXTI_ClearITPendingBit(EXTI_Line2);   
    EXTI_ClearITPendingBit(EXTI_Line3);  
    EXTI_ClearITPendingBit(EXTI_Line5); 
//選擇中斷管腳PC.2 PC.3 PC.5 
    GPIO_EXTILineConfig(GPIO_PortSourceGPIOC, GPIO_PinSource2);   
    GPIO_EXTILineConfig(GPIO_PortSourceGPIOC, GPIO_PinSource3);   
    GPIO_EXTILineConfig(GPIO_PortSourceGPIOC, GPIO_PinSource5); 
    EXTI_InitStructure.EXTI_Line = EXTI_Line2 | EXTI_Line3 | EXTI_Line5; //選擇中斷線路2 3 5   
    EXTI_InitStructure.EXTI_Mode = EXTI_Mode_Interrupt; //設置為插斷要求,非事件請求  
    EXTI_InitStructure.EXTI_Trigger = EXTI_Trigger_Rising_Falling; //設置中斷觸發方式為上下降沿觸發   
    EXTI_InitStructure.EXTI_LineCmd = ENABLE;      
//外部中斷使能 
  EXTI_Init(&EXTI_InitStructure); 


//EXTI_cfg中需要調用到的函數都在stm32f10x_exti.c。 

//5. 寫中斷回應函數 
//STM32不像C51單片機那樣,可以用過interrupt關鍵字來定義中斷回應函數,STM32的中斷回應函數介面存在中斷向量表中,是由啟動代碼給出的。
//預設的中斷回應函數在stm32f10x_it.c中。因此我們需要把這個檔加入到工程中來。 
//在這個檔中,我們發現,很多函數都是只有一個函數名,並沒有函數體。
我們找到EXTI2_IRQHandler()這個函數,這就是EXTI2中斷回應的函數。
//我的目標是將LED燈點亮,所以函數體其實很簡單: 

void EXTI2_IRQHandler(void) {             //點亮LED燈 
    GPIO_SetBits(GPIOD,GPIO_Pin_6);   //清空中斷標誌位元,防止持續進入中斷   
    EXTI_ClearITPendingBit(EXTI_Line2); 


void EXTI3_IRQHandler(void) {   
    GPIO_SetBits(GPIOD,GPIO_Pin_3);   
    EXTI_ClearITPendingBit(EXTI_Line3); 


void EXTI9_5_IRQHandler(void) {   
    GPIO_SetBits(GPIOB,GPIO_Pin_5);   
    EXTI_ClearITPendingBit(EXTI_Line5); 

//由於EXTI5-9是共用一個中斷響應函數,因此所有的EXTI5 - EXTI9的回應函數都寫在這個裡面。 

//6. 寫主函數 
#include "stm32f10x_lib.h" 
    void RCC_cfg(); 
    void IO_cfg(); 
    void EXTI_cfg(); 
    void NVIC_cfg(); 
    int main() {   
        RCC_cfg();   
        IO_cfg();   
        NVIC_cfg();   
        EXTI_cfg();   
        while(1);  

    } 



//main函數前是函式宣告,main函數函數體中都是調用初始化配置函數,然後進入閉環,等待中斷回應。 

沒有留言: