变量究竟是存在寄存器还是堆栈?

嵌入式电子 2023-06-01 18:47

1. 变量是放在寄存器里还是堆栈里?

堆栈对于处理器来说就是一块内存区域,而寄存器是处理器触手可及的存储,对于RISC 处理器而言,堆栈中的数据CPU并不能直接进行运算,还是要先加载到寄存器中才行。对于编译器而言,我猜测还是优先会选择将变量用寄存器保存。那什么时候需要用到堆栈呢?什么东西需要保存到堆栈呢?一种是需要切换上下文的地方,另一种是需要传参的地方。函数调用就是一种典型应用。

2. 函数调用时的栈与寄存器


一个典型的函数调用流程如上图所示,关键的涉及栈和寄存器的步骤如下:

  1. 首先,在调用其他函数前,Caller需要保存自身的上下文,比如某个变量的值存在寄存器x1,该寄存器有可能在被调函数Callee中用到,那就需要存到堆栈中;

  2. 调用函数时的传参,参数的传递有两个选择,一个是将参数放到寄存器,另一个则是将参数放到堆栈中。

  3. Caller调用函数的时候需要将PC+4,即函数的返回地址存到寄存器或堆栈,被调函数执行完调到该返回地址执行;

  4. 对于Callee而言,首先要保存上文中的某些寄存器,有的同志可能会问,步骤1中Caller不是已经保存了一波寄存器了,为什么Callee中又来一波?以RISC-V为例,约定了两类寄存器——临时寄存器和保存寄存器,其中临时寄存器可以由Callee随意使用,所以就需要Caller来保存;而保存寄存器需要Callee来保证其值在调用前后不能改变,所以需要Callee存储。

  5. Callee在执行完之后,自然需要将“保存寄存器”的值恢复。此外,返回参数也需要存放至寄存器或堆栈中。

从上面的描述中可以看到,保存寄存器只能存到堆栈中,而保存参数和函数返回地址则既可以放在寄存器,也可以放在堆栈。RISC-V对此进行了如下约定:

寄存器名ABI名(编程用名)用途约定谁负责在函数调用过程中维护这些寄存器
x0zero读取时总为 0, 写入时不起任何效果N/A
x1ra存放函数返回值(return address)Caller
x2sp存放栈指针(stack pointer)Callee
x5~x7, x28~x31t0~t2, t3~t6临时(temporaries)寄存器,Callee 可能会使用这些寄存器,所以Callee 不保证这些寄存器中的值在函数调用过程中保持不变,这意味着对于 Caller 来说,如果需要的话,Caller 需要自己在调用 Callee 之前保存临时寄存器中的值。Caller
x8, x9, x18~x27s0, s1, s2~s11保存(saved)寄存器,Callee 需要保证这些寄存器的值在函数返回后仍然维持函数调用之前的原值,所以一旦 Callee 在自己的函数中会用到这些寄存器则需要在栈中备份并在退出函数时进行恢复。Callee
x10 , x11a0 , a1参数(argument)寄存器,用于在函数调用过程中保存第一个和第二个参数,以及在函数返回时传递返回值。Caller
x12 ~ x17a2 ~ a7参数(argument)寄存器,如果函数调用时需要传递更多的参数,则可以用这些寄存器,但注意用于传递参数的寄存器最多只有 8 个(a0 ~ a7),如果还有更多的参数则要利用栈。Caller

由于编译器做出了以上函数调用约定,这就意味着汇编指令中本来要带的参数不需要了,比如原来跳转并链接到某个label的指令是jal x1, offset,意思是跳转到 offset 制定位置,返回地址保存在 x1 (ra)中,现在由于做好了约定,可以直接使用伪指令jal offset来替代。

3. 函数调用实例

下面举个例子来说明一下小节2中的过程。
首先给出C语言代码如下,代码入口是_start函数,其中调用了aa_bb函数,aa_bb又调用了square函数。

void _start()
{
// calling nested routine
aa_bb(3, 4);
}

int aa_bb(int a, int b)
{
return square(a) + square(b);
}

int square(int num)
{
return num * num;
}

让我们根据2中的一些约定来写RISC-V汇编指令:

_start:
la sp, stack_end # 首先初始化堆栈指针
li a0, 3 # 第一个参数放入寄存器a0
li a0, 4 # 第二个参数放入寄存器a1
call aa_bb # Caller中没用到什么临时寄存器,所以不需要保存,直接跳转; call伪指令会将跳转地址放到寄存器ra中
stop:
j stop # 死循环结束代码

aa_bb:
addi sp, sp, -16 # 栈指针初始化时指在栈去末尾,要存放4个word的数据,所以减16
sw s0, 0(sp) # 考虑到后面会用到保存寄存器,s0~s2,需要Callee进行保存
sw s1, 4(sp)
sw s2, 8(sp)
sw ra, 12(sp) # ra中目前存放的是_start函数下一条指令的地址,由于在aa_bb函数中还要调用别的函数,会覆盖掉ra寄存器的值,因此也要存下来

mv s0, a0 # a0寄存器的值挪到s0中
mv s1, a1 # a1寄存器的值挪到s1中

li s2, 0 # 将最终结果放在s2中,因此先将s2清零

mv a0, s0 # 准备调用square函数啦,传参放入寄存器a0; 作为一个Caller没有需要保存的临时寄存器
jal square # 跳转到square函数中执行,返回地址会放在ra寄存器,覆盖了原来的ra
add s2, s2, a0 # 计算的返回参数在a0中,加到s2上

mv a1, s1 # 与上面计算同理
jal square
add s2, s2, a0

mv a0, s2 # 最终的返回参数防止到a0寄存器
lw s0, 0(sp) # 从栈区中恢复各个保存寄存器的值
lw s1, 4(sp)
lw s2, 8(sp)
lw ra, 12(sp) # ra寄存器的值恢复回来
addi sp, sp, 16 # 释放栈空间,栈指针依旧指向进入函数前的位置
ret

square:
addi sp, sp, -8
sw s0, 0(sp)
sw s1, 4(sp)

mv s0, a0
mul s1, s0, s0
mv a0, s1

lw s0, 0(sp)
lw s1, 4(sp)
addi sp, sp, 8
ret

stack_start:
.rept 10 # 定义10个word大小的栈空间
.word 0
.endr
stack_end:

如果有人读了上面的代码,可能会发出一个疑问——为什么运算的时候非得用s0~s2这种保存寄存器,这不是脱裤子放屁呢吗?因为这个是教学演示代码,为了说明小节2中的函数调用流程。

4. volatile关键字的原理

volatile关键字存在的根本原因正是变量会存在寄存器还是内存中。假如一个变量会在多线程中用到,该变量如果在一个线程运行中被加载到了寄存器中,则显然后续的代码也会继续使用这个寄存器中的变量值,而不会使用内存中的该值。如果别的线程中的代码修改了内存中该变量的值,而本线性依旧使用寄存器中的值就会出错。
volatile就是告诉编译器,每次用到这个变量的时,都去内存中把这个值重新加载一次,而不是沿用之前的寄存器中的值。因为这个值可能被中途改变了。


定期以通俗易懂的方式分享嵌入式知识,关注公众号,加星标,每天进步一点点。


声明:

本号原创、转载的文章、图片等版权归原作者所有,如有侵权,请联系删除。

关注、点赞、在看、转发,支持优质内容! 

评论
  • 这篇内容主要讨论三个基本问题,硅电容是什么,为什么要使用硅电容,如何正确使用硅电容?1.  硅电容是什么首先我们需要了解电容是什么?物理学上电容的概念指的是给定电位差下自由电荷的储藏量,记为C,单位是F,指的是容纳电荷的能力,C=εS/d=ε0εrS/4πkd(真空)=Q/U。百度百科上电容器的概念指的是两个相互靠近的导体,中间夹一层不导电的绝缘介质。通过观察电容本身的定义公式中可以看到,在各个变量中比较能够改变的就是εr,S和d,也就是介质的介电常数,金属板有效相对面积以及距离。当前
    知白 2025-01-06 12:04 173浏览
  • 在智能家居领域中,Wi-Fi、蓝牙、Zigbee、Thread与Z-Wave等无线通信协议是构建短距物联局域网的关键手段,它们常在实际应用中交叉运用,以满足智能家居生态系统多样化的功能需求。然而,这些协议之间并未遵循统一的互通标准,缺乏直接的互操作性,在进行组网时需要引入额外的网关作为“翻译桥梁”,极大地增加了系统的复杂性。 同时,Apple HomeKit、SamSung SmartThings、Amazon Alexa、Google Home等主流智能家居平台为了提升市占率与消费者
    华普微HOPERF 2025-01-06 17:23 145浏览
  • 本文介绍Linux系统更换开机logo方法教程,通用RK3566、RK3568、RK3588、RK3576等开发板,触觉智能RK3562开发板演示,搭载4核A53处理器,主频高达2.0GHz;内置独立1Tops算力NPU,可应用于物联网网关、平板电脑、智能家居、教育电子、工业显示与控制等行业。制作图片开机logo图片制作注意事项(1)图片必须为bmp格式;(2)图片大小不能大于4MB;(3)BMP位深最大是32,建议设置为8;(4)图片名称为logo.bmp和logo_kernel.bmp;开机
    Industio_触觉智能 2025-01-06 10:43 87浏览
  • By Toradex 秦海1). 简介嵌入式平台设备基于Yocto Linux 在开发后期量产前期,为了安全以及提高启动速度等考虑,希望将 ARM 处理器平台的 Debug Console 输出关闭,本文就基于 NXP i.MX8MP ARM 处理器平台来演示相关流程。 本文所示例的平台来自于 Toradex Verdin i.MX8MP 嵌入式平台。  2. 准备a). Verdin i.MX8MP ARM核心版配合Dahlia载板并
    hai.qin_651820742 2025-01-07 14:52 45浏览
  • 村田是目前全球量产硅电容的领先企业,其在2016年收购了法国IPDiA头部硅电容器公司,并于2023年6月宣布投资约100亿日元将硅电容产能提升两倍。以下内容主要来自村田官网信息整理,村田高密度硅电容器采用半导体MOS工艺开发,并使用3D结构来大幅增加电极表面,因此在给定的占位面积内增加了静电容量。村田的硅技术以嵌入非结晶基板的单片结构为基础(单层MIM和多层MIM—MIM是指金属 / 绝缘体/ 金属) 村田硅电容采用先进3D拓扑结构在100um内,使开发的有效静电容量面积相当于80个
    知白 2025-01-07 15:02 75浏览
  • 根据Global Info Research项目团队最新调研,预计2030年全球封闭式电机产值达到1425百万美元,2024-2030年期间年复合增长率CAGR为3.4%。 封闭式电机是一种电动机,其外壳设计为密闭结构,通常用于要求较高的防护等级的应用场合。封闭式电机可以有效防止外部灰尘、水分和其他污染物进入内部,从而保护电机的内部组件,延长其使用寿命。 环洋市场咨询机构出版的调研分析报告【全球封闭式电机行业总体规模、主要厂商及IPO上市调研报告,2025-2031】研究全球封闭式电机总体规
    GIRtina 2025-01-06 11:10 104浏览
  • 每日可见的315MHz和433MHz遥控模块,你能分清楚吗?众所周知,一套遥控设备主要由发射部分和接收部分组成,发射器可以将控制者的控制按键经过编码,调制到射频信号上面,然后经天线发射出无线信号。而接收器是将天线接收到的无线信号进行解码,从而得到与控制按键相对应的信号,然后再去控制相应的设备工作。当前,常见的遥控设备主要分为红外遥控与无线电遥控两大类,其主要区别为所采用的载波频率及其应用场景不一致。红外遥控设备所采用的射频信号频率一般为38kHz,通常应用在电视、投影仪等设备中;而无线电遥控设备
    华普微HOPERF 2025-01-06 15:29 127浏览
  • 大模型的赋能是指利用大型机器学习模型(如深度学习模型)来增强或改进各种应用和服务。这种技术在许多领域都显示出了巨大的潜力,包括但不限于以下几个方面: 1. 企业服务:大模型可以用于构建智能客服系统、知识库问答系统等,提升企业的服务质量和运营效率。 2. 教育服务:在教育领域,大模型被应用于个性化学习、智能辅导、作业批改等,帮助教师减轻工作负担,提高教学质量。 3. 工业智能化:大模型有助于解决工业领域的复杂性和不确定性问题,尽管在认知能力方面尚未完全具备专家级的复杂决策能力。 4. 消费
    丙丁先生 2025-01-07 09:25 80浏览
  • 彼得·德鲁克被誉为“现代管理学之父”,他的管理思想影响了无数企业和管理者。然而,关于他的书籍分类,一种流行的说法令人感到困惑:德鲁克一生写了39本书,其中15本是关于管理的,而其中“专门写工商企业或为企业管理者写的”只有两本——《为成果而管理》和《创新与企业家精神》。这样的表述广为流传,但深入探讨后却发现并不完全准确。让我们一起重新审视这一说法,解析其中的矛盾与根源,进而重新认识德鲁克的管理思想及其著作的真正价值。从《创新与企业家精神》看德鲁克的视角《创新与企业家精神》通常被认为是一本专为企业管
    优思学院 2025-01-06 12:03 119浏览
  • 根据环洋市场咨询(Global Info Research)项目团队最新调研,预计2030年全球无人机锂电池产值达到2457百万美元,2024-2030年期间年复合增长率CAGR为9.6%。 无人机锂电池是无人机动力系统中存储并释放能量的部分。无人机使用的动力电池,大多数是锂聚合物电池,相较其他电池,锂聚合物电池具有较高的能量密度,较长寿命,同时也具有良好的放电特性和安全性。 全球无人机锂电池核心厂商有宁德新能源科技、欣旺达、鹏辉能源、深圳格瑞普和EaglePicher等,前五大厂商占有全球
    GIRtina 2025-01-07 11:02 71浏览
我要评论
0
点击右上角,分享到朋友圈 我知道啦
请使用浏览器分享功能 我知道啦