2 //===----------------------------------------------------------------------===//
4 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
5 // See https://llvm.org/LICENSE.txt for license information.
6 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
8 //===----------------------------------------------------------------------===//
10 // libsupc++ does not implement the dependent EH ABI and the functionality
11 // it uses to implement std::exception_ptr (which it declares as an alias of
12 // std::__exception_ptr::exception_ptr) is not directly exported to clients. So
13 // we have little choice but to hijack std::__exception_ptr::exception_ptr's
14 // (which fortunately has the same layout as our std::exception_ptr) copy
15 // constructor, assignment operator and destructor (which are part of its
16 // stable ABI), and its rethrow_exception(std::__exception_ptr::exception_ptr)
21 namespace __exception_ptr {
23 struct exception_ptr {
26 explicit exception_ptr(void*) noexcept;
27 exception_ptr(const exception_ptr&) noexcept;
28 exception_ptr& operator=(const exception_ptr&) noexcept;
29 ~exception_ptr() noexcept;
32 } // namespace __exception_ptr
34 [[noreturn]] void rethrow_exception(__exception_ptr::exception_ptr);
36 exception_ptr::~exception_ptr() noexcept { reinterpret_cast<__exception_ptr::exception_ptr*>(this)->~exception_ptr(); }
38 exception_ptr::exception_ptr(const exception_ptr& other) noexcept : __ptr_(other.__ptr_) {
39 new (reinterpret_cast<void*>(this))
40 __exception_ptr::exception_ptr(reinterpret_cast<const __exception_ptr::exception_ptr&>(other));
43 exception_ptr& exception_ptr::operator=(const exception_ptr& other) noexcept {
44 *reinterpret_cast<__exception_ptr::exception_ptr*>(this) =
45 reinterpret_cast<const __exception_ptr::exception_ptr&>(other);
49 exception_ptr exception_ptr::__from_native_exception_pointer(void* __e) noexcept {
51 new (reinterpret_cast<void*>(&ptr)) __exception_ptr::exception_ptr(__e);
56 nested_exception::nested_exception() noexcept : __ptr_(current_exception()) {}
58 [[noreturn]] void nested_exception::rethrow_nested() const {
59 if (__ptr_ == nullptr)
61 rethrow_exception(__ptr_);
64 [[noreturn]] void rethrow_exception(exception_ptr p) {
65 rethrow_exception(reinterpret_cast<__exception_ptr::exception_ptr&>(p));