HaikuDepot: notify work status from main window
[haiku.git] / src / libs / libc++ / exception.cpp
blobe172f642d4830590e8840332514c557296002c37
1 //===------------------------ exception.cpp -------------------------------===//
2 //
3 // The LLVM Compiler Infrastructure
4 //
5 // This file is dual licensed under the MIT and the University of Illinois Open
6 // Source Licenses. See LICENSE.TXT for details.
7 //
8 //===----------------------------------------------------------------------===//
9 #include <stdlib.h>
10 #include <stdio.h>
12 #include "exception"
13 #include "new"
15 #if defined(__APPLE__) && !defined(LIBCXXRT)
16 #include <cxxabi.h>
18 using namespace __cxxabiv1;
19 #define HAVE_DEPENDENT_EH_ABI 1
20 #ifndef _LIBCPPABI_VERSION
21 using namespace __cxxabiapple;
22 // On Darwin, there are two STL shared libraries and a lower level ABI
23 // shared library. The globals holding the current terminate handler and
24 // current unexpected handler are in the ABI library.
25 #define __terminate_handler __cxxabiapple::__cxa_terminate_handler
26 #define __unexpected_handler __cxxabiapple::__cxa_unexpected_handler
27 #endif // _LIBCPPABI_VERSION
28 #elif defined(LIBCXXRT) || defined(LIBCXX_BUILDING_LIBCXXABI)
29 #include <cxxabi.h>
30 using namespace __cxxabiv1;
31 #if defined(LIBCXXRT) || defined(_LIBCPPABI_VERSION)
32 #define HAVE_DEPENDENT_EH_ABI 1
33 #endif
34 #elif !defined(__GLIBCXX__) // defined(LIBCXX_BUILDING_LIBCXXABI)
35 static std::terminate_handler __terminate_handler;
36 static std::unexpected_handler __unexpected_handler;
37 #endif // defined(LIBCXX_BUILDING_LIBCXXABI)
39 namespace std
42 #if !defined(LIBCXXRT) && !defined(_LIBCPPABI_VERSION) && !defined(__GLIBCXX__)
44 // libcxxrt provides implementations of these functions itself.
45 unexpected_handler
46 set_unexpected(unexpected_handler func) _NOEXCEPT
48 return __sync_lock_test_and_set(&__unexpected_handler, func);
51 unexpected_handler
52 get_unexpected() _NOEXCEPT
54 return __sync_fetch_and_add(&__unexpected_handler, (unexpected_handler)0);
57 _LIBCPP_NORETURN
58 void
59 unexpected()
61 (*get_unexpected())();
62 // unexpected handler should not return
63 terminate();
66 terminate_handler
67 set_terminate(terminate_handler func) _NOEXCEPT
69 return __sync_lock_test_and_set(&__terminate_handler, func);
72 terminate_handler
73 get_terminate() _NOEXCEPT
75 return __sync_fetch_and_add(&__terminate_handler, (terminate_handler)0);
78 #ifndef __EMSCRIPTEN__ // We provide this in JS
79 _LIBCPP_NORETURN
80 void
81 terminate() _NOEXCEPT
83 #ifndef _LIBCPP_NO_EXCEPTIONS
84 try
86 #endif // _LIBCPP_NO_EXCEPTIONS
87 (*get_terminate())();
88 // handler should not return
89 fprintf(stderr, "terminate_handler unexpectedly returned\n");
90 ::abort();
91 #ifndef _LIBCPP_NO_EXCEPTIONS
93 catch (...)
95 // handler should not throw exception
96 fprintf(stderr, "terminate_handler unexpectedly threw an exception\n");
97 ::abort();
99 #endif // _LIBCPP_NO_EXCEPTIONS
101 #endif // !__EMSCRIPTEN__
102 #endif // !defined(LIBCXXRT) && !defined(_LIBCPPABI_VERSION)
104 #if !defined(LIBCXXRT) && !defined(__GLIBCXX__) && !defined(__EMSCRIPTEN__)
105 bool uncaught_exception() _NOEXCEPT { return uncaught_exceptions() > 0; }
107 int uncaught_exceptions() _NOEXCEPT
109 #if defined(__APPLE__) || defined(_LIBCPPABI_VERSION)
110 // on Darwin, there is a helper function so __cxa_get_globals is private
111 # if _LIBCPPABI_VERSION > 1101
112 return __cxa_uncaught_exceptions();
113 # else
114 return __cxa_uncaught_exception() ? 1 : 0;
115 # endif
116 #else // __APPLE__
117 # if defined(_MSC_VER) && ! defined(__clang__)
118 _LIBCPP_WARNING("uncaught_exceptions not yet implemented")
119 # else
120 # warning uncaught_exception not yet implemented
121 # endif
122 fprintf(stderr, "uncaught_exceptions not yet implemented\n");
123 ::abort();
124 #endif // __APPLE__
128 #ifndef _LIBCPPABI_VERSION
130 exception::~exception() _NOEXCEPT
134 const char* exception::what() const _NOEXCEPT
136 return "std::exception";
139 #endif // _LIBCPPABI_VERSION
140 #endif //LIBCXXRT
141 #if !defined(_LIBCPPABI_VERSION) && !defined(__GLIBCXX__)
143 bad_exception::~bad_exception() _NOEXCEPT
147 const char* bad_exception::what() const _NOEXCEPT
149 return "std::bad_exception";
152 #endif
154 #if defined(__GLIBCXX__)
156 // libsupc++ does not implement the dependent EH ABI and the functionality
157 // it uses to implement std::exception_ptr (which it declares as an alias of
158 // std::__exception_ptr::exception_ptr) is not directly exported to clients. So
159 // we have little choice but to hijack std::__exception_ptr::exception_ptr's
160 // (which fortunately has the same layout as our std::exception_ptr) copy
161 // constructor, assignment operator and destructor (which are part of its
162 // stable ABI), and its rethrow_exception(std::__exception_ptr::exception_ptr)
163 // function.
165 namespace __exception_ptr
168 struct exception_ptr
170 void* __ptr_;
172 exception_ptr(const exception_ptr&) _NOEXCEPT;
173 exception_ptr& operator=(const exception_ptr&) _NOEXCEPT;
174 ~exception_ptr() _NOEXCEPT;
179 _LIBCPP_NORETURN void rethrow_exception(__exception_ptr::exception_ptr);
181 #endif
183 exception_ptr::~exception_ptr() _NOEXCEPT
185 #if HAVE_DEPENDENT_EH_ABI
186 __cxa_decrement_exception_refcount(__ptr_);
187 #elif defined(__GLIBCXX__)
188 reinterpret_cast<__exception_ptr::exception_ptr*>(this)->~exception_ptr();
189 #else
190 # if defined(_MSC_VER) && ! defined(__clang__)
191 _LIBCPP_WARNING("exception_ptr not yet implemented")
192 # else
193 # warning exception_ptr not yet implemented
194 # endif
195 fprintf(stderr, "exception_ptr not yet implemented\n");
196 ::abort();
197 #endif
200 exception_ptr::exception_ptr(const exception_ptr& other) _NOEXCEPT
201 : __ptr_(other.__ptr_)
203 #if HAVE_DEPENDENT_EH_ABI
204 __cxa_increment_exception_refcount(__ptr_);
205 #elif defined(__GLIBCXX__)
206 new (reinterpret_cast<void*>(this)) __exception_ptr::exception_ptr(
207 reinterpret_cast<const __exception_ptr::exception_ptr&>(other));
208 #else
209 # if defined(_MSC_VER) && ! defined(__clang__)
210 _LIBCPP_WARNING("exception_ptr not yet implemented")
211 # else
212 # warning exception_ptr not yet implemented
213 # endif
214 fprintf(stderr, "exception_ptr not yet implemented\n");
215 ::abort();
216 #endif
219 exception_ptr& exception_ptr::operator=(const exception_ptr& other) _NOEXCEPT
221 #if HAVE_DEPENDENT_EH_ABI
222 if (__ptr_ != other.__ptr_)
224 __cxa_increment_exception_refcount(other.__ptr_);
225 __cxa_decrement_exception_refcount(__ptr_);
226 __ptr_ = other.__ptr_;
228 return *this;
229 #elif defined(__GLIBCXX__)
230 *reinterpret_cast<__exception_ptr::exception_ptr*>(this) =
231 reinterpret_cast<const __exception_ptr::exception_ptr&>(other);
232 return *this;
233 #else
234 # if defined(_MSC_VER) && ! defined(__clang__)
235 _LIBCPP_WARNING("exception_ptr not yet implemented")
236 # else
237 # warning exception_ptr not yet implemented
238 # endif
239 fprintf(stderr, "exception_ptr not yet implemented\n");
240 ::abort();
241 #endif
244 nested_exception::nested_exception() _NOEXCEPT
245 : __ptr_(current_exception())
249 #if !defined(__GLIBCXX__)
251 nested_exception::~nested_exception() _NOEXCEPT
255 #endif
257 _LIBCPP_NORETURN
258 void
259 nested_exception::rethrow_nested() const
261 if (__ptr_ == nullptr)
262 terminate();
263 rethrow_exception(__ptr_);
266 #if !defined(__GLIBCXX__)
268 exception_ptr current_exception() _NOEXCEPT
270 #if HAVE_DEPENDENT_EH_ABI
271 // be nicer if there was a constructor that took a ptr, then
272 // this whole function would be just:
273 // return exception_ptr(__cxa_current_primary_exception());
274 exception_ptr ptr;
275 ptr.__ptr_ = __cxa_current_primary_exception();
276 return ptr;
277 #else
278 # if defined(_MSC_VER) && ! defined(__clang__)
279 _LIBCPP_WARNING( "exception_ptr not yet implemented" )
280 # else
281 # warning exception_ptr not yet implemented
282 # endif
283 fprintf(stderr, "exception_ptr not yet implemented\n");
284 ::abort();
285 #endif
288 #endif // !__GLIBCXX__
290 _LIBCPP_NORETURN
291 void rethrow_exception(exception_ptr p)
293 #if HAVE_DEPENDENT_EH_ABI
294 __cxa_rethrow_primary_exception(p.__ptr_);
295 // if p.__ptr_ is NULL, above returns so we terminate
296 terminate();
297 #elif defined(__GLIBCXX__)
298 rethrow_exception(reinterpret_cast<__exception_ptr::exception_ptr&>(p));
299 #else
300 # if defined(_MSC_VER) && ! defined(__clang__)
301 _LIBCPP_WARNING("exception_ptr not yet implemented")
302 # else
303 # warning exception_ptr not yet implemented
304 # endif
305 fprintf(stderr, "exception_ptr not yet implemented\n");
306 ::abort();
307 #endif
309 } // std