File indexing completed on 2025-04-04 08:33:21
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_COM_CODE_HPP
0032 #define BOOST_OUTCOME_SYSTEM_ERROR2_COM_CODE_HPP
0033
0034 #if !defined(_WIN32) && !defined(BOOST_OUTCOME_STANDARDESE_IS_IN_THE_HOUSE)
0035 #error This file should only be included on Windows
0036 #endif
0037
0038 #include "nt_code.hpp"
0039 #include "win32_code.hpp"
0040
0041 #ifndef BOOST_OUTCOME_STANDARDESE_IS_IN_THE_HOUSE
0042 #include <stdio.h> // needed by mingw for comdef.h to work
0043
0044 #include <comdef.h>
0045 #endif
0046
0047 BOOST_OUTCOME_SYSTEM_ERROR2_NAMESPACE_BEGIN
0048
0049 class _com_code_domain;
0050
0051
0052
0053
0054
0055
0056
0057 using com_code = status_code<_com_code_domain>;
0058
0059 using com_error = status_error<_com_code_domain>;
0060
0061
0062
0063 class _com_code_domain : public status_code_domain
0064 {
0065 template <class DomainType> friend class status_code;
0066 using _base = status_code_domain;
0067
0068
0069 #ifdef _COMDEF_NOT_WINAPI_FAMILY_DESKTOP_APP
0070 static _base::string_ref _make_string_ref(HRESULT c, wchar_t *perrinfo = nullptr) noexcept
0071 #else
0072 static _base::string_ref _make_string_ref(HRESULT c, IErrorInfo *perrinfo = nullptr) noexcept
0073 #endif
0074 {
0075 _com_error ce(c, perrinfo);
0076 #ifdef _UNICODE
0077 win32::DWORD wlen = (win32::DWORD) wcslen(ce.ErrorMessage());
0078 size_t allocation = wlen + (wlen >> 1);
0079 win32::DWORD bytes;
0080 if(wlen == 0)
0081 {
0082 return _base::string_ref("failed to get message from system");
0083 }
0084 for(;;)
0085 {
0086 auto *p = static_cast<char *>(malloc(allocation));
0087 if(p == nullptr)
0088 {
0089 return _base::string_ref("failed to get message from system");
0090 }
0091 bytes = win32::WideCharToMultiByte(65001 , 0, ce.ErrorMessage(), (int) (wlen + 1), p, (int) allocation, nullptr, nullptr);
0092 if(bytes != 0)
0093 {
0094 char *end = strchr(p, 0);
0095 while(end[-1] == 10 || end[-1] == 13)
0096 {
0097 --end;
0098 }
0099 *end = 0;
0100 return _base::atomic_refcounted_string_ref(p, end - p);
0101 }
0102 free(p);
0103 if(win32::GetLastError() == 0x7a )
0104 {
0105 allocation += allocation >> 2;
0106 continue;
0107 }
0108 return _base::string_ref("failed to get message from system");
0109 }
0110 #else
0111 auto wlen = static_cast<win32::DWORD>(strlen(ce.ErrorMessage()));
0112 auto *p = static_cast<char *>(malloc(wlen + 1));
0113 if(p == nullptr)
0114 {
0115 return _base::string_ref("failed to get message from system");
0116 }
0117 memcpy(p, ce.ErrorMessage(), wlen + 1);
0118 char *end = strchr(p, 0);
0119 while(end[-1] == 10 || end[-1] == 13)
0120 {
0121 --end;
0122 }
0123 *end = 0;
0124 return _base::atomic_refcounted_string_ref(p, end - p);
0125 #endif
0126 }
0127
0128 public:
0129
0130 using value_type = HRESULT;
0131 using _base::string_ref;
0132
0133 public:
0134
0135 constexpr explicit _com_code_domain(typename _base::unique_id_type id = 0xdc8275428b4effac) noexcept
0136 : _base(id)
0137 {
0138 }
0139 _com_code_domain(const _com_code_domain &) = default;
0140 _com_code_domain(_com_code_domain &&) = default;
0141 _com_code_domain &operator=(const _com_code_domain &) = default;
0142 _com_code_domain &operator=(_com_code_domain &&) = default;
0143 ~_com_code_domain() = default;
0144
0145
0146 static inline constexpr const _com_code_domain &get();
0147
0148 virtual string_ref name() const noexcept override { return string_ref("COM domain"); }
0149
0150 virtual payload_info_t payload_info() const noexcept override
0151 {
0152 return {sizeof(value_type), sizeof(status_code_domain *) + sizeof(value_type),
0153 (alignof(value_type) > alignof(status_code_domain *)) ? alignof(value_type) : alignof(status_code_domain *)};
0154 }
0155
0156 protected:
0157 virtual bool _do_failure(const status_code<void> &code) const noexcept override
0158 {
0159 assert(code.domain() == *this);
0160 return static_cast<const com_code &>(code).value() < 0;
0161 }
0162
0163
0164 virtual bool _do_equivalent(const status_code<void> &code1, const status_code<void> &code2) const noexcept override
0165 {
0166 assert(code1.domain() == *this);
0167 const auto &c1 = static_cast<const com_code &>(code1);
0168 if(code2.domain() == *this)
0169 {
0170 const auto &c2 = static_cast<const com_code &>(code2);
0171 return c1.value() == c2.value();
0172 }
0173 if((c1.value() & FACILITY_NT_BIT) != 0)
0174 {
0175 if(code2.domain() == nt_code_domain)
0176 {
0177 const auto &c2 = static_cast<const nt_code &>(code2);
0178 if(c2.value() == (c1.value() & ~FACILITY_NT_BIT))
0179 {
0180 return true;
0181 }
0182 }
0183 else if(code2.domain() == generic_code_domain)
0184 {
0185 const auto &c2 = static_cast<const generic_code &>(code2);
0186 if(static_cast<int>(c2.value()) == _nt_code_domain::_nt_code_to_errno(c1.value() & ~FACILITY_NT_BIT))
0187 {
0188 return true;
0189 }
0190 }
0191 }
0192 else if(HRESULT_FACILITY(c1.value()) == FACILITY_WIN32)
0193 {
0194 if(code2.domain() == win32_code_domain)
0195 {
0196 const auto &c2 = static_cast<const win32_code &>(code2);
0197 if(c2.value() == HRESULT_CODE(c1.value()))
0198 {
0199 return true;
0200 }
0201 }
0202 else if(code2.domain() == generic_code_domain)
0203 {
0204 const auto &c2 = static_cast<const generic_code &>(code2);
0205 if(static_cast<int>(c2.value()) == _win32_code_domain::_win32_code_to_errno(HRESULT_CODE(c1.value())))
0206 {
0207 return true;
0208 }
0209 }
0210 }
0211 return false;
0212 }
0213 virtual generic_code _generic_code(const status_code<void> &code) const noexcept override
0214 {
0215 assert(code.domain() == *this);
0216 const auto &c1 = static_cast<const com_code &>(code);
0217 if(c1.value() == S_OK)
0218 {
0219 return generic_code(errc::success);
0220 }
0221 if((c1.value() & FACILITY_NT_BIT) != 0)
0222 {
0223 return generic_code(static_cast<errc>(_nt_code_domain::_nt_code_to_errno(c1.value() & ~FACILITY_NT_BIT)));
0224 }
0225 if(HRESULT_FACILITY(c1.value()) == FACILITY_WIN32)
0226 {
0227 return generic_code(static_cast<errc>(_win32_code_domain::_win32_code_to_errno(HRESULT_CODE(c1.value()))));
0228 }
0229 return generic_code(errc::unknown);
0230 }
0231 virtual string_ref _do_message(const status_code<void> &code) const noexcept override
0232 {
0233 assert(code.domain() == *this);
0234 const auto &c = static_cast<const com_code &>(code);
0235 return _make_string_ref(c.value());
0236 }
0237 #if defined(_CPPUNWIND) || defined(__EXCEPTIONS) || defined(BOOST_OUTCOME_STANDARDESE_IS_IN_THE_HOUSE)
0238 BOOST_OUTCOME_SYSTEM_ERROR2_NORETURN virtual void _do_throw_exception(const status_code<void> &code) const override
0239 {
0240 assert(code.domain() == *this);
0241 const auto &c = static_cast<const com_code &>(code);
0242 throw status_error<_com_code_domain>(c);
0243 }
0244 #endif
0245 };
0246
0247 constexpr _com_code_domain com_code_domain;
0248 inline constexpr const _com_code_domain &_com_code_domain::get()
0249 {
0250 return com_code_domain;
0251 }
0252
0253 BOOST_OUTCOME_SYSTEM_ERROR2_NAMESPACE_END
0254
0255 #endif