2017-09-27 02:20:39 +08:00
|
|
|
// RUN: %clang_cc1 %s -fdelayed-template-parsing -fcxx-exceptions -fsyntax-only -Wexceptions -verify -fdeclspec -std=c++11
|
2017-06-24 04:30:33 +08:00
|
|
|
struct A_ShouldDiag {
|
|
|
|
~A_ShouldDiag(); // implicitly noexcept(true)
|
|
|
|
};
|
2017-07-06 00:43:45 +08:00
|
|
|
A_ShouldDiag::~A_ShouldDiag() { // expected-note {{destructor has a implicit non-throwing exception specification}}
|
|
|
|
throw 1; // expected-warning {{has a non-throwing exception specification but can still throw}}
|
2017-06-24 04:30:33 +08:00
|
|
|
}
|
|
|
|
struct B_ShouldDiag {
|
|
|
|
int i;
|
|
|
|
~B_ShouldDiag() noexcept(true) {} //no disg, no throw stmt
|
|
|
|
};
|
|
|
|
struct R_ShouldDiag : A_ShouldDiag {
|
|
|
|
B_ShouldDiag b;
|
2017-07-06 00:43:45 +08:00
|
|
|
~R_ShouldDiag() { // expected-note {{destructor has a implicit non-throwing exception specification}}
|
2017-06-24 04:30:33 +08:00
|
|
|
throw 1; // expected-warning {{has a non-throwing exception specification but}}
|
|
|
|
}
|
2017-09-27 02:20:39 +08:00
|
|
|
__attribute__((nothrow)) R_ShouldDiag() {// expected-note {{function declared non-throwing here}}
|
|
|
|
throw 1;// expected-warning {{has a non-throwing exception specification but}}
|
|
|
|
}
|
|
|
|
void __attribute__((nothrow)) SomeThrow() {// expected-note {{function declared non-throwing here}}
|
|
|
|
throw 1; // expected-warning {{has a non-throwing exception specification but}}
|
|
|
|
}
|
|
|
|
void __declspec(nothrow) SomeDeclspecThrow() {// expected-note {{function declared non-throwing here}}
|
|
|
|
throw 1; // expected-warning {{has a non-throwing exception specification but}}
|
|
|
|
}
|
2017-06-24 04:30:33 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
struct M_ShouldNotDiag {
|
|
|
|
B_ShouldDiag b;
|
|
|
|
~M_ShouldNotDiag() noexcept(false);
|
|
|
|
};
|
|
|
|
|
|
|
|
M_ShouldNotDiag::~M_ShouldNotDiag() noexcept(false) {
|
|
|
|
throw 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
struct N_ShouldDiag {
|
|
|
|
B_ShouldDiag b;
|
|
|
|
~N_ShouldDiag(); //implicitly noexcept(true)
|
|
|
|
};
|
|
|
|
|
2017-07-06 00:43:45 +08:00
|
|
|
N_ShouldDiag::~N_ShouldDiag() { // expected-note {{destructor has a implicit non-throwing exception specification}}
|
2017-06-24 04:30:33 +08:00
|
|
|
throw 1; // expected-warning {{has a non-throwing exception specification but}}
|
|
|
|
}
|
|
|
|
struct X_ShouldDiag {
|
|
|
|
B_ShouldDiag b;
|
2017-07-06 00:43:45 +08:00
|
|
|
~X_ShouldDiag() noexcept { // expected-note {{destructor has a non-throwing exception}}
|
|
|
|
throw 1; // expected-warning {{has a non-throwing exception specification but}}
|
2017-06-24 04:30:33 +08:00
|
|
|
}
|
|
|
|
};
|
|
|
|
struct Y_ShouldDiag : A_ShouldDiag {
|
2017-07-06 00:43:45 +08:00
|
|
|
~Y_ShouldDiag() noexcept(true) { // expected-note {{destructor has a non-throwing exception specification}}
|
|
|
|
throw 1; // expected-warning {{has a non-throwing exception specification but}}
|
2017-06-24 04:30:33 +08:00
|
|
|
}
|
|
|
|
};
|
|
|
|
struct C_ShouldNotDiag {
|
|
|
|
int i;
|
|
|
|
~C_ShouldNotDiag() noexcept(false) {}
|
|
|
|
};
|
|
|
|
struct D_ShouldNotDiag {
|
|
|
|
C_ShouldNotDiag c;
|
|
|
|
~D_ShouldNotDiag() { //implicitly noexcept(false)
|
|
|
|
throw 1;
|
|
|
|
}
|
|
|
|
};
|
2017-07-06 00:43:45 +08:00
|
|
|
struct E_ShouldNotDiag {
|
2017-06-24 04:30:33 +08:00
|
|
|
C_ShouldNotDiag c;
|
|
|
|
~E_ShouldNotDiag(); //implicitly noexcept(false)
|
|
|
|
};
|
|
|
|
E_ShouldNotDiag::~E_ShouldNotDiag() //implicitly noexcept(false)
|
|
|
|
{
|
|
|
|
throw 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename T>
|
|
|
|
class A1_ShouldDiag {
|
|
|
|
T b;
|
|
|
|
|
|
|
|
public:
|
2017-07-06 00:43:45 +08:00
|
|
|
~A1_ShouldDiag() { // expected-note {{destructor has a implicit non-throwing exception specification}}
|
2017-06-24 04:30:33 +08:00
|
|
|
throw 1; // expected-warning {{has a non-throwing exception specification but}}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
template <typename T>
|
|
|
|
struct B1_ShouldDiag {
|
|
|
|
T i;
|
|
|
|
~B1_ShouldDiag() noexcept(true) {}
|
|
|
|
};
|
|
|
|
template <typename T>
|
|
|
|
struct R1_ShouldDiag : A1_ShouldDiag<T> //expected-note {{in instantiation of member function}}
|
|
|
|
{
|
|
|
|
B1_ShouldDiag<T> b;
|
2017-07-06 00:43:45 +08:00
|
|
|
~R1_ShouldDiag() { // expected-note {{destructor has a implicit non-throwing exception specification}}
|
2017-06-24 04:30:33 +08:00
|
|
|
throw 1; // expected-warning {{has a non-throwing exception specification but}}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
template <typename T>
|
|
|
|
struct S1_ShouldDiag : A1_ShouldDiag<T> {
|
|
|
|
B1_ShouldDiag<T> b;
|
2017-07-06 00:43:45 +08:00
|
|
|
~S1_ShouldDiag() noexcept { // expected-note {{destructor has a non-throwing exception specification}}
|
|
|
|
throw 1; // expected-warning {{has a non-throwing exception specification but}}
|
2017-06-24 04:30:33 +08:00
|
|
|
}
|
|
|
|
};
|
2017-07-06 00:43:45 +08:00
|
|
|
void operator delete(void *ptr) noexcept { // expected-note {{deallocator has a non-throwing exception specification}}
|
|
|
|
throw 1; // expected-warning {{has a non-throwing exception specification but}}
|
2017-06-24 04:30:33 +08:00
|
|
|
}
|
|
|
|
struct except_fun {
|
|
|
|
static const bool i = false;
|
|
|
|
};
|
|
|
|
struct noexcept_fun {
|
|
|
|
static const bool i = true;
|
|
|
|
};
|
|
|
|
template <typename T>
|
|
|
|
struct dependent_warn {
|
|
|
|
~dependent_warn() noexcept(T::i) {
|
|
|
|
throw 1;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
template <typename T>
|
|
|
|
struct dependent_warn_noexcept {
|
2017-07-06 00:43:45 +08:00
|
|
|
~dependent_warn_noexcept() noexcept(T::i) { // expected-note {{destructor has a non-throwing exception specification}}
|
|
|
|
throw 1; // expected-warning {{has a non-throwing exception specification but}}
|
2017-06-24 04:30:33 +08:00
|
|
|
}
|
|
|
|
};
|
|
|
|
template <typename T>
|
|
|
|
struct dependent_warn_both {
|
2017-07-06 00:43:45 +08:00
|
|
|
~dependent_warn_both() noexcept(T::i) { // expected-note {{destructor has a non-throwing exception specification}}
|
|
|
|
throw 1; // expected-warning {{has a non-throwing exception specification but}}
|
2017-06-24 04:30:33 +08:00
|
|
|
}
|
|
|
|
};
|
2017-07-06 00:43:45 +08:00
|
|
|
void foo() noexcept { //expected-note {{function declared non-throwing here}}
|
|
|
|
throw 1; // expected-warning {{has a non-throwing exception specification but}}
|
2017-06-24 04:30:33 +08:00
|
|
|
}
|
|
|
|
struct Throws {
|
|
|
|
~Throws() noexcept(false);
|
|
|
|
};
|
|
|
|
|
|
|
|
struct ShouldDiagnose {
|
|
|
|
Throws T;
|
2017-07-06 00:43:45 +08:00
|
|
|
~ShouldDiagnose() noexcept { //expected-note {{destructor has a non-throwing exception specification}}
|
2017-06-24 04:30:33 +08:00
|
|
|
throw; // expected-warning {{has a non-throwing exception specification but}}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
struct ShouldNotDiagnose {
|
|
|
|
Throws T;
|
2017-07-06 00:43:45 +08:00
|
|
|
~ShouldNotDiagnose() {
|
|
|
|
throw;
|
2017-06-24 04:30:33 +08:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
void bar_ShouldNotDiag() noexcept {
|
|
|
|
try {
|
|
|
|
throw 1;
|
|
|
|
} catch (...) {
|
|
|
|
}
|
|
|
|
}
|
|
|
|
void f_ShouldNotDiag() noexcept {
|
|
|
|
try {
|
|
|
|
throw 12;
|
|
|
|
} catch (int) {
|
|
|
|
}
|
|
|
|
}
|
|
|
|
void g_ShouldNotDiag() noexcept {
|
|
|
|
try {
|
|
|
|
throw 12;
|
|
|
|
} catch (...) {
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-07-06 00:43:45 +08:00
|
|
|
void h_ShouldDiag() noexcept { //expected-note {{function declared non-throwing here}}
|
2017-06-24 04:30:33 +08:00
|
|
|
try {
|
|
|
|
throw 12; // expected-warning {{has a non-throwing exception specification but}}
|
|
|
|
} catch (const char *) {
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-07-06 00:43:45 +08:00
|
|
|
void i_ShouldDiag() noexcept { //expected-note {{function declared non-throwing here}}
|
2017-06-24 04:30:33 +08:00
|
|
|
try {
|
|
|
|
throw 12;
|
|
|
|
} catch (int) {
|
|
|
|
throw; // expected-warning {{has a non-throwing exception specification but}}
|
|
|
|
}
|
|
|
|
}
|
2017-07-06 00:43:45 +08:00
|
|
|
void j_ShouldDiag() noexcept { //expected-note {{function declared non-throwing here}}
|
2017-06-24 04:30:33 +08:00
|
|
|
try {
|
|
|
|
throw 12;
|
|
|
|
} catch (int) {
|
|
|
|
throw "haha"; // expected-warning {{has a non-throwing exception specification but}}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-07-06 00:43:45 +08:00
|
|
|
void k_ShouldDiag() noexcept { //expected-note {{function declared non-throwing here}}
|
2017-06-24 04:30:33 +08:00
|
|
|
try {
|
|
|
|
throw 12;
|
|
|
|
} catch (...) {
|
|
|
|
throw; // expected-warning {{has a non-throwing exception specification but}}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-07-06 00:43:45 +08:00
|
|
|
void loo_ShouldDiag(int i) noexcept { //expected-note {{function declared non-throwing here}}
|
2017-06-24 04:30:33 +08:00
|
|
|
if (i)
|
|
|
|
try {
|
|
|
|
throw 12;
|
|
|
|
} catch (int) {
|
|
|
|
throw "haha"; //expected-warning {{has a non-throwing exception specification but}}
|
|
|
|
}
|
|
|
|
i = 10;
|
|
|
|
}
|
|
|
|
|
|
|
|
void loo1_ShouldNotDiag() noexcept {
|
|
|
|
if (0)
|
|
|
|
throw 12;
|
|
|
|
}
|
|
|
|
|
2017-07-06 00:43:45 +08:00
|
|
|
void loo2_ShouldDiag() noexcept { //expected-note {{function declared non-throwing here}}
|
2017-06-24 04:30:33 +08:00
|
|
|
if (1)
|
|
|
|
throw 12; // expected-warning {{has a non-throwing exception specification but}}
|
|
|
|
}
|
|
|
|
struct S {};
|
|
|
|
|
2017-07-06 00:43:45 +08:00
|
|
|
void l_ShouldDiag() noexcept { //expected-note {{function declared non-throwing here}}
|
2017-06-24 04:30:33 +08:00
|
|
|
try {
|
|
|
|
throw S{}; //expected-warning {{has a non-throwing exception specification but}}
|
|
|
|
} catch (S *s) {
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void m_ShouldNotDiag() noexcept {
|
|
|
|
try {
|
|
|
|
const S &s = S{};
|
|
|
|
throw s;
|
|
|
|
} catch (S s) {
|
|
|
|
}
|
|
|
|
}
|
|
|
|
void n_ShouldNotDiag() noexcept {
|
|
|
|
try {
|
|
|
|
S s = S{};
|
|
|
|
throw s;
|
|
|
|
} catch (const S &s) {
|
|
|
|
}
|
|
|
|
}
|
2017-10-18 04:57:24 +08:00
|
|
|
// As seen in p34973, this should not throw the warning. If there is an active
|
|
|
|
// exception, catch(...) catches everything.
|
|
|
|
void o_ShouldNotDiag() noexcept {
|
2017-06-24 04:30:33 +08:00
|
|
|
try {
|
2017-10-18 04:57:24 +08:00
|
|
|
throw;
|
2017-06-24 04:30:33 +08:00
|
|
|
} catch (...) {
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-10-18 04:57:24 +08:00
|
|
|
void p_ShouldDiag() noexcept { //expected-note {{function declared non-throwing here}}
|
|
|
|
try {
|
|
|
|
throw; //expected-warning {{has a non-throwing exception specification but}}
|
|
|
|
} catch (int){
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void q_ShouldNotDiag() noexcept {
|
|
|
|
try {
|
|
|
|
throw;
|
|
|
|
} catch (int){
|
|
|
|
} catch (...){
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-06-24 04:30:33 +08:00
|
|
|
#define NOEXCEPT noexcept
|
2017-07-06 00:43:45 +08:00
|
|
|
void with_macro() NOEXCEPT { //expected-note {{function declared non-throwing here}}
|
2017-06-24 04:30:33 +08:00
|
|
|
throw 1; // expected-warning {{has a non-throwing exception specification but}}
|
|
|
|
}
|
|
|
|
|
|
|
|
void with_try_block() try {
|
|
|
|
throw 2;
|
|
|
|
} catch (...) {
|
|
|
|
}
|
|
|
|
|
2017-07-06 00:43:45 +08:00
|
|
|
void with_try_block1() noexcept try { //expected-note {{function declared non-throwing here}}
|
|
|
|
throw 2; // expected-warning {{has a non-throwing exception specification but}}
|
2017-06-24 04:30:33 +08:00
|
|
|
} catch (char *) {
|
|
|
|
}
|
|
|
|
|
2017-06-30 15:22:02 +08:00
|
|
|
namespace derived {
|
|
|
|
struct B {};
|
|
|
|
struct D: B {};
|
|
|
|
void goodPlain() noexcept {
|
|
|
|
try {
|
|
|
|
throw D();
|
|
|
|
} catch (B) {}
|
|
|
|
}
|
|
|
|
void goodReference() noexcept {
|
|
|
|
try {
|
|
|
|
throw D();
|
|
|
|
} catch (B &) {}
|
|
|
|
}
|
|
|
|
void goodPointer() noexcept {
|
|
|
|
D d;
|
|
|
|
try {
|
|
|
|
throw &d;
|
|
|
|
} catch (B *) {}
|
|
|
|
}
|
2017-07-06 00:43:45 +08:00
|
|
|
void badPlain() noexcept { //expected-note {{function declared non-throwing here}}
|
2017-06-30 15:22:02 +08:00
|
|
|
try {
|
2017-07-06 00:43:45 +08:00
|
|
|
throw B(); // expected-warning {{'badPlain' has a non-throwing exception specification but can still throw}}
|
2017-06-30 15:22:02 +08:00
|
|
|
} catch (D) {}
|
|
|
|
}
|
2017-07-06 00:43:45 +08:00
|
|
|
void badReference() noexcept { //expected-note {{function declared non-throwing here}}
|
2017-06-30 15:22:02 +08:00
|
|
|
try {
|
2017-07-06 00:43:45 +08:00
|
|
|
throw B(); // expected-warning {{'badReference' has a non-throwing exception specification but can still throw}}
|
2017-06-30 15:22:02 +08:00
|
|
|
} catch (D &) {}
|
|
|
|
}
|
2017-07-06 00:43:45 +08:00
|
|
|
void badPointer() noexcept { //expected-note {{function declared non-throwing here}}
|
2017-06-30 15:22:02 +08:00
|
|
|
B b;
|
|
|
|
try {
|
2017-07-06 00:43:45 +08:00
|
|
|
throw &b; // expected-warning {{'badPointer' has a non-throwing exception specification but can still throw}}
|
2017-06-30 15:22:02 +08:00
|
|
|
} catch (D *) {}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-06-24 04:30:33 +08:00
|
|
|
int main() {
|
|
|
|
R1_ShouldDiag<int> o; //expected-note {{in instantiation of member function}}
|
|
|
|
S1_ShouldDiag<int> b; //expected-note {{in instantiation of member function}}
|
|
|
|
dependent_warn<except_fun> f;
|
|
|
|
dependent_warn_noexcept<noexcept_fun> f1; //expected-note {{in instantiation of member function}}
|
|
|
|
dependent_warn_both<except_fun> f2;
|
|
|
|
dependent_warn_both<noexcept_fun> f3; //expected-note {{in instantiation of member function}}
|
|
|
|
ShouldDiagnose obj;
|
|
|
|
ShouldNotDiagnose obj1;
|
|
|
|
}
|