From: Argyrios Kyrtzidis Date: Sat, 17 Dec 2011 04:13:22 +0000 (+0000) Subject: In Parser::SkipUntil do not stop at '@' unconditionally. X-Git-Url: https://granicus.if.org/sourcecode?a=commitdiff_plain;h=0db9f4dad563a335641f5b9d4a42504d638b6c85;p=clang In Parser::SkipUntil do not stop at '@' unconditionally. Stopping at '@' was originally intended to avoid skipping an '@' at the @interface context when doing parser recovery, but we should not stop at all '@' tokens because they may be part of expressions (e.g. in @"string", @selector(), etc.), so in most cases we will want to skip them. This commit caused 'test/Parser/method-def-in-class.m' to fail for the cases where we tried to recover from unmatched angle bracket but IMO it is not a big deal to not have good recovery from such broken code and the way we did recovery would not always work anyway (e.g. if there was '@' in an expression). The case that rdar://7029784 is about still passes. git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@146815 91177308-0d34-0410-b5e6-96231b3b80d8 --- diff --git a/lib/Parse/ParseObjc.cpp b/lib/Parse/ParseObjc.cpp index 56d5c57c05..1f20924c9b 100644 --- a/lib/Parse/ParseObjc.cpp +++ b/lib/Parse/ParseObjc.cpp @@ -366,8 +366,12 @@ void Parser::ParseObjCInterfaceDeclList(tok::ObjCKeywordKind contextKey, allMethods.push_back(methodPrototype); // Consume the ';' here, since ParseObjCMethodPrototype() is re-used for // method definitions. - ExpectAndConsume(tok::semi, diag::err_expected_semi_after_method_proto, - "", tok::semi); + if (ExpectAndConsumeSemi(diag::err_expected_semi_after_method_proto)) { + // We didn't find a semi and we error'ed out. Skip until a ';' or '@'. + SkipUntil(tok::at, /*StopAtSemi=*/true, /*DontConsume=*/true); + if (Tok.is(tok::semi)) + ConsumeToken(); + } continue; } if (Tok.is(tok::l_paren)) { diff --git a/lib/Parse/Parser.cpp b/lib/Parse/Parser.cpp index 95ce065ff7..c57711dbc6 100644 --- a/lib/Parse/Parser.cpp +++ b/lib/Parse/Parser.cpp @@ -284,9 +284,6 @@ bool Parser::SkipUntil(const tok::TokenKind *Toks, unsigned NumToks, ConsumeStringToken(); break; - case tok::at: - return false; - case tok::semi: if (StopAtSemi) return false; diff --git a/test/Parser/method-def-in-class.m b/test/Parser/method-def-in-class.m index 490c0559d1..476ab9ba20 100644 --- a/test/Parser/method-def-in-class.m +++ b/test/Parser/method-def-in-class.m @@ -7,19 +7,8 @@ } @end -@interface B --(id) f0 { // expected-error {{expected ';' after method prototype}} - assert(0); -@end - @interface C - (id) f0 { // expected-error {{expected ';' after method prototype}} assert(0); }; @end - -@interface D -- (id) f0 { // expected-error {{expected ';' after method prototype}} - assert(0); -@property int P; -@end