- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
我试图理解一些汇编代码并设法完成除了几行之外的大部分代码。我能够理解内部发生的大部分事情,但无法完全理解代码开头和结尾处发生了什么(以及为什么)。有人可以阐明这一点吗?
int main() {
int a, b;
a = 12;
b = 20;
b = a + 123;
return 0;
}
反汇编版本:
8048394:8d 4c 24 04 lea 0x4(%esp),%ecx ; ??
8048398:83 e4 f0 and $0xfffffff0,%esp ; ??
804839b:ff 71 fc pushl -0x4(%ecx) ; ??
804839e:55 push %ebp ; Store the Base pointer
804839f:89 e5 mov %esp,%ebp ; Initialize the Base pointer with the stack pointer
80483a1:51 push %ecx ; ??
80483a2:83 ec 4c sub $0x4c,%esp ; ??
80483a5:c7 45 f8 0c 00 00 00 movl $0xc,-0x8(%ebp) ; Move 12 into -0x8(%ebp)
80483ac:c7 45 f4 14 00 00 00 movl $0x14,-0xc(%ebp) ; Move 20 into -0xc(%ebp)
80483b3:8b 45 f8 mov -0x8(%ebp),%eax ; Move 12@-0x8(%ebp) into eax
80483b6:83 c0 7b add $0x7b,%eax ; Add 123 to 12@eax
80483b9:89 45 f4 mov %eax,-0xc(%ebp) ; Store the result into b@-0xc(%ebp)
80483bc:b8 00 00 00 00 mov $0x0,%eax ; Move 0 into eax
80483c1:83 c4 10 add $0x10,%esp ; ??
80483c4:59 pop %ecx ; ??
80483c5:5d pop %ebp ; ??
80483c6:8d 61 fc lea -0x4(%ecx),%esp ; ??
最佳答案
堆栈向下增长。 push
从堆栈指针 (esp) 减去,pop
添加到 esp。你必须牢记这一点才能理解其中的很多内容。
8048394:8d 4c 24 04 lea 0x4(%esp),%ecx ; ??
lea = 加载有效地址
这会将位于堆栈中 4 个字节的东西的地址保存到堆栈中。由于这是 32 位(4 字节字)x86 代码,这意味着堆栈中的第二项。由于这是函数的代码(在本例中为 main),堆栈顶部的 4 个字节是返回地址。
8048398:83 e4 f0 and $0xfffffff0,%esp ; ??
此代码确保堆栈对齐到 16 字节。在此操作之后 esp 将小于或等于此操作之前的值,因此堆栈可能会增长,从而保护堆栈上可能已经存在的任何内容。这有时是在 main
中完成的,以防使用未对齐的堆栈调用函数,这可能会导致事情变得非常慢(我认为 16 字节是 x86 上的缓存行宽度,虽然 4 字节对齐在这里非常重要)。如果 main 有一个未对齐的堆栈,程序的其余部分也将如此。
804839b:ff 71 fc pushl -0x4(%ecx) ; ??
因为 ecx 之前是作为指向返回地址另一边的指针从栈顶加载的,所以因为它有一个 -4 索引,所以它指的是返回当前函数的返回地址被推回栈顶,以便 main 可以正常返回。 (推送很神奇,似乎能够在同一条指令中从 RAM 的不同位置加载和存储)。
804839e:55 push %ebp ; Store the Base pointer
804839f:89 e5 mov %esp,%ebp ; Initialize the Base pointer with the stack pointer
80483a1:51 push %ecx ; ??
80483a2:83 ec 4c sub $0x4c,%esp ; ??
这主要是标准的函数序言(之前的内容是 main 的特殊内容)。这正在创建一个堆栈框架(ebp 和 esp 之间的区域),局部变量可以存在于其中。 ebp 被压入,以便可以在结语中(在当前函数的末尾)恢复旧的堆栈帧。
80483a5:c7 45 f8 0c 00 00 00 movl $0xc,-0x8(%ebp) ; Move 12 into -0x8(%ebp)
80483ac:c7 45 f4 14 00 00 00 movl $0x14,-0xc(%ebp) ; Move 20 into -0xc(%ebp)
80483b3:8b 45 f8 mov -0x8(%ebp),%eax ; Move 12@-0x8(%ebp) into eax
80483b6:83 c0 7b add $0x7b,%eax ; Add 123 to 12@eax
80483b9:89 45 f4 mov %eax,-0xc(%ebp) ; Store the result into b@-0xc(%ebp)
80483bc:b8 00 00 00 00 mov $0x0,%eax ; Move 0 into eax
eax 是存储整数函数返回值的地方。这是设置从 main 返回 0。
80483c1:83 c4 10 add $0x10,%esp ; ??
80483c4:59 pop %ecx ; ??
80483c5:5d pop %ebp ; ??
80483c6:8d 61 fc lea -0x4(%ecx),%esp ; ??
这是函数结尾。因为一开始的奇怪的堆栈对齐代码,所以比较难理解。不过,我很难弄清楚为什么这次堆栈的调整量比序言中的少。
很明显,此特定代码未在优化的情况下编译。如果它在那里可能不会太多,因为编译器可以看到即使它没有执行 main
中列出的数学运算,程序的最终结果也是相同的。对于实际做某事(有副作用或结果)的程序,有时更容易阅读轻微优化的代码(gcc 的 -O1 或 -0s 参数)。
对于不是 main
的函数,读取由编译器生成的程序集通常要容易得多。如果您想通过阅读来理解代码,那么您可以自己编写一个函数,该函数采用一些参数来生成结果或对全局变量起作用,您将能够更好地理解它。
另一件可能对您有帮助的事情是让 gcc 为您生成汇编文件,而不是反汇编它们。 -S
标志告诉它生成此文件(但不生成其他文件),并在末尾使用 .s
命名程序集文件。这应该比反汇编版本更容易阅读。
关于c - 理解一些汇编语句的目的,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/4228261/
我正致力于通过 OAuth 合并外部 API,但对 expires_in 属性的用途有点迷惑。通过阅读,应该对 api token 的使用进行防御性编码,因为您应该预料到 token 在任何时候都可能
有人可以概述或总结一下 Spring 框架上下文中 bean 的用途吗? 我了解标准的 Java bean(没有 arg 构造函数、getter/setter,通常是序列化的),但 Spring be
使用 OpenGL 4.1 和 ARB_separate_shader_objects,我们能够在着色器程序中存储着色管道的不同阶段。众所周知,要使用这些,我们需要将它们附加到程序管道对象,然后绑定(
正如我从文档中了解到的那样,“MoveIteratorFactory”的目的是生成每一步都需要执行的 Action 。 “getSize”方法的移动子集有多大? “createOriginalMove
请解释 CMakeLists.txt 中这一行的目的是什么: 包括(InstallRequiredSystemLibraries) 我在 CMake 示例中看到这一行,但找不到好的解释,为什么我需要它
这里是新手。我仍在尝试理解在多个布局中运行单个进程或目的的概念。 例如,我想在我的申请中添加“提交后”功能。有一个包含标题、内容等文本框的主布局,以及一个链接到另一个布局以选择类别的按钮。我的问题是,
我在看 Box Oauth2.0 View Controller : https://github.com/box/box-ios-sdk-v2/blob/master/BoxSDK/OAuth2/B
我编写了一个将字符串复制到系统剪贴板的 Java 应用程序。构造函数使用 Clipboard.setContents(Transferable contents, ClipboardOwner own
阅读此文后:http://sourcemaking.com/design_patterns/command 我还是不太明白为什么我们需要这个。 最佳答案 想法是,如果命令被封装为对象,那么这些命令可以
我知道 c++ 中的模板是做什么的,但是今天我看到了一些奇怪的代码: template <> void swap(foo &a, foo &b) { a.name = b.name; a.
我不太明白 C# Collections 中 IEnumerator 的用途是什么。它的用途是什么,为什么要使用它? 我试着在线查看 http://msdn.microsoft.com/en-us/l
不幸的是,我今天做了一些代码考古(同时重构了一些旧的危险代码)并发现了这样的小化石: # line 7 "foo.y" 能在里面找到如此古老的宝藏,我完全惊呆了。我在 C 编程的网站上阅读了它。然而,
您能否澄清一下此注释的实际用途? - 如果我们没有使用数据库中的 SQL 表定义定义相应的约束,会发生什么情况。当我们尝试插入时,hibernate 会检查唯一性吗?或者这就是DB的目的吗?如果 hi
我在视频教程中看到过这段代码: const navToggle = ["Menu"].join(""); $(".site-header").prepend(navToggle); 我明白它的基本作用
我想知道这个成员函数的 scroll_to(TextBuffer::iterator& iter, double within_margin = 0)参数 within_margin。 API 是这样
我想知道是否可以将子目录提交到目录例如,假设您有 site.com/directory 可以将子目录提交到目录。我即将开始为希望她的网站在搜索引擎中排名靠前的客户进行一些搜索引擎优化。我知道实现此目的
STL 迭代器的用途是什么?为什么程序员要创造这个概念? 最佳答案 迭代器允许您将算法与容器分开。只要您有开始和结束迭代器,并且知道迭代器的功能(随机访问等),您就可以在迭代器指定的范围内进行操作。例
NSData *responseData = [NSURLConnection sendSynchronousRequest:theRequest returningResponse:&respons
我正在编写代码,使用通用的 linux i2c 驱动程序 linux/i2c-dev.h 实现一个简单的 i2c 读/写功能 我对 ioctl 感到困惑:I2C_SLAVE 内核文档说明如下: You
在尝试克隆可变集合时,我最初的方法是对 mutable.Cloneable 特征使用 clone() 方法。但是,这取决于创建引用副本的 java.Object.clone 实现,而不是深拷贝。通过测
我是一名优秀的程序员,十分优秀!