Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 1 | // -*- C++ -*- |
| 2 | //===---------------------------- system_error ----------------------------===// |
| 3 | // |
Howard Hinnant | f5256e1 | 2010-05-11 21:36:01 +0000 | [diff] [blame] | 4 | // The LLVM Compiler Infrastructure |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 5 | // |
Howard Hinnant | b64f8b0 | 2010-11-16 22:09:02 +0000 | [diff] [blame] | 6 | // This file is dual licensed under the MIT and the University of Illinois Open |
| 7 | // Source Licenses. See LICENSE.TXT for details. |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 8 | // |
| 9 | //===----------------------------------------------------------------------===// |
| 10 | |
| 11 | #ifndef _LIBCPP_SYSTEM_ERROR |
| 12 | #define _LIBCPP_SYSTEM_ERROR |
| 13 | |
| 14 | /* |
| 15 | system_error synopsis |
| 16 | |
| 17 | namespace std |
| 18 | { |
| 19 | |
| 20 | class error_category |
| 21 | { |
| 22 | public: |
Howard Hinnant | 1e15fd1 | 2011-05-26 19:48:01 +0000 | [diff] [blame] | 23 | virtual ~error_category() noexcept; |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 24 | |
| 25 | error_category(const error_category&) = delete; |
| 26 | error_category& operator=(const error_category&) = delete; |
| 27 | |
Howard Hinnant | 1e15fd1 | 2011-05-26 19:48:01 +0000 | [diff] [blame] | 28 | virtual const char* name() const noexcept = 0; |
| 29 | virtual error_condition default_error_condition(int ev) const noexcept; |
| 30 | virtual bool equivalent(int code, const error_condition& condition) const noexcept; |
| 31 | virtual bool equivalent(const error_code& code, int condition) const noexcept; |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 32 | virtual string message(int ev) const = 0; |
| 33 | |
Howard Hinnant | 1e15fd1 | 2011-05-26 19:48:01 +0000 | [diff] [blame] | 34 | bool operator==(const error_category& rhs) const noexcept; |
| 35 | bool operator!=(const error_category& rhs) const noexcept; |
| 36 | bool operator<(const error_category& rhs) const noexcept; |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 37 | }; |
| 38 | |
Howard Hinnant | 1e15fd1 | 2011-05-26 19:48:01 +0000 | [diff] [blame] | 39 | const error_category& generic_category() noexcept; |
| 40 | const error_category& system_category() noexcept; |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 41 | |
| 42 | template <class T> struct is_error_code_enum |
| 43 | : public false_type {}; |
| 44 | |
| 45 | template <class T> struct is_error_condition_enum |
| 46 | : public false_type {}; |
| 47 | |
| 48 | class error_code |
| 49 | { |
| 50 | public: |
| 51 | // constructors: |
Howard Hinnant | 1e15fd1 | 2011-05-26 19:48:01 +0000 | [diff] [blame] | 52 | error_code() noexcept; |
| 53 | error_code(int val, const error_category& cat) noexcept; |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 54 | template <class ErrorCodeEnum> |
Howard Hinnant | 1e15fd1 | 2011-05-26 19:48:01 +0000 | [diff] [blame] | 55 | error_code(ErrorCodeEnum e) noexcept; |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 56 | |
| 57 | // modifiers: |
Howard Hinnant | 1e15fd1 | 2011-05-26 19:48:01 +0000 | [diff] [blame] | 58 | void assign(int val, const error_category& cat) noexcept; |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 59 | template <class ErrorCodeEnum> |
Howard Hinnant | 1e15fd1 | 2011-05-26 19:48:01 +0000 | [diff] [blame] | 60 | error_code& operator=(ErrorCodeEnum e) noexcept; |
| 61 | void clear() noexcept; |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 62 | |
| 63 | // observers: |
Howard Hinnant | 1e15fd1 | 2011-05-26 19:48:01 +0000 | [diff] [blame] | 64 | int value() const noexcept; |
| 65 | const error_category& category() const noexcept; |
| 66 | error_condition default_error_condition() const noexcept; |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 67 | string message() const; |
Howard Hinnant | 1e15fd1 | 2011-05-26 19:48:01 +0000 | [diff] [blame] | 68 | explicit operator bool() const noexcept; |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 69 | }; |
| 70 | |
| 71 | // non-member functions: |
Howard Hinnant | 1e15fd1 | 2011-05-26 19:48:01 +0000 | [diff] [blame] | 72 | bool operator<(const error_code& lhs, const error_code& rhs) noexcept; |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 73 | template <class charT, class traits> |
| 74 | basic_ostream<charT,traits>& |
| 75 | operator<<(basic_ostream<charT,traits>& os, const error_code& ec); |
| 76 | |
| 77 | class error_condition |
| 78 | { |
| 79 | public: |
| 80 | // constructors: |
Howard Hinnant | 1e15fd1 | 2011-05-26 19:48:01 +0000 | [diff] [blame] | 81 | error_condition() noexcept; |
| 82 | error_condition(int val, const error_category& cat) noexcept; |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 83 | template <class ErrorConditionEnum> |
Howard Hinnant | 1e15fd1 | 2011-05-26 19:48:01 +0000 | [diff] [blame] | 84 | error_condition(ErrorConditionEnum e) noexcept; |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 85 | |
| 86 | // modifiers: |
Howard Hinnant | 1e15fd1 | 2011-05-26 19:48:01 +0000 | [diff] [blame] | 87 | void assign(int val, const error_category& cat) noexcept; |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 88 | template <class ErrorConditionEnum> |
Howard Hinnant | 1e15fd1 | 2011-05-26 19:48:01 +0000 | [diff] [blame] | 89 | error_condition& operator=(ErrorConditionEnum e) noexcept; |
| 90 | void clear() noexcept; |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 91 | |
| 92 | // observers: |
Howard Hinnant | 1e15fd1 | 2011-05-26 19:48:01 +0000 | [diff] [blame] | 93 | int value() const noexcept; |
| 94 | const error_category& category() const noexcept; |
| 95 | string message() const noexcept; |
| 96 | explicit operator bool() const noexcept; |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 97 | }; |
| 98 | |
Howard Hinnant | 1e15fd1 | 2011-05-26 19:48:01 +0000 | [diff] [blame] | 99 | bool operator<(const error_condition& lhs, const error_condition& rhs) noexcept; |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 100 | |
| 101 | class system_error |
| 102 | : public runtime_error |
| 103 | { |
| 104 | public: |
| 105 | system_error(error_code ec, const string& what_arg); |
| 106 | system_error(error_code ec, const char* what_arg); |
| 107 | system_error(error_code ec); |
| 108 | system_error(int ev, const error_category& ecat, const string& what_arg); |
| 109 | system_error(int ev, const error_category& ecat, const char* what_arg); |
| 110 | system_error(int ev, const error_category& ecat); |
| 111 | |
Howard Hinnant | 1e15fd1 | 2011-05-26 19:48:01 +0000 | [diff] [blame] | 112 | const error_code& code() const noexcept; |
| 113 | const char* what() const noexcept; |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 114 | }; |
| 115 | |
| 116 | enum class errc |
| 117 | { |
| 118 | address_family_not_supported, // EAFNOSUPPORT |
| 119 | address_in_use, // EADDRINUSE |
| 120 | address_not_available, // EADDRNOTAVAIL |
| 121 | already_connected, // EISCONN |
| 122 | argument_list_too_long, // E2BIG |
| 123 | argument_out_of_domain, // EDOM |
| 124 | bad_address, // EFAULT |
| 125 | bad_file_descriptor, // EBADF |
| 126 | bad_message, // EBADMSG |
| 127 | broken_pipe, // EPIPE |
| 128 | connection_aborted, // ECONNABORTED |
| 129 | connection_already_in_progress, // EALREADY |
| 130 | connection_refused, // ECONNREFUSED |
| 131 | connection_reset, // ECONNRESET |
| 132 | cross_device_link, // EXDEV |
| 133 | destination_address_required, // EDESTADDRREQ |
| 134 | device_or_resource_busy, // EBUSY |
| 135 | directory_not_empty, // ENOTEMPTY |
| 136 | executable_format_error, // ENOEXEC |
| 137 | file_exists, // EEXIST |
| 138 | file_too_large, // EFBIG |
| 139 | filename_too_long, // ENAMETOOLONG |
| 140 | function_not_supported, // ENOSYS |
| 141 | host_unreachable, // EHOSTUNREACH |
| 142 | identifier_removed, // EIDRM |
| 143 | illegal_byte_sequence, // EILSEQ |
| 144 | inappropriate_io_control_operation, // ENOTTY |
| 145 | interrupted, // EINTR |
| 146 | invalid_argument, // EINVAL |
| 147 | invalid_seek, // ESPIPE |
| 148 | io_error, // EIO |
| 149 | is_a_directory, // EISDIR |
| 150 | message_size, // EMSGSIZE |
| 151 | network_down, // ENETDOWN |
| 152 | network_reset, // ENETRESET |
| 153 | network_unreachable, // ENETUNREACH |
| 154 | no_buffer_space, // ENOBUFS |
| 155 | no_child_process, // ECHILD |
| 156 | no_link, // ENOLINK |
| 157 | no_lock_available, // ENOLCK |
| 158 | no_message_available, // ENODATA |
| 159 | no_message, // ENOMSG |
| 160 | no_protocol_option, // ENOPROTOOPT |
| 161 | no_space_on_device, // ENOSPC |
| 162 | no_stream_resources, // ENOSR |
| 163 | no_such_device_or_address, // ENXIO |
| 164 | no_such_device, // ENODEV |
| 165 | no_such_file_or_directory, // ENOENT |
| 166 | no_such_process, // ESRCH |
| 167 | not_a_directory, // ENOTDIR |
| 168 | not_a_socket, // ENOTSOCK |
| 169 | not_a_stream, // ENOSTR |
| 170 | not_connected, // ENOTCONN |
| 171 | not_enough_memory, // ENOMEM |
| 172 | not_supported, // ENOTSUP |
| 173 | operation_canceled, // ECANCELED |
| 174 | operation_in_progress, // EINPROGRESS |
| 175 | operation_not_permitted, // EPERM |
| 176 | operation_not_supported, // EOPNOTSUPP |
| 177 | operation_would_block, // EWOULDBLOCK |
| 178 | owner_dead, // EOWNERDEAD |
| 179 | permission_denied, // EACCES |
| 180 | protocol_error, // EPROTO |
| 181 | protocol_not_supported, // EPROTONOSUPPORT |
| 182 | read_only_file_system, // EROFS |
| 183 | resource_deadlock_would_occur, // EDEADLK |
| 184 | resource_unavailable_try_again, // EAGAIN |
| 185 | result_out_of_range, // ERANGE |
| 186 | state_not_recoverable, // ENOTRECOVERABLE |
| 187 | stream_timeout, // ETIME |
| 188 | text_file_busy, // ETXTBSY |
| 189 | timed_out, // ETIMEDOUT |
| 190 | too_many_files_open_in_system, // ENFILE |
| 191 | too_many_files_open, // EMFILE |
| 192 | too_many_links, // EMLINK |
| 193 | too_many_symbolic_link_levels, // ELOOP |
| 194 | value_too_large, // EOVERFLOW |
| 195 | wrong_protocol_type // EPROTOTYPE |
| 196 | }; |
| 197 | |
| 198 | template <> struct is_error_condition_enum<errc> |
| 199 | : true_type { } |
| 200 | |
Howard Hinnant | 1e15fd1 | 2011-05-26 19:48:01 +0000 | [diff] [blame] | 201 | error_code make_error_code(errc e) noexcept; |
| 202 | error_condition make_error_condition(errc e) noexcept; |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 203 | |
| 204 | // Comparison operators: |
Howard Hinnant | 1e15fd1 | 2011-05-26 19:48:01 +0000 | [diff] [blame] | 205 | bool operator==(const error_code& lhs, const error_code& rhs) noexcept; |
| 206 | bool operator==(const error_code& lhs, const error_condition& rhs) noexcept; |
| 207 | bool operator==(const error_condition& lhs, const error_code& rhs) noexcept; |
| 208 | bool operator==(const error_condition& lhs, const error_condition& rhs) noexcept; |
| 209 | bool operator!=(const error_code& lhs, const error_code& rhs) noexcept; |
| 210 | bool operator!=(const error_code& lhs, const error_condition& rhs) noexcept; |
| 211 | bool operator!=(const error_condition& lhs, const error_code& rhs) noexcept; |
| 212 | bool operator!=(const error_condition& lhs, const error_condition& rhs) noexcept; |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 213 | |
| 214 | template <> struct hash<std::error_code>; |
| 215 | |
| 216 | } // std |
| 217 | |
| 218 | */ |
| 219 | |
| 220 | #include <__config> |
| 221 | #include <cerrno> |
| 222 | #include <type_traits> |
| 223 | #include <stdexcept> |
| 224 | #include <__functional_base> |
| 225 | |
Howard Hinnant | 08e1747 | 2011-10-17 20:05:10 +0000 | [diff] [blame] | 226 | #if !defined(_LIBCPP_HAS_NO_PRAGMA_SYSTEM_HEADER) |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 227 | #pragma GCC system_header |
Howard Hinnant | 08e1747 | 2011-10-17 20:05:10 +0000 | [diff] [blame] | 228 | #endif |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 229 | |
| 230 | _LIBCPP_BEGIN_NAMESPACE_STD |
| 231 | |
| 232 | // is_error_code_enum |
| 233 | |
Howard Hinnant | 8d7a955 | 2010-09-23 17:31:07 +0000 | [diff] [blame] | 234 | template <class _Tp> |
| 235 | struct _LIBCPP_VISIBLE is_error_code_enum |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 236 | : public false_type {}; |
| 237 | |
| 238 | // is_error_condition_enum |
| 239 | |
Howard Hinnant | 8d7a955 | 2010-09-23 17:31:07 +0000 | [diff] [blame] | 240 | template <class _Tp> |
| 241 | struct _LIBCPP_VISIBLE is_error_condition_enum |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 242 | : public false_type {}; |
| 243 | |
David Chisnall | 81e6858 | 2010-08-11 16:52:41 +0000 | [diff] [blame] | 244 | // Some error codes are not present on all platforms, so we provide equivalents |
| 245 | // for them: |
| 246 | |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 247 | //enum class errc |
| 248 | struct errc |
| 249 | { |
| 250 | enum _ { |
| 251 | address_family_not_supported = EAFNOSUPPORT, |
| 252 | address_in_use = EADDRINUSE, |
| 253 | address_not_available = EADDRNOTAVAIL, |
| 254 | already_connected = EISCONN, |
| 255 | argument_list_too_long = E2BIG, |
| 256 | argument_out_of_domain = EDOM, |
| 257 | bad_address = EFAULT, |
| 258 | bad_file_descriptor = EBADF, |
| 259 | bad_message = EBADMSG, |
| 260 | broken_pipe = EPIPE, |
| 261 | connection_aborted = ECONNABORTED, |
| 262 | connection_already_in_progress = EALREADY, |
| 263 | connection_refused = ECONNREFUSED, |
| 264 | connection_reset = ECONNRESET, |
| 265 | cross_device_link = EXDEV, |
| 266 | destination_address_required = EDESTADDRREQ, |
| 267 | device_or_resource_busy = EBUSY, |
| 268 | directory_not_empty = ENOTEMPTY, |
| 269 | executable_format_error = ENOEXEC, |
| 270 | file_exists = EEXIST, |
| 271 | file_too_large = EFBIG, |
| 272 | filename_too_long = ENAMETOOLONG, |
| 273 | function_not_supported = ENOSYS, |
| 274 | host_unreachable = EHOSTUNREACH, |
| 275 | identifier_removed = EIDRM, |
| 276 | illegal_byte_sequence = EILSEQ, |
| 277 | inappropriate_io_control_operation = ENOTTY, |
| 278 | interrupted = EINTR, |
| 279 | invalid_argument = EINVAL, |
| 280 | invalid_seek = ESPIPE, |
| 281 | io_error = EIO, |
| 282 | is_a_directory = EISDIR, |
| 283 | message_size = EMSGSIZE, |
| 284 | network_down = ENETDOWN, |
| 285 | network_reset = ENETRESET, |
| 286 | network_unreachable = ENETUNREACH, |
| 287 | no_buffer_space = ENOBUFS, |
| 288 | no_child_process = ECHILD, |
| 289 | no_link = ENOLINK, |
| 290 | no_lock_available = ENOLCK, |
David Chisnall | 81e6858 | 2010-08-11 16:52:41 +0000 | [diff] [blame] | 291 | #ifdef ENODATA |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 292 | no_message_available = ENODATA, |
David Chisnall | 81e6858 | 2010-08-11 16:52:41 +0000 | [diff] [blame] | 293 | #else |
| 294 | no_message_available = ENOMSG, |
| 295 | #endif |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 296 | no_message = ENOMSG, |
| 297 | no_protocol_option = ENOPROTOOPT, |
| 298 | no_space_on_device = ENOSPC, |
David Chisnall | 81e6858 | 2010-08-11 16:52:41 +0000 | [diff] [blame] | 299 | #ifdef ENOSR |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 300 | no_stream_resources = ENOSR, |
David Chisnall | 81e6858 | 2010-08-11 16:52:41 +0000 | [diff] [blame] | 301 | #else |
| 302 | no_stream_resources = ENOMEM, |
| 303 | #endif |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 304 | no_such_device_or_address = ENXIO, |
| 305 | no_such_device = ENODEV, |
| 306 | no_such_file_or_directory = ENOENT, |
| 307 | no_such_process = ESRCH, |
| 308 | not_a_directory = ENOTDIR, |
| 309 | not_a_socket = ENOTSOCK, |
David Chisnall | 81e6858 | 2010-08-11 16:52:41 +0000 | [diff] [blame] | 310 | #ifdef ENOSTR |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 311 | not_a_stream = ENOSTR, |
David Chisnall | 81e6858 | 2010-08-11 16:52:41 +0000 | [diff] [blame] | 312 | #else |
| 313 | not_a_stream = EINVAL, |
| 314 | #endif |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 315 | not_connected = ENOTCONN, |
| 316 | not_enough_memory = ENOMEM, |
| 317 | not_supported = ENOTSUP, |
| 318 | operation_canceled = ECANCELED, |
| 319 | operation_in_progress = EINPROGRESS, |
| 320 | operation_not_permitted = EPERM, |
| 321 | operation_not_supported = EOPNOTSUPP, |
| 322 | operation_would_block = EWOULDBLOCK, |
| 323 | owner_dead = EOWNERDEAD, |
| 324 | permission_denied = EACCES, |
| 325 | protocol_error = EPROTO, |
| 326 | protocol_not_supported = EPROTONOSUPPORT, |
| 327 | read_only_file_system = EROFS, |
| 328 | resource_deadlock_would_occur = EDEADLK, |
| 329 | resource_unavailable_try_again = EAGAIN, |
| 330 | result_out_of_range = ERANGE, |
| 331 | state_not_recoverable = ENOTRECOVERABLE, |
David Chisnall | 81e6858 | 2010-08-11 16:52:41 +0000 | [diff] [blame] | 332 | #ifdef ETIME |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 333 | stream_timeout = ETIME, |
David Chisnall | 81e6858 | 2010-08-11 16:52:41 +0000 | [diff] [blame] | 334 | #else |
| 335 | stream_timeout = ETIMEDOUT, |
| 336 | #endif |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 337 | text_file_busy = ETXTBSY, |
| 338 | timed_out = ETIMEDOUT, |
| 339 | too_many_files_open_in_system = ENFILE, |
| 340 | too_many_files_open = EMFILE, |
| 341 | too_many_links = EMLINK, |
| 342 | too_many_symbolic_link_levels = ELOOP, |
| 343 | value_too_large = EOVERFLOW, |
| 344 | wrong_protocol_type = EPROTOTYPE |
| 345 | }; |
| 346 | |
| 347 | _ __v_; |
| 348 | |
Howard Hinnant | 8d7a955 | 2010-09-23 17:31:07 +0000 | [diff] [blame] | 349 | _LIBCPP_ALWAYS_INLINE |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 350 | errc(_ __v) : __v_(__v) {} |
Howard Hinnant | 8d7a955 | 2010-09-23 17:31:07 +0000 | [diff] [blame] | 351 | _LIBCPP_ALWAYS_INLINE |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 352 | operator int() const {return __v_;} |
| 353 | |
| 354 | }; |
| 355 | |
Howard Hinnant | 8d7a955 | 2010-09-23 17:31:07 +0000 | [diff] [blame] | 356 | template <> |
| 357 | struct _LIBCPP_VISIBLE is_error_condition_enum<errc> |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 358 | : true_type { }; |
| 359 | |
Howard Hinnant | 8d7a955 | 2010-09-23 17:31:07 +0000 | [diff] [blame] | 360 | template <> |
| 361 | struct _LIBCPP_VISIBLE is_error_condition_enum<errc::_> |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 362 | : true_type { }; |
| 363 | |
| 364 | class error_condition; |
| 365 | class error_code; |
| 366 | |
| 367 | // class error_category |
| 368 | |
| 369 | class __do_message; |
| 370 | |
Howard Hinnant | 8d7a955 | 2010-09-23 17:31:07 +0000 | [diff] [blame] | 371 | class _LIBCPP_VISIBLE error_category |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 372 | { |
| 373 | public: |
Howard Hinnant | 1e15fd1 | 2011-05-26 19:48:01 +0000 | [diff] [blame] | 374 | virtual ~error_category() _NOEXCEPT; |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 375 | |
| 376 | private: |
Howard Hinnant | 1e15fd1 | 2011-05-26 19:48:01 +0000 | [diff] [blame] | 377 | error_category() _NOEXCEPT; |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 378 | error_category(const error_category&);// = delete; |
| 379 | error_category& operator=(const error_category&);// = delete; |
| 380 | |
| 381 | public: |
Howard Hinnant | 1e15fd1 | 2011-05-26 19:48:01 +0000 | [diff] [blame] | 382 | virtual const char* name() const _NOEXCEPT = 0; |
| 383 | virtual error_condition default_error_condition(int __ev) const _NOEXCEPT; |
| 384 | virtual bool equivalent(int __code, const error_condition& __condition) const _NOEXCEPT; |
| 385 | virtual bool equivalent(const error_code& __code, int __condition) const _NOEXCEPT; |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 386 | virtual string message(int __ev) const = 0; |
| 387 | |
| 388 | _LIBCPP_ALWAYS_INLINE |
Howard Hinnant | 1e15fd1 | 2011-05-26 19:48:01 +0000 | [diff] [blame] | 389 | bool operator==(const error_category& __rhs) const _NOEXCEPT {return this == &__rhs;} |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 390 | |
| 391 | _LIBCPP_ALWAYS_INLINE |
Howard Hinnant | 1e15fd1 | 2011-05-26 19:48:01 +0000 | [diff] [blame] | 392 | bool operator!=(const error_category& __rhs) const _NOEXCEPT {return !(*this == __rhs);} |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 393 | |
| 394 | _LIBCPP_ALWAYS_INLINE |
Howard Hinnant | 1e15fd1 | 2011-05-26 19:48:01 +0000 | [diff] [blame] | 395 | bool operator< (const error_category& __rhs) const _NOEXCEPT {return this < &__rhs;} |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 396 | |
| 397 | friend class __do_message; |
| 398 | }; |
| 399 | |
| 400 | class _LIBCPP_HIDDEN __do_message |
| 401 | : public error_category |
| 402 | { |
| 403 | public: |
| 404 | virtual string message(int ev) const; |
| 405 | }; |
| 406 | |
Howard Hinnant | 1e15fd1 | 2011-05-26 19:48:01 +0000 | [diff] [blame] | 407 | const error_category& generic_category() _NOEXCEPT; |
| 408 | const error_category& system_category() _NOEXCEPT; |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 409 | |
Howard Hinnant | 8d7a955 | 2010-09-23 17:31:07 +0000 | [diff] [blame] | 410 | class _LIBCPP_VISIBLE error_condition |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 411 | { |
| 412 | int __val_; |
| 413 | const error_category* __cat_; |
| 414 | public: |
| 415 | _LIBCPP_ALWAYS_INLINE |
Howard Hinnant | 1e15fd1 | 2011-05-26 19:48:01 +0000 | [diff] [blame] | 416 | error_condition() _NOEXCEPT : __val_(0), __cat_(&generic_category()) {} |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 417 | |
| 418 | _LIBCPP_ALWAYS_INLINE |
Howard Hinnant | 1e15fd1 | 2011-05-26 19:48:01 +0000 | [diff] [blame] | 419 | error_condition(int __val, const error_category& __cat) _NOEXCEPT |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 420 | : __val_(__val), __cat_(&__cat) {} |
| 421 | |
Howard Hinnant | 9996844 | 2011-11-29 18:15:50 +0000 | [diff] [blame] | 422 | template <class _Ep> |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 423 | _LIBCPP_ALWAYS_INLINE |
Howard Hinnant | 9996844 | 2011-11-29 18:15:50 +0000 | [diff] [blame] | 424 | error_condition(_Ep __e, |
| 425 | typename enable_if<is_error_condition_enum<_Ep>::value>::type* = 0 |
Howard Hinnant | 1e15fd1 | 2011-05-26 19:48:01 +0000 | [diff] [blame] | 426 | ) _NOEXCEPT |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 427 | {*this = make_error_condition(__e);} |
| 428 | |
| 429 | _LIBCPP_ALWAYS_INLINE |
Howard Hinnant | 1e15fd1 | 2011-05-26 19:48:01 +0000 | [diff] [blame] | 430 | void assign(int __val, const error_category& __cat) _NOEXCEPT |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 431 | { |
| 432 | __val_ = __val; |
| 433 | __cat_ = &__cat; |
| 434 | } |
| 435 | |
Howard Hinnant | 9996844 | 2011-11-29 18:15:50 +0000 | [diff] [blame] | 436 | template <class _Ep> |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 437 | _LIBCPP_ALWAYS_INLINE |
| 438 | typename enable_if |
| 439 | < |
Howard Hinnant | 9996844 | 2011-11-29 18:15:50 +0000 | [diff] [blame] | 440 | is_error_condition_enum<_Ep>::value, |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 441 | error_condition& |
| 442 | >::type |
Howard Hinnant | 9996844 | 2011-11-29 18:15:50 +0000 | [diff] [blame] | 443 | operator=(_Ep __e) _NOEXCEPT |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 444 | {*this = make_error_condition(__e); return *this;} |
| 445 | |
| 446 | _LIBCPP_ALWAYS_INLINE |
Howard Hinnant | 1e15fd1 | 2011-05-26 19:48:01 +0000 | [diff] [blame] | 447 | void clear() _NOEXCEPT |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 448 | { |
| 449 | __val_ = 0; |
| 450 | __cat_ = &generic_category(); |
| 451 | } |
| 452 | |
| 453 | _LIBCPP_ALWAYS_INLINE |
Howard Hinnant | 1e15fd1 | 2011-05-26 19:48:01 +0000 | [diff] [blame] | 454 | int value() const _NOEXCEPT {return __val_;} |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 455 | |
| 456 | _LIBCPP_ALWAYS_INLINE |
Howard Hinnant | 1e15fd1 | 2011-05-26 19:48:01 +0000 | [diff] [blame] | 457 | const error_category& category() const _NOEXCEPT {return *__cat_;} |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 458 | string message() const; |
| 459 | |
| 460 | _LIBCPP_ALWAYS_INLINE |
| 461 | //explicit |
Howard Hinnant | 1e15fd1 | 2011-05-26 19:48:01 +0000 | [diff] [blame] | 462 | operator bool() const _NOEXCEPT {return __val_ != 0;} |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 463 | }; |
| 464 | |
| 465 | inline _LIBCPP_INLINE_VISIBILITY |
| 466 | error_condition |
Howard Hinnant | 1e15fd1 | 2011-05-26 19:48:01 +0000 | [diff] [blame] | 467 | make_error_condition(errc __e) _NOEXCEPT |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 468 | { |
| 469 | return error_condition(static_cast<int>(__e), generic_category()); |
| 470 | } |
| 471 | |
| 472 | inline _LIBCPP_INLINE_VISIBILITY |
| 473 | bool |
Howard Hinnant | 1e15fd1 | 2011-05-26 19:48:01 +0000 | [diff] [blame] | 474 | operator<(const error_condition& __x, const error_condition& __y) _NOEXCEPT |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 475 | { |
| 476 | return __x.category() < __y.category() |
Howard Hinnant | ec3773c | 2011-12-01 20:21:04 +0000 | [diff] [blame] | 477 | || (__x.category() == __y.category() && __x.value() < __y.value()); |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 478 | } |
| 479 | |
| 480 | // error_code |
| 481 | |
Howard Hinnant | 8d7a955 | 2010-09-23 17:31:07 +0000 | [diff] [blame] | 482 | class _LIBCPP_VISIBLE error_code |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 483 | { |
| 484 | int __val_; |
| 485 | const error_category* __cat_; |
| 486 | public: |
| 487 | _LIBCPP_ALWAYS_INLINE |
Howard Hinnant | 1e15fd1 | 2011-05-26 19:48:01 +0000 | [diff] [blame] | 488 | error_code() _NOEXCEPT : __val_(0), __cat_(&system_category()) {} |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 489 | |
| 490 | _LIBCPP_ALWAYS_INLINE |
Howard Hinnant | 1e15fd1 | 2011-05-26 19:48:01 +0000 | [diff] [blame] | 491 | error_code(int __val, const error_category& __cat) _NOEXCEPT |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 492 | : __val_(__val), __cat_(&__cat) {} |
| 493 | |
Howard Hinnant | 9996844 | 2011-11-29 18:15:50 +0000 | [diff] [blame] | 494 | template <class _Ep> |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 495 | _LIBCPP_ALWAYS_INLINE |
Howard Hinnant | 9996844 | 2011-11-29 18:15:50 +0000 | [diff] [blame] | 496 | error_code(_Ep __e, |
| 497 | typename enable_if<is_error_code_enum<_Ep>::value>::type* = 0 |
Howard Hinnant | 1e15fd1 | 2011-05-26 19:48:01 +0000 | [diff] [blame] | 498 | ) _NOEXCEPT |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 499 | {*this = make_error_code(__e);} |
| 500 | |
| 501 | _LIBCPP_ALWAYS_INLINE |
Howard Hinnant | 1e15fd1 | 2011-05-26 19:48:01 +0000 | [diff] [blame] | 502 | void assign(int __val, const error_category& __cat) _NOEXCEPT |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 503 | { |
| 504 | __val_ = __val; |
| 505 | __cat_ = &__cat; |
| 506 | } |
| 507 | |
Howard Hinnant | 9996844 | 2011-11-29 18:15:50 +0000 | [diff] [blame] | 508 | template <class _Ep> |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 509 | _LIBCPP_ALWAYS_INLINE |
| 510 | typename enable_if |
| 511 | < |
Howard Hinnant | 9996844 | 2011-11-29 18:15:50 +0000 | [diff] [blame] | 512 | is_error_code_enum<_Ep>::value, |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 513 | error_code& |
| 514 | >::type |
Howard Hinnant | 9996844 | 2011-11-29 18:15:50 +0000 | [diff] [blame] | 515 | operator=(_Ep __e) _NOEXCEPT |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 516 | {*this = make_error_code(__e); return *this;} |
| 517 | |
| 518 | _LIBCPP_ALWAYS_INLINE |
Howard Hinnant | 1e15fd1 | 2011-05-26 19:48:01 +0000 | [diff] [blame] | 519 | void clear() _NOEXCEPT |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 520 | { |
| 521 | __val_ = 0; |
| 522 | __cat_ = &system_category(); |
| 523 | } |
| 524 | |
| 525 | _LIBCPP_ALWAYS_INLINE |
Howard Hinnant | 1e15fd1 | 2011-05-26 19:48:01 +0000 | [diff] [blame] | 526 | int value() const _NOEXCEPT {return __val_;} |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 527 | |
| 528 | _LIBCPP_ALWAYS_INLINE |
Howard Hinnant | 1e15fd1 | 2011-05-26 19:48:01 +0000 | [diff] [blame] | 529 | const error_category& category() const _NOEXCEPT {return *__cat_;} |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 530 | |
| 531 | _LIBCPP_ALWAYS_INLINE |
Howard Hinnant | 1e15fd1 | 2011-05-26 19:48:01 +0000 | [diff] [blame] | 532 | error_condition default_error_condition() const _NOEXCEPT |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 533 | {return __cat_->default_error_condition(__val_);} |
| 534 | |
| 535 | string message() const; |
| 536 | |
| 537 | _LIBCPP_ALWAYS_INLINE |
| 538 | //explicit |
Howard Hinnant | 1e15fd1 | 2011-05-26 19:48:01 +0000 | [diff] [blame] | 539 | operator bool() const _NOEXCEPT {return __val_ != 0;} |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 540 | }; |
| 541 | |
| 542 | inline _LIBCPP_INLINE_VISIBILITY |
| 543 | error_code |
Howard Hinnant | 1e15fd1 | 2011-05-26 19:48:01 +0000 | [diff] [blame] | 544 | make_error_code(errc __e) _NOEXCEPT |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 545 | { |
| 546 | return error_code(static_cast<int>(__e), generic_category()); |
| 547 | } |
| 548 | |
| 549 | inline _LIBCPP_INLINE_VISIBILITY |
| 550 | bool |
Howard Hinnant | 1e15fd1 | 2011-05-26 19:48:01 +0000 | [diff] [blame] | 551 | operator<(const error_code& __x, const error_code& __y) _NOEXCEPT |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 552 | { |
| 553 | return __x.category() < __y.category() |
Howard Hinnant | ec3773c | 2011-12-01 20:21:04 +0000 | [diff] [blame] | 554 | || (__x.category() == __y.category() && __x.value() < __y.value()); |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 555 | } |
| 556 | |
| 557 | inline _LIBCPP_INLINE_VISIBILITY |
| 558 | bool |
Howard Hinnant | 1e15fd1 | 2011-05-26 19:48:01 +0000 | [diff] [blame] | 559 | operator==(const error_code& __x, const error_code& __y) _NOEXCEPT |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 560 | { |
| 561 | return __x.category() == __y.category() && __x.value() == __y.value(); |
| 562 | } |
| 563 | |
| 564 | inline _LIBCPP_INLINE_VISIBILITY |
| 565 | bool |
Howard Hinnant | 1e15fd1 | 2011-05-26 19:48:01 +0000 | [diff] [blame] | 566 | operator==(const error_code& __x, const error_condition& __y) _NOEXCEPT |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 567 | { |
| 568 | return __x.category().equivalent(__x.value(), __y) |
| 569 | || __y.category().equivalent(__x, __y.value()); |
| 570 | } |
| 571 | |
| 572 | inline _LIBCPP_INLINE_VISIBILITY |
| 573 | bool |
Howard Hinnant | 1e15fd1 | 2011-05-26 19:48:01 +0000 | [diff] [blame] | 574 | operator==(const error_condition& __x, const error_code& __y) _NOEXCEPT |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 575 | { |
| 576 | return __y == __x; |
| 577 | } |
| 578 | |
| 579 | inline _LIBCPP_INLINE_VISIBILITY |
| 580 | bool |
Howard Hinnant | 1e15fd1 | 2011-05-26 19:48:01 +0000 | [diff] [blame] | 581 | operator==(const error_condition& __x, const error_condition& __y) _NOEXCEPT |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 582 | { |
| 583 | return __x.category() == __y.category() && __x.value() == __y.value(); |
| 584 | } |
| 585 | |
| 586 | inline _LIBCPP_INLINE_VISIBILITY |
| 587 | bool |
Howard Hinnant | 1e15fd1 | 2011-05-26 19:48:01 +0000 | [diff] [blame] | 588 | operator!=(const error_code& __x, const error_code& __y) _NOEXCEPT |
| 589 | {return !(__x == __y);} |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 590 | |
| 591 | inline _LIBCPP_INLINE_VISIBILITY |
| 592 | bool |
Howard Hinnant | 1e15fd1 | 2011-05-26 19:48:01 +0000 | [diff] [blame] | 593 | operator!=(const error_code& __x, const error_condition& __y) _NOEXCEPT |
| 594 | {return !(__x == __y);} |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 595 | |
| 596 | inline _LIBCPP_INLINE_VISIBILITY |
| 597 | bool |
Howard Hinnant | 1e15fd1 | 2011-05-26 19:48:01 +0000 | [diff] [blame] | 598 | operator!=(const error_condition& __x, const error_code& __y) _NOEXCEPT |
| 599 | {return !(__x == __y);} |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 600 | |
| 601 | inline _LIBCPP_INLINE_VISIBILITY |
| 602 | bool |
Howard Hinnant | 1e15fd1 | 2011-05-26 19:48:01 +0000 | [diff] [blame] | 603 | operator!=(const error_condition& __x, const error_condition& __y) _NOEXCEPT |
| 604 | {return !(__x == __y);} |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 605 | |
| 606 | template <> |
Howard Hinnant | 8d7a955 | 2010-09-23 17:31:07 +0000 | [diff] [blame] | 607 | struct _LIBCPP_VISIBLE hash<error_code> |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 608 | : public unary_function<error_code, size_t> |
| 609 | { |
Howard Hinnant | 8d7a955 | 2010-09-23 17:31:07 +0000 | [diff] [blame] | 610 | _LIBCPP_INLINE_VISIBILITY |
Howard Hinnant | 1e15fd1 | 2011-05-26 19:48:01 +0000 | [diff] [blame] | 611 | size_t operator()(const error_code& __ec) const _NOEXCEPT |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 612 | { |
| 613 | return static_cast<size_t>(__ec.value()); |
| 614 | } |
| 615 | }; |
| 616 | |
| 617 | // system_error |
| 618 | |
Howard Hinnant | 8d7a955 | 2010-09-23 17:31:07 +0000 | [diff] [blame] | 619 | class _LIBCPP_VISIBLE system_error |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 620 | : public runtime_error |
| 621 | { |
| 622 | error_code __ec_; |
| 623 | public: |
| 624 | system_error(error_code __ec, const string& __what_arg); |
| 625 | system_error(error_code __ec, const char* __what_arg); |
| 626 | system_error(error_code __ec); |
| 627 | system_error(int __ev, const error_category& __ecat, const string& __what_arg); |
| 628 | system_error(int __ev, const error_category& __ecat, const char* __what_arg); |
| 629 | system_error(int __ev, const error_category& __ecat); |
Howard Hinnant | 1e15fd1 | 2011-05-26 19:48:01 +0000 | [diff] [blame] | 630 | ~system_error() _NOEXCEPT; |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 631 | |
| 632 | _LIBCPP_ALWAYS_INLINE |
Howard Hinnant | 1e15fd1 | 2011-05-26 19:48:01 +0000 | [diff] [blame] | 633 | const error_code& code() const _NOEXCEPT {return __ec_;} |
Howard Hinnant | bc8d3f9 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 634 | |
| 635 | private: |
| 636 | static string __init(const error_code&, string); |
| 637 | }; |
| 638 | |
| 639 | void __throw_system_error(int ev, const char* what_arg); |
| 640 | |
| 641 | _LIBCPP_END_NAMESPACE_STD |
| 642 | |
| 643 | #endif // _LIBCPP_SYSTEM_ERROR |