2010-05-12 03:42:16 +08:00
|
|
|
// -*- C++ -*-
|
2021-11-18 05:25:01 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
2010-05-12 03:42:16 +08:00
|
|
|
//
|
2019-01-19 18:56:40 +08:00
|
|
|
// 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
|
2010-05-12 03:42:16 +08:00
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#ifndef __LIBCPP_TYPEINFO
|
|
|
|
#define __LIBCPP_TYPEINFO
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
|
|
|
typeinfo synopsis
|
|
|
|
|
|
|
|
namespace std {
|
|
|
|
|
|
|
|
class type_info
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
virtual ~type_info();
|
2010-08-22 08:02:43 +08:00
|
|
|
|
2011-05-27 02:23:59 +08:00
|
|
|
bool operator==(const type_info& rhs) const noexcept;
|
2022-08-06 21:09:50 +08:00
|
|
|
bool operator!=(const type_info& rhs) const noexcept; // removed in C++20
|
2010-08-22 08:02:43 +08:00
|
|
|
|
2011-05-27 02:23:59 +08:00
|
|
|
bool before(const type_info& rhs) const noexcept;
|
|
|
|
size_t hash_code() const noexcept;
|
|
|
|
const char* name() const noexcept;
|
2010-08-22 08:02:43 +08:00
|
|
|
|
2010-05-12 03:42:16 +08:00
|
|
|
type_info(const type_info& rhs) = delete;
|
|
|
|
type_info& operator=(const type_info& rhs) = delete;
|
|
|
|
};
|
|
|
|
|
|
|
|
class bad_cast
|
|
|
|
: public exception
|
|
|
|
{
|
|
|
|
public:
|
2011-05-27 02:23:59 +08:00
|
|
|
bad_cast() noexcept;
|
|
|
|
bad_cast(const bad_cast&) noexcept;
|
|
|
|
bad_cast& operator=(const bad_cast&) noexcept;
|
|
|
|
virtual const char* what() const noexcept;
|
2010-05-12 03:42:16 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
class bad_typeid
|
|
|
|
: public exception
|
|
|
|
{
|
|
|
|
public:
|
2011-05-27 02:23:59 +08:00
|
|
|
bad_typeid() noexcept;
|
|
|
|
bad_typeid(const bad_typeid&) noexcept;
|
|
|
|
bad_typeid& operator=(const bad_typeid&) noexcept;
|
|
|
|
virtual const char* what() const noexcept;
|
2010-05-12 03:42:16 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
} // std
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
2022-03-26 00:55:36 +08:00
|
|
|
#include <__assert> // all public C++ headers provide the assertion handler
|
2020-11-05 04:01:25 +08:00
|
|
|
#include <__availability>
|
2021-05-19 23:57:04 +08:00
|
|
|
#include <__config>
|
2010-05-12 03:42:16 +08:00
|
|
|
#include <cstddef>
|
2014-03-30 19:34:26 +08:00
|
|
|
#include <cstdint>
|
2021-05-19 23:57:04 +08:00
|
|
|
#include <exception>
|
2020-02-14 23:38:16 +08:00
|
|
|
#include <type_traits>
|
2021-05-19 23:57:04 +08:00
|
|
|
|
2016-09-07 05:25:27 +08:00
|
|
|
#ifdef _LIBCPP_NO_EXCEPTIONS
|
|
|
|
#include <cstdlib>
|
|
|
|
#endif
|
2010-05-12 03:42:16 +08:00
|
|
|
|
2011-10-18 04:05:10 +08:00
|
|
|
#if !defined(_LIBCPP_HAS_NO_PRAGMA_SYSTEM_HEADER)
|
2022-02-02 09:16:40 +08:00
|
|
|
# pragma GCC system_header
|
2011-10-18 04:05:10 +08:00
|
|
|
#endif
|
2010-05-12 03:42:16 +08:00
|
|
|
|
2019-03-05 09:57:01 +08:00
|
|
|
#if defined(_LIBCPP_ABI_VCRUNTIME)
|
2018-01-26 09:22:17 +08:00
|
|
|
#include <vcruntime_typeinfo.h>
|
|
|
|
#else
|
|
|
|
|
2010-05-12 03:42:16 +08:00
|
|
|
namespace std // purposefully not using versioning namespace
|
|
|
|
{
|
|
|
|
|
2019-05-29 10:21:37 +08:00
|
|
|
|
2019-02-02 04:00:13 +08:00
|
|
|
#if defined(_LIBCPP_ABI_MICROSOFT)
|
|
|
|
|
2010-08-22 08:02:43 +08:00
|
|
|
class _LIBCPP_EXCEPTION_ABI type_info
|
2010-05-12 03:42:16 +08:00
|
|
|
{
|
|
|
|
type_info& operator=(const type_info&);
|
|
|
|
type_info(const type_info&);
|
2016-08-30 04:43:38 +08:00
|
|
|
|
2017-09-15 13:42:39 +08:00
|
|
|
mutable struct {
|
|
|
|
const char *__undecorated_name;
|
|
|
|
const char __decorated_name[1];
|
|
|
|
} __data;
|
|
|
|
|
2017-09-15 14:19:31 +08:00
|
|
|
int __compare(const type_info &__rhs) const _NOEXCEPT;
|
2010-05-12 03:42:16 +08:00
|
|
|
|
|
|
|
public:
|
2017-05-05 01:08:54 +08:00
|
|
|
_LIBCPP_AVAILABILITY_TYPEINFO_VTABLE
|
2010-05-12 03:42:16 +08:00
|
|
|
virtual ~type_info();
|
|
|
|
|
2017-09-15 13:42:39 +08:00
|
|
|
const char *name() const _NOEXCEPT;
|
|
|
|
|
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
|
|
|
bool before(const type_info& __arg) const _NOEXCEPT {
|
|
|
|
return __compare(__arg) < 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
size_t hash_code() const _NOEXCEPT;
|
|
|
|
|
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
|
|
|
bool operator==(const type_info& __arg) const _NOEXCEPT {
|
|
|
|
return __compare(__arg) == 0;
|
|
|
|
}
|
2019-02-02 04:00:13 +08:00
|
|
|
|
2022-08-06 21:09:50 +08:00
|
|
|
#if _LIBCPP_STD_VER <= 17
|
2019-02-02 04:00:13 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
|
|
|
bool operator!=(const type_info& __arg) const _NOEXCEPT
|
|
|
|
{ return !operator==(__arg); }
|
2022-08-06 21:09:50 +08:00
|
|
|
#endif
|
2019-02-02 04:00:13 +08:00
|
|
|
};
|
|
|
|
|
2019-05-29 10:21:37 +08:00
|
|
|
#else // !defined(_LIBCPP_ABI_MICROSOFT)
|
2019-02-02 04:00:13 +08:00
|
|
|
|
2019-05-29 10:21:37 +08:00
|
|
|
// ========================================================================== //
|
|
|
|
// Implementations
|
|
|
|
// ========================================================================== //
|
|
|
|
// ------------------------------------------------------------------------- //
|
|
|
|
// Unique
|
2020-05-16 03:58:19 +08:00
|
|
|
// (_LIBCPP_TYPEINFO_COMPARISON_IMPLEMENTATION = 1)
|
2019-05-29 10:21:37 +08:00
|
|
|
// ------------------------------------------------------------------------- //
|
|
|
|
// This implementation of type_info assumes a unique copy of the RTTI for a
|
2021-03-03 05:18:37 +08:00
|
|
|
// given type inside a program. This is a valid assumption when abiding to the
|
2019-05-29 10:21:37 +08:00
|
|
|
// Itanium ABI (http://itanium-cxx-abi.github.io/cxx-abi/abi.html#vtable-components).
|
|
|
|
// Under this assumption, we can always compare the addresses of the type names
|
|
|
|
// to implement equality-comparison of type_infos instead of having to perform
|
|
|
|
// a deep string comparison.
|
|
|
|
// -------------------------------------------------------------------------- //
|
|
|
|
// NonUnique
|
2020-05-16 03:58:19 +08:00
|
|
|
// (_LIBCPP_TYPEINFO_COMPARISON_IMPLEMENTATION = 2)
|
2019-05-29 10:21:37 +08:00
|
|
|
// -------------------------------------------------------------------------- //
|
|
|
|
// This implementation of type_info does not assume there is always a unique
|
|
|
|
// copy of the RTTI for a given type inside a program. For various reasons
|
|
|
|
// the linker may have failed to merge every copy of a types RTTI
|
|
|
|
// (For example: -Bsymbolic or llvm.org/PR37398). Under this assumption, two
|
|
|
|
// type_infos are equal if their addresses are equal or if a deep string
|
|
|
|
// comparison is equal.
|
|
|
|
// -------------------------------------------------------------------------- //
|
|
|
|
// NonUniqueARMRTTIBit
|
2020-11-17 07:13:43 +08:00
|
|
|
// (_LIBCPP_TYPEINFO_COMPARISON_IMPLEMENTATION = 3)
|
2019-05-29 10:21:37 +08:00
|
|
|
// -------------------------------------------------------------------------- //
|
2021-03-03 05:18:37 +08:00
|
|
|
// This implementation is specific to ARM64 on Apple platforms.
|
|
|
|
//
|
2019-02-02 04:00:13 +08:00
|
|
|
// This implementation of type_info does not assume always a unique copy of
|
2021-03-03 05:18:37 +08:00
|
|
|
// the RTTI for a given type inside a program. When constructing the type_info,
|
|
|
|
// the compiler packs the pointer to the type name into a uintptr_t and reserves
|
|
|
|
// the high bit of that pointer, which is assumed to be free for use under that
|
|
|
|
// ABI. If that high bit is set, that specific copy of the RTTI can't be assumed
|
|
|
|
// to be unique within the program. If the high bit is unset, then the RTTI can
|
|
|
|
// be assumed to be unique within the program.
|
2019-02-02 04:00:13 +08:00
|
|
|
//
|
2021-03-03 05:18:37 +08:00
|
|
|
// When comparing type_infos, if both RTTIs can be assumed to be unique, it
|
|
|
|
// suffices to compare their addresses. If both the RTTIs can't be assumed to
|
|
|
|
// be unique, we must perform a deep string comparison of the type names.
|
|
|
|
// However, if one of the RTTIs is guaranteed unique and the other one isn't,
|
|
|
|
// then both RTTIs are necessarily not to be considered equal.
|
2019-05-29 10:21:37 +08:00
|
|
|
//
|
2021-03-03 05:18:37 +08:00
|
|
|
// The intent of this design is to remove the need for weak symbols. Specifically,
|
|
|
|
// if a type would normally have a default-visibility RTTI emitted as a weak
|
|
|
|
// symbol, it is given hidden visibility instead and the non-unique bit is set.
|
|
|
|
// Otherwise, types declared with hidden visibility are always considered to have
|
|
|
|
// a unique RTTI: the RTTI is emitted with linkonce_odr linkage and is assumed
|
|
|
|
// to be deduplicated by the linker within the linked image. Across linked image
|
|
|
|
// boundaries, such types are thus considered different types.
|
2019-02-02 04:00:13 +08:00
|
|
|
|
2020-11-17 07:13:43 +08:00
|
|
|
// This value can be overriden in the __config_site. When it's not overriden,
|
|
|
|
// we pick a default implementation based on the platform here.
|
|
|
|
#ifndef _LIBCPP_TYPEINFO_COMPARISON_IMPLEMENTATION
|
|
|
|
|
2022-07-29 01:17:12 +08:00
|
|
|
// Windows and AIX binaries can't merge typeinfos, so use the NonUnique implementation.
|
|
|
|
# if defined(_LIBCPP_OBJECT_FORMAT_COFF) || defined(_LIBCPP_OBJECT_FORMAT_XCOFF)
|
2020-11-17 07:13:43 +08:00
|
|
|
# define _LIBCPP_TYPEINFO_COMPARISON_IMPLEMENTATION 2
|
|
|
|
|
|
|
|
// On arm64 on Apple platforms, use the special NonUniqueARMRTTIBit implementation.
|
|
|
|
# elif defined(__APPLE__) && defined(__LP64__) && !defined(__x86_64__)
|
|
|
|
# define _LIBCPP_TYPEINFO_COMPARISON_IMPLEMENTATION 3
|
|
|
|
|
|
|
|
// On all other platforms, assume the Itanium C++ ABI and use the Unique implementation.
|
|
|
|
# else
|
|
|
|
# define _LIBCPP_TYPEINFO_COMPARISON_IMPLEMENTATION 1
|
|
|
|
# endif
|
|
|
|
#endif
|
|
|
|
|
2019-05-29 10:21:37 +08:00
|
|
|
struct __type_info_implementations {
|
|
|
|
struct __string_impl_base {
|
|
|
|
typedef const char* __type_name_t;
|
|
|
|
_LIBCPP_INLINE_VISIBILITY _LIBCPP_ALWAYS_INLINE
|
|
|
|
_LIBCPP_CONSTEXPR static const char* __type_name_to_string(__type_name_t __v) _NOEXCEPT {
|
|
|
|
return __v;
|
2017-01-06 05:22:22 +08:00
|
|
|
}
|
2019-05-29 10:21:37 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY _LIBCPP_ALWAYS_INLINE
|
|
|
|
_LIBCPP_CONSTEXPR static __type_name_t __string_to_type_name(const char* __v) _NOEXCEPT {
|
|
|
|
return __v;
|
2017-01-06 05:22:22 +08:00
|
|
|
}
|
2019-05-29 10:21:37 +08:00
|
|
|
};
|
2014-03-30 19:34:26 +08:00
|
|
|
|
2019-05-29 10:21:37 +08:00
|
|
|
struct __unique_impl : __string_impl_base {
|
|
|
|
_LIBCPP_INLINE_VISIBILITY _LIBCPP_ALWAYS_INLINE
|
|
|
|
static size_t __hash(__type_name_t __v) _NOEXCEPT {
|
|
|
|
return reinterpret_cast<size_t>(__v);
|
|
|
|
}
|
|
|
|
_LIBCPP_INLINE_VISIBILITY _LIBCPP_ALWAYS_INLINE
|
|
|
|
static bool __eq(__type_name_t __lhs, __type_name_t __rhs) _NOEXCEPT {
|
|
|
|
return __lhs == __rhs;
|
|
|
|
}
|
|
|
|
_LIBCPP_INLINE_VISIBILITY _LIBCPP_ALWAYS_INLINE
|
|
|
|
static bool __lt(__type_name_t __lhs, __type_name_t __rhs) _NOEXCEPT {
|
|
|
|
return __lhs < __rhs;
|
|
|
|
}
|
|
|
|
};
|
2017-01-06 05:22:22 +08:00
|
|
|
|
2019-05-29 10:21:37 +08:00
|
|
|
struct __non_unique_impl : __string_impl_base {
|
|
|
|
_LIBCPP_INLINE_VISIBILITY _LIBCPP_ALWAYS_INLINE
|
|
|
|
static size_t __hash(__type_name_t __ptr) _NOEXCEPT {
|
2017-01-06 05:22:22 +08:00
|
|
|
size_t __hash = 5381;
|
|
|
|
while (unsigned char __c = static_cast<unsigned char>(*__ptr++))
|
|
|
|
__hash = (__hash * 33) ^ __c;
|
|
|
|
return __hash;
|
|
|
|
}
|
2019-05-29 10:21:37 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY _LIBCPP_ALWAYS_INLINE
|
|
|
|
static bool __eq(__type_name_t __lhs, __type_name_t __rhs) _NOEXCEPT {
|
|
|
|
return __lhs == __rhs || __builtin_strcmp(__lhs, __rhs) == 0;
|
|
|
|
}
|
|
|
|
_LIBCPP_INLINE_VISIBILITY _LIBCPP_ALWAYS_INLINE
|
|
|
|
static bool __lt(__type_name_t __lhs, __type_name_t __rhs) _NOEXCEPT {
|
|
|
|
return __builtin_strcmp(__lhs, __rhs) < 0;
|
|
|
|
}
|
|
|
|
};
|
2010-05-12 03:42:16 +08:00
|
|
|
|
2019-05-29 10:21:37 +08:00
|
|
|
struct __non_unique_arm_rtti_bit_impl {
|
|
|
|
typedef uintptr_t __type_name_t;
|
|
|
|
|
|
|
|
_LIBCPP_INLINE_VISIBILITY _LIBCPP_ALWAYS_INLINE
|
|
|
|
static const char* __type_name_to_string(__type_name_t __v) _NOEXCEPT {
|
|
|
|
return reinterpret_cast<const char*>(__v &
|
|
|
|
~__non_unique_rtti_bit::value);
|
|
|
|
}
|
|
|
|
_LIBCPP_INLINE_VISIBILITY _LIBCPP_ALWAYS_INLINE
|
|
|
|
static __type_name_t __string_to_type_name(const char* __v) _NOEXCEPT {
|
|
|
|
return reinterpret_cast<__type_name_t>(__v);
|
|
|
|
}
|
2017-01-06 05:22:22 +08:00
|
|
|
|
2019-05-29 10:21:37 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY _LIBCPP_ALWAYS_INLINE
|
|
|
|
static size_t __hash(__type_name_t __v) _NOEXCEPT {
|
|
|
|
if (__is_type_name_unique(__v))
|
2021-06-28 21:53:28 +08:00
|
|
|
return __v;
|
2019-05-29 10:21:37 +08:00
|
|
|
return __non_unique_impl::__hash(__type_name_to_string(__v));
|
|
|
|
}
|
|
|
|
_LIBCPP_INLINE_VISIBILITY _LIBCPP_ALWAYS_INLINE
|
|
|
|
static bool __eq(__type_name_t __lhs, __type_name_t __rhs) _NOEXCEPT {
|
|
|
|
if (__lhs == __rhs)
|
|
|
|
return true;
|
2021-03-03 05:18:37 +08:00
|
|
|
if (__is_type_name_unique(__lhs) || __is_type_name_unique(__rhs))
|
|
|
|
// Either both are unique and have a different address, or one of them
|
|
|
|
// is unique and the other one isn't. In both cases they are unequal.
|
2017-01-06 05:22:22 +08:00
|
|
|
return false;
|
2019-05-29 10:21:37 +08:00
|
|
|
return __builtin_strcmp(__type_name_to_string(__lhs), __type_name_to_string(__rhs)) == 0;
|
2017-01-06 05:22:22 +08:00
|
|
|
}
|
2019-05-29 10:21:37 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY _LIBCPP_ALWAYS_INLINE
|
|
|
|
static bool __lt(__type_name_t __lhs, __type_name_t __rhs) _NOEXCEPT {
|
2021-03-03 05:18:37 +08:00
|
|
|
if (__is_type_name_unique(__lhs) || __is_type_name_unique(__rhs))
|
2019-05-29 10:21:37 +08:00
|
|
|
return __lhs < __rhs;
|
|
|
|
return __builtin_strcmp(__type_name_to_string(__lhs), __type_name_to_string(__rhs)) < 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
2019-05-29 10:38:19 +08:00
|
|
|
// The unique bit is the top bit. It is expected that __type_name_t is 64 bits when
|
|
|
|
// this implementation is actually used.
|
[libc++] Consistently replace `std::` qualification with `_VSTD::` or nothing. NFCI.
I used a lot of `git grep` to find places where `std::` was being used
outside of comments and assert-messages. There were three outcomes:
- Qualified function calls, e.g. `std::move` becomes `_VSTD::move`.
This is the most common case.
- Typenames that don't need qualification, e.g. `std::allocator` becomes `allocator`.
Leaving these as `_VSTD::allocator` would also be fine, but I decided
that removing the qualification is more consistent with existing practice.
- Names that specifically need un-versioned `std::` qualification,
or that I wasn't sure about. For example, I didn't touch any code in
<atomic>, <math.h>, <new>, or any ext/ or experimental/ headers;
and I didn't touch any instances of `std::type_info`.
In some deduction guides, we were accidentally using `class Alloc = typename std::allocator<T>`,
despite `std::allocator<T>`'s type-ness not being template-dependent.
Because `std::allocator` is a qualified name, this did parse as we intended;
but what we meant was simply `class Alloc = allocator<T>`.
Differential Revision: https://reviews.llvm.org/D92250
2020-11-28 00:02:06 +08:00
|
|
|
typedef integral_constant<__type_name_t,
|
2019-05-29 10:38:19 +08:00
|
|
|
(1ULL << ((__CHAR_BIT__ * sizeof(__type_name_t)) - 1))> __non_unique_rtti_bit;
|
2019-02-02 04:00:13 +08:00
|
|
|
|
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2019-05-29 10:21:37 +08:00
|
|
|
static bool __is_type_name_unique(__type_name_t __lhs) _NOEXCEPT {
|
|
|
|
return !(__lhs & __non_unique_rtti_bit::value);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
typedef
|
2020-11-17 07:13:43 +08:00
|
|
|
#if _LIBCPP_TYPEINFO_COMPARISON_IMPLEMENTATION == 1
|
2019-05-29 10:21:37 +08:00
|
|
|
__unique_impl
|
2020-05-16 03:58:19 +08:00
|
|
|
#elif _LIBCPP_TYPEINFO_COMPARISON_IMPLEMENTATION == 2
|
|
|
|
__non_unique_impl
|
2020-11-17 07:13:43 +08:00
|
|
|
#elif _LIBCPP_TYPEINFO_COMPARISON_IMPLEMENTATION == 3
|
|
|
|
__non_unique_arm_rtti_bit_impl
|
2019-05-29 10:21:37 +08:00
|
|
|
#else
|
2020-05-16 03:58:19 +08:00
|
|
|
# error invalid configuration for _LIBCPP_TYPEINFO_COMPARISON_IMPLEMENTATION
|
2019-05-29 10:21:37 +08:00
|
|
|
#endif
|
|
|
|
__impl;
|
2019-02-02 04:00:13 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
class _LIBCPP_EXCEPTION_ABI type_info
|
|
|
|
{
|
2019-05-29 10:21:37 +08:00
|
|
|
type_info& operator=(const type_info&);
|
|
|
|
type_info(const type_info&);
|
|
|
|
|
|
|
|
protected:
|
|
|
|
typedef __type_info_implementations::__impl __impl;
|
2019-02-02 04:00:13 +08:00
|
|
|
|
2019-05-29 10:21:37 +08:00
|
|
|
__impl::__type_name_t __type_name;
|
2019-02-02 04:00:13 +08:00
|
|
|
|
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2019-05-29 10:21:37 +08:00
|
|
|
explicit type_info(const char* __n)
|
|
|
|
: __type_name(__impl::__string_to_type_name(__n)) {}
|
2019-02-02 04:00:13 +08:00
|
|
|
|
|
|
|
public:
|
|
|
|
_LIBCPP_AVAILABILITY_TYPEINFO_VTABLE
|
|
|
|
virtual ~type_info();
|
|
|
|
|
2010-09-24 02:58:28 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2017-01-06 05:22:22 +08:00
|
|
|
const char* name() const _NOEXCEPT
|
2019-05-29 10:21:37 +08:00
|
|
|
{
|
|
|
|
return __impl::__type_name_to_string(__type_name);
|
|
|
|
}
|
2010-05-12 03:42:16 +08:00
|
|
|
|
2014-03-30 19:34:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2017-01-06 05:22:22 +08:00
|
|
|
bool before(const type_info& __arg) const _NOEXCEPT
|
2019-05-29 10:21:37 +08:00
|
|
|
{
|
|
|
|
return __impl::__lt(__type_name, __arg.__type_name);
|
|
|
|
}
|
2017-01-06 05:22:22 +08:00
|
|
|
|
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
|
|
|
size_t hash_code() const _NOEXCEPT
|
2019-05-29 10:21:37 +08:00
|
|
|
{
|
|
|
|
return __impl::__hash(__type_name);
|
|
|
|
}
|
2017-01-06 05:22:22 +08:00
|
|
|
|
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
|
|
|
bool operator==(const type_info& __arg) const _NOEXCEPT
|
2019-05-29 10:21:37 +08:00
|
|
|
{
|
|
|
|
return __impl::__eq(__type_name, __arg.__type_name);
|
|
|
|
}
|
2017-01-06 05:22:22 +08:00
|
|
|
|
2022-08-06 21:09:50 +08:00
|
|
|
#if _LIBCPP_STD_VER <= 17
|
2017-01-06 05:22:22 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
|
|
|
bool operator!=(const type_info& __arg) const _NOEXCEPT
|
|
|
|
{ return !operator==(__arg); }
|
2022-08-06 21:09:50 +08:00
|
|
|
#endif
|
2010-05-12 03:42:16 +08:00
|
|
|
};
|
2019-05-29 10:21:37 +08:00
|
|
|
#endif // defined(_LIBCPP_ABI_MICROSOFT)
|
2019-02-02 04:00:13 +08:00
|
|
|
|
2010-05-15 04:19:37 +08:00
|
|
|
class _LIBCPP_EXCEPTION_ABI bad_cast
|
2010-08-22 08:02:43 +08:00
|
|
|
: public exception
|
2010-05-12 03:42:16 +08:00
|
|
|
{
|
2019-05-29 10:21:37 +08:00
|
|
|
public:
|
|
|
|
bad_cast() _NOEXCEPT;
|
2020-03-14 02:36:26 +08:00
|
|
|
bad_cast(const bad_cast&) _NOEXCEPT = default;
|
2022-08-24 08:14:29 +08:00
|
|
|
~bad_cast() _NOEXCEPT override;
|
|
|
|
const char* what() const _NOEXCEPT override;
|
2010-05-12 03:42:16 +08:00
|
|
|
};
|
|
|
|
|
2010-05-15 04:19:37 +08:00
|
|
|
class _LIBCPP_EXCEPTION_ABI bad_typeid
|
2010-08-22 08:02:43 +08:00
|
|
|
: public exception
|
2010-05-12 03:42:16 +08:00
|
|
|
{
|
2019-05-29 10:21:37 +08:00
|
|
|
public:
|
|
|
|
bad_typeid() _NOEXCEPT;
|
2022-08-24 08:14:29 +08:00
|
|
|
~bad_typeid() _NOEXCEPT override;
|
|
|
|
const char* what() const _NOEXCEPT override;
|
2010-05-12 03:42:16 +08:00
|
|
|
};
|
|
|
|
|
2021-12-02 21:12:51 +08:00
|
|
|
} // namespace std
|
2010-05-12 03:42:16 +08:00
|
|
|
|
2019-03-05 09:57:01 +08:00
|
|
|
#endif // defined(_LIBCPP_ABI_VCRUNTIME)
|
2018-01-26 09:22:17 +08:00
|
|
|
|
[libcxx] Fix using the vcruntime ABI with _HAS_EXCEPTIONS=0 defined
_HAS_EXCEPTIONS=0 allows disabling the exception parts of the MS STL
and vcruntime, and e.g. compiler-rt/lib/fuzzer sets this define (to
work around issues with MS STL). If using libc++ instead of MS STL,
this define previously broke the libc++ headers.
If _HAS_EXCEPTIONS is set to 0, the vcruntime_exception.h header
doesn't define the ABI base class std::exception. If no exceptions
are going to be thrown, this probably is fine (although it also
breaks using subclasses of it as regular objects that aren't thrown),
but it requires ifdeffing out all subclasses of all exception/error
derived objects (which are sprinkled throughout the headers).
Instead, libc++ will supply an ABI compatible definition when
_HAS_EXCEPTIONS is set to 0, which will make the class hierarchies
complete.
In this build configuration, one can still create instances of
exception subclasses, and those objects will be ABI incompatible
with the ones from when _HAS_EXCEPTIONS isn't defined to 0 - but
one may argue that's a pathological/self-imposed problem in that case.
Reviewed By: #libc, ldionne
Differential Revision: https://reviews.llvm.org/D103947
2022-08-18 04:57:59 +08:00
|
|
|
#if defined(_LIBCPP_ABI_VCRUNTIME) && _HAS_EXCEPTIONS == 0
|
|
|
|
|
|
|
|
namespace std {
|
|
|
|
|
|
|
|
class bad_cast : public exception {
|
|
|
|
public:
|
|
|
|
bad_cast() _NOEXCEPT : exception("bad cast") {}
|
|
|
|
|
|
|
|
private:
|
|
|
|
bad_cast(const char* const __message) _NOEXCEPT : exception(__message) {}
|
|
|
|
};
|
|
|
|
|
|
|
|
class bad_typeid : public exception {
|
|
|
|
public:
|
|
|
|
bad_typeid() _NOEXCEPT : exception("bad typeid") {}
|
|
|
|
|
|
|
|
private:
|
|
|
|
bad_typeid(const char* const __message) _NOEXCEPT : exception(__message) {}
|
|
|
|
};
|
|
|
|
|
|
|
|
} // namespace std
|
|
|
|
|
|
|
|
#endif // defined(_LIBCPP_ABI_VCRUNTIME) && _HAS_EXCEPTIONS == 0
|
|
|
|
|
2016-08-25 23:09:01 +08:00
|
|
|
_LIBCPP_BEGIN_NAMESPACE_STD
|
2018-07-12 07:14:33 +08:00
|
|
|
_LIBCPP_NORETURN inline _LIBCPP_INLINE_VISIBILITY
|
2016-08-25 23:09:01 +08:00
|
|
|
void __throw_bad_cast()
|
|
|
|
{
|
|
|
|
#ifndef _LIBCPP_NO_EXCEPTIONS
|
|
|
|
throw bad_cast();
|
|
|
|
#else
|
2018-08-04 06:36:53 +08:00
|
|
|
_VSTD::abort();
|
2016-08-25 23:09:01 +08:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
_LIBCPP_END_NAMESPACE_STD
|
|
|
|
|
2021-04-21 00:03:32 +08:00
|
|
|
#endif // __LIBCPP_TYPEINFO
|