2012-11-01 15:19:01 +00:00
|
|
|
// Copyright (C) 2003 Dolphin Project.
|
|
|
|
|
|
|
|
// This program is free software: you can redistribute it and/or modify
|
|
|
|
// it under the terms of the GNU General Public License as published by
|
2012-11-04 22:01:49 +00:00
|
|
|
// the Free Software Foundation, version 2.0 or later versions.
|
2012-11-01 15:19:01 +00:00
|
|
|
|
|
|
|
// This program is distributed in the hope that it will be useful,
|
|
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
// GNU General Public License 2.0 for more details.
|
|
|
|
|
|
|
|
// A copy of the GPL 2.0 should have been included with the program.
|
|
|
|
// If not, see http://www.gnu.org/licenses/
|
|
|
|
|
|
|
|
// Official SVN repository and contact information can be found at
|
|
|
|
// http://code.google.com/p/dolphin-emu/
|
|
|
|
|
2017-11-20 10:57:54 +00:00
|
|
|
#include <string>
|
2022-12-16 13:55:56 +00:00
|
|
|
#include <mutex>
|
2017-11-20 10:57:54 +00:00
|
|
|
|
2017-03-05 01:21:25 +00:00
|
|
|
#include "ppsspp_config.h"
|
|
|
|
|
2022-01-30 23:49:02 +00:00
|
|
|
#include "Common/CommonTypes.h"
|
2020-08-15 10:25:39 +00:00
|
|
|
#include "Common/Log.h"
|
2013-05-03 06:22:15 +00:00
|
|
|
#include "StringUtils.h"
|
2020-10-01 11:05:04 +00:00
|
|
|
#include "Common/Data/Encoding/Utf8.h"
|
2023-03-15 21:26:23 +00:00
|
|
|
#include "Common/Thread/ThreadUtil.h"
|
2023-09-23 06:53:24 +00:00
|
|
|
#include "Common/TimeUtil.h"
|
2012-11-01 15:19:01 +00:00
|
|
|
|
2020-08-16 11:41:16 +00:00
|
|
|
#if PPSSPP_PLATFORM(ANDROID)
|
|
|
|
#include <android/log.h>
|
|
|
|
#elif PPSSPP_PLATFORM(WINDOWS)
|
2020-08-16 11:27:28 +00:00
|
|
|
#include "CommonWindows.h"
|
|
|
|
#endif
|
2012-11-01 15:19:01 +00:00
|
|
|
|
2020-08-16 12:11:56 +00:00
|
|
|
#define LOG_BUF_SIZE 2048
|
2020-08-16 11:48:31 +00:00
|
|
|
|
2022-01-30 02:32:49 +00:00
|
|
|
static bool hitAnyAsserts = false;
|
|
|
|
|
2024-04-16 17:54:48 +00:00
|
|
|
static std::mutex g_extraAssertInfoMutex;
|
|
|
|
static std::string g_extraAssertInfo = "menu";
|
|
|
|
static double g_assertInfoTime = 0.0;
|
|
|
|
static bool g_exitOnAssert;
|
2022-12-16 13:55:56 +00:00
|
|
|
|
|
|
|
void SetExtraAssertInfo(const char *info) {
|
|
|
|
std::lock_guard<std::mutex> guard(g_extraAssertInfoMutex);
|
2022-12-17 21:17:29 +00:00
|
|
|
g_extraAssertInfo = info ? info : "menu";
|
2023-09-23 06:53:24 +00:00
|
|
|
g_assertInfoTime = time_now_d();
|
2022-12-16 13:55:56 +00:00
|
|
|
}
|
|
|
|
|
2024-04-16 17:54:48 +00:00
|
|
|
void SetCleanExitOnAssert() {
|
|
|
|
g_exitOnAssert = true;
|
|
|
|
}
|
|
|
|
|
2020-08-16 12:11:56 +00:00
|
|
|
bool HandleAssert(const char *function, const char *file, int line, const char *expression, const char* format, ...) {
|
2012-11-01 15:19:01 +00:00
|
|
|
// Read message and write it to the log
|
2020-08-16 12:11:56 +00:00
|
|
|
char text[LOG_BUF_SIZE];
|
2020-08-16 11:27:28 +00:00
|
|
|
const char *caption = "Critical";
|
2012-11-01 15:19:01 +00:00
|
|
|
va_list args;
|
|
|
|
va_start(args, format);
|
2020-08-16 11:41:16 +00:00
|
|
|
vsnprintf(text, sizeof(text), format, args);
|
2012-11-01 15:19:01 +00:00
|
|
|
va_end(args);
|
2020-08-16 11:41:16 +00:00
|
|
|
|
2020-08-16 12:11:56 +00:00
|
|
|
// Secondary formatting. Wonder if this can be combined into the vsnprintf somehow.
|
2021-05-01 06:31:54 +00:00
|
|
|
char formatted[LOG_BUF_SIZE + 128];
|
2022-12-16 13:55:56 +00:00
|
|
|
{
|
|
|
|
std::lock_guard<std::mutex> guard(g_extraAssertInfoMutex);
|
2023-09-23 06:53:24 +00:00
|
|
|
double delta = time_now_d() - g_assertInfoTime;
|
|
|
|
snprintf(formatted, sizeof(formatted), "(%s:%s:%d): [%s] (%s, %0.1fs) %s", file, function, line, expression, g_extraAssertInfo.c_str(), delta, text);
|
2022-12-16 13:55:56 +00:00
|
|
|
}
|
2020-08-16 12:11:56 +00:00
|
|
|
|
2017-11-20 10:57:54 +00:00
|
|
|
// Normal logging (will also log to Android log)
|
2024-07-14 12:42:59 +00:00
|
|
|
ERROR_LOG(Log::System, "%s", formatted);
|
|
|
|
// Also do a simple printf for good measure, in case logging of System is disabled (should we disallow that?)
|
2022-01-22 04:22:12 +00:00
|
|
|
fprintf(stderr, "%s\n", formatted);
|
2012-11-01 15:19:01 +00:00
|
|
|
|
2022-01-30 02:32:49 +00:00
|
|
|
hitAnyAsserts = true;
|
|
|
|
|
2016-01-03 23:10:50 +00:00
|
|
|
#if defined(USING_WIN_UI)
|
2022-01-30 02:32:49 +00:00
|
|
|
// Avoid hanging on CI.
|
|
|
|
if (!getenv("CI")) {
|
|
|
|
int msgBoxStyle = MB_ICONINFORMATION | MB_YESNO;
|
|
|
|
std::wstring wtext = ConvertUTF8ToWString(formatted) + L"\n\nTry to continue?";
|
2023-03-15 21:26:23 +00:00
|
|
|
std::wstring wcaption = ConvertUTF8ToWString(std::string(caption) + " " + GetCurrentThreadName());
|
2022-01-30 02:32:49 +00:00
|
|
|
OutputDebugString(wtext.c_str());
|
2024-04-16 17:54:48 +00:00
|
|
|
printf("%s\n", formatted);
|
2022-01-30 02:32:49 +00:00
|
|
|
if (IDYES != MessageBox(0, wtext.c_str(), wcaption.c_str(), msgBoxStyle)) {
|
2024-04-16 17:54:48 +00:00
|
|
|
if (g_exitOnAssert) {
|
|
|
|
// Hard exit.
|
|
|
|
ExitProcess(1);
|
|
|
|
return false;
|
|
|
|
}
|
2022-01-30 02:32:49 +00:00
|
|
|
} else {
|
|
|
|
return true;
|
|
|
|
}
|
2020-08-16 12:11:56 +00:00
|
|
|
}
|
2022-01-30 02:32:49 +00:00
|
|
|
return false;
|
2020-08-16 12:11:56 +00:00
|
|
|
#elif PPSSPP_PLATFORM(ANDROID)
|
|
|
|
__android_log_assert(expression, "PPSSPP", "%s", formatted);
|
|
|
|
// Doesn't matter what we return here.
|
|
|
|
return false;
|
2012-11-01 15:19:01 +00:00
|
|
|
#else
|
2020-08-16 11:27:28 +00:00
|
|
|
OutputDebugStringUTF8(text);
|
2017-11-20 10:57:54 +00:00
|
|
|
return false;
|
2012-11-01 15:19:01 +00:00
|
|
|
#endif
|
|
|
|
}
|
2022-01-30 02:32:49 +00:00
|
|
|
|
|
|
|
// These are mainly used for unit testing.
|
|
|
|
bool HitAnyAsserts() {
|
|
|
|
return hitAnyAsserts;
|
|
|
|
}
|
|
|
|
void ResetHitAnyAsserts() {
|
|
|
|
hitAnyAsserts = false;
|
|
|
|
}
|