Improve attribute parsing & tests.

- Support noreturn on function-typed variables.

 - Extend isFunctionOrMethod to return true for K&R functions and
   provide hasFunctionProto to check if a decl has information about
   its arguments. This code needs some serious cleaning, but works.

 - Add/improve test cases for noreturn and unused.


git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@57778 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
Daniel Dunbar 2008-10-19 02:04:16 +00:00
Родитель 6dfe2f544a
Коммит d3f2c10f88
4 изменённых файлов: 67 добавлений и 31 удалений

Просмотреть файл

@ -25,7 +25,7 @@ using namespace clang;
// Helper functions
//===----------------------------------------------------------------------===//
static const FunctionTypeProto *getFunctionProto(Decl *d) {
static const FunctionType *getFunctionType(Decl *d) {
QualType Ty;
if (ValueDecl *decl = dyn_cast<ValueDecl>(d))
Ty = decl->getType();
@ -39,24 +39,37 @@ static const FunctionTypeProto *getFunctionProto(Decl *d) {
if (Ty->isFunctionPointerType())
Ty = Ty->getAsPointerType()->getPointeeType();
if (const FunctionType *FnTy = Ty->getAsFunctionType())
return dyn_cast<FunctionTypeProto>(FnTy->getAsFunctionType());
return 0;
return Ty->getAsFunctionType();
}
// FIXME: We should provide an abstraction around a method or function
// to provide the following bits of information.
/// isFunctionOrMethod - Return true if the given decl is a (non-K&R)
/// function or an Objective-C method.
/// isFunctionOrMethod - Return true if the given decl has function
/// type (function or function-typed variable) or an Objective-C
/// method.
static bool isFunctionOrMethod(Decl *d) {
return getFunctionProto(d) || isa<ObjCMethodDecl>(d);
return getFunctionType(d) || isa<ObjCMethodDecl>(d);
}
/// hasFunctionProto - Return true if the given decl has a argument
/// information. This decl should have already passed
/// isFunctionOrMethod.
static bool hasFunctionProto(Decl *d) {
if (const FunctionType *FnTy = getFunctionType(d)) {
return isa<FunctionTypeProto>(FnTy);
} else {
assert(isa<ObjCMethodDecl>(d));
return true;
}
}
/// getFunctionOrMethodNumArgs - Return number of function or method
/// arguments. It is an error to call this on a K&R function (use
/// hasFunctionProto first).
static unsigned getFunctionOrMethodNumArgs(Decl *d) {
if (const FunctionTypeProto *proto = getFunctionProto(d)) {
if (const FunctionType *FnTy = getFunctionType(d)) {
const FunctionTypeProto *proto = cast<FunctionTypeProto>(FnTy);
return proto->getNumArgs();
} else {
return cast<ObjCMethodDecl>(d)->getNumParams();
@ -64,7 +77,8 @@ static unsigned getFunctionOrMethodNumArgs(Decl *d) {
}
static QualType getFunctionOrMethodArgType(Decl *d, unsigned Idx) {
if (const FunctionTypeProto *proto = getFunctionProto(d)) {
if (const FunctionType *FnTy = getFunctionType(d)) {
const FunctionTypeProto *proto = cast<FunctionTypeProto>(FnTy);
return proto->getArgType(Idx);
} else {
return cast<ObjCMethodDecl>(d)->getParamDecl(Idx)->getType();
@ -72,7 +86,8 @@ static QualType getFunctionOrMethodArgType(Decl *d, unsigned Idx) {
}
static bool isFunctionOrMethodVariadic(Decl *d) {
if (const FunctionTypeProto *proto = getFunctionProto(d)) {
if (const FunctionType *FnTy = getFunctionType(d)) {
const FunctionTypeProto *proto = cast<FunctionTypeProto>(FnTy);
return proto->isVariadic();
} else {
return cast<ObjCMethodDecl>(d)->isVariadic();
@ -289,17 +304,15 @@ static void HandleIBOutletAttr(Decl *d, const AttributeList &Attr, Sema &S) {
}
static void HandleNonNullAttr(Decl *d, const AttributeList &Attr, Sema &S) {
// GCC ignores the nonnull attribute on K&R style function
// prototypes, so we ignore it as well
const FunctionTypeProto *proto = getFunctionProto(d);
if (!proto) {
if (!isFunctionOrMethod(d) || !hasFunctionProto(d)) {
S.Diag(Attr.getLoc(), diag::warn_attribute_wrong_decl_type,
"nonnull", "function");
return;
}
unsigned NumArgs = proto->getNumArgs();
unsigned NumArgs = getFunctionOrMethodNumArgs(d);
// The nonnull attribute only applies to pointers.
llvm::SmallVector<unsigned, 10> NonNullArgs;
@ -328,7 +341,7 @@ static void HandleNonNullAttr(Decl *d, const AttributeList &Attr, Sema &S) {
--x;
// Is the function argument a pointer type?
if (!proto->getArgType(x)->isPointerType()) {
if (!getFunctionOrMethodArgType(d, x)->isPointerType()) {
// FIXME: Should also highlight argument in decl.
S.Diag(Attr.getLoc(), diag::err_nonnull_pointers_only,
"nonnull", Ex->getSourceRange());
@ -341,12 +354,9 @@ static void HandleNonNullAttr(Decl *d, const AttributeList &Attr, Sema &S) {
// If no arguments were specified to __attribute__((nonnull)) then all
// pointer arguments have a nonnull attribute.
if (NonNullArgs.empty()) {
unsigned idx = 0;
for (FunctionTypeProto::arg_type_iterator
I=proto->arg_type_begin(), E=proto->arg_type_end(); I!=E; ++I, ++idx)
if ((*I)->isPointerType())
NonNullArgs.push_back(idx);
for (unsigned I = 0, E = getFunctionOrMethodNumArgs(d); I != E; ++I)
if (getFunctionOrMethodArgType(d, I)->isPointerType())
NonNullArgs.push_back(I);
if (NonNullArgs.empty()) {
S.Diag(Attr.getLoc(), diag::warn_attribute_nonnull_no_pointers);
@ -394,7 +404,7 @@ static void HandleNoReturnAttr(Decl *d, const AttributeList &Attr, Sema &S) {
return;
}
if (!isa<FunctionDecl>(d) && !isa<ObjCMethodDecl>(d)) {
if (!isFunctionOrMethod(d)) {
S.Diag(Attr.getLoc(), diag::warn_attribute_wrong_decl_type,
"noreturn", "function");
return;
@ -411,7 +421,7 @@ static void HandleUnusedAttr(Decl *d, const AttributeList &Attr, Sema &S) {
return;
}
if (!isa<VarDecl>(d) && !getFunctionProto(d)) {
if (!isa<VarDecl>(d) && !isFunctionOrMethod(d)) {
S.Diag(Attr.getLoc(), diag::warn_attribute_wrong_decl_type,
"unused", "variable and function");
return;
@ -755,9 +765,7 @@ static void HandleFormatAttr(Decl *d, const AttributeList &Attr, Sema &S) {
return;
}
// GCC ignores the format attribute on K&R style function
// prototypes, so we ignore it as well
if (!isFunctionOrMethod(d)) {
if (!isFunctionOrMethod(d) || !hasFunctionProto(d)) {
S.Diag(Attr.getLoc(), diag::warn_attribute_wrong_decl_type,
"format", "function");
return;

14
test/Sema/attr-noreturn.c Normal file
Просмотреть файл

@ -0,0 +1,14 @@
// RUN: clang -verify -fsyntax-only %s
static void (*fp0)(void) __attribute__((noreturn));
static void __attribute__((noreturn)) f0(void) {
fatal();
}
// On K&R
int f1() __attribute__((noreturn));
int g0 __attribute__((noreturn)); // expected-warning {{'noreturn' attribute only applies to function types}}
int f2() __attribute__((noreturn(1, 2))); // expected-error {{attribute requires 0 argument(s)}}

12
test/Sema/attr-unused.c Normal file
Просмотреть файл

@ -0,0 +1,12 @@
// RUN: clang -verify -fsyntax-only %s
static void (*fp0)(void) __attribute__((unused));
static void __attribute__((unused)) f0(void);
// On K&R
int f1() __attribute__((unused));
int g0 __attribute__((unused));
int f2() __attribute__((unused(1, 2))); // expected-error {{attribute requires 0 argument(s)}}

Просмотреть файл

@ -1,8 +1,10 @@
// RUN: clang -fsyntax-only %s
// RUN: clang -verify -fsyntax-only %s
@class NSString;
@interface A
-t1 __attribute__((noreturn));
- (NSString *)stringByAppendingFormat:(NSString *)format, ... __attribute__((format(__NSString__, 1, 2)));
-(void) m0 __attribute__((noreturn));
-(void) m1 __attribute__((unused));
@end