summaryrefslogtreecommitdiffstats
path: root/clang
diff options
context:
space:
mode:
authorFariborz Jahanian <fjahanian@apple.com>2012-09-17 19:15:26 +0000
committerFariborz Jahanian <fjahanian@apple.com>2012-09-17 19:15:26 +0000
commitf4ffdf357c98e30fd8a67d41ab51e55932f8d2d4 (patch)
tree69301404a4e37a67f4877489dcd4b1d064f76942 /clang
parent42782343606a4c50fb5f237bbda37823b67e845b (diff)
downloadbcm5719-llvm-f4ffdf357c98e30fd8a67d41ab51e55932f8d2d4.tar.gz
bcm5719-llvm-f4ffdf357c98e30fd8a67d41ab51e55932f8d2d4.zip
objective-C: issue warning when there is no whitespace
between objc method parameter name and colon. // rdar://12263549 llvm-svn: 164047
Diffstat (limited to 'clang')
-rw-r--r--clang/include/clang/Basic/DiagnosticParseKinds.td6
-rw-r--r--clang/lib/Parse/ParseObjc.cpp26
-rw-r--r--clang/test/SemaObjC/unused.m5
-rw-r--r--clang/test/SemaObjC/warning-missing-selector-name.m20
4 files changed, 54 insertions, 3 deletions
diff --git a/clang/include/clang/Basic/DiagnosticParseKinds.td b/clang/include/clang/Basic/DiagnosticParseKinds.td
index e4e339aefc4..218a6a31da4 100644
--- a/clang/include/clang/Basic/DiagnosticParseKinds.td
+++ b/clang/include/clang/Basic/DiagnosticParseKinds.td
@@ -213,6 +213,12 @@ def err_expected_semi_after_static_assert : Error<
"expected ';' after static_assert">;
def err_expected_semi_for : Error<"expected ';' in 'for' statement specifier">;
def err_expected_colon_after : Error<"expected ':' after %0">;
+def missing_selector_name : Warning<
+ "parameter name used as selector"
+ " may result in incomplete method selector name">,
+ InGroup<DiagGroup<"missing-argument-name-in-selector">>;
+def note_missing_argument_name : Note<
+ "did you mean to use %0 as the selector name instead of %1">;
def err_label_end_of_compound_statement : Error<
"label at end of compound statement: expected statement">;
def err_address_of_label_outside_fn : Error<
diff --git a/clang/lib/Parse/ParseObjc.cpp b/clang/lib/Parse/ParseObjc.cpp
index 1465a730e44..48c5efac6ba 100644
--- a/clang/lib/Parse/ParseObjc.cpp
+++ b/clang/lib/Parse/ParseObjc.cpp
@@ -18,6 +18,7 @@
#include "clang/Sema/PrettyDeclStackTrace.h"
#include "clang/Sema/Scope.h"
#include "llvm/ADT/SmallVector.h"
+#include "llvm/ADT/StringExtras.h"
using namespace clang;
@@ -1031,7 +1032,7 @@ Decl *Parser::ParseObjCMethodDecl(SourceLocation mLoc,
Scope::FunctionPrototypeScope|Scope::DeclScope);
AttributePool allParamAttrs(AttrFactory);
-
+ bool warnSelectorName = false;
while (1) {
ParsedAttributes paramAttrs(AttrFactory);
Sema::ObjCArgInfo ArgInfo;
@@ -1102,6 +1103,13 @@ Decl *Parser::ParseObjCMethodDecl(SourceLocation mLoc,
SelIdent = ParseObjCSelectorPiece(selLoc);
if (!SelIdent && Tok.isNot(tok::colon))
break;
+ if (MethodDefinition && !SelIdent) {
+ SourceLocation ColonLoc = Tok.getLocation();
+ if (PP.getLocForEndOfToken(ArgInfo.NameLoc) == ColonLoc) {
+ warnSelectorName = true;
+ Diag(ArgInfo.NameLoc, diag::missing_selector_name);
+ }
+ }
// We have a selector or a colon, continue parsing.
}
@@ -1142,6 +1150,22 @@ Decl *Parser::ParseObjCMethodDecl(SourceLocation mLoc,
Selector Sel = PP.getSelectorTable().getSelector(KeyIdents.size(),
&KeyIdents[0]);
+ if (warnSelectorName) {
+ SmallVector<IdentifierInfo *, 12> DiagKeyIdents;
+ for (unsigned i = 0, size = KeyIdents.size(); i < size; i++)
+ if (KeyIdents[i])
+ DiagKeyIdents.push_back(KeyIdents[i]);
+ else {
+ std::string name = "Name";
+ name += llvm::utostr(i+1);
+ IdentifierInfo *NamedMissingId = &PP.getIdentifierTable().get(name);
+ DiagKeyIdents.push_back(NamedMissingId);
+ }
+ Selector NewSel =
+ PP.getSelectorTable().getSelector(DiagKeyIdents.size(), &DiagKeyIdents[0]);
+ Diag(mLoc, diag::note_missing_argument_name)
+ << NewSel.getAsString() << Sel.getAsString();
+ }
Decl *Result
= Actions.ActOnMethodDeclaration(getCurScope(), mLoc, Tok.getLocation(),
mType, DSRet, ReturnType,
diff --git a/clang/test/SemaObjC/unused.m b/clang/test/SemaObjC/unused.m
index efaf9c8f3ef..23d7e8cbe50 100644
--- a/clang/test/SemaObjC/unused.m
+++ b/clang/test/SemaObjC/unused.m
@@ -29,8 +29,9 @@ void test2() {
@end
@implementation foo
-- (int) meth: (int)x:
-(int)y: // expected-warning{{unused}}
+- (int) meth: (int)x: // expected-warning {{parameter name used as selector may result in incomplete method selector name}} \
+ // expected-note {{did you mean to use meth:Name2:Name3: as the selector name instead of meth:::}}
+(int)y: // expected-warning{{unused}} expected-warning {{parameter name used as selector may result in incomplete method selector name}}
(int) __attribute__((unused))z { return x; }
@end
diff --git a/clang/test/SemaObjC/warning-missing-selector-name.m b/clang/test/SemaObjC/warning-missing-selector-name.m
new file mode 100644
index 00000000000..e4bde0b7466
--- /dev/null
+++ b/clang/test/SemaObjC/warning-missing-selector-name.m
@@ -0,0 +1,20 @@
+// RUN: %clang_cc1 -fsyntax-only -verify -Wno-objc-root-class %s
+// RUN: %clang_cc1 -x objective-c++ -fsyntax-only -verify -Wno-objc-root-class -Wmissing-argument-name-in-selector %s
+// rdar://12263549
+
+@interface Super @end
+@interface INTF : Super
+-(void) Name1:(id)Arg1 Name2:(id)Arg2; // Name1:Name2:
+-(void) Name1:(id) Name2:(id)Arg2;
+-(void) Name1:(id)Arg1 Name2:(id)Arg2 Name3:(id)Arg3; // Name1:Name2:Name3:
+-(void) Name1:(id)Arg1 Name2:(id) Name3:(id)Arg3;
+@end
+
+@implementation INTF
+-(void) Name1:(id)Arg1 Name2:(id)Arg2{}
+-(void) Name1:(id) Name2:(id)Arg2 {} // expected-warning {{parameter name used as selector may result in incomplete method selector name}} \
+ // expected-note {{did you mean to use Name1:Name2: as the selector name instead of Name1::}}
+-(void) Name1:(id)Arg1 Name2:(id)Arg2 Name3:(id)Arg3 {}
+-(void) Name1:(id)Arg1 Name2:(id) Name3:(id)Arg3 {} // expected-warning {{parameter name used as selector may result in incomplete method selector name}} \
+ // expected-note {{did you mean to use Name1:Name2:Name3: as the selector name instead of Name1:Name2::}}
+@end
OpenPOWER on IntegriCloud