blob: 600b54838b2d855c831fdef18460089d10de3d0f [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
118class exception_ptr
119{
120 void* __ptr_;
121public:
122 exception_ptr() : __ptr_() {}
123 exception_ptr(nullptr_t) : __ptr_() {}
124 exception_ptr(const exception_ptr&);
125 exception_ptr& operator=(const exception_ptr&);
126 ~exception_ptr();
127
128 // explicit
129 operator bool() const {return __ptr_ != nullptr;}
130
131 friend bool operator==(const exception_ptr& __x, const exception_ptr& __y)
132 {return __x.__ptr_ == __y.__ptr_;}
133 friend bool operator!=(const exception_ptr& __x, const exception_ptr& __y)
134 {return !(__x == __y);}
135
136 friend exception_ptr current_exception();
137 friend void rethrow_exception(exception_ptr); // noreturn
138};
139
140template<class _E>
141exception_ptr
142make_exception_ptr(_E __e)
143{
144 try
145 {
146 throw __e;
147 }
148 catch (...)
149 {
150 return current_exception();
151 }
152}
153
Howard Hinnanted2c2912010-05-27 17:06:52 +0000154// nested_exception
155
156class _LIBCPP_EXCEPTION_ABI nested_exception
157{
158 exception_ptr __ptr_;
159public:
160 nested_exception();
161// nested_exception(const nested_exception&) throw() = default;
162// nested_exception& operator=(const nested_exception&) throw() = default;
163 virtual ~nested_exception();
164
165 // access functions
166 void rethrow_nested /*[[noreturn]]*/ () const;
167 exception_ptr nested_ptr() const {return __ptr_;}
168};
169
170template <class _Tp>
171struct __nested
172 : public _Tp,
173 public nested_exception
174{
175 explicit __nested(const _Tp& __t) : _Tp(__t) {}
176};
177
178template <class _Tp>
179void
180#ifdef _LIBCPP_MOVE
181throw_with_nested /*[[noreturn]]*/ (_Tp&& __t, typename enable_if<
182 is_class<typename remove_reference<_Tp>::type>::value &&
183 !is_base_of<nested_exception, typename remove_reference<_Tp>::type>::value
184 >::type* = 0)
185#else
186throw_with_nested (_Tp& __t, typename enable_if<
187 is_class<_Tp>::value && !is_base_of<nested_exception, _Tp>::value
188 >::type* = 0)
189#endif
190{
191 throw __nested<typename remove_reference<_Tp>::type>(_STD::forward<_Tp>(__t));
192}
193
194template <class _Tp>
195void
196#ifdef _LIBCPP_MOVE
197throw_with_nested /*[[noreturn]]*/ (_Tp&& __t, typename enable_if<
198 !is_class<typename remove_reference<_Tp>::type>::value ||
199 is_base_of<nested_exception, typename remove_reference<_Tp>::type>::value
200 >::type* = 0)
201#else
202throw_with_nested (_Tp& __t, typename enable_if<
203 !is_class<_Tp>::value || is_base_of<nested_exception, _Tp>::value
204 >::type* = 0)
205#endif
206{
207 throw _STD::forward<_Tp>(__t);
208}
209
210template <class _E>
211inline
212void
213rethrow_if_nested(const _E& __e, typename enable_if<
214 !is_same<_E, nested_exception>::value &&
215 is_convertible<_E*, nested_exception*>::value
216 >::type* = 0)
217{
218 static_cast<const nested_exception&>(__e).rethrow_nested();
219}
220
221template <class _E>
222inline
223void
224rethrow_if_nested(const _E& __e, typename enable_if<
225 is_same<_E, nested_exception>::value ||
226 !is_convertible<_E*, nested_exception*>::value
227 >::type* = 0)
228{
229}
230
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000231} // std
232
233#endif // _LIBCPP_EXCEPTION