1
0
mirror of https://github.com/OpenRCT2/OpenRCT2 synced 2026-01-19 13:03:11 +01:00
Files
OpenRCT2/src/openrct2/core/Guard.cpp
2025-01-19 15:59:45 +00:00

156 lines
4.1 KiB
C++

/*****************************************************************************
* Copyright (c) 2014-2025 OpenRCT2 developers
*
* For a complete list of all authors, please refer to contributors.md
* Interested in contributing? Visit https://github.com/OpenRCT2/OpenRCT2
*
* OpenRCT2 is licensed under the GNU General Public License version 3.
*****************************************************************************/
#ifdef _WIN32
#include <cassert>
#include <windows.h>
#endif
#include "../Version.h"
#include "Console.hpp"
#include "Diagnostics.hpp"
#include "Guard.hpp"
#include "String.hpp"
#include "StringBuilder.h"
#include <cassert>
#include <cstdarg>
#include <cstdio>
#include <cstdlib>
namespace OpenRCT2::Guard
{
static constexpr const utf8* kAssertionMessage = "An assertion failed, please report this to the OpenRCT2 developers.";
// The default behaviour when an assertion is raised.
static ASSERT_BEHAVIOUR _assertBehaviour =
#ifdef _WIN32
ASSERT_BEHAVIOUR::MESSAGE_BOX
#else
ASSERT_BEHAVIOUR::CASSERT
#endif
;
static std::optional<std::string> _lastAssertMessage = std::nullopt;
#ifdef _WIN32
[[nodiscard]] static std::wstring CreateDialogAssertMessage(std::string_view);
static void ForceCrash();
#endif
ASSERT_BEHAVIOUR GetAssertBehaviour()
{
return _assertBehaviour;
}
void SetAssertBehaviour(ASSERT_BEHAVIOUR behaviour)
{
_assertBehaviour = behaviour;
}
void Assert(bool expression, const char* message, ...)
{
va_list args;
va_start(args, message);
Assert_VA(expression, message, args);
va_end(args);
}
void Assert_VA(bool expression, const char* message, va_list args)
{
if (expression)
return;
Console::Error::WriteLine(kAssertionMessage);
Console::Error::WriteLine("Version: %s", gVersionInfoFull);
// This is never freed, but acceptable considering we are about to crash out
std::string formattedMessage;
if (message != nullptr)
{
formattedMessage = String::formatVA(message, args);
Console::Error::WriteLine(formattedMessage.c_str());
_lastAssertMessage = std::make_optional(formattedMessage);
}
#ifdef DEBUG
Debug::Break();
#endif
switch (_assertBehaviour)
{
case ASSERT_BEHAVIOUR::ABORT:
abort();
default:
case ASSERT_BEHAVIOUR::CASSERT:
assert(false);
break;
#ifdef _WIN32
case ASSERT_BEHAVIOUR::MESSAGE_BOX:
{
// Show message box if we are not building for testing
auto buffer = CreateDialogAssertMessage(formattedMessage);
int32_t result = MessageBoxW(
nullptr, buffer.c_str(), L"" OPENRCT2_NAME, MB_ABORTRETRYIGNORE | MB_ICONEXCLAMATION);
if (result == IDABORT)
{
ForceCrash();
}
break;
}
#endif
}
}
void Fail(const char* message, ...)
{
va_list args;
va_start(args, message);
Assert_VA(false, message, args);
va_end(args);
}
void Fail_VA(const char* message, va_list args)
{
Assert_VA(false, message, args);
}
std::optional<std::string> GetLastAssertMessage()
{
return _lastAssertMessage;
}
#ifdef _WIN32
[[nodiscard]] static std::wstring CreateDialogAssertMessage(std::string_view formattedMessage)
{
StringBuilder sb;
sb.Append(kAssertionMessage);
sb.Append("\n\n");
sb.Append("Version: ");
sb.Append(gVersionInfoFull);
if (!formattedMessage.empty())
{
sb.Append("\n");
sb.Append(formattedMessage);
}
return String::toWideChar({ sb.GetBuffer(), sb.GetLength() });
}
static void ForceCrash()
{
#ifdef USE_BREAKPAD
// Force a crash that breakpad will handle allowing us to get a dump
*((void**)0) = 0;
#else
assert(false);
#endif
}
#endif
} // namespace OpenRCT2::Guard