|

关注+星标公众号,不错过精彩内容2 r& s) v% h7 t9 [. s% p) t
yzxb310q35i64013517640.gif
2 X4 F' ~0 h2 @9 k8 L* \5 t
作者 | strongerHuang
; O' G8 _2 ]/ r微信公众号 | 嵌入式专栏
% B# B' P9 U. {. s8 W: }
- Q) s+ ^# e5 t0 c# j/ c有小伙伴问了这样的问题:我有个任务中的代码量很多,是不是这个任务的堆栈需要分配很大才行?5 ]3 R) q ?8 i; I# @
+ t/ F0 A! ~$ i
其实,并不是代码量越大,就分配更多堆栈空间,主要取决于你任务中包含的“临时变量”。/ O, q- M! K& }5 D
qk4zhe54zgz64013517741.jpg
; Q4 ?& }4 x V7 F0 L(慕尼黑华南电子展将在月底举办,大家可以提前注册预约)
0 H! b, q( b* `: O7 ~8 d% }% Z! _, ~4 Q3 q+ Y7 C
1RTOS任务堆栈分配市面上很多RTOS的任务都是需要提前分配堆栈大小,也就是在创建任务的时候分配好堆栈的大小。$ n' u# F1 ?# Q; X
比如uCOS创建一个检测(Check)任务:// 任务优先级#define TASK_CHECK_PRIO 6
3 t7 z4 J) h' Z6 N4 M0 W// 任务堆栈大小#define TASK_CHECK_STK_SIZE 128# k5 j" f# C0 T5 O- g- o
// 堆栈OS_STK TaskCheckStk[TASK_CHECK_STK_SIZE];3 I8 N# y& R$ P
// 创建任务 - 信号检测OSTaskCreateExt((void (*)(void *)) AppTaskCheck, (void *) 0, (OS_STK *)&TaskCheckStk[TASK_CHECK_STK_SIZE-1], (INT8U ) TASK_CHECK_PRIO, (INT16U ) TASK_CHECK_PRIO, (OS_STK *)&TaskCheckStk[0], (INT32U ) TASK_CHECK_STK_SIZE, (void *) 0, (INT16U )(OS_TASK_OPT_STK_CHK | OS_TASK_OPT_STK_CLR));
J( g7 F6 t4 _8 c2 p// 任务应用实现void AppTaskCheck(void *p_arg){ // 代码···
8 }* ?& Z* N7 O3 E6 E7 m (void)p_arg;
' G y1 v0 H/ o for(;;) { // 代码··· }}
+ I0 p6 j: O8 r9 b0 b4 I1 y/ sFreeRRTOS创建一个任务类似,也是在创建时分配堆栈大小:#define TASK_CHECK_PRIO 6#define TASK_CHECK_STK_SIZE 128
0 v A) n# p. ABaseType_t xReturn; `: l, w9 u# B. ^ w6 b/ q
xReturn = xTaskCreate(AppTaskCheck, "AppTaskCheck", TASK_CHECK_STK_SIZE, NULL, TASK_CHECK_PRIO, NULL);1 l" Q0 R: ?' [4 q/ n2 ^( N( x- [
除了堆栈,其实还有像消息队列、消息邮箱等也是需要提前分配堆栈。0 }9 J5 V& X5 Z! j+ t$ r% F
比如FreeRTOS创建CLI消息队列:4 k% Y N0 W5 `! H. v* U
#define CLI_QUEUE_NUM 256 //CLI接收队列数#define CLI_PACKAGE_LEN 2 //CLI数据包长度 ]+ F# P" G' P! F+ c% D
QueueHandle_t xCLIRcvQueue = NULL;
5 K) e# I( t; o* V1 d/* 创建队列 */if(xCLIRcvQueue == NULL){ xCLIRcvQueue = xQueueCreate(CLI_QUEUE_NUM, CLI_PACKAGE_LEN);}8 v* r9 n: s" x A9 s8 I
这就是创建任务(或队列)的分配堆栈,至于具体分配多少,与你实际情况有关,下面章节我会描述。( x# B3 u$ X+ D8 X8 y
7 v3 p% j' e9 a ]3 y, ~1 G; E
2任务代码量一个任务的代码量,就是你任务中调用的那些代码。
, T8 P& G: W$ b, ]比如上面例子中的代码:// 任务应用实现void AppTaskCheck(void *p_arg){ // 代码···8 ^ Y, x( b3 E+ c1 {0 e3 A' K0 e
(void)p_arg;
7 D3 A7 R C4 ^* o5 Z: v- H7 k for(;;) { // 代码··· }}8 Q' X* C$ E3 o
这里可能写了几千行代码,或者调用了上百个函数,每个函数里面都有不少代码。
2 R( w- s: Q5 L$ K! A& t
! p0 ~" |! g! [. f* a% e2 }8 Z& |这样下来,这一个任务的代码量就很大了。
/ x$ k W/ j# H( R* ^" m; l+ z I2 q/ v* t( V
3任务代码量和堆栈大小有关系吗?很多人就存在一个疑惑:任务挂起,要在堆栈中临时保存任务,如果这个任务的代码量很大,是不需要很大堆栈空间才行?
9 b8 r/ N) u4 l. H答案:不一定需要很大堆栈空间,任务代码量和堆栈也没有直接关系。
# R# V1 s6 p& f+ ]' o. [9 ^
' ?% s, B5 K7 p S可能很多初学者存在这么一个误区:保存一个任务,就是把这个任务所有代码都保存起来(在堆栈中)。! ?, J2 C; G, ?4 _: v! |* |! R
8 t) W1 n1 u+ d1 Y8 b堆栈主要保存是这个任务自身的变量(控制块),还有临时变量等这些关键变量信息,而并非要保存所有代码。2 P4 d5 O+ o: a' X* C, _
% p7 ~& d# Z- Z/ b: H4堆栈分配多大才合适?任务堆栈大小,主要取决于你任务中【临时变量】的多少。
* D& h3 j/ I0 b$ {7 @" ^5 V- M注意:临时变量包含你代码中所有嵌套函数中的临时变量。$ ^3 b& \+ `5 H' |6 P
对于RAM资源相对较大的处理器,你可以尽量分配多一点堆栈资源。# K, W1 c$ T' K( a7 {; i. d
但是,很多时候,我们的RAM资源都是相对比较紧张的。这个时候,就需要你综合平衡。
0 g) r( A/ ^* X$ K! r比如静态局部变量:void AppTaskCheck(void *p_arg){ static uint8_t aaa; //静态局部变量0 a) y5 B% D- K! a; X9 M
(void)p_arg;
% H' S! y$ F$ @; H for(;;) { // 代码··· }}
0 V& O6 u) r3 M3 u5 k7 Q- w这里的aaa变量就不会占用该任务的堆栈空间,但是它会占用全局变量(RAM)空间。
$ x) u n" ?( g4 W( i) ]5 t: Y用静态局部变量,还是临时变量,要牵涉到你项目具体情况,比如:RAM资源、代码运行效率等。(临时变量还会有一个数据拷贝过程)- l. R `" U! l: _' Q1 x
所以,该如何分配堆栈,该用静态还是临时变量,需要综合考虑你项目的情况而定。
% u$ {( h1 x9 Q6 \* E& i. l9 h. r3 N2 v# F- [+ I
(慕尼黑华南电子展将在月底举办,大家可以提前注册预约)9 l; V$ j. E5 C6 G' c8 g% Q5 d
------------ END ------------
4 J8 H! t! M3 m) x# A( H关注公众号后台回复『嵌入式开发』『单片机』阅读更多相关文章。
- A: u$ ~- t$ B k- C回复“加群”按规则加入技术交流群,回复“1024”查看更多内容。/ ~9 c4 {- q' b8 ?/ g
dwaltqzlpgr64013517841.jpg
* W: }. a. p# \ }
3 |. a; }3 ^% J" @# O1 ~2 W
* t# D/ P, F: d9 [
z41yh3ht34s64013517941.png
. o6 x& Y# O& v- g
( ]% F4 B" u3 i' Z$ w点击“阅读原文”查看更多分享。 |
|