mirror of
https://github.com/tromey/gdb-gui.git
synced 2025-07-21 13:01:16 +02:00
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
320 lines
11 KiB
Python
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)
|