/* Python interface to MI commands Copyright (C) 2023 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 . */ #include "defs.h" #include "python-internal.h" #include "ui-out.h" #include "mi/mi-parse.h" /* A ui_out subclass that creates a Python object based on the data that is passed in. */ class py_ui_out : public ui_out { public: py_ui_out () : ui_out (fix_multi_location_breakpoint_output | fix_breakpoint_script_output) { do_begin (ui_out_type_tuple, nullptr); } bool can_emit_style_escape () const override { return false; } bool do_is_mi_like_p () const override { return true; } /* Return the Python object that was created. If a Python error occurred during the processing, set the Python error and return nullptr. */ PyObject *result () { if (m_error.has_value ()) { m_error->restore (); return nullptr; } return current ().obj.release (); } protected: void do_progress_end () override { } void do_progress_start () override { } void do_progress_notify (const std::string &, const char *, double, double) override { } void do_table_begin (int nbrofcols, int nr_rows, const char *tblid) override { do_begin (ui_out_type_list, tblid); } void do_table_body () override { } void do_table_end () override { do_end (ui_out_type_list); } void do_table_header (int width, ui_align align, const std::string &col_name, const std::string &col_hdr) override { } void do_begin (ui_out_type type, const char *id) override; void do_end (ui_out_type type) override; void do_field_signed (int fldno, int width, ui_align align, const char *fldname, LONGEST value) override; void do_field_unsigned (int fldno, int width, ui_align align, const char *fldname, ULONGEST value) override; void do_field_skip (int fldno, int width, ui_align align, const char *fldname) override { } void do_field_string (int fldno, int width, ui_align align, const char *fldname, const char *string, const ui_file_style &style) override; void do_field_fmt (int fldno, int width, ui_align align, const char *fldname, const ui_file_style &style, const char *format, va_list args) override ATTRIBUTE_PRINTF (7, 0); void do_spaces (int numspaces) override { } void do_text (const char *string) override { } void do_message (const ui_file_style &style, const char *format, va_list args) override ATTRIBUTE_PRINTF (3,0) { } void do_wrap_hint (int indent) override { } void do_flush () override { } void do_redirect (struct ui_file *outstream) override { } private: /* When constructing Python objects, this class keeps a stack of objects being constructed. Each such object has this type. */ struct object_desc { /* Name of the field (or empty for lists) that this object will eventually become. */ std::string field_name; /* The object under construction. */ gdbpy_ref<> obj; /* The type of structure being created. Note that tables are treated as lists here. */ ui_out_type type; }; /* The stack of objects being created. */ std::vector m_objects; /* If an error occurred, this holds the exception information for use by the 'release' method. */ gdb::optional m_error; /* Return a reference to the object under construction. */ object_desc ¤t () { return m_objects.back (); } /* Add a new field to the current object under construction. */ void add_field (const char *name, const gdbpy_ref<> &obj); }; void py_ui_out::add_field (const char *name, const gdbpy_ref<> &obj) { if (obj == nullptr) { m_error.emplace (); return; } object_desc &desc = current (); if (desc.type == ui_out_type_list) { if (PyList_Append (desc.obj.get (), obj.get ()) < 0) m_error.emplace (); } else { if (PyDict_SetItemString (desc.obj.get (), name, obj.get ()) < 0) m_error.emplace (); } } void py_ui_out::do_begin (ui_out_type type, const char *id) { if (m_error.has_value ()) return; gdbpy_ref<> new_obj (type == ui_out_type_list ? PyList_New (0) : PyDict_New ()); if (new_obj == nullptr) { m_error.emplace (); return; } object_desc new_desc; if (id != nullptr) new_desc.field_name = id; new_desc.obj = std::move (new_obj); new_desc.type = type; m_objects.push_back (std::move (new_desc)); } void py_ui_out::do_end (ui_out_type type) { if (m_error.has_value ()) return; object_desc new_obj = std::move (current ()); m_objects.pop_back (); add_field (new_obj.field_name.c_str (), new_obj.obj); } void py_ui_out::do_field_signed (int fldno, int width, ui_align align, const char *fldname, LONGEST value) { if (m_error.has_value ()) return; gdbpy_ref<> val = gdb_py_object_from_longest (value); add_field (fldname, val); } void py_ui_out::do_field_unsigned (int fldno, int width, ui_align align, const char *fldname, ULONGEST value) { if (m_error.has_value ()) return; gdbpy_ref<> val = gdb_py_object_from_ulongest (value); add_field (fldname, val); } void py_ui_out::do_field_string (int fldno, int width, ui_align align, const char *fldname, const char *string, const ui_file_style &style) { if (m_error.has_value ()) return; gdbpy_ref<> val = host_string_to_python_string (string); add_field (fldname, val); } void py_ui_out::do_field_fmt (int fldno, int width, ui_align align, const char *fldname, const ui_file_style &style, const char *format, va_list args) { if (m_error.has_value ()) return; std::string str = string_vprintf (format, args); do_field_string (fldno, width, align, fldname, str.c_str (), style); } /* Implementation of the gdb.execute_mi command. */ PyObject * gdbpy_execute_mi_command (PyObject *self, PyObject *args, PyObject *kw) { gdb::unique_xmalloc_ptr mi_command; std::vector> arg_strings; Py_ssize_t n_args = PyTuple_Size (args); if (n_args < 0) return nullptr; for (Py_ssize_t i = 0; i < n_args; ++i) { /* Note this returns a borrowed reference. */ PyObject *arg = PyTuple_GetItem (args, i); if (arg == nullptr) return nullptr; gdb::unique_xmalloc_ptr str = python_string_to_host_string (arg); if (str == nullptr) return nullptr; if (i == 0) mi_command = std::move (str); else arg_strings.push_back (std::move (str)); } py_ui_out uiout; try { scoped_restore save_uiout = make_scoped_restore (¤t_uiout, &uiout); std::unique_ptr parser = mi_parse::make (std::move (mi_command), std::move (arg_strings)); mi_execute_command (parser.get ()); } catch (const gdb_exception &except) { gdbpy_convert_exception (except); return nullptr; } return uiout.result (); }