summaryrefslogblamecommitdiff
path: root/src/main-gdb-head.py
blob: a465c973b3d0cd748bdec92f4216229939a63504 (plain) (tree)
1
2
3
4
5
6
7
8
9






                                                                   
                                                       
 

         



















                                                                 





















                                                               
             
                                                                             
 
                                                         
                                                                    






                                      
                       
                    
                         
 
                                                                 
                                                                          
 

                            

                                                             



                                                                        
 






                                  
                                
                                  



                                              
























                                           
 















                                          

                                           




                                                      











                                                  




                                       


                











                                                          

                                                          
                                                                 


                          
                                           
 
                                                                      





                                                                      









                                              
                              

                                                                



                                           
                                             


                                       











                                        
# Work around awkwardness in gdb's python printers:
# 1. In src/main-gdb-head.py, define the printer mechanism.
# 2. In src/*/*.py, define all the printer classes.
# 3. In src/main-gdb-tail.py, reflect to actually add the printers.

# gdb sticks everything in one scope.
# This lets us enumerate what *we* added.
initial_globals = {id(v):v for v in globals().values()}

import re

# copied from gdb/types.py for compatibility with old gdb
def get_basic_type(type_):
    """Return the "basic" type of a type.

    Arguments:
        type_: The type to reduce to its basic type.

    Returns:
        type_ with const/volatile is stripped away,
        and typedefs/references converted to the underlying type.
    """

    while (type_.code == gdb.TYPE_CODE_REF or
           type_.code == gdb.TYPE_CODE_TYPEDEF):
        if type_.code == gdb.TYPE_CODE_REF:
            type_ = type_.target()
        else:
            type_ = type_.strip_typedefs()
    return type_.unqualified()

def info_symbol(addr):
    ''' returns (symbol, offset, section, lib or None) or None?
    '''
    info = gdb.execute('info symbol %d' % addr, to_string=True)
    try:
        sym_and_off, sec_and_lib = info.split(' in section ')
    except ValueError:
        return None
    try:
        sym, off = sym_and_off.split(' + ')
    except ValueError:
        sym = sym_and_off
        off = 0
    else:
        off = int(off, 10)
    try:
        sec, lib = sec_and_lib.split(' of ')
    except ValueError:
        sec = sec_and_lib
        lib = None
    return (sym, off, sec, lib)

def finish():
    global finish, initial_globals, FastPrinters, EnumPrinter, PointerPrinter

    final_globals = {id(v):v for v in globals().values()}
    diff = set(final_globals.keys()) - set(initial_globals.keys()) \
            - {
                    'finish',
                    'initial_globals',
                    'FastPrinters',
                    'EnumPrinter',
                    'PointerPrinter',
            }
    fp = FastPrinters()
    ep = EnumPrinter
    ptrp = PointerPrinter

    # After this, don't access any more globals in this function.
    del finish, initial_globals, FastPrinters, EnumPrinter, PointerPrinter

    for i in diff:
        v = final_globals[i]
        if hasattr(v, 'children') or hasattr(v, 'to_string'):
            fp.add_printer(v)
        # TODO see if there's a way to detect the top-level printers too
        # the problem is that some of them collide and the order *is*
        # important, but sets and dicts don't preserve order.
        # Particularly, 'PointerPrinter' must come after 'FastPrinters'.

    obj = gdb.current_objfile()
    if obj is None:
        obj = gdb
        filename = '<unknown>'
    else:
        filename = obj.filename
    obj.pretty_printers.append(fp)
    n = len(obj.pretty_printers)
    obj.pretty_printers.append(ep)
    obj.pretty_printers.append(ptrp)
    n = len(obj.pretty_printers) - n
    print('Added %d+%d custom printers for %s'
            % (len(fp.printers), n, filename))

class EnumPrinter(object):
    __slots__ = ('_value')
    name = 'enum-class'
    enabled = True

    def __new__(cls, v):
        type = get_basic_type(v.type)
        if type.code != gdb.TYPE_CODE_ENUM:
            return None
        return object.__new__(cls)

    def __init__(self, v):
        self._value = v

    def to_string(self):
        v = self._value
        self.__class__.enabled = False
        try:
            name = str(v)
        finally:
            self.__class__.enabled = True
        name = name.split('::')[-1]
        scope = get_basic_type(v.type).tag
        return '%s::%s' % (scope, name)

class PointerPrinter(object):
    __slots__ = ('_value')
    name = 'any-symbol-pointer'
    enabled = True

    def __new__(cls, v):
        type = get_basic_type(v.type)
        if type.code != gdb.TYPE_CODE_PTR:
            return None
        return object.__new__(cls)

    def __init__(self, v):
        self._value = v

    def to_string(self):
        v = self._value
        uptr = gdb.lookup_type('uintptr_t')
        addr = int(v.cast(uptr))
        if not addr:
            s = 'nullptr'
        else:
            try:
                sym, off, sec, lib = info_symbol(addr)
            except TypeError:
                sp = gdb.parse_and_eval('$sp')
                sp = int(sp.cast(uptr))
                LOTS = 8 * 1024 * 1024
                diff = addr - sp
                if +diff >= 0 and +diff <= LOTS:
                    a = '<$sp+0x%x>' % +diff
                elif -diff >= 0 and -diff <= LOTS:
                    a = '<$sp-0x%x>' % -diff
                else:
                    a = '<heap 0x%x>' % addr
                s = '(%s)%s' % (v.type, a)
            else:
                if off:
                    s = '<%s+%d>' % off
                else:
                    s = '<%s>' % sym
        return s


class FastPrinters(object):
    ''' printer dispatch the way gdb *should* have done it
    '''
    __slots__ = ('name', 'enabled', 'printers')

    def __init__(self):
        self.name = 'tmwa'
        self.enabled = True
        self.printers = {}

    def add_printer(self, cls):
        assert hasattr(cls, 'enabled')
        # TODO: check if the class name exists
        # this is really hard since templates are involved
        self.printers[(cls.name, getattr(cls, 'depth', 0))] = cls

    @property
    def subprinters(self):
        return list(self.printers.values())

    def strip_templates(self, name, __pattern=re.compile('<[^<>]*>')):
        # TODO what about '<' and '>' as non-type template parameters?
        changed = 1
        while changed:
            name, changed = __pattern.subn('', name)
        return name

    def get_tag_and_depth(self, type):
        depth = 0
        while True:
            type = get_basic_type(type)
            if type.code != gdb.TYPE_CODE_PTR:
                break
            type = type.target()
            depth += 1
        return (str(type), depth)

    def __call__(self, value):
        (stype, depth) = self.get_tag_and_depth(value.type)
        #(dtype, _) = self.get_tag_and_depth(value.dynamic_type)
        if stype is None:
            return

        stype = self.strip_templates(stype)
        p = self.printers.get((stype, depth))
        if p is not None and p.enabled:
            return p(value)
        return None

class char(object):
    __slots__ = ('_value')
    name = 'char'
    depth = 1
    enabled = True

    def __init__(self, value):
        self._value = value

    def to_string(self):
        return self._value.lazy_string()