- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
我正在关注 this tutorial关于 assembly 。
根据教程(我在本地也试过,也得到了类似的结果),源码如下:
int natural_generator()
{
int a = 1;
static int b = -1;
b += 1; /* (1, 2) */
return a + b;
}
编译为这些汇编指令:
$ gdb static
(gdb) break natural_generator
(gdb) run
(gdb) disassemble
Dump of assembler code for function natural_generator:
push %rbp
mov %rsp,%rbp
movl $0x1,-0x4(%rbp)
mov 0x177(%rip),%eax # (1)
add $0x1,%eax
mov %eax,0x16c(%rip) # (2)
mov -0x4(%rbp),%eax
add 0x163(%rip),%eax # 0x100001018 <natural_generator.b>
pop %rbp
retq
End of assembler dump.
(我添加的行号注释 (1)
、(2)
和 (1, 2)
。)
问题:为什么在编译后的代码中,静态变量b
的地址是相对于指令指针(RIP)的,它不断变化(参见 (1)
和 (2)
行),因此生成更复杂的汇编代码,而不是相对于可执行文件的特定部分,其中存储这样的变量?
根据提到的教程,有这样的部分:
This is because the value for
b
is hardcoded in a different section of the sample executable, and it’s loaded into memory along with all the machine code by the operating system’s loader when the process is launched.
(强调我的。)
最佳答案
使用RIP 相对寻址访问静态变量b
的主要原因有两个。首先是它使代码位置独立,这意味着它是否用于共享库或 position independent executable代码可以更容易地重新定位。第二个是它允许将代码加载到 64 位地址空间中的任何位置,而无需在指令中编码巨大的 8 字节(64 位)位移,而 64 位 x86 CPU 无论如何都不支持这种位移。
您提到编译器可以改为生成引用变量的代码,该代码相对于它所在的部分的开头。虽然这样做也具有与上面给出的相同的优点,但它不会使程序集任何不那么复杂。事实上,这会使它变得更复杂。生成的汇编代码首先必须计算变量所在段的地址,因为它只知道它相对于指令指针的位置。然后它必须将它存储在寄存器中,因此可以相对于该地址访问 b
(以及该部分中的任何其他变量)。
由于 32 位 x86 代码不支持 RIP 相对寻址,您的替代解决方案实际上是编译器在生成 32 位位置独立代码时所做的。它将变量b
放在全局偏移表(GOT)中,然后访问相对于GOT基址的变量。这是使用 gcc -m32 -O3 -fPIC -S test.c
编译时您的代码生成的程序集:
natural_generator:
call __x86.get_pc_thunk.cx
addl $_GLOBAL_OFFSET_TABLE_, %ecx
movl b.1392@GOTOFF(%ecx), %eax
leal 1(%eax), %edx
addl $2, %eax
movl %edx, b.1392@GOTOFF(%ecx)
ret
第一个函数调用将后续指令的地址放在 ECX 中。下一条指令通过添加 GOT 从指令开始的相对偏移量来计算 GOT 的地址。变量 ECX 现在包含 GOT 的地址,并在访问其余代码中的变量 b
时用作基础。
将其与 gcc -m64 -O3 -S test.c
生成的 64 位代码进行比较:
natural_generator:
movl b.1745(%rip), %eax
leal 1(%rax), %edx
addl $2, %eax
movl %edx, b.1745(%rip)
ret
(该代码与您问题中的示例不同,因为启用了优化。一般来说,只查看优化输出是个好主意,因为如果不进行优化,编译器通常会生成糟糕的代码,这些代码会做很多无用的事情。另请注意,不需要使用 -fPIC
标志,因为无论如何编译器都会生成 64 位位置独立代码。)
注意 64 位版本中如何少了两条汇编指令,使其成为不太复杂的版本。您还可以看到代码使用了一个较少的寄存器 (ECX)。虽然它对您的代码没有太大影响,但在一个更复杂的示例中,它是一个可以用于其他用途的寄存器。这使得代码更加复杂,因为编译器需要处理更多的寄存器。
关于assembly - 为什么静态变量的地址是相对于指令指针的?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/40329260/
当我编译我的 ionic 4 应用程序时,我得到这个错误:不开始。但是,如果我更改代码中的某些内容并再次编译,代码将成功编译。我查了一下它可能与路径(绝对/相对)有关。但在这些问题中,解决方案是使用相
现在我发现当我打开终端时,它总是显示这些消息: bash: /usr/local/texlive/2012/texmf/doc/man:: No such file or directory bash
我有一些具有不同 url 前缀的嵌套模块。现在我想在一个模块中导航而不指定前缀(在我的模块中,我不想知道在哪个前缀下可以访问该模块)。 这些是我的 app.module 的路线: const APP_
我想在java中进行日期范围搜索假设我想搜索从2019年10月22日到当前日期。但问题是在两周的 block 大小中进行日期范围搜索(考虑到这可能会有所不同,但以周为单位),例如这里开始日期将为 20
我正在尝试实现移动到单击鼠标的点。 但是我遇到了 X 轴镜像行为的问题。当我点击顶部 -> 它移动到底部,当我点击底部 -> 它移动到顶部。 这里是原始位置的例子 我点击了屏幕上有红十字的位置。 但它
尝试使用以下方法让对象跟随光标: int mx =(int) MouseInfo.getPointerInfo().getLocation().getX()-50; Player.setX(mx);
我有 4 个 JButton 设置在彼此下方。我希望当用户水平调整框架大小时它们左右移动。 例如:帧大小:400,400 按钮位置:300,200 现在我将框架大小调整为:600,400 Button
我想要做的是将一个元素(我用作背景的彩色 UIView)定位到我的 Storyboard 中,以便它从 ImageField 的中间开始。并填充所有内容,直到屏幕底部。我正在使用 xcode7 和 s
我正在编写一些 C++ 代码,它主要为共享它的其他两个项目提供一个类,但也包括一个小程序,以便在需要时可以从命令行使用它。该类必须加载一些资源,这些资源被写入资源文件夹中的多个文件。这些文件的路径当然
我能以某种方式随时获得相对于帧初始引用的加速度矢量吗? (我的意思是:xArbitraryZVertical 模式下的帧引用,我第一次得到 Core Motion 数据)我试图做什么:每次我得到 CM
saved 我希望 div#one 在父 div 的左边缘和提交按钮的左边缘之间的空间居中。 最佳答案 还有几种方法: saved 很难说 saved 文本没有居中在容器 d
所以我在页面上有一个带有背景图像的对象,在 mousemove 上它移动了相对于鼠标的背景位置。但我遇到的唯一问题是在鼠标进入对象时将背景图像动画化到鼠标的当前位置。 我能够将背景位置动画化回其原始位
我的一个网站中有一个图像缩放属性。我想相对于 div 的中心缩放图像。 缩放功能如下。 function zoom(zm) { img=document.getElementById("pic"
我正在尝试调整水平导航栏左侧的导航栏 Logo 的大小,然后让其余导航栏元素占据相同的垂直空间并在空间中垂直居中。导航栏元素目前不使用完整的垂直空间。我尝试过的每个尺寸属性都产生了另一个问题。感谢所有
我有一些 h2 文本当前在移动 View 中左对齐,位于居中的 div 上方。我怎样才能将它对齐到移动 View 中相对于 div 的左对齐(应用下面的 CSS 中提供的媒体查询)? CodePen
我想让文本元素(在本例中为 h2 副标题)居中,方法是让它忽略左侧的 float 图像。我更喜欢 h2 副标题与 h1 标题垂直居中对齐。有没有什么办法可以单独使用 CSS 来做到这一点? 这是一个示
您好! 我在这个网站上工作,但我一直遇到同样的问题。当我把边距放在百分比而不是像素时,它似乎是从包装器或页面中获取百分比。可能是一些愚蠢的错误,但我并没有真正使用过百分比。无论如何,我所说的类是“ L
我很好奇这是怎么做到的,我做了类似的东西,例如 Home About Search bar Container
我希望我的 firstLabel 位于同一行文本字段的右侧。 //css input[type="text"]{ display:block; margin-botto
我正在尝试绘制多边形,并希望能够单击我的框架以获取鼠标坐标,以便更快地将心理图像转换为 x/y 值。 我在用 System.out.println("("+ MouseInfo.getPointerI
我是一名优秀的程序员,十分优秀!