// RUN: %clang_cc1 -fsyntax-only -verify %s -Wmissing-noreturn -Wreturn-type
void f() __attribute__((noreturn));
template<typename T> void g(T) {
f();
}
template void g<int>(int);
template<typename T> struct A {
void g() {
f();
}
};
template struct A<int>;
struct B {
template<typename T> void g(T) {
f();
}
};
template void B::g<int>(int);
// We don't want a warning here.
struct X {
virtual void g() { f(); }
};
namespace test1 {
bool condition();
// We don't want a warning here.
void foo() {
while (condition()) {}
}
}
// This test case previously had a false "missing return" warning.
struct R7880658 {
R7880658 &operator++();
bool operator==(const R7880658 &) const;
bool operator!=(const R7880658 &) const;
};
void f_R7880658(R7880658 f, R7880658 l) { // no-warning
for (; f != l; ++f) {
}
}
namespace test2 {
bool g();
void *h() __attribute__((noreturn));
void *j();
struct A {
void *f;
A() : f(0) { }
A(int) : f(h()) { } // expected-warning {{function 'A' could be declared with attribute 'noreturn'}}
A(char) : f(j()) { }
A(bool b) : f(b ? h() : j()) { }
};
}
namespace test3 {
struct A {
~A();
};
struct B {
~B() { }
A a;
};
struct C : A {
~C() { }
};
}
// Properly handle CFGs with destructors.
struct rdar8875247 {
~rdar8875247 ();
};
void rdar8875247_aux();
int rdar8875247_test() {
rdar8875247 f;
} // expected-warning{{non-void function does not return a value}}
struct rdar8875247_B {
rdar8875247_B();
~rdar8875247_B();
};
rdar8875247_B test_rdar8875247_B() {
rdar8875247_B f;
return f;
} // no-warning
namespace PR10801 {
struct Foo {
void wibble() __attribute((__noreturn__));
};
struct Bar {
void wibble();
};
template <typename T> void thingy(T thing) {
thing.wibble();
}
void test() {
Foo f;
Bar b;
thingy(f);
thingy(b);
}
}