diff options
author | Chris Lattner <sabre@nondot.org> | 2010-02-02 17:32:27 +0000 |
---|---|---|
committer | Chris Lattner <sabre@nondot.org> | 2010-02-02 17:32:27 +0000 |
commit | 99c952046600f6bfccf315aa7ad5b1be2d242cc3 (patch) | |
tree | d1f1bcc3ebad22a05f7a64967154e6013118176f | |
parent | 39053672d459bff1d3ab1f72a7901f24e3fa542e (diff) |
the declspec of a declaration can have storage-class specifiers,
type qualifiers and type specifiers in any order. For example,
this is valid: struct x {...} typedef y;
This fixes PR6208.
git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@95094 91177308-0d34-0410-b5e6-96231b3b80d8
-rw-r--r-- | lib/Parse/ParseDeclCXX.cpp | 25 | ||||
-rw-r--r-- | test/Parser/declarators.c | 4 | ||||
-rw-r--r-- | test/Sema/declspec.c | 3 |
3 files changed, 25 insertions, 7 deletions
diff --git a/lib/Parse/ParseDeclCXX.cpp b/lib/Parse/ParseDeclCXX.cpp index 234ec4c8b5..371aba1719 100644 --- a/lib/Parse/ParseDeclCXX.cpp +++ b/lib/Parse/ParseDeclCXX.cpp @@ -935,13 +935,24 @@ void Parser::ParseClassSpecifier(tok::TokenKind TagTokKind, if (TUK == Action::TUK_Definition) { switch (Tok.getKind()) { case tok::semi: // struct foo {...} ; - case tok::star: // struct foo {...} * P; - case tok::amp: // struct foo {...} & R = ... - case tok::identifier: // struct foo {...} V ; - case tok::r_paren: //(struct foo {...} ) {4} - case tok::annot_cxxscope: // struct foo {...} a:: b; - case tok::annot_typename: // struct foo {...} a ::b; - case tok::annot_template_id: // struct foo {...} a<int> ::b; + case tok::star: // struct foo {...} * P; + case tok::amp: // struct foo {...} & R = ... + case tok::identifier: // struct foo {...} V ; + case tok::r_paren: //(struct foo {...} ) {4} + case tok::annot_cxxscope: // struct foo {...} a:: b; + case tok::annot_typename: // struct foo {...} a ::b; + case tok::annot_template_id: // struct foo {...} a<int> ::b; + // Storage-class specifiers + case tok::kw_static: // struct foo {...} static x; + case tok::kw_extern: // struct foo {...} extern x; + case tok::kw_typedef: // struct foo {...} typedef x; + case tok::kw_register: // struct foo {...} register x; + case tok::kw_auto: // struct foo {...} auto x; + // Type qualifiers + case tok::kw_const: // struct foo {...} const x; + case tok::kw_volatile: // struct foo {...} volatile x; + case tok::kw_restrict: // struct foo {...} restrict x; + case tok::kw_inline: // struct foo {...} inline foo() {}; break; case tok::r_brace: // struct bar { struct foo {...} } diff --git a/test/Parser/declarators.c b/test/Parser/declarators.c index 074e2ad7cb..edd68cc28c 100644 --- a/test/Parser/declarators.c +++ b/test/Parser/declarators.c @@ -71,3 +71,7 @@ struct test9 { int y; int z // expected-warning {{expected ';' at end of declaration list}} }; + +// PR6208 +struct test10 { int a; } static test10x; +struct test11 { int a; } const test11x; diff --git a/test/Sema/declspec.c b/test/Sema/declspec.c index e1f2bf4911..d9f4157ab3 100644 --- a/test/Sema/declspec.c +++ b/test/Sema/declspec.c @@ -8,8 +8,11 @@ void foof(const char *, ...) __attribute__((__format__(__printf__, 1, 2))), barf int typedef validTypeDecl() { } // expected-error {{function definition declared 'typedef'}} struct _zend_module_entry { } // expected-error {{expected ';' after struct}} +int gv1; typedef struct _zend_function_entry { } // expected-error {{expected ';' after struct}} \ // expected-error {{declaration does not declare anything}} +int gv2; + static void buggy(int *x) { } // Type qualifiers. |