mirror of
https://foundry.openuru.org/gitblit/r/CWE-ou-minkata.git
synced 2025-07-18 11:19:10 +00:00
Implement pfCrashHandler
pfCrashHandler includes a client that watches plClient for crashes. When it detects a crash, it lets the plCrashSrv (plCrashHandler.exe) know about it. We then produce a mninidump, then signal the client that it's OK to show a crash dialog. See http://www.nynaeve.net/?p=128 for a good explanation of why I split the crash logic into another process.
This commit is contained in:
@ -1,5 +1,6 @@
|
||||
add_subdirectory(plClient)
|
||||
add_subdirectory(plClientPatcher)
|
||||
add_subdirectory(plCrashHandler)
|
||||
add_subdirectory(plPythonPack)
|
||||
add_subdirectory(plUruLauncher)
|
||||
add_subdirectory(plFileSecure)
|
||||
|
@ -79,6 +79,7 @@ target_link_libraries(plClient pfCharacter)
|
||||
target_link_libraries(plClient pfConditional)
|
||||
target_link_libraries(plClient pfConsole)
|
||||
target_link_libraries(plClient pfConsoleCore)
|
||||
target_link_libraries(plClient pfCrashHandler)
|
||||
target_link_libraries(plClient pfCsrSrv)
|
||||
target_link_libraries(plClient pfGameGUIMgr)
|
||||
target_link_libraries(plClient pfGameMgr)
|
||||
|
@ -56,6 +56,7 @@ You can contact Cyan Worlds, Inc. by email legal@cyan.com
|
||||
|
||||
#include "plClient.h"
|
||||
#include "plClientResMgr/plClientResMgr.h"
|
||||
#include "pfCrashHandler/plCrashCli.h"
|
||||
#include "plNetClient/plNetClientMgr.h"
|
||||
#include "plNetClient/plNetLinkingMgr.h"
|
||||
#include "plInputCore/plInputManager.h"
|
||||
@ -120,6 +121,10 @@ plClient *gClient;
|
||||
bool gPendingActivate = false;
|
||||
bool gPendingActivateFlag = false;
|
||||
|
||||
#ifndef HS_DEBUGGING
|
||||
static plCrashCli s_crash;
|
||||
#endif
|
||||
|
||||
static bool s_loginDlgRunning = false;
|
||||
static CEvent s_statusEvent(kEventManualReset);
|
||||
static UINT s_WmTaskbarList = RegisterWindowMessage("TaskbarButtonCreated");
|
||||
@ -1343,10 +1348,6 @@ BOOL CALLBACK SplashDialogProc( HWND hwndDlg, UINT uMsg, WPARAM wParam, LPARAM l
|
||||
return 0;
|
||||
}
|
||||
|
||||
static char sStackTraceMsg[ 10240 ] = "";
|
||||
void printStackTrace( char* buffer, int bufferSize, unsigned long stackPtr = 0, unsigned long opPtr = 0 );
|
||||
//void StackTraceFromContext( HANDLE hThread, CONTEXT *context, char *outputBuffer );
|
||||
|
||||
BOOL CALLBACK ExceptionDialogProc( HWND hwndDlg, UINT uMsg, WPARAM wParam, LPARAM lParam )
|
||||
{
|
||||
static char *sLastMsg = nil;
|
||||
@ -1383,63 +1384,21 @@ BOOL CALLBACK ExceptionDialogProc( HWND hwndDlg, UINT uMsg, WPARAM wParam, LPARA
|
||||
return 0;
|
||||
}
|
||||
|
||||
|
||||
#ifndef HS_DEBUGGING
|
||||
LONG WINAPI plCustomUnhandledExceptionFilter( struct _EXCEPTION_POINTERS *ExceptionInfo )
|
||||
{
|
||||
const char *type = nil;
|
||||
switch( ExceptionInfo->ExceptionRecord->ExceptionCode )
|
||||
{
|
||||
case EXCEPTION_ACCESS_VIOLATION: type = "Access violation"; break;
|
||||
case EXCEPTION_ARRAY_BOUNDS_EXCEEDED: type = "Array bounds exceeded"; break;
|
||||
case EXCEPTION_BREAKPOINT: type = "Breakpoint"; break;
|
||||
case EXCEPTION_DATATYPE_MISALIGNMENT: type = "Datatype misalignment"; break;
|
||||
case EXCEPTION_FLT_DENORMAL_OPERAND: type = "Floating operand denormal"; break;
|
||||
case EXCEPTION_FLT_DIVIDE_BY_ZERO: type = "Floating-point divide-by-zero"; break;
|
||||
case EXCEPTION_FLT_INEXACT_RESULT: type = "Floating-point inexact result"; break;
|
||||
case EXCEPTION_FLT_INVALID_OPERATION: type = "Floating-point invalid operation"; break;
|
||||
case EXCEPTION_FLT_OVERFLOW: type = "Floating-point overflow"; break;
|
||||
case EXCEPTION_FLT_STACK_CHECK: type = "Floating-point stack error"; break;
|
||||
case EXCEPTION_FLT_UNDERFLOW: type = "Floating-point underflow"; break;
|
||||
case EXCEPTION_ILLEGAL_INSTRUCTION: type = "Illegal instruction"; break;
|
||||
case EXCEPTION_IN_PAGE_ERROR: type = "Exception in page"; break;
|
||||
case EXCEPTION_INT_DIVIDE_BY_ZERO: type = "Integer divide-by-zero"; break;
|
||||
case EXCEPTION_INT_OVERFLOW: type = "Integer overflow"; break;
|
||||
case EXCEPTION_INVALID_DISPOSITION: type = "Invalid disposition"; break;
|
||||
case EXCEPTION_NONCONTINUABLE_EXCEPTION: type = "Noncontinuable exception"; break;
|
||||
case EXCEPTION_PRIV_INSTRUCTION: type = "Private instruction"; break;
|
||||
case EXCEPTION_SINGLE_STEP: type = "Single-step"; break;
|
||||
case EXCEPTION_STACK_OVERFLOW: type = "Stack overflow"; break;
|
||||
}
|
||||
// Before we do __ANYTHING__, pass the exception to plCrashHandler
|
||||
s_crash.ReportCrash(ExceptionInfo);
|
||||
|
||||
char prodName[256];
|
||||
wchar_t productString[256];
|
||||
ProductString(productString, arrsize(productString));
|
||||
StrToAnsi(prodName, productString, arrsize(prodName));
|
||||
// Now, try to create a nice exception dialog after plCrashHandler is done.
|
||||
s_crash.WaitForHandle();
|
||||
HWND parentHwnd = (gClient == nil) ? GetActiveWindow() : gClient->GetWindowHandle();
|
||||
DialogBoxParam(gHInst, MAKEINTRESOURCE(IDD_EXCEPTION), parentHwnd, ExceptionDialogProc, NULL);
|
||||
|
||||
sprintf( sStackTraceMsg, "%s\r\nException type: %s\r\n", prodName, ( type != nil ) ? type : "(unknown)" );
|
||||
|
||||
printStackTrace( sStackTraceMsg, sizeof( sStackTraceMsg ), ExceptionInfo->ContextRecord->Ebp, (unsigned long)ExceptionInfo->ExceptionRecord->ExceptionAddress );
|
||||
|
||||
/// Print the info out to a log file as well
|
||||
hsUNIXStream log;
|
||||
wchar_t fileAndPath[MAX_PATH];
|
||||
PathGetLogDirectory(fileAndPath, arrsize(fileAndPath));
|
||||
PathAddFilename(fileAndPath, fileAndPath, L"stackDump.log", arrsize(fileAndPath));
|
||||
if( log.Open( fileAndPath, L"wt" ) )
|
||||
{
|
||||
log.WriteString( sStackTraceMsg );
|
||||
log.Close();
|
||||
}
|
||||
|
||||
/// Hopefully we can access this resource, even given the exception (i.e. very-bad-error) we just experienced
|
||||
if(TGIsCider || (::DialogBoxParam( gHInst, MAKEINTRESOURCE( IDD_EXCEPTION ), ( gClient != nil ) ? gClient->GetWindowHandle() : nil,
|
||||
ExceptionDialogProc, (LPARAM)sStackTraceMsg ) == -1) )
|
||||
{
|
||||
// The dialog failed, so just fallback to a standard message box
|
||||
hsMessageBox( sStackTraceMsg, "UruExplorer Exception", hsMessageBoxNormal );
|
||||
}
|
||||
// Trickle up the handlers
|
||||
return EXCEPTION_EXECUTE_HANDLER;
|
||||
}
|
||||
#endif
|
||||
|
||||
#include "pfConsoleCore/pfConsoleEngine.h"
|
||||
PF_CONSOLE_LINK_ALL()
|
||||
|
22
Sources/Plasma/Apps/plCrashHandler/CMakeLists.txt
Normal file
22
Sources/Plasma/Apps/plCrashHandler/CMakeLists.txt
Normal file
@ -0,0 +1,22 @@
|
||||
include_directories("../../CoreLib")
|
||||
include_directories("../../NucleusLib")
|
||||
include_directories("../../FeatureLib")
|
||||
|
||||
set(plCrashHandler_SOURCES
|
||||
main.cpp
|
||||
)
|
||||
|
||||
add_executable(plCrashHandler ${plCrashHandler_SOURCES})
|
||||
if(PLASMA_EXTERNAL_RELEASE)
|
||||
set_target_properties(plClient PROPERTIES OUTPUT_NAME "UruCrashHandler")
|
||||
endif(PLASMA_EXTERNAL_RELEASE)
|
||||
target_link_libraries(plCrashHandler CoreLib)
|
||||
target_link_libraries(plCrashHandler pfCrashHandler)
|
||||
target_link_libraries(plCrashHandler plFile)
|
||||
target_link_libraries(plCrashHandler pnProduct)
|
||||
target_link_libraries(plCrashHandler pnUtils)
|
||||
|
||||
# Platform specific libs
|
||||
if(WIN32)
|
||||
target_link_libraries(plCrashHandler Dbghelp)
|
||||
endif(WIN32)
|
72
Sources/Plasma/Apps/plCrashHandler/main.cpp
Normal file
72
Sources/Plasma/Apps/plCrashHandler/main.cpp
Normal file
@ -0,0 +1,72 @@
|
||||
/*==LICENSE==*
|
||||
|
||||
CyanWorlds.com Engine - MMOG client, server and tools
|
||||
Copyright (C) 2011 Cyan Worlds, Inc.
|
||||
|
||||
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, either version 3 of the License, or
|
||||
(at your option) any later version.
|
||||
|
||||
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 for more details.
|
||||
|
||||
You should have received a copy of the GNU General Public License
|
||||
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||
|
||||
Additional permissions under GNU GPL version 3 section 7
|
||||
|
||||
If you modify this Program, or any covered work, by linking or
|
||||
combining it with any of RAD Game Tools Bink SDK, Autodesk 3ds Max SDK,
|
||||
NVIDIA PhysX SDK, Microsoft DirectX SDK, OpenSSL library, Independent
|
||||
JPEG Group JPEG library, Microsoft Windows Media SDK, or Apple QuickTime SDK
|
||||
(or a modified version of those libraries),
|
||||
containing parts covered by the terms of the Bink SDK EULA, 3ds Max EULA,
|
||||
PhysX SDK EULA, DirectX SDK EULA, OpenSSL and SSLeay licenses, IJG
|
||||
JPEG Library README, Windows Media SDK EULA, or QuickTime SDK EULA, the
|
||||
licensors of this Program grant you additional
|
||||
permission to convey the resulting work. Corresponding Source for a
|
||||
non-source form of such a combination shall include the source code for
|
||||
the parts of OpenSSL and IJG JPEG Library used as well as that of the covered
|
||||
work.
|
||||
|
||||
You can contact Cyan Worlds, Inc. by email legal@cyan.com
|
||||
or by snail mail at:
|
||||
Cyan Worlds, Inc.
|
||||
14617 N Newport Hwy
|
||||
Mead, WA 99021
|
||||
|
||||
*==LICENSE==*/
|
||||
|
||||
#include "HeadSpin.h"
|
||||
#include "pfCrashHandler/plCrashSrv.h"
|
||||
#include "pnUtils/pnUtils.h"
|
||||
|
||||
enum
|
||||
{
|
||||
kArgMemFile
|
||||
};
|
||||
|
||||
static const CmdArgDef s_cmdLineArgs[] =
|
||||
{
|
||||
{ (kCmdArgRequired | kCmdTypeString), nil, kArgMemFile },
|
||||
};
|
||||
|
||||
int main(int argc, char* argv[])
|
||||
{
|
||||
// Parse command line arguments. We MUST have the file argument
|
||||
CCmdParser cmdParser(s_cmdLineArgs, arrsize(s_cmdLineArgs));
|
||||
if (!cmdParser.Parse())
|
||||
{
|
||||
hsMessageBox("You should never run this manually.", "Error", hsMessageBoxNormal, hsMessageBoxIconExclamation);
|
||||
return 1;
|
||||
}
|
||||
|
||||
char* file = hsWStringToString(cmdParser.GetString(kArgMemFile));
|
||||
plCrashSrv srv(file);
|
||||
delete[] file;
|
||||
srv.HandleCrash();
|
||||
return 0;
|
||||
}
|
Reference in New Issue
Block a user