- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
堆栈帧的结构是什么以及在汇编中调用函数时如何使用它?
最佳答案
每个例程都使用堆栈的一部分,我们将其称为堆栈帧。尽管汇编程序员并不被迫遵循以下风格,但强烈建议将其作为良好实践。
每个例程的堆栈帧分为三部分:函数参数、指向前一个堆栈帧的反向指针和局部变量。
第 1 部分:函数参数
例程堆栈帧的这一部分由调用者设置。使用“push”指令,调用者将参数压入堆栈。不同的语言可能会以不同的顺序推送参数。 C,如果我没记错的话,将它们从右推到左。也就是说,如果您打电话...
foo (a, b, c);
调用者会将其转换为...
push c
push b
push a
call foo
当每个项目被插入堆栈时,堆栈会向下增长。也就是说,堆栈指针寄存器递减四 (4) 个字节(在 32 位模式下),并且该项目被复制到堆栈指针寄存器指向的内存位置。请注意,“call”指令将隐式地将返回地址压入堆栈。参数的清理将在第 5 部分中解决。
第 2 部分:Stackframe 后退指针
此时,“call”指令已发出,我们现在正处于被调用例程的开始处。如果我们想访问我们的参数,我们可以像...
[esp + 0] - return address
[esp + 4] - parameter 'a'
[esp + 8] - parameter 'b'
[esp + 12] - parameter 'c'
但是,在我们为局部变量和其他内容留出空间后,这可能会变得很笨拙。因此,除了堆栈指针寄存器之外,我们还使用堆栈基指针寄存器。但是,我们希望将堆栈基指针寄存器设置为当前帧,而不是前一个函数。因此,我们将旧的保存在堆栈上(这会修改堆栈上参数的偏移量),然后将当前堆栈指针寄存器复制到堆栈基指针寄存器。
push ebp ; save previous stackbase-pointer register
mov ebp, esp ; ebp = esp
有时您可能会看到仅使用“ENTER”指令即可完成此操作。
第三部分:为局部变量留出空间
局部变量存储在堆栈中。由于堆栈向下增长,我们减去一些字节(足以存储我们的局部变量):
sub esp, n_bytes ; n_bytes = 局部变量所需的字节数
第 4 部分:将所有内容放在一起。使用栈基指针寄存器访问参数...
[ebp + 16] - parameter 'c'
[ebp + 12] - parameter 'b'
[ebp + 8] - parameter 'a'
[ebp + 4] - return address
[ebp + 0] - saved stackbase-pointer register
使用堆栈指针寄存器访问局部变量...
[esp + (# - 4)] - top of local variables section
[esp + 0] - bottom of local variables section
第 5 部分:堆栈框架清理
当我们离开例程时,必须清理堆栈帧。
mov esp, ebp ; undo the carving of space for the local variables
pop ebp ; restore the previous stackbase-pointer register
有时您可能会看到“LEAVE”指令取代了这两条指令。
根据您使用的语言,您可能会看到“RET”指令的两种形式之一。
ret
ret <some #>
选择哪一个将取决于语言的选择(或者如果使用汇编程序编写,您希望遵循的风格)。第一种情况表明调用者负责从堆栈中删除参数(在 foo(a,b,c) 示例中,它将通过 ... add esp, 12 执行此操作),这就是“C”的方式它。第二种情况表示返回指令返回时会从堆栈中弹出 # 个字(或 # 个字节,我不记得是哪一个),从而从堆栈中删除参数。如果我没记错的话,这是 Pascal 使用的风格。
关于assembly - 汇编中的栈帧是什么?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/3699283/
栈一种常见的特殊线性数据结构,其特殊之处在于其操作顺序,下面会详细介绍,也正因为其特性,因此栈可以轻松解决表达式求值、括号匹配、递归算法、回溯算法等等问题。 01、定义 栈的特殊性表现为操作受
目录 实践1 —— 从字符串中移除星号 栈和数组存储数据的方式一样,它们都只是元素的列表。不同之处在于栈的以下3个限制: 数据只能从栈末插入; 数据只
准备工作 工具:idea+jdk8 技术要求:java基础语法 编码环节 首先,我们得先确定下来,用什么数据来模拟栈的操作。由于是一个一个的元素放入栈里面,我们可以考虑用数组来实现。
0. 学习目标 栈和队列是在程序设计中常见的数据类型,从数据结构的角度来讲,栈和队列也是线性表,是操作受限的线性表,它们的基本操作是线性表操作的子集,但从数据类型的角度来讲,它们与线性表又有着巨大的不
在可以使用递归(在堆栈中存储状态)和对象创建(在堆中创建新对象)的场景中。 问题 在对象创建和递归之间进行选择时应考虑哪些参数? 我的研究得出以下结论(需要验证这一点) 当可用内存较少时:使用递归 可
下面是我编写的用于检查内存对齐的示例程序。 Pavan@Pavan-pc:~/working_dir/pavan/C$ cat mem3.c #include #include
Instapaper 和 Twitterrific 等应用启动的 View 不是其导航堆栈的 Root View 。我们知道这一点,因为初始 View 已经有一个后退按钮。 Instapaper 推出
有没有办法在调试或正常运行期间的某个时刻可视化 Activity 堆栈? 最佳答案 您可以通过 Activity 管理器获取一些有用的信息。 ActivityManager manag
我想编写一个应用层协议(protocol),在发送 GET 请求时使用 TCP 返回特定的 ASCII 文本。我读了第一HTTP specification和 the SMTP specificati
1、堆和栈的速度性能分析 堆和栈是jvm内存模型中的2个重要组成部分,自己很早以前也总结过堆和栈的区别,基本都是从存储内
一: 概念 栈,同样是一种特殊的线性表,是一种last in first out(lifo)的形式,现实中有很多这样的例子,
java中stack类继承于vector,其特性为后进先出(lastinfirstout). 入栈和出栈实例图: 实例图的java代码实例: ?
1、单链表 1、在我们数据结构中,单链表非常重要。它里面的数据元素是以结点为单位,每个结点是由数据元素的数据和下一个结点的地址组成,在java集合框架里面 LinkedList、Ha
本文实例讲述了Python编程实现双链表,栈,队列及二叉树的方法。分享给大家供大家参考,具体如下: 1.双链表 ?
我一遍又一遍地阅读定义,但我仍然不明白ARM中的SP和LR是什么?我了解PC(它显示下一条指令的地址),SP和LR可能类似,但我只是不明白它是什么。你能帮我一下吗? 编辑:如果你能用例子来解释它,那就
我必须使用索引 0 作为堆栈的顶部,并且在实现此操作时遇到问题。我得到了所有 null,但输出 100、200 和 300 是我得到的唯一数字。我忽略的实现有什么问题吗? push 方法应该实现 Ar
我正在用 Java 解决汉诺塔问题。我选择使用 Stacks 作为钉子,除了 move 方法之外,一切都正常。我有规范和 JUnit 测试类,目前通过了 7 项测试中的 6 项,但在移动测试中失败了。
这个问题在这里已经有了答案: 关闭 11 年前。 Possible Duplicate: Does this type of memory get allocated on the heap or
首先,抱歉我的英语不好。我将尝试解释我的问题: 我有一个 RootViewController(基于导航的项目)。因此,它显示了表格 View ,当用户选择表格的一行 (didSelectRowAtI
我有一个看起来像这样的类 class A { int b; void B() { int c; } } int main() { A asdf;
我是一名优秀的程序员,十分优秀!