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_NT_CODE_HPP
0032 #define BOOST_OUTCOME_SYSTEM_ERROR2_NT_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 "win32_code.hpp"
0039
0040 BOOST_OUTCOME_SYSTEM_ERROR2_NAMESPACE_BEGIN
0041
0042
0043 namespace win32
0044 {
0045 #ifdef __MINGW32__
0046 extern "C"
0047 {
0048 #endif
0049
0050 using NTSTATUS = long;
0051
0052 using HMODULE = void *;
0053
0054 extern HMODULE __stdcall GetModuleHandleW(const wchar_t *lpModuleName);
0055 #ifdef __MINGW32__
0056 }
0057 #else
0058 #pragma comment(lib, "kernel32.lib")
0059 #if(defined(__x86_64__) || defined(_M_X64)) || (defined(__aarch64__) || defined(_M_ARM64))
0060 #pragma comment(linker, "/alternatename:?GetModuleHandleW@win32@system_error2@@YAPEAXPEB_W@Z=GetModuleHandleW")
0061 #elif defined(__x86__) || defined(_M_IX86) || defined(__i386__)
0062 #pragma comment(linker, "/alternatename:?GetModuleHandleW@win32@system_error2@@YGPAXPB_W@Z=_GetModuleHandleW@4")
0063 #elif defined(__arm__) || defined(_M_ARM)
0064 #pragma comment(linker, "/alternatename:?GetModuleHandleW@win32@system_error2@@YAPAXPB_W@Z=GetModuleHandleW")
0065 #else
0066 #error Unknown architecture
0067 #endif
0068 #endif
0069 }
0070
0071 class _nt_code_domain;
0072
0073 using nt_code = status_code<_nt_code_domain>;
0074
0075 using nt_error = status_error<_nt_code_domain>;
0076
0077
0078
0079 class _nt_code_domain : public status_code_domain
0080 {
0081 template <class DomainType> friend class status_code;
0082 friend class _com_code_domain;
0083 using _base = status_code_domain;
0084 static int _nt_code_to_errno(win32::NTSTATUS c)
0085 {
0086 if(c >= 0)
0087 {
0088 return 0;
0089 }
0090 switch(static_cast<unsigned>(c))
0091 {
0092 #include "detail/nt_code_to_generic_code.ipp"
0093 }
0094 return -1;
0095 }
0096 static win32::DWORD _nt_code_to_win32_code(win32::NTSTATUS c)
0097 {
0098 if(c >= 0)
0099 {
0100 return 0;
0101 }
0102 switch(static_cast<unsigned>(c))
0103 {
0104 #include "detail/nt_code_to_win32_code.ipp"
0105 }
0106 return static_cast<win32::DWORD>(-1);
0107 }
0108
0109 static _base::string_ref _make_string_ref(win32::NTSTATUS c) noexcept
0110 {
0111 wchar_t buffer[32768];
0112 static win32::HMODULE ntdll = win32::GetModuleHandleW(L"NTDLL.DLL");
0113 win32::DWORD wlen =
0114 win32::FormatMessageW(0x00000800 | 0x00001000 | 0x00000200 ,
0115 ntdll, c, (1 << 10) , buffer, 32768, nullptr);
0116 size_t allocation = wlen + (wlen >> 1);
0117 win32::DWORD bytes;
0118 if(wlen == 0)
0119 {
0120 return _base::string_ref("failed to get message from system");
0121 }
0122 for(;;)
0123 {
0124 auto *p = static_cast<char *>(malloc(allocation));
0125 if(p == nullptr)
0126 {
0127 return _base::string_ref("failed to get message from system");
0128 }
0129 bytes = win32::WideCharToMultiByte(65001 , 0, buffer, (int) (wlen + 1), p, (int) allocation, nullptr, nullptr);
0130 if(bytes != 0)
0131 {
0132 char *end = strchr(p, 0);
0133 while(end[-1] == 10 || end[-1] == 13)
0134 {
0135 --end;
0136 }
0137 *end = 0;
0138 return _base::atomic_refcounted_string_ref(p, end - p);
0139 }
0140 free(p);
0141 if(win32::GetLastError() == 0x7a )
0142 {
0143 allocation += allocation >> 2;
0144 continue;
0145 }
0146 return _base::string_ref("failed to get message from system");
0147 }
0148 }
0149
0150 public:
0151
0152 using value_type = win32::NTSTATUS;
0153 using _base::string_ref;
0154
0155 public:
0156
0157 constexpr explicit _nt_code_domain(typename _base::unique_id_type id = 0x93f3b4487e4af25b) noexcept
0158 : _base(id)
0159 {
0160 }
0161 _nt_code_domain(const _nt_code_domain &) = default;
0162 _nt_code_domain(_nt_code_domain &&) = default;
0163 _nt_code_domain &operator=(const _nt_code_domain &) = default;
0164 _nt_code_domain &operator=(_nt_code_domain &&) = default;
0165 ~_nt_code_domain() = default;
0166
0167
0168 static inline constexpr const _nt_code_domain &get();
0169
0170 virtual string_ref name() const noexcept override { return string_ref("NT domain"); }
0171
0172 virtual payload_info_t payload_info() const noexcept override
0173 {
0174 return {sizeof(value_type), sizeof(status_code_domain *) + sizeof(value_type),
0175 (alignof(value_type) > alignof(status_code_domain *)) ? alignof(value_type) : alignof(status_code_domain *)};
0176 }
0177
0178 protected:
0179 virtual bool _do_failure(const status_code<void> &code) const noexcept override
0180 {
0181 assert(code.domain() == *this);
0182 return static_cast<const nt_code &>(code).value() < 0;
0183 }
0184 virtual bool _do_equivalent(const status_code<void> &code1, const status_code<void> &code2) const noexcept override
0185 {
0186 assert(code1.domain() == *this);
0187 const auto &c1 = static_cast<const nt_code &>(code1);
0188 if(code2.domain() == *this)
0189 {
0190 const auto &c2 = static_cast<const nt_code &>(code2);
0191 return c1.value() == c2.value();
0192 }
0193 if(code2.domain() == generic_code_domain)
0194 {
0195 const auto &c2 = static_cast<const generic_code &>(code2);
0196 if(static_cast<int>(c2.value()) == _nt_code_to_errno(c1.value()))
0197 {
0198 return true;
0199 }
0200 }
0201 if(code2.domain() == win32_code_domain)
0202 {
0203 const auto &c2 = static_cast<const win32_code &>(code2);
0204 if(c2.value() == _nt_code_to_win32_code(c1.value()))
0205 {
0206 return true;
0207 }
0208 }
0209 return false;
0210 }
0211 virtual generic_code _generic_code(const status_code<void> &code) const noexcept override
0212 {
0213 assert(code.domain() == *this);
0214 const auto &c = static_cast<const nt_code &>(code);
0215 return generic_code(static_cast<errc>(_nt_code_to_errno(c.value())));
0216 }
0217 virtual string_ref _do_message(const status_code<void> &code) const noexcept override
0218 {
0219 assert(code.domain() == *this);
0220 const auto &c = static_cast<const nt_code &>(code);
0221 return _make_string_ref(c.value());
0222 }
0223 #if defined(_CPPUNWIND) || defined(__EXCEPTIONS) || defined(BOOST_OUTCOME_STANDARDESE_IS_IN_THE_HOUSE)
0224 BOOST_OUTCOME_SYSTEM_ERROR2_NORETURN virtual void _do_throw_exception(const status_code<void> &code) const override
0225 {
0226 assert(code.domain() == *this);
0227 const auto &c = static_cast<const nt_code &>(code);
0228 throw status_error<_nt_code_domain>(c);
0229 }
0230 #endif
0231 };
0232
0233 constexpr _nt_code_domain nt_code_domain;
0234 inline constexpr const _nt_code_domain &_nt_code_domain::get()
0235 {
0236 return nt_code_domain;
0237 }
0238
0239 BOOST_OUTCOME_SYSTEM_ERROR2_NAMESPACE_END
0240
0241 #endif