我正在运行 python 3.6.6-debug(通过 pyenv 安装)并且我已经将关联的 libpython.py 从复制cpython/Tools/gdb/libpython.py
到~/.config/gdb
(签出 v3.6.6 标记)。
在我的 .gdbinit 中,我有:
source ~/.config/gdb/libpython.py
对于简单的过程,我可以毫无问题地使用py-list
,等,但是我目前在 py.test 下测试的程序对于任何 python gdb 帮助器命令都会给我这个错误:py-bt
(gdb) py-list
Python Exception <class 'RuntimeError'> Type does not have a target.:
Error occurred in Python command: Type does not have a target.
这个错误是什么意思,我该如何解决?
更新
我深入研究了 libpython.py 以确切了解py-list
/py-bt
做他们的事情,然后从 gdb 中手动运行相关的 gdb python 命令以重现问题并准确隔离 libpython.py 中问题发生的位置。在进行下面的调试后,我能够在 gdb 中获得更详细的回溯:
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
File "~/git/cpython/Tools/gdb/libpython.py", line 916, in filename
File "~/git/cpython/Tools/gdb/libpython.py", line 1158, in proxyval
RuntimeError: Type does not have a target.
该问题在 libpython.py 第1158行触发,即
fields = gdb.lookup_type('PyUnicodeObject').target().fields()
这澄清了一些事情:libpython.py 获取 PyUnicodeObject 的 Type 对象,然后尝试调用target
它的方法,但是 PyUnicodeObject 的 Type 对象没有 target。根据gdb 文档:
— 函数:Type.target()
返回一个新的 gdb.Type 对象,它表示该类型的目标类型。
对于指针类型,目标类型是指向对象的类型。对于数组类型(意思是类 C 数组),目标类型是数组元素的类型。对于函数或方法类型,目标类型是返回值的类型。对于复杂类型,目标类型是元素的类型。对于 typedef,目标类型是别名类型。
如果该类型没有目标,此方法将抛出异常。
这绝对看起来像一个错误,尽管我在其他任何在线、python 问题跟踪器或 python 提交历史中都找不到任何提及此问题的内容。我将在 python 跟踪器上打开一个问题,看看维护人员怎么说(除非有人之前遇到过这个问题并提交了答案)。
我是如何调试的
配置ptrace
为允许在没有 sudo 的情况下进行调试
$ sudo sh -c 'echo 0 > /proc/sys/kernel/yama/ptrace_scope
确定挂起(多处理)程序的父python进程
$ pstree -p -s 22391
systemd(1)───tmux(31719)───bash(5161)───py.test(22391)─┬─py.test(22478)
├─py.test(24577)
├─py.test(24578)
├─python3.6(25427)
├─python3.6(25545)
├─python3.6(25546)
├─python3.6(25547)
├─python3.6(27376)───{python3.6}(27393)
├─python3.6(30563)───{python3.6}(30580)
├─{py.test}(27368)
├─{py.test}(30562)
├─{py.test}(629)
└─{py.test}(630)
(我只是在上面猜对了,对我的程序使用任何正在运行的 python 进程的 pid 都可以)
附加到父进程
$ gdb -p 22391
确定最近的python执行帧并切换到它
(gdb) bt 10
#0 0x00007fec7309a5d3 in select () at ../sysdeps/unix/syscall-template.S:84
#1 0x00007fec738692aa in pysleep (secs=50000000) at ./Modules/timemodule.c:1417
#2 0x00007fec738671a3 in time_sleep (self=0x7fec71a00458, obj=0x7fec6cf728b0) at ./Modules/timemodule.c:235
#3 0x00007fec7368513e in _PyCFunction_FastCallDict (func_obj=0x7fec719ff5f8, args=0x7fec406fac08, nargs=1, kwargs=0x0) at Objects/methodobject.c:209
#4 0x00007fec73685535 in _PyCFunction_FastCallKeywords (func=0x7fec719ff5f8, stack=0x7fec406fac08, nargs=1, kwnames=0x0) at Objects/methodobject.c:294
#5 0x00007fec7379ab0d in call_function (pp_stack=0x7ffc37032440, oparg=1, kwnames=0x0) at Python/ceval.c:4830
#6 0x00007fec737927ca in _PyEval_EvalFrameDefault (f=0x7fec406faa58, throwflag=0) at Python/ceval.c:3328
===> #7 0x00007fec7377eb3b in PyEval_EvalFrameEx (f=0x7fec406faa58, throwflag=0) at Python/ceval.c:754
#8 0x00007fec7363a208 in gen_send_ex (gen=0x7fec3d0b88d8, arg=0x0, exc=0, closing=0) at Objects/genobject.c:189
#9 0x00007fec7363bca6 in gen_iternext (gen=0x7fec3d0b88d8) at Objects/genobject.c:563
(More stack frames follow...)
(gdb) frame 7
#7 0x00007fec7377eb3b in PyEval_EvalFrameEx (f=0x7fec406faa58, throwflag=0) at Python/ceval.c:754
754 Python/ceval.c: No such file or directory.
添加 python 源目录并用于tui enable
获取一些上下文
(gdb) dir ~/git/cpython
Source directories searched: /home/calid/git/cpython:$cdir:$cwd
(gdb) tui enable
启动gdb的交互式python解释器,手动输入libpython行获取当前python脚本/行号
(gdb) pi
>>> gdbframe = gdb.selected_frame()
>>> f = gdbframe.read_var('f')
>>> pyframe = PyFrameObjectPtr.from_pyobject_ptr(f)
>>> pyframe.filename()
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
File "~/git/cpython/Tools/gdb/libpython.py", line 916, in filename
File "~/git/cpython/Tools/gdb/libpython.py", line 1158, in proxyval
RuntimeError: Type does not have a target.
这重现了我用py-list
and看到的异常py-bt
,但这次我也得到了一个非常有用的回溯。
默认开启 gdb python 堆栈跟踪
set python print-stack full
毕竟,我偶然发现了上述选项的文档。设置默认打开堆栈跟踪打印并且可以避免进行所有手动调试的需要......所以事后看来我做了很多我不需要做的额外工作:)(虽然我确实学到了很多东西进行中)。
我现在已经将此添加到我的 gdbinit 中以备将来使用。
资源
- 使用 gdb 调试 Python
- 使用 GDB 进行低级 Python 调试(这个非常棒,对我有很大帮助)
- Python 的内脏:你好,ceval.c!