|
oowb3nazstz64012791135.gif
/ z9 v. x! O5 R" J7 n. X' v
点击上方蓝色字体,关注我们
/ j% \. y2 E7 o; q/ R' Z
% l' C, l4 J! `( w0 r1
T4 B# W6 I6 i: @* ?任务调度延迟
" s2 y' V1 k( K( X/ W任务调度器是RTOS的核心,当一个高优先级任务准备就绪时,调度器需要及时切换到该任务。! \2 r6 U6 n1 g2 [
/ ?0 r/ z% g# ~/ ]9 l) F1 _调度延迟是指从高优先级任务就绪到开始执行的时间。; @( t& x: }! {; Z' Z
0 P6 ?: K& X) a' x0 \& c8 j调度器算法(如时间片轮转、优先级调度)对调度延迟有很大影响。3 B4 W8 S7 B. Y* z
# D% b0 O/ U: o. O- q/ K& d
使用FreeRTOS中的任务优先级来示例调度的影响,例子中,vTask2的优先级更高,因此每次调度时RTOS都会优先执行它,体现了任务调度对实时性的影响。
! G+ q( o# U) {2 t* }! ~) M) C
! n0 I4 O+ E. ]( x#include "FreeRTOS.h"#include "task.h"
6 O9 i3 [; o8 Pvoid vTask1(void *pvParameters) { for (;;) { // Task 1 has a lower priority printf("Task 1 running. m I6 r& d+ c- N9 G! b: h/ Z
"); vTaskDelay(500 / portTICK_PERIOD_MS); // Simulate some work }}
% R% o1 ^) p' z) w$ Y1 Nvoid vTask2(void *pvParameters) { for (;;) { // Task 2 has a higher priority printf("Task 2 running
! Q3 w- t7 H: \% N"); vTaskDelay(500 / portTICK_PERIOD_MS); // Simulate some work }}4 u' H4 u) L& _& x, X5 |# B! R
int main(void) { // Create tasks with different priorities xTaskCreate(vTask1, "Task 1", 1000, NULL, 1, NULL); xTaskCreate(vTask2, "Task 2", 1000, NULL, 2, NULL); // Higher priority vTaskStartScheduler(); // Start the scheduler for (;;) {}}, ?& v- n# B# h. ?4 D% x
2
5 q2 B& O- C1 U中断处理延迟
9 D+ D$ g. R7 J! b5 d6 Q( ]中断处理是RTOS实时性的重要因素。& N4 K4 `: |! D" d. O6 b
3 o. g8 O9 c* u0 h系统需要能够快速响应中断,但过多的中断可能导致高优先级任务延迟执行。( V3 `% A; L Q7 a$ b
0 {8 f7 j* u+ Z
因此,中断服务程序(ISR)的设计应尽量短小,避免耗时的操作放在ISR中。
; J/ b/ D) W; v8 m* J: {' E& z" U
! q3 T1 u( n3 u$ YISR通过vTaskNotifyGiveFromISR将处理任务交给高优先级任务,确保ISR本身尽量短,从而减少中断对实时性的影响。
& A5 y' c6 j- T& @7 e" B* A8 L2 l" w- S8 J- X: J
void vShortISR(void) { BaseType_t xHigherPriorityTaskWoken = pdFALSE;2 @% N" A% ]$ b6 G
// Notify a task that an interrupt has occurred (short ISR) vTaskNotifyGiveFromISR(xTaskHandle, &xHigherPriorityTaskWoken);
$ \) x5 H2 K2 j; c& M1 R% j4 K6 o" c // Perform context switch if necessary portYIELD_FROM_ISR(xHigherPriorityTaskWoken);}0 s9 V3 e6 V3 B: k; h) E( v
3" F3 K. S' T9 T/ ]% q5 f3 a% k6 b
系统负载# ^& s0 ^: ]' M m$ }$ g. Z4 p& J
如果系统负载过高,任务和中断处理可能会被推迟。7 T1 k4 z; j* X1 M
# t# V$ J9 b9 D! l f1 N合理的负载分配和资源使用可以提高实时性,确保高优先级任务和中断能够及时响应。
: n( C* h4 I- Z' s6 ^- ^) R* N0 m/ x( F4 M% g, a$ H/ M
在高负载任务运行期间,系统可能难以及时响应其他任务。2 \" v, t: P- H( l' G
+ G# ~' p& E+ G" K
可通过vTaskDelay()等函数将任务划分为多个小段,避免单个任务占用过多时间。- _4 O$ f2 @+ }9 M* X. b
6 t" d/ S! d7 F
void vHeavyTask(void *pvParameters) { while(1) { // Simulate heavy computation for (int i = 0; i 1000000; i++) { // Do some heavy work } printf("Heavy task completed/ F+ S2 d% I s2 t
"); }}
Y+ X3 [8 @7 Y, E4 k4# }8 x i) h0 Q7 J+ @+ l) Y+ {' Y
任务优先级反转; x$ G* F8 l0 @0 k+ O
优先级反转发生在低优先级任务持有资源而高优先级任务需要该资源的情况,导致高优先级任务被阻塞。" j M- v8 d' ]& d
4 A. `: d( h1 N9 Q+ J& A9 D m
通过使用优先级继承机制,RTOS可以暂时提升低优先级任务的优先级,解决优先级反转问题。
- u: m8 {/ G2 t2 ?9 M& H
- q* v" [7 z, [, l) n" p, U例子中,优先级反转可能发生,RTOS可以通过优先级继承机制暂时提升低优先级任务的优先级,解决这个问题。
- X0 o2 d( w; x! j2 c* I3 e' S2 b. R) L+ n- `8 C' U7 d
SemaphoreHandle_t xMutex;
7 ~- B) Q( P2 }' q |/ f' v5 {( S) K Ovoid vLowPriorityTask(void *pvParameters) { // Low-priority task acquires the mutex xSemaphoreTake(xMutex, portMAX_DELAY); printf("Low-priority task holding mutex# K4 c$ F5 o! Q& z0 `% u4 \/ \
"); vTaskDelay(1000 / portTICK_PERIOD_MS); // Simulate some work xSemaphoreGive(xMutex);}" [ |4 G0 _7 Q! U8 g
void vHighPriorityTask(void *pvParameters) { // High-priority task tries to take the mutex xSemaphoreTake(xMutex, portMAX_DELAY); printf("High-priority task acquired mutex
- X4 J. j7 c% P$ S+ ~5 Q. ^"); xSemaphoreGive(xMutex);}
( B8 f3 a! x( f0 x3 sint main(void) { xMutex = xSemaphoreCreateMutex(); // Create tasks with different priorities xTaskCreate(vLowPriorityTask, "Low", 1000, NULL, 1, NULL); xTaskCreate(vHighPriorityTask, "High", 1000, NULL, 2, NULL); vTaskStartScheduler(); for (;;) {}}
4 {. y5 c: x0 J4 I5/ {3 B9 Y% U! X5 A8 G
时钟精度: u+ s; `# L6 h5 y# [+ N
时钟是实时操作系统的基础,所有定时相关的操作依赖时钟。8 c& N C* [" K( S" W
& m. p& O% _, O( y8 _时钟的精度和抖动会直接影响任务的准确性,特别是在定时器任务和延迟任务的场景下。
; x0 ~7 K8 T& \9 ?# c. m
6 p# P: |1 q9 U$ u7 G9 k例子中,确保任务每100毫秒精确执行一次,如果时钟不够精确或存在抖动,任务的执行时间间隔将会受到影响。- T# h4 v' O$ v5 D; G' M @5 s
# n: u/ R" c' n; f$ }
void vTask(void *pvParameters) { TickType_t xLastWakeTime; const TickType_t xFrequency = 100; // 100 ms% R- `+ Z2 |" d6 y! V5 ~
xLastWakeTime = xTaskGetTickCount();
0 G5 ]( `, l- d for (;;) { // Wait for the next cycle vTaskDelayUntil(&xLastWakeTime, xFrequency); printf("Task running at precise intervals
! g! B5 B/ R0 B& r$ }" P3 N8 y1 b"); }}( X# o, O3 G4 E7 R. b
6
+ c$ H L. i% ^4 S内存管理2 V* Y5 C3 j3 [: m: L L$ Z
实时系统的内存管理需要高效且可控。动态内存分配(如malloc())可能导致内存碎片化,降低系统性能。: @3 H% F- |' S1 d
+ i+ j7 M6 B% o \- T* c& @1 u
为避免内存管理对实时性造成负面影响,通常建议使用静态内存分配。
6 D) u* U2 R% ?3 v% x$ i6 q4 {7 }* | {& ?
例子中,使用静态内存分配来避免动态分配带来的内存碎片问题,从而提高实时性。
: }9 |% u3 ^' n" R# }* K8 W- n& b; i- a
static StackType_t xStackBuffer[1000];static StaticTask_t xTaskBuffer;
( Y' J, |7 B t/ a# K4 D" h j4 Pvoid vStaticTask(void *pvParameters) { for (;;) { printf("Static task running/ a- h) }, K4 T' b
"); vTaskDelay(1000 / portTICK_PERIOD_MS); }} X. \: Z% p2 G- x# q$ V! C2 q' g
int main(void) { // Create a task with statically allocated memory xTaskCreateStatic(vStaticTask, "StaticTask", 1000, NULL, 1, xStackBuffer, &xTaskBuffer); vTaskStartScheduler(); for (;;) {}}- z" F$ ]- M. S& r1 u
通过合理的设计和优化,可以有效提升RTOS的实时性能,确保系统能够及时响应任务和中断。+ |) L' H" B4 M8 @
psy1upvhwn464012791236.jpg
. n! c* \: ^6 |# p. q% W
uutzlalz5nl64012791336.gif
$ j( D! K1 n2 A5 h点击阅读原文,更精彩~ |
|