Files
gdb-gui/gui/commands.py
Tom Tromey 22bdecb10d make it work with both python2 and python3
I ran 2to3, fixed up the output a tiny bit, and fixed the C module as
well.  I smoke tested with both versions of Python but didn't use it in
anger yet.

Fix #29
2015-05-19 09:44:09 -06:00

320 lines
11 KiB
Python

# Copyright (C) 2012, 2013, 2015 Tom Tromey <tom@tromey.com>
# This program is free software; you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation; either version 3 of the License, or
# (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program. If not, see <http://www.gnu.org/licenses/>.
import gdb
import gui.startup
import gui.source
import gui.logwindow
import gui.toplevel
import gui.dprintf
import gui.events
import gui.display
import gui.gdbutil
import gui.adapt
import re
from gui.startup import in_gtk_thread
class GuiCommand(gdb.Command):
def __init__(self):
super(GuiCommand, self).__init__('gui', gdb.COMMAND_SUPPORT,
prefix = True)
class GuiSourceCommand(gdb.Command):
"""Create a new source window.
Usage: gui source
This creates a new source window in the GUI. Any number of source
windows can be created."""
def __init__(self):
super(GuiSourceCommand, self).__init__('gui source',
gdb.COMMAND_SUPPORT)
def invoke(self, arg, from_tty):
self.dont_repeat()
gui.source.lru_handler.new_source_window()
class GuiListCommand(gdb.Command):
"""List some source code in a source window.
Usage: gui list LINESPEC
This command uses LINESPEC to show some source code in a source
window. If a source window is already available, the source is
displayed there. Otherwise, a new source window is made.
LINESPEC is a line specification of the form given to 'break'."""
def __init__(self):
super(GuiListCommand, self).__init__('gui list',
gdb.COMMAND_SUPPORT)
def invoke(self, arg, from_tty):
self.dont_repeat()
(extra, sals) = gdb.decode_line(arg)
if extra is not None:
raise gdb.GdbError('unrecognized junk at end of command: ' + extra)
if sals is None:
raise gdb.GdbError('not found')
if len(sals) > 1:
print("Ambiguous linespec, only showing first result")
sal = sals[0]
if sal.symtab is None or sal.symtab.filename is None:
raise gdb.GdbError('could not find file for symbol')
gui.source.lru_handler.show_source_gdb(None, sal.symtab,
sal.symtab.fullname(),
sal.line)
class GuiShowCommand(gdb.Command):
"""Show the source for a symbol in a source window.
Usage: gui show SYMBOL
This command looks for the definition of SYMBOL in the program and
shows its source location in a source window. If a source window is
already available, the source is displayed there. Otherwise, a new
source window is made."""
def __init__(self):
super(GuiShowCommand, self).__init__('gui show',
gdb.COMMAND_SUPPORT)
def invoke(self, arg, from_tty):
self.dont_repeat()
try:
(symbol, ignore) = gdb.lookup_symbol(arg)
except gdb.error as e:
if gui.gdbutil.is_running():
raise
gui.adapt.notify_bug(13351)
symbol = gdb.lookup_global_symbol(arg)
if symbol is None:
raise gdb.GdbError('symbol ' + arg + ' not found')
if symbol.symtab is None or symbol.symtab.filename is None:
raise gdb.GdbError('symbol ' + arg
+ ' does not seem to have an associated file')
gui.source.lru_handler.show_source_gdb(None, symbol.symtab,
symbol.symtab.fullname(),
symbol.line)
class GuiLogWindowCommand(gdb.Command):
"""Create a new log window.
Usage: gui log
This creates a new "log" window in the GUI. A log window is used
to display output from "gui print", "gui printf", "gui output",
and "gui dprintf".
Multiple log windows can be created and output can be directed to
a given instance using the "@" syntax, like:
gui print @5 variable"""
def __init__(self):
super(GuiLogWindowCommand, self).__init__('gui log',
gdb.COMMAND_SUPPORT)
def invoke(self, arg, from_tty):
self.dont_repeat()
window = gui.logwindow.LogWindow()
print("Created log window %d; now the default" % window.number)
class GuiPrintBase(gdb.Command):
def __init__(self, command):
super(GuiPrintBase, self).__init__('gui ' + command,
gdb.COMMAND_SUPPORT)
self.command = command
# Given ARG, return a pair (WINDOW, NEW_ARG).
def _parse_arg(self, arg, do_default = True):
arg = arg.strip()
match = re.match('@(\\d+)\\s+(.*)$', arg)
if match is not None:
winno = int(match.group(1))
arg = match.group(2)
window = gui.toplevel.state.get(winno)
if window is None:
raise gdb.GdbError('could not find window %d' % winno)
if not isinstance(window, gui.logwindow.LogWindow):
raise gdb.GdbError('window %d is not a log window' % winno)
elif do_default:
window = gui.logwindow.default_log_window
if window is None:
raise gdb.GdbError('no default log window')
else:
window = None
return (window, arg)
def invoke(self, arg, from_tty):
(window, arg) = self._parse_arg(arg)
text = gdb.execute(self.command + ' ' + arg, from_tty, True)
window.append(text)
class GuiPrintCommand(GuiPrintBase):
"""Print to a gui log window.
Usage: gui print [@N] ARGS
This is a wrapper for the "print" command that redirects its output
to a "gui log" window. If "@N" is given, then output goes to that
window; otherwise, output goes to the most recently created log window."""
def __init__(self):
super(GuiPrintCommand, self).__init__('print')
class GuiOutputCommand(GuiPrintBase):
"""Output to a gui log window.
Usage: gui output [@N] ARGS
This is a wrapper for the "output" command that redirects its output
to a "gui log" window. If "@N" is given, then output goes to that
window; otherwise, output goes to the most recently created log window."""
def __init__(self):
super(GuiOutputCommand, self).__init__('output')
class GuiPrintfCommand(GuiPrintBase):
"""printf to a gui log window.
Usage: gui printf [@N] ARGS
This is a wrapper for the "printf" command that redirects its output
to a "gui log" window. If "@N" is given, then output goes to that
window; otherwise, output goes to the most recently created log window."""
def __init__(self):
super(GuiPrintfCommand, self).__init__('printf')
class GuiDprintfCommand(GuiPrintBase):
"""dprintf to a gui log window.
Usage: gui dprintf [@N] ARGS
This is a wrapper for the "dprintf" command that redirects its output
to a "gui log" window. If "@N" is given, then output goes to that
window; otherwise, output goes to the most recently created log window."""
def __init__(self):
super(GuiDprintfCommand, self).__init__('dprintf')
def invoke(self, arg, from_tty):
(window, arg) = self._parse_arg(arg, False)
orig_arg = arg
(ignore, arg) = gdb.decode_line(arg)
if arg is None:
raise gdb.GdbError("no printf arguments to 'gui dprintf'")
arg = arg.strip()
if not arg.startswith(','):
raise gdb.GdbError("comma expected after linespec")
arg = arg[1:]
spec = arg[0 : -len(arg)]
DPrintfBreakpoint(spec, window, arg)
class GuiDisplayCommand(gdb.Command):
"""Create a new display window.
Usage: gui display [-diff] COMMAND
A display window runs a gdb command after operations that change the
current frame or that cause the inferior to stop. If "-diff" is
given, then every time the display is updated, changed lines are
highlighted."""
def __init__(self):
super(GuiDisplayCommand, self).__init__('gui display',
gdb.COMMAND_SUPPORT)
def invoke(self, arg, from_tty):
self.dont_repeat()
diff = False
if arg.startswith('-diff '):
diff = True
arg = arg[6:]
gui.display.DisplayWindow(arg, diff)
def complete(self, text, word):
# FIXME, see
# https://sourceware.org/bugzilla/show_bug.cgi?id=13077
return None
class InfoWindowsCommand(gdb.Command):
"""List all the GUI windows.
Usage: info windows
This lists all the GUI windows.
Note that this should not be confused with "info win", which is
part of the TUI."""
def __init__(self):
super(InfoWindowsCommand, self).__init__('info windows',
gdb.COMMAND_SUPPORT)
def invoke(self, arg, from_tty):
self.dont_repeat()
gui.toplevel.state.display()
class DeleteWindowsCommand(gdb.Command):
"""Delete a GUI window.
Usage: delete window N
Delete GUI window number N.
A window's number appears in its title bar, and can also be
found using "info windows"."""
def __init__(self):
super(DeleteWindowsCommand, self).__init__('delete window',
gdb.COMMAND_SUPPORT)
def invoke(self, arg, from_tty):
self.dont_repeat()
winno = int(arg)
window = gui.toplevel.state.get(winno)
if window is not None:
window.destroy()
GuiCommand()
GuiSourceCommand()
GuiLogWindowCommand()
GuiPrintCommand()
GuiOutputCommand()
GuiPrintfCommand()
GuiDprintfCommand()
GuiDisplayCommand()
GuiListCommand()
GuiShowCommand()
InfoWindowsCommand()
DeleteWindowsCommand()
_can_override = False
# A temporary test to see if you have a gdb that supports this.
class TestCommand(gdb.Command):
"""A temporary test command created for the GUI.
This does nothing, the GUI startup code uses it to see if
your copy of gdb has some command-overriding support."""
def __init__(self, set_it):
super(TestCommand, self).__init__('maint gui-test', gdb.COMMAND_DATA)
self.set_it = set_it
def invoke(self, arg, from_tty):
if self.set_it:
global _can_override
_can_override = True
else:
try:
super(TestCommand, self).invoke(arg, from_tty)
except:
pass
TestCommand(True)
TestCommand(False).invoke('', 0)
# See framecache.py - we prefer the before_prompt event if it exists;
# but otherwise try the overriding approach. Both of these rely on a
# hacked gdb :-(
if _can_override and not hasattr(gdb.events, 'before_prompt'):
class Overrider(gdb.Command):
def __init__(self, name, event):
super(Overrider, self).__init__(name, gdb.COMMAND_DATA)
self.event = event
def invoke(self, arg, from_tty):
super(Overrider, self).invoke(arg, from_tty)
self.event.post()
Overrider('up', gui.events.frame_changed)
Overrider('down', gui.events.frame_changed)
Overrider('frame', gui.events.frame_changed)