180b64f08SRichard Smith //===--------------- catch_member_function_pointer_02.cpp -----------------===//
280b64f08SRichard Smith //
380b64f08SRichard Smith //                     The LLVM Compiler Infrastructure
480b64f08SRichard Smith //
580b64f08SRichard Smith // This file is dual licensed under the MIT and the University of Illinois Open
680b64f08SRichard Smith // Source Licenses. See LICENSE.TXT for details.
780b64f08SRichard Smith //
880b64f08SRichard Smith //===----------------------------------------------------------------------===//
980b64f08SRichard Smith 
1080b64f08SRichard Smith // Can a noexcept member function pointer be caught by a non-noexcept catch
1180b64f08SRichard Smith // clause?
12366bb54fSRichard Smith // UNSUPPORTED: libcxxabi-no-exceptions, libcxxabi-no-noexcept-function-type
1380b64f08SRichard Smith 
140af53563SEric Fiselier // GCC 7 and 8 support noexcept function types but this test still fails.
150af53563SEric Fiselier // This is likely a bug in their implementation. Investigation needed.
16*6fa95b87SEric Fiselier // XFAIL: gcc-7, gcc-8, gcc-9
170af53563SEric Fiselier 
1880b64f08SRichard Smith #include <cassert>
1980b64f08SRichard Smith 
2080b64f08SRichard Smith struct X {
2180b64f08SRichard Smith   template<bool Noexcept> void f() noexcept(Noexcept) {}
2280b64f08SRichard Smith };
2380b64f08SRichard Smith template<bool Noexcept> using FnType = void (X::*)() noexcept(Noexcept);
2480b64f08SRichard Smith 
2580b64f08SRichard Smith template<bool ThrowNoexcept, bool CatchNoexcept>
2680b64f08SRichard Smith void check()
2780b64f08SRichard Smith {
2880b64f08SRichard Smith     try
2980b64f08SRichard Smith     {
3080b64f08SRichard Smith         auto p = &X::f<ThrowNoexcept>;
3180b64f08SRichard Smith         throw p;
3280b64f08SRichard Smith         assert(false);
3380b64f08SRichard Smith     }
3480b64f08SRichard Smith     catch (FnType<CatchNoexcept> p)
3580b64f08SRichard Smith     {
3680b64f08SRichard Smith         assert(ThrowNoexcept || !CatchNoexcept);
3780b64f08SRichard Smith         assert(p == &X::f<ThrowNoexcept>);
3880b64f08SRichard Smith     }
3980b64f08SRichard Smith     catch (...)
4080b64f08SRichard Smith     {
4180b64f08SRichard Smith         assert(!ThrowNoexcept && CatchNoexcept);
4280b64f08SRichard Smith     }
4380b64f08SRichard Smith }
4480b64f08SRichard Smith 
4580b64f08SRichard Smith void check_deep() {
4680b64f08SRichard Smith     FnType<true> p = &X::f<true>;
4780b64f08SRichard Smith     try
4880b64f08SRichard Smith     {
4980b64f08SRichard Smith         throw &p;
5080b64f08SRichard Smith     }
5180b64f08SRichard Smith     catch (FnType<false> *q)
5280b64f08SRichard Smith     {
5380b64f08SRichard Smith         assert(false);
5480b64f08SRichard Smith     }
5580b64f08SRichard Smith     catch (FnType<true> *q)
5680b64f08SRichard Smith     {
5780b64f08SRichard Smith     }
5880b64f08SRichard Smith     catch (...)
5980b64f08SRichard Smith     {
6080b64f08SRichard Smith         assert(false);
6180b64f08SRichard Smith     }
6280b64f08SRichard Smith }
6380b64f08SRichard Smith 
6480b64f08SRichard Smith int main()
6580b64f08SRichard Smith {
6680b64f08SRichard Smith     check<false, false>();
6780b64f08SRichard Smith     check<false, true>();
6880b64f08SRichard Smith     check<true, false>();
6980b64f08SRichard Smith     check<true, true>();
7080b64f08SRichard Smith     check_deep();
7180b64f08SRichard Smith }
72