229 lines
8.5 KiB
C++
229 lines
8.5 KiB
C++
/* Proposed SG14 status_code
|
|
(C) 2018-2022 Niall Douglas <http://www.nedproductions.biz/> (5 commits)
|
|
File Created: Feb 2018
|
|
|
|
|
|
Boost Software License - Version 1.0 - August 17th, 2003
|
|
|
|
Permission is hereby granted, free of charge, to any person or organization
|
|
obtaining a copy of the software and accompanying documentation covered by
|
|
this license (the "Software") to use, reproduce, display, distribute,
|
|
execute, and transmit the Software, and to prepare derivative works of the
|
|
Software, and to permit third-parties to whom the Software is furnished to
|
|
do so, all subject to the following:
|
|
|
|
The copyright notices in the Software and this entire statement, including
|
|
the above license grant, this restriction and the following disclaimer,
|
|
must be included in all copies of the Software, in whole or in part, and
|
|
all derivative works of the Software, unless such copies or derivative
|
|
works are solely in the form of machine-executable object code generated by
|
|
a source language processor.
|
|
|
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
FITNESS FOR A PARTICULAR PURPOSE, TITLE AND NON-INFRINGEMENT. IN NO EVENT
|
|
SHALL THE COPYRIGHT HOLDERS OR ANYONE DISTRIBUTING THE SOFTWARE BE LIABLE
|
|
FOR ANY DAMAGES OR OTHER LIABILITY, WHETHER IN CONTRACT, TORT OR OTHERWISE,
|
|
ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
|
|
DEALINGS IN THE SOFTWARE.
|
|
*/
|
|
|
|
#ifndef BOOST_OUTCOME_SYSTEM_ERROR2_NT_CODE_HPP
|
|
#define BOOST_OUTCOME_SYSTEM_ERROR2_NT_CODE_HPP
|
|
|
|
#if !defined(_WIN32) && !defined(BOOST_OUTCOME_STANDARDESE_IS_IN_THE_HOUSE)
|
|
#error This file should only be included on Windows
|
|
#endif
|
|
|
|
#include "win32_code.hpp"
|
|
|
|
BOOST_OUTCOME_SYSTEM_ERROR2_NAMESPACE_BEGIN
|
|
|
|
//! \exclude
|
|
namespace win32
|
|
{
|
|
// A Win32 NTSTATUS
|
|
using NTSTATUS = long;
|
|
// A Win32 HMODULE
|
|
using HMODULE = void *;
|
|
// Used to retrieve where the NTDLL DLL is mapped into memory
|
|
extern HMODULE __stdcall GetModuleHandleW(const wchar_t *lpModuleName);
|
|
#pragma comment(lib, "kernel32.lib")
|
|
#if(defined(__x86_64__) || defined(_M_X64)) || (defined(__aarch64__) || defined(_M_ARM64))
|
|
#pragma comment(linker, "/alternatename:?GetModuleHandleW@win32@system_error2@@YAPEAXPEB_W@Z=GetModuleHandleW")
|
|
#elif defined(__x86__) || defined(_M_IX86) || defined(__i386__)
|
|
#pragma comment(linker, "/alternatename:?GetModuleHandleW@win32@system_error2@@YGPAXPB_W@Z=_GetModuleHandleW@4")
|
|
#elif defined(__arm__) || defined(_M_ARM)
|
|
#pragma comment(linker, "/alternatename:?GetModuleHandleW@win32@system_error2@@YAPAXPB_W@Z=GetModuleHandleW")
|
|
#else
|
|
#error Unknown architecture
|
|
#endif
|
|
} // namespace win32
|
|
|
|
class _nt_code_domain;
|
|
//! (Windows only) A NT error code, those returned by NT kernel functions.
|
|
using nt_code = status_code<_nt_code_domain>;
|
|
//! (Windows only) A specialisation of `status_error` for the NT error code domain.
|
|
using nt_error = status_error<_nt_code_domain>;
|
|
|
|
/*! (Windows only) The implementation of the domain for NT error codes, those returned by NT kernel functions.
|
|
*/
|
|
class _nt_code_domain : public status_code_domain
|
|
{
|
|
template <class DomainType> friend class status_code;
|
|
template <class StatusCode> friend class detail::indirecting_domain;
|
|
friend class _com_code_domain;
|
|
using _base = status_code_domain;
|
|
static int _nt_code_to_errno(win32::NTSTATUS c)
|
|
{
|
|
if(c >= 0)
|
|
{
|
|
return 0; // success
|
|
}
|
|
switch(static_cast<unsigned>(c))
|
|
{
|
|
#include "detail/nt_code_to_generic_code.ipp"
|
|
}
|
|
return -1;
|
|
}
|
|
static win32::DWORD _nt_code_to_win32_code(win32::NTSTATUS c) // NOLINT
|
|
{
|
|
if(c >= 0)
|
|
{
|
|
return 0; // success
|
|
}
|
|
switch(static_cast<unsigned>(c))
|
|
{
|
|
#include "detail/nt_code_to_win32_code.ipp"
|
|
}
|
|
return static_cast<win32::DWORD>(-1);
|
|
}
|
|
//! Construct from a NT error code
|
|
static _base::string_ref _make_string_ref(win32::NTSTATUS c) noexcept
|
|
{
|
|
wchar_t buffer[32768];
|
|
static win32::HMODULE ntdll = win32::GetModuleHandleW(L"NTDLL.DLL");
|
|
win32::DWORD wlen = win32::FormatMessageW(0x00000800 /*FORMAT_MESSAGE_FROM_HMODULE*/ | 0x00001000 /*FORMAT_MESSAGE_FROM_SYSTEM*/ | 0x00000200 /*FORMAT_MESSAGE_IGNORE_INSERTS*/, ntdll, c, (1 << 10) /*MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT)*/, buffer, 32768, nullptr);
|
|
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)); // NOLINT
|
|
if(p == nullptr)
|
|
{
|
|
return _base::string_ref("failed to get message from system");
|
|
}
|
|
bytes = win32::WideCharToMultiByte(65001 /*CP_UTF8*/, 0, buffer, (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; // NOLINT
|
|
return _base::atomic_refcounted_string_ref(p, end - p);
|
|
}
|
|
free(p); // NOLINT
|
|
if(win32::GetLastError() == 0x7a /*ERROR_INSUFFICIENT_BUFFER*/)
|
|
{
|
|
allocation += allocation >> 2;
|
|
continue;
|
|
}
|
|
return _base::string_ref("failed to get message from system");
|
|
}
|
|
}
|
|
|
|
public:
|
|
//! The value type of the NT code, which is a `win32::NTSTATUS`
|
|
using value_type = win32::NTSTATUS;
|
|
using _base::string_ref;
|
|
|
|
public:
|
|
//! Default constructor
|
|
constexpr explicit _nt_code_domain(typename _base::unique_id_type id = 0x93f3b4487e4af25b) noexcept
|
|
: _base(id)
|
|
{
|
|
}
|
|
_nt_code_domain(const _nt_code_domain &) = default;
|
|
_nt_code_domain(_nt_code_domain &&) = default;
|
|
_nt_code_domain &operator=(const _nt_code_domain &) = default;
|
|
_nt_code_domain &operator=(_nt_code_domain &&) = default;
|
|
~_nt_code_domain() = default;
|
|
|
|
//! Constexpr singleton getter. Returns the constexpr nt_code_domain variable.
|
|
static inline constexpr const _nt_code_domain &get();
|
|
|
|
virtual string_ref name() const noexcept override { return string_ref("NT domain"); } // NOLINT
|
|
|
|
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 // NOLINT
|
|
{
|
|
assert(code.domain() == *this);
|
|
return static_cast<const nt_code &>(code).value() < 0; // NOLINT
|
|
}
|
|
virtual bool _do_equivalent(const status_code<void> &code1, const status_code<void> &code2) const noexcept override // NOLINT
|
|
{
|
|
assert(code1.domain() == *this);
|
|
const auto &c1 = static_cast<const nt_code &>(code1); // NOLINT
|
|
if(code2.domain() == *this)
|
|
{
|
|
const auto &c2 = static_cast<const nt_code &>(code2); // NOLINT
|
|
return c1.value() == c2.value();
|
|
}
|
|
if(code2.domain() == generic_code_domain)
|
|
{
|
|
const auto &c2 = static_cast<const generic_code &>(code2); // NOLINT
|
|
if(static_cast<int>(c2.value()) == _nt_code_to_errno(c1.value()))
|
|
{
|
|
return true;
|
|
}
|
|
}
|
|
if(code2.domain() == win32_code_domain)
|
|
{
|
|
const auto &c2 = static_cast<const win32_code &>(code2); // NOLINT
|
|
if(c2.value() == _nt_code_to_win32_code(c1.value()))
|
|
{
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
virtual generic_code _generic_code(const status_code<void> &code) const noexcept override // NOLINT
|
|
{
|
|
assert(code.domain() == *this);
|
|
const auto &c = static_cast<const nt_code &>(code); // NOLINT
|
|
return generic_code(static_cast<errc>(_nt_code_to_errno(c.value())));
|
|
}
|
|
virtual string_ref _do_message(const status_code<void> &code) const noexcept override // NOLINT
|
|
{
|
|
assert(code.domain() == *this);
|
|
const auto &c = static_cast<const nt_code &>(code); // NOLINT
|
|
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 // NOLINT
|
|
{
|
|
assert(code.domain() == *this);
|
|
const auto &c = static_cast<const nt_code &>(code); // NOLINT
|
|
throw status_error<_nt_code_domain>(c);
|
|
}
|
|
#endif
|
|
};
|
|
//! (Windows only) A constexpr source variable for the NT code domain, which is that of NT kernel functions. Returned by `_nt_code_domain::get()`.
|
|
constexpr _nt_code_domain nt_code_domain;
|
|
inline constexpr const _nt_code_domain &_nt_code_domain::get()
|
|
{
|
|
return nt_code_domain;
|
|
}
|
|
|
|
BOOST_OUTCOME_SYSTEM_ERROR2_NAMESPACE_END
|
|
|
|
#endif
|