123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190 |
- //===----------------------------------------------------------------------===//
- //
- // 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
- //
- //
- // This file implements the "Exception Handling APIs"
- // https://itanium-cxx-abi.github.io/cxx-abi/abi-eh.html
- //
- //===----------------------------------------------------------------------===//
- #ifndef _CXA_EXCEPTION_H
- #define _CXA_EXCEPTION_H
- #include <exception> // for std::unexpected_handler and std::terminate_handler
- #include "cxxabi.h"
- #include "unwind.h"
- namespace __cxxabiv1 {
- #ifdef __USING_EMSCRIPTEN_EXCEPTIONS__
- struct _LIBCXXABI_HIDDEN __cxa_exception {
- size_t referenceCount;
- std::type_info *exceptionType;
- void (*exceptionDestructor)(void *);
- uint8_t caught;
- uint8_t rethrown;
- void *adjustedPtr;
- // Add padding to ensure that the size of __cxa_exception is a multiple of
- // the maximum useful alignment for the target machine. This ensures that
- // the thrown object that follows has that correct alignment.
- void *padding;
- };
- static_assert(sizeof(__cxa_exception) % alignof(max_align_t) == 0, "__cxa_exception must have a size that is multipl of max alignment");
- #else
- static const uint64_t kOurExceptionClass = 0x434C4E47432B2B00; // CLNGC++\0
- static const uint64_t kOurDependentExceptionClass = 0x434C4E47432B2B01; // CLNGC++\1
- static const uint64_t get_vendor_and_language = 0xFFFFFFFFFFFFFF00; // mask for CLNGC++
- _LIBCXXABI_HIDDEN uint64_t __getExceptionClass (const _Unwind_Exception*);
- _LIBCXXABI_HIDDEN void __setExceptionClass ( _Unwind_Exception*, uint64_t);
- _LIBCXXABI_HIDDEN bool __isOurExceptionClass(const _Unwind_Exception*);
- struct _LIBCXXABI_HIDDEN __cxa_exception {
- #if defined(__LP64__) || defined(_WIN64) || defined(_LIBCXXABI_ARM_EHABI)
- // Now _Unwind_Exception is marked with __attribute__((aligned)),
- // which implies __cxa_exception is also aligned. Insert padding
- // in the beginning of the struct, rather than before unwindHeader.
- void *reserve;
- // This is a new field to support C++ 0x exception_ptr.
- // For binary compatibility it is at the start of this
- // struct which is prepended to the object thrown in
- // __cxa_allocate_exception.
- size_t referenceCount;
- #endif
- // Manage the exception object itself.
- std::type_info *exceptionType;
- #ifdef __USING_WASM_EXCEPTIONS__
- // In wasm, destructors return their argument
- void *(*exceptionDestructor)(void *);
- #else
- void (*exceptionDestructor)(void *);
- #endif
- std::unexpected_handler unexpectedHandler;
- std::terminate_handler terminateHandler;
- __cxa_exception *nextException;
- int handlerCount;
- #if defined(_LIBCXXABI_ARM_EHABI)
- __cxa_exception* nextPropagatingException;
- int propagationCount;
- #else
- int handlerSwitchValue;
- const unsigned char *actionRecord;
- const unsigned char *languageSpecificData;
- void *catchTemp;
- void *adjustedPtr;
- #endif
- #if !defined(__LP64__) && !defined(_WIN64) && !defined(_LIBCXXABI_ARM_EHABI)
- // This is a new field to support C++ 0x exception_ptr.
- // For binary compatibility it is placed where the compiler
- // previously adding padded to 64-bit align unwindHeader.
- size_t referenceCount;
- #endif
- _Unwind_Exception unwindHeader;
- };
- // http://sourcery.mentor.com/archives/cxx-abi-dev/msg01924.html
- // The layout of this structure MUST match the layout of __cxa_exception, with
- // primaryException instead of referenceCount.
- struct _LIBCXXABI_HIDDEN __cxa_dependent_exception {
- #if defined(__LP64__) || defined(_WIN64) || defined(_LIBCXXABI_ARM_EHABI)
- void* reserve; // padding.
- void* primaryException;
- #endif
- std::type_info *exceptionType;
- void (*exceptionDestructor)(void *);
- std::unexpected_handler unexpectedHandler;
- std::terminate_handler terminateHandler;
- __cxa_exception *nextException;
- int handlerCount;
- #if defined(_LIBCXXABI_ARM_EHABI)
- __cxa_exception* nextPropagatingException;
- int propagationCount;
- #else
- int handlerSwitchValue;
- const unsigned char *actionRecord;
- const unsigned char *languageSpecificData;
- void * catchTemp;
- void *adjustedPtr;
- #endif
- #if !defined(__LP64__) && !defined(_WIN64) && !defined(_LIBCXXABI_ARM_EHABI)
- void* primaryException;
- #endif
- _Unwind_Exception unwindHeader;
- };
- // Verify the negative offsets of different fields.
- static_assert(sizeof(_Unwind_Exception) +
- offsetof(__cxa_exception, unwindHeader) ==
- sizeof(__cxa_exception),
- "unwindHeader has wrong negative offsets");
- static_assert(sizeof(_Unwind_Exception) +
- offsetof(__cxa_dependent_exception, unwindHeader) ==
- sizeof(__cxa_dependent_exception),
- "unwindHeader has wrong negative offsets");
- #if defined(_LIBCXXABI_ARM_EHABI)
- static_assert(offsetof(__cxa_exception, propagationCount) +
- sizeof(_Unwind_Exception) + sizeof(void*) ==
- sizeof(__cxa_exception),
- "propagationCount has wrong negative offset");
- static_assert(offsetof(__cxa_dependent_exception, propagationCount) +
- sizeof(_Unwind_Exception) + sizeof(void*) ==
- sizeof(__cxa_dependent_exception),
- "propagationCount has wrong negative offset");
- #elif defined(__LP64__) || defined(_WIN64)
- static_assert(offsetof(__cxa_exception, adjustedPtr) +
- sizeof(_Unwind_Exception) + sizeof(void*) ==
- sizeof(__cxa_exception),
- "adjustedPtr has wrong negative offset");
- static_assert(offsetof(__cxa_dependent_exception, adjustedPtr) +
- sizeof(_Unwind_Exception) + sizeof(void*) ==
- sizeof(__cxa_dependent_exception),
- "adjustedPtr has wrong negative offset");
- #else
- static_assert(offsetof(__cxa_exception, referenceCount) +
- sizeof(_Unwind_Exception) + sizeof(void*) ==
- sizeof(__cxa_exception),
- "referenceCount has wrong negative offset");
- static_assert(offsetof(__cxa_dependent_exception, primaryException) +
- sizeof(_Unwind_Exception) + sizeof(void*) ==
- sizeof(__cxa_dependent_exception),
- "primaryException has wrong negative offset");
- #endif
- struct _LIBCXXABI_HIDDEN __cxa_eh_globals {
- __cxa_exception * caughtExceptions;
- unsigned int uncaughtExceptions;
- #if defined(_LIBCXXABI_ARM_EHABI)
- __cxa_exception* propagatingExceptions;
- #endif
- };
- extern "C" _LIBCXXABI_FUNC_VIS __cxa_eh_globals * __cxa_get_globals ();
- extern "C" _LIBCXXABI_FUNC_VIS __cxa_eh_globals * __cxa_get_globals_fast ();
- extern "C" _LIBCXXABI_FUNC_VIS void * __cxa_allocate_dependent_exception ();
- extern "C" _LIBCXXABI_FUNC_VIS void __cxa_free_dependent_exception (void * dependent_exception);
- #endif // !__USING_EMSCRIPTEN_EXCEPTIONS__
- } // namespace __cxxabiv1
- #endif // _CXA_EXCEPTION_H
|