#pragma once #include "../DefaultInclude.hpp" #include namespace geode { class Mod; Mod* getMod(); } namespace geode::utils { #ifdef GEODE_IS_WINDOWS static constexpr size_t GEODE_TERMINATE_EXCEPTION_CODE = 0x4000; static constexpr size_t GEODE_UNREACHABLE_EXCEPTION_CODE = 0x4001; static constexpr bool isGeodeExceptionCode(size_t code) { return GEODE_TERMINATE_EXCEPTION_CODE <= code && code <= GEODE_UNREACHABLE_EXCEPTION_CODE; } #else static constexpr size_t GEODE_TERMINATE_EXCEPTION_CODE = 0; static constexpr size_t GEODE_UNREACHABLE_EXCEPTION_CODE = 0; static constexpr bool isGeodeExceptionCode(size_t code) { return false; } #endif namespace detail { // This needs to do stuff with `Mod*` which is not included in the file GEODE_DLL void logTerminationError(const char* reason, Mod* mod); } template [[noreturn]] void terminate(std::string const& reason, Mod* mod = getMod(), size_t platformCode = GEODE_TERMINATE_EXCEPTION_CODE) { // Add the error to the logfile detail::logTerminationError(reason.c_str(), mod); #ifdef GEODE_IS_WINDOWS // If a debugger is attached, start debugging if (IsDebuggerPresent()) { OutputDebugStringA(reason.c_str()); DebugBreak(); } // Otherwise terminate by raising an exception (which is caught by the crashlog handler) else { std::array errorList { static_cast(reason.c_str()), mod }; RaiseException( platformCode, EXCEPTION_NONCONTINUABLE, 2, reinterpret_cast(errorList.data()) ); } #endif std::terminate(); } template [[noreturn]] void unreachable(std::string const& reason = "Unspecified", Mod* mod = getMod()) { terminate(reason, mod, GEODE_UNREACHABLE_EXCEPTION_CODE); } }