Use an accessor function for general_symbol_info::language
[deliverable/binutils-gdb.git] / gdb / python / py-framefilter.c
index 6d22321ef1c6dc89da7612bdee503c72b6962033..fd03d313e9ea6260e474e1b52f292adee3f0544f 100644 (file)
@@ -1,6 +1,6 @@
 /* Python frame filters
 
-   Copyright (C) 2013-2017 Free Software Foundation, Inc.
+   Copyright (C) 2013-2019 Free Software Foundation, Inc.
 
    This file is part of GDB.
 
 #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 "py-ref.h"
+#include "gdbsupport/gdb_optional.h"
+#include "cli/cli-style.h"
 
 enum mi_print_types
 {
@@ -54,10 +57,10 @@ enum mi_print_types
 
 static enum ext_lang_bt_status
 extract_sym (PyObject *obj, gdb::unique_xmalloc_ptr<char> *name,
-            struct symbol **sym, struct block **sym_block,
+            struct symbol **sym, const struct block **sym_block,
             const struct language_defn **language)
 {
-  gdbpy_ref result (PyObject_CallMethod (obj, "symbol", NULL));
+  gdbpy_ref<> result (PyObject_CallMethod (obj, "symbol", NULL));
 
   if (result == NULL)
     return EXT_LANG_BT_ERROR;
@@ -100,13 +103,13 @@ extract_sym (PyObject *obj, gdb::unique_xmalloc_ptr<char> *name,
 
       /* Duplicate the symbol name, so the caller has consistency
         in garbage collection.  */
-      name->reset (xstrdup (SYMBOL_PRINT_NAME (*sym)));
+      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 (SYMBOL_LANGUAGE (*sym));
+       *language = language_def ((*sym)->language ());
       else
        *language = current_language;
     }
@@ -128,7 +131,7 @@ extract_value (PyObject *obj, struct value **value)
 {
   if (PyObject_HasAttrString (obj, "value"))
     {
-      gdbpy_ref vresult (PyObject_CallMethod (obj, "value", NULL));
+      gdbpy_ref<> vresult (PyObject_CallMethod (obj, "value", NULL));
 
       if (vresult == NULL)
        return EXT_LANG_BT_ERROR;
@@ -198,34 +201,16 @@ mi_should_print (struct symbol *sym, enum mi_print_types type)
 /* 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 EXT_LANG_BT_ERROR on error, with
-   any GDB exceptions converted to a Python exception, or EXT_LANG_BT_OK on
-   success.  */
+   type will be extracted from.  */
 
-static enum ext_lang_bt_status
+static void
 py_print_type (struct ui_out *out, struct value *val)
 {
+  check_typedef (value_type (val));
 
-  TRY
-    {
-      struct ui_file *stb;
-      struct cleanup *cleanup;
-
-      stb = mem_fileopen ();
-      cleanup = make_cleanup_ui_file_delete (stb);
-      check_typedef (value_type (val));
-      type_print (value_type (val), "", stb, -1);
-      out->field_stream ("type", stb);
-      do_cleanups (cleanup);
-    }
-  CATCH (except, RETURN_MASK_ALL)
-    {
-      gdbpy_convert_exception (except);
-      return EXT_LANG_BT_ERROR;
-    }
-  END_CATCH
-
-  return EXT_LANG_BT_OK;
+  string_file stb;
+  type_print (value_type (val), "", &stb, -1);
+  out->field_stream ("type", stb);
 }
 
 /* Helper function which outputs a value to an output field in a
@@ -233,11 +218,9 @@ py_print_type (struct ui_out *out, struct value *val)
    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 EXT_LANG_BT_ERROR on error, with any GDB
-   exceptions converted to a Python exception, or EXT_LANG_BT_OK on
-   success. */
+   will be printed with.  */
 
-static enum ext_lang_bt_status
+static void
 py_print_value (struct ui_out *out, struct value *val,
                const struct value_print_options *opts,
                int indent,
@@ -252,18 +235,7 @@ py_print_value (struct ui_out *out, struct value *val,
   if (args_type == MI_PRINT_SIMPLE_VALUES
       || args_type == MI_PRINT_ALL_VALUES)
     {
-      struct type *type = NULL;
-
-      TRY
-       {
-         type = check_typedef (value_type (val));
-       }
-      CATCH (except, RETURN_MASK_ALL)
-       {
-         gdbpy_convert_exception (except);
-         return EXT_LANG_BT_ERROR;
-       }
-      END_CATCH
+      struct type *type = check_typedef (value_type (val));
 
       if (args_type == MI_PRINT_ALL_VALUES)
        should_print = 1;
@@ -278,26 +250,11 @@ py_print_value (struct ui_out *out, struct value *val,
 
   if (should_print)
     {
-      TRY
-       {
-         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);
-         out->field_stream ("value", stb);
-         do_cleanups (cleanup);
-       }
-      CATCH (except, RETURN_MASK_ALL)
-       {
-         gdbpy_convert_exception (except);
-         return EXT_LANG_BT_ERROR;
-       }
-      END_CATCH
-    }
+      string_file stb;
 
-  return EXT_LANG_BT_OK;
+      common_val_print (val, &stb, indent, opts, language);
+      out->field_stream ("value", stb);
+    }
 }
 
 /* Helper function to call a Python method and extract an iterator
@@ -308,11 +265,11 @@ py_print_value (struct ui_out *out, struct value *val,
    This function can return an iterator, or NULL.  */
 
 static PyObject *
-get_py_iter_from_func (PyObject *filter, char *func)
+get_py_iter_from_func (PyObject *filter, const char *func)
 {
   if (PyObject_HasAttrString (filter, func))
     {
-      gdbpy_ref result (PyObject_CallMethod (filter, func, NULL));
+      gdbpy_ref<> result (PyObject_CallMethod (filter, func, NULL));
 
       if (result != NULL)
        {
@@ -345,10 +302,9 @@ get_py_iter_from_func (PyObject *filter, char *func)
     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 EXT_LANG_BT_ERROR on error, with any GDB exceptions
-    converted to a Python exception, or EXT_LANG_BT_OK on success.  */
+    printed.  */
 
-static enum ext_lang_bt_status
+static void
 py_print_single_arg (struct ui_out *out,
                     const char *sym_name,
                     struct frame_arg *fa,
@@ -359,124 +315,96 @@ py_print_single_arg (struct ui_out *out,
                     const struct language_defn *language)
 {
   struct value *val;
-  enum ext_lang_bt_status retval = EXT_LANG_BT_OK;
 
   if (fa != NULL)
     {
       if (fa->val == NULL && fa->error == NULL)
-       return EXT_LANG_BT_OK;
-      language = language_def (SYMBOL_LANGUAGE (fa->sym));
+       return;
+      language = language_def (fa->sym->language ());
       val = fa->val;
     }
   else
     val = fv;
 
-  TRY
-    {
-      struct cleanup *cleanups = make_cleanup (null_cleanup, NULL);
+  gdb::optional<ui_out_emit_tuple> maybe_tuple;
 
-      /*  MI has varying rules for tuples, but generally if there is only
+  /*  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)
-           make_cleanup_ui_out_tuple_begin_end (out, NULL);
-       }
+  if (out->is_mi_like_p ())
+    {
+      if (print_args_field || args_type != NO_VALUES)
+       maybe_tuple.emplace (out, nullptr);
+    }
 
-      annotate_arg_begin ();
+  annotate_arg_begin ();
+
+  /* If frame argument is populated, check for entry-values and the
+     entry value options.  */
+  if (fa != NULL)
+    {
+      string_file stb;
 
-      /* If frame argument is populated, check for entry-values and the
-        entry value options.  */
-      if (fa != NULL)
+      fprintf_symbol_filtered (&stb, fa->sym->print_name (),
+                              fa->sym->language (),
+                              DMGL_PARAMS | DMGL_ANSI);
+      if (fa->entry_kind == print_entry_values_compact)
        {
-         struct ui_file *stb;
+         stb.puts ("=");
 
-         stb = mem_fileopen ();
-         make_cleanup_ui_file_delete (stb);
-         fprintf_symbol_filtered (stb, SYMBOL_PRINT_NAME (fa->sym),
-                                  SYMBOL_LANGUAGE (fa->sym),
+         fprintf_symbol_filtered (&stb, fa->sym->print_name (),
+                                  fa->sym->language (),
                                   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);
-           }
-         out->field_stream ("name", stb);
        }
-      else
-       /* Otherwise, just output the name.  */
-       out->field_string ("name", sym_name);
+      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 ();
+  annotate_arg_name_end ();
 
-      if (! out->is_mi_like_p ())
-       out->text ("=");
+  out->text ("=");
 
-      if (print_args_field)
-       out->field_int ("arg", 1);
+  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)
-       {
-         if (py_print_type (out, val) == EXT_LANG_BT_ERROR)
-           {
-             retval = EXT_LANG_BT_ERROR;
-             do_cleanups (cleanups);
-           }
-       }
+  /* 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 (retval != EXT_LANG_BT_ERROR)
-       {
-         if (val != NULL)
-           annotate_arg_value (value_type (val));
+  if (val != NULL)
+    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 (! out->is_mi_like_p () && args_type == NO_VALUES)
-           out->field_string ("value", "...");
-         else
+  /* 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)
            {
-             /* 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",
-                                       _("<error reading variable: %s>"),
-                                       fa->error);
-                   }
-                 else if (py_print_value (out, val, opts, 0, args_type, language)
-                          == EXT_LANG_BT_ERROR)
-                   retval = EXT_LANG_BT_ERROR;
-               }
+             gdb_assert (fa != NULL && fa->error != NULL);
+             out->field_fmt ("value", metadata_style.style (),
+                             _("<error reading variable: %s>"),
+                             fa->error.get ());
            }
-
-         do_cleanups (cleanups);
+         else
+           py_print_value (out, val, opts, 0, args_type, language);
        }
     }
-  CATCH (except, RETURN_MASK_ERROR)
-    {
-      gdbpy_convert_exception (except);
-    }
-  END_CATCH
-
-  return retval;
 }
 
 /* Helper function to loop over frame arguments provided by the
@@ -497,7 +425,6 @@ enumerate_args (PyObject *iter,
                int print_args_field,
                struct frame_info *frame)
 {
-  PyObject *item;
   struct value_print_options opts;
 
   get_user_print_options (&opts);
@@ -510,50 +437,33 @@ enumerate_args (PyObject *iter,
 
   opts.deref_ref = 1;
 
-  TRY
-    {
-      annotate_frame_args ();
-    }
-  CATCH (except, RETURN_MASK_ALL)
-    {
-      gdbpy_convert_exception (except);
-      goto error;
-    }
-  END_CATCH
+  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.  */
-  item = PyIter_Next (iter);
+  gdbpy_ref<> item (PyIter_Next (iter));
   if (item == NULL && PyErr_Occurred ())
-    goto error;
+    return EXT_LANG_BT_ERROR;
 
-  while (item)
+  while (item != NULL)
     {
       const struct language_defn *language;
       gdb::unique_xmalloc_ptr<char> sym_name;
       struct symbol *sym;
-      struct block *sym_block;
+      const struct block *sym_block;
       struct value *val;
       enum ext_lang_bt_status success = EXT_LANG_BT_ERROR;
 
-      success = extract_sym (item, &sym_name, &sym, &sym_block, &language);
+      success = extract_sym (item.get (), &sym_name, &sym, &sym_block,
+                            &language);
       if (success == EXT_LANG_BT_ERROR)
-       {
-         Py_DECREF (item);
-         goto error;
-       }
+       return EXT_LANG_BT_ERROR;
 
-      success = extract_value (item, &val);
+      success = extract_value (item.get (), &val);
       if (success == EXT_LANG_BT_ERROR)
-       {
-         Py_DECREF (item);
-         goto error;
-       }
-
-      Py_DECREF (item);
-      item = NULL;
+       return EXT_LANG_BT_ERROR;
 
       if (sym && out->is_mi_like_p ()
          && ! mi_should_print (sym, MI_PRINT_ARGS))
@@ -571,19 +481,11 @@ enumerate_args (PyObject *iter,
            {
              PyErr_SetString (PyExc_RuntimeError,
                               _("No symbol or value provided."));
-             goto error;
+             return EXT_LANG_BT_ERROR;
            }
 
-         TRY
-           {
-             read_frame_arg (sym, frame, &arg, &entryarg);
-           }
-         CATCH (except, RETURN_MASK_ALL)
-           {
-             gdbpy_convert_exception (except);
-             goto error;
-           }
-         END_CATCH
+         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
@@ -591,100 +493,47 @@ enumerate_args (PyObject *iter,
 
          if (arg.entry_kind != print_entry_values_only)
            {
-             if (py_print_single_arg (out, NULL, &arg,
-                                      NULL, &opts,
-                                      args_type,
-                                      print_args_field,
-                                      NULL) == EXT_LANG_BT_ERROR)
-               {
-                 xfree (arg.error);
-                 xfree (entryarg.error);
-                 goto error;
-               }
+             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)
                {
-                 TRY
-                   {
-                     out->text (", ");
-                     out->wrap_hint ("    ");
-                   }
-                 CATCH (except, RETURN_MASK_ALL)
-                   {
-                     xfree (arg.error);
-                     xfree (entryarg.error);
-                     gdbpy_convert_exception (except);
-                     goto error;
-                   }
-                 END_CATCH
+                 out->text (", ");
+                 out->wrap_hint ("    ");
                }
 
-             if (py_print_single_arg (out, NULL, &entryarg, NULL, &opts,
-                                      args_type, print_args_field, NULL)
-                 == EXT_LANG_BT_ERROR)
-               {
-                     xfree (arg.error);
-                     xfree (entryarg.error);
-                     goto error;
-               }
+             py_print_single_arg (out, NULL, &entryarg, NULL, &opts,
+                                  args_type, print_args_field, NULL);
            }
-
-         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.get (), NULL, val, &opts,
-                                      args_type, print_args_field,
-                                      language) == EXT_LANG_BT_ERROR)
-               goto error;
-           }
+           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 = PyIter_Next (iter);
+      item.reset (PyIter_Next (iter));
       if (item != NULL)
-       {
-         TRY
-           {
-             out->text (", ");
-           }
-         CATCH (except, RETURN_MASK_ALL)
-           {
-             Py_DECREF (item);
-             gdbpy_convert_exception (except);
-             goto error;
-           }
-         END_CATCH
-       }
+       out->text (", ");
       else if (PyErr_Occurred ())
-       goto error;
+       return EXT_LANG_BT_ERROR;
 
-      TRY
-       {
-         annotate_arg_end ();
-       }
-      CATCH (except, RETURN_MASK_ALL)
-       {
-         Py_DECREF (item);
-         gdbpy_convert_exception (except);
-         goto error;
-       }
-      END_CATCH
+      annotate_arg_end ();
     }
 
   return EXT_LANG_BT_OK;
-
- error:
-  return EXT_LANG_BT_ERROR;
 }
 
 
@@ -707,61 +556,42 @@ enumerate_locals (PyObject *iter,
                  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)))
+  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;
-      struct block *sym_block;
+      const struct block *sym_block;
       int local_indent = 8 + (8 * indent);
-      struct cleanup *locals_cleanups;
+      gdb::optional<ui_out_emit_tuple> tuple;
 
-      locals_cleanups = make_cleanup_py_decref (item);
+      gdbpy_ref<> item (PyIter_Next (iter));
+      if (item == NULL)
+       break;
 
-      success = extract_sym (item, &sym_name, &sym, &sym_block, &language);
+      success = extract_sym (item.get (), &sym_name, &sym, &sym_block,
+                            &language);
       if (success == EXT_LANG_BT_ERROR)
-       {
-         do_cleanups (locals_cleanups);
-         goto error;
-       }
+       return EXT_LANG_BT_ERROR;
 
-      success = extract_value (item, &val);
+      success = extract_value (item.get (), &val);
       if (success == EXT_LANG_BT_ERROR)
-       {
-         do_cleanups (locals_cleanups);
-         goto error;
-       }
+       return EXT_LANG_BT_ERROR;
 
       if (sym != NULL && out->is_mi_like_p ()
          && ! mi_should_print (sym, MI_PRINT_LOCALS))
-       {
-         do_cleanups (locals_cleanups);
-         continue;
-       }
+       continue;
 
       /* If the object did not provide a value, read it.  */
       if (val == NULL)
-       {
-         TRY
-           {
-             val = read_var_value (sym, sym_block, frame);
-           }
-         CATCH (except, RETURN_MASK_ERROR)
-           {
-             gdbpy_convert_exception (except);
-             do_cleanups (locals_cleanups);
-             goto error;
-           }
-         END_CATCH
-       }
+       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
@@ -769,37 +599,16 @@ enumerate_locals (PyObject *iter,
       if (out->is_mi_like_p ())
        {
          if (print_args_field || args_type != NO_VALUES)
-           make_cleanup_ui_out_tuple_begin_end (out, NULL);
+           tuple.emplace (out, nullptr);
        }
-      TRY
-       {
-         if (! out->is_mi_like_p ())
-           {
-             /* If the output is not MI we indent locals.  */
-             out->spaces (local_indent);
-           }
 
-         out->field_string ("name", sym_name.get ());
-
-         if (! out->is_mi_like_p ())
-           out->text (" = ");
-       }
-      CATCH (except, RETURN_MASK_ERROR)
-       {
-         gdbpy_convert_exception (except);
-         do_cleanups (locals_cleanups);
-         goto error;
-       }
-      END_CATCH
+      /* 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)
-       {
-         if (py_print_type (out, val) == EXT_LANG_BT_ERROR)
-           {
-             do_cleanups (locals_cleanups);
-             goto error;
-           }
-       }
+       py_print_type (out, val);
 
       /* CLI always prints values for locals.  MI uses the
         simple/no/all system.  */
@@ -807,46 +616,22 @@ enumerate_locals (PyObject *iter,
        {
          int val_indent = (indent + 1) * 4;
 
-         if (py_print_value (out, val, &opts, val_indent, args_type,
-                             language) == EXT_LANG_BT_ERROR)
-           {
-             do_cleanups (locals_cleanups);
-             goto error;
-           }
+         py_print_value (out, val, &opts, val_indent, args_type,
+                         language);
        }
       else
        {
          if (args_type != NO_VALUES)
-           {
-             if (py_print_value (out, val, &opts, 0, args_type,
-                                 language) == EXT_LANG_BT_ERROR)
-               {
-                 do_cleanups (locals_cleanups);
-                 goto error;
-               }
-           }
+           py_print_value (out, val, &opts, 0, args_type,
+                           language);
        }
 
-      do_cleanups (locals_cleanups);
-
-      TRY
-       {
-         out->text ("\n");
-       }
-      CATCH (except, RETURN_MASK_ERROR)
-       {
-         gdbpy_convert_exception (except);
-         goto error;
-       }
-      END_CATCH
+      out->text ("\n");
     }
 
-  if (item == NULL && PyErr_Occurred ())
-    goto error;
-
-  return EXT_LANG_BT_OK;
+  if (!PyErr_Occurred ())
+    return EXT_LANG_BT_OK;
 
- error:
   return EXT_LANG_BT_ERROR;
 }
 
@@ -859,38 +644,27 @@ py_mi_print_variables (PyObject *filter, struct ui_out *out,
                       enum ext_lang_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);
+  gdbpy_ref<> args_iter (get_py_iter_from_func (filter, "frame_args"));
   if (args_iter == NULL)
-    goto error;
+    return EXT_LANG_BT_ERROR;
 
-  locals_iter = get_py_iter_from_func (filter, "frame_locals");
+  gdbpy_ref<> locals_iter (get_py_iter_from_func (filter, "frame_locals"));
   if (locals_iter == NULL)
-    goto error;
+    return EXT_LANG_BT_ERROR;
 
-  make_cleanup_py_decref (locals_iter);
-  make_cleanup_ui_out_list_begin_end (out, "variables");
+  ui_out_emit_list list_emitter (out, "variables");
 
-  if (args_iter != Py_None)
-    if (enumerate_args (args_iter, out, args_type, 1, frame)
-       == EXT_LANG_BT_ERROR)
-      goto error;
+  if (args_iter != Py_None
+      && (enumerate_args (args_iter.get (), out, args_type, 1, frame)
+         == EXT_LANG_BT_ERROR))
+    return EXT_LANG_BT_ERROR;
 
-  if (locals_iter != Py_None)
-    if (enumerate_locals (locals_iter, out, 1, args_type, 1, frame)
-       == EXT_LANG_BT_ERROR)
-      goto 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;
 
-  do_cleanups (old_chain);
   return EXT_LANG_BT_OK;
-
- error:
-  do_cleanups (old_chain);
-  return EXT_LANG_BT_ERROR;
 }
 
 /* Helper function for printing locals.  This function largely just
@@ -904,26 +678,18 @@ py_print_locals (PyObject *filter,
                 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);
-
+  gdbpy_ref<> locals_iter (get_py_iter_from_func (filter, "frame_locals"));
   if (locals_iter == NULL)
-    goto locals_error;
+    return EXT_LANG_BT_ERROR;
 
-  make_cleanup_ui_out_list_begin_end (out, "locals");
+  ui_out_emit_list list_emitter (out, "locals");
 
-  if (locals_iter != Py_None)
-    if (enumerate_locals (locals_iter, out, indent, args_type,
-                         0, frame) == EXT_LANG_BT_ERROR)
-      goto locals_error;
+  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;
 
-  do_cleanups (old_chain);
   return EXT_LANG_BT_OK;
-
- locals_error:
-  do_cleanups (old_chain);
-  return EXT_LANG_BT_ERROR;
 }
 
 /* Helper function for printing frame arguments.  This function
@@ -937,50 +703,36 @@ py_print_args (PyObject *filter,
               enum ext_lang_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);
-
+  gdbpy_ref<> args_iter (get_py_iter_from_func (filter, "frame_args"));
   if (args_iter == NULL)
-    goto args_error;
-
-  make_cleanup_ui_out_list_begin_end (out, "args");
+    return EXT_LANG_BT_ERROR;
 
-  TRY
-    {
-      annotate_frame_args ();
-      if (! out->is_mi_like_p ())
-       out->text (" (");
-    }
-  CATCH (except, RETURN_MASK_ALL)
-    {
-      gdbpy_convert_exception (except);
-      goto args_error;
-    }
-  END_CATCH
+  ui_out_emit_list list_emitter (out, "args");
 
-  if (args_iter != Py_None)
-    if (enumerate_args (args_iter, out, args_type, 0, frame)
-       == EXT_LANG_BT_ERROR)
-      goto args_error;
+  out->wrap_hint ("   ");
+  annotate_frame_args ();
+  out->text (" (");
 
-  TRY
-    {
-      if (! out->is_mi_like_p ())
-       out->text (")");
-    }
-  CATCH (except, RETURN_MASK_ALL)
+  if (args_type == CLI_PRESENCE)
     {
-      gdbpy_convert_exception (except);
-      goto args_error;
+      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;
+       }
     }
-  END_CATCH
+  else if (args_iter != Py_None
+          && (enumerate_args (args_iter.get (), out, args_type, 0, frame)
+              == EXT_LANG_BT_ERROR))
+    return EXT_LANG_BT_ERROR;
 
-  do_cleanups (old_chain);
-  return EXT_LANG_BT_OK;
+  out->text (")");
 
- args_error:
-  do_cleanups (old_chain);
-  return EXT_LANG_BT_ERROR;
+  return EXT_LANG_BT_OK;
 }
 
 /*  Print a single frame to the designated output stream, detecting
@@ -995,11 +747,11 @@ py_print_args (PyObject *filter,
     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_COMPLETED
+    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, int flags,
+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)
 {
@@ -1007,10 +759,17 @@ py_print_frame (PyObject *filter, int flags,
   CORE_ADDR address = 0;
   struct gdbarch *gdbarch = NULL;
   struct frame_info *frame = NULL;
-  struct cleanup *cleanup_stack;
   struct value_print_options opts;
-  PyObject *py_inf_frame;
+
   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.  */
@@ -1020,31 +779,33 @@ py_print_frame (PyObject *filter, int flags,
   print_locals = (flags & PRINT_LOCALS) ? 1 : 0;
 
   get_user_print_options (&opts);
+  if (print_frame_info)
+  {
+    gdb::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.  */
-  py_inf_frame = PyObject_CallMethod (filter, "inferior_frame", NULL);
+  gdbpy_ref<> py_inf_frame (PyObject_CallMethod (filter, "inferior_frame",
+                                                NULL));
   if (py_inf_frame == NULL)
     return EXT_LANG_BT_ERROR;
 
-  frame = frame_object_to_frame_info (py_inf_frame);;
-
-  Py_DECREF (py_inf_frame);
-
+  frame = frame_object_to_frame_info (py_inf_frame.get ());
   if (frame == NULL)
     return EXT_LANG_BT_ERROR;
 
-  TRY
-    {
-      gdbarch = get_frame_arch (frame);
-    }
-  CATCH (except, RETURN_MASK_ERROR)
-    {
-      gdbpy_convert_exception (except);
-      return EXT_LANG_BT_ERROR;
-    }
-  END_CATCH
+  symtab_and_line sal = find_frame_sal (frame);
+
+  gdbarch = get_frame_arch (frame);
 
   /* stack-list-variables.  */
   if (print_locals && print_args && ! print_frame_info)
@@ -1052,379 +813,269 @@ py_print_frame (PyObject *filter, int flags,
       if (py_mi_print_variables (filter, out, &opts,
                                 args_type, frame) == EXT_LANG_BT_ERROR)
        return EXT_LANG_BT_ERROR;
-      return EXT_LANG_BT_COMPLETED;
+      return EXT_LANG_BT_OK;
     }
 
-  cleanup_stack = make_cleanup (null_cleanup, NULL);
+  gdb::optional<ui_out_emit_tuple> tuple;
 
   /* -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");
+    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)
-       {
-         TRY
-           {
-             out->spaces (indent * 4);
-           }
-         CATCH (except, RETURN_MASK_ERROR)
-           {
-             gdbpy_convert_exception (except);
-             do_cleanups (cleanup_stack);
-             return EXT_LANG_BT_ERROR;
-           }
-         END_CATCH
-       }
+       out->spaces (indent * 4);
 
       /* The address is required for frame annotations, and also for
         address printing.  */
       if (PyObject_HasAttrString (filter, "address"))
        {
-         PyObject *paddr = PyObject_CallMethod (filter, "address", NULL);
+         gdbpy_ref<> paddr (PyObject_CallMethod (filter, "address", NULL));
 
          if (paddr == NULL)
-           {
-             do_cleanups (cleanup_stack);
-             return EXT_LANG_BT_ERROR;
-           }
+           return EXT_LANG_BT_ERROR;
 
          if (paddr != Py_None)
            {
-             if (get_addr_from_python (paddr, &address) < 0)
-               {
-                 Py_DECREF (paddr);
-                 do_cleanups (cleanup_stack);
-                 return EXT_LANG_BT_ERROR;
-               }
+             if (get_addr_from_python (paddr.get (), &address) < 0)
+               return EXT_LANG_BT_ERROR;
 
              has_addr = 1;
            }
-         Py_DECREF (paddr);
        }
     }
 
+  /* 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_frame_info || print_args) && print_level)
+  if (print_level
+      && (location_print
+         || (out->is_mi_like_p () && (print_frame_info || print_args))))
     {
       struct frame_info **slot;
       int level;
 
       slot = (struct frame_info **) htab_find_slot (levels_printed,
                                                    frame, INSERT);
-      TRY
-       {
-         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;
-             annotate_frame_begin (print_level ? level : 0,
-                                   gdbarch, address);
-             out->text ("#");
-             out->field_fmt_int (2, ui_left, "level",
-                                   level);
-           }
-       }
-      CATCH (except, RETURN_MASK_ERROR)
+
+      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
        {
-         gdbpy_convert_exception (except);
-         do_cleanups (cleanup_stack);
-         return EXT_LANG_BT_ERROR;
+         *slot = frame;
+         annotate_frame_begin (print_level ? level : 0,
+                               gdbarch, address);
+         out->text ("#");
+         out->field_fmt_signed (2, ui_left, "level", level);
        }
-      END_CATCH
     }
 
-  if (print_frame_info)
+  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)
        {
-         TRY
+         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 ");
            }
-         CATCH (except, RETURN_MASK_ERROR)
-           {
-             gdbpy_convert_exception (except);
-             do_cleanups (cleanup_stack);
-             return EXT_LANG_BT_ERROR;
-           }
-         END_CATCH
        }
 
       /* Print frame function name.  */
       if (PyObject_HasAttrString (filter, "function"))
        {
-         PyObject *py_func = PyObject_CallMethod (filter, "function", NULL);
-         struct cleanup *py_func_cleanup;
+         gdbpy_ref<> py_func (PyObject_CallMethod (filter, "function", NULL));
          const char *function = NULL;
 
          if (py_func == NULL)
-           {
-             do_cleanups (cleanup_stack);
-             return EXT_LANG_BT_ERROR;
-           }
-         py_func_cleanup = make_cleanup_py_decref (py_func);
+           return EXT_LANG_BT_ERROR;
 
-         if (gdbpy_is_string (py_func))
+         if (gdbpy_is_string (py_func.get ()))
            {
-             function_to_free = python_string_to_host_string (py_func);
+             function_to_free = python_string_to_host_string (py_func.get ());
 
              if (function_to_free == NULL)
-               {
-                 do_cleanups (cleanup_stack);
-                 return EXT_LANG_BT_ERROR;
-               }
+               return EXT_LANG_BT_ERROR;
 
              function = function_to_free.get ();
            }
-         else if (PyLong_Check (py_func))
+         else if (PyLong_Check (py_func.get ()))
            {
              CORE_ADDR addr;
              struct bound_minimal_symbol msymbol;
 
-             if (get_addr_from_python (py_func, &addr) < 0)
-               {
-                 do_cleanups (cleanup_stack);
-                 return EXT_LANG_BT_ERROR;
-               }
+             if (get_addr_from_python (py_func.get (), &addr) < 0)
+               return EXT_LANG_BT_ERROR;
 
              msymbol = lookup_minimal_symbol_by_pc (addr);
              if (msymbol.minsym != NULL)
-               function = MSYMBOL_PRINT_NAME (msymbol.minsym);
+               function = msymbol.minsym->print_name ();
            }
          else if (py_func != Py_None)
            {
              PyErr_SetString (PyExc_RuntimeError,
                               _("FrameDecorator.function: expecting a " \
                                 "String, integer or None."));
-             do_cleanups (cleanup_stack);
-             return EXT_LANG_BT_ERROR;
-           }
-
-         TRY
-           {
-             annotate_frame_function_name ();
-             if (function == NULL)
-               out->field_skip ("func");
-             else
-               out->field_string ("func", function);
-           }
-         CATCH (except, RETURN_MASK_ERROR)
-           {
-             gdbpy_convert_exception (except);
-             do_cleanups (cleanup_stack);
              return EXT_LANG_BT_ERROR;
            }
-         END_CATCH
 
-         do_cleanups (py_func_cleanup);
+         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)
+  if (print_args && (location_print || out->is_mi_like_p ()))
     {
       if (py_print_args (filter, out, args_type, frame) == EXT_LANG_BT_ERROR)
-       {
-         do_cleanups (cleanup_stack);
-         return EXT_LANG_BT_ERROR;
-       }
+       return EXT_LANG_BT_ERROR;
     }
 
   /* File name/source/line number information.  */
-  if (print_frame_info)
+  bool print_location_source
+    = ((location_print && print_what != SHORT_LOCATION)
+       || (out->is_mi_like_p () && print_frame_info));
+  if (print_location_source)
     {
-      TRY
-       {
-         annotate_frame_source_begin ();
-       }
-      CATCH (except, RETURN_MASK_ERROR)
-       {
-         gdbpy_convert_exception (except);
-         do_cleanups (cleanup_stack);
-         return EXT_LANG_BT_ERROR;
-       }
-      END_CATCH
+      annotate_frame_source_begin ();
 
       if (PyObject_HasAttrString (filter, "filename"))
        {
-         PyObject *py_fn = PyObject_CallMethod (filter, "filename", NULL);
-         struct cleanup *py_fn_cleanup;
+         gdbpy_ref<> py_fn (PyObject_CallMethod (filter, "filename", NULL));
 
          if (py_fn == NULL)
-           {
-             do_cleanups (cleanup_stack);
-             return EXT_LANG_BT_ERROR;
-           }
-         py_fn_cleanup = make_cleanup_py_decref (py_fn);
+           return EXT_LANG_BT_ERROR;
 
          if (py_fn != Py_None)
            {
              gdb::unique_xmalloc_ptr<char>
-               filename (python_string_to_host_string (py_fn));
+               filename (python_string_to_host_string (py_fn.get ()));
 
              if (filename == NULL)
-               {
-                 do_cleanups (cleanup_stack);
-                 return EXT_LANG_BT_ERROR;
-               }
+               return EXT_LANG_BT_ERROR;
 
-             TRY
-               {
-                 out->wrap_hint ("   ");
-                 out->text (" at ");
-                 annotate_frame_source_file ();
-                 out->field_string ("file", filename.get ());
-                 annotate_frame_source_file_end ();
-               }
-             CATCH (except, RETURN_MASK_ERROR)
-               {
-                 gdbpy_convert_exception (except);
-                 do_cleanups (cleanup_stack);
-                 return EXT_LANG_BT_ERROR;
-               }
-             END_CATCH
+             out->wrap_hint ("   ");
+             out->text (" at ");
+             annotate_frame_source_file ();
+             out->field_string ("file", filename.get (),
+                                file_name_style.style ());
+             annotate_frame_source_file_end ();
            }
-         do_cleanups (py_fn_cleanup);
        }
 
       if (PyObject_HasAttrString (filter, "line"))
        {
-         PyObject *py_line = PyObject_CallMethod (filter, "line", NULL);
-         struct cleanup *py_line_cleanup;
+         gdbpy_ref<> py_line (PyObject_CallMethod (filter, "line", NULL));
          int line;
 
          if (py_line == NULL)
-           {
-             do_cleanups (cleanup_stack);
-             return EXT_LANG_BT_ERROR;
-           }
-         py_line_cleanup = make_cleanup_py_decref (py_line);
+           return EXT_LANG_BT_ERROR;
 
          if (py_line != Py_None)
            {
-             line = PyLong_AsLong (py_line);
+             line = PyLong_AsLong (py_line.get ());
              if (PyErr_Occurred ())
-               {
-                 do_cleanups (cleanup_stack);
-                 return EXT_LANG_BT_ERROR;
-               }
+               return EXT_LANG_BT_ERROR;
 
-             TRY
-               {
-                 out->text (":");
-                 annotate_frame_source_line ();
-                 out->field_int ("line", line);
-               }
-             CATCH (except, RETURN_MASK_ERROR)
-               {
-                 gdbpy_convert_exception (except);
-                 do_cleanups (cleanup_stack);
-                 return EXT_LANG_BT_ERROR;
-               }
-             END_CATCH
+             out->text (":");
+             annotate_frame_source_line ();
+             out->field_signed ("line", line);
            }
-         do_cleanups (py_line_cleanup);
        }
+      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 ())
     {
-      TRY
-       {
-         annotate_frame_end ();
-         out->text ("\n");
-       }
-      CATCH (except, RETURN_MASK_ERROR)
-       {
-         gdbpy_convert_exception (except);
-         do_cleanups (cleanup_stack);
-         return EXT_LANG_BT_ERROR;
-       }
-      END_CATCH
+      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)
-       {
-         do_cleanups (cleanup_stack);
-         return EXT_LANG_BT_ERROR;
-       }
+       return EXT_LANG_BT_ERROR;
     }
 
-  {
-    PyObject *elided;
-    struct cleanup *elided_cleanup;
-
-    /* Finally recursively print elided frames, if any.  */
-    elided = get_py_iter_from_func (filter, "elided");
-    if (elided == NULL)
-      {
-       do_cleanups (cleanup_stack);
+  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;
-      }
-    elided_cleanup = make_cleanup_py_decref (elided);
 
-    if (elided != Py_None)
-      {
-       PyObject *item;
+      if (elided != Py_None)
+       {
+         PyObject *item;
 
-       make_cleanup_ui_out_list_begin_end (out, "children");
+         ui_out_emit_list inner_list_emiter (out, "children");
 
-       if (! out->is_mi_like_p ())
          indent++;
 
-       while ((item = PyIter_Next (elided)))
-         {
-           struct cleanup *item_cleanup = make_cleanup_py_decref (item);
-
-           enum ext_lang_bt_status success = py_print_frame (item, flags,
-                                                             args_type, out,
-                                                             indent,
-                                                             levels_printed);
+         while ((item = PyIter_Next (elided.get ())))
+           {
+             gdbpy_ref<> item_ref (item);
 
-           do_cleanups (item_cleanup);
+             enum ext_lang_bt_status success
+               = py_print_frame (item, flags, args_type, out, indent,
+                                 levels_printed);
 
-           if (success == EXT_LANG_BT_ERROR)
-             {
-               do_cleanups (cleanup_stack);
+             if (success == EXT_LANG_BT_ERROR)
                return EXT_LANG_BT_ERROR;
-             }
-         }
-       if (item == NULL && PyErr_Occurred ())
-         {
-           do_cleanups (cleanup_stack);
+           }
+         if (item == NULL && PyErr_Occurred ())
            return EXT_LANG_BT_ERROR;
-         }
-      }
-    do_cleanups (elided_cleanup);
-  }
+       }
+    }
 
-  do_cleanups (cleanup_stack);
-  return EXT_LANG_BT_COMPLETED;
+  return EXT_LANG_BT_OK;
 }
 
 /* Helper function to initiate frame filter invocation at starting
@@ -1434,32 +1085,32 @@ static PyObject *
 bootstrap_python_frame_filters (struct frame_info *frame,
                                int frame_low, int frame_high)
 {
-  gdbpy_ref frame_obj (frame_info_to_frame_object (frame));
+  gdbpy_ref<> frame_obj (frame_info_to_frame_object (frame));
   if (frame_obj == NULL)
     return NULL;
 
-  gdbpy_ref module (PyImport_ImportModule ("gdb.frames"));
+  gdbpy_ref<> module (PyImport_ImportModule ("gdb.frames"));
   if (module == NULL)
     return NULL;
 
-  gdbpy_ref sort_func (PyObject_GetAttrString (module.get (),
-                                              "execute_frame_filters"));
+  gdbpy_ref<> sort_func (PyObject_GetAttrString (module.get (),
+                                                "execute_frame_filters"));
   if (sort_func == NULL)
     return NULL;
 
-  gdbpy_ref py_frame_low (PyInt_FromLong (frame_low));
+  gdbpy_ref<> py_frame_low (PyInt_FromLong (frame_low));
   if (py_frame_low == NULL)
     return NULL;
 
-  gdbpy_ref py_frame_high (PyInt_FromLong (frame_high));
+  gdbpy_ref<> py_frame_high (PyInt_FromLong (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));
+  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;
 
@@ -1482,38 +1133,46 @@ bootstrap_python_frame_filters (struct frame_info *frame,
     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_COMPLETED on success.  */
+    or EXT_LANG_BT_OK on success.  */
 
 enum ext_lang_bt_status
 gdbpy_apply_frame_filter (const struct extension_language_defn *extlang,
-                         struct frame_info *frame, int flags,
+                         struct frame_info *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;
-  struct cleanup *cleanups;
   enum ext_lang_bt_status success = EXT_LANG_BT_ERROR;
-  PyObject *iterable;
-  PyObject *item;
-  htab_t levels_printed;
 
   if (!gdb_python_initialized)
     return EXT_LANG_BT_NO_FILTERS;
 
-  TRY
+  try
     {
       gdbarch = get_frame_arch (frame);
     }
-  CATCH (except, RETURN_MASK_ALL)
+  catch (const gdb_exception_error &except)
     {
       /* Let gdb try to print the stack trace.  */
       return EXT_LANG_BT_NO_FILTERS;
     }
-  END_CATCH
 
-  cleanups = ensure_python_env (gdbarch, current_language);
+  gdbpy_enter enter_py (gdbarch, current_language);
 
-  iterable = bootstrap_python_frame_filters (frame, frame_low, frame_high);
+  /* 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)
     {
@@ -1530,53 +1189,64 @@ gdbpy_apply_frame_filter (const struct extension_language_defn *extlang,
         initialization error.  This return code will trigger a
         default backtrace.  */
 
-      gdbpy_print_stack ();
-      do_cleanups (cleanups);
+      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.  */
-  make_cleanup_py_decref (iterable);
   if (iterable == Py_None)
-    {
-      success = EXT_LANG_BT_NO_FILTERS;
-      goto done;
-    }
+    return EXT_LANG_BT_NO_FILTERS;
 
-  levels_printed = htab_create (20,
-                               htab_hash_pointer,
-                               htab_eq_pointer,
-                               NULL);
-  make_cleanup_htab_delete (levels_printed);
+  htab_up levels_printed (htab_create (20,
+                                      htab_hash_pointer,
+                                      htab_eq_pointer,
+                                      NULL));
 
-  while ((item = PyIter_Next (iterable)))
+  while (true)
     {
-      struct cleanup *item_cleanup = make_cleanup_py_decref (item);
+      gdbpy_ref<> item (PyIter_Next (iterable.get ()));
+
+      if (item == NULL)
+       {
+         if (PyErr_Occurred ())
+           {
+             gdbpy_print_stack_or_quit ();
+             return EXT_LANG_BT_ERROR;
+           }
+         break;
+       }
 
-      success = py_print_frame (item, flags, args_type, out, 0,
-                               levels_printed);
+      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.  */
+             printf_filtered (_("(More stack frames follow...)\n"));
+             break;
+           }
+       }
 
-      do_cleanups (item_cleanup);
+      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 ();
+       gdbpy_print_stack_or_quit ();
     }
 
-  if (item == NULL && PyErr_Occurred ())
-    goto error;
-
- done:
-  do_cleanups (cleanups);
   return success;
-
-  /* Exit and abandon backtrace on error, printing the exception that
-     is set.  */
- error:
-  gdbpy_print_stack ();
-  do_cleanups (cleanups);
-  return EXT_LANG_BT_ERROR;
 }
This page took 0.043822 seconds and 4 git commands to generate.