diff options
author | Eric Fiselier <eric@efcs.ca> | 2016-01-19 23:37:51 +0000 |
---|---|---|
committer | Eric Fiselier <eric@efcs.ca> | 2016-01-19 23:37:51 +0000 |
commit | ccd5d8d8019162e5b294aa788bc30c8a533a0e58 (patch) | |
tree | 9c4459af4f5fe1a9e7c7b3614754bf6f8b14924d | |
parent | 4cf56917ea2ffb22fca7989e75f1a9c6460a61f7 (diff) | |
download | bcm5719-llvm-ccd5d8d8019162e5b294aa788bc30c8a533a0e58.tar.gz bcm5719-llvm-ccd5d8d8019162e5b294aa788bc30c8a533a0e58.zip |
Recommit r256322: Fix PR25898 - Check for incomplete pointers types in can_catch(...)
This patch re-commits r256322 and r256323. They were reverted due to a OS X
test failure. The test failure has been fixed by libc++ commit r258217.
This patch also adds some additional tests.
llvm-svn: 258222
-rw-r--r-- | libcxxabi/src/private_typeinfo.cpp | 34 |
1 files changed, 17 insertions, 17 deletions
diff --git a/libcxxabi/src/private_typeinfo.cpp b/libcxxabi/src/private_typeinfo.cpp index beacd701d51..df03596e8b6 100644 --- a/libcxxabi/src/private_typeinfo.cpp +++ b/libcxxabi/src/private_typeinfo.cpp @@ -34,9 +34,12 @@ // // _LIBCXX_DYNAMIC_FALLBACK is currently off by default. + +#include <string.h> + + #ifdef _LIBCXX_DYNAMIC_FALLBACK #include "abort_message.h" -#include <string.h> #include <sys/syslog.h> #endif @@ -57,31 +60,19 @@ namespace __cxxabiv1 #pragma GCC visibility push(hidden) -#ifdef _LIBCXX_DYNAMIC_FALLBACK - inline bool is_equal(const std::type_info* x, const std::type_info* y, bool use_strcmp) { +#ifndef _WIN32 if (!use_strcmp) return x == y; return strcmp(x->name(), y->name()) == 0; -} - -#else // !_LIBCXX_DYNAMIC_FALLBACK - -inline -bool -is_equal(const std::type_info* x, const std::type_info* y, bool) -{ -#ifndef _WIN32 - return x == y; #else return (x == y) || (strcmp(x->name(), y->name()) == 0); -#endif +#endif } -#endif // _LIBCXX_DYNAMIC_FALLBACK // __shim_type_info @@ -351,8 +342,17 @@ bool __pbase_type_info::can_catch(const __shim_type_info* thrown_type, void*&) const { - return is_equal(this, thrown_type, false) || - is_equal(thrown_type, &typeid(std::nullptr_t), false); + if (is_equal(thrown_type, &typeid(std::nullptr_t), false)) return true; + bool use_strcmp = this->__flags & (__incomplete_class_mask | + __incomplete_mask); + if (!use_strcmp) { + const __pbase_type_info* thrown_pbase = dynamic_cast<const __pbase_type_info*>( + thrown_type); + if (!thrown_pbase) return false; + use_strcmp = thrown_pbase->__flags & (__incomplete_class_mask | + __incomplete_mask); + } + return is_equal(this, thrown_type, use_strcmp); } #ifdef __clang__ |