aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorJakub Jelinek <jakub@redhat.com>2008-11-10 14:41:37 +0100
committerJakub Jelinek <jakub@gcc.gnu.org>2008-11-10 14:41:37 +0100
commit5072c6bd2e2e365b1cc92e921cbf75d34fb26c0f (patch)
tree4ee43bf33873d343cbe16be7dc0e94d1b1199e81
parent94b5f4db225e10e4d0a93f7dba46bc6ddf8e193a (diff)
downloadgcc-5072c6bd2e2e365b1cc92e921cbf75d34fb26c0f.zip
gcc-5072c6bd2e2e365b1cc92e921cbf75d34fb26c0f.tar.gz
gcc-5072c6bd2e2e365b1cc92e921cbf75d34fb26c0f.tar.bz2
re PR c++/38021 (C++ hang for new keywords)
PR c++/38021 * parser.c (cp_parser_enum_specifier): After parsing :, parse definitely. Don't return early if type specifier is erroneous. * g++.dg/cpp0x/enum1.C: New test. From-SVN: r141739
-rw-r--r--gcc/cp/ChangeLog7
-rw-r--r--gcc/cp/parser.c23
-rw-r--r--gcc/testsuite/ChangeLog5
-rw-r--r--gcc/testsuite/g++.dg/cpp0x/enum1.C6
4 files changed, 33 insertions, 8 deletions
diff --git a/gcc/cp/ChangeLog b/gcc/cp/ChangeLog
index ec7a1aa..81941e2 100644
--- a/gcc/cp/ChangeLog
+++ b/gcc/cp/ChangeLog
@@ -1,3 +1,10 @@
+2008-11-10 Jakub Jelinek <jakub@redhat.com>
+
+ PR c++/38021
+ * parser.c (cp_parser_enum_specifier): After parsing :,
+ parse definitely. Don't return early if type specifier
+ is erroneous.
+
2008-11-06 David Edelsohn <edelsohn@gnu.org>
PR target/26397
diff --git a/gcc/cp/parser.c b/gcc/cp/parser.c
index 40f2a3a..c819082 100644
--- a/gcc/cp/parser.c
+++ b/gcc/cp/parser.c
@@ -11780,6 +11780,7 @@ cp_parser_enum_specifier (cp_parser* parser)
tree type;
tree attributes;
bool scoped_enum_p = false;
+ bool has_underlying_type = false;
tree underlying_type = NULL_TREE;
/* Parse tentatively so that we can back up if we don't find a
@@ -11805,7 +11806,7 @@ cp_parser_enum_specifier (cp_parser* parser)
scoped_enum_p = true;
}
-
+
attributes = cp_parser_attributes_opt (parser);
if (cp_lexer_next_token_is (parser->lexer, CPP_NAME))
@@ -11818,18 +11819,22 @@ cp_parser_enum_specifier (cp_parser* parser)
{
cp_decl_specifier_seq type_specifiers;
+ /* At this point this is surely not elaborated type specifier. */
+ if (!cp_parser_parse_definitely (parser))
+ return NULL_TREE;
+
if (cxx_dialect == cxx98)
maybe_warn_cpp0x ("scoped enums");
/* Consume the `:'. */
cp_lexer_consume_token (parser->lexer);
+ has_underlying_type = true;
+
/* Parse the type-specifier-seq. */
cp_parser_type_specifier_seq (parser, /*is_condition=*/false,
&type_specifiers);
- if (type_specifiers.type == error_mark_node)
- return error_mark_node;
-
+
/* If that didn't work, stop. */
if (type_specifiers.type != error_mark_node)
{
@@ -11838,15 +11843,17 @@ cp_parser_enum_specifier (cp_parser* parser)
if (underlying_type == error_mark_node)
underlying_type = NULL_TREE;
}
- else
- cp_parser_error (parser, "expected underlying type of enumeration");
}
/* Look for the `{' but don't consume it yet. */
if (!cp_lexer_next_token_is (parser->lexer, CPP_OPEN_BRACE))
- cp_parser_simulate_error (parser);
+ {
+ cp_parser_error (parser, "expected %<{%>");
+ if (has_underlying_type)
+ return NULL_TREE;
+ }
- if (!cp_parser_parse_definitely (parser))
+ if (!has_underlying_type && !cp_parser_parse_definitely (parser))
return NULL_TREE;
/* Issue an error message if type-definitions are forbidden here. */
diff --git a/gcc/testsuite/ChangeLog b/gcc/testsuite/ChangeLog
index b341ead..412c77a 100644
--- a/gcc/testsuite/ChangeLog
+++ b/gcc/testsuite/ChangeLog
@@ -1,3 +1,8 @@
+2008-11-10 Jakub Jelinek <jakub@redhat.com>
+
+ PR c++/38021
+ * g++.dg/cpp0x/enum1.C: New test.
+
2008-11-02 Ralph Loader <suckfish@ihug.co.nz>
PR middle-end/37807
diff --git a/gcc/testsuite/g++.dg/cpp0x/enum1.C b/gcc/testsuite/g++.dg/cpp0x/enum1.C
new file mode 100644
index 0000000..af691f0
--- /dev/null
+++ b/gcc/testsuite/g++.dg/cpp0x/enum1.C
@@ -0,0 +1,6 @@
+// PR c++/38021
+// { dg-do compile }
+// { dg-options "-std=gnu++0x" }
+
+enum : { }; // { dg-error "expected type-specifier" }
+enum : 3 { }; // { dg-error "expected" }