1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81
|
// RUN: %clang_cc1 -fsyntax-only -verify -Wformat-nonliteral -pedantic %s
#include <stdarg.h>
extern "C" {
extern int scanf(const char *restrict, ...);
extern int printf(const char *restrict, ...);
extern int vprintf(const char *restrict, va_list);
}
@class NSString;
@interface Format
+ (void)print:(NSString *)format, ... __attribute__((format(NSString, 1, 2)));
@end
namespace Templates {
template<typename T>
void my_uninstantiated_print(const T &arg) {
[Format print:@"%d", arg];
}
template<typename T>
void my_print(const T &arg) {
[Format print:@"%d", arg]; // expected-warning {{format specifies type 'int' but the argument has type 'const char *'}}
}
void use_my_print() {
my_print("abc"); // expected-note {{requested here}}
}
template<typename T>
class UninstantiatedPrinter {
public:
static void print(const T &arg) {
[Format print:@"%d", arg]; // no-warning
}
};
template<typename T>
class Printer {
public:
void print(const T &arg) {
[Format print:@"%d", arg]; // expected-warning {{format specifies type 'int' but the argument has type 'const char *'}}
}
};
void use_class(Printer<const char *> &p) {
p.print("abc"); // expected-note {{requested here}}
}
template<typename T>
class UninstantiatedWrapper {
public:
class Printer {
public:
void print(const T &arg) {
[Format print:@"%d", arg]; // no-warning
}
};
};
template<typename T>
class Wrapper {
public:
class Printer {
public:
void print(const T &arg) {
[Format print:@"%d", arg]; // expected-warning {{format specifies type 'int' but the argument has type 'const char *'}}
}
};
};
void use_class(Wrapper<const char *>::Printer &p) {
p.print("abc"); // expected-note {{requested here}}
}
}
|