| /* Support for printing Ada types for GDB, the GNU debugger. | 
 |    Copyright (C) 1986-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/>.  */ | 
 |  | 
 | #include "event-top.h" | 
 | #include "gdbtypes.h" | 
 | #include "value.h" | 
 | #include "c-lang.h" | 
 | #include "cli/cli-style.h" | 
 | #include "typeprint.h" | 
 | #include "ada-lang.h" | 
 | #include <ctype.h> | 
 |  | 
 | static int print_selected_record_field_types (struct type *, struct type *, | 
 | 					      int, int, | 
 | 					      struct ui_file *, int, int, | 
 | 					      const struct type_print_options *); | 
 |  | 
 | static int print_record_field_types (struct type *, struct type *, | 
 | 				     struct ui_file *, int, int, | 
 | 				     const struct type_print_options *); | 
 |  | 
 |  | 
 |  | 
 | static char *name_buffer; | 
 | static int name_buffer_len; | 
 |  | 
 | /* The (decoded) Ada name of TYPE.  This value persists until the | 
 |    next call.  */ | 
 |  | 
 | static char * | 
 | decoded_type_name (struct type *type) | 
 | { | 
 |   if (ada_type_name (type) == NULL) | 
 |     return NULL; | 
 |   else | 
 |     { | 
 |       const char *raw_name = ada_type_name (type); | 
 |       char *s, *q; | 
 |  | 
 |       if (name_buffer == NULL || name_buffer_len <= strlen (raw_name)) | 
 | 	{ | 
 | 	  name_buffer_len = 16 + 2 * strlen (raw_name); | 
 | 	  name_buffer = (char *) xrealloc (name_buffer, name_buffer_len); | 
 | 	} | 
 |       strcpy (name_buffer, raw_name); | 
 |  | 
 |       s = (char *) strstr (name_buffer, "___"); | 
 |       if (s != NULL) | 
 | 	*s = '\0'; | 
 |  | 
 |       s = name_buffer + strlen (name_buffer) - 1; | 
 |       while (s > name_buffer && (s[0] != '_' || s[-1] != '_')) | 
 | 	s -= 1; | 
 |  | 
 |       if (s == name_buffer) | 
 | 	return name_buffer; | 
 |  | 
 |       if (!islower (s[1])) | 
 | 	return NULL; | 
 |  | 
 |       for (s = q = name_buffer; *s != '\0'; q += 1) | 
 | 	{ | 
 | 	  if (s[0] == '_' && s[1] == '_') | 
 | 	    { | 
 | 	      *q = '.'; | 
 | 	      s += 2; | 
 | 	    } | 
 | 	  else | 
 | 	    { | 
 | 	      *q = *s; | 
 | 	      s += 1; | 
 | 	    } | 
 | 	} | 
 |       *q = '\0'; | 
 |       return name_buffer; | 
 |     } | 
 | } | 
 |  | 
 | /* Return nonzero if TYPE is a subrange type, and its bounds | 
 |    are identical to the bounds of its subtype.  */ | 
 |  | 
 | static int | 
 | type_is_full_subrange_of_target_type (struct type *type) | 
 | { | 
 |   struct type *subtype; | 
 |  | 
 |   if (type->code () != TYPE_CODE_RANGE) | 
 |     return 0; | 
 |  | 
 |   subtype = type->target_type (); | 
 |   if (subtype == NULL) | 
 |     return 0; | 
 |  | 
 |   if (is_dynamic_type (type)) | 
 |     return 0; | 
 |  | 
 |   if (ada_discrete_type_low_bound (type) | 
 |       != ada_discrete_type_low_bound (subtype)) | 
 |     return 0; | 
 |  | 
 |   if (ada_discrete_type_high_bound (type) | 
 |       != ada_discrete_type_high_bound (subtype)) | 
 |     return 0; | 
 |  | 
 |   return 1; | 
 | } | 
 |  | 
 | /* Print TYPE on STREAM, preferably as a range if BOUNDS_PREFERRED_P | 
 |    is nonzero.  */ | 
 |  | 
 | static void | 
 | print_range (struct type *type, struct ui_file *stream, | 
 | 	     int bounds_preferred_p) | 
 | { | 
 |   if (!bounds_preferred_p) | 
 |     { | 
 |       /* Try stripping all TYPE_CODE_RANGE layers whose bounds | 
 | 	 are identical to the bounds of their subtype.  When | 
 | 	 the bounds of both types match, it can allow us to | 
 | 	 print a range using the name of its base type, which | 
 | 	 is easier to read.  For instance, we would print... | 
 |  | 
 | 	     array (character) of ... | 
 |  | 
 | 	 ... instead of... | 
 |  | 
 | 	     array ('["00"]' .. '["ff"]') of ...  */ | 
 |       while (type_is_full_subrange_of_target_type (type)) | 
 | 	type = type->target_type (); | 
 |     } | 
 |  | 
 |   switch (type->code ()) | 
 |     { | 
 |     case TYPE_CODE_RANGE: | 
 |     case TYPE_CODE_ENUM: | 
 |       { | 
 | 	LONGEST lo = 0, hi = 0; /* init for gcc -Wall */ | 
 | 	int got_error = 0; | 
 |  | 
 | 	try | 
 | 	  { | 
 | 	    lo = ada_discrete_type_low_bound (type); | 
 | 	    hi = ada_discrete_type_high_bound (type); | 
 | 	  } | 
 | 	catch (const gdb_exception_error &e) | 
 | 	  { | 
 | 	    /* This can happen when the range is dynamic.  Sometimes, | 
 | 	       resolving dynamic property values requires us to have | 
 | 	       access to an actual object, which is not available | 
 | 	       when the user is using the "ptype" command on a type. | 
 | 	       Print the range as an unbounded range.  */ | 
 | 	    gdb_printf (stream, "<>"); | 
 | 	    got_error = 1; | 
 | 	  } | 
 |  | 
 | 	if (!got_error) | 
 | 	  { | 
 | 	    ada_print_scalar (type, lo, stream); | 
 | 	    gdb_printf (stream, " .. "); | 
 | 	    ada_print_scalar (type, hi, stream); | 
 | 	  } | 
 |       } | 
 |       break; | 
 |     default: | 
 |       gdb_printf (stream, "%.*s", | 
 | 		  ada_name_prefix_len (type->name ()), | 
 | 		  type->name ()); | 
 |       break; | 
 |     } | 
 | } | 
 |  | 
 | /* Print the number or discriminant bound at BOUNDS+*N on STREAM, and | 
 |    set *N past the bound and its delimiter, if any.  */ | 
 |  | 
 | static void | 
 | print_range_bound (struct type *type, const char *bounds, int *n, | 
 | 		   struct ui_file *stream) | 
 | { | 
 |   LONGEST B; | 
 |  | 
 |   if (ada_scan_number (bounds, *n, &B, n)) | 
 |     { | 
 |       /* STABS decodes all range types which bounds are 0 .. -1 as | 
 | 	 unsigned integers (ie. the type code is TYPE_CODE_INT, not | 
 | 	 TYPE_CODE_RANGE).  Unfortunately, ada_print_scalar() relies | 
 | 	 on the unsigned flag to determine whether the bound should | 
 | 	 be printed as a signed or an unsigned value.  This causes | 
 | 	 the upper bound of the 0 .. -1 range types to be printed as | 
 | 	 a very large unsigned number instead of -1. | 
 | 	 To workaround this stabs deficiency, we replace the TYPE by NULL | 
 | 	 to indicate default output when we detect that the bound is negative, | 
 | 	 and the type is a TYPE_CODE_INT.  The bound is negative when | 
 | 	 'm' is the last character of the number scanned in BOUNDS.  */ | 
 |       if (bounds[*n - 1] == 'm' && type->code () == TYPE_CODE_INT) | 
 | 	type = NULL; | 
 |       ada_print_scalar (type, B, stream); | 
 |       if (bounds[*n] == '_') | 
 | 	*n += 2; | 
 |     } | 
 |   else | 
 |     { | 
 |       int bound_len; | 
 |       const char *bound = bounds + *n; | 
 |       const char *pend; | 
 |  | 
 |       pend = strstr (bound, "__"); | 
 |       if (pend == NULL) | 
 | 	*n += bound_len = strlen (bound); | 
 |       else | 
 | 	{ | 
 | 	  bound_len = pend - bound; | 
 | 	  *n += bound_len + 2; | 
 | 	} | 
 |       gdb_printf (stream, "%.*s", bound_len, bound); | 
 |     } | 
 | } | 
 |  | 
 | /* Assuming NAME[0 .. NAME_LEN-1] is the name of a range type, print | 
 |    the value (if found) of the bound indicated by SUFFIX ("___L" or | 
 |    "___U") according to the ___XD conventions.  */ | 
 |  | 
 | static void | 
 | print_dynamic_range_bound (struct type *type, const char *name, int name_len, | 
 | 			   const char *suffix, struct ui_file *stream) | 
 | { | 
 |   LONGEST B; | 
 |   std::string name_buf (name, name_len); | 
 |   name_buf += suffix; | 
 |  | 
 |   if (get_int_var_value (name_buf.c_str (), B)) | 
 |     ada_print_scalar (type, B, stream); | 
 |   else | 
 |     gdb_printf (stream, "?"); | 
 | } | 
 |  | 
 | /* Print RAW_TYPE as a range type, using any bound information | 
 |    following the GNAT encoding (if available). | 
 |  | 
 |    If BOUNDS_PREFERRED_P is nonzero, force the printing of the range | 
 |    using its bounds.  Otherwise, try printing the range without | 
 |    printing the value of the bounds, if possible (this is only | 
 |    considered a hint, not a guaranty).  */ | 
 |  | 
 | static void | 
 | print_range_type (struct type *raw_type, struct ui_file *stream, | 
 | 		  int bounds_preferred_p) | 
 | { | 
 |   const char *name; | 
 |   struct type *base_type; | 
 |   const char *subtype_info; | 
 |  | 
 |   gdb_assert (raw_type != NULL); | 
 |   name = raw_type->name (); | 
 |   gdb_assert (name != NULL); | 
 |  | 
 |   if (raw_type->code () == TYPE_CODE_RANGE) | 
 |     base_type = raw_type->target_type (); | 
 |   else | 
 |     base_type = raw_type; | 
 |  | 
 |   subtype_info = strstr (name, "___XD"); | 
 |   if (subtype_info == NULL) | 
 |     print_range (raw_type, stream, bounds_preferred_p); | 
 |   else | 
 |     { | 
 |       int prefix_len = subtype_info - name; | 
 |       const char *bounds_str; | 
 |       int n; | 
 |  | 
 |       subtype_info += 5; | 
 |       bounds_str = strchr (subtype_info, '_'); | 
 |       n = 1; | 
 |  | 
 |       if (*subtype_info == 'L') | 
 | 	{ | 
 | 	  print_range_bound (base_type, bounds_str, &n, stream); | 
 | 	  subtype_info += 1; | 
 | 	} | 
 |       else | 
 | 	print_dynamic_range_bound (base_type, name, prefix_len, "___L", | 
 | 				   stream); | 
 |  | 
 |       gdb_printf (stream, " .. "); | 
 |  | 
 |       if (*subtype_info == 'U') | 
 | 	print_range_bound (base_type, bounds_str, &n, stream); | 
 |       else | 
 | 	print_dynamic_range_bound (base_type, name, prefix_len, "___U", | 
 | 				   stream); | 
 |     } | 
 | } | 
 |  | 
 | /* Print enumerated type TYPE on STREAM.  */ | 
 |  | 
 | static void | 
 | print_enum_type (struct type *type, struct ui_file *stream) | 
 | { | 
 |   int len = type->num_fields (); | 
 |   int i; | 
 |   LONGEST lastval; | 
 |  | 
 |   gdb_printf (stream, "("); | 
 |   stream->wrap_here (1); | 
 |  | 
 |   lastval = 0; | 
 |   for (i = 0; i < len; i++) | 
 |     { | 
 |       QUIT; | 
 |       if (i) | 
 | 	gdb_printf (stream, ", "); | 
 |       stream->wrap_here (4); | 
 |       fputs_styled (ada_enum_name (type->field (i).name ()), | 
 | 		    variable_name_style.style (), 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 += 1; | 
 |     } | 
 |   gdb_printf (stream, ")"); | 
 | } | 
 |  | 
 | /* Print simple (constrained) array type TYPE on STREAM.  LEVEL is the | 
 |    recursion (indentation) level, in case the element type itself has | 
 |    nested structure, and SHOW is the number of levels of internal | 
 |    structure to show (see ada_print_type).  */ | 
 |  | 
 | static void | 
 | print_array_type (struct type *type, struct ui_file *stream, int show, | 
 | 		  int level, const struct type_print_options *flags) | 
 | { | 
 |   int bitsize; | 
 |   int n_indices; | 
 |   struct type *elt_type = NULL; | 
 |  | 
 |   if (ada_is_constrained_packed_array_type (type)) | 
 |     type = ada_coerce_to_simple_array_type (type); | 
 |  | 
 |   bitsize = 0; | 
 |   gdb_printf (stream, "array ("); | 
 |  | 
 |   if (type == NULL) | 
 |     { | 
 |       fprintf_styled (stream, metadata_style.style (), | 
 | 		      _("<undecipherable array type>")); | 
 |       return; | 
 |     } | 
 |  | 
 |   n_indices = -1; | 
 |   if (ada_is_simple_array_type (type)) | 
 |     { | 
 |       struct type *range_desc_type; | 
 |       struct type *arr_type; | 
 |  | 
 |       range_desc_type = ada_find_parallel_type (type, "___XA"); | 
 |       ada_fixup_array_indexes_type (range_desc_type); | 
 |  | 
 |       bitsize = 0; | 
 |       if (range_desc_type == NULL) | 
 | 	{ | 
 | 	  for (arr_type = type; arr_type->code () == TYPE_CODE_ARRAY; ) | 
 | 	    { | 
 | 	      if (arr_type != type) | 
 | 		gdb_printf (stream, ", "); | 
 | 	      print_range (arr_type->index_type (), stream, | 
 | 			   0 /* bounds_preferred_p */); | 
 | 	      if (arr_type->field (0).bitsize () > 0) | 
 | 		bitsize = arr_type->field (0).bitsize (); | 
 | 	      /* A multi-dimensional array is represented using a | 
 | 		 sequence of array types.  If one of these types has a | 
 | 		 name, then it is not another dimension of the outer | 
 | 		 array, but rather the element type of the outermost | 
 | 		 array.  */ | 
 | 	      arr_type = arr_type->target_type (); | 
 | 	      if (arr_type->name () != nullptr) | 
 | 		break; | 
 | 	    } | 
 | 	} | 
 |       else | 
 | 	{ | 
 | 	  int k; | 
 |  | 
 | 	  n_indices = range_desc_type->num_fields (); | 
 | 	  for (k = 0, arr_type = type; | 
 | 	       k < n_indices; | 
 | 	       k += 1, arr_type = arr_type->target_type ()) | 
 | 	    { | 
 | 	      if (k > 0) | 
 | 		gdb_printf (stream, ", "); | 
 | 	      print_range_type (range_desc_type->field (k).type (), | 
 | 				stream, 0 /* bounds_preferred_p */); | 
 | 	      if (arr_type->field (0).bitsize () > 0) | 
 | 		bitsize = arr_type->field (0).bitsize (); | 
 | 	    } | 
 | 	} | 
 |     } | 
 |   else | 
 |     { | 
 |       int i, i0; | 
 |  | 
 |       for (i = i0 = ada_array_arity (type); i > 0; i -= 1) | 
 | 	gdb_printf (stream, "%s<>", i == i0 ? "" : ", "); | 
 |     } | 
 |  | 
 |   elt_type = ada_array_element_type (type, n_indices); | 
 |   gdb_printf (stream, ") of "); | 
 |   stream->wrap_here (0); | 
 |   ada_print_type (elt_type, "", stream, show == 0 ? 0 : show - 1, level + 1, | 
 | 		  flags); | 
 |   /* Arrays with variable-length elements are never bit-packed in practice but | 
 |      compilers have to describe their stride so that we can properly fetch | 
 |      individual elements.  Do not say the array is packed in this case.  */ | 
 |   if (bitsize > 0 && !is_dynamic_type (elt_type)) | 
 |     gdb_printf (stream, " <packed: %d-bit elements>", bitsize); | 
 | } | 
 |  | 
 | /* Print the choices encoded by field FIELD_NUM of variant-part TYPE on | 
 |    STREAM, assuming that VAL_TYPE (if non-NULL) is the type of the | 
 |    values.  Return non-zero if the field is an encoding of | 
 |    discriminant values, as in a standard variant record, and 0 if the | 
 |    field is not so encoded (as happens with single-component variants | 
 |    in types annotated with pragma Unchecked_Union).  */ | 
 |  | 
 | static int | 
 | print_choices (struct type *type, int field_num, struct ui_file *stream, | 
 | 	       struct type *val_type) | 
 | { | 
 |   int have_output; | 
 |   int p; | 
 |   const char *name = type->field (field_num).name (); | 
 |  | 
 |   have_output = 0; | 
 |  | 
 |   /* Skip over leading 'V': NOTE soon to be obsolete.  */ | 
 |   if (name[0] == 'V') | 
 |     { | 
 |       if (!ada_scan_number (name, 1, NULL, &p)) | 
 | 	goto Huh; | 
 |     } | 
 |   else | 
 |     p = 0; | 
 |  | 
 |   while (1) | 
 |     { | 
 |       switch (name[p]) | 
 | 	{ | 
 | 	default: | 
 | 	  goto Huh; | 
 | 	case '_': | 
 | 	case '\0': | 
 | 	  gdb_printf (stream, " =>"); | 
 | 	  return 1; | 
 | 	case 'S': | 
 | 	case 'R': | 
 | 	case 'O': | 
 | 	  if (have_output) | 
 | 	    gdb_printf (stream, " | "); | 
 | 	  have_output = 1; | 
 | 	  break; | 
 | 	} | 
 |  | 
 |       switch (name[p]) | 
 | 	{ | 
 | 	case 'S': | 
 | 	  { | 
 | 	    LONGEST W; | 
 |  | 
 | 	    if (!ada_scan_number (name, p + 1, &W, &p)) | 
 | 	      goto Huh; | 
 | 	    ada_print_scalar (val_type, W, stream); | 
 | 	    break; | 
 | 	  } | 
 | 	case 'R': | 
 | 	  { | 
 | 	    LONGEST L, U; | 
 |  | 
 | 	    if (!ada_scan_number (name, p + 1, &L, &p) | 
 | 		|| name[p] != 'T' || !ada_scan_number (name, p + 1, &U, &p)) | 
 | 	      goto Huh; | 
 | 	    ada_print_scalar (val_type, L, stream); | 
 | 	    gdb_printf (stream, " .. "); | 
 | 	    ada_print_scalar (val_type, U, stream); | 
 | 	    break; | 
 | 	  } | 
 | 	case 'O': | 
 | 	  gdb_printf (stream, "others"); | 
 | 	  p += 1; | 
 | 	  break; | 
 | 	} | 
 |     } | 
 |  | 
 | Huh: | 
 |   gdb_printf (stream, "? =>"); | 
 |   return 0; | 
 | } | 
 |  | 
 | /* A helper for print_variant_clauses that prints the members of | 
 |    VAR_TYPE.  DISCR_TYPE is the type of the discriminant (or nullptr | 
 |    if not available).  The discriminant is contained in OUTER_TYPE. | 
 |    STREAM, LEVEL, SHOW, and FLAGS are the same as for | 
 |    ada_print_type.  */ | 
 |  | 
 | static void | 
 | print_variant_clauses (struct type *var_type, struct type *discr_type, | 
 | 		       struct type *outer_type, struct ui_file *stream, | 
 | 		       int show, int level, | 
 | 		       const struct type_print_options *flags) | 
 | { | 
 |   for (int i = 0; i < var_type->num_fields (); i += 1) | 
 |     { | 
 |       gdb_printf (stream, "\n%*swhen ", level, ""); | 
 |       if (print_choices (var_type, i, stream, discr_type)) | 
 | 	{ | 
 | 	  if (print_record_field_types (var_type->field (i).type (), | 
 | 					outer_type, stream, show, level, | 
 | 					flags) | 
 | 	      <= 0) | 
 | 	    gdb_printf (stream, " null;"); | 
 | 	} | 
 |       else | 
 | 	print_selected_record_field_types (var_type, outer_type, i, i, | 
 | 					   stream, show, level, flags); | 
 |     } | 
 | } | 
 |  | 
 | /* Assuming that field FIELD_NUM of TYPE represents variants whose | 
 |    discriminant is contained in OUTER_TYPE, print its components on STREAM. | 
 |    LEVEL is the recursion (indentation) level, in case any of the fields | 
 |    themselves have nested structure, and SHOW is the number of levels of  | 
 |    internal structure to show (see ada_print_type).  For this purpose, | 
 |    fields nested in a variant part are taken to be at the same level as | 
 |    the fields immediately outside the variant part.  */ | 
 |  | 
 | static void | 
 | print_variant_clauses (struct type *type, int field_num, | 
 | 		       struct type *outer_type, struct ui_file *stream, | 
 | 		       int show, int level, | 
 | 		       const struct type_print_options *flags) | 
 | { | 
 |   struct type *var_type, *par_type; | 
 |   struct type *discr_type; | 
 |  | 
 |   var_type = type->field (field_num).type (); | 
 |   discr_type = ada_variant_discrim_type (var_type, outer_type); | 
 |  | 
 |   if (var_type->code () == TYPE_CODE_PTR) | 
 |     { | 
 |       var_type = var_type->target_type (); | 
 |       if (var_type == NULL || var_type->code () != TYPE_CODE_UNION) | 
 | 	return; | 
 |     } | 
 |  | 
 |   par_type = ada_find_parallel_type (var_type, "___XVU"); | 
 |   if (par_type != NULL) | 
 |     var_type = par_type; | 
 |  | 
 |   print_variant_clauses (var_type, discr_type, outer_type, stream, show, | 
 | 			 level + 4, flags); | 
 | } | 
 |  | 
 | /* Assuming that field FIELD_NUM of TYPE is a variant part whose | 
 |    discriminants are contained in OUTER_TYPE, print a description of it | 
 |    on STREAM.  LEVEL is the recursion (indentation) level, in case any of | 
 |    the fields themselves have nested structure, and SHOW is the number of | 
 |    levels of internal structure to show (see ada_print_type).  For this | 
 |    purpose, fields nested in a variant part are taken to be at the same | 
 |    level as the fields immediately outside the variant part.  */ | 
 |  | 
 | static void | 
 | print_variant_part (struct type *type, int field_num, struct type *outer_type, | 
 | 		    struct ui_file *stream, int show, int level, | 
 | 		    const struct type_print_options *flags) | 
 | { | 
 |   const char *variant | 
 |     = ada_variant_discrim_name (type->field (field_num).type ()); | 
 |   if (*variant == '\0') | 
 |     variant = "?"; | 
 |  | 
 |   gdb_printf (stream, "\n%*scase %s is", level + 4, "", variant); | 
 |   print_variant_clauses (type, field_num, outer_type, stream, show, | 
 | 			 level + 4, flags); | 
 |   gdb_printf (stream, "\n%*send case;", level + 4, ""); | 
 | } | 
 |  | 
 | /* Print a description on STREAM of the fields FLD0 through FLD1 in | 
 |    record or union type TYPE, whose discriminants are in OUTER_TYPE. | 
 |    LEVEL is the recursion (indentation) level, in case any of the | 
 |    fields themselves have nested structure, and SHOW is the number of | 
 |    levels of internal structure to show (see ada_print_type).  Does | 
 |    not print parent type information of TYPE.  Returns 0 if no fields | 
 |    printed, -1 for an incomplete type, else > 0.  Prints each field | 
 |    beginning on a new line, but does not put a new line at end.  */ | 
 |  | 
 | static int | 
 | print_selected_record_field_types (struct type *type, struct type *outer_type, | 
 | 				   int fld0, int fld1, | 
 | 				   struct ui_file *stream, int show, int level, | 
 | 				   const struct type_print_options *flags) | 
 | { | 
 |   int i, flds; | 
 |  | 
 |   flds = 0; | 
 |  | 
 |   if (fld0 > fld1 && type->is_stub ()) | 
 |     return -1; | 
 |  | 
 |   for (i = fld0; i <= fld1; i += 1) | 
 |     { | 
 |       QUIT; | 
 |  | 
 |       if (ada_is_parent_field (type, i) || ada_is_ignored_field (type, i)) | 
 | 	; | 
 |       else if (ada_is_wrapper_field (type, i)) | 
 | 	flds += print_record_field_types (type->field (i).type (), type, | 
 | 					  stream, show, level, flags); | 
 |       else if (ada_is_variant_part (type, i)) | 
 | 	{ | 
 | 	  print_variant_part (type, i, outer_type, stream, show, level, flags); | 
 | 	  flds = 1; | 
 | 	} | 
 |       else | 
 | 	{ | 
 | 	  flds += 1; | 
 | 	  gdb_printf (stream, "\n%*s", level + 4, ""); | 
 | 	  ada_print_type (type->field (i).type (), | 
 | 			  type->field (i).name (), | 
 | 			  stream, show - 1, level + 4, flags); | 
 | 	  gdb_printf (stream, ";"); | 
 | 	} | 
 |     } | 
 |  | 
 |   return flds; | 
 | } | 
 |  | 
 | static void print_record_field_types_dynamic | 
 |   (const gdb::array_view<variant_part> &parts, | 
 |    int from, int to, struct type *type, struct ui_file *stream, | 
 |    int show, int level, const struct type_print_options *flags); | 
 |  | 
 | /* Print the choices encoded by VARIANT on STREAM.  LEVEL is the | 
 |    indentation level.  The type of the discriminant for VARIANT is | 
 |    given by DISR_TYPE.  */ | 
 |  | 
 | static void | 
 | print_choices (struct type *discr_type, const variant &variant, | 
 | 	       struct ui_file *stream, int level) | 
 | { | 
 |   gdb_printf (stream, "\n%*swhen ", level, ""); | 
 |   if (variant.is_default ()) | 
 |     gdb_printf (stream, "others"); | 
 |   else | 
 |     { | 
 |       bool first = true; | 
 |       for (const discriminant_range &range : variant.discriminants) | 
 | 	{ | 
 | 	  if (!first) | 
 | 	    gdb_printf (stream, " | "); | 
 | 	  first = false; | 
 |  | 
 | 	  ada_print_scalar (discr_type, range.low, stream); | 
 | 	  if (range.low != range.high) | 
 | 	    ada_print_scalar (discr_type, range.high, stream); | 
 | 	} | 
 |     } | 
 |  | 
 |   gdb_printf (stream, " =>"); | 
 | } | 
 |  | 
 | /* Print a single variant part, PART, on STREAM.  TYPE is the | 
 |    enclosing type.  SHOW, LEVEL, and FLAGS are the usual type-printing | 
 |    settings.  This prints information about PART and the fields it | 
 |    controls.  It returns the index of the next field that should be | 
 |    shown -- that is, one after the last field printed by this | 
 |    call.  */ | 
 |  | 
 | static int | 
 | print_variant_part (const variant_part &part, | 
 | 		    struct type *type, struct ui_file *stream, | 
 | 		    int show, int level, | 
 | 		    const struct type_print_options *flags) | 
 | { | 
 |   struct type *discr_type = nullptr; | 
 |   const char *name; | 
 |   if (part.discriminant_index == -1) | 
 |     name = "?"; | 
 |   else | 
 |     { | 
 |       name = type->field (part.discriminant_index).name ();; | 
 |       discr_type = type->field (part.discriminant_index).type (); | 
 |     } | 
 |  | 
 |   gdb_printf (stream, "\n%*scase %s is", level + 4, "", name); | 
 |  | 
 |   int last_field = -1; | 
 |   for (const variant &variant : part.variants) | 
 |     { | 
 |       print_choices (discr_type, variant, stream, level + 8); | 
 |  | 
 |       if (variant.first_field == variant.last_field) | 
 | 	gdb_printf (stream, " null;"); | 
 |       else | 
 | 	{ | 
 | 	  print_record_field_types_dynamic (variant.parts, | 
 | 					    variant.first_field, | 
 | 					    variant.last_field, type, stream, | 
 | 					    show, level + 8, flags); | 
 | 	  last_field = variant.last_field; | 
 | 	} | 
 |     } | 
 |  | 
 |   gdb_printf (stream, "\n%*send case;", level + 4, ""); | 
 |  | 
 |   return last_field; | 
 | } | 
 |  | 
 | /* Print some fields of TYPE to STREAM.  SHOW, LEVEL, and FLAGS are | 
 |    the usual type-printing settings.  PARTS is the array of variant | 
 |    parts that correspond to the range of fields to be printed.  FROM | 
 |    and TO are the range of fields to print.  */ | 
 |  | 
 | static void | 
 | print_record_field_types_dynamic (const gdb::array_view<variant_part> &parts, | 
 | 				  int from, int to, | 
 | 				  struct type *type, struct ui_file *stream, | 
 | 				  int show, int level, | 
 | 				  const struct type_print_options *flags) | 
 | { | 
 |   int field = from; | 
 |  | 
 |   for (const variant_part &part : parts) | 
 |     { | 
 |       if (part.variants.empty ()) | 
 | 	continue; | 
 |  | 
 |       /* Print any non-varying fields.  */ | 
 |       int first_varying = part.variants[0].first_field; | 
 |       print_selected_record_field_types (type, type, field, | 
 | 					 first_varying - 1, stream, | 
 | 					 show, level, flags); | 
 |  | 
 |       field = print_variant_part (part, type, stream, show, level, flags); | 
 |     } | 
 |  | 
 |   /* Print any trailing fields that we were asked to print.  */ | 
 |   print_selected_record_field_types (type, type, field, to - 1, stream, show, | 
 | 				     level, flags); | 
 | } | 
 |  | 
 | /* Print a description on STREAM of all fields of record or union type | 
 |    TYPE, as for print_selected_record_field_types, above.  */ | 
 |  | 
 | static int | 
 | print_record_field_types (struct type *type, struct type *outer_type, | 
 | 			  struct ui_file *stream, int show, int level, | 
 | 			  const struct type_print_options *flags) | 
 | { | 
 |   struct dynamic_prop *prop = type->dyn_prop (DYN_PROP_VARIANT_PARTS); | 
 |   if (prop != nullptr) | 
 |     { | 
 |       if (prop->kind () == PROP_TYPE) | 
 | 	{ | 
 | 	  type = prop->original_type (); | 
 | 	  prop = type->dyn_prop (DYN_PROP_VARIANT_PARTS); | 
 | 	} | 
 |       gdb_assert (prop->kind () == PROP_VARIANT_PARTS); | 
 |       print_record_field_types_dynamic (*prop->variant_parts (), | 
 | 					0, type->num_fields (), | 
 | 					type, stream, show, level, flags); | 
 |       return type->num_fields (); | 
 |     } | 
 |  | 
 |   return print_selected_record_field_types (type, outer_type, | 
 | 					    0, type->num_fields () - 1, | 
 | 					    stream, show, level, flags); | 
 | } | 
 |     | 
 |  | 
 | /* Print record type TYPE on STREAM.  LEVEL is the recursion (indentation) | 
 |    level, in case the element type itself has nested structure, and SHOW is | 
 |    the number of levels of internal structure to show (see ada_print_type).  */ | 
 |  | 
 | static void | 
 | print_record_type (struct type *type0, struct ui_file *stream, int show, | 
 | 		   int level, const struct type_print_options *flags) | 
 | { | 
 |   struct type *parent_type; | 
 |   struct type *type; | 
 |  | 
 |   type = ada_find_parallel_type (type0, "___XVE"); | 
 |   if (type == NULL) | 
 |     type = type0; | 
 |  | 
 |   parent_type = ada_parent_type (type); | 
 |   if (ada_type_name (parent_type) != NULL) | 
 |     { | 
 |       const char *parent_name = decoded_type_name (parent_type); | 
 |  | 
 |       /* If we fail to decode the parent type name, then use the parent | 
 | 	 type name as is.  Not pretty, but should never happen except | 
 | 	 when the debugging info is incomplete or incorrect.  This | 
 | 	 prevents a crash trying to print a NULL pointer.  */ | 
 |       if (parent_name == NULL) | 
 | 	parent_name = ada_type_name (parent_type); | 
 |       gdb_printf (stream, "new %s with record", parent_name); | 
 |     } | 
 |   else if (parent_type == NULL && ada_is_tagged_type (type, 0)) | 
 |     gdb_printf (stream, "tagged record"); | 
 |   else | 
 |     gdb_printf (stream, "record"); | 
 |  | 
 |   if (show < 0) | 
 |     gdb_printf (stream, " ... end record"); | 
 |   else | 
 |     { | 
 |       int flds; | 
 |  | 
 |       flds = 0; | 
 |       if (parent_type != NULL && ada_type_name (parent_type) == NULL) | 
 | 	flds += print_record_field_types (parent_type, parent_type, | 
 | 					  stream, show, level, flags); | 
 |       flds += print_record_field_types (type, type, stream, show, level, | 
 | 					flags); | 
 |  | 
 |       if (flds > 0) | 
 | 	gdb_printf (stream, "\n%*send record", level, ""); | 
 |       else if (flds < 0) | 
 | 	gdb_printf (stream, _(" <incomplete type> end record")); | 
 |       else | 
 | 	gdb_printf (stream, " null; end record"); | 
 |     } | 
 | } | 
 |  | 
 | /* Print the unchecked union type TYPE in something resembling Ada | 
 |    format on STREAM.  LEVEL is the recursion (indentation) level | 
 |    in case the element type itself has nested structure, and SHOW is the | 
 |    number of levels of internal structure to show (see ada_print_type).  */ | 
 | static void | 
 | print_unchecked_union_type (struct type *type, struct ui_file *stream, | 
 | 			    int show, int level, | 
 | 			    const struct type_print_options *flags) | 
 | { | 
 |   if (show < 0) | 
 |     gdb_printf (stream, "record (?) is ... end record"); | 
 |   else if (type->num_fields () == 0) | 
 |     gdb_printf (stream, "record (?) is null; end record"); | 
 |   else | 
 |     { | 
 |       gdb_printf (stream, "record (?) is\n%*scase ? is", level + 4, ""); | 
 |  | 
 |       print_variant_clauses (type, nullptr, type, stream, show, level + 8, flags); | 
 |  | 
 |       gdb_printf (stream, "\n%*send case;\n%*send record", | 
 | 		  level + 4, "", level, ""); | 
 |     } | 
 | } | 
 |  | 
 |  | 
 |  | 
 | /* Print function or procedure type TYPE on STREAM.  Make it a header | 
 |    for function or procedure NAME if NAME is not null.  */ | 
 |  | 
 | static void | 
 | print_func_type (struct type *type, struct ui_file *stream, const char *name, | 
 | 		 const struct type_print_options *flags) | 
 | { | 
 |   int i, len = type->num_fields (); | 
 |  | 
 |   if (type->target_type () != NULL | 
 |       && type->target_type ()->code () == TYPE_CODE_VOID) | 
 |     gdb_printf (stream, "procedure"); | 
 |   else | 
 |     gdb_printf (stream, "function"); | 
 |  | 
 |   if (name != NULL && name[0] != '\0') | 
 |     { | 
 |       gdb_puts (" ", stream); | 
 |       fputs_styled (name, function_name_style.style (), stream); | 
 |     } | 
 |  | 
 |   if (len > 0) | 
 |     { | 
 |       gdb_printf (stream, " ("); | 
 |       for (i = 0; i < len; i += 1) | 
 | 	{ | 
 | 	  if (i > 0) | 
 | 	    { | 
 | 	      gdb_puts ("; ", stream); | 
 | 	      stream->wrap_here (4); | 
 | 	    } | 
 | 	  gdb_printf (stream, "a%d: ", i + 1); | 
 | 	  ada_print_type (type->field (i).type (), "", stream, -1, 0, | 
 | 			  flags); | 
 | 	} | 
 |       gdb_printf (stream, ")"); | 
 |     } | 
 |  | 
 |   if (type->target_type () == NULL) | 
 |     gdb_printf (stream, " return <unknown return type>"); | 
 |   else if (type->target_type ()->code () != TYPE_CODE_VOID) | 
 |     { | 
 |       gdb_printf (stream, " return "); | 
 |       ada_print_type (type->target_type (), "", stream, 0, 0, flags); | 
 |     } | 
 | } | 
 |  | 
 |  | 
 | /* Print a description of a type TYPE0. | 
 |    Output goes to STREAM (via stdio). | 
 |    If VARSTRING is a non-NULL, non-empty string, print as an Ada | 
 |        variable/field declaration. | 
 |    SHOW+1 is the maximum number of levels of internal type structure | 
 |       to show (this applies to record types, enumerated types, and | 
 |       array types). | 
 |    SHOW is the number of levels of internal type structure to show | 
 |       when there is a type name for the SHOWth deepest level (0th is | 
 |       outer level). | 
 |    When SHOW<0, no inner structure is shown. | 
 |    LEVEL indicates level of recursion (for nested definitions).  */ | 
 |  | 
 | void | 
 | ada_print_type (struct type *type0, const char *varstring, | 
 | 		struct ui_file *stream, int show, int level, | 
 | 		const struct type_print_options *flags) | 
 | { | 
 |   if (type0->code () == TYPE_CODE_INTERNAL_FUNCTION) | 
 |     { | 
 |       c_print_type (type0, "", stream, show, level, | 
 | 		    language_ada, flags); | 
 |       return; | 
 |     } | 
 |  | 
 |   struct type *type = ada_check_typedef (ada_get_base_type (type0)); | 
 |   /* If we can decode the original type name, use it.  However, there | 
 |      are cases where the original type is an internally-generated type | 
 |      with a name that can't be decoded (and whose encoded name might | 
 |      not actually bear any relation to the type actually declared in | 
 |      the sources). In that case, try using the name of the base type | 
 |      in its place. | 
 |  | 
 |      Note that we looked at the possibility of always using the name | 
 |      of the base type. This does not always work, unfortunately, as | 
 |      there are situations where it's the base type which has an | 
 |      internally-generated name.  */ | 
 |   const char *type_name = decoded_type_name (type0); | 
 |   if (type_name == nullptr) | 
 |     type_name = decoded_type_name (type); | 
 |   int is_var_decl = (varstring != NULL && varstring[0] != '\0'); | 
 |  | 
 |   if (type == NULL) | 
 |     { | 
 |       if (is_var_decl) | 
 | 	gdb_printf (stream, "%.*s: ", | 
 | 		    ada_name_prefix_len (varstring), varstring); | 
 |       fprintf_styled (stream, metadata_style.style (), "<null type?>"); | 
 |       return; | 
 |     } | 
 |  | 
 |   if (is_var_decl && type->code () != TYPE_CODE_FUNC) | 
 |     gdb_printf (stream, "%.*s: ", | 
 | 		ada_name_prefix_len (varstring), varstring); | 
 |  | 
 |   if (type_name != NULL && show <= 0 && !ada_is_aligner_type (type)) | 
 |     { | 
 |       gdb_printf (stream, "%.*s", | 
 | 		  ada_name_prefix_len (type_name), type_name); | 
 |       return; | 
 |     } | 
 |  | 
 |   if (ada_is_aligner_type (type)) | 
 |     ada_print_type (ada_aligned_type (type), "", stream, show, level, flags); | 
 |   else if (ada_is_constrained_packed_array_type (type) | 
 | 	   && type->code () != TYPE_CODE_PTR) | 
 |     print_array_type (type, stream, show, level, flags); | 
 |   else | 
 |     switch (type->code ()) | 
 |       { | 
 |       default: | 
 | 	gdb_printf (stream, "<"); | 
 | 	c_print_type (type, "", stream, show, level, language_ada, flags); | 
 | 	gdb_printf (stream, ">"); | 
 | 	break; | 
 |       case TYPE_CODE_PTR: | 
 |       case TYPE_CODE_TYPEDEF: | 
 | 	/* An __XVL field is not truly a pointer, so don't print | 
 | 	   "access" in this case.  */ | 
 | 	if (type->code () != TYPE_CODE_PTR | 
 | 	    || (varstring != nullptr | 
 | 		&& strstr (varstring, "___XVL") == nullptr)) | 
 | 	  gdb_printf (stream, "access "); | 
 | 	ada_print_type (type->target_type (), "", stream, show, level, | 
 | 			flags); | 
 | 	break; | 
 |       case TYPE_CODE_REF: | 
 | 	gdb_printf (stream, "<ref> "); | 
 | 	ada_print_type (type->target_type (), "", stream, show, level, | 
 | 			flags); | 
 | 	break; | 
 |       case TYPE_CODE_ARRAY: | 
 | 	print_array_type (type, stream, show, level, flags); | 
 | 	break; | 
 |       case TYPE_CODE_BOOL: | 
 | 	gdb_printf (stream, "(false, true)"); | 
 | 	break; | 
 |       case TYPE_CODE_INT: | 
 | 	{ | 
 | 	  const char *name = ada_type_name (type); | 
 |  | 
 | 	  if (!ada_is_range_type_name (name)) | 
 | 	    fprintf_styled (stream, metadata_style.style (), | 
 | 			    _("<%s-byte integer>"), | 
 | 			    pulongest (type->length ())); | 
 | 	  else | 
 | 	    { | 
 | 	      gdb_printf (stream, "range "); | 
 | 	      print_range_type (type, stream, 1 /* bounds_preferred_p */); | 
 | 	    } | 
 | 	} | 
 | 	break; | 
 |       case TYPE_CODE_RANGE: | 
 | 	if (is_fixed_point_type (type)) | 
 | 	  { | 
 | 	    gdb_printf (stream, "<"); | 
 | 	    print_type_fixed_point (type, stream); | 
 | 	    gdb_printf (stream, ">"); | 
 | 	  } | 
 | 	else if (ada_is_modular_type (type)) | 
 | 	  gdb_printf (stream, "mod %s",  | 
 | 		      int_string (ada_modulus (type), 10, 0, 0, 1)); | 
 | 	else | 
 | 	  { | 
 | 	    gdb_printf (stream, "range "); | 
 | 	    print_range (type, stream, 1 /* bounds_preferred_p */); | 
 | 	  } | 
 | 	break; | 
 |       case TYPE_CODE_FLT: | 
 | 	fprintf_styled (stream, metadata_style.style (), | 
 | 			_("<%s-byte float>"), | 
 | 			pulongest (type->length ())); | 
 | 	break; | 
 |       case TYPE_CODE_ENUM: | 
 | 	if (show < 0) | 
 | 	  gdb_printf (stream, "(...)"); | 
 | 	else | 
 | 	  print_enum_type (type, stream); | 
 | 	break; | 
 |       case TYPE_CODE_STRUCT: | 
 | 	if (ada_is_array_descriptor_type (type)) | 
 | 	  print_array_type (type, stream, show, level, flags); | 
 | 	else | 
 | 	  print_record_type (type, stream, show, level, flags); | 
 | 	break; | 
 |       case TYPE_CODE_UNION: | 
 | 	print_unchecked_union_type (type, stream, show, level, flags); | 
 | 	break; | 
 |       case TYPE_CODE_FUNC: | 
 | 	print_func_type (type, stream, varstring, flags); | 
 | 	break; | 
 |       } | 
 | } | 
 |  | 
 | /* Implement the la_print_typedef language method for Ada.  */ | 
 |  | 
 | void | 
 | ada_print_typedef (struct type *type, struct symbol *new_symbol, | 
 | 		   struct ui_file *stream) | 
 | { | 
 |   type = ada_check_typedef (type); | 
 |   ada_print_type (type, "", stream, 0, 0, &type_print_raw_options); | 
 | } |