2021-11-18 05:25:01 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
2016-11-03 07:41:51 +08:00
|
|
|
//
|
2019-01-19 18:56:40 +08:00
|
|
|
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
|
|
|
|
// See https://llvm.org/LICENSE.txt for license information.
|
|
|
|
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
|
2016-11-03 07:41:51 +08:00
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2021-07-15 21:43:47 +08:00
|
|
|
// Can a noexcept member function pointer be caught by a non-noexcept catch clause?
|
2021-11-09 06:00:43 +08:00
|
|
|
// UNSUPPORTED: c++03, c++11, c++14
|
|
|
|
// UNSUPPORTED: no-exceptions
|
2021-07-15 21:43:47 +08:00
|
|
|
|
|
|
|
// Support for catching a function pointer including noexcept was shipped in macOS 10.13
|
|
|
|
// XFAIL: use_system_cxx_lib && {{.+}}-apple-macosx10.{{9|10|11|12}}
|
2016-11-03 07:41:51 +08:00
|
|
|
|
2021-08-10 06:08:39 +08:00
|
|
|
// GCC supports noexcept function types but this test still fails.
|
2017-05-09 08:11:02 +08:00
|
|
|
// This is likely a bug in their implementation. Investigation needed.
|
2022-05-31 00:38:27 +08:00
|
|
|
// XFAIL: gcc-11, gcc-12
|
2017-05-09 08:11:02 +08:00
|
|
|
|
2016-11-03 07:41:51 +08:00
|
|
|
#include <cassert>
|
|
|
|
|
|
|
|
struct X {
|
|
|
|
template<bool Noexcept> void f() noexcept(Noexcept) {}
|
|
|
|
};
|
|
|
|
template<bool Noexcept> using FnType = void (X::*)() noexcept(Noexcept);
|
|
|
|
|
|
|
|
template<bool ThrowNoexcept, bool CatchNoexcept>
|
|
|
|
void check()
|
|
|
|
{
|
|
|
|
try
|
|
|
|
{
|
|
|
|
auto p = &X::f<ThrowNoexcept>;
|
|
|
|
throw p;
|
|
|
|
assert(false);
|
|
|
|
}
|
|
|
|
catch (FnType<CatchNoexcept> p)
|
|
|
|
{
|
|
|
|
assert(ThrowNoexcept || !CatchNoexcept);
|
|
|
|
assert(p == &X::f<ThrowNoexcept>);
|
|
|
|
}
|
|
|
|
catch (...)
|
|
|
|
{
|
|
|
|
assert(!ThrowNoexcept && CatchNoexcept);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void check_deep() {
|
|
|
|
FnType<true> p = &X::f<true>;
|
|
|
|
try
|
|
|
|
{
|
|
|
|
throw &p;
|
|
|
|
}
|
|
|
|
catch (FnType<false> *q)
|
|
|
|
{
|
|
|
|
assert(false);
|
|
|
|
}
|
|
|
|
catch (FnType<true> *q)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
catch (...)
|
|
|
|
{
|
|
|
|
assert(false);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-10-09 01:36:33 +08:00
|
|
|
int main(int, char**)
|
2016-11-03 07:41:51 +08:00
|
|
|
{
|
|
|
|
check<false, false>();
|
|
|
|
check<false, true>();
|
|
|
|
check<true, false>();
|
|
|
|
check<true, true>();
|
|
|
|
check_deep();
|
2020-10-09 01:36:33 +08:00
|
|
|
|
|
|
|
return 0;
|
2016-11-03 07:41:51 +08:00
|
|
|
}
|