c - 如何从 `push %ebp`获取汇编指令的相对偏移量

标签 c assembly linker operating-system stack

当我使用disas时在gdb 。我可能会得到这样的东西。

(gdb) disas bar
Dump of assembler code for function bar:
   0x08048e84 <+0>: push   %ebp
   0x08048e85 <+1>: mov    %esp,%ebp
   0x08048e87 <+3>: sub    $0x8,%esp
   0x08048e8a <+6>: mov    0xc(%ebp),%eax
   0x08048e8d <+9>: mov    0x8(%ebp),%edx
   0x08048e90 <+12>:    add    %edx,%eax
   0x08048e92 <+14>:    mov    %eax,-0x4(%ebp)
   0x08048e95 <+17>:    mov    0x81f4074,%eax
   0x08048e9a <+22>:    mov    %eax,(%esp)
   0x08048e9d <+25>:    call   0x8048ed8 <traceback>
   0x08048ea2 <+30>:    mov    -0x4(%ebp),%eax
   0x08048ea5 <+33>:    mov    %eax,0x8(%ebp)
   0x08048ea8 <+36>:    leave  
   0x08048ea9 <+37>:    ret    
End of assembler dump.

假设我有0x08048ea2在我的C程序中。我怎样才能获得偏移量<+30>并得到0x08048e84

最佳答案

很难说清楚你到底在追求什么,但回溯函数通常会查看“堆栈帧”并从那里找到返回地址。然后,他们解析函数列表,查找每个返回地址的前一个函数。如果您在调试器中,那么也可能可以知道返回地址代表哪一行代码。

从这里我不确定问题是“如何评估堆栈帧”还是“如何找到“紧邻的前一个函数”。

堆栈帧的生成是特定于平台的,但通常使用特定的寄存器来保存地址,该地址是在函数入口处建立的堆栈上的固定位置。堆栈帧通常指向堆栈中前一个堆栈帧指针处(或旁边)的位置。这允许使用帧指针遍历堆栈并评估返回地址(通常位于相对于堆栈帧信息的固定位置。

请注意,编译器可以进行各种优化,这些优化会影响堆栈帧(具体有两种:禁用堆栈帧生成和返回值优化)。

确定函数地址也取决于平台。如果需要,编译器和链接器将创建和管理调试符号。这本质上嵌入了一个函数名称和起始地址的查找表,该查找表被加载到内存中供调试器访问。更好的系统还将提供在发布版本中查找符号名称的能力。他们通常通过让 IDE 提供函数基地址(映射文件)来实现此目的,然后找出代码段加载到内存中的位置。我怀疑此信息是由操作系统调用提供的,但也许还有其他机制。

关于c - 如何从 `push %ebp`获取汇编指令的相对偏移量,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/20921878/

相关文章:

c - 定义 ARM MCU 内存边界

c - c99和c11的区别

c - USB驱动编译错误

组装:有人可以解释一下带有 leaq 指令的这一行的作用吗?

assembly - "Divide error -Overflow"8086 emu 错误

string - 错误 : unable to find valid values for all labels after 1003 passes; stalled for 1001, 放弃

全局变量的 C ld 链接时错误

c - #define 在 C 中的效率

c++ - 如果我创建了一个进程,是否意味着我总能终止它?

C 包含库失败