Commit dcd7f2a7 authored by Mark Florisson's avatar Mark Florisson

Added 'cy exec' and 'py-exec' commands

parent ad032f3f
...@@ -346,11 +346,7 @@ class CythonBase(object): ...@@ -346,11 +346,7 @@ class CythonBase(object):
selected_frame.select() selected_frame.select()
def get_cython_globals_dict(self): def get_remote_cython_globals_dict(self):
"""
Get the Cython globals dict where the remote names are turned into
local strings.
"""
m = gdb.parse_and_eval('__pyx_m') m = gdb.parse_and_eval('__pyx_m')
try: try:
...@@ -361,7 +357,16 @@ class CythonBase(object): ...@@ -361,7 +357,16 @@ class CythonBase(object):
with debugging support (-g)?""")) with debugging support (-g)?"""))
m = m.cast(PyModuleObject.pointer()) m = m.cast(PyModuleObject.pointer())
pyobject_dict = libpython.PyObjectPtr.from_pyobject_ptr(m['md_dict']) return m['md_dict']
def get_cython_globals_dict(self):
"""
Get the Cython globals dict where the remote names are turned into
local strings.
"""
remote_dict = self.get_remote_cython_globals_dict()
pyobject_dict = libpython.PyObjectPtr.from_pyobject_ptr(remote_dict)
result = {} result = {}
seen = set() seen = set()
...@@ -598,6 +603,8 @@ class CyCy(CythonCommand): ...@@ -598,6 +603,8 @@ class CyCy(CythonCommand):
print_ = CyPrint.register(), print_ = CyPrint.register(),
locals = CyLocals.register(), locals = CyLocals.register(),
globals = CyGlobals.register(), globals = CyGlobals.register(),
exec_ = libpython.FixGdbCommand('cy exec', '-cy-exec'),
_exec = CyExec.register(),
cy_cname = CyCName('cy_cname'), cy_cname = CyCName('cy_cname'),
cy_cvalue = CyCValue('cy_cvalue'), cy_cvalue = CyCValue('cy_cvalue'),
cy_lineno = CyLine('cy_lineno'), cy_lineno = CyLine('cy_lineno'),
...@@ -1092,6 +1099,75 @@ class CyGlobals(CyLocals): ...@@ -1092,6 +1099,75 @@ class CyGlobals(CyLocals):
prefix=' ') prefix=' ')
class CyExec(CythonCommand):
name = '-cy-exec'
command_class = gdb.COMMAND_STACK
completer_class = gdb.COMPLETE_NONE
def _fill_locals_dict(self, executor, local_dict_pointer):
"Fill a remotely allocated dict with values from the Cython C stack"
cython_func = self.get_cython_function()
for name, cyvar in cython_func.locals.iteritems():
if cyvar.type == PythonObject:
# skip unitialized Cython variables
try:
val = gdb.parse_and_eval(cyvar.cname)
except RuntimeError:
continue
else:
# Fortunately, Cython initializes all local (automatic)
# variables to NULL
if libpython.pointervalue(val) == 0:
continue
pystringp = executor.alloc_pystring(name)
code = '''
PyDict_SetItem(
(PyObject *) %d,
(PyObject *) %d,
(PyObject *) %s)
''' % (local_dict_pointer, pystringp, cyvar.cname)
# PyDict_SetItem doesn't steal our reference
executor.decref(pystringp)
if gdb.parse_and_eval(code) < 0:
gdb.parse_and_eval('PyErr_Print()')
raise gdb.GdbError("Unable to execute Python code.")
def _find_first_cython_or_python_frame(self):
frame = gdb.selected_frame()
while frame:
if (self.is_cython_function(frame) or
self.is_python_function(frame)):
return frame
frame = frame.older()
raise gdb.GdbError("There is no Cython or Python frame on the stack.")
def invoke(self, expr, from_tty):
frame = self._find_first_cython_or_python_frame()
if self.is_python_function(frame):
libpython.py_exec.invoke(expr, from_tty)
return
executor = libpython.PythonCodeExecutor()
# get the dict of Cython globals and construct a dict in the inferior
# with Cython locals
global_dict = gdb.parse_and_eval(
'(PyObject *) PyModule_GetDict(__pyx_m)')
local_dict = gdb.parse_and_eval('(PyObject *) PyDict_New()')
try:
self._fill_locals_dict(executor, libpython.pointervalue(local_dict))
executor.evalcode(expr, global_dict, local_dict)
finally:
executor.decref(libpython.pointervalue(local_dict))
# Functions # Functions
class CyCName(gdb.Function, CythonBase): class CyCName(gdb.Function, CythonBase):
......
...@@ -49,6 +49,7 @@ import os ...@@ -49,6 +49,7 @@ import os
import re import re
import sys import sys
import atexit import atexit
import warnings
import tempfile import tempfile
import itertools import itertools
...@@ -1591,6 +1592,22 @@ gdb.execute = execute ...@@ -1591,6 +1592,22 @@ gdb.execute = execute
_logging_state = _LoggingState() _logging_state = _LoggingState()
def get_selected_inferior():
"""
Return the selected inferior in gdb.
"""
# Woooh, another bug in gdb! Is there an end in sight?
# http://sourceware.org/bugzilla/show_bug.cgi?id=12212
return gdb.inferiors()[0]
selected_thread = gdb.selected_thread()
for inferior in gdb.inferiors():
for thread in inferior.threads():
if thread == selected_thread:
return inferior
class GenericCodeStepper(gdb.Command): class GenericCodeStepper(gdb.Command):
""" """
Superclass for code stepping. Subclasses must implement the following Superclass for code stepping. Subclasses must implement the following
...@@ -1687,7 +1704,7 @@ class GenericCodeStepper(gdb.Command): ...@@ -1687,7 +1704,7 @@ class GenericCodeStepper(gdb.Command):
""" """
def stopped(self): def stopped(self):
return gdb.inferiors()[0].pid == 0 return get_selected_inferior().pid == 0
def _stackdepth(self, frame): def _stackdepth(self, frame):
depth = 0 depth = 0
...@@ -1882,3 +1899,191 @@ py_run = PyRun('py-run') ...@@ -1882,3 +1899,191 @@ py_run = PyRun('py-run')
py_cont = PyCont('py-cont') py_cont = PyCont('py-cont')
py_step.init_breakpoints() py_step.init_breakpoints()
Py_single_input = 256
Py_eval_input = 258
def pointervalue(gdbval):
"""
Return the value of the pionter as a Python int.
gdbval.type must be a pointer type
"""
# don't convert with int() as it will raise a RuntimeError
if gdbval.address is not None:
return long(gdbval.address)
else:
# the address attribute is None sometimes, in which case we can
# still convert the pointer to an int
return long(gdbval)
class PythonCodeExecutor(object):
def malloc(self, size):
chunk = (gdb.parse_and_eval("(void *) malloc(%d)" % size))
pointer = pointervalue(chunk)
if pointer == 0:
err("No memory could be allocated in the inferior.")
return pointer
def alloc_string(self, string):
pointer = self.malloc(len(string))
get_selected_inferior().write_memory(pointer, string)
return pointer
def alloc_pystring(self, string):
stringp = self.alloc_string(string)
try:
result = gdb.parse_and_eval(
'PyString_FromStringAndSize((char *) %d, (size_t) %d)' %
(stringp, len(string)))
finally:
self.free(stringp)
pointer = pointervalue(result)
if pointer == 0:
err("Unable to allocate Python string in "
"the inferior.")
return pointer
def free(self, pointer):
gdb.parse_and_eval("free((void *) %d)" % pointer)
def decref(self, pointer):
"Decrement the reference count of a Python object in the inferior."
# Py_DecRef is like Py_XDECREF, but a function. So we don't have
# to check for NULL. This should also decref all our allocated
# Python strings.
gdb.parse_and_eval('Py_DecRef((PyObject *) %d)' % pointer)
def evalcode(self, code, global_dict=None, local_dict=None):
"""
Evaluate python code `code` given as a string in the inferior and
return the result as a gdb.Value. Returns a new reference in the
inferior.
Of course, executing any code in the inferior may be dangerous and may
leave the debuggee in an unsafe state or terminate it alltogether.
"""
if '\0' in code:
err("String contains NUL byte.")
code += '\0'
pointer = self.alloc_string(code)
globalsp = pointervalue(global_dict)
localsp = pointervalue(local_dict)
if globalp == 0 or localp == 0:
raise gdb.GdbError("Unable to obtain or create locals or globals.")
code = """
PyRun_String(
(PyObject *) %(code)d,
(int) %(start)d,
(PyObject *) %(globals)s,
(PyObject *) %(locals)d)
""" % dict(code=pointer, start=Py_single_input,
globals=globalsp, locals=localsp)
with FetchAndRestoreError():
try:
self.decref(gdb.parse_and_eval(code))
finally:
self.free(pointer)
class FetchAndRestoreError(PythonCodeExecutor):
"""
Context manager that fetches the error indicator in the inferior and
restores it on exit.
"""
def __init__(self):
self.sizeof_PyObjectPtr = gdb.lookup_type('PyObject').pointer().sizeof
self.pointer = self.malloc(self.sizeof_PyObjectPtr * 3)
type = self.pointer
value = self.pointer + self.sizeof_PyObjectPtr
traceback = self.pointer + self.sizeof_PyObjectPtr * 2
self.errstate = type, value, traceback
def __enter__(self):
gdb.parse_and_eval("PyErr_Fetch(%d, %d, %d)" % self.errstate)
def __exit__(self, *args):
if gdb.parse_and_eval("(int) PyErr_Occurred()"):
gdb.parse_and_eval("PyErr_Print()")
pyerr_restore = ("PyErr_Restore("
"(PyObject *) *%d,"
"(PyObject *) *%d,"
"(PyObject *) *%d)")
try:
gdb.parse_and_eval(pyerr_restore % self.errstate)
finally:
self.free(self.pointer)
class FixGdbCommand(gdb.Command):
def __init__(self, command, actual_command):
super(FixGdbCommand, self).__init__(command, gdb.COMMAND_DATA,
gdb.COMPLETE_NONE)
self.actual_command = actual_command
def fix_gdb(self):
"""
So, you must be wondering what the story is this time! Yeeees, indeed,
I have quite the story for you! It seems that invoking either 'cy exec'
and 'py-exec' work perfectly fine, but after this gdb's python API is
entirely broken. Some unset exception value is still set?
sys.exc_clear() didn't help. A demonstration:
(gdb) cy exec 'hello'
'hello'
(gdb) python gdb.execute('cont')
RuntimeError: Cannot convert value to int.
Error while executing Python code.
(gdb) python gdb.execute('cont')
[15148 refs]
Program exited normally.
"""
warnings.filterwarnings('ignore', r'.*', RuntimeWarning, re.escape(__name__))
try:
long(gdb.parse_and_eval("(void *) 0")) == 0
except RuntimeError:
pass
# warnings.resetwarnings()
def invoke(self, args, from_tty):
self.fix_gdb()
gdb.execute('%s %s' % (self.actual_command, args))
self.fix_gdb()
class PyExec(gdb.Command):
def invoke(self, expr, from_tty):
executor = PythonCodeExecutor()
global_dict = gdb.parse_and_eval('PyEval_GetGlobals()')
local_dict = gdb.parse_and_eval('PyEval_GetLocals()')
if pointervalue(global_dict) == 0 or pointervalue(local_dict) == 0:
raise gdb.GdbError("Unable to find the locals or globals of the "
"most recent Python function (relative to the "
"selected frame).")
executor.evalcode(expr, global_dict, local_dict)
py_exec = FixGdbCommand('py-exec', '-py-exec')
_py_exec = PyExec("-py-exec", gdb.COMMAND_DATA, gdb.COMPLETE_NONE)
\ No newline at end of file
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment