180b64f08SRichard Smith //===--------------- catch_member_function_pointer_02.cpp -----------------===// 280b64f08SRichard Smith // 357b08b09SChandler Carruth // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. 457b08b09SChandler Carruth // See https://llvm.org/LICENSE.txt for license information. 557b08b09SChandler Carruth // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception 680b64f08SRichard Smith // 780b64f08SRichard Smith //===----------------------------------------------------------------------===// 880b64f08SRichard Smith 9*8fb47456SLouis Dionne // Can a noexcept member function pointer be caught by a non-noexcept catch clause? 10*8fb47456SLouis Dionne // UNSUPPORTED: no-exceptions, no-noexcept-function-type 11*8fb47456SLouis Dionne 12*8fb47456SLouis Dionne // Support for catching a function pointer including noexcept was shipped in macOS 10.13 13*8fb47456SLouis Dionne // XFAIL: use_system_cxx_lib && {{.+}}-apple-macosx10.{{9|10|11|12}} 1480b64f08SRichard Smith 150af53563SEric Fiselier // GCC 7 and 8 support noexcept function types but this test still fails. 160af53563SEric Fiselier // This is likely a bug in their implementation. Investigation needed. 174f194d0dSLouis Dionne // XFAIL: gcc-7, gcc-8, gcc-9, gcc-10, gcc-11 180af53563SEric Fiselier 1980b64f08SRichard Smith #include <cassert> 2080b64f08SRichard Smith 2180b64f08SRichard Smith struct X { 2280b64f08SRichard Smith template<bool Noexcept> void f() noexcept(Noexcept) {} 2380b64f08SRichard Smith }; 2480b64f08SRichard Smith template<bool Noexcept> using FnType = void (X::*)() noexcept(Noexcept); 2580b64f08SRichard Smith 2680b64f08SRichard Smith template<bool ThrowNoexcept, bool CatchNoexcept> 2780b64f08SRichard Smith void check() 2880b64f08SRichard Smith { 2980b64f08SRichard Smith try 3080b64f08SRichard Smith { 3180b64f08SRichard Smith auto p = &X::f<ThrowNoexcept>; 3280b64f08SRichard Smith throw p; 3380b64f08SRichard Smith assert(false); 3480b64f08SRichard Smith } 3580b64f08SRichard Smith catch (FnType<CatchNoexcept> p) 3680b64f08SRichard Smith { 3780b64f08SRichard Smith assert(ThrowNoexcept || !CatchNoexcept); 3880b64f08SRichard Smith assert(p == &X::f<ThrowNoexcept>); 3980b64f08SRichard Smith } 4080b64f08SRichard Smith catch (...) 4180b64f08SRichard Smith { 4280b64f08SRichard Smith assert(!ThrowNoexcept && CatchNoexcept); 4380b64f08SRichard Smith } 4480b64f08SRichard Smith } 4580b64f08SRichard Smith 4680b64f08SRichard Smith void check_deep() { 4780b64f08SRichard Smith FnType<true> p = &X::f<true>; 4880b64f08SRichard Smith try 4980b64f08SRichard Smith { 5080b64f08SRichard Smith throw &p; 5180b64f08SRichard Smith } 5280b64f08SRichard Smith catch (FnType<false> *q) 5380b64f08SRichard Smith { 5480b64f08SRichard Smith assert(false); 5580b64f08SRichard Smith } 5680b64f08SRichard Smith catch (FnType<true> *q) 5780b64f08SRichard Smith { 5880b64f08SRichard Smith } 5980b64f08SRichard Smith catch (...) 6080b64f08SRichard Smith { 6180b64f08SRichard Smith assert(false); 6280b64f08SRichard Smith } 6380b64f08SRichard Smith } 6480b64f08SRichard Smith 65504bc07dSLouis Dionne int main(int, char**) 6680b64f08SRichard Smith { 6780b64f08SRichard Smith check<false, false>(); 6880b64f08SRichard Smith check<false, true>(); 6980b64f08SRichard Smith check<true, false>(); 7080b64f08SRichard Smith check<true, true>(); 7180b64f08SRichard Smith check_deep(); 72504bc07dSLouis Dionne 73504bc07dSLouis Dionne return 0; 7480b64f08SRichard Smith } 75