Home | History | Annotate | Download | only in test
      1 //===--------------------- 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 
     18 void 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 
     25 struct A {};
     26 struct Base {};
     27 struct Derived : public Base {};
     28 
     29 template <class To>
     30 bool test_conversion(To) { return true; }
     31 
     32 template <class To>
     33 bool test_conversion(...) { return false; }
     34 
     35 template <class Pointer>
     36 struct CreatePointer {
     37   Pointer operator()() const {
     38       return (Pointer)0;
     39   }
     40 };
     41 
     42 template <class Tp>
     43 struct CreatePointer<Tp*> {
     44   Tp* operator()() const {
     45       return (Tp*)42;
     46   }
     47 };
     48 
     49 template <class Throw, class Catch>
     50 void 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.
     69 template <class Tp, bool First = false>
     70 struct 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.
     79 template <class Member, class Class>
     80 struct 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 
     87 template <class Throw, class Catch, int level, bool first = false>
     88 struct 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 
    122 template <class Throw, class Catch, bool first>
    123 struct generate_tests_imp<Throw, Catch, 0, first> {
    124   void operator()() {
    125       catch_pointer_test<Throw, Catch>();
    126   }
    127 };
    128 
    129 template <class Throw, class Catch, int level>
    130 struct generate_tests : generate_tests_imp<Throw, Catch, level, true> {};
    131 
    132 int 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 }
    144