新聞中心

        STM32中斷向量嵌套NVIC理解

        作者: 時(shí)間:2016-11-25 來(lái)源:網(wǎng)絡(luò) 收藏
        一,中斷優(yōu)先級(jí):

        STM32(Cortex-M3)中的優(yōu)先級(jí)概念
        STM32(Cortex-M3)中有兩個(gè)優(yōu)先級(jí)的概念——搶占式優(yōu)先級(jí)和響應(yīng)優(yōu)先級(jí),有人把響應(yīng)優(yōu)先級(jí)稱(chēng)作亞優(yōu)先級(jí)或副優(yōu)先級(jí),每個(gè)中斷源都需要被指定這兩種優(yōu)先級(jí)。

        本文引用地址:http://www.104case.com/article/201611/321552.htm

        具有高搶占式優(yōu)先級(jí)的中斷可以在具有低搶占式優(yōu)先級(jí)的中斷處理過(guò)程中被響應(yīng),即中斷嵌套,或者說(shuō)高搶占式優(yōu)先級(jí)的中斷可以嵌套低搶占式優(yōu)先級(jí)的中斷。

        當(dāng)兩個(gè)中斷源的搶占式優(yōu)先級(jí)相同時(shí),這兩個(gè)中斷將沒(méi)有嵌套關(guān)系,當(dāng)一個(gè)中斷到來(lái)后,如果正在處理另一個(gè)中斷,這個(gè)后到來(lái)的中斷就要等到前一個(gè)中斷處理完之后才能被處理。如果這兩個(gè)中斷同時(shí)到達(dá),則中斷控制器根據(jù)他們的響應(yīng)優(yōu)先級(jí)高低來(lái)決定先處理哪一個(gè);如果他們的搶占式優(yōu)先級(jí)和響應(yīng)優(yōu)先級(jí)都相等,則根據(jù)他們?cè)谥袛啾碇械呐盼豁樞驔Q定先處理哪一個(gè)。

        既然每個(gè)中斷源都需要被指定這兩種優(yōu)先級(jí),就需要有相應(yīng)的寄存器位記錄每個(gè)中斷的優(yōu)先級(jí);在Cortex-M3中定義了8個(gè)比特位用于設(shè)置中斷源的優(yōu)先級(jí),這8個(gè)比特位可以有8種分配方式,如下:

        所有8位用于指定響應(yīng)優(yōu)先級(jí)
        最高1位用于指定搶占式優(yōu)先級(jí),最低7位用于指定響應(yīng)優(yōu)先級(jí)
        最高2位用于指定搶占式優(yōu)先級(jí),最低6位用于指定響應(yīng)優(yōu)先級(jí)
        最高3位用于指定搶占式優(yōu)先級(jí),最低5位用于指定響應(yīng)優(yōu)先級(jí)
        最高4位用于指定搶占式優(yōu)先級(jí),最低4位用于指定響應(yīng)優(yōu)先級(jí)
        最高5位用于指定搶占式優(yōu)先級(jí),最低3位用于指定響應(yīng)優(yōu)先級(jí)
        最高6位用于指定搶占式優(yōu)先級(jí),最低2位用于指定響應(yīng)優(yōu)先級(jí)
        最高7位用于指定搶占式優(yōu)先級(jí),最低1位用于指定響應(yīng)優(yōu)先級(jí)

        這就是優(yōu)先級(jí)分組的概念。

        --------------------------------------------------------------------------------
        Cortex-M3允許具有較少中斷源時(shí)使用較少的寄存器位指定中斷源的優(yōu)先級(jí),因此STM32把指定中斷優(yōu)先級(jí)的寄存器位減少到4位,這4個(gè)寄存器位的分組方式如下:

        第0組:所有4位用于指定響應(yīng)優(yōu)先級(jí)
        第1組:最高1位用于指定搶占式優(yōu)先級(jí),最低3位用于指定響應(yīng)優(yōu)先級(jí)
        第2組:最高2位用于指定搶占式優(yōu)先級(jí),最低2位用于指定響應(yīng)優(yōu)先級(jí)
        第3組:最高3位用于指定搶占式優(yōu)先級(jí),最低1位用于指定響應(yīng)優(yōu)先級(jí)
        第4組:所有4位用于指定搶占式優(yōu)先級(jí)

        可以通過(guò)調(diào)用STM32的固件庫(kù)中的函數(shù)NVIC_PriorityGroupConfig()選擇使用哪種優(yōu)先級(jí)分組方式,這個(gè)函數(shù)的參數(shù)有下列5種:

        NVIC_PriorityGroup_0 => 選擇第0組
        NVIC_PriorityGroup_1 => 選擇第1組
        NVIC_PriorityGroup_2 => 選擇第2組
        NVIC_PriorityGroup_3 => 選擇第3組
        NVIC_PriorityGroup_4 => 選擇第4組

        接下來(lái)就是指定中斷源的優(yōu)先級(jí),下面以一個(gè)簡(jiǎn)單的例子說(shuō)明如何指定中斷源的搶占式優(yōu)先級(jí)和響應(yīng)優(yōu)先級(jí):

        // 選擇使用優(yōu)先級(jí)分組第1組
        NVIC_PriorityGroupConfig(NVIC_PriorityGroup_1);
        中斷設(shè)置:時(shí)能中斷->優(yōu)先級(jí)分組方式(對(duì)應(yīng)的每個(gè)中斷都有)->設(shè)定搶占式優(yōu)先級(jí)別->設(shè)定響應(yīng)優(yōu)先級(jí)別->調(diào)用NVIC_Init(&xx)
        // 使能EXTI0中斷
        NVIC_InitStructure.NVIC_IRQChannel = EXTI0_IRQChannel;
        NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 1; // 指定搶占式優(yōu)先級(jí)別1

        NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0; // 指定響應(yīng)優(yōu)先級(jí)別0
        NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
        NVIC_Init(&NVIC_InitStructure);

        // 使能EXTI9_5中斷
        NVIC_InitStructure.NVIC_IRQChannel = EXTI9_5_IRQChannel;
        NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0; // 指定搶占式優(yōu)先級(jí)別0
        NVIC_InitStructure.NVIC_IRQChannelSubPriority = 1; // 指定響應(yīng)優(yōu)先級(jí)別1
        NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
        NVIC_Init(&NVIC_InitStructure);

        要注意的幾點(diǎn)是:

        1)如果指定的搶占式優(yōu)先級(jí)別或響應(yīng)優(yōu)先級(jí)別超出了選定的優(yōu)先級(jí)分組所限定的范圍,將可能得到意想不到的結(jié)果;

        2)搶占式優(yōu)先級(jí)別相同的中斷源之間沒(méi)有嵌套關(guān)系;

        3)如果某個(gè)中斷源被指定為某個(gè)搶占式優(yōu)先級(jí)別,又沒(méi)有其它中斷源處于同一個(gè)搶占式優(yōu)先級(jí)別,則可以為這個(gè)中斷源指定任意有效的響應(yīng)優(yōu)先級(jí)別。

        二,開(kāi)關(guān)總中斷:

        在STM32/Cortex-M3中是通過(guò)改變CPU的當(dāng)前優(yōu)先級(jí)來(lái)允許或禁止中斷。
        PRIMASK位:只允許NMI和hard fault異常,其他中斷/異常都被屏蔽(當(dāng)前CPU優(yōu)先級(jí)=0)。
        FAULTMASK位:只允許NMI,其他所有中斷/異常都被屏蔽(當(dāng)前CPU優(yōu)先級(jí)=-1)。

        在STM32固件庫(kù)中(stm32f10x_nvic.c和stm32f10x_nvic.h) 定義了四個(gè)函數(shù)操作PRIMASK位和FAULTMASK位,改變CPU的當(dāng)前優(yōu)先級(jí),從而達(dá)到控制所有中斷的目的。

        下面兩個(gè)函數(shù)等效于關(guān)閉總中斷:
        void NVIC_SETPRIMASK(void);
        void NVIC_SETFAULTMASK(void);

        下面兩個(gè)函數(shù)等效于開(kāi)放總中斷:
        void NVIC_RESETPRIMASK(void);
        void NVIC_RESETFAULTMASK(void);

        上面兩組函數(shù)要成對(duì)使用,不能交叉使用。

        例如:

        第一種方法:
        NVIC_SETPRIMASK(); //關(guān)閉總中斷
        NVIC_RESETPRIMASK();//開(kāi)放總中斷

        第二種方法:
        NVIC_SETFAULTMASK(); //關(guān)閉總中斷
        NVIC_RESETFAULTMASK();//開(kāi)放總中斷

        常常使用

        NVIC_SETPRIMASK(); // Disable Interrupts
        NVIC_RESETPRIMASK(); // Enable Interrupts

        STM32中斷流程處理

        作為我的一個(gè)習(xí)慣,學(xué)習(xí)某一個(gè)平臺(tái)的東西,總是先要摸清楚中斷的處理流程,當(dāng)然是從文件代碼級(jí)的流程分析了。

        下面就說(shuō)下stm32的中斷流程。我們知道,stm32的庫(kù)中寫(xiě)好了很多的驅(qū)動(dòng)程序,可以說(shuō)包括了所有的。同時(shí)也提供很多數(shù)據(jù)處理方式,例如串口的讀寫(xiě),用戶可以選擇輪詢、中斷、DMA等3中方式來(lái)處理。

        關(guān)于中斷,stm32的庫(kù)中做好了框架,用戶只要填寫(xiě)好幾個(gè)函數(shù)的實(shí)現(xiàn)就ok了,就像網(wǎng)上說(shuō)的,這就是傻瓜式開(kāi)發(fā)。

        了解中斷,首先要知道stm32f10x_it.c這個(gè)文件,一般情況下是和main文件在同一個(gè)目錄下的。打開(kāi)這個(gè)文件,我們可以看到xyz_IRQHandler函數(shù)的實(shí)現(xiàn),雖然說(shuō)是實(shí)現(xiàn),但是幾乎都是空的。對(duì)了,這些函數(shù)就是要用戶填寫(xiě)的中斷處理函數(shù),如果你用到了哪個(gè)中斷來(lái)做相應(yīng)的處理,你就要填寫(xiě)相應(yīng)的中斷處理函數(shù),需要根據(jù)各外設(shè)的實(shí)際情況來(lái)填寫(xiě),但是一般都會(huì)有關(guān)閉和開(kāi)啟中斷。在這個(gè)文件中還有很多系統(tǒng)相關(guān)的中斷處理函數(shù),例如系統(tǒng)時(shí)鐘SysTickHandler。具體的實(shí)現(xiàn)可以參考stm32fwlibFWLibexamples下的各例子。

        到這里,我們也只不過(guò)看了中斷的處理函數(shù),而這些處理函數(shù)是如何被硬件中斷調(diào)用的呢?嗯,說(shuō)到這里就不得不提一下stm32f10x_vector.c這個(gè)文件了。內(nèi)容如下:
        typedef void( *intfunc )( void );
        typedef union { intfunc __fun; void * __ptr; } intvec_elem;



        //IAR對(duì)所用語(yǔ)言(這里是C)做的一些擴(kuò)展,也就是說(shuō)這里可以用擴(kuò)展的功能
        #pragma language=extended#pragma segment="CSTACK"

        void __iar_program_start( void );


        #pragma location = ".intvec"


        const intvec_elem __vector_table[] =
        {
        { .__ptr = __sfe( "CSTACK" ) },
        &__iar_program_start,
        NMIException,
        HardFaultException,
        MemManageException,
        BusFaultException,
        UsageFaultException,
        0, 0, 0, 0,
        vPortSVCHandler,
        DebugMonitor,
        0,
        xPortPendSVHandler,
        xPortSysTickHandler,
        WWDG_IRQHandler,
        PVD_IRQHandler,
        TAMPER_IRQHandler,
        RTC_IRQHandler,
        FLASH_IRQHandler,
        RCC_IRQHandler,
        EXTI0_IRQHandler,
        EXTI1_IRQHandler,
        EXTI2_IRQHandler,
        EXTI3_IRQHandler,
        EXTI4_IRQHandler,
        DMAChannel1_IRQHandler,
        DMAChannel2_IRQHandler,
        DMAChannel3_IRQHandler,
        DMAChannel4_IRQHandler,
        DMAChannel5_IRQHandler,
        DMAChannel6_IRQHandler,
        DMAChannel7_IRQHandler,
        ADC_IRQHandler,
        USB_HP_CAN_TX_IRQHandler,
        USB_LP_CAN_RX0_IRQHandler,
        CAN_RX1_IRQHandler,
        CAN_SCE_IRQHandler,
        EXTI9_5_IRQHandler,
        TIM1_BRK_IRQHandler,
        TIM1_UP_IRQHandler,
        TIM1_TRG_COM_IRQHandler,
        TIM1_CC_IRQHandler,
        vTimer2IntHandler,
        TIM3_IRQHandler,
        TIM4_IRQHandler,
        I2C1_EV_IRQHandler,
        I2C1_ER_IRQHandler,
        I2C2_EV_IRQHandler,
        I2C2_ER_IRQHandler,
        SPI1_IRQHandler,
        SPI2_IRQHandler,
        vUARTInterruptHandler,
        USART2_IRQHandler,
        USART3_IRQHandler,
        EXTI15_10_IRQHandler,
        RTCAlarm_IRQHandler,
        USBWakeUp_IRQHandler,
        };
        現(xiàn)在我們清楚了,這兒就是中斷向量表,每一個(gè)item對(duì)應(yīng)一個(gè)中斷或異常處理,這里item的填寫(xiě)要和stm32spec中的Interrupt and exception vectors一節(jié)中的列表中的順序一致。

        說(shuō)道這里,又有一個(gè)問(wèn)題,這個(gè)向量表是放在何處的呢?上面對(duì).intvec的解釋可以看出是被鏈接器放到了一個(gè)地址上(這里是0x08000000,NVIC_VectTab_FLASH)。但是stm32是怎么知道這個(gè)地址的呢,也許有個(gè)默認(rèn)值,或者是就這一個(gè)固定值?)。我們?cè)趕tm32f10x_nvic.c文件中發(fā)現(xiàn)下面這樣的一個(gè)函數(shù)
        void NVIC_SetVectorTable(u32 NVIC_VectTab, u32 Offset)
        {

        assert(IS_NVIC_VECTTAB(NVIC_VectTab));
        assert(IS_NVIC_OFFSET(Offset));

        SCB->ExceptionTableOffset = (((u32)Offset << 0x07) & (u32)0x1FFFFF80);
        SCB->ExceptionTableOffset |= NVIC_VectTab;
        }
        同時(shí)在example目錄下有vectortable_relocation這樣的一個(gè)例子:This example describes how to use the NVIC firmware library to set the CortexM3 vector table in a specific address other than default.
        在這個(gè)例子里面就是直接調(diào)用了上面的那個(gè)函數(shù),似乎意思很明顯了。但是SCB->ExceptionTableOffset是如何起作用的呢?

        著重解釋這個(gè)問(wèn)題,先看一組定義:【stm32f10x_map.b】

        #define SCS_BASE ((u32)0xE000E000)
        #define SysTick_BASE (SCS_BASE + 0x0010)
        #define NVIC_BASE (SCS_BASE + 0x0100)
        #define SCB_BASE (SCS_BASE + 0x0D00)
        #ifdef _SCB
        #define SCB ((SCB_TypeDef *) SCB_BASE)
        #endif
        typedef struct
        {
        vu32 CPUID;
        vu32 IRQControlState;
        vu32 ExceptionTableOffset;
        vu32 AIRC;
        vu32 SysCtrl;
        vu32 ConfigCtrl;
        vu32 SystemPriority[3];
        vu32 SysHandlerCtrl;
        vu32 ConfigFaultStatus;
        vu32 HardFaultStatus;
        vu32 DebugFaultStatus;
        vu32 MemoryManageFaultAddr;
        vu32 BusFaultAddr;
        } SCB_TypeDef;

        其實(shí)這里主要就是要弄清楚這個(gè)SCB是什么意思,因?yàn)檫@個(gè)結(jié)構(gòu)是映射到一個(gè)物理地址上的。像別的控制寄存器都是這么個(gè)玩法,莫非這也是個(gè)某類(lèi)控制器。google一下,果然對(duì)于系統(tǒng)控制寄存器組【上篇文章有提到】STM32的固件庫(kù)中有如下定義:
        typedef struct
        {
        vuc32 CPUID;
        vu32 ICSR;
        vu32 VTOR;
        vu32 AIRCR;
        vu32 SCR;
        vu32 CCR;
        vu32 SHPR[3];
        vu32 SHCSR;
        vu32 CFSR;
        vu32 HFSR;
        vu32 DFSR;
        vu32 MMFAR;
        vu32 BFAR;
        vu32 AFSR;
        } SCB_TypeDef;
        它們對(duì)應(yīng)ARM手冊(cè)中的名稱(chēng)為
        CPUID = CPUID Base Register
        ICSR = Interrupt Control State Register
        VTOR = Vector Table Offset Register
        AIRCR = Application Interrupt/Reset Control Register
        SCR = System Control Register
        CCR = Configuration Control Register
        SHPR = System Handlers Priority Register
        SHCSR = System Handler Control and State Register
        CFSR = Configurable Fault Status Registers
        HFSR = Hard Fault Status Register
        DFSR = Debug Fault Status Register
        MMFAR = Mem Manage Address Register
        BFAR = Bus Fault Address Register
        AFSR = Auxiliary Fault Status Register

        至此,我們終于清楚了,這個(gè)中斷向量表的地址,最終是要寫(xiě)到某個(gè)控制器中去。那這么說(shuō)來(lái),上述的0x08000000可以是個(gè)別的值了,只要保證這一處的地址不能被別的程序訪問(wèn)就行了。




        關(guān)鍵詞: STM32中斷向量嵌套NVI

        評(píng)論


        技術(shù)專(zhuān)區(qū)

        關(guān)閉
        主站蜘蛛池模板: 岱山县| 乐亭县| 保德县| 邢台县| 梁河县| 封开县| 如皋市| 浦北县| 琼海市| 乌苏市| 巴彦淖尔市| 五台县| 轮台县| 禄劝| 江川县| 雅江县| 玉山县| 华安县| 许昌市| 南通市| 庆阳市| 南开区| 大英县| 调兵山市| 墨玉县| 云浮市| 湟中县| 漯河市| 疏附县| 大同市| 南阳市| 荣昌县| 长阳| 靖宇县| 卢湾区| 武城县| 益阳市| 邵阳市| 婺源县| 海伦市| 襄垣县|