172 lines
2.4 KiB
C++
172 lines
2.4 KiB
C++
|
//===--------------- catch_member_function_pointer_01.cpp -----------------===//
|
||
|
//
|
||
|
// The LLVM Compiler Infrastructure
|
||
|
//
|
||
|
// This file is dual licensed under the MIT and the University of Illinois Open
|
||
|
// Source Licenses. See LICENSE.TXT for details.
|
||
|
//
|
||
|
//===----------------------------------------------------------------------===//
|
||
|
|
||
|
// GCC incorrectly allows PMF type "void (T::*)()" to be caught as "void (T::*)() const"
|
||
|
// See https://gcc.gnu.org/bugzilla/show_bug.cgi?id=69375
|
||
|
// XFAIL: gcc
|
||
|
// UNSUPPORTED: libcxxabi-no-exceptions
|
||
|
#include <cassert>
|
||
|
|
||
|
struct A
|
||
|
{
|
||
|
void foo() {}
|
||
|
void bar() const {}
|
||
|
};
|
||
|
|
||
|
typedef void (A::*mf1)();
|
||
|
typedef void (A::*mf2)() const;
|
||
|
|
||
|
struct B : public A
|
||
|
{
|
||
|
};
|
||
|
|
||
|
typedef void (B::*dmf1)();
|
||
|
typedef void (B::*dmf2)() const;
|
||
|
|
||
|
template <class Tp>
|
||
|
bool can_convert(Tp) { return true; }
|
||
|
|
||
|
template <class>
|
||
|
bool can_convert(...) { return false; }
|
||
|
|
||
|
|
||
|
void test1()
|
||
|
{
|
||
|
try
|
||
|
{
|
||
|
throw &A::foo;
|
||
|
assert(false);
|
||
|
}
|
||
|
catch (mf2)
|
||
|
{
|
||
|
assert(false);
|
||
|
}
|
||
|
catch (mf1)
|
||
|
{
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void test2()
|
||
|
{
|
||
|
try
|
||
|
{
|
||
|
throw &A::bar;
|
||
|
assert(false);
|
||
|
}
|
||
|
catch (mf1)
|
||
|
{
|
||
|
assert(false);
|
||
|
}
|
||
|
catch (mf2)
|
||
|
{
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
void test_derived()
|
||
|
{
|
||
|
try
|
||
|
{
|
||
|
throw (mf1)0;
|
||
|
assert(false);
|
||
|
}
|
||
|
catch (dmf2)
|
||
|
{
|
||
|
assert(false);
|
||
|
}
|
||
|
catch (dmf1)
|
||
|
{
|
||
|
assert(false);
|
||
|
}
|
||
|
catch (mf1)
|
||
|
{
|
||
|
}
|
||
|
|
||
|
try
|
||
|
{
|
||
|
throw (mf2)0;
|
||
|
assert(false);
|
||
|
}
|
||
|
catch (dmf1)
|
||
|
{
|
||
|
assert(false);
|
||
|
}
|
||
|
catch (dmf2)
|
||
|
{
|
||
|
assert(false);
|
||
|
}
|
||
|
catch (mf2)
|
||
|
{
|
||
|
}
|
||
|
|
||
|
assert(!can_convert<mf1>((dmf1)0));
|
||
|
assert(!can_convert<mf2>((dmf1)0));
|
||
|
try
|
||
|
{
|
||
|
throw (dmf1)0;
|
||
|
assert(false);
|
||
|
}
|
||
|
catch (mf2)
|
||
|
{
|
||
|
assert(false);
|
||
|
}
|
||
|
catch (mf1)
|
||
|
{
|
||
|
assert(false);
|
||
|
}
|
||
|
catch (...)
|
||
|
{
|
||
|
}
|
||
|
|
||
|
assert(!can_convert<mf1>((dmf2)0));
|
||
|
assert(!can_convert<mf2>((dmf2)0));
|
||
|
try
|
||
|
{
|
||
|
throw (dmf2)0;
|
||
|
assert(false);
|
||
|
}
|
||
|
catch (mf2)
|
||
|
{
|
||
|
assert(false);
|
||
|
}
|
||
|
catch (mf1)
|
||
|
{
|
||
|
assert(false);
|
||
|
}
|
||
|
catch (...)
|
||
|
{
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void test_void()
|
||
|
{
|
||
|
assert(!can_convert<void*>(&A::foo));
|
||
|
try
|
||
|
{
|
||
|
throw &A::foo;
|
||
|
assert(false);
|
||
|
}
|
||
|
catch (void*)
|
||
|
{
|
||
|
assert(false);
|
||
|
}
|
||
|
catch(...)
|
||
|
{
|
||
|
}
|
||
|
}
|
||
|
|
||
|
int main()
|
||
|
{
|
||
|
test1();
|
||
|
test2();
|
||
|
test_derived();
|
||
|
test_void();
|
||
|
}
|