diff options
author | Argyrios Kyrtzidis <akyrtzi@gmail.com> | 2013-12-03 21:11:36 +0000 |
---|---|---|
committer | Argyrios Kyrtzidis <akyrtzi@gmail.com> | 2013-12-03 21:11:36 +0000 |
commit | 22bfa2c28bdbca09949ba4cf8670c310d3e42c71 (patch) | |
tree | 27e9efff069b7c1a7bb8b948c83913b77a67fd76 /clang/lib | |
parent | 9ed9e5f31c1fe946a5c6c1ffb651052925570fd4 (diff) | |
download | bcm5719-llvm-22bfa2c28bdbca09949ba4cf8670c310d3e42c71.tar.gz bcm5719-llvm-22bfa2c28bdbca09949ba4cf8670c310d3e42c71.zip |
[objc] Emit a warning when the implementation of a designated initializer does not chain to
an init method that is a designated initializer for the superclass.
llvm-svn: 196316
Diffstat (limited to 'clang/lib')
-rw-r--r-- | clang/lib/AST/DeclObjC.cpp | 45 | ||||
-rw-r--r-- | clang/lib/Sema/ScopeInfo.cpp | 4 | ||||
-rw-r--r-- | clang/lib/Sema/SemaDecl.cpp | 12 | ||||
-rw-r--r-- | clang/lib/Sema/SemaDeclObjC.cpp | 6 | ||||
-rw-r--r-- | clang/lib/Sema/SemaExprObjC.cpp | 10 |
5 files changed, 75 insertions, 2 deletions
diff --git a/clang/lib/AST/DeclObjC.cpp b/clang/lib/AST/DeclObjC.cpp index 528b52fe5d5..cb58ad482e1 100644 --- a/clang/lib/AST/DeclObjC.cpp +++ b/clang/lib/AST/DeclObjC.cpp @@ -398,12 +398,39 @@ void ObjCInterfaceDecl::getDesignatedInitializers( for (instmeth_iterator I = IFace->instmeth_begin(), E = IFace->instmeth_end(); I != E; ++I) { const ObjCMethodDecl *MD = *I; - if (MD->getMethodFamily() == OMF_init && - MD->hasAttr<ObjCDesignatedInitializerAttr>()) + if (MD->isThisDeclarationADesignatedInitializer()) Methods.push_back(MD); } } +bool ObjCInterfaceDecl::isDesignatedInitializer(Selector Sel, + const ObjCMethodDecl **InitMethod) const { + assert(hasDefinition()); + if (data().ExternallyCompleted) + LoadExternalDefinition(); + + const ObjCInterfaceDecl *IFace = this; + while (IFace) { + if (IFace->data().HasDesignatedInitializers) + break; + IFace = IFace->getSuperClass(); + } + + if (!IFace) + return false; + + if (const ObjCMethodDecl *MD = IFace->lookupMethod(Sel, /*isInstance=*/true, + /*shallowCategoryLookup=*/true, + /*followSuper=*/false)) { + if (MD->isThisDeclarationADesignatedInitializer()) { + if (InitMethod) + *InitMethod = MD; + return true; + } + } + return false; +} + void ObjCInterfaceDecl::allocateDefinitionData() { assert(!hasDefinition() && "ObjC class already has a definition"); Data.setPointer(new (getASTContext()) DefinitionData()); @@ -623,6 +650,20 @@ ObjCMethodDecl *ObjCMethodDecl::CreateDeserialized(ASTContext &C, unsigned ID) { Selector(), QualType(), 0, 0); } +bool ObjCMethodDecl::isThisDeclarationADesignatedInitializer() const { + return getMethodFamily() == OMF_init && + hasAttr<ObjCDesignatedInitializerAttr>(); +} + +bool ObjCMethodDecl::isDesignatedInitializerForTheInterface() const { + const DeclContext *DC = getDeclContext(); + if (isa<ObjCProtocolDecl>(DC)) + return false; + if (const ObjCInterfaceDecl *ID = getClassInterface()) + return ID->isDesignatedInitializer(getSelector()); + return false; +} + Stmt *ObjCMethodDecl::getBody() const { return Body.get(getASTContext().getExternalSource()); } diff --git a/clang/lib/Sema/ScopeInfo.cpp b/clang/lib/Sema/ScopeInfo.cpp index 8b3493ebfef..cf2b5ebe0e5 100644 --- a/clang/lib/Sema/ScopeInfo.cpp +++ b/clang/lib/Sema/ScopeInfo.cpp @@ -26,6 +26,10 @@ void FunctionScopeInfo::Clear() { HasBranchProtectedScope = false; HasBranchIntoScope = false; HasIndirectGoto = false; + HasDroppedStmt = false; + ObjCShouldCallSuper = false; + ObjCIsDesignatedInit = false; + ObjCWarnForNoDesignatedInitChain = false; SwitchStack.clear(); Returns.clear(); diff --git a/clang/lib/Sema/SemaDecl.cpp b/clang/lib/Sema/SemaDecl.cpp index 8229a6224a2..c14cc4b8619 100644 --- a/clang/lib/Sema/SemaDecl.cpp +++ b/clang/lib/Sema/SemaDecl.cpp @@ -9814,6 +9814,18 @@ Decl *Sema::ActOnFinishFunctionBody(Decl *dcl, Stmt *Body, << MD->getSelector().getAsString(); getCurFunction()->ObjCShouldCallSuper = false; } + if (getCurFunction()->ObjCWarnForNoDesignatedInitChain) { + const ObjCMethodDecl *InitMethod = 0; + bool isDesignated = MD->getClassInterface() + ->isDesignatedInitializer(MD->getSelector(), &InitMethod); + assert(isDesignated && InitMethod); + (void)isDesignated; + Diag(MD->getLocation(), + diag::warn_objc_designated_init_missing_super_call); + Diag(InitMethod->getLocation(), + diag::note_objc_designated_init_marked_here); + getCurFunction()->ObjCWarnForNoDesignatedInitChain = false; + } } else { return 0; } diff --git a/clang/lib/Sema/SemaDeclObjC.cpp b/clang/lib/Sema/SemaDeclObjC.cpp index d9b6378269a..53c11d59e66 100644 --- a/clang/lib/Sema/SemaDeclObjC.cpp +++ b/clang/lib/Sema/SemaDeclObjC.cpp @@ -391,6 +391,9 @@ void Sema::ActOnStartOfObjCMethodDef(Scope *FnBodyScope, Decl *D) { MDecl->getLocation(), 0); } + if (MDecl->isDesignatedInitializerForTheInterface()) + getCurFunction()->ObjCIsDesignatedInit = true; + // If this is "dealloc" or "finalize", set some bit here. // Then in ActOnSuperMessage() (SemaExprObjC), set it back to false. // Finally, in ActOnFinishFunctionBody() (SemaDecl), warn if flag is set. @@ -413,6 +416,9 @@ void Sema::ActOnStartOfObjCMethodDef(Scope *FnBodyScope, Decl *D) { getCurFunction()->ObjCShouldCallSuper = (SuperMethod && SuperMethod->hasAttr<ObjCRequiresSuperAttr>()); } + + if (getCurFunction()->ObjCIsDesignatedInit) + getCurFunction()->ObjCWarnForNoDesignatedInitChain = true; } } } diff --git a/clang/lib/Sema/SemaExprObjC.cpp b/clang/lib/Sema/SemaExprObjC.cpp index a60749b0224..20f4b4f7248 100644 --- a/clang/lib/Sema/SemaExprObjC.cpp +++ b/clang/lib/Sema/SemaExprObjC.cpp @@ -2447,6 +2447,16 @@ ExprResult Sema::BuildInstanceMessage(Expr *Receiver, } } + if (SuperLoc.isValid() && getCurFunction()->ObjCIsDesignatedInit) { + if (const ObjCObjectPointerType * + OCIType = ReceiverType->getAsObjCInterfacePointerType()) { + if (const ObjCInterfaceDecl *ID = OCIType->getInterfaceDecl()) { + if (ID->isDesignatedInitializer(Sel)) + getCurFunction()->ObjCWarnForNoDesignatedInitChain = false; + } + } + } + // Check the message arguments. unsigned NumArgs = ArgsIn.size(); Expr **Args = ArgsIn.data(); |