接上篇:RTOS超低功耗平台应用---FreeRTOS启动&任务创建1 O* \" t. X7 v7 @
/ G$ {7 ]% c1 @7 Q) w& A& L# ]# K' O
8 K& i, }% B& Q# h# X+ e' t8 X 首先了解一下信号量,一般信号量主要实现两个功能:
1 k1 G- v/ i: S- E* F! y: [ 1,两个任务之间或者中断函数跟任务之间的同步功能# ~8 V2 @2 e+ d. ?( t; `# d p. b
2,多个共享资源的管理8 ^0 c) D4 ~: S# E# {
FreeRTOS针对这两种功能分别提供了二值信号量和计数信号量,其中二值信号量可以理解成计数,即初始化为仅有一个资源可以使用。
2 t/ V7 Q$ u4 k- d F 实验是创建一个线程,该线程通过ISR发出的信号量打印按钮按下计数,包含一个LED闪烁任务。/ w* \4 O& G6 {$ }9 k
- /**' P, G* H& a) {; y8 l
- * @brief Start a Task.
* m& N& t* ?0 ~ - * @param argument: Not used" Q/ e% S$ }& m
- * @retval None
9 L& @$ X( I( j& ]( V; l - */
3 \, s4 o) r' J9 n% n& S+ g - static void vAppStartTask( void *pvParameters )
; e& [8 e( i( J4 P+ k; x' f - {7 r9 |9 z& K4 X7 ~( a& {! s
- (void) pvParameters;$ Y5 n/ N4 L/ F: f4 ?8 a
- 1 @6 |& n- Y E4 n
- for(;;)
7 v3 v5 j3 o4 }1 ]& {( g; v6 ~9 [9 q - {
6 ]' [% C( q* G9 [( w - BSP_LED_Toggle(LED2);5 P3 b' a8 d) q$ |& {4 A
-
$ b" g% |; j& i0 `. J - vTaskDelay(200);
% {( W# A& A" u) k+ v - }* u8 f- f9 H, e+ M+ c/ H* T. j2 Q
- }
0 J" h7 K0 ^4 a+ {: [
9 ]! v& Z$ l j- F [6 n- /**$ @) Y( X. X2 J/ K, ] m
- * @brief Button Task.
) E6 F2 V* R. K9 I% O - * @param argument: Not used) G" O) F) p6 q/ {7 }, o3 ]
- * @retval None: S& V" k/ x3 X8 e G
- */) h( [% N; m7 E2 k$ G+ P0 L
- static void prvAppButtonTask( void *pvParameters )
4 S4 l9 p: l' i1 f( f; ^ - {
7 Z0 A; Z$ l o5 H# Z7 d8 G' ^* \ J+ j - configASSERT( xTestSemaphore ); /* The assert function determines whether the semaphore is valid. */
, N {6 }: K* E+ p& P% p$ R4 S - $ O& H; y# o; R& |1 e9 P
- /* This is the task used as an example of how to synchronise a task with
# k9 \* j1 V+ ]: z) F) G - an interrupt. Each time the button interrupt gives the semaphore, this task
; u" `3 V; n0 D( z - will unblock, increment its execution counter, then return to block
) z$ q0 u2 ^+ v T; P! V: \ - again. */0 ^! a! M2 w0 @ q4 T% j+ l6 N- P+ K2 s
-
' F# b3 v. K! f: ?# \( w v8 G - /* Take the semaphore before started to ensure it is in the correct
: ?# _3 } F& Q; S6 y' s) V- L - state. */
1 C( K( z J! @9 k - xSemaphoreTake( xTestSemaphore, mainDONT_BLOCK );
! T6 ]) ~0 @# c) G+ K% R -
/ F$ k: d! X& m4 H% E( w - for( ;; )
2 P$ p* ` d; _$ Z - {& E: \5 E. g) O4 r- b& {2 W" l
- xSemaphoreTake( xTestSemaphore, portMAX_DELAY );
5 i" B2 ^1 i0 U( h8 A - ulButtonPressCounts++;
5 a: r+ ]' Z% ^8 _3 C) k - printf("Button Press Counts:%d\r\n",ulButtonPressCounts);7 E3 Y8 r) f( q. F' j+ i E2 s% l" l3 ?
- }/ y v( J& H% g
- }
- c7 D6 f+ [5 @& C
4 ]1 Z, `; P3 D: |* I0 [3 d- /**; x1 ?" W- [8 g7 Y
- * @brief EXTI line detection callback.
! }" Q g3 z7 t' t8 ?% u - * @param GPIO_Pin: Specifies the pins connected EXTI line
) w/ {* E" L8 g$ J4 r2 y) x - * @retval None
* Z4 H% l) {5 p - */3 T" C1 D6 Y- ?. J$ g0 O
- void HAL_GPIO_EXTI_Callback(uint16_t GPIO_Pin)9 S% B9 J5 t3 A" f: \
- {
* O- s! ^; l( l: W - if(GPIO_Pin == KEY_BUTTON_PIN)
2 C( J4 u1 X! ]0 E - {
! e# ]/ u$ P7 A& l, z$ B - if (uwIncrementState == 0)
: H: ~$ q1 a+ K- h3 U2 t2 Z - {
* {: j$ a: w7 F - /* Change the Push button state *// K" l3 y& t( M5 b
- uwIncrementState = 1;1 P7 k' b- F! Q T- o; l- \
- }
6 B0 C8 G4 q6 v: ^ - else! M* [; b' X- C4 g
- {3 g7 Q; a) L: @+ W( Y8 _
- /* Change the Push button state */
- @7 C1 g9 T8 O/ `* ^" b - uwIncrementState = 0;
9 F, M+ r r2 ]2 A" X - }; N2 B) _/ A7 z7 w3 I
- }
% o5 T. M0 g/ Z E2 e - }
, m3 k& G1 ?) r9 d5 F1 e
复制代码 主函数中使用xSemaphoreCreateBinary() 创建信号量,- xTestSemaphore = xSemaphoreCreateBinary();
复制代码 按键检测任务中使用xSemaphoreTake()函数来获取信号量。按键检测使用外部中断,信号量使用完毕后的释放也是在终端服务函数中,如下
) `- n, j" H+ i6 k% x- xSemaphoreTake( xTestSemaphore, mainDONT_BLOCK ); //获取信号,传入创建的信号量句柄xTestSemaphore
复制代码- /**
$ @( Q8 b( J. @ - * @brief EXTI15_10 IRQ5 S+ Y4 }8 Z3 w a6 S
- * @param None
1 A3 e4 s0 X+ T - * @retval None
; L, i2 [/ R8 R - */
6 ]& [' B% j' s - void EXTI15_10_IRQHandler(void)( Z: K6 O$ {% }2 p9 G4 B
- {5 D: P- D1 F0 o( a' {
-
- r, M2 a: M% Q5 J( [6 z! X: I- Y - static BaseType_t xHigherPriorityTaskWoken;
) ~# U: D% f. c, V' V/ d" { -
* E' O( u% _8 Y, j6 q' ~( K - /* Inerrupt message handling */9 K8 n A7 C- `# B: J3 U
- HAL_GPIO_EXTI_IRQHandler(KEY_BUTTON_PIN); % E; @( R2 `6 M6 f4 R
-
. U4 D" m! N/ {- P - /* The handler releases a semaphore.
! u3 O; T! ^; h6 r* b8 r4 ] - xHigherPriorityTaskWoken has been initialised to zero. */
8 N- p8 M- P' E - - N/ b" T5 z, j% L/ f% P+ q* U; w- J
- if (utime_tick >= 200)7 K, q0 L6 f' g0 t7 E) z+ c% y
- {
: }( S1 b) y2 F3 s# [: q3 l - /* The second parameter of the send sync signal is used to save ' A- a: f* B4 y: x% B# I
- whether a high-priority task is ready.*/
3 A1 B F- I9 P2 ]5 O8 L - xSemaphoreGiveFromISR( xTestSemaphore, &xHigherPriorityTaskWoken ); * P S% G( u' q3 _3 o T8 M
- utime_tick = 0;! R& z G h5 X5 ^" O4 P' A* g P
- }
复制代码
2 }+ o/ k# ]6 s! @0 ?) V& O% t* v5 {1 m' \' }0 s
程序运行后会执行LED闪烁任务按键计数任务,当获取到信号量后计数值加1,同时释放信号量共下次资源可用。运行效果如下
! r, X8 p; U7 V$ s1 a0 ^, T0 {% R+ r
4 R, k+ Q' }6 C |