|  | /* Support for printing Pascal types for GDB, the GNU debugger. | 
|  | Copyright (C) 2000-2024 Free Software Foundation, Inc. | 
|  |  | 
|  | This file is part of GDB. | 
|  |  | 
|  | This program is free software; you can redistribute it and/or modify | 
|  | it under the terms of the GNU General Public License as published by | 
|  | the Free Software Foundation; either version 3 of the License, or | 
|  | (at your option) any later version. | 
|  |  | 
|  | This program is distributed in the hope that it will be useful, | 
|  | but WITHOUT ANY WARRANTY; without even the implied warranty of | 
|  | MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 
|  | GNU General Public License for more details. | 
|  |  | 
|  | You should have received a copy of the GNU General Public License | 
|  | along with this program.  If not, see <http://www.gnu.org/licenses/>.  */ | 
|  |  | 
|  | /* This file is derived from p-typeprint.c */ | 
|  |  | 
|  | #include "event-top.h" | 
|  | #include "gdbsupport/gdb_obstack.h" | 
|  | #include "bfd.h" | 
|  | #include "symtab.h" | 
|  | #include "gdbtypes.h" | 
|  | #include "expression.h" | 
|  | #include "value.h" | 
|  | #include "gdbcore.h" | 
|  | #include "target.h" | 
|  | #include "language.h" | 
|  | #include "p-lang.h" | 
|  | #include "typeprint.h" | 
|  | #include "gdb-demangle.h" | 
|  | #include <ctype.h> | 
|  | #include "cli/cli-style.h" | 
|  |  | 
|  | /* See language.h.  */ | 
|  |  | 
|  | void | 
|  | pascal_language::print_type (struct type *type, const char *varstring, | 
|  | struct ui_file *stream, int show, int level, | 
|  | const struct type_print_options *flags) const | 
|  | { | 
|  | enum type_code code; | 
|  | int demangled_args; | 
|  |  | 
|  | code = type->code (); | 
|  |  | 
|  | if (show > 0) | 
|  | type = check_typedef (type); | 
|  |  | 
|  | if ((code == TYPE_CODE_FUNC | 
|  | || code == TYPE_CODE_METHOD)) | 
|  | { | 
|  | type_print_varspec_prefix (type, stream, show, 0, flags); | 
|  | } | 
|  | /* first the name */ | 
|  | if (varstring != nullptr) | 
|  | gdb_puts (varstring, stream); | 
|  |  | 
|  | if ((varstring != NULL && *varstring != '\0') | 
|  | && !(code == TYPE_CODE_FUNC | 
|  | || code == TYPE_CODE_METHOD)) | 
|  | { | 
|  | gdb_puts (" : ", stream); | 
|  | } | 
|  |  | 
|  | if (!(code == TYPE_CODE_FUNC | 
|  | || code == TYPE_CODE_METHOD)) | 
|  | { | 
|  | type_print_varspec_prefix (type, stream, show, 0, flags); | 
|  | } | 
|  |  | 
|  | type_print_base (type, stream, show, level, flags); | 
|  | /* For demangled function names, we have the arglist as part of the name, | 
|  | so don't print an additional pair of ()'s.  */ | 
|  |  | 
|  | demangled_args = varstring ? strchr (varstring, '(') != NULL : 0; | 
|  | type_print_varspec_suffix (type, stream, show, 0, demangled_args, | 
|  | flags); | 
|  |  | 
|  | } | 
|  |  | 
|  | /* See language.h.  */ | 
|  |  | 
|  | void | 
|  | pascal_language::print_typedef (struct type *type, struct symbol *new_symbol, | 
|  | struct ui_file *stream) const | 
|  | { | 
|  | type = check_typedef (type); | 
|  | gdb_printf (stream, "type "); | 
|  | gdb_printf (stream, "%s = ", new_symbol->print_name ()); | 
|  | type_print (type, "", stream, 0); | 
|  | gdb_printf (stream, ";"); | 
|  | } | 
|  |  | 
|  | /* See p-lang.h.  */ | 
|  |  | 
|  | void | 
|  | pascal_language::type_print_derivation_info (struct ui_file *stream, | 
|  | struct type *type) const | 
|  | { | 
|  | const char *name; | 
|  | int i; | 
|  |  | 
|  | for (i = 0; i < TYPE_N_BASECLASSES (type); i++) | 
|  | { | 
|  | gdb_puts (i == 0 ? ": " : ", ", stream); | 
|  | gdb_printf (stream, "%s%s ", | 
|  | BASETYPE_VIA_PUBLIC (type, i) ? "public" : "private", | 
|  | BASETYPE_VIA_VIRTUAL (type, i) ? " virtual" : ""); | 
|  | name = TYPE_BASECLASS (type, i)->name (); | 
|  | gdb_printf (stream, "%s", name ? name : "(null)"); | 
|  | } | 
|  | if (i > 0) | 
|  | { | 
|  | gdb_puts (" ", stream); | 
|  | } | 
|  | } | 
|  |  | 
|  | /* See p-lang.h.  */ | 
|  |  | 
|  | void | 
|  | pascal_language::type_print_method_args (const char *physname, | 
|  | const char *methodname, | 
|  | struct ui_file *stream) const | 
|  | { | 
|  | int is_constructor = (startswith (physname, "__ct__")); | 
|  | int is_destructor = (startswith (physname, "__dt__")); | 
|  |  | 
|  | if (is_constructor || is_destructor) | 
|  | { | 
|  | physname += 6; | 
|  | } | 
|  |  | 
|  | gdb_puts (methodname, stream); | 
|  |  | 
|  | if (physname && (*physname != 0)) | 
|  | { | 
|  | gdb_puts (" (", stream); | 
|  | /* We must demangle this.  */ | 
|  | while (isdigit (physname[0])) | 
|  | { | 
|  | int len = 0; | 
|  | int i, j; | 
|  | char *argname; | 
|  |  | 
|  | while (isdigit (physname[len])) | 
|  | { | 
|  | len++; | 
|  | } | 
|  | i = strtol (physname, &argname, 0); | 
|  | physname += len; | 
|  |  | 
|  | for (j = 0; j < i; ++j) | 
|  | gdb_putc (physname[j], stream); | 
|  |  | 
|  | physname += i; | 
|  | if (physname[0] != 0) | 
|  | { | 
|  | gdb_puts (", ", stream); | 
|  | } | 
|  | } | 
|  | gdb_puts (")", stream); | 
|  | } | 
|  | } | 
|  |  | 
|  | /* See p-lang.h.  */ | 
|  |  | 
|  | void | 
|  | pascal_language::type_print_varspec_prefix (struct type *type, | 
|  | struct ui_file *stream, | 
|  | int show, int passed_a_ptr, | 
|  | const struct type_print_options *flags) const | 
|  | { | 
|  | if (type == 0) | 
|  | return; | 
|  |  | 
|  | if (type->name () && show <= 0) | 
|  | return; | 
|  |  | 
|  | QUIT; | 
|  |  | 
|  | switch (type->code ()) | 
|  | { | 
|  | case TYPE_CODE_PTR: | 
|  | gdb_printf (stream, "^"); | 
|  | type_print_varspec_prefix (type->target_type (), stream, 0, 1, | 
|  | flags); | 
|  | break;			/* Pointer should be handled normally | 
|  | in pascal.  */ | 
|  |  | 
|  | case TYPE_CODE_METHOD: | 
|  | if (passed_a_ptr) | 
|  | gdb_printf (stream, "("); | 
|  | if (type->target_type () != NULL | 
|  | && type->target_type ()->code () != TYPE_CODE_VOID) | 
|  | { | 
|  | gdb_printf (stream, "function  "); | 
|  | } | 
|  | else | 
|  | { | 
|  | gdb_printf (stream, "procedure "); | 
|  | } | 
|  |  | 
|  | if (passed_a_ptr) | 
|  | { | 
|  | gdb_printf (stream, " "); | 
|  | type_print_base (TYPE_SELF_TYPE (type), | 
|  | stream, 0, passed_a_ptr, flags); | 
|  | gdb_printf (stream, "::"); | 
|  | } | 
|  | break; | 
|  |  | 
|  | case TYPE_CODE_REF: | 
|  | type_print_varspec_prefix (type->target_type (), stream, 0, 1, | 
|  | flags); | 
|  | gdb_printf (stream, "&"); | 
|  | break; | 
|  |  | 
|  | case TYPE_CODE_FUNC: | 
|  | if (passed_a_ptr) | 
|  | gdb_printf (stream, "("); | 
|  |  | 
|  | if (type->target_type () != NULL | 
|  | && type->target_type ()->code () != TYPE_CODE_VOID) | 
|  | { | 
|  | gdb_printf (stream, "function  "); | 
|  | } | 
|  | else | 
|  | { | 
|  | gdb_printf (stream, "procedure "); | 
|  | } | 
|  |  | 
|  | break; | 
|  |  | 
|  | case TYPE_CODE_ARRAY: | 
|  | if (passed_a_ptr) | 
|  | gdb_printf (stream, "("); | 
|  | gdb_printf (stream, "array "); | 
|  | if (type->target_type ()->length () > 0 | 
|  | && type->bounds ()->high.is_constant ()) | 
|  | gdb_printf (stream, "[%s..%s] ", | 
|  | plongest (type->bounds ()->low.const_val ()), | 
|  | plongest (type->bounds ()->high.const_val ())); | 
|  | gdb_printf (stream, "of "); | 
|  | break; | 
|  | } | 
|  | } | 
|  |  | 
|  | /* See p-lang.h.  */ | 
|  |  | 
|  | void | 
|  | pascal_language::print_func_args (struct type *type, struct ui_file *stream, | 
|  | const struct type_print_options *flags) const | 
|  | { | 
|  | int i, len = type->num_fields (); | 
|  |  | 
|  | if (len) | 
|  | { | 
|  | gdb_printf (stream, "("); | 
|  | } | 
|  | for (i = 0; i < len; i++) | 
|  | { | 
|  | if (i > 0) | 
|  | { | 
|  | gdb_puts (", ", stream); | 
|  | stream->wrap_here (4); | 
|  | } | 
|  | /*  Can we find if it is a var parameter ?? | 
|  | if ( TYPE_FIELD(type, i) == ) | 
|  | { | 
|  | gdb_printf (stream, "var "); | 
|  | } */ | 
|  | print_type (type->field (i).type (), ""	/* TYPE_FIELD_NAME | 
|  | seems invalid!  */ | 
|  | ,stream, -1, 0, flags); | 
|  | } | 
|  | if (len) | 
|  | { | 
|  | gdb_printf (stream, ")"); | 
|  | } | 
|  | } | 
|  |  | 
|  | /* See p-lang.h.  */ | 
|  |  | 
|  | void | 
|  | pascal_language::type_print_func_varspec_suffix  (struct type *type, | 
|  | struct ui_file *stream, | 
|  | int show, int passed_a_ptr, | 
|  | int demangled_args, | 
|  | const struct type_print_options *flags) const | 
|  | { | 
|  | if (type->target_type () == NULL | 
|  | || type->target_type ()->code () != TYPE_CODE_VOID) | 
|  | { | 
|  | gdb_printf (stream, " : "); | 
|  | type_print_varspec_prefix (type->target_type (), | 
|  | stream, 0, 0, flags); | 
|  |  | 
|  | if (type->target_type () == NULL) | 
|  | type_print_unknown_return_type (stream); | 
|  | else | 
|  | type_print_base (type->target_type (), stream, show, 0, | 
|  | flags); | 
|  |  | 
|  | type_print_varspec_suffix (type->target_type (), stream, 0, | 
|  | passed_a_ptr, 0, flags); | 
|  | } | 
|  | } | 
|  |  | 
|  | /* See p-lang.h.  */ | 
|  |  | 
|  | void | 
|  | pascal_language::type_print_varspec_suffix (struct type *type, | 
|  | struct ui_file *stream, | 
|  | int show, int passed_a_ptr, | 
|  | int demangled_args, | 
|  | const struct type_print_options *flags) const | 
|  | { | 
|  | if (type == 0) | 
|  | return; | 
|  |  | 
|  | if (type->name () && show <= 0) | 
|  | return; | 
|  |  | 
|  | QUIT; | 
|  |  | 
|  | switch (type->code ()) | 
|  | { | 
|  | case TYPE_CODE_ARRAY: | 
|  | if (passed_a_ptr) | 
|  | gdb_printf (stream, ")"); | 
|  | break; | 
|  |  | 
|  | case TYPE_CODE_METHOD: | 
|  | if (passed_a_ptr) | 
|  | gdb_printf (stream, ")"); | 
|  | type_print_method_args ("", "", stream); | 
|  | type_print_func_varspec_suffix (type, stream, show, | 
|  | passed_a_ptr, 0, flags); | 
|  | break; | 
|  |  | 
|  | case TYPE_CODE_PTR: | 
|  | case TYPE_CODE_REF: | 
|  | type_print_varspec_suffix (type->target_type (), | 
|  | stream, 0, 1, 0, flags); | 
|  | break; | 
|  |  | 
|  | case TYPE_CODE_FUNC: | 
|  | if (passed_a_ptr) | 
|  | gdb_printf (stream, ")"); | 
|  | if (!demangled_args) | 
|  | print_func_args (type, stream, flags); | 
|  | type_print_func_varspec_suffix (type, stream, show, | 
|  | passed_a_ptr, 0, flags); | 
|  | break; | 
|  | } | 
|  | } | 
|  |  | 
|  | /* See p-lang.h.  */ | 
|  |  | 
|  | void | 
|  | pascal_language::type_print_base (struct type *type, struct ui_file *stream, int show, | 
|  | int level, const struct type_print_options *flags) const | 
|  | { | 
|  | int i; | 
|  | int len; | 
|  | LONGEST lastval; | 
|  | enum | 
|  | { | 
|  | s_none, s_public, s_private, s_protected | 
|  | } | 
|  | section_type; | 
|  |  | 
|  | QUIT; | 
|  | stream->wrap_here (4); | 
|  | if (type == NULL) | 
|  | { | 
|  | fputs_styled ("<type unknown>", metadata_style.style (), stream); | 
|  | return; | 
|  | } | 
|  |  | 
|  | /* void pointer */ | 
|  | if ((type->code () == TYPE_CODE_PTR) | 
|  | && (type->target_type ()->code () == TYPE_CODE_VOID)) | 
|  | { | 
|  | gdb_puts (type->name () ? type->name () : "pointer", | 
|  | stream); | 
|  | return; | 
|  | } | 
|  | /* When SHOW is zero or less, and there is a valid type name, then always | 
|  | just print the type name directly from the type.  */ | 
|  |  | 
|  | if (show <= 0 | 
|  | && type->name () != NULL) | 
|  | { | 
|  | gdb_puts (type->name (), stream); | 
|  | return; | 
|  | } | 
|  |  | 
|  | type = check_typedef (type); | 
|  |  | 
|  | switch (type->code ()) | 
|  | { | 
|  | case TYPE_CODE_TYPEDEF: | 
|  | case TYPE_CODE_PTR: | 
|  | case TYPE_CODE_REF: | 
|  | type_print_base (type->target_type (), stream, show, level, | 
|  | flags); | 
|  | break; | 
|  |  | 
|  | case TYPE_CODE_ARRAY: | 
|  | print_type (type->target_type (), NULL, stream, 0, 0, flags); | 
|  | break; | 
|  |  | 
|  | case TYPE_CODE_FUNC: | 
|  | case TYPE_CODE_METHOD: | 
|  | break; | 
|  | case TYPE_CODE_STRUCT: | 
|  | if (type->name () != NULL) | 
|  | { | 
|  | gdb_puts (type->name (), stream); | 
|  | gdb_puts (" = ", stream); | 
|  | } | 
|  | if (HAVE_CPLUS_STRUCT (type)) | 
|  | { | 
|  | gdb_printf (stream, "class "); | 
|  | } | 
|  | else | 
|  | { | 
|  | gdb_printf (stream, "record "); | 
|  | } | 
|  | goto struct_union; | 
|  |  | 
|  | case TYPE_CODE_UNION: | 
|  | if (type->name () != NULL) | 
|  | { | 
|  | gdb_puts (type->name (), stream); | 
|  | gdb_puts (" = ", stream); | 
|  | } | 
|  | gdb_printf (stream, "case <?> of "); | 
|  |  | 
|  | struct_union: | 
|  | stream->wrap_here (4); | 
|  | if (show < 0) | 
|  | { | 
|  | /* If we just printed a tag name, no need to print anything else.  */ | 
|  | if (type->name () == NULL) | 
|  | gdb_printf (stream, "{...}"); | 
|  | } | 
|  | else if (show > 0 || type->name () == NULL) | 
|  | { | 
|  | type_print_derivation_info (stream, type); | 
|  |  | 
|  | gdb_printf (stream, "\n"); | 
|  | if ((type->num_fields () == 0) && (TYPE_NFN_FIELDS (type) == 0)) | 
|  | { | 
|  | if (type->is_stub ()) | 
|  | gdb_printf (stream, "%*s<incomplete type>\n", | 
|  | level + 4, ""); | 
|  | else | 
|  | gdb_printf (stream, "%*s<no data fields>\n", | 
|  | level + 4, ""); | 
|  | } | 
|  |  | 
|  | /* Start off with no specific section type, so we can print | 
|  | one for the first field we find, and use that section type | 
|  | thereafter until we find another type.  */ | 
|  |  | 
|  | section_type = s_none; | 
|  |  | 
|  | /* If there is a base class for this type, | 
|  | do not print the field that it occupies.  */ | 
|  |  | 
|  | len = type->num_fields (); | 
|  | for (i = TYPE_N_BASECLASSES (type); i < len; i++) | 
|  | { | 
|  | QUIT; | 
|  | /* Don't print out virtual function table.  */ | 
|  | if ((startswith (type->field (i).name (), "_vptr")) | 
|  | && is_cplus_marker ((type->field (i).name ())[5])) | 
|  | continue; | 
|  |  | 
|  | /* If this is a pascal object or class we can print the | 
|  | various section labels.  */ | 
|  |  | 
|  | if (HAVE_CPLUS_STRUCT (type)) | 
|  | { | 
|  | field &fld = type->field (i); | 
|  |  | 
|  | if (fld.is_protected ()) | 
|  | { | 
|  | if (section_type != s_protected) | 
|  | { | 
|  | section_type = s_protected; | 
|  | gdb_printf (stream, "%*sprotected\n", | 
|  | level + 2, ""); | 
|  | } | 
|  | } | 
|  | else if (fld.is_private ()) | 
|  | { | 
|  | if (section_type != s_private) | 
|  | { | 
|  | section_type = s_private; | 
|  | gdb_printf (stream, "%*sprivate\n", | 
|  | level + 2, ""); | 
|  | } | 
|  | } | 
|  | else | 
|  | { | 
|  | if (section_type != s_public) | 
|  | { | 
|  | section_type = s_public; | 
|  | gdb_printf (stream, "%*spublic\n", | 
|  | level + 2, ""); | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | print_spaces (level + 4, stream); | 
|  | if (type->field (i).is_static ()) | 
|  | gdb_printf (stream, "static "); | 
|  | print_type (type->field (i).type (), | 
|  | type->field (i).name (), | 
|  | stream, show - 1, level + 4, flags); | 
|  | if (!type->field (i).is_static () | 
|  | && type->field (i).is_packed ()) | 
|  | { | 
|  | /* It is a bitfield.  This code does not attempt | 
|  | to look at the bitpos and reconstruct filler, | 
|  | unnamed fields.  This would lead to misleading | 
|  | results if the compiler does not put out fields | 
|  | for such things (I don't know what it does).  */ | 
|  | gdb_printf (stream, " : %d", type->field (i).bitsize ()); | 
|  | } | 
|  | gdb_printf (stream, ";\n"); | 
|  | } | 
|  |  | 
|  | /* If there are both fields and methods, put a space between.  */ | 
|  | len = TYPE_NFN_FIELDS (type); | 
|  | if (len && section_type != s_none) | 
|  | gdb_printf (stream, "\n"); | 
|  |  | 
|  | /* Object pascal: print out the methods.  */ | 
|  |  | 
|  | for (i = 0; i < len; i++) | 
|  | { | 
|  | struct fn_field *f = TYPE_FN_FIELDLIST1 (type, i); | 
|  | int j, len2 = TYPE_FN_FIELDLIST_LENGTH (type, i); | 
|  | const char *method_name = TYPE_FN_FIELDLIST_NAME (type, i); | 
|  |  | 
|  | /* this is GNU C++ specific | 
|  | how can we know constructor/destructor? | 
|  | It might work for GNU pascal.  */ | 
|  | for (j = 0; j < len2; j++) | 
|  | { | 
|  | const char *physname = TYPE_FN_FIELD_PHYSNAME (f, j); | 
|  |  | 
|  | int is_constructor = (startswith (physname, "__ct__")); | 
|  | int is_destructor = (startswith (physname, "__dt__")); | 
|  |  | 
|  | QUIT; | 
|  | if (TYPE_FN_FIELD_PROTECTED (f, j)) | 
|  | { | 
|  | if (section_type != s_protected) | 
|  | { | 
|  | section_type = s_protected; | 
|  | gdb_printf (stream, "%*sprotected\n", | 
|  | level + 2, ""); | 
|  | } | 
|  | } | 
|  | else if (TYPE_FN_FIELD_PRIVATE (f, j)) | 
|  | { | 
|  | if (section_type != s_private) | 
|  | { | 
|  | section_type = s_private; | 
|  | gdb_printf (stream, "%*sprivate\n", | 
|  | level + 2, ""); | 
|  | } | 
|  | } | 
|  | else | 
|  | { | 
|  | if (section_type != s_public) | 
|  | { | 
|  | section_type = s_public; | 
|  | gdb_printf (stream, "%*spublic\n", | 
|  | level + 2, ""); | 
|  | } | 
|  | } | 
|  |  | 
|  | print_spaces (level + 4, stream); | 
|  | if (TYPE_FN_FIELD_STATIC_P (f, j)) | 
|  | gdb_printf (stream, "static "); | 
|  | if (TYPE_FN_FIELD_TYPE (f, j)->target_type () == 0) | 
|  | { | 
|  | /* Keep GDB from crashing here.  */ | 
|  | gdb_printf (stream, "<undefined type> %s;\n", | 
|  | TYPE_FN_FIELD_PHYSNAME (f, j)); | 
|  | break; | 
|  | } | 
|  |  | 
|  | if (is_constructor) | 
|  | { | 
|  | gdb_printf (stream, "constructor "); | 
|  | } | 
|  | else if (is_destructor) | 
|  | { | 
|  | gdb_printf (stream, "destructor  "); | 
|  | } | 
|  | else if (TYPE_FN_FIELD_TYPE (f, j)->target_type () != 0 | 
|  | && (TYPE_FN_FIELD_TYPE(f, j)->target_type ()->code () | 
|  | != TYPE_CODE_VOID)) | 
|  | { | 
|  | gdb_printf (stream, "function  "); | 
|  | } | 
|  | else | 
|  | { | 
|  | gdb_printf (stream, "procedure "); | 
|  | } | 
|  | /* This does not work, no idea why !!  */ | 
|  |  | 
|  | type_print_method_args (physname, method_name, stream); | 
|  |  | 
|  | if (TYPE_FN_FIELD_TYPE (f, j)->target_type () != 0 | 
|  | && (TYPE_FN_FIELD_TYPE(f, j)->target_type ()->code () | 
|  | != TYPE_CODE_VOID)) | 
|  | { | 
|  | gdb_puts (" : ", stream); | 
|  | type_print (TYPE_FN_FIELD_TYPE (f, j)->target_type (), | 
|  | "", stream, -1); | 
|  | } | 
|  | if (TYPE_FN_FIELD_VIRTUAL_P (f, j)) | 
|  | gdb_printf (stream, "; virtual"); | 
|  |  | 
|  | gdb_printf (stream, ";\n"); | 
|  | } | 
|  | } | 
|  | gdb_printf (stream, "%*send", level, ""); | 
|  | } | 
|  | break; | 
|  |  | 
|  | case TYPE_CODE_ENUM: | 
|  | if (type->name () != NULL) | 
|  | { | 
|  | gdb_puts (type->name (), stream); | 
|  | if (show > 0) | 
|  | gdb_puts (" ", stream); | 
|  | } | 
|  | /* enum is just defined by | 
|  | type enume_name = (enum_member1,enum_member2,...)  */ | 
|  | gdb_printf (stream, " = "); | 
|  | stream->wrap_here (4); | 
|  | if (show < 0) | 
|  | { | 
|  | /* If we just printed a tag name, no need to print anything else.  */ | 
|  | if (type->name () == NULL) | 
|  | gdb_printf (stream, "(...)"); | 
|  | } | 
|  | else if (show > 0 || type->name () == NULL) | 
|  | { | 
|  | gdb_printf (stream, "("); | 
|  | len = type->num_fields (); | 
|  | lastval = 0; | 
|  | for (i = 0; i < len; i++) | 
|  | { | 
|  | QUIT; | 
|  | if (i) | 
|  | gdb_printf (stream, ", "); | 
|  | stream->wrap_here (4); | 
|  | gdb_puts (type->field (i).name (), stream); | 
|  | if (lastval != type->field (i).loc_enumval ()) | 
|  | { | 
|  | gdb_printf (stream, | 
|  | " := %s", | 
|  | plongest (type->field (i).loc_enumval ())); | 
|  | lastval = type->field (i).loc_enumval (); | 
|  | } | 
|  | lastval++; | 
|  | } | 
|  | gdb_printf (stream, ")"); | 
|  | } | 
|  | break; | 
|  |  | 
|  | case TYPE_CODE_VOID: | 
|  | gdb_printf (stream, "void"); | 
|  | break; | 
|  |  | 
|  | case TYPE_CODE_UNDEF: | 
|  | gdb_printf (stream, "record <unknown>"); | 
|  | break; | 
|  |  | 
|  | case TYPE_CODE_ERROR: | 
|  | gdb_printf (stream, "%s", TYPE_ERROR_NAME (type)); | 
|  | break; | 
|  |  | 
|  | /* this probably does not work for enums.  */ | 
|  | case TYPE_CODE_RANGE: | 
|  | { | 
|  | struct type *target = type->target_type (); | 
|  |  | 
|  | print_type_scalar (target, type->bounds ()->low.const_val (), stream); | 
|  | gdb_puts ("..", stream); | 
|  | print_type_scalar (target, type->bounds ()->high.const_val (), stream); | 
|  | } | 
|  | break; | 
|  |  | 
|  | case TYPE_CODE_SET: | 
|  | gdb_puts ("set of ", stream); | 
|  | print_type (type->index_type (), "", stream, | 
|  | show - 1, level, flags); | 
|  | break; | 
|  |  | 
|  | case TYPE_CODE_STRING: | 
|  | gdb_puts ("String", stream); | 
|  | break; | 
|  |  | 
|  | default: | 
|  | /* Handle types not explicitly handled by the other cases, | 
|  | such as fundamental types.  For these, just print whatever | 
|  | the type name is, as recorded in the type itself.  If there | 
|  | is no type name, then complain.  */ | 
|  | if (type->name () != NULL) | 
|  | { | 
|  | gdb_puts (type->name (), stream); | 
|  | } | 
|  | else | 
|  | { | 
|  | /* At least for dump_symtab, it is important that this not be | 
|  | an error ().  */ | 
|  | fprintf_styled (stream, metadata_style.style (), | 
|  | "<invalid unnamed pascal type code %d>", | 
|  | type->code ()); | 
|  | } | 
|  | break; | 
|  | } | 
|  | } |