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 #ifndef BOOST_OUTCOME_SYSTEM_ERROR2_STD_ERROR_CODE_HPP
0026 #define BOOST_OUTCOME_SYSTEM_ERROR2_STD_ERROR_CODE_HPP
0027
0028 #ifndef BOOST_OUTCOME_SYSTEM_ERROR2_NOT_POSIX
0029 #include "posix_code.hpp"
0030 #endif
0031
0032 #if defined(_WIN32) || defined(BOOST_OUTCOME_STANDARDESE_IS_IN_THE_HOUSE)
0033 #include "win32_code.hpp"
0034 #endif
0035
0036 #include <system_error>
0037
0038 BOOST_OUTCOME_SYSTEM_ERROR2_NAMESPACE_BEGIN
0039
0040 class _std_error_code_domain;
0041
0042 using std_error_code = status_code<_std_error_code_domain>;
0043
0044 namespace mixins
0045 {
0046 template <class Base> struct mixin<Base, _std_error_code_domain> : public Base
0047 {
0048 using Base::Base;
0049
0050
0051 inline mixin(std::error_code ec);
0052
0053
0054 inline const std::error_category &category() const noexcept;
0055 };
0056 }
0057
0058
0059
0060
0061 class _std_error_code_domain final : public status_code_domain
0062 {
0063 template <class DomainType> friend class status_code;
0064 using _base = status_code_domain;
0065 using _error_code_type = std::error_code;
0066 using _error_category_type = std::error_category;
0067
0068 std::string _name;
0069
0070 static _base::string_ref _make_string_ref(_error_code_type c) noexcept
0071 {
0072 #if defined(__cpp_exceptions) || defined(__EXCEPTIONS) || defined(_CPPUNWIND)
0073 try
0074 #endif
0075 {
0076 std::string msg = c.message();
0077 auto *p = static_cast<char *>(malloc(msg.size() + 1));
0078 if(p == nullptr)
0079 {
0080 return _base::string_ref("failed to allocate message");
0081 }
0082 memcpy(p, msg.c_str(), msg.size() + 1);
0083 return _base::atomic_refcounted_string_ref(p, msg.size());
0084 }
0085 #if defined(__cpp_exceptions) || defined(__EXCEPTIONS) || defined(_CPPUNWIND)
0086 catch(...)
0087 {
0088 return _base::string_ref("failed to allocate message");
0089 }
0090 #endif
0091 }
0092
0093 public:
0094
0095 using value_type = int;
0096 using _base::string_ref;
0097
0098
0099 const _error_category_type &error_category() const noexcept
0100 {
0101 auto ptr = 0x223a160d20de97b4 ^ this->id();
0102 return *reinterpret_cast<const _error_category_type *>(ptr);
0103 }
0104
0105
0106 explicit _std_error_code_domain(const _error_category_type &category) noexcept
0107 : _base(0x223a160d20de97b4 ^ reinterpret_cast<_base::unique_id_type>(&category))
0108 , _name("std_error_code_domain(")
0109 {
0110 _name.append(category.name());
0111 _name.push_back(')');
0112 }
0113 _std_error_code_domain(const _std_error_code_domain &) = default;
0114 _std_error_code_domain(_std_error_code_domain &&) = default;
0115 _std_error_code_domain &operator=(const _std_error_code_domain &) = default;
0116 _std_error_code_domain &operator=(_std_error_code_domain &&) = default;
0117 ~_std_error_code_domain() = default;
0118
0119 static inline const _std_error_code_domain *get(_error_code_type ec);
0120
0121 virtual string_ref name() const noexcept override { return string_ref(_name.c_str(), _name.size()); }
0122
0123 virtual payload_info_t payload_info() const noexcept override
0124 {
0125 return {sizeof(value_type), sizeof(status_code_domain *) + sizeof(value_type),
0126 (alignof(value_type) > alignof(status_code_domain *)) ? alignof(value_type) : alignof(status_code_domain *)};
0127 }
0128
0129 protected:
0130 virtual bool _do_failure(const status_code<void> &code) const noexcept override;
0131 virtual bool _do_equivalent(const status_code<void> &code1, const status_code<void> &code2) const noexcept override;
0132 virtual generic_code _generic_code(const status_code<void> &code) const noexcept override;
0133 virtual string_ref _do_message(const status_code<void> &code) const noexcept override;
0134 #if defined(_CPPUNWIND) || defined(__EXCEPTIONS) || defined(BOOST_OUTCOME_STANDARDESE_IS_IN_THE_HOUSE)
0135 BOOST_OUTCOME_SYSTEM_ERROR2_NORETURN virtual void _do_throw_exception(const status_code<void> &code) const override;
0136 #endif
0137 };
0138
0139 namespace detail
0140 {
0141 extern inline _std_error_code_domain *std_error_code_domain_from_category(const std::error_category &category)
0142 {
0143 static constexpr size_t max_items = 64;
0144 static struct storage_t
0145 {
0146 std::atomic<unsigned> _lock;
0147 union item_t
0148 {
0149 int _init;
0150 _std_error_code_domain domain;
0151 constexpr item_t()
0152 : _init(0)
0153 {
0154 }
0155 ~item_t() {}
0156 } items[max_items];
0157 size_t count{0};
0158
0159 void lock()
0160 {
0161 while(_lock.exchange(1, std::memory_order_acquire) != 0)
0162 ;
0163 }
0164 void unlock() { _lock.store(0, std::memory_order_release); }
0165
0166 storage_t() {}
0167 ~storage_t()
0168 {
0169 lock();
0170 for(size_t n = 0; n < count; n++)
0171 {
0172 items[n].domain.~_std_error_code_domain();
0173 }
0174 unlock();
0175 }
0176 _std_error_code_domain *add(const std::error_category &category)
0177 {
0178 _std_error_code_domain *ret = nullptr;
0179 lock();
0180 for(size_t n = 0; n < count; n++)
0181 {
0182 if(items[n].domain.error_category() == category)
0183 {
0184 ret = &items[n].domain;
0185 break;
0186 }
0187 }
0188 if(ret == nullptr && count < max_items)
0189 {
0190 ret = new(std::addressof(items[count++].domain)) _std_error_code_domain(category);
0191 }
0192 unlock();
0193 return ret;
0194 }
0195 } storage;
0196 return storage.add(category);
0197 }
0198 }
0199
0200 namespace mixins
0201 {
0202 template <class Base>
0203 inline mixin<Base, _std_error_code_domain>::mixin(std::error_code ec)
0204 : Base(typename Base::_value_type_constructor{}, _std_error_code_domain::get(ec), ec.value())
0205 {
0206 }
0207
0208 template <class Base> inline const std::error_category &mixin<Base, _std_error_code_domain>::category() const noexcept
0209 {
0210 const auto &domain = static_cast<const _std_error_code_domain &>(this->domain());
0211 return domain.error_category();
0212 };
0213 }
0214
0215 inline const _std_error_code_domain *_std_error_code_domain::get(std::error_code ec)
0216 {
0217 auto *p = detail::std_error_code_domain_from_category(ec.category());
0218 assert(p != nullptr);
0219 if(p == nullptr)
0220 {
0221 abort();
0222 }
0223 return p;
0224 }
0225
0226
0227 inline bool _std_error_code_domain::_do_failure(const status_code<void> &code) const noexcept
0228 {
0229 assert(code.domain() == *this);
0230 return static_cast<const std_error_code &>(code).value() != 0;
0231 }
0232
0233 inline bool _std_error_code_domain::_do_equivalent(const status_code<void> &code1, const status_code<void> &code2) const noexcept
0234 {
0235 assert(code1.domain() == *this);
0236 const auto &c1 = static_cast<const std_error_code &>(code1);
0237 const auto &cat1 = c1.category();
0238
0239 if(code2.domain() == *this)
0240 {
0241 const auto &c2 = static_cast<const std_error_code &>(code2);
0242 const auto &cat2 = c2.category();
0243
0244 if(cat1 == cat2)
0245 {
0246 return c1.value() == c2.value();
0247 }
0248
0249 return false;
0250 }
0251
0252 if(cat1 == std::generic_category())
0253 {
0254
0255 generic_code _c1(static_cast<errc>(c1.value()));
0256 return _c1 == code2;
0257 }
0258
0259 if(cat1 == std::system_category())
0260 {
0261
0262 #ifdef _WIN32
0263 win32_code _c1((win32::DWORD) c1.value());
0264 return _c1 == code2;
0265 #elif !defined(BOOST_OUTCOME_SYSTEM_ERROR2_NOT_POSIX)
0266 posix_code _c1(c1.value());
0267 return _c1 == code2;
0268 #endif
0269 }
0270 return false;
0271 }
0272
0273 inline generic_code _std_error_code_domain::_generic_code(const status_code<void> &code) const noexcept
0274 {
0275 assert(code.domain() == *this);
0276 const auto &c = static_cast<const std_error_code &>(code);
0277
0278 std::error_condition cond(c.category().default_error_condition(c.value()));
0279 if(cond.category() == std::generic_category())
0280 {
0281 return generic_code(static_cast<errc>(cond.value()));
0282 }
0283 #if !defined(BOOST_OUTCOME_SYSTEM_ERROR2_NOT_POSIX) && !defined(_WIN32)
0284 if(cond.category() == std::system_category())
0285 {
0286 return generic_code(static_cast<errc>(cond.value()));
0287 }
0288 #endif
0289 return errc::unknown;
0290 }
0291
0292 inline _std_error_code_domain::string_ref _std_error_code_domain::_do_message(const status_code<void> &code) const noexcept
0293 {
0294 assert(code.domain() == *this);
0295 const auto &c = static_cast<const std_error_code &>(code);
0296 return _make_string_ref(_error_code_type(c.value(), c.category()));
0297 }
0298
0299 #if defined(_CPPUNWIND) || defined(__EXCEPTIONS) || defined(BOOST_OUTCOME_STANDARDESE_IS_IN_THE_HOUSE)
0300 BOOST_OUTCOME_SYSTEM_ERROR2_NORETURN inline void _std_error_code_domain::_do_throw_exception(const status_code<void> &code) const
0301 {
0302 assert(code.domain() == *this);
0303 const auto &c = static_cast<const std_error_code &>(code);
0304 throw std::system_error(std::error_code(c.value(), c.category()));
0305 }
0306 #endif
0307
0308 static_assert(sizeof(std_error_code) <= sizeof(void *) * 2, "std_error_code does not fit into a system_code!");
0309
0310 BOOST_OUTCOME_SYSTEM_ERROR2_NAMESPACE_END
0311
0312
0313 namespace std
0314 {
0315 inline BOOST_OUTCOME_SYSTEM_ERROR2_NAMESPACE::std_error_code make_status_code(error_code c) noexcept
0316 {
0317 return BOOST_OUTCOME_SYSTEM_ERROR2_NAMESPACE::std_error_code(c);
0318 }
0319 }
0320
0321 #endif