diff options
author | Jakub Jelinek <jakub@redhat.com> | 2008-11-10 14:41:37 +0100 |
---|---|---|
committer | Jakub Jelinek <jakub@gcc.gnu.org> | 2008-11-10 14:41:37 +0100 |
commit | 5072c6bd2e2e365b1cc92e921cbf75d34fb26c0f (patch) | |
tree | 4ee43bf33873d343cbe16be7dc0e94d1b1199e81 | |
parent | 94b5f4db225e10e4d0a93f7dba46bc6ddf8e193a (diff) | |
download | gcc-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/ChangeLog | 7 | ||||
-rw-r--r-- | gcc/cp/parser.c | 23 | ||||
-rw-r--r-- | gcc/testsuite/ChangeLog | 5 | ||||
-rw-r--r-- | gcc/testsuite/g++.dg/cpp0x/enum1.C | 6 |
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" } |