File indexing completed on 2025-04-04 08:33:22
0001
0002
0003
0004
0005
0006
0007
0008
0009
0010
0011
0012
0013
0014
0015
0016
0017
0018
0019
0020
0021
0022
0023
0024
0025
0026
0027
0028
0029
0030
0031 #ifndef BOOST_OUTCOME_SYSTEM_ERROR2_STATUS_CODE_DOMAIN_HPP
0032 #define BOOST_OUTCOME_SYSTEM_ERROR2_STATUS_CODE_DOMAIN_HPP
0033
0034 #include "config.hpp"
0035
0036 #include <cstring> // for strchr
0037
0038 BOOST_OUTCOME_SYSTEM_ERROR2_NAMESPACE_BEGIN
0039
0040
0041
0042
0043
0044
0045
0046
0047
0048
0049 template <class DomainType> class status_code;
0050 class _generic_code_domain;
0051
0052 using generic_code = status_code<_generic_code_domain>;
0053
0054 namespace detail
0055 {
0056 template <class StatusCode, class Allocator> class indirecting_domain;
0057
0058
0059
0060
0061
0062
0063
0064
0065 #ifdef __GNUC__
0066 #pragma GCC diagnostic push
0067 #pragma GCC diagnostic ignored "-Wdiv-by-zero"
0068 #endif
0069 #if defined(__cpp_exceptions) || (defined(_MSC_VER) && !defined(__clang__))
0070 #define BOOST_OUTCOME_SYSTEM_ERROR2_FAIL_CONSTEXPR(msg) throw msg
0071 #else
0072 #define BOOST_OUTCOME_SYSTEM_ERROR2_FAIL_CONSTEXPR(msg) ((void) msg, 1 / 0)
0073 #endif
0074 constexpr inline unsigned long long parse_hex_byte(char c)
0075 {
0076 return ('0' <= c && c <= '9') ? (c - '0') :
0077 ('a' <= c && c <= 'f') ? (10 + c - 'a') :
0078 ('A' <= c && c <= 'F') ? (10 + c - 'A') :
0079 BOOST_OUTCOME_SYSTEM_ERROR2_FAIL_CONSTEXPR("Invalid character in UUID");
0080 }
0081 constexpr inline unsigned long long parse_uuid2(const char *s)
0082 {
0083 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) |
0084 (parse_hex_byte(s[4]) << 16) | (parse_hex_byte(s[5]) << 20) | (parse_hex_byte(s[6]) << 24) | (parse_hex_byte(s[7]) << 28) |
0085 (parse_hex_byte(s[9]) << 32) | (parse_hex_byte(s[10]) << 36) | (parse_hex_byte(s[11]) << 40) | (parse_hex_byte(s[12]) << 44) |
0086 (parse_hex_byte(s[14]) << 48) | (parse_hex_byte(s[15]) << 52) | (parse_hex_byte(s[16]) << 56) | (parse_hex_byte(s[17]) << 60))
0087 ^
0088 ((parse_hex_byte(s[19]) << 0) | (parse_hex_byte(s[20]) << 4) | (parse_hex_byte(s[21]) << 8) | (parse_hex_byte(s[22]) << 12) |
0089 (parse_hex_byte(s[24]) << 16) | (parse_hex_byte(s[25]) << 20) | (parse_hex_byte(s[26]) << 24) | (parse_hex_byte(s[27]) << 28) |
0090 (parse_hex_byte(s[28]) << 32) | (parse_hex_byte(s[29]) << 36) | (parse_hex_byte(s[30]) << 40) | (parse_hex_byte(s[31]) << 44) |
0091 (parse_hex_byte(s[32]) << 48) | (parse_hex_byte(s[33]) << 52) | (parse_hex_byte(s[34]) << 56) | (parse_hex_byte(s[35]) << 60));
0092 }
0093 template <size_t N> constexpr inline unsigned long long parse_uuid_from_array(const char (&uuid)[N])
0094 {
0095 return (N == 37) ? parse_uuid2(uuid) : ((N == 39) ? parse_uuid2(uuid + 1) : BOOST_OUTCOME_SYSTEM_ERROR2_FAIL_CONSTEXPR("UUID does not have correct length"));
0096 }
0097 template <size_t N> constexpr inline unsigned long long parse_uuid_from_pointer(const char *uuid)
0098 {
0099 return (N == 36) ? parse_uuid2(uuid) : ((N == 38) ? parse_uuid2(uuid + 1) : BOOST_OUTCOME_SYSTEM_ERROR2_FAIL_CONSTEXPR("UUID does not have correct length"));
0100 }
0101 #ifdef __GNUC__
0102 #pragma GCC diagnostic pop
0103 #endif
0104 static constexpr unsigned long long test_uuid_parse = parse_uuid_from_array("430f1201-94fc-06c7-430f-120194fc06c7");
0105
0106 }
0107
0108
0109
0110 class status_code_domain
0111 {
0112 template <class DomainType> friend class status_code;
0113 template <class StatusCode, class Allocator> friend class detail::indirecting_domain;
0114
0115 public:
0116
0117 using unique_id_type = unsigned long long;
0118
0119
0120
0121
0122
0123
0124
0125
0126
0127 class string_ref
0128 {
0129 public:
0130
0131 using value_type = const char;
0132
0133 using size_type = size_t;
0134
0135 using pointer = const char *;
0136
0137 using const_pointer = const char *;
0138
0139 using iterator = const char *;
0140
0141 using const_iterator = const char *;
0142
0143 protected:
0144
0145 enum class _thunk_op
0146 {
0147 copy,
0148 move,
0149 destruct
0150 };
0151
0152 using _thunk_spec = void (*)(string_ref *dest, const string_ref *src, _thunk_op op);
0153 #ifndef NDEBUG
0154 private:
0155 static void _checking_string_thunk(string_ref *dest, const string_ref *src, _thunk_op ) noexcept
0156 {
0157 (void) dest;
0158 (void) src;
0159 assert(dest->_thunk == _checking_string_thunk);
0160 assert(src == nullptr || src->_thunk == _checking_string_thunk);
0161
0162 }
0163
0164 protected:
0165 #endif
0166
0167 pointer _begin{}, _end{};
0168
0169 void *_state[3]{};
0170
0171 const _thunk_spec _thunk{nullptr};
0172
0173 constexpr explicit string_ref(_thunk_spec thunk) noexcept
0174 : _thunk(thunk)
0175 {
0176 }
0177
0178 public:
0179
0180 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,
0181 void *state2 = nullptr,
0182 #ifndef NDEBUG
0183 _thunk_spec thunk = _checking_string_thunk
0184 #else
0185 _thunk_spec thunk = nullptr
0186 #endif
0187 ) noexcept
0188 : _begin(str)
0189 , _end((len == static_cast<size_type>(-1)) ? (str + detail::cstrlen(str)) : (str + len))
0190 ,
0191 _state{state0, state1, state2}
0192 , _thunk(thunk)
0193 {
0194 }
0195
0196 BOOST_OUTCOME_SYSTEM_ERROR2_CONSTEXPR20 string_ref(const string_ref &o)
0197 : _begin(o._begin)
0198 , _end(o._end)
0199 , _state{o._state[0], o._state[1], o._state[2]}
0200 , _thunk(o._thunk)
0201 {
0202 if(_thunk != nullptr)
0203 {
0204 _thunk(this, &o, _thunk_op::copy);
0205 }
0206 }
0207
0208 BOOST_OUTCOME_SYSTEM_ERROR2_CONSTEXPR20 string_ref(string_ref &&o) noexcept
0209 : _begin(o._begin)
0210 , _end(o._end)
0211 , _state{o._state[0], o._state[1], o._state[2]}
0212 , _thunk(o._thunk)
0213 {
0214 if(_thunk != nullptr)
0215 {
0216 _thunk(this, &o, _thunk_op::move);
0217 }
0218 }
0219
0220 BOOST_OUTCOME_SYSTEM_ERROR2_CONSTEXPR20 string_ref &operator=(const string_ref &o)
0221 {
0222 if(this != &o)
0223 {
0224 #if defined(__cpp_exceptions) || defined(__EXCEPTIONS) || defined(_CPPUNWIND)
0225 string_ref temp(static_cast<string_ref &&>(*this));
0226 this->~string_ref();
0227 try
0228 {
0229 new(this) string_ref(o);
0230 }
0231 catch(...)
0232 {
0233 new(this) string_ref(static_cast<string_ref &&>(temp));
0234 throw;
0235 }
0236 #else
0237 this->~string_ref();
0238 new(this) string_ref(o);
0239 #endif
0240 }
0241 return *this;
0242 }
0243
0244 BOOST_OUTCOME_SYSTEM_ERROR2_CONSTEXPR20 string_ref &operator=(string_ref &&o) noexcept
0245 {
0246 if(this != &o)
0247 {
0248 this->~string_ref();
0249 new(this) string_ref(static_cast<string_ref &&>(o));
0250 }
0251 return *this;
0252 }
0253
0254 BOOST_OUTCOME_SYSTEM_ERROR2_CONSTEXPR20 ~string_ref()
0255 {
0256 if(_thunk != nullptr)
0257 {
0258 _thunk(this, nullptr, _thunk_op::destruct);
0259 }
0260 _begin = _end = nullptr;
0261 }
0262
0263
0264 BOOST_OUTCOME_SYSTEM_ERROR2_NODISCARD constexpr bool empty() const noexcept { return _begin == _end; }
0265
0266 constexpr size_type size() const noexcept { return _end - _begin; }
0267
0268 constexpr const_pointer c_str() const noexcept { return _begin; }
0269
0270 constexpr const_pointer data() const noexcept { return _begin; }
0271
0272 BOOST_OUTCOME_SYSTEM_ERROR2_CONSTEXPR14 iterator begin() noexcept { return _begin; }
0273
0274 BOOST_OUTCOME_SYSTEM_ERROR2_CONSTEXPR14 const_iterator begin() const noexcept { return _begin; }
0275
0276 constexpr const_iterator cbegin() const noexcept { return _begin; }
0277
0278 BOOST_OUTCOME_SYSTEM_ERROR2_CONSTEXPR14 iterator end() noexcept { return _end; }
0279
0280 BOOST_OUTCOME_SYSTEM_ERROR2_CONSTEXPR14 const_iterator end() const noexcept { return _end; }
0281
0282 constexpr const_iterator cend() const noexcept { return _end; }
0283 };
0284
0285
0286
0287 class atomic_refcounted_string_ref : public string_ref
0288 {
0289 struct _allocated_msg
0290 {
0291 mutable std::atomic<unsigned> count{1};
0292 };
0293 _allocated_msg *&_msg() noexcept { return reinterpret_cast<_allocated_msg *&>(this->_state[0]); }
0294 const _allocated_msg *_msg() const noexcept { return reinterpret_cast<const _allocated_msg *>(this->_state[0]); }
0295
0296 static BOOST_OUTCOME_SYSTEM_ERROR2_CONSTEXPR20 void _refcounted_string_thunk(string_ref *_dest, const string_ref *_src, _thunk_op op) noexcept
0297 {
0298 auto dest = static_cast<atomic_refcounted_string_ref *>(_dest);
0299 auto src = static_cast<const atomic_refcounted_string_ref *>(_src);
0300 (void) src;
0301 assert(dest->_thunk == _refcounted_string_thunk);
0302 assert(src == nullptr || src->_thunk == _refcounted_string_thunk);
0303 switch(op)
0304 {
0305 case _thunk_op::copy:
0306 {
0307 if(dest->_msg() != nullptr)
0308 {
0309 auto count = dest->_msg()->count.fetch_add(1, std::memory_order_relaxed);
0310 (void) count;
0311 assert(count != 0);
0312 }
0313 return;
0314 }
0315 case _thunk_op::move:
0316 {
0317 assert(src);
0318 auto msrc = const_cast<atomic_refcounted_string_ref *>(src);
0319 msrc->_begin = msrc->_end = nullptr;
0320 msrc->_state[0] = msrc->_state[1] = msrc->_state[2] = nullptr;
0321 return;
0322 }
0323 case _thunk_op::destruct:
0324 {
0325 if(dest->_msg() != nullptr)
0326 {
0327 auto count = dest->_msg()->count.fetch_sub(1, std::memory_order_release);
0328 if(count == 1)
0329 {
0330 std::atomic_thread_fence(std::memory_order_acquire);
0331 free((void *) dest->_begin);
0332 delete dest->_msg();
0333 }
0334 }
0335 }
0336 }
0337 }
0338
0339 public:
0340
0341 explicit atomic_refcounted_string_ref(const char *str, size_type len = static_cast<size_type>(-1), void *state1 = nullptr, void *state2 = nullptr) noexcept
0342 : string_ref(str, len, new(std::nothrow) _allocated_msg, state1, state2, _refcounted_string_thunk)
0343 {
0344 if(_msg() == nullptr)
0345 {
0346 free((void *) this->_begin);
0347 _msg() = nullptr;
0348 this->_begin = "failed to get message from system";
0349 this->_end = strchr(this->_begin, 0);
0350 return;
0351 }
0352 }
0353 };
0354
0355 private:
0356 unique_id_type _id;
0357
0358 protected:
0359
0360
0361
0362
0363 constexpr explicit status_code_domain(unique_id_type id) noexcept
0364 : _id(id)
0365 {
0366 }
0367
0368
0369 template <size_t N>
0370 constexpr explicit status_code_domain(const char (&uuid)[N]) noexcept
0371 : _id(detail::parse_uuid_from_array<N>(uuid))
0372 {
0373 }
0374 template <size_t N> struct _uuid_size
0375 {
0376 };
0377
0378 template <size_t N>
0379 constexpr explicit status_code_domain(const char *uuid, _uuid_size<N> ) noexcept
0380 : _id(detail::parse_uuid_from_pointer<N>(uuid))
0381 {
0382 }
0383
0384 status_code_domain(const status_code_domain &) = default;
0385
0386 status_code_domain(status_code_domain &&) = default;
0387
0388 status_code_domain &operator=(const status_code_domain &) = default;
0389
0390 status_code_domain &operator=(status_code_domain &&) = default;
0391
0392 ~status_code_domain() = default;
0393
0394 public:
0395
0396 constexpr bool operator==(const status_code_domain &o) const noexcept { return _id == o._id; }
0397
0398 constexpr bool operator!=(const status_code_domain &o) const noexcept { return _id != o._id; }
0399
0400 constexpr bool operator<(const status_code_domain &o) const noexcept { return _id < o._id; }
0401
0402
0403 constexpr unique_id_type id() const noexcept { return _id; }
0404
0405 BOOST_OUTCOME_SYSTEM_ERROR2_CONSTEXPR20 virtual string_ref name() const noexcept = 0;
0406
0407 struct payload_info_t
0408 {
0409 size_t payload_size{0};
0410 size_t total_size{0};
0411 size_t total_alignment{1};
0412
0413 payload_info_t() = default;
0414 constexpr payload_info_t(size_t _payload_size, size_t _total_size, size_t _total_alignment)
0415 : payload_size(_payload_size)
0416 , total_size(_total_size)
0417 , total_alignment(_total_alignment)
0418 {
0419 }
0420 };
0421
0422 BOOST_OUTCOME_SYSTEM_ERROR2_CONSTEXPR20 virtual payload_info_t payload_info() const noexcept = 0;
0423
0424 protected:
0425
0426 BOOST_OUTCOME_SYSTEM_ERROR2_CONSTEXPR20 virtual bool _do_failure(const status_code<void> &code) const noexcept = 0;
0427
0428 BOOST_OUTCOME_SYSTEM_ERROR2_CONSTEXPR20 virtual bool _do_equivalent(const status_code<void> &code1, const status_code<void> &code2) const noexcept = 0;
0429
0430 BOOST_OUTCOME_SYSTEM_ERROR2_CONSTEXPR20 virtual generic_code _generic_code(const status_code<void> &code) const noexcept = 0;
0431
0432 BOOST_OUTCOME_SYSTEM_ERROR2_CONSTEXPR20 virtual string_ref _do_message(const status_code<void> &code) const noexcept = 0;
0433 #if defined(_CPPUNWIND) || defined(__EXCEPTIONS) || defined(BOOST_OUTCOME_STANDARDESE_IS_IN_THE_HOUSE)
0434
0435 BOOST_OUTCOME_SYSTEM_ERROR2_NORETURN BOOST_OUTCOME_SYSTEM_ERROR2_CONSTEXPR20 virtual void _do_throw_exception(const status_code<void> &code) const = 0;
0436 #else
0437
0438 BOOST_OUTCOME_SYSTEM_ERROR2_NORETURN virtual void _do_throw_exception(const status_code<void> & ) const { abort(); }
0439 #endif
0440
0441
0442 virtual bool _do_erased_copy(status_code<void> &dst, const status_code<void> &src, payload_info_t dstinfo) const
0443 {
0444
0445 const auto srcinfo = payload_info();
0446 if(dstinfo.total_size < srcinfo.total_size)
0447 {
0448 return false;
0449 }
0450 const auto tocopy = (dstinfo.total_size > srcinfo.total_size) ? srcinfo.total_size : dstinfo.total_size;
0451 memcpy(&dst, &src, tocopy);
0452 return true;
0453 }
0454
0455 BOOST_OUTCOME_SYSTEM_ERROR2_CONSTEXPR20 virtual void _do_erased_destroy(status_code<void> &code, payload_info_t info) const noexcept
0456 {
0457 (void) code;
0458 (void) info;
0459 }
0460 };
0461
0462 BOOST_OUTCOME_SYSTEM_ERROR2_NAMESPACE_END
0463
0464 #endif