* doc/gdb.texinfo (appendix "Installing GDB"): changes in configure.
[deliverable/binutils-gdb.git] / gdb / symtab.c
index 5b36c71f1f88229dc42f316e4fdb868b79175b45..2d6cd05d755df162494bc806b238a849a6abcec4 100644 (file)
 /* Symbol table lookup for the GNU debugger, GDB.
-   Copyright (C) 1986, 1987, 1988 Free Software Foundation, Inc.
-
-GDB is distributed in the hope that it will be useful, but WITHOUT ANY
-WARRANTY.  No author or distributor accepts responsibility to anyone
-for the consequences of using it or for whether it serves any
-particular purpose or works at all, unless he says so in writing.
-Refer to the GDB General Public License for full details.
-
-Everyone is granted permission to copy, modify and redistribute GDB,
-but only under the conditions described in the GDB General Public
-License.  A copy of this license is supposed to have been given to you
-along with GDB so you can know your rights and responsibilities.  It
-should be in a file named COPYING.  Among other things, the copyright
-notice and this notice must be preserved on all copies.
-
-In other words, go ahead and share GDB, but don't try to stop
-anyone else from sharing it farther.  Help stamp out software hoarding!
-*/
+   Copyright 1986, 1987, 1988, 1989, 1990, 1991 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 2 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, write to the Free Software
+Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.  */
+
+#include <stdio.h>
 #include "defs.h"
 #include "symtab.h"
 #include "param.h"
+#include "gdbcore.h"
+#include "frame.h"
+#include "target.h"
+#include "value.h"
+#include "symfile.h"
+#include "gdbcmd.h"
+#include "regex.h"
+#include "language.h"
 
-#include <stdio.h>
 #include <obstack.h>
 #include <assert.h>
 
+#include <sys/types.h>
+#include <fcntl.h>
+#include <string.h>
+#include <sys/stat.h>
 
-/* Allocate an obstack to hold objects that should be freed
-   when we load a new symbol table.
-   This includes the symbols made by dbxread
-   and the types that are not permanent.  */
-
-struct obstack obstack1;
-
-struct obstack *symbol_obstack = &obstack1;
-
-/* This obstack will be used for partial_symbol objects.  It can
-   probably actually be the same as the symbol_obstack above, but I'd
-   like to keep them seperate for now.  If I want to later, I'll
-   replace one with the other.  */
-
-struct obstack obstack2;
+extern char *getenv ();
 
-struct obstack *psymbol_obstack = &obstack2;
+extern char *cplus_demangle ();
+extern struct value *value_of_this ();
+extern void break_command ();
+extern void select_source_symtab ();
 
-/* These variables point to the objects
-   representing the predefined C data types.  */
+/* Functions this file defines */
+static int find_line_common ();
+struct partial_symtab *lookup_partial_symtab ();
+static struct partial_symbol *lookup_partial_symbol ();
+static struct partial_symbol *lookup_demangled_partial_symbol ();
+static struct symbol *lookup_demangled_block_symbol ();
 
-struct type *builtin_type_void;
-struct type *builtin_type_char;
-struct type *builtin_type_short;
-struct type *builtin_type_int;
-struct type *builtin_type_long;
-#ifdef LONG_LONG
-struct type *builtin_type_long_long;
-#endif
-struct type *builtin_type_unsigned_char;
-struct type *builtin_type_unsigned_short;
-struct type *builtin_type_unsigned_int;
-struct type *builtin_type_unsigned_long;
-#ifdef LONG_LONG
-struct type *builtin_type_unsigned_long_long;
-#endif
-struct type *builtin_type_float;
-struct type *builtin_type_double;
+/* The single non-language-specific builtin type */
+struct type *builtin_type_error;
 
 /* Block in which the most recently searched-for symbol was found.
    Might be better to make this a parameter to lookup_symbol and 
    value_of_this. */
 struct block *block_found;
 
-/* Functions */
-static int find_line_common ();
-static int lookup_misc_func ();
-struct partial_symtab *lookup_partial_symtab ();
-struct symtab *psymtab_to_symtab ();
-static struct partial_symbol *lookup_partial_symbol ();
+char no_symtab_msg[] = "No symbol table is loaded.  Use the \"file\" command.";
 
-/* Lookup the symbol table of a source file named NAME.  If there
-   isn't any symtab for it, lookup the psymtab and read in the
-   symbtab.  */
+/* Check for a symtab of a specific name; first in symtabs, then in
+   psymtabs.  *If* there is no '/' in the name, a match after a '/'
+   in the symtab filename will also work.  */
 
-struct symtab *
-lookup_symtab (name)
+static struct symtab *
+lookup_symtab_1 (name)
      char *name;
 {
   register struct symtab *s;
   register struct partial_symtab *ps;
-  register char *copy;
+  register char *slash = strchr (name, '/');
+  register int len = strlen (name);
 
   for (s = symtab_list; s; s = s->next)
     if (!strcmp (name, s->filename))
       return s;
 
+  for (ps = partial_symtab_list; ps; ps = ps->next)
+    if (!strcmp (name, ps->filename))
+      {
+       if (ps->readin)
+         fatal ("Internal: readin pst found when no symtab found.");
+       return PSYMTAB_TO_SYMTAB (ps);
+      }
+
+  if (!slash)
+    {
+      for (s = symtab_list; s; s = s->next)
+       {
+         int l = strlen (s->filename);
+
+         if (s->filename[l - len -1] == '/'
+             && !strcmp (s->filename + l - len, name))
+           return s;
+       }
+
+      for (ps = partial_symtab_list; ps; ps = ps->next)
+       {
+         int l = strlen (ps->filename);
+
+         if (ps->filename[l - len - 1] == '/'
+             && !strcmp (ps->filename + l - len, name))
+           {
+             if (ps->readin)
+               fatal ("Internal: readin pst found when no symtab found.");
+             return PSYMTAB_TO_SYMTAB (ps);
+           }
+       }
+    }
+  return 0;
+}
+
+/* Lookup the symbol table of a source file named NAME.  Try a couple
+   of variations if the first lookup doesn't work.  */
+
+struct symtab *
+lookup_symtab (name)
+     char *name;
+{
+  register struct symtab *s;
+  register char *copy;
+
+  s = lookup_symtab_1 (name);
+  if (s) return s;
+
   /* If name not found as specified, see if adding ".c" helps.  */
 
   copy = (char *) alloca (strlen (name) + 3);
   strcpy (copy, name);
   strcat (copy, ".c");
-  for (s = symtab_list; s; s = s->next)
-    if (!strcmp (copy, s->filename))
-      return s;
-
-  ps = lookup_partial_symtab (name);
-  if (ps)
-    {
-      s = psymtab_to_symtab (ps);
-      return s;
-    }
+  s = lookup_symtab_1 (copy);
+  if (s) return s;
 
+  /* We didn't find anything; die.  */
   return 0;
 }
 
-/* Lookup the partial symbol table of a source file named NAME.  */
+/* Lookup the partial symbol table of a source file named NAME.  This
+   only returns true on an exact match (ie. this semantics are
+   different from lookup_symtab.  */
 
 struct partial_symtab *
 lookup_partial_symtab (name)
 char *name;
 {
   register struct partial_symtab *s;
-  register char *copy;
   
   for (s = partial_symtab_list; s; s = s->next)
     if (!strcmp (name, s->filename))
       return s;
   
-  /* If name not found as specified, see if adding ".c" helps.  */
-  
-  copy = (char *) alloca (strlen (name) + 3);
-  strcpy (copy, name);
-  strcat (copy, ".c");
-  for (s = partial_symtab_list; s; s = s->next)
-    if (!strcmp (copy, s->filename))
-      return s;
-  
   return 0;
 }
 \f
+/* Return a typename for a struct/union/enum type
+   without the tag qualifier.  If the type has a NULL name,
+   NULL is returned.  */
+char *
+type_name_no_tag (type)
+     register struct type *type;
+{
+  register char *name = TYPE_NAME (type);
+  char *strchr ();
+  if (name == 0)
+    return 0;
+
+#if 0
+  switch (TYPE_CODE (type))
+    {
+    case TYPE_CODE_STRUCT:
+      return name + 7;
+    case TYPE_CODE_UNION:
+      return name + 6;
+    case TYPE_CODE_ENUM:
+      return name + 5;
+    }
+#endif
+
+  name = strchr (name, ' ');
+  if (name)
+    return name + 1;
+
+  return TYPE_NAME (type);
+}
+
+/* Added by Bryan Boreham, Kewill, Sun Sep 17 18:07:17 1989.
+
+   If this is a stubbed struct (i.e. declared as struct foo *), see if
+   we can find a full definition in some other file. If so, copy this
+   definition, so we can use it in future.  If not, set a flag so we 
+   don't waste too much time in future.  (FIXME, this doesn't seem
+   to be happening...)
+
+   This used to be coded as a macro, but I don't think it is called 
+   often enough to merit such treatment.
+*/
+
+struct complaint stub_noname_complaint =
+  {"stub type has NULL name", 0, 0};
+
+void 
+check_stub_type(type)
+     struct type *type;
+{
+  if (TYPE_FLAGS(type) & TYPE_FLAG_STUB)
+    {
+      char* name= type_name_no_tag (type);
+      struct symbol *sym;
+      if (name == 0)
+       {
+         complain (&stub_noname_complaint, 0);
+         return;
+       }
+      sym = lookup_symbol (name, 0, STRUCT_NAMESPACE, 0, 
+                          (struct symtab **)NULL);
+      if (sym)
+       bcopy (SYMBOL_TYPE(sym), type, sizeof (struct type));
+    }
+}
+
+/* Demangle a GDB method stub type.  */
+char *
+gdb_mangle_typename (type)
+     struct type *type;
+{
+  static struct type *last_type;
+  static char *mangled_typename;
+
+  if (type != last_type)
+    {
+      /* Need a new type prefix.  */
+      char *strchr ();
+      char *newname = type_name_no_tag (type);
+      char buf[20];
+      int len;
+
+      if (mangled_typename)
+       free (mangled_typename);
+
+      len = strlen (newname);
+      sprintf (buf, "__%d", len);
+      mangled_typename = (char *)xmalloc (strlen (buf) + len + 1);
+      strcpy (mangled_typename, buf);
+      strcat (mangled_typename, newname);
+      /* Now we have built "__#newname".  */
+    }
+  return mangled_typename;
+}
+
+/* Lookup a primitive type named NAME. 
+   Return zero if NAME is not a primitive type.*/
+
+struct type *
+lookup_primitive_typename (name)
+     char *name;
+{
+   struct type ** const *p;
+
+   for (p = current_language->la_builtin_type_vector; *p; p++)
+      if(!strcmp((**p)->name, name))
+        return **p;
+   return 0; 
+}
+
 /* Lookup a typedef or primitive type named NAME,
    visible in lexical block BLOCK.
    If NOERR is nonzero, return zero if NAME is not suitably defined.  */
@@ -148,27 +276,18 @@ lookup_typename (name, block, noerr)
      struct block *block;
      int noerr;
 {
-  register struct symbol *sym = lookup_symbol (name, block, VAR_NAMESPACE, 0);
+  register struct symbol *sym =
+    lookup_symbol (name, block, VAR_NAMESPACE, 0, (struct symtab **)NULL);
   if (sym == 0 || SYMBOL_CLASS (sym) != LOC_TYPEDEF)
     {
-      if (!strcmp (name, "int"))
-       return builtin_type_int;
-      if (!strcmp (name, "long"))
-       return builtin_type_long;
-      if (!strcmp (name, "short"))
-       return builtin_type_short;
-      if (!strcmp (name, "char"))
-       return builtin_type_char;
-      if (!strcmp (name, "float"))
-       return builtin_type_float;
-      if (!strcmp (name, "double"))
-       return builtin_type_double;
-      if (!strcmp (name, "void"))
-       return builtin_type_void;
-
-      if (noerr)
+      struct type *tmp;
+      tmp = lookup_primitive_typename (name);
+      if(tmp)
+        return tmp;
+      else if (!tmp && noerr)
        return 0;
-      error ("No type named %s.", name);
+      else
+        error ("No type named %s.", name);
     }
   return SYMBOL_TYPE (sym);
 }
@@ -177,15 +296,11 @@ struct type *
 lookup_unsigned_typename (name)
      char *name;
 {
-  if (!strcmp (name, "int"))
-    return builtin_type_unsigned_int;
-  if (!strcmp (name, "long"))
-    return builtin_type_unsigned_long;
-  if (!strcmp (name, "short"))
-    return builtin_type_unsigned_short;
-  if (!strcmp (name, "char"))
-    return builtin_type_unsigned_char;
-  error ("No type named unsigned %s.", name);
+  char *uns = alloca (strlen(name) + 10);
+
+  strcpy (uns, "unsigned ");
+  strcpy (uns+9, name);
+  return lookup_typename (uns, (struct block *)0, 0);
 }
 
 /* Lookup a structure type named "struct NAME",
@@ -197,7 +312,7 @@ lookup_struct (name, block)
      struct block *block;
 {
   register struct symbol *sym 
-    = lookup_symbol (name, block, STRUCT_NAMESPACE, 0);
+    = lookup_symbol (name, block, STRUCT_NAMESPACE, 0, (struct symtab **)NULL);
 
   if (sym == 0)
     error ("No struct type named %s.", name);
@@ -215,7 +330,7 @@ lookup_union (name, block)
      struct block *block;
 {
   register struct symbol *sym 
-    = lookup_symbol (name, block, STRUCT_NAMESPACE, 0);
+    = lookup_symbol (name, block, STRUCT_NAMESPACE, 0, (struct symtab **)NULL);
 
   if (sym == 0)
     error ("No union type named %s.", name);
@@ -233,7 +348,7 @@ lookup_enum (name, block)
      struct block *block;
 {
   register struct symbol *sym 
-    = lookup_symbol (name, block, STRUCT_NAMESPACE, 0);
+    = lookup_symbol (name, block, STRUCT_NAMESPACE, 0, (struct symtab **)NULL);
   if (sym == 0)
     error ("No enum type named %s.", name);
   if (TYPE_CODE (SYMBOL_TYPE (sym)) != TYPE_CODE_ENUM)
@@ -241,6 +356,59 @@ lookup_enum (name, block)
   return SYMBOL_TYPE (sym);
 }
 
+/* Given a type TYPE, lookup the type of the component of type named
+   NAME.  
+   If NOERR is nonzero, return zero if NAME is not suitably defined.  */
+
+struct type *
+lookup_struct_elt_type (type, name, noerr)
+     struct type *type;
+     char *name;
+     int noerr;
+{
+  int i;
+
+  if (TYPE_CODE (type) != TYPE_CODE_STRUCT
+      && TYPE_CODE (type) != TYPE_CODE_UNION)
+    {
+      target_terminal_ours ();
+      fflush (stdout);
+      fprintf (stderr, "Type ");
+      type_print (type, "", stderr, -1);
+      error (" is not a structure or union type.");
+    }
+
+  check_stub_type (type);
+
+  for (i = TYPE_NFIELDS (type) - 1; i >= TYPE_N_BASECLASSES (type); i--)
+    {
+      char *t_field_name = TYPE_FIELD_NAME (type, i);
+
+      if (t_field_name && !strcmp (t_field_name, name))
+       return TYPE_FIELD_TYPE (type, i);
+    }
+  /* OK, it's not in this class.  Recursively check the baseclasses.  */
+  for (i = TYPE_N_BASECLASSES (type) - 1; i >= 0; i--)
+    {
+      struct type *t = lookup_struct_elt_type (TYPE_BASECLASS (type, i),
+                                              name, 0);
+      if (t != NULL)
+       return t;
+    }
+
+  if (noerr)
+    return NULL;
+  
+  target_terminal_ours ();
+  fflush (stdout);
+  fprintf (stderr, "Type ");
+  type_print (type, "", stderr, -1);
+  fprintf (stderr, " has no component named ");
+  fputs_filtered (name, stderr);
+  error (".");
+  return (struct type *)-1;    /* For lint */
+}
+
 /* Given a type TYPE, return a type of pointers to that type.
    May need to construct such a type if this is the first use.
 
@@ -269,6 +437,8 @@ lookup_pointer_type (type)
   if (TYPE_FLAGS (type) & TYPE_FLAG_PERM)
     TYPE_FLAGS (ptype) |= TYPE_FLAG_PERM;
   /* We assume the machine has only one representation for pointers!  */
+  /* FIXME:  This confuses host<->target data representations, and is a
+     poor assumption besides. */
   TYPE_LENGTH (ptype) = sizeof (char *);
   TYPE_CODE (ptype) = TYPE_CODE_PTR;
   return ptype;
@@ -361,6 +531,97 @@ lookup_member_type (type, domain)
   return mtype;
 }
 
+/* Allocate a stub method whose return type is
+   TYPE.  We will fill in arguments later.  This always
+   returns a fresh type.  If we unify this type with
+   an existing type later, the storage allocated
+   here can be freed.  */
+struct type *
+allocate_stub_method (type)
+     struct type *type;
+{
+  struct type *mtype = (struct type *)xmalloc (sizeof (struct type));
+  bzero (mtype, sizeof (struct type));
+  TYPE_MAIN_VARIANT (mtype) = mtype;
+  TYPE_TARGET_TYPE (mtype) = type;
+  TYPE_FLAGS (mtype) = TYPE_FLAG_STUB;
+  TYPE_CODE (mtype) = TYPE_CODE_METHOD;
+  TYPE_LENGTH (mtype) = 1;
+  return mtype;
+}
+
+/* Lookup a method type belonging to class DOMAIN, returning type TYPE,
+   and taking a list of arguments ARGS.
+   If one is not found, allocate a new one.  */
+
+struct type *
+lookup_method_type (domain, type, args)
+     struct type *domain, *type, **args;
+{
+  register struct type *mtype = TYPE_MAIN_VARIANT (type);
+  struct type *main_type;
+
+  main_type = mtype;
+  while (mtype)
+    {
+      if (TYPE_DOMAIN_TYPE (mtype) == domain)
+       {
+         struct type **t1 = args;
+         struct type **t2 = TYPE_ARG_TYPES (mtype);
+         if (t2)
+           {
+             int i;
+             for (i = 0; t1[i] != 0 && t1[i]->code != TYPE_CODE_VOID; i++)
+               if (t1[i] != t2[i])
+                 break;
+             if (t1[i] == t2[i])
+               return mtype;
+           }
+       }
+      mtype = TYPE_NEXT_VARIANT (mtype);
+    }
+
+  /* This is the first time anyone wanted this member type.  */
+  if (TYPE_FLAGS (type) & TYPE_FLAG_PERM)
+    mtype  = (struct type *) xmalloc (sizeof (struct type));
+  else
+    mtype  = (struct type *) obstack_alloc (symbol_obstack,
+                                           sizeof (struct type));
+
+  bzero (mtype, sizeof (struct type));
+  if (main_type == 0)
+    main_type = mtype;
+  else
+    {
+      TYPE_NEXT_VARIANT (mtype) = TYPE_NEXT_VARIANT (main_type);
+      TYPE_NEXT_VARIANT (main_type) = mtype;
+    }
+  TYPE_MAIN_VARIANT (mtype) = main_type;
+  TYPE_TARGET_TYPE (mtype) = type;
+  TYPE_DOMAIN_TYPE (mtype) = domain;
+  TYPE_ARG_TYPES (mtype) = args;
+  /* New type is permanent if type pointed to is permanent.  */
+  if (TYPE_FLAGS (type) & TYPE_FLAG_PERM)
+    TYPE_FLAGS (mtype) |= TYPE_FLAG_PERM;
+
+  /* In practice, this is never used.  */
+  TYPE_LENGTH (mtype) = 1;
+  TYPE_CODE (mtype) = TYPE_CODE_METHOD;
+
+#if 0
+  /* Now splice in the new member pointer type.  */
+  if (main_type)
+    {
+      /* This type was not "smashed".  */
+      TYPE_CHAIN (mtype) = TYPE_CHAIN (main_type);
+      TYPE_CHAIN (main_type) = mtype;
+    }
+#endif
+
+  return mtype;
+}
+
+#if 0
 /* Given a type TYPE, return a type which has offset OFFSET,
    via_virtual VIA_VIRTUAL, and via_public VIA_PUBLIC.
    May need to construct such a type if none exists.  */
@@ -375,7 +636,7 @@ lookup_basetype_type (type, offset, via_virtual, via_public)
 
   if (offset != 0)
     {
-      printf ("type offset non-zero in lookup_basetype_type");
+      printf ("Internal error: type offset non-zero in lookup_basetype_type");
       offset = 0;
     }
 
@@ -422,6 +683,7 @@ lookup_basetype_type (type, offset, via_virtual, via_public)
 
   return btype;
 }
+#endif
 
 /* Given a type TYPE, return a type of functions that return that type.
    May need to construct such a type if this is the first use.  */
@@ -452,29 +714,59 @@ lookup_function_type (type)
   return ptype;
 }
 \f
-/* Smash TYPE to be a type of pointers to TO_TYPE.
-   If TO_TYPE is not permanent and has no pointer-type yet,
-   record TYPE as its pointer-type.  */
+/* Create an array type.  Elements will be of type TYPE, and there will
+   be NUM of them.
 
-void
-smash_to_pointer_type (type, to_type)
-     struct type *type, *to_type;
-{
-  bzero (type, sizeof (struct type));
-  TYPE_TARGET_TYPE (type) = to_type;
-  /* We assume the machine has only one representation for pointers!  */
-  TYPE_LENGTH (type) = sizeof (char *);
-  TYPE_CODE (type) = TYPE_CODE_PTR;
+   Eventually this should be extended to take two more arguments which
+   specify the bounds of the array and the type of the index.
+   It should also be changed to be a "lookup" function, with the
+   appropriate data structures added to the type field.
+   Then read array type should call here.  */
 
-  TYPE_MAIN_VARIANT (type) = type;
-
-  if (TYPE_POINTER_TYPE (to_type) == 0
-      && !(TYPE_FLAGS (type) & TYPE_FLAG_PERM))
-    {
-      TYPE_POINTER_TYPE (to_type) = type;
-    }
+struct type *
+create_array_type (element_type, number)
+     struct type *element_type;
+     int number;
+{
+  struct type *result_type = (struct type *)
+    obstack_alloc (symbol_obstack, sizeof (struct type));
+  struct type *range_type;
+
+  bzero (result_type, sizeof (struct type));
+
+  TYPE_CODE (result_type) = TYPE_CODE_ARRAY;
+  TYPE_TARGET_TYPE (result_type) = element_type;
+  TYPE_LENGTH (result_type) = number * TYPE_LENGTH (element_type);
+  TYPE_NFIELDS (result_type) = 1;
+  TYPE_FIELDS (result_type) =
+    (struct field *) obstack_alloc (symbol_obstack, sizeof (struct field));
+
+  {
+    /* Create range type.  */
+    range_type = (struct type *) obstack_alloc (symbol_obstack,
+                                               sizeof (struct type));
+    TYPE_CODE (range_type) = TYPE_CODE_RANGE;
+    TYPE_TARGET_TYPE (range_type) = builtin_type_int;  /* FIXME */
+
+    /* This should never be needed.  */
+    TYPE_LENGTH (range_type) = sizeof (int);
+
+    TYPE_NFIELDS (range_type) = 2;
+    TYPE_FIELDS (range_type) =
+      (struct field *) obstack_alloc (symbol_obstack,
+                                     2 * sizeof (struct field));
+    TYPE_FIELD_BITPOS (range_type, 0) = 0; /* FIXME */
+    TYPE_FIELD_BITPOS (range_type, 1) = number-1; /* FIXME */
+    TYPE_FIELD_TYPE (range_type, 0) = builtin_type_int; /* FIXME */
+    TYPE_FIELD_TYPE (range_type, 1) = builtin_type_int; /* FIXME */
+  }
+  TYPE_FIELD_TYPE(result_type,0)=range_type;
+  TYPE_VPTR_FIELDNO (result_type) = -1;
+
+  return result_type;
 }
 
+\f
 /* Smash TYPE to be a type of members of DOMAIN with type TO_TYPE.  */
 
 void
@@ -492,55 +784,80 @@ smash_to_member_type (type, domain, to_type)
   TYPE_MAIN_VARIANT (type) = lookup_member_type (domain, to_type);
 }
 
-/* Smash TYPE to be a type of reference to TO_TYPE.
-   If TO_TYPE is not permanent and has no pointer-type yet,
-   record TYPE as its pointer-type.  */
+/* Smash TYPE to be a type of method of DOMAIN with type TO_TYPE.  */
 
 void
-smash_to_reference_type (type, to_type)
-     struct type *type, *to_type;
+smash_to_method_type (type, domain, to_type, args)
+     struct type *type, *domain, *to_type, **args;
 {
   bzero (type, sizeof (struct type));
   TYPE_TARGET_TYPE (type) = to_type;
-  /* We assume the machine has only one representation for pointers!  */
-  TYPE_LENGTH (type) = sizeof (char *);
-  TYPE_CODE (type) = TYPE_CODE_REF;
+  TYPE_DOMAIN_TYPE (type) = domain;
+  TYPE_ARG_TYPES (type) = args;
 
-  TYPE_MAIN_VARIANT (type) = type;
+  /* In practice, this is never needed.  */
+  TYPE_LENGTH (type) = 1;
+  TYPE_CODE (type) = TYPE_CODE_METHOD;
 
-  if (TYPE_REFERENCE_TYPE (to_type) == 0
-      && !(TYPE_FLAGS (type) & TYPE_FLAG_PERM))
-    {
-      TYPE_REFERENCE_TYPE (to_type) = type;
-    }
+  TYPE_MAIN_VARIANT (type) = lookup_method_type (domain, to_type, args);
 }
+\f
+/* Find which partial symtab on the partial_symtab_list contains
+   PC.  Return 0 if none.  */
 
-/* Smash TYPE to be a type of functions returning TO_TYPE.
-   If TO_TYPE is not permanent and has no function-type yet,
-   record TYPE as its function-type.  */
+struct partial_symtab *
+find_pc_psymtab (pc)
+     register CORE_ADDR pc;
+{
+  register struct partial_symtab *ps;
 
-void
-smash_to_function_type (type, to_type)
-     struct type *type, *to_type;
+  for (ps = partial_symtab_list; ps; ps = ps->next)
+    if (pc >= ps->textlow && pc < ps->texthigh)
+      return ps;
+
+  return 0;
+}
+
+/* Find which partial symbol within a psymtab contains PC.  Return 0
+   if none.  Check all psymtabs if PSYMTAB is 0.  */
+struct partial_symbol *
+find_pc_psymbol (psymtab, pc)
+     struct partial_symtab *psymtab;
+     CORE_ADDR pc;
 {
-  bzero (type, sizeof (struct type));
-  TYPE_TARGET_TYPE (type) = to_type;
-  TYPE_LENGTH (type) = 1;
-  TYPE_CODE (type) = TYPE_CODE_FUNC;
-  TYPE_NFIELDS (type) = 0;
+  struct partial_symbol *best, *p;
+  CORE_ADDR best_pc;
+  
+  if (!psymtab)
+    psymtab = find_pc_psymtab (pc);
+  if (!psymtab)
+    return 0;
 
-  if (TYPE_FUNCTION_TYPE (to_type) == 0
-      && !(TYPE_FLAGS (type) & TYPE_FLAG_PERM))
-    {
-      TYPE_FUNCTION_TYPE (to_type) = type;
-    }
+  best_pc = psymtab->textlow - 1;
+
+  for (p = static_psymbols.list + psymtab->statics_offset;
+       (p - (static_psymbols.list + psymtab->statics_offset)
+       < psymtab->n_static_syms);
+       p++)
+    if (SYMBOL_NAMESPACE (p) == VAR_NAMESPACE
+       && SYMBOL_CLASS (p) == LOC_BLOCK
+       && pc >= SYMBOL_VALUE_ADDRESS (p)
+       && SYMBOL_VALUE_ADDRESS (p) > best_pc)
+      {
+       best_pc = SYMBOL_VALUE_ADDRESS (p);
+       best = p;
+      }
+  if (best_pc == psymtab->textlow - 1)
+    return 0;
+  return best;
 }
-\f
-static struct symbol *lookup_block_symbol ();
 
+\f
 /* Find the definition for a specified symbol name NAME
    in namespace NAMESPACE, visible from lexical block BLOCK.
-   Returns the struct symbol pointer, or zero if no symbol is found. 
+   Returns the struct symbol pointer, or zero if no symbol is found.
+   If SYMTAB is non-NULL, store the symbol table in which the
+   symbol was found there, or NULL if not found.
    C++: if IS_A_FIELD_OF_THIS is nonzero on entry, check to see if
    NAME is a field of the current implied argument `this'.  If so set
    *IS_A_FIELD_OF_THIS to 1, otherwise set it to zero. 
@@ -548,13 +865,13 @@ static struct symbol *lookup_block_symbol ();
    a field of `this', value_of_this sets BLOCK_FOUND to the proper value.) */
 
 struct symbol *
-lookup_symbol (name, block, namespace, is_a_field_of_this)
+lookup_symbol (name, block, namespace, is_a_field_of_this, symtab)
      char *name;
      register struct block *block;
      enum namespace namespace;
      int *is_a_field_of_this;
+     struct symtab **symtab;
 {
-  register int i, n;
   register struct symbol *sym;
   register struct symtab *s;
   register struct partial_symtab *ps;
@@ -568,21 +885,70 @@ lookup_symbol (name, block, namespace, is_a_field_of_this)
       if (sym) 
        {
          block_found = block;
+         if (symtab != NULL)
+           {
+             /* Search the list of symtabs for one which contains the
+                address of the start of this block.  */
+             struct block *b;
+             for (s = symtab_list; s; s = s->next)
+               {
+                 bv = BLOCKVECTOR (s);
+                 b = BLOCKVECTOR_BLOCK (bv, GLOBAL_BLOCK);
+                 if (BLOCK_START (b) <= BLOCK_START (block)
+                     && BLOCK_END (b) > BLOCK_START (block))
+                   break;
+               }
+             *symtab = s;
+           }
+
          return sym;
        }
       block = BLOCK_SUPERBLOCK (block);
     }
 
+  /* But that doesn't do any demangling for the STATIC_BLOCK.
+     I'm not sure whether demangling is needed in the case of
+     nested function in inner blocks; if so this needs to be changed.
+     
+     Don't need to mess with the psymtabs; if we have a block,
+     that file is read in.  If we don't, then we deal later with
+     all the psymtab stuff that needs checking.  */
+  if (namespace == VAR_NAMESPACE && block != NULL)
+    {
+      struct block *b;
+      /* Find the right symtab.  */
+      for (s = symtab_list; s; s = s->next)
+       {
+         bv = BLOCKVECTOR (s);
+         b = BLOCKVECTOR_BLOCK (bv, STATIC_BLOCK);
+         if (BLOCK_START (b) <= BLOCK_START (block)
+             && BLOCK_END (b) > BLOCK_START (block))
+           {
+             sym = lookup_demangled_block_symbol (b, name);
+             if (sym)
+               {
+                 block_found = b;
+                 if (symtab != NULL)
+                   *symtab = s;
+                 return sym;
+               }
+           }
+       }
+    }
+
+
   /* C++: If requested to do so by the caller, 
      check to see if NAME is a field of `this'. */
   if (is_a_field_of_this)
     {
-      int v = (int) value_of_this (0);
+      struct value *v = value_of_this (0);
       
       *is_a_field_of_this = 0;
       if (v && check_field (v, name))
        {
-         *is_a_field_of_this = 1;          
+         *is_a_field_of_this = 1;
+         if (symtab != NULL)
+           *symtab = NULL;
          return 0;
        }
     }
@@ -593,24 +959,89 @@ lookup_symbol (name, block, namespace, is_a_field_of_this)
   for (s = symtab_list; s; s = s->next)
     {
       bv = BLOCKVECTOR (s);
-      block = BLOCKVECTOR_BLOCK (bv, 0);
+      block = BLOCKVECTOR_BLOCK (bv, GLOBAL_BLOCK);
       sym = lookup_block_symbol (block, name, namespace);
       if (sym) 
        {
          block_found = block;
+         if (symtab != NULL)
+           *symtab = s;
          return sym;
        }
     }
 
+  /* Check for the possibility of the symbol being a global function
+     that is stored on the misc function vector.  Eventually, all
+     global symbols might be resolved in this way.  */
+  
+  if (namespace == VAR_NAMESPACE)
+    {
+      int ind = lookup_misc_func (name);
+
+      /* Look for a mangled C++ name for NAME. */
+      if (ind == -1)
+       {
+         int name_len = strlen (name);
+
+         for (ind = misc_function_count; --ind >= 0; )
+             /* Assume orginal name is prefix of mangled name. */
+             if (!strncmp (misc_function_vector[ind].name, name, name_len))
+               {
+                 char *demangled =
+                     cplus_demangle(misc_function_vector[ind].name, -1);
+                 if (demangled != NULL)
+                   {
+                     int cond = strcmp (demangled, name);
+                     free (demangled);
+                     if (!cond)
+                         break;
+                   }
+               }
+         /* Loop terminates on no match with ind == -1. */
+        }
+
+      if (ind != -1)
+       {
+         s = find_pc_symtab (misc_function_vector[ind].address);
+         /* If S is zero, there are no debug symbols for this file.
+            Skip this stuff and check for matching static symbols below. */
+         if (s)
+           {
+             bv = BLOCKVECTOR (s);
+             block = BLOCKVECTOR_BLOCK (bv, GLOBAL_BLOCK);
+             sym = lookup_block_symbol (block, misc_function_vector[ind].name,
+                                        namespace);
+             /* sym == 0 if symbol was found in the misc_function_vector
+                but not in the symtab.
+                Return 0 to use the misc_function definition of "foo_".
+
+                This happens for Fortran  "foo_" symbols,
+                which are "foo" in the symtab.
+
+                This can also happen if "asm" is used to make a
+                regular symbol but not a debugging symbol, e.g.
+                asm(".globl _main");
+                asm("_main:");
+                */
+
+             if (symtab != NULL)
+               *symtab = s;
+             return sym;
+           }
+       }
+    }
+      
   for (ps = partial_symtab_list; ps; ps = ps->next)
-    if (lookup_partial_symbol (ps, name, 1, namespace))
+    if (!ps->readin && lookup_partial_symbol (ps, name, 1, namespace))
       {
-       s = psymtab_to_symtab(ps);
+       s = PSYMTAB_TO_SYMTAB(ps);
        bv = BLOCKVECTOR (s);
-       block = BLOCKVECTOR_BLOCK (bv, 0);
+       block = BLOCKVECTOR_BLOCK (bv, GLOBAL_BLOCK);
        sym = lookup_block_symbol (block, name, namespace);
        if (!sym)
          fatal ("Internal: global symbol found in psymtab but not in symtab");
+       if (symtab != NULL)
+         *symtab = s;
        return sym;
       }
 
@@ -618,31 +1049,140 @@ lookup_symbol (name, block, namespace, is_a_field_of_this)
      Not strictly correct, but more useful than an error.
      Do the symtabs first, then check the psymtabs */
 
-  for (s = symtab_list; s; s = s->next)
+  for (s = symtab_list; s; s = s->next)
+    {
+      bv = BLOCKVECTOR (s);
+      block = BLOCKVECTOR_BLOCK (bv, STATIC_BLOCK);
+      sym = lookup_block_symbol (block, name, namespace);
+      if (sym) 
+       {
+         block_found = block;
+         if (symtab != NULL)
+           *symtab = s;
+         return sym;
+       }
+    }
+
+  for (ps = partial_symtab_list; ps; ps = ps->next)
+    if (!ps->readin && lookup_partial_symbol (ps, name, 0, namespace))
+      {
+       s = PSYMTAB_TO_SYMTAB(ps);
+       bv = BLOCKVECTOR (s);
+       block = BLOCKVECTOR_BLOCK (bv, STATIC_BLOCK);
+       sym = lookup_block_symbol (block, name, namespace);
+       if (!sym)
+         fatal ("Internal: static symbol found in psymtab but not in symtab");
+       if (symtab != NULL)
+         *symtab = s;
+       return sym;
+      }
+
+  /* Now search all per-file blocks for static mangled symbols.
+     Do the symtabs first, then check the psymtabs.  */
+
+  if (namespace ==  VAR_NAMESPACE)
+    {
+      for (s = symtab_list; s; s = s->next)
+       {
+         bv = BLOCKVECTOR (s);
+         block = BLOCKVECTOR_BLOCK (bv, STATIC_BLOCK);
+         sym = lookup_demangled_block_symbol (block, name);
+         if (sym) 
+           {
+             block_found = block;
+             if (symtab != NULL)
+               *symtab = s;
+             return sym;
+           }
+       }
+
+      for (ps = partial_symtab_list; ps; ps = ps->next)
+       if (!ps->readin && lookup_demangled_partial_symbol (ps, name))
+         {
+           s = PSYMTAB_TO_SYMTAB(ps);
+           bv = BLOCKVECTOR (s);
+           block = BLOCKVECTOR_BLOCK (bv, STATIC_BLOCK);
+           sym = lookup_demangled_block_symbol (block, name);
+           if (!sym)
+             fatal ("Internal: static symbol found in psymtab but not in symtab");
+           if (symtab != NULL)
+             *symtab = s;
+           return sym;
+         }
+    }
+
+  if (symtab != NULL)
+    *symtab = NULL;
+  return 0;
+}
+
+/* Look for a static demangled symbol in block BLOCK.  */
+
+static struct symbol *
+lookup_demangled_block_symbol (block, name)
+     register struct block *block;
+     char *name;
+{
+  register int bot, top, inc;
+  register struct symbol *sym;
+
+  bot = 0;
+  top = BLOCK_NSYMS (block);
+  inc = name[0];
+
+  while (bot < top)
+    {
+      sym = BLOCK_SYM (block, bot);
+      if (SYMBOL_NAME (sym)[0] == inc
+         && SYMBOL_NAMESPACE (sym) == VAR_NAMESPACE)
+       {
+         char *demangled = cplus_demangle(SYMBOL_NAME (sym), -1);
+         if (demangled != NULL)
+           {
+             int cond = strcmp (demangled, name);
+             free (demangled);
+             if (!cond)
+               return sym;
+           }
+       }
+      bot++;
+    }
+
+  return 0;
+}
+
+/* Look, in partial_symtab PST, for static mangled symbol NAME. */
+
+static struct partial_symbol *
+lookup_demangled_partial_symbol (pst, name)
+     struct partial_symtab *pst;
+     char *name;
+{
+  struct partial_symbol *start, *psym;
+  int length = pst->n_static_syms;
+  register int inc = name[0];
+
+  if (!length)
+    return (struct partial_symbol *) 0;
+  
+  start = static_psymbols.list + pst->statics_offset;
+  for (psym = start; psym < start + length; psym++)
     {
-      bv = BLOCKVECTOR (s);
-      block = BLOCKVECTOR_BLOCK (bv, 1);
-      sym = lookup_block_symbol (block, name, namespace);
-      if (sym) 
+      if (SYMBOL_NAME (psym)[0] == inc
+         && SYMBOL_NAMESPACE (psym) == VAR_NAMESPACE)
        {
-         block_found = block;
-         return sym;
+         char *demangled = cplus_demangle(SYMBOL_NAME (psym), -1);
+         if (demangled != NULL)
+           {
+             int cond = strcmp (demangled, name);
+             free (demangled);
+             if (!cond)
+               return psym;
+           }
        }
     }
 
-  for (ps = partial_symtab_list; ps; ps = ps->next)
-    if (lookup_partial_symbol (ps, name, 0, namespace))
-      {
-       s = psymtab_to_symtab(ps);
-       bv = BLOCKVECTOR (s);
-       block = BLOCKVECTOR_BLOCK (bv, 1);
-       sym = lookup_block_symbol (block, name, namespace);
-       if (!sym)
-         fatal ("Internal: static symbol found in psymtab but not in symtab");
-       return sym;
-      }
-
-  return 0;
+  return (struct partial_symbol *) 0;
 }
 
 /* Look, in partial_symtab PST, for symbol NAME.  Check the global
@@ -658,19 +1198,19 @@ lookup_partial_symbol (pst, name, global, namespace)
   struct partial_symbol *start, *psym;
   int length = (global ? pst->n_global_syms : pst->n_static_syms);
 
-  start = (global ?
-          global_psymbols + pst->globals_offset :
-          static_psymbols + pst->statics_offset  );
-
   if (!length)
     return (struct partial_symbol *) 0;
   
+  start = (global ?
+          global_psymbols.list + pst->globals_offset :
+          static_psymbols.list + pst->statics_offset  );
+
   if (global)                  /* This means we can use a binary */
                                /* search.  */
     {
       struct partial_symbol *top, *bottom, *center;
 
-      /* Binary search.  This search is guarranteed to end with center
+      /* Binary search.  This search is guaranteed to end with center
          pointing at the earliest partial symbol with the correct
         name.  At that point *all* partial symbols with that name
         will be checked against the correct namespace. */
@@ -710,7 +1250,7 @@ lookup_partial_symbol (pst, name, global, namespace)
 
 /* Look for a symbol in block BLOCK.  */
 
-static struct symbol *
+struct symbol *
 lookup_block_symbol (block, name, namespace)
      register struct block *block;
      char *name;
@@ -790,8 +1330,10 @@ lookup_block_symbol (block, name, namespace)
          && !strcmp (SYMBOL_NAME (sym), name)
          && SYMBOL_NAMESPACE (sym) == namespace)
        {
-         if (SYMBOL_CLASS (sym) == LOC_ARG ||
-             SYMBOL_CLASS (sym) == LOC_REGPARM)
+         if (SYMBOL_CLASS (sym) == LOC_ARG
+             || SYMBOL_CLASS (sym) == LOC_LOCAL_ARG
+             || SYMBOL_CLASS (sym) == LOC_REF_ARG
+             || SYMBOL_CLASS (sym) == LOC_REGPARM)
            parameter_sym = sym;
          else
            return sym;
@@ -816,7 +1358,7 @@ block_function (bl)
 
 /* Subroutine of find_pc_line */
 
-static struct symtab *
+struct symtab *
 find_pc_symtab (pc)
      register CORE_ADDR pc;
 {
@@ -830,19 +1372,22 @@ find_pc_symtab (pc)
   for (s = symtab_list; s; s = s->next)
     {
       bv = BLOCKVECTOR (s);
-      b = BLOCKVECTOR_BLOCK (bv, 0);
+      b = BLOCKVECTOR_BLOCK (bv, GLOBAL_BLOCK);
       if (BLOCK_START (b) <= pc
          && BLOCK_END (b) > pc)
        break;
     }
 
   if (!s)
-    for (ps = partial_symtab_list; ps; ps = ps->next)
-      if (pc >= ps->textlow && pc < ps->texthigh)
-       {
-         s = psymtab_to_symtab (ps);
-         break;
-       }
+    {
+      ps = find_pc_psymtab (pc);
+      if (ps && ps->readin)
+       printf_filtered (
+         "(Internal error: pc in read in psymtab, but not in symtab.)\n");
+
+      if (ps)
+       s = PSYMTAB_TO_SYMTAB (ps);
+    }
 
   return s;
 }
@@ -865,7 +1410,7 @@ find_pc_line (pc, notcurrent)
   register int len;
   register int i;
   register struct linetable_entry *item;
-  struct symtab_and_line value;
+  struct symtab_and_line val;
   struct blockvector *bv;
 
   /* Info on best line seen so far, and where it starts, and its file.  */
@@ -905,11 +1450,11 @@ find_pc_line (pc, notcurrent)
   s = find_pc_symtab (pc);
   if (s == 0)
     {
-      value.symtab = 0;
-      value.line = 0;
-      value.pc = pc;
-      value.end = 0;
-      return value;
+      val.symtab = 0;
+      val.line = 0;
+      val.pc = pc;
+      val.end = 0;
+      return val;
     }
 
   bv = BLOCKVECTOR (s);
@@ -968,21 +1513,21 @@ find_pc_line (pc, notcurrent)
     }
   if (best_symtab == 0)
     {
-      value.symtab = alt_symtab;
-      value.line = alt_line - 1;
-      value.pc = BLOCK_END (BLOCKVECTOR_BLOCK (bv, 0));
-      value.end = alt_pc;
+      val.symtab = alt_symtab;
+      val.line = alt_line - 1;
+      val.pc = BLOCK_END (BLOCKVECTOR_BLOCK (bv, GLOBAL_BLOCK));
+      val.end = alt_pc;
     }
   else
     {
-      value.symtab = best_symtab;
-      value.line = best_line;
-      value.pc = best_pc;
-      value.end = (best_end ? best_end
+      val.symtab = best_symtab;
+      val.line = best_line;
+      val.pc = best_pc;
+      val.end = (best_end ? best_end
                   : (alt_pc ? alt_pc
-                     : BLOCK_END (BLOCKVECTOR_BLOCK (bv, 0))));
+                     : BLOCK_END (BLOCKVECTOR_BLOCK (bv, GLOBAL_BLOCK))));
     }
-  return value;
+  return val;
 }
 \f
 /* Find the PC value for a given source file and line number.
@@ -995,14 +1540,14 @@ find_line_pc (symtab, line)
      int line;
 {
   register struct linetable *l;
-  register int index;
+  register int ind;
   int dummy;
 
   if (symtab == 0)
     return 0;
   l = LINETABLE (symtab);
-  index = find_line_common(l, line, &dummy);
-  return index ? l->item[index].pc : 0;
+  ind = find_line_common(l, line, &dummy);
+  return (ind >= 0) ? l->item[ind].pc : 0;
 }
 
 /* Find the range of pc values in a line.
@@ -1018,28 +1563,27 @@ find_line_pc_range (symtab, thisline, startptr, endptr)
      CORE_ADDR *startptr, *endptr;
 {
   register struct linetable *l;
-  register int index;
+  register int ind;
   int exact_match;             /* did we get an exact linenumber match */
-  register CORE_ADDR prev_pc;
-  CORE_ADDR last_pc;
 
   if (symtab == 0)
     return 0;
 
   l = LINETABLE (symtab);
-  index = find_line_common (l, thisline, &exact_match);
-  if (index)
+  ind = find_line_common (l, thisline, &exact_match);
+  if (ind >= 0)
     {
-      *startptr = l->item[index].pc;
+      *startptr = l->item[ind].pc;
       /* If we have not seen an entry for the specified line,
         assume that means the specified line has zero bytes.  */
-      if (!exact_match || index == l->nitems-1)
+      if (!exact_match || ind == l->nitems-1)
        *endptr = *startptr;
       else
        /* Perhaps the following entry is for the following line.
           It's worth a try.  */
-       if (l->item[index+1].line == thisline + 1)
-         *endptr = l->item[index+1].pc;
+       if (ind+1 < l->nitems
+        && l->item[ind+1].line == thisline + 1)
+         *endptr = l->item[ind+1].pc;
        else
          *endptr = find_line_pc (symtab, thisline+1);
       return 1;
@@ -1050,7 +1594,7 @@ find_line_pc_range (symtab, thisline, startptr, endptr)
 
 /* Given a line table and a line number, return the index into the line
    table for the pc of the nearest line whose number is >= the specified one.
-   Return 0 if none is found.  The value is never zero is it is an index.
+   Return -1 if none is found.  The value is >= 0 if it is an index.
 
    Set *EXACT_MATCH nonzero if the value returned is an exact match.  */
 
@@ -1067,13 +1611,11 @@ find_line_common (l, lineno, exact_match)
      or 0 if none has been seen so far.
      BEST_INDEX identifies the item for it.  */
 
-  int best_index = 0;
+  int best_index = -1;
   int best = 0;
 
-  int nextline = -1;
-
   if (lineno <= 0)
-    return 0;
+    return -1;
 
   len = l->nitems;
   for (i = 0; i < len; i++)
@@ -1111,6 +1653,168 @@ find_pc_line_pc_range (pc, startptr, endptr)
   return sal.symtab != 0;
 }
 \f
+/* If P is of the form "operator[ \t]+..." where `...' is
+   some legitimate operator text, return a pointer to the
+   beginning of the substring of the operator text.
+   Otherwise, return "".  */
+static char *
+operator_chars (p, end)
+     char *p;
+     char **end;
+{
+  *end = "";
+  if (strncmp (p, "operator", 8))
+    return *end;
+  p += 8;
+
+  /* Don't get faked out by `operator' being part of a longer
+     identifier.  */
+  if ((*p >= 'A' && *p <= 'Z') || (*p >= 'a' && *p <= 'z')
+      || *p == '_' || *p == '$' || *p == '\0')
+    return *end;
+
+  /* Allow some whitespace between `operator' and the operator symbol.  */
+  while (*p == ' ' || *p == '\t')
+    p++;
+
+  switch (*p)
+    {
+    case '!':
+    case '=':
+    case '*':
+    case '/':
+    case '%':
+    case '^':
+      if (p[1] == '=')
+       *end = p+2;
+      else
+       *end = p+1;
+      return p;
+    case '<':
+    case '>':
+    case '+':
+    case '-':
+    case '&':
+    case '|':
+      if (p[1] == '=' || p[1] == p[0])
+       *end = p+2;
+      else
+       *end = p+1;
+      return p;
+    case '~':
+    case ',':
+      *end = p+1;
+      return p;
+    case '(':
+      if (p[1] != ')')
+       error ("`operator ()' must be specified without whitespace in `()'");
+      *end = p+2;
+      return p;
+    case '?':
+      if (p[1] != ':')
+       error ("`operator ?:' must be specified without whitespace in `?:'");
+      *end = p+2;
+      return p;
+    case '[':
+      if (p[1] != ']')
+       error ("`operator []' must be specified without whitespace in `[]'");
+      *end = p+2;
+      return p;
+    default:
+      error ("`operator %s' not supported", p);
+      break;
+    }
+  *end = "";
+  return *end;
+}
+
+/* Recursive helper function for decode_line_1.
+ * Look for methods named NAME in type T.
+ * Return number of matches.
+ * Put matches in PHYSNAMES and SYM_ARR (which better be big enough!).
+ * These allocations seem to define "big enough":
+ * sym_arr = (struct symbol **) alloca(TYPE_NFN_FIELDS_TOTAL (t) * sizeof(struct symbol*));
+ * physnames = (char **) alloca (TYPE_NFN_FIELDS_TOTAL (t) * sizeof(char*));
+ */
+
+int
+find_methods(t, name, physnames, sym_arr)
+     struct type *t;
+     char *name;
+     char **physnames;
+     struct symbol **sym_arr;
+{
+  int i1 = 0;
+  int ibase;
+  struct symbol *sym_class;
+  char *class_name = type_name_no_tag (t);
+  /* Ignore this class if it doesn't have a name.
+     This prevents core dumps, but is just a workaround
+     because we might not find the function in
+     certain cases, such as
+     struct D {virtual int f();}
+     struct C : D {virtual int g();}
+     (in this case g++ 1.35.1- does not put out a name
+     for D as such, it defines type 19 (for example) in
+     the same stab as C, and then does a
+     .stabs "D:T19" and a .stabs "D:t19".
+     Thus
+     "break C::f" should not be looking for field f in
+     the class named D, 
+     but just for the field f in the baseclasses of C
+     (no matter what their names).
+     
+     However, I don't know how to replace the code below
+     that depends on knowing the name of D.  */
+  if (class_name
+      && (sym_class = lookup_symbol (class_name,
+                                    (struct block *)NULL,
+                                    STRUCT_NAMESPACE,
+                                    (int *)NULL,
+                                    (struct symtab **)NULL)))
+    {
+      int method_counter;
+      t = SYMBOL_TYPE (sym_class);
+      for (method_counter = TYPE_NFN_FIELDS (t) - 1;
+          method_counter >= 0;
+          --method_counter)
+       {
+         int field_counter;
+         struct fn_field *f = TYPE_FN_FIELDLIST1 (t, method_counter);
+
+         char *method_name = TYPE_FN_FIELDLIST_NAME (t, method_counter);
+         if (!strcmp (name, method_name))
+           /* Find all the fields with that name.  */
+           for (field_counter = TYPE_FN_FIELDLIST_LENGTH (t, method_counter) - 1;
+                field_counter >= 0;
+                --field_counter)
+             {
+               char *phys_name;
+               if (TYPE_FLAGS (TYPE_FN_FIELD_TYPE (f, field_counter)) & TYPE_FLAG_STUB)
+                 check_stub_method (t, method_counter, field_counter);
+               phys_name = TYPE_FN_FIELD_PHYSNAME (f, field_counter);
+               physnames[i1] = (char*) alloca (strlen (phys_name) + 1);
+               strcpy (physnames[i1], phys_name);
+               sym_arr[i1] = lookup_symbol (phys_name,
+                                            SYMBOL_BLOCK_VALUE (sym_class),
+                                            VAR_NAMESPACE,
+                                            (int *) NULL,
+                                            (struct symtab **) NULL);
+               if (sym_arr[i1]) i1++;
+             }
+       }
+    }
+  /* Only search baseclasses if there is no match yet,
+   * since names in derived classes override those in baseclasses.
+   */
+  if (i1)
+    return i1;
+  for (ibase = 0; ibase < TYPE_N_BASECLASSES (t); ibase++)
+    i1 += find_methods(TYPE_BASECLASS(t, ibase), name,
+                      physnames + i1, sym_arr + i1);
+  return i1;
+}
+
 /* Parse a string that specifies a line number.
    Pass the address of a char * variable; that variable will be
    advanced over the characters actually parsed.
@@ -1121,6 +1825,8 @@ find_pc_line_pc_range (pc, startptr, endptr)
    FILE:LINENUM -- that line in that file.  PC returned is 0.
    FUNCTION -- line number of openbrace of that function.
       PC returned is the start of the function.
+   VARIABLE -- line number of definition of that variable.
+      PC returned is 0.
    FILE:FUNCTION -- likewise, but prefer functions in that file.
    *EXPR -- line in which address EXPR appears.
 
@@ -1147,19 +1853,22 @@ decode_line_1 (argptr, funfirstline, default_symtab, default_line)
 {
   struct symtabs_and_lines decode_line_2 ();
   struct symtabs_and_lines values;
-  struct symtab_and_line value;
+  struct symtab_and_line val;
   register char *p, *p1;
+  char *q, *q1;
   register struct symtab *s;
+
   register struct symbol *sym;
+  /* The symtab that SYM was found in.  */
+  struct symtab *sym_symtab;
+
   register CORE_ADDR pc;
   register int i;
   char *copy;
   struct symbol *sym_class;
-  char *class_name, *method_name, *phys_name;
-  int method_counter;
   int i1;
   struct symbol **sym_arr;
-  struct type *t, *field;
+  struct type *t;
   char **physnames;
   
   /* Defaults have defaults.  */
@@ -1176,7 +1885,8 @@ decode_line_1 (argptr, funfirstline, default_symtab, default_line)
     {
       (*argptr)++;
       pc = parse_and_eval_address_1 (argptr);
-      values.sals = (struct symtab_and_line *)malloc (sizeof (struct symtab_and_line));
+      values.sals = (struct symtab_and_line *)
+       xmalloc (sizeof (struct symtab_and_line));
       values.nelts = 1;
       values.sals[0] = find_pc_line (pc, 0);
       values.sals[0].pc = pc;
@@ -1194,6 +1904,7 @@ decode_line_1 (argptr, funfirstline, default_symtab, default_line)
     }
   while (p[0] == ' ' || p[0] == '\t') p++;
 
+  q = operator_chars (*argptr, &q1);
   if (p[0] == ':')
     {
 
@@ -1212,7 +1923,8 @@ decode_line_1 (argptr, funfirstline, default_symtab, default_line)
          while (*p == ' ' || *p == '\t') p++;
          *argptr = p;
 
-         sym_class = lookup_symbol (copy, 0, STRUCT_NAMESPACE, 0);
+         sym_class = lookup_symbol (copy, 0, STRUCT_NAMESPACE, 0, 
+                                    (struct symtab **)NULL);
        
          if (sym_class &&
              (TYPE_CODE (SYMBOL_TYPE (sym_class)) == TYPE_CODE_STRUCT
@@ -1222,9 +1934,23 @@ decode_line_1 (argptr, funfirstline, default_symtab, default_line)
                 Find the next token (everything up to end or next whitespace). */
              p = *argptr;
              while (*p && *p != ' ' && *p != '\t' && *p != ',' && *p !=':') p++;
-             copy = (char *) alloca (p - *argptr + 1);
-             bcopy (*argptr, copy, p - *argptr);
-             copy[p - *argptr] = '\0';
+             q = operator_chars (*argptr, &q1);
+
+             copy = (char *) alloca (p - *argptr + 1 + (q1 - q));
+             if (q1 - q)
+               {
+                 copy[0] = 'o';
+                 copy[1] = 'p';
+                 copy[2] = CPLUS_MARKER;
+                 bcopy (q, copy + 3, q1 - q);
+                 copy[3 + (q1 - q)] = '\0';
+                 p = q1;
+               }
+             else
+               {
+                 bcopy (*argptr, copy, p - *argptr);
+                 copy[p - *argptr] = '\0';
+               }
 
              /* no line number may be specified */
              while (*p == ' ' || *p == '\t') p++;
@@ -1241,47 +1967,19 @@ decode_line_1 (argptr, funfirstline, default_symtab, default_line)
                  /* destructors are a special case.  */
                  struct fn_field *f = TYPE_FN_FIELDLIST1 (t, 0);
                  int len = TYPE_FN_FIELDLIST_LENGTH (t, 0) - 1;
-                 phys_name = TYPE_FN_FIELD_PHYSNAME (f, len);
+                 char *phys_name = TYPE_FN_FIELD_PHYSNAME (f, len);
                  physnames[i1] = (char *)alloca (strlen (phys_name) + 1);
                  strcpy (physnames[i1], phys_name);
-                 sym_arr[i1] = lookup_symbol (phys_name, SYMBOL_BLOCK_VALUE (sym_class), VAR_NAMESPACE, 0);
+                 sym_arr[i1] =
+                   lookup_symbol (phys_name, SYMBOL_BLOCK_VALUE (sym_class),
+                                  VAR_NAMESPACE, 0, (struct symtab **)NULL);
                  if (sym_arr[i1]) i1++;
                }
-             else while (t)
-               {
-                 class_name = TYPE_NAME (t);
-                 while (*class_name++ != ' ');
-
-                 sym_class = lookup_symbol (class_name, 0, STRUCT_NAMESPACE, 0);
-                 for (method_counter = TYPE_NFN_FIELDS (SYMBOL_TYPE (sym_class)) - 1;
-                      method_counter >= 0;
-                      --method_counter)
-                   {
-                     int field_counter;
-                     struct fn_field *f =
-                       TYPE_FN_FIELDLIST1 (SYMBOL_TYPE (sym_class), method_counter);
-
-                     method_name = TYPE_FN_FIELDLIST_NAME (SYMBOL_TYPE (sym_class), method_counter);
-                     if (!strcmp (copy, method_name))
-                       for (field_counter = TYPE_FN_FIELDLIST_LENGTH (SYMBOL_TYPE (sym_class), method_counter) - 1;
-                            field_counter >= 0;
-                            --field_counter)
-                         {
-                           phys_name = TYPE_FN_FIELD_PHYSNAME (f, field_counter);
-                           physnames[i1] = (char*) alloca (strlen (phys_name) + 1);
-                           strcpy (physnames[i1], phys_name);
-                           sym_arr[i1] = lookup_symbol (phys_name, SYMBOL_BLOCK_VALUE (sym_class), VAR_NAMESPACE, 0);
-                           if (sym_arr[i1]) i1++;
-                         }
-                   }
-                 if (TYPE_N_BASECLASSES (t))
-                   t = TYPE_BASECLASS(t, 1);
-                 else
-                   break;
-               }
-
+             else
+               i1 = find_methods (t, copy, physnames, sym_arr);
              if (i1 == 1)
                {
+                 /* There is exactly one field with that name.  */
                  sym = sym_arr[0];
 
                  if (sym && SYMBOL_CLASS (sym) == LOC_BLOCK)
@@ -1290,7 +1988,7 @@ decode_line_1 (argptr, funfirstline, default_symtab, default_line)
                      pc = BLOCK_START (SYMBOL_BLOCK_VALUE (sym)) + FUNCTION_START_OFFSET;
                      if (funfirstline)
                        SKIP_PROLOGUE (pc);
-                     values.sals = (struct symtab_and_line *)malloc (sizeof (struct symtab_and_line));
+                     values.sals = (struct symtab_and_line *)xmalloc (sizeof (struct symtab_and_line));
                      values.nelts = 1;
                      values.sals[0] = find_pc_line (pc, 0);
                      values.sals[0].pc = (values.sals[0].end && values.sals[0].pc != pc) ? values.sals[0].end : pc;
@@ -1303,14 +2001,28 @@ decode_line_1 (argptr, funfirstline, default_symtab, default_line)
                }
              if (i1 > 0)
                {
-                 return decode_line_2 (argptr, sym_arr, physnames,
-                                       i1, funfirstline);
+                 /* There is more than one field with that name
+                    (overloaded).  Ask the user which one to use.  */
+                 return decode_line_2 (sym_arr, i1, funfirstline);
                }
              else
-               error ("that class does not have any method named %s",copy);
+               {
+                 char *tmp;
+
+                 if (OPNAME_PREFIX_P (copy))
+                   {
+                     tmp = (char *)alloca (strlen (copy+3) + 9);
+                     strcpy (tmp, "operator ");
+                     strcat (tmp, copy+3);
+                   }
+                 else
+                   tmp = copy;
+                 error ("that class does not have any method named %s", tmp);
+               }
            }
          else
-           error("no class, struct, or union named %s", copy );
+           /* The quotes are important if copy is empty.  */
+           error("No class, struct, or union named \"%s\"", copy );
        }
       /*  end of C++  */
 
@@ -1318,16 +2030,28 @@ decode_line_1 (argptr, funfirstline, default_symtab, default_line)
       /* Extract the file name.  */
       p1 = p;
       while (p != *argptr && p[-1] == ' ') --p;
-      copy = (char *) alloca (p - *argptr + 1);
-      bcopy (*argptr, copy, p - *argptr);
-      copy[p - *argptr] = 0;
+      copy = (char *) alloca (p - *argptr + 1 + (q1 - q));
+      if (q1 - q)
+       {
+         copy[0] = 'o';
+         copy[1] = 'p';
+         copy[2] = CPLUS_MARKER;
+         bcopy (q, copy + 3, q1-q);
+         copy[3 + (q1-q)] = 0;
+         p = q1;
+       }
+      else
+       {
+         bcopy (*argptr, copy, p - *argptr);
+         copy[p - *argptr] = 0;
+       }
 
       /* Find that file's data.  */
       s = lookup_symtab (copy);
       if (s == 0)
        {
          if (symtab_list == 0 && partial_symtab_list == 0)
-           error ("No symbol table is loaded.  Use the \"symbol-file\" command.");
+           error (no_symtab_msg);
          error ("No source file named %s.", copy);
        }
 
@@ -1353,15 +2077,13 @@ decode_line_1 (argptr, funfirstline, default_symtab, default_line)
       enum sign {none, plus, minus} sign = none;
 
       /* This is where we need to make sure that we have good defaults.
-        We must guarrantee that this section of code is never executed
+        We must guarantee that this section of code is never executed
         when we are called with just a function name, since
         select_source_symtab calls us with such an argument  */
 
       if (s == 0 && default_symtab == 0)
        {
-         if (symtab_list == 0 && partial_symtab_list == 0)
-           error ("No symbol table is loaded.  Use the \"symbol-file\" command.");
-         select_source_symtab (symtab_list);
+         select_source_symtab (0);
          default_symtab = current_source_symtab;
          default_line = current_source_line;
        }
@@ -1370,38 +2092,40 @@ decode_line_1 (argptr, funfirstline, default_symtab, default_line)
        sign = plus, (*argptr)++;
       else if (**argptr == '-')
        sign = minus, (*argptr)++;
-      value.line = atoi (*argptr);
+      val.line = atoi (*argptr);
       switch (sign)
        {
        case plus:
          if (p == *argptr)
-           value.line = 5;
+           val.line = 5;
          if (s == 0)
-           value.line = default_line + value.line;
+           val.line = default_line + val.line;
          break;
        case minus:
          if (p == *argptr)
-           value.line = 15;
+           val.line = 15;
          if (s == 0)
-           value.line = default_line - value.line;
+           val.line = default_line - val.line;
          else
-           value.line = 1;
+           val.line = 1;
          break;
+       case none:
+         break;        /* No need to adjust val.line.  */
        }
 
       while (*p == ' ' || *p == '\t') p++;
       *argptr = p;
       if (s == 0)
        s = default_symtab;
-      value.symtab = s;
-      value.pc = 0;
-      values.sals = (struct symtab_and_line *)malloc (sizeof (struct symtab_and_line));
-      values.sals[0] = value;
+      val.symtab = s;
+      val.pc = 0;
+      values.sals = (struct symtab_and_line *)xmalloc (sizeof (struct symtab_and_line));
+      values.sals[0] = val;
       values.nelts = 1;
       return values;
     }
 
-  /* Arg token is not digits => try it as a function name
+  /* Arg token is not digits => try it as a variable name
      Find the next token (everything up to end or next whitespace).  */
   p = *argptr;
   while (*p && *p != ' ' && *p != '\t' && *p != ',') p++;
@@ -1411,52 +2135,77 @@ decode_line_1 (argptr, funfirstline, default_symtab, default_line)
   while (*p == ' ' || *p == '\t') p++;
   *argptr = p;
 
-  /* Look up that token as a function.
+  /* Look up that token as a variable.
      If file specified, use that file's per-file block to start with.  */
 
-  sym = lookup_symbol (copy, s ? BLOCKVECTOR_BLOCK (BLOCKVECTOR (s), 1) : 0,
-                      VAR_NAMESPACE, 0);
+  sym = lookup_symbol (copy,
+                      (s ? BLOCKVECTOR_BLOCK (BLOCKVECTOR (s), STATIC_BLOCK)
+                       : get_selected_block ()),
+                      VAR_NAMESPACE, 0, &sym_symtab);
 
-  if (sym && SYMBOL_CLASS (sym) == LOC_BLOCK)
+  if (sym != NULL)
     {
-      /* Arg is the name of a function */
-      pc = BLOCK_START (SYMBOL_BLOCK_VALUE (sym)) + FUNCTION_START_OFFSET;
-      if (funfirstline)
-       SKIP_PROLOGUE (pc);
-      value = find_pc_line (pc, 0);
-#ifdef convex
-      /* Convex: no need to suppress code on first line, if any */
-      value.pc = pc;
+      if (SYMBOL_CLASS (sym) == LOC_BLOCK)
+       {
+         /* Arg is the name of a function */
+         pc = BLOCK_START (SYMBOL_BLOCK_VALUE (sym)) + FUNCTION_START_OFFSET;
+         if (funfirstline)
+           SKIP_PROLOGUE (pc);
+         val = find_pc_line (pc, 0);
+#ifdef PROLOGUE_FIRSTLINE_OVERLAP
+         /* Convex: no need to suppress code on first line, if any */
+         val.pc = pc;
 #else
-      value.pc = (value.end && value.pc != pc) ? value.end : pc;
+         val.pc = (val.end && val.pc != pc) ? val.end : pc;
 #endif
-      values.sals = (struct symtab_and_line *)malloc (sizeof (struct symtab_and_line));
-      values.sals[0] = value;
-      values.nelts = 1;
-      return values;
+         values.sals = (struct symtab_and_line *)xmalloc (sizeof (struct symtab_and_line));
+         values.sals[0] = val;
+         values.nelts = 1;
+         
+         /* I think this is always the same as the line that
+            we calculate above, but the general principle is
+            "trust the symbols more than stuff like
+            SKIP_PROLOGUE".  */
+         if (SYMBOL_LINE (sym) != 0)
+           values.sals[0].line = SYMBOL_LINE (sym);
+         
+         return values;
+       }
+      else if (SYMBOL_LINE (sym) != 0)
+       {
+         /* We know its line number.  */
+         values.sals = (struct symtab_and_line *)
+           xmalloc (sizeof (struct symtab_and_line));
+         values.nelts = 1;
+         bzero (&values.sals[0], sizeof (values.sals[0]));
+         values.sals[0].symtab = sym_symtab;
+         values.sals[0].line = SYMBOL_LINE (sym);
+         return values;
+       }
+      else
+       /* This can happen if it is compiled with a compiler which doesn't
+          put out line numbers for variables.  */
+       error ("Line number not known for symbol \"%s\"", copy);
     }
 
-  if (sym)
-    error ("%s is not a function.", copy);
-
-  if ((i = lookup_misc_func (copy)) < 0)
-    error ("Function %s not defined.", copy);
-  else
+  if ((i = lookup_misc_func (copy)) >= 0)
     {
-      value.symtab = 0;
-      value.line = 0;
-      value.pc = misc_function_vector[i].address + FUNCTION_START_OFFSET;
+      val.symtab = 0;
+      val.line = 0;
+      val.pc = misc_function_vector[i].address + FUNCTION_START_OFFSET;
       if (funfirstline)
-       SKIP_PROLOGUE (value.pc);
-      values.sals = (struct symtab_and_line *)malloc (sizeof (struct symtab_and_line));
-      values.sals[0] = value;
+       SKIP_PROLOGUE (val.pc);
+      values.sals = (struct symtab_and_line *)xmalloc (sizeof (struct symtab_and_line));
+      values.sals[0] = val;
       values.nelts = 1;
       return values;
     }
 
-  if (symtab_list == 0 && partial_symtab_list == 0)
-    error ("No symbol table is loaded.  Use the \"symbol-file\" command.");
+  if (symtab_list == 0 && partial_symtab_list == 0 && misc_function_count == 0)
+    error (no_symtab_msg);
+
   error ("Function %s not defined.", copy);
+  return values;       /* for lint */
 }
 
 struct symtabs_and_lines
@@ -1474,23 +2223,23 @@ decode_line_spec (string, funfirstline)
   return sals;
 }
 
+/* Given a list of NELTS symbols in sym_arr (with corresponding
+   mangled names in physnames), return a list of lines to operate on
+   (ask user if necessary).  */
 struct symtabs_and_lines
-decode_line_2 (argptr, sym_arr, physnames, nelts, funfirstline)
-     char **argptr;
+decode_line_2 (sym_arr, nelts, funfirstline)
      struct symbol *sym_arr[];
-     char *physnames[];
      int nelts;
      int funfirstline;
 {
-  char *getenv();
   struct symtabs_and_lines values, return_values;
   register CORE_ADDR pc;
-  char *args, *arg1, *gdb_read_line ();
+  char *args, *arg1, *command_line_input ();
   int i;
   char *prompt;
 
   values.sals = (struct symtab_and_line *) alloca (nelts * sizeof(struct symtab_and_line));
-  return_values.sals = (struct symtab_and_line *) malloc (nelts * sizeof(struct symtab_and_line));
+  return_values.sals = (struct symtab_and_line *) xmalloc (nelts * sizeof(struct symtab_and_line));
 
   i = 0;
   printf("[0] cancel\n[1] all\n");
@@ -1504,6 +2253,8 @@ decode_line_2 (argptr, sym_arr, physnames, nelts, funfirstline)
          if (funfirstline)
            SKIP_PROLOGUE (pc);
          values.sals[i] = find_pc_line (pc, 0);
+         values.sals[i].pc = (values.sals[i].end && values.sals[i].pc != pc) ?
+                              values.sals[i].end                      :  pc;
          printf("[%d] file:%s; line number:%d\n",
                 (i+2), values.sals[i].symtab->filename, values.sals[i].line);
        }
@@ -1518,7 +2269,7 @@ decode_line_2 (argptr, sym_arr, physnames, nelts, funfirstline)
   printf("%s ",prompt);
   fflush(stdout);
 
-  args = gdb_read_line (0, 0);
+  args = command_line_input (0, 0);
   
   if (args == 0)
     error_no_arg ("one or more choice numbers");
@@ -1571,7 +2322,7 @@ decode_line_2 (argptr, sym_arr, physnames, nelts, funfirstline)
 
 /* Return the index of misc function named NAME.  */
 
-static int
+int
 lookup_misc_func (name)
      register char *name;
 {
@@ -1583,44 +2334,76 @@ lookup_misc_func (name)
   return -1;           /* not found */
 }
 \f
-/*
- * Slave routine for sources_info
- */
-static int rows_output;
-
+/* Slave routine for sources_info.  Force line breaks at ,'s.
+   NAME is the name to print and *FIRST is nonzero if this is the first
+   name printed.  Set *FIRST to zero.  */
 static void
-output_source_filename (name, next)
-char *name;
-int next;
+output_source_filename (name, first)
+     char *name;
+     int *first;
 {
-  static int column = 0;
-  
-  if (column != 0 && column + strlen (name) >= 70)
+  static int column;
+  /* Table of files printed so far.  Since a single source file can
+     result in several partial symbol tables, we need to avoid printing
+     it more than once.  Note: if some of the psymtabs are read in and
+     some are not, it gets printed both under "Source files for which
+     symbols have been read" and "Source files for which symbols will
+     be read in on demand".  I consider this a reasonable way to deal
+     with the situation.  I'm not sure whether this can also happen for
+     symtabs; it doesn't hurt to check.  */
+  static char **tab = NULL;
+  /* Allocated size of tab in elements.
+     Start with one 256-byte block (when using GNU malloc.c).
+     24 is the malloc overhead when range checking is in effect.  */
+  static int tab_alloc_size = (256 - 24) / sizeof (char *);
+  /* Current size of tab in elements.  */
+  static int tab_cur_size;
+
+  char **p;
+
+  if (*first)
+    {
+      if (tab == NULL)
+       tab = (char **) xmalloc (tab_alloc_size * sizeof (*tab));
+      tab_cur_size = 0;
+    }
+
+  /* Is NAME in tab?  */
+  for (p = tab; p < tab + tab_cur_size; p++)
+    if (strcmp (*p, name) == 0)
+      /* Yes; don't print it again.  */
+      return;
+  /* No; add it to tab.  */
+  if (tab_cur_size == tab_alloc_size)
+    {
+      tab_alloc_size *= 2;
+      tab = (char **) xrealloc (tab, tab_alloc_size * sizeof (*tab));
+    }
+  tab[tab_cur_size++] = name;
+
+  if (*first)
     {
-      printf ("\n");
       column = 0;
-      if (++rows_output >= 21)
-       {
-         printf ("--Type Return to print more--");
-         fflush (stdout);
-         gdb_read_line (0, 0);
-         rows_output = 0;
-       }
+      *first = 0;
     }
-  else if (column != 0)
+  else
     {
-      printf (" ");
+      printf_filtered (",");
       column++;
     }
-  printf ("%s", name);
-  column += strlen (name);
-  if (next)
+
+  if (column != 0 && column + strlen (name) >= 70)
+    {
+      printf_filtered ("\n");
+      column = 0;
+    }
+  else if (column != 0)
     {
-      printf (",");
+      printf_filtered (" ");
       column++;
     }
-  
-  if (!next) column = 0;
+  fputs_filtered (name, stdout);
+  column += strlen (name);
 }  
 
 static void
@@ -1628,46 +2411,44 @@ sources_info ()
 {
   register struct symtab *s;
   register struct partial_symtab *ps;
-  register int column = 0;
-
+  int first;
+  
   if (symtab_list == 0 && partial_symtab_list == 0)
     {
-      printf ("No symbol table is loaded.\n");
+      printf (no_symtab_msg);
       return;
     }
   
-  printf ("Source files for which symbols have been read in:\n\n");
-  rows_output = 2;
+  printf_filtered ("Source files for which symbols have been read in:\n\n");
+
+  first = 1;
   for (s = symtab_list; s; s = s->next)
-    output_source_filename (s->filename, s->next);
-  printf ("\n\n");
+    output_source_filename (s->filename, &first);
+  printf_filtered ("\n\n");
   
-  printf ("Source files for which symbols will be read in on demand:\n\n");
-  rows_output += 2;
+  printf_filtered ("Source files for which symbols will be read in on demand:\n\n");
+
+  first = 1;
   for (ps = partial_symtab_list; ps; ps = ps->next)
-    output_source_filename (ps->filename, ps->next);
-  printf ("\n");
+    if (!ps->readin)
+      output_source_filename (ps->filename, &first);
+  printf_filtered ("\n");
 }
 
 /* List all symbols (if REGEXP is 0) or all symbols matching REGEXP.
    If CLASS is zero, list all symbols except functions and type names.
    If CLASS is 1, list only functions.
-   If CLASS is 2, list only type names.  */
-
-#define MORE  \
-{ print_count++;               \
-  if (print_count >= 21)       \
-    { printf ("--Type Return to print more--");        \
-      print_count = 0;         \
-      fflush (stdout);         \
-      gdb_read_line (0, 0); } }
+   If CLASS is 2, list only type names.
+   If CLASS is 3, list only method names.
 
-static void sort_block_syms ();
+   BPT is non-zero if we should set a breakpoint at the functions
+   we find.  */
 
 static void
-list_symbols (regexp, class)
+list_symbols (regexp, class, bpt)
      char *regexp;
      int class;
+     int bpt;
 {
   register struct symtab *s;
   register struct partial_symtab *ps;
@@ -1680,18 +2461,18 @@ list_symbols (regexp, class)
   char *val;
   static char *classnames[]
     = {"variable", "function", "type", "method"};
-  int print_count = 0;
   int found_in_file = 0;
+  int found_misc = 0;
+  static enum misc_function_type types[]
+    = {mf_data, mf_text, mf_abs, mf_unknown};
+  static enum misc_function_type types2[]
+    = {mf_bss,  mf_text, mf_abs, mf_unknown};
+  enum misc_function_type ourtype = types[class];
+  enum misc_function_type ourtype2 = types2[class];
 
   if (regexp)
-    if (val = (char *) re_comp (regexp))
-      error ("Invalid regexp: %s", val);
-
-  printf (regexp
-         ? "All %ss matching regular expression \"%s\":\n"
-         : "All defined %ss:\n",
-         classnames[class],
-         regexp);
+    if (0 != (val = re_comp (regexp)))
+      error ("Invalid regexp (%s): %s", val, regexp);
 
   /* Search through the partial_symtab_list *first* for all symbols
      matching the regexp.  That way we don't have to reproduce all of
@@ -1701,24 +2482,27 @@ list_symbols (regexp, class)
       struct partial_symbol *bound, *gbound, *sbound;
       int keep_going = 1;
 
-      gbound = global_psymbols + ps->globals_offset + ps->n_global_syms;
-      sbound = static_psymbols + ps->statics_offset + ps->n_static_syms;
+      if (ps->readin) continue;
+      
+      gbound = global_psymbols.list + ps->globals_offset + ps->n_global_syms;
+      sbound = static_psymbols.list + ps->statics_offset + ps->n_static_syms;
       bound = gbound;
 
       /* Go through all of the symbols stored in a partial
         symtab in one loop. */
-      psym = global_psymbols + ps->globals_offset;
+      psym = global_psymbols.list + ps->globals_offset;
       while (keep_going)
        {
          if (psym >= bound)
            {
              if (bound == gbound && ps->n_static_syms != 0)
                {
-                 psym = static_psymbols + ps->statics_offset;
+                 psym = static_psymbols.list + ps->statics_offset;
                  bound = sbound;
                }
              else
                keep_going = 0;
+             continue;
            }
          else
            {
@@ -1728,12 +2512,12 @@ list_symbols (regexp, class)
                 load the file and go on to the next one */
              if ((regexp == 0 || re_exec (SYMBOL_NAME (psym)))
                  && ((class == 0 && SYMBOL_CLASS (psym) != LOC_TYPEDEF
-                      && SYMBOL_CLASS (psym) != LOC_BLOCK)
+                                 && SYMBOL_CLASS (psym) != LOC_BLOCK)
                      || (class == 1 && SYMBOL_CLASS (psym) == LOC_BLOCK)
                      || (class == 2 && SYMBOL_CLASS (psym) == LOC_TYPEDEF)
                      || (class == 3 && SYMBOL_CLASS (psym) == LOC_BLOCK)))
                {
-                 psymtab_to_symtab(ps);
+                 (void) PSYMTAB_TO_SYMTAB(ps);
                  keep_going = 0;
                }
            }
@@ -1741,6 +2525,33 @@ list_symbols (regexp, class)
        }
     }
 
+  /* Here, we search through the misc function vector for functions that
+     match, and call find_pc_symtab on them to force their symbols to
+     be read.  The symbol will then be found during the scan of symtabs
+     below.  If find_pc_symtab fails, set found_misc so that we will
+     rescan to print any matching symbols without debug info.  */
+
+  if (class == 1) {
+    for (i = 0; i < misc_function_count; i++) {
+      if (misc_function_vector[i].type != ourtype
+       && misc_function_vector[i].type != ourtype2)
+       continue;
+      if (regexp == 0 || re_exec (misc_function_vector[i].name)) {
+         if (0 == find_pc_symtab (misc_function_vector[i].address))
+           found_misc = 1;
+      }
+    }
+  }
+
+  /* Printout here so as to get after the "Reading in symbols"
+     messages which will be generated above.  */
+  if (!bpt)
+    printf_filtered (regexp
+         ? "All %ss matching regular expression \"%s\":\n"
+         : "All defined %ss:\n",
+         classnames[class],
+         regexp);
+
   for (s = symtab_list; s; s = s->next)
     {
       found_in_file = 0;
@@ -1751,7 +2562,7 @@ list_symbols (regexp, class)
         It happens that the first symtab in the list
         for any given blockvector is the main file.  */
       if (bv != prev_bv)
-       for (i = 0; i < 2; i++)
+       for (i = GLOBAL_BLOCK; i <= STATIC_BLOCK; i++)
          {
            b = BLOCKVECTOR_BLOCK (bv, i);
            /* Skip the sort if this block is always sorted.  */
@@ -1763,121 +2574,132 @@ list_symbols (regexp, class)
                sym = BLOCK_SYM (b, j);
                if ((regexp == 0 || re_exec (SYMBOL_NAME (sym)))
                    && ((class == 0 && SYMBOL_CLASS (sym) != LOC_TYPEDEF
-                        && SYMBOL_CLASS (sym) != LOC_BLOCK)
+                                   && SYMBOL_CLASS (sym) != LOC_BLOCK)
                        || (class == 1 && SYMBOL_CLASS (sym) == LOC_BLOCK)
                        || (class == 2 && SYMBOL_CLASS (sym) == LOC_TYPEDEF)
                        || (class == 3 && SYMBOL_CLASS (sym) == LOC_BLOCK)))
                  {
-                   if (!found_in_file)
+                   if (bpt)
                      {
-                       printf ("\nFile %s:\n", s->filename);
-                       print_count += 2;
+                       /* Set a breakpoint here, if it's a function */
+                       if (class == 1)
+                         break_command (SYMBOL_NAME(sym), 0);
+                     }
+                   else if (!found_in_file)
+                     {
+                       fputs_filtered ("\nFile ", stdout);
+                       fputs_filtered (s->filename, stdout);
+                       fputs_filtered (":\n", stdout);
                      }
                    found_in_file = 1;
-#if 0
-                   MORE;
-#else
-                   {
-                     print_count++;
-                     if (print_count >= 21)
-                       {
-                         printf ("--Type Return to print more--");
-                         print_count = 0;
-                         fflush (stdout);
-                         gdb_read_line (0, 0);
-                       }
-                   }
-#endif
-                   if (class != 2 && i == 1)
-                     printf ("static ");
+
+                   if (class != 2 && i == STATIC_BLOCK)
+                     printf_filtered ("static ");
+
+                   /* Typedef that is not a C++ class */
                    if (class == 2
                        && SYMBOL_NAMESPACE (sym) != STRUCT_NAMESPACE)
-                     printf ("typedef ");
-
-                   if (class < 3)
+                      typedef_print (SYMBOL_TYPE(sym), sym, stdout);
+                   /* variable, func, or typedef-that-is-c++-class */
+                   else if (class < 2 || 
+                            (class == 2 && 
+                               SYMBOL_NAMESPACE(sym) == STRUCT_NAMESPACE))
                      {
-                       type_print (SYMBOL_TYPE (sym),
-                                   (SYMBOL_CLASS (sym) == LOC_TYPEDEF
-                                    ? "" : SYMBOL_NAME (sym)),
-                                   stdout, 0);
-                       printf (";\n");
+                        type_print (SYMBOL_TYPE (sym),
+                                    (SYMBOL_CLASS (sym) == LOC_TYPEDEF
+                                     ? "" : SYMBOL_NAME (sym)),
+                                    stdout, 0);
+                        
+                        printf_filtered (";\n");
                      }
                    else
                      {
-                       char buf[1024];
 # if 0
+                       char buf[1024];
                        type_print_base (TYPE_FN_FIELD_TYPE(t, i), stdout, 0, 0); 
                        type_print_varspec_prefix (TYPE_FN_FIELD_TYPE(t, i), stdout, 0); 
-                       sprintf (buf, " %s::", TYPE_NAME (t));
+                       sprintf (buf, " %s::", type_name_no_tag (t));
                        type_print_method_args (TYPE_FN_FIELD_ARGS (t, i), buf, name, stdout);
 # endif
                      }
-                   if (class == 2
-                       && SYMBOL_NAMESPACE (sym) != STRUCT_NAMESPACE
-                       && (TYPE_NAME ((SYMBOL_TYPE (sym))) == 0
-                           || 0 != strcmp (TYPE_NAME ((SYMBOL_TYPE (sym))),
-                                           SYMBOL_NAME (sym))))
-                     printf (" %s", SYMBOL_NAME (sym));
                  }
              }
          }
       prev_bv = bv;
     }
+
+
+  /* If there are no eyes, avoid all contact.  I mean, if there are
+     no debug symbols, then print directly from the misc_function_vector.  */
+
+  if (found_misc || class != 1) {
+    found_in_file = 0;
+    for (i = 0; i < misc_function_count; i++) {
+      if (misc_function_vector[i].type != ourtype
+       && misc_function_vector[i].type != ourtype2)
+       continue;
+      if (regexp == 0 || re_exec (misc_function_vector[i].name)) {
+       /* Functions:  Look up by address. */
+       if (class == 1)
+         if (0 != find_pc_symtab (misc_function_vector[i].address))
+           continue;
+       /* Variables/Absolutes:  Look up by name */
+       if (0 != lookup_symbol (misc_function_vector[i].name, 
+               (struct block *)0, VAR_NAMESPACE, 0, (struct symtab **)0))
+         continue;
+       if (!found_in_file) {
+         printf_filtered ("\nNon-debugging symbols:\n");
+         found_in_file = 1;
+       }
+       printf_filtered ("      %08x  %s\n",
+             misc_function_vector[i].address,
+             misc_function_vector[i].name);
+      }
+    }
+  }
 }
 
 static void
 variables_info (regexp)
      char *regexp;
 {
-  list_symbols (regexp, 0);
+  list_symbols (regexp, 0, 0);
 }
 
 static void
 functions_info (regexp)
      char *regexp;
 {
-  list_symbols (regexp, 1);
+  list_symbols (regexp, 1, 0);
 }
 
 static void
 types_info (regexp)
      char *regexp;
 {
-  list_symbols (regexp, 2);
+  list_symbols (regexp, 2, 0);
 }
 
+#if 0
+/* Tiemann says: "info methods was never implemented."  */
 static void
 methods_info (regexp)
      char *regexp;
 {
-  list_symbols (regexp, 3);
-}
-\f
-/* Call sort_block_syms to sort alphabetically the symbols of one block.  */
-
-static int
-compare_symbols (s1, s2)
-     struct symbol **s1, **s2;
-{
-  /* Names that are less should come first.  */
-  register int namediff = strcmp (SYMBOL_NAME (*s1), SYMBOL_NAME (*s2));
-  if (namediff != 0) return namediff;
-  /* For symbols of the same name, registers should come first.  */
-  return ((SYMBOL_CLASS (*s2) == LOC_REGISTER)
-         - (SYMBOL_CLASS (*s1) == LOC_REGISTER));
+  list_symbols (regexp, 3, 0);
 }
+#endif /* 0 */
 
+/* Breakpoint all functions matching regular expression. */
 static void
-sort_block_syms (b)
-     register struct block *b;
+rbreak_command (regexp)
+     char *regexp;
 {
-  qsort (&BLOCK_SYM (b, 0), BLOCK_NSYMS (b),
-        sizeof (struct symbol *), compare_symbols);
+  list_symbols (regexp, 1, 1);
 }
 \f
-/* Initialize the standard C scalar types.  */
+/* Helper function to initialize the standard scalar types.  */
 
-static
 struct type *
 init_type (code, length, uns, name)
      enum type_code code;
@@ -1899,7 +2721,6 @@ init_type (code, length, uns, name)
   /* C++ fancies.  */
   TYPE_NFN_FIELDS (type) = 0;
   TYPE_N_BASECLASSES (type) = 0;
-  TYPE_BASECLASSES (type) = 0;
   return type;
 }
 
@@ -1912,9 +2733,201 @@ contained_in (a, b)
 {
   if (!a || !b)
     return 0;
-  return a->startaddr >= b->startaddr && a->endaddr <= b->endaddr;
+  return BLOCK_START (a) >= BLOCK_START (b)
+      && BLOCK_END (a)   <= BLOCK_END (b);
+}
+
+\f
+/* Helper routine for make_symbol_completion_list.  */
+
+int return_val_size, return_val_index;
+char **return_val;
+
+void
+completion_list_add_symbol (symname)
+     char *symname;
+{
+  if (return_val_index + 3 > return_val_size)
+    return_val =
+      (char **)xrealloc (return_val,
+                        (return_val_size *= 2) * sizeof (char *));
+  
+  return_val[return_val_index] =
+    (char *)xmalloc (1 + strlen (symname));
+  
+  strcpy (return_val[return_val_index], symname);
+  
+  return_val[++return_val_index] = (char *)NULL;
+}
+
+/* Return a NULL terminated array of all symbols (regardless of class) which
+   begin by matching TEXT.  If the answer is no symbols, then the return value
+   is an array which contains only a NULL pointer.
+
+   Problem: All of the symbols have to be copied because readline
+   frees them.  I'm not going to worry about this; hopefully there
+   won't be that many.  */
+
+char **
+make_symbol_completion_list (text)
+  char *text;
+{
+  register struct symtab *s;
+  register struct partial_symtab *ps;
+  register struct block *b, *surrounding_static_block = 0;
+  extern struct block *get_selected_block ();
+  register int i, j;
+  struct partial_symbol *psym;
+
+  int text_len = strlen (text);
+  return_val_size = 100;
+  return_val_index = 0;
+  return_val =
+    (char **)xmalloc ((1 + return_val_size) *sizeof (char *));
+  return_val[0] = (char *)NULL;
+
+  /* Look through the partial symtabs for all symbols which begin
+     by matching TEXT.  Add each one that you find to the list.  */
+
+  for (ps = partial_symtab_list; ps; ps = ps->next)
+    {
+      /* If the psymtab's been read in we'll get it when we search
+        through the blockvector.  */
+      if (ps->readin) continue;
+
+      for (psym = global_psymbols.list + ps->globals_offset;
+          psym < (global_psymbols.list + ps->globals_offset
+                  + ps->n_global_syms);
+          psym++)
+       {
+         QUIT;                 /* If interrupted, then quit. */
+         if ((strncmp (SYMBOL_NAME (psym), text, text_len) == 0))
+           completion_list_add_symbol (SYMBOL_NAME (psym));
+       }
+      
+      for (psym = static_psymbols.list + ps->statics_offset;
+          psym < (static_psymbols.list + ps->statics_offset
+                  + ps->n_static_syms);
+          psym++)
+       {
+         QUIT;
+         if ((strncmp (SYMBOL_NAME (psym), text, text_len) == 0))
+           completion_list_add_symbol (SYMBOL_NAME (psym));
+       }
+    }
+
+  /* At this point scan through the misc function vector and add each
+     symbol you find to the list.  Eventually we want to ignore
+     anything that isn't a text symbol (everything else will be
+     handled by the psymtab code above).  */
+
+  for (i = 0; i < misc_function_count; i++)
+    if (!strncmp (text, misc_function_vector[i].name, text_len))
+      completion_list_add_symbol (misc_function_vector[i].name);
+
+  /* Search upwards from currently selected frame (so that we can
+     complete on local vars.  */
+  for (b = get_selected_block (); b; b = BLOCK_SUPERBLOCK (b))
+    {
+      if (!BLOCK_SUPERBLOCK (b))
+       surrounding_static_block = b; /* For elmin of dups */
+      
+      /* Also catch fields of types defined in this places which
+        match our text string.  Only complete on types visible
+        from current context.  */
+      for (i = 0; i < BLOCK_NSYMS (b); i++)
+       {
+         register struct symbol *sym = BLOCK_SYM (b, i);
+         
+         if (!strncmp (SYMBOL_NAME (sym), text, text_len))
+           completion_list_add_symbol (SYMBOL_NAME (sym));
+
+         if (SYMBOL_CLASS (sym) == LOC_TYPEDEF)
+           {
+             struct type *t = SYMBOL_TYPE (sym);
+             enum type_code c = TYPE_CODE (t);
+
+             if (c == TYPE_CODE_UNION || c == TYPE_CODE_STRUCT)
+               for (j = TYPE_N_BASECLASSES (t); j < TYPE_NFIELDS (t); j++)
+                 if (TYPE_FIELD_NAME (t, j) &&
+                     !strncmp (TYPE_FIELD_NAME (t, j), text, text_len))
+                   completion_list_add_symbol (TYPE_FIELD_NAME (t, j));
+           }
+       }
+    }
+
+  /* Go through the symtabs and check the externs and statics for
+     symbols which match.  */
+
+  for (s = symtab_list; s; s = s->next)
+    {
+      b = BLOCKVECTOR_BLOCK (BLOCKVECTOR (s), GLOBAL_BLOCK);
+      
+      for (i = 0; i < BLOCK_NSYMS (b); i++)
+       if (!strncmp (SYMBOL_NAME (BLOCK_SYM (b, i)), text, text_len))
+         completion_list_add_symbol (SYMBOL_NAME (BLOCK_SYM (b, i)));
+    }
+
+  for (s = symtab_list; s; s = s->next)
+    {
+      b = BLOCKVECTOR_BLOCK (BLOCKVECTOR (s), STATIC_BLOCK);
+
+      /* Don't do this block twice.  */
+      if (b == surrounding_static_block) continue;
+      
+      for (i = 0; i < BLOCK_NSYMS (b); i++)
+       if (!strncmp (SYMBOL_NAME (BLOCK_SYM (b, i)), text, text_len))
+         completion_list_add_symbol (SYMBOL_NAME (BLOCK_SYM (b, i)));
+    }
+
+  return (return_val);
 }
+\f
+#if 0
+/* Add the type of the symbol sym to the type of the current
+   function whose block we are in (assumed).  The type of
+   this current function is contained in *TYPE.
+   
+   This basically works as follows:  When we find a function
+   symbol (N_FUNC with a 'f' or 'F' in the symbol name), we record
+   a pointer to its type in the global in_function_type.  Every 
+   time we come across a parameter symbol ('p' in its name), then
+   this procedure adds the name and type of that parameter
+   to the function type pointed to by *TYPE.  (Which should correspond
+   to in_function_type if it was called correctly).
+
+   Note that since we are modifying a type, the result of 
+   lookup_function_type() should be bcopy()ed before calling
+   this.  When not in strict typing mode, the expression
+   evaluator can choose to ignore this.
+
+   Assumption:  All of a function's parameter symbols will
+   appear before another function symbol is found.  The parameters 
+   appear in the same order in the argument list as they do in the
+   symbol table. */
 
+void
+add_param_to_type (type,sym)
+   struct type **type;
+   struct symbol *sym;
+{
+   int num = ++(TYPE_NFIELDS(*type));
+
+   if(TYPE_NFIELDS(*type)-1)
+      TYPE_FIELDS(*type) = 
+        (struct field *)xrealloc((char *)(TYPE_FIELDS(*type)),
+                                 num*sizeof(struct field));
+   else
+      TYPE_FIELDS(*type) =
+        (struct field *)xmalloc(num*sizeof(struct field));
+   
+   TYPE_FIELD_BITPOS(*type,num-1) = num-1;
+   TYPE_FIELD_BITSIZE(*type,num-1) = 0;
+   TYPE_FIELD_TYPE(*type,num-1) = SYMBOL_TYPE(sym);
+   TYPE_FIELD_NAME(*type,num-1) = SYMBOL_NAME(sym);
+}
+#endif 
+\f
 void
 _initialize_symtab ()
 {
@@ -1922,38 +2935,31 @@ _initialize_symtab ()
            "All global and static variable names, or those matching REGEXP.");
   add_info ("functions", functions_info,
            "All function names, or those matching REGEXP.");
+
+  /* FIXME:  This command has at least the following problems:
+     1.  It prints builtin types (in a very strange and confusing fashion).
+     2.  It doesn't print right, e.g. with
+         typedef struct foo *FOO
+        type_print prints "FOO" when we want to make it (in this situation)
+        print "struct foo *".
+     I also think "ptype" or "whatis" is more likely to be useful (but if
+     there is much disagreement "info types" can be fixed).  */
   add_info ("types", types_info,
            "All types names, or those matching REGEXP.");
+
+#if 0
   add_info ("methods", methods_info,
            "All method names, or those matching REGEXP::REGEXP.\n\
 If the class qualifier is ommited, it is assumed to be the current scope.\n\
 If the first REGEXP is ommited, then all methods matching the second REGEXP\n\
 are listed.");
+#endif
   add_info ("sources", sources_info,
            "Source files in the program.");
 
-  obstack_init (symbol_obstack);
-  obstack_init (psymbol_obstack);
-
-  builtin_type_void = init_type (TYPE_CODE_VOID, 0, 0, "void");
+  add_com ("rbreak", no_class, rbreak_command,
+           "Set a breakpoint for all functions matching REGEXP.");
 
-  builtin_type_float = init_type (TYPE_CODE_FLT, sizeof (float), 0, "float");
-  builtin_type_double = init_type (TYPE_CODE_FLT, sizeof (double), 0, "double");
-
-  builtin_type_char = init_type (TYPE_CODE_INT, sizeof (char), 0, "char");
-  builtin_type_short = init_type (TYPE_CODE_INT, sizeof (short), 0, "short");
-  builtin_type_long = init_type (TYPE_CODE_INT, sizeof (long), 0, "long");
-  builtin_type_int = init_type (TYPE_CODE_INT, sizeof (int), 0, "int");
-
-  builtin_type_unsigned_char = init_type (TYPE_CODE_INT, sizeof (char), 1, "unsigned char");
-  builtin_type_unsigned_short = init_type (TYPE_CODE_INT, sizeof (short), 1, "unsigned short");
-  builtin_type_unsigned_long = init_type (TYPE_CODE_INT, sizeof (long), 1, "unsigned long");
-  builtin_type_unsigned_int = init_type (TYPE_CODE_INT, sizeof (int), 1, "unsigned int");
-#ifdef LONG_LONG
-  builtin_type_long_long =
-    init_type (TYPE_CODE_INT, sizeof (long long), 0, "long long");
-  builtin_type_unsigned_long_long = 
-    init_type (TYPE_CODE_INT, sizeof (long long), 1, "unsigned long long");
-#endif
+  /* Initialize the one built-in type that isn't language dependent... */
+  builtin_type_error = init_type (TYPE_CODE_ERROR, 0, 0, "<unknown type>");
 }
-
This page took 0.053515 seconds and 4 git commands to generate.