自动化GDB:每10毫秒显示回溯

3
我想为gdb编写一个脚本,每10毫秒保存进程的backtrace(堆栈)。我该怎么做?
这可以看作是针对“一文不值”的人(即不能使用任何高级分析器)的调用图剖析。
是的,有很多高级分析器。适用于流行的CPU和流行的操作系统。Shark非常令人印象深刻且易于使用,但我想通过这样的脚本获得基本功能,使用gdb工具。

1
如果你的平台支持,最好使用Dtrace。 - Joshua Smith
我没有适用于该平台的dtrace。我有适用于该平台的gdb。 - osgx
3个回答

3

你能获取lsstack吗?也许你可以从应用程序外部运行它。为什么是10毫秒?在100毫秒或更长时间内,百分比将大致相同。如果应用程序太快,你可以通过外部循环人为地减慢它的速度,这也不会改变百分比。此外,如果应用程序运行足够长,并且你的目标是找出性能问题所在,你还可以使用Ctrl-C在gdb下手动获取样本。


我想要自动定期堆栈转储。对于运行几分钟的大型应用程序来说,按下ctrl-c需要10毫秒对我来说是不可能的。 - osgx
在手动模式下,当应用程序中存在>2-3个这样的热点时,存在无法获取堆栈的风险。 - osgx
不,我不能获取一个lsstack。我可以使用gdb,而且gdb在处理我的应用程序方面拥有更好的能力。 - osgx
@osgx:关于您的第二条评论,我认为您会发现它是有效的,原因如下。当修复了你所称的热点时,将会减少执行时间一定的百分比X(大概在5%至90%之间)。然后,任何随机堆栈采样遇到的问题的概率都大于等于X。例如,如果X是20%,则无论您以多慢的速度取样,您都会在约4个样本中看到该问题,并且这将告诉您需要修复的内容。然后重复进行。 - Mike Dunlavey
我不想手动做这件事!(Ctrl-C 将在一段时间后被终止)。我想用脚本来完成。使用脚本,我将获得相当类似的结果,但手动工作时,我将得到非常不同的点。如果您无法帮助我自动化 gdb,请随意删除此答案。 - osgx
显示剩余5条评论

1

(1) 手动。在shell中执行以下操作。在shell提示符上反复按Ctrl+C。

gdb -x print_callstack.gdb -p pid

或者,(2)在另一个 shell 上重复地向 pid 发送相同次数的信号,如下循环。
let count=0; \
while [ $count -le 100 ]; do \
  kill -INT pid ; sleep 0.10; \
  let $count=$count+1; \
done

(1) 中 print_callstack.gdb 的源代码如下:

set pagination 0
set $count = 0
while $count < 100
    backtrace
    continue
    set $count = $count + 1
end
detach
quit

pstack的man页面https://linux.die.net/man/1/pstack


备选方案是pstack或gstack。 - S R Bandi
S R Bandi,您能否将http链接添加到手册中(例如作为纯文本)? - osgx

0
cat > gdb.run
set pagination 0 
backtrace 
continue 
backtrace 
continue 
... as many more backtrace + continue's as needed
backtrace 
continue 
detach 
quit

当然,省略重复的换行符,但在这个论坛软件中如何输入单个换行符呢?:(

gdb -x gdb.run -p $pid

然后只需使用 do。
kill -INT $pid ; sleep 0.01

在另一个脚本的循环中。

kill -INT 是当你按下 ctrl-C 时操作系统执行的命令。读者的练习:让 gdb 脚本使用 $n 次迭代的循环。


网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接