status_code_domain.hpp 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417
  1. /* Proposed SG14 status_code
  2. (C) 2018-2020 Niall Douglas <http://www.nedproductions.biz/> (5 commits)
  3. File Created: Feb 2018
  4. Boost Software License - Version 1.0 - August 17th, 2003
  5. Permission is hereby granted, free of charge, to any person or organization
  6. obtaining a copy of the software and accompanying documentation covered by
  7. this license (the "Software") to use, reproduce, display, distribute,
  8. execute, and transmit the Software, and to prepare derivative works of the
  9. Software, and to permit third-parties to whom the Software is furnished to
  10. do so, all subject to the following:
  11. The copyright notices in the Software and this entire statement, including
  12. the above license grant, this restriction and the following disclaimer,
  13. must be included in all copies of the Software, in whole or in part, and
  14. all derivative works of the Software, unless such copies or derivative
  15. works are solely in the form of machine-executable object code generated by
  16. a source language processor.
  17. THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
  18. IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
  19. FITNESS FOR A PARTICULAR PURPOSE, TITLE AND NON-INFRINGEMENT. IN NO EVENT
  20. SHALL THE COPYRIGHT HOLDERS OR ANYONE DISTRIBUTING THE SOFTWARE BE LIABLE
  21. FOR ANY DAMAGES OR OTHER LIABILITY, WHETHER IN CONTRACT, TORT OR OTHERWISE,
  22. ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
  23. DEALINGS IN THE SOFTWARE.
  24. */
  25. #ifndef BOOST_OUTCOME_SYSTEM_ERROR2_STATUS_CODE_DOMAIN_HPP
  26. #define BOOST_OUTCOME_SYSTEM_ERROR2_STATUS_CODE_DOMAIN_HPP
  27. #include "config.hpp"
  28. #include <cstring> // for strchr
  29. BOOST_OUTCOME_SYSTEM_ERROR2_NAMESPACE_BEGIN
  30. /*! The main workhorse of the system_error2 library, can be typed (`status_code<DomainType>`), erased-immutable (`status_code<void>`) or erased-mutable (`status_code<erased<T>>`).
  31. Be careful of placing these into containers! Equality and inequality operators are
  32. *semantic* not exact. Therefore two distinct items will test true! To help prevent
  33. surprise on this, `operator<` and `std::hash<>` are NOT implemented in order to
  34. trap potential incorrectness. Define your own custom comparison functions for your
  35. container which perform exact comparisons.
  36. */
  37. template <class DomainType> class status_code;
  38. class _generic_code_domain;
  39. //! The generic code is a status code with the generic code domain, which is that of `errc` (POSIX).
  40. using generic_code = status_code<_generic_code_domain>;
  41. namespace detail
  42. {
  43. template <class StatusCode> class indirecting_domain;
  44. template <class T> struct status_code_sizer
  45. {
  46. void *a;
  47. T b;
  48. };
  49. template <class To, class From> struct type_erasure_is_safe
  50. {
  51. static constexpr bool value = traits::is_move_bitcopying<From>::value //
  52. && (sizeof(status_code_sizer<From>) <= sizeof(status_code_sizer<To>));
  53. };
  54. /* We are severely limited by needing to retain C++ 11 compatibility when doing
  55. constexpr string parsing. MSVC lets you throw exceptions within a constexpr
  56. evaluation context when exceptions are globally disabled, but won't let you
  57. divide by zero, even if never evaluated, ever in constexpr. GCC and clang won't
  58. let you throw exceptions, ever, if exceptions are globally disabled. So let's
  59. use the trick of divide by zero in constexpr on GCC and clang if and only if
  60. exceptions are globally disabled.
  61. */
  62. #ifdef __GNUC__
  63. #pragma GCC diagnostic push
  64. #pragma GCC diagnostic ignored "-Wdiv-by-zero"
  65. #endif
  66. #if defined(__cpp_exceptions) || (defined(_MSC_VER) && !defined(__clang__))
  67. #define BOOST_OUTCOME_SYSTEM_ERROR2_FAIL_CONSTEXPR(msg) throw msg
  68. #else
  69. #define BOOST_OUTCOME_SYSTEM_ERROR2_FAIL_CONSTEXPR(msg) ((void) msg, 1 / 0)
  70. #endif
  71. constexpr inline unsigned long long parse_hex_byte(char c) { return ('0' <= c && c <= '9') ? (c - '0') : ('a' <= c && c <= 'f') ? (10 + c - 'a') : ('A' <= c && c <= 'F') ? (10 + c - 'A') : BOOST_OUTCOME_SYSTEM_ERROR2_FAIL_CONSTEXPR("Invalid character in UUID"); }
  72. constexpr inline unsigned long long parse_uuid2(const char *s)
  73. {
  74. return ((parse_hex_byte(s[0]) << 0) | (parse_hex_byte(s[1]) << 4) | (parse_hex_byte(s[2]) << 8) | (parse_hex_byte(s[3]) << 12) | (parse_hex_byte(s[4]) << 16) | (parse_hex_byte(s[5]) << 20) | (parse_hex_byte(s[6]) << 24) | (parse_hex_byte(s[7]) << 28) | (parse_hex_byte(s[9]) << 32) | (parse_hex_byte(s[10]) << 36) |
  75. (parse_hex_byte(s[11]) << 40) | (parse_hex_byte(s[12]) << 44) | (parse_hex_byte(s[14]) << 48) | (parse_hex_byte(s[15]) << 52) | (parse_hex_byte(s[16]) << 56) | (parse_hex_byte(s[17]) << 60)) //
  76. ^ //
  77. ((parse_hex_byte(s[19]) << 0) | (parse_hex_byte(s[20]) << 4) | (parse_hex_byte(s[21]) << 8) | (parse_hex_byte(s[22]) << 12) | (parse_hex_byte(s[24]) << 16) | (parse_hex_byte(s[25]) << 20) | (parse_hex_byte(s[26]) << 24) | (parse_hex_byte(s[27]) << 28) | (parse_hex_byte(s[28]) << 32) |
  78. (parse_hex_byte(s[29]) << 36) | (parse_hex_byte(s[30]) << 40) | (parse_hex_byte(s[31]) << 44) | (parse_hex_byte(s[32]) << 48) | (parse_hex_byte(s[33]) << 52) | (parse_hex_byte(s[34]) << 56) | (parse_hex_byte(s[35]) << 60));
  79. }
  80. template <size_t N> constexpr inline unsigned long long parse_uuid(const char (&uuid)[N]) { return (N == 37) ? parse_uuid2(uuid) : ((N == 39) ? parse_uuid2(uuid + 1) : BOOST_OUTCOME_SYSTEM_ERROR2_FAIL_CONSTEXPR("UUID does not have correct length")); }
  81. #ifdef __GNUC__
  82. #pragma GCC diagnostic pop
  83. #endif
  84. static constexpr unsigned long long test_uuid_parse = parse_uuid("430f1201-94fc-06c7-430f-120194fc06c7");
  85. //static constexpr unsigned long long test_uuid_parse2 = parse_uuid("x30f1201-94fc-06c7-430f-120194fc06c7");
  86. } // namespace detail
  87. /*! Abstract base class for a coding domain of a status code.
  88. */
  89. class status_code_domain
  90. {
  91. template <class DomainType> friend class status_code;
  92. template <class StatusCode> friend class indirecting_domain;
  93. public:
  94. //! Type of the unique id for this domain.
  95. using unique_id_type = unsigned long long;
  96. /*! (Potentially thread safe) Reference to a message string.
  97. Be aware that you cannot add payload to implementations of this class.
  98. You get exactly the `void *[3]` array to keep state, this is usually
  99. sufficient for a `std::shared_ptr<>` or a `std::string`.
  100. You can install a handler to be called when this object is copied,
  101. moved and destructed. This takes the form of a C function pointer.
  102. */
  103. class string_ref
  104. {
  105. public:
  106. //! The value type
  107. using value_type = const char;
  108. //! The size type
  109. using size_type = size_t;
  110. //! The pointer type
  111. using pointer = const char *;
  112. //! The const pointer type
  113. using const_pointer = const char *;
  114. //! The iterator type
  115. using iterator = const char *;
  116. //! The const iterator type
  117. using const_iterator = const char *;
  118. protected:
  119. //! The operation occurring
  120. enum class _thunk_op
  121. {
  122. copy,
  123. move,
  124. destruct
  125. };
  126. //! The prototype of the handler function. Copies can throw, moves and destructs cannot.
  127. using _thunk_spec = void (*)(string_ref *dest, const string_ref *src, _thunk_op op);
  128. #ifndef NDEBUG
  129. private:
  130. static void _checking_string_thunk(string_ref *dest, const string_ref *src, _thunk_op /*unused*/) noexcept
  131. {
  132. (void) dest;
  133. (void) src;
  134. assert(dest->_thunk == _checking_string_thunk); // NOLINT
  135. assert(src == nullptr || src->_thunk == _checking_string_thunk); // NOLINT
  136. // do nothing
  137. }
  138. protected:
  139. #endif
  140. //! Pointers to beginning and end of character range
  141. pointer _begin{}, _end{};
  142. //! Three `void*` of state
  143. void *_state[3]{}; // at least the size of a shared_ptr
  144. //! Handler for when operations occur
  145. const _thunk_spec _thunk{nullptr};
  146. constexpr explicit string_ref(_thunk_spec thunk) noexcept
  147. : _thunk(thunk)
  148. {
  149. }
  150. public:
  151. //! Construct from a C string literal
  152. BOOST_OUTCOME_SYSTEM_ERROR2_CONSTEXPR14 explicit string_ref(const char *str, size_type len = static_cast<size_type>(-1), void *state0 = nullptr, void *state1 = nullptr, void *state2 = nullptr,
  153. #ifndef NDEBUG
  154. _thunk_spec thunk = _checking_string_thunk
  155. #else
  156. _thunk_spec thunk = nullptr
  157. #endif
  158. ) noexcept
  159. : _begin(str)
  160. , _end((len == static_cast<size_type>(-1)) ? (str + detail::cstrlen(str)) : (str + len))
  161. , // NOLINT
  162. _state{state0, state1, state2}
  163. , _thunk(thunk)
  164. {
  165. }
  166. //! Copy construct the derived implementation.
  167. string_ref(const string_ref &o)
  168. : _begin(o._begin)
  169. , _end(o._end)
  170. , _state{o._state[0], o._state[1], o._state[2]}
  171. , _thunk(o._thunk)
  172. {
  173. if(_thunk != nullptr)
  174. {
  175. _thunk(this, &o, _thunk_op::copy);
  176. }
  177. }
  178. //! Move construct the derived implementation.
  179. string_ref(string_ref &&o) noexcept
  180. : _begin(o._begin)
  181. , _end(o._end)
  182. , _state{o._state[0], o._state[1], o._state[2]}
  183. , _thunk(o._thunk)
  184. {
  185. if(_thunk != nullptr)
  186. {
  187. _thunk(this, &o, _thunk_op::move);
  188. }
  189. }
  190. //! Copy assignment
  191. string_ref &operator=(const string_ref &o)
  192. {
  193. if(this != &o)
  194. {
  195. #if defined(__cpp_exceptions) || defined(__EXCEPTIONS) || defined(_CPPUNWIND)
  196. string_ref temp(static_cast<string_ref &&>(*this));
  197. this->~string_ref();
  198. try
  199. {
  200. new(this) string_ref(o); // may throw
  201. }
  202. catch(...)
  203. {
  204. new(this) string_ref(static_cast<string_ref &&>(temp));
  205. throw;
  206. }
  207. #else
  208. this->~string_ref();
  209. new(this) string_ref(o);
  210. #endif
  211. }
  212. return *this;
  213. }
  214. //! Move assignment
  215. string_ref &operator=(string_ref &&o) noexcept
  216. {
  217. if(this != &o)
  218. {
  219. this->~string_ref();
  220. new(this) string_ref(static_cast<string_ref &&>(o));
  221. }
  222. return *this;
  223. }
  224. //! Destruction
  225. ~string_ref()
  226. {
  227. if(_thunk != nullptr)
  228. {
  229. _thunk(this, nullptr, _thunk_op::destruct);
  230. }
  231. _begin = _end = nullptr;
  232. }
  233. //! Returns whether the reference is empty or not
  234. BOOST_OUTCOME_SYSTEM_ERROR2_NODISCARD bool empty() const noexcept { return _begin == _end; }
  235. //! Returns the size of the string
  236. size_type size() const noexcept { return _end - _begin; }
  237. //! Returns a null terminated C string
  238. const_pointer c_str() const noexcept { return _begin; }
  239. //! Returns a null terminated C string
  240. const_pointer data() const noexcept { return _begin; }
  241. //! Returns the beginning of the string
  242. iterator begin() noexcept { return _begin; }
  243. //! Returns the beginning of the string
  244. const_iterator begin() const noexcept { return _begin; }
  245. //! Returns the beginning of the string
  246. const_iterator cbegin() const noexcept { return _begin; }
  247. //! Returns the end of the string
  248. iterator end() noexcept { return _end; }
  249. //! Returns the end of the string
  250. const_iterator end() const noexcept { return _end; }
  251. //! Returns the end of the string
  252. const_iterator cend() const noexcept { return _end; }
  253. };
  254. /*! A reference counted, threadsafe reference to a message string.
  255. */
  256. class atomic_refcounted_string_ref : public string_ref
  257. {
  258. struct _allocated_msg
  259. {
  260. mutable std::atomic<unsigned> count{1};
  261. };
  262. _allocated_msg *&_msg() noexcept { return reinterpret_cast<_allocated_msg *&>(this->_state[0]); } // NOLINT
  263. const _allocated_msg *_msg() const noexcept { return reinterpret_cast<const _allocated_msg *>(this->_state[0]); } // NOLINT
  264. static void _refcounted_string_thunk(string_ref *_dest, const string_ref *_src, _thunk_op op) noexcept
  265. {
  266. auto dest = static_cast<atomic_refcounted_string_ref *>(_dest); // NOLINT
  267. auto src = static_cast<const atomic_refcounted_string_ref *>(_src); // NOLINT
  268. (void) src;
  269. assert(dest->_thunk == _refcounted_string_thunk); // NOLINT
  270. assert(src == nullptr || src->_thunk == _refcounted_string_thunk); // NOLINT
  271. switch(op)
  272. {
  273. case _thunk_op::copy:
  274. {
  275. if(dest->_msg() != nullptr)
  276. {
  277. auto count = dest->_msg()->count.fetch_add(1, std::memory_order_relaxed);
  278. (void) count;
  279. assert(count != 0); // NOLINT
  280. }
  281. return;
  282. }
  283. case _thunk_op::move:
  284. {
  285. assert(src); // NOLINT
  286. auto msrc = const_cast<atomic_refcounted_string_ref *>(src); // NOLINT
  287. msrc->_begin = msrc->_end = nullptr;
  288. msrc->_state[0] = msrc->_state[1] = msrc->_state[2] = nullptr;
  289. return;
  290. }
  291. case _thunk_op::destruct:
  292. {
  293. if(dest->_msg() != nullptr)
  294. {
  295. auto count = dest->_msg()->count.fetch_sub(1, std::memory_order_release);
  296. if(count == 1)
  297. {
  298. std::atomic_thread_fence(std::memory_order_acquire);
  299. free((void *) dest->_begin); // NOLINT
  300. delete dest->_msg(); // NOLINT
  301. }
  302. }
  303. }
  304. }
  305. }
  306. public:
  307. //! Construct from a C string literal allocated using `malloc()`.
  308. explicit atomic_refcounted_string_ref(const char *str, size_type len = static_cast<size_type>(-1), void *state1 = nullptr, void *state2 = nullptr) noexcept
  309. : string_ref(str, len, new(std::nothrow) _allocated_msg, state1, state2, _refcounted_string_thunk)
  310. {
  311. if(_msg() == nullptr)
  312. {
  313. free((void *) this->_begin); // NOLINT
  314. _msg() = nullptr; // disabled
  315. this->_begin = "failed to get message from system";
  316. this->_end = strchr(this->_begin, 0);
  317. return;
  318. }
  319. }
  320. };
  321. private:
  322. unique_id_type _id;
  323. protected:
  324. /*! Use [https://www.random.org/cgi-bin/randbyte?nbytes=8&format=h](https://www.random.org/cgi-bin/randbyte?nbytes=8&format=h) to get a random 64 bit id.
  325. Do NOT make up your own value. Do NOT use zero.
  326. */
  327. constexpr explicit status_code_domain(unique_id_type id) noexcept
  328. : _id(id)
  329. {
  330. }
  331. /*! UUID constructor, where input is constexpr parsed into a `unique_id_type`.
  332. */
  333. template <size_t N>
  334. constexpr explicit status_code_domain(const char (&uuid)[N]) noexcept
  335. : _id(detail::parse_uuid<N>(uuid))
  336. {
  337. }
  338. //! No public copying at type erased level
  339. status_code_domain(const status_code_domain &) = default;
  340. //! No public moving at type erased level
  341. status_code_domain(status_code_domain &&) = default;
  342. //! No public assignment at type erased level
  343. status_code_domain &operator=(const status_code_domain &) = default;
  344. //! No public assignment at type erased level
  345. status_code_domain &operator=(status_code_domain &&) = default;
  346. //! No public destruction at type erased level
  347. ~status_code_domain() = default;
  348. public:
  349. //! True if the unique ids match.
  350. constexpr bool operator==(const status_code_domain &o) const noexcept { return _id == o._id; }
  351. //! True if the unique ids do not match.
  352. constexpr bool operator!=(const status_code_domain &o) const noexcept { return _id != o._id; }
  353. //! True if this unique is lower than the other's unique id.
  354. constexpr bool operator<(const status_code_domain &o) const noexcept { return _id < o._id; }
  355. //! Returns the unique id used to identify identical category instances.
  356. constexpr unique_id_type id() const noexcept { return _id; }
  357. //! Name of this category.
  358. virtual string_ref name() const noexcept = 0;
  359. protected:
  360. //! True if code means failure.
  361. virtual bool _do_failure(const status_code<void> &code) const noexcept = 0;
  362. //! True if code is (potentially non-transitively) equivalent to another code in another domain.
  363. virtual bool _do_equivalent(const status_code<void> &code1, const status_code<void> &code2) const noexcept = 0;
  364. //! Returns the generic code closest to this code, if any.
  365. virtual generic_code _generic_code(const status_code<void> &code) const noexcept = 0;
  366. //! Return a reference to a string textually representing a code.
  367. virtual string_ref _do_message(const status_code<void> &code) const noexcept = 0;
  368. #if defined(_CPPUNWIND) || defined(__EXCEPTIONS) || defined(BOOST_OUTCOME_STANDARDESE_IS_IN_THE_HOUSE)
  369. //! Throw a code as a C++ exception.
  370. BOOST_OUTCOME_SYSTEM_ERROR2_NORETURN virtual void _do_throw_exception(const status_code<void> &code) const = 0;
  371. #else
  372. // Keep a vtable slot for binary compatibility
  373. BOOST_OUTCOME_SYSTEM_ERROR2_NORETURN virtual void _do_throw_exception(const status_code<void> & /*code*/) const { abort(); }
  374. #endif
  375. // For a `status_code<erased<T>>` only, copy from `src` to `dst`. Default implementation uses `memcpy()`.
  376. virtual void _do_erased_copy(status_code<void> &dst, const status_code<void> &src, size_t bytes) const { memcpy(&dst, &src, bytes); } // NOLINT
  377. // For a `status_code<erased<T>>` only, destroy the erased value type. Default implementation does nothing.
  378. virtual void _do_erased_destroy(status_code<void> &code, size_t bytes) const noexcept // NOLINT
  379. {
  380. (void) code;
  381. (void) bytes;
  382. }
  383. };
  384. BOOST_OUTCOME_SYSTEM_ERROR2_NAMESPACE_END
  385. #endif