blob: d722ea07b7a23472178cee814d01e8e98d7caf0c [file] [log] [blame]
Eric Fiselier0cb62d12015-04-02 23:26:37 +00001//===--------------------- catch_pointer_nullptr.cpp ----------------------===//
2//
3// The LLVM Compiler Infrastructure
4//
5// This file is dual licensed under the MIT and the University of Illinois Open
6// Source Licenses. See LICENSE.TXT for details.
7//
8//===----------------------------------------------------------------------===//
9
10#include <cassert>
11#include <cstdlib>
12#include <iostream>
13
14// Roll our own assertion macro to get better error messages out of the tests.
15// In particular on systems that don't use __PRETTY_FUNCTION__ in assertions.
16#define my_assert(pred, msg) do_assert(pred, msg, __LINE__, __PRETTY_FUNCTION__)
17
18void do_assert(bool assert_passed, const char* msg, int line, const char* func) {
19 if (assert_passed) return;
20 std::cerr << __FILE__ << ":" << line << " " << func
21 << ": Assertion Failed `" << msg << "'\n\n";
22 std::abort();
23}
24
25struct A {};
26struct Base {};
27struct Derived : public Base {};
28
29template <class To>
30bool test_conversion(To) { return true; }
31
32template <class To>
33bool test_conversion(...) { return false; }
34
35template <class Pointer>
36struct CreatePointer {
37 Pointer operator()() const {
38 return (Pointer)0;
39 }
40};
41
42template <class Tp>
43struct CreatePointer<Tp*> {
44 Tp* operator()() const {
45 return (Tp*)42;
46 }
47};
48
49template <class Throw, class Catch>
50void catch_pointer_test() {
51 Throw throw_ptr = CreatePointer<Throw>()();
52 // Use the compiler to determine if the exception of type Throw can be
53 // implicitly converted to type Catch.
54 const bool can_convert = test_conversion<Catch>(throw_ptr);
55 try {
56 throw throw_ptr;
57 assert(false);
58 } catch (Catch catch_ptr) {
59 Catch catch2 = CreatePointer<Catch>()();
60 my_assert(can_convert, "non-convertible type incorrectly caught");
61 my_assert(catch_ptr == catch2,
62 "Thrown pointer does not match caught ptr");
63 } catch (...) {
64 my_assert(!can_convert, "convertible type incorrectly not caught");
65 }
66}
67
68// Generate CV qualified pointer typedefs.
69template <class Tp, bool First = false>
70struct TestTypes {
71 typedef Tp* Type;
72 typedef Tp const* CType;
73 typedef Tp volatile* VType;
74 typedef Tp const volatile* CVType;
75};
76
77// Special case for cv-qualifying a pointer-to-member without adding an extra
78// pointer to it.
79template <class Member, class Class>
80struct TestTypes<Member Class::*, true> {
81 typedef Member (Class::*Type);
82 typedef const Member (Class::*CType);
83 typedef volatile Member (Class::*VType);
84 typedef const volatile Member (Class::*CVType);
85};
86
87template <class Throw, class Catch, int level, bool first = false>
88struct generate_tests_imp {
89 typedef TestTypes<Throw, first> ThrowTypes;
90 typedef TestTypes<Catch, first> CatchTypes;
91 void operator()() {
92 typedef typename ThrowTypes::Type Type;
93 typedef typename ThrowTypes::CType CType;
94 typedef typename ThrowTypes::VType VType;
95 typedef typename ThrowTypes::CVType CVType;
96
97 run_catch_tests<Type>();
98 run_catch_tests<CType>();
99 run_catch_tests<VType>();
100 run_catch_tests<CVType>();
101 }
102
103 template <class ThrowTp>
104 void run_catch_tests() {
105 typedef typename CatchTypes::Type Type;
106 typedef typename CatchTypes::CType CType;
107 typedef typename CatchTypes::VType VType;
108 typedef typename CatchTypes::CVType CVType;
109
110 catch_pointer_test<ThrowTp, Type>();
111 catch_pointer_test<ThrowTp, CType>();
112 catch_pointer_test<ThrowTp, VType>();
113 catch_pointer_test<ThrowTp, CVType>();
114
115 generate_tests_imp<ThrowTp, Type, level-1>()();
116 generate_tests_imp<ThrowTp, CType, level-1>()();
117 generate_tests_imp<ThrowTp, VType, level-1>()();
118 generate_tests_imp<ThrowTp, CVType, level-1>()();
119 }
120};
121
122template <class Throw, class Catch, bool first>
123struct generate_tests_imp<Throw, Catch, 0, first> {
124 void operator()() {
125 catch_pointer_test<Throw, Catch>();
126 }
127};
128
129template <class Throw, class Catch, int level>
130struct generate_tests : generate_tests_imp<Throw, Catch, level, true> {};
131
132int main()
133{
134 generate_tests<int, int, 3>()();
135 generate_tests<Base, Derived, 2>()();
136 generate_tests<Derived, Base, 2>()();
137 generate_tests<int, void, 2>()();
138 generate_tests<void, int, 2>()();
139
140 generate_tests<int A::*, int A::*, 3>()();
141 generate_tests<int A::*, void, 2>()();
142 generate_tests<void, int A::*, 2>()();
143 generate_tests<int Base::*, int Derived::*, 2>()();
144 generate_tests<int Derived::*, int Base::*, 2>()();
145}