深入RTOS(3):无敌重要的堆与栈

1.堆

堆是一个空闲的内存,我们可以利用一部分内存,然后再释放回去;
例如c语言中的malloc和free函数就可以实现这个功能;

2.栈

栈是RTOS的基础,栈也是一个空闲的内存,但我们经常会感受不到它的存在,它像一个幕后英雄,但它却是不可缺少的一部分;

2.1 函数嵌套实例

void c_fun(void){}
void b_fun(void){}
int a_fun(int val)
{
	int a = 8;
	a +=val;
	b_fun();
	c_fun();
	return a;
}
 
int main(void)
{
	a_fun(46);
	return 0;
}

main函数调用了a_fun函数,a_fun函数调用了b_fun,c_fun函数;

问题是返回地址保存在哪里呢?

main——>a_fun:LR寄存器保存下一个语句的地址(return 0这句) 
                              然后,调用a_fun函数(这个时候会将LR --> 栈中,不然LR会被覆盖,就回不去了)
a_fun——>b_fun:LR寄存器保存下一个语句的地址(c_fun这句) 
                               然后,调用b_fun函数(这个时候会将LR --> 栈中,不然LR会被覆盖,就回不去了)
a_fun——>c_fun:LR寄存器保存下一个语句的地址(return a这句) 
                               然后,调用c_fun函数(这个时候会将LR --> 栈中,不然LR会被覆盖,就回不去了)    
这个时候我们会发现,每一次LR写入新值都会覆盖旧值,所以我们每次都需要将LR中的值写入栈中,LR——>栈;                      
所以返回地址保存在栈中!!

调用过程细节如下:

①启动文件START.o ——>main

②设置SP指针(0x20000000)之后,通过汇编BL语句跳转到main,main划分n个字节作为栈,包含了LR等一系列的寄存器和局部变量;

③main——>a_fun
在函数a_fun的开头,划分了m字节的空间,并将LR指向(Return 0的地址),为局部变量a划分出了局部空间

④a_fun——>b_fun
在函数b_fun的开头,划分了p字节的空间,并将LR指向(c_fun();的地址)

⑤a_fun——>c_fun     

2.2 结论

栈的使用过程,是RTOS多任务的核心,例如分配了taskA和taskB两个任务,那么将会为两个任务分别创建Stack栈,来存储任务中使用到的函数的返回地址和局部变量,两个栈互不干扰和影响;

3.问答

1.调用函数的时候,除了返回地址LR入栈,还有什么需要入栈?

  • 被调用者保存寄存器(Callee-saved registers):这些寄存器在函数调用过程中需要被保存,以防止被调用者函数修改它们的值。在ARM汇编中,被调用者保存寄存器通常包括R4-R11。这些寄存器的值在函数调用前需要入栈,以便在函数返回时恢复它们的原始值。

  • 栈指针(Stack Pointer,简称SP):在函数调用前,需要将当前的栈指针值入栈,以便在函数返回时恢复栈指针的原始值。这对于正确地管理堆栈和恢复寄存器状态至关重要。

  • 局部变量和临时数据:在函数调用前,需要将局部变量和临时数据入栈,以便在函数返回时恢复它们的原始值。这对于正确地管理函数的局部数据和避免数据污染至关重要。

参考链接

RTOS(2)最重要的基础:堆与栈-CSDN博客

  • 21
    点赞
  • 22
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值