一、概述 1、中断简介 中断是指计算机运行过程中,出现某些意外情况需主机干预时,机器能自动停止正在运行的程序并转入处理新情况的程序,处理完毕后又返回原被暂停的程序继续运行。 中断的分类: (1)硬件中断 - 可屏蔽中断:硬件中断的一类,可通过在中断屏蔽寄存器中设定位掩码来关闭。
- 非可屏蔽中断:硬件中断的一类,无法通过在中断屏蔽寄存器中设定位掩码来关闭。典型例子是时钟中断(一个硬件时钟以恒定频率—如50Hz—发出的中断)。
- 处理器中断:一种特殊的硬件中断。由处理器发出,被其它处理器接收。仅见于多处理器系统,以便于处理器间通信或同步。
- 伪中断:一类不希望被产生的硬件中断。发生的原因有很多种,如中断线路上电气信号异常,或是中断请求设备本身有问题。( p; F. @+ u- D* [1 h
(2)软件中断: 是一条CPU指令,用以自陷一个中断。由于软中断指令通常运行一个切换CPU至内核态的子例程,它常被用作实现系统调用。 中断优先权: 在某一时刻有几个中断源同时发出中断请求时,处理器只响应其中优先权最高的中断源。当处理机正在运行某个中断服务程序期间出现另一个中断源的请求时,如果后者的优先权低于前者,处理机不予理睬,反之,处理机立即响应后者,进入所谓的“嵌套中断”。中断优先权的排序按其性质、重要性以及处理的方便性决定,由硬件的优先权仲裁逻辑或软件的顺序询问程序来实现。 中断过程: (1)中断源发出中断请求; (2)判断当前处理机是否允许中断和该中断源是否被屏蔽; (3)优先权排队; (4)处理机执行完当前指令或当前指令无法执行完,则立即停止当前程序保护断点地址和处理机当前状态,转入相应的中断服务程序。 (5)执行中断服务程序; (6)恢复被保护的状态,执行“中断返回”指令回到被中断的程序或转入其他程序。 2、外部中断/时间控制器(EXTI)外部中断 EXTI主要特性 (1)每个中断/事件线上都具有独立的触发和屏蔽。 (2)每个中断线都具有专用的状态位。 (3)支持多达23个软件事件/中断请求。 图3_0外部中断/事件控制器框图 外部中断/事件线映射: 多达140个的GPIO通过以下方式连接到16个外部中断/事件线: 图3_1外部中断/事件线映射图 另外七根EXTI线连接方式如下: - EXTI线16连接到PVD输出。
- EXTI线17连接到RTC闹钟事件。
- EXTI线18连接到USB OTG FS 唤醒事件。
- EXTI线19连接到以太网唤醒事件。
- EXTI线20连接到USB OTG HS (在FS中配置)唤醒事件。
- EXTI线21连接到RTC入侵和时间戳事件。
- EXTI线22连接到RTC唤醒事件。
' c( @. L( n1 l& Y/ @9 q3 H3 k
3、STM32中断优先级的判断
; d# }+ B- J! r' z* v STM32 目前支持的中断共为 84 个(16 个内核+68 个外部), 16 级可编程中断优先级的设置(仅使用中断优先级设置 8bit 中的高 4 位)和16个抢占优先级(因为抢占优先级最多可以有四位数)。 STM32(Cortex-M3)中有两个优先级的概念——抢占式优先级和响应优先级,有人把响应优先级称作'亚优先级'或'副优先级',每个中断源都需要被指定这两种优先级。 具有高抢占式优先级的中断可以在具有低抢占式优先级的中断处理过程中被响应,即中断嵌套,或者说高抢占式优先级的中断可以嵌套低抢占式优先级的中断。 当两个中断源的抢占式优先级相同时,这两个中断将没有嵌套关系,当一个中断到来后,如果正在处理另一个中断,这个后到来的中断就要等到前一个中断处理完之后才能被处理。如果这两个中断同时到达,则中断控制器根据他们的响应优先级高低来决定先处理哪一个;如果他们的抢占式优先级和响应优先级都相等,则根据他们在中断表中的排位顺序决定先处理哪一个。 优先级分组: 中断优先级分组域中,可以有8种分配方式,如下: 所有8位用于指定响应优先级。 最高1位用于指定抢占式优先级。 最低7位用于指定响应优先级。 最高2位用于指定抢占式优先级。 最低6位用于指定响应优先级。 最高3位用于指定抢占式优先级。 最低5位用于指定响应优先级。 最高4位用于指定抢占式优先级。 最低4位用于指定响应优先级。 最高5位用于指定抢占式优先级。 最低3位用于指定响应优先级。 最高6位用于指定抢占式优先级。 最低2位用于指定响应优先级。 最高7位用于指定抢占式优先级。 最低1位用于指定响应优先级。 二、硬件电路 图3_2引脚分布图
, O& B5 J: a, c' }& T' ?) S0 k 图3_3电路原理图 在iCore3双核心开发板中,采用一个独立的按键与STM32F407的GPIO相连,另一端接地,且PH15外接一个1k电阻大小的上拉电阻。
' h. L3 }' n; _- [三、实验原理 按键的一端与SMT32的GPIO(PH15)相连,另一端接地,且PH15外接一个1K电流大小的上拉电阻。初始化时把PH15设置成输入模式。当按键弹起时候,PH15由于上拉电阻的作用呈高电平(3.3V);当按键按下时候,PH15直接被按键短接到GND,呈低电平。因此PH15的电平变化产生下降沿,从而进入中断函数,可得到按键状态。原理图如下: 图3_4实验原理图 四、源代码 1、主函数 - /*
, ]/ p0 i8 g. Y' ~, z! P - * Name : main# v2 {& y1 S& w/ N& _ m5 { f5 y
- * Description : ---; c( J) @! s: l1 g1 _
- * Author : ysloveivy. c+ ~) B$ @; x; h3 l: I
- *
) C; S/ n# U" d - * History1 F% Z3 ~% l6 @9 @; \' S6 c
- * --------------------
+ P; R# F3 L+ ?' I - * Rev : 0.00
# h) z) H: z' y( q - * Date : 11/21/2015, w5 d5 Z6 b7 F& D+ l ` H& [8 I
- * - h: @& l0 Z* [# `
- * create.2 J& F3 z( l1 s8 r( B3 d3 F
- * --------------------
" o( F3 ~6 s6 | - */* w0 \* B0 A9 D3 x6 `5 `' e
- int main(void)
* i/ h" a* I; @/ u/ d - {7 _% |! @* g+ m- y
- led.initialize();
5 n8 o1 ]: ]3 j( f7 Q - exti.initialize();' C& ^8 p5 N5 }) {' A, Y
- LED_RED_ON;
* [( e. F' q, R: s) ^) S4 s - LED_GREEN_OFF;; l7 ^ n5 E5 H" C8 V: }* z+ x
- LED_BLUE_OFF;4 A. e" T9 f# Q9 D
- + n- D2 \! h. V1 X
- while(1);: e! W, z$ C' H$ d5 z/ O- ^2 x
- }
1 V! }1 F* h# W' Z* x1 _
复制代码2、GPIO初始化 - /*9 ]$ t. L4 [: M, ]+ ~. N4 g1 _9 r9 I
- * Name : initialize
) U, |/ ~4 l' i ]* q4 l) D; l - * Description : ---
, |. S$ y& x! _' n! B - * Author : ysloveivy.& X! F/ v) w" v
- */ A! A5 [( F5 b7 G" u; a$ a
- * History
7 B: s+ w" S, }0 ?1 b - * -------------------
7 A+ V; }8 Y2 \+ y9 `" p6 z* m - * Rev : 0.00: U* F( L4 E: m+ ]
- * Date : 11/21/2015
* i: v! g: l q# }) ^, x" g+ @ - *
B- G/ v) p, ` K) q J b- W) P - * create.5 s! \+ Y) z; ~" o
- * -------------------( |2 ^( f* t9 V! a, \
- */
3 X$ Q" l i0 f7 D3 j/ F0 P - static int initialize(void)
0 u, X, e3 b2 Y' d7 y6 r - {7 I+ ]4 J3 l4 v5 f) y
- EXTI_InitTypeDef EXTI_InitStructure;
( M- @% j' J4 U - GPIO_InitTypeDef GPIO_InitStructure;4 L: r% T; ~# O. v5 |; V0 `7 ?4 e
- NVIC_InitTypeDef NVIC_InitStructure;
6 P! Y( X8 L& R7 T! {3 B
- R; s; ]; X+ O; [ W- RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOH,ENABLE); //使能GPIOH时钟
# D; M/ d! {) n& i/ I( }& s - RCC_APB2PeriphClockCmd(RCC_APB2Periph_SYSCFG, ENABLE); //使能SYSCFG时钟, ~2 M( C$ \6 R. k* I" [
- ; P$ e6 q0 d R: p2 X3 o; d+ B
- GPIO_InitStructure.GPIO_Pin = GPIO_Pin_15;0 ?/ g+ t& T+ n. p$ p7 D$ N
- GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN;% w: V I R: D( r0 @, o+ H( _0 M
- GPIO_InitStructure.GPIO_Speed = GPIO_Speed_100MHz;
# ]/ k9 }$ C! L. K5 Y) [1 Q! k7 R - GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_NOPULL;
' N& E; K# ], `) C9 d9 }7 Q1 c - GPIO_Init(GPIOH, &GPIO_InitStructure);5 ]- B, _' R& \' W
- % F1 |' {$ D( E
- SYSCFG_EXTILineConfig(EXTI_PortSourceGPIOH, EXTI_PinSource15); //PH15与EXTI15连接
' d9 o) T3 {' V. k4 W; U. ]
% p! k+ Y/ n8 U' Z' U4 C6 C- EXTI_InitStructure.EXTI_Line = EXTI_Line15; //设置连接中断线15
# [2 v; ?7 k7 @' ?! s - EXTI_InitStructure.EXTI_Mode = EXTI_Mode_Interrupt; //设置中断模式为中断) B ]9 d" {( Z# U& G" s: W
- EXTI_InitStructure.EXTI_Trigger = EXTI_Trigger_Falling; //下降沿触发
+ O6 ~1 A; t% j! a7 i: C - EXTI_InitStructure.EXTI_LineCmd = ENABLE;
$ H+ B( b* B2 M5 Q7 [& S - EXTI_Init(&EXTI_InitStructure);
, w" W( V/ |$ F3 ?4 Z, U - 1 T. q' q2 Y: B3 ^2 z( e
- //设置中断优先级分组. v, [1 o- c1 l3 S1 X
- NVIC_InitStructure.NVIC_IRQChannel = EXTI15_10_IRQn;
( z& B4 N0 ]0 v3 [7 h0 a$ B - NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0x0F;. ~, h0 D* Z ~$ K$ y
- NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0x0F;
# ] a# [' y1 Z - NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
+ I* E. k* [9 k, W0 M; B( D
R8 O' P7 H% z" H4 w5 U2 U- NVIC_Init(&NVIC_InitStructure);
7 V( j f I2 z# X- ?4 ?+ d
: e5 H; c! n0 u: Y/ d- return 0;
0 V5 k+ l* n" k7 S3 G - }* l" J8 ~5 q2 z2 j. d m
- H; m+ o8 ?: o. q8 U$ P. K- /*
' x: F. ~5 E8 |/ ^: p# {# t - * Name : EXTI15_10_IRQHandler& L7 z% b1 T) W# R
- * Description : ---
7 j! n1 }4 r. z8 S - * Author : ysloveivy.
/ e+ l- D# A# K2 d - *
& w1 ]: k& x; y) M: ] - * History
$ B2 ~# T$ t" { - * -------------------3 B# O8 \7 i) O
- * Rev : 0.00, v' w# @ d3 o, h
- * Date : 11/21/2015' [, s% F: Q+ {6 b7 Q1 t
- * 2 J& Y# U" f6 M/ x+ b3 M8 \8 M
- * create.
2 h1 u& J5 B7 [; P! x# p - * -------------------
- C. A2 t7 d5 k9 m" D- E - */
; O5 W) P" v) }& H - void EXTI15_10_IRQHandler(void)
$ G' K( X% i9 j% I - {' f6 k) u5 t! Z# ?3 d; C e
- int i;
1 `7 \2 R( |! w8 {# @) h$ X - static int work_status = 0;$ O" l+ q# ?' v
- if(EXTI_GetITStatus(EXTI_Line15) != RESET)' h* G7 k; X( u
- {
3 b% D0 m7 a4 U5 }9 @ - for(i = 0;i < 1000;i++);) r1 M# B/ b. p% T
- if(!KEY_INPUT){
6 T. v4 ? c0 T - work_status += 1;' v/ K' }% d; K' T
- if(work_status > 2)work_status = 0;8 r/ K/ r4 u {; c' x" P8 [
- }
1 ]; ]6 J4 [5 l5 Z, V A - switch(work_status){
: Z/ k: [) [: O' s& Z" Z - case 0:
6 ]- ?2 Y+ \: G/ u) z4 b @9 k - LED_RED_ON;
# y: N+ z$ F \- g& i* l0 {" d - LED_GREEN_OFF;6 M) `8 t' q1 A# c0 {' i& Y$ H
- LED_BLUE_OFF;* g5 c: s( U1 |4 q& r6 W
- break;7 q2 `6 n* |6 b0 u$ [1 s
- case 1:: Q; |4 J' z! O9 ]) d9 ^
- LED_RED_OFF;
/ B/ @) d1 i+ F- \6 m - LED_GREEN_ON;
5 m D! ?" p) j: r! O) V - LED_BLUE_OFF;
8 b6 a' B8 w1 w( n5 g1 | - break;7 E! T1 C+ `' I$ I$ z
- case 2:- P, G9 x5 ^0 c& c0 y7 Y
- LED_RED_OFF;
" O2 J" \3 T* Q3 z& P5 B+ c - LED_GREEN_OFF;
8 u+ ?8 U. E* n( i4 D4 x- D - LED_BLUE_ON;
2 K* A. C& c3 s - break;$ g5 ?- z$ @- E9 g' U) w
- default:
F; a. j" H( I8 J$ s
, M' \! J" L0 B1 h$ o- break;4 I4 [- ?9 M5 G! O) p
- }
4 V7 }& V5 N, J" X* { - EXTI_ClearITPendingBit(EXTI_Line15);3 v9 R" f R: K8 Z
- }) z8 s% H3 }$ G
- }
复制代码3、小知识: 设置中断和优先级主要用到以下两个结构体: 优先设置结构体: - typedef struct
* j1 y8 P5 m& d2 |* M( U$ G6 u - {
& z m# j" L/ n2 _+ j7 A, I% Q - uint32_t EXTI_Line; //设置连接中断线+ N2 V/ [, w+ T' m; }1 o9 \4 q
- EXTIMode_TypeDef EXTI_Mode; //设置中断模式
! U+ B* t7 h, ?2 ?- c0 W- X - EXTITrigger_TypeDef EXTI_Trigger; //触发方式 : n& ]5 T9 q3 _
- FunctionalState EXTI_LineCmd; //使能与否+ e. Y) ]" f' s, ^6 q
8 Z4 q8 E% |5 e' J% o- }EXTI_InitTypeDef;
复制代码 优先级设置结构体: - typedef struct
1 K3 v) T" L; ]( D Q# c - {: @' g% j& N6 h* q
- uint8_t NVIC_IRQChannel; //嵌套向量中断控制器通道
9 m; u# B4 w! F- c7 Q7 T- P5 p
/ v9 \' p+ K2 G, ^, x! Z1 v- uint8_t NVIC_IRQChannelPreemptionPriority; //抢占优先级% f7 P' e7 f' R) f( U+ n; u W
" y8 V1 R4 L1 F5 U2 q- uint8_t NVIC_IRQChannelSubPriority; //相应优先级
/ e4 t% C& P* y2 `4 U2 j) ^9 v - F# B0 B8 x2 P0 S" p2 @, r/ i
- FunctionalState NVIC_IRQChannelCmd; //使能与否
* m/ C v, Y" V1 Z. G0 _ - } NVIC_InitTypeDef;
复制代码 可以通过以上两个结构体来配置中断和优先级。 五、实验现象 按键每按下一次,三色LED灯变换(红色、绿色、蓝色轮流变换)一次。 六、代码包下载 . S8 v: D' M( I% w8 K9 {" v6 y
1 ~- r( C) W% @. p% Z
6 `! v; I4 u2 v7 H5 P' N5 {
# W Z) u& P0 q
1 R, A4 @5 `6 w7 W( q3 v% q
& H& F: T- B3 ^# |
2 y% f6 n g2 x, C/ Z |