diff options
| author | Erik Verbruggen <erikjv@me.com> | 2011-12-06 09:25:23 +0000 | 
|---|---|---|
| committer | Erik Verbruggen <erikjv@me.com> | 2011-12-06 09:25:23 +0000 | 
| commit | c6c8d9356dc04afc149d3bab16c99ccefa2cbc5e (patch) | |
| tree | 86ac1194f34efe6109075ff1958c2a0202ed3d4c /clang/lib | |
| parent | 6572e0f2033dc950d1dafbc60365f7f5534dac32 (diff) | |
| download | bcm5719-llvm-c6c8d9356dc04afc149d3bab16c99ccefa2cbc5e.tar.gz bcm5719-llvm-c6c8d9356dc04afc149d3bab16c99ccefa2cbc5e.zip | |
Extend warnings for missing '@end'.
Fixes PR2709.
llvm-svn: 145928
Diffstat (limited to 'clang/lib')
| -rw-r--r-- | clang/lib/Parse/ParseObjc.cpp | 67 | ||||
| -rw-r--r-- | clang/lib/Sema/SemaDeclObjC.cpp | 47 | 
2 files changed, 83 insertions, 31 deletions
| diff --git a/clang/lib/Parse/ParseObjc.cpp b/clang/lib/Parse/ParseObjc.cpp index 10c74ffdfdb..737f2b858d3 100644 --- a/clang/lib/Parse/ParseObjc.cpp +++ b/clang/lib/Parse/ParseObjc.cpp @@ -113,6 +113,23 @@ Parser::ParseObjCAtClassDeclaration(SourceLocation atLoc) {                                                ClassNames.size());  } +void Parser::CheckNestedObjCContexts(SourceLocation AtLoc) +{ +  Sema::ObjCContainerKind ock = Actions.getObjCContainerKind(); +  if (ock == Sema::OCK_None) +    return; + +  Decl *Decl = Actions.ActOnAtEnd(getCurScope(), AtLoc); +  Diag(AtLoc, diag::err_objc_missing_end) +      << FixItHint::CreateInsertion(AtLoc, "@end\n"); +  if (Decl) +    Diag(Decl->getLocStart(), diag::note_objc_container_start) +        << (int) ock; +  if (!PendingObjCImpDecl.empty()) +    PendingObjCImpDecl.pop_back(); +  ObjCImpDecl = 0; +} +  ///  ///   objc-interface:  ///     objc-class-interface-attributes[opt] objc-class-interface @@ -141,10 +158,11 @@ Parser::ParseObjCAtClassDeclaration(SourceLocation atLoc) {  ///     __attribute__((unavailable))  ///     __attribute__((objc_exception)) - used by NSException on 64-bit  /// -Decl *Parser::ParseObjCAtInterfaceDeclaration(SourceLocation atLoc, +Decl *Parser::ParseObjCAtInterfaceDeclaration(SourceLocation AtLoc,                                                ParsedAttributes &attrs) {    assert(Tok.isObjCAtKeyword(tok::objc_interface) &&           "ParseObjCAtInterfaceDeclaration(): Expected @interface"); +  CheckNestedObjCContexts(AtLoc);    ConsumeToken(); // the "interface" identifier    // Code completion after '@interface'. @@ -205,7 +223,7 @@ Decl *Parser::ParseObjCAtInterfaceDeclaration(SourceLocation atLoc,        return 0;      Decl *CategoryType = -    Actions.ActOnStartCategoryInterface(atLoc, +    Actions.ActOnStartCategoryInterface(AtLoc,                                          nameId, nameLoc,                                          categoryId, categoryLoc,                                          ProtocolRefs.data(), @@ -214,7 +232,7 @@ Decl *Parser::ParseObjCAtInterfaceDeclaration(SourceLocation atLoc,                                          EndProtoLoc);      if (Tok.is(tok::l_brace)) -      ParseObjCClassInstanceVariables(CategoryType, tok::objc_private, atLoc); +      ParseObjCClassInstanceVariables(CategoryType, tok::objc_private, AtLoc);      ParseObjCInterfaceDeclList(tok::objc_not_keyword, CategoryType);      return CategoryType; @@ -250,14 +268,14 @@ Decl *Parser::ParseObjCAtInterfaceDeclaration(SourceLocation atLoc,      return 0;    Decl *ClsType = -    Actions.ActOnStartClassInterface(atLoc, nameId, nameLoc, +    Actions.ActOnStartClassInterface(AtLoc, nameId, nameLoc,                                       superClassId, superClassLoc,                                       ProtocolRefs.data(), ProtocolRefs.size(),                                       ProtocolLocs.data(),                                       EndProtoLoc, attrs.getList());    if (Tok.is(tok::l_brace)) -    ParseObjCClassInstanceVariables(ClsType, tok::objc_protected, atLoc); +    ParseObjCClassInstanceVariables(ClsType, tok::objc_protected, AtLoc);    ParseObjCInterfaceDeclList(tok::objc_interface, ClsType);    return ClsType; @@ -425,7 +443,10 @@ void Parser::ParseObjCInterfaceDeclList(tok::ObjCKeywordKind contextKey,      case tok::objc_implementation:      case tok::objc_interface: -      Diag(Tok, diag::err_objc_missing_end); +      Diag(AtLoc, diag::err_objc_missing_end) +          << FixItHint::CreateInsertion(AtLoc, "@end\n"); +      Diag(CDecl->getLocStart(), diag::note_objc_container_start) +          << (int) Actions.getObjCContainerKind();        ConsumeToken();        break; @@ -465,10 +486,16 @@ void Parser::ParseObjCInterfaceDeclList(tok::ObjCKeywordKind contextKey,    if (Tok.is(tok::code_completion)) {      Actions.CodeCompleteObjCAtDirective(getCurScope());      return cutOffParsing(); -  } else if (Tok.isObjCAtKeyword(tok::objc_end)) +  } else if (Tok.isObjCAtKeyword(tok::objc_end)) {      ConsumeToken(); // the "end" identifier -  else -    Diag(Tok, diag::err_objc_missing_end); +  } else { +    Diag(Tok, diag::err_objc_missing_end) +        << FixItHint::CreateInsertion(Tok.getLocation(), "\n@end\n"); +    Diag(CDecl->getLocStart(), diag::note_objc_container_start) +        << (int) Actions.getObjCContainerKind(); +    AtEnd.setBegin(Tok.getLocation()); +    AtEnd.setEnd(Tok.getLocation()); +  }    // Insert collected methods declarations into the @interface object.    // This passes in an invalid SourceLocation for AtEndLoc when EOF is hit. @@ -1316,6 +1343,7 @@ Decl *Parser::ParseObjCAtProtocolDeclaration(SourceLocation AtLoc,                                               ParsedAttributes &attrs) {    assert(Tok.isObjCAtKeyword(tok::objc_protocol) &&           "ParseObjCAtProtocolDeclaration(): Expected @protocol"); +  CheckNestedObjCContexts(AtLoc);    ConsumeToken(); // the "protocol" identifier    if (Tok.is(tok::code_completion)) { @@ -1399,10 +1427,10 @@ Decl *Parser::ParseObjCAtProtocolDeclaration(SourceLocation AtLoc,  ///  ///   objc-category-implementation-prologue:  ///     @implementation identifier ( identifier ) -Decl *Parser::ParseObjCAtImplementationDeclaration( -  SourceLocation atLoc) { +Decl *Parser::ParseObjCAtImplementationDeclaration(SourceLocation AtLoc) {    assert(Tok.isObjCAtKeyword(tok::objc_implementation) &&           "ParseObjCAtImplementationDeclaration(): Expected @implementation"); +  CheckNestedObjCContexts(AtLoc);    ConsumeToken(); // the "implementation" identifier    // Code completion after '@implementation'. @@ -1446,7 +1474,7 @@ Decl *Parser::ParseObjCAtImplementationDeclaration(      }      rparenLoc = ConsumeParen();      Decl *ImplCatType = Actions.ActOnStartCategoryImplementation( -                                    atLoc, nameId, nameLoc, categoryId, +                                    AtLoc, nameId, nameLoc, categoryId,                                      categoryLoc);      ObjCImpDecl = ImplCatType; @@ -1467,11 +1495,11 @@ Decl *Parser::ParseObjCAtImplementationDeclaration(      superClassLoc = ConsumeToken(); // Consume super class name    }    Decl *ImplClsType = Actions.ActOnStartClassImplementation( -                                  atLoc, nameId, nameLoc, +                                  AtLoc, nameId, nameLoc,                                    superClassId, superClassLoc);    if (Tok.is(tok::l_brace)) // we have ivars -    ParseObjCClassInstanceVariables(ImplClsType, tok::objc_private, atLoc); +    ParseObjCClassInstanceVariables(ImplClsType, tok::objc_private, AtLoc);    ObjCImpDecl = ImplClsType;    PendingObjCImpDecl.push_back(ObjCImpDecl); @@ -1498,7 +1526,7 @@ Parser::ParseObjCAtEndDeclaration(SourceRange atEnd) {    }    else      // missing @implementation -    Diag(atEnd.getBegin(), diag::err_expected_implementation); +    Diag(atEnd.getBegin(), diag::err_expected_objc_container);    clearLateParsedObjCMethods();    ObjCImpDecl = 0; @@ -1510,8 +1538,15 @@ Parser::DeclGroupPtrTy Parser::FinishPendingObjCActions() {    Actions.DiagnoseUseOfUnimplementedSelectors();    if (PendingObjCImpDecl.empty())      return Actions.ConvertDeclToDeclGroup(0); +    Decl *ImpDecl = PendingObjCImpDecl.pop_back_val(); -  Actions.ActOnAtEnd(getCurScope(), SourceRange()); +  Actions.ActOnAtEnd(getCurScope(), SourceRange(Tok.getLocation())); +  Diag(Tok, diag::err_objc_missing_end) +      << FixItHint::CreateInsertion(Tok.getLocation(), "\n@end\n"); +  if (ImpDecl) +    Diag(ImpDecl->getLocStart(), diag::note_objc_container_start) +        << Sema::OCK_Implementation; +    return Actions.ConvertDeclToDeclGroup(ImpDecl);  } diff --git a/clang/lib/Sema/SemaDeclObjC.cpp b/clang/lib/Sema/SemaDeclObjC.cpp index afcf3cc63ae..b271ae66a10 100644 --- a/clang/lib/Sema/SemaDeclObjC.cpp +++ b/clang/lib/Sema/SemaDeclObjC.cpp @@ -2148,15 +2148,39 @@ void Sema::DiagnoseDuplicateIvars(ObjCInterfaceDecl *ID,    }  } +Sema::ObjCContainerKind Sema::getObjCContainerKind() const { +  switch (CurContext->getDeclKind()) { +    case Decl::ObjCInterface: +      return Sema::OCK_Interface; +    case Decl::ObjCProtocol: +      return Sema::OCK_Protocol; +    case Decl::ObjCCategory: +      if (dyn_cast<ObjCCategoryDecl>(CurContext)->IsClassExtension()) +        return Sema::OCK_ClassExtension; +      else +        return Sema::OCK_Category; +    case Decl::ObjCImplementation: +      return Sema::OCK_Implementation; +    case Decl::ObjCCategoryImpl: +      return Sema::OCK_CategoryImplementation; + +    default: +      return Sema::OCK_None; +  } +} +  // Note: For class/category implemenations, allMethods/allProperties is  // always null. -void Sema::ActOnAtEnd(Scope *S, SourceRange AtEnd, -                      Decl **allMethods, unsigned allNum, -                      Decl **allProperties, unsigned pNum, -                      DeclGroupPtrTy *allTUVars, unsigned tuvNum) { +Decl *Sema::ActOnAtEnd(Scope *S, SourceRange AtEnd, +                       Decl **allMethods, unsigned allNum, +                       Decl **allProperties, unsigned pNum, +                       DeclGroupPtrTy *allTUVars, unsigned tuvNum) { + +  if (getObjCContainerKind() == Sema::OCK_None) +    return 0; + +  assert(AtEnd.isValid() && "Invalid location for '@end'"); -  if (!CurContext->isObjCContainer()) -    return;    ObjCContainerDecl *OCD = dyn_cast<ObjCContainerDecl>(CurContext);    Decl *ClassDecl = cast<Decl>(OCD); @@ -2165,15 +2189,6 @@ void Sema::ActOnAtEnd(Scope *S, SourceRange AtEnd,           || isa<ObjCProtocolDecl>(ClassDecl);    bool checkIdenticalMethods = isa<ObjCImplementationDecl>(ClassDecl); -  if (!isInterfaceDeclKind && AtEnd.isInvalid()) { -    // FIXME: This is wrong.  We shouldn't be pretending that there is -    //  an '@end' in the declaration. -    SourceLocation L = OCD->getAtStartLoc(); -    AtEnd.setBegin(L); -    AtEnd.setEnd(L); -    Diag(L, diag::err_missing_atend); -  } -      // FIXME: Remove these and use the ObjCContainerDecl/DeclContext.    llvm::DenseMap<Selector, const ObjCMethodDecl*> InsMap;    llvm::DenseMap<Selector, const ObjCMethodDecl*> ClsMap; @@ -2335,6 +2350,8 @@ void Sema::ActOnAtEnd(Scope *S, SourceRange AtEnd,        (*I)->setTopLevelDeclInObjCContainer();      Consumer.HandleTopLevelDeclInObjCContainer(DG);    } + +  return ClassDecl;  } | 

