9

我的 Mono 应用程序在 Mac 上崩溃并显示此消息(完整日志):

$ mono --debug bin/Debug/SparkleShare.app/Contents/MonoBundle/SparkleShare.exe
[...]
Stack overflow in unmanaged: IP: 0x26eb76, fault addr: 0xbf808ffc
[...]

“非托管”意味着堆栈溢出不在我的代码中(我只有托管代码),而是在我嵌入的库(SQLite、DotCmis、NewtonSoft.Json)或 Mono 的代码中。

即使我在调试模式下编译和运行,我得到的只是这两个十六进制。

问题:如何调查此堆栈溢出?有什么诀窍吗?

注意:相同的库(具有几乎相同的代码)在 Linux 和 Windows 上运行良好。

4

1 回答 1

5

处理堆栈溢出非常棘手(对于单声道),因此很可能堆栈溢出实际上是你的。问题在于找出堆栈跟踪。

我通常使用 gdb 运行:

gdb --args mono --debug bin/Debug/SparkleShare.app/Contents/MonoBundle/SparkleShare.exe

然后在堆栈开始增长之后尝试按 Ctrl+C,但在它实际溢出之前(gdb 与堆栈溢出严重混淆,发生这种情况时通常必须退出 gdb,这就是为什么你会需要在行动中捕捉溢出)。

按下 Ctrl+C 后,执行 a thread apply all backtrace,您将知道是否即将发生堆栈溢出(一个线程将有数千帧)。

一旦你在 gdb 中有一个巨大的堆栈跟踪,你需要确定循环。只需查看堆栈跟踪的地址,这通常很容易。获得此数据后,您可以像这样获得托管框架:

(gdb) p mono_pmip (0xdeaddead)
$1 = 0x0000dead  "Managed frame information shows up here"

然后对您找到的循环中的所有帧执行相同的操作。

这里有更多使用 gdb 调试单声道的技巧。

于 2012-12-03T13:09:05.480 回答