AddFactoryMethodToGlobalPool(MDecl, true);
}
+/// HasExplicitOwnershipAttr - returns true when pointer to ObjC pointer
+/// has explicit ownership attribute; false otherwise.
+static bool
+HasExplicitOwnershipAttr(Sema &S, ParmVarDecl *Param) {
+ QualType T = Param->getType();
+
+ if (!T->isPointerType() && !T->isReferenceType())
+ return true;
+
+ if (const PointerType *PT = T->getAs<PointerType>()) {
+ T = PT->getPointeeType();
+ } else if (const ReferenceType *RT = T->getAs<ReferenceType>()) {
+ T = RT->getPointeeType();
+ } else {
+ return true;
+ }
+
+ // If we have a lifetime qualifier, but it's local, we must have
+ // inferred it. So, it is implicit.
+ return !T.getLocalQualifiers().hasObjCLifetime();
+}
+
/// ActOnStartOfObjCMethodDef - This routine sets up parameters; invisible
/// and user declared, in the method definition's AST.
void Sema::ActOnStartOfObjCMethodDef(Scope *FnBodyScope, Decl *D) {
RequireCompleteType(Param->getLocation(), Param->getType(),
diag::err_typecheck_decl_incomplete_type))
Param->setInvalidDecl();
+ if (!Param->isInvalidDecl() &&
+ getLangOpts().ObjCAutoRefCount &&
+ !HasExplicitOwnershipAttr(*this, Param))
+ Diag(Param->getLocation(), diag::warn_arc_strong_pointer_objc_pointer) <<
+ Param->getType();
if ((*PI)->getIdentifier())
PushOnScopeChains(*PI, FnBodyScope);
return TInfo;
}
-
-/// checkImplicitObjCParamAttribute - diagnoses when pointer to ObjC pointer
-/// has implicit ownership attribute.
-static void
-checkImplicitObjCParamAttribute(Sema &S, Declarator &D, QualType T) {
- if (!S.getLangOpts().ObjCAutoRefCount ||
- !S.OriginalLexicalContext ||
- (S.OriginalLexicalContext->getDeclKind() != Decl::ObjCImplementation &&
- S.OriginalLexicalContext->getDeclKind() != Decl::ObjCCategoryImpl))
- return;
-
- if (!T->isObjCIndirectLifetimeType())
- return;
- if (!T->isPointerType() && !T->isReferenceType())
- return;
- QualType OrigT = T;
- T = T->isPointerType()
- ? T->getAs<PointerType>()->getPointeeType()
- : T->getAs<ReferenceType>()->getPointeeType();
- if (T->isObjCLifetimeType()) {
- // when lifetime is Qualifiers::OCL_None it means that it has
- // no implicit ownership qualifier (which means it is explicit).
- Qualifiers::ObjCLifetime lifetime =
- T.getLocalQualifiers().getObjCLifetime();
- if (lifetime != Qualifiers::OCL_None)
- S.Diag(D.getLocStart(), diag::warn_arc_strong_pointer_objc_pointer)
- << OrigT;
- }
-}
-
/// \brief Create a LocInfoType to hold the given QualType and TypeSourceInfo.
ParsedType Sema::CreateParsedType(QualType T, TypeSourceInfo *TInfo) {
// FIXME: LocInfoTypes are "transient", only needed for passing to/from Parser
// to apply them to the actual parameter declaration.
if (D.getContext() != Declarator::ObjCParameterContext)
checkUnusedDeclAttributes(D);
- else
- checkImplicitObjCParamAttribute(*this, D, T);
if (getLangOpts().CPlusPlus) {
// Check that there are no default arguments (C++ only).