OSDN Git Service

Add framework support for darwin.
[pf3gnuchains/gcc-fork.git] / gcc / c-decl.c
index a346516..1145363 100644 (file)
@@ -1,6 +1,6 @@
 /* Process declarations and variables for C compiler.
    Copyright (C) 1988, 1992, 1993, 1994, 1995, 1996, 1997, 1998, 1999, 2000,
-   2001, 2002, 2003 Free Software Foundation, Inc.
+   2001, 2002, 2003, 2004 Free Software Foundation, Inc.
 
 This file is part of GCC.
 
@@ -61,7 +61,6 @@ enum decl_context
   FUNCDEF,                     /* Function definition */
   PARM,                                /* Declaration of parm before function body */
   FIELD,                       /* Declaration inside struct or union */
-  BITFIELD,                    /* Likewise but with specified width */
   TYPENAME};                   /* Typename (inside cast or sizeof)  */
 
 \f
@@ -126,6 +125,10 @@ static GTY(()) struct stmt_tree_s c_stmt_tree;
 
 static GTY(()) tree c_scope_stmt_stack;
 
+/* State saving variables.  */
+int c_in_iteration_stmt;
+int c_in_case_stmt;
+
 /* A list of external DECLs that appeared at block scope when there was
    some other global meaning for that identifier.  */
 static GTY(()) tree truly_local_externals;
@@ -227,21 +230,21 @@ struct c_scope GTY(())
 
   /* True if we are currently filling this scope with parameter
      declarations.  */
-  bool parm_flag : 1;
+  BOOL_BITFIELD parm_flag : 1;
 
   /* True if we already complained about forward parameter decls
      in this scope.  This prevents double warnings on
      foo (int a; int b; ...)  */
-  bool warned_forward_parm_decls : 1;
+  BOOL_BITFIELD warned_forward_parm_decls : 1;
 
   /* True if this is the outermost block scope of a function body.
      This scope contains the parameters, the local variables declared
      in the outermost block, and all the labels (except those in
      nested functions, or declared at block scope with __label__).  */
-  bool function_body : 1;
+  BOOL_BITFIELD function_body : 1;
 
   /* True means make a BLOCK for this scope no matter what.  */
-  bool keep : 1;
+  BOOL_BITFIELD keep : 1;
 };
 
 /* The scope currently in effect.  */
@@ -303,15 +306,12 @@ tree static_ctors, static_dtors;
 
 static struct c_scope *make_scope (void);
 static void pop_scope (void);
-static tree match_builtin_function_types (tree, tree);
-static int duplicate_decls (tree, tree, int, int);
-static int redeclaration_error_message (tree, tree);
 static tree make_label (tree, location_t);
 static void bind_label (tree, tree, struct c_scope *);
 static void implicit_decl_warning (tree);
 static tree lookup_tag (enum tree_code, tree, int);
 static tree lookup_name_current_level (tree);
-static tree grokdeclarator (tree, tree, enum decl_context, int);
+static tree grokdeclarator (tree, tree, enum decl_context, int, tree *);
 static tree grokparms (tree, int);
 static void layout_array_type (tree);
 static void store_parm_decls_newstyle (void);
@@ -321,6 +321,7 @@ static void c_expand_body_1 (tree, int);
 static tree any_external_decl (tree);
 static void record_external_decl (tree);
 static void warn_if_shadowing (tree, tree);
+static void check_bitfield_type_and_width (tree *, tree *, const char *);
 static void clone_underlying_type (tree);
 static bool flexible_array_type_p (tree);
 static hashval_t link_hash_hash        (const void *);
@@ -367,8 +368,7 @@ c_finish_incomplete_decl (tree decl)
          && ! DECL_EXTERNAL (decl)
          && TYPE_DOMAIN (type) == 0)
        {
-         warning ("%Harray '%D' assumed to have one element",
-                   &DECL_SOURCE_LOCATION (decl), decl);
+         warning ("%Jarray '%D' assumed to have one element", decl, decl);
 
          complete_array_type (type, NULL_TREE, 1);
 
@@ -411,6 +411,39 @@ pop_scope (void)
   scope_freelist = scope;
 }
 
+/* The Objective-C front-end often needs to determine the current scope.  */
+
+void *
+get_current_scope (void)
+{
+  return current_scope;
+}
+
+/* The following function is used only by Objective-C.  It needs to live here
+   because it accesses the innards of c_scope.  */
+
+void
+objc_mark_locals_volatile (void *enclosing_blk)
+{
+  struct c_scope *scope;
+
+  for (scope = current_scope;
+       scope && scope != enclosing_blk;
+       scope = scope->outer)
+    {
+      tree decl;
+
+      for (decl = scope->names; decl; decl = TREE_CHAIN (decl))
+       {
+         DECL_REGISTER (decl) = 0;
+         TREE_THIS_VOLATILE (decl) = 1;
+       }
+      /* Do not climb up past the current function.  */
+      if (scope->function_body)
+       break;
+    }
+}
+
 /* Nonzero if we are currently in the global scope.  */
 
 int
@@ -505,7 +538,8 @@ poplevel (int keep, int dummy ATTRIBUTE_UNUSED, int functionbody)
   tree decl;
   tree p;
 
-  scope->function_body |= functionbody;
+  /* The following line does not use |= due to a bug in HP's C compiler.  */
+  scope->function_body = scope->function_body | functionbody;
 
   if (keep == KEEP_MAYBE)
     keep = (scope->names || scope->tags);
@@ -538,22 +572,20 @@ poplevel (int keep, int dummy ATTRIBUTE_UNUSED, int functionbody)
 
   for (p = scope->names; p; p = TREE_CHAIN (p))
     {
-      const location_t *locus = &DECL_SOURCE_LOCATION (p);
-
       switch (TREE_CODE (p))
        {
        case LABEL_DECL:
          if (TREE_USED (p) && !DECL_INITIAL (p))
            {
-             error ("%Hlabel `%D' used but not defined", locus, p);
+             error ("%Jlabel `%D' used but not defined", p, p);
              DECL_INITIAL (p) = error_mark_node;
            }
          else if (!TREE_USED (p) && warn_unused_label)
            {
              if (DECL_INITIAL (p))
-               warning ("%Hlabel `%D' defined but not used", locus, p);
+               warning ("%Jlabel `%D' defined but not used", p, p);
              else
-               warning ("%Hlabel `%D' declared but not defined", locus, p);
+               warning ("%Jlabel `%D' declared but not defined", p, p);
            }
 
          IDENTIFIER_LABEL_VALUE (DECL_NAME (p)) = 0;
@@ -578,8 +610,8 @@ poplevel (int keep, int dummy ATTRIBUTE_UNUSED, int functionbody)
              && !DECL_IN_SYSTEM_HEADER (p)
              && DECL_NAME (p)
              && !DECL_ARTIFICIAL (p))
-           warning ("%Hunused variable `%D'", locus, p);
-         /* fall through */
+           warning ("%Junused variable `%D'", p, p);
+         /* Fall through.  */
 
        default:
        normal:
@@ -641,7 +673,7 @@ poplevel (int keep, int dummy ATTRIBUTE_UNUSED, int functionbody)
     IDENTIFIER_TAG_VALUE (TREE_PURPOSE (p)) = TREE_VALUE (p);
 
   /* Dispose of the block that we just made inside some higher level.  */
-  if (scope->function_body)
+  if (scope->function_body && current_function_decl)
     DECL_INITIAL (current_function_decl) = block;
   else if (scope->outer)
     {
@@ -720,12 +752,12 @@ pushtag (tree name, tree type)
   TYPE_CONTEXT (type) = DECL_CONTEXT (TYPE_STUB_DECL (type));
 }
 \f
-/* Subroutine of duplicate_decls.  Allow harmless mismatches in return
+/* Subroutine of compare_decls.  Allow harmless mismatches in return
    and argument types provided that the type modes match.  This function
    return a unified type given a suitable match, and 0 otherwise.  */
 
 static tree
-match_builtin_function_types (tree oldtype, tree newtype)
+match_builtin_function_types (tree newtype, tree oldtype)
 {
   tree newrettype, oldrettype;
   tree newargs, oldargs;
@@ -760,597 +792,577 @@ match_builtin_function_types (tree oldtype, tree newtype)
   return build_type_attribute_variant (trytype, TYPE_ATTRIBUTES (oldtype));
 }
 
-/* Handle when a new declaration NEWDECL
-   has the same name as an old one OLDDECL
-   in the same binding contour.
-   Prints an error message if appropriate.
+/* Subroutine of diagnose_mismatched_decls.  Check for function type
+   mismatch involving an empty arglist vs a nonempty one and give clearer
+   diagnostics.  */
+static void
+diagnose_arglist_conflict (tree newdecl, tree olddecl,
+                          tree newtype, tree oldtype)
+{
+  tree t;
+
+  if (TREE_CODE (olddecl) != FUNCTION_DECL
+      || !comptypes (TREE_TYPE (oldtype), TREE_TYPE (newtype), COMPARE_STRICT)
+      || !((TYPE_ARG_TYPES (oldtype) == 0 && DECL_INITIAL (olddecl) == 0)
+          ||
+          (TYPE_ARG_TYPES (newtype) == 0 && DECL_INITIAL (newdecl) == 0)))
+    return;
+
+  t = TYPE_ARG_TYPES (oldtype);
+  if (t == 0)
+    t = TYPE_ARG_TYPES (newtype);
+  for (; t; t = TREE_CHAIN (t))
+    {
+      tree type = TREE_VALUE (t);
 
-   If safely possible, alter OLDDECL to look like NEWDECL, and return 1.
-   Otherwise, return 0.
+      if (TREE_CHAIN (t) == 0
+         && TYPE_MAIN_VARIANT (type) != void_type_node)
+       {
+         inform ("a parameter list with an ellipsis can't match "
+                 "an empty parameter name list declaration");
+         break;
+       }
 
-   When DIFFERENT_BINDING_LEVEL is true, NEWDECL is an external declaration,
-   and OLDDECL is in an outer scope and should thus not be changed.  */
+      if (c_type_promotes_to (type) != type)
+       {
+         inform ("an argument type that has a default promotion can't match "
+                 "an empty parameter name list declaration");
+         break;
+       }
+    }
+}
 
-static int
-duplicate_decls (tree newdecl, tree olddecl, int different_binding_level,
-                int different_tu)
+/* Another subroutine of diagnose_mismatched_decls.  OLDDECL is an
+   old-style function definition, NEWDECL is a prototype declaration.
+   Diagnose inconsistencies in the argument list.  Returns TRUE if
+   the prototype is compatible, FALSE if not.  */
+static bool
+validate_proto_after_old_defn (tree newdecl, tree newtype, tree oldtype)
 {
-  int comptype_flags = (different_tu ? COMPARE_DIFFERENT_TU
-                       : COMPARE_STRICT);
-  int types_match = comptypes (TREE_TYPE (newdecl), TREE_TYPE (olddecl),
-                              comptype_flags);
-  int new_is_definition = (TREE_CODE (newdecl) == FUNCTION_DECL
-                          && DECL_INITIAL (newdecl) != 0);
-  tree oldtype = TREE_TYPE (olddecl);
-  tree newtype = TREE_TYPE (newdecl);
-  int errmsg = 0;
+  tree newargs, oldargs;
+  int i;
 
-  if (DECL_P (olddecl))
+  /* ??? Elsewhere TYPE_MAIN_VARIANT is not used in this context.  */
+#define END_OF_ARGLIST(t) (TYPE_MAIN_VARIANT (t) == void_type_node)
+
+  oldargs = TYPE_ACTUAL_ARG_TYPES (oldtype);
+  newargs = TYPE_ARG_TYPES (newtype);
+  i = 1;
+
+  for (;;)
     {
-      if (TREE_CODE (newdecl) == FUNCTION_DECL
-         && TREE_CODE (olddecl) == FUNCTION_DECL
-         && (DECL_UNINLINABLE (newdecl) || DECL_UNINLINABLE (olddecl)))
+      tree oldargtype = TREE_VALUE (oldargs);
+      tree newargtype = TREE_VALUE (newargs);
+
+      if (END_OF_ARGLIST (oldargtype) && END_OF_ARGLIST (newargtype))
+       break;
+
+      /* Reaching the end of just one list means the two decls don't
+        agree on the number of arguments.  */
+      if (END_OF_ARGLIST (oldargtype))
        {
-         if (DECL_DECLARED_INLINE_P (newdecl)
-             && DECL_UNINLINABLE (newdecl)
-             && lookup_attribute ("noinline", DECL_ATTRIBUTES (newdecl)))
-           /* Already warned elsewhere.  */;
-         else if (DECL_DECLARED_INLINE_P (olddecl)
-                  && DECL_UNINLINABLE (olddecl)
-                  && lookup_attribute ("noinline", DECL_ATTRIBUTES (olddecl)))
-           /* Already warned.  */;
-         else if (DECL_DECLARED_INLINE_P (newdecl)
-                  && ! DECL_DECLARED_INLINE_P (olddecl)
-                  && DECL_UNINLINABLE (olddecl)
-                  && lookup_attribute ("noinline", DECL_ATTRIBUTES (olddecl)))
-           {
-             warning ("%Hfunction '%D' redeclared as inline",
-                       &DECL_SOURCE_LOCATION (newdecl), newdecl);
-             warning ("%Hprevious declaration of function '%D' "
-                       "with attribute noinline",
-                       &DECL_SOURCE_LOCATION (olddecl), olddecl);
-           }
-         else if (DECL_DECLARED_INLINE_P (olddecl)
-                  && DECL_UNINLINABLE (newdecl)
-                  && lookup_attribute ("noinline", DECL_ATTRIBUTES (newdecl)))
-           {
-             warning ("%Hfunction '%D' redeclared with attribute noinline",
-                       &DECL_SOURCE_LOCATION (newdecl), newdecl);
-             warning ("%Hprevious declaration of function '%D' was inline",
-                       &DECL_SOURCE_LOCATION (olddecl), olddecl);
-           }
+         error ("%Jprototype for '%D' declares more arguments "
+                "than previous old-style definition", newdecl, newdecl);
+         return false;
+       }
+      else if (END_OF_ARGLIST (newargtype))
+       {
+         error ("%Jprototype for '%D' declares fewer arguments "
+                "than previous old-style definition", newdecl, newdecl);
+         return false;
        }
 
-      DECL_ATTRIBUTES (newdecl)
-       = (*targetm.merge_decl_attributes) (olddecl, newdecl);
+      /* Type for passing arg must be consistent with that declared
+        for the arg.  */
+      else if (! comptypes (oldargtype, newargtype, COMPARE_STRICT))
+       {
+         error ("%Jprototype for '%D' declares arg %d with incompatible type",
+                newdecl, newdecl, i);
+         return false;
+       }
+
+      oldargs = TREE_CHAIN (oldargs);
+      newargs = TREE_CHAIN (newargs);
+      i++;
     }
 
-  if (TREE_CODE (newtype) == ERROR_MARK
-      || TREE_CODE (oldtype) == ERROR_MARK)
-    types_match = 0;
+  /* If we get here, no errors were found, but do issue a warning
+     for this poor-style construct.  */
+  warning ("%Jprototype for '%D' follows non-prototype definition",
+          newdecl, newdecl);
+  return true;
+#undef END_OF_ARGLIST
+}
+
+/* Subroutine of diagnose_mismatched_decls.  Report the location of DECL,
+   first in a pair of mismatched declarations, using the diagnostic
+   function DIAG.  */
+static void
+locate_old_decl (tree decl, void (*diag)(const char *, ...))
+{
+  if (TREE_CODE (decl) == FUNCTION_DECL && DECL_BUILT_IN (decl))
+    ;
+  else if (DECL_INITIAL (decl))
+    diag (N_("%Jprevious definition of '%D' was here"), decl, decl);
+  else if (C_DECL_IMPLICIT (decl))
+    diag (N_("%Jprevious implicit declaration of '%D' was here"), decl, decl);
+  else
+    diag (N_("%Jprevious declaration of '%D' was here"), decl, decl);
+}
+
+/* Subroutine of duplicate_decls.  Compare NEWDECL to OLDDECL.
+   Returns true if the caller should proceed to merge the two, false
+   if OLDDECL should simply be discarded.  As a side effect, issues
+   all necessary diagnostics for invalid or poor-style combinations.
+   If it returns true, writes the types of NEWDECL and OLDDECL to
+   *NEWTYPEP and *OLDTYPEP - these may have been adjusted from
+   TREE_TYPE (NEWDECL, OLDDECL) respectively.  */
+
+static bool
+diagnose_mismatched_decls (tree newdecl, tree olddecl,
+                          tree *newtypep, tree *oldtypep)
+{
+  tree newtype, oldtype;
+  bool pedwarned = false;
+  bool warned = false;
+
+  /* If we have error_mark_node for either decl or type, just discard
+     the previous decl - we're in an error cascade already.  */
+  if (olddecl == error_mark_node || newdecl == error_mark_node)
+    return false;
+  *oldtypep = oldtype = TREE_TYPE (olddecl);
+  *newtypep = newtype = TREE_TYPE (newdecl);
+  if (oldtype == error_mark_node || newtype == error_mark_node)
+    return false;
 
-  /* New decl is completely inconsistent with the old one =>
-     tell caller to replace the old one.
-     This is always an error except in the case of shadowing a builtin.  */
+  /* Two different categories of symbol altogether.  This is an error
+     unless OLDDECL is a builtin.  OLDDECL will be discarded in any case.  */
   if (TREE_CODE (olddecl) != TREE_CODE (newdecl))
     {
+      if (TREE_CODE (olddecl) != FUNCTION_DECL
+          || !DECL_BUILT_IN (olddecl) || !C_DECL_INVISIBLE (olddecl))
+       {
+         error ("%J'%D' redeclared as different kind of symbol",
+                newdecl, newdecl);
+         locate_old_decl (olddecl, error);
+       }
+      else if (TREE_PUBLIC (newdecl))
+       warning ("%Jbuilt-in function '%D' declared as non-function",
+                newdecl, newdecl);
+      else if (warn_shadow)
+       warning ("%Jshadowing built-in function '%D'",
+                newdecl, newdecl);
+      return false;
+    }
+
+  if (!comptypes (oldtype, newtype, COMPARE_STRICT))
+    {
       if (TREE_CODE (olddecl) == FUNCTION_DECL
-         && DECL_BUILT_IN (olddecl))
+         && DECL_BUILT_IN (olddecl) && C_DECL_INVISIBLE (olddecl))
        {
-         /* If you declare a built-in or predefined function name as static,
-            the old definition is overridden,
-            but optionally warn this was a bad choice of name.  */
-         if (!TREE_PUBLIC (newdecl))
+         /* Accept harmless mismatch in function types.
+            This is for the ffs and fprintf builtins.  */
+         tree trytype = match_builtin_function_types (newtype, oldtype);
+
+         if (trytype && comptypes (newtype, trytype, COMPARE_STRICT))
+           *oldtypep = oldtype = trytype;
+         else
            {
-             if (warn_shadow)
-               warning ("%Hshadowing built-in function '%D'",
-                         &DECL_SOURCE_LOCATION (newdecl), newdecl);
+             /* If types don't match for a built-in, throw away the
+                built-in.  No point in calling locate_old_decl here, it
+                won't print anything.  */
+             warning ("%Jconflicting types for built-in function '%D'",
+                      newdecl, newdecl);
+             return false;
            }
-         else
-           warning ("%Hbuilt-in function '%D' declared as non-function",
-                     &DECL_SOURCE_LOCATION (newdecl), newdecl);
+       }
+      else if (TREE_CODE (olddecl) == FUNCTION_DECL
+              && DECL_SOURCE_LINE (olddecl) == 0)
+       {
+         /* A conflicting function declaration for a predeclared
+            function that isn't actually built in.  Objective C uses
+            these.  The new declaration silently overrides everything
+            but the volatility (i.e. noreturn) indication.  See also
+            below.  FIXME: Make Objective C use normal builtins.  */
+         TREE_THIS_VOLATILE (newdecl) |= TREE_THIS_VOLATILE (olddecl);
+         return false;
+       }
+      /* Permit void foo (...) to match int foo (...) if the latter is
+        the definition and implicit int was used.  See
+        c-torture/compile/920625-2.c.  */
+      else if (TREE_CODE (newdecl) == FUNCTION_DECL && DECL_INITIAL (newdecl)
+              && TYPE_MAIN_VARIANT (TREE_TYPE (oldtype)) == void_type_node
+              && TYPE_MAIN_VARIANT (TREE_TYPE (newtype)) == integer_type_node
+              && C_FUNCTION_IMPLICIT_INT (newdecl))
+       {
+         pedwarn ("%Jconflicting types for '%D'", newdecl, newdecl);
+         /* Make sure we keep void as the return type.  */
+         TREE_TYPE (newdecl) = *newtypep = newtype = oldtype;
+         C_FUNCTION_IMPLICIT_INT (newdecl) = 0;
+         pedwarned = true;
        }
       else
        {
-         error ("%H'%D' redeclared as different kind of symbol",
-                 &DECL_SOURCE_LOCATION (newdecl), newdecl);
-         error ("%Hprevious declaration of '%D'",
-                 &DECL_SOURCE_LOCATION (olddecl), olddecl);
+         error ("%Jconflicting types for '%D'", newdecl, newdecl);
+         diagnose_arglist_conflict (newdecl, olddecl, newtype, oldtype);
+         locate_old_decl (olddecl, error);
+         return false;
        }
-
-      return 0;
     }
 
-  /* For real parm decl following a forward decl, return 1 so old decl
-     will be reused.  Only allow this to happen once.  */
-  if (types_match && TREE_CODE (newdecl) == PARM_DECL
-      && TREE_ASM_WRITTEN (olddecl) && ! TREE_ASM_WRITTEN (newdecl))
+  /* Redeclaration of a type is a constraint violation (6.7.2.3p1),
+     but silently ignore the redeclaration if either is in a system
+     header.  (Conflicting redeclarations were handled above.)  */
+  if (TREE_CODE (newdecl) == TYPE_DECL)
     {
-      TREE_ASM_WRITTEN (olddecl) = 0;
-      return 1;
+      if (DECL_IN_SYSTEM_HEADER (newdecl) || DECL_IN_SYSTEM_HEADER (olddecl))
+       return true;  /* Allow OLDDECL to continue in use.  */
+      
+      error ("%Jredefinition of typedef '%D'", newdecl, newdecl);
+      locate_old_decl (olddecl, error);
+      return false;
     }
 
-  /* The new declaration is the same kind of object as the old one.
-     The declarations may partially match.  Print warnings if they don't
-     match enough.  Ultimately, copy most of the information from the new
-     decl to the old one, and keep using the old one.  */
-
-  if (TREE_CODE (olddecl) == FUNCTION_DECL && DECL_BUILT_IN (olddecl))
+  /* Function declarations can either be 'static' or 'extern' (no
+     qualifier is equivalent to 'extern' - C99 6.2.2p5) and therefore
+     can never conflict with each other on account of linkage (6.2.2p4).
+     Multiple definitions are not allowed (6.9p3,5) but GCC permits
+     two definitions if one is 'extern inline' and one is not.  The non-
+     extern-inline definition supersedes the extern-inline definition.  */
+  else if (TREE_CODE (newdecl) == FUNCTION_DECL)
     {
-      /* A function declaration for a built-in function.  */
-      if (!TREE_PUBLIC (newdecl))
+      /* If you declare a built-in function name as static, or
+        define the built-in with an old-style definition (so we
+        can't validate the argument list) the built-in definition is
+        overridden, but optionally warn this was a bad choice of name.  */
+      if (DECL_BUILT_IN (olddecl)
+         && C_DECL_INVISIBLE (olddecl)
+         && (!TREE_PUBLIC (newdecl)
+             || (DECL_INITIAL (newdecl)
+                 && !TYPE_ARG_TYPES (TREE_TYPE (newdecl)))))
        {
-         /* If you declare a built-in function name as static, the
-            built-in definition is overridden,
-            but optionally warn this was a bad choice of name.  */
          if (warn_shadow)
-           warning ("%Hshadowing built-in function '%D'",
-                     &DECL_SOURCE_LOCATION (newdecl), newdecl);
+           warning ("%Jshadowing built-in function '%D'", newdecl, newdecl);
          /* Discard the old built-in function.  */
-         return 0;
+         return false;
        }
-      if (!types_match)
+      
+      if (DECL_INITIAL (newdecl))
        {
-         /* Accept harmless mismatch in function types.
-            This is for the ffs and fprintf builtins.  */
-         tree trytype = match_builtin_function_types (oldtype, newtype);
-
-         if (trytype)
+         if (DECL_INITIAL (olddecl)
+             && !(DECL_DECLARED_INLINE_P (olddecl)
+                  && DECL_EXTERNAL (olddecl)
+                  && !(DECL_DECLARED_INLINE_P (newdecl)
+                       && DECL_EXTERNAL (newdecl))))
            {
-             types_match = comptypes (newtype, trytype, comptype_flags);
-             if (types_match)
-               oldtype = trytype;
-             if (! different_binding_level)
-               TREE_TYPE (olddecl) = oldtype;
+             error ("%Jredefinition of '%D'", newdecl, newdecl);
+             locate_old_decl (olddecl, error);
+             return false;
            }
        }
-      if (!types_match)
+      /* If we have a prototype after an old-style function definition,
+        the argument types must be checked specially.  */
+      else if (DECL_INITIAL (olddecl)
+              && !TYPE_ARG_TYPES (oldtype) && TYPE_ARG_TYPES (newtype)
+              && TYPE_ACTUAL_ARG_TYPES (oldtype)
+              && !validate_proto_after_old_defn (newdecl, newtype, oldtype))
        {
-         /* If types don't match for a built-in, throw away the built-in.  */
-         warning ("%Hconflicting types for built-in function '%D'",
-                   &DECL_SOURCE_LOCATION (newdecl), newdecl);
-         return 0;
+         locate_old_decl (olddecl, error);
+         return false;
+       }
+      /* Mismatched non-static and static is considered poor style.
+         We only diagnose static then non-static if -Wtraditional,
+        because it is the most convenient way to get some effects
+        (see e.g.  what unwind-dw2-fde-glibc.c does to the definition
+        of _Unwind_Find_FDE in unwind-dw2-fde.c).  Revisit?  */
+      if (TREE_PUBLIC (olddecl) && !TREE_PUBLIC (newdecl))
+       {
+         /* A static function declaration for a predeclared function
+            that isn't actually built in, silently overrides the
+            default.  Objective C uses these.  See also above.
+            FIXME: Make Objective C use normal builtins.  */
+         if (TREE_CODE (olddecl) == FUNCTION_DECL
+             && DECL_SOURCE_LINE (olddecl) == 0)
+           return false;
+         else
+           {
+             warning ("%Jstatic declaration of '%D' follows "
+                      "non-static declaration", newdecl, newdecl);
+             warned = true;
+           }
+       }
+      else if (TREE_PUBLIC (newdecl) && !TREE_PUBLIC (olddecl)
+              && warn_traditional)
+       {
+         warning ("%Jnon-static declaration of '%D' follows "
+                  "static declaration", newdecl, newdecl);
+         warned = true;
        }
     }
-  else if (TREE_CODE (olddecl) == FUNCTION_DECL
-          && DECL_SOURCE_LINE (olddecl) == 0)
+  else if (TREE_CODE (newdecl) == VAR_DECL)
     {
-      /* A function declaration for a predeclared function
-        that isn't actually built in.  */
-      if (!TREE_PUBLIC (newdecl))
+      /* Only variables can be thread-local, and all declarations must
+        agree on this property.  */
+      if (DECL_THREAD_LOCAL (newdecl) != DECL_THREAD_LOCAL (olddecl))
        {
-         /* If you declare it as static, the
-            default definition is overridden.  */
-         return 0;
+         if (DECL_THREAD_LOCAL (newdecl))
+           error ("%Jthread-local declaration of '%D' follows "
+                  "non-thread-local declaration", newdecl, newdecl);
+         else
+           error ("%Jnon-thread-local declaration of '%D' follows "
+                  "thread-local declaration", newdecl, newdecl);
+
+         locate_old_decl (olddecl, error);
+         return false;
        }
-      else if (!types_match)
+
+      /* Multiple initialized definitions are not allowed (6.9p3,5).  */
+      if (DECL_INITIAL (newdecl) && DECL_INITIAL (olddecl))
        {
-         /* If the types don't match, preserve volatility indication.
-            Later on, we will discard everything else about the
-            default declaration.  */
-         TREE_THIS_VOLATILE (newdecl) |= TREE_THIS_VOLATILE (olddecl);
+         error ("%Jredefinition of '%D'", newdecl, newdecl);
+         locate_old_decl (olddecl, error);
+         return false;
        }
-    }
-  /* Permit char *foo () to match void *foo (...) if not pedantic,
-     if one of them came from a system header file.  */
-  else if (!types_match
-          && TREE_CODE (olddecl) == FUNCTION_DECL
-          && TREE_CODE (newdecl) == FUNCTION_DECL
-          && TREE_CODE (TREE_TYPE (oldtype)) == POINTER_TYPE
-          && TREE_CODE (TREE_TYPE (newtype)) == POINTER_TYPE
-          && (DECL_IN_SYSTEM_HEADER (olddecl)
-              || DECL_IN_SYSTEM_HEADER (newdecl))
-          && ((TYPE_MAIN_VARIANT (TREE_TYPE (TREE_TYPE (newtype))) == void_type_node
-               && TYPE_ARG_TYPES (oldtype) == 0
-               && self_promoting_args_p (TYPE_ARG_TYPES (newtype))
-               && TREE_TYPE (TREE_TYPE (oldtype)) == char_type_node)
-              ||
-              (TREE_TYPE (TREE_TYPE (newtype)) == char_type_node
-               && TYPE_ARG_TYPES (newtype) == 0
-               && self_promoting_args_p (TYPE_ARG_TYPES (oldtype))
-               && TYPE_MAIN_VARIANT (TREE_TYPE (TREE_TYPE (oldtype))) == void_type_node)))
-    {
-      if (pedantic)
-       pedwarn ("%Hconflicting types for '%D'",
-                 &DECL_SOURCE_LOCATION (newdecl), newdecl);
-      /* Make sure we keep void * as ret type, not char *.  */
-      if (TYPE_MAIN_VARIANT (TREE_TYPE (TREE_TYPE (oldtype))) == void_type_node)
-       TREE_TYPE (newdecl) = newtype = oldtype;
-
-      /* Set DECL_IN_SYSTEM_HEADER, so that if we see another declaration
-        we will come back here again.  */
-      DECL_IN_SYSTEM_HEADER (newdecl) = 1;
-    }
-  /* Permit void foo (...) to match int foo (...) if the latter is the
-     definition and implicit int was used.  See c-torture/compile/920625-2.c.  */
-  else if (!types_match        && new_is_definition
-          && TREE_CODE (olddecl) == FUNCTION_DECL
-          && TREE_CODE (newdecl) == FUNCTION_DECL
-          && TYPE_MAIN_VARIANT (TREE_TYPE (oldtype)) == void_type_node
-          && TYPE_MAIN_VARIANT (TREE_TYPE (newtype)) == integer_type_node
-          && C_FUNCTION_IMPLICIT_INT (newdecl))
-    {
-      pedwarn ("%Hconflicting types for '%D'",
-               &DECL_SOURCE_LOCATION (newdecl), newdecl);
-      /* Make sure we keep void as the return type.  */
-      TREE_TYPE (newdecl) = newtype = oldtype;
-      C_FUNCTION_IMPLICIT_INT (newdecl) = 0;
-    }
-  else if (!types_match
-          /* Permit char *foo (int, ...); followed by char *foo ();
-             if not pedantic.  */
-          && ! (TREE_CODE (olddecl) == FUNCTION_DECL
-                && ! pedantic
-                /* Return types must still match.  */
-                && comptypes (TREE_TYPE (oldtype),
-                              TREE_TYPE (newtype), comptype_flags)
-                && TYPE_ARG_TYPES (newtype) == 0))
-    {
-      error ("%Hconflicting types for '%D'",
-             &DECL_SOURCE_LOCATION (newdecl), newdecl);
-      /* Check for function type mismatch
-        involving an empty arglist vs a nonempty one.  */
-      if (TREE_CODE (olddecl) == FUNCTION_DECL
-         && comptypes (TREE_TYPE (oldtype),
-                       TREE_TYPE (newtype), comptype_flags)
-         && ((TYPE_ARG_TYPES (oldtype) == 0
-              && DECL_INITIAL (olddecl) == 0)
-             ||
-             (TYPE_ARG_TYPES (newtype) == 0
-              && DECL_INITIAL (newdecl) == 0)))
+
+      /* Objects declared at file scope: if at least one is 'extern',
+        it's fine (6.2.2p4); otherwise the linkage must agree (6.2.2p7).  */
+      if (DECL_FILE_SCOPE_P (newdecl))
        {
-         /* Classify the problem further.  */
-         tree t = TYPE_ARG_TYPES (oldtype);
-         if (t == 0)
-           t = TYPE_ARG_TYPES (newtype);
-         for (; t; t = TREE_CHAIN (t))
+         if (!DECL_EXTERNAL (newdecl)
+             && !DECL_EXTERNAL (olddecl)
+             && TREE_PUBLIC (newdecl) != TREE_PUBLIC (olddecl))
            {
-             tree type = TREE_VALUE (t);
-
-             if (TREE_CHAIN (t) == 0
-                 && TYPE_MAIN_VARIANT (type) != void_type_node)
-               {
-                 error ("a parameter list with an ellipsis can't match an empty parameter name list declaration");
-                 break;
-               }
+             if (TREE_PUBLIC (newdecl))
+               error ("%Jnon-static declaration of '%D' follows "
+                      "static declaration", newdecl, newdecl);
+             else
+               error ("%Jstatic declaration of '%D' follows "
+                      "non-static declaration", newdecl, newdecl);
 
-             if (c_type_promotes_to (type) != type)
-               {
-                 error ("an argument type that has a default promotion can't match an empty parameter name list declaration");
-                 break;
-               }
+             locate_old_decl (olddecl, error);
+             return false;
            }
        }
-      if (C_DECL_IMPLICIT (olddecl))
-       error ("%Hprevious implicit declaration of '%D'",
-               &DECL_SOURCE_LOCATION (olddecl), olddecl);
-      else
-       error ("%Hprevious declaration of '%D'",
-               &DECL_SOURCE_LOCATION (olddecl), olddecl);
+      /* Two objects with the same name declared at the same block
+        scope must both be external references (6.7p3).  */
+      else if (DECL_CONTEXT (newdecl) == DECL_CONTEXT (olddecl)
+              && (!DECL_EXTERNAL (newdecl) || !DECL_EXTERNAL (olddecl)))
+       {
+         if (DECL_EXTERNAL (newdecl))
+           error ("%Jextern declaration of '%D' follows "
+                  "declaration with no linkage", newdecl, newdecl);
+         else if (DECL_EXTERNAL (olddecl))
+           error ("%Jdeclaration of '%D' with no linkage follows "
+                  "extern declaration", newdecl, newdecl);
+         else
+           error ("%Jredeclaration of '%D' with no linkage",
+                  newdecl, newdecl);
 
-      /* This is safer because the initializer might contain references
-        to variables that were declared between olddecl and newdecl. This
-        will make the initializer invalid for olddecl in case it gets
-        assigned to olddecl below.  */
-      if (TREE_CODE (newdecl) == VAR_DECL)
-       DECL_INITIAL (newdecl) = 0;
-    }
-  /* TLS cannot follow non-TLS declaration.  */
-  else if (TREE_CODE (olddecl) == VAR_DECL && TREE_CODE (newdecl) == VAR_DECL
-          && !DECL_THREAD_LOCAL (olddecl) && DECL_THREAD_LOCAL (newdecl))
-    {
-      error ("%Hthread-local declaration of '%D' follows non thread-local "
-             "declaration", &DECL_SOURCE_LOCATION (newdecl), newdecl);
-      error ("%Hprevious declaration of '%D'",
-             &DECL_SOURCE_LOCATION (olddecl), olddecl);
+         locate_old_decl (olddecl, error);
+         return false;
+       }
     }
-  /* non-TLS declaration cannot follow TLS declaration.  */
-  else if (TREE_CODE (olddecl) == VAR_DECL && TREE_CODE (newdecl) == VAR_DECL
-          && DECL_THREAD_LOCAL (olddecl) && !DECL_THREAD_LOCAL (newdecl))
+
+  /* warnings */
+  /* All decls must agree on a non-default visibility.  */
+  if (DECL_VISIBILITY (newdecl) != VISIBILITY_DEFAULT
+      && DECL_VISIBILITY (olddecl) != VISIBILITY_DEFAULT
+      && DECL_VISIBILITY (newdecl) != DECL_VISIBILITY (olddecl))
     {
-      error ("%Hnon thread-local declaration of '%D' follows "
-             "thread-local declaration",
-             &DECL_SOURCE_LOCATION (newdecl), newdecl);
-      error ("%Hprevious declaration of '%D'",
-             &DECL_SOURCE_LOCATION (olddecl), olddecl);
+      warning ("%Jredeclaration of '%D' with different visibility "
+              "(old visibility preserved)", newdecl, newdecl);
+      warned = true;
     }
-  else
+
+  if (TREE_CODE (newdecl) == FUNCTION_DECL)
     {
-      errmsg = redeclaration_error_message (newdecl, olddecl);
-      if (errmsg)
+      /* Diagnose inline __attribute__ ((noinline)) which is silly.  */
+      if (DECL_DECLARED_INLINE_P (newdecl)
+         && lookup_attribute ("noinline", DECL_ATTRIBUTES (olddecl)))
        {
-          const location_t *locus = &DECL_SOURCE_LOCATION (newdecl);
-         switch (errmsg)
-           {
-           case 1:
-             error ("%Hredefinition of '%D'", locus, newdecl);
-             break;
-           case 2:
-             error ("%Hredeclaration of '%D'", locus, newdecl);
-             break;
-           case 3:
-             error ("%Hconflicting declarations of '%D'", locus, newdecl);
-             break;
-           default:
-             abort ();
-           }
-
-          locus = &DECL_SOURCE_LOCATION (olddecl);
-          if (DECL_INITIAL (olddecl)
-              && current_scope == global_scope)
-            error ("%H'%D' previously defined here", locus, olddecl);
-          else
-            error ("%H'%D' previously declared here", locus, olddecl);
-         return 0;
+         warning ("%Jinline declaration of '%D' follows "
+                  "declaration with attribute noinline", newdecl, newdecl);
+         warned = true;
        }
-      else if (TREE_CODE (newdecl) == TYPE_DECL
-               && (DECL_IN_SYSTEM_HEADER (olddecl)
-                   || DECL_IN_SYSTEM_HEADER (newdecl)))
+      else if (DECL_DECLARED_INLINE_P (olddecl)
+              && lookup_attribute ("noinline", DECL_ATTRIBUTES (newdecl)))
        {
-          const location_t *locus = &DECL_SOURCE_LOCATION (newdecl);
-         warning ("%Hredefinition of '%D'", locus, newdecl);
-          locus = &DECL_SOURCE_LOCATION (olddecl);
-          if (DECL_INITIAL (olddecl)
-              && current_scope == global_scope)
-            warning ("%H'%D' previously defined here", locus, olddecl);
-          else
-            warning ("%H'%D' previously declared here", locus, olddecl);
+         warning ("%Jdeclaration of '%D' with attribute noinline follows "
+                  "inline declaration ", newdecl, newdecl);
+         warned = true;
        }
-      else if (TREE_CODE (olddecl) == FUNCTION_DECL
-              && DECL_INITIAL (olddecl) != 0
-              && TYPE_ARG_TYPES (oldtype) == 0
-              && TYPE_ARG_TYPES (newtype) != 0
-              && TYPE_ACTUAL_ARG_TYPES (oldtype) != 0)
+
+      /* Inline declaration after use or definition.
+        ??? Should we still warn about this now we have unit-at-a-time
+        mode and can get it right?  */
+      if (DECL_DECLARED_INLINE_P (newdecl) && !DECL_DECLARED_INLINE_P (olddecl))
        {
-         tree type, parm;
-         int nargs;
-         /* Prototype decl follows defn w/o prototype.  */
-
-         for (parm = TYPE_ACTUAL_ARG_TYPES (oldtype),
-              type = TYPE_ARG_TYPES (newtype),
-              nargs = 1;
-              ;
-              parm = TREE_CHAIN (parm), type = TREE_CHAIN (type), nargs++)
+         if (TREE_USED (olddecl))
            {
-             if (TYPE_MAIN_VARIANT (TREE_VALUE (parm)) == void_type_node
-                 && TYPE_MAIN_VARIANT (TREE_VALUE (type)) == void_type_node)
-               {
-                  const location_t *locus = &DECL_SOURCE_LOCATION (newdecl);
-                 warning ("%Hprototype for '%D' follows", locus, newdecl);
-                  locus = &DECL_SOURCE_LOCATION (olddecl);
-                 warning ("%Hnon-prototype definition here", locus);
-                 break;
-               }
-             if (TYPE_MAIN_VARIANT (TREE_VALUE (parm)) == void_type_node
-                 || TYPE_MAIN_VARIANT (TREE_VALUE (type)) == void_type_node)
-               {
-                  const location_t *locus = &DECL_SOURCE_LOCATION (newdecl);
-                 error ("%Hprototype for '%D' follows and number of "
-                         "arguments doesn't match", locus, newdecl);
-                  locus = &DECL_SOURCE_LOCATION (olddecl);
-                 error ("%Hnon-prototype definition here", locus);
-                 errmsg = 1;
-                 break;
-               }
-             /* Type for passing arg must be consistent
-                with that declared for the arg.  */
-             if (! comptypes (TREE_VALUE (parm), TREE_VALUE (type),
-                              comptype_flags))
-               {
-                  const location_t *locus = &DECL_SOURCE_LOCATION (newdecl);
-                 error ("%Hprototype for '%D' follows and argument %d "
-                         "doesn't match", locus, newdecl, nargs);
-                  locus = &DECL_SOURCE_LOCATION (olddecl);
-                 error ("%Hnon-prototype definition here", locus);
-                 errmsg = 1;
-                 break;
-               }
+             warning ("%J'%D' declared inline after being called",
+                      olddecl, olddecl);
+             warned = true;
+           }
+         else if (DECL_INITIAL (olddecl))
+           {
+             warning ("%J'%D' declared inline after its definition",
+                      olddecl, olddecl);
+             warned = true;
            }
        }
-      /* Warn about mismatches in various flags.  */
-      else
+    }
+  else /* VAR_DECL */
+    {
+      /* These bits are only type qualifiers when applied to objects.  */
+      if (TREE_THIS_VOLATILE (newdecl) != TREE_THIS_VOLATILE (olddecl))
        {
-         const location_t *locus = &DECL_SOURCE_LOCATION (newdecl);
-
-         /* Warn if function is now inline
-            but was previously declared not inline and has been called.  */
-         if (TREE_CODE (olddecl) == FUNCTION_DECL
-             && ! DECL_DECLARED_INLINE_P (olddecl)
-             && DECL_DECLARED_INLINE_P (newdecl)
-             && TREE_USED (olddecl))
-           warning ("%H'%D' declared inline after being called",
-                    locus, newdecl);
-         if (TREE_CODE (olddecl) == FUNCTION_DECL
-             && ! DECL_DECLARED_INLINE_P (olddecl)
-             && DECL_DECLARED_INLINE_P (newdecl)
-             && DECL_INITIAL (olddecl) != 0)
-           warning ("%H'%D' declared inline after its definition",
-                    locus, newdecl);
-
-         /* If pedantic, warn when static declaration follows a non-static
-            declaration.  Otherwise, do so only for functions.  */
-         if ((pedantic || TREE_CODE (olddecl) == FUNCTION_DECL)
-             && TREE_PUBLIC (olddecl)
-             && !TREE_PUBLIC (newdecl))
-           warning ("%Hstatic declaration for '%D' follows non-static",
-                    locus, newdecl);
-
-         /* If warn_traditional, warn when a non-static function
-            declaration follows a static one.  */
-         if (warn_traditional && !in_system_header
-             && TREE_CODE (olddecl) == FUNCTION_DECL
-             && !TREE_PUBLIC (olddecl)
-             && TREE_PUBLIC (newdecl))
-           warning ("%Hnon-static declaration for '%D' follows static",
-                    locus, newdecl);
-
-         /* Warn when const declaration follows a non-const
-            declaration, but not for functions.  */
-         if (TREE_CODE (olddecl) != FUNCTION_DECL
-             && !TREE_READONLY (olddecl)
-             && TREE_READONLY (newdecl))
-           warning ("%Hconst declaration for '%D' follows non-const",
-                    locus, newdecl);
-         /* These bits are logically part of the type, for variables.
-            But not for functions
-            (where qualifiers are not valid ANSI anyway).  */
-         else if (pedantic && TREE_CODE (olddecl) != FUNCTION_DECL
-             && (TREE_READONLY (newdecl) != TREE_READONLY (olddecl)
-                 || TREE_THIS_VOLATILE (newdecl) != TREE_THIS_VOLATILE (olddecl)))
-           pedwarn ("%Htype qualifiers for '%D' conflict with previous "
-                    "declaration", locus, newdecl);
+         if (TREE_THIS_VOLATILE (newdecl))
+           pedwarn ("%Jvolatile declaration of '%D' follows "
+                    "non-volatile declaration", newdecl, newdecl);
+         else
+           pedwarn ("%Jnon-volatile declaration of '%D' follows "
+                    "volatile declaration", newdecl, newdecl);
+         pedwarned = true;
+       }
+      if (TREE_READONLY (newdecl) != TREE_READONLY (olddecl))
+       {
+         if (TREE_READONLY (newdecl))
+           pedwarn ("%Jconst declaration of '%D' follows "
+                    "non-const declaration", newdecl, newdecl);
+         else
+           pedwarn ("%Jnon-const declaration of '%D' follows "
+                    "const declaration", newdecl, newdecl);
+         pedwarned = true;
        }
     }
 
-  /* Optionally warn about more than one declaration for the same name.  */
-  if (errmsg == 0 && warn_redundant_decls && DECL_SOURCE_LINE (olddecl) != 0
-      /* Don't warn about a function declaration
-        followed by a definition.  */
-      && !(TREE_CODE (newdecl) == FUNCTION_DECL && DECL_INITIAL (newdecl) != 0
-          && DECL_INITIAL (olddecl) == 0)
-      /* Don't warn about extern decl followed by (tentative) definition.  */
-      && !(DECL_EXTERNAL (olddecl) && ! DECL_EXTERNAL (newdecl)))
+  /* Optional warning for completely redundant decls.  */
+  if (!warned && !pedwarned
+      && warn_redundant_decls
+      /* Don't warn about a function declaration followed by a
+        definition.  */
+    && !(TREE_CODE (newdecl) == FUNCTION_DECL
+        && DECL_INITIAL (newdecl) && !DECL_INITIAL (olddecl))
+    /* Don't warn about an extern followed by a definition.  */
+    && !(DECL_EXTERNAL (olddecl) && !DECL_EXTERNAL (newdecl)))
     {
-      warning ("%Hredundant redeclaration of '%D' in same scope",
-               &DECL_SOURCE_LOCATION (newdecl), newdecl);
-      warning ("%Hprevious declaration of '%D'",
-               &DECL_SOURCE_LOCATION (olddecl), olddecl);
+      warning ("%Jredundant redeclaration of '%D'", newdecl, newdecl);
+      warned = true;
     }
 
-  /* Copy all the DECL_... slots specified in the new decl
-     except for any that we copy here from the old type.
+  /* Report location of previous decl/defn in a consistent manner.  */
+  if (warned || pedwarned)
+    locate_old_decl (olddecl, pedwarned ? pedwarn : warning);
 
-     Past this point, we don't change OLDTYPE and NEWTYPE
-     even if we change the types of NEWDECL and OLDDECL.  */
+  return true;
+}
 
-  if (types_match)
-    {
-      /* When copying info to olddecl, we store into write_olddecl
-        instead.  This allows us to avoid modifying olddecl when
-        different_binding_level is true.  */
-      tree write_olddecl = different_binding_level ? newdecl : olddecl;
+/* Subroutine of duplicate_decls.  NEWDECL has been found to be
+   consistent with OLDDECL, but carries new information.  Merge the
+   new information into OLDDECL.  This function issues no
+   diagnostics.  */
 
-      /* Merge the data types specified in the two decls.  */
-      if (TREE_CODE (newdecl) != FUNCTION_DECL || !DECL_BUILT_IN (olddecl))
-       {
-         if (different_binding_level)
-           {
-             if (TYPE_ARG_TYPES (oldtype) != 0
-                 && TYPE_ARG_TYPES (newtype) == 0)
-               TREE_TYPE (newdecl) = common_type (newtype, oldtype);
-             else
-               TREE_TYPE (newdecl)
-                 = build_type_attribute_variant
-                   (newtype,
-                    merge_attributes (TYPE_ATTRIBUTES (newtype),
-                                      TYPE_ATTRIBUTES (oldtype)));
-           }
-         else
-           TREE_TYPE (newdecl)
-             = TREE_TYPE (olddecl)
-               = common_type (newtype, oldtype);
-       }
+static void
+merge_decls (tree newdecl, tree olddecl, tree newtype, tree oldtype)
+{
+  int new_is_definition = (TREE_CODE (newdecl) == FUNCTION_DECL
+                          && DECL_INITIAL (newdecl) != 0);
 
-      /* Lay the type out, unless already done.  */
-      if (oldtype != TREE_TYPE (newdecl))
-       {
-         if (TREE_TYPE (newdecl) != error_mark_node)
-           layout_type (TREE_TYPE (newdecl));
-         if (TREE_CODE (newdecl) != FUNCTION_DECL
-             && TREE_CODE (newdecl) != TYPE_DECL
-             && TREE_CODE (newdecl) != CONST_DECL)
-           layout_decl (newdecl, 0);
-       }
-      else
-       {
-         /* Since the type is OLDDECL's, make OLDDECL's size go with.  */
-         DECL_SIZE (newdecl) = DECL_SIZE (olddecl);
-         DECL_SIZE_UNIT (newdecl) = DECL_SIZE_UNIT (olddecl);
-         DECL_MODE (newdecl) = DECL_MODE (olddecl);
-         if (TREE_CODE (olddecl) != FUNCTION_DECL)
-           if (DECL_ALIGN (olddecl) > DECL_ALIGN (newdecl))
-             {
-               DECL_ALIGN (newdecl) = DECL_ALIGN (olddecl);
-               DECL_USER_ALIGN (newdecl) |= DECL_ALIGN (olddecl);
-             }
-       }
+  /* For real parm decl following a forward decl, return 1 so old decl
+     will be reused.  Only allow this to happen once.  */
+  if (TREE_CODE (newdecl) == PARM_DECL
+      && TREE_ASM_WRITTEN (olddecl) && ! TREE_ASM_WRITTEN (newdecl))
+    {
+      TREE_ASM_WRITTEN (olddecl) = 0;
+      return;
+    }
 
-      /* Keep the old rtl since we can safely use it.  */
-      COPY_DECL_RTL (olddecl, newdecl);
+  DECL_ATTRIBUTES (newdecl)
+    = (*targetm.merge_decl_attributes) (olddecl, newdecl);
 
-      /* Merge the type qualifiers.  */
-      if (TREE_READONLY (newdecl))
-       TREE_READONLY (write_olddecl) = 1;
+  /* Merge the data types specified in the two decls.  */
+  TREE_TYPE (newdecl)
+    = TREE_TYPE (olddecl)
+    = common_type (newtype, oldtype);
 
-      if (TREE_THIS_VOLATILE (newdecl))
-       {
-         TREE_THIS_VOLATILE (write_olddecl) = 1;
-         if (TREE_CODE (newdecl) == VAR_DECL
-             /* If an automatic variable is re-declared in the same
-                function scope, but the old declaration was not
-                volatile, make_var_volatile() would crash because the
-                variable would have been assigned to a pseudo, not a
-                MEM.  Since this duplicate declaration is invalid
-                anyway, we just skip the call.  */
-             && errmsg == 0)
-           make_var_volatile (newdecl);
-       }
+  /* Lay the type out, unless already done.  */
+  if (oldtype != TREE_TYPE (newdecl))
+    {
+      if (TREE_TYPE (newdecl) != error_mark_node)
+       layout_type (TREE_TYPE (newdecl));
+      if (TREE_CODE (newdecl) != FUNCTION_DECL
+         && TREE_CODE (newdecl) != TYPE_DECL
+         && TREE_CODE (newdecl) != CONST_DECL)
+       layout_decl (newdecl, 0);
+    }
+  else
+    {
+      /* Since the type is OLDDECL's, make OLDDECL's size go with.  */
+      DECL_SIZE (newdecl) = DECL_SIZE (olddecl);
+      DECL_SIZE_UNIT (newdecl) = DECL_SIZE_UNIT (olddecl);
+      DECL_MODE (newdecl) = DECL_MODE (olddecl);
+      if (TREE_CODE (olddecl) != FUNCTION_DECL)
+       if (DECL_ALIGN (olddecl) > DECL_ALIGN (newdecl))
+         {
+           DECL_ALIGN (newdecl) = DECL_ALIGN (olddecl);
+           DECL_USER_ALIGN (newdecl) |= DECL_ALIGN (olddecl);
+         }
+    }
 
-      /* Keep source location of definition rather than declaration.  */
-      /* When called with different_binding_level set, keep the old
-        information so that meaningful diagnostics can be given.  */
-      if (DECL_INITIAL (newdecl) == 0 && DECL_INITIAL (olddecl) != 0
-         && ! different_binding_level)
-       {
-         DECL_SOURCE_LINE (newdecl) = DECL_SOURCE_LINE (olddecl);
-         DECL_SOURCE_FILE (newdecl) = DECL_SOURCE_FILE (olddecl);
-       }
+  /* Keep the old rtl since we can safely use it.  */
+  COPY_DECL_RTL (olddecl, newdecl);
 
-      /* Merge the unused-warning information.  */
-      if (DECL_IN_SYSTEM_HEADER (olddecl))
-       DECL_IN_SYSTEM_HEADER (newdecl) = 1;
-      else if (DECL_IN_SYSTEM_HEADER (newdecl))
-       DECL_IN_SYSTEM_HEADER (write_olddecl) = 1;
-
-      /* Merge the initialization information.  */
-      /* When called with different_binding_level set, don't copy over
-        DECL_INITIAL, so that we don't accidentally change function
-        declarations into function definitions.  */
-      if (DECL_INITIAL (newdecl) == 0 && ! different_binding_level)
-       DECL_INITIAL (newdecl) = DECL_INITIAL (olddecl);
-
-      /* Merge the section attribute.
-         We want to issue an error if the sections conflict but that must be
-        done later in decl_attributes since we are called before attributes
-        are assigned.  */
-      if (DECL_SECTION_NAME (newdecl) == NULL_TREE)
-       DECL_SECTION_NAME (newdecl) = DECL_SECTION_NAME (olddecl);
-
-      /* Copy the assembler name.
-        Currently, it can only be defined in the prototype.  */
-      COPY_DECL_ASSEMBLER_NAME (olddecl, newdecl);
-
-      if (TREE_CODE (newdecl) == FUNCTION_DECL)
-       {
-         DECL_STATIC_CONSTRUCTOR(newdecl) |= DECL_STATIC_CONSTRUCTOR(olddecl);
-         DECL_STATIC_DESTRUCTOR (newdecl) |= DECL_STATIC_DESTRUCTOR (olddecl);
-         DECL_NO_LIMIT_STACK (newdecl) |= DECL_NO_LIMIT_STACK (olddecl);
-         DECL_NO_INSTRUMENT_FUNCTION_ENTRY_EXIT (newdecl)
-           |= DECL_NO_INSTRUMENT_FUNCTION_ENTRY_EXIT (olddecl);
-         TREE_THIS_VOLATILE (newdecl) |= TREE_THIS_VOLATILE (olddecl);
-         TREE_READONLY (newdecl) |= TREE_READONLY (olddecl);
-         DECL_IS_MALLOC (newdecl) |= DECL_IS_MALLOC (olddecl);
-         DECL_IS_PURE (newdecl) |= DECL_IS_PURE (olddecl);
-       }
+  /* Merge the type qualifiers.  */
+  if (TREE_READONLY (newdecl))
+    TREE_READONLY (olddecl) = 1;
+
+  if (TREE_THIS_VOLATILE (newdecl))
+    {
+      TREE_THIS_VOLATILE (olddecl) = 1;
+      if (TREE_CODE (newdecl) == VAR_DECL)
+       make_var_volatile (newdecl);
     }
-  /* If cannot merge, then use the new type and qualifiers,
-     and don't preserve the old rtl.  */
-  else if (! different_binding_level)
+
+  /* Keep source location of definition rather than declaration.  */
+  if (DECL_INITIAL (newdecl) == 0 && DECL_INITIAL (olddecl) != 0)
+    DECL_SOURCE_LOCATION (newdecl) = DECL_SOURCE_LOCATION (olddecl);
+
+  /* Merge the unused-warning information.  */
+  if (DECL_IN_SYSTEM_HEADER (olddecl))
+    DECL_IN_SYSTEM_HEADER (newdecl) = 1;
+  else if (DECL_IN_SYSTEM_HEADER (newdecl))
+    DECL_IN_SYSTEM_HEADER (olddecl) = 1;
+
+  /* Merge the initialization information.  */
+   if (DECL_INITIAL (newdecl) == 0)
+    DECL_INITIAL (newdecl) = DECL_INITIAL (olddecl);
+
+  /* Merge the section attribute.
+     We want to issue an error if the sections conflict but that must be
+     done later in decl_attributes since we are called before attributes
+     are assigned.  */
+  if (DECL_SECTION_NAME (newdecl) == NULL_TREE)
+    DECL_SECTION_NAME (newdecl) = DECL_SECTION_NAME (olddecl);
+
+  /* Copy the assembler name.
+     Currently, it can only be defined in the prototype.  */
+  COPY_DECL_ASSEMBLER_NAME (olddecl, newdecl);
+
+  /* If either declaration has a nondefault visibility, use it.  */
+  if (DECL_VISIBILITY (olddecl) != VISIBILITY_DEFAULT)
+    DECL_VISIBILITY (newdecl) = DECL_VISIBILITY (olddecl);
+
+  if (TREE_CODE (newdecl) == FUNCTION_DECL)
     {
-      TREE_TYPE (olddecl) = TREE_TYPE (newdecl);
-      TREE_READONLY (olddecl) = TREE_READONLY (newdecl);
-      TREE_THIS_VOLATILE (olddecl) = TREE_THIS_VOLATILE (newdecl);
-      TREE_SIDE_EFFECTS (olddecl) = TREE_SIDE_EFFECTS (newdecl);
+      DECL_STATIC_CONSTRUCTOR(newdecl) |= DECL_STATIC_CONSTRUCTOR(olddecl);
+      DECL_STATIC_DESTRUCTOR (newdecl) |= DECL_STATIC_DESTRUCTOR (olddecl);
+      DECL_NO_LIMIT_STACK (newdecl) |= DECL_NO_LIMIT_STACK (olddecl);
+      DECL_NO_INSTRUMENT_FUNCTION_ENTRY_EXIT (newdecl)
+       |= DECL_NO_INSTRUMENT_FUNCTION_ENTRY_EXIT (olddecl);
+      TREE_THIS_VOLATILE (newdecl) |= TREE_THIS_VOLATILE (olddecl);
+      TREE_READONLY (newdecl) |= TREE_READONLY (olddecl);
+      DECL_IS_MALLOC (newdecl) |= DECL_IS_MALLOC (olddecl);
+      DECL_IS_PURE (newdecl) |= DECL_IS_PURE (olddecl);
     }
 
   /* Merge the storage class information.  */
@@ -1362,8 +1374,6 @@ duplicate_decls (tree newdecl, tree olddecl, int different_binding_level,
       TREE_PUBLIC (newdecl) &= TREE_PUBLIC (olddecl);
       /* This is since we don't automatically
         copy the attributes of NEWDECL into OLDDECL.  */
-      /* No need to worry about different_binding_level here because
-        then TREE_PUBLIC (newdecl) was true.  */
       TREE_PUBLIC (olddecl) = TREE_PUBLIC (newdecl);
       /* If this clears `static', clear it in the identifier too.  */
       if (! TREE_PUBLIC (olddecl))
@@ -1371,23 +1381,15 @@ duplicate_decls (tree newdecl, tree olddecl, int different_binding_level,
     }
   if (DECL_EXTERNAL (newdecl))
     {
-      if (! different_binding_level || different_tu)
-       {
-         /* Don't mess with these flags on local externs; they remain
-            external even if there's a declaration at file scope which
-            isn't.  */
-         TREE_STATIC (newdecl) = TREE_STATIC (olddecl);
-         DECL_EXTERNAL (newdecl) = DECL_EXTERNAL (olddecl);
-       }
+      TREE_STATIC (newdecl) = TREE_STATIC (olddecl);
+      DECL_EXTERNAL (newdecl) = DECL_EXTERNAL (olddecl);
+
       /* An extern decl does not override previous storage class.  */
       TREE_PUBLIC (newdecl) = TREE_PUBLIC (olddecl);
       if (! DECL_EXTERNAL (newdecl))
        {
          DECL_CONTEXT (newdecl) = DECL_CONTEXT (olddecl);
-         /* If we have two non-EXTERNAL file-scope decls that are
-            the same, only one of them should be written out.  */
-         if (different_tu)
-           TREE_ASM_WRITTEN (newdecl) = 1;
+         DECL_COMMON (newdecl) = DECL_COMMON (olddecl);
        }
     }
   else
@@ -1404,7 +1406,11 @@ duplicate_decls (tree newdecl, tree olddecl, int different_binding_level,
         been written out yet.  */
       if (new_is_definition && DECL_INITIAL (olddecl))
        {
-         if (TREE_USED (olddecl))
+         if (TREE_USED (olddecl)
+             /* In unit-at-a-time mode we never inline re-defined extern
+                inline functions.  */
+             && !flag_unit_at_a_time
+             && cgraph_function_possibly_inlined_p (olddecl))
            (*debug_hooks->outlining_inline_function) (olddecl);
 
          /* The new defn must not be inline.  */
@@ -1425,37 +1431,18 @@ duplicate_decls (tree newdecl, tree olddecl, int different_binding_level,
 
       if (DECL_BUILT_IN (olddecl))
        {
-         /* Get rid of any built-in function if new arg types don't match it
-            or if we have a function definition.  */
-         if (! types_match || new_is_definition)
-           {
-             if (! different_binding_level)
-               {
-                 TREE_TYPE (olddecl) = TREE_TYPE (newdecl);
-                 DECL_BUILT_IN_CLASS (olddecl) = NOT_BUILT_IN;
-               }
-           }
-         else
-           {
-             /* If redeclaring a builtin function, and not a definition,
-                it stays built in.  */
-             DECL_BUILT_IN_CLASS (newdecl) = DECL_BUILT_IN_CLASS (olddecl);
-             DECL_FUNCTION_CODE (newdecl) = DECL_FUNCTION_CODE (olddecl);
-           }
+         /* If redeclaring a builtin function, it stays built in.  */
+         DECL_BUILT_IN_CLASS (newdecl) = DECL_BUILT_IN_CLASS (olddecl);
+         DECL_FUNCTION_CODE (newdecl) = DECL_FUNCTION_CODE (olddecl);
        }
 
       /* Also preserve various other info from the definition.  */
       if (! new_is_definition)
        {
          DECL_RESULT (newdecl) = DECL_RESULT (olddecl);
-         /* When called with different_binding_level set, don't copy over
-            DECL_INITIAL, so that we don't accidentally change function
-            declarations into function definitions.  */
-         if (! different_binding_level)
-           DECL_INITIAL (newdecl) = DECL_INITIAL (olddecl);
-         DECL_SAVED_INSNS (newdecl) = DECL_SAVED_INSNS (olddecl);
+         DECL_INITIAL (newdecl) = DECL_INITIAL (olddecl);
+         DECL_STRUCT_FUNCTION (newdecl) = DECL_STRUCT_FUNCTION (olddecl);
          DECL_SAVED_TREE (newdecl) = DECL_SAVED_TREE (olddecl);
-         DECL_ESTIMATED_INSNS (newdecl) = DECL_ESTIMATED_INSNS (olddecl);
          DECL_ARGUMENTS (newdecl) = DECL_ARGUMENTS (olddecl);
 
          /* Set DECL_INLINE on the declaration if we've got a body
@@ -1464,9 +1451,7 @@ duplicate_decls (tree newdecl, tree olddecl, int different_binding_level,
            {
              DECL_INLINE (newdecl) = 1;
              DECL_ABSTRACT_ORIGIN (newdecl)
-               = (different_binding_level
-                  ? DECL_ORIGIN (olddecl)
-                  : DECL_ABSTRACT_ORIGIN (olddecl));
+               = DECL_ABSTRACT_ORIGIN (olddecl);
            }
        }
       else
@@ -1478,8 +1463,6 @@ duplicate_decls (tree newdecl, tree olddecl, int different_binding_level,
            DECL_INLINE (newdecl) = 1;
        }
     }
-  if (different_binding_level)
-    return 0;
 
   /* Copy most of the decl-specific fields of NEWDECL into OLDDECL.
      But preserve OLDDECL's DECL_UID.  */
@@ -1492,10 +1475,6 @@ duplicate_decls (tree newdecl, tree olddecl, int different_binding_level,
     DECL_UID (olddecl) = olddecl_uid;
   }
 
-  /* NEWDECL contains the merged attribute lists.
-     Update OLDDECL to be the same.  */
-  DECL_ATTRIBUTES (olddecl) = DECL_ATTRIBUTES (newdecl);
-
   /* If OLDDECL had its DECL_RTL instantiated, re-invoke make_decl_rtl
      so that encode_section_info has a chance to look at the new decl
      flags and attributes.  */
@@ -1504,10 +1483,28 @@ duplicate_decls (tree newdecl, tree olddecl, int different_binding_level,
          || (TREE_CODE (olddecl) == VAR_DECL
              && TREE_STATIC (olddecl))))
     make_decl_rtl (olddecl, NULL);
-
-  return 1;
 }
 
+/* Handle when a new declaration NEWDECL has the same name as an old
+   one OLDDECL in the same binding contour.  Prints an error message
+   if appropriate.
+
+   If safely possible, alter OLDDECL to look like NEWDECL, and return
+   true.  Otherwise, return false.  */
+
+static bool
+duplicate_decls (tree newdecl, tree olddecl)
+{
+  tree newtype, oldtype;
+
+  if (!diagnose_mismatched_decls (newdecl, olddecl, &newtype, &oldtype))
+    return false;
+
+  merge_decls (newdecl, olddecl, newtype, oldtype);
+  return true;
+}
+  
+\f
 /* Return any external DECL associated with ID, whether or not it is
    currently in scope.  */
 
@@ -1547,8 +1544,6 @@ record_external_decl (tree decl)
 static void
 warn_if_shadowing (tree x, tree old)
 {
-  const char *name;
-
   /* Nothing to shadow?  */
   if (old == 0
       /* Shadow warnings not wanted?  */
@@ -1565,13 +1560,14 @@ warn_if_shadowing (tree x, tree old)
       || (TREE_CODE (x) == PARM_DECL && current_scope->outer->parm_flag))
     return;
 
-  name = IDENTIFIER_POINTER (DECL_NAME (x));
   if (TREE_CODE (old) == PARM_DECL)
-    shadow_warning (SW_PARAM, name, old);
+    warning ("%Jdeclaration of '%D' shadows a parameter", x, x);
   else if (DECL_FILE_SCOPE_P (old))
-    shadow_warning (SW_GLOBAL, name, old);
+    warning ("%Jdeclaration of '%D' shadows a global declaration", x, x);
   else
-    shadow_warning (SW_LOCAL, name, old);
+    warning ("%Jdeclaration of '%D' shadows a previous local", x, x);
+
+  warning ("%Jshadowed declaration is here", old);
 }
 
 
@@ -1672,7 +1668,7 @@ pushdecl (tree x)
     DECL_CONTEXT (x) = current_file_decl;
   else
     DECL_CONTEXT (x) = current_function_decl;
-  
+
   if (name)
     {
       tree old;
@@ -1685,7 +1681,7 @@ pushdecl (tree x)
                 IDENTIFIER_POINTER (name));
 
       old = lookup_name_current_level (name);
-      if (old && duplicate_decls (x, old, 0, false))
+      if (old && duplicate_decls (x, old))
        {
          /* For PARM_DECLs, old may be a forward declaration.
             If so, we want to remove it from its old location
@@ -1707,14 +1703,13 @@ pushdecl (tree x)
       if (DECL_EXTERNAL (x) || scope == global_scope)
        {
          /* Find and check against a previous, not-in-scope, external
-            decl for this identifier.  (C99 s???: If two declarations
-            with external linkage, referring to the same object, have
-            incompatible types, the behavior is undefined).  */
-         tree ext = any_external_decl (name);
+            decl for this identifier.  (C99 6.2.7p2: All declarations
+            that refer to the same object or function shall have
+            compatible type; otherwise, the behavior is undefined.)  */
+         tree ext = any_external_decl (name);
          if (ext)
            {
-             if (duplicate_decls (x, ext, scope != global_scope, 
-                                  false))
+             if (duplicate_decls (x, ext))
                x = copy_node (ext);
            }
          else
@@ -1763,7 +1758,8 @@ pushdecl (tree x)
          if ((TREE_CODE (element) == RECORD_TYPE
               || TREE_CODE (element) == UNION_TYPE)
              && (TREE_CODE (x) != TYPE_DECL
-                 || TREE_CODE (TREE_TYPE (x)) == ARRAY_TYPE))
+                 || TREE_CODE (TREE_TYPE (x)) == ARRAY_TYPE)
+             && !COMPLETE_TYPE_P (element))
            C_TYPE_INCOMPLETE_VARS (element)
              = tree_cons (NULL_TREE, x, C_TYPE_INCOMPLETE_VARS (element));
        }
@@ -1820,8 +1816,7 @@ implicitly_declare (tree functionid)
        {
          implicit_decl_warning (DECL_NAME (decl));
          if (! DECL_FILE_SCOPE_P (decl))
-           warning ("%Hprevious declaration of '%D'",
-                     &DECL_SOURCE_LOCATION (decl), decl);
+           warning ("%Jprevious declaration of '%D'", decl, decl);
          C_DECL_IMPLICIT (decl) = 1;
        }
       /* If this function is global, then it must already be in the
@@ -1867,69 +1862,6 @@ implicit_decl_warning (tree id)
     warning ("implicit declaration of function `%s'", name);
 }
 
-/* Return zero if the declaration NEWDECL is valid
-   when the declaration OLDDECL (assumed to be for the same name)
-   has already been seen.
-   Otherwise return 1 if NEWDECL is a redefinition, 2 if it is a redeclaration,
-   and 3 if it is a conflicting declaration.  */
-
-static int
-redeclaration_error_message (tree newdecl, tree olddecl)
-{
-  if (TREE_CODE (newdecl) == TYPE_DECL)
-    {
-      /* Do not complain about type redeclarations where at least one
-        declaration was in a system header.  */
-      if (DECL_IN_SYSTEM_HEADER (olddecl) || DECL_IN_SYSTEM_HEADER (newdecl))
-       return 0;
-      return 1;
-    }
-  else if (TREE_CODE (newdecl) == FUNCTION_DECL)
-    {
-      /* Declarations of functions can insist on internal linkage
-        but they can't be inconsistent with internal linkage,
-        so there can be no error on that account.
-        However defining the same name twice is no good.  */
-      if (DECL_INITIAL (olddecl) != 0 && DECL_INITIAL (newdecl) != 0
-         /* However, defining once as extern inline and a second
-            time in another way is ok.  */
-         && ! (DECL_DECLARED_INLINE_P (olddecl) && DECL_EXTERNAL (olddecl)
-              && ! (DECL_DECLARED_INLINE_P (newdecl)
-                    && DECL_EXTERNAL (newdecl))))
-       return 1;
-      return 0;
-    }
-  else if (DECL_FILE_SCOPE_P (newdecl))
-    {
-      /* Objects declared at file scope:  */
-      /* If at least one is a reference, it's ok.  */
-      if (DECL_EXTERNAL (newdecl) || DECL_EXTERNAL (olddecl))
-       return 0;
-      /* Reject two definitions.  */
-      if (DECL_INITIAL (olddecl) != 0 && DECL_INITIAL (newdecl) != 0)
-       return 1;
-      /* Now we have two tentative defs, or one tentative and one real def.  */
-      /* Insist that the linkage match.  */
-      if (TREE_PUBLIC (olddecl) != TREE_PUBLIC (newdecl))
-       return 3;
-      return 0;
-    }
-  else if (current_scope->parm_flag
-          && TREE_ASM_WRITTEN (olddecl) && !TREE_ASM_WRITTEN (newdecl))
-    return 0;
-  else
-    {
-      /* Newdecl has block scope.  If olddecl has block scope also, then
-        reject two definitions, and reject a definition together with an
-        external reference.  Otherwise, it is OK, because newdecl must
-        be an extern reference to olddecl.  */
-      if (!(DECL_EXTERNAL (newdecl) && DECL_EXTERNAL (olddecl))
-         && DECL_CONTEXT (newdecl) == DECL_CONTEXT (olddecl))
-       return 2;
-      return 0;
-    }
-}
-
 /* Issue an error message for a reference to an undeclared variable
    ID, including a reference to a builtin outside of function-call
    context.  Establish a binding of the identifier to error_mark_node
@@ -2056,8 +1988,7 @@ declare_label (tree name)
     if (dup == label)
       {
        error ("duplicate label declaration `%s'", IDENTIFIER_POINTER (name));
-       error ("%Hthis is a previous declaration",
-              &DECL_SOURCE_LOCATION (dup));
+       error ("%Jthis is a previous declaration", dup);
 
        /* Just use the previous declaration.  */
        return dup;
@@ -2092,12 +2023,11 @@ define_label (location_t location, tree name)
          || (DECL_CONTEXT (label) != current_function_decl
              && C_DECLARED_LABEL_FLAG (label))))
     {
-      location_t *prev_loc = &DECL_SOURCE_LOCATION (label);
       error ("%Hduplicate label `%D'", &location, label);
       if (DECL_INITIAL (label))
-       error ("%H`%D' previously defined here", prev_loc, label);
+       error ("%J`%D' previously defined here", label, label);
       else
-       error ("%H`%D' previously declared here", prev_loc, label);
+       error ("%J`%D' previously declared here", label, label);
       return 0;
     }
   else if (label && DECL_CONTEXT (label) == current_function_decl)
@@ -2126,12 +2056,13 @@ define_label (location_t location, tree name)
   return label;
 }
 \f
-/* Return the list of declarations of the current scope.  */
+/* Return the list of declarations of the current scope.
+   This hook is optional and not implemented for C.  */
 
 tree
 getdecls (void)
 {
-  return current_scope->names;
+  return 0;
 }
 
 \f
@@ -2247,7 +2178,7 @@ c_init_decl_processing (void)
   tree endlink;
   tree ptr_ftype_void, ptr_ftype_ptr;
   location_t save_loc = input_location;
-  
+
   /* Adds some ggc roots, and reserved words for c-parse.in.  */
   c_parse_init ();
 
@@ -2287,7 +2218,7 @@ c_init_decl_processing (void)
 
   input_location = save_loc;
 
-  pedantic_lvalues = pedantic;
+  pedantic_lvalues = true;
 
   make_fname_decl = c_make_fname_decl;
   start_fname_decls ();
@@ -2520,7 +2451,8 @@ groktypename (tree typename)
 
   split_specs_attrs (TREE_PURPOSE (typename), &specs, &attrs);
 
-  typename = grokdeclarator (TREE_VALUE (typename), specs, TYPENAME, 0);
+  typename = grokdeclarator (TREE_VALUE (typename), specs, TYPENAME, 0,
+                            NULL);
 
   /* Apply attributes.  */
   decl_attributes (&typename, attrs, 0);
@@ -2537,7 +2469,7 @@ groktypename_in_parm_context (tree typename)
     return typename;
   return grokdeclarator (TREE_VALUE (typename),
                         TREE_PURPOSE (typename),
-                        PARM, 0);
+                        PARM, 0, NULL);
 }
 
 /* Decode a declarator in an ordinary declaration or data definition.
@@ -2567,14 +2499,13 @@ start_decl (tree declarator, tree declspecs, int initialized, tree attributes)
     deprecated_state = DEPRECATED_SUPPRESS;
 
   decl = grokdeclarator (declarator, declspecs,
-                        NORMAL, initialized);
+                        NORMAL, initialized, NULL);
 
   deprecated_state = DEPRECATED_NORMAL;
 
   if (warn_main > 0 && TREE_CODE (decl) != FUNCTION_DECL
       && MAIN_NAME_P (DECL_NAME (decl)))
-    warning ("%H'%D' is usually a function",
-             &DECL_SOURCE_LOCATION (decl), decl);
+    warning ("%J'%D' is usually a function", decl, decl);
 
   if (initialized)
     /* Is it valid for this decl to have an initializer at all?
@@ -2674,11 +2605,30 @@ start_decl (tree declarator, tree declspecs, int initialized, tree attributes)
   decl_attributes (&decl, attributes, 0);
 
   if (TREE_CODE (decl) == FUNCTION_DECL
+      && targetm.calls.promote_prototypes (TREE_TYPE (decl)))
+    {
+      tree ce = declarator;
+
+      if (TREE_CODE (ce) == INDIRECT_REF)
+       ce = TREE_OPERAND (declarator, 0);
+      if (TREE_CODE (ce) == CALL_EXPR)
+       {
+         tree args = TREE_PURPOSE (TREE_OPERAND (ce, 1));
+         for (; args; args = TREE_CHAIN (args))
+           {
+             tree type = TREE_TYPE (args);
+             if (INTEGRAL_TYPE_P (type)
+                 && TYPE_PRECISION (type) < TYPE_PRECISION (integer_type_node))
+               DECL_ARG_TYPE (args) = integer_type_node;
+           }
+       }
+    }
+
+  if (TREE_CODE (decl) == FUNCTION_DECL
       && DECL_DECLARED_INLINE_P (decl)
       && DECL_UNINLINABLE (decl)
       && lookup_attribute ("noinline", DECL_ATTRIBUTES (decl)))
-    warning ("%Hinline function '%D' given attribute noinline",
-             &DECL_SOURCE_LOCATION (decl), decl);
+    warning ("%Jinline function '%D' given attribute noinline", decl, decl);
 
   /* Add this decl to the current scope.
      TEM may equal DECL or it may be a previous decl of the same name.  */
@@ -2731,7 +2681,12 @@ finish_decl (tree decl, tree init, tree asmspec_tree)
   if (init)
     store_init_value (decl, init);
 
-  /* Deduce size of array from initialization, if not already known */
+  if (c_dialect_objc () && (TREE_CODE (decl) == VAR_DECL
+                   || TREE_CODE (decl) == FUNCTION_DECL
+                   || TREE_CODE (decl) == FIELD_DECL))
+    objc_check_decl (decl);
+
+  /* Deduce size of array from initialization, if not already known.  */
   if (TREE_CODE (type) == ARRAY_TYPE
       && TYPE_DOMAIN (type) == 0
       && TREE_CODE (decl) != TYPE_DECL)
@@ -2749,14 +2704,12 @@ finish_decl (tree decl, tree init, tree asmspec_tree)
       type = TREE_TYPE (decl);
 
       if (failure == 1)
-       error ("%Hinitializer fails to determine size of '%D'",
-               &DECL_SOURCE_LOCATION (decl), decl);
+       error ("%Jinitializer fails to determine size of '%D'", decl, decl);
 
       else if (failure == 2)
        {
          if (do_default)
-           error ("%Harray size missing in '%D'",
-                   &DECL_SOURCE_LOCATION (decl), decl);
+           error ("%Jarray size missing in '%D'", decl, decl);
          /* If a `static' var's size isn't known,
             make it extern as well as static, so it does not get
             allocated.
@@ -2772,8 +2725,7 @@ finish_decl (tree decl, tree init, tree asmspec_tree)
         warn only if the value is less than zero.  */
       else if (pedantic && TYPE_DOMAIN (type) != 0
              && tree_int_cst_sgn (TYPE_MAX_VALUE (TYPE_DOMAIN (type))) < 0)
-       error ("%Hzero or negative size array '%D'",
-               &DECL_SOURCE_LOCATION (decl), decl);
+       error ("%Jzero or negative size array '%D'", decl, decl);
 
       layout_decl (decl, 0);
     }
@@ -2801,8 +2753,7 @@ finish_decl (tree decl, tree init, tree asmspec_tree)
                   is an error.  */
                !DECL_EXTERNAL (decl)))
        {
-         error ("%Hstorage size of '%D' isn't known",
-                 &DECL_SOURCE_LOCATION (decl), decl);
+         error ("%Jstorage size of '%D' isn't known", decl, decl);
          TREE_TYPE (decl) = error_mark_node;
        }
 
@@ -2812,8 +2763,7 @@ finish_decl (tree decl, tree init, tree asmspec_tree)
          if (TREE_CODE (DECL_SIZE (decl)) == INTEGER_CST)
            constant_expression_warning (DECL_SIZE (decl));
          else
-           error ("%Hstorage size of '%D' isn't constant",
-                   &DECL_SOURCE_LOCATION (decl), decl);
+           error ("%Jstorage size of '%D' isn't constant", decl, decl);
        }
 
       if (TREE_USED (type))
@@ -2849,7 +2799,7 @@ finish_decl (tree decl, tree init, tree asmspec_tree)
 #endif
        }
       SET_DECL_RTL (decl, NULL_RTX);
-      SET_DECL_ASSEMBLER_NAME (decl, get_identifier (starred));
+      change_decl_assembler_name (decl, get_identifier (starred));
     }
 
   /* If #pragma weak was used, mark the decl weak now.  */
@@ -2896,10 +2846,10 @@ finish_decl (tree decl, tree init, tree asmspec_tree)
              if (TREE_CODE (decl) == VAR_DECL
                  && !DECL_REGISTER (decl)
                  && !TREE_STATIC (decl))
-               warning ("%Hignoring asm-specifier for non-static local "
-                         "variable '%D'", &DECL_SOURCE_LOCATION (decl), decl);
+               warning ("%Jignoring asm-specifier for non-static local "
+                         "variable '%D'", decl, decl);
              else
-               SET_DECL_ASSEMBLER_NAME (decl, get_identifier (asmspec));
+               change_decl_assembler_name (decl, get_identifier (asmspec));
            }
 
          if (TREE_CODE (decl) != FUNCTION_DECL)
@@ -2927,12 +2877,7 @@ finish_decl (tree decl, tree init, tree asmspec_tree)
     mark_referenced (DECL_ASSEMBLER_NAME (decl));
 
   if (TREE_CODE (decl) == TYPE_DECL)
-    {
-      /* This is a no-op in c-lang.c or something real in objc-act.c.  */
-      if (c_dialect_objc ())
-       objc_check_decl (decl);
-      rest_of_decl_compilation (decl, NULL, DECL_FILE_SCOPE_P (decl), 0);
-    }
+    rest_of_decl_compilation (decl, NULL, DECL_FILE_SCOPE_P (decl), 0);
 
   /* At the end of a declaration, throw away any variable type sizes
      of types defined inside that declaration.  There is no use
@@ -2990,7 +2935,8 @@ push_parm_decl (tree parm)
   immediate_size_expand = 0;
 
   decl = grokdeclarator (TREE_VALUE (TREE_PURPOSE (parm)),
-                        TREE_PURPOSE (TREE_PURPOSE (parm)), PARM, 0);
+                        TREE_PURPOSE (TREE_PURPOSE (parm)),
+                        PARM, 0, NULL);
   decl_attributes (&decl, TREE_VALUE (parm), 0);
 
   decl = pushdecl (decl);
@@ -3185,6 +3131,77 @@ flexible_array_type_p (tree type)
   }
 }
 \f
+/* Performs sanity checks on the TYPE and WIDTH of the bit-field NAME,
+   replacing with appropriate values if they are invalid.  */
+static void
+check_bitfield_type_and_width (tree *type, tree *width, const char *orig_name)
+{
+  tree type_mv;
+  unsigned int max_width;
+  unsigned HOST_WIDE_INT w;
+  const char *name = orig_name ? orig_name: _("<anonymous>");
+
+  /* Necessary?  */
+  STRIP_NOPS (*width);
+
+  /* Detect and ignore out of range field width and process valid
+     field widths.  */
+  if (TREE_CODE (*width) != INTEGER_CST)
+    {
+      error ("bit-field `%s' width not an integer constant", name);
+      *width = integer_one_node;
+    }
+  else
+    {
+      constant_expression_warning (*width);
+      if (tree_int_cst_sgn (*width) < 0)
+       {
+         error ("negative width in bit-field `%s'", name);
+         *width = integer_one_node;
+       }
+      else if (integer_zerop (*width) && orig_name)
+       {
+         error ("zero width for bit-field `%s'", name);
+         *width = integer_one_node;
+       }
+    }
+
+  /* Detect invalid bit-field type.  */
+  if (TREE_CODE (*type) != INTEGER_TYPE
+      && TREE_CODE (*type) != BOOLEAN_TYPE
+      && TREE_CODE (*type) != ENUMERAL_TYPE)
+    {
+      error ("bit-field `%s' has invalid type", name);
+      *type = unsigned_type_node;
+    }
+
+  type_mv = TYPE_MAIN_VARIANT (*type);
+  if (pedantic
+      && type_mv != integer_type_node
+      && type_mv != unsigned_type_node
+      && type_mv != boolean_type_node)
+    pedwarn ("type of bit-field `%s' is a GCC extension", name);
+
+  if (type_mv == boolean_type_node)
+    max_width = CHAR_TYPE_SIZE;
+  else
+    max_width = TYPE_PRECISION (*type);
+
+  if (0 < compare_tree_int (*width, max_width))
+    {
+      error ("width of `%s' exceeds its type", name);
+      w = max_width;
+      *width = build_int_2 (w, 0);
+    }
+  else
+    w = tree_low_cst (*width, 1);
+
+  if (TREE_CODE (*type) == ENUMERAL_TYPE
+      && (w < min_precision (TYPE_MIN_VALUE (*type), TREE_UNSIGNED (*type))
+         || w < min_precision (TYPE_MAX_VALUE (*type), TREE_UNSIGNED (*type))))
+    warning ("`%s' is narrower than values of its type", name);
+}
+\f
 /* Given declspecs and a declarator,
    determine the name and type of the object declared
    and construct a ..._DECL node for it.
@@ -3204,8 +3221,9 @@ flexible_array_type_p (tree type)
      TYPENAME if for a typename (in a cast or sizeof).
       Don't make a DECL node; just return the ..._TYPE node.
      FIELD for a struct or union field; make a FIELD_DECL.
-     BITFIELD for a field with specified width.
    INITIALIZED is 1 if the decl has an initializer.
+   WIDTH is non-NULL for bit-fields, and is a pointer to an INTEGER_CST node
+   representing the width of the bit-field.
 
    In the TYPENAME case, DECLARATOR is really an absolute declarator.
    It may also be so in the PARM case, for a prototype where the
@@ -3216,7 +3234,7 @@ flexible_array_type_p (tree type)
 
 static tree
 grokdeclarator (tree declarator, tree declspecs,
-               enum decl_context decl_context, int initialized)
+               enum decl_context decl_context, int initialized, tree *width)
 {
   int specbits = 0;
   tree spec;
@@ -3231,19 +3249,17 @@ grokdeclarator (tree declarator, tree declspecs,
   int explicit_char = 0;
   int defaulted_int = 0;
   tree typedef_decl = 0;
-  const char *name;
+  const char *name, *orig_name;
   tree typedef_type = 0;
   int funcdef_flag = 0;
   enum tree_code innermost_code = ERROR_MARK;
-  int bitfield = 0;
   int size_varies = 0;
   tree decl_attr = NULL_TREE;
   tree array_ptr_quals = NULL_TREE;
   int array_parm_static = 0;
   tree returned_attrs = NULL_TREE;
-
-  if (decl_context == BITFIELD)
-    bitfield = 1, decl_context = FIELD;
+  bool bitfield = width != NULL;
+  tree element_type;
 
   if (decl_context == FUNCDEF)
     funcdef_flag = 1, decl_context = NORMAL;
@@ -3276,6 +3292,7 @@ grokdeclarator (tree declarator, tree declspecs,
        default:
          abort ();
        }
+    orig_name = name;
     if (name == 0)
       name = "type name";
   }
@@ -3508,7 +3525,7 @@ grokdeclarator (tree declarator, tree declspecs,
     }
 
   /* Decide whether an integer type is signed or not.
-     Optionally treat bitfields as signed by default.  */
+     Optionally treat bit-fields as signed by default.  */
   if (specbits & 1 << (int) RID_UNSIGNED
       || (bitfield && ! flag_signed_bitfields
          && (explicit_int || defaulted_int || explicit_char
@@ -3580,14 +3597,27 @@ grokdeclarator (tree declarator, tree declspecs,
        }
     }
 
+  /* Check the type and width of a bit-field.  */
+  if (bitfield)
+    check_bitfield_type_and_width (&type, width, orig_name);
+
   /* Figure out the type qualifiers for the declaration.  There are
      two ways a declaration can become qualified.  One is something
      like `const int i' where the `const' is explicit.  Another is
      something like `typedef const int CI; CI i' where the type of the
-     declaration contains the `const'.  */
-  constp = !! (specbits & 1 << (int) RID_CONST) + TYPE_READONLY (type);
-  restrictp = !! (specbits & 1 << (int) RID_RESTRICT) + TYPE_RESTRICT (type);
-  volatilep = !! (specbits & 1 << (int) RID_VOLATILE) + TYPE_VOLATILE (type);
+     declaration contains the `const'.  A third possibility is that
+     there is a type qualifier on the element type of a typedefed
+     array type, in which case we should extract that qualifier so
+     that c_apply_type_quals_to_decls receives the full list of
+     qualifiers to work with (C90 is not entirely clear about whether
+     duplicate qualifiers should be diagnosed in this case, but it
+     seems most appropriate to do so).  */
+  element_type = strip_array_types (type);
+  constp = !! (specbits & 1 << (int) RID_CONST) + TYPE_READONLY (element_type);
+  restrictp
+    = !! (specbits & 1 << (int) RID_RESTRICT) + TYPE_RESTRICT (element_type);
+  volatilep
+    = !! (specbits & 1 << (int) RID_VOLATILE) + TYPE_VOLATILE (element_type);
   inlinep = !! (specbits & (1 << (int) RID_INLINE));
   if (constp > 1 && ! flag_isoc99)
     pedwarn ("duplicate `const'");
@@ -4382,8 +4412,7 @@ grokdeclarator (tree declarator, tree declspecs,
          C_DECL_VARIABLE_SIZE (decl) = 1;
 
        if (inlinep)
-         pedwarn ("%Hvariable '%D' declared `inline'",
-                   &DECL_SOURCE_LOCATION (decl), decl);
+         pedwarn ("%Jvariable '%D' declared `inline'", decl, decl);
 
        DECL_EXTERNAL (decl) = extern_ref;
 
@@ -4575,10 +4604,6 @@ get_parm_info (int void_at_end)
         declared types.  The back end may override this.  */
       type = TREE_TYPE (decl);
       DECL_ARG_TYPE (decl) = type;
-      if (PROMOTE_PROTOTYPES
-         && INTEGRAL_TYPE_P (type)
-         && TYPE_PRECISION (type) < TYPE_PRECISION (integer_type_node))
-       DECL_ARG_TYPE (decl) = integer_type_node;
 
       /* Check for (..., void, ...) and issue an error.  */
       if (VOID_TYPE_P (type) && !DECL_NAME (decl) && !gave_void_only_once_err)
@@ -4600,8 +4625,8 @@ get_parm_info (int void_at_end)
        if (!TREE_ASM_WRITTEN (decl))
          abort ();
 
-         error ("%Hparameter \"%D\" has just a forward declaration",
-                &DECL_SOURCE_LOCATION (decl), decl);
+         error ("%Jparameter \"%D\" has just a forward declaration",
+                decl, decl);
       }
 
   /* Warn about any struct, union or enum tags defined within this
@@ -4626,7 +4651,7 @@ get_parm_info (int void_at_end)
        default: abort ();
        }
 
-      if (TREE_PURPOSE (decl)) 
+      if (TREE_PURPOSE (decl))
        /* The first %s will be one of 'struct', 'union', or 'enum'.  */
        warning ("\"%s %s\" declared inside parameter list",
                 keyword, IDENTIFIER_POINTER (TREE_PURPOSE (decl)));
@@ -4741,7 +4766,7 @@ start_struct (enum tree_code code, tree name)
 
 /* Process the specs, declarator (NULL if omitted) and width (NULL if omitted)
    of a structure component, returning a FIELD_DECL node.
-   WIDTH is non-NULL for bit fields only, and is an INTEGER_CST node.
+   WIDTH is non-NULL for bit-fields only, and is an INTEGER_CST node.
 
    This is done during the parsing of the struct declaration.
    The FIELD_DECL nodes are chained together and the lot of them
@@ -4796,13 +4821,12 @@ grokfield (tree declarator, tree declspecs, tree width)
        }
     }
 
-  value = grokdeclarator (declarator, declspecs, width ? BITFIELD : FIELD, 0);
+  value = grokdeclarator (declarator, declspecs, FIELD, 0,
+                         width ? &width : NULL);
 
   finish_decl (value, NULL_TREE, NULL_TREE);
   DECL_INITIAL (value) = width;
 
-  if (c_dialect_objc ())
-    objc_check_decl (value);
   return value;
 }
 \f
@@ -4837,8 +4861,7 @@ detect_field_duplicates (tree fieldlist)
            for (y = fieldlist; y != x; y = TREE_CHAIN (y))
              if (DECL_NAME (y) == DECL_NAME (x))
                {
-                 error ("%Hduplicate member '%D'",
-                         &DECL_SOURCE_LOCATION (x), x);
+                 error ("%Jduplicate member '%D'", x, x);
                  DECL_NAME (x) = NULL_TREE;
                }
          }
@@ -4854,8 +4877,7 @@ detect_field_duplicates (tree fieldlist)
            slot = htab_find_slot (htab, y, INSERT);
            if (*slot)
              {
-               error ("%Hduplicate member '%D'",
-                       &DECL_SOURCE_LOCATION (x), x);
+               error ("%Jduplicate member '%D'", x, x);
                DECL_NAME (x) = NULL_TREE;
              }
            *slot = y;
@@ -4947,79 +4969,12 @@ finish_struct (tree t, tree fieldlist, tree attributes)
        error ("nested redefinition of `%s'",
               IDENTIFIER_POINTER (TYPE_NAME (t)));
 
-      /* Detect invalid bit-field size.  */
-      if (DECL_INITIAL (x))
-       STRIP_NOPS (DECL_INITIAL (x));
-      if (DECL_INITIAL (x))
-       {
-         if (TREE_CODE (DECL_INITIAL (x)) == INTEGER_CST)
-           constant_expression_warning (DECL_INITIAL (x));
-         else
-           {
-             error ("%Hbit-field '%D' width not an integer constant",
-                     &DECL_SOURCE_LOCATION (x), x);
-             DECL_INITIAL (x) = NULL;
-           }
-       }
-
-      /* Detect invalid bit-field type.  */
-      if (DECL_INITIAL (x)
-         && TREE_CODE (TREE_TYPE (x)) != INTEGER_TYPE
-         && TREE_CODE (TREE_TYPE (x)) != BOOLEAN_TYPE
-         && TREE_CODE (TREE_TYPE (x)) != ENUMERAL_TYPE)
-       {
-         error ("%Hbit-field '%D' has invalid type",
-                 &DECL_SOURCE_LOCATION (x), x);
-         DECL_INITIAL (x) = NULL;
-       }
-
-      if (DECL_INITIAL (x) && pedantic
-         && TYPE_MAIN_VARIANT (TREE_TYPE (x)) != integer_type_node
-         && TYPE_MAIN_VARIANT (TREE_TYPE (x)) != unsigned_type_node
-         && TYPE_MAIN_VARIANT (TREE_TYPE (x)) != boolean_type_node
-         /* Accept an enum that's equivalent to int or unsigned int.  */
-         && !(TREE_CODE (TREE_TYPE (x)) == ENUMERAL_TYPE
-              && (TYPE_PRECISION (TREE_TYPE (x))
-                  == TYPE_PRECISION (integer_type_node))))
-       pedwarn ("%Hbit-field '%D' type invalid in ISO C",
-                 &DECL_SOURCE_LOCATION (x), x);
-
-      /* Detect and ignore out of range field width and process valid
-        field widths.  */
       if (DECL_INITIAL (x))
        {
-         int max_width
-           = (TYPE_MAIN_VARIANT (TREE_TYPE (x)) == boolean_type_node
-              ? CHAR_TYPE_SIZE : TYPE_PRECISION (TREE_TYPE (x)));
-
-         if (tree_int_cst_sgn (DECL_INITIAL (x)) < 0)
-           error ("%Hnegative width in bit-field '%D'",
-                   &DECL_SOURCE_LOCATION (x), x);
-         else if (0 < compare_tree_int (DECL_INITIAL (x), max_width))
-           pedwarn ("%Hwidth of '%D' exceeds its type",
-                     &DECL_SOURCE_LOCATION (x), x);
-         else if (integer_zerop (DECL_INITIAL (x)) && DECL_NAME (x) != 0)
-           error ("%Hzero width for bit-field '%D'",
-                   &DECL_SOURCE_LOCATION (x), x);
-         else
-           {
-             /* The test above has assured us that TREE_INT_CST_HIGH is 0.  */
-             unsigned HOST_WIDE_INT width
-               = tree_low_cst (DECL_INITIAL (x), 1);
-
-             if (TREE_CODE (TREE_TYPE (x)) == ENUMERAL_TYPE
-                 && (width < min_precision (TYPE_MIN_VALUE (TREE_TYPE (x)),
-                                            TREE_UNSIGNED (TREE_TYPE (x)))
-                     || (width
-                         < min_precision (TYPE_MAX_VALUE (TREE_TYPE (x)),
-                                          TREE_UNSIGNED (TREE_TYPE (x))))))
-               warning ("%H'%D' is narrower than values of its type",
-                         &DECL_SOURCE_LOCATION (x), x);
-
-             DECL_SIZE (x) = bitsize_int (width);
-             DECL_BIT_FIELD (x) = 1;
-             SET_DECL_C_BIT_FIELD (x);
-           }
+         unsigned HOST_WIDE_INT width = tree_low_cst (DECL_INITIAL (x), 1);
+         DECL_SIZE (x) = bitsize_int (width);
+         DECL_BIT_FIELD (x) = 1;
+         SET_DECL_C_BIT_FIELD (x);
        }
 
       DECL_INITIAL (x) = 0;
@@ -5031,20 +4986,16 @@ finish_struct (tree t, tree fieldlist, tree attributes)
          && TYPE_MAX_VALUE (TYPE_DOMAIN (TREE_TYPE (x))) == NULL_TREE)
        {
          if (TREE_CODE (t) == UNION_TYPE)
-           error ("%Hflexible array member in union",
-                   &DECL_SOURCE_LOCATION (x));
+           error ("%Jflexible array member in union", x);
          else if (TREE_CHAIN (x) != NULL_TREE)
-           error ("%Hflexible array member not at end of struct",
-                   &DECL_SOURCE_LOCATION (x));
+           error ("%Jflexible array member not at end of struct", x);
          else if (! saw_named_field)
-           error ("%Hflexible array member in otherwise empty struct",
-                   &DECL_SOURCE_LOCATION (x));
+           error ("%Jflexible array member in otherwise empty struct", x);
        }
 
       if (pedantic && TREE_CODE (t) == RECORD_TYPE
          && flexible_array_type_p (TREE_TYPE (x)))
-       pedwarn ("%Hinvalid use of structure with flexible array member",
-                 &DECL_SOURCE_LOCATION (x));
+       pedwarn ("%Jinvalid use of structure with flexible array member", x);
 
       if (DECL_NAME (x))
        saw_named_field = 1;
@@ -5059,7 +5010,7 @@ finish_struct (tree t, tree fieldlist, tree attributes)
 
   layout_type (t);
 
-  /* Delete all zero-width bit-fields from the fieldlist */
+  /* Delete all zero-width bit-fields from the fieldlist */
   {
     tree *fieldlistp = &fieldlist;
     while (*fieldlistp)
@@ -5075,7 +5026,7 @@ finish_struct (tree t, tree fieldlist, tree attributes)
   TYPE_FIELDS (t) = fieldlist;
 
   /* If there are lots of fields, sort so we can look through them fast.
-    We arbitrarily consider 16 or more elts to be "a lot".  */
+     We arbitrarily consider 16 or more elts to be "a lot".  */
 
   {
     int len = 0;
@@ -5092,23 +5043,23 @@ finish_struct (tree t, tree fieldlist, tree attributes)
         tree *field_array;
         struct lang_type *space;
         struct sorted_fields_type *space2;
-        
+
         len += list_length (x);
-  
+
         /* Use the same allocation policy here that make_node uses, to
           ensure that this lives as long as the rest of the struct decl.
           All decls in an inline function need to be saved.  */
-  
+
         space = ggc_alloc (sizeof (struct lang_type));
         space2 = ggc_alloc (sizeof (struct sorted_fields_type) + len * sizeof (tree));
-        
+
         len = 0;
        space->s = space2;
        field_array = &space2->elts[0];
         for (x = fieldlist; x; x = TREE_CHAIN (x))
           {
             field_array[len++] = x;
-          
+
             /* If there is anonymous struct or union, break out of the loop.  */
             if (DECL_NAME (x) == NULL)
               break;
@@ -5123,7 +5074,7 @@ finish_struct (tree t, tree fieldlist, tree attributes)
           }
       }
   }
-  
+
   for (x = TYPE_MAIN_VARIANT (t); x; x = TYPE_NEXT_VARIANT (x))
     {
       TYPE_FIELDS (x) = TYPE_FIELDS (t);
@@ -5442,11 +5393,13 @@ start_function (tree declspecs, tree declarator, tree attributes)
   current_function_returns_abnormally = 0;
   warn_about_return_type = 0;
   current_extern_inline = 0;
+  c_in_iteration_stmt = 0;
+  c_in_case_stmt = 0;
 
   /* Don't expand any sizes in the return type of the function.  */
   immediate_size_expand = 0;
 
-  decl1 = grokdeclarator (declarator, declspecs, FUNCDEF, 1);
+  decl1 = grokdeclarator (declarator, declspecs, FUNCDEF, 1, NULL);
 
   /* If the declarator is not suitable for a function definition,
      cause a syntax error.  */
@@ -5461,8 +5414,7 @@ start_function (tree declspecs, tree declarator, tree attributes)
   if (DECL_DECLARED_INLINE_P (decl1)
       && DECL_UNINLINABLE (decl1)
       && lookup_attribute ("noinline", DECL_ATTRIBUTES (decl1)))
-    warning ("%Hinline function '%D' given attribute noinline",
-             &DECL_SOURCE_LOCATION (decl1), decl1);
+    warning ("%Jinline function '%D' given attribute noinline", decl1, decl1);
 
   announce_function (decl1);
 
@@ -5512,29 +5464,27 @@ start_function (tree declspecs, tree declarator, tree attributes)
           && TREE_PUBLIC (decl1)
           && ! MAIN_NAME_P (DECL_NAME (decl1))
           && C_DECL_ISNT_PROTOTYPE (old_decl))
-    warning ("%Hno previous prototype for '%D'",
-             &DECL_SOURCE_LOCATION (decl1), decl1);
+    warning ("%Jno previous prototype for '%D'", decl1, decl1);
   /* Optionally warn of any def with no previous prototype
      if the function has already been used.  */
   else if (warn_missing_prototypes
           && old_decl != 0 && TREE_USED (old_decl)
           && TYPE_ARG_TYPES (TREE_TYPE (old_decl)) == 0)
-    warning ("%H'%D' was used with no prototype before its definition",
-             &DECL_SOURCE_LOCATION (decl1), decl1);
+    warning ("%J'%D' was used with no prototype before its definition",
+            decl1, decl1);
   /* Optionally warn of any global def with no previous declaration.  */
   else if (warn_missing_declarations
           && TREE_PUBLIC (decl1)
           && old_decl == 0
           && ! MAIN_NAME_P (DECL_NAME (decl1)))
-    warning ("%Hno previous declaration for '%D'",
-             &DECL_SOURCE_LOCATION (decl1), decl1);
+    warning ("%Jno previous declaration for '%D'", decl1, decl1);
   /* Optionally warn of any def with no previous declaration
      if the function has already been used.  */
   else if (warn_missing_declarations
           && old_decl != 0 && TREE_USED (old_decl)
           && C_DECL_IMPLICIT (old_decl))
-    warning ("%H`%D' was used with no declaration before its definition",
-             &DECL_SOURCE_LOCATION (decl1), decl1);
+    warning ("%J`%D' was used with no declaration before its definition",
+            decl1, decl1);
 
   /* This is a definition, not a reference.
      So normally clear DECL_EXTERNAL.
@@ -5566,11 +5516,10 @@ start_function (tree declspecs, tree declarator, tree attributes)
     {
       tree args;
       int argct = 0;
-      const location_t *locus = &DECL_SOURCE_LOCATION (decl1);
 
       if (TYPE_MAIN_VARIANT (TREE_TYPE (TREE_TYPE (decl1)))
          != integer_type_node)
-       pedwarn ("%Hreturn type of '%D' is not `int'", locus, decl1);
+       pedwarn ("%Jreturn type of '%D' is not `int'", decl1, decl1);
 
       for (args = TYPE_ARG_TYPES (TREE_TYPE (decl1)); args;
           args = TREE_CHAIN (args))
@@ -5585,8 +5534,8 @@ start_function (tree declspecs, tree declarator, tree attributes)
            {
            case 1:
              if (TYPE_MAIN_VARIANT (type) != integer_type_node)
-               pedwarn ("%Hfirst argument of '%D' should be `int'",
-                         locus, decl1);
+               pedwarn ("%Jfirst argument of '%D' should be `int'",
+                        decl1, decl1);
              break;
 
            case 2:
@@ -5594,8 +5543,8 @@ start_function (tree declspecs, tree declarator, tree attributes)
                  || TREE_CODE (TREE_TYPE (type)) != POINTER_TYPE
                  || (TYPE_MAIN_VARIANT (TREE_TYPE (TREE_TYPE (type)))
                      != char_type_node))
-               pedwarn ("%Hsecond argument of '%D' should be 'char **'",
-                         locus, decl1);
+               pedwarn ("%Jsecond argument of '%D' should be 'char **'",
+                         decl1, decl1);
              break;
 
            case 3:
@@ -5603,8 +5552,8 @@ start_function (tree declspecs, tree declarator, tree attributes)
                  || TREE_CODE (TREE_TYPE (type)) != POINTER_TYPE
                  || (TYPE_MAIN_VARIANT (TREE_TYPE (TREE_TYPE (type)))
                      != char_type_node))
-               pedwarn ("%Hthird argument of '%D' should probably be "
-                         "'char **'", locus, decl1);
+               pedwarn ("%Jthird argument of '%D' should probably be "
+                         "'char **'", decl1, decl1);
              break;
            }
        }
@@ -5613,10 +5562,10 @@ start_function (tree declspecs, tree declarator, tree attributes)
         argument because it's only mentioned in an appendix of the
         standard.  */
       if (argct > 0 && (argct < 2 || argct > 3))
-       pedwarn ("%H'%D' takes only zero or two arguments", locus, decl1);
+       pedwarn ("%J'%D' takes only zero or two arguments", decl1, decl1);
 
       if (! TREE_PUBLIC (decl1))
-       pedwarn ("%H'%D' is normally a non-static function", locus, decl1);
+       pedwarn ("%J'%D' is normally a non-static function", decl1, decl1);
     }
 
   /* Record the decl so that the function name is defined.
@@ -5672,8 +5621,8 @@ store_parm_decls_newstyle (void)
 
   if (current_scope->parms || current_scope->names || current_scope->tags)
     {
-      error ("%Hold-style parameter declarations in prototyped "
-            "function definition", &DECL_SOURCE_LOCATION (fndecl));
+      error ("%Jold-style parameter declarations in prototyped "
+            "function definition", fndecl);
 
       /* Get rid of the old-style declarations.  */
       poplevel (0, 0, 0);
@@ -5686,7 +5635,7 @@ store_parm_decls_newstyle (void)
     {
       DECL_CONTEXT (decl) = current_function_decl;
       if (DECL_NAME (decl) == 0)
-       error ("%Hparameter name omitted", &DECL_SOURCE_LOCATION (decl));
+       error ("%Jparameter name omitted", decl);
       else
        {
          if (IDENTIFIER_SYMBOL_VALUE (DECL_NAME (decl)))
@@ -5761,8 +5710,7 @@ store_parm_decls_oldstyle (void)
     {
       if (TREE_VALUE (parm) == 0)
        {
-         error ("%Hparameter name missing from parameter list",
-                &DECL_SOURCE_LOCATION (fndecl));
+         error ("%Jparameter name missing from parameter list", fndecl);
          TREE_PURPOSE (parm) = 0;
          continue;
        }
@@ -5770,15 +5718,14 @@ store_parm_decls_oldstyle (void)
       decl = IDENTIFIER_SYMBOL_VALUE (TREE_VALUE (parm));
       if (decl && DECL_CONTEXT (decl) == fndecl)
        {
-         const location_t *locus = &DECL_SOURCE_LOCATION (decl);
          /* If we got something other than a PARM_DECL it is an error.  */
          if (TREE_CODE (decl) != PARM_DECL)
-           error ("%H\"%D\" declared as a non-parameter", locus, decl);
+           error ("%J\"%D\" declared as a non-parameter", decl, decl);
          /* If the declaration is already marked, we have a duplicate
             name.  Complain and ignore the duplicate.  */
          else if (DECL_WEAK (decl))
            {
-             error ("%Hmultiple parameters named \"%D\"", locus, decl);
+             error ("%Jmultiple parameters named \"%D\"", decl, decl);
              TREE_PURPOSE (parm) = 0;
              continue;
            }
@@ -5786,7 +5733,7 @@ store_parm_decls_oldstyle (void)
             an int.  */
          else if (VOID_TYPE_P (TREE_TYPE (decl)))
            {
-             error ("%Hparameter \"%D\" declared void", locus, decl);
+             error ("%Jparameter \"%D\" declared void", decl, decl);
              TREE_TYPE (decl) = integer_type_node;
              DECL_ARG_TYPE (decl) = integer_type_node;
              layout_decl (decl, 0);
@@ -5795,16 +5742,15 @@ store_parm_decls_oldstyle (void)
       /* If no declaration found, default to int.  */
       else
        {
-         const location_t *locus = &DECL_SOURCE_LOCATION (fndecl);
          decl = build_decl (PARM_DECL, TREE_VALUE (parm), integer_type_node);
          DECL_ARG_TYPE (decl) = TREE_TYPE (decl);
-         DECL_SOURCE_LOCATION (decl) = *locus;
+         DECL_SOURCE_LOCATION (decl) = DECL_SOURCE_LOCATION (fndecl);
          pushdecl (decl);
 
          if (flag_isoc99)
-           pedwarn ("%Htype of \"%D\" defaults to \"int\"", locus, decl);
+           pedwarn ("%Jtype of \"%D\" defaults to \"int\"", decl, decl);
          else if (extra_warnings)
-           warning ("%Htype of \"%D\" defaults to \"int\"", locus, decl);
+           warning ("%Jtype of \"%D\" defaults to \"int\"", decl, decl);
        }
 
       TREE_PURPOSE (parm) = decl;
@@ -5816,18 +5762,16 @@ store_parm_decls_oldstyle (void)
 
   for (parm = current_scope->parms; parm; parm = TREE_CHAIN (parm))
     {
-      const location_t *locus = &DECL_SOURCE_LOCATION (parm);
-
       if (!COMPLETE_TYPE_P (TREE_TYPE (parm)))
        {
-         error ("%Hparameter \"%D\" has incomplete type", locus, parm);
+         error ("%Jparameter \"%D\" has incomplete type", parm, parm);
          TREE_TYPE (parm) = error_mark_node;
        }
 
       if (! DECL_WEAK (parm))
        {
-         error ("%Hdeclaration for parameter \"%D\" but no such parameter",
-                locus, parm);
+         error ("%Jdeclaration for parameter \"%D\" but no such parameter",
+                parm, parm);
 
          /* Pretend the parameter was not missing.
             This gets us to a standard state and minimizes
@@ -5900,7 +5844,7 @@ store_parm_decls_oldstyle (void)
                     useful for argument types like uid_t.  */
                  DECL_ARG_TYPE (parm) = TREE_TYPE (parm);
 
-                 if (PROMOTE_PROTOTYPES
+                 if (targetm.calls.promote_prototypes (TREE_TYPE (current_function_decl))
                      && INTEGRAL_TYPE_P (TREE_TYPE (parm))
                      && TYPE_PRECISION (TREE_TYPE (parm))
                      < TYPE_PRECISION (integer_type_node))
@@ -6029,13 +5973,10 @@ store_parm_decls (void)
    all the way to assembler language output.  The free the storage
    for the function definition.
 
-   This is called after parsing the body of the function definition.
-
-   NESTED is nonzero if the function being finished is nested in another.
-   CAN_DEFER_P is nonzero if the function may be deferred.  */
+   This is called after parsing the body of the function definition.  */
 
 void
-finish_function (int nested, int can_defer_p)
+finish_function (void)
 {
   tree fndecl = current_function_decl;
 
@@ -6055,11 +5996,26 @@ finish_function (int nested, int can_defer_p)
       poplevel (0, 0, 0);
     }
 
-  BLOCK_SUPERCONTEXT (DECL_INITIAL (fndecl)) = fndecl;
+  if (TREE_CODE (fndecl) == FUNCTION_DECL
+      && targetm.calls.promote_prototypes (TREE_TYPE (fndecl)))
+    {
+      tree args = DECL_ARGUMENTS (fndecl);
+      for (; args; args = TREE_CHAIN (args))
+       {
+         tree type = TREE_TYPE (args);
+         if (INTEGRAL_TYPE_P (type)
+             && TYPE_PRECISION (type) < TYPE_PRECISION (integer_type_node))
+           DECL_ARG_TYPE (args) = integer_type_node;
+       }
+    }
+
+  if (DECL_INITIAL (fndecl) && DECL_INITIAL (fndecl) != error_mark_node)
+    BLOCK_SUPERCONTEXT (DECL_INITIAL (fndecl)) = fndecl;
 
   /* Must mark the RESULT_DECL as being in this function.  */
 
-  DECL_CONTEXT (DECL_RESULT (fndecl)) = fndecl;
+  if (DECL_RESULT (fndecl) && DECL_RESULT (fndecl) != error_mark_node)
+    DECL_CONTEXT (DECL_RESULT (fndecl)) = fndecl;
 
   if (MAIN_NAME_P (DECL_NAME (fndecl)) && flag_hosted)
     {
@@ -6069,8 +6025,7 @@ finish_function (int nested, int can_defer_p)
          /* If warn_main is 1 (-Wmain) or 2 (-Wall), we have already warned.
             If warn_main is -1 (-Wno-main) we don't want to be warned.  */
          if (!warn_main)
-           pedwarn ("%Hreturn type of '%D' is not `int'",
-                     &DECL_SOURCE_LOCATION (fndecl), fndecl);
+           pedwarn ("%Jreturn type of '%D' is not `int'", fndecl, fndecl);
        }
       else
        {
@@ -6104,104 +6059,30 @@ finish_function (int nested, int can_defer_p)
       && DECL_INLINE (fndecl))
     warning ("no return statement in function returning non-void");
 
-  /* We're leaving the context of this function, so zap cfun.  It's still in
-     DECL_SAVED_INSNS, and we'll restore it in tree_rest_of_compilation.  */
-  cfun = NULL;
-
-  if (flag_unit_at_a_time && can_defer_p)
-    {
-      cgraph_finalize_function (fndecl, DECL_SAVED_TREE (fndecl));
-      current_function_decl = NULL;
-      return;
-    }
-
-  if (! nested)
-    {
-      /* Function is parsed.
-        Generate RTL for the body of this function or defer
-        it for later expansion.  */
-      bool uninlinable = true;
-
-      /* There's no reason to do any of the work here if we're only doing
-        semantic analysis; this code just generates RTL.  */
-      if (flag_syntax_only)
-       {
-         current_function_decl = NULL;
-         DECL_SAVED_TREE (fndecl) = NULL_TREE;
-         return;
-       }
-
-      if (flag_inline_trees)
-       {
-         /* First, cache whether the current function is inlinable.  Some
-            predicates depend on cfun and current_function_decl to
-            function completely.  */
-         timevar_push (TV_INTEGRATION);
-         uninlinable = !tree_inlinable_function_p (fndecl);
-
-         if (can_defer_p
-             /* We defer functions marked inline *even if* the function
-                itself is not inlinable.  This is because we don't yet
-                know if the function will actually be used; we may be
-                able to avoid emitting it entirely.  */
-             && (!uninlinable || DECL_DECLARED_INLINE_P (fndecl))
-             /* Save function tree for inlining.  Should return 0 if the
-                language does not support function deferring or the
-                function could not be deferred.  */
-             && defer_fn (fndecl))
-           {
-             /* Let the back-end know that this function exists.  */
-             (*debug_hooks->deferred_inline_function) (fndecl);
-             timevar_pop (TV_INTEGRATION);
-             current_function_decl = NULL;
-             return;
-           }
-
-         /* Then, inline any functions called in it.  */
-         optimize_inline_calls (fndecl);
-         timevar_pop (TV_INTEGRATION);
-       }
-
-      c_expand_body (fndecl);
-
-      /* Keep the function body if it's needed for inlining or dumping.  */
-      if (uninlinable && !dump_enabled_p (TDI_all))
-       {
-         /* Allow the body of the function to be garbage collected.  */
-         DECL_SAVED_TREE (fndecl) = NULL_TREE;
-       }
-
-      /* Let the error reporting routines know that we're outside a
-        function.  For a nested function, this value is used in
-        c_pop_function_context and then reset via pop_function_context.  */
-      current_function_decl = NULL;
-    }
-}
+  /* With just -Wextra, complain only if function returns both with
+     and without a value.  */
+  if (extra_warnings
+      && current_function_returns_value
+      && current_function_returns_null)
+    warning ("this function may return with or without a value");
 
-/* Generate the RTL for a deferred function FNDECL.  */
+  /* We're leaving the context of this function, so zap cfun.
+     It's still in DECL_STRUCT_FUNCTION , and we'll restore it in
+     tree_rest_of_compilation.  */
+  cfun = NULL;
 
-void
-c_expand_deferred_function (tree fndecl)
-{
-  /* DECL_INLINE or DECL_RESULT might got cleared after the inline
-     function was deferred, e.g. in duplicate_decls.  */
-  if (DECL_INLINE (fndecl) && DECL_RESULT (fndecl))
-    {
-      if (flag_inline_trees)
-       {
-         timevar_push (TV_INTEGRATION);
-         optimize_inline_calls (fndecl);
-         timevar_pop (TV_INTEGRATION);
-       }
-      c_expand_body (fndecl);
-      current_function_decl = NULL;
-    }
+  /* ??? Objc emits functions after finalizing the compilation unit.
+     This should be cleaned up later and this conditional removed.  */
+  if (!cgraph_global_info_ready)
+    cgraph_finalize_function (fndecl, false);
+  else
+    c_expand_body (fndecl);
+  current_function_decl = NULL;
 }
 
 /* Generate the RTL for the body of FNDECL.  If NESTED_P is nonzero,
    then we are already in the process of generating RTL for another
-   function.  If can_defer_p is zero, we won't attempt to defer the
-   generation of RTL.  */
+   function.  */
 
 static void
 c_expand_body_1 (tree fndecl, int nested_p)
@@ -6211,16 +6092,16 @@ c_expand_body_1 (tree fndecl, int nested_p)
       /* Make sure that we will evaluate variable-sized types involved
         in our function's type.  */
       expand_pending_sizes (DECL_LANG_SPECIFIC (fndecl)->pending_sizes);
-    }
 
-  tree_rest_of_compilation (fndecl);
+      /* Squirrel away our current state.  */
+      push_function_context ();
+    }
+    
+  tree_rest_of_compilation (fndecl, nested_p);
 
-  /* With just -Wextra, complain only if function returns both with
-     and without a value.  */
-  if (extra_warnings
-      && current_function_returns_value
-      && current_function_returns_null)
-    warning ("this function may return with or without a value");
+  if (nested_p)
+    /* Return to the enclosing function.  */
+    pop_function_context ();
 
   if (DECL_STATIC_CONSTRUCTOR (fndecl))
     {
@@ -6246,7 +6127,9 @@ c_expand_body_1 (tree fndecl, int nested_p)
 void
 c_expand_body (tree fndecl)
 {
-  c_expand_body_1 (fndecl, 0);
+
+  if (DECL_INITIAL (fndecl) && DECL_INITIAL (fndecl) != error_mark_node)
+    c_expand_body_1 (fndecl, 0);
 }
 \f
 /* Check the declarations given in a for-loop for satisfying the C99
@@ -6296,35 +6179,20 @@ check_for_loop_decls (void)
         }
     }
 
-  for (t = getdecls (); t; t = TREE_CHAIN (t))
+  for (t = current_scope->names; t; t = TREE_CHAIN (t))
     {
-      const location_t *locus = &DECL_SOURCE_LOCATION (t);
       if (TREE_CODE (t) != VAR_DECL && DECL_NAME (t))
-       error ("%Hdeclaration of non-variable '%D' in 'for' loop "
-               "initial declaration", locus, t);
+       error ("%Jdeclaration of non-variable '%D' in 'for' loop "
+               "initial declaration", t, t);
       else if (TREE_STATIC (t))
-       error ("%Hdeclaration of static variable '%D' in 'for' loop "
-              "initial declaration", locus, t);
+       error ("%Jdeclaration of static variable '%D' in 'for' loop "
+              "initial declaration", t, t);
       else if (DECL_EXTERNAL (t))
-       error ("%Hdeclaration of 'extern' variable '%D' in 'for' loop "
-               "initial declaration", locus, t);
+       error ("%Jdeclaration of 'extern' variable '%D' in 'for' loop "
+               "initial declaration", t, t);
     }
 }
 \f
-/* Save and restore the variables in this file and elsewhere
-   that keep track of the progress of compilation of the current function.
-   Used for nested functions.  */
-
-struct language_function GTY(())
-{
-  struct c_language_function base;
-  int returns_value;
-  int returns_null;
-  int returns_abnormally;
-  int warn_about_return_type;
-  int extern_inline;
-};
-
 /* Save and reinitialize the variables
    used during compilation of a C function.  */
 
@@ -6337,6 +6205,8 @@ c_push_function_context (struct function *f)
 
   p->base.x_stmt_tree = c_stmt_tree;
   p->base.x_scope_stmt_stack = c_scope_stmt_stack;
+  p->x_in_iteration_stmt = c_in_iteration_stmt;
+  p->x_in_case_stmt = c_in_case_stmt;
   p->returns_value = current_function_returns_value;
   p->returns_null = current_function_returns_null;
   p->returns_abnormally = current_function_returns_abnormally;
@@ -6351,7 +6221,7 @@ c_pop_function_context (struct function *f)
 {
   struct language_function *p = f->language;
 
-  if (DECL_SAVED_INSNS (current_function_decl) == 0
+  if (DECL_STRUCT_FUNCTION (current_function_decl) == 0
       && DECL_SAVED_TREE (current_function_decl) == NULL_TREE)
     {
       /* Stop pointing to the local nodes about to be freed.  */
@@ -6363,6 +6233,8 @@ c_pop_function_context (struct function *f)
 
   c_stmt_tree = p->base.x_stmt_tree;
   c_scope_stmt_stack = p->base.x_scope_stmt_stack;
+  c_in_iteration_stmt = p->x_in_iteration_stmt;
+  c_in_case_stmt = p->x_in_case_stmt;
   current_function_returns_value = p->returns_value;
   current_function_returns_null = p->returns_null;
   current_function_returns_abnormally = p->returns_abnormally;
@@ -6490,11 +6362,14 @@ void
 record_builtin_type (enum rid rid_index, const char *name, tree type)
 {
   tree id;
+  tree tdecl;
   if (name == 0)
     id = ridpointers[(int) rid_index];
   else
     id = get_identifier (name);
-  pushdecl (build_decl (TYPE_DECL, id, type));
+  tdecl = build_decl (TYPE_DECL, id, type);
+  pushdecl (tdecl);
+  debug_hooks->type_decl (tdecl, 0);
 }
 
 /* Build the void_list_node (void_type_node having been created).  */
@@ -6570,7 +6445,7 @@ merge_translation_unit_decls (void)
   tree decl;
   htab_t link_hash_table;
   tree block;
-  
+
   /* Create the BLOCK that poplevel would have created, but don't
      actually call poplevel since that's expensive.  */
   block = make_node (BLOCK);
@@ -6611,7 +6486,7 @@ merge_translation_unit_decls (void)
                DECL_EXTERNAL (decl) = 1;
              else if (DECL_COMMON (old_decl) || DECL_ONE_ONLY (old_decl))
                DECL_EXTERNAL (old_decl) = 1;
-             
+
              if (DECL_EXTERNAL (decl))
                {
                  DECL_INITIAL (decl) = NULL_TREE;
@@ -6629,10 +6504,8 @@ merge_translation_unit_decls (void)
                }
              else
                {
-                 error ("%Hredefinition of global '%D'",
-                         &DECL_SOURCE_LOCATION (decl), decl);
-                 error ("%H'%D' previously defined here",
-                         &DECL_SOURCE_LOCATION (old_decl), old_decl);
+                 error ("%Jredefinition of global '%D'", decl, decl);
+                 error ("%J'%D' previously defined here", old_decl, old_decl);
                }
            }
          else
@@ -6647,13 +6520,13 @@ merge_translation_unit_decls (void)
        {
          tree global_decl;
          global_decl = htab_find (link_hash_table, decl);
-         
+
          if (! global_decl)
            continue;
-         
+
          /* Print any appropriate error messages, and partially merge
             the decls.  */
-         (void) duplicate_decls (decl, global_decl, true, true);
+         (void) duplicate_decls (decl, global_decl);
        }
 
   htab_delete (link_hash_table);
@@ -6665,7 +6538,7 @@ void
 c_write_global_declarations(void)
 {
   tree link;
-  
+
   for (link = current_file_decl; link; link = TREE_CHAIN (link))
     {
       tree globals = BLOCK_VARS (DECL_INITIAL (link));
@@ -6673,16 +6546,16 @@ c_write_global_declarations(void)
       tree *vec = xmalloc (sizeof (tree) * len);
       int i;
       tree decl;
-      
+
       /* Process the decls in the order they were written.  */
 
       for (i = 0, decl = globals; i < len; i++, decl = TREE_CHAIN (decl))
        vec[i] = decl;
-      
+
       wrapup_global_declarations (vec, len);
-      
+
       check_global_declarations (vec, len);
-      
+
       /* Clean up.  */
       free (vec);
     }
@@ -6695,12 +6568,13 @@ c_reset_state (void)
 {
   tree link;
   tree file_scope_decl;
-  
+
   /* Pop the global scope.  */
   if (current_scope != global_scope)
       current_scope = global_scope;
   file_scope_decl = current_file_decl;
   DECL_INITIAL (file_scope_decl) = poplevel (1, 0, 0);
+  BLOCK_SUPERCONTEXT (DECL_INITIAL (file_scope_decl)) = file_scope_decl;
   truly_local_externals = NULL_TREE;
 
   /* Start a new global binding level.  */