1// -*- C++ -*-
2//===----------------------------------------------------------------------===//
3//
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
7//
8//===----------------------------------------------------------------------===//
9
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)
17// function.
18
19namespace std {
20
21namespace __exception_ptr
22{
23
24struct exception_ptr
25{
26    void* __ptr_;
27
28    exception_ptr(const exception_ptr&) noexcept;
29    exception_ptr& operator=(const exception_ptr&) noexcept;
30    ~exception_ptr() noexcept;
31};
32
33}
34
35_LIBCPP_NORETURN void rethrow_exception(__exception_ptr::exception_ptr);
36
37exception_ptr::~exception_ptr() noexcept
38{
39    reinterpret_cast<__exception_ptr::exception_ptr*>(this)->~exception_ptr();
40}
41
42exception_ptr::exception_ptr(const exception_ptr& other) noexcept
43    : __ptr_(other.__ptr_)
44{
45    new (reinterpret_cast<void*>(this)) __exception_ptr::exception_ptr(
46        reinterpret_cast<const __exception_ptr::exception_ptr&>(other));
47}
48
49exception_ptr& exception_ptr::operator=(const exception_ptr& other) noexcept
50{
51    *reinterpret_cast<__exception_ptr::exception_ptr*>(this) =
52        reinterpret_cast<const __exception_ptr::exception_ptr&>(other);
53    return *this;
54}
55
56nested_exception::nested_exception() noexcept
57    : __ptr_(current_exception())
58{
59}
60
61
62_LIBCPP_NORETURN
63void
64nested_exception::rethrow_nested() const
65{
66    if (__ptr_ == nullptr)
67        terminate();
68    rethrow_exception(__ptr_);
69}
70
71_LIBCPP_NORETURN
72void rethrow_exception(exception_ptr p)
73{
74    rethrow_exception(reinterpret_cast<__exception_ptr::exception_ptr&>(p));
75}
76
77} // namespace std
78