首页
学习
活动
专区
圈层
工具
发布
社区首页 >问答首页 >GDB-Python脚本:遍历C/C++ struct字段的任何示例

GDB-Python脚本:遍历C/C++ struct字段的任何示例
EN

Stack Overflow用户
提问于 2013-02-05 00:13:44
回答 1查看 8.1K关注 0票数 13

新的GDB-Python脚本API看起来相当强大,应该非常有用。但是,编写一个有用的脚本来迭代C或C++结构中的字段并不容易。有谁知道一些固体样品能做到这一点吗?

提前谢谢。

更新最后一个示例:替换早期示例中的_print_fields()

代码语言:javascript
复制
    if l.type.code == gdb.TYPE_CODE_STRUCT:
        print "Found a struct  %s " % n
        #self._print_fields(n, t)
        self._print_deep_items(n, t, l)
    else:
        print "Found no struct"

def _print_deep_items (self, n_, type_, instance_):
    for fld in type_.fields():
        fn = fld.name
        ft = fld.type
        fv = instance_[fn]
        if fv.type.code == gdb.TYPE_CODE_STRUCT:
            print "  Found a sub struct  %s " % fn
            self._print_deep_items(fn, ft, fv)
        else:
            print "    Field %s " % fn, " type %s " % ft.tag, " value %s " % fv

以及产出:

代码语言:javascript
复制
  variable s1   type S1
Found a struct  s1
    Field v1   type None   value 0
    Field v2   type None   value 0
  Found a sub struct  v3
    Field w3   type None   value 0

使用第一个示例进行更新:使以下示例代码正常工作。这不是最优的,因为它在组成字符串字段名后对每个字段进行查找。abarnert显示了一种很有前途和优雅的方法,工作代码在上面最后的更新部分中被更新。

代码语言:javascript
复制
import gdb
class PrintGList(gdb.Command):
    """print fields of a struct: wzd struct_object

Iterate through the fields of a struct, and display
a human-readable form of the objects."""
    def __init__(self):
        gdb.Command.__init__(self, "wzd", gdb.COMMAND_DATA, gdb.COMPLETE_SYMBOL, True)

    def invoke(self, arg, from_tty):

        arg_list = gdb.string_to_argv(arg)
        if len(arg_list) < 1:
            print "usage: wzd struct"
            return

        n = arg_list[0]
        l = gdb.parse_and_eval(arg_list[0])
        m = l.type.tag

        print "  variable %s " % n, " type %s " % m
        try:
            t = gdb.lookup_type(m)
        except RuntimeError, e:
            print "type %s not found" % t
            return

        if l.type.code == gdb.TYPE_CODE_STRUCT:
            print "Found a struct  %s " % n
            self._print_fields(n, t)
        else:
            print "Found no struct"

    def _print_fields(self, n, typeobject):
        print typeobject
        flds = typeobject.fields()
        for x in flds:
            sn = n + "." + x.name
            print "  field %s" % sn, " code %s " % x.type.code, " type %s " % x.type.tag
            if x.type.code == gdb.TYPE_CODE_STRUCT:
                print "Found sub level struct  %s " % sn
                sl = gdb.parse_and_eval(sn)
                sm = sl.type.tag
                st = gdb.lookup_type( sm )
                self._print_fields(sn, x.type)

    def _deep_items (self, type_):
        for k, v in type_.iteritems():
            if k:
                print " k v %s " % k , " %s " % v
            else:
                print "   v    ",      " %s " % v

PrintGList()

要测试的源文件:

代码语言:javascript
复制
struct S2 {        int w3;    };
struct S1 {        int v1, v2;      struct S2 v3; } s1;
int main(int argc, char *argv[]) {   return 0; }

示例输出:

代码语言:javascript
复制
  variable s1   type S1
Found a struct  s1
S1
  field s1.v1  typecode 8   type None
  field s1.v2  typecode 8   type None
  field s1.v3  typecode 3   type S2
Found sub level struct  s1.v3
S2
  field s1.v3.w3  typecode 8   type None

获得: source /home/me/testpath/wzdfile.py文件a.out b主r wzd s1退出的GDB会话

EN

回答 1

Stack Overflow用户

回答已采纳

发布于 2013-02-05 01:08:43

根据医生们,迭代C结构的字段应该非常简单:

如果类型是结构或类类型,或者枚举类型,则可以使用Python字典语法访问该类型的字段。例如,如果some_type是一个包含结构类型的gdb.Type实例,则可以使用以下方法访问其foo字段: bar =某类‘’foo‘ bar将是一个gdb.Field对象;有关gdb.Field类的描述,请参见下面对Type.fields方法的描述。

您也可以使用Type.fields显式地获取struct的字段,但是(从7.4开始)也可以使用普通的dict方法,因此可以获得名称/Field对的列表:

代码语言:javascript
复制
for name, field in foo.type.iteritems():

或者,仅仅因为名字:

代码语言:javascript
复制
for name, field in foo.type.iterkeys():

诸若此类。

这似乎并没有直接记录在页面上,但是当gdb.typesdeep_items的时候,它非常强烈地暗示了这一点

返回类似于标准gdb.Type.iteritems方法的Python迭代器。

例如,考虑到这种C类型:

代码语言:javascript
复制
struct S {
    int x;
    int y;
};

你可以这样做:

代码语言:javascript
复制
(gdb) python struct_S = my_s.type # or gdb.lookup_type("struct S"), etc.
(gdb) python print struct_S.keys()
{['a', 'b']}
(gdb) python print my_s['a']
0

通过对来源的快速浏览,可以看到gdb.types.deep_item(type_)是如何实现的,而这似乎就是它的全部。

在gdb7.4之前,不能将类型直接视为dict。也就是说,没有for name in instance_.type:instance_.type.iteritems()等,您必须显式地调用fields。总之,将所有这些放在一起,下面是一个用gdb 7.2迭代结构所有字段的简单示例:

代码语言:javascript
复制
for field in inst.fields:
    fname = field.name
    ftype = field.type
    fval = inst[fname]

但是,如果您的struct中包含一个匿名struct,则这是行不通的。为此,您将需要deep_items (如果7.2中没有这样的代码,则需要查看代码并找出如何自己实现)。

所以,在7.2中并不是很琐碎,但是很简单。而且,如果您想要琐碎,只需升级到7.4。

票数 10
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/14698086

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档