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 980b64f08SRichard Smith // Can a noexcept member function pointer be caught by a non-noexcept catch 1080b64f08SRichard Smith // clause? 118c61114cSLouis Dionne // UNSUPPORTED: no-exceptions, libcxxabi-no-noexcept-function-type 1280b64f08SRichard Smith 130af53563SEric Fiselier // GCC 7 and 8 support noexcept function types but this test still fails. 140af53563SEric Fiselier // This is likely a bug in their implementation. Investigation needed. 15*4f194d0dSLouis Dionne // XFAIL: gcc-7, gcc-8, gcc-9, gcc-10, gcc-11 160af53563SEric Fiselier 1780b64f08SRichard Smith #include <cassert> 1880b64f08SRichard Smith 1980b64f08SRichard Smith struct X { 2080b64f08SRichard Smith template<bool Noexcept> void f() noexcept(Noexcept) {} 2180b64f08SRichard Smith }; 2280b64f08SRichard Smith template<bool Noexcept> using FnType = void (X::*)() noexcept(Noexcept); 2380b64f08SRichard Smith 2480b64f08SRichard Smith template<bool ThrowNoexcept, bool CatchNoexcept> 2580b64f08SRichard Smith void check() 2680b64f08SRichard Smith { 2780b64f08SRichard Smith try 2880b64f08SRichard Smith { 2980b64f08SRichard Smith auto p = &X::f<ThrowNoexcept>; 3080b64f08SRichard Smith throw p; 3180b64f08SRichard Smith assert(false); 3280b64f08SRichard Smith } 3380b64f08SRichard Smith catch (FnType<CatchNoexcept> p) 3480b64f08SRichard Smith { 3580b64f08SRichard Smith assert(ThrowNoexcept || !CatchNoexcept); 3680b64f08SRichard Smith assert(p == &X::f<ThrowNoexcept>); 3780b64f08SRichard Smith } 3880b64f08SRichard Smith catch (...) 3980b64f08SRichard Smith { 4080b64f08SRichard Smith assert(!ThrowNoexcept && CatchNoexcept); 4180b64f08SRichard Smith } 4280b64f08SRichard Smith } 4380b64f08SRichard Smith 4480b64f08SRichard Smith void check_deep() { 4580b64f08SRichard Smith FnType<true> p = &X::f<true>; 4680b64f08SRichard Smith try 4780b64f08SRichard Smith { 4880b64f08SRichard Smith throw &p; 4980b64f08SRichard Smith } 5080b64f08SRichard Smith catch (FnType<false> *q) 5180b64f08SRichard Smith { 5280b64f08SRichard Smith assert(false); 5380b64f08SRichard Smith } 5480b64f08SRichard Smith catch (FnType<true> *q) 5580b64f08SRichard Smith { 5680b64f08SRichard Smith } 5780b64f08SRichard Smith catch (...) 5880b64f08SRichard Smith { 5980b64f08SRichard Smith assert(false); 6080b64f08SRichard Smith } 6180b64f08SRichard Smith } 6280b64f08SRichard Smith 63504bc07dSLouis Dionne int main(int, char**) 6480b64f08SRichard Smith { 6580b64f08SRichard Smith check<false, false>(); 6680b64f08SRichard Smith check<false, true>(); 6780b64f08SRichard Smith check<true, false>(); 6880b64f08SRichard Smith check<true, true>(); 6980b64f08SRichard Smith check_deep(); 70504bc07dSLouis Dionne 71504bc07dSLouis Dionne return 0; 7280b64f08SRichard Smith } 73