/vendor/boost/libs/system/error_code.cpp

https://bitbucket.org/genericcontainer/goblin-camp/ · C++ · 426 lines · 357 code · 33 blank · 36 comment · 56 complexity · ac5038c3dc2909b5c3bf27141ca93225 MD5 · raw file

  1. // error_code support implementation file ----------------------------------//
  2. // Copyright Beman Dawes 2002, 2006
  3. // Distributed under the Boost Software License, Version 1.0. (See accompanying
  4. // file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
  5. // See library home page at http://www.boost.org/libs/system
  6. //----------------------------------------------------------------------------//
  7. #include <boost/config/warning_disable.hpp>
  8. // define BOOST_SYSTEM_SOURCE so that <boost/system/config.hpp> knows
  9. // the library is being built (possibly exporting rather than importing code)
  10. #define BOOST_SYSTEM_SOURCE
  11. #include <boost/system/config.hpp>
  12. #include <boost/system/error_code.hpp>
  13. #include <boost/cerrno.hpp>
  14. #include <vector>
  15. #include <cstdlib>
  16. #include <cassert>
  17. using namespace boost::system;
  18. using namespace boost::system::errc;
  19. #include <cstring> // for strerror/strerror_r
  20. # if defined( BOOST_WINDOWS_API )
  21. # include <windows.h>
  22. # include "local_free_on_destruction.hpp"
  23. # ifndef ERROR_INCORRECT_SIZE
  24. # define ERROR_INCORRECT_SIZE ERROR_BAD_ARGUMENTS
  25. # endif
  26. # endif
  27. //----------------------------------------------------------------------------//
  28. namespace
  29. {
  30. // standard error categories ---------------------------------------------//
  31. class generic_error_category : public error_category
  32. {
  33. public:
  34. generic_error_category(){}
  35. const char * name() const;
  36. std::string message( int ev ) const;
  37. };
  38. class system_error_category : public error_category
  39. {
  40. public:
  41. system_error_category(){}
  42. const char * name() const;
  43. std::string message( int ev ) const;
  44. error_condition default_error_condition( int ev ) const;
  45. };
  46. // generic_error_category implementation ---------------------------------//
  47. const char * generic_error_category::name() const
  48. {
  49. return "generic";
  50. }
  51. std::string generic_error_category::message( int ev ) const
  52. {
  53. static std::string unknown_err( "Unknown error" );
  54. // strerror_r is preferred because it is always thread safe,
  55. // however, we fallback to strerror in certain cases because:
  56. // -- Windows doesn't provide strerror_r.
  57. // -- HP and Sundo provide strerror_r on newer systems, but there is
  58. // no way to tell if is available at runtime and in any case their
  59. // versions of strerror are thread safe anyhow.
  60. // -- Linux only sometimes provides strerror_r.
  61. // -- Tru64 provides strerror_r only when compiled -pthread.
  62. // -- VMS doesn't provide strerror_r, but on this platform, strerror is
  63. // thread safe.
  64. # if defined(BOOST_WINDOWS_API) || defined(__hpux) || defined(__sun)\
  65. || (defined(__linux) && (!defined(__USE_XOPEN2K) || defined(BOOST_SYSTEM_USE_STRERROR)))\
  66. || (defined(__osf__) && !defined(_REENTRANT))\
  67. || (defined(__vms))\
  68. || (defined(__QNXNTO__))
  69. const char * c_str = std::strerror( ev );
  70. return c_str
  71. ? std::string( c_str )
  72. : unknown_err;
  73. # else // use strerror_r
  74. char buf[64];
  75. char * bp = buf;
  76. std::size_t sz = sizeof(buf);
  77. # if defined(__CYGWIN__) || defined(__USE_GNU)
  78. // Oddball version of strerror_r
  79. const char * c_str = strerror_r( ev, bp, sz );
  80. return c_str
  81. ? std::string( c_str )
  82. : unknown_err;
  83. # else
  84. // POSIX version of strerror_r
  85. int result;
  86. for (;;)
  87. {
  88. // strerror_r returns 0 on success, otherwise ERANGE if buffer too small,
  89. // invalid_argument if ev not a valid error number
  90. # if defined (__sgi)
  91. const char * c_str = strerror( ev );
  92. result = 0;
  93. return c_str
  94. ? std::string( c_str )
  95. : unknown_err;
  96. # else
  97. result = strerror_r( ev, bp, sz );
  98. # endif
  99. if (result == 0 )
  100. break;
  101. else
  102. {
  103. # if defined(__linux)
  104. // Linux strerror_r returns -1 on error, with error number in errno
  105. result = errno;
  106. # endif
  107. if ( result != ERANGE ) break;
  108. if ( sz > sizeof(buf) ) std::free( bp );
  109. sz *= 2;
  110. if ( (bp = static_cast<char*>(std::malloc( sz ))) == 0 )
  111. return std::string( "ENOMEM" );
  112. }
  113. }
  114. std::string msg;
  115. try
  116. {
  117. msg = ( ( result == invalid_argument ) ? "Unknown error" : bp );
  118. }
  119. # ifndef BOOST_NO_EXCEPTIONS
  120. // See ticket #2098
  121. catch(...)
  122. {
  123. // just eat the exception
  124. }
  125. # endif
  126. if ( sz > sizeof(buf) ) std::free( bp );
  127. sz = 0;
  128. return msg;
  129. # endif // else POSIX version of strerror_r
  130. # endif // else use strerror_r
  131. }
  132. // system_error_category implementation --------------------------------//
  133. const char * system_error_category::name() const
  134. {
  135. return "system";
  136. }
  137. error_condition system_error_category::default_error_condition( int ev ) const
  138. {
  139. switch ( ev )
  140. {
  141. case 0: return make_error_condition( success );
  142. # if defined(BOOST_POSIX_API)
  143. // POSIX-like O/S -> posix_errno decode table ---------------------------//
  144. case E2BIG: return make_error_condition( argument_list_too_long );
  145. case EACCES: return make_error_condition( permission_denied );
  146. case EADDRINUSE: return make_error_condition( address_in_use );
  147. case EADDRNOTAVAIL: return make_error_condition( address_not_available );
  148. case EAFNOSUPPORT: return make_error_condition( address_family_not_supported );
  149. case EAGAIN: return make_error_condition( resource_unavailable_try_again );
  150. # if EALREADY != EBUSY // EALREADY and EBUSY are the same on QNX Neutrino
  151. case EALREADY: return make_error_condition( connection_already_in_progress );
  152. # endif
  153. case EBADF: return make_error_condition( bad_file_descriptor );
  154. case EBADMSG: return make_error_condition( bad_message );
  155. case EBUSY: return make_error_condition( device_or_resource_busy );
  156. case ECANCELED: return make_error_condition( operation_canceled );
  157. case ECHILD: return make_error_condition( no_child_process );
  158. case ECONNABORTED: return make_error_condition( connection_aborted );
  159. case ECONNREFUSED: return make_error_condition( connection_refused );
  160. case ECONNRESET: return make_error_condition( connection_reset );
  161. case EDEADLK: return make_error_condition( resource_deadlock_would_occur );
  162. case EDESTADDRREQ: return make_error_condition( destination_address_required );
  163. case EDOM: return make_error_condition( argument_out_of_domain );
  164. case EEXIST: return make_error_condition( file_exists );
  165. case EFAULT: return make_error_condition( bad_address );
  166. case EFBIG: return make_error_condition( file_too_large );
  167. case EHOSTUNREACH: return make_error_condition( host_unreachable );
  168. case EIDRM: return make_error_condition( identifier_removed );
  169. case EILSEQ: return make_error_condition( illegal_byte_sequence );
  170. case EINPROGRESS: return make_error_condition( operation_in_progress );
  171. case EINTR: return make_error_condition( interrupted );
  172. case EINVAL: return make_error_condition( invalid_argument );
  173. case EIO: return make_error_condition( io_error );
  174. case EISCONN: return make_error_condition( already_connected );
  175. case EISDIR: return make_error_condition( is_a_directory );
  176. case ELOOP: return make_error_condition( too_many_symbolic_link_levels );
  177. case EMFILE: return make_error_condition( too_many_files_open );
  178. case EMLINK: return make_error_condition( too_many_links );
  179. case EMSGSIZE: return make_error_condition( message_size );
  180. case ENAMETOOLONG: return make_error_condition( filename_too_long );
  181. case ENETDOWN: return make_error_condition( network_down );
  182. case ENETRESET: return make_error_condition( network_reset );
  183. case ENETUNREACH: return make_error_condition( network_unreachable );
  184. case ENFILE: return make_error_condition( too_many_files_open_in_system );
  185. case ENOBUFS: return make_error_condition( no_buffer_space );
  186. case ENODATA: return make_error_condition( no_message_available );
  187. case ENODEV: return make_error_condition( no_such_device );
  188. case ENOENT: return make_error_condition( no_such_file_or_directory );
  189. case ENOEXEC: return make_error_condition( executable_format_error );
  190. case ENOLCK: return make_error_condition( no_lock_available );
  191. case ENOLINK: return make_error_condition( no_link );
  192. case ENOMEM: return make_error_condition( not_enough_memory );
  193. case ENOMSG: return make_error_condition( no_message );
  194. case ENOPROTOOPT: return make_error_condition( no_protocol_option );
  195. case ENOSPC: return make_error_condition( no_space_on_device );
  196. case ENOSR: return make_error_condition( no_stream_resources );
  197. case ENOSTR: return make_error_condition( not_a_stream );
  198. case ENOSYS: return make_error_condition( function_not_supported );
  199. case ENOTCONN: return make_error_condition( not_connected );
  200. case ENOTDIR: return make_error_condition( not_a_directory );
  201. # if ENOTEMPTY != EEXIST // AIX treats ENOTEMPTY and EEXIST as the same value
  202. case ENOTEMPTY: return make_error_condition( directory_not_empty );
  203. # endif // ENOTEMPTY != EEXIST
  204. # if ENOTRECOVERABLE != ECONNRESET // the same on some Broadcom chips
  205. case ENOTRECOVERABLE: return make_error_condition( state_not_recoverable );
  206. # endif // ENOTRECOVERABLE != ECONNRESET
  207. case ENOTSOCK: return make_error_condition( not_a_socket );
  208. case ENOTSUP: return make_error_condition( not_supported );
  209. case ENOTTY: return make_error_condition( inappropriate_io_control_operation );
  210. case ENXIO: return make_error_condition( no_such_device_or_address );
  211. # if EOPNOTSUPP != ENOTSUP
  212. case EOPNOTSUPP: return make_error_condition( operation_not_supported );
  213. # endif // EOPNOTSUPP != ENOTSUP
  214. case EOVERFLOW: return make_error_condition( value_too_large );
  215. # if EOWNERDEAD != ECONNABORTED // the same on some Broadcom chips
  216. case EOWNERDEAD: return make_error_condition( owner_dead );
  217. # endif // EOWNERDEAD != ECONNABORTED
  218. case EPERM: return make_error_condition( operation_not_permitted );
  219. case EPIPE: return make_error_condition( broken_pipe );
  220. case EPROTO: return make_error_condition( protocol_error );
  221. case EPROTONOSUPPORT: return make_error_condition( protocol_not_supported );
  222. case EPROTOTYPE: return make_error_condition( wrong_protocol_type );
  223. case ERANGE: return make_error_condition( result_out_of_range );
  224. case EROFS: return make_error_condition( read_only_file_system );
  225. case ESPIPE: return make_error_condition( invalid_seek );
  226. case ESRCH: return make_error_condition( no_such_process );
  227. case ETIME: return make_error_condition( stream_timeout );
  228. case ETIMEDOUT: return make_error_condition( timed_out );
  229. case ETXTBSY: return make_error_condition( text_file_busy );
  230. # if EAGAIN != EWOULDBLOCK
  231. case EWOULDBLOCK: return make_error_condition( operation_would_block );
  232. # endif // EAGAIN != EWOULDBLOCK
  233. case EXDEV: return make_error_condition( cross_device_link );
  234. #else
  235. // Windows system -> posix_errno decode table ---------------------------//
  236. // see WinError.h comments for descriptions of errors
  237. case ERROR_ACCESS_DENIED: return make_error_condition( permission_denied );
  238. case ERROR_ALREADY_EXISTS: return make_error_condition( file_exists );
  239. case ERROR_BAD_UNIT: return make_error_condition( no_such_device );
  240. case ERROR_BUFFER_OVERFLOW: return make_error_condition( filename_too_long );
  241. case ERROR_BUSY: return make_error_condition( device_or_resource_busy );
  242. case ERROR_BUSY_DRIVE: return make_error_condition( device_or_resource_busy );
  243. case ERROR_CANNOT_MAKE: return make_error_condition( permission_denied );
  244. case ERROR_CANTOPEN: return make_error_condition( io_error );
  245. case ERROR_CANTREAD: return make_error_condition( io_error );
  246. case ERROR_CANTWRITE: return make_error_condition( io_error );
  247. case ERROR_CURRENT_DIRECTORY: return make_error_condition( permission_denied );
  248. case ERROR_DEV_NOT_EXIST: return make_error_condition( no_such_device );
  249. case ERROR_DEVICE_IN_USE: return make_error_condition( device_or_resource_busy );
  250. case ERROR_DIR_NOT_EMPTY: return make_error_condition( directory_not_empty );
  251. case ERROR_DIRECTORY: return make_error_condition( invalid_argument ); // WinError.h: "The directory name is invalid"
  252. case ERROR_DISK_FULL: return make_error_condition( no_space_on_device );
  253. case ERROR_FILE_EXISTS: return make_error_condition( file_exists );
  254. case ERROR_FILE_NOT_FOUND: return make_error_condition( no_such_file_or_directory );
  255. case ERROR_HANDLE_DISK_FULL: return make_error_condition( no_space_on_device );
  256. case ERROR_INVALID_ACCESS: return make_error_condition( permission_denied );
  257. case ERROR_INVALID_DRIVE: return make_error_condition( no_such_device );
  258. case ERROR_INVALID_FUNCTION: return make_error_condition( function_not_supported );
  259. case ERROR_INVALID_HANDLE: return make_error_condition( invalid_argument );
  260. case ERROR_INVALID_NAME: return make_error_condition( invalid_argument );
  261. case ERROR_LOCK_VIOLATION: return make_error_condition( no_lock_available );
  262. case ERROR_LOCKED: return make_error_condition( no_lock_available );
  263. case ERROR_NEGATIVE_SEEK: return make_error_condition( invalid_argument );
  264. case ERROR_NOACCESS: return make_error_condition( permission_denied );
  265. case ERROR_NOT_ENOUGH_MEMORY: return make_error_condition( not_enough_memory );
  266. case ERROR_NOT_READY: return make_error_condition( resource_unavailable_try_again );
  267. case ERROR_NOT_SAME_DEVICE: return make_error_condition( cross_device_link );
  268. case ERROR_OPEN_FAILED: return make_error_condition( io_error );
  269. case ERROR_OPEN_FILES: return make_error_condition( device_or_resource_busy );
  270. case ERROR_OPERATION_ABORTED: return make_error_condition( operation_canceled );
  271. case ERROR_OUTOFMEMORY: return make_error_condition( not_enough_memory );
  272. case ERROR_PATH_NOT_FOUND: return make_error_condition( no_such_file_or_directory );
  273. case ERROR_READ_FAULT: return make_error_condition( io_error );
  274. case ERROR_RETRY: return make_error_condition( resource_unavailable_try_again );
  275. case ERROR_SEEK: return make_error_condition( io_error );
  276. case ERROR_SHARING_VIOLATION: return make_error_condition( permission_denied );
  277. case ERROR_TOO_MANY_OPEN_FILES: return make_error_condition( too_many_files_open );
  278. case ERROR_WRITE_FAULT: return make_error_condition( io_error );
  279. case ERROR_WRITE_PROTECT: return make_error_condition( permission_denied );
  280. case WSAEACCES: return make_error_condition( permission_denied );
  281. case WSAEADDRINUSE: return make_error_condition( address_in_use );
  282. case WSAEADDRNOTAVAIL: return make_error_condition( address_not_available );
  283. case WSAEAFNOSUPPORT: return make_error_condition( address_family_not_supported );
  284. case WSAEALREADY: return make_error_condition( connection_already_in_progress );
  285. case WSAEBADF: return make_error_condition( bad_file_descriptor );
  286. case WSAECONNABORTED: return make_error_condition( connection_aborted );
  287. case WSAECONNREFUSED: return make_error_condition( connection_refused );
  288. case WSAECONNRESET: return make_error_condition( connection_reset );
  289. case WSAEDESTADDRREQ: return make_error_condition( destination_address_required );
  290. case WSAEFAULT: return make_error_condition( bad_address );
  291. case WSAEHOSTUNREACH: return make_error_condition( host_unreachable );
  292. case WSAEINPROGRESS: return make_error_condition( operation_in_progress );
  293. case WSAEINTR: return make_error_condition( interrupted );
  294. case WSAEINVAL: return make_error_condition( invalid_argument );
  295. case WSAEISCONN: return make_error_condition( already_connected );
  296. case WSAEMFILE: return make_error_condition( too_many_files_open );
  297. case WSAEMSGSIZE: return make_error_condition( message_size );
  298. case WSAENAMETOOLONG: return make_error_condition( filename_too_long );
  299. case WSAENETDOWN: return make_error_condition( network_down );
  300. case WSAENETRESET: return make_error_condition( network_reset );
  301. case WSAENETUNREACH: return make_error_condition( network_unreachable );
  302. case WSAENOBUFS: return make_error_condition( no_buffer_space );
  303. case WSAENOPROTOOPT: return make_error_condition( no_protocol_option );
  304. case WSAENOTCONN: return make_error_condition( not_connected );
  305. case WSAENOTSOCK: return make_error_condition( not_a_socket );
  306. case WSAEOPNOTSUPP: return make_error_condition( operation_not_supported );
  307. case WSAEPROTONOSUPPORT: return make_error_condition( protocol_not_supported );
  308. case WSAEPROTOTYPE: return make_error_condition( wrong_protocol_type );
  309. case WSAETIMEDOUT: return make_error_condition( timed_out );
  310. case WSAEWOULDBLOCK: return make_error_condition( operation_would_block );
  311. #endif
  312. default: return error_condition( ev, system_category() );
  313. }
  314. }
  315. # if !defined( BOOST_WINDOWS_API )
  316. std::string system_error_category::message( int ev ) const
  317. {
  318. return generic_category().message( ev );
  319. }
  320. # else
  321. std::string system_error_category::message( int ev ) const
  322. {
  323. # ifndef BOOST_NO_ANSI_APIS
  324. LPVOID lpMsgBuf = 0;
  325. DWORD retval = ::FormatMessageA(
  326. FORMAT_MESSAGE_ALLOCATE_BUFFER |
  327. FORMAT_MESSAGE_FROM_SYSTEM |
  328. FORMAT_MESSAGE_IGNORE_INSERTS,
  329. NULL,
  330. ev,
  331. MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT), // Default language
  332. (LPSTR) &lpMsgBuf,
  333. 0,
  334. NULL
  335. );
  336. detail::local_free_on_destruction lfod(lpMsgBuf);
  337. if (retval == 0)
  338. return std::string("Unknown error");
  339. std::string str( static_cast<LPCSTR>(lpMsgBuf) );
  340. # else // WinCE workaround
  341. LPVOID lpMsgBuf = 0;
  342. DWORD retval = ::FormatMessageW(
  343. FORMAT_MESSAGE_ALLOCATE_BUFFER |
  344. FORMAT_MESSAGE_FROM_SYSTEM |
  345. FORMAT_MESSAGE_IGNORE_INSERTS,
  346. NULL,
  347. ev,
  348. MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT), // Default language
  349. (LPWSTR) &lpMsgBuf,
  350. 0,
  351. NULL
  352. );
  353. detail::local_free_on_destruction lfod(lpMsgBuf);
  354. if (retval == 0)
  355. return std::string("Unknown error");
  356. int num_chars = (wcslen( static_cast<LPCWSTR>(lpMsgBuf) ) + 1) * 2;
  357. LPSTR narrow_buffer = (LPSTR)_alloca( num_chars );
  358. if (::WideCharToMultiByte(CP_ACP, 0, static_cast<LPCWSTR>(lpMsgBuf), -1, narrow_buffer, num_chars, NULL, NULL) == 0)
  359. return std::string("Unknown error");
  360. std::string str( narrow_buffer );
  361. # endif
  362. while ( str.size()
  363. && (str[str.size()-1] == '\n' || str[str.size()-1] == '\r') )
  364. str.erase( str.size()-1 );
  365. if ( str.size() && str[str.size()-1] == '.' )
  366. { str.erase( str.size()-1 ); }
  367. return str;
  368. }
  369. # endif
  370. } // unnamed namespace
  371. namespace boost
  372. {
  373. namespace system
  374. {
  375. # ifndef BOOST_SYSTEM_NO_DEPRECATED
  376. BOOST_SYSTEM_DECL error_code throws; // "throw on error" special error_code;
  377. // note that it doesn't matter if this
  378. // isn't initialized before use since
  379. // the only use is to take its
  380. // address for comparison purposes
  381. # endif
  382. BOOST_SYSTEM_DECL const error_category & system_category()
  383. {
  384. static const system_error_category system_category_const;
  385. return system_category_const;
  386. }
  387. BOOST_SYSTEM_DECL const error_category & generic_category()
  388. {
  389. static const generic_error_category generic_category_const;
  390. return generic_category_const;
  391. }
  392. } // namespace system
  393. } // namespace boost