
This is needed when running the tests in Freestanding mode, where main() isn't treated specially. In Freestanding, main() doesn't get mangled as extern "C", so whatever runtime we're using fails to find the entry point. One way to solve this problem is to define a symbol alias from __Z4mainiPPc to _main, however this requires all definitions of main() to have the same mangling. Hence this commit.
173 lines
2.5 KiB
C++
173 lines
2.5 KiB
C++
//===--------------- catch_member_function_pointer_01.cpp -----------------===//
|
|
//
|
|
// 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
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// 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: 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(int, char**)
|
|
{
|
|
test1();
|
|
test2();
|
|
test_derived();
|
|
test_void();
|
|
|
|
return 0;
|
|
}
|