如何从调试符号中查找已编译的变量/函数地址

时间:2016-04-19 18:23:08

标签: c gcc

我找到了关于如何拆分编译文件和调试符号的以下帖子(How to generate gcc debug symbol outside the build target?)。

但是,我在调试文件中找不到任何有用的信息。

例如,

我的helloWorld代码是:

#include<stdio.h>

int main(void) {

    int a;
    a = 5;
    printf("The memory address of a is: %p\n", (void*) &a);
    return 0;
}

我跑了gcc -g -o hello hello.c

objcopy --only-keep-debug hello hello.debug
gdb -s main.debug -e main

在gdb中,我试过的任何东西都没有给我任何关于a的信息,我找不到它的地址,我找不到主函数地址

例如:

(gdb) info variables
All defined variables:

Non-debugging symbols:
0x0000000000400618  _IO_stdin_used
0x0000000000400710  __FRAME_END__
0x0000000000600e3c  __init_array_end
0x0000000000600e3c  __init_array_start
0x0000000000600e40  __CTOR_LIST__
0x0000000000600e48  __CTOR_END__
0x0000000000600e50  __DTOR_LIST__
0x0000000000600e58  __DTOR_END__
0x0000000000600e60  __JCR_END__
0x0000000000600e60  __JCR_LIST__
0x0000000000600e68  _DYNAMIC
0x0000000000601000  _GLOBAL_OFFSET_TABLE_
0x0000000000601028  __data_start
0x0000000000601028  data_start
0x0000000000601030  __dso_handle
0x0000000000601038  __bss_start
0x0000000000601038  _edata
0x0000000000601038  completed.6603
0x0000000000601040  dtor_idx.6605
0x0000000000601048  _end

我做错了吗?我是否错误地理解了调试文件?有没有办法从保存的调试信息中找出已编译变量/函数的地址?

1 个答案:

答案 0 :(得分:2)

int a是一个堆栈变量,因此除非您正在调用该特定函数,否则它没有固定的地址。此外,对该函数的每次调用都将分配自己的变量。

当我们说“调试符号”时,我们通常指的是函数和全局变量。在这种情况下,局部变量不是“符号”。实际上,如果使用优化启用int a进行编译,几乎肯定会优化为寄存器变量,因此根本不会有地址,除非您强制将其写入内存some_function(&a)或类似。

只需在GDB中编写main,即可找到print main的地址。这是因为当函数出现在值上下文中时,函数被隐式转换为C中的指针,而GDB的print使用C语义。