// Copyright (C) 1994-2019 Free Software Foundation, Inc. // // This file is part of GCC. // // GCC is free software; you can redistribute it and/or modify // it under the terms of the GNU General Public License as published by // the Free Software Foundation; either version 3, or (at your option) // any later version. // GCC is distributed in the hope that it will be useful, // but WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the // GNU General Public License for more details. // Under Section 7 of GPL version 3, you are granted additional // permissions described in the GCC Runtime Library Exception, version // 3.1, as published by the Free Software Foundation. // You should have received a copy of the GNU General Public License and // a copy of the GCC Runtime Library Exception along with this program; // see the files COPYING3 and COPYING.RUNTIME respectively. If not, see // . #include "tinfo.h" namespace __cxxabiv1 { __pbase_type_info:: ~__pbase_type_info () {} bool __pbase_type_info:: __do_catch (const type_info *thr_type, void **thr_obj, unsigned outer) const { if (*this == *thr_type) return true; // same type #if __cpp_rtti if (*thr_type == typeid (nullptr)) { // A catch handler for any pointer type matches nullptr_t. if (typeid (*this) == typeid(__pointer_type_info)) { *thr_obj = nullptr; return true; } else if (typeid (*this) == typeid(__pointer_to_member_type_info)) { if (__pointee->__is_function_p ()) { using pmf_type = void (__pbase_type_info::*)(); static const pmf_type pmf = nullptr; *thr_obj = const_cast(&pmf); return true; } else { using pm_type = int __pbase_type_info::*; static const pm_type pm = nullptr; *thr_obj = const_cast(&pm); return true; } } } if (typeid (*this) != typeid (*thr_type)) return false; // not both same kind of pointers #endif if (!(outer & 1)) // We're not the same and our outer pointers are not all const qualified // Therefore there must at least be a qualification conversion involved // But for that to be valid, our outer pointers must be const qualified. return false; const __pbase_type_info *thrown_type = static_cast (thr_type); unsigned tflags = thrown_type->__flags; const unsigned fqual_mask = __transaction_safe_mask|__noexcept_mask; unsigned throw_fqual = (tflags & fqual_mask); unsigned catch_fqual = (__flags & fqual_mask); if (throw_fqual & ~catch_fqual) /* Catch can perform a function pointer conversion. */ tflags &= catch_fqual; if (catch_fqual & ~throw_fqual) /* But not the reverse. */ return false; if (tflags & ~__flags) // We're less qualified. return false; if (!(__flags & __const_mask)) outer &= ~1; return __pointer_catch (thrown_type, thr_obj, outer); } }