X-Git-Url: http://git.efficios.com/?a=blobdiff_plain;f=gdb%2Fui-out.c;h=38190811ab2b42ea04c3c698a9b68b2119bf510c;hb=2af46ca02947874cbd188f0933427086d82cf388;hp=0d49aba73808e57e42e80c8de9d7ba53878cc30d;hpb=bafdd3b3f6cad9754b1ee83a0f14839b59a0a188;p=deliverable%2Fbinutils-gdb.git diff --git a/gdb/ui-out.c b/gdb/ui-out.c index 0d49aba738..38190811ab 100644 --- a/gdb/ui-out.c +++ b/gdb/ui-out.c @@ -1,5 +1,8 @@ /* Output generating routines for GDB. - Copyright 1999, 2000, 2001 Free Software Foundation, Inc. + + Copyright (C) 1999, 2000, 2001, 2002, 2004, 2005, 2007 + Free Software Foundation, Inc. + Contributed by Cygnus Solutions. Written by Fernando Nasser for Cygnus. @@ -7,7 +10,7 @@ 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 2 of the License, or + 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, @@ -16,9 +19,7 @@ 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, write to the Free Software - Foundation, Inc., 59 Temple Place - Suite 330, - Boston, MA 02111-1307, USA. */ + along with this program. If not, see . */ #include "defs.h" #include "gdb_string.h" @@ -27,11 +28,6 @@ #include "ui-out.h" #include "gdb_assert.h" -/* Convenience macro for allocting typesafe memory. */ - -#undef XMALLOC -#define XMALLOC(TYPE) (TYPE*) xmalloc (sizeof (TYPE)) - /* table header structures */ struct ui_out_hdr @@ -48,7 +44,7 @@ struct ui_out_hdr is always available. Stack/nested level 0 is reserved for the top-level result. */ -enum { MAX_UI_OUT_LEVELS = 5 }; +enum { MAX_UI_OUT_LEVELS = 6 }; struct ui_out_level { @@ -71,6 +67,11 @@ struct ui_out_table header is being generated. */ int body_flag; + /* 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 entry_level; + /* Number of table columns (as specified in the table_begin call). */ int columns; @@ -176,12 +177,12 @@ static void default_field_fmt (struct ui_out *uiout, int fldno, int width, enum ui_align align, const char *fldname, const char *format, - va_list args); + va_list args) ATTR_FORMAT (printf, 6, 0); static void default_spaces (struct ui_out *uiout, int numspaces); static void default_text (struct ui_out *uiout, const char *string); static void default_message (struct ui_out *uiout, int verbosity, const char *format, - va_list args); + va_list args) ATTR_FORMAT (printf, 3, 0); static void default_wrap_hint (struct ui_out *uiout, char *identstring); static void default_flush (struct ui_out *uiout); @@ -204,6 +205,7 @@ struct ui_out_impl default_ui_out_impl = default_message, default_wrap_hint, default_flush, + NULL, 0, /* Does not need MI hacks. */ }; @@ -245,13 +247,16 @@ static void uo_field_string (struct ui_out *uiout, int fldno, int width, const char *string); static void uo_field_fmt (struct ui_out *uiout, int fldno, int width, enum ui_align align, const char *fldname, - const char *format, va_list args); + const char *format, va_list args) + ATTR_FORMAT (printf, 6, 0); static void uo_spaces (struct ui_out *uiout, int numspaces); static void uo_text (struct ui_out *uiout, const char *string); static void uo_message (struct ui_out *uiout, int verbosity, - const char *format, va_list args); + const char *format, va_list args) + ATTR_FORMAT (printf, 3, 0); static void uo_wrap_hint (struct ui_out *uiout, char *identstring); static void uo_flush (struct ui_out *uiout); +static int uo_redirect (struct ui_out *uiout, struct ui_file *outstream); /* Prototypes for local functions */ @@ -262,27 +267,26 @@ static void append_header_to_list (struct ui_out *uiout, int width, static int get_next_header (struct ui_out *uiout, int *colno, int *width, int *alignment, char **colhdr); static void clear_header_list (struct ui_out *uiout); -static void verify_field_proper_position (struct ui_out *uiout); -static void verify_field_alignment (struct ui_out *uiout, int fldno, int *width, int *alignment); - -static void init_ui_out_state (struct ui_out *uiout); +static void verify_field (struct ui_out *uiout, int *fldno, int *width, + int *align); /* exported functions (ui_out API) */ /* Mark beginning of a table */ -void +static void ui_out_table_begin (struct ui_out *uiout, int nbrofcols, int nr_rows, const char *tblid) { if (uiout->table.flag) internal_error (__FILE__, __LINE__, - "tables cannot be nested; table_begin found before \ -previous table_end."); + _("tables cannot be nested; table_begin found before \ +previous table_end.")); uiout->table.flag = 1; uiout->table.body_flag = 0; + uiout->table.entry_level = uiout->level + 1; uiout->table.columns = nbrofcols; if (tblid != NULL) uiout->table.id = xstrdup (tblid); @@ -298,16 +302,16 @@ ui_out_table_body (struct ui_out *uiout) { if (!uiout->table.flag) internal_error (__FILE__, __LINE__, - "table_body outside a table is not valid; it must be \ -after a table_begin and before a table_end."); + _("table_body outside a table is not valid; it must be \ +after a table_begin and before a table_end.")); if (uiout->table.body_flag) internal_error (__FILE__, __LINE__, - "extra table_body call not allowed; there must be \ -only one table_body after a table_begin and before a table_end."); + _("extra table_body call not allowed; there must be \ +only one table_body after a table_begin and before a table_end.")); if (uiout->table.header_next->colno != uiout->table.columns) internal_error (__FILE__, __LINE__, - "number of headers differ from number of table \ -columns."); + _("number of headers differ from number of table \ +columns.")); uiout->table.body_flag = 1; uiout->table.header_next = uiout->table.header_first; @@ -315,13 +319,14 @@ columns."); uo_table_body (uiout); } -void +static void ui_out_table_end (struct ui_out *uiout) { if (!uiout->table.flag) internal_error (__FILE__, __LINE__, - "misplaced table_end or missing table_begin."); + _("misplaced table_end or missing table_begin.")); + uiout->table.entry_level = 0; uiout->table.body_flag = 0; uiout->table.flag = 0; @@ -339,14 +344,30 @@ ui_out_table_header (struct ui_out *uiout, int width, enum ui_align alignment, { if (!uiout->table.flag || uiout->table.body_flag) internal_error (__FILE__, __LINE__, - "table header must be specified after table_begin \ -and before table_body."); + _("table header must be specified after table_begin \ +and before table_body.")); append_header_to_list (uiout, width, alignment, col_name, colhdr); uo_table_header (uiout, width, alignment, col_name, colhdr); } +static void +do_cleanup_table_end (void *data) +{ + struct ui_out *ui_out = data; + + ui_out_table_end (ui_out); +} + +struct cleanup * +make_cleanup_ui_out_table_begin_end (struct ui_out *ui_out, int nr_cols, + int nr_rows, const char *tblid) +{ + ui_out_table_begin (ui_out, nr_cols, nr_rows, tblid); + return make_cleanup (do_cleanup_table_end, ui_out); +} + void ui_out_begin (struct ui_out *uiout, enum ui_out_type type, @@ -355,25 +376,32 @@ ui_out_begin (struct ui_out *uiout, int new_level; if (uiout->table.flag && !uiout->table.body_flag) internal_error (__FILE__, __LINE__, - "table header or table_body expected; lists must be \ -specified after table_body."); + _("table header or table_body expected; lists must be \ +specified after table_body.")); + + /* 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; + int align; + verify_field (uiout, &fldno, &width, &align); + } + new_level = push_level (uiout, type, id); - if (uiout->table.flag && (new_level == 1)) - uiout->table.header_next = uiout->table.header_first; - uo_begin (uiout, type, new_level, id); -} -void -ui_out_list_begin (struct ui_out *uiout, - const char *id) -{ - ui_out_begin (uiout, ui_out_type_list, id); -} + /* 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 (uiout->table.body_flag + && uiout->table.entry_level == new_level) + uiout->table.header_next = uiout->table.header_first; -void -ui_out_tuple_begin (struct ui_out *uiout, const char *id) -{ - ui_out_begin (uiout, ui_out_type_tuple, id); + uo_begin (uiout, type, new_level, id); } void @@ -384,18 +412,6 @@ ui_out_end (struct ui_out *uiout, uo_end (uiout, type, old_level); } -void -ui_out_list_end (struct ui_out *uiout) -{ - ui_out_end (uiout, ui_out_type_list); -} - -void -ui_out_tuple_end (struct ui_out *uiout) -{ - ui_out_end (uiout, ui_out_type_tuple); -} - struct ui_out_end_cleanup_data { struct ui_out *uiout; @@ -421,20 +437,11 @@ make_cleanup_ui_out_end (struct ui_out *uiout, return make_cleanup (do_cleanup_end, end_cleanup_data); } -struct cleanup * -make_cleanup_ui_out_begin_end (struct ui_out *uiout, - enum ui_out_type type, - const char *id) -{ - ui_out_begin (uiout, type, id); - return make_cleanup_ui_out_end (uiout, type); -} - struct cleanup * make_cleanup_ui_out_tuple_begin_end (struct ui_out *uiout, const char *id) { - ui_out_tuple_begin (uiout, id); + ui_out_begin (uiout, ui_out_type_tuple, id); return make_cleanup_ui_out_end (uiout, ui_out_type_tuple); } @@ -442,7 +449,7 @@ struct cleanup * make_cleanup_ui_out_list_begin_end (struct ui_out *uiout, const char *id) { - ui_out_list_begin (uiout, id); + ui_out_begin (uiout, ui_out_type_list, id); return make_cleanup_ui_out_end (uiout, ui_out_type_list); } @@ -456,14 +463,26 @@ ui_out_field_int (struct ui_out *uiout, int align; struct ui_out_level *current = current_level (uiout); - verify_field_proper_position (uiout); + verify_field (uiout, &fldno, &width, &align); - current->field_count += 1; - fldno = current->field_count; + uo_field_int (uiout, fldno, width, align, fldname, value); +} - verify_field_alignment (uiout, fldno, &width, &align); +void +ui_out_field_fmt_int (struct ui_out *uiout, + int input_width, + enum ui_align input_align, + const char *fldname, + int value) +{ + int fldno; + int width; + int align; + struct ui_out_level *current = current_level (uiout); - uo_field_int (uiout, fldno, width, align, fldname, value); + verify_field (uiout, &fldno, &width, &align); + + uo_field_int (uiout, fldno, input_width, input_align, fldname, value); } void @@ -472,10 +491,20 @@ ui_out_field_core_addr (struct ui_out *uiout, CORE_ADDR address) { char addstr[20]; - - /* FIXME-32x64: need a print_address_numeric with field width */ - /* print_address_numeric (address, 1, local_stream); */ - strcpy (addstr, local_hex_string_custom ((unsigned long) address, "08l")); + int addr_bit = gdbarch_addr_bit (current_gdbarch); + + /* Truncate address to match deprecated_print_address_numeric(). */ + if (addr_bit < (sizeof (CORE_ADDR) * HOST_CHAR_BIT)) + address &= ((CORE_ADDR) 1 << addr_bit) - 1; + + /* FIXME: cagney/2002-05-03: Need local_address_string() function + that returns the language localized string formatted to a width + based on gdbarch_addr_bit. */ + /* deprecated_print_address_numeric (address, 1, local_stream); */ + if (addr_bit <= 32) + strcpy (addstr, hex_string_custom (address, 8)); + else + strcpy (addstr, hex_string_custom (address, 16)); ui_out_field_string (uiout, fldname, addstr); } @@ -505,14 +534,8 @@ ui_out_field_skip (struct ui_out *uiout, int fldno; int width; int align; - struct ui_out_level *current = current_level (uiout); - - verify_field_proper_position (uiout); - - current->field_count += 1; - fldno = current->field_count; - verify_field_alignment (uiout, fldno, &width, &align); + verify_field (uiout, &fldno, &width, &align); uo_field_skip (uiout, fldno, width, align, fldname); } @@ -525,14 +548,8 @@ ui_out_field_string (struct ui_out *uiout, int fldno; int width; int align; - struct ui_out_level *current = current_level (uiout); - - verify_field_proper_position (uiout); - current->field_count += 1; - fldno = current->field_count; - - verify_field_alignment (uiout, fldno, &width, &align); + verify_field (uiout, &fldno, &width, &align); uo_field_string (uiout, fldno, width, align, fldname, string); } @@ -547,15 +564,9 @@ ui_out_field_fmt (struct ui_out *uiout, int fldno; int width; int align; - struct ui_out_level *current = current_level (uiout); - - verify_field_proper_position (uiout); - - current->field_count += 1; - fldno = current->field_count; /* will not align, but has to call anyway */ - verify_field_alignment (uiout, fldno, &width, &align); + verify_field (uiout, &fldno, &width, &align); va_start (args, format); @@ -633,6 +644,12 @@ ui_out_flush (struct ui_out *uiout) uo_flush (uiout); } +int +ui_out_redirect (struct ui_out *uiout, struct ui_file *outstream) +{ + return uo_redirect (uiout, outstream); +} + /* set the flags specified by the mask given */ int ui_out_set_flags (struct ui_out *uiout, int mask) @@ -976,6 +993,15 @@ uo_flush (struct ui_out *uiout) uiout->impl->flush (uiout); } +int +uo_redirect (struct ui_out *uiout, struct ui_file *outstream) +{ + if (!uiout->impl->redirect) + return -1; + uiout->impl->redirect (uiout, outstream); + return 0; +} + /* local functions */ /* list of column headers manipulation routines */ @@ -1008,15 +1034,20 @@ append_header_to_list (struct ui_out *uiout, temphdr = XMALLOC (struct ui_out_hdr); temphdr->width = width; temphdr->alignment = alignment; - /* we have to copy the column title as the original may be an automatic */ + /* We have to copy the column title as the original may be an + automatic. */ if (colhdr != NULL) temphdr->colhdr = xstrdup (colhdr); else temphdr->colhdr = NULL; + if (col_name != NULL) + temphdr->col_name = xstrdup (col_name); + else if (colhdr != NULL) temphdr->col_name = xstrdup (colhdr); else - temphdr->col_name = xstrdup (colhdr); + temphdr->col_name = NULL; + temphdr->next = NULL; if (uiout->table.header_first == NULL) { @@ -1055,49 +1086,49 @@ get_next_header (struct ui_out *uiout, return 1; } -/* makes sure the field_* calls were properly placed */ + +/* Verify that the field/tuple/list is correctly positioned. Return + the field number and corresponding alignment (if + available/applicable). */ static void -verify_field_proper_position (struct ui_out *uiout) +verify_field (struct ui_out *uiout, int *fldno, int *width, int *align) { + struct ui_out_level *current = current_level (uiout); + char *text; + if (uiout->table.flag) { if (!uiout->table.body_flag) internal_error (__FILE__, __LINE__, - "table_body missing; table fields must be \ -specified after table_body and inside a list."); - if (uiout->level == 0) - internal_error (__FILE__, __LINE__, - "list_begin missing; table fields must be \ -specified after table_body and inside a list."); + _("table_body missing; table fields must be \ +specified after table_body and inside a list.")); + /* NOTE: cagney/2001-12-08: There was a check here to ensure + that this code was only executed when uiout->level was + greater than zero. That no longer applies - this code is run + before each table row tuple is started and at that point the + level is zero. */ } -} -/* determines what is the alignment policy */ - -static void -verify_field_alignment (struct ui_out *uiout, - int fldno, - int *width, - int *align) -{ - int colno; - char *text; + current->field_count += 1; - if (uiout->table.flag - && get_next_header (uiout, &colno, width, align, &text)) + if (uiout->table.body_flag + && uiout->table.entry_level == uiout->level + && get_next_header (uiout, fldno, width, align, &text)) { - if (fldno != colno) + if (*fldno != current->field_count) internal_error (__FILE__, __LINE__, - "ui-out internal error in handling headers."); + _("ui-out internal error in handling headers.")); } else { *width = 0; *align = ui_noalign; + *fldno = current->field_count; } } + /* access to ui_out format private members */ void