|  | /* Output generating routines for GDB. | 
|  |  | 
|  | Copyright (C) 1999-2024 Free Software Foundation, Inc. | 
|  |  | 
|  | Contributed by Cygnus Solutions. | 
|  | Written by Fernando Nasser for Cygnus. | 
|  |  | 
|  | 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 "expression.h" | 
|  | #include "language.h" | 
|  | #include "ui-out.h" | 
|  | #include "gdbsupport/format.h" | 
|  | #include "cli/cli-style.h" | 
|  | #include "diagnostics.h" | 
|  |  | 
|  | #include <vector> | 
|  | #include <memory> | 
|  | #include <string> | 
|  |  | 
|  | namespace { | 
|  |  | 
|  | /* A header of a ui_out_table.  */ | 
|  |  | 
|  | class ui_out_hdr | 
|  | { | 
|  | public: | 
|  |  | 
|  | explicit ui_out_hdr (int number, int min_width, ui_align alignment, | 
|  | const std::string &name, const std::string &header) | 
|  | : m_number (number), | 
|  | m_min_width (min_width), | 
|  | m_alignment (alignment), | 
|  | m_name (name), | 
|  | m_header (header) | 
|  | { | 
|  | } | 
|  |  | 
|  | int number () const | 
|  | { | 
|  | return m_number; | 
|  | } | 
|  |  | 
|  | int min_width () const | 
|  | { | 
|  | return m_min_width; | 
|  | } | 
|  |  | 
|  | ui_align alignment () const | 
|  | { | 
|  | return m_alignment; | 
|  | } | 
|  |  | 
|  | const std::string &header () const | 
|  | { | 
|  | return m_header; | 
|  | } | 
|  |  | 
|  | const std::string &name () const | 
|  | { | 
|  | return m_name; | 
|  | } | 
|  |  | 
|  | private: | 
|  |  | 
|  | /* The number of the table column this header represents, 1-based.  */ | 
|  | int m_number; | 
|  |  | 
|  | /* Minimal column width in characters.  May or may not be applicable, | 
|  | depending on the actual implementation of ui_out.  */ | 
|  | int m_min_width; | 
|  |  | 
|  | /* Alignment of the content in the column.  May or may not be applicable, | 
|  | depending on the actual implementation of ui_out.  */ | 
|  | ui_align m_alignment; | 
|  |  | 
|  | /* Internal column name, used to internally refer to the column.  */ | 
|  | std::string m_name; | 
|  |  | 
|  | /* Printed header text of the column.  */ | 
|  | std::string m_header; | 
|  | }; | 
|  |  | 
|  | } // namespace | 
|  |  | 
|  | /* A level of nesting (either a list or a tuple) in a ui_out output.  */ | 
|  |  | 
|  | class ui_out_level | 
|  | { | 
|  | public: | 
|  |  | 
|  | explicit ui_out_level (ui_out_type type) | 
|  | : m_type (type), | 
|  | m_field_count (0) | 
|  | { | 
|  | } | 
|  |  | 
|  | ui_out_type type () const | 
|  | { | 
|  | return m_type; | 
|  | } | 
|  |  | 
|  | int field_count () const | 
|  | { | 
|  | return m_field_count; | 
|  | } | 
|  |  | 
|  | void inc_field_count () | 
|  | { | 
|  | m_field_count++; | 
|  | } | 
|  |  | 
|  | private: | 
|  |  | 
|  | /* The type of this level.  */ | 
|  | ui_out_type m_type; | 
|  |  | 
|  | /* Count each field; the first element is for non-list fields.  */ | 
|  | int m_field_count; | 
|  | }; | 
|  |  | 
|  | /* Tables are special.  Maintain a separate structure that tracks | 
|  | their state.  At present an output can only contain a single table | 
|  | but that restriction might eventually be lifted.  */ | 
|  |  | 
|  | class ui_out_table | 
|  | { | 
|  | public: | 
|  |  | 
|  | /* States (steps) of a table generation.  */ | 
|  |  | 
|  | enum class state | 
|  | { | 
|  | /* We are generating the table headers.  */ | 
|  | HEADERS, | 
|  |  | 
|  | /* We are generating the table body.  */ | 
|  | BODY, | 
|  | }; | 
|  |  | 
|  | explicit ui_out_table (int entry_level, int nr_cols, const std::string &id) | 
|  | : m_state (state::HEADERS), | 
|  | m_entry_level (entry_level), | 
|  | m_nr_cols (nr_cols), | 
|  | m_id (id) | 
|  | { | 
|  | } | 
|  |  | 
|  | /* Start building the body of the table.  */ | 
|  |  | 
|  | void start_body (); | 
|  |  | 
|  | /* Add a new header to the table.  */ | 
|  |  | 
|  | void append_header (int width, ui_align alignment, | 
|  | const std::string &col_name, const std::string &col_hdr); | 
|  |  | 
|  | void start_row (); | 
|  |  | 
|  | /* Extract the format information for the next header and advance | 
|  | the header iterator.  Return false if there was no next header.  */ | 
|  |  | 
|  | bool get_next_header (int *colno, int *width, ui_align *alignment, | 
|  | const char **col_hdr); | 
|  |  | 
|  | bool query_field (int colno, int *width, int *alignment, | 
|  | const char **col_name) const; | 
|  |  | 
|  | state current_state () const; | 
|  |  | 
|  | int entry_level () const; | 
|  |  | 
|  | private: | 
|  |  | 
|  | state m_state; | 
|  |  | 
|  | /* The level at which each entry of the table is to be found.  A row | 
|  | (a tuple) is made up of entries.  Consequently ENTRY_LEVEL is one | 
|  | above that of the table.  */ | 
|  | int m_entry_level; | 
|  |  | 
|  | /* Number of table columns (as specified in the table_begin call).  */ | 
|  | int m_nr_cols; | 
|  |  | 
|  | /* String identifying the table (as specified in the table_begin | 
|  | call).  */ | 
|  | std::string m_id; | 
|  |  | 
|  | /* Pointers to the column headers.  */ | 
|  | std::vector<std::unique_ptr<ui_out_hdr>> m_headers; | 
|  |  | 
|  | /* Iterator over the headers vector, used when printing successive fields.  */ | 
|  | std::vector<std::unique_ptr<ui_out_hdr>>::const_iterator m_headers_iterator; | 
|  | }; | 
|  |  | 
|  | /* See ui-out.h.  */ | 
|  |  | 
|  | void ui_out_table::start_body () | 
|  | { | 
|  | if (m_state != state::HEADERS) | 
|  | internal_error (_("extra table_body call not allowed; there must be only " | 
|  | "one table_body after a table_begin and before a " | 
|  | "table_end.")); | 
|  |  | 
|  | /* Check if the number of defined headers matches the number of expected | 
|  | columns.  */ | 
|  | if (m_headers.size () != m_nr_cols) | 
|  | internal_error (_("number of headers differ from number of table " | 
|  | "columns.")); | 
|  |  | 
|  | m_state = state::BODY; | 
|  | m_headers_iterator = m_headers.begin (); | 
|  | } | 
|  |  | 
|  | /* See ui-out.h.  */ | 
|  |  | 
|  | void ui_out_table::append_header (int width, ui_align alignment, | 
|  | const std::string &col_name, | 
|  | const std::string &col_hdr) | 
|  | { | 
|  | if (m_state != state::HEADERS) | 
|  | internal_error (_("table header must be specified after table_begin and " | 
|  | "before table_body.")); | 
|  |  | 
|  | auto header = std::make_unique<ui_out_hdr> (m_headers.size () + 1, | 
|  | width, alignment, | 
|  | col_name, col_hdr); | 
|  |  | 
|  | m_headers.push_back (std::move (header)); | 
|  | } | 
|  |  | 
|  | /* See ui-out.h.  */ | 
|  |  | 
|  | void ui_out_table::start_row () | 
|  | { | 
|  | m_headers_iterator = m_headers.begin (); | 
|  | } | 
|  |  | 
|  | /* See ui-out.h.  */ | 
|  |  | 
|  | bool ui_out_table::get_next_header (int *colno, int *width, ui_align *alignment, | 
|  | const char **col_hdr) | 
|  | { | 
|  | /* There may be no headers at all or we may have used all columns.  */ | 
|  | if (m_headers_iterator == m_headers.end ()) | 
|  | return false; | 
|  |  | 
|  | ui_out_hdr *hdr = m_headers_iterator->get (); | 
|  |  | 
|  | *colno = hdr->number (); | 
|  | *width = hdr->min_width (); | 
|  | *alignment = hdr->alignment (); | 
|  | *col_hdr = hdr->header ().c_str (); | 
|  |  | 
|  | /* Advance the header pointer to the next entry.  */ | 
|  | m_headers_iterator++; | 
|  |  | 
|  | return true; | 
|  | } | 
|  |  | 
|  | /* See ui-out.h.  */ | 
|  |  | 
|  | bool ui_out_table::query_field (int colno, int *width, int *alignment, | 
|  | const char **col_name) const | 
|  | { | 
|  | /* Column numbers are 1-based, so convert to 0-based index.  */ | 
|  | int index = colno - 1; | 
|  |  | 
|  | if (index >= 0 && index < m_headers.size ()) | 
|  | { | 
|  | ui_out_hdr *hdr = m_headers[index].get (); | 
|  |  | 
|  | gdb_assert (colno == hdr->number ()); | 
|  |  | 
|  | *width = hdr->min_width (); | 
|  | *alignment = hdr->alignment (); | 
|  | *col_name = hdr->name ().c_str (); | 
|  |  | 
|  | return true; | 
|  | } | 
|  | else | 
|  | return false; | 
|  | } | 
|  |  | 
|  | /* See ui-out.h.  */ | 
|  |  | 
|  | ui_out_table::state ui_out_table::current_state () const | 
|  | { | 
|  | return m_state; | 
|  | } | 
|  |  | 
|  | /* See ui-out.h.  */ | 
|  |  | 
|  | int ui_out_table::entry_level () const | 
|  | { | 
|  | return m_entry_level; | 
|  | } | 
|  |  | 
|  | int | 
|  | ui_out::level () const | 
|  | { | 
|  | return m_levels.size (); | 
|  | } | 
|  |  | 
|  | /* The current (inner most) level.  */ | 
|  |  | 
|  | ui_out_level * | 
|  | ui_out::current_level () const | 
|  | { | 
|  | return m_levels.back ().get (); | 
|  | } | 
|  |  | 
|  | /* Create a new level, of TYPE.  */ | 
|  | void | 
|  | ui_out::push_level (ui_out_type type) | 
|  | { | 
|  | auto level = std::make_unique<ui_out_level> (type); | 
|  |  | 
|  | m_levels.push_back (std::move (level)); | 
|  | } | 
|  |  | 
|  | /* Discard the current level.  TYPE is the type of the level being | 
|  | discarded.  */ | 
|  | void | 
|  | ui_out::pop_level (ui_out_type type) | 
|  | { | 
|  | /* We had better not underflow the buffer.  */ | 
|  | gdb_assert (m_levels.size () > 0); | 
|  | gdb_assert (current_level ()->type () == type); | 
|  |  | 
|  | m_levels.pop_back (); | 
|  | } | 
|  |  | 
|  | /* Mark beginning of a table.  */ | 
|  |  | 
|  | void | 
|  | ui_out::table_begin (int nr_cols, int nr_rows, const std::string &tblid) | 
|  | { | 
|  | if (m_table_up != nullptr) | 
|  | internal_error (_("tables cannot be nested; table_begin found before \ | 
|  | previous table_end.")); | 
|  |  | 
|  | m_table_up = std::make_unique<ui_out_table> (level () + 1, nr_cols, tblid); | 
|  |  | 
|  | do_table_begin (nr_cols, nr_rows, tblid.c_str ()); | 
|  | } | 
|  |  | 
|  | void | 
|  | ui_out::table_header (int width, ui_align alignment, | 
|  | const std::string &col_name, const std::string &col_hdr) | 
|  | { | 
|  | if (m_table_up == nullptr) | 
|  | internal_error (_("table_header outside a table is not valid; it must be \ | 
|  | after a table_begin and before a table_body.")); | 
|  |  | 
|  | m_table_up->append_header (width, alignment, col_name, col_hdr); | 
|  |  | 
|  | do_table_header (width, alignment, col_name, col_hdr); | 
|  | } | 
|  |  | 
|  | void | 
|  | ui_out::table_body () | 
|  | { | 
|  | if (m_table_up == nullptr) | 
|  | internal_error (_("table_body outside a table is not valid; it must be " | 
|  | "after a table_begin and before a table_end.")); | 
|  |  | 
|  | m_table_up->start_body (); | 
|  |  | 
|  | do_table_body (); | 
|  | } | 
|  |  | 
|  | void | 
|  | ui_out::table_end () | 
|  | { | 
|  | if (m_table_up == nullptr) | 
|  | internal_error (_("misplaced table_end or missing table_begin.")); | 
|  |  | 
|  | do_table_end (); | 
|  |  | 
|  | m_table_up = nullptr; | 
|  | } | 
|  |  | 
|  | void | 
|  | ui_out::begin (ui_out_type type, const char *id) | 
|  | { | 
|  | /* Be careful to verify the ``field'' before the new tuple/list is | 
|  | pushed onto the stack.  That way the containing list/table/row is | 
|  | verified and not the newly created tuple/list.  This verification | 
|  | is needed (at least) for the case where a table row entry | 
|  | contains either a tuple/list.  For that case bookkeeping such as | 
|  | updating the column count or advancing to the next heading still | 
|  | needs to be performed.  */ | 
|  | { | 
|  | int fldno; | 
|  | int width; | 
|  | ui_align align; | 
|  |  | 
|  | verify_field (&fldno, &width, &align); | 
|  | } | 
|  |  | 
|  | push_level (type); | 
|  |  | 
|  | /* If the push puts us at the same level as a table row entry, we've | 
|  | got a new table row.  Put the header pointer back to the start.  */ | 
|  | if (m_table_up != nullptr | 
|  | && m_table_up->current_state () == ui_out_table::state::BODY | 
|  | && m_table_up->entry_level () == level ()) | 
|  | m_table_up->start_row (); | 
|  |  | 
|  | do_begin (type, id); | 
|  | } | 
|  |  | 
|  | void | 
|  | ui_out::end (ui_out_type type) | 
|  | { | 
|  | pop_level (type); | 
|  |  | 
|  | do_end (type); | 
|  | } | 
|  |  | 
|  | void | 
|  | ui_out::field_signed (const char *fldname, LONGEST value, | 
|  | const ui_file_style &style) | 
|  | { | 
|  | int fldno; | 
|  | int width; | 
|  | ui_align align; | 
|  |  | 
|  | verify_field (&fldno, &width, &align); | 
|  |  | 
|  | do_field_signed (fldno, width, align, fldname, value, style); | 
|  | } | 
|  |  | 
|  | void | 
|  | ui_out::field_fmt_signed (int input_width, ui_align input_align, | 
|  | const char *fldname, LONGEST value) | 
|  | { | 
|  | int fldno; | 
|  | int width; | 
|  | ui_align align; | 
|  |  | 
|  | verify_field (&fldno, &width, &align); | 
|  |  | 
|  | do_field_signed (fldno, input_width, input_align, fldname, value, | 
|  | ui_file_style ()); | 
|  | } | 
|  |  | 
|  | /* See ui-out.h.  */ | 
|  |  | 
|  | void | 
|  | ui_out::field_unsigned (const char *fldname, ULONGEST value) | 
|  | { | 
|  | int fldno; | 
|  | int width; | 
|  | ui_align align; | 
|  |  | 
|  | verify_field (&fldno, &width, &align); | 
|  |  | 
|  | do_field_unsigned (fldno, width, align, fldname, value); | 
|  | } | 
|  |  | 
|  | /* Documented in ui-out.h.  */ | 
|  |  | 
|  | void | 
|  | ui_out::field_core_addr (const char *fldname, struct gdbarch *gdbarch, | 
|  | CORE_ADDR address) | 
|  | { | 
|  | field_string (fldname, print_core_address (gdbarch, address), | 
|  | address_style.style ()); | 
|  | } | 
|  |  | 
|  | void | 
|  | ui_out::field_stream (const char *fldname, string_file &stream, | 
|  | const ui_file_style &style) | 
|  | { | 
|  | if (!stream.empty ()) | 
|  | field_string (fldname, stream.c_str (), style); | 
|  | else | 
|  | field_skip (fldname); | 
|  | stream.clear (); | 
|  | } | 
|  |  | 
|  | /* Used to omit a field.  */ | 
|  |  | 
|  | void | 
|  | ui_out::field_skip (const char *fldname) | 
|  | { | 
|  | int fldno; | 
|  | int width; | 
|  | ui_align align; | 
|  |  | 
|  | verify_field (&fldno, &width, &align); | 
|  |  | 
|  | do_field_skip (fldno, width, align, fldname); | 
|  | } | 
|  |  | 
|  | void | 
|  | ui_out::field_string (const char *fldname, const char *string, | 
|  | const ui_file_style &style) | 
|  | { | 
|  | int fldno; | 
|  | int width; | 
|  | ui_align align; | 
|  |  | 
|  | verify_field (&fldno, &width, &align); | 
|  |  | 
|  | do_field_string (fldno, width, align, fldname, string, style); | 
|  | } | 
|  |  | 
|  | /* VARARGS */ | 
|  | void | 
|  | ui_out::field_fmt (const char *fldname, const char *format, ...) | 
|  | { | 
|  | va_list args; | 
|  | int fldno; | 
|  | int width; | 
|  | ui_align align; | 
|  |  | 
|  | verify_field (&fldno, &width, &align); | 
|  |  | 
|  | va_start (args, format); | 
|  |  | 
|  | do_field_fmt (fldno, width, align, fldname, ui_file_style (), format, args); | 
|  |  | 
|  | va_end (args); | 
|  | } | 
|  |  | 
|  | void | 
|  | ui_out::field_fmt (const char *fldname, const ui_file_style &style, | 
|  | const char *format, ...) | 
|  | { | 
|  | va_list args; | 
|  | int fldno; | 
|  | int width; | 
|  | ui_align align; | 
|  |  | 
|  | verify_field (&fldno, &width, &align); | 
|  |  | 
|  | va_start (args, format); | 
|  |  | 
|  | do_field_fmt (fldno, width, align, fldname, style, format, args); | 
|  |  | 
|  | va_end (args); | 
|  | } | 
|  |  | 
|  | void | 
|  | ui_out::call_do_message (const ui_file_style &style, const char *format, | 
|  | ...) | 
|  | { | 
|  | va_list args; | 
|  |  | 
|  | va_start (args, format); | 
|  |  | 
|  | /* Since call_do_message is only used as a helper of vmessage, silence the | 
|  | warning here once instead of at all call sites in vmessage, if we were | 
|  | to put a "format" attribute on call_do_message.  */ | 
|  | DIAGNOSTIC_PUSH | 
|  | DIAGNOSTIC_IGNORE_FORMAT_NONLITERAL | 
|  | do_message (style, format, args); | 
|  | DIAGNOSTIC_POP | 
|  |  | 
|  | va_end (args); | 
|  | } | 
|  |  | 
|  | void | 
|  | ui_out::vmessage (const ui_file_style &in_style, const char *format, | 
|  | va_list args) | 
|  | { | 
|  | format_pieces fpieces (&format, true); | 
|  |  | 
|  | ui_file_style style = in_style; | 
|  |  | 
|  | for (auto &&piece : fpieces) | 
|  | { | 
|  | const char *current_substring = piece.string; | 
|  |  | 
|  | gdb_assert (piece.n_int_args >= 0 && piece.n_int_args <= 2); | 
|  | int intvals[2] = { 0, 0 }; | 
|  | for (int i = 0; i < piece.n_int_args; ++i) | 
|  | intvals[i] = va_arg (args, int); | 
|  |  | 
|  | /* The only ones we support for now.  */ | 
|  | gdb_assert (piece.n_int_args == 0 | 
|  | || piece.argclass == string_arg | 
|  | || piece.argclass == int_arg | 
|  | || piece.argclass == long_arg); | 
|  |  | 
|  | switch (piece.argclass) | 
|  | { | 
|  | case string_arg: | 
|  | { | 
|  | const char *str = va_arg (args, const char *); | 
|  | switch (piece.n_int_args) | 
|  | { | 
|  | case 0: | 
|  | call_do_message (style, current_substring, str); | 
|  | break; | 
|  | case 1: | 
|  | call_do_message (style, current_substring, intvals[0], str); | 
|  | break; | 
|  | case 2: | 
|  | call_do_message (style, current_substring, | 
|  | intvals[0], intvals[1], str); | 
|  | break; | 
|  | } | 
|  | } | 
|  | break; | 
|  | case wide_string_arg: | 
|  | gdb_assert_not_reached ("wide_string_arg not supported in vmessage"); | 
|  | break; | 
|  | case wide_char_arg: | 
|  | gdb_assert_not_reached ("wide_char_arg not supported in vmessage"); | 
|  | break; | 
|  | case long_long_arg: | 
|  | call_do_message (style, current_substring, va_arg (args, long long)); | 
|  | break; | 
|  | case int_arg: | 
|  | { | 
|  | int val = va_arg (args, int); | 
|  | switch (piece.n_int_args) | 
|  | { | 
|  | case 0: | 
|  | call_do_message (style, current_substring, val); | 
|  | break; | 
|  | case 1: | 
|  | call_do_message (style, current_substring, intvals[0], val); | 
|  | break; | 
|  | case 2: | 
|  | call_do_message (style, current_substring, | 
|  | intvals[0], intvals[1], val); | 
|  | break; | 
|  | } | 
|  | } | 
|  | break; | 
|  | case long_arg: | 
|  | { | 
|  | long val = va_arg (args, long); | 
|  | switch (piece.n_int_args) | 
|  | { | 
|  | case 0: | 
|  | call_do_message (style, current_substring, val); | 
|  | break; | 
|  | case 1: | 
|  | call_do_message (style, current_substring, intvals[0], val); | 
|  | break; | 
|  | case 2: | 
|  | call_do_message (style, current_substring, | 
|  | intvals[0], intvals[1], val); | 
|  | break; | 
|  | } | 
|  | } | 
|  | break; | 
|  | case size_t_arg: | 
|  | { | 
|  | size_t val = va_arg (args, size_t); | 
|  | switch (piece.n_int_args) | 
|  | { | 
|  | case 0: | 
|  | call_do_message (style, current_substring, val); | 
|  | break; | 
|  | case 1: | 
|  | call_do_message (style, current_substring, intvals[0], val); | 
|  | break; | 
|  | case 2: | 
|  | call_do_message (style, current_substring, | 
|  | intvals[0], intvals[1], val); | 
|  | break; | 
|  | } | 
|  | } | 
|  | break; | 
|  | case double_arg: | 
|  | call_do_message (style, current_substring, va_arg (args, double)); | 
|  | break; | 
|  | case long_double_arg: | 
|  | gdb_assert_not_reached ("long_double_arg not supported in vmessage"); | 
|  | break; | 
|  | case dec32float_arg: | 
|  | gdb_assert_not_reached ("dec32float_arg not supported in vmessage"); | 
|  | break; | 
|  | case dec64float_arg: | 
|  | gdb_assert_not_reached ("dec64float_arg not supported in vmessage"); | 
|  | break; | 
|  | case dec128float_arg: | 
|  | gdb_assert_not_reached ("dec128float_arg not supported in vmessage"); | 
|  | break; | 
|  | case ptr_arg: | 
|  | switch (current_substring[2]) | 
|  | { | 
|  | case 'F': | 
|  | { | 
|  | gdb_assert (!test_flags (disallow_ui_out_field)); | 
|  | base_field_s *bf = va_arg (args, base_field_s *); | 
|  | switch (bf->kind) | 
|  | { | 
|  | case field_kind::FIELD_SIGNED: | 
|  | { | 
|  | auto *f = (signed_field_s *) bf; | 
|  | field_signed (f->name, f->val); | 
|  | } | 
|  | break; | 
|  | case field_kind::FIELD_STRING: | 
|  | { | 
|  | auto *f = (string_field_s *) bf; | 
|  | field_string (f->name, f->str); | 
|  | } | 
|  | break; | 
|  | } | 
|  | } | 
|  | break; | 
|  | case 's': | 
|  | { | 
|  | styled_string_s *ss = va_arg (args, styled_string_s *); | 
|  | call_do_message (ss->style, "%s", ss->str); | 
|  | } | 
|  | break; | 
|  | case '[': | 
|  | style = *va_arg (args, const ui_file_style *); | 
|  | break; | 
|  | case ']': | 
|  | { | 
|  | void *arg = va_arg (args, void *); | 
|  | gdb_assert (arg == nullptr); | 
|  |  | 
|  | style = {}; | 
|  | } | 
|  | break; | 
|  | default: | 
|  | call_do_message (style, current_substring, va_arg (args, void *)); | 
|  | break; | 
|  | } | 
|  | break; | 
|  | case literal_piece: | 
|  | /* Print a portion of the format string that has no | 
|  | directives.  Note that this will not include any ordinary | 
|  | %-specs, but it might include "%%".  That is why we use | 
|  | call_do_message here.  Also, we pass a dummy argument | 
|  | because some platforms have modified GCC to include | 
|  | -Wformat-security by default, which will warn here if | 
|  | there is no argument.  */ | 
|  | call_do_message (style, current_substring, 0); | 
|  | break; | 
|  | default: | 
|  | internal_error (_("failed internal consistency check")); | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | void | 
|  | ui_out::message (const char *format, ...) | 
|  | { | 
|  | va_list args; | 
|  | va_start (args, format); | 
|  |  | 
|  | vmessage (ui_file_style (), format, args); | 
|  |  | 
|  | va_end (args); | 
|  | } | 
|  |  | 
|  | /* Verify that the field/tuple/list is correctly positioned.  Return | 
|  | the field number and corresponding alignment (if | 
|  | available/applicable).  */ | 
|  |  | 
|  | void | 
|  | ui_out::verify_field (int *fldno, int *width, ui_align *align) | 
|  | { | 
|  | ui_out_level *current = current_level (); | 
|  | const char *text; | 
|  |  | 
|  | if (m_table_up != nullptr | 
|  | && m_table_up->current_state () != ui_out_table::state::BODY) | 
|  | { | 
|  | internal_error (_("table_body missing; table fields must be \ | 
|  | specified after table_body and inside a list.")); | 
|  | } | 
|  |  | 
|  | current->inc_field_count (); | 
|  |  | 
|  | if (m_table_up != nullptr | 
|  | && m_table_up->current_state () == ui_out_table::state::BODY | 
|  | && m_table_up->entry_level () == level () | 
|  | && m_table_up->get_next_header (fldno, width, align, &text)) | 
|  | { | 
|  | if (*fldno != current->field_count ()) | 
|  | internal_error (_("ui-out internal error in handling headers.")); | 
|  | } | 
|  | else | 
|  | { | 
|  | *width = 0; | 
|  | *align = ui_noalign; | 
|  | *fldno = current->field_count (); | 
|  | } | 
|  | } | 
|  |  | 
|  | /* Access table field parameters.  */ | 
|  |  | 
|  | bool | 
|  | ui_out::query_table_field (int colno, int *width, int *alignment, | 
|  | const char **col_name) | 
|  | { | 
|  | if (m_table_up == nullptr) | 
|  | return false; | 
|  |  | 
|  | return m_table_up->query_field (colno, width, alignment, col_name); | 
|  | } | 
|  |  | 
|  | /* The constructor.  */ | 
|  |  | 
|  | ui_out::ui_out (ui_out_flags flags) | 
|  | : m_flags (flags) | 
|  | { | 
|  | /* Create the ui-out level #1, the default level.  */ | 
|  | push_level (ui_out_type_tuple); | 
|  | } | 
|  |  | 
|  | ui_out::~ui_out () | 
|  | { | 
|  | } | 
|  |  | 
|  | /* See ui-out.h.  */ | 
|  |  | 
|  | void | 
|  | buffer_group::output_unit::flush () const | 
|  | { | 
|  | if (!m_msg.empty ()) | 
|  | m_stream->puts (m_msg.c_str ()); | 
|  |  | 
|  | if (m_wrap_hint >= 0) | 
|  | m_stream->wrap_here (m_wrap_hint); | 
|  |  | 
|  | if (m_flush) | 
|  | m_stream->flush (); | 
|  | } | 
|  |  | 
|  | /* See ui-out.h.  */ | 
|  |  | 
|  | void | 
|  | buffer_group::write (const char *buf, long length_buf, ui_file *stream) | 
|  | { | 
|  | /* Record each line separately.  */ | 
|  | for (size_t prev = 0, cur = 0; cur < length_buf; ++cur) | 
|  | if (buf[cur] == '\n' || cur == length_buf - 1) | 
|  | { | 
|  | std::string msg (buf + prev, cur - prev + 1); | 
|  |  | 
|  | if (m_buffered_output.size () > 0 | 
|  | && m_buffered_output.back ().m_wrap_hint == -1 | 
|  | && m_buffered_output.back ().m_stream == stream | 
|  | && m_buffered_output.back ().m_msg.size () > 0 | 
|  | && m_buffered_output.back ().m_msg.back () != '\n') | 
|  | m_buffered_output.back ().m_msg.append (msg); | 
|  | else | 
|  | m_buffered_output.emplace_back (msg).m_stream = stream; | 
|  | prev = cur + 1; | 
|  | } | 
|  | } | 
|  |  | 
|  | /* See ui-out.h.  */ | 
|  |  | 
|  | void | 
|  | buffer_group::wrap_here (int indent, ui_file *stream) | 
|  | { | 
|  | m_buffered_output.emplace_back ("", indent).m_stream = stream; | 
|  | } | 
|  |  | 
|  | /* See ui-out.h.  */ | 
|  |  | 
|  | void | 
|  | buffer_group::flush_here (ui_file *stream) | 
|  | { | 
|  | m_buffered_output.emplace_back ("", -1, true).m_stream = stream; | 
|  | } | 
|  |  | 
|  | /* See ui-out.h.  */ | 
|  |  | 
|  | ui_file * | 
|  | get_unbuffered (ui_file *stream) | 
|  | { | 
|  | buffering_file *buf = dynamic_cast<buffering_file *> (stream); | 
|  |  | 
|  | if (buf == nullptr) | 
|  | return stream; | 
|  |  | 
|  | return get_unbuffered (buf->stream ()); | 
|  | } | 
|  |  | 
|  | buffered_streams::buffered_streams (buffer_group *group, ui_out *uiout) | 
|  | : m_buffered_stdout (group, gdb_stdout), | 
|  | m_buffered_stderr (group, gdb_stderr), | 
|  | m_buffered_stdlog (group, gdb_stdlog), | 
|  | m_buffered_stdtarg (group, gdb_stdtarg), | 
|  | m_uiout (uiout) | 
|  | { | 
|  | gdb_stdout = &m_buffered_stdout; | 
|  | gdb_stderr = &m_buffered_stderr; | 
|  | gdb_stdlog = &m_buffered_stdlog; | 
|  | gdb_stdtarg = &m_buffered_stdtarg; | 
|  |  | 
|  | ui_file *stream = current_uiout->current_stream (); | 
|  | if (stream != nullptr) | 
|  | { | 
|  | m_buffered_current_uiout.emplace (group, stream); | 
|  | current_uiout->redirect (&(*m_buffered_current_uiout)); | 
|  | } | 
|  |  | 
|  | stream = m_uiout->current_stream (); | 
|  | if (stream != nullptr && current_uiout != m_uiout) | 
|  | { | 
|  | m_buffered_uiout.emplace (group, stream); | 
|  | m_uiout->redirect (&(*m_buffered_uiout)); | 
|  | } | 
|  |  | 
|  | m_buffers_in_place = true; | 
|  | } | 
|  |  | 
|  | /* See ui-out.h.  */ | 
|  |  | 
|  | void | 
|  | buffered_streams::remove_buffers () | 
|  | { | 
|  | if (!m_buffers_in_place) | 
|  | return; | 
|  |  | 
|  | m_buffers_in_place = false; | 
|  |  | 
|  | gdb_stdout = m_buffered_stdout.stream (); | 
|  | gdb_stderr = m_buffered_stderr.stream (); | 
|  | gdb_stdlog = m_buffered_stdlog.stream (); | 
|  | gdb_stdtarg = m_buffered_stdtarg.stream (); | 
|  |  | 
|  | if (m_buffered_current_uiout.has_value ()) | 
|  | current_uiout->redirect (nullptr); | 
|  |  | 
|  | if (m_buffered_uiout.has_value ()) | 
|  | m_uiout->redirect (nullptr); | 
|  | } | 
|  |  | 
|  | buffer_group::buffer_group (ui_out *uiout) | 
|  | : m_buffered_streams (new buffered_streams (this, uiout)) | 
|  | { /* Nothing.  */ } | 
|  |  | 
|  | /* See ui-out.h.  */ | 
|  |  | 
|  | void | 
|  | buffer_group::flush () const | 
|  | { | 
|  | m_buffered_streams->remove_buffers (); | 
|  |  | 
|  | for (const output_unit &ou : m_buffered_output) | 
|  | ou.flush (); | 
|  | } |