Add warning for functions/blocks that have attribute 'noreturn' but return a non-void result. (<rdar://problem/7562925>)

llvm-svn: 111492
This commit is contained in:
Ted Kremenek 2010-08-19 00:52:13 +00:00
parent 5295ce8120
commit 50e0105f1c
10 changed files with 28 additions and 12 deletions

View File

@ -3135,6 +3135,9 @@ def ext_return_has_void_expr : Extension<
def warn_noreturn_function_has_return_expr : Warning<
"function %0 declared 'noreturn' should not return">,
InGroup<DiagGroup<"invalid-noreturn">>;
def warn_noreturn_function_has_nonvoid_result : Warning<
"%select{functions|blocks}0 declared 'noreturn' should have a 'void' result type">,
InGroup<DiagGroup<"invalid-noreturn">>;
def warn_falloff_noreturn_function : Warning<
"function declared 'noreturn' should not return">,
InGroup<DiagGroup<"invalid-noreturn">>;

View File

@ -1833,6 +1833,14 @@ static void HandleObjCGCTypeAttribute(QualType &Type,
Type = S.Context.getObjCGCQualType(Type, GCAttr);
}
static QualType GetResultType(QualType T) {
if (const PointerType *PT = T->getAs<PointerType>())
T = PT->getPointeeType();
else if (const BlockPointerType *BT = T->getAs<BlockPointerType>())
T = BT->getPointeeType();
return T->getAs<FunctionType>()->getResultType();
}
/// Process an individual function attribute. Returns true if the
/// attribute does not make sense to apply to this type.
bool ProcessFnAttr(Sema &S, QualType &Type, const AttributeList &Attr) {
@ -1849,7 +1857,12 @@ bool ProcessFnAttr(Sema &S, QualType &Type, const AttributeList &Attr) {
&& !Type->isBlockPointerType()
&& !Type->isFunctionType())
return true;
if (!GetResultType(Type)->isVoidType()) {
S.Diag(Attr.getLoc(), diag::warn_noreturn_function_has_nonvoid_result)
<< (Type->isBlockPointerType() ? /* blocks */ 1 : /* functions */ 0);
}
// Otherwise we can process right away.
Type = S.Context.getNoReturnType(Type);
return false;

View File

@ -323,7 +323,7 @@ int test_invalidate_by_ref() {
// was the block containing the merge for '?', which would trigger an
// assertion failure.
int rdar_7027684_aux();
int rdar_7027684_aux_2() __attribute__((noreturn));
int rdar_7027684_aux_2() __attribute__((noreturn)); // expected-warning{{functions declared 'noreturn' should have a 'void' result type}}
void rdar_7027684(int x, int y) {
{}; // this empty compound statement is critical.
(rdar_7027684_aux() ? rdar_7027684_aux_2() : (void) 0);

View File

@ -9,7 +9,7 @@ static void __attribute__((noreturn)) f0(void) {
} // expected-warning {{function declared 'noreturn' should not return}}
// On K&R
int f1() __attribute__((noreturn));
int f1() __attribute__((noreturn)); // expected-warning{{functions declared 'noreturn' should have a 'void' result type}}
int g0 __attribute__((noreturn)); // expected-warning {{'noreturn' only applies to function types; type here is 'int'}}

View File

@ -100,10 +100,10 @@ int (*funcptr3[5])(long);
int sz8 = sizeof(^int (*[5])(long) {return funcptr3;}); // expected-error {{block declared as returning an array}}
void foo6() {
int (^b)(int) __attribute__((noreturn));
int (^b)(int) __attribute__((noreturn)); // expected-warning{{blocks declared 'noreturn' should have a 'void' result type}}
b = ^ (int i) __attribute__((noreturn)) { return 1; }; // expected-error {{block declared 'noreturn' should not return}}
b(1);
int (^c)(void) __attribute__((noreturn)) = ^ __attribute__((noreturn)) { return 100; }; // expected-error {{block declared 'noreturn' should not return}}
int (^c)(void) __attribute__((noreturn)) = ^ __attribute__((noreturn)) { return 100; }; // expected-error {{block declared 'noreturn' should not return}} expected-warning{{blocks declared 'noreturn' should have a 'void' result type}}
}

View File

@ -20,11 +20,11 @@ void test2_positive() {
// This test case illustrates that we don't warn about the missing return
// because the function is marked noreturn and there is an infinite loop.
extern int foo_test_3();
__attribute__((__noreturn__)) void* test3(int arg) {
__attribute__((__noreturn__)) void* test3(int arg) { // expected-warning{{functions declared 'noreturn' should have a 'void' result type}}
while (1) foo_test_3();
}
__attribute__((__noreturn__)) void* test3_positive(int arg) {
__attribute__((__noreturn__)) void* test3_positive(int arg) { // expected-warning{{functions declared 'noreturn' should have a 'void' result type}}
while (0) foo_test_3();
} // expected-warning{{function declared 'noreturn' should not return}}

View File

@ -60,7 +60,7 @@ int test8() {
(void)(1 + unknown());
} // expected-warning {{control reaches end of non-void function}}
int halt3() __attribute__((noreturn));
int halt3() __attribute__((noreturn)); // expected-warning{{functions declared 'noreturn' should have a 'void' result type}}
int test9() {
(void)(halt3() + unknown());

View File

@ -1,6 +1,6 @@
// RUN: %clang %s -fsyntax-only -Xclang -verify -fblocks -Wunreachable-code -Wno-unused-value
int halt() __attribute__((noreturn));
int halt() __attribute__((noreturn)); // expected-warning{{functions declared 'noreturn' should have a 'void' result type}}
int live();
int dead();

View File

@ -31,7 +31,7 @@ void test_f3() {
class xpto {
int blah() __attribute__((noreturn));
int blah() __attribute__((noreturn)); // expected-warning{{functions declared 'noreturn' should have a 'void' result type}}
};
int xpto::blah() {

View File

@ -1,6 +1,6 @@
// RUN: %clang_cc1 %s -fsyntax-only -verify -fblocks -Wunreachable-code -Wno-unused-value
int &halt() __attribute__((noreturn));
int &halt() __attribute__((noreturn)); // expected-warning{{functions declared 'noreturn' should have a 'void' result type}}
int &live();
int dead();
int liveti() throw(int);
@ -60,7 +60,7 @@ void test5() {
struct S {
int mem;
} s;
S &foor() __attribute__((noreturn));
S &foor() __attribute__((noreturn)); // expected-warning{{functions declared 'noreturn' should have a 'void' result type}}
foor()
.mem; // expected-warning {{will never be executed}}
}