105 lines
4.0 KiB
C++
105 lines
4.0 KiB
C++
|
// RUN: %clang_cc1 %s -fcxx-exceptions -fdeclspec -fsyntax-only -Wexceptions -verify -std=c++14
|
||
|
// RUN: %clang_cc1 %s -fcxx-exceptions -fdeclspec -fsyntax-only -Wexceptions -verify -std=c++17 -DCPP17
|
||
|
|
||
|
__attribute__((nothrow)) void f1();
|
||
|
static_assert(noexcept(f1()), "");
|
||
|
void f1() noexcept;
|
||
|
// expected-error@+2 {{exception specification in declaration does not match previous declaration}}
|
||
|
// expected-note@-2 {{previous declaration is here}}
|
||
|
void f1() noexcept(false);
|
||
|
|
||
|
__attribute__((nothrow)) void f2();
|
||
|
static_assert(noexcept(f2()), "");
|
||
|
// expected-error@+2 {{exception specification in declaration does not match previous declaration}}
|
||
|
// expected-note@-3 {{previous declaration is here}}
|
||
|
void f2() noexcept(false);
|
||
|
|
||
|
void f3() __attribute__((nothrow));
|
||
|
static_assert(noexcept(f3()), "");
|
||
|
void f3() noexcept;
|
||
|
// expected-error@+2 {{exception specification in declaration does not match previous declaration}}
|
||
|
// expected-note@-2 {{previous declaration is here}}
|
||
|
void f3() noexcept(false);
|
||
|
|
||
|
// Still noexcept due to throw()
|
||
|
__attribute__((nothrow)) void f4() throw();
|
||
|
static_assert(noexcept(f4()), "");
|
||
|
|
||
|
// Still noexcept due to noexcept
|
||
|
__attribute__((nothrow)) void f5() noexcept;
|
||
|
static_assert(noexcept(f5()), "");
|
||
|
|
||
|
// Still noexcept due to noexcept(true)
|
||
|
__attribute__((nothrow)) void f6() noexcept(true);
|
||
|
static_assert(noexcept(f6()), "");
|
||
|
|
||
|
#ifndef CPP17
|
||
|
// Doesn't override C++ implementation.
|
||
|
// expected-warning@+1{{'nothrow' attribute conflicts with exception specification; attribute ignored}}
|
||
|
__attribute__((nothrow)) void f7() throw(int);
|
||
|
static_assert(!noexcept(f7()), "");
|
||
|
#endif
|
||
|
|
||
|
// Doesn't override C++ implementation.
|
||
|
// expected-warning@+1{{'nothrow' attribute conflicts with exception specification; attribute ignored}}
|
||
|
__attribute__((nothrow)) void f8() noexcept(false);
|
||
|
static_assert(!noexcept(f8()), "");
|
||
|
|
||
|
__declspec(nothrow) void foo1() noexcept;
|
||
|
__declspec(nothrow) void foo2() noexcept(true);
|
||
|
// expected-warning@+1{{'nothrow' attribute conflicts with exception specification; attribute ignored}}
|
||
|
__declspec(nothrow) void foo3() noexcept(false);
|
||
|
__declspec(nothrow) void foo4() noexcept(noexcept(foo1()));
|
||
|
__declspec(nothrow) void foo5() noexcept(noexcept(foo2()));
|
||
|
// expected-warning@+1{{'nothrow' attribute conflicts with exception specification; attribute ignored}}
|
||
|
__declspec(nothrow) void foo6() noexcept(noexcept(foo3()));
|
||
|
|
||
|
template<typename F>
|
||
|
__declspec(nothrow) void foo7() noexcept(noexcept(F()));
|
||
|
|
||
|
// FIXME: It would be nice to be able to warn on these, however at the time we
|
||
|
// evaluate the nothrow, these have yet to be parsed, so the data is not yet
|
||
|
// there.
|
||
|
struct S {
|
||
|
__declspec(nothrow) void f1();
|
||
|
#ifndef CPP17
|
||
|
__declspec(nothrow) void f2() throw();
|
||
|
__declspec(nothrow) void f3() throw(int);
|
||
|
#endif
|
||
|
__declspec(nothrow) void f4() noexcept(true);
|
||
|
__declspec(nothrow) void f5() noexcept(false);
|
||
|
};
|
||
|
|
||
|
namespace PR42100 {
|
||
|
class Base {
|
||
|
public:
|
||
|
// expected-note@+1{{overridden virtual function is here}}
|
||
|
virtual __declspec(nothrow) void foo() = 0;
|
||
|
// expected-note@+1{{previous declaration is here}}
|
||
|
__declspec(nothrow) void bar();
|
||
|
};
|
||
|
|
||
|
// expected-warning@+1{{'bar' is missing exception specification '__attribute__((nothrow))'}}
|
||
|
void Base::bar() {}
|
||
|
|
||
|
class Sub : public Base {
|
||
|
public:
|
||
|
// expected-warning@+1{{exception specification of overriding function is more lax than base version}}
|
||
|
void foo() {}
|
||
|
};
|
||
|
}
|
||
|
|
||
|
namespace FuncPointerReferenceConverts {
|
||
|
void FuncToBeRefed();
|
||
|
|
||
|
#ifndef CPP17
|
||
|
// expected-error@+6{{target exception specification is not superset of source}}
|
||
|
// expected-error@+6{{target exception specification is not superset of source}}
|
||
|
#else
|
||
|
// expected-error@+3{{non-const lvalue reference to type 'void () __attribute__((nothrow))' cannot bind to a value of unrelated type 'void ()'}}
|
||
|
// expected-error@+3{{cannot initialize a variable of type 'void (*)() __attribute__((nothrow))' with an lvalue of type 'void ()': different exception specifications}}
|
||
|
#endif
|
||
|
__declspec(nothrow) void (&FuncRef)() = FuncToBeRefed;
|
||
|
__declspec(nothrow) void (*FuncPtr)() = FuncToBeRefed;
|
||
|
}
|