| /* Python frame filters | 
 |  | 
 |    Copyright (C) 2013-2024 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 "objfiles.h" | 
 | #include "symtab.h" | 
 | #include "language.h" | 
 | #include "arch-utils.h" | 
 | #include "python.h" | 
 | #include "ui-out.h" | 
 | #include "valprint.h" | 
 | #include "stack.h" | 
 | #include "source.h" | 
 | #include "annotate.h" | 
 | #include "hashtab.h" | 
 | #include "demangle.h" | 
 | #include "mi/mi-cmds.h" | 
 | #include "python-internal.h" | 
 | #include <optional> | 
 | #include "cli/cli-style.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 and  SYM_BLOCK is a pass-through argument to | 
 |    write  the block where the symbol lies in.  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  EXT_LANG_BT_ERROR  on  error  with the | 
 |    appropriate Python exception set, and EXT_LANG_BT_OK on success.  */ | 
 |  | 
 | static enum ext_lang_bt_status | 
 | extract_sym (PyObject *obj, gdb::unique_xmalloc_ptr<char> *name, | 
 | 	     struct symbol **sym, const struct block **sym_block, | 
 | 	     const struct language_defn **language) | 
 | { | 
 |   gdbpy_ref<> result = gdbpy_call_method (obj, "symbol"); | 
 |  | 
 |   if (result == NULL) | 
 |     return EXT_LANG_BT_ERROR; | 
 |  | 
 |   /* For 'symbol' callback, the function can return a symbol or a | 
 |      string.  */ | 
 |   if (gdbpy_is_string (result.get ())) | 
 |     { | 
 |       *name = python_string_to_host_string (result.get ()); | 
 |  | 
 |       if (*name == NULL) | 
 | 	return EXT_LANG_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 | 
 | 	 the current language.  */ | 
 |       *language = current_language; | 
 |       *sym = NULL; | 
 |       *sym_block = 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.get ()); | 
 |       /* TODO: currently, we have no way to recover the block in which SYMBOL | 
 | 	 was found, so we have no block to return.  Trying to evaluate SYMBOL | 
 | 	 will yield an incorrect value when it's located in a FRAME and | 
 | 	 evaluated from another frame (as permitted in nested functions).  */ | 
 |       *sym_block = NULL; | 
 |  | 
 |       if (*sym == NULL) | 
 | 	{ | 
 | 	  PyErr_SetString (PyExc_RuntimeError, | 
 | 			   _("Unexpected value.  Expecting a " | 
 | 			     "gdb.Symbol or a Python string.")); | 
 | 	  return EXT_LANG_BT_ERROR; | 
 | 	} | 
 |  | 
 |       /* Duplicate the symbol name, so the caller has consistency | 
 | 	 in garbage collection.  */ | 
 |       name->reset (xstrdup ((*sym)->print_name ())); | 
 |  | 
 |       /* 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 ((*sym)->language ()); | 
 |       else | 
 | 	*language = current_language; | 
 |     } | 
 |  | 
 |   return EXT_LANG_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 EXT_LANG_BT_ERROR | 
 |    on error with the appropriate Python exception set, and EXT_LANG_BT_OK on | 
 |    success.  */ | 
 |  | 
 | static enum ext_lang_bt_status | 
 | extract_value (PyObject *obj, struct value **value) | 
 | { | 
 |   if (PyObject_HasAttrString (obj, "value")) | 
 |     { | 
 |       gdbpy_ref<> vresult = gdbpy_call_method (obj, "value"); | 
 |  | 
 |       if (vresult == NULL) | 
 | 	return EXT_LANG_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) | 
 | 	{ | 
 | 	  *value = NULL; | 
 | 	  return EXT_LANG_BT_OK; | 
 | 	} | 
 |       else | 
 | 	{ | 
 | 	  *value = convert_value_from_python (vresult.get ()); | 
 |  | 
 | 	  if (*value == NULL) | 
 | 	    return EXT_LANG_BT_ERROR; | 
 |  | 
 | 	  return EXT_LANG_BT_OK; | 
 | 	} | 
 |     } | 
 |   else | 
 |     *value = NULL; | 
 |  | 
 |   return EXT_LANG_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 (sym->aclass ()) | 
 |     { | 
 |     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 = ! sym->is_argument (); | 
 |       else | 
 | 	print_me = sym->is_argument (); | 
 |     } | 
 |   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.  */ | 
 |  | 
 | static void | 
 | py_print_type (struct ui_out *out, struct value *val) | 
 | { | 
 |   check_typedef (val->type ()); | 
 |  | 
 |   string_file stb; | 
 |   type_print (val->type (), "", &stb, -1); | 
 |   out->field_stream ("type", stb); | 
 | } | 
 |  | 
 | /* 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.  */ | 
 |  | 
 | static void | 
 | py_print_value (struct ui_out *out, struct value *val, | 
 | 		const struct value_print_options *opts, | 
 | 		int indent, | 
 | 		enum ext_lang_frame_args args_type, | 
 | 		const struct language_defn *language) | 
 | { | 
 |   int should_print = 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) | 
 |     { | 
 |       if (args_type == MI_PRINT_ALL_VALUES) | 
 | 	should_print = 1; | 
 |       else if (args_type == MI_PRINT_SIMPLE_VALUES | 
 | 	       && mi_simple_type_p (val->type ())) | 
 | 	should_print = 1; | 
 |     } | 
 |   else if (args_type != NO_VALUES) | 
 |     should_print = 1; | 
 |  | 
 |   if (should_print) | 
 |     { | 
 |       string_file stb; | 
 |  | 
 |       common_val_print (val, &stb, indent, opts, language); | 
 |       out->field_stream ("value", stb); | 
 |     } | 
 | } | 
 |  | 
 | /* 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, const char *func) | 
 | { | 
 |   if (PyObject_HasAttrString (filter, func)) | 
 |     { | 
 |       gdbpy_ref<> result = gdbpy_call_method (filter, func); | 
 |  | 
 |       if (result != NULL) | 
 | 	{ | 
 | 	  if (result == Py_None) | 
 | 	    { | 
 | 	      return result.release (); | 
 | 	    } | 
 | 	  else | 
 | 	    { | 
 | 	      return PyObject_GetIter (result.get ()); | 
 | 	    } | 
 | 	} | 
 |     } | 
 |   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.  */ | 
 |  | 
 | static void | 
 | 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 ext_lang_frame_args args_type, | 
 | 		     int print_args_field, | 
 | 		     const struct language_defn *language) | 
 | { | 
 |   struct value *val; | 
 |  | 
 |   if (fa != NULL) | 
 |     { | 
 |       if (fa->val == NULL && fa->error == NULL) | 
 | 	return; | 
 |       language = language_def (fa->sym->language ()); | 
 |       val = fa->val; | 
 |     } | 
 |   else | 
 |     val = fv; | 
 |  | 
 |   std::optional<ui_out_emit_tuple> maybe_tuple; | 
 |  | 
 |   /*  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 (out->is_mi_like_p ()) | 
 |     { | 
 |       if (print_args_field || args_type != NO_VALUES) | 
 | 	maybe_tuple.emplace (out, nullptr); | 
 |     } | 
 |  | 
 |   annotate_arg_begin (); | 
 |  | 
 |   /* If frame argument is populated, check for entry-values and the | 
 |      entry value options.  */ | 
 |   if (fa != NULL) | 
 |     { | 
 |       string_file stb; | 
 |  | 
 |       gdb_puts (fa->sym->print_name (), &stb); | 
 |       if (fa->entry_kind == print_entry_values_compact) | 
 | 	{ | 
 | 	  stb.puts ("="); | 
 |  | 
 | 	  gdb_puts (fa->sym->print_name (), &stb); | 
 | 	} | 
 |       if (fa->entry_kind == print_entry_values_only | 
 | 	  || fa->entry_kind == print_entry_values_compact) | 
 | 	stb.puts ("@entry"); | 
 |       out->field_stream ("name", stb); | 
 |     } | 
 |   else | 
 |     /* Otherwise, just output the name.  */ | 
 |     out->field_string ("name", sym_name); | 
 |  | 
 |   annotate_arg_name_end (); | 
 |  | 
 |   out->text ("="); | 
 |  | 
 |   if (print_args_field) | 
 |     out->field_signed ("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 && val != NULL) | 
 |     py_print_type (out, val); | 
 |  | 
 |   if (val != NULL) | 
 |     annotate_arg_value (val->type ()); | 
 |  | 
 |   /* If the output is to the CLI, and the user option "set print | 
 |      frame-arguments" is set to none, just output "...".  */ | 
 |   if (! out->is_mi_like_p () && args_type == NO_VALUES) | 
 |     out->field_string ("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 (val == NULL) | 
 | 	    { | 
 | 	      gdb_assert (fa != NULL && fa->error != NULL); | 
 | 	      out->field_fmt ("value", metadata_style.style (), | 
 | 			      _("<error reading variable: %s>"), | 
 | 			      fa->error.get ()); | 
 | 	    } | 
 | 	  else | 
 | 	    py_print_value (out, val, opts, 0, args_type, language); | 
 | 	} | 
 |     } | 
 | } | 
 |  | 
 | /* 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 EXT_LANG_BT_ERROR on error, with any GDB | 
 |    exceptions converted to a Python exception, or EXT_LANG_BT_OK on | 
 |    success.  */ | 
 |  | 
 | static enum ext_lang_bt_status | 
 | enumerate_args (PyObject *iter, | 
 | 		struct ui_out *out, | 
 | 		enum ext_lang_frame_args args_type, | 
 | 		bool raw_frame_args, | 
 | 		int print_args_field, | 
 | 		const frame_info_ptr &frame) | 
 | { | 
 |   struct value_print_options opts; | 
 |  | 
 |   get_user_print_options (&opts); | 
 |   opts.raw = raw_frame_args; | 
 |  | 
 |   if (args_type == CLI_SCALAR_VALUES) | 
 |     { | 
 |       /* True in "summary" mode, false otherwise.  */ | 
 |       opts.summary = true; | 
 |     } | 
 |  | 
 |   opts.deref_ref = true; | 
 |  | 
 |   annotate_frame_args (); | 
 |  | 
 |   /*  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.  */ | 
 |   gdbpy_ref<> item (PyIter_Next (iter)); | 
 |   if (item == NULL && PyErr_Occurred ()) | 
 |     return EXT_LANG_BT_ERROR; | 
 |  | 
 |   while (item != NULL) | 
 |     { | 
 |       const struct language_defn *language; | 
 |       gdb::unique_xmalloc_ptr<char> sym_name; | 
 |       struct symbol *sym; | 
 |       const struct block *sym_block; | 
 |       struct value *val; | 
 |       enum ext_lang_bt_status success = EXT_LANG_BT_ERROR; | 
 |  | 
 |       success = extract_sym (item.get (), &sym_name, &sym, &sym_block, | 
 | 			     &language); | 
 |       if (success == EXT_LANG_BT_ERROR) | 
 | 	return EXT_LANG_BT_ERROR; | 
 |  | 
 |       success = extract_value (item.get (), &val); | 
 |       if (success == EXT_LANG_BT_ERROR) | 
 | 	return EXT_LANG_BT_ERROR; | 
 |  | 
 |       if (sym && out->is_mi_like_p () | 
 | 	  && ! mi_should_print (sym, MI_PRINT_ARGS)) | 
 | 	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.")); | 
 | 	      return EXT_LANG_BT_ERROR; | 
 | 	    } | 
 |  | 
 | 	  read_frame_arg (user_frame_print_options, | 
 | 			  sym, frame, &arg, &entryarg); | 
 |  | 
 | 	  /* 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) | 
 | 	    { | 
 | 	      py_print_single_arg (out, NULL, &arg, | 
 | 				   NULL, &opts, | 
 | 				   args_type, | 
 | 				   print_args_field, | 
 | 				   NULL); | 
 | 	    } | 
 |  | 
 | 	  if (entryarg.entry_kind != print_entry_values_no) | 
 | 	    { | 
 | 	      if (arg.entry_kind != print_entry_values_only) | 
 | 		{ | 
 | 		  out->text (", "); | 
 | 		  out->wrap_hint (4); | 
 | 		} | 
 |  | 
 | 	      py_print_single_arg (out, NULL, &entryarg, NULL, &opts, | 
 | 				   args_type, print_args_field, NULL); | 
 | 	    } | 
 | 	} | 
 |       else | 
 | 	{ | 
 | 	  /* If the object has provided a value, we just print that.  */ | 
 | 	  if (val != NULL) | 
 | 	    py_print_single_arg (out, sym_name.get (), NULL, val, &opts, | 
 | 				 args_type, print_args_field, | 
 | 				 language); | 
 | 	} | 
 |  | 
 |       /* Collect the next item from the iterator.  If | 
 | 	 this is the last item, do not print the | 
 | 	 comma.  */ | 
 |       item.reset (PyIter_Next (iter)); | 
 |       if (item != NULL) | 
 | 	out->text (", "); | 
 |       else if (PyErr_Occurred ()) | 
 | 	return EXT_LANG_BT_ERROR; | 
 |  | 
 |       annotate_arg_end (); | 
 |     } | 
 |  | 
 |   return EXT_LANG_BT_OK; | 
 | } | 
 |  | 
 |  | 
 | /* 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 | 
 |    EXT_LANG_BT_ERROR on error, with any GDB exceptions converted to a Python | 
 |    exception, or EXT_LANG_BT_OK on success.  */ | 
 |  | 
 | static enum ext_lang_bt_status | 
 | enumerate_locals (PyObject *iter, | 
 | 		  struct ui_out *out, | 
 | 		  int indent, | 
 | 		  enum ext_lang_frame_args args_type, | 
 | 		  int print_args_field, | 
 | 		  const frame_info_ptr &frame) | 
 | { | 
 |   struct value_print_options opts; | 
 |  | 
 |   get_user_print_options (&opts); | 
 |   opts.deref_ref = true; | 
 |  | 
 |   while (true) | 
 |     { | 
 |       const struct language_defn *language; | 
 |       gdb::unique_xmalloc_ptr<char> sym_name; | 
 |       struct value *val; | 
 |       enum ext_lang_bt_status success = EXT_LANG_BT_ERROR; | 
 |       struct symbol *sym; | 
 |       const struct block *sym_block; | 
 |       int local_indent = 8 + (8 * indent); | 
 |       std::optional<ui_out_emit_tuple> tuple; | 
 |  | 
 |       gdbpy_ref<> item (PyIter_Next (iter)); | 
 |       if (item == NULL) | 
 | 	break; | 
 |  | 
 |       success = extract_sym (item.get (), &sym_name, &sym, &sym_block, | 
 | 			     &language); | 
 |       if (success == EXT_LANG_BT_ERROR) | 
 | 	return EXT_LANG_BT_ERROR; | 
 |  | 
 |       success = extract_value (item.get (), &val); | 
 |       if (success == EXT_LANG_BT_ERROR) | 
 | 	return EXT_LANG_BT_ERROR; | 
 |  | 
 |       if (sym != NULL && out->is_mi_like_p () | 
 | 	  && ! mi_should_print (sym, MI_PRINT_LOCALS)) | 
 | 	continue; | 
 |  | 
 |       /* If the object did not provide a value, read it.  */ | 
 |       if (val == NULL) | 
 | 	val = read_var_value (sym, sym_block, frame); | 
 |  | 
 |       /* 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 (out->is_mi_like_p ()) | 
 | 	{ | 
 | 	  if (print_args_field || args_type != NO_VALUES) | 
 | 	    tuple.emplace (out, nullptr); | 
 | 	} | 
 |  | 
 |       /* If the output is not MI we indent locals.  */ | 
 |       out->spaces (local_indent); | 
 |       out->field_string ("name", sym_name.get ()); | 
 |       out->text (" = "); | 
 |  | 
 |       if (args_type == MI_PRINT_SIMPLE_VALUES) | 
 | 	py_print_type (out, val); | 
 |  | 
 |       /* CLI always prints values for locals.  MI uses the | 
 | 	 simple/no/all system.  */ | 
 |       if (! out->is_mi_like_p ()) | 
 | 	{ | 
 | 	  int val_indent = (indent + 1) * 4; | 
 |  | 
 | 	  py_print_value (out, val, &opts, val_indent, args_type, | 
 | 			  language); | 
 | 	} | 
 |       else | 
 | 	{ | 
 | 	  if (args_type != NO_VALUES) | 
 | 	    py_print_value (out, val, &opts, 0, args_type, | 
 | 			    language); | 
 | 	} | 
 |  | 
 |       out->text ("\n"); | 
 |     } | 
 |  | 
 |   if (!PyErr_Occurred ()) | 
 |     return EXT_LANG_BT_OK; | 
 |  | 
 |   return EXT_LANG_BT_ERROR; | 
 | } | 
 |  | 
 | /*  Helper function for -stack-list-variables.  Returns EXT_LANG_BT_ERROR on | 
 |     error, or EXT_LANG_BT_OK on success.  */ | 
 |  | 
 | static enum ext_lang_bt_status | 
 | py_mi_print_variables (PyObject *filter, struct ui_out *out, | 
 | 		       struct value_print_options *opts, | 
 | 		       enum ext_lang_frame_args args_type, | 
 | 		       const frame_info_ptr &frame, | 
 | 		       bool raw_frame_args_p) | 
 | { | 
 |   gdbpy_ref<> args_iter (get_py_iter_from_func (filter, "frame_args")); | 
 |   if (args_iter == NULL) | 
 |     return EXT_LANG_BT_ERROR; | 
 |  | 
 |   gdbpy_ref<> locals_iter (get_py_iter_from_func (filter, "frame_locals")); | 
 |   if (locals_iter == NULL) | 
 |     return EXT_LANG_BT_ERROR; | 
 |  | 
 |   ui_out_emit_list list_emitter (out, "variables"); | 
 |  | 
 |   if (args_iter != Py_None | 
 |       && (enumerate_args (args_iter.get (), out, args_type, raw_frame_args_p, | 
 | 			  1, frame) == EXT_LANG_BT_ERROR)) | 
 |     return EXT_LANG_BT_ERROR; | 
 |  | 
 |   if (locals_iter != Py_None | 
 |       && (enumerate_locals (locals_iter.get (), out, 1, args_type, 1, frame) | 
 | 	  == EXT_LANG_BT_ERROR)) | 
 |     return EXT_LANG_BT_ERROR; | 
 |  | 
 |   return EXT_LANG_BT_OK; | 
 | } | 
 |  | 
 | /* Helper function for printing locals.  This function largely just | 
 |    creates the wrapping tuple, and calls enumerate_locals.  Returns | 
 |    EXT_LANG_BT_ERROR on error, or EXT_LANG_BT_OK on success.  */ | 
 |  | 
 | static enum ext_lang_bt_status | 
 | py_print_locals (PyObject *filter, | 
 | 		 struct ui_out *out, | 
 | 		 enum ext_lang_frame_args args_type, | 
 | 		 int indent, | 
 | 		 const frame_info_ptr &frame) | 
 | { | 
 |   gdbpy_ref<> locals_iter (get_py_iter_from_func (filter, "frame_locals")); | 
 |   if (locals_iter == NULL) | 
 |     return EXT_LANG_BT_ERROR; | 
 |  | 
 |   ui_out_emit_list list_emitter (out, "locals"); | 
 |  | 
 |   if (locals_iter != Py_None | 
 |       && (enumerate_locals (locals_iter.get (), out, indent, args_type, | 
 | 			    0, frame) == EXT_LANG_BT_ERROR)) | 
 |     return EXT_LANG_BT_ERROR; | 
 |  | 
 |   return EXT_LANG_BT_OK; | 
 | } | 
 |  | 
 | /* Helper function for printing frame arguments.  This function | 
 |    largely just creates the wrapping tuple, and calls enumerate_args. | 
 |    Returns EXT_LANG_BT_ERROR on error, with any GDB exceptions converted to | 
 |    a Python exception, or EXT_LANG_BT_OK on success.  */ | 
 |  | 
 | static enum ext_lang_bt_status | 
 | py_print_args (PyObject *filter, | 
 | 	       struct ui_out *out, | 
 | 	       enum ext_lang_frame_args args_type, | 
 | 	       bool raw_frame_args, | 
 | 	       const frame_info_ptr &frame) | 
 | { | 
 |   gdbpy_ref<> args_iter (get_py_iter_from_func (filter, "frame_args")); | 
 |   if (args_iter == NULL) | 
 |     return EXT_LANG_BT_ERROR; | 
 |  | 
 |   ui_out_emit_list list_emitter (out, "args"); | 
 |  | 
 |   out->wrap_hint (3); | 
 |   annotate_frame_args (); | 
 |   out->text (" ("); | 
 |  | 
 |   if (args_type == CLI_PRESENCE) | 
 |     { | 
 |       if (args_iter != Py_None) | 
 | 	{ | 
 | 	  gdbpy_ref<> item (PyIter_Next (args_iter.get ())); | 
 |  | 
 | 	  if (item != NULL) | 
 | 	    out->text ("..."); | 
 | 	  else if (PyErr_Occurred ()) | 
 | 	    return EXT_LANG_BT_ERROR; | 
 | 	} | 
 |     } | 
 |   else if (args_iter != Py_None | 
 | 	   && (enumerate_args (args_iter.get (), out, args_type, | 
 | 			       raw_frame_args, 0, frame) | 
 | 	       == EXT_LANG_BT_ERROR)) | 
 |     return EXT_LANG_BT_ERROR; | 
 |  | 
 |   out->text (")"); | 
 |  | 
 |   return EXT_LANG_BT_OK; | 
 | } | 
 |  | 
 | /*  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 EXT_LANG_BT_ERROR on error, with any | 
 |     GDB exceptions converted to a Python exception, or EXT_LANG_BT_OK | 
 |     on success.  It can also throw an exception RETURN_QUIT.  */ | 
 |  | 
 | static enum ext_lang_bt_status | 
 | py_print_frame (PyObject *filter, frame_filter_flags flags, | 
 | 		enum ext_lang_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; | 
 |   frame_info_ptr frame = NULL; | 
 |   struct value_print_options opts; | 
 |  | 
 |   int print_level, print_frame_info, print_args, print_locals; | 
 |   /* Note that the below default in non-mi mode is the same as the | 
 |      default value for the backtrace command (see the call to print_frame_info | 
 |      in backtrace_command_1). | 
 |      Having the same default ensures that 'bt' and 'bt no-filters' | 
 |      have the same behaviour when some filters exist but do not apply | 
 |      to a frame.  */ | 
 |   enum print_what print_what | 
 |     = out->is_mi_like_p () ? LOC_AND_ADDRESS : LOCATION; | 
 |   gdb::unique_xmalloc_ptr<char> function_to_free; | 
 |  | 
 |   /* 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); | 
 |   if (print_frame_info) | 
 |     { | 
 |       std::optional<enum print_what> user_frame_info_print_what; | 
 |  | 
 |       get_user_print_what_frame_info (&user_frame_info_print_what); | 
 |       if (!out->is_mi_like_p () && user_frame_info_print_what.has_value ()) | 
 | 	{ | 
 | 	  /* Use the specific frame information desired by the user.  */ | 
 | 	  print_what = *user_frame_info_print_what; | 
 | 	} | 
 |     } | 
 |  | 
 |   /* 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.  */ | 
 |   gdbpy_ref<> py_inf_frame = gdbpy_call_method (filter, "inferior_frame"); | 
 |   if (py_inf_frame == NULL) | 
 |     return EXT_LANG_BT_ERROR; | 
 |  | 
 |   frame = frame_object_to_frame_info (py_inf_frame.get ()); | 
 |   if (frame == NULL) | 
 |     return EXT_LANG_BT_ERROR; | 
 |  | 
 |   symtab_and_line sal = find_frame_sal (frame); | 
 |  | 
 |   gdbarch = get_frame_arch (frame); | 
 |  | 
 |   /* stack-list-variables.  */ | 
 |   if (print_locals && print_args && ! print_frame_info) | 
 |     { | 
 |       bool raw_frame_args = (flags & PRINT_RAW_FRAME_ARGUMENTS) != 0; | 
 |       if (py_mi_print_variables (filter, out, &opts, args_type, frame, | 
 | 				 raw_frame_args) == EXT_LANG_BT_ERROR) | 
 | 	return EXT_LANG_BT_ERROR; | 
 |       return EXT_LANG_BT_OK; | 
 |     } | 
 |  | 
 |   std::optional<ui_out_emit_tuple> tuple; | 
 |  | 
 |   /* -stack-list-locals does not require a | 
 |      wrapping frame attribute.  */ | 
 |   if (print_frame_info || (print_args && ! print_locals)) | 
 |     tuple.emplace (out, "frame"); | 
 |  | 
 |   if (print_frame_info) | 
 |     { | 
 |       /* Elided frames are also printed with this function (recursively) | 
 | 	 and are printed with indention.  */ | 
 |       if (indent > 0) | 
 | 	out->spaces (indent * 4); | 
 |  | 
 |       /* The address is required for frame annotations, and also for | 
 | 	 address printing.  */ | 
 |       if (PyObject_HasAttrString (filter, "address")) | 
 | 	{ | 
 | 	  gdbpy_ref<> paddr = gdbpy_call_method (filter, "address"); | 
 |  | 
 | 	  if (paddr == NULL) | 
 | 	    return EXT_LANG_BT_ERROR; | 
 |  | 
 | 	  if (paddr != Py_None) | 
 | 	    { | 
 | 	      if (get_addr_from_python (paddr.get (), &address) < 0) | 
 | 		return EXT_LANG_BT_ERROR; | 
 |  | 
 | 	      has_addr = 1; | 
 | 	    } | 
 | 	} | 
 |     } | 
 |  | 
 |   /* For MI, each piece is controlled individually.  */ | 
 |   bool location_print = (print_frame_info | 
 | 			 && !out->is_mi_like_p () | 
 | 			 && (print_what == LOCATION | 
 | 			     || print_what == SRC_AND_LOC | 
 | 			     || print_what == LOC_AND_ADDRESS | 
 | 			     || print_what == SHORT_LOCATION)); | 
 |  | 
 |   /* Print frame level.  MI does not require the level if | 
 |      locals/variables only are being printed.  */ | 
 |   if (print_level | 
 |       && (location_print | 
 | 	  || (out->is_mi_like_p () && (print_frame_info || print_args)))) | 
 |     { | 
 |       struct frame_info **slot; | 
 |       int level; | 
 |  | 
 |       slot = (frame_info **) htab_find_slot (levels_printed, | 
 | 						   frame.get(), INSERT); | 
 |  | 
 |       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) | 
 | 	out->field_skip ("level"); | 
 |       else | 
 | 	{ | 
 | 	  *slot = frame.get (); | 
 | 	  annotate_frame_begin (print_level ? level : 0, | 
 | 				gdbarch, address); | 
 | 	  out->text ("#"); | 
 | 	  out->field_fmt_signed (2, ui_left, "level", level); | 
 | 	} | 
 |     } | 
 |  | 
 |   if (location_print || (out->is_mi_like_p () && print_frame_info)) | 
 |     { | 
 |       /* Print address to the address field.  If an address is not provided, | 
 | 	 print nothing.  */ | 
 |       if (opts.addressprint && has_addr) | 
 | 	{ | 
 | 	  if (!sal.symtab | 
 | 	      || frame_show_address (frame, sal) | 
 | 	      || print_what == LOC_AND_ADDRESS) | 
 | 	    { | 
 | 	      annotate_frame_address (); | 
 | 	      out->field_core_addr ("addr", gdbarch, address); | 
 | 	      if (get_frame_pc_masked (frame)) | 
 | 		out->field_string ("pac", " [PAC]"); | 
 | 	      annotate_frame_address_end (); | 
 | 	      out->text (" in "); | 
 | 	    } | 
 | 	} | 
 |  | 
 |       /* Print frame function name.  */ | 
 |       if (PyObject_HasAttrString (filter, "function")) | 
 | 	{ | 
 | 	  gdbpy_ref<> py_func = gdbpy_call_method (filter, "function"); | 
 | 	  const char *function = NULL; | 
 |  | 
 | 	  if (py_func == NULL) | 
 | 	    return EXT_LANG_BT_ERROR; | 
 |  | 
 | 	  if (gdbpy_is_string (py_func.get ())) | 
 | 	    { | 
 | 	      function_to_free = python_string_to_host_string (py_func.get ()); | 
 |  | 
 | 	      if (function_to_free == NULL) | 
 | 		return EXT_LANG_BT_ERROR; | 
 |  | 
 | 	      function = function_to_free.get (); | 
 | 	    } | 
 | 	  else if (PyLong_Check (py_func.get ())) | 
 | 	    { | 
 | 	      CORE_ADDR addr; | 
 |  | 
 | 	      if (get_addr_from_python (py_func.get (), &addr) < 0) | 
 | 		return EXT_LANG_BT_ERROR; | 
 |  | 
 | 	      bound_minimal_symbol msymbol | 
 | 		= lookup_minimal_symbol_by_pc (addr); | 
 | 	      if (msymbol.minsym != NULL) | 
 | 		function = msymbol.minsym->print_name (); | 
 | 	    } | 
 | 	  else if (py_func != Py_None) | 
 | 	    { | 
 | 	      PyErr_SetString (PyExc_RuntimeError, | 
 | 			       _("FrameDecorator.function: expecting a " \ | 
 | 				 "String, integer or None.")); | 
 | 	      return EXT_LANG_BT_ERROR; | 
 | 	    } | 
 |  | 
 | 	  annotate_frame_function_name (); | 
 | 	  if (function == NULL) | 
 | 	    out->field_skip ("func"); | 
 | 	  else | 
 | 	    out->field_string ("func", function, function_name_style.style ()); | 
 | 	} | 
 |     } | 
 |  | 
 |  | 
 |   /* Frame arguments.  Check the result, and error if something went | 
 |      wrong.  */ | 
 |   if (print_args && (location_print || out->is_mi_like_p ())) | 
 |     { | 
 |       bool raw_frame_args = (flags & PRINT_RAW_FRAME_ARGUMENTS) != 0; | 
 |       if (py_print_args (filter, out, args_type, raw_frame_args, frame) | 
 | 	  == EXT_LANG_BT_ERROR) | 
 | 	return EXT_LANG_BT_ERROR; | 
 |     } | 
 |  | 
 |   /* File name/source/line number information.  */ | 
 |   bool print_location_source | 
 |     = ((location_print && print_what != SHORT_LOCATION) | 
 |        || (out->is_mi_like_p () && print_frame_info)); | 
 |   if (print_location_source) | 
 |     { | 
 |       annotate_frame_source_begin (); | 
 |  | 
 |       if (PyObject_HasAttrString (filter, "filename")) | 
 | 	{ | 
 | 	  gdbpy_ref<> py_fn = gdbpy_call_method (filter, "filename"); | 
 |  | 
 | 	  if (py_fn == NULL) | 
 | 	    return EXT_LANG_BT_ERROR; | 
 |  | 
 | 	  if (py_fn != Py_None) | 
 | 	    { | 
 | 	      gdb::unique_xmalloc_ptr<char> | 
 | 		filename (python_string_to_host_string (py_fn.get ())); | 
 |  | 
 | 	      if (filename == NULL) | 
 | 		return EXT_LANG_BT_ERROR; | 
 |  | 
 | 	      out->wrap_hint (3); | 
 | 	      out->text (" at "); | 
 | 	      annotate_frame_source_file (); | 
 | 	      out->field_string ("file", filename.get (), | 
 | 				 file_name_style.style ()); | 
 | 	      annotate_frame_source_file_end (); | 
 | 	    } | 
 | 	} | 
 |  | 
 |       if (PyObject_HasAttrString (filter, "line")) | 
 | 	{ | 
 | 	  gdbpy_ref<> py_line = gdbpy_call_method (filter, "line"); | 
 | 	  int line; | 
 |  | 
 | 	  if (py_line == NULL) | 
 | 	    return EXT_LANG_BT_ERROR; | 
 |  | 
 | 	  if (py_line != Py_None) | 
 | 	    { | 
 | 	      line = PyLong_AsLong (py_line.get ()); | 
 | 	      if (PyErr_Occurred ()) | 
 | 		return EXT_LANG_BT_ERROR; | 
 |  | 
 | 	      out->text (":"); | 
 | 	      annotate_frame_source_line (); | 
 | 	      out->field_signed ("line", line); | 
 | 	    } | 
 | 	} | 
 |       if (out->is_mi_like_p ()) | 
 | 	out->field_string ("arch", | 
 | 			   (gdbarch_bfd_arch_info (gdbarch))->printable_name); | 
 |     } | 
 |  | 
 |   bool source_print | 
 |     = (! out->is_mi_like_p () | 
 |        && (print_what == SRC_LINE || print_what == SRC_AND_LOC)); | 
 |   if (source_print) | 
 |     { | 
 |       if (print_location_source) | 
 | 	out->text ("\n"); /* Newline after the location source.  */ | 
 |       print_source_lines (sal.symtab, sal.line, sal.line + 1, 0); | 
 |     } | 
 |  | 
 |   /* For MI we need to deal with the "children" list population of | 
 |      elided frames, so if MI output detected do not send newline.  */ | 
 |   if (! out->is_mi_like_p ()) | 
 |     { | 
 |       annotate_frame_end (); | 
 |       /* print_source_lines has already printed a newline.  */ | 
 |       if (!source_print) | 
 | 	out->text ("\n"); | 
 |     } | 
 |  | 
 |   if (print_locals) | 
 |     { | 
 |       if (py_print_locals (filter, out, args_type, indent, | 
 | 			   frame) == EXT_LANG_BT_ERROR) | 
 | 	return EXT_LANG_BT_ERROR; | 
 |     } | 
 |  | 
 |   if ((flags & PRINT_HIDE) == 0) | 
 |     { | 
 |       /* Finally recursively print elided frames, if any.  */ | 
 |       gdbpy_ref<> elided (get_py_iter_from_func (filter, "elided")); | 
 |       if (elided == NULL) | 
 | 	return EXT_LANG_BT_ERROR; | 
 |  | 
 |       if (elided != Py_None) | 
 | 	{ | 
 | 	  PyObject *item; | 
 |  | 
 | 	  ui_out_emit_list inner_list_emiter (out, "children"); | 
 |  | 
 | 	  indent++; | 
 |  | 
 | 	  while ((item = PyIter_Next (elided.get ()))) | 
 | 	    { | 
 | 	      gdbpy_ref<> item_ref (item); | 
 |  | 
 | 	      enum ext_lang_bt_status success | 
 | 		= py_print_frame (item, flags, args_type, out, indent, | 
 | 				  levels_printed); | 
 |  | 
 | 	      if (success == EXT_LANG_BT_ERROR) | 
 | 		return EXT_LANG_BT_ERROR; | 
 | 	    } | 
 | 	  if (item == NULL && PyErr_Occurred ()) | 
 | 	    return EXT_LANG_BT_ERROR; | 
 | 	} | 
 |     } | 
 |  | 
 |   return EXT_LANG_BT_OK; | 
 | } | 
 |  | 
 | /* Helper function to initiate frame filter invocation at starting | 
 |    frame FRAME.  */ | 
 |  | 
 | static PyObject * | 
 | bootstrap_python_frame_filters (const frame_info_ptr &frame, | 
 | 				int frame_low, int frame_high) | 
 | { | 
 |   gdbpy_ref<> frame_obj (frame_info_to_frame_object (frame)); | 
 |   if (frame_obj == NULL) | 
 |     return NULL; | 
 |  | 
 |   gdbpy_ref<> module (PyImport_ImportModule ("gdb.frames")); | 
 |   if (module == NULL) | 
 |     return NULL; | 
 |  | 
 |   gdbpy_ref<> sort_func (PyObject_GetAttrString (module.get (), | 
 | 						 "execute_frame_filters")); | 
 |   if (sort_func == NULL) | 
 |     return NULL; | 
 |  | 
 |   gdbpy_ref<> py_frame_low = gdb_py_object_from_longest (frame_low); | 
 |   if (py_frame_low == NULL) | 
 |     return NULL; | 
 |  | 
 |   gdbpy_ref<> py_frame_high = gdb_py_object_from_longest (frame_high); | 
 |   if (py_frame_high == NULL) | 
 |     return NULL; | 
 |  | 
 |   gdbpy_ref<> iterable (PyObject_CallFunctionObjArgs (sort_func.get (), | 
 | 						      frame_obj.get (), | 
 | 						      py_frame_low.get (), | 
 | 						      py_frame_high.get (), | 
 | 						      NULL)); | 
 |   if (iterable == NULL) | 
 |     return NULL; | 
 |  | 
 |   if (iterable != Py_None) | 
 |     return PyObject_GetIter (iterable.get ()); | 
 |   else | 
 |     return iterable.release (); | 
 | } | 
 |  | 
 | /*  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 EXT_LANG_BT_ERROR on error, | 
 |     or EXT_LANG_BT_OK on success.  */ | 
 |  | 
 | enum ext_lang_bt_status | 
 | gdbpy_apply_frame_filter (const struct extension_language_defn *extlang, | 
 | 			  const frame_info_ptr &frame, frame_filter_flags flags, | 
 | 			  enum ext_lang_frame_args args_type, | 
 | 			  struct ui_out *out, int frame_low, int frame_high) | 
 | { | 
 |   struct gdbarch *gdbarch = NULL; | 
 |   enum ext_lang_bt_status success = EXT_LANG_BT_ERROR; | 
 |  | 
 |   if (!gdb_python_initialized) | 
 |     return EXT_LANG_BT_NO_FILTERS; | 
 |  | 
 |   try | 
 |     { | 
 |       gdbarch = get_frame_arch (frame); | 
 |     } | 
 |   catch (const gdb_exception_error &except) | 
 |     { | 
 |       /* Let gdb try to print the stack trace.  */ | 
 |       return EXT_LANG_BT_NO_FILTERS; | 
 |     } | 
 |  | 
 |   gdbpy_enter enter_py (gdbarch); | 
 |  | 
 |   /* When we're limiting the number of frames, be careful to request | 
 |      one extra frame, so that we can print a message if there are more | 
 |      frames.  */ | 
 |   int frame_countdown = -1; | 
 |   if ((flags & PRINT_MORE_FRAMES) != 0 && frame_low >= 0 && frame_high >= 0) | 
 |     { | 
 |       ++frame_high; | 
 |       /* This has an extra +1 because it is checked before a frame is | 
 | 	 printed.  */ | 
 |       frame_countdown = frame_high - frame_low + 1; | 
 |     } | 
 |  | 
 |   gdbpy_ref<> iterable (bootstrap_python_frame_filters (frame, frame_low, | 
 | 							frame_high)); | 
 |  | 
 |   if (iterable == NULL) | 
 |     { | 
 |       /* 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 EXT_LANG_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_or_quit (); | 
 |       return EXT_LANG_BT_NO_FILTERS; | 
 |     } | 
 |  | 
 |   /* 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.  */ | 
 |   if (iterable == Py_None) | 
 |     return EXT_LANG_BT_NO_FILTERS; | 
 |  | 
 |   htab_up levels_printed (htab_create (20, | 
 | 				       htab_hash_pointer, | 
 | 				       htab_eq_pointer, | 
 | 				       NULL)); | 
 |  | 
 |   while (true) | 
 |     { | 
 |       gdbpy_ref<> item (PyIter_Next (iterable.get ())); | 
 |  | 
 |       if (item == NULL) | 
 | 	{ | 
 | 	  if (PyErr_Occurred ()) | 
 | 	    { | 
 | 	      gdbpy_print_stack_or_quit (); | 
 | 	      return EXT_LANG_BT_ERROR; | 
 | 	    } | 
 | 	  break; | 
 | 	} | 
 |  | 
 |       if (frame_countdown != -1) | 
 | 	{ | 
 | 	  gdb_assert ((flags & PRINT_MORE_FRAMES) != 0); | 
 | 	  --frame_countdown; | 
 | 	  if (frame_countdown == 0) | 
 | 	    { | 
 | 	      /* We've printed all the frames we were asked to | 
 | 		 print, but more frames existed.  */ | 
 | 	      gdb_printf (_("(More stack frames follow...)\n")); | 
 | 	      break; | 
 | 	    } | 
 | 	} | 
 |  | 
 |       try | 
 | 	{ | 
 | 	  success = py_print_frame (item.get (), flags, args_type, out, 0, | 
 | 				    levels_printed.get ()); | 
 | 	} | 
 |       catch (const gdb_exception_error &except) | 
 | 	{ | 
 | 	  gdbpy_convert_exception (except); | 
 | 	  success = EXT_LANG_BT_ERROR; | 
 | 	} | 
 |  | 
 |       /* Do not exit on error printing a single frame.  Print the | 
 | 	 error and continue with other frames.  */ | 
 |       if (success == EXT_LANG_BT_ERROR) | 
 | 	gdbpy_print_stack_or_quit (); | 
 |     } | 
 |  | 
 |   return success; | 
 | } |