一、什么是PendSV PendSV是可悬起异常,如果我们把它配置最低优先级,那么如果同时有多个异常被触发,它会在其他异常执行完毕后再执行,而且任何异常都可以中断它。更详细的内容在《Cortex-M3 权威指南》里有介绍,下面我摘抄了一段。
OS 可以利用它“缓期执行”一个异常——直到其它重要的任务完成后才执行动 作。悬起 PendSV 的方法是:手工往 NVIC的 PendSV悬起寄存器中写 1。悬起后,如果优先级不够 高,则将缓期等待执行。 PendSV的典型使用场合是在上下文切换时(在不同任务之间切换)。例如,一个系统中有两个就绪的任务,上下文切换被触发的场合可以是:
0 v( e+ |1 B2 I+ f' W1、执行一个系统调用
' T) h) f1 ~# o- w9 ]9 e0 d2、系统滴答定时器(SYSTICK)中断,(轮转调度中需要) 让我们举个简单的例子来辅助理解。假设有这么一个系统,里面有两个就绪的任务,并且通过SysTick异常启动上下文切换。但若在产生 SysTick 异常时正在响应一个中断,则 SysTick异常会抢占其 ISR。在这种情况下,OS是不能执行上下文切换的,否则将使中断请求被延迟,而且在真实系统中延迟时间还往往不可预知——任何有一丁点实时要求的系统都决不能容忍这 种事。因此,在 CM3 中也是严禁没商量——如果 OS 在某中断活跃时尝试切入线程模式,将触犯用法fault异常。 为解决此问题,早期的 OS 大多会检测当前是否有中断在活跃中,只有在无任何中断需要响应 时,才执行上下文切换(切换期间无法响应中断)。然而,这种方法的弊端在于,它可以把任务切 换动作拖延很久(因为如果抢占了 IRQ,则本次 SysTick在执行后不得作上下文切换,只能等待下 一次SysTick异常),尤其是当某中断源的频率和SysTick异常的频率比较接近时,会发生“共振”, 使上下文切换迟迟不能进行。现在好了,PendSV来完美解决这个问题了。PendSV异常会自动延迟上下文切换的请求,直到 其它的 ISR都完成了处理后才放行。为实现这个机制,需要把 PendSV编程为最低优先级的异常。如果 OS检测到某 IRQ正在活动并且被 SysTick抢占,它将悬起一个 PendSV异常,以便缓期执行 上下文切换。 使用 PendSV 控制上下文切换个中事件的流水账记录如下: 1. 任务 A呼叫 SVC来请求任务切换(例如,等待某些工作完成) 2. OS接收到请求,做好上下文切换的准备,并且悬起一个 PendSV异常。 3. 当 CPU退出 SVC后,它立即进入 PendSV,从而执行上下文切换。 4. 当 PendSV执行完毕后,将返回到任务 B,同时进入线程模式。 5. 发生了一个中断,并且中断服务程序开始执行 6. 在 ISR执行过程中,发生 SysTick异常,并且抢占了该 ISR。 7. OS执行必要的操作,然后悬起 PendSV异常以作好上下文切换的准备。 8. 当 SysTick退出后,回到先前被抢占的 ISR中,ISR继续执行 9. ISR执行完毕并退出后,PendSV服务例程开始执行,并且在里面执行上下文切换 10. 当 PendSV执行完毕后,回到任务 A,同时系统再次进入线程模式。
我们在uCOS的PendSV的处理代码中可以看到: - OS_CPU_PendSVHandler
/ {2 ?8 x% |6 u0 G' @8 E - CPSID I ; 关中断3 D/ P; t" C' `8 V5 J
- ;保存上文 ' i3 n* m8 A0 ` c3 j) V
- ;....................... * z; ?0 i' L8 y$ b: g( U9 N& P
- ;切换下文 0 V& y% I. R; P8 W9 A
- CPSIE I ;开中断& w9 X: O6 u$ h
- BX LR ;异常返回
复制代码 & \( R3 A. c2 O9 x
它在异常一开始就关闭了中端,结束时开启中断,中间的代码为临界区代码,即不可被中断的操作。PendSV异常是任务切换的堆栈部分的核心,由他来完成上下文切换。PendSV的操作也很简单,主要有设置优先级和触发异常两部分: - NVIC_INT_CTRL EQU 0xE000ED04 ; 中断控制寄存器
/ p& e l4 E& ]+ U1 J - NVIC_SYSPRI14 EQU 0xE000ED22 ; 系统优先级寄存器(优先级14).
( D2 q8 W) a2 |5 g# @. K - NVIC_PENDSV_PRI EQU 0xFF ; PendSV优先级(最低). NVIC_PENDSVSET EQU 0x10000000 ; PendSV触发值2 K8 T& r/ j) @ L8 y) l; w: \: D
- : @! ?$ ~. Z: R! f
- ; 设置PendSV的异常中断优先级
# Q! y# a4 g4 b% p
- j \# c% D* p" D M( j; F6 x7 D- LDR R0, =NVIC_SYSPRI14
5 n9 p4 w# b9 R# B - LDR R1, =NVIC_PENDSV_PRI # F8 o7 Z0 X: R3 d7 d" Y7 ^
- STRB R1, [R0] ; 触发PendSV异常& l0 ^* m. i+ z4 p d9 d5 k; B# J
- LDR R0, =NVIC_INT_CTRL ' D- X1 {/ q/ K3 b/ p3 J
- LDR R1, =NVIC_PENDSVSET % i7 x: Y5 b) B, M
- STR R1, [R0]
复制代码
: c* Y, j( j2 T6 I1 J 二、堆栈操作 Cortex M4有两个堆栈寄存器,主堆栈指针(MSP)与进程堆栈指针(PSP),而且任一时刻只能使用其中的一个。MSP为复位后缺省使用的堆栈指针,异常永远使用MSP,如果手动开启PSP,那么线程使用PSP,否则也使用MSP。怎么开启PSP? - MSR PSP, R0 ; Load PSP with new process SP! f* S N: F/ l l! C0 [* f4 \
- ORR LR, LR, #0x04 ; Ensure exception return uses process stack
复制代码
1 {; J7 r ]* s2 d7 r! i; u* u很容易就看出来了,置LR的位2为1,那么异常返回后,线程使用PSP。 写OS首先要将内存分配搞明白,单片机内存本来就很小,所以我们当然要斤斤计较一下。在OS运行之前,我们首先要初始化MSP和PSP, - EXTERN OS_CPU_ExceptStkBase m0 ~- d/ h$ s6 L3 Y
- ;PSP清零,作为首次上下文切换的标志: k( {! I% Z) W1 g) }4 @/ ^
- MOVS R0, #0
6 w. v% ]- d3 Z& d! f - MSR PSP, R0; j4 U9 g0 u& Q x0 [+ T
- ;将MSP设为我们为其分配的内存地址8 j0 @- _7 Y' v$ Z% _, U+ v4 }1 y
- LDR R0, =OS_CPU_ExceptStkBase/ g, w- e4 ?/ P
- LDR R1, [R0]
8 I1 j# f# a: n$ _. ~9 J - MSR MSP, R1
复制代码 4 n2 T$ F" `* R( R" e# z% s5 ?' t8 ~, Q
然后就是PendSV上下文切换中的堆栈操作了,如果不使用FPU,则进入异常自动压栈xPSR,PC,LR,R12,R0-R3,我们还要把R4-R11入栈。如果开启了FPU,自动压栈的寄存器还有S0-S15,还需吧S16-S31压栈。 - MRS R0, PSP3 Y6 |6 T2 y! n# c
- SUBS R0, R0, #0x20 ;压入R4-R11
7 v0 C% F* ^- \ ~1 W' M: p - STM R0, {R4-R11}2 m) s5 X, k5 v& P1 v# j8 s
+ ?' \/ M# Z$ f! H; X, f+ D- LDR R1, =Cur_TCB_Point ;当前任务的指针
2 D2 m; i3 a& H1 |: v8 U2 [% C* H - LDR R1, [R1]& }$ b( u$ X. y! N0 L; j4 {' J
- STR R0, [R1] ; 更新任务堆栈指针
复制代码
$ `* k3 i) h; u* C/ ^# p出栈类似,但要注意顺序 - LDR R1, =TCB_Point ;要切换的任务指针3 ^! Y- E( L: o; J, F% B( _
- LDR R2, [R1]" Y2 z. q v& T r! t) B
- LDR R0, [R2] ; R0为要切换的任务堆栈地址, e1 k. o7 H* X9 Z0 d! B4 |
-
% g1 C) [' O, O! H/ b - LDM R0, {R4-R11} ; 弹出R4-R11! U" d) f/ Z; t
- ADDS R0, R0, #0x20; |0 X) s" ~0 L
- 1 V, R0 V$ {: Z; L3 b# `
- MSR PSP, R0 ;更新PSP
复制代码 6 j% K7 n0 O8 m" K, z* @
三、OS实战 新建os_port.asm文件,内容如下: - NVIC_INT_CTRL EQU 0xE000ED04 ; Interrupt control state register.( y: K2 f# k& w& s- @
- NVIC_SYSPRI14 EQU 0xE000ED22 ; System priority register (priority 14).2 f0 n) ^* \! F7 ]
- NVIC_PENDSV_PRI EQU 0xFF ; PendSV priority value (lowest).: [# W6 @# x; m7 X' `3 x. R
- NVIC_PENDSVSET EQU 0x10000000 ; Value to trigger PendSV exception.% A5 X7 Z% a5 J3 q; t
- 9 ~; ` c1 Z/ I
- RSEG CODE:CODE:NOROOT(2)! X4 j2 E- l0 o7 O$ ?) n. ^: M
- THUMB0 P E, M4 _% ^% B- a
/ Y) a& p, M8 i& J! I3 L- $ Z5 R9 G- c: O
- EXTERN g_OS_CPU_ExceptStkBase5 s4 z* _( O4 d8 p# I4 a* k% x( G
-
$ A' A# S, S/ K8 m9 [: ~ - EXTERN g_OS_Tcb_CurP
& `; v. X1 t: ~' C% k+ q - EXTERN g_OS_Tcb_HighRdyP
* Z2 Q% {7 o. c+ K - # p6 A/ R! j; n! ?2 W$ t
- PUBLIC OSStart_Asm! [6 \3 M4 x; K
- PUBLIC PendSV_Handler5 M9 S' F: k- x% ?: e& D4 v
- PUBLIC OSCtxSw" s4 e/ v8 ?7 I8 o
& E N0 V# }+ c* H! n8 S, n6 h- OSCtxSw2 {* V+ @& z0 l9 D# X
- LDR R0, =NVIC_INT_CTRL; f: ~- ]' f; C
- LDR R1, =NVIC_PENDSVSET) D; ^; t4 Q9 g* Z0 s: h z0 \1 O
- STR R1, [R0]) r( g6 l# J: q( v- M* e
- BX LR ; Enable interrupts at processor level
6 A# F: t6 b# V6 w/ V: h4 K; d
: @" {- w: P0 ~7 Q( @' {- OSStart_Asm8 s9 s& j6 K/ k+ P. B
- LDR R0, =NVIC_SYSPRI14 ; Set the PendSV exception priority
+ |" i& \: A; S- r' S - LDR R1, =NVIC_PENDSV_PRI
" G8 E2 b, T& ]9 h0 {# _# z - STRB R1, [R0]* B8 n* @0 T s7 j8 ^
- & o {* K' Y! H3 O8 Z
- MOVS R0, #0 ; Set the PSP to 0 for initial context switch call# y( \; j) _6 g
- MSR PSP, R0
: R* `6 P# M8 C, h( T& _, i
6 N! T1 v' W! q' m, m% s( a- LDR R0, =g_OS_CPU_ExceptStkBase ; Initialize the MSP to the OS_CPU_ExceptStkBase
8 X$ `9 M! o) c! @8 U/ X - LDR R1, [R0]
{* Q) Q& s- A6 o" w c' Z - MSR MSP, R1 4 z+ }* w" e! Q) _
- & ~3 ]; r" _) `( f
- LDR R0, =NVIC_INT_CTRL ; Trigger the PendSV exception (causes context switch)
/ M9 N$ J9 \/ @ - LDR R1, =NVIC_PENDSVSET
* S# b: e& S4 ~% G+ `$ H+ H - STR R1, [R0]
, ~2 b/ W( m1 E( }' X - ; ]' U, j6 V3 D+ i$ U" o
- CPSIE I ; Enable interrupts at processor level
# S" e6 s2 d+ S9 P4 L# l - 4 d$ |0 H' ^( K- v- x$ I
- OSStartHang
0 H$ |4 _) v1 M, O7 O - B OSStartHang ; Should never get here
2 t; u- D6 V0 f$ I0 U( w0 F" _ - % W& q4 t2 ~- N6 w, V6 K) o" U
-
1 Z8 H( m ^; f$ V% d, B
/ m0 m- M! I; [% r- PendSV_Handler
3 ?8 E/ K8 @* S8 W& s - CPSID I ; Prevent interruption during context switch
7 b& C/ d* `' P- j - MRS R0, PSP ; PSP is process stack pointer0 U7 n8 |2 {+ M, v8 E b' B" n0 u# O
- CBZ R0, OS_CPU_PendSVHandler_nosave ; Skip register save the first time; k+ I6 g. _- \3 Q: Q
-
8 U# H$ c, o) K# i" x0 \* H) I - SUBS R0, R0, #0x20 ; Save remaining regs r4-11 on process stack
/ ?2 F1 J: t- k - STM R0, {R4-R11}
0 \, t; q: C8 B' E- G& g/ v
9 l8 s! ^- G- x8 u7 B- LDR R1, =g_OS_Tcb_CurP ; OSTCBCur->OSTCBStkPtr = SP;; x9 Z3 ^- }" t; g# J2 |% s$ F
- LDR R1, [R1]
' j! e# E3 I) ?5 I( w3 b L( F$ R, t - STR R0, [R1] ; R0 is SP of process being switched out. r7 U: [1 y' X% ^! v6 {; a- O8 y
2 u$ O. T: }" M, w9 d( \7 |- ; At this point, entire context of process has been saved7 G% k; T) S5 `7 G
- OS_CPU_PendSVHandler_nosave
7 K1 t$ H. n+ Z* V - LDR R0, =g_OS_Tcb_CurP ; OSTCBCur = OSTCBHighRdy;; d' B, ?8 x5 p
- LDR R1, =g_OS_Tcb_HighRdyP
; c% W: ?- \# B! a; f' { - LDR R2, [R1]5 _# d3 D+ T) {. |$ L
- STR R2, [R0]
* i7 [% r0 N. _ |0 {5 |
2 J6 v/ s4 r8 J2 ? x; j- LDR R0, [R2] ; R0 is new process SP; SP = OSTCBHighRdy->OSTCBStkPtr;
; U+ l( z& y# Q -
- n$ o+ | R( a- F1 X6 h! Q: Y. F; a$ g - LDM R0, {R4-R11} ; Restore r4-11 from new process stack
|* f' u+ Y2 L+ S# S! x - ADDS R0, R0, #0x200 C2 U8 M+ h9 @# `
-
! d* R4 F) l( B" C) ~ - MSR PSP, R0 ; Load PSP with new process SP& o5 e% w/ V7 [1 }+ f, X# w+ I
- ORR LR, LR, #0x04 ; Ensure exception return uses process stack" Y' R6 a4 ~3 D+ ~/ H! q5 E
- ' M: L+ f" @: H0 F
- CPSIE I
9 l1 G' U. c+ Z! L: \ - BX LR ; Exception return will restore remaining context9 p# r: g' d$ M( C/ r. s
- + P4 a' g, h: V
- END
复制代码
+ P3 K2 A- f, \0 y% U+ Rmain.c内容如下: - #include "stdio.h"5 l; ?! F- z2 C& i6 X6 D* J2 e
- #define OS_EXCEPT_STK_SIZE 1024( o1 c/ B( Y9 y5 k* I1 @1 K8 Z' g5 B
- #define TASK_1_STK_SIZE 1024
3 y( f& x5 L2 g& H. |0 D% G - #define TASK_2_STK_SIZE 1024' s: x7 G7 I" H8 w
/ @- r7 Q: j/ ~! ~& p1 Q8 ]( I- typedef unsigned int OS_STK;2 O1 s- E1 v% ?+ B/ P9 L
- typedef void (*OS_TASK)(void);& s1 O, t/ O$ \: A- g ]/ A. n
- - y) B" p* ?/ a, f
- typedef struct OS_TCB
7 H5 s: S9 M; X, _; N; f - {! q+ c. C. s' T1 y- D( x: _1 J
- OS_STK *StkAddr;
# }. S4 h! Q- \, F6 b - }OS_TCB,*OS_TCBP;
* [5 c$ r, |0 @ o - ' \2 ?/ q. K1 G# w. \7 Z
5 _4 G: q: g$ ], H( t: Y- OS_TCBP g_OS_Tcb_CurP; ) E6 y) \/ O$ V4 P+ O4 l
- OS_TCBP g_OS_Tcb_HighRdyP;
* _9 n9 N* ?' g - 7 x J. x1 H3 H9 z3 y# q7 T
- static OS_STK OS_CPU_ExceptStk[OS_EXCEPT_STK_SIZE];
) x- r( ]# L/ r- o, }0 U" X - OS_STK *g_OS_CPU_ExceptStkBase;
: H5 n G3 X- Q1 Z, i# N3 p" s
! ~& m5 I( t- j- static OS_TCB TCB_1;9 m Q# W% X" {. ^9 |
- static OS_TCB TCB_2;2 q x8 m8 {) P! F# e( ~
- static OS_STK TASK_1_STK[TASK_1_STK_SIZE];
8 W3 J+ m- ^, W' h: Z1 j - static OS_STK TASK_2_STK[TASK_2_STK_SIZE];) S/ K4 @9 [" S' |# y
7 H( p# c9 f* b9 D" Z: |0 {9 a- extern void OSStart_Asm(void);
% z' ~5 p! E6 s" K J. K - extern void OSCtxSw(void);) ~1 f: C5 B' A! P7 h) |, b
- ! R/ V) t7 |5 I3 ]
- void Task_Switch()
3 `# _" Y/ h3 O# J* D - {$ l) b! z+ f; e- v8 T- n) L
- if(g_OS_Tcb_CurP == &TCB_1)# O/ y7 B/ k+ F9 U; z: ?
- g_OS_Tcb_HighRdyP=&TCB_2;, p( o5 M m: F* |, x2 m: c
- else
0 u: H! R+ ~' ] - g_OS_Tcb_HighRdyP=&TCB_1;) u9 n1 T: n$ n4 c2 Q
-
$ p5 C0 z# B3 I - OSCtxSw();
% E8 n; O% I$ j& O1 ?& _ y - }
4 d) t; J% I0 W w- M& Q4 D, u - . _9 L3 B0 U: K: p( m# A H
- 8 d; U- f" P2 j# q7 P4 D* n
- void task_1(); t: f0 B |+ x+ S' w
- {3 |6 b/ l/ L7 s8 _ Q3 K
- printf("Task 1 Running!!!\n");
" x; C. T4 E" C: q - Task_Switch();* _- `! f+ d& I( H/ d
- printf("Task 1 Running!!!\n");5 i' V9 k7 D0 I& |6 W
- Task_Switch();
1 n# f! Q0 C, e! _ z - }
) t4 z( w" E' H5 u: V1 R2 }% n
; r9 o4 u- V. E |7 K3 r3 c- void task_2()
& S% v9 ]' P" y, k - {
/ V) X, X: `6 r7 I -
% d) v, W! o7 k" K. ^2 z - printf("Task 2 Running!!!\n");
# Y5 p& j, e; U \7 o! } - Task_Switch();: ~0 o+ b/ O1 F
- printf("Task 2 Running!!!\n");
& Q6 r, ~! |: L3 D1 P I - Task_Switch();
" r9 ]4 Y6 a/ r* ]6 R. b% q0 r$ _ - }
# b; @9 N! B$ }& f: O' ^4 J
2 q7 ~" f9 V2 t2 ]- j* c- void Task_End(void)5 o- \ \" J( g6 W1 E
- {5 M( d e; n5 S2 e8 q% k
- printf("Task End\n");
- J; `5 j2 n7 f$ Q - while(1)
% `/ c6 F2 U, t! {2 ? - {}3 f7 i5 y5 Z, j3 O5 e
- }3 C% C& B7 f! g
6 A( v& M1 P; q0 ?- void Task_Create(OS_TCB *tcb,OS_TASK task,OS_STK *stk)
0 ]# y5 a& S8 [; t, R - {
8 [, S% ?) {: l7 w# C" I - OS_STK *p_stk;
9 ~! C/ y0 H6 u# g* c v - p_stk = stk;; [# K0 [# t$ R8 ~" D5 q
- p_stk = (OS_STK *)((OS_STK)(p_stk) & 0xFFFFFFF8u);4 G" p K @& t! G+ T
- - [' c( A, F1 \: H$ I- m- k) K
- *(--p_stk) = (OS_STK)0x01000000uL; //xPSR
7 D; G! n7 Y# l3 @8 Y- k6 h) i - *(--p_stk) = (OS_STK)task; // Entry Point
+ E- p8 d* j) v* w* z, j2 o3 Q - *(--p_stk) = (OS_STK)Task_End; // R14 (LR)
) f4 c# y* r$ E8 s, {- [ - *(--p_stk) = (OS_STK)0x12121212uL; // R12% W2 Z# ?, U9 n2 ?
- *(--p_stk) = (OS_STK)0x03030303uL; // R3
4 V( ]. r) |; d/ [) u3 ]; M - *(--p_stk) = (OS_STK)0x02020202uL; // R2) l2 m. A5 x; |- k7 n# j/ b
- *(--p_stk) = (OS_STK)0x01010101uL; // R1
, k2 U+ n# F% l& q$ ~: U - *(--p_stk) = (OS_STK)0x00000000u; // R0
! x$ g# `9 E7 U- D -
0 K- |1 D5 z# n8 q* I+ w - *(--p_stk) = (OS_STK)0x11111111uL; // R113 x3 T7 \9 n4 L! d
- *(--p_stk) = (OS_STK)0x10101010uL; // R10
2 ?0 n+ D) L% c; g( C4 M$ ? - *(--p_stk) = (OS_STK)0x09090909uL; // R9
& D* O6 J* s& l' A3 m - *(--p_stk) = (OS_STK)0x08080808uL; // R8
! W. Y1 ?2 o4 g7 ?: w: u% k - *(--p_stk) = (OS_STK)0x07070707uL; // R7
, r) @$ G2 ~7 Q6 q) N - *(--p_stk) = (OS_STK)0x06060606uL; // R66 R _. M* n4 ~: o% B9 V$ j
- *(--p_stk) = (OS_STK)0x05050505uL; // R5
) H+ {% i3 R) n: M! B9 m - *(--p_stk) = (OS_STK)0x04040404uL; // R4- m' F0 w1 n7 W7 F
- 9 U) [+ U0 L; m
- tcb->StkAddr=p_stk;6 `3 W2 Y' e: n" ]3 {
- }$ H, O8 K7 h& l1 P+ X3 f8 c
- / B7 [8 _& p/ l
- x7 D. r9 V+ \5 N- int main(), ~8 |( B+ [0 Z. b9 q8 D3 d; ?, B
- {
' q; P* F2 v& x - 9 }& X' y: P; u1 R) ^$ g
- g_OS_CPU_ExceptStkBase = OS_CPU_ExceptStk + OS_EXCEPT_STK_SIZE - 1;
9 R6 h$ {) t1 C- V' ~4 J" H - ; }8 ^+ i. `' y1 @% i2 z
- Task_Create(&TCB_1,task_1,&TASK_1_STK[TASK_1_STK_SIZE-1]);
* u3 N) U$ U: q0 I" J - Task_Create(&TCB_2,task_2,&TASK_2_STK[TASK_1_STK_SIZE-1]);0 O$ h7 p7 q* W( `& P; [ P7 N
- ) Q$ D1 Q' s' D5 s& f
- g_OS_Tcb_HighRdyP=&TCB_1;
% O* J; q0 K( E' D' V- O - W9 H) E9 x- A/ i: |
- OSStart_Asm();
# p9 h& D, ]: U6 f - 7 T0 G: X; ?7 H6 \# q- _+ T" |
- return 0;; t- S9 G2 c$ Z8 X
- }
复制代码
! k' d: E- S% g编译下载并调试:
: C4 k6 P( R5 p7 g2 O2 b4 W" T1 t' o
3 `( b4 ]: L$ i* v4 X5 ^2 e
在此处设置断点 此时寄存器的值,可以看到R4-R11正是我们给的值,单步运行几次,可以看到进入了我们的任务task_1或task_2,任务里打印信息,然后调用Task_Switch进行切换,OSCtxSw触发PendSV异常。
/ M" D- ]! b% C# t6 \. A& K
+ `2 `, A* J! ?2 o; v! Y$ q
IO输出如下: ) N' `9 @2 t( y: Y
* a/ T1 c- |2 D0 N- y, m: y
至此我们成功实现了使用PenSV进行两个任务的互相切换。之后,我们使用使用SysTick实现比较完整的多任务切换。 : I- T+ P7 O# h* C' |
|