blob: a69f30be3ed8837bde0c2156736928a4245b1b14 [file] [log] [blame]
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001// -*- C++ -*-
2//===-------------------------- exception ---------------------------------===//
3//
Howard Hinnantf5256e12010-05-11 21:36:01 +00004// The LLVM Compiler Infrastructure
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00005//
6// This file is distributed under the University of Illinois Open Source
7// License. See LICENSE.TXT for details.
8//
9//===----------------------------------------------------------------------===//
10
11#ifndef _LIBCPP_EXCEPTION
12#define _LIBCPP_EXCEPTION
13
14/*
15 exception synopsis
16
17namespace std
18{
19
20class exception
21{
22public:
23 exception() throw();
24 exception(const exception&) throw();
25 exception& operator=(const exception&) throw();
26 virtual ~exception() throw();
27 virtual const char* what() const throw();
28};
29
30class bad_exception
31 : public exception
32{
33public:
34 bad_exception() throw();
35 bad_exception(const bad_exception&) throw();
36 bad_exception& operator=(const bad_exception&) throw();
37 virtual ~bad_exception() throw();
38 virtual const char* what() const throw();
39};
40
41typedef void (*unexpected_handler)();
42unexpected_handler set_unexpected(unexpected_handler f ) throw();
43void unexpected [[noreturn]] ();
44
45typedef void (*terminate_handler)();
46terminate_handler set_terminate(terminate_handler f ) throw();
47void terminate [[noreturn]] ();
48
49bool uncaught_exception() throw();
50
51typedef unspecified exception_ptr;
52
53exception_ptr current_exception();
54void rethrow_exception [[noreturn]] (exception_ptr p);
55template<class E> exception_ptr make_exception_ptr(E e);
56
57class nested_exception
58{
59public:
60 nested_exception() throw();
61 nested_exception(const nested_exception&) throw() = default;
62 nested_exception& operator=(const nested_exception&) throw() = default;
63 virtual ~nested_exception() = default;
64
65 // access functions
66 void rethrow_nested [[noreturn]] () const;
67 exception_ptr nested_ptr() const;
68};
69
70template <class T> void throw_with_nested [[noreturn]] (T&& t);
71template <class E> void rethrow_if_nested(const E& e);
72
73} // std
74
75*/
76
77#include <__config>
78#include <cstddef>
Howard Hinnanted2c2912010-05-27 17:06:52 +000079#include <type_traits>
Howard Hinnantbc8d3f92010-05-11 19:42:16 +000080
81#pragma GCC system_header
82
83namespace std // purposefully not using versioning namespace
84{
85
86class _LIBCPP_EXCEPTION_ABI exception
87{
88public:
89 _LIBCPP_INLINE_VISIBILITY exception() throw() {}
90 virtual ~exception() throw();
91 virtual const char* what() const throw();
92};
93
94class _LIBCPP_EXCEPTION_ABI bad_exception
95 : public exception
96{
97public:
98 _LIBCPP_INLINE_VISIBILITY bad_exception() throw() {}
99 virtual ~bad_exception() throw();
100 virtual const char* what() const throw();
101};
102
103typedef void (*unexpected_handler)();
104_LIBCPP_VISIBLE unexpected_handler set_unexpected(unexpected_handler) throw();
105_LIBCPP_VISIBLE void unexpected();
106
107typedef void (*terminate_handler)();
108_LIBCPP_VISIBLE terminate_handler set_terminate(terminate_handler) throw();
109_LIBCPP_VISIBLE void terminate() __attribute__((__noreturn__));
110
111_LIBCPP_VISIBLE bool uncaught_exception() throw();
112
113class exception_ptr;
114
115exception_ptr current_exception();
116void rethrow_exception(exception_ptr); // noreturn
117
Howard Hinnant422a53f2010-09-21 21:28:23 +0000118class _LIBCPP_VISIBLE exception_ptr
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000119{
120 void* __ptr_;
121public:
Howard Hinnant422a53f2010-09-21 21:28:23 +0000122 _LIBCPP_INLINE_VISIBILITY exception_ptr() : __ptr_() {}
123 _LIBCPP_INLINE_VISIBILITY exception_ptr(nullptr_t) : __ptr_() {}
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000124 exception_ptr(const exception_ptr&);
125 exception_ptr& operator=(const exception_ptr&);
126 ~exception_ptr();
127
Howard Hinnant422a53f2010-09-21 21:28:23 +0000128 _LIBCPP_INLINE_VISIBILITY
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000129 // explicit
130 operator bool() const {return __ptr_ != nullptr;}
131
Howard Hinnant422a53f2010-09-21 21:28:23 +0000132 friend _LIBCPP_INLINE_VISIBILITY
133 bool operator==(const exception_ptr& __x, const exception_ptr& __y)
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000134 {return __x.__ptr_ == __y.__ptr_;}
Howard Hinnant422a53f2010-09-21 21:28:23 +0000135 friend _LIBCPP_INLINE_VISIBILITY
136 bool operator!=(const exception_ptr& __x, const exception_ptr& __y)
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000137 {return !(__x == __y);}
138
139 friend exception_ptr current_exception();
140 friend void rethrow_exception(exception_ptr); // noreturn
141};
142
143template<class _E>
144exception_ptr
145make_exception_ptr(_E __e)
146{
Howard Hinnantd4444702010-08-11 17:04:31 +0000147#ifndef _LIBCPP_NO_EXCEPTIONS
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000148 try
149 {
150 throw __e;
151 }
152 catch (...)
153 {
154 return current_exception();
155 }
Howard Hinnant324bb032010-08-22 00:02:43 +0000156#endif // _LIBCPP_NO_EXCEPTIONS
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000157}
158
Howard Hinnanted2c2912010-05-27 17:06:52 +0000159// nested_exception
160
161class _LIBCPP_EXCEPTION_ABI nested_exception
162{
163 exception_ptr __ptr_;
164public:
165 nested_exception();
166// nested_exception(const nested_exception&) throw() = default;
167// nested_exception& operator=(const nested_exception&) throw() = default;
168 virtual ~nested_exception();
169
170 // access functions
171 void rethrow_nested /*[[noreturn]]*/ () const;
Howard Hinnant422a53f2010-09-21 21:28:23 +0000172 _LIBCPP_INLINE_VISIBILITY exception_ptr nested_ptr() const {return __ptr_;}
Howard Hinnanted2c2912010-05-27 17:06:52 +0000173};
174
175template <class _Tp>
176struct __nested
177 : public _Tp,
178 public nested_exception
179{
Howard Hinnant422a53f2010-09-21 21:28:23 +0000180 _LIBCPP_INLINE_VISIBILITY explicit __nested(const _Tp& __t) : _Tp(__t) {}
Howard Hinnanted2c2912010-05-27 17:06:52 +0000181};
182
183template <class _Tp>
Howard Hinnant324bb032010-08-22 00:02:43 +0000184void
Howard Hinnant73d21a42010-09-04 23:28:19 +0000185#ifndef _LIBCPP_HAS_NO_RVALUE_REFERENCES
Howard Hinnanted2c2912010-05-27 17:06:52 +0000186throw_with_nested /*[[noreturn]]*/ (_Tp&& __t, typename enable_if<
187 is_class<typename remove_reference<_Tp>::type>::value &&
188 !is_base_of<nested_exception, typename remove_reference<_Tp>::type>::value
189 >::type* = 0)
Howard Hinnant73d21a42010-09-04 23:28:19 +0000190#else // _LIBCPP_HAS_NO_RVALUE_REFERENCES
Howard Hinnanted2c2912010-05-27 17:06:52 +0000191throw_with_nested (_Tp& __t, typename enable_if<
192 is_class<_Tp>::value && !is_base_of<nested_exception, _Tp>::value
193 >::type* = 0)
Howard Hinnant73d21a42010-09-04 23:28:19 +0000194#endif // _LIBCPP_HAS_NO_RVALUE_REFERENCES
Howard Hinnanted2c2912010-05-27 17:06:52 +0000195{
Howard Hinnantd4444702010-08-11 17:04:31 +0000196#ifndef _LIBCPP_NO_EXCEPTIONS
Howard Hinnanted2c2912010-05-27 17:06:52 +0000197 throw __nested<typename remove_reference<_Tp>::type>(_STD::forward<_Tp>(__t));
Howard Hinnantd4444702010-08-11 17:04:31 +0000198#endif
Howard Hinnanted2c2912010-05-27 17:06:52 +0000199}
200
201template <class _Tp>
Howard Hinnant324bb032010-08-22 00:02:43 +0000202void
Howard Hinnant73d21a42010-09-04 23:28:19 +0000203#ifndef _LIBCPP_HAS_NO_RVALUE_REFERENCES
Howard Hinnanted2c2912010-05-27 17:06:52 +0000204throw_with_nested /*[[noreturn]]*/ (_Tp&& __t, typename enable_if<
205 !is_class<typename remove_reference<_Tp>::type>::value ||
206 is_base_of<nested_exception, typename remove_reference<_Tp>::type>::value
207 >::type* = 0)
Howard Hinnant73d21a42010-09-04 23:28:19 +0000208#else // _LIBCPP_HAS_NO_RVALUE_REFERENCES
Howard Hinnanted2c2912010-05-27 17:06:52 +0000209throw_with_nested (_Tp& __t, typename enable_if<
210 !is_class<_Tp>::value || is_base_of<nested_exception, _Tp>::value
211 >::type* = 0)
Howard Hinnant73d21a42010-09-04 23:28:19 +0000212#endif // _LIBCPP_HAS_NO_RVALUE_REFERENCES
Howard Hinnanted2c2912010-05-27 17:06:52 +0000213{
Howard Hinnantd4444702010-08-11 17:04:31 +0000214#ifndef _LIBCPP_NO_EXCEPTIONS
Howard Hinnanted2c2912010-05-27 17:06:52 +0000215 throw _STD::forward<_Tp>(__t);
Howard Hinnantd4444702010-08-11 17:04:31 +0000216#endif
Howard Hinnanted2c2912010-05-27 17:06:52 +0000217}
218
219template <class _E>
Howard Hinnant422a53f2010-09-21 21:28:23 +0000220inline _LIBCPP_INLINE_VISIBILITY
Howard Hinnanted2c2912010-05-27 17:06:52 +0000221void
222rethrow_if_nested(const _E& __e, typename enable_if<
Howard Hinnant6bb9f582010-05-28 13:35:41 +0000223 is_polymorphic<_E>::value
Howard Hinnanted2c2912010-05-27 17:06:52 +0000224 >::type* = 0)
225{
Howard Hinnant6bb9f582010-05-28 13:35:41 +0000226 const nested_exception* __nep = dynamic_cast<const nested_exception*>(&__e);
227 if (__nep)
228 __nep->rethrow_nested();
Howard Hinnanted2c2912010-05-27 17:06:52 +0000229}
230
231template <class _E>
Howard Hinnant422a53f2010-09-21 21:28:23 +0000232inline _LIBCPP_INLINE_VISIBILITY
Howard Hinnanted2c2912010-05-27 17:06:52 +0000233void
234rethrow_if_nested(const _E& __e, typename enable_if<
Howard Hinnant6bb9f582010-05-28 13:35:41 +0000235 !is_polymorphic<_E>::value
Howard Hinnanted2c2912010-05-27 17:06:52 +0000236 >::type* = 0)
237{
238}
239
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000240} // std
241
242#endif // _LIBCPP_EXCEPTION