// 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 // the Free Software Foundation, version 2.0 or later versions. // 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/ #include #include #include "ppsspp_config.h" #include "Common/CommonTypes.h" #include "Common/Log.h" #include "StringUtils.h" #include "Common/Data/Encoding/Utf8.h" #include "Common/Thread/ThreadUtil.h" #include "Common/TimeUtil.h" #if PPSSPP_PLATFORM(ANDROID) #include #elif PPSSPP_PLATFORM(WINDOWS) #include "CommonWindows.h" static HWND g_dialogParent; #endif #define LOG_BUF_SIZE 2048 static bool hitAnyAsserts = false; static std::mutex g_extraAssertInfoMutex; static std::string g_extraAssertInfo = "menu"; static double g_assertInfoTime = 0.0; static bool g_exitOnAssert; static AssertNoCallbackFunc g_assertCancelCallback = 0; static void *g_assertCancelCallbackUserData = 0; void SetAssertDialogParent(void *handle) { #if PPSSPP_PLATFORM(WINDOWS) g_dialogParent = (HWND)handle; #endif } void SetExtraAssertInfo(const char *info) { std::lock_guard guard(g_extraAssertInfoMutex); g_extraAssertInfo = info ? info : "menu"; g_assertInfoTime = time_now_d(); } void SetAssertCancelCallback(AssertNoCallbackFunc callback, void *userdata) { g_assertCancelCallback = callback; g_assertCancelCallbackUserData = userdata; } void SetCleanExitOnAssert() { g_exitOnAssert = true; } void BreakIntoPSPDebugger(const char *reason) { if (g_assertCancelCallback) { g_assertCancelCallback(reason, g_assertCancelCallbackUserData); } } bool HandleAssert(const char *function, const char *file, int line, const char *expression, const char* format, ...) { // Read message and write it to the log char text[LOG_BUF_SIZE]; const char *caption = "Critical"; va_list args; va_start(args, format); vsnprintf(text, sizeof(text), format, args); va_end(args); // Secondary formatting. Wonder if this can be combined into the vsnprintf somehow. char formatted[LOG_BUF_SIZE + 128]; { std::lock_guard guard(g_extraAssertInfoMutex); 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); } // Normal logging (will also log to Android log) 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?) fprintf(stderr, "%s\n", formatted); hitAnyAsserts = true; #if defined(USING_WIN_UI) // Avoid hanging on CI. if (!getenv("CI")) { const int msgBoxStyle = MB_ICONINFORMATION | MB_YESNOCANCEL; std::string text = formatted; text += "\n\nTry to continue?"; if (IsDebuggerPresent()) { text += "\n\nNo: break directly into the native debugger"; text += "\n\nCancel: skip and break into PPSSPP debugger"; } else { text += "\n\nNo: exit"; text += "\n\nCancel: skip and break into PPSSPP debugger"; } const char *threadName = GetCurrentThreadName(); OutputDebugStringA(formatted); printf("%s\n", formatted); std::wstring wcaption = ConvertUTF8ToWString(std::string(caption) + " " + (threadName ? threadName : "(unknown thread)")); switch (MessageBox(g_dialogParent, ConvertUTF8ToWString(text).c_str(), wcaption.c_str(), msgBoxStyle)) { case IDYES: return true; case IDNO: if (g_exitOnAssert || !IsDebuggerPresent()) { // Hard exit. ExitProcess(1); } return false; // Break into the native debugger. case IDCANCEL: g_assertCancelCallback(formatted, g_assertCancelCallbackUserData); return true; // don't crash! } } return false; #elif PPSSPP_PLATFORM(ANDROID) __android_log_assert(expression, "PPSSPP", "%s", formatted); // Doesn't matter what we return here. return false; #else OutputDebugStringUTF8(text); return false; #endif } // These are mainly used for unit testing. bool HitAnyAsserts() { return hitAnyAsserts; } void ResetHitAnyAsserts() { hitAnyAsserts = false; }