181 lines
3.9 KiB
C
181 lines
3.9 KiB
C
|
/*
|
||
|
* mptOSException.h
|
||
|
* ----------------
|
||
|
* Purpose: platform-specific exception/signal handling
|
||
|
* Notes : (currently none)
|
||
|
* Authors: OpenMPT Devs
|
||
|
* The OpenMPT source code is released under the BSD license. Read LICENSE for more details.
|
||
|
*/
|
||
|
|
||
|
|
||
|
#pragma once
|
||
|
|
||
|
#include "openmpt/all/BuildSettings.hpp"
|
||
|
|
||
|
#include "mptBaseMacros.h"
|
||
|
#include "mptBaseTypes.h"
|
||
|
|
||
|
#if MPT_OS_WINDOWS
|
||
|
#include <windows.h>
|
||
|
#endif // MPT_OS_WINDOWS
|
||
|
|
||
|
|
||
|
OPENMPT_NAMESPACE_BEGIN
|
||
|
|
||
|
|
||
|
#if MPT_OS_WINDOWS
|
||
|
|
||
|
|
||
|
namespace Windows
|
||
|
{
|
||
|
|
||
|
|
||
|
namespace SEH
|
||
|
{
|
||
|
|
||
|
|
||
|
struct Code
|
||
|
{
|
||
|
private:
|
||
|
DWORD m_Code;
|
||
|
public:
|
||
|
constexpr Code(DWORD code) noexcept
|
||
|
: m_Code(code)
|
||
|
{
|
||
|
return;
|
||
|
}
|
||
|
public:
|
||
|
constexpr DWORD code() const noexcept
|
||
|
{
|
||
|
return m_Code;
|
||
|
}
|
||
|
};
|
||
|
|
||
|
|
||
|
template <typename Tfn, typename Tfilter, typename Thandler>
|
||
|
auto TryFilterHandleThrow(const Tfn &fn, const Tfilter &filter, const Thandler &handler) -> decltype(fn())
|
||
|
{
|
||
|
static_assert(std::is_trivially_copy_assignable<decltype(fn())>::value);
|
||
|
static_assert(std::is_trivially_copy_constructible<decltype(fn())>::value);
|
||
|
static_assert(std::is_trivially_move_assignable<decltype(fn())>::value);
|
||
|
static_assert(std::is_trivially_move_constructible<decltype(fn())>::value);
|
||
|
DWORD code = 0;
|
||
|
__try
|
||
|
{
|
||
|
return fn();
|
||
|
} __except(filter(GetExceptionCode(), GetExceptionInformation()))
|
||
|
{
|
||
|
code = GetExceptionCode();
|
||
|
}
|
||
|
throw Windows::SEH::Code(code);
|
||
|
}
|
||
|
|
||
|
|
||
|
template <typename Tfn, typename Tfilter, typename Thandler>
|
||
|
void TryFilterHandleVoid(const Tfn &fn, const Tfilter &filter, const Thandler &handler)
|
||
|
{
|
||
|
static_assert(std::is_same<decltype(fn()), void>::value || std::is_trivially_copy_assignable<decltype(fn())>::value);
|
||
|
static_assert(std::is_same<decltype(fn()), void>::value || std::is_trivially_copy_constructible<decltype(fn())>::value);
|
||
|
static_assert(std::is_same<decltype(fn()), void>::value || std::is_trivially_move_assignable<decltype(fn())>::value);
|
||
|
static_assert(std::is_same<decltype(fn()), void>::value || std::is_trivially_move_constructible<decltype(fn())>::value);
|
||
|
__try
|
||
|
{
|
||
|
fn();
|
||
|
return;
|
||
|
} __except(filter(GetExceptionCode(), GetExceptionInformation()))
|
||
|
{
|
||
|
DWORD code = GetExceptionCode();
|
||
|
handler(code);
|
||
|
}
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
|
||
|
template <typename Tfn, typename Tfilter, typename Thandler>
|
||
|
auto TryFilterHandleDefault(const Tfn &fn, const Tfilter &filter, const Thandler &handler, decltype(fn()) def = decltype(fn()){}) -> decltype(fn())
|
||
|
{
|
||
|
static_assert(std::is_trivially_copy_assignable<decltype(fn())>::value);
|
||
|
static_assert(std::is_trivially_copy_constructible<decltype(fn())>::value);
|
||
|
static_assert(std::is_trivially_move_assignable<decltype(fn())>::value);
|
||
|
static_assert(std::is_trivially_move_constructible<decltype(fn())>::value);
|
||
|
auto result = def;
|
||
|
__try
|
||
|
{
|
||
|
result = fn();
|
||
|
} __except(filter(GetExceptionCode(), GetExceptionInformation()))
|
||
|
{
|
||
|
DWORD code = GetExceptionCode();
|
||
|
result = handler(code);
|
||
|
}
|
||
|
return result;
|
||
|
}
|
||
|
|
||
|
|
||
|
template <typename Tfn>
|
||
|
auto TryReturnOrThrow(const Tfn &fn) -> decltype(fn())
|
||
|
{
|
||
|
return TryFilterHandleThrow(
|
||
|
fn,
|
||
|
[](auto code, auto eptr)
|
||
|
{
|
||
|
MPT_UNREFERENCED_PARAMETER(code);
|
||
|
MPT_UNREFERENCED_PARAMETER(eptr);
|
||
|
return EXCEPTION_EXECUTE_HANDLER;
|
||
|
},
|
||
|
[](auto code)
|
||
|
{
|
||
|
throw Windows::SEH::Code(code);
|
||
|
});
|
||
|
}
|
||
|
|
||
|
|
||
|
template <typename Tfn>
|
||
|
DWORD TryOrError(const Tfn &fn)
|
||
|
{
|
||
|
DWORD result = DWORD{0};
|
||
|
TryFilterHandleVoid(
|
||
|
fn,
|
||
|
[](auto code, auto eptr)
|
||
|
{
|
||
|
MPT_UNREFERENCED_PARAMETER(code);
|
||
|
MPT_UNREFERENCED_PARAMETER(eptr);
|
||
|
return EXCEPTION_EXECUTE_HANDLER;
|
||
|
},
|
||
|
[&result](auto code)
|
||
|
{
|
||
|
result = code;
|
||
|
});
|
||
|
return result;
|
||
|
}
|
||
|
|
||
|
|
||
|
template <typename Tfn>
|
||
|
auto TryReturnOrDefault(const Tfn &fn, decltype(fn()) def = decltype(fn()){}) -> decltype(fn())
|
||
|
{
|
||
|
return TryFilterHandleDefault(
|
||
|
fn,
|
||
|
[](auto code, auto eptr)
|
||
|
{
|
||
|
MPT_UNREFERENCED_PARAMETER(code);
|
||
|
MPT_UNREFERENCED_PARAMETER(eptr);
|
||
|
return EXCEPTION_EXECUTE_HANDLER;
|
||
|
},
|
||
|
[def](auto code)
|
||
|
{
|
||
|
MPT_UNREFERENCED_PARAMETER(code);
|
||
|
return def;
|
||
|
});
|
||
|
}
|
||
|
|
||
|
|
||
|
} // namspace SEH
|
||
|
|
||
|
|
||
|
} // namespace Windows
|
||
|
|
||
|
|
||
|
#endif // MPT_OS_WINDOWS
|
||
|
|
||
|
|
||
|
OPENMPT_NAMESPACE_END
|