<div dir="ltr">Rather than hardcoding all these symbol names in the compiler (CFStringCreateWithFormatAndArguments etc), maybe it'd be nicer to provide an attribute that has this effect and then make the declarations of these functions use the new attribute?</div><div class="gmail_extra"><br><div class="gmail_quote">On Tue, Sep 9, 2014 at 4:10 PM, Fariborz Jahanian <span dir="ltr"><<a href="mailto:fjahanian@apple.com" target="_blank">fjahanian@apple.com</a>></span> wrote:<br><blockquote class="gmail_quote" style="margin:0 0 0 .8ex;border-left:1px #ccc solid;padding-left:1ex">Author: fjahanian<br>
Date: Tue Sep  9 18:10:54 2014<br>
New Revision: 217467<br>
<br>
URL: <a href="http://llvm.org/viewvc/llvm-project?rev=217467&view=rev" target="_blank">http://llvm.org/viewvc/llvm-project?rev=217467&view=rev</a><br>
Log:<br>
Objective-C. Under a special flag, -Wcstring-format-directive,<br>
off by default, issue a warning if %s directive is used in<br>
certain CF/NS formatting APIs, to assist user in deprecating<br>
use of such %s in these APIs. rdar://18182443<br>
<br>
Added:<br>
    cfe/trunk/test/SemaObjC/format-cstrings-warning.m<br>
Modified:<br>
    cfe/trunk/include/clang/AST/Decl.h<br>
    cfe/trunk/include/clang/Analysis/Analyses/FormatString.h<br>
    cfe/trunk/include/clang/Basic/DiagnosticGroups.td<br>
    cfe/trunk/include/clang/Basic/DiagnosticSemaKinds.td<br>
    cfe/trunk/include/clang/Basic/IdentifierTable.h<br>
    cfe/trunk/include/clang/Sema/Sema.h<br>
    cfe/trunk/lib/AST/Decl.cpp<br>
    cfe/trunk/lib/Analysis/PrintfFormatString.cpp<br>
    cfe/trunk/lib/Basic/IdentifierTable.cpp<br>
    cfe/trunk/lib/Sema/SemaChecking.cpp<br>
    cfe/trunk/lib/Sema/SemaExprObjC.cpp<br>
<br>
Modified: cfe/trunk/include/clang/AST/Decl.h<br>
URL: <a href="http://llvm.org/viewvc/llvm-project/cfe/trunk/include/clang/AST/Decl.h?rev=217467&r1=217466&r2=217467&view=diff" target="_blank">http://llvm.org/viewvc/llvm-project/cfe/trunk/include/clang/AST/Decl.h?rev=217467&r1=217466&r2=217467&view=diff</a><br>
==============================================================================<br>
--- cfe/trunk/include/clang/AST/Decl.h (original)<br>
+++ cfe/trunk/include/clang/AST/Decl.h Tue Sep  9 18:10:54 2014<br>
@@ -289,6 +289,8 @@ public:<br>
     return const_cast<NamedDecl*>(this)->getMostRecentDecl();<br>
   }<br>
<br>
+  ObjCStringFormatFamily getObjCFStringFormattingFamily() const;<br>
+<br>
   static bool classof(const Decl *D) { return classofKind(D->getKind()); }<br>
   static bool classofKind(Kind K) { return K >= firstNamed && K <= lastNamed; }<br>
 };<br>
<br>
Modified: cfe/trunk/include/clang/Analysis/Analyses/FormatString.h<br>
URL: <a href="http://llvm.org/viewvc/llvm-project/cfe/trunk/include/clang/Analysis/Analyses/FormatString.h?rev=217467&r1=217466&r2=217467&view=diff" target="_blank">http://llvm.org/viewvc/llvm-project/cfe/trunk/include/clang/Analysis/Analyses/FormatString.h?rev=217467&r1=217466&r2=217467&view=diff</a><br>
==============================================================================<br>
--- cfe/trunk/include/clang/Analysis/Analyses/FormatString.h (original)<br>
+++ cfe/trunk/include/clang/Analysis/Analyses/FormatString.h Tue Sep  9 18:10:54 2014<br>
@@ -647,6 +647,9 @@ public:<br>
 bool ParsePrintfString(FormatStringHandler &H,<br>
                        const char *beg, const char *end, const LangOptions &LO,<br>
                        const TargetInfo &Target);<br>
+<br>
+bool ParseFormatStringHasSArg(const char *beg, const char *end, const LangOptions &LO,<br>
+                              const TargetInfo &Target);<br>
<br>
 bool ParseScanfString(FormatStringHandler &H,<br>
                       const char *beg, const char *end, const LangOptions &LO,<br>
<br>
Modified: cfe/trunk/include/clang/Basic/DiagnosticGroups.td<br>
URL: <a href="http://llvm.org/viewvc/llvm-project/cfe/trunk/include/clang/Basic/DiagnosticGroups.td?rev=217467&r1=217466&r2=217467&view=diff" target="_blank">http://llvm.org/viewvc/llvm-project/cfe/trunk/include/clang/Basic/DiagnosticGroups.td?rev=217467&r1=217466&r2=217467&view=diff</a><br>
==============================================================================<br>
--- cfe/trunk/include/clang/Basic/DiagnosticGroups.td (original)<br>
+++ cfe/trunk/include/clang/Basic/DiagnosticGroups.td Tue Sep  9 18:10:54 2014<br>
@@ -350,6 +350,7 @@ def InvalidOffsetof : DiagGroup<"invalid<br>
 def : DiagGroup<"strict-prototypes">;<br>
 def StrictSelector : DiagGroup<"strict-selector-match">;<br>
 def MethodDuplicate : DiagGroup<"duplicate-method-match">;<br>
+def ObjCCStringFormat : DiagGroup<"cstring-format-directive">;<br>
 def CoveredSwitchDefault : DiagGroup<"covered-switch-default">;<br>
 def SwitchBool     : DiagGroup<"switch-bool">;<br>
 def SwitchEnum     : DiagGroup<"switch-enum">;<br>
<br>
Modified: cfe/trunk/include/clang/Basic/DiagnosticSemaKinds.td<br>
URL: <a href="http://llvm.org/viewvc/llvm-project/cfe/trunk/include/clang/Basic/DiagnosticSemaKinds.td?rev=217467&r1=217466&r2=217467&view=diff" target="_blank">http://llvm.org/viewvc/llvm-project/cfe/trunk/include/clang/Basic/DiagnosticSemaKinds.td?rev=217467&r1=217466&r2=217467&view=diff</a><br>
==============================================================================<br>
--- cfe/trunk/include/clang/Basic/DiagnosticSemaKinds.td (original)<br>
+++ cfe/trunk/include/clang/Basic/DiagnosticSemaKinds.td Tue Sep  9 18:10:54 2014<br>
@@ -719,6 +719,11 @@ def err_duplicate_method_decl : Error<"d<br>
 def warn_duplicate_method_decl :<br>
   Warning<"multiple declarations of method %0 found and ignored">,<br>
   InGroup<MethodDuplicate>, DefaultIgnore;<br>
+def warn_objc_cdirective_format_string :<br>
+  Warning<"using %0 directive in %select{NSString|CFString}1 "<br>
+          "which is being passed as a formatting argument to the formatting "<br>
+          "%select{method|CFfunction}2">,<br>
+  InGroup<ObjCCStringFormat>, DefaultIgnore;<br>
 def err_objc_var_decl_inclass :<br>
     Error<"cannot declare variable inside @interface or @protocol">;<br>
 def error_missing_method_context : Error<<br>
<br>
Modified: cfe/trunk/include/clang/Basic/IdentifierTable.h<br>
URL: <a href="http://llvm.org/viewvc/llvm-project/cfe/trunk/include/clang/Basic/IdentifierTable.h?rev=217467&r1=217466&r2=217467&view=diff" target="_blank">http://llvm.org/viewvc/llvm-project/cfe/trunk/include/clang/Basic/IdentifierTable.h?rev=217467&r1=217466&r2=217467&view=diff</a><br>
==============================================================================<br>
--- cfe/trunk/include/clang/Basic/IdentifierTable.h (original)<br>
+++ cfe/trunk/include/clang/Basic/IdentifierTable.h Tue Sep  9 18:10:54 2014<br>
@@ -589,6 +589,12 @@ enum ObjCInstanceTypeFamily {<br>
   OIT_ReturnsSelf<br>
 };<br>
<br>
+enum ObjCStringFormatFamily {<br>
+  SFF_None,<br>
+  SFF_NSString,<br>
+  SFF_CFString<br>
+};<br>
+<br>
 /// \brief Smart pointer class that efficiently represents Objective-C method<br>
 /// names.<br>
 ///<br>
@@ -634,6 +640,8 @@ class Selector {<br>
   }<br>
<br>
   static ObjCMethodFamily getMethodFamilyImpl(Selector sel);<br>
+<br>
+  static ObjCStringFormatFamily getStringFormatFamilyImpl(Selector sel);<br>
<br>
 public:<br>
   friend class SelectorTable; // only the SelectorTable can create these<br>
@@ -704,7 +712,11 @@ public:<br>
   ObjCMethodFamily getMethodFamily() const {<br>
     return getMethodFamilyImpl(*this);<br>
   }<br>
-<br>
+<br>
+  ObjCStringFormatFamily getStringFormatFamily() const {<br>
+    return getStringFormatFamilyImpl(*this);<br>
+  }<br>
+<br>
   static Selector getEmptyMarker() {<br>
     return Selector(uintptr_t(-1));<br>
   }<br>
<br>
Modified: cfe/trunk/include/clang/Sema/Sema.h<br>
URL: <a href="http://llvm.org/viewvc/llvm-project/cfe/trunk/include/clang/Sema/Sema.h?rev=217467&r1=217466&r2=217467&view=diff" target="_blank">http://llvm.org/viewvc/llvm-project/cfe/trunk/include/clang/Sema/Sema.h?rev=217467&r1=217466&r2=217467&view=diff</a><br>
==============================================================================<br>
--- cfe/trunk/include/clang/Sema/Sema.h (original)<br>
+++ cfe/trunk/include/clang/Sema/Sema.h Tue Sep  9 18:10:54 2014<br>
@@ -8403,6 +8403,8 @@ public:<br>
                          FormatStringType Type, bool inFunctionCall,<br>
                          VariadicCallType CallType,<br>
                          llvm::SmallBitVector &CheckedVarArgs);<br>
+<br>
+  bool FormatStringHasSArg(const StringLiteral *FExpr);<br>
<br>
 private:<br>
   bool CheckFormatArguments(const FormatAttr *Format,<br>
<br>
Modified: cfe/trunk/lib/AST/Decl.cpp<br>
URL: <a href="http://llvm.org/viewvc/llvm-project/cfe/trunk/lib/AST/Decl.cpp?rev=217467&r1=217466&r2=217467&view=diff" target="_blank">http://llvm.org/viewvc/llvm-project/cfe/trunk/lib/AST/Decl.cpp?rev=217467&r1=217466&r2=217467&view=diff</a><br>
==============================================================================<br>
--- cfe/trunk/lib/AST/Decl.cpp (original)<br>
+++ cfe/trunk/lib/AST/Decl.cpp Tue Sep  9 18:10:54 2014<br>
@@ -1001,6 +1001,19 @@ bool NamedDecl::isLinkageValid() const {<br>
          getCachedLinkage();<br>
 }<br>
<br>
+ObjCStringFormatFamily NamedDecl::getObjCFStringFormattingFamily() const {<br>
+  StringRef name = getName();<br>
+  if (name.empty()) return SFF_None;<br>
+<br>
+  if (name.front() == 'C')<br>
+    if (name == "CFStringCreateWithFormat" ||<br>
+        name == "CFStringCreateWithFormatAndArguments" ||<br>
+        name == "CFStringAppendFormat" ||<br>
+        name == "CFStringAppendFormatAndArguments")<br>
+      return SFF_CFString;<br>
+  return SFF_None;<br>
+}<br>
+<br>
 Linkage NamedDecl::getLinkageInternal() const {<br>
   // We don't care about visibility here, so ask for the cheapest<br>
   // possible visibility analysis.<br>
<br>
Modified: cfe/trunk/lib/Analysis/PrintfFormatString.cpp<br>
URL: <a href="http://llvm.org/viewvc/llvm-project/cfe/trunk/lib/Analysis/PrintfFormatString.cpp?rev=217467&r1=217466&r2=217467&view=diff" target="_blank">http://llvm.org/viewvc/llvm-project/cfe/trunk/lib/Analysis/PrintfFormatString.cpp?rev=217467&r1=217466&r2=217467&view=diff</a><br>
==============================================================================<br>
--- cfe/trunk/lib/Analysis/PrintfFormatString.cpp (original)<br>
+++ cfe/trunk/lib/Analysis/PrintfFormatString.cpp Tue Sep  9 18:10:54 2014<br>
@@ -54,7 +54,8 @@ static PrintfSpecifierResult ParsePrintf<br>
                                                   const char *E,<br>
                                                   unsigned &argIndex,<br>
                                                   const LangOptions &LO,<br>
-                                                  const TargetInfo &Target) {<br>
+                                                  const TargetInfo &Target,<br>
+                                                  bool Warn) {<br>
<br>
   using namespace clang::analyze_format_string;<br>
   using namespace clang::analyze_printf;<br>
@@ -83,7 +84,8 @@ static PrintfSpecifierResult ParsePrintf<br>
<br>
   if (I == E) {<br>
     // No more characters left?<br>
-    H.HandleIncompleteSpecifier(Start, E - Start);<br>
+    if (Warn)<br>
+      H.HandleIncompleteSpecifier(Start, E - Start);<br>
     return true;<br>
   }<br>
<br>
@@ -93,7 +95,8 @@ static PrintfSpecifierResult ParsePrintf<br>
<br>
   if (I == E) {<br>
     // No more characters left?<br>
-    H.HandleIncompleteSpecifier(Start, E - Start);<br>
+    if (Warn)<br>
+      H.HandleIncompleteSpecifier(Start, E - Start);<br>
     return true;<br>
   }<br>
<br>
@@ -118,7 +121,8 @@ static PrintfSpecifierResult ParsePrintf<br>
<br>
   if (I == E) {<br>
     // No more characters left?<br>
-    H.HandleIncompleteSpecifier(Start, E - Start);<br>
+    if (Warn)<br>
+      H.HandleIncompleteSpecifier(Start, E - Start);<br>
     return true;<br>
   }<br>
<br>
@@ -129,7 +133,8 @@ static PrintfSpecifierResult ParsePrintf<br>
<br>
   if (I == E) {<br>
     // No more characters left?<br>
-    H.HandleIncompleteSpecifier(Start, E - Start);<br>
+    if (Warn)<br>
+      H.HandleIncompleteSpecifier(Start, E - Start);<br>
     return true;<br>
   }<br>
<br>
@@ -137,7 +142,8 @@ static PrintfSpecifierResult ParsePrintf<br>
   if (*I == '.') {<br>
     ++I;<br>
     if (I == E) {<br>
-      H.HandleIncompleteSpecifier(Start, E - Start);<br>
+      if (Warn)<br>
+        H.HandleIncompleteSpecifier(Start, E - Start);<br>
       return true;<br>
     }<br>
<br>
@@ -147,7 +153,8 @@ static PrintfSpecifierResult ParsePrintf<br>
<br>
     if (I == E) {<br>
       // No more characters left?<br>
-      H.HandleIncompleteSpecifier(Start, E - Start);<br>
+      if (Warn)<br>
+        H.HandleIncompleteSpecifier(Start, E - Start);<br>
       return true;<br>
     }<br>
   }<br>
@@ -155,7 +162,8 @@ static PrintfSpecifierResult ParsePrintf<br>
   // Look for the length modifier.<br>
   if (ParseLengthModifier(FS, I, E, LO) && I == E) {<br>
     // No more characters left?<br>
-    H.HandleIncompleteSpecifier(Start, E - Start);<br>
+    if (Warn)<br>
+      H.HandleIncompleteSpecifier(Start, E - Start);<br>
     return true;<br>
   }<br>
<br>
@@ -239,7 +247,7 @@ bool clang::analyze_format_string::Parse<br>
   // Keep looking for a format specifier until we have exhausted the string.<br>
   while (I != E) {<br>
     const PrintfSpecifierResult &FSR = ParsePrintfSpecifier(H, I, E, argIndex,<br>
-                                                            LO, Target);<br>
+                                                            LO, Target, true);<br>
     // Did a fail-stop error of any kind occur when parsing the specifier?<br>
     // If so, don't do any more processing.<br>
     if (FSR.shouldStop())<br>
@@ -257,6 +265,34 @@ bool clang::analyze_format_string::Parse<br>
   return false;<br>
 }<br>
<br>
+bool clang::analyze_format_string::ParseFormatStringHasSArg(const char *I,<br>
+                                                            const char *E,<br>
+                                                            const LangOptions &LO,<br>
+                                                            const TargetInfo &Target) {<br>
+<br>
+  unsigned argIndex = 0;<br>
+<br>
+  // Keep looking for a %s format specifier until we have exhausted the string.<br>
+  FormatStringHandler H;<br>
+  while (I != E) {<br>
+    const PrintfSpecifierResult &FSR = ParsePrintfSpecifier(H, I, E, argIndex,<br>
+                                                            LO, Target, false);<br>
+    // Did a fail-stop error of any kind occur when parsing the specifier?<br>
+    // If so, don't do any more processing.<br>
+    if (FSR.shouldStop())<br>
+      return false;<br>
+    // Did we exhaust the string or encounter an error that<br>
+    // we can recover from?<br>
+    if (!FSR.hasValue())<br>
+      continue;<br>
+    const analyze_printf::PrintfSpecifier &FS = FSR.getValue();<br>
+    // Return true if this a %s format specifier.<br>
+    if (FS.getConversionSpecifier().getKind() == ConversionSpecifier::Kind::sArg)<br>
+      return true;<br>
+  }<br>
+  return false;<br>
+}<br>
+<br>
 //===----------------------------------------------------------------------===//<br>
 // Methods on PrintfSpecifier.<br>
 //===----------------------------------------------------------------------===//<br>
<br>
Modified: cfe/trunk/lib/Basic/IdentifierTable.cpp<br>
URL: <a href="http://llvm.org/viewvc/llvm-project/cfe/trunk/lib/Basic/IdentifierTable.cpp?rev=217467&r1=217466&r2=217467&view=diff" target="_blank">http://llvm.org/viewvc/llvm-project/cfe/trunk/lib/Basic/IdentifierTable.cpp?rev=217467&r1=217466&r2=217467&view=diff</a><br>
==============================================================================<br>
--- cfe/trunk/lib/Basic/IdentifierTable.cpp (original)<br>
+++ cfe/trunk/lib/Basic/IdentifierTable.cpp Tue Sep  9 18:10:54 2014<br>
@@ -487,6 +487,33 @@ ObjCInstanceTypeFamily Selector::getInst<br>
   return OIT_None;<br>
 }<br>
<br>
+ObjCStringFormatFamily Selector::getStringFormatFamilyImpl(Selector sel) {<br>
+  IdentifierInfo *first = sel.getIdentifierInfoForSlot(0);<br>
+  if (!first) return SFF_None;<br>
+<br>
+  StringRef name = first->getName();<br>
+<br>
+  switch (name.front()) {<br>
+    case 'a':<br>
+      if (name == "appendFormat") return SFF_NSString;<br>
+      break;<br>
+<br>
+    case 'i':<br>
+      if (name == "initWithFormat") return SFF_NSString;<br>
+      break;<br>
+<br>
+    case 'l':<br>
+      if (name == "localizedStringWithFormat") return SFF_NSString;<br>
+      break;<br>
+<br>
+    case 's':<br>
+      if (name == "stringByAppendingFormat" ||<br>
+          name == "stringWithFormat") return SFF_NSString;<br>
+      break;<br>
+  }<br>
+  return SFF_None;<br>
+}<br>
+<br>
 namespace {<br>
   struct SelectorTableImpl {<br>
     llvm::FoldingSet<MultiKeywordSelector> Table;<br>
<br>
Modified: cfe/trunk/lib/Sema/SemaChecking.cpp<br>
URL: <a href="http://llvm.org/viewvc/llvm-project/cfe/trunk/lib/Sema/SemaChecking.cpp?rev=217467&r1=217466&r2=217467&view=diff" target="_blank">http://llvm.org/viewvc/llvm-project/cfe/trunk/lib/Sema/SemaChecking.cpp?rev=217467&r1=217466&r2=217467&view=diff</a><br>
==============================================================================<br>
--- cfe/trunk/lib/Sema/SemaChecking.cpp (original)<br>
+++ cfe/trunk/lib/Sema/SemaChecking.cpp Tue Sep  9 18:10:54 2014<br>
@@ -767,6 +767,37 @@ static void CheckNonNullArgument(Sema &S<br>
     S.Diag(CallSiteLoc, diag::warn_null_arg) << ArgExpr->getSourceRange();<br>
 }<br>
<br>
+/// \brief Diagnose use of %s directive in an NSString which is being passed<br>
+/// as formatting string to formatting method.<br>
+static void<br>
+DiagnoseCStringFormatDirectiveInCFAPI(Sema &S,<br>
+                                        const NamedDecl *FDecl,<br>
+                                        Expr **Args,<br>
+                                        unsigned NumArgs) {<br>
+  if (NumArgs < 3)<br>
+    return;<br>
+  ObjCStringFormatFamily SFFamily = FDecl->getObjCFStringFormattingFamily();<br>
+  if (SFFamily == ObjCStringFormatFamily::SFF_CFString) {<br>
+    const Expr *FormatExpr = Args[2];<br>
+    if (const CStyleCastExpr *CSCE = dyn_cast<CStyleCastExpr>(FormatExpr))<br>
+      FormatExpr = CSCE->getSubExpr();<br>
+    const StringLiteral *FormatString;<br>
+    if (const ObjCStringLiteral *OSL =<br>
+        dyn_cast<ObjCStringLiteral>(FormatExpr->IgnoreParenImpCasts()))<br>
+      FormatString = OSL->getString();<br>
+    else<br>
+      FormatString = dyn_cast<StringLiteral>(FormatExpr->IgnoreParenImpCasts());<br>
+    if (!FormatString)<br>
+      return;<br>
+    if (S.FormatStringHasSArg(FormatString)) {<br>
+      S.Diag(FormatExpr->getExprLoc(), diag::warn_objc_cdirective_format_string)<br>
+        << "%s" << 1 << 1;<br>
+        S.Diag(FDecl->getLocation(), diag::note_entity_declared_at)<br>
+          << FDecl->getDeclName();<br>
+    }<br>
+  }<br>
+}<br>
+<br>
 static void CheckNonNullArguments(Sema &S,<br>
                                   const NamedDecl *FDecl,<br>
                                   ArrayRef<const Expr *> Args,<br>
@@ -899,6 +930,8 @@ bool Sema::CheckFunctionCall(FunctionDec<br>
     return false;<br>
<br>
   CheckAbsoluteValueFunction(TheCall, FDecl, FnInfo);<br>
+<br>
+  DiagnoseCStringFormatDirectiveInCFAPI(*this, FDecl, Args, NumArgs);<br>
<br>
   unsigned CMId = FDecl->getMemoryFunctionKind();<br>
   if (CMId == 0)<br>
@@ -3744,6 +3777,20 @@ void Sema::CheckFormatString(const Strin<br>
   } // TODO: handle other formats<br>
 }<br>
<br>
+bool Sema::FormatStringHasSArg(const StringLiteral *FExpr) {<br>
+  // Str - The format string.  NOTE: this is NOT null-terminated!<br>
+  StringRef StrRef = FExpr->getString();<br>
+  const char *Str = StrRef.data();<br>
+  // Account for cases where the string literal is truncated in a declaration.<br>
+  const ConstantArrayType *T = Context.getAsConstantArrayType(FExpr->getType());<br>
+  assert(T && "String literal not of constant array type!");<br>
+  size_t TypeSize = T->getSize().getZExtValue();<br>
+  size_t StrLen = std::min(std::max(TypeSize, size_t(1)) - 1, StrRef.size());<br>
+  return analyze_format_string::ParseFormatStringHasSArg(Str, Str + StrLen,<br>
+                                                         getLangOpts(),<br>
+                                                         Context.getTargetInfo());<br>
+}<br>
+<br>
 //===--- CHECK: Warn on use of wrong absolute value function. -------------===//<br>
<br>
 // Returns the related absolute value function that is larger, of 0 if one<br>
<br>
Modified: cfe/trunk/lib/Sema/SemaExprObjC.cpp<br>
URL: <a href="http://llvm.org/viewvc/llvm-project/cfe/trunk/lib/Sema/SemaExprObjC.cpp?rev=217467&r1=217466&r2=217467&view=diff" target="_blank">http://llvm.org/viewvc/llvm-project/cfe/trunk/lib/Sema/SemaExprObjC.cpp?rev=217467&r1=217466&r2=217467&view=diff</a><br>
==============================================================================<br>
--- cfe/trunk/lib/Sema/SemaExprObjC.cpp (original)<br>
+++ cfe/trunk/lib/Sema/SemaExprObjC.cpp Tue Sep  9 18:10:54 2014<br>
@@ -2114,6 +2114,32 @@ static void checkCocoaAPI(Sema &S, const<br>
                      edit::rewriteObjCRedundantCallWithLiteral);<br>
 }<br>
<br>
+/// \brief Diagnose use of %s directive in an NSString which is being passed<br>
+/// as formatting string to formatting method.<br>
+static void<br>
+DiagnoseCStringFormatDirectiveInObjCAPI(Sema &S,<br>
+                                        ObjCMethodDecl *Method,<br>
+                                        Selector Sel,<br>
+                                        Expr **Args, unsigned NumArgs) {<br>
+  if (NumArgs == 0)<br>
+    return;<br>
+  ObjCStringFormatFamily SFFamily = Sel.getStringFormatFamily();<br>
+  if (SFFamily == ObjCStringFormatFamily::SFF_NSString) {<br>
+    Expr *FormatExpr = Args[0];<br>
+    if (ObjCStringLiteral *OSL =<br>
+        dyn_cast<ObjCStringLiteral>(FormatExpr->IgnoreParenImpCasts())) {<br>
+      StringLiteral *FormatString = OSL->getString();<br>
+      if (S.FormatStringHasSArg(FormatString)) {<br>
+        S.Diag(FormatExpr->getExprLoc(), diag::warn_objc_cdirective_format_string)<br>
+        << "%s" << 0 << 0;<br>
+        if (Method)<br>
+          S.Diag(Method->getLocation(), diag::note_method_declared_at)<br>
+          << Method->getDeclName();<br>
+      }<br>
+    }<br>
+  }<br>
+}<br>
+<br>
 /// \brief Build an Objective-C class message expression.<br>
 ///<br>
 /// This routine takes care of both normal class messages and<br>
@@ -2258,7 +2284,9 @@ ExprResult Sema::BuildClassMessage(TypeS<br>
       }<br>
     }<br>
   }<br>
-<br>
+<br>
+  DiagnoseCStringFormatDirectiveInObjCAPI(*this, Method, Sel, Args, NumArgs);<br>
+<br>
   // Construct the appropriate ObjCMessageExpr.<br>
   ObjCMessageExpr *Result;<br>
   if (SuperLoc.isValid())<br>
@@ -2743,6 +2771,8 @@ ExprResult Sema::BuildInstanceMessage(Ex<br>
     }<br>
   }<br>
<br>
+  DiagnoseCStringFormatDirectiveInObjCAPI(*this, Method, Sel, Args, NumArgs);<br>
+<br>
   // Construct the appropriate ObjCMessageExpr instance.<br>
   ObjCMessageExpr *Result;<br>
   if (SuperLoc.isValid())<br>
<br>
Added: cfe/trunk/test/SemaObjC/format-cstrings-warning.m<br>
URL: <a href="http://llvm.org/viewvc/llvm-project/cfe/trunk/test/SemaObjC/format-cstrings-warning.m?rev=217467&view=auto" target="_blank">http://llvm.org/viewvc/llvm-project/cfe/trunk/test/SemaObjC/format-cstrings-warning.m?rev=217467&view=auto</a><br>
==============================================================================<br>
--- cfe/trunk/test/SemaObjC/format-cstrings-warning.m (added)<br>
+++ cfe/trunk/test/SemaObjC/format-cstrings-warning.m Tue Sep  9 18:10:54 2014<br>
@@ -0,0 +1,60 @@<br>
+// RUN: %clang_cc1 -Wcstring-format-directive -verify -fsyntax-only %s<br>
+// rdar://18182443<br>
+<br>
+typedef __builtin_va_list __darwin_va_list;<br>
+typedef __builtin_va_list va_list;<br>
+<br>
+@interface NSString @end<br>
+<br>
+@interface NSString (NSStringExtensionMethods)<br>
+- (NSString *)stringByAppendingFormat:(NSString *)format, ... __attribute__((format(__NSString__, 1, 2)));<br>
+- (instancetype)initWithFormat:(NSString *)format, ... __attribute__((format(__NSString__, 1, 2))); // expected-note {{method 'initWithFormat:' declared here}}<br>
+- (instancetype)initWithFormat:(NSString *)format arguments:(va_list)argList __attribute__((format(__NSString__, 1, 0)));<br>
+- (instancetype)initWithFormat:(NSString *)format locale:(id)locale, ... __attribute__((format(__NSString__, 1, 3)));<br>
+- (instancetype)initWithFormat:(NSString *)format locale:(id)locale arguments:(va_list)argList __attribute__((format(__NSString__, 1, 0)));<br>
++ (instancetype)stringWithFormat:(NSString *)format, ... __attribute__((format(__NSString__, 1, 2))); // expected-note {{method 'stringWithFormat:' declared here}}<br>
++ (instancetype)localizedStringWithFormat:(NSString *)format, ... __attribute__((format(__NSString__, 1, 2)));<br>
+@end<br>
+<br>
+@interface NSMutableString : NSString<br>
+@end<br>
+<br>
+@interface NSMutableString (NSMutableStringExtensionMethods)<br>
+<br>
+- (void)appendFormat:(NSString *)format, ... __attribute__((format(__NSString__, 1, 2)));<br>
+<br>
+@end<br>
+<br>
+NSString *ns(NSString *pns) {<br>
+  [pns initWithFormat: @"Number %d length %c name %s", 1, 'a', "something"]; // expected-warning {{using %s directive in NSString which is being passed as a formatting argument to the formatting method}}<br>
+  return [NSString stringWithFormat : @"Hello%s", " There"]; // expected-warning {{using %s directive in NSString which is being passed as a formatting argument to the formatting method}}<br>
+}<br>
+<br>
+<br>
+typedef const struct __CFString * CFStringRef;<br>
+typedef struct __CFString * CFMutableStringRef;<br>
+typedef const struct __CFAllocator * CFAllocatorRef;<br>
+<br>
+<br>
+typedef const struct __CFDictionary * CFDictionaryRef;<br>
+<br>
+<br>
+extern<br>
+CFStringRef CFStringCreateWithFormat(CFAllocatorRef alloc, CFDictionaryRef formatOptions, CFStringRef format, ...) __attribute__((format(CFString, 3, 4)));<br>
+<br>
+extern<br>
+CFStringRef CFStringCreateWithFormatAndArguments(CFAllocatorRef alloc, CFDictionaryRef formatOptions, CFStringRef format, va_list arguments) __attribute__((format(CFString, 3, 0))); // expected-note {{'CFStringCreateWithFormatAndArguments' declared here}}<br>
+<br>
+extern<br>
+void CFStringAppendFormat(CFMutableStringRef theString, CFDictionaryRef formatOptions, CFStringRef format, ...) __attribute__((format(CFString, 3, 4)));<br>
+<br>
+extern<br>
+void CFStringAppendFormatAndArguments(CFMutableStringRef theString, CFDictionaryRef formatOptions, CFStringRef format, va_list arguments) __attribute__((format(CFString, 3, 0))); // expected-note {{'CFStringAppendFormatAndArguments' declared here}}<br>
+<br>
+void foo(va_list argList) {<br>
+  CFAllocatorRef alloc;<br>
+  CFStringCreateWithFormatAndArguments (alloc, 0, (CFStringRef)@"%s\n", argList); // expected-warning {{using %s directive in CFString which is being passed as a formatting argument to the formatting CFfunction}}<br>
+  CFStringAppendFormatAndArguments ((CFMutableStringRef)@"AAAA", 0, (CFStringRef)"Hello %s there %d\n", argList); // expected-warning {{using %s directive in CFString which is being passed as a formatting argument to the formatting CFfunction}}<br>
+  CFStringCreateWithFormatAndArguments (alloc, 0, (CFStringRef)@"%c\n", argList);<br>
+  CFStringAppendFormatAndArguments ((CFMutableStringRef)@"AAAA", 0, (CFStringRef)"%d\n", argList);<br>
+}<br>
<br>
<br>
_______________________________________________<br>
cfe-commits mailing list<br>
<a href="mailto:cfe-commits@cs.uiuc.edu">cfe-commits@cs.uiuc.edu</a><br>
<a href="http://lists.cs.uiuc.edu/mailman/listinfo/cfe-commits" target="_blank">http://lists.cs.uiuc.edu/mailman/listinfo/cfe-commits</a><br>
</blockquote></div><br></div>