你的浏览器版本过低,可能导致网站不能正常访问!
为了你能正常使用网站功能,请使用这些浏览器。

STM32 NVIC与中断控制

[复制链接]
STMCU小助手 发布时间:2022-12-29 17:00
STM32 NVIC与中断控制学习
9 Y$ D9 }3 K, e0 d. ^" F. H1 c+ r- K) eNVIC全称向量中断控制器,NVIC 共支持 1 至 240 个外部中断输入(通常外部中断写作 IRQs)。NVIC 还支持一个“永垂不朽”的不可屏蔽中断(NMI)输入。NVIC 的访问地址是 0xE000_E000。所有 NVIC 的中断控制/状态寄存器都只能在特权级下访问。不过有一个例外——软件触发中断寄存器可以在用户级下访问以产生软件中断。所有的中断控制/状态寄存器均可按字/半字/字节的方式访问。此外,有几个中断屏蔽寄存器也与中断控制密切相关,它们是第三章中讲到的“特殊功能寄存器”,只能通过 MRS/MSR及 CPS 来访问。4 s- M7 }0 R6 F0 `

1 K& n( B5 d) D! I! ~1 U中断配置基础. x" M$ u0 S8 m+ f# v1 b
每个外部中断都在 NVIC 的下列寄存器中“挂号”:, ^2 M2 Z  B$ K% ~9 A* ^
使能与除能寄存器, s9 |/ {7 {3 |# _+ ~/ ~
悬起与“解悬”寄存器7 ~6 ?7 _) |% ~0 a% b& m; {! E) C: r
优先级寄存器* U  _! b# m$ Z: U2 F! k; ~
活动状态寄存器& Q  Y2 o/ T! W! o
中断的使能与除能中断的使能与除能分别使用各自的寄存器来控制,CM3 中可以有 240 对使能位/除能位,每个中断拥有一对。这240 个对子分布在 8 对 32 位寄存器中(最后一对没有用完)。欲使能一个中断,你需要写 1 到对应 SETENA 的位中;欲除能一个中断,你需要写 1 到对应的 CLRENA 位中;如果往它们中写 0,不会有任何效果。通过这种方式,使能/除能中断时只需把“当事位”写成1,其它的位可以全部为零。再也不用像以前那样,害怕有些位被写入0 而破坏其对应的中断设置(写 0 没有效果),从而实现每个中断都可以自顾地设置,而互不侵犯——只需单一的写指令,不再需要读‐改‐写。如上所述,SETENA 位和 CLRENA 位可以有 240 对,对应的 32 位寄存器可以有8 对,因此使用数字后缀来区分这些寄存器,如 SETENA0, SETENA1…SETENA7,如表 8.1 所示。但是在特定的芯片中,只有该芯片实现的中断,其对应的位才有意义。因此,如果你使用的芯片支持 32 个中断,则只有SETENA0/CLRENA0 才需要使用。SETENA/CLRENA 可以按字/半字/字节的方式来访问。又因为前 16 个异常已经分配给系统异常,故而中断 0 的异常号是 16。7 P: ^  e/ j9 D9 S3 @! k

% Y! Y8 S, A$ D$ [/ g* [
ba82ea5527534e779a8be89abf0b59c3.png & Y& q- a4 `! _

7 v+ ~  L: |5 N' l2 r2 f中断的悬起与解悬
* x# q9 @5 a8 Y3 d1 I1 S' C+ S如果中断发生时,正在处理同级或高优先级异常,或者被掩蔽,则中断不能立即得到响应。此时中断被悬起。
. }; Z# h( t# S: h3 z$ B中断的悬起状态可以通过“中断设置悬起寄存器(SETPEND)”和“中断悬起清除寄存器(CLRPEND)”来读取,还可以写它们来手工悬起中断。悬起寄存器和“解悬”寄存器也可以有 8 对,其用法和用量都与前面介绍的使能/除能寄存器完全相同。
. d* k* k5 K8 y0 j2 p- C  n+ K' ?0 Q, k. q& E  [: l
8794fa80a10448b1bd24cd29a4278a8e.png
8 u6 h/ D+ r) ^: L/ e0 U

% d1 k6 ~- ^; k3 ?2 D优先级. r5 o# `9 [5 j# Z
每个外部中断都有一个对应的优先级寄存器,每个寄存器占用 8 位,但是允许最少只使用最高 3 位。4 个相临的优先级寄存器拼成一个 32 位寄存器。如前所述,根据优先级组设置,优先级可以被分为高低两个位段,分别是抢占优先级和亚优先级。优先级寄存器都可以按字节访问,当然也可以按半字/字来访问。有意义的优先级寄存器数目由芯片厂商实现的中断数目决定。+ m& A! Z' d, y5 C% R' D) h

* x% y4 Q8 d% [% g1 A% N) F. Z
236a6aa8042d40a593a2b76c0b3ecb9c.png * S( B+ e/ G& b; }
5 }0 v8 b1 w( J9 Z% f
活动状态
9 Q# S: i" d6 c. Q4 D5 B
每个外部中断都有一个活动状态位。在处理器执行了其 ISR 的第一条指令后,它的活动位就被置 1,并且直到 ISR 返回时才硬件清零。由于支持嵌套,允许高优先级异常抢占某个ISR。然而,哪怕一个中断被抢占,其活动状态也依然为 1(请仔细琢磨前文讲到的“直到 ISR返回时才清零)。活动状态寄存器的定义,与前面讲的使能/除能和悬起/解悬寄存器相同,只是不再成对出现。它们也能按字/半字/字节访问,但他们是只读的。
( ?* E0 e" j3 V9 K6 w
# G9 v6 }* g7 ]5 }! V
30101b011f26409cb60809d807d93cf2.png
- X  J- M& q: e) U1 F& \
5 r1 j* r: P' h* {( d% ^
这里提一下这个优先级分组,这个优先级分组寄存器SBC->AIRCR作用就是指定NVIC中抢占优先级和次优先级分别占几位,如果是grop1,那么就表示抢占优先级占1位,子优先级占3bit
, }! t- w$ n. t' L
, S! P8 t. T) ]; A
c4f1b550fd464ebfa00b791505ed73f0.png 4 S$ D. T1 ?2 e4 K

: N4 E- C. r3 r# `9 H, Z1 ?- o两个重要的库文件:core_cm4.h和misc.c
2 [7 W; ?2 ~# ]! j- j, [* i8 r
  1. typedef struct0 a  @) l4 ]  G( k$ {, G
  2. {( Q5 i% K1 z: C- [- a( h+ s
  3.   __IO uint32_t ISER[8];                 /*!< Offset: 0x000 (R/W)  Interrupt Set Enable Register           */8 Z* V3 s- S7 V: n
  4.        uint32_t RESERVED0[24];' P# O6 G: O' [
  5.   __IO uint32_t ICER[8];                 /*!< Offset: 0x080 (R/W)  Interrupt Clear Enable Register         */
    ; G) M) c6 J+ l$ @! G! ^! F7 b
  6.        uint32_t RSERVED1[24];. T6 M$ ~5 A" |! E
  7.   __IO uint32_t ISPR[8];                 /*!< Offset: 0x100 (R/W)  Interrupt Set Pending Register          */- [( R5 r' \9 F( \& i
  8.        uint32_t RESERVED2[24];( J  {. p6 p7 O5 j& k
  9.   __IO uint32_t ICPR[8];                 /*!< Offset: 0x180 (R/W)  Interrupt Clear Pending Register        */5 L& d) N/ G! C: t7 n7 E
  10.        uint32_t RESERVED3[24];$ B, n) X- e# [0 X! P
  11.   __IO uint32_t IABR[8];                 /*!< Offset: 0x200 (R/W)  Interrupt Active bit Register           */3 f+ q6 g( i: ?8 T- `/ t7 Q. V' M
  12.        uint32_t RESERVED4[56];
    + d. o! _! ~5 k
  13.   __IO uint8_t  IP[240];                 /*!< Offset: 0x300 (R/W)  Interrupt Priority Register (8Bit wide) */$ [* C) T  |  E
  14.        uint32_t RESERVED5[644];
    " k7 B" H$ x0 _1 ?5 e0 c
  15.   __O  uint32_t STIR;                    /*!< Offset: 0xE00 ( /W)  Software Trigger Interrupt Register     */( V9 \! X; n, `* @: E+ Y9 n. M
  16. }  NVIC_Type;. x7 z  X. T- {! A$ K, R/ t: x
复制代码

# C; b5 `7 k0 P/ {. h' p& T1 A. B8 p! i( b1 v9 f
以串口中断初始化例程说明:" l4 {3 j! ^2 q' F5 P
  1. NVIC_InitTypeDef NVIC_InitStructure;
    : f* r! J8 o5 B
  2.         NVIC_PriorityGroupConfig(NVIC_PriorityGroup_1);2 P( U, X  R9 d5 }. `
  3.        
    . [7 J6 n2 D0 i$ Y
  4.         NVIC_InitStructure.NVIC_IRQChannel=USART1_IRQn;
    # h7 `$ \( m5 G: k
  5.         NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority=0x00;7 q3 E$ _1 X1 z  _
  6.         NVIC_InitStructure.NVIC_IRQChannelSubPriority=0x02;+ p! b( c+ H. W3 g7 r/ L
  7.         NVIC_InitStructure.NVIC_IRQChannelCmd=ENABLE;
    7 m' v( L3 D- {5 G0 w( A* G
  8.         NVIC_Init(&NVIC_InitStructure);
    4 m. N$ Q+ D8 K
复制代码

4 }# n# x0 ]- W& b. g" g3 A  p所以总结流程如下:
4 D- y) `, P; B3 @8 w1-使能中断请求, f' P& X+ {# V
2-配置中断优先级分组(只需配置一次即可)
: F3 z, h' y/ I* [9 {3-配置NVIC寄存器,初始化NVIC_InitTypeDef;+ u8 G+ F; {" t8 \5 ~' {9 H2 E
4-编写中断服务函数9 [7 l5 z  Z' A6 B& w; {% G1 Y  \
————————————————
/ U6 _5 y& y  ], m* l$ L# C5 ~. ?: s: H版权声明:tony++
3 g- @( k) I0 K" ?$ e, f: D- ^7 s  U
% n/ H; E; `3 f  a
& ^- x7 }+ [2 e4 b+ @2 I
收藏 评论0 发布时间:2022-12-29 17:00

举报

0个回答
关于
我们是谁
投资者关系
意法半导体可持续发展举措
创新与技术
意法半导体官网
联系我们
联系ST分支机构
寻找销售人员和分销渠道
社区
媒体中心
活动与培训
隐私策略
隐私策略
Cookies管理
行使您的权利
官方最新发布
STM32N6 AI生态系统
STM32MCU,MPU高性能GUI
ST ACEPACK电源模块
意法半导体生物传感器
STM32Cube扩展软件包
关注我们
st-img 微信公众号
st-img 手机版