在线看毛片网站电影-亚洲国产欧美日韩精品一区二区三区,国产欧美乱夫不卡无乱码,国产精品欧美久久久天天影视,精品一区二区三区视频在线观看,亚洲国产精品人成乱码天天看,日韩久久久一区,91精品国产91免费

<menu id="6qfwx"><li id="6qfwx"></li></menu>
    1. <menu id="6qfwx"><dl id="6qfwx"></dl></menu>

      <label id="6qfwx"><ol id="6qfwx"></ol></label><menu id="6qfwx"></menu><object id="6qfwx"><strike id="6qfwx"><noscript id="6qfwx"></noscript></strike></object>
        1. <center id="6qfwx"><dl id="6qfwx"></dl></center>

            新聞中心

            EEPW首頁 > 嵌入式系統(tǒng) > 設(shè)計應(yīng)用 > STM32中斷向量嵌套NVIC理解

            STM32中斷向量嵌套NVIC理解

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

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

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

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

            當(dāng)兩個中斷源的搶占式優(yōu)先級相同時,這兩個中斷將沒有嵌套關(guān)系,當(dāng)一個中斷到來后,如果正在處理另一個中斷,這個后到來的中斷就要等到前一個中斷處理完之后才能被處理。如果這兩個中斷同時到達(dá),則中斷控制器根據(jù)他們的響應(yīng)優(yōu)先級高低來決定先處理哪一個;如果他們的搶占式優(yōu)先級和響應(yīng)優(yōu)先級都相等,則根據(jù)他們在中斷表中的排位順序決定先處理哪一個。

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

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

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

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

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

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

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

            接下來就是指定中斷源的優(yōu)先級,下面以一個簡單的例子說明如何指定中斷源的搶占式優(yōu)先級和響應(yīng)優(yōu)先級:

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

            NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0; // 指定響應(yīng)優(yōu)先級別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)先級別0
            NVIC_InitStructure.NVIC_IRQChannelSubPriority = 1; // 指定響應(yīng)優(yōu)先級別1
            NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
            NVIC_Init(&NVIC_InitStructure);

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

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

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

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

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

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

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

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

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

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

            例如:

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

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

            常常使用

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

            STM32中斷流程處理

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

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

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

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

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



            //IAR對所用語言(這里是C)做的一些擴(kuò)展,也就是說這里可以用擴(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)在我們清楚了,這兒就是中斷向量表,每一個item對應(yīng)一個中斷或異常處理,這里item的填寫要和stm32spec中的Interrupt and exception vectors一節(jié)中的列表中的順序一致。

            說道這里,又有一個問題,這個向量表是放在何處的呢?上面對.intvec的解釋可以看出是被鏈接器放到了一個地址上(這里是0x08000000,NVIC_VectTab_FLASH)。但是stm32是怎么知道這個地址的呢,也許有個默認(rèn)值,或者是就這一個固定值?)。我們在stm32f10x_nvic.c文件中發(fā)現(xiàn)下面這樣的一個函數(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;
            }
            同時在example目錄下有vectortable_relocation這樣的一個例子:This example describes how to use the NVIC firmware library to set the CortexM3 vector table in a specific address other than default.
            在這個例子里面就是直接調(diào)用了上面的那個函數(shù),似乎意思很明顯了。但是SCB->ExceptionTableOffset是如何起作用的呢?

            著重解釋這個問題,先看一組定義:【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;

            其實這里主要就是要弄清楚這個SCB是什么意思,因為這個結(jié)構(gòu)是映射到一個物理地址上的。像別的控制寄存器都是這么個玩法,莫非這也是個某類控制器。google一下,果然對于系統(tǒng)控制寄存器組【上篇文章有提到】STM32的固件庫中有如下定義:
            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;
            它們對應(yīng)ARM手冊中的名稱為
            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

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




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

            評論


            技術(shù)專區(qū)

            關(guān)閉