2013-05-10 Phil Muldoon <pmuldoon@redhat.com>
* stack.c (backtrace_command_1): Add "no-filters", and Python frame
filter logic.
(backtrace_command): Add "no-filters" option parsing.
(_initialize_stack): Alter help to reflect "no-filters" option.
* Makefile.in (SUBDIR_PYTHON_OBS): Add py-framefilter.o
(SUBDIR_PYTHON_SRCS): Add py-framefilter.c
(py-frame.o): Add target
* data-directory/Makefile.in (PYTHON_DIR): Add Python frame
filter files.
* python/python.h: Add new frame filter constants, and flag enum.
(apply_frame_filter): Add definition.
* python/python.c (apply_frame_filter): New non-Python
enabled function.
* python/py-utils.c (py_xdecref): New function.
(make_cleanup_py_xdecref): Ditto.
* python/py-objfile.c: Declare frame_filters dictionary.
(objfpy_dealloc): Add frame_filters dealloc.
(objfpy_new): Initialize frame_filters attribute.
(objfile_to_objfile_object): Ditto.
(objfpy_get_frame_filters): New function.
(objfpy_set_frame_filters): New function.
* python/py-progspace.c: Declare frame_filters dictionary.
(pspy_dealloc): Add frame_filters dealloc.
(pspy_new): Initialize frame_filters attribute.
(pspacee_to_pspace_object): Ditto.
(pspy_get_frame_filters): New function.
(pspy_set_frame_filters): New function.
* python/py-framefilter.c: New file.
* python/lib/gdb/command/frame_filters.py: New file.
* python/lib/gdb/frames.py: New file.
* python/lib/gdb/__init__.py: Initialize global frame_filters
dictionary
* python/lib/gdb/FrameDecorator.py: New file.
* python/lib/gdb/FrameIterator.py: New file.
* mi/mi-cmds.c (mi_cmds): Add frame filters command.
* mi/mi-cmds.h: Declare.
* mi/mi-cmd-stack.c (mi_cmd_stack_list_frames): Add
--no-frame-filter logic, and Python frame filter logic.
(stack_enable_frame_filters): New function.
(parse_no_frame_option): Ditto.
(mi_cmd_stack_list_frames): Add --no-frame-filter and Python frame
filter logic.
(mi_cmd_stack_list_locals): Ditto.
(mi_cmd_stack_list_args): Ditto.
(mi_cmd_stack_list_variables): Ditto.
* NEWS: Add frame filter note.
2013-05-10 Phil Muldoon <pmuldoon@redhat.com>
* gdb.python/py-framefilter.py: New File.
* gdb.python/py-framefilter-mi.exp: Ditto.
* gdb.python/py-framefilter.c: Ditto.
* gdb.python/py-framefilter-mi.exp: Ditto.
* gdb.python/py-framefilter-mi.c: Ditto,
* gdb.python/py-framefilter-gdb.py.in: Ditto.
2013-05-10 Phil Muldoon <pmuldoon@redhat.com>
* gdb.texinfo (Backtrace): Add "no-filter" argument.
(Python API): Add Frame Filters API, Frame Wrapper API,
Writing a Frame Filter/Wrapper, Managing Management of Frame
Filters chapter entries.
(Frame Filters API): New Node.
(Frame Wrapper API): New Node.
(Writing a Frame Filter): New Node.
(Managing Frame Filters): New Node.
(Progspaces In Python): Add note about frame_filters attribute.
(Objfiles in Python): Ditto.
(GDB/MI Stack Manipulation): Add -enable-frame-filters command,
@anchors and --no-frame-filters option to -stack-list-variables,
-stack-list-frames, -stack-list-locals and -stack-list-arguments
commands.
2013-05-10 10:26:03 +00:00
|
|
|
/* Python frame filters
|
|
|
|
|
|
|
|
Copyright (C) 2013 Free Software Foundation, Inc.
|
|
|
|
|
|
|
|
This file is part of GDB.
|
|
|
|
|
|
|
|
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/>. */
|
|
|
|
|
|
|
|
#include "defs.h"
|
|
|
|
#include "objfiles.h"
|
|
|
|
#include "symtab.h"
|
|
|
|
#include "language.h"
|
|
|
|
#include "exceptions.h"
|
|
|
|
#include "arch-utils.h"
|
|
|
|
#include "python.h"
|
|
|
|
#include "ui-out.h"
|
|
|
|
#include "valprint.h"
|
|
|
|
#include "annotate.h"
|
|
|
|
#include "hashtab.h"
|
|
|
|
#include "demangle.h"
|
|
|
|
#include "mi/mi-cmds.h"
|
|
|
|
#include "python-internal.h"
|
|
|
|
|
|
|
|
enum mi_print_types
|
|
|
|
{
|
|
|
|
MI_PRINT_ARGS,
|
|
|
|
MI_PRINT_LOCALS
|
|
|
|
};
|
|
|
|
|
|
|
|
/* Helper function to extract a symbol, a name and a language
|
|
|
|
definition from a Python object that conforms to the "Symbol Value"
|
|
|
|
interface. OBJ is the Python object to extract the values from.
|
|
|
|
NAME is a pass-through argument where the name of the symbol will
|
|
|
|
be written. NAME is allocated in this function, but the caller is
|
|
|
|
responsible for clean up. SYM is a pass-through argument where the
|
|
|
|
symbol will be written. In the case of the API returning a string,
|
|
|
|
this will be set to NULL. LANGUAGE is also a pass-through argument
|
|
|
|
denoting the language attributed to the Symbol. In the case of SYM
|
|
|
|
being NULL, this will be set to the current language. Returns
|
|
|
|
PY_BT_ERROR on error with the appropriate Python exception set, and
|
|
|
|
PY_BT_OK on success. */
|
|
|
|
|
|
|
|
static enum py_bt_status
|
|
|
|
extract_sym (PyObject *obj, char **name, struct symbol **sym,
|
|
|
|
const struct language_defn **language)
|
|
|
|
{
|
|
|
|
PyObject *result = PyObject_CallMethod (obj, "symbol", NULL);
|
|
|
|
|
|
|
|
if (result == NULL)
|
|
|
|
return PY_BT_ERROR;
|
|
|
|
|
|
|
|
/* For 'symbol' callback, the function can return a symbol or a
|
|
|
|
string. */
|
|
|
|
if (gdbpy_is_string (result))
|
|
|
|
{
|
|
|
|
*name = python_string_to_host_string (result);
|
|
|
|
Py_DECREF (result);
|
|
|
|
|
|
|
|
if (*name == NULL)
|
|
|
|
return PY_BT_ERROR;
|
|
|
|
/* If the API returns a string (and not a symbol), then there is
|
|
|
|
no symbol derived language available and the frame filter has
|
|
|
|
either overridden the symbol with a string, or supplied a
|
|
|
|
entirely synthetic symbol/value pairing. In that case, use
|
|
|
|
python_language. */
|
|
|
|
*language = python_language;
|
|
|
|
*sym = NULL;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/* This type checks 'result' during the conversion so we
|
|
|
|
just call it unconditionally and check the return. */
|
|
|
|
*sym = symbol_object_to_symbol (result);
|
|
|
|
|
|
|
|
Py_DECREF (result);
|
|
|
|
|
|
|
|
if (*sym == NULL)
|
|
|
|
{
|
|
|
|
PyErr_SetString (PyExc_RuntimeError,
|
|
|
|
_("Unexpected value. Expecting a "
|
|
|
|
"gdb.Symbol or a Python string."));
|
|
|
|
return PY_BT_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Duplicate the symbol name, so the caller has consistency
|
|
|
|
in garbage collection. */
|
|
|
|
*name = xstrdup (SYMBOL_PRINT_NAME (*sym));
|
|
|
|
|
|
|
|
/* If a symbol is specified attempt to determine the language
|
|
|
|
from the symbol. If mode is not "auto", then the language
|
|
|
|
has been explicitly set, use that. */
|
|
|
|
if (language_mode == language_mode_auto)
|
|
|
|
*language = language_def (SYMBOL_LANGUAGE (*sym));
|
|
|
|
else
|
|
|
|
*language = current_language;
|
|
|
|
}
|
|
|
|
|
|
|
|
return PY_BT_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Helper function to extract a value from an object that conforms to
|
|
|
|
the "Symbol Value" interface. OBJ is the Python object to extract
|
|
|
|
the value from. VALUE is a pass-through argument where the value
|
|
|
|
will be written. If the object does not have the value attribute,
|
|
|
|
or provides the Python None for a value, VALUE will be set to NULL
|
|
|
|
and this function will return as successful. Returns PY_BT_ERROR
|
|
|
|
on error with the appropriate Python exception set, and PY_BT_OK on
|
|
|
|
success. */
|
|
|
|
|
|
|
|
static enum py_bt_status
|
|
|
|
extract_value (PyObject *obj, struct value **value)
|
|
|
|
{
|
|
|
|
if (PyObject_HasAttrString (obj, "value"))
|
|
|
|
{
|
|
|
|
PyObject *vresult = PyObject_CallMethod (obj, "value", NULL);
|
|
|
|
|
|
|
|
if (vresult == NULL)
|
|
|
|
return PY_BT_ERROR;
|
|
|
|
|
|
|
|
/* The Python code has returned 'None' for a value, so we set
|
|
|
|
value to NULL. This flags that GDB should read the
|
|
|
|
value. */
|
|
|
|
if (vresult == Py_None)
|
|
|
|
{
|
|
|
|
Py_DECREF (vresult);
|
|
|
|
*value = NULL;
|
|
|
|
return PY_BT_OK;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
*value = convert_value_from_python (vresult);
|
|
|
|
Py_DECREF (vresult);
|
|
|
|
|
|
|
|
if (*value == NULL)
|
|
|
|
return PY_BT_ERROR;
|
|
|
|
|
|
|
|
return PY_BT_OK;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
*value = NULL;
|
|
|
|
|
|
|
|
return PY_BT_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* MI prints only certain values according to the type of symbol and
|
|
|
|
also what the user has specified. SYM is the symbol to check, and
|
|
|
|
MI_PRINT_TYPES is an enum specifying what the user wants emitted
|
|
|
|
for the MI command in question. */
|
|
|
|
static int
|
|
|
|
mi_should_print (struct symbol *sym, enum mi_print_types type)
|
|
|
|
{
|
|
|
|
int print_me = 0;
|
|
|
|
|
|
|
|
switch (SYMBOL_CLASS (sym))
|
|
|
|
{
|
|
|
|
default:
|
|
|
|
case LOC_UNDEF: /* catches errors */
|
|
|
|
case LOC_CONST: /* constant */
|
|
|
|
case LOC_TYPEDEF: /* local typedef */
|
|
|
|
case LOC_LABEL: /* local label */
|
|
|
|
case LOC_BLOCK: /* local function */
|
|
|
|
case LOC_CONST_BYTES: /* loc. byte seq. */
|
|
|
|
case LOC_UNRESOLVED: /* unresolved static */
|
|
|
|
case LOC_OPTIMIZED_OUT: /* optimized out */
|
|
|
|
print_me = 0;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case LOC_ARG: /* argument */
|
|
|
|
case LOC_REF_ARG: /* reference arg */
|
|
|
|
case LOC_REGPARM_ADDR: /* indirect register arg */
|
|
|
|
case LOC_LOCAL: /* stack local */
|
|
|
|
case LOC_STATIC: /* static */
|
|
|
|
case LOC_REGISTER: /* register */
|
|
|
|
case LOC_COMPUTED: /* computed location */
|
|
|
|
if (type == MI_PRINT_LOCALS)
|
|
|
|
print_me = ! SYMBOL_IS_ARGUMENT (sym);
|
|
|
|
else
|
|
|
|
print_me = SYMBOL_IS_ARGUMENT (sym);
|
|
|
|
}
|
|
|
|
return print_me;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Helper function which outputs a type name extracted from VAL to a
|
|
|
|
"type" field in the output stream OUT. OUT is the ui-out structure
|
|
|
|
the type name will be output too, and VAL is the value that the
|
|
|
|
type will be extracted from. Returns PY_BT_ERROR on error, with
|
|
|
|
any GDB exceptions converted to a Python exception, or PY_BT_OK on
|
|
|
|
success. */
|
|
|
|
|
|
|
|
static enum py_bt_status
|
|
|
|
py_print_type (struct ui_out *out, struct value *val)
|
|
|
|
{
|
|
|
|
volatile struct gdb_exception except;
|
|
|
|
|
|
|
|
TRY_CATCH (except, RETURN_MASK_ALL)
|
|
|
|
{
|
|
|
|
struct type *type;
|
|
|
|
struct ui_file *stb;
|
|
|
|
struct cleanup *cleanup;
|
|
|
|
|
|
|
|
stb = mem_fileopen ();
|
|
|
|
cleanup = make_cleanup_ui_file_delete (stb);
|
|
|
|
type = check_typedef (value_type (val));
|
|
|
|
type_print (value_type (val), "", stb, -1);
|
|
|
|
ui_out_field_stream (out, "type", stb);
|
|
|
|
do_cleanups (cleanup);
|
|
|
|
}
|
|
|
|
if (except.reason < 0)
|
|
|
|
{
|
|
|
|
gdbpy_convert_exception (except);
|
|
|
|
return PY_BT_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
return PY_BT_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Helper function which outputs a value to an output field in a
|
|
|
|
stream. OUT is the ui-out structure the value will be output to,
|
|
|
|
VAL is the value that will be printed, OPTS contains the value
|
|
|
|
printing options, ARGS_TYPE is an enumerator describing the
|
|
|
|
argument format, and LANGUAGE is the language_defn that the value
|
|
|
|
will be printed with. Returns PY_BT_ERROR on error, with any GDB
|
|
|
|
exceptions converted to a Python exception, or PY_BT_OK on
|
|
|
|
success. */
|
|
|
|
|
|
|
|
static enum py_bt_status
|
|
|
|
py_print_value (struct ui_out *out, struct value *val,
|
|
|
|
const struct value_print_options *opts,
|
|
|
|
int indent,
|
|
|
|
enum py_frame_args args_type,
|
|
|
|
const struct language_defn *language)
|
|
|
|
{
|
|
|
|
int should_print = 0;
|
|
|
|
volatile struct gdb_exception except;
|
|
|
|
int local_indent = (4 * indent);
|
|
|
|
|
|
|
|
/* Never set an indent level for common_val_print if MI. */
|
|
|
|
if (ui_out_is_mi_like_p (out))
|
|
|
|
local_indent = 0;
|
|
|
|
|
|
|
|
/* MI does not print certain values, differentiated by type,
|
|
|
|
depending on what ARGS_TYPE indicates. Test type against option.
|
|
|
|
For CLI print all values. */
|
|
|
|
if (args_type == MI_PRINT_SIMPLE_VALUES
|
|
|
|
|| args_type == MI_PRINT_ALL_VALUES)
|
|
|
|
{
|
|
|
|
struct type *type = NULL;
|
|
|
|
|
|
|
|
TRY_CATCH (except, RETURN_MASK_ALL)
|
|
|
|
{
|
|
|
|
type = check_typedef (value_type (val));
|
|
|
|
}
|
|
|
|
if (except.reason < 0)
|
|
|
|
{
|
|
|
|
gdbpy_convert_exception (except);
|
|
|
|
return PY_BT_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (args_type == MI_PRINT_ALL_VALUES)
|
|
|
|
should_print = 1;
|
|
|
|
else if (args_type == MI_PRINT_SIMPLE_VALUES
|
|
|
|
&& TYPE_CODE (type) != TYPE_CODE_ARRAY
|
|
|
|
&& TYPE_CODE (type) != TYPE_CODE_STRUCT
|
|
|
|
&& TYPE_CODE (type) != TYPE_CODE_UNION)
|
|
|
|
should_print = 1;
|
|
|
|
}
|
|
|
|
else if (args_type != NO_VALUES)
|
|
|
|
should_print = 1;
|
|
|
|
|
|
|
|
if (should_print)
|
|
|
|
{
|
|
|
|
TRY_CATCH (except, RETURN_MASK_ALL)
|
|
|
|
{
|
|
|
|
struct ui_file *stb;
|
|
|
|
struct cleanup *cleanup;
|
|
|
|
|
|
|
|
stb = mem_fileopen ();
|
|
|
|
cleanup = make_cleanup_ui_file_delete (stb);
|
|
|
|
common_val_print (val, stb, indent, opts, language);
|
|
|
|
ui_out_field_stream (out, "value", stb);
|
|
|
|
do_cleanups (cleanup);
|
|
|
|
}
|
|
|
|
if (except.reason < 0)
|
|
|
|
{
|
|
|
|
gdbpy_convert_exception (except);
|
|
|
|
return PY_BT_ERROR;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return PY_BT_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Helper function to call a Python method and extract an iterator
|
|
|
|
from the result. If the function returns anything but an iterator
|
|
|
|
the exception is preserved and NULL is returned. FILTER is the
|
|
|
|
Python object to call, and FUNC is the name of the method. Returns
|
|
|
|
a PyObject, or NULL on error with the appropriate exception set.
|
|
|
|
This function can return an iterator, or NULL. */
|
|
|
|
|
|
|
|
static PyObject *
|
|
|
|
get_py_iter_from_func (PyObject *filter, char *func)
|
|
|
|
{
|
|
|
|
if (PyObject_HasAttrString (filter, func))
|
|
|
|
{
|
|
|
|
PyObject *result = PyObject_CallMethod (filter, func, NULL);
|
|
|
|
|
|
|
|
if (result != NULL)
|
|
|
|
{
|
|
|
|
if (result == Py_None)
|
|
|
|
{
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
PyObject *iterator = PyObject_GetIter (result);
|
|
|
|
|
|
|
|
Py_DECREF (result);
|
|
|
|
return iterator;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
Py_RETURN_NONE;
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Helper function to output a single frame argument and value to an
|
|
|
|
output stream. This function will account for entry values if the
|
|
|
|
FV parameter is populated, the frame argument has entry values
|
|
|
|
associated with them, and the appropriate "set entry-value"
|
|
|
|
options are set. Will output in CLI or MI like format depending
|
|
|
|
on the type of output stream detected. OUT is the output stream,
|
|
|
|
SYM_NAME is the name of the symbol. If SYM_NAME is populated then
|
|
|
|
it must have an accompanying value in the parameter FV. FA is a
|
|
|
|
frame argument structure. If FA is populated, both SYM_NAME and
|
|
|
|
FV are ignored. OPTS contains the value printing options,
|
|
|
|
ARGS_TYPE is an enumerator describing the argument format,
|
|
|
|
PRINT_ARGS_FIELD is a flag which indicates if we output "ARGS=1"
|
|
|
|
in MI output in commands where both arguments and locals are
|
|
|
|
printed. Returns PY_BT_ERROR on error, with any GDB exceptions
|
|
|
|
converted to a Python exception, or PY_BT_OK on success. */
|
|
|
|
|
|
|
|
static enum py_bt_status
|
|
|
|
py_print_single_arg (struct ui_out *out,
|
|
|
|
const char *sym_name,
|
|
|
|
struct frame_arg *fa,
|
|
|
|
struct value *fv,
|
|
|
|
const struct value_print_options *opts,
|
|
|
|
enum py_frame_args args_type,
|
|
|
|
int print_args_field,
|
|
|
|
const struct language_defn *language)
|
|
|
|
{
|
|
|
|
struct value *val;
|
|
|
|
volatile struct gdb_exception except;
|
|
|
|
|
|
|
|
if (fa != NULL)
|
|
|
|
{
|
|
|
|
language = language_def (SYMBOL_LANGUAGE (fa->sym));
|
|
|
|
val = fa->val;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
val = fv;
|
|
|
|
|
|
|
|
TRY_CATCH (except, RETURN_MASK_ALL)
|
|
|
|
{
|
|
|
|
struct cleanup *cleanups = make_cleanup (null_cleanup, NULL);
|
|
|
|
|
|
|
|
/* MI has varying rules for tuples, but generally if there is only
|
|
|
|
one element in each item in the list, do not start a tuple. The
|
|
|
|
exception is -stack-list-variables which emits an ARGS="1" field
|
|
|
|
if the value is a frame argument. This is denoted in this
|
|
|
|
function with PRINT_ARGS_FIELD which is flag from the caller to
|
|
|
|
emit the ARGS field. */
|
|
|
|
if (ui_out_is_mi_like_p (out))
|
|
|
|
{
|
|
|
|
if (print_args_field || args_type != NO_VALUES)
|
|
|
|
make_cleanup_ui_out_tuple_begin_end (out, NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
annotate_arg_begin ();
|
|
|
|
|
|
|
|
/* If frame argument is populated, check for entry-values and the
|
|
|
|
entry value options. */
|
|
|
|
if (fa != NULL)
|
|
|
|
{
|
|
|
|
struct ui_file *stb;
|
|
|
|
|
|
|
|
stb = mem_fileopen ();
|
|
|
|
make_cleanup_ui_file_delete (stb);
|
|
|
|
fprintf_symbol_filtered (stb, SYMBOL_PRINT_NAME (fa->sym),
|
|
|
|
SYMBOL_LANGUAGE (fa->sym),
|
|
|
|
DMGL_PARAMS | DMGL_ANSI);
|
|
|
|
if (fa->entry_kind == print_entry_values_compact)
|
|
|
|
{
|
|
|
|
fputs_filtered ("=", stb);
|
|
|
|
|
|
|
|
fprintf_symbol_filtered (stb, SYMBOL_PRINT_NAME (fa->sym),
|
|
|
|
SYMBOL_LANGUAGE (fa->sym),
|
|
|
|
DMGL_PARAMS | DMGL_ANSI);
|
|
|
|
}
|
|
|
|
if (fa->entry_kind == print_entry_values_only
|
|
|
|
|| fa->entry_kind == print_entry_values_compact)
|
|
|
|
{
|
|
|
|
fputs_filtered ("@entry", stb);
|
|
|
|
}
|
|
|
|
ui_out_field_stream (out, "name", stb);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
/* Otherwise, just output the name. */
|
|
|
|
ui_out_field_string (out, "name", sym_name);
|
|
|
|
|
|
|
|
annotate_arg_name_end ();
|
|
|
|
|
|
|
|
if (! ui_out_is_mi_like_p (out))
|
|
|
|
ui_out_text (out, "=");
|
|
|
|
|
|
|
|
if (print_args_field)
|
|
|
|
ui_out_field_int (out, "arg", 1);
|
|
|
|
|
|
|
|
/* For MI print the type, but only for simple values. This seems
|
|
|
|
weird, but this is how MI choose to format the various output
|
|
|
|
types. */
|
|
|
|
if (args_type == MI_PRINT_SIMPLE_VALUES)
|
|
|
|
{
|
|
|
|
if (py_print_type (out, val) == PY_BT_ERROR)
|
|
|
|
{
|
|
|
|
do_cleanups (cleanups);
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
annotate_arg_value (value_type (val));
|
|
|
|
|
|
|
|
/* If the output is to the CLI, and the user option "set print
|
|
|
|
frame-arguments" is set to none, just output "...". */
|
|
|
|
if (! ui_out_is_mi_like_p (out) && args_type == NO_VALUES)
|
|
|
|
ui_out_field_string (out, "value", "...");
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/* Otherwise, print the value for both MI and the CLI, except
|
|
|
|
for the case of MI_PRINT_NO_VALUES. */
|
|
|
|
if (args_type != NO_VALUES)
|
|
|
|
{
|
|
|
|
if (py_print_value (out, val, opts, 0, args_type, language)
|
|
|
|
== PY_BT_ERROR)
|
|
|
|
{
|
|
|
|
do_cleanups (cleanups);
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
do_cleanups (cleanups);
|
|
|
|
}
|
|
|
|
if (except.reason < 0)
|
|
|
|
{
|
|
|
|
gdbpy_convert_exception (except);
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
|
|
|
return PY_BT_OK;
|
|
|
|
|
|
|
|
error:
|
|
|
|
return PY_BT_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Helper function to loop over frame arguments provided by the
|
|
|
|
"frame_arguments" Python API. Elements in the iterator must
|
|
|
|
conform to the "Symbol Value" interface. ITER is the Python
|
|
|
|
iterable object, OUT is the output stream, ARGS_TYPE is an
|
|
|
|
enumerator describing the argument format, PRINT_ARGS_FIELD is a
|
|
|
|
flag which indicates if we output "ARGS=1" in MI output in commands
|
|
|
|
where both arguments and locals are printed, and FRAME is the
|
|
|
|
backing frame. Returns PY_BT_ERROR on error, with any GDB
|
|
|
|
exceptions converted to a Python exception, or PY_BT_OK on
|
|
|
|
success. */
|
|
|
|
|
|
|
|
static enum py_bt_status
|
|
|
|
enumerate_args (PyObject *iter,
|
|
|
|
struct ui_out *out,
|
|
|
|
enum py_frame_args args_type,
|
|
|
|
int print_args_field,
|
|
|
|
struct frame_info *frame)
|
|
|
|
{
|
|
|
|
PyObject *item;
|
|
|
|
struct value_print_options opts;
|
|
|
|
volatile struct gdb_exception except;
|
|
|
|
|
|
|
|
get_user_print_options (&opts);
|
|
|
|
|
|
|
|
if (args_type == CLI_SCALAR_VALUES)
|
|
|
|
{
|
|
|
|
/* True in "summary" mode, false otherwise. */
|
|
|
|
opts.summary = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
opts.deref_ref = 1;
|
|
|
|
|
|
|
|
TRY_CATCH (except, RETURN_MASK_ALL)
|
|
|
|
{
|
|
|
|
annotate_frame_args ();
|
|
|
|
}
|
|
|
|
if (except.reason < 0)
|
|
|
|
{
|
|
|
|
gdbpy_convert_exception (except);
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Collect the first argument outside of the loop, so output of
|
|
|
|
commas in the argument output is correct. At the end of the
|
|
|
|
loop block collect another item from the iterator, and, if it is
|
|
|
|
not null emit a comma. */
|
|
|
|
item = PyIter_Next (iter);
|
|
|
|
if (item == NULL && PyErr_Occurred ())
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
while (item)
|
|
|
|
{
|
|
|
|
const struct language_defn *language;
|
|
|
|
char *sym_name;
|
|
|
|
struct symbol *sym;
|
|
|
|
struct value *val;
|
|
|
|
enum py_bt_status success = PY_BT_ERROR;
|
|
|
|
|
|
|
|
success = extract_sym (item, &sym_name, &sym, &language);
|
|
|
|
if (success == PY_BT_ERROR)
|
|
|
|
{
|
|
|
|
Py_DECREF (item);
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
|
|
|
success = extract_value (item, &val);
|
|
|
|
if (success == PY_BT_ERROR)
|
|
|
|
{
|
|
|
|
xfree (sym_name);
|
|
|
|
Py_DECREF (item);
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
|
|
|
Py_DECREF (item);
|
|
|
|
item = NULL;
|
|
|
|
|
|
|
|
if (sym && ui_out_is_mi_like_p (out)
|
|
|
|
&& ! mi_should_print (sym, MI_PRINT_ARGS))
|
|
|
|
{
|
|
|
|
xfree (sym_name);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* If the object did not provide a value, read it using
|
|
|
|
read_frame_args and account for entry values, if any. */
|
|
|
|
if (val == NULL)
|
|
|
|
{
|
|
|
|
struct frame_arg arg, entryarg;
|
|
|
|
|
|
|
|
/* If there is no value, and also no symbol, set error and
|
|
|
|
exit. */
|
|
|
|
if (sym == NULL)
|
|
|
|
{
|
|
|
|
PyErr_SetString (PyExc_RuntimeError,
|
|
|
|
_("No symbol or value provided."));
|
|
|
|
xfree (sym_name);
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
|
|
|
TRY_CATCH (except, RETURN_MASK_ALL)
|
|
|
|
{
|
|
|
|
read_frame_arg (sym, frame, &arg, &entryarg);
|
|
|
|
}
|
|
|
|
if (except.reason < 0)
|
|
|
|
{
|
|
|
|
xfree (sym_name);
|
|
|
|
gdbpy_convert_exception (except);
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* The object has not provided a value, so this is a frame
|
|
|
|
argument to be read by GDB. In this case we have to
|
|
|
|
account for entry-values. */
|
|
|
|
|
|
|
|
if (arg.entry_kind != print_entry_values_only)
|
|
|
|
{
|
|
|
|
if (py_print_single_arg (out, NULL, &arg,
|
|
|
|
NULL, &opts,
|
|
|
|
args_type,
|
|
|
|
print_args_field,
|
|
|
|
NULL) == PY_BT_ERROR)
|
|
|
|
{
|
|
|
|
xfree (arg.error);
|
|
|
|
xfree (entryarg.error);
|
|
|
|
xfree (sym_name);
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (entryarg.entry_kind != print_entry_values_no)
|
|
|
|
{
|
|
|
|
if (arg.entry_kind != print_entry_values_only)
|
|
|
|
{
|
|
|
|
TRY_CATCH (except, RETURN_MASK_ALL)
|
|
|
|
{
|
|
|
|
ui_out_text (out, ", ");
|
|
|
|
ui_out_wrap_hint (out, " ");
|
|
|
|
}
|
|
|
|
if (except.reason < 0)
|
|
|
|
{
|
|
|
|
xfree (arg.error);
|
|
|
|
xfree (entryarg.error);
|
|
|
|
xfree (sym_name);
|
|
|
|
gdbpy_convert_exception (except);
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (py_print_single_arg (out, NULL, &entryarg, NULL,
|
|
|
|
&opts, args_type,
|
|
|
|
print_args_field, NULL) == PY_BT_ERROR)
|
|
|
|
{
|
|
|
|
xfree (arg.error);
|
|
|
|
xfree (entryarg.error);
|
|
|
|
xfree (sym_name);
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
xfree (arg.error);
|
|
|
|
xfree (entryarg.error);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/* If the object has provided a value, we just print that. */
|
|
|
|
if (val != NULL)
|
|
|
|
{
|
|
|
|
if (py_print_single_arg (out, sym_name, NULL, val, &opts,
|
|
|
|
args_type, print_args_field,
|
|
|
|
language) == PY_BT_ERROR)
|
|
|
|
{
|
|
|
|
xfree (sym_name);
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
xfree (sym_name);
|
|
|
|
|
|
|
|
/* Collect the next item from the iterator. If
|
|
|
|
this is the last item, do not print the
|
|
|
|
comma. */
|
|
|
|
item = PyIter_Next (iter);
|
|
|
|
if (item != NULL)
|
|
|
|
{
|
|
|
|
TRY_CATCH (except, RETURN_MASK_ALL)
|
|
|
|
{
|
|
|
|
ui_out_text (out, ", ");
|
|
|
|
}
|
|
|
|
if (except.reason < 0)
|
|
|
|
{
|
|
|
|
Py_DECREF (item);
|
|
|
|
gdbpy_convert_exception (except);
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (PyErr_Occurred ())
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
TRY_CATCH (except, RETURN_MASK_ALL)
|
|
|
|
{
|
|
|
|
annotate_arg_end ();
|
|
|
|
}
|
|
|
|
if (except.reason < 0)
|
|
|
|
{
|
|
|
|
Py_DECREF (item);
|
|
|
|
gdbpy_convert_exception (except);
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return PY_BT_OK;
|
|
|
|
|
|
|
|
error:
|
|
|
|
return PY_BT_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* Helper function to loop over variables provided by the
|
|
|
|
"frame_locals" Python API. Elements in the iterable must conform
|
|
|
|
to the "Symbol Value" interface. ITER is the Python iterable
|
|
|
|
object, OUT is the output stream, INDENT is whether we should
|
|
|
|
indent the output (for CLI), ARGS_TYPE is an enumerator describing
|
|
|
|
the argument format, PRINT_ARGS_FIELD is flag which indicates
|
|
|
|
whether to output the ARGS field in the case of
|
|
|
|
-stack-list-variables and FRAME is the backing frame. Returns
|
|
|
|
PY_BT_ERROR on error, with any GDB exceptions converted to a Python
|
|
|
|
exception, or PY_BT_OK on success. */
|
|
|
|
|
|
|
|
static enum py_bt_status
|
|
|
|
enumerate_locals (PyObject *iter,
|
|
|
|
struct ui_out *out,
|
|
|
|
int indent,
|
|
|
|
enum py_frame_args args_type,
|
|
|
|
int print_args_field,
|
|
|
|
struct frame_info *frame)
|
|
|
|
{
|
|
|
|
PyObject *item;
|
|
|
|
struct value_print_options opts;
|
|
|
|
|
|
|
|
get_user_print_options (&opts);
|
|
|
|
opts.deref_ref = 1;
|
|
|
|
|
|
|
|
while ((item = PyIter_Next (iter)))
|
|
|
|
{
|
|
|
|
const struct language_defn *language;
|
|
|
|
char *sym_name;
|
|
|
|
struct value *val;
|
|
|
|
enum py_bt_status success = PY_BT_ERROR;
|
|
|
|
struct symbol *sym;
|
|
|
|
volatile struct gdb_exception except;
|
|
|
|
int local_indent = 8 + (8 * indent);
|
|
|
|
struct cleanup *locals_cleanups;
|
|
|
|
|
|
|
|
locals_cleanups = make_cleanup_py_decref (item);
|
|
|
|
|
|
|
|
success = extract_sym (item, &sym_name, &sym, &language);
|
|
|
|
if (success == PY_BT_ERROR)
|
|
|
|
{
|
|
|
|
do_cleanups (locals_cleanups);
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
|
|
|
make_cleanup (xfree, sym_name);
|
|
|
|
|
|
|
|
success = extract_value (item, &val);
|
|
|
|
if (success == PY_BT_ERROR)
|
|
|
|
{
|
|
|
|
do_cleanups (locals_cleanups);
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (sym != NULL && ui_out_is_mi_like_p (out)
|
|
|
|
&& ! mi_should_print (sym, MI_PRINT_LOCALS))
|
|
|
|
{
|
|
|
|
do_cleanups (locals_cleanups);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* If the object did not provide a value, read it. */
|
|
|
|
if (val == NULL)
|
|
|
|
{
|
|
|
|
TRY_CATCH (except, RETURN_MASK_ALL)
|
|
|
|
{
|
|
|
|
val = read_var_value (sym, frame);
|
|
|
|
}
|
|
|
|
if (except.reason < 0)
|
|
|
|
{
|
|
|
|
gdbpy_convert_exception (except);
|
|
|
|
do_cleanups (locals_cleanups);
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* With PRINT_NO_VALUES, MI does not emit a tuple normally as
|
|
|
|
each output contains only one field. The exception is
|
|
|
|
-stack-list-variables, which always provides a tuple. */
|
|
|
|
if (ui_out_is_mi_like_p (out))
|
|
|
|
{
|
|
|
|
if (print_args_field || args_type != NO_VALUES)
|
|
|
|
make_cleanup_ui_out_tuple_begin_end (out, NULL);
|
|
|
|
}
|
|
|
|
TRY_CATCH (except, RETURN_MASK_ALL)
|
|
|
|
{
|
|
|
|
if (! ui_out_is_mi_like_p (out))
|
|
|
|
{
|
|
|
|
/* If the output is not MI we indent locals. */
|
|
|
|
ui_out_spaces (out, local_indent);
|
|
|
|
}
|
|
|
|
|
|
|
|
ui_out_field_string (out, "name", sym_name);
|
|
|
|
|
|
|
|
if (! ui_out_is_mi_like_p (out))
|
|
|
|
ui_out_text (out, " = ");
|
|
|
|
}
|
|
|
|
if (except.reason < 0)
|
|
|
|
{
|
|
|
|
gdbpy_convert_exception (except);
|
|
|
|
do_cleanups (locals_cleanups);
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (args_type == MI_PRINT_SIMPLE_VALUES)
|
|
|
|
{
|
|
|
|
if (py_print_type (out, val) == PY_BT_ERROR)
|
|
|
|
{
|
|
|
|
do_cleanups (locals_cleanups);
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* CLI always prints values for locals. MI uses the
|
|
|
|
simple/no/all system. */
|
|
|
|
if (! ui_out_is_mi_like_p (out))
|
|
|
|
{
|
|
|
|
int val_indent = (indent + 1) * 4;
|
|
|
|
|
|
|
|
if (py_print_value (out, val, &opts, val_indent, args_type,
|
|
|
|
language) == PY_BT_ERROR)
|
|
|
|
{
|
|
|
|
do_cleanups (locals_cleanups);
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (args_type != NO_VALUES)
|
|
|
|
{
|
|
|
|
if (py_print_value (out, val, &opts, 0, args_type,
|
|
|
|
language) == PY_BT_ERROR)
|
|
|
|
{
|
|
|
|
do_cleanups (locals_cleanups);
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
do_cleanups (locals_cleanups);
|
|
|
|
|
|
|
|
TRY_CATCH (except, RETURN_MASK_ALL)
|
|
|
|
{
|
|
|
|
ui_out_text (out, "\n");
|
|
|
|
}
|
|
|
|
if (except.reason < 0)
|
|
|
|
{
|
|
|
|
gdbpy_convert_exception (except);
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (item == NULL && PyErr_Occurred ())
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
return PY_BT_OK;
|
|
|
|
|
|
|
|
error:
|
|
|
|
return PY_BT_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Helper function for -stack-list-variables. Returns PY_BT_ERROR on
|
|
|
|
error, or PY_BT_OK on success. */
|
|
|
|
|
|
|
|
static enum py_bt_status
|
|
|
|
py_mi_print_variables (PyObject *filter, struct ui_out *out,
|
|
|
|
struct value_print_options *opts,
|
|
|
|
enum py_frame_args args_type,
|
|
|
|
struct frame_info *frame)
|
|
|
|
{
|
|
|
|
struct cleanup *old_chain;
|
|
|
|
PyObject *args_iter;
|
|
|
|
PyObject *locals_iter;
|
|
|
|
|
|
|
|
args_iter = get_py_iter_from_func (filter, "frame_args");
|
|
|
|
old_chain = make_cleanup_py_xdecref (args_iter);
|
|
|
|
if (args_iter == NULL)
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
locals_iter = get_py_iter_from_func (filter, "frame_locals");
|
|
|
|
if (locals_iter == NULL)
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
make_cleanup_py_decref (locals_iter);
|
|
|
|
make_cleanup_ui_out_list_begin_end (out, "variables");
|
|
|
|
|
|
|
|
if (args_iter != Py_None)
|
|
|
|
if (enumerate_args (args_iter, out, args_type, 1, frame) == PY_BT_ERROR)
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
if (locals_iter != Py_None)
|
|
|
|
if (enumerate_locals (locals_iter, out, 1, args_type, 1, frame)
|
|
|
|
== PY_BT_ERROR)
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
do_cleanups (old_chain);
|
|
|
|
return PY_BT_OK;
|
|
|
|
|
|
|
|
error:
|
|
|
|
do_cleanups (old_chain);
|
|
|
|
return PY_BT_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Helper function for printing locals. This function largely just
|
|
|
|
creates the wrapping tuple, and calls enumerate_locals. Returns
|
|
|
|
PY_BT_ERROR on error, or PY_BT_OK on success.*/
|
|
|
|
|
|
|
|
static enum py_bt_status
|
|
|
|
py_print_locals (PyObject *filter,
|
|
|
|
struct ui_out *out,
|
|
|
|
enum py_frame_args args_type,
|
|
|
|
int indent,
|
|
|
|
struct frame_info *frame)
|
|
|
|
{
|
|
|
|
PyObject *locals_iter = get_py_iter_from_func (filter,
|
|
|
|
"frame_locals");
|
|
|
|
struct cleanup *old_chain = make_cleanup_py_xdecref (locals_iter);
|
|
|
|
|
|
|
|
if (locals_iter == NULL)
|
|
|
|
goto locals_error;
|
|
|
|
|
|
|
|
make_cleanup_ui_out_list_begin_end (out, "locals");
|
|
|
|
|
|
|
|
if (locals_iter != Py_None)
|
|
|
|
if (enumerate_locals (locals_iter, out, indent, args_type,
|
|
|
|
0, frame) == PY_BT_ERROR)
|
|
|
|
goto locals_error;
|
|
|
|
|
|
|
|
do_cleanups (old_chain);
|
|
|
|
return PY_BT_OK;;
|
|
|
|
|
|
|
|
locals_error:
|
|
|
|
do_cleanups (old_chain);
|
|
|
|
return PY_BT_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Helper function for printing frame arguments. This function
|
|
|
|
largely just creates the wrapping tuple, and calls enumerate_args.
|
|
|
|
Returns PY_BT_ERROR on error, with any GDB exceptions converted to
|
|
|
|
a Python exception, or PY_BT_OK on success. */
|
|
|
|
|
|
|
|
static enum py_bt_status
|
|
|
|
py_print_args (PyObject *filter,
|
|
|
|
struct ui_out *out,
|
|
|
|
enum py_frame_args args_type,
|
|
|
|
struct frame_info *frame)
|
|
|
|
{
|
|
|
|
PyObject *args_iter = get_py_iter_from_func (filter, "frame_args");
|
|
|
|
struct cleanup *old_chain = make_cleanup_py_xdecref (args_iter);
|
|
|
|
volatile struct gdb_exception except;
|
|
|
|
|
|
|
|
if (args_iter == NULL)
|
|
|
|
goto args_error;
|
|
|
|
|
|
|
|
make_cleanup_ui_out_list_begin_end (out, "args");
|
|
|
|
|
|
|
|
TRY_CATCH (except, RETURN_MASK_ALL)
|
|
|
|
{
|
|
|
|
annotate_frame_args ();
|
|
|
|
if (! ui_out_is_mi_like_p (out))
|
|
|
|
ui_out_text (out, " (");
|
|
|
|
}
|
|
|
|
if (except.reason < 0)
|
|
|
|
{
|
|
|
|
gdbpy_convert_exception (except);
|
|
|
|
goto args_error;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (args_iter != Py_None)
|
|
|
|
if (enumerate_args (args_iter, out, args_type, 0, frame) == PY_BT_ERROR)
|
|
|
|
goto args_error;
|
|
|
|
|
|
|
|
TRY_CATCH (except, RETURN_MASK_ALL)
|
|
|
|
{
|
|
|
|
if (! ui_out_is_mi_like_p (out))
|
|
|
|
ui_out_text (out, ")");
|
|
|
|
}
|
|
|
|
if (except.reason < 0)
|
|
|
|
{
|
|
|
|
gdbpy_convert_exception (except);
|
|
|
|
goto args_error;
|
|
|
|
}
|
|
|
|
|
|
|
|
do_cleanups (old_chain);
|
|
|
|
return PY_BT_OK;
|
|
|
|
|
|
|
|
args_error:
|
|
|
|
do_cleanups (old_chain);
|
|
|
|
return PY_BT_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Print a single frame to the designated output stream, detecting
|
|
|
|
whether the output is MI or console, and formatting the output
|
|
|
|
according to the conventions of that protocol. FILTER is the
|
|
|
|
frame-filter associated with this frame. FLAGS is an integer
|
|
|
|
describing the various print options. The FLAGS variables is
|
|
|
|
described in "apply_frame_filter" function. ARGS_TYPE is an
|
|
|
|
enumerator describing the argument format. OUT is the output
|
|
|
|
stream to print, INDENT is the level of indention for this frame
|
|
|
|
(in the case of elided frames), and LEVELS_PRINTED is a hash-table
|
|
|
|
containing all the frames level that have already been printed.
|
|
|
|
If a frame level has been printed, do not print it again (in the
|
|
|
|
case of elided frames). Returns PY_BT_ERROR on error, with any
|
|
|
|
GDB exceptions converted to a Python exception, or PY_BT_COMPLETED
|
|
|
|
on success. */
|
|
|
|
|
|
|
|
static enum py_bt_status
|
|
|
|
py_print_frame (PyObject *filter, int flags, enum py_frame_args args_type,
|
|
|
|
struct ui_out *out, int indent, htab_t levels_printed)
|
|
|
|
{
|
|
|
|
int has_addr = 0;
|
|
|
|
CORE_ADDR address = 0;
|
|
|
|
struct gdbarch *gdbarch = NULL;
|
|
|
|
struct frame_info *frame = NULL;
|
|
|
|
struct cleanup *cleanup_stack = make_cleanup (null_cleanup, NULL);
|
|
|
|
struct value_print_options opts;
|
|
|
|
PyObject *py_inf_frame, *elided;
|
|
|
|
int print_level, print_frame_info, print_args, print_locals;
|
|
|
|
volatile struct gdb_exception except;
|
|
|
|
|
|
|
|
/* Extract print settings from FLAGS. */
|
|
|
|
print_level = (flags & PRINT_LEVEL) ? 1 : 0;
|
|
|
|
print_frame_info = (flags & PRINT_FRAME_INFO) ? 1 : 0;
|
|
|
|
print_args = (flags & PRINT_ARGS) ? 1 : 0;
|
|
|
|
print_locals = (flags & PRINT_LOCALS) ? 1 : 0;
|
|
|
|
|
|
|
|
get_user_print_options (&opts);
|
|
|
|
|
|
|
|
/* Get the underlying frame. This is needed to determine GDB
|
|
|
|
architecture, and also, in the cases of frame variables/arguments to
|
|
|
|
read them if they returned filter object requires us to do so. */
|
|
|
|
py_inf_frame = PyObject_CallMethod (filter, "inferior_frame", NULL);
|
|
|
|
if (py_inf_frame == NULL)
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
frame = frame_object_to_frame_info (py_inf_frame);;
|
|
|
|
|
|
|
|
Py_DECREF (py_inf_frame);
|
|
|
|
|
|
|
|
if (frame == NULL)
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
TRY_CATCH (except, RETURN_MASK_ALL)
|
|
|
|
{
|
|
|
|
gdbarch = get_frame_arch (frame);
|
|
|
|
}
|
|
|
|
if (except.reason < 0)
|
|
|
|
{
|
|
|
|
gdbpy_convert_exception (except);
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* stack-list-variables. */
|
|
|
|
if (print_locals && print_args && ! print_frame_info)
|
|
|
|
{
|
|
|
|
if (py_mi_print_variables (filter, out, &opts,
|
|
|
|
args_type, frame) == PY_BT_ERROR)
|
|
|
|
goto error;
|
|
|
|
else
|
|
|
|
{
|
|
|
|
do_cleanups (cleanup_stack);
|
|
|
|
return PY_BT_COMPLETED;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* -stack-list-locals does not require a
|
|
|
|
wrapping frame attribute. */
|
|
|
|
if (print_frame_info || (print_args && ! print_locals))
|
|
|
|
make_cleanup_ui_out_tuple_begin_end (out, "frame");
|
|
|
|
|
|
|
|
if (print_frame_info)
|
|
|
|
{
|
|
|
|
/* Elided frames are also printed with this function (recursively)
|
|
|
|
and are printed with indention. */
|
|
|
|
if (indent > 0)
|
|
|
|
{
|
|
|
|
TRY_CATCH (except, RETURN_MASK_ALL)
|
|
|
|
{
|
|
|
|
ui_out_spaces (out, indent*4);
|
|
|
|
}
|
|
|
|
if (except.reason < 0)
|
|
|
|
{
|
|
|
|
gdbpy_convert_exception (except);
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* The address is required for frame annotations, and also for
|
|
|
|
address printing. */
|
|
|
|
if (PyObject_HasAttrString (filter, "address"))
|
|
|
|
{
|
|
|
|
PyObject *paddr = PyObject_CallMethod (filter, "address", NULL);
|
|
|
|
if (paddr != NULL)
|
|
|
|
{
|
|
|
|
if (paddr != Py_None)
|
|
|
|
{
|
|
|
|
address = PyLong_AsLong (paddr);
|
|
|
|
has_addr = 1;
|
|
|
|
}
|
|
|
|
Py_DECREF (paddr);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Print frame level. MI does not require the level if
|
|
|
|
locals/variables only are being printed. */
|
|
|
|
if ((print_frame_info || print_args) && print_level)
|
|
|
|
{
|
|
|
|
struct frame_info **slot;
|
|
|
|
int level;
|
|
|
|
volatile struct gdb_exception except;
|
|
|
|
|
|
|
|
slot = (struct frame_info **) htab_find_slot (levels_printed,
|
|
|
|
frame, INSERT);
|
|
|
|
TRY_CATCH (except, RETURN_MASK_ALL)
|
|
|
|
{
|
|
|
|
level = frame_relative_level (frame);
|
|
|
|
|
|
|
|
/* Check if this frame has already been printed (there are cases
|
|
|
|
where elided synthetic dummy-frames have to 'borrow' the frame
|
|
|
|
architecture from the eliding frame. If that is the case, do
|
|
|
|
not print 'level', but print spaces. */
|
|
|
|
if (*slot == frame)
|
|
|
|
ui_out_field_skip (out, "level");
|
|
|
|
else
|
|
|
|
{
|
|
|
|
*slot = frame;
|
|
|
|
annotate_frame_begin (print_level ? level : 0,
|
|
|
|
gdbarch, address);
|
|
|
|
ui_out_text (out, "#");
|
|
|
|
ui_out_field_fmt_int (out, 2, ui_left, "level",
|
|
|
|
level);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (except.reason < 0)
|
|
|
|
{
|
|
|
|
gdbpy_convert_exception (except);
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (print_frame_info)
|
|
|
|
{
|
|
|
|
/* Print address to the address field. If an address is not provided,
|
|
|
|
print nothing. */
|
|
|
|
if (opts.addressprint && has_addr)
|
|
|
|
{
|
|
|
|
TRY_CATCH (except, RETURN_MASK_ALL)
|
|
|
|
{
|
|
|
|
annotate_frame_address ();
|
|
|
|
ui_out_field_core_addr (out, "addr", gdbarch, address);
|
|
|
|
annotate_frame_address_end ();
|
|
|
|
ui_out_text (out, " in ");
|
|
|
|
}
|
|
|
|
if (except.reason < 0)
|
|
|
|
{
|
|
|
|
gdbpy_convert_exception (except);
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Print frame function name. */
|
|
|
|
if (PyObject_HasAttrString (filter, "function"))
|
|
|
|
{
|
|
|
|
PyObject *py_func = PyObject_CallMethod (filter, "function", NULL);
|
|
|
|
|
|
|
|
if (py_func != NULL)
|
|
|
|
{
|
|
|
|
const char *function = NULL;
|
|
|
|
|
|
|
|
if (gdbpy_is_string (py_func))
|
|
|
|
{
|
2013-08-29 10:06:18 +00:00
|
|
|
char *function_to_free = NULL;
|
|
|
|
|
|
|
|
function = function_to_free =
|
|
|
|
python_string_to_host_string (py_func);
|
2013-05-10 Phil Muldoon <pmuldoon@redhat.com>
* stack.c (backtrace_command_1): Add "no-filters", and Python frame
filter logic.
(backtrace_command): Add "no-filters" option parsing.
(_initialize_stack): Alter help to reflect "no-filters" option.
* Makefile.in (SUBDIR_PYTHON_OBS): Add py-framefilter.o
(SUBDIR_PYTHON_SRCS): Add py-framefilter.c
(py-frame.o): Add target
* data-directory/Makefile.in (PYTHON_DIR): Add Python frame
filter files.
* python/python.h: Add new frame filter constants, and flag enum.
(apply_frame_filter): Add definition.
* python/python.c (apply_frame_filter): New non-Python
enabled function.
* python/py-utils.c (py_xdecref): New function.
(make_cleanup_py_xdecref): Ditto.
* python/py-objfile.c: Declare frame_filters dictionary.
(objfpy_dealloc): Add frame_filters dealloc.
(objfpy_new): Initialize frame_filters attribute.
(objfile_to_objfile_object): Ditto.
(objfpy_get_frame_filters): New function.
(objfpy_set_frame_filters): New function.
* python/py-progspace.c: Declare frame_filters dictionary.
(pspy_dealloc): Add frame_filters dealloc.
(pspy_new): Initialize frame_filters attribute.
(pspacee_to_pspace_object): Ditto.
(pspy_get_frame_filters): New function.
(pspy_set_frame_filters): New function.
* python/py-framefilter.c: New file.
* python/lib/gdb/command/frame_filters.py: New file.
* python/lib/gdb/frames.py: New file.
* python/lib/gdb/__init__.py: Initialize global frame_filters
dictionary
* python/lib/gdb/FrameDecorator.py: New file.
* python/lib/gdb/FrameIterator.py: New file.
* mi/mi-cmds.c (mi_cmds): Add frame filters command.
* mi/mi-cmds.h: Declare.
* mi/mi-cmd-stack.c (mi_cmd_stack_list_frames): Add
--no-frame-filter logic, and Python frame filter logic.
(stack_enable_frame_filters): New function.
(parse_no_frame_option): Ditto.
(mi_cmd_stack_list_frames): Add --no-frame-filter and Python frame
filter logic.
(mi_cmd_stack_list_locals): Ditto.
(mi_cmd_stack_list_args): Ditto.
(mi_cmd_stack_list_variables): Ditto.
* NEWS: Add frame filter note.
2013-05-10 Phil Muldoon <pmuldoon@redhat.com>
* gdb.python/py-framefilter.py: New File.
* gdb.python/py-framefilter-mi.exp: Ditto.
* gdb.python/py-framefilter.c: Ditto.
* gdb.python/py-framefilter-mi.exp: Ditto.
* gdb.python/py-framefilter-mi.c: Ditto,
* gdb.python/py-framefilter-gdb.py.in: Ditto.
2013-05-10 Phil Muldoon <pmuldoon@redhat.com>
* gdb.texinfo (Backtrace): Add "no-filter" argument.
(Python API): Add Frame Filters API, Frame Wrapper API,
Writing a Frame Filter/Wrapper, Managing Management of Frame
Filters chapter entries.
(Frame Filters API): New Node.
(Frame Wrapper API): New Node.
(Writing a Frame Filter): New Node.
(Managing Frame Filters): New Node.
(Progspaces In Python): Add note about frame_filters attribute.
(Objfiles in Python): Ditto.
(GDB/MI Stack Manipulation): Add -enable-frame-filters command,
@anchors and --no-frame-filters option to -stack-list-variables,
-stack-list-frames, -stack-list-locals and -stack-list-arguments
commands.
2013-05-10 10:26:03 +00:00
|
|
|
|
|
|
|
if (function == NULL)
|
|
|
|
{
|
|
|
|
Py_DECREF (py_func);
|
|
|
|
goto error;
|
|
|
|
}
|
2013-08-29 10:06:18 +00:00
|
|
|
make_cleanup (xfree, function_to_free);
|
2013-05-10 Phil Muldoon <pmuldoon@redhat.com>
* stack.c (backtrace_command_1): Add "no-filters", and Python frame
filter logic.
(backtrace_command): Add "no-filters" option parsing.
(_initialize_stack): Alter help to reflect "no-filters" option.
* Makefile.in (SUBDIR_PYTHON_OBS): Add py-framefilter.o
(SUBDIR_PYTHON_SRCS): Add py-framefilter.c
(py-frame.o): Add target
* data-directory/Makefile.in (PYTHON_DIR): Add Python frame
filter files.
* python/python.h: Add new frame filter constants, and flag enum.
(apply_frame_filter): Add definition.
* python/python.c (apply_frame_filter): New non-Python
enabled function.
* python/py-utils.c (py_xdecref): New function.
(make_cleanup_py_xdecref): Ditto.
* python/py-objfile.c: Declare frame_filters dictionary.
(objfpy_dealloc): Add frame_filters dealloc.
(objfpy_new): Initialize frame_filters attribute.
(objfile_to_objfile_object): Ditto.
(objfpy_get_frame_filters): New function.
(objfpy_set_frame_filters): New function.
* python/py-progspace.c: Declare frame_filters dictionary.
(pspy_dealloc): Add frame_filters dealloc.
(pspy_new): Initialize frame_filters attribute.
(pspacee_to_pspace_object): Ditto.
(pspy_get_frame_filters): New function.
(pspy_set_frame_filters): New function.
* python/py-framefilter.c: New file.
* python/lib/gdb/command/frame_filters.py: New file.
* python/lib/gdb/frames.py: New file.
* python/lib/gdb/__init__.py: Initialize global frame_filters
dictionary
* python/lib/gdb/FrameDecorator.py: New file.
* python/lib/gdb/FrameIterator.py: New file.
* mi/mi-cmds.c (mi_cmds): Add frame filters command.
* mi/mi-cmds.h: Declare.
* mi/mi-cmd-stack.c (mi_cmd_stack_list_frames): Add
--no-frame-filter logic, and Python frame filter logic.
(stack_enable_frame_filters): New function.
(parse_no_frame_option): Ditto.
(mi_cmd_stack_list_frames): Add --no-frame-filter and Python frame
filter logic.
(mi_cmd_stack_list_locals): Ditto.
(mi_cmd_stack_list_args): Ditto.
(mi_cmd_stack_list_variables): Ditto.
* NEWS: Add frame filter note.
2013-05-10 Phil Muldoon <pmuldoon@redhat.com>
* gdb.python/py-framefilter.py: New File.
* gdb.python/py-framefilter-mi.exp: Ditto.
* gdb.python/py-framefilter.c: Ditto.
* gdb.python/py-framefilter-mi.exp: Ditto.
* gdb.python/py-framefilter-mi.c: Ditto,
* gdb.python/py-framefilter-gdb.py.in: Ditto.
2013-05-10 Phil Muldoon <pmuldoon@redhat.com>
* gdb.texinfo (Backtrace): Add "no-filter" argument.
(Python API): Add Frame Filters API, Frame Wrapper API,
Writing a Frame Filter/Wrapper, Managing Management of Frame
Filters chapter entries.
(Frame Filters API): New Node.
(Frame Wrapper API): New Node.
(Writing a Frame Filter): New Node.
(Managing Frame Filters): New Node.
(Progspaces In Python): Add note about frame_filters attribute.
(Objfiles in Python): Ditto.
(GDB/MI Stack Manipulation): Add -enable-frame-filters command,
@anchors and --no-frame-filters option to -stack-list-variables,
-stack-list-frames, -stack-list-locals and -stack-list-arguments
commands.
2013-05-10 10:26:03 +00:00
|
|
|
}
|
|
|
|
else if (PyLong_Check (py_func))
|
|
|
|
{
|
|
|
|
CORE_ADDR addr = PyLong_AsUnsignedLongLong (py_func);
|
|
|
|
struct bound_minimal_symbol msymbol;
|
|
|
|
|
|
|
|
if (PyErr_Occurred ())
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
msymbol = lookup_minimal_symbol_by_pc (addr);
|
|
|
|
if (msymbol.minsym != NULL)
|
|
|
|
function = SYMBOL_PRINT_NAME (msymbol.minsym);
|
|
|
|
}
|
|
|
|
else if (py_func != Py_None)
|
|
|
|
{
|
|
|
|
PyErr_SetString (PyExc_RuntimeError,
|
|
|
|
_("FrameDecorator.function: expecting a " \
|
|
|
|
"String, integer or None."));
|
|
|
|
Py_DECREF (py_func);
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
TRY_CATCH (except, RETURN_MASK_ALL)
|
|
|
|
{
|
|
|
|
annotate_frame_function_name ();
|
|
|
|
if (function == NULL)
|
|
|
|
ui_out_field_skip (out, "func");
|
|
|
|
else
|
|
|
|
ui_out_field_string (out, "func", function);
|
|
|
|
}
|
|
|
|
if (except.reason < 0)
|
|
|
|
{
|
|
|
|
Py_DECREF (py_func);
|
|
|
|
gdbpy_convert_exception (except);
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Py_DECREF (py_func);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* Frame arguments. Check the result, and error if something went
|
|
|
|
wrong. */
|
|
|
|
if (print_args)
|
|
|
|
{
|
|
|
|
if (py_print_args (filter, out, args_type, frame) == PY_BT_ERROR)
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* File name/source/line number information. */
|
|
|
|
if (print_frame_info)
|
|
|
|
{
|
|
|
|
TRY_CATCH (except, RETURN_MASK_ALL)
|
|
|
|
{
|
|
|
|
annotate_frame_source_begin ();
|
|
|
|
}
|
|
|
|
if (except.reason < 0)
|
|
|
|
{
|
|
|
|
gdbpy_convert_exception (except);
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (PyObject_HasAttrString (filter, "filename"))
|
|
|
|
{
|
|
|
|
PyObject *py_fn = PyObject_CallMethod (filter, "filename",
|
|
|
|
NULL);
|
|
|
|
if (py_fn != NULL)
|
|
|
|
{
|
|
|
|
if (py_fn != Py_None)
|
|
|
|
{
|
2013-08-29 10:06:18 +00:00
|
|
|
char *filename = python_string_to_host_string (py_fn);
|
2013-05-10 Phil Muldoon <pmuldoon@redhat.com>
* stack.c (backtrace_command_1): Add "no-filters", and Python frame
filter logic.
(backtrace_command): Add "no-filters" option parsing.
(_initialize_stack): Alter help to reflect "no-filters" option.
* Makefile.in (SUBDIR_PYTHON_OBS): Add py-framefilter.o
(SUBDIR_PYTHON_SRCS): Add py-framefilter.c
(py-frame.o): Add target
* data-directory/Makefile.in (PYTHON_DIR): Add Python frame
filter files.
* python/python.h: Add new frame filter constants, and flag enum.
(apply_frame_filter): Add definition.
* python/python.c (apply_frame_filter): New non-Python
enabled function.
* python/py-utils.c (py_xdecref): New function.
(make_cleanup_py_xdecref): Ditto.
* python/py-objfile.c: Declare frame_filters dictionary.
(objfpy_dealloc): Add frame_filters dealloc.
(objfpy_new): Initialize frame_filters attribute.
(objfile_to_objfile_object): Ditto.
(objfpy_get_frame_filters): New function.
(objfpy_set_frame_filters): New function.
* python/py-progspace.c: Declare frame_filters dictionary.
(pspy_dealloc): Add frame_filters dealloc.
(pspy_new): Initialize frame_filters attribute.
(pspacee_to_pspace_object): Ditto.
(pspy_get_frame_filters): New function.
(pspy_set_frame_filters): New function.
* python/py-framefilter.c: New file.
* python/lib/gdb/command/frame_filters.py: New file.
* python/lib/gdb/frames.py: New file.
* python/lib/gdb/__init__.py: Initialize global frame_filters
dictionary
* python/lib/gdb/FrameDecorator.py: New file.
* python/lib/gdb/FrameIterator.py: New file.
* mi/mi-cmds.c (mi_cmds): Add frame filters command.
* mi/mi-cmds.h: Declare.
* mi/mi-cmd-stack.c (mi_cmd_stack_list_frames): Add
--no-frame-filter logic, and Python frame filter logic.
(stack_enable_frame_filters): New function.
(parse_no_frame_option): Ditto.
(mi_cmd_stack_list_frames): Add --no-frame-filter and Python frame
filter logic.
(mi_cmd_stack_list_locals): Ditto.
(mi_cmd_stack_list_args): Ditto.
(mi_cmd_stack_list_variables): Ditto.
* NEWS: Add frame filter note.
2013-05-10 Phil Muldoon <pmuldoon@redhat.com>
* gdb.python/py-framefilter.py: New File.
* gdb.python/py-framefilter-mi.exp: Ditto.
* gdb.python/py-framefilter.c: Ditto.
* gdb.python/py-framefilter-mi.exp: Ditto.
* gdb.python/py-framefilter-mi.c: Ditto,
* gdb.python/py-framefilter-gdb.py.in: Ditto.
2013-05-10 Phil Muldoon <pmuldoon@redhat.com>
* gdb.texinfo (Backtrace): Add "no-filter" argument.
(Python API): Add Frame Filters API, Frame Wrapper API,
Writing a Frame Filter/Wrapper, Managing Management of Frame
Filters chapter entries.
(Frame Filters API): New Node.
(Frame Wrapper API): New Node.
(Writing a Frame Filter): New Node.
(Managing Frame Filters): New Node.
(Progspaces In Python): Add note about frame_filters attribute.
(Objfiles in Python): Ditto.
(GDB/MI Stack Manipulation): Add -enable-frame-filters command,
@anchors and --no-frame-filters option to -stack-list-variables,
-stack-list-frames, -stack-list-locals and -stack-list-arguments
commands.
2013-05-10 10:26:03 +00:00
|
|
|
|
|
|
|
if (filename == NULL)
|
|
|
|
{
|
|
|
|
Py_DECREF (py_fn);
|
|
|
|
goto error;
|
|
|
|
}
|
2013-08-29 10:06:18 +00:00
|
|
|
|
|
|
|
make_cleanup (xfree, filename);
|
2013-05-10 Phil Muldoon <pmuldoon@redhat.com>
* stack.c (backtrace_command_1): Add "no-filters", and Python frame
filter logic.
(backtrace_command): Add "no-filters" option parsing.
(_initialize_stack): Alter help to reflect "no-filters" option.
* Makefile.in (SUBDIR_PYTHON_OBS): Add py-framefilter.o
(SUBDIR_PYTHON_SRCS): Add py-framefilter.c
(py-frame.o): Add target
* data-directory/Makefile.in (PYTHON_DIR): Add Python frame
filter files.
* python/python.h: Add new frame filter constants, and flag enum.
(apply_frame_filter): Add definition.
* python/python.c (apply_frame_filter): New non-Python
enabled function.
* python/py-utils.c (py_xdecref): New function.
(make_cleanup_py_xdecref): Ditto.
* python/py-objfile.c: Declare frame_filters dictionary.
(objfpy_dealloc): Add frame_filters dealloc.
(objfpy_new): Initialize frame_filters attribute.
(objfile_to_objfile_object): Ditto.
(objfpy_get_frame_filters): New function.
(objfpy_set_frame_filters): New function.
* python/py-progspace.c: Declare frame_filters dictionary.
(pspy_dealloc): Add frame_filters dealloc.
(pspy_new): Initialize frame_filters attribute.
(pspacee_to_pspace_object): Ditto.
(pspy_get_frame_filters): New function.
(pspy_set_frame_filters): New function.
* python/py-framefilter.c: New file.
* python/lib/gdb/command/frame_filters.py: New file.
* python/lib/gdb/frames.py: New file.
* python/lib/gdb/__init__.py: Initialize global frame_filters
dictionary
* python/lib/gdb/FrameDecorator.py: New file.
* python/lib/gdb/FrameIterator.py: New file.
* mi/mi-cmds.c (mi_cmds): Add frame filters command.
* mi/mi-cmds.h: Declare.
* mi/mi-cmd-stack.c (mi_cmd_stack_list_frames): Add
--no-frame-filter logic, and Python frame filter logic.
(stack_enable_frame_filters): New function.
(parse_no_frame_option): Ditto.
(mi_cmd_stack_list_frames): Add --no-frame-filter and Python frame
filter logic.
(mi_cmd_stack_list_locals): Ditto.
(mi_cmd_stack_list_args): Ditto.
(mi_cmd_stack_list_variables): Ditto.
* NEWS: Add frame filter note.
2013-05-10 Phil Muldoon <pmuldoon@redhat.com>
* gdb.python/py-framefilter.py: New File.
* gdb.python/py-framefilter-mi.exp: Ditto.
* gdb.python/py-framefilter.c: Ditto.
* gdb.python/py-framefilter-mi.exp: Ditto.
* gdb.python/py-framefilter-mi.c: Ditto,
* gdb.python/py-framefilter-gdb.py.in: Ditto.
2013-05-10 Phil Muldoon <pmuldoon@redhat.com>
* gdb.texinfo (Backtrace): Add "no-filter" argument.
(Python API): Add Frame Filters API, Frame Wrapper API,
Writing a Frame Filter/Wrapper, Managing Management of Frame
Filters chapter entries.
(Frame Filters API): New Node.
(Frame Wrapper API): New Node.
(Writing a Frame Filter): New Node.
(Managing Frame Filters): New Node.
(Progspaces In Python): Add note about frame_filters attribute.
(Objfiles in Python): Ditto.
(GDB/MI Stack Manipulation): Add -enable-frame-filters command,
@anchors and --no-frame-filters option to -stack-list-variables,
-stack-list-frames, -stack-list-locals and -stack-list-arguments
commands.
2013-05-10 10:26:03 +00:00
|
|
|
TRY_CATCH (except, RETURN_MASK_ALL)
|
|
|
|
{
|
|
|
|
ui_out_wrap_hint (out, " ");
|
|
|
|
ui_out_text (out, " at ");
|
|
|
|
annotate_frame_source_file ();
|
|
|
|
ui_out_field_string (out, "file", filename);
|
|
|
|
annotate_frame_source_file_end ();
|
|
|
|
}
|
|
|
|
if (except.reason < 0)
|
|
|
|
{
|
|
|
|
Py_DECREF (py_fn);
|
|
|
|
gdbpy_convert_exception (except);
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Py_DECREF (py_fn);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (PyObject_HasAttrString (filter, "line"))
|
|
|
|
{
|
|
|
|
PyObject *py_line = PyObject_CallMethod (filter, "line", NULL);
|
|
|
|
int line;
|
|
|
|
|
|
|
|
if (py_line != NULL)
|
|
|
|
{
|
|
|
|
if (py_line != Py_None)
|
|
|
|
{
|
|
|
|
line = PyLong_AsLong (py_line);
|
|
|
|
TRY_CATCH (except, RETURN_MASK_ALL)
|
|
|
|
{
|
|
|
|
ui_out_text (out, ":");
|
|
|
|
annotate_frame_source_line ();
|
|
|
|
ui_out_field_int (out, "line", line);
|
|
|
|
}
|
|
|
|
if (except.reason < 0)
|
|
|
|
{
|
|
|
|
Py_DECREF (py_line);
|
|
|
|
gdbpy_convert_exception (except);
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Py_DECREF (py_line);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* For MI we need to deal with the "children" list population of
|
|
|
|
elided frames, so if MI output detected do not send newline. */
|
|
|
|
if (! ui_out_is_mi_like_p (out))
|
|
|
|
{
|
|
|
|
TRY_CATCH (except, RETURN_MASK_ALL)
|
|
|
|
{
|
|
|
|
annotate_frame_end ();
|
|
|
|
ui_out_text (out, "\n");
|
|
|
|
}
|
|
|
|
if (except.reason < 0)
|
|
|
|
{
|
|
|
|
gdbpy_convert_exception (except);
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (print_locals)
|
|
|
|
{
|
|
|
|
if (py_print_locals (filter, out, args_type, indent,
|
|
|
|
frame) == PY_BT_ERROR)
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Finally recursively print elided frames, if any. */
|
|
|
|
elided = get_py_iter_from_func (filter, "elided");
|
|
|
|
if (elided == NULL)
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
make_cleanup_py_decref (elided);
|
|
|
|
if (elided != Py_None)
|
|
|
|
{
|
|
|
|
PyObject *item;
|
|
|
|
|
|
|
|
make_cleanup_ui_out_list_begin_end (out, "children");
|
|
|
|
|
|
|
|
if (! ui_out_is_mi_like_p (out))
|
|
|
|
indent++;
|
|
|
|
|
|
|
|
while ((item = PyIter_Next (elided)))
|
|
|
|
{
|
|
|
|
enum py_bt_status success = py_print_frame (item, flags,
|
|
|
|
args_type, out,
|
|
|
|
indent,
|
|
|
|
levels_printed);
|
|
|
|
|
|
|
|
if (success == PY_BT_ERROR)
|
|
|
|
{
|
|
|
|
Py_DECREF (item);
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
|
|
|
Py_DECREF (item);
|
|
|
|
}
|
|
|
|
if (item == NULL && PyErr_Occurred ())
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
do_cleanups (cleanup_stack);
|
|
|
|
return PY_BT_COMPLETED;
|
|
|
|
|
|
|
|
error:
|
|
|
|
do_cleanups (cleanup_stack);
|
|
|
|
return PY_BT_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Helper function to initiate frame filter invocation at starting
|
|
|
|
frame FRAME. */
|
|
|
|
|
|
|
|
static PyObject *
|
|
|
|
bootstrap_python_frame_filters (struct frame_info *frame,
|
|
|
|
int frame_low, int frame_high)
|
|
|
|
{
|
|
|
|
struct cleanup *cleanups =
|
|
|
|
make_cleanup (null_cleanup, NULL);
|
|
|
|
PyObject *module, *sort_func, *iterable, *frame_obj, *iterator;
|
|
|
|
PyObject *py_frame_low, *py_frame_high;
|
|
|
|
|
|
|
|
frame_obj = frame_info_to_frame_object (frame);
|
|
|
|
if (frame_obj == NULL)
|
|
|
|
goto error;
|
|
|
|
make_cleanup_py_decref (frame_obj);
|
|
|
|
|
|
|
|
module = PyImport_ImportModule ("gdb.frames");
|
|
|
|
if (module == NULL)
|
|
|
|
goto error;
|
|
|
|
make_cleanup_py_decref (module);
|
|
|
|
|
|
|
|
sort_func = PyObject_GetAttrString (module, "execute_frame_filters");
|
|
|
|
if (sort_func == NULL)
|
|
|
|
goto error;
|
|
|
|
make_cleanup_py_decref (sort_func);
|
|
|
|
|
|
|
|
py_frame_low = PyInt_FromLong (frame_low);
|
|
|
|
if (py_frame_low == NULL)
|
|
|
|
goto error;
|
|
|
|
make_cleanup_py_decref (py_frame_low);
|
|
|
|
|
|
|
|
py_frame_high = PyInt_FromLong (frame_high);
|
|
|
|
if (py_frame_high == NULL)
|
|
|
|
goto error;
|
|
|
|
make_cleanup_py_decref (py_frame_high);
|
|
|
|
|
|
|
|
iterable = PyObject_CallFunctionObjArgs (sort_func, frame_obj,
|
|
|
|
py_frame_low,
|
|
|
|
py_frame_high,
|
|
|
|
NULL);
|
|
|
|
if (iterable == NULL)
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
do_cleanups (cleanups);
|
|
|
|
|
|
|
|
if (iterable != Py_None)
|
|
|
|
{
|
|
|
|
iterator = PyObject_GetIter (iterable);
|
|
|
|
Py_DECREF (iterable);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
return iterable;
|
|
|
|
}
|
|
|
|
|
|
|
|
return iterator;
|
|
|
|
|
|
|
|
error:
|
|
|
|
do_cleanups (cleanups);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* This is the only publicly exported function in this file. FRAME
|
|
|
|
is the source frame to start frame-filter invocation. FLAGS is an
|
|
|
|
integer holding the flags for printing. The following elements of
|
|
|
|
the FRAME_FILTER_FLAGS enum denotes the make-up of FLAGS:
|
|
|
|
PRINT_LEVEL is a flag indicating whether to print the frame's
|
|
|
|
relative level in the output. PRINT_FRAME_INFO is a flag that
|
|
|
|
indicates whether this function should print the frame
|
|
|
|
information, PRINT_ARGS is a flag that indicates whether to print
|
|
|
|
frame arguments, and PRINT_LOCALS, likewise, with frame local
|
|
|
|
variables. ARGS_TYPE is an enumerator describing the argument
|
|
|
|
format, OUT is the output stream to print. FRAME_LOW is the
|
|
|
|
beginning of the slice of frames to print, and FRAME_HIGH is the
|
|
|
|
upper limit of the frames to count. Returns PY_BT_ERROR on error,
|
|
|
|
or PY_BT_COMPLETED on success.*/
|
|
|
|
|
|
|
|
enum py_bt_status
|
|
|
|
apply_frame_filter (struct frame_info *frame, int flags,
|
|
|
|
enum py_frame_args args_type,
|
|
|
|
struct ui_out *out, int frame_low,
|
|
|
|
int frame_high)
|
|
|
|
|
|
|
|
{
|
|
|
|
struct gdbarch *gdbarch = NULL;
|
|
|
|
struct cleanup *cleanups;
|
|
|
|
enum py_bt_status success = PY_BT_ERROR;
|
|
|
|
PyObject *iterable;
|
|
|
|
volatile struct gdb_exception except;
|
|
|
|
PyObject *item;
|
|
|
|
htab_t levels_printed;
|
|
|
|
|
2013-08-29 09:24:33 +00:00
|
|
|
if (!gdb_python_initialized)
|
|
|
|
return PY_BT_NO_FILTERS;
|
|
|
|
|
2013-05-10 Phil Muldoon <pmuldoon@redhat.com>
* stack.c (backtrace_command_1): Add "no-filters", and Python frame
filter logic.
(backtrace_command): Add "no-filters" option parsing.
(_initialize_stack): Alter help to reflect "no-filters" option.
* Makefile.in (SUBDIR_PYTHON_OBS): Add py-framefilter.o
(SUBDIR_PYTHON_SRCS): Add py-framefilter.c
(py-frame.o): Add target
* data-directory/Makefile.in (PYTHON_DIR): Add Python frame
filter files.
* python/python.h: Add new frame filter constants, and flag enum.
(apply_frame_filter): Add definition.
* python/python.c (apply_frame_filter): New non-Python
enabled function.
* python/py-utils.c (py_xdecref): New function.
(make_cleanup_py_xdecref): Ditto.
* python/py-objfile.c: Declare frame_filters dictionary.
(objfpy_dealloc): Add frame_filters dealloc.
(objfpy_new): Initialize frame_filters attribute.
(objfile_to_objfile_object): Ditto.
(objfpy_get_frame_filters): New function.
(objfpy_set_frame_filters): New function.
* python/py-progspace.c: Declare frame_filters dictionary.
(pspy_dealloc): Add frame_filters dealloc.
(pspy_new): Initialize frame_filters attribute.
(pspacee_to_pspace_object): Ditto.
(pspy_get_frame_filters): New function.
(pspy_set_frame_filters): New function.
* python/py-framefilter.c: New file.
* python/lib/gdb/command/frame_filters.py: New file.
* python/lib/gdb/frames.py: New file.
* python/lib/gdb/__init__.py: Initialize global frame_filters
dictionary
* python/lib/gdb/FrameDecorator.py: New file.
* python/lib/gdb/FrameIterator.py: New file.
* mi/mi-cmds.c (mi_cmds): Add frame filters command.
* mi/mi-cmds.h: Declare.
* mi/mi-cmd-stack.c (mi_cmd_stack_list_frames): Add
--no-frame-filter logic, and Python frame filter logic.
(stack_enable_frame_filters): New function.
(parse_no_frame_option): Ditto.
(mi_cmd_stack_list_frames): Add --no-frame-filter and Python frame
filter logic.
(mi_cmd_stack_list_locals): Ditto.
(mi_cmd_stack_list_args): Ditto.
(mi_cmd_stack_list_variables): Ditto.
* NEWS: Add frame filter note.
2013-05-10 Phil Muldoon <pmuldoon@redhat.com>
* gdb.python/py-framefilter.py: New File.
* gdb.python/py-framefilter-mi.exp: Ditto.
* gdb.python/py-framefilter.c: Ditto.
* gdb.python/py-framefilter-mi.exp: Ditto.
* gdb.python/py-framefilter-mi.c: Ditto,
* gdb.python/py-framefilter-gdb.py.in: Ditto.
2013-05-10 Phil Muldoon <pmuldoon@redhat.com>
* gdb.texinfo (Backtrace): Add "no-filter" argument.
(Python API): Add Frame Filters API, Frame Wrapper API,
Writing a Frame Filter/Wrapper, Managing Management of Frame
Filters chapter entries.
(Frame Filters API): New Node.
(Frame Wrapper API): New Node.
(Writing a Frame Filter): New Node.
(Managing Frame Filters): New Node.
(Progspaces In Python): Add note about frame_filters attribute.
(Objfiles in Python): Ditto.
(GDB/MI Stack Manipulation): Add -enable-frame-filters command,
@anchors and --no-frame-filters option to -stack-list-variables,
-stack-list-frames, -stack-list-locals and -stack-list-arguments
commands.
2013-05-10 10:26:03 +00:00
|
|
|
cleanups = ensure_python_env (gdbarch, current_language);
|
|
|
|
|
|
|
|
TRY_CATCH (except, RETURN_MASK_ALL)
|
|
|
|
{
|
|
|
|
gdbarch = get_frame_arch (frame);
|
|
|
|
}
|
|
|
|
if (except.reason < 0)
|
|
|
|
{
|
|
|
|
gdbpy_convert_exception (except);
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
|
|
|
iterable = bootstrap_python_frame_filters (frame, frame_low, frame_high);
|
|
|
|
|
|
|
|
if (iterable == NULL)
|
2013-08-29 09:24:33 +00:00
|
|
|
{
|
|
|
|
/* Normally if there is an error GDB prints the exception,
|
|
|
|
abandons the backtrace and exits. The user can then call "bt
|
|
|
|
no-filters", and get a default backtrace (it would be
|
|
|
|
confusing to automatically start a standard backtrace halfway
|
|
|
|
through a Python filtered backtrace). However in the case
|
|
|
|
where GDB cannot initialize the frame filters (most likely
|
|
|
|
due to incorrect auto-load paths), GDB has printed nothing.
|
|
|
|
In this case it is OK to print the default backtrace after
|
|
|
|
printing the error message. GDB returns PY_BT_NO_FILTERS
|
|
|
|
here to signify there are no filters after printing the
|
|
|
|
initialization error. This return code will trigger a
|
|
|
|
default backtrace. */
|
|
|
|
|
|
|
|
gdbpy_print_stack ();
|
|
|
|
do_cleanups (cleanups);
|
|
|
|
return PY_BT_NO_FILTERS;
|
|
|
|
}
|
2013-05-10 Phil Muldoon <pmuldoon@redhat.com>
* stack.c (backtrace_command_1): Add "no-filters", and Python frame
filter logic.
(backtrace_command): Add "no-filters" option parsing.
(_initialize_stack): Alter help to reflect "no-filters" option.
* Makefile.in (SUBDIR_PYTHON_OBS): Add py-framefilter.o
(SUBDIR_PYTHON_SRCS): Add py-framefilter.c
(py-frame.o): Add target
* data-directory/Makefile.in (PYTHON_DIR): Add Python frame
filter files.
* python/python.h: Add new frame filter constants, and flag enum.
(apply_frame_filter): Add definition.
* python/python.c (apply_frame_filter): New non-Python
enabled function.
* python/py-utils.c (py_xdecref): New function.
(make_cleanup_py_xdecref): Ditto.
* python/py-objfile.c: Declare frame_filters dictionary.
(objfpy_dealloc): Add frame_filters dealloc.
(objfpy_new): Initialize frame_filters attribute.
(objfile_to_objfile_object): Ditto.
(objfpy_get_frame_filters): New function.
(objfpy_set_frame_filters): New function.
* python/py-progspace.c: Declare frame_filters dictionary.
(pspy_dealloc): Add frame_filters dealloc.
(pspy_new): Initialize frame_filters attribute.
(pspacee_to_pspace_object): Ditto.
(pspy_get_frame_filters): New function.
(pspy_set_frame_filters): New function.
* python/py-framefilter.c: New file.
* python/lib/gdb/command/frame_filters.py: New file.
* python/lib/gdb/frames.py: New file.
* python/lib/gdb/__init__.py: Initialize global frame_filters
dictionary
* python/lib/gdb/FrameDecorator.py: New file.
* python/lib/gdb/FrameIterator.py: New file.
* mi/mi-cmds.c (mi_cmds): Add frame filters command.
* mi/mi-cmds.h: Declare.
* mi/mi-cmd-stack.c (mi_cmd_stack_list_frames): Add
--no-frame-filter logic, and Python frame filter logic.
(stack_enable_frame_filters): New function.
(parse_no_frame_option): Ditto.
(mi_cmd_stack_list_frames): Add --no-frame-filter and Python frame
filter logic.
(mi_cmd_stack_list_locals): Ditto.
(mi_cmd_stack_list_args): Ditto.
(mi_cmd_stack_list_variables): Ditto.
* NEWS: Add frame filter note.
2013-05-10 Phil Muldoon <pmuldoon@redhat.com>
* gdb.python/py-framefilter.py: New File.
* gdb.python/py-framefilter-mi.exp: Ditto.
* gdb.python/py-framefilter.c: Ditto.
* gdb.python/py-framefilter-mi.exp: Ditto.
* gdb.python/py-framefilter-mi.c: Ditto,
* gdb.python/py-framefilter-gdb.py.in: Ditto.
2013-05-10 Phil Muldoon <pmuldoon@redhat.com>
* gdb.texinfo (Backtrace): Add "no-filter" argument.
(Python API): Add Frame Filters API, Frame Wrapper API,
Writing a Frame Filter/Wrapper, Managing Management of Frame
Filters chapter entries.
(Frame Filters API): New Node.
(Frame Wrapper API): New Node.
(Writing a Frame Filter): New Node.
(Managing Frame Filters): New Node.
(Progspaces In Python): Add note about frame_filters attribute.
(Objfiles in Python): Ditto.
(GDB/MI Stack Manipulation): Add -enable-frame-filters command,
@anchors and --no-frame-filters option to -stack-list-variables,
-stack-list-frames, -stack-list-locals and -stack-list-arguments
commands.
2013-05-10 10:26:03 +00:00
|
|
|
|
|
|
|
/* If iterable is None, then there are no frame filters registered.
|
|
|
|
If this is the case, defer to default GDB printing routines in MI
|
|
|
|
and CLI. */
|
|
|
|
make_cleanup_py_decref (iterable);
|
|
|
|
if (iterable == Py_None)
|
|
|
|
{
|
|
|
|
success = PY_BT_NO_FILTERS;
|
|
|
|
goto done;
|
|
|
|
}
|
|
|
|
|
|
|
|
levels_printed = htab_create (20,
|
|
|
|
htab_hash_pointer,
|
|
|
|
htab_eq_pointer,
|
|
|
|
NULL);
|
|
|
|
make_cleanup_htab_delete (levels_printed);
|
|
|
|
|
|
|
|
while ((item = PyIter_Next (iterable)))
|
|
|
|
{
|
|
|
|
success = py_print_frame (item, flags, args_type, out, 0,
|
|
|
|
levels_printed);
|
|
|
|
|
|
|
|
/* Do not exit on error printing a single frame. Print the
|
|
|
|
error and continue with other frames. */
|
|
|
|
if (success == PY_BT_ERROR)
|
|
|
|
gdbpy_print_stack ();
|
|
|
|
|
|
|
|
Py_DECREF (item);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (item == NULL && PyErr_Occurred ())
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
done:
|
|
|
|
do_cleanups (cleanups);
|
|
|
|
return success;
|
|
|
|
|
2013-08-29 09:24:33 +00:00
|
|
|
/* Exit and abandon backtrace on error, printing the exception that
|
|
|
|
is set. */
|
2013-05-10 Phil Muldoon <pmuldoon@redhat.com>
* stack.c (backtrace_command_1): Add "no-filters", and Python frame
filter logic.
(backtrace_command): Add "no-filters" option parsing.
(_initialize_stack): Alter help to reflect "no-filters" option.
* Makefile.in (SUBDIR_PYTHON_OBS): Add py-framefilter.o
(SUBDIR_PYTHON_SRCS): Add py-framefilter.c
(py-frame.o): Add target
* data-directory/Makefile.in (PYTHON_DIR): Add Python frame
filter files.
* python/python.h: Add new frame filter constants, and flag enum.
(apply_frame_filter): Add definition.
* python/python.c (apply_frame_filter): New non-Python
enabled function.
* python/py-utils.c (py_xdecref): New function.
(make_cleanup_py_xdecref): Ditto.
* python/py-objfile.c: Declare frame_filters dictionary.
(objfpy_dealloc): Add frame_filters dealloc.
(objfpy_new): Initialize frame_filters attribute.
(objfile_to_objfile_object): Ditto.
(objfpy_get_frame_filters): New function.
(objfpy_set_frame_filters): New function.
* python/py-progspace.c: Declare frame_filters dictionary.
(pspy_dealloc): Add frame_filters dealloc.
(pspy_new): Initialize frame_filters attribute.
(pspacee_to_pspace_object): Ditto.
(pspy_get_frame_filters): New function.
(pspy_set_frame_filters): New function.
* python/py-framefilter.c: New file.
* python/lib/gdb/command/frame_filters.py: New file.
* python/lib/gdb/frames.py: New file.
* python/lib/gdb/__init__.py: Initialize global frame_filters
dictionary
* python/lib/gdb/FrameDecorator.py: New file.
* python/lib/gdb/FrameIterator.py: New file.
* mi/mi-cmds.c (mi_cmds): Add frame filters command.
* mi/mi-cmds.h: Declare.
* mi/mi-cmd-stack.c (mi_cmd_stack_list_frames): Add
--no-frame-filter logic, and Python frame filter logic.
(stack_enable_frame_filters): New function.
(parse_no_frame_option): Ditto.
(mi_cmd_stack_list_frames): Add --no-frame-filter and Python frame
filter logic.
(mi_cmd_stack_list_locals): Ditto.
(mi_cmd_stack_list_args): Ditto.
(mi_cmd_stack_list_variables): Ditto.
* NEWS: Add frame filter note.
2013-05-10 Phil Muldoon <pmuldoon@redhat.com>
* gdb.python/py-framefilter.py: New File.
* gdb.python/py-framefilter-mi.exp: Ditto.
* gdb.python/py-framefilter.c: Ditto.
* gdb.python/py-framefilter-mi.exp: Ditto.
* gdb.python/py-framefilter-mi.c: Ditto,
* gdb.python/py-framefilter-gdb.py.in: Ditto.
2013-05-10 Phil Muldoon <pmuldoon@redhat.com>
* gdb.texinfo (Backtrace): Add "no-filter" argument.
(Python API): Add Frame Filters API, Frame Wrapper API,
Writing a Frame Filter/Wrapper, Managing Management of Frame
Filters chapter entries.
(Frame Filters API): New Node.
(Frame Wrapper API): New Node.
(Writing a Frame Filter): New Node.
(Managing Frame Filters): New Node.
(Progspaces In Python): Add note about frame_filters attribute.
(Objfiles in Python): Ditto.
(GDB/MI Stack Manipulation): Add -enable-frame-filters command,
@anchors and --no-frame-filters option to -stack-list-variables,
-stack-list-frames, -stack-list-locals and -stack-list-arguments
commands.
2013-05-10 10:26:03 +00:00
|
|
|
error:
|
|
|
|
gdbpy_print_stack ();
|
|
|
|
do_cleanups (cleanups);
|
|
|
|
return PY_BT_ERROR;
|
|
|
|
}
|