Add support to GDB for the Renesas rl78 architecture.
[external/binutils.git] / gdb / ada-typeprint.c
index b69deab..97ab937 100644 (file)
@@ -1,6 +1,6 @@
 /* Support for printing Ada types for GDB, the GNU debugger.
-   Copyright (C) 1986, 1988, 1989, 1991, 1997, 1998, 1999, 2000, 2001, 2002,
-   2003, 2004, 2007, 2008, 2009, 2010 Free Software Foundation, Inc.
+   Copyright (C) 1986, 1988-1989, 1991, 1997-2004, 2007-2012 Free
+   Software Foundation, Inc.
 
    This file is part of GDB.
 
 #include "gdb_string.h"
 #include <errno.h>
 
+static int print_selected_record_field_types (struct type *, struct type *,
+                                             int, int,
+                                             struct ui_file *, int, int);
+   
 static int print_record_field_types (struct type *, struct type *,
                                     struct ui_file *, int, int);
 
 static void print_array_type (struct type *, struct ui_file *, int, int);
 
-static void print_choices (struct type *, int, struct ui_file *,
-                          struct type *);
+static int print_choices (struct type *, int, struct ui_file *,
+                         struct type *);
 
 static void print_range (struct type *, struct ui_file *);
 
@@ -151,6 +155,7 @@ print_range_bound (struct type *type, 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
@@ -335,55 +340,52 @@ print_array_type (struct type *type, struct ui_file *stream, int show,
     }
 
   n_indices = -1;
-  if (show < 0)
-    fprintf_filtered (stream, "...");
-  else
+  if (ada_is_simple_array_type (type))
     {
-      if (ada_is_simple_array_type (type))
-       {
-         struct type *range_desc_type;
-         struct type *arr_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);
+      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; TYPE_CODE (arr_type) == TYPE_CODE_ARRAY;
-                  arr_type = TYPE_TARGET_TYPE (arr_type))
-               {
-                 if (arr_type != type)
-                   fprintf_filtered (stream, ", ");
-                 print_range (TYPE_INDEX_TYPE (arr_type), stream);
-                 if (TYPE_FIELD_BITSIZE (arr_type, 0) > 0)
-                   bitsize = TYPE_FIELD_BITSIZE (arr_type, 0);
-               }
-           }
-         else
+      bitsize = 0;
+      if (range_desc_type == NULL)
+       {
+         for (arr_type = type; TYPE_CODE (arr_type) == TYPE_CODE_ARRAY;
+              arr_type = TYPE_TARGET_TYPE (arr_type))
            {
-             int k;
-             n_indices = TYPE_NFIELDS (range_desc_type);
-             for (k = 0, arr_type = type;
-                  k < n_indices;
-                  k += 1, arr_type = TYPE_TARGET_TYPE (arr_type))
-               {
-                 if (k > 0)
-                   fprintf_filtered (stream, ", ");
-                 print_range_type (TYPE_FIELD_TYPE (range_desc_type, k),
-                                   stream);
-                 if (TYPE_FIELD_BITSIZE (arr_type, 0) > 0)
-                   bitsize = TYPE_FIELD_BITSIZE (arr_type, 0);
-               }
+             if (arr_type != type)
+               fprintf_filtered (stream, ", ");
+             print_range (TYPE_INDEX_TYPE (arr_type), stream);
+             if (TYPE_FIELD_BITSIZE (arr_type, 0) > 0)
+               bitsize = TYPE_FIELD_BITSIZE (arr_type, 0);
            }
        }
       else
        {
-         int i, i0;
-         for (i = i0 = ada_array_arity (type); i > 0; i -= 1)
-           fprintf_filtered (stream, "%s<>", i == i0 ? "" : ", ");
+         int k;
+
+         n_indices = TYPE_NFIELDS (range_desc_type);
+         for (k = 0, arr_type = type;
+              k < n_indices;
+              k += 1, arr_type = TYPE_TARGET_TYPE (arr_type))
+           {
+             if (k > 0)
+               fprintf_filtered (stream, ", ");
+             print_range_type (TYPE_FIELD_TYPE (range_desc_type, k),
+                               stream);
+             if (TYPE_FIELD_BITSIZE (arr_type, 0) > 0)
+               bitsize = TYPE_FIELD_BITSIZE (arr_type, 0);
+           }
        }
     }
+  else
+    {
+      int i, i0;
+
+      for (i = i0 = ada_array_arity (type); i > 0; i -= 1)
+       fprintf_filtered (stream, "%s<>", i == i0 ? "" : ", ");
+    }
 
   fprintf_filtered (stream, ") of ");
   wrap_here ("");
@@ -394,9 +396,13 @@ print_array_type (struct type *type, struct ui_file *stream, int show,
 }
 
 /* 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.  */
+   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_Variant).  */
 
-static void
+static int
 print_choices (struct type *type, int field_num, struct ui_file *stream,
               struct type *val_type)
 {
@@ -420,7 +426,11 @@ print_choices (struct type *type, int field_num, struct ui_file *stream,
       switch (name[p])
        {
        default:
-         return;
+         goto Huh;
+       case '_':
+       case '\0':
+         fprintf_filtered (stream, " =>");
+         return 1;
        case 'S':
        case 'R':
        case 'O':
@@ -435,6 +445,7 @@ print_choices (struct type *type, int field_num, struct ui_file *stream,
        case 'S':
          {
            LONGEST W;
+
            if (!ada_scan_number (name, p + 1, &W, &p))
              goto Huh;
            ada_print_scalar (val_type, W, stream);
@@ -443,6 +454,7 @@ print_choices (struct type *type, int field_num, struct ui_file *stream,
        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;
@@ -459,18 +471,17 @@ print_choices (struct type *type, int field_num, struct ui_file *stream,
     }
 
 Huh:
-  fprintf_filtered (stream, "??");
-
+  fprintf_filtered (stream, "?? =>");
+  return 0;
 }
 
-/* Assuming that field FIELD_NUM of TYPE is a VARIANTS field whose
-   discriminant is contained in OUTER_TYPE, print its variants 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.  */
+/* 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,
@@ -498,11 +509,16 @@ print_variant_clauses (struct type *type, int field_num,
   for (i = 0; i < TYPE_NFIELDS (var_type); i += 1)
     {
       fprintf_filtered (stream, "\n%*swhen ", level + 4, "");
-      print_choices (var_type, i, stream, discr_type);
-      fprintf_filtered (stream, " =>");
-      if (print_record_field_types (TYPE_FIELD_TYPE (var_type, i),
-                                   outer_type, stream, show, level + 4) <= 0)
-       fprintf_filtered (stream, " null;");
+      if (print_choices (var_type, i, stream, discr_type))
+       {
+         if (print_record_field_types (TYPE_FIELD_TYPE (var_type, i),
+                                       outer_type, stream, show, level + 4) 
+             <= 0)
+           fprintf_filtered (stream, " null;");
+       }
+      else
+       print_selected_record_field_types (var_type, outer_type, i, i,
+                                          stream, show, level + 4);
     }
 }
 
@@ -526,28 +542,28 @@ print_variant_part (struct type *type, int field_num, struct type *outer_type,
   fprintf_filtered (stream, "\n%*send case;", level + 4, "");
 }
 
-/* Print a description on STREAM of the fields in record 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.  */
+/* 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_record_field_types (struct type *type, struct type *outer_type,
-                         struct ui_file *stream, int show, int level)
+print_selected_record_field_types (struct type *type, struct type *outer_type,
+                                  int fld0, int fld1,
+                                  struct ui_file *stream, int show, int level)
 {
-  int len, i, flds;
+  int i, flds;
 
   flds = 0;
-  len = TYPE_NFIELDS (type);
 
-  if (len == 0 && TYPE_STUB (type))
+  if (fld0 > fld1 && TYPE_STUB (type))
     return -1;
 
-  for (i = 0; i < len; i += 1)
+  for (i = fld0; i <= fld1; i += 1)
     {
       QUIT;
 
@@ -575,6 +591,19 @@ print_record_field_types (struct type *type, struct type *outer_type,
   return flds;
 }
 
+/* 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)
+{
+  return print_selected_record_field_types (type, outer_type,
+                                           0, TYPE_NFIELDS (type) - 1,
+                                           stream, show, level);
+}
+   
+
 /* 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).  */
@@ -592,8 +621,17 @@ print_record_type (struct type *type0, struct ui_file *stream, int show,
 
   parent_type = ada_parent_type (type);
   if (ada_type_name (parent_type) != NULL)
-    fprintf_filtered (stream, "new %s with record",
-                     decoded_type_name (parent_type));
+    {
+      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);
+      fprintf_filtered (stream, "new %s with record", parent_name);
+    }
   else if (parent_type == NULL && ada_is_tagged_type (type, 0))
     fprintf_filtered (stream, "tagged record");
   else
@@ -659,7 +697,7 @@ print_unchecked_union_type (struct type *type, struct ui_file *stream,
    for function or procedure NAME if NAME is not null.  */
 
 static void
-print_func_type (struct type *type, struct ui_file *stream, char *name)
+print_func_type (struct type *type, struct ui_file *stream, const char *name)
 {
   int i, len = TYPE_NFIELDS (type);
 
@@ -709,8 +747,8 @@ print_func_type (struct type *type, struct ui_file *stream, char *name)
    LEVEL indicates level of recursion (for nested definitions).  */
 
 void
-ada_print_type (struct type *type0, char *varstring, struct ui_file *stream,
-               int show, int level)
+ada_print_type (struct type *type0, const char *varstring,
+               struct ui_file *stream, int show, int level)
 {
   struct type *type = ada_check_typedef (ada_get_base_type (type0));
   char *type_name = decoded_type_name (type0);
@@ -732,7 +770,7 @@ ada_print_type (struct type *type0, char *varstring, struct ui_file *stream,
     fprintf_filtered (stream, "%.*s: ",
                      ada_name_prefix_len (varstring), varstring);
 
-  if (type_name != NULL && show <= 0)
+  if (type_name != NULL && show <= 0 && !ada_is_aligner_type (type))
     {
       fprintf_filtered (stream, "%.*s",
                        ada_name_prefix_len (type_name), type_name);
@@ -741,18 +779,9 @@ ada_print_type (struct type *type0, char *varstring, struct ui_file *stream,
 
   if (ada_is_aligner_type (type))
     ada_print_type (ada_aligned_type (type), "", stream, show, level);
-  else if (ada_is_constrained_packed_array_type (type))
-    {
-      if (TYPE_CODE (type) == TYPE_CODE_PTR)
-        {
-          fprintf_filtered (stream, "access ");
-          print_array_type (TYPE_TARGET_TYPE (type), stream, show, level);
-        }
-      else
-        {
-          print_array_type (type, stream, show, level);
-        }
-    }
+  else if (ada_is_constrained_packed_array_type (type)
+          && TYPE_CODE (type) != TYPE_CODE_PTR)
+    print_array_type (type, stream, show, level);
   else
     switch (TYPE_CODE (type))
       {
@@ -762,6 +791,7 @@ ada_print_type (struct type *type0, char *varstring, struct ui_file *stream,
        fprintf_filtered (stream, ">");
        break;
       case TYPE_CODE_PTR:
+      case TYPE_CODE_TYPEDEF:
        fprintf_filtered (stream, "access ");
        ada_print_type (TYPE_TARGET_TYPE (type), "", stream, show, level);
        break;
@@ -781,6 +811,7 @@ ada_print_type (struct type *type0, char *varstring, struct ui_file *stream,
        else
          {
            char *name = ada_type_name (type);
+
            if (!ada_is_range_type_name (name))
              fprintf_filtered (stream, _("<%d-byte integer>"),
                                TYPE_LENGTH (type));