c++ - 如何使用 GDB 从 C++ 功能逐步调试到 std::function 用户代码?
问题描述
例如,使用:
#include <functional>
#include <iostream>
int myfunc(int i){ return i + 1; }
int main() {
std::function<int(int)> f = myfunc;
int i = f(1);
std::cout << i << std::endl;
}
编译:
g++ -ggdb3 -O0 -std=c++11 -Wall -Wextra -pedantic -o main.out main.cpp
如果我尝试step
进入调用f(1)
,那么它首先会引导我进入 C++ 标准库代码,我要么必须非常努力地思考并做正确的next
/step
顺序,要么step
在到达实际myfunc
调用之前愚蠢并点击 17 次。
有没有更简单的方法来解决这个问题,也许使用一些现有的 GDB/Python 脚本?
这基本上是由 Microsoft 人员为 Visual Studio 完成的,如下所述:https ://devblogs.microsoft.com/cppblog/improving-the-debugging-experience-for-stdfunction/
我喜欢它可以在 Ubuntu 中默认进入 stdlibc++ 并且之前已经使用过它,但是如果 GDB 可以std::function
默认进入用户代码,并且有一些其他机制可以进入 libstdc++ 这个特定的案子。
我很想有点厚颜无耻,只使用以下重复命令 n 次的 Python GDB 脚本:gdb - 执行命令 n 次,它允许我这样做:
repeat-cmd 17 s
相关:QtCreator 调试器:进入 std::function
在 Ubuntu 18.04、GDB 8.1、GCC 7.4 中测试。
解决方案
在您的用例中不起作用的第一件事是
gdb 跳过 -gfile bits/*.h
正如您已经提到的,这也将跳过从 std 库中调用的代码。
下一个想法:跳过具有“已知名称”的函数。正如我们通常std::
在函数名前面使用的那样.gdbinit
。它看起来好多了:
sys.path.insert(0, '/usr/share/gcc-9/python/')
from libstdcxx.v6.printers import register_libstdcxx_printers
register_libstdcxx_printers (None)
import gdb
import re
def stop_handler(event):
frame_name = gdb.selected_frame().name();
if re.search("(operator new.*)|(^__gnu_cxx::.*)|(^std::.*)", frame_name) != None:
gdb.execute("step")
gdb.events.stop.connect(stop_handler)
end
我现在搜索了一段时间以找到将当前文件/行号组合放入 python 的内容,但没有找到任何内容。是否有任何 python 访问当前文件名和行号?如果是这样,在这种情况下这将简单得多。如果有人可以在此处添加一些信息,那就太好了!
备注:在 ddd 中,它不会很好地工作,因为来自 python 的内部步骤将不会被 gui 识别。所以绿色箭头停留在最后一个阶梯线上,而不是当前阶梯线上。如果有人在这种情况下也知道一个技巧,欢迎提供任何帮助!
更新:
如果您使用以下内容.gdbinit
python
import sys
sys.path.insert(0, '/usr/share/gcc-9/python/')
from libstdcxx.v6.printers import register_libstdcxx_printers
register_libstdcxx_printers (None)
import gdb
import re
def stop_handler(event):
sal = gdb.selected_frame().find_sal()
if sal == None:
gdb.execute("step")
else:
symtab = sal.symtab;
if symtab == None:
gdb.execute("step")
else:
file_name = symtab.fullname();
if re.search("(.*bits.*)", file_name ) != None:
gdb.execute("step")
gdb.events.stop.connect(stop_handler)
end
一切都在我的电脑上完美运行。作为附加提示:如果使用ddd
,您可以打开回溯窗口,然后绿色箭头也将跟随 python 内部 gdb 步骤。这对我来说非常有效!感谢您的提问!:-)
也许可以更新正则表达式以适应过滤更多外部库或更具选择性。
推荐阅读
- java - 如何在 Spring-Boot 中使用 SSL 证书并为 Android 客户端生成公钥
- cassandra - cassandra的多节点集群-无法互相闲聊
- c# - AspNetCore 中缺少 Range 标头
- swiftui - DragGesture 没有更新
- php - 错误:在此服务器上找不到请求的 URL /login。(拉拉维尔)
- django - Django 如何在会话中将 JSON 数据保存在视图之外并在视图内检索保存的数据?
- kubernetes - 获取特定的 Kubernetes pod
- monaco-editor - 我应该如何在摩纳哥编辑器中为令牌添加悬停
- python - 如何进行条件 JSON 模式验证
- vue.js - 在动态加载的组件中获取自我