6

在 Solaris 10 上的 C 中,我想从进程中的任意线程获取调用堆栈。

我有许多工作线程和一个线程来监视它们以检测紧密循环和死锁。我想实现的功能是让监控线程在杀死它之前多次从“挂起”线程打印调用堆栈。

我知道如何通过让监控线程执行 pstack(使用 system() 或通过分叉)来实现这一点。但我希望能够在 C 中实现这个功能。有没有办法做到这一点?

我知道如何通过遍历堆栈来使线程打印其 OWN 调用堆栈,这在它遇到断言时很有用,但不知道如何为同一进程中的另一个线程执行此操作。

谢谢你的帮助。尼克B

4

2 回答 2

4

您可以使用walkcontext()遍历堆栈,使用dladdr()/dladdr1()将地址转换为函数名称。 walkcontext()为线程获取一个ucontext。如果您没有该线程的合作,那么您可以通过停止线程(例如使用PCTWSTOP)然后从该pr_oldcontext线程的 lwpstatus 结构的字段中读取其地址来获取它的 ucontext,从/proc/self/lstatus.

于 2009-08-21T15:46:09.593 回答
3

如果您使用 gcc,则可以使用内置函数 __builtin_return_address。void * __builtin_return_address(无符号整数级别)

该函数返回调用该函数的函数的地址。即函数的调用者。

级别指定了多少级别。0 表示当前函数 1 表示调用者,2 表示调用者调用者。以下示例将提供用法。通过打印函数的地址,可以确定调用堆栈。

int calla()
{
   printf("Inside calla\n");
   printf("A1=%x\n",__builtin_return_address (0));
   printf("A2=%x\n",__builtin_return_address (1) );
   printf("A3=%x\n",__builtin_return_address (2) );
}
int callb()
{
    printf("Inside callb\n");
    calle();
    printf("B1=%x\n",__builtin_return_address (0) );
    printf("B2=%x\n",__builtin_return_address (1) );
    printf("B3=%x\n",__builtin_return_address (2) );
}
int callc()
{
    printf("Inside callc\n");
    printf("C1=%x\n",__builtin_return_address (0) );
    printf("C2=%x\n",__builtin_return_address (1) );
    printf("C3=%x\n",__builtin_return_address (2) );
}
int calld()
{
    printf("Inside calld\n");
    printf("D1=%x\n",__builtin_return_address (0) );
    printf("D2=%x\n",__builtin_return_address (1) );
    printf("D3=%x\n",__builtin_return_address (2) );
}
int calle()
{
    printf("Inside calle\n");
    printf("E1=%x\n",__builtin_return_address (0) );
    printf("E2=%x\n",__builtin_return_address (1) );
    printf("E3=%x\n",__builtin_return_address (2) );
}
main()
{
    printf("Address of main=%x calla=%x callb=%x callc=%x calld=%x calle=%x\n",main,calla,callb,callc,calld,calle);
    calla();
    callb();
    calld();
}
于 2009-08-21T13:40:48.730 回答