summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--gcc/ChangeLog10
-rw-r--r--gcc/c-decl.c18
-rw-r--r--gcc/c-parse.in26
-rw-r--r--gcc/c-typeck.c2
-rw-r--r--gcc/testsuite/ChangeLog9
-rw-r--r--gcc/testsuite/gcc.dg/funcdef-storage-1.c6
-rw-r--r--gcc/testsuite/gcc.dg/pr18596-1.c36
-rw-r--r--gcc/testsuite/gcc.dg/pr18596-2.c39
-rw-r--r--gcc/testsuite/gcc.dg/pr18596-3.c14
9 files changed, 137 insertions, 23 deletions
diff --git a/gcc/ChangeLog b/gcc/ChangeLog
index f65773c70c0..3fc04174332 100644
--- a/gcc/ChangeLog
+++ b/gcc/ChangeLog
@@ -1,3 +1,13 @@
+2004-12-21 James A. Morrison <phython@gcc.gnu.org>
+
+ PR c/18596
+ * c-parse.in (initdcl): Don't process a declaration if start_decl fails.
+ (notype_initdcl): Don't process a declaration if start_decl fails.
+ * c-decl.c (start_decl): Fail if grokdeclarator fails.
+ (grokdeclarator): Fail if a function definition has an invalid storage
+ class.
+ * c-typeck.c (start_init): Treat error_mark_node the same as 0.
+
2004-12-21 Richard Henderson <rth@redhat.com>
* config/i386/i386.c (x86_sse_split_regs): Rename from
diff --git a/gcc/c-decl.c b/gcc/c-decl.c
index 97420d16cc2..f881ea8e052 100644
--- a/gcc/c-decl.c
+++ b/gcc/c-decl.c
@@ -2964,6 +2964,8 @@ start_decl (struct c_declarator *declarator, struct c_declspecs *declspecs,
decl = grokdeclarator (declarator, declspecs,
NORMAL, initialized, NULL);
+ if (!decl)
+ return 0;
deprecated_state = DEPRECATED_NORMAL;
@@ -4437,14 +4439,9 @@ grokdeclarator (const struct c_declarator *declarator,
}
else if (TREE_CODE (type) == FUNCTION_TYPE)
{
- decl = build_decl (FUNCTION_DECL, declarator->u.id, type);
- decl = build_decl_attribute_variant (decl, decl_attr);
-
if (storage_class == csc_register || threadp)
{
error ("invalid storage class for function %qs", name);
- if (DECL_INITIAL (decl) != NULL_TREE)
- DECL_INITIAL (decl) = error_mark_node;
}
else if (current_scope != file_scope)
{
@@ -4458,14 +4455,19 @@ grokdeclarator (const struct c_declarator *declarator,
if (pedantic)
pedwarn ("invalid storage class for function %qs", name);
}
- if (storage_class == csc_static)
+ else if (storage_class == csc_static)
{
error ("invalid storage class for function %qs", name);
- if (DECL_INITIAL (decl) != NULL_TREE)
- DECL_INITIAL (decl) = error_mark_node;
+ if (funcdef_flag)
+ storage_class = declspecs->storage_class = csc_none;
+ else
+ return 0;
}
}
+ decl = build_decl (FUNCTION_DECL, declarator->u.id, type);
+ decl = build_decl_attribute_variant (decl, decl_attr);
+
DECL_LANG_SPECIFIC (decl) = GGC_CNEW (struct lang_decl);
if (pedantic && type_quals && !DECL_IN_SYSTEM_HEADER (decl))
diff --git a/gcc/c-parse.in b/gcc/c-parse.in
index 86880ac6153..94df4bbc3b0 100644
--- a/gcc/c-parse.in
+++ b/gcc/c-parse.in
@@ -1321,16 +1321,23 @@ initdcl:
declarator maybeasm maybe_attribute '='
{ $<ttype>$ = start_decl ($1, current_declspecs, true,
chainon ($3, all_prefix_attributes));
+ if (!$<ttype>$)
+ $<ttype>$ = error_mark_node;
start_init ($<ttype>$, $2, global_bindings_p ()); }
init
/* Note how the declaration of the variable is in effect while its init is parsed! */
{ finish_init ();
- maybe_warn_string_init (TREE_TYPE ($<ttype>5), $6);
- finish_decl ($<ttype>5, $6.value, $2); }
+ if ($<ttype>5 != error_mark_node)
+ {
+ maybe_warn_string_init (TREE_TYPE ($<ttype>5), $6);
+ finish_decl ($<ttype>5, $6.value, $2);
+ }
+ }
| declarator maybeasm maybe_attribute
{ tree d = start_decl ($1, current_declspecs, false,
chainon ($3, all_prefix_attributes));
- finish_decl (d, NULL_TREE, $2);
+ if (d)
+ finish_decl (d, NULL_TREE, $2);
}
;
@@ -1338,16 +1345,23 @@ notype_initdcl:
notype_declarator maybeasm maybe_attribute '='
{ $<ttype>$ = start_decl ($1, current_declspecs, true,
chainon ($3, all_prefix_attributes));
+ if (!$<ttype>$)
+ $<ttype>$ = error_mark_node;
start_init ($<ttype>$, $2, global_bindings_p ()); }
init
/* Note how the declaration of the variable is in effect while its init is parsed! */
{ finish_init ();
- maybe_warn_string_init (TREE_TYPE ($<ttype>5), $6);
- finish_decl ($<ttype>5, $6.value, $2); }
+ if ($<ttype>5 != error_mark_node)
+ {
+ maybe_warn_string_init (TREE_TYPE ($<ttype>5), $6);
+ finish_decl ($<ttype>5, $6.value, $2);
+ }
+ }
| notype_declarator maybeasm maybe_attribute
{ tree d = start_decl ($1, current_declspecs, false,
chainon ($3, all_prefix_attributes));
- finish_decl (d, NULL_TREE, $2); }
+ if (d)
+ finish_decl (d, NULL_TREE, $2); }
;
/* the * rules are dummies to accept the Apollo extended syntax
so that the header files compile. */
diff --git a/gcc/c-typeck.c b/gcc/c-typeck.c
index b3f18727585..d2963b37454 100644
--- a/gcc/c-typeck.c
+++ b/gcc/c-typeck.c
@@ -4470,7 +4470,7 @@ start_init (tree decl, tree asmspec_tree ATTRIBUTE_UNUSED, int top_level)
constructor_designated = 0;
constructor_top_level = top_level;
- if (decl != 0)
+ if (decl != 0 && decl != error_mark_node)
{
require_constant_value = TREE_STATIC (decl);
require_constant_elements
diff --git a/gcc/testsuite/ChangeLog b/gcc/testsuite/ChangeLog
index 138785b2a89..0c742af3358 100644
--- a/gcc/testsuite/ChangeLog
+++ b/gcc/testsuite/ChangeLog
@@ -1,3 +1,12 @@
+2004-12-21 James A. Morrison <phython@gcc.gnu.org>
+
+ PR c/18596
+ * gcc.dg/funcdef-storage-1.c (foo): Remove.
+ * gcc.dg/pr18596-1.c: Use dg-error.
+ (dg-options): Use -fno-unit-at-a-time.
+ * gcc.dg/pr18596-2.c: New test.
+ * gcc.dg/pr18596-3.c: New test.
+
2004-12-20 Roger Sayle <roger@eyesopen.com>
PR middle-end/18683
diff --git a/gcc/testsuite/gcc.dg/funcdef-storage-1.c b/gcc/testsuite/gcc.dg/funcdef-storage-1.c
index 7ced3a14b2e..9aa963a0288 100644
--- a/gcc/testsuite/gcc.dg/funcdef-storage-1.c
+++ b/gcc/testsuite/gcc.dg/funcdef-storage-1.c
@@ -1,5 +1,4 @@
/* { dg-do compile } */
-/* { dg-options "" } */
void
flarm(void)
@@ -8,8 +7,3 @@ flarm(void)
foo();
}
-
-static void
-foo(void)
-{
-}
diff --git a/gcc/testsuite/gcc.dg/pr18596-1.c b/gcc/testsuite/gcc.dg/pr18596-1.c
index 055d60a4a75..dc34f444e5d 100644
--- a/gcc/testsuite/gcc.dg/pr18596-1.c
+++ b/gcc/testsuite/gcc.dg/pr18596-1.c
@@ -1,7 +1,39 @@
/* { dg-do compile } */
+/* { dg-options "-fno-unit-at-a-time" } */
+
int f(int i)
{
- static int g(); /* { dg-warning "invalid storage class" } */
- static int g() { return i; } /* { dg-warning "invalid storage class" } */
+ static int g(); /* { dg-error "invalid storage class" } */
+ static int g() { return i; } /* { dg-error "invalid storage class" } */
return g();
}
+
+int k (int i)
+{
+ static int g (); /* { dg-error "invalid storage class" } */
+ int g () {
+ return i;
+ }
+
+ return g ();
+}
+
+int l (int i)
+{
+ auto int g ();
+ static int g () { /* { dg-error "invalid storage class" } */
+ return i;
+ }
+
+ static int h () { /* { dg-error "invalid storage class" } */
+ return 3;
+ }
+ return g () + h ();
+}
+
+int m (int i)
+{
+ static g (); /* { dg-error "invalid storage class" } */
+ static g () { return i; } /* { dg-error "invalid storage class" } */
+ return g ();
+}
diff --git a/gcc/testsuite/gcc.dg/pr18596-2.c b/gcc/testsuite/gcc.dg/pr18596-2.c
new file mode 100644
index 00000000000..7a52e807d38
--- /dev/null
+++ b/gcc/testsuite/gcc.dg/pr18596-2.c
@@ -0,0 +1,39 @@
+/* { dg-do compile } */
+/* { dg-options "-funit-at-a-time" } */
+
+int f(int i)
+{
+ static int g(); /* { dg-error "invalid storage class" } */
+ static int g() { return i; } /* { dg-error "invalid storage class" } */
+ return g();
+}
+
+int k (int i)
+{
+ static int g (); /* { dg-error "invalid storage class" } */
+ int g () {
+ return i;
+ }
+
+ return g ();
+}
+
+int l (int i)
+{
+ auto int g ();
+ static int g () { /* { dg-error "invalid storage class" } */
+ return i;
+ }
+
+ static int h () { /* { dg-error "invalid storage class" } */
+ return 3;
+ }
+ return g () + h ();
+}
+
+int m (int i)
+{
+ static g (); /* { dg-error "invalid storage class" } */
+ static g () { return i; } /* { dg-error "invalid storage class" } */
+ return g ();
+}
diff --git a/gcc/testsuite/gcc.dg/pr18596-3.c b/gcc/testsuite/gcc.dg/pr18596-3.c
new file mode 100644
index 00000000000..be17e7ca4b3
--- /dev/null
+++ b/gcc/testsuite/gcc.dg/pr18596-3.c
@@ -0,0 +1,14 @@
+/* { dg-do compile } */
+/* { dg-options "" } */
+
+int foo ()
+{
+ static g () = 0; /* { dg-error "invalid storage class" } */
+ static int f () = 1; /* { dg-error "invalid storage class" } */
+ auto int h () = 0; /* { dg-error "initialized like a variable" } */
+ static int i () = { 0 }; /* { dg-error "invalid storage class" } */
+ static int j () = /* { dg-error "invalid storage class" } */
+ { 0, 0.0 };
+}
+/* { dg-warning "excess elements" "" { target *-*-* } 11 } */
+/* { dg-warning "near initialization" "" { target *-*-* } 11 } */