如何将通用链接列表中的数据类型传递给GDB打印命令?

时间:2018-12-03 21:18:40

标签: c linked-list gdb pretty-print gdb-python

我正在使用Python为此链接列表数据结构编写GDB漂亮打印机:

struct listNode {
    void                 *data;         /* node's data                       */
    struct listNode      *next;         /* next node in list                 */
};


struct xlist {
    struct listNode      *head;         /* head of the list                  */
    struct listNode     **tail;         /* addr of last node's 'next' ptr    */
    struct listIterator  *iNext;        /* iterator chain for list_destroy() */
    ListDelF              fDel;         /* function to delete node data      */
    int                   count;        /* number of nodes in list           */
};

typedef struct xlist * List;

数据字段为空*,因为它是通用的,因此根据需要将其强制转换为适当的指针类型。下面是我漂亮的打印机,它可以很好地为某种类型(print_field_t)进行硬编码,但是我正在寻找一种将所需类型作为参数传递给打印机的方法。据我所知,API不支持此功能,但是有没有办法实现此功能或等效功能?

#!/usr/bin/env python3
import gdb.printing
import gdb

class ListPrinter(object):
    '''Print a List object'''

    class _iterator:
        def __init__(self, current):
            self.current = current
            self.count = 0
            self.end = False

        def __iter__(self):
            return self

        def __next__(self):
            if self.end:
                raise StopIteration

            type = gdb.lookup_type('print_field_t').pointer()
            value = self.current['data'].cast(type).dereference()

            self.current = self.current['next']
            self.count += 1
            if self.current == 0:
                self.end = True
            return (str(self.count), value)

    def __init__(self, val):
        self.val = val

    def to_string(self):
        return 'List({}):'.format(self.val['count'])

    def children(self):
        return self._iterator(self.val['head'])

    def display_hint(self):
        return 'array'

def build_pretty_printer():
    pp = gdb.printing.RegexpCollectionPrettyPrinter('List')
    pp.add_printer('List Printer', '^List$', ListPrinter)
    return pp

gdb.printing.register_pretty_printer(gdb.current_objfile(), build_pretty_printer())

0 个答案:

没有答案