一、嵌套终端向量控制器NVIC
1.1NVIC介绍
NVIC(Nest Vector Interrupt Controller),嵌套中断向量控制器,作用是管理中断嵌套 先级。 核心任务是管理中断优
管理中断嵌套:我们在处理某个中断的过程中还没处理完这个中断,又来了一个中断,是否可以打断这个中断,不能就存在中断嵌套
特点:
68个可屏蔽中断通道(不包含16个Cortex-M3的中断线)
16个可编程的优先等级(使用了4位中断优先级)
低延迟的异常和中断处理
电源管理控制
系统控制寄存器的实现
12.优先级
抢占优先级和响应优先级
拥有较高抢占优先级的中断可以打断抢占优先级较低的中断
若两个抢占优先级的中断同时挂起,则优先执行响应优先级较高的中断
若两个挂起的中断优先级都一致,则优先执行位于中断向量表中位置较高的中断
响应优先级不会造成中断嵌套,也就是说中断嵌套是由抢占优先级决定的(抢占优先级数字越小,优先级越高)
8种分配方式:
所有8位全部用于指定响应优先级
第7位用于指定抢占式优先级,0-6位用于指定响应优先级
6-7位用于指定抢占式优先级,0-5位用于指定响应优先级
5-7位用于指定抢占式优先级,0-4位用于指定响应优先级
4-7位用于指定抢占式优先级,0-3位用于指定响应优先级
3-7位用于指定抢占式优先级,0-2位用于指定响应优先级
2-7位用于指定抢占式优先级,0-1位用于指定响应优先级
1-8位田干指定给上式优牛级 0位田干指定响应伏牛级
Cortex-M3允许具有较少中断源时使用较少的寄存器位指定中断源的优先级,因此STM32中断优先级的寄存器位只 用到AIRCR高四位,共有以下几种组合:
第0组:所有4位用于指定响应优先级
第1组:第7位用于指定抢占式优先级, 4-6位用于指定响应优先级 '
第2组:第6-7位用于指定抢占式优先级, 4-5位用于指定响应优先级
第3组:第5-7位用于指定抢占式优先级,第4位用于指定响应优先级 第4组:
所有4位用于指定抢占式优先级
使用固件库函数配置中断优先级得方法(misc.c中定义):
1
/**
2
*
@brief Configures the priority grouping
:
pre
‐
emption priority and
3
*
subpriority
.
4
*
@param NVIC_PriorityGroup
:
specifies the priority grouping bits
5
*
length
.
6
*
This parameter can be one
of
the following values
:
7
*
@arg NVIC_PriorityGroup_0
:
0
bits
for
pre
‐
emption priority
8
*
4
bits
for
subpriority
9
*
@arg NVIC_PriorityGroup_1
:
1
bits
for
pre
‐
emption priority
10
*
3
bits
for
subpriority
11
*
@arg NVIC_PriorityGroup_2
:
2
bits
for
pre
‐
emption priority
12
*
2
bits
for
subpriority
13
*
@arg NVIC_PriorityGroup_3
:
3
bits
for
pre
‐
emption priority
14
*
1
bits
for
subpriority
15
*
@arg NVIC_PriorityGroup_4
:
4
bits
for
pre
‐
emption priority
16
*
0
bits
for
subpriority
17
*
@retval None
18
*/
19
void
NVIC_PriorityGroupConfig
(
uint32_t NVIC_PriorityGroup
)
优先级组确定后,可以根据优先级组来配置对应IRQ的抢占优先级和响应优先级
1.3NVIC初始化
每个外部中断都由NVIC统一进行管理,所以NVIC包含了中断功能的使能和失能,优先级的
配置等功能。
固件库中关于NVIC的初始化函数
1
/**
2
*
@brief Initializes the
NVIC
peripheral according to the specified
3
*
parameters
in
the NVIC_InitStruct
.
4
*
@param NVIC_InitStruct
:
pointer to a NVIC_InitTypeDef structure
5
*
that contains
6
*
the configuration information
for
the specified
NVIC
7
*
peripheral
.
8
*
@retval None
9
*/
10
void
NVIC_Init
(
NVIC_InitTypeDef
*
NVIC_InitStruct
)
11
形参变量说明:
1
typedef struct
2
{
3
uint8_t NVIC_IRQChannel
;
4
uint8_t NVIC_IRQChannelPreemptionPriority
;
5
uint8_t NVIC_IRQChannelSubPriority
;
6
FunctionalState NVIC_IRQChannelCmd
;
7
}
1.NVIC_IRQChannel:外部中断通道(在stm32f10x.h中定义)
1
typedef
enum
IRQn
2
{
3
NonMaskableInt_IRQn
= ‐
14
,
/*!< 2 Non Maskable Interrupt*/
4
MemoryManagement_IRQn
= ‐
12
,
/*!< 4 Cortex‐M3 Memory Management Interrup
t*/
5
BusFault_IRQn
= ‐
11
,
/*!< 5 Cortex‐M3 Bus Fault Interrupt*/
6
UsageFault_IRQn
= ‐
10
,
/*!< 6 Cortex‐M3 Usage Fault Interrupt*/
7
SVCall_IRQn
= ‐
5
,
/*!< 11 Cortex‐M3 SV Call Interrupt*/
8
DebugMonitor_IRQn
= ‐
4
,
/*!< 12 Cortex‐M3 Debug Monitor Interrupt*/
9
PendSV_IRQn
= ‐
2
,
/*!< 14 Cortex‐M3 Pend SV Interrupt*/
10
SysTick_IRQn
= ‐
1
,
/*!< 15 Cortex‐M3 System Tick Interrupt*/
11
12
WWDG_IRQn
=
0
,
/*!< Window WatchDog Interrupt*/
13
PVD_IRQn
=
1
,
/*!< PVD through EXTI Line detection Interrupt*/
14
TAMPER_IRQn
=
2
,
/*!< Tamper Interrupt*/
15
RTC_IRQn
=
3
,
/*!< RTC global Interrupt*/
16
FLASH_IRQn
=
4
,
/*!< FLASH global Interrupt*/
17
RCC_IRQn
=
5
,
/*!< RCC global Interrupt*/
18
EXTI0_IRQn
=
6
,
/*!< EXTI Line0 Interrupt*/
19
EXTI1_IRQn
=
7
,
/*!< EXTI Line1 Interrupt*/
20
EXTI2_IRQn
=
8
,
/*!< EXTI Line2 Interrupt*/
21
EXTI3_IRQn
=
9
,
/*!< EXTI Line3 Interrupt*/
22
EXTI4_IRQn
=
10
,
/*!< EXTI Line4 Interrupt*/
23
DMA1_Channel1_IRQn
=
11
,
/*!< DMA1 Channel 1 global Interrupt*/
24
DMA1_Channel2_IRQn
=
12
,
/*!< DMA1 Channel 2 global Interrupt*/
25
DMA1_Channel3_IRQn
=
13
,
/*!< DMA1 Channel 3 global Interrupt*/
26
DMA1_Channel4_IRQn
=
14
,
/*!< DMA1 Channel 4 global Interrupt*/
27
DMA1_Channel5_IRQn
=
15
,
/*!< DMA1 Channel 5 global Interrupt*/
28
DMA1_Channel6_IRQn
=
16
,
/*!< DMA1 Channel 6 global Interrupt*/
29
DMA1_Channel7_IRQn
=
17
,
/*!< DMA1 Channel 7 global Interrupt*/
30
...
31
}
2.NVIC_IRQChannelPreemptionPriority:抢占优先级(最大取值15)
3.NVIC_IRQChannelSubPriority:响应优先级(最大取值15)
4.NVIC_IRQChannelCmd:(ENABLE/DISABLE)
使能/失能对应的中断通道
GPIO的外部中断触发方式为高电平触发上升沿触发,上升沿触发一个中断交由EXTI中断控制器进行管理,再往上给到nvic,你想由cpu响应中断,首先由nvic管理,nvic给没个中断一个管理开关,没有nvic的管理就没有cpu对于中断的响应
1.4中断的具体行为
当CM3开始响应一个中断时,会做如下动作:
入栈: 把8个寄存器的值压入栈
取向量:从向量表中找出对应的服务程序入口地址
选择堆栈指针MSP(主堆栈)/PSP(进程堆栈),更新堆栈指针SP,更新链接寄存器
LR,更新程序计数器PC
1.4.1、入栈
响应异常的第一个动作,就是自动保存现场,依次把xPSR、PC, LR, R12以及R3-R0由
硬
件寄存器自动压入适当的堆栈中。
1.4.2、取向量
数据总线(系统总线)在执行入栈的时候,指令总线从向量表中找出正确的异常向量,
然后在服务程序的入口处预取指。
(由此可以看到各自都有专用总线的好处:入栈与取指这两个工作能同时进行)
1.4.3、更新寄存器
在入栈和取向量操作完成之后,执行服务例程之前,还要更新一系列的寄存器.
SP:在入栈后会把堆栈指针更新到新的位置。在执行服务例程时,将由MSP负
责对堆栈的
访问。
PSR:更新IPSR位段的值为新响应的异常编号。
PC:在取向量完成后,PC将指向服务例程的入口地址,
LR:在出入ISR(Interrupt Service Routines)中断服务程序的时候,LR的值
将得到更新
(在异常进入时由系统计算并赋给LR,并在异常返回时使用它
1.5、异常/中断返回
当异常服务例程执行完毕后,需要恢复先前的系统状态,才能使被中断的程序得以继续执
行。
异常/中断处理完成后,执行如下处理:
出栈:恢复先前压入栈中的寄存器,堆栈指针的值也改回先前的值
更新NVIC寄存器:伴随着异常的返回,它的活动位也被硬件清除
二、外部中断
对于互联型产品,外部中断/事件控制器由20个产生事件/中断请求的边沿检测器组成,
对于其它
产品,则有19个能产生事件/中断请求的边沿检测器。每个输入线可以独立地配置输入类型
和对应的触发事件(上升沿或下降沿或者双边沿都触发)。每个输入线都可以独立地被屏蔽。
挂起寄存器保持着状态线的中断请求
EXTI 可分为两大部分功能,一个是产生中断,另一个是产生事件:
中断:
信号从输入线输入,经过边沿检测电路来控制信号触发(根据上升沿下降沿触发选择寄
存器的设置来控制),如果检测到有效信号后,将该有效信号输出到或门电路,由红色箭头
方向经过请求挂起寄存器和中断屏蔽寄存器,到达与门电路,条件满足送至NVIC
事件:
信号从输入线输入,经过边沿检测电路来控制信号触发(根据上升沿下降沿触发选择寄存器
的设置来控制),如果检测到有效信号后,将该有效信号输出到或门电路,由蓝色箭头方向
经过与门电路,送至买脉冲发生器,产生脉冲。这个脉冲信号可以给其他外设电路使用,比
如定时器 TIM、模拟数字转换器 ADC等等,这样的脉冲信号一般用来触发 TIM 或者 ADC
开始转换。
事件和中断的区别:
产生中断线路目的是把输入信号输入到 NVIC,进一步会运行中断服务函数,实现功能,这
样是软件级的。而产生事件线路目的就是传输一个脉冲信号给其他外设使用,并且是电路级
别的信号传输,属于硬件级的。
2.1、主要特性
每个中断/事件都有独立的触发和屏蔽
每个中断线都有专用的状态位
支持多达20个软件的中断/事件请求
检测脉冲宽度低于APB2时钟宽度的外部信号
2.2、外部中断/事件线路映像
通用I/O端口以下图的方式连接到16个外部中断/事件线上
另外四个EXTI线的连接方式如下:
● EXTI线16连接到PVD输出
● EXTI线17连接到RTC闹钟事件
● EXTI线18连接到USB唤醒事件
● EXTI线19连接到以太网唤醒事件(只适用于互联型产品)
2.1、主要特性
每个中断/事件都有独立的触发和屏蔽
每个中断线都有专用的状态位
支持多达20个软件的中断/事件请求
检测脉冲宽度低于APB2时钟宽度的外部信号
2.2、外部中断/事件线路映像
通用I/O端口以下图的方式连接到16个外部中断/事件线上
另外四个EXTI线的连接方式如下:
● EXTI线16连接到PVD输出
● EXTI线17连接到RTC闹钟事件
● EXTI线18连接到USB唤醒事件
● EXTI线19连接到以太网唤醒事件(只适用于互联型产品
2.3、EXTI相关数据结构与函数说明
2.3.1、初始化函数
1
void
EXTI_Init
(
EXTI_InitTypeDef
*
EXTI_InitStruct
)
参数:EXTI_InitStruct
1
typedef struct
2
{
3
uint32_t EXTI_Line
;
//
中断线
4
EXTIMode_TypeDef EXTI_Mode
;
//
中断模式
5
EXTITrigger_TypeDef EXTI_Trigger
;
//
中断触发方式
6
FunctionalState EXTI_LineCmd
;
//
中断功能使能
7
}
@EXTI_Line
1
#define
EXTI_Line0
((
uint32_t
)
0x00001
)
/*!< External interrupt line 0 */
2
#define
EXTI_Line1
((
uint32_t
)
0x00002
)
/*!< External interrupt line 1 */
3
#define
EXTI_Line2
((
uint32_t
)
0x00004
)
/*!< External interrupt line 2 */
4
#define
EXTI_Line3
((
uint32_t
)
0x00008
)
/*!< External interrupt line 3 */
5
#define
EXTI_Line4
((
uint32_t
)
0x00010
)
/*!< External interrupt line 4 */
6
#define
EXTI_Line5
((
uint32_t
)
0x00020
)
/*!< External interrupt line 5 */
7
#define
EXTI_Line6
((
uint32_t
)
0x00040
)
/*!< External interrupt line 6 */
8
#define
EXTI_Line7
((
uint32_t
)
0x00080
)
/*!< External interrupt line 7 */
9
#define
EXTI_Line8
((
uint32_t
)
0x00100
)
/*!< External interrupt line 8 */
10
#define
EXTI_Line9
((
uint32_t
)
0x00200
)
/*!< External interrupt line 9 */
11
#define
EXTI_Line10
((
uint32_t
)
0x00400
)
/*!< External interrupt line 10
*/
12
#define
EXTI_Line11
((
uint32_t
)
0x00800
)
/*!< External interrupt line 11
*/
13
#define
EXTI_Line12
((
uint32_t
)
0x01000
)
/*!< External interrupt line 12
*/
14
#define
EXTI_Line13
((
uint32_t
)
0x02000
)
/*!< External interrupt line 13
*/
15
#define
EXTI_Line14
((
uint32_t
)
0x04000
)
/*!< External interrupt line 14
*/
16
#define
EXTI_Line15
((
uint32_t
)
0x08000
)
/*!< External interrupt line 15
*/
17
#define
EXTI_Line16
((
uint32_t
)
0x10000
)
/*!< External interrupt line 16
Connected to the PVD Output */
18
#define
EXTI_Line17
((
uint32_t
)
0x20000
)
/*!< External interrupt line 17
Connected to the RTC Alarm event */
19
#define
EXTI_Line18
((
uint32_t
)
0x40000
)
/*!< External interrupt line 18
Connected to the USB Device/USB OTG FS
20
Wakeup
from
suspend event
*/
21
#define
EXTI_Line19
((
uint32_t
)
0x80000
)
/*!< External interrupt line 19
Connected to the Ethernet Wakeup event */
@EXTI_Mode
1
typedef
enum
2
{
3
EXTI_Mode_Interrupt
=
0x00
,
//
中断模式
4
EXTI_Mode_Event
=
0x04
//
事件模式
5
}
EXTIMode_TypeDef
;
@EXTI_Trigger
1
typedef
enum
2
{
3
EXTI_Trigger_Rising
=
0x08
,
//
上升沿触发
4
EXTI_Trigger_Falling
=
0x0C
,
//
下降沿触发
5
EXTI_Trigger_Rising_Falling
=
0x10
//
双边沿触发
6
}
EXTITrigger_TypeDef
;
@EXTI_LineCmd
1
ENABLE
2
DISABLE
2.3.2、获取中断状态
1
//
参数:中断线
2
ITStatus
EXTI_GetITStatus
(
uint32_t EXTI_Line
);
2.3.3、清空中断标志位
1
//
参数:中断线
2
void
EXTI_ClearITPendingBit
(
uint32_t EXTI_Line
);
2.3.4、软件产生中断
1
void
EXTI_GenerateSWInterrupt
(
uint32_t EXTI_Line
)
2.4、中断回调函数
在startup_stm32f10x_xx.s中的中断向量表里:
1
DCD
EXTI0_IRQHandler
;
EXTI
Line
0
2
DCD
EXTI1_IRQHandler
;
EXTI
Line
1
3
DCD
EXTI2_IRQHandler
;
EXTI
Line
2
4
DCD
EXTI3_IRQHandler
;
EXTI
Line
3
5
DCD
EXTI4_IRQHandler
;
EXTI
Line
4
6
DCD
EXTI9_5_IRQHandler
;
EXTI
Line
9..5
7
DCD
EXTI15_10_IRQHandler
;
EXTI
Line
15..10