forked from OSchip/llvm-project
Implements support of format_arg attribute on C++ member.
llvm-svn: 149998
This commit is contained in:
parent
57d2c78572
commit
6255bd14f0
|
@ -1459,21 +1459,20 @@ bool Sema::SemaCheckStringLiteral(const Expr *E, Expr **Args,
|
|||
return false;
|
||||
}
|
||||
|
||||
case Stmt::CallExprClass: {
|
||||
case Stmt::CallExprClass:
|
||||
case Stmt::CXXMemberCallExprClass: {
|
||||
const CallExpr *CE = cast<CallExpr>(E);
|
||||
if (const ImplicitCastExpr *ICE
|
||||
= dyn_cast<ImplicitCastExpr>(CE->getCallee())) {
|
||||
if (const DeclRefExpr *DRE = dyn_cast<DeclRefExpr>(ICE->getSubExpr())) {
|
||||
if (const FunctionDecl *FD = dyn_cast<FunctionDecl>(DRE->getDecl())) {
|
||||
if (const FormatArgAttr *FA = FD->getAttr<FormatArgAttr>()) {
|
||||
unsigned ArgIndex = FA->getFormatIdx();
|
||||
const Expr *Arg = CE->getArg(ArgIndex - 1);
|
||||
if (const NamedDecl *ND = dyn_cast_or_null<NamedDecl>(CE->getCalleeDecl())) {
|
||||
if (const FormatArgAttr *FA = ND->getAttr<FormatArgAttr>()) {
|
||||
unsigned ArgIndex = FA->getFormatIdx();
|
||||
if (const CXXMethodDecl *MD = dyn_cast<CXXMethodDecl>(ND))
|
||||
if (MD->isInstance())
|
||||
--ArgIndex;
|
||||
const Expr *Arg = CE->getArg(ArgIndex - 1);
|
||||
|
||||
return SemaCheckStringLiteral(Arg, Args, NumArgs, HasVAListArg,
|
||||
format_idx, firstDataArg, Type,
|
||||
inFunctionCall);
|
||||
}
|
||||
}
|
||||
return SemaCheckStringLiteral(Arg, Args, NumArgs, HasVAListArg,
|
||||
format_idx, firstDataArg, Type,
|
||||
inFunctionCall);
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -1534,11 +1533,7 @@ void Sema::CheckFormatArguments(const FormatAttr *Format, CallExpr *TheCall) {
|
|||
// The way the format attribute works in GCC, the implicit this argument
|
||||
// of member functions is counted. However, it doesn't appear in our own
|
||||
// lists, so decrement format_idx in that case.
|
||||
if (isa<CXXMemberCallExpr>(TheCall)) {
|
||||
const CXXMethodDecl *method_decl =
|
||||
dyn_cast<CXXMethodDecl>(TheCall->getCalleeDecl());
|
||||
IsCXXMember = method_decl && method_decl->isInstance();
|
||||
}
|
||||
IsCXXMember = isa<CXXMemberCallExpr>(TheCall);
|
||||
CheckFormatArguments(Format, TheCall->getArgs(), TheCall->getNumArgs(),
|
||||
IsCXXMember, TheCall->getRParenLoc(),
|
||||
TheCall->getCallee()->getSourceRange());
|
||||
|
|
|
@ -1,4 +1,4 @@
|
|||
// RUN: %clang_cc1 -fsyntax-only -verify -pedantic %s
|
||||
// RUN: %clang_cc1 -fsyntax-only -verify -Wformat-nonliteral -pedantic %s
|
||||
|
||||
extern "C" {
|
||||
extern int scanf(const char *restrict, ...);
|
||||
|
@ -17,3 +17,25 @@ void f(char **sp, float *fp) {
|
|||
void g() {
|
||||
printf("%ls", "foo"); // expected-warning{{format specifies type 'wchar_t *' but the argument has type 'const char *'}}
|
||||
}
|
||||
|
||||
// Test that we properly handle format_idx on C++ members.
|
||||
class Foo {
|
||||
public:
|
||||
const char *gettext(const char *fmt) __attribute__((format_arg(2)));
|
||||
|
||||
int scanf(const char *restrict, ...) __attribute__((format(scanf, 2, 3)));
|
||||
int printf(const char *restrict, ...) __attribute__((format(printf, 2, 3)));
|
||||
|
||||
static const char *gettext_static(const char *fmt) __attribute__((format_arg(1)));
|
||||
static int printf_static(const char *restrict, ...) __attribute__((format(printf, 1, 2)));
|
||||
};
|
||||
|
||||
void h(int *i) {
|
||||
Foo foo;
|
||||
foo.scanf("%d"); // expected-warning{{more '%' conversions than data arguments}}
|
||||
foo.printf("%d", i); // expected-warning{{format specifies type 'int' but the argument has type 'int *'}}
|
||||
Foo::printf_static("%d", i); // expected-warning{{format specifies type 'int' but the argument has type 'int *'}}
|
||||
|
||||
printf(foo.gettext("%d"), i); // expected-warning{{format specifies type 'int' but the argument has type 'int *'}}
|
||||
printf(Foo::gettext_static("%d"), i); // expected-warning{{format specifies type 'int' but the argument has type 'int *'}}
|
||||
}
|
||||
|
|
Loading…
Reference in New Issue