]> granicus.if.org Git - clang/commitdiff
Warn if forward class is used as a receiver.
authorFariborz Jahanian <fjahanian@apple.com>
Fri, 8 May 2009 23:02:36 +0000 (23:02 +0000)
committerFariborz Jahanian <fjahanian@apple.com>
Fri, 8 May 2009 23:02:36 +0000 (23:02 +0000)
git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@71278 91177308-0d34-0410-b5e6-96231b3b80d8

include/clang/Basic/DiagnosticSemaKinds.td
lib/Sema/SemaExprObjC.cpp
test/SemaObjC/forward-class-receiver.m [new file with mode: 0644]

index a8d079d720a7810149e0fb7a73f1bc0c92a5dc62..f491a43e3b6e6c9f0755574c3910021d47816e4c 100644 (file)
@@ -1690,6 +1690,8 @@ def err_collection_expr_type : Error<
 
 // Type
 def ext_invalid_sign_spec : Extension<"'%0' cannot be signed or unsigned">;
+def warn_receiver_forward_class : Warning<
+    "receiver %0 is a forward class and corresponding @interface may not exist">;
 def warn_missing_declspec : Warning<
   "declaration specifier missing, defaulting to 'int'">;
 def warn_missing_type_specifier : Warning<
index e8ff18ba0c6743c0df9fff6bcb173037c47c7fbe..3dbc2cf015f06189f6eeb672803c012787ad8f05 100644 (file)
@@ -422,7 +422,15 @@ Sema::ExprResult Sema::ActOnClassMessage(
   assert(ClassDecl && "missing interface declaration");
   ObjCMethodDecl *Method = 0;
   QualType returnType;
-  Method = ClassDecl->lookupClassMethod(Context, Sel);
+  if (ClassDecl->isForwardDecl()) {
+    // A forward class used in messaging is tread as a 'Class'
+    Method = LookupFactoryMethodInGlobalPool(Sel, SourceRange(lbrac,rbrac));
+    if (Method)
+      Diag(lbrac, diag::warn_receiver_forward_class) 
+        << ClassDecl->getDeclName();
+  }
+  if (!Method)
+    Method = ClassDecl->lookupClassMethod(Context, Sel);
   
   // If we have an implementation in scope, check "private" methods.
   if (!Method)
diff --git a/test/SemaObjC/forward-class-receiver.m b/test/SemaObjC/forward-class-receiver.m
new file mode 100644 (file)
index 0000000..8353f39
--- /dev/null
@@ -0,0 +1,13 @@
+// RUN: clang-cc  -fsyntax-only -verify %s
+
+@interface I
++ new;
+@end
+Class isa;
+
+@class NotKnown;
+
+void foo(NotKnown *n) {
+  [isa new];
+  [NotKnown new];         /* expected-warning {{receiver 'NotKnown' is a forward class and corresponding}} */
+}