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

调试调到怀疑人生,heap我对你的误解有多深

[复制链接]
watershade 发布时间:2019-4-2 21:41
这几天在调试自己写的一个FreeRTOS的程序。程序稍微有点大,用的是F0内核的MCU。ROM只有16K。
) V* R' t. }6 _' @0 l. l因为我的程序内部要自己用allloc开辟一部分空间,而因为使用了多个任务和MailQueue整个空间的占用可想而知。
0 h9 b& O$ z& k7 Y+ w# m所以我设置FreeRTOS的heap空间5120,考虑到别的占用。所以总共分的heap是5K+2K。这时候RTOS跑不起来了。
4 _9 H/ G6 G# l; K8 ?, T当然我的Stack也分了2K。所以我谨慎的分了8K,还不行。这时候我把FreeRTOS的heap空间扩大了。这时候竟然连编译都通不过了。
7 A: r; _$ o' G+ t我还在怀疑是不是我的程序有什么问题。: Y5 O0 |' w6 F( ?
直到今晚,我才解开这个问题。只怪我学艺不精,对heap的理解有点想当然。(差不多有快一年没有耐心做大项目了,所以之前掌握的本就似是而非的知识就成了埋在调试路上的陷阱。)
* u4 Q9 l! c0 j/ I5 @原来,F0和其它的系列是一样的。都是双堆栈指针:MSP(Main Stack Pointer主堆栈指针)和PSP(Process Stack Pointer)。这种设计非常适合RTOS的设计。MSP顾名思义是给系统栈使用的。PSP是给进程栈使用的。在RTOS运行中,整个栈空间是由PSP管理的。而切换到ISR中则使用MSP。1 B- a+ e" K) I/ c
这点其实我是知道的,知识印象有些模糊。但我真正不知道的在于下面一点。.s文件中的heap只管分配给MSP。 . N  D+ y: `7 s$ ]1 M/ ]$ n. E  A
STM32cubemx中有个简单的估算heap大小的方法。(因为不包含MailQueue,所以尺寸你可以适当在上面增加一下mempool的大小。)我习惯的做法就是将两个数加起来设置一下heap但是这次遇见问题了。我给.s文件中的heap设置的越大,就表示msp管理的heap越大,因为整个空间有限。我的psp占用的栈就小了。所以在我上面的列子中。psp也许需要超过5K的空间,这时候因为我的MSP分配的多了。就导致编译通不过。这时候我就设法扩大PSP的空间,但是因为理解错误,我就需要确保.s分配的空间大于5K。所以这时候我发现我的FreeRTOS的最大空间不能扩大了。
" m! m$ l" H& R) o0 y
( G3 Q& N2 b& A. u' k9 L' D那现在整么解决呐。很简单就是把.s的空间设置小一点。不如2K。一般够我的程序用了。而stack也是2k。所以我的FreeRTOS的heap配置可以扩大。比如8K.这时候我发现问题解决了。那一切问题就要有官方介绍才成。
6 u  A# R! |* h1 r2 P
/ P6 \7 ?' k' e: Tarm官网的介绍:http://www.keil.com/support/docs/4049.htm 这里面给了几个链接说明。可以看一下。& t  z& Y) r  p& x& Z4 _9 h- l
mainSteak.jpg
7 o" b7 p& F- q( c6 ]+ H  q5 U5 y) a8 W+ y" r7 J
这个问题无意间才发现的。要不是调试调到怀疑人生,我将会对.s的heap误解越来越深。
( y& I% P) U8 [- F
收藏 3 评论12 发布时间:2019-4-2 21:41

举报

12个回答
watershade 回答时间:2019-4-3 00:34:06
本帖最后由 watershade 于 2019-4-3 17:19 编辑
' N3 F8 z* Y$ e5 M% v! D/ s, @1 |
帖子应该移到哪个区?(谢谢管理员将它移动到了RTOS区)
' q* V0 w& H% L" ~2 C
jeffhe1 回答时间:2019-4-3 08:39:26
真是大神, 厲害的文章
一代睡神的崛起 回答时间:2019-4-3 08:41:56
谢谢分享,学习了2 z( j- R+ y- K+ i( @
sky_han 回答时间:2019-4-3 08:58:11
。MSP顾名思义是给系统栈使用的。PSP是给进程栈使用的。在RTOS运行中,整个栈空间是由PSP管理的。而切换到ISR中则使用MSP。' |0 v: e9 J# x. v9 C

6 x7 J  D3 Z  ]7 a! T这个是大家都知道的,不知道的是?  u' Y& ~" s( y& s& w

' h7 |' A5 c* X5 E% u* n- u; g! TMSP和PSP各有各自的HEAP。
" i  x) T  T1 L0 j2 u8 ]* m9 m& B9 E. E+ S9 s- X% V% Z- W
tanic 回答时间:2019-4-3 09:46:59
MSP用的.s分配的   PSP用的rtos 通过宏分配的?
七哥 回答时间:2019-4-3 10:02:07
tanic 发表于 2019-4-3 09:46
  r2 X& d7 a7 _4 jMSP用的.s分配的   PSP用的rtos 通过宏分配的?

$ W- m; G7 P$ G. r" p1 o' t: T学习了
Kevin_G 回答时间:2019-4-3 10:11:40
厉害
watershade 回答时间:2019-4-3 17:17:25
tanic 发表于 2019-4-3 09:46
* t6 z# C' G: P, FMSP用的.s分配的   PSP用的rtos 通过宏分配的?
. J3 r% K' U2 x6 n" z: v9 @
是的
watershade 回答时间:2019-4-3 17:18:00
sky_han 发表于 2019-4-3 08:58; e+ N4 U; o/ c) H0 K
。MSP顾名思义是给系统栈使用的。PSP是给进程栈使用的。在RTOS运行中,整个栈空间是由PSP管理的。而切换到I ...

. O4 a& B0 W" `% R' l6 M1 I+ O0 q我不知道的是.s只管msp
Cortexxx 回答时间:2019-4-9 09:30:01
有个小问题
/ {  b0 l0 l* o  s
  1. xTaskCreate(FactoryTask,"Factory",128, 0, 3, NULL);
复制代码
  1.         BaseType_t xTaskCreate(        TaskFunction_t pxTaskCode,* Z# Y8 Y/ a. B% y$ M
  2.                                                         const char * const pcName,6 e8 u' [' d' k8 X/ m
  3.                                                         const uint16_t usStackDepth,
    9 _* E$ D  A' ?( ?' _, ~
  4.                                                         void * const pvParameters,
    : m: ]! [/ ^3 {8 w# w8 y
  5.                                                         UBaseType_t uxPriority,
    " s) m1 m' z. ]/ h! I" f: D
  6.                                                         TaskHandle_t * const pxCreatedTask ) /*lint !e971 Unqualified char types are allowed for strings and single characters only. */
复制代码

* F) w6 d3 u6 S* Q: m
. T. B& y$ {% C0 x: {7 k0 r这个代码的第三个参数 128 申请的是stack还是Heap?
/ k2 d3 P# i: n我进到这个function里面看到它用malloc开了一块内存,故认为是开辟的是heap空间,
) v( h/ U3 \" X: E. _8 \; U但是他的形参名称为什么是 StackDepth。 这点我就很糊涂了,到底这个128*4 stack 还是heap?
watershade 回答时间:2019-4-15 13:06:24
Cortexxx 发表于 2019-4-9 09:30
- e8 f0 `. \2 t+ a有个小问题
0 F+ _0 [0 {" S9 A/ f, T
不好意思,这几天没有看论坛。
1 w# u. H6 c) L. N下图是从freeRTOS v10.0.0的手册上截图的。(这个手册freeRTOS官方可免费下载) frtos_heap_stack.jpg * f2 c- h. I7 t! `3 A& E
手册提到了动态创建和静态创建两种内存分配方法。我的上面提到的TOTAL_HEAP_SIZE其实只在动态创建的时候才有效.
) j. R6 v8 p& ^4 L+ ^在动态分配的时候,需要先指定heap的总共大小。但这段memory并不是划出来不用的。所有的task,queue和semaphore创建的时候都要从这里面划分。而对于任务的动态创建,TCB和stack都是从heap里面分配的。但如果使用静态创建,那这个任务就是在编译环节分配的。不是程序在heap里面动态分配的。
2 Q5 ~4 M1 B6 g# V$ Y* o/ c
0 ]. j7 O7 [% E; v; O" I~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~3 p6 q4 l  v  _5 }; z
如果你看st封装的cmsis-rtos,他的程序创建使用的是osThreadCreate.而freeRTOS的memory allocation分为三种:一种是Dynamic,另一种是Static,第三种是Dynamic/Static。但本质上使用的还是上图提到的两个functions:xTaskCreate和xTaskCreateStatic
+ s! ^" |3 d7 |/ U, u图片里展示了主要的内容。你可以试着将memory allocate改成static,你会看到heap的大小选择没有了。
! F* ~$ ?* j+ [, G. y" N( } frtos_heap_stack_2.jpg frtos_heap_stack_3.jpg # \# o9 F/ v6 f2 J& I
7 \6 C; I! V) `
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
) |8 Q: i  b& F( n这是我个人的理解,如果你有疑问或者不同意。可以留言给我,咱们交流交流。谢谢: `+ C1 J# @2 Y8 n+ |% p
Cortexxx 回答时间:2019-4-16 08:56:04
watershade 发表于 2019-4-15 13:06
- ?! d8 v5 r. S; ^不好意思,这几天没有看论坛。
- D  _5 }( r- |* k- S: w; S下图是从freeRTOS v10.0.0的手册上截图的。(这个手册freeRTOS官方可免费 ...
, h: C8 t2 M/ o6 P  c* e
感谢回复 了解了 XDDD
; [. P# |$ c# @( a

所属标签

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