• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1// RUN: %clang_cc1 -triple x86_64-apple-darwin -Wformat-nonliteral -fsyntax-only -fblocks -verify -Wno-objc-root-class %s
2
3//===----------------------------------------------------------------------===//
4// The following code is reduced using delta-debugging from
5// Foundation.h (Mac OS X).
6//
7// It includes the basic definitions for the test cases below.
8// Not including Foundation.h directly makes this test case both svelt and
9// portable to non-Mac platforms.
10//===----------------------------------------------------------------------===//
11
12#include <stdarg.h>
13
14typedef signed char BOOL;
15typedef unsigned int NSUInteger;
16typedef long NSInteger;
17@class NSString, Protocol;
18extern void NSLog(NSString *format, ...);
19extern void NSLogv(NSString *format, va_list args);
20typedef struct _NSZone NSZone;
21@class NSInvocation, NSMethodSignature, NSCoder, NSString, NSEnumerator;
22@protocol NSObject  - (BOOL)isEqual:(id)object; @end
23@protocol NSCopying  - (id)copyWithZone:(NSZone *)zone; @end
24@protocol NSMutableCopying  - (id)mutableCopyWithZone:(NSZone *)zone; @end
25@protocol NSCoding  - (void)encodeWithCoder:(NSCoder *)aCoder; @end
26@interface NSObject <NSObject> {} @end
27typedef float CGFloat;
28@interface NSString : NSObject <NSCopying, NSMutableCopying, NSCoding>    - (NSUInteger)length; @end
29@interface NSSimpleCString : NSString {} @end
30@interface NSConstantString : NSSimpleCString @end
31extern void *_NSConstantStringClassReference;
32
33typedef const struct __CFString * CFStringRef;
34extern void CFStringCreateWithFormat(CFStringRef format, ...) __attribute__((format(CFString, 1, 2)));
35#define CFSTR(cStr)  ((CFStringRef) __builtin___CFStringMakeConstantString ("" cStr ""))
36
37// This function is used instead of the builtin if -fno-constant-cfstrings.
38// The definition on Mac OS X is NOT annotated with format_arg as of 10.8,
39// but clang will implicitly add the attribute if it's not written.
40extern CFStringRef __CFStringMakeConstantString(const char *);
41
42int printf(const char * restrict, ...) ;
43
44//===----------------------------------------------------------------------===//
45// Test cases.
46//===----------------------------------------------------------------------===//
47
48void check_nslog(unsigned k) {
49  NSLog(@"%d%%", k); // no-warning
50  NSLog(@"%s%lb%d", "unix", 10,20); // expected-warning {{invalid conversion specifier 'b'}}
51}
52
53// Check type validation
54extern void NSLog2(int format, ...) __attribute__((format(__NSString__, 1, 2))); // expected-error {{format argument not an NSString}}
55extern void CFStringCreateWithFormat2(int *format, ...) __attribute__((format(CFString, 1, 2))); // expected-error {{format argument not a CFString}}
56
57// <rdar://problem/7068334> - Catch use of long long with int arguments.
58void rdar_7068334() {
59  long long test = 500;
60  printf("%i ",test); // expected-warning{{format specifies type 'int' but the argument has type 'long long'}}
61  NSLog(@"%i ",test); // expected-warning{{format specifies type 'int' but the argument has type 'long long'}}
62  CFStringCreateWithFormat(CFSTR("%i"),test); // expected-warning{{format specifies type 'int' but the argument has type 'long long'}}
63}
64
65// <rdar://problem/7697748>
66void rdar_7697748() {
67  NSLog(@"%@!"); // expected-warning{{more '%' conversions than data arguments}}
68}
69
70@protocol Foo;
71
72void test_p_conversion_with_objc_pointer(id x, id<Foo> y) {
73  printf("%p", x); // no-warning
74  printf("%p", y); // no-warning
75}
76
77// <rdar://problem/10696348>, PR 10274 - CFString and NSString formats are ignored
78extern void MyNSLog(NSString *format, ...) __attribute__((format(__NSString__, 1, 2)));
79extern void MyCFStringCreateWithFormat(CFStringRef format, ...) __attribute__((format(__CFString__, 1, 2)));
80
81void check_mylog() {
82  MyNSLog(@"%@"); // expected-warning {{more '%' conversions than data arguments}}
83  MyCFStringCreateWithFormat(CFSTR("%@")); // expected-warning {{more '%' conversions than data arguments}}
84}
85
86// PR 10275 - format function attribute isn't checked in Objective-C methods
87@interface Foo
88+ (id)fooWithFormat:(NSString *)fmt, ... __attribute__((format(__NSString__, 1, 2)));
89+ (id)fooWithCStringFormat:(const char *)format, ... __attribute__((format(__printf__, 1, 2)));
90@end
91
92void check_method() {
93  [Foo fooWithFormat:@"%@"]; // expected-warning {{more '%' conversions than data arguments}}
94  [Foo fooWithCStringFormat:"%@"]; // expected-warning {{invalid conversion specifier '@'}}
95}
96
97// Warn about using BOOL with %@
98void rdar10743758(id x) {
99  NSLog(@"%@ %@", x, (BOOL) 1); // expected-warning {{format specifies type 'id' but the argument has type 'BOOL' (aka 'signed char')}}
100}
101
102NSString *test_literal_propagation(void) {
103  const char * const s1 = "constant string %s"; // expected-note {{format string is defined here}}
104  printf(s1); // expected-warning {{more '%' conversions than data arguments}}
105  const char * const s5 = "constant string %s"; // expected-note {{format string is defined here}}
106  const char * const s2 = s5;
107  printf(s2); // expected-warning {{more '%' conversions than data arguments}}
108
109  const char * const s3 = (const char *)0;
110  printf(s3); // no-warning (NULL is a valid format string)
111
112  NSString * const ns1 = @"constant string %s"; // expected-note {{format string is defined here}}
113  NSLog(ns1); // expected-warning {{more '%' conversions than data arguments}}
114  NSString * const ns5 = @"constant string %s"; // expected-note {{format string is defined here}}
115  NSString * const ns2 = ns5;
116  NSLog(ns2); // expected-warning {{more '%' conversions than data arguments}}
117  NSString * ns3 = ns1;
118  NSLog(ns3); // expected-warning {{format string is not a string literal}}}
119  // expected-note@-1{{treat the string as an argument to avoid this}}
120
121  NSString * const ns6 = @"split" " string " @"%s"; // expected-note {{format string is defined here}}
122  NSLog(ns6); // expected-warning {{more '%' conversions than data arguments}}
123}
124
125// Do not emit warnings when using NSLocalizedString
126#include "format-strings-system.h"
127
128// Test it inhibits diag only for macros in system headers
129#define MyNSLocalizedString(key) GetLocalizedString(key)
130#define MyNSAssert(fmt, arg) NSLog(fmt, arg, 0, 0)
131
132void check_NSLocalizedString() {
133  [Foo fooWithFormat:NSLocalizedString(@"format"), @"arg"]; // no-warning
134  [Foo fooWithFormat:MyNSLocalizedString(@"format"), @"arg"]; // expected-warning {{format string is not a string literal}}}
135}
136
137void check_NSAssert() {
138  NSAssert(@"Hello %@", @"World"); // no-warning
139  MyNSAssert(@"Hello %@", @"World"); // expected-warning  {{data argument not used by format string}}
140}
141
142typedef __WCHAR_TYPE__ wchar_t;
143
144// Test that %S, %C, %ls check for 16 bit types in ObjC strings, as described at
145// http://developer.apple.com/library/mac/#documentation/Cocoa/Conceptual/Strings/Articles/formatSpecifiers.html#//apple_ref/doc/uid/TP40004265
146
147void test_percent_S() {
148  const unsigned short data[] = { 'a', 'b', 0 };
149  const unsigned short* ptr = data;
150  NSLog(@"%S", ptr);  // no-warning
151
152  const wchar_t* wchar_ptr = L"ab";
153  NSLog(@"%S", wchar_ptr);  // expected-warning{{format specifies type 'const unichar *' (aka 'const unsigned short *') but the argument has type 'const wchar_t *'}}
154}
155
156void test_percent_ls() {
157  const unsigned short data[] = { 'a', 'b', 0 };
158  const unsigned short* ptr = data;
159  NSLog(@"%ls", ptr);  // no-warning
160
161  const wchar_t* wchar_ptr = L"ab";
162  NSLog(@"%ls", wchar_ptr);  // expected-warning{{format specifies type 'const unichar *' (aka 'const unsigned short *') but the argument has type 'const wchar_t *'}}
163}
164
165void test_percent_C() {
166  const unsigned short data = 'a';
167  NSLog(@"%C", data);  // no-warning
168
169  const wchar_t wchar_data = L'a';
170  NSLog(@"%C", wchar_data);  // expected-warning{{format specifies type 'unichar' (aka 'unsigned short') but the argument has type 'wchar_t'}}
171}
172
173// Test that %@ works with toll-free bridging (<rdar://problem/10814120>).
174void test_toll_free_bridging(CFStringRef x, id y) {
175  NSLog(@"%@", x); // no-warning
176  CFStringCreateWithFormat(CFSTR("%@"), x); // no-warning
177
178  NSLog(@"%@", y); // no-warning
179  CFStringCreateWithFormat(CFSTR("%@"), y); // no-warning
180}
181
182@interface Bar
183+ (void)log:(NSString *)fmt, ...;
184+ (void)log2:(NSString *)fmt, ... __attribute__((format(NSString, 1, 2)));
185@end
186
187@implementation Bar
188
189+ (void)log:(NSString *)fmt, ... {
190  va_list ap;
191  va_start(ap,fmt);
192  NSLogv(fmt, ap); // expected-warning{{format string is not a string literal}}
193  va_end(ap);
194}
195
196+ (void)log2:(NSString *)fmt, ... {
197  va_list ap;
198  va_start(ap,fmt);
199  NSLogv(fmt, ap); // no-warning
200  va_end(ap);
201}
202
203@end
204
205
206// Test that it is okay to use %p with the address of a block.
207void rdar11049844_aux();
208int rdar11049844() {
209  typedef void (^MyBlock)(void);
210  MyBlock x = ^void() { rdar11049844_aux(); };
211  printf("%p", x);  // no-warning
212}
213
214void test_nonBuiltinCFStrings() {
215  CFStringCreateWithFormat(__CFStringMakeConstantString("%@"), 1); // expected-warning{{format specifies type 'id' but the argument has type 'int'}}
216}
217
218
219// Don't crash on an invalid argument expression.
220// <rdar://problem/11890818>
221@interface NSDictionary : NSObject
222- (id)objectForKeyedSubscript:(id)key;
223@end
224
225void testInvalidFormatArgument(NSDictionary *dict) {
226  NSLog(@"no specifiers", dict[CFSTR("abc")]); // expected-error{{indexing expression is invalid because subscript type 'CFStringRef' (aka 'const struct __CFString *') is not an integral or Objective-C pointer type}}
227  NSLog(@"%@", dict[CFSTR("abc")]); // expected-error{{indexing expression is invalid because subscript type 'CFStringRef' (aka 'const struct __CFString *') is not an integral or Objective-C pointer type}}
228  NSLog(@"%@ %@", dict[CFSTR("abc")]); // expected-error{{indexing expression is invalid because subscript type 'CFStringRef' (aka 'const struct __CFString *') is not an integral or Objective-C pointer type}}
229
230  [Foo fooWithFormat:@"no specifiers", dict[CFSTR("abc")]]; // expected-error{{indexing expression is invalid because subscript type 'CFStringRef' (aka 'const struct __CFString *') is not an integral or Objective-C pointer type}}
231  [Foo fooWithFormat:@"%@", dict[CFSTR("abc")]]; // expected-error{{indexing expression is invalid because subscript type 'CFStringRef' (aka 'const struct __CFString *') is not an integral or Objective-C pointer type}}
232  [Foo fooWithFormat:@"%@ %@", dict[CFSTR("abc")]]; // expected-error{{indexing expression is invalid because subscript type 'CFStringRef' (aka 'const struct __CFString *') is not an integral or Objective-C pointer type}} expected-warning{{more '%' conversions than data arguments}}
233}
234
235
236// <rdar://problem/11825593>
237void testByValueObjectInFormat(Foo *obj) {
238  printf("%d %d %d", 1L, *obj, 1L); // expected-error {{cannot pass object with interface type 'Foo' by value to variadic function; expected type from format string was 'int'}} expected-warning 2 {{format specifies type 'int' but the argument has type 'long'}}
239  printf("%!", *obj); // expected-error {{cannot pass object with interface type 'Foo' by value through variadic function}} expected-warning {{invalid conversion specifier}}
240  printf(0, *obj); // expected-error {{cannot pass object with interface type 'Foo' by value through variadic function}}
241
242  [Bar log2:@"%d", *obj]; // expected-error {{cannot pass object with interface type 'Foo' by value to variadic method; expected type from format string was 'int'}}
243}
244
245// <rdar://problem/13557053>
246void testTypeOf(NSInteger dW, NSInteger dH) {
247  NSLog(@"dW %d  dH %d",({ __typeof__(dW) __a = (dW); __a < 0 ? -__a : __a; }),({ __typeof__(dH) __a = (dH); __a < 0 ? -__a : __a; })); // expected-warning 2 {{format specifies type 'int' but the argument has type 'long'}}
248}
249
250void testUnicode() {
251  NSLog(@"%C", 0x2022); // no-warning
252  NSLog(@"%C", 0x202200); // expected-warning{{format specifies type 'unichar' (aka 'unsigned short') but the argument has type 'int'}}
253}
254
255// Test Objective-C modifier flags.
256void testObjCModifierFlags() {
257  NSLog(@"%[]@", @"Foo"); // expected-warning {{missing object format flag}}
258  NSLog(@"%[", @"Foo"); // expected-warning {{incomplete format specifier}}
259  NSLog(@"%[tt", @"Foo");  // expected-warning {{incomplete format specifier}}
260  NSLog(@"%[tt]@", @"Foo"); // no-warning
261  NSLog(@"%[tt]@ %s", @"Foo", "hello"); // no-warning
262  NSLog(@"%s %[tt]@", "hello", @"Foo"); // no-warning
263  NSLog(@"%[blark]@", @"Foo"); // expected-warning {{'blark' is not a valid object format flag}}
264  NSLog(@"%2$[tt]@ %1$[tt]@", @"Foo", @"Bar"); // no-warning
265  NSLog(@"%2$[tt]@ %1$[tt]s", @"Foo", @"Bar"); // expected-warning {{object format flags cannot be used with 's' conversion specifier}}
266}
267