123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255 |
- #ifndef BOOST_OUTCOME_SYSTEM_ERROR2_COM_CODE_HPP
- #define BOOST_OUTCOME_SYSTEM_ERROR2_COM_CODE_HPP
- #if !defined(_WIN32) && !defined(BOOST_OUTCOME_STANDARDESE_IS_IN_THE_HOUSE)
- #error This file should only be included on Windows
- #endif
- #include "nt_code.hpp"
- #include "win32_code.hpp"
- #ifndef BOOST_OUTCOME_STANDARDESE_IS_IN_THE_HOUSE
- #include <stdio.h> // needed by mingw for comdef.h to work
- #include <comdef.h>
- #endif
- BOOST_OUTCOME_SYSTEM_ERROR2_NAMESPACE_BEGIN
- class _com_code_domain;
- using com_code = status_code<_com_code_domain>;
- using com_error = status_error<_com_code_domain>;
- class _com_code_domain : public status_code_domain
- {
- template <class DomainType> friend class status_code;
- using _base = status_code_domain;
-
- #ifdef _COMDEF_NOT_WINAPI_FAMILY_DESKTOP_APP
- static _base::string_ref _make_string_ref(HRESULT c, wchar_t *perrinfo = nullptr) noexcept
- #else
- static _base::string_ref _make_string_ref(HRESULT c, IErrorInfo *perrinfo = nullptr) noexcept
- #endif
- {
- _com_error ce(c, perrinfo);
- #ifdef _UNICODE
- win32::DWORD wlen = (win32::DWORD) wcslen(ce.ErrorMessage());
- size_t allocation = wlen + (wlen >> 1);
- win32::DWORD bytes;
- if(wlen == 0)
- {
- return _base::string_ref("failed to get message from system");
- }
- for(;;)
- {
- auto *p = static_cast<char *>(malloc(allocation));
- if(p == nullptr)
- {
- return _base::string_ref("failed to get message from system");
- }
- bytes = win32::WideCharToMultiByte(65001 , 0, ce.ErrorMessage(), (int) (wlen + 1), p, (int) allocation, nullptr, nullptr);
- if(bytes != 0)
- {
- char *end = strchr(p, 0);
- while(end[-1] == 10 || end[-1] == 13)
- {
- --end;
- }
- *end = 0;
- return _base::atomic_refcounted_string_ref(p, end - p);
- }
- free(p);
- if(win32::GetLastError() == 0x7a )
- {
- allocation += allocation >> 2;
- continue;
- }
- return _base::string_ref("failed to get message from system");
- }
- #else
- auto wlen = static_cast<win32::DWORD>(strlen(ce.ErrorMessage()));
- auto *p = static_cast<char *>(malloc(wlen + 1));
- if(p == nullptr)
- {
- return _base::string_ref("failed to get message from system");
- }
- memcpy(p, ce.ErrorMessage(), wlen + 1);
- char *end = strchr(p, 0);
- while(end[-1] == 10 || end[-1] == 13)
- {
- --end;
- }
- *end = 0;
- return _base::atomic_refcounted_string_ref(p, end - p);
- #endif
- }
- public:
-
- using value_type = HRESULT;
- using _base::string_ref;
- public:
-
- constexpr explicit _com_code_domain(typename _base::unique_id_type id = 0xdc8275428b4effac) noexcept
- : _base(id)
- {
- }
- _com_code_domain(const _com_code_domain &) = default;
- _com_code_domain(_com_code_domain &&) = default;
- _com_code_domain &operator=(const _com_code_domain &) = default;
- _com_code_domain &operator=(_com_code_domain &&) = default;
- ~_com_code_domain() = default;
-
- static inline constexpr const _com_code_domain &get();
- virtual string_ref name() const noexcept override { return string_ref("COM domain"); }
- virtual payload_info_t payload_info() const noexcept override
- {
- return {sizeof(value_type), sizeof(status_code_domain *) + sizeof(value_type),
- (alignof(value_type) > alignof(status_code_domain *)) ? alignof(value_type) : alignof(status_code_domain *)};
- }
- protected:
- virtual bool _do_failure(const status_code<void> &code) const noexcept override
- {
- assert(code.domain() == *this);
- return static_cast<const com_code &>(code).value() < 0;
- }
-
- virtual bool _do_equivalent(const status_code<void> &code1, const status_code<void> &code2) const noexcept override
- {
- assert(code1.domain() == *this);
- const auto &c1 = static_cast<const com_code &>(code1);
- if(code2.domain() == *this)
- {
- const auto &c2 = static_cast<const com_code &>(code2);
- return c1.value() == c2.value();
- }
- if((c1.value() & FACILITY_NT_BIT) != 0)
- {
- if(code2.domain() == nt_code_domain)
- {
- const auto &c2 = static_cast<const nt_code &>(code2);
- if(c2.value() == (c1.value() & ~FACILITY_NT_BIT))
- {
- return true;
- }
- }
- else if(code2.domain() == generic_code_domain)
- {
- const auto &c2 = static_cast<const generic_code &>(code2);
- if(static_cast<int>(c2.value()) == _nt_code_domain::_nt_code_to_errno(c1.value() & ~FACILITY_NT_BIT))
- {
- return true;
- }
- }
- }
- else if(HRESULT_FACILITY(c1.value()) == FACILITY_WIN32)
- {
- if(code2.domain() == win32_code_domain)
- {
- const auto &c2 = static_cast<const win32_code &>(code2);
- if(c2.value() == HRESULT_CODE(c1.value()))
- {
- return true;
- }
- }
- else if(code2.domain() == generic_code_domain)
- {
- const auto &c2 = static_cast<const generic_code &>(code2);
- if(static_cast<int>(c2.value()) == _win32_code_domain::_win32_code_to_errno(HRESULT_CODE(c1.value())))
- {
- return true;
- }
- }
- }
- return false;
- }
- virtual generic_code _generic_code(const status_code<void> &code) const noexcept override
- {
- assert(code.domain() == *this);
- const auto &c1 = static_cast<const com_code &>(code);
- if(c1.value() == S_OK)
- {
- return generic_code(errc::success);
- }
- if((c1.value() & FACILITY_NT_BIT) != 0)
- {
- return generic_code(static_cast<errc>(_nt_code_domain::_nt_code_to_errno(c1.value() & ~FACILITY_NT_BIT)));
- }
- if(HRESULT_FACILITY(c1.value()) == FACILITY_WIN32)
- {
- return generic_code(static_cast<errc>(_win32_code_domain::_win32_code_to_errno(HRESULT_CODE(c1.value()))));
- }
- return generic_code(errc::unknown);
- }
- virtual string_ref _do_message(const status_code<void> &code) const noexcept override
- {
- assert(code.domain() == *this);
- const auto &c = static_cast<const com_code &>(code);
- return _make_string_ref(c.value());
- }
- #if defined(_CPPUNWIND) || defined(__EXCEPTIONS) || defined(BOOST_OUTCOME_STANDARDESE_IS_IN_THE_HOUSE)
- BOOST_OUTCOME_SYSTEM_ERROR2_NORETURN virtual void _do_throw_exception(const status_code<void> &code) const override
- {
- assert(code.domain() == *this);
- const auto &c = static_cast<const com_code &>(code);
- throw status_error<_com_code_domain>(c);
- }
- #endif
- };
- constexpr _com_code_domain com_code_domain;
- inline constexpr const _com_code_domain &_com_code_domain::get()
- {
- return com_code_domain;
- }
- BOOST_OUTCOME_SYSTEM_ERROR2_NAMESPACE_END
- #endif
|