python # GDB dashboard - Modular visual interface for GDB in Python. # # https://github.com/cyrus-and/gdb-dashboard # License ---------------------------------------------------------------------- # Copyright (c) 2015-2023 Andrea Cardaci # # Permission is hereby granted, free of charge, to any person obtaining a copy # of this software and associated documentation files (the "Software"), to deal # in the Software without restriction, including without limitation the rights # to use, copy, modify, merge, publish, distribute, sublicense, and/or sell # copies of the Software, and to permit persons to whom the Software is # furnished to do so, subject to the following conditions: # # The above copyright notice and this permission notice shall be included in all # copies or substantial portions of the Software. # # THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR # IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, # FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE # AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER # LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, # OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE # SOFTWARE. # Imports ---------------------------------------------------------------------- import ast import io import itertools import math import os import re import struct import traceback # Common attributes ------------------------------------------------------------ class R(): @staticmethod def attributes(): return { # miscellaneous 'ansi': { 'doc': 'Control the ANSI output of the dashboard.', 'default': True, 'type': bool }, 'syntax_highlighting': { 'doc': '''Pygments style to use for syntax highlighting. Using an empty string (or a name not in the list) disables this feature. The list of all the available styles can be obtained with (from GDB itself): python from pygments.styles import * python for style in get_all_styles(): print(style)''', 'default': 'monokai' }, 'discard_scrollback': { 'doc': '''Discard the scrollback buffer at each redraw. This makes scrolling less confusing by discarding the previously printed dashboards but only works with certain terminals.''', 'default': True, 'type': bool }, # values formatting 'compact_values': { 'doc': 'Display complex objects in a single line.', 'default': True, 'type': bool }, 'max_value_length': { 'doc': 'Maximum length of displayed values before truncation.', 'default': 100, 'type': int }, 'value_truncation_string': { 'doc': 'String to use to mark value truncation.', 'default': '…', }, 'dereference': { 'doc': 'Annotate pointers with the pointed value.', 'default': True, 'type': bool }, # prompt 'prompt': { 'doc': '''GDB prompt. This value is used as a Python format string where `{status}` is expanded with the substitution of either `prompt_running` or `prompt_not_running` attributes, according to the target program status. The resulting string must be a valid GDB prompt, see the command `python print(gdb.prompt.prompt_help())`''', 'default': '{status}' }, 'prompt_running': { 'doc': '''Define the value of `{status}` when the target program is running. See the `prompt` attribute. This value is used as a Python format string where `{pid}` is expanded with the process identifier of the target program.''', 'default': '\[\e[1;35m\]>>>\[\e[0m\]' }, 'prompt_not_running': { 'doc': '''Define the value of `{status}` when the target program is running. See the `prompt` attribute. This value is used as a Python format string.''', 'default': '\[\e[90m\]>>>\[\e[0m\]' }, # divider 'omit_divider': { 'doc': 'Omit the divider in external outputs when only one module is displayed.', 'default': False, 'type': bool }, 'divider_fill_char_primary': { 'doc': 'Filler around the label for primary dividers', 'default': '─' }, 'divider_fill_char_secondary': { 'doc': 'Filler around the label for secondary dividers', 'default': '─' }, 'divider_fill_style_primary': { 'doc': 'Style for `divider_fill_char_primary`', 'default': '36' }, 'divider_fill_style_secondary': { 'doc': 'Style for `divider_fill_char_secondary`', 'default': '90' }, 'divider_label_style_on_primary': { 'doc': 'Label style for non-empty primary dividers', 'default': '1;33' }, 'divider_label_style_on_secondary': { 'doc': 'Label style for non-empty secondary dividers', 'default': '1;37' }, 'divider_label_style_off_primary': { 'doc': 'Label style for empty primary dividers', 'default': '33' }, 'divider_label_style_off_secondary': { 'doc': 'Label style for empty secondary dividers', 'default': '90' }, 'divider_label_skip': { 'doc': 'Gap between the aligning border and the label.', 'default': 3, 'type': int, 'check': check_ge_zero }, 'divider_label_margin': { 'doc': 'Number of spaces around the label.', 'default': 1, 'type': int, 'check': check_ge_zero }, 'divider_label_align_right': { 'doc': 'Label alignment flag.', 'default': False, 'type': bool }, # common styles 'style_selected_1': { 'default': '1;32' }, 'style_selected_2': { 'default': '32' }, 'style_low': { 'default': '90' }, 'style_high': { 'default': '1;37' }, 'style_error': { 'default': '31' }, 'style_critical': { 'default': '0;41' } } # Common ----------------------------------------------------------------------- class Beautifier(): def __init__(self, hint, tab_size=4): self.tab_spaces = ' ' * tab_size if tab_size else None self.active = False if not R.ansi or not R.syntax_highlighting: return # attempt to set up Pygments try: import pygments from pygments.lexers import GasLexer, NasmLexer from pygments.formatters import Terminal256Formatter if hint == 'att': self.lexer = GasLexer() elif hint == 'intel': self.lexer = NasmLexer() else: from pygments.lexers import get_lexer_for_filename self.lexer = get_lexer_for_filename(hint, stripnl=False) self.formatter = Terminal256Formatter(style=R.syntax_highlighting) self.active = True except ImportError: # Pygments not available pass except pygments.util.ClassNotFound: # no lexer for this file or invalid style pass def process(self, source): # convert tabs if requested if self.tab_spaces: source = source.replace('\t', self.tab_spaces) if self.active: import pygments source = pygments.highlight(source, self.lexer, self.formatter) return source.rstrip('\n') def run(command): return gdb.execute(command, to_string=True) def ansi(string, style): if R.ansi: return '\x1b[{}m{}\x1b[0m'.format(style, string) else: return string def divider(width, label='', primary=False, active=True): if primary: divider_fill_style = R.divider_fill_style_primary divider_fill_char = R.divider_fill_char_primary divider_label_style_on = R.divider_label_style_on_primary divider_label_style_off = R.divider_label_style_off_primary else: divider_fill_style = R.divider_fill_style_secondary divider_fill_char = R.divider_fill_char_secondary divider_label_style_on = R.divider_label_style_on_secondary divider_label_style_off = R.divider_label_style_off_secondary if label: if active: divider_label_style = divider_label_style_on else: divider_label_style = divider_label_style_off skip = R.divider_label_skip margin = R.divider_label_margin before = ansi(divider_fill_char * skip, divider_fill_style) middle = ansi(label, divider_label_style) after_length = width - len(label) - skip - 2 * margin after = ansi(divider_fill_char * after_length, divider_fill_style) if R.divider_label_align_right: before, after = after, before return ''.join([before, ' ' * margin, middle, ' ' * margin, after]) else: return ansi(divider_fill_char * width, divider_fill_style) def check_gt_zero(x): return x > 0 def check_ge_zero(x): return x >= 0 def to_unsigned(value, size=8): # values from GDB can be used transparently but are not suitable for # being printed as unsigned integers, so a conversion is needed mask = (2 ** (size * 8)) - 1 return int(value.cast(gdb.Value(mask).type)) & mask def to_string(value): # attempt to convert an inferior value to string; OK when (Python 3 || # simple ASCII); otherwise (Python 2.7 && not ASCII) encode the string as # utf8 try: value_string = str(value) except UnicodeEncodeError: value_string = unicode(value).encode('utf8') except gdb.error as e: value_string = ansi(e, R.style_error) return value_string def format_address(address): pointer_size = gdb.parse_and_eval('$pc').type.sizeof return ('0x{{:0{}x}}').format(pointer_size * 2).format(address) def format_value(value, compact=None): # format references as referenced values # (TYPE_CODE_RVALUE_REF is not supported by old GDB) if value.type.code in (getattr(gdb, 'TYPE_CODE_REF', None), getattr(gdb, 'TYPE_CODE_RVALUE_REF', None)): try: value = value.referenced_value() except gdb.error as e: return ansi(e, R.style_error) # format the value out = to_string(value) # dereference up to the actual value if requested if R.dereference and value.type.code == gdb.TYPE_CODE_PTR: while value.type.code == gdb.TYPE_CODE_PTR: try: value = value.dereference() except gdb.error as e: break else: formatted = to_string(value) out += '{} {}'.format(ansi(':', R.style_low), formatted) # compact the value if compact is not None and compact or R.compact_values: out = re.sub(r'$\s*', '', out, flags=re.MULTILINE) # truncate the value if R.max_value_length > 0 and len(out) > R.max_value_length: out = out[0:R.max_value_length] + ansi(R.value_truncation_string, R.style_critical) return out # XXX parsing the output of `info breakpoints` is apparently the best option # right now, see: https://sourceware.org/bugzilla/show_bug.cgi?id=18385 # XXX GDB version 7.11 (quire recent) does not have the pending field, so # fall back to the parsed information def fetch_breakpoints(watchpoints=False, pending=False): # fetch breakpoints addresses parsed_breakpoints = dict() catch_what_regex = re.compile(r'([^,]+".*")?[^,]*') for line in run('info breakpoints').split('\n'): # just keep numbered lines if not line or not line[0].isdigit(): continue # extract breakpoint number, address and pending status fields = line.split() number = int(fields[0].split('.')[0]) try: if len(fields) >= 5 and fields[1] == 'breakpoint': # multiple breakpoints have no address yet is_pending = fields[4] == '' is_multiple = fields[4] == '' address = None if is_multiple or is_pending else int(fields[4], 16) is_enabled = fields[3] == 'y' address_info = address, is_enabled parsed_breakpoints[number] = [address_info], is_pending, '' elif len(fields) >= 5 and fields[1] == 'catchpoint': # only take before comma, but ignore commas in quotes what = catch_what_regex.search(' '.join(fields[4:]))[0].strip() parsed_breakpoints[number] = [], False, what elif len(fields) >= 3 and number in parsed_breakpoints: # add this address to the list of multiple locations address = int(fields[2], 16) is_enabled = fields[1] == 'y' address_info = address, is_enabled parsed_breakpoints[number][0].append(address_info) else: # watchpoints parsed_breakpoints[number] = [], False, '' except ValueError: pass # fetch breakpoints from the API and complement with address and source # information breakpoints = [] # XXX in older versions gdb.breakpoints() returns None for gdb_breakpoint in gdb.breakpoints() or []: # skip internal breakpoints if gdb_breakpoint.number < 0: continue addresses, is_pending, what = parsed_breakpoints[gdb_breakpoint.number] is_pending = getattr(gdb_breakpoint, 'pending', is_pending) if not pending and is_pending: continue if not watchpoints and gdb_breakpoint.type != gdb.BP_BREAKPOINT: continue # add useful fields to the object breakpoint = dict() breakpoint['number'] = gdb_breakpoint.number breakpoint['type'] = gdb_breakpoint.type breakpoint['enabled'] = gdb_breakpoint.enabled breakpoint['location'] = gdb_breakpoint.location breakpoint['expression'] = gdb_breakpoint.expression breakpoint['condition'] = gdb_breakpoint.condition breakpoint['temporary'] = gdb_breakpoint.temporary breakpoint['hit_count'] = gdb_breakpoint.hit_count breakpoint['pending'] = is_pending breakpoint['what'] = what # add addresses and source information breakpoint['addresses'] = [] for address, is_enabled in addresses: if address: sal = gdb.find_pc_line(address) breakpoint['addresses'].append({ 'address': address, 'enabled': is_enabled, 'file_name': sal.symtab.filename if address and sal.symtab else None, 'file_line': sal.line if address else None }) breakpoints.append(breakpoint) return breakpoints # Dashboard -------------------------------------------------------------------- class Dashboard(gdb.Command): '''Redisplay the dashboard.''' def __init__(self): gdb.Command.__init__(self, 'dashboard', gdb.COMMAND_USER, gdb.COMPLETE_NONE, True) # setup subcommands Dashboard.ConfigurationCommand(self) Dashboard.OutputCommand(self) Dashboard.EnabledCommand(self) Dashboard.LayoutCommand(self) # setup style commands Dashboard.StyleCommand(self, 'dashboard', R, R.attributes()) # main terminal self.output = None # used to inhibit redisplays during init parsing self.inhibited = None # enabled by default self.enabled = None self.enable() def on_continue(self, _): # try to contain the GDB messages in a specified area unless the # dashboard is printed to a separate file (dashboard -output ...) # or there are no modules to display in the main terminal enabled_modules = list(filter(lambda m: not m.output and m.enabled, self.modules)) if self.is_running() and not self.output and len(enabled_modules) > 0: width, _ = Dashboard.get_term_size() gdb.write(Dashboard.clear_screen()) gdb.write(divider(width, 'Output/messages', True)) gdb.write('\n') gdb.flush() def on_stop(self, _): if self.is_running(): self.render(clear_screen=False) def on_exit(self, _): if not self.is_running(): return # collect all the outputs outputs = set() outputs.add(self.output) outputs.update(module.output for module in self.modules) outputs.remove(None) # reset the terminal status for output in outputs: try: with open(output, 'w') as fs: fs.write(Dashboard.reset_terminal()) except: # skip cleanup for invalid outputs pass def enable(self): if self.enabled: return self.enabled = True # setup events gdb.events.cont.connect(self.on_continue) gdb.events.stop.connect(self.on_stop) gdb.events.exited.connect(self.on_exit) def disable(self): if not self.enabled: return self.enabled = False # setup events gdb.events.cont.disconnect(self.on_continue) gdb.events.stop.disconnect(self.on_stop) gdb.events.exited.disconnect(self.on_exit) def load_modules(self, modules): self.modules = [] for module in modules: info = Dashboard.ModuleInfo(self, module) self.modules.append(info) def redisplay(self, style_changed=False): # manually redisplay the dashboard if self.is_running() and not self.inhibited: self.render(True, style_changed) def inferior_pid(self): return gdb.selected_inferior().pid def is_running(self): return self.inferior_pid() != 0 def render(self, clear_screen, style_changed=False): # fetch module content and info all_disabled = True display_map = dict() for module in self.modules: # fall back to the global value output = module.output or self.output # add the instance or None if disabled if module.enabled: all_disabled = False instance = module.instance else: instance = None display_map.setdefault(output, []).append(instance) # process each display info for output, instances in display_map.items(): try: buf = '' # use GDB stream by default fs = None if output: fs = open(output, 'w') fd = fs.fileno() fs.write(Dashboard.setup_terminal()) else: fs = gdb fd = 1 # stdout # get the terminal size (default main terminal if either the # output is not a file) try: width, height = Dashboard.get_term_size(fd) except: width, height = Dashboard.get_term_size() # clear the "screen" if requested for the main terminal, # auxiliary terminals are always cleared if fs is not gdb or clear_screen: buf += Dashboard.clear_screen() # show message if all the modules in this output are disabled if not any(instances): # skip the main terminal if fs is gdb: continue # write the error message buf += divider(width, 'Warning', True) buf += '\n' if self.modules: buf += 'No module to display (see `dashboard -layout`)' else: buf += 'No module loaded' buf += '\n' fs.write(buf) continue # process all the modules for that output for n, instance in enumerate(instances, 1): # skip disabled modules if not instance: continue try: # ask the module to generate the content lines = instance.lines(width, height, style_changed) except Exception as e: # allow to continue on exceptions in modules stacktrace = traceback.format_exc().strip() lines = [ansi(stacktrace, R.style_error)] # create the divider if needed div = [] if not R.omit_divider or len(instances) > 1 or fs is gdb: div = [divider(width, instance.label(), True, lines)] # write the data buf += '\n'.join(div + lines) # write the newline for all but last unless main terminal if n != len(instances) or fs is gdb: buf += '\n' # write the final newline and the terminator only if it is the # main terminal to allow the prompt to display correctly (unless # there are no modules to display) if fs is gdb and not all_disabled: buf += divider(width, primary=True) buf += '\n' fs.write(buf) except Exception as e: cause = traceback.format_exc().strip() Dashboard.err('Cannot write the dashboard\n{}'.format(cause)) finally: # don't close gdb stream if fs and fs is not gdb: fs.close() # Utility methods -------------------------------------------------------------- @staticmethod def start(): # save the instance for customization convenience global dashboard # initialize the dashboard dashboard = Dashboard() Dashboard.set_custom_prompt(dashboard) # parse Python inits, load modules then parse GDB inits dashboard.inhibited = True Dashboard.parse_inits(True) modules = Dashboard.get_modules() dashboard.load_modules(modules) Dashboard.parse_inits(False) dashboard.inhibited = False # GDB overrides run('set pagination off') # display if possible (program running and not explicitly disabled by # some configuration file) if dashboard.enabled: dashboard.redisplay() @staticmethod def get_term_size(fd=1): # defaults to the main terminal try: if sys.platform == 'win32': import curses # XXX always neglects the fd parameter height, width = curses.initscr().getmaxyx() curses.endwin() return int(width), int(height) else: import termios import fcntl # first 2 shorts (4 byte) of struct winsize raw = fcntl.ioctl(fd, termios.TIOCGWINSZ, ' ' * 4) height, width = struct.unpack('hh', raw) return int(width), int(height) except (ImportError, OSError): # this happens when no curses library is found on windows or when # the terminal is not properly configured return 80, 24 # hardcoded fallback value @staticmethod def set_custom_prompt(dashboard): def custom_prompt(_): # render thread status indicator if dashboard.is_running(): pid = dashboard.inferior_pid() status = R.prompt_running.format(pid=pid) else: status = R.prompt_not_running # build prompt prompt = R.prompt.format(status=status) prompt = gdb.prompt.substitute_prompt(prompt) return prompt + ' ' # force trailing space gdb.prompt_hook = custom_prompt @staticmethod def parse_inits(python): # paths where the .gdbinit.d directory might be search_paths = [ '/etc/gdb-dashboard', '{}/gdb-dashboard'.format(os.getenv('XDG_CONFIG_HOME', '~/.config')), '~/Library/Preferences/gdb-dashboard', '~/.gdbinit.d' ] # expand the tilde and walk the paths inits_dirs = (os.walk(os.path.expanduser(path)) for path in search_paths) # process all the init files in order for root, dirs, files in itertools.chain.from_iterable(inits_dirs): dirs.sort() # skipping dotfiles for init in sorted(file for file in files if not file.startswith('.')): path = os.path.join(root, init) _, ext = os.path.splitext(path) # either load Python files or GDB if python == (ext == '.py'): gdb.execute('source ' + path) @staticmethod def get_modules(): # scan the scope for modules modules = [] for name in globals(): obj = globals()[name] try: if issubclass(obj, Dashboard.Module): modules.append(obj) except TypeError: continue # sort modules alphabetically modules.sort(key=lambda x: x.__name__) return modules @staticmethod def create_command(name, invoke, doc, is_prefix, complete=None): if callable(complete): Class = type('', (gdb.Command,), { '__doc__': doc, 'invoke': invoke, 'complete': complete }) Class(name, gdb.COMMAND_USER, prefix=is_prefix) else: Class = type('', (gdb.Command,), { '__doc__': doc, 'invoke': invoke }) Class(name, gdb.COMMAND_USER, complete or gdb.COMPLETE_NONE, is_prefix) @staticmethod def err(string): print(ansi(string, R.style_error)) @staticmethod def complete(word, candidates): return filter(lambda candidate: candidate.startswith(word), candidates) @staticmethod def parse_arg(arg): # encode unicode GDB command arguments as utf8 in Python 2.7 if type(arg) is not str: arg = arg.encode('utf8') return arg @staticmethod def clear_screen(): # ANSI: move the cursor to top-left corner and clear the screen # (optionally also clear the scrollback buffer if supported by the # terminal) return '\x1b[H\x1b[2J' + ('\x1b[3J' if R.discard_scrollback else '') @staticmethod def setup_terminal(): # ANSI: enable alternative screen buffer and hide cursor return '\x1b[?1049h\x1b[?25l' @staticmethod def reset_terminal(): # ANSI: disable alternative screen buffer and show cursor return '\x1b[?1049l\x1b[?25h' # Module descriptor ------------------------------------------------------------ class ModuleInfo: def __init__(self, dashboard, module): self.name = module.__name__.lower() # from class to module name self.enabled = True self.output = None # value from the dashboard by default self.instance = module() self.doc = self.instance.__doc__ or '(no documentation)' self.prefix = 'dashboard {}'.format(self.name) # add GDB commands self.add_main_command(dashboard) self.add_output_command(dashboard) self.add_style_command(dashboard) self.add_subcommands(dashboard) def add_main_command(self, dashboard): module = self def invoke(self, arg, from_tty, info=self): arg = Dashboard.parse_arg(arg) if arg == '': info.enabled ^= True if dashboard.is_running(): dashboard.redisplay() else: status = 'enabled' if info.enabled else 'disabled' print('{} module {}'.format(module.name, status)) else: Dashboard.err('Wrong argument "{}"'.format(arg)) doc_brief = 'Configure the {} module, with no arguments toggles its visibility.'.format(self.name) doc = '{}\n\n{}'.format(doc_brief, self.doc) Dashboard.create_command(self.prefix, invoke, doc, True) def add_output_command(self, dashboard): Dashboard.OutputCommand(dashboard, self.prefix, self) def add_style_command(self, dashboard): Dashboard.StyleCommand(dashboard, self.prefix, self.instance, self.instance.attributes()) def add_subcommands(self, dashboard): for name, command in self.instance.commands().items(): self.add_subcommand(dashboard, name, command) def add_subcommand(self, dashboard, name, command): action = command['action'] doc = command['doc'] complete = command.get('complete') def invoke(self, arg, from_tty, info=self): arg = Dashboard.parse_arg(arg) if info.enabled: try: action(arg) except Exception as e: Dashboard.err(e) return # don't catch redisplay errors dashboard.redisplay() else: Dashboard.err('Module disabled') prefix = '{} {}'.format(self.prefix, name) Dashboard.create_command(prefix, invoke, doc, False, complete) # GDB commands ----------------------------------------------------------------- # handler for the `dashboard` command itself def invoke(self, arg, from_tty): arg = Dashboard.parse_arg(arg) # show messages for checks in redisplay if arg != '': Dashboard.err('Wrong argument "{}"'.format(arg)) elif not self.is_running(): Dashboard.err('Is the target program running?') else: self.redisplay() class ConfigurationCommand(gdb.Command): '''Dump or save the dashboard configuration. With an optional argument the configuration will be written to the specified file. This command allows to configure the dashboard live then make the changes permanent, for example: dashboard -configuration ~/.gdbinit.d/init At startup the `~/.gdbinit.d/` directory tree is walked and files are evaluated in alphabetical order but giving priority to Python files. This is where user configuration files must be placed.''' def __init__(self, dashboard): gdb.Command.__init__(self, 'dashboard -configuration', gdb.COMMAND_USER, gdb.COMPLETE_FILENAME) self.dashboard = dashboard def invoke(self, arg, from_tty): arg = Dashboard.parse_arg(arg) if arg: with open(os.path.expanduser(arg), 'w') as fs: fs.write('# auto generated by GDB dashboard\n\n') self.dump(fs) self.dump(gdb) def dump(self, fs): # dump layout self.dump_layout(fs) # dump styles self.dump_style(fs, R) for module in self.dashboard.modules: self.dump_style(fs, module.instance, module.prefix) # dump outputs self.dump_output(fs, self.dashboard) for module in self.dashboard.modules: self.dump_output(fs, module, module.prefix) def dump_layout(self, fs): layout = ['dashboard -layout'] for module in self.dashboard.modules: mark = '' if module.enabled else '!' layout.append('{}{}'.format(mark, module.name)) fs.write(' '.join(layout)) fs.write('\n') def dump_style(self, fs, obj, prefix='dashboard'): attributes = getattr(obj, 'attributes', lambda: dict())() for name, attribute in attributes.items(): real_name = attribute.get('name', name) default = attribute.get('default') value = getattr(obj, real_name) if value != default: fs.write('{} -style {} {!r}\n'.format(prefix, name, value)) def dump_output(self, fs, obj, prefix='dashboard'): output = getattr(obj, 'output') if output: fs.write('{} -output {}\n'.format(prefix, output)) class OutputCommand(gdb.Command): '''Set the output file/TTY for the whole dashboard or single modules. The dashboard/module will be written to the specified file, which will be created if it does not exist. If the specified file identifies a terminal then its geometry will be used, otherwise it falls back to the geometry of the main GDB terminal. When invoked without argument on the dashboard, the output/messages and modules which do not specify an output themselves will be printed on standard output (default). When invoked without argument on a module, it will be printed where the dashboard will be printed. An overview of all the outputs can be obtained with the `dashboard -layout` command.''' def __init__(self, dashboard, prefix=None, obj=None): if not prefix: prefix = 'dashboard' if not obj: obj = dashboard prefix = prefix + ' -output' gdb.Command.__init__(self, prefix, gdb.COMMAND_USER, gdb.COMPLETE_FILENAME) self.dashboard = dashboard self.obj = obj # None means the dashboard itself def invoke(self, arg, from_tty): arg = Dashboard.parse_arg(arg) # reset the terminal status if self.obj.output: try: with open(self.obj.output, 'w') as fs: fs.write(Dashboard.reset_terminal()) except: # just do nothing if the file is not writable pass # set or open the output file if arg == '': self.obj.output = None else: self.obj.output = arg # redisplay the dashboard in the new output self.dashboard.redisplay() class EnabledCommand(gdb.Command): '''Enable or disable the dashboard. The current status is printed if no argument is present.''' def __init__(self, dashboard): gdb.Command.__init__(self, 'dashboard -enabled', gdb.COMMAND_USER) self.dashboard = dashboard def invoke(self, arg, from_tty): arg = Dashboard.parse_arg(arg) if arg == '': status = 'enabled' if self.dashboard.enabled else 'disabled' print('The dashboard is {}'.format(status)) elif arg == 'on': self.dashboard.enable() self.dashboard.redisplay() elif arg == 'off': self.dashboard.disable() else: msg = 'Wrong argument "{}"; expecting "on" or "off"' Dashboard.err(msg.format(arg)) def complete(self, text, word): return Dashboard.complete(word, ['on', 'off']) class LayoutCommand(gdb.Command): '''Set or show the dashboard layout. Accepts a space-separated list of directive. Each directive is in the form "[!]". Modules in the list are placed in the dashboard in the same order as they appear and those prefixed by "!" are disabled by default. Omitted modules are hidden and placed at the bottom in alphabetical order. Without arguments the current layout is shown where the first line uses the same form expected by the input while the remaining depict the current status of output files. Passing `!` as a single argument resets the dashboard original layout.''' def __init__(self, dashboard): gdb.Command.__init__(self, 'dashboard -layout', gdb.COMMAND_USER) self.dashboard = dashboard def invoke(self, arg, from_tty): arg = Dashboard.parse_arg(arg) directives = str(arg).split() if directives: # apply the layout if directives == ['!']: self.reset() else: if not self.layout(directives): return # in case of errors # redisplay or otherwise notify if from_tty: if self.dashboard.is_running(): self.dashboard.redisplay() else: self.show() else: self.show() def reset(self): modules = self.dashboard.modules modules.sort(key=lambda module: module.name) for module in modules: module.enabled = True def show(self): global_str = 'Dashboard' default = '(default TTY)' max_name_len = max(len(module.name) for module in self.dashboard.modules) max_name_len = max(max_name_len, len(global_str)) fmt = '{{}}{{:{}s}}{{}}'.format(max_name_len + 2) print((fmt + '\n').format(' ', global_str, self.dashboard.output or default)) for module in self.dashboard.modules: mark = ' ' if module.enabled else '!' style = R.style_high if module.enabled else R.style_low line = fmt.format(mark, module.name, module.output or default) print(ansi(line, style)) def layout(self, directives): modules = self.dashboard.modules # parse and check directives parsed_directives = [] selected_modules = set() for directive in directives: enabled = (directive[0] != '!') name = directive[not enabled:] if name in selected_modules: Dashboard.err('Module "{}" already set'.format(name)) return False if next((False for module in modules if module.name == name), True): Dashboard.err('Cannot find module "{}"'.format(name)) return False parsed_directives.append((name, enabled)) selected_modules.add(name) # reset visibility for module in modules: module.enabled = False # move and enable the selected modules on top last = 0 for name, enabled in parsed_directives: todo = enumerate(modules[last:], start=last) index = next(index for index, module in todo if name == module.name) modules[index].enabled = enabled modules.insert(last, modules.pop(index)) last += 1 return True def complete(self, text, word): all_modules = (m.name for m in self.dashboard.modules) return Dashboard.complete(word, all_modules) class StyleCommand(gdb.Command): '''Access the stylable attributes. Without arguments print all the stylable attributes. When only the name is specified show the current value. With name and value set the stylable attribute. Values are parsed as Python literals and converted to the proper type. ''' def __init__(self, dashboard, prefix, obj, attributes): self.prefix = prefix + ' -style' gdb.Command.__init__(self, self.prefix, gdb.COMMAND_USER, gdb.COMPLETE_NONE, True) self.dashboard = dashboard self.obj = obj self.attributes = attributes self.add_styles() def add_styles(self): this = self for name, attribute in self.attributes.items(): # fetch fields attr_name = attribute.get('name', name) attr_type = attribute.get('type', str) attr_check = attribute.get('check', lambda _: True) attr_default = attribute['default'] # set the default value (coerced to the type) value = attr_type(attr_default) setattr(self.obj, attr_name, value) # create the command def invoke(self, arg, from_tty, name=name, attr_name=attr_name, attr_type=attr_type, attr_check=attr_check): new_value = Dashboard.parse_arg(arg) if new_value == '': # print the current value value = getattr(this.obj, attr_name) print('{} = {!r}'.format(name, value)) else: try: # convert and check the new value parsed = ast.literal_eval(new_value) value = attr_type(parsed) if not attr_check(value): msg = 'Invalid value "{}" for "{}"' raise Exception(msg.format(new_value, name)) except Exception as e: Dashboard.err(e) else: # set and redisplay setattr(this.obj, attr_name, value) this.dashboard.redisplay(True) prefix = self.prefix + ' ' + name doc = attribute.get('doc', 'This style is self-documenting') Dashboard.create_command(prefix, invoke, doc, False) def invoke(self, arg, from_tty): # an argument here means that the provided attribute is invalid if arg: Dashboard.err('Invalid argument "{}"'.format(arg)) return # print all the pairs for name, attribute in self.attributes.items(): attr_name = attribute.get('name', name) value = getattr(self.obj, attr_name) print('{} = {!r}'.format(name, value)) # Base module ------------------------------------------------------------------ # just a tag class Module(): '''Base class for GDB dashboard modules. Modules are instantiated once at initialization time and kept during the whole the GDB session. The name of a module is automatically obtained by the class name. Optionally, a module may include a description which will appear in the GDB help system by specifying a Python docstring for the class. By convention the first line should contain a brief description.''' def label(self): '''Return the module label which will appear in the divider.''' pass def lines(self, term_width, term_height, style_changed): '''Return a list of strings which will form the module content. When a module is temporarily unable to produce its content, it should return an empty list; its divider will then use the styles with the "off" qualifier. term_width and term_height are the dimension of the terminal where this module will be displayed. If `style_changed` is `True` then some attributes have changed since the last time so the implementation may want to update its status.''' pass def attributes(self): '''Return the dictionary of available attributes. The key is the attribute name and the value is another dictionary with items: - `default` is the initial value for this attribute; - `doc` is the optional documentation of this attribute which will appear in the GDB help system; - `name` is the name of the attribute of the Python object (defaults to the key value); - `type` is the Python type of this attribute defaulting to the `str` type, it is used to coerce the value passed as an argument to the proper type, or raise an exception; - `check` is an optional control callback which accept the coerced value and returns `True` if the value satisfies the constraint and `False` otherwise. Those attributes can be accessed from the implementation using instance variables named `name`.''' return {} def commands(self): '''Return the dictionary of available commands. The key is the attribute name and the value is another dictionary with items: - `action` is the callback to be executed which accepts the raw input string from the GDB prompt, exceptions in these functions will be shown automatically to the user; - `doc` is the documentation of this command which will appear in the GDB help system; - `completion` is the optional completion policy, one of the `gdb.COMPLETE_*` constants defined in the GDB reference manual (https://sourceware.org/gdb/onlinedocs/gdb/Commands-In-Python.html).''' return {} # Default modules -------------------------------------------------------------- class Source(Dashboard.Module): '''Show the program source code, if available.''' def __init__(self): self.file_name = None self.source_lines = [] self.ts = None self.highlighted = False self.offset = 0 def label(self): label = 'Source' if self.show_path and self.file_name: label += ': {}'.format(self.file_name) return label def lines(self, term_width, term_height, style_changed): # skip if the current thread is not stopped if not gdb.selected_thread().is_stopped(): return [] # try to fetch the current line (skip if no line information) sal = gdb.selected_frame().find_sal() current_line = sal.line if current_line == 0: self.file_name = None return [] # try to lookup the source file candidates = [ sal.symtab.fullname(), sal.symtab.filename, # XXX GDB also uses absolute filename but it is harder to implement # properly and IMHO useless os.path.basename(sal.symtab.filename)] for candidate in candidates: file_name = candidate ts = None try: ts = os.path.getmtime(file_name) break except: # try another or delay error check to open() continue # style changed, different file name or file modified in the meanwhile if style_changed or file_name != self.file_name or ts and ts > self.ts: try: # reload the source file if changed with io.open(file_name, errors='replace') as source_file: highlighter = Beautifier(file_name, self.tab_size) self.highlighted = highlighter.active source = highlighter.process(source_file.read()) self.source_lines = source.split('\n') # store file name and timestamp only if success to have # persistent errors self.file_name = file_name self.ts = ts except IOError as e: msg = 'Cannot display "{}"'.format(file_name) return [ansi(msg, R.style_error)] # compute the line range height = self.height or (term_height - 1) start = current_line - 1 - int(height / 2) + self.offset end = start + height # extra at start extra_start = 0 if start < 0: extra_start = min(-start, height) start = 0 # extra at end extra_end = 0 if end > len(self.source_lines): extra_end = min(end - len(self.source_lines), height) end = len(self.source_lines) else: end = max(end, 0) # return the source code listing breakpoints = fetch_breakpoints() out = [] number_format = '{{:>{}}}'.format(len(str(end))) for number, line in enumerate(self.source_lines[start:end], start + 1): # properly handle UTF-8 source files line = to_string(line) if int(number) == current_line: # the current line has a different style without ANSI if R.ansi: if self.highlighted and not self.highlight_line: line_format = '{}' + ansi(number_format, R.style_selected_1) + ' {}' else: line_format = '{}' + ansi(number_format + ' {}', R.style_selected_1) else: # just show a plain text indicator line_format = '{}' + number_format + '> {}' else: line_format = '{}' + ansi(number_format, R.style_low) + ' {}' # check for breakpoint presence enabled = None for breakpoint in breakpoints: addresses = breakpoint['addresses'] is_root_enabled = addresses[0]['enabled'] for address in addresses: # note, despite the lookup path always use the relative # (sal.symtab.filename) file name to match source files with # breakpoints if address['file_line'] == number and address['file_name'] == sal.symtab.filename: enabled = enabled or (address['enabled'] and is_root_enabled) if enabled is None: breakpoint = ' ' else: breakpoint = ansi('!', R.style_critical) if enabled else ansi('-', R.style_low) out.append(line_format.format(breakpoint, number, line.rstrip('\n'))) # return the output along with scroll indicators if len(out) <= height: extra = [ansi('~', R.style_low)] return extra_start * extra + out + extra_end * extra else: return out def commands(self): return { 'scroll': { 'action': self.scroll, 'doc': 'Scroll by relative steps or reset if invoked without argument.' } } def attributes(self): return { 'height': { 'doc': '''Height of the module. A value of 0 uses the whole height.''', 'default': 10, 'type': int, 'check': check_ge_zero }, 'tab-size': { 'doc': 'Number of spaces used to display the tab character.', 'default': 4, 'name': 'tab_size', 'type': int, 'check': check_gt_zero }, 'path': { 'doc': 'Path visibility flag in the module label.', 'default': False, 'name': 'show_path', 'type': bool }, 'highlight-line': { 'doc': 'Decide whether the whole current line should be highlighted.', 'default': False, 'name': 'highlight_line', 'type': bool } } def scroll(self, arg): if arg: self.offset += int(arg) else: self.offset = 0 class Assembly(Dashboard.Module): '''Show the disassembled code surrounding the program counter. The instructions constituting the current statement are marked, if available.''' def __init__(self): self.offset = 0 self.cache_key = None self.cache_asm = None def label(self): return 'Assembly' def lines(self, term_width, term_height, style_changed): # skip if the current thread is not stopped if not gdb.selected_thread().is_stopped(): return [] # flush the cache if the style is changed if style_changed: self.cache_key = None # prepare the highlighter try: flavor = gdb.parameter('disassembly-flavor') except: flavor = 'att' # not always defined (see #36) highlighter = Beautifier(flavor, tab_size=None) # fetch the assembly code line_info = None frame = gdb.selected_frame() # PC is here height = self.height or (term_height - 1) try: # disassemble the current block asm_start, asm_end = self.fetch_function_boundaries() asm = self.fetch_asm(asm_start, asm_end, False, highlighter) # find the location of the PC pc_index = next(index for index, instr in enumerate(asm) if instr['addr'] == frame.pc()) # compute the instruction range start = pc_index - int(height / 2) + self.offset end = start + height # extra at start extra_start = 0 if start < 0: extra_start = min(-start, height) start = 0 # extra at end extra_end = 0 if end > len(asm): extra_end = min(end - len(asm), height) end = len(asm) else: end = max(end, 0) # fetch actual interval asm = asm[start:end] # if there are line information then use it, it may be that # line_info is not None but line_info.last is None line_info = gdb.find_pc_line(frame.pc()) line_info = line_info if line_info.last else None except (gdb.error, RuntimeError, StopIteration): # if it is not possible (stripped binary or the PC is not present in # the output of `disassemble` as per issue #31) start from PC try: extra_start = 0 extra_end = 0 # allow to scroll down nevertheless clamped_offset = min(self.offset, 0) asm = self.fetch_asm(frame.pc(), height - clamped_offset, True, highlighter) asm = asm[-clamped_offset:] except gdb.error as e: msg = '{}'.format(e) return [ansi(msg, R.style_error)] # fetch function start if available (e.g., not with @plt) func_start = None if self.show_function and frame.function(): func_start = to_unsigned(frame.function().value()) # compute the maximum offset size if asm and func_start: max_offset = max(len(str(abs(asm[0]['addr'] - func_start))), len(str(abs(asm[-1]['addr'] - func_start)))) # return the machine code breakpoints = fetch_breakpoints() max_length = max(instr['length'] for instr in asm) if asm else 0 inferior = gdb.selected_inferior() out = [] for index, instr in enumerate(asm): addr = instr['addr'] length = instr['length'] text = instr['asm'] addr_str = format_address(addr) if self.show_opcodes: # fetch and format opcode region = inferior.read_memory(addr, length) opcodes = (' '.join('{:02x}'.format(ord(byte)) for byte in region)) opcodes += (max_length - len(region)) * 3 * ' ' + ' ' else: opcodes = '' # compute the offset if available if self.show_function: if func_start: offset = '{:+d}'.format(addr - func_start) offset = offset.ljust(max_offset + 1) # sign func_info = '{}{}'.format(frame.function(), offset) else: func_info = '?' else: func_info = '' format_string = '{}{}{}{}{}{}' indicator = ' ' text = ' ' + text if addr == frame.pc(): if not R.ansi: indicator = '> ' addr_str = ansi(addr_str, R.style_selected_1) indicator = ansi(indicator, R.style_selected_1) opcodes = ansi(opcodes, R.style_selected_1) func_info = ansi(func_info, R.style_selected_1) if not highlighter.active or self.highlight_line: text = ansi(text, R.style_selected_1) elif line_info and line_info.pc <= addr < line_info.last: if not R.ansi: indicator = ': ' addr_str = ansi(addr_str, R.style_selected_2) indicator = ansi(indicator, R.style_selected_2) opcodes = ansi(opcodes, R.style_selected_2) func_info = ansi(func_info, R.style_selected_2) if not highlighter.active or self.highlight_line: text = ansi(text, R.style_selected_2) else: addr_str = ansi(addr_str, R.style_low) func_info = ansi(func_info, R.style_low) # check for breakpoint presence enabled = None for breakpoint in breakpoints: addresses = breakpoint['addresses'] is_root_enabled = addresses[0]['enabled'] for address in addresses: if address['address'] == addr: enabled = enabled or (address['enabled'] and is_root_enabled) if enabled is None: breakpoint = ' ' else: breakpoint = ansi('!', R.style_critical) if enabled else ansi('-', R.style_low) out.append(format_string.format(breakpoint, addr_str, indicator, opcodes, func_info, text)) # return the output along with scroll indicators if len(out) <= height: extra = [ansi('~', R.style_low)] return extra_start * extra + out + extra_end * extra else: return out def commands(self): return { 'scroll': { 'action': self.scroll, 'doc': 'Scroll by relative steps or reset if invoked without argument.' } } def attributes(self): return { 'height': { 'doc': '''Height of the module. A value of 0 uses the whole height.''', 'default': 10, 'type': int, 'check': check_ge_zero }, 'opcodes': { 'doc': 'Opcodes visibility flag.', 'default': False, 'name': 'show_opcodes', 'type': bool }, 'function': { 'doc': 'Function information visibility flag.', 'default': True, 'name': 'show_function', 'type': bool }, 'highlight-line': { 'doc': 'Decide whether the whole current line should be highlighted.', 'default': False, 'name': 'highlight_line', 'type': bool } } def scroll(self, arg): if arg: self.offset += int(arg) else: self.offset = 0 def fetch_function_boundaries(self): frame = gdb.selected_frame() # parse the output of the disassemble GDB command to find the function # boundaries, this should handle cases in which a function spans # multiple discontinuous blocks disassemble = run('disassemble') for block_start, block_end in re.findall(r'Address range 0x([0-9a-f]+) to 0x([0-9a-f]+):', disassemble): block_start = int(block_start, 16) block_end = int(block_end, 16) if block_start <= frame.pc() < block_end: return block_start, block_end - 1 # need to be inclusive # if function information is available then try to obtain the # boundaries by looking at the superblocks block = frame.block() if frame.function(): while block and (not block.function or block.function.name != frame.function().name): block = block.superblock block = block or frame.block() return block.start, block.end - 1 def fetch_asm(self, start, end_or_count, relative, highlighter): # fetch asm from cache or disassemble if self.cache_key == (start, end_or_count): asm = self.cache_asm else: kwargs = { 'start_pc': start, 'count' if relative else 'end_pc': end_or_count } asm = gdb.selected_frame().architecture().disassemble(**kwargs) self.cache_key = (start, end_or_count) self.cache_asm = asm # syntax highlight the cached entry for instr in asm: instr['asm'] = highlighter.process(instr['asm']) return asm class Variables(Dashboard.Module): '''Show arguments and locals of the selected frame.''' def label(self): return 'Variables' def lines(self, term_width, term_height, style_changed): return Variables.format_frame( gdb.selected_frame(), self.show_arguments, self.show_locals, self.compact, self.align, self.sort) def attributes(self): return { 'arguments': { 'doc': 'Frame arguments visibility flag.', 'default': True, 'name': 'show_arguments', 'type': bool }, 'locals': { 'doc': 'Frame locals visibility flag.', 'default': True, 'name': 'show_locals', 'type': bool }, 'compact': { 'doc': 'Single-line display flag.', 'default': True, 'type': bool }, 'align': { 'doc': 'Align variables in column flag (only if not compact).', 'default': False, 'type': bool }, 'sort': { 'doc': 'Sort variables by name.', 'default': False, 'type': bool } } @staticmethod def format_frame(frame, show_arguments, show_locals, compact, align, sort): out = [] # fetch frame arguments and locals decorator = gdb.FrameDecorator.FrameDecorator(frame) separator = ansi(', ', R.style_low) if show_arguments: def prefix(line): return Stack.format_line('arg', line) frame_args = decorator.frame_args() args_lines = Variables.fetch(frame, frame_args, compact, align, sort) if args_lines: if compact: args_line = separator.join(args_lines) single_line = prefix(args_line) out.append(single_line) else: out.extend(map(prefix, args_lines)) if show_locals: def prefix(line): return Stack.format_line('loc', line) frame_locals = decorator.frame_locals() locals_lines = Variables.fetch(frame, frame_locals, compact, align, sort) if locals_lines: if compact: locals_line = separator.join(locals_lines) single_line = prefix(locals_line) out.append(single_line) else: out.extend(map(prefix, locals_lines)) return out @staticmethod def fetch(frame, data, compact, align, sort): lines = [] name_width = 0 if align and not compact: name_width = max(len(str(elem.sym)) for elem in data) if data else 0 for elem in data or []: name = ansi(elem.sym, R.style_high) + ' ' * (name_width - len(str(elem.sym))) equal = ansi('=', R.style_low) value = format_value(elem.sym.value(frame), compact) lines.append('{} {} {}'.format(name, equal, value)) if sort: lines.sort() return lines class Stack(Dashboard.Module): '''Show the current stack trace including the function name and the file location, if available. Optionally list the frame arguments and locals too.''' def label(self): return 'Stack' def lines(self, term_width, term_height, style_changed): # skip if the current thread is not stopped if not gdb.selected_thread().is_stopped(): return [] # find the selected frame level (XXX Frame.level() is a recent addition) start_level = 0 frame = gdb.newest_frame() while frame: if frame == gdb.selected_frame(): break frame = frame.older() start_level += 1 # gather the frames more = False frames = [gdb.selected_frame()] going_down = True while True: # stack frames limit reached if len(frames) == self.limit: more = True break # zigzag the frames starting from the selected one if going_down: frame = frames[-1].older() if frame: frames.append(frame) else: frame = frames[0].newer() if frame: frames.insert(0, frame) start_level -= 1 else: break else: frame = frames[0].newer() if frame: frames.insert(0, frame) start_level -= 1 else: frame = frames[-1].older() if frame: frames.append(frame) else: break # switch direction going_down = not going_down # format the output lines = [] for number, frame in enumerate(frames, start=start_level): selected = frame == gdb.selected_frame() lines.extend(self.get_frame_lines(number, frame, selected)) # add the placeholder if more: lines.append('[{}]'.format(ansi('+', R.style_selected_2))) return lines def attributes(self): return { 'limit': { 'doc': 'Maximum number of displayed frames (0 means no limit).', 'default': 10, 'type': int, 'check': check_ge_zero }, 'arguments': { 'doc': 'Frame arguments visibility flag.', 'default': False, 'name': 'show_arguments', 'type': bool }, 'locals': { 'doc': 'Frame locals visibility flag.', 'default': False, 'name': 'show_locals', 'type': bool }, 'compact': { 'doc': 'Single-line display flag.', 'default': False, 'type': bool }, 'align': { 'doc': 'Align variables in column flag (only if not compact).', 'default': False, 'type': bool }, 'sort': { 'doc': 'Sort variables by name.', 'default': False, 'type': bool } } def get_frame_lines(self, number, frame, selected=False): # fetch frame info style = R.style_selected_1 if selected else R.style_selected_2 frame_id = ansi(str(number), style) info = Stack.get_pc_line(frame, style) frame_lines = [] frame_lines.append('[{}] {}'.format(frame_id, info)) # add frame arguments and locals variables = Variables.format_frame( frame, self.show_arguments, self.show_locals, self.compact, self.align, self.sort) frame_lines.extend(variables) return frame_lines @staticmethod def format_line(prefix, line): prefix = ansi(prefix, R.style_low) return '{} {}'.format(prefix, line) @staticmethod def get_pc_line(frame, style): frame_pc = ansi(format_address(frame.pc()), style) info = 'from {}'.format(frame_pc) # if a frame function symbol is available then use it to fetch the # current function name and address, otherwise fall back relying on the # frame name if frame.function(): name = ansi(frame.function(), style) func_start = to_unsigned(frame.function().value()) offset = ansi(str(frame.pc() - func_start), style) info += ' in {}+{}'.format(name, offset) elif frame.name(): name = ansi(frame.name(), style) info += ' in {}'.format(name) sal = frame.find_sal() if sal and sal.symtab: file_name = ansi(sal.symtab.filename, style) file_line = ansi(str(sal.line), style) info += ' at {}:{}'.format(file_name, file_line) return info class History(Dashboard.Module): '''List the last entries of the value history.''' def label(self): return 'History' def lines(self, term_width, term_height, style_changed): out = [] # fetch last entries for i in range(-self.limit + 1, 1): try: value = format_value(gdb.history(i)) value_id = ansi('$${}', R.style_high).format(abs(i)) equal = ansi('=', R.style_low) line = '{} {} {}'.format(value_id, equal, value) out.append(line) except gdb.error: continue return out def attributes(self): return { 'limit': { 'doc': 'Maximum number of values to show.', 'default': 3, 'type': int, 'check': check_gt_zero } } class Memory(Dashboard.Module): '''Allow to inspect memory regions.''' DEFAULT_LENGTH = 16 class Region(): def __init__(self, expression, length, module): self.expression = expression self.length = length self.module = module self.original = None self.latest = None def reset(self): self.original = None self.latest = None def format(self, per_line): # fetch the memory content try: address = Memory.parse_as_address(self.expression) inferior = gdb.selected_inferior() memory = inferior.read_memory(address, self.length) # set the original memory snapshot if needed if not self.original: self.original = memory except gdb.error as e: msg = 'Cannot access {} bytes starting at {}: {}' msg = msg.format(self.length, self.expression, e) return [ansi(msg, R.style_error)] # format the memory content out = [] for i in range(0, len(memory), per_line): region = memory[i:i + per_line] pad = per_line - len(region) address_str = format_address(address + i) # compute changes hexa = [] text = [] for j in range(len(region)): rel = i + j byte = memory[rel] hexa_byte = '{:02x}'.format(ord(byte)) text_byte = self.module.format_byte(byte) # differences against the latest have the highest priority if self.latest and memory[rel] != self.latest[rel]: hexa_byte = ansi(hexa_byte, R.style_selected_1) text_byte = ansi(text_byte, R.style_selected_1) # cumulative changes if enabled elif self.module.cumulative and memory[rel] != self.original[rel]: hexa_byte = ansi(hexa_byte, R.style_selected_2) text_byte = ansi(text_byte, R.style_selected_2) # format the text differently for clarity else: text_byte = ansi(text_byte, R.style_high) hexa.append(hexa_byte) text.append(text_byte) # output the formatted line hexa_placeholder = ' {}'.format(self.module.placeholder[0] * 2) text_placeholder = self.module.placeholder[0] out.append('{} {}{} {}{}'.format( ansi(address_str, R.style_low), ' '.join(hexa), ansi(pad * hexa_placeholder, R.style_low), ''.join(text), ansi(pad * text_placeholder, R.style_low))) # update the latest memory snapshot self.latest = memory return out def __init__(self): self.table = {} def label(self): return 'Memory' def lines(self, term_width, term_height, style_changed): out = [] for expression, region in self.table.items(): out.append(divider(term_width, expression)) out.extend(region.format(self.get_per_line(term_width))) return out def commands(self): return { 'watch': { 'action': self.watch, 'doc': '''Watch a memory region by expression and length. The length defaults to 16 bytes.''', 'complete': gdb.COMPLETE_EXPRESSION }, 'unwatch': { 'action': self.unwatch, 'doc': 'Stop watching a memory region by expression.', 'complete': gdb.COMPLETE_EXPRESSION }, 'clear': { 'action': self.clear, 'doc': 'Clear all the watched regions.' } } def attributes(self): return { 'cumulative': { 'doc': 'Highlight changes cumulatively, watch again to reset.', 'default': False, 'type': bool }, 'full': { 'doc': 'Take the whole horizontal space.', 'default': False, 'type': bool }, 'placeholder': { 'doc': 'Placeholder used for missing items and unprintable characters.', 'default': '·' } } def watch(self, arg): if arg: expression, _, length_str = arg.partition(' ') length = Memory.parse_as_address(length_str) if length_str else Memory.DEFAULT_LENGTH # keep the length when the memory is watched to reset the changes region = self.table.get(expression) if region and not length_str: region.reset() else: self.table[expression] = Memory.Region(expression, length, self) else: raise Exception('Specify a memory location') def unwatch(self, arg): if arg: try: del self.table[arg] except KeyError: raise Exception('Memory expression not watched') else: raise Exception('Specify a matched memory expression') def clear(self, arg): self.table.clear() def format_byte(self, byte): # `type(byte) is bytes` in Python 3 if 0x20 < ord(byte) < 0x7f: return chr(ord(byte)) else: return self.placeholder[0] def get_per_line(self, term_width): if self.full: padding = 3 # two double spaces separator (one is part of below) elem_size = 4 # HH + 1 space + T address_length = gdb.parse_and_eval('$pc').type.sizeof * 2 + 2 # 0x return max(int((term_width - address_length - padding) / elem_size), 1) else: return Memory.DEFAULT_LENGTH @staticmethod def parse_as_address(expression): value = gdb.parse_and_eval(expression) return to_unsigned(value) class Registers(Dashboard.Module): '''Show the CPU registers and their values.''' def __init__(self): self.table = {} def label(self): return 'Registers' def lines(self, term_width, term_height, style_changed): # skip if the current thread is not stopped if not gdb.selected_thread().is_stopped(): return [] # obtain the registers to display if style_changed: self.table = {} if self.register_list: register_list = self.register_list.split() else: register_list = Registers.fetch_register_list() # fetch registers status registers = [] for name in register_list: # exclude registers with a dot '.' or parse_and_eval() will fail if '.' in name: continue value = gdb.parse_and_eval('${}'.format(name)) string_value = Registers.format_value(value) # exclude unavailable registers (see #255) if string_value == '': continue changed = self.table and (self.table.get(name, '') != string_value) self.table[name] = string_value registers.append((name, string_value, changed)) # handle the empty register list if not registers: msg = 'No registers to show (check the "dashboard registers -style list" attribute)' return [ansi(msg, R.style_error)] # compute lengths considering an extra space between and around the # entries (hence the +2 and term_width - 1) max_name = max(len(name) for name, _, _ in registers) max_value = max(len(value) for _, value, _ in registers) max_width = max_name + max_value + 2 columns = min(int((term_width - 1) / max_width) or 1, len(registers)) rows = int(math.ceil(float(len(registers)) / columns)) # build the registers matrix if self.column_major: matrix = list(registers[i:i + rows] for i in range(0, len(registers), rows)) else: matrix = list(registers[i::columns] for i in range(columns)) # compute the lengths column wise max_names_column = list(max(len(name) for name, _, _ in column) for column in matrix) max_values_column = list(max(len(value) for _, value, _ in column) for column in matrix) line_length = sum(max_names_column) + columns + sum(max_values_column) extra = term_width - line_length # compute padding as if there were one more column base_padding = int(extra / (columns + 1)) padding_column = [base_padding] * columns # distribute the remainder among columns giving the precedence to # internal padding rest = extra % (columns + 1) while rest: padding_column[rest % columns] += 1 rest -= 1 # format the registers out = [''] * rows for i, column in enumerate(matrix): max_name = max_names_column[i] max_value = max_values_column[i] for j, (name, value, changed) in enumerate(column): name = ' ' * (max_name - len(name)) + ansi(name, R.style_low) style = R.style_selected_1 if changed else '' value = ansi(value, style) + ' ' * (max_value - len(value)) padding = ' ' * padding_column[i] item = '{}{} {}'.format(padding, name, value) out[j] += item return out def attributes(self): return { 'column-major': { 'doc': 'Show registers in columns instead of rows.', 'default': False, 'name': 'column_major', 'type': bool }, 'list': { 'doc': '''String of space-separated register names to display. The empty list (default) causes to show all the available registers. For architectures different from x86 setting this attribute might be mandatory.''', 'default': '', 'name': 'register_list', } } @staticmethod def format_value(value): try: if value.type.code in [gdb.TYPE_CODE_INT, gdb.TYPE_CODE_PTR]: int_value = to_unsigned(value, value.type.sizeof) value_format = '0x{{:0{}x}}'.format(2 * value.type.sizeof) return value_format.format(int_value) except (gdb.error, ValueError): # convert to unsigned but preserve code and flags information pass return str(value) @staticmethod def fetch_register_list(*match_groups): names = [] for line in run('maintenance print register-groups').split('\n'): fields = line.split() if len(fields) != 7: continue name, _, _, _, _, _, groups = fields if not re.match('\w', name): continue for group in groups.split(','): if group in (match_groups or ('general',)): names.append(name) break return names class Threads(Dashboard.Module): '''List the currently available threads.''' def label(self): return 'Threads' def lines(self, term_width, term_height, style_changed): out = [] selected_thread = gdb.selected_thread() # do not restore the selected frame if the thread is not stopped restore_frame = gdb.selected_thread().is_stopped() if restore_frame: selected_frame = gdb.selected_frame() # fetch the thread list threads = [] for inferior in gdb.inferiors(): if self.all_inferiors or inferior == gdb.selected_inferior(): threads += gdb.Inferior.threads(inferior) for thread in threads: # skip running threads if requested if self.skip_running and thread.is_running(): continue is_selected = (thread.ptid == selected_thread.ptid) style = R.style_selected_1 if is_selected else R.style_selected_2 if self.all_inferiors: number = '{}.{}'.format(thread.inferior.num, thread.num) else: number = str(thread.num) number = ansi(number, style) tid = ansi(str(thread.ptid[1] or thread.ptid[2]), style) info = '[{}] id {}'.format(number, tid) if thread.name: info += ' name {}'.format(ansi(thread.name, style)) # switch thread to fetch info (unless is running in non-stop mode) try: thread.switch() frame = gdb.newest_frame() info += ' ' + Stack.get_pc_line(frame, style) except gdb.error: info += ' (running)' out.append(info) # restore thread and frame selected_thread.switch() if restore_frame: selected_frame.select() return out def attributes(self): return { 'skip-running': { 'doc': 'Skip running threads.', 'default': False, 'name': 'skip_running', 'type': bool }, 'all-inferiors': { 'doc': 'Show threads from all inferiors.', 'default': False, 'name': 'all_inferiors', 'type': bool }, } class Expressions(Dashboard.Module): '''Watch user expressions.''' def __init__(self): self.table = [] def label(self): return 'Expressions' def lines(self, term_width, term_height, style_changed): out = [] label_width = 0 if self.align: label_width = max(len(expression) for expression in self.table) if self.table else 0 default_radix = Expressions.get_default_radix() for number, expression in enumerate(self.table, start=1): label = expression match = re.match('^/(\d+) +(.+)$', expression) try: if match: radix, expression = match.groups() run('set output-radix {}'.format(radix)) value = format_value(gdb.parse_and_eval(expression)) except gdb.error as e: value = ansi(e, R.style_error) finally: if match: run('set output-radix {}'.format(default_radix)) number = ansi(str(number), R.style_selected_2) label = ansi(expression, R.style_high) + ' ' * (label_width - len(expression)) equal = ansi('=', R.style_low) out.append('[{}] {} {} {}'.format(number, label, equal, value)) return out def commands(self): return { 'watch': { 'action': self.watch, 'doc': 'Watch an expression using the format `[/] `.', 'complete': gdb.COMPLETE_EXPRESSION }, 'unwatch': { 'action': self.unwatch, 'doc': 'Stop watching an expression by index.' }, 'clear': { 'action': self.clear, 'doc': 'Clear all the watched expressions.' } } def attributes(self): return { 'align': { 'doc': 'Align variables in column flag.', 'default': False, 'type': bool } } def watch(self, arg): if arg: if arg not in self.table: self.table.append(arg) else: raise Exception('Expression already watched') else: raise Exception('Specify an expression') def unwatch(self, arg): if arg: try: number = int(arg) - 1 except: number = -1 if 0 <= number < len(self.table): self.table.pop(number) else: raise Exception('Expression not watched') else: raise Exception('Specify an expression') def clear(self, arg): self.table.clear() @staticmethod def get_default_radix(): try: return gdb.parameter('output-radix') except RuntimeError: # XXX this is a fix for GDB <8.1.x see #161 message = run('show output-radix') match = re.match('^Default output radix for printing of values is (\d+)\.$', message) return match.groups()[0] if match else 10 # fallback # XXX workaround to support BP_BREAKPOINT in older GDB versions setattr(gdb, 'BP_CATCHPOINT', getattr(gdb, 'BP_CATCHPOINT', 26)) class Breakpoints(Dashboard.Module): '''Display the breakpoints list.''' NAMES = { gdb.BP_BREAKPOINT: 'break', gdb.BP_WATCHPOINT: 'watch', gdb.BP_HARDWARE_WATCHPOINT: 'write watch', gdb.BP_READ_WATCHPOINT: 'read watch', gdb.BP_ACCESS_WATCHPOINT: 'access watch', gdb.BP_CATCHPOINT: 'catch' } def label(self): return 'Breakpoints' def lines(self, term_width, term_height, style_changed): out = [] breakpoints = fetch_breakpoints(watchpoints=True, pending=self.show_pending) for breakpoint in breakpoints: sub_lines = [] # format common information style = R.style_selected_1 if breakpoint['enabled'] else R.style_selected_2 number = ansi(breakpoint['number'], style) bp_type = ansi(Breakpoints.NAMES[breakpoint['type']], style) if breakpoint['temporary']: bp_type = bp_type + ' {}'.format(ansi('once', style)) if not R.ansi and breakpoint['enabled']: bp_type = 'disabled ' + bp_type line = '[{}] {}'.format(number, bp_type) if breakpoint['type'] == gdb.BP_BREAKPOINT: for i, address in enumerate(breakpoint['addresses']): addr = address['address'] if i == 0 and addr: # this is a regular breakpoint line += ' at {}'.format(ansi(format_address(addr), style)) # format source information file_name = address.get('file_name') file_line = address.get('file_line') if file_name and file_line: file_name = ansi(file_name, style) file_line = ansi(file_line, style) line += ' in {}:{}'.format(file_name, file_line) elif i > 0: # this is a sub breakpoint sub_style = R.style_selected_1 if address['enabled'] else R.style_selected_2 sub_number = ansi('{}.{}'.format(breakpoint['number'], i), sub_style) sub_line = '[{}]'.format(sub_number) sub_line += ' at {}'.format(ansi(format_address(addr), sub_style)) # format source information file_name = address.get('file_name') file_line = address.get('file_line') if file_name and file_line: file_name = ansi(file_name, sub_style) file_line = ansi(file_line, sub_style) sub_line += ' in {}:{}'.format(file_name, file_line) sub_lines += [sub_line] # format user location location = breakpoint['location'] line += ' for {}'.format(ansi(location, style)) elif breakpoint['type'] == gdb.BP_CATCHPOINT: what = breakpoint['what'] line += ' {}'.format(ansi(what, style)) else: # format user expression expression = breakpoint['expression'] line += ' for {}'.format(ansi(expression, style)) # format condition condition = breakpoint['condition'] if condition: line += ' if {}'.format(ansi(condition, style)) # format hit count hit_count = breakpoint['hit_count'] if hit_count: word = 'time{}'.format('s' if hit_count > 1 else '') line += ' hit {} {}'.format(ansi(breakpoint['hit_count'], style), word) # append the main line and possibly sub breakpoints out.append(line) out.extend(sub_lines) return out def attributes(self): return { 'pending': { 'doc': 'Also show pending breakpoints.', 'default': True, 'name': 'show_pending', 'type': bool } } # XXX traceback line numbers in this Python block must be increased by 1 end # Better GDB defaults ---------------------------------------------------------- set history save set verbose off set print pretty on set print array off set print array-indexes on set python print-stack full # Start ------------------------------------------------------------------------ python Dashboard.start() # File variables --------------------------------------------------------------- # vim: filetype=python # Local Variables: # mode: python # End: