• 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;
16@class NSString, Protocol;
17extern void NSLog(NSString *format, ...);
18extern void NSLogv(NSString *format, va_list args);
19typedef struct _NSZone NSZone;
20@class NSInvocation, NSMethodSignature, NSCoder, NSString, NSEnumerator;
21@protocol NSObject  - (BOOL)isEqual:(id)object; @end
22@protocol NSCopying  - (id)copyWithZone:(NSZone *)zone; @end
23@protocol NSMutableCopying  - (id)mutableCopyWithZone:(NSZone *)zone; @end
24@protocol NSCoding  - (void)encodeWithCoder:(NSCoder *)aCoder; @end
25@interface NSObject <NSObject> {} @end
26typedef float CGFloat;
27@interface NSString : NSObject <NSCopying, NSMutableCopying, NSCoding>    - (NSUInteger)length; @end
28@interface NSSimpleCString : NSString {} @end
29@interface NSConstantString : NSSimpleCString @end
30extern void *_NSConstantStringClassReference;
31
32typedef const struct __CFString * CFStringRef;
33extern void CFStringCreateWithFormat(CFStringRef format, ...) __attribute__((format(CFString, 1, 2)));
34
35int printf(const char * restrict, ...) ;
36
37//===----------------------------------------------------------------------===//
38// Test cases.
39//===----------------------------------------------------------------------===//
40
41void check_nslog(unsigned k) {
42  NSLog(@"%d%%", k); // no-warning
43  NSLog(@"%s%lb%d", "unix", 10,20); // expected-warning {{invalid conversion specifier 'b'}}
44}
45
46// Check type validation
47extern void NSLog2(int format, ...) __attribute__((format(__NSString__, 1, 2))); // expected-error {{format argument not an NSString}}
48extern void CFStringCreateWithFormat2(int *format, ...) __attribute__((format(CFString, 1, 2))); // expected-error {{format argument not a CFString}}
49
50// <rdar://problem/7068334> - Catch use of long long with int arguments.
51void rdar_7068334() {
52  long long test = 500;
53  printf("%i ",test); // expected-warning{{format specifies type 'int' but the argument has type 'long long'}}
54  NSLog(@"%i ",test); // expected-warning{{format specifies type 'int' but the argument has type 'long long'}}
55}
56
57// <rdar://problem/7697748>
58void rdar_7697748() {
59  NSLog(@"%@!"); // expected-warning{{more '%' conversions than data arguments}}
60}
61
62@protocol Foo;
63
64void test_p_conversion_with_objc_pointer(id x, id<Foo> y) {
65  printf("%p", x); // no-warning
66  printf("%p", y); // no-warning
67}
68
69// <rdar://problem/10696348>, PR 10274 - CFString and NSString formats are ignored
70extern void MyNSLog(NSString *format, ...) __attribute__((format(__NSString__, 1, 2)));
71extern void MyCFStringCreateWithFormat(CFStringRef format, ...) __attribute__((format(__CFString__, 1, 2)));
72
73void check_mylog() {
74  MyNSLog(@"%@"); // expected-warning {{more '%' conversions than data arguments}}
75  // FIXME: find a way to test CFString too, but I don't know how to create constant CFString.
76}
77
78// PR 10275 - format function attribute isn't checked in Objective-C methods
79@interface Foo
80+ (id)fooWithFormat:(NSString *)fmt, ... __attribute__((format(__NSString__, 1, 2)));
81+ (id)fooWithCStringFormat:(const char *)format, ... __attribute__((format(__printf__, 1, 2)));
82@end
83
84void check_method() {
85  [Foo fooWithFormat:@"%@"]; // expected-warning {{more '%' conversions than data arguments}}
86  [Foo fooWithCStringFormat:"%@"]; // expected-warning {{invalid conversion specifier '@'}}
87}
88
89// Warn about using BOOL with %@
90void rdar10743758(id x) {
91  NSLog(@"%@ %@", x, (BOOL) 1); // expected-warning {{format specifies type 'id' but the argument has type 'BOOL' (aka 'signed char')}}
92}
93
94NSString *test_literal_propagation(void) {
95  const char * const s1 = "constant string %s"; // expected-note {{format string is defined here}}
96  printf(s1); // expected-warning {{more '%' conversions than data arguments}}
97  const char * const s5 = "constant string %s"; // expected-note {{format string is defined here}}
98  const char * const s2 = s5;
99  printf(s2); // expected-warning {{more '%' conversions than data arguments}}
100
101  const char * const s3 = (const char *)0;
102  printf(s3); // no-warning (NULL is a valid format string)
103
104  NSString * const ns1 = @"constant string %s"; // expected-note {{format string is defined here}}
105  NSLog(ns1); // expected-warning {{more '%' conversions than data arguments}}
106  NSString * const ns5 = @"constant string %s"; // expected-note {{format string is defined here}}
107  NSString * const ns2 = ns5;
108  NSLog(ns2); // expected-warning {{more '%' conversions than data arguments}}
109  NSString * ns3 = ns1;
110  NSLog(ns3); // expected-warning {{format string is not a string literal}}}
111}
112
113// Do not emit warnings when using NSLocalizedString
114extern NSString *GetLocalizedString(NSString *str);
115#define NSLocalizedString(key) GetLocalizedString(key)
116
117void check_NSLocalizedString() {
118  [Foo fooWithFormat:NSLocalizedString(@"format"), @"arg"]; // no-warning
119}
120
121typedef __WCHAR_TYPE__ wchar_t;
122
123// Test that %S, %C, %ls check for 16 bit types in ObjC strings, as described at
124// http://developer.apple.com/library/mac/#documentation/Cocoa/Conceptual/Strings/Articles/formatSpecifiers.html#//apple_ref/doc/uid/TP40004265
125
126void test_percent_S() {
127  const unsigned short data[] = { 'a', 'b', 0 };
128  const unsigned short* ptr = data;
129  NSLog(@"%S", ptr);  // no-warning
130
131  const wchar_t* wchar_ptr = L"ab";
132  NSLog(@"%S", wchar_ptr);  // expected-warning{{format specifies type 'const unsigned short *' but the argument has type 'const wchar_t *'}}
133}
134
135void test_percent_ls() {
136  const unsigned short data[] = { 'a', 'b', 0 };
137  const unsigned short* ptr = data;
138  NSLog(@"%ls", ptr);  // no-warning
139
140  const wchar_t* wchar_ptr = L"ab";
141  NSLog(@"%ls", wchar_ptr);  // expected-warning{{format specifies type 'const unsigned short *' but the argument has type 'const wchar_t *'}}
142}
143
144void test_percent_C() {
145  const unsigned short data = 'a';
146  NSLog(@"%C", data);  // no-warning
147
148  const wchar_t wchar_data = L'a';
149  NSLog(@"%C", wchar_data);  // expected-warning{{format specifies type 'unsigned short' but the argument has type 'wchar_t'}}
150}
151
152// Test that %@ works with toll-free bridging (<rdar://problem/10814120>).
153void test_toll_free_bridging(CFStringRef x) {
154  NSLog(@"%@", x); // no-warning
155}
156
157@interface Bar
158+ (void)log:(NSString *)fmt, ...;
159+ (void)log2:(NSString *)fmt, ... __attribute__((format(NSString, 1, 2)));
160@end
161
162@implementation Bar
163
164+ (void)log:(NSString *)fmt, ... {
165  va_list ap;
166  va_start(ap,fmt);
167  NSLogv(fmt, ap); // expected-warning{{format string is not a string literal}}
168  va_end(ap);
169}
170
171+ (void)log2:(NSString *)fmt, ... {
172  va_list ap;
173  va_start(ap,fmt);
174  NSLogv(fmt, ap); // no-warning
175  va_end(ap);
176}
177
178@end
179
180
181// Test that it is okay to use %p with the address of a block.
182void rdar11049844_aux();
183int rdar11049844() {
184  typedef void (^MyBlock)(void);
185  MyBlock x = ^void() { rdar11049844_aux(); };
186  printf("%p", x);  // no-warning
187}
188
189