error_code.hpp 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520
  1. // boost/system/error_code.hpp ---------------------------------------------//
  2. // Copyright Beman Dawes 2006, 2007
  3. // Copyright Christoper Kohlhoff 2007
  4. // Distributed under the Boost Software License, Version 1.0. (See accompanying
  5. // file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
  6. // See library home page at http://www.boost.org/libs/system
  7. #ifndef BOOST_ERROR_CODE_HPP
  8. #define BOOST_ERROR_CODE_HPP
  9. #include <boost/system/config.hpp>
  10. #include <boost/cstdint.hpp>
  11. #include <boost/assert.hpp>
  12. #include <boost/noncopyable.hpp>
  13. #include <boost/utility/enable_if.hpp>
  14. #include <ostream>
  15. #include <string>
  16. #include <stdexcept>
  17. #include <functional>
  18. // TODO: undef these macros if not already defined
  19. #include <boost/cerrno.hpp>
  20. #if !defined(BOOST_POSIX_API) && !defined(BOOST_WINDOWS_API)
  21. # error BOOST_POSIX_API or BOOST_WINDOWS_API must be defined
  22. #endif
  23. #include <boost/config/abi_prefix.hpp> // must be the last #include
  24. #ifndef BOOST_SYSTEM_NOEXCEPT
  25. #define BOOST_SYSTEM_NOEXCEPT BOOST_NOEXCEPT
  26. #endif
  27. namespace boost
  28. {
  29. namespace system
  30. {
  31. class error_code;
  32. class error_condition;
  33. // "Concept" helpers ---------------------------------------------------//
  34. template< class T >
  35. struct is_error_code_enum { static const bool value = false; };
  36. template< class T >
  37. struct is_error_condition_enum { static const bool value = false; };
  38. // generic error_conditions --------------------------------------------//
  39. namespace errc
  40. {
  41. enum errc_t
  42. {
  43. success = 0,
  44. address_family_not_supported = EAFNOSUPPORT,
  45. address_in_use = EADDRINUSE,
  46. address_not_available = EADDRNOTAVAIL,
  47. already_connected = EISCONN,
  48. argument_list_too_long = E2BIG,
  49. argument_out_of_domain = EDOM,
  50. bad_address = EFAULT,
  51. bad_file_descriptor = EBADF,
  52. bad_message = EBADMSG,
  53. broken_pipe = EPIPE,
  54. connection_aborted = ECONNABORTED,
  55. connection_already_in_progress = EALREADY,
  56. connection_refused = ECONNREFUSED,
  57. connection_reset = ECONNRESET,
  58. cross_device_link = EXDEV,
  59. destination_address_required = EDESTADDRREQ,
  60. device_or_resource_busy = EBUSY,
  61. directory_not_empty = ENOTEMPTY,
  62. executable_format_error = ENOEXEC,
  63. file_exists = EEXIST,
  64. file_too_large = EFBIG,
  65. filename_too_long = ENAMETOOLONG,
  66. function_not_supported = ENOSYS,
  67. host_unreachable = EHOSTUNREACH,
  68. identifier_removed = EIDRM,
  69. illegal_byte_sequence = EILSEQ,
  70. inappropriate_io_control_operation = ENOTTY,
  71. interrupted = EINTR,
  72. invalid_argument = EINVAL,
  73. invalid_seek = ESPIPE,
  74. io_error = EIO,
  75. is_a_directory = EISDIR,
  76. message_size = EMSGSIZE,
  77. network_down = ENETDOWN,
  78. network_reset = ENETRESET,
  79. network_unreachable = ENETUNREACH,
  80. no_buffer_space = ENOBUFS,
  81. no_child_process = ECHILD,
  82. no_link = ENOLINK,
  83. no_lock_available = ENOLCK,
  84. no_message_available = ENODATA,
  85. no_message = ENOMSG,
  86. no_protocol_option = ENOPROTOOPT,
  87. no_space_on_device = ENOSPC,
  88. no_stream_resources = ENOSR,
  89. no_such_device_or_address = ENXIO,
  90. no_such_device = ENODEV,
  91. no_such_file_or_directory = ENOENT,
  92. no_such_process = ESRCH,
  93. not_a_directory = ENOTDIR,
  94. not_a_socket = ENOTSOCK,
  95. not_a_stream = ENOSTR,
  96. not_connected = ENOTCONN,
  97. not_enough_memory = ENOMEM,
  98. not_supported = ENOTSUP,
  99. operation_canceled = ECANCELED,
  100. operation_in_progress = EINPROGRESS,
  101. operation_not_permitted = EPERM,
  102. operation_not_supported = EOPNOTSUPP,
  103. operation_would_block = EWOULDBLOCK,
  104. owner_dead = EOWNERDEAD,
  105. permission_denied = EACCES,
  106. protocol_error = EPROTO,
  107. protocol_not_supported = EPROTONOSUPPORT,
  108. read_only_file_system = EROFS,
  109. resource_deadlock_would_occur = EDEADLK,
  110. resource_unavailable_try_again = EAGAIN,
  111. result_out_of_range = ERANGE,
  112. state_not_recoverable = ENOTRECOVERABLE,
  113. stream_timeout = ETIME,
  114. text_file_busy = ETXTBSY,
  115. timed_out = ETIMEDOUT,
  116. too_many_files_open_in_system = ENFILE,
  117. too_many_files_open = EMFILE,
  118. too_many_links = EMLINK,
  119. too_many_symbolic_link_levels = ELOOP,
  120. value_too_large = EOVERFLOW,
  121. wrong_protocol_type = EPROTOTYPE
  122. };
  123. } // namespace errc
  124. # ifndef BOOST_SYSTEM_NO_DEPRECATED
  125. namespace posix = errc;
  126. namespace posix_error = errc;
  127. # endif
  128. template<> struct is_error_condition_enum<errc::errc_t>
  129. { static const bool value = true; };
  130. // ----------------------------------------------------------------------//
  131. // Operating system specific interfaces --------------------------------//
  132. // The interface is divided into general and system-specific portions to
  133. // meet these requirements:
  134. //
  135. // * Code calling an operating system API can create an error_code with
  136. // a single category (system_category), even for POSIX-like operating
  137. // systems that return some POSIX errno values and some native errno
  138. // values. This code should not have to pay the cost of distinguishing
  139. // between categories, since it is not yet known if that is needed.
  140. //
  141. // * Users wishing to write system-specific code should be given enums for
  142. // at least the common error cases.
  143. //
  144. // * System specific code should fail at compile time if moved to another
  145. // operating system.
  146. // The system specific portions of the interface are located in headers
  147. // with names reflecting the operating system. For example,
  148. //
  149. // <boost/system/cygwin_error.hpp>
  150. // <boost/system/linux_error.hpp>
  151. // <boost/system/windows_error.hpp>
  152. //
  153. // These headers are effectively empty for compiles on operating systems
  154. // where they are not applicable.
  155. // ----------------------------------------------------------------------//
  156. // class error_category ------------------------------------------------//
  157. class error_category : public noncopyable
  158. {
  159. public:
  160. virtual ~error_category(){}
  161. virtual const char * name() const BOOST_SYSTEM_NOEXCEPT = 0;
  162. virtual std::string message( int ev ) const = 0;
  163. inline virtual error_condition default_error_condition( int ev ) const BOOST_SYSTEM_NOEXCEPT;
  164. inline virtual bool equivalent( int code,
  165. const error_condition & condition ) const BOOST_SYSTEM_NOEXCEPT;
  166. inline virtual bool equivalent( const error_code & code,
  167. int condition ) const BOOST_SYSTEM_NOEXCEPT;
  168. bool operator==(const error_category & rhs) const BOOST_SYSTEM_NOEXCEPT { return this == &rhs; }
  169. bool operator!=(const error_category & rhs) const BOOST_SYSTEM_NOEXCEPT { return this != &rhs; }
  170. bool operator<( const error_category & rhs ) const BOOST_SYSTEM_NOEXCEPT
  171. {
  172. return std::less<const error_category*>()( this, &rhs );
  173. }
  174. };
  175. // predefined error categories -----------------------------------------//
  176. # ifdef BOOST_ERROR_CODE_HEADER_ONLY
  177. inline const error_category & system_category() BOOST_SYSTEM_NOEXCEPT;
  178. inline const error_category & generic_category() BOOST_SYSTEM_NOEXCEPT;
  179. #else
  180. BOOST_SYSTEM_DECL const error_category & system_category() BOOST_SYSTEM_NOEXCEPT;
  181. BOOST_SYSTEM_DECL const error_category & generic_category() BOOST_SYSTEM_NOEXCEPT;
  182. #endif
  183. // deprecated synonyms --------------------------------------------------//
  184. # ifndef BOOST_SYSTEM_NO_DEPRECATED
  185. inline const error_category & get_system_category() { return system_category(); }
  186. inline const error_category & get_generic_category() { return generic_category(); }
  187. inline const error_category & get_posix_category() { return generic_category(); }
  188. static const error_category & posix_category = generic_category();
  189. static const error_category & errno_ecat = generic_category();
  190. static const error_category & native_ecat = system_category();
  191. # endif
  192. // class error_condition -----------------------------------------------//
  193. // error_conditions are portable, error_codes are system or library specific
  194. class error_condition
  195. {
  196. public:
  197. // constructors:
  198. error_condition() BOOST_SYSTEM_NOEXCEPT : m_val(0), m_cat(&generic_category()) {}
  199. error_condition( int val, const error_category & cat ) BOOST_SYSTEM_NOEXCEPT : m_val(val), m_cat(&cat) {}
  200. template <class ErrorConditionEnum>
  201. error_condition(ErrorConditionEnum e,
  202. typename boost::enable_if<is_error_condition_enum<ErrorConditionEnum> >::type* = 0) BOOST_SYSTEM_NOEXCEPT
  203. {
  204. *this = make_error_condition(e);
  205. }
  206. // modifiers:
  207. void assign( int val, const error_category & cat ) BOOST_SYSTEM_NOEXCEPT
  208. {
  209. m_val = val;
  210. m_cat = &cat;
  211. }
  212. template<typename ErrorConditionEnum>
  213. typename boost::enable_if<is_error_condition_enum<ErrorConditionEnum>, error_condition>::type &
  214. operator=( ErrorConditionEnum val ) BOOST_SYSTEM_NOEXCEPT
  215. {
  216. *this = make_error_condition(val);
  217. return *this;
  218. }
  219. void clear() BOOST_SYSTEM_NOEXCEPT
  220. {
  221. m_val = 0;
  222. m_cat = &generic_category();
  223. }
  224. // observers:
  225. int value() const BOOST_SYSTEM_NOEXCEPT { return m_val; }
  226. const error_category & category() const BOOST_SYSTEM_NOEXCEPT { return *m_cat; }
  227. std::string message() const { return m_cat->message(value()); }
  228. typedef void (*unspecified_bool_type)();
  229. static void unspecified_bool_true() {}
  230. operator unspecified_bool_type() const BOOST_SYSTEM_NOEXCEPT // true if error
  231. {
  232. return m_val == 0 ? 0 : unspecified_bool_true;
  233. }
  234. bool operator!() const BOOST_SYSTEM_NOEXCEPT // true if no error
  235. {
  236. return m_val == 0;
  237. }
  238. // relationals:
  239. // the more symmetrical non-member syntax allows enum
  240. // conversions work for both rhs and lhs.
  241. inline friend bool operator==( const error_condition & lhs,
  242. const error_condition & rhs ) BOOST_SYSTEM_NOEXCEPT
  243. {
  244. return lhs.m_cat == rhs.m_cat && lhs.m_val == rhs.m_val;
  245. }
  246. inline friend bool operator<( const error_condition & lhs,
  247. const error_condition & rhs ) BOOST_SYSTEM_NOEXCEPT
  248. // the more symmetrical non-member syntax allows enum
  249. // conversions work for both rhs and lhs.
  250. {
  251. return lhs.m_cat < rhs.m_cat
  252. || (lhs.m_cat == rhs.m_cat && lhs.m_val < rhs.m_val);
  253. }
  254. private:
  255. int m_val;
  256. const error_category * m_cat;
  257. };
  258. // class error_code ----------------------------------------------------//
  259. // We want error_code to be a value type that can be copied without slicing
  260. // and without requiring heap allocation, but we also want it to have
  261. // polymorphic behavior based on the error category. This is achieved by
  262. // abstract base class error_category supplying the polymorphic behavior,
  263. // and error_code containing a pointer to an object of a type derived
  264. // from error_category.
  265. class error_code
  266. {
  267. public:
  268. // constructors:
  269. error_code() BOOST_SYSTEM_NOEXCEPT : m_val(0), m_cat(&system_category()) {}
  270. error_code( int val, const error_category & cat ) BOOST_SYSTEM_NOEXCEPT : m_val(val), m_cat(&cat) {}
  271. template <class ErrorCodeEnum>
  272. error_code(ErrorCodeEnum e,
  273. typename boost::enable_if<is_error_code_enum<ErrorCodeEnum> >::type* = 0) BOOST_SYSTEM_NOEXCEPT
  274. {
  275. *this = make_error_code(e);
  276. }
  277. // modifiers:
  278. void assign( int val, const error_category & cat ) BOOST_SYSTEM_NOEXCEPT
  279. {
  280. m_val = val;
  281. m_cat = &cat;
  282. }
  283. template<typename ErrorCodeEnum>
  284. typename boost::enable_if<is_error_code_enum<ErrorCodeEnum>, error_code>::type &
  285. operator=( ErrorCodeEnum val ) BOOST_SYSTEM_NOEXCEPT
  286. {
  287. *this = make_error_code(val);
  288. return *this;
  289. }
  290. void clear() BOOST_SYSTEM_NOEXCEPT
  291. {
  292. m_val = 0;
  293. m_cat = &system_category();
  294. }
  295. // observers:
  296. int value() const BOOST_SYSTEM_NOEXCEPT { return m_val; }
  297. const error_category & category() const BOOST_SYSTEM_NOEXCEPT { return *m_cat; }
  298. error_condition default_error_condition() const BOOST_SYSTEM_NOEXCEPT { return m_cat->default_error_condition(value()); }
  299. std::string message() const { return m_cat->message(value()); }
  300. typedef void (*unspecified_bool_type)();
  301. static void unspecified_bool_true() {}
  302. operator unspecified_bool_type() const BOOST_SYSTEM_NOEXCEPT // true if error
  303. {
  304. return m_val == 0 ? 0 : unspecified_bool_true;
  305. }
  306. bool operator!() const BOOST_SYSTEM_NOEXCEPT // true if no error
  307. {
  308. return m_val == 0;
  309. }
  310. // relationals:
  311. inline friend bool operator==( const error_code & lhs,
  312. const error_code & rhs ) BOOST_SYSTEM_NOEXCEPT
  313. // the more symmetrical non-member syntax allows enum
  314. // conversions work for both rhs and lhs.
  315. {
  316. return lhs.m_cat == rhs.m_cat && lhs.m_val == rhs.m_val;
  317. }
  318. inline friend bool operator<( const error_code & lhs,
  319. const error_code & rhs ) BOOST_SYSTEM_NOEXCEPT
  320. // the more symmetrical non-member syntax allows enum
  321. // conversions work for both rhs and lhs.
  322. {
  323. return lhs.m_cat < rhs.m_cat
  324. || (lhs.m_cat == rhs.m_cat && lhs.m_val < rhs.m_val);
  325. }
  326. private:
  327. int m_val;
  328. const error_category * m_cat;
  329. };
  330. // predefined error_code object used as "throw on error" tag
  331. # ifndef BOOST_SYSTEM_NO_DEPRECATED
  332. BOOST_SYSTEM_DECL extern error_code throws;
  333. # endif
  334. // Moving from a "throws" object to a "throws" function without breaking
  335. // existing code is a bit of a problem. The workaround is to place the
  336. // "throws" function in namespace boost rather than namespace boost::system.
  337. } // namespace system
  338. namespace detail { inline system::error_code * throws() { return 0; } }
  339. // Misuse of the error_code object is turned into a noisy failure by
  340. // poisoning the reference. This particular implementation doesn't
  341. // produce warnings or errors from popular compilers, is very efficient
  342. // (as determined by inspecting generated code), and does not suffer
  343. // from order of initialization problems. In practice, it also seems
  344. // cause user function error handling implementation errors to be detected
  345. // very early in the development cycle.
  346. inline system::error_code & throws()
  347. { return *detail::throws(); }
  348. namespace system
  349. {
  350. // non-member functions ------------------------------------------------//
  351. inline bool operator!=( const error_code & lhs,
  352. const error_code & rhs ) BOOST_SYSTEM_NOEXCEPT
  353. {
  354. return !(lhs == rhs);
  355. }
  356. inline bool operator!=( const error_condition & lhs,
  357. const error_condition & rhs ) BOOST_SYSTEM_NOEXCEPT
  358. {
  359. return !(lhs == rhs);
  360. }
  361. inline bool operator==( const error_code & code,
  362. const error_condition & condition ) BOOST_SYSTEM_NOEXCEPT
  363. {
  364. return code.category().equivalent( code.value(), condition )
  365. || condition.category().equivalent( code, condition.value() );
  366. }
  367. inline bool operator!=( const error_code & lhs,
  368. const error_condition & rhs ) BOOST_SYSTEM_NOEXCEPT
  369. {
  370. return !(lhs == rhs);
  371. }
  372. inline bool operator==( const error_condition & condition,
  373. const error_code & code ) BOOST_SYSTEM_NOEXCEPT
  374. {
  375. return condition.category().equivalent( code, condition.value() )
  376. || code.category().equivalent( code.value(), condition );
  377. }
  378. inline bool operator!=( const error_condition & lhs,
  379. const error_code & rhs ) BOOST_SYSTEM_NOEXCEPT
  380. {
  381. return !(lhs == rhs);
  382. }
  383. // TODO: both of these may move elsewhere, but the LWG hasn't spoken yet.
  384. template <class charT, class traits>
  385. inline std::basic_ostream<charT,traits>&
  386. operator<< (std::basic_ostream<charT,traits>& os, error_code ec)
  387. {
  388. os << ec.category().name() << ':' << ec.value();
  389. return os;
  390. }
  391. inline std::size_t hash_value( const error_code & ec )
  392. {
  393. return static_cast<std::size_t>(ec.value())
  394. + reinterpret_cast<std::size_t>(&ec.category());
  395. }
  396. // make_* functions for errc::errc_t -----------------------------//
  397. namespace errc
  398. {
  399. // explicit conversion:
  400. inline error_code make_error_code( errc_t e ) BOOST_SYSTEM_NOEXCEPT
  401. { return error_code( e, generic_category() ); }
  402. // implicit conversion:
  403. inline error_condition make_error_condition( errc_t e ) BOOST_SYSTEM_NOEXCEPT
  404. { return error_condition( e, generic_category() ); }
  405. }
  406. // error_category default implementation -------------------------------//
  407. error_condition error_category::default_error_condition( int ev ) const BOOST_SYSTEM_NOEXCEPT
  408. {
  409. return error_condition( ev, *this );
  410. }
  411. bool error_category::equivalent( int code,
  412. const error_condition & condition ) const BOOST_SYSTEM_NOEXCEPT
  413. {
  414. return default_error_condition( code ) == condition;
  415. }
  416. bool error_category::equivalent( const error_code & code,
  417. int condition ) const BOOST_SYSTEM_NOEXCEPT
  418. {
  419. return *this == code.category() && code.value() == condition;
  420. }
  421. } // namespace system
  422. } // namespace boost
  423. #include <boost/config/abi_suffix.hpp> // pops abi_prefix.hpp pragmas
  424. # ifdef BOOST_ERROR_CODE_HEADER_ONLY
  425. # include <boost/system/detail/error_code.ipp>
  426. # endif
  427. #endif // BOOST_ERROR_CODE_HPP