如何获取调用函数的名称?

bin*_*inW 18 c gcc function-pointers

我正在使用gnu工具链.我怎样才能在运行时找到函数的调用者?即,例如函数B()被许多函数使用函数指针调用.现在,每当B被调用时,我想打印调用者名称.我需要这个来调试某个问题.

ric*_*chq 18

如果您使用的是GNU,则可以使用回溯功能.在该手册页上有一个使用示例.


Fra*_*kH. 9

函数调用的代码位置__builtin_return_address()内部的gcc保存.要检索该名称,您必须解析程序的符号表; 虽然这是可能的,通过dladdr(),这是有限制的:

  1. 在所有上下文中调用backtrace()/ dladdr()(例如,从信号处理程序,或在多线程程序中同时执行,或从您无法调用的上下文中调用malloc())可能不安全.
  2. 操作不是即时的,但可能需要将调用线程置于睡眠状态; 如果在呼叫点保持任何锁定,这是一件坏事.
  3. 返回地址在名称中"可解析"的程度有限; 例如,如果有内联函数调用你的函数,那么你的调用者的调用者就会出现(backtrace()manpage也说明了这一点,就像dladdr()"BUGS"部分中的那样).
  4. 在C++中,还有通过con/destructors/initializers进行调用的额外挑战,其中函数的"调用者"最终可能是某些编译器生成的元代码,而您感兴趣的实际位置可能在其他地方(调用者的调用者) ,甚至更深入的调用堆栈).

这通常是解耦跟踪和函数名称解析的更好方法; 即只输出返回地址(作为十六进制/二进制),然后根据程序运行时检索到的符号表对结果日志进行后处理.


Dou*_*all 5

Vasil Dimov在回答类似问题时指出的另一种方法是用报告或传递调用函数名称的包装宏替换函数调用.这将适用于内联函数,其中backtrace不会.另一方面,如果您通过引用调用该函数或以其他方式获取其地址,则它将不起作用.

例如:

int B(int x){
    ...
}
Run Code Online (Sandbox Code Playgroud)

可能成为:

int _B(int x, char *caller){
    printf("caller is %s\n", caller);
    ...
}

#define B(x) _B((x), __func__)
Run Code Online (Sandbox Code Playgroud)

每次调用B()都会打印出来的姓名.Vasil Dimov以不同的方式构造它,直接在宏中打印名称并保持函数不变.