diff options
author | Richard Smith <richard-llvm@metafoo.co.uk> | 2011-11-29 05:27:40 +0000 |
---|---|---|
committer | Richard Smith <richard-llvm@metafoo.co.uk> | 2011-11-29 05:27:40 +0000 |
commit | 874d253668f9ed183ca409cdff9d424925ee7800 (patch) | |
tree | 637fc2dc1c9b792795c6f63d5cf68061c347caa2 | |
parent | b49e728a4d1a84b72f3aebf60ff494684f9cb004 (diff) |
PR10101: Recover better from a common copy-paste error: if a function
declaration at namespace scope is followed by a semicolon and an open-brace
(or in C++, a 'try', ':' or '='), then the error is probably a function
definition with a spurious ';', rather than a mysterious '{'.
git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@145372 91177308-0d34-0410-b5e6-96231b3b80d8
-rw-r--r-- | include/clang/Basic/DiagnosticParseKinds.td | 2 | ||||
-rw-r--r-- | lib/Parse/ParseDecl.cpp | 17 | ||||
-rw-r--r-- | lib/Parse/ParseTemplate.cpp | 26 | ||||
-rw-r--r-- | test/FixIt/fixit-cxx0x.cpp | 23 | ||||
-rw-r--r-- | test/FixIt/fixit.c | 8 | ||||
-rw-r--r-- | test/FixIt/fixit.cpp | 10 | ||||
-rw-r--r-- | test/SemaCXX/typedef-redecl.cpp | 5 |
7 files changed, 80 insertions, 11 deletions
diff --git a/include/clang/Basic/DiagnosticParseKinds.td b/include/clang/Basic/DiagnosticParseKinds.td index e0b10fa87c..9b5540d4c9 100644 --- a/include/clang/Basic/DiagnosticParseKinds.td +++ b/include/clang/Basic/DiagnosticParseKinds.td @@ -138,6 +138,8 @@ def ext_expected_semi_decl_list : ExtWarn< "expected ';' at end of declaration list">; def err_expected_member_name_or_semi : Error< "expected member name or ';' after declaration specifiers">; +def err_stray_semi_function_definition : Error< + "stray ';' in function definition">; def err_function_declared_typedef : Error< "function definition declared 'typedef'">; def err_iboutletcollection_builtintype : Error< diff --git a/lib/Parse/ParseDecl.cpp b/lib/Parse/ParseDecl.cpp index 3bddd3de13..75c423340c 100644 --- a/lib/Parse/ParseDecl.cpp +++ b/lib/Parse/ParseDecl.cpp @@ -1041,13 +1041,28 @@ Parser::DeclGroupPtrTy Parser::ParseDeclGroup(ParsingDeclSpec &DS, return DeclGroupPtrTy(); } + // Do we have a stray semicolon in the middle of a function definition? + if (AllowFunctionDefinitions && D.isFunctionDeclarator() && + Tok.is(tok::semi) && Context == Declarator::FileContext) { + const Token &Next = NextToken(); + if (Next.is(tok::l_brace) || Next.is(tok::kw_try) || + (getLang().CPlusPlus && + (Next.is(tok::colon) || Next.is(tok::equal)))) { + // Pretend we didn't see the semicolon. + SourceLocation SemiLoc = ConsumeToken(); + Diag(SemiLoc, diag::err_stray_semi_function_definition) + << FixItHint::CreateRemoval(SemiLoc); + assert(isStartOfFunctionDefinition(D) && "expected a function defn"); + } + } + // Check to see if we have a function *definition* which must have a body. if (AllowFunctionDefinitions && D.isFunctionDeclarator() && // Look at the next token to make sure that this isn't a function // declaration. We have to check this because __attribute__ might be the // start of a function definition in GCC-extended K&R C. !isDeclarationAfterDeclarator()) { - + if (isStartOfFunctionDefinition(D)) { if (DS.getStorageClassSpec() == DeclSpec::SCS_typedef) { Diag(Tok, diag::err_function_declared_typedef); diff --git a/lib/Parse/ParseTemplate.cpp b/lib/Parse/ParseTemplate.cpp index b485f1e977..91cf8e81c2 100644 --- a/lib/Parse/ParseTemplate.cpp +++ b/lib/Parse/ParseTemplate.cpp @@ -240,6 +240,20 @@ Parser::ParseSingleDeclarationAfterTemplate( return 0; } + // Check for a stray semicolon in a function definition. + if (DeclaratorInfo.isFunctionDeclarator() && Tok.is(tok::semi) && + Context == Declarator::FileContext) { + const Token &Next = NextToken(); + if (Next.is(tok::l_brace) || Next.is(tok::kw_try) || + Next.is(tok::equal) || Next.is(tok::colon)) { + SourceLocation SemiLoc = ConsumeToken(); + Diag(SemiLoc, diag::err_stray_semi_function_definition) + << FixItHint::CreateRemoval(SemiLoc); + assert(!isDeclarationAfterDeclarator() && + isStartOfFunctionDefinition(DeclaratorInfo)); + } + } + // If we have a declaration or declarator list, handle it. if (isDeclarationAfterDeclarator()) { // Parse this declaration. @@ -264,16 +278,8 @@ Parser::ParseSingleDeclarationAfterTemplate( if (DS.getStorageClassSpec() == DeclSpec::SCS_typedef) { Diag(Tok, diag::err_function_declared_typedef); - if (Tok.is(tok::l_brace)) { - // This recovery skips the entire function body. It would be nice - // to simply call ParseFunctionDefinition() below, however Sema - // assumes the declarator represents a function, not a typedef. - ConsumeBrace(); - SkipUntil(tok::r_brace, true); - } else { - SkipUntil(tok::semi); - } - return 0; + // Recover by ignoring the 'typedef'. + DS.ClearStorageClassSpecs(); } return ParseFunctionDefinition(DeclaratorInfo, TemplateInfo); } diff --git a/test/FixIt/fixit-cxx0x.cpp b/test/FixIt/fixit-cxx0x.cpp index 9fb647d03f..d9b8763b05 100644 --- a/test/FixIt/fixit-cxx0x.cpp +++ b/test/FixIt/fixit-cxx0x.cpp @@ -58,3 +58,26 @@ namespace SemiCommaTypo { n [[]], // expected-error {{expected ';' at end of declaration}} int o; } + +int extraSemi(); // expected-error {{stray ';' in function definition}} + = delete; + +class ExtraSemi { +public: + ExtraSemi(); + ExtraSemi(const ExtraSemi &); + int n; +}; +ExtraSemi::ExtraSemi(); // expected-error {{stray ';'}} + : n(0) { +} +ExtraSemi::ExtraSemi(const ExtraSemi &); // expected-error {{stray ';'}} + = default; + +template<typename T> T extraSemi(T t); + +template<typename T> T extraSemi(T t); // expected-error {{stray ';'}} +{ + return t; +} +template int extraSemi(int); diff --git a/test/FixIt/fixit.c b/test/FixIt/fixit.c index 967ae23c18..da1144da0f 100644 --- a/test/FixIt/fixit.c +++ b/test/FixIt/fixit.c @@ -77,3 +77,11 @@ void oopsMoreCommas() { static int b[] = { 3, 4, 5 }, &a == &b ? oopsMoreCommas() : removeUnusedLabels(a[0]); } + +void extraSemicolon(); +{ + void extraSemicolon(); + { + return; + } +} diff --git a/test/FixIt/fixit.cpp b/test/FixIt/fixit.cpp index 86b397777c..7c8ba9ac6f 100644 --- a/test/FixIt/fixit.cpp +++ b/test/FixIt/fixit.cpp @@ -125,3 +125,13 @@ AD oopsMoreCommas() { AD ad, // expected-error {{expected ';' at end of declaration}} return ad; } + +int extraSemi1(); // expected-error {{stray ';' in function definition}} +{ + return 0; +} + +int extraSemi2(); // expected-error {{stray ';' in function definition}} +try { +} catch (...) { +} diff --git a/test/SemaCXX/typedef-redecl.cpp b/test/SemaCXX/typedef-redecl.cpp index 7db1970a70..31de9c00c1 100644 --- a/test/SemaCXX/typedef-redecl.cpp +++ b/test/SemaCXX/typedef-redecl.cpp @@ -54,3 +54,8 @@ namespace PR7462 { typedef int operator! (A); // expected-error{{typedef name must be an identifier}} int i = !A(); // expected-error{{invalid argument type}} } + +template<typename T> +typedef T f(T t) { return t; } // expected-error {{function definition declared 'typedef'}} +int k = f(0); +int k2 = k; |