764 lines
20 KiB
C++
Raw Permalink Normal View History

2020-04-22 12:56:21 -04:00
//========= Copyright Valve Corporation, All rights reserved. ============//
//
// Purpose: Xbox console link
//
//=====================================================================================//
#include "xbox/xbox_console.h"
#include "xbox/xbox_vxconsole.h"
#include "tier0/threadtools.h"
#include "tier0/tslist.h"
#include "tier0/ICommandLine.h"
#include "tier0/memdbgon.h"
// all redirecting funneled here, stop redirecting in this module only
#undef OutputDebugStringA
#pragma comment( lib, "xbdm.lib" )
struct DebugString_t
{
unsigned int color;
char *pString;
};
#define XBX_DBGCOMMANDPREFIX "XCMD"
#define XBX_DBGRESPONSEPREFIX "XACK"
#define XBX_DBGPRINTPREFIX "XPRT"
#define XBX_DBGCOLORPREFIX "XCLR"
#define XBX_MAX_RCMDLENGTH 256
#define XBX_MAX_MESSAGE 2048
CThreadFastMutex g_xbx_dbgChannelMutex;
CThreadFastMutex g_xbx_dbgCommandHandlerMutex;
static char g_xbx_dbgRemoteBuf[XBX_MAX_RCMDLENGTH];
static HANDLE g_xbx_dbgValidEvent;
static HANDLE g_xbx_dbgCmdCompleteEvent;
bool g_xbx_bUseVXConsoleOutput = true;
bool g_xbx_bDoSyncOutput;
static ThreadHandle_t g_xbx_hDebugThread;
CTSQueue<DebugString_t> g_xbx_DebugStringQueue;
extern CInterlockedInt g_xbx_numProfileCounters;
extern unsigned int g_xbx_profileCounters[];
extern char g_xbx_profileName[];
int g_xbx_freeMemory;
_inline bool XBX_NoXBDM() { return false; }
static CXboxConsole XboxConsole;
DLL_EXPORT IXboxConsole *GetConsoleInterface()
{
return &XboxConsole;
}
//-----------------------------------------------------------------------------
// Low level string output.
// Input string should be stack based, can get clobbered.
//-----------------------------------------------------------------------------
static void OutputStringToDevice( unsigned int color, char *pString, bool bRemoteValid )
{
if ( !bRemoteValid )
{
// local debug only
OutputDebugStringA( pString );
return;
}
// remote debug valid
// non pure colors don't translate well - find closest pure hue
unsigned int bestColor = color;
int r = ( bestColor & 0xFF );
int g = ( bestColor >> 8 ) & 0xFF;
int b = ( bestColor >> 16 ) & 0xFF;
if ( ( r && r != 255 ) || ( g && g != 255 ) || ( b && b != 255 ) )
{
int r0, g0, b0;
unsigned int minDist = 0xFFFFFFFF;
for ( int i=0; i<8; i++ )
{
r0 = g0 = b0 = 0;
if ( i&4 )
r0 = 255;
if ( i&2 )
g0 = 255;
if ( i&1 )
b0 = 255;
unsigned int d = ( r-r0 )*( r-r0 ) + ( g-g0 )*( g-g0 ) + ( b-b0 )*( b-b0 );
if ( minDist > d )
{
minDist = d;
bestColor = XMAKECOLOR( r0, g0, b0 );
}
}
}
// create color string
char colorString[16];
sprintf( colorString, XBX_DBGCOLORPREFIX "[%8.8x]", bestColor );
// chunk line out, for each cr
char strBuffer[XBX_MAX_RCMDLENGTH];
char *pStart = pString;
char *pEnd = pStart + strlen( pStart );
char *pNext;
while ( pStart < pEnd )
{
pNext = strchr( pStart, '\n' );
if ( !pNext )
pNext = pEnd;
else
*pNext = '\0';
int length = _snprintf( strBuffer, XBX_MAX_RCMDLENGTH, "%s!%s%s", XBX_DBGPRINTPREFIX, colorString, pStart );
if ( length == -1 )
{
strBuffer[sizeof( strBuffer )-1] = '\0';
}
// Send the string
DmSendNotificationString( strBuffer );
// advance past cr
pStart = pNext+1;
}
}
//-----------------------------------------------------------------------------
// XBX_IsConsoleConnected
//
//-----------------------------------------------------------------------------
bool CXboxConsole::IsConsoleConnected()
{
bool bConnected;
if ( g_xbx_dbgValidEvent == NULL )
{
// init was never called
return false;
}
AUTO_LOCK_FM( g_xbx_dbgChannelMutex );
bConnected = ( WaitForSingleObject( g_xbx_dbgValidEvent, 0 ) == WAIT_OBJECT_0 );
return bConnected;
}
//-----------------------------------------------------------------------------
// Output string to listening console. Queues output for slave thread.
// Needs to be lightweight.
//-----------------------------------------------------------------------------
void CXboxConsole::DebugString( unsigned int color, const char* pFormat, ... )
{
if ( XBX_NoXBDM() )
return;
va_list args;
char szStringBuffer[XBX_MAX_MESSAGE];
int length;
// resolve string
va_start( args, pFormat );
length = _vsnprintf( szStringBuffer, sizeof( szStringBuffer ), pFormat, args );
if ( length == -1 )
{
szStringBuffer[sizeof( szStringBuffer ) - 1] = '\0';
}
va_end( args );
if ( !g_xbx_bDoSyncOutput )
{
// queue string for delayed output
DebugString_t debugString;
debugString.color = color;
debugString.pString = strdup( szStringBuffer );
g_xbx_DebugStringQueue.PushItem( debugString );
}
else
{
bool bRemoteValid = g_xbx_bUseVXConsoleOutput && XBX_IsConsoleConnected();
OutputStringToDevice( color, szStringBuffer, bRemoteValid );
}
}
//-----------------------------------------------------------------------------
// Waits for debug queue to drain.
//-----------------------------------------------------------------------------
void CXboxConsole::FlushDebugOutput()
{
while ( g_xbx_DebugStringQueue.Count() != 0 )
{
Sleep( 1 );
}
}
//-----------------------------------------------------------------------------
// _xdbg_strlen
//
// Critical section safe.
//-----------------------------------------------------------------------------
int _xdbg_strlen( const CHAR* str )
{
const CHAR* strEnd = str;
while( *strEnd )
strEnd++;
return strEnd - str;
}
//-----------------------------------------------------------------------------
// _xdbg_tolower
//
// Critical section safe.
//-----------------------------------------------------------------------------
inline CHAR _xdbg_tolower( CHAR ch )
{
if( ch >= 'A' && ch <= 'Z' )
return ch - ( 'A' - 'a' );
else
return ch;
}
//-----------------------------------------------------------------------------
// _xdbg_strnicmp
//
// Critical section safe.
//-----------------------------------------------------------------------------
BOOL _xdbg_strnicmp( const CHAR* str1, const CHAR* str2, int n )
{
while ( ( _xdbg_tolower( *str1 ) == _xdbg_tolower( *str2 ) ) && *str1 && n > 0 )
{
--n;
++str1;
++str2;
}
return ( !n || _xdbg_tolower( *str1 ) == _xdbg_tolower( *str2 ) );
}
//-----------------------------------------------------------------------------
// _xdbg_strcpy
//
// Critical section safe.
//-----------------------------------------------------------------------------
VOID _xdbg_strcpy( CHAR* strDest, const CHAR* strSrc )
{
while ( ( *strDest++ = *strSrc++ ) != 0 );
}
//-----------------------------------------------------------------------------
// _xdbg_strcpyn
//
// Critical section safe.
//-----------------------------------------------------------------------------
VOID _xdbg_strcpyn( CHAR* strDest, const CHAR* strSrc, int numChars )
{
while ( numChars>0 && ( *strDest++ = *strSrc++ ) != 0 )
numChars--;
}
//-----------------------------------------------------------------------------
// _xdbg_gettoken
//
// Critical section safe.
//-----------------------------------------------------------------------------
void _xdbg_gettoken( CHAR** tokenStream, CHAR* token, int tokenSize )
{
int c;
int len;
CHAR* data;
len = 0;
// skip prefix whitespace
data = *tokenStream;
while ( ( c = *data ) <= ' ' )
{
if ( !c )
goto cleanUp;
data++;
}
// parse a token
do
{
if ( len < tokenSize )
token[len++] = c;
data++;
c = *data;
} while ( c > ' ' );
if ( len >= tokenSize )
len = 0;
cleanUp:
token[len] = '\0';
*tokenStream = data;
}
//-----------------------------------------------------------------------------
// _xdbg_tokenize
//
// Critical section safe.
//-----------------------------------------------------------------------------
int _xdbg_tokenize( CHAR* tokenStream, CHAR** tokens, int maxTokens )
{
char token[64];
// tokenize stream into seperate tokens
int numTokens = 0;
while ( 1 )
{
tokens[numTokens++] = tokenStream;
if ( numTokens >= maxTokens )
break;
_xdbg_gettoken( &tokenStream, token, sizeof( token ) );
if ( !tokenStream[0] || !token[0] )
break;
*tokenStream = '\0';
tokenStream++;
}
return ( numTokens );
}
//-----------------------------------------------------------------------------
// _xdbg_findtoken
//
// Critical section safe. Returns -1 if not found
//-----------------------------------------------------------------------------
int _xdbg_findtoken( CHAR** tokens, int numTokens, CHAR* token )
{
int i;
int len;
len = _xdbg_strlen( token );
for ( i=0; i<numTokens; i++ )
{
if ( _xdbg_strnicmp( tokens[i], token, len ) )
return i;
}
// not found
return -1;
}
//-----------------------------------------------------------------------------
// _DebugCommandHandler
//
//-----------------------------------------------------------------------------
HRESULT __stdcall _DebugCommandHandler( const CHAR* strCommand, CHAR* strResponse, DWORD dwResponseLen, PDM_CMDCONT pdmcc )
{
CHAR buff[256];
CHAR* args[8];
int numArgs;
AUTO_LOCK_FM( g_xbx_dbgCommandHandlerMutex );
// skip over the command prefix and the exclamation mark
strCommand += _xdbg_strlen( XBX_DBGCOMMANDPREFIX ) + 1;
if ( strCommand[0] == '\0' )
{
// just a ping
goto cleanUp;
}
// get command and optional arguments
_xdbg_strcpyn( buff, strCommand, sizeof( buff ) );
numArgs = _xdbg_tokenize( buff, args, sizeof( args )/sizeof( CHAR* ) );
if ( _xdbg_strnicmp( args[0], "__connect__", 11 ) )
{
if ( numArgs > 1 && atoi( args[1] ) == VXCONSOLE_PROTOCOL_VERSION )
{
// initial connect - respond that we're connected
_xdbg_strcpyn( strResponse, XBX_DBGRESPONSEPREFIX "Connected To Application.", dwResponseLen );
SetEvent( g_xbx_dbgValidEvent );
// notify convar system to send its commands
// allows vxconsole to re-connect during game
_xdbg_strcpy( g_xbx_dbgRemoteBuf, "getcvars" );
XBX_QueueEvent( XEV_REMOTECMD, ( int )g_xbx_dbgRemoteBuf, 0, 0 );
}
else
{
_xdbg_strcpyn( strResponse, XBX_DBGRESPONSEPREFIX "Rejecting Connection: Wrong Protocol Version.", dwResponseLen );
}
goto cleanUp;
}
if ( _xdbg_strnicmp( args[0], "__disconnect__", 14 ) )
{
// respond that we're disconnected
_xdbg_strcpyn( strResponse, XBX_DBGRESPONSEPREFIX "Disconnected.", dwResponseLen );
ResetEvent( g_xbx_dbgValidEvent );
goto cleanUp;
}
if ( _xdbg_strnicmp( args[0], "__complete__", 12 ) )
{
// remote server has finished command - respond to acknowledge
_xdbg_strcpyn( strResponse, XBX_DBGRESPONSEPREFIX "OK", dwResponseLen );
// set the complete event - allows expected synchronous calling mechanism
SetEvent( g_xbx_dbgCmdCompleteEvent );
goto cleanUp;
}
if ( _xdbg_strnicmp( args[0], "__memory__", 10 ) )
{
// get a current stat of available memory
MEMORYSTATUS stat;
GlobalMemoryStatus( &stat );
g_xbx_freeMemory = stat.dwAvailPhys;
if ( _xdbg_findtoken( args, numArgs, "quiet" ) > 0 )
{
_xdbg_strcpyn( strResponse, XBX_DBGRESPONSEPREFIX "OK", dwResponseLen );
}
else
{
// 32 MB is reserved and fixed by OS, so not reporting
_snprintf( strResponse, dwResponseLen, XBX_DBGRESPONSEPREFIX "Available: %.2f MB, Used: %.2f MB, Free: %.2f MB",
stat.dwTotalPhys/( 1024.0f*1024.0f ) - 32.0f,
( stat.dwTotalPhys - stat.dwAvailPhys )/( 1024.0f*1024.0f ) - 32.0f,
stat.dwAvailPhys/( 1024.0f*1024.0f ) );
}
goto cleanUp;
}
if ( g_xbx_dbgRemoteBuf[0] )
{
// previous command still pending
_xdbg_strcpyn( strResponse, XBX_DBGRESPONSEPREFIX "Cannot execute: Previous command still pending", dwResponseLen );
}
else
{
// add the command to the event queue to be processed by main app
_xdbg_strcpy( g_xbx_dbgRemoteBuf, strCommand );
XBX_QueueEvent( XEV_REMOTECMD, ( int )g_xbx_dbgRemoteBuf, 0, 0 );
_xdbg_strcpyn( strResponse, XBX_DBGRESPONSEPREFIX "OK", dwResponseLen );
}
cleanUp:
return XBDM_NOERR;
}
//-----------------------------------------------------------------------------
// XBX_SendRemoteCommand
//
//-----------------------------------------------------------------------------
void CXboxConsole::SendRemoteCommand( const char *pCommand, bool async )
{
char cmdString[XBX_MAX_RCMDLENGTH];
if ( XBX_NoXBDM() || !IsConsoleConnected() )
return;
AUTO_LOCK_FM( g_xbx_dbgChannelMutex );
_snprintf( cmdString, sizeof( cmdString ), "%s!%s", XBX_DBGCOMMANDPREFIX, pCommand );
HRESULT hr = DmSendNotificationString( cmdString );
if ( FAILED( hr ) )
{
XBX_Error( "XBX_SendRemoteCommand: failed on %s", cmdString );
}
// wait for command completion
if ( !async )
{
DWORD timeout;
if ( !strnicmp( pCommand, "Assert()", 8 ) )
{
// the assert is waiting for user to make selection
timeout = INFINITE;
}
else
{
// no vxconsole operation should take this long
timeout = 15000;
}
if ( WaitForSingleObject( g_xbx_dbgCmdCompleteEvent, timeout ) == WAIT_TIMEOUT )
{
// we have no choice but to dump core
DmCrashDump( false );
}
}
}
//-----------------------------------------------------------------------------
// Handle delayed VXConsole transactions
//
//-----------------------------------------------------------------------------
static unsigned _DebugThreadFunc( void *pParam )
{
while ( 1 )
{
Sleep( 10 );
if ( !g_xbx_DebugStringQueue.Count() && !g_xbx_numProfileCounters && !g_xbx_freeMemory )
{
continue;
}
if ( g_xbx_numProfileCounters )
{
// build and send asynchronously
char dbgCommand[XBX_MAX_RCMDLENGTH];
_snprintf( dbgCommand, sizeof( dbgCommand ), "SetProfileData() %s 0x%8.8x", g_xbx_profileName, g_xbx_profileCounters );
XBX_SendRemoteCommand( dbgCommand, true );
// mark as sent
g_xbx_numProfileCounters = 0;
}
if ( g_xbx_freeMemory )
{
// build and send asynchronously
char dbgCommand[XBX_MAX_RCMDLENGTH];
_snprintf( dbgCommand, sizeof( dbgCommand ), "FreeMemory() 0x%8.8x", g_xbx_freeMemory );
XBX_SendRemoteCommand( dbgCommand, true );
// mark as sent
g_xbx_freeMemory = 0;
}
bool bRemoteValid = g_xbx_bUseVXConsoleOutput && XBX_IsConsoleConnected();
while ( 1 )
{
DebugString_t debugString;
if ( !g_xbx_DebugStringQueue.PopItem( &debugString ) )
{
break;
}
OutputStringToDevice( debugString.color, debugString.pString, bRemoteValid );
free( debugString.pString );
}
}
return 0;
}
//-----------------------------------------------------------------------------
// XBX_InitConsoleMonitor
//
//-----------------------------------------------------------------------------
void CXboxConsole::InitConsoleMonitor( bool bWaitForConnect )
{
if ( XBX_NoXBDM() )
return;
// create our events
g_xbx_dbgValidEvent = CreateEvent( XBOX_DONTCARE, TRUE, FALSE, NULL );
g_xbx_dbgCmdCompleteEvent = CreateEvent( XBOX_DONTCARE, FALSE, FALSE, NULL );
// register our command handler with the debug monitor
HRESULT hr = DmRegisterCommandProcessor( XBX_DBGCOMMANDPREFIX, _DebugCommandHandler );
if ( FAILED( hr ) )
{
XBX_Error( "XBX_InitConsoleMonitor: failed to register command processor" );
}
// user can have output bypass slave thread
g_xbx_bDoSyncOutput = CommandLine()->FindParm( "-syncoutput" ) != 0;
// create a slave thread to do delayed VXConsole transactions
ThreadId_t threadID;
g_xbx_hDebugThread = CreateSimpleThread( _DebugThreadFunc, NULL, &threadID, 16*1024 );
ThreadSetDebugName( threadID, "DebugThread" );
ThreadSetAffinity( g_xbx_hDebugThread, XBOX_PROCESSOR_5 );
if ( bWaitForConnect )
{
XBX_DebugString( XBX_CLR_DEFAULT, "Waiting For VXConsole Connection...\n" );
WaitForSingleObject( g_xbx_dbgValidEvent, INFINITE );
}
}
//-----------------------------------------------------------------------------
// Sends a disconnect signal to possibly attached VXConsole.
//-----------------------------------------------------------------------------
void CXboxConsole::DisconnectConsoleMonitor()
{
if ( XBX_NoXBDM() )
return;
// caller is trying to safely stop vxconsole traffic, disconnect must be synchronous
XBX_SendRemoteCommand( "Disconnect()", false );
}
bool CXboxConsole::GetXboxName( char *pName, unsigned *pLength )
{
return ( DmGetXboxName( pName, (DWORD *)pLength ) == XBDM_NOERR );
}
void CXboxConsole::CrashDump( bool b )
{
DmCrashDump(b);
}
//-----------------------------------------------------------------------------
// Walk to a specific module and dump size info
//-----------------------------------------------------------------------------
int CXboxConsole::DumpModuleSize( const char *pName )
{
HRESULT error;
PDM_WALK_MODULES pWalkMod = NULL;
DMN_MODLOAD modLoad;
int size = 0;
// iterate and find match
do
{
error = DmWalkLoadedModules( &pWalkMod, &modLoad );
if ( XBDM_NOERR == error && !stricmp( modLoad.Name, pName ) )
{
Msg( "0x%8.8x, %5.2f MB, %s\n", modLoad.BaseAddress, modLoad.Size/( 1024.0f*1024.0f ), modLoad.Name );
size = modLoad.Size;
error = XBDM_ENDOFLIST;
}
}
while ( XBDM_NOERR == error );
DmCloseLoadedModules( pWalkMod );
if ( error != XBDM_ENDOFLIST )
{
Warning( "DmWalkLoadedModules() failed.\n" );
}
return size;
}
//-----------------------------------------------------------------------------
// 360 spew sizes of dll modules
//-----------------------------------------------------------------------------
char const* HACK_stristr( char const* pStr, char const* pSearch ) // hack because moved code from above vstdlib
{
AssertValidStringPtr(pStr);
AssertValidStringPtr(pSearch);
if (!pStr || !pSearch)
return 0;
char const* pLetter = pStr;
// Check the entire string
while (*pLetter != 0)
{
// Skip over non-matches
if (tolower((unsigned char)*pLetter) == tolower((unsigned char)*pSearch))
{
// Check for match
char const* pMatch = pLetter + 1;
char const* pTest = pSearch + 1;
while (*pTest != 0)
{
// We've run off the end; don't bother.
if (*pMatch == 0)
return 0;
if (tolower((unsigned char)*pMatch) != tolower((unsigned char)*pTest))
break;
++pMatch;
++pTest;
}
// Found a match!
if (*pTest == 0)
return pLetter;
}
++pLetter;
}
return 0;
}
void CXboxConsole::DumpDllInfo( const char *pBasePath )
{
// Directories containing dlls
static char *dllDirs[] =
{
"bin",
"hl2\\bin",
"tf\\bin",
"portal\\bin",
"episodic\\bin"
};
char binPath[MAX_PATH];
char dllPath[MAX_PATH];
char searchPath[MAX_PATH];
HMODULE hModule;
WIN32_FIND_DATA wfd;
HANDLE hFind;
Msg( "Dumping Module Sizes...\n" );
for ( int i = 0; i < ARRAYSIZE( dllDirs ); ++i )
{
int totalSize = 0;
_snprintf( binPath, sizeof( binPath ), "%s\\%s", pBasePath, dllDirs[i] );
_snprintf( searchPath, sizeof( binPath ), "%s\\*.dll", binPath );
// show the directory we're searching
Msg( "\nDirectory: %s\n\n", binPath );
// Start the find and check for failure.
hFind = FindFirstFile( searchPath, &wfd );
if ( INVALID_HANDLE_VALUE == hFind )
{
Warning( "No Files Found.\n" );
}
else
{
// Load and unload each dll individually.
do
{
if ( !HACK_stristr( wfd.cFileName, "_360.dll" ) )
{
// exclude explicit pc dlls
// FindFirstFile does not support a spec mask of *_360.dll on the Xbox HDD
continue;
}
_snprintf( dllPath, sizeof( dllPath ), "%s\\%s", binPath, wfd.cFileName );
hModule = LoadLibrary( dllPath );
if ( hModule )
{
totalSize += DumpModuleSize( wfd.cFileName );
FreeLibrary( hModule );
}
else
{
Warning( "Failed to load: %s\n", dllPath );
}
}
while( FindNextFile( hFind, &wfd ) );
FindClose( hFind );
Msg( "Total Size: %.2f MB\n", totalSize/( 1024.0f*1024.0f ) );
}
}
}
void CXboxConsole::OutputDebugString( const char *p )
{
::OutputDebugStringA( p );
}
bool CXboxConsole::IsDebuggerPresent()
{
return ( DmIsDebuggerPresent() != 0 );
}