如何在gdb中打印<incomplete type>变量

问题描述 投票:0回答:6

有时 gdb 对于某些类型的变量会打印“不完整类型”。这意味着什么?我们如何才能看到该值?

gdb
6个回答
37
投票

这意味着该变量的类型尚未完全指定。例如:

struct hatstand;
struct hatstand *foo;

GDB 知道

foo
是指向
hatstand
结构的指针,但该结构的成员尚未定义。因此,“不完整类型”。

要打印该值,您可以将其转换为兼容类型。

例如,如果您知道

foo
实际上是指向
lampshade
结构的指针:

print (struct lampshade *)foo

或者,您可以将其打印为通用指针,或将其视为整数:

print (void *)foo
print (int)foo

另请参阅 GDB 手册中的这些页面:


9
投票

我发现,如果您反汇编一个使用不完整结构类型的函数,gdb 会“发现”结构成员并随后显示它们。例如,假设您有一个字符串结构:

struct my_string {
    char * _string,
    int _size
} ;

一些通过指针创建和获取字符串的函数:

my_string * create_string(const char *) {...}
const char * get_string(my_string *){...}

以及创建字符串的测试:

int main(int argc, char *argv[]) {
    my_string *str = create_string("Hello World!") ;
    printf("String value: %s\n", get_string(str)) ;
    ...
}

在 gdb 中运行它并尝试“print *str”,您将得到“不完整类型”响应。但是,尝试“反汇编 get_string”,然后“打印 *str”,它将正确显示结构和值。我不知道为什么会这样,但确实如此。


2
投票

免责声明:我是一名 Python 开发人员,对 C++ 和 Linux 操作系统的功能只有一些最基本的了解,所以我下面描述的只是我个人遇到的问题的解决方案。

如果您尝试使用来自第 3 方库的类型,请确保这些库没有缺少调试信息。

示例

(gdb) info share Qt
From                To                  Syms Read   Shared Object Library
0x00007ffff5336080  0x00007ffff56ba585  Yes (*)     /usr/lib/x86_64-linux-gnu/libQt5Widgets.so.5
0x00007ffff4ad3510  0x00007ffff4ef0cbe  Yes (*)     /usr/lib/x86_64-linux-gnu/libQt5Gui.so.5
0x00007ffff47829c0  0x00007ffff47e1ba1  Yes (*)     /usr/lib/x86_64-linux-gnu/libQt5DBus.so.5
0x00007ffff40bb5e0  0x00007ffff439dd92  Yes (*)     /usr/lib/x86_64-linux-gnu/libQt5Core.so.5
0x00007ffff2e581e0  0x00007ffff2e78e4f  Yes (*)     /usr/lib/x86_64-linux-gnu/libQt5Xml.so.5
0x00007ffff28c8a00  0x00007ffff29d9999  Yes (*)     /usr/lib/x86_64-linux-gnu/libQt5Network.so.5
0x00007ffff2251750  0x00007ffff2252a46  Yes (*)     /usr/lib/x86_64-linux-gnu/libQt5X11Extras.so.5
0x00007ffff1cc9f80  0x00007ffff1cfc861  Yes (*)     /usr/lib/x86_64-linux-gnu/libQt5PrintSupport.so.5
0x00007fffee269c10  0x00007fffee297b57  Yes (*)     /usr/lib/x86_64-linux-gnu/libQt5Svg.so.5
0x00007fffed987560  0x00007fffed98b6a8  Yes (*)     /usr/lib/x86_64-linux-gnu/libQt5TextToSpeech.so.5
0x00007fffe980e130  0x00007fffe9900c0c  Yes (*)     /usr/lib/x86_64-linux-gnu/libQt5XcbQpa.so.5
0x00007fffe69ef650  0x00007fffe69ffe0d  Yes (*)     /usr/lib/x86_64-linux-gnu/libQt5QuickControls2.so.5
0x00007fffe5c0f890  0x00007fffe5eae1c1  Yes (*)     /usr/lib/x86_64-linux-gnu/libQt5Quick.so.5
0x00007fffe5522690  0x00007fffe581f636  Yes (*)     /usr/lib/x86_64-linux-gnu/libQt5Qml.so.5
0x00007fffe51996b0  0x00007fffe5221363  Yes (*)     /usr/lib/x86_64-linux-gnu/libQt5QuickTemplates2.so.5
(*): Shared library is missing debugging information.

^ 当前示例中的所有 Qt 库都缺少调试信息。这是因为 qt 库的调试信息位于单独的包中,但未安装。

每当我这样做时

whatis
一切都工作正常:

(gdb) whatis e
type = QEvent *

但是当我尝试访问它的成员时

(gdb) p e->type()
Couldn't find method QEvent::type

并尝试获取详细的类型描述

(gdb) ptype e
type = class QEvent {
  <incomplete type>
} *

解决方案(适用于 Ubuntu 上的 Qt)

  1. 查找该文件属于操作系统发行版中的哪个包
$ dpkg -S /usr/lib/x86_64-linux-gnu/libQt5Core.so.5
libqt5core5a:amd64: /usr/lib/x86_64-linux-gnu/libQt5Core.so.5
  1. 搜索相关包

    用keywork

    libqt5core5a
    搜索返回了2个包,一个是
    libqt5core5a
    本身,另一个是
    libqt5core5a-dbgsym
    。后者的描述为:“libqt5core5a 的调试符号”

  2. 安装带有调试符号的包(我还为其他一些重要的 Qt 库安装了调试符号)

$ sudo apt install libqt5core5a-dbgsym libqt5widgets5-dbgsym libqt5gui5-dbgsym
  1. 确保
    gdb
    中的库现在具有调试信息
(gdb) info share Qt
From                To                  Syms Read   Shared Object Library
0x00007ffff5336080  0x00007ffff56ba585  Yes         /usr/lib/x86_64-linux-gnu/libQt5Widgets.so.5
0x00007ffff4ad3510  0x00007ffff4ef0cbe  Yes         /usr/lib/x86_64-linux-gnu/libQt5Gui.so.5
0x00007ffff47829c0  0x00007ffff47e1ba1  Yes (*)     /usr/lib/x86_64-linux-gnu/libQt5DBus.so.5
0x00007ffff40bb5e0  0x00007ffff439dd92  Yes         /usr/lib/x86_64-linux-gnu/libQt5Core.so.5
0x00007ffff2e571e0  0x00007ffff2e77e4f  Yes (*)     /usr/lib/x86_64-linux-gnu/libQt5Xml.so.5
0x00007ffff28c7a00  0x00007ffff29d8999  Yes (*)     /usr/lib/x86_64-linux-gnu/libQt5Network.so.5
0x00007ffff2250750  0x00007ffff2251a46  Yes (*)     /usr/lib/x86_64-linux-gnu/libQt5X11Extras.so.5
0x00007ffff1cc8f80  0x00007ffff1cfb861  Yes (*)     /usr/lib/x86_64-linux-gnu/libQt5PrintSupport.so.5
0x00007fffee268c10  0x00007fffee296b57  Yes (*)     /usr/lib/x86_64-linux-gnu/libQt5Svg.so.5
0x00007fffed985560  0x00007fffed9896a8  Yes (*)     /usr/lib/x86_64-linux-gnu/libQt5TextToSpeech.so.5
0x00007fffe95fc130  0x00007fffe96eec0c  Yes (*)     /usr/lib/x86_64-linux-gnu/libQt5XcbQpa.so.5
0x00007fffe701f650  0x00007fffe702fe0d  Yes (*)     /usr/lib/x86_64-linux-gnu/libQt5QuickControls2.so.5
0x00007fffe623c890  0x00007fffe64db1c1  Yes (*)     /usr/lib/x86_64-linux-gnu/libQt5Quick.so.5
0x00007fffe5b4f690  0x00007fffe5e4c636  Yes (*)     /usr/lib/x86_64-linux-gnu/libQt5Qml.so.5
0x00007fffe57c66b0  0x00007fffe584e363  Yes (*)     /usr/lib/x86_64-linux-gnu/libQt5QuickTemplates2.so.5
(*): Shared library is missing debugging information.
  1. 现在使用 Qt 类型可以按预期工作
(gdb) p e->type()
$4 = QEvent::Paint
(gdb) ptype e
type = class QEvent {
  public:
...
}

0
投票

我也有同样的问题。如果您手动从库中加载符号:

set auto-solib-add off
attach thread_id
shared any_lib
shared another_lib

您还需要使用相同的命令从声明该对象的库中加载符号。


0
投票

我不知道该错误的完整含义,但正如 Peter 指出的那样,相关方法的反汇编会导致其中一些类型定义可用。

我的例子:

在类的 .h 中,该类包含内部辅助类的前向声明,以便外部类可以包含指向它的指针。相应的 .cpp 具有完整的内部辅助类定义。

在破坏外部类的方法时,gdb 报告通过外部类的实例取消引用内部类实例的指针的不完整类型。

对外部类中的方法之一发出 disasemble 命令允许 gdb 使用先前失败的相同指针来理解内部类的结构。


0
投票

我在尝试在 Docker 容器内调试

libpam
时遇到了同样的问题。当尝试从
<incomplete type>
打印
pamh
参数时,它会显示
pam_sm_authenticate

最初,调试符号丢失,但从源代码构建库后仍然无法工作。

当我尝试使用 LLDB 而不是 GDB 时,我收到此错误:

error: 'A' packet returned an error: 8

这让我找到了这个答案,其中指出

ptrace
系统调用似乎默认在容器内被禁用。

就我而言,我并不真正关心安全性,因为我只是使用容器来创建可重现的环境,因此,我通过将以下参数

--cap-add=SYS_PTRACE --security-opt seccomp=unconfined
添加到
docker run
来允许执行。

© www.soinside.com 2019 - 2024. All rights reserved.